19
19
pygtk.require("2.0")
25
from bzrlib import trace
23
26
from bzrlib.osutils import format_date
26
class LogView(gtk.ScrolledWindow):
28
from bzrlib.bencode import bdecode
30
from bzrlib.util.bencode import bdecode
31
from bzrlib.testament import Testament
33
from bzrlib.plugins.gtk import icon_path
35
from bzrlib.plugins.gtk.avatarsbox import AvatarsBox
38
from bzrlib.plugins.gtk import seahorse
50
def _open_link(widget, uri):
51
for cmd in ['sensible-browser', 'xdg-open']:
52
if webbrowser._iscommand(cmd):
53
webbrowser._tryorder.insert(0, '%s "%%s"' % cmd)
56
if getattr(gtk, 'link_button_set_uri_hook', None) is not None:
57
# Not available before PyGtk-2.10
58
gtk.link_button_set_uri_hook(_open_link)
60
class BugsTab(gtk.VBox):
63
super(BugsTab, self).__init__(False, 6)
65
table = gtk.Table(rows=2, columns=2)
67
table.set_row_spacings(6)
68
table.set_col_spacing(0, 16)
71
image.set_from_file(icon_path("bug.png"))
72
table.attach(image, 0, 1, 0, 1, gtk.FILL)
74
align = gtk.Alignment(0.0, 0.1)
75
self.label = gtk.Label()
77
table.attach(align, 1, 2, 0, 1, gtk.FILL)
79
treeview = self.construct_treeview()
80
table.attach(treeview, 1, 2, 1, 2, gtk.FILL | gtk.EXPAND)
82
self.set_border_width(6)
83
self.pack_start(table, expand=False)
88
def set_revision(self, revision):
93
bugs_text = revision.properties.get('bugs', '')
94
for bugline in bugs_text.splitlines():
95
(url, status) = bugline.split(" ")
97
self.add_bug(url, status)
99
if self.num_bugs == 0:
101
elif self.num_bugs == 1:
106
self.label.set_markup("<b>Bugs fixed</b>\n" +
107
"This revision claims to fix " +
108
"%d %s." % (self.num_bugs, label))
110
def construct_treeview(self):
111
self.bugs = gtk.ListStore(gobject.TYPE_STRING, gobject.TYPE_STRING)
112
self.treeview = gtk.TreeView(self.bugs)
113
self.treeview.set_headers_visible(False)
115
uri_column = gtk.TreeViewColumn('Bug URI', gtk.CellRendererText(), text=0)
116
self.treeview.append_column(uri_column)
118
self.treeview.connect('row-activated', self.on_row_activated)
120
win = gtk.ScrolledWindow()
121
win.set_policy(gtk.POLICY_AUTOMATIC, gtk.POLICY_AUTOMATIC)
122
win.set_shadow_type(gtk.SHADOW_IN)
123
win.add(self.treeview)
130
self.set_sensitive(False)
131
self.label.set_markup("<b>No bugs fixed</b>\n" +
132
"This revision does not claim to fix any bugs.")
134
def add_bug(self, url, status):
136
self.bugs.append([url, status])
137
self.set_sensitive(True)
139
def get_num_bugs(self):
142
def on_row_activated(self, treeview, path, column):
143
uri = self.bugs.get_value(self.bugs.get_iter(path), 0)
144
_open_link(self, uri)
147
class SignatureTab(gtk.VBox):
149
def __init__(self, repository):
152
self.repository = repository
154
super(SignatureTab, self).__init__(False, 6)
155
signature_box = gtk.Table(rows=3, columns=3)
156
signature_box.set_col_spacing(0, 16)
157
signature_box.set_col_spacing(1, 12)
158
signature_box.set_row_spacings(6)
160
self.signature_image = gtk.Image()
161
signature_box.attach(self.signature_image, 0, 1, 0, 1, gtk.FILL)
163
align = gtk.Alignment(0.0, 0.1)
164
self.signature_label = gtk.Label()
165
align.add(self.signature_label)
166
signature_box.attach(align, 1, 3, 0, 1, gtk.FILL)
168
align = gtk.Alignment(0.0, 0.5)
169
self.signature_key_id_label = gtk.Label()
170
self.signature_key_id_label.set_markup("<b>Key Id:</b>")
171
align.add(self.signature_key_id_label)
172
signature_box.attach(align, 1, 2, 1, 2, gtk.FILL, gtk.FILL)
174
align = gtk.Alignment(0.0, 0.5)
175
self.signature_key_id = gtk.Label()
176
self.signature_key_id.set_selectable(True)
177
align.add(self.signature_key_id)
178
signature_box.attach(align, 2, 3, 1, 2, gtk.EXPAND | gtk.FILL, gtk.FILL)
180
align = gtk.Alignment(0.0, 0.5)
181
self.signature_fingerprint_label = gtk.Label()
182
self.signature_fingerprint_label.set_markup("<b>Fingerprint:</b>")
183
align.add(self.signature_fingerprint_label)
184
signature_box.attach(align, 1, 2, 2, 3, gtk.FILL, gtk.FILL)
186
align = gtk.Alignment(0.0, 0.5)
187
self.signature_fingerprint = gtk.Label()
188
self.signature_fingerprint.set_selectable(True)
189
align.add(self.signature_fingerprint)
190
signature_box.attach(align, 2, 3, 2, 3, gtk.EXPAND | gtk.FILL, gtk.FILL)
192
align = gtk.Alignment(0.0, 0.5)
193
self.signature_trust_label = gtk.Label()
194
self.signature_trust_label.set_markup("<b>Trust:</b>")
195
align.add(self.signature_trust_label)
196
signature_box.attach(align, 1, 2, 3, 4, gtk.FILL, gtk.FILL)
198
align = gtk.Alignment(0.0, 0.5)
199
self.signature_trust = gtk.Label()
200
self.signature_trust.set_selectable(True)
201
align.add(self.signature_trust)
202
signature_box.attach(align, 2, 3, 3, 4, gtk.EXPAND | gtk.FILL, gtk.FILL)
204
self.set_border_width(6)
205
self.pack_start(signature_box, expand=False)
208
def set_revision(self, revision):
209
self.revision = revision
210
revid = revision.revision_id
212
if self.repository.has_signature_for_revision_id(revid):
213
crypttext = self.repository.get_signature_text(revid)
214
self.show_signature(crypttext)
216
self.show_no_signature()
218
def show_no_signature(self):
219
self.signature_key_id_label.hide()
220
self.signature_key_id.set_text("")
222
self.signature_fingerprint_label.hide()
223
self.signature_fingerprint.set_text("")
225
self.signature_trust_label.hide()
226
self.signature_trust.set_text("")
228
self.signature_image.set_from_file(icon_path("sign-unknown.png"))
229
self.signature_label.set_markup("<b>Authenticity unknown</b>\n" +
230
"This revision has not been signed.")
232
def show_signature(self, crypttext):
233
(cleartext, key) = seahorse.verify(crypttext)
235
assert cleartext is not None
237
inv = self.repository.get_inventory(self.revision.revision_id)
238
expected_testament = Testament(self.revision, inv).as_short_text()
239
if expected_testament != cleartext:
240
self.signature_image.set_from_file(icon_path("sign-bad.png"))
241
self.signature_label.set_markup("<b>Signature does not match repository data</b>\n" +
242
"The signature plaintext is different from the expected testament plaintext.")
245
if key and key.is_available():
247
if key.get_display_name() == self.revision.committer:
248
self.signature_image.set_from_file(icon_path("sign-ok.png"))
249
self.signature_label.set_markup("<b>Authenticity confirmed</b>\n" +
250
"This revision has been signed with " +
253
self.signature_image.set_from_file(icon_path("sign-bad.png"))
254
self.signature_label.set_markup("<b>Authenticity cannot be confirmed</b>\n" +
255
"Revision committer is not the same as signer.")
257
self.signature_image.set_from_file(icon_path("sign-bad.png"))
258
self.signature_label.set_markup("<b>Authenticity cannot be confirmed</b>\n" +
259
"This revision has been signed, but the " +
260
"key is not trusted.")
262
self.show_no_signature()
263
self.signature_image.set_from_file(icon_path("sign-bad.png"))
264
self.signature_label.set_markup("<b>Authenticity cannot be confirmed</b>\n" +
265
"Signature key not available.")
268
trust = key.get_trust()
270
if trust <= seahorse.TRUST_NEVER:
271
trust_text = 'never trusted'
272
elif trust == seahorse.TRUST_UNKNOWN:
273
trust_text = 'not trusted'
274
elif trust == seahorse.TRUST_MARGINAL:
275
trust_text = 'marginally trusted'
276
elif trust == seahorse.TRUST_FULL:
277
trust_text = 'fully trusted'
278
elif trust == seahorse.TRUST_ULTIMATE:
279
trust_text = 'ultimately trusted'
281
self.signature_key_id_label.show()
282
self.signature_key_id.set_text(key.get_id())
284
fingerprint = key.get_fingerprint()
285
if fingerprint == "":
286
fingerprint = '<span foreground="dim grey">N/A</span>'
288
self.signature_fingerprint_label.show()
289
self.signature_fingerprint.set_markup(fingerprint)
291
self.signature_trust_label.show()
292
self.signature_trust.set_text('This key is ' + trust_text)
295
class RevisionView(gtk.Notebook):
27
296
""" Custom widget for commit log details.
29
298
A variety of bzr tools may need to implement such a thing. This is a
33
def __init__(self, revision=None, scroll=True, tags=[]):
34
super(LogView, self).__init__()
36
self.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
304
gobject.TYPE_PYOBJECT,
306
'The branch holding the revision being displayed',
307
gobject.PARAM_CONSTRUCT_ONLY | gobject.PARAM_WRITABLE
311
gobject.TYPE_PYOBJECT,
313
'The revision being displayed',
314
gobject.PARAM_READWRITE
318
gobject.TYPE_PYOBJECT,
321
gobject.PARAM_READWRITE
325
gobject.TYPE_PYOBJECT,
328
gobject.PARAM_READWRITE
332
def __init__(self, branch=None, repository=None):
333
gtk.Notebook.__init__(self)
335
self._revision = None
336
self._branch = branch
337
if branch is not None:
338
self._repository = branch.repository
38
self.set_policy(gtk.POLICY_NEVER, gtk.POLICY_NEVER)
39
self.set_shadow_type(gtk.SHADOW_NONE)
340
self._repository = repository
342
self._create_general()
343
self._create_relations()
344
# Disabled because testaments aren't verified yet:
346
self._create_signature()
347
self._create_file_info_view()
350
self.set_current_page(PAGE_GENERAL)
351
self.connect_after('switch-page', self._switch_page_cb)
41
353
self._show_callback = None
42
self._go_callback = None
43
354
self._clicked_callback = None
45
if revision is not None:
46
self.set_revision(revision, tags=tags)
356
self._revision = None
357
self._branch = branch
361
self.set_file_id(None)
363
def do_get_property(self, property):
364
if property.name == 'branch':
366
elif property.name == 'revision':
367
return self._revision
368
elif property.name == 'children':
369
return self._children
370
elif property.name == 'file-id':
373
raise AttributeError, 'unknown property %s' % property.name
375
def do_set_property(self, property, value):
376
if property.name == 'branch':
378
elif property.name == 'revision':
379
self._set_revision(value)
380
elif property.name == 'children':
381
self.set_children(value)
382
elif property.name == 'file-id':
383
self._file_id = value
385
raise AttributeError, 'unknown property %s' % property.name
48
387
def set_show_callback(self, callback):
49
388
self._show_callback = callback
51
def set_go_callback(self, callback):
52
self._go_callback = callback
54
def set_revision(self, revision, tags=[]):
390
def set_file_id(self, file_id):
391
"""Set a specific file id that we want to track.
393
This just effects the display of a per-file commit message.
394
If it is set to None, then all commit messages will be shown.
396
self.set_property('file-id', file_id)
398
def set_revision(self, revision):
399
if revision != self._revision:
400
self.set_property('revision', revision)
402
def get_revision(self):
403
return self.get_property('revision')
405
def _set_revision(self, revision):
406
if revision is None: return
408
self.avatarsbox.reset()
55
410
self._revision = revision
56
self.revision_id.set_text(revision.revision_id)
57
411
if revision.committer is not None:
58
412
self.committer.set_text(revision.committer)
413
self.avatarsbox.add(revision.committer, "committer")
60
415
self.committer.set_text("")
416
self.avatarsbox.hide()
61
417
author = revision.properties.get('author', '')
418
self.avatarsbox.merge(revision.get_apparent_authors(), "author")
63
420
self.author.set_text(author)
64
421
self.author.show()
134
549
button = gtk.Button()
135
550
button.add(image)
136
551
button.connect("clicked", self._show_clicked_cb,
137
self._revision.revision_id, parent_id)
552
self._revision.revision_id, revid)
138
553
hbox.pack_start(button, expand=False, fill=True)
141
if self._go_callback is not None:
142
button = gtk.Button(parent_id)
143
button.connect("clicked", self._go_clicked_cb, parent_id)
145
button = gtk.Label(parent_id)
556
button = gtk.Button()
557
revid_label = gtk.Label(str(revid))
558
revid_label.set_ellipsize(pango.ELLIPSIZE_MIDDLE)
559
revid_label.set_alignment(0.0, 0.5)
560
button.add(revid_label)
561
button.connect("clicked",
562
lambda w, r: self.set_revision(self._repository.get_revision(r)), revid)
146
563
button.set_use_underline(False)
147
hbox.pack_start(button, expand=False, fill=True)
564
hbox.pack_start(button, expand=True, fill=True)
567
def _create_general(self):
151
568
vbox = gtk.VBox(False, 6)
152
569
vbox.set_border_width(6)
153
570
vbox.pack_start(self._create_headers(), expand=False, fill=True)
154
vbox.pack_start(self._create_parents_table(), expand=False, fill=True)
155
571
vbox.pack_start(self._create_message_view())
156
self.add_with_viewport(vbox)
572
self.append_page(vbox, tab_label=gtk.Label("General"))
575
def _create_relations(self):
576
vbox = gtk.VBox(False, 6)
577
vbox.set_border_width(6)
578
vbox.pack_start(self._create_parents(), expand=False, fill=True)
579
vbox.pack_start(self._create_children(), expand=False, fill=True)
580
self.append_page(vbox, tab_label=gtk.Label("Relations"))
583
def _create_signature(self):
584
self.signature_table = SignatureTab(self._repository)
585
self.append_page(self.signature_table, tab_label=gtk.Label('Signature'))
586
self.connect_after('notify::revision', self._update_signature)
159
588
def _create_headers(self):
589
self.avatarsbox = AvatarsBox()
160
591
self.table = gtk.Table(rows=5, columns=2)
161
592
self.table.set_row_spacings(6)
162
593
self.table.set_col_spacings(6)
163
594
self.table.show()
165
align = gtk.Alignment(1.0, 0.5)
596
self.avatarsbox.pack_start(self.table)
597
self.avatarsbox.show()
166
601
label = gtk.Label()
602
label.set_alignment(1.0, 0.5)
167
603
label.set_markup("<b>Revision Id:</b>")
169
self.table.attach(align, 0, 1, 0, 1, gtk.FILL, gtk.FILL)
604
self.table.attach(label, 0, 1, row, row+1, gtk.FILL, gtk.FILL)
173
align = gtk.Alignment(0.0, 0.5)
174
self.revision_id = gtk.Label()
175
self.revision_id.set_selectable(True)
176
align.add(self.revision_id)
177
self.table.attach(align, 1, 2, 0, 1, gtk.EXPAND | gtk.FILL, gtk.FILL)
179
self.revision_id.show()
607
revision_id = gtk.Label()
608
revision_id.set_ellipsize(pango.ELLIPSIZE_MIDDLE)
609
revision_id.set_alignment(0.0, 0.5)
610
revision_id.set_selectable(True)
611
self.connect('notify::revision',
612
lambda w, p: revision_id.set_text(self._revision.revision_id))
613
self.table.attach(revision_id, 1, 2, row, row+1, gtk.EXPAND | gtk.FILL, gtk.FILL)
181
align = gtk.Alignment(1.0, 0.5)
182
617
self.author_label = gtk.Label()
618
self.author_label.set_alignment(1.0, 0.5)
183
619
self.author_label.set_markup("<b>Author:</b>")
184
align.add(self.author_label)
185
self.table.attach(align, 0, 1, 1, 2, gtk.FILL, gtk.FILL)
620
self.table.attach(self.author_label, 0, 1, row, row+1, gtk.FILL, gtk.FILL)
187
621
self.author_label.show()
189
align = gtk.Alignment(0.0, 0.5)
190
623
self.author = gtk.Label()
624
self.author.set_ellipsize(pango.ELLIPSIZE_END)
625
self.author.set_alignment(0.0, 0.5)
191
626
self.author.set_selectable(True)
192
align.add(self.author)
193
self.table.attach(align, 1, 2, 1, 2, gtk.EXPAND | gtk.FILL, gtk.FILL)
627
self.table.attach(self.author, 1, 2, row, row+1, gtk.EXPAND | gtk.FILL, gtk.FILL)
195
628
self.author.show()
196
629
self.author.hide()
198
align = gtk.Alignment(1.0, 0.5)
199
632
label = gtk.Label()
633
label.set_alignment(1.0, 0.5)
200
634
label.set_markup("<b>Committer:</b>")
202
self.table.attach(align, 0, 1, 2, 3, gtk.FILL, gtk.FILL)
635
self.table.attach(label, 0, 1, row, row+1, gtk.FILL, gtk.FILL)
206
align = gtk.Alignment(0.0, 0.5)
207
638
self.committer = gtk.Label()
639
self.committer.set_ellipsize(pango.ELLIPSIZE_END)
640
self.committer.set_alignment(0.0, 0.5)
208
641
self.committer.set_selectable(True)
209
align.add(self.committer)
210
self.table.attach(align, 1, 2, 2, 3, gtk.EXPAND | gtk.FILL, gtk.FILL)
642
self.table.attach(self.committer, 1, 2, row, row+1, gtk.EXPAND | gtk.FILL, gtk.FILL)
212
643
self.committer.show()
214
align = gtk.Alignment(0.0, 0.5)
216
label.set_markup("<b>Branch nick:</b>")
218
self.table.attach(align, 0, 1, 3, 4, gtk.FILL, gtk.FILL)
222
align = gtk.Alignment(0.0, 0.5)
223
646
self.branchnick_label = gtk.Label()
224
self.branchnick_label.set_selectable(True)
225
align.add(self.branchnick_label)
226
self.table.attach(align, 1, 2, 3, 4, gtk.EXPAND | gtk.FILL, gtk.FILL)
647
self.branchnick_label.set_alignment(1.0, 0.5)
648
self.branchnick_label.set_markup("<b>Branch nick:</b>")
649
self.table.attach(self.branchnick_label, 0, 1, row, row+1, gtk.FILL, gtk.FILL)
227
650
self.branchnick_label.show()
230
align = gtk.Alignment(1.0, 0.5)
652
self.branchnick = gtk.Label()
653
self.branchnick.set_ellipsize(pango.ELLIPSIZE_MIDDLE)
654
self.branchnick.set_alignment(0.0, 0.5)
655
self.branchnick.set_selectable(True)
656
self.table.attach(self.branchnick, 1, 2, row, row+1, gtk.EXPAND | gtk.FILL, gtk.FILL)
657
self.branchnick.show()
231
660
label = gtk.Label()
661
label.set_alignment(1.0, 0.5)
232
662
label.set_markup("<b>Timestamp:</b>")
234
self.table.attach(align, 0, 1, 4, 5, gtk.FILL, gtk.FILL)
663
self.table.attach(label, 0, 1, row, row+1, gtk.FILL, gtk.FILL)
238
align = gtk.Alignment(0.0, 0.5)
239
666
self.timestamp = gtk.Label()
667
self.timestamp.set_ellipsize(pango.ELLIPSIZE_END)
668
self.timestamp.set_alignment(0.0, 0.5)
240
669
self.timestamp.set_selectable(True)
241
align.add(self.timestamp)
242
self.table.attach(align, 1, 2, 4, 5, gtk.EXPAND | gtk.FILL, gtk.FILL)
670
self.table.attach(self.timestamp, 1, 2, row, row+1, gtk.EXPAND | gtk.FILL, gtk.FILL)
244
671
self.timestamp.show()
246
align = gtk.Alignment(1.0, 0.5)
247
674
self.tags_label = gtk.Label()
675
self.tags_label.set_alignment(1.0, 0.5)
248
676
self.tags_label.set_markup("<b>Tags:</b>")
249
align.add(self.tags_label)
251
self.table.attach(align, 0, 1, 5, 6, gtk.FILL, gtk.FILL)
677
self.table.attach(self.tags_label, 0, 1, row, row+1, gtk.FILL, gtk.FILL)
252
678
self.tags_label.show()
254
align = gtk.Alignment(0.0, 0.5)
255
self.tags_list = gtk.VBox()
256
align.add(self.tags_list)
257
self.table.attach(align, 1, 2, 5, 6, gtk.EXPAND | gtk.FILL, gtk.FILL)
680
self.tags_list = gtk.Label()
681
self.tags_list.set_ellipsize(pango.ELLIPSIZE_MIDDLE)
682
self.tags_list.set_alignment(0.0, 0.5)
683
self.table.attach(self.tags_list, 1, 2, row, row+1, gtk.EXPAND | gtk.FILL, gtk.FILL)
259
684
self.tags_list.show()
260
self.tags_widgets = []
264
def _create_parents_table(self):
265
self.parents_table = gtk.Table(rows=1, columns=2)
266
self.parents_table.set_row_spacings(3)
267
self.parents_table.set_col_spacings(6)
268
self.parents_table.show()
686
self.connect('notify::revision', self._add_tags)
688
self.avatarsbox.show()
689
return self.avatarsbox
691
def _create_parents(self):
692
hbox = gtk.HBox(True, 3)
694
self.parents_table = self._create_parents_or_children_table(
269
696
self.parents_widgets = []
697
hbox.pack_start(self.parents_table)
702
def _create_children(self):
703
hbox = gtk.HBox(True, 3)
704
self.children_table = self._create_parents_or_children_table(
706
self.children_widgets = []
707
hbox.pack_start(self.children_table)
711
def _create_parents_or_children_table(self, text):
712
table = gtk.Table(rows=1, columns=2)
713
table.set_row_spacings(3)
714
table.set_col_spacings(6)
271
717
label = gtk.Label()
272
label.set_markup("<b>Parents:</b>")
718
label.set_markup(text)
273
719
align = gtk.Alignment(0.0, 0.5)
275
self.parents_table.attach(align, 0, 1, 0, 1, gtk.FILL, gtk.FILL)
721
table.attach(align, 0, 1, 0, 1, gtk.FILL, gtk.FILL)
279
return self.parents_table
281
727
def _create_message_view(self):
282
self.message_buffer = gtk.TextBuffer()
283
tv = gtk.TextView(self.message_buffer)
284
tv.set_editable(False)
285
tv.set_wrap_mode(gtk.WRAP_WORD)
286
tv.modify_font(pango.FontDescription("Monospace"))
728
msg_buffer = gtk.TextBuffer()
729
self.connect('notify::revision',
730
lambda w, p: msg_buffer.set_text(self._revision.message))
731
window = gtk.ScrolledWindow()
732
window.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
733
window.set_shadow_type(gtk.SHADOW_IN)
734
tv = gtk.TextView(msg_buffer)
735
tv.set_editable(False)
736
tv.set_wrap_mode(gtk.WRAP_WORD)
738
tv.modify_font(pango.FontDescription("Monospace"))
744
def _create_bugs(self):
745
self.bugs_page = BugsTab()
746
self.connect_after('notify::revision', self._update_bugs)
747
self.append_page(self.bugs_page, tab_label=gtk.Label('Bugs'))
749
def _create_file_info_view(self):
750
self.file_info_box = gtk.VBox(False, 6)
751
self.file_info_box.set_border_width(6)
752
self.file_info_buffer = gtk.TextBuffer()
753
window = gtk.ScrolledWindow()
754
window.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
755
window.set_shadow_type(gtk.SHADOW_IN)
756
tv = gtk.TextView(self.file_info_buffer)
757
tv.set_editable(False)
758
tv.set_wrap_mode(gtk.WRAP_WORD)
759
tv.modify_font(pango.FontDescription("Monospace"))
763
self.file_info_box.pack_start(window)
764
self.file_info_box.hide() # Only shown when there are per-file messages
765
self.append_page(self.file_info_box, tab_label=gtk.Label('Per-file'))