#!/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, DefaultHint from pysollib.layout import Layout from pysollib.mygettext import _ from pysollib.pysoltk import MfxCanvasText from pysollib.stack import \ AC_FoundationStack, \ BasicRowStack, \ DealRowTalonStack, \ InitialDealTalonStack, \ InvisibleStack, \ RK_RowStack, \ SC_RowStack, \ SS_FoundationStack, \ SS_RowStack, \ StackWrapper, \ WasteStack, \ WasteTalonStack from pysollib.util import ACE, ANY_SUIT, BLACK, JACK, KING, QUEEN, RANKS, RED class GrandfathersClock_Hint(CautiousDefaultHint): # FIXME: demo is not too clever in this game def _getDropCardScore(self, score, color, r, t, ncards): # drop all cards immediately return 92000, color # ************************************************************************ # * Grandfather's Clock # ************************************************************************ class GrandfathersClock(Game): Hint_Class = GrandfathersClock_Hint # # game layout # def createGame(self): # create layout l, s = Layout(self), self.s # set window # (piles up to 9 cards are fully playable in default window size) dh = max(3*l.YS//2+l.CH, l.YS+(9-1)*l.YOFFSET) self.setSize(10*l.XS+l.XM, l.YM+2*dh) # create stacks for i in range(2): x, y = l.XM, l.YM + i*dh for j in range(4): s.rows.append( RK_RowStack(x, y, self, max_move=1, max_accept=1)) x = x + l.XS y = l.YM + dh - l.CH // 2 self.setRegion(s.rows[:4], (-999, -999, x - l.XM // 2, y)) self.setRegion(s.rows[4:], (-999, y, x - l.XM // 2, 999999)) d = [(0, 0), (1, 0.15), (2, 0.5), (2.5, 1.5), (2, 2.5), (1, 2.85)] for i in range(len(d)): d.append((0 - d[i][0], 3 - d[i][1])) x0, y0 = l.XM, l.YM + dh - l.CH for i in range(12): j = (i + 5) % 12 x = int(round(x0 + (6.5+d[j][0]) * l.XS)) y = int(round(y0 + (-1.5+d[j][1]) * l.YS)) suit = (1, 2, 0, 3)[i % 4] s.foundations.append(SS_FoundationStack(x, y, self, suit, base_rank=i+1, mod=13, max_move=0)) s.talon = InitialDealTalonStack( self.width-l.XS, self.height-l.YS, self) # define stack-groups self.sg.openstacks = s.foundations + s.rows self.sg.talonstacks = [s.talon] self.sg.dropstacks = s.rows # # game overrides # def _shuffleHook(self, cards): # move clock cards to bottom of the Talon (i.e. last cards to be dealt) C, S, H, D = 0*13, 1*13, 2*13, 3*13 ids = (1+S, 2+H, 3+C, 4+D, 5+S, 6+H, 7+C, 8+D, 9+S, 10+H, 11+C, 12+D) clocks = [] for c in cards[:]: if c.id in ids: clocks.append(c) cards.remove(c) # sort clocks reverse by rank clocks.sort(key=lambda x: -x.rank) return clocks + cards def startGame(self): self.startDealSample() self.playSample("grandfathersclock", loop=1, priority=200) self._dealNumRows(4) self.s.talon.dealRow() self.s.talon.dealRow(rows=self.s.foundations) shallHighlightMatch = Game._shallHighlightMatch_RK def getHighlightPilesStacks(self): return () def getAutoStacks(self, event=None): # disable auto drop - this would ruin the whole gameplay return ((), (), ()) # ************************************************************************ # * Dial # ************************************************************************ class Dial(Game): def createGame(self): l, s = Layout(self), self.s self.setSize(l.XM+8*l.XS, l.YM+4*l.YS) x0, y0 = l.XM+2*l.XS, l.YM rank = 0 font = self.app.getFont("canvas_default") for xx, yy in ((3.5, 0.15), (4.5, 0.5), (5, 1.5), (4.5, 2.5), (3.5, 2.85), (2.5, 3), (1.5, 2.85), (0.5, 2.5), (0, 1.5), (0.5, 0.5), (1.5, 0.15), (2.5, 0), (2.5, 1.5), ): x = int(x0 + xx*l.XS) y = int(y0 + yy*l.YS) stack = AC_FoundationStack( x, y, self, suit=ANY_SUIT, dir=0, max_cards=4, base_rank=rank, max_move=0) stack.getBottomImage = stack._getReserveBottomImage s.foundations.append(stack) if self.preview <= 1: label = RANKS[rank][0] if label == "1": label = "10" stack.texts.misc = MfxCanvasText(self.canvas, x + l.CW // 2, y + l.CH // 2, anchor="center", font=font) stack.texts.misc.config(text=label) rank += 1 x, y = l.XM, l.YM s.talon = WasteTalonStack(x, y, self, max_rounds=2) l.createText(s.talon, 's') x += l.XS s.waste = WasteStack(x, y, self) l.createText(s.waste, 's') l.createRoundText(s.talon, 'ne', dx=l.XS) l.defaultStackGroups() def startGame(self): self.startDealSample() self.s.talon.dealCards() # deal first card to WasteStack # ************************************************************************ # * Hemispheres # ************************************************************************ class Hemispheres_Hint(DefaultHint): def shallMovePile(self, from_stack, to_stack, pile, rpile): if not self._defaultShallMovePile(from_stack, to_stack, pile, rpile): return False if from_stack in self.game.s.rows and to_stack in self.game.s.rows: # check for loops return len(from_stack.cards) == 1 return True class Hemispheres_RowStack(SC_RowStack): def _canSwapPair(self, from_stack): if from_stack not in self.game.s.rows[4:]: return False if len(self.cards) != 1 or len(from_stack.cards) != 1: return False if self in self.game.s.rows[4:10]: alt_rows = self.game.s.rows[10:] color = RED else: alt_rows = self.game.s.rows[4:10] color = BLACK if from_stack not in alt_rows: return False c0, c1 = from_stack.cards[0], self.cards[0] return c0.color == color and c1.color != color def acceptsCards(self, from_stack, cards): if self._canSwapPair(from_stack): return True if not SC_RowStack.acceptsCards(self, from_stack, cards): return False if self in self.game.s.rows[4:10]: if cards[0].color == BLACK: return False return (from_stack in self.game.s.rows[4:10] or from_stack is self.game.s.waste) if self in self.game.s.rows[10:]: if cards[0].color == RED: return False return (from_stack in self.game.s.rows[10:] or from_stack is self.game.s.waste) return False def moveMove(self, ncards, to_stack, frames=-1, shadow=-1): if self._canSwapPair(to_stack): self._swapPairMove(ncards, to_stack, frames=-1, shadow=0) else: SC_RowStack.moveMove(self, ncards, to_stack, frames=frames, shadow=shadow) def _swapPairMove(self, n, other_stack, frames=-1, shadow=-1): game = self.game old_state = game.enterState(game.S_FILL) swap = game.s.internals[0] game.moveMove(n, self, swap, frames=0) game.moveMove(n, other_stack, self, frames=frames, shadow=shadow) game.moveMove(n, swap, other_stack, frames=0) game.leaveState(old_state) class Hemispheres(Game): Hint_Class = Hemispheres_Hint def createGame(self): l, s = Layout(self), self.s self.setSize(l.XM+9.5*l.XS, l.YM+5*l.YS) # internal stack (for swap) s.internals.append(InvisibleStack(self)) x0, y0 = l.XM+1.5*l.XS, l.YM # barriers for xx, yy in ((0, 2), (7, 2), (3.5, 0), (3.5, 4), ): x, y = x0+xx*l.XS, y0+yy*l.YS s.rows.append(BasicRowStack(x, y, self, max_accept=0)) # northern hemisphere (red) for xx, yy in ((0.5, 1), (1.5, 0.5), (2.5, 0.3), (4.5, 0.3), (5.5, 0.5), (6.5, 1), ): x, y = x0+xx*l.XS, y0+yy*l.YS stack = Hemispheres_RowStack(x, y, self, base_color=RED, max_move=1) stack.CARD_YOFFSET = 0 s.rows.append(stack) # southern hemisphere (black) for xx, yy in ((6.5, 3), (5.5, 3.5), (4.5, 3.8), (2.5, 3.8), (1.5, 3.5), (0.5, 3), ): x, y = x0+xx*l.XS, y0+yy*l.YS stack = Hemispheres_RowStack(x, y, self, base_color=BLACK, max_move=1, dir=1) stack.CARD_YOFFSET = 0 s.rows.append(stack) # foundations x, y = x0+2*l.XS, y0+1.5*l.YS for i in range(4): s.foundations.append(SS_FoundationStack(x, y, self, suit=2+i//2, max_move=0)) x += l.XS x, y = x0+2*l.XS, y+l.YS for i in range(4): s.foundations.append(SS_FoundationStack(x, y, self, suit=i//2, max_move=0, base_rank=KING, dir=-1)) x += l.XS # talon & waste x, y = l.XM, l.YM s.talon = WasteTalonStack(x, y, self, max_rounds=1) l.createText(s.talon, 'ne') y += l.YS s.waste = WasteStack(x, y, self) l.createText(s.waste, 'ne') l.defaultStackGroups() def _shuffleHook(self, cards): founds_cards = [] # foundations rows_cards = [] # rows for c in cards[:]: if c.rank in (ACE, KING): cond = ((c.rank == ACE and c.color == RED) or (c.rank == KING and c.color == BLACK)) if cond: cards.remove(c) founds_cards.append(c) elif c.deck == 0: cards.remove(c) rows_cards.append(c) founds_cards.sort(key=lambda x: (-x.rank, -x.suit)) rows_cards.sort(key=lambda x: (x.rank, x.suit)) return cards+rows_cards+founds_cards def startGame(self): self.s.talon.dealRow(rows=self.s.foundations, frames=0) self._startAndDealRowAndCards() def fillStack(self, stack): if stack in self.s.rows[4:] and not stack.cards: old_state = self.enterState(self.S_FILL) if not self.s.waste.cards: self.s.talon.dealCards() if self.s.waste.cards: self.s.waste.moveMove(1, stack) self.leaveState(old_state) def shallHighlightMatch(self, stack1, card1, stack2, card2): # by color return card1.color == card2.color and abs(card1.rank-card2.rank) == 1 # ************************************************************************ # * Big Ben # ************************************************************************ class BigBen_Talon(DealRowTalonStack): def dealCards(self, sound=False): if not self.cards: return 0 rows = [s for s in self.game.s.rows if len(s.cards) < 3] if not rows: # deal to the waste if sound and not self.game.demo: self.game.playSample("dealwaste") self.game.flipAndMoveMove(self, self.game.s.waste) return 1 # deal to the rows if sound and self.game.app.opt.animations: self.game.startDealSample() ncards = 0 while rows: n = self.dealRowAvail(rows=rows, sound=False) if not n: break ncards += n rows = [s for s in self.game.s.rows if len(s.cards) < 3] if sound: self.game.stopSamples() return ncards class BigBen_RowStack(SS_RowStack): def acceptsCards(self, from_stack, cards): if not SS_RowStack.acceptsCards(self, from_stack, cards): return False if len(self.cards) < 3: return False return True class BigBen(Game): Hint_Class = CautiousDefaultHint def createGame(self): l, s = Layout(self), self.s self.setSize(l.XM+12*l.XS, l.YM+5.5*l.YS) y = l.YM for i in range(2): x = l.XM for j in range(6): s.rows.append(BigBen_RowStack(x, y, self, max_move=1, mod=13)) x += l.XS y += 2.75*l.YS x0, y0 = l.XM+6*l.XS, l.YM rank = 1 for xx, yy in ( (0, 1.5), (0.5, 0.5), (1.5, 0.15), (2.5, 0), (3.5, 0.15), (4.5, 0.5), (5, 1.5), (4.5, 2.5), (3.5, 2.85), (2.5, 3), (1.5, 2.85), (0.5, 2.5), ): x = int(x0 + xx*l.XS) y = int(y0 + yy*l.YS) suit = (3, 0, 2, 1)[rank % 4] max_cards = rank <= 4 and 8 or 9 s.foundations.append(SS_FoundationStack(x, y, self, suit=suit, max_cards=max_cards, base_rank=rank, mod=13, max_move=0)) rank += 1 x, y = self.width-l.XS, self.height-l.YS s.talon = BigBen_Talon(x, y, self, max_rounds=1) l.createText(s.talon, 'n') x -= l.XS s.waste = WasteStack(x, y, self) l.createText(s.waste, 'n') l.defaultStackGroups() def _shuffleHook(self, cards): # move clock cards to top of the Talon (i.e. first cards to be dealt) C, S, H, D = list(range(4)) # suits t = [(1, C), (2, H), (3, S), (4, D), (5, C), (6, H), (7, S), (8, D), (9, C), (JACK, H), (QUEEN, S), (KING, D)] clocks = [] for c in cards[:]: if (c.rank, c.suit) in t: t.remove((c.rank, c.suit)) cards.remove(c) clocks.append(c) if not t: break # sort clocks reverse by rank clocks.sort(key=lambda x: -x.rank) return cards+clocks def startGame(self): self.startDealSample() self.s.talon.dealRow(rows=self.s.foundations, frames=4) for i in range(3): self.s.talon.dealRow(frames=4) def _autoDeal(self, sound=True): # don't deal a card to the waste if the waste is empty return 0 shallHighlightMatch = Game._shallHighlightMatch_SSW # ************************************************************************ # * Clock # * Relaxed Clock # ************************************************************************ class Clock_RowStack(RK_RowStack): getBottomImage = RK_RowStack._getReserveBottomImage def _numFaceDown(self): ncards = 0 for c in self.cards: if not c.face_up: ncards += 1 return ncards def clickHandler(self, event): game = self.game if game.s.rows[12].cards[-1].rank == KING \ and game.DrawsLeft > 0 and not self.cards[-1].face_up: old_state = game.enterState(game.S_FILL) game.flipMove(self) game.saveStateMove(2 | 16) game.playSample("move", priority=10) game.moveMove(1, self, game.s.rows[12]) game.DrawsLeft -= 1 game.saveStateMove(1 | 16) game.leaveState(old_state) game.finishMove() else: return RK_RowStack.clickHandler(self, event) def acceptsCards(self, from_stack, cards): return cards[0].rank == self.id or \ (self.cards[-1].face_up and self.cards[-1].rank == KING) def moveMove(self, ncards, to_stack, frames=-1, shadow=-1): self._swapPairMove(ncards, to_stack, frames=-1, shadow=0) def _swapPairMove(self, n, other_stack, frames=-1, shadow=-1): is_king = other_stack.cards[-1].rank == KING game = self.game old_state = game.enterState(game.S_FILL) swap = game.s.internals[0] ncards = other_stack._numFaceDown() for i in range(ncards): game.moveMove(n, other_stack, swap, frames=0) from pysollib.settings import TOOLKIT if TOOLKIT == 'kivy': game.moveMove(n, self, other_stack, frames=-1) else: game.moveMove(n, self, other_stack, frames=0) if ncards > 0: for i in range(ncards): game.moveMove(n, swap, other_stack, frames=0) game.flipMove(other_stack) game.moveMove(n, other_stack, self) if is_king: self._moveKingToBottom() game.leaveState(old_state) def _moveKingToBottom(self): # move king to bottom of stack game = self.game swap, swap2 = game.s.internals game.moveMove(1, self, swap2, frames=0) ncards = self._numFaceDown() for i in range(ncards): game.moveMove(1, self, swap, frames=0) game.moveMove(1, swap2, self, frames=0) for i in range(ncards): game.moveMove(1, swap, self, frames=0) if not self.cards[-1].face_up: game.flipMove(self) self._fillStack() def _fillStack(self): c = self.cards[-1] n = self._numFaceDown() if n == 0: return if c.face_up and c.rank == KING: self._moveKingToBottom() def canFlipCard(self): return False def getHelp(self): return _('Tableau. Build by same rank.') class Clock(Game): RowStack_Class = Clock_RowStack Talon_Class = InitialDealTalonStack HAS_WASTE = False Draws = 0 DrawsLeft = 0 def createGame(self): # create layout l, s = Layout(self), self.s # set window dx = l.XS + 3*l.XOFFSET w = max(5.25*dx + l.XS, 5.5*dx) if self.HAS_WASTE: w += (1.5 * l.XS) self.setSize(l.XM + w, l.YM + 4*l.YS) font = self.app.getFont("canvas_default") # create stacks row_rank = 0 for xx, yy in ( (3.25, 0.15), (4.25, 0.5), (4.5, 1.5), (4.25, 2.5), (3.25, 2.85), (2.25, 3), (1.25, 2.85), (0.25, 2.5), (0, 1.5), (0.25, 0.5), (1.25, 0.15), (2.25, 0), ): x = l.XM + xx*dx y = l.YM + yy*l.YS if self.HAS_WASTE: x += (2 * l.XS) stack = self.RowStack_Class(x, y, self, max_move=0, base_rank=row_rank) stack.CARD_XOFFSET, stack.CARD_YOFFSET = l.XOFFSET, 0 stack.SHRINK_FACTOR = 1 s.rows.append(stack) if self.preview <= 1: label = RANKS[row_rank][0] if label == "1": label = "10" stack.texts.misc = MfxCanvasText(self.canvas, x + l.CW // 2, y + l.CH // 2, anchor="center", font=font) stack.texts.misc.config(text=label) row_rank += 1 x, y = l.XM + 2.25*dx, l.YM + 1.5*l.YS if self.HAS_WASTE: x += (2 * l.XS) stack = self.RowStack_Class(x, y, self, max_move=1, base_rank=row_rank) stack.CARD_XOFFSET, stack.CARD_YOFFSET = l.XOFFSET, 0 stack.SHRINK_FACTOR = 1 s.rows.append(stack) if self.preview <= 1: stack.texts.misc = MfxCanvasText(self.canvas, x + l.CW // 2, y + l.CH // 2, anchor="center", font=font) stack.texts.misc.config(text=(RANKS[row_rank][0])) if self.HAS_WASTE: x, y = l.XM, l.YM s.talon = self.Talon_Class(x, y, self) l.createText(s.talon, 's') x += l.XS s.waste = WasteStack(x, y, self) l.createText(s.waste, 's') l.createRoundText(s.talon, 'ne', dx=l.XS) else: x, y = self.width - l.XS, self.height - l.YS s.talon = self.Talon_Class(x, y, self) # create an invisible stacks s.internals.append(InvisibleStack(self)) s.internals.append(InvisibleStack(self)) # default l.defaultStackGroups() def startGame(self): self.DrawsLeft = self.Draws for i in range(3): self.s.talon.dealRow(frames=0, flip=False) self.startDealSample() self.s.talon.dealRow(flip=False) self.flipMove(self.s.rows[-1]) self.s.rows[-1]._fillStack() def isGameWon(self): for r in self.s.rows: if len(r.cards) != 4 or not r.cards[-1].face_up: return False return True def _restoreGameHook(self, game): if self.Draws > 0: self.DrawsLeft = game.loadinfo.DrawsLeft def _loadGameHook(self, p): if self.Draws > 0: self.loadinfo.addattr(DrawsLeft=p.load()) def _saveGameHook(self, p): if self.Draws > 0: DrawsLeft = self.DrawsLeft p.dump(DrawsLeft) def setState(self, state): # restore saved vars (from undo/redo) self.DrawsLeft = state[0] for s in self.s.foundations: s.cap.DrawsLeft = state[0] break def getState(self): # save vars (for undo/redo) return [self.DrawsLeft] def getHighlightPilesStacks(self): return () def getAutoStacks(self, event=None): return (), (), () def getStuck(self): if self.DrawsLeft > 0: return True return Game.getStuck(self) class RelaxedClock(Clock): Draws = 1 # ************************************************************************ # * German Clock # ************************************************************************ class GermanClock_RowStack(AC_FoundationStack): getBottomImage = AC_FoundationStack._getReserveBottomImage def acceptsCards(self, from_stack, cards): num_cards = len(self.cards) for i in range(13): check_seq = self.game.s.rows[i].cards if len(check_seq) > num_cards: if check_seq[num_cards].suit != cards[0].suit: return False return AC_FoundationStack.acceptsCards(self, from_stack, cards) class GermanClock(Clock): RowStack_Class = StackWrapper(GermanClock_RowStack, dir=0, max_move=0, suit=ANY_SUIT) Talon_Class = StackWrapper(WasteTalonStack, max_rounds=2) HAS_WASTE = True def startGame(self): pass def isGameWon(self): for r in self.s.rows: if len(r.cards) < 4: return False return True def getAutoStacks(self, event=None): return Game.getAutoStacks(self, event) # register the game registerGame(GameInfo(261, GrandfathersClock, "Grandfather's Clock", GI.GT_1DECK_TYPE | GI.GT_OPEN, 1, 0, GI.SL_BALANCED)) registerGame(GameInfo(682, Dial, "Dial", GI.GT_1DECK_TYPE, 1, 1, GI.SL_LUCK)) registerGame(GameInfo(690, Hemispheres, "Hemispheres", GI.GT_2DECK_TYPE, 2, 0, GI.SL_BALANCED, altnames=("The Four Continents",))) registerGame(GameInfo(697, BigBen, "Big Ben", GI.GT_2DECK_TYPE, 2, 0, GI.SL_BALANCED, altnames=("Father Time"))) registerGame(GameInfo(737, Clock, "Clock", GI.GT_1DECK_TYPE | GI.GT_CHILDREN, 1, 0, GI.SL_LUCK, altnames=("Travellers", "Sundial"))) registerGame(GameInfo(827, GermanClock, "German Clock", GI.GT_1DECK_TYPE, 1, 1, GI.SL_MOSTLY_LUCK, altnames=("Die Uhr",))) registerGame(GameInfo(915, RelaxedClock, "Relaxed Clock", GI.GT_1DECK_TYPE | GI.GT_RELAXED | GI.GT_CHILDREN, 1, 0, GI.SL_LUCK, altnames=("Watch")))