/b-gtk/fix-viz

To get this branch, use:
bzr branch http://gegoxaren.bato24.eu/bzr/b-gtk/fix-viz

« back to all changes in this revision

Viewing changes to viz/branchwin.py

  • Committer: Daniel Schierbeck
  • Date: 2007-10-14 15:54:57 UTC
  • mto: This revision was merged to the branch mainline in revision 317.
  • Revision ID: daniel.schierbeck@gmail.com-20071014155457-m3ek29p4ima8ev7d
Added the new Window base class.

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
# -*- coding: UTF-8 -*-
1
2
"""Branch window.
2
3
 
3
4
This module contains the code to manage the branch information window,
4
5
which contains both the revision graph and details panes.
5
6
"""
6
7
 
7
 
__copyright__ = "Copyright (c) 2005 Canonical Ltd."
 
8
__copyright__ = "Copyright © 2005 Canonical Ltd."
8
9
__author__    = "Scott James Remnant <scott@ubuntu.com>"
9
10
 
10
11
 
11
 
from gi.repository import Gdk
12
 
from gi.repository import Gtk
 
12
import gtk
 
13
import gobject
 
14
import pango
 
15
import treemodel
13
16
 
14
 
from bzrlib.plugins.gtk import icon_path
15
 
from bzrlib.plugins.gtk.branchview import TreeView
16
 
from bzrlib.plugins.gtk.preferences import PreferencesWindow
17
 
from bzrlib.plugins.gtk.revisionmenu import RevisionMenu
18
17
from bzrlib.plugins.gtk.window import Window
 
18
from bzrlib.osutils import format_date
19
19
 
20
 
from bzrlib.config import GlobalConfig
21
 
from bzrlib.revision import NULL_REVISION
22
 
from bzrlib.trace import mutter
 
20
from linegraph import linegraph, same_branch
 
21
from graphcell import CellRendererGraph
 
22
from treemodel import TreeModel
23
23
 
24
24
 
25
25
class BranchWindow(Window):
29
29
    for a particular branch.
30
30
    """
31
31
 
32
 
    def __init__(self, branch, start_revs, maxnum, parent=None):
33
 
        """Create a new BranchWindow.
34
 
 
35
 
        :param branch: Branch object for branch to show.
36
 
        :param start_revs: Revision ids of top revisions.
37
 
        :param maxnum: Maximum number of revisions to display, 
38
 
                       None for no limit.
39
 
        """
40
 
 
41
 
        super(BranchWindow, self).__init__(parent=parent)
 
32
    def __init__(self, parent=None):
 
33
        Window.__init__(self, parent=parent)
42
34
        self.set_border_width(0)
43
 
 
44
 
        self.branch      = branch
45
 
        self.start_revs  = start_revs
46
 
        self.maxnum      = maxnum
47
 
        self.config      = GlobalConfig()
48
 
 
49
 
        if self.config.get_user_option('viz-compact-view') == 'yes':
50
 
            self.compact_view = True
51
 
        else:
52
 
            self.compact_view = False
53
 
 
54
 
        self.set_title(branch._get_nick(local=True) + " - revision history")
55
 
 
56
 
        # user-configured window size
57
 
        size = self._load_size('viz-window-size')
58
 
        if size:
59
 
            width, height = size
60
 
        else:
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)
 
35
        self.set_title("Revision history")
 
36
 
 
37
        # Use three-quarters of the screen by default
 
38
        screen = self.get_screen()
 
39
        monitor = screen.get_monitor_geometry(0)
 
40
        width = int(monitor.width * 0.75)
 
41
        height = int(monitor.height * 0.75)
66
42
        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')
69
43
 
70
44
        # FIXME AndyFitz!
71
 
        icon = self.render_icon_pixbuf(Gtk.STOCK_INDEX, Gtk.IconSize.BUTTON)
 
45
        icon = self.render_icon(gtk.STOCK_INDEX, gtk.ICON_SIZE_BUTTON)
72
46
        self.set_icon(icon)
73
47
 
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)
77
 
 
78
 
        self.accel_group = Gtk.AccelGroup()
 
48
        self.accel_group = gtk.AccelGroup()
79
49
        self.add_accel_group(self.accel_group)
80
50
 
81
 
        self.prev_rev_action = Gtk.Action("prev-rev", "_Previous Revision", "Go to the previous revision", Gtk.STOCK_GO_DOWN)
82
 
        self.prev_rev_action.set_accel_path("<viz>/Go/Previous Revision")
83
 
        self.prev_rev_action.set_accel_group(self.accel_group)
84
 
        self.prev_rev_action.connect("activate", self._back_clicked_cb)
85
 
        self.prev_rev_action.connect_accelerator()
86
 
 
87
 
        self.next_rev_action = Gtk.Action("next-rev", "_Next Revision", "Go to the next revision", Gtk.STOCK_GO_UP)
88
 
        self.next_rev_action.set_accel_path("<viz>/Go/Next Revision")
89
 
        self.next_rev_action.set_accel_group(self.accel_group)
90
 
        self.next_rev_action.connect("activate", self._fwd_clicked_cb)
91
 
        self.next_rev_action.connect_accelerator()
92
 
 
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()
98
 
 
99
 
        self.vbox = self.construct()
100
 
 
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."""
104
 
        def save_size(src):
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)
110
 
 
111
 
    def set_revision(self, revid):
