Current Path: > > opt > alt > python27 > lib64 > python2.7 > idlelib
Operation : Linux premium131.web-hosting.com 4.18.0-553.44.1.lve.el8.x86_64 #1 SMP Thu Mar 13 14:29:12 UTC 2025 x86_64 Software : Apache Server IP : 162.0.232.56 | Your IP: 216.73.216.111 Domains : 1034 Domain(s) Permission : [ 0755 ]
Name | Type | Size | Last Modified | Actions |
---|---|---|---|---|
Icons | Directory | - | - | |
idle_test | Directory | - | - | |
AutoComplete.py | File | 8964 bytes | January 08 2025 10:43:10. | |
AutoComplete.pyc | File | 8181 bytes | January 08 2025 10:43:10. | |
AutoComplete.pyo | File | 8181 bytes | January 08 2025 10:43:10. | |
AutoCompleteWindow.py | File | 17318 bytes | January 08 2025 10:43:09. | |
AutoCompleteWindow.pyc | File | 12714 bytes | January 08 2025 10:43:10. | |
AutoCompleteWindow.pyo | File | 12655 bytes | January 08 2025 10:43:11. | |
AutoExpand.py | File | 3395 bytes | January 08 2025 10:43:10. | |
AutoExpand.pyc | File | 3580 bytes | January 08 2025 10:43:11. | |
AutoExpand.pyo | File | 3580 bytes | January 08 2025 10:43:11. | |
Bindings.py | File | 2976 bytes | January 08 2025 10:43:09. | |
Bindings.pyc | File | 4708 bytes | January 08 2025 10:43:09. | |
Bindings.pyo | File | 4708 bytes | January 08 2025 10:43:09. | |
CREDITS.txt | File | 1866 bytes | January 08 2025 10:43:11. | |
CallTipWindow.py | File | 6065 bytes | January 08 2025 10:43:09. | |
CallTipWindow.pyc | File | 6292 bytes | January 08 2025 10:43:09. | |
CallTipWindow.pyo | File | 6292 bytes | January 08 2025 10:43:09. | |
CallTips.py | File | 7740 bytes | January 08 2025 10:43:09. | |
CallTips.pyc | File | 8354 bytes | January 08 2025 10:43:10. | |
CallTips.pyo | File | 8354 bytes | January 08 2025 10:43:10. | |
ChangeLog | File | 56393 bytes | January 08 2025 10:43:11. | |
ClassBrowser.py | File | 6999 bytes | January 08 2025 10:43:11. | |
ClassBrowser.pyc | File | 9890 bytes | January 08 2025 10:43:10. | |
ClassBrowser.pyo | File | 9890 bytes | January 08 2025 10:43:10. | |
CodeContext.py | File | 8342 bytes | January 08 2025 10:43:09. | |
CodeContext.pyc | File | 6788 bytes | January 08 2025 10:43:11. | |
CodeContext.pyo | File | 6744 bytes | January 08 2025 10:43:09. | |
ColorDelegator.py | File | 9762 bytes | January 08 2025 10:43:09. | |
ColorDelegator.pyc | File | 9119 bytes | January 08 2025 10:43:10. | |
ColorDelegator.pyo | File | 9119 bytes | January 08 2025 10:43:10. | |
Debugger.py | File | 18236 bytes | January 08 2025 10:43:10. | |
Debugger.pyc | File | 18144 bytes | January 08 2025 10:43:10. | |
Debugger.pyo | File | 18144 bytes | January 08 2025 10:43:10. | |
Delegator.py | File | 665 bytes | January 08 2025 10:43:10. | |
Delegator.pyc | File | 1346 bytes | January 08 2025 10:43:09. | |
Delegator.pyo | File | 1346 bytes | January 08 2025 10:43:09. | |
EditorWindow.py | File | 65498 bytes | January 08 2025 10:43:09. | |
EditorWindow.pyc | File | 58522 bytes | January 08 2025 10:43:09. | |
EditorWindow.pyo | File | 58420 bytes | January 08 2025 10:43:10. | |
FileList.py | File | 3718 bytes | January 08 2025 10:43:10. | |
FileList.pyc | File | 4168 bytes | January 08 2025 10:43:11. | |
FileList.pyo | File | 4135 bytes | January 08 2025 10:43:10. | |
FormatParagraph.py | File | 7287 bytes | January 08 2025 10:43:11. | |
FormatParagraph.pyc | File | 7311 bytes | January 08 2025 10:43:09. | |
FormatParagraph.pyo | File | 7311 bytes | January 08 2025 10:43:09. | |
GrepDialog.py | File | 5145 bytes | January 08 2025 10:43:11. | |
GrepDialog.pyc | File | 6607 bytes | January 08 2025 10:43:11. | |
GrepDialog.pyo | File | 6607 bytes | January 08 2025 10:43:11. | |
HISTORY.txt | File | 10317 bytes | January 08 2025 10:43:09. | |
HyperParser.py | File | 10497 bytes | January 08 2025 10:43:09. | |
HyperParser.pyc | File | 6803 bytes | January 08 2025 10:43:09. | |
HyperParser.pyo | File | 6803 bytes | January 08 2025 10:43:09. | |
IOBinding.py | File | 21911 bytes | January 08 2025 10:43:09. | |
IOBinding.pyc | File | 19068 bytes | January 08 2025 10:43:11. | |
IOBinding.pyo | File | 19068 bytes | January 08 2025 10:43:11. | |
IdleHistory.py | File | 4052 bytes | January 08 2025 10:43:11. | |
IdleHistory.pyc | File | 4151 bytes | January 08 2025 10:43:11. | |
IdleHistory.pyo | File | 4151 bytes | January 08 2025 10:43:11. | |
MultiCall.py | File | 17701 bytes | January 08 2025 10:43:10. | |
MultiCall.pyc | File | 16793 bytes | January 08 2025 10:43:09. | |
MultiCall.pyo | File | 16719 bytes | January 08 2025 10:43:09. | |
MultiStatusBar.py | File | 1348 bytes | January 08 2025 10:43:09. | |
MultiStatusBar.pyc | File | 2357 bytes | January 08 2025 10:43:11. | |
MultiStatusBar.pyo | File | 2357 bytes | January 08 2025 10:43:11. | |
NEWS.txt | File | 47247 bytes | January 08 2025 10:43:10. | |
ObjectBrowser.py | File | 4376 bytes | January 08 2025 10:43:10. | |
ObjectBrowser.pyc | File | 7431 bytes | January 08 2025 10:43:10. | |
ObjectBrowser.pyo | File | 7431 bytes | January 08 2025 10:43:10. | |
OutputWindow.py | File | 4576 bytes | January 08 2025 10:43:11. | |
OutputWindow.pyc | File | 5427 bytes | January 08 2025 10:43:10. | |
OutputWindow.pyo | File | 5427 bytes | January 08 2025 10:43:10. | |
ParenMatch.py | File | 6714 bytes | January 08 2025 10:43:10. | |
ParenMatch.pyc | File | 7350 bytes | January 08 2025 10:43:11. | |
ParenMatch.pyo | File | 7350 bytes | January 08 2025 10:43:11. | |
PathBrowser.py | File | 3009 bytes | January 08 2025 10:43:09. | |
PathBrowser.pyc | File | 4681 bytes | January 08 2025 10:43:11. | |
PathBrowser.pyo | File | 4681 bytes | January 08 2025 10:43:11. | |
Percolator.py | File | 3222 bytes | January 08 2025 10:43:11. | |
Percolator.pyc | File | 4803 bytes | January 08 2025 10:43:11. | |
Percolator.pyo | File | 4619 bytes | January 08 2025 10:43:09. | |
PyParse.py | File | 19510 bytes | January 08 2025 10:43:10. | |
PyParse.pyc | File | 10226 bytes | January 08 2025 10:43:09. | |
PyParse.pyo | File | 9788 bytes | January 08 2025 10:43:10. | |
PyShell.py | File | 58872 bytes | January 08 2025 10:43:09. | |
PyShell.pyc | File | 54269 bytes | January 08 2025 10:43:10. | |
PyShell.pyo | File | 54167 bytes | January 08 2025 10:43:10. | |
README.txt | File | 7890 bytes | January 08 2025 10:43:10. | |
RemoteDebugger.py | File | 11632 bytes | January 08 2025 10:43:09. | |
RemoteDebugger.pyc | File | 17131 bytes | January 08 2025 10:43:10. | |
RemoteDebugger.pyo | File | 16977 bytes | January 08 2025 10:43:09. | |
RemoteObjectBrowser.py | File | 942 bytes | January 08 2025 10:43:09. | |
RemoteObjectBrowser.pyc | File | 2280 bytes | January 08 2025 10:43:10. | |
RemoteObjectBrowser.pyo | File | 2280 bytes | January 08 2025 10:43:10. | |
ReplaceDialog.py | File | 6639 bytes | January 08 2025 10:43:10. | |
ReplaceDialog.pyc | File | 8016 bytes | January 08 2025 10:43:09. | |
ReplaceDialog.pyo | File | 8016 bytes | January 08 2025 10:43:09. | |
RstripExtension.py | File | 1050 bytes | January 08 2025 10:43:10. | |
RstripExtension.pyc | File | 1665 bytes | January 08 2025 10:43:10. | |
RstripExtension.pyo | File | 1665 bytes | January 08 2025 10:43:10. | |
ScriptBinding.py | File | 8459 bytes | January 08 2025 10:43:10. | |
ScriptBinding.pyc | File | 8370 bytes | January 08 2025 10:43:09. | |
ScriptBinding.pyo | File | 8370 bytes | January 08 2025 10:43:09. | |
ScrolledList.py | File | 4373 bytes | January 08 2025 10:43:10. | |
ScrolledList.pyc | File | 6768 bytes | January 08 2025 10:43:09. | |
ScrolledList.pyo | File | 6768 bytes | January 08 2025 10:43:09. | |
SearchDialog.py | File | 2630 bytes | January 08 2025 10:43:09. | |
SearchDialog.pyc | File | 4139 bytes | January 08 2025 10:43:09. | |
SearchDialog.pyo | File | 4139 bytes | January 08 2025 10:43:09. | |
SearchDialogBase.py | File | 7094 bytes | January 08 2025 10:43:10. | |
SearchDialogBase.pyc | File | 8631 bytes | January 08 2025 10:43:11. | |
SearchDialogBase.pyo | File | 8631 bytes | January 08 2025 10:43:11. | |
SearchEngine.py | File | 7463 bytes | January 08 2025 10:43:11. | |
SearchEngine.pyc | File | 8575 bytes | January 08 2025 10:43:10. | |
SearchEngine.pyo | File | 8575 bytes | January 08 2025 10:43:10. | |
StackViewer.py | File | 4431 bytes | January 08 2025 10:43:11. | |
StackViewer.pyc | File | 6677 bytes | January 08 2025 10:43:09. | |
StackViewer.pyo | File | 6677 bytes | January 08 2025 10:43:09. | |
TODO.txt | File | 8478 bytes | January 08 2025 10:43:09. | |
ToolTip.py | File | 3173 bytes | January 08 2025 10:43:11. | |
ToolTip.pyc | File | 4890 bytes | January 08 2025 10:43:10. | |
ToolTip.pyo | File | 4890 bytes | January 08 2025 10:43:10. | |
TreeWidget.py | File | 15037 bytes | January 08 2025 10:43:09. | |
TreeWidget.pyc | File | 18383 bytes | January 08 2025 10:43:10. | |
TreeWidget.pyo | File | 18383 bytes | January 08 2025 10:43:10. | |
UndoDelegator.py | File | 10787 bytes | January 08 2025 10:43:09. | |
UndoDelegator.pyc | File | 14218 bytes | January 08 2025 10:43:09. | |
UndoDelegator.pyo | File | 14218 bytes | January 08 2025 10:43:09. | |
WidgetRedirector.py | File | 6906 bytes | January 08 2025 10:43:09. | |
WidgetRedirector.pyc | File | 7951 bytes | January 08 2025 10:43:11. | |
WidgetRedirector.pyo | File | 7951 bytes | January 08 2025 10:43:11. | |
WindowList.py | File | 2473 bytes | January 08 2025 10:43:09. | |
WindowList.pyc | File | 3831 bytes | January 08 2025 10:43:10. | |
WindowList.pyo | File | 3831 bytes | January 08 2025 10:43:10. | |
ZoomHeight.py | File | 1300 bytes | January 08 2025 10:43:11. | |
ZoomHeight.pyc | File | 1711 bytes | January 08 2025 10:43:09. | |
ZoomHeight.pyo | File | 1711 bytes | January 08 2025 10:43:09. | |
__init__.py | File | 288 bytes | January 08 2025 10:43:09. | |
__init__.pyc | File | 444 bytes | January 08 2025 10:43:11. | |
__init__.pyo | File | 444 bytes | January 08 2025 10:43:11. | |
aboutDialog.py | File | 7014 bytes | January 08 2025 10:43:09. | |
aboutDialog.pyc | File | 7017 bytes | January 08 2025 10:43:09. | |
aboutDialog.pyo | File | 7017 bytes | January 08 2025 10:43:09. | |
config-extensions.def | File | 2965 bytes | January 08 2025 10:43:10. | |
config-highlight.def | File | 2515 bytes | January 08 2025 10:43:11. | |
config-keys.def | File | 7777 bytes | January 08 2025 10:43:10. | |
config-main.def | File | 2561 bytes | January 08 2025 10:43:09. | |
configDialog.py | File | 65958 bytes | January 08 2025 10:43:09. | |
configDialog.pyc | File | 54435 bytes | January 08 2025 10:43:11. | |
configDialog.pyo | File | 54435 bytes | January 08 2025 10:43:11. | |
configHandler.py | File | 32485 bytes | January 08 2025 10:43:10. | |
configHandler.pyc | File | 29985 bytes | January 08 2025 10:43:11. | |
configHandler.pyo | File | 29985 bytes | January 08 2025 10:43:11. | |
configHelpSourceEdit.py | File | 6686 bytes | January 08 2025 10:43:10. | |
configHelpSourceEdit.pyc | File | 6712 bytes | January 08 2025 10:43:10. | |
configHelpSourceEdit.pyo | File | 6712 bytes | January 08 2025 10:43:10. | |
configSectionNameDialog.py | File | 4040 bytes | January 08 2025 10:43:09. | |
configSectionNameDialog.pyc | File | 4510 bytes | January 08 2025 10:43:11. | |
configSectionNameDialog.pyo | File | 4510 bytes | January 08 2025 10:43:11. | |
dynOptionMenuWidget.py | File | 1984 bytes | January 08 2025 10:43:09. | |
dynOptionMenuWidget.pyc | File | 2868 bytes | January 08 2025 10:43:09. | |
dynOptionMenuWidget.pyo | File | 2868 bytes | January 08 2025 10:43:09. | |
extend.txt | File | 3642 bytes | January 08 2025 10:43:11. | |
help.html | File | 42414 bytes | January 08 2025 10:43:10. | |
help.py | File | 11035 bytes | January 08 2025 10:43:10. | |
help.pyc | File | 12543 bytes | January 08 2025 10:43:09. | |
help.pyo | File | 12543 bytes | January 08 2025 10:43:09. | |
help.txt | File | 12144 bytes | January 08 2025 10:43:09. | |
idle.py | File | 453 bytes | January 08 2025 10:43:09. | |
idle.pyc | File | 423 bytes | January 08 2025 10:43:09. | |
idle.pyo | File | 423 bytes | January 08 2025 10:43:09. | |
idle.pyw | File | 563 bytes | January 08 2025 10:43:10. | |
idlever.py | File | 415 bytes | January 08 2025 10:43:10. | |
idlever.pyc | File | 591 bytes | January 08 2025 10:43:10. | |
idlever.pyo | File | 591 bytes | January 08 2025 10:43:10. | |
keybindingDialog.py | File | 12468 bytes | January 08 2025 10:43:09. | |
keybindingDialog.pyc | File | 12368 bytes | January 08 2025 10:43:10. | |
keybindingDialog.pyo | File | 12368 bytes | January 08 2025 10:43:10. | |
macosxSupport.py | File | 8435 bytes | January 08 2025 10:43:09. | |
macosxSupport.pyc | File | 8585 bytes | January 08 2025 10:43:10. | |
macosxSupport.pyo | File | 8447 bytes | January 08 2025 10:43:09. | |
rpc.py | File | 20150 bytes | January 08 2025 10:43:10. | |
rpc.pyc | File | 22495 bytes | January 08 2025 10:43:10. | |
rpc.pyo | File | 22389 bytes | January 08 2025 10:43:09. | |
run.py | File | 12917 bytes | January 08 2025 10:43:10. | |
run.pyc | File | 13770 bytes | January 08 2025 10:43:11. | |
run.pyo | File | 13712 bytes | January 08 2025 10:43:09. | |
tabbedpages.py | File | 18439 bytes | January 08 2025 10:43:09. | |
tabbedpages.pyc | File | 19081 bytes | January 08 2025 10:43:11. | |
tabbedpages.pyo | File | 19081 bytes | January 08 2025 10:43:11. | |
textView.py | File | 3520 bytes | January 08 2025 10:43:09. | |
textView.pyc | File | 4116 bytes | January 08 2025 10:43:10. | |
textView.pyo | File | 4116 bytes | January 08 2025 10:43:10. |
"""An implementation of tabbed pages using only standard Tkinter. Originally developed for use in IDLE. Based on tabpage.py. Classes exported: TabbedPageSet -- A Tkinter implementation of a tabbed-page widget. TabSet -- A widget containing tabs (buttons) in one or more rows. """ from Tkinter import * class InvalidNameError(Exception): pass class AlreadyExistsError(Exception): pass class TabSet(Frame): """A widget containing tabs (buttons) in one or more rows. Only one tab may be selected at a time. """ def __init__(self, page_set, select_command, tabs=None, n_rows=1, max_tabs_per_row=5, expand_tabs=False, **kw): """Constructor arguments: select_command -- A callable which will be called when a tab is selected. It is called with the name of the selected tab as an argument. tabs -- A list of strings, the names of the tabs. Should be specified in the desired tab order. The first tab will be the default and first active tab. If tabs is None or empty, the TabSet will be initialized empty. n_rows -- Number of rows of tabs to be shown. If n_rows <= 0 or is None, then the number of rows will be decided by TabSet. See _arrange_tabs() for details. max_tabs_per_row -- Used for deciding how many rows of tabs are needed, when the number of rows is not constant. See _arrange_tabs() for details. """ Frame.__init__(self, page_set, **kw) self.select_command = select_command self.n_rows = n_rows self.max_tabs_per_row = max_tabs_per_row self.expand_tabs = expand_tabs self.page_set = page_set self._tabs = {} self._tab2row = {} if tabs: self._tab_names = list(tabs) else: self._tab_names = [] self._selected_tab = None self._tab_rows = [] self.padding_frame = Frame(self, height=2, borderwidth=0, relief=FLAT, background=self.cget('background')) self.padding_frame.pack(side=TOP, fill=X, expand=False) self._arrange_tabs() def add_tab(self, tab_name): """Add a new tab with the name given in tab_name.""" if not tab_name: raise InvalidNameError("Invalid Tab name: '%s'" % tab_name) if tab_name in self._tab_names: raise AlreadyExistsError("Tab named '%s' already exists" %tab_name) self._tab_names.append(tab_name) self._arrange_tabs() def remove_tab(self, tab_name): """Remove the tab named <tab_name>""" if not tab_name in self._tab_names: raise KeyError("No such Tab: '%s" % page_name) self._tab_names.remove(tab_name) self._arrange_tabs() def set_selected_tab(self, tab_name): """Show the tab named <tab_name> as the selected one""" if tab_name == self._selected_tab: return if tab_name is not None and tab_name not in self._tabs: raise KeyError("No such Tab: '%s" % page_name) # deselect the current selected tab if self._selected_tab is not None: self._tabs[self._selected_tab].set_normal() self._selected_tab = None if tab_name is not None: # activate the tab named tab_name self._selected_tab = tab_name tab = self._tabs[tab_name] tab.set_selected() # move the tab row with the selected tab to the bottom tab_row = self._tab2row[tab] tab_row.pack_forget() tab_row.pack(side=TOP, fill=X, expand=0) def _add_tab_row(self, tab_names, expand_tabs): if not tab_names: return tab_row = Frame(self) tab_row.pack(side=TOP, fill=X, expand=0) self._tab_rows.append(tab_row) for tab_name in tab_names: tab = TabSet.TabButton(tab_name, self.select_command, tab_row, self) if expand_tabs: tab.pack(side=LEFT, fill=X, expand=True) else: tab.pack(side=LEFT) self._tabs[tab_name] = tab self._tab2row[tab] = tab_row # tab is the last one created in the above loop tab.is_last_in_row = True def _reset_tab_rows(self): while self._tab_rows: tab_row = self._tab_rows.pop() tab_row.destroy() self._tab2row = {} def _arrange_tabs(self): """ Arrange the tabs in rows, in the order in which they were added. If n_rows >= 1, this will be the number of rows used. Otherwise the number of rows will be calculated according to the number of tabs and max_tabs_per_row. In this case, the number of rows may change when adding/removing tabs. """ # remove all tabs and rows for tab_name in self._tabs.keys(): self._tabs.pop(tab_name).destroy() self._reset_tab_rows() if not self._tab_names: return if self.n_rows is not None and self.n_rows > 0: n_rows = self.n_rows else: # calculate the required number of rows n_rows = (len(self._tab_names) - 1) // self.max_tabs_per_row + 1 # not expanding the tabs with more than one row is very ugly expand_tabs = self.expand_tabs or n_rows > 1 i = 0 # index in self._tab_names for row_index in xrange(n_rows): # calculate required number of tabs in this row n_tabs = (len(self._tab_names) - i - 1) // (n_rows - row_index) + 1 tab_names = self._tab_names[i:i + n_tabs] i += n_tabs self._add_tab_row(tab_names, expand_tabs) # re-select selected tab so it is properly displayed selected = self._selected_tab self.set_selected_tab(None) if selected in self._tab_names: self.set_selected_tab(selected) class TabButton(Frame): """A simple tab-like widget.""" bw = 2 # borderwidth def __init__(self, name, select_command, tab_row, tab_set): """Constructor arguments: name -- The tab's name, which will appear in its button. select_command -- The command to be called upon selection of the tab. It is called with the tab's name as an argument. """ Frame.__init__(self, tab_row, borderwidth=self.bw, relief=RAISED) self.name = name self.select_command = select_command self.tab_set = tab_set self.is_last_in_row = False self.button = Radiobutton( self, text=name, command=self._select_event, padx=5, pady=1, takefocus=FALSE, indicatoron=FALSE, highlightthickness=0, selectcolor='', borderwidth=0) self.button.pack(side=LEFT, fill=X, expand=True) self._init_masks() self.set_normal() def _select_event(self, *args): """Event handler for tab selection. With TabbedPageSet, this calls TabbedPageSet.change_page, so that selecting a tab changes the page. Note that this does -not- call set_selected -- it will be called by TabSet.set_selected_tab, which should be called when whatever the tabs are related to changes. """ self.select_command(self.name) return def set_selected(self): """Assume selected look""" self._place_masks(selected=True) def set_normal(self): """Assume normal look""" self._place_masks(selected=False) def _init_masks(self): page_set = self.tab_set.page_set background = page_set.pages_frame.cget('background') # mask replaces the middle of the border with the background color self.mask = Frame(page_set, borderwidth=0, relief=FLAT, background=background) # mskl replaces the bottom-left corner of the border with a normal # left border self.mskl = Frame(page_set, borderwidth=0, relief=FLAT, background=background) self.mskl.ml = Frame(self.mskl, borderwidth=self.bw, relief=RAISED) self.mskl.ml.place(x=0, y=-self.bw, width=2*self.bw, height=self.bw*4) # mskr replaces the bottom-right corner of the border with a normal # right border self.mskr = Frame(page_set, borderwidth=0, relief=FLAT, background=background) self.mskr.mr = Frame(self.mskr, borderwidth=self.bw, relief=RAISED) def _place_masks(self, selected=False): height = self.bw if selected: height += self.bw self.mask.place(in_=self, relx=0.0, x=0, rely=1.0, y=0, relwidth=1.0, width=0, relheight=0.0, height=height) self.mskl.place(in_=self, relx=0.0, x=-self.bw, rely=1.0, y=0, relwidth=0.0, width=self.bw, relheight=0.0, height=height) page_set = self.tab_set.page_set if selected and ((not self.is_last_in_row) or (self.winfo_rootx() + self.winfo_width() < page_set.winfo_rootx() + page_set.winfo_width()) ): # for a selected tab, if its rightmost edge isn't on the # rightmost edge of the page set, the right mask should be one # borderwidth shorter (vertically) height -= self.bw self.mskr.place(in_=self, relx=1.0, x=0, rely=1.0, y=0, relwidth=0.0, width=self.bw, relheight=0.0, height=height) self.mskr.mr.place(x=-self.bw, y=-self.bw, width=2*self.bw, height=height + self.bw*2) # finally, lower the tab set so that all of the frames we just # placed hide it self.tab_set.lower() class TabbedPageSet(Frame): """A Tkinter tabbed-pane widget. Constains set of 'pages' (or 'panes') with tabs above for selecting which page is displayed. Only one page will be displayed at a time. Pages may be accessed through the 'pages' attribute, which is a dictionary of pages, using the name given as the key. A page is an instance of a subclass of Tk's Frame widget. The page widgets will be created (and destroyed when required) by the TabbedPageSet. Do not call the page's pack/place/grid/destroy methods. Pages may be added or removed at any time using the add_page() and remove_page() methods. """ class Page(object): """Abstract base class for TabbedPageSet's pages. Subclasses must override the _show() and _hide() methods. """ uses_grid = False def __init__(self, page_set): self.frame = Frame(page_set, borderwidth=2, relief=RAISED) def _show(self): raise NotImplementedError def _hide(self): raise NotImplementedError class PageRemove(Page): """Page class using the grid placement manager's "remove" mechanism.""" uses_grid = True def _show(self): self.frame.grid(row=0, column=0, sticky=NSEW) def _hide(self): self.frame.grid_remove() class PageLift(Page): """Page class using the grid placement manager's "lift" mechanism.""" uses_grid = True def __init__(self, page_set): super(TabbedPageSet.PageLift, self).__init__(page_set) self.frame.grid(row=0, column=0, sticky=NSEW) self.frame.lower() def _show(self): self.frame.lift() def _hide(self): self.frame.lower() class PagePackForget(Page): """Page class using the pack placement manager's "forget" mechanism.""" def _show(self): self.frame.pack(fill=BOTH, expand=True) def _hide(self): self.frame.pack_forget() def __init__(self, parent, page_names=None, page_class=PageLift, n_rows=1, max_tabs_per_row=5, expand_tabs=False, **kw): """Constructor arguments: page_names -- A list of strings, each will be the dictionary key to a page's widget, and the name displayed on the page's tab. Should be specified in the desired page order. The first page will be the default and first active page. If page_names is None or empty, the TabbedPageSet will be initialized empty. n_rows, max_tabs_per_row -- Parameters for the TabSet which will manage the tabs. See TabSet's docs for details. page_class -- Pages can be shown/hidden using three mechanisms: * PageLift - All pages will be rendered one on top of the other. When a page is selected, it will be brought to the top, thus hiding all other pages. Using this method, the TabbedPageSet will not be resized when pages are switched. (It may still be resized when pages are added/removed.) * PageRemove - When a page is selected, the currently showing page is hidden, and the new page shown in its place. Using this method, the TabbedPageSet may resize when pages are changed. * PagePackForget - This mechanism uses the pack placement manager. When a page is shown it is packed, and when it is hidden it is unpacked (i.e. pack_forget). This mechanism may also cause the TabbedPageSet to resize when the page is changed. """ Frame.__init__(self, parent, **kw) self.page_class = page_class self.pages = {} self._pages_order = [] self._current_page = None self._default_page = None self.columnconfigure(0, weight=1) self.rowconfigure(1, weight=1) self.pages_frame = Frame(self) self.pages_frame.grid(row=1, column=0, sticky=NSEW) if self.page_class.uses_grid: self.pages_frame.columnconfigure(0, weight=1) self.pages_frame.rowconfigure(0, weight=1) # the order of the following commands is important self._tab_set = TabSet(self, self.change_page, n_rows=n_rows, max_tabs_per_row=max_tabs_per_row, expand_tabs=expand_tabs) if page_names: for name in page_names: self.add_page(name) self._tab_set.grid(row=0, column=0, sticky=NSEW) self.change_page(self._default_page) def add_page(self, page_name): """Add a new page with the name given in page_name.""" if not page_name: raise InvalidNameError("Invalid TabPage name: '%s'" % page_name) if page_name in self.pages: raise AlreadyExistsError( "TabPage named '%s' already exists" % page_name) self.pages[page_name] = self.page_class(self.pages_frame) self._pages_order.append(page_name) self._tab_set.add_tab(page_name) if len(self.pages) == 1: # adding first page self._default_page = page_name self.change_page(page_name) def remove_page(self, page_name): """Destroy the page whose name is given in page_name.""" if not page_name in self.pages: raise KeyError("No such TabPage: '%s" % page_name) self._pages_order.remove(page_name) # handle removing last remaining, default, or currently shown page if len(self._pages_order) > 0: if page_name == self._default_page: # set a new default page self._default_page = self._pages_order[0] else: self._default_page = None if page_name == self._current_page: self.change_page(self._default_page) self._tab_set.remove_tab(page_name) page = self.pages.pop(page_name) page.frame.destroy() def change_page(self, page_name): """Show the page whose name is given in page_name.""" if self._current_page == page_name: return if page_name is not None and page_name not in self.pages: raise KeyError("No such TabPage: '%s'" % page_name) if self._current_page is not None: self.pages[self._current_page]._hide() self._current_page = None if page_name is not None: self._current_page = page_name self.pages[page_name]._show() self._tab_set.set_selected_tab(page_name) def _tabbed_pages(parent): # test dialog root=Tk() width, height, x, y = list(map(int, re.split('[x+]', parent.geometry()))) root.geometry("+%d+%d"%(x, y + 175)) root.title("Test tabbed pages") tabPage=TabbedPageSet(root, page_names=['Foobar','Baz'], n_rows=0, expand_tabs=False, ) tabPage.pack(side=TOP, expand=TRUE, fill=BOTH) Label(tabPage.pages['Foobar'].frame, text='Foo', pady=20).pack() Label(tabPage.pages['Foobar'].frame, text='Bar', pady=20).pack() Label(tabPage.pages['Baz'].frame, text='Baz').pack() entryPgName=Entry(root) buttonAdd=Button(root, text='Add Page', command=lambda:tabPage.add_page(entryPgName.get())) buttonRemove=Button(root, text='Remove Page', command=lambda:tabPage.remove_page(entryPgName.get())) labelPgName=Label(root, text='name of page to add/remove:') buttonAdd.pack(padx=5, pady=5) buttonRemove.pack(padx=5, pady=5) labelPgName.pack(padx=5) entryPgName.pack(padx=5) root.mainloop() if __name__ == '__main__': from idlelib.idle_test.htest import run run(_tabbed_pages)
SILENT KILLER Tool