mirror of
https://github.com/shlomif/PySolFC.git
synced 2025-04-05 00:02:29 -04:00
2381 lines
84 KiB
Python
2381 lines
84 KiB
Python
#!/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
|
|
# Copyright (C) 2017 LB
|
|
#
|
|
# 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/>.
|
|
#
|
|
# ---------------------------------------------------------------------------#
|
|
|
|
import math
|
|
import os
|
|
import re
|
|
|
|
from kivy.event import EventDispatcher
|
|
from kivy.properties import BooleanProperty
|
|
from kivy.properties import NumericProperty
|
|
from kivy.properties import StringProperty
|
|
|
|
from pysollib.gamedb import GI
|
|
from pysollib.kivy.LApp import LMenu
|
|
from pysollib.kivy.LApp import LMenuItem
|
|
from pysollib.kivy.LApp import LScrollView
|
|
from pysollib.kivy.LApp import LTopLevel
|
|
from pysollib.kivy.LApp import LTreeNode
|
|
from pysollib.kivy.LApp import LTreeRoot
|
|
from pysollib.kivy.findcarddialog import destroy_find_card_dialog
|
|
from pysollib.kivy.selectcardset import SelectCardsetDialogWithPreview
|
|
from pysollib.kivy.selectgame import SelectGameDialog
|
|
from pysollib.kivy.solverdialog import connect_game_solver_dialog
|
|
from pysollib.kivy.tkconst import CURSOR_WATCH, EVENT_HANDLED, EVENT_PROPAGATE
|
|
from pysollib.kivy.tkconst import TOOLBAR_BUTTONS
|
|
from pysollib.kivy.tkutil import after_idle
|
|
from pysollib.kivy.tkutil import bind
|
|
from pysollib.mfxutil import Struct
|
|
from pysollib.mygettext import _
|
|
from pysollib.pysoltk import MfxMessageDialog
|
|
from pysollib.pysoltk import connect_game_find_card_dialog
|
|
from pysollib.settings import SELECT_GAME_MENU
|
|
from pysollib.settings import TITLE
|
|
|
|
|
|
# ************************************************************************
|
|
# * tk emuls:
|
|
# ************************************************************************
|
|
|
|
|
|
class TkVarObj(EventDispatcher):
|
|
def __init(self):
|
|
self.value = None
|
|
|
|
def set(self, v):
|
|
if v is None:
|
|
if type(self.value) is str:
|
|
v = ''
|
|
self.value = v
|
|
|
|
def get(self):
|
|
return self.value
|
|
|
|
|
|
class BooleanVar(TkVarObj):
|
|
value = BooleanProperty(False)
|
|
|
|
|
|
class IntVar(TkVarObj):
|
|
value = NumericProperty(0)
|
|
|
|
|
|
class StringVar(TkVarObj):
|
|
value = StringProperty('')
|
|
|
|
# ************************************************************************
|
|
# * Menu Dialogs
|
|
# ************************************************************************
|
|
|
|
|
|
class LMenuDialog(object):
|
|
|
|
dialogCache = {}
|
|
|
|
def make_pop_command(self, parent, title):
|
|
def pop_command(event):
|
|
print('event = %s' % event)
|
|
parent.popWork(title)
|
|
return pop_command
|
|
|
|
def __init__(self, menubar, parent, title, app, **kw):
|
|
super(LMenuDialog, self).__init__()
|
|
|
|
self.menubar = menubar
|
|
self.parent = parent
|
|
self.app = app
|
|
self.title = title
|
|
self.window = None
|
|
self.running = False
|
|
self.persist = False
|
|
if 'persist' in kw:
|
|
self.persist = kw['persist']
|
|
|
|
# prüfen ob noch aktiv - toggle.
|
|
|
|
if parent.workStack.peek(title) is not None:
|
|
parent.popWork(title)
|
|
return
|
|
|
|
if self.persist and title in self.dialogCache:
|
|
parent.pushWork(title, self.dialogCache[title])
|
|
return
|
|
|
|
pc = self.closeWindow = self.make_pop_command(parent, title)
|
|
|
|
# neuen Dialog aufbauen.
|
|
|
|
window = LTopLevel(parent, title, **kw)
|
|
window.titleline.bind(on_press=pc)
|
|
self.parent.pushWork(title, window)
|
|
self.window = window
|
|
self.running = True
|
|
|
|
if self.persist:
|
|
self.dialogCache[title] = window
|
|
|
|
# Tree skelett.
|
|
|
|
tv = self.tvroot = LTreeRoot(root_options=dict(text='EditTree'))
|
|
tv.hide_root = True
|
|
tv.size_hint = 1, None
|
|
tv.bind(minimum_height=tv.setter('height'))
|
|
|
|
# menupunkte aufbauen.
|
|
|
|
self.buildTree(tv, None)
|
|
|
|
# tree in einem Scrollwindow präsentieren.
|
|
|
|
root = LScrollView(pos=(0, 0))
|
|
root.add_widget(tv)
|
|
self.window.content.add_widget(root)
|
|
|
|
def buildTree(self, tree, node):
|
|
print('buildTree base')
|
|
# to implement in dervied class
|
|
pass
|
|
|
|
# ************************************************************************
|
|
|
|
|
|
class MainMenuDialog(LMenuDialog):
|
|
|
|
def __init__(self, menubar, parent, title, app, **kw):
|
|
kw['size_hint'] = (0.2, 1)
|
|
kw['persist'] = True
|
|
super(MainMenuDialog, self).__init__(
|
|
menubar, parent, title, app, **kw)
|
|
|
|
def make_game_command(self, command):
|
|
def game_command():
|
|
command()
|
|
self.closeWindow(0)
|
|
return game_command
|
|
|
|
def buildTree(self, tv, node):
|
|
rg = tv.add_node(
|
|
LTreeNode(
|
|
text=_("File"),
|
|
command=self.make_game_command(self.menubar.mFileMenuDialog)))
|
|
rg = tv.add_node(
|
|
LTreeNode(
|
|
text=_("Games"),
|
|
command=self.make_game_command(
|
|
self.menubar.mSelectGameDialog)))
|
|
rg = tv.add_node(
|
|
LTreeNode(
|
|
text=_("Tools"),
|
|
command=self.make_game_command(self.menubar.mEditMenuDialog)))
|
|
rg = tv.add_node(
|
|
LTreeNode(
|
|
text=_("Statistics"),
|
|
command=self.make_game_command(self.menubar.mGameMenuDialog)))
|
|
rg = tv.add_node(
|
|
LTreeNode(
|
|
text=_("Assist"),
|
|
command=self.make_game_command(
|
|
self.menubar.mAssistMenuDialog)))
|
|
rg = tv.add_node(
|
|
LTreeNode(
|
|
text=_("Options"),
|
|
command=self.make_game_command(
|
|
self.menubar.mOptionsMenuDialog)))
|
|
rg = tv.add_node(
|
|
LTreeNode(
|
|
text=_("Help"),
|
|
command=self.make_game_command(self.menubar.mHelpMenuDialog)))
|
|
del rg
|
|
|
|
# ************************************************************************
|
|
|
|
|
|
class FileMenuDialog(LMenuDialog):
|
|
|
|
def __init__(self, menubar, parent, title, app, **kw):
|
|
kw['size_hint'] = (0.3, 1)
|
|
super(FileMenuDialog, self).__init__(
|
|
menubar, parent, title, app, **kw)
|
|
|
|
def make_game_command(self, key, command):
|
|
def game_command():
|
|
command(key)
|
|
return game_command
|
|
|
|
def buildTree(self, tv, node):
|
|
rg = tv.add_node(
|
|
LTreeNode(text=_('Recent games')))
|
|
# Recent Liste
|
|
recids = self.app.opt.recent_gameid
|
|
# recgames = []
|
|
for rid in recids:
|
|
gi = self.app.getGameInfo(rid)
|
|
if gi:
|
|
command = self.make_game_command(
|
|
rid, self.menubar._mSelectGame)
|
|
tv.add_node(
|
|
LTreeNode(text=gi.name, command=command), rg)
|
|
|
|
rg = tv.add_node(
|
|
LTreeNode(text=_('Favorite games')))
|
|
if rg:
|
|
tv.add_node(LTreeNode(
|
|
text=_('<Add>'), command=self.menubar.mAddFavor), rg)
|
|
tv.add_node(LTreeNode(
|
|
text=_('<Remove>'), command=self.menubar.mDelFavor), rg)
|
|
|
|
# Recent Liste
|
|
favids = self.app.opt.favorite_gameid
|
|
# favgames = []
|
|
for fid in favids:
|
|
gi = self.app.getGameInfo(fid)
|
|
if gi:
|
|
command = self.make_game_command(
|
|
fid, self.menubar._mSelectGame)
|
|
tv.add_node(
|
|
LTreeNode(text=gi.name, command=command), rg)
|
|
|
|
tv.add_node(LTreeNode(
|
|
text=_('Load'), command=self.menubar.mOpen))
|
|
tv.add_node(LTreeNode(
|
|
text=_('Save'), command=self.menubar.mSaveAs))
|
|
|
|
tv.add_node(LTreeNode(
|
|
text=_('Quit'), command=self.menubar.mHoldAndQuit))
|
|
|
|
# ************************************************************************
|
|
|
|
|
|
class EditMenuDialog(LMenuDialog): # Tools
|
|
|
|
def __init__(self, menubar, parent, title, app, **kw):
|
|
kw['size_hint'] = (0.2, 1)
|
|
kw['persist'] = True
|
|
super(EditMenuDialog, self).__init__(
|
|
menubar, parent, title, app, **kw)
|
|
|
|
def make_auto_command(self, variable, command):
|
|
def auto_command():
|
|
variable.set(not variable.get())
|
|
command()
|
|
return auto_command
|
|
|
|
def addCheckNode(self, tv, rg, title, auto_var, auto_com):
|
|
command = self.make_auto_command(auto_var, auto_com)
|
|
rg1 = tv.add_node(
|
|
LTreeNode(text=title, command=command, variable=auto_var), rg)
|
|
return rg1
|
|
|
|
def buildTree(self, tv, node):
|
|
tv.add_node(LTreeNode(
|
|
text=_('New game'), command=self.menubar.mNewGame))
|
|
tv.add_node(LTreeNode(
|
|
text=_('Restart game'), command=self.menubar.mRestart))
|
|
|
|
tv.add_node(LTreeNode(
|
|
text=_('Undo'), command=self.menubar.mUndo))
|
|
tv.add_node(LTreeNode(
|
|
text=_('Redo'), command=self.menubar.mRedo))
|
|
tv.add_node(LTreeNode(
|
|
text=_('Redo all'), command=self.menubar.mRedoAll))
|
|
|
|
tv.add_node(LTreeNode(
|
|
text=_('Auto drop'), command=self.menubar.mDrop))
|
|
tv.add_node(LTreeNode(
|
|
text=_('Shuffle tiles'), command=self.menubar.mShuffle))
|
|
tv.add_node(LTreeNode(
|
|
text=_('Deal cards'), command=self.menubar.mDeal))
|
|
|
|
self.addCheckNode(tv, None,
|
|
_('Pause'),
|
|
self.menubar.tkopt.pause,
|
|
self.menubar.mPause)
|
|
|
|
tv.add_node(LTreeNode(
|
|
text=_('Load game'), command=self.menubar.mOpen))
|
|
tv.add_node(LTreeNode(
|
|
text=_('Save game'), command=self.menubar.mSaveAs))
|
|
|
|
tv.add_node(LTreeNode(
|
|
text=_('Help'), command=self.menubar.mHelpRules))
|
|
|
|
# -------------------------------------------
|
|
# TBD ?
|
|
'''
|
|
menu.add_separator()
|
|
submenu = MfxMenu(menu, label=n_("&Set bookmark"))
|
|
for i in range(9):
|
|
label = _("Bookmark %d") % (i + 1)
|
|
submenu.add_command(
|
|
label=label, command=lambda i=i: self.mSetBookmark(i))
|
|
submenu = MfxMenu(menu, label=n_("Go&to bookmark"))
|
|
for i in range(9):
|
|
label = _("Bookmark %d") % (i + 1)
|
|
acc = m + "%d" % (i + 1)
|
|
submenu.add_command(
|
|
label=label,
|
|
command=lambda i=i: self.mGotoBookmark(i),
|
|
accelerator=acc)
|
|
menu.add_command(
|
|
label=n_("&Clear bookmarks"), command=self.mClearBookmarks)
|
|
menu.add_separator()
|
|
'''
|
|
# und solitär wizard (-> custom games).
|
|
'''
|
|
tv.add_node(LTreeNode(
|
|
text='Solitaire &Wizard', command=self.menubar.mWizard))
|
|
tv.add_node(LTreeNode(
|
|
text='Edit current game', command=self.menubar.mWizardEdit))
|
|
'''
|
|
|
|
# ************************************************************************
|
|
|
|
|
|
class GameMenuDialog(LMenuDialog):
|
|
|
|
def __init__(self, menubar, parent, title, app, **kw):
|
|
kw['size_hint'] = (0.2, 1)
|
|
kw['persist'] = True
|
|
super(GameMenuDialog, self).__init__(
|
|
menubar, parent, title, app, **kw)
|
|
|
|
def make_command(self, key, command):
|
|
def stats_command():
|
|
kw = {}
|
|
kw['mode'] = key
|
|
command(**kw)
|
|
return stats_command
|
|
|
|
def buildTree(self, tv, node):
|
|
tv.add_node(LTreeNode(
|
|
text=_('Current game...'),
|
|
command=self.make_command(101, self.menubar.mPlayerStats)), None)
|
|
|
|
# tv.add_node(LTreeNode(
|
|
# text='All games ...',
|
|
# command=self.make_command(102, self.menubar.mPlayerStats)), None)
|
|
|
|
# -------------------------------------------
|
|
# TBD ? - just to remember original tk code.
|
|
'''
|
|
menu.add_command(
|
|
label=n_("S&tatus..."),
|
|
command=lambda x: self.mPlayerStats(mode=100), accelerator=m+"Y")
|
|
menu.add_checkbutton(
|
|
label=n_("&Comments..."), variable=self.tkopt.comment,
|
|
command=self.mEditGameComment)
|
|
'''
|
|
'''
|
|
menu.add_separator()
|
|
submenu = MfxMenu(menu, label=n_("&Statistics"))
|
|
submenu.add_command(
|
|
label=n_("Current game..."),
|
|
command=lambda x: self.mPlayerStats(mode=101))
|
|
submenu.add_command(
|
|
label=n_("All games..."),
|
|
command=lambda x: self.mPlayerStats(mode=102))
|
|
submenu.add_separator()
|
|
submenu.add_command(
|
|
label=n_("Session log..."),
|
|
command=lambda x: self.mPlayerStats(mode=104))
|
|
submenu.add_command(
|
|
label=n_("Full log..."),
|
|
command=lambda x: self.mPlayerStats(mode=103))
|
|
submenu.add_separator()
|
|
submenu.add_command(
|
|
label=TOP_TITLE+"...",
|
|
command=lambda x: self.mPlayerStats(mode=105),
|
|
accelerator=m+"T")
|
|
submenu.add_command(
|
|
label=n_("Progression..."),
|
|
command=lambda x: self.mPlayerStats(mode=107))
|
|
submenu = MfxMenu(menu, label=n_("D&emo statistics"))
|
|
submenu.add_command(
|
|
label=n_("Current game..."),
|
|
command=lambda x: self.mPlayerStats(mode=1101))
|
|
submenu.add_command(
|
|
label=n_("All games..."),
|
|
command=lambda x: self.mPlayerStats(mode=1102))
|
|
'''
|
|
|
|
# ************************************************************************
|
|
|
|
|
|
class AssistMenuDialog(LMenuDialog):
|
|
|
|
def __init__(self, menubar, parent, title, app, **kw):
|
|
kw['size_hint'] = (0.2, 1)
|
|
kw['persist'] = True
|
|
super(AssistMenuDialog, self).__init__(
|
|
menubar, parent, title, app, **kw)
|
|
|
|
def buildTree(self, tv, node):
|
|
tv.add_node(LTreeNode(
|
|
text=_('Hint'), command=self.menubar.mHint))
|
|
|
|
tv.add_node(LTreeNode(
|
|
text=_('Highlight piles'), command=self.menubar.mHighlightPiles))
|
|
|
|
# tv.add_node(LTreeNode(
|
|
# text='Find Card', command=self.menubar.mFindCard))
|
|
|
|
tv.add_node(LTreeNode(
|
|
text=_('Demo'), command=self.menubar.mDemo))
|
|
|
|
# -------------------------------------------
|
|
# TBD. How ?
|
|
|
|
'''
|
|
menu.add_command(
|
|
label=n_("Demo (&all games)"), command=self.mMixedDemo)
|
|
if USE_FREECELL_SOLVER:
|
|
menu.add_command(label=n_("&Solver"), command=self.mSolver)
|
|
else:
|
|
menu.add_command(label=n_("&Solver"), state='disabled')
|
|
menu.add_separator()
|
|
menu.add_command(
|
|
label=n_("&Piles description"),
|
|
command=self.mStackDesk, accelerator="F2")
|
|
'''
|
|
|
|
# ************************************************************************
|
|
|
|
|
|
class OptionsMenuDialog(LMenuDialog):
|
|
|
|
def __init__(self, menubar, parent, title, app, **kw):
|
|
kw['size_hint'] = (0.5, 1)
|
|
kw['persist'] = True
|
|
super(OptionsMenuDialog, self).__init__(
|
|
menubar, parent, title, app, **kw)
|
|
|
|
def make_auto_command(self, variable, command):
|
|
def auto_command():
|
|
variable.set(not variable.get())
|
|
command()
|
|
return auto_command
|
|
|
|
def addCheckNode(self, tv, rg, title, auto_var, auto_com):
|
|
command = self.make_auto_command(auto_var, auto_com)
|
|
rg1 = tv.add_node(
|
|
LTreeNode(text=title, command=command, variable=auto_var), rg)
|
|
return rg1
|
|
|
|
def make_val_command(self, variable, value, command):
|
|
def val_command():
|
|
variable.set(value)
|
|
command()
|
|
return val_command
|
|
|
|
def make_vars_command(self, command, key):
|
|
def vars_command():
|
|
command(key)
|
|
return vars_command
|
|
|
|
def addRadioNode(self, tv, rg, title, auto_var, auto_val, auto_com):
|
|
command = self.make_val_command(auto_var, auto_val, auto_com)
|
|
rg1 = tv.add_node(
|
|
LTreeNode(text=title,
|
|
command=command,
|
|
variable=auto_var, value=auto_val), rg)
|
|
return rg1
|
|
|
|
def buildTree(self, tv, node):
|
|
|
|
# -------------------------------------------
|
|
# Automatic play settings
|
|
|
|
rg = tv.add_node(
|
|
LTreeNode(text=_('Automatic play')))
|
|
if rg:
|
|
self.addCheckNode(tv, rg,
|
|
_('Auto face up'),
|
|
self.menubar.tkopt.autofaceup,
|
|
self.menubar.mOptAutoFaceUp)
|
|
|
|
self.addCheckNode(tv, rg,
|
|
_('Auto drop'),
|
|
self.menubar.tkopt.autodrop,
|
|
self.menubar.mOptAutoDrop)
|
|
|
|
self.addCheckNode(tv, rg,
|
|
_('Auto deal'),
|
|
self.menubar.tkopt.autodeal,
|
|
self.menubar.mOptAutoDeal)
|
|
|
|
# submenu.add_separator()
|
|
|
|
self.addCheckNode(tv, rg,
|
|
_('Quick play'),
|
|
self.menubar.tkopt.quickplay,
|
|
self.menubar.mOptQuickPlay)
|
|
|
|
# -------------------------------------------
|
|
# Player assistance
|
|
|
|
rg = tv.add_node(
|
|
LTreeNode(text=_('Assist level')))
|
|
if rg:
|
|
self.addCheckNode(tv, rg,
|
|
_('Enable undo'),
|
|
self.menubar.tkopt.undo,
|
|
self.menubar.mOptEnableUndo)
|
|
|
|
self.addCheckNode(tv, rg,
|
|
_('Enable bookmarks'),
|
|
self.menubar.tkopt.bookmarks,
|
|
self.menubar.mOptEnableBookmarks)
|
|
|
|
self.addCheckNode(tv, rg,
|
|
_('Enable hint'),
|
|
self.menubar.tkopt.hint,
|
|
self.menubar.mOptEnableHint)
|
|
|
|
self.addCheckNode(tv, rg,
|
|
_('Enable shuffle'),
|
|
self.menubar.tkopt.shuffle,
|
|
self.menubar.mOptEnableShuffle)
|
|
|
|
self.addCheckNode(tv, rg,
|
|
_('Enable highlight piles'),
|
|
self.menubar.tkopt.highlight_piles,
|
|
self.menubar.mOptEnableHighlightPiles)
|
|
|
|
self.addCheckNode(tv, rg,
|
|
_('Enable highlight cards'),
|
|
self.menubar.tkopt.highlight_cards,
|
|
self.menubar.mOptEnableHighlightCards)
|
|
|
|
self.addCheckNode(tv, rg,
|
|
_('Enable highlight same rank'),
|
|
self.menubar.tkopt.highlight_samerank,
|
|
self.menubar.mOptEnableHighlightSameRank)
|
|
|
|
self.addCheckNode(tv, rg,
|
|
_('Highlight no matching'),
|
|
self.menubar.tkopt.highlight_not_matching,
|
|
self.menubar.mOptEnableHighlightNotMatching)
|
|
|
|
# submenu.add_separator()
|
|
|
|
self.addCheckNode(tv, rg,
|
|
_('Show removed tiles (in Mahjongg games)'),
|
|
self.menubar.tkopt.mahjongg_show_removed,
|
|
self.menubar.mOptMahjonggShowRemoved)
|
|
|
|
self.addCheckNode(tv, rg,
|
|
_('Show hint arrow (in Shisen-Sho games)'),
|
|
self.menubar.tkopt.shisen_show_hint,
|
|
self.menubar.mOptShisenShowHint)
|
|
|
|
self.addCheckNode(tv, rg,
|
|
_('Deal all cards (in Accordion type games)'),
|
|
self.menubar.tkopt.accordion_deal_all,
|
|
self.menubar.mOptAccordionDealAll)
|
|
|
|
# submenu.add_separator()
|
|
|
|
# -------------------------------------------
|
|
# Language options
|
|
|
|
rg = tv.add_node(
|
|
LTreeNode(text=_('Language')))
|
|
if rg:
|
|
self.addRadioNode(tv, rg,
|
|
_('Default'),
|
|
self.menubar.tkopt.language, '',
|
|
self.menubar.mOptLanguage)
|
|
self.addRadioNode(tv, rg,
|
|
_('English'),
|
|
self.menubar.tkopt.language, 'en',
|
|
self.menubar.mOptLanguage)
|
|
self.addRadioNode(tv, rg,
|
|
_('German'),
|
|
self.menubar.tkopt.language, 'de',
|
|
self.menubar.mOptLanguage)
|
|
self.addRadioNode(tv, rg,
|
|
_('Italian'),
|
|
self.menubar.tkopt.language, 'it',
|
|
self.menubar.mOptLanguage)
|
|
self.addRadioNode(tv, rg,
|
|
_('Polish'),
|
|
self.menubar.tkopt.language, 'pl',
|
|
self.menubar.mOptLanguage)
|
|
self.addRadioNode(tv, rg,
|
|
_('Russian'),
|
|
self.menubar.tkopt.language, 'ru',
|
|
self.menubar.mOptLanguage)
|
|
|
|
# -------------------------------------------
|
|
# Sound options
|
|
|
|
rg = tv.add_node(
|
|
LTreeNode(text=_('Sound')))
|
|
if rg:
|
|
self.addCheckNode(tv, rg,
|
|
_('Enable'),
|
|
self.menubar.tkopt.sound,
|
|
self.menubar.mOptSoundDialog)
|
|
|
|
rg1 = tv.add_node(
|
|
LTreeNode(text=_('Volume')), rg)
|
|
if rg1:
|
|
self.addRadioNode(tv, rg1,
|
|
_('100%'),
|
|
self.menubar.tkopt.sound_sample_volume, 100,
|
|
self.menubar.mOptSoundSampleVol)
|
|
self.addRadioNode(tv, rg1,
|
|
_('75%'),
|
|
self.menubar.tkopt.sound_sample_volume, 75,
|
|
self.menubar.mOptSoundSampleVol)
|
|
self.addRadioNode(tv, rg1,
|
|
_('50%'),
|
|
self.menubar.tkopt.sound_sample_volume, 50,
|
|
self.menubar.mOptSoundSampleVol)
|
|
self.addRadioNode(tv, rg1,
|
|
_('25%'),
|
|
self.menubar.tkopt.sound_sample_volume, 25,
|
|
self.menubar.mOptSoundSampleVol)
|
|
|
|
rg1 = tv.add_node(
|
|
LTreeNode(text=_('Samples')), rg)
|
|
if rg1:
|
|
key = 'areyousure'
|
|
self.addCheckNode(
|
|
tv, rg1,
|
|
_('are you sure'),
|
|
self.menubar.tkopt.sound_sample_vars[key],
|
|
self.make_vars_command(self.menubar.mOptSoundSample, key))
|
|
key = 'autodrop'
|
|
self.addCheckNode(
|
|
tv, rg1,
|
|
_('auto drop'),
|
|
self.menubar.tkopt.sound_sample_vars[key],
|
|
self.make_vars_command(self.menubar.mOptSoundSample, key))
|
|
key = 'autoflip'
|
|
self.addCheckNode(
|
|
tv, rg1,
|
|
_('auto flip'),
|
|
self.menubar.tkopt.sound_sample_vars[key],
|
|
self.make_vars_command(self.menubar.mOptSoundSample, key))
|
|
key = 'autopilotlost'
|
|
self.addCheckNode(
|
|
tv, rg1,
|
|
_('auto pilot lost'),
|
|
self.menubar.tkopt.sound_sample_vars[key],
|
|
self.make_vars_command(self.menubar.mOptSoundSample, key))
|
|
key = 'autopilotwon'
|
|
self.addCheckNode(
|
|
tv, rg1,
|
|
_('auto pilot won'),
|
|
self.menubar.tkopt.sound_sample_vars[key],
|
|
self.make_vars_command(self.menubar.mOptSoundSample, key))
|
|
key = 'deal'
|
|
self.addCheckNode(
|
|
tv, rg1,
|
|
_('deal'),
|
|
self.menubar.tkopt.sound_sample_vars[key],
|
|
self.make_vars_command(self.menubar.mOptSoundSample, key))
|
|
key = 'dealwaste'
|
|
self.addCheckNode(
|
|
tv, rg1,
|
|
_('deal waste'),
|
|
self.menubar.tkopt.sound_sample_vars[key],
|
|
self.make_vars_command(self.menubar.mOptSoundSample, key))
|
|
key = 'droppair'
|
|
self.addCheckNode(
|
|
tv, rg1,
|
|
_('drop pair'),
|
|
self.menubar.tkopt.sound_sample_vars[key],
|
|
self.make_vars_command(self.menubar.mOptSoundSample, key))
|
|
key = 'drop'
|
|
self.addCheckNode(
|
|
tv, rg1,
|
|
_('drop'),
|
|
self.menubar.tkopt.sound_sample_vars[key],
|
|
self.make_vars_command(self.menubar.mOptSoundSample, key))
|
|
key = 'flip'
|
|
self.addCheckNode(
|
|
tv, rg1,
|
|
_('flip'),
|
|
self.menubar.tkopt.sound_sample_vars[key],
|
|
self.make_vars_command(self.menubar.mOptSoundSample, key))
|
|
key = 'move'
|
|
self.addCheckNode(
|
|
tv, rg1,
|
|
_('move'),
|
|
self.menubar.tkopt.sound_sample_vars[key],
|
|
self.make_vars_command(self.menubar.mOptSoundSample, key))
|
|
key = 'nomove'
|
|
self.addCheckNode(
|
|
tv, rg1,
|
|
_('no move'),
|
|
self.menubar.tkopt.sound_sample_vars[key],
|
|
self.make_vars_command(self.menubar.mOptSoundSample, key))
|
|
key = 'redo'
|
|
self.addCheckNode(
|
|
tv, rg1,
|
|
_('redo'),
|
|
self.menubar.tkopt.sound_sample_vars[key],
|
|
self.make_vars_command(self.menubar.mOptSoundSample, key))
|
|
key = 'startdrag'
|
|
self.addCheckNode(
|
|
tv, rg1,
|
|
_('start drag'),
|
|
self.menubar.tkopt.sound_sample_vars[key],
|
|
self.make_vars_command(self.menubar.mOptSoundSample, key))
|
|
key = 'turnwaste'
|
|
self.addCheckNode(
|
|
tv, rg1,
|
|
_('turn waste'),
|
|
self.menubar.tkopt.sound_sample_vars[key],
|
|
self.make_vars_command(self.menubar.mOptSoundSample, key))
|
|
key = 'undo'
|
|
self.addCheckNode(
|
|
tv, rg1,
|
|
_('undo'),
|
|
self.menubar.tkopt.sound_sample_vars[key],
|
|
self.make_vars_command(self.menubar.mOptSoundSample, key))
|
|
key = 'gamefinished'
|
|
self.addCheckNode(
|
|
tv, rg1,
|
|
_('game finished'),
|
|
self.menubar.tkopt.sound_sample_vars[key],
|
|
self.make_vars_command(self.menubar.mOptSoundSample, key))
|
|
key = 'gamelost'
|
|
self.addCheckNode(
|
|
tv, rg1,
|
|
_('game lost'),
|
|
self.menubar.tkopt.sound_sample_vars[key],
|
|
self.make_vars_command(self.menubar.mOptSoundSample, key))
|
|
key = 'gameperfect'
|
|
self.addCheckNode(
|
|
tv, rg1,
|
|
_('game perfect'),
|
|
self.menubar.tkopt.sound_sample_vars[key],
|
|
self.make_vars_command(self.menubar.mOptSoundSample, key))
|
|
key = 'gamewon'
|
|
self.addCheckNode(
|
|
tv, rg1,
|
|
_('game won'),
|
|
self.menubar.tkopt.sound_sample_vars[key],
|
|
self.make_vars_command(self.menubar.mOptSoundSample, key))
|
|
key = 'extra'
|
|
self.addCheckNode(
|
|
tv, rg1,
|
|
_('Other'),
|
|
self.menubar.tkopt.sound_sample_vars[key],
|
|
self.make_vars_command(self.menubar.mOptSoundSample, key))
|
|
|
|
# -------------------------------------------
|
|
# Cardsets and card backside options
|
|
|
|
rg = tv.add_node(
|
|
LTreeNode(text=_('Cardsets')))
|
|
if rg:
|
|
self.menubar.tkopt.cardset.set(self.app.cardset.index)
|
|
|
|
csm = self.app.cardset_manager
|
|
# cnt = csm.len()
|
|
i = 0
|
|
while 1:
|
|
cs = csm.get(i)
|
|
if cs is None:
|
|
break
|
|
rg1 = self.addRadioNode(tv, rg,
|
|
cs.name,
|
|
self.menubar.tkopt.cardset, i,
|
|
self.menubar.mOptCardset)
|
|
if rg1:
|
|
cbs = cs.backnames
|
|
self.menubar.tkopt.cardbacks[i] = IntVar()
|
|
self.menubar.tkopt.cardbacks[i].set(cs.backindex)
|
|
|
|
bcnt = len(cbs)
|
|
bi = 0
|
|
while 1:
|
|
if bi == bcnt:
|
|
break
|
|
cb = cbs[bi]
|
|
self.addRadioNode(
|
|
tv, rg1,
|
|
cb,
|
|
self.menubar.tkopt.cardbacks[i], bi,
|
|
self.make_vars_command(
|
|
self.menubar.mOptSetCardback, i))
|
|
bi += 1
|
|
|
|
i += 1
|
|
|
|
# -------------------------------------------
|
|
# Table background settings
|
|
|
|
rg = tv.add_node(
|
|
LTreeNode(text=_('Table')))
|
|
if rg:
|
|
rg1 = tv.add_node(
|
|
LTreeNode(text=_('Solid colors')), rg)
|
|
if rg1:
|
|
key = 'table'
|
|
self.addRadioNode(
|
|
tv, rg1,
|
|
_('Blue'),
|
|
self.menubar.tkopt.color_vars[key], '#0082df',
|
|
self.menubar.mOptTableColor)
|
|
self.addRadioNode(
|
|
tv, rg1,
|
|
_('Green'),
|
|
self.menubar.tkopt.color_vars[key], '#008200',
|
|
self.menubar.mOptTableColor)
|
|
self.addRadioNode(
|
|
tv, rg1,
|
|
_('Navy'),
|
|
self.menubar.tkopt.color_vars[key], '#000086',
|
|
self.menubar.mOptTableColor)
|
|
self.addRadioNode(
|
|
tv, rg1,
|
|
_('Olive'),
|
|
self.menubar.tkopt.color_vars[key], '#868200',
|
|
self.menubar.mOptTableColor)
|
|
self.addRadioNode(
|
|
tv, rg1,
|
|
_('Orange'),
|
|
self.menubar.tkopt.color_vars[key], '#f79600',
|
|
self.menubar.mOptTableColor)
|
|
self.addRadioNode(
|
|
tv, rg1,
|
|
_('Teal'),
|
|
self.menubar.tkopt.color_vars[key], '#008286',
|
|
self.menubar.mOptTableColor)
|
|
|
|
rg1 = tv.add_node(
|
|
LTreeNode(text=_('Tiles and Images')), rg)
|
|
|
|
if rg1:
|
|
tm = self.app.tabletile_manager
|
|
# cnt = tm.len()
|
|
i = 1
|
|
while True:
|
|
ti = tm.get(i)
|
|
if ti is None:
|
|
break
|
|
self.addRadioNode(tv, rg1,
|
|
ti.name,
|
|
self.menubar.tkopt.tabletile, i,
|
|
self.menubar.mOptTileSet)
|
|
i += 1
|
|
|
|
# -------------------------------------------
|
|
# Card view options
|
|
|
|
rg = tv.add_node(
|
|
LTreeNode(text=_('Card view')))
|
|
if rg:
|
|
self.addCheckNode(tv, rg,
|
|
_('Card shadow'),
|
|
self.menubar.tkopt.shadow,
|
|
self.menubar.mOptShadow)
|
|
|
|
self.addCheckNode(tv, rg,
|
|
_('Shade legal moves'),
|
|
self.menubar.tkopt.shade,
|
|
self.menubar.mOptShade)
|
|
|
|
self.addCheckNode(tv, rg,
|
|
_('Negative cards bottom'),
|
|
self.menubar.tkopt.negative_bottom,
|
|
self.menubar.mOptNegativeBottom)
|
|
|
|
self.addCheckNode(tv, rg,
|
|
_('Shrink face-down cards'),
|
|
self.menubar.tkopt.shrink_face_down,
|
|
self.menubar.mOptShrinkFaceDown)
|
|
|
|
self.addCheckNode(tv, rg,
|
|
_('Shade filled stacks'),
|
|
self.menubar.tkopt.shade_filled_stacks,
|
|
self.menubar.mOptShadeFilledStacks)
|
|
|
|
# -------------------------------------------
|
|
# Animation settins
|
|
|
|
rg = tv.add_node(
|
|
LTreeNode(text=_('Animations')))
|
|
if rg:
|
|
self.addRadioNode(tv, rg,
|
|
_('None'),
|
|
self.menubar.tkopt.animations, 0,
|
|
self.menubar.mOptAnimations)
|
|
|
|
self.addRadioNode(tv, rg,
|
|
_('Very fast'),
|
|
self.menubar.tkopt.animations, 1,
|
|
self.menubar.mOptAnimations)
|
|
|
|
self.addRadioNode(tv, rg,
|
|
_('Fast'),
|
|
self.menubar.tkopt.animations, 2,
|
|
self.menubar.mOptAnimations)
|
|
|
|
self.addRadioNode(tv, rg,
|
|
_('Medium'),
|
|
self.menubar.tkopt.animations, 3,
|
|
self.menubar.mOptAnimations)
|
|
|
|
self.addRadioNode(tv, rg,
|
|
_('Slow'),
|
|
self.menubar.tkopt.animations, 4,
|
|
self.menubar.mOptAnimations)
|
|
|
|
self.addRadioNode(tv, rg,
|
|
_('Very slow'),
|
|
self.menubar.tkopt.animations, 5,
|
|
self.menubar.mOptAnimations)
|
|
|
|
# submenu.add_separator()
|
|
|
|
self.addCheckNode(tv, rg,
|
|
_('Redeal animation'),
|
|
self.menubar.tkopt.redeal_animation,
|
|
self.menubar.mRedealAnimation)
|
|
|
|
self.addCheckNode(tv, rg,
|
|
_('Winning animation'),
|
|
self.menubar.tkopt.win_animation,
|
|
self.menubar.mWinAnimation)
|
|
|
|
# -------------------------------------------
|
|
# Touch mode settings
|
|
|
|
rg = tv.add_node(
|
|
LTreeNode(text=_('Touch mode')))
|
|
if rg:
|
|
self.addRadioNode(tv, rg,
|
|
_('Drag-and-Drop'),
|
|
self.menubar.tkopt.mouse_type, 'drag-n-drop',
|
|
self.menubar.mOptMouseType)
|
|
|
|
self.addRadioNode(tv, rg,
|
|
_('Point-and-Click'),
|
|
self.menubar.tkopt.mouse_type, 'point-n-click',
|
|
self.menubar.mOptMouseType)
|
|
|
|
# sinnlos mit touch-device:
|
|
# self.addRadioNode(tv, rg,
|
|
# 'Sticky mouse',
|
|
# self.menubar.tkopt.mouse_type, u'sticky-mouse',
|
|
# self.menubar.mOptMouseType)
|
|
|
|
# submenu.add_separator()
|
|
|
|
# sinnlos mit touch-device:
|
|
# self.addCheckNode(tv, rg,
|
|
# 'Use mouse for undo/redo',
|
|
# self.menubar.tkopt.mouse_undo,
|
|
# self.menubar.mOptMouseUndo)
|
|
|
|
# submenu.add_separator()
|
|
|
|
# -------------------------------------------
|
|
# TBD ?
|
|
|
|
'''
|
|
menu.add_command(label=n_("&Fonts..."), command=self.mOptFonts)
|
|
menu.add_command(label=n_("&Colors..."), command=self.mOptColors)
|
|
menu.add_command(label=n_("Time&outs..."), command=self.mOptTimeouts)
|
|
menu.add_separator()
|
|
'''
|
|
|
|
# -------------------------------------------
|
|
# Toolbar options
|
|
|
|
rg = tv.add_node(
|
|
LTreeNode(text=_('Toolbar')))
|
|
if rg:
|
|
self.addRadioNode(tv, rg,
|
|
_('Hide'),
|
|
self.menubar.tkopt.toolbar, 0,
|
|
self.menubar.mOptToolbar)
|
|
|
|
# not supported: Top, Bottom
|
|
# self.addRadioNode(tv, rg,
|
|
# 'Top',
|
|
# self.menubar.tkopt.toolbar, 1,
|
|
# self.menubar.mOptToolbar)
|
|
# self.addRadioNode(tv, rg,
|
|
# 'Bottom',
|
|
# self.menubar.tkopt.toolbar, 2,
|
|
# self.menubar.mOptToolbar)
|
|
|
|
self.addRadioNode(tv, rg,
|
|
_('Left'),
|
|
self.menubar.tkopt.toolbar, 3,
|
|
self.menubar.mOptToolbar)
|
|
self.addRadioNode(tv, rg,
|
|
_('Right'),
|
|
self.menubar.tkopt.toolbar, 4,
|
|
self.menubar.mOptToolbar)
|
|
|
|
# -------------------------------------------
|
|
# Statusbar - not implemented
|
|
|
|
'''
|
|
submenu = MfxMenu(menu, label=n_("Stat&usbar"))
|
|
submenu.add_checkbutton(
|
|
label=n_("Show &statusbar"),
|
|
variable=self.tkopt.statusbar,
|
|
command=self.mOptStatusbar)
|
|
submenu.add_checkbutton(
|
|
label=n_("Show &number of cards"),
|
|
variable=self.tkopt.num_cards,
|
|
command=self.mOptNumCards)
|
|
submenu.add_checkbutton(
|
|
label=n_("Show &help bar"),
|
|
variable=self.tkopt.helpbar,
|
|
command=self.mOptHelpbar)
|
|
'''
|
|
|
|
# -------------------------------------------
|
|
# general options
|
|
|
|
# self.addCheckNode(tv, None,
|
|
# 'Save games geometry',
|
|
# self.menubar.tkopt.save_games_geometry,
|
|
# self.menubar.mOptSaveGamesGeometry)
|
|
|
|
# self.addCheckNode(tv, None,
|
|
# 'Demo logo',
|
|
# self.menubar.tkopt.demo_logo,
|
|
# self.menubar.mOptDemoLogo)
|
|
|
|
self.addCheckNode(tv, None,
|
|
_('Startup splash screen'),
|
|
self.menubar.tkopt.splashscreen,
|
|
self.menubar.mOptSplashscreen)
|
|
|
|
self.addCheckNode(tv, None,
|
|
_('Winning splash'),
|
|
self.menubar.tkopt.display_win_message,
|
|
self.menubar.mWinDialog)
|
|
|
|
|
|
# ************************************************************************
|
|
|
|
|
|
class HelpMenuDialog(LMenuDialog):
|
|
def __init__(self, menubar, parent, title, app, **kw):
|
|
kw['size_hint'] = (0.3, 1)
|
|
kw['persist'] = True
|
|
super(HelpMenuDialog, self).__init__(menubar, parent, title, app, **kw)
|
|
|
|
def make_help_command(self, command):
|
|
def help_command():
|
|
command()
|
|
self.closeWindow(0)
|
|
return help_command
|
|
|
|
def buildTree(self, tv, node):
|
|
tv.add_node(
|
|
LTreeNode(
|
|
text=_('Contents'),
|
|
command=self.make_help_command(self.menubar.mHelp)))
|
|
tv.add_node(
|
|
LTreeNode(
|
|
text=_('How to use PySol'),
|
|
command=self.make_help_command(self.menubar.mHelpHowToPlay)))
|
|
tv.add_node(
|
|
LTreeNode(
|
|
text=_('Rules for this game'),
|
|
command=self.make_help_command(self.menubar.mHelpRules)))
|
|
tv.add_node(
|
|
LTreeNode(
|
|
text=_('License terms'),
|
|
command=self.make_help_command(self.menubar.mHelpLicense)))
|
|
tv.add_node(
|
|
LTreeNode(
|
|
text=_('About %s...') % TITLE,
|
|
command=self.make_help_command(self.menubar.mHelpAbout)))
|
|
|
|
# tv.add_node(LTreeNode(
|
|
# text='AboutKivy ...',
|
|
# command=self.makeHtmlCommand(self.menubar, "kivy.html")))
|
|
|
|
def makeHtmlCommand(self, bar, htmlfile):
|
|
def htmlCommand():
|
|
bar.mHelpHtml(htmlfile)
|
|
|
|
return htmlCommand
|
|
|
|
|
|
# ************************************************************************
|
|
# *
|
|
# ************************************************************************
|
|
|
|
|
|
class EmulTkMenu(object):
|
|
|
|
def __init__(self, master, **kw):
|
|
|
|
self.name = kw["name"]
|
|
self.n = 0
|
|
self._w = None
|
|
if (self.name):
|
|
if master._w == '.':
|
|
self._w = '.' + self.name
|
|
else:
|
|
self._w = master._w + '.' + self.name
|
|
else:
|
|
self.name = "<>"
|
|
|
|
def labeltoname(self, label):
|
|
name = re.sub(r"[^0-9a-zA-Z]", "", label).lower()
|
|
label = _(label)
|
|
underline = label.find('&')
|
|
if underline >= 0:
|
|
label = label.replace('&', '')
|
|
return name, label, underline
|
|
|
|
def add_cascade(self, cnf={}, **kw):
|
|
self.add('cascade', cnf or kw)
|
|
pass
|
|
|
|
def add(self, itemType, cnf={}):
|
|
label = cnf.get("label")
|
|
if label:
|
|
name = cnf.get('name')
|
|
if name:
|
|
name, label, underline = self.labeltoname(label)
|
|
cnf["underline"] = cnf.get("underline", underline)
|
|
cnf["label"] = label
|
|
if name and self.addPath:
|
|
path = str(self._w) + "." + name
|
|
self.addPath(path, self, self.n, cnf.get("menu"))
|
|
|
|
def cget(self, key):
|
|
return key
|
|
|
|
# ************************************************************************
|
|
|
|
|
|
class MfxMenubar(EmulTkMenu):
|
|
addPath = None
|
|
|
|
def __init__(self, master, **kw):
|
|
super(MfxMenubar, self).__init__(master, **kw)
|
|
topmenu = self.name == 'menubar'
|
|
|
|
self.menu = LMenu(not topmenu, text=self.name)
|
|
if topmenu:
|
|
master.setMenu(self.menu)
|
|
|
|
# ************************************************************************
|
|
# * - create menubar
|
|
# * - update menubar
|
|
# * - menu actions
|
|
# ************************************************************************
|
|
|
|
|
|
class PysolMenubarTk:
|
|
def __init__(self, app, top, progress=None):
|
|
self._createTkOpt()
|
|
self._setOptions()
|
|
# init columnbreak
|
|
# self.__cb_max = int(self.top.winfo_screenheight()/23)
|
|
self.__cb_max = 8
|
|
# sh = self.top.winfo_screenheight()
|
|
# self.__cb_max = 22
|
|
# if sh >= 600: self.__cb_max = 27
|
|
# if sh >= 768: self.__cb_max = 32
|
|
# if sh >= 1024: self.__cb_max = 40
|
|
self.progress = progress
|
|
# create menus
|
|
self.__menubar = None
|
|
self.__menupath = {}
|
|
self.__keybindings = {}
|
|
self._createMenubar()
|
|
self.top = top
|
|
|
|
if self.progress:
|
|
self.progress.update(step=1)
|
|
|
|
# set the menubar
|
|
# self.updateBackgroundImagesMenu()
|
|
# self.top.config(menu=self.__menubar)
|
|
|
|
def _createTkOpt(self):
|
|
# structure to convert menu-options to Toolkit variables
|
|
self.tkopt = Struct(
|
|
gameid=IntVar(),
|
|
gameid_popular=IntVar(),
|
|
comment=BooleanVar(),
|
|
autofaceup=BooleanVar(),
|
|
autodrop=BooleanVar(),
|
|
autodeal=BooleanVar(),
|
|
quickplay=BooleanVar(),
|
|
undo=BooleanVar(),
|
|
bookmarks=BooleanVar(),
|
|
hint=BooleanVar(),
|
|
shuffle=BooleanVar(),
|
|
highlight_piles=BooleanVar(),
|
|
highlight_cards=BooleanVar(),
|
|
highlight_samerank=BooleanVar(),
|
|
highlight_not_matching=BooleanVar(),
|
|
mahjongg_show_removed=BooleanVar(),
|
|
shisen_show_hint=BooleanVar(),
|
|
accordion_deal_all=BooleanVar(),
|
|
sound=BooleanVar(),
|
|
sound_sample_volume=IntVar(),
|
|
sound_music_volume=IntVar(),
|
|
cardback=IntVar(),
|
|
tabletile=IntVar(),
|
|
animations=IntVar(),
|
|
redeal_animation=BooleanVar(),
|
|
win_animation=BooleanVar(),
|
|
shadow=BooleanVar(),
|
|
shade=BooleanVar(),
|
|
shade_filled_stacks=BooleanVar(),
|
|
shrink_face_down=BooleanVar(),
|
|
toolbar=IntVar(),
|
|
toolbar_style=StringVar(),
|
|
toolbar_relief=StringVar(),
|
|
toolbar_compound=StringVar(),
|
|
toolbar_size=IntVar(),
|
|
statusbar=BooleanVar(),
|
|
num_cards=BooleanVar(),
|
|
helpbar=BooleanVar(),
|
|
save_games_geometry=BooleanVar(),
|
|
splashscreen=BooleanVar(),
|
|
demo_logo=BooleanVar(),
|
|
mouse_type=StringVar(),
|
|
mouse_undo=BooleanVar(),
|
|
negative_bottom=BooleanVar(),
|
|
display_win_message=BooleanVar(),
|
|
pause=BooleanVar(),
|
|
cardset=IntVar(),
|
|
cardbacks={},
|
|
toolbar_vars={},
|
|
sound_sample_vars={},
|
|
color_vars={},
|
|
language=StringVar(),
|
|
)
|
|
for w in TOOLBAR_BUTTONS:
|
|
self.tkopt.toolbar_vars[w] = BooleanVar()
|
|
for k in self.app.opt.sound_samples:
|
|
self.tkopt.sound_sample_vars[k] = BooleanVar()
|
|
for k in self.app.opt.colors:
|
|
self.tkopt.color_vars[k] = StringVar()
|
|
|
|
def _setOptions(self):
|
|
tkopt, opt = self.tkopt, self.app.opt
|
|
# set state of the menu items
|
|
tkopt.autofaceup.set(opt.autofaceup)
|
|
tkopt.autodrop.set(opt.autodrop)
|
|
tkopt.autodeal.set(opt.autodeal)
|
|
tkopt.quickplay.set(opt.quickplay)
|
|
tkopt.undo.set(opt.undo)
|
|
tkopt.hint.set(opt.hint)
|
|
tkopt.shuffle.set(opt.shuffle)
|
|
tkopt.bookmarks.set(opt.bookmarks)
|
|
tkopt.highlight_piles.set(opt.highlight_piles)
|
|
tkopt.highlight_cards.set(opt.highlight_cards)
|
|
tkopt.highlight_samerank.set(opt.highlight_samerank)
|
|
tkopt.highlight_not_matching.set(opt.highlight_not_matching)
|
|
tkopt.shrink_face_down.set(opt.shrink_face_down)
|
|
tkopt.shade_filled_stacks.set(opt.shade_filled_stacks)
|
|
tkopt.mahjongg_show_removed.set(opt.mahjongg_show_removed)
|
|
tkopt.shisen_show_hint.set(opt.shisen_show_hint)
|
|
tkopt.accordion_deal_all.set(opt.accordion_deal_all)
|
|
tkopt.sound.set(opt.sound)
|
|
tkopt.sound_sample_volume.set(opt.sound_sample_volume)
|
|
tkopt.sound_music_volume.set(opt.sound_music_volume)
|
|
tkopt.cardback.set(self.app.cardset.backindex)
|
|
tkopt.tabletile.set(self.app.tabletile_index)
|
|
tkopt.animations.set(opt.animations)
|
|
tkopt.redeal_animation.set(opt.redeal_animation)
|
|
tkopt.win_animation.set(opt.win_animation)
|
|
tkopt.shadow.set(opt.shadow)
|
|
tkopt.shade.set(opt.shade)
|
|
tkopt.toolbar.set(opt.toolbar)
|
|
tkopt.toolbar_style.set(opt.toolbar_style)
|
|
tkopt.toolbar_relief.set(opt.toolbar_relief)
|
|
tkopt.toolbar_compound.set(opt.toolbar_compound)
|
|
tkopt.toolbar_size.set(opt.toolbar_size)
|
|
tkopt.toolbar_relief.set(opt.toolbar_relief)
|
|
tkopt.statusbar.set(opt.statusbar)
|
|
tkopt.num_cards.set(opt.num_cards)
|
|
tkopt.helpbar.set(opt.helpbar)
|
|
tkopt.save_games_geometry.set(opt.save_games_geometry)
|
|
tkopt.demo_logo.set(opt.demo_logo)
|
|
tkopt.splashscreen.set(opt.splashscreen)
|
|
tkopt.mouse_type.set(opt.mouse_type)
|
|
tkopt.mouse_undo.set(opt.mouse_undo)
|
|
tkopt.negative_bottom.set(opt.negative_bottom)
|
|
tkopt.display_win_message.set(opt.display_win_message)
|
|
tkopt.cardset.set(self.app.cardset_manager.getSelected())
|
|
tkopt.language.set(opt.language)
|
|
|
|
for w in TOOLBAR_BUTTONS:
|
|
tkopt.toolbar_vars[w].set(opt.toolbar_vars.get(w, False))
|
|
for k in self.app.opt.sound_samples:
|
|
self.tkopt.sound_sample_vars[k].set(
|
|
opt.sound_samples.get(k, False))
|
|
for k in self.app.opt.colors:
|
|
self.tkopt.color_vars[k].set(opt.colors.get(k, '#000000'))
|
|
|
|
def connectGame(self, game):
|
|
self.game = game
|
|
if game is None:
|
|
return
|
|
assert self.app is game.app
|
|
tkopt = self.tkopt
|
|
# opt = self.app.opt
|
|
tkopt.gameid.set(game.id)
|
|
tkopt.gameid_popular.set(game.id)
|
|
tkopt.comment.set(bool(game.gsaveinfo.comment))
|
|
tkopt.pause.set(self.game.pause)
|
|
if game.canFindCard():
|
|
connect_game_find_card_dialog(game)
|
|
else:
|
|
destroy_find_card_dialog()
|
|
connect_game_solver_dialog(game)
|
|
|
|
# create a GTK-like path
|
|
def _addPath(self, path, menu, index, submenu):
|
|
# print ('MfxMenubar: _addPath %s, %s' % (path, menu))
|
|
# y = self.yy
|
|
if path not in self.__menupath:
|
|
# print path, menu, index, submenu
|
|
self.__menupath[path] = (menu, index, submenu)
|
|
|
|
def _getEnabledState(self, enabled):
|
|
print('_getEnabledState: %s' % enabled)
|
|
if enabled:
|
|
return "normal"
|
|
return "disabled"
|
|
|
|
def updateProgress(self):
|
|
if self.progress:
|
|
self.progress.update(step=1)
|
|
|
|
#
|
|
# create the menubar
|
|
#
|
|
|
|
def _createMenubar(self):
|
|
MfxMenubar.addPath = self._addPath
|
|
kw = {"name": "menubar"}
|
|
self.__menubar = MfxMenubar(self.top, **kw)
|
|
|
|
# init keybindings
|
|
bind(self.top, "<KeyPress>", self._keyPressHandler)
|
|
|
|
# LMainMenuDialog()
|
|
LMenuItem(self.__menubar.menu,
|
|
text=_("Menu"), command=self.mMainMenuDialog)
|
|
|
|
MfxMenubar.addPath = None
|
|
|
|
#
|
|
# key binding utility
|
|
#
|
|
|
|
def _bindKey(self, modifier, key, func):
|
|
# if 0 and not modifier and len(key) == 1:
|
|
# self.__keybindings[key.lower()] = func
|
|
# self.__keybindings[key.upper()] = func
|
|
# return
|
|
if not modifier and len(key) == 1:
|
|
# ignore Ctrl/Shift/Alt
|
|
# but don't ignore NumLock (state == 16)
|
|
def lfunc(e, func=func):
|
|
return e.state in (0, 16) and func(e)
|
|
func = lfunc
|
|
# func = lambda e, func=func: e.state in (0, 16) and func(e)
|
|
sequence = "<" + modifier + "KeyPress-" + key + ">"
|
|
bind(self.top, sequence, func)
|
|
if len(key) == 1 and key != key.upper():
|
|
key = key.upper()
|
|
sequence = "<" + modifier + "KeyPress-" + key + ">"
|
|
bind(self.top, sequence, func)
|
|
|
|
def _keyPressHandler(self, event):
|
|
r = EVENT_PROPAGATE
|
|
if event and self.game:
|
|
# print event.__dict__
|
|
if self.game.demo:
|
|
# stop the demo by setting self.game.demo.keypress
|
|
if event.char: # ignore Ctrl/Shift/etc.
|
|
self.game.demo.keypress = event.char
|
|
r = EVENT_HANDLED
|
|
# func = self.__keybindings.get(event.char)
|
|
# if func and (event.state & ~2) == 0:
|
|
# func(event)
|
|
# r = EVENT_HANDLED
|
|
return r
|
|
|
|
#
|
|
# Select Game menu creation
|
|
#
|
|
'''
|
|
def _addSelectGameMenu(self, menu):
|
|
games = map(self.app.gdb.get, self.app.gdb.getGamesIdSortedByName())
|
|
m = "Ctrl-"
|
|
if sys.platform == "darwin":
|
|
m = "Cmd-"
|
|
menu.add_command(label=n_("All &games..."), accelerator=m + "W",
|
|
command=self.mSelectGameDialog)
|
|
|
|
def _addSelectGameSubMenu(self, games, menu, select_data,
|
|
command, variable):
|
|
# print select_data
|
|
need_sep = 0
|
|
for label, select_func in select_data:
|
|
if label is None:
|
|
need_sep = 1
|
|
continue
|
|
g = filter(select_func, games)
|
|
if not g:
|
|
continue
|
|
if need_sep:
|
|
menu.add_separator()
|
|
need_sep = 0
|
|
submenu = MfxMenu(menu, label=label)
|
|
self._addSelectGameSubSubMenu(g, submenu, command, variable)
|
|
|
|
def _getNumGames(self, games, select_data):
|
|
ngames = 0
|
|
for label, select_func in select_data:
|
|
ngames += len(filter(select_func, games))
|
|
return ngames
|
|
|
|
def _addSelectMahjonggGameSubMenu(self, games, menu, command, variable):
|
|
def select_func(gi): return gi.si.game_type == GI.GT_MAHJONGG
|
|
mahjongg_games = filter(select_func, games)
|
|
if len(mahjongg_games) == 0:
|
|
return
|
|
#
|
|
menu = MfxMenu(menu, label=n_("&Mahjongg games"))
|
|
|
|
def add_menu(games, c0, c1, menu=menu,
|
|
variable=variable, command=command):
|
|
if not games:
|
|
return
|
|
label = c0 + ' - ' + c1
|
|
if c0 == c1:
|
|
label = c0
|
|
submenu = MfxMenu(menu, label=label, name=None)
|
|
self._addSelectGameSubSubMenu(games, submenu, command,
|
|
variable, short_name=True)
|
|
|
|
games = {}
|
|
for gi in mahjongg_games:
|
|
c = gi.short_name.strip()[0]
|
|
if c in games:
|
|
games[c].append(gi)
|
|
else:
|
|
games[c] = [gi]
|
|
games = games.items()
|
|
games.sort()
|
|
g0 = []
|
|
c0 = c1 = games[0][0]
|
|
for c, g1 in games:
|
|
if len(g0) + len(g1) >= self.__cb_max:
|
|
add_menu(g0, c0, c1)
|
|
g0 = g1
|
|
c0 = c1 = c
|
|
else:
|
|
g0 += g1
|
|
c1 = c
|
|
add_menu(g0, c0, c1)
|
|
|
|
def _addSelectPopularGameSubMenu(self, games, menu, command, variable):
|
|
def select_func(gi): return gi.si.game_flags & GI.GT_POPULAR
|
|
if len(filter(select_func, games)) == 0:
|
|
return
|
|
data = (n_("&Popular games"), select_func)
|
|
self._addSelectGameSubMenu(games, menu, (data, ),
|
|
self.mSelectGamePopular,
|
|
self.tkopt.gameid_popular)
|
|
|
|
def _addSelectFrenchGameSubMenu(self, games, menu, command, variable):
|
|
if self._getNumGames(games, GI.SELECT_GAME_BY_TYPE) == 0:
|
|
return
|
|
submenu = MfxMenu(menu, label=n_("&French games"))
|
|
self._addSelectGameSubMenu(games, submenu, GI.SELECT_GAME_BY_TYPE,
|
|
self.mSelectGame, self.tkopt.gameid)
|
|
|
|
def _addSelectOrientalGameSubMenu(self, games, menu, command, variable):
|
|
if self._getNumGames(games, GI.SELECT_ORIENTAL_GAME_BY_TYPE) == 0:
|
|
return
|
|
submenu = MfxMenu(menu, label=n_("&Oriental games"))
|
|
self._addSelectGameSubMenu(games, submenu,
|
|
GI.SELECT_ORIENTAL_GAME_BY_TYPE,
|
|
self.mSelectGame, self.tkopt.gameid)
|
|
|
|
def _addSelectSpecialGameSubMenu(self, games, menu, command, variable):
|
|
if self._getNumGames(games, GI.SELECT_ORIENTAL_GAME_BY_TYPE) == 0:
|
|
return
|
|
submenu = MfxMenu(menu, label=n_("&Special games"))
|
|
self._addSelectGameSubMenu(games, submenu,
|
|
GI.SELECT_SPECIAL_GAME_BY_TYPE,
|
|
self.mSelectGame, self.tkopt.gameid)
|
|
|
|
def _addSelectCustomGameSubMenu(self, games, menu, command, variable):
|
|
submenu = MfxMenu(menu, label=n_("&Custom games"))
|
|
|
|
def select_func(gi): return gi.si.game_type == GI.GT_CUSTOM
|
|
games = filter(select_func, games)
|
|
self.updateGamesMenu(submenu, games)
|
|
'''
|
|
|
|
def _addSelectAllGameSubMenu(self, games, menu, command, variable):
|
|
# LB
|
|
# herausgenommen: zu aufwendig !
|
|
return
|
|
'''
|
|
menu = MfxMenu(menu, label=n_("&All games by name"))
|
|
n, d = 0, self.__cb_max
|
|
i = 0
|
|
while True:
|
|
if self.progress:
|
|
self.progress.update(step=1)
|
|
columnbreak = i > 0 and (i % d) == 0
|
|
i += 1
|
|
if not games[n:n + d]:
|
|
break
|
|
m = min(n + d - 1, len(games) - 1)
|
|
label = games[n].name[:3] + ' - ' + games[m].name[:3]
|
|
|
|
submenu = MfxMenu(menu, label=label, name=None)
|
|
self._addSelectGameSubSubMenu(games[n:n + d], submenu,
|
|
command, variable)
|
|
n += d
|
|
# if columnbreak:
|
|
# menu.entryconfigure(i, columnbreak=columnbreak)
|
|
'''
|
|
|
|
# Eine 'closure' in Python? - voila!
|
|
def make_gamesetter(self, n, variable, command):
|
|
def gamesetter(x):
|
|
variable.set(n)
|
|
command()
|
|
return gamesetter
|
|
|
|
def _addSelectGameSubSubMenu(self, games, menu, command, variable,
|
|
short_name=False):
|
|
|
|
# cb = self.__cb_max
|
|
for i in range(len(games)):
|
|
gi = games[i]
|
|
# columnbreak = i > 0 and (i % cb) == 0
|
|
if short_name:
|
|
label = gi.short_name
|
|
else:
|
|
label = gi.name
|
|
|
|
# optimized by inlining
|
|
|
|
# geht nicht mehr 'optimiert' mit kivy
|
|
# die Funktionalität des tk.calls kann mit hilfe
|
|
# einer 'closure' rekonstruiert werden (s.o).
|
|
# LB
|
|
|
|
gsetter = self.make_gamesetter(gi.id, variable, command)
|
|
menu.add_command(label=label, command=gsetter)
|
|
|
|
# menu.tk.call((menu._w, 'add', 'radiobutton') +
|
|
# menu._options({'command': command,
|
|
# 'variable': variable,
|
|
# 'columnbreak': columnbreak,
|
|
# 'value': gi.id,
|
|
# 'label': label}))
|
|
|
|
def updateGamesMenu(self, menu, games):
|
|
|
|
def cmp2(a, b):
|
|
"""python 3 replacement for python 2 cmp function"""
|
|
return (a > b) - (a < b)
|
|
|
|
menu.delete(0, 'last')
|
|
|
|
if len(games) == 0:
|
|
menu.add_radiobutton(label=_('<none>'), name=None,
|
|
state='disabled')
|
|
elif len(games) > self.__cb_max * 4:
|
|
games.sort(lambda a, b: cmp2(a.name, b.name))
|
|
self._addSelectAllGameSubMenu(games, menu,
|
|
command=self.mSelectGame,
|
|
variable=self.tkopt.gameid)
|
|
else:
|
|
self._addSelectGameSubSubMenu(games, menu,
|
|
command=self.mSelectGame,
|
|
variable=self.tkopt.gameid)
|
|
|
|
def mMainMenuDialog(self, *event):
|
|
MainMenuDialog(self, self.top, title=_("Main Menu"), app=self.app)
|
|
return EVENT_HANDLED
|
|
|
|
def mFileMenuDialog(self, *event):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.game.setCursor(cursor=CURSOR_WATCH)
|
|
after_idle(self.top, self.__restoreCursor)
|
|
FileMenuDialog(self, self.top, title=_("File Menu"), app=self.app)
|
|
return EVENT_HANDLED
|
|
|
|
def mEditMenuDialog(self, *event):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.game.setCursor(cursor=CURSOR_WATCH)
|
|
after_idle(self.top, self.__restoreCursor)
|
|
EditMenuDialog(self, self.top, title=_("Tools"), app=self.app)
|
|
return EVENT_HANDLED
|
|
|
|
def mGameMenuDialog(self, *event):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.game.setCursor(cursor=CURSOR_WATCH)
|
|
after_idle(self.top, self.__restoreCursor)
|
|
GameMenuDialog(self, self.top, title=_("Statistics"), app=self.app)
|
|
return EVENT_HANDLED
|
|
|
|
def mAssistMenuDialog(self, *event):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.game.setCursor(cursor=CURSOR_WATCH)
|
|
after_idle(self.top, self.__restoreCursor)
|
|
AssistMenuDialog(self, self.top, title=_("Assists"), app=self.app)
|
|
return EVENT_HANDLED
|
|
|
|
def mOptionsMenuDialog(self, *event):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.game.setCursor(cursor=CURSOR_WATCH)
|
|
after_idle(self.top, self.__restoreCursor)
|
|
OptionsMenuDialog(self, self.top, title=_("Options"), app=self.app)
|
|
return EVENT_HANDLED
|
|
|
|
def mHelpMenuDialog(self, *event):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.game.setCursor(cursor=CURSOR_WATCH)
|
|
after_idle(self.top, self.__restoreCursor)
|
|
HelpMenuDialog(self, self.top, title=_("Help"), app=self.app)
|
|
return EVENT_HANDLED
|
|
#
|
|
# Select Game menu actions
|
|
#
|
|
|
|
def mSelectGame(self, *args):
|
|
print('mSelectGame %s' % self)
|
|
self._mSelectGame(self.tkopt.gameid.get())
|
|
|
|
def mSelectGamePopular(self, *args):
|
|
self._mSelectGame(self.tkopt.gameid_popular.get())
|
|
|
|
def _mSelectGameDialog(self, d):
|
|
if d.gameid != self.game.id:
|
|
self.tkopt.gameid.set(d.gameid)
|
|
self.tkopt.gameid_popular.set(d.gameid)
|
|
self._cancelDrag()
|
|
self.game.endGame()
|
|
self.game.quitGame(d.gameid, random=d.random)
|
|
return EVENT_HANDLED
|
|
|
|
def __restoreCursor(self, *event):
|
|
self.game.setCursor(cursor=self.app.top_cursor)
|
|
|
|
def mSelectGameDialog(self, *event):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.game.setCursor(cursor=CURSOR_WATCH)
|
|
after_idle(self.top, self.__restoreCursor)
|
|
d = SelectGameDialog(self.top, title=_("Select game"),
|
|
app=self.app, gameid=self.game.id)
|
|
return self._mSelectGameDialog(d)
|
|
|
|
#
|
|
# menubar overrides
|
|
#
|
|
|
|
def updateFavoriteGamesMenu(self):
|
|
return
|
|
|
|
# TBD ?
|
|
'''
|
|
gameids = self.app.opt.favorite_gameid
|
|
|
|
print('favorite_gameids = %s' % gameids)
|
|
|
|
submenu = self.__menupath[".menubar.file.favoritegames"][2]
|
|
games = []
|
|
for id in gameids:
|
|
gi = self.app.getGameInfo(id)
|
|
if gi:
|
|
games.append(gi)
|
|
self.updateGamesMenu(submenu, games)
|
|
|
|
# das folgende ist nur das enable/disable des add/remove buttons.
|
|
# geht mit kivy nicht so.
|
|
|
|
# state = self._getEnabledState
|
|
# in_favor = self.app.game.id in gameids
|
|
|
|
# menu, index, submenu = self.__menupath[".menubar.file.addtofavorites"]
|
|
# menu.entryconfig(index, state=state(not in_favor))
|
|
|
|
# menu, index, submenu = self.__menupath[".menubar.file.removefromfavorites"]
|
|
# menu.entryconfig(index, state=state(in_favor))
|
|
'''
|
|
|
|
def updateRecentGamesMenu(self, gameids):
|
|
return
|
|
|
|
# TBD ?
|
|
'''
|
|
submenu = self.__menupath[".menubar.file.recentgames"][2]
|
|
games = []
|
|
for id in gameids:
|
|
gi = self.app.getGameInfo(id)
|
|
if gi:
|
|
games.append(gi)
|
|
self.updateGamesMenu(submenu, games)
|
|
'''
|
|
|
|
def updateBookmarkMenuState(self):
|
|
# LB:
|
|
print('updateBookmarkMenuState - fake')
|
|
return
|
|
|
|
state = self._getEnabledState
|
|
mp1 = self.__menupath.get(".menubar.edit.setbookmark")
|
|
mp2 = self.__menupath.get(".menubar.edit.gotobookmark")
|
|
mp3 = self.__menupath.get(".menubar.edit.clearbookmarks")
|
|
if mp1 is None or mp2 is None or mp3 is None:
|
|
return
|
|
x = self.app.opt.bookmarks and self.game.canSetBookmark()
|
|
#
|
|
menu, index, submenu = mp1
|
|
for i in range(9):
|
|
submenu.entryconfig(i, state=state(x))
|
|
menu.entryconfig(index, state=state(x))
|
|
#
|
|
menu, index, submenu = mp2
|
|
ms = 0
|
|
for i in range(9):
|
|
s = self.game.gsaveinfo.bookmarks.get(i) is not None
|
|
submenu.entryconfig(i, state=state(s and x))
|
|
ms = ms or s
|
|
menu.entryconfig(index, state=state(ms and x))
|
|
#
|
|
menu, index, submenu = mp3
|
|
menu.entryconfig(index, state=state(ms and x))
|
|
|
|
def updateBackgroundImagesMenu(self):
|
|
# LB:
|
|
print('updateBackgroundImagesMenu - fake')
|
|
return
|
|
|
|
mp = self.__menupath.get(".menubar.options.cardbackground")
|
|
# delete all entries
|
|
submenu = mp[2]
|
|
submenu.delete(0, "last")
|
|
# insert new cardbacks
|
|
mbacks = self.app.images.getCardbacks()
|
|
cb = int(math.ceil(math.sqrt(len(mbacks))))
|
|
for i in range(len(mbacks)):
|
|
columnbreak = i > 0 and (i % cb) == 0
|
|
submenu.add_radiobutton(
|
|
label=mbacks[i].name,
|
|
image=mbacks[i].menu_image,
|
|
variable=self.tkopt.cardback,
|
|
value=i,
|
|
command=self.mOptCardback,
|
|
columnbreak=columnbreak,
|
|
indicatoron=0,
|
|
hidemargin=0)
|
|
#
|
|
# menu updates
|
|
#
|
|
|
|
def setMenuState(self, state, path):
|
|
# LB: not used
|
|
return
|
|
|
|
def setToolbarState(self, state, path):
|
|
# LB: not used
|
|
return
|
|
|
|
def _setCommentMenu(self, v):
|
|
self.tkopt.comment.set(v)
|
|
|
|
def _setPauseMenu(self, v):
|
|
self.tkopt.pause.set(v)
|
|
|
|
#
|
|
# menu actions
|
|
#
|
|
|
|
DEFAULTEXTENSION = ".pso"
|
|
# TRANSLATORS: Usually, 'PySol files'
|
|
FILETYPES = ((_("%s files") % TITLE, "*" + DEFAULTEXTENSION),
|
|
(_("All files"), "*"))
|
|
|
|
def mAddFavor(self, *event):
|
|
gameid = self.app.game.id
|
|
if gameid not in self.app.opt.favorite_gameid:
|
|
self.app.opt.favorite_gameid.append(gameid)
|
|
self.updateFavoriteGamesMenu()
|
|
|
|
def mDelFavor(self, *event):
|
|
gameid = self.app.game.id
|
|
if gameid in self.app.opt.favorite_gameid:
|
|
self.app.opt.favorite_gameid.remove(gameid)
|
|
self.updateFavoriteGamesMenu()
|
|
|
|
def mOpen(self, *event):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
# filename = self.game.filename
|
|
filename = "lastgame.pso"
|
|
if filename:
|
|
idir, ifile = os.path.split(os.path.normpath(filename))
|
|
else:
|
|
idir, ifile = "", ""
|
|
if not idir:
|
|
idir = self.app.dn.savegames
|
|
# d = tkFileDialog.Open()
|
|
# filename = d.show(filetypes=self.FILETYPES,
|
|
# defaultextension=self.DEFAULTEXTENSION,
|
|
# initialdir=idir, initialfile=ifile)
|
|
filename = idir + "/" + ifile
|
|
|
|
print('filename = %s' % filename)
|
|
if filename:
|
|
filename = os.path.normpath(filename)
|
|
# filename = os.path.normcase(filename)
|
|
if os.path.isfile(filename):
|
|
self.game.loadGame(filename)
|
|
|
|
def mSaveAs(self, *event):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
if not self.menustate.save_as:
|
|
return
|
|
# filename = self.game.filename
|
|
filename = "lastgame.pso"
|
|
if not filename:
|
|
filename = self.app.getGameSaveName(self.game.id)
|
|
if os.name == "posix":
|
|
filename = filename + "-" + self.game.getGameNumber(format=0)
|
|
elif os.path.supports_unicode_filenames: # new in python 2.3
|
|
filename = filename + "-" + self.game.getGameNumber(format=0)
|
|
else:
|
|
filename = filename + "-01"
|
|
filename = filename + self.DEFAULTEXTENSION
|
|
idir, ifile = os.path.split(os.path.normpath(filename))
|
|
if not idir:
|
|
idir = self.app.dn.savegames
|
|
# print self.game.filename, ifile
|
|
# d = tkFileDialog.SaveAs()
|
|
# filename = d.show(filetypes=self.FILETYPES,
|
|
# defaultextension=self.DEFAULTEXTENSION,
|
|
# initialdir=idir, initialfile=ifile)
|
|
filename = idir + "/" + ifile
|
|
if filename:
|
|
filename = os.path.normpath(filename)
|
|
# filename = os.path.normcase(filename)
|
|
self.game.saveGame(filename)
|
|
self.updateMenus()
|
|
|
|
def mPause(self, *args):
|
|
if not self.game:
|
|
return
|
|
if not self.game.pause:
|
|
if self._cancelDrag():
|
|
return
|
|
self.game.doPause()
|
|
self.tkopt.pause.set(self.game.pause)
|
|
|
|
def mOptLanguage(self, *args):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.language = self.tkopt.language.get()
|
|
MfxMessageDialog(
|
|
self.app.top, title=_("Note"),
|
|
text=_("""\
|
|
These settings will take effect
|
|
the next time you restart the %(app)s""") % {'app': TITLE})
|
|
|
|
def mOptSoundDialog(self, *args):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.sound = self.tkopt.sound.get()
|
|
|
|
def mOptSoundSampleVol(self, *args):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.sound_sample_volume = self.tkopt.sound_sample_volume.get()
|
|
|
|
def mOptSoundMusicVol(self, *args):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.sound_music_volume = self.tkopt.sound_music_volume.get()
|
|
|
|
def mOptSoundSample(self, key, *args):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.sound_samples[key] = \
|
|
self.tkopt.sound_sample_vars[key].get()
|
|
|
|
def mOptTableColor(self, *args):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
nv = self.tkopt.color_vars['table'].get()
|
|
ov = self.app.opt.colors['table']
|
|
self.app.opt.colors['table'] = nv
|
|
if ov != nv:
|
|
self.app.top_bg = nv
|
|
self.app.tabletile_index = 0
|
|
self.app.setTile(0, force=True)
|
|
self.tkopt.tabletile.set(0)
|
|
|
|
def mOptTileSet(self, *args):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
idx = self.tkopt.tabletile.get()
|
|
if idx > 0 and idx != self.app.tabletile_index:
|
|
self.app.setTile(idx)
|
|
self.tkopt.color_vars['table'].set('#008285')
|
|
|
|
def mOptAutoFaceUp(self, *args):
|
|
if self._cancelDrag():
|
|
return
|
|
self.app.opt.autofaceup = self.tkopt.autofaceup.get()
|
|
if self.app.opt.autofaceup:
|
|
self.game.autoPlay()
|
|
|
|
def mOptAutoDrop(self, *args):
|
|
if self._cancelDrag():
|
|
return
|
|
self.app.opt.autodrop = self.tkopt.autodrop.get()
|
|
if self.app.opt.autodrop:
|
|
self.game.autoPlay()
|
|
|
|
def mOptAutoDeal(self, *args):
|
|
if self._cancelDrag():
|
|
return
|
|
self.app.opt.autodeal = self.tkopt.autodeal.get()
|
|
if self.app.opt.autodeal:
|
|
self.game.autoPlay()
|
|
|
|
def mOptQuickPlay(self, *args):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.quickplay = self.tkopt.quickplay.get()
|
|
|
|
def mOptEnableUndo(self, *args):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.undo = self.tkopt.undo.get()
|
|
self.game.updateMenus()
|
|
|
|
def mOptEnableBookmarks(self, *args):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.bookmarks = self.tkopt.bookmarks.get()
|
|
self.game.updateMenus()
|
|
|
|
def mOptEnableHint(self, *args):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.hint = self.tkopt.hint.get()
|
|
self.game.updateMenus()
|
|
|
|
def mOptEnableShuffle(self, *args):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.shuffle = self.tkopt.shuffle.get()
|
|
self.game.updateMenus()
|
|
|
|
def mOptEnableHighlightPiles(self, *args):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.highlight_piles = self.tkopt.highlight_piles.get()
|
|
self.game.updateMenus()
|
|
|
|
def mOptEnableHighlightCards(self, *args):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.highlight_cards = self.tkopt.highlight_cards.get()
|
|
self.game.updateMenus()
|
|
|
|
def mOptEnableHighlightSameRank(self, *args):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.highlight_samerank = self.tkopt.highlight_samerank.get()
|
|
# self.game.updateMenus()
|
|
|
|
def mOptEnableHighlightNotMatching(self, *args):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.highlight_not_matching = \
|
|
self.tkopt.highlight_not_matching.get()
|
|
# self.game.updateMenus()
|
|
|
|
def mOptAnimations(self, *args):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.animations = self.tkopt.animations.get()
|
|
|
|
def mRedealAnimation(self, *args):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.redeal_animation = self.tkopt.redeal_animation.get()
|
|
|
|
def mWinAnimation(self, *args):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.win_animation = self.tkopt.win_animation.get()
|
|
|
|
def mWinDialog(self, *args):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.display_win_message = self.tkopt.display_win_message.get()
|
|
|
|
def mOptShadow(self, *args):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.shadow = self.tkopt.shadow.get()
|
|
|
|
def mOptShade(self, *args):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.shade = self.tkopt.shade.get()
|
|
|
|
def mOptShrinkFaceDown(self, *args):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.shrink_face_down = self.tkopt.shrink_face_down.get()
|
|
self.game.endGame(bookmark=1)
|
|
self.game.quitGame(bookmark=1)
|
|
|
|
def mOptShadeFilledStacks(self, *args):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.shade_filled_stacks = self.tkopt.shade_filled_stacks.get()
|
|
self.game.endGame(bookmark=1)
|
|
self.game.quitGame(bookmark=1)
|
|
|
|
def mOptMahjonggShowRemoved(self, *args):
|
|
if self._cancelDrag():
|
|
return
|
|
self.app.opt.mahjongg_show_removed = \
|
|
self.tkopt.mahjongg_show_removed.get()
|
|
# self.game.updateMenus()
|
|
self.game.endGame(bookmark=1)
|
|
self.game.quitGame(bookmark=1)
|
|
|
|
def mOptShisenShowHint(self, *args):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.shisen_show_hint = self.tkopt.shisen_show_hint.get()
|
|
# self.game.updateMenus()
|
|
|
|
def mOptAccordionDealAll(self, *args):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.accordion_deal_all = self.tkopt.accordion_deal_all.get()
|
|
# self.game.updateMenus()
|
|
|
|
def mOptCardset(self, *event):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
idx = self.tkopt.cardset.get()
|
|
cs = self.app.cardset_manager.get(idx)
|
|
if cs is None or idx == self.app.cardset.index:
|
|
return
|
|
if idx >= 0:
|
|
self.app.nextgame.cardset = cs
|
|
self._cancelDrag()
|
|
self.game.endGame(bookmark=1)
|
|
self.game.quitGame(bookmark=1)
|
|
|
|
def mSelectCardsetDialog(self, *event):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
# strings, default = ("&OK", "&Load", "&Cancel"), 0
|
|
strings, default = (None, _("&Load"), _("&Cancel"), ), 1
|
|
# if os.name == "posix":
|
|
strings, default = (None, _("&Load"), _(
|
|
"&Cancel"), _("&Info..."), ), 1
|
|
key = self.app.nextgame.cardset.index
|
|
d = SelectCardsetDialogWithPreview(
|
|
self.top, title=_("Select cardset"),
|
|
app=self.app, manager=self.app.cardset_manager, key=key,
|
|
strings=strings, default=default)
|
|
|
|
cs = self.app.cardset_manager.get(d.key)
|
|
if cs is None or d.key == self.app.cardset.index:
|
|
return
|
|
if d.status == 0 and d.button in (0, 1) and d.key >= 0:
|
|
self.app.nextgame.cardset = cs
|
|
if d.button == 1:
|
|
self._cancelDrag()
|
|
self.game.endGame(bookmark=1)
|
|
self.game.quitGame(bookmark=1)
|
|
|
|
def mOptSetCardback(self, key, *event):
|
|
val = self.tkopt.cardbacks[key].get()
|
|
cs = self.app.cardset_manager.get(key)
|
|
cs.updateCardback(backindex=val)
|
|
# ANM: wir können den Background nur für das aktuell
|
|
# selektierte Cardset wirklich ändern. Nur dieses wird
|
|
# wird in den Optionen gespeichert.
|
|
if (cs == self.app.cardset):
|
|
self.app.updateCardset(self.game.id)
|
|
self.app.cardset.backindex = val
|
|
image = self.app.images.getBack(update=True)
|
|
for card in self.game.cards:
|
|
card.updateCardBackground(image=image)
|
|
self.app.canvas.update_idletasks()
|
|
|
|
def _mOptCardback(self, index):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
cs = self.app.cardset
|
|
old_index = cs.backindex
|
|
cs.updateCardback(backindex=index)
|
|
if cs.backindex == old_index:
|
|
return
|
|
self.app.updateCardset(self.game.id)
|
|
image = self.app.images.getBack(update=True)
|
|
for card in self.game.cards:
|
|
card.updateCardBackground(image=image)
|
|
self.app.canvas.update_idletasks()
|
|
self.tkopt.cardback.set(cs.backindex)
|
|
|
|
def mOptCardback(self, *event):
|
|
self._mOptCardback(self.tkopt.cardback.get())
|
|
|
|
def mOptChangeCardback(self, *event):
|
|
self._mOptCardback(self.app.cardset.backindex + 1)
|
|
|
|
def mOptToolbar(self, *event):
|
|
# if self._cancelDrag(break_pause=False): return
|
|
self.setToolbarSide(self.tkopt.toolbar.get())
|
|
|
|
def mOptToolbarStyle(self, *event):
|
|
# if self._cancelDrag(break_pause=False): return
|
|
self.setToolbarStyle(self.tkopt.toolbar_style.get())
|
|
|
|
def mOptToolbarCompound(self, *event):
|
|
# if self._cancelDrag(break_pause=False): return
|
|
self.setToolbarCompound(self.tkopt.toolbar_compound.get())
|
|
|
|
def mOptToolbarSize(self, *event):
|
|
# if self._cancelDrag(break_pause=False): return
|
|
self.setToolbarSize(self.tkopt.toolbar_size.get())
|
|
|
|
def mOptToolbarRelief(self, *event):
|
|
# if self._cancelDrag(break_pause=False): return
|
|
self.setToolbarRelief(self.tkopt.toolbar_relief.get())
|
|
|
|
def mOptToolbarConfig(self, w):
|
|
self.toolbarConfig(w, self.tkopt.toolbar_vars[w].get())
|
|
|
|
def mOptStatusbar(self, *event):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
if not self.app.statusbar:
|
|
return
|
|
side = self.tkopt.statusbar.get()
|
|
self.app.opt.statusbar = side
|
|
resize = not self.app.opt.save_games_geometry
|
|
if self.app.statusbar.show(side, resize=resize):
|
|
self.top.update_idletasks()
|
|
|
|
def mOptNumCards(self, *event):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.num_cards = self.tkopt.num_cards.get()
|
|
|
|
def mOptHelpbar(self, *event):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
if not self.app.helpbar:
|
|
return
|
|
show = self.tkopt.helpbar.get()
|
|
self.app.opt.helpbar = show
|
|
resize = not self.app.opt.save_games_geometry
|
|
if self.app.helpbar.show(show, resize=resize):
|
|
self.top.update_idletasks()
|
|
|
|
def mOptSaveGamesGeometry(self, *event):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.save_games_geometry = self.tkopt.save_games_geometry.get()
|
|
|
|
def mOptDemoLogo(self, *event):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.demo_logo = self.tkopt.demo_logo.get()
|
|
|
|
def mOptSplashscreen(self, *event):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.splashscreen = self.tkopt.splashscreen.get()
|
|
|
|
def mOptMouseType(self, *event):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.mouse_type = self.tkopt.mouse_type.get()
|
|
|
|
def mOptMouseUndo(self, *event):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.mouse_undo = self.tkopt.mouse_undo.get()
|
|
|
|
def mOptNegativeBottom(self, *event):
|
|
if self._cancelDrag():
|
|
return
|
|
self.app.opt.negative_bottom = self.tkopt.negative_bottom.get()
|
|
self.app.updateCardset()
|
|
self.game.endGame(bookmark=1)
|
|
self.game.quitGame(bookmark=1)
|
|
|
|
#
|
|
# toolbar support
|
|
#
|
|
|
|
def setToolbarSide(self, side):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.toolbar = side
|
|
self.tkopt.toolbar.set(side) # update radiobutton
|
|
resize = not self.app.opt.save_games_geometry
|
|
if self.app.toolbar.show(side, resize=resize):
|
|
self.top.update_idletasks()
|
|
|
|
def setToolbarSize(self, size):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.toolbar_size = size
|
|
self.tkopt.toolbar_size.set(size) # update radiobutton
|
|
dir = self.app.getToolbarImagesDir()
|
|
if self.app.toolbar.updateImages(dir, size):
|
|
self.game.updateStatus(player=self.app.opt.player)
|
|
self.top.update_idletasks()
|
|
|
|
def setToolbarStyle(self, style):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.toolbar_style = style
|
|
# update radiobutton
|
|
self.tkopt.toolbar_style.set(style)
|
|
dir = self.app.getToolbarImagesDir()
|
|
size = self.app.opt.toolbar_size
|
|
if self.app.toolbar.updateImages(dir, size):
|
|
# self.game.updateStatus(player=self.app.opt.player)
|
|
self.top.update_idletasks()
|
|
|
|
def setToolbarCompound(self, compound):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.toolbar_compound = compound
|
|
self.tkopt.toolbar_compound.set(
|
|
compound) # update radiobutton
|
|
if self.app.toolbar.setCompound(compound):
|
|
self.game.updateStatus(player=self.app.opt.player)
|
|
self.top.update_idletasks()
|
|
|
|
def setToolbarRelief(self, relief):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.toolbar_relief = relief
|
|
self.tkopt.toolbar_relief.set(relief) # update radiobutton
|
|
self.app.toolbar.setRelief(relief)
|
|
self.top.update_idletasks()
|
|
|
|
def toolbarConfig(self, w, v):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.app.opt.toolbar_vars[w] = v
|
|
self.app.toolbar.config(w, v)
|
|
self.top.update_idletasks()
|
|
|
|
#
|
|
# stacks descriptions
|
|
#
|
|
|
|
def mStackDesk(self, *event):
|
|
if self.game.stackdesc_list:
|
|
self.game.deleteStackDesc()
|
|
else:
|
|
if self._cancelDrag(break_pause=True):
|
|
return
|
|
self.game.showStackDesc()
|
|
|
|
def wizardDialog(self, edit=False):
|
|
from pysollib.wizardutil import write_game, reset_wizard
|
|
from wizarddialog import WizardDialog
|
|
|
|
if edit:
|
|
reset_wizard(self.game)
|
|
else:
|
|
reset_wizard(None)
|
|
d = WizardDialog(self.top, _('Solitaire Wizard'), self.app)
|
|
if d.status == 0 and d.button == 0:
|
|
try:
|
|
if edit:
|
|
gameid = write_game(self.app, game=self.game)
|
|
else:
|
|
gameid = write_game(self.app)
|
|
except Exception:
|
|
return
|
|
if SELECT_GAME_MENU:
|
|
menu = self.__menupath[".menubar.select.customgames"][2]
|
|
|
|
def select_func(gi): return gi.si.game_type == GI.GT_CUSTOM
|
|
games = map(self.app.gdb.get,
|
|
self.app.gdb.getGamesIdSortedByName())
|
|
games = filter(select_func, games)
|
|
self.updateGamesMenu(menu, games)
|
|
|
|
self.tkopt.gameid.set(gameid)
|
|
self._mSelectGame(gameid, force=True)
|
|
|
|
def mWizard(self, *event):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.wizardDialog()
|
|
|
|
def mWizardEdit(self, *event):
|
|
if self._cancelDrag(break_pause=False):
|
|
return
|
|
self.wizardDialog(edit=True)
|
|
|
|
|
|
'''
|
|
'''
|