112
 
        self.treeview.set_revision_id(revid)
 
51
        self.construct()
113
52
 
114
53
    def construct(self):
115
54
        """Construct the window contents."""
116
 
        vbox = Gtk.VBox(spacing=0)
 
55
        vbox = gtk.VBox(spacing=0)
117
56
        self.add(vbox)
118
57
 
119
 
        # order is important here
120
 
        paned = self.construct_paned()
121
 
        nav = self.construct_navigation()
122
 
        menubar = self.construct_menubar()
123
 
 
124
 
        vbox.pack_start(menubar, False, True, 0)
125
 
        vbox.pack_start(nav, False, True, 0)
126
 
        vbox.pack_start(paned, True, True, 0)
 
58
        vbox.pack_start(self.construct_navigation(), expand=False, fill=True)
 
59
        vbox.pack_start(self.construct_loading_msg(), expand=False, fill=True)
 
60
        
 
61
        paned = gtk.VPaned()
 
62
        paned.pack1(self.construct_top(), resize=True, shrink=False)
 
63
        paned.pack2(self.construct_bottom(), resize=False, shrink=True)
 
64
        paned.show()
 
65
        vbox.pack_start(paned, expand=True, fill=True)
127
66
        vbox.set_focus_child(paned)
128
67
 
129
 
 
130
68
        vbox.show()
131
 
 
132
 
        return vbox
133
 
 
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)
138
 
        else:
139
 
            self.paned = Gtk.Paned.new(Gtk.Orientation.VERTICAL)
140
 
 
141
 
        self.paned.pack1(self.construct_top(), resize=False, shrink=True)
142
 
        self.paned.pack2(self.construct_bottom(), resize=True, shrink=False)
143
 
        self.paned.show()
144
 
 
145
 
        return self.paned
146
 
 
147
 
    def construct_menubar(self):
148
 
        menubar = Gtk.MenuBar()
149
 
 
150
 
        file_menu = Gtk.Menu()
151
 
        file_menuitem = Gtk.MenuItem.new_with_mnemonic("_File")
152
 
        file_menuitem.set_submenu(file_menu)
153
 
 
154
 
        file_menu_close = Gtk.ImageMenuItem.new_from_stock(
155
 
            Gtk.STOCK_CLOSE, self.accel_group)
156
 
        file_menu_close.connect('activate', lambda x: self.destroy())
157
 
 
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())
161
 
 
162
 
        if self._parent is not None:
163
 
            file_menu.add(file_menu_close)
164
 
        file_menu.add(file_menu_quit)
165
 
 
166
 
        edit_menu = Gtk.Menu()
167
 
        edit_menuitem = Gtk.MenuItem.new_with_mnemonic("_Edit")
168
 
        edit_menuitem.set_submenu(edit_menu)
169
 
 
170
 
        edit_menu_branchopts = Gtk.MenuItem(label="Branch Settings")
171
 
        edit_menu_branchopts.connect('activate', lambda x: PreferencesWindow(self.branch.get_config()).show())
172
 
 
173
 
        edit_menu_globopts = Gtk.MenuItem(label="Global Settings")
174
 
        edit_menu_globopts.connect('activate', lambda x: PreferencesWindow().show())
175
 
 
176
 
        edit_menu.add(edit_menu_branchopts)
177
 
        edit_menu.add(edit_menu_globopts)
178
 
 
179
 
        view_menu = Gtk.Menu()
180
 
        view_menuitem = Gtk.MenuItem.new_with_mnemonic("_View")
181
 
        view_menuitem.set_submenu(view_menu)
182
 
 
183
 
        view_menu_refresh = self.refresh_action.create_menu_item()
184
 
        view_menu_refresh.connect('activate', self._refresh_clicked)
185
 
 
186
 
        view_menu.add(view_menu_refresh)
187
 
        view_menu.add(Gtk.SeparatorMenuItem())
188
 
 
189
 
        view_menu_toolbar = Gtk.CheckMenuItem(label="Show Toolbar")
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)
193
 
            self.toolbar.hide()
194
 
        view_menu_toolbar.connect('toggled', self._toolbar_visibility_changed)
195
 
 
196
 
        view_menu_compact = Gtk.CheckMenuItem(label="Show Compact Graph")
