help_about.py 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211
  1. """About Dialog for IDLE
  2. """
  3. import os
  4. import sys
  5. import webbrowser
  6. from platform import python_version, architecture
  7. from tkinter import Toplevel, Frame, Label, Button, PhotoImage
  8. from tkinter import SUNKEN, TOP, BOTTOM, LEFT, X, BOTH, W, EW, NSEW, E
  9. from idlelib import textview
  10. version = python_version()
  11. def build_bits():
  12. "Return bits for platform."
  13. if sys.platform == 'darwin':
  14. return '64' if sys.maxsize > 2**32 else '32'
  15. else:
  16. return architecture()[0][:2]
  17. class AboutDialog(Toplevel):
  18. """Modal about dialog for idle
  19. """
  20. def __init__(self, parent, title=None, *, _htest=False, _utest=False):
  21. """Create popup, do not return until tk widget destroyed.
  22. parent - parent of this dialog
  23. title - string which is title of popup dialog
  24. _htest - bool, change box location when running htest
  25. _utest - bool, don't wait_window when running unittest
  26. """
  27. Toplevel.__init__(self, parent)
  28. self.configure(borderwidth=5)
  29. # place dialog below parent if running htest
  30. self.geometry("+%d+%d" % (
  31. parent.winfo_rootx()+30,
  32. parent.winfo_rooty()+(30 if not _htest else 100)))
  33. self.bg = "#bbbbbb"
  34. self.fg = "#000000"
  35. self.create_widgets()
  36. self.resizable(height=False, width=False)
  37. self.title(title or
  38. f'About IDLE {version} ({build_bits()} bit)')
  39. self.transient(parent)
  40. self.grab_set()
  41. self.protocol("WM_DELETE_WINDOW", self.ok)
  42. self.parent = parent
  43. self.button_ok.focus_set()
  44. self.bind('<Return>', self.ok) # dismiss dialog
  45. self.bind('<Escape>', self.ok) # dismiss dialog
  46. self._current_textview = None
  47. self._utest = _utest
  48. if not _utest:
  49. self.deiconify()
  50. self.wait_window()
  51. def create_widgets(self):
  52. frame = Frame(self, borderwidth=2, relief=SUNKEN)
  53. frame_buttons = Frame(self)
  54. frame_buttons.pack(side=BOTTOM, fill=X)
  55. frame.pack(side=TOP, expand=True, fill=BOTH)
  56. self.button_ok = Button(frame_buttons, text='Close',
  57. command=self.ok)
  58. self.button_ok.pack(padx=5, pady=5)
  59. frame_background = Frame(frame, bg=self.bg)
  60. frame_background.pack(expand=True, fill=BOTH)
  61. header = Label(frame_background, text='IDLE', fg=self.fg,
  62. bg=self.bg, font=('courier', 24, 'bold'))
  63. header.grid(row=0, column=0, sticky=E, padx=10, pady=10)
  64. tk_patchlevel = self.tk.call('info', 'patchlevel')
  65. ext = '.png' if tk_patchlevel >= '8.6' else '.gif'
  66. icon = os.path.join(os.path.abspath(os.path.dirname(__file__)),
  67. 'Icons', f'idle_48{ext}')
  68. self.icon_image = PhotoImage(master=self._root(), file=icon)
  69. logo = Label(frame_background, image=self.icon_image, bg=self.bg)
  70. logo.grid(row=0, column=0, sticky=W, rowspan=2, padx=10, pady=10)
  71. byline_text = "Python's Integrated Development\nand Learning Environment" + 5*'\n'
  72. byline = Label(frame_background, text=byline_text, justify=LEFT,
  73. fg=self.fg, bg=self.bg)
  74. byline.grid(row=2, column=0, sticky=W, columnspan=3, padx=10, pady=5)
  75. email = Label(frame_background, text='email: idle-dev@python.org',
  76. justify=LEFT, fg=self.fg, bg=self.bg)
  77. email.grid(row=6, column=0, columnspan=2, sticky=W, padx=10, pady=0)
  78. docs = Label(frame_background, text="https://docs.python.org/"
  79. f"{version[:version.rindex('.')]}/library/idle.html",
  80. justify=LEFT, fg=self.fg, bg=self.bg)
  81. docs.grid(row=7, column=0, columnspan=2, sticky=W, padx=10, pady=0)
  82. docs.bind("<Button-1>", lambda event: webbrowser.open(docs['text']))
  83. Frame(frame_background, borderwidth=1, relief=SUNKEN,
  84. height=2, bg=self.bg).grid(row=8, column=0, sticky=EW,
  85. columnspan=3, padx=5, pady=5)
  86. pyver = Label(frame_background,
  87. text='Python version: ' + version,
  88. fg=self.fg, bg=self.bg)
  89. pyver.grid(row=9, column=0, sticky=W, padx=10, pady=0)
  90. tkver = Label(frame_background, text='Tk version: ' + tk_patchlevel,
  91. fg=self.fg, bg=self.bg)
  92. tkver.grid(row=9, column=1, sticky=W, padx=2, pady=0)
  93. py_buttons = Frame(frame_background, bg=self.bg)
  94. py_buttons.grid(row=10, column=0, columnspan=2, sticky=NSEW)
  95. self.py_license = Button(py_buttons, text='License', width=8,
  96. highlightbackground=self.bg,
  97. command=self.show_py_license)
  98. self.py_license.pack(side=LEFT, padx=10, pady=10)
  99. self.py_copyright = Button(py_buttons, text='Copyright', width=8,
  100. highlightbackground=self.bg,
  101. command=self.show_py_copyright)
  102. self.py_copyright.pack(side=LEFT, padx=10, pady=10)
  103. self.py_credits = Button(py_buttons, text='Credits', width=8,
  104. highlightbackground=self.bg,
  105. command=self.show_py_credits)
  106. self.py_credits.pack(side=LEFT, padx=10, pady=10)
  107. Frame(frame_background, borderwidth=1, relief=SUNKEN,
  108. height=2, bg=self.bg).grid(row=11, column=0, sticky=EW,
  109. columnspan=3, padx=5, pady=5)
  110. idlever = Label(frame_background,
  111. text='IDLE version: ' + version,
  112. fg=self.fg, bg=self.bg)
  113. idlever.grid(row=12, column=0, sticky=W, padx=10, pady=0)
  114. idle_buttons = Frame(frame_background, bg=self.bg)
  115. idle_buttons.grid(row=13, column=0, columnspan=3, sticky=NSEW)
  116. self.readme = Button(idle_buttons, text='README', width=8,
  117. highlightbackground=self.bg,
  118. command=self.show_readme)
  119. self.readme.pack(side=LEFT, padx=10, pady=10)
  120. self.idle_news = Button(idle_buttons, text='NEWS', width=8,
  121. highlightbackground=self.bg,
  122. command=self.show_idle_news)
  123. self.idle_news.pack(side=LEFT, padx=10, pady=10)
  124. self.idle_credits = Button(idle_buttons, text='Credits', width=8,
  125. highlightbackground=self.bg,
  126. command=self.show_idle_credits)
  127. self.idle_credits.pack(side=LEFT, padx=10, pady=10)
  128. # License, copyright, and credits are of type _sitebuiltins._Printer
  129. def show_py_license(self):
  130. "Handle License button event."
  131. self.display_printer_text('About - License', license)
  132. def show_py_copyright(self):
  133. "Handle Copyright button event."
  134. self.display_printer_text('About - Copyright', copyright)
  135. def show_py_credits(self):
  136. "Handle Python Credits button event."
  137. self.display_printer_text('About - Python Credits', credits)
  138. # Encode CREDITS.txt to utf-8 for proper version of Loewis.
  139. # Specify others as ascii until need utf-8, so catch errors.
  140. def show_idle_credits(self):
  141. "Handle Idle Credits button event."
  142. self.display_file_text('About - Credits', 'CREDITS.txt', 'utf-8')
  143. def show_readme(self):
  144. "Handle Readme button event."
  145. self.display_file_text('About - Readme', 'README.txt', 'ascii')
  146. def show_idle_news(self):
  147. "Handle News button event."
  148. self.display_file_text('About - NEWS', 'NEWS.txt', 'utf-8')
  149. def display_printer_text(self, title, printer):
  150. """Create textview for built-in constants.
  151. Built-in constants have type _sitebuiltins._Printer. The
  152. text is extracted from the built-in and then sent to a text
  153. viewer with self as the parent and title as the title of
  154. the popup.
  155. """
  156. printer._Printer__setup()
  157. text = '\n'.join(printer._Printer__lines)
  158. self._current_textview = textview.view_text(
  159. self, title, text, _utest=self._utest)
  160. def display_file_text(self, title, filename, encoding=None):
  161. """Create textview for filename.
  162. The filename needs to be in the current directory. The path
  163. is sent to a text viewer with self as the parent, title as
  164. the title of the popup, and the file encoding.
  165. """
  166. fn = os.path.join(os.path.abspath(os.path.dirname(__file__)), filename)
  167. self._current_textview = textview.view_file(
  168. self, title, fn, encoding, _utest=self._utest)
  169. def ok(self, event=None):
  170. "Dismiss help_about dialog."
  171. self.grab_release()
  172. self.destroy()
  173. if __name__ == '__main__':
  174. from unittest import main
  175. main('idlelib.idle_test.test_help_about', verbosity=2, exit=False)
  176. from idlelib.idle_test.htest import run
  177. run(AboutDialog)