// Copyright 2025 Bill Rossi // // This file is part of Hanafuda Hachi-Hachi. // // Hanafuda Hachi-Hachi is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. // // Hanafuda Hachi-Hachi is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. // // You should have received a copy of the GNU General Public License along with Hanafuda Hachi-Hachi. If not, see . #include #include "teyaku.h" #include "card.h" SetTeyaku calculate_set_teyaku(const Hand h) { int month_counts[12] = { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 }; int month_stands[12] = { 0, 0, 0, 1, 1, 0, 1, 0, 0, 0, 0, 1 }; int pawlonia_chaff = 0; for (int i = 0; i < h.count; i++) { Card c = *h.cards[i]; month_counts[c.month]++; if (c.index >= 45) pawlonia_chaff++; } int triplets = 0, standing_triplets = 0, pairs = 0, four_of_a_kind = 0; for (int i = 0; i < 12; i++) { if (month_counts[i] == 4) { four_of_a_kind++; } else if (month_counts[i] == 3) { if (month_stands[i] && (i != 12 || pawlonia_chaff == 3)) { standing_triplets++; } else { triplets++; } } else if (month_counts[i] == 2) { pairs++; } } int total_triplets = triplets + standing_triplets; if (four_of_a_kind && total_triplets) return FOUR_THREE; else if (four_of_a_kind && pairs) return ONE_TWO_FOUR; else if (standing_triplets == 2) return TWO_STANDING_TRIPLETS; else if (total_triplets && pairs == 2) return TRIPLET_AND_TWO_PAIRS; else if (triplets == 1 && standing_triplets == 1) return TRIPLET_AND_STANDING_TRIPLET; else if (four_of_a_kind) return FOUR_OF_A_KIND; else if (triplets == 2) return TWO_TRIPLETS; else if (pairs == 3) return THREE_PAIRS; else if (standing_triplets) return STANDING_TRIPLET; else if (triplets) return TRIPLET; return SET_TEYAKU_NONE; } ChaffTeyaku calculate_chaff_teyaku(const Hand h) { int ribbons = 0; int animals = 0; int brights = 0; int chaff = 0; for (int i = 0; i < h.count; i++) { Card c = *h.cards[i]; if (c.month == NOVEMBER) chaff++; // November cards are all counted as chaff here else if (c.type == BRIGHT) brights++; else if (c.type == RIBBON) ribbons++; else if (c.type == ANIMAL) animals++; else chaff++; } if (chaff == 7) return EMPTY_HAND; else if (chaff == 6 && brights == 1) return ONE_BRIGHT; else if (chaff == 6 && animals == 1) return ONE_ANIMAL; else if (chaff == 6 && ribbons == 1) return ONE_RIBBON; else if (ribbons >= 2 && chaff == 7 - ribbons) return CHAFF_TEYAKU_RED; return CHAFF_TEYAKU_NONE; } static int set_teyaku_points_array[11] = { 0, 2, 3, 6, 7, 8, 4, 6, 7, 8, 20 }; static char *set_teyaku_english_array[11] = { "None", "Triplet", "Standing Triplet", "Two Triplets", "Triplet and Standing Triplet", "Two Standing Triplets", "Three Pairs", "Four of a Kind", "Triplet and Two Pairs", "One-Two-Four", "Four-Three" }; static int chaff_teyaku_points_array[6] = { 0, 2, 3, 3, 4, 4 }; static char *chaff_teyaku_english_array[6] = { "None", "Red", "One Ribbon", "One Animal", "One Bright", "Empty Hand" }; int set_teyaku_points(SetTeyaku st) { return set_teyaku_points_array[st]; } int chaff_teyaku_points(ChaffTeyaku ct) { return chaff_teyaku_points_array[ct]; } int teyaku_points(Teyaku *t) { return set_teyaku_points(t->set) + chaff_teyaku_points(t->chaff); } char *set_teyaku_english(SetTeyaku st) { return set_teyaku_english_array[st]; } char *chaff_teyaku_english(ChaffTeyaku ct) { return chaff_teyaku_english_array[ct]; } void calculate_teyaku(const Hand h, Teyaku *t) { t->chaff = calculate_chaff_teyaku(h); t->set = calculate_set_teyaku(h); t->calculated = true; } void set_teyaku_to_string(Teyaku *t, char *str) { sprintf(str, "Set: %s(%d)", set_teyaku_english(t->set), set_teyaku_points(t->set)); } void chaff_teyaku_to_string(Teyaku *t, char *str) { sprintf(str, "Chaff: %s(%d)", chaff_teyaku_english(t->chaff), chaff_teyaku_points(t->chaff)); } void teyaku_to_string(Teyaku *t, char *str) { int set_points = set_teyaku_points(t->set); int chaff_points = chaff_teyaku_points(t->chaff); // sprintf(str, "Set: %s(%d) / Chaff: %s(%d) / Total: %d", set_teyaku_english(t->set), set_points, chaff_teyaku_english(t->chaff), chaff_points, set_points + chaff_points); if (set_points + chaff_points > 0) sprintf(str, "Teyaku: %d", set_points + chaff_points); else sprintf(str, ""); }