123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164 |
- """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)
|