From eddc5bfafd8f302cb190a98b3bec86550f0fad7e Mon Sep 17 00:00:00 2001 From: Dominic Masters Date: Fri, 13 Aug 2021 09:32:15 -0700 Subject: [PATCH] Added basic betting, turns, and added queue restacking --- include/dawn/dawn.h | 1 + include/dawn/poker/bet.h | 7 ++- include/dawn/poker/player.h | 4 +- include/dawn/poker/turn.h | 19 +++++++ src/display/animation/queue.c | 24 +++++++++ src/display/animation/queue.h | 10 +++- src/game/poker/actions/bet.c | 98 +++++++++++++++++++++++++++++++++- src/game/poker/actions/bet.h | 2 + src/game/poker/actions/round.c | 5 +- src/game/poker/actions/round.h | 2 + src/poker/bet.c | 6 +++ src/poker/bet.h | 8 +++ src/poker/turn.c | 28 ++++++++++ src/poker/turn.h | 11 ++++ 14 files changed, 217 insertions(+), 8 deletions(-) create mode 100644 include/dawn/poker/turn.h create mode 100644 src/poker/turn.c create mode 100644 src/poker/turn.h diff --git a/include/dawn/dawn.h b/include/dawn/dawn.h index c52c29be..a6024a19 100644 --- a/include/dawn/dawn.h +++ b/include/dawn/dawn.h @@ -52,6 +52,7 @@ #include "locale/language.h" // Poker Game Logic +#include "poker/turn.h" #include "poker/bet.h" #include "poker/card.h" #include "poker/dealer.h" diff --git a/include/dawn/poker/bet.h b/include/dawn/poker/bet.h index 878626d1..69e90b1f 100644 --- a/include/dawn/poker/bet.h +++ b/include/dawn/poker/bet.h @@ -19,11 +19,14 @@ typedef struct { /** Blinds */ - uint32_t blindSmall, blindBig; + int32_t blindSmall, blindBig; + + /** How big the current bet is for the round. */ + int32_t currentBet; /** For Betting round, which player is currently betting */ uint8_t better; /** Current pot of chips */ - uint32_t pot; + int32_t pot; } pokerbet_t; \ No newline at end of file diff --git a/include/dawn/poker/player.h b/include/dawn/poker/player.h index e2da4c35..02b7ca89 100644 --- a/include/dawn/poker/player.h +++ b/include/dawn/poker/player.h @@ -48,8 +48,8 @@ typedef struct { uint8_t state; /** Chips in players' posession */ - uint32_t chips; + int32_t chips; /** Current bet in current round player has placed */ - uint32_t currentBet; + int32_t currentBet; } pokerplayer_t; \ No newline at end of file diff --git a/include/dawn/poker/turn.h b/include/dawn/poker/turn.h new file mode 100644 index 00000000..4704fd1e --- /dev/null +++ b/include/dawn/poker/turn.h @@ -0,0 +1,19 @@ +/** + * Copyright (c) 2021 Dominic Masters + * + * This software is released under the MIT License. + * https://opensource.org/licenses/MIT + */ + +#pragma once +#include "../libs.h" + +#define POKER_TURN_TYPE_OUT 0x00 +#define POKER_TURN_TYPE_FOLD 0x01 +#define POKER_TURN_TYPE_BET 0x02 + +typedef struct { + uint8_t type; + int32_t chips; + float confidence; +} pokerturn_t; \ No newline at end of file diff --git a/src/display/animation/queue.c b/src/display/animation/queue.c index c103f862..a7b62e37 100644 --- a/src/display/animation/queue.c +++ b/src/display/animation/queue.c @@ -65,6 +65,30 @@ void queueDispose(queue_t *queue) { if(action->onEnd != NULL) action->onEnd(queue, action, queue->current); } +void queueRestack(queue_t *queue) { + queueaction_t *action; + uint8_t i; + queueaction_t items[ANIMATION_QUEUE_ITEM_MAX]; + + // Take the current queue and copy it. + arrayCopy(sizeof(queueaction_t), + queue->items+queue->current, queue->count - queue->current, + items + ); + + // Now rewind the stack + queue->count -= queue->current; + queue->current = 0; + + // Now copy back + arrayCopy(sizeof(queueaction_t), items, queue->count, queue->items); + + // Now fix indexes + for(i = 0; i < queue->count; i++) { + queue->items[i].index = i; + } +} + void _queueDelayUpdate(queue_t *queue, queueaction_t *action, uint8_t i) { float n = queue->timeline - queue->actionStarted; diff --git a/src/display/animation/queue.h b/src/display/animation/queue.h index 3a461761..46948fdb 100644 --- a/src/display/animation/queue.h +++ b/src/display/animation/queue.h @@ -7,6 +7,7 @@ #pragma once #include +#include "../../util/array.h" /** * Initialize the queue set. @@ -41,10 +42,17 @@ void queueUpdate(queue_t *queue, engine_t *engine); */ void queueDispose(queue_t *queue); +/** + * Restacks the queue. The restack process essentially rewinds the queue so that + * the current items move back to position 0, and allows you to add more items + * to the queue again. + * + * @param queue Queue to restack. + */ +void queueRestack(queue_t *queue); /** Callbacks for Queue Delay Action */ void _queueDelayUpdate(queue_t *queue, queueaction_t *action, uint8_t i); - /** * Adds a delay action to a queue. * @param queue Queue to add to. diff --git a/src/game/poker/actions/bet.c b/src/game/poker/actions/bet.c index e31af3f1..05a0b66b 100644 --- a/src/game/poker/actions/bet.c +++ b/src/game/poker/actions/bet.c @@ -7,8 +7,102 @@ #include "bet.h" -queueaction_t * pokerGameActionBetAdd(pokergame_t *game) { +void _pokerGameActionBetOnStart( + queue_t *queue, queueaction_t *action, uint8_t i +) { +} + +void _pokerGameActionBetOnUpdate( + queue_t *queue, queueaction_t *action, uint8_t i +) { + bool isHuman; + bool turnMade = false; + pokerturn_t turn; + pokergame_t *game = (pokergame_t *)action->data; + pokerplayer_t *player; + + + // As of right now the queue should basically be empty besides this item, so + // let's restack + queueRestack(queue); + + // Are they human? + player = game->poker.players + game->poker.bet.better; + isHuman = game->poker.bet.better == POKER_PLAYER_HUMAN_INDEX; + + // Handle as an AI + if(isHuman) { + turn.type = POKER_TURN_TYPE_OUT; + turnMade = true; + } else { + turn = pokerTurnGet(&game->poker, game->poker.bet.better); + turnMade = true; + } + + // Now decide if we should do something. + if(!turnMade) return; + + // What happened? + char *debugAction; + switch(turn.type) { + // Player bets + case POKER_TURN_TYPE_BET: + debugAction = "betting"; + pokerBetPlayer(&game->poker, player, turn.chips); + break; + + // Player folds + case POKER_TURN_TYPE_FOLD: + debugAction = "folding"; + player->state |= POKER_PLAYER_STATE_FOLDED; + break; + + // Player may be out + default: + debugAction = "doing nothing"; + break; + } + + printf("Player %i is %s.\n", game->poker.bet.better, debugAction); + queueNext(queue); +} + +void _pokerGameActionBetOnEnd( + queue_t *queue, queueaction_t *action, uint8_t i +) { + uint8_t j; + pokerplayer_t *player; + pokergame_t *game = (pokergame_t *)action->data; + bool playersPending; + + // Go to the next better + game->poker.bet.better = (game->poker.bet.better + 1) % POKER_PLAYER_COUNT; + playersPending = false; + + // Check if each player needs to action. + for(j = 0; j < POKER_PLAYER_COUNT; j++) { + player = game->poker.players + j; + if(player->state & POKER_PLAYER_STATE_FOLDED) continue; + if(player->state & POKER_PLAYER_STATE_OUT) continue; + if(player->currentBet >= game->poker.bet.currentBet) continue; + + // Yes, this player needs to check, raise or fold + pokerGameActionBetAdd(game); + playersPending = true; + break; + } + + // Are we waiting on any players? + if(playersPending) return; + + // No! Begin the next flop. + printf("Not waiting on anything\n"); +} + +queueaction_t * pokerGameActionBetAdd(pokergame_t *game) { queueaction_t *action = pokerGameActionAdd(game); - + action->onStart = &_pokerGameActionBetOnStart; + action->onUpdate = &_pokerGameActionBetOnUpdate; + action->onEnd = &_pokerGameActionBetOnEnd; return action; } \ No newline at end of file diff --git a/src/game/poker/actions/bet.h b/src/game/poker/actions/bet.h index 4454e432..fa90da21 100644 --- a/src/game/poker/actions/bet.h +++ b/src/game/poker/actions/bet.h @@ -8,5 +8,7 @@ #pragma once #include #include "action.h" +#include "../../../poker/turn.h" +#include "../../../poker/bet.h" queueaction_t * pokerGameActionBetAdd(pokergame_t *game); \ No newline at end of file diff --git a/src/game/poker/actions/round.c b/src/game/poker/actions/round.c index 9142bc94..6ef7f547 100644 --- a/src/game/poker/actions/round.c +++ b/src/game/poker/actions/round.c @@ -35,7 +35,10 @@ void _pokerGameActionRoundOnEnd(queue_t *queue,queueaction_t *action,uint8_t i){ // Deal pokerActionDealAdd(queue, &game->poker); - // Begin Betting Round + // Begin Betting Round. This will queue for one player only and then the round + // will take over. + pokerBetResetBetter(&game->poker); + pokerGameActionBetAdd(game); } queueaction_t * pokerGameActionRoundAdd(pokergame_t *game) { diff --git a/src/game/poker/actions/round.h b/src/game/poker/actions/round.h index f548a77d..632f9372 100644 --- a/src/game/poker/actions/round.h +++ b/src/game/poker/actions/round.h @@ -6,9 +6,11 @@ #pragma once #include #include "action.h" +#include "../../../poker/bet.h" #include "../../../poker/actions/round.h" #include "../../../poker/actions/blinds.h" #include "../../../poker/actions/deal.h" +#include "bet.h" void _pokerGameActionRoundOnStart( queue_t *queue, queueaction_t *action, uint8_t i diff --git a/src/poker/bet.c b/src/poker/bet.c index e5625536..6b12fed7 100644 --- a/src/poker/bet.c +++ b/src/poker/bet.c @@ -14,12 +14,18 @@ void pokerBetInit(pokerbet_t *bet) { void pokerBetReset(pokerbet_t *bet) { bet->pot = 0; + bet->currentBet = 0; } void pokerBetPlayer(poker_t *poker, pokerplayer_t *player, int32_t chips) { poker->bet.pot += chips; player->chips -= chips; player->currentBet += chips; + poker->bet.currentBet = mathMax(poker->bet.currentBet, player->currentBet); +} + +void pokerBetResetBetter(poker_t *poker) { + poker->bet.better = (poker->roundSmallBlind + 1) % POKER_PLAYER_COUNT; } void pokerBetTakeBlinds(poker_t *poker) { diff --git a/src/poker/bet.h b/src/poker/bet.h index 72a32c7a..16e605f0 100644 --- a/src/poker/bet.h +++ b/src/poker/bet.h @@ -29,6 +29,14 @@ void pokerBetReset(pokerbet_t *bet); */ void pokerBetPlayer(poker_t *poker, pokerplayer_t *player, int32_t chips); +/** + * Reset the current better back to the round/turns default. The better will + * always be the player to the right of the small blind player. + * + * @param poker Poker game to update for. + */ +void pokerBetResetBetter(poker_t *poker); + /** * Takes the current blinds from the correct players. * diff --git a/src/poker/turn.c b/src/poker/turn.c new file mode 100644 index 00000000..86e570b6 --- /dev/null +++ b/src/poker/turn.c @@ -0,0 +1,28 @@ +/** + * Copyright (c) 2021 Dominic Masters + * + * This software is released under the MIT License. + * https://opensource.org/licenses/MIT + */ + +#include "turn.h" + +pokerturn_t pokerTurnGet(poker_t *poker, uint8_t playerIndex) { + pokerturn_t turn; + pokerplayer_t *player; + + player = poker->players + playerIndex; + + // Can the player do anything? + turn.type = POKER_TURN_TYPE_OUT; + if(player->state & POKER_PLAYER_STATE_FOLDED) return turn; + if(player->state & POKER_PLAYER_STATE_OUT) return turn; + + + // I have nfi + turn.type = POKER_TURN_TYPE_BET; + turn.chips = 1; + turn.confidence = 1; + + return turn; +} \ No newline at end of file diff --git a/src/poker/turn.h b/src/poker/turn.h new file mode 100644 index 00000000..e8f5f68d --- /dev/null +++ b/src/poker/turn.h @@ -0,0 +1,11 @@ +/** + * Copyright (c) 2021 Dominic Masters + * + * This software is released under the MIT License. + * https://opensource.org/licenses/MIT + */ + +#pragma once +#include + +pokerturn_t pokerTurnGet(poker_t *poker, uint8_t playerIndex); \ No newline at end of file