45
class GAnnotateWindow(Window):
44
class GAnnotateWindow(gtk.Window):
46
45
"""Annotate window."""
48
def __init__(self, all=False, plain=False, parent=None, branch=None):
47
def __init__(self, all=False, plain=False):
53
Window.__init__(self, parent)
51
gtk.Window.__init__(self, gtk.WINDOW_TOPLEVEL)
55
53
self.set_icon(self.render_icon(gtk.STOCK_FIND, gtk.ICON_SIZE_BUTTON))
56
54
self.annotate_colormap = AnnotateColorSaturation()
59
57
self.revisions = {}
63
59
def annotate(self, tree, branch, file_id):
64
60
self.annotations = []
65
61
self.branch = branch
67
63
self.file_id = file_id
68
self.revisionview.set_file_id(file_id)
69
64
self.revision_id = getattr(tree, 'get_revision_id',
70
65
lambda: CURRENT_REVISION)()
72
# [revision id, line number, author, revno, highlight color, line]
67
# [revision id, line number, committer, revno, highlight color, line]
73
68
self.annomodel = gtk.ListStore(gobject.TYPE_STRING,
74
69
gobject.TYPE_STRING,
75
70
gobject.TYPE_STRING,
83
78
branch.repository.lock_read()
85
revno_map = self.branch.get_revision_id_to_revno_map()
86
for revision_id, revno in revno_map.iteritems():
87
self.dotted[revision_id] = '.'.join(str(num) for num in revno)
88
79
for line_no, (revision, revno, line)\
89
80
in enumerate(self._annotate(tree, file_id)):
90
81
if revision.revision_id == last_seen and not self.all:
82
revno = committer = ""
93
84
last_seen = revision.revision_id
94
author = ", ".join(revision.get_apparent_authors())
85
committer = revision.committer
96
87
if revision.revision_id not in self.revisions:
97
88
self.revisions[revision.revision_id] = revision
99
90
self.annomodel.append([revision.revision_id,
104
95
line.rstrip("\r\n")
132
120
self.annoview.set_cursor(row)
133
121
self.annoview.scroll_to_cell(row, use_align=True)
123
def _dotted_revnos(self, repository, revision_id):
124
"""Return a dict of revision_id -> dotted revno
126
:param repository: The repository to get the graph from
127
:param revision_id: The last revision for which this info is needed
129
graph = repository.get_revision_graph(revision_id)
131
for n, revision_id, d, revno, e in tsort.merge_sort(graph,
132
revision_id, generate_revno=True):
133
dotted[revision_id] = '.'.join(str(num) for num in revno)
136
136
def _annotate(self, tree, file_id):
137
137
current_revision = FakeRevision(CURRENT_REVISION)
139
139
current_revision.timestamp = time.time()
140
140
current_revision.message = '[Not yet committed]'
141
141
current_revision.parent_ids = tree.get_parent_ids()
142
current_revision.properties['branch-nick'] = self.branch._get_nick(local=True)
143
142
current_revno = '%d?' % (self.branch.revno() + 1)
144
143
repository = self.branch.repository
145
144
if self.revision_id == CURRENT_REVISION:
146
145
revision_id = self.branch.last_revision()
148
147
revision_id = self.revision_id
148
dotted = self._dotted_revnos(repository, revision_id)
149
149
revision_cache = RevisionCache(repository, self.revisions)
150
150
for origin, text in tree.annotate_iter(file_id):
177
177
return self.annomodel[path][REVISION_ID_COL]
179
def _activate_selected_revision(self, w):
179
def _show_log(self, w):
180
180
rev_id = self._selected_revision()
181
if not rev_id or rev_id == NULL_REVISION:
183
selected = self.revisions[rev_id]
184
self.revisionview.set_revision(selected)
185
if (len(selected.parent_ids) != 0 and selected.parent_ids[0] not in
190
self.back_button.set_sensitive(enable_back)
183
self.logview.set_revision(self.revisions[rev_id])
192
185
def _create(self):
193
self.revisionview = self._create_log_view()
186
self.logview = self._create_log_view()
194
187
self.annoview = self._create_annotate_view()
196
vbox = gtk.VBox(False)
189
vbox = gtk.VBox(False, 12)
190
vbox.set_border_width(12)
199
193
sw = gtk.ScrolledWindow()
202
196
sw.add(self.annoview)
203
197
self.annoview.gwindow = self
210
hbox = gtk.HBox(False, 6)
211
self.back_button = self._create_back_button()
212
hbox.pack_start(self.back_button, expand=False, fill=True)
213
self.forward_button = self._create_forward_button()
214
hbox.pack_start(self.forward_button, expand=False, fill=True)
215
self.find_button = self._create_find_button()
216
hbox.pack_start(self.find_button, expand=False, fill=True)
217
self.goto_button = self._create_goto_button()
218
hbox.pack_start(self.goto_button, expand=False, fill=True)
220
vbox.pack_start(hbox, expand=False, fill=True)
222
200
self.pane = pane = gtk.VPaned()
224
pane.add2(self.revisionview)
202
pane.add2(self.logview)
226
204
vbox.pack_start(pane, expand=True, fill=True)
228
206
self._search = SearchBox()
229
swbox.pack_start(self._search, expand=False, fill=True)
207
vbox.pack_start(self._search, expand=False, fill=True)
230
208
accels = gtk.AccelGroup()
231
209
accels.connect_group(gtk.keysyms.f, gtk.gdk.CONTROL_MASK,
232
210
gtk.ACCEL_LOCKED,
236
214
self._search_by_line)
237
215
self.add_accel_group(accels)
217
hbox = gtk.HBox(True, 6)
218
hbox.pack_start(self._create_prev_button(), expand=False, fill=True)
219
hbox.pack_end(self._create_button_box(), expand=False, fill=True)
221
vbox.pack_start(hbox, expand=False, fill=True)
241
def _search_by_text(self, *ignored): # (accel_group, window, key, modifiers):
225
def _search_by_text(self, accel_group, window, key, modifiers):
242
226
self._search.show_for('text')
243
227
self._search.set_target(self.annoview, TEXT_LINE_COL)
245
def _search_by_line(self, *ignored): # accel_group, window, key, modifiers):
229
def _search_by_line(self, accel_group, window, key, modifiers):
246
230
self._search.show_for('line')
247
231
self._search.set_target(self.annoview, LINE_NUM_COL)
249
def line_diff(self, tv, path, tvc):
233
def row_diff(self, tv, path, tvc):
251
235
revision = self.annotations[row]
252
236
repository = self.branch.repository
327
311
def _create_log_view(self):
328
lv = RevisionView(self._branch)
332
def _create_back_button(self):
316
def _create_button_box(self):
317
box = gtk.HButtonBox()
318
box.set_layout(gtk.BUTTONBOX_END)
321
button = gtk.Button()
322
button.set_use_stock(True)
323
button.set_label("gtk-close")
324
button.connect("clicked", lambda w: self.destroy())
327
box.pack_start(button, expand=False, fill=False)
331
def _create_prev_button(self):
332
box = gtk.HButtonBox()
333
box.set_layout(gtk.BUTTONBOX_START)
333
336
button = gtk.Button()
334
337
button.set_use_stock(True)
335
338
button.set_label("gtk-go-back")
336
339
button.connect("clicked", lambda w: self.go_back())
337
button.set_relief(gtk.RELIEF_NONE)
341
def _create_forward_button(self):
342
button = gtk.Button()
343
button.set_use_stock(True)
344
button.set_label("gtk-go-forward")
345
button.connect("clicked", lambda w: self.go_forward())
346
button.set_relief(gtk.RELIEF_NONE)
348
button.set_sensitive(False)
351
def _create_find_button(self):
352
button = gtk.Button()
353
button.set_use_stock(True)
354
button.set_label("gtk-find")
355
button.set_tooltip_text("Search for text (Ctrl+F)")
356
button.connect("clicked", self._search_by_text)
357
button.set_relief(gtk.RELIEF_NONE)
359
button.set_sensitive(True)
362
def _create_goto_button(self):
363
button = gtk.Button()
364
button.set_label("Goto Line")
365
button.set_tooltip_text("Scroll to a line by entering its number (Ctrl+G)")
366
button.connect("clicked", self._search_by_line)
367
button.set_relief(gtk.RELIEF_NONE)
369
button.set_sensitive(True)
341
box.pack_start(button, expand=False, fill=False)
372
344
def go_back(self):
373
last_tree = self.tree
374
345
rev_id = self._selected_revision()
375
346
parent_id = self.revisions[rev_id].parent_ids[0]
376
target_tree = self.branch.repository.revision_tree(parent_id)
377
if self._go(target_tree):
378
self.history.append(last_tree)
379
self.forward_button.set_sensitive(True)
381
self._no_back.add(parent_id)
382
self.back_button.set_sensitive(False)
384
def go_forward(self):
385
if len(self.history) == 0:
387
target_tree = self.history.pop()
388
if len(self.history) == 0:
389
self.forward_button.set_sensitive(False)
390
self._go(target_tree)
392
def _go(self, target_tree):
393
rev_id = self._selected_revision()
394
if self.file_id in target_tree:
395
offset = self.get_scroll_offset(target_tree)
347
tree = self.branch.repository.revision_tree(parent_id)
348
if self.file_id in tree:
349
offset = self.get_scroll_offset(tree)
396
350
(row,), col = self.annoview.get_cursor()
397
self.annotate(target_tree, self.branch, self.file_id)
401
self.annoview.set_cursor(new_row)
351
self.annotate(tree, self.branch, self.file_id)
352
self.annoview.set_cursor(row+offset)
406
354
def get_scroll_offset(self, tree):
407
355
old = self.tree.get_file(self.file_id)
417
class FakeRevision(object):
418
367
""" A fake revision.
420
369
For when a revision is referenced but not present.
423
def __init__(self, revision_id, committer='?', nick=None):
372
def __init__(self, revision_id, committer='?'):
424
373
self.revision_id = revision_id
425
374
self.parent_ids = []
426
375
self.committer = committer
427
376
self.message = "?"
428
377
self.timestamp = 0.0
429
378
self.timezone = 0
432
def get_apparent_authors(self):
433
return [self.committer]
436
382
class RevisionCache(object):
437
383
"""A caching revision source"""
439
384
def __init__(self, real_source, seed_cache=None):
440
385
self.__real_source = real_source
441
386
if seed_cache is None: