1
0
Fork 0
mirror of https://github.com/shlomif/PySolFC.git synced 2025-04-05 00:02:29 -04:00

Compare commits

...

8 commits

Author SHA1 Message Date
Shlomi Fish
7934babad2 Extract a method or a function.
This is Refactoring / code cleanup.

See:

* https://refactoring.com/catalog/extractMethod.html

* https://en.wikipedia.org/wiki/Code_refactoring

* https://www.refactoring.com/

* https://www.joelonsoftware.com/2002/01/23/rub-a-dub-dub/

Some small optimisations may have slipped in as well.
2019-07-15 17:17:04 +03:00
Shlomi Fish
62005c7c42 Extract a common class/struct.
This is Refactoring / code cleanup.

See:

* https://en.wikipedia.org/wiki/God_object

* https://en.wikipedia.org/wiki/Extract_class

* https://en.wikipedia.org/wiki/Code_refactoring

* https://www.refactoring.com/

* https://www.joelonsoftware.com/2002/01/23/rub-a-dub-dub/

Some small optimisations may have slipped in as well.
2019-07-15 14:17:03 +03:00
Shlomi Fish
c30d131181 install the attrs.py deps. 2019-07-15 12:52:19 +03:00
Shlomi Fish
18efc0e5d3 Extract a common class/struct.
This is Refactoring / code cleanup.

See:

* https://en.wikipedia.org/wiki/God_object

* https://en.wikipedia.org/wiki/Extract_class

* https://en.wikipedia.org/wiki/Code_refactoring

* https://www.refactoring.com/

* https://www.joelonsoftware.com/2002/01/23/rub-a-dub-dub/

Some small optimisations may have slipped in as well.
2019-07-14 22:56:06 +03:00
Juhani Numminen
63c6aa5816 Don't duplicate locales, e.g. de -> de_DE
The duplication seems to be unnecessary. Setting environment variable
LANG=it_CH gives Italian as PySolFC language, even if we didn't explicitly
create 'it_CH'.
2019-07-14 14:02:43 +03:00
Juhani Numminen
5062f0e0d9 Install all translations, not only Russian 2019-07-14 14:02:32 +03:00
Shlomi Fish
0aad4261d8 appveyor fix #1 - os.sep 2019-07-14 08:58:46 +03:00
Juhani Numminen
c13b8587fc
Update pysol.desktop for unix desktop integration
Encoding has been deprecated - see https://specifications.freedesktop.org/desktop-entry-spec/1.0/apc.html

StartupWMClass adds a startup animation. xprop was used to find out the value for WM_CLASS; it was tested on KDE Plasma 5.
2019-07-13 16:53:14 +03:00
7 changed files with 122 additions and 118 deletions

View file

@ -8,15 +8,7 @@ install:
- choco install strawberryperl
- copy %PYTHON%\python.exe %PYTHON%\python3.exe
- SET PATH=%PYTHON%;C:\strawberry\c\bin;C:\strawberry\perl\site\bin;C:\strawberry\perl\bin;%PATH%
- python3 -mpip install Pillow
- python3 -mpip install flake8
- python3 -mpip install flake8-import-order
- python3 -mpip install py2exe
- python3 -mpip install pycotap
- python3 -mpip install pygame
- python3 -mpip install pysol-cards
- python3 -mpip install random2
- python3 -mpip install six
- python3 -mpip install Pillow attrs flake8 flake8-import-order py2exe pycotap pygame pysol-cards random2 six
- perl -v
- copy C:\msys64\mingw64\bin\mingw32-make.exe C:\msys64\mingw64\bin\make.exe
- SET PATH=C:\strawberry\c\bin;C:\strawberry\perl\site\bin;C:\strawberry\perl\bin;%PATH%

View file

@ -39,7 +39,7 @@ before_install:
install:
- sudo cpanm --notest Capture::Tiny
- sudo cpanm Code::TidyAll::Plugin::Flake8 Perl::Tidy Test::Code::TidyAll Test::Differences Test::TrailingSpace
- export PY_MODS='pycotap pysol-cards random2 six'
- export PY_MODS='attrs pycotap pysol-cards random2 six'
- "`which python3` -m pip install --upgrade flake8 flake8-import-order $PY_MODS"
- "sudo /usr/bin/python3 -m pip install --upgrade $PY_MODS || true"
- "sudo `which python2` -m pip install --upgrade $PY_MODS"

View file

