Skip to content

Commit 35e0131

Browse files
rust port of Amazing
1 parent dd65000 commit 35e0131

File tree

2 files changed

+206
-0
lines changed

2 files changed

+206
-0
lines changed

02_Amazing/rust/Cargo.toml

+9
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
[package]
2+
name = "rust"
3+
version = "0.1.0"
4+
edition = "2018"
5+
6+
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
7+
8+
[dependencies]
9+
rand = "0.8.5"

02_Amazing/rust/src/main.rs

+197
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,197 @@
1+
use rand::{Rng, thread_rng, prelude::SliceRandom};
2+
use std::{io, collections::HashSet};
3+
4+
fn main() {
5+
//DATA
6+
enum Direction {
7+
LEFT=0,
8+
UP=1,
9+
RIGHT=2,
10+
DOWN=3,
11+
}
12+
impl Direction {
13+
fn val(&self) -> usize {
14+
match self {
15+
Direction::LEFT=>0,
16+
Direction::UP=>1,
17+
Direction::RIGHT=>2,
18+
Direction::DOWN=>3,
19+
}
20+
}
21+
}
22+
const EXIT_DOWN:usize = 1;
23+
const EXIT_RIGHT:usize = 2;
24+
let mut rng = thread_rng(); //rng
25+
/*
26+
vector of:
27+
vectors of:
28+
integers
29+
Initially set to 0, unprocessed cells.
30+
Filled in with consecutive non-zero numbers as cells are processed
31+
*/
32+
let mut used; //2d vector
33+
/*
34+
vector of:
35+
vectors of:
36+
integers
37+
Remains 0 if there is no exit down or right
38+
Set to 1 if there is an exit down
39+
Set to 2 if there is an exit right
40+
Set to 3 if there are exits down and right
41+
*/
42+
let mut walls; //2d vector
43+
let width;
44+
let height;
45+
let entrance_column; //rng, column of entrance
46+
let mut row;
47+
let mut col;
48+
let mut count;
49+
50+
51+
52+
//print welcome message
53+
println!("
54+
AMAZING PROGRAM
55+
CREATIVE COMPUTING MORRISTOWN, NEW JERSEY\n\n\n\n");
56+
57+
//prompt for input
58+
width = get_user_input("What is your width?");
59+
print!("\n"); //one blank line below
60+
height = get_user_input("What is your height?");
61+
print!("\n\n\n\n");//4 blank lines below
62+
63+
//generate maze
64+
//initialize used and wall vectors
65+
//2d vectors when you don't know the sizes at compile time are wierd, but here's how it's done :)
66+
used = vec![0; (width * height) as usize];
67+
let mut used: Vec<_> = used.as_mut_slice().chunks_mut(width as usize).collect();
68+
let used = used.as_mut_slice(); //accessible as used[][]
69+
//2d vectors when you don't know the sizes at compile time are wierd, but here's how it's done :)
70+
walls = vec![0; (width * height) as usize];
71+
let mut walls: Vec<_> = walls.as_mut_slice().chunks_mut(width as usize).collect();
72+
let walls = walls.as_mut_slice(); //accessible as walls[][]
73+
74+
entrance_column=rng.gen_range(0..width-1);
75+
row = 0;
76+
col = entrance_column;
77+
count = 1;
78+
used[row][col] = count;
79+
count += 1;
80+
81+
while count != width*height + 1 {
82+
//remove possible directions that are blocked or
83+
//hit cells already processed
84+
let mut possible_directions: HashSet<usize> = vec![Direction::LEFT.val(),Direction::UP.val(),Direction::RIGHT.val(),Direction::DOWN.val()].into_iter().collect();
85+
if col==0 || used[row][col-1]!=0 {
86+
possible_directions.remove(&Direction::LEFT.val());
87+
}
88+
if row==0 || used[row-1][col]!=0 {
89+
possible_directions.remove(&Direction::UP.val());
90+
}
91+
if col==width-1 || used[row][col+1]!=0 {
92+
possible_directions.remove(&Direction::RIGHT.val());
93+
}
94+
if row==height-1 || used[row+1][col]!=0 {
95+
possible_directions.remove(&Direction::DOWN.val());
96+
}
97+
98+
//If we can move in a direction, move and make opening
99+
if possible_directions.len() != 0 { //all values in possible_directions are not NONE
100+
let pos_dir_vec: Vec<_> = possible_directions.into_iter().collect(); // convert the set to a vector to get access to the choose method
101+
//select a random direction
102+
match pos_dir_vec.choose(&mut rng).expect("error") {
103+
0=> {
104+
col -= 1;
105+
walls[row][col] = EXIT_RIGHT;
106+
},
107+
1=> {
108+
row -= 1;
109+
walls[row][col] = EXIT_DOWN;
110+
},
111+
2=>{
112+
walls[row][col] = walls[row][col] + EXIT_RIGHT;
113+
col += 1;
114+
},
115+
3=>{
116+
walls[row][col] = walls[row][col] + EXIT_DOWN;
117+
row += 1;
118+
},
119+
_=>{},
120+
}
121+
used[row][col]=count;
122+
count += 1;
123+
}
124+
//otherwise, move to the next used cell, and try again
125+
else {
126+
loop {
127+
if col != width-1 {col += 1;}
128+
else if row != height-1 {row+=1; col=0;}
129+
else {row=0;col=0;}
130+
131+
if used[row][col] != 0 {break;}
132+
}
133+
}
134+
135+
}
136+
// Add a random exit
137+
col=rng.gen_range(0..width);
138+
row=height-1;
139+
walls[row][col]+=1;
140+
141+
//print maze
142+
//first line
143+
for c in 0..width {
144+
if c == entrance_column {
145+
print!(". ");
146+
}
147+
else {
148+
print!(".--");
149+
}
150+
}
151+
println!(".");
152+
//rest of maze
153+
for r in 0..height {
154+
print!("I");
155+
for c in 0..width {
156+
if walls[r][c]<2 {print!(" I");}
157+
else {print!(" ");}
158+
}
159+
println!();
160+
for c in 0..width {
161+
if walls[r][c] == 0 || walls[r][c]==2 {print!(":--");}
162+
else {print!(": ");}
163+
}
164+
println!(".");
165+
}
166+
167+
168+
169+
170+
171+
172+
}
173+
174+
fn get_user_input(prompt: &str) -> usize {
175+
//DATA
176+
let mut raw_input = String::new(); // temporary variable for user input that can be parsed later
177+
178+
//input loop
179+
return loop {
180+
181+
//print prompt
182+
println!("{}", prompt);
183+
184+
//read user input from standard input, and store it to raw_input
185+
raw_input.clear(); //clear input
186+
io::stdin().read_line(&mut raw_input).expect( "CANNOT READ INPUT!");
187+
188+
//from input, try to read a number
189+
match raw_input.trim().parse::<usize>() {
190+
Ok(i) => break i, // this escapes the loop, returning i
191+
Err(e) => {
192+
println!("MEANINGLESS DIMENSION. TRY AGAIN. {}", e.to_string().to_uppercase());
193+
continue; // run the loop again
194+
}
195+
};
196+
}
197+
}

0 commit comments

Comments
 (0)