mirror of
				https://github.com/python/cpython.git
				synced 2025-11-04 11:49:12 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			164 lines
		
	
	
	
		
			5.4 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			164 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 seach 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)
 |