/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 branchview/treeview.py

Reuse the viz treeview in the revision browser.

Show diffs side-by-side

added added

removed removed

Lines of Context:
3
3
 
4
4
"""
5
5
 
6
 
__copyright__ = "Copyright © 2005 Canonical Ltd."
 
6
__copyright__ = "Copyright � 2005 Canonical Ltd."
7
7
__author__    = "Daniel Schierbeck <daniel.schierbeck@gmail.com>"
8
8
 
9
 
from gi.repository import Gtk
10
 
from gi.repository import GObject
11
 
from gi.repository import Pango
 
9
import sys
 
10
import string
 
11
import gtk
 
12
import gobject
 
13
import pango
 
14
import re
 
15
import treemodel
12
16
 
13
 
from bzrlib import ui
 
17
from linegraph import linegraph, same_branch
 
18
from graphcell import CellRendererGraph
 
19
from treemodel import TreeModel
14
20
from bzrlib.revision import NULL_REVISION
15
21
 
16
 
from bzrlib.plugins.gtk import lock
17
 
from bzrlib.plugins.gtk.ui import ProgressPanel
18
 
from bzrlib.plugins.gtk.branchview import treemodel
19
 
from bzrlib.plugins.gtk.branchview.linegraph import linegraph, same_branch
20
 
from bzrlib.plugins.gtk.branchview.graphcell import CellRendererGraph
21
 
 
22
 
 
23
 
class TreeView(Gtk.VBox):
 
22
class TreeView(gtk.VBox):
24
23
 
25
24
    __gproperties__ = {
26
 
        'branch': (GObject.TYPE_PYOBJECT,
 
25
        'branch': (gobject.TYPE_PYOBJECT,
27
26
                   'Branch',
28
27
                   'The Bazaar branch being visualized',
29
 
                   GObject.PARAM_CONSTRUCT_ONLY | GObject.PARAM_WRITABLE),
 
28
                   gobject.PARAM_CONSTRUCT_ONLY | gobject.PARAM_WRITABLE),
30
29
 
31
 
        'revision': (GObject.TYPE_PYOBJECT,
 
30
        'revision': (gobject.TYPE_PYOBJECT,
32
31
                     'Revision',
33
32
                     'The currently selected revision',
34
 
                     GObject.PARAM_READWRITE),
 
33
                     gobject.PARAM_READWRITE),
35
34
 
36
 
        'revision-number': (GObject.TYPE_STRING,
 
35
        'revision-number': (gobject.TYPE_STRING,
37
36
                            'Revision number',
38
37
                            'The number of the selected revision',
39
38
                            '',
40
 
                            GObject.PARAM_READABLE),
 
39
                            gobject.PARAM_READABLE),
41
40
 
42
 
        'children': (GObject.TYPE_PYOBJECT,
 
41
        'children': (gobject.TYPE_PYOBJECT,
43
42
                     'Child revisions',
44
43
                     'Children of the currently selected revision',
45
 
                     GObject.PARAM_READABLE),
 
44
                     gobject.PARAM_READABLE),
46
45
 
47
 
        'parents': (GObject.TYPE_PYOBJECT,
 
46
        'parents': (gobject.TYPE_PYOBJECT,
48
47
                    'Parent revisions',
49
48
                    'Parents to the currently selected revision',
50
 
                    GObject.PARAM_READABLE),
 
49
                    gobject.PARAM_READABLE),
51
50
 
52
 
        'revno-column-visible': (GObject.TYPE_BOOLEAN,
53
 
                                 'Revision number column',
 
51
        'revno-column-visible': (gobject.TYPE_BOOLEAN,
 
52
                                 'Revision number',
54
53
                                 'Show revision number column',
55
54
                                 True,
56
 
                                 GObject.PARAM_READWRITE),
57
 
 
58
 
        'graph-column-visible': (GObject.TYPE_BOOLEAN,
59
 
                                 'Graph column',
60
 
                                 'Show graph column',
61
 
                                 True,
62
 
                                 GObject.PARAM_READWRITE),
63
 
 
64
 
        'date-column-visible': (GObject.TYPE_BOOLEAN,
 
55
                                 gobject.PARAM_READWRITE),
 
56
 
 
57
        'date-column-visible': (gobject.TYPE_BOOLEAN,
65
58
                                 'Date',
66
59
                                 'Show date column',
67
60
                                 False,
68
 
                                 GObject.PARAM_READWRITE),
 
61
                                 gobject.PARAM_READWRITE),
69
62
 
70
 
        'compact': (GObject.TYPE_BOOLEAN,
 
63
        'compact': (gobject.TYPE_BOOLEAN,
71
64
                    'Compact view',
72
65
                    'Break ancestry lines to save space',
73
66
                    True,
74
 
                    GObject.PARAM_CONSTRUCT | GObject.PARAM_READWRITE),
75
 
 
76
 
        'mainline-only': (GObject.TYPE_BOOLEAN,
77
 
                    'Mainline only',
78
 
                    'Only show the mainline history.',
79
 
                    False,
80
 
                    GObject.PARAM_CONSTRUCT | GObject.PARAM_READWRITE),
 
67
                    gobject.PARAM_CONSTRUCT | gobject.PARAM_READWRITE)
81
68
 
82
69
    }
83
70
 
84
71
    __gsignals__ = {
85
 
        'revision-selected': (GObject.SignalFlags.RUN_FIRST,
86
 
                              None,
 
72
        'revisions-loaded': (gobject.SIGNAL_RUN_FIRST, 
 
73
                             gobject.TYPE_NONE,
 
74
                             ()),
 
75
        'revision-selected': (gobject.SIGNAL_RUN_FIRST,
 
76
                              gobject.TYPE_NONE,
87
77
                              ()),
88
 
        'revision-activated': (GObject.SignalFlags.RUN_FIRST,
89
 
                              None,
90
 
                              (GObject.TYPE_PYOBJECT, GObject.TYPE_PYOBJECT)),
91
 
        'tag-added': (GObject.SignalFlags.RUN_FIRST,
92
 
                              None,
93
 
                              (GObject.TYPE_STRING, GObject.TYPE_STRING)),
94
 
        'refreshed': (GObject.SignalFlags.RUN_FIRST, None,
95
 
                              ())
 
78
        'revision-activated': (gobject.SIGNAL_RUN_FIRST,
 
79
                              gobject.TYPE_NONE,
 
80
                              (gobject.TYPE_PYOBJECT, gobject.TYPE_PYOBJECT)),
 
81
        'tag-added': (gobject.SIGNAL_RUN_FIRST,
 
82
                              gobject.TYPE_NONE,
 
83
                              (gobject.TYPE_STRING, gobject.TYPE_STRING))
96
84
    }
97
85
 
98
86
    def __init__(self, branch, start, maxnum, compact=True):
105
93
        :param broken_line_length: After how much lines to break 
106
94
                                   branches.
107
95
        """