197
 
        view_menu_compact.set_active(self.compact_view)
198
 
        view_menu_compact.connect('activate', self._brokenlines_toggled_cb)
199
 
 
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)
205
 
 
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)
211
 
 
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)
217
 
 
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)
224
 
 
225
 
        view_menu.add(view_menu_toolbar)
226
 
        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())
233
 
 
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(
237
 
            "Show _Date Column")
238
 
 
239
 
        # Revision numbers are pointless if there are multiple branches
240
 
        if len(self.start_revs) > 1:
241
 
            self.mnu_show_revno_column.set_sensitive(False)
242
 
            self.treeview.set_property('revno-column-visible', False)
243
 
 
244
 
        for (col, name) in [(self.mnu_show_revno_column, "revno"), 
245
 
                            (self.mnu_show_date_column, "date")]:
246
 
            col.set_active(self.treeview.get_property(name + "-column-visible"))
247
 
            col.connect('toggled', self._col_visibility_changed, name)
248
 
            view_menu.add(col)
249
 
 
250
 
        go_menu = Gtk.Menu()
251
 
        go_menu.set_accel_group(self.accel_group)
252
 
        go_menuitem = Gtk.MenuItem.new_with_mnemonic("_Go")
253
 
        go_menuitem.set_submenu(go_menu)
254
 
 
255
 
        go_menu_next = self.next_rev_action.create_menu_item()
256
 
        go_menu_prev = self.prev_rev_action.create_menu_item()
257
 
 
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())
263
 
 
264
 
        go_menu.add(go_menu_next)
265
 
        go_menu.add(go_menu_prev)
266
 
        go_menu.add(Gtk.SeparatorMenuItem())
267
 
        go_menu.add(self.go_menu_tags)
268
 
 
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)
273
 
 
274
 
        branch_menu = Gtk.Menu()
275
 
        branch_menuitem = Gtk.MenuItem.new_with_mnemonic("_Branch")
276
 
        branch_menuitem.set_submenu(branch_menu)
277
 
 
278
 
        branch_menu.add(Gtk.MenuItem.new_with_mnemonic("Pu_ll Revisions"))
279
 
        branch_menu.add(Gtk.MenuItem.new_with_mnemonic("Pu_sh Revisions"))
280
 
 
281
 
        try:
282
 
            from bzrlib.plugins import search
283
 
        except ImportError:
284
 
            mutter("Didn't find search plugin")
285
 
        else:
286
 
            branch_menu.add(Gtk.SeparatorMenuItem())
287
 
 
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)
291
 
 
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)
295
 
 
296
 
        help_menu = Gtk.Menu()
297
 
        help_menuitem = Gtk.MenuItem.new_with_mnemonic("_Help")
298
 
        help_menuitem.set_submenu(help_menu)
299
 
 
300
 
        help_about_menuitem = Gtk.ImageMenuItem.new_from_stock(
301
 
            Gtk.STOCK_ABOUT, self.accel_group)
302
 
        help_about_menuitem.connect('activate', self._about_dialog_cb)
303
 
 
304
 
        help_menu.add(help_about_menuitem)
305
 
 
306
 
        menubar.add(file_menuitem)
307
 
        menubar.add(edit_menuitem)
308
 
        menubar.add(view_menuitem)
309
 
        menubar.add(go_menuitem)
310
 
        menubar.add(revision_menuitem)
311
 
        menubar.add(branch_menuitem)
312
 
        menubar.add(help_menuitem)
313
 
        menubar.show_all()
314
 
 
315
 
        return menubar
 
69
    
 
70
    def construct_loading_msg(self):
 
71
        image_loading = gtk.image_new_from_stock(gtk.STOCK_REFRESH,
 
72
                                                 gtk.ICON_SIZE_BUTTON)
 
73
        image_loading.show()
 
74
        
 
75
        label_loading = gtk.Label(_("Please wait, loading ancestral graph..."))
 
76
        label_loading.set_alignment(0.0, 0.5)        
 
77
        label_loading.show()
 
78
        
 
79
        self.loading_msg_box = gtk.HBox()
 
80
        self.loading_msg_box.set_spacing(5)
 
81
        self.loading_msg_box.set_border_width(5)        
 
82
        self.loading_msg_box.pack_start(image_loading, False, False)
 
83
        self.loading_msg_box.pack_start(label_loading, True, True)
 
84
        self.loading_msg_box.show()
 
85
        
 
86
        return self.loading_msg_box
316
87
 
317
88
    def construct_top(self):
318
89
        """Construct the top-half of the window."""
319
 
        # FIXME: Make broken_line_length configurable
320
 
 
321
 
        self.treeview = TreeView(self.branch, self.start_revs, self.maxnum,
322
 
            self.compact_view)
323
 
 
324
 
        for col in ["revno", "date"]:
