GrepDialog.py 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. from __future__ import print_function
  2. import os
  3. import fnmatch
  4. import re # for htest
  5. import sys
  6. from Tkinter import StringVar, BooleanVar, Checkbutton # for GrepDialog
  7. from Tkinter import Tk, Text, Button, SEL, END # for htest
  8. from idlelib import SearchEngine
  9. from idlelib.SearchDialogBase import SearchDialogBase
  10. # Importing OutputWindow fails due to import loop
  11. # EditorWindow -> GrepDialop -> OutputWindow -> EditorWindow
  12. def grep(text, io=None, flist=None):
  13. root = text._root()
  14. engine = SearchEngine.get(root)
  15. if not hasattr(engine, "_grepdialog"):
  16. engine._grepdialog = GrepDialog(root, engine, flist)
  17. dialog = engine._grepdialog
  18. searchphrase = text.get("sel.first", "sel.last")
  19. dialog.open(text, searchphrase, io)
  20. class GrepDialog(SearchDialogBase):
  21. title = "Find in Files Dialog"
  22. icon = "Grep"
  23. needwrapbutton = 0
  24. def __init__(self, root, engine, flist):
  25. SearchDialogBase.__init__(self, root, engine)
  26. self.flist = flist
  27. self.globvar = StringVar(root)
  28. self.recvar = BooleanVar(root)
  29. def open(self, text, searchphrase, io=None):
  30. SearchDialogBase.open(self, text, searchphrase)
  31. if io:
  32. path = io.filename or ""
  33. else:
  34. path = ""
  35. dir, base = os.path.split(path)
  36. head, tail = os.path.splitext(base)
  37. if not tail:
  38. tail = ".py"
  39. self.globvar.set(os.path.join(dir, "*" + tail))
  40. def create_entries(self):
  41. SearchDialogBase.create_entries(self)
  42. self.globent = self.make_entry("In files:", self.globvar)[0]
  43. def create_other_buttons(self):
  44. f = self.make_frame()[0]
  45. btn = Checkbutton(f, anchor="w",
  46. variable=self.recvar,
  47. text="Recurse down subdirectories")
  48. btn.pack(side="top", fill="both")
  49. btn.select()
  50. def create_command_buttons(self):
  51. SearchDialogBase.create_command_buttons(self)
  52. self.make_button("Search Files", self.default_command, 1)
  53. def default_command(self, event=None):
  54. prog = self.engine.getprog()
  55. if not prog:
  56. return
  57. path = self.globvar.get()
  58. if not path:
  59. self.top.bell()
  60. return
  61. from idlelib.OutputWindow import OutputWindow # leave here!
  62. save = sys.stdout
  63. try:
  64. sys.stdout = OutputWindow(self.flist)
  65. self.grep_it(prog, path)
  66. finally:
  67. sys.stdout = save
  68. def grep_it(self, prog, path):
  69. dir, base = os.path.split(path)
  70. list = self.findfiles(dir, base, self.recvar.get())
  71. list.sort()
  72. self.close()
  73. pat = self.engine.getpat()
  74. print("Searching %r in %s ..." % (pat, path))
  75. hits = 0
  76. try:
  77. for fn in list:
  78. try:
  79. with open(fn) as f:
  80. for lineno, line in enumerate(f, 1):
  81. if line[-1:] == '\n':
  82. line = line[:-1]
  83. if prog.search(line):
  84. sys.stdout.write("%s: %s: %s\n" %
  85. (fn, lineno, line))
  86. hits += 1
  87. except IOError as msg:
  88. print(msg)
  89. print(("Hits found: %s\n"
  90. "(Hint: right-click to open locations.)"
  91. % hits) if hits else "No hits.")
  92. except AttributeError:
  93. # Tk window has been closed, OutputWindow.text = None,
  94. # so in OW.write, OW.text.insert fails.
  95. pass
  96. def findfiles(self, dir, base, rec):
  97. try:
  98. names = os.listdir(dir or os.curdir)
  99. except os.error as msg:
  100. print(msg)
  101. return []
  102. list = []
  103. subdirs = []
  104. for name in names:
  105. fn = os.path.join(dir, name)
  106. if os.path.isdir(fn):
  107. subdirs.append(fn)
  108. else:
  109. if fnmatch.fnmatch(name, base):
  110. list.append(fn)
  111. if rec:
  112. for subdir in subdirs:
  113. list.extend(self.findfiles(subdir, base, rec))
  114. return list
  115. def close(self, event=None):
  116. if self.top:
  117. self.top.grab_release()
  118. self.top.withdraw()
  119. def _grep_dialog(parent): # htest #
  120. from idlelib.PyShell import PyShellFileList
  121. root = Tk()
  122. root.title("Test GrepDialog")
  123. width, height, x, y = list(map(int, re.split('[x+]', parent.geometry())))
  124. root.geometry("+%d+%d"%(x, y + 150))
  125. flist = PyShellFileList(root)
  126. text = Text(root, height=5)
  127. text.pack()
  128. def show_grep_dialog():
  129. text.tag_add(SEL, "1.0", END)
  130. grep(text, flist=flist)
  131. text.tag_remove(SEL, "1.0", END)
  132. button = Button(root, text="Show GrepDialog", command=show_grep_dialog)
  133. button.pack()
  134. root.mainloop()
  135. if __name__ == "__main__":
  136. import unittest
  137. unittest.main('idlelib.idle_test.test_grep', verbosity=2, exit=False)
  138. from idlelib.idle_test.htest import run
  139. run(_grep_dialog)