blob: 64b13ac2abb3b2eb76fa35d9eee9b2eff0bd70ae [file] [log] [blame]
Kurt B. Kaiser09cb74b2003-06-12 04:20:56 +00001"""About Dialog for IDLE
2
Steven M. Gava44d3d1a2001-07-31 06:59:02 +00003"""
Neal Norwitz9d72bb42007-04-17 08:48:32 +00004import os
csabella9a02ae32017-06-26 22:28:58 -04005import sys
6from platform import python_version, architecture
Terry Jan Reedybfbaa6b2016-08-31 00:50:55 -04007
csabellad352d682017-06-23 12:00:29 -04008from tkinter import Toplevel, Frame, Label, Button, PhotoImage
9from tkinter import SUNKEN, TOP, BOTTOM, LEFT, X, BOTH, W, EW, NSEW, E
Terry Jan Reedybfbaa6b2016-08-31 00:50:55 -040010
Terry Jan Reedy6fa5bdc2016-05-28 13:22:31 -040011from idlelib import textview
Neal Norwitz539594f2002-11-30 19:12:41 +000012
Terry Jan Reedybfbaa6b2016-08-31 00:50:55 -040013
csabella9a02ae32017-06-26 22:28:58 -040014def build_bits():
15 "Return bits for platform."
16 if sys.platform == 'darwin':
17 return '64' if sys.maxsize > 2**32 else '32'
18 else:
19 return architecture()[0][:2]
20
21
Steven M. Gava44d3d1a2001-07-31 06:59:02 +000022class AboutDialog(Toplevel):
Kurt B. Kaiser09cb74b2003-06-12 04:20:56 +000023 """Modal about dialog for idle
24
Kurt B. Kaiser6655e4b2002-12-31 16:03:23 +000025 """
Terry Jan Reedybfebfd82017-09-30 17:37:53 -040026 def __init__(self, parent, title=None, *, _htest=False, _utest=False):
csabella5a346d52017-05-27 15:49:26 -040027 """Create popup, do not return until tk widget destroyed.
28
29 parent - parent of this dialog
30 title - string which is title of popup dialog
Terry Jan Reedy1b392ff2014-05-24 18:48:18 -040031 _htest - bool, change box location when running htest
mlouielu054e0912017-05-22 06:19:35 +080032 _utest - bool, don't wait_window when running unittest
Terry Jan Reedy1b392ff2014-05-24 18:48:18 -040033 """
Steven M. Gava885c0bb2001-07-31 10:44:35 +000034 Toplevel.__init__(self, parent)
35 self.configure(borderwidth=5)
Terry Jan Reedy1b392ff2014-05-24 18:48:18 -040036 # place dialog below parent if running htest
37 self.geometry("+%d+%d" % (
38 parent.winfo_rootx()+30,
39 parent.winfo_rooty()+(30 if not _htest else 100)))
csabella9a02ae32017-06-26 22:28:58 -040040 self.bg = "#bbbbbb"
41 self.fg = "#000000"
csabella5a346d52017-05-27 15:49:26 -040042 self.create_widgets()
43 self.resizable(height=False, width=False)
csabella9a02ae32017-06-26 22:28:58 -040044 self.title(title or
45 f'About IDLE {python_version()} ({build_bits()} bit)')
Steven M. Gava885c0bb2001-07-31 10:44:35 +000046 self.transient(parent)
47 self.grab_set()
csabella5a346d52017-05-27 15:49:26 -040048 self.protocol("WM_DELETE_WINDOW", self.ok)
Steven M. Gava885c0bb2001-07-31 10:44:35 +000049 self.parent = parent
csabella5a346d52017-05-27 15:49:26 -040050 self.button_ok.focus_set()
51 self.bind('<Return>', self.ok) # dismiss dialog
52 self.bind('<Escape>', self.ok) # dismiss dialog
mlouielu054e0912017-05-22 06:19:35 +080053 self._current_textview = None
54 self._utest = _utest
55
56 if not _utest:
57 self.deiconify()
58 self.wait_window()
Kurt B. Kaiser6655e4b2002-12-31 16:03:23 +000059
csabella5a346d52017-05-27 15:49:26 -040060 def create_widgets(self):
csabella5a346d52017-05-27 15:49:26 -040061 frame = Frame(self, borderwidth=2, relief=SUNKEN)
62 frame_buttons = Frame(self)
63 frame_buttons.pack(side=BOTTOM, fill=X)
64 frame.pack(side=TOP, expand=True, fill=BOTH)
65 self.button_ok = Button(frame_buttons, text='Close',
66 command=self.ok)
67 self.button_ok.pack(padx=5, pady=5)
68
69 frame_background = Frame(frame, bg=self.bg)
70 frame_background.pack(expand=True, fill=BOTH)
71
72 header = Label(frame_background, text='IDLE', fg=self.fg,
73 bg=self.bg, font=('courier', 24, 'bold'))
csabellad352d682017-06-23 12:00:29 -040074 header.grid(row=0, column=0, sticky=E, padx=10, pady=10)
75
76 tk_patchlevel = self.tk.call('info', 'patchlevel')
77 ext = '.png' if tk_patchlevel >= '8.6' else '.gif'
78 icon = os.path.join(os.path.abspath(os.path.dirname(__file__)),
79 'Icons', f'idle_48{ext}')
80 self.icon_image = PhotoImage(master=self._root(), file=icon)
81 logo = Label(frame_background, image=self.icon_image, bg=self.bg)
82 logo.grid(row=0, column=0, sticky=W, rowspan=2, padx=10, pady=10)
83
Mark Roseman592eda12017-06-27 19:42:10 -070084 byline_text = "Python's Integrated Development\nand Learning Environment" + 5*'\n'
csabella5a346d52017-05-27 15:49:26 -040085 byline = Label(frame_background, text=byline_text, justify=LEFT,
86 fg=self.fg, bg=self.bg)
87 byline.grid(row=2, column=0, sticky=W, columnspan=3, padx=10, pady=5)
88 email = Label(frame_background, text='email: idle-dev@python.org',
89 justify=LEFT, fg=self.fg, bg=self.bg)
90 email.grid(row=6, column=0, columnspan=2, sticky=W, padx=10, pady=0)
91 docs = Label(frame_background, text='https://docs.python.org/' +
csabella18ede062017-06-23 20:00:58 -040092 python_version()[:3] + '/library/idle.html',
csabella5a346d52017-05-27 15:49:26 -040093 justify=LEFT, fg=self.fg, bg=self.bg)
94 docs.grid(row=7, column=0, columnspan=2, sticky=W, padx=10, pady=0)
95
96 Frame(frame_background, borderwidth=1, relief=SUNKEN,
Kurt B. Kaiser09cb74b2003-06-12 04:20:56 +000097 height=2, bg=self.bg).grid(row=8, column=0, sticky=EW,
98 columnspan=3, padx=5, pady=5)
csabella5a346d52017-05-27 15:49:26 -040099
csabella18ede062017-06-23 20:00:58 -0400100 pyver = Label(frame_background,
101 text='Python version: ' + python_version(),
csabella5a346d52017-05-27 15:49:26 -0400102 fg=self.fg, bg=self.bg)
103 pyver.grid(row=9, column=0, sticky=W, padx=10, pady=0)
csabella5a346d52017-05-27 15:49:26 -0400104 tkver = Label(frame_background, text='Tk version: ' + tk_patchlevel,
105 fg=self.fg, bg=self.bg)
106 tkver.grid(row=9, column=1, sticky=W, padx=2, pady=0)
107 py_buttons = Frame(frame_background, bg=self.bg)
108 py_buttons.grid(row=10, column=0, columnspan=2, sticky=NSEW)
109 self.py_license = Button(py_buttons, text='License', width=8,
110 highlightbackground=self.bg,
111 command=self.show_py_license)
112 self.py_license.pack(side=LEFT, padx=10, pady=10)
113 self.py_copyright = Button(py_buttons, text='Copyright', width=8,
114 highlightbackground=self.bg,
115 command=self.show_py_copyright)
116 self.py_copyright.pack(side=LEFT, padx=10, pady=10)
117 self.py_credits = Button(py_buttons, text='Credits', width=8,
118 highlightbackground=self.bg,
119 command=self.show_py_credits)
120 self.py_credits.pack(side=LEFT, padx=10, pady=10)
121
122 Frame(frame_background, borderwidth=1, relief=SUNKEN,
Kurt B. Kaiser09cb74b2003-06-12 04:20:56 +0000123 height=2, bg=self.bg).grid(row=11, column=0, sticky=EW,
124 columnspan=3, padx=5, pady=5)
Steven M. Gava885c0bb2001-07-31 10:44:35 +0000125
csabella18ede062017-06-23 20:00:58 -0400126 idlever = Label(frame_background,
127 text='IDLE version: ' + python_version(),
csabella5a346d52017-05-27 15:49:26 -0400128 fg=self.fg, bg=self.bg)
129 idlever.grid(row=12, column=0, sticky=W, padx=10, pady=0)
130 idle_buttons = Frame(frame_background, bg=self.bg)
131 idle_buttons.grid(row=13, column=0, columnspan=3, sticky=NSEW)
132 self.readme = Button(idle_buttons, text='README', width=8,
133 highlightbackground=self.bg,
134 command=self.show_readme)
135 self.readme.pack(side=LEFT, padx=10, pady=10)
136 self.idle_news = Button(idle_buttons, text='NEWS', width=8,
137 highlightbackground=self.bg,
138 command=self.show_idle_news)
139 self.idle_news.pack(side=LEFT, padx=10, pady=10)
140 self.idle_credits = Button(idle_buttons, text='Credits', width=8,
141 highlightbackground=self.bg,
142 command=self.show_idle_credits)
143 self.idle_credits.pack(side=LEFT, padx=10, pady=10)
144
145 # License, copyright, and credits are of type _sitebuiltins._Printer
146 def show_py_license(self):
147 "Handle License button event."
Guido van Rossum8ce8a782007-11-01 19:42:39 +0000148 self.display_printer_text('About - License', license)
Kurt B. Kaiser6655e4b2002-12-31 16:03:23 +0000149
csabella5a346d52017-05-27 15:49:26 -0400150 def show_py_copyright(self):
151 "Handle Copyright button event."
Guido van Rossum8ce8a782007-11-01 19:42:39 +0000152 self.display_printer_text('About - Copyright', copyright)
Kurt B. Kaiser09cb74b2003-06-12 04:20:56 +0000153
csabella5a346d52017-05-27 15:49:26 -0400154 def show_py_credits(self):
155 "Handle Python Credits button event."
Guido van Rossum8ce8a782007-11-01 19:42:39 +0000156 self.display_printer_text('About - Python Credits', credits)
Kurt B. Kaiser09cb74b2003-06-12 04:20:56 +0000157
Terry Jan Reedy35676512016-01-16 23:44:04 -0500158 # Encode CREDITS.txt to utf-8 for proper version of Loewis.
159 # Specify others as ascii until need utf-8, so catch errors.
csabella5a346d52017-05-27 15:49:26 -0400160 def show_idle_credits(self):
161 "Handle Idle Credits button event."
Terry Jan Reedy35676512016-01-16 23:44:04 -0500162 self.display_file_text('About - Credits', 'CREDITS.txt', 'utf-8')
Steven M. Gava885c0bb2001-07-31 10:44:35 +0000163
csabella5a346d52017-05-27 15:49:26 -0400164 def show_readme(self):
165 "Handle Readme button event."
Terry Jan Reedy35676512016-01-16 23:44:04 -0500166 self.display_file_text('About - Readme', 'README.txt', 'ascii')
Kurt B. Kaiser09cb74b2003-06-12 04:20:56 +0000167
csabella5a346d52017-05-27 15:49:26 -0400168 def show_idle_news(self):
169 "Handle News button event."
Terry Jan Reedy6ff7a142016-06-22 03:55:20 -0400170 self.display_file_text('About - NEWS', 'NEWS.txt', 'utf-8')
Kurt B. Kaiser09cb74b2003-06-12 04:20:56 +0000171
Guido van Rossum8ce8a782007-11-01 19:42:39 +0000172 def display_printer_text(self, title, printer):
csabella5a346d52017-05-27 15:49:26 -0400173 """Create textview for built-in constants.
174
175 Built-in constants have type _sitebuiltins._Printer. The
176 text is extracted from the built-in and then sent to a text
177 viewer with self as the parent and title as the title of
178 the popup.
179 """
Kurt B. Kaiser09cb74b2003-06-12 04:20:56 +0000180 printer._Printer__setup()
Guido van Rossum8ce8a782007-11-01 19:42:39 +0000181 text = '\n'.join(printer._Printer__lines)
mlouielu054e0912017-05-22 06:19:35 +0800182 self._current_textview = textview.view_text(
183 self, title, text, _utest=self._utest)
Kurt B. Kaiser09cb74b2003-06-12 04:20:56 +0000184
Guido van Rossum8ce8a782007-11-01 19:42:39 +0000185 def display_file_text(self, title, filename, encoding=None):
csabella5a346d52017-05-27 15:49:26 -0400186 """Create textview for filename.
187
188 The filename needs to be in the current directory. The path
189 is sent to a text viewer with self as the parent, title as
190 the title of the popup, and the file encoding.
191 """
Guido van Rossum8ce8a782007-11-01 19:42:39 +0000192 fn = os.path.join(os.path.abspath(os.path.dirname(__file__)), filename)
mlouielu054e0912017-05-22 06:19:35 +0800193 self._current_textview = textview.view_file(
194 self, title, fn, encoding, _utest=self._utest)
Steven M. Gava885c0bb2001-07-31 10:44:35 +0000195
csabella5a346d52017-05-27 15:49:26 -0400196 def ok(self, event=None):
197 "Dismiss help_about dialog."
Tal Einat10ea9402018-08-02 09:18:29 +0300198 self.grab_release()
Steven M. Gava885c0bb2001-07-31 10:44:35 +0000199 self.destroy()
Kurt B. Kaiser6655e4b2002-12-31 16:03:23 +0000200
Terry Jan Reedybfbaa6b2016-08-31 00:50:55 -0400201
Steven M. Gava44d3d1a2001-07-31 06:59:02 +0000202if __name__ == '__main__':
Terry Jan Reedyea3dc802018-06-18 04:47:59 -0400203 from unittest import main
204 main('idlelib.idle_test.test_help_about', verbosity=2, exit=False)
205
Terry Jan Reedy06313b72014-05-11 23:32:32 -0400206 from idlelib.idle_test.htest import run
207 run(AboutDialog)