@ -47,7 +47,7 @@ pot:
done
mo:
for loc in ru ru_RU de de_AT de_BE de_DE de_LU de_CH pl pl_PL it it_IT; do \
for loc in ru de pl it; do \
test -d locale/$${loc}/LC_MESSAGES || mkdir -p locale/$${loc}/LC_MESSAGES; \
done
for lang in ru pl it; do \
@ -56,12 +56,6 @@ mo:
for lang in ru de pl it; do \
msgfmt -o locale/$${lang}/LC_MESSAGES/pysol.mo po/$${lang}.po; \
done
cp -f locale/ru/LC_MESSAGES/pysol.mo locale/ru_RU/LC_MESSAGES/pysol.mo
for dir in de_AT de_BE de_DE de_LU de_CH; do \
cp -f locale/de/LC_MESSAGES/pysol.mo locale/$${dir}/LC_MESSAGES/pysol.mo; \
done
cp -f locale/pl/LC_MESSAGES/pysol.mo locale/pl_PL/LC_MESSAGES/pysol.mo
cp -f locale/it/LC_MESSAGES/pysol.mo locale/it_IT/LC_MESSAGES/pysol.mo
pretest:
@rm -f tests/individually-importing/*.py # To avoid stray files

View file

@ -1,8 +1,10 @@
[Desktop Entry]
Encoding=UTF-8
Name=PySol Fan Club Edition
Comment=More than 1000 solitaire card games
Exec=pysol.py
StartupWMClass=Pysol
Terminal=false
Type=Application
Categories=Game;CardGame;
Keywords=solitaire;patience;cards;pysolfc;klondike;
Icon=/usr/share/icons/pysol01.png

View file

@ -27,6 +27,8 @@ import time
import traceback
from pickle import Pickler, Unpickler, UnpicklingError
import attr
from pysol_cards.cards import ms_rearrange
from pysollib.game.dump import pysolDumpGame
@ -256,6 +258,65 @@ def _highlightCards__calc_item(canvas, delta, cw, ch, s, c1, c2, color):
return r
@attr.s
class StackGroups(object):
dropstacks = attr.ib(factory=list)
hp_stacks = attr.ib(factory=list) # for getHightlightPilesStacks()
openstacks = attr.ib(factory=list)
reservestacks = attr.ib(factory=list)
talonstacks = attr.ib(factory=list)
def to_tuples(self):
"""docstring for to_tuples"""
self.openstacks = [s for s in self.openstacks
if s.cap.max_accept >= s.cap.min_accept]
self.hp_stacks = [s for s in self.dropstacks
if s.cap.max_move >= 2]
self.openstacks = tuple(self.openstacks)
self.talonstacks = tuple(self.talonstacks)
self.dropstacks = tuple(self.dropstacks)
self.reservestacks = tuple(self.reservestacks)
self.hp_stacks = tuple(self.hp_stacks)
@attr.s
class StackRegions(object):
# list of tuples(stacks, rect)
info = attr.ib(factory=list)
# list of stacks in no region
remaining = attr.ib(factory=list)
data = attr.ib(factory=list)
# init info (at the start)
init_info = attr.ib(factory=list)
def calc_info(self, xf, yf):
"""docstring for calc_info"""
info = []
for stacks, rect in self.init_info:
newrect = (int(round(rect[0]*xf)), int(round(rect[1]*yf)),
int(round(rect[2]*xf)), int(round(rect[3]*yf)))
info.append((stacks, newrect))
self.info = tuple(info)
def optimize(self, remaining):
"""docstring for optimize"""
# sort data by priority
self.data.sort()
self.data.reverse()
# copy (stacks, rect) to info
self.info = []
for d in self.data:
self.info.append((d[2], d[3]))
self.info = tuple(self.info)
# determine remaining stacks
for stacks, rect in self.info:
for stack in stacks:
while stack in remaining:
remaining.remove(stack)
self.remaining = tuple(remaining)
self.init_info = self.info
class Game(object):
# for self.gstats.updated
U_PLAY = 0
@ -305,23 +366,8 @@ class Game(object):
reserves=[],
internals=[],
)
self.sg = Struct( # stack-groups
openstacks=[], # for getClosestStack(): only on
# these stacks the player can place a card
talonstacks=[], # for Hint
dropstacks=[], # for Hint & getAutoStacks()
reservestacks=[], # for Hint
# hint=Struct(), # extra info for class Hint
hp_stacks=[], # for getHightlightPilesStacks()
)
self.regions = Struct( # for getClosestStack()
# set by optimizeRegions():
info=[], # list of tuples(stacks, rect)
remaining=[], # list of stacks in no region
#
data=[], # raw data
init_info=[], # init info (at the start)
)
self.sg = StackGroups()
self.regions = StackRegions()
self.init_size = (0, 0)
self.event_handled = False # if click event handled by Stack (???)
self.reset()
@ -340,10 +386,6 @@ class Game(object):
self.app.intro.progress.update(step=1)
self.createGame()
# set some defaults
self.sg.openstacks = [s for s in self.sg.openstacks
if s.cap.max_accept >= s.cap.min_accept]
self.sg.hp_stacks = [s for s in self.sg.dropstacks
if s.cap.max_move >= 2]
self.createSnGroups()
# convert stackgroups to tuples (speed)
self.allstacks = tuple(self.allstacks)
@ -351,11 +393,7 @@ class Game(object):
self.s.rows = tuple(self.s.rows)
self.s.reserves = tuple(self.s.reserves)
self.s.internals = tuple(self.s.internals)
self.sg.openstacks = tuple(self.sg.openstacks)
self.sg.talonstacks = tuple(self.sg.talonstacks)
self.sg.dropstacks = tuple(self.sg.dropstacks)
self.sg.reservestacks = tuple(self.sg.reservestacks)
self.sg.hp_stacks = tuple(self.sg.hp_stacks)
self.sg.to_tuples()
# init the stack view
for stack in self.allstacks:
stack.prepareStack()
@ -365,7 +403,6 @@ class Game(object):
assert hasattr(self.s.talon, "max_rounds")
if DEBUG:
self._checkGame()
# optimize regions
self.optimizeRegions()
# create cards
if not self.cards:
@ -541,7 +578,6 @@ class Game(object):
for stack in self.allstacks:
stack.prepareStack()
stack.assertStack()
# optimize regions
self.optimizeRegions()
# create cards
self.cards = self.createCards()
@ -905,12 +941,7 @@ class Game(object):
x, y = int(round(x0*xf)), int(round(y0*yf))
stack.resize(xf, yf)
stack.updatePositions()
info = []
for stacks, rect in self.regions.init_info:
rect = (int(round(rect[0]*xf)), int(round(rect[1]*yf)),
int(round(rect[2]*xf)), int(round(rect[3]*yf)))
info.append((stacks, rect))
self.regions.info = tuple(info)
self.regions.calc_info(xf, yf)
# texts
for t in ('info', 'help', 'misc', 'score', 'base_rank'):
init_coord = getattr(self.init_texts, t)
@ -1719,22 +1750,7 @@ class Game(object):
# as getClosestStack() is called within the mouse motion handler
# event it is worth optimizing a little bit
def optimizeRegions(self):
# sort regions.data by priority
self.regions.data.sort()
self.regions.data.reverse()
# copy (stacks, rect) to regions.info
self.regions.info = []
for d in self.regions.data:
self.regions.info.append((d[2], d[3]))
self.regions.info = tuple(self.regions.info)
# determine remaining stacks
remaining = list(self.sg.openstacks)
for stacks, rect in self.regions.info:
for stack in stacks:
while stack in remaining:
remaining.remove(stack)
self.regions.remaining = tuple(remaining)
self.regions.init_info = self.regions.info
return self.regions.optimize(list(self.sg.openstacks))
def getInvisibleCoords(self):
# for InvisibleStack, etc
@ -1861,18 +1877,20 @@ class Game(object):
top_msg = ''
if ret:
if ret[0] and ret[1]:
top_msg = _('''
You have reached
# %d in the %s of playing time
and # %d in the %s of moves.''') % (ret[0], TOP_TITLE, ret[1], TOP_TITLE)
top_msg = _(
'''\nYou have reached\n''' +
'# %d in the %s of playing time' +
'\nand # %d in the %s of moves.') % \
(ret[0], TOP_TITLE, ret[1], TOP_TITLE)
elif ret[0]: # playing time
top_msg = _('''
You have reached
# %d in the %s of playing time.''') % (ret[0], TOP_TITLE)
top_msg = _(
'''\nYou have reached\n''' +
'''# %d in the %s of playing time.''') \
% (ret[0], TOP_TITLE)
elif ret[1]: # moves
top_msg = _('''
You have reached
# %d in the %s of moves.''') % (ret[1], TOP_TITLE)
top_msg = _(
'''\nYou have reached\n''' +
'# %d in the %s of moves.') % (ret[1], TOP_TITLE)
return top_msg
elif not demo:
# only update the session log
@ -1908,16 +1926,12 @@ You have reached
'''Your playing time is %s\nfor %d moves.''',
self.moves.index)
text = text % (time, self.moves.index)
d = MfxMessageDialog(self.top, title=_("Game won"),
text=_('''
Congratulations, this
was a truly perfect game !
%s
%s
''') % (text, top_msg),
strings=(_("&New game"), None, _("&Cancel")),
image=self.app.gimages.logos[5])
d = MfxMessageDialog(
self.top, title=_("Game won"),
text=_('\nCongratulations, this\nwas a truly perfect game !' +
'\n\n%s\n%s\n') % (text, top_msg),
strings=(_("&New game"), None, _("&Cancel")),
image=self.app.gimages.logos[5])
elif status == 1:
top_msg = self.updateStats()
time = self.getTime()
@ -1928,15 +1942,14 @@ was a truly perfect game !
'''Your playing time is %s\nfor %d moves.''',
self.moves.index)
text = text % (time, self.moves.index)
d = MfxMessageDialog(self.top, title=_("Game won"),
text=_('''
Congratulations, you did it !
%s
%s
''') % (text, top_msg),
strings=(_("&New game"), None, _("&Cancel")),
image=self.app.gimages.logos[4])
d = MfxMessageDialog(
self.top, title=_("Game won"),
text=(
_('\nCongratulations, you did it !\n\n%s\n%s\n') %
(text, top_msg)
),
strings=(_("&New game"), None, _("&Cancel")),
image=self.app.gimages.logos[4])
elif self.gstats.updated < 0:
self.finished = True
self.playSample("gamefinished", priority=1000)
@ -3007,11 +3020,10 @@ Congratulations, you did it !
self.setCursor(cursor=self.app.top_cursor)
MfxMessageDialog(
self.top, title=_("Load game error"), bitmap="error",
text=_("""\
Error while loading game.
Probably the game file is damaged,
but this could also be a bug you might want to report."""))
text=_(
"Error while loading game.\n\n" +
"Probably the game file is damaged,\n" +
"but this could also be a bug you might want to report."))
traceback.print_exc()
except UnpicklingError as ex:
self.updateMenus()
@ -3021,11 +3033,11 @@ but this could also be a bug you might want to report."""))
except Exception:
self.updateMenus()
self.setCursor(cursor=self.app.top_cursor)
MfxMessageDialog(self.top, title=_("Load game error"),
bitmap="error", text=_("""\
Internal error while loading game.
Please report this bug."""))
MfxMessageDialog(
self.top, title=_("Load game error"),
bitmap="error", text=_(
"""Internal error while loading game.\n\n""" +
"Please report this bug."))
traceback.print_exc()
else:
if self.pause:
@ -3088,9 +3100,11 @@ Please report this bug."""))
version = pload(str)
# validate(isinstance(version, str) and len(version) <= 20, err_txt)
version_tuple = pload(tuple)
validate(version_tuple >= (1, 0), _('''\
Cannot load games saved with
%s version %s''') % (PACKAGE, version))
validate(
version_tuple >= (1, 0),
_('''Cannot load games saved with\n%s version %s''') % (
PACKAGE,
version))
game_version = 1
bookmark = pload(int)
validate(0 <= bookmark <= 2, err_txt)
@ -3105,10 +3119,11 @@ Cannot load games saved with
if not game.canLoadGame(version_tuple, game_version):
destruct(game)
game = None
validate(game is not None, _('''\
Cannot load this game from version %s
as the game rules have changed
in the current implementation.''') % version)
validate(
game is not None,
_('Cannot load this game from version %s\n' +
'as the game rules have changed\n' +
'in the current implementation.') % version)
game.version = version
game.version_tuple = version_tuple
#

View file

@ -14,7 +14,8 @@ for d, _, files in os.walk("pysollib"):
for f in files:
if re.search("\\.py$", f):
module_names.append(
(d + "/" + re.sub("\\.py$", "", f)).replace("/", "."))
(d + "/" + re.sub("\\.py$", "", f))
.replace("/", ".").replace(os.sep, "."))
module_names.sort()
for module_name in module_names:

View file

@ -3,6 +3,7 @@
import os
from distutils.core import setup
from glob import glob
from pysollib.settings import PACKAGE_URL
from pysollib.settings import VERSION
@ -45,9 +46,8 @@ if os.name == 'posix':
data_files.append(('share/icons',
['data/images/misc/pysol01.png',
'data/images/misc/pysol02.png', ]))
for l in ('ru', 'ru_RU'):
data_files.append(('share/locale/%s/LC_MESSAGES' % l,
['locale/%s/LC_MESSAGES/pysol.mo' % l]))
for mofile in glob('locale/*/*/*.mo'):
data_files.append(('share/' + os.path.dirname(mofile), [mofile]))
data_files.append((data_dir, ['data/pysolfc.glade']))
data_files.append(('share/applications', ['data/pysol.desktop']))