# 牌与牌型
import os
import time
import random


class PokerCard:
    SUIT_SYMBOLS = {
        'S': "♠",  # 黑桃 Spade
        'H': "♥",  # 红心 Heart
        'C': "♣",  # 梅花 Club
        'D': "♦",  # 方块 Diamond
        'Joker': "🃏"  # 王
    }
    SUIT_SYMBOLS_ZH = {
        'S': "黑桃",  # 黑桃 Spade
        'H': "红心",  # 红心 Heart
        'C': "梅花",  # 梅花 Club
        'D': "方块",  # 方块 Diamond
        'Joker': "王"  # 王
    }
    SUIT_SYMBOLS_ZH_R = {
        "黑桃": 'S',  # 黑桃 Spade
        "红心": 'H',  # 红心 Heart
        "梅花": 'C',  # 梅花 Club
        "方块": 'D',  # 方块 Diamond
        "王": 'Joker'  # 王
    }
    
    RANK_NAMES = {
        '3': '3', '4': '4', '5': '5', '6': '6', '7': '7', '8': '8', '9': '9',
        '10': '10', 'J': 'J', 'Q': 'Q', 'K': 'K', 'A': 'A', '2': '2',
        'small': '小', 'big': '大'
    }
    
    RANK_WEIGHTS = {
        '3': 3, '4': 4, '5': 5, '6': 6, '7': 7, '8': 8, '9': 9, '10': 10,
        'J': 11, 'Q': 12, 'K': 13, 'A': 14, '2': 15,
        'small': 16, 'big': 17
    }

    def __init__(self, suit: str, rank: str):
        """
        初始化一张扑克牌
        
        :param suit: 花色 ('S', 'H', 'C', 'D', 'Joker')
        :param rank: 点数 
         普通牌: '3'-'10','J','Q','K','A','2'
         王: 'small'(小王), 'big'(大王)
        """
        self.suit = suit
        self.rank = rank
        self.weight = self.RANK_WEIGHTS[rank]
        self.cards_path = os.path.join(os.path.dirname(os.path.abspath(__file__)), 'cards')
        self.image_file = f"{self.cards_path}/{self.suit}_{self.rank}.png"
        if self.suit == 'Joker':
            self.image_file = f"{self.cards_path}/Joker_{self.rank}.png"
        
    def __str__(self):
        """返回符号显示格式"""
        if self.suit == 'Joker':
            return f"{self.RANK_NAMES[self.rank]}🃏"
        return f"{self.SUIT_SYMBOLS[self.suit]}{self.RANK_NAMES[self.rank]}"
    
    def str_zh(self):
        """返回中文显示格式"""
        if self.suit == 'Joker':
            return f"{self.RANK_NAMES[self.rank]}王"
        return f"{self.SUIT_SYMBOLS_ZH[self.suit]}{self.RANK_NAMES[self.rank]}"
    
    def __repr__(self):
        return f"<PokerCard {self.suit}-{self.rank}>"
    
    def __eq__(self, other):
        """比较两张牌是否相同（花色和点数都相同）"""
        return self.suit == other.suit and self.rank == other.rank
    
    def __lt__(self, other):
        """比较两张牌的权重（用于排序）"""
        return self.weight < other.weight
    
    def __hash__(self):
        """使牌对象可哈希（可用于放入集合或作为字典键）"""
        return hash((self.suit, self.rank))
    
    @staticmethod
    def create_deck():
        """创建一副完整的斗地主扑克牌（54张）"""
        suits = ['S', 'H', 'C', 'D']
        ranks = ['3', '4', '5', '6', '7', '8', '9', '10', 'J', 'Q', 'K', 'A', '2']
        
        # 添加普通牌
        deck = [PokerCard(suit, rank) for suit in suits for rank in ranks]
        
        # 添加大小王
        deck.append(PokerCard('Joker', 'small'))
        deck.append(PokerCard('Joker', 'big'))
        
        return deck

        
