updates
This commit is contained in:
parent
ad6fb0c03e
commit
ed5a425dab
5
engine/ecs/entity.go
Normal file
5
engine/ecs/entity.go
Normal file
@ -0,0 +1,5 @@
|
||||
package ecs
|
||||
|
||||
type Entity struct {
|
||||
ID int
|
||||
}
|
88
engine/fov/fov.go
Normal file
88
engine/fov/fov.go
Normal file
@ -0,0 +1,88 @@
|
||||
package fov
|
||||
|
||||
import (
|
||||
"github.com/jcerise/gogue/gamemap"
|
||||
"math"
|
||||
)
|
||||
|
||||
type FieldOfVision struct {
|
||||
cosTable map[int]float64
|
||||
sinTable map[int]float64
|
||||
torchRadius int
|
||||
}
|
||||
|
||||
func (f *FieldOfVision) Initialize() {
|
||||
|
||||
f.cosTable = make(map[int]float64)
|
||||
f.sinTable = make(map[int]float64)
|
||||
|
||||
for i := 0; i < 360; i++ {
|
||||
ax := math.Sin(float64(i) / (float64(180) / math.Pi))
|
||||
ay := math.Cos(float64(i) / (float64(180) / math.Pi))
|
||||
|
||||
f.sinTable[i] = ax
|
||||
f.cosTable[i] = ay
|
||||
}
|
||||
}
|
||||
|
||||
func (f *FieldOfVision) SetTorchRadius(radius int) {
|
||||
if radius > 1 {
|
||||
f.torchRadius = radius
|
||||
}
|
||||
}
|
||||
|
||||
func (f *FieldOfVision) SetAllInvisible(gameMap *gamemap.Map) {
|
||||
for x := 0; x < gameMap.Width; x++ {
|
||||
for y := 0; y < gameMap.Height; y++ {
|
||||
gameMap.Tiles[x][y].Visible = false
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func (f *FieldOfVision) RayCast(playerX, playerY int, gameMap *gamemap.Map) {
|
||||
// Cast out rays each degree in a 360 circle from the player. If a ray passes over a floor (does not block sight)
|
||||
// tile, keep going, up to the maximum torch radius (view radius) of the player. If the ray intersects a wall
|
||||
// (blocks sight), stop, as the player will not be able to see past that. Every visible tile will get the Visible
|
||||
// and Explored properties set to true.
|
||||
|
||||
for i := 0; i < 360; i++ {
|
||||
|
||||
ax := f.sinTable[i]
|
||||
ay := f.cosTable[i]
|
||||
|
||||
x := float64(playerX)
|
||||
y := float64(playerY)
|
||||
|
||||
// Mark the players current position as explored
|
||||
tile := gameMap.Tiles[playerX][playerY]
|
||||
tile.Explored = true
|
||||
tile.Visible = true
|
||||
|
||||
for j := 0; j < f.torchRadius; j++ {
|
||||
x -= ax
|
||||
y -= ay
|
||||
|
||||
roundedX := int(Round(x))
|
||||
roundedY := int(Round(y))
|
||||
|
||||
if x < 0 || x > float64(gameMap.Width-1) || y < 0 || y > float64(gameMap.Height-1) {
|
||||
// If the ray is cast outside of the gamemap, stop
|
||||
break
|
||||
}
|
||||
|
||||
tile := gameMap.Tiles[roundedX][roundedY]
|
||||
|
||||
tile.Explored = true
|
||||
tile.Visible = true
|
||||
|
||||
if gameMap.Tiles[roundedX][roundedY].BlocksSight == true {
|
||||
// The ray hit a wall, go no further
|
||||
break
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func Round(f float64) float64 {
|
||||
return math.Floor(f + .5)
|
||||
}
|
14
engine/gamemap/level.go
Normal file
14
engine/gamemap/level.go
Normal file
@ -0,0 +1,14 @@
|
||||
package gamemap
|
||||
|
||||
import "lab.zaar.be/thefish/alchemyst-go/engine/ecs"
|
||||
|
||||
type Level struct {
|
||||
Name string
|
||||
Branch string
|
||||
Depth int
|
||||
MaxRooms int
|
||||
Width int
|
||||
Height int
|
||||
Objects []ecs.Entity
|
||||
Tiles [][]*Tile
|
||||
}
|
20
engine/gamemap/mapgen.go
Normal file
20
engine/gamemap/mapgen.go
Normal file
@ -0,0 +1,20 @@
|
||||
package gamemap
|
||||
|
||||
type mapGen interface {
|
||||
generate(l *Level) *Level
|
||||
}
|
||||
|
||||
type defaultGen struct {}
|
||||
|
||||
func (d defaultGen) generate(l *Level) *Level {
|
||||
|
||||
l.Tiles, rooms = addRooms(l)
|
||||
l. Tiles = connectRooms(rooms)
|
||||
l.Objects = populate(rooms)
|
||||
|
||||
return l
|
||||
}
|
||||
|
||||
func addRooms(l *Level) {
|
||||
|
||||
}
|
143
engine/gamemap/tile.go
Normal file
143
engine/gamemap/tile.go
Normal file
@ -0,0 +1,143 @@
|
||||
package gamemap
|
||||
|
||||
import "lab.zaar.be/thefish/alchemyst-go/util"
|
||||
import blt "lab.zaar.be/thefish/bearlibterminal"
|
||||
|
||||
type ColorHolder struct {
|
||||
R uint8
|
||||
G uint8
|
||||
B uint8
|
||||
}
|
||||
|
||||
type TileColorSet struct {
|
||||
Fg func() uint32
|
||||
Bg func() uint32
|
||||
DarkFg func() uint32
|
||||
DarkBg func() uint32
|
||||
current *ColorHolder
|
||||
}
|
||||
|
||||
type Appearance struct {
|
||||
Char string
|
||||
ColorSet *TileColorSet
|
||||
}
|
||||
|
||||
var crng = util.NewRNG()
|
||||
|
||||
func colordance(colorValue uint8, minGlow, maxGlow, step int) uint8 {
|
||||
color := crng.Range(0, step) + int(colorValue)
|
||||
if color > maxGlow {
|
||||
color = crng.Range(0, step) + minGlow
|
||||
}
|
||||
return uint8(color)
|
||||
}
|
||||
|
||||
type Tile struct {
|
||||
*Appearance
|
||||
Name string
|
||||
Description string
|
||||
BlocksPass bool
|
||||
BlocksSight bool
|
||||
Explored bool
|
||||
MustDraw bool
|
||||
}
|
||||
|
||||
|
||||
func NewWall() *Tile {
|
||||
return &Tile{
|
||||
Name: "Wall",
|
||||
Description: "A dull rock wall",
|
||||
BlocksPass: true,
|
||||
BlocksSight: true,
|
||||
Explored: false,
|
||||
MustDraw: false,
|
||||
Appearance: &Appearance{
|
||||
Char: "#",
|
||||
ColorSet: &TileColorSet{
|
||||
Fg: func() uint32 {return blt.ColorFromARGB(255, 130,110,150)},
|
||||
Bg: func() uint32 {return blt.ColorFromARGB(255, 172,170,173)},
|
||||
DarkFg: func() uint32 {return blt.ColorFromARGB(255, 20,20,68)},
|
||||
DarkBg: func() uint32 {return blt.ColorFromARGB(255, 7,7,30)},
|
||||
},
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
func NewFloor() *Tile {
|
||||
return &Tile{
|
||||
Name: "Floor",
|
||||
Description: "Dusty rock floor",
|
||||
BlocksPass: false,
|
||||
BlocksSight: false,
|
||||
Explored: false,
|
||||
MustDraw: false,
|
||||
Appearance: &Appearance{
|
||||
Char: ".",
|
||||
ColorSet: &TileColorSet{
|
||||
Fg: func() uint32 {return blt.ColorFromARGB(255, 220,220,250)},
|
||||
Bg: func() uint32 {return blt.ColorFromARGB(255, 19,19,70)},
|
||||
DarkFg: func() uint32 {return blt.ColorFromARGB(255, 30,20,50)},
|
||||
DarkBg: func() uint32 {return blt.ColorFromARGB(255, 7,7,30)},
|
||||
},
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
func NewWaterTile() *Tile {
|
||||
ch := &ColorHolder{19,19,70}
|
||||
return &Tile {
|
||||
Name: "Water",
|
||||
Description: "Murky water",
|
||||
BlocksPass: false,
|
||||
BlocksSight: false,
|
||||
Explored: false,
|
||||
MustDraw: true, //fixme debug
|
||||
Appearance: &Appearance{
|
||||
Char: ".",
|
||||
ColorSet: &TileColorSet{
|
||||
current: ch,
|
||||
Fg: func() uint32 {return blt.ColorFromARGB(255, 220,220,250)},
|
||||
Bg: func() uint32 {
|
||||
return blt.ColorFromARGB(
|
||||
255,
|
||||
ch.R,
|
||||
colordance(ch.G, 2, 42, 4 ),
|
||||
colordance(ch.B, 180,229,12),
|
||||
)
|
||||
},
|
||||
DarkFg: func() uint32 {return blt.ColorFromARGB(255, 30,20,50)},
|
||||
DarkBg: func() uint32 {return blt.ColorFromARGB(255, 7,7,30)},
|
||||
},
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
func NewDeepWaterTile() *Tile {
|
||||
ch := &ColorHolder{5,2,154}
|
||||
return &Tile {
|
||||
Name: "Deep Water",
|
||||
Description: "Deep water",
|
||||
BlocksPass: false,
|
||||
BlocksSight: false,
|
||||
Explored: false,
|
||||
MustDraw: true, //fixme debug
|
||||
Appearance: &Appearance{
|
||||
Char: " ",
|
||||
ColorSet: &TileColorSet{
|
||||
current: ch,
|
||||
Fg: func() uint32 {return blt.ColorFromARGB(255, 220,220,250)},
|
||||
Bg: func() uint32 {
|
||||
return blt.ColorFromARGB(
|
||||
255,
|
||||
ch.R,
|
||||
colordance(ch.G, 0, 15, 2 ),
|
||||
colordance(ch.B, 120,180,5),
|
||||
)
|
||||
},
|
||||
DarkFg: func() uint32 {return blt.ColorFromARGB(255, 30,20,50)},
|
||||
DarkBg: func() uint32 {return blt.ColorFromARGB(255, 7,7,30)},
|
||||
},
|
||||
},
|
||||
}
|
||||
}
|
||||
|
5
engine/gamemap/types.go
Normal file
5
engine/gamemap/types.go
Normal file
@ -0,0 +1,5 @@
|
||||
package gamemap
|
||||
|
||||
type Coords struct {
|
||||
x,y int
|
||||
}
|
1
main.go
1
main.go
@ -59,7 +59,6 @@ func main() {
|
||||
|
||||
}
|
||||
mainCtx.Logger().Info().Msg("pre-shutdown sequence")
|
||||
|
||||
}
|
||||
|
||||
// do runs f on the main thread.
|
||||
|
43
ui/mainwindow/camera.go
Normal file
43
ui/mainwindow/camera.go
Normal file
@ -0,0 +1,43 @@
|
||||
package mainwindow
|
||||
|
||||
type GameCamera struct {
|
||||
X int
|
||||
Y int
|
||||
Width int
|
||||
Height int
|
||||
}
|
||||
|
||||
func (c *GameCamera) MoveCamera(targetX int, targetY int, mapWidth int, mapHeight int) {
|
||||
// Update the camera coordinates to the target coordinates
|
||||
x := targetX - c.Width/2
|
||||
y := targetY - c.Height/2
|
||||
|
||||
if x < 0 {
|
||||
x = 0
|
||||
}
|
||||
|
||||
if y < 0 {
|
||||
y = 0
|
||||
}
|
||||
|
||||
if x > mapWidth - c.Width {
|
||||
x = mapWidth - c.Width
|
||||
}
|
||||
|
||||
if y > mapHeight - c.Height {
|
||||
y = mapHeight - c.Height
|
||||
}
|
||||
|
||||
c.X, c.Y = x, y
|
||||
}
|
||||
|
||||
func (c *GameCamera) ToCameraCoordinates(mapX int, mapY int) (cameraX int, cameraY int) {
|
||||
// Convert coordinates on the gamemap, to coordinates on the viewport
|
||||
x, y := mapX-c.X, mapY-c.Y
|
||||
|
||||
if x < 0 || y < 0 || x >= c.Width || y >= c.Height {
|
||||
return -1, -1
|
||||
}
|
||||
|
||||
return x, y
|
||||
}
|
@ -1,12 +0,0 @@
|
||||
package mainwindow
|
||||
|
||||
//Console is a pair of layers (BG and FG) used to render something
|
||||
// All because of lack of background colors in libbearterminal
|
||||
|
||||
type Console struct {
|
||||
x,y,w,h int
|
||||
FgLayer *Layer
|
||||
BgLayer *Layer
|
||||
}
|
||||
|
||||
func NewConsole
|
@ -26,6 +26,7 @@ func (mw *MainWindow) Open() {
|
||||
fmt.Sprintf(
|
||||
//"window: size=%dx%d, title='%s v%s'; font: ./resources/fonts-bitmap/ibmnew8x12.png, size=8x12;",
|
||||
"window: size=%dx%d, title='%s v%s'; font: %s, size=8x16;",
|
||||
//"window: size=%dx%d, title='%s v%s'",
|
||||
config.MainWindowSizeX,
|
||||
config.MainWindowSizeY,
|
||||
config.Title,
|
||||
|
94
util/rng.go
Normal file
94
util/rng.go
Normal file
@ -0,0 +1,94 @@
|
||||
package util
|
||||
|
||||
import (
|
||||
"math"
|
||||
"math/rand"
|
||||
"time"
|
||||
)
|
||||
|
||||
type RNG struct {
|
||||
seed int64
|
||||
rand *rand.Rand
|
||||
}
|
||||
|
||||
func NewRNG() *RNG {
|
||||
rng := RNG{}
|
||||
|
||||
// Set the seed to the current time. This can be updated later by the user.
|
||||
rng.seed = time.Now().UTC().UnixNano()
|
||||
rng.rand = rand.New(rand.NewSource(rng.seed))
|
||||
|
||||
return &rng
|
||||
}
|
||||
|
||||
func (rng *RNG) GetSeed() int64 {
|
||||
return rng.seed
|
||||
}
|
||||
|
||||
func (rng *RNG) SetSeed(seed int64) {
|
||||
rng.seed = seed
|
||||
}
|
||||
|
||||
func (rng *RNG) Uniform() float64 {
|
||||
return rng.rand.Float64()
|
||||
}
|
||||
|
||||
func (rng *RNG) UniformRange(a, b float64) float64 {
|
||||
return a + rng.Uniform() * (b - a)
|
||||
}
|
||||
|
||||
func (rng *RNG) Normal(mean, stddev float64) float64 {
|
||||
|
||||
var r, x float64
|
||||
|
||||
for r >= 1 || r == 0 {
|
||||
x = rng.UniformRange(-1.0, 1.0)
|
||||
y := rng.UniformRange(-1.0, 1.0)
|
||||
r = x*x + y*y
|
||||
}
|
||||
|
||||
result := x * math.Sqrt(-2 * math.Log(r) / r)
|
||||
|
||||
return mean + stddev * result
|
||||
|
||||
}
|
||||
|
||||
func (rng *RNG) Percentage() int {
|
||||
return rng.rand.Intn(100)
|
||||
}
|
||||
|
||||
func (rng *RNG) Range(min, max int) int {
|
||||
if min == max {
|
||||
return min
|
||||
} else {
|
||||
return rng.rand.Intn(max - min) + min
|
||||
}
|
||||
}
|
||||
|
||||
func (rng *RNG) RangeNegative(min, max int) int {
|
||||
if min == max {
|
||||
return min
|
||||
} else {
|
||||
return rng.rand.Intn(max - min + 1) + min
|
||||
}
|
||||
}
|
||||
|
||||
func (rng *RNG) GetWeightedEntity(values map[int]int) int {
|
||||
// First up, get the total weight value from the gamemap
|
||||
totalWeight := 0
|
||||
for weight := range values {
|
||||
totalWeight += weight
|
||||
}
|
||||
|
||||
// Next, get a random integer in the range of the total weight
|
||||
r := rng.Range(0, totalWeight)
|
||||
|
||||
for weight, value := range values {
|
||||
r -= value
|
||||
if r <= 0 {
|
||||
return weight
|
||||
}
|
||||
}
|
||||
|
||||
return -1
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user