2025-02-26 20:05:31 -05:00
// 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/>.
2025-02-01 08:28:52 -05:00
# include <stdio.h>
2025-02-01 07:04:28 -05:00
# include "teyaku.h"
2025-02-01 06:40:33 -05:00
# include "card.h"
2025-02-01 08:32:38 -05:00
SetTeyaku calculate_set_teyaku ( const Hand h ) {
2025-02-01 06:40:33 -05:00
int month_counts [ 12 ] = { 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 } ;
2025-02-01 07:04:28 -05:00
int month_stands [ 12 ] = { 0 , 0 , 0 , 1 , 1 , 0 , 1 , 0 , 0 , 0 , 0 , 1 } ;
2025-02-25 19:57:09 -05:00
int pawlonia_chaff = 0 ;
2025-02-01 06:40:33 -05:00
for ( int i = 0 ; i < h . count ; i + + ) {
2025-02-02 18:13:19 -05:00
Card c = * h . cards [ i ] ;
2025-02-01 06:40:33 -05:00
month_counts [ c . month ] + + ;
2025-02-25 19:57:09 -05:00
if ( c . index > = 45 ) pawlonia_chaff + + ;
2025-02-01 06:40:33 -05:00
}
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 ) {
2025-02-25 19:57:09 -05:00
if ( month_stands [ i ] & & ( i ! = 12 | | pawlonia_chaff = = 3 ) ) {
standing_triplets + + ;
} else {
triplets + + ;
}
2025-02-01 06:40:33 -05:00
} else if ( month_counts [ i ] = = 2 ) {
pairs + + ;
}
}
int total_triplets = triplets + standing_triplets ;
2025-02-01 07:04:28 -05:00
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 ;
2025-02-01 06:40:33 -05:00
}
2025-02-01 08:32:38 -05:00
ChaffTeyaku calculate_chaff_teyaku ( const Hand h ) {
2025-02-01 06:40:33 -05:00
int ribbons = 0 ;
int animals = 0 ;
int brights = 0 ;
int chaff = 0 ;
for ( int i = 0 ; i < h . count ; i + + ) {
2025-02-02 18:13:19 -05:00
Card c = * h . cards [ i ] ;
2025-02-01 06:40:33 -05:00
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 + + ;
}
2025-02-01 07:04:28 -05:00
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 " } ;
2025-02-01 11:23:01 -05:00
int set_teyaku_points ( SetTeyaku st ) {
return set_teyaku_points_array [ st ] ;
2025-02-01 07:04:28 -05:00
}
2025-02-01 11:23:01 -05:00
int chaff_teyaku_points ( ChaffTeyaku ct ) {
return chaff_teyaku_points_array [ ct ] ;
2025-02-01 07:04:28 -05:00
}
2025-02-23 05:42:31 -05:00
int teyaku_points ( Teyaku * t ) {
return set_teyaku_points ( t - > set ) + chaff_teyaku_points ( t - > chaff ) ;
}
2025-02-01 11:23:01 -05:00
char * set_teyaku_english ( SetTeyaku st ) {
return set_teyaku_english_array [ st ] ;
2025-02-01 07:04:28 -05:00
}
2025-02-01 11:23:01 -05:00
char * chaff_teyaku_english ( ChaffTeyaku ct ) {
return chaff_teyaku_english_array [ ct ] ;
2025-02-01 06:40:33 -05:00
}
2025-02-01 11:23:01 -05:00
void calculate_teyaku ( const Hand h , Teyaku * t ) {
t - > chaff = calculate_chaff_teyaku ( h ) ;
t - > set = calculate_set_teyaku ( h ) ;
2025-02-04 18:47:56 -05:00
t - > calculated = true ;
2025-02-01 11:23:01 -05:00
}
2025-02-23 05:42:31 -05:00
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 ) ) ;
}
2025-02-01 11:23:01 -05:00
void teyaku_to_string ( Teyaku * t , char * str ) {
int set_points = set_teyaku_points ( t - > set ) ;
int chaff_points = chaff_teyaku_points ( t - > chaff ) ;
2025-02-23 05:10:42 -05:00
// 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 , " " ) ;
2025-02-01 06:40:33 -05:00
}