resolve conflicts
This commit is contained in:
		
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -1,4 +1,5 @@ | |||||||
| .idea/* | .idea/* | ||||||
| tmp/* | tmp/* | ||||||
| logs/* | logs/* | ||||||
|  | dist/* | ||||||
| *.swp | *.swp | ||||||
|  | |||||||
							
								
								
									
										21
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										21
									
								
								Makefile
									
									
									
									
									
								
							| @ -2,26 +2,27 @@ PROJECT_NAME := "alchemyst-go" | |||||||
| PKG := "lab.zaar.be/thefish/$(PROJECT_NAME)" | PKG := "lab.zaar.be/thefish/$(PROJECT_NAME)" | ||||||
| GO=$(shell which go) | GO=$(shell which go) | ||||||
| CWD=$(shell pwd) | CWD=$(shell pwd) | ||||||
|  | DISTFOLDER=$(CWD)/dist | ||||||
|  | CP=$(shell cp) | ||||||
| GLIDE=$(shell which glide) | GLIDE=$(shell which glide) | ||||||
|  |  | ||||||
| LDFLAGS="-X \"main.versionInfo=${PKG_VER}-${CI_PIPELINE_ID} built at $(shell date) on $(shell hostname) with $(shell go version)\"" | LDFLAGS="-X \"main.versionInfo=${PKG_VER}-${CI_PIPELINE_ID} built at $(shell date) on $(shell hostname) with $(shell go version)\"" | ||||||
|  |  | ||||||
| .PHONY: all build test get-dep | .PHONY: all build test | ||||||
|  |  | ||||||
| all: build test | all: build test | ||||||
|  |  | ||||||
| #build: build.spec build.server | #build: build.spec build.server | ||||||
| build: build.server | build: build.game | ||||||
|  |  | ||||||
| #build.spec: | build.deps: | ||||||
| #	$(CWD)/bin/oapi-codegen --generate types,spec -o $(CWD)/api/inner/inner.gen.go $(CWD)/public/schema/api.yaml | 	GIT_SSL_NO_VERIFY=true $(GO) mod vendor | ||||||
|  |  | ||||||
| build.server: | build.game: | ||||||
| 	$(GO) build -ldflags $(LDFLAGS) -o $(CWD)/bin/game $(CWD)/cmd/game/main.go | 	cp $(CWD)/vendor/lab.zaar.be/thefish/bearlibterminal/libBearLibTerminal.so $(DISTFOLDER) | ||||||
|  | 	cp $(CWD)/config.json $(DISTFOLDER) | ||||||
|  | 	cp -r $(CWD)/resources $(DISTFOLDER) | ||||||
|  | 	$(GO) build -ldflags $(LDFLAGS) -o $(DISTFOLDER)/game $(CWD)/cmd/game/main.go | ||||||
|  |  | ||||||
| test: | test: | ||||||
| 	$(GO) test -v $(go list ./... | grep -v /vendor/) | 	$(GO) test -v $(go list ./... | grep -v /vendor/) | ||||||
|  |  | ||||||
| get-dep: |  | ||||||
| 	GIT_SSL_NO_VERIFY=true go mod vendor |  | ||||||
|  |  | ||||||
|  | |||||||
							
								
								
									
										100
									
								
								cmd/game/main.go
									
									
									
									
									
								
							
							
						
						
									
										100
									
								
								cmd/game/main.go
									
									
									
									
									
								
							| @ -3,8 +3,13 @@ package main | |||||||
| import ( | import ( | ||||||
| 	"github.com/rs/zerolog" | 	"github.com/rs/zerolog" | ||||||
| 	"github.com/rs/zerolog/log" | 	"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" | ||||||
| 	"lab.zaar.be/thefish/alchemyst-go/engine/gamemap/mapgens" | 	"lab.zaar.be/thefish/alchemyst-go/engine/gamemap/mapgens" | ||||||
|  | 	"lab.zaar.be/thefish/alchemyst-go/engine/gamestate" | ||||||
|  | 	"lab.zaar.be/thefish/alchemyst-go/engine/mob" | ||||||
|  | 	"lab.zaar.be/thefish/alchemyst-go/engine/screens" | ||||||
|  | 	"lab.zaar.be/thefish/alchemyst-go/engine/types" | ||||||
| 	"lab.zaar.be/thefish/alchemyst-go/ui" | 	"lab.zaar.be/thefish/alchemyst-go/ui" | ||||||
| 	"lab.zaar.be/thefish/alchemyst-go/ui/mainwindow" | 	"lab.zaar.be/thefish/alchemyst-go/ui/mainwindow" | ||||||
| 	"lab.zaar.be/thefish/alchemyst-go/util" | 	"lab.zaar.be/thefish/alchemyst-go/util" | ||||||
| @ -22,23 +27,6 @@ func init() { | |||||||
| 	runtime.LockOSThread() | 	runtime.LockOSThread() | ||||||
| } | } | ||||||
|  |  | ||||||
| type GameState struct { |  | ||||||
| 	mainfunc chan func() |  | ||||||
| 	exit     chan struct{} |  | ||||||
| 	input    chan string |  | ||||||
| 	rawInput chan int |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // do runs f on the main thread. |  | ||||||
| func (*GameState) Do(f func()) { |  | ||||||
| 	done := make(chan struct{}, 1) |  | ||||||
| 	State.mainfunc <- func() { |  | ||||||
| 		f() |  | ||||||
| 		done <- struct{}{} |  | ||||||
| 	} |  | ||||||
| 	<-done |  | ||||||
| } |  | ||||||
|  |  | ||||||
| //we can run logic in separate goroutines | //we can run logic in separate goroutines | ||||||
| // | // | ||||||
| //    go doSometing(State,...) | //    go doSometing(State,...) | ||||||
| @ -47,17 +35,18 @@ func (*GameState) Do(f func()) { | |||||||
| // func doSomething(State main.GameState, args...) { | // func doSomething(State main.GameState, args...) { | ||||||
| // 		... | // 		... | ||||||
| // 			State.Do(func() { | // 			State.Do(func() { | ||||||
| //				...do stuff in main thread, ie render something | //				...do stuff in main thread | ||||||
| //			}) | //			}) | ||||||
| // 		... | // 		... | ||||||
| // } | // } | ||||||
| // Use this trick CAREFULLY, cause closures may cause memleaks |  | ||||||
|  |  | ||||||
| var State = GameState{ | var State = gamestate.GameState{ | ||||||
| 	mainfunc: make(chan func()), | 	Mainfunc:     make(chan func()), | ||||||
| 	exit:     make(chan struct{}, 1), | 	Exit:         make(chan struct{}, 1), | ||||||
| 	input:    make(chan string, 1), | 	Input:        make(chan string, 1), | ||||||
| 	rawInput: make(chan int, 1), | 	RawInput:     make(chan int, 1), | ||||||
|  | 	FovRecompute: make(chan struct{}, 1), | ||||||
|  | 	Redraw:       make(chan struct{}, 1), | ||||||
| } | } | ||||||
|  |  | ||||||
| func main() { | func main() { | ||||||
| @ -79,29 +68,53 @@ func main() { | |||||||
| 	vp.Render() | 	vp.Render() | ||||||
|  |  | ||||||
| 	go decodeInput(mainCtx, mw.GetLayer("base")) | 	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 | 	//but every call to bearlibterminal must be wrapped to closure and passed to mainfunc | ||||||
| 	var exit = false | 	var exit = false | ||||||
| 	for !exit { | 	for !exit { | ||||||
|  |  | ||||||
| 		select { | 		select { | ||||||
| 		case State.rawInput <- ui.ReadKeyCode(): | 		case State.RawInput <- ui.ReadKeyCode(): | ||||||
| 			break | 			break | ||||||
| 		case pressed := <-State.input: | 		case pressed := <-State.Input: | ||||||
| 			mw.GetLayer("base").ClearArea(0, 3, 40, 1) | 			screenMgr.CurrentScreen.HandleInput(pressed) | ||||||
| 			mw.GetLayer("base").Print(1, 3, "Key: "+pressed) |  | ||||||
| 			mw.GetLayer("base").Print(1, 6, "█") |  | ||||||
| 			break | 			break | ||||||
| 			//case f := <-State.mainfunc: | 			//case f := <-State.mainfunc: | ||||||
| 			//	f() | 			//	f() | ||||||
| 			//	break | 			//	break | ||||||
| 		case <-State.exit: | 		case <-State.Exit: | ||||||
| 			mainCtx.Logger().Warn().Msg("quitting NOW") | 			mainCtx.Logger().Warn().Msg("quitting NOW") | ||||||
| 			exit = true | 			exit = true | ||||||
| 			break | 			break | ||||||
| 			// не оставляйте default в бесконесчном select {} - сожрет всё CPU | 			// не оставляйте default в бесконесчном select {} - сожрет всё CPU | ||||||
| 		default: | 		default: | ||||||
| 			vp.Render() | 			screenMgr.CurrentScreen.Render() | ||||||
| 			blt.Refresh() | 			blt.Refresh() | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| @ -117,23 +130,21 @@ func setupLayers(mainwindow *mainwindow.MainWindow) { | |||||||
|  |  | ||||||
| func decodeInput(ctx util.ClientCtx, baseLayer *mainwindow.Layer) { | func decodeInput(ctx util.ClientCtx, baseLayer *mainwindow.Layer) { | ||||||
| 	var exit = false | 	var exit = false | ||||||
| 	//for some reason blt's input queue gots spammed with 0xE0 on start. | 	var waitForWCspam = true | ||||||
| 	//with this crutch we can wait out this WindowCloseEvent burst. | 	for !exit { | ||||||
| 	var waitForStartingWindowCloseBurst = true |  | ||||||
| 	for !exit{ |  | ||||||
| 		select { | 		select { | ||||||
| 		case keycode := <-State.rawInput: | 		case keycode := <-State.RawInput: | ||||||
| 			if keycode == blt.TK_NONE { | 			if keycode == blt.TK_NONE { | ||||||
| 				continue | 				continue | ||||||
| 			} | 			} | ||||||
| 			if keycode == blt.TK_CLOSE && !waitForStartingWindowCloseBurst { | 			if keycode == blt.TK_CLOSE && !waitForWCspam { | ||||||
| 				ctx.Logger().Warn().Msg("exiting on window close...") | 				ctx.Logger().Warn().Msg("exiting on window close...") | ||||||
| 				State.exit <- struct{}{} | 				State.Exit <- struct{}{} | ||||||
| 				ctx.Logger().Warn().Msg("...done") | 				ctx.Logger().Warn().Msg("...done") | ||||||
| 				return | 				return | ||||||
| 			} | 			} | ||||||
| 			var pressed= "" | 			var pressed = "" | ||||||
| 			var isModifier, _= util.InArray(keycode, modifiers) | 			var isModifier, _ = util.InArray(keycode, modifiers) | ||||||
| 			if !isModifier { | 			if !isModifier { | ||||||
|  |  | ||||||
| 				pressed = ui.Scancodemap[keycode] | 				pressed = ui.Scancodemap[keycode] | ||||||
| @ -150,6 +161,7 @@ func decodeInput(ctx util.ClientCtx, baseLayer *mainwindow.Layer) { | |||||||
|  |  | ||||||
| 				//global hotkeys | 				//global hotkeys | ||||||
| 				switch pressed { | 				switch pressed { | ||||||
|  | 				//fixme testing only | ||||||
| 				case "F10": | 				case "F10": | ||||||
| 					State.Do(func() { | 					State.Do(func() { | ||||||
| 						blt.Set("window: size=100x47; font: ./resources/fonts-ttf/UbuntuMono-R.ttf, size=11;") | 						blt.Set("window: size=100x47; font: ./resources/fonts-ttf/UbuntuMono-R.ttf, size=11;") | ||||||
| @ -158,13 +170,15 @@ func decodeInput(ctx util.ClientCtx, baseLayer *mainwindow.Layer) { | |||||||
| 					fallthrough | 					fallthrough | ||||||
| 				case "Escape": | 				case "Escape": | ||||||
| 					ctx.Logger().Info().Msg("exiting on quit command...") | 					ctx.Logger().Info().Msg("exiting on quit command...") | ||||||
| 					State.exit <- struct{}{} | 					State.Exit <- struct{}{} | ||||||
| 					ctx.Logger().Info().Msg("...done") | 					ctx.Logger().Info().Msg("...done") | ||||||
| 					exit = true | 					exit = true | ||||||
| 					return | 					return | ||||||
| 				default: | 				default: | ||||||
| 					waitForStartingWindowCloseBurst = false | 					if pressed != "" { | ||||||
| 					State.input <- pressed | 						waitForWCspam = false; | ||||||
|  | 						State.Input <- pressed | ||||||
|  | 					} | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  | |||||||
| @ -1,9 +1,10 @@ | |||||||
| { | { | ||||||
|   "version": "0.0.0.1", |   "version": "0.0.0.1", | ||||||
|   "title" : "Test Go+BLT App", |   "title" : "Alchemyst", | ||||||
|   "sizeX": 100, |   "sizeX": 100, | ||||||
|   "sizeY": 47, |   "sizeY": 47, | ||||||
|   "fpsLimit" : 60, |   "fpsLimit" : 60, | ||||||
|   "font": "./resources/fonts-ttf/LiberationMono-Bold.ttf", |   "font": "./resources/fonts-ttf/LiberationMono-Bold.ttf", | ||||||
|  |   "fontSize": "8x12", | ||||||
|   "verbosity": "debug" |   "verbosity": "debug" | ||||||
| } | } | ||||||
							
								
								
									
										0
									
								
								dist/.gitkeep
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								dist/.gitkeep
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
								
								
									
										9
									
								
								engine/ecs/component.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								engine/ecs/component.go
									
									
									
									
									
										Normal 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
									
								
							
							
						
						
									
										247
									
								
								engine/ecs/controller.go
									
									
									
									
									
										Normal 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() | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -1,3 +1,9 @@ | |||||||
| package ecs | package ecs | ||||||
|  |  | ||||||
|  | // ECS system by jcerise, github.com/jcerise/gogue | ||||||
|  |  | ||||||
| type Entity int | type Entity int | ||||||
|  |  | ||||||
|  | func (e *Entity) HasComponent(c Component) bool { | ||||||
|  | 	return true | ||||||
|  | } | ||||||
							
								
								
									
										7
									
								
								engine/ecs/system.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								engine/ecs/system.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,7 @@ | |||||||
|  | package ecs | ||||||
|  |  | ||||||
|  | // ECS system by jcerise, github.com/jcerise/gogue | ||||||
|  |  | ||||||
|  | type System interface { | ||||||
|  | 	Process() | ||||||
|  | } | ||||||
							
								
								
									
										87
									
								
								engine/ecs/systemMessages.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								engine/ecs/systemMessages.go
									
									
									
									
									
										Normal 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
									
								
							
							
						
						
									
										25
									
								
								engine/ecs/util.go
									
									
									
									
									
										Normal 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 | ||||||
|  | } | ||||||
| @ -195,7 +195,9 @@ func (ps *precomputedShade) PrecomputeFovMap() { | |||||||
| } | } | ||||||
|  |  | ||||||
| func (ps *precomputedShade) recalc(level *gamemap.Level, initCoords types.Coords, radius int) { | func (ps *precomputedShade) recalc(level *gamemap.Level, initCoords types.Coords, radius int) { | ||||||
|  | 	for i, _ := range ps.CellList { | ||||||
|  | 		ps.CellList[i].lit = 0 | ||||||
|  | 	} | ||||||
| 	ps.originCoords = initCoords | 	ps.originCoords = initCoords | ||||||
|  |  | ||||||
| 	if radius > ps.MaxTorchRadius { | 	if radius > ps.MaxTorchRadius { | ||||||
| @ -250,6 +252,7 @@ func (ps *precomputedShade) recalc(level *gamemap.Level, initCoords types.Coords | |||||||
|  |  | ||||||
| func (ps *precomputedShade) ComputeFov(level *gamemap.Level, initCoords types.Coords, radius int) { | func (ps *precomputedShade) ComputeFov(level *gamemap.Level, initCoords types.Coords, radius int) { | ||||||
|  |  | ||||||
|  | 	level.SetAllInvisible() | ||||||
| 	ps.recalc(level, initCoords, radius) | 	ps.recalc(level, initCoords, radius) | ||||||
|  |  | ||||||
| 	for _, cell := range ps.CellList { | 	for _, cell := range ps.CellList { | ||||||
| @ -260,6 +263,7 @@ func (ps *precomputedShade) ComputeFov(level *gamemap.Level, initCoords types.Co | |||||||
| 				continue | 				continue | ||||||
| 			} | 			} | ||||||
| 			level.GetTile(cs).Visible = true | 			level.GetTile(cs).Visible = true | ||||||
|  | 			level.GetTile(cs).Explored = true | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		//light walls, crutch | 		//light walls, crutch | ||||||
| @ -270,9 +274,10 @@ func (ps *precomputedShade) ComputeFov(level *gamemap.Level, initCoords types.Co | |||||||
| 				for _, maybeNb := range ps.CellList { | 				for _, maybeNb := range ps.CellList { | ||||||
| 					if  //int(maybeNb.distance) == int(cell.distance-1) && | 					if  //int(maybeNb.distance) == int(cell.distance-1) && | ||||||
| 						maybeNb.IsAdjacentTo(&cell.Coords) && | 						maybeNb.IsAdjacentTo(&cell.Coords) && | ||||||
| 						(maybeNb.X == cell.X || maybeNb.Y == cell.Y) && | 						//(maybeNb.X == cell.X || maybeNb.Y == cell.Y) && | ||||||
| 						maybeNb.lit > 0 { //magic constant! | 						maybeNb.lit > 0 { //magic constant! | ||||||
| 						level.GetTile(cs).Visible = true | 						level.GetTile(cs).Visible = true | ||||||
|  | 						level.GetTile(cs).Explored = true | ||||||
| 					} | 					} | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
|  | |||||||
| @ -67,7 +67,7 @@ func TestPrecompShade(t *testing.T) { | |||||||
| 		if playerCoords.X == x && playerCoords.Y == y { | 		if playerCoords.X == x && playerCoords.Y == y { | ||||||
| 			return "@" | 			return "@" | ||||||
| 		} | 		} | ||||||
| 		result := level.GetTileByXY(x, y).Char | 		result := level.GetTileByXY(x, y).Glyph.GetGlyph() | ||||||
| 		if !level.GetTileByXY(x, y).Visible { | 		if !level.GetTileByXY(x, y).Visible { | ||||||
| 			result = "?" | 			result = "?" | ||||||
| 		} | 		} | ||||||
|  | |||||||
| @ -8,7 +8,7 @@ import ( | |||||||
|  |  | ||||||
| //fixme move to config | //fixme move to config | ||||||
| var mapWidth = 150 | var mapWidth = 150 | ||||||
| var mapHeight = 100 | var mapHeight = 90 | ||||||
|  |  | ||||||
|  |  | ||||||
| type Level struct { | type Level struct { | ||||||
| @ -17,7 +17,7 @@ type Level struct { | |||||||
| 	Name     string | 	Name     string | ||||||
| 	Branch   string | 	Branch   string | ||||||
| 	Depth    int | 	Depth    int | ||||||
| 	Objects  []ecs.Entity | 	Objects  *[]ecs.Entity | ||||||
| 	Tiles    []*Tile | 	Tiles    []*Tile | ||||||
| } | } | ||||||
|  |  | ||||||
| @ -49,6 +49,7 @@ func (l *Level) Put (x, y int, tileFunc interface{}) { | |||||||
|  |  | ||||||
| func NewLevel(ctx util.ClientCtx, branch string, depth int) *Level { | func NewLevel(ctx util.ClientCtx, branch string, depth int) *Level { | ||||||
| 	l := &Level{ | 	l := &Level{ | ||||||
|  | 		ctx: ctx, | ||||||
| 		Name:     branch + string(depth), | 		Name:     branch + string(depth), | ||||||
| 		Depth:    depth, | 		Depth:    depth, | ||||||
| 		Rect: types.NewRect(0,0, mapWidth, mapHeight), | 		Rect: types.NewRect(0,0, mapWidth, mapHeight), | ||||||
| @ -58,6 +59,12 @@ func NewLevel(ctx util.ClientCtx, branch string, depth int) *Level { | |||||||
| 	return l | 	return l | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (l *Level) SetAllInvisible() { | ||||||
|  | 	for idx, _ := range l.Tiles { | ||||||
|  | 		l.Tiles[idx].Visible = false | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
| type Room struct { | type Room struct { | ||||||
| 	*types.Rect | 	*types.Rect | ||||||
| 	Center types.Coords | 	Center types.Coords | ||||||
|  | |||||||
| @ -8,10 +8,10 @@ import ( | |||||||
| //fixme move to config | //fixme move to config | ||||||
| var minRoomSize = 3 | var minRoomSize = 3 | ||||||
| var maxRoomSize = 22 | var maxRoomSize = 22 | ||||||
| var maxrooms = 50 | var maxrooms = 30 | ||||||
|  |  | ||||||
| //fixme make closure to stack them | //fixme make closure to stack them | ||||||
| func DefaultGen(l *gamemap.Level) *gamemap.Level { | func DefaultGen(l *gamemap.Level) (*gamemap.Level, []*gamemap.Room) { | ||||||
|  |  | ||||||
| 	rng := util.NewRNG() | 	rng := util.NewRNG() | ||||||
|  |  | ||||||
| @ -56,53 +56,49 @@ func DefaultGen(l *gamemap.Level) *gamemap.Level { | |||||||
| 		if !failed { | 		if !failed { | ||||||
| 			rooms = append(rooms, newRoom) | 			rooms = append(rooms, newRoom) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | 		//addStairs(rooms) | ||||||
|  | 		//itemize(rooms) | ||||||
|  | 	} | ||||||
|  | 	fges := map[int]types.RectFill{ | ||||||
|  | 		1: 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() }, | ||||||
|  | 		}, | ||||||
|  |  | ||||||
|  | 		2: types.RectFill{ | ||||||
|  | 			Top:         func() *gamemap.Tile { return gamemap.NewWaterTile() }, | ||||||
|  | 			Bottom:      func() *gamemap.Tile { return gamemap.NewWaterTile() }, | ||||||
|  | 			Left:        func() *gamemap.Tile { return gamemap.NewWaterTile() }, | ||||||
|  | 			Right:       func() *gamemap.Tile { return gamemap.NewWaterTile() }, | ||||||
|  | 			BottomLeft:  func() *gamemap.Tile { return gamemap.NewWaterTile() }, | ||||||
|  | 			BottomRight: func() *gamemap.Tile { return gamemap.NewWaterTile() }, | ||||||
|  | 			TopLeft:     func() *gamemap.Tile { return gamemap.NewWaterTile() }, | ||||||
|  | 			TopRight:    func() *gamemap.Tile { return gamemap.NewWaterTile() }, | ||||||
|  | 			Body:        func() *gamemap.Tile { return gamemap.NewDeepWaterTile() }, | ||||||
|  | 		}, | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	//fillage := types.RectFill{ | 	var fillage types.RectFill | ||||||
| 	//	Top: func() *gamemap.Tile {return gamemap.NewWall()}, | 	for _, room := range rooms { | ||||||
| 	//	Bottom: func() *gamemap.Tile {return gamemap.NewWall()}, | 		fillage = fges[rng.GetWeightedEntity(map[int]int{1:10, 2:1})] | ||||||
| 	//	Left: func() *gamemap.Tile {return gamemap.NewWall()}, | 		room.Blit(fillage, l) | ||||||
| 	//	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()}, |  | ||||||
| 	//} |  | ||||||
|  |  | ||||||
| 	//fillage := types.RectFill{ |  | ||||||
| 	//	Top: func() *gamemap.Tile {return gamemap.NewWaterTile()}, |  | ||||||
| 	//	Bottom: func() *gamemap.Tile {return gamemap.NewWaterTile()}, |  | ||||||
| 	//	Left: func() *gamemap.Tile {return gamemap.NewWaterTile()}, |  | ||||||
| 	//	Right: func() *gamemap.Tile {return gamemap.NewWaterTile()}, |  | ||||||
| 	//	BottomLeft: func() *gamemap.Tile {return gamemap.NewWaterTile()}, |  | ||||||
| 	//	BottomRight: func() *gamemap.Tile {return gamemap.NewWaterTile()}, |  | ||||||
| 	//	TopLeft: func() *gamemap.Tile {return gamemap.NewWaterTile()}, |  | ||||||
| 	//	TopRight: func() *gamemap.Tile {return gamemap.NewWaterTile()}, |  | ||||||
| 	//	Body: func() *gamemap.Tile {return gamemap.NewDeepWaterTile()}, |  | ||||||
| 	//} |  | ||||||
|  |  | ||||||
| 	fillage := types.RectFill{ |  | ||||||
| 		Top: func() *gamemap.Tile {return gamemap.NewFloor()}, |  | ||||||
| 		Bottom: func() *gamemap.Tile {return gamemap.NewFloor()}, |  | ||||||
| 		Left: func() *gamemap.Tile {return gamemap.NewFloor()}, |  | ||||||
| 		Right: func() *gamemap.Tile {return gamemap.NewFloor()}, |  | ||||||
| 		BottomLeft: func() *gamemap.Tile {return gamemap.NewFloor()}, |  | ||||||
| 		BottomRight: func() *gamemap.Tile {return gamemap.NewFloor()}, |  | ||||||
| 		TopLeft: func() *gamemap.Tile {return gamemap.NewFloor()}, |  | ||||||
| 		TopRight: func() *gamemap.Tile {return gamemap.NewFloor()}, |  | ||||||
| 		Body: func() *gamemap.Tile {return gamemap.NewFloor()}, |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  |  | ||||||
| 	for idx, room := range rooms { | 	for idx, room := range rooms { | ||||||
| 		room.Blit(fillage, l) |  | ||||||
| 		if idx > 0 { | 		if idx > 0 { | ||||||
| 			connectRooms(l, room, rooms[idx-1], fillage, rng.Range(0,1)) | 			connectRooms(l, room, rooms[idx-1], fillage, rng.Range(0,1)) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return l | 	return l, rooms | ||||||
| } | } | ||||||
|  |  | ||||||
| func connectRooms (l *gamemap.Level, room, otherRoom *gamemap.Room, fillage types.RectFill, toss int) { | func connectRooms (l *gamemap.Level, room, otherRoom *gamemap.Room, fillage types.RectFill, toss int) { | ||||||
|  | |||||||
| @ -1,69 +1,8 @@ | |||||||
| package gamemap | package gamemap | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"github.com/gammazero/deque" | 	. "lab.zaar.be/thefish/alchemyst-go/engine/types" | ||||||
| 	"lab.zaar.be/thefish/alchemyst-go/util" |  | ||||||
| ) | ) | ||||||
| import blt "lab.zaar.be/thefish/bearlibterminal" |  | ||||||
|  |  | ||||||
| type ColorHolder interface { |  | ||||||
| 	GetColor() uint32 |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type cdeque struct { |  | ||||||
| 	deque.Deque |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (c *cdeque) Next() uint8 { |  | ||||||
| 	c.Rotate(1) |  | ||||||
| 	return c.Front().(uint8) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type DanceColorHolder struct { |  | ||||||
| 	A uint8 |  | ||||||
| 	R *cdeque |  | ||||||
| 	G *cdeque |  | ||||||
| 	B *cdeque |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (chd *DanceColorHolder) GetColor() uint32 { |  | ||||||
| 	return blt.ColorFromARGB( |  | ||||||
| 		chd.A, |  | ||||||
| 		chd.R.Next(), |  | ||||||
| 		chd.G.Next(), |  | ||||||
| 		chd.B.Next(), |  | ||||||
| 	) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type PlainColorHolder struct { |  | ||||||
| 	A uint8 |  | ||||||
| 	R uint8 |  | ||||||
| 	G uint8 |  | ||||||
| 	B uint8 |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (chb *PlainColorHolder) GetColor() uint32 { |  | ||||||
| 	return blt.ColorFromARGB( |  | ||||||
| 		chb.A, |  | ||||||
| 		chb.R, |  | ||||||
| 		chb.G, |  | ||||||
| 		chb.B, |  | ||||||
| 	) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type TileColorSet struct { |  | ||||||
| 	Fg     ColorHolder |  | ||||||
| 	Bg     ColorHolder |  | ||||||
| 	DarkFg ColorHolder |  | ||||||
| 	DarkBg ColorHolder |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type Appearance struct { |  | ||||||
| 	Char     string        `json:"char"` |  | ||||||
| 	ColorSet *TileColorSet `json:"colorSet"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| var crng = util.NewRNG() |  | ||||||
|  |  | ||||||
| type Tile struct { | type Tile struct { | ||||||
| 	*Appearance `json:"app"` | 	*Appearance `json:"app"` | ||||||
| @ -78,12 +17,11 @@ type Tile struct { | |||||||
| } | } | ||||||
|  |  | ||||||
| func (t *Tile) GetChar() string { | func (t *Tile) GetChar() string { | ||||||
| 	return t.Char | 	return t.Glyph.GetGlyph() | ||||||
| } | } | ||||||
|  |  | ||||||
| func (t *Tile) GetRawColor() uint32 { | func (t *Tile) GetRawColor() uint32 { | ||||||
| 	//if !t.Visible { | 	if t.Visible { | ||||||
| 	if !t.Visible { |  | ||||||
| 		return t.Appearance.ColorSet.Fg.GetColor() | 		return t.Appearance.ColorSet.Fg.GetColor() | ||||||
| 	} else { | 	} else { | ||||||
| 		return t.Appearance.ColorSet.DarkFg.GetColor() | 		return t.Appearance.ColorSet.DarkFg.GetColor() | ||||||
| @ -91,7 +29,6 @@ func (t *Tile) GetRawColor() uint32 { | |||||||
| } | } | ||||||
|  |  | ||||||
| func (t *Tile) GetRawBgColor() uint32 { | func (t *Tile) GetRawBgColor() uint32 { | ||||||
| 	//if !t.Visible { |  | ||||||
| 	if t.Visible { | 	if t.Visible { | ||||||
| 		return t.Appearance.ColorSet.Bg.GetColor() | 		return t.Appearance.ColorSet.Bg.GetColor() | ||||||
| 	} else { | 	} else { | ||||||
| @ -99,38 +36,6 @@ func (t *Tile) GetRawBgColor() uint32 { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func singleColorRing(colorValue uint8) *cdeque { |  | ||||||
| 	c := &cdeque{} |  | ||||||
| 	c.PushBack(colorValue) |  | ||||||
| 	return c |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func fillColorRing(colorValue uint8, minGlow, maxGlow, step int) *cdeque { |  | ||||||
| 	q := make([]uint8, 0) |  | ||||||
| 	color := int(colorValue) |  | ||||||
| 	for color < maxGlow { |  | ||||||
| 		q = append(q, uint8(color)) |  | ||||||
| 		color = crng.Range(1, step) + color |  | ||||||
| 	} |  | ||||||
| 	color = crng.Range(0, step+minGlow) |  | ||||||
| 	q = append(q, uint8(color)) |  | ||||||
| 	//for uint8(color) < uint8(colorValue) { |  | ||||||
| 	//	q = append(q, uint8(color)) |  | ||||||
| 	//	color = crng.Range(1, step+minGlow) |  | ||||||
| 	//} |  | ||||||
|  |  | ||||||
| 	c := &cdeque{} |  | ||||||
| 	toss := crng.Range(0, 1) //Хаха |  | ||||||
| 	for _, v := range q { |  | ||||||
| 		if toss == 1 { |  | ||||||
| 			c.PushBack(uint8(v)) |  | ||||||
| 		} else { |  | ||||||
| 			c.PushFront(uint8(v)) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return c |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func NewWall() *Tile { | func NewWall() *Tile { | ||||||
| 	return &Tile{ | 	return &Tile{ | ||||||
| 		Name:        "Wall", | 		Name:        "Wall", | ||||||
| @ -140,7 +45,7 @@ func NewWall() *Tile { | |||||||
| 		Explored:    false, | 		Explored:    false, | ||||||
| 		MustDraw:    false, | 		MustDraw:    false, | ||||||
| 		Appearance: &Appearance{ | 		Appearance: &Appearance{ | ||||||
| 			Char: "#", | 			Glyph: &PlainGlyphHolder{"#"}, | ||||||
| 			ColorSet: &TileColorSet{ | 			ColorSet: &TileColorSet{ | ||||||
| 				Fg:     &PlainColorHolder{255, 130, 110, 150}, | 				Fg:     &PlainColorHolder{255, 130, 110, 150}, | ||||||
| 				Bg:     &PlainColorHolder{255, 172, 170, 173}, | 				Bg:     &PlainColorHolder{255, 172, 170, 173}, | ||||||
| @ -160,7 +65,7 @@ func NewFloor() *Tile { | |||||||
| 		Explored:    false, | 		Explored:    false, | ||||||
| 		MustDraw:    false, | 		MustDraw:    false, | ||||||
| 		Appearance: &Appearance{ | 		Appearance: &Appearance{ | ||||||
| 			Char: ".", | 			Glyph: &PlainGlyphHolder{"."}, | ||||||
| 			ColorSet: &TileColorSet{ | 			ColorSet: &TileColorSet{ | ||||||
| 				Fg:     &PlainColorHolder{255, 220, 220, 250}, | 				Fg:     &PlainColorHolder{255, 220, 220, 250}, | ||||||
| 				Bg:     &PlainColorHolder{255, 19, 19, 70}, | 				Bg:     &PlainColorHolder{255, 19, 19, 70}, | ||||||
| @ -183,17 +88,17 @@ func NewWaterTile() *Tile { | |||||||
| 		Colordance:  true, | 		Colordance:  true, | ||||||
|  |  | ||||||
| 		Appearance: &Appearance{ | 		Appearance: &Appearance{ | ||||||
| 			Char: " ", | 			Glyph: &PlainGlyphHolder{" "}, | ||||||
| 			ColorSet: &TileColorSet{ | 			ColorSet: &TileColorSet{ | ||||||
| 				Fg: &PlainColorHolder{255, 220, 220, 250}, | 				Fg: &PlainColorHolder{255, 220, 220, 250}, | ||||||
| 				Bg: &DanceColorHolder{ | 				Bg: &DanceColorHolder{ | ||||||
| 					255, | 					255, | ||||||
| 					singleColorRing(19), | 					SingleColorRing(19), | ||||||
| 					fillColorRing(22, 2, 42, 4), | 					FillColorRing(19, 0, 15, 2), | ||||||
| 					fillColorRing(204, 180, 229, 12), | 					FillColorRing(127, 120, 176, 12), | ||||||
| 				}, | 				}, | ||||||
| 				DarkFg: &PlainColorHolder{255, 30, 20, 50 }, | 				DarkFg: &PlainColorHolder{255, 30, 20, 50}, | ||||||
| 				DarkBg: &PlainColorHolder{255, 7, 7, 80}, | 				DarkBg: &PlainColorHolder{255, 7, 7, 30}, | ||||||
| 			}, | 			}, | ||||||
| 		}, | 		}, | ||||||
| 	} | 	} | ||||||
| @ -210,17 +115,17 @@ func NewDeepWaterTile() *Tile { | |||||||
| 		MustDraw:    true, //fixme debug | 		MustDraw:    true, //fixme debug | ||||||
| 		Colordance:  true, | 		Colordance:  true, | ||||||
| 		Appearance: &Appearance{ | 		Appearance: &Appearance{ | ||||||
| 			Char: " ", | 			Glyph: &PlainGlyphHolder{" "}, | ||||||
| 			ColorSet: &TileColorSet{ | 			ColorSet: &TileColorSet{ | ||||||
| 				Fg: &PlainColorHolder{255, 220, 220, 250}, | 				Fg: &PlainColorHolder{255, 220, 220, 250}, | ||||||
| 				Bg: &DanceColorHolder{ | 				Bg: &DanceColorHolder{ | ||||||
| 					255, | 					255, | ||||||
| 					singleColorRing(5), | 					SingleColorRing(5), | ||||||
| 					fillColorRing(2,0,15,2), | 					FillColorRing(2, 2, 42, 4), | ||||||
| 					fillColorRing(154, 120, 180, 5), | 					FillColorRing(154, 150, 229, 12), | ||||||
| 				}, | 				}, | ||||||
| 				DarkFg: &PlainColorHolder{255, 30, 20, 50}, | 				DarkFg: &PlainColorHolder{255, 30, 20, 50}, | ||||||
| 				DarkBg: &PlainColorHolder{255, 7, 7, 80}, | 				DarkBg: &PlainColorHolder{255, 7, 7, 30}, | ||||||
| 			}, | 			}, | ||||||
| 		}, | 		}, | ||||||
| 	} | 	} | ||||||
|  | |||||||
							
								
								
									
										28
									
								
								engine/gamestate/gamestate.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								engine/gamestate/gamestate.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,28 @@ | |||||||
|  | package gamestate | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"lab.zaar.be/thefish/alchemyst-go/engine/gamemap" | ||||||
|  | 	"lab.zaar.be/thefish/alchemyst-go/engine/mob" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | type GameState struct { | ||||||
|  | 	Mainfunc     chan func() | ||||||
|  | 	Exit         chan struct{} | ||||||
|  | 	Input        chan string | ||||||
|  | 	RawInput     chan int | ||||||
|  | 	FovRecompute chan struct{} | ||||||
|  | 	Redraw		 chan struct{} | ||||||
|  | 	Level        *gamemap.Level | ||||||
|  | 	Player 		 *mob.Player | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // do runs f on the main thread. | ||||||
|  | func (g *GameState) Do(f func()) { | ||||||
|  | 	done := make(chan struct{}, 1) | ||||||
|  | 	g.Mainfunc <- func() { | ||||||
|  | 		f() | ||||||
|  | 		f = nil //zero pointer for closure function | ||||||
|  | 		done <- struct{}{} | ||||||
|  | 	} | ||||||
|  | 	<-done | ||||||
|  | } | ||||||
| @ -1,19 +1,27 @@ | |||||||
| package mob | package mob | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"fmt" | ||||||
| 	"lab.zaar.be/thefish/alchemyst-go/engine/gamemap" | 	"lab.zaar.be/thefish/alchemyst-go/engine/gamemap" | ||||||
| 	"lab.zaar.be/thefish/alchemyst-go/engine/types" | 	"lab.zaar.be/thefish/alchemyst-go/engine/types" | ||||||
|  | 	"reflect" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type Mob struct { | type Mob struct { | ||||||
| 	*gamemap.Appearance | 	*types.Appearance | ||||||
| 	*types.Coords | 	types.Coords | ||||||
| 	BlocksPass bool | 	BlocksPass bool | ||||||
| } | } | ||||||
|  |  | ||||||
|  | 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") { | ||||||
|  |  | ||||||
| func (m *Mob) Walk(dx, dy int) { | 	} | ||||||
|  | 	fmt.Printf("new coords: %d, %d\n", m.Coords.X, m.Coords.Y) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *Mob) Render() { | func (m *Mob) Render() { | ||||||
| @ -23,3 +31,7 @@ func (m *Mob) Render() { | |||||||
| func (m *Mob) MoveToCoords(c types.Coords) { | func (m *Mob) MoveToCoords(c types.Coords) { | ||||||
|  |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (mob Mob) TypeOf() reflect.Type { | ||||||
|  | 	return reflect.TypeOf(mob) | ||||||
|  | } | ||||||
							
								
								
									
										29
									
								
								engine/mob/mob_render_system.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								engine/mob/mob_render_system.go
									
									
									
									
									
										Normal 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) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @ -1,5 +1,5 @@ | |||||||
| package mob | package mob | ||||||
|  |  | ||||||
| type Player struct { | type Player struct { | ||||||
| 	*Mob | 	Mob | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										11
									
								
								engine/screens/character.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								engine/screens/character.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,11 @@ | |||||||
|  | package screens | ||||||
|  |  | ||||||
|  | type CharacterScreen struct { | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (ts *CharacterScreen) UseEcs() bool {return false} | ||||||
|  | func (ts *CharacterScreen) Enter() {} | ||||||
|  | func (ts *CharacterScreen) HandleInput(input string) {} | ||||||
|  | func (ts *CharacterScreen) Exit() {} | ||||||
|  | func (ts *CharacterScreen) Render() {} | ||||||
							
								
								
									
										58
									
								
								engine/screens/game.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								engine/screens/game.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,58 @@ | |||||||
|  | package screens | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"lab.zaar.be/thefish/alchemyst-go/engine/gamestate" | ||||||
|  | 	"lab.zaar.be/thefish/alchemyst-go/ui/mainwindow" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | type GameScreen struct { | ||||||
|  | 	mw    *mainwindow.MainWindow | ||||||
|  | 	state *gamestate.GameState | ||||||
|  | 	vp    *mainwindow.ViewPort | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func NewGameScreen(mw *mainwindow.MainWindow, state *gamestate.GameState, viewPort *mainwindow.ViewPort) *GameScreen { | ||||||
|  | 	return &GameScreen{mw: mw, state: state, vp: viewPort} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (ts *GameScreen) UseEcs() bool { return true } | ||||||
|  | func (ts *GameScreen) Enter()       {} | ||||||
|  | func (ts *GameScreen) HandleInput(input string) { | ||||||
|  | 	//ts.state.Do(func(){ | ||||||
|  | 	switch input { | ||||||
|  | 	case "Up", "k", "8": | ||||||
|  | 		ts.state.Player.Walk(ts.state.Level, 0, -1) | ||||||
|  | 		break | ||||||
|  | 	case "Down", "j", "2": | ||||||
|  | 		ts.state.Player.Walk(ts.state.Level,0, 1) | ||||||
|  | 		break | ||||||
|  | 	case "Left", "h", "4": | ||||||
|  | 		ts.state.Player.Walk(ts.state.Level,-1, 0) | ||||||
|  | 		break | ||||||
|  | 	case "Right", "l", "6": | ||||||
|  | 		ts.state.Player.Walk(ts.state.Level,1, 0) | ||||||
|  | 		break | ||||||
|  | 	case "y", "7": | ||||||
|  | 		ts.state.Player.Walk(ts.state.Level,-1, -1) | ||||||
|  | 		break | ||||||
|  | 	case "u", "9": | ||||||
|  | 		ts.state.Player.Walk(ts.state.Level,1, -1) | ||||||
|  | 		break | ||||||
|  | 	case "b", "1": | ||||||
|  | 		ts.state.Player.Walk(ts.state.Level,-1, 1) | ||||||
|  | 		break | ||||||
|  | 	case "n", "3": | ||||||
|  | 		ts.state.Player.Walk(ts.state.Level,1, 1) | ||||||
|  | 		break | ||||||
|  | 	default: | ||||||
|  | 		ts.mw.GetLayer("base").ClearArea(0, 3, 40, 1) | ||||||
|  | 		ts.mw.GetLayer("base").Print(1, 3, "Key: "+input) | ||||||
|  | 		ts.mw.GetLayer("base").Print(1, 6, "█") | ||||||
|  |  | ||||||
|  | 	} | ||||||
|  | 	//}) | ||||||
|  | } | ||||||
|  | func (ts *GameScreen) Exit()   {} | ||||||
|  | func (ts *GameScreen) Render() { | ||||||
|  | 	ts.vp.Render(ts.state) | ||||||
|  | } | ||||||
							
								
								
									
										11
									
								
								engine/screens/inventory.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								engine/screens/inventory.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,11 @@ | |||||||
|  | package screens | ||||||
|  |  | ||||||
|  | type InventoryScreen struct { | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (ts *InventoryScreen) UseEcs() bool {return true} | ||||||
|  | func (ts *InventoryScreen) Enter() {} | ||||||
|  | func (ts *InventoryScreen) HandleInput(input string) {} | ||||||
|  | func (ts *InventoryScreen) Exit() {} | ||||||
|  | func (ts *InventoryScreen) Render() {} | ||||||
							
								
								
									
										11
									
								
								engine/screens/target.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								engine/screens/target.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,11 @@ | |||||||
|  | package screens | ||||||
|  |  | ||||||
|  | type TargetScreen struct { | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (ts *TargetScreen) UseEcs() bool {return true} | ||||||
|  | func (ts *TargetScreen) Enter() {} | ||||||
|  | func (ts *TargetScreen) HandleInput(input string) {} | ||||||
|  | func (ts *TargetScreen) Exit() {} | ||||||
|  | func (ts *TargetScreen) Render() {} | ||||||
							
								
								
									
										11
									
								
								engine/screens/title.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								engine/screens/title.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,11 @@ | |||||||
|  | package screens | ||||||
|  |  | ||||||
|  | type TitleScreen struct { | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (ts *TitleScreen) UseEcs() bool {return false} | ||||||
|  | func (ts *TitleScreen) Enter() {} | ||||||
|  | func (ts *TitleScreen) HandleInput(input string) {} | ||||||
|  | func (ts *TitleScreen) Exit() {} | ||||||
|  | func (ts *TitleScreen) Render() {} | ||||||
							
								
								
									
										111
									
								
								engine/types/appearance.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										111
									
								
								engine/types/appearance.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,111 @@ | |||||||
|  | package types | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"github.com/gammazero/deque" | ||||||
|  | 	"lab.zaar.be/thefish/alchemyst-go/util" | ||||||
|  | 	"reflect" | ||||||
|  | ) | ||||||
|  | import blt "lab.zaar.be/thefish/bearlibterminal" | ||||||
|  |  | ||||||
|  | var crng = util.NewRNG() | ||||||
|  |  | ||||||
|  | type ColorHolder interface { | ||||||
|  | 	GetColor() uint32 | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type cdeque struct { | ||||||
|  | 	deque.Deque | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *cdeque) Next() uint8 { | ||||||
|  | 	c.Rotate(1) | ||||||
|  | 	return c.Front().(uint8) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type DanceColorHolder struct { | ||||||
|  | 	A uint8 | ||||||
|  | 	R *cdeque | ||||||
|  | 	G *cdeque | ||||||
|  | 	B *cdeque | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (chd *DanceColorHolder) GetColor() uint32 { | ||||||
|  | 	return blt.ColorFromARGB( | ||||||
|  | 		chd.A, | ||||||
|  | 		chd.R.Next(), | ||||||
|  | 		chd.G.Next(), | ||||||
|  | 		chd.B.Next(), | ||||||
|  | 	) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type PlainColorHolder struct { | ||||||
|  | 	A uint8 | ||||||
|  | 	R uint8 | ||||||
|  | 	G uint8 | ||||||
|  | 	B uint8 | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (chb *PlainColorHolder) GetColor() uint32 { | ||||||
|  | 	return blt.ColorFromARGB( | ||||||
|  | 		chb.A, | ||||||
|  | 		chb.R, | ||||||
|  | 		chb.G, | ||||||
|  | 		chb.B, | ||||||
|  | 	) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type TileColorSet struct { | ||||||
|  | 	Fg     ColorHolder | ||||||
|  | 	Bg     ColorHolder | ||||||
|  | 	DarkFg ColorHolder | ||||||
|  | 	DarkBg ColorHolder | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | type GlyphHolder interface { | ||||||
|  | 	GetGlyph() string | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type PlainGlyphHolder struct { | ||||||
|  | 	Glyph string | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (pch *PlainGlyphHolder) GetGlyph() string { | ||||||
|  | 	return pch.Glyph | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type Appearance struct { | ||||||
|  | 	Glyph    GlyphHolder   `json:"char"` | ||||||
|  | 	ColorSet *TileColorSet `json:"colorSet"` | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func SingleColorRing(colorValue uint8) *cdeque { | ||||||
|  | 	c := &cdeque{} | ||||||
|  | 	c.PushBack(colorValue) | ||||||
|  | 	return c | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func FillColorRing(colorValue uint8, minGlow, maxGlow, step int) *cdeque { | ||||||
|  | 	q := make([]uint8, 0) | ||||||
|  | 	color := int(colorValue) | ||||||
|  | 	for color < maxGlow { | ||||||
|  | 		q = append(q, uint8(color)) | ||||||
|  | 		color = crng.Range(1, step) + color | ||||||
|  | 	} | ||||||
|  | 	color = crng.Range(0, step+minGlow) | ||||||
|  | 	q = append(q, colorValue) | ||||||
|  | 	//for uint8(color) < uint8(colorValue) { | ||||||
|  | 	//	q = append(q, uint8(color)) | ||||||
|  | 	//	color = crng.Range(1, step+minGlow) | ||||||
|  | 	//} | ||||||
|  |  | ||||||
|  | 	c := &cdeque{} | ||||||
|  | 	for _, v := range q { | ||||||
|  | 		c.PushBack(uint8(v)) | ||||||
|  | 	} | ||||||
|  | 	return c | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (app Appearance) TypeOf() reflect.Type { | ||||||
|  | 	return reflect.TypeOf(app) | ||||||
|  | } | ||||||
| @ -1,11 +1,18 @@ | |||||||
| package types | package types | ||||||
|  |  | ||||||
| import "math" | import ( | ||||||
|  | 	"math" | ||||||
|  | 	"reflect" | ||||||
|  | ) | ||||||
|  |  | ||||||
| type Coords struct { | type Coords struct { | ||||||
| 	X, Y int | 	X, Y int | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (сс Coords) TypeOf() reflect.Type { | ||||||
|  | 	return reflect.TypeOf(сс) | ||||||
|  | } | ||||||
|  |  | ||||||
| func (c *Coords) Get() (int, int) { | func (c *Coords) Get() (int, int) { | ||||||
| 	return c.X, c.Y | 	return c.X, c.Y | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										88
									
								
								engine/types/screen.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										88
									
								
								engine/types/screen.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,88 @@ | |||||||
|  | package types | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"lab.zaar.be/thefish/alchemyst-go/util" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | type Screen interface { | ||||||
|  | 	Enter() | ||||||
|  | 	Exit() | ||||||
|  | 	Render() | ||||||
|  | 	HandleInput(input string) | ||||||
|  | 	UseEcs() bool | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type ScreenManager struct { | ||||||
|  | 	ctx util.ClientCtx | ||||||
|  | 	Screens map[string]Screen | ||||||
|  | 	CurrentScreen Screen | ||||||
|  | 	PreviousScreen Screen | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // NewScreenManager is a convenience/constructor method to properly initialize a new ScreenManager | ||||||
|  | func NewScreenManager(ctx util.ClientCtx) *ScreenManager { | ||||||
|  | 	manager := ScreenManager{ | ||||||
|  | 		ctx:ctx, | ||||||
|  | 		Screens: make(map[string]Screen), | ||||||
|  | 		CurrentScreen: nil, | ||||||
|  | 	} | ||||||
|  | 	return &manager | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (sm *ScreenManager) AddScreen(screenName string, screen Screen) { | ||||||
|  | 	// Check to see if a screen with the given screenName has already been added | ||||||
|  | 	if _, ok := sm.Screens[screenName]; !ok { | ||||||
|  | 		// A screen with the given name does not yet exist on the ScreenManager, go ahead and add it | ||||||
|  | 		sm.Screens[screenName] = screen | ||||||
|  | 	} else { | ||||||
|  | 		sm.ctx.Logger().Warn().Msgf("A screen with name %v was already added to the ScreenManager %v!", screenName, sm) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // RemoveScreen will remove a screen from the ScreenManager. This can be useful when a temporary screen needs to be | ||||||
|  | // created, as it can be quickly added (rather than registering at game creation), and then removed when it is no | ||||||
|  | // longer needed | ||||||
|  | func (sm *ScreenManager) RemoveScreen(screenName string, screen Screen) { | ||||||
|  | 	// Check if the given screenName exists in the ScreenManager | ||||||
|  | 	if _, ok := sm.Screens[screenName]; ok { | ||||||
|  | 		delete(sm.Screens, screenName) | ||||||
|  | 	} else { | ||||||
|  | 		// A screen with the given name does not exist | ||||||
|  | 		sm.ctx.Logger().Warn().Msgf("A screen with name %v was not found on ScreenManager %v!", screenName, sm) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // SetScreen will set the current screen property of the screen manager to the provided screen | ||||||
|  | func (sm *ScreenManager) SetScreen(screen Screen) { | ||||||
|  | 	// Call the exit function of the currentScreen, and set the currentScreen as the previousScreen | ||||||
|  | 	// Only do this if there is a currentScreen | ||||||
|  | 	if sm.CurrentScreen != nil { | ||||||
|  | 		sm.CurrentScreen.Exit() | ||||||
|  | 		sm.PreviousScreen = sm.CurrentScreen | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Set the provided screen as the currentScreen, and call the enter() function of the new currentScreen | ||||||
|  | 	sm.CurrentScreen = screen | ||||||
|  | 	sm.CurrentScreen.Enter() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // SetScreenByName takes a string representing the screen desired to navigate to. It will then transition the | ||||||
|  | // ScreenManager to the specified screen, if one is found. | ||||||
|  | func (sm *ScreenManager) SetScreenByName(screenName string) { | ||||||
|  | 	// Check if the given screenName exists in the ScreenManager | ||||||
|  | 	if _, ok := sm.Screens[screenName]; ok { | ||||||
|  | 		// Call the exit function of the currentScreen, and set the currentScreen as the previousScreen | ||||||
|  | 		// Only do this if there is a currentScreen | ||||||
|  | 		if sm.CurrentScreen != nil { | ||||||
|  | 			sm.CurrentScreen.Exit() | ||||||
|  | 			sm.PreviousScreen = sm.CurrentScreen | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// Set the provided screen as the currentScreen, and call the enter() function of the new currentScreen | ||||||
|  | 		sm.CurrentScreen = sm.Screens[screenName] | ||||||
|  | 		sm.CurrentScreen.Enter() | ||||||
|  | 	} else { | ||||||
|  | 		// A screen with the given name does not exist | ||||||
|  | 		sm.ctx.Logger().Warn().Msgf("A screen with name %v was not found on ScreenManager %v!", screenName, sm) | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								link.sh
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								link.sh
									
									
									
									
									
								
							| @ -1,2 +1,2 @@ | |||||||
| #!/bin/bash | #!/bin/bash | ||||||
| ln -s src/bearlibterminal/libBearLibTerminal.so . | ln -s vendor/lab.zaar.be/thefish/bearlibterminal/libBearLibTerminal.so . | ||||||
|  | |||||||
| @ -98,7 +98,7 @@ func (layer *Layer) Decorate(f func(args ...interface{})) func(args ...interface | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func (layer *Layer) Clear(r *types.Rect) { | func (layer *Layer) ClearRect(r *types.Rect) { | ||||||
| 	blt.ClearArea(r.X, r.Y, r.W, r.H) | 	blt.ClearArea(r.X, r.Y, r.W, r.H) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | |||||||
| @ -42,13 +42,14 @@ func (mw *MainWindow) Open() { | |||||||
| 	blt.Set( | 	blt.Set( | ||||||
| 		fmt.Sprintf( | 		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: ./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'; font: %s, size=%s;", | ||||||
| 			//"window: size=%dx%d, title='%s v%s'", | 			//"window: size=%dx%d, title='%s v%s'", | ||||||
| 			config.MainWindowSizeX, | 			config.MainWindowSizeX, | ||||||
| 			config.MainWindowSizeY, | 			config.MainWindowSizeY, | ||||||
| 			config.Title, | 			config.Title, | ||||||
| 			config.Version, | 			config.Version, | ||||||
| 			config.Font, | 			config.Font, | ||||||
|  | 			config.FontSize, | ||||||
| 		), | 		), | ||||||
| 	) | 	) | ||||||
| } | } | ||||||
|  | |||||||
| @ -2,43 +2,51 @@ package mainwindow | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"errors" | 	"errors" | ||||||
|  | 	"fmt" | ||||||
| 	"lab.zaar.be/thefish/alchemyst-go/engine/fov" | 	"lab.zaar.be/thefish/alchemyst-go/engine/fov" | ||||||
| 	"lab.zaar.be/thefish/alchemyst-go/engine/fov/precomputed_shade" | 	"lab.zaar.be/thefish/alchemyst-go/engine/fov/precomputed_shade" | ||||||
| 	"lab.zaar.be/thefish/alchemyst-go/engine/gamemap" | 	"lab.zaar.be/thefish/alchemyst-go/engine/gamestate" | ||||||
| 	"lab.zaar.be/thefish/alchemyst-go/engine/types" | 	"lab.zaar.be/thefish/alchemyst-go/engine/types" | ||||||
|  | 	"time" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var NotInViewError = errors.New("not in ViewPort") | var NotInViewError = errors.New("not in ViewPort") | ||||||
|  |  | ||||||
| const FPS_LIMIT = 60 |  | ||||||
|  |  | ||||||
| type ViewPort struct { | type ViewPort struct { | ||||||
| 	*types.Rect | 	*types.Rect | ||||||
| 	level             *gamemap.Level | 	cameraCoords      types.Coords | ||||||
| 	layer             *Layer | 	layer             *Layer | ||||||
| 	Fov               fov.Fov | 	Fov               fov.Fov | ||||||
| 	playerCoords      types.Coords | 	PlayerCoords      types.Coords | ||||||
| 	playerTorchRadius int | 	PlayerTorchRadius int | ||||||
|  | 	animateTiles      *time.Ticker | ||||||
| } | } | ||||||
|  |  | ||||||
| func NewViewPort(x, y, w, h int, level *gamemap.Level, layer *Layer) *ViewPort { | func NewViewPort(x, y, w, h int, layer *Layer) *ViewPort { | ||||||
| 	//fixme |  | ||||||
| 	fov := precomputed_shade.NewPrecomputedShade(15) | 	computedFov := precomputed_shade.NewPrecomputedShade(15) | ||||||
| 	fov.Init() | 	computedFov.Init() | ||||||
| 	vp := ViewPort{ | 	vp := ViewPort{ | ||||||
| 		Rect:  &types.Rect{x, y, w, h}, | 		Rect:  &types.Rect{x, y, w, h}, | ||||||
| 		level: level, |  | ||||||
| 		layer: layer, | 		layer: layer, | ||||||
| 		Fov:   fov, | 		Fov:   computedFov, | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	vp.playerCoords = types.Coords{10, 10} | 	vp.PlayerTorchRadius = 11 | ||||||
| 	vp.playerTorchRadius = 10 | 	vp.animateTiles = time.NewTicker(time.Second / 12) | ||||||
|  |  | ||||||
| 	return &vp | 	return &vp | ||||||
| } | } | ||||||
|  |  | ||||||
| func (vp *ViewPort) Move(c *types.Coords) { | func (vp *ViewPort) Close() { | ||||||
|  | 	vp.animateTiles.Stop() | ||||||
|  | 	vp.animateTiles = nil //free pointer to ticker | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (vp *ViewPort) Move(state *gamestate.GameState) { | ||||||
|  |  | ||||||
|  | 	c := &state.Player.Coords | ||||||
|  |  | ||||||
| 	x := c.X - vp.Rect.W/2 | 	x := c.X - vp.Rect.W/2 | ||||||
| 	y := c.Y - vp.Rect.H/2 | 	y := c.Y - vp.Rect.H/2 | ||||||
|  |  | ||||||
| @ -48,138 +56,84 @@ func (vp *ViewPort) Move(c *types.Coords) { | |||||||
| 	if y < 0 { | 	if y < 0 { | ||||||
| 		y = 0 | 		y = 0 | ||||||
| 	} | 	} | ||||||
| 	if x > vp.level.W-vp.W { | 	if x > state.Level.W - vp.W - 1 { | ||||||
| 		x = vp.level.W - vp.W | 		x = state.Level.W - vp.W | ||||||
| 	} | 	} | ||||||
| 	if y > vp.level.H-vp.H { | 	if y > state.Level.H-vp.H - 1 { | ||||||
| 		x = vp.level.H - vp.H | 		y = state.Level.H - vp.H | ||||||
| 	} | 	} | ||||||
| 	//if x != vp.X || y != vp.Y { State.FovRecompute <- struct{}{}} | 	if x != vp.cameraCoords.X || y != vp.cameraCoords.Y { | ||||||
| 	vp.X, vp.Y = x, y | 		state.FovRecompute <- struct{}{} | ||||||
|  | 	} | ||||||
|  | 	vp.cameraCoords.X = x | ||||||
|  | 	vp.cameraCoords.Y = y | ||||||
|  |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (vp *ViewPort) ToVPCoords(c *types.Coords) (newCoords *types.Coords, err error) { | func (vp *ViewPort) ToVPCoords(c types.Coords) (newCoords types.Coords, err error) { | ||||||
| 	//coords on map to coords on vp | 	//coords on map to coords on vp | ||||||
| 	x, y := c.X-vp.X, c.Y-vp.Y | 	x, y := c.X-vp.cameraCoords.X, c.Y-vp.cameraCoords.Y | ||||||
| 	if x < 0 || y < 0 || x >= vp.W || y >= vp.H { | 	if x < 0 || y < 0 || x > vp.W || y > vp.H { | ||||||
| 		return &types.Coords{-1, -1}, NotInViewError | 		return types.Coords{-1, -1}, NotInViewError | ||||||
| 	} | 	} | ||||||
| 	return &types.Coords{x, y}, nil | 	return types.Coords{x, y}, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| ////call only from main thread |  | ||||||
| //func (vp *ViewPort) Render() { |  | ||||||
| //	//fixme get these from state chan(s) |  | ||||||
| //	var fpsTicker int |  | ||||||
| //	var fovRecompute bool = true |  | ||||||
| //	redraw := false |  | ||||||
| //	//fixme get player instance |  | ||||||
| // |  | ||||||
| //	vp.Move(&vp.playerCoords) |  | ||||||
| //	//fixme detect fovRecompute |  | ||||||
| //	if fovRecompute { |  | ||||||
| //		vp.layer.Clear(vp.Rect) |  | ||||||
| //		fovRecompute = false |  | ||||||
| //		redraw = true |  | ||||||
| //		//fixme |  | ||||||
| // |  | ||||||
| //		vp.Fov.ComputeFov(vp.level, vp.playerCoords, vp.playerTorchRadius) |  | ||||||
| //	} |  | ||||||
| //	//increase ticker |  | ||||||
| //	fpsTicker++ |  | ||||||
| // |  | ||||||
| //	if redraw || fpsTicker%(FPS_LIMIT/10) == 0 { |  | ||||||
| //		fpsTicker = 0 |  | ||||||
| // |  | ||||||
| //		for y := 0; y < vp.H; y++ { |  | ||||||
| //			for x := 0; x < vp.W; x++ { |  | ||||||
| //				mapCoords := types.Coords{vp.X + x, vp.Y + y} |  | ||||||
| //				tile := vp.level.Tiles[mapCoords.X][mapCoords.Y] |  | ||||||
| //				visible := vp.Fov.IsInFov(mapCoords) |  | ||||||
| //				if !visible { |  | ||||||
| //					if tile.MustDraw { |  | ||||||
| //						//darkened version of landscape |  | ||||||
| //						vp.layer.WithRawColor(tile.ColorSet.DarkFg()). |  | ||||||
| //							PutWithRawBackground(mapCoords.X, mapCoords.Y, tile.Char, tile.ColorSet.DarkBg()) |  | ||||||
| //					} |  | ||||||
| //				} else { |  | ||||||
| //					if redraw == true || tile.Colordance { |  | ||||||
| //						vp.layer.WithRawColor(tile.ColorSet.Fg()). |  | ||||||
| //							PutWithRawBackground(mapCoords.X, mapCoords.Y, tile.Char, tile.ColorSet.Bg()) |  | ||||||
| //						tile.Explored = true |  | ||||||
| //						tile.MustDraw = true |  | ||||||
| //					} |  | ||||||
| //				} |  | ||||||
| //			} |  | ||||||
| //		} |  | ||||||
| // |  | ||||||
| //	} |  | ||||||
| //} |  | ||||||
|  |  | ||||||
| var redraw = true | var redraw = true | ||||||
| var fovRecompute = true | var fovRecompute = true | ||||||
|  |  | ||||||
| func (vp *ViewPort) Render() { | func (vp *ViewPort) Listen(state gamestate.GameState) { | ||||||
|  | 	for { | ||||||
|  | 		select { | ||||||
|  | 		case <-state.FovRecompute: | ||||||
|  | 			fovRecompute = true | ||||||
|  | 		case <-state.Redraw: | ||||||
|  | 			redraw = true | ||||||
|  | 		case <-vp.animateTiles.C: | ||||||
|  | 			redraw = true | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (vp *ViewPort) Render(state *gamestate.GameState) { | ||||||
|  |  | ||||||
|  | 	vp.Move(state) | ||||||
|  |  | ||||||
| 	if fovRecompute { | 	if fovRecompute { | ||||||
| 		vp.layer.Clear(vp.Rect) | 		vp.layer.ClearRect(vp.Rect) | ||||||
| 		fovRecompute = false | 		fovRecompute = true | ||||||
| 		redraw = true | 		redraw = true | ||||||
| 		//fixme | 		vp.Fov.ComputeFov(state.Level, state.Player.Coords, vp.PlayerTorchRadius) | ||||||
|  |  | ||||||
| 		vp.Fov.ComputeFov(vp.level, vp.playerCoords, vp.playerTorchRadius) |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	//for y := 0; y < vp.H; y++ { | 	//if redraw { | ||||||
| 	//	for x := 0; x < vp.W; x++ { | 		//terrain | ||||||
| 	//		mapCoords := types.Coords{vp.X + x, vp.Y + y} |  | ||||||
| 	//		tile := vp.level.Tiles[mapCoords.X][mapCoords.Y] |  | ||||||
| 	// |  | ||||||
| 	//		if tile.Visible { |  | ||||||
| 	//			if tile.MustDraw { |  | ||||||
| 	//				//darkened version of landscape |  | ||||||
| 	//				vp.layer.WithRawColor(tile.ColorSet.DarkFg()). |  | ||||||
| 	//					PutWithRawBackground(mapCoords.X, mapCoords.Y, tile.Char, tile.ColorSet.DarkBg()) |  | ||||||
| 	//			} |  | ||||||
| 	//		} else { |  | ||||||
| 	//			if redraw == true || tile.Colordance { |  | ||||||
| 	//				vp.layer.WithRawColor(tile.ColorSet.Fg()). |  | ||||||
| 	//					PutWithRawBackground(mapCoords.X, mapCoords.Y, tile.Char, tile.ColorSet.Bg()) |  | ||||||
| 	//				tile.Explored = true |  | ||||||
| 	//				tile.MustDraw = true |  | ||||||
| 	//			} |  | ||||||
| 	//		} |  | ||||||
| 	//	} |  | ||||||
| 	//} |  | ||||||
|  |  | ||||||
| 		for y := 0; y < vp.H; y++ { | 		for y := 0; y < vp.H; y++ { | ||||||
| 			for x := 0; x < vp.W; x++ { | 			for x := 0; x < vp.W; x++ { | ||||||
| 			mapCoords := types.Coords{vp.X + x, vp.Y + y} | 				mapCoords := types.Coords{vp.cameraCoords.X + x, vp.cameraCoords.Y + y} | ||||||
| 			//vp.level.GetTile(mapCoords).Render() |  | ||||||
|  |  | ||||||
| 			tile := vp.level.GetTile(mapCoords) | 				if state.Level.InBounds(mapCoords) { | ||||||
|  | 					tile := state.Level.GetTile(mapCoords) | ||||||
|  | 					if tile.Explored || tile.Visible { | ||||||
|  | 						vp.layer.PutToBase(x+vp.X, y+vp.Y, tile.GetChar(), tile.GetRawColor(), tile.GetRawBgColor()) | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		//mobs | ||||||
|  | 		pc, err := vp.ToVPCoords(state.Player.Coords) | ||||||
|  | 		_ = pc | ||||||
|  | 		if err != nil { | ||||||
|  | 			fmt.Println("error on getting player position") | ||||||
|  | 		} else { | ||||||
|  | 			vp.layer.WithColor("white").Put(pc.X+vp.X, pc.Y+vp.Y, "@") | ||||||
|  | 			//mw.GetLayer("base").WithColor("white").Put(42, 10, "B") | ||||||
|  | 			//mw.GetLayer("overlay").WithColor("white").Put(59, 10, "O") | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 			//vp.layer. | 		//redraw = true | ||||||
| 			//	WithRawColor(tile.GetRawColor()). | 		//redraw = false | ||||||
| 			//	PutWithBackground(mapCoords.X, mapCoords.Y, tile.GetChar(), "grey") |  | ||||||
|  |  | ||||||
| 			vp.layer.PutToBase(mapCoords.X,mapCoords.Y,tile.GetChar(), tile.GetRawColor(), tile.GetRawBgColor()) |  | ||||||
|  |  | ||||||
| 			//if tile.Visible { |  | ||||||
| 			//	if tile.MustDraw { |  | ||||||
| 			//		//darkened version of landscape |  | ||||||
| 			//		vp.layer.WithColor("green"). |  | ||||||
| 			//			Put(mapCoords.X, mapCoords.Y, tile.Char) |  | ||||||
| 			//	} |  | ||||||
| 			//} else { |  | ||||||
| 			//	if redraw == true || tile.Colordance { |  | ||||||
| 			//		vp.layer.WithColor("grey"). |  | ||||||
| 			//			Put(mapCoords.X, mapCoords.Y, tile.Char) |  | ||||||
| 			//		tile.Explored = true |  | ||||||
| 			//		tile.MustDraw = true |  | ||||||
| 			//	} |  | ||||||
| 	//} | 	//} | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| } | } | ||||||
|  | |||||||
| @ -14,6 +14,7 @@ type Config struct { | |||||||
| 	MainWindowSizeX int    `json:"sizeX"` | 	MainWindowSizeX int    `json:"sizeX"` | ||||||
| 	MainWindowSizeY int    `json:"sizeY"` | 	MainWindowSizeY int    `json:"sizeY"` | ||||||
| 	Font            string `json:"font"` | 	Font            string `json:"font"` | ||||||
|  | 	FontSize            string `json:"fontSize"` //format is "8x12" | ||||||
| 	Verbosity       string `json:"verbosity"` | 	Verbosity       string `json:"verbosity"` | ||||||
| } | } | ||||||
|  |  | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user