mirror of
				https://github.com/python/cpython.git
				synced 2025-11-04 11:49:12 +00:00 
			
		
		
		
	This follows the previous patch that changed idlelib file names. Class names that matched old module names are not changed. Change idlelib imports in turtledemo.__main__. Exception: config-extensions.def. Previously, extension section names, file names, and class names had to match. Changing section names would create cross-version conflicts in config-extensions.cfg (user customizations). Instead map old names to new file names at point of import in editor.EditorWindow.load_extension. Patch extensively tested with test_idle, idle_test.htest.py, a custom import-all test, running IDLE in a console to catch messages, and testing each menu item. Based on a patch by Al Sweigart.
		
			
				
	
	
		
			158 lines
		
	
	
	
		
			5 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			158 lines
		
	
	
	
		
			5 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
import os
 | 
						|
import fnmatch
 | 
						|
import re  # for htest
 | 
						|
import sys
 | 
						|
from tkinter import StringVar, BooleanVar, Checkbutton  # for GrepDialog
 | 
						|
from tkinter import Tk, Text, Button, SEL, END  # for htest
 | 
						|
from idlelib import searchengine
 | 
						|
from idlelib.searchbase import SearchDialogBase
 | 
						|
# Importing OutputWindow fails due to import loop
 | 
						|
# EditorWindow -> GrepDialop -> OutputWindow -> EditorWindow
 | 
						|
 | 
						|
def grep(text, io=None, flist=None):
 | 
						|
    root = text._root()
 | 
						|
    engine = searchengine.get(root)
 | 
						|
    if not hasattr(engine, "_grepdialog"):
 | 
						|
        engine._grepdialog = GrepDialog(root, engine, flist)
 | 
						|
    dialog = engine._grepdialog
 | 
						|
    searchphrase = text.get("sel.first", "sel.last")
 | 
						|
    dialog.open(text, searchphrase, io)
 | 
						|
 | 
						|
