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 | """ |
csabella | 42bc8be | 2017-06-29 18:42:17 -0400 | [diff] [blame] | 4 | from tkinter import Toplevel, Text |
| 5 | from tkinter.ttk import Frame, Scrollbar, Button |
Terry Jan Reedy | 82c4615 | 2016-06-22 04:54:18 -0400 | [diff] [blame] | 6 | from tkinter.messagebox import showerror |
Steven M. Gava | 44d3d1a | 2001-07-31 06:59:02 +0000 | [diff] [blame] | 7 | |
Terry Jan Reedy | bfbaa6b | 2016-08-31 00:50:55 -0400 | [diff] [blame] | 8 | |
csabella | 42bc8be | 2017-06-29 18:42:17 -0400 | [diff] [blame] | 9 | class TextFrame(Frame): |
| 10 | "Display text with scrollbar." |
| 11 | |
| 12 | def __init__(self, parent, rawtext): |
| 13 | """Create a frame for Textview. |
| 14 | |
| 15 | parent - parent widget for this frame |
| 16 | rawtext - text to display |
| 17 | """ |
| 18 | super().__init__(parent) |
| 19 | self['relief'] = 'sunken' |
| 20 | self['height'] = 700 |
| 21 | # TODO: get fg/bg from theme. |
| 22 | self.bg = '#ffffff' |
| 23 | self.fg = '#000000' |
| 24 | |
| 25 | self.text = text = Text(self, wrap='word', highlightthickness=0, |
| 26 | fg=self.fg, bg=self.bg) |
| 27 | self.scroll = scroll = Scrollbar(self, orient='vertical', |
| 28 | takefocus=False, command=text.yview) |
| 29 | text['yscrollcommand'] = scroll.set |
| 30 | text.insert(0.0, rawtext) |
| 31 | text['state'] = 'disabled' |
| 32 | text.focus_set() |
| 33 | |
| 34 | scroll.pack(side='right', fill='y') |
| 35 | text.pack(side='left', expand=True, fill='both') |
| 36 | |
| 37 | |
| 38 | class ViewFrame(Frame): |
| 39 | "Display TextFrame and Close button." |
| 40 | def __init__(self, parent, text): |
| 41 | super().__init__(parent) |
| 42 | self.parent = parent |
| 43 | self.bind('<Return>', self.ok) |
| 44 | self.bind('<Escape>', self.ok) |
| 45 | self.textframe = TextFrame(self, text) |
| 46 | self.button_ok = button_ok = Button( |
| 47 | self, text='Close', command=self.ok, takefocus=False) |
| 48 | self.textframe.pack(side='top', expand=True, fill='both') |
| 49 | button_ok.pack(side='bottom') |
| 50 | |
| 51 | def ok(self, event=None): |
| 52 | """Dismiss text viewer dialog.""" |
| 53 | self.parent.destroy() |
| 54 | |
| 55 | |
| 56 | class ViewWindow(Toplevel): |
Terry Jan Reedy | a3623c8 | 2016-08-31 19:45:39 -0400 | [diff] [blame] | 57 | "A simple text viewer dialog for IDLE." |
Kurt B. Kaiser | 09cb74b | 2003-06-12 04:20:56 +0000 | [diff] [blame] | 58 | |
Louie Lu | ba365da | 2017-05-18 05:51:31 +0800 | [diff] [blame] | 59 | def __init__(self, parent, title, text, modal=True, |
Terry Jan Reedy | bfebfd8 | 2017-09-30 17:37:53 -0400 | [diff] [blame] | 60 | *, _htest=False, _utest=False): |
Louie Lu | ba365da | 2017-05-18 05:51:31 +0800 | [diff] [blame] | 61 | """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] | 62 | |
Louie Lu | ba365da | 2017-05-18 05:51:31 +0800 | [diff] [blame] | 63 | If modal is left True, users cannot interact with other windows |
| 64 | until the textview window is closed. |
Terry Jan Reedy | 537e2c8 | 2014-06-05 03:38:34 -0400 | [diff] [blame] | 65 | |
csabella | 0aa0a06 | 2017-05-28 06:50:55 -0400 | [diff] [blame] | 66 | parent - parent of this dialog |
| 67 | title - string which is title of popup dialog |
| 68 | text - text to display in dialog |
Terry Jan Reedy | 537e2c8 | 2014-06-05 03:38:34 -0400 | [diff] [blame] | 69 | _htest - bool; change box location when running htest. |
Louie Lu | ba365da | 2017-05-18 05:51:31 +0800 | [diff] [blame] | 70 | _utest - bool; don't wait_window when running unittest. |
Steven M. Gava | d721c48 | 2001-07-31 10:46:53 +0000 | [diff] [blame] | 71 | """ |
csabella | 42bc8be | 2017-06-29 18:42:17 -0400 | [diff] [blame] | 72 | super().__init__(parent) |
| 73 | self['borderwidth'] = 5 |
Terry Jan Reedy | a3623c8 | 2016-08-31 19:45:39 -0400 | [diff] [blame] | 74 | # Place dialog below parent if running htest. |
csabella | 42bc8be | 2017-06-29 18:42:17 -0400 | [diff] [blame] | 75 | x = parent.winfo_rootx() + 10 |
| 76 | y = parent.winfo_rooty() + (10 if not _htest else 100) |
| 77 | self.geometry(f'=750x500+{x}+{y}') |
Steven M. Gava | 44d3d1a | 2001-07-31 06:59:02 +0000 | [diff] [blame] | 78 | |
Steven M. Gava | d721c48 | 2001-07-31 10:46:53 +0000 | [diff] [blame] | 79 | self.title(title) |
csabella | 42bc8be | 2017-06-29 18:42:17 -0400 | [diff] [blame] | 80 | self.viewframe = ViewFrame(self, text) |
csabella | 0aa0a06 | 2017-05-28 06:50:55 -0400 | [diff] [blame] | 81 | self.protocol("WM_DELETE_WINDOW", self.ok) |
csabella | 42bc8be | 2017-06-29 18:42:17 -0400 | [diff] [blame] | 82 | self.button_ok = button_ok = Button(self, text='Close', |
| 83 | command=self.ok, takefocus=False) |
| 84 | self.viewframe.pack(side='top', expand=True, fill='both') |
Terry Jan Reedy | e91e763 | 2012-02-05 15:14:20 -0500 | [diff] [blame] | 85 | |
| 86 | if modal: |
| 87 | self.transient(parent) |
| 88 | self.grab_set() |
Louie Lu | ba365da | 2017-05-18 05:51:31 +0800 | [diff] [blame] | 89 | if not _utest: |
| 90 | self.wait_window() |
Kurt B. Kaiser | 6655e4b | 2002-12-31 16:03:23 +0000 | [diff] [blame] | 91 | |
csabella | 0aa0a06 | 2017-05-28 06:50:55 -0400 | [diff] [blame] | 92 | def ok(self, event=None): |
| 93 | """Dismiss text viewer dialog.""" |
Steven M. Gava | d721c48 | 2001-07-31 10:46:53 +0000 | [diff] [blame] | 94 | self.destroy() |
Steven M. Gava | 44d3d1a | 2001-07-31 06:59:02 +0000 | [diff] [blame] | 95 | |
Guido van Rossum | 8ce8a78 | 2007-11-01 19:42:39 +0000 | [diff] [blame] | 96 | |
Louie Lu | ba365da | 2017-05-18 05:51:31 +0800 | [diff] [blame] | 97 | def view_text(parent, title, text, modal=True, _utest=False): |
csabella | 42bc8be | 2017-06-29 18:42:17 -0400 | [diff] [blame] | 98 | """Create text viewer for given text. |
csabella | 0aa0a06 | 2017-05-28 06:50:55 -0400 | [diff] [blame] | 99 | |
| 100 | parent - parent of this dialog |
| 101 | title - string which is the title of popup dialog |
| 102 | text - text to display in this dialog |
| 103 | modal - controls if users can interact with other windows while this |
| 104 | dialog is displayed |
| 105 | _utest - bool; controls wait_window on unittest |
| 106 | """ |
csabella | 42bc8be | 2017-06-29 18:42:17 -0400 | [diff] [blame] | 107 | return ViewWindow(parent, title, text, modal, _utest=_utest) |
Guido van Rossum | 8ce8a78 | 2007-11-01 19:42:39 +0000 | [diff] [blame] | 108 | |
csabella | 0aa0a06 | 2017-05-28 06:50:55 -0400 | [diff] [blame] | 109 | |
Louie Lu | ba365da | 2017-05-18 05:51:31 +0800 | [diff] [blame] | 110 | def view_file(parent, title, filename, encoding=None, modal=True, _utest=False): |
csabella | 42bc8be | 2017-06-29 18:42:17 -0400 | [diff] [blame] | 111 | """Create text viewer for text in filename. |
csabella | 0aa0a06 | 2017-05-28 06:50:55 -0400 | [diff] [blame] | 112 | |
| 113 | Return error message if file cannot be read. Otherwise calls view_text |
| 114 | with contents of the file. |
| 115 | """ |
Guido van Rossum | 8ce8a78 | 2007-11-01 19:42:39 +0000 | [diff] [blame] | 116 | try: |
Éric Araujo | ccf03a1 | 2011-08-01 17:29:36 +0200 | [diff] [blame] | 117 | with open(filename, 'r', encoding=encoding) as file: |
| 118 | contents = file.read() |
Terry Jan Reedy | 82c4615 | 2016-06-22 04:54:18 -0400 | [diff] [blame] | 119 | except OSError: |
| 120 | showerror(title='File Load Error', |
csabella | 42bc8be | 2017-06-29 18:42:17 -0400 | [diff] [blame] | 121 | message=f'Unable to load file {filename!r} .', |
Terry Jan Reedy | 82c4615 | 2016-06-22 04:54:18 -0400 | [diff] [blame] | 122 | parent=parent) |
| 123 | except UnicodeDecodeError as err: |
| 124 | showerror(title='Unicode Decode Error', |
| 125 | message=str(err), |
| 126 | parent=parent) |
Guido van Rossum | 8ce8a78 | 2007-11-01 19:42:39 +0000 | [diff] [blame] | 127 | else: |
Louie Lu | ba365da | 2017-05-18 05:51:31 +0800 | [diff] [blame] | 128 | return view_text(parent, title, contents, modal, _utest=_utest) |
terryjreedy | 295304d | 2017-05-17 22:59:46 -0400 | [diff] [blame] | 129 | return None |
Louie Lu | ba365da | 2017-05-18 05:51:31 +0800 | [diff] [blame] | 130 | |
Guido van Rossum | 8ce8a78 | 2007-11-01 19:42:39 +0000 | [diff] [blame] | 131 | |
Steven M. Gava | 44d3d1a | 2001-07-31 06:59:02 +0000 | [diff] [blame] | 132 | if __name__ == '__main__': |
Terry Jan Reedy | 537e2c8 | 2014-06-05 03:38:34 -0400 | [diff] [blame] | 133 | import unittest |
| 134 | unittest.main('idlelib.idle_test.test_textview', verbosity=2, exit=False) |
Terry Jan Reedy | 1b392ff | 2014-05-24 18:48:18 -0400 | [diff] [blame] | 135 | from idlelib.idle_test.htest import run |
csabella | 42bc8be | 2017-06-29 18:42:17 -0400 | [diff] [blame] | 136 | run(ViewWindow) |