Kurt B. Kaiser | 09cb74b | 2003-06-12 04:20:56 +0000 | [diff] [blame] | 1 | """Simple text browser for IDLE |
| 2 | |
Steven M. Gava | 44d3d1a | 2001-07-31 06:59:02 +0000 | [diff] [blame] | 3 | """ |
Tal Einat | 604e7b9 | 2018-09-25 15:10:14 +0300 | [diff] [blame] | 4 | from tkinter import Toplevel, Text, TclError,\ |
| 5 | HORIZONTAL, VERTICAL, N, S, E, W |
csabella | 42bc8be | 2017-06-29 18:42:17 -0400 | [diff] [blame] | 6 | from tkinter.ttk import Frame, Scrollbar, Button |
Terry Jan Reedy | 82c4615 | 2016-06-22 04:54:18 -0400 | [diff] [blame] | 7 | from tkinter.messagebox import showerror |
Steven M. Gava | 44d3d1a | 2001-07-31 06:59:02 +0000 | [diff] [blame] | 8 | |
Tal Einat | c87d9f4 | 2018-09-23 15:23:15 +0300 | [diff] [blame] | 9 | from idlelib.colorizer import color_config |
| 10 | |
Terry Jan Reedy | bfbaa6b | 2016-08-31 00:50:55 -0400 | [diff] [blame] | 11 | |
Tal Einat | 604e7b9 | 2018-09-25 15:10:14 +0300 | [diff] [blame] | 12 | class AutoHiddenScrollbar(Scrollbar): |
| 13 | """A scrollbar that is automatically hidden when not needed. |
| 14 | |
| 15 | Only the grid geometry manager is supported. |
| 16 | """ |
| 17 | def set(self, lo, hi): |
| 18 | if float(lo) > 0.0 or float(hi) < 1.0: |
| 19 | self.grid() |
| 20 | else: |
| 21 | self.grid_remove() |
| 22 | super().set(lo, hi) |
| 23 | |
| 24 | def pack(self, **kwargs): |
| 25 | raise TclError(f'{self.__class__.__name__} does not support "pack"') |
| 26 | |
| 27 | def place(self, **kwargs): |
| 28 | raise TclError(f'{self.__class__.__name__} does not support "place"') |
| 29 | |
| 30 | |
csabella | 42bc8be | 2017-06-29 18:42:17 -0400 | [diff] [blame] | 31 | class TextFrame(Frame): |
| 32 | "Display text with scrollbar." |
| 33 | |
Tal Einat | 604e7b9 | 2018-09-25 15:10:14 +0300 | [diff] [blame] | 34 | def __init__(self, parent, rawtext, wrap='word'): |
csabella | 42bc8be | 2017-06-29 18:42:17 -0400 | [diff] [blame] | 35 | """Create a frame for Textview. |
| 36 | |
| 37 | parent - parent widget for this frame |
| 38 | rawtext - text to display |
| 39 | """ |
| 40 | super().__init__(parent) |
| 41 | self['relief'] = 'sunken' |
| 42 | self['height'] = 700 |
csabella | 42bc8be | 2017-06-29 18:42:17 -0400 | [diff] [blame] | 43 | |
Tal Einat | 604e7b9 | 2018-09-25 15:10:14 +0300 | [diff] [blame] | 44 | self.text = text = Text(self, wrap=wrap, highlightthickness=0) |
Tal Einat | c87d9f4 | 2018-09-23 15:23:15 +0300 | [diff] [blame] | 45 | color_config(text) |
Tal Einat | 604e7b9 | 2018-09-25 15:10:14 +0300 | [diff] [blame] | 46 | text.grid(row=0, column=0, sticky=N+S+E+W) |
| 47 | self.grid_rowconfigure(0, weight=1) |
| 48 | self.grid_columnconfigure(0, weight=1) |
csabella | 42bc8be | 2017-06-29 18:42:17 -0400 | [diff] [blame] | 49 | text.insert(0.0, rawtext) |
| 50 | text['state'] = 'disabled' |
| 51 | text.focus_set() |
| 52 | |
Tal Einat | 604e7b9 | 2018-09-25 15:10:14 +0300 | [diff] [blame] | 53 | # vertical scrollbar |
| 54 | self.yscroll = yscroll = AutoHiddenScrollbar(self, orient=VERTICAL, |
| 55 | takefocus=False, |
| 56 | command=text.yview) |
| 57 | text['yscrollcommand'] = yscroll.set |
| 58 | yscroll.grid(row=0, column=1, sticky=N+S) |
| 59 | |
| 60 | if wrap == 'none': |
| 61 | # horizontal scrollbar |
| 62 | self.xscroll = xscroll = AutoHiddenScrollbar(self, orient=HORIZONTAL, |
| 63 | takefocus=False, |
| 64 | command=text.xview) |
| 65 | text['xscrollcommand'] = xscroll.set |
| 66 | xscroll.grid(row=1, column=0, sticky=E+W) |
csabella | 42bc8be | 2017-06-29 18:42:17 -0400 | [diff] [blame] | 67 | |
| 68 | |
| 69 | class ViewFrame(Frame): |
| 70 | "Display TextFrame and Close button." |
Tal Einat | 604e7b9 | 2018-09-25 15:10:14 +0300 | [diff] [blame] | 71 | def __init__(self, parent, text, wrap='word'): |
csabella | 42bc8be | 2017-06-29 18:42:17 -0400 | [diff] [blame] | 72 | super().__init__(parent) |
| 73 | self.parent = parent |
| 74 | self.bind('<Return>', self.ok) |
| 75 | self.bind('<Escape>', self.ok) |
Tal Einat | 604e7b9 | 2018-09-25 15:10:14 +0300 | [diff] [blame] | 76 | self.textframe = TextFrame(self, text, wrap=wrap) |
csabella | 42bc8be | 2017-06-29 18:42:17 -0400 | [diff] [blame] | 77 | self.button_ok = button_ok = Button( |
| 78 | self, text='Close', command=self.ok, takefocus=False) |
| 79 | self.textframe.pack(side='top', expand=True, fill='both') |
| 80 | button_ok.pack(side='bottom') |
| 81 | |
| 82 | def ok(self, event=None): |
| 83 | """Dismiss text viewer dialog.""" |
| 84 | self.parent.destroy() |
| 85 | |
| 86 | |
| 87 | class ViewWindow(Toplevel): |
Terry Jan Reedy | a3623c8 | 2016-08-31 19:45:39 -0400 | [diff] [blame] | 88 | "A simple text viewer dialog for IDLE." |
Kurt B. Kaiser | 09cb74b | 2003-06-12 04:20:56 +0000 | [diff] [blame] | 89 | |
Tal Einat | 604e7b9 | 2018-09-25 15:10:14 +0300 | [diff] [blame] | 90 | def __init__(self, parent, title, text, modal=True, wrap='word', |
Terry Jan Reedy | bfebfd8 | 2017-09-30 17:37:53 -0400 | [diff] [blame] | 91 | *, _htest=False, _utest=False): |
Louie Lu | ba365da | 2017-05-18 05:51:31 +0800 | [diff] [blame] | 92 | """Show the given text in a scrollable window with a 'close' button. |
Guido van Rossum | 8ce8a78 | 2007-11-01 19:42:39 +0000 | [diff] [blame] | 93 | |
Louie Lu | ba365da | 2017-05-18 05:51:31 +0800 | [diff] [blame] | 94 | If modal is left True, users cannot interact with other windows |
| 95 | until the textview window is closed. |
Terry Jan Reedy | 537e2c8 | 2014-06-05 03:38:34 -0400 | [diff] [blame] | 96 | |
csabella | 0aa0a06 | 2017-05-28 06:50:55 -0400 | [diff] [blame] | 97 | parent - parent of this dialog |
| 98 | title - string which is title of popup dialog |
| 99 | text - text to display in dialog |
Tal Einat | 604e7b9 | 2018-09-25 15:10:14 +0300 | [diff] [blame] | 100 | wrap - type of text wrapping to use ('word', 'char' or 'none') |
Terry Jan Reedy | 537e2c8 | 2014-06-05 03:38:34 -0400 | [diff] [blame] | 101 | _htest - bool; change box location when running htest. |
Louie Lu | ba365da | 2017-05-18 05:51:31 +0800 | [diff] [blame] | 102 | _utest - bool; don't wait_window when running unittest. |
Steven M. Gava | d721c48 | 2001-07-31 10:46:53 +0000 | [diff] [blame] | 103 | """ |
csabella | 42bc8be | 2017-06-29 18:42:17 -0400 | [diff] [blame] | 104 | super().__init__(parent) |
| 105 | self['borderwidth'] = 5 |
Terry Jan Reedy | a3623c8 | 2016-08-31 19:45:39 -0400 | [diff] [blame] | 106 | # Place dialog below parent if running htest. |
csabella | 42bc8be | 2017-06-29 18:42:17 -0400 | [diff] [blame] | 107 | x = parent.winfo_rootx() + 10 |
| 108 | y = parent.winfo_rooty() + (10 if not _htest else 100) |
| 109 | self.geometry(f'=750x500+{x}+{y}') |
Steven M. Gava | 44d3d1a | 2001-07-31 06:59:02 +0000 | [diff] [blame] | 110 | |
Steven M. Gava | d721c48 | 2001-07-31 10:46:53 +0000 | [diff] [blame] | 111 | self.title(title) |
Tal Einat | 604e7b9 | 2018-09-25 15:10:14 +0300 | [diff] [blame] | 112 | self.viewframe = ViewFrame(self, text, wrap=wrap) |
csabella | 0aa0a06 | 2017-05-28 06:50:55 -0400 | [diff] [blame] | 113 | self.protocol("WM_DELETE_WINDOW", self.ok) |
csabella | 42bc8be | 2017-06-29 18:42:17 -0400 | [diff] [blame] | 114 | self.button_ok = button_ok = Button(self, text='Close', |
| 115 | command=self.ok, takefocus=False) |
| 116 | self.viewframe.pack(side='top', expand=True, fill='both') |
Terry Jan Reedy | e91e763 | 2012-02-05 15:14:20 -0500 | [diff] [blame] | 117 | |
Tal Einat | dd74369 | 2018-08-02 10:30:06 +0300 | [diff] [blame] | 118 | self.is_modal = modal |
| 119 | if self.is_modal: |
Terry Jan Reedy | e91e763 | 2012-02-05 15:14:20 -0500 | [diff] [blame] | 120 | self.transient(parent) |
| 121 | self.grab_set() |
Louie Lu | ba365da | 2017-05-18 05:51:31 +0800 | [diff] [blame] | 122 | if not _utest: |
| 123 | self.wait_window() |
Kurt B. Kaiser | 6655e4b | 2002-12-31 16:03:23 +0000 | [diff] [blame] | 124 | |
csabella | 0aa0a06 | 2017-05-28 06:50:55 -0400 | [diff] [blame] | 125 | def ok(self, event=None): |
| 126 | """Dismiss text viewer dialog.""" |
Tal Einat | dd74369 | 2018-08-02 10:30:06 +0300 | [diff] [blame] | 127 | if self.is_modal: |
| 128 | self.grab_release() |
Steven M. Gava | d721c48 | 2001-07-31 10:46:53 +0000 | [diff] [blame] | 129 | self.destroy() |
Steven M. Gava | 44d3d1a | 2001-07-31 06:59:02 +0000 | [diff] [blame] | 130 | |
Guido van Rossum | 8ce8a78 | 2007-11-01 19:42:39 +0000 | [diff] [blame] | 131 | |
Tal Einat | 604e7b9 | 2018-09-25 15:10:14 +0300 | [diff] [blame] | 132 | def view_text(parent, title, text, modal=True, wrap='word', _utest=False): |
csabella | 42bc8be | 2017-06-29 18:42:17 -0400 | [diff] [blame] | 133 | """Create text viewer for given text. |
csabella | 0aa0a06 | 2017-05-28 06:50:55 -0400 | [diff] [blame] | 134 | |
| 135 | parent - parent of this dialog |
| 136 | title - string which is the title of popup dialog |
| 137 | text - text to display in this dialog |
Tal Einat | 604e7b9 | 2018-09-25 15:10:14 +0300 | [diff] [blame] | 138 | wrap - type of text wrapping to use ('word', 'char' or 'none') |
csabella | 0aa0a06 | 2017-05-28 06:50:55 -0400 | [diff] [blame] | 139 | modal - controls if users can interact with other windows while this |
| 140 | dialog is displayed |
| 141 | _utest - bool; controls wait_window on unittest |
| 142 | """ |
Tal Einat | 604e7b9 | 2018-09-25 15:10:14 +0300 | [diff] [blame] | 143 | return ViewWindow(parent, title, text, modal, wrap=wrap, _utest=_utest) |
Guido van Rossum | 8ce8a78 | 2007-11-01 19:42:39 +0000 | [diff] [blame] | 144 | |
csabella | 0aa0a06 | 2017-05-28 06:50:55 -0400 | [diff] [blame] | 145 | |
Tal Einat | 604e7b9 | 2018-09-25 15:10:14 +0300 | [diff] [blame] | 146 | def view_file(parent, title, filename, encoding, modal=True, wrap='word', |
| 147 | _utest=False): |
csabella | 42bc8be | 2017-06-29 18:42:17 -0400 | [diff] [blame] | 148 | """Create text viewer for text in filename. |
csabella | 0aa0a06 | 2017-05-28 06:50:55 -0400 | [diff] [blame] | 149 | |
| 150 | Return error message if file cannot be read. Otherwise calls view_text |
| 151 | with contents of the file. |
| 152 | """ |
Guido van Rossum | 8ce8a78 | 2007-11-01 19:42:39 +0000 | [diff] [blame] | 153 | try: |
Éric Araujo | ccf03a1 | 2011-08-01 17:29:36 +0200 | [diff] [blame] | 154 | with open(filename, 'r', encoding=encoding) as file: |
| 155 | contents = file.read() |
Terry Jan Reedy | 82c4615 | 2016-06-22 04:54:18 -0400 | [diff] [blame] | 156 | except OSError: |
| 157 | showerror(title='File Load Error', |
csabella | 42bc8be | 2017-06-29 18:42:17 -0400 | [diff] [blame] | 158 | message=f'Unable to load file {filename!r} .', |
Terry Jan Reedy | 82c4615 | 2016-06-22 04:54:18 -0400 | [diff] [blame] | 159 | parent=parent) |
| 160 | except UnicodeDecodeError as err: |
| 161 | showerror(title='Unicode Decode Error', |
| 162 | message=str(err), |
| 163 | parent=parent) |
Guido van Rossum | 8ce8a78 | 2007-11-01 19:42:39 +0000 | [diff] [blame] | 164 | else: |
Tal Einat | 604e7b9 | 2018-09-25 15:10:14 +0300 | [diff] [blame] | 165 | return view_text(parent, title, contents, modal, wrap=wrap, |
| 166 | _utest=_utest) |
terryjreedy | 295304d | 2017-05-17 22:59:46 -0400 | [diff] [blame] | 167 | return None |
Louie Lu | ba365da | 2017-05-18 05:51:31 +0800 | [diff] [blame] | 168 | |
Guido van Rossum | 8ce8a78 | 2007-11-01 19:42:39 +0000 | [diff] [blame] | 169 | |
Steven M. Gava | 44d3d1a | 2001-07-31 06:59:02 +0000 | [diff] [blame] | 170 | if __name__ == '__main__': |
Terry Jan Reedy | 4d92158 | 2018-06-19 19:12:52 -0400 | [diff] [blame] | 171 | from unittest import main |
| 172 | main('idlelib.idle_test.test_textview', verbosity=2, exit=False) |
| 173 | |
Terry Jan Reedy | 1b392ff | 2014-05-24 18:48:18 -0400 | [diff] [blame] | 174 | from idlelib.idle_test.htest import run |
csabella | 42bc8be | 2017-06-29 18:42:17 -0400 | [diff] [blame] | 175 | run(ViewWindow) |