From b8c8a65fa7fea5787b86e653d85718fff7c6bfeb Mon Sep 17 00:00:00 2001 From: thefish Date: Sun, 27 Oct 2019 01:08:06 +0300 Subject: [PATCH] basic mapgen --- engine/gamemap/level.go | 46 ++++++------- engine/gamemap/mapgens/default.go | 108 ++++++++++++++++++++++++++++++ engine/gamemap/types.go | 5 -- engine/types/coords.go | 9 +++ engine/types/rect.go | 16 ++++- ui/mainwindow/uiwindow.go | 6 +- 6 files changed, 154 insertions(+), 36 deletions(-) create mode 100644 engine/gamemap/mapgens/default.go delete mode 100644 engine/gamemap/types.go create mode 100644 engine/types/coords.go diff --git a/engine/gamemap/level.go b/engine/gamemap/level.go index 9af8793..a65036a 100644 --- a/engine/gamemap/level.go +++ b/engine/gamemap/level.go @@ -2,59 +2,51 @@ package gamemap import ( "lab.zaar.be/thefish/alchemyst-go/engine/ecs" + "lab.zaar.be/thefish/alchemyst-go/engine/types" "lab.zaar.be/thefish/alchemyst-go/util" ) -var maxrooms = 100 +//fixme move to config var mapWidth = 150 var mapHeight = 100 + type Level struct { + *types.Rect + ctx util.ClientCtx Name string Branch string Depth int MaxRooms int - Width int - Height int Objects []ecs.Entity Tiles [][]*Tile } +func (l *Level) Put (x, y int, tileFunc interface{}) { + tf := tileFunc.(func() *Tile)() + if tf == nil { + l.ctx.Logger().Fatal().Msgf("Got non-tile type to put into level: %v", tf) + } + l.Tiles[x][y] = tf +} + func NewLevel(ctx util.ClientCtx, branch string, depth int) *Level { l := &Level{ Name: branch + string(depth), Depth: depth, MaxRooms: maxrooms, - Width: mapWidth, - Height: mapHeight, + Rect: types.NewRect(0,0, mapWidth, mapHeight), } - l.Tiles = make([][]*Tile, l.Width) + l.Tiles = make([][]*Tile, l.W) for i := range l.Tiles { - l.Tiles[i] = make([]*Tile, l.Height) - } - - return l -} - -func Generate(l *Level) (l *Level) { - - for i := 0; i < l.MaxRooms; i++ { - + l.Tiles[i] = make([]*Tile, l.H) } return l } type Room struct { - x, y, w, h int -} - -func (self *Room) intersects(other *Room) bool { - if self.x <= (other.x+other.w) && - (self.x+self.w) >= other.x && - self.y <= (other.y+other.h) && - (self.y+self.h) >= other.y { - - } -} + *types.Rect + Center *types.Coords +} \ No newline at end of file diff --git a/engine/gamemap/mapgens/default.go b/engine/gamemap/mapgens/default.go new file mode 100644 index 0000000..bc6ddc3 --- /dev/null +++ b/engine/gamemap/mapgens/default.go @@ -0,0 +1,108 @@ +package mapgens + +import ( + "lab.zaar.be/thefish/alchemyst-go/engine/gamemap" + "lab.zaar.be/thefish/alchemyst-go/engine/types" + "lab.zaar.be/thefish/alchemyst-go/util" +) +//fixme move to config +var maxrooms = 100 +var minRoomSize = 3 +var maxRoomSize = 11 + +func DefaultGen(l *gamemap.Level) *gamemap.Level { + + rng := util.NewRNG() + + //fill with walls + for i := 0; i < l.W; i ++ { + for j := 0; j < l.H; j++ { + l.Tiles[i][j] = gamemap.NewWall() + } + } + + rooms := make([]*gamemap.Room, maxrooms) + + for i := 0; i < l.MaxRooms; i++ { + newRoom := &gamemap.Room{ + Rect: types.NewRect( + rng.Range(l.X, l.W), + rng.Range(l.Y, l.H), + rng.Range(minRoomSize, maxRoomSize), + rng.Range(minRoomSize, maxRoomSize), + )} + newRoom.Center.X = newRoom.X + newRoom.W / 2 + newRoom.Center.Y = newRoom.Y + newRoom.H / 2 + + failed := false + for _, otherRoom := range rooms { + if otherRoom.Intersects(newRoom.Rect) { + failed = true + break + } + } + if !failed { + rooms = append(rooms, newRoom) + } + } + + fillage := types.RectFill{ + Top: func() *gamemap.Tile {return gamemap.NewWall()}, + Bottom: func() *gamemap.Tile {return gamemap.NewWall()}, + Left: func() *gamemap.Tile {return gamemap.NewWall()}, + Right: func() *gamemap.Tile {return gamemap.NewWall()}, + BottomLeft: func() *gamemap.Tile {return gamemap.NewWall()}, + BottomRight: func() *gamemap.Tile {return gamemap.NewWall()}, + TopLeft: func() *gamemap.Tile {return gamemap.NewWall()}, + TopRight: func() *gamemap.Tile {return gamemap.NewWall()}, + Body: func() *gamemap.Tile {return gamemap.NewFloor()}, + } + + for idx, room := range rooms { + room.Blit(fillage, l) + if idx > 0 { + connectRooms(l, room, rooms[idx-1], fillage, rng.Range(0,1)) + } + } + + return l +} + +func connectRooms (l *gamemap.Level, room, otherRoom *gamemap.Room, fillage types.RectFill, toss int) { + if toss == 0 { + digHTunnel(l, room.Center.X,otherRoom.Center.X,room.Center.Y, fillage) + digVTunnel(l, room.Center.Y,otherRoom.Center.Y,otherRoom.Center.X, fillage) + } else { + digVTunnel(l, room.Center.Y, otherRoom.Center.Y, room.Center.Y, fillage) + digHTunnel(l, room.Center.X, otherRoom.Center.X, otherRoom.Center.Y, fillage) + } +} + +func digHTunnel(l *gamemap.Level, x1,x2,y int, fillage types.RectFill) { + var start, finish int + if x1 > x2 { + start = x1 + finish = x2 + } else { + start = x2 + finish = x1 + } + for i := start; i <= finish; i++ { + l.Tiles[i][y] = fillage.Body.(func() *gamemap.Tile)() + } +} + + +func digVTunnel(l *gamemap.Level, y1,y2,x int, fillage types.RectFill) { + var start, finish int + if y1 > y2 { + start = y1 + finish = y2 + } else { + start = y2 + finish = y1 + } + for i := start; i <= finish; i++ { + l.Tiles[x][i] = fillage.Body.(func() *gamemap.Tile)() + } +} diff --git a/engine/gamemap/types.go b/engine/gamemap/types.go deleted file mode 100644 index 816bf62..0000000 --- a/engine/gamemap/types.go +++ /dev/null @@ -1,5 +0,0 @@ -package gamemap - -type Coords struct { - x, y int -} diff --git a/engine/types/coords.go b/engine/types/coords.go new file mode 100644 index 0000000..097d2e9 --- /dev/null +++ b/engine/types/coords.go @@ -0,0 +1,9 @@ +package types + +type Coords struct { + X, Y int +} + +func (c *Coords) Get() (int,int) { + return c.X,c.Y +} \ No newline at end of file diff --git a/engine/types/rect.go b/engine/types/rect.go index 5bb0917..bbfaa0e 100644 --- a/engine/types/rect.go +++ b/engine/types/rect.go @@ -12,7 +12,7 @@ func NewRect(x, y, w, h int) *Rect { return &Rect{x, y, w, h} } -func (r *Rect) RenderToLayer(fillage RectFill, layer Putable) { +func (r *Rect) Blit(fillage RectFill, layer Putable) { if fillage.Body != "" { for i := r.X + 1; i < r.X+r.W; i++ { @@ -45,3 +45,17 @@ func (r *Rect) RenderToLayer(fillage RectFill, layer Putable) { //lii.Put(X+W, Y-1, "L"); layer.Put(r.X+r.W, r.Y+r.H, fillage.BottomRight) } + +func (self *Rect) Intersects(other *Rect) bool { + if self.X <= (other.X+other.W) && + (self.X+self.W) >= other.X && + self.Y <= (other.Y+other.Y) && + (self.Y+self.H) >= other.Y { + return true + } + return false +} + +func (r *Rect) InBounds (x,y int) bool { + return x >= r.X && x <= (r.X + r.W) && y >= r.Y && y <= (r.Y + r.H) +} \ No newline at end of file diff --git a/ui/mainwindow/uiwindow.go b/ui/mainwindow/uiwindow.go index b01c6e2..6d0a621 100644 --- a/ui/mainwindow/uiwindow.go +++ b/ui/mainwindow/uiwindow.go @@ -51,15 +51,15 @@ func (layer *Layer) NewWindow(x, y, w, h int) *UiWindow { } func (uiw *UiWindow) NoBorder() { - uiw.RenderToLayer(noborder, uiw.layer) + uiw.Blit(noborder, uiw.layer) } func (uiw *UiWindow) Splash() { - uiw.RenderToLayer(splash, uiw.layer) + uiw.Blit(splash, uiw.layer) } func (uiw *UiWindow) DoubleBordered(title string) { - uiw.RenderToLayer(doubleBorder, uiw.layer) + uiw.Blit(doubleBorder, uiw.layer) if len(title) > (uiw.W - 2) { title = title[:(uiw.W - 2)] }