master
Rostyslav Hnatyshyn 1 year ago
parent 2fd7c9c9d5
commit 4f467758d8
  1. 82
      src/lib/ai.rs
  2. 126
      src/lib/entity.rs
  3. 13
      src/lib/point.rs
  4. 37
      src/lib/screen.rs
  5. 31
      src/lib/world.rs
  6. 294
      src/main.rs

@ -0,0 +1,82 @@
use crate::lib::screen::{Screen, BoardCommand};
use crate::World;
use crate::Point;
use crate::lib::entity::{Ant, Egg, Queen};
use rand::thread_rng;
use rand::prelude::SliceRandom;
pub trait AI {
fn step(&mut self, b: &Screen, w: &World) -> BoardCommand;
fn get_position(&self) -> Point;
}
impl AI for Ant {
fn get_position(&self) -> Point {
self.pos.clone()
}
// return the next move for this ant
fn step(&mut self, b: &Screen, w: &World) -> BoardCommand {
let valid = w.get_valid_movements(&self.pos, b);
let mut rng = thread_rng();
let choice = valid.choose(&mut rng).cloned();
if choice.is_none() {
return BoardCommand::Noop;
} else {
let pos = choice.unwrap();
if w.cleared.contains(&pos) {
self.pos = pos;
return BoardCommand::Noop;
} else {
return BoardCommand::Dig(pos);
}
}
}
}
impl AI for Egg {
fn step(&mut self, b: &Screen, w: &World) -> BoardCommand {
if self.counter > 0 {
self.counter -= 1;
return BoardCommand::Noop;
} else {
BoardCommand::Hatch(self.id, self.queen_id)
}
}
fn get_position(&self) -> Point {
self.pos.clone()
}
}
impl AI for Queen {
fn get_position(&self) -> Point {
self.pos.clone()
}
fn step(&mut self, b: &Screen, w: &World) -> BoardCommand {
let valid: Vec<Point> = w.get_valid_movements(&self.pos, b);
let mut rng = thread_rng();
let choice = valid.choose(&mut rng).cloned();
if choice.is_none() {
return BoardCommand::Noop;
} else {
let pos = choice.unwrap();
if w.cleared.contains(&pos) {
// choose between laying an egg and moving
if self.egg_count < 3 {
self.egg_count += 1;
return BoardCommand::LayEgg(pos, self.id);
} else {
self.pos = pos;
return BoardCommand::Noop;
}
} else {
return BoardCommand::Noop;
}
}
}
}

@ -0,0 +1,126 @@
use crate::Point;
use std::collections::HashMap;
use crate::lib::ai::AI;
use downcast_rs::{impl_downcast, Downcast};
pub trait Entity: AI + Renderable + Downcast {}
impl_downcast!(Entity);
pub struct Ant {
pub pos: Point,
pub id: u32,
}
impl Ant {
pub fn new(x: i32, y: i32, id: u32) -> Ant {
Ant {
pos: Point(x, y),
id,
}
}
}
impl Entity for Ant {}
impl Renderable for Ant {
fn render(&self) -> &str {
"o"
}
}
pub struct Queen {
pub pos: Point,
pub egg_count: u8,
pub id: u32,
}
impl Renderable for Queen {
fn render(&self) -> &str {
"q"
}
}
pub trait Renderable {
fn render(&self) -> &str {
"z"
}
}
impl Queen {
pub fn new(x: i32, y: i32, id: u32) -> Queen {
Queen {
pos: Point(x, y),
egg_count: 0,
id,
}
}
}
#[derive(Debug)]
pub struct Egg {
pub pos: Point,
pub counter: u8,
pub queen_id: u32,
pub id: u32,
}
impl Renderable for Egg {
fn render(&self) -> &str {
"e"
}
}
impl Egg {
pub fn new(x: i32, y: i32, id: u32, queen_id: u32) -> Egg {
Egg {
pos: Point(x, y),
counter: 10,
id,
queen_id,
}
}
}
impl Entity for Queen {}
impl Entity for Egg {}
pub struct Entities {
pub data: HashMap<u32, Box<dyn Entity>>,
id_counter: u32,
}
impl Entities {
pub fn new() -> Entities {
Entities {
id_counter: 0,
data: HashMap::new(),
}
}
pub fn add_ant(&mut self, x: i32, y: i32) -> u32 {
let a = Ant::new(x, y, self.id_counter);
let id = a.id;
self.data.insert(a.id, Box::new(a));
self.id_counter += 1;
id
}
pub fn add_queen(&mut self, x: i32, y: i32) -> u32 {
let q = Queen::new(x, y, self.id_counter);
let id = q.id;
self.data.insert(q.id, Box::new(q));
self.id_counter += 1;
id
}
pub fn add_egg(&mut self, x: i32, y: i32, queen_id: u32) -> u32 {
let e = Egg::new(x, y, self.id_counter, queen_id);
let id = e.id;
self.data.insert(e.id, Box::new(e));
self.id_counter += 1;
id
}
}

