mirror of
https://github.com/shlomif/PySolFC.git
synced 2025-04-05 00:02:29 -04:00
Added Sarlacc game.
This commit is contained in:
parent
97e90f147c
commit
caa85016b1
3 changed files with 135 additions and 23 deletions
29
html-src/rules/sarlacc.html
Normal file
29
html-src/rules/sarlacc.html
Normal file
|
@ -0,0 +1,29 @@
|
||||||
|
<h1>Sarlacc</h1>
|
||||||
|
<p>
|
||||||
|
FreeCell type. 1 deck. No redeal.
|
||||||
|
|
||||||
|
<h3>Object</h3>
|
||||||
|
<p>
|
||||||
|
Move all cards to the foundations.
|
||||||
|
|
||||||
|
<h3>Rules</h3>
|
||||||
|
<p>
|
||||||
|
The tableau piles are built in a pyramid-esque layout, with all cards dealt
|
||||||
|
to alternating rows of ten and eleven cards. Each card forms a tableau pile
|
||||||
|
which can be built down by alternate color. Only single cards can be moved,
|
||||||
|
though sequences can be moved based on the number of six open free cells.
|
||||||
|
The bottom layer cards can be filled with a king when empty, but the remaining
|
||||||
|
layers cannot be filled.
|
||||||
|
<p>
|
||||||
|
Though all cards are dealt face-up, cards in the lower layers cannot
|
||||||
|
be moved until the cards on top of them are moved. However, if the
|
||||||
|
only card or sequence overlapping a card can legally be played on it,
|
||||||
|
that sequence can be moved down in a dropdown move.
|
||||||
|
<p>
|
||||||
|
Foundations are built up in suit from Ace to King. The game is won when
|
||||||
|
all cards are moved to the foundations.
|
||||||
|
|
||||||
|
<h3>Notes</h3>
|
||||||
|
<p>
|
||||||
|
Sarlacc was invented by Jan Wolter, and first appeared in his app
|
||||||
|
<a href="https://politaire.com">Politaire</a>.
|
|
@ -510,7 +510,7 @@ class GI:
|
||||||
800, 814, 820, 825, 889, 911, 926,
|
800, 814, 820, 825, 889, 911, 926,
|
||||||
941)),
|
941)),
|
||||||
("Mary Whitmore Jones", (421, 624,)),
|
("Mary Whitmore Jones", (421, 624,)),
|
||||||
("Jan Wolter", (917, 939,)),
|
("Jan Wolter", (917, 939, 946,)),
|
||||||
)
|
)
|
||||||
|
|
||||||
GAMES_BY_PYSOL_VERSION = (
|
GAMES_BY_PYSOL_VERSION = (
|
||||||
|
@ -593,7 +593,7 @@ class GI:
|
||||||
('fc-2.20', tuple(range(855, 897))),
|
('fc-2.20', tuple(range(855, 897))),
|
||||||
('fc-2.21', tuple(range(897, 900)) + tuple(range(11014, 11017)) +
|
('fc-2.21', tuple(range(897, 900)) + tuple(range(11014, 11017)) +
|
||||||
tuple(range(13160, 13163)) + (16682,)),
|
tuple(range(13160, 13163)) + (16682,)),
|
||||||
('dev', tuple(range(906, 946)) + tuple(range(11017, 11020)) +
|
('dev', tuple(range(906, 947)) + tuple(range(11017, 11020)) +
|
||||||
tuple(range(5600, 5624)) + tuple(range(18000, 18005)) +
|
tuple(range(5600, 5624)) + tuple(range(18000, 18005)) +
|
||||||
tuple(range(22303, 22311)) + tuple(range(22353, 22361))),
|
tuple(range(22303, 22311)) + tuple(range(22353, 22361))),
|
||||||
)
|
)
|
||||||
|
|
|
@ -28,11 +28,14 @@ from pysollib.layout import Layout
|
||||||
from pysollib.stack import \
|
from pysollib.stack import \
|
||||||
AC_RowStack, \
|
AC_RowStack, \
|
||||||
InitialDealTalonStack, \
|
InitialDealTalonStack, \
|
||||||
|
ReserveStack, \
|
||||||
SS_FoundationStack, \
|
SS_FoundationStack, \
|
||||||
StackWrapper, \
|
StackWrapper, \
|
||||||
|
SuperMoveAC_RowStack, \
|
||||||
WasteStack, \
|
WasteStack, \
|
||||||
WasteTalonStack, \
|
WasteTalonStack, \
|
||||||
Yukon_AC_RowStack
|
Yukon_AC_RowStack, \
|
||||||
|
getNumberOfFreeStacks
|
||||||
from pysollib.util import ANY_RANK, KING
|
from pysollib.util import ANY_RANK, KING
|
||||||
|
|
||||||
|
|
||||||
|
@ -41,17 +44,17 @@ from pysollib.util import ANY_RANK, KING
|
||||||
# ************************************************************************
|
# ************************************************************************
|
||||||
|
|
||||||
class Interlock_StackMethods:
|
class Interlock_StackMethods:
|
||||||
STEP = (9, 9, 9, 9, 9, 9, 9, 9, 9,
|
STEP = ((9, 10), (9, 10), (9, 10), (9, 10), (9, 10),
|
||||||
10, 10, 10, 10, 10, 10, 10, 10, 10, 10)
|
(9, 10), (9, 10), (9, 10), (9, 10),
|
||||||
|
(10, 11), (10, 11), (10, 11), (10, 11), (10, 11),
|
||||||
|
(10, 11), (10, 11), (10, 11), (10, 11), (10, 11))
|
||||||
|
|
||||||
def basicIsBlocked(self):
|
def basicIsBlocked(self):
|
||||||
r, step = self.game.s.rows, self.STEP
|
r, step = self.game.s.rows, self.STEP
|
||||||
i, n, mylen = self.id, 1, len(step)
|
i, mylen = self.id, len(step)
|
||||||
while i < mylen:
|
if i < mylen:
|
||||||
i = i + step[i]
|
for j in step[i]:
|
||||||
n = n + 1
|
if r[j + i].cards:
|
||||||
for j in range(i, i + n):
|
|
||||||
if r[j].cards:
|
|
||||||
return True
|
return True
|
||||||
return False
|
return False
|
||||||
|
|
||||||
|
@ -61,13 +64,11 @@ class Interlock_StackMethods:
|
||||||
or not self.cards[0].face_up:
|
or not self.cards[0].face_up:
|
||||||
return False
|
return False
|
||||||
r, step = self.game.s.rows, self.STEP
|
r, step = self.game.s.rows, self.STEP
|
||||||
i, n, mylen = self.id, 1, len(step)
|
i, mylen = self.id, len(step)
|
||||||
while i < mylen:
|
if i < mylen:
|
||||||
i = i + step[i]
|
for j in step[i]:
|
||||||
n = n + 1
|
if r[j + i].cards:
|
||||||
for j in range(i, i + n):
|
if r[j + i] != other_stack:
|
||||||
if r[j].cards:
|
|
||||||
if r[j] != other_stack:
|
|
||||||
return False
|
return False
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
@ -86,7 +87,7 @@ class Interlock_StackMethods:
|
||||||
|
|
||||||
class Interlock_RowStack(Interlock_StackMethods, AC_RowStack):
|
class Interlock_RowStack(Interlock_StackMethods, AC_RowStack):
|
||||||
def acceptsCards(self, from_stack, cards):
|
def acceptsCards(self, from_stack, cards):
|
||||||
if len(self.cards) == 0 and self.id > self.STEP[0] - 1:
|
if len(self.cards) == 0 and self.id > self.STEP[0][0] - 1:
|
||||||
return False
|
return False
|
||||||
if (self.isDropdownMove(from_stack) and
|
if (self.isDropdownMove(from_stack) and
|
||||||
len(cards) == len(from_stack.cards)):
|
len(cards) == len(from_stack.cards)):
|
||||||
|
@ -107,8 +108,9 @@ class Interlock(Game):
|
||||||
|
|
||||||
def createGame(self):
|
def createGame(self):
|
||||||
lay, s = Layout(self), self.s
|
lay, s = Layout(self), self.s
|
||||||
self.setSize((max(self.MAX_ROWS, 7) * lay.XS) + lay.XM,
|
w = (max(self.MAX_ROWS, 7) * lay.XS) + lay.XM
|
||||||
(2.5 * lay.YS) + (self.PLAYCARDS * lay.YOFFSET) + lay.YM)
|
h = (2.5 * lay.YS) + (self.PLAYCARDS * lay.YOFFSET) + lay.YM
|
||||||
|
self.setSize(w, h)
|
||||||
|
|
||||||
self.min_rows = self.MAX_ROWS - 2
|
self.min_rows = self.MAX_ROWS - 2
|
||||||
gap = max(7, self.MAX_ROWS) - self.min_rows
|
gap = max(7, self.MAX_ROWS) - self.min_rows
|
||||||
|
@ -202,7 +204,7 @@ class LoveADuck(Interlock):
|
||||||
# ************************************************************************
|
# ************************************************************************
|
||||||
|
|
||||||
class Guardian_RowStack(Interlock_RowStack):
|
class Guardian_RowStack(Interlock_RowStack):
|
||||||
STEP = (3, 3, 3, 4, 4, 4, 4)
|
STEP = ((3, 4), (3, 4), (3, 4), (4, 5), (4, 5), (4, 5), (4, 5))
|
||||||
|
|
||||||
|
|
||||||
class Guardian(Interlock):
|
class Guardian(Interlock):
|
||||||
|
@ -218,10 +220,91 @@ class Guardian(Interlock):
|
||||||
self.s.talon.dealCards() # deal first card to WasteStack
|
self.s.talon.dealCards() # deal first card to WasteStack
|
||||||
|
|
||||||
|
|
||||||
|
# ************************************************************************
|
||||||
|
# * Sarlacc
|
||||||
|
# ************************************************************************
|
||||||
|
|
||||||
|
class Sarlacc_RowStack(Interlock_StackMethods, SuperMoveAC_RowStack):
|
||||||
|
STEP = ((10, 11), (10, 11), (10, 11), (10, 11), (10, 11),
|
||||||
|
(10, 11), (10, 11), (10, 11), (10, 11), (10, 11),
|
||||||
|
(11,), (10, 11), (10, 11), (10, 11), (10, 11), (10, 11),
|
||||||
|
(10, 11), (10, 11), (10, 11), (10, 11), (10,),
|
||||||
|
(10, 11), (10, 11), (10, 11), (10, 11), (10, 11),
|
||||||
|
(10, 11), (10, 11), (10, 11), (10, 11), (10, 11),
|
||||||
|
(11,), (10, 11), (10, 11), (10, 11), (10, 11), (10, 11),
|
||||||
|
(10, 11), (10, 11), (10, 11), (10, 11), (10,))
|
||||||
|
|
||||||
|
def acceptsCards(self, from_stack, cards):
|
||||||
|
if len(self.cards) == 0 and self.id > 9:
|
||||||
|
return False
|
||||||
|
if (self.isDropdownMove(from_stack) and
|
||||||
|
len(cards) == len(from_stack.cards)):
|
||||||
|
return self.dropdownAcceptsCards(cards)
|
||||||
|
|
||||||
|
return SuperMoveAC_RowStack.acceptsCards(self, from_stack, cards)
|
||||||
|
|
||||||
|
def _getMaxMove(self, to_stack_ncards):
|
||||||
|
max_move = getNumberOfFreeStacks(self.game.s.reserves) + 1
|
||||||
|
if self.cap.base_rank != ANY_RANK:
|
||||||
|
return max_move
|
||||||
|
n = getNumberOfFreeStacks(self.game.s.rows[:10])
|
||||||
|
if to_stack_ncards == 0:
|
||||||
|
n -= 1
|
||||||
|
return max_move << max(n, 0)
|
||||||
|
|
||||||
|
|
||||||
|
class Sarlacc(Interlock):
|
||||||
|
RowStack_Class = Sarlacc_RowStack
|
||||||
|
|
||||||
|
MAX_ROWS = 11
|
||||||
|
PLAYCARDS = 13
|
||||||
|
|
||||||
|
def createGame(self):
|
||||||
|
lay, s = Layout(self), self.s
|
||||||
|
w = (11 * lay.XS) + lay.XM
|
||||||
|
h = (4.5 * lay.YS) + (self.PLAYCARDS * lay.YOFFSET) + lay.YM
|
||||||
|
self.setSize(w, h)
|
||||||
|
|
||||||
|
self.min_rows = self.MAX_ROWS - 2
|
||||||
|
# create stacks
|
||||||
|
for i in range(5):
|
||||||
|
if i % 2 == 0:
|
||||||
|
x = lay.XM + lay.XS // 2
|
||||||
|
else:
|
||||||
|
x = lay.XM
|
||||||
|
|
||||||
|
y = lay.YM + lay.YS + i * lay.YS // 4
|
||||||
|
for j in range(10 + (i % 2)):
|
||||||
|
s.rows.append(self.RowStack_Class(x, y, self))
|
||||||
|
x = x + lay.XS
|
||||||
|
|
||||||
|
x, y = lay.XM, h - lay.YS
|
||||||
|
s.talon = InitialDealTalonStack(x, y, self)
|
||||||
|
|
||||||
|
x, y = lay.XM, lay.YM
|
||||||
|
for i in range(6):
|
||||||
|
s.reserves.append(ReserveStack(x, y, self))
|
||||||
|
x += lay.XS
|
||||||
|
|
||||||
|
for i in range(4):
|
||||||
|
x += lay.XS
|
||||||
|
s.foundations.append(SS_FoundationStack(x, y, self, i,
|
||||||
|
mod=13, max_move=0))
|
||||||
|
|
||||||
|
lay.defaultStackGroups()
|
||||||
|
|
||||||
|
def startGame(self):
|
||||||
|
self.startDealSample()
|
||||||
|
self.s.talon.dealRow(rows=self.s.rows[:42], flip=1, frames=0)
|
||||||
|
self.s.talon.dealRow(rows=self.s.rows[42:])
|
||||||
|
|
||||||
|
|
||||||
# register the game
|
# register the game
|
||||||
registerGame(GameInfo(852, Guardian, "Guardian",
|
registerGame(GameInfo(852, Guardian, "Guardian",
|
||||||
GI.GT_KLONDIKE, 1, -1, GI.SL_BALANCED))
|
GI.GT_KLONDIKE, 1, -1, GI.SL_BALANCED))
|
||||||
registerGame(GameInfo(938, Interlock, "Interlock",
|
registerGame(GameInfo(938, Interlock, "Interlock",
|
||||||
GI.GT_KLONDIKE | GI.GT_ORIGINAL, 1, -1, GI.SL_BALANCED))
|
GI.GT_KLONDIKE | GI.GT_ORIGINAL, 1, -1, GI.SL_BALANCED))
|
||||||
registerGame(GameInfo(939, LoveADuck, "Love a Duck",
|
registerGame(GameInfo(939, LoveADuck, "Love a Duck",
|
||||||
GI.GT_YUKON | GI.GT_OPEN, 1, 0, GI.SL_BALANCED))
|
GI.GT_YUKON | GI.GT_OPEN, 1, 0, GI.SL_MOSTLY_SKILL))
|
||||||
|
registerGame(GameInfo(946, Sarlacc, "Sarlacc",
|
||||||
|
GI.GT_FREECELL | GI.GT_OPEN, 1, 0, GI.SL_MOSTLY_SKILL))
|
||||||
|
|
Loading…
Add table
Reference in a new issue