From e18890ab0033583e67d0a81d185e93d4fbcb01f6 Mon Sep 17 00:00:00 2001 From: Luis Ferro Date: Tue, 12 Nov 2019 11:27:47 +0100 Subject: Add noecho to supress keys going to screen --- src/main.rs | 3 +++ 1 file changed, 3 insertions(+) diff --git a/src/main.rs b/src/main.rs index 4426a93..df82f42 100644 --- a/src/main.rs +++ b/src/main.rs @@ -87,6 +87,9 @@ fn main() { render_level(&window, &dungeon.levels[0]); + window.keypad(true); + noecho(); + loop { // update actors // update character -- cgit v1.2.3 From 1c2ffcff2371182c83ff411c27ce742a4b9432a8 Mon Sep 17 00:00:00 2001 From: Luis Ferro Date: Tue, 12 Nov 2019 11:32:24 +0100 Subject: The actual fix. --- src/main.rs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/main.rs b/src/main.rs index df82f42..49e8a8c 100644 --- a/src/main.rs +++ b/src/main.rs @@ -98,12 +98,12 @@ fn main() { // get input and execute it match window.getch() { - Some(Input::Character('h')) => { window.addstr("DEL: quit\n"); }, + Some(Input::Character('h')) => { window.addstr("q: quit\n"); }, // Some(Input::KeyDown) => { window.addstr("down\n"); }, // Some(Input::KeyUp) => { window.addch('b'); }, // Some(Input::KeyLeft) => { window.addch('c'); }, // Some(Input::KeyRight) => { window.addch('d'); }, - Some(Input::KeyDC) => break, + Some(Input::Character('q')) => break, Some(_) => (), None => (), } -- cgit v1.2.3 From 5fd6433d88be76844a643738b5d3befc137ca6e5 Mon Sep 17 00:00:00 2001 From: Guillaume Pasquet Date: Tue, 12 Nov 2019 11:47:29 +0100 Subject: Better aligned room generation --- .vscode/launch.json | 2 +- src/world.rs | 60 ++++++++++++++++++++++++++++++++++++++++++++++------- 2 files changed, 54 insertions(+), 8 deletions(-) diff --git a/.vscode/launch.json b/.vscode/launch.json index dfa5895..e2e657d 100644 --- a/.vscode/launch.json +++ b/.vscode/launch.json @@ -16,7 +16,7 @@ "externalConsole": true }, { - "name": "(OSX) Launch", + "name": "(Unix) Launch", "type": "lldb", "request": "launch", "program": "${workspaceRoot}/target/debug/roguelike", diff --git a/src/world.rs b/src/world.rs index 24a4c31..2fa72ae 100644 --- a/src/world.rs +++ b/src/world.rs @@ -186,7 +186,9 @@ pub struct Level { xsize: usize, ysize: usize, rooms: Vec, - corridors: Vec + corridors: Vec, + entrance: Point, + exit: Point } pub struct Dungeon { @@ -230,9 +232,15 @@ impl Dungeon { impl Generable for Dungeon { fn generate(&mut self) { - for _ in 0..self.depth { - let mut level = Level::new(self.xsize, self.ysize, None); + let mut level = Level::new(self.xsize, self.ysize, None); + level.generate(); + let mut next_entrance = level.get_exit(); + self.levels.push(level); + + for _ in 1..self.depth { + level = Level::new(self.xsize, self.ysize, Some(next_entrance)); level.generate(); + next_entrance = level.get_exit(); self.levels.push(level); } } @@ -247,7 +255,12 @@ impl Level { xsize, ysize, rooms: Vec::new(), - corridors: Vec::new() + corridors: Vec::new(), + entrance: match start { + Some(st) => st, + None => (0, 0) + }, + exit: (0, 0) } } @@ -262,6 +275,9 @@ impl Level { corridor.tile(&mut grid)?; } + grid.set_tile(self.entrance.0, self.entrance.1, TileType::StairsUp); + grid.set_tile(self.exit.0, self.exit.1, TileType::StairsDown); + Ok(grid) } @@ -272,6 +288,14 @@ impl Level { return (0,0) } + pub fn get_entrance(&self) -> Point { + self.entrance + } + + pub fn get_exit(&self) -> Point { + self.exit + } + fn overlaps(&self, start: Point, width: usize, height: usize, padding: usize) -> bool { for room in &self.rooms { if room.start.0 < start.0 + width + padding && @@ -300,8 +324,8 @@ impl Level { fn random_room(&self) -> Result { // TODO: Detect when not enough space is left to allocate a room. let mut rng = rand::thread_rng(); - let room_width = rng.gen_range(3, 12); - let room_height = rng.gen_range(3, 12); + let room_width = rng.gen_range(4, 12); + let room_height = rng.gen_range(4, 12); // TODO: Find a way to write a lambda to generate the start point. let mut start: Point = ( @@ -318,6 +342,19 @@ impl Level { Ok(Room::new(start, room_width, room_height)) } + + fn centered_room(&self, center: Point) -> Room { + let mut rng = rand::thread_rng(); + let room_width = rng.gen_range(3, 12); + let room_height = rng.gen_range(3, 12); + + let start = ( + (center.0 as f32 - (room_width as f32 / 2f32)).round() as usize, + (center.1 as f32 - (room_height as f32 / 2f32)).round() as usize + ); + + Room::new(start, room_width, room_height) + } } impl Generable for Level { @@ -325,8 +362,12 @@ impl Generable for Level { let mut rng = rand::thread_rng(); let room_number = rng.gen_range(3, 5); + if self.entrance != (0, 0) { + self.rooms.push(self.centered_room(self.entrance)); + } + // Generate rooms - for _ in 0..room_number { + for _ in self.rooms.len()..room_number { self.rooms.push(self.random_room().unwrap()); } @@ -365,6 +406,11 @@ impl Generable for Level { CorridorType::Vertical )); } + + if self.entrance == (0, 0) { + self.entrance = self.rooms[0].center; + } + self.exit = self.rooms.last().unwrap().center; } } -- cgit v1.2.3 From ec671aa9b56c53d76ce310f0772ee05c97064d3f Mon Sep 17 00:00:00 2001 From: Guillaume Pasquet Date: Tue, 12 Nov 2019 11:59:03 +0100 Subject: Dynamic terminal size. --- src/main.rs | 21 +++++++-------------- 1 file changed, 7 insertions(+), 14 deletions(-) diff --git a/src/main.rs b/src/main.rs index 49e8a8c..c4c6f4f 100644 --- a/src/main.rs +++ b/src/main.rs @@ -55,8 +55,14 @@ fn debug_level(level: Level) { fn main() { + let window = initscr(); let mut level = 0; - let mut dungeon = Dungeon::new(80, 24, 5); + + let mut dungeon = Dungeon::new( + window.get_max_x() as usize, + window.get_max_y() as usize - 2, + 5 + ); dungeon.generate(); let start_location = dungeon.levels[0].get_start_point(); @@ -72,19 +78,6 @@ fn main() { ); character.place(start_location); - // Dump the whole dungeon structure in terminal for debugging - match env::var("DEBUG") { - Ok(_) => { - for l in dungeon.levels { - debug_level(l); - } - return - }, - Err(_) => () - }; - - let window = initscr(); - render_level(&window, &dungeon.levels[0]); window.keypad(true); -- cgit v1.2.3 From 8fa3fa881bc3b954e136295fe6cc7022737ae9db Mon Sep 17 00:00:00 2001 From: Guillaume Pasquet Date: Tue, 12 Nov 2019 14:22:19 +0100 Subject: Refactor all the things! --- src/character.rs | 146 ------------------------------------------------ src/computer.rs | 36 ------------ src/entities.rs | 167 +++++++++++++++++++++++++++++++++++++++++++++++++++++++ src/main.rs | 24 ++------ src/tiling.rs | 52 +++++++++++++++++ src/world.rs | 73 ++++++------------------ 6 files changed, 240 insertions(+), 258 deletions(-) delete mode 100644 src/character.rs delete mode 100644 src/computer.rs create mode 100644 src/entities.rs create mode 100644 src/tiling.rs diff --git a/src/character.rs b/src/character.rs deleted file mode 100644 index 3e6c4a7..0000000 --- a/src/character.rs +++ /dev/null @@ -1,146 +0,0 @@ -use std::cmp; - -use crate::world::Point; - -pub struct Character { - pub name: String, - pub class: String, - pub health: i32, - max_health: i32, - attack: i32, - dodge: i32, - luck: i32, - xp: i32, - location: Point -} - -pub trait Player { - fn new( - name: String, - class: String, - health: i32, - attack: i32, - dodge: i32, - luck: i32, - location: Point - ) -> Character; - - fn place(&mut self, location: Point); - - fn select(&self, player_name: String, player_luck: i32) -> Self; - - fn damage(&mut self, damage_amount: i32); - - fn heal(&mut self, heal_amount: i32); - - fn attack(&self) -> i32; - - fn dodge(&self) -> i32; - - fn info(&self) -> String; - - fn stats(&self) -> String; - - fn walk(&mut self, dir: Direction); -} - -pub enum Direction { - North, - South, - East, - West -} - -impl Player for Character { - fn new( - name: String, - class: String, - health: i32, - attack: i32, - dodge: i32, - luck: i32, - location: Point - ) -> Character { - Character { - name: name, - class: class, - max_health: health, - health: health, - attack: attack, - dodge: dodge, - luck: luck, - xp: 0, - location: location - } - } - - fn place(&mut self, location: Point) { - self.location = location; - } - - fn select(&self, player_name: String, player_luck: i32) -> Self { - Self::new( - player_name, - self.class.to_string(), - self.health, - self.attack, - self.dodge, - self.luck + player_luck, - (0,0) - ) - } - - fn walk(&mut self, dir: Direction) { - match dir { - Direction::North => { (); }, - Direction::South => { (); }, - Direction::East => { (); }, - Direction::West => { (); } - } - } - - fn damage(&mut self, damage_amount: i32) { - self.health = cmp::max(0, self.health - damage_amount); - self.xp += 2; - } - - fn heal(&mut self, heal_amount: i32) { - if (self.health) <= self.max_health { - self.health = cmp::min(self.health + heal_amount, self.max_health); - self.xp += 1; - } - } - - fn attack(&self) -> i32 { - self.xp + self.attack + self.luck / 2 - } - - fn dodge(&self) -> i32 { - self.xp + self.dodge + self.luck / 2 - } - - fn info(&self) -> String { - format!( - "{} \thp: {} attack: {} dodge: {} luck: {}", - self.class, self.health, self.attack, self.dodge, self.luck - ) - } - - fn stats(&self) -> String { - format!( - "{} - hp: {} attack: {} dodge: {} luck: {} experience: {}", - self.class, self.health, self.attack, self.dodge, self.luck, self.xp - ) - } -} - -#[cfg(test)] -mod tests { - use super::*; - - fn test_attack() { - let player = Character::new("".to_string(), "Rogue".to_string(), 1, 4, 1, 4, (0,0)); - - assert_eq!(player.attack(), 6); - } -} diff --git a/src/computer.rs b/src/computer.rs deleted file mode 100644 index a8b9b3e..0000000 --- a/src/computer.rs +++ /dev/null @@ -1,36 +0,0 @@ -pub struct Computer { - level: i32, - difficulty: i32, -} - -pub trait Enemy { - fn new(level: i32, difficulty: i32) -> Self; - - fn action(&self) -> (i32, i32); - - fn level_up(&mut self); - - fn stats(&self) -> String; -} - -impl Enemy for Computer { - fn new(level: i32, difficulty: i32) -> Computer { - Computer { - level: level, - difficulty: difficulty - } - } - - fn action(&self) -> (i32, i32) { - (self.level, self.difficulty) - } - - fn level_up(&mut self) { - self.level += 1; - self.difficulty += 3; - } - - fn stats(&self) -> String { - format!("level: {} difficulty: {}", self.level, self.difficulty) - } -} diff --git a/src/entities.rs b/src/entities.rs new file mode 100644 index 0000000..51cefc1 --- /dev/null +++ b/src/entities.rs @@ -0,0 +1,167 @@ +use std::cmp; + +use crate::world::{Point, Direction}; +use crate::tiling::TileType; + +pub trait Entity { + fn info(&self) -> String; + fn place(&mut self, location: Point); +} + +#[derive(Clone)] +pub struct Character { + pub name: String, + pub class: String, + pub health: i32, + max_health: i32, + attack: i32, + dodge: i32, + luck: i32, + xp: i32, + location: Point, + tile_type: TileType +} + +pub trait Enemy { + fn new( + class: String, + health: i32, + attack: i32, + dodge: i32, + luck: i32, + location: Point + ) -> Self; + + fn set_tile_type(&mut self, tile_type: TileType); +} + +pub trait Player { + fn new( + name: String, + class: String, + health: i32, + attack: i32, + dodge: i32, + luck: i32, + location: Point + ) -> Self; + fn damage(&mut self, damage_amount: i32); + fn heal(&mut self, heal_amount: i32); + fn attack(&self) -> i32; + fn dodge(&self) -> i32; + fn stats(&self) -> String; + fn walk(&mut self, dir: Direction); +} + +impl Entity for Character { + fn place(&mut self, location: Point) { + self.location = location; + } + + fn info(&self) -> String { + format!( + "{} \thp: {} attack: {} dodge: {} luck: {}", + self.class, self.health, self.attack, self.dodge, self.luck + ) + } +} + +impl Enemy for Character { + fn new( + class: String, + health: i32, + attack: i32, + dodge: i32, + luck: i32, + location: Point + ) -> Character { + Character { + name: class.clone(), + class: class.clone(), + max_health: health, + health, + attack, + dodge, + luck, + xp: 0, + location: location, + tile_type: TileType::Character + } + } + + fn set_tile_type(&mut self, tile_type: TileType) { + self.tile_type = tile_type + } +} + +impl Player for Character { + fn new( + name: String, + class: String, + health: i32, + attack: i32, + dodge: i32, + luck: i32, + location: Point + ) -> Character { + Character { + name: name, + class: class, + max_health: health, + health: health, + attack: attack, + dodge: dodge, + luck: luck, + xp: 0, + location: location, + tile_type: TileType::Player + } + } + + fn walk(&mut self, dir: Direction) { + match dir { + Direction::North => { (); }, + Direction::South => { (); }, + Direction::East => { (); }, + Direction::West => { (); } + } + } + + fn damage(&mut self, damage_amount: i32) { + self.health = cmp::max(0, self.health - damage_amount); + self.xp += 2; + } + + fn heal(&mut self, heal_amount: i32) { + if (self.health) <= self.max_health { + self.health = cmp::min(self.health + heal_amount, self.max_health); + self.xp += 1; + } + } + + fn attack(&self) -> i32 { + self.xp + self.attack + self.luck / 2 + } + + fn dodge(&self) -> i32 { + self.xp + self.dodge + self.luck / 2 + } + + fn stats(&self) -> String { + format!( + "{} - hp: {} attack: {} dodge: {} luck: {} experience: {}", + self.class, self.health, self.attack, self.dodge, self.luck, self.xp + ) + } +} + +#[cfg(test)] +mod tests { + use super::*; + + fn test_attack() { + let player: Character = Player::new("".to_string(), "Rogue".to_string(), 1, 4, 1, 4, (0,0)); + + assert_eq!(player.attack(), 6); + } +} diff --git a/src/main.rs b/src/main.rs index c4c6f4f..f7e7f0f 100644 --- a/src/main.rs +++ b/src/main.rs @@ -4,16 +4,14 @@ extern crate pancurses; #[macro_use] extern crate text_io; -mod character; -mod computer; +mod entities; mod world; +mod tiling; -use std::env; -use character::Player; -use character::Character; -use computer::Enemy; +use entities::{Character, Player, Entity}; use pancurses::{Window, initscr, endwin, Input, noecho}; -use world::{Dungeon, Level, Generable, TileType}; +use world::{Dungeon, Level, Generatable}; +use tiling::TileType; fn tile_to_str(tile: &TileType) -> &str { match tile { @@ -42,18 +40,6 @@ fn render_level(window: &Window, level: &Level) { } } -fn debug_level(level: Level) { - let grid = level.to_tilegrid().unwrap(); - - for line in grid.raw_data().iter() { - for block in line.iter() { - print!("{}", tile_to_str(block)); - } - print!("\n"); - } -} - - fn main() { let window = initscr(); let mut level = 0; diff --git a/src/tiling.rs b/src/tiling.rs new file mode 100644 index 0000000..139af24 --- /dev/null +++ b/src/tiling.rs @@ -0,0 +1,52 @@ +pub struct TileGrid { + grid: Vec> +} + +impl<'a> TileGrid { + pub fn new(xsize: usize, ysize: usize) -> TileGrid { + let mut grid = TileGrid { + grid: Vec::with_capacity(ysize) + }; + + for _ in 0..ysize { + let mut subvec = Vec::with_capacity(xsize); + for _ in 0..xsize { + subvec.push(TileType::Empty); + } + grid.grid.push(subvec); + } + + return grid; + } + + pub fn set_tile(&mut self, x: usize, y: usize, tile: TileType) { + self.grid[y][x] = tile; + } + + /// Sets a tile if nothing lies underneath it. + pub fn set_empty_tile(&mut self, x: usize, y: usize, tile: TileType) { + self.set_tile(x, y, match self.grid[y][x] { + TileType::Empty => tile, + _ => self.grid[y][x].clone() + }) + } + + pub fn raw_data(&'a self) -> &'a Vec> { + &self.grid + } +} + +pub trait Tileable { + fn tile(&self, grid: &mut TileGrid) -> Result<(), String>; +} + +#[derive(Clone)] +pub enum TileType { + Empty, + Wall, + Floor, + StairsUp, + StairsDown, + Character, + Player +} diff --git a/src/world.rs b/src/world.rs index 2fa72ae..abaffa1 100644 --- a/src/world.rs +++ b/src/world.rs @@ -1,26 +1,21 @@ use rand::Rng; +use crate::entities::{Entity}; +use crate::tiling::{TileGrid, Tileable, TileType}; + +pub enum Direction { + North, + South, + East, + West +} pub type Point = (usize, usize); -#[derive(Clone)] -pub enum TileType { - Empty, - Wall, - Floor, - StairsUp, - StairsDown, - Character, -} - enum CorridorType { Horizontal, Vertical } -trait Tileable { - fn tile(&self, grid: &mut TileGrid) -> Result<(), String>; -} - const LEFT: (i8, i8) = (-1i8, 0); const RIGHT: (i8, i8) = (1i8, 0); const UP: (i8, i8) = (0, -1i8); @@ -144,49 +139,12 @@ impl Tileable for Corridor { } } -pub struct TileGrid { - grid: Vec> -} - -impl<'a> TileGrid { - pub fn new(xsize: usize, ysize: usize) -> TileGrid { - let mut grid = TileGrid { - grid: Vec::with_capacity(ysize) - }; - - for _ in 0..ysize { - let mut subvec = Vec::with_capacity(xsize); - for _ in 0..xsize { - subvec.push(TileType::Empty); - } - grid.grid.push(subvec); - } - - return grid; - } - - fn set_tile(&mut self, x: usize, y: usize, tile: TileType) { - self.grid[y][x] = tile; - } - - /// Sets a tile if nothing lies underneath it. - fn set_empty_tile(&mut self, x: usize, y: usize, tile: TileType) { - self.set_tile(x, y, match self.grid[y][x] { - TileType::Empty => tile, - _ => self.grid[y][x].clone() - }) - } - - pub fn raw_data(&'a self) -> &'a Vec> { - &self.grid - } -} - pub struct Level { xsize: usize, ysize: usize, rooms: Vec, corridors: Vec, + entities: Vec>, entrance: Point, exit: Point } @@ -198,7 +156,7 @@ pub struct Dungeon { pub levels: Vec } -pub trait Generable { +pub trait Generatable { fn generate(&mut self); } @@ -230,7 +188,7 @@ impl Dungeon { } } -impl Generable for Dungeon { +impl Generatable for Dungeon { fn generate(&mut self) { let mut level = Level::new(self.xsize, self.ysize, None); level.generate(); @@ -254,8 +212,9 @@ impl Level { Level { xsize, ysize, - rooms: Vec::new(), - corridors: Vec::new(), + rooms: vec![], + corridors: vec![], + entities: vec![], entrance: match start { Some(st) => st, None => (0, 0) @@ -357,7 +316,7 @@ impl Level { } } -impl Generable for Level { +impl Generatable for Level { fn generate(&mut self) { let mut rng = rand::thread_rng(); let room_number = rng.gen_range(3, 5); -- cgit v1.2.3