mirror of
https://github.com/shlomif/PySolFC.git
synced 2025-04-05 00:02:29 -04:00
Compare commits
8 commits
9bf8f475bb
...
9af86cddfb
Author | SHA1 | Date | |
---|---|---|---|
|
9af86cddfb | ||
|
90323173a7 | ||
|
9e148b88fe | ||
|
da256a30b0 | ||
|
cb25f39136 | ||
|
4cb94702f3 | ||
|
436bca2852 | ||
|
918f177acc |
10 changed files with 189 additions and 114 deletions
|
@ -33,3 +33,18 @@ Otherwise, the games' sources live under
|
||||||
written in Python 2.7/3.x and you can try inheriting from an existing
|
written in Python 2.7/3.x and you can try inheriting from an existing
|
||||||
variant [class](https://en.wikipedia.org/wiki/Class_%28computer_programming%29).
|
variant [class](https://en.wikipedia.org/wiki/Class_%28computer_programming%29).
|
||||||
|
|
||||||
|
# The Release Process
|
||||||
|
|
||||||
|
In order to publish a new version, follow these steps:
|
||||||
|
|
||||||
|
1. Update `NEWS.asciidoc`.
|
||||||
|
2. Update the `VERSION_TUPLE =` line in `pysollib/settings.py`.
|
||||||
|
3. Test using `gmake test` .
|
||||||
|
4. `git commit` the changes .
|
||||||
|
5. `git tag pysolfc-2.6.5` (or equivalent version).
|
||||||
|
6. `git push` and `git push --tags` to https://github.com/shlomif/PySolFC .
|
||||||
|
7. Wait for the AppVeyor build for the tag to complete and scan the .exe using https://www.virustotal.com/ .
|
||||||
|
8. Run `gmake dist`.
|
||||||
|
9. Use [rexz9](https://github.com/shlomif/shlomif-computer-settings/blob/567b6ab3f4272ad66bf331536dc80bf58bfff3af/shlomif-settings/bash-aliases/user_aliases.bash#L57) on `dist/PySol*.tar.xz`.
|
||||||
|
10. Go to https://sourceforge.net/projects/pysolfc/files/PySolFC/ and add a folder called PySolFC-2.6.5 (note the capitalisation).
|
||||||
|
11. Add the tar.xz and the .exe there and mark them as defaults for the right OSes.
|
||||||
|
|
|
@ -4,7 +4,8 @@
|
||||||
## code
|
## code
|
||||||
##
|
##
|
||||||
include pysol.py setup.py setup_osx.py setup.cfg MANIFEST.in Makefile
|
include pysol.py setup.py setup_osx.py setup.cfg MANIFEST.in Makefile
|
||||||
include COPYING README.md AUTHORS README.android README.kivy
|
include COPYING README.md AUTHORS.md README.android README.kivy
|
||||||
|
include NEWS.asciidoc
|
||||||
#recursive-include pysollib *.py
|
#recursive-include pysollib *.py
|
||||||
include pysollib/*.py pysollib/macosx/*.py pysollib/configobj/*.py
|
include pysollib/*.py pysollib/macosx/*.py pysollib/configobj/*.py
|
||||||
include pysollib/winsystems/*.py
|
include pysollib/winsystems/*.py
|
||||||
|
|
|
@ -1,6 +1,12 @@
|
||||||
[[news]]
|
[[news]]
|
||||||
=== News
|
=== News
|
||||||
|
|
||||||
|
* _25 April, 2019:_ There is a new stable release
|
||||||
|
https://sourceforge.net/projects/pysolfc/files/PySolFC/PySolFC-2.6.4/[PySolFC
|
||||||
|
v2.6.4]. New in this release:
|
||||||
|
** A https://github.com/shlomif/PySolFC/issues/107[new logo] thanks to @ghostofiht.
|
||||||
|
** Fix for https://github.com/shlomif/PySolFC/issues/110[Freezes] (pygame upgrade)
|
||||||
|
** Fixed https://github.com/shlomif/PySolFC/issues/111[Three Peaks scoring].
|
||||||
* _7 April, 2019:_ There is a new stable release
|
* _7 April, 2019:_ There is a new stable release
|
||||||
https://sourceforge.net/projects/pysolfc/files/PySolFC/PySolFC-2.6.3/[PySolFC
|
https://sourceforge.net/projects/pysolfc/files/PySolFC/PySolFC-2.6.3/[PySolFC
|
||||||
v2.6.3]. New in this release:
|
v2.6.3]. New in this release:
|
||||||
|
|
121
pysollib/app.py
121
pysollib/app.py
|
@ -25,9 +25,12 @@
|
||||||
# imports
|
# imports
|
||||||
import os
|
import os
|
||||||
import re
|
import re
|
||||||
|
import sys
|
||||||
import traceback
|
import traceback
|
||||||
from pickle import UnpicklingError
|
from pickle import UnpicklingError
|
||||||
|
|
||||||
|
from pysollib.app_stat import GameStat
|
||||||
|
from pysollib.app_stat_result import GameStatResult
|
||||||
from pysollib.gamedb import GAME_DB, GI, loadGame
|
from pysollib.gamedb import GAME_DB, GI, loadGame
|
||||||
from pysollib.images import Images, SubsampledImages
|
from pysollib.images import Images, SubsampledImages
|
||||||
from pysollib.mfxutil import Struct, destruct
|
from pysollib.mfxutil import Struct, destruct
|
||||||
|
@ -37,7 +40,7 @@ from pysollib.mfxutil import latin1_normalize, print_err
|
||||||
from pysollib.mfxutil import pickle, unpickle
|
from pysollib.mfxutil import pickle, unpickle
|
||||||
from pysollib.mygettext import _
|
from pysollib.mygettext import _
|
||||||
from pysollib.options import Options
|
from pysollib.options import Options
|
||||||
from pysollib.pysolrandom import PysolRandom
|
from pysollib.pysolrandom import PysolRandom, constructRandom
|
||||||
from pysollib.pysoltk import HTMLViewer
|
from pysollib.pysoltk import HTMLViewer
|
||||||
from pysollib.pysoltk import HelpStatusbar, PysolStatusbar
|
from pysollib.pysoltk import HelpStatusbar, PysolStatusbar
|
||||||
from pysollib.pysoltk import MfxDialog, MfxExceptionDialog, MfxMessageDialog
|
from pysollib.pysoltk import MfxDialog, MfxExceptionDialog, MfxMessageDialog
|
||||||
|
@ -53,7 +56,7 @@ from pysollib.resource import Sample, SampleManager
|
||||||
from pysollib.resource import Tile, TileManager
|
from pysollib.resource import Tile, TileManager
|
||||||
from pysollib.settings import DEBUG
|
from pysollib.settings import DEBUG
|
||||||
from pysollib.settings import PACKAGE, VERSION_TUPLE, WIN_SYSTEM
|
from pysollib.settings import PACKAGE, VERSION_TUPLE, WIN_SYSTEM
|
||||||
from pysollib.settings import TOOLKIT, TOP_SIZE
|
from pysollib.settings import TOOLKIT
|
||||||
from pysollib.util import CARDSET, IMAGE_EXTENSIONS
|
from pysollib.util import CARDSET, IMAGE_EXTENSIONS
|
||||||
from pysollib.winsystems import TkSettings
|
from pysollib.winsystems import TkSettings
|
||||||
if TOOLKIT == 'tk':
|
if TOOLKIT == 'tk':
|
||||||
|
@ -70,112 +73,7 @@ if True: # This prevents from travis 'error' E402.
|
||||||
# ************************************************************************
|
# ************************************************************************
|
||||||
# * Statistics
|
# * Statistics
|
||||||
# ************************************************************************
|
# ************************************************************************
|
||||||
|
_GameStatResult = GameStatResult
|
||||||
|
|
||||||
class _GameStatResult:
|
|
||||||
def __init__(self):
|
|
||||||
self.min = 0
|
|
||||||
self.max = 0
|
|
||||||
self.top = []
|
|
||||||
self.num = 0
|
|
||||||
self.total = 0 # sum of all values
|
|
||||||
self.average = 0
|
|
||||||
|
|
||||||
def update(self, gameid, value, game_number, game_start_time):
|
|
||||||
# update min & max
|
|
||||||
if not self.min or value < self.min:
|
|
||||||
self.min = value
|
|
||||||
if not self.max or value > self.max:
|
|
||||||
self.max = value
|
|
||||||
# calculate position & update top
|
|
||||||
position = None
|
|
||||||
n = 0
|
|
||||||
for i in self.top:
|
|
||||||
if value < i.value:
|
|
||||||
position = n+1
|
|
||||||
v = Struct(gameid=gameid,
|
|
||||||
value=value,
|
|
||||||
game_number=game_number,
|
|
||||||
game_start_time=game_start_time)
|
|
||||||
self.top.insert(n, v)
|
|
||||||
del self.top[TOP_SIZE:]
|
|
||||||
break
|
|
||||||
n += 1
|
|
||||||
if not position and len(self.top) < TOP_SIZE:
|
|
||||||
v = Struct(gameid=gameid,
|
|
||||||
value=value,
|
|
||||||
game_number=game_number,
|
|
||||||
game_start_time=game_start_time)
|
|
||||||
self.top.append(v)
|
|
||||||
position = len(self.top)
|
|
||||||
# update average
|
|
||||||
self.total += value
|
|
||||||
self.num += 1
|
|
||||||
self.average = float(self.total)/self.num
|
|
||||||
return position
|
|
||||||
|
|
||||||
|
|
||||||
class GameStat:
|
|
||||||
def __init__(self, id):
|
|
||||||
self.gameid = id
|
|
||||||
#
|
|
||||||
self.num_total = 0
|
|
||||||
# self.num_not_won = 0
|
|
||||||
self.num_lost = 0
|
|
||||||
self.num_won = 0
|
|
||||||
self.num_perfect = 0
|
|
||||||
#
|
|
||||||
self.time_result = _GameStatResult()
|
|
||||||
self.moves_result = _GameStatResult()
|
|
||||||
self.total_moves_result = _GameStatResult()
|
|
||||||
self.score_result = _GameStatResult()
|
|
||||||
self.score_casino_result = _GameStatResult()
|
|
||||||
|
|
||||||
def update(self, game, status):
|
|
||||||
#
|
|
||||||
game_number = game.getGameNumber(format=0)
|
|
||||||
game_start_time = game.gstats.start_time
|
|
||||||
# update number of games
|
|
||||||
# status:
|
|
||||||
# -1 - NOT WON (not played)
|
|
||||||
# 0 - LOST
|
|
||||||
# 1 - WON
|
|
||||||
# 2 - PERFECT
|
|
||||||
self.num_total += 1
|
|
||||||
assert status in (0, 1, 2)
|
|
||||||
if status == 0:
|
|
||||||
self.num_lost += 1
|
|
||||||
return
|
|
||||||
elif status == 1:
|
|
||||||
self.num_won += 1
|
|
||||||
else: # status == 2
|
|
||||||
self.num_perfect += 1
|
|
||||||
|
|
||||||
score = game.getGameScore()
|
|
||||||
# print 'GameScore:', score
|
|
||||||
score_p = None
|
|
||||||
if score is not None:
|
|
||||||
score_p = self.score_result.update(
|
|
||||||
game.id, score, game_number, game_start_time)
|
|
||||||
score = game.getGameScoreCasino()
|
|
||||||
# print 'GameScoreCasino:', score
|
|
||||||
score_casino_p = None
|
|
||||||
if score is not None:
|
|
||||||
score_casino_p = self.score_casino_result.update(
|
|
||||||
game.id, score, game_number, game_start_time)
|
|
||||||
|
|
||||||
if status == 0:
|
|
||||||
return
|
|
||||||
|
|
||||||
game.updateTime()
|
|
||||||
time_p = self.time_result.update(
|
|
||||||
game.id, game.stats.elapsed_time, game_number, game_start_time)
|
|
||||||
moves_p = self.moves_result.update(
|
|
||||||
game.id, game.moves.index, game_number, game_start_time)
|
|
||||||
total_moves_p = self.total_moves_result.update(
|
|
||||||
game.id, game.stats.total_moves, game_number, game_start_time)
|
|
||||||
|
|
||||||
return time_p, moves_p, total_moves_p, score_p, score_casino_p
|
|
||||||
|
|
||||||
|
|
||||||
class Statistics:
|
class Statistics:
|
||||||
|
@ -418,6 +316,7 @@ class Application:
|
||||||
loadgame=None, # load a game ?
|
loadgame=None, # load a game ?
|
||||||
game=None,
|
game=None,
|
||||||
gameid=None,
|
gameid=None,
|
||||||
|
deal=None,
|
||||||
)
|
)
|
||||||
self.demo_counter = 0
|
self.demo_counter = 0
|
||||||
|
|
||||||
|
@ -487,8 +386,12 @@ class Application:
|
||||||
gameid = self.gdb.getGameByName(self.commandline.game)
|
gameid = self.gdb.getGameByName(self.commandline.game)
|
||||||
if gameid is None:
|
if gameid is None:
|
||||||
print_err(_("can't find game: ") + self.commandline.game)
|
print_err(_("can't find game: ") + self.commandline.game)
|
||||||
|
sys.exit(-1)
|
||||||
else:
|
else:
|
||||||
self.nextgame.id, self.nextgame.random = gameid, None
|
self.nextgame.id = gameid
|
||||||
|
deal = self.commandline.deal
|
||||||
|
self.nextgame.random = \
|
||||||
|
None if deal is None else constructRandom(deal)
|
||||||
elif self.commandline.gameid is not None:
|
elif self.commandline.gameid is not None:
|
||||||
self.nextgame.id, self.nextgame.random = \
|
self.nextgame.id, self.nextgame.random = \
|
||||||
self.commandline.gameid, None
|
self.commandline.gameid, None
|
||||||
|
|
82
pysollib/app_stat.py
Normal file
82
pysollib/app_stat.py
Normal file
|
@ -0,0 +1,82 @@
|
||||||
|
# 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 <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
|
||||||
|
from pysollib.app_stat_result import GameStatResult
|
||||||
|
|
||||||
|
|
||||||
|
class GameStat:
|
||||||
|
def __init__(self, id):
|
||||||
|
self.gameid = id
|
||||||
|
#
|
||||||
|
self.num_total = 0
|
||||||
|
# self.num_not_won = 0
|
||||||
|
self.num_lost = 0
|
||||||
|
self.num_won = 0
|
||||||
|
self.num_perfect = 0
|
||||||
|
#
|
||||||
|
self.time_result = GameStatResult()
|
||||||
|
self.moves_result = GameStatResult()
|
||||||
|
self.total_moves_result = GameStatResult()
|
||||||
|
self.score_result = GameStatResult()
|
||||||
|
self.score_casino_result = GameStatResult()
|
||||||
|
|
||||||
|
def update(self, game, status):
|
||||||
|
#
|
||||||
|
game_number = game.getGameNumber(format=0)
|
||||||
|
game_start_time = game.gstats.start_time
|
||||||
|
# update number of games
|
||||||
|
# status:
|
||||||
|
# -1 - NOT WON (not played)
|
||||||
|
# 0 - LOST
|
||||||
|
# 1 - WON
|
||||||
|
# 2 - PERFECT
|
||||||
|
self.num_total += 1
|
||||||
|
assert status in (0, 1, 2)
|
||||||
|
if status == 0:
|
||||||
|
self.num_lost += 1
|
||||||
|
return
|
||||||
|
elif status == 1:
|
||||||
|
self.num_won += 1
|
||||||
|
else: # status == 2
|
||||||
|
self.num_perfect += 1
|
||||||
|
|
||||||
|
score = game.getGameScore()
|
||||||
|
# print 'GameScore:', score
|
||||||
|
score_p = None
|
||||||
|
if score is not None:
|
||||||
|
score_p = self.score_result.update(
|
||||||
|
game.id, score, game_number, game_start_time)
|
||||||
|
score = game.getGameScoreCasino()
|
||||||
|
# print 'GameScoreCasino:', score
|
||||||
|
score_casino_p = None
|
||||||
|
if score is not None:
|
||||||
|
score_casino_p = self.score_casino_result.update(
|
||||||
|
game.id, score, game_number, game_start_time)
|
||||||
|
|
||||||
|
if status == 0:
|
||||||
|
return
|
||||||
|
|
||||||
|
game.updateTime()
|
||||||
|
time_p = self.time_result.update(
|
||||||
|
game.id, game.stats.elapsed_time, game_number, game_start_time)
|
||||||
|
moves_p = self.moves_result.update(
|
||||||
|
game.id, game.moves.index, game_number, game_start_time)
|
||||||
|
total_moves_p = self.total_moves_result.update(
|
||||||
|
game.id, game.stats.total_moves, game_number, game_start_time)
|
||||||
|
|
||||||
|
return time_p, moves_p, total_moves_p, score_p, score_casino_p
|
62
pysollib/app_stat_result.py
Normal file
62
pysollib/app_stat_result.py
Normal file
|
@ -0,0 +1,62 @@
|
||||||
|
# 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 <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
from pysollib.mfxutil import Struct
|
||||||
|
from pysollib.settings import TOP_SIZE
|
||||||
|
|
||||||
|
|
||||||
|
class GameStatResult:
|
||||||
|
def __init__(self):
|
||||||
|
self.min = 0
|
||||||
|
self.max = 0
|
||||||
|
self.top = []
|
||||||
|
self.num = 0
|
||||||
|
self.total = 0 # sum of all values
|
||||||
|
self.average = 0
|
||||||
|
|
||||||
|
def update(self, gameid, value, game_number, game_start_time):
|
||||||
|
# update min & max
|
||||||
|
if not self.min or value < self.min:
|
||||||
|
self.min = value
|
||||||
|
if not self.max or value > self.max:
|
||||||
|
self.max = value
|
||||||
|
# calculate position & update top
|
||||||
|
position = None
|
||||||
|
n = 0
|
||||||
|
for i in self.top:
|
||||||
|
if value < i.value:
|
||||||
|
position = n+1
|
||||||
|
v = Struct(gameid=gameid,
|
||||||
|
value=value,
|
||||||
|
game_number=game_number,
|
||||||
|
game_start_time=game_start_time)
|
||||||
|
self.top.insert(n, v)
|
||||||
|
del self.top[TOP_SIZE:]
|
||||||
|
break
|
||||||
|
n += 1
|
||||||
|
if not position and len(self.top) < TOP_SIZE:
|
||||||
|
v = Struct(gameid=gameid,
|
||||||
|
value=value,
|
||||||
|
game_number=game_number,
|
||||||
|
game_start_time=game_start_time)
|
||||||
|
self.top.append(v)
|
||||||
|
position = len(self.top)
|
||||||
|
# update average
|
||||||
|
self.total += value
|
||||||
|
self.num += 1
|
||||||
|
self.average = float(self.total)/self.num
|
||||||
|
return position
|
|
@ -86,7 +86,7 @@ def parse_option(argv):
|
||||||
prog_name = argv[0]
|
prog_name = argv[0]
|
||||||
try:
|
try:
|
||||||
optlist, args = getopt.getopt(argv[1:], "g:i:hD:",
|
optlist, args = getopt.getopt(argv[1:], "g:i:hD:",
|
||||||
["game=", "gameid=",
|
["deal=", "game=", "gameid=",
|
||||||
"french-only",
|
"french-only",
|
||||||
"noplugins",
|
"noplugins",
|
||||||
"nosound",
|
"nosound",
|
||||||
|
@ -97,6 +97,7 @@ def parse_option(argv):
|
||||||
(err, prog_name), 0)
|
(err, prog_name), 0)
|
||||||
return None
|
return None
|
||||||
opts = {"help": False,
|
opts = {"help": False,
|
||||||
|
"deal": None,
|
||||||
"game": None,
|
"game": None,
|
||||||
"gameid": None,
|
"gameid": None,
|
||||||
"french-only": False,
|
"french-only": False,
|
||||||
|
@ -107,6 +108,8 @@ def parse_option(argv):
|
||||||
for i in optlist:
|
for i in optlist:
|
||||||
if i[0] in ("-h", "--help"):
|
if i[0] in ("-h", "--help"):
|
||||||
opts["help"] = True
|
opts["help"] = True
|
||||||
|
elif i[0] in ("--deal"):
|
||||||
|
opts["deal"] = i[1]
|
||||||
elif i[0] in ("-g", "--game"):
|
elif i[0] in ("-g", "--game"):
|
||||||
opts["game"] = i[1]
|
opts["game"] = i[1]
|
||||||
elif i[0] in ("-i", "--gameid"):
|
elif i[0] in ("-i", "--gameid"):
|
||||||
|
@ -164,6 +167,7 @@ def pysol_init(app, args):
|
||||||
opts, filename = opts
|
opts, filename = opts
|
||||||
if filename:
|
if filename:
|
||||||
app.commandline.loadgame = filename
|
app.commandline.loadgame = filename
|
||||||
|
app.commandline.deal = opts['deal']
|
||||||
app.commandline.game = opts['game']
|
app.commandline.game = opts['game']
|
||||||
if opts['gameid'] is not None:
|
if opts['gameid'] is not None:
|
||||||
try:
|
try:
|
||||||
|
|
|
@ -107,9 +107,9 @@ def print_err(s, level=1):
|
||||||
elif level == 2:
|
elif level == 2:
|
||||||
ss = PACKAGE+': DEBUG WARNING:'
|
ss = PACKAGE+': DEBUG WARNING:'
|
||||||
try:
|
try:
|
||||||
print_(ss, s.encode(locale.getpreferredencoding()), file=sys.stderr)
|
|
||||||
except Exception:
|
|
||||||
print_(ss, s, file=sys.stderr)
|
print_(ss, s, file=sys.stderr)
|
||||||
|
except Exception:
|
||||||
|
print_(ss, s.encode(locale.getpreferredencoding()), file=sys.stderr)
|
||||||
sys.stderr.flush()
|
sys.stderr.flush()
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -32,7 +32,7 @@ PACKAGE = 'PySolFC'
|
||||||
TITLE = 'PySol'
|
TITLE = 'PySol'
|
||||||
PACKAGE_URL = 'http://pysolfc.sourceforge.net/'
|
PACKAGE_URL = 'http://pysolfc.sourceforge.net/'
|
||||||
|
|
||||||
VERSION_TUPLE = (2, 6, 3)
|
VERSION_TUPLE = (2, 6, 4)
|
||||||
VERSION = '.'.join(map(str, VERSION_TUPLE))
|
VERSION = '.'.join(map(str, VERSION_TUPLE))
|
||||||
|
|
||||||
# Tk windowing system (auto set up in init.py)
|
# Tk windowing system (auto set up in init.py)
|
||||||
|
|
|
@ -14,6 +14,8 @@ for module_name in \
|
||||||
'pysollib.acard',
|
'pysollib.acard',
|
||||||
'pysollib.actions',
|
'pysollib.actions',
|
||||||
'pysollib.app',
|
'pysollib.app',
|
||||||
|
'pysollib.app_stat',
|
||||||
|
'pysollib.app_stat_result',
|
||||||
'pysollib.configobj.configobj',
|
'pysollib.configobj.configobj',
|
||||||
'pysollib.configobj.validate',
|
'pysollib.configobj.validate',
|
||||||
'pysollib.customgame',
|
'pysollib.customgame',
|
||||||
|
|
Loading…
Add table
Reference in a new issue