| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  | """Line numbering implementation for IDLE as an extension.
 | 
					
						
							|  |  |  | Includes BaseSideBar which can be extended for other sidebar based extensions | 
					
						
							|  |  |  | """
 | 
					
						
							| 
									
										
										
										
											2021-04-29 01:27:55 +03:00
										 |  |  | import contextlib | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  | import functools | 
					
						
							|  |  |  | import itertools | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import tkinter as tk | 
					
						
							| 
									
										
										
										
											2021-04-29 01:27:55 +03:00
										 |  |  | from tkinter.font import Font | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  | from idlelib.config import idleConf | 
					
						
							|  |  |  | from idlelib.delegator import Delegator | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  | from idlelib import macosx | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-29 01:27:55 +03:00
										 |  |  | def get_lineno(text, index): | 
					
						
							|  |  |  |     """Return the line number of an index in a Tk text widget.""" | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |     text_index = text.index(index) | 
					
						
							|  |  |  |     return int(float(text_index)) if text_index else None | 
					
						
							| 
									
										
										
										
											2021-04-29 01:27:55 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  | def get_end_linenumber(text): | 
					
						
							| 
									
										
										
										
											2021-04-29 01:27:55 +03:00
										 |  |  |     """Return the number of the last line in a Tk text widget.""" | 
					
						
							|  |  |  |     return get_lineno(text, 'end-1c') | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-29 01:27:55 +03:00
										 |  |  | def get_displaylines(text, index): | 
					
						
							|  |  |  |     """Display height, in lines, of a logical line in a Tk text widget.""" | 
					
						
							| 
									
										
										
										
											2023-10-24 12:59:19 +03:00
										 |  |  |     return text.count(f"{index} linestart", | 
					
						
							|  |  |  |                       f"{index} lineend", | 
					
						
							|  |  |  |                       "displaylines") | 
					
						
							| 
									
										
										
										
											2021-04-29 01:27:55 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  | def get_widget_padding(widget): | 
					
						
							|  |  |  |     """Get the total padding of a Tk widget, including its border.""" | 
					
						
							|  |  |  |     # TODO: use also in codecontext.py | 
					
						
							|  |  |  |     manager = widget.winfo_manager() | 
					
						
							|  |  |  |     if manager == 'pack': | 
					
						
							|  |  |  |         info = widget.pack_info() | 
					
						
							|  |  |  |     elif manager == 'grid': | 
					
						
							|  |  |  |         info = widget.grid_info() | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         raise ValueError(f"Unsupported geometry manager: {manager}") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # All values are passed through getint(), since some | 
					
						
							|  |  |  |     # values may be pixel objects, which can't simply be added to ints. | 
					
						
							|  |  |  |     padx = sum(map(widget.tk.getint, [ | 
					
						
							|  |  |  |         info['padx'], | 
					
						
							|  |  |  |         widget.cget('padx'), | 
					
						
							|  |  |  |         widget.cget('border'), | 
					
						
							|  |  |  |     ])) | 
					
						
							|  |  |  |     pady = sum(map(widget.tk.getint, [ | 
					
						
							|  |  |  |         info['pady'], | 
					
						
							|  |  |  |         widget.cget('pady'), | 
					
						
							|  |  |  |         widget.cget('border'), | 
					
						
							|  |  |  |     ])) | 
					
						
							|  |  |  |     return padx, pady | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-29 01:27:55 +03:00
										 |  |  | @contextlib.contextmanager | 
					
						
							|  |  |  | def temp_enable_text_widget(text): | 
					
						
							|  |  |  |     text.configure(state=tk.NORMAL) | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         yield | 
					
						
							|  |  |  |     finally: | 
					
						
							|  |  |  |         text.configure(state=tk.DISABLED) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  | class BaseSideBar: | 
					
						
							| 
									
										
										
										
											2021-04-29 01:27:55 +03:00
										 |  |  |     """A base class for sidebars using Text.""" | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  |     def __init__(self, editwin): | 
					
						
							|  |  |  |         self.editwin = editwin | 
					
						
							|  |  |  |         self.parent = editwin.text_frame | 
					
						
							|  |  |  |         self.text = editwin.text | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |         self.is_shown = False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.main_widget = self.init_widgets() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.bind_events() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  |         self.update_font() | 
					
						
							|  |  |  |         self.update_colors() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |     def init_widgets(self): | 
					
						
							|  |  |  |         """Initialize the sidebar's widgets, returning the main widget.""" | 
					
						
							|  |  |  |         raise NotImplementedError | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def update_font(self): | 
					
						
							|  |  |  |         """Update the sidebar text font, usually after config changes.""" | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |         raise NotImplementedError | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def update_colors(self): | 
					
						
							|  |  |  |         """Update the sidebar text colors, usually after config changes.""" | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |         raise NotImplementedError | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |     def grid(self): | 
					
						
							|  |  |  |         """Layout the widget, always using grid layout.""" | 
					
						
							|  |  |  |         raise NotImplementedError | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def show_sidebar(self): | 
					
						
							|  |  |  |         if not self.is_shown: | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |             self.grid() | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  |             self.is_shown = True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def hide_sidebar(self): | 
					
						
							|  |  |  |         if self.is_shown: | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |             self.main_widget.grid_forget() | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  |             self.is_shown = False | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |     def yscroll_event(self, *args, **kwargs): | 
					
						
							|  |  |  |         """Hook for vertical scrolling for sub-classes to override.""" | 
					
						
							|  |  |  |         raise NotImplementedError | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  |     def redirect_yscroll_event(self, *args, **kwargs): | 
					
						
							|  |  |  |         """Redirect vertical scrolling to the main editor text widget.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         The scroll bar is also updated. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         self.editwin.vbar.set(*args) | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |         return self.yscroll_event(*args, **kwargs) | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def redirect_focusin_event(self, event): | 
					
						
							|  |  |  |         """Redirect focus-in events to the main editor text widget.""" | 
					
						
							|  |  |  |         self.text.focus_set() | 
					
						
							|  |  |  |         return 'break' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def redirect_mousebutton_event(self, event, event_name): | 
					
						
							|  |  |  |         """Redirect mouse button events to the main editor text widget.""" | 
					
						
							|  |  |  |         self.text.focus_set() | 
					
						
							|  |  |  |         self.text.event_generate(event_name, x=0, y=event.y) | 
					
						
							|  |  |  |         return 'break' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def redirect_mousewheel_event(self, event): | 
					
						
							|  |  |  |         """Redirect mouse wheel events to the editwin text widget.""" | 
					
						
							|  |  |  |         self.text.event_generate('<MouseWheel>', | 
					
						
							|  |  |  |                                  x=0, y=event.y, delta=event.delta) | 
					
						
							|  |  |  |         return 'break' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def bind_events(self): | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |         self.text['yscrollcommand'] = self.redirect_yscroll_event | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  |         # Ensure focus is always redirected to the main editor text widget. | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |         self.main_widget.bind('<FocusIn>', self.redirect_focusin_event) | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # Redirect mouse scrolling to the main editor text widget. | 
					
						
							|  |  |  |         # | 
					
						
							|  |  |  |         # Note that without this, scrolling with the mouse only scrolls | 
					
						
							|  |  |  |         # the line numbers. | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |         self.main_widget.bind('<MouseWheel>', self.redirect_mousewheel_event) | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # Redirect mouse button events to the main editor text widget, | 
					
						
							|  |  |  |         # except for the left mouse button (1). | 
					
						
							|  |  |  |         # | 
					
						
							|  |  |  |         # Note: X-11 sends Button-4 and Button-5 events for the scroll wheel. | 
					
						
							|  |  |  |         def bind_mouse_event(event_name, target_event_name): | 
					
						
							|  |  |  |             handler = functools.partial(self.redirect_mousebutton_event, | 
					
						
							|  |  |  |                                         event_name=target_event_name) | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |             self.main_widget.bind(event_name, handler) | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         for button in [2, 3, 4, 5]: | 
					
						
							|  |  |  |             for event_name in (f'<Button-{button}>', | 
					
						
							|  |  |  |                                f'<ButtonRelease-{button}>', | 
					
						
							|  |  |  |                                f'<B{button}-Motion>', | 
					
						
							|  |  |  |                                ): | 
					
						
							|  |  |  |                 bind_mouse_event(event_name, target_event_name=event_name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Convert double- and triple-click events to normal click events, | 
					
						
							|  |  |  |             # since event_generate() doesn't allow generating such events. | 
					
						
							|  |  |  |             for event_name in (f'<Double-Button-{button}>', | 
					
						
							|  |  |  |                                f'<Triple-Button-{button}>', | 
					
						
							|  |  |  |                                ): | 
					
						
							|  |  |  |                 bind_mouse_event(event_name, | 
					
						
							|  |  |  |                                  target_event_name=f'<Button-{button}>') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |         # start_line is set upon <Button-1> to allow selecting a range of rows | 
					
						
							|  |  |  |         # by dragging.  It is cleared upon <ButtonRelease-1>. | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  |         start_line = None | 
					
						
							| 
									
										
										
										
											2019-08-04 19:25:27 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |         # last_y is initially set upon <B1-Leave> and is continuously updated | 
					
						
							|  |  |  |         # upon <B1-Motion>, until <B1-Enter> or the mouse button is released. | 
					
						
							|  |  |  |         # It is used in text_auto_scroll(), which is called repeatedly and | 
					
						
							|  |  |  |         # does have a mouse event available. | 
					
						
							|  |  |  |         last_y = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # auto_scrolling_after_id is set whenever text_auto_scroll is | 
					
						
							|  |  |  |         # scheduled via .after().  It is used to stop the auto-scrolling | 
					
						
							|  |  |  |         # upon <B1-Enter>, as well as to avoid scheduling the function several | 
					
						
							|  |  |  |         # times in parallel. | 
					
						
							|  |  |  |         auto_scrolling_after_id = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def drag_update_selection_and_insert_mark(y_coord): | 
					
						
							|  |  |  |             """Helper function for drag and selection event handlers.""" | 
					
						
							|  |  |  |             lineno = get_lineno(self.text, f"@0,{y_coord}") | 
					
						
							|  |  |  |             a, b = sorted([start_line, lineno]) | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  |             self.text.tag_remove("sel", "1.0", "end") | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |             self.text.tag_add("sel", f"{a}.0", f"{b+1}.0") | 
					
						
							|  |  |  |             self.text.mark_set("insert", | 
					
						
							|  |  |  |                                f"{lineno if lineno == a else lineno + 1}.0") | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |         def b1_mousedown_handler(event): | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  |             nonlocal start_line | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |             nonlocal last_y | 
					
						
							|  |  |  |             start_line = int(float(self.text.index(f"@0,{event.y}"))) | 
					
						
							|  |  |  |             last_y = event.y | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             drag_update_selection_and_insert_mark(event.y) | 
					
						
							|  |  |  |         self.main_widget.bind('<Button-1>', b1_mousedown_handler) | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-04 19:25:27 +03:00
										 |  |  |         def b1_mouseup_handler(event): | 
					
						
							|  |  |  |             # On mouse up, we're no longer dragging.  Set the shared persistent | 
					
						
							|  |  |  |             # variables to None to represent this. | 
					
						
							|  |  |  |             nonlocal start_line | 
					
						
							|  |  |  |             nonlocal last_y | 
					
						
							|  |  |  |             start_line = None | 
					
						
							|  |  |  |             last_y = None | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |             self.text.event_generate('<ButtonRelease-1>', x=0, y=event.y) | 
					
						
							|  |  |  |         self.main_widget.bind('<ButtonRelease-1>', b1_mouseup_handler) | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |         def b1_drag_handler(event): | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  |             nonlocal last_y | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |             if last_y is None:  # i.e. if not currently dragging | 
					
						
							|  |  |  |                 return | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  |             last_y = event.y | 
					
						
							|  |  |  |             drag_update_selection_and_insert_mark(event.y) | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |         self.main_widget.bind('<B1-Motion>', b1_drag_handler) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def text_auto_scroll(): | 
					
						
							|  |  |  |             """Mimic Text auto-scrolling when dragging outside of it.""" | 
					
						
							|  |  |  |             # See: https://github.com/tcltk/tk/blob/064ff9941b4b80b85916a8afe86a6c21fd388b54/library/text.tcl#L670 | 
					
						
							|  |  |  |             nonlocal auto_scrolling_after_id | 
					
						
							|  |  |  |             y = last_y | 
					
						
							|  |  |  |             if y is None: | 
					
						
							|  |  |  |                 self.main_widget.after_cancel(auto_scrolling_after_id) | 
					
						
							|  |  |  |                 auto_scrolling_after_id = None | 
					
						
							| 
									
										
										
										
											2019-08-04 19:25:27 +03:00
										 |  |  |                 return | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |             elif y < 0: | 
					
						
							|  |  |  |                 self.text.yview_scroll(-1 + y, 'pixels') | 
					
						
							|  |  |  |                 drag_update_selection_and_insert_mark(y) | 
					
						
							|  |  |  |             elif y > self.main_widget.winfo_height(): | 
					
						
							|  |  |  |                 self.text.yview_scroll(1 + y - self.main_widget.winfo_height(), | 
					
						
							|  |  |  |                                        'pixels') | 
					
						
							|  |  |  |                 drag_update_selection_and_insert_mark(y) | 
					
						
							|  |  |  |             auto_scrolling_after_id = \ | 
					
						
							|  |  |  |                 self.main_widget.after(50, text_auto_scroll) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def b1_leave_handler(event): | 
					
						
							|  |  |  |             # Schedule the initial call to text_auto_scroll(), if not already | 
					
						
							|  |  |  |             # scheduled. | 
					
						
							|  |  |  |             nonlocal auto_scrolling_after_id | 
					
						
							|  |  |  |             if auto_scrolling_after_id is None: | 
					
						
							|  |  |  |                 nonlocal last_y | 
					
						
							|  |  |  |                 last_y = event.y | 
					
						
							|  |  |  |                 auto_scrolling_after_id = \ | 
					
						
							|  |  |  |                     self.main_widget.after(0, text_auto_scroll) | 
					
						
							|  |  |  |         self.main_widget.bind('<B1-Leave>', b1_leave_handler) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def b1_enter_handler(event): | 
					
						
							|  |  |  |             # Cancel the scheduling of text_auto_scroll(), if it exists. | 
					
						
							|  |  |  |             nonlocal auto_scrolling_after_id | 
					
						
							|  |  |  |             if auto_scrolling_after_id is not None: | 
					
						
							|  |  |  |                 self.main_widget.after_cancel(auto_scrolling_after_id) | 
					
						
							|  |  |  |                 auto_scrolling_after_id = None | 
					
						
							|  |  |  |         self.main_widget.bind('<B1-Enter>', b1_enter_handler) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class EndLineDelegator(Delegator): | 
					
						
							|  |  |  |     """Generate callbacks with the current end line number.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     The provided callback is called after every insert and delete. | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     def __init__(self, changed_callback): | 
					
						
							|  |  |  |         Delegator.__init__(self) | 
					
						
							|  |  |  |         self.changed_callback = changed_callback | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def insert(self, index, chars, tags=None): | 
					
						
							|  |  |  |         self.delegate.insert(index, chars, tags) | 
					
						
							|  |  |  |         self.changed_callback(get_end_linenumber(self.delegate)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def delete(self, index1, index2=None): | 
					
						
							|  |  |  |         self.delegate.delete(index1, index2) | 
					
						
							|  |  |  |         self.changed_callback(get_end_linenumber(self.delegate)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class LineNumbers(BaseSideBar): | 
					
						
							|  |  |  |     """Line numbers support for editor windows.""" | 
					
						
							|  |  |  |     def __init__(self, editwin): | 
					
						
							|  |  |  |         super().__init__(editwin) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         end_line_delegator = EndLineDelegator(self.update_sidebar_text) | 
					
						
							|  |  |  |         # Insert the delegator after the undo delegator, so that line numbers | 
					
						
							|  |  |  |         # are properly updated after undo and redo actions. | 
					
						
							|  |  |  |         self.editwin.per.insertfilterafter(end_line_delegator, | 
					
						
							|  |  |  |                                            after=self.editwin.undo) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def init_widgets(self): | 
					
						
							|  |  |  |         _padx, pady = get_widget_padding(self.text) | 
					
						
							|  |  |  |         self.sidebar_text = tk.Text(self.parent, width=1, wrap=tk.NONE, | 
					
						
							|  |  |  |                                     padx=2, pady=pady, | 
					
						
							|  |  |  |                                     borderwidth=0, highlightthickness=0) | 
					
						
							|  |  |  |         self.sidebar_text.config(state=tk.DISABLED) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.prev_end = 1 | 
					
						
							|  |  |  |         self._sidebar_width_type = type(self.sidebar_text['width']) | 
					
						
							|  |  |  |         with temp_enable_text_widget(self.sidebar_text): | 
					
						
							|  |  |  |             self.sidebar_text.insert('insert', '1', 'linenumber') | 
					
						
							|  |  |  |         self.sidebar_text.config(takefocus=False, exportselection=False) | 
					
						
							|  |  |  |         self.sidebar_text.tag_config('linenumber', justify=tk.RIGHT) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         end = get_end_linenumber(self.text) | 
					
						
							|  |  |  |         self.update_sidebar_text(end) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return self.sidebar_text | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def grid(self): | 
					
						
							|  |  |  |         self.sidebar_text.grid(row=1, column=0, sticky=tk.NSEW) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def update_font(self): | 
					
						
							|  |  |  |         font = idleConf.GetFont(self.text, 'main', 'EditorWindow') | 
					
						
							|  |  |  |         self.sidebar_text['font'] = font | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def update_colors(self): | 
					
						
							|  |  |  |         """Update the sidebar text colors, usually after config changes.""" | 
					
						
							|  |  |  |         colors = idleConf.GetHighlight(idleConf.CurrentTheme(), 'linenumber') | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |         foreground = colors['foreground'] | 
					
						
							|  |  |  |         background = colors['background'] | 
					
						
							|  |  |  |         self.sidebar_text.config( | 
					
						
							|  |  |  |             fg=foreground, bg=background, | 
					
						
							|  |  |  |             selectforeground=foreground, selectbackground=background, | 
					
						
							|  |  |  |             inactiveselectbackground=background, | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def update_sidebar_text(self, end): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Perform the following action: | 
					
						
							|  |  |  |         Each line sidebar_text contains the linenumber for that line | 
					
						
							|  |  |  |         Synchronize with editwin.text so that both sidebar_text and | 
					
						
							|  |  |  |         editwin.text contain the same number of lines"""
 | 
					
						
							|  |  |  |         if end == self.prev_end: | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         width_difference = len(str(end)) - len(str(self.prev_end)) | 
					
						
							|  |  |  |         if width_difference: | 
					
						
							|  |  |  |             cur_width = int(float(self.sidebar_text['width'])) | 
					
						
							|  |  |  |             new_width = cur_width + width_difference | 
					
						
							|  |  |  |             self.sidebar_text['width'] = self._sidebar_width_type(new_width) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-29 01:27:55 +03:00
										 |  |  |         with temp_enable_text_widget(self.sidebar_text): | 
					
						
							|  |  |  |             if end > self.prev_end: | 
					
						
							|  |  |  |                 new_text = '\n'.join(itertools.chain( | 
					
						
							|  |  |  |                     [''], | 
					
						
							|  |  |  |                     map(str, range(self.prev_end + 1, end + 1)), | 
					
						
							|  |  |  |                 )) | 
					
						
							|  |  |  |                 self.sidebar_text.insert(f'end -1c', new_text, 'linenumber') | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 self.sidebar_text.delete(f'{end+1}.0 -1c', 'end -1c') | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         self.prev_end = end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |     def yscroll_event(self, *args, **kwargs): | 
					
						
							|  |  |  |         self.sidebar_text.yview_moveto(args[0]) | 
					
						
							|  |  |  |         return 'break' | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-29 01:27:55 +03:00
										 |  |  | class WrappedLineHeightChangeDelegator(Delegator): | 
					
						
							|  |  |  |     def __init__(self, callback): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         callback - Callable, will be called when an insert, delete or replace | 
					
						
							|  |  |  |                    action on the text widget may require updating the shell | 
					
						
							|  |  |  |                    sidebar. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Delegator.__init__(self) | 
					
						
							|  |  |  |         self.callback = callback | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def insert(self, index, chars, tags=None): | 
					
						
							|  |  |  |         is_single_line = '\n' not in chars | 
					
						
							|  |  |  |         if is_single_line: | 
					
						
							|  |  |  |             before_displaylines = get_displaylines(self, index) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.delegate.insert(index, chars, tags) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if is_single_line: | 
					
						
							|  |  |  |             after_displaylines = get_displaylines(self, index) | 
					
						
							|  |  |  |             if after_displaylines == before_displaylines: | 
					
						
							|  |  |  |                 return  # no need to update the sidebar | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.callback() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def delete(self, index1, index2=None): | 
					
						
							|  |  |  |         if index2 is None: | 
					
						
							|  |  |  |             index2 = index1 + "+1c" | 
					
						
							|  |  |  |         is_single_line = get_lineno(self, index1) == get_lineno(self, index2) | 
					
						
							|  |  |  |         if is_single_line: | 
					
						
							|  |  |  |             before_displaylines = get_displaylines(self, index1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.delegate.delete(index1, index2) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if is_single_line: | 
					
						
							|  |  |  |             after_displaylines = get_displaylines(self, index1) | 
					
						
							|  |  |  |             if after_displaylines == before_displaylines: | 
					
						
							|  |  |  |                 return  # no need to update the sidebar | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.callback() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  | class ShellSidebar(BaseSideBar): | 
					
						
							| 
									
										
										
										
											2021-04-29 01:27:55 +03:00
										 |  |  |     """Sidebar for the PyShell window, for prompts etc.""" | 
					
						
							|  |  |  |     def __init__(self, editwin): | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |         self.canvas = None | 
					
						
							|  |  |  |         self.line_prompts = {} | 
					
						
							| 
									
										
										
										
											2021-04-29 01:27:55 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |         super().__init__(editwin) | 
					
						
							| 
									
										
										
										
											2021-04-29 01:27:55 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         change_delegator = \ | 
					
						
							|  |  |  |             WrappedLineHeightChangeDelegator(self.change_callback) | 
					
						
							|  |  |  |         # Insert the TextChangeDelegator after the last delegator, so that | 
					
						
							|  |  |  |         # the sidebar reflects final changes to the text widget contents. | 
					
						
							|  |  |  |         d = self.editwin.per.top | 
					
						
							|  |  |  |         if d.delegate is not self.text: | 
					
						
							|  |  |  |             while d.delegate is not self.editwin.per.bottom: | 
					
						
							|  |  |  |                 d = d.delegate | 
					
						
							|  |  |  |         self.editwin.per.insertfilterafter(change_delegator, after=d) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |         self.is_shown = True | 
					
						
							| 
									
										
										
										
											2021-04-29 01:27:55 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |     def init_widgets(self): | 
					
						
							|  |  |  |         self.canvas = tk.Canvas(self.parent, width=30, | 
					
						
							|  |  |  |                                 borderwidth=0, highlightthickness=0, | 
					
						
							|  |  |  |                                 takefocus=False) | 
					
						
							| 
									
										
										
										
											2021-04-29 01:27:55 +03:00
										 |  |  |         self.update_sidebar() | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |         self.grid() | 
					
						
							|  |  |  |         return self.canvas | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def bind_events(self): | 
					
						
							|  |  |  |         super().bind_events() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.main_widget.bind( | 
					
						
							|  |  |  |             # AquaTk defines <2> as the right button, not <3>. | 
					
						
							|  |  |  |             "<Button-2>" if macosx.isAquaTk() else "<Button-3>", | 
					
						
							|  |  |  |             self.context_menu_event, | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def context_menu_event(self, event): | 
					
						
							|  |  |  |         rmenu = tk.Menu(self.main_widget, tearoff=0) | 
					
						
							|  |  |  |         has_selection = bool(self.text.tag_nextrange('sel', '1.0')) | 
					
						
							|  |  |  |         def mkcmd(eventname): | 
					
						
							|  |  |  |             return lambda: self.text.event_generate(eventname) | 
					
						
							|  |  |  |         rmenu.add_command(label='Copy', | 
					
						
							|  |  |  |                           command=mkcmd('<<copy>>'), | 
					
						
							|  |  |  |                           state='normal' if has_selection else 'disabled') | 
					
						
							|  |  |  |         rmenu.add_command(label='Copy with prompts', | 
					
						
							|  |  |  |                           command=mkcmd('<<copy-with-prompts>>'), | 
					
						
							|  |  |  |                           state='normal' if has_selection else 'disabled') | 
					
						
							|  |  |  |         rmenu.tk_popup(event.x_root, event.y_root) | 
					
						
							|  |  |  |         return "break" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def grid(self): | 
					
						
							| 
									
										
										
										
											2021-04-29 01:27:55 +03:00
										 |  |  |         self.canvas.grid(row=1, column=0, sticky=tk.NSEW, padx=2, pady=0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def change_callback(self): | 
					
						
							|  |  |  |         if self.is_shown: | 
					
						
							|  |  |  |             self.update_sidebar() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def update_sidebar(self): | 
					
						
							|  |  |  |         text = self.text | 
					
						
							|  |  |  |         text_tagnames = text.tag_names | 
					
						
							|  |  |  |         canvas = self.canvas | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |         line_prompts = self.line_prompts = {} | 
					
						
							| 
									
										
										
										
											2021-04-29 01:27:55 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         canvas.delete(tk.ALL) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         index = text.index("@0,0") | 
					
						
							|  |  |  |         if index.split('.', 1)[1] != '0': | 
					
						
							|  |  |  |             index = text.index(f'{index}+1line linestart') | 
					
						
							| 
									
										
										
										
											2022-02-02 19:59:24 -06:00
										 |  |  |         while (lineinfo := text.dlineinfo(index)) is not None: | 
					
						
							| 
									
										
										
										
											2021-04-29 01:27:55 +03:00
										 |  |  |             y = lineinfo[1] | 
					
						
							|  |  |  |             prev_newline_tagnames = text_tagnames(f"{index} linestart -1c") | 
					
						
							|  |  |  |             prompt = ( | 
					
						
							|  |  |  |                 '>>>' if "console" in prev_newline_tagnames else | 
					
						
							|  |  |  |                 '...' if "stdin" in prev_newline_tagnames else | 
					
						
							|  |  |  |                 None | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |             if prompt: | 
					
						
							|  |  |  |                 canvas.create_text(2, y, anchor=tk.NW, text=prompt, | 
					
						
							|  |  |  |                                    font=self.font, fill=self.colors[0]) | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |                 lineno = get_lineno(text, index) | 
					
						
							|  |  |  |                 line_prompts[lineno] = prompt | 
					
						
							| 
									
										
										
										
											2021-04-29 01:27:55 +03:00
										 |  |  |             index = text.index(f'{index}+1line') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def yscroll_event(self, *args, **kwargs): | 
					
						
							|  |  |  |         """Redirect vertical scrolling to the main editor text widget.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         The scroll bar is also updated. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         self.change_callback() | 
					
						
							|  |  |  |         return 'break' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def update_font(self): | 
					
						
							|  |  |  |         """Update the sidebar text font, usually after config changes.""" | 
					
						
							|  |  |  |         font = idleConf.GetFont(self.text, 'main', 'EditorWindow') | 
					
						
							|  |  |  |         tk_font = Font(self.text, font=font) | 
					
						
							|  |  |  |         char_width = max(tk_font.measure(char) for char in ['>', '.']) | 
					
						
							|  |  |  |         self.canvas.configure(width=char_width * 3 + 4) | 
					
						
							|  |  |  |         self.font = font | 
					
						
							|  |  |  |         self.change_callback() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def update_colors(self): | 
					
						
							|  |  |  |         """Update the sidebar text colors, usually after config changes.""" | 
					
						
							|  |  |  |         linenumbers_colors = idleConf.GetHighlight(idleConf.CurrentTheme(), 'linenumber') | 
					
						
							|  |  |  |         prompt_colors = idleConf.GetHighlight(idleConf.CurrentTheme(), 'console') | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |         foreground = prompt_colors['foreground'] | 
					
						
							|  |  |  |         background = linenumbers_colors['background'] | 
					
						
							| 
									
										
										
										
											2021-04-29 01:27:55 +03:00
										 |  |  |         self.colors = (foreground, background) | 
					
						
							| 
									
										
										
										
											2021-05-03 05:27:38 +03:00
										 |  |  |         self.canvas.configure(background=background) | 
					
						
							| 
									
										
										
										
											2021-04-29 01:27:55 +03:00
										 |  |  |         self.change_callback() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  | def _linenumbers_drag_scrolling(parent):  # htest # | 
					
						
							|  |  |  |     from idlelib.idle_test.test_sidebar import Dummy_editwin | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-03 04:28:37 -05:00
										 |  |  |     top = tk.Toplevel(parent) | 
					
						
							|  |  |  |     text_frame = tk.Frame(top) | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  |     text_frame.pack(side=tk.LEFT, fill=tk.BOTH, expand=True) | 
					
						
							|  |  |  |     text_frame.rowconfigure(1, weight=1) | 
					
						
							|  |  |  |     text_frame.columnconfigure(1, weight=1) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-03 04:28:37 -05:00
										 |  |  |     font = idleConf.GetFont(top, 'main', 'EditorWindow') | 
					
						
							| 
									
										
										
										
											2019-07-23 15:22:11 +03:00
										 |  |  |     text = tk.Text(text_frame, width=80, height=24, wrap=tk.NONE, font=font) | 
					
						
							|  |  |  |     text.grid(row=1, column=1, sticky=tk.NSEW) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     editwin = Dummy_editwin(text) | 
					
						
							|  |  |  |     editwin.vbar = tk.Scrollbar(text_frame) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     linenumbers = LineNumbers(editwin) | 
					
						
							|  |  |  |     linenumbers.show_sidebar() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     text.insert('1.0', '\n'.join('a'*i for i in range(1, 101))) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if __name__ == '__main__': | 
					
						
							|  |  |  |     from unittest import main | 
					
						
							|  |  |  |     main('idlelib.idle_test.test_sidebar', verbosity=2, exit=False) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     from idlelib.idle_test.htest import run | 
					
						
							|  |  |  |     run(_linenumbers_drag_scrolling) |