108
 
        super(TreeView, self).__init__(homogeneous=False, spacing=0)
109
 
 
110
 
        self.progress_widget = ProgressPanel()
111
 
        self.pack_start(self.progress_widget, False, True, 0)
112
 
        if getattr(ui.ui_factory, "set_progress_bar_widget", None) is not None:
113
 
            # We'are using our own ui, let's tell it to use our widget.
114
 
            ui.ui_factory.set_progress_bar_widget(self.progress_widget)
115
 
 
116
 
        self.scrolled_window = Gtk.ScrolledWindow()
117
 
        self.scrolled_window.set_policy(Gtk.PolicyType.AUTOMATIC,
118
 
                                        Gtk.PolicyType.AUTOMATIC)
119
 
        self.scrolled_window.set_shadow_type(Gtk.ShadowType.IN)
 
96
        gtk.VBox.__init__(self, spacing=0)
 
97
 
 
98
        self.pack_start(self.construct_loading_msg(), expand=False, fill=True)
 
99
        self.connect('revisions-loaded', 
 
100
                lambda x: self.loading_msg_box.hide())
 
101
 
 
102
        self.scrolled_window = gtk.ScrolledWindow()
 
103
        self.scrolled_window.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
 
104
        self.scrolled_window.set_shadow_type(gtk.SHADOW_IN)
