Py-Hanabi/compress.py

92 lines
2.4 KiB
Python
Raw Normal View History

import json
from enum import Enum
from typing import List, Optional
BASE62 = "abcdefghijklmnopqrstuvwxyz0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ";
COLORS = 'rygbp'
class DeckCard():
def __init__(self, suitIndex: int, rank: int):
self.suitIndex: int = suitIndex
self.rank: int = rank
def __eq__(self, other):
return self.suitIndex == other.suitIndex and self.rank == other.rank
def __repr__(self):
return COLORS[self.suitIndex] + str(self.rank)
class ActionType(Enum):
Play = 0
Discard = 1
ColorClue = 2
RankClue = 3
EndGame = 4
class Action():
def __init__(self, type_: ActionType, target: int, value: Optional[int] = None):
self.type = type_
self.target = target
self.value = value
def compress_actions(actions: List[Action]) -> str:
minType = 0
maxType = 0
if len(actions) != 0:
minType = min(map(lambda a: a.type.value, actions))
maxType = max(map(lambda a: a.type.value, actions))
typeRange = maxType - minType + 1
def compress_action(action):
value = 0 if action.value is None else action.value
a = BASE62[typeRange * value + (action.type.value - minType)]
b = BASE62[action.target]
return a + b
out = str(minType) + str(maxType)
out += ''.join(map(compress_action, actions))
return out
def compress_deck(deck: List[DeckCard]) -> str:
assert(len(deck) != 0)
minRank = min(map(lambda c: c.rank, deck))
maxRank = max(map(lambda c: c.rank, deck))
rankRange = maxRank - minRank + 1
def compress_card(card):
return BASE62[rankRange * card.suitIndex + (card.rank - minRank)]
out = str(minRank) + str(maxRank)
out += ''.join(map(compress_card, deck))
return out
def decompress_deck(deck_str: str) -> List[DeckCard]:
assert(len(deck_str) >= 2)
minRank = int(deck_str[0])
maxRank = int(deck_str[1])
assert(maxRank >= minRank)
rankRange = maxRank - minRank + 1
def decompress_card(card_char):
index = BASE62.index(card_char)
suitIndex = index // rankRange
rank = index % rankRange + minRank
return DeckCard(suitIndex, rank)
return [decompress_card(c) for c in deck_str[2:]]
## test
deck = [DeckCard(0,1), DeckCard(2,4), DeckCard(4,5)]
c = compress_deck(deck)
l = decompress_deck(c)
print(deck, l)
f = [Action(ActionType.Discard, 2), Action(ActionType.Play, 3, 8)]
a = compress_actions(f)
print(a)