Added temporary CLI status indicator
[findit] / src / files / search.py
1 #!/usr/bin/env python
2 # -*-coding: utf-8 -*-
3 # vim: sw=4 ts=4 expandtab ai
4
5 from os import walk
6 from os.path import join, abspath, normcase, basename, isdir, getsize
7 from heapq import nlargest
8
9 from misc import size_hum_read, _
10
11 #==============================================================================
12
13 class Control(object):
14
15     def __init__(self, ui, config):
16         self.config = config
17
18         self.present = eval(ui + '_Presentation(config, self.start_search)')
19         self.abstrac = Abstraction(self.config, self.present)
20
21         self.toplevel = self.present.toplevel
22
23     def start_search(self, get_data, get_stopit):
24         filelist = []
25         outtype, start_path, count = get_data()
26         search_func = self.abstrac.filegetter(start_path, get_stopit)
27         for fsize, fpath in nlargest(count, search_func):
28             filelist.append([int(fsize), fpath, size_hum_read(fsize)])
29         self.present.show_out_toplevel(None, outtype, filelist)
30
31     def run(self):
32         self.present.run()
33
34 #==============================================================================
35
36 class Abstraction(object):
37
38     def __init__(self, config, presentation):
39         self.ignore_dirs = config['ignore_dirs']
40         self.presentation = presentation
41
42     def filegetter(self, startdir, get_stopit):
43         """Generator of file sizes and paths based on os.walk."""
44
45         # Walk across directory tree
46         for dirpath, dirnames, fnames in walk(startdir):
47             # Eliminate unnecessary directories
48             ignore_dirs = self.ignore_dirs
49             for ign_dir in ignore_dirs[:]:
50                 for dirname in dirnames[:]:
51                     if ign_dir == normcase(join(abspath(dirpath), dirname)):
52                         dirnames.remove(dirname)
53                         ignore_dirs.remove(ign_dir)
54
55             for fname in fnames:
56                 flpath = abspath(join(dirpath, fname))
57                 self.presentation.show_current_status(flpath)
58
59                 # Stop search via 'stopit' signal
60                 stopit = get_stopit()
61                 if stopit:
62                     stopit = False
63                     print 'Stopped'
64                     raise StopIteration
65                 # Query only valid files
66                 try:
67                     # Return results (bytesize, path)
68                     yield getsize(flpath), flpath
69                 except OSError:
70                     continue
71
72 #==============================================================================
73
74 class Cli_Presentation(object):
75     def __init__(self, config, start_func):
76         import time; global time
77
78         self.start_func = start_func
79
80         self.outtype = config['outtype']
81         self.start_path = config['start_path']
82         self.count = config['count']
83         self.stopit = False
84
85         self.toplevel = None
86
87     def get_data(self):
88         return self.outtype, self.start_path, int(self.count)
89
90     def get_stopit(self):
91         return False
92
93     def show_out_toplevel(self, _, outtype, results):
94         out_submodule = __import__('files.' + outtype, None, None, outtype)
95         out_submodule.Cli_Presentation(results).toplevel
96
97     def show_current_status(self, current_path):
98         #pass
99         print '|' + '\r',
100         print '/' + '\r',
101         print '-' + '\r',
102         print '\\' + '\r',
103         ### print current_path
104
105     def run(self):
106         self.start_func(self.get_data, self.get_stopit)
107
108 #==============================================================================
109
110 class Gtk_Presentation(object):
111
112     def __init__(self, config, start_func):
113         import gtk
114
115         self.config = config
116
117         # "Start path" entry
118         self.path_entry = gtk.Entry()
119         self.path_entry.set_text(self.config['start_path'])
120
121         # "Files quantity" label
122         qty_label = gtk.Label(_('Files quantity'))
123
124         # "Files quantity" spin
125         self.qty_spin = gtk.SpinButton()
126         self.qty_spin.set_numeric(True)
127         self.qty_spin.set_range(0, 65536)
128         self.qty_spin.set_increments(1, 10)
129         self.qty_spin.set_value(self.config['count'])
130
131         # "Start" button
132         self.start_btn = gtk.Button(_('Start'))
133         self.start_btn.connect('released', self.start_btn_released, start_func)
134
135         # "Stop" button
136         self.stop_btn = gtk.Button(_('Stop'))
137         self.stop_btn.set_sensitive(False)
138         self.stop_btn.connect('clicked', self.stop_btn_clicked)
139
140         # Output selection
141         outtable_rbtn = gtk.RadioButton(None, _('Table'))
142         outtable_rbtn.set_name('outtable')
143         outdiagram_rbtn = gtk.RadioButton(outtable_rbtn, _('Diagram'))
144         outdiagram_rbtn.set_name('outdiagram')
145         out1_rbtn = gtk.RadioButton(outtable_rbtn, 'Another 1')
146         out1_rbtn.set_name('outanother1')
147         self.out_rbtns = [outtable_rbtn, outdiagram_rbtn, out1_rbtn]
148
149         hbox = gtk.HBox(False, 4)
150         hbox.pack_start(qty_label, False, False, 0)
151         hbox.pack_start(self.qty_spin, False, False, 0)
152         hbox.pack_start(self.start_btn, False, False, 0)
153         hbox.pack_start(self.stop_btn, False, False, 0)
154         for btn in reversed(self.out_rbtns):
155             hbox.pack_end(btn, False, False, 0)
156             # Activate radio button
157             if btn.get_name() == self.config['outtype']:
158                 btn.set_active(True)
159
160         self.statusbar = gtk.Statusbar()
161         self.context_id = self.statusbar.get_context_id('Current walked file')
162
163         self.vbox = gtk.VBox(False, 4)
164         self.vbox.pack_start(self.path_entry, False, False, 0)
165         self.vbox.pack_start(hbox, False, False, 0)
166         self.vbox.pack_end(self.statusbar, False, False, 0)
167
168         self.toplevel = self.vbox
169
170         # For importing gtk only once (lambda not work)
171         def show_current_status(current_path):
172             self.statusbar.push(self.context_id, current_path)
173             gtk.main_iteration()
174         self.show_current_status = show_current_status
175
176         self.show_out_toplevel(None, self.config['outtype'], [(1, 'path', 'bytesize')])
177
178     #=== Functions ============================================================
179     def start_btn_released(self, btn, start_func):
180         self.stopit = False
181         self.stop_btn.set_sensitive(True)
182         self.start_btn.set_sensitive(False)
183         start_func(self.get_data, self.get_stopit)
184         self.stop_btn.set_sensitive(False)
185         self.start_btn.set_sensitive(True)
186
187     def stop_btn_clicked(self, widget):
188         self.stopit = True
189         self.stop_btn.set_sensitive(False)
190         self.start_btn.set_sensitive(True)
191
192     def get_data(self):
193         for btn in self.out_rbtns:
194             if btn.get_active():
195                 out = btn.get_name()
196         return out, self.path_entry.get_text(), int(self.qty_spin.get_value())
197
198     def get_stopit(self):
199         return self.stopit
200
201     def run(self):
202         pass
203
204     #=== Output type selecting ================================================
205     def show_out_toplevel(self, btn, outtype, results):
206         print 'Entering <' + outtype + '> output mode...'
207         out_submodule = __import__('files.' + outtype, None, None, outtype)
208
209         try:
210             self.out_toplevel.destroy()
211         except:
212             pass
213
214         self.out_toplevel = out_submodule.Gtk_Presentation(results).toplevel
215         self.vbox.add(self.out_toplevel)
216         self.out_toplevel.show_all()
217 ###        out_submodule.Gtk_Presentation().show_results(results)
218
219 #==============================================================================
220
221 class Hildon_Presentation(object):
222
223     def __init__(self, config, start_func):
224         import gtk
225         import hildon
226
227         self.config = config
228
229     def run(self):
230         pass