| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  | """ParenMatch -- An IDLE extension for parenthesis matching.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | When you hit a right paren, the cursor should move briefly to the left | 
					
						
							|  |  |  | paren.  Paren here is used generically; the matching applies to | 
					
						
							|  |  |  | parentheses, square brackets, and curly braces. | 
					
						
							|  |  |  | """
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-11-18 22:05:48 +00:00
										 |  |  | from HyperParser import HyperParser | 
					
						
							| 
									
										
										
										
											2002-09-14 02:46:19 +00:00
										 |  |  | from configHandler import idleConf | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-07-20 22:22:52 +00:00
										 |  |  | _openers = {')':'(',']':'[','}':'{'} | 
					
						
							| 
									
										
										
										
											2005-11-18 22:05:48 +00:00
										 |  |  | CHECK_DELAY = 100 # miliseconds | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  | class ParenMatch: | 
					
						
							|  |  |  |     """Highlight matching parentheses
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     There are three supported style of paren matching, based loosely | 
					
						
							| 
									
										
										
										
											2001-07-13 19:49:27 +00:00
										 |  |  |     on the Emacs options.  The style is select based on the | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  |     HILITE_STYLE attribute; it can be changed used the set_style | 
					
						
							|  |  |  |     method. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     The supported styles are: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     default -- When a right paren is typed, highlight the matching | 
					
						
							|  |  |  |         left paren for 1/2 sec. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     expression -- When a right paren is typed, highlight the entire | 
					
						
							|  |  |  |         expression from the left paren to the right paren. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     TODO: | 
					
						
							|  |  |  |         - extend IDLE with configuration dialog to change options | 
					
						
							|  |  |  |         - implement rest of Emacs highlight styles (see below) | 
					
						
							|  |  |  |         - print mismatch warning in IDLE status window | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Note: In Emacs, there are several styles of highlight where the | 
					
						
							|  |  |  |     matching paren is highlighted whenever the cursor is immediately | 
					
						
							|  |  |  |     to the right of a right paren.  I don't know how to do that in Tk, | 
					
						
							|  |  |  |     so I haven't bothered. | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2005-11-18 22:05:48 +00:00
										 |  |  |     menudefs = [ | 
					
						
							|  |  |  |         ('edit', [ | 
					
						
							|  |  |  |             ("Show surrounding parens", "<<flash-paren>>"), | 
					
						
							|  |  |  |         ]) | 
					
						
							|  |  |  |     ] | 
					
						
							| 
									
										
										
										
											2002-09-14 02:46:19 +00:00
										 |  |  |     STYLE = idleConf.GetOption('extensions','ParenMatch','style', | 
					
						
							|  |  |  |             default='expression') | 
					
						
							|  |  |  |     FLASH_DELAY = idleConf.GetOption('extensions','ParenMatch','flash-delay', | 
					
						
							|  |  |  |             type='int',default=500) | 
					
						
							|  |  |  |     HILITE_CONFIG = idleConf.GetHighlight(idleConf.CurrentTheme(),'hilite') | 
					
						
							|  |  |  |     BELL = idleConf.GetOption('extensions','ParenMatch','bell', | 
					
						
							|  |  |  |             type='bool',default=1) | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-11-18 22:05:48 +00:00
										 |  |  |     RESTORE_VIRTUAL_EVENT_NAME = "<<parenmatch-check-restore>>" | 
					
						
							|  |  |  |     # We want the restore event be called before the usual return and | 
					
						
							|  |  |  |     # backspace events. | 
					
						
							|  |  |  |     RESTORE_SEQUENCES = ("<KeyPress>", "<ButtonPress>", | 
					
						
							|  |  |  |                          "<Key-Return>", "<Key-BackSpace>") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  |     def __init__(self, editwin): | 
					
						
							|  |  |  |         self.editwin = editwin | 
					
						
							|  |  |  |         self.text = editwin.text | 
					
						
							| 
									
										
										
										
											2005-11-18 22:05:48 +00:00
										 |  |  |         # Bind the check-restore event to the function restore_event, | 
					
						
							|  |  |  |         # so that we can then use activate_restore (which calls event_add) | 
					
						
							|  |  |  |         # and deactivate_restore (which calls event_delete). | 
					
						
							|  |  |  |         editwin.text.bind(self.RESTORE_VIRTUAL_EVENT_NAME, | 
					
						
							|  |  |  |                           self.restore_event) | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  |         self.counter = 0 | 
					
						
							| 
									
										
										
										
											2005-11-18 22:05:48 +00:00
										 |  |  |         self.is_restore_active = 0 | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  |         self.set_style(self.STYLE) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-11-18 22:05:48 +00:00
										 |  |  |     def activate_restore(self): | 
					
						
							|  |  |  |         if not self.is_restore_active: | 
					
						
							|  |  |  |             for seq in self.RESTORE_SEQUENCES: | 
					
						
							|  |  |  |                 self.text.event_add(self.RESTORE_VIRTUAL_EVENT_NAME, seq) | 
					
						
							|  |  |  |             self.is_restore_active = True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def deactivate_restore(self): | 
					
						
							|  |  |  |         if self.is_restore_active: | 
					
						
							|  |  |  |             for seq in self.RESTORE_SEQUENCES: | 
					
						
							|  |  |  |                 self.text.event_delete(self.RESTORE_VIRTUAL_EVENT_NAME, seq) | 
					
						
							|  |  |  |             self.is_restore_active = False | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  |     def set_style(self, style): | 
					
						
							|  |  |  |         self.STYLE = style | 
					
						
							|  |  |  |         if style == "default": | 
					
						
							|  |  |  |             self.create_tag = self.create_tag_default | 
					
						
							|  |  |  |             self.set_timeout = self.set_timeout_last | 
					
						
							|  |  |  |         elif style == "expression": | 
					
						
							|  |  |  |             self.create_tag = self.create_tag_expression | 
					
						
							|  |  |  |             self.set_timeout = self.set_timeout_none | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-11-18 22:05:48 +00:00
										 |  |  |     def flash_paren_event(self, event): | 
					
						
							|  |  |  |         indices = HyperParser(self.editwin, "insert").get_surrounding_brackets() | 
					
						
							|  |  |  |         if indices is None: | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  |             self.warn_mismatched() | 
					
						
							|  |  |  |             return | 
					
						
							| 
									
										
										
										
											2005-11-18 22:05:48 +00:00
										 |  |  |         self.activate_restore() | 
					
						
							|  |  |  |         self.create_tag(indices) | 
					
						
							|  |  |  |         self.set_timeout_last() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def paren_closed_event(self, event): | 
					
						
							|  |  |  |         # If it was a shortcut and not really a closing paren, quit. | 
					
						
							| 
									
										
										
										
											2006-07-20 22:22:52 +00:00
										 |  |  |         closer = self.text.get("insert-1c") | 
					
						
							|  |  |  |         if closer not in _openers: | 
					
						
							| 
									
										
										
										
											2005-11-18 22:05:48 +00:00
										 |  |  |             return | 
					
						
							|  |  |  |         hp = HyperParser(self.editwin, "insert-1c") | 
					
						
							|  |  |  |         if not hp.is_in_code(): | 
					
						
							|  |  |  |             return | 
					
						
							| 
									
										
										
										
											2006-07-20 22:22:52 +00:00
										 |  |  |         indices = hp.get_surrounding_brackets(_openers[closer], True) | 
					
						
							| 
									
										
										
										
											2005-11-18 22:05:48 +00:00
										 |  |  |         if indices is None: | 
					
						
							|  |  |  |             self.warn_mismatched() | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         self.activate_restore() | 
					
						
							|  |  |  |         self.create_tag(indices) | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  |         self.set_timeout() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-11-18 22:05:48 +00:00
										 |  |  |     def restore_event(self, event=None): | 
					
						
							|  |  |  |         self.text.tag_delete("paren") | 
					
						
							|  |  |  |         self.deactivate_restore() | 
					
						
							|  |  |  |         self.counter += 1   # disable the last timer, if there is one. | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def handle_restore_timer(self, timer_count): | 
					
						
							| 
									
										
										
										
											2005-11-18 22:05:48 +00:00
										 |  |  |         if timer_count == self.counter: | 
					
						
							|  |  |  |             self.restore_event() | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def warn_mismatched(self): | 
					
						
							|  |  |  |         if self.BELL: | 
					
						
							|  |  |  |             self.text.bell() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # any one of the create_tag_XXX methods can be used depending on | 
					
						
							|  |  |  |     # the style | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-11-18 22:05:48 +00:00
										 |  |  |     def create_tag_default(self, indices): | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  |         """Highlight the single paren that matches""" | 
					
						
							| 
									
										
										
										
											2005-11-18 22:05:48 +00:00
										 |  |  |         self.text.tag_add("paren", indices[0]) | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  |         self.text.tag_config("paren", self.HILITE_CONFIG) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-11-18 22:05:48 +00:00
										 |  |  |     def create_tag_expression(self, indices): | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  |         """Highlight the entire expression""" | 
					
						
							| 
									
										
										
										
											2005-11-18 22:05:48 +00:00
										 |  |  |         if self.text.get(indices[1]) in (')', ']', '}'): | 
					
						
							|  |  |  |             rightindex = indices[1]+"+1c" | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             rightindex = indices[1] | 
					
						
							|  |  |  |         self.text.tag_add("paren", indices[0], rightindex) | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  |         self.text.tag_config("paren", self.HILITE_CONFIG) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # any one of the set_timeout_XXX methods can be used depending on | 
					
						
							|  |  |  |     # the style | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def set_timeout_none(self): | 
					
						
							| 
									
										
										
										
											2005-11-18 22:05:48 +00:00
										 |  |  |         """Highlight will remain until user input turns it off
 | 
					
						
							|  |  |  |         or the insert has moved"""
 | 
					
						
							|  |  |  |         # After CHECK_DELAY, call a function which disables the "paren" tag | 
					
						
							|  |  |  |         # if the event is for the most recent timer and the insert has changed, | 
					
						
							|  |  |  |         # or schedules another call for itself. | 
					
						
							|  |  |  |         self.counter += 1 | 
					
						
							|  |  |  |         def callme(callme, self=self, c=self.counter, | 
					
						
							|  |  |  |                    index=self.text.index("insert")): | 
					
						
							|  |  |  |             if index != self.text.index("insert"): | 
					
						
							|  |  |  |                 self.handle_restore_timer(c) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 self.editwin.text_frame.after(CHECK_DELAY, callme, callme) | 
					
						
							|  |  |  |         self.editwin.text_frame.after(CHECK_DELAY, callme, callme) | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def set_timeout_last(self): | 
					
						
							|  |  |  |         """The last highlight created will be removed after .5 sec""" | 
					
						
							|  |  |  |         # associate a counter with an event; only disable the "paren" | 
					
						
							|  |  |  |         # tag if the event is for the most recent timer. | 
					
						
							| 
									
										
										
										
											2005-11-18 22:05:48 +00:00
										 |  |  |         self.counter += 1 | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  |         self.editwin.text_frame.after(self.FLASH_DELAY, | 
					
						
							|  |  |  |                                       lambda self=self, c=self.counter: \ | 
					
						
							|  |  |  |                                       self.handle_restore_timer(c)) |