class GrepDialog(SearchDialogBase):
 | 
						|
 | 
						|
    title = "Find in Files Dialog"
 | 
						|
    icon = "Grep"
 | 
						|
    needwrapbutton = 0
 | 
						|
 | 
						|
    def __init__(self, root, engine, flist):
 | 
						|
        SearchDialogBase.__init__(self, root, engine)
 | 
						|
        self.flist = flist
 | 
						|
        self.globvar = StringVar(root)
 | 
						|
        self.recvar = BooleanVar(root)
 | 
						|
 | 
						|
    def open(self, text, searchphrase, io=None):
 | 
						|
        SearchDialogBase.open(self, text, searchphrase)
 | 
						|
        if io:
 | 
						|
            path = io.filename or ""
 | 
						|
        else:
 | 
						|
            path = ""
 | 
						|
        dir, base = os.path.split(path)
 | 
						|
        head, tail = os.path.splitext(base)
 | 
						|
        if not tail:
 | 
						|
            tail = ".py"
 | 
						|
        self.globvar.set(os.path.join(dir, "*" + tail))
 | 
						|
 | 
						|
    def create_entries(self):
 | 
						|
        SearchDialogBase.create_entries(self)
 | 
						|
        self.globent = self.make_entry("In files:", self.globvar)[0]
 | 
						|
 | 
						|
    def create_other_buttons(self):
 | 
						|
        f = self.make_frame()[0]
 | 
						|
 | 
						|
        btn = Checkbutton(f, anchor="w",
 | 
						|
                variable=self.recvar,
 | 
						|
                text="Recurse down subdirectories")
 | 
						|
        btn.pack(side="top", fill="both")
 | 
						|
        btn.select()
 | 
						|
 | 
						|
    def create_command_buttons(self):
 | 
						|
        SearchDialogBase.create_command_buttons(self)
 | 
						|
        self.make_button("Search Files", self.default_command, 1)
 | 
						|
 | 
						|
    def default_command(self, event=None):
 | 
						|
        prog = self.engine.getprog()
 | 
						|
        if not prog:
 | 
						|
            return
 | 
						|
        path = self.globvar.get()
 | 
						|
        if not path:
 | 
						|
            self.top.bell()
 | 
						|
            return
 | 
						|
        from idlelib.outwin import OutputWindow  # leave here!
 | 
						|
        save = sys.stdout
 | 
						|
        try:
 | 
						|
            sys.stdout = OutputWindow(self.flist)
 | 
						|
            self.grep_it(prog, path)
 | 
						|
        finally:
 | 
						|
            sys.stdout = save
 | 
						|
 | 
						|
    def grep_it(self, prog, path):
 | 
						|
        dir, base = os.path.split(path)
 | 
						|
        list = self.findfiles(dir, base, self.recvar.get())
 | 
						|
        list.sort()
 | 
						|
        self.close()
 | 
						|
        pat = self.engine.getpat()
 | 
						|
        print("Searching %r in %s ..." % (pat, path))
 | 
						|
        hits = 0
 | 
						|
        try:
 | 
						|
            for fn in list:
 | 
						|
                try:
 | 
						|
                    with open(fn, errors='replace') as f:
 | 
						|
                        for lineno, line in enumerate(f, 1):
 | 
						|
                            if line[-1:] == '\n':
 | 
						|
                                line = line[:-1]
 | 
						|
                            if prog.search(line):
 | 
						|
                                sys.stdout.write("%s: %s: %s\n" %
 | 
						|
                                                 (fn, lineno, line))
 | 
						|
                                hits += 1
 | 
						|
                except OSError as msg:
 | 
						|
                    print(msg)
 | 
						|
            print(("Hits found: %s\n"
 | 
						|
                  "(Hint: right-click to open locations.)"
 | 
						|
                  % hits) if hits else "No hits.")
 | 
						|
        except AttributeError:
 | 
						|
            # Tk window has been closed, OutputWindow.text = None,
 | 
						|
            # so in OW.write, OW.text.insert fails.
 | 
						|
            pass
 | 
						|
 | 
						|
    def findfiles(self, dir, base, rec):
 | 
						|
        try:
 | 
						|
            names = os.listdir(dir or os.curdir)
 | 
						|
        except OSError as msg:
 | 
						|
            print(msg)
 | 
						|
            return []
 | 
						|
        list = []
 | 
						|
        subdirs = []
 | 
						|
        for name in names:
 | 
						|
            fn = os.path.join(dir, name)
 | 
						|
            if os.path.isdir(fn):
 | 
						|
                subdirs.append(fn)
 | 
						|
            else:
 | 
						|
                if fnmatch.fnmatch(name, base):
 | 
						|
                    list.append(fn)
 | 
						|
        if rec:
 | 
						|
            for subdir in subdirs:
 | 
						|
                list.extend(self.findfiles(subdir, base, rec))
 | 
						|
        return list
 | 
						|
 | 
						|
    def close(self, event=None):
 | 
						|
        if self.top:
 | 
						|
            self.top.grab_release()
 | 
						|
            self.top.withdraw()
 | 
						|
 | 
						|
 | 
						|
def _grep_dialog(parent):  # htest #
 | 
						|
    from idlelib.pyshell import PyShellFileList
 | 
						|
    root = Tk()
 | 
						|
    root.title("Test GrepDialog")
 | 
						|
    width, height, x, y = list(map(int, re.split('[x+]', parent.geometry())))
 | 
						|
    root.geometry("+%d+%d"%(x, y + 150))
 | 
						|
 | 
						|
    flist = PyShellFileList(root)
 | 
						|
    text = Text(root, height=5)
 | 
						|
    text.pack()
 | 
						|
 | 
						|
    def show_grep_dialog():
 | 
						|
        text.tag_add(SEL, "1.0", END)
 | 
						|
        grep(text, flist=flist)
 | 
						|
        text.tag_remove(SEL, "1.0", END)
 | 
						|
 | 
						|
    button = Button(root, text="Show GrepDialog", command=show_grep_dialog)
 | 
						|
    button.pack()
 | 
						|
    root.mainloop()
 | 
						|
 | 
						|
if __name__ == "__main__":
 | 
						|
    import unittest
 | 
						|
    unittest.main('idlelib.idle_test.test_grep', verbosity=2, exit=False)
 | 
						|
 | 
						|
    from idlelib.idle_test.htest import run
 | 
						|
    run(_grep_dialog)
 |