1
0
Fork 0
mirror of https://github.com/shlomif/PySolFC.git synced 2025-04-05 00:02:29 -04:00
PySolFC/pysollib/games/grandfathersclock.py

786 lines
26 KiB
Python

#!/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 <http://www.gnu.org/licenses/>.
#
# ---------------------------------------------------------------------------##
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")))