63
65
icon = self.render_icon(gtk.STOCK_INDEX, gtk.ICON_SIZE_BUTTON)
64
66
self.set_icon(icon)
68
gtk.accel_map_add_entry("<viz>/Go/Next Revision", gtk.keysyms.Up, gtk.gdk.MOD1_MASK)
69
gtk.accel_map_add_entry("<viz>/Go/Previous Revision", gtk.keysyms.Down, gtk.gdk.MOD1_MASK)
70
gtk.accel_map_add_entry("<viz>/View/Refresh", gtk.keysyms.F5, 0)
66
72
self.accel_group = gtk.AccelGroup()
67
73
self.add_accel_group(self.accel_group)
75
gtk.Action.set_tool_item_type(gtk.MenuToolButton)
77
self.prev_rev_action = gtk.Action("prev-rev", "_Previous Revision", "Go to the previous revision", gtk.STOCK_GO_DOWN)
78
self.prev_rev_action.set_accel_path("<viz>/Go/Previous Revision")
79
self.prev_rev_action.set_accel_group(self.accel_group)
80
self.prev_rev_action.connect("activate", self._back_clicked_cb)
81
self.prev_rev_action.connect_accelerator()
83
self.next_rev_action = gtk.Action("next-rev", "_Next Revision", "Go to the next revision", gtk.STOCK_GO_UP)
84
self.next_rev_action.set_accel_path("<viz>/Go/Next Revision")
85
self.next_rev_action.set_accel_group(self.accel_group)
86
self.next_rev_action.connect("activate", self._fwd_clicked_cb)
87
self.next_rev_action.connect_accelerator()
89
self.refresh_action = gtk.Action("refresh", "_Refresh", "Refresh view", gtk.STOCK_REFRESH)
90
self.refresh_action.set_accel_path("<viz>/View/Refresh")
91
self.refresh_action.set_accel_group(self.accel_group)
92
self.refresh_action.connect("activate", self._refresh_clicked)
93
self.refresh_action.connect_accelerator()
71
def set_revision(self, revision):
72
self.treeview.set_revision(revision)
97
def set_revision(self, revid):
98
self.treeview.set_revision_id(revid)
74
100
def construct(self):
75
101
"""Construct the window contents."""
76
102
vbox = gtk.VBox(spacing=0)
79
top = self.construct_top()
105
self.paned = gtk.VPaned()
106
self.paned.pack1(self.construct_top(), resize=True, shrink=False)
107
self.paned.pack2(self.construct_bottom(), resize=False, shrink=True)
81
110
vbox.pack_start(self.construct_menubar(), expand=False, fill=True)
82
111
vbox.pack_start(self.construct_navigation(), expand=False, fill=True)
83
vbox.pack_start(self.construct_loading_msg(), expand=False, fill=True)
86
paned.pack1(top, resize=True, shrink=False)
87
paned.pack2(self.construct_bottom(), resize=False, shrink=True)
89
vbox.pack_start(paned, expand=True, fill=True)
90
vbox.set_focus_child(paned)
113
vbox.pack_start(self.paned, expand=True, fill=True)
114
vbox.set_focus_child(self.paned)
116
141
edit_menu_branchopts = gtk.MenuItem("Branch Settings")
117
142
edit_menu_branchopts.connect('activate', lambda x: PreferencesWindow(self.branch.get_config()).show())
119
edit_menu_prefs = gtk.ImageMenuItem(gtk.STOCK_PREFERENCES)
120
edit_menu_prefs.connect('activate', lambda x: PreferencesWindow().show())
144
edit_menu_globopts = gtk.MenuItem("Global Settings")
145
edit_menu_globopts.connect('activate', lambda x: PreferencesWindow().show())
122
147
edit_menu.add(edit_menu_find)
123
148
edit_menu.add(edit_menu_branchopts)
124
edit_menu.add(edit_menu_prefs)
149
edit_menu.add(edit_menu_globopts)
126
151
view_menu = gtk.Menu()
127
152
view_menuitem = gtk.MenuItem("_View")
128
153
view_menuitem.set_submenu(view_menu)
155
view_menu_refresh = self.refresh_action.create_menu_item()
156
view_menu_refresh.connect('activate', self._refresh_clicked)
158
view_menu.add(view_menu_refresh)
159
view_menu.add(gtk.SeparatorMenuItem())
130
161
view_menu_toolbar = gtk.CheckMenuItem("Show Toolbar")
131
162
view_menu_toolbar.set_active(True)
132
163
view_menu_toolbar.connect('toggled', self._toolbar_visibility_changed)
165
view_menu_compact = gtk.CheckMenuItem("Show Compact Graph")
166
view_menu_compact.set_active(self.compact_view)
167
view_menu_compact.connect('activate', self._brokenlines_toggled_cb)
134
169
view_menu.add(view_menu_toolbar)
170
view_menu.add(view_menu_compact)
135
171
view_menu.add(gtk.SeparatorMenuItem())
137
for (label, name) in [("Revision _Number", "revno"), ("_Date", "date")]:
138
col = gtk.CheckMenuItem("Show " + label + " Column")
173
self.mnu_show_revno_column = gtk.CheckMenuItem("Show Revision _Number Column")
174
self.mnu_show_date_column = gtk.CheckMenuItem("Show _Date Column")
176
# Revision numbers are pointless if there are multiple branches
177
if len(self.start_revs) > 1:
178
self.mnu_show_revno_column.set_sensitive(False)
179
self.treeview.set_property('revno-column-visible', False)
181
for (col, name) in [(self.mnu_show_revno_column, "revno"),
182
(self.mnu_show_date_column, "date")]:
139
183
col.set_active(self.treeview.get_property(name + "-column-visible"))
140
184
col.connect('toggled', self._col_visibility_changed, name)
141
185
view_menu.add(col)
143
187
go_menu = gtk.Menu()
188
go_menu.set_accel_group(self.accel_group)
144
189
go_menuitem = gtk.MenuItem("_Go")
145
190
go_menuitem.set_submenu(go_menu)
147
go_menu_back = gtk.ImageMenuItem(gtk.STOCK_GO_DOWN)
148
go_menu_back.connect("activate", self._back_clicked_cb)
150
go_menu_forward = gtk.ImageMenuItem(gtk.STOCK_GO_UP)
151
go_menu_forward.connect("activate", self._fwd_clicked_cb)
153
tags_menu = gtk.Menu()
154
go_menu_tags = gtk.MenuItem("_Tags")
155
go_menu_tags.set_submenu(tags_menu)
157
if self.branch.supports_tags():
158
for (tag, revid) in self.branch.tags.get_tag_dict().items():
159
tag_item = gtk.MenuItem(tag)
160
tag_item.connect('activate', self._tag_selected_cb, revid)
161
tags_menu.add(tag_item)
163
go_menu.add(go_menu_back)
164
go_menu.add(go_menu_forward)
192
go_menu_next = self.next_rev_action.create_menu_item()
193
go_menu_prev = self.prev_rev_action.create_menu_item()
195
tag_image = gtk.Image()
196
tag_image.set_from_file(icon_path("tag-16.png"))
197
self.go_menu_tags = gtk.ImageMenuItem("_Tags")
198
self.go_menu_tags.set_image(tag_image)
201
go_menu.add(go_menu_next)
202
go_menu.add(go_menu_prev)
165
203
go_menu.add(gtk.SeparatorMenuItem())
166
go_menu.add(go_menu_tags)
204
go_menu.add(self.go_menu_tags)
168
206
revision_menu = gtk.Menu()
169
207
revision_menuitem = gtk.MenuItem("_Revision")
186
229
branch_menu.add(gtk.MenuItem("Pu_ll Revisions"))
187
230
branch_menu.add(gtk.MenuItem("Pu_sh Revisions"))
233
from bzrlib.plugins import search
235
mutter("Didn't find search plugin")
237
branch_index_menuitem = gtk.MenuItem("_Index")
238
branch_index_menuitem.connect('activate', self._branch_index_cb)
239
branch_menu.add(branch_index_menuitem)
241
help_menu = gtk.Menu()
242
help_menuitem = gtk.MenuItem("_Help")
243
help_menuitem.set_submenu(help_menu)
245
help_about_menuitem = gtk.ImageMenuItem(gtk.STOCK_ABOUT, self.accel_group)
246
help_about_menuitem.connect('activate', self._about_dialog_cb)
248
help_menu.add(help_about_menuitem)
189
250
menubar.add(file_menuitem)
190
251
menubar.add(edit_menuitem)
191
252
menubar.add(view_menuitem)
192
253
menubar.add(go_menuitem)
193
254
menubar.add(revision_menuitem)
194
255
menubar.add(branch_menuitem)
256
menubar.add(help_menuitem)
195
257
menubar.show_all()
199
def construct_loading_msg(self):
200
image_loading = gtk.image_new_from_stock(gtk.STOCK_REFRESH,
201
gtk.ICON_SIZE_BUTTON)
204
label_loading = gtk.Label(_("Please wait, loading ancestral graph..."))
205
label_loading.set_alignment(0.0, 0.5)
208
self.loading_msg_box = gtk.HBox()
209
self.loading_msg_box.set_spacing(5)
210
self.loading_msg_box.set_border_width(5)
211
self.loading_msg_box.pack_start(image_loading, False, False)
212
self.loading_msg_box.pack_start(label_loading, True, True)
213
self.loading_msg_box.show()
215
return self.loading_msg_box
217
261
def construct_top(self):
218
262
"""Construct the top-half of the window."""
219
263
# FIXME: Make broken_line_length configurable
220
if self.compact_view:
225
self.treeview = TreeView(self.branch, self.start, self.maxnum, brokenlines)
227
self.treeview.connect("revision-selected",
265
self.treeview = TreeView(self.branch, self.start_revs, self.maxnum, self.compact_view)
267
self.treeview.connect('revision-selected',
228
268
self._treeselection_changed_cb)
230
self.treeview.connect('revisions-loaded',
231
lambda x: self.loading_msg_box.hide())
269
self.treeview.connect('revision-activated',
270
self._tree_revision_activated)
272
self.treeview.connect('tag-added', lambda w, t, r: self._update_tags())
274
for col in ["revno", "date"]:
275
option = self.config.get_user_option(col + '-column-visible')
276
if option is not None:
277
self.treeview.set_property(col + '-column-visible', option == 'True')
279
self.treeview.set_property(col + '-column-visible', False)
233
281
self.treeview.show()
283
align = gtk.Alignment(0.0, 0.0, 1.0, 1.0)
284
align.set_padding(5, 0, 0, 0)
285
align.add(self.treeview)
237
290
def construct_navigation(self):
238
291
"""Construct the navigation buttons."""
239
292
self.toolbar = gtk.Toolbar()
240
293
self.toolbar.set_style(gtk.TOOLBAR_BOTH_HORIZ)
242
self.back_button = gtk.MenuToolButton(stock_id=gtk.STOCK_GO_DOWN)
243
self.back_button.add_accelerator("clicked", self.accel_group, ord('['),
245
self.back_button.connect("clicked", self._back_clicked_cb)
246
self.toolbar.insert(self.back_button, -1)
295
self.prev_button = self.prev_rev_action.create_tool_item()
296
self.toolbar.insert(self.prev_button, -1)
248
self.fwd_button = gtk.MenuToolButton(stock_id=gtk.STOCK_GO_UP)
249
self.fwd_button.add_accelerator("clicked", self.accel_group, ord(']'),
251
self.fwd_button.connect("clicked", self._fwd_clicked_cb)
252
self.toolbar.insert(self.fwd_button, -1)
298
self.next_button = self.next_rev_action.create_tool_item()
299
self.toolbar.insert(self.next_button, -1)
254
301
self.toolbar.insert(gtk.SeparatorToolItem(), -1)
256
brokenlines_button = gtk.ToggleToolButton()
257
brokenlines_button.set_label("Compact View")
258
brokenlines_button.set_active(self.compact_view)
259
brokenlines_button.set_is_important(True)
260
brokenlines_button.connect('toggled', self._brokenlines_toggled_cb)
261
self.toolbar.insert(brokenlines_button, -1)
303
refresh_button = gtk.ToolButton(gtk.STOCK_REFRESH)
304
refresh_button.connect('clicked', self._refresh_clicked)
305
self.toolbar.insert(refresh_button, -1)
263
307
self.toolbar.show_all()
267
311
def construct_bottom(self):
268
312
"""Construct the bottom half of the window."""
269
313
from bzrlib.plugins.gtk.revisionview import RevisionView
270
self.revisionview = RevisionView(None, tags=[], show_children=True, branch=self.branch)
314
self.revisionview = RevisionView(branch=self.branch)
271
315
(width, height) = self.get_size()
272
316
self.revisionview.set_size_request(width, int(height / 2.5))
273
317
self.revisionview.show()
274
318
self.revisionview.set_show_callback(self._show_clicked_cb)
275
self.revisionview.set_go_callback(self._go_clicked_cb)
319
self.revisionview.connect('notify::revision', self._go_clicked_cb)
320
self.treeview.connect('tag-added', lambda w, t, r: self.revisionview.update_tags())
276
321
return self.revisionview
278
323
def _tag_selected_cb(self, menuitem, revid):
284
329
parents = self.treeview.get_parents()
285
330
children = self.treeview.get_children()
287
if revision is not None:
288
back_menu = gtk.Menu()
332
if revision and revision != NULL_REVISION:
333
prev_menu = gtk.Menu()
289
334
if len(parents) > 0:
290
self.back_button.set_sensitive(True)
335
self.prev_rev_action.set_sensitive(True)
291
336
for parent_id in parents:
292
parent = self.branch.repository.get_revision(parent_id)
294
str = ' (' + parent.properties['branch-nick'] + ')'
337
if parent_id and parent_id != NULL_REVISION:
338
parent = self.branch.repository.get_revision(parent_id)
340
str = ' (' + parent.properties['branch-nick'] + ')'
298
item = gtk.MenuItem(parent.message.split("\n")[0] + str)
299
item.connect('activate', self._set_revision_cb, parent_id)
344
item = gtk.MenuItem(parent.message.split("\n")[0] + str)
345
item.connect('activate', self._set_revision_cb, parent_id)
303
self.back_button.set_sensitive(False)
306
self.back_button.set_menu(back_menu)
308
fwd_menu = gtk.Menu()
349
self.prev_rev_action.set_sensitive(False)
352
self.prev_button.set_menu(prev_menu)
354
next_menu = gtk.Menu()
309
355
if len(children) > 0:
310
self.fwd_button.set_sensitive(True)
356
self.next_rev_action.set_sensitive(True)
311
357
for child_id in children:
312
358
child = self.branch.repository.get_revision(child_id)
318
364
item = gtk.MenuItem(child.message.split("\n")[0] + str)
319
365
item.connect('activate', self._set_revision_cb, child_id)
323
self.fwd_button.set_sensitive(False)
326
self.fwd_button.set_menu(fwd_menu)
329
if self.branch.supports_tags():
330
tagdict = self.branch.tags.get_reverse_tag_dict()
331
if tagdict.has_key(revision.revision_id):
332
tags = tagdict[revision.revision_id]
333
self.revisionview.set_revision(revision, tags, children)
369
self.next_rev_action.set_sensitive(False)
372
self.next_button.set_menu(next_menu)
374
self.revisionview.set_revision(revision)
375
self.revisionview.set_children(children)
377
def _tree_revision_activated(self, widget, path, col):
378
# TODO: more than one parent
379
"""Callback for when a treeview row gets activated."""
380
revision = self.treeview.get_revision()
381
parents = self.treeview.get_parents()
383
if len(parents) == 0:
386
parent_id = parents[0]
388
self.show_diff(revision.revision_id, parent_id)
389
self.treeview.grab_focus()
391
def _menu_diff_cb(self,w):
392
(path, focus) = self.treeview.treeview.get_cursor()
393
revid = self.treeview.model[path][treemodel.REVID]
395
parentids = self.branch.repository.revision_parents(revid)
397
if len(parentids) == 0:
398
parentid = NULL_REVISION
400
parentid = parentids[0]
402
self.show_diff(revid,parentid)
335
404
def _back_clicked_cb(self, *args):
336
405
"""Callback for when the back button is clicked."""
340
409
"""Callback for when the forward button is clicked."""
341
410
self.treeview.forward()
343
def _go_clicked_cb(self, revid):
412
def _go_clicked_cb(self, w, p):
344
413
"""Callback for when the go button for a parent is clicked."""
345
self.treeview.set_revision_id(revid)
414
if self.revisionview.get_revision() is not None:
415
self.treeview.set_revision(self.revisionview.get_revision())
347
417
def _show_clicked_cb(self, revid, parentid):
348
418
"""Callback for when the show button for a parent is clicked."""
349
self.treeview.show_diff(revid, parentid)
419
self.show_diff(revid, parentid)
350
420
self.treeview.grab_focus()
422
def _compare_with_cb(self,w):
423
"""Callback for revision 'compare with' menu. Will show a small
424
dialog with branch revisions to compare with selected revision in TreeView"""
426
from bzrlib.plugins.gtk.revbrowser import RevisionBrowser
428
rb = RevisionBrowser(self.branch,self)
431
if ret == gtk.RESPONSE_OK:
432
(path, focus) = self.treeview.treeview.get_cursor()
433
revid = self.treeview.model[path][treemodel.REVID]
434
self.show_diff(revid, rb.selected_revid)
352
438
def _set_revision_cb(self, w, revision_id):
353
439
self.treeview.set_revision_id(revision_id)
363
449
self.config.set_user_option('viz-compact-view', option)
365
revision = self.treeview.get_revision()
367
self.treeview.destroy()
368
self.paned.pack1(self.construct_top(), resize=True, shrink=False)
370
gobject.idle_add(self.set_revision, revision.revision_id)
450
self.treeview.set_property('compact', self.compact_view)
451
self.treeview.refresh()
372
453
def _tag_revision_cb(self, w):
373
dialog = AddTagDialog(self.branch.repository, self.treeview.get_revision().revision_id, self.branch)
374
response = dialog.run()
375
if response != gtk.RESPONSE_NONE:
378
if response == gtk.RESPONSE_OK:
380
self.branch.lock_write()
381
self.branch.tags.set_tag(dialog.tagname, dialog._revid)
455
self.treeview.set_sensitive(False)
456
dialog = AddTagDialog(self.branch.repository, self.treeview.get_revision().revision_id, self.branch)
457
response = dialog.run()
458
if response != gtk.RESPONSE_NONE:
461
if response == gtk.RESPONSE_OK:
462
self.treeview.add_tag(dialog.tagname, dialog._revid)
467
self.treeview.set_sensitive(True)
469
def _branch_index_cb(self, w):
470
from bzrlib.plugins.search import index as _mod_index
471
_mod_index.index_url(self.branch.base)
473
def _about_dialog_cb(self, w):
474
from bzrlib.plugins.gtk.about import AboutDialog
387
478
def _col_visibility_changed(self, col, property):
479
self.config.set_user_option(property + '-column-visible', col.get_active())
388
480
self.treeview.set_property(property + '-column-visible', col.get_active())
390
482
def _toolbar_visibility_changed(self, col):
392
484
self.toolbar.show()
394
486
self.toolbar.hide()
488
def _show_about_cb(self, w):
489
dialog = AboutDialog()
490
dialog.connect('response', lambda d,r: d.destroy())
493
def _refresh_clicked(self, w):
494
self.treeview.refresh()
496
def _update_tags(self):
499
if self.branch.supports_tags():
500
tags = self.branch.tags.get_tag_dict().items()
503
for tag, revid in tags:
504
tag_image = gtk.Image()
505
tag_image.set_from_file(icon_path('tag-16.png'))
506
tag_item = gtk.ImageMenuItem(tag.replace('_', '__'))
507
tag_item.set_image(tag_image)
508
tag_item.connect('activate', self._tag_selected_cb, revid)
510
self.go_menu_tags.set_submenu(menu)
512
self.go_menu_tags.set_sensitive(len(tags) != 0)
514
self.go_menu_tags.set_sensitive(False)
516
self.go_menu_tags.show_all()
518
def show_diff(self, revid=None, parentid=None):
519
"""Open a new window to show a diff between the given revisions."""
520
from bzrlib.plugins.gtk.diff import DiffWindow
521
window = DiffWindow(parent=self)
524
parentid = NULL_REVISION
526
rev_tree = self.branch.repository.revision_tree(revid)
527
parent_tree = self.branch.repository.revision_tree(parentid)
529
description = revid + " - " + self.branch.nick
530
window.set_diff(description, rev_tree, parent_tree)