325
 
            option = self.config.get_user_option(col + '-column-visible')
326
 
            if option is not None:
327
 
                self.treeview.set_property(col + '-column-visible',
328
 
                    option == 'True')
329
 
            else:
330
 
                self.treeview.set_property(col + '-column-visible', False)
331
 
 
 
90
        scrollwin = gtk.ScrolledWindow()
 
91
        scrollwin.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
 
92
        scrollwin.set_shadow_type(gtk.SHADOW_IN)
 
93
        scrollwin.show()
 
94
 
 
95
        self.treeview = gtk.TreeView()
 
96
        self.treeview.set_rules_hint(True)
 
97
        self.treeview.set_search_column(4)
 
98
        self.treeview.get_selection().connect("changed", self._treeselection_changed_cb)
 
99
        self.treeview.connect("row-activated", self._treeview_row_activated_cb)
 
100
        self.treeview.connect("button-release-event", 
 
101
                self._treeview_row_mouseclick)
 
102
        self.treeview.set_property('fixed-height-mode', True)
 
103
        scrollwin.add(self.treeview)
332
104
        self.treeview.show()
333
105
 
334
 
        align = Gtk.Alignment.new(0.0, 0.0, 1.0, 1.0)
335
 
        align.set_padding(5, 0, 0, 0)
336
 
        align.add(self.treeview)
337
 
        # user-configured size
338
 
        size = self._load_size('viz-graph-size')
339
 
        if size:
340
 
            width, height = size
341
 
            align.set_size_request(width, height)
342
 
        else:
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')
346
 
        align.show()
347
 
 
348
 
        return align
 
106
        cell = gtk.CellRendererText()
 
107
        cell.set_property("width-chars", 15)
 
108
        cell.set_property("ellipsize", pango.ELLIPSIZE_END)
 
109
        column = gtk.TreeViewColumn("Revision No")
 
110
        column.set_resizable(True)
 
111
        column.set_sizing(gtk.TREE_VIEW_COLUMN_FIXED)
 
112
        column.set_fixed_width(cell.get_size(self.treeview)[2])
 
113
        column.pack_start(cell, expand=True)
 
114
        column.add_attribute(cell, "text", treemodel.REVNO)
 
115
        self.treeview.append_column(column)
 
116
 
 
117
        self.graph_cell = CellRendererGraph()
 
118
        self.graph_column = gtk.TreeViewColumn()
 
119
        self.graph_column.set_resizable(True)
 
120
        self.graph_column.set_sizing(gtk.TREE_VIEW_COLUMN_FIXED)
 
121
        self.graph_column.pack_start(self.graph_cell, expand=False)
 
122
        self.graph_column.add_attribute(self.graph_cell, "node", treemodel.NODE)
 
123
        self.graph_column.add_attribute(self.graph_cell, "in-lines", treemodel.LAST_LINES)
 
124
        self.graph_column.add_attribute(self.graph_cell, "out-lines", treemodel.LINES)
 
125
        self.treeview.append_column(self.graph_column)
 
126
 
 
127
        cell = gtk.CellRendererText()
 
128
        cell.set_property("width-chars", 65)
 
129
        cell.set_property("ellipsize", pango.ELLIPSIZE_END)
 
130
        column = gtk.TreeViewColumn("Message")
 
131
        column.set_resizable(True)
 
132
        column.set_sizing(gtk.TREE_VIEW_COLUMN_FIXED)
 
133
        column.set_fixed_width(cell.get_size(self.treeview)[2])
 
134
        column.pack_start(cell, expand=True)
 
135
        column.add_attribute(cell, "text", treemodel.MESSAGE)
 
136
        self.treeview.append_column(column)
 
137
 
 
138
        cell = gtk.CellRendererText()
 
139
        cell.set_property("width-chars", 15)
 
140
        cell.set_property("ellipsize", pango.ELLIPSIZE_END)
 
141
        column = gtk.TreeViewColumn("Committer")
 
142
        column.set_resizable(True)
 
143
        column.set_sizing(gtk.TREE_VIEW_COLUMN_FIXED)
 
144
        column.set_fixed_width(cell.get_size(self.treeview)[2])
 
145
        column.pack_start(cell, expand=True)
 
146
        column.add_attribute(cell, "text", treemodel.COMMITER)
 
147
        self.treeview.append_column(column)
 
148
 
 
149
        return scrollwin
349
150
 
350
151
    def construct_navigation(self):
351
152
        """Construct the navigation buttons."""
352
 
        self.toolbar = Gtk.Toolbar()
353
 
        self.toolbar.set_style(Gtk.ToolbarStyle.BOTH_HORIZ)
354
 
 
355
 
        self.prev_button = self.prev_rev_action.create_tool_item()
356
 
        self.toolbar.insert(self.prev_button, -1)
357
 
 
358
 
        self.next_button = self.next_rev_action.create_tool_item()
359
 
        self.toolbar.insert(self.next_button, -1)
360
 
 
361
 
        self.toolbar.insert(Gtk.SeparatorToolItem(), -1)
362
 
 
363
 
        refresh_button = Gtk.ToolButton.new_from_stock(Gtk.STOCK_REFRESH)
364
 
        refresh_button.connect('clicked', self._refresh_clicked)
365
 
        self.toolbar.insert(refresh_button, -1)
366
 
 
367
 
        self.toolbar.show_all()
368
 
 
369
 
        return self.toolbar
 
153
        frame = gtk.Frame()
 
154
        frame.set_shadow_type(gtk.SHADOW_OUT)
 
155
        frame.show()
 
156
        
 
157
        hbox = gtk.HBox(spacing=12)
 
158
        frame.add(hbox)
 
159
        hbox.show()
 
160
 
 
161
        self.back_button = gtk.Button(stock=gtk.STOCK_GO_BACK)
 
162
        self.back_button.set_relief(gtk.RELIEF_NONE)
 
163
        self.back_button.add_accelerator("clicked", self.accel_group, ord('['),
 
164
                                         0, 0)
 
165
        self.back_button.connect("clicked", self._back_clicked_cb)
 
166
        hbox.pack_start(self.back_button, expand=False, fill=True)
 
167
        self.back_button.show()
 
168
 
 
169
        self.fwd_button = gtk.Button(stock=gtk.STOCK_GO_FORWARD)
 
170
        self.fwd_button.set_relief(gtk.RELIEF_NONE)
 
171
        self.fwd_button.add_accelerator("clicked", self.accel_group, ord(']'),
 
172
                                        0, 0)
 
173
        self.fwd_button.connect("clicked", self._fwd_clicked_cb)
 
174
        hbox.pack_start(self.fwd_button, expand=False, fill=True)
 
175
        self.fwd_button.show()
 
176
 
 
177
        return frame
370
178
 
371
179
    def construct_bottom(self):
372
180
        """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)
375
 
        else:
376
 
            self.diff_paned = Gtk.Paned.new(Gtk.Orientation.HORIZONTAL)
 
181
        from bzrlib.plugins.gtk.logview import LogView
 
182
        self.logview = LogView(None, True, [], True)
377
183
        (width, height) = self.get_size()
378
 
        self.diff_paned.set_size_request(20, 20) # shrinkable
379
 
 
380
 
        from bzrlib.plugins.gtk.revisionview import RevisionView
381
 
        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')
385
 
        if size:
386
 
            width, height = size
387
 
            self.revisionview.set_size_request(width, height)
388
 
        self._save_size_on_destroy(self.revisionview, 'viz-revisionview-size')
389
 
        self.revisionview.show()
390
 
        self.revisionview.set_show_callback(self._show_clicked_cb)
391
 
        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)
399
 
 
400
 
        from bzrlib.plugins.gtk.diff import DiffWidget
401
 
        self.diff = DiffWidget()
402
 
        self.diff_paned.pack2(self.diff)
403
 
 
404
 
        self.diff_paned.show_all()
405
 
        if self.config.get_user_option('viz-show-diffs') != 'True':
406
 
            self.diff.hide()
407
 
 
408
 
        return self.diff_paned
409
 
 
410
 
    def _tag_selected_cb(self, menuitem, revid):
411
 
        self.treeview.set_revision_id(revid)
412
 
 
 
184
        self.logview.set_size_request(width, int(height / 2.5))
 
185
        self.logview.show()
 
186
        self.logview.set_show_callback(self._show_clicked_cb)
 
187
        self.logview.set_go_callback(self._go_clicked_cb)
 
188
        return self.logview
 
189
 
 
190
    def set_branch(self, branch, start, maxnum):
 
191
        """Set the branch and start position for this window.
 
192
 
 
193
        Creates a new TreeModel and populates it with information about
 
194
        the new branch before updating the window title and model of the
 
195
        treeview itself.
 
196
        """
 
197
        self.branch = branch
 
198
        self.set_title(branch.nick + " - revision history")
 
199
        gobject.idle_add(self.populate_model, start, maxnum)
 
200
 
 
201
    def populate_model(self, start, maxnum):
 
202
        (linegraphdata, index, columns_len) = linegraph(self.branch,
 
203
                                                        start,
 
204
                                                        maxnum)
 
205
        self.model = TreeModel(self.branch, linegraphdata)
 
206
        self.graph_cell.columns_len = columns_len
 
207
        width = self.graph_cell.get_size(self.treeview)[2]
 
208
        self.graph_column.set_fixed_width(width)
 
209
        self.graph_column.set_max_width(width)
 
210
        self.index = index
 
211
        self.treeview.set_model(self.model)
 
212
        self.treeview.get_selection().select_path(0)
 
213
        self.loading_msg_box.hide()
 
214
        return False
 
215
    
413
216
    def _treeselection_changed_cb(self, selection, *args):
414
 
        """callback for when the treeview changes."""
415
 
        revision = self.treeview.get_revision()
416
 
        parents  = self.treeview.get_parents()
417
 
        children = self.treeview.get_children()
418
 
 
419
 
        if revision and revision.revision_id != NULL_REVISION:
420
 
            self.revision_menu.set_revision_ids([revision.revision_id])
421
 
            prev_menu = Gtk.Menu()
422
 
            if len(parents) > 0:
423
 
                self.prev_rev_action.set_sensitive(True)
424
 
                for parent_id in parents:
425
 
                    if parent_id and parent_id != NULL_REVISION:
426
 
                        parent = self.branch.repository.get_revision(parent_id)
427
 
                        try:
428
 
                            str = ' (%s)' % parent.properties['branch-nick']
429
 
                        except KeyError:
430
 
                            str = ""
431
 
 
432
 
                        item = Gtk.MenuItem(
433
 
                            label=parent.message.split("\n")[0] + str)
434
 
                        item.connect('activate', self._set_revision_cb, parent_id)
435
 
                        prev_menu.add(item)
436
 
                prev_menu.show_all()
437
 
            else:
438
 
                self.prev_rev_action.set_sensitive(False)
439
 
                prev_menu.hide()
440
 
 
441
 
            if getattr(self.prev_button, 'set_menu', None) is not None:
442
 
                self.prev_button.set_menu(prev_menu)
443
 
 
444
 
            next_menu = Gtk.Menu()
445
 
            if len(children) > 0:
446
 
                self.next_rev_action.set_sensitive(True)
447
 
                for child_id in children:
448
 
                    child = self.branch.repository.get_revision(child_id)
449
 
                    try:
450
 
                        str = ' (%s)' % child.properties['branch-nick']
451
 
                    except KeyError:
452
 
                        str = ""
453
 
 
454
 
                    item = Gtk.MenuItem(
455
 
                        label=child.message.split("\n")[0] + str)
456
 
                    item.connect('activate', self._set_revision_cb, child_id)
457
 
                    next_menu.add(item)
458
 
                next_menu.show_all()
459
 
            else:
460
 
                self.next_rev_action.set_sensitive(False)
461
 
                next_menu.hide()
462
 
 
463
 
            if getattr(self.next_button, 'set_menu', None) is not None:
464
 
                self.next_button.set_menu(next_menu)
465
 
 
466
 
            self.revisionview.set_revision(revision)
467
 
            self.revisionview.set_children(children)
468
 
            self.update_diff_panel(revision, parents)
469
 
 
470
 
    def _tree_revision_activated(self, widget, path, col):
471
 
        # TODO: more than one parent
472
 
        """Callback for when a treeview row gets activated."""
473
 
        revision = self.treeview.get_revision()
474
 
        parents  = self.treeview.get_parents()
475
 
 
476
 
        if len(parents) == 0:
477
 
            parent_id = NULL_REVISION
478
 
        else:
479
 
            parent_id = parents[0]
480
 
 
481
 
        if revision is not None:
482
 
            self.show_diff(revision.revision_id, parent_id)
483
 
        else:
484
 
            self.show_diff(NULL_REVISION)
485
 
        self.treeview.grab_focus()
 
217
        """Callback for when the treeview changes."""
 
218
        (model, selected_rows) = selection.get_selected_rows()
 
219
        if len(selected_rows) > 0:
 
220
            iter = self.model.get_iter(selected_rows[0])
 
221
            revision = self.model.get_value(iter, treemodel.REVISION)
 
222
            parents = self.model.get_value(iter, treemodel.PARENTS)
 
223
            children = self.model.get_value(iter, treemodel.CHILDREN)
 
224
            
 
225
            self.back_button.set_sensitive(len(parents) > 0)
 
226
            self.fwd_button.set_sensitive(len(children) > 0)
 
227
            tags = []
 
228
            if self.branch.supports_tags():
 
229
                tagdict = self.branch.tags.get_reverse_tag_dict()
 
230
                if tagdict.has_key(revision.revision_id):
 
231
                    tags = tagdict[revision.revision_id]
 
232
            self.logview.set_revision(revision, tags, children)
486
233
 
487
234
    def _back_clicked_cb(self, *args):
488
235
        """Callback for when the back button is clicked."""
489
 
        self.treeview.back()
 
236
        (path, col) = self.treeview.get_cursor()
 
237
        revision = self.model[path][0]
 
238
        parents = self.model[path][4]
 
239
        if not len(parents):
 
240
            return
 
241
 
 
242
        for parent_id in parents:
 
243
            parent = self.revisions[self.index[parent_id]]
 
244
            if same_branch(revision, parent):
 
245
                self.treeview.set_cursor(self.index[parent_id])
 
246
                break
 
247
        else:
 
248
            self.treeview.set_cursor(self.index[parents[0]])
 
249
        self.treeview.grab_focus()
490
250
 
491
251
    def _fwd_clicked_cb(self, *args):
492
252
        """Callback for when the forward button is clicked."""
493
 
        self.treeview.forward()
494
 
 
495
 
    def _go_clicked_cb(self, w, p):
 
253
        (path, col) = self.treeview.get_cursor()
 
254
        revision = self.model[path][0]
 
255
        children = self.model[path][5]
 
256
        if not len(children):
 
257
            return
 
258
 
 
259
        for child_id in children:
 
260
            child = self.revisions[self.index[child_id]]
 
261
            if same_branch(child, revision):
 
262
                self.treeview.set_cursor(self.index[child_id])
 
263
                break
 
264
        else:
 
265
            self.treeview.set_cursor(self.index[children[0]])
 
266
        self.treeview.grab_focus()
 
267
 
 
268
    def _go_clicked_cb(self, revid):
496
269
        """Callback for when the go button for a parent is clicked."""
497
 
        if self.revisionview.get_revision() is not None:
498
 
            self.treeview.set_revision(self.revisionview.get_revision())
499
 
 
500
 
    def _show_clicked_cb(self, revid, parentid):
501
 
        """Callback for when the show button for a parent is clicked."""
502
 
        self.show_diff(revid, parentid)
 
270
        self.treeview.set_cursor(self.index[revid])
503
271
        self.treeview.grab_focus()
504
272
 
505
 
    def _set_revision_cb(self, w, revision_id):
506
 
        self.treeview.set_revision_id(revision_id)
507
 
 
508
 
    def _brokenlines_toggled_cb(self, button):
509
 
        self.compact_view = button.get_active()
510
 
 
511
 
        if self.compact_view:
512
 
            option = 'yes'
513
 
        else:
514
 
            option = 'no'
515
 
 
516
 
        self.config.set_user_option('viz-compact-view', option)
517
 
        self.treeview.set_property('compact', self.compact_view)
518
 
        self.treeview.refresh()
519
 
 
520
 
    def _branch_index_cb(self, w):
521
 
        from bzrlib.plugins.search import index as _mod_index
522
 
        _mod_index.index_url(self.branch.base)
523
 
 
524
 
    def _branch_search_cb(self, w):
525
 
        from bzrlib.plugins.search import (
526
 
            index as _mod_index,
527
 
            errors as search_errors,
528
 
            )
529
 
        from bzrlib.plugins.gtk.search import SearchDialog
530
 
 
531
 
        try:
532
 
            index = _mod_index.open_index_url(self.branch.base)
533
 
        except search_errors.NoSearchIndex:
534
 
            dialog = Gtk.MessageDialog(self, type=Gtk.MessageType.QUESTION, 
535
 
                buttons=Gtk.ButtonsType.OK_CANCEL, 
536
 
                message_format="This branch has not been indexed yet. "
537
 
                               "Index now?")
538
 
            if dialog.run() == Gtk.ResponseType.OK:
539
 
                dialog.destroy()
540
 
                index = _mod_index.index_url(self.branch.base)
541
 
            else:
542
 
                dialog.destroy()
543
 
                return
544
 
 
545
 
        dialog = SearchDialog(index)
546
 
 
547
 
        if dialog.run() == Gtk.ResponseType.OK:
548
 
            revid = dialog.get_revision()
549
 
            if revid is not None:
550
 
                self.set_revision(revid)
551
 
 
552
 
        dialog.destroy()
553
 
 
554
 
    def _about_dialog_cb(self, w):
555
 
        from bzrlib.plugins.gtk.about import AboutDialog
556
 
        AboutDialog().run()
557
 
 
558
 
    def _col_visibility_changed(self, col, property):
559
 
        self.config.set_user_option(property + '-column-visible', col.get_active())
560
 
        self.treeview.set_property(property + '-column-visible', col.get_active())
561
 
 
562
 
    def _toolbar_visibility_changed(self, col):
563
 
        if col.get_active():
564
 
            self.toolbar.show()
565
 
        else:
566
 
            self.toolbar.hide()
567
 
        self.config.set_user_option('viz-toolbar-visible', col.get_active())
568
 
 
569
 
    def _vertical_layout(self, col):
570
 
        """Toggle the layout vertical/horizontal"""
571
 
        self.config.set_user_option('viz-vertical', str(col.get_active()))
572
 
 
573
 
        old = self.paned
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()
579
 
 
580
 
        self.treeview.emit('revision-selected')
581
 
 
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))
588
 
 
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))
595
 
 
596
 
    def _diff_visibility_changed(self, col):
597
 
        """Hide or show the diff panel."""
598
 
        if col.get_active():
599
 
            self.diff.show()
600
 
            self._make_diff_nonzero_size()
601
 
        else:
602
 
            self.diff.hide()
603
 
        self.config.set_user_option('viz-show-diffs', str(col.get_active()))
604
 
        self.update_diff_panel()
605
 
 
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()))
609
 
 
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()
614
 
 
615
 
        self.treeview.emit('revision-selected')
616
 
 
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)
621
 
 
622
 
    def _refresh_clicked(self, w):
623
 
        self.treeview.refresh()
624
 
 
625
 
    def _update_tags(self):
626
 
        menu = Gtk.Menu()
627
 
 
628
 
        if self.branch.supports_tags():
629
 
            tags = self.branch.tags.get_tag_dict().items()
630
 
            tags.sort(reverse=True)
631
 
            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)
637
 
                tag_item.connect('activate', self._tag_selected_cb, revid)
638
 
                tag_item.set_sensitive(self.treeview.has_revision_id(revid))
639
 
                menu.add(tag_item)
640
 
            self.go_menu_tags.set_submenu(menu)
641
 
 
642
 
            self.go_menu_tags.set_sensitive(len(tags) != 0)
643
 
        else:
644
 
            self.go_menu_tags.set_sensitive(False)
645
 
 
646
 
        self.go_menu_tags.show_all()
647
 
 
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)
652
 
        """
653
 
        size = self.config.get_user_option(name)
654
 
        if size:
655
 
            width, height = [int(num) for num in size.split('x')]
656
 
            # avoid writing config every time we start
657
 
            return width, height
658
 
        return None
659
 
 
660
 
    def show_diff(self, revid, parentid=NULL_REVISION):
 
273
    def show_diff(self, branch, revid, parentid):
661
274
        """Open a new window to show a diff between the given revisions."""
662
275
        from bzrlib.plugins.gtk.diff import DiffWindow
663
276
        window = DiffWindow(parent=self)
664
 
 
665
 
        rev_tree    = self.branch.repository.revision_tree(revid)
666
 
        parent_tree = self.branch.repository.revision_tree(parentid)
667
 
 
668
 
        description = revid + " - " + self.branch._get_nick(local=True)
 
277
        (parent_tree, rev_tree) = branch.repository.revision_trees([parentid, 
 
278
                                                                   revid])
 
279
        description = revid + " - " + branch.nick
669
280
        window.set_diff(description, rev_tree, parent_tree)
670
281
        window.show()
671
282
 
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':
675
 
            return
676
 
 
677
 
        if not revision: # default to selected row
678
 
            revision = self.treeview.get_revision()
679
 
        if revision == NULL_REVISION:
680
 
            return
