165 lines
5.4 KiB
165 lines
5.4 KiB
3 years ago
|
"""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)
|