diff --git a/html-src/rules/crossword.html b/html-src/rules/crossword.html new file mode 100644 index 00000000..7397aa25 --- /dev/null +++ b/html-src/rules/crossword.html @@ -0,0 +1,20 @@ +

Crossword

+

+One-Deck game type. 1 deck. No redeal. + +

Object

+

+Arrange 49 cards into a seven by seven grid. + +

Rules

+

A single card is dealt in the center of a seven by seven grid. One by one, deal the remaining +cards into the grid, each next to another card, either horizontally, vertically, or diagonally. +Non-face cards in the grid must be played in sequences that total up to even numbers. Once a card +is played, it cannot be moved. +

Face cards played in the grid separate the different sequences, similar to how the black +spaces separate words in an actual crossword puzzle. As such, when a face card is played, it is +flipped face down. Two face cards cannot be played directly adjacent to each other horizontally or +vertically (though them touching diagonally is allowed). +

Once there is only one open space in the grid remaining, the last four cards in the deck are dealt +out and you can play any one of them into the last space. The game is won if 49 cards can be +successfully played into the grid. diff --git a/pysollib/games/__init__.py b/pysollib/games/__init__.py index 9e54dbf9..be7ab71d 100644 --- a/pysollib/games/__init__.py +++ b/pysollib/games/__init__.py @@ -35,6 +35,7 @@ from . import calculation # noqa: F401 from . import camelot # noqa: F401 from . import canfield # noqa: F401 from . import capricieuse # noqa: F401 +from . import crossword # noqa: F401 from . import curdsandwhey # noqa: F401 from . import daddylonglegs # noqa: F401 from . import dieboesesieben # noqa: F401 diff --git a/pysollib/games/crossword.py b/pysollib/games/crossword.py new file mode 100644 index 00000000..6a10f48a --- /dev/null +++ b/pysollib/games/crossword.py @@ -0,0 +1,219 @@ +#!/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 . +# +# --------------------------------------------------------------------------- + +from pysollib.game import Game +from pysollib.gamedb import GI, GameInfo, registerGame +from pysollib.layout import Layout +from pysollib.stack import \ + OpenTalonStack, \ + ReserveStack, \ + Stack, \ + StackWrapper + + +# ************************************************************************ +# * Crossword +# ************************************************************************ + +class Crossword_RowStack(ReserveStack): + def clickHandler(self, event): + if (not self.cards and self.game.s.talon.cards and + self.game.isValidPlay(self.id, + self.game.s.talon.getCard().rank + 1)): + self.game.s.talon.playMoveMove(1, self) + return 1 + return ReserveStack.clickHandler(self, event) + + rightclickHandler = clickHandler + + def acceptsCards(self, from_stack, cards): + return not self.cards and self.game.isValidPlay(self.id, + cards[0].rank + 1) + + def canFlipCard(self): + return False + + def closeStack(self): + if (self.cards[0].rank >= 10): + self.flipMove() + if len(self.game.s.talon.cards) == 4: + self.game.s.talon.flipMove() + for r in self.game.s.reserves: + self.game.s.talon.moveMove(1, r) + + +class Crossword_FinalCard(ReserveStack): + def rightclickHandler(self, event): + if (self.cards): + for r in self.game.s.rows: + if (not r.cards and + self.game.isValidPlay(r.id, self.cards[0].rank + 1)): + self.playMoveMove(1, r) + + def acceptsCards(self, from_stack, cards): + return (len(self.game.s.talon.cards) <= 4 and + from_stack == self.game.s.talon) + + def canMoveCards(self, cards): + return True + + getBottomImage = Stack._getNoneBottomImage + + +class Crossword(Game): + Talon_Class = OpenTalonStack + RowStack_Class = StackWrapper(Crossword_RowStack, max_move=0) + FinalCards_Class = StackWrapper(Crossword_FinalCard, max_move=0) + Hint_Class = None + + # + # game layout + # + + def createGame(self): + # create layout + l, s = Layout(self), self.s + + ta = "ss" + x, y = l.XM, l.YM + 2 * l.YS + + # set window + w = max(2 * l.XS, x) + self.setSize(l.XM + w + 7 * l.XS + 50, l.YM + 7 * l.YS + 30) + + # create stacks + for i in range(7): + for j in range(7): + 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 + + # set up spots for final cards + for i in range(4): + x, y = l.XM, w + l.YM + i * l.YS + s.reserves.append(self.FinalCards_Class(x, y, self)) + x, y = l.XM, l.YM + + s.talon = self.Talon_Class(x, y, self) + l.createText(s.talon, anchor=ta) + + # define rows to check for sequences + r = s.rows + self.crossword_rows = [ + r[0:7], r[7:14], r[14:21], r[21:28], + r[28:35], r[35:42], r[42:49], + (r[0], r[0+7], r[0+14], r[0+21], r[0+28], r[0+35], r[0+42]), + (r[1], r[1+7], r[1+14], r[1+21], r[1+28], r[1+35], r[1+42]), + (r[2], r[2+7], r[2+14], r[2+21], r[2+28], r[2+35], r[2+42]), + (r[3], r[3+7], r[3+14], r[3+21], r[3+28], r[3+35], r[3+42]), + (r[4], r[4+7], r[4+14], r[4+21], r[4+28], r[4+35], r[4+42]), + (r[5], r[5+7], r[5+14], r[5+21], r[5+28], r[5+35], r[5+42]), + (r[6], r[6+7], r[6+14], r[6+21], r[6+28], r[6+35], r[6+42]) + ] + self.crossword_rows = list(map(tuple, self.crossword_rows)) + + # define stack-groups + l.defaultStackGroups() + return l + + def startGame(self): + self.moveMove(1, self.s.talon, self.s.rows[24], frames=0) + self.s.rows[24].flipMove() + self.s.talon.fillStack() + + def isGameWon(self): + if len(self.s.talon.cards) == 0: + for r in self.s.reserves: + if (not r.cards): + return True + return False + + def isValidPlay(self, playSpace, playRank): + # check that there's an adjacent card + if (not self.adjacentCard(playSpace)): + return False + + # check the totals + for hand in self.crossword_rows: + count = 0 # count of the sequence + hasEmpties = False # Whether the sequence still has empty spaces + lastFace = False # Was the last card a face card? + for s in hand: + if s.id == playSpace: + rank = playRank + elif s.cards: + rank = s.cards[0].rank + 1 + else: + rank = -1 + hasEmpties = True + lastFace = False + if (rank > -1): + if (rank < 11): + count += rank + lastFace = False + else: + if ((count % 2) != 0 and not hasEmpties) or lastFace: + return False + else: + count = 0 + hasEmpties = False + lastFace = True + if (count % 2) != 0 and not hasEmpties: + return False + return True + + def adjacentCard(self, playSpace): + if (playSpace % 7 != 6 and self.s.rows[playSpace + 1].cards): + return True + + if (playSpace % 7 != 0 and self.s.rows[playSpace - 1].cards): + return True + + if (playSpace + 7 < 49 and self.s.rows[playSpace + 7].cards): + return True + + if (playSpace - 7 > 0 and self.s.rows[playSpace - 7].cards): + return True + + if (playSpace % 7 != 6 and playSpace - 6 > 0 + and self.s.rows[playSpace - 6].cards): + return True + + if (playSpace % 7 != 0 and playSpace - 8 > 0 + and self.s.rows[playSpace - 8].cards): + return True + + if (playSpace % 7 != 0 and playSpace + 6 < 49 + and self.s.rows[playSpace + 6].cards): + return True + + if (playSpace % 7 != 6 and playSpace + 8 < 49 + and self.s.rows[playSpace + 8].cards): + return True + + return False + + +# register the game +registerGame(GameInfo(778, Crossword, "Crossword", + GI.GT_1DECK_TYPE, 1, 0, GI.SL_BALANCED))