minicrossterm/examples/program_examples/snake/src/main.rs

135 lines
3.6 KiB
Rust
Raw Normal View History

2018-08-20 07:13:21 +10:00
extern crate crossterm;
extern crate rand;
mod map;
mod messages;
2018-08-20 07:13:21 +10:00
mod snake;
mod variables;
use self::crossterm::{input, terminal, ClearType, Color, Crossterm, Screen};
2018-08-20 07:13:21 +10:00
use map::Map;
use snake::Snake;
use variables::{Direction, Position, Size};
2018-08-20 07:13:21 +10:00
use std::collections::HashMap;
use std::io::Read;
use std::io::Write;
use std::iter::Iterator;
use std::{thread, time};
2018-08-20 07:13:21 +10:00
fn main() {
let map_size = title_screen();
{
let mut screen = Screen::new(true);
let crossterm = Crossterm::from_screen(&screen);
2018-08-20 07:13:21 +10:00
let cursor = crossterm.cursor();
let mut input = crossterm.input();
cursor.hide();
2018-08-20 07:13:21 +10:00
let mut stdin = input.read_async().bytes();
let mut free_positions: HashMap<String, Position> =
HashMap::with_capacity((map_size.width * map_size.height) as usize);
2018-08-20 07:13:21 +10:00
let mut map = Map::new(map_size.clone());
map.render_map(&screen, &mut free_positions);
let mut direction = Direction::Right;
let mut snake = Snake::new(map_size.clone());
for part in snake.get_parts().iter() {
free_positions
.remove_entry(format!("{},{}", part.position.x, part.position.y).as_str());
2018-08-20 07:13:21 +10:00
}
map.spawn_food(&free_positions, &screen);
loop {
2018-08-25 02:49:25 +10:00
thread::sleep(time::Duration::from_millis(200));
2018-08-20 07:13:21 +10:00
let pressed_key = stdin.next();
if let Some(Ok(key)) = pressed_key {
match key as char {
'w' => direction = Direction::Up,
'a' => direction = Direction::Left,
's' => direction = Direction::Down,
'd' => direction = Direction::Right,
_ => {}
}
}
snake.move_snake(&direction, &screen, &mut free_positions);
if map.is_out_of_bounds(snake.snake_parts[0].position) {
2018-08-20 07:13:21 +10:00
break;
}
snake.draw_snake(&screen);
if snake.has_eaten_food(map.foot_pos) {
2018-08-20 07:13:21 +10:00
map.spawn_food(&free_positions, &screen);
}
}
2018-08-24 02:15:18 +10:00
drop(screen);
2018-08-20 07:13:21 +10:00
}
2018-08-24 02:15:18 +10:00
2018-08-20 07:13:21 +10:00
game_over_screen();
}
fn title_screen() -> Size {
let crossterm = Crossterm::new();
2018-08-20 07:13:21 +10:00
let cursor = crossterm.cursor();
let terminal = crossterm.terminal().clear(ClearType::All);
println!("{}", messages::SNAKERS.join("\n\r"));
2018-08-20 07:13:21 +10:00
cursor.goto(0, 15);
println!("Enter map width:");
cursor.goto(17, 15);
let width = crossterm.input().read_line().unwrap();
2018-08-25 19:04:42 +10:00
println!("\r\nEnter map height:");
cursor.goto(17, 17);
2018-08-20 07:13:21 +10:00
let height = crossterm.input().read_line().unwrap();
let parsed_width = width.parse::<usize>().unwrap();
let parsed_height = height.parse::<usize>().unwrap();
let terminal = crossterm.terminal().clear(ClearType::All);
return Size::new(parsed_width, parsed_height);
}
fn print_game_stats(map_size: Size, snake_lenght: usize, food_aten: usize, screen: &mut Screen) {
let crossterm = Crossterm::new();
2018-08-20 07:13:21 +10:00
let cursor = crossterm.cursor();
let terminal = crossterm.terminal().clear(ClearType::All);
screen.write(format!("Snake Lenght: {}\n\r", snake_lenght).as_ref());
screen.write(format!("Food aten: {}\n\r", snake_lenght).as_ref());
cursor.goto(0, map_size.height as u16);
cursor.goto(0, map_size.height as u16);
2018-08-20 07:13:21 +10:00
}
fn game_over_screen() {
let crossterm = Crossterm::new();
2018-08-20 07:13:21 +10:00
let cursor = crossterm.cursor();
let terminal = crossterm.terminal();
terminal.clear(ClearType::All);
println!(
"{}",
crossterm
.style(format!("{}", messages::END_MESSAGE.join("\n\r")))
.with(Color::Red)
);
// cursor.goto()
2018-08-20 07:13:21 +10:00
cursor.show();
}