From 3ef949d4611cd98e2447da17761305d1e574875b Mon Sep 17 00:00:00 2001 From: Joe R Date: Sun, 27 Aug 2023 12:20:04 -0400 Subject: [PATCH] Added support for Picture Puzzle games, based on a concept from UltraSol. --- contrib/customize_cardset.asciidoc | 4 +- .../cards/bottoms/puzzle/bottom02-n.png | Bin 0 -> 129 bytes data/images/cards/bottoms/puzzle/bottom02.png | Bin 0 -> 129 bytes html-src/cardset_customization.html | 9 +- html-src/gen-html.py | 1 + html-src/rules/matrix.html | 28 ++-- html-src/rules/tilepuzzle.html | 11 ++ po/de_pysol.po | 8 +- po/fr_pysol.po | 8 +- po/it_pysol.po | 8 +- po/pl_pysol.po | 8 +- po/pt_BR_pysol.po | 8 +- po/pysol.pot | 6 + po/ru_pysol.po | 8 +- pysollib/app.py | 4 + pysollib/game/__init__.py | 2 +- pysollib/gamedb.py | 15 +- pysollib/games/ultra/__init__.py | 1 + pysollib/games/ultra/matrix.py | 26 ++++ pysollib/games/ultra/tilepuzzle.py | 144 ++++++++++++++++++ pysollib/options.py | 28 +++- pysollib/resource.py | 30 +++- scripts/cardset_viewer.py | 3 +- 23 files changed, 331 insertions(+), 29 deletions(-) create mode 100644 data/images/cards/bottoms/puzzle/bottom02-n.png create mode 100644 data/images/cards/bottoms/puzzle/bottom02.png create mode 100644 html-src/rules/tilepuzzle.html create mode 100644 pysollib/games/ultra/tilepuzzle.py diff --git a/contrib/customize_cardset.asciidoc b/contrib/customize_cardset.asciidoc index 1807da8c..1fdcb6ec 100644 --- a/contrib/customize_cardset.asciidoc +++ b/contrib/customize_cardset.asciidoc @@ -42,6 +42,7 @@ Cardsets Types: * Dashavatara Ganjifa = 8 * Trumps Only = 9 * Matching = 10 +* Puzzle = 11 *$C:* The number of cards in the cardset, generally 42, 48, 52, 68, 78, 96 or 120 @@ -76,6 +77,7 @@ Cardsets Styles: * Places = 20 * Plain = 21 * Products = 22 +* Puzzle = 33 * Round Cardsets = 18 * Science Fiction = 23 * Sports = 24 @@ -112,7 +114,7 @@ Cardsets Origins: *$F:* The Year the cardset was created (in the range 1000 to 2299) -*$G:* The subtype of the cardset. Usually 0 - for French type cardsets, a value of 1 is used if there are jokers. +*$G:* The subtype of the cardset. Usually 0 - for French type cardsets, a value of 1 is used if there are jokers. For Puzzle type cardsets, it is the number of pieces per row/column. *$H:* Whether the cardset is a 3D Mahjongg cardset - 1 if it is, 0 if it isn't. For cardsets with a version less than 7, version 6 cardsets treat this value as 1, and older version cardsets treat it as 0. diff --git a/data/images/cards/bottoms/puzzle/bottom02-n.png b/data/images/cards/bottoms/puzzle/bottom02-n.png new file mode 100644 index 0000000000000000000000000000000000000000..732d76760990a15d0d7f21c259a27535503be838 GIT binary patch literal 129 zcmeAS@N?(olHy`uVBq!ia0vp^A|TAc1|)ksWqE-VV{wqX6T`Z5GB1G~mUKs7M+SzC z{oH>NS%G}c0*}aI1_r)EAj~ML;ne^X6!Ua(46*P}mSA0+AkxzmXu!3CNkW2wErfy5 TN~X^rsDi=M)z4*}Q$iB}CEpr} literal 0 HcmV?d00001 diff --git a/data/images/cards/bottoms/puzzle/bottom02.png b/data/images/cards/bottoms/puzzle/bottom02.png new file mode 100644 index 0000000000000000000000000000000000000000..732d76760990a15d0d7f21c259a27535503be838 GIT binary patch literal 129 zcmeAS@N?(olHy`uVBq!ia0vp^A|TAc1|)ksWqE-VV{wqX6T`Z5GB1G~mUKs7M+SzC z{oH>NS%G}c0*}aI1_r)EAj~ML;ne^X6!Ua(46*P}mSA0+AkxzmXu!3CNkW2wErfy5 TN~X^rsDi=M)z4*}Q$iB}CEpr} literal 0 HcmV?d00001 diff --git a/html-src/cardset_customization.html b/html-src/cardset_customization.html index 3ffdcd26..3dbd6165 100644 --- a/html-src/cardset_customization.html +++ b/html-src/cardset_customization.html @@ -47,6 +47,9 @@ back01.ext;back02.ext;back03.ext
  • Matching = 10

  • +
  • +

    Puzzle = 11

    +
  • $C: The number of cards in the cardset, generally 42, 48, 52, 68, 78, 96 or 120

    @@ -132,6 +135,9 @@ back01.ext;back02.ext;back03.ext
  • Products = 22

  • +
  • +

    Puzzle = 33

    +
  • Round Cardsets = 18

  • @@ -221,7 +227,8 @@ back01.ext;back02.ext;back03.ext

    $F: The Year the cardset was created (in the range 1000 to 2299)

    $G: The subtype of the cardset. Usually 0 - for French type - cardsets, a value of 1 is used if there are jokers.

    + cardsets, a value of 1 is used if there are jokers. For Puzzle type + cardsets, it is the number of pieces per row/column.

    $H: Whether the cardset is a 3D Mahjongg cardset - 1 if it is, 0 if it isn't. For cardsets with a version less than 7, version 6 cardsets treat this value as 1, and older version cardsets treat it as 0.

    diff --git a/html-src/gen-html.py b/html-src/gen-html.py index 5ee1a5df..4b240960 100755 --- a/html-src/gen-html.py +++ b/html-src/gen-html.py @@ -79,6 +79,7 @@ rules_files = [ ('freecell.html', 'PySol - Rules for FreeCell'), ('lightsout.html', 'PySol - Rules for Lights Out'), ('fourrivers.html', 'PySol - Rules for Four Rivers'), + ('tilepuzzle.html', 'PySol - Rules for Tile Puzzle'), ] diff --git a/html-src/rules/matrix.html b/html-src/rules/matrix.html index 84174029..7da34351 100644 --- a/html-src/rules/matrix.html +++ b/html-src/rules/matrix.html @@ -5,19 +5,21 @@ Matrix type. One deck. No redeals. Restore game pieces to their proper order.

    Rules

    -When the game opens the all game pieces except for the final piece are -dealt to the tableau in their proper order. The pieces are then scrambled -to a random pattern. With the larger grids it may take several seconds for -the scrambling to be completed. Pieces may not be lifted from the canvas. -They may be moved by clicking on a piece which is in either the row or the -column which has the empty spot. That piece and all intervening pieces will -move one space towards the empty spot. The image game piece sets are best used -with the grid size for which they are designed. The size is indicated in the -name of the set. King of Hearts 4x4, Players Trumps 10x10, etc. The default -set of numbered pieces works with any grid size. When all the pieces have been -restored to their correct order the final piece will be dealt to the tableau and -the game has been won. Every Matrix game can be solved. +

    +The pieces are dealt to a grid, based on the size of the matrix, with a single +blank space left. Pieces may not be lifted from the canvas. They may be moved +by clicking on a piece which is in either the row or the column which has the +empty spot. That piece and all intervening pieces will move one space towards +the empty spot. +

    +Matrix games can be played with a sequential numbered cardset (Matrix), or with +a picture puzzle cardset (Picture Matrix). With the sequential cardset, your +goal is to organize the cards in sequence, left to right, top to bottom. With +a picture puzzle cardset, your goal is to assemble the picture correctly. +

    +When all the pieces have been restored to their correct order, with the blank space +in the bottom right corner, the final piece will be dealt to the tableau and the +game has been won. Every Matrix game can be solved.

    Strategy

    Begin in the upper left hand corner and complete one row before starting the next. -Take a screen shot of the image sets before the game is scrambled. diff --git a/html-src/rules/tilepuzzle.html b/html-src/rules/tilepuzzle.html new file mode 100644 index 00000000..207d759a --- /dev/null +++ b/html-src/rules/tilepuzzle.html @@ -0,0 +1,11 @@ +

    Tile Puzzle

    +Puzzle type. One deck. No redeals. + +

    Object

    +Restore game pieces to their proper order. + +

    Rules

    +

    +The pieces are dealt to a grid, based on the size of the puzzle. You may +swap any two pieces freely. Your goal is to arrange all the pieces to +successfully, and accurately, complete the picture. diff --git a/po/de_pysol.po b/po/de_pysol.po index d5449c1c..ca9f7f29 100644 --- a/po/de_pysol.po +++ b/po/de_pysol.po @@ -3,7 +3,7 @@ msgstr "" "Project-Id-Version: PySol 0.0.1\n" "Report-Msgid-Bugs-To: \n" "POT-Creation-Date: 2021-12-10 10:19-0500\n" -"PO-Revision-Date: 2023-06-29 22:25-0400\n" +"PO-Revision-Date: 2023-08-27 12:10-0400\n" "Last-Translator: H. Schaekel \n" "Language-Team: German\n" "Language: de\n" @@ -469,6 +469,9 @@ msgstr "" msgid "Poker" msgstr "" +msgid "Puzzle" +msgstr "" + msgid "Shisen-Sho" msgstr "" @@ -871,6 +874,9 @@ msgstr "Trumpf nur Typ (variable Karten)" msgid "Matching type (variable cards)" msgstr "" +msgid "Puzzle type (variable pieces)" +msgstr "" + #: pysollib/resource.py:194 msgid "French" msgstr "Französisch" diff --git a/po/fr_pysol.po b/po/fr_pysol.po index 8596b878..1a48c563 100644 --- a/po/fr_pysol.po +++ b/po/fr_pysol.po @@ -8,7 +8,7 @@ msgstr "" "Project-Id-Version: 1.02\n" "Report-Msgid-Bugs-To: \n" "POT-Creation-Date: 2021-12-10 10:19-0500\n" -"PO-Revision-Date: 2023-06-29 22:24-0400\n" +"PO-Revision-Date: 2023-08-27 12:14-0400\n" "Last-Translator: Eric Rausch \n" "Language-Team: French\n" "Language: fr\n" @@ -475,6 +475,9 @@ msgstr "" msgid "Poker" msgstr "" +msgid "Puzzle" +msgstr "" + msgid "Shisen-Sho" msgstr "" @@ -901,6 +904,9 @@ msgstr "Type Atouts seulement (cartes variables)" msgid "Matching type (variable cards)" msgstr "" +msgid "Puzzle type (variable pieces)" +msgstr "" + #: pysollib/resource.py:194 msgid "French" msgstr "Français" diff --git a/po/it_pysol.po b/po/it_pysol.po index ca0b269f..99fc1de4 100644 --- a/po/it_pysol.po +++ b/po/it_pysol.po @@ -12,7 +12,7 @@ msgstr "" "Project-Id-Version: it_pysol\n" "Report-Msgid-Bugs-To: \n" "POT-Creation-Date: 2021-12-10 10:19-0500\n" -"PO-Revision-Date: 2023-06-29 22:24-0400\n" +"PO-Revision-Date: 2023-08-27 12:15-0400\n" "Last-Translator: Giuliano Colla \n" "Language-Team: Italiano \n" "Language: it\n" @@ -481,6 +481,9 @@ msgstr "" msgid "Poker" msgstr "" +msgid "Puzzle" +msgstr "" + msgid "Shisen-Sho" msgstr "" @@ -909,6 +912,9 @@ msgstr "Tipo Trionfi (carte variabili)" msgid "Matching type (variable cards)" msgstr "" +msgid "Puzzle type (variable pieces)" +msgstr "" + #: pysollib/resource.py:194 msgid "French" msgstr "Francesi" diff --git a/po/pl_pysol.po b/po/pl_pysol.po index e33fc9f4..671750cc 100644 --- a/po/pl_pysol.po +++ b/po/pl_pysol.po @@ -8,7 +8,7 @@ msgstr "" "Project-Id-Version: PySolFC\n" "Report-Msgid-Bugs-To: \n" "POT-Creation-Date: 2021-12-10 10:19-0500\n" -"PO-Revision-Date: 2023-06-29 22:22-0400\n" +"PO-Revision-Date: 2023-08-27 12:15-0400\n" "Last-Translator: Jerzy Trzeciak \n" "Language-Team: Polish \n" "Language: pl\n" @@ -475,6 +475,9 @@ msgstr "Samotnik" msgid "Poker" msgstr "Poker" +msgid "Puzzle" +msgstr "" + msgid "Shisen-Sho" msgstr "" @@ -906,6 +909,9 @@ msgstr "Typ tylko Atu (różne karty)" msgid "Matching type (variable cards)" msgstr "" +msgid "Puzzle type (variable pieces)" +msgstr "" + #: pysollib/resource.py:194 msgid "French" msgstr "Francuski" diff --git a/po/pt_BR_pysol.po b/po/pt_BR_pysol.po index edfe3eb0..29017ca3 100644 --- a/po/pt_BR_pysol.po +++ b/po/pt_BR_pysol.po @@ -8,7 +8,7 @@ msgstr "" "Project-Id-Version: \n" "Report-Msgid-Bugs-To: \n" "POT-Creation-Date: 2021-12-10 10:19-0500\n" -"PO-Revision-Date: 2023-06-29 22:23-0400\n" +"PO-Revision-Date: 2023-08-27 12:16-0400\n" "Last-Translator: Matheus Knack \n" "Language-Team: \n" "Language: pt_BR\n" @@ -474,6 +474,9 @@ msgstr "Resta um" msgid "Poker" msgstr "Pôquer" +msgid "Puzzle" +msgstr "" + msgid "Shisen-Sho" msgstr "Mahjong Shinsen-Sho" @@ -900,6 +903,9 @@ msgstr "Apenas tipos de trunfos (cartas variáveis)" msgid "Matching type (variable cards)" msgstr "Estilos Combinados (cartas variáveis)" +msgid "Puzzle type (variable pieces)" +msgstr "" + #: pysollib/resource.py:194 msgid "French" msgstr "Francês" diff --git a/po/pysol.pot b/po/pysol.pot index 60f6ec47..668245f9 100644 --- a/po/pysol.pot +++ b/po/pysol.pot @@ -449,6 +449,9 @@ msgstr "" msgid "Poker" msgstr "" +msgid "Puzzle" +msgstr "" + msgid "Shisen-Sho" msgstr "" @@ -815,6 +818,9 @@ msgstr "" msgid "Matching type (variable cards)" msgstr "" +msgid "Puzzle type (variable pieces)" +msgstr "" + #: pysollib/resource.py:194 msgid "French" msgstr "" diff --git a/po/ru_pysol.po b/po/ru_pysol.po index d51b4021..3047fea4 100644 --- a/po/ru_pysol.po +++ b/po/ru_pysol.po @@ -7,7 +7,7 @@ msgstr "" "Project-Id-Version: \n" "Report-Msgid-Bugs-To: \n" "POT-Creation-Date: 2021-12-10 10:19-0500\n" -"PO-Revision-Date: 2023-06-29 22:23-0400\n" +"PO-Revision-Date: 2023-08-27 12:16-0400\n" "Last-Translator: Skomoroh \n" "Language-Team: Russian \n" "Language: ru\n" @@ -478,6 +478,9 @@ msgstr "" msgid "Poker" msgstr "" +msgid "Puzzle" +msgstr "" + msgid "Shisen-Sho" msgstr "" @@ -896,6 +899,9 @@ msgstr "Без мастей (переменное количество карт) msgid "Matching type (variable cards)" msgstr "" +msgid "Puzzle type (variable pieces)" +msgstr "" + #: pysollib/resource.py:194 msgid "French" msgstr "Классические" diff --git a/pysollib/app.py b/pysollib/app.py index f9ae6369..52941829 100644 --- a/pysollib/app.py +++ b/pysollib/app.py @@ -820,6 +820,10 @@ class Application: t0 = "Matching" if cs.ncards < (gi.ncards / 2): # not enough cards t1 = t0 + elif gc == GI.GC_PUZZLE: + t0 = "Puzzle" + if cs_type not in (CSI.TYPE_PUZZLE,) or cs_subtype != gs: + t1 = t0 else: # we should not come here t0 = t1 = "Unknown" diff --git a/pysollib/game/__init__.py b/pysollib/game/__init__.py index 159a5fcb..b0a0c511 100644 --- a/pysollib/game/__init__.py +++ b/pysollib/game/__init__.py @@ -3481,7 +3481,7 @@ class Game(object): # for find_card_dialog def canFindCard(self): - return self.gameinfo.category != GI.GC_MATCHING + return self.gameinfo.category not in (GI.GC_MATCHING, GI.GC_PUZZLE) # # subclass hooks diff --git a/pysollib/gamedb.py b/pysollib/gamedb.py index 4fb35b3d..38633416 100644 --- a/pysollib/gamedb.py +++ b/pysollib/gamedb.py @@ -56,12 +56,21 @@ class GI: GC_DASHAVATARA_GANJIFA = CSI.TYPE_DASHAVATARA_GANJIFA GC_TRUMP_ONLY = CSI.TYPE_TRUMP_ONLY GC_MATCHING = CSI.TYPE_MATCHING + GC_PUZZLE = CSI.TYPE_PUZZLE - NUM_CATEGORIES = CSI.TYPE_MATCHING + NUM_CATEGORIES = CSI.TYPE_PUZZLE # game subcategory GS_NONE = CSI.SUBTYPE_NONE GS_JOKER_DECK = CSI.SUBTYPE_JOKER_DECK + GS_3X3 = CSI.SUBTYPE_3X3 + GS_4X4 = CSI.SUBTYPE_4X4 + GS_5X5 = CSI.SUBTYPE_5X5 + GS_6X6 = CSI.SUBTYPE_6X6 + GS_7X7 = CSI.SUBTYPE_7X7 + GS_8X8 = CSI.SUBTYPE_8X8 + GS_9X9 = CSI.SUBTYPE_9X9 + GS_10X10 = CSI.SUBTYPE_10X10 # game type GT_1DECK_TYPE = 0 @@ -162,6 +171,7 @@ class GI: GT_MEMORY: n_("Memory"), GT_PEGGED: n_("Pegged"), GT_POKER_TYPE: n_("Poker"), + GT_PUZZLE_TYPE: n_("Puzzle"), GT_SHISEN_SHO: n_("Shisen-Sho"), GT_TAROCK: n_("Tarock"), GT_HANOI: n_("Tower of Hanoi"), @@ -565,7 +575,8 @@ 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, 913)) + tuple(range(11017, 11020))), + ('dev', tuple(range(906, 913)) + tuple(range(11017, 11020)) + + tuple(range(22303, 22311)) + tuple(range(22353, 22361))), ) # deprecated - the correct way is to or a GI.GT_XXX flag diff --git a/pysollib/games/ultra/__init__.py b/pysollib/games/ultra/__init__.py index f4f130de..2aa7997c 100644 --- a/pysollib/games/ultra/__init__.py +++ b/pysollib/games/ultra/__init__.py @@ -28,3 +28,4 @@ from . import larasgame # noqa: F401 from . import matrix # noqa: F401 from . import mughal # noqa: F401 from . import tarock # noqa: F401 +from . import tilepuzzle # noqa: F401 diff --git a/pysollib/games/ultra/matrix.py b/pysollib/games/ultra/matrix.py index 1f8bb322..bce2348d 100644 --- a/pysollib/games/ultra/matrix.py +++ b/pysollib/games/ultra/matrix.py @@ -227,6 +227,22 @@ def r(id, short_name, width): return gi +def r2(id, short_name, width): + name = short_name + ncards = width ** 2 + gi = GameInfo( + id, Matrix, name, + GI.GT_MATRIX, 1, 0, GI.SL_SKILL, + category=GI.GC_PUZZLE, short_name=short_name, + subcategory=width, + suits=(), ranks=(), trumps=list(range(ncards)), + si={"decks": 1, "ncards": ncards}) + gi.ncards = ncards + gi.rules_filename = "matrix.html" + registerGame(gi) + return gi + + r(22223, "Matrix 3x3", 3) r(22224, "Matrix 4x4", 4) r(22225, "Matrix 5x5", 5) @@ -237,4 +253,14 @@ r(22229, "Matrix 9x9", 9) r(22230, "Matrix 10x10", 10) # r(22240, "Matrix 20x20", 20) +r2(22303, "Picture Matrix 3x3", 3) +r2(22304, "Picture Matrix 4x4", 4) +r2(22305, "Picture Matrix 5x5", 5) +r2(22306, "Picture Matrix 6x6", 6) +r2(22307, "Picture Matrix 7x7", 7) +r2(22308, "Picture Matrix 8x8", 8) +r2(22309, "Picture Matrix 9x9", 9) +r2(22310, "Picture Matrix 10x10", 10) + del r +del r2 diff --git a/pysollib/games/ultra/tilepuzzle.py b/pysollib/games/ultra/tilepuzzle.py new file mode 100644 index 00000000..abc9dc2e --- /dev/null +++ b/pysollib/games/ultra/tilepuzzle.py @@ -0,0 +1,144 @@ +#!/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 . +# +# ---------------------------------------------------------------------------## + +import math + +from pysollib.game import Game +from pysollib.gamedb import GI, GameInfo, registerGame +from pysollib.layout import Layout +from pysollib.stack import \ + InitialDealTalonStack, \ + InvisibleStack, \ + OpenStack + +# ************************************************************************ +# * Tile Puzzle Row Stack +# ************************************************************************ + + +class TilePuzzle_RowStack(OpenStack): + + def moveMove(self, ncards, to_stack, frames=-1, shadow=-1): + self._swapPairMove(ncards, to_stack, frames=-1, shadow=0) + + def _swapPairMove(self, n, other_stack, frames=-1, shadow=-1): + game = self.game + old_state = game.enterState(game.S_FILL) + swap = game.s.internals[0] + game.moveMove(n, self, swap, frames=0) + game.moveMove(n, other_stack, self, frames=frames, shadow=shadow) + game.moveMove(n, swap, other_stack, frames=0) + game.leaveState(old_state) + + +# ************************************************************************ +# * Tile Puzzle Game +# ************************************************************************ + +class TilePuzzle(Game): + + # + # Game layout + # + + def createGame(self): + l, s = Layout(self), self.s + grid = math.sqrt(self.gameinfo.ncards) + assert grid == int(grid) + grid = int(grid) + + # Set window size + w, h = l.XM * 2 + l.CW * grid, l.YM * 2 + l.CH * grid + self.setSize(w, h) + + # Create rows + for j in range(grid): + x, y = l.XM, l.YM + l.CH * j + for i in range(grid): + s.rows.append(TilePuzzle_RowStack(x, y, self, + max_accept=1, max_cards=2)) + x = x + l.CW + + # Create talon + x, y = -2*l.XS, 0 # invisible + s.talon = InitialDealTalonStack(x, y, self) + + s.internals.append(InvisibleStack(self)) + + # Define stack groups + l.defaultStackGroups() + + # + # Game over rides + # + + def startGame(self): + assert len(self.s.talon.cards) == self.gameinfo.ncards + self.startDealSample() + self.s.talon.dealRow(rows=self.s.rows, frames=3) + + def isGameWon(self): + if self.busy: + return 0 + s = self.s.rows + for r in s[:len(s)]: + if not r.cards[0].rank == r.id: + return 0 + self.s.talon.dealRow(rows=s, frames=0) + return 1 + + def shallHighlightMatch(self, stack1, card1, stack2, card2): + return ((card1.rank + 1 == card2.rank) or + (card1.rank - 1 == card2.rank)) + +# ************************************************************************ +# * Register a Tile Puzzle game +# ************************************************************************ + + +def r(id, short_name, width): + name = short_name + ncards = width ** 2 + gi = GameInfo( + id, TilePuzzle, name, + GI.GT_PUZZLE_TYPE, 1, 0, GI.SL_SKILL, + category=GI.GC_PUZZLE, short_name=short_name, + subcategory=width, + suits=(), ranks=(), trumps=list(range(ncards)), + si={"decks": 1, "ncards": ncards}) + gi.ncards = ncards + gi.rules_filename = "tilepuzzle.html" + registerGame(gi) + return gi + + +r(22353, "Tile Puzzle 3x3", 3) +r(22354, "Tile Puzzle 4x4", 4) +r(22355, "Tile Puzzle 5x5", 5) +r(22356, "Tile Puzzle 6x6", 6) +r(22357, "Tile Puzzle 7x7", 7) +r(22358, "Tile Puzzle 8x8", 8) +r(22359, "Tile Puzzle 9x9", 9) +r(22360, "Tile Puzzle 10x10", 10) + +del r diff --git a/pysollib/options.py b/pysollib/options.py index 974337a2..e66e1f21 100644 --- a/pysollib/options.py +++ b/pysollib/options.py @@ -202,6 +202,14 @@ highlight_piles = float(0.2, 9.9) 8 = string_list(min=2, max=2) 9 = string_list(min=2, max=2) 10 = string_list(min=2, max=2) +11_3 = string_list(min=2, max=2) +11_4 = string_list(min=2, max=2) +11_5 = string_list(min=2, max=2) +11_6 = string_list(min=2, max=2) +11_7 = string_list(min=2, max=2) +11_8 = string_list(min=2, max=2) +11_9 = string_list(min=2, max=2) +11_10 = string_list(min=2, max=2) scale_cards = boolean scale_x = float scale_y = float @@ -560,7 +568,15 @@ class Options: CSI.TYPE_DASHAVATARA_GANJIFA: {0: ("Dashavatara Ganjifa XL", "")}, CSI.TYPE_TRUMP_ONLY: {0: ("Next Matrix", "")}, - CSI.TYPE_MATCHING: {0: ("Neo", "")} + CSI.TYPE_MATCHING: {0: ("Neo", "")}, + CSI.TYPE_PUZZLE: {3: ("Dojouji Ukiyo E (3x3)", ""), + 4: ("Knave of Hearts (4x4)", ""), + 5: ("Victoria Falls (5x5)", ""), + 6: ("Hokusai Ukiyo E (6x6)", ""), + 7: ("Blaren (7x7)", ""), + 8: ("Mid Winter's Eve (8x8)", ""), + 9: ("Hofamterspiel (9x9)", ""), + 10: ("Eternal Dragon (10x10)", "")}, } else: self.cardset = { @@ -576,7 +592,15 @@ class Options: CSI.TYPE_NAVAGRAHA_GANJIFA: {0: ("Dashavatara Ganjifa", "")}, CSI.TYPE_DASHAVATARA_GANJIFA: {0: ("Dashavatara Ganjifa", "")}, CSI.TYPE_TRUMP_ONLY: {0: ("Matrix", "")}, - CSI.TYPE_MATCHING: {0: (c, "")} + CSI.TYPE_MATCHING: {0: (c, "")}, + CSI.TYPE_PUZZLE: {3: ("Dojouji Ukiyo E (3x3)", ""), + 4: ("Knave of Hearts (4x4)", ""), + 5: ("Victoria Falls (5x5)", ""), + 6: ("Hokusai Ukiyo E (6x6)", ""), + 7: ("Blaren (7x7)", ""), + 8: ("Mid Winter's Eve (8x8)", ""), + 9: ("Hofamterspiel (9x9)", ""), + 10: ("Eternal Dragon (10x10)", "")}, } # not changeable options diff --git a/pysollib/resource.py b/pysollib/resource.py index 73651c06..f061e0ba 100644 --- a/pysollib/resource.py +++ b/pysollib/resource.py @@ -181,10 +181,19 @@ class CSI: TYPE_DASHAVATARA_GANJIFA = 8 TYPE_TRUMP_ONLY = 9 TYPE_MATCHING = 10 + TYPE_PUZZLE = 11 # cardset subtypes SUBTYPE_NONE = 0 SUBTYPE_JOKER_DECK = 1 + SUBTYPE_3X3 = 3 + SUBTYPE_4X4 = 4 + SUBTYPE_5X5 = 5 + SUBTYPE_6X6 = 6 + SUBTYPE_7X7 = 7 + SUBTYPE_8X8 = 8 + SUBTYPE_9X9 = 9 + SUBTYPE_10X10 = 10 TYPE = { 1: _("French type (52-54 cards)"), @@ -197,6 +206,7 @@ class CSI: 8: _("Dashavatara Ganjifa type (120 cards)"), 9: _("Trumps only type (variable cards)"), 10: _("Matching type (variable cards)"), + 11: _("Puzzle type (variable pieces)") } TYPE_NAME = { @@ -210,6 +220,7 @@ class CSI: 8: _("Dashavatara Ganjifa"), 9: _("Trumps only"), 10: _("Matching"), + 11: _("Puzzle") } TYPE_ID = { @@ -222,7 +233,8 @@ class CSI: 7: "navagraha-ganjifa", 8: "dashavatara-ganjifa", 9: "trumps-only", - 10: "matching" + 10: "matching", + 11: "puzzle" } TYPE_SUITS = { @@ -235,7 +247,8 @@ class CSI: 7: "abcdefghi", 8: "abcdefghij", 9: "", - 10: "" + 10: "", + 11: "" } TYPE_RANKS = { @@ -249,6 +262,7 @@ class CSI: 8: list(range(12)), 9: list(range(0)), 10: list(range(0)), + 11: list(range(0)), } TYPE_TRUMPS = { @@ -262,6 +276,7 @@ class CSI: 8: (), 9: (), 10: (), + 11: (), } # cardset styles @@ -292,6 +307,7 @@ class CSI: 20: _("Places"), # 21: _("Plain"), # 22: _("Products"), # + 33: _("Puzzle"), # 18: _("Round Cardsets"), # 23: _("Science Fiction"), # 24: _("Sports"), # @@ -503,6 +519,16 @@ class CardsetManager(ResourceManager): cs.nletters = 0 cs.nshadows = 0 cs.trumps = list(range(cs.ncards)) + elif s == CSI.TYPE_PUZZLE: + # ???return 0 ## FIXME + # cs.nbottoms = 7 + # cs.ranks = () + # cs.suits = "" + # cs.trumps = range(cs.ncards) + cs.nbottoms = 1 + cs.nletters = 0 + cs.nshadows = 0 + cs.trumps = list(range(cs.ncards)) else: return 0 diff --git a/scripts/cardset_viewer.py b/scripts/cardset_viewer.py index 04ca11d2..d6a17244 100755 --- a/scripts/cardset_viewer.py +++ b/scripts/cardset_viewer.py @@ -26,7 +26,8 @@ cardset_type = { '7': 'Navagraha Ganjifa', '8': 'Dashavatara Ganjifa', '9': 'Trump only', - '10': 'Matching' + '10': 'Matching', + '11': 'Puzzle' } ALL_IMGS = False