1
# -*- coding: UTF-8 -*-
4
3
This module contains the code to manage the branch information window,
5
4
which contains both the revision graph and details panes.
8
__copyright__ = "Copyright © 2005 Canonical Ltd."
7
__copyright__ = "Copyright (c) 2005 Canonical Ltd."
9
8
__author__ = "Scott James Remnant <scott@ubuntu.com>"
16
from bzrlib.plugins.gtk.window import Window
17
13
from bzrlib.plugins.gtk import icon_path
18
from bzrlib.plugins.gtk.tags import AddTagDialog
14
from bzrlib.plugins.gtk.branchview import TreeView
19
15
from bzrlib.plugins.gtk.preferences import PreferencesWindow
20
from bzrlib.plugins.gtk.branchview import TreeView, treemodel
21
from bzrlib.revision import Revision, NULL_REVISION
22
from bzrlib.config import BranchConfig
16
from bzrlib.plugins.gtk.revisionmenu import RevisionMenu
17
from bzrlib.plugins.gtk.window import Window
23
19
from bzrlib.config import GlobalConfig
20
from bzrlib.revision import NULL_REVISION
21
from bzrlib.trace import mutter
25
24
class BranchWindow(Window):
52
51
self.compact_view = False
54
self.set_title(branch.nick + " - revision history")
53
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)
55
# user-configured window size
56
size = self._load_size('viz-window-size')
60
# Use three-quarters of the screen by default
61
screen = self.get_screen()
62
monitor = screen.get_monitor_geometry(0)
63
width = int(monitor.width * 0.75)
64
height = int(monitor.height * 0.75)
61
65
self.set_default_size(width, height)
66
self.set_size_request(width/3, height/3)
67
self._save_size_on_destroy(self, 'viz-window-size')
64
70
icon = self.render_icon(gtk.STOCK_INDEX, gtk.ICON_SIZE_BUTTON)
71
77
self.accel_group = gtk.AccelGroup()
72
78
self.add_accel_group(self.accel_group)
74
gtk.Action.set_tool_item_type(gtk.MenuToolButton)
80
if getattr(gtk.Action, 'set_tool_item_type', None) is not None:
81
# Not available before PyGtk-2.10
82
gtk.Action.set_tool_item_type(gtk.MenuToolButton)
76
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")
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)
101
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, expand=False, fill=True)
127
vbox.pack_start(nav, expand=False, fill=True)
128
vbox.pack_start(paned, expand=True, fill=True)
129
vbox.set_focus_child(paned)
131
self.treeview.connect('revision-selected',
132
self._treeselection_changed_cb)
133
self.treeview.connect('revision-activated',
134
self._tree_revision_activated)
136
self.treeview.connect('tag-added', lambda w, t, r: self._update_tags())
141
def construct_paned(self):
142
"""Construct the main HPaned/VPaned contents."""
143
if self.config.get_user_option('viz-vertical') == 'True':
144
self.paned = gtk.HPaned()
146
self.paned = gtk.VPaned()
148
self.paned.pack1(self.construct_top(), resize=False, shrink=True)
149
self.paned.pack2(self.construct_bottom(), resize=True, shrink=False)
107
150
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
154
def construct_menubar(self):
118
155
menubar = gtk.MenuBar()
124
161
file_menu_close = gtk.ImageMenuItem(gtk.STOCK_CLOSE, self.accel_group)
125
162
file_menu_close.connect('activate', lambda x: self.destroy())
127
164
file_menu_quit = gtk.ImageMenuItem(gtk.STOCK_QUIT, self.accel_group)
128
165
file_menu_quit.connect('activate', lambda x: gtk.main_quit())
130
167
if self._parent is not None:
131
168
file_menu.add(file_menu_close)
132
169
file_menu.add(file_menu_quit)
135
172
edit_menuitem = gtk.MenuItem("_Edit")
136
173
edit_menuitem.set_submenu(edit_menu)
138
edit_menu_find = gtk.ImageMenuItem(gtk.STOCK_FIND)
140
175
edit_menu_branchopts = gtk.MenuItem("Branch Settings")
141
176
edit_menu_branchopts.connect('activate', lambda x: PreferencesWindow(self.branch.get_config()).show())
143
178
edit_menu_globopts = gtk.MenuItem("Global Settings")
144
179
edit_menu_globopts.connect('activate', lambda x: PreferencesWindow().show())
146
edit_menu.add(edit_menu_find)
147
181
edit_menu.add(edit_menu_branchopts)
148
182
edit_menu.add(edit_menu_globopts)
160
194
view_menu_toolbar = gtk.CheckMenuItem("Show Toolbar")
161
195
view_menu_toolbar.set_active(True)
196
if self.config.get_user_option('viz-toolbar-visible') == 'False':
197
view_menu_toolbar.set_active(False)
162
199
view_menu_toolbar.connect('toggled', self._toolbar_visibility_changed)
164
201
view_menu_compact = gtk.CheckMenuItem("Show Compact Graph")
165
202
view_menu_compact.set_active(self.compact_view)
166
203
view_menu_compact.connect('activate', self._brokenlines_toggled_cb)
205
view_menu_vertical = gtk.CheckMenuItem("Side-by-side Layout")
206
view_menu_vertical.set_active(False)
207
if self.config.get_user_option('viz-vertical') == 'True':
208
view_menu_vertical.set_active(True)
209
view_menu_vertical.connect('toggled', self._vertical_layout)
211
view_menu_diffs = gtk.CheckMenuItem("Show Diffs")
212
view_menu_diffs.set_active(False)
213
if self.config.get_user_option('viz-show-diffs') == 'True':
214
view_menu_diffs.set_active(True)
215
view_menu_diffs.connect('toggled', self._diff_visibility_changed)
217
view_menu_wide_diffs = gtk.CheckMenuItem("Wide Diffs")
218
view_menu_wide_diffs.set_active(False)
219
if self.config.get_user_option('viz-wide-diffs') == 'True':
220
view_menu_wide_diffs.set_active(True)
221
view_menu_wide_diffs.connect('toggled', self._diff_placement_changed)
223
view_menu_wrap_diffs = gtk.CheckMenuItem("Wrap _Long Lines in Diffs")
224
view_menu_wrap_diffs.set_active(False)
225
if self.config.get_user_option('viz-wrap-diffs') == 'True':
226
view_menu_wrap_diffs.set_active(True)
227
view_menu_wrap_diffs.connect('toggled', self._diff_wrap_changed)
168
229
view_menu.add(view_menu_toolbar)
169
230
view_menu.add(view_menu_compact)
231
view_menu.add(view_menu_vertical)
232
view_menu.add(gtk.SeparatorMenuItem())
233
view_menu.add(view_menu_diffs)
234
view_menu.add(view_menu_wide_diffs)
235
view_menu.add(view_menu_wrap_diffs)
170
236
view_menu.add(gtk.SeparatorMenuItem())
172
238
self.mnu_show_revno_column = gtk.CheckMenuItem("Show Revision _Number Column")
195
261
tag_image.set_from_file(icon_path("tag-16.png"))
196
262
self.go_menu_tags = gtk.ImageMenuItem("_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
268
go_menu.add(gtk.SeparatorMenuItem())
203
269
go_menu.add(self.go_menu_tags)
205
revision_menu = gtk.Menu()
271
self.revision_menu = RevisionMenu(self.branch.repository, [],
272
self.branch, parent=self)
206
273
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)
274
revision_menuitem.set_submenu(self.revision_menu)
224
276
branch_menu = gtk.Menu()
225
277
branch_menuitem = gtk.MenuItem("_Branch")
228
280
branch_menu.add(gtk.MenuItem("Pu_ll Revisions"))
229
281
branch_menu.add(gtk.MenuItem("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("_Index")
291
branch_index_menuitem.connect('activate', self._branch_index_cb)
292
branch_menu.add(branch_index_menuitem)
294
branch_search_menuitem = gtk.MenuItem("_Search")
295
branch_search_menuitem.connect('activate', self._branch_search_cb)
296
branch_menu.add(branch_search_menuitem)
231
298
help_menu = gtk.Menu()
232
299
help_menuitem = gtk.MenuItem("_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(gtk.STOCK_ABOUT,
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)
273
336
align = gtk.Alignment(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')
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.diff_paned.pack1(self.revisionview)
398
from bzrlib.plugins.gtk.diff import DiffWidget
399
self.diff = DiffWidget()
400
self.diff_paned.pack2(self.diff)
402
self.diff_paned.show_all()
403
if self.config.get_user_option('viz-show-diffs') != 'True':
406
return self.diff_paned
313
408
def _tag_selected_cb(self, menuitem, revid):
314
409
self.treeview.set_revision_id(revid)
359
457
self.next_rev_action.set_sensitive(False)
362
self.next_button.set_menu(next_menu)
460
if getattr(self.next_button, 'set_menu', None) is not None:
461
self.next_button.set_menu(next_menu)
364
463
self.revisionview.set_revision(revision)
365
464
self.revisionview.set_children(children)
465
self.update_diff_panel(revision, parents)
367
467
def _tree_revision_activated(self, widget, path, col):
368
468
# TODO: more than one parent
369
469
"""Callback for when a treeview row gets activated."""
371
471
parents = self.treeview.get_parents()
373
473
if len(parents) == 0:
474
parent_id = NULL_REVISION
376
476
parent_id = parents[0]
378
478
self.show_diff(revision.revision_id, parent_id)
379
479
self.treeview.grab_focus()
381
def _menu_diff_cb(self,w):
382
(path, focus) = self.treeview.treeview.get_cursor()
383
revid = self.treeview.model[path][treemodel.REVID]
385
parentids = self.branch.repository.revision_parents(revid)
387
if len(parentids) == 0:
388
parentid = NULL_REVISION
390
parentid = parentids[0]
392
self.show_diff(revid,parentid)
394
481
def _back_clicked_cb(self, *args):
395
482
"""Callback for when the back button is clicked."""
396
483
self.treeview.back()
398
485
def _fwd_clicked_cb(self, *args):
399
486
"""Callback for when the forward button is clicked."""
400
487
self.treeview.forward()
409
496
self.show_diff(revid, parentid)
410
497
self.treeview.grab_focus()
412
def _compare_with_cb(self,w):
413
"""Callback for revision 'compare with' menu. Will show a small
414
dialog with branch revisions to compare with selected revision in TreeView"""
416
from bzrlib.plugins.gtk.revbrowser import RevisionBrowser
418
rb = RevisionBrowser(self.branch,self)
421
if ret == gtk.RESPONSE_OK:
422
(path, focus) = self.treeview.treeview.get_cursor()
423
revid = self.treeview.model[path][treemodel.REVID]
424
self.show_diff(revid, rb.selected_revid)
428
499
def _set_revision_cb(self, w, revision_id):
429
500
self.treeview.set_revision_id(revision_id)
440
511
self.treeview.set_property('compact', self.compact_view)
441
512
self.treeview.refresh()
443
def _tag_revision_cb(self, w):
514
def _branch_index_cb(self, w):
515
from bzrlib.plugins.search import index as _mod_index
516
_mod_index.index_url(self.branch.base)
518
def _branch_search_cb(self, w):
519
from bzrlib.plugins.search import (
521
errors as search_errors,
523
from bzrlib.plugins.gtk.search import SearchDialog
445
self.treeview.set_sensitive(False)
446
dialog = AddTagDialog(self.branch.repository, self.treeview.get_revision().revision_id, self.branch)
447
response = dialog.run()
448
if response != gtk.RESPONSE_NONE:
451
if response == gtk.RESPONSE_OK:
452
self.treeview.add_tag(dialog.tagname, dialog._revid)
457
self.treeview.set_sensitive(True)
526
index = _mod_index.open_index_url(self.branch.base)
527
except search_errors.NoSearchIndex:
528
dialog = gtk.MessageDialog(self, type=gtk.MESSAGE_QUESTION,
529
buttons=gtk.BUTTONS_OK_CANCEL,
530
message_format="This branch has not been indexed yet. "
532
if dialog.run() == gtk.RESPONSE_OK:
534
index = _mod_index.index_url(self.branch.base)
539
dialog = SearchDialog(index)
541
if dialog.run() == gtk.RESPONSE_OK:
542
self.set_revision(dialog.get_revision())
459
546
def _about_dialog_cb(self, w):
460
547
from bzrlib.plugins.gtk.about import AboutDialog
462
548
AboutDialog().run()
464
550
def _col_visibility_changed(self, col, property):
468
554
def _toolbar_visibility_changed(self, col):
469
555
if col.get_active():
472
558
self.toolbar.hide()
474
def _show_about_cb(self, w):
475
dialog = AboutDialog()
476
dialog.connect('response', lambda d,r: d.destroy())
559
self.config.set_user_option('viz-toolbar-visible', col.get_active())
561
def _vertical_layout(self, col):
562
"""Toggle the layout vertical/horizontal"""
563
self.config.set_user_option('viz-vertical', str(col.get_active()))
566
self.vbox.remove(old)
567
self.vbox.pack_start(self.construct_paned(), expand=True, fill=True)
568
self._make_diff_paned_nonzero_size()
569
self._make_diff_nonzero_size()
571
self.treeview.emit('revision-selected')
573
def _make_diff_paned_nonzero_size(self):
574
"""make sure the diff/revision pane isn't zero-width or zero-height"""
575
alloc = self.diff_paned.get_allocation()
576
if (alloc.width < 10) or (alloc.height < 10):
577
width, height = self.get_size()
578
self.diff_paned.set_size_request(width/3, int(height / 2.5))
580
def _make_diff_nonzero_size(self):
581
"""make sure the diff isn't zero-width or zero-height"""
582
alloc = self.diff.get_allocation()
583
if (alloc.width < 10) or (alloc.height < 10):
584
width, height = self.get_size()
585
self.revisionview.set_size_request(width/3, int(height / 2.5))
587
def _diff_visibility_changed(self, col):
588
"""Hide or show the diff panel."""
591
self._make_diff_nonzero_size()
594
self.config.set_user_option('viz-show-diffs', str(col.get_active()))
595
self.update_diff_panel()
597
def _diff_placement_changed(self, col):
598
"""Toggle the diff panel's position."""
599
self.config.set_user_option('viz-wide-diffs', str(col.get_active()))
601
old = self.paned.get_child2()
602
self.paned.remove(old)
603
self.paned.pack2(self.construct_bottom(), resize=True, shrink=False)
604
self._make_diff_nonzero_size()
606
self.treeview.emit('revision-selected')
608
def _diff_wrap_changed(self, widget):
609
"""Toggle word wrap in the diff widget."""
610
self.config.set_user_option('viz-wrap-diffs', widget.get_active())
611
self.diff._on_wraplines_toggled(widget)
479
613
def _refresh_clicked(self, w):
480
614
self.treeview.refresh()
485
619
if self.branch.supports_tags():
486
620
tags = self.branch.tags.get_tag_dict().items()
621
tags.sort(reverse=True)
489
622
for tag, revid in tags:
490
623
tag_image = gtk.Image()
491
624
tag_image.set_from_file(icon_path('tag-16.png'))
492
625
tag_item = gtk.ImageMenuItem(tag.replace('_', '__'))
493
626
tag_item.set_image(tag_image)
494
627
tag_item.connect('activate', self._tag_selected_cb, revid)
628
tag_item.set_sensitive(self.treeview.has_revision_id(revid))
495
629
menu.add(tag_item)
496
630
self.go_menu_tags.set_submenu(menu)
502
636
self.go_menu_tags.show_all()
504
def show_diff(self, revid=None, parentid=None):
638
def _load_size(self, name):
639
"""Read and parse 'name' from self.config.
640
The value is a string, formatted as WIDTHxHEIGHT
641
Returns None, or (width, height)
643
size = self.config.get_user_option(name)
645
width, height = [int(num) for num in size.split('x')]
646
# avoid writing config every time we start
650
def show_diff(self, revid=None, parentid=NULL_REVISION):
505
651
"""Open a new window to show a diff between the given revisions."""
506
652
from bzrlib.plugins.gtk.diff import DiffWindow
507
653
window = DiffWindow(parent=self)
510
parentid = NULL_REVISION
512
655
rev_tree = self.branch.repository.revision_tree(revid)
513
656
parent_tree = self.branch.repository.revision_tree(parentid)
515
description = revid + " - " + self.branch.nick
658
description = revid + " - " + self.branch._get_nick(local=True)
516
659
window.set_diff(description, rev_tree, parent_tree)
662
def update_diff_panel(self, revision=None, parents=None):
663
"""Show the current revision in the diff panel."""
664
if self.config.get_user_option('viz-show-diffs') != 'True':
667
if not revision: # default to selected row
668
revision = self.treeview.get_revision()
669
if revision == NULL_REVISION:
672
if not parents: # default to selected row's parents
673
parents = self.treeview.get_parents()
674
if len(parents) == 0:
675
parent_id = NULL_REVISION
677
parent_id = parents[0]
679
rev_tree = self.branch.repository.revision_tree(revision.revision_id)
680
parent_tree = self.branch.repository.revision_tree(parent_id)
682
self.diff.set_diff(rev_tree, parent_tree)
683
if self.config.get_user_option('viz-wrap-diffs') == 'True':
684
self.diff._on_wraplines_toggled(wrap=True)