from tkinter import * from idlelib import SearchEngine from idlelib.SearchDialogBase import SearchDialogBase import re def replace(text): 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): 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): 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): 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): 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: text.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: text.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): 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): root = Tk() root.title("Test ReplaceDialog") width, height, x, y = list(map(int, re.split('[x+]', parent.geometry()))) root.geometry("+%d+%d"%(x, y + 150)) # mock undo delegator methods def undo_block_start(): pass def undo_block_stop(): pass text = Text(root) text.undo_block_start = undo_block_start text.undo_block_stop = undo_block_stop text.pack() text.insert("insert","This is a sample string.\n"*10) def show_replace(): text.tag_add(SEL, "1.0", END) replace(text) text.tag_remove(SEL, "1.0", END) button = Button(root, text="Replace", command=show_replace) button.pack() if __name__ == '__main__': 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 |
|
|
AutoComplete.py | File | 8.99 KB | 0644 |
|
AutoCompleteWindow.py | File | 17.32 KB | 0644 |
|
AutoExpand.py | File | 3.32 KB | 0644 |
|
Bindings.py | File | 2.97 KB | 0644 |
|
CREDITS.txt | File | 1.82 KB | 0644 |
|
CallTipWindow.py | File | 5.89 KB | 0644 |
|
CallTips.py | File | 5.79 KB | 0644 |
|
ChangeLog | File | 55.07 KB | 0644 |
|
ClassBrowser.py | File | 6.83 KB | 0644 |
|
CodeContext.py | File | 8.15 KB | 0644 |
|
ColorDelegator.py | File | 9.47 KB | 0644 |
|
Debugger.py | File | 18.32 KB | 0644 |
|
Delegator.py | File | 665 B | 0644 |
|
EditorWindow.py | File | 64.51 KB | 0644 |
|
FileList.py | File | 3.72 KB | 0644 |
|
FormatParagraph.py | File | 7.12 KB | 0644 |
|
GrepDialog.py | File | 5 KB | 0644 |
|
HISTORY.txt | File | 10.08 KB | 0644 |
|
HyperParser.py | File | 12.58 KB | 0644 |
|
IOBinding.py | File | 19.31 KB | 0644 |
|
IdleHistory.py | File | 3.96 KB | 0644 |
|
MultiCall.py | File | 18.14 KB | 0644 |
|
MultiStatusBar.py | File | 1.32 KB | 0644 |
|
NEWS.txt | File | 38.9 KB | 0644 |
|
ObjectBrowser.py | File | 3.88 KB | 0644 |
|
OutputWindow.py | File | 4.29 KB | 0644 |
|
ParenMatch.py | File | 6.56 KB | 0644 |
|
PathBrowser.py | File | 3.13 KB | 0644 |
|
Percolator.py | File | 3.17 KB | 0644 |
|
PyParse.py | File | 19.98 KB | 0644 |
|
PyShell.py | File | 57.08 KB | 0755 |
|
README.txt | File | 7.53 KB | 0644 |
|
RemoteDebugger.py | File | 11.73 KB | 0644 |
|
RemoteObjectBrowser.py | File | 964 B | 0644 |
|
ReplaceDialog.py | File | 6.48 KB | 0644 |
|
RstripExtension.py | File | 1.03 KB | 0644 |
|
ScriptBinding.py | File | 7.87 KB | 0644 |
|
ScrolledList.py | File | 4.27 KB | 0644 |
|
SearchDialog.py | File | 2.57 KB | 0644 |
|
SearchDialogBase.py | File | 6.84 KB | 0644 |
|
SearchEngine.py | File | 7.31 KB | 0644 |
|
StackViewer.py | File | 4.32 KB | 0644 |
|
TODO.txt | File | 8.28 KB | 0644 |
|
ToolTip.py | File | 3.1 KB | 0644 |
|
TreeWidget.py | File | 14.67 KB | 0644 |
|
UndoDelegator.py | File | 10.56 KB | 0644 |
|
WidgetRedirector.py | File | 6.71 KB | 0644 |
|
WindowList.py | File | 2.41 KB | 0644 |
|
ZoomHeight.py | File | 1.27 KB | 0644 |
|
__init__.py | File | 288 B | 0644 |
|
__main__.py | File | 159 B | 0644 |
|
aboutDialog.py | File | 6.53 KB | 0644 |
|
config-extensions.def | File | 2.9 KB | 0644 |
|
config-highlight.def | File | 2.46 KB | 0644 |
|
config-keys.def | File | 7.59 KB | 0644 |
|
config-main.def | File | 2.5 KB | 0644 |
|
configDialog.py | File | 62.8 KB | 0644 |
|
configHandler.py | File | 31.44 KB | 0644 |
|
configHelpSourceEdit.py | File | 6.51 KB | 0644 |
|
configSectionNameDialog.py | File | 3.91 KB | 0644 |
|
dynOptionMenuWidget.py | File | 1.94 KB | 0644 |
|
extend.txt | File | 3.56 KB | 0644 |
|
help.html | File | 40.16 KB | 0644 |
|
help.py | File | 9.47 KB | 0644 |
|
help.txt | File | 17.48 KB | 0644 |
|
idle.py | File | 400 B | 0644 |
|
idle.pyw | File | 570 B | 0644 |
|
idlever.py | File | 415 B | 0644 |
|
keybindingDialog.py | File | 12.13 KB | 0644 |
|
macosxSupport.py | File | 8.48 KB | 0644 |
|
rpc.py | File | 20.29 KB | 0644 |
|
run.py | File | 13.35 KB | 0644 |
|
tabbedpages.py | File | 17.99 KB | 0644 |
|
textView.py | File | 3.15 KB | 0644 |
|