class Hand:
    # 牌型常量
    """
    ROCKET = 'rocket'  # 火箭
    BOMB = 'bomb'  # 炸弹
    SINGLE = 'single'  # 单牌
    PAIR = 'pair'  # 对子
    TRIPLE = 'triple'  # 三张
    TRIPLE_WITH_SINGLE = 'triple_with_single'  # 三带单
    TRIPLE_WITH_PAIR = 'triple_with_pair'  # 三带对
    STRAIGHT = 'straight'  # 顺子
    CONSECUTIVE_PAIRS = 'consecutive_pairs'  # 连对
    AIRPLANE = 'airplane'  # 飞机
    AIRPLANE_WITH_SINGLES = 'airplane_with_singles'  # 飞机带单翅膀
    AIRPLANE_WITH_PAIRS = 'airplane_with_pairs'  # 飞机带对翅膀
    FOUR_WITH_TWO_SINGLES = 'four_with_two_singles'  # 四带二单
    FOUR_WITH_TWO_PAIRS = 'four_with_two_pairs'  # 四带二对
    INVALID = 'invalid'  # 无效牌型
    """
    ROCKET = '王炸'  # 火箭
    BOMB = '炸弹'  # 炸弹
    SINGLE = '单牌'  # 单牌
    PAIR = '对子'  # 对子
    TRIPLE = '三张'  # 三张
    TRIPLE_WITH_SINGLE = '三带单'  # 三带单
    TRIPLE_WITH_PAIR = '三带对'  # 三带对
    STRAIGHT = '顺子'  # 顺子
    CONSECUTIVE_PAIRS = '连对'  # 连对
    AIRPLANE = '飞机'  # 飞机
    AIRPLANE_WITH_SINGLES = '飞机带单翅膀'  # 飞机带单翅膀
    AIRPLANE_WITH_PAIRS = '飞机带对翅膀'  # 飞机带对翅膀
    FOUR_WITH_TWO_SINGLES = '四带二单'  # 四带二单
    FOUR_WITH_TWO_PAIRS = '四带二对'  # 四带二对
    INVALID = '无效'  # 无效牌型
    
    # 牌型等级
    TYPE_LEVELS = {
        ROCKET: 9,
        BOMB: 8,
        # 其他牌型等级为0
    }
    
    def __init__(self, cards):
        self.cards = sorted(cards)  # 按权重排序
        self.type = self.INVALID
        self.key_point = None  # 用于比较的关键点数
        self.level = 0  # 牌型等级
        self._detect_hand()  # 检测牌型
    
    def __repr__(self):
        return f"<Hand {[str(c) for c in self.cards]} | Type: {self.type} | Key: {self.key_point}>"
    
    def __iter__(self):
        return iter(self.cards)

    def _detect_hand(self):
        """检测牌型"""
        num_cards = len(self.cards)
        
        # 0张牌无效
        if num_cards == 0:
            return
        
        # 1. 火箭检测 (大王+小王)
        if num_cards == 2:
            if {c.rank for c in self.cards} == {'small', 'big'}:
                self.type = self.ROCKET
                self.key_point = 18  # 火箭特殊权重
                self.level = self.TYPE_LEVELS[self.ROCKET]
                return
        
        # 2. 炸弹检测 (四张相同点数)
        if num_cards == 4:
            ranks = {c.rank for c in self.cards}
            if len(ranks) == 1:
                self.type = self.BOMB
                self.key_point = self.cards[0].weight
                self.level = self.TYPE_LEVELS[self.BOMB]
                return
        
        # 3. 单牌检测
        if num_cards == 1:
            self.type = self.SINGLE
            self.key_point = self.cards[0].weight
            return
        
        # 4. 对子检测
        if num_cards == 2:
            if self.cards[0].rank == self.cards[1].rank:
                self.type = self.PAIR
                self.key_point = self.cards[0].weight
                return
        
        # 5. 三张检测
        if num_cards == 3:
            ranks = {c.rank for c in self.cards}
            if len(ranks) == 1:
                self.type = self.TRIPLE
                self.key_point = self.cards[0].weight
                return
        
        # 6. 三带单检测 (4张牌)
        if num_cards == 4:
            rank_count = {}
            for card in self.cards:
                rank_count[card.rank] = rank_count.get(card.rank, 0) + 1
            
            if sorted(rank_count.values()) == [1, 3]:
                for rank, count in rank_count.items():
                    if count == 3:
                        self.type = self.TRIPLE_WITH_SINGLE
                        self.key_point = PokerCard.RANK_WEIGHTS[rank]
                        return
        
        # 7. 三带对检测 (5张牌)
        if num_cards == 5:
            rank_count = {}
            for card in self.cards:
                rank_count[card.rank] = rank_count.get(card.rank, 0) + 1
            
            if sorted(rank_count.values()) == [2, 3]:
                for rank, count in rank_count.items():
                    if count == 3:
                        self.type = self.TRIPLE_WITH_PAIR
                        self.key_point = PokerCard.RANK_WEIGHTS[rank]
                        return
        
        # 8. 顺子检测 (5张或更多连续单牌) （不包含 2和双王，最大到A，最小到3，不分花色）
        if num_cards >= 5:
            # 排除2和王
            if any(card.rank in ['2', 'small', 'big'] for card in self.cards):
                pass
            else:
                weights = [c.weight for c in self.cards]
                # 检查是否连续且无重复
                if weights == list(range(weights[0], weights[0] + num_cards)):
                    self.type = self.STRAIGHT
                    self.key_point = weights[-1]  # 最大牌的点数
                    return
        
        # 9. 连对检测 (3对或更多连续对子) （不包含 2和双王，最大到A，最小到3，不分花色）
        if num_cards >= 6 and num_cards % 2 == 0:
            # 排除2和王
            if any(card.rank in ['2', 'small', 'big'] for card in self.cards):
                pass
            else:
                pairs = []
                valid = True
                # 检查是否全是对子
                for i in range(0, num_cards, 2):
                    if self.cards[i].rank != self.cards[i+1].rank:
                        valid = False
                        break
                    pairs.append(self.cards[i].rank)
                
                if valid:
                    weights = [PokerCard.RANK_WEIGHTS[r] for r in pairs]
                    # 检查对子是否连续
                    if weights == list(range(weights[0], weights[0] + len(weights))):
                        self.type = self.CONSECUTIVE_PAIRS
                        self.key_point = weights[-1]  # 最大对子的点数
                        return
        
        # 10. 飞机检测 (2个或更多连续三张)  不包括 2和双王，不分花色
        if num_cards >= 6 and num_cards % 3 == 0:
            triple_count = num_cards // 3
            # 排除2和王
            if any(card.rank in ['2', 'small', 'big'] for card in self.cards):
                pass
            else:
                triples = []
                valid = True
                # 检查是否全是三张
                for i in range(0, num_cards, 3):
                    ranks = {self.cards[i].rank, self.cards[i+1].rank, self.cards[i+2].rank}
                    if len(ranks) != 1:
                        valid = False
                        break
                    triples.append(self.cards[i].rank)
                
                if valid:
                    weights = [PokerCard.RANK_WEIGHTS[r] for r in triples]
                    # 检查三张是否连续
                    if weights == list(range(weights[0], weights[0] + len(weights))):
                        self.type = self.AIRPLANE
                        self.key_point = weights[-1]  # 最大三张的点数
                        return
        
        # 11. 飞机带单翅膀检测 # 所带牌中不包含双王、炸弹以及
        #与飞机连续的三张指的是例如 333444555 带 777是可以的，666那就只是飞机了 这个不用管
        if num_cards >= 8 and num_cards % 4 == 0:
            n = num_cards // 4 # 3牌和单牌成对，单牌可相同
            # 排除2和王
            if any(card.rank in ['small', 'big'] for card in self.cards):
                pass
            else:
                rank_count = {}
                for card in self.cards:
                    rank_count[card.rank] = rank_count.get(card.rank, 0) + 1

                bombs = [rank for rank, count in rank_count.items() if count == 4]
                # 找出三张的牌
                triples = [rank for rank, count in rank_count.items() if count == 3]
                others = [rank for rank, count in rank_count.items() if count == 1] + [rank for rank, count in rank_count.items() if count == 2] * 2
                if '2' in triples or bombs: # 飞机本身不包括 2
                    pass
                else:
                    # 检查三张是否连续
                    if len(triples) >= n:
                        # 按权重排序
                        triples.sort(key=lambda r: PokerCard.RANK_WEIGHTS[r])
                        # 取连续的n个三张
                        for i in range(len(triples) - n + 1):
                            sub_triples = triples[i:i+n]
                            weights = [PokerCard.RANK_WEIGHTS[r] for r in sub_triples]
                            if weights == list(range(weights[0], weights[0] + n)):
                                # 检查翅膀数量是否匹配
                                if len(triples) > n and n == 3: # 多出来一组3牌
                                    self.type = self.AIRPLANE_WITH_SINGLES
                                    self.key_point = weights[-1]
                                    return
                                elif len(others) >= n:
                                    self.type = self.AIRPLANE_WITH_SINGLES
                                    self.key_point = weights[-1]
                                    return
        
        # 12. 飞机带对翅膀检测  所带牌中不包含双王、炸弹以及与飞机连续的三张
        if num_cards >= 10 and num_cards % 5 == 0:
            n = num_cards // 5 # 3牌和双牌成对，双牌不可相同
            # 排除2和王
            if any(card.rank in ['small', 'big'] for card in self.cards):
                pass
            else:
                rank_count = {}
                for card in self.cards:
                    rank_count[card.rank] = rank_count.get(card.rank, 0) + 1

                bombs = [rank for rank, count in rank_count.items() if count == 4]
                # 找出三张和对子
                triples = [rank for rank, count in rank_count.items() if count == 3]
                pairs = [rank for rank, count in rank_count.items() if count == 2]
                if '2' in triples or bombs: # 飞机本身不包括 2
                    pass
                else:
                    # 检查三张是否连续
                    if len(triples) >= n:
                        triples.sort(key=lambda r: PokerCard.RANK_WEIGHTS[r])
                        for i in range(len(triples) - n + 1):
                            sub_triples = triples[i:i+n]
                            weights = [PokerCard.RANK_WEIGHTS[r] for r in sub_triples]
                            if weights == list(range(weights[0], weights[0] + n)):
                                # 检查对子数量是否匹配
                                if len(pairs) >= n:
                                    self.type = self.AIRPLANE_WITH_PAIRS
                                    self.key_point = weights[-1]
                                    return
        
        # 13. 四带二单检测 （所带牌中不包括双王及炸弹）
        if num_cards == 6:
            if any(card.rank in ['small', 'big'] for card in self.cards):
                pass
            else:
                rank_count = {}
                for card in self.cards:
                    rank_count[card.rank] = rank_count.get(card.rank, 0) + 1
                
                # 找出四张和单张
                fours = [rank for rank, count in rank_count.items() if count == 4]
                singles = [rank for rank, count in rank_count.items() if count == 1] + [rank for rank, count in rank_count.items() if count == 2]*2
                
                if len(fours) == 1 and len(singles) == 2:
                    self.type = self.FOUR_WITH_TWO_SINGLES
                    self.key_point = PokerCard.RANK_WEIGHTS[fours[0]]
                    return
        
        # 14. 四带二对检测 （所带牌中不包括双王及炸弹）
        if num_cards == 8:
            if any(card.rank in ['small', 'big'] for card in self.cards):
                pass
            else:
                rank_count = {}
                for card in self.cards:
                    rank_count[card.rank] = rank_count.get(card.rank, 0) + 1
                
                # 找出四张和对子
                fours = [rank for rank, count in rank_count.items() if count == 4]
                pairs = [rank for rank, count in rank_count.items() if count == 2]
                
                if len(fours) == 1 and len(pairs) == 2:
                    self.type = self.FOUR_WITH_TWO_PAIRS
                    self.key_point = PokerCard.RANK_WEIGHTS[fours[0]]
                    return
    
    def is_valid(self):
        """检查牌型是否有效"""
        return self.type != self.INVALID
    
    def __gt__(self, other):
        """比较两手牌的大小"""
        if not self.is_valid() or not other.is_valid():
            return False
        
        # 火箭最大
        if self.type == self.ROCKET:
            return other.type != self.ROCKET
        
        # 炸弹可以炸除火箭外的所有牌
        if self.type == self.BOMB:
            if other.type == self.ROCKET:
                return False
            if other.type == self.BOMB:
                return self.key_point > other.key_point
            return True
        
        # 其他牌比较
        if other.type == self.ROCKET or other.type == self.BOMB:
            return False
        
        # 同类型同数量才能比较
        if self.type == other.type and len(self.cards) == len(other.cards):
            return self.key_point > other.key_point
        
        return False
    
    def __eq__(self, other):
        """比较两手牌是否相等（仅用于测试）"""
        if not self.is_valid() or not other.is_valid():
            return False
        return self.type == other.type and self.key_point == other.key_point and len(self.cards) == len(other.cards)
    
    @staticmethod
    def create_hand(cards):
        """创建手牌对象（方便测试）"""
        return Hand(cards)

