"""Simple text browser for IDLE """ from tkinter import Toplevel, Text, TclError,\ HORIZONTAL, VERTICAL, N, S, E, W from tkinter.ttk import Frame, Scrollbar, Button from tkinter.messagebox import showerror from idlelib.colorizer import color_config class AutoHiddenScrollbar(Scrollbar): """A scrollbar that is automatically hidden when not needed. Only the grid geometry manager is supported. """ def set(self, lo, hi): if float(lo) > 0.0 or float(hi) < 1.0: self.grid() else: self.grid_remove() super().set(lo, hi) def pack(self, **kwargs): raise TclError(f'{self.__class__.__name__} does not support "pack"') def place(self, **kwargs): raise TclError(f'{self.__class__.__name__} does not support "place"') class TextFrame(Frame): "Display text with scrollbar." def __init__(self, parent, rawtext, wrap='word'): """Create a frame for Textview. parent - parent widget for this frame rawtext - text to display """ super().__init__(parent) self['relief'] = 'sunken' self['height'] = 700 self.text = text = Text(self, wrap=wrap, highlightthickness=0) color_config(text) text.grid(row=0, column=0, sticky=N+S+E+W) self.grid_rowconfigure(0, weight=1) self.grid_columnconfigure(0, weight=1) text.insert(0.0, rawtext) text['state'] = 'disabled' text.focus_set() # vertical scrollbar self.yscroll = yscroll = AutoHiddenScrollbar(self, orient=VERTICAL, takefocus=False, command=text.yview) text['yscrollcommand'] = yscroll.set yscroll.grid(row=0, column=1, sticky=N+S) if wrap == 'none': # horizontal scrollbar self.xscroll = xscroll = AutoHiddenScrollbar(self, orient=HORIZONTAL, takefocus=False, command=text.xview) text['xscrollcommand'] = xscroll.set xscroll.grid(row=1, column=0, sticky=E+W) class ViewFrame(Frame): "Display TextFrame and Close button." def __init__(self, parent, text, wrap='word'): super().__init__(parent) self.parent = parent self.bind('<Return>', self.ok) self.bind('<Escape>', self.ok) self.textframe = TextFrame(self, text, wrap=wrap) self.button_ok = button_ok = Button( self, text='Close', command=self.ok, takefocus=False) self.textframe.pack(side='top', expand=True, fill='both') button_ok.pack(side='bottom') def ok(self, event=None): """Dismiss text viewer dialog.""" self.parent.destroy() class ViewWindow(Toplevel): "A simple text viewer dialog for IDLE." def __init__(self, parent, title, text, modal=True, wrap='word', *, _htest=False, _utest=False): """Show the given text in a scrollable window with a 'close' button. If modal is left True, users cannot interact with other windows until the textview window is closed. parent - parent of this dialog title - string which is title of popup dialog text - text to display in dialog wrap - type of text wrapping to use ('word', 'char' or 'none') _htest - bool; change box location when running htest. _utest - bool; don't wait_window when running unittest. """ super().__init__(parent) self['borderwidth'] = 5 # Place dialog below parent if running htest. x = parent.winfo_rootx() + 10 y = parent.winfo_rooty() + (10 if not _htest else 100) self.geometry(f'=750x500+{x}+{y}') self.title(title) self.viewframe = ViewFrame(self, text, wrap=wrap) self.protocol("WM_DELETE_WINDOW", self.ok) self.button_ok = button_ok = Button(self, text='Close', command=self.ok, takefocus=False) self.viewframe.pack(side='top', expand=True, fill='both') self.is_modal = modal if self.is_modal: self.transient(parent) self.grab_set() if not _utest: self.wait_window() def ok(self, event=None): """Dismiss text viewer dialog.""" if self.is_modal: self.grab_release() self.destroy() def view_text(parent, title, text, modal=True, wrap='word', _utest=False): """Create text viewer for given text. parent - parent of this dialog title - string which is the title of popup dialog text - text to display in this dialog wrap - type of text wrapping to use ('word', 'char' or 'none') modal - controls if users can interact with other windows while this dialog is displayed _utest - bool; controls wait_window on unittest """ return ViewWindow(parent, title, text, modal, wrap=wrap, _utest=_utest) def view_file(parent, title, filename, encoding, modal=True, wrap='word', _utest=False): """Create text viewer for text in filename. Return error message if file cannot be read. Otherwise calls view_text with contents of the file. """ try: with open(filename, 'r', encoding=encoding) as file: contents = file.read() except OSError: showerror(title='File Load Error', message=f'Unable to load file {filename!r} .', parent=parent) except UnicodeDecodeError as err: showerror(title='Unicode Decode Error', message=str(err), parent=parent) else: return view_text(parent, title, contents, modal, wrap=wrap, _utest=_utest) return None if __name__ == '__main__': from unittest import main main('idlelib.idle_test.test_textview', verbosity=2, exit=False) from idlelib.idle_test.htest import run run(ViewWindow)
Name | Type | Size | Permission | Actions |
---|---|---|---|---|
Icons | Folder | 0755 |
|
|
__pycache__ | Folder | 0755 |
|
|
idle_test | Folder | 0755 |
|
|
CREDITS.txt | File | 1.82 KB | 0644 |
|
ChangeLog | File | 55.04 KB | 0644 |
|
HISTORY.txt | File | 10.07 KB | 0644 |
|
NEWS.txt | File | 38.91 KB | 0644 |
|
NEWS2x.txt | File | 26.54 KB | 0644 |
|
README.txt | File | 9.37 KB | 0644 |
|
TODO.txt | File | 8.28 KB | 0644 |
|
__init__.py | File | 396 B | 0644 |
|
__main__.py | File | 159 B | 0644 |
|
_pyclbr.py | File | 14.84 KB | 0644 |
|
autocomplete.py | File | 9.11 KB | 0644 |
|
autocomplete_w.py | File | 19.36 KB | 0644 |
|
autoexpand.py | File | 3.14 KB | 0644 |
|
browser.py | File | 8.09 KB | 0644 |
|
calltip.py | File | 5.92 KB | 0644 |
|
calltip_w.py | File | 6.94 KB | 0644 |
|
codecontext.py | File | 10.24 KB | 0644 |
|
colorizer.py | File | 11.01 KB | 0644 |
|
config-extensions.def | File | 2.21 KB | 0644 |
|
config-highlight.def | File | 2.62 KB | 0644 |
|
config-keys.def | File | 10.52 KB | 0644 |
|
config-main.def | File | 3.05 KB | 0644 |
|
config.py | File | 37.97 KB | 0644 |
|
config_key.py | File | 13.09 KB | 0644 |
|
configdialog.py | File | 98.69 KB | 0644 |
|
debugger.py | File | 18.65 KB | 0644 |
|
debugger_r.py | File | 11.86 KB | 0644 |
|
debugobj.py | File | 3.96 KB | 0644 |
|
debugobj_r.py | File | 1.06 KB | 0644 |
|
delegator.py | File | 1.02 KB | 0644 |
|
dynoption.py | File | 1.97 KB | 0644 |
|
editor.py | File | 65.7 KB | 0644 |
|
extend.txt | File | 3.56 KB | 0644 |
|
filelist.py | File | 3.8 KB | 0644 |
|
grep.py | File | 6.58 KB | 0644 |
|
help.html | File | 53.82 KB | 0644 |
|
help.py | File | 11.06 KB | 0644 |
|
help_about.py | File | 8.77 KB | 0644 |
|
history.py | File | 3.95 KB | 0644 |
|
hyperparser.py | File | 12.58 KB | 0644 |
|
idle.py | File | 454 B | 0644 |
|
idle.pyw | File | 570 B | 0644 |
|
iomenu.py | File | 20.25 KB | 0644 |
|
macosx.py | File | 9.43 KB | 0644 |
|
mainmenu.py | File | 3.62 KB | 0644 |
|
multicall.py | File | 18.21 KB | 0644 |
|
outwin.py | File | 5.67 KB | 0644 |
|
paragraph.py | File | 7 KB | 0644 |
|
parenmatch.py | File | 7.04 KB | 0644 |
|
pathbrowser.py | File | 3.12 KB | 0644 |
|
percolator.py | File | 3.06 KB | 0644 |
|
pyparse.py | File | 19.65 KB | 0644 |
|
pyshell.py | File | 56.38 KB | 0755 |
|
query.py | File | 12.14 KB | 0644 |
|
redirector.py | File | 6.71 KB | 0644 |
|
replace.py | File | 7.33 KB | 0644 |
|
rpc.py | File | 20.64 KB | 0644 |
|
rstrip.py | File | 868 B | 0644 |
|
run.py | File | 16.87 KB | 0644 |
|
runscript.py | File | 7.66 KB | 0644 |
|
scrolledlist.py | File | 4.35 KB | 0644 |
|
search.py | File | 3.09 KB | 0644 |
|
searchbase.py | File | 7.28 KB | 0644 |
|
searchengine.py | File | 7.3 KB | 0644 |
|
squeezer.py | File | 13 KB | 0644 |
|
stackviewer.py | File | 4.35 KB | 0644 |
|
statusbar.py | File | 1.41 KB | 0644 |
|
textview.py | File | 5.98 KB | 0644 |
|
tooltip.py | File | 6.33 KB | 0644 |
|
tree.py | File | 14.74 KB | 0644 |
|
undo.py | File | 10.79 KB | 0644 |
|
window.py | File | 2.53 KB | 0644 |
|
zoomheight.py | File | 1.31 KB | 0644 |
|
zzdummy.py | File | 961 B | 0644 |
|