diff --git a/README b/README index 6c9e4aa2..890880c5 100644 --- a/README +++ b/README @@ -14,7 +14,7 @@ Requirements. - PyGame: http://www.pygame.org/ (mp3, ogg, wav, midi, tracker music) ** other packages (optional) ** - - Tile: http://tktable.sourceforge.net/tile/ (0.7.8 or later) + - Tile (ttk): http://tktable.sourceforge.net/tile/ (0.8.0 or later) - PIL (Python Imaging Library): http://www.pythonware.com/products/pil - Freecell Solver: http://vipe.technion.ac.il/~shlomif/freecell-solver/ diff --git a/pysollib/pysolaudio.py b/pysollib/pysolaudio.py index db61be2f..8bb1ce64 100644 --- a/pysollib/pysolaudio.py +++ b/pysollib/pysolaudio.py @@ -418,7 +418,7 @@ class OSSAudioClient(AbstractAudioClient): class PyGameAudioClient(AbstractAudioClient): - EXTENSIONS = r'\.((ogg)|(mp3)|(wav)|(it)|(mod)|(s3m)|(xm)|(mid))$' + EXTENSIONS = r'\.((ogg)|(mp3)|(wav)|(it)|(mod)|(s3m)|(xm)|(mid)|(midi))$' CAN_PLAY_SOUND = True CAN_PLAY_MUSIC = True @@ -492,18 +492,22 @@ class PyGameAudioClient(AbstractAudioClient): self.music.set_volume(vol) self.music.play() while self.music and self.music.get_busy(): - self.time.wait(200) - if self.time: - self.time.wait(300) + self._wait(200) + self._wait(300) except: ##if traceback: traceback.print_exc() - self.time.wait(1000) + self._wait(1000) def _destroy(self): self.mixer.stop() self.mixer.quit() self.music = None + def _wait(self, s): + # sometime time or time.wait is None (threading) + if self.time and self.time.wait: + self.time.wait(s) + def playContinuousMusic(self, music_list): ##print 'playContinuousMusic' self.music_list = music_list diff --git a/pysollib/tile/Tile.py b/pysollib/tile/Tile.py deleted file mode 100644 index ba81291d..00000000 --- a/pysollib/tile/Tile.py +++ /dev/null @@ -1,618 +0,0 @@ -# http://tkinter.unpythonic.net/wiki/TileWrapper - -import Tkinter - -TileVersion = None -_tile_prefix = '' # XXX - -def initialize(root=None): - global TileVersion, _tile_prefix - if root is None: - root = Tkinter._default_root - TileVersion = root.tk.call("package", "require", "tile", "0.7.8") - if TileVersion >= '0.8': - _tile_prefix = 'ttk::' # XXX - -def availableThemes(root=None): - if root is None: - root = Tkinter._default_root - if TileVersion >= '0.8': - return root.tk.call("ttk::themes") - return root.tk.call("tile::availableThemes") - -def setTheme(root=None, theme=None): - if root is None: - root = Tkinter._default_root - if TileVersion >= '0.8': - return root.tk.call("ttk::setTheme", theme) - return root.tk.call("tile::setTheme", theme) - - -class Style(Tkinter.Misc): - def __init__(self, master=None): - if master is None: - master = Tkinter._default_root - self.tk = master.tk - - def default(self, style, **kw): - """Sets the default value of the specified option(s) in style""" - assert TileVersion < '0.8' # removed in Tile-0.8.0 - opts = self._options(kw) - return self.tk.call(_tile_prefix+"style", "default", style, *opts) - - def map_style(self, **kw): - """Sets dynamic values of the specified option(s) in style. See - "STATE MAPS", below. - """ - raise NotImplementedError() - - def layout(self, style, layoutSpec): - """Define the widget layout for style style. See "LAYOUTS" below - for the format of layoutSpec. If layoutSpec is omitted, return the - layout specification for style style. - """ - raise NotImplementedError() - - def element_create(self, name, type, *args): - """Creates a new element in the current theme of type type. The - only built-in element type is image (see image(n)), although - themes may define other element types (see - Ttk_RegisterElementFactory). - """ - raise NotImplementedError() - - def element_names(self): - """Returns a list of all elements defined in the current theme.""" - return self.tk.call(_tile_prefix+"style", "elements", "names") - - def theme_create(self, name, parent=None, basedon=None): - """Creates a new theme. It is an error if themeName already exists. - If -parent is specified, the new theme will inherit styles, elements, - and layouts from the parent theme basedon. If -settings is present, - script is evaluated in the context of the new theme as per style theme - settings. - """ - raise NotImplementedError() - - def theme_settings(self, name, script): - """Temporarily sets the current theme to themeName, evaluate script, - then restore the previous theme. Typically script simply defines - styles and elements, though arbitrary Tcl code may appear. - """ - raise NotImplementedError() - - def theme_names(self): - """Returns a list of the available themes.""" - return self.tk.call(_tile_prefix+"style", "theme", "names") - - def theme_use(self, theme): - """Sets the current theme to themeName, and refreshes all widgets.""" - return self.tk.call(_tile_prefix+"style", "theme", "use", theme) - - def configure(self, style, cnf={}, **kw): - """Sets the default value of the specified option(s) - in style.""" - opts = self._options(cnf, kw) - return self.tk.call(_tile_prefix+"style", "configure", style, *opts) - config = configure - - def lookup(self, style, option, state=None, default=None): - """Returns the value specified for -option in style - style in state state, using the standard lookup - rules for element options. state is a list of - state names; if omitted, it defaults to all bits - off (the ``normal'' state). If the default argu- - ment is present, it is used as a fallback value in - case no specification for -option is found.""" - opts = [] - if state: - opts = [state] - if default: - opts.append(default) - return self.tk.call(_tile_prefix+"style", "lookup", style, - "-"+option, *opts) - - - -class Widget(Tkinter.Widget, Style): - def __init__(self, master, widgetName=None, cnf={}, kw={}, extra=()): - if not widgetName: - ## why you would ever want to create a Tile Widget is behond me! - widgetName="ttk::widget" - Tkinter.Widget.__init__(self, master, widgetName, cnf, kw) - - def instate(self, spec=None, script=None): - """Test the widget's state. If script is not specified, returns 1 - if the widget state matches statespec and 0 otherwise. If script - is specified, equivalent to if {[pathName instate stateSpec]} - script. - """ - return self.tk.call(self._w, "instate", spec, script) - - def state(self, spec=None): - """Modify or inquire widget state. If stateSpec is present, sets - the widget state: for each flag in stateSpec, sets the corresponding - flag or clears it if prefixed by an exclamation point. Returns a new - state spec indicating which flags were changed: ''set changes - [pathName state spec] ; pathName state $changes'' will restore - pathName to the original state. If stateSpec is not specified, - returns a list of the currently-enabled state flags. - """ - return self.tk.call(self._w, "state", spec) - - -class Button(Widget, Tkinter.Button): - def __init__(self, master=None, cnf={}, **kw): - Widget.__init__(self, master, "ttk::button", cnf, kw) - - -class Checkbutton(Widget, Tkinter.Checkbutton): - def __init__(self, master=None, cnf={}, **kw): - Widget.__init__(self, master, "ttk::checkbutton", cnf, kw) - - -class Combobox(Widget, Tkinter.Entry): - def __init__(self, master=None, cnf={}, **kw): - Widget.__init__(self, master, "ttk::combobox", cnf, kw) - - def current(self, index=None): - """If index is supplied, sets the combobox value to the element - at position newIndex in the list of -values. Otherwise, returns - the index of the current value in the list of -values or -1 if - the current value does not appear in the list. - """ - return self.tk.call(self._w, "current", index) - - -class Entry(Widget, Tkinter.Entry): - def __init__(self, master=None, cnf={}, **kw): - Widget.__init__(self, master, "ttk::entry", cnf, kw) - - def validate(self): - """Force revalidation, independent of the conditions specified by - the -validate option. Returns 0 if the -validatecommand returns a - false value, or 1 if it returns a true value or is not specified. - """ - return self.tk.call(self._w, "validate") - - -class Label(Widget, Tkinter.Label): - def __init__(self, master=None, cnf={}, **kw): - Widget.__init__(self, master, "ttk::label", cnf, kw) - - -class Frame(Widget, Tkinter.Frame): - def __init__(self, master=None, cnf={}, **kw): - Widget.__init__(self, master, "ttk::frame", cnf, kw) - - -class Sizegrip(Widget): - def __init__(self, master=None, cnf={}, **kw): - Widget.__init__(self, master, "ttk::sizegrip", cnf, kw) - - -class LabelFrame(Widget, Tkinter.LabelFrame): - def __init__(self, master=None, cnf={}, **kw): - Widget.__init__(self, master, "ttk::labelframe", cnf, kw) - - -class Menubutton(Widget, Tkinter.Menubutton): - def __init__(self, master=None, cnf={}, **kw): - Widget.__init__(self, master, "ttk::menubutton", cnf, kw) - - -class Scale(Widget, Tkinter.Scale): - def __init__(self, master=None, cnf={}, **kw): - Widget.__init__(self, master, "ttk::scale", cnf, kw) - - -class Notebook(Widget): - def __init__(self, master=None, cnf={}, **kw): - Widget.__init__(self, master, "ttk::notebook", cnf, kw) - - def add(self, child, cnf=(), **kw): - """Adds a new tab to the notebook. When the tab is selected, the - child window will be displayed. child must be a direct child of - the notebook window. See TAB OPTIONS for the list of available - options. - """ - return self.tk.call((self._w, "add", child) + self._options(cnf, kw)) - - def forget(self, index): - """Removes the tab specified by index, unmaps and unmanages the - associated child window. - """ - return self.tk.call(self._w, "forget", index) - - def index(self, index): - """Returns the numeric index of the tab specified by index, or - the total number of tabs if index is the string "end". - """ - return self.tk.call(self._w, "index") - - def select(self, index=None): - """Selects the specified tab. The associated child pane will be - displayed, and the previously-selected pane (if different) is - unmapped. If tabid is omitted, returns the widget name of the - currently selected pane. - """ - return self.tk.call(self._w, "select", index) - - def tab(self, index, **kw): - """Query or modify the options of the specific tab. If no - -option is specified, returns a dictionary of the tab option - values. If one -option is specified, returns the value of tha - t option. Otherwise, sets the -options to the corresponding - values. See TAB OPTIONS for the available options. - """ - return self.tk.call((self._w, "tab", index) + self._options(kw)) - - def tabs(self): - """Returns a list of all pane windows managed by the widget.""" - return self.tk.call(self._w, "tabs") - - def enableTraversal(self): - """To enable keyboard traversal for a toplevel window containing a - notebook widget $nb, call: - - ttk::notebook::enableTraversal $nb - - This will extend the bindings for the toplevel widget containing the - notebook as follows: - - * Control-Tab selects the tab following the currently selected one. - * Shift-Control-Tab selects the tab preceding the currently selected - one. - * Alt-K, where K is the mnemonic (underlined) character of any tab, - will select that tab. - - Multiple notebooks in a single toplevel may be enabled for traversal, - including nested notebooks. However, notebook traversal only works - properly if all panes are direct children of the notebook.""" - return self.tk.call("ttk::notebook::enableTraversal", self._w) - - -class Paned(Widget): - """ - WIDGET OPTIONS - Name Database name Database class - -orient orient Orient - Specifies the orientation of the window. If vertical, subpanes - are stacked top-to-bottom; if horizontal, subpanes are stacked - left-to-right. - - PANE OPTIONS - The following options may be specified for each pane: - Name Database name Database class - -weight weight Weight - An integer specifying the relative stretchability of the pane. - When the paned window is resized, the extra space is added or - subracted to each pane proportionally to its -weight - """ - def __init__(self, master=None, cnf={}, **kw): - if 'orient' not in kw: - kw['orient'] = 'horizontal' - ##Widget.__init__(self, master, "ttk::paned", cnf, kw) - Widget.__init__(self, master, "ttk::panedwindow", cnf, kw) - - def add(self, subwindow, **kw): - """Adds a new pane to the window. subwindow must be a direct child of - the paned window pathname. See PANE OPTIONS for the list of available - options. - """ - return self.tk.call((self._w, "add", subwindow) + self._options(kw)) - - def forget(self, pane): - """Removes the specified subpane from the widget. pane is either an - integer index or the name of a managed subwindow. - """ - self.tk.call(self._w, "forget", pane) - - def insert(self, pos, subwindow, **kw): - """Inserts a pane at the specified position. pos is either the string - end, an integer index, or the name of a managed subwindow. If - subwindow is already managed by the paned window, moves it to the - specified position. See PANE OPTIONS for the list of available - options. - """ - return self.tk.call((self._w, "insert", pos, subwindow) + self._options(kw)) - - def pane(self, pane, **kw): - """Query or modify the options of the specified pane, where pane is - either an integer index or the name of a managed subwindow. If no - -option is specified, returns a dictionary of the pane option values. - If one -option is specified, returns the value of that option. - Otherwise, sets the -options to the corresponding values. - """ - return self.tk.call((self._w, "pane", pane) + self._options(kw)) - -PanedWindow = Paned - - -class Progressbar(Widget): - def __init__(self, master=None, cnf={}, **kw): - Widget.__init__(self, master, "ttk::progressbar", cnf, kw) - - def step(self, amount=1.0): - """Increments the -value by amount. amount defaults to 1.0 - if omitted. - """ - return self.tk.call(self._w, "step", amount) - - def start(self, interval=None): - """Begin autoincrement mode: schedules a recurring timer event that - calls step every interval milliseconds. If omitted, interval defaults - to 50 milliseconds (20 steps/second). - """ - self.tk.call("ttk::progressbar::start", self._w, interval) - - def stop(self): - """Stop autoincrement mode: cancels any recurring timer event - initiated by pathName start. - """ - self.tk.call("ttk::progressbar::stop", self._w) - - -class Radiobutton(Widget, Tkinter.Radiobutton): - def __init__(self, master=None, cnf={}, **kw): - Widget.__init__(self, master, "ttk::radiobutton", cnf, kw) - - -class Scrollbar(Widget, Tkinter.Scrollbar): - def __init__(self, master=None, cnf={}, **kw): - Widget.__init__(self, master, "ttk::scrollbar", cnf, kw) - - -class Separator(Widget): - def __init__(self, master=None, cnf={}, **kw): - Widget.__init__(self, master, "ttk::separator", cnf, kw) - - -class Treeview(Widget, Tkinter.Listbox): - def __init__(self, master=None, cnf={}, **kw): - Widget.__init__(self, master, 'ttk::treeview', cnf, kw) - - def children(self, item, newchildren=None): - """If newchildren is not specified, returns the list of - children belonging to item. - - If newchildren is specified, replaces item's child list - with newchildren. Items in the old child list not present - in the new child list are detached from the tree. None of - the items in newchildren may be an ancestor of item. - """ - return self.tk.call(self._w, "children", item, newchildren) - # Workaround: `children' overwrite in Tkinter.BaseWidget.__init__ - child = children - tree_children = children - - def column(self, column, **kw): - """Query or modify the options for the specified column. - If no options are specified, returns a dictionary of - option/value pairs. If a single option is specified, - returns the value of that option. Otherwise, the options - are updated with the specified values. The following - options may be set on each column: - - -id name - The column name. This is a read-only option. For example, - [$pathname column #n -id] returns the data column - associated with data column #n. - -anchor - Specifies how the text in this column should be aligned - with respect to the cell. One of n, ne, e, se, s, sw, w, - nw, or center. - -width w - The width of the column in pixels. Default is something - reasonable, probably 200 or so. - """ - return self.tk.call((self._w, 'column', column) + self._options(kw)) - - def delete(self, items): - """Deletes each of the items and all of their descendants. - The root item may not be deleted. See also: detach. - """ - return self.tk.call(self._w, "delete", items) - - def detach(self, items): - """Unlinks all of the specified items from the tree. The - items and all of their descendants are still present and - may be reinserted at another point in the tree but will - not be displayed. The root item may not be detached. See - also: delete. - """ - return self.tk.call(self._w, "detach", items) - - def exists(self, item): - """Returns 1 if the specified item is present in the - tree, 0 otherwise. - """ - return self.tk.call(self._w, "exists", item) - - def focus(self, item=None): - """If item is specified, sets the focus item to item. - Otherwise, returns the current focus item, or {} if there - is none. - """ - return self.tk.call(self._w, "focus", item) - - def heading(self, column, **kw): - """Query or modify the heading options for the specified - column. Valid options are: - - -text text - The text to display in the column heading. - -image imageName - Specifies an image to display to the right of the column heading. - -command script - A script to evaluate when the heading label is pressed. - """ - return self.tk.call((self._w, 'heading', column) + self._options(kw)) - - def identify(self, component, x, y): - """Returns a description of the specified component under the point - given by x and y, or the empty string if no such component is - present at that position. The following subcommands are - supported: - - pathname identify row x y - Returns the item ID of the item at position y. - - pathname identify column x y - Returns the data column identifier of the cell at position x. - The tree column has ID #0. - - See COLUMN IDENTIFIERS for a discussion of display columns and data - columns. - """ - return self.tk.call(self._w, "identify", component, x, y) - - def identify_row(self, x, y): - return self.identify('row', x, y) - - def identify_column(self, x, y): - return self.identify('column', x, y) - - def index(self, item): - """Returns the integer index of item within its parent's list of - children. - """ - return self.tk.call(self._w, "index", item) - - def insert(self, parent, index, id=None, **kw): - """Creates a new item. parent is the item ID of the parent item, or - the empty string {} to create a new top-level item. index is an - integer, or the value end, specifying where in the list of parent's - children to insert the new item. If index is less than or equal to - zero, the new node is inserted at the beginning; if index is greater - than or equal to the current number of children, it is inserted at the - end. If -id is specified, it is used as the item identifier; id must - not already exist in the tree. Otherwise, a new unique identifier is - generated. - returns the item identifier of the newly created item. See ITEM - OPTIONS for the list of available options. - """ - if not parent: parent = '' - args = (self._w, 'insert', parent, index) - if id: args = args + ('-id', id) - return self.tk.call(args + self._options(kw)) - - def item(self, item, **kw): - """Query or modify the options for the specified item. If no -option - is specified, returns a dictionary of option/value pairs. If a single - -option is specified, returns the value of that option. Otherwise, the - item's options are updated with the specified values. See ITEM OPTIONS - for the list of available options. - """ - return self.tk.call((self._w, 'item') + self._options(kw)) - - def move(self, item, parent, index): - """Moves item to position index in parent's list of children. It is - illegal to move an item under one of its descendants. - - If index is less than or equal to zero, item is moved to the - beginning; if greater than or equal to the number of children, it's - moved to the end. - """ - return self.tk.call(self._w, 'move', item, parent, index) - - def next(self, item): - """Returns the identifier of item's next sibling, or {} if item is the - last child of its parent. - """ - return self.tk.call(self._w, 'next', item) - - def parent(self, item): - """Returns the ID of the parent of item, or {} if item is at the top - level of the hierarchy. - """ - return self.tk.call(self._w, 'parent', item) - - def prev(self, item): - """Returns the identifier of item's previous sibling, or {} if item is - the first child of its parent. - """ - return self.tk.call(self._w, 'prev', item) - - def selection(self): - """Returns the list of selected items.""" - return self.tk.call(self._w, "selection") - - def selection_set(self, items): - """items becomes the new selection.""" - return self.tk.call(self._w, "selection", "set", items) - - def selection_add(self, items): - """Add items to the selection.""" - return self.tk.call(self._w, "selection", "add", items) - - def selection_remove(self, items): - """Remove items from the selection.""" - return self.tk.call(self._w, "selection", "remove", items) - - def selection_toggle(self, items): - """Toggle the selection state of each item in items.""" - return self.tk.call(self._w, "selection", "toggle", items) - - def set(self, item, column, value=None): - """If value is specified, sets the value of column column in item - item, otherwise returns the current value. See COLUMN IDENTIFIERS. - """ - raise NotImplementedError() - - -def test(): - import sys - root = Tkinter.Tk() - initialize() - root.option_add('*Toolbar.relief', 'groove') - root.option_add('*Toolbar.borderWidth', 2) - root.option_add('*Toolbar.Button.Pad', 2) - base = Frame(root) - base.pack(expand=True, fill='both') - tb_frame = Frame(base, class_='Toolbar') - tb_frame.pack(side='top', expand=False, fill='x') - b = Button(tb_frame, text='Open', style='Toolbutton') - b.grid(row=0, column=0, sticky='news') - b = Button(tb_frame, text='Save', style='Toolbutton') - b.grid(row=0, column=1, sticky='news') - b = Checkbutton(tb_frame, text='Bold', style='Toolbutton') - b.grid(row=0, column=2, sticky='news') - b = Checkbutton(tb_frame, text='Italic', style='Toolbutton') - b.grid(row=0, column=3, sticky='news') - tb_frame.grid_columnconfigure(4, weight=1) - theme_frame = LabelFrame(base, text='Theme') - theme_frame.pack(side='left', expand=False, fill='y', padx=4, pady=4) - theme_var = Tkinter.StringVar() - theme_var.set('default') - for theme in availableThemes(): - command = lambda theme=theme: setTheme(theme=theme) - b = Radiobutton(theme_frame, text=theme, variable=theme_var, - value=theme, command=command) - b.pack(side='top', expand=False, fill='x', padx=4) - right_frame = Frame(base) - right_frame.pack(side='right', expand=True, fill='both') - b = Checkbutton(right_frame, text='Checkbutton') - b.pack(expand=False, fill='x') - b = Button(right_frame, text='Button') - b.pack(expand=False, fill='x') - text_frame = Frame(right_frame) - text_frame.pack(expand=True, fill='both') - text = Tkinter.Text(text_frame, width=40, height=20, - fg='black', bg='white', wrap='none') - hsb = Scrollbar(text_frame, orient='horizontal', command=text.xview) - vsb = Scrollbar(text_frame, orient='vertical', command=text.yview) - text.grid(row=0, column=0, sticky='nwse') - hsb.grid(row=1, column=0, sticky='we') - vsb.grid(row=0, column=1, sticky='ns') - text.configure(xscrollcommand=hsb.set) - text.configure(yscrollcommand=vsb.set) - text.insert('end', open(sys.argv[0]).read()) - grip = Sizegrip(text_frame) - grip.grid(row=1, column=1, sticky='se') - text_frame.grid_columnconfigure(0, weight=1) - text_frame.grid_rowconfigure(0, weight=1) - root.mainloop() - -if __name__ == '__main__': - test() diff --git a/pysollib/tile/colorsdialog.py b/pysollib/tile/colorsdialog.py index 614f1dd7..6ecdd8e8 100644 --- a/pysollib/tile/colorsdialog.py +++ b/pysollib/tile/colorsdialog.py @@ -23,7 +23,7 @@ __all__ = ['ColorsDialog'] # imports import Tkinter -import Tile +import ttk from tkColorChooser import askcolor # PySol imports @@ -43,7 +43,7 @@ class ColorsDialog(MfxDialog): top_frame, bottom_frame = self.createFrames(kw) self.createBitmaps(top_frame, kw) - frame = Tile.Frame(top_frame) + frame = ttk.Frame(top_frame) frame.pack(expand=True, fill='both', padx=5, pady=10) frame.columnconfigure(0, weight=1) @@ -75,13 +75,13 @@ class ColorsDialog(MfxDialog): (_('Hint arrow:'), self.hintarrow_var), (_('Highlight not matching:'), self.not_matching_var), ): - Tile.Label(frame, text=title, anchor='w', - ).grid(row=row, column=0, sticky='we') + ttk.Label(frame, text=title, anchor='w', + ).grid(row=row, column=0, sticky='we') l = Tkinter.Label(frame, width=10, height=2, bg=var.get(), textvariable=var) l.grid(row=row, column=1, padx=5) - b = Tile.Button(frame, text=_('Change...'), width=10, - command=lambda l=l: self.selectColor(l)) + b = ttk.Button(frame, text=_('Change...'), width=10, + command=lambda l=l: self.selectColor(l)) b.grid(row=row, column=2) row += 1 # diff --git a/pysollib/tile/edittextdialog.py b/pysollib/tile/edittextdialog.py index 52212403..b5309138 100644 --- a/pysollib/tile/edittextdialog.py +++ b/pysollib/tile/edittextdialog.py @@ -37,7 +37,7 @@ __all__ = ['EditTextDialog'] # imports import Tkinter -import Tile +import ttk # PySol imports from pysollib.mfxutil import KwStruct @@ -61,7 +61,7 @@ class EditTextDialog(MfxDialog): wrap="word", width=64, height=16) self.text_w.pack(side='left', fill="both", expand=True) ###self.text_w.pack(side='top', padx=kw.padx, pady=kw.pady) - vbar = Tile.Scrollbar(top_frame) + vbar = ttk.Scrollbar(top_frame) vbar.pack(side='right', fill='y') self.text_w["yscrollcommand"] = vbar.set vbar["command"] = self.text_w.yview diff --git a/pysollib/tile/fontsdialog.py b/pysollib/tile/fontsdialog.py index 1c1e96fb..30602da2 100644 --- a/pysollib/tile/fontsdialog.py +++ b/pysollib/tile/fontsdialog.py @@ -23,7 +23,7 @@ __all__ = ['FontsDialog'] # imports import Tkinter -import Tile +import ttk import tkFont # PySol imports @@ -79,28 +79,28 @@ class FontChooserDialog(MfxDialog): self.size_var = Tkinter.IntVar() self.size_var.set(self.font_size) # - frame = Tile.Frame(top_frame) + frame = ttk.Frame(top_frame) frame.pack(expand=True, fill='both', padx=5, pady=10) frame.columnconfigure(0, weight=1) #frame.rowconfigure(1, weight=1) - self.entry = Tile.Entry(frame) + self.entry = ttk.Entry(frame) self.entry.grid(row=0, column=0, columnspan=2, sticky='news') self.entry.insert('end', _('abcdefghABCDEFGH')) self.list_box = Tkinter.Listbox(frame, width=36, exportselection=False) - sb = Tile.Scrollbar(frame) + sb = ttk.Scrollbar(frame) self.list_box.configure(yscrollcommand=sb.set) sb.configure(command=self.list_box.yview) self.list_box.grid(row=1, column=0, sticky='news') # rowspan=4 sb.grid(row=1, column=1, sticky='ns') bind(self.list_box, '<>', self.fontupdate) ##self.list_box.focus() - cb1 = Tile.Checkbutton(frame, text=_('Bold'), - command=self.fontupdate, - variable=self.weight_var) + cb1 = ttk.Checkbutton(frame, text=_('Bold'), + command=self.fontupdate, + variable=self.weight_var) cb1.grid(row=2, column=0, columnspan=2, sticky='we') - cb2 = Tile.Checkbutton(frame, text=_('Italic'), - command=self.fontupdate, - variable=self.slant_var) + cb2 = ttk.Checkbutton(frame, text=_('Italic'), + command=self.fontupdate, + variable=self.slant_var) cb2.grid(row=3, column=0, columnspan=2, sticky='we') sc = PysolScale(frame, from_=6, to=40, resolution=1, @@ -155,7 +155,7 @@ class FontsDialog(MfxDialog): top_frame, bottom_frame = self.createFrames(kw) self.createBitmaps(top_frame, kw) - frame = Tile.Frame(top_frame) + frame = ttk.Frame(top_frame) frame.pack(expand=True, fill='both', padx=5, pady=10) frame.columnconfigure(0, weight=1) @@ -172,16 +172,16 @@ class FontsDialog(MfxDialog): ): font = app.opt.fonts[fn] self.fonts[fn] = font - Tile.Label(frame, text=title, anchor='w' - ).grid(row=row, column=0, sticky='we') + ttk.Label(frame, text=title, anchor='w' + ).grid(row=row, column=0, sticky='we') if font: title = ' '.join([str(i) for i in font if i not in ('roman', 'normal')]) elif font is None: title = 'Default' - l = Tile.Label(frame, font=font, text=title) + l = ttk.Label(frame, font=font, text=title) l.grid(row=row, column=1, padx=8) - b = Tile.Button(frame, text=_('Change...'), width=10, - command=lambda l=l, fn=fn: self.selectFont(l, fn)) + b = ttk.Button(frame, text=_('Change...'), width=10, + command=lambda l=l, fn=fn: self.selectFont(l, fn)) b.grid(row=row, column=2) row += 1 # diff --git a/pysollib/tile/gameinfodialog.py b/pysollib/tile/gameinfodialog.py index b0b51b76..ee0c9344 100644 --- a/pysollib/tile/gameinfodialog.py +++ b/pysollib/tile/gameinfodialog.py @@ -23,7 +23,7 @@ __all__ = ['GameInfoDialog'] # imports -import Tile +import ttk # PySol imports from pysollib.mfxutil import KwStruct @@ -43,7 +43,7 @@ class GameInfoDialog(MfxDialog): top_frame, bottom_frame = self.createFrames(kw) self.createBitmaps(top_frame, kw) - frame = Tile.Frame(top_frame) + frame = ttk.Frame(top_frame) frame.pack(expand=True, fill='both', padx=5, pady=10) frame.columnconfigure(0, weight=1) @@ -107,10 +107,10 @@ class GameInfoDialog(MfxDialog): ('Hint:', hint), ): if t: - Tile.Label(frame, text=n, anchor='w' - ).grid(row=row, column=0, sticky='nw') - Tile.Label(frame, text=t, anchor='w', justify='left' - ).grid(row=row, column=1, sticky='nw') + ttk.Label(frame, text=n, anchor='w' + ).grid(row=row, column=0, sticky='nw') + ttk.Label(frame, text=t, anchor='w', justify='left' + ).grid(row=row, column=1, sticky='nw') row += 1 if game.s.talon: @@ -133,8 +133,8 @@ class GameInfoDialog(MfxDialog): self.mainloop(focus, kw.timeout) def showStacks(self, frame, row, title, stacks): - Tile.Label(frame, text=title, anchor='w' - ).grid(row=row, column=0, sticky='nw') + ttk.Label(frame, text=title, anchor='w' + ).grid(row=row, column=0, sticky='nw') if isinstance(stacks, (list, tuple)): fs = {} for f in stacks: @@ -146,8 +146,8 @@ class GameInfoDialog(MfxDialog): t = '\n'.join(['%s (%d)' % (i[0], i[1]) for i in fs.items()]) else: t = stacks.__class__.__name__ - Tile.Label(frame, text=t, anchor='w', justify='left' - ).grid(row=row, column=1, sticky='nw') + ttk.Label(frame, text=t, anchor='w', justify='left' + ).grid(row=row, column=1, sticky='nw') def initKw(self, kw): kw = KwStruct(kw, diff --git a/pysollib/tile/menubar.py b/pysollib/tile/menubar.py index 708e2ac1..07517902 100644 --- a/pysollib/tile/menubar.py +++ b/pysollib/tile/menubar.py @@ -38,8 +38,8 @@ __all__ = ['PysolMenubarTk'] # imports import math, os, sys, re, traceback -import Tile import Tkinter +import ttk import tkFileDialog # PySol imports @@ -1351,7 +1351,7 @@ the next time you restart """)+TITLE, def createThemesMenu(self, menu): submenu = MfxMenu(menu, label=n_("Set t&heme")) - all_themes = list(Tile.availableThemes()) + all_themes = list(ttk.Style(self.top).theme_names()) all_themes.sort() # tn = { diff --git a/pysollib/tile/playeroptionsdialog.py b/pysollib/tile/playeroptionsdialog.py index e34dc106..75279b5a 100644 --- a/pysollib/tile/playeroptionsdialog.py +++ b/pysollib/tile/playeroptionsdialog.py @@ -37,7 +37,7 @@ __all__ = ['PlayerOptionsDialog'] # imports import Tkinter -import Tile +import ttk # PySol imports from pysollib.mfxutil import KwStruct @@ -65,25 +65,25 @@ class PlayerOptionsDialog(MfxDialog): self.win_animation_var = Tkinter.BooleanVar() self.win_animation_var.set(app.opt.win_animation != 0) # - frame = Tile.Frame(top_frame) + frame = ttk.Frame(top_frame) frame.pack(expand=True, fill='both', padx=5, pady=10) - widget = Tile.Label(frame, text=_("\nPlease enter your name"), - takefocus=0) + widget = ttk.Label(frame, text=_("\nPlease enter your name"), + takefocus=0) widget.grid(row=0, column=0, columnspan=2, sticky='ew', padx=0, pady=5) # w = kw.get("e_width", 30) # width in characters names = self.app.getAllUserNames() - self.player_var = Tile.Combobox(frame, width=w, values=tuple(names)) + self.player_var = ttk.Combobox(frame, width=w, values=tuple(names)) self.player_var.current(names.index(app.opt.player)) self.player_var.grid(row=1, column=0, sticky='ew', padx=0, pady=5) # - widget = Tile.Checkbutton(frame, variable=self.confirm_var, - text=_("Confirm quit")) + widget = ttk.Checkbutton(frame, variable=self.confirm_var, + text=_("Confirm quit")) widget.grid(row=2, column=0, columnspan=2, sticky='ew', padx=0, pady=5) - widget = Tile.Checkbutton(frame, variable=self.update_stats_var, - text=_("Update statistics and logs")) + widget = ttk.Checkbutton(frame, variable=self.update_stats_var, + text=_("Update statistics and logs")) widget.grid(row=3, column=0, columnspan=2, sticky='ew', padx=0, pady=5) -### widget = Tile.Checkbutton(frame, variable=self.win_animation_var, +### widget = ttk.Checkbutton(frame, variable=self.win_animation_var, ### text="Win animation") ### widget.pack(side='top', padx=kw.padx, pady=kw.pady) frame.columnconfigure(0, weight=1) diff --git a/pysollib/tile/progressbar.py b/pysollib/tile/progressbar.py index a10262ba..2cd08c5e 100644 --- a/pysollib/tile/progressbar.py +++ b/pysollib/tile/progressbar.py @@ -37,7 +37,7 @@ __all__ = ['PysolProgressBar'] # imports import Tkinter -import Tile +import ttk # Toolkit imports from tkconst import EVENT_HANDLED @@ -59,15 +59,15 @@ class PysolProgressBar: self.top.wm_resizable(False, False) self.top.config(cursor="watch") # - self.frame = Tile.Frame(self.top, relief='flat', borderwidth=0) - self.progress = Tile.Progressbar(self.frame, maximum=100, length=250) - ##style = Tile.Style(self.progress) + self.frame = ttk.Frame(self.top, relief='flat', borderwidth=0) + self.progress = ttk.Progressbar(self.frame, maximum=100, length=250) + ##style = ttk.Style(self.progress) ##style.configure('TProgressbar', background=color) if images: - self.f1 = Tile.Label(self.frame, image=images[0]) + self.f1 = ttk.Label(self.frame, image=images[0]) self.f1.pack(side='left', ipadx=8, ipady=4) self.progress.pack(side='left', expand=True, fill='x') - self.f2 = Tile.Label(self.frame, image=images[1]) + self.f2 = ttk.Label(self.frame, image=images[1]) self.f2.pack(side='left', ipadx=8, ipady=4) else: self.progress.pack(expand=True, fill='x') diff --git a/pysollib/tile/selectcardset.py b/pysollib/tile/selectcardset.py index 0efd6662..3362f3ba 100644 --- a/pysollib/tile/selectcardset.py +++ b/pysollib/tile/selectcardset.py @@ -38,7 +38,7 @@ __all__ = ['SelectCardsetDialogWithPreview'] # imports import os import Tkinter -import Tile +import ttk # PySol imports from pysollib.mfxutil import KwStruct @@ -201,10 +201,10 @@ class SelectCardsetDialogWithPreview(MfxDialog): w1, w2 = 216, 400 else: w1, w2 = 200, 300 - paned_window = Tile.PanedWindow(top_frame) + paned_window = ttk.PanedWindow(top_frame, orient='horizontal') paned_window.pack(expand=True, fill='both') - left_frame = Tile.Frame(paned_window) - right_frame = Tile.Frame(paned_window) + left_frame = ttk.Frame(paned_window) + right_frame = ttk.Frame(paned_window) paned_window.add(left_frame) paned_window.add(right_frame) font = app.getFont("default") @@ -308,11 +308,11 @@ class CardsetInfoDialog(MfxDialog): MfxDialog.__init__(self, parent, title, kw.resizable, kw.default) top_frame, bottom_frame = self.createFrames(kw) self.createBitmaps(top_frame, kw) - frame = Tile.Frame(top_frame) + frame = ttk.Frame(top_frame) frame.pack(fill="both", expand=True, padx=5, pady=10) # # - info_frame = Tile.LabelFrame(frame, text=_('About cardset')) + info_frame = ttk.LabelFrame(frame, text=_('About cardset')) info_frame.grid(row=0, column=0, columnspan=2, sticky='ew', padx=0, pady=5, ipadx=5, ipady=5) styles = nationalities = year = None @@ -334,11 +334,11 @@ class CardsetInfoDialog(MfxDialog): (_('Size:'), '%d x %d' % (cardset.CARDW, cardset.CARDH)), ): if t is not None: - l = Tile.Label(info_frame, text=n, - anchor='w', justify='left') + l = ttk.Label(info_frame, text=n, + anchor='w', justify='left') l.grid(row=row, column=0, sticky='nw', padx=4) - l = Tile.Label(info_frame, text=t, - anchor='w', justify='left') + l = ttk.Label(info_frame, text=t, + anchor='w', justify='left') l.grid(row=row, column=1, sticky='nw', padx=4) row += 1 if images: @@ -347,10 +347,10 @@ class CardsetInfoDialog(MfxDialog): im = choice(images) f = os.path.join(cardset.dir, cardset.backname) self.back_image = loadImage(file=f) # store the image - l = Tile.Label(info_frame, image=im, padding=5) + l = ttk.Label(info_frame, image=im, padding=5) l.grid(row=0, column=2, rowspan=row+1, sticky='ne') - l = Tile.Label(info_frame, image=self.back_image, - padding=(0,5,5,5)) # left margin = 0 + l = ttk.Label(info_frame, image=self.back_image, + padding=(0,5,5,5)) # left margin = 0 l.grid(row=0, column=3, rowspan=row+1, sticky='ne') info_frame.columnconfigure(2, weight=1) @@ -362,7 +362,7 @@ class CardsetInfoDialog(MfxDialog): text_w = Tkinter.Text(frame, bd=1, relief="sunken", wrap="word", padx=4, width=64, height=16, bg=bg) text_w.grid(row=1, column=0, sticky='nsew') - sb = Tile.Scrollbar(frame) + sb = ttk.Scrollbar(frame) sb.grid(row=1, column=1, sticky='ns') text_w.configure(yscrollcommand=sb.set) sb.configure(command=text_w.yview) diff --git a/pysollib/tile/selectgame.py b/pysollib/tile/selectgame.py index 0caf34ce..0d1617d6 100644 --- a/pysollib/tile/selectgame.py +++ b/pysollib/tile/selectgame.py @@ -36,7 +36,7 @@ # imports import os -import Tile +import ttk from UserList import UserList # PySol imports @@ -363,10 +363,10 @@ class SelectGameDialogWithPreview(SelectGameDialog): #padx, pady = kw.padx/2, kw.pady/2 padx, pady = 4, 4 # PanedWindow - paned_window = Tile.PanedWindow(top_frame) + paned_window = ttk.PanedWindow(top_frame, orient='horizontal') paned_window.pack(expand=True, fill='both', padx=8, pady=8) - left_frame = Tile.Frame(paned_window) - right_frame = Tile.Frame(paned_window) + left_frame = ttk.Frame(paned_window) + right_frame = ttk.Frame(paned_window) paned_window.add(left_frame) paned_window.add(right_frame) # Tree @@ -375,10 +375,10 @@ class SelectGameDialogWithPreview(SelectGameDialog): default=kw.default, font=font, width=w1) self.tree.frame.pack(padx=padx, pady=pady, expand=True, fill='both') # LabelFrame - info_frame = Tile.LabelFrame(right_frame, text=_('About game')) + info_frame = ttk.LabelFrame(right_frame, text=_('About game')) info_frame.grid(row=0, column=0, padx=padx, pady=pady, ipadx=4, ipady=4, sticky='nws') - stats_frame = Tile.LabelFrame(right_frame, text=_('Statistics')) + stats_frame = ttk.LabelFrame(right_frame, text=_('Statistics')) stats_frame.grid(row=0, column=1, padx=padx, pady=pady, ipadx=4, ipady=4, sticky='nws') # Info @@ -400,9 +400,9 @@ class SelectGameDialogWithPreview(SelectGameDialog): ('moves', _('Moves:'), stats_frame, 4), ('percent', _('% won:'), stats_frame, 5), ): - title_label = Tile.Label(f, text=t, justify='left', anchor='w') + title_label = ttk.Label(f, text=t, justify='left', anchor='w') title_label.grid(row=row, column=0, sticky='nw', padx=4) - text_label = Tile.Label(f, justify='left', anchor='w') + text_label = ttk.Label(f, justify='left', anchor='w') text_label.grid(row=row, column=1, sticky='nw', padx=4) self.info_labels[n] = (title_label, text_label) ##info_frame.columnconfigure(1, weight=1) diff --git a/pysollib/tile/selecttile.py b/pysollib/tile/selecttile.py index 9ed8c6f5..626314ed 100644 --- a/pysollib/tile/selecttile.py +++ b/pysollib/tile/selecttile.py @@ -36,7 +36,7 @@ # imports import Tkinter -import Tile +import ttk import tkColorChooser # PySol imports @@ -131,7 +131,7 @@ class SelectTileDialogWithPreview(MfxDialog): w1, w2 = 200, 300 font = app.getFont("default") padx, pady = 4, 4 - frame = Tile.Frame(top_frame) + frame = ttk.Frame(top_frame) frame.pack(fill='both', expand=True, padx=kw.padx-padx, pady=kw.pady-pady) self.tree = self.Tree_Class(self, frame, key=key, default=kw.default, diff --git a/pysollib/tile/solverdialog.py b/pysollib/tile/solverdialog.py index ac5620d5..cb0378c6 100644 --- a/pysollib/tile/solverdialog.py +++ b/pysollib/tile/solverdialog.py @@ -29,7 +29,7 @@ __all__ = [ # imports import Tkinter -import Tile +import ttk # PySol imports from pysollib.settings import TITLE @@ -66,14 +66,14 @@ class SolverDialog(MfxDialog): self.games = {} # key: gamename; value: gameid # - frame = Tile.Frame(top_frame) + frame = ttk.Frame(top_frame) frame.pack(expand=True, fill='both', padx=4, pady=4) frame.columnconfigure(1, weight=1) # row = 0 - Tile.Label(frame, text=_('Game:'), anchor='w' - ).grid(row=row, column=0, sticky='ew', padx=2, pady=2) + ttk.Label(frame, text=_('Game:'), anchor='w' + ).grid(row=row, column=0, sticky='ew', padx=2, pady=2) games = app.getGamesForSolver() gamenames = [''] for id in games: @@ -90,8 +90,8 @@ class SolverDialog(MfxDialog): # row += 1 - Tile.Label(frame, text=_('Solving method:'), anchor='w' - ).grid(row=row, column=0, sticky='ew', padx=2, pady=2) + ttk.Label(frame, text=_('Solving method:'), anchor='w' + ).grid(row=row, column=0, sticky='ew', padx=2, pady=2) ##sm = self.solving_methods.values() ##sm.sort() sm = ['A*', @@ -107,8 +107,8 @@ class SolverDialog(MfxDialog): # row += 1 - Tile.Label(frame, text=_('Preset:'), anchor='w' - ).grid(row=row, column=0, sticky='ew', padx=2, pady=2) + ttk.Label(frame, text=_('Preset:'), anchor='w' + ).grid(row=row, column=0, sticky='ew', padx=2, pady=2) presets = [ 'none', 'abra-kadabra', @@ -131,8 +131,8 @@ class SolverDialog(MfxDialog): row += 1 self.max_iters_var = Tkinter.IntVar() self.max_iters_var.set(10e4) - Tile.Label(frame, text=_('Max iterations:'), anchor='w' - ).grid(row=row, column=0, sticky='ew', padx=2, pady=2) + ttk.Label(frame, text=_('Max iterations:'), anchor='w' + ).grid(row=row, column=0, sticky='ew', padx=2, pady=2) spin = Tkinter.Spinbox(frame, bg='white', from_=1000, to=10e6, increment=1000, textvariable=self.max_iters_var) spin.grid(row=row, column=1, sticky='w', padx=2, pady=2) @@ -141,8 +141,8 @@ class SolverDialog(MfxDialog): row += 1 self.max_depth_var = Tkinter.IntVar() self.max_depth_var.set(1000) - Tile.Label(frame, text=_('Max depth:'), anchor='w' - ).grid(row=row, column=0, sticky='ew', padx=2, pady=2) + ttk.Label(frame, text=_('Max depth:'), anchor='w' + ).grid(row=row, column=0, sticky='ew', padx=2, pady=2) spin = Tkinter.Spinbox(frame, bg='white', from_=100, to=10000, increment=100, textvariable=self.max_depth_var) spin.grid(row=row, column=1, sticky='w', padx=2, pady=2) @@ -151,38 +151,38 @@ class SolverDialog(MfxDialog): row += 1 self.progress_var = Tkinter.BooleanVar() self.progress_var.set(True) - w = Tile.Checkbutton(frame, variable=self.progress_var, - text=_('Show progress')) + w = ttk.Checkbutton(frame, variable=self.progress_var, + text=_('Show progress')) w.grid(row=row, column=0, columnspan=2, sticky='ew', padx=2, pady=2) # - label_frame = Tile.LabelFrame(top_frame, text=_('Progress')) + label_frame = ttk.LabelFrame(top_frame, text=_('Progress')) label_frame.pack(expand=True, fill='both', padx=6, pady=2) #label_frame.columnconfigure(0, weight=1) label_frame.columnconfigure(1, weight=1) # frow = 0 - Tile.Label(label_frame, text=_('Iteration:'), anchor='w' - ).grid(row=frow, column=0, sticky='ew', padx=4, pady=2) - lb = Tile.Label(label_frame, anchor='w') + ttk.Label(label_frame, text=_('Iteration:'), anchor='w' + ).grid(row=frow, column=0, sticky='ew', padx=4, pady=2) + lb = ttk.Label(label_frame, anchor='w') lb.grid(row=frow, column=1, sticky='ew', padx=4, pady=2) self.iter_label = lb frow += 1 - Tile.Label(label_frame, text=_('Depth:'), anchor='w' - ).grid(row=frow, column=0, sticky='ew', padx=4, pady=2) - lb = Tile.Label(label_frame, anchor='w') + ttk.Label(label_frame, text=_('Depth:'), anchor='w' + ).grid(row=frow, column=0, sticky='ew', padx=4, pady=2) + lb = ttk.Label(label_frame, anchor='w') lb.grid(row=frow, column=1, sticky='ew', padx=4, pady=2) self.depth_label = lb frow += 1 - Tile.Label(label_frame, text=_('Stored-States:'), anchor='w' - ).grid(row=frow, column=0, sticky='ew', padx=4, pady=2) - lb = Tile.Label(label_frame, anchor='w') + ttk.Label(label_frame, text=_('Stored-States:'), anchor='w' + ).grid(row=frow, column=0, sticky='ew', padx=4, pady=2) + lb = ttk.Label(label_frame, anchor='w') lb.grid(row=frow, column=1, sticky='ew', padx=4, pady=2) self.states_label = lb # - lb = Tile.Label(top_frame, anchor='w') + lb = ttk.Label(top_frame, anchor='w') lb.pack(expand=True, fill='x', padx=6, pady=4) self.result_label = lb diff --git a/pysollib/tile/soundoptionsdialog.py b/pysollib/tile/soundoptionsdialog.py index e45f05d7..cdbe0d12 100644 --- a/pysollib/tile/soundoptionsdialog.py +++ b/pysollib/tile/soundoptionsdialog.py @@ -38,7 +38,7 @@ __all__ = ['SoundOptionsDialog'] # imports import os import Tkinter -import Tile +import ttk # PySol imports from pysollib.mfxutil import KwStruct @@ -104,34 +104,34 @@ class SoundOptionsDialog(MfxDialog): ] # - frame = Tile.Frame(top_frame) + frame = ttk.Frame(top_frame) frame.pack(expand=True, fill='both', padx=5, pady=5) frame.columnconfigure(1, weight=1) # row = 0 - w = Tile.Checkbutton(frame, variable=self.sound, - text=_("Sound enabled")) + w = ttk.Checkbutton(frame, variable=self.sound, + text=_("Sound enabled")) w.grid(row=row, column=0, columnspan=2, sticky='ew') # if os.name == "nt" and pysolsoundserver: row += 1 - w = Tile.Checkbutton(frame, variable=self.sound_mode, - text=_("Use DirectX for sound playing"), - command=self.mOptSoundDirectX) + w = ttk.Checkbutton(frame, variable=self.sound_mode, + text=_("Use DirectX for sound playing"), + command=self.mOptSoundDirectX) w.grid(row=row, column=0, columnspan=2, sticky='ew') # if app.audio.CAN_PLAY_MUSIC: # and app.startup_opt.sound_mode > 0: row += 1 - Tile.Label(frame, text=_('Sample volume:'), anchor='w' - ).grid(row=row, column=0, sticky='ew') + ttk.Label(frame, text=_('Sample volume:'), anchor='w' + ).grid(row=row, column=0, sticky='ew') w = PysolScale(frame, from_=0, to=128, resolution=1, orient='horizontal', takefocus=0, length="3i", #label=_('Sample volume'), variable=self.sample_volume) w.grid(row=row, column=1, sticky='w', padx=5) row += 1 - Tile.Label(frame, text=_('Music volume:'), anchor='w' - ).grid(row=row, column=0, sticky='ew') + ttk.Label(frame, text=_('Music volume:'), anchor='w' + ).grid(row=row, column=0, sticky='ew') w = PysolScale(frame, from_=0, to=128, resolution=1, orient='horizontal', takefocus=0, length="3i", #label=_('Music volume'), @@ -142,7 +142,7 @@ class SoundOptionsDialog(MfxDialog): # remove "Apply" button kw.strings[1] = None # - frame = Tile.LabelFrame(top_frame, text=_('Enable samples')) + frame = ttk.LabelFrame(top_frame, text=_('Enable samples')) frame.pack(expand=True, fill='both', padx=5, pady=5) frame.columnconfigure(0, weight=1) frame.columnconfigure(1, weight=1) @@ -151,7 +151,7 @@ class SoundOptionsDialog(MfxDialog): col = 0 for n, t, v in self.samples: v.set(app.opt.sound_samples[n]) - w = Tile.Checkbutton(frame, text=t, variable=v) + w = ttk.Checkbutton(frame, text=t, variable=v) w.grid(row=row, column=col, sticky='ew', padx=3, pady=1) if col == 1: col = 0 diff --git a/pysollib/tile/statusbar.py b/pysollib/tile/statusbar.py index f1a19cfa..a650b708 100644 --- a/pysollib/tile/statusbar.py +++ b/pysollib/tile/statusbar.py @@ -39,7 +39,7 @@ __all__ = ['PysolStatusbar', # imports import os, sys import Tkinter -import Tile +import ttk if __name__ == '__main__': d = os.path.abspath(os.path.join(sys.path[0], os.pardir, os.pardir)) @@ -70,15 +70,15 @@ class MfxStatusbar: # self.padx = 1 self.label_relief = 'sunken' - self.top_frame = Tile.Frame(self.top) + self.top_frame = ttk.Frame(self.top) self.top_frame.grid(row=self._row, column=self._column, columnspan=self._columnspan, sticky='ew') - self.frame = Tile.Frame(self.top_frame) + self.frame = ttk.Frame(self.top_frame) self.frame.pack(side='left', expand=True, fill='both', padx=0, pady=1) # util def _createLabel(self, name, expand=False, width=0, tooltip=None): - frame = Tile.Frame(self.frame, borderwidth=1, relief=self.label_relief) + frame = ttk.Frame(self.frame, borderwidth=1, relief=self.label_relief) frame.grid(row=0, column=self._label_column, sticky='nsew', padx=self.padx) if expand: @@ -87,7 +87,7 @@ class MfxStatusbar: self._label_column += 1 setattr(self, name + '_frame', frame) self._widgets.append(frame) - label = Tile.Label(frame, width=width, anchor='center') + label = ttk.Label(frame, width=width, anchor='center') label.pack(expand=True, fill='both') setattr(self, name + '_label', label) self._widgets.append(label) @@ -98,7 +98,7 @@ class MfxStatusbar: return label def _createSizegrip(self): - sg = Tile.Sizegrip(self.top_frame) + sg = ttk.Sizegrip(self.top_frame) sg.pack(side='right', anchor='se') diff --git a/pysollib/tile/timeoutsdialog.py b/pysollib/tile/timeoutsdialog.py index 8d2faf76..e5edc93e 100644 --- a/pysollib/tile/timeoutsdialog.py +++ b/pysollib/tile/timeoutsdialog.py @@ -23,7 +23,7 @@ __all__ = ['TimeoutsDialog'] # imports import Tkinter -import Tile +import ttk # PySol imports from pysollib.mfxutil import KwStruct @@ -43,7 +43,7 @@ class TimeoutsDialog(MfxDialog): top_frame, bottom_frame = self.createFrames(kw) #self.createBitmaps(top_frame, kw) - frame = Tile.Frame(top_frame) + frame = ttk.Frame(top_frame) frame.pack(expand=True, fill='both', padx=5, pady=10) frame.columnconfigure(0, weight=1) @@ -60,7 +60,7 @@ class TimeoutsDialog(MfxDialog): self.highlight_samerank_sleep_var = Tkinter.DoubleVar() self.highlight_samerank_sleep_var.set(app.opt.timeouts['highlight_samerank']) # - lframe = Tile.LabelFrame(frame, text=_('Set delays in seconds'), + lframe = ttk.LabelFrame(frame, text=_('Set delays in seconds'), padding=(10, 5)) lframe.pack(expand=True, fill='both', padx=4) row = 0 @@ -72,7 +72,7 @@ class TimeoutsDialog(MfxDialog): (_('Highlight cards:'), self.highlight_cards_sleep_var), (_('Highlight same rank:'), self.highlight_samerank_sleep_var), ): - Tile.Label(lframe, text=title, anchor='w' + ttk.Label(lframe, text=title, anchor='w' ).grid(row=row, column=0, sticky='we') widget = PysolScale(lframe, from_=0.2, to=9.9, value=var.get(), resolution=0.1, orient='horizontal', diff --git a/pysollib/tile/tkhtml.py b/pysollib/tile/tkhtml.py index 4d9e3c40..78facadf 100644 --- a/pysollib/tile/tkhtml.py +++ b/pysollib/tile/tkhtml.py @@ -39,7 +39,7 @@ __all__ = ['HTMLViewer'] import os, sys import htmllib, formatter import Tkinter -import Tile +import ttk if __name__ == '__main__': d = os.path.abspath(os.path.join(sys.path[0], '..', '..')) @@ -248,34 +248,34 @@ class HTMLViewer: ##self.defcursor = 'xterm' self.handcursor = "hand2" - frame = Tile.Frame(parent, width=640, height=440) + frame = ttk.Frame(parent, width=640, height=440) frame.pack(expand=True, fill='both') frame.grid_propagate(False) # create buttons button_width = 8 - self.homeButton = Tile.Button(frame, text=_("Index"), - width=button_width, - command=self.goHome) + self.homeButton = ttk.Button(frame, text=_("Index"), + width=button_width, + command=self.goHome) self.homeButton.grid(row=0, column=0, sticky='w') - self.backButton = Tile.Button(frame, text=_("Back"), - width=button_width, - command=self.goBack) + self.backButton = ttk.Button(frame, text=_("Back"), + width=button_width, + command=self.goBack) self.backButton.grid(row=0, column=1, sticky='w') - self.forwardButton = Tile.Button(frame, text=_("Forward"), - width=button_width, - command=self.goForward) + self.forwardButton = ttk.Button(frame, text=_("Forward"), + width=button_width, + command=self.goForward) self.forwardButton.grid(row=0, column=2, sticky='w') - self.closeButton = Tile.Button(frame, text=_("Close"), - width=button_width, - command=self.destroy) + self.closeButton = ttk.Button(frame, text=_("Close"), + width=button_width, + command=self.destroy) self.closeButton.grid(row=0, column=3, sticky='e') # create text widget - text_frame = Tile.Frame(frame) + text_frame = ttk.Frame(frame) text_frame.grid(row=1, column=0, columnspan=4, sticky='nsew', padx=1, pady=1) - vbar = Tile.Scrollbar(text_frame) + vbar = ttk.Scrollbar(text_frame) vbar.pack(side='right', fill='y') self.text = Tkinter.Text(text_frame, fg='black', bg='white', diff --git a/pysollib/tile/tkstats.py b/pysollib/tile/tkstats.py index bbcc499b..278a5d65 100644 --- a/pysollib/tile/tkstats.py +++ b/pysollib/tile/tkstats.py @@ -46,7 +46,7 @@ __all__ = ['SingleGame_StatsDialog', import os import time import Tkinter -import Tile +import ttk import tkFont # PySol imports @@ -77,14 +77,14 @@ class StatsDialog(MfxDialog): self.font = app.getFont('default') self.tkfont = tkFont.Font(parent, self.font) self.font_metrics = self.tkfont.metrics() - style = Tile.Style() + style = ttk.Style(parent) heading_font = style.lookup('Heading', 'font') # treeview heading self.heading_tkfont = tkFont.Font(parent, heading_font) self.selected_game = None top_frame, bottom_frame = self.createFrames(kw) - notebook = Tile.Notebook(top_frame) + notebook = ttk.Notebook(top_frame) notebook.pack(expand=True, fill='both', padx=10, pady=10) self.notebook_tabs = [] @@ -167,20 +167,21 @@ SingleGame_StatsDialog = AllGames_StatsDialog = Top_StatsDialog = ProgressionDia # // # ************************************************************************/ -class SingleGameFrame(Tile.Frame): +class SingleGameFrame(ttk.Frame): def __init__(self, dialog, parent, app, player, gameid, **kw): - Tile.Frame.__init__(self, parent) + ttk.Frame.__init__(self, parent) self.oval_width = 120 self.oval_height = 60 - left_label = Tile.Label(self, image=app.gimages.logos[5]) + left_label = ttk.Label(self, image=app.gimages.logos[5]) left_label.pack(side='left', expand=True, fill='both') - self.right_frame = Tile.Frame(self) + self.right_frame = ttk.Frame(self) self.right_frame.pack(side='right', expand=True) self.dialog = dialog self.app = app + self.parent = parent self.player = player or _("Demo games") # self._calc_tabs() @@ -232,9 +233,9 @@ class SingleGameFrame(Tile.Frame): return pwon, plost def _createChartInit(self, text): - frame = Tile.LabelFrame(self.right_frame, text=text) + frame = ttk.LabelFrame(self.right_frame, text=text) frame.pack(side='top', fill='both', expand=False, padx=20, pady=10) - style = Tile.Style(self.right_frame) + style = ttk.Style(self.parent) fg = style.lookup('.', 'foreground') or None # use default if fg == '' bg = style.lookup('.', 'background') or None self.fg = fg @@ -383,14 +384,14 @@ class TreeFormatter(PysolStatsFormatter): for result in self.getStatResults(player, sort_by): # result == [name, won+lost, won, lost, time, moves, perc, id] t1, t2, t3, t4, t5, t6, t7, t8 = result - id = self.tree.insert(None, "end", text=t1, + id = self.tree.insert("", "end", text=t1, values=(t2, t3, t4, t5, t6, t7)) self.parent_window.tree_items.append(id) self.parent_window.games[id] = t8 total, played, won, lost, time_, moves, perc = self.getStatSummary() text = _("Total (%d out of %d games)") % (played, total) - id = self.tree.insert(None, "end", text=text, + id = self.tree.insert("", "end", text=text, values=(won+lost, won, lost, time_, moves, perc)) self.parent_window.tree_items.append(id) return 1 @@ -401,7 +402,7 @@ class TreeFormatter(PysolStatsFormatter): num_rows = 0 for result in self.getLogResults(player, prev_games): t1, t2, t3, t4, t5, t6 = result - id = self.tree.insert(None, "end", text=t1, values=(t2, t3, t4)) + id = self.tree.insert("", "end", text=t1, values=(t2, t3, t4)) self.parent_window.tree_items.append(id) num_rows += 1 if num_rows > self.MAX_ROWS: @@ -421,12 +422,12 @@ class TreeFormatter(PysolStatsFormatter): # // # ************************************************************************/ -class AllGamesFrame(Tile.Frame): +class AllGamesFrame(ttk.Frame): COLUMNS = ('played', 'won', 'lost', 'time', 'moves', 'percent') def __init__(self, dialog, parent, app, player, **kw): - Tile.Frame.__init__(self, parent) + ttk.Frame.__init__(self, parent) # self.dialog = dialog self.app = app @@ -439,22 +440,21 @@ class AllGamesFrame(Tile.Frame): self.tree_tabs = None self.games = {} # tree_itemid: gameid # - frame = Tile.Frame(self) + frame = ttk.Frame(self) frame.pack(fill='both', expand=True, padx=10, pady=10) - vsb = Tile.Scrollbar(frame) + vsb = ttk.Scrollbar(frame) vsb.grid(row=0, column=1, sticky='ns') - self.tree = Tile.Treeview(frame, columns=self.COLUMNS, - selectmode='browse') + self.tree = ttk.Treeview(frame, columns=self.COLUMNS, + selectmode='browse') self.tree.grid(row=0, column=0, sticky='nsew') self.tree.config(yscrollcommand=vsb.set) vsb.config(command=self.tree.yview) frame.rowconfigure(0, weight=1) frame.columnconfigure(0, weight=1) - if Tile.TileVersion >= '0.8': - hsb = Tile.Scrollbar(frame, orient='horizontal') - hsb.grid(row=1, column=0, sticky='ew') - self.tree.config(xscrollcommand=hsb.set) - hsb.config(command=self.tree.xview) + hsb = ttk.Scrollbar(frame, orient='horizontal') + hsb.grid(row=1, column=0, sticky='ew') + self.tree.config(xscrollcommand=hsb.set) + hsb.config(command=self.tree.xview) bind(self.tree, '<>', self.treeviewSelected) # self.formatter = TreeFormatter(self.app, self.tree, self, @@ -519,7 +519,7 @@ class LogDialog(MfxDialog): self.font = app.getFont('default') self.tkfont = tkFont.Font(parent, self.font) - style = Tile.Style() + style = ttk.Style(parent) heading_font = style.lookup('Heading', 'font') # treeview heading self.heading_tkfont = tkFont.Font(parent, heading_font) self.font_metrics = self.tkfont.metrics() @@ -534,7 +534,7 @@ class LogDialog(MfxDialog): ##self.selected_game = None top_frame, bottom_frame = self.createFrames(kw) - notebook = Tile.Notebook(top_frame) + notebook = ttk.Notebook(top_frame) notebook.pack(expand=True, fill='both', padx=10, pady=10) self.notebook_tabs = [] @@ -680,48 +680,48 @@ class _TopDialog(MfxDialog): cnf = {'master': top_frame, 'padding': (4, 1), } - frame = Tile.Frame(**cnf) + frame = ttk.Frame(**cnf) frame.pack(expand=True, fill='both', padx=10, pady=10) frame.columnconfigure(0, weight=1) cnf['master'] = frame cnf['text'] = _('N') - l = Tile.Label(**cnf) + l = ttk.Label(**cnf) l.grid(row=0, column=0, sticky='ew') if gameid == 'all': cnf['text'] = _('Game') - l = Tile.Label(**cnf) + l = ttk.Label(**cnf) l.grid(row=0, column=1, sticky='ew') cnf['text'] = _('Game number') - l = Tile.Label(**cnf) + l = ttk.Label(**cnf) l.grid(row=0, column=2, sticky='ew') cnf['text'] = _('Started at') - l = Tile.Label(**cnf) + l = ttk.Label(**cnf) l.grid(row=0, column=3, sticky='ew') cnf['text'] = _('Result') - l = Tile.Label(**cnf) + l = ttk.Label(**cnf) l.grid(row=0, column=4, sticky='ew') row = 1 for i in top: # N cnf['text'] = str(row) - l = Tile.Label(**cnf) + l = ttk.Label(**cnf) l.grid(row=row, column=0, sticky='ew') if gameid == 'all': name = app.getGameTitleName(i.gameid) if name is None: name = _("** UNKNOWN %d **") % i.gameid cnf['text'] = name - l = Tile.Label(**cnf) + l = ttk.Label(**cnf) l.grid(row=row, column=1, sticky='ew') # Game number cnf['text'] = '#'+str(i.game_number) - l = Tile.Label(**cnf) + l = ttk.Label(**cnf) l.grid(row=row, column=2, sticky='ew') # Start time t = time.strftime('%Y-%m-%d %H:%M', time.localtime(i.game_start_time)) cnf['text'] = t - l = Tile.Label(**cnf) + l = ttk.Label(**cnf) l.grid(row=row, column=3, sticky='ew') # Result if isinstance(i.value, float): @@ -731,7 +731,7 @@ class _TopDialog(MfxDialog): # moves s = str(i.value) cnf['text'] = s - l = Tile.Label(**cnf) + l = ttk.Label(**cnf) l.grid(row=row, column=4, sticky='ew') row += 1 @@ -744,31 +744,31 @@ class _TopDialog(MfxDialog): return MfxDialog.initKw(self, kw) -class TopFrame(Tile.Frame): +class TopFrame(ttk.Frame): def __init__(self, dialog, parent, app, player, gameid): - Tile.Frame.__init__(self, parent) + ttk.Frame.__init__(self, parent) self.app = app self.dialog = dialog - left_label = Tile.Label(self, image=app.gimages.logos[5]) + left_label = ttk.Label(self, image=app.gimages.logos[5]) left_label.pack(side='left', expand=True, fill='both') - frame = Tile.LabelFrame(self, text=_('Current game'), - padding=(10,5,10,10)) + frame = ttk.LabelFrame(self, text=_('Current game'), + padding=(10,5,10,10)) frame.pack(side='top', expand=True, fill='x', padx=10, pady=10) ##frame.columnconfigure(0, weight=1) if not self.createTopFrame(frame, player, gameid): - Tile.Label(frame, text=_('No TOP for this game') - ).pack(padx=10, pady=10) + ttk.Label(frame, text=_('No TOP for this game') + ).pack(padx=10, pady=10) - frame = Tile.LabelFrame(self, text=_('All games'), - padding=(10,5,10,10)) + frame = ttk.LabelFrame(self, text=_('All games'), + padding=(10,5,10,10)) frame.pack(side='top', expand=True, fill='x', padx=10, pady=10) ##frame.columnconfigure(0, weight=1) if not self.createTopFrame(frame, player, 'all'): - Tile.Label(frame, text=_('No TOP for all games') - ).pack(padx=10, pady=10) + ttk.Label(frame, text=_('No TOP for all games') + ).pack(padx=10, pady=10) def createTopFrame(self, frame, player, gameid): app = self.app @@ -778,13 +778,13 @@ class TopFrame(Tile.Frame): not app.stats.games_stats[player][gameid].time_result.top): return False - Tile.Label(frame, text=_('Minimum') - ).grid(row=0, column=1, padx=5, pady=5) - Tile.Label(frame, text=_('Maximum') - ).grid(row=0, column=2, padx=5, pady=5) - Tile.Label(frame, text=_('Average') - ).grid(row=0, column=3, padx=5, pady=5) - ##Tile.Label(frame, text=_('Total')).grid(row=0, column=4) + ttk.Label(frame, text=_('Minimum') + ).grid(row=0, column=1, padx=5, pady=5) + ttk.Label(frame, text=_('Maximum') + ).grid(row=0, column=2, padx=5, pady=5) + ttk.Label(frame, text=_('Average') + ).grid(row=0, column=3, padx=5, pady=5) + ##ttk.Label(frame, text=_('Total')).grid(row=0, column=4) s = app.stats.games_stats[player][gameid] @@ -825,19 +825,19 @@ class TopFrame(Tile.Frame): ## s.score_casino_result.max, ## round(s.score_casino_result.average, 2), )) for l, min, max, avr, tot, top in ll: - Tile.Label(frame, text=l - ).grid(row=row, column=0, padx=5, pady=5) - Tile.Label(frame, text=str(min) - ).grid(row=row, column=1, padx=5, pady=5) - Tile.Label(frame, text=str(max) - ).grid(row=row, column=2, padx=5, pady=5) - Tile.Label(frame, text=str(avr) - ).grid(row=row, column=3, padx=5, pady=5) - ##Tile.Label(frame, text=str(tot)).grid(row=row, column=4) + ttk.Label(frame, text=l + ).grid(row=row, column=0, padx=5, pady=5) + ttk.Label(frame, text=str(min) + ).grid(row=row, column=1, padx=5, pady=5) + ttk.Label(frame, text=str(max) + ).grid(row=row, column=2, padx=5, pady=5) + ttk.Label(frame, text=str(avr) + ).grid(row=row, column=3, padx=5, pady=5) + ##ttk.Label(frame, text=str(tot)).grid(row=row, column=4) def command(gameid=gameid, top=top): self.showTop(gameid, top) - b = Tile.Button(frame, text=TOP_TITLE+' ...', - width=10, command=command) + b = ttk.Button(frame, text=TOP_TITLE+' ...', + width=10, command=command) b.grid(row=row, column=5) row += 1 return True @@ -850,10 +850,10 @@ class TopFrame(Tile.Frame): # // # ************************************************************************/ -class ProgressionFrame(Tile.Frame): +class ProgressionFrame(ttk.Frame): def __init__(self, dialog, parent, app, player, gameid, **kw): - Tile.Frame.__init__(self, parent) + ttk.Frame.__init__(self, parent) self.mapped = False @@ -864,7 +864,7 @@ class ProgressionFrame(Tile.Frame): self.items = [] self.formatter = ProgressionFormatter(app, player, gameid) - frame = Tile.Frame(self) + frame = ttk.Frame(self) frame.pack(expand=True, fill='both', padx=5, pady=10) frame.columnconfigure(0, weight=1) @@ -890,19 +890,19 @@ class ProgressionFrame(Tile.Frame): canvas.pack(side='left', padx=5) # right frame - right_frame = Tile.Frame(frame) + right_frame = ttk.Frame(frame) right_frame.pack(side='left', fill='x', padx=5) self.all_games_variable = var = Tkinter.StringVar() var.set('all') - b = Tile.Radiobutton(right_frame, text=_('All games'), - variable=var, value='all', - command=self.updateGraph) + b = ttk.Radiobutton(right_frame, text=_('All games'), + variable=var, value='all', + command=self.updateGraph) b.pack(fill='x', expand=True, padx=3, pady=1) - b = Tile.Radiobutton(right_frame, text=_('Current game'), - variable=var, value='current', - command=self.updateGraph) + b = ttk.Radiobutton(right_frame, text=_('Current game'), + variable=var, value='current', + command=self.updateGraph) b.pack(fill='x', expand=True, padx=3, pady=1) - label_frame = Tile.LabelFrame(right_frame, text=_('Statistics for')) + label_frame = ttk.LabelFrame(right_frame, text=_('Statistics for')) label_frame.pack(side='top', fill='x', pady=10) self.variable = var = Tkinter.StringVar() var.set('week') @@ -912,28 +912,28 @@ class ProgressionFrame(Tile.Frame): ('year', _('Last year')), ('all', _('All time')), ): - b = Tile.Radiobutton(label_frame, text=t, variable=var, - value=v, command=self.updateGraph) + b = ttk.Radiobutton(label_frame, text=t, variable=var, + value=v, command=self.updateGraph) b.pack(fill='x', expand=True, padx=3, pady=1) - label_frame = Tile.LabelFrame(right_frame, text=_('Show graphs')) + label_frame = ttk.LabelFrame(right_frame, text=_('Show graphs')) label_frame.pack(side='top', fill='x') self.played_graph_var = Tkinter.BooleanVar() self.played_graph_var.set(True) - b = Tile.Checkbutton(label_frame, text=_('Played'), - command=self.updateGraph, - variable=self.played_graph_var) + b = ttk.Checkbutton(label_frame, text=_('Played'), + command=self.updateGraph, + variable=self.played_graph_var) b.pack(fill='x', expand=True, padx=3, pady=1) self.won_graph_var = Tkinter.BooleanVar() self.won_graph_var.set(True) - b = Tile.Checkbutton(label_frame, text=_('Won'), - command=self.updateGraph, - variable=self.won_graph_var) + b = ttk.Checkbutton(label_frame, text=_('Won'), + command=self.updateGraph, + variable=self.won_graph_var) b.pack(fill='x', expand=True, padx=3, pady=1) self.percent_graph_var = Tkinter.BooleanVar() self.percent_graph_var.set(True) - b = Tile.Checkbutton(label_frame, text=_('% won'), - command=self.updateGraph, - variable=self.percent_graph_var) + b = ttk.Checkbutton(label_frame, text=_('% won'), + command=self.updateGraph, + variable=self.percent_graph_var) b.pack(fill='x', expand=True, padx=3, pady=1) #self.createGraph() diff --git a/pysollib/tile/tkwidget.py b/pysollib/tile/tkwidget.py index e2753ea3..a5c3dab0 100644 --- a/pysollib/tile/tkwidget.py +++ b/pysollib/tile/tkwidget.py @@ -46,7 +46,7 @@ __all__ = ['MfxDialog', # imports import sys, os, time, locale import Tkinter -import Tile +import ttk import tkFont import traceback @@ -77,7 +77,7 @@ class MfxDialog: # ex. _ToplevelDialog self.buttons = [] self.accel_keys = {} self.top = makeToplevel(parent, title=title) - #self._frame = Tile.Frame(self.top) + #self._frame = ttk.Frame(self.top) #self._frame.pack(expand=True, fill='both') self._frame = self.top self.top.wm_resizable(resizable, resizable) @@ -167,23 +167,23 @@ class MfxDialog: # ex. _ToplevelDialog return kw def createFrames(self, kw): - bottom_frame = Tile.Frame(self._frame, relief='flat', borderwidth=4) + bottom_frame = ttk.Frame(self._frame, relief='flat', borderwidth=4) bottom_frame.pack(side='bottom', fill='both', expand=False) if kw.separator: - separator = Tile.Separator(self._frame) + separator = ttk.Separator(self._frame) separator.pack(side='bottom', fill='x') - top_frame = Tile.Frame(self._frame) + top_frame = ttk.Frame(self._frame) top_frame.pack(side='top', fill='both', expand=1) return top_frame, bottom_frame def createBitmaps(self, frame, kw): if kw.bitmap: ## in ("error", "info", "question", "warning") img = self.img.get(kw.bitmap) - b = Tile.Label(frame, image=img) + b = ttk.Label(frame, image=img) b.pack(side=kw.bitmap_side, padx=kw.bitmap_padx, pady=kw.bitmap_pady) elif kw.image: - b = Tile.Label(frame, image=kw.image) + b = ttk.Label(frame, image=kw.image) b.pack(side=kw.image_side, padx=kw.image_padx, pady=kw.image_pady) def createButtons(self, frame, kw): @@ -229,9 +229,9 @@ class MfxDialog: # ex. _ToplevelDialog button_img = MfxDialog.button_img.get(s) s = s.replace('&', '') if button < 0: - widget = Tile.Button(frame, text=s, state="disabled") + widget = ttk.Button(frame, text=s, state="disabled") else: - widget = Tile.Button(frame, text=s, default="normal", + widget = ttk.Button(frame, text=s, default="normal", command = lambda self=self, button=button: \ self.mDone(button)) if button == kw.default: @@ -274,8 +274,8 @@ class MfxMessageDialog(MfxDialog): self.createBitmaps(top_frame, kw) # self.button = kw.default - msg = Tile.Label(top_frame, text=kw.text, justify=kw.justify, - width=kw.width) + msg = ttk.Label(top_frame, text=kw.text, justify=kw.justify, + width=kw.width) msg.pack(fill='both', expand=True, padx=kw.padx, pady=kw.pady) # focus = self.createButtons(bottom_frame, kw) @@ -314,10 +314,10 @@ class PysolAboutDialog(MfxMessageDialog): self.createBitmaps(top_frame, kw) # self.button = kw.default - frame = Tile.Frame(top_frame) + frame = ttk.Frame(top_frame) frame.pack(fill='both', expand=True, padx=kw.padx, pady=kw.pady) - msg = Tile.Label(frame, text=kw.text, justify=kw.justify, - width=kw.width) + msg = ttk.Label(frame, text=kw.text, justify=kw.justify, + width=kw.width) msg.pack(fill='both', expand=True) if sys.version_info >= (2, 4): @@ -328,8 +328,8 @@ class PysolAboutDialog(MfxMessageDialog): else: font = tkFont.Font(parent, app.getFont('default')) font.configure(underline=True) - url_label = Tile.Label(frame, text=kw.url, font=font, - foreground='blue', cursor='hand2') + url_label = ttk.Label(frame, text=kw.url, font=font, + foreground='blue', cursor='hand2') url_label.pack() url_label.bind('<1>', self._urlClicked) # @@ -353,10 +353,10 @@ class MfxSimpleEntry(MfxDialog): # self.value = value if label: - label = Tile.Label(top_frame, text=label, takefocus=0) + label = ttk.Label(top_frame, text=label, takefocus=0) label.pack(pady=5) w = kw.get("e_width", 0) # width in characters - self.var = Tile.Entry(top_frame, exportselection=1, width=w) + self.var = ttk.Entry(top_frame, exportselection=1, width=w) self.var.insert(0, value) self.var.pack(side='top', padx=kw.padx, pady=kw.pady) # @@ -447,7 +447,7 @@ class MfxTooltip: self.timer = None if self.tooltip or not self.text: return -## if isinstance(self.widget, (Tile.Button, Tile.Checkbutton)): +## if isinstance(self.widget, (ttk.Button, ttk.Checkbutton)): ## if self.widget["state"] == 'disabled': ## return ##x = self.widget.winfo_rootx() @@ -559,7 +559,7 @@ class MfxScrolledCanvas: def createFrame(self, kw): width = kw.get("width") height = kw.get("height") - self.frame = Tile.Frame(self.parent, width=width, height=height) + self.frame = ttk.Frame(self.parent, width=width, height=height) def createCanvas(self, kw): bd = kw['bd'] kw['bd'] = 0 @@ -570,14 +570,14 @@ class MfxScrolledCanvas: self.canvas = MfxCanvas(frame, **kw) self.canvas.pack(expand=True, fill='both') def createHbar(self): - self.hbar = Tile.Scrollbar(self.frame, takefocus=0, - orient="horizontal") + self.hbar = ttk.Scrollbar(self.frame, takefocus=0, + orient="horizontal") self.canvas["xscrollcommand"] = self._setHbar self.hbar["command"] = self.canvas.xview self.hbar.grid(row=1, column=0, sticky="we") self.hbar.grid_remove() def createVbar(self): - self.vbar = Tile.Scrollbar(self.frame, takefocus=0) + self.vbar = ttk.Scrollbar(self.frame, takefocus=0) self.canvas["yscrollcommand"] = self._setVbar self.vbar["command"] = self.canvas.yview self.vbar.grid(row=0, column=1, sticky="ns") @@ -713,7 +713,7 @@ class StackDesc: # /*********************************************************************** -# // Tile.Scale workaround (label and resolution) +# // ttk.Scale workaround (label and resolution) # ************************************************************************/ class MyPysolScale: @@ -758,11 +758,11 @@ class MyPysolScale: # create widgets side = 'left' # 'top' - self.frame = Tile.Frame(parent) - self.label = Tile.Label(self.frame, anchor='w', - width=width, padding=(5,0)) + self.frame = ttk.Frame(parent) + self.label = ttk.Label(self.frame, anchor='w', + width=width, padding=(5,0)) self.label.pack(side=side, expand=False, fill='x') - self.scale = Tile.Scale(self.frame, **kw) + self.scale = ttk.Scale(self.frame, **kw) self.scale.pack(side=side, expand=True, fill='both', pady=4) if value is not None: @@ -812,16 +812,16 @@ PysolScale = MyPysolScale # /*********************************************************************** -# // Tile.Combobox workaround (clear selection) +# // ttk.Combobox workaround (clear selection) # ************************************************************************/ -class PysolCombo(Tile.Combobox): +class PysolCombo(ttk.Combobox): def __init__(self, master=None, **kw): self._command = None if 'selectcommand' in kw: self._command = kw['selectcommand'] del kw['selectcommand'] - Tile.Combobox.__init__(self, master, **kw) + ttk.Combobox.__init__(self, master, **kw) self.bind('<>', self._callback) def _callback(self, *args): diff --git a/pysollib/tile/tkwrap.py b/pysollib/tile/tkwrap.py index e5ae7c6e..a71b462d 100644 --- a/pysollib/tile/tkwrap.py +++ b/pysollib/tile/tkwrap.py @@ -39,7 +39,6 @@ __all__ = ['TclError', # imports import Tkinter TclError = Tkinter.TclError -import Tile # PySol imports from tkconst import EVENT_PROPAGATE @@ -53,7 +52,6 @@ from tkconst import EVENT_PROPAGATE class MfxRoot(Tkinter.Tk): def __init__(self, **kw): Tkinter.Tk.__init__(self, **kw) - Tile.initialize(self) self.app = None self.wm_protocol('WM_DELETE_WINDOW', self.wmDeleteWindow) # for interruptible sleep diff --git a/pysollib/tile/toolbar.py b/pysollib/tile/toolbar.py index c34c7f81..226a8d58 100644 --- a/pysollib/tile/toolbar.py +++ b/pysollib/tile/toolbar.py @@ -38,7 +38,7 @@ __all__ = ['PysolToolbarTk'] # imports import os import Tkinter -import Tile +import ttk # PySol imports from pysollib.mfxutil import destruct @@ -88,25 +88,25 @@ class AbstractToolbarButton: self.grid_forget() -class ToolbarCheckbutton(AbstractToolbarButton, Tile.Checkbutton): +class ToolbarCheckbutton(AbstractToolbarButton, ttk.Checkbutton): def __init__(self, parent, toolbar, toolbar_name, position, **kwargs): kwargs['style'] = 'Toolbutton' - Tile.Checkbutton.__init__(self, parent, **kwargs) + ttk.Checkbutton.__init__(self, parent, **kwargs) AbstractToolbarButton.__init__(self, parent, toolbar, toolbar_name, position) -class ToolbarButton(AbstractToolbarButton, Tile.Button): +class ToolbarButton(AbstractToolbarButton, ttk.Button): def __init__(self, parent, toolbar, toolbar_name, position, **kwargs): kwargs['style'] = 'Toolbutton' - Tile.Button.__init__(self, parent, **kwargs) + ttk.Button.__init__(self, parent, **kwargs) AbstractToolbarButton.__init__(self, parent, toolbar, toolbar_name, position) -class ToolbarSeparator(Tile.Separator): +class ToolbarSeparator(ttk.Separator): def __init__(self, parent, toolbar, position, **kwargs): kwargs['orient'] = 'vertical' - Tile.Separator.__init__(self, parent, **kwargs) + ttk.Separator.__init__(self, parent, **kwargs) self.toolbar = toolbar self.position = position self.visible = False @@ -179,9 +179,9 @@ class PysolToolbarTk: self.compound = compound self.orient='horizontal' # - self.frame = Tile.Frame(top, class_='Toolbar', - relief=TkSettings.toolbar_relief, - borderwidth=TkSettings.toolbar_borderwidth) + self.frame = ttk.Frame(top, class_='Toolbar', + relief=TkSettings.toolbar_relief, + borderwidth=TkSettings.toolbar_borderwidth) # for l, f, t in ( (n_("New"), self.mNewGame, _("New game")), diff --git a/pysollib/tile/ttk.py b/pysollib/tile/ttk.py new file mode 100644 index 00000000..6f6723e1 --- /dev/null +++ b/pysollib/tile/ttk.py @@ -0,0 +1,1486 @@ +"""Ttk wrapper. + +This module provides classes to allow using Tk themed widget set. + +Ttk is based on a revised and enhanced version of +TIP #48 (http://tip.tcl.tk/48) specified style engine. + +Its basic idea is to separate, to the extent possible, the code +implementing a widget's behavior from the code implementing its +appearance. Widget class bindings are primarily responsible for +maintaining the widget state and invoking callbacks, all aspects +of the widgets appearance lies at Themes. +""" + +__version__ = "0.1.2" + +__author__ = "Guilherme Polo " + +__all__ = ["Button", "Checkbutton", "Combobox", "Entry", "Frame", "Label", + "Labelframe", "LabelFrame", "Menubutton", "Notebook", "Panedwindow", + "PanedWindow", "Progressbar", "Radiobutton", "Scale", "Scrollbar", + "Separator", "Sizegrip", "Style", "Treeview", + # Extensions + "LabeledScale", "OptionMenu"] + +import Tkinter + +_flatten = Tkinter._flatten + +# Verify if Tk is new enough to not need Tile checking +_REQUIRE_TILE = True if Tkinter.TkVersion < 8.5 else False + +def _loadttk(loadtk): + # This extends the default Tkinter.Tk._loadtk method so we can be + # sure that ttk is available for use, or not. + def _wrapper(self): + loadtk(self) + + if _REQUIRE_TILE: + import os + tilelib = os.environ.get('TILE_LIBRARY') + if tilelib: + # append custom tile path to the the list of directories that + # Tcl uses when attempting to resolve packages with the package + # command + self.tk.eval('global auto_path; ' + 'lappend auto_path {%s}' % tilelib) + self.tk.eval('package require tile') # TclError may be raised here + + return _wrapper + +Tkinter.Tk._loadtk = _loadttk(Tkinter.Tk._loadtk) + +def _format_optdict(optdict, script=False, ignore=None): + """Formats optdict to pass it to tk.call. + + E.g. (script=False): + {'foreground': 'blue', 'padding': [1, 2, 3, 4]} returns: + ('-foreground', 'blue', '-padding', '1 2 3 4')""" + format = "%s" if not script else "{%s}" + + opts = [] + for opt, value in optdict.iteritems(): + if ignore and opt in ignore: + continue + + if isinstance(value, (list, tuple)): + v = [] + for val in value: + if isinstance(val, basestring): + v.append(unicode(val) if val else '{}') + else: + v.append(str(val)) + value = format % ' '.join(v) + + if script and value == '': + value = '{}' # empty string in Python is equivalent to {} in Tcl + + opts.append(("-%s" % opt, value)) + + return _flatten(opts) + +def _format_mapdict(mapdict, script=False): + """Formats mapdict to pass it to tk.call. + + E.g. (script=False): + {'expand': [('active', 'selected', 'grey'), ('focus', [1, 2, 3, 4])]} + + returns: + + ('-expand', '{active selected} grey focus {1, 2, 3, 4}')""" + # if caller passes a Tcl script to tk.call, all the values need to + # be grouped into words (arguments to a command in Tcl dialect) + format = "%s" if not script else "{%s}" + + opts = [] + for opt, value in mapdict.iteritems(): + + opt_val = [] + # each value in mapdict is expected to be a sequence, where each item + # is another sequence containing a state (or several) and a value + for statespec in value: + state, val = statespec[:-1], statespec[-1] + + if len(state) > 1: # group multiple states + state = "{%s}" % ' '.join(state) + else: # single state + # if it is empty (something that evaluates to False), then + # format it to Tcl code to denote the "normal" state + state = state[0] or '{}' + + if isinstance(val, (list, tuple)): # val needs to be grouped + val = "{%s}" % ' '.join(map(str, val)) + + opt_val.append("%s %s" % (state, val)) + + opts.append(("-%s" % opt, format % ' '.join(opt_val))) + + return _flatten(opts) + +def _format_elemcreate(etype, script=False, *args, **kw): + """Formats args and kw according to the given element factory etype.""" + spec = None + opts = () + if etype in ("image", "vsapi"): + if etype == "image": # define an element based on an image + # first arg should be the default image name + iname = args[0] + # next args, if any, are statespec/value pairs which is almost + # a mapdict, but we just need the value + imagespec = _format_mapdict({None: args[1:]})[1] + spec = "%s %s" % (iname, imagespec) + + else: + # define an element whose visual appearance is drawn using the + # Microsoft Visual Styles API which is responsible for the + # themed styles on Windows XP and Vista. + # Availability: Tk 8.6, Windows XP and Vista. + class_name, part_id = args[:2] + statemap = _format_mapdict({None: args[2:]})[1] + spec = "%s %s %s" % (class_name, part_id, statemap) + + opts = _format_optdict(kw, script) + + elif etype == "from": # clone an element + # it expects a themename and optionally an element to clone from, + # otherwise it will clone {} (empty element) + spec = args[0] # theme name + if len(args) > 1: # elementfrom specified + opts = (args[1], ) + + if script: + spec = '{%s}' % spec + opts = ' '.join(map(str, opts)) + + return spec, opts + +def _format_layoutlist(layout, indent=2, indent_size=2): + """Formats a layout list so we can pass the result to ttk::style + layout and ttk::style settings. Note that the layout doesn't has to + be a list necessarily. + + E.g.: + [("Menubutton.background", None), + ("Menubutton.button", {"children": + [("Menubutton.focus", {"children": + [("Menubutton.padding", {"children": + [("Menubutton.label", {"side": "left", "expand": 1})] + })] + })] + }), + ("Menubutton.indicator", {"side": "right"}) + ] + + returns: + + Menubutton.background + Menubutton.button -children { + Menubutton.focus -children { + Menubutton.padding -children { + Menubutton.label -side left -expand 1 + } + } + } + Menubutton.indicator -side right""" + script = [] + + for layout_elem in layout: + elem, opts = layout_elem + opts = opts or {} + fopts = ' '.join(map(str, _format_optdict(opts, True, "children"))) + head = "%s%s%s" % (' ' * indent, elem, (" %s" % fopts) if fopts else '') + + if "children" in opts: + script.append(head + " -children {") + indent += indent_size + newscript, indent = _format_layoutlist(opts['children'], indent) + script.append(newscript) + indent -= indent_size + script.append('%s}' % (' ' * indent)) + else: + script.append(head) + + return '\n'.join(script), indent + +def _script_from_settings(settings): + """Returns an appropriate script, based on settings, according to + theme_settings definition to be used by theme_settings and + theme_create.""" + script = [] + # a script will be generated according to settings passed, which + # will then be evaluated by Tcl + for name, opts in settings.iteritems(): + # will format specific keys according to Tcl code + if opts.get('configure'): # format 'configure' + s = ' '.join(map(str, _format_optdict(opts['configure'], True))) + script.append("ttk::style configure %s %s;" % (name, s)) + + if opts.get('map'): # format 'map' + s = ' '.join(map(str, _format_mapdict(opts['map'], True))) + script.append("ttk::style map %s %s;" % (name, s)) + + if 'layout' in opts: # format 'layout' which may be empty + if not opts['layout']: + s = 'null' # could be any other word, but this one makes sense + else: + s, _ = _format_layoutlist(opts['layout']) + script.append("ttk::style layout %s {\n%s\n}" % (name, s)) + + if opts.get('element create'): # format 'element create' + eopts = opts['element create'] + etype = eopts[0] + + # find where args end, and where kwargs start + argc = 1 # etype was the first one + while argc < len(eopts) and not hasattr(eopts[argc], 'iteritems'): + argc += 1 + + elemargs = eopts[1:argc] + elemkw = eopts[argc] if argc < len(eopts) and eopts[argc] else {} + spec, opts = _format_elemcreate(etype, True, *elemargs, **elemkw) + + script.append("ttk::style element create %s %s %s %s" % ( + name, etype, spec, opts)) + + return '\n'.join(script) + +def _dict_from_tcltuple(ttuple, cut_minus=True): + """Break tuple in pairs, format it properly, then build the return + dict. If cut_minus is True, the supposed '-' prefixing options will + be removed. + + ttuple is expected to contain an even number of elements.""" + opts = [] + opt_start = 1 if cut_minus else 0 + + for opt, val in zip(iter(ttuple[::2]), iter(ttuple[1::2])): + if isinstance(val, basestring): + try: + if ' ' in val: # could be the padding option + val = map(int, val.split()) + elif val.isdigit(): + val = int(val) + except ValueError: # leave val untouched for now + pass + + elif val and hasattr(val, '__len__') and hasattr(val[0], 'typename'): + # could be a statespec + val = _list_from_statespec(val) + + elif hasattr(val, 'typename'): + # some other Tcl object + val = str(val) + + opts.append((str(opt)[opt_start:], val)) + + return dict(opts) + +def _list_from_statespec(stuple): + """Construct a list from the given statespec tuple according to the + accepted statespec accepted by _format_mapdict.""" + split_it = lambda ob: getattr(ob, 'typename', None) == 'StateSpec' + val = [str(val).split() if split_it(val) else val for val in stuple] + return [_flatten(spec) for spec in zip(iter(val[::2]), iter(val[1::2]))] + +def _list_from_layouttuple(ltuple): + """Construct a list from the tuple returned by ttk::layout, this is + somewhat the reverse of _format_layoutlist.""" + res = [] + + indx = 0 + while indx < len(ltuple): + name = ltuple[indx] + opts = {} + res.append((name, opts)) + indx += 1 + + while indx < len(ltuple): # grab name's options + opt, val = ltuple[indx:indx + 2] + if not opt.startswith('-'): # found next name + break + + opt = opt[1:] # remove the '-' from the option + indx += 2 + + if opt == 'children': + val = _list_from_layouttuple(val) + + opts[opt] = val + + return res + +def _val_or_dict(options, func, *args): + """Format options then call func with args and options and return + the appropriate result. + + If no option is specified, a dict is returned. If a option is + specified with the None value, the value for that option is returned. + Otherwise, the function just sets the passed options and the caller + shouldn't be expecting a return value anyway.""" + options = _format_optdict(options) + res = func(*(args + options)) + + if len(options) % 2: # option specified without a value, return its value + return res + + return _dict_from_tcltuple(res) + + +class Style(object): + """Manipulate style database.""" + + _name = "ttk::style" + + def __init__(self, master=None): + if not master: + if not Tkinter._default_root: + Tkinter._default_root = Tkinter.Tk() + master = Tkinter._default_root + + self.tk = master.tk + + + def configure(self, style, query_opt=None, **kw): + """Query or sets the default value of the specified option(s) in + style. + + Each key in kw is an option and each value is either a string or + a sequence identifying the value for that option.""" + if query_opt: + query_opt = ('' if query_opt.startswith('-') else '-') + query_opt + return self.tk.call(self._name, "configure", style, query_opt) + + return _dict_from_tcltuple(self.tk.call(self._name, "configure", style, + *(_format_optdict(kw)))) + + + def map(self, style, query_opt=None, **kw): + """Query or sets dynamic values of the specified option(s) in + style. + + Each key in kw is an option and each value should be a list or a + tuple (usually) containing statespecs grouped in tuples, or list, + or something else of your preference. A statespec is compound of + one or more states and then a value.""" + if query_opt: + query_opt = ('' if query_opt.startswith('-') else '-') + query_opt + return _list_from_statespec(self.tk.call(self._name, "map", style, + query_opt)) + + return _dict_from_tcltuple(self.tk.call(self._name, "map", style, + *(_format_mapdict(kw)))) + + + def lookup(self, style, option, state=None, default=None): + """Returns the value specified for option in style. + + If state is specified it is expected to be a sequence of one + or more states. If the default argument is set, it is used as + a fallback value in case no specification for option is found.""" + state = ' '.join(state) if state else '' + option = ('-' if not option.startswith('-') else '') + option + + return self.tk.call(self._name, "lookup", style, option, state, default) + + + def layout(self, style, layoutspec=None): + """Define the widget layout for given style. If layoutspec is + omitted, return the layout specification for given style. + + layoutspec is expected to be a list or an object different than + None that evaluates to False if you want to "turn off" that style. + If it is a list (or tuple, or something else), each item should be + a tuple where the first item is the layout name and the second item + should have the format described below: + + LAYOUTS + + A layout can contain the value None, if takes no options, or + a dict of options specifying how to arrange the element. + The layout mechanism uses a simplified version of the pack + geometry manager: given an initial cavity, each element is + allocated a parcel. Valid options/values are: + + side: whichside + Specifies which side of the cavity to place the + element; one of top, right, bottom or left. If + omitted, the element occupies the entire cavity. + + sticky: nswe + Specifies where the element is placed inside its + allocated parcel. + + children: [sublayout... ] + Specifies a list of elements to place inside the + element. Each element is a tuple (or other sequence) + where the first item is the layout name, and the other + is a LAYOUT.""" + lspec = None + if layoutspec: + lspec = _format_layoutlist(layoutspec)[0] + elif layoutspec is not None: # will disable the layout ({}, '', etc) + lspec = "null" # could be any other word, but this may make sense + # when calling layout(style) later + + return _list_from_layouttuple(self.tk.call(self._name, "layout", style, + lspec)) + + + def element_create(self, elementname, etype, *args, **kw): + """Create a new element in the current theme of given etype.""" + spec, opts = _format_elemcreate(etype, False, *args, **kw) + self.tk.call(self._name, "element", "create", elementname, etype, + spec, *opts) + + + def element_names(self): + """Returns the list of elements defined in the current theme.""" + return self.tk.call(self._name, "element", "names") + + + def element_options(self, elementname): + """Return the list of elementname's options.""" + return self.tk.call(self._name, "element", "options", elementname) + + + def theme_create(self, themename, parent=None, settings=None): + """Creates a new theme. + + It is an error if themename already exists. If parent is + specified, the new theme will inherit styles, elements and + layouts from the specified parent theme. If settings are present, + they are expected to have the same syntax used for theme_settings.""" + script = _script_from_settings(settings) if settings else '' + + if parent: + self.tk.call(self._name, "theme", "create", themename, + "-parent", parent, "-settings", script) + else: + self.tk.call(self._name, "theme", "create", themename, + "-settings", script) + + + def theme_settings(self, themename, settings): + """Temporarily sets the current theme to themename, apply specified + settings and then restore the previous theme. + + Each key in settings is a style and each value may contain the + keys 'configure', 'map', 'layout' and 'element create' and they + are expected to have the same format as specified by the methods + configure, map, layout and element_create respectively.""" + script = _script_from_settings(settings) + self.tk.call(self._name, "theme", "settings", themename, script) + + + def theme_names(self): + """Returns a list of all known themes.""" + return self.tk.call(self._name, "theme", "names") + + + def theme_use(self, themename): + """Sets the current theme to themename and refreshes all widgets.""" + self.tk.call(self._name, "theme", "use", themename) + + +class Widget(Tkinter.Widget): + """Base class for Tk themed widgets.""" + + def __init__(self, master, widgetname, kw=None): + """Constructs a Ttk Widget with the parent master. + + STANDARD OPTIONS + + class, cursor, takefocus, style + + SCROLLABLE WIDGET OPTIONS + + xscrollcommand, yscrollcommand + + LABEL WIDGET OPTIONS + + text, textvariable, underline, image, compound, width + + WIDGET STATES + + active, disabled, focus, pressed, selected, background, + readonly, alternate, invalid + """ + Tkinter.Widget.__init__(self, master, widgetname, kw=kw) + + + def identify(self, x, y): + """Returns the name of the element at position x, y, or the empty + string if the point does not lie within any element. + + x and y are pixel coordinates relative to the widget.""" + return self.tk.call(self._w, "identify", x, y) + + + def instate(self, statespec, callback=None, *args, **kw): + """Test the widget's state. + + If callback is not specified, returns 1 if the widget state + matches statespec and 0 otherwise. If callback is specified, + then it will be invoked with *args, **kw if the widget state + matches statespec. statespec is expected to be a sequence.""" + ret = self.tk.call(self._w, "instate", ' '.join(statespec)) + if ret and callback: + return callback(*args, **kw) + + return ret + + + def state(self, statespec=None): + """Modify or inquire widget state. + + Widget state is returned if statespec is None, otherwise it is + set according to the statespec flags and then a new state spec + is returned indicating which flags were changed. statespec is + expected to be a sequence.""" + statespec = statespec or '' + + return self.tk.splitlist( + str(self.tk.call(self._w, "state", ' '.join(statespec)))) + + +class Button(Widget): + """Ttk Button widget, displays a textual label and/or image, and + evaluates a command when pressed.""" + + def __init__(self, master=None, **kw): + """Construct a Ttk Button widget with the parent master. + + STANDARD OPTIONS + + class, compound, cursor, image, state, style, takefocus, + text, textvariable, underline, width + + WIDGET-SPECIFIC OPTIONS + + command, default, width + """ + Widget.__init__(self, master, "ttk::button", kw) + + + def invoke(self): + """Invokes the command associated with the button.""" + return self.tk.call(self._w, "invoke") + + +class Checkbutton(Widget): + """Ttk Checkbutton widget which is either in on- or off-state.""" + + def __init__(self, master=None, **kw): + """Construct a Ttk Checkbutton widget with the parent master. + + STANDARD OPTIONS + + class, compound, cursor, image, state, style, takefocus, + text, textvariable, underline, width + + WIDGET-SPECIFIC OPTIONS + + command, offvalue, onvalue, variable + """ + Widget.__init__(self, master, "ttk::checkbutton", kw) + + + def invoke(self): + """Toggles between the selected and deselected states and + invokes the associated command. If the widget is currently + selected, sets the option variable to the offvalue option + and deselects the widget; otherwise, sets the option variable + to the option onvalue. + + Returns the result of the associated command.""" + return self.tk.call(self._w, "invoke") + + +class Entry(Widget, Tkinter.Entry): + """Ttk Entry widget displays a one-line text string and allows that + string to be edited by the user.""" + + def __init__(self, master=None, widget=None, **kw): + """Constructs a Ttk Entry widget with the parent master. + + STANDARD OPTIONS + + class, cursor, style, takefocus, xscrollcommand + + WIDGET-SPECIFIC OPTIONS + + exportselection, invalidcommand, justify, show, state, + textvariable, validate, validatecommand, width + + VALIDATION MODES + + none, key, focus, focusin, focusout, all + """ + Widget.__init__(self, master, widget or "ttk::entry", kw) + + + def bbox(self, index): + """Return a tuple of (x, y, width, height) which describes the + bounding box of the character given by index.""" + return self.tk.call(self._w, "bbox", index) + + + def identify(self, x, y): + """Returns the name of the element at position x, y, or the + empty string if the coordinates are outside the window.""" + return self.tk.call(self._w, "identify", x, y) + + + def validate(self): + """Force revalidation, independent of the conditions specified + by the validate option. Returns 0 if validation fails, 1 if it + succeeds. Sets or clears the invalid state accordingly.""" + return self.tk.call(self._w, "validate") + + +class Combobox(Entry): + """Ttk Combobox widget combines a text field with a pop-down list of + values.""" + + def __init__(self, master=None, **kw): + """Construct a Ttk Combobox widget with the parent master. + + STANDARD OPTIONS + + class, cursor, style, takefocus + + WIDGET-SPECIFIC OPTIONS + + exportselection, justify, height, postcommand, state, + textvariable, values, width + """ + Entry.__init__(self, master, "ttk::combobox", **kw) + + + def current(self, newindex=None): + """If newindex is supplied, sets the combobox value to the + element at position newindex in the list of values. Otherwise, + returns the index of the current value in the list of values + or -1 if the current value does not appear in the list.""" + return self.tk.call(self._w, "current", newindex) + + + def set(self, value): + """Sets the value of the combobox to value.""" + self.tk.call(self._w, "set", value) + + +class Frame(Widget): + """Ttk Frame widget is a container, used to group other widgets + together.""" + + def __init__(self, master=None, **kw): + """Construct a Ttk Frame with parent master. + + STANDARD OPTIONS + + class, cursor, style, takefocus + + WIDGET-SPECIFIC OPTIONS + + borderwidth, relief, padding, width, height + """ + Widget.__init__(self, master, "ttk::frame", kw) + + +class Label(Widget): + """Ttk Label widget displays a textual label and/or image.""" + + def __init__(self, master=None, **kw): + """Construct a Ttk Label with parent master. + + STANDARD OPTIONS + + class, compound, cursor, image, style, takefocus, text, + textvariable, underline, width + + WIDGET-SPECIFIC OPTIONS + + anchor, background, font, foreground, justify, padding, + relief, text, wraplength + """ + Widget.__init__(self, master, "ttk::label", kw) + + +class Labelframe(Widget): + """Ttk Labelframe widget is a container used to group other widgets + together. It has an optional label, which may be a plain text string + or another widget.""" + + def __init__(self, master=None, **kw): + """Construct a Ttk Labelframe with parent master. + + STANDARD OPTIONS + + class, cursor, style, takefocus + + WIDGET-SPECIFIC OPTIONS + labelanchor, text, underline, padding, labelwidget, width, + height + """ + Widget.__init__(self, master, "ttk::labelframe", kw) + +LabelFrame = Labelframe # Tkinter name compatibility + + +class Menubutton(Widget): + """Ttk Menubutton widget displays a textual label and/or image, and + displays a menu when pressed.""" + + def __init__(self, master=None, **kw): + """Construct a Ttk Menubutton with parent master. + + STANDARD OPTIONS + + class, compound, cursor, image, state, style, takefocus, + text, textvariable, underline, width + + WIDGET-SPECIFIC OPTIONS + + direction, menu + """ + Widget.__init__(self, master, "ttk::menubutton", kw) + + +class Notebook(Widget): + """Ttk Notebook widget manages a collection of windows and displays + a single one at a time. Each child window is associated with a tab, + which the user may select to change the currently-displayed window.""" + + def __init__(self, master=None, **kw): + """Construct a Ttk Notebook with parent master. + + STANDARD OPTIONS + + class, cursor, style, takefocus + + WIDGET-SPECIFIC OPTIONS + + height, padding, width + + TAB OPTIONS + + state, sticky, padding, text, image, compound, underline + + TAB IDENTIFIERS (tab_id) + + The tab_id argument found in several methods may take any of + the following forms: + + * An integer between zero and the number of tabs + * The name of a child window + * A positional specification of the form "@x,y", which + defines the tab + * The string "current", which identifies the + currently-selected tab + * The string "end", which returns the number of tabs (only + valid for method index) + """ + Widget.__init__(self, master, "ttk::notebook", kw) + + + def add(self, child, **kw): + """Adds a new tab to the notebook. + + If window is currently managed by the notebook but hidden, it is + restored to its previous position.""" + self.tk.call(self._w, "add", child, *(_format_optdict(kw))) + + + def forget(self, tab_id): + """Removes the tab specified by tab_id, unmaps and unmanages the + associated window.""" + self.tk.call(self._w, "forget", tab_id) + + + def hide(self, tab_id): + """Hides the tab specified by tab_id. + + The tab will not be displayed, but the associated window remains + managed by the notebook and its configuration remembered. Hidden + tabs may be restored with the add command.""" + self.tk.call(self._w, "hide", tab_id) + + + def identify(self, x, y): + """Returns the name of the tab element at position x, y, or the + empty string if none.""" + return self.tk.call(self._w, "identify", x, y) + + + def index(self, tab_id): + """Returns the numeric index of the tab specified by tab_id, or + the total number of tabs if tab_id is the string "end".""" + return self.tk.call(self._w, "index", tab_id) + + + def insert(self, pos, child, **kw): + """Inserts a pane at the specified position. + + pos is either the string end, an integer index, or the name of + a managed child. If child is already managed by the notebook, + moves it to the specified position.""" + self.tk.call(self._w, "insert", pos, child, *(_format_optdict(kw))) + + + def select(self, tab_id=None): + """Selects the specified tab. + + The associated child window will be displayed, and the + previously-selected window (if different) is unmapped. If tab_id + is omitted, returns the widget name of the currently selected + pane.""" + return self.tk.call(self._w, "select", tab_id) + + + def tab(self, tab_id, **kw): + """Query or modify the options of the specific tab. + + If no option is specified, returns a dictionary of the tab option + values. If one option is specified, returns the value of that + option. Otherwise, sets the options to the corresponding values.""" + return _val_or_dict(kw, self.tk.call, self._w, "tab", tab_id) + + + def tabs(self): + """Returns a list of windows managed by the notebook.""" + return self.tk.call(self._w, "tabs") + + + def enable_traversal(self): + """Enable keyboard traversal for a toplevel window containing + this notebook. + + This will extend the bindings for the toplevel window containing + this notebook as follows: + + Control-Tab: selects the tab following the currently selected + one + + Shift-Control-Tab: selects the tab preceding the currently + selected one + + Alt-K: where K is the mnemonic (underlined) character of any + tab, will select that tab. + + Multiple notebooks in a single toplevel may be enabled for + traversal, including nested notebooks. However, notebook traversal + only works properly if all panes are direct children of the + notebook.""" + # The only, and good, difference I see is about mnemonics, which works + # after calling this method. Control-Tab and Shift-Control-Tab always + # works (here at least). + self.tk.call("ttk::notebook::enableTraversal", self._w) + + +class Panedwindow(Widget, Tkinter.PanedWindow): + """Ttk Panedwindow widget displays a number of subwindows, stacked + either vertically or horizontally.""" + + def __init__(self, master=None, **kw): + """Construct a Ttk Panedwindow with parent master. + + STANDARD OPTIONS + + class, cursor, style, takefocus + + WIDGET-SPECIFIC OPTIONS + + orient, width, height + + PANE OPTIONS + + weight + """ + Widget.__init__(self, master, "ttk::panedwindow", kw) + + + def insert(self, pos, child, **kw): + """Inserts a pane at the specified positions. + + pos is either the string end, and integer index, or the name + of a child. If child is already managed by the paned window, + moves it to the specified position.""" + self.tk.call(self._w, "insert", pos, child, *(_format_optdict(kw))) + + + def pane(self, pane, **kw): + """Query or modify the options of the specified pane. + + pane is either an integer index or the name of a managed + subwindow. If no options are specified, returns a dict of the + pane option values, if an option is specified without a value, + returns the value of that option. Otherwise, the options are set.""" + return _val_or_dict(kw, self.tk.call, self._w, "pane", pane) + + + def sashpos(self, index, newpos=None): + """If newpos is specified, sets the position of sash number index. + + May adjust the positions of adjacent sashes to ensure that + positions are monotonically increasing. Sash positions are further + constrained to be between 0 and the total size of the widget. + + Returns the new position of sash number index.""" + return self.tk.call(self._w, "sashpos", index, newpos) + +PanedWindow = Panedwindow # Tkinter name compatibility + + +class Progressbar(Widget): + """Ttk Progressbar widget shows the status of a long-running + operation. They can operate in two modes: determinate mode shows the + amount completed relative to the total amount of work to be done, and + indeterminate mode provides an animated display to let the user know + that something is happening.""" + + def __init__(self, master=None, **kw): + """Construct a Ttk Progressbar with parent master. + + STANDARD OPTIONS + + class, cursor, style, takefocus + + WIDGET-SPECIFIC OPTIONS + + orient, length, mode, maximum, value, variable, phase + """ + Widget.__init__(self, master, "ttk::progressbar", kw) + + + def start(self, interval=None): + """Begin autoincrement mode: schedules a recurring timer event + that calls method step every interval milliseconds. + + interval defaults to 50 milliseconds (20 steps/second) if ommited.""" + self.tk.call(self._w, "start", interval) + + + def step(self, amount=None): + """Increments the value option by amount. + + amount defaults to 1.0 if omitted.""" + self.tk.call(self._w, "step", amount) + + + def stop(self): + """Stop autoincrement mode: cancels any recurring timer event + initiated by start.""" + self.tk.call(self._w, "stop") + + +class Radiobutton(Widget): + """Ttk Radiobutton widgets are used in groups to show or change a + set of mutually-exclusive options.""" + + def __init__(self, master=None, **kw): + """Construct a Ttk Radiobutton with parent master. + + STANDARD OPTIONS + + class, compound, cursor, image, state, style, takefocus, + text, textvariable, underline, width + + WIDGET-SPECIFIC OPTIONS + + command, value, variable + """ + Widget.__init__(self, master, "ttk::radiobutton", kw) + + + def invoke(self): + """Sets the option variable to the option value, selects the + widget, and invokes the associated command. + + Returns the result of the command, or an empty string if + no command is specified.""" + return self.tk.call(self._w, "invoke") + + +class Scale(Widget, Tkinter.Scale): + """Ttk Scale widget is typically used to control the numeric value of + a linked variable that varies uniformly over some range.""" + + def __init__(self, master=None, **kw): + """Construct a Ttk Scale with parent master. + + STANDARD OPTIONS + + class, cursor, style, takefocus + + WIDGET-SPECIFIC OPTIONS + + command, from, length, orient, to, value, variable + """ + Widget.__init__(self, master, "ttk::scale", kw) + + + def get(self, x=None, y=None): + """Get the current value of the value option, or the value + corresponding to the coordinates x, y if they are specified. + + x and y are pixel coordinates relative to the scale widget + origin.""" + return self.tk.call(self._w, 'get', x, y) + + +class Scrollbar(Widget, Tkinter.Scrollbar): + """Ttk Scrollbar controls the viewport of a scrollable widget.""" + + def __init__(self, master=None, **kw): + """Construct a Ttk Scrollbar with parent master. + + STANDARD OPTIONS + + class, cursor, style, takefocus + + WIDGET-SPECIFIC OPTIONS + + command, orient + """ + Widget.__init__(self, master, "ttk::scrollbar", kw) + + +class Separator(Widget): + """Ttk Separator widget displays a horizontal or vertical separator + bar.""" + + def __init__(self, master=None, **kw): + """Construct a Ttk Separator with parent master. + + STANDARD OPTIONS + + class, cursor, style, takefocus + + WIDGET-SPECIFIC OPTIONS + + orient + """ + Widget.__init__(self, master, "ttk::separator", kw) + + +class Sizegrip(Widget): + """Ttk Sizegrip allows the user to resize the containing toplevel + window by pressing and dragging the grip.""" + + def __init__(self, master=None, **kw): + """Construct a Ttk Sizegrip with parent master. + + STANDARD OPTIONS + + class, cursor, state, style, takefocus + """ + Widget.__init__(self, master, "ttk::sizegrip", kw) + + +class Treeview(Widget): + """Ttk Treeview widget displays a hierarchical collection of items. + + Each item has a textual label, an optional image, and an optional list + of data values. The data values are displayed in successive columns + after the tree label.""" + + def __init__(self, master=None, **kw): + """Construct a Ttk Treeview with parent master. + + STANDARD OPTIONS + + class, cursor, style, takefocus, xscrollcommand, + yscrollcommand + + WIDGET-SPECIFIC OPTIONS + + columns, displaycolumns, height, padding, selectmode, show + + ITEM OPTIONS + + text, image, values, open, tags + + TAG OPTIONS + + foreground, background, font, image + """ + Widget.__init__(self, master, "ttk::treeview", kw) + + + def bbox(self, item, column=None): + """Returns the bounding box (relative to the treeview widget's + window) of the specified item in the form x y width height. + + If column is specified, returns the bounding box of that cell. + If the item is not visible (i.e., if it is a descendant of a + closed item or is scrolled offscreen), returns an empty string.""" + return self.tk.call(self._w, "bbox", item, column) + + + def get_children(self, item): + """Returns a tuple of children belonging to item.""" + return self.tk.call(self._w, "children", item) + + + def set_children(self, item, newchildren): + """Replaces item's child with newchildren. + + Children present in item that are not present in newchildren + are detached from tree. No items in newchildren may be an + ancestor of item.""" + self.tk.call(self._w, "children", item, newchildren) + + + def column(self, column, **kw): + """Query or modify the options for the specified column. + + If no options are specified, returns a dictionary of option/value + pairs. If an option is specified with value None, returns the + value of that option. Otherwise, the options are updated with the + specified values.""" + return _val_or_dict(kw, self.tk.call, self._w, "column", column) + + + def delete(self, items): + """Delete all specified items and all their descendants. The root + item may not be deleted.""" + self.tk.call(self._w, "delete", items) + + + def detach(self, items): + """Unlinks all of the specified items from the tree. + + The items and all of their descendants are still present, and may + be reinserted at another point in the tree, but will not be + displayed. The root item may not be detached.""" + self.tk.call(self._w, "detach", items) + + + def exists(self, item): + """Returns 1 if the specified item is present in the three, + 0 otherwise.""" + return self.tk.call(self._w, "exists", item) + + + def focus(self, item=None): + """If item is specified, sets the focus item to item. Otherwise, + returns the current focus item, or '' if there is none.""" + return self.tk.call(self._w, "focus", item) + + + def heading(self, column, **kw): + """Query or modify the heading options for the specified column. + + Valid options/values are: + text: text + The text to display in the column heading + image: image_name + Specifies an image to display to the right of the column + heading + anchor: anchor + Specifies how the heading text should be aligned. One of + the standard Tk anchor values + command: callback + A callback to be invoked when the heading label is + pressed. + + To configure the tree column heading, call this with column = "#0" """ + cmd = kw.get('command') + if cmd and not isinstance(cmd, basestring): + # callback not registered yet, do it now + kw['command'] = self.master.register(cmd, self._substitute) + + return _val_or_dict(kw, self.tk.call, self._w, 'heading', column) + + + def identify(self, component, x, y): + """Returns a description of the specified component under the + point given by x and y, or the empty string if no such component + is present at that position.""" + return self.tk.call(self._w, "identify", component, x, y) + + + def identify_row(self, y): + """Returns the item ID of the item at position y.""" + return self.identify("row", 0, y) + + + def identify_column(self, x): + """Returns the data column identifier of the cell at position x. + + The tree column has ID #0.""" + return self.identify("column", x, 0) + + + def index(self, item): + """Returns the integer index of item within its parent's list + of children.""" + return self.tk.call(self._w, "index", item) + + + def insert(self, parent, index, iid=None, **kw): + """Creates a new item and return the item identifier of the newly + created item. + + parent is the item ID of the parent item, or the empty string + to create a new top-level item. index is an integer, or the value + end, specifying where in the list of parent's children to insert + the new item. If index is less than or equal to zero, the new node + is inserted at the beginning, if index is greater than or equal to + the current number of children, it is inserted at the end. If iid + is specified, it is used as the item identifier, iid must not + already exist in the tree. Otherwise, a new unique identifier + is generated.""" + opts, values = _format_optdict(kw, ignore='values'), kw.get('values') + # values may need special formatting if any value contains a space + if values: + values = map(unicode, values) + opts += ("-values", + ' '.join(('{%s}' if ' ' in v else '%s') % v for v in values)) + if iid: + res = self.tk.call(self._w, "insert", parent, index, "-id", iid, + *opts) + else: + res = self.tk.call(self._w, "insert", parent, index, *opts) + + return res + + + def item(self, item, **kw): + """Query or modify the options for the specified item. + + If no options are specified, returns a dictionary of option/value + pairs. If a single option is specified with value None, returns + the value of that option. Otherwise, the item's options are + updated with the specified values.""" + return _val_or_dict(kw, self.tk.call, self._w, "item", item) + + + def move(self, item, parent, index): + """Moves item to position index in parent's list of children. + + It is illegal to move an item under one of its descendants. If + index is less than or equal to zero, item is moved to the + beginning, if greater than or equal to the number of children, + it is moved to the end. If item was detached it is reattached.""" + self.tk.call(self._w, "move", item, parent, index) + + reattach = move # A sensible method name for reattaching detached items + + + def next(self, item): + """Returns the identifier of item's next sibling, or '' if item + is the last child of its parent.""" + return self.tk.call(self._w, "next", item) + + + def parent(self, item): + """Returns the ID of the parent of item, or '' if item is at the + top level of the hierarchy.""" + return self.tk.call(self._w, "parent", item) + + + def prev(self, item): + """Returns the identifier of item's previous sibling, or '' if + item is the first child of its parent.""" + return self.tk.call(self._w, "prev", item) + + + def see(self, item): + """Ensure that item is visible. + + Sets all of item's ancestors open option to True, and scrolls + the widget if necessary so that item is within the visible + portion of the tree.""" + self.tk.call(self._w, "see", item) + + + def selection(self, selop=None, items=None): + """If selop is not specified, returns selected items.""" + return self.tk.call(self._w, "selection", selop, items) + + + def selection_set(self, items): + """items becomes the new selection.""" + self.selection("set", items) + + + def selection_add(self, items): + """Add items to the selection.""" + self.selection("add", items) + + + def selection_remove(self, items): + """Remove items from the selection.""" + self.selection("remove", items) + + + def selection_toggle(self, items): + """Toggle the selection state of each item in items.""" + self.selection("toggle", items) + + + def set(self, item, column=None, value=None): + """With one argument, returns a dictionary of column/value pairs + for the specified item. With two arguments, returns the current + value of the specified column. With three arguments, sets the + value of given column in given item to the specified value.""" + res = self.tk.call(self._w, "set", item, column, value) + if column is None and value is None: + return _dict_from_tcltuple(res, False) + else: + return res + + + def tag_bind(self, tagname, sequence=None, callback=None): + """Bind a callback for the given event sequence to the tag tagname. + When an event is delivered to an item, the callbacks for each + of the item's tags option are called.""" + self._bind((self._w, "tag", "bind", tagname), sequence, callback, add=0) + + + def tag_configure(self, tagname, **kw): + """Query or modify the options for the specified tagname. + + If one or more option/value pairs are specified, sets the value + of those options for the specified tag. If a single option + is specified with value None, returns the value of that option + (or the empty string if the option has not been specified for + tagname). With no additional arguments, returns a dictionary of + the option settings for tagname.""" + return _val_or_dict(kw, self.tk.call, self._w, "tag", "configure", + tagname) + + + def xview(self, *args): + """Query or modify horizontal position of the treeview.""" + return self.tk.call(self._w, "xview", *args) + + + def yview(self, *args): + """Query or modify vertical position of the treeview.""" + return self.tk.call(self._w, "yview", *args) + + +# Extensions + +class LabeledScale(Frame, object): + """A Ttk Scale widget with a Ttk Label widget over it indicating its + current value. + + The Ttk Scale can be accessed through instance.scale, and Ttk Label + can be accessed through instance.label""" + + def __init__(self, master=None, variable=None, from_=0, to=10, **kw): + """Construct a LabeledScale with parent master, a variable to be + associated with the Ttk Scale widget and its range. If variable is + not specified, a Tkinter.IntVar is created. + + WIDGET-SPECIFIC OPTIONS + + compound: 'top' or 'bottom' + Specifies how to display the label relative to the scale. + Defaults to 'top'. + """ + self._label_top = kw.pop('compound', 'top') == 'top' + Frame.__init__(self, master, **kw) + self._variable = variable or Tkinter.IntVar(master, value=from_) + + self.label = Label(self) + self.scale = Scale(self, variable=self._variable, from_=from_, to=to) + + # position scale and label according to the compound option + scale_side = 'bottom' if self._label_top else 'top' + label_side = 'top' if scale_side == 'bottom' else 'bottom' + self.scale.pack(side=scale_side, fill='x') + tmp = Label(self).pack(side=label_side) # place holder + self.label.place(anchor='n' if label_side == 'top' else 's') + + # update the label as scale or variable changes + self._variable.trace_variable('w', self._adjust) + self.scale.bind('', self._adjust) + self.scale.bind('', self._adjust) + + + def _adjust(self, *args): + """Adjust the label position according to the scale.""" + newval = self._variable.get() + if not self.scale['from'] <= newval <= self.scale['to']: + return + + self.update() + self.label['text'] = newval + + x, y = self.scale.coords() + if self._label_top: + y = self.scale.winfo_y() - self.label.winfo_reqheight() + else: + y = self.scale.winfo_reqheight() + self.label.winfo_reqheight() + x = x + self.scale.winfo_x() + + self.label.place_configure(x=x, y=y) + + + def _get_value(self): + """Return current scale value.""" + return self._variable.get() + + + def _set_value(self, val): + """Set new scale value.""" + self._variable.set(val) + + + value = property(_get_value, _set_value) + + +class OptionMenu(Menubutton): + """Themed OptionMenu which allows the user to select a value from a + menu.""" + + def __init__(self, master, variable, default=None, *values, **kwargs): + """Construct a themed OptionMenu widget with the parent master, + the resource textvariable set to variable, the initially selected + value specified by the default parameter, the other menu values + given by *values and an additional keyword argument command.""" + kw = {'textvariable': variable, 'style': kwargs.pop('style', None), + 'direction': kwargs.pop('direction', None)} + Menubutton.__init__(self, master, **kw) + + self._menu = Tkinter.Menu(self, name="menu", tearoff=0) + self._callback = kwargs.pop('command', None) + self._variable = variable + if kwargs: + raise Tkinter.TclError('unknown option -%s' % ( + kwargs.iterkeys().next())) + + self['menu'] = self._menu + self.set_menu(default, *((default, ) + values)) + + + def __getitem__(self, item): + if item == 'menu': + return self._menu + + return Menubutton.__getitem__(self, item) + + + def set_menu(self, default=None, *values): + """Build a new menu of radiobuttons with *values and optionally + a default value.""" + menu = self['menu'] + menu.delete(0, 'end') + for val in values: + menu.add_radiobutton(label=val, + command=Tkinter._setit(self._variable, val, self._callback)) + + if default: + self._variable.set(default) + + + def destroy(self): + """Destroy this widget and the associated menu.""" + Menubutton.destroy(self) + self._menu = None diff --git a/pysollib/tile/wizarddialog.py b/pysollib/tile/wizarddialog.py index 67997407..a17b6ac2 100644 --- a/pysollib/tile/wizarddialog.py +++ b/pysollib/tile/wizarddialog.py @@ -24,7 +24,7 @@ __all__ = ['WizardDialog'] # imports import Tkinter -import Tile +import ttk # PySol imports from pysollib.mfxutil import KwStruct @@ -47,22 +47,22 @@ class WizardDialog(MfxDialog): top_frame, bottom_frame = self.createFrames(kw) self.createBitmaps(top_frame, kw) - frame = Tile.Frame(top_frame) + frame = ttk.Frame(top_frame) frame.pack(expand=True, fill='both', padx=10, pady=10) frame.columnconfigure(0, weight=1) - notebook = Tile.Notebook(frame) + notebook = ttk.Notebook(frame) notebook.pack(expand=True, fill='both') for w in WizardWidgets: if isinstance(w, basestring): - frame = Tile.Frame(notebook) + frame = ttk.Frame(notebook) notebook.add(frame, text=w, sticky='nsew', padding=5) frame.columnconfigure(1, weight=1) row = 0 continue - Tile.Label(frame, text=w.label).grid(row=row, column=0) + ttk.Label(frame, text=w.label).grid(row=row, column=0) if w.widget == 'preset': if w.variable is None: @@ -82,7 +82,7 @@ class WizardDialog(MfxDialog): elif w.widget == 'entry': if w.variable is None: w.variable = Tkinter.StringVar() - en = Tile.Entry(frame, textvariable=w.variable) + en = ttk.Entry(frame, textvariable=w.variable) en.grid(row=row, column=1, sticky='ew', padx=2, pady=2) elif w.widget == 'menu': if w.variable is None: @@ -109,8 +109,8 @@ class WizardDialog(MfxDialog): elif w.widget == 'check': if w.variable is None: w.variable = Tkinter.BooleanVar() - ch = Tile.Checkbutton(frame, variable=w.variable, - takefocus=False) + ch = ttk.Checkbutton(frame, variable=w.variable, + takefocus=False) ch.grid(row=row, column=1, sticky='ew', padx=2, pady=2) if w.current_value is None: diff --git a/pysollib/winsystems/aqua.py b/pysollib/winsystems/aqua.py index d4eb1dbb..bc9aadd8 100644 --- a/pysollib/winsystems/aqua.py +++ b/pysollib/winsystems/aqua.py @@ -23,7 +23,7 @@ import sys, os import Tkinter from pysollib.settings import TOOLKIT, USE_TILE -from pysollib.tile import Tile +from pysollib.tile import ttk from pysollib.macosx.appSupport import hideTkConsole from common import base_init_root_window, BaseTkSettings @@ -36,14 +36,14 @@ def init_root_window(root, app): if TOOLKIT == 'gtk': pass elif USE_TILE: - style = Tile.Style(root) + style = ttk.Style(root) color = style.lookup('.', 'background') if color: - root.tk_setPalette(color) # for non-Tile widgets + root.tk_setPalette(color) # for non-ttk widgets if app.opt.tile_theme == 'aqua': - # standard Tk scrollbars work on OS X, but Tile ones look weird - Tile.Scrollbar = Tkinter.Scrollbar + # standard Tk scrollbars work on OS X, but ttk ones look weird + ttk.Scrollbar = Tkinter.Scrollbar else: # pure Tk #root.option_add(...) diff --git a/pysollib/winsystems/common.py b/pysollib/winsystems/common.py index 5556afae..5c38f34f 100644 --- a/pysollib/winsystems/common.py +++ b/pysollib/winsystems/common.py @@ -26,11 +26,10 @@ from pysollib.settings import VERSION from pysollib.settings import TOOLKIT, USE_TILE from pysollib.settings import DEBUG from pysollib.mfxutil import print_err -from pysollib.tile import Tile +from pysollib.tile import ttk def init_tile(app, top): - Tile.initialize(top) # load available themes d = os.path.join(app.dataloader.dir, 'themes') if os.path.isdir(d): @@ -38,10 +37,7 @@ def init_tile(app, top): for t in os.listdir(d): if os.path.exists(os.path.join(d, t, 'pkgIndex.tcl')): try: - if Tile.TileVersion < '0.8': - top.tk.eval('package require tile::theme::'+t) - else: - top.tk.eval('package require ttk::theme::'+t) + top.tk.eval('package require ttk::theme::'+t) #print 'load theme:', t except: traceback.print_exc() @@ -50,17 +46,12 @@ def init_tile(app, top): def set_theme(app, top, theme): # set theme + style = ttk.Style(top) try: - Tile.setTheme(top, theme) + style.theme_use(theme) except: print_err(_('invalid theme name: ') + theme) - Tile.setTheme(top, app.opt.default_tile_theme) -## style = Tile.Style(top) -## #all_themes = style.theme_names() -## if theme not in all_themes: -## print_err(_('invalid theme name: ') + theme) -## theme = app.opt.default_tile_theme -## style.theme_use(theme) + style.theme_use(app.opt.default_tile_theme) def get_font_name(font): diff --git a/pysollib/winsystems/win32.py b/pysollib/winsystems/win32.py index 2cc8fa36..2cc0ad54 100644 --- a/pysollib/winsystems/win32.py +++ b/pysollib/winsystems/win32.py @@ -22,7 +22,7 @@ import sys, os from pysollib.settings import TOOLKIT, USE_TILE -from pysollib.tile import Tile +from pysollib.tile import ttk from common import base_init_root_window, BaseTkSettings @@ -33,7 +33,7 @@ def init_root_window(root, app): pass elif USE_TILE: theme = app.opt.tile_theme - style = Tile.Style(root) + style = ttk.Style(root) if theme not in ('winnative', 'xpnative'): color = style.lookup('.', 'background') if color: diff --git a/pysollib/winsystems/x11.py b/pysollib/winsystems/x11.py index 96937e96..1d86dcf6 100644 --- a/pysollib/winsystems/x11.py +++ b/pysollib/winsystems/x11.py @@ -26,7 +26,7 @@ import tkFont from pysollib.settings import TITLE from pysollib.settings import TOOLKIT, USE_TILE -from pysollib.tile import Tile +from pysollib.tile import ttk from common import base_init_root_window, BaseTkSettings, get_font_name @@ -53,18 +53,14 @@ def init_root_window(root, app): root.tk.evalfile(f) except: traceback.print_exc() - f = 'clrpick8.4.tcl' - if Tile.TileVersion >= '0.8': - f = 'clrpick8.5.tcl' + f = 'clrpick8.5.tcl' f = os.path.join(app.dataloader.dir, 'tcl', f) if os.path.exists(f): try: root.tk.evalfile(f) except: traceback.print_exc() - f = 'fsdialog8.4.tcl' - if Tile.TileVersion >= '0.8': - f = 'fsdialog8.5.tcl' + f = 'fsdialog8.5.tcl' f = os.path.join(app.dataloader.dir, 'tcl', f) if os.path.exists(f): try: @@ -77,14 +73,14 @@ def init_root_window(root, app): tkFileDialog.SaveAs.command = 'ttk::getSaveFile' tkFileDialog.Directory.command = 'ttk::chooseDirectory' - style = Tile.Style(root) + style = ttk.Style(root) color = style.lookup('.', 'background') if color: root.tk_setPalette(color) root.option_add('*Menu.borderWidth', 1, 60) root.option_add('*Menu.activeBorderWidth', 1, 60) - color = style.lookup('.', 'background', 'active') + color = style.lookup('.', 'background', ['active']) if color: root.option_add('*Menu.activeBackground', color, 60) @@ -96,11 +92,11 @@ def init_root_window(root, app): root.option_add('*selectBackground', '#0a5f89', 60) root.option_add('*inactiveSelectBackground', '#0a5f89', 60) # Tk-8.5 - color = style.lookup('TEntry', 'selectbackground', 'focus') + color = style.lookup('TEntry', 'selectbackground', ['focus']) if color: root.option_add('*selectBackground', color, 60) root.option_add('*inactiveSelectBackground', color, 60) - color = style.lookup('TEntry', 'selectforeground', 'focus') + color = style.lookup('TEntry', 'selectforeground', ['focus']) if color: root.option_add('*selectForeground', color, 60) @@ -118,7 +114,7 @@ def init_root_window(root, app): f = root.tk.splitlist(root.tk.call('font', 'actual', fn)) root.tk.call('font', 'configure', 'TkHeadingFont', *f) else: - # use font from Tile settings + # use font from ttk settings font = style.lookup('.', 'font') if font: fn = get_font_name(font)