mirror of
https://github.com/shlomif/PySolFC.git
synced 2025-04-05 00:02:29 -04:00
643 lines
21 KiB
Python
643 lines
21 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.layout import Layout
|
|
from pysollib.mygettext import _
|
|
from pysollib.pysoltk import MfxCanvasText
|
|
from pysollib.stack import \
|
|
InitialDealTalonStack, \
|
|
InvisibleStack, \
|
|
OpenTalonStack, \
|
|
RedealTalonStack, \
|
|
ReserveStack, \
|
|
Stack, \
|
|
StackWrapper
|
|
|
|
# ************************************************************************
|
|
# * Cribbage Square
|
|
# ************************************************************************
|
|
|
|
|
|
class CribbageSquare_RowStack(ReserveStack):
|
|
def clickHandler(self, event):
|
|
if not self.cards:
|
|
self.game.s.talon.playMoveMove(1, self)
|
|
return 1
|
|
return ReserveStack.clickHandler(self, event)
|
|
|
|
rightclickHandler = clickHandler
|
|
|
|
|
|
class CribbageSquare_Talon(OpenTalonStack):
|
|
def canMoveCards(self, cards):
|
|
if self.game.isBoardFull():
|
|
return False
|
|
return OpenTalonStack.canMoveCards(self, cards)
|
|
|
|
|
|
class CribbageSquare(Game):
|
|
Talon_Class = CribbageSquare_Talon
|
|
RowStack_Class = StackWrapper(CribbageSquare_RowStack, max_move=0)
|
|
Hint_Class = None
|
|
|
|
WIN_SCORE = 61
|
|
NUM_RESERVE = 0
|
|
RESERVE_STACK = StackWrapper(ReserveStack, max_cards=5)
|
|
|
|
#
|
|
# game layout
|
|
#
|
|
|
|
def createGame(self):
|
|
# create layout
|
|
l, s = Layout(self), self.s
|
|
|
|
# create texts 1)
|
|
ta = "ss"
|
|
x, y = l.XM, l.YM + 2 * l.YS
|
|
|
|
# set window
|
|
w = max(2 * l.XS, x, ((self.NUM_RESERVE + 1) * l.XS) + (4 * l.XM))
|
|
self.setSize(l.XM + w + 4 * l.XS + 50, l.YM + 4 * l.YS + 30)
|
|
|
|
# create stacks
|
|
for i in range(4):
|
|
for j in range(4):
|
|
x, y = l.XM + w + j * l.XS, l.YM + i * l.YS
|
|
s.rows.append(self.RowStack_Class(x, y, self))
|
|
x, y = l.XM, l.YM
|
|
s.talon = self.Talon_Class(x, y, self)
|
|
l.createText(s.talon, anchor=ta)
|
|
s.internals.append(InvisibleStack(self)) # for _swapPairMove()
|
|
|
|
for i in range(self.NUM_RESERVE):
|
|
x, y = ((i + 1) * l.XS) + (2 * l.XM), l.YM
|
|
s.reserves.append(self.RESERVE_STACK(x, y, self))
|
|
|
|
# create texts 2)
|
|
if self.preview <= 1:
|
|
for i in (3, 7, 11, 15):
|
|
tx, ty, ta, tf = l.getTextAttr(s.rows[i], anchor="e")
|
|
t = MfxCanvasText(self.canvas, tx + 8, ty,
|
|
anchor=ta,
|
|
font=self.app.getFont("canvas_default"))
|
|
self.texts.list.append(t)
|
|
for i in range(12, 16):
|
|
tx, ty, ta, tf = l.getTextAttr(s.rows[i], anchor="ss")
|
|
t = MfxCanvasText(self.canvas, tx, ty, anchor=ta,
|
|
font=self.app.getFont("canvas_default"))
|
|
self.texts.list.append(t)
|
|
self.texts.score = MfxCanvasText(
|
|
self.canvas, l.XM, 4 * l.YS, anchor="sw",
|
|
font=self.app.getFont("canvas_large"))
|
|
|
|
# define hands for scoring
|
|
r = s.rows
|
|
self.cribbage_hands = [
|
|
r[0:4], r[4:8], r[8:12], r[12:16],
|
|
(r[0], r[0+4], r[0+8], r[0+12]),
|
|
(r[1], r[1+4], r[1+8], r[1+12]),
|
|
(r[2], r[2+4], r[2+8], r[2+12]),
|
|
(r[3], r[3+4], r[3+8], r[3+12])
|
|
]
|
|
self.cribbage_hands = list(map(tuple, self.cribbage_hands))
|
|
|
|
# define stack-groups
|
|
l.defaultStackGroups()
|
|
return l
|
|
|
|
#
|
|
# game overrides
|
|
#
|
|
|
|
def startGame(self):
|
|
self.moveMove(35 - (5 * self.NUM_RESERVE), self.s.talon,
|
|
self.s.internals[0], frames=0)
|
|
self.s.talon.fillStack()
|
|
|
|
def isBoardFull(self):
|
|
for i in range(16):
|
|
if len(self.s.rows[i].cards) == 0:
|
|
return False
|
|
return True
|
|
|
|
def isGameWon(self):
|
|
if self.isBoardFull():
|
|
return self.getGameScore() >= self.WIN_SCORE
|
|
|
|
return False
|
|
|
|
def getAutoStacks(self, event=None):
|
|
return ((), (), ())
|
|
|
|
#
|
|
# scoring
|
|
#
|
|
|
|
def updateText(self):
|
|
if self.preview > 1:
|
|
return
|
|
score = 0
|
|
for i in range(8):
|
|
value = self.getHandScore(self.cribbage_hands[i])
|
|
|
|
self.texts.list[i].config(text=str(value))
|
|
score += value
|
|
#
|
|
score = self.checkHisHeels(score)
|
|
t = ""
|
|
if score >= self.WIN_SCORE:
|
|
t = _("WON\n\n")
|
|
if not self.isBoardFull():
|
|
t += _("Points: %d") % score
|
|
else:
|
|
t += _("Total: %d") % score
|
|
self.texts.score.config(text=t)
|
|
|
|
def getGameScore(self):
|
|
score = 0
|
|
for hand in self.cribbage_hands:
|
|
value = self.getHandScore(hand)
|
|
score += value
|
|
|
|
score = self.checkHisHeels(score)
|
|
return score
|
|
|
|
def getAllCombinations(self, hand):
|
|
if hand == ():
|
|
return ((),)
|
|
|
|
x = self.getAllCombinations(hand[1:])
|
|
|
|
return x + tuple([(hand[0],) + y for y in x])
|
|
|
|
def getHandScore(self, hand):
|
|
same_suit = [0] * 4
|
|
hand_score = 0
|
|
|
|
upcard = None
|
|
upcard_talon = None
|
|
if self.isBoardFull():
|
|
upcard_talon = self.getUpcardStack()
|
|
upcard = upcard_talon.cards[0]
|
|
|
|
# First get flushes and his nobs, as these can only be
|
|
# scored once per hand.
|
|
for s in hand:
|
|
if s.cards:
|
|
suit = s.cards[0].suit
|
|
same_suit[suit] = same_suit[suit] + 1
|
|
if upcard is not None and s.cards[0].rank == 10 \
|
|
and s.cards[0].suit == upcard.suit:
|
|
hand_score += 1 # His nobs
|
|
#
|
|
if max(same_suit) == 4:
|
|
hand_score += 4 # Flush
|
|
if upcard is not None and upcard.suit == hand[0].cards[0].suit:
|
|
hand_score += 1 # Flush of five
|
|
|
|
if upcard is not None:
|
|
hand = hand + (upcard_talon,)
|
|
combos = self.getAllCombinations(hand)
|
|
|
|
longest_run = 3
|
|
run_score = 0
|
|
# The remaining hands can be scored for every combination.
|
|
for c in combos:
|
|
c_same_rank = [0] * 13
|
|
c_ranks = []
|
|
total = 0
|
|
incomplete = False
|
|
|
|
for s in c:
|
|
if s.cards:
|
|
rank = s.cards[0].rank
|
|
c_same_rank[rank] = c_same_rank[rank] + 1
|
|
c_ranks.append(rank)
|
|
|
|
if rank < 10:
|
|
total += (rank + 1)
|
|
else:
|
|
total += 10
|
|
else:
|
|
incomplete = True
|
|
break
|
|
|
|
if incomplete:
|
|
continue
|
|
|
|
if total == 15:
|
|
hand_score += 2 # Fifteen
|
|
|
|
if len(c) == 2 and max(c_same_rank) == 2:
|
|
hand_score += 2 # Pair
|
|
|
|
# For runs, we only want to consider the longest run
|
|
if len(c) >= longest_run:
|
|
if c_same_rank.count(1) == len(c):
|
|
d = max(c_ranks) - min(c_ranks)
|
|
if d == len(c) - 1:
|
|
if len(c) > longest_run:
|
|
run_score = 0
|
|
longest_run = len(c)
|
|
run_score += longest_run # Runs
|
|
|
|
hand_score += run_score
|
|
|
|
return hand_score
|
|
|
|
def getUpcardStack(self):
|
|
return self.s.talon
|
|
|
|
def checkHisHeels(self, score):
|
|
if self.isBoardFull() and self.getUpcardStack().cards[0].rank == 10:
|
|
return score + 2
|
|
return score
|
|
|
|
|
|
# ************************************************************************
|
|
# * Cribbage Shuffle
|
|
# ************************************************************************
|
|
|
|
class CribbageShuffle_RowStack(ReserveStack):
|
|
def moveMove(self, ncards, to_stack, frames=-1, shadow=-1):
|
|
assert ncards == 1 and to_stack in self.game.s.rows
|
|
assert len(to_stack.cards) == 1
|
|
self._swapPairMove(ncards, to_stack, frames=-1, shadow=0)
|
|
|
|
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 CribbageShuffle(CribbageSquare):
|
|
Talon_Class = InitialDealTalonStack
|
|
RowStack_Class = StackWrapper(
|
|
CribbageShuffle_RowStack, max_accept=1, max_cards=2)
|
|
|
|
WIN_SCORE = 61
|
|
|
|
def createGame(self):
|
|
CribbageSquare.createGame(self)
|
|
if self.s.talon.texts.ncards:
|
|
self.s.talon.texts.ncards.text_format = "%D"
|
|
|
|
def startGame(self):
|
|
self.moveMove(35, self.s.talon, self.s.internals[0], frames=0)
|
|
self._startAndDealRow()
|
|
self.s.talon.flipMove()
|
|
|
|
|
|
# ************************************************************************
|
|
# * Cribbage Square (Waste)
|
|
# * Cribbage Square (1 Reserve)
|
|
# * Cribbage Square (2 Reserves)
|
|
# ************************************************************************
|
|
|
|
class CribbageSquareWaste(CribbageSquare):
|
|
NUM_RESERVE = 1
|
|
RESERVE_STACK = StackWrapper(ReserveStack, max_cards=5, max_move=0)
|
|
|
|
|
|
class CribbageSquare1Reserve(CribbageSquare):
|
|
NUM_RESERVE = 1
|
|
|
|
|
|
class CribbageSquare2Reserves(CribbageSquare):
|
|
NUM_RESERVE = 2
|
|
|
|
|
|
# ************************************************************************
|
|
# * Cribbage Patience
|
|
# ************************************************************************
|
|
|
|
class CribbagePatience_Talon(RedealTalonStack):
|
|
|
|
def canDealCards(self):
|
|
return self.game.isFinalizedHand and len(self.cards) >= 9
|
|
|
|
def dealCards(self, sound=False):
|
|
old_state = self.game.enterState(self.game.S_FILL)
|
|
self.game.dealHand()
|
|
self.game.leaveState(old_state)
|
|
|
|
|
|
class CribbagePatience_HandStack(ReserveStack):
|
|
getBottomImage = Stack._getNoneBottomImage
|
|
|
|
def clickHandler(self, event):
|
|
for s in self.game.s.rows[0:4]:
|
|
if len(s.cards) == 0 and s.acceptsCards(self, self.cards):
|
|
return self.playMoveMove(1, s)
|
|
return 0
|
|
|
|
rightclickHandler = clickHandler
|
|
|
|
def acceptsCards(self, from_stack, cards):
|
|
return False
|
|
|
|
def moveMove(self, ncards, to_stack, frames=-1, shadow=-1):
|
|
ReserveStack.moveMove(self, ncards, to_stack, frames=frames,
|
|
shadow=shadow)
|
|
self.game.dealAdditionalHand()
|
|
if self.game.isBoardFull():
|
|
self.game.finalizeHand()
|
|
|
|
|
|
class CribbagePatience_CribStack(ReserveStack):
|
|
def acceptsCards(self, from_stack, cards):
|
|
if from_stack not in self.game.s.rows[4:]:
|
|
return False
|
|
if self.game.HANDS > 1:
|
|
if (len(self.game.s.rows[10].cards) > 0 and
|
|
from_stack not in self.game.s.rows[10:]):
|
|
return False
|
|
return ReserveStack.acceptsCards(self, from_stack, cards)
|
|
|
|
|
|
class CribbagePatience(CribbageShuffle):
|
|
WIN_SCORE = 80
|
|
HANDS = 1
|
|
|
|
def createGame(self):
|
|
self.score = 0
|
|
self.isFinalizedHand = False
|
|
l, s = Layout(self), self.s
|
|
self.setSize((2 * l.XM) + 8 * l.XS,
|
|
l.YM + ((self.HANDS + 1) * l.YS))
|
|
x, y = self.getInvisibleCoords()
|
|
s.waste = ReserveStack(x, y, self)
|
|
x, y = l.XM, l.YM
|
|
s.talon = CribbagePatience_Talon(x, y, self)
|
|
l.createText(s.talon, "se")
|
|
x += 2 * l.XS
|
|
for i in range(4):
|
|
s.rows.append(CribbagePatience_CribStack(x, y, self, max_move=1))
|
|
x += l.XS
|
|
x, y = l.XM, l.YM + l.YS
|
|
s.reserves.append(ReserveStack(x, y, self))
|
|
for i in range(self.HANDS):
|
|
x += 2 * l.XS
|
|
for i in range(6):
|
|
s.rows.append(CribbagePatience_HandStack(x, y, self,
|
|
max_move=1))
|
|
x += l.XS
|
|
x, y = l.XM, y + l.YS
|
|
|
|
# define hands for scoring
|
|
r = s.rows
|
|
if self.HANDS == 1:
|
|
self.cribbage_hands = [
|
|
r[0:4], r[4:8]
|
|
]
|
|
else:
|
|
self.cribbage_hands = [
|
|
r[0:4], r[4:8], r[10:14]
|
|
]
|
|
self.cribbage_hands = list(map(tuple, self.cribbage_hands))
|
|
|
|
if self.preview <= 1:
|
|
if self.HANDS == 2:
|
|
scores = (0, 4, 10)
|
|
else:
|
|
scores = (0, 4)
|
|
|
|
for i in scores:
|
|
tx, ty, ta, tf = l.getTextAttr(s.rows[i], anchor="w")
|
|
t = MfxCanvasText(self.canvas, tx - 8, ty,
|
|
anchor=ta,
|
|
font=self.app.getFont("canvas_default"))
|
|
self.texts.list.append(t)
|
|
self.texts.score = MfxCanvasText(
|
|
self.canvas, l.XM + 6 * l.XS, 1 * l.YS, anchor="sw",
|
|
font=self.app.getFont("canvas_large"))
|
|
|
|
l.defaultStackGroups()
|
|
|
|
def fillStack(self, stack):
|
|
if not stack.cards:
|
|
old_state = self.enterState(self.S_FILL)
|
|
hand = self.s.rows[4:10]
|
|
if stack in hand:
|
|
i = list(hand).index(stack)
|
|
if i < len(hand)-1:
|
|
from_stack = hand[i+1]
|
|
pile = from_stack.getPile()
|
|
if pile:
|
|
from_stack.moveMove(len(pile), stack)
|
|
self.leaveState(old_state)
|
|
|
|
def startGame(self):
|
|
self.moveMove(7, self.s.talon, self.s.waste, frames=0)
|
|
self.score = 0
|
|
self.isFinalizedHand = False
|
|
self.dealHand()
|
|
|
|
def dealHand(self):
|
|
self.startDealSample()
|
|
self.saveStateMove(2 | 16)
|
|
if self.isFinalizedHand:
|
|
for r in reversed(self.s.rows):
|
|
if (len(r.cards) > 0):
|
|
r.moveMove(1, self.s.waste)
|
|
self.s.reserves[0].moveMove(1, self.s.waste)
|
|
self.isFinalizedHand = False
|
|
self.saveStateMove(1 | 16)
|
|
|
|
self.s.talon.dealRow(rows=self.s.rows[:2], flip=0)
|
|
self.s.talon.dealRow(rows=self.s.rows[4:10])
|
|
self.stopSamples()
|
|
|
|
def dealAdditionalHand(self):
|
|
return
|
|
|
|
def isBoardFull(self):
|
|
for i in range(8):
|
|
if len(self.s.rows[i].cards) == 0:
|
|
return False
|
|
return True
|
|
|
|
def finalizeHand(self):
|
|
if self.isFinalizedHand:
|
|
return
|
|
self.saveStateMove(2 | 16)
|
|
self.isFinalizedHand = True
|
|
old_state = self.enterState(self.S_FILL)
|
|
for c in self.s.rows[0:2]:
|
|
if not c.cards[0].face_up:
|
|
c.flipMove()
|
|
self.s.talon.flipMove()
|
|
self.s.talon.moveMove(1, self.s.reserves[0])
|
|
self.leaveState(old_state)
|
|
for i in range(len(self.cribbage_hands)):
|
|
value = self.getHandScore(self.cribbage_hands[i])
|
|
self.texts.list[i].config(text=str(value))
|
|
self.score += value
|
|
self.score = self.checkHisHeels(self.score)
|
|
self.saveStateMove(1 | 16)
|
|
|
|
def updateText(self):
|
|
if self.preview > 1:
|
|
return
|
|
if self.isBoardFull():
|
|
for i in range(len(self.cribbage_hands)):
|
|
value = self.getHandScore(self.cribbage_hands[i])
|
|
|
|
self.texts.list[i].config(text=str(value))
|
|
else:
|
|
for i in range(len(self.cribbage_hands)):
|
|
self.texts.list[i].config(text="")
|
|
#
|
|
t = ""
|
|
if self.score >= self.WIN_SCORE and len(self.s.talon.cards) == 0:
|
|
t = _("WON\n\n")
|
|
t += _("Total: %d") % self.score
|
|
self.texts.score.config(text=t)
|
|
|
|
def _autoDeal(self, sound=True):
|
|
# don't deal a card to the waste if the waste is empty
|
|
return 0
|
|
|
|
def getUpcardStack(self):
|
|
return self.s.reserves[0]
|
|
|
|
def getGameScore(self):
|
|
return self.score
|
|
|
|
def _restoreGameHook(self, game):
|
|
self.score = game.loadinfo.dval.get('Score')
|
|
self.isFinalizedHand = game.loadinfo.dval.get('Finalized')
|
|
|
|
def _loadGameHook(self, p):
|
|
self.loadinfo.addattr(dval=p.load())
|
|
|
|
def _saveGameHook(self, p):
|
|
dval = {'Score': self.score, 'Finalized': self.isFinalizedHand}
|
|
p.dump(dval)
|
|
|
|
def setState(self, state):
|
|
# restore saved vars (from undo/redo)
|
|
self.score = state[0]
|
|
self.isFinalizedHand = state[1]
|
|
|
|
def getState(self):
|
|
# save vars (for undo/redo)
|
|
return [self.score, self.isFinalizedHand]
|
|
|
|
|
|
class CribbagePatienceII(CribbagePatience):
|
|
WIN_SCORE = 80
|
|
HANDS = 2
|
|
|
|
def startGame(self):
|
|
self.score = 0
|
|
self.isFinalizedHand = False
|
|
self.dealHand()
|
|
|
|
def fillStack(self, stack):
|
|
if not stack.cards:
|
|
old_state = self.enterState(self.S_FILL)
|
|
hand = self.s.rows[4:10] + self.s.rows[10:16]
|
|
if stack in hand:
|
|
i = list(hand).index(stack)
|
|
if i < len(hand)-1:
|
|
from_stack = hand[i+1]
|
|
pile = from_stack.getPile()
|
|
if pile:
|
|
from_stack.moveMove(len(pile), stack)
|
|
self.leaveState(old_state)
|
|
|
|
def dealHand(self):
|
|
self.startDealSample()
|
|
if self.isFinalizedHand:
|
|
for r in reversed(self.s.rows):
|
|
if (len(r.cards) > 0):
|
|
r.moveMove(1, self.s.waste)
|
|
self.s.reserves[0].moveMove(1, self.s.waste)
|
|
self.saveStateMove(2 | 16)
|
|
self.isFinalizedHand = False
|
|
self.saveStateMove(1 | 16)
|
|
|
|
self.s.talon.dealRow(rows=self.s.rows[4:10])
|
|
self.stopSamples()
|
|
|
|
def dealAdditionalHand(self):
|
|
for i in range(2):
|
|
if len(self.s.rows[i].cards) == 0:
|
|
return
|
|
for i in range(2, 4):
|
|
if len(self.s.rows[i].cards) > 0:
|
|
return
|
|
if len(self.s.rows[10].cards) == 0:
|
|
self.startDealSample()
|
|
self.saveStateMove(2 | 16)
|
|
self.s.talon.dealRow(rows=self.s.rows[10:16])
|
|
self.saveStateMove(1 | 16)
|
|
self.stopSamples()
|
|
|
|
def isBoardFull(self):
|
|
for i in range(8):
|
|
if len(self.s.rows[i].cards) == 0:
|
|
return False
|
|
for i in range(10, 14):
|
|
if len(self.s.rows[i].cards) == 0:
|
|
return False
|
|
return True
|
|
|
|
|
|
# register the game
|
|
registerGame(GameInfo(805, CribbageSquare, "Cribbage Square",
|
|
GI.GT_CRIBBAGE_TYPE | GI.GT_SCORE, 1, 0,
|
|
GI.SL_MOSTLY_SKILL, si={"ncards": 17}))
|
|
registerGame(GameInfo(806, CribbageSquareWaste, "Cribbage Square (Waste)",
|
|
GI.GT_CRIBBAGE_TYPE | GI.GT_SCORE, 1, 0,
|
|
GI.SL_MOSTLY_SKILL, si={"ncards": 22},
|
|
rules_filename="cribbagesquare.html"))
|
|
registerGame(GameInfo(807, CribbageSquare1Reserve,
|
|
"Cribbage Square (1 Reserve)",
|
|
GI.GT_CRIBBAGE_TYPE | GI.GT_SCORE, 1, 0,
|
|
GI.SL_MOSTLY_SKILL, si={"ncards": 22},
|
|
rules_filename="cribbagesquare.html"))
|
|
registerGame(GameInfo(808, CribbageSquare2Reserves,
|
|
"Cribbage Square (2 Reserves)",
|
|
GI.GT_CRIBBAGE_TYPE | GI.GT_SCORE, 1, 0,
|
|
GI.SL_MOSTLY_SKILL, si={"ncards": 27},
|
|
rules_filename="cribbagesquare.html"))
|
|
registerGame(GameInfo(809, CribbageShuffle, "Cribbage Shuffle",
|
|
GI.GT_CRIBBAGE_TYPE | GI.GT_SCORE | GI.GT_OPEN, 1, 0,
|
|
GI.SL_MOSTLY_SKILL,
|
|
si={"ncards": 17}))
|
|
registerGame(GameInfo(955, CribbagePatience, "Cribbage Patience",
|
|
GI.GT_CRIBBAGE_TYPE | GI.GT_SCORE, 1, 0,
|
|
GI.SL_MOSTLY_SKILL))
|
|
registerGame(GameInfo(980, CribbagePatienceII, "Cribbage Patience II",
|
|
GI.GT_CRIBBAGE_TYPE | GI.GT_SCORE, 1, 0,
|
|
GI.SL_MOSTLY_SKILL))
|