1
# Copyright (C) 2006 by Szilveszter Farkas (Phanatic) <szilveszter.farkas@gmail.com>
3
# This program is free software; you can redistribute it and/or modify
4
# it under the terms of the GNU General Public License as published by
5
# the Free Software Foundation; either version 2 of the License, or
6
# (at your option) any later version.
8
# This program is distributed in the hope that it will be useful,
9
# but WITHOUT ANY WARRANTY; without even the implied warranty of
10
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11
# GNU General Public License for more details.
13
# You should have received a copy of the GNU General Public License
14
# along with this program; if not, write to the Free Software
15
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
30
from bzrlib.branch import Branch
31
import bzrlib.errors as errors
32
from bzrlib.workingtree import WorkingTree
34
# Olive GTK UI version
35
__version__ = '0.11.0'
38
if sys.platform == 'win32':
39
gladefile = os.path.dirname(sys.executable) + "/share/olive/olive.glade"
41
gladefile = "/usr/share/olive/olive.glade"
43
if not os.path.exists(gladefile):
44
# Load from current directory if not installed
45
gladefile = "olive.glade"
47
if not os.path.exists(gladefile):
49
print _('Glade file cannot be found.')
52
from dialog import error_dialog, info_dialog
55
""" The main Olive GTK frontend class. This is called when launching the
59
self.toplevel = gtk.glade.XML(gladefile, 'window_main', 'olive-gtk')
61
self.window = self.toplevel.get_widget('window_main')
63
self.pref = OlivePreferences()
65
# Initialize the statusbar
66
self.statusbar = self.toplevel.get_widget('statusbar')
67
self.context_id = self.statusbar.get_context_id('olive')
70
self.window_main = self.toplevel.get_widget('window_main')
72
self.hpaned_main = self.toplevel.get_widget('hpaned_main')
74
self.treeview_left = self.toplevel.get_widget('treeview_left')
75
self.treeview_right = self.toplevel.get_widget('treeview_right')
76
# Get some important menu items
77
self.menuitem_add_files = self.toplevel.get_widget('menuitem_add_files')
78
self.menuitem_remove_files = self.toplevel.get_widget('menuitem_remove_file')
79
self.menuitem_file_make_directory = self.toplevel.get_widget('menuitem_file_make_directory')
80
self.menuitem_file_rename = self.toplevel.get_widget('menuitem_file_rename')
81
self.menuitem_file_move = self.toplevel.get_widget('menuitem_file_move')
82
self.menuitem_view_show_hidden_files = self.toplevel.get_widget('menuitem_view_show_hidden_files')
83
self.menuitem_branch = self.toplevel.get_widget('menuitem_branch')
84
self.menuitem_branch_init = self.toplevel.get_widget('menuitem_branch_initialize')
85
self.menuitem_branch_get = self.toplevel.get_widget('menuitem_branch_get')
86
self.menuitem_branch_checkout = self.toplevel.get_widget('menuitem_branch_checkout')
87
self.menuitem_branch_pull = self.toplevel.get_widget('menuitem_branch_pull')
88
self.menuitem_branch_push = self.toplevel.get_widget('menuitem_branch_push')
89
self.menuitem_branch_commit = self.toplevel.get_widget('menuitem_branch_commit')
90
self.menuitem_branch_status = self.toplevel.get_widget('menuitem_branch_status')
91
self.menuitem_branch_missing = self.toplevel.get_widget('menuitem_branch_missing_revisions')
92
self.menuitem_stats = self.toplevel.get_widget('menuitem_stats')
93
self.menuitem_stats_diff = self.toplevel.get_widget('menuitem_stats_diff')
94
self.menuitem_stats_log = self.toplevel.get_widget('menuitem_stats_log')
95
# Get some toolbuttons
96
#self.menutoolbutton_diff = self.toplevel.get_widget('menutoolbutton_diff')
97
self.toolbutton_diff = self.toplevel.get_widget('toolbutton_diff')
98
self.toolbutton_log = self.toplevel.get_widget('toolbutton_log')
99
self.toolbutton_commit = self.toplevel.get_widget('toolbutton_commit')
100
self.toolbutton_pull = self.toplevel.get_widget('toolbutton_pull')
101
self.toolbutton_push = self.toplevel.get_widget('toolbutton_push')
102
# Get the drive selector
103
self.combobox_drive = gtk.combo_box_new_text()
104
self.combobox_drive.connect("changed", self._refresh_drives)
106
self.vbox_main_right = self.toplevel.get_widget('vbox_main_right')
109
# Dictionary for signal_autoconnect
110
dic = { "on_window_main_destroy": gtk.main_quit,
111
"on_window_main_delete_event": self.on_window_main_delete_event,
112
"on_quit_activate": self.on_window_main_delete_event,
113
"on_about_activate": self.on_about_activate,
114
"on_menuitem_add_files_activate": self.on_menuitem_add_files_activate,
115
"on_menuitem_remove_file_activate": self.on_menuitem_remove_file_activate,
116
"on_menuitem_file_make_directory_activate": self.on_menuitem_file_make_directory_activate,
117
"on_menuitem_file_move_activate": self.on_menuitem_file_move_activate,
118
"on_menuitem_file_rename_activate": self.on_menuitem_file_rename_activate,
119
"on_menuitem_view_show_hidden_files_activate": self.on_menuitem_view_show_hidden_files_activate,
120
"on_menuitem_view_refresh_activate": self.on_menuitem_view_refresh_activate,
121
"on_menuitem_branch_initialize_activate": self.on_menuitem_branch_initialize_activate,
122
"on_menuitem_branch_get_activate": self.on_menuitem_branch_get_activate,
123
"on_menuitem_branch_checkout_activate": self.on_menuitem_branch_checkout_activate,
124
"on_menuitem_branch_commit_activate": self.on_menuitem_branch_commit_activate,
125
"on_menuitem_branch_push_activate": self.on_menuitem_branch_push_activate,
126
"on_menuitem_branch_pull_activate": self.on_menuitem_branch_pull_activate,
127
"on_menuitem_branch_status_activate": self.on_menuitem_branch_status_activate,
128
"on_menuitem_branch_missing_revisions_activate": self.on_menuitem_branch_missing_revisions_activate,
129
"on_menuitem_stats_diff_activate": self.on_menuitem_stats_diff_activate,
130
"on_menuitem_stats_log_activate": self.on_menuitem_stats_log_activate,
131
"on_menuitem_stats_infos_activate": self.on_menuitem_stats_infos_activate,
132
"on_toolbutton_refresh_clicked": self.on_menuitem_view_refresh_activate,
133
"on_toolbutton_log_clicked": self.on_menuitem_stats_log_activate,
134
#"on_menutoolbutton_diff_clicked": self.on_menuitem_stats_diff_activate,
135
"on_toolbutton_diff_clicked": self.on_menuitem_stats_diff_activate,
136
"on_toolbutton_commit_clicked": self.on_menuitem_branch_commit_activate,
137
"on_toolbutton_pull_clicked": self.on_menuitem_branch_pull_activate,
138
"on_toolbutton_push_clicked": self.on_menuitem_branch_push_activate,
139
"on_treeview_right_button_press_event": self.on_treeview_right_button_press_event,
140
"on_treeview_right_row_activated": self.on_treeview_right_row_activated,
141
"on_treeview_left_button_press_event": self.on_treeview_left_button_press_event,
142
"on_treeview_left_row_activated": self.on_treeview_left_row_activated }
144
# Connect the signals to the handlers
145
self.toplevel.signal_autoconnect(dic)
147
# Apply window size and position
148
width = self.pref.get_preference('window_width', 'int')
149
height = self.pref.get_preference('window_height', 'int')
150
self.window.resize(width, height)
151
x = self.pref.get_preference('window_x', 'int')
152
y = self.pref.get_preference('window_y', 'int')
153
self.window.move(x, y)
154
# Apply paned position
155
pos = self.pref.get_preference('paned_position', 'int')
156
self.hpaned_main.set_position(pos)
158
# Apply menu to the toolbutton
159
#menubutton = self.toplevel.get_widget('menutoolbutton_diff')
160
#menubutton.set_menu(handler.menu.toolbar_diff)
162
# Now we can show the window
165
# Show drive selector if under Win32
166
if sys.platform == 'win32':
167
self.vbox_main_right.pack_start(self.combobox_drive, False, True, 0)
168
self.vbox_main_right.reorder_child(self.combobox_drive, 0)
169
self.combobox_drive.show()
170
self.gen_hard_selector()
175
self.menuitem_view_show_hidden_files.set_active(self.pref.get_preference('dotted_files', 'bool'))
177
self.set_path(os.getcwd())
180
def set_path(self, path):
182
self.notbranch = False
184
self.wt, self.wtpath = WorkingTree.open_containing(self.path)
185
except errors.NotBranchError:
186
self.notbranch = True
191
def on_about_activate(self, widget):
192
from dialog import about
195
def on_menuitem_add_files_activate(self, widget):
196
""" Add file(s)... menu handler. """
197
from add import OliveAdd
198
add = OliveAdd(self.wt, self.wtpath, self.get_selected_right())
201
def on_menuitem_branch_get_activate(self, widget):
202
""" Branch/Get... menu handler. """
203
from branch import OliveBranch
204
branch = OliveBranch(self.get_path())
207
def on_menuitem_branch_checkout_activate(self, widget):
208
""" Branch/Checkout... menu handler. """
209
from checkout import OliveCheckout
210
checkout = OliveCheckout(self.get_path())
213
def on_menuitem_branch_commit_activate(self, widget):
214
""" Branch/Commit... menu handler. """
215
from commit import OliveCommit
216
commit = OliveCommit(self.wt, self.wtpath)
219
def on_menuitem_branch_missing_revisions_activate(self, widget):
220
""" Branch/Missing revisions menu handler. """
221
local_branch = self.wt.branch
223
other_branch = local_branch.get_parent()
224
if other_branch is None:
225
error_dialog(_('Parent location is unknown'),
226
_('Cannot determine missing revisions if no parent location is known.'))
229
remote_branch = Branch.open(other_branch)
231
if remote_branch.base == local_branch.base:
232
remote_branch = local_branch
234
ret = len(local_branch.missing_revisions(remote_branch))
237
info_dialog(_('There are missing revisions'),
238
_('%d revision(s) missing.') % ret)
240
info_dialog(_('Local branch up to date'),
241
_('There are no missing revisions.'))
243
def on_menuitem_branch_pull_activate(self, widget):
244
""" Branch/Pull menu handler. """
245
branch_to = self.wt.branch
247
location = branch_to.get_parent()
249
error_dialog(_('Parent location is unknown'),
250
_('Pulling is not possible until there is a parent location.'))
254
branch_from = Branch.open(location)
255
except errors.NotBranchError:
256
error_dialog(_('Directory is not a branch'),
257
_('You can perform this action only in a branch.'))
259
if branch_to.get_parent() is None:
260
branch_to.set_parent(branch_from.base)
262
#old_rh = branch_to.revision_history()
263
#if tree_to is not None:
264
# tree_to.pull(branch_from)
266
# branch_to.pull(branch_from)
267
branch_to.pull(branch_from)
269
# TODO: get the number of pulled revisions
272
info_dialog(_('Pull successful'), _('%d revision(s) pulled.') % ret)
274
def on_menuitem_branch_push_activate(self, widget):
275
""" Branch/Push... menu handler. """
276
from push import OlivePush
277
push = OlivePush(self.wt.branch)
280
def on_menuitem_branch_status_activate(self, widget):
281
""" Branch/Status... menu handler. """
282
from status import OliveStatus
283
status = OliveStatus(self.wt, self.wtpath)
286
def on_menuitem_branch_initialize_activate(self, widget):
287
""" Initialize current directory. """
288
import bzrlib.bzrdir as bzrdir
291
if not os.path.exists(self.path):
295
existing_bzrdir = bzrdir.BzrDir.open(self.path)
296
except errors.NotBranchError:
297
bzrdir.BzrDir.create_branch_convenience(self.path)
299
if existing_bzrdir.has_branch():
300
if existing_bzrdir.has_workingtree():
301
raise errors.AlreadyBranchError(self.path)
303
raise errors.BranchExistsWithoutWorkingTree(self.path)
305
existing_bzrdir.create_branch()
306
existing_bzrdir.create_workingtree()
307
except errors.AlreadyBranchError, errmsg:
308
error_dialog(_('Directory is already a branch'),
309
_('The current directory (%s) is already a branch.\nYou can start using it, or initialize another directory.') % errmsg)
310
except errors.BranchExistsWithoutWorkingTree, errmsg:
311
error_dialog(_('Branch without a working tree'),
312
_('The current directory (%s)\nis a branch without a working tree.') % errmsg)
314
info_dialog(_('Initialize successful'),
315
_('Directory successfully initialized.'))
318
def on_menuitem_file_make_directory_activate(self, widget):
319
""" File/Make directory... menu handler. """
320
from mkdir import OliveMkdir
321
mkdir = OliveMkdir(self.wt, self.wtpath)
324
def on_menuitem_file_move_activate(self, widget):
325
""" File/Move... menu handler. """
326
from move import OliveMove
327
move = OliveMove(self.wt, self.wtpath, self.get_selected_right())
330
def on_menuitem_file_rename_activate(self, widget):
331
""" File/Rename... menu handler. """
332
from rename import OliveRename
333
rename = OliveRename(self.wt, self.wtpath, self.get_selected_right())
336
def on_menuitem_remove_file_activate(self, widget):
337
""" Remove (unversion) selected file. """
338
from remove import OliveRemove
339
remove = OliveRemove(self.wt, self.wtpath, self.get_selected_right())
342
def on_menuitem_stats_diff_activate(self, widget):
343
""" Statistics/Differences... menu handler. """
344
from bzrlib.plugins.gtk.viz.diffwin import DiffWindow
345
window = DiffWindow()
346
parent_tree = self.wt.branch.repository.revision_tree(self.wt.branch.last_revision())
347
window.set_diff(self.wt.branch.nick, self.wt, parent_tree)
350
def on_menuitem_stats_infos_activate(self, widget):
351
""" Statistics/Informations... menu handler. """
352
from info import OliveInfo
353
info = OliveInfo(self.wt)
356
def on_menuitem_stats_log_activate(self, widget):
357
""" Statistics/Log... menu handler. """
358
from bzrlib.plugins.gtk.viz.branchwin import BranchWindow
359
window = BranchWindow()
360
window.set_branch(self.wt.branch, self.wt.branch.last_revision(), None)
363
def on_menuitem_view_refresh_activate(self, widget):
364
""" View/Refresh menu handler. """
365
# Refresh the left pane
367
# Refresh the right pane
370
def on_menuitem_view_show_hidden_files_activate(self, widget):
371
""" View/Show hidden files menu handler. """
372
self.pref.set_preference('dotted_files', widget.get_active())
374
def on_treeview_left_button_press_event(self, widget, event):
375
""" Occurs when somebody right-clicks in the bookmark list. """
376
if event.button == 3:
377
# Don't show context with nothing selected
378
if self.get_selected_left() == None:
382
from menu import OliveMenu
383
menu = OliveMenu(self.get_path(), self.get_selected_left())
385
menu.left_context_menu().popup(None, None, None, 0,
388
def on_treeview_left_row_activated(self, treeview, path, view_column):
389
""" Occurs when somebody double-clicks or enters an item in the
392
newdir = self.get_selected_left()
396
self.set_path(newdir)
399
def on_treeview_right_button_press_event(self, widget, event):
400
""" Occurs when somebody right-clicks in the file list. """
401
if event.button == 3:
403
from menu import OliveMenu
404
menu = OliveMenu(self.get_path(), self.get_selected_right())
406
m_add = menu.ui.get_widget('/context_right/add')
407
m_remove = menu.ui.get_widget('/context_right/remove')
408
m_commit = menu.ui.get_widget('/context_right/commit')
409
m_diff = menu.ui.get_widget('/context_right/diff')
410
# check if we're in a branch
412
from bzrlib.branch import Branch
413
Branch.open_containing(self.get_path())
414
m_add.set_sensitive(True)
415
m_remove.set_sensitive(True)
416
m_commit.set_sensitive(True)
417
m_diff.set_sensitive(True)
418
except errors.NotBranchError:
419
m_add.set_sensitive(False)
420
m_remove.set_sensitive(False)
421
m_commit.set_sensitive(False)
422
m_diff.set_sensitive(False)
423
menu.right_context_menu().popup(None, None, None, 0,
426
def on_treeview_right_row_activated(self, treeview, path, view_column):
427
""" Occurs when somebody double-clicks or enters an item in the
431
from launch import launch
433
newdir = self.get_selected_right()
436
self.set_path(os.path.split(self.get_path())[0])
438
fullpath = self.get_path() + os.sep + newdir
439
if os.path.isdir(fullpath):
440
# selected item is an existant directory
441
self.set_path(fullpath)
447
def on_window_main_delete_event(self, widget, event=None):
448
""" Do some stuff before exiting. """
449
width, height = self.window_main.get_size()
450
self.pref.set_preference('window_width', width)
451
self.pref.set_preference('window_height', height)
452
x, y = self.window_main.get_position()
453
self.pref.set_preference('window_x', x)
454
self.pref.set_preference('window_y', y)
455
self.pref.set_preference('paned_position',
456
self.hpaned_main.get_position())
459
self.window_main.destroy()
461
def _load_left(self):
462
""" Load data into the left panel. (Bookmarks) """
464
treestore = gtk.TreeStore(str, str)
467
bookmarks = self.pref.get_bookmarks()
469
# Add them to the TreeStore
470
titer = treestore.append(None, [_('Bookmarks'), None])
471
for item in bookmarks:
472
title = self.pref.get_bookmark_title(item)
473
treestore.append(titer, [title, item])
475
# Create the column and add it to the TreeView
476
self.treeview_left.set_model(treestore)
477
tvcolumn_bookmark = gtk.TreeViewColumn(_('Bookmark'))
478
self.treeview_left.append_column(tvcolumn_bookmark)
481
cell = gtk.CellRendererText()
482
tvcolumn_bookmark.pack_start(cell, True)
483
tvcolumn_bookmark.add_attribute(cell, 'text', 0)
486
self.treeview_left.expand_all()
488
def _load_right(self):
489
""" Load data into the right panel. (Filelist) """
491
liststore = gtk.ListStore(str, str, str)
496
# Fill the appropriate lists
497
dotted_files = self.pref.get_preference('dotted_files', 'bool')
498
for item in os.listdir(self.path):
499
if not dotted_files and item[0] == '.':
501
if os.path.isdir(self.path + os.sep + item):
510
if not self.notbranch:
511
branch = self.wt.branch
512
tree2 = self.wt.branch.repository.revision_tree(branch.last_revision())
514
delta = self.wt.changes_from(tree2, want_unchanged=True)
516
# Add'em to the ListStore
518
liststore.append([gtk.STOCK_DIRECTORY, item, ''])
521
if not self.notbranch:
522
filename = self.wt.relpath(self.path + os.sep + item)
524
for rpath, rpathnew, id, kind, text_modified, meta_modified in delta.renamed:
525
if rpathnew == filename:
527
for rpath, id, kind in delta.added:
528
if rpath == filename:
530
for rpath, id, kind in delta.removed:
531
if rpath == filename:
533
for rpath, id, kind, text_modified, meta_modified in delta.modified:
534
if rpath == filename:
536
for rpath, id, kind in delta.unchanged:
537
if rpath == filename:
541
# status = fileops.status(path + os.sep + item)
542
#except errors.PermissionDenied:
545
if status == 'renamed':
547
elif status == 'removed':
549
elif status == 'added':
551
elif status == 'modified':
553
elif status == 'unchanged':
557
liststore.append([gtk.STOCK_FILE, item, st])
559
# Create the columns and add them to the TreeView
560
self.treeview_right.set_model(liststore)
561
tvcolumn_filename = gtk.TreeViewColumn(_('Filename'))
562
tvcolumn_status = gtk.TreeViewColumn(_('Status'))
563
self.treeview_right.append_column(tvcolumn_filename)
564
self.treeview_right.append_column(tvcolumn_status)
567
cellpb = gtk.CellRendererPixbuf()
568
cell = gtk.CellRendererText()
569
tvcolumn_filename.pack_start(cellpb, False)
570
tvcolumn_filename.pack_start(cell, True)
571
tvcolumn_filename.set_attributes(cellpb, stock_id=0)
572
tvcolumn_filename.add_attribute(cell, 'text', 1)
573
tvcolumn_status.pack_start(cell, True)
574
tvcolumn_status.add_attribute(cell, 'text', 2)
577
self.set_sensitivity()
579
def get_selected_right(self):
580
""" Get the selected filename. """
581
treeselection = self.treeview_right.get_selection()
582
(model, iter) = treeselection.get_selected()
587
return model.get_value(iter, 1)
589
def get_selected_left(self):
590
""" Get the selected bookmark. """
591
treeselection = self.treeview_left.get_selection()
592
(model, iter) = treeselection.get_selected()
597
return model.get_value(iter, 1)
599
def set_statusbar(self, message):
600
""" Set the statusbar message. """
601
self.statusbar.push(self.context_id, message)
603
def clear_statusbar(self):
604
""" Clean the last message from the statusbar. """
605
self.statusbar.pop(self.context_id)
607
def set_sensitivity(self):
608
""" Set menu and toolbar sensitivity. """
609
self.menuitem_branch_init.set_sensitive(self.notbranch)
610
self.menuitem_branch_get.set_sensitive(self.notbranch)
611
self.menuitem_branch_checkout.set_sensitive(self.notbranch)
612
self.menuitem_branch_pull.set_sensitive(not self.notbranch)
613
self.menuitem_branch_push.set_sensitive(not self.notbranch)
614
self.menuitem_branch_commit.set_sensitive(not self.notbranch)
615
self.menuitem_branch_status.set_sensitive(not self.notbranch)
616
self.menuitem_branch_missing.set_sensitive(not self.notbranch)
617
self.menuitem_stats.set_sensitive(not self.notbranch)
618
self.menuitem_add_files.set_sensitive(not self.notbranch)
619
self.menuitem_remove_files.set_sensitive(not self.notbranch)
620
self.menuitem_file_make_directory.set_sensitive(not self.notbranch)
621
self.menuitem_file_rename.set_sensitive(not self.notbranch)
622
self.menuitem_file_move.set_sensitive(not self.notbranch)
623
#self.menutoolbutton_diff.set_sensitive(True)
624
self.toolbutton_diff.set_sensitive(not self.notbranch)
625
self.toolbutton_log.set_sensitive(not self.notbranch)
626
self.toolbutton_commit.set_sensitive(not self.notbranch)
627
self.toolbutton_pull.set_sensitive(not self.notbranch)
628
self.toolbutton_push.set_sensitive(not self.notbranch)
630
def refresh_left(self):
631
""" Refresh the bookmark list. """
633
# Get TreeStore and clear it
634
treestore = self.treeview_left.get_model()
637
# Re-read preferences
641
bookmarks = self.pref.get_bookmarks()
643
# Add them to the TreeStore
644
titer = treestore.append(None, [_('Bookmarks'), None])
645
for item in bookmarks:
646
title = self.pref.get_bookmark_title(item)
647
treestore.append(titer, [title, item])
649
# Add the TreeStore to the TreeView
650
self.treeview_left.set_model(treestore)
653
self.treeview_left.expand_all()
655
def refresh_right(self, path=None):
656
""" Refresh the file list. """
657
from bzrlib.workingtree import WorkingTree
660
path = self.get_path()
662
# A workaround for double-clicking Bookmarks
663
if not os.path.exists(path):
666
# Get ListStore and clear it
667
liststore = self.treeview_right.get_model()
673
# Fill the appropriate lists
674
dotted_files = self.pref.get_preference('dotted_files', 'bool')
675
for item in os.listdir(path):
676
if not dotted_files and item[0] == '.':
678
if os.path.isdir(path + os.sep + item):
687
# Try to open the working tree
690
tree1 = WorkingTree.open_containing(path)[0]
691
except errors.NotBranchError:
693
except errors.PermissionDenied:
694
print "DEBUG: permission denied."
697
branch = tree1.branch
698
tree2 = tree1.branch.repository.revision_tree(branch.last_revision())
700
delta = tree1.changes_from(tree2, want_unchanged=True)
702
# Add'em to the ListStore
704
liststore.append([gtk.STOCK_DIRECTORY, item, ''])
708
filename = tree1.relpath(path + os.sep + item)
710
for rpath, rpathnew, id, kind, text_modified, meta_modified in delta.renamed:
711
if rpathnew == filename:
713
for rpath, id, kind in delta.added:
714
if rpath == filename:
716
for rpath, id, kind, text_modified, meta_modified in delta.removed:
717
if rpath == filename:
719
for rpath, id, kind, text_modified, meta_modified in delta.modified:
720
if rpath == filename:
722
for rpath, id, kind in delta.unchanged:
723
if rpath == filename:
727
# status = fileops.status(path + os.sep + item)
728
#except errors.PermissionDenied:
731
if status == 'renamed':
733
elif status == 'removed':
735
elif status == 'added':
737
elif status == 'modified':
739
elif status == 'unchanged':
743
liststore.append([gtk.STOCK_FILE, item, st])
745
# Add the ListStore to the TreeView
746
self.treeview_right.set_model(liststore)
749
self.set_sensitivity()
751
def _harddisks(self):
752
""" Returns hard drive letters under Win32. """
757
if sys.platform == 'win32':
758
print "pyWin32 modules needed to run Olive on Win32."
764
for drive in string.ascii_uppercase:
765
if win32file.GetDriveType(drive+':') == win32file.DRIVE_FIXED:
766
driveletters.append(drive+':')
769
def gen_hard_selector(self):
770
""" Generate the hard drive selector under Win32. """
771
drives = self._harddisks()
773
self.combobox_drive.append_text(drive)
775
def _refresh_drives(self, combobox):
776
model = combobox.get_model()
777
active = combobox.get_active()
779
drive = model[active][0]
780
self.refresh_right(drive + '\\')
784
class OlivePreferences:
785
""" A class which handles Olive's preferences. """
787
""" Initialize the Preferences class. """
788
# Some default options
789
self.defaults = { 'strict_commit' : False,
790
'dotted_files' : False,
791
'window_width' : 700,
792
'window_height' : 400,
795
'paned_position': 200 }
797
# Create a config parser object
798
self.config = ConfigParser.RawConfigParser()
800
# Load the configuration
803
def _get_default(self, option):
804
""" Get the default option for a preference. """
806
ret = self.defaults[option]
813
""" Refresh the configuration. """
814
# First write out the changes
816
# Then load the configuration again
820
""" Just read the configuration. """
821
if sys.platform == 'win32':
822
# Windows - no dotted files
823
self.config.read([os.path.expanduser('~/olive.conf')])
825
self.config.read([os.path.expanduser('~/.olive.conf')])
828
""" Write the configuration to the appropriate files. """
829
if sys.platform == 'win32':
830
# Windows - no dotted files
831
fp = open(os.path.expanduser('~/olive.conf'), 'w')
832
self.config.write(fp)
835
fp = open(os.path.expanduser('~/.olive.conf'), 'w')
836
self.config.write(fp)
839
def get_bookmarks(self):
840
""" Return the list of bookmarks. """
841
bookmarks = self.config.sections()
842
if self.config.has_section('preferences'):
843
bookmarks.remove('preferences')
846
def add_bookmark(self, path):
847
""" Add bookmark. """
849
self.config.add_section(path)
850
except ConfigParser.DuplicateSectionError:
855
def get_bookmark_title(self, path):
856
""" Get bookmark title. """
858
ret = self.config.get(path, 'title')
859
except ConfigParser.NoOptionError:
864
def set_bookmark_title(self, path, title):
865
""" Set bookmark title. """
866
self.config.set(path, 'title', title)
868
def remove_bookmark(self, path):
869
""" Remove bookmark. """
870
return self.config.remove_section(path)
872
def set_preference(self, option, value):
873
""" Set the value of the given option. """
874
if self.config.has_section('preferences'):
875
self.config.set('preferences', option, value)
877
self.config.add_section('preferences')
878
self.config.set('preferences', option, value)
880
def get_preference(self, option, kind='str'):
881
""" Get the value of the given option.
883
:param kind: str/bool/int/float. default: str
885
if self.config.has_option('preferences', option):
887
#return self.config.getboolean('preferences', option)
890
return self.config.getint('preferences', option)
891
elif kind == 'float':
892
return self.config.getfloat('preferences', option)
894
return self.config.get('preferences', option)
897
return self._get_default(option)