From 2fbb50151a1453d4fad4f457e1b2ff79d4186721 Mon Sep 17 00:00:00 2001 From: limepotato Date: Mon, 12 Feb 2024 18:45:21 -0700 Subject: [PATCH] wha? --- README.md | 2 - bin/main/assets/icon.png | Bin 0 -> 724 bytes .../assets/tilesets/potrogue_grunge_16x16.png | Bin 0 -> 2570 bytes bin/main/assets/tilesets/rogue_yun_16x16.png | Bin 0 -> 9709 bytes bin/main/data/values.conf | 1 + .../ouroboros/potrogue/blocks/GameBlock.kt | 86 ++++++++++ .../potrogue/builders/EntityFactory.kt | 64 +++++++ .../potrogue/builders/GameBlockFactory.kt | 9 + .../ouroboros/potrogue/builders/GameColors.kt | 18 ++ .../potrogue/builders/GameTileRepository.kt | 46 +++++ .../potrogue/builders/WorldBuilder.kt | 74 ++++++++ .../ouroboros/potrogue/data/config/Config.kt | 69 ++++++++ .../potrogue/data/config/GameConfig.kt | 39 +++++ .../entity/attributes/CreatureSpread.kt | 9 + .../entity/attributes/EntityActions.kt | 40 +++++ .../entity/attributes/EntityPosition.kt | 26 +++ .../potrogue/entity/attributes/EntityTile.kt | 7 + .../entity/attributes/flags/BlockOccupier.kt | 5 + .../entity/attributes/types/EntityTypes.kt | 15 ++ .../potrogue/entity/messages/Attack.kt | 11 ++ .../ouroboros/potrogue/entity/messages/Dig.kt | 11 ++ .../potrogue/entity/messages/EntityAction.kt | 34 ++++ .../potrogue/entity/messages/MoveCamera.kt | 13 ++ .../potrogue/entity/messages/MoveTo.kt | 13 ++ .../potrogue/entity/systems/Attackable.kt | 15 ++ .../potrogue/entity/systems/CameraMover.kt | 44 +++++ .../potrogue/entity/systems/CreatureGrowth.kt | 42 +++++ .../potrogue/entity/systems/Diggable.kt | 15 ++ .../potrogue/entity/systems/InputReceiver.kt | 43 +++++ .../potrogue/entity/systems/Movable.kt | 87 ++++++++++ .../potrogue/extensions/EntityExtensions.kt | 34 ++++ .../potrogue/extensions/PositionExtensions.kt | 20 +++ .../potrogue/extensions/TypeAliases.kt | 35 ++++ bin/main/group/ouroboros/potrogue/main.kt | 22 +++ .../ouroboros/potrogue/util/ResourceGetter.kt | 14 ++ .../ouroboros/potrogue/view/ConfigView.kt | 56 ++++++ .../group/ouroboros/potrogue/view/LoseView.kt | 49 ++++++ .../ouroboros/potrogue/view/PauseView.kt | 54 ++++++ .../group/ouroboros/potrogue/view/PlayView.kt | 77 +++++++++ .../ouroboros/potrogue/view/StartView.kt | 73 ++++++++ .../group/ouroboros/potrogue/view/WinView.kt | 49 ++++++ .../group/ouroboros/potrogue/world/Game.kt | 28 +++ .../ouroboros/potrogue/world/GameBuilder.kt | 83 +++++++++ .../ouroboros/potrogue/world/GameContext.kt | 19 ++ .../group/ouroboros/potrogue/world/World.kt | 162 ++++++++++++++++++ bin/main/logback.xml | 22 +++ build.gradle.kts | 4 +- .../assets/tilesets/potrogue_grunge_16x16.png | Bin 2481 -> 2570 bytes .../assets/tilesets/rogue_yun_16x16.png | Bin 9520 -> 9709 bytes 49 files changed, 1635 insertions(+), 4 deletions(-) create mode 100644 bin/main/assets/icon.png create mode 100644 bin/main/assets/tilesets/potrogue_grunge_16x16.png create mode 100644 bin/main/assets/tilesets/rogue_yun_16x16.png create mode 100644 bin/main/data/values.conf create mode 100644 bin/main/group/ouroboros/potrogue/blocks/GameBlock.kt create mode 100644 bin/main/group/ouroboros/potrogue/builders/EntityFactory.kt create mode 100644 bin/main/group/ouroboros/potrogue/builders/GameBlockFactory.kt create mode 100644 bin/main/group/ouroboros/potrogue/builders/GameColors.kt create mode 100644 bin/main/group/ouroboros/potrogue/builders/GameTileRepository.kt create mode 100644 bin/main/group/ouroboros/potrogue/builders/WorldBuilder.kt create mode 100644 bin/main/group/ouroboros/potrogue/data/config/Config.kt create mode 100644 bin/main/group/ouroboros/potrogue/data/config/GameConfig.kt create mode 100644 bin/main/group/ouroboros/potrogue/entity/attributes/CreatureSpread.kt create mode 100644 bin/main/group/ouroboros/potrogue/entity/attributes/EntityActions.kt create mode 100644 bin/main/group/ouroboros/potrogue/entity/attributes/EntityPosition.kt create mode 100644 bin/main/group/ouroboros/potrogue/entity/attributes/EntityTile.kt create mode 100644 bin/main/group/ouroboros/potrogue/entity/attributes/flags/BlockOccupier.kt create mode 100644 bin/main/group/ouroboros/potrogue/entity/attributes/types/EntityTypes.kt create mode 100644 bin/main/group/ouroboros/potrogue/entity/messages/Attack.kt create mode 100644 bin/main/group/ouroboros/potrogue/entity/messages/Dig.kt create mode 100644 bin/main/group/ouroboros/potrogue/entity/messages/EntityAction.kt create mode 100644 bin/main/group/ouroboros/potrogue/entity/messages/MoveCamera.kt create mode 100644 bin/main/group/ouroboros/potrogue/entity/messages/MoveTo.kt create mode 100644 bin/main/group/ouroboros/potrogue/entity/systems/Attackable.kt create mode 100644 bin/main/group/ouroboros/potrogue/entity/systems/CameraMover.kt create mode 100644 bin/main/group/ouroboros/potrogue/entity/systems/CreatureGrowth.kt create mode 100644 bin/main/group/ouroboros/potrogue/entity/systems/Diggable.kt create mode 100644 bin/main/group/ouroboros/potrogue/entity/systems/InputReceiver.kt create mode 100644 bin/main/group/ouroboros/potrogue/entity/systems/Movable.kt create mode 100644 bin/main/group/ouroboros/potrogue/extensions/EntityExtensions.kt create mode 100644 bin/main/group/ouroboros/potrogue/extensions/PositionExtensions.kt create mode 100644 bin/main/group/ouroboros/potrogue/extensions/TypeAliases.kt create mode 100644 bin/main/group/ouroboros/potrogue/main.kt create mode 100644 bin/main/group/ouroboros/potrogue/util/ResourceGetter.kt create mode 100644 bin/main/group/ouroboros/potrogue/view/ConfigView.kt create mode 100644 bin/main/group/ouroboros/potrogue/view/LoseView.kt create mode 100644 bin/main/group/ouroboros/potrogue/view/PauseView.kt create mode 100644 bin/main/group/ouroboros/potrogue/view/PlayView.kt create mode 100644 bin/main/group/ouroboros/potrogue/view/StartView.kt create mode 100644 bin/main/group/ouroboros/potrogue/view/WinView.kt create mode 100644 bin/main/group/ouroboros/potrogue/world/Game.kt create mode 100644 bin/main/group/ouroboros/potrogue/world/GameBuilder.kt create mode 100644 bin/main/group/ouroboros/potrogue/world/GameContext.kt create mode 100644 bin/main/group/ouroboros/potrogue/world/World.kt create mode 100644 bin/main/logback.xml diff --git a/README.md b/README.md index 704e7e1..befbf5c 100644 --- a/README.md +++ b/README.md @@ -1,8 +1,6 @@ # PotRogue ### A WIP, opensource, roguelike project built in [Kotlin](https://kotlinlang.org/), utilizing [Zircon](https://hexworks.org/projects/zircon/). -### For now, please make issues on [The mirror repo](https://next.forgejo.org/Ouroboros/potrogue/issues) as ForgeFed has not yet been implemented in mainline ForgeJo, and this instance does not have an open registration - ## Installation 1. Make sure you have installed Java 20/21 2. Clone this repo and run `./gradlew clean build` || ~~Download the latest .jar in the Releases page [Self-Hosted ForgeJo](https://git.ouroboros.group/Ouroboros/potrogue/releases) | ~~[Mirror](https://next.forgejo.org/Ouroboros/potrogue/releases)~~~~ diff --git a/bin/main/assets/icon.png b/bin/main/assets/icon.png new file mode 100644 index 0000000000000000000000000000000000000000..1473d199f3cfb02e031a8843ba4d39671942cb75 GIT binary patch literal 724 zcmeAS@N?(olHy`uVBq!ia0vp^4j|0I3?%1nZ+ru!7>k44ofy`glX(f`6b1N%xPG~P z?ZdHSU-v#cw&2^__G#~CF8I2#{@s=N%Wh5WzItln^~beGo^-AKvVHH5QkY6yve`0_|z3Eb*+wXb0IEF;DzMX!$@2~=o z>+WPbMbRr=Iv@Y(o1svy(49`S{~VeOI!!)vxtycL?+`ROq=g z$Xx&P`j;$$kr1!})HC=jkp(mALb>~LT6;C-gvUXJ6S6##CbLIE6 zgz1cGci$;Laf)D&aQ^=L1=AW=*9v}vRlgRvGq}XCANiuFzk!9Ji8mmofx@jA=71_Q@w_OlC+xwv*+p|5p&L7$QD$?@!DdCWx>UQI6cwP&)KyV}etH*U53Km8qHGZuyVGaPJ@ULMPs;?~poi#I{DPQKuG t==W!*zrDTw_qV*vcl%1akL@3M5AEhyxAtLsJuq!Bc)I$ztaD0e0s#HQVCMh; literal 0 HcmV?d00001 diff --git a/bin/main/assets/tilesets/potrogue_grunge_16x16.png b/bin/main/assets/tilesets/potrogue_grunge_16x16.png new file mode 100644 index 0000000000000000000000000000000000000000..14e889fdb70f8ff65e3ff898d0dbc7560f3a1680 GIT binary patch literal 2570 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 literal 0 HcmV?d00001 diff --git a/bin/main/assets/tilesets/rogue_yun_16x16.png b/bin/main/assets/tilesets/rogue_yun_16x16.png new file mode 100644 index 0000000000000000000000000000000000000000..6e20af16db06766ad4049737dde575b940686e30 GIT binary patch literal 9709 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 = mutableListOf(), +) : BaseBlock( + emptyTile = Tile.empty(), + tiles = persistentMapOf(BlockTileType.CONTENT to defaultTile) +) { + init { + updateContent() + } + + val isWall: Boolean + get() = defaultTile == WALL + + val isFloor: Boolean + get() = defaultTile == FLOOR + + + + // We add a property which tells whether this block is just a floor (similar to isWall) + val isEmptyFloor: Boolean + get() = currentEntities.isEmpty() + + // occupier will return the first entity which has the BlockOccupier flag or an empty Maybe if there is none + val occupier: Maybe> + get() = Maybe.ofNullable(currentEntities.firstOrNull { it.occupiesBlock }) + + val isOccupied: Boolean + get() = occupier.isPresent + // Note how we tell whether a block is occupied by checking for the presence of an occupier + + // Exposed a getter for entities which takes a snapshot (defensive copy) of the current entities and returns them. + // We do this because we don’t want to expose the internals of + // GameBlock which would make currentEntities mutable to the outside world + val entities: Iterable> + get() = currentEntities.toList() + + // We expose a function for adding an Entity to our block + fun addEntity(entity: GameEntity) { + currentEntities.add(entity) + updateContent() + } + + // And also for removing one + fun removeEntity(entity: GameEntity) { + currentEntities.remove(entity) + updateContent() + } + + // Incorporated our entities to how we display a block by + private fun updateContent() { + val entityTiles = currentEntities.map { it.tile } + content = when { + // Checking if the player is at this block. If yes, it is displayed on top + entityTiles.contains(PLAYER) -> PLAYER + entityTiles.contains(WALL) -> WALL + // Otherwise, the first Entity is displayed if present + entityTiles.isNotEmpty() -> entityTiles.first() + // Or the default tile if not + else -> defaultTile + } + } + + companion object { + + fun createWith(entity: GameEntity) = GameBlock( + currentEntities = mutableListOf(entity) + ) + } +} diff --git a/bin/main/group/ouroboros/potrogue/builders/EntityFactory.kt b/bin/main/group/ouroboros/potrogue/builders/EntityFactory.kt new file mode 100644 index 0000000..2c43482 --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/builders/EntityFactory.kt @@ -0,0 +1,64 @@ +package group.ouroboros.potrogue.builders + +import group.ouroboros.potrogue.entity.attributes.CreatureSpread +import group.ouroboros.potrogue.entity.attributes.EntityActions +import group.ouroboros.potrogue.entity.attributes.EntityPosition +import group.ouroboros.potrogue.entity.attributes.EntityTile +import group.ouroboros.potrogue.entity.attributes.flags.BlockOccupier +import group.ouroboros.potrogue.entity.attributes.types.Creature +import group.ouroboros.potrogue.entity.attributes.types.Player +import group.ouroboros.potrogue.entity.attributes.types.Wall +import group.ouroboros.potrogue.entity.messages.Attack +import group.ouroboros.potrogue.entity.messages.Dig +import group.ouroboros.potrogue.entity.systems.* +import group.ouroboros.potrogue.world.GameContext +import org.hexworks.amethyst.api.builder.EntityBuilder +import org.hexworks.amethyst.api.entity.EntityType +import org.hexworks.amethyst.api.newEntityOfType + +// We add a function which calls Entities.newEntityOfType +// and pre-fills the generic type parameter for Context with GameContext. +fun newGameEntityOfType( + type: T, + init: EntityBuilder.() -> Unit +) = newEntityOfType(type, init) + +// We define our factory as an object since we’ll only ever have a single instance of it. +object EntityFactory { + + // WALLS! + fun newWall() = newGameEntityOfType(Wall) { + attributes( + EntityPosition(), + BlockOccupier, + EntityTile(GameTileRepository.WALL) + ) + facets(Diggable) + } + + // We add a function for creating a newPlayer and call newGameEntityOfType with our previously created Player type. + fun newPlayer() = newGameEntityOfType(Player) { + // We specify our Attributes, Behaviors, and Facets. We only have Attributes so far though. + attributes( + EntityPosition(), + EntityTile(GameTileRepository.PLAYER), + EntityActions(Dig::class, Attack::class) + ) + behaviors(InputReceiver) + facets(Movable, CameraMover) + } + + // We added the creatureSpread as a parameter to newCreature and it also has a default value. + // This enables us to call it with a CreatureSpread object when The Creature grows and use the default when we create the first one in the builder + fun newCreature(creatureSpread: CreatureSpread = CreatureSpread()) = newGameEntityOfType(Creature) { + attributes( + BlockOccupier, + EntityPosition(), + EntityTile(GameTileRepository.CREATURE), + // We pass the creatureSPread parameter to our builder so it will use whatever we supplied instead of creating one by hand + creatureSpread + ) + facets(Attackable) + behaviors(CreatureGrowth) + } +} \ No newline at end of file diff --git a/bin/main/group/ouroboros/potrogue/builders/GameBlockFactory.kt b/bin/main/group/ouroboros/potrogue/builders/GameBlockFactory.kt new file mode 100644 index 0000000..a482e72 --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/builders/GameBlockFactory.kt @@ -0,0 +1,9 @@ +package group.ouroboros.potrogue.builders + +import group.ouroboros.potrogue.blocks.GameBlock + +object GameBlockFactory { + fun floor() = GameBlock(GameTileRepository.FLOOR) + + fun wall() = GameBlock.createWith(EntityFactory.newWall()) +} \ No newline at end of file diff --git a/bin/main/group/ouroboros/potrogue/builders/GameColors.kt b/bin/main/group/ouroboros/potrogue/builders/GameColors.kt new file mode 100644 index 0000000..254b742 --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/builders/GameColors.kt @@ -0,0 +1,18 @@ +package group.ouroboros.potrogue.builders + +import org.hexworks.zircon.api.color.TileColor + +object GameColors { + //We set some colors for tiles + val wallForegroundColor = TileColor.fromString("#1e1e2e") + val wallBackgroundColor = TileColor.fromString("#cba6f7") + + val floorForegroundColor = TileColor.fromString("#1e1e2e") + val floorBackgroundColor = TileColor.fromString("#11111b") + + // Player Color? + val accentColor = TileColor.fromString("#94e2d5") + + //The Creature Color + val creatureColor = TileColor.fromString("#f9e2af") +} \ No newline at end of file diff --git a/bin/main/group/ouroboros/potrogue/builders/GameTileRepository.kt b/bin/main/group/ouroboros/potrogue/builders/GameTileRepository.kt new file mode 100644 index 0000000..3c679fe --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/builders/GameTileRepository.kt @@ -0,0 +1,46 @@ +package group.ouroboros.potrogue.builders + +import group.ouroboros.potrogue.builders.GameColors.accentColor +import group.ouroboros.potrogue.builders.GameColors.floorBackgroundColor +import group.ouroboros.potrogue.builders.GameColors.floorForegroundColor +import group.ouroboros.potrogue.builders.GameColors.wallBackgroundColor +import group.ouroboros.potrogue.builders.GameColors.wallForegroundColor +import org.hexworks.zircon.api.data.CharacterTile +import org.hexworks.zircon.api.data.Tile +import org.hexworks.zircon.api.graphics.Symbols + +object GameTileRepository { + // Factory for creating tile objects, we use basic CharacterTiles here, + // but Zircon can indeed use GraphicalTiles(textured) which will come later. + + //Empty Tile + val EMPTY: CharacterTile = Tile.empty() + + //Floor Tile + val FLOOR: CharacterTile = Tile.newBuilder() + .withCharacter(Symbols.INTERPUNCT) + .withForegroundColor(floorForegroundColor) + .withBackgroundColor(floorBackgroundColor) + .buildCharacterTile() + + //Wall Tile + val WALL: CharacterTile = Tile.newBuilder() + .withCharacter('▒') + .withForegroundColor(wallForegroundColor) + .withBackgroundColor(wallBackgroundColor) + .buildCharacterTile() + + //Player Tile + val PLAYER: CharacterTile = Tile.newBuilder() + .withCharacter('☺') + .withBackgroundColor(floorBackgroundColor) + .withForegroundColor(accentColor) + .buildCharacterTile() + + //The Creature Tile + val CREATURE = Tile.newBuilder() + .withCharacter('☻') + .withBackgroundColor(GameColors.floorBackgroundColor) + .withForegroundColor(GameColors.creatureColor) + .buildCharacterTile() +} \ No newline at end of file diff --git a/bin/main/group/ouroboros/potrogue/builders/WorldBuilder.kt b/bin/main/group/ouroboros/potrogue/builders/WorldBuilder.kt new file mode 100644 index 0000000..20d56ec --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/builders/WorldBuilder.kt @@ -0,0 +1,74 @@ +package group.ouroboros.potrogue.builders + +import group.ouroboros.potrogue.blocks.GameBlock +import group.ouroboros.potrogue.extensions.sameLevelNeighborsShuffled +import group.ouroboros.potrogue.world.World +import org.hexworks.zircon.api.data.Position3D +import org.hexworks.zircon.api.data.Size3D + +// We take the worldSize from the outside world. This is useful because later it can be parameterized. +class WorldBuilder (private val worldSize: Size3D) { + + private val width = worldSize.xLength + private val height = worldSize.zLength + // We maintain a Map of Blocks which we will use when we build the World + private var blocks: MutableMap = mutableMapOf() + + // With makeCaves we create a fluent interface so that the users of WorldBuilder can use it in a similar manner as we build Tiles and Components in Zircon. + fun makeCaves(): WorldBuilder { + return randomizeTiles() + .smooth(8) + } + + // When we build the World we take a visible size which will be used by the GameArea. + fun build(visibleSize: Size3D): World = World(blocks, visibleSize, worldSize) + + private fun randomizeTiles(): WorldBuilder { + forAllPositions { pos -> + // In Kotlin if is not a statement but an expression. This means that it returns a value so we can assign it to our Map. + blocks[pos] = if (Math.random() < 0.5) { + GameBlockFactory.floor() + } else GameBlockFactory.wall() + } + return this + } + + private fun smooth(iterations: Int): WorldBuilder { + // We are going to need a new Map of blocks for our smoothing because we can’t do it in place. Modifying the original Map would render our cellular automata algorithm useless because it needs to calculate the new state from the old state. + val newBlocks = mutableMapOf() + repeat(iterations) { + forAllPositions { pos -> + // We create a 3D world, so we need not only x and y, but also z. What you see here is called destructuring + val (x, y, z) = pos + var floors = 0 + var rocks = 0 + // Here we iterate over a list of the current position and all its neighbors + pos.sameLevelNeighborsShuffled().plus(pos).forEach { neighbor -> + // And we only care about the positions which have a corresponding block (when they are not outside the game world) + blocks.whenPresent(neighbor) { block -> + if (block.isEmptyFloor) { + floors++ + } else rocks++ + } + } + newBlocks[Position3D.create(x, y, z)] = + if (floors >= rocks) GameBlockFactory.floor() else GameBlockFactory.wall() + } + // When we’re done with smoothing we replace the old Map with the new one. + blocks = newBlocks + } + return this + } + + // This is just a convenience function for iterating over all of the world’s positions which I added as a demonstration of how functions with lambdas work. Here you can pass any function which takes a Position3D and returns Unit (Unit is the equivalent of Java’s Void). + private fun forAllPositions(fn: (Position3D) -> Unit) { + worldSize.fetchPositions().forEach(fn) + } + + // This function is an example of defining an extension function which takes a function as a parameter. What the header of the function means here is: + // + // Augment all MutableMaps which are holding Position3D to GameBlock mappings to have a function named “whenPresent” which takes a position and a function. + private fun MutableMap.whenPresent(pos: Position3D, fn: (GameBlock) -> Unit) { + this[pos]?.let(fn) + } +} \ No newline at end of file diff --git a/bin/main/group/ouroboros/potrogue/data/config/Config.kt b/bin/main/group/ouroboros/potrogue/data/config/Config.kt new file mode 100644 index 0000000..56f80ab --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/data/config/Config.kt @@ -0,0 +1,69 @@ +package group.ouroboros.potrogue.data.config + +import dev.dirs.ProjectDirectories +import java.io.File +import java.io.FileInputStream +import java.io.FileOutputStream +import java.io.OutputStream +import java.nio.file.Files +import java.nio.file.Path +import java.nio.file.Paths +import java.util.* + + +val prop = Properties() +class Config { + val confDir = ProjectDirectories.from("xyz", "limepot", "potrogue") + val runDir = File(confDir.configDir) + val confFile = File(confDir.configDir + "/potrogue.conf") + private val prop = Properties() + private var runDirExists = runDir.exists() + init { + //Check if the directories and files exist, if not, create them. Also check if config version is incorrect. + //TODO: DataPacks and Advanced configuration system (see values.conf in jar) + //Files.createDirectories(Paths.get("./run/data")) + if(!runDirExists){ + Files.createDirectories(Paths.get(confDir.configDir)) + } + if(confFile.exists()) { + FileInputStream(confFile).use { prop.load(it) } + } + //Otherwise create the necessary directories + else{ + Files.createFile(Path.of(confDir.configDir + "/potrogue.conf")) + FileInputStream(confFile).use { + prop.load(it) + prop.setProperty("configVersion", "1") + prop.setProperty("windowWidth", "80") + prop.setProperty("windowHeight", "54") + prop.setProperty("dungeonLevels", "2") + prop.setProperty("sidebarWidth", "18") + prop.setProperty("logAreaHeight", "12") + prop.setProperty("helpTipHeight", "3") + prop.setProperty("creaturesPerLevel", "15") + prop.setProperty("creatureMaxSpread", "20") + } + val out: OutputStream = FileOutputStream(confFile) + prop.store(out, "PotRogue Configuration File, restart game if changed value. HERE BE DRAGONS.") + } +} + //Convert values from the config file to in-code variables, + // so we can use them later, also make them public because I said so. + val windowWidth: Int = (prop.getProperty("windowWidth")).toInt() + + val windowHeight: Int = (prop.getProperty("windowHeight")).toInt() + + val dungeonLevels: Int = (prop.getProperty("dungeonLevels")).toInt() + + val sidebarWidth: Int = (prop.getProperty("sidebarWidth")).toInt() + + val logAreaHeight: Int = (prop.getProperty("logAreaHeight")).toInt() + + val helpTipHeight: Int = (prop.getProperty("helpTipHeight")).toInt() + + val creaturesPerLevel: Int = (prop.getProperty("creaturesPerLevel")).toInt() + + val creatureMaxSpread: Int = (prop.getProperty("creatureMaxSpread")).toInt() + + val configVersion: Int = (prop.getProperty("configVersion")).toInt() +} \ No newline at end of file diff --git a/bin/main/group/ouroboros/potrogue/data/config/GameConfig.kt b/bin/main/group/ouroboros/potrogue/data/config/GameConfig.kt new file mode 100644 index 0000000..0109799 --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/data/config/GameConfig.kt @@ -0,0 +1,39 @@ +package group.ouroboros.potrogue.data.config + +import group.ouroboros.potrogue.GAME_ID +import group.ouroboros.potrogue.GAME_VER +import org.hexworks.zircon.api.CP437TilesetResources +import org.hexworks.zircon.api.ColorThemes.newBuilder +import org.hexworks.zircon.api.application.AppConfig +import org.hexworks.zircon.api.color.TileColor +import org.hexworks.zircon.api.data.Size3D + + +object GameConfig { + // look & feel + var TILESET = CP437TilesetResources.loadTilesetFromJar(16, 16, "/assets/tilesets/potrogue_grunge_16x16.png") + + val WORLD_SIZE = Size3D.create(Config().windowWidth * 3, Config().windowHeight * 3 , Config().dungeonLevels) + val GAME_AREA_SIZE = Size3D.create( + xLength = Config().windowWidth - Config().sidebarWidth, + yLength = Config().windowHeight - Config().logAreaHeight, + zLength = Config().dungeonLevels + ) + + fun buildAppConfig() = AppConfig.newBuilder() + .withDefaultTileset(TILESET) + .withSize(Config().windowWidth, Config().windowHeight) + .withTitle("$GAME_ID | $GAME_VER") + .withIcon("assets/icon.png") + .build() + + var catppuccinMocha = newBuilder() + .withAccentColor(TileColor.fromString("#b4befe")) + .withPrimaryForegroundColor(TileColor.fromString("#f5c2e7")) + .withSecondaryForegroundColor(TileColor.fromString("#cba6f7")) + .withPrimaryBackgroundColor(TileColor.fromString("#1e1e2e")) + .withSecondaryBackgroundColor(TileColor.fromString("#11111b")) + .build() + + val THEME = catppuccinMocha +} diff --git a/bin/main/group/ouroboros/potrogue/entity/attributes/CreatureSpread.kt b/bin/main/group/ouroboros/potrogue/entity/attributes/CreatureSpread.kt new file mode 100644 index 0000000..a1ae661 --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/entity/attributes/CreatureSpread.kt @@ -0,0 +1,9 @@ +package group.ouroboros.potrogue.entity.attributes + +import group.ouroboros.potrogue.data.config.Config +import org.hexworks.amethyst.api.base.BaseAttribute + +data class CreatureSpread( + var spreadCount: Int = 0, + val maximumSpread: Int = Config().creatureMaxSpread +) : BaseAttribute() diff --git a/bin/main/group/ouroboros/potrogue/entity/attributes/EntityActions.kt b/bin/main/group/ouroboros/potrogue/entity/attributes/EntityActions.kt new file mode 100644 index 0000000..618dcb0 --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/entity/attributes/EntityActions.kt @@ -0,0 +1,40 @@ +package group.ouroboros.potrogue.entity.attributes + +import group.ouroboros.potrogue.entity.messages.EntityAction +import group.ouroboros.potrogue.extensions.GameEntity +import group.ouroboros.potrogue.world.GameContext +import org.hexworks.amethyst.api.base.BaseAttribute +import org.hexworks.amethyst.api.entity.EntityType +import kotlin.reflect.KClass + +class EntityActions ( + // This Attribute is capable of holding classes of any kind of EntityAction. + // We use vararg here which is similar to how varargs work in Java: + // we can create the EntityActions object with any number of constructor parameters like this: + // EntityActions(Dig::class, Look::class). + // We need to use the class objects (KClass) here instead of the actual EntityAction objects because each time we perform an action + // a new EntityAction has to be created. + // So you can think about actions here as templates. + private vararg val actions: KClass> +) : BaseAttribute() { + + // This function can be used to create the actual EntityAction objects by using the given context, source and target + fun createActionsFor( + context: GameContext, + source: GameEntity, + target: GameEntity + ): Iterable> { + return actions.map { + try { + // When we create the actions we just call the first constructor of the class and hope for the best. + // There is no built-in way in Kotlin (nor in Java) to make sure that a class has a specific constructor in compile time so that’s why + it.constructors.first().call(context, source, target) + + // We catch any exceptions and rethrow them here stating that the operation failed. + // We just have to remember that whenever we create an EntityAction it has a constructor for the 3 mandatory fields. + } catch (e: Exception) { + throw IllegalArgumentException("Can't create EntityAction. Does it have the proper constructor?") + } + } + } +} \ No newline at end of file diff --git a/bin/main/group/ouroboros/potrogue/entity/attributes/EntityPosition.kt b/bin/main/group/ouroboros/potrogue/entity/attributes/EntityPosition.kt new file mode 100644 index 0000000..0880505 --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/entity/attributes/EntityPosition.kt @@ -0,0 +1,26 @@ +package group.ouroboros.potrogue.entity.attributes + +import org.hexworks.amethyst.api.base.BaseAttribute +import org.hexworks.cobalt.databinding.api.extension.toProperty +import org.hexworks.zircon.api.data.Position3D +class EntityPosition( + // We add initialPosition as a constructor parameter to our class and its default value is unknown. + // What’s this? Position3D comes from Zircon + // and can be used to represent a point in 3D space (as we have discussed before), + // and unknown implements the Null Object Pattern for us. + initialPosition: Position3D = Position3D.unknown() +) : BaseAttribute() { + + // Here we create a private Property from the initialPosition. + // What’s a Property you might ask? Well, it is used for data binding. + // A Property is a wrapper for a value that can change over time. + // It can be bound to other Property objects + // so their values change together, and you can also add change listeners to them. + // Property comes from the Cobalt library we use, and it works in a very similar way as properties work in JavaFX. + private val positionProperty = initialPosition.toProperty() + + // We create a Kotlin delegate from our Property. + // This means that position will be accessible to the outside world + // as if it was a simple field, but it takes its value from our Property under the hood. + var position: Position3D by positionProperty.asDelegate() +} \ No newline at end of file diff --git a/bin/main/group/ouroboros/potrogue/entity/attributes/EntityTile.kt b/bin/main/group/ouroboros/potrogue/entity/attributes/EntityTile.kt new file mode 100644 index 0000000..8d2b899 --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/entity/attributes/EntityTile.kt @@ -0,0 +1,7 @@ +package group.ouroboros.potrogue.entity.attributes + +import org.hexworks.amethyst.api.base.BaseAttribute +import org.hexworks.zircon.api.data.Tile + +// EntityTile is an Attribute that holds the Tile of an Entity we use to display it in our world +data class EntityTile(val tile: Tile = Tile.empty()) : BaseAttribute() diff --git a/bin/main/group/ouroboros/potrogue/entity/attributes/flags/BlockOccupier.kt b/bin/main/group/ouroboros/potrogue/entity/attributes/flags/BlockOccupier.kt new file mode 100644 index 0000000..49e16cc --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/entity/attributes/flags/BlockOccupier.kt @@ -0,0 +1,5 @@ +package group.ouroboros.potrogue.entity.attributes.flags + +import org.hexworks.amethyst.api.base.BaseAttribute + +object BlockOccupier : BaseAttribute() \ No newline at end of file diff --git a/bin/main/group/ouroboros/potrogue/entity/attributes/types/EntityTypes.kt b/bin/main/group/ouroboros/potrogue/entity/attributes/types/EntityTypes.kt new file mode 100644 index 0000000..6428c24 --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/entity/attributes/types/EntityTypes.kt @@ -0,0 +1,15 @@ +package group.ouroboros.potrogue.entity.attributes.types + +import org.hexworks.amethyst.api.base.BaseEntityType + +object Player : BaseEntityType( + name = "player" +) + +object Wall : BaseEntityType( + name = "wall" +) + +object Creature : BaseEntityType( + name = "creature" +) \ No newline at end of file diff --git a/bin/main/group/ouroboros/potrogue/entity/messages/Attack.kt b/bin/main/group/ouroboros/potrogue/entity/messages/Attack.kt new file mode 100644 index 0000000..1262c00 --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/entity/messages/Attack.kt @@ -0,0 +1,11 @@ +package group.ouroboros.potrogue.entity.messages + +import group.ouroboros.potrogue.extensions.GameEntity +import group.ouroboros.potrogue.world.GameContext +import org.hexworks.amethyst.api.entity.EntityType + +data class Attack( + override val context: GameContext, + override val source: GameEntity, + override val target: GameEntity +) : EntityAction \ No newline at end of file diff --git a/bin/main/group/ouroboros/potrogue/entity/messages/Dig.kt b/bin/main/group/ouroboros/potrogue/entity/messages/Dig.kt new file mode 100644 index 0000000..a5caeb2 --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/entity/messages/Dig.kt @@ -0,0 +1,11 @@ +package group.ouroboros.potrogue.entity.messages + +import group.ouroboros.potrogue.extensions.GameEntity +import group.ouroboros.potrogue.world.GameContext +import org.hexworks.amethyst.api.entity.EntityType + +data class Dig( + override val context: GameContext, + override val source: GameEntity, + override val target: GameEntity +) : EntityAction diff --git a/bin/main/group/ouroboros/potrogue/entity/messages/EntityAction.kt b/bin/main/group/ouroboros/potrogue/entity/messages/EntityAction.kt new file mode 100644 index 0000000..50238c2 --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/entity/messages/EntityAction.kt @@ -0,0 +1,34 @@ +package group.ouroboros.potrogue.entity.messages + +import group.ouroboros.potrogue.extensions.GameEntity +import group.ouroboros.potrogue.extensions.GameMessage +import group.ouroboros.potrogue.world.GameContext +import org.hexworks.amethyst.api.entity.EntityType + +// Our EntityAction is different from a regular GameMessage in a way that it also has a target. +// So an EntityAction represents a source trying to perform an action on target. + +// We have two generic type parameters, S and T. +// S is the EntityType of the source, T is the EntityType of the target. +// This will be useful later on as we’ll see. +interface EntityAction : GameMessage { + + // We save the reference to target in all EntityActions + val target: GameEntity + + // The component1, component2 … componentN methods implement destructuring in Kotlin. + // Since destructuring is positional as we’ve seen previously by implementing the + // component* functions, we can control how an EntityAction can be destructured. + // In our case with these 3 operator functions, we can destructure any EntityActions like this: + // + //val (context, source, target) = entityAction + operator fun component1() = context + operator fun component2() = source + operator fun component3() = target + + data class Attack( + override val context: GameContext, + override val source: GameEntity, + override val target: GameEntity + ) : EntityAction +} diff --git a/bin/main/group/ouroboros/potrogue/entity/messages/MoveCamera.kt b/bin/main/group/ouroboros/potrogue/entity/messages/MoveCamera.kt new file mode 100644 index 0000000..cd283ad --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/entity/messages/MoveCamera.kt @@ -0,0 +1,13 @@ +package group.ouroboros.potrogue.entity.messages + +import group.ouroboros.potrogue.extensions.GameEntity +import group.ouroboros.potrogue.extensions.GameMessage +import group.ouroboros.potrogue.world.GameContext +import org.hexworks.amethyst.api.entity.EntityType +import org.hexworks.zircon.api.data.Position3D + +data class MoveCamera( + override val context: GameContext, + override val source: GameEntity, + val previousPosition: Position3D +) : GameMessage diff --git a/bin/main/group/ouroboros/potrogue/entity/messages/MoveTo.kt b/bin/main/group/ouroboros/potrogue/entity/messages/MoveTo.kt new file mode 100644 index 0000000..4dd765d --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/entity/messages/MoveTo.kt @@ -0,0 +1,13 @@ +package group.ouroboros.potrogue.entity.messages + +import group.ouroboros.potrogue.extensions.GameEntity +import group.ouroboros.potrogue.extensions.GameMessage +import group.ouroboros.potrogue.world.GameContext +import org.hexworks.amethyst.api.entity.EntityType +import org.hexworks.zircon.api.data.Position3D + +data class MoveTo( + override val context: GameContext, + override val source: GameEntity, + val position: Position3D +) : GameMessage diff --git a/bin/main/group/ouroboros/potrogue/entity/systems/Attackable.kt b/bin/main/group/ouroboros/potrogue/entity/systems/Attackable.kt new file mode 100644 index 0000000..a86dfec --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/entity/systems/Attackable.kt @@ -0,0 +1,15 @@ +package group.ouroboros.potrogue.entity.systems + +import group.ouroboros.potrogue.entity.messages.Attack +import group.ouroboros.potrogue.world.GameContext +import org.hexworks.amethyst.api.Consumed +import org.hexworks.amethyst.api.Response +import org.hexworks.amethyst.api.base.BaseFacet + +object Attackable : BaseFacet(Attack::class) { + override suspend fun receive(message: Attack): Response { + val (context, _, target) = message + context.world.removeEntity(target) + return Consumed + } +} \ No newline at end of file diff --git a/bin/main/group/ouroboros/potrogue/entity/systems/CameraMover.kt b/bin/main/group/ouroboros/potrogue/entity/systems/CameraMover.kt new file mode 100644 index 0000000..f80311e --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/entity/systems/CameraMover.kt @@ -0,0 +1,44 @@ +package group.ouroboros.potrogue.entity.systems + +import group.ouroboros.potrogue.entity.messages.MoveCamera +import group.ouroboros.potrogue.extensions.position +import group.ouroboros.potrogue.world.GameContext +import org.hexworks.amethyst.api.Consumed +import org.hexworks.amethyst.api.Response +import org.hexworks.amethyst.api.base.BaseFacet + +object CameraMover : BaseFacet(MoveCamera::class) { + + override suspend fun receive(message: MoveCamera): Response { + val (context, source, previousPosition) = message + val world = context.world + // The player’s position on the screen can be calculated + // by subtracting the World’s visibleOffset from the player’s position. + + // The visibleOffset is the top left position of the + // visible part of the World relative to the top left corner of the whole World (which is 0, 0). + val screenPos = source.position - world.visibleOffset + // We calculate the center position of the visible part of the world here + val halfHeight = world.visibleSize.yLength / 2 + val halfWidth = world.visibleSize.xLength / 2 + val currentPosition = source.position + // And we only move the camera if we moved in a certain direction + // (left, for example) and the Entity’s position on the screen is left of the middle position. + // The logic is the same for all directions, but we use the corresponding x or y coordinate + when { + previousPosition.y > currentPosition.y && screenPos.y < halfHeight -> { + world.scrollOneBackward() + } + previousPosition.y < currentPosition.y && screenPos.y > halfHeight -> { + world.scrollOneForward() + } + previousPosition.x > currentPosition.x && screenPos.x < halfWidth -> { + world.scrollOneLeft() + } + previousPosition.x < currentPosition.x && screenPos.x > halfWidth -> { + world.scrollOneRight() + } + } + return Consumed + } +} \ No newline at end of file diff --git a/bin/main/group/ouroboros/potrogue/entity/systems/CreatureGrowth.kt b/bin/main/group/ouroboros/potrogue/entity/systems/CreatureGrowth.kt new file mode 100644 index 0000000..8c30555 --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/entity/systems/CreatureGrowth.kt @@ -0,0 +1,42 @@ +package group.ouroboros.potrogue.entity.systems + +import group.ouroboros.potrogue.builders.EntityFactory +import group.ouroboros.potrogue.entity.attributes.CreatureSpread +import group.ouroboros.potrogue.extensions.position +import group.ouroboros.potrogue.extensions.tryToFindAttribute +import group.ouroboros.potrogue.world.GameContext +import org.hexworks.amethyst.api.base.BaseBehavior +import org.hexworks.amethyst.api.entity.Entity +import org.hexworks.amethyst.api.entity.EntityType +import org.hexworks.zircon.api.data.Size3D + +// We create a Behavior and supply CreatureSpread as a mandatory Attribute to it +object CreatureGrowth : BaseBehavior(CreatureSpread::class) { + + override suspend fun update(entity: Entity, context: GameContext): Boolean { + val world = context.world + // When update is called with an entity we try to find its CreatureSpread Attribute. + // We know that it is there so we don’t have to use the findAttribute method. + val creatureSpread = entity.tryToFindAttribute(CreatureSpread::class) + // Destructuring works for CreatureSpread because it is a data class + val (spreadCount, maxSpread) = creatureSpread + // You can specify any probability here. + // It will have a direct effect on how often The Creature spreads. + // Feel free to tinker with this number but don’t be surprised if you find yourself in a creaturesplosion! + return if (spreadCount < maxSpread && Math.random() < 0.015) { + world.findEmptyLocationWithin( + offset = entity.position + .withRelativeX(-1) + .withRelativeY(-1), + size = Size3D.create(3, 3, 0) + ).map { emptyLocation -> + // Note that we pass creatureSpread as a parameter to newCreature + // so that all Creatures in the same Creature colony can share this Attribute. + // This makes sure that Creatures won’t spread all over the place and the size of a colony is controlled + world.addEntity(EntityFactory.newCreature(creatureSpread), emptyLocation) + creatureSpread.spreadCount++ + } + true + } else false + } +} \ No newline at end of file diff --git a/bin/main/group/ouroboros/potrogue/entity/systems/Diggable.kt b/bin/main/group/ouroboros/potrogue/entity/systems/Diggable.kt new file mode 100644 index 0000000..9cd4c46 --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/entity/systems/Diggable.kt @@ -0,0 +1,15 @@ +package group.ouroboros.potrogue.entity.systems + +import group.ouroboros.potrogue.entity.messages.Dig +import group.ouroboros.potrogue.world.GameContext +import org.hexworks.amethyst.api.Consumed +import org.hexworks.amethyst.api.Response +import org.hexworks.amethyst.api.base.BaseFacet + +object Diggable : BaseFacet(Dig::class) { + override suspend fun receive(message: Dig): Response { + val (context, _, target) = message + context.world.removeEntity(target) + return Consumed + } +} \ No newline at end of file diff --git a/bin/main/group/ouroboros/potrogue/entity/systems/InputReceiver.kt b/bin/main/group/ouroboros/potrogue/entity/systems/InputReceiver.kt new file mode 100644 index 0000000..40abd6e --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/entity/systems/InputReceiver.kt @@ -0,0 +1,43 @@ +package group.ouroboros.potrogue.entity.systems + +import group.ouroboros.potrogue.entity.messages.MoveTo +import group.ouroboros.potrogue.extensions.position +import group.ouroboros.potrogue.world.GameContext +import org.hexworks.amethyst.api.base.BaseBehavior +import org.hexworks.amethyst.api.entity.Entity +import org.hexworks.amethyst.api.entity.EntityType +import org.hexworks.zircon.api.uievent.KeyCode +import org.hexworks.zircon.api.uievent.KeyboardEvent + +// InputReceiver checks for WASD, and acts accordingly +object InputReceiver : BaseBehavior() { + + override suspend fun update(entity: Entity, context: GameContext): Boolean { + // We destructure our context object so its properties are easier to access. + // Destructuring is positional, so here _ means that we don’t care about that specific property. + val (_, _, uiEvent, player) = context + val currentPos = player.position + // We only want KeyboardEvents for now so we check with the is operator. + // This is similar as the instanceof operator in Java but a bit more useful. + + if (uiEvent is KeyboardEvent) { + // We use when which is similar to switch in Java to check which key was pressed. + // Zircon has a KeyCode for all keys which can be pressed. when in Kotlin is also an expression, + // and not a statement, so it returns a value. We can change it into our newPosition variable. + + val newPosition = when (uiEvent.code) { + KeyCode.KEY_W -> currentPos.withRelativeY(-1) + KeyCode.KEY_A -> currentPos.withRelativeX(-1) + KeyCode.KEY_S -> currentPos.withRelativeY(1) + KeyCode.KEY_D -> currentPos.withRelativeX(1) + else -> { + // If some key is pressed other than WASD, then we just return the current position, so no movement will happen + currentPos + } + } + // We receive the MoveTo message on our player here. + player.receiveMessage(MoveTo(context, player, newPosition)) + } + return true + } +} diff --git a/bin/main/group/ouroboros/potrogue/entity/systems/Movable.kt b/bin/main/group/ouroboros/potrogue/entity/systems/Movable.kt new file mode 100644 index 0000000..21c50bd --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/entity/systems/Movable.kt @@ -0,0 +1,87 @@ +package group.ouroboros.potrogue.entity.systems + +import group.ouroboros.potrogue.entity.attributes.types.Player +import group.ouroboros.potrogue.entity.messages.MoveCamera +import group.ouroboros.potrogue.entity.messages.MoveTo +import group.ouroboros.potrogue.extensions.position +import group.ouroboros.potrogue.extensions.tryActionsOn +import group.ouroboros.potrogue.world.GameContext +import org.hexworks.amethyst.api.Consumed +import org.hexworks.amethyst.api.MessageResponse +import org.hexworks.amethyst.api.Pass +import org.hexworks.amethyst.api.Response +import org.hexworks.amethyst.api.base.BaseFacet + +/* +* Hey, what’s Pass and Consumed? +* Why do we have to return anything? Good question! When an Entity receives a Message it tries to send the given message to its Facets in order. +* Each Facet has to return a Response. There are 3 kinds: Pass, Consumed and MessageResponse. If we return Pass, the loop continues and the entity tries the next Facet. +* If we return Consumed, the loop stops. MessageResponse is special, we can return a new message using it and the entity will continue the loop using the new Message! +* This is useful for implementing complex interactions between entities + */ + +// A Facet accepts only a specific message, so we have to indicate that we only handle MoveTo. +object Movable : BaseFacet(MoveTo::class) { + + override suspend fun receive(message: MoveTo): Response { + /* + * This funky (context, entity, position) code is called Destructuring. + * This might be familiar for Python folks and what it does is that it unpacks the values from an object which supports it. So writing this: + * val (context, entity, position) = myObj + * + * Is the equivalent of writing this: + * val context = myObj.context + * val entity = myObj.entity + * val position = myObj.position + */ + val (context, entity, position) = message + val world = context.world + // we save the previous position before we change it + val previousPosition = entity.position + // Here we say that we’ll return Pass as a default + var result: Response = Pass + /* + // Then we check whether moving the entity was successful or not (remember the success return value?) + if (world.moveEntity(entity, position)) { + // If the move was successful and the entity we moved is the player + result = if (entity.type == Player) { + MessageResponse( + // We return the MessageResponse + MoveCamera( + context = context, + source = entity, + previousPosition = previousPosition + ) + ) + // Otherwise we keep the Consumed response + } else Consumed + } + // Finally we return the result + return result + }*/ + + // We will only do anything if there is a block at the given position. + // It is possible that there are no blocks at the edge of the map for example (if we want to move off the map) + world.fetchBlockAtOrNull(position)?.let { block -> + if (block.isOccupied) { + // If the block is occupied we try our actions on the block + result = entity.tryActionsOn(context, block.occupier.get()) + } else { + //Otherwise we do what we were doing before + if (world.moveEntity(entity, position)) { + result = Consumed + if (entity.type == Player) { + result = MessageResponse( + MoveCamera( + context = context, + source = entity, + previousPosition = previousPosition + ) + ) + } + } + } + } + return result + } +} \ No newline at end of file diff --git a/bin/main/group/ouroboros/potrogue/extensions/EntityExtensions.kt b/bin/main/group/ouroboros/potrogue/extensions/EntityExtensions.kt new file mode 100644 index 0000000..821f2f2 --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/extensions/EntityExtensions.kt @@ -0,0 +1,34 @@ +package group.ouroboros.potrogue.extensions + +import group.ouroboros.potrogue.entity.attributes.EntityActions +import group.ouroboros.potrogue.entity.attributes.flags.BlockOccupier +import group.ouroboros.potrogue.world.GameContext +import org.hexworks.amethyst.api.Consumed +import org.hexworks.amethyst.api.Pass +import org.hexworks.amethyst.api.Response + +// We define this function as an extension function on AnyGameEntity. +// This means that from now on we can call tryActionsOn on any of our entities! +// It is also suspending fun because the receiveMessage function we call later is also a suspending function. +// Suspending is part of the Kotlin Coroutines API, and it is a deep topic. +// We’re not going to cover it here as we don’t take advantage of it +suspend fun AnyGameEntity.tryActionsOn(context: GameContext, target: AnyGameEntity): Response { + var result: Response = Pass + // We can only try the actions of an entity which has at least one, so we try to find the attribute. + findAttributeOrNull(EntityActions::class)?.let { + // if we find the attribute, we just create the actions for our context/source/target combination + it.createActionsFor(context, this, target).forEach { action -> + // And we then send the message to the target for + // immediate processing, and if the message is Consumed, it means that + if (target.receiveMessage(action) is Consumed) { + result = Consumed + // We can break out of the forEach block. + return@forEach + } + } + } + return result +} + +val AnyGameEntity.occupiesBlock: Boolean + get() = findAttribute(BlockOccupier::class).isPresent diff --git a/bin/main/group/ouroboros/potrogue/extensions/PositionExtensions.kt b/bin/main/group/ouroboros/potrogue/extensions/PositionExtensions.kt new file mode 100644 index 0000000..f161f76 --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/extensions/PositionExtensions.kt @@ -0,0 +1,20 @@ +package group.ouroboros.potrogue.extensions + +import org.hexworks.zircon.api.data.Position3D + +// We add the extension function to Position3D. +// We do it by defining a function not with a simple name, but by the format: +// fun .: return type { // .... +fun Position3D.sameLevelNeighborsShuffled(): List { + return (-1..1).flatMap { x -> + // We use functional programming here. + // flatMap and map work in a similar way as you might've been used to it in Java 8’s Stream API. + (-1..1).map { y -> + // When you write extension functions, this will be bound to the class being extended. + // So this here will point to the Position3D instance on which sameLevelNeighborsShuffled is called. + this.withRelativeX(x).withRelativeY(y) + } + // minus here will remove this position from the List and return a new List. + // shuffled will also return a new list which contains the same elements but shuffled. + }.minus(this).shuffled() +} diff --git a/bin/main/group/ouroboros/potrogue/extensions/TypeAliases.kt b/bin/main/group/ouroboros/potrogue/extensions/TypeAliases.kt new file mode 100644 index 0000000..620fbf0 --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/extensions/TypeAliases.kt @@ -0,0 +1,35 @@ +package group.ouroboros.potrogue.extensions + +import group.ouroboros.potrogue.entity.attributes.EntityPosition +import group.ouroboros.potrogue.entity.attributes.EntityTile +import group.ouroboros.potrogue.world.GameContext +import org.hexworks.amethyst.api.Attribute +import org.hexworks.amethyst.api.Message +import org.hexworks.amethyst.api.entity.Entity +import org.hexworks.amethyst.api.entity.EntityType +import org.hexworks.zircon.api.data.Tile +import kotlin.reflect.KClass + +typealias AnyGameEntity = GameEntity +typealias GameEntity = Entity +typealias GameMessage = Message + +// Create an extension property (works the same way as an extension function) on AnyGameEntity. +var AnyGameEntity.position + // Define a getter for it which tries to find the + // EntityPosition attribute in our Entity and throws and exception if the Entity has no position. + get() = tryToFindAttribute(EntityPosition::class).position + // We also define a setter for it which sets the Property we defined before + set(value) { + findAttribute(EntityPosition::class).map { + it.position = value + } + } + +val AnyGameEntity.tile: Tile + get() = this.tryToFindAttribute(EntityTile::class).tile + +// Define a function which implements the “try to find or throw an exception” logic for both of our properties. +fun AnyGameEntity.tryToFindAttribute(klass: KClass): T = findAttribute(klass).orElseThrow { + NoSuchElementException("Entity '$this' has no property with type '${klass.simpleName}'.") +} \ No newline at end of file diff --git a/bin/main/group/ouroboros/potrogue/main.kt b/bin/main/group/ouroboros/potrogue/main.kt new file mode 100644 index 0000000..4218176 --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/main.kt @@ -0,0 +1,22 @@ +package group.ouroboros.potrogue + +import group.ouroboros.potrogue.data.config.Config +import group.ouroboros.potrogue.data.config.GameConfig +import group.ouroboros.potrogue.view.StartView +import org.hexworks.zircon.api.SwingApplications +import org.hexworks.zircon.api.VirtualApplications + +// Important Values +const val GAME_ID = "PotRogue" +const val GAME_VER = "0.1.0-DEV" +const val confVers = 1 + +fun main() { + Config() + if (Config().configVersion != confVers){ + Config().confFile.delete() + } + // Start Application + val grid = SwingApplications.startTileGrid(GameConfig.buildAppConfig()) + StartView(grid).dock() +} diff --git a/bin/main/group/ouroboros/potrogue/util/ResourceGetter.kt b/bin/main/group/ouroboros/potrogue/util/ResourceGetter.kt new file mode 100644 index 0000000..a0b9313 --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/util/ResourceGetter.kt @@ -0,0 +1,14 @@ +package group.ouroboros.potrogue.util + +import java.net.URL +import java.nio.file.Files +import java.nio.file.Paths + +class ResourceGetter { + fun downloadFile(url: URL, fileName: String) { + url.openStream().use { Files.copy(it, Paths.get(fileName)) } + } + + //EXAMPLE USAGE + // ResourceGetter().downloadFile(URL("https://url.to/resource.txt"), "location/to/store/resource.txt") +} \ No newline at end of file diff --git a/bin/main/group/ouroboros/potrogue/view/ConfigView.kt b/bin/main/group/ouroboros/potrogue/view/ConfigView.kt new file mode 100644 index 0000000..84ffc97 --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/view/ConfigView.kt @@ -0,0 +1,56 @@ +package group.ouroboros.potrogue.view + +import group.ouroboros.potrogue.data.config.GameConfig +import org.hexworks.zircon.api.CP437TilesetResources +import org.hexworks.zircon.api.ComponentDecorations +import org.hexworks.zircon.api.Components +import org.hexworks.zircon.api.component.ColorTheme +import org.hexworks.zircon.api.component.ComponentAlignment +import org.hexworks.zircon.api.grid.TileGrid +import org.hexworks.zircon.api.view.base.BaseView + +class ConfigView (private val grid: TileGrid, theme: ColorTheme = GameConfig.THEME) : BaseView(grid, theme) { + init { + val msg = "Pre-Game Configuration" + + // a text box can hold headers, paragraphs and list items + // `contentWidth = ` here is a so-called keyword parameter + // using them you can pass parameters not by their order + // but by their name. + // this might be familiar for Python programmers + val header = Components.textBox(contentWidth = msg.length) + // we add a header + .addHeader(msg) + // and a new line + .addNewLine() + // and align it to center + .withAlignmentWithin(screen, ComponentAlignment.TOP_CENTER) + .build() // finally, we build the component + + //TODO: Options: world size, character tile (smiley, @, &), character customizations (class, looks, stats, start), + + val tilesetButton = Components.button() + .withAlignmentWithin(screen, ComponentAlignment.CENTER) + .withText("CHANGE TILESET") + .withDecorations(ComponentDecorations.box(), ComponentDecorations.shadow()) + .build() + + val backButton = Components.button() + .withAlignmentWithin(screen, ComponentAlignment.BOTTOM_CENTER) + .withText("BACK") + .withDecorations(ComponentDecorations.box(), ComponentDecorations.shadow()) + .build() + + tilesetButton.onActivated { + GameConfig.TILESET = CP437TilesetResources.anikki16x16() + } + + //Once the back button is activated, go back to startView + backButton.onActivated { + replaceWith(StartView(grid)) + } + + // We can add multiple components at once + //Bake The Cake + screen.addComponents(header,backButton,tilesetButton) + }} \ No newline at end of file diff --git a/bin/main/group/ouroboros/potrogue/view/LoseView.kt b/bin/main/group/ouroboros/potrogue/view/LoseView.kt new file mode 100644 index 0000000..4e14bf6 --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/view/LoseView.kt @@ -0,0 +1,49 @@ +package group.ouroboros.potrogue.view + + +import group.ouroboros.potrogue.data.config.GameConfig +import org.hexworks.zircon.api.ComponentDecorations.box +import org.hexworks.zircon.api.Components +import org.hexworks.zircon.api.component.ColorTheme +import org.hexworks.zircon.api.component.ComponentAlignment +import org.hexworks.zircon.api.grid.TileGrid +import org.hexworks.zircon.api.view.base.BaseView +import kotlin.system.exitProcess + +class LoseView (private val grid: TileGrid, theme: ColorTheme = GameConfig.THEME) : BaseView(grid, theme) { + + init { + //Title + val header = Components.header() + .withText("Game Over") + .withAlignmentWithin(screen, ComponentAlignment.CENTER) + .build() + + //Reset Button + val restartButton = Components.button() + .withAlignmentAround(header, ComponentAlignment.BOTTOM_LEFT) + .withText("Restart") + .withDecorations(box()) + .build() + + //Quit Button + val exitButton = Components.button() + .withAlignmentAround(header, ComponentAlignment.BOTTOM_RIGHT) + .withText("Quit") + .withDecorations(box()) + .build() + + //On Reset Button activated, move back to PlayView + restartButton.onActivated { + replaceWith(PlayView(grid)) + } + + //On Quit BButton activated, exit program + exitButton.onActivated { + exitProcess(0) + } + + //Bake the cake + screen.addComponents(header, restartButton, exitButton) + } +} \ No newline at end of file diff --git a/bin/main/group/ouroboros/potrogue/view/PauseView.kt b/bin/main/group/ouroboros/potrogue/view/PauseView.kt new file mode 100644 index 0000000..f625a0d --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/view/PauseView.kt @@ -0,0 +1,54 @@ +package group.ouroboros.potrogue.view + +import group.ouroboros.potrogue.data.config.GameConfig +import org.hexworks.zircon.api.ComponentDecorations +import org.hexworks.zircon.api.Components +import org.hexworks.zircon.api.component.ColorTheme +import org.hexworks.zircon.api.component.ComponentAlignment +import org.hexworks.zircon.api.grid.TileGrid +import org.hexworks.zircon.api.view.base.BaseView + +class PauseView(private val grid: TileGrid, theme: ColorTheme = GameConfig.THEME) : BaseView(grid, theme) { + init { + val msg = "Pre-Game Configuration" + + // a text box can hold headers, paragraphs and list items + // `contentWidth = ` here is a so-called keyword parameter + // using them you can pass parameters not by their order + // but by their name. + // this might be familiar for Python programmers + val header = Components.textBox(contentWidth = msg.length) + // we add a header + .addHeader(msg) + // and a new line + .addNewLine() + // and align it to center + .withAlignmentWithin(screen, ComponentAlignment.TOP_CENTER) + .build() // finally, we build the component + + val backButton = Components.button() + .withAlignmentWithin(screen, ComponentAlignment.BOTTOM_CENTER) + .withText("RESUME") + .withDecorations(ComponentDecorations.box(), ComponentDecorations.shadow()) + .build() + + val resumeButton = Components.button() + .withAlignmentWithin(screen, ComponentAlignment.BOTTOM_CENTER) + .withText("RESUME") + .withDecorations(ComponentDecorations.box(), ComponentDecorations.shadow()) + .build() + + //Once the back button is activated, go back to startView + backButton.onActivated { + replaceWith(StartView(grid)) + } + + resumeButton.onActivated { + replaceWith(PlayView(grid)) + } + + // We can add multiple components at once + //Bake The Cake + screen.addComponents(header,backButton) + } +} \ No newline at end of file diff --git a/bin/main/group/ouroboros/potrogue/view/PlayView.kt b/bin/main/group/ouroboros/potrogue/view/PlayView.kt new file mode 100644 index 0000000..5d0b06e --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/view/PlayView.kt @@ -0,0 +1,77 @@ +package group.ouroboros.potrogue.view + +import group.ouroboros.potrogue.builders.GameTileRepository +import group.ouroboros.potrogue.data.config.Config +import group.ouroboros.potrogue.data.config.GameConfig +import group.ouroboros.potrogue.world.Game +import group.ouroboros.potrogue.world.GameBuilder +import org.hexworks.cobalt.databinding.api.extension.toProperty +import org.hexworks.zircon.api.ComponentDecorations.box +import org.hexworks.zircon.api.Components +import org.hexworks.zircon.api.component.ColorTheme +import org.hexworks.zircon.api.component.ComponentAlignment +import org.hexworks.zircon.api.game.ProjectionMode +import org.hexworks.zircon.api.grid.TileGrid +import org.hexworks.zircon.api.uievent.* +import org.hexworks.zircon.api.view.base.BaseView +import org.hexworks.zircon.internal.game.impl.GameAreaComponentRenderer + + +class PlayView (private val grid: TileGrid, private val game: Game = GameBuilder.create(), theme: ColorTheme = GameConfig.THEME) : BaseView(grid, theme) { + init { + //Create Sidebar + val sidebar = Components.panel() + .withPreferredSize(Config().sidebarWidth, Config().windowHeight - Config().logAreaHeight) + .withDecorations(box()) + .build() + + //Create area for logging + val logArea = Components.logArea() + .withDecorations(box(title = "Log")) + .withPreferredSize(Config().windowWidth, Config().logAreaHeight) + .withAlignmentWithin(screen, ComponentAlignment.BOTTOM_RIGHT) + .build() + + //Create help tooltip + val helpTip = Components.panel() + .withPreferredSize(Config().windowWidth - Config().sidebarWidth, Config().helpTipHeight) + .withPosition(Config().sidebarWidth, 42 - Config().helpTipHeight) + .withDecorations(box(title = "Help")) + .build() + + //Create Game view + val gameComponent = Components.panel() + .withPreferredSize(game.world.visibleSize.to2DSize()) + .withComponentRenderer( + GameAreaComponentRenderer( + gameArea = game.world, + projectionMode = ProjectionMode.TOP_DOWN.toProperty(), + fillerTile = GameTileRepository.FLOOR + ) + ) + .withAlignmentWithin(screen, ComponentAlignment.TOP_RIGHT) + .build() + + screen.addComponents(sidebar, logArea, helpTip, gameComponent) + + // modify our PlayView to update our world whenever the user presses a key + screen.handleKeyboardEvents(KeyboardEventType.KEY_PRESSED) { event, _ -> + game.world.update(screen, event, game) + Processed + } + + grid.handleKeyboardEvents(KeyboardEventType.KEY_PRESSED) label@{ event: KeyboardEvent, phase: UIEventPhase? -> + // we filter for KeyCode.ESCAPE only + if (event.code == KeyCode.ESCAPE) { + // only prints it when we press Arrow Up + replaceWith(PauseView(grid)) + return@label UIEventResponse.processed() + } else { + // otherwise we just pass on it + return@label UIEventResponse.pass() // we didn't handle it so we pass on the event + } + } + + + } +} \ No newline at end of file diff --git a/bin/main/group/ouroboros/potrogue/view/StartView.kt b/bin/main/group/ouroboros/potrogue/view/StartView.kt new file mode 100644 index 0000000..b99f622 --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/view/StartView.kt @@ -0,0 +1,73 @@ +package group.ouroboros.potrogue.view + +import group.ouroboros.potrogue.GAME_ID +import group.ouroboros.potrogue.data.config.GameConfig +import org.hexworks.zircon.api.ComponentDecorations.box +import org.hexworks.zircon.api.ComponentDecorations.shadow +import org.hexworks.zircon.api.Components +import org.hexworks.zircon.api.component.ColorTheme +import org.hexworks.zircon.api.component.ComponentAlignment +import org.hexworks.zircon.api.grid.TileGrid +import org.hexworks.zircon.api.view.base.BaseView +import kotlin.system.exitProcess + +class StartView (private val grid: TileGrid, theme: ColorTheme = GameConfig.THEME) : BaseView(grid, theme) { + init { + val msg = "Welcome to $GAME_ID." + + // a text box can hold headers, paragraphs and list items + // `contentWidth = ` here is a so-called keyword parameter + // using them you can pass parameters not by their order + // but by their name. + // this might be familiar for Python programmers + val header = Components.textBox(contentWidth = msg.length) + // we add a header + .addHeader(msg) + // and a new line + .addNewLine() + // and align it to center + .withAlignmentWithin(screen, ComponentAlignment.CENTER) + .build() // finally, we build the component + + val startButton = Components.button() + // we align the button to the bottom center of our header + .withAlignmentAround(header, ComponentAlignment.BOTTOM_CENTER) + // its text is "Start!" + .withText("QUICK PLAY!") + // we want a box and some shadow around it + .withDecorations(box(), shadow()) + .build() + + val configButton = Components.button() + .withAlignmentAround(startButton, ComponentAlignment.BOTTOM_CENTER) + .withText("PLAY") + .withDecorations(box(), shadow()) + .build() + + val exitButton = Components.button() + .withAlignmentAround(configButton, ComponentAlignment.BOTTOM_CENTER) + .withText("EXIT") + .withDecorations(box(), shadow()) + .build() + + //TODO: move this on to a configuration screen for world/player customization before PlayView, + // for now basic gameplay is in order though. + + //Once the start button is pressed, move on to the PlayView + startButton.onActivated { + replaceWith(PlayView(grid)) + } + + configButton.onActivated { + replaceWith(ConfigView(grid)) + } + + exitButton.onActivated { + exitProcess(0) + } + + // We can add multiple components at once + //Bake The Cake + screen.addComponents(header, startButton, configButton, exitButton) + } +} \ No newline at end of file diff --git a/bin/main/group/ouroboros/potrogue/view/WinView.kt b/bin/main/group/ouroboros/potrogue/view/WinView.kt new file mode 100644 index 0000000..4be1dc1 --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/view/WinView.kt @@ -0,0 +1,49 @@ +package group.ouroboros.potrogue.view + +import group.ouroboros.potrogue.data.config.GameConfig +import org.hexworks.zircon.api.ComponentDecorations.box +import org.hexworks.zircon.api.Components +import org.hexworks.zircon.api.component.ColorTheme +import org.hexworks.zircon.api.component.ComponentAlignment +import org.hexworks.zircon.api.grid.TileGrid +import org.hexworks.zircon.api.view.base.BaseView +import kotlin.system.exitProcess + +// For if winning… just a test. +class WinView(private val grid: TileGrid, theme: ColorTheme = GameConfig.THEME) : BaseView(grid, theme) { + + init { + // Title + val header = Components.header() + .withText("You won!") + .withAlignmentWithin(screen, ComponentAlignment.CENTER) + .build() + + // Create Reset Button + val restartButton = Components.button() + .withAlignmentAround(header, ComponentAlignment.BOTTOM_LEFT) + .withText("Restart") + .withDecorations(box()) + .build() + + // Create Quit Button + val exitButton = Components.button() + .withAlignmentAround(header, ComponentAlignment.BOTTOM_RIGHT) + .withText("Quit") + .withDecorations(box()) + .build() + + // On Reset Button activated, move back to PlayView + restartButton.onActivated { + replaceWith(PlayView(grid)) + } + + // On Quit Button activated, exit program + exitButton.onActivated { + exitProcess(0) + } + + // Bake The Cake + screen.addComponents(header, restartButton, exitButton) + } +} diff --git a/bin/main/group/ouroboros/potrogue/world/Game.kt b/bin/main/group/ouroboros/potrogue/world/Game.kt new file mode 100644 index 0000000..9db46f6 --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/world/Game.kt @@ -0,0 +1,28 @@ +package group.ouroboros.potrogue.world + +import group.ouroboros.potrogue.entity.attributes.types.Player +import group.ouroboros.potrogue.extensions.GameEntity + +/* + * The TL;DR for DIP is this: By stating what we need (the World here) but not how we get it we let the outside world decide how to provide it for us. + * This is also called “Wishful Thinking.” + * This kind of dependency inversion lets the users of our program inject any kind of object that corresponds to the World contract. + * For example, we can create an in-memory world, one which is stored in a database or one which is generated on the fly. Game won’t care! + * This is in stark contrast to what we had before: an explicit instantiation of the World by using the WorldBuilder. + */ + +class Game ( + val world: World, + val player: GameEntity +) { + companion object { + + fun create( + player: GameEntity, + world: World + ) = Game( + world = world, + player = player + ) + } +} diff --git a/bin/main/group/ouroboros/potrogue/world/GameBuilder.kt b/bin/main/group/ouroboros/potrogue/world/GameBuilder.kt new file mode 100644 index 0000000..c755b5f --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/world/GameBuilder.kt @@ -0,0 +1,83 @@ +package group.ouroboros.potrogue.world + +import group.ouroboros.potrogue.builders.EntityFactory +import group.ouroboros.potrogue.builders.WorldBuilder +import group.ouroboros.potrogue.data.config.Config +import group.ouroboros.potrogue.data.config.GameConfig.WORLD_SIZE +import group.ouroboros.potrogue.entity.attributes.types.Player +import group.ouroboros.potrogue.extensions.GameEntity +import org.hexworks.amethyst.api.entity.EntityType +import org.hexworks.zircon.api.data.Position3D +import org.hexworks.zircon.api.data.Size +import org.hexworks.zircon.api.data.Size3D + +// Take the size of the World as a parameter +class GameBuilder(val worldSize: Size3D) { + + // We define the visible size which is our viewport of the world + private val visibleSize = Size3D.create( + xLength = Config().windowWidth - Config().sidebarWidth, + yLength = Config().windowHeight - Config().logAreaHeight - Config().helpTipHeight, + zLength = 1 + ) + + // We build our World here as part of the Game + val world = WorldBuilder(worldSize) + .makeCaves() + .build(visibleSize = visibleSize) + + fun buildGame(): Game { + prepareWorld() + + val player = addPlayer() + addCreature() + + return Game.create( + player = player, + world = world + ) + } + + // prepareWorld can be called with method chaining here, since also will return the GameBuilder object + private fun prepareWorld() = also { + world.scrollUpBy(world.actualSize.zLength) + } + + + // Add this extension method to any GameEntity and we use the T generic type parameter to preserve the type in the return value to out function + private fun GameEntity.addToWorld( + // atLevel will be used to supply the level at which we want to add the Entity + atLevel: Int, + // atArea specifies the size of the area at which we want to add the Entity this defaults to the actual size of the world (the whole level). + // this function returns the GameEntity which we called this function on which allows us to perform Method Chaining + atArea: Size = world.actualSize.to2DSize()): GameEntity { + world.addAtEmptyPosition(this, + // We call addAtEmptyPosition with the supplied level + offset = Position3D.defaultPosition().withZ(atLevel), + // and we set the size using the supplied Size + size = Size3D.from2DSize(atArea)) + return this + } + + // Create Player using addToWorld Function + private fun addPlayer(): GameEntity { + return EntityFactory.newPlayer().addToWorld( + atLevel = Config().dungeonLevels - 1, + atArea = world.visibleSize.to2DSize()) + } + + private fun addCreature() = also { + repeat(world.actualSize.zLength) { level -> + repeat(Config().creaturesPerLevel) { + EntityFactory.newCreature().addToWorld(level) + } + } + } + + companion object { + + fun create() = GameBuilder( + worldSize = WORLD_SIZE + ).buildGame() + } +} diff --git a/bin/main/group/ouroboros/potrogue/world/GameContext.kt b/bin/main/group/ouroboros/potrogue/world/GameContext.kt new file mode 100644 index 0000000..5dd5af4 --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/world/GameContext.kt @@ -0,0 +1,19 @@ +package group.ouroboros.potrogue.world + +import group.ouroboros.potrogue.entity.attributes.types.Player +import group.ouroboros.potrogue.extensions.GameEntity +import org.hexworks.amethyst.api.Context +import org.hexworks.zircon.api.screen.Screen +import org.hexworks.zircon.api.uievent.UIEvent + +data class GameContext( + // The world itself + val world: World, + // The Screen object which we can use to open dialogs and interact with the UI in general + val screen: Screen, + // The UIEvent which caused the update of the world (a key press for example) + val uiEvent: UIEvent, + // The object representing the player. This is optional, but because we use the player in a lot of places it makes sense to add it here + val player: GameEntity + +) : Context diff --git a/bin/main/group/ouroboros/potrogue/world/World.kt b/bin/main/group/ouroboros/potrogue/world/World.kt new file mode 100644 index 0000000..4fe784d --- /dev/null +++ b/bin/main/group/ouroboros/potrogue/world/World.kt @@ -0,0 +1,162 @@ +package group.ouroboros.potrogue.world + +import group.ouroboros.potrogue.blocks.GameBlock +import group.ouroboros.potrogue.extensions.GameEntity +import group.ouroboros.potrogue.extensions.position +import org.hexworks.amethyst.api.entity.Entity +import org.hexworks.amethyst.api.entity.EntityType +import org.hexworks.amethyst.internal.TurnBasedEngine +import org.hexworks.amethyst.platform.Dispatchers +import org.hexworks.cobalt.datatypes.Maybe +import org.hexworks.zircon.api.builder.game.GameAreaBuilder +import org.hexworks.zircon.api.data.Position3D +import org.hexworks.zircon.api.data.Size3D +import org.hexworks.zircon.api.data.Tile +import org.hexworks.zircon.api.game.GameArea +import org.hexworks.zircon.api.screen.Screen +import org.hexworks.zircon.api.uievent.UIEvent + +class World ( + // A World object is about holding the world data in memory, but it is not about generating it, so we take the initial state of the world as a parameter. + startingBlocks: Map, + visibleSize: Size3D, + actualSize: Size3D + // We implement the GameArea which we’ll use with the GameComponent +) : GameArea by GameAreaBuilder.newBuilder() + // We set its visibleSize. This is the size of the area which will be visible on our screen + .withVisibleSize(visibleSize) + // We set the actualSize. This is the size of the whole world which can be multiple times bigger than the visible part. + // GameArea supports scrolling so we’ll be able to scroll through our caves soon + .withActualSize(actualSize) + .build() { + + // We added the Engine to the world which handles our entities. + // We could have used dependency inversion here, but this is not likely to change in the future so we’re keeping it simple. + private val engine: TurnBasedEngine = TurnBasedEngine(Dispatchers.Single) + + init { + startingBlocks.forEach { (pos, block) -> + // A World takes a Map of GameBlocks, so we need to add them to the GameArea. + // Where these blocks come from? We’ll see soon enough wen we implement the WorldBuilder! + setBlockAt(pos, block) + block.entities.forEach { entity -> + // Also added the Entities in the starting blocks to our engine + engine.addEntity(entity) + // Saved their position + entity.position = pos + } + } +} + /** + * Adds the given [Entity] at the given [Position3D]. + * Has no effect if this world already contains the + * given [Entity]. + */ + // Added a function for adding new entities + fun addEntity(entity: Entity, position: Position3D) { + entity.position = position + engine.addEntity(entity) + fetchBlockAt(position).map { + it.addEntity(entity) + } + } + + fun removeEntity(entity: Entity) { + fetchBlockAt(entity.position).map { + it.removeEntity(entity) + } + engine.removeEntity(entity) + entity.position = Position3D.unknown() + } + + // Added a function for adding an Entity at an empty position. + // This function needs a little explanation though. + // What happens here is that we try to find and empty position in our World within the given bounds (offset and size). + // Using this function we can limit the search for empty positions to a single level or multiple levels, and also within a given level. + // This will be very useful later. + fun addAtEmptyPosition( + entity: GameEntity, + offset: Position3D = Position3D.create(0, 0, 0), + size: Size3D = actualSize + ): Boolean { + return findEmptyLocationWithin(offset, size).fold( + // If we didn’t find an empty position, then we return with false indicating that we were not successful + whenEmpty = { + false + }, + // Otherwise we add the Entity at the position which was found. + whenPresent = { location -> + addEntity(entity, location) + true + }) + + } + + /** + * Finds an empty location within the given area (offset and size) on this [World]. + */ + // This function performs a random serach for an empty position. + // To prevent seraching endlessly in a World which has none, we limit the maximum number of tries to 10. + fun findEmptyLocationWithin(offset: Position3D, size: Size3D): Maybe { + var position = Maybe.empty() + val maxTries = 10 + var currentTry = 0 + while (position.isPresent.not() && currentTry < maxTries) { + val pos = Position3D.create( + x = (Math.random() * size.xLength).toInt() + offset.x, + y = (Math.random() * size.yLength).toInt() + offset.y, + z = (Math.random() * size.zLength).toInt() + offset.z + ) + fetchBlockAt(pos).map { + if (it.isEmptyFloor) { + position = Maybe.of(pos) + } + } + currentTry++ + } + return position + } + + // Create the function update which takes all the necessary objects as parameters + fun update(screen: Screen, uiEvent: UIEvent, game: Game) { + // We use the context object which we created before to update the engine. + // If you were wondering before why this class will be necessary now you know: + // a Context object holds all the information which might be necessary to update the entity objects within our world + engine.executeTurn(GameContext( + world = this, + // We pass the screen because we’ll be using it to display dialogs and similar things + screen = screen, + // We’ll inspect the UIEvent to determine what the user wants to do (like moving around). + // We’re using UIEvent instead of KeyboardEvent here because it is possible that at some time we also want to use mouse events. + uiEvent = uiEvent, + // Adding the player entity to the context is not mandatory, + // but since we use it almost everywhere this little optimization will make our life easier. + player = game.player)) + } + + // We pass the entity we want to move and the position where we want to move it. + fun moveEntity(entity: GameEntity, position: Position3D): Boolean { + // We create a success variable which holds a Boolean value representing whether the operation was successful + var success = false + // We fetch both blocks + val oldBlock = fetchBlockAt(entity.position) + val newBlock = fetchBlockAt(position) + + // We only proceed if both blocks are present + if (bothBlocksPresent(oldBlock, newBlock)) { + // In that case success is true + success = true + oldBlock.get().removeEntity(entity) + entity.position = position + newBlock.get().addEntity(entity) + } + //Then we return success + return success + } + + // This is an example of giving a name to a logical operation. + // In this case it is very simple but sometimes logical operations become very complex and it makes sense to give them a name like this (“both blocks present?”) + // so they are easy to reason about. + private fun bothBlocksPresent(oldBlock: Maybe, newBlock: Maybe) = + oldBlock.isPresent && newBlock.isPresent +} \ No newline at end of file diff --git a/bin/main/logback.xml b/bin/main/logback.xml new file mode 100644 index 0000000..8e16f8f --- /dev/null +++ b/bin/main/logback.xml @@ -0,0 +1,22 @@ + + + + + + %d{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n + + + + + + + + + + + + + + + diff --git a/build.gradle.kts b/build.gradle.kts index a05c3e0..71f23e0 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -7,7 +7,6 @@ val junit_version: String by project val mockito_version: String by project val assertj_version: String by project val game_name: String by project -val version: String by project plugins { kotlin("jvm") version "1.9.20" @@ -46,6 +45,8 @@ dependencies { implementation("dev.dirs:directories:26") } + + tasks { named("shadowJar") { mergeServiceFiles() @@ -67,4 +68,3 @@ val jar by tasks.getting(Jar::class) { } } - diff --git a/src/main/resources/assets/tilesets/potrogue_grunge_16x16.png b/src/main/resources/assets/tilesets/potrogue_grunge_16x16.png index 3579c62179a6a9c568fa6a632adaa53d576ae133..14e889fdb70f8ff65e3ff898d0dbc7560f3a1680 100644 GIT binary patch delta 2366 zcmV-E3BmTU6N(g&QGcXKL_t(|ob6qScH=k>6#xIHceX8(K;a@S#ZK(1*N(*tM-c#- z?s=o*crzMccl<5*XU9#(+(bZb%r1BT&3*hq8~nCs0JLNMo`YS_Sa}YCl-ckcY&hII z{(a3Kxk+C8cA!ZU358-KyLAuy)(PllLJ<|PPccDWU1kfby2?)?T<`AHB zydq%c@GS(`4M*#I0tOK^0oq}Y04Lv~6zCZwY`>NQUc{p}83D)OIDkWdBUP;sZdik) z1CyZ(MbJjP*^rcgq6i*iw;)Kv2EGSu)a0AQjpwtp+`t{=fMIQaDSMfo69Iu;`V zMG9yQRU1~zKzGJ-4tJL8@`klk1O{CV6&0P)*U?NwScoOW{R-2oe0*aJmPclkv! zvk`_(bbnj*rmpQ#P+H?D468m6U=y^eu^wH1<#v1aYk^j09)g24F?jwI-WmeplC1=K z70&ZtAIpL4=~-xh6MupkuK(a)M!U~GvFk(EuF_VnxCab9B{lidOt z0bY~$0!)98^I`nSBxrl=6lkDG!w7BW!EAtCZD-(ELV!MEf&d6p zAR0%XjiWobw%kbpGeeW3i=cJT2n+H$943e#M8oseI|Z_v|6c?i5~}c60?Ml76xnR} z{aKz6;8%6E0%fdJb<{6bcUkc<_7uvSolsU10LFhm?AE7V95J3d`XD_t!#5DnFbIu) zG&V^Avwi0>Wv#gyNCmkGK7on#E7-_cCY2t^z_}s- z;J^5AYM>4Hc*`pS(BXi9b%MGN@o;_3(*zg>tc#ivBQH=RR#yb*l(UqtaJ~xnDy;P3 zzfFIoCFhnD(C$JC!1}BH`vlmUV2U7ay!- zmKVoh;lGWbWNvJdmZoH)g7e!hSyrb*_LLds^ezGQy}gR4`E!U|I90N!s(W(rmHOSR zU$`advVQemLJ-6Yv0oG3jrh{h+r=pWO>=*4nDyqGXGGv)dsPFCu>sya$DDJOJ;lvM zHBm%GNMT?_z}0q1z|uXy=~a)JKA4Uj4gsj&HA+1uZAW{}M@pbE0A(%^fOl(jz*pt= zEfA2x570Ei(ExLovo4sPSfQ^!Eh66hRRUJ#dIrcwL4Zpjj_>5O32=7{bB{9XoaukL zeh_I0$SrU?Hgehx`P`{A!WUA&iRb(jaBWE2_Y#n&q}|?pf-f&QcfH}HsGvi;1laj` z33=ieEaKNCBq3;6EtkhduGCj;h}(`CSSb>}9Ywy?6{C>%b@+M$2!fcf7f<&h^|586 z-<#rJM*cm0E(51yP5uWBqTgwM7}_)A9}6o2J~{&0(SIjCdoTq)yatef(?3A~{Bm=A zcnb7~CVRf92;xEbWkt|l1S|m`T?3P$1wDU)Ahz-K|I5~|C*bSx^$|bAk5AX zZIgh7`ALZp?mxB{azMv_odWjdeY*s-O#&9?7d2Mo|Dp1~@4fyA9nuP+Yli^$U5B7T zuE)1REzghLZ_0g{|A#JtNeX}ht=V6O;vR^1{sALAmZ zdxIDh^1J|@!l4Fa_Ws%jKr(;NfD|D`k zJ=|RH+zoyK0f(@vX}Fyz&mVtZS$__qy@iA6xI?g$0(hj1NwJX`0U-m`dTX&BTk_D1 z8b)P9i%1K`0xlhhQt1jdr@QxtNq}F!H~qhdfK`|FX0H;E7o^(sXB^(s(3%P9J5vBB z@0zza{|Oj>izxNx$3;-a(;QzbYP<8Jv6_LrUC%cT+Nui! zK1J5&KZ836fWZXjZ2V^lAd43t0$9;1<0L>qB`} z5v%&W^(5dH0yY@%k3naOyY|--u+D&4Z*6`b0hI#R0XXMnPqjYK_eugL5C8vOBcS1) zoB7H5sV{UWQ2ZkV;P-z59QC+1k);Vwr=+@XB4AaZ8n^RtrNakG*>fQB;5`J`7|025 z%}({L`X6GIm>PXuz8=R^+dA$oDXjomKSY2Pv}Q0UK$qom*}GJ94-rvsAYk#HkP@Jm zFC2qbdSKt}1c=$FeDAWSQa~?XxCmK!YU)oD;Mu=T09v0AP}gFER}*kp`*#U2jf{X& zHVJq&0R%x11VIo4K@bE%5ClOGFT}lnefjtPLe$!O$@)J>fHnA<`G22)i}sU%KTp87 krNGx0K^}tyLA(n81F#1k_X8#Bc>n+a07*qoM6N<$f_p_?k^lez delta 2278 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 RnH&HB002ovPDHLkV1h<$U5x+$