use gogues ecs, not working, needs rethinking

This commit is contained in:
thefish 2019-11-04 19:07:16 +03:00
parent fd27dfd636
commit f9ebcefc86
13 changed files with 490 additions and 33 deletions

View File

@ -3,6 +3,7 @@ package main
import (
"github.com/rs/zerolog"
"github.com/rs/zerolog/log"
"lab.zaar.be/thefish/alchemyst-go/engine/ecs"
"lab.zaar.be/thefish/alchemyst-go/engine/gamemap"
"lab.zaar.be/thefish/alchemyst-go/engine/gamemap/mapgens"
"lab.zaar.be/thefish/alchemyst-go/engine/gamestate"
@ -44,8 +45,8 @@ var State = gamestate.GameState{
Exit: make(chan struct{}, 1),
Input: make(chan string, 1),
RawInput: make(chan int, 1),
FovRecompute: make(chan struct{},1),
Redraw: make(chan struct{},1),
FovRecompute: make(chan struct{}, 1),
Redraw: make(chan struct{}, 1),
}
func main() {
@ -66,35 +67,58 @@ func main() {
State.Level = level
vp := mainwindow.NewViewPort(30, 0, 70, 47, mw.GetLayer("base"))
screenMgr := types.NewScreenManager(mainCtx)
screenMgr.AddScreen("title", &screens.TitleScreen{})
screenMgr.AddScreen("game", screens.NewGameScreen(mw, &State, vp))
screenMgr.SetScreenByName("game")
//fixme
player := &mob.Player{
Mob: mob.Mob{
Appearance: &types.Appearance{
Glyph: &types.PlainGlyphHolder{"@"},
ColorSet: &types.TileColorSet{
Fg: &types.PlainColorHolder{255, 255, 255, 255},
},
},
Coords: rooms[0].Center,
BlocksPass: true,
},
}
State.Player = player
//player := &mob.Player{
// Mob: mob.Mob{
// Appearance: &types.Appearance{
// Glyph: &types.PlainGlyphHolder{"@"},
// ColorSet: &types.TileColorSet{
// Fg: &types.PlainColorHolder{255, 255, 255, 255},
// },
// },
// Coords: rooms[0].Center,
// BlocksPass: true,
// },
//}
//State.Player = player
vp.PlayerCoords = player.Coords
vp.Render(&State)
//vp.PlayerCoords = player.Coords
//vp.Render(&State)
go decodeInput(mainCtx, mw.GetLayer("base"))
go vp.Listen(State)
controller := ecs.NewController()
controller.MapComponentClass("coords", types.Coords{})
controller.MapComponentClass("appearance", types.Appearance{})
controller.MapComponentClass("mob", mob.Mob{})
player := controller.CreateEntity([]ecs.Component{})
controller.AddComponent(player, &types.Appearance{
Glyph: &types.PlainGlyphHolder{"@"},
ColorSet: &types.TileColorSet{
Fg: &types.PlainColorHolder{255, 255, 255, 255},
},
})
controller.AddComponent(player, rooms[0].Center) //implicit Coords
render := mob.MobRenderSystem{EntityController: controller}
controller.AddSystem(render, 1)
//but every call to bearlibterminal must be wrapped to closure and passed to mainfunc
var exit = false
for !exit {
@ -112,7 +136,7 @@ func main() {
mainCtx.Logger().Warn().Msg("quitting NOW")
exit = true
break
// не оставляйте default в бесконесчном select {} - сожрет всё CPU
// не оставляйте default в бесконесчном select {} - сожрет всё CPU
default:
screenMgr.CurrentScreen.Render()
blt.Refresh()

9
engine/ecs/component.go Normal file
View File

@ -0,0 +1,9 @@
package ecs
// ECS system by jcerise, github.com/jcerise/gogue
import "reflect"
type Component interface {
TypeOf() reflect.Type
}

247
engine/ecs/controller.go Normal file
View File

@ -0,0 +1,247 @@
package ecs
// ECS system by jcerise, github.com/jcerise/gogue
import (
"fmt"
"reflect"
"sort"
)
type Controller struct {
systems map[reflect.Type]System
sortedSystems map[int][]System
priorityKeys []int
nextEntityID int
components map[reflect.Type][]int
entities map[int]map[reflect.Type]Component
deadEntities []int
// The component map will keep track of what components are available
componentMap map[string]Component
}
// NewController is a convenience/constructor method to properly initialize a new processor
func NewController() *Controller {
controller := Controller{}
controller.systems = make(map[reflect.Type]System)
controller.sortedSystems = make(map[int][]System)
controller.priorityKeys = []int{}
controller.nextEntityID = 0
controller.components = make(map[reflect.Type][]int)
controller.entities = make(map[int]map[reflect.Type]Component)
controller.deadEntities = []int{}
controller.componentMap = make(map[string]Component)
return &controller
}
// Create a new entity in the world. An entity is simply a unique integer.
// If any components are provided, they will be associated with the created entity
func (c *Controller) CreateEntity(components []Component) int {
c.nextEntityID += 1
if len(components) > 0 {
for _, v := range components {
c.AddComponent(c.nextEntityID, v)
}
}
c.entities[c.nextEntityID] = make(map[reflect.Type]Component)
return c.nextEntityID
}
// DeleteEntity removes an entity, all component instances attached to that entity, and any components associations with
// that entity
func (c *Controller) DeleteEntity(entity int) {
// First, delete all the component associations for the entity to be removed
for k, _ := range c.entities[entity] {
c.RemoveComponent(entity, k)
}
// Then, delete the entity itself. The components have already been removed and disassociated with it, so a simple
// delete will do here
delete(c.entities, entity)
}
// MapComponent registers a component with the controller. This map of components gives the controller access to the
// valid components for a game system, and allows for dynamic loading of components from the data loader.
func (c *Controller) MapComponentClass(componentName string, component Component) {
// TODO: Possible to overwrite existing components with old name...
c.componentMap[componentName] = component
}
// GetMappedComponentClass returns a component class based on the name it was registered under. This allows for dyanamic
// mapping of components to entities, for example, from the data loader.
func (c *Controller) GetMappedComponentClass(componentName string) Component {
if _, ok := c.componentMap[componentName]; ok {
return c.componentMap[componentName]
} else {
// TODO: Add better (read: actual) error handling here
fmt.Printf("Component[%s] not registered on Controller.\n", componentName)
return nil
}
}
// AddComponent adds a component to an entity. The component is added to the global list of components for the
// processor, and also directly associated with the entity itself. This allows for flexible checking of components,
// as you can check which entites are associated with a component, and vice versa.
func (c *Controller) AddComponent(entity int, component Component) {
// First, get the type of the component
componentType := reflect.TypeOf(component)
// Record that the component type is associated with the entity.
c.components[componentType] = append(c.components[componentType], entity)
// Now, check to see if the entity is already tracked in the controller entity list. If it is not, add it, and
// associate the component with it
if _, ok := c.entities[entity]; !ok {
c.entities[entity] = make(map[reflect.Type]Component)
}
c.entities[entity][componentType] = component
}
// HasComponent checks a given entity to see if it has a given component associated with it
func (c *Controller) HasComponent(entity int, componentType reflect.Type) bool {
if _, ok := c.entities[entity][componentType]; ok {
return true
} else {
return false
}
}
// GetComponent returns the component instance for a component type, if one exists for the provided entity
func (c *Controller) GetComponent(entity int, componentType reflect.Type) Component {
// Check the given entity has the provided component
if c.HasComponent(entity, componentType) {
return c.entities[entity][componentType]
}
return nil
}
// GetEntity gets a specific entity, and all of its component instances
func (c *Controller) GetEntity(entity int) map[reflect.Type]Component {
for i, _ := range c.entities {
if i == entity {
return c.entities[entity]
}
}
return nil
}
// GetEntities returns a map of all entities and their component instances
func (c *Controller) GetEntities() map[int]map[reflect.Type]Component {
return c.entities
}
// GetEntitiesWithComponent returns a list of all entities with a given component attached
// TODO: Allow for passing a list of components
func (c *Controller) GetEntitiesWithComponent(componentType reflect.Type) []int {
entitiesWithComponent := make([]int, 0)
for entity := range c.entities {
if c.HasComponent(entity, componentType) {
entitiesWithComponent = append(entitiesWithComponent, entity)
}
}
return entitiesWithComponent
}
// UpdateComponent updates a component on an entity with a new version of the same component
func (c *Controller) UpdateComponent(entity int, componentType reflect.Type, newComponent Component) int {
// First, remove the component in question (Don't actually update things, but rather remove and replace)
c.RemoveComponent(entity, componentType)
// Next, replace the removed component with the updated one
c.AddComponent(entity, newComponent)
return entity
}
// DeleteComponent will delete a component instance from an entity, based on component type. It will also remove the
// association between the component and the entity, and remove the component from the processor completely if no
// other entities are using it.
func (c *Controller) RemoveComponent(entity int, componentType reflect.Type) int {
// Find the index of the entity to operate on in the components slice
index := -1
for i, v := range c.components[componentType] {
if (v == entity) {
index = i
}
}
// If the component was found on the entity, remove the association between the component and the entity
if index != -1 {
c.components[componentType] = append(c.components[componentType][:index], c.components[componentType][index+1:]...)
// If this was the last entity associated with the component, remove the component entry as well
if len(c.components[componentType]) == 0 {
delete(c.components, componentType)
}
}
// Now, remove the component instance from the entity
delete(c.entities[entity], componentType)
return entity
}
// AddSystem registers a system to the controller. A priority can be provided, and systems will be processed in
// numeric order, low to high. If multiple systems are registered as the same priority, they will be randomly run within
// that priority group.
func (c *Controller) AddSystem(system System, priority int) {
systemType := reflect.TypeOf(system)
if _, ok := c.systems[systemType]; !ok {
// A system of this type has not been added yet, so add it to the systems list
c.systems[systemType] = system
// Now, append the system to a special list that will be used for sorting by priority
if !IntInSlice(priority, c.priorityKeys) {
c.priorityKeys = append(c.priorityKeys, priority)
}
c.sortedSystems[priority] = append(c.sortedSystems[priority], system)
sort.Ints(c.priorityKeys)
} else {
fmt.Printf("A system of type %v was already added to the controller %v!", systemType, c)
}
}
// Process kicks off system processing for all systems attached to the controller. Systems will be processed in the
// order they are found, or if they have a priority, in priority order. If there is a mix of systems with priority and
// without, systems with priority will be processed first (in order).
func (c *Controller) Process(excludedSystems []reflect.Type) {
for _, key := range c.priorityKeys {
for _, system := range c.sortedSystems[key] {
systemType := reflect.TypeOf(system)
// Check if the current system type was marked as excluded on this call. If it was, not process it.
if !TypeInSlice(systemType, excludedSystems) {
system.Process()
}
}
}
}
// HasSystem checks the controller to see if it has a given system associated with it
func (c *Controller) HasSystem(systemType reflect.Type) bool {
if _, ok := c.systems[systemType]; ok {
return true
} else {
return false
}
}
// ProcessSystem allows for on demand processing of individual systems, rather than processing all at once via Process
func (c *Controller) ProcessSystem(systemType reflect.Type) {
if c.HasSystem(systemType) {
system := c.systems[systemType]
system.Process()
}
}

View File

@ -1,3 +1,9 @@
package ecs
// ECS system by jcerise, github.com/jcerise/gogue
type Entity int
func (e *Entity) HasComponent(c Component) bool {
return true
}

7
engine/ecs/system.go Normal file
View File

@ -0,0 +1,7 @@
package ecs
// ECS system by jcerise, github.com/jcerise/gogue
type System interface {
Process()
}

View File

@ -0,0 +1,87 @@
package ecs
type SystemMessageType struct {
Name string
}
type SystemMessage struct {
MessageType SystemMessageType
Originator System
MessageContent map[string]string
}
// SystemMessageQueue is a super simple way of messaging between systems. Essentially, it is nothing more than a list of
// messages. Each message has a type, and an originator. Each system can "subscribe" to a type of message, which
// basically just means that it will check the queue for any messages of that type before it does anything else.
// Messages can contain a map of information, which each system that creates messages of that type, and those that
// subscribe to it should know how to handle any information contained in the message. Ideally, the message queue will
// be cleared out occasionally, either by the subscribing systems, or the game loop. Pretty simple for now, but should
// solve a subset of problems nicely.
type SystemMessageQueue struct {
Messages map[System][]SystemMessage
Subscriptions map[System][]SystemMessageType
}
func InitializeSystemMessageQueue() *SystemMessageQueue {
smq := SystemMessageQueue{}
smq.Messages = make(map[System][]SystemMessage)
smq.Subscriptions = make(map[System][]SystemMessageType)
return &smq
}
// BroadcastMessage appends a system message onto the games SystemMessageQueue, allowing it to consumed by a service
// subscribes to the MessageType.
func (smq *SystemMessageQueue) BroadcastMessage(messageType SystemMessageType, messageContent map[string]string, originator System) {
newMessage := SystemMessage{MessageType: messageType, MessageContent: messageContent, Originator: originator}
// Find all subscriptions to this message type, and add this message to the subscribers message queue
for subscribedSystem, typeList := range smq.Subscriptions {
if MessageTypeInSlice(messageType, typeList) {
smq.Messages[subscribedSystem] = append(smq.Messages[subscribedSystem], newMessage)
}
}
}
// GetSubscribedMessages returns a list of SystemMessages that have messageType. Can return an empty list
func (smq *SystemMessageQueue) GetSubscribedMessages(system System) []SystemMessage {
messages := []SystemMessage{}
for _, message := range smq.Messages[system] {
messages = append(messages, message)
}
return messages
}
// DeleteMessages deletes a processed message from the queue (for example, if the event has been processed)
func (smq *SystemMessageQueue) DeleteMessages(messageName string, system System) {
modifiedQueue := smq.Messages[system]
for index, message := range smq.Messages[system] {
if message.MessageType.Name == messageName {
modifiedQueue[index] = modifiedQueue[len(modifiedQueue)-1]
modifiedQueue = modifiedQueue[:len(modifiedQueue)-1]
}
}
smq.Messages[system] = modifiedQueue
}
//MessageTypeInSlice will return true if the MessageType provided is present in the slice provided, false otherwise
func MessageTypeInSlice(a SystemMessageType, list []SystemMessageType) bool {
for _, b := range list {
if b == a {
return true
}
}
return false
}
//MessageTypeInSliceOfMessages will return true if the MessageType provided is present in the slice provided, false otherwise
func MessageTypeInSliceOfMessages(a SystemMessageType, list []SystemMessage) bool {
for _, b := range list {
if b.MessageType == a {
return true
}
}
return false
}

25
engine/ecs/util.go Normal file
View File

@ -0,0 +1,25 @@
package ecs
// ECS system by jcerise, github.com/jcerise/gogue
import "reflect"
// IntInSlice will return true if the integer value provided is present in the slice provided, false otherwise.
func IntInSlice(a int, list []int) bool {
for _, b := range list {
if b == a {
return true
}
}
return false
}
// TypeInSlice will return true if the reflect.Type provided is present in the slice provided, false otherwise.
func TypeInSlice(a reflect.Type, list []reflect.Type) bool {
for _, b := range list {
if b == a {
return true
}
}
return false
}

View File

@ -17,7 +17,7 @@ type Level struct {
Name string
Branch string
Depth int
Objects []ecs.Entity
Objects *[]ecs.Entity
Tiles []*Tile
}

View File

@ -2,7 +2,9 @@ package mob
import (
"fmt"
"lab.zaar.be/thefish/alchemyst-go/engine/gamemap"
"lab.zaar.be/thefish/alchemyst-go/engine/types"
"reflect"
)
type Mob struct {
@ -11,8 +13,14 @@ type Mob struct {
BlocksPass bool
}
func (m *Mob) Walk(dx, dy int) {
m.Coords = types.Coords{m.X + dx, m.Y + dy}
func (m *Mob) Walk(level *gamemap.Level, dx, dy int) {
newCoords := types.Coords{m.X + dx, m.Y + dy}
if level.GetTile(newCoords).BlocksPass {
return
}
if level.Objects.At(newCoords).HasComponent("block_pass") {
}
fmt.Printf("new coords: %d, %d\n", m.Coords.X, m.Coords.Y)
}
@ -22,4 +30,8 @@ func (m *Mob) Render() {
func (m *Mob) MoveToCoords(c types.Coords) {
}
func (mob Mob) TypeOf() reflect.Type {
return reflect.TypeOf(mob)
}

View File

@ -0,0 +1,29 @@
package mob
import (
"lab.zaar.be/thefish/alchemyst-go/engine/ecs"
"lab.zaar.be/thefish/alchemyst-go/engine/types"
)
type MobRenderSystem struct {
EntityController *ecs.Controller
}
func (mrs MobRenderSystem) Process(){
for e := range mrs.EntityController.GetEntities() {
if mrs.EntityController.HasComponent(e, types.Coords{}.TypeOf()) &&
mrs.EntityController.HasComponent(e, types.Appearance{}.TypeOf()) {
pos := mrs.EntityController.GetComponent(e, types.Coords{}.TypeOf()).(types.Coords)
appearance := mrs.EntityController.GetComponent(e, types.Appearance{}.TypeOf()).(types.Appearance)
// Clear the cell this entity occupies, so it is the only glyph drawn there
for i := 0; i <= 2; i++ {
//fixme
gogue.ClearArea(pos.X, pos.Y, 1, 1, i)
}
//fixme
gogue.PrintGlyph(pos.X, pos.Y, appearance.Glyph, "", appearance.Layer)
}
}
}

View File

@ -21,28 +21,28 @@ func (ts *GameScreen) HandleInput(input string) {
//ts.state.Do(func(){
switch input {
case "Up", "k", "8":
ts.state.Player.Walk(0, -1)
ts.state.Player.Walk(ts.state.Level, 0, -1)
break
case "Down", "j", "2":
ts.state.Player.Walk(0, 1)
ts.state.Player.Walk(ts.state.Level,0, 1)
break
case "Left", "h", "4":
ts.state.Player.Walk(-1, 0)
ts.state.Player.Walk(ts.state.Level,-1, 0)
break
case "Right", "l", "6":
ts.state.Player.Walk(1, 0)
ts.state.Player.Walk(ts.state.Level,1, 0)
break
case "y", "7":
ts.state.Player.Walk(-1, -1)
ts.state.Player.Walk(ts.state.Level,-1, -1)
break
case "u", "9":
ts.state.Player.Walk(1, -1)
ts.state.Player.Walk(ts.state.Level,1, -1)
break
case "b", "1":
ts.state.Player.Walk(-1, 1)
ts.state.Player.Walk(ts.state.Level,-1, 1)
break
case "n", "3":
ts.state.Player.Walk(1, 1)
ts.state.Player.Walk(ts.state.Level,1, 1)
break
default:
ts.mw.GetLayer("base").ClearArea(0, 3, 40, 1)

View File

@ -3,6 +3,7 @@ package types
import (
"github.com/gammazero/deque"
"lab.zaar.be/thefish/alchemyst-go/util"
"reflect"
)
import blt "lab.zaar.be/thefish/bearlibterminal"
@ -78,7 +79,6 @@ type Appearance struct {
ColorSet *TileColorSet `json:"colorSet"`
}
func SingleColorRing(colorValue uint8) *cdeque {
c := &cdeque{}
c.PushBack(colorValue)
@ -104,4 +104,8 @@ func FillColorRing(colorValue uint8, minGlow, maxGlow, step int) *cdeque {
c.PushBack(uint8(v))
}
return c
}
func (app Appearance) TypeOf() reflect.Type {
return reflect.TypeOf(app)
}

View File

@ -1,11 +1,18 @@
package types
import "math"
import (
"math"
"reflect"
)
type Coords struct {
X, Y int
}
func (сс Coords) TypeOf() reflect.Type {
return reflect.TypeOf(сс)
}
func (c *Coords) Get() (int, int) {
return c.X, c.Y
}