1
0
Fork 0
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:
Joe R 2024-01-09 18:49:58 -05:00
parent 97e90f147c
commit caa85016b1
3 changed files with 135 additions and 23 deletions

View 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>.

View file

@ -510,7 +510,7 @@ class GI:
800, 814, 820, 825, 889, 911, 926,
941)),
("Mary Whitmore Jones", (421, 624,)),
("Jan Wolter", (917, 939,)),
("Jan Wolter", (917, 939, 946,)),
)
GAMES_BY_PYSOL_VERSION = (
@ -593,7 +593,7 @@ class GI:
('fc-2.20', tuple(range(855, 897))),
('fc-2.21', tuple(range(897, 900)) + tuple(range(11014, 11017)) +
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(22303, 22311)) + tuple(range(22353, 22361))),
)

View file

@ -28,11 +28,14 @@ from pysollib.layout import Layout
from pysollib.stack import \
AC_RowStack, \
InitialDealTalonStack, \
ReserveStack, \
SS_FoundationStack, \
StackWrapper, \
SuperMoveAC_RowStack, \
WasteStack, \
WasteTalonStack, \
Yukon_AC_RowStack
Yukon_AC_RowStack, \
getNumberOfFreeStacks
from pysollib.util import ANY_RANK, KING
@ -41,17 +44,17 @@ from pysollib.util import ANY_RANK, KING
# ************************************************************************
class Interlock_StackMethods:
STEP = (9, 9, 9, 9, 9, 9, 9, 9, 9,
10, 10, 10, 10, 10, 10, 10, 10, 10, 10)
STEP = ((9, 10), (9, 10), (9, 10), (9, 10), (9, 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):
r, step = self.game.s.rows, self.STEP
i, n, mylen = self.id, 1, len(step)
while i < mylen:
i = i + step[i]
n = n + 1
for j in range(i, i + n):
if r[j].cards:
i, mylen = self.id, len(step)
if i < mylen:
for j in step[i]:
if r[j + i].cards:
return True
return False
@ -61,13 +64,11 @@ class Interlock_StackMethods:
or not self.cards[0].face_up:
return False
r, step = self.game.s.rows, self.STEP
i, n, mylen = self.id, 1, len(step)
while i < mylen:
i = i + step[i]
n = n + 1
for j in range(i, i + n):
if r[j].cards:
if r[j] != other_stack:
i, mylen = self.id, len(step)
if i < mylen:
for j in step[i]:
if r[j + i].cards:
if r[j + i] != other_stack:
return False
return True
@ -86,7 +87,7 @@ class Interlock_StackMethods:
class Interlock_RowStack(Interlock_StackMethods, AC_RowStack):
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
if (self.isDropdownMove(from_stack) and
len(cards) == len(from_stack.cards)):
@ -107,8 +108,9 @@ class Interlock(Game):
def createGame(self):
lay, s = Layout(self), self.s
self.setSize((max(self.MAX_ROWS, 7) * lay.XS) + lay.XM,
(2.5 * lay.YS) + (self.PLAYCARDS * lay.YOFFSET) + lay.YM)
w = (max(self.MAX_ROWS, 7) * lay.XS) + lay.XM
h = (2.5 * lay.YS) + (self.PLAYCARDS * lay.YOFFSET) + lay.YM
self.setSize(w, h)
self.min_rows = self.MAX_ROWS - 2
gap = max(7, self.MAX_ROWS) - self.min_rows
@ -202,7 +204,7 @@ class LoveADuck(Interlock):
# ************************************************************************
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):
@ -218,10 +220,91 @@ class Guardian(Interlock):
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
registerGame(GameInfo(852, Guardian, "Guardian",
GI.GT_KLONDIKE, 1, -1, GI.SL_BALANCED))
registerGame(GameInfo(938, Interlock, "Interlock",
GI.GT_KLONDIKE | GI.GT_ORIGINAL, 1, -1, GI.SL_BALANCED))
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))