hanafuda/game.c

339 lines
9.2 KiB
C
Raw Normal View History

2025-02-02 18:13:19 -05:00
#include <string.h>
2025-02-02 18:30:45 -05:00
#include <stdio.h>
2025-02-02 18:13:19 -05:00
#include "game.h"
#include "card.h"
#include "teyaku.h"
#include "dekiyaku.h"
2025-02-04 18:47:56 -05:00
#include "field_multiplier.h"
2025-02-08 07:55:28 -05:00
#include "play.h"
2025-02-02 18:13:19 -05:00
Vector2 mouse_pos;
char teyaku_calculation[400];
void initialize_game(Game *g) {
2025-02-02 19:07:49 -05:00
g->deck.count = 0;
2025-02-10 17:27:02 -05:00
g->deck.position = (Vector2) { 800, 400 };
g->deck.display_type = HAND_DISPLAY_DECK;
2025-02-02 18:13:19 -05:00
g->should_close = false;
2025-02-03 20:07:22 -05:00
g->state = GAME_STATE_INITIALIZING;
2025-02-04 18:47:56 -05:00
g->field_multiplier = NULL;
g->player_teyaku.calculated = false;
g->left_teyaku.calculated = false;
g->right_teyaku.calculated = false;
2025-02-02 18:13:19 -05:00
for (int i = 0; i < 48; i++) {
CardType t = CHAFF;
RibbonType rt = RIBBON_NONE;
Month month = i / 4;
switch (i) {
case 0:
case 8:
case 28:
case 40:
case 44:
t = BRIGHT; break;
case 1:
case 5:
case 9:
t = RIBBON; rt = RIBBON_POETRY; break;
case 21:
case 33:
case 37:
t = RIBBON; rt = RIBBON_BLUE; break;
case 13:
case 17:
case 25:
case 42:
t = RIBBON; rt = RIBBON_PLAIN; break;
case 4:
case 12:
case 16:
case 20:
case 24:
case 29:
case 32:
case 36:
case 41:
t = ANIMAL; break;
}
2025-02-06 19:44:48 -05:00
g->cards[i] = (Card) { i, t, rt, month, { 800, 100 }, false };
2025-02-03 20:07:22 -05:00
g->cards[i].move.end_time = 0.;
g->cards[i].move.position = &g->cards[i].position;
2025-02-06 18:20:16 -05:00
g->cards[i].move.destination = (Vector2) { 800, 400 };
g->cards[i].order = i;
2025-02-02 18:30:45 -05:00
g->deck.cards[i] = &g->cards[i];
g->deck.count++;
2025-02-02 18:13:19 -05:00
}
2025-02-02 18:30:45 -05:00
shuffle_hand(&g->deck);
2025-02-02 19:07:49 -05:00
g->player_hand.count = 0;
2025-02-06 18:20:16 -05:00
g->player_hand.position = (Vector2) { 300, 600 };
2025-02-05 21:19:05 -05:00
g->player_hand.display_type = HAND_DISPLAY_ROW;
2025-02-02 19:07:49 -05:00
g->right_hand.count = 0;
2025-02-06 18:20:16 -05:00
g->right_hand.position = (Vector2) { 50, 125 };
2025-02-05 21:19:05 -05:00
g->right_hand.display_type = HAND_DISPLAY_ROW;
2025-02-02 19:07:49 -05:00
g->left_hand.count = 0;
2025-02-06 18:20:16 -05:00
g->left_hand.position = (Vector2) { 750, 125 };
2025-02-05 21:19:05 -05:00
g->left_hand.display_type = HAND_DISPLAY_ROW;
2025-02-04 18:47:56 -05:00
g->field.count = 0;
2025-02-06 18:20:16 -05:00
g->field.position = (Vector2) { 400, 300 };
2025-02-05 21:19:05 -05:00
g->field.display_type = HAND_DISPLAY_FIELD;
2025-02-08 08:15:43 -05:00
g->player_scored.count = 0;
g->player_scored.position = (Vector2) { 300, 750 };
g->player_scored.display_type = HAND_DISPLAY_ROW;
g->right_scored.count = 0;
g->right_scored.position = (Vector2) { 50, 25 };
g->right_scored.display_type = HAND_DISPLAY_ROW;
g->left_scored.count = 0;
g->left_scored.position = (Vector2) { 750, 25 };
g->left_scored.display_type = HAND_DISPLAY_ROW;
2025-02-02 19:07:49 -05:00
2025-02-02 18:13:19 -05:00
strcpy(teyaku_calculation, "");
Image cards_image = LoadImage("img/cards.png");
g->cards_texture = LoadTextureFromImage(cards_image);
UnloadImage(cards_image);
2025-02-03 20:07:22 -05:00
g->state = GAME_STATE_DEALING;
2025-02-02 18:13:19 -05:00
}
bool stale_calculation = true;
2025-02-02 19:07:49 -05:00
void handle_input(Game *g) {
2025-02-06 19:44:48 -05:00
switch (g->state) {
case GAME_STATE_PLAYER_CHOOSING_FROM_HAND:
if (IsMouseButtonPressed(0)) {
mouse_pos = GetMousePosition();
for (int i = 0; i < g->player_hand.count; i++) {
if (point_within_card(g->player_hand.cards[i], mouse_pos)) {
2025-02-07 19:07:17 -05:00
for (int j = 0; j < 48; j++) {
g->cards[j].selected = false;
}
g->player_hand.cards[i]->selected = true;
2025-02-06 19:44:48 -05:00
break;
}
2025-02-02 18:13:19 -05:00
}
}
2025-02-06 19:44:48 -05:00
break;
case GAME_STATE_PLAYER_CHOOSING_TARGET:
2025-02-07 19:07:17 -05:00
if (IsMouseButtonPressed(0)) {
mouse_pos = GetMousePosition();
for (int i = 0; i < g->player_hand.count; i++) {
if (point_within_card(g->player_hand.cards[i], mouse_pos)) {
if (g->player_hand.cards[i]->selected) {
g->player_hand.cards[i]->selected = false;
} else {
for (int j = 0; j < g->player_hand.count; j++) {
g->player_hand.cards[j]->selected = false;
}
g->player_hand.cards[i]->selected = true;
}
}
}
2025-02-08 07:55:28 -05:00
Card *selected_card = NULL;
for (int i = 0; i < g->player_hand.count; i++) {
if (g->player_hand.cards[i]->selected) {
selected_card = g->player_hand.cards[i];
break;
}
}
for (int i = 0; i < g->field.count; i++) {
if (point_within_card(g->field.cards[i], mouse_pos)) {
if (selected_card == NULL) printf("No card selected, whoops");
if (valid_play(&g->field, selected_card, g->field.cards[i])) {
2025-02-08 08:15:43 -05:00
g->current_play_from_hand = selected_card;
g->current_play_target = g->field.cards[i];
2025-02-08 07:55:28 -05:00
} else {
printf("Invalid\n");
}
2025-02-10 17:27:02 -05:00
break;
2025-02-08 07:55:28 -05:00
}
}
if (CheckCollisionPointRec(mouse_pos, next_card_position(&g->field))) {
if (valid_play(&g->field, selected_card, NULL)) {
2025-02-08 08:15:43 -05:00
g->current_play_from_hand = selected_card;
g->current_play_target = NULL;
2025-02-08 07:55:28 -05:00
} else {
printf("Invalid\n");
}
}
2025-02-07 19:07:17 -05:00
}
2025-02-06 19:44:48 -05:00
break;
default:
break;
2025-02-02 18:13:19 -05:00
}
}
2025-02-03 20:07:22 -05:00
void run_frame_dealing(Game *g) {
if (g->player_hand.count < 4) {
deal(&g->deck, &g->player_hand, 4, true);
} else if (g->left_hand.count < 4) {
deal(&g->deck, &g->left_hand, 4, false);
2025-02-05 21:19:05 -05:00
} else if (g->right_hand.count < 4) {
deal(&g->deck, &g->right_hand, 4, false);
2025-02-04 18:47:56 -05:00
} else if (g->field.count < 3) {
deal(&g->deck, &g->field, 3, true);
2025-02-03 20:07:22 -05:00
} else if (g->player_hand.count < 7) {
deal(&g->deck, &g->player_hand, 3, true);
} else if (g->left_hand.count < 7) {
deal(&g->deck, &g->left_hand, 3, false);
2025-02-05 21:19:05 -05:00
} else if (g->right_hand.count < 7) {
deal(&g->deck, &g->right_hand, 3, false);
2025-02-04 18:47:56 -05:00
} else if (g->field.count < 6) {
deal(&g->deck, &g->field, 3, true);
2025-02-03 20:07:22 -05:00
} else {
2025-02-04 18:47:56 -05:00
g->state = GAME_STATE_CALCULATING_FIELD_MULTIPLIER;
2025-02-03 20:07:22 -05:00
}
}
2025-02-04 18:47:56 -05:00
void run_frame_calculating_field_multiplier(Game *g) {
g->field_multiplier = calculate_field_multiplier(&g->field);
g->state = GAME_STATE_CALCULATING_TEYAKU;
}
void run_frame_calculating_teyaku(Game *g) {
calculate_teyaku(g->player_hand, &g->player_teyaku);
2025-02-07 19:07:17 -05:00
for (int i = 0; i < 48; i++) {
g->cards[i].selected = false;
}
2025-02-06 19:44:48 -05:00
g->state = GAME_STATE_PLAYER_CHOOSING_FROM_HAND;
}
void run_frame_player_choosing_from_hand(Game *g) {
2025-02-07 19:07:17 -05:00
for (int i = 0; i < g->player_hand.count; i++) {
if (g->player_hand.cards[i]->selected) {
g->state = GAME_STATE_PLAYER_CHOOSING_TARGET;
2025-02-08 08:15:43 -05:00
return;
2025-02-07 19:07:17 -05:00
}
}
2025-02-06 19:44:48 -05:00
}
2025-02-07 19:07:17 -05:00
2025-02-06 19:44:48 -05:00
void run_frame_player_choosing_target(Game *g) {
2025-02-07 19:07:17 -05:00
bool no_cards_selected = true;
for (int i = 0; i < g->player_hand.count; i++) {
if (g->player_hand.cards[i]->selected) {
no_cards_selected = false;
break;
}
}
if (no_cards_selected) {
g->state = GAME_STATE_PLAYER_CHOOSING_FROM_HAND;
2025-02-08 08:15:43 -05:00
return;
}
if (g->current_play_from_hand) {
if (g->current_play_target) {
g->current_play_from_hand->selected = false;
remove_from_hand(&g->player_hand, g->current_play_from_hand);
add_to_hand(&g->player_scored, g->current_play_from_hand);
remove_from_hand(&g->field, g->current_play_target);
add_to_hand(&g->player_scored, g->current_play_target);
g->current_play_from_hand = NULL;
g->current_play_target = NULL;
} else {
g->current_play_from_hand->selected = false;
remove_from_hand(&g->player_hand, g->current_play_from_hand);
add_to_hand(&g->field, g->current_play_from_hand);
g->current_play_from_hand = NULL;
}
2025-02-10 17:27:02 -05:00
g->state = GAME_STATE_PLAYER_FROM_DECK;
}
}
void run_frame_player_from_deck(Game *g) {
Card *top_card = g->deck.cards[g->deck.count - 1];
if (top_card->visible) {
Card *target = valid_target(top_card, &g->field);
if (target) {
remove_from_hand(&g->field, target);
add_to_hand(&g->player_scored, target);
remove_from_hand(&g->deck, top_card);
add_to_hand(&g->player_scored, top_card);
} else {
remove_from_hand(&g->deck, top_card);
add_to_hand(&g->field, top_card);
}
2025-02-08 08:15:43 -05:00
g->state = GAME_STATE_INITIALIZING;
2025-02-10 17:27:02 -05:00
} else {
remove_from_hand(&g->deck, top_card);
add_to_hand(&g->deck, top_card);
top_card->visible = true;
top_card->move.end_time = 0.3;
top_card->move.destination.x = top_card->move.destination.x + 100;
2025-02-07 19:07:17 -05:00
}
2025-02-04 18:47:56 -05:00
}
2025-02-03 20:07:22 -05:00
void run_frame(Game *g) {
2025-02-06 19:44:48 -05:00
handle_input(g);
2025-02-03 20:07:22 -05:00
float delta = GetFrameTime();
bool done_moving = true;
for (int i = 0; i < 48; i++) {
move_position(&g->cards[i].move, delta);
if (!card_done_moving(&g->cards[i])) done_moving = false;
}
if (!done_moving) return;
switch (g->state) {
case GAME_STATE_INITIALIZING:
return;
case GAME_STATE_DEALING:
run_frame_dealing(g);
break;
2025-02-04 18:47:56 -05:00
case GAME_STATE_CALCULATING_FIELD_MULTIPLIER:
run_frame_calculating_field_multiplier(g);
break;
case GAME_STATE_CALCULATING_TEYAKU:
run_frame_calculating_teyaku(g);
break;
2025-02-06 19:44:48 -05:00
case GAME_STATE_PLAYER_CHOOSING_FROM_HAND:
run_frame_player_choosing_from_hand(g);
break;
case GAME_STATE_PLAYER_CHOOSING_TARGET:
run_frame_player_choosing_target(g);
break;
2025-02-10 17:27:02 -05:00
case GAME_STATE_PLAYER_FROM_DECK:
run_frame_player_from_deck(g);
break;
2025-02-03 20:07:22 -05:00
}
}
2025-02-02 18:13:19 -05:00
void draw_frame(Game *g) {
BeginDrawing();
ClearBackground(RAYWHITE);
for (int i = 0; i < 48; i++) {
draw_card(&g->cards[i], &g->cards_texture);
}
2025-02-05 21:19:05 -05:00
if (g->field_multiplier) DrawText(g->field_multiplier->name, 60, 385, 40, BLACK);
2025-02-04 18:47:56 -05:00
if (g->player_teyaku.calculated) {
char s[200];
teyaku_to_string(&g->player_teyaku, s);
DrawText(s, 5, 25, 30, BLACK);
}
2025-02-06 19:44:48 -05:00
switch (g->state) {
case GAME_STATE_PLAYER_CHOOSING_FROM_HAND:
DrawText("Choose a card to play", 60, 485, 20, BLACK);
2025-02-06 19:44:48 -05:00
break;
case GAME_STATE_PLAYER_CHOOSING_TARGET:
DrawText("Choose a target on the field", 60, 485, 20, BLACK);
DrawRectangleRec(next_card_position(&g->field), BLUE);
2025-02-06 19:44:48 -05:00
break;
default:
break;
}
2025-02-02 18:13:19 -05:00
EndDrawing();
}
void run_until_closing(Game *g) {
while (!WindowShouldClose() && !g->should_close) {
run_frame(g);
draw_frame(g);
}
}