"""Replace dialog for IDLE. Inherits SearchDialogBase for GUI. Uses idlelib.SearchEngine for search capability. Defines various replace related functions like replace, replace all, replace+find. """ import re from tkinter import StringVar, TclError from idlelib.searchbase import SearchDialogBase from idlelib import searchengine def replace(text): """Returns a singleton ReplaceDialog instance.The single dialog saves user entries and preferences across instances.""" root = text._root() engine = searchengine.get(root) if not hasattr(engine, "_replacedialog"): engine._replacedialog = ReplaceDialog(root, engine) dialog = engine._replacedialog dialog.open(text) class ReplaceDialog(SearchDialogBase): title = "Replace Dialog" icon = "Replace" def __init__(self, root, engine): SearchDialogBase.__init__(self, root, engine) self.replvar = StringVar(root) def open(self, text): """Display the replace dialog""" SearchDialogBase.open(self, text) try: first = text.index("sel.first") except TclError: first = None try: last = text.index("sel.last") except TclError: last = None first = first or text.index("insert") last = last or first self.show_hit(first, last) self.ok = 1 def create_entries(self): """Create label and text entry widgets""" SearchDialogBase.create_entries(self) self.replent = self.make_entry("Replace with:", self.replvar)[0] def create_command_buttons(self): SearchDialogBase.create_command_buttons(self) self.make_button("Find", self.find_it) self.make_button("Replace", self.replace_it) self.make_button("Replace+Find", self.default_command, 1) self.make_button("Replace All", self.replace_all) def find_it(self, event=None): self.do_find(0) def replace_it(self, event=None): if self.do_find(self.ok): self.do_replace() def default_command(self, event=None): "Replace and find next." if self.do_find(self.ok): if self.do_replace(): # Only find next match if replace succeeded. # A bad re can cause it to fail. self.do_find(0) def _replace_expand(self, m, repl): """ Helper function for expanding a regular expression in the replace field, if needed. """ if self.engine.isre(): try: new = m.expand(repl) except re.error: self.engine.report_error(repl, 'Invalid Replace Expression') new = None else: new = repl return new def replace_all(self, event=None): """Replace all instances of patvar with replvar in text""" prog = self.engine.getprog() if not prog: return repl = self.replvar.get() text = self.text res = self.engine.search_text(text, prog) if not res: self.bell() return text.tag_remove("sel", "1.0", "end") text.tag_remove("hit", "1.0", "end") line = res[0] col = res[1].start() if self.engine.iswrap(): line = 1 col = 0 ok = 1 first = last = None # XXX ought to replace circular instead of top-to-bottom when wrapping text.undo_block_start() while 1: res = self.engine.search_forward(text, prog, line, col, 0, ok) if not res: break line, m = res chars = text.get("%d.0" % line, "%d.0" % (line+1)) orig = m.group() new = self._replace_expand(m, repl) if new is None: break i, j = m.span() first = "%d.%d" % (line, i) last = "%d.%d" % (line, j) if new == orig: text.mark_set("insert", last) else: text.mark_set("insert", first) if first != last: text.delete(first, last) if new: text.insert(first, new) col = i + len(new) ok = 0 text.undo_block_stop() if first and last: self.show_hit(first, last) self.close() def do_find(self, ok=0): if not self.engine.getprog(): return False text = self.text res = self.engine.search_text(text, None, ok) if not res: self.bell() return False line, m = res i, j = m.span() first = "%d.%d" % (line, i) last = "%d.%d" % (line, j) self.show_hit(first, last) self.ok = 1 return True def do_replace(self): prog = self.engine.getprog() if not prog: return False text = self.text try: first = pos = text.index("sel.first") last = text.index("sel.last") except TclError: pos = None if not pos: first = last = pos = text.index("insert") line, col = searchengine.get_line_col(pos) chars = text.get("%d.0" % line, "%d.0" % (line+1)) m = prog.match(chars, col) if not prog: return False new = self._replace_expand(m, self.replvar.get()) if new is None: return False text.mark_set("insert", first) text.undo_block_start() if m.group(): text.delete(first, last) if new: text.insert(first, new) text.undo_block_stop() self.show_hit(first, text.index("insert")) self.ok = 0 return True def show_hit(self, first, last): """Highlight text from 'first' to 'last'. 'first', 'last' - Text indices""" text = self.text text.mark_set("insert", first) text.tag_remove("sel", "1.0", "end") text.tag_add("sel", first, last) text.tag_remove("hit", "1.0", "end") if first == last: text.tag_add("hit", first) else: text.tag_add("hit", first, last) text.see("insert") text.update_idletasks() def close(self, event=None): SearchDialogBase.close(self, event) self.text.tag_remove("hit", "1.0", "end") def _replace_dialog(parent): # htest # from tkinter import Toplevel, Text, END, SEL from tkinter.ttk import Button box = Toplevel(parent) box.title("Test ReplaceDialog") x, y = map(int, parent.geometry().split('+')[1:]) box.geometry("+%d+%d" % (x, y + 175)) # mock undo delegator methods def undo_block_start(): pass def undo_block_stop(): pass text = Text(box, inactiveselectbackground='gray') text.undo_block_start = undo_block_start text.undo_block_stop = undo_block_stop text.pack() text.insert("insert","This is a sample sTring\nPlus MORE.") text.focus_set() def show_replace(): text.tag_add(SEL, "1.0", END) replace(text) text.tag_remove(SEL, "1.0", END) button = Button(box, text="Replace", command=show_replace) button.pack() if __name__ == '__main__': from unittest import main main('idlelib.idle_test.test_replace', verbosity=2, exit=False) from idlelib.idle_test.htest import run run(_replace_dialog)
Name | Type | Size | Permission | Actions |
---|---|---|---|---|
Icons | Folder | 0755 |
|
|
__pycache__ | Folder | 0755 |
|
|
idle_test | Folder | 0755 |
|
|
CREDITS.txt | File | 1.82 KB | 0644 |
|
ChangeLog | File | 55.04 KB | 0644 |
|
HISTORY.txt | File | 10.07 KB | 0644 |
|
NEWS.txt | File | 38.91 KB | 0644 |
|
NEWS2x.txt | File | 26.54 KB | 0644 |
|
README.txt | File | 9.37 KB | 0644 |
|
TODO.txt | File | 8.28 KB | 0644 |
|
__init__.py | File | 396 B | 0644 |
|
__main__.py | File | 159 B | 0644 |
|
_pyclbr.py | File | 14.84 KB | 0644 |
|
autocomplete.py | File | 9.11 KB | 0644 |
|
autocomplete_w.py | File | 19.36 KB | 0644 |
|
autoexpand.py | File | 3.14 KB | 0644 |
|
browser.py | File | 8.09 KB | 0644 |
|
calltip.py | File | 5.92 KB | 0644 |
|
calltip_w.py | File | 6.94 KB | 0644 |
|
codecontext.py | File | 10.24 KB | 0644 |
|
colorizer.py | File | 11.01 KB | 0644 |
|
config-extensions.def | File | 2.21 KB | 0644 |
|
config-highlight.def | File | 2.62 KB | 0644 |
|
config-keys.def | File | 10.52 KB | 0644 |
|
config-main.def | File | 3.05 KB | 0644 |
|
config.py | File | 37.97 KB | 0644 |
|
config_key.py | File | 13.09 KB | 0644 |
|
configdialog.py | File | 98.69 KB | 0644 |
|
debugger.py | File | 18.65 KB | 0644 |
|
debugger_r.py | File | 11.86 KB | 0644 |
|
debugobj.py | File | 3.96 KB | 0644 |
|
debugobj_r.py | File | 1.06 KB | 0644 |
|
delegator.py | File | 1.02 KB | 0644 |
|
dynoption.py | File | 1.97 KB | 0644 |
|
editor.py | File | 65.7 KB | 0644 |
|
extend.txt | File | 3.56 KB | 0644 |
|
filelist.py | File | 3.8 KB | 0644 |
|
grep.py | File | 6.58 KB | 0644 |
|
help.html | File | 53.82 KB | 0644 |
|
help.py | File | 11.06 KB | 0644 |
|
help_about.py | File | 8.77 KB | 0644 |
|
history.py | File | 3.95 KB | 0644 |
|
hyperparser.py | File | 12.58 KB | 0644 |
|
idle.py | File | 454 B | 0644 |
|
idle.pyw | File | 570 B | 0644 |
|
iomenu.py | File | 20.25 KB | 0644 |
|
macosx.py | File | 9.43 KB | 0644 |
|
mainmenu.py | File | 3.62 KB | 0644 |
|
multicall.py | File | 18.21 KB | 0644 |
|
outwin.py | File | 5.67 KB | 0644 |
|
paragraph.py | File | 7 KB | 0644 |
|
parenmatch.py | File | 7.04 KB | 0644 |
|
pathbrowser.py | File | 3.12 KB | 0644 |
|
percolator.py | File | 3.06 KB | 0644 |
|
pyparse.py | File | 19.65 KB | 0644 |
|
pyshell.py | File | 56.38 KB | 0755 |
|
query.py | File | 12.14 KB | 0644 |
|
redirector.py | File | 6.71 KB | 0644 |
|
replace.py | File | 7.33 KB | 0644 |
|
rpc.py | File | 20.64 KB | 0644 |
|
rstrip.py | File | 868 B | 0644 |
|
run.py | File | 16.87 KB | 0644 |
|
runscript.py | File | 7.66 KB | 0644 |
|
scrolledlist.py | File | 4.35 KB | 0644 |
|
search.py | File | 3.09 KB | 0644 |
|
searchbase.py | File | 7.28 KB | 0644 |
|
searchengine.py | File | 7.3 KB | 0644 |
|
squeezer.py | File | 13 KB | 0644 |
|
stackviewer.py | File | 4.35 KB | 0644 |
|
statusbar.py | File | 1.41 KB | 0644 |
|
textview.py | File | 5.98 KB | 0644 |
|
tooltip.py | File | 6.33 KB | 0644 |
|
tree.py | File | 14.74 KB | 0644 |
|
undo.py | File | 10.79 KB | 0644 |
|
window.py | File | 2.53 KB | 0644 |
|
zoomheight.py | File | 1.31 KB | 0644 |
|
zzdummy.py | File | 961 B | 0644 |
|