work on level generator
This commit is contained in:
parent
7f288dbcd1
commit
c6492c28c2
@ -73,7 +73,8 @@ pub fn get_room_type_per_level() -> Vec<HashMap<RoomType, std::ops::RangeInclusi
|
||||
let tmp = [
|
||||
// level 1
|
||||
vec![
|
||||
(RoomType::EmptyRoom, 75),
|
||||
(RoomType::EmptyRoom, 50),
|
||||
(RoomType::ArtifactRoom, 25),
|
||||
(RoomType::MonsterRoom, 5),
|
||||
(RoomType::BasicRoom, 20),
|
||||
],
|
||||
|
@ -57,6 +57,7 @@ impl Level {
|
||||
return (None, None, None);
|
||||
}
|
||||
if !self.discovered[x][y] {
|
||||
#[cfg(test)]
|
||||
return (Some(StructureElement::Unknown), None, None);
|
||||
}
|
||||
let search_pos = &Position::new(self.level, x, y);
|
||||
|
@ -2,15 +2,19 @@ use petgraph::algo::min_spanning_tree;
|
||||
use petgraph::data::*;
|
||||
use petgraph::graph::Graph;
|
||||
use petgraph::graph::UnGraph;
|
||||
use std::cmp::max;
|
||||
use std::cmp::min;
|
||||
use std::ops::Range;
|
||||
|
||||
use rand::Rng;
|
||||
use rand::{rngs::ThreadRng, seq::SliceRandom};
|
||||
|
||||
use crate::artifacts::Artifact;
|
||||
use crate::artifacts::Chest;
|
||||
use crate::artifacts::Potion;
|
||||
use crate::constants::get_monsters_per_level;
|
||||
use crate::constants::ROOM_HEIGHT;
|
||||
use crate::constants::ROOM_WIDTH;
|
||||
use crate::monster::create_monster_by_type;
|
||||
use crate::monster::Monster;
|
||||
use crate::position::Position;
|
||||
use crate::room::Connection;
|
||||
use crate::{
|
||||
constants::{
|
||||
@ -23,6 +27,7 @@ use crate::{
|
||||
pub struct LevelGenerator {
|
||||
level: usize,
|
||||
rooms: [[Room; ROOMS_VERTICAL]; ROOMS_HORIZONAL],
|
||||
rng: ThreadRng,
|
||||
}
|
||||
enum Direction {
|
||||
Horizontal,
|
||||
@ -103,7 +108,7 @@ impl LevelGenerator {
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
// debug print a text view of the dungeon
|
||||
println!(" 0 1 2 3 4 5 6 7");
|
||||
for r in 0..ROOMS_VERTICAL {
|
||||
print!("{} ", r);
|
||||
@ -174,20 +179,20 @@ impl LevelGenerator {
|
||||
|
||||
if src_node_col == tgt_node_col {
|
||||
// println!("Down");
|
||||
let start_col = src_node_col * ROOM_WIDTH+ROOM_WIDTH/2;
|
||||
let start_row = src_node_row * ROOM_HEIGHT+ROOM_HEIGHT;
|
||||
let end_col = tgt_node_col * ROOM_WIDTH+ROOM_WIDTH/2;
|
||||
let end_row = tgt_node_row * ROOM_HEIGHT;
|
||||
let start_col = src_node_col * ROOM_WIDTH + ROOM_WIDTH / 2;
|
||||
let start_row = src_node_row * ROOM_HEIGHT + ROOM_HEIGHT;
|
||||
let end_col = tgt_node_col * ROOM_WIDTH + ROOM_WIDTH / 2;
|
||||
let end_row = tgt_node_row * ROOM_HEIGHT;
|
||||
rooms[src_node_col][src_node_row].connection_down = Some(Connection {
|
||||
start_pos: (start_col, start_row),
|
||||
end_pos: (end_col, end_row),
|
||||
});
|
||||
} else {
|
||||
// println!("Right");
|
||||
let start_col = src_node_col * ROOM_WIDTH+ROOM_WIDTH;
|
||||
let start_row = src_node_row * ROOM_HEIGHT+ROOM_HEIGHT/2;
|
||||
let start_col = src_node_col * ROOM_WIDTH + ROOM_WIDTH;
|
||||
let start_row = src_node_row * ROOM_HEIGHT + ROOM_HEIGHT / 2;
|
||||
let end_col = tgt_node_col * ROOM_WIDTH;
|
||||
let end_row = tgt_node_row * ROOM_HEIGHT+ROOM_HEIGHT/2;
|
||||
let end_row = tgt_node_row * ROOM_HEIGHT + ROOM_HEIGHT / 2;
|
||||
rooms[src_node_col][src_node_row].connection_right = Some(Connection {
|
||||
start_pos: (start_col, start_row),
|
||||
end_pos: (end_col, end_row),
|
||||
@ -195,7 +200,26 @@ impl LevelGenerator {
|
||||
}
|
||||
}
|
||||
|
||||
LevelGenerator { level, rooms }
|
||||
LevelGenerator {
|
||||
level,
|
||||
rooms,
|
||||
rng,
|
||||
}
|
||||
}
|
||||
|
||||
fn select_monster(position: Position, rng: &mut ThreadRng) -> Box<dyn Monster> {
|
||||
let level = position.get_level();
|
||||
let value = rng.gen_range(1..=100);
|
||||
|
||||
let t = get_monsters_per_level();
|
||||
if level < t.len() {
|
||||
for (mtype, range) in &t[level] {
|
||||
if range.contains(&value) {
|
||||
return create_monster_by_type(mtype, position);
|
||||
}
|
||||
}
|
||||
}
|
||||
panic!("no monster selectable!");
|
||||
}
|
||||
|
||||
fn select_room_type(level: usize, rng: &mut ThreadRng) -> RoomType {
|
||||
@ -216,18 +240,37 @@ impl LevelGenerator {
|
||||
let mut structure = [[StructureElement::Wall; LEVEL_HEIGHT]; LEVEL_WIDTH];
|
||||
let mut start_pos = (0, 0);
|
||||
let mut end_pos = (0, 0);
|
||||
let mut monsters: Vec<Box<dyn Monster>> = Vec::with_capacity(10);
|
||||
let mut artifacts: Vec<Box<dyn Artifact>> = Vec::with_capacity(10);
|
||||
for col in 0..ROOMS_HORIZONAL {
|
||||
for row in 0..ROOMS_VERTICAL {
|
||||
let position = self.rooms[col][row].render(&mut structure, col, row);
|
||||
if self.rooms[col][row].kind == RoomType::Start
|
||||
|| self.rooms[col][row].kind == RoomType::StairUp
|
||||
{
|
||||
start_pos = position;
|
||||
}
|
||||
if self.rooms[col][row].kind == RoomType::End
|
||||
|| self.rooms[col][row].kind == RoomType::StairDown
|
||||
{
|
||||
end_pos = position;
|
||||
let room = self.rooms[col][row];
|
||||
let position = room.render(&mut structure, col, row);
|
||||
match room.kind {
|
||||
RoomType::Start => {start_pos=position},
|
||||
RoomType::End => {end_pos=position},
|
||||
RoomType::StairUp => {start_pos=position},
|
||||
RoomType::StairDown => {end_pos=position},
|
||||
RoomType::BasicRoom => {},
|
||||
RoomType::ArtifactRoom => {
|
||||
match self.rng.gen_range(1..=100) {
|
||||
1..=50 => {
|
||||
artifacts
|
||||
.push(Box::new(Chest::new(Position::new(self.level, position.0, position.1))));
|
||||
}
|
||||
_ => {
|
||||
artifacts
|
||||
.push(Box::new(Potion::new(Position::new(self.level, position.0, position.1))));
|
||||
}
|
||||
};
|
||||
},
|
||||
RoomType::MonsterRoom => {
|
||||
monsters.push(LevelGenerator::select_monster(
|
||||
Position::new(self.level, position.0, position.1),
|
||||
&mut self.rng,
|
||||
));
|
||||
},
|
||||
RoomType::EmptyRoom => {},
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -247,8 +290,8 @@ impl LevelGenerator {
|
||||
level: self.level,
|
||||
structure,
|
||||
discovered: [[false; LEVEL_HEIGHT]; LEVEL_WIDTH],
|
||||
monsters: vec![],
|
||||
artifacts: vec![],
|
||||
monsters,
|
||||
artifacts,
|
||||
start: start_pos,
|
||||
end: end_pos,
|
||||
rng: rand::thread_rng(),
|
||||
|
@ -1,6 +1,6 @@
|
||||
use ratatui::buffer::Buffer;
|
||||
use ratatui::layout::Rect;
|
||||
use ratatui::style::Color;
|
||||
use ratatui::style::{Color, Modifier, Style};
|
||||
use ratatui::widgets::{StatefulWidget, Widget};
|
||||
|
||||
use crate::game::Game;
|
||||
@ -15,6 +15,9 @@ impl LevelWidget {
|
||||
fn set_cell(&self, buf: &mut Buffer, x: u16, y: u16, symbol: &str, fg: Color, bg: Color) {
|
||||
buf[(x, y)].set_symbol(symbol).set_bg(bg).set_fg(fg);
|
||||
}
|
||||
fn set_bold_cell(&self, buf: &mut Buffer, x: u16, y: u16, symbol: &str, fg: Color, bg: Color) {
|
||||
buf[(x, y)].set_symbol(symbol).set_bg(bg).set_fg(fg).set_style(Style::new().add_modifier(Modifier::BOLD));
|
||||
}
|
||||
}
|
||||
|
||||
impl StatefulWidget for LevelWidget {
|
||||
@ -46,7 +49,7 @@ impl StatefulWidget for LevelWidget {
|
||||
}
|
||||
StructureElement::Wall => {
|
||||
// TODO add fancy walls with https://en.wikipedia.org/wiki/Box-drawing_characters
|
||||
self.set_cell(buf, x, y, "#", FG_BROWN, Color::Gray);
|
||||
self.set_cell(buf, x, y, "#", FG_BROWN, FG_BROWN);
|
||||
}
|
||||
StructureElement::Floor => {
|
||||
self.set_cell(buf, x, y, " ", FG_BROWN, Color::Gray);
|
||||
@ -68,7 +71,7 @@ impl StatefulWidget for LevelWidget {
|
||||
}
|
||||
(_, _, Some(t)) => {
|
||||
let (s, c) = t.get_representation();
|
||||
self.set_cell(buf, x, y, s, c, Color::Gray);
|
||||
self.set_bold_cell(buf, x, y, s, c, Color::Gray);
|
||||
}
|
||||
(None, None, None) => {}
|
||||
};
|
||||
|
12
src/room.rs
12
src/room.rs
@ -119,25 +119,19 @@ impl Room {
|
||||
match self.kind {
|
||||
RoomType::Start => {
|
||||
tgt[left + self.special.0][top + self.special.1] = StructureElement::Start;
|
||||
(left + self.special.0, top + self.special.1)
|
||||
}
|
||||
RoomType::End => {
|
||||
tgt[left + self.special.0][top + self.special.1] = StructureElement::End;
|
||||
(left + self.special.0, top + self.special.1)
|
||||
}
|
||||
RoomType::StairUp => {
|
||||
tgt[left + self.special.0][top + self.special.1] = StructureElement::StairUp;
|
||||
(left + self.special.0, top + self.special.1)
|
||||
}
|
||||
RoomType::StairDown => {
|
||||
tgt[left + self.special.0][top + self.special.1] = StructureElement::StairDown;
|
||||
(left + self.special.0, top + self.special.1)
|
||||
}
|
||||
RoomType::BasicRoom => (0, 0),
|
||||
RoomType::ArtifactRoom => (0, 0),
|
||||
RoomType::MonsterRoom => (0, 0),
|
||||
RoomType::EmptyRoom => (0, 0),
|
||||
}
|
||||
_ => {}
|
||||
};
|
||||
(left + self.special.0, top + self.special.1)
|
||||
}
|
||||
}
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user