mirror of
https://github.com/shlomif/PySolFC.git
synced 2025-04-05 00:02:29 -04:00
758 lines
26 KiB
Python
758 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 AbstractHint
|
|
from pysollib.layout import Layout
|
|
from pysollib.stack import \
|
|
BasicRowStack, \
|
|
DealRowTalonStack, \
|
|
InvisibleStack, \
|
|
RK_FoundationStack, \
|
|
SS_FoundationStack, \
|
|
SS_RowStack, \
|
|
Stack, \
|
|
StackWrapper, \
|
|
WasteStack, \
|
|
WasteTalonStack
|
|
from pysollib.util import ACE, KING, QUEEN
|
|
|
|
# ************************************************************************
|
|
# *
|
|
# ************************************************************************
|
|
|
|
|
|
class PictureGallery_Hint(AbstractHint):
|
|
def computeHints(self):
|
|
game = self.game
|
|
|
|
# 1) try if we can drop a card (i.e. an Ace)
|
|
for r in game.sg.dropstacks:
|
|
t, n = r.canDropCards(game.s.foundations)
|
|
if t and n == 1:
|
|
c = r.getCard()
|
|
assert t is not r and c
|
|
assert c.rank == ACE
|
|
if r in game.s.tableaux:
|
|
base_score = 90000 + (4 - r.cap.base_rank)
|
|
else:
|
|
base_score = 90000
|
|
score = base_score + 100 * (self.K - c.rank)
|
|
self.addHint(score, 1, r, t)
|
|
|
|
# 2) try if we can move a card to the tableaux
|
|
if not self.hints:
|
|
for r in game.sg.dropstacks:
|
|
pile = r.getPile()
|
|
if not pile or len(pile) != 1:
|
|
continue
|
|
if r in game.s.tableaux:
|
|
rr = self.ClonedStack(r, stackcards=r.cards[:-1])
|
|
if rr.acceptsCards(None, pile):
|
|
# do not move a card that is already in correct place
|
|
continue
|
|
base_score = 80000 + (4 - r.cap.base_rank)
|
|
else:
|
|
base_score = 80000
|
|
# find a stack that would accept this card
|
|
for t in game.s.tableaux:
|
|
if t is not r and t.acceptsCards(r, pile):
|
|
score = base_score + 100 * (self.K - pile[0].rank)
|
|
self.addHint(score, 1, r, t)
|
|
break
|
|
|
|
# 3) Try if we can move a card from the tableaux
|
|
# to a row stack. This can only happen if there are
|
|
# no more cards to deal.
|
|
if not self.hints:
|
|
for r in game.s.tableaux:
|
|
pile = r.getPile()
|
|
if not pile or len(pile) != 1:
|
|
continue
|
|
rr = self.ClonedStack(r, stackcards=r.cards[:-1])
|
|
if rr.acceptsCards(None, pile):
|
|
# do not move a card that is already in correct place
|
|
continue
|
|
# find a stack that would accept this card
|
|
for t in game.s.rows:
|
|
if t is not r and t.acceptsCards(r, pile):
|
|
score = 70000 + 100 * (self.K - pile[0].rank)
|
|
self.addHint(score, 1, r, t)
|
|
break
|
|
|
|
# 4) try if we can move a card within the row stacks
|
|
if not self.hints:
|
|
for r in game.s.rows:
|
|
pile = r.getPile()
|
|
if not pile:
|
|
continue
|
|
lp = len(pile)
|
|
lr = len(r.cards)
|
|
assert 1 <= lp <= lr
|
|
rpile = r.cards[:(lr-lp)] # remaining pile
|
|
if not pile or len(pile) != 1 or len(pile) == len(r.cards):
|
|
continue
|
|
base_score = 60000
|
|
# find a stack that would accept this card
|
|
for t in game.s.rows:
|
|
if self.shallMovePile(r, t, pile, rpile):
|
|
score = base_score + 100 * (self.K - pile[0].rank)
|
|
self.addHint(score, 1, r, t)
|
|
break
|
|
|
|
# 5) try if we can deal cards
|
|
if self.level >= 2:
|
|
if game.canDealCards():
|
|
self.addHint(self.SCORE_DEAL, 0, game.s.talon, None)
|
|
|
|
|
|
# ************************************************************************
|
|
# * Picture Gallery
|
|
# ************************************************************************
|
|
|
|
# this Foundation only accepts Aces
|
|
class PictureGallery_Foundation(RK_FoundationStack):
|
|
def __init__(self, x, y, game):
|
|
RK_FoundationStack.__init__(
|
|
self, x, y, game, base_rank=ACE, dir=0, max_move=0,
|
|
max_cards=(4 * game.gameinfo.decks))
|
|
self.CARD_YOFFSET = min(30, self.game.app.images.CARD_YOFFSET + 10)
|
|
|
|
def getBottomImage(self):
|
|
return self.game.app.images.getLetter(ACE)
|
|
|
|
def closeStack(self):
|
|
if len(self.cards) == (4 * self.game.gameinfo.decks):
|
|
if self.game.moves.state not in \
|
|
(self.game.S_REDO, self.game.S_RESTORE):
|
|
self.game.flipAllMove(self)
|
|
|
|
def canFlipCard(self):
|
|
return False
|
|
|
|
|
|
class PictureGallery_TableauStack(SS_RowStack):
|
|
max_accept = 1
|
|
|
|
def __init__(self, x, y, game, base_rank, yoffset, dir=3, max_cards=4):
|
|
SS_RowStack.__init__(
|
|
self, x, y, game,
|
|
base_rank=base_rank, dir=dir, max_cards=max_cards,
|
|
max_accept=self.max_accept)
|
|
self.CARD_YOFFSET = yoffset
|
|
|
|
def acceptsCards(self, from_stack, cards):
|
|
if not SS_RowStack.acceptsCards(self, from_stack, cards):
|
|
return False
|
|
# check that the base card is correct
|
|
if self.cards and self.cards[0].rank != self.cap.base_rank:
|
|
return False
|
|
return True
|
|
|
|
getBottomImage = Stack._getLetterImage
|
|
|
|
|
|
class PictureGallery_RowStack(BasicRowStack):
|
|
def acceptsCards(self, from_stack, cards):
|
|
if not BasicRowStack.acceptsCards(self, from_stack, cards):
|
|
return False
|
|
# check
|
|
if self.cards or self.game.s.talon.cards:
|
|
return False
|
|
return True
|
|
|
|
getBottomImage = Stack._getTalonBottomImage
|
|
|
|
|
|
# ************************************************************************
|
|
# *
|
|
# ************************************************************************
|
|
|
|
class PictureGallery(Game):
|
|
Hint_Class = PictureGallery_Hint
|
|
|
|
Foundation_Class = PictureGallery_Foundation
|
|
TableauStack_Classes = [
|
|
StackWrapper(
|
|
PictureGallery_TableauStack, base_rank=3, max_cards=4, dir=3),
|
|
StackWrapper(
|
|
PictureGallery_TableauStack, base_rank=2, max_cards=4, dir=3),
|
|
StackWrapper(
|
|
PictureGallery_TableauStack, base_rank=1, max_cards=4, dir=3),
|
|
]
|
|
RowStack_Class = StackWrapper(PictureGallery_RowStack, max_accept=1)
|
|
Talon_Class = DealRowTalonStack
|
|
|
|
NORMAL_OFFSET = False
|
|
|
|
#
|
|
# game layout
|
|
#
|
|
|
|
def createGame(self, waste=False, numstacks=8):
|
|
rows = len(self.TableauStack_Classes)
|
|
# create layout
|
|
l, s = Layout(self), self.s
|
|
numtableau = (4 * self.gameinfo.decks)
|
|
if not self.NORMAL_OFFSET:
|
|
TABLEAU_YOFFSET = min(numtableau + 1, max(3, l.YOFFSET // 3))
|
|
else:
|
|
TABLEAU_YOFFSET = l.YOFFSET
|
|
|
|
# set window
|
|
th = l.YS + ((numtableau + 4) // rows - 1) * TABLEAU_YOFFSET
|
|
if self.Foundation_Class is None and self.RowStack_Class is None:
|
|
h = 0
|
|
else:
|
|
# (set piles so at least 2/3 of a card is visible with 10 cards)
|
|
h = ((numtableau + 2) - 1) * l.YOFFSET + l.CH * 2 // 3
|
|
self.setSize((numtableau + 2) * l.XS + l.XM, l.YM + 3 * th + l.YM + h)
|
|
|
|
# create stacks
|
|
s.addattr(tableaux=[]) # register extra stack variable
|
|
x = l.XM + numtableau * l.XS + l.XS // 2
|
|
y = l.YM + l.CH // 2
|
|
if self.Foundation_Class is not None:
|
|
s.foundations.append(self.Foundation_Class(x, y, self))
|
|
y = l.YM
|
|
for cl in self.TableauStack_Classes:
|
|
x = l.XM
|
|
for j in range(numtableau):
|
|
s.tableaux.append(cl(x, y, self, yoffset=TABLEAU_YOFFSET))
|
|
x = x + l.XS
|
|
y = y + th
|
|
if self.Foundation_Class is not None:
|
|
self.setRegion(s.foundations, (x - l.CW // 2, -999, 999999,
|
|
y - l.CH))
|
|
x, y = l.XM, y + l.YM
|
|
if self.RowStack_Class is not None:
|
|
for i in range(numstacks):
|
|
s.rows.append(self.RowStack_Class(x, y, self))
|
|
x = x + l.XS
|
|
# self.setRegion(s.rows, (-999, -999, x - l.CW // 2, 999999))
|
|
x = l.XM + numstacks * l.XS + l.XS // 2
|
|
y = self.height - l.YS
|
|
if self.RowStack_Class is None and self.Foundation_Class is None:
|
|
y = l.YM + l.YS + l.CH // 2
|
|
s.talon = self.Talon_Class(x, y, self)
|
|
l.createText(s.talon, "se")
|
|
if waste:
|
|
y -= l.YS
|
|
s.waste = WasteStack(x, y, self)
|
|
l.createText(s.waste, "se")
|
|
|
|
# define stack-groups
|
|
if waste:
|
|
ws = [s.waste]
|
|
else:
|
|
ws = []
|
|
self.sg.openstacks = s.foundations + s.tableaux + s.rows + ws
|
|
self.sg.talonstacks = [s.talon] + ws
|
|
self.sg.dropstacks = s.tableaux + s.rows + ws
|
|
|
|
#
|
|
# game overrides
|
|
#
|
|
|
|
def startGame(self):
|
|
self.s.talon.dealRow(rows=self.s.tableaux, frames=0)
|
|
self._startAndDealRow()
|
|
|
|
def isGameWon(self):
|
|
if len(self.s.foundations[0].cards) != (4 * self.gameinfo.decks):
|
|
return False
|
|
for stack in self.s.tableaux:
|
|
if len(stack.cards) != 4:
|
|
return False
|
|
return True
|
|
|
|
def fillStack(self, stack):
|
|
if self.s.talon.cards:
|
|
if stack in self.s.rows and len(stack.cards) == 0:
|
|
self.s.talon.dealRow(rows=[stack])
|
|
|
|
def shallHighlightMatch(self, stack1, card1, stack2, card2):
|
|
if card1.rank == ACE or card2.rank == ACE:
|
|
return False
|
|
return (card1.suit == card2.suit and
|
|
(card1.rank + 3 == card2.rank or card2.rank + 3 == card1.rank))
|
|
|
|
def getHighlightPilesStacks(self):
|
|
return ()
|
|
|
|
|
|
class BigPictureGallery(PictureGallery):
|
|
|
|
def createGame(self):
|
|
PictureGallery.createGame(self, numstacks=12)
|
|
|
|
|
|
class HugePictureGallery(PictureGallery):
|
|
|
|
def createGame(self):
|
|
PictureGallery.createGame(self, numstacks=16)
|
|
|
|
|
|
# ************************************************************************
|
|
# * Great Wheel
|
|
# * Greater Wheel
|
|
# * Carousel
|
|
# ************************************************************************
|
|
|
|
|
|
class GreatWheel_Hint(PictureGallery_Hint):
|
|
shallMovePile = PictureGallery_Hint._cautiousShallMovePile
|
|
|
|
|
|
class GreatWheel_Foundation(PictureGallery_Foundation):
|
|
def acceptsCards(self, from_stack, cards):
|
|
if not PictureGallery_Foundation.acceptsCards(self, from_stack, cards):
|
|
return False
|
|
if self.cards and self.cards[-1].color == cards[0].color:
|
|
return False
|
|
return True
|
|
|
|
|
|
class GreatWheel_RowStack(BasicRowStack):
|
|
def acceptsCards(self, from_stack, cards):
|
|
if not BasicRowStack.acceptsCards(self, from_stack, cards):
|
|
return False
|
|
if self.game.s.talon.cards:
|
|
return False
|
|
if not self.cards:
|
|
return True
|
|
c1, c2 = self.cards[-1], cards[0]
|
|
return c1.suit == c2.suit and c1.rank == c2.rank+1
|
|
|
|
getBottomImage = Stack._getTalonBottomImage
|
|
|
|
|
|
class GreatWheel(PictureGallery):
|
|
|
|
Hint_Class = GreatWheel_Hint
|
|
Foundation_Class = GreatWheel_Foundation
|
|
TableauStack_Classes = [
|
|
StackWrapper(
|
|
PictureGallery_TableauStack, base_rank=2, max_cards=5, dir=2),
|
|
StackWrapper(
|
|
PictureGallery_TableauStack, base_rank=1, max_cards=6, dir=2),
|
|
]
|
|
RowStack_Class = StackWrapper(GreatWheel_RowStack, max_accept=1)
|
|
Talon_Class = StackWrapper(WasteTalonStack, max_rounds=1)
|
|
|
|
def createGame(self):
|
|
PictureGallery.createGame(self, waste=True, numstacks=8)
|
|
|
|
def fillStack(self, stack):
|
|
if stack is self.s.waste and not stack.cards:
|
|
self.s.talon.dealCards()
|
|
if self.s.talon.cards or self.s.waste.cards:
|
|
if stack in self.s.rows and len(stack.cards) == 0:
|
|
old_state = self.enterState(self.S_FILL)
|
|
for i in range(2 + self.gameinfo.decks):
|
|
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 startGame(self):
|
|
for i in range(1 + self.gameinfo.decks):
|
|
self.s.talon.dealRow(frames=0)
|
|
self.startDealSample()
|
|
self.s.talon.dealRow()
|
|
self.s.talon.dealCards()
|
|
|
|
def isGameWon(self):
|
|
if len(self.s.foundations[0].cards) != (4 * self.gameinfo.decks):
|
|
return False
|
|
if self.s.talon.cards or self.s.waste.cards:
|
|
return False
|
|
for stack in self.s.rows:
|
|
if stack.cards:
|
|
return False
|
|
return True
|
|
|
|
def shallHighlightMatch(self, stack1, card1, stack2, card2):
|
|
if card1.rank == ACE or card2.rank == ACE:
|
|
return False
|
|
return (card1.suit == card2.suit and
|
|
(card1.rank + 2 == card2.rank or card2.rank + 2 == card1.rank))
|
|
|
|
|
|
class GreaterWheel(GreatWheel):
|
|
|
|
def createGame(self):
|
|
PictureGallery.createGame(self, waste=True, numstacks=12)
|
|
|
|
|
|
class Carousel_RowStack(BasicRowStack):
|
|
def acceptsCards(self, from_stack, cards):
|
|
if not self.cards:
|
|
return True
|
|
c1, c2 = self.cards[-1], cards[0]
|
|
return c1.suit == c2.suit and c1.rank == c2.rank+1
|
|
|
|
getBottomImage = Stack._getTalonBottomImage
|
|
|
|
|
|
class Carousel(GreatWheel):
|
|
RowStack_Class = StackWrapper(Carousel_RowStack, max_accept=1)
|
|
|
|
# ************************************************************************
|
|
# * Mount Olympus
|
|
# * Zeus
|
|
# ************************************************************************
|
|
|
|
|
|
class MountOlympus_Foundation(SS_FoundationStack):
|
|
def getHelp(self):
|
|
return 'Build up in suit by twos.'
|
|
|
|
|
|
class MountOlympus_RowStack(SS_RowStack):
|
|
def getHelp(self):
|
|
return 'Build down in suit by twos.'
|
|
|
|
|
|
class MountOlympus(Game):
|
|
RowStack_Class = MountOlympus_RowStack
|
|
|
|
def createGame(self):
|
|
# create layout
|
|
l, s = Layout(self), self.s
|
|
|
|
# set window
|
|
self.setSize(l.XM+9*l.XS, l.YM+3*l.YS+12*l.YOFFSET)
|
|
|
|
# create stacks
|
|
x, y = l.XM+l.XS, l.YM
|
|
for i in range(8):
|
|
s.foundations.append(
|
|
MountOlympus_Foundation(
|
|
x, y, self,
|
|
suit=i//2, base_rank=ACE, dir=2, max_move=0, max_cards=7))
|
|
x += l.XS
|
|
x, y = l.XM+l.XS, l.YM+l.YS
|
|
for i in range(8):
|
|
s.foundations.append(
|
|
MountOlympus_Foundation(
|
|
x, y, self,
|
|
suit=i//2, base_rank=1, dir=2, max_move=0, max_cards=6))
|
|
x += l.XS
|
|
x, y = l.XM, l.YM+2*l.YS
|
|
for i in range(9):
|
|
s.rows.append(self.RowStack_Class(x, y, self, dir=-2))
|
|
x += l.XS
|
|
s.talon = DealRowTalonStack(l.XM, l.YM, self)
|
|
l.createText(s.talon, 's')
|
|
|
|
# define stack-groups
|
|
l.defaultStackGroups()
|
|
|
|
def _shuffleHook(self, cards):
|
|
return self._shuffleHookMoveToTop(
|
|
cards,
|
|
lambda c: (c.rank in (ACE, 1), (c.rank, c.suit)))
|
|
|
|
def startGame(self):
|
|
self.startDealSample()
|
|
self.s.talon.dealRow(rows=self.s.foundations)
|
|
self.s.talon.dealRow()
|
|
|
|
def fillStack(self, stack):
|
|
if self.s.talon.cards:
|
|
if stack in self.s.rows and len(stack.cards) == 0:
|
|
self.s.talon.dealRow(rows=[stack])
|
|
|
|
def shallHighlightMatch(self, stack1, card1, stack2, card2):
|
|
return (card1.suit == card2.suit and
|
|
(card1.rank + 2 == card2.rank or card2.rank + 2 == card1.rank))
|
|
|
|
|
|
class Zeus_RowStack(MountOlympus_RowStack):
|
|
def acceptsCards(self, from_stack, cards):
|
|
if not MountOlympus_RowStack.acceptsCards(self, from_stack, cards):
|
|
return False
|
|
if not self.cards:
|
|
return cards[0].rank in (QUEEN, KING)
|
|
return True
|
|
|
|
|
|
class Zeus(MountOlympus):
|
|
RowStack_Class = Zeus_RowStack
|
|
|
|
def startGame(self):
|
|
self.s.talon.dealRow(rows=self.s.foundations, frames=0)
|
|
self.startDealSample()
|
|
for i in range(4):
|
|
self.s.talon.dealRow()
|
|
|
|
|
|
# ************************************************************************
|
|
# * Royal Parade
|
|
# * Big Parade
|
|
# ************************************************************************
|
|
|
|
class RoyalParade_TableauStack(PictureGallery_TableauStack):
|
|
|
|
def _canSwapPair(self, from_stack):
|
|
if from_stack not in self.game.s.tableaux:
|
|
return False
|
|
if len(self.cards) != 1 or len(from_stack.cards) != 1:
|
|
return False
|
|
c0, c1 = from_stack.cards[0], self.cards[0]
|
|
return (c0.rank == self.cap.base_rank and
|
|
c1.rank == from_stack.cap.base_rank)
|
|
|
|
def acceptsCards(self, from_stack, cards):
|
|
if self._canSwapPair(from_stack):
|
|
return True
|
|
return PictureGallery_TableauStack.acceptsCards(
|
|
self, from_stack, cards)
|
|
|
|
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:
|
|
PictureGallery_TableauStack.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 RoyalParade(PictureGallery):
|
|
Talon_Class = DealRowTalonStack
|
|
TableauStack_Classes = [
|
|
StackWrapper(RoyalParade_TableauStack,
|
|
base_rank=1, max_cards=4, dir=3),
|
|
StackWrapper(RoyalParade_TableauStack,
|
|
base_rank=2, max_cards=4, dir=3),
|
|
StackWrapper(RoyalParade_TableauStack,
|
|
base_rank=3, max_cards=4, dir=3),
|
|
]
|
|
RowStack_Class = StackWrapper(BasicRowStack, max_accept=0)
|
|
|
|
def createGame(self, waste=False, numstacks=8):
|
|
PictureGallery.createGame(self, waste=waste, numstacks=numstacks)
|
|
self.s.internals.append(InvisibleStack(self))
|
|
|
|
def startGame(self):
|
|
self.startDealSample()
|
|
self.s.talon.dealRow(rows=self.s.tableaux, frames=0)
|
|
self.s.talon.dealRow()
|
|
|
|
|
|
class BigParade(RoyalParade):
|
|
|
|
def createGame(self):
|
|
RoyalParade.createGame(self, numstacks=12)
|
|
|
|
|
|
# ************************************************************************
|
|
# * Virginia Reel
|
|
# * Three Up
|
|
# * Blue Jacket
|
|
# ************************************************************************
|
|
|
|
class VirginiaReel_Talon(DealRowTalonStack):
|
|
|
|
def canDealCards(self):
|
|
if not DealRowTalonStack.canDealCards(self):
|
|
return False
|
|
for s in self.game.s.tableaux:
|
|
if not s.cards:
|
|
return False
|
|
return True
|
|
|
|
|
|
class VirginiaReel(RoyalParade):
|
|
Talon_Class = VirginiaReel_Talon
|
|
|
|
def _shuffleHook(self, cards):
|
|
bottom_cards = []
|
|
ranks = []
|
|
for c in cards[:]:
|
|
if c.rank in (1, 2, 3) and c.rank not in ranks:
|
|
ranks.append(c.rank)
|
|
cards.remove(c)
|
|
bottom_cards.append(c)
|
|
if len(ranks) == 3:
|
|
break
|
|
bottom_cards.sort(key=lambda x: -x.rank)
|
|
return cards+bottom_cards
|
|
|
|
def startGame(self):
|
|
numdeal = (4 * self.gameinfo.decks)
|
|
self.s.talon.dealRow(rows=self.s.tableaux[0::numdeal], frames=0)
|
|
self.startDealSample()
|
|
for i in range(3):
|
|
rows = self.s.tableaux[i*numdeal+1:i*numdeal+numdeal]
|
|
self.s.talon.dealRow(rows=rows, frames=0)
|
|
self.s.talon.dealRow()
|
|
|
|
def fillStack(self, stack):
|
|
pass
|
|
|
|
|
|
class ThreeUp(VirginiaReel):
|
|
|
|
def createGame(self):
|
|
VirginiaReel.createGame(self, numstacks=12)
|
|
|
|
|
|
class BlueJacket_RowStack(BasicRowStack):
|
|
def acceptsCards(self, from_stack, cards):
|
|
return len(self.cards) == 0 and len(cards) == 1
|
|
|
|
|
|
class BlueJacket(VirginiaReel):
|
|
RowStack_Class = StackWrapper(BlueJacket_RowStack, max_accept=1)
|
|
|
|
|
|
# ************************************************************************
|
|
# * Devil's Grip
|
|
# ************************************************************************
|
|
|
|
class DevilsGrip_TableauStack(RoyalParade_TableauStack):
|
|
max_accept = 4
|
|
|
|
def _canSwapPair(self, from_stack):
|
|
if from_stack not in self.game.s.tableaux:
|
|
return False
|
|
if len(self.cards) == 0 or len(from_stack.cards) == 0:
|
|
return False
|
|
if self.cap.base_rank == from_stack.cap.base_rank:
|
|
return False
|
|
c0, c1 = from_stack.cards[0], self.cards[0]
|
|
return (c0.rank != c1.rank and
|
|
(c0.rank == self.cap.base_rank or
|
|
c1.rank == from_stack.cap.base_rank))
|
|
|
|
def acceptsCards(self, from_stack, cards):
|
|
if self._canSwapPair(from_stack):
|
|
return True
|
|
return SS_RowStack.acceptsCards(
|
|
self, from_stack, cards)
|
|
|
|
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(len(self.cards), self, swap, frames=0)
|
|
game.moveMove(len(other_stack.cards), other_stack, self,
|
|
frames=frames, shadow=shadow)
|
|
game.moveMove(len(swap.cards), swap, other_stack, frames=0)
|
|
game.leaveState(old_state)
|
|
|
|
|
|
class DevilsGrip(RoyalParade):
|
|
Foundation_Class = None
|
|
RowStack_Class = None
|
|
TableauStack_Classes = [
|
|
StackWrapper(DevilsGrip_TableauStack,
|
|
base_rank=1, max_cards=4, dir=3),
|
|
StackWrapper(DevilsGrip_TableauStack,
|
|
base_rank=2, max_cards=4, dir=3),
|
|
StackWrapper(DevilsGrip_TableauStack,
|
|
base_rank=3, max_cards=4, dir=3),
|
|
]
|
|
Talon_Class = StackWrapper(WasteTalonStack, max_rounds=-1, num_deal=3)
|
|
|
|
NORMAL_OFFSET = True
|
|
|
|
def createGame(self):
|
|
RoyalParade.createGame(self, waste=True, numstacks=8)
|
|
|
|
def startGame(self):
|
|
self.startDealSample()
|
|
self.s.talon.dealRow(rows=self.s.tableaux, frames=0)
|
|
self.s.talon.dealCards()
|
|
|
|
def isGameWon(self):
|
|
for stack in self.s.tableaux:
|
|
if len(stack.cards) != 4 or \
|
|
stack.cards[0].rank != stack.cap.base_rank:
|
|
return False
|
|
return True
|
|
|
|
def fillStack(self, stack):
|
|
if not stack.cards and stack in self.s.tableaux:
|
|
if self.s.talon.cards:
|
|
old_state = self.enterState(self.S_FILL)
|
|
self.s.talon.moveMove(1, stack)
|
|
self.leaveState(old_state)
|
|
elif self.s.waste.cards:
|
|
old_state = self.enterState(self.S_FILL)
|
|
self.s.waste.moveMove(1, stack)
|
|
self.leaveState(old_state)
|
|
|
|
|
|
# register the game
|
|
registerGame(GameInfo(7, PictureGallery, "Picture Gallery",
|
|
GI.GT_PICTURE_GALLERY, 2, 0, GI.SL_BALANCED,
|
|
altnames=("Die Bildgallerie", "Mod-3")))
|
|
registerGame(GameInfo(397, GreatWheel, "Great Wheel",
|
|
GI.GT_PICTURE_GALLERY | GI.GT_STRIPPED, 2, 0,
|
|
GI.SL_BALANCED, ranks=list(range(12)) # without Kings
|
|
))
|
|
registerGame(GameInfo(398, MountOlympus, "Mount Olympus",
|
|
GI.GT_PICTURE_GALLERY, 2, 0, GI.SL_BALANCED))
|
|
registerGame(GameInfo(399, Zeus, "Zeus",
|
|
GI.GT_PICTURE_GALLERY, 2, 0, GI.SL_BALANCED))
|
|
registerGame(GameInfo(546, RoyalParade, "Royal Parade",
|
|
GI.GT_PICTURE_GALLERY, 2, 0, GI.SL_MOSTLY_SKILL,
|
|
altnames=("Hussars", "Financier", "Cavalcade")))
|
|
registerGame(GameInfo(547, VirginiaReel, "Virginia Reel",
|
|
GI.GT_PICTURE_GALLERY, 2, 0, GI.SL_MOSTLY_SKILL))
|
|
registerGame(GameInfo(782, GreaterWheel, "Greater Wheel",
|
|
GI.GT_PICTURE_GALLERY | GI.GT_STRIPPED, 4, 0,
|
|
GI.SL_BALANCED, ranks=list(range(12)) # without Kings
|
|
))
|
|
registerGame(GameInfo(803, BigParade, "Big Parade",
|
|
GI.GT_PICTURE_GALLERY, 3, 0, GI.SL_MOSTLY_SKILL))
|
|
registerGame(GameInfo(804, ThreeUp, "Three Up",
|
|
GI.GT_PICTURE_GALLERY, 3, 0, GI.SL_MOSTLY_SKILL))
|
|
registerGame(GameInfo(927, BigPictureGallery, "Big Picture Gallery",
|
|
GI.GT_PICTURE_GALLERY, 3, 0, GI.SL_BALANCED))
|
|
registerGame(GameInfo(928, HugePictureGallery, "Huge Picture Gallery",
|
|
GI.GT_PICTURE_GALLERY, 4, 0, GI.SL_BALANCED))
|
|
registerGame(GameInfo(932, DevilsGrip, "Devil's Grip",
|
|
GI.GT_PICTURE_GALLERY | GI.GT_STRIPPED, 2, -1,
|
|
GI.SL_MOSTLY_LUCK,
|
|
ranks=list(range(1, 13)) # without Aces
|
|
))
|
|
registerGame(GameInfo(944, BlueJacket, "Blue Jacket",
|
|
GI.GT_PICTURE_GALLERY, 2, 0, GI.SL_MOSTLY_SKILL))
|
|
registerGame(GameInfo(962, Carousel, "Carousel",
|
|
GI.GT_PICTURE_GALLERY | GI.GT_STRIPPED, 2, 0,
|
|
GI.SL_BALANCED, ranks=list(range(12)) # without Kings
|
|
))
|