120
105
        self.scrolled_window.show()
121
 
        self.pack_start(self.scrolled_window, True, True, 0)
 
106
        self.pack_start(self.scrolled_window, expand=True, fill=True)
122
107
 
123
108
        self.scrolled_window.add(self.construct_treeview())
 
109
        
124
110
 
125
 
        self.path = None
 
111
        self.iter = None
126
112
        self.branch = branch
127
113
        self.revision = None
128
 
        self.index = {}
129
114
 
130
115
        self.start = start
131
116
        self.maxnum = maxnum
132
117
        self.compact = compact
133
118
 
134
 
        self.model = treemodel.BranchTreeModel(self.branch, [])
135
 
        GObject.idle_add(self.populate)
136
 
 
137
 
        self.connect("destroy", self._on_destroy)
138
 
 
139
 
    def _on_destroy(self, *ignored):
140
 
        self.branch.unlock()
141
 
        if getattr(ui.ui_factory, "set_progress_bar_widget", None) is not None:
142
 
            # We'are using our own ui, let's tell it to stop using our widget.
143
 
            ui.ui_factory.set_progress_bar_widget(None)
 
119
        gobject.idle_add(self.populate)
 
120
 
 
121
        self.connect("destroy", lambda x: self.branch.unlock())
144
122
 
145
123
    def do_get_property(self, property):
146
124
        if property.name == 'revno-column-visible':
147
125
            return self.revno_column.get_visible()
148
 
        elif property.name == 'graph-column-visible':
149
 
            return self.graph_column.get_visible()
150
126
        elif property.name == 'date-column-visible':
151
127
            return self.date_column.get_visible()
152
128
        elif property.name == 'compact':
153
129
            return self.compact
154
 
        elif property.name == 'mainline-only':
155
 
            return self.mainline_only
156
130
        elif property.name == 'branch':
157
131
            return self.branch
158
132
        elif property.name == 'revision':
159
 
            if self.path is None:
160
 
                return None
161
 
            return self.model.get_value(self.model.get_iter(self.path),
162
 
                                        treemodel.REVISION)
 
133
            return self.model.get_value(self.iter, treemodel.REVISION)
163
134
        elif property.name == 'revision-number':
164
 
            if self.path is None:
165
 
                return None
166
 
            return self.model.get_value(self.model.get_iter(self.path),
167
 
                                        treemodel.REVNO)
 
135
            return self.model.get_value(self.iter, treemodel.REVNO)
168
136
        elif property.name == 'children':
169
 
            if self.path is None:
170
 
                return None
171
 
            return self.model.get_value(self.model.get_iter(self.path),
172
 
                                        treemodel.CHILDREN)
 
137
            return self.model.get_value(self.iter, treemodel.CHILDREN)
173
138
        elif property.name == 'parents':
174
 
            if self.path is None:
175
 
                return None
176
 
            return self.model.get_value(self.model.get_iter(self.path),
177
 
                                        treemodel.PARENTS)
 
139
            return self.model.get_value(self.iter, treemodel.PARENTS)
178
140
        else:
179
141
            raise AttributeError, 'unknown property %s' % property.name
180
142
 
181
143
    def do_set_property(self, property, value):
182
144
        if property.name == 'revno-column-visible':
183
145
            self.revno_column.set_visible(value)
184
 
        elif property.name == 'graph-column-visible':
185
 
            self.graph_column.set_visible(value)
186
146
        elif property.name == 'date-column-visible':
187
147
            self.date_column.set_visible(value)
188
148
        elif property.name == 'compact':
189
149
            self.compact = value
190
 
        elif property.name == 'mainline-only':
191
 
            self.mainline_only = value
192
150
        elif property.name == 'branch':
193
151
            self.branch = value
194
152
        elif property.name == 'revision':
200
158
        """Return revision id of currently selected revision, or None."""
201
159
        return self.get_property('revision')
202
160
 
