From d0a2b97a9e9bb796268f0e111ef2c27d64650b34 Mon Sep 17 00:00:00 2001 From: Joe R Date: Wed, 3 Jan 2024 19:30:57 -0500 Subject: [PATCH] Added Rosamund's Bower game. --- html-src/rules/rosamundsbower.html | 47 ++++++++++ pysollib/gamedb.py | 2 +- pysollib/games/royalcotillion.py | 135 +++++++++++++++++++++++++++++ 3 files changed, 183 insertions(+), 1 deletion(-) create mode 100644 html-src/rules/rosamundsbower.html diff --git a/html-src/rules/rosamundsbower.html b/html-src/rules/rosamundsbower.html new file mode 100644 index 00000000..d4d00743 --- /dev/null +++ b/html-src/rules/rosamundsbower.html @@ -0,0 +1,47 @@ +

Rosamund's Bower

+

+One-Deck game type. 1 deck. 3 redeals. + +

Object

+

+Move all cards to the single foundation. + +

Rules

+

+At the start of the game, the Jack of Spades is placed aside, along with +the King of Spades and Queen of Hearts. Eight cards are dealt in a circle +around the queen, and a pile of seven cards is dealt face-down below the +king. +

+The foundation is built down from the jack, turning the corner from ace to +king as necessary. When a card is moved from the piles around the queen, +it is replaced with the top card from the pile below the king. +

+When there are no moves left, you can draw cards from the talon one at a +time, and move them to one of three waste piles, or the foundation directly. +When the talon is empty, you can place the three waste piles on top of each +other to form a new talon and redeal. You can redeal three times per game. +

+Only once all the other cards have been moved to the foundation can the +Queen of Hearts or King of Spades be moved. Once they are moved to the +foundation, and all cards have been moved to the foundation, the game is +won. + +

Notes

+

+This game is based on the story of Rosamund Clifford, the mistress +of King Henry II of England. As per the story, they conducted their +affairs inside a maze called "Rosamund's Bower". +

+In the game, the Queen of Hearts represents Fair Rosamund, the King +represents King Henry II, with the Jack representing a sinister assailant +who wants to take Rosamund for himself. The circle of cards around the +queen are the guards of her bower, and the face-down stack under the king +are the king's guards. At the end of the game, the king and queen are +together at the bottom of the deck, symbolizing Henry II united in love +with Fair Rosamund. +

