1
0
Fork 0
mirror of https://github.com/shlomif/PySolFC.git synced 2025-04-05 00:02:29 -04:00
PySolFC/pysollib/tile/toolbar.py
skomoroh 5175caf86a + separated demo-mode and solver; solver dialog
+ added statistics progression; statistics progression dialog
+ new animation speed: `medium'; renamed `timer based' to `fast'
+ added flip animation (animatedFlip)
+ added move to waste animation (animatedFlipAndMove)
+ added cancel-drag animation (moveCardsBackHandler)
* improved demo game (snapshots based check for loop)
- removed setting text color from file name
- removed auto generation shadows (too slow)
* optimized menu creation
* changed some keybindings
* updated russian translation
* many bugfixes


git-svn-id: file:///home/shlomif/Backup/svn-dumps/PySolFC/svnsync-repos/pysolfc/PySolFC/trunk@138 efabe8c0-fbe8-4139-b769-b5e6d273206e
2007-02-16 23:20:02 +00:00

470 lines
16 KiB
Python

## vim:ts=4:et:nowrap
##
##---------------------------------------------------------------------------##
##
## PySol -- a Python Solitaire game
##
## Copyright (C) 2003 Markus Franz Xaver Johannes Oberhumer
## Copyright (C) 2002 Markus Franz Xaver Johannes Oberhumer
## Copyright (C) 2001 Markus Franz Xaver Johannes Oberhumer
## Copyright (C) 2000 Markus Franz Xaver Johannes Oberhumer
## Copyright (C) 1999 Markus Franz Xaver Johannes Oberhumer
## Copyright (C) 1998 Markus Franz Xaver Johannes Oberhumer
## All Rights Reserved.
##
## 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 2 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; see the file COPYING.
## If not, write to the Free Software Foundation, Inc.,
## 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
##
## Markus F.X.J. Oberhumer
## <markus@oberhumer.com>
## http://www.oberhumer.com/pysol
##
##---------------------------------------------------------------------------##
__all__ = ['PysolToolbar']
# imports
import os
import Tile as Tkinter
# PySol imports
from pysollib.mfxutil import destruct
from pysollib.mfxutil import Image, ImageTk, ImageOps
from pysollib.util import IMAGE_EXTENSIONS
from pysollib.settings import PACKAGE
from pysollib.actions import PysolToolbarActions
from pysollib.winsystems import TkSettings
# Toolkit imports
from tkconst import EVENT_HANDLED, EVENT_PROPAGATE
from tkwidget import MfxTooltip
from menubar import createToolbarMenu, MfxMenu
gettext = _
n_ = lambda x: x
# /***********************************************************************
# //
# ************************************************************************/
class AbstractToolbarButton:
def __init__(self, parent, toolbar, toolbar_name, position):
self.toolbar = toolbar
self.toolbar_name = toolbar_name
self.position = position
self.visible = False
def show(self, orient, force=False):
if self.visible and not force:
return
self.visible = True
if orient == 'horizontal':
padx, pady = TkSettings.toolbar_button_padding
self.grid(row=0,
column=self.position,
padx=padx, pady=pady,
sticky='nsew')
else:
pady, padx = TkSettings.toolbar_button_padding
self.grid(row=self.position,
column=0,
padx=padx, pady=pady,
sticky='nsew')
def hide(self):
if not self.visible: return
self.visible = False
self.grid_forget()
class ToolbarCheckbutton(AbstractToolbarButton, Tkinter.Checkbutton):
def __init__(self, parent, toolbar, toolbar_name, position, **kwargs):
kwargs['style'] = 'Toolbutton'
Tkinter.Checkbutton.__init__(self, parent, **kwargs)
AbstractToolbarButton.__init__(self, parent, toolbar, toolbar_name, position)
class ToolbarButton(AbstractToolbarButton, Tkinter.Button):
def __init__(self, parent, toolbar, toolbar_name, position, **kwargs):
kwargs['style'] = 'Toolbutton'
Tkinter.Button.__init__(self, parent, **kwargs)
AbstractToolbarButton.__init__(self, parent, toolbar, toolbar_name, position)
class ToolbarSeparator(Tkinter.Separator):
def __init__(self, parent, toolbar, position, **kwargs):
kwargs['orient'] = 'vertical'
Tkinter.Separator.__init__(self, parent, **kwargs)
self.toolbar = toolbar
self.position = position
self.visible = False
def show(self, orient, force=False):
if self.visible and not force:
return
self.visible = True
if orient == 'horizontal':
padx, pady = 4, 6
self.config(orient='vertical')
self.grid(row=0,
column=self.position,
padx=padx, pady=pady,
sticky='nsew')
else:
padx, pady = 4, 6
self.config(orient='horizontal')
self.grid(row=self.position,
column=0,
padx=pady, pady=padx,
sticky='nsew')
def hide(self):
if not self.visible: return
self.visible = False
self.grid_forget()
class ToolbarLabel(Tkinter.Message):
def __init__(self, parent, toolbar, toolbar_name, position, **kwargs):
Tkinter.Message.__init__(self, parent, **kwargs)
self.toolbar = toolbar
self.toolbar_name = toolbar_name
self.position = position
self.visible = False
def show(self, orient, force=False):
if self.visible and not force:
return
self.visible = True
padx, pady = TkSettings.toolbar_label_padding
if orient == 'horizontal':
self.grid(row=0,
column=self.position,
padx=padx, pady=pady,
sticky='nsew')
else:
self.grid(row=self.position,
column=0,
padx=padx, pady=pady,
sticky='nsew')
def hide(self):
if not self.visible: return
self.visible = False
self.grid_forget()
# /***********************************************************************
# // Note: Applications should call show/hide after constructor.
# ************************************************************************/
class PysolToolbar(PysolToolbarActions):
def __init__(self, top, dir, size=0, relief=Tkinter.FLAT,
compound=Tkinter.NONE):
PysolToolbarActions.__init__(self)
self.top = top
self.side = -1
self._tooltips = []
self._widgets = []
self.dir = dir
self.size = size
self.compound = compound
self.orient='horizontal'
#
self.frame = Tkinter.Frame(top, class_='Toolbar',
relief=TkSettings.toolbar_relief,
borderwidth=TkSettings.toolbar_borderwidth)
#
for l, f, t in (
(n_("New"), self.mNewGame, _("New game")),
(n_("Restart"), self.mRestart, _("Restart the\ncurrent game")),
(None, None, None),
(n_("Open"), self.mOpen, _("Open a\nsaved game")),
(n_("Save"), self.mSave, _("Save game")),
(None, None, None),
(n_("Undo"), self.mUndo, _("Undo last move")),
(n_("Redo"), self.mRedo, _("Redo last move")),
(n_("Autodrop"), self.mDrop, _("Auto drop cards")),
(n_("Pause"), self.mPause, _("Pause game")),
(None, None, None),
(n_("Statistics"), self.mPlayerStats, _("View statistics")),
(n_("Rules"), self.mHelpRules, _("Rules for this game")),
(None, None, None),
(n_("Quit"), self.mQuit, _("Quit ")+PACKAGE),
):
if l is None:
sep = self._createSeparator()
sep.bind("<3>", self.rightclickHandler)
elif l == 'Pause':
self._createButton(l, f, check=True, tooltip=t)
else:
self._createButton(l, f, tooltip=t)
position=len(self._widgets)
self.frame.rowconfigure(position, weight=1)
self.frame.columnconfigure(position, weight=1)
#
self._createLabel("player", label=n_('Player'),
tooltip=_("Player options"))
#
self.player_label.bind("<1>",self.mOptPlayerOptions)
self.popup = None
self.frame.bind("<3>", self.rightclickHandler)
#
self.setCompound(compound, force=True)
def config(self, w, v):
if w == 'player':
# label
if v:
self.player_label.show(orient=self.orient)
else:
self.player_label.hide()
else:
# button
widget = getattr(self, w+'_button')
position = widget.position
if v:
widget.show(orient=self.orient)
else:
widget.hide()
#
prev_visible = None
last_visible = None
for w in self._widgets:
if isinstance(w, ToolbarSeparator):
if prev_visible is None or isinstance(prev_visible, ToolbarSeparator):
w.hide()
else:
w.show(orient=self.orient)
if w.visible:
prev_visible = w
if not isinstance(w, ToolbarLabel):
last_visible = w
if isinstance(last_visible, ToolbarSeparator):
last_visible.hide()
# util
def _loadImage(self, name):
file = os.path.join(self.dir, name)
image = None
for ext in IMAGE_EXTENSIONS:
file = os.path.join(self.dir, name+ext)
if os.path.isfile(file):
if Image:
image = ImageTk.PhotoImage(Image.open(file))
else:
image = Tkinter.PhotoImage(file=file)
break
return image
def _createSeparator(self):
position=len(self._widgets)
sep = ToolbarSeparator(self.frame,
position=position,
toolbar=self,
takefocus=0)
sep.show(orient=self.orient)
self._widgets.append(sep)
return sep
def _createButton(self, label, command, check=False, tooltip=None):
name = label.lower()
image = self._loadImage(name)
position = len(self._widgets)
kw = {
'position' : position,
'toolbar' : self,
'toolbar_name' : name,
'command' : command,
'takefocus' : 0,
'text' : gettext(label),
}
if image:
kw['image'] = image
if check:
button = ToolbarCheckbutton(self.frame, **kw)
else:
button = ToolbarButton(self.frame, **kw)
button.show(orient=self.orient)
setattr(self, name + "_image", image)
setattr(self, name + "_button", button)
self._widgets.append(button)
if tooltip:
b = MfxTooltip(button)
self._tooltips.append(b)
b.setText(tooltip)
return button
def _createLabel(self, name, label=None, tooltip=None):
aspect = (400, 300) [self.getSize() != 0]
position = len(self._widgets)+1
label = ToolbarLabel(self.frame,
position=position,
toolbar=self,
toolbar_name=name,
relief="ridge",
justify="center",
aspect=aspect)
label.show(orient=self.orient)
setattr(self, name + "_label", label)
self._widgets.append(label)
if tooltip:
b = MfxTooltip(label)
self._tooltips.append(b)
b.setText(tooltip)
return label
def _busy(self):
if not self.side or not self.game or not self.menubar:
return 1
self.game.stopDemo()
self.game.interruptSleep()
return self.game.busy
#
# public methods
#
def show(self, side=1, resize=1):
if self.side == side:
return 0
if resize:
self.top.wm_geometry("") # cancel user-specified geometry
if not side:
# hide
self.frame.grid_forget()
else:
# show
pack_func = self.frame.grid_configure
if side == 1:
# top
pack_func(row=0, column=1, sticky='ew', padx=0, pady=0)
elif side == 2:
# bottom
pack_func(row=2, column=1, sticky='ew', padx=0, pady=0)
elif side == 3:
# left
pack_func(row=1, column=0, sticky='ns', padx=0, pady=1)
else:
# right
pack_func(row=1, column=2, sticky='ns', padx=0, pady=1)
# set orient
orient = side in (1, 2) and Tkinter.HORIZONTAL or Tkinter.VERTICAL
self._setOrient(orient)
self.side = side
return 1
def hide(self, resize=1):
self.show(0, resize)
def destroy(self):
for w in self._tooltips:
if w: w.destroy()
self._tooltips = []
for w in self._widgets:
if w: w.destroy()
self._widgets = []
def setCursor(self, cursor):
if self.side:
self.frame.config(cursor=cursor)
self.frame.update_idletasks()
def connectGame(self, game, menubar):
PysolToolbarActions.connectGame(self, game, menubar)
if self.popup:
self.popup.destroy()
destruct(self.popup)
self.popup = None
if menubar:
tkopt = menubar.tkopt
self.pause_button.config(variable=tkopt.pause)
self.popup = MfxMenu(master=None, label=n_('Toolbar'), tearoff=0)
createToolbarMenu(menubar, self.popup)
def updateText(self, **kw):
for name in kw.keys():
label = getattr(self, name + "_label")
label["text"] = kw[name]
def updateImages(self, dir, size):
if dir == self.dir and size == self.size:
return 0
if not os.path.isdir(dir):
return 0
old_dir, old_size = self.dir, self.size
self.dir, self.size = dir, size
data = []
try:
for w in self._widgets:
if not isinstance(w, (ToolbarButton, ToolbarCheckbutton)):
continue
name = w.toolbar_name
image = self._loadImage(name)
data.append((name, w, image))
except:
self.dir, self.size = old_dir, old_size
return 0
l = self.player_label
aspect = (400, 300) [size != 0]
l.config(aspect=aspect)
for name, w, image in data:
w.config(image=image)
setattr(self, name + "_image", image)
self.setCompound(self.compound, force=True)
return 1
def setCompound(self, compound, force=False):
if not force and self.compound == compound:
return False
for w in self._widgets:
if not isinstance(w, (ToolbarButton, ToolbarCheckbutton)):
continue
if compound == 'text':
w.config(compound='none', image='')
else:
image = getattr(self, w.toolbar_name+'_image')
w.config(compound=compound, image=image)
self.compound = compound
return True
def _setOrient(self, orient='horizontal', force=False):
if not force and self.orient == orient:
return False
for w in self._widgets:
if w.visible:
w.show(orient=orient, force=True)
self.orient = orient
return True
#
# Mouse event handlers
#
def rightclickHandler(self, event):
if self._busy(): return EVENT_HANDLED
if self.popup:
##print event.x, event.y, event.x_root, event.y_root, event.__dict__
self.popup.tk_popup(event.x_root, event.y_root)
return EVENT_HANDLED
def getSize(self):
if self.compound == 'text':
return 0
size = self.size
comp = int(self.compound in (Tkinter.TOP, Tkinter.BOTTOM))
return int((size+comp) != 0)