diff --git a/card.c b/card.c index 148b44c..f2cf3d3 100644 --- a/card.c +++ b/card.c @@ -6,7 +6,7 @@ static Vector2 card_size = (Vector2) { CARD_WIDTH, CARD_HEIGHT }; static char *month_english_abbr[12] = { "Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec" }; Color BRICKRED = (Color) { 136, 12, 2, 255 }; -void draw_card(Card *c, Texture2D **cards_texture, int index) { +void draw_card(Card *c, Texture2D *cards_texture, int index) { int i_vert = c->index % 4; int i_horiz = c->index / 4; int pos_vert = i_vert * CARD_HEIGHT; @@ -15,8 +15,9 @@ void draw_card(Card *c, Texture2D **cards_texture, int index) { Color color = index == 1 ? BLACK : BRICKRED; if (c->visible) { + printf("draw card %p\n", cards_texture); fflush(stdout); DrawTexturePro( - *cards_texture[index ? 1 : 0], + *cards_texture, (Rectangle) { pos_horiz, pos_vert, CARD_WIDTH, CARD_HEIGHT }, (Rectangle) { c->position.x, c->position.y, card_size.x, card_size.y }, (Vector2) { 0, 0 }, diff --git a/card.h b/card.h index f57bfbf..23db92f 100644 --- a/card.h +++ b/card.h @@ -73,7 +73,7 @@ struct Hand { HandDisplayType display_type; }; -void draw_card(Card *c, Texture2D **cards_texture, int index); +void draw_card(Card *c, Texture2D *cards_texture, int index); bool point_within_card(Card *c, Vector2 v); void shuffle_hand(Hand *h); void deal(Hand *from, Hand *to, int count, bool up, float deal_speed); diff --git a/game.c b/game.c index 73ce60c..12b2ebf 100644 --- a/game.c +++ b/game.c @@ -15,15 +15,16 @@ Vector2 mouse_pos; char teyaku_calculation[400]; void initialize_game(Game *g) { - g->cards_texture = malloc(2 * sizeof(Texture)); Image cards_image_red = LoadImage("img/cards_red.png"); - g->cards_texture[0] = LoadTextureFromImage(cards_image_red); + g->cards_texture_red = LoadTextureFromImage(cards_image_red); UnloadImage(cards_image_red); Image cards_image_black = LoadImage("img/cards_black.png"); - g->cards_texture[1] = LoadTextureFromImage(cards_image_black); + g->cards_texture_black = LoadTextureFromImage(cards_image_black); UnloadImage(cards_image_black); + printf("%p %p\n", &g->cards_texture_black, &g->cards_texture_red); + g->deck.count = 0; g->deck.position = (Vector2) { 500, 300 }; g->deck.display_type = HAND_DISPLAY_DECK; @@ -37,6 +38,7 @@ void initialize_game(Game *g) { g->black_card_backs = true; g->deal_speed = 0.2; g->options = malloc(sizeof(Options)); + initialize_title(g); load_options_from_game(g); init_dialogs(g); @@ -76,14 +78,18 @@ void initialize_game(Game *g) { case 41: t = ANIMAL; break; } - g->cards[i] = (Card) { i, t, rt, month, { 500, 300 }, false }; + g->cards[i] = (Card) { i, t, rt, month, { -500, -300 }, false }; g->cards[i].move.end_time = 0.; g->cards[i].move.position = &g->cards[i].position; - g->cards[i].move.destination = (Vector2) { 500, 300 }; + g->cards[i].move.destination = (Vector2) { -500, -300 }; g->cards[i].order = i; g->cards[i].selected = false; + g->cards[i].visible = false; + g->deck.count++; } + printf("DONE CARDS\n"); + g->player.points = 100 * g->kan_value; g->right.points = 100 * g->kan_value; g->left.points = 100 * g->kan_value; @@ -148,7 +154,7 @@ void initialize_game(Game *g) { } g->current_round = 0; - g->state = GAME_STATE_OPTIONS; + g->state = GAME_STATE_TITLE_SCREEN; } Player *current_player(Game *g) { @@ -173,6 +179,10 @@ void handle_input(Game *g) { return options_handle_input(g); } + if (g->state == GAME_STATE_TITLE_SCREEN) { + return title_handle_input(g); + } + if (g->dialog) { return dialog_handle_input(g->dialog); } @@ -778,6 +788,7 @@ void run_frame(Game *g) { if (g->dialog) return; move_cards(g); + if (g->state == GAME_STATE_TITLE_SCREEN) run_frame_title(g); if (!done_moving(g)) return; switch (g->state) { @@ -835,18 +846,24 @@ void run_frame(Game *g) { case GAME_STATE_NEW_GAME: run_frame_new_game(g); break; + case GAME_STATE_TITLE_SCREEN: + break; case GAME_STATE_OPTIONS: break; } } +Texture *cards_texture_fun(Game *g) { + return g->black_card_backs ? &g->cards_texture_black : &g->cards_texture_red; +} + void draw_player_cards(Game *g, Player *p) { for (int i = 0; i < p->hand.count; i++) { - draw_card(p->hand.cards[i], &g->cards_texture, g->black_card_backs); + draw_card(p->hand.cards[i], cards_texture_fun(g), g->black_card_backs); } for (int i = 0; i < p->scored.count; i++) { - draw_card(p->scored.cards[i], &g->cards_texture, g->black_card_backs); + draw_card(p->scored.cards[i], cards_texture_fun(g), g->black_card_backs); } } @@ -855,11 +872,11 @@ void draw_cards(Game *g) { draw_player_cards(g, &g->right); draw_player_cards(g, &g->left); for (int i = 0; i < g->field.count; i++) { - draw_card(g->field.cards[i], &g->cards_texture, g->black_card_backs); + draw_card(g->field.cards[i], cards_texture_fun(g), g->black_card_backs); } for (int i = 0; i < g->deck.count; i++) { - draw_card(g->deck.cards[i], &g->cards_texture, g->black_card_backs); + draw_card(g->deck.cards[i], cards_texture_fun(g), g->black_card_backs); } } @@ -867,6 +884,12 @@ void draw_frame(Game *g) { BeginDrawing(); ClearBackground(RAYWHITE); + if (g->state == GAME_STATE_TITLE_SCREEN) { + title_draw(g); + EndDrawing(); + return; + } + draw_cards(g); if (g->state == GAME_STATE_OPTIONS) { diff --git a/game.h b/game.h index 44ccc15..8cf0fcf 100644 --- a/game.h +++ b/game.h @@ -13,6 +13,7 @@ typedef struct Game Game; #include "player.h" #include "dialog.h" #include "options.h" +#include "title.h" typedef enum GameState { GAME_STATE_INITIALIZING, @@ -41,7 +42,7 @@ struct Game { GameState state; bool should_close; Card cards[48]; - Texture2D *cards_texture; + Texture2D cards_texture_red, cards_texture_black; Hand deck, field; FieldMultiplier *field_multiplier; Card *current_play_from_hand, *current_play_target; @@ -56,6 +57,7 @@ struct Game { bool black_card_backs; float deal_speed; Options *options; + Title *title; }; void initialize_game(Game *g); diff --git a/img/rules_qr.png b/img/rules_qr.png new file mode 100644 index 0000000..df39fdc Binary files /dev/null and b/img/rules_qr.png differ diff --git a/main.c b/main.c index 69404e0..864cd2b 100644 --- a/main.c +++ b/main.c @@ -8,9 +8,6 @@ #include "game.h" -char *text = "こんにちわ、 世界!"; -Texture2D cards_texture; - int main(int argc, char** argv) { srand(time(NULL)); InitWindow(1400, 900, "Hanafuda Hachi-Hachi"); diff --git a/options.c b/options.c index 0cecc2a..4f0962c 100644 --- a/options.c +++ b/options.c @@ -29,12 +29,12 @@ void save_options_to_game(Game *g) { void handle_options_save(Game *g) { save_options_to_game(g); - g->state = GAME_STATE_INITIALIZING; + g->state = GAME_STATE_TITLE_SCREEN; } void handle_options_cancel(Game *g) { load_options_from_game(g); - g->state = GAME_STATE_INITIALIZING; + g->state = GAME_STATE_TITLE_SCREEN; } void handle_select_kan(Game *g, int index) { diff --git a/title.c b/title.c new file mode 100644 index 0000000..2d0fab8 --- /dev/null +++ b/title.c @@ -0,0 +1,163 @@ +#include +#include +#include "title.h" + +void initialize_title(Game *g) { + g->title = malloc(sizeof(Title)); + Image rules_qr_img = LoadImage("img/rules_qr.png"); + g->title->rules_qr = LoadTextureFromImage(rules_qr_img); + UnloadImage(rules_qr_img); +} + +void title_handle_click_start(Game *g) { + g->state = GAME_STATE_INITIALIZING; +} + +void title_handle_click_options(Game *g) { + g->state = GAME_STATE_OPTIONS; +} + +void title_handle_click_quit(Game *g) { + g->should_close = true; +} + +Vector2 tmp; // stands for "title mouse position" +void title_handle_input(Game *g) { + tmp = GetMousePosition(); + Title *t = g->title; + t->hover_start = false; + t->hover_options = false; + t->hover_quit = false; + t->hover_credits = false; + t->hover_rules = false; + + int half_start_width = MeasureText("Start", 60) / 2; + if (tmp.x > 700-half_start_width && tmp.x < 700+half_start_width && tmp.y > 350 && tmp.y < 410) { + t->hover_start = true; + if (IsMouseButtonPressed(0)) title_handle_click_start(g); + } + int half_options_width = MeasureText("Options", 60) / 2; + if (tmp.x > 700-half_options_width && tmp.x < 700+half_options_width && tmp.y > 500 && tmp.y < 560) { + t->hover_options = true; + if (IsMouseButtonPressed(0)) title_handle_click_options(g); + } + int half_quit_width = MeasureText("Quit", 60) / 2; + if (tmp.x > 700-half_quit_width && tmp.x < 700+half_quit_width && tmp.y > 650 && tmp.y < 710) { + t->hover_quit = true; + if (IsMouseButtonPressed(0)) title_handle_click_quit(g); + } + int half_credits_width = MeasureText("Credits", 40) / 2; + if (tmp.x > 1100-half_credits_width && tmp.x < 1100+half_credits_width && tmp.y > 600 && tmp.y < 640) t->hover_credits = true; + int half_rules_width = MeasureText("Rules", 40) / 2; + if (tmp.x > 300-half_rules_width && tmp.x < 300+half_rules_width && tmp.y > 600 && tmp.y < 640) t->hover_rules = true; +} + +void title_fly_card(Game *g, Card *card) { + int x, y; + switch ((int) rand() % 4) { + case 0: + x = (rand() % 1600) - 100; + y = -100; + break; + case 1: + x = (rand() % 1600) - 100; + y = 1000; + break; + case 2: + x = -100; + y = (rand() % 1100) - 100;; + break; + case 3: + x = 1500; + y = (rand() % 1100) - 100;; + break; + } + + card->position.x = x; + card->position.y = y; + Move *move = &card->move; + move->position->x = x; + move->position->y = y; + move->origin.x = x; + move->origin.y = y; + + + switch ((int) rand() % 4) { + case 0: + x = (rand() % 1600) - 100; + y = -100; + break; + case 1: + x = (rand() % 1600) - 100; + y = 1000; + break; + case 2: + x = -100; + y = (rand() % 1100) - 100;; + break; + case 3: + x = 1500; + y = (rand() % 1100) - 100;; + break; + } + + move->destination.x = x; + move->destination.y = y; + + move->curve = CURVE_LINEAR; + + move->current_time = 0.; + move->end_time = (rand() % 5) + 1; + printf("current: %f, end: %f\n", card->move.current_time, card->move.end_time); +} + +void run_frame_title(Game *g) { + if (rand() % 20 > 3) return; + + Card *card = NULL; + for (int i = 0; i < 48; i++) { + int index = rand() % 48; + if (g->cards[index].move.current_time > g->cards[index].move.end_time) { + card = &g->cards[index]; + break; + } + } + + if (!card) return; + + title_fly_card(g, card); +} + +Texture *cards_texture_fun2(Game *g) { + return g->black_card_backs ? &g->cards_texture_black : &g->cards_texture_red; +} + +void title_draw(Game *g) { + for (int i = 0; i < 48; i++) { + g->cards[i].visible = true; + draw_card(&g->cards[i], cards_texture_fun2(g), g->black_card_backs); + } + + Title *t = g->title; + DrawTextCentered("Hanafuda Hachi-Hachi", 700, 100, 90, BLACK); + DrawTextCentered("Start", 700, 350, 60, t->hover_start ? RED : BLACK); + DrawTextCentered("Options", 700, 500, 60, t->hover_options ? RED : BLACK); + DrawTextCentered("Quit", 700, 650, 60, t->hover_quit ? RED : BLACK); + DrawTextCentered("Credits", 1100, 600, 40, BLACK); + DrawTextCentered("Rules", 300, 600, 40, BLACK); + + if (t->hover_rules) DrawTextureEx(t->rules_qr, (Vector2) { 135, 250 }, 0., 3., WHITE); + + if (t->hover_credits) { + DrawRectangle(870, 200, 460, 380, BLACK); + DrawRectangle(873, 203, 454, 374, WHITE); + DrawTextCentered("Programmed by bassguitarbill", 1100, 210, 25, BLACK); + DrawTextCentered("https://bassguitarbill.rocks", 1100, 240, 25, BLACK); + DrawTextCentered("Running on raylib", 1100, 310, 25, BLACK); + DrawTextCentered("https://www.raylib.com", 1100, 340, 25, BLACK); + DrawTextCentered("Drawn with Aseprite", 1100, 410, 25, BLACK); + DrawTextCentered("https://www.aseprite.org", 1100, 440, 25, BLACK); + DrawTextCentered("Art adapted from Louiemantia", 1100, 510, 25, BLACK); + DrawTextCentered("https://commons.wikimedia.org/wiki/User:Louiemantia", 1100, 540, 18, BLACK); + } +} diff --git a/title.h b/title.h new file mode 100644 index 0000000..4930ec2 --- /dev/null +++ b/title.h @@ -0,0 +1,22 @@ +#ifndef _HF_TITLE_ +#define _HF_TITLE_ + +typedef struct Title Title; + +#include "game.h" + +struct Title { + bool hover_start; + bool hover_options; + bool hover_quit; + bool hover_credits; + bool hover_rules; + Texture2D rules_qr; +}; + +void title_handle_input(Game *g); +void title_draw(Game *g); +void initialize_title(Game *g); +void run_frame_title(Game *g); + +#endif