+Another interpretation of the game involves the Jack being a suitor +trying to save the queen of hearts from her father, the evil king. +In this variant, the foundation is built separately from the Jack, +and the Jack and Queen remain the last two cards at the end of the game. diff --git a/pysollib/gamedb.py b/pysollib/gamedb.py index df23b3a1..f78de4ac 100644 --- a/pysollib/gamedb.py +++ b/pysollib/gamedb.py @@ -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, 943)) + tuple(range(11017, 11020)) + + ('dev', tuple(range(906, 944)) + tuple(range(11017, 11020)) + tuple(range(5600, 5624)) + tuple(range(18000, 18005)) + tuple(range(22303, 22311)) + tuple(range(22353, 22361))), ) diff --git a/pysollib/games/royalcotillion.py b/pysollib/games/royalcotillion.py index 4e22a784..36897acb 100644 --- a/pysollib/games/royalcotillion.py +++ b/pysollib/games/royalcotillion.py @@ -1355,6 +1355,138 @@ class TwilightZone(Game): shallHighlightMatch = Game._shallHighlightMatch_AC +# ************************************************************************ +# * Rosamund's Bower +# ************************************************************************ + +class RosamundsBower_Talon(WasteTalonStack): + def _redeal(self): + game, num_cards = self.game, len(self.cards) + if len(self.waste.cards) > 0: + game.moveMove(1, self.waste, game.s.reserves[3], frames=2) + rows = list(game.s.reserves)[3:6] + rows.reverse() + for r in rows: + while r.cards: + num_cards = num_cards + 1 + game.moveMove(1, r, self, frames=2) + if self.cards[-1].face_up: + game.flipMove(self) + assert len(self.cards) == num_cards + self.game.nextRoundMove(self) + + def canDealCards(self): + return (len(self.cards) > 0 or self.round != self.max_rounds) and \ + (len(self.cards) == 0 or len(self.waste.cards) == 0) + + def dealCards(self, sound=False): + if self.cards: + return WasteTalonStack.dealCards(self, sound=sound) + if sound: + self.game.startDealSample() + self._redeal() + if sound: + self.game.stopSamples() + return + + +class RosamundsBower_Guards(OpenStack): + def canFlipCard(self): + return False + + +class RosamundsBower_WasteReserve(ReserveStack): + def acceptsCards(self, from_stack, cards): + return from_stack == self.game.s.waste + + +class RosamundsBower_KingQueen(OpenStack): + def acceptsCards(self, from_stack, cards): + return False + + def canMoveCards(self, cards): + return len(self.game.s.foundations[0].cards) > 49 + + +class RosamundsBower(Game): + + def createGame(self): + l, s = Layout(self), self.s + self.setSize(l.XM + 7.5 * l.XS, l.YM + l.TEXT_HEIGHT + 5.5 * l.YS) + + # vertical rows + x = l.XM + l.XS * 2 + for i in (0, 1): + y = l.YM + l.YS + for j in range(2): + stack = BasicRowStack(x, y, self) + s.rows.append(stack) + y += l.YS + x += 3 * l.XS + # horizontal rows + y = l.YM + for i in (2, 3): + x = l.XM + l.XS * 3 + for j in range(2): + stack = BasicRowStack(x, y, self) + s.rows.append(stack) + x += l.XS + y += 3 * l.YS + # Fair Rosamund and the king + x, y = l.XM + 6.5 * l.XS, l.YM + s.reserves.append(RosamundsBower_KingQueen(x, y, self)) + x, y = l.XM + 3.5 * l.XS, l.YM + 1.5 * l.YS + s.reserves.append(RosamundsBower_KingQueen(x, y, self)) + + # Other stacks + x, y = l.XM + 6.5 * l.XS, l.YM + l.YS + s.reserves.append(RosamundsBower_Guards(x, y, self)) + l.createText(s.reserves[2], 's') + + x, y = l.XM, l.YM + 3 * l.YS + s.foundations.append(RK_FoundationStack(x, y, self, + dir=-1, max_cards=52, + mod=13)) + y = l.YM + 4.5 * l.YS + s.talon = RosamundsBower_Talon(x, y, self, max_rounds=4) + l.createText(s.talon, 's') + l.createRoundText(self.s.talon, 'n') + x += l.XS + s.waste = WasteStack(x, y, self) + l.createText(s.waste, 's') + x += .5 * l.XS + + for i in range(3): + x += l.XS + s.reserves.append(RosamundsBower_WasteReserve(x, y, self, + max_cards=52)) + + l.defaultStackGroups() + + def startGame(self): + self.s.talon.dealRow(rows=self.s.foundations, frames=0) + self.s.talon.dealRow(rows=self.s.reserves[:2], frames=0) + for i in range(7): + self.s.talon.dealRow(rows=[self.s.reserves[2]], frames=0, flip=0) + self.startDealSample() + self.s.talon.dealRow() + self.s.talon.dealCards() + + def fillStack(self, stack): + if not stack.cards and stack in self.s.rows: + if self.s.reserves[2].cards: + old_state = self.enterState(self.S_FILL) + self.s.reserves[2].flipMove() + self.s.reserves[2].moveMove(1, stack) + self.leaveState(old_state) + + def _shuffleHook(self, cards): + # move Fair Rosamund, the king, and the knave to the top of + # the deck. + return self._shuffleHookMoveToTop( + cards, lambda c: (c.id in (23, 37, 25), c.id), 3) + + # register the game registerGame(GameInfo(54, RoyalCotillion, "Royal Cotillion", GI.GT_2DECK_TYPE, 2, 0, GI.SL_LUCK, @@ -1405,3 +1537,6 @@ registerGame(GameInfo(748, TwilightZone, "Twilight Zone", GI.GT_2DECK_TYPE, 2, 1, GI.SL_BALANCED)) registerGame(GameInfo(752, Reserves, "Reserves", GI.GT_2DECK_TYPE, 2, 2, GI.SL_BALANCED)) +registerGame(GameInfo(943, RosamundsBower, "Rosamund's Bower", + GI.GT_1DECK_TYPE, 1, 3, GI.SL_BALANCED, + altnames=("Rosamund",)))