hanafuda/teyaku.c
2025-02-26 20:05:31 -05:00

125 lines
4.7 KiB
C

// 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 <https://www.gnu.org/licenses/>.
#include <stdio.h>
#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, "");
}