52
52
self.compact_view = False
54
self.set_title(branch.nick + " - revision history")
54
self.set_title(branch._get_nick(local=True) + " - revision history")
56
# Use three-quarters of the screen by default
57
screen = self.get_screen()
58
monitor = screen.get_monitor_geometry(0)
59
width = int(monitor.width * 0.75)
60
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)
61
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')
64
icon = self.render_icon(gtk.STOCK_INDEX, gtk.ICON_SIZE_BUTTON)
71
icon = self.render_icon_pixbuf(Gtk.STOCK_INDEX, Gtk.IconSize.BUTTON)
65
72
self.set_icon(icon)
67
gtk.accel_map_add_entry("<viz>/Go/Next Revision", gtk.keysyms.Up, gtk.gdk.MOD1_MASK)
68
gtk.accel_map_add_entry("<viz>/Go/Previous Revision", gtk.keysyms.Down, gtk.gdk.MOD1_MASK)
69
gtk.accel_map_add_entry("<viz>/View/Refresh", gtk.keysyms.F5, 0)
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)
71
self.accel_group = gtk.AccelGroup()
78
self.accel_group = Gtk.AccelGroup()
72
79
self.add_accel_group(self.accel_group)
74
gtk.Action.set_tool_item_type(gtk.MenuToolButton)
81
if getattr(Gtk.Action, 'set_tool_item_type', None) is not None:
82
Gtk.Action.set_tool_item_type(Gtk.MenuToolButton)
76
self.prev_rev_action = gtk.Action("prev-rev", "_Previous Revision", "Go to the previous revision", gtk.STOCK_GO_DOWN)
84
self.prev_rev_action = Gtk.Action("prev-rev", "_Previous Revision", "Go to the previous revision", Gtk.STOCK_GO_DOWN)
77
85
self.prev_rev_action.set_accel_path("<viz>/Go/Previous Revision")
78
86
self.prev_rev_action.set_accel_group(self.accel_group)
79
87
self.prev_rev_action.connect("activate", self._back_clicked_cb)
80
88
self.prev_rev_action.connect_accelerator()
82
self.next_rev_action = gtk.Action("next-rev", "_Next Revision", "Go to the next revision", gtk.STOCK_GO_UP)
90
self.next_rev_action = Gtk.Action("next-rev", "_Next Revision", "Go to the next revision", Gtk.STOCK_GO_UP)
83
91
self.next_rev_action.set_accel_path("<viz>/Go/Next Revision")
84
92
self.next_rev_action.set_accel_group(self.accel_group)
85
93
self.next_rev_action.connect("activate", self._fwd_clicked_cb)
86
94
self.next_rev_action.connect_accelerator()
88
self.refresh_action = gtk.Action("refresh", "_Refresh", "Refresh view", gtk.STOCK_REFRESH)
96
self.refresh_action = Gtk.Action("refresh", "_Refresh", "Refresh view", Gtk.STOCK_REFRESH)
89
97
self.refresh_action.set_accel_path("<viz>/View/Refresh")
90
98
self.refresh_action.set_accel_group(self.accel_group)
91
99
self.refresh_action.connect("activate", self._refresh_clicked)
92
100
self.refresh_action.connect_accelerator()
102
self.vbox = self.construct()
104
def _save_size_on_destroy(self, widget, config_name):
105
"""Creates a hook that saves the size of widget to config option
106
config_name when the window is destroyed/closed."""
108
width, height = widget.allocation.width, widget.allocation.height
109
value = '%sx%s' % (width, height)
110
self.config.set_user_option(config_name, value)
111
self.connect("destroy", save_size)
96
113
def set_revision(self, revid):
97
114
self.treeview.set_revision_id(revid)
99
116
def construct(self):
100
117
"""Construct the window contents."""
101
vbox = gtk.VBox(spacing=0)
118
vbox = Gtk.VBox(spacing=0)
104
self.paned = gtk.VPaned()
105
self.paned.pack1(self.construct_top(), resize=True, shrink=False)
106
self.paned.pack2(self.construct_bottom(), resize=False, shrink=True)
121
# order is important here
122
paned = self.construct_paned()
123
nav = self.construct_navigation()
124
menubar = self.construct_menubar()
126
vbox.pack_start(menubar, False, True, 0)
127
vbox.pack_start(nav, False, True, 0)
128
vbox.pack_start(paned, True, True, 0)
129
vbox.set_focus_child(paned)
136
def construct_paned(self):
137
"""Construct the main HPaned/VPaned contents."""
138
if self.config.get_user_option('viz-vertical') == 'True':
139
self.paned = Gtk.HPaned()
141
self.paned = Gtk.VPaned()
143
self.paned.pack1(self.construct_top(), resize=False, shrink=True)
144
self.paned.pack2(self.construct_bottom(), resize=True, shrink=False)
107
145
self.paned.show()
109
vbox.pack_start(self.construct_menubar(), expand=False, fill=True)
110
vbox.pack_start(self.construct_navigation(), expand=False, fill=True)
112
vbox.pack_start(self.paned, expand=True, fill=True)
113
vbox.set_focus_child(self.paned)
117
149
def construct_menubar(self):
118
menubar = gtk.MenuBar()
150
menubar = Gtk.MenuBar()
120
file_menu = gtk.Menu()
121
file_menuitem = gtk.MenuItem("_File")
152
file_menu = Gtk.Menu()
153
file_menuitem = Gtk.MenuItem.new_with_mnemonic("_File")
122
154
file_menuitem.set_submenu(file_menu)
124
file_menu_close = gtk.ImageMenuItem(gtk.STOCK_CLOSE, self.accel_group)
156
file_menu_close = Gtk.ImageMenuItem.new_from_stock(
157
Gtk.STOCK_CLOSE, self.accel_group)
125
158
file_menu_close.connect('activate', lambda x: self.destroy())
127
file_menu_quit = gtk.ImageMenuItem(gtk.STOCK_QUIT, self.accel_group)
128
file_menu_quit.connect('activate', lambda x: gtk.main_quit())
160
file_menu_quit = Gtk.ImageMenuItem.new_from_stock(
161
Gtk.STOCK_QUIT, self.accel_group)
162
file_menu_quit.connect('activate', lambda x: Gtk.main_quit())
130
164
if self._parent is not None:
131
165
file_menu.add(file_menu_close)
132
166
file_menu.add(file_menu_quit)
134
edit_menu = gtk.Menu()
135
edit_menuitem = gtk.MenuItem("_Edit")
168
edit_menu = Gtk.Menu()
169
edit_menuitem = Gtk.MenuItem.new_with_mnemonic("_Edit")
136
170
edit_menuitem.set_submenu(edit_menu)
138
edit_menu_find = gtk.ImageMenuItem(gtk.STOCK_FIND)
140
edit_menu_branchopts = gtk.MenuItem("Branch Settings")
172
edit_menu_branchopts = Gtk.MenuItem(label="Branch Settings")
141
173
edit_menu_branchopts.connect('activate', lambda x: PreferencesWindow(self.branch.get_config()).show())
143
edit_menu_globopts = gtk.MenuItem("Global Settings")
175
edit_menu_globopts = Gtk.MenuItem(label="Global Settings")
144
176
edit_menu_globopts.connect('activate', lambda x: PreferencesWindow().show())
146
edit_menu.add(edit_menu_find)
147
178
edit_menu.add(edit_menu_branchopts)
148
179
edit_menu.add(edit_menu_globopts)
150
view_menu = gtk.Menu()
151
view_menuitem = gtk.MenuItem("_View")
181
view_menu = Gtk.Menu()
182
view_menuitem = Gtk.MenuItem.new_with_mnemonic("_View")
152
183
view_menuitem.set_submenu(view_menu)
154
185
view_menu_refresh = self.refresh_action.create_menu_item()
155
186
view_menu_refresh.connect('activate', self._refresh_clicked)
157
188
view_menu.add(view_menu_refresh)
158
view_menu.add(gtk.SeparatorMenuItem())
189
view_menu.add(Gtk.SeparatorMenuItem())
160
view_menu_toolbar = gtk.CheckMenuItem("Show Toolbar")
191
view_menu_toolbar = Gtk.CheckMenuItem(label="Show Toolbar")
161
192
view_menu_toolbar.set_active(True)
193
if self.config.get_user_option('viz-toolbar-visible') == 'False':
194
view_menu_toolbar.set_active(False)
162
196
view_menu_toolbar.connect('toggled', self._toolbar_visibility_changed)
164
view_menu_compact = gtk.CheckMenuItem("Show Compact Graph")
198
view_menu_compact = Gtk.CheckMenuItem(label="Show Compact Graph")
165
199
view_menu_compact.set_active(self.compact_view)
166
200
view_menu_compact.connect('activate', self._brokenlines_toggled_cb)
202
view_menu_vertical = Gtk.CheckMenuItem(label="Side-by-side Layout")
203
view_menu_vertical.set_active(False)
204
if self.config.get_user_option('viz-vertical') == 'True':
205
view_menu_vertical.set_active(True)
206
view_menu_vertical.connect('toggled', self._vertical_layout)
208
view_menu_diffs = Gtk.CheckMenuItem(label="Show Diffs")
209
view_menu_diffs.set_active(False)
210
if self.config.get_user_option('viz-show-diffs') == 'True':
211
view_menu_diffs.set_active(True)
212
view_menu_diffs.connect('toggled', self._diff_visibility_changed)
214
view_menu_wide_diffs = Gtk.CheckMenuItem(label="Wide Diffs")
215
view_menu_wide_diffs.set_active(False)
216
if self.config.get_user_option('viz-wide-diffs') == 'True':
217
view_menu_wide_diffs.set_active(True)
218
view_menu_wide_diffs.connect('toggled', self._diff_placement_changed)
220
view_menu_wrap_diffs = Gtk.CheckMenuItem.new_with_mnemonic(
221
"Wrap _Long Lines in Diffs")
222
view_menu_wrap_diffs.set_active(False)
223
if self.config.get_user_option('viz-wrap-diffs') == 'True':
224
view_menu_wrap_diffs.set_active(True)
225
view_menu_wrap_diffs.connect('toggled', self._diff_wrap_changed)
168
227
view_menu.add(view_menu_toolbar)
169
228
view_menu.add(view_menu_compact)
170
view_menu.add(gtk.SeparatorMenuItem())
229
view_menu.add(view_menu_vertical)
230
view_menu.add(Gtk.SeparatorMenuItem())
231
view_menu.add(view_menu_diffs)
232
view_menu.add(view_menu_wide_diffs)
233
view_menu.add(view_menu_wrap_diffs)
234
view_menu.add(Gtk.SeparatorMenuItem())
172
self.mnu_show_revno_column = gtk.CheckMenuItem("Show Revision _Number Column")
173
self.mnu_show_date_column = gtk.CheckMenuItem("Show _Date Column")
236
self.mnu_show_revno_column = Gtk.CheckMenuItem.new_with_mnemonic(
237
"Show Revision _Number Column")
238
self.mnu_show_date_column = Gtk.CheckMenuItem.new_with_mnemonic(
175
241
# Revision numbers are pointless if there are multiple branches
176
242
if len(self.start_revs) > 1:
183
249
col.connect('toggled', self._col_visibility_changed, name)
184
250
view_menu.add(col)
187
253
go_menu.set_accel_group(self.accel_group)
188
go_menuitem = gtk.MenuItem("_Go")
254
go_menuitem = Gtk.MenuItem.new_with_mnemonic("_Go")
189
255
go_menuitem.set_submenu(go_menu)
191
257
go_menu_next = self.next_rev_action.create_menu_item()
192
258
go_menu_prev = self.prev_rev_action.create_menu_item()
194
tag_image = gtk.Image()
260
tag_image = Gtk.Image()
195
261
tag_image.set_from_file(icon_path("tag-16.png"))
196
self.go_menu_tags = gtk.ImageMenuItem("_Tags")
262
self.go_menu_tags = Gtk.ImageMenuItem.new_with_mnemonic("_Tags")
197
263
self.go_menu_tags.set_image(tag_image)
264
self.treeview.connect('refreshed', lambda w: self._update_tags())
200
266
go_menu.add(go_menu_next)
201
267
go_menu.add(go_menu_prev)
202
go_menu.add(gtk.SeparatorMenuItem())
268
go_menu.add(Gtk.SeparatorMenuItem())
203
269
go_menu.add(self.go_menu_tags)
205
revision_menu = gtk.Menu()
206
revision_menuitem = gtk.MenuItem("_Revision")
207
revision_menuitem.set_submenu(revision_menu)
209
revision_menu_diff = gtk.MenuItem("View Changes")
210
revision_menu_diff.connect('activate',
213
revision_menu_compare = gtk.MenuItem("Compare with...")
214
revision_menu_compare.connect('activate',
215
self._compare_with_cb)
217
revision_menu_tag = gtk.MenuItem("Tag Revision")
218
revision_menu_tag.connect('activate', self._tag_revision_cb)
220
revision_menu.add(revision_menu_tag)
221
revision_menu.add(revision_menu_diff)
222
revision_menu.add(revision_menu_compare)
224
branch_menu = gtk.Menu()
225
branch_menuitem = gtk.MenuItem("_Branch")
271
self.revision_menu = RevisionMenu(self.branch.repository, [],
272
self.branch, parent=self)
273
revision_menuitem = Gtk.MenuItem.new_with_mnemonic("_Revision")
274
revision_menuitem.set_submenu(self.revision_menu)
276
branch_menu = Gtk.Menu()
277
branch_menuitem = Gtk.MenuItem.new_with_mnemonic("_Branch")
226
278
branch_menuitem.set_submenu(branch_menu)
228
branch_menu.add(gtk.MenuItem("Pu_ll Revisions"))
229
branch_menu.add(gtk.MenuItem("Pu_sh Revisions"))
231
help_menu = gtk.Menu()
232
help_menuitem = gtk.MenuItem("_Help")
280
branch_menu.add(Gtk.MenuItem.new_with_mnemonic("Pu_ll Revisions"))
281
branch_menu.add(Gtk.MenuItem.new_with_mnemonic("Pu_sh Revisions"))
284
from bzrlib.plugins import search
286
mutter("Didn't find search plugin")
288
branch_menu.add(Gtk.SeparatorMenuItem())
290
branch_index_menuitem = Gtk.MenuItem.new_with_mnemonic("_Index")
291
branch_index_menuitem.connect('activate', self._branch_index_cb)
292
branch_menu.add(branch_index_menuitem)
294
branch_search_menuitem = Gtk.MenuItem.new_with_mnemonic("_Search")
295
branch_search_menuitem.connect('activate', self._branch_search_cb)
296
branch_menu.add(branch_search_menuitem)
298
help_menu = Gtk.Menu()
299
help_menuitem = Gtk.MenuItem.new_with_mnemonic("_Help")
233
300
help_menuitem.set_submenu(help_menu)
235
help_about_menuitem = gtk.ImageMenuItem(gtk.STOCK_ABOUT, self.accel_group)
302
help_about_menuitem = Gtk.ImageMenuItem.new_from_stock(
303
Gtk.STOCK_ABOUT, self.accel_group)
236
304
help_about_menuitem.connect('activate', self._about_dialog_cb)
238
306
help_menu.add(help_about_menuitem)
252
320
"""Construct the top-half of the window."""
253
321
# FIXME: Make broken_line_length configurable
255
self.treeview = TreeView(self.branch, self.start_revs, self.maxnum, self.compact_view)
257
self.treeview.connect('revision-selected',
258
self._treeselection_changed_cb)
259
self.treeview.connect('revision-activated',
260
self._tree_revision_activated)
262
self.treeview.connect('tag-added', lambda w, t, r: self._update_tags())
323
self.treeview = TreeView(self.branch, self.start_revs, self.maxnum,
264
326
for col in ["revno", "date"]:
265
327
option = self.config.get_user_option(col + '-column-visible')
266
328
if option is not None:
267
self.treeview.set_property(col + '-column-visible', option == 'True')
329
self.treeview.set_property(col + '-column-visible',
269
332
self.treeview.set_property(col + '-column-visible', False)
271
334
self.treeview.show()
273
align = gtk.Alignment(0.0, 0.0, 1.0, 1.0)
336
align = Gtk.Alignment.new(0.0, 0.0, 1.0, 1.0)
274
337
align.set_padding(5, 0, 0, 0)
275
338
align.add(self.treeview)
339
# user-configured size
340
size = self._load_size('viz-graph-size')
343
align.set_size_request(width, height)
345
(width, height) = self.get_size()
346
align.set_size_request(width, int(height / 2.5))
347
self._save_size_on_destroy(align, 'viz-graph-size')
280
352
def construct_navigation(self):
281
353
"""Construct the navigation buttons."""
282
self.toolbar = gtk.Toolbar()
283
self.toolbar.set_style(gtk.TOOLBAR_BOTH_HORIZ)
354
self.toolbar = Gtk.Toolbar()
355
self.toolbar.set_style(Gtk.ToolbarStyle.BOTH_HORIZ)
285
357
self.prev_button = self.prev_rev_action.create_tool_item()
286
358
self.toolbar.insert(self.prev_button, -1)
301
373
def construct_bottom(self):
302
374
"""Construct the bottom half of the window."""
375
if self.config.get_user_option('viz-wide-diffs') == 'True':
376
self.diff_paned = Gtk.VPaned()
378
self.diff_paned = Gtk.HPaned()
379
(width, height) = self.get_size()
380
self.diff_paned.set_size_request(20, 20) # shrinkable
303
382
from bzrlib.plugins.gtk.revisionview import RevisionView
304
383
self.revisionview = RevisionView(branch=self.branch)
305
(width, height) = self.get_size()
306
self.revisionview.set_size_request(width, int(height / 2.5))
384
self.revisionview.set_size_request(width/3, int(height / 2.5))
385
# user-configured size
386
size = self._load_size('viz-revisionview-size')
389
self.revisionview.set_size_request(width, height)
390
self._save_size_on_destroy(self.revisionview, 'viz-revisionview-size')
307
391
self.revisionview.show()
308
392
self.revisionview.set_show_callback(self._show_clicked_cb)
309
393
self.revisionview.connect('notify::revision', self._go_clicked_cb)
310
self.treeview.connect('tag-added', lambda w, t, r: self.revisionview.update_tags())
311
return self.revisionview
394
self.treeview.connect('tag-added',
395
lambda w, t, r: self.revisionview.update_tags())
396
self.treeview.connect('revision-selected',
397
self._treeselection_changed_cb)
398
self.treeview.connect('revision-activated',
399
self._tree_revision_activated)
400
self.diff_paned.pack1(self.revisionview)
402
from bzrlib.plugins.gtk.diff import DiffWidget
403
self.diff = DiffWidget()
404
self.diff_paned.pack2(self.diff)
406
self.diff_paned.show_all()
407
if self.config.get_user_option('viz-show-diffs') != 'True':
410
return self.diff_paned
313
412
def _tag_selected_cb(self, menuitem, revid):
314
413
self.treeview.set_revision_id(revid)
468
557
def _toolbar_visibility_changed(self, col):
469
558
if col.get_active():
472
561
self.toolbar.hide()
474
def _show_about_cb(self, w):
475
dialog = AboutDialog()
476
dialog.connect('response', lambda d,r: d.destroy())
562
self.config.set_user_option('viz-toolbar-visible', col.get_active())
564
def _vertical_layout(self, col):
565
"""Toggle the layout vertical/horizontal"""
566
self.config.set_user_option('viz-vertical', str(col.get_active()))
569
self.vbox.remove(old)
570
self.vbox.pack_start(
571
self.construct_paned(), True, True, True, 0)
572
self._make_diff_paned_nonzero_size()
573
self._make_diff_nonzero_size()
575
self.treeview.emit('revision-selected')
577
def _make_diff_paned_nonzero_size(self):
578
"""make sure the diff/revision pane isn't zero-width or zero-height"""
579
alloc = self.diff_paned.get_allocation()
580
if (alloc.width < 10) or (alloc.height < 10):
581
width, height = self.get_size()
582
self.diff_paned.set_size_request(width/3, int(height / 2.5))
584
def _make_diff_nonzero_size(self):
585
"""make sure the diff isn't zero-width or zero-height"""
586
alloc = self.diff.get_allocation()
587
if (alloc.width < 10) or (alloc.height < 10):
588
width, height = self.get_size()
589
self.revisionview.set_size_request(width/3, int(height / 2.5))
591
def _diff_visibility_changed(self, col):
592
"""Hide or show the diff panel."""
595
self._make_diff_nonzero_size()
598
self.config.set_user_option('viz-show-diffs', str(col.get_active()))
599
self.update_diff_panel()
601
def _diff_placement_changed(self, col):
602
"""Toggle the diff panel's position."""
603
self.config.set_user_option('viz-wide-diffs', str(col.get_active()))
605
old = self.paned.get_child2()
606
self.paned.remove(old)
607
self.paned.pack2(self.construct_bottom(), resize=True, shrink=False)
608
self._make_diff_nonzero_size()
610
self.treeview.emit('revision-selected')
612
def _diff_wrap_changed(self, widget):
613
"""Toggle word wrap in the diff widget."""
614
self.config.set_user_option('viz-wrap-diffs', widget.get_active())
615
self.diff._on_wraplines_toggled(widget)
479
617
def _refresh_clicked(self, w):
480
618
self.treeview.refresh()
482
620
def _update_tags(self):
485
623
if self.branch.supports_tags():
486
624
tags = self.branch.tags.get_tag_dict().items()
625
tags.sort(reverse=True)
489
626
for tag, revid in tags:
490
tag_image = gtk.Image()
627
tag_image = Gtk.Image()
491
628
tag_image.set_from_file(icon_path('tag-16.png'))
492
tag_item = gtk.ImageMenuItem(tag.replace('_', '__'))
629
tag_item = Gtk.ImageMenuItem.new_with_mnemonic(
630
tag.replace('_', '__'))
493
631
tag_item.set_image(tag_image)
494
632
tag_item.connect('activate', self._tag_selected_cb, revid)
633
tag_item.set_sensitive(self.treeview.has_revision_id(revid))
495
634
menu.add(tag_item)
496
635
self.go_menu_tags.set_submenu(menu)