X-Git-Url: http://git.scottworley.com/pluta-lesnura/blobdiff_plain/aa0622ab28e4bf320db9a2ff747e5004923e2570..c5e4b1eb46c33d636e95f3dbbbdf0bb15ce5d2ef:/src/lib.rs diff --git a/src/lib.rs b/src/lib.rs index 09bd5fb..c46aa7d 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -181,6 +181,17 @@ impl Hand { fn random(&self) -> Option<&Card> { self.cards.choose(&mut rand::thread_rng()) } + /// Make a new Hand that contains only cards of the requested suit + fn filter_by_suit(&self, suit: Suit) -> Self { + Self { + cards: self + .cards + .iter() + .filter(|c| c.suit().expect("I shouldn't have jokers in my hand") == suit) + .copied() + .collect(), + } + } } #[derive(Copy, Clone)] @@ -405,8 +416,9 @@ impl Player { } } -pub fn random_player(draw_chance: f64) -> impl FnMut(&Game) -> Play { - move |game: &Game| -> Play { +#[must_use] +pub fn random_player(draw_chance: f64) -> Player { + Player(Box::new(move |game: &Game| -> Play { match game.phase { Phase::Play => Play::Play( *game @@ -425,7 +437,22 @@ pub fn random_player(draw_chance: f64) -> impl FnMut(&Game) -> Play { } } } - } + })) +} + +/// When available, make plays that grant momentum. +#[must_use] +pub fn momentum_player(mut fallback: Player) -> Player { + Player(Box::new(move |game: &Game| -> Play { + if game.phase == Phase::Play { + if let Some(suit) = game.discard.top().and_then(Card::suit) { + if let Some(card) = game.current_player_hand().filter_by_suit(suit).random() { + return Play::Play(*card); + } + } + } + fallback.0(game) + })) } /// # Errors @@ -501,7 +528,7 @@ mod tests { #[test] fn test_game() { for num_players in 1..10 { - let players: Vec<_> = std::iter::from_fn(|| Some(Player::new(random_player(0.5)))) + let players: Vec<_> = std::iter::from_fn(|| Some(momentum_player(random_player(0.5)))) .take(num_players) .collect(); let mut game = Game::default();