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