203
 
    def has_revision_id(self, revision_id):
204
 
        return (revision_id in self.index)
205
 
 
206
161
    def set_revision(self, revision):
207
162
        self.set_property('revision', revision)
208
163
 
211
166
 
212
167
        :param revid: Revision id of revision to display.
213
168
        """
214
 
        self.treeview.set_cursor(
215
 
            Gtk.TreePath(path=self.index[revid]), None, False)
 
169
        self.treeview.set_cursor(self.index[revid])
216
170
        self.treeview.grab_focus()
217
171
 
218
172
    def get_children(self):
232
186
    def add_tag(self, tag, revid=None):
233
187
        if revid is None: revid = self.revision.revision_id
234
188
 
235
 
        if lock.release(self.branch):
 
189
        try:
 
190
            self.branch.unlock()
 
191
 
236
192
            try:
237
 
                lock.acquire(self.branch, lock.WRITE)
 
193
                self.branch.lock_write()
238
194
                self.model.add_tag(tag, revid)
239
195
            finally:
240
 
                lock.release(self.branch)
241
 
 
242
 
            lock.acquire(self.branch, lock.READ)
243
 
 
244
 
            self.emit('tag-added', tag, revid)
245
 
 
 
196
                self.branch.unlock()
 
197
 
 
198
        finally:
 
199
            self.branch.lock_read()
 
200
 
 
201
        self.emit('tag-added', tag, revid)
 
202
        
246
203
    def refresh(self):
247
 
        GObject.idle_add(self.populate, self.get_revision())
 
204
        self.loading_msg_box.show()
 
205
        gobject.idle_add(self.populate, self.get_revision())
248
206
 
249
207
    def update(self):
250
208
        try:
260
218
    def back(self):
261
219
        """Signal handler for the Back button."""
262
220
        parents = self.get_parents()
263
 
        if not parents:
 
221
        if not len(parents):
264
222
            return
265
223
 
266
224
        for parent_id in parents:
275
233
    def forward(self):
276
234
        """Signal handler for the Forward button."""
277
235
        children = self.get_children()
278
 
        if not children:
 
236
        if not len(children):
279
237
            return
280
238
 
281
239
        for child_id in children:
297
255
                       should be broken.
298
256
        """
299
257
 
300
 
        if getattr(ui.ui_factory, "set_progress_bar_widget", None) is not None:
301
 
            # We'are using our own ui, let's tell it to use our widget.
302
 
            ui.ui_factory.set_progress_bar_widget(self.progress_widget)
303
 
        self.progress_bar = ui.ui_factory.nested_progress_bar()
304
 
        self.progress_bar.update("Loading ancestry graph", 0, 5)
305
 
 
306
 
        try:
307
 
            if self.compact:
308
 
                broken_line_length = 32
309
 
            else:
310
 
                broken_line_length = None
311
 
 
312
 
            show_graph = self.graph_column.get_visible()
313
 
 
314
 
            self.branch.lock_read()
315
 
            (linegraphdata, index, columns_len) = linegraph(
316
 
                self.branch.repository.get_graph(),
317
 
                self.start,
318
 
                self.maxnum, 
319
 
                broken_line_length,
320
 
                show_graph,
321
 
                self.mainline_only,
322
 
                self.progress_bar)
323
 
 
324
 
            self.model.set_line_graph_data(linegraphdata)
325
 
            self.graph_cell.columns_len = columns_len
326
 
            width = self.graph_cell.get_preferred_width(self.treeview)[1]
327
 
            if width > 500:
328
 
                width = 500
329
 
            elif width == 0:
330
 
                # The get_preferred_width() call got an insane value.
331
 
                width = 200
332
 
            self.graph_column.set_fixed_width(width)
333
 
            self.graph_column.set_max_width(width)
334
 
            self.index = index
335
 
            self.treeview.set_model(self.model)
336
 
 
337
 
            if not revision or revision == NULL_REVISION:
338
 
                self.treeview.set_cursor(Gtk.TreePath(path=0), None, False)
339
 
            else:
340
 
                self.set_revision(revision)
