Discard class
This commit is contained in:
parent
093a65ae96
commit
1a0cad3b87
1 changed files with 131 additions and 15 deletions
146
src/game.rs
146
src/game.rs
|
@ -14,11 +14,21 @@ pub type Color = &'static str;
|
|||
pub const COLORS: [Color; 5] = ["blue", "red", "yellow", "white", "green"];
|
||||
|
||||
pub type Value = u32;
|
||||
// list of (value, count) pairs
|
||||
// list of values, assumed to be small to large
|
||||
pub const VALUES : [Value; 5] = [1, 2, 3, 4, 5];
|
||||
pub const VALUE_COUNTS : [(Value, u32); 5] = [(1, 3), (2, 2), (3, 2), (4, 2), (5, 1)];
|
||||
pub const FINAL_VALUE : Value = 5;
|
||||
|
||||
pub fn get_count_for_value(value: &Value) -> usize {
|
||||
match *value {
|
||||
1 => 3,
|
||||
2 | 3 | 4 => 2,
|
||||
5 => 1,
|
||||
_ => { panic!(format!("Unexpected value: {}", value)); }
|
||||
}
|
||||
}
|
||||
|
||||
pub type Player = u32;
|
||||
|
||||
#[derive(Debug)]
|
||||
pub struct Card {
|
||||
pub color: Color,
|
||||
|
@ -68,7 +78,7 @@ impl <T> fmt::Display for Pile<T> where T: fmt::Display {
|
|||
for item in &self.0 {
|
||||
try!(f.write_str(&format!("{}, ", item)));
|
||||
}
|
||||
try!(f.write_str(""));
|
||||
try!(f.write_str("]"));
|
||||
Ok(())
|
||||
}
|
||||
}
|
||||
|
@ -77,8 +87,6 @@ pub type Cards = Pile<Card>;
|
|||
|
||||
pub type CardsInfo = Pile<CardInfo>;
|
||||
|
||||
pub type Player = u32;
|
||||
|
||||
#[derive(Debug)]
|
||||
pub struct Firework {
|
||||
pub color: Color,
|
||||
|
@ -101,11 +109,7 @@ impl Firework {
|
|||
}
|
||||
|
||||
fn desired_value(&self) -> Option<Value> {
|
||||
if self.complete() {
|
||||
None
|
||||
} else {
|
||||
Some(self.top_value() + 1)
|
||||
}
|
||||
if self.complete() { None } else { Some(self.top_value() + 1) }
|
||||
}
|
||||
|
||||
fn score(&self) -> usize {
|
||||
|
@ -140,6 +144,75 @@ impl fmt::Display for Firework {
|
|||
}
|
||||
}
|
||||
|
||||
#[derive(Debug)]
|
||||
pub struct Discard {
|
||||
pub cards: Cards,
|
||||
counts: HashMap<Color, HashMap<Value, usize>>,
|
||||
}
|
||||
impl Discard {
|
||||
fn new() -> Discard {
|
||||
let mut counts = HashMap::new();
|
||||
for color in COLORS.iter() {
|
||||
let mut color_count = HashMap::new();
|
||||
for value in VALUES.iter() {
|
||||
color_count.insert(*value, 0);
|
||||
}
|
||||
counts.insert(*color, color_count);
|
||||
}
|
||||
Discard {
|
||||
cards: Cards::new(),
|
||||
counts: counts,
|
||||
}
|
||||
}
|
||||
|
||||
fn get_count(&self, card: &Card) -> usize {
|
||||
let color_count = self.counts.get(card.color).unwrap();
|
||||
color_count.get(&card.value).unwrap().clone()
|
||||
}
|
||||
|
||||
fn has_all(&self, card: &Card) -> bool {
|
||||
self.remaining(card) == 0
|
||||
}
|
||||
|
||||
fn remaining(&self, card: &Card) -> usize {
|
||||
let count = self.get_count(&card);
|
||||
get_count_for_value(&card.value) - count
|
||||
}
|
||||
|
||||
fn place(&mut self, card: Card) {
|
||||
let count = self.get_count(&card);
|
||||
let ref mut color_count = self.counts.get_mut(card.color).unwrap();
|
||||
color_count.insert(card.value, count + 1);
|
||||
self.cards.place(card);
|
||||
}
|
||||
}
|
||||
impl fmt::Display for Discard {
|
||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||
// try!(f.write_str(&format!(
|
||||
// "{}", self.cards,
|
||||
// )));
|
||||
for color in COLORS.iter() {
|
||||
let colorchar = color.chars().next().unwrap();
|
||||
try!(f.write_str(&format!(
|
||||
"{}: ", colorchar,
|
||||
)));
|
||||
let color_count = self.counts.get(color).unwrap();
|
||||
for value in VALUES.iter() {
|
||||
let count = color_count.get(value).unwrap();
|
||||
let total = get_count_for_value(value);
|
||||
try!(f.write_str(&format!(
|
||||
"{}/{} {}s", count, total, value
|
||||
)));
|
||||
if *value != FINAL_VALUE {
|
||||
try!(f.write_str(", "));
|
||||
}
|
||||
}
|
||||
try!(f.write_str("\n"));
|
||||
}
|
||||
Ok(())
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Debug)]
|
||||
pub enum Hinted {
|
||||
Color(Color),
|
||||
|
@ -260,9 +333,10 @@ fn new_deck() -> Cards {
|
|||
let mut deck: Cards = Cards::from(Vec::new());
|
||||
|
||||
for color in COLORS.iter() {
|
||||
for &(value, count) in VALUE_COUNTS.iter() {
|
||||
for value in VALUES.iter() {
|
||||
let count = get_count_for_value(value);
|
||||
for _ in 0..count {
|
||||
deck.place(Card {color: color, value: value});
|
||||
deck.place(Card {color: color, value: value.clone()});
|
||||
}
|
||||
}
|
||||
};
|
||||
|
@ -276,7 +350,7 @@ fn new_deck() -> Cards {
|
|||
#[derive(Debug)]
|
||||
pub struct BoardState {
|
||||
deck: Cards,
|
||||
pub discard: Cards,
|
||||
pub discard: Discard,
|
||||
pub fireworks: HashMap<Color, Firework>,
|
||||
|
||||
pub num_players: u32,
|
||||
|
@ -305,7 +379,7 @@ impl BoardState {
|
|||
BoardState {
|
||||
deck: new_deck(),
|
||||
fireworks: fireworks,
|
||||
discard: Cards::new(),
|
||||
discard: Discard::new(),
|
||||
num_players: opts.num_players,
|
||||
player: 0,
|
||||
turn: 1,
|
||||
|
@ -330,6 +404,47 @@ impl BoardState {
|
|||
Some(card.value) == firework.desired_value()
|
||||
}
|
||||
|
||||
pub fn was_played(&self, card: &Card) -> bool {
|
||||
let firework = self.fireworks.get(card.color).unwrap();
|
||||
if firework.complete() {
|
||||
true
|
||||
} else {
|
||||
card.value < firework.desired_value().unwrap()
|
||||
}
|
||||
}
|
||||
|
||||
// is never going to play, based on discard + fireworks
|
||||
pub fn is_unplayable(&self, card: &Card) -> bool {
|
||||
let firework = self.fireworks.get(card.color).unwrap();
|
||||
if firework.complete() {
|
||||
true
|
||||
} else {
|
||||
let desired = firework.desired_value().unwrap();
|
||||
if card.value < desired {
|
||||
true
|
||||
} else {
|
||||
let mut playable = true;
|
||||
for value in VALUES.iter() {
|
||||
if *value < desired {
|
||||
// already have these cards
|
||||
continue
|
||||
} else if *value > card.value {
|
||||
// don't care about these cards
|
||||
break
|
||||
} else {
|
||||
// need these cards
|
||||
let needed_card = Card {color: card.color, value: value.clone()};
|
||||
if self.discard.has_all(&needed_card) {
|
||||
// already discarded all of these
|
||||
playable = false;
|
||||
}
|
||||
}
|
||||
}
|
||||
playable
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
pub fn get_players(&self) -> Vec<Player> {
|
||||
(0..self.num_players).collect::<Vec<_>>()
|
||||
}
|
||||
|
@ -377,7 +492,8 @@ impl fmt::Display for BoardState {
|
|||
for (_, firework) in &self.fireworks {
|
||||
try!(f.write_str(&format!(" {}\n", firework)));
|
||||
}
|
||||
// discard: Cards::new(),
|
||||
try!(f.write_str("Discard:\n"));
|
||||
try!(f.write_str(&format!("{}\n", self.discard)));
|
||||
|
||||
Ok(())
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue