17
18
from bzrlib.plugins.gtk import _i18n
19
from bzrlib.plugins.gtk.ui import GtkProgressBar, ProgressPanel
18
20
from linegraph import linegraph, same_branch
19
21
from graphcell import CellRendererGraph
20
22
from treemodel import TreeModel
21
23
from bzrlib.revision import NULL_REVISION
24
from bzrlib.plugins.gtk import lock
23
27
class TreeView(gtk.VBox):
93
94
(gobject.TYPE_PYOBJECT, gobject.TYPE_PYOBJECT)),
94
95
'tag-added': (gobject.SIGNAL_RUN_FIRST,
96
(gobject.TYPE_STRING, gobject.TYPE_STRING))
97
(gobject.TYPE_STRING, gobject.TYPE_STRING)),
98
'refreshed': (gobject.SIGNAL_RUN_FIRST, gobject.TYPE_NONE,
99
102
def __init__(self, branch, start, maxnum, compact=True):
109
112
gtk.VBox.__init__(self, spacing=0)
111
self.pack_start(self.construct_loading_msg(), expand=False, fill=True)
112
self.connect('revisions-loaded',
113
lambda x: self.loading_msg_box.hide())
114
self.progress_widget = ProgressPanel()
115
self.pack_start(self.progress_widget, expand=False, fill=True)
116
if getattr(ui.ui_factory, "set_progress_bar_widget", None) is not None:
117
# We'are using our own ui, let's tell it to use our widget.
118
ui.ui_factory.set_progress_bar_widget(self.progress_widget)
115
120
self.scrolled_window = gtk.ScrolledWindow()
116
121
self.scrolled_window.set_policy(gtk.POLICY_AUTOMATIC,
133
138
gobject.idle_add(self.populate)
135
self.connect("destroy", lambda x: self.branch.unlock())
140
self.connect("destroy", self._on_destroy)
142
def _on_destroy(self, *ignored):
144
if getattr(ui.ui_factory, "set_progress_bar_widget", None) is not None:
145
# We'are using our own ui, let's tell it to stop using our widget.
146
ui.ui_factory.set_progress_bar_widget(None)
137
148
def do_get_property(self, property):
138
149
if property.name == 'revno-column-visible':
148
159
elif property.name == 'branch':
149
160
return self.branch
150
161
elif property.name == 'revision':
151
return self.model.get_value(self.iter, treemodel.REVISION)
162
return self.model.get_value(self.model.get_iter(self.path),
152
164
elif property.name == 'revision-number':
153
return self.model.get_value(self.iter, treemodel.REVNO)
165
return self.model.get_value(self.model.get_iter(self.path),
154
167
elif property.name == 'children':
155
return self.model.get_value(self.iter, treemodel.CHILDREN)
168
return self.model.get_value(self.model.get_iter(self.path),
156
170
elif property.name == 'parents':
157
return self.model.get_value(self.iter, treemodel.PARENTS)
171
return self.model.get_value(self.model.get_iter(self.path),
159
174
raise AttributeError, 'unknown property %s' % property.name
180
195
"""Return revision id of currently selected revision, or None."""
181
196
return self.get_property('revision')
198
def has_revision_id(self, revision_id):
199
return (revision_id in self.index)
183
201
def set_revision(self, revision):
184
202
self.set_property('revision', revision)
208
226
def add_tag(self, tag, revid=None):
209
227
if revid is None: revid = self.revision.revision_id
229
if lock.release(self.branch):
215
self.branch.lock_write()
231
lock.acquire(self.branch, lock.WRITE)
216
232
self.model.add_tag(tag, revid)
221
self.branch.lock_read()
223
self.emit('tag-added', tag, revid)
234
lock.release(self.branch)
236
lock.acquire(self.branch, lock.READ)
238
self.emit('tag-added', tag, revid)
225
240
def refresh(self):
226
self.loading_msg_box.show()
227
241
gobject.idle_add(self.populate, self.get_revision())
229
243
def update(self):
277
291
should be broken.
281
broken_line_length = 32
283
broken_line_length = None
285
show_graph = self.graph_column.get_visible()
287
self.branch.lock_read()
288
(linegraphdata, index, columns_len) = linegraph(self.branch.repository.get_graph(),
289
(self.start,) , # Sequence of start revisions
295
self.model = TreeModel(self.branch, linegraphdata)
296
self.graph_cell.columns_len = columns_len
297
width = self.graph_cell.get_size(self.treeview)[2]
300
self.graph_column.set_fixed_width(width)
301
self.graph_column.set_max_width(width)
303
self.treeview.set_model(self.model)
305
if not revision or revision == NULL_REVISION:
306
self.treeview.set_cursor(0)
308
self.set_revision(revision)
310
self.emit('revisions-loaded')
294
if getattr(ui.ui_factory, "set_progress_bar_widget", None) is not None:
295
# We'are using our own ui, let's tell it to use our widget.
296
ui.ui_factory.set_progress_bar_widget(self.progress_widget)
297
self.progress_bar = ui.ui_factory.nested_progress_bar()
298
self.progress_bar.update("Loading ancestry graph", 0, 5)
302
broken_line_length = 32
304
broken_line_length = None
306
show_graph = self.graph_column.get_visible()
308
self.branch.lock_read()
309
(linegraphdata, index, columns_len) = linegraph(self.branch.repository.get_graph(),
317
self.model = TreeModel(self.branch, linegraphdata)
318
self.graph_cell.columns_len = columns_len
319
width = self.graph_cell.get_size(self.treeview)[2]
322
self.graph_column.set_fixed_width(width)
323
self.graph_column.set_max_width(width)
325
self.treeview.set_model(self.model)
327
if not revision or revision == NULL_REVISION:
328
self.treeview.set_cursor(0)
330
self.set_revision(revision)
332
self.emit('refreshed')
335
self.progress_bar.finished()
314
337
def construct_treeview(self):
315
338
self.treeview = gtk.TreeView()
322
345
if set_tooltip is not None:
323
346
set_tooltip(treemodel.MESSAGE)
348
self._prev_cursor_path = None
325
349
self.treeview.connect("cursor-changed",
326
350
self._on_selection_changed)
395
419
return self.treeview
397
def construct_loading_msg(self):
398
image_loading = gtk.image_new_from_stock(gtk.STOCK_REFRESH,
399
gtk.ICON_SIZE_BUTTON)
402
label_loading = gtk.Label(
403
_i18n("Please wait, loading ancestral graph..."))
404
label_loading.set_alignment(0.0, 0.5)
407
self.loading_msg_box = gtk.HBox()
408
self.loading_msg_box.set_spacing(5)
409
self.loading_msg_box.set_border_width(5)
410
self.loading_msg_box.pack_start(image_loading, False, False)
411
self.loading_msg_box.pack_start(label_loading, True, True)
412
self.loading_msg_box.show()
414
return self.loading_msg_box
416
421
def _on_selection_changed(self, treeview):
417
422
"""callback for when the treeview changes."""
418
423
(path, focus) = treeview.get_cursor()
420
self.iter = self.model.get_iter(path)
424
if (path is not None) and (path != self._prev_cursor_path):
425
self._prev_cursor_path = path # avoid emitting twice per click
421
427
self.emit('revision-selected')
423
429
def _on_revision_selected(self, widget, event):
424
from bzrlib.plugins.gtk.revisionmenu import RevisionPopupMenu
430
from bzrlib.plugins.gtk.revisionmenu import RevisionMenu
425
431
if event.button == 3:
426
menu = RevisionPopupMenu(self.branch.repository,
432
menu = RevisionMenu(self.branch.repository,
427
433
[self.get_revision().revision_id],
429
435
menu.connect('tag-added', lambda w, t, r: self.add_tag(t, r))