341
 
 
342
 
            self.emit('refreshed')
343
 
            return False
344
 
        finally:
345
 
            self.progress_bar.finished()
 
258
        if self.compact:
 
259
            broken_line_length = 32
 
260
        else:
 
261
            broken_line_length = None
 
262
 
 
263
        self.branch.lock_read()
 
264
        (linegraphdata, index, columns_len) = linegraph(self.branch.repository,
 
265
                                                        self.start,
 
266
                                                        self.maxnum, 
 
267
                                                        broken_line_length)
 
268
 
 
269
        self.model = TreeModel(self.branch, linegraphdata)
 
270
        self.graph_cell.columns_len = columns_len
 
271
        width = self.graph_cell.get_size(self.treeview)[2]
 
272
        if width > 500:
 
273
            width = 500
 
274
        self.graph_column.set_fixed_width(width)
 
275
        self.graph_column.set_max_width(width)
 
276
        self.index = index
 
277
        self.treeview.set_model(self.model)
 
278
 
 
279
        if revision is None:
 
280
            self.treeview.set_cursor(0)
 
281
        else:
 
282
            self.set_revision(revision)
 
283
 
 
284
        self.emit('revisions-loaded')
 
285
 
 
286
        return False
346
287
 
347
288
    def construct_treeview(self):
348
 
        self.treeview = Gtk.TreeView()
 
289
        self.treeview = gtk.TreeView()
349
290
 
350
291
        self.treeview.set_rules_hint(True)
351
 
        # combined revno/summary interactive search
352
 
        #
353
 
        # the row in a treemodel is considered "matched" if a REVNO *starts*
354
 
        # from the key (that is the key is found in a REVNO at the offset 0)
355
 
        # or if a MESSAGE *contains* the key anywhere (that is, the key is
356
 
        # found case insensitively in a MESSAGE at any offset)
357
 
        def search_equal_func(model, column, key, iter, ignored):
358
 
            return (model.get_value(iter, treemodel.REVNO).find(key) != 0
359
 
                and model.get_value(iter, treemodel.MESSAGE).lower().find(key.lower()) == -1)
360
 
 
361
 
        self.treeview.set_search_equal_func(search_equal_func, None)
362
 
        self.treeview.set_enable_search(True)
363
 
 
364
 
        self.treeview.set_tooltip_column(treemodel.MESSAGE)
365
 
        self.treeview.set_headers_visible(True)
366
 
 
367
 
        self._prev_cursor_path = None
 
292
        self.treeview.set_search_column(treemodel.REVNO)
 
293
        
 
294
        # Fix old PyGTK bug - by JAM
 
295
        set_tooltip = getattr(self.treeview, 'set_tooltip_column', None)
 
296
        if set_tooltip is not None:
 
297
            set_tooltip(treemodel.MESSAGE)
 
298
 
368
299
        self.treeview.connect("cursor-changed",
369
300
                self._on_selection_changed)
370
301
 
378
309
 
379
310
        self.treeview.show()
380
311
 
381
 
        cell = Gtk.CellRendererText()
 
312
        cell = gtk.CellRendererText()
382
313
        cell.set_property("width-chars", 15)
383
 
        cell.set_property("ellipsize", Pango.EllipsizeMode.END)
384
 
        self.revno_column = Gtk.TreeViewColumn("Revision No")
 
314
        cell.set_property("ellipsize", pango.ELLIPSIZE_END)
 
315
        self.revno_column = gtk.TreeViewColumn("Revision No")
385
316
        self.revno_column.set_resizable(True)
386
 
        self.revno_column.set_sizing(Gtk.TreeViewColumnSizing.FIXED)
387
 
        self.revno_column.set_fixed_width(
388
 
            cell.get_preferred_width(self.treeview)[1])
389
 
        self.revno_column.pack_start(cell, True)
 
317
        self.revno_column.set_sizing(gtk.TREE_VIEW_COLUMN_FIXED)
 
318
        self.revno_column.set_fixed_width(cell.get_size(self.treeview)[2])
 
319
        self.revno_column.pack_start(cell, expand=True)
