aboutsummaryrefslogtreecommitdiff
path: root/src/world.rs
blob: fbbc6bea47b75468006e99d6ecdfe2011cf49c56 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
pub enum BlockType {
    Nothing,
    Wall,
    Floor,
}

pub struct World {
    size: usize,
    world: Vec<Vec<BlockType>>
}

pub trait GameWorld<'a> {
    fn new(size: usize) -> Self;

    fn generate(&mut self);

    fn get_world(&'a self) -> &'a Vec<Vec<BlockType>>;

    fn get_item(&'a self, x: usize, y: usize) -> &'a BlockType;
}

impl World {
    fn make_vertical_corridor(&mut self, start: (usize, usize), length: usize) {
        let x = start.0;
        let endy = start.1 + length;
        for y in start.1..endy {
            self.wall_up(x - 1, y);
            self.set_tile(x, y, BlockType::Floor);
            self.wall_up(x + 1, y);
        }
    }

    fn make_horizontal_corridor(&mut self, start: (usize, usize), length: usize) {
        let y = start.1;
        let endx = start.0 + length;
        for x in start.0..endx {
            self.wall_up(x, y - 1);
            self.set_tile(x, y, BlockType::Floor);
            self.wall_up(x, y - 1);
        }
    }

    fn set_tile(&mut self, x: usize, y: usize, block: BlockType) {
        self.world[y][x] = block;
    }

    /// Puts a wall on the coordinates if it isn't a floor.
    fn wall_up(&mut self, x: usize, y: usize) {
        self.set_tile(x, y, match self.world[y][x] {
            BlockType::Floor => BlockType::Floor,
            _ => BlockType::Wall
        })
    }

    /// Creates a room at the given coordinates of the given size.
    fn make_room(&mut self, start: (usize, usize), width: usize, height: usize) {
        let endx = start.0 + width;
        let endy = start.1 + height;

        // Draw the walls
        for x in start.0..endx {
            self.wall_up(x, start.1);
            self.wall_up(x, endy);
        }

        for y in start.1..endy {
            self.wall_up(start.0, y);
            self.wall_up(endx, y);
        }

        // Fill the room
        for x in (start.0 + 1)..(endx) {
            for y in (start.1 + 1)..(endy) {
                self.set_tile(x, y, BlockType::Floor);
            }
        }
    }
}

impl<'a> GameWorld<'a> for World {
    fn new(size: usize) -> World {
        World {
            size: size,
            world: Vec::with_capacity(size)
        }
    }
    
    fn generate(&mut self) {
        for _ in 0..self.size {
            let mut subvec = Vec::with_capacity(self.size);
            for _ in 0..self.size {
                subvec.push(BlockType::Nothing);
            }
            self.world.push(subvec);
        }

        self.make_room((1, 13), 5, 7);
        self.make_vertical_corridor((3, 5), 10);
    }

    fn get_world(&'a self) -> &'a Vec<Vec<BlockType>> {
        &self.world
    }

    fn get_item(&'a self, x: usize, y: usize) -> &'a BlockType {
        &self.world[x][y]
    }
}

#[cfg(test)]
mod tests {
    use super::*;

    #[test]
    fn test_generates_world() {
        let mut world = World::new(128);
        world.generate();

        assert_eq!(world.get_world().len(), 128);
        assert_eq!(world.get_world()[0].len(), 128);
        match world.get_world()[0][0] {
            BlockType::Nothing => assert!(true),
            _ => assert!(false)
        }
    }
}