#!/usr/bin/env python # -*- mode: python; coding: utf-8; -*- # ---------------------------------------------------------------------------## # # Copyright (C) 1998-2003 Markus Franz Xaver Johannes Oberhumer # Copyright (C) 2003 Mt. Hood Playing Card Co. # Copyright (C) 2005-2009 Skomoroh # # This program 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. # # This program 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 this program. If not, see . # # ---------------------------------------------------------------------------## from pysollib.game import Game from pysollib.gamedb import GI, GameInfo, registerGame from pysollib.hint import CautiousDefaultHint from pysollib.layout import Layout from pysollib.stack import \ BasicRowStack, \ DealRowTalonStack, \ InitialDealTalonStack, \ RedealTalonStack, \ SS_FoundationStack, \ StackWrapper, \ TalonStack, \ UD_RK_RowStack, \ UD_SS_RowStack, \ WasteStack, \ WasteTalonStack from pysollib.util import ACE, JACK, KING, NO_RANK, QUEEN class StHelena_Talon(TalonStack): def canDealCards(self): if self.round == self.max_rounds: return False return not self.game.isGameWon() def dealCards(self, sound=False): # move all cards to the Talon and redeal lr = len(self.game.s.rows) num_cards = 0 if len(self.cards) > 0: num_cards = len(self.cards) self.game.startDealSample() for i in range(lr): k = min(lr, len(self.cards)) for j in range(k): self.game.flipAndMoveMove(self, self.game.s.rows[j], 4) self.game.stopSamples() return num_cards for r in self.game.s.rows[::-1]: for i in range(len(r.cards)): num_cards += 1 self.game.moveMove(1, r, self, frames=0) assert len(self.cards) == num_cards if num_cards == 0: # game already finished return 0 # redeal self.cards.reverse() self.game.nextRoundMove(self) self.game.startDealSample() for i in range(lr): k = min(lr, len(self.cards)) for j in range(k): self.game.moveMove(1, self, self.game.s.rows[j], frames=4) # done self.game.stopSamples() assert len(self.cards) == 0 return num_cards class StHelena_FoundationStack(SS_FoundationStack): def acceptsCards(self, from_stack, cards): if not SS_FoundationStack.acceptsCards(self, from_stack, cards): return False if self.game.s.talon.round == 1: if (self.cap.base_rank == KING and from_stack in self.game.s.rows[6:10:]): return False if (self.cap.base_rank == ACE and from_stack in self.game.s.rows[:4]): return False return True class StHelena(Game): Hint_Class = CautiousDefaultHint Talon_Class = StackWrapper(StHelena_Talon, max_rounds=3) Foundation_Class = StHelena_FoundationStack RowStack_Class = StackWrapper(UD_RK_RowStack, base_rank=NO_RANK) # # game layout # def createGame(self): # create layout l, s = Layout(self), self.s # set window w, h = 3*l.XM+6*l.XS, 3*l.YM+4*l.YS self.setSize(w, h) # create stacks lay = ( (2, 1, 1, 0), (2, 2, 1, 0), (2, 3, 1, 0), (2, 4, 1, 0), (3, 5, 2, 1), (3, 5, 2, 2), (2, 4, 3, 3), (2, 3, 3, 3), (2, 2, 3, 3), (2, 1, 3, 3), (1, 0, 2, 2), (1, 0, 2, 1), ) for xm, xs, ym, ys in lay: x, y = xm*l.XM+xs*l.XS, ym*l.YM+ys*l.YS stack = self.RowStack_Class(x, y, self, max_move=1, max_accept=1) stack.CARD_XOFFSET = stack.CARD_YOFFSET = 0 s.rows.append(stack) x, y = 2*l.XM+l.XS, 2*l.YM+l.YS for i in range(4): s.foundations.append(self.Foundation_Class(x, y, self, suit=i, base_rank=KING, dir=-1)) x = x + l.XS x, y = 2*l.XM+l.XS, 2*l.YM+2*l.YS for i in range(4): s.foundations.append(self.Foundation_Class(x, y, self, suit=i)) x = x + l.XS s.talon = self.Talon_Class(l.XM, l.YM, self) # default l.defaultAll() # # game overrides # def _shuffleHook(self, cards): return self._shuffleHookMoveToBottom( cards, lambda c: (c.deck == 0 and c.rank in (0, 12), (-c.rank, c.suit)), 8) def startGame(self): self._startDealNumRows(7) self.s.talon.dealRow() self.s.talon.dealRow(self.s.foundations) shallHighlightMatch = Game._shallHighlightMatch_RK # ************************************************************************ # * Box Kite # ************************************************************************ class BoxKite(StHelena): Talon_Class = InitialDealTalonStack Foundation_Class = SS_FoundationStack RowStack_Class = StackWrapper(UD_RK_RowStack, base_rank=NO_RANK, mod=13) shallHighlightMatch = Game._shallHighlightMatch_RKW # ************************************************************************ # * Louis # ************************************************************************ class Louis(StHelena): Foundation_Class = SS_FoundationStack RowStack_Class = StackWrapper(UD_SS_RowStack, base_rank=NO_RANK, mod=13) shallHighlightMatch = Game._shallHighlightMatch_RKW def startGame(self): self.startDealSample() self.s.talon.dealRow(self.s.foundations) self.s.talon.dealRow() def _shuffleHook(self, cards): return self._shuffleHookMoveToTop( cards, lambda c: (c.deck == 0 and c.rank in (0, 12), (-c.rank, c.suit)), 8) def fillStack(self, stack): if (self.s.talon.cards and stack in self.s.rows and len(stack.cards) == 0): self.s.talon.dealRow(rows=[stack]) # ************************************************************************ # * Les Quatre Coins # ************************************************************************ class LesQuatreCoins_RowStack(UD_RK_RowStack): def acceptsCards(self, from_stack, cards): if not UD_RK_RowStack.acceptsCards(self, from_stack, cards): return False return len(self.game.s.talon.cards) == 0 class LesQuatreCoins_Talon(RedealTalonStack): def canDealCards(self): if self.round == self.max_rounds: return len(self.cards) != 0 return not self.game.isGameWon() def dealCards(self, sound=False): if not self.cards: RedealTalonStack.redealCards(self, sound=False) if sound and not self.game.demo: self.game.startDealSample() rows = self.game.s.rows rows = rows[:1]+rows[4:8]+rows[2:3]+rows[1:2]+rows[8:]+rows[3:4] num_cards = self.dealRowAvail(rows=rows) if sound and not self.game.demo: self.game.stopSamples() return num_cards class LesQuatreCoins_Foundation(SS_FoundationStack): def acceptsCards(self, from_stack, cards): if not SS_FoundationStack.acceptsCards(self, from_stack, cards): return False if not self.cards: return True if self.game.s.talon.cards: if from_stack in self.game.s.rows[4:]: i = list(self.game.s.foundations).index(self) j = list(self.game.s.rows).index(from_stack) return i == j-4 return True class LesQuatreCoins(Game): Hint_Class = CautiousDefaultHint def createGame(self): l, s = Layout(self), self.s self.setSize(l.XM+7*l.XS, l.YM+5*l.YS) for i, j in ((0, 0), (5, 0), (0, 4), (5, 4)): x, y = l.XM+l.XS+i*l.XS, l.YM+j*l.YS stack = LesQuatreCoins_RowStack(x, y, self, max_move=1, base_rank=NO_RANK) s.rows.append(stack) stack.CARD_YOFFSET = 0 for x in (l.XM+2*l.XS, l.XM+5*l.XS): y = l.YM+l.YS/2 for j in range(4): stack = LesQuatreCoins_RowStack(x, y, self, max_move=1, base_rank=NO_RANK) s.rows.append(stack) stack.CARD_YOFFSET = 0 y += l.YS x, y = l.XM+3*l.XS, l.YM+l.YS/2 for i in range(4): s.foundations.append(LesQuatreCoins_Foundation(x, y, self, suit=i)) y += l.YS x, y = l.XM+4*l.XS, l.YM+l.YS/2 for i in range(4): s.foundations.append(LesQuatreCoins_Foundation(x, y, self, suit=i, base_rank=KING, dir=-1)) y += l.YS x, y = l.XM, l.YM+2*l.YS s.talon = LesQuatreCoins_Talon(x, y, self, max_rounds=3) l.createText(s.talon, 's') l.createRoundText(s.talon, 'nn') l.defaultStackGroups() def startGame(self): self.startDealSample() self.s.talon.dealCards() shallHighlightMatch = Game._shallHighlightMatch_RK # ************************************************************************ # * Regal Family # ************************************************************************ class RegalFamily_RowStack(UD_SS_RowStack): def acceptsCards(self, from_stack, cards): if not UD_SS_RowStack.acceptsCards(self, from_stack, cards): return False return len(self.game.s.talon.cards) == 0 class RegalFamily(Game): Hint_Class = CautiousDefaultHint def createGame(self): l, s = Layout(self), self.s self.setSize(l.XM+8*l.XS, l.YM+5*l.YS) for i, j in ((0, 0), (1, 0), (2, 0), (3, 0), (4, 0), (5, 0), (6, 0), (6, 1), (6, 2), (6, 3), (6, 4), (5, 4), (4, 4), (3, 4), (2, 4), (1, 4), (0, 4), (0, 3), (0, 2), (0, 1) ): x, y = l.XM+l.XS+i*l.XS, l.YM+j*l.YS stack = RegalFamily_RowStack(x, y, self, max_move=1, base_rank=NO_RANK) s.rows.append(stack) stack.CARD_YOFFSET = 0 x, y = l.XM+3*l.XS, l.YM+l.YS for i in range(3): s.foundations.append(SS_FoundationStack(x, y, self, suit=i, base_rank=9, mod=13, dir=-1)) s.foundations.append(SS_FoundationStack(x, y+2*l.YS, self, suit=i, base_rank=9, mod=13, dir=-1)) x += l.XS x, y = l.XM+3*l.XS, l.YM+2*l.YS s.foundations.append(SS_FoundationStack(x, y, self, suit=3, base_rank=ACE, mod=13)) x += 2*l.XS s.foundations.append(SS_FoundationStack(x, y, self, suit=3, base_rank=JACK, mod=13, dir=-1)) x, y = l.XM, l.YM+2*l.YS s.talon = DealRowTalonStack(x, y, self) l.createText(s.talon, 's') l.defaultStackGroups() def startGame(self): self._startAndDealRow() shallHighlightMatch = Game._shallHighlightMatch_SS # ************************************************************************ # * King's Audience # ************************************************************************ class KingsAudience_RowStack(BasicRowStack): def acceptsCards(self, from_stack, cards): if len(self.cards) == 0: return from_stack == self.game.s.waste c = self.cards[-1] if cards[0].suit != c.suit: return False if ((cards[0].rank == JACK and c.rank == ACE) or (cards[0].rank == ACE and c.rank == JACK)): return True if ((cards[0].rank == QUEEN and c.rank == KING) or (cards[0].rank == KING and c.rank == QUEEN)): return True return False class KingsAudience_Foundation(SS_FoundationStack): def acceptsCards(self, from_stack, cards): if len(self.cards) == 0: return False return SS_FoundationStack.acceptsCards(self, from_stack, cards) class KingsAudience(Game): Hint_Class = CautiousDefaultHint def createGame(self): # create layout l, s = Layout(self), self.s # set window w, h = 3 * l.XM + 6 * l.XS, 3 * l.YM + 6 * l.YS self.setSize(w, h) # create stacks lay = ( (2, 1, 1, 0), (2, 2, 1, 0), (2, 3, 1, 0), (2, 4, 1, 0), (3, 5, 2, 1), (3, 5, 2, 2), (3, 5, 2, 3), (3, 5, 2, 4), (2, 4, 3, 5), (2, 3, 3, 5), (2, 2, 3, 5), (2, 1, 3, 5), (1, 0, 2, 1), (1, 0, 2, 2), (1, 0, 2, 3), (1, 0, 2, 4), ) for xm, xs, ym, ys in lay: x, y = xm * l.XM + xs * l.XS, ym * l.YM + ys * l.YS stack = KingsAudience_RowStack(x, y, self, max_move=1, max_accept=1) stack.CARD_XOFFSET = stack.CARD_YOFFSET = 0 s.rows.append(stack) x, y = 2 * l.XM + l.XS, 2 * l.YM + l.YS for i in range(4): s.foundations.append(KingsAudience_Foundation(x, y, self, suit=i, max_cards=2)) x = x + l.XS x, y = 2 * l.XM + l.XS, 2 * l.YM + 4 * l.YS for i in range(4): s.foundations.append(KingsAudience_Foundation(x, y, self, suit=i, max_cards=11, dir=-1)) x = x + l.XS tx = (2 * (l.XM + l.XS)) ty = (2 * (l.YM + l.YS)) s.talon = WasteTalonStack(tx, ty, self, max_rounds=1) l.createText(s.talon, "s") s.waste = WasteStack(tx + l.XS, ty, self) l.createText(s.waste, "s") # define stack-groups l.defaultStackGroups() def fillStack(self, stack): for s in self.s.rows: if len(s.cards) > 1: if not self.demo: self.playSample("droppair", priority=200) old_state = self.enterState(self.S_FILL) if s.cards[0].rank in (KING, QUEEN): s.moveMove(2, self.s.foundations[s.cards[0].suit]) elif s.cards[0].rank == ACE: s.moveMove(2, self.s.foundations[s.cards[0].suit + 4]) elif s.cards[0].rank == JACK: s.moveMove(1, self.s.foundations[s.cards[0].suit + 4]) s.moveMove(1, self.s.foundations[s.cards[0].suit + 4]) self.leaveState(old_state) if len(s.cards) == 0: if len(self.s.waste.cards) == 0: self.s.talon.dealCards() if len(self.s.waste.cards) > 0: self.s.waste.moveMove(1, s) if len(stack.cards) == 0: if stack is self.s.waste and self.s.talon.cards: self.s.talon.dealCards() elif stack in self.s.rows and self.s.waste.cards: self.s.waste.moveMove(1, stack) def startGame(self): self._startAndDealRow() self.s.talon.dealCards() shallHighlightMatch = Game._shallHighlightMatch_SS # register the game registerGame(GameInfo(302, StHelena, "St. Helena", GI.GT_2DECK_TYPE, 2, 2, GI.SL_BALANCED, altnames=("Napoleon's Favorite", "Washington's Favorite"))) registerGame(GameInfo(408, BoxKite, "Box Kite", GI.GT_2DECK_TYPE, 2, 0, GI.SL_BALANCED)) registerGame(GameInfo(620, LesQuatreCoins, "Les Quatre Coins", GI.GT_2DECK_TYPE, 2, 2, GI.SL_MOSTLY_SKILL, altnames=("Four Corners", "Cornerstones", "Corner Patience"))) registerGame(GameInfo(621, RegalFamily, "Regal Family", GI.GT_2DECK_TYPE, 2, 0, GI.SL_BALANCED)) registerGame(GameInfo(859, KingsAudience, "King's Audience", GI.GT_1DECK_TYPE, 1, 0, GI.SL_MOSTLY_LUCK, altnames=("Queen's Audience"))) registerGame(GameInfo(975, Louis, "Louis", GI.GT_2DECK_TYPE, 2, 2, GI.SL_BALANCED))