390
320
        self.revno_column.add_attribute(cell, "text", treemodel.REVNO)
391
321
        self.treeview.append_column(self.revno_column)
392
322
 
393
323
        self.graph_cell = CellRendererGraph()
394
 
        self.graph_column = Gtk.TreeViewColumn()
 
324
        self.graph_column = gtk.TreeViewColumn()
395
325
        self.graph_column.set_resizable(True)
396
 
        self.graph_column.set_sizing(Gtk.TreeViewColumnSizing.FIXED)
397
 
        self.graph_column.pack_start(self.graph_cell, True)
398
 
        self.graph_column.add_attribute(
399
 
            self.graph_cell, "node", treemodel.NODE)
400
 
        self.graph_column.add_attribute(
401
 
            self.graph_cell, "tags", treemodel.TAGS)
402
 
        self.graph_column.add_attribute(
403
 
            self.graph_cell, "in-lines", treemodel.LAST_LINES)
404
 
        self.graph_column.add_attribute(
405
 
            self.graph_cell, "out-lines", treemodel.LINES)
 
326
        self.graph_column.set_sizing(gtk.TREE_VIEW_COLUMN_FIXED)
 
327
        self.graph_column.pack_start(self.graph_cell, expand=False)
 
328
        self.graph_column.add_attribute(self.graph_cell, "node", treemodel.NODE)
 
329
        self.graph_column.add_attribute(self.graph_cell, "tags", treemodel.TAGS)
 
330
        self.graph_column.add_attribute(self.graph_cell, "in-lines", treemodel.LAST_LINES)
 
331
        self.graph_column.add_attribute(self.graph_cell, "out-lines", treemodel.LINES)
406
332
        self.treeview.append_column(self.graph_column)
407
333
 
408
 
        cell = Gtk.CellRendererText()
 
334
        cell = gtk.CellRendererText()
409
335
        cell.set_property("width-chars", 65)
410
 
        cell.set_property("ellipsize", Pango.EllipsizeMode.END)
411
 
        self.summary_column = Gtk.TreeViewColumn("Summary")
 
336
        cell.set_property("ellipsize", pango.ELLIPSIZE_END)
 
337
        self.summary_column = gtk.TreeViewColumn("Summary")
412
338
        self.summary_column.set_resizable(True)
413
 
        self.summary_column.set_expand(True)
414
 
        self.summary_column.set_sizing(Gtk.TreeViewColumnSizing.FIXED)
415
 
        self.summary_column.set_fixed_width(
416
 
            cell.get_preferred_width(self.treeview)[1])
417
 
        self.summary_column.pack_start(cell, True)
 
339
        self.summary_column.set_sizing(gtk.TREE_VIEW_COLUMN_FIXED)
 
340
        self.summary_column.set_fixed_width(cell.get_size(self.treeview)[2])
 
341
        self.summary_column.pack_start(cell, expand=True)
418
342
        self.summary_column.add_attribute(cell, "markup", treemodel.SUMMARY)
419
343
        self.treeview.append_column(self.summary_column)
420
344
 
421
 
        cell = Gtk.CellRendererText()
 
345
        cell = gtk.CellRendererText()
422
346
        cell.set_property("width-chars", 15)
423
 
        cell.set_property("ellipsize", Pango.EllipsizeMode.END)
424
 
        self.authors_column = Gtk.TreeViewColumn("Author(s)")
425
 
        self.authors_column.set_resizable(False)
426
 
        self.authors_column.set_sizing(Gtk.TreeViewColumnSizing.FIXED)
427
 
        self.authors_column.set_fixed_width(200)
428
 
        self.authors_column.pack_start(cell, True)
429
 
        self.authors_column.add_attribute(cell, "text", treemodel.AUTHORS)
430
 
        self.treeview.append_column(self.authors_column)
 
347
        cell.set_property("ellipsize", pango.ELLIPSIZE_END)
 
348
        self.committer_column = gtk.TreeViewColumn("Committer")
 
349
        self.committer_column.set_resizable(True)
 
