add src/searchfile.py - first approximation
[findit] / src / searchfile.py
1 #!/usr/bin/env python
2 # -*-coding: utf-8 -*-
3 # vim: sw=4 ts=4 expandtab ai
4
5 import gtk
6 import pango
7 from os import walk
8 from os.path import join, abspath, normcase, basename, \
9                             isdir, getsize
10 from heapq import nlargest
11
12 class Search_File_Control(object):
13     
14     def __init__(self, start_dir, ignore_dirs, count):
15         self.filelist = []
16         self.srch_present = Search_File_Presentation(self.start_search)
17         self.srch_abs = Search_File_Abstraction(start_dir, ignore_dirs)
18         self.fl_cnt = count
19
20     def get_ui(self):
21         return self.srch_present.get_ui()
22
23     def start_search(self, label):
24         count, start_path = self.srch_present() 
25         for fsize, fpath in nlargest(self.fl_cnt, \
26                 self.srch_abs.filegetter(self.srch_present.stopit, label) ):
27             self.filelist.append([fpath, self.srch_abs.size_convert(fsize), fsize])
28         return self.filelist
29
30
31 class Search_File_Abstraction(object):
32     
33     def __init__(self, startdir, ignore_dirs):
34         self.startdir = startdir
35         self.ignore_dirs = ignore_dirs
36         self.label = label
37         
38     def size_convert(self, size):
39         """Return string with file size in b or Kb or Mb or Gb or Tb."""
40         for i, unit in enumerate(['%d b', '%.1f Kb', '%.2f Mb', '%.3f Gb', '%.4f Tb']):
41             if size < 1024**(i+1):
42                 return unit % (size/1024.**i)
43         return '>1024 Tb'
44
45     def filegetter(self, stopit, label):
46         """Generator of file sizes and paths based on os.walk."""
47         # Проходим по всем папкам вглубь от заданного пути
48         for dirpath, dirnames, fnames in walk(self.startdir):
49         # Исключаем каталоги из поиска в соответствии со списком исключений
50             for ign_dir in self.ignore_dirs[:]:
51                 for dirname in dirnames[:]:
52                     if ign_dir == normcase(join(abspath(dirpath), dirname)):
53                         dirnames.remove(dirname)
54                         ignore_dirs.remove(ign_dir)
55
56             for fname in fnames:
57                 flpath = abspath(join(dirpath, fname))
58                 # Выводим текущий опрашиваемый файл в строку статуса
59                 label.set_text(flpath)
60                 # обновляем окно
61                 gtk.main_iteration()
62                 # Останавливаем цикл по нажатию кнопки стоп
63                 if stopit:
64                     stopit = False
65                     raise StopIteration
66                 # Проверяем можем ли мы определить размер файла - иначе пропускаем его
67                 try:
68                     # Возвращаем размер и полный путь файла
69                     yield getsize(flpath), flpath
70                 except OSError:
71                     continue
72   
73 class Search_File_Presentation(object):
74     
75     def __init__(self, st_func):
76         # Строка вывода текущего осматриваемого файла
77         self.currfilelbl = gtk.Label('---')
78         self.currfilelbl.set_alignment(0, 0.5)
79         self.currfilelbl.set_ellipsize(pango.ELLIPSIZE_MIDDLE)
80         self.currfilelbl.set_padding(2, 2)
81         currfilefrm = gtk.Frame()
82         currfilefrm.add(self.currfilelbl)
83
84         # Кнопка "Показать"
85         self.butt_start = gtk.Button('Go')
86         self.butt_start.connect('released', self.start_srch(st_func))
87
88         # Кнопка "Остановить"
89         self.butt_stop = gtk.Button('Stop')
90         self.butt_stop.set_sensitive(False)
91         self.butt_stop.connect('clicked', self.stop_srch)
92         self.stopit = False
93
94         self.search_hbox = gtk.HBox(False, 10)
95         self.search_hbox.pack_start(currfilefrm, False, False, 0)
96         self.search_hbox.pack_end(self.butt_start, False, False, 0)
97         self.search_hbox.pack_end(self.butt_stop, False, False, 0)
98
99     def get_ui(self):
100         return self.search_hbox
101
102     def start_srch(self, start_func):
103         self.butt_stop.set_sensitive(True)
104         self.butt_start.set_sensitive(False)
105         start_func(self.currfilelbl)
106         self.butt_stop.set_sensitive(False)
107         self.butt_start.set_sensitive(True)
108
109     def stop_srch(self):
110         self.stopit = True
111         self.butt_stop.set_sensitive(False)
112         self.butt_start.set_sensitive(True)