@ -0,0 +1,13 @@
#[derive(Hash, PartialEq, Eq, Clone, Debug, Copy)]
pub struct Point(pub i32, pub i32);
impl Point {
pub fn get_neighbors(&self) -> Vec<Point> {
vec![
Point(self.0, self.1 + 1),
Point(self.0, self.1 - 1),
Point(self.0 - 1, self.1),
Point(self.0 + 1, self.1),
]
}
}

@ -0,0 +1,37 @@
use crate::Point;
pub struct Screen {
pub center: Point,
max_x: i32,
max_y: i32,
}
impl Screen {
pub fn new(max_x: i32, max_y: i32) -> Screen {
Screen {
center: Point(max_x / 2, max_y / 2),
max_x,
max_y,
}
}
pub fn is_in_bounds(&self, pos: &Point) -> bool {
(pos.0 > -self.max_x && pos.0 < self.max_x) && (pos.1 > -self.max_y && pos.1 < self.max_y)
}
pub fn get_valid_movements(&self, pos: &Point) -> Vec<Point> {
let binding = pos.get_neighbors();
binding
.iter()
.filter(|e| self.is_in_bounds(e))
.map(|e| e.clone())
.collect()
}
}
pub enum BoardCommand {
Dig(Point),
LayEgg(Point, u32),
Hatch(u32, u32),
Noop,
}

@ -0,0 +1,31 @@
use crate::Screen;
use crate::Point;
use std::collections::HashSet;
#[derive(Clone)]
pub struct World {
pub cleared: HashSet<Point>,
pub occupied: HashSet<Point>,
}
impl World {
pub fn new() -> World {
World {
cleared: HashSet::new(),
occupied: HashSet::new(),
}
}
pub fn clear(&mut self, pos: Point) {
self.cleared.insert(pos);
}
pub fn get_valid_movements(&self, pos: &Point, b: &Screen) -> Vec<Point> {
let moves = b.get_valid_movements(pos);
moves
.iter()
.filter(|p| !self.occupied.contains(p))
.map(|p| p.clone())
.collect()
}
}