350
        self.committer_column.set_sizing(gtk.TREE_VIEW_COLUMN_FIXED)
 
351
        self.committer_column.set_fixed_width(cell.get_size(self.treeview)[2])
 
352
        self.committer_column.pack_start(cell, expand=True)
 
353
        self.committer_column.add_attribute(cell, "text", treemodel.COMMITTER)
 
354
        self.treeview.append_column(self.committer_column)
431
355
 
432
 
        cell = Gtk.CellRendererText()
 
356
        cell = gtk.CellRendererText()
433
357
        cell.set_property("width-chars", 20)
434
 
        cell.set_property("ellipsize", Pango.EllipsizeMode.END)
435
 
        self.date_column = Gtk.TreeViewColumn("Date")
 
358
        cell.set_property("ellipsize", pango.ELLIPSIZE_END)
 
359
        self.date_column = gtk.TreeViewColumn("Date")
436
360
        self.date_column.set_visible(False)
437
361
        self.date_column.set_resizable(True)
438
 
        self.date_column.set_sizing(Gtk.TreeViewColumnSizing.FIXED)
439
 
        self.date_column.set_fixed_width(130)
440
 
        self.date_column.pack_start(cell, True)
 
362
        self.date_column.set_sizing(gtk.TREE_VIEW_COLUMN_FIXED)
 
363
        self.date_column.set_fixed_width(cell.get_size(self.treeview)[2])
 
364
        self.date_column.pack_start(cell, expand=True)
441
365
        self.date_column.add_attribute(cell, "text", treemodel.TIMESTAMP)
442
366
        self.treeview.append_column(self.date_column)
443
 
 
 
367
        
444
368
        return self.treeview
 
369
    
 
370
    def construct_loading_msg(self):
 
371
        image_loading = gtk.image_new_from_stock(gtk.STOCK_REFRESH,
 
372
                                                 gtk.ICON_SIZE_BUTTON)
 
373
        image_loading.show()
 
374
        
 
375
        label_loading = gtk.Label(_("Please wait, loading ancestral graph..."))
 
376
        label_loading.set_alignment(0.0, 0.5)
 
377
        label_loading.show()
 
378
        
 
379
        self.loading_msg_box = gtk.HBox()
 
380
        self.loading_msg_box.set_spacing(5)
 
381
        self.loading_msg_box.set_border_width(5)        
 
382
        self.loading_msg_box.pack_start(image_loading, False, False)
 
383
        self.loading_msg_box.pack_start(label_loading, True, True)
 
384
        self.loading_msg_box.show()
 
385
        
 
386
        return self.loading_msg_box
445
387
 
446
388
    def _on_selection_changed(self, treeview):
447
389
        """callback for when the treeview changes."""
448
390
        (path, focus) = treeview.get_cursor()
449
 
        if (path is not None) and (path != self._prev_cursor_path):
450
 
            self._prev_cursor_path = path # avoid emitting twice per click
451
 
            self.path = path
 
391
        if path is not None:
 
392
            self.iter = self.model.get_iter(path)
452
393
            self.emit('revision-selected')
453
394
 
454
395
    def _on_revision_selected(self, widget, event):
455
 
        from bzrlib.plugins.gtk.revisionmenu import RevisionMenu
 
396
        from bzrlib.plugins.gtk.revisionmenu import RevisionPopupMenu
456
397
        if event.button == 3:
457
 
            revs = []
458
 
            rev = self.get_revision()
459
 
            if rev is not None:
460
 
                revs.append(rev.revision_id)
461
 
            menu = RevisionMenu(self.branch.repository, revs, self.branch)
 
398
            menu = RevisionPopupMenu(self.branch.repository, 
 
399
                [self.get_revision().revision_id],
 
400
                self.branch)
462
401
            menu.connect('tag-added', lambda w, t, r: self.add_tag(t, r))
463
 
            menu.popup(None, None, None, None, event.button, event.get_time())
 
402
            menu.popup(None, None, None, event.button, event.get_time())
464
403
 
465
404
    def _on_revision_activated(self, widget, path, col):
466
405
        self.emit('revision-activated', path, col)