681
 
 
682
 
        if not parents: # default to selected row's parents
683
 
            parents  = self.treeview.get_parents()
 
283
    def _show_clicked_cb(self, revid, parentid):
 
284
        """Callback for when the show button for a parent is clicked."""
 
285
        self.show_diff(self.branch, revid, parentid)
 
286
        self.treeview.grab_focus()
 
287
 
 
288
    def _treeview_row_mouseclick(self, widget, event):
 
289
        from bzrlib.plugins.gtk.revisionmenu import RevisionPopupMenu
 
290
        if event.button == 3:
 
291
            menu = RevisionPopupMenu(self.branch.repository, 
 
292
                [x.revision_id for x in self.selected_revisions()],
 
293
                self.branch)
 
294
            menu.popup(None, None, None, event.button, event.get_time())
 
295
 
 
296
    def selected_revision(self, path):
 
297
        return self.model[path][treemodel.REVISION]
 
298
 
 
299
    def selected_revisions(self):
 
300
        return [self.selected_revision(path) for path in \
 
301
                self.treeview.get_selection().get_selected_rows()[1]]
 
302
 
 
303
    def _treeview_row_activated_cb(self, widget, path, col):
 
304
        # TODO: more than one parent
 
305
        """Callback for when a treeview row gets activated."""
 
306
        revision_id = self.model[path][treemodel.REVID]
 
307
        parents = self.model[path][treemodel.PARENTS]
684
308
        if len(parents) == 0:
685
 
            parent_id = NULL_REVISION
686
 
        else:
687
 
            parent_id = parents[0]
688
 
 
689
 
        rev_tree    = self.branch.repository.revision_tree(revision.revision_id)
690
 
        parent_tree = self.branch.repository.revision_tree(parent_id)
691
 
 
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)
695
 
        self.diff.show_all()
 
309
            # Ignore revisions without parent
 
310
            return
 
311
        parent_id = parents[0]
 
312
        self.show_diff(self.branch, revision_id, parent_id)
 
313
        self.treeview.grab_focus()