51
52
self.compact_view = False
53
self.set_title(branch.nick + " - revision history")
54
self.set_title(branch._get_nick(local=True) + " - revision history")
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)
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)
60
66
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')
63
icon = self.render_icon(gtk.STOCK_INDEX, gtk.ICON_SIZE_BUTTON)
71
icon = self.render_icon_pixbuf(Gtk.STOCK_INDEX, Gtk.IconSize.BUTTON)
64
72
self.set_icon(icon)
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)
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)
69
self.accel_group = gtk.AccelGroup()
78
self.accel_group = Gtk.AccelGroup()
70
79
self.add_accel_group(self.accel_group)
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)
81
self.prev_rev_action = Gtk.Action("prev-rev", "_Previous Revision", "Go to the previous revision", Gtk.STOCK_GO_DOWN)
75
82
self.prev_rev_action.set_accel_path("<viz>/Go/Previous Revision")
76
83
self.prev_rev_action.set_accel_group(self.accel_group)
77
84
self.prev_rev_action.connect("activate", self._back_clicked_cb)
78
85
self.prev_rev_action.connect_accelerator()
80
self.next_rev_action = gtk.Action("next-rev", "_Next Revision", "Go to the next revision", gtk.STOCK_GO_UP)
87
self.next_rev_action = Gtk.Action("next-rev", "_Next Revision", "Go to the next revision", Gtk.STOCK_GO_UP)
81
88
self.next_rev_action.set_accel_path("<viz>/Go/Next Revision")
82
89
self.next_rev_action.set_accel_group(self.accel_group)
83
90
self.next_rev_action.connect("activate", self._fwd_clicked_cb)
84
91
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)
88
111
def set_revision(self, revid):
89
112
self.treeview.set_revision_id(revid)
91
114
def construct(self):
92
115
"""Construct the window contents."""
93
vbox = gtk.VBox(spacing=0)
116
vbox = Gtk.VBox(spacing=0)
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)
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)
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)
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)
109
147
def construct_menubar(self):
110
menubar = gtk.MenuBar()
148
menubar = Gtk.MenuBar()
112
file_menu = gtk.Menu()
113
file_menuitem = gtk.MenuItem("_File")
150
file_menu = Gtk.Menu()
151
file_menuitem = Gtk.MenuItem.new_with_mnemonic("_File")
114
152
file_menuitem.set_submenu(file_menu)
116
file_menu_close = gtk.ImageMenuItem(gtk.STOCK_CLOSE, self.accel_group)
154
file_menu_close = Gtk.ImageMenuItem.new_from_stock(
155
Gtk.STOCK_CLOSE, self.accel_group)
117
156
file_menu_close.connect('activate', lambda x: self.destroy())
119
file_menu_quit = gtk.ImageMenuItem(gtk.STOCK_QUIT, self.accel_group)
120
file_menu_quit.connect('activate', lambda x: gtk.main_quit())
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())
122
162
if self._parent is not None:
123
163
file_menu.add(file_menu_close)
124
164
file_menu.add(file_menu_quit)
126
edit_menu = gtk.Menu()
127
edit_menuitem = gtk.MenuItem("_Edit")
166
edit_menu = Gtk.Menu()
167
edit_menuitem = Gtk.MenuItem.new_with_mnemonic("_Edit")
128
168
edit_menuitem.set_submenu(edit_menu)
130
edit_menu_find = gtk.ImageMenuItem(gtk.STOCK_FIND)
132
edit_menu_branchopts = gtk.MenuItem("Branch Settings")
170
edit_menu_branchopts = Gtk.MenuItem(label="Branch Settings")
133
171
edit_menu_branchopts.connect('activate', lambda x: PreferencesWindow(self.branch.get_config()).show())
135
edit_menu_globopts = gtk.MenuItem("Global Settings")
173
edit_menu_globopts = Gtk.MenuItem(label="Global Settings")
136
174
edit_menu_globopts.connect('activate', lambda x: PreferencesWindow().show())
138
edit_menu.add(edit_menu_find)
139
176
edit_menu.add(edit_menu_branchopts)
140
177
edit_menu.add(edit_menu_globopts)
142
view_menu = gtk.Menu()
143
view_menuitem = gtk.MenuItem("_View")
179
view_menu = Gtk.Menu()
180
view_menuitem = Gtk.MenuItem.new_with_mnemonic("_View")
144
181
view_menuitem.set_submenu(view_menu)
146
view_menu_toolbar = gtk.CheckMenuItem("Show Toolbar")
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")
147
190
view_menu_toolbar.set_active(True)
191
if self.config.get_user_option('viz-toolbar-visible') == 'False':
192
view_menu_toolbar.set_active(False)
148
194
view_menu_toolbar.connect('toggled', self._toolbar_visibility_changed)
150
view_menu_compact = gtk.CheckMenuItem("Show Compact Graph")
196
view_menu_compact = Gtk.CheckMenuItem(label="Show Compact Graph")
151
197
view_menu_compact.set_active(self.compact_view)
152
198
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)
154
225
view_menu.add(view_menu_toolbar)
155
226
view_menu.add(view_menu_compact)
156
view_menu.add(gtk.SeparatorMenuItem())
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())
158
self.mnu_show_revno_column = gtk.CheckMenuItem("Show Revision _Number Column")
159
self.mnu_show_date_column = gtk.CheckMenuItem("Show _Date Column")
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(
161
239
# Revision numbers are pointless if there are multiple branches
162
240
if len(self.start_revs) > 1:
169
247
col.connect('toggled', self._col_visibility_changed, name)
170
248
view_menu.add(col)
173
251
go_menu.set_accel_group(self.accel_group)
174
go_menuitem = gtk.MenuItem("_Go")
252
go_menuitem = Gtk.MenuItem.new_with_mnemonic("_Go")
175
253
go_menuitem.set_submenu(go_menu)
177
255
go_menu_next = self.next_rev_action.create_menu_item()
178
256
go_menu_prev = self.prev_rev_action.create_menu_item()
180
self.go_menu_tags = gtk.MenuItem("_Tags")
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())
183
264
go_menu.add(go_menu_next)
184
265
go_menu.add(go_menu_prev)
185
go_menu.add(gtk.SeparatorMenuItem())
266
go_menu.add(Gtk.SeparatorMenuItem())
186
267
go_menu.add(self.go_menu_tags)
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")
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")
204
276
branch_menuitem.set_submenu(branch_menu)
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")
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")
211
298
help_menuitem.set_submenu(help_menu)
213
help_about_menuitem = gtk.ImageMenuItem(gtk.STOCK_ABOUT, self.accel_group)
300
help_about_menuitem = Gtk.ImageMenuItem.new_from_stock(
301
Gtk.STOCK_ABOUT, self.accel_group)
214
302
help_about_menuitem.connect('activate', self._about_dialog_cb)
216
304
help_menu.add(help_about_menuitem)
230
318
"""Construct the top-half of the window."""
231
319
# FIXME: Make broken_line_length configurable
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())
321
self.treeview = TreeView(self.branch, self.start_revs, self.maxnum,
242
324
for col in ["revno", "date"]:
243
325
option = self.config.get_user_option(col + '-column-visible')
244
326
if option is not None:
245
self.treeview.set_property(col + '-column-visible', option == 'True')
327
self.treeview.set_property(col + '-column-visible',
330
self.treeview.set_property(col + '-column-visible', False)
247
self.treeview.set_property(col + '-column-visible', False)
248
332
self.treeview.show()
250
align = gtk.Alignment(0.0, 0.0, 1.0, 1.0)
334
align = Gtk.Alignment.new(0.0, 0.0, 1.0, 1.0)
251
335
align.set_padding(5, 0, 0, 0)
252
336
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')
257
350
def construct_navigation(self):
258
351
"""Construct the navigation buttons."""
259
self.toolbar = gtk.Toolbar()
260
self.toolbar.set_style(gtk.TOOLBAR_BOTH_HORIZ)
352
self.toolbar = Gtk.Toolbar()
353
self.toolbar.set_style(Gtk.ToolbarStyle.BOTH_HORIZ)
262
355
self.prev_button = self.prev_rev_action.create_tool_item()
263
356
self.toolbar.insert(self.prev_button, -1)
278
371
def construct_bottom(self):
279
372
"""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
280
380
from bzrlib.plugins.gtk.revisionview import RevisionView
281
381
self.revisionview = RevisionView(branch=self.branch)
282
(width, height) = self.get_size()
283
self.revisionview.set_size_request(width, int(height / 2.5))
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')
284
389
self.revisionview.show()
285
390
self.revisionview.set_show_callback(self._show_clicked_cb)
286
391
self.revisionview.connect('notify::revision', self._go_clicked_cb)
287
self.treeview.connect('tag-added', lambda w, t, r: self.revisionview.update_tags())
288
return self.revisionview
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
290
410
def _tag_selected_cb(self, menuitem, revid):
291
411
self.treeview.set_revision_id(revid)
416
562
def _toolbar_visibility_changed(self, col):
417
563
if col.get_active():
420
566
self.toolbar.hide()
422
def _show_about_cb(self, w):
423
dialog = AboutDialog()
424
dialog.connect('response', lambda d,r: d.destroy())
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)
427
622
def _refresh_clicked(self, w):
428
623
self.treeview.refresh()
430
625
def _update_tags(self):
433
628
if self.branch.supports_tags():
434
629
tags = self.branch.tags.get_tag_dict().items()
630
tags.sort(reverse=True)
437
631
for tag, revid in tags:
438
tag_item = gtk.MenuItem(tag, use_underline=False)
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)
439
637
tag_item.connect('activate', self._tag_selected_cb, revid)
638
tag_item.set_sensitive(self.treeview.has_revision_id(revid))
440
639
menu.add(tag_item)
441
640
self.go_menu_tags.set_submenu(menu)
447
646
self.go_menu_tags.show_all()
449
def show_diff(self, revid=None, parentid=None):
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):
450
661
"""Open a new window to show a diff between the given revisions."""
451
662
from bzrlib.plugins.gtk.diff import DiffWindow
452
663
window = DiffWindow(parent=self)
455
parentid = NULL_REVISION
457
665
rev_tree = self.branch.repository.revision_tree(revid)
458
666
parent_tree = self.branch.repository.revision_tree(parentid)
460
description = revid + " - " + self.branch.nick
668
description = revid + " - " + self.branch._get_nick(local=True)
461
669
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)