52
51
self.compact_view = False
54
self.set_title(branch._get_nick(local=True) + " - revision history")
53
self.set_title(branch.nick + " - revision history")
56
# user-configured window size
57
size = self._load_size('viz-window-size')
61
# Use three-quarters of the screen by default
62
screen = self.get_screen()
63
monitor = screen.get_monitor_geometry(0)
64
width = int(monitor.width * 0.75)
65
height = int(monitor.height * 0.75)
55
# Use three-quarters of the screen by default
56
screen = self.get_screen()
57
monitor = screen.get_monitor_geometry(0)
58
width = int(monitor.width * 0.75)
59
height = int(monitor.height * 0.75)
66
60
self.set_default_size(width, height)
67
self.set_size_request(width/3, height/3)
68
self._save_size_on_destroy(self, 'viz-window-size')
71
icon = self.render_icon_pixbuf(Gtk.STOCK_INDEX, Gtk.IconSize.BUTTON)
63
icon = self.render_icon(gtk.STOCK_INDEX, gtk.ICON_SIZE_BUTTON)
72
64
self.set_icon(icon)
74
Gtk.AccelMap.add_entry("<viz>/Go/Next Revision", Gdk.KEY_Up, Gdk.ModifierType.MOD1_MASK)
75
Gtk.AccelMap.add_entry("<viz>/Go/Previous Revision", Gdk.KEY_Down, Gdk.ModifierType.MOD1_MASK)
76
Gtk.AccelMap.add_entry("<viz>/View/Refresh", Gdk.KEY_F5, 0)
66
gtk.accel_map_add_entry("<viz>/Go/Next Revision", gtk.keysyms.Up, gtk.gdk.MOD1_MASK)
67
gtk.accel_map_add_entry("<viz>/Go/Previous Revision", gtk.keysyms.Down, gtk.gdk.MOD1_MASK)
78
self.accel_group = Gtk.AccelGroup()
69
self.accel_group = gtk.AccelGroup()
79
70
self.add_accel_group(self.accel_group)
81
self.prev_rev_action = Gtk.Action("prev-rev", "_Previous Revision", "Go to the previous revision", Gtk.STOCK_GO_DOWN)
72
gtk.Action.set_tool_item_type(gtk.MenuToolButton)
74
self.prev_rev_action = gtk.Action("prev-rev", "_Previous Revision", "Go to the previous revision", gtk.STOCK_GO_DOWN)
82
75
self.prev_rev_action.set_accel_path("<viz>/Go/Previous Revision")
83
76
self.prev_rev_action.set_accel_group(self.accel_group)
84
77
self.prev_rev_action.connect("activate", self._back_clicked_cb)
85
78
self.prev_rev_action.connect_accelerator()
87
self.next_rev_action = Gtk.Action("next-rev", "_Next Revision", "Go to the next revision", Gtk.STOCK_GO_UP)
80
self.next_rev_action = gtk.Action("next-rev", "_Next Revision", "Go to the next revision", gtk.STOCK_GO_UP)
88
81
self.next_rev_action.set_accel_path("<viz>/Go/Next Revision")
89
82
self.next_rev_action.set_accel_group(self.accel_group)
90
83
self.next_rev_action.connect("activate", self._fwd_clicked_cb)
91
84
self.next_rev_action.connect_accelerator()
93
self.refresh_action = Gtk.Action("refresh", "_Refresh", "Refresh view", Gtk.STOCK_REFRESH)
94
self.refresh_action.set_accel_path("<viz>/View/Refresh")
95
self.refresh_action.set_accel_group(self.accel_group)
96
self.refresh_action.connect("activate", self._refresh_clicked)
97
self.refresh_action.connect_accelerator()
99
self.vbox = self.construct()
101
def _save_size_on_destroy(self, widget, config_name):
102
"""Creates a hook that saves the size of widget to config option
103
config_name when the window is destroyed/closed."""
105
allocation = widget.get_allocation()
106
width, height = allocation.width, allocation.height
107
value = '%sx%s' % (width, height)
108
self.config.set_user_option(config_name, value)
109
self.connect("destroy", save_size)
111
88
def set_revision(self, revid):
112
89
self.treeview.set_revision_id(revid)
114
91
def construct(self):
115
92
"""Construct the window contents."""
116
vbox = Gtk.VBox(spacing=0)
93
vbox = gtk.VBox(spacing=0)
119
# order is important here
120
paned = self.construct_paned()
121
nav = self.construct_navigation()
122
menubar = self.construct_menubar()
124
vbox.pack_start(menubar, False, True, 0)
125
vbox.pack_start(nav, False, True, 0)
126
vbox.pack_start(paned, True, True, 0)
127
vbox.set_focus_child(paned)
96
self.paned = gtk.VPaned()
97
self.paned.pack1(self.construct_top(), resize=True, shrink=False)
98
self.paned.pack2(self.construct_bottom(), resize=False, shrink=True)
101
vbox.pack_start(self.construct_menubar(), expand=False, fill=True)
102
vbox.pack_start(self.construct_navigation(), expand=False, fill=True)
104
vbox.pack_start(self.paned, expand=True, fill=True)
105
vbox.set_focus_child(self.paned)
134
def construct_paned(self):
135
"""Construct the main HPaned/VPaned contents."""
136
if self.config.get_user_option('viz-vertical') == 'True':
137
self.paned = Gtk.Paned.new(Gtk.Orientation.HORIZONTAL)
139
self.paned = Gtk.Paned.new(Gtk.Orientation.VERTICAL)
141
self.paned.pack1(self.construct_top(), resize=False, shrink=True)
142
self.paned.pack2(self.construct_bottom(), resize=True, shrink=False)
147
109
def construct_menubar(self):
148
menubar = Gtk.MenuBar()
110
menubar = gtk.MenuBar()
150
file_menu = Gtk.Menu()
151
file_menuitem = Gtk.MenuItem.new_with_mnemonic("_File")
112
file_menu = gtk.Menu()
113
file_menuitem = gtk.MenuItem("_File")
152
114
file_menuitem.set_submenu(file_menu)
154
file_menu_close = Gtk.ImageMenuItem.new_from_stock(
155
Gtk.STOCK_CLOSE, self.accel_group)
116
file_menu_close = gtk.ImageMenuItem(gtk.STOCK_CLOSE, self.accel_group)
156
117
file_menu_close.connect('activate', lambda x: self.destroy())
158
file_menu_quit = Gtk.ImageMenuItem.new_from_stock(
159
Gtk.STOCK_QUIT, self.accel_group)
160
file_menu_quit.connect('activate', lambda x: Gtk.main_quit())
119
file_menu_quit = gtk.ImageMenuItem(gtk.STOCK_QUIT, self.accel_group)
120
file_menu_quit.connect('activate', lambda x: gtk.main_quit())
162
122
if self._parent is not None:
163
123
file_menu.add(file_menu_close)
164
124
file_menu.add(file_menu_quit)
166
edit_menu = Gtk.Menu()
167
edit_menuitem = Gtk.MenuItem.new_with_mnemonic("_Edit")
126
edit_menu = gtk.Menu()
127
edit_menuitem = gtk.MenuItem("_Edit")
168
128
edit_menuitem.set_submenu(edit_menu)
170
edit_menu_branchopts = Gtk.MenuItem(label="Branch Settings")
130
edit_menu_find = gtk.ImageMenuItem(gtk.STOCK_FIND)
132
edit_menu_branchopts = gtk.MenuItem("Branch Settings")
171
133
edit_menu_branchopts.connect('activate', lambda x: PreferencesWindow(self.branch.get_config()).show())
173
edit_menu_globopts = Gtk.MenuItem(label="Global Settings")
135
edit_menu_globopts = gtk.MenuItem("Global Settings")
174
136
edit_menu_globopts.connect('activate', lambda x: PreferencesWindow().show())
138
edit_menu.add(edit_menu_find)
176
139
edit_menu.add(edit_menu_branchopts)
177
140
edit_menu.add(edit_menu_globopts)
179
view_menu = Gtk.Menu()
180
view_menuitem = Gtk.MenuItem.new_with_mnemonic("_View")
142
view_menu = gtk.Menu()
143
view_menuitem = gtk.MenuItem("_View")
181
144
view_menuitem.set_submenu(view_menu)
183
view_menu_refresh = self.refresh_action.create_menu_item()
184
view_menu_refresh.connect('activate', self._refresh_clicked)
186
view_menu.add(view_menu_refresh)
187
view_menu.add(Gtk.SeparatorMenuItem())
189
view_menu_toolbar = Gtk.CheckMenuItem(label="Show Toolbar")
146
view_menu_toolbar = gtk.CheckMenuItem("Show Toolbar")
190
147
view_menu_toolbar.set_active(True)
191
if self.config.get_user_option('viz-toolbar-visible') == 'False':
192
view_menu_toolbar.set_active(False)
194
148
view_menu_toolbar.connect('toggled', self._toolbar_visibility_changed)
196
view_menu_compact = Gtk.CheckMenuItem(label="Show Compact Graph")
150
view_menu_compact = gtk.CheckMenuItem("Show Compact Graph")
197
151
view_menu_compact.set_active(self.compact_view)
198
152
view_menu_compact.connect('activate', self._brokenlines_toggled_cb)
200
view_menu_vertical = Gtk.CheckMenuItem(label="Side-by-side Layout")
201
view_menu_vertical.set_active(False)
202
if self.config.get_user_option('viz-vertical') == 'True':
203
view_menu_vertical.set_active(True)
204
view_menu_vertical.connect('toggled', self._vertical_layout)
206
view_menu_diffs = Gtk.CheckMenuItem(label="Show Diffs")
207
view_menu_diffs.set_active(False)
208
if self.config.get_user_option('viz-show-diffs') == 'True':
209
view_menu_diffs.set_active(True)
210
view_menu_diffs.connect('toggled', self._diff_visibility_changed)
212
view_menu_wide_diffs = Gtk.CheckMenuItem(label="Wide Diffs")
213
view_menu_wide_diffs.set_active(False)
214
if self.config.get_user_option('viz-wide-diffs') == 'True':
215
view_menu_wide_diffs.set_active(True)
216
view_menu_wide_diffs.connect('toggled', self._diff_placement_changed)
218
view_menu_wrap_diffs = Gtk.CheckMenuItem.new_with_mnemonic(
219
"Wrap _Long Lines in Diffs")
220
view_menu_wrap_diffs.set_active(False)
221
if self.config.get_user_option('viz-wrap-diffs') == 'True':
222
view_menu_wrap_diffs.set_active(True)
223
view_menu_wrap_diffs.connect('toggled', self._diff_wrap_changed)
225
154
view_menu.add(view_menu_toolbar)
226
155
view_menu.add(view_menu_compact)
227
view_menu.add(view_menu_vertical)
228
view_menu.add(Gtk.SeparatorMenuItem())
229
view_menu.add(view_menu_diffs)
230
view_menu.add(view_menu_wide_diffs)
231
view_menu.add(view_menu_wrap_diffs)
232
view_menu.add(Gtk.SeparatorMenuItem())
156
view_menu.add(gtk.SeparatorMenuItem())
234
self.mnu_show_revno_column = Gtk.CheckMenuItem.new_with_mnemonic(
235
"Show Revision _Number Column")
236
self.mnu_show_date_column = Gtk.CheckMenuItem.new_with_mnemonic(
158
self.mnu_show_revno_column = gtk.CheckMenuItem("Show Revision _Number Column")
159
self.mnu_show_date_column = gtk.CheckMenuItem("Show _Date Column")
239
161
# Revision numbers are pointless if there are multiple branches
240
162
if len(self.start_revs) > 1:
247
169
col.connect('toggled', self._col_visibility_changed, name)
248
170
view_menu.add(col)
251
173
go_menu.set_accel_group(self.accel_group)
252
go_menuitem = Gtk.MenuItem.new_with_mnemonic("_Go")
174
go_menuitem = gtk.MenuItem("_Go")
253
175
go_menuitem.set_submenu(go_menu)
255
177
go_menu_next = self.next_rev_action.create_menu_item()
256
178
go_menu_prev = self.prev_rev_action.create_menu_item()
258
tag_image = Gtk.Image()
259
tag_image.set_from_file(icon_path("tag-16.png"))
260
self.go_menu_tags = Gtk.ImageMenuItem.new_with_mnemonic("_Tags")
261
self.go_menu_tags.set_image(tag_image)
262
self.treeview.connect('refreshed', lambda w: self._update_tags())
180
self.go_menu_tags = gtk.MenuItem("_Tags")
264
183
go_menu.add(go_menu_next)
265
184
go_menu.add(go_menu_prev)
266
go_menu.add(Gtk.SeparatorMenuItem())
185
go_menu.add(gtk.SeparatorMenuItem())
267
186
go_menu.add(self.go_menu_tags)
269
self.revision_menu = RevisionMenu(self.branch.repository, [],
270
self.branch, parent=self)
271
revision_menuitem = Gtk.MenuItem.new_with_mnemonic("_Revision")
272
revision_menuitem.set_submenu(self.revision_menu)
274
branch_menu = Gtk.Menu()
275
branch_menuitem = Gtk.MenuItem.new_with_mnemonic("_Branch")
188
revision_menu = gtk.Menu()
189
revision_menuitem = gtk.MenuItem("_Revision")
190
revision_menuitem.set_submenu(revision_menu)
192
revision_menu_diff = gtk.MenuItem("View Changes")
193
revision_menu_diff.connect('activate',
194
lambda w: self.treeview.show_diff())
196
revision_menu_tag = gtk.MenuItem("Tag Revision")
197
revision_menu_tag.connect('activate', self._tag_revision_cb)
199
revision_menu.add(revision_menu_tag)
200
revision_menu.add(revision_menu_diff)
202
branch_menu = gtk.Menu()
203
branch_menuitem = gtk.MenuItem("_Branch")
276
204
branch_menuitem.set_submenu(branch_menu)
278
branch_menu.add(Gtk.MenuItem.new_with_mnemonic("Pu_ll Revisions"))
279
branch_menu.add(Gtk.MenuItem.new_with_mnemonic("Pu_sh Revisions"))
282
from bzrlib.plugins import search
284
mutter("Didn't find search plugin")
286
branch_menu.add(Gtk.SeparatorMenuItem())
288
branch_index_menuitem = Gtk.MenuItem.new_with_mnemonic("_Index")
289
branch_index_menuitem.connect('activate', self._branch_index_cb)
290
branch_menu.add(branch_index_menuitem)
292
branch_search_menuitem = Gtk.MenuItem.new_with_mnemonic("_Search")
293
branch_search_menuitem.connect('activate', self._branch_search_cb)
294
branch_menu.add(branch_search_menuitem)
296
help_menu = Gtk.Menu()
297
help_menuitem = Gtk.MenuItem.new_with_mnemonic("_Help")
206
branch_menu.add(gtk.MenuItem("Pu_ll Revisions"))
207
branch_menu.add(gtk.MenuItem("Pu_sh Revisions"))
209
help_menu = gtk.Menu()
210
help_menuitem = gtk.MenuItem("_Help")
298
211
help_menuitem.set_submenu(help_menu)
300
help_about_menuitem = Gtk.ImageMenuItem.new_from_stock(
301
Gtk.STOCK_ABOUT, self.accel_group)
213
help_about_menuitem = gtk.ImageMenuItem(gtk.STOCK_ABOUT, self.accel_group)
302
214
help_about_menuitem.connect('activate', self._about_dialog_cb)
304
216
help_menu.add(help_about_menuitem)
318
230
"""Construct the top-half of the window."""
319
231
# FIXME: Make broken_line_length configurable
321
self.treeview = TreeView(self.branch, self.start_revs, self.maxnum,
233
self.treeview = TreeView(self.branch, self.start_revs, self.maxnum, self.compact_view)
235
self.treeview.connect('revision-selected',
236
self._treeselection_changed_cb)
237
self.treeview.connect('revision-activated',
238
self._tree_revision_activated)
240
self.treeview.connect('tag-added', lambda w, t, r: self._update_tags())
324
242
for col in ["revno", "date"]:
325
243
option = self.config.get_user_option(col + '-column-visible')
326
244
if option is not None:
327
self.treeview.set_property(col + '-column-visible',
330
self.treeview.set_property(col + '-column-visible', False)
245
self.treeview.set_property(col + '-column-visible', option == 'True')
247
self.treeview.set_property(col + '-column-visible', False)
332
248
self.treeview.show()
334
align = Gtk.Alignment.new(0.0, 0.0, 1.0, 1.0)
250
align = gtk.Alignment(0.0, 0.0, 1.0, 1.0)
335
251
align.set_padding(5, 0, 0, 0)
336
252
align.add(self.treeview)
337
# user-configured size
338
size = self._load_size('viz-graph-size')
341
align.set_size_request(width, height)
343
(width, height) = self.get_size()
344
align.set_size_request(width, int(height / 2.5))
345
self._save_size_on_destroy(align, 'viz-graph-size')
350
257
def construct_navigation(self):
351
258
"""Construct the navigation buttons."""
352
self.toolbar = Gtk.Toolbar()
353
self.toolbar.set_style(Gtk.ToolbarStyle.BOTH_HORIZ)
259
self.toolbar = gtk.Toolbar()
260
self.toolbar.set_style(gtk.TOOLBAR_BOTH_HORIZ)
355
262
self.prev_button = self.prev_rev_action.create_tool_item()
356
263
self.toolbar.insert(self.prev_button, -1)
371
278
def construct_bottom(self):
372
279
"""Construct the bottom half of the window."""
373
if self.config.get_user_option('viz-wide-diffs') == 'True':
374
self.diff_paned = Gtk.Paned.new(Gtk.Orientation.VERTICAL)
376
self.diff_paned = Gtk.Paned.new(Gtk.Orientation.HORIZONTAL)
377
(width, height) = self.get_size()
378
self.diff_paned.set_size_request(20, 20) # shrinkable
380
280
from bzrlib.plugins.gtk.revisionview import RevisionView
381
281
self.revisionview = RevisionView(branch=self.branch)
382
self.revisionview.set_size_request(width/3, int(height / 2.5))
383
# user-configured size
384
size = self._load_size('viz-revisionview-size')
387
self.revisionview.set_size_request(width, height)
388
self._save_size_on_destroy(self.revisionview, 'viz-revisionview-size')
282
(width, height) = self.get_size()
283
self.revisionview.set_size_request(width, int(height / 2.5))
389
284
self.revisionview.show()
390
285
self.revisionview.set_show_callback(self._show_clicked_cb)
391
286
self.revisionview.connect('notify::revision', self._go_clicked_cb)
392
self.treeview.connect('tag-added',
393
lambda w, t, r: self.revisionview.update_tags())
394
self.treeview.connect('revision-selected',
395
self._treeselection_changed_cb)
396
self.treeview.connect('revision-activated',
397
self._tree_revision_activated)
398
self.diff_paned.pack1(self.revisionview)
400
from bzrlib.plugins.gtk.diff import DiffWidget
401
self.diff = DiffWidget()
402
self.diff_paned.pack2(self.diff)
404
self.diff_paned.show_all()
405
if self.config.get_user_option('viz-show-diffs') != 'True':
408
return self.diff_paned
287
self.treeview.connect('tag-added', lambda w, t, r: self.revisionview.update_tags())
288
return self.revisionview
410
290
def _tag_selected_cb(self, menuitem, revid):
411
291
self.treeview.set_revision_id(revid)
562
416
def _toolbar_visibility_changed(self, col):
563
417
if col.get_active():
566
420
self.toolbar.hide()
567
self.config.set_user_option('viz-toolbar-visible', col.get_active())
569
def _vertical_layout(self, col):
570
"""Toggle the layout vertical/horizontal"""
571
self.config.set_user_option('viz-vertical', str(col.get_active()))
574
self.vbox.remove(old)
575
self.vbox.pack_start(
576
self.construct_paned(), True, True, 0)
577
self._make_diff_paned_nonzero_size()
578
self._make_diff_nonzero_size()
580
self.treeview.emit('revision-selected')
582
def _make_diff_paned_nonzero_size(self):
583
"""make sure the diff/revision pane isn't zero-width or zero-height"""
584
alloc = self.diff_paned.get_allocation()
585
if (alloc.width < 10) or (alloc.height < 10):
586
width, height = self.get_size()
587
self.diff_paned.set_size_request(width/3, int(height / 2.5))
589
def _make_diff_nonzero_size(self):
590
"""make sure the diff isn't zero-width or zero-height"""
591
alloc = self.diff.get_allocation()
592
if (alloc.width < 10) or (alloc.height < 10):
593
width, height = self.get_size()
594
self.revisionview.set_size_request(width/3, int(height / 2.5))
596
def _diff_visibility_changed(self, col):
597
"""Hide or show the diff panel."""
600
self._make_diff_nonzero_size()
603
self.config.set_user_option('viz-show-diffs', str(col.get_active()))
604
self.update_diff_panel()
606
def _diff_placement_changed(self, col):
607
"""Toggle the diff panel's position."""
608
self.config.set_user_option('viz-wide-diffs', str(col.get_active()))
610
old = self.paned.get_child2()
611
self.paned.remove(old)
612
self.paned.pack2(self.construct_bottom(), resize=True, shrink=False)
613
self._make_diff_nonzero_size()
615
self.treeview.emit('revision-selected')
617
def _diff_wrap_changed(self, widget):
618
"""Toggle word wrap in the diff widget."""
619
self.config.set_user_option('viz-wrap-diffs', widget.get_active())
620
self.diff._on_wraplines_toggled(widget)
422
def _show_about_cb(self, w):
423
dialog = AboutDialog()
424
dialog.connect('response', lambda d,r: d.destroy())
622
427
def _refresh_clicked(self, w):
623
428
self.treeview.refresh()
625
430
def _update_tags(self):
628
433
if self.branch.supports_tags():
629
434
tags = self.branch.tags.get_tag_dict().items()
630
tags.sort(reverse=True)
631
437
for tag, revid in tags:
632
tag_image = Gtk.Image()
633
tag_image.set_from_file(icon_path('tag-16.png'))
634
tag_item = Gtk.ImageMenuItem.new_with_mnemonic(
635
tag.replace('_', '__'))
636
tag_item.set_image(tag_image)
438
tag_item = gtk.MenuItem(tag, use_underline=False)
637
439
tag_item.connect('activate', self._tag_selected_cb, revid)
638
tag_item.set_sensitive(self.treeview.has_revision_id(revid))
639
440
menu.add(tag_item)
640
441
self.go_menu_tags.set_submenu(menu)
646
447
self.go_menu_tags.show_all()
648
def _load_size(self, name):
649
"""Read and parse 'name' from self.config.
650
The value is a string, formatted as WIDTHxHEIGHT
651
Returns None, or (width, height)
653
size = self.config.get_user_option(name)
655
width, height = [int(num) for num in size.split('x')]
656
# avoid writing config every time we start
660
def show_diff(self, revid, parentid=NULL_REVISION):
449
def show_diff(self, revid=None, parentid=None):
661
450
"""Open a new window to show a diff between the given revisions."""
662
451
from bzrlib.plugins.gtk.diff import DiffWindow
663
452
window = DiffWindow(parent=self)
455
parentid = NULL_REVISION
665
457
rev_tree = self.branch.repository.revision_tree(revid)
666
458
parent_tree = self.branch.repository.revision_tree(parentid)
668
description = revid + " - " + self.branch._get_nick(local=True)
460
description = revid + " - " + self.branch.nick
669
461
window.set_diff(description, rev_tree, parent_tree)
672
def update_diff_panel(self, revision=None, parents=None):
673
"""Show the current revision in the diff panel."""
674
if self.config.get_user_option('viz-show-diffs') != 'True':
677
if not revision: # default to selected row
678
revision = self.treeview.get_revision()
679
if revision == NULL_REVISION:
682
if not parents: # default to selected row's parents
683
parents = self.treeview.get_parents()
684
if len(parents) == 0:
685
parent_id = NULL_REVISION
687
parent_id = parents[0]
689
rev_tree = self.branch.repository.revision_tree(revision.revision_id)
690
parent_tree = self.branch.repository.revision_tree(parent_id)
692
self.diff.set_diff(rev_tree, parent_tree)
693
if self.config.get_user_option('viz-wrap-diffs') == 'True':
694
self.diff._on_wraplines_toggled(wrap=True)