165 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			165 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| """Search dialog for Find, Find Again, and Find Selection
 | |
|    functionality.
 | |
| 
 | |
|    Inherits from SearchDialogBase for GUI and uses searchengine
 | |
|    to prepare search pattern.
 | |
| """
 | |
| from tkinter import TclError
 | |
| 
 | |
| from idlelib import searchengine
 | |
| from idlelib.searchbase import SearchDialogBase
 | |
| 
 | |
| def _setup(text):
 | |
|     """Return the new or existing singleton SearchDialog instance.
 | |
| 
 | |
|     The singleton dialog saves user entries and preferences
 | |
|     across instances.
 | |
| 
 | |
|     Args:
 | |
|         text: Text widget containing the text to be searched.
 | |
|     """
 | |
|     root = text._root()
 | |
|     engine = searchengine.get(root)
 | |
|     if not hasattr(engine, "_searchdialog"):
 | |
|         engine._searchdialog = SearchDialog(root, engine)
 | |
|     return engine._searchdialog
 | |
| 
 | |
| def find(text):
 | |
|     """Open the search dialog.
 | |
| 
 | |
|     Module-level function to access the singleton SearchDialog
 | |
|     instance and open the dialog.  If text is selected, it is
 | |
|     used as the search phrase; otherwise, the previous entry
 | |
|     is used.  No search is done with this command.
 | |
|     """
 | |
|     pat = text.get("sel.first", "sel.last")
 | |
|     return _setup(text).open(text, pat)  # Open is inherited from SDBase.
 | |
| 
 | |
| def find_again(text):
 | |
|     """Repeat the search for the last pattern and preferences.
 | |
| 
 | |
|     Module-level function to access the singleton SearchDialog
 | |
|     instance to search again using the user entries and preferences
 | |
|     from the last dialog.  If there was no prior search, open the
 | |
|     search dialog; otherwise, perform the search without showing the
 | |
|     dialog.
 | |
|     """
 | |
|     return _setup(text).find_again(text)
 | |
| 
 | |
| def find_selection(text):
 | |
|     """Search for the selected pattern in the text.
 | |
| 
 | |
|     Module-level function to access the singleton SearchDialog
 | |
|     instance to search using the selected text.  With a text
 | |
|     selection, perform the search without displaying the dialog.
 | |
|     Without a selection, use the prior entry as the search phrase
 | |
|     and don't display the dialog.  If there has been no prior
 | |
|     search, open the search dialog.
 | |
|     """
 | |
|     return _setup(text).find_selection(text)
 | |
| 
 | |
| 
 | |
| class SearchDialog(SearchDialogBase):
 | |
|     "Dialog for finding a pattern in text."
 | |
| 
 | |
|     def create_widgets(self):
 | |
|         "Create the base search dialog and add a button for Find Next."
 | |
|         SearchDialogBase.create_widgets(self)
 | |
|         # TODO - why is this here and not in a create_command_buttons?
 | |
|         self.make_button("Find Next", self.default_command, isdef=True)
 | |
| 
 | |
|     def default_command(self, event=None):
 | |
|         "Handle the Find Next button as the default command."
 | |
|         if not self.engine.getprog():
 | |
|             return
 | |
|         self.find_again(self.text)
 | |
| 
 | |
|     def find_again(self, text):
 | |
|         """Repeat the last search.
 | |
| 
 | |
|         If no search was previously run, open a new search dialog.  In
 | |
|         this case, no search is done.
 | |
| 
 | |
|         If a search was previously run, the search dialog won't be
 | |
|         shown and the options from the previous search (including the
 | |
|         search pattern) will be used to find the next occurrence
 | |
|         of the pattern.  Next is relative based on direction.
 | |
| 
 | |
|         Position the window to display the located occurrence in the
 | |
|         text.
 | |
| 
 | |
|         Return True if the search was successful and False otherwise.
 | |
|         """
 | |
|         if not self.engine.getpat():
 | |
|             self.open(text)
 | |
|             return False
 | |
|         if not self.engine.getprog():
 | |
|             return False
 | |
|         res = self.engine.search_text(text)
 | |
|         if res:
 | |
|             line, m = res
 | |
|             i, j = m.span()
 | |
|             first = "%d.%d" % (line, i)
 | |
|             last = "%d.%d" % (line, j)
 | |
|             try:
 | |
|                 selfirst = text.index("sel.first")
 | |
|                 sellast = text.index("sel.last")
 | |
|                 if selfirst == first and sellast == last:
 | |
|                     self.bell()
 | |
|                     return False
 | |
|             except TclError:
 | |
|                 pass
 | |
|             text.tag_remove("sel", "1.0", "end")
 | |
|             text.tag_add("sel", first, last)
 | |
|             text.mark_set("insert", self.engine.isback() and first or last)
 | |
|             text.see("insert")
 | |
|             return True
 | |
|         else:
 | |
|             self.bell()
 | |
|             return False
 | |
| 
 | |
|     def find_selection(self, text):
 | |
|         """Search for selected text with previous dialog preferences.
 | |
| 
 | |
|         Instead of using the same pattern for searching (as Find
 | |
|         Again does), this first resets the pattern to the currently
 | |
|         selected text.  If the selected text isn't changed, then use
 | |
|         the prior search phrase.
 | |
|         """
 | |
|         pat = text.get("sel.first", "sel.last")
 | |
|         if pat:
 | |
|             self.engine.setcookedpat(pat)
 | |
|         return self.find_again(text)
 | |
| 
 | |
| 
 | |
| def _search_dialog(parent):  # htest #
 | |
|     "Display search test box."
 | |
|     from tkinter import Toplevel, Text
 | |
|     from tkinter.ttk import Frame, Button
 | |
| 
 | |
|     top = Toplevel(parent)
 | |
|     top.title("Test SearchDialog")
 | |
|     x, y = map(int, parent.geometry().split('+')[1:])
 | |
|     top.geometry("+%d+%d" % (x, y + 175))
 | |
| 
 | |
|     frame = Frame(top)
 | |
|     frame.pack()
 | |
|     text = Text(frame, inactiveselectbackground='gray')
 | |
|     text.pack()
 | |
|     text.insert("insert","This is a sample string.\n"*5)
 | |
| 
 | |
|     def show_find():
 | |
|         text.tag_add('sel', '1.0', 'end')
 | |
|         _setup(text).open(text)
 | |
|         text.tag_remove('sel', '1.0', 'end')
 | |
| 
 | |
|     button = Button(frame, text="Search (selection ignored)", command=show_find)
 | |
|     button.pack()
 | |
| 
 | |
| if __name__ == '__main__':
 | |
|     from unittest import main
 | |
|     main('idlelib.idle_test.test_search', verbosity=2, exit=False)
 | |
| 
 | |
|     from idlelib.idle_test.htest import run
 | |
|     run(_search_dialog)
 |