@ -2,246 +2,24 @@ extern crate downcast_rs;
extern crate ncurses; extern crate ncurses;
use ncurses::*; use ncurses::*;
use rand::seq::SliceRandom;
use rand::thread_rng;
use std::collections::{HashMap, HashSet};
use std::thread::sleep; use std::thread::sleep;
use std::time; use std::time;
#[derive(Hash, PartialEq, Eq, Clone, Debug, Copy)] mod lib {
struct Point(i32, i32); pub mod point;
pub mod screen;
struct Board { pub mod world;
center: Point, pub mod entity;
max_x: i32, pub mod ai;
max_y: i32,
}
fn get_neighbors(pos: &Point) -> Vec<Point> {
vec![
Point(pos.0, pos.1 + 1),
Point(pos.0, pos.1 - 1),
Point(pos.0 - 1, pos.1),
Point(pos.0 + 1, pos.1),
]
}
impl Board {
fn new(max_x: i32, max_y: i32) -> Board {
Board {
center: Point(max_x / 2, max_y / 2),
max_x,
max_y,
}
}
fn is_in_bounds(&self, pos: &Point) -> bool {
(pos.0 > -self.max_x && pos.0 < self.max_x) && (pos.1 > -self.max_y && pos.1 < self.max_y)
}
fn get_valid_movements(&self, pos: &Point) -> Vec<Point> {
let binding = get_neighbors(pos);
binding
.iter()
.filter(|e| self.is_in_bounds(e))
.map(|e| e.clone())
.collect()
}
}
#[derive(Clone)]
struct World {
cleared: HashSet<Point>,
occupied: HashSet<Point>,
}
impl World {
fn new() -> World {
World {
cleared: HashSet::new(),
occupied: HashSet::new(),
}
}
fn clear(&mut self, pos: Point) {
self.cleared.insert(pos);
} }
fn get_valid_movements(&self, pos: &Point, b: &Board) -> Vec<Point> { use lib::point::Point;
let moves = b.get_valid_movements(pos); use lib::screen::{BoardCommand, Screen};
moves use lib::world::World;
.iter() use lib::entity::{Queen, Entities};
.filter(|p| !self.occupied.contains(p))
.map(|p| p.clone())
.collect()
}
}
use downcast_rs::{impl_downcast, Downcast};
trait Entity: AI + Renderable + Downcast {}
impl_downcast!(Entity);
trait AI {
fn step(&mut self, b: &Board, w: &World) -> BoardCommand;
fn get_position(&self) -> Point;
}
struct Ant { fn render(e: &Entities, w: &World, b: &Screen) {
pos: Point,
id: u32,
}
impl Ant {
fn new(x: i32, y: i32, id: u32) -> Ant {
Ant {
pos: Point(x, y),
id,
}
}
}
impl Entity for Ant {}
impl Renderable for Ant {
fn render(&self) -> &str {
"o"
}
}
struct Queen {
pos: Point,
egg_count: u8,
id: u32,
}
impl Renderable for Queen {
fn render(&self) -> &str {
"q"
}
}
trait Renderable {
fn render(&self) -> &str {
"z"
}
}
impl Queen {
fn new(x: i32, y: i32, id: u32) -> Queen {
Queen {
pos: Point(x, y),
egg_count: 0,
id,
}
}
}
#[derive(Debug)]
struct Egg {
pos: Point,
counter: u8,
queen_id: u32,
id: u32,
}
impl Renderable for Egg {
fn render(&self) -> &str {
"e"
}
}
impl Egg {
fn new(x: i32, y: i32, id: u32, queen_id: u32) -> Egg {
Egg {
pos: Point(x, y),
counter: 10,
id,
queen_id,
}
}
}
impl AI for Egg {
fn step(&mut self, b: &Board, w: &World) -> BoardCommand {
if self.counter > 0 {
self.counter -= 1;
return BoardCommand::Noop;
} else {
BoardCommand::Hatch(self.id, self.queen_id)
}
}
fn get_position(&self) -> Point {
self.pos.clone()
}
}
enum BoardCommand {
Dig(Point),
LayEgg(Point, u32),
Hatch(u32, u32),
Noop,
}
impl AI for Ant {
fn get_position(&self) -> Point {
self.pos.clone()
}
// return the next move for this ant
fn step(&mut self, b: &Board, w: &World) -> BoardCommand {
let valid = w.get_valid_movements(&self.pos, b);
let mut rng = thread_rng();
let choice = valid.choose(&mut rng).cloned();
if choice.is_none() {
return BoardCommand::Noop;
} else {
let pos = choice.unwrap();
if w.cleared.contains(&pos) {
self.pos = pos;
return BoardCommand::Noop;
} else {
return BoardCommand::Dig(pos);
}
}
}
}
impl Entity for Queen {}
impl Entity for Egg {}
impl AI for Queen {
fn get_position(&self) -> Point {
self.pos.clone()
}
fn step(&mut self, b: &Board, w: &World) -> BoardCommand {
let valid: Vec<Point> = w.get_valid_movements(&self.pos, b);
let mut rng = thread_rng();
let choice = valid.choose(&mut rng).cloned();
if choice.is_none() {
return BoardCommand::Noop;
} else {
let pos = choice.unwrap();
if w.cleared.contains(&pos) {
// choose between laying an egg and moving
if self.egg_count < 3 {
self.egg_count += 1;
return BoardCommand::LayEgg(pos, self.id);
} else {
self.pos = pos;
return BoardCommand::Noop;
}
} else {
return BoardCommand::Noop;
}
}
}
}
fn render(e: &Entities, w: &World, b: &Board) {
for c in w.cleared.iter() { for c in w.cleared.iter() {
mvprintw(c.1 + b.center.1, c.0 + b.center.0, "x"); mvprintw(c.1 + b.center.1, c.0 + b.center.0, "x");
} }
@ -262,7 +40,7 @@ mod tests {
let max_x = 20; let max_x = 20;
let max_y = 20; let max_y = 20;
let mut board = Board::new(max_x, max_y); let mut board = Screen::new(max_x, max_y);
let mut world = World::new(); let mut world = World::new();
dbg!(board.is_in_bounds(&Point(0, 0))); dbg!(board.is_in_bounds(&Point(0, 0)));
@ -270,7 +48,7 @@ mod tests {
} }
} }
fn simulate(e: &mut Entities, w: &mut World, b: &mut Board) { fn simulate(e: &mut Entities, w: &mut World, b: &mut Screen) {
let cmds: Vec<BoardCommand> = e let cmds: Vec<BoardCommand> = e
.data .data
.values_mut() .values_mut()
@ -301,43 +79,6 @@ fn simulate(e: &mut Entities, w: &mut World, b: &mut Board) {
let _ = e.data.values().map(|p| w.occupied.insert(p.get_position())); let _ = e.data.values().map(|p| w.occupied.insert(p.get_position()));
} }
struct Entities {
data: HashMap<u32, Box<dyn Entity>>,
id_counter: u32,
}
impl Entities {
fn new() -> Entities {
Entities {
id_counter: 0,
data: HashMap::new(),
}
}
fn add_ant(&mut self, x: i32, y: i32) -> u32 {
let a = Ant::new(x, y, self.id_counter);
let id = a.id;
self.data.insert(a.id, Box::new(a));
self.id_counter += 1;
id
}
fn add_queen(&mut self, x: i32, y: i32) -> u32 {
let q = Queen::new(x, y, self.id_counter);
let id = q.id;
self.data.insert(q.id, Box::new(q));
self.id_counter += 1;
id
}
fn add_egg(&mut self, x: i32, y: i32, queen_id: u32) -> u32 {
let e = Egg::new(x, y, self.id_counter, queen_id);
let id = e.id;
self.data.insert(e.id, Box::new(e));
self.id_counter += 1;
id
}
}
fn main() { fn main() {
initscr(); initscr();
@ -350,14 +91,9 @@ fn main() {
getmaxyx(stdscr(), &mut max_y, &mut max_x); getmaxyx(stdscr(), &mut max_y, &mut max_x);
// might move all of contents of board into world
// TODO: rename board to screen -> will keep track of where we are looking at in the world
// TODO: create entity factory to build new entities, will make it easier to have multiple
// colonies because no ids will ever overlap
// TODO: fix renderable to render different colors // TODO: fix renderable to render different colors
let mut board = Board::new(max_x, max_y); let mut board = Screen::new(max_x, max_y);
let mut world = World::new(); let mut world = World::new();
let mut entities = Entities::new(); let mut entities = Entities::new();

Loading…
Cancel
Save