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))