19
19
pygtk.require("2.0")
25
23
from bzrlib.osutils import format_date
26
from bzrlib.util.bencode import bdecode
27
from bzrlib.testament import Testament
29
from bzrlib.plugins.gtk import icon_path
32
from bzrlib.plugins.gtk import seahorse
44
def _open_link(widget, uri):
45
for cmd in ['sensible-browser', 'xdg-open']:
46
if webbrowser._iscommand(cmd):
47
webbrowser._tryorder.insert(0, '%s "%%s"' % cmd)
50
gtk.link_button_set_uri_hook(_open_link)
52
class BugsTab(gtk.VBox):
55
super(BugsTab, self).__init__(False, 6)
57
table = gtk.Table(rows=2, columns=2)
59
table.set_row_spacings(6)
60
table.set_col_spacing(0, 16)
63
image.set_from_file(icon_path("bug.png"))
64
table.attach(image, 0, 1, 0, 1, gtk.FILL)
66
align = gtk.Alignment(0.0, 0.1)
67
self.label = gtk.Label()
69
table.attach(align, 1, 2, 0, 1, gtk.FILL)
71
treeview = self.construct_treeview()
72
table.attach(treeview, 1, 2, 1, 2, gtk.FILL | gtk.EXPAND)
74
self.set_border_width(6)
75
self.pack_start(table, expand=False)
80
def set_revision(self, revision):
85
bugs_text = revision.properties.get('bugs', '')
86
for bugline in bugs_text.splitlines():
87
(url, status) = bugline.split(" ")
89
self.add_bug(url, status)
91
if self.num_bugs == 0:
93
elif self.num_bugs == 1:
98
self.label.set_markup("<b>Bugs fixed</b>\n" +
99
"This revision claims to fix " +
100
"%d %s." % (self.num_bugs, label))
102
def construct_treeview(self):
103
self.bugs = gtk.ListStore(gobject.TYPE_STRING, gobject.TYPE_STRING)
104
self.treeview = gtk.TreeView(self.bugs)
105
self.treeview.set_headers_visible(False)
107
uri_column = gtk.TreeViewColumn('Bug URI', gtk.CellRendererText(), text=0)
108
self.treeview.append_column(uri_column)
110
self.treeview.connect('row-activated', self.on_row_activated)
112
win = gtk.ScrolledWindow()
113
win.set_policy(gtk.POLICY_AUTOMATIC, gtk.POLICY_AUTOMATIC)
114
win.set_shadow_type(gtk.SHADOW_IN)
115
win.add(self.treeview)
122
self.set_sensitive(False)
123
self.label.set_markup("<b>No bugs fixed</b>\n" +
124
"This revision does not claim to fix any bugs.")
126
def add_bug(self, url, status):
128
self.bugs.append([url, status])
129
self.set_sensitive(True)
131
def get_num_bugs(self):
134
def on_row_activated(self, treeview, path, column):
135
uri = self.bugs.get_value(self.bugs.get_iter(path), 0)
136
_open_link(self, uri)
139
class SignatureTab(gtk.VBox):
141
def __init__(self, repository):
144
self.repository = repository
146
super(SignatureTab, self).__init__(False, 6)
147
signature_box = gtk.Table(rows=3, columns=3)
148
signature_box.set_col_spacing(0, 16)
149
signature_box.set_col_spacing(1, 12)
150
signature_box.set_row_spacings(6)
152
self.signature_image = gtk.Image()
153
signature_box.attach(self.signature_image, 0, 1, 0, 1, gtk.FILL)
155
align = gtk.Alignment(0.0, 0.1)
156
self.signature_label = gtk.Label()
157
align.add(self.signature_label)
158
signature_box.attach(align, 1, 3, 0, 1, gtk.FILL)
160
align = gtk.Alignment(0.0, 0.5)
161
self.signature_key_id_label = gtk.Label()
162
self.signature_key_id_label.set_markup("<b>Key Id:</b>")
163
align.add(self.signature_key_id_label)
164
signature_box.attach(align, 1, 2, 1, 2, gtk.FILL, gtk.FILL)
166
align = gtk.Alignment(0.0, 0.5)
167
self.signature_key_id = gtk.Label()
168
self.signature_key_id.set_selectable(True)
169
align.add(self.signature_key_id)
170
signature_box.attach(align, 2, 3, 1, 2, gtk.EXPAND | gtk.FILL, gtk.FILL)
172
align = gtk.Alignment(0.0, 0.5)
173
self.signature_fingerprint_label = gtk.Label()
174
self.signature_fingerprint_label.set_markup("<b>Fingerprint:</b>")
175
align.add(self.signature_fingerprint_label)
176
signature_box.attach(align, 1, 2, 2, 3, gtk.FILL, gtk.FILL)
178
align = gtk.Alignment(0.0, 0.5)
179
self.signature_fingerprint = gtk.Label()
180
self.signature_fingerprint.set_selectable(True)
181
align.add(self.signature_fingerprint)
182
signature_box.attach(align, 2, 3, 2, 3, gtk.EXPAND | gtk.FILL, gtk.FILL)
184
align = gtk.Alignment(0.0, 0.5)
185
self.signature_trust_label = gtk.Label()
186
self.signature_trust_label.set_markup("<b>Trust:</b>")
187
align.add(self.signature_trust_label)
188
signature_box.attach(align, 1, 2, 3, 4, gtk.FILL, gtk.FILL)
190
align = gtk.Alignment(0.0, 0.5)
191
self.signature_trust = gtk.Label()
192
self.signature_trust.set_selectable(True)
193
align.add(self.signature_trust)
194
signature_box.attach(align, 2, 3, 3, 4, gtk.EXPAND | gtk.FILL, gtk.FILL)
196
self.set_border_width(6)
197
self.pack_start(signature_box, expand=False)
200
def set_revision(self, revision):
201
self.revision = revision
202
revid = revision.revision_id
204
if self.repository.has_signature_for_revision_id(revid):
205
crypttext = self.repository.get_signature_text(revid)
206
self.show_signature(crypttext)
208
self.show_no_signature()
210
def show_no_signature(self):
211
self.signature_key_id_label.hide()
212
self.signature_key_id.set_text("")
214
self.signature_fingerprint_label.hide()
215
self.signature_fingerprint.set_text("")
217
self.signature_trust_label.hide()
218
self.signature_trust.set_text("")
220
self.signature_image.set_from_file(icon_path("sign-unknown.png"))
221
self.signature_label.set_markup("<b>Authenticity unknown</b>\n" +
222
"This revision has not been signed.")
224
def show_signature(self, crypttext):
225
(cleartext, key) = seahorse.verify(crypttext)
227
assert cleartext is not None
229
inv = self.repository.get_inventory(self.revision.revision_id)
230
expected_testament = Testament(self.revision, inv).as_short_text()
231
if expected_testament != cleartext:
232
self.signature_image.set_from_file(icon_path("sign-bad.png"))
233
self.signature_label.set_markup("<b>Signature does not match repository data</b>\n" +
234
"The signature plaintext is different from the expected testament plaintext.")
237
if key and key.is_available():
239
if key.get_display_name() == self.revision.committer:
240
self.signature_image.set_from_file(icon_path("sign-ok.png"))
241
self.signature_label.set_markup("<b>Authenticity confirmed</b>\n" +
242
"This revision has been signed with " +
245
self.signature_image.set_from_file(icon_path("sign-bad.png"))
246
self.signature_label.set_markup("<b>Authenticity cannot be confirmed</b>\n" +
247
"Revision committer is not the same as signer.")
249
self.signature_image.set_from_file(icon_path("sign-bad.png"))
250
self.signature_label.set_markup("<b>Authenticity cannot be confirmed</b>\n" +
251
"This revision has been signed, but the " +
252
"key is not trusted.")
254
self.show_no_signature()
255
self.signature_image.set_from_file(icon_path("sign-bad.png"))
256
self.signature_label.set_markup("<b>Authenticity cannot be confirmed</b>\n" +
257
"Signature key not available.")
260
trust = key.get_trust()
262
if trust <= seahorse.TRUST_NEVER:
263
trust_text = 'never trusted'
264
elif trust == seahorse.TRUST_UNKNOWN:
265
trust_text = 'not trusted'
266
elif trust == seahorse.TRUST_MARGINAL:
267
trust_text = 'marginally trusted'
268
elif trust == seahorse.TRUST_FULL:
269
trust_text = 'fully trusted'
270
elif trust == seahorse.TRUST_ULTIMATE:
271
trust_text = 'ultimately trusted'
273
self.signature_key_id_label.show()
274
self.signature_key_id.set_text(key.get_id())
276
fingerprint = key.get_fingerprint()
277
if fingerprint == "":
278
fingerprint = '<span foreground="dim grey">N/A</span>'
280
self.signature_fingerprint_label.show()
281
self.signature_fingerprint.set_markup(fingerprint)
283
self.signature_trust_label.show()
284
self.signature_trust.set_text('This key is ' + trust_text)
287
class RevisionView(gtk.Notebook):
26
class LogView(gtk.ScrolledWindow):
288
27
""" Custom widget for commit log details.
290
29
A variety of bzr tools may need to implement such a thing. This is a
296
gobject.TYPE_PYOBJECT,
298
'The branch holding the revision being displayed',
299
gobject.PARAM_CONSTRUCT_ONLY | gobject.PARAM_WRITABLE
303
gobject.TYPE_PYOBJECT,
305
'The revision being displayed',
306
gobject.PARAM_READWRITE
310
gobject.TYPE_PYOBJECT,
313
gobject.PARAM_READWRITE
317
gobject.TYPE_PYOBJECT,
320
gobject.PARAM_READWRITE
324
def __init__(self, branch=None, repository=None):
325
gtk.Notebook.__init__(self)
327
self._revision = None
328
self._branch = branch
329
if branch is not None:
330
self._repository = branch.repository
33
def __init__(self, revision=None, scroll=True, tags=[]):
34
super(LogView, self).__init__()
36
self.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
332
self._repository = repository
334
self._create_general()
335
self._create_relations()
336
# Disabled because testaments aren't verified yet:
338
self._create_signature()
339
self._create_file_info_view()
342
self.set_current_page(PAGE_GENERAL)
343
self.connect_after('switch-page', self._switch_page_cb)
38
self.set_policy(gtk.POLICY_NEVER, gtk.POLICY_NEVER)
39
self.set_shadow_type(gtk.SHADOW_NONE)
345
41
self._show_callback = None
42
self._go_callback = None
346
43
self._clicked_callback = None
348
self._revision = None
349
self._branch = branch
353
self.set_file_id(None)
355
def do_get_property(self, property):
356
if property.name == 'branch':
358
elif property.name == 'revision':
359
return self._revision
360
elif property.name == 'children':
361
return self._children
362
elif property.name == 'file-id':
365
raise AttributeError, 'unknown property %s' % property.name
367
def do_set_property(self, property, value):
368
if property.name == 'branch':
370
elif property.name == 'revision':
371
self._set_revision(value)
372
elif property.name == 'children':
373
self.set_children(value)
374
elif property.name == 'file-id':
375
self._file_id = value
377
raise AttributeError, 'unknown property %s' % property.name
45
if revision is not None:
46
self.set_revision(revision, tags=tags)
379
48
def set_show_callback(self, callback):
380
49
self._show_callback = callback
382
def set_file_id(self, file_id):
383
"""Set a specific file id that we want to track.
385
This just effects the display of a per-file commit message.
386
If it is set to None, then all commit messages will be shown.
388
self.set_property('file-id', file_id)
390
def set_revision(self, revision):
391
if revision != self._revision:
392
self.set_property('revision', revision)
394
def get_revision(self):
395
return self.get_property('revision')
397
def _set_revision(self, revision):
398
if revision is None: return
51
def set_go_callback(self, callback):
52
self._go_callback = callback
54
def set_revision(self, revision, tags=[]):
400
55
self._revision = revision
56
self.revision_id.set_text(revision.revision_id)
401
57
if revision.committer is not None:
402
58
self.committer.set_text(revision.committer)
570
162
self.table.set_col_spacings(6)
571
163
self.table.show()
165
align = gtk.Alignment(1.0, 0.5)
575
166
label = gtk.Label()
576
label.set_alignment(1.0, 0.5)
577
167
label.set_markup("<b>Revision Id:</b>")
578
self.table.attach(label, 0, 1, row, row+1, gtk.FILL, gtk.FILL)
169
self.table.attach(align, 0, 1, 0, 1, gtk.FILL, gtk.FILL)
581
revision_id = gtk.Label()
582
revision_id.set_ellipsize(pango.ELLIPSIZE_MIDDLE)
583
revision_id.set_alignment(0.0, 0.5)
584
revision_id.set_selectable(True)
585
self.connect('notify::revision',
586
lambda w, p: revision_id.set_text(self._revision.revision_id))
587
self.table.attach(revision_id, 1, 2, row, row+1, gtk.EXPAND | 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()
181
align = gtk.Alignment(1.0, 0.5)
591
182
self.author_label = gtk.Label()
592
self.author_label.set_alignment(1.0, 0.5)
593
183
self.author_label.set_markup("<b>Author:</b>")
594
self.table.attach(self.author_label, 0, 1, row, row+1, gtk.FILL, gtk.FILL)
184
align.add(self.author_label)
185
self.table.attach(align, 0, 1, 1, 2, gtk.FILL, gtk.FILL)
595
187
self.author_label.show()
189
align = gtk.Alignment(0.0, 0.5)
597
190
self.author = gtk.Label()
598
self.author.set_ellipsize(pango.ELLIPSIZE_END)
599
self.author.set_alignment(0.0, 0.5)
600
191
self.author.set_selectable(True)
601
self.table.attach(self.author, 1, 2, row, row+1, gtk.EXPAND | gtk.FILL, gtk.FILL)
192
align.add(self.author)
193
self.table.attach(align, 1, 2, 1, 2, gtk.EXPAND | gtk.FILL, gtk.FILL)
602
195
self.author.show()
603
196
self.author.hide()
198
align = gtk.Alignment(1.0, 0.5)
606
199
label = gtk.Label()
607
label.set_alignment(1.0, 0.5)
608
200
label.set_markup("<b>Committer:</b>")
609
self.table.attach(label, 0, 1, row, row+1, gtk.FILL, gtk.FILL)
202
self.table.attach(align, 0, 1, 2, 3, gtk.FILL, gtk.FILL)
206
align = gtk.Alignment(0.0, 0.5)
612
207
self.committer = gtk.Label()
613
self.committer.set_ellipsize(pango.ELLIPSIZE_END)
614
self.committer.set_alignment(0.0, 0.5)
615
208
self.committer.set_selectable(True)
616
self.table.attach(self.committer, 1, 2, row, row+1, gtk.EXPAND | gtk.FILL, gtk.FILL)
209
align.add(self.committer)
210
self.table.attach(align, 1, 2, 2, 3, gtk.EXPAND | gtk.FILL, gtk.FILL)
617
212
self.committer.show()
214
align = gtk.Alignment(0.0, 0.5)
620
215
label = gtk.Label()
621
label.set_alignment(1.0, 0.5)
622
216
label.set_markup("<b>Branch nick:</b>")
623
self.table.attach(label, 0, 1, row, row+1, gtk.FILL, gtk.FILL)
218
self.table.attach(align, 0, 1, 3, 4, gtk.FILL, gtk.FILL)
222
align = gtk.Alignment(0.0, 0.5)
626
223
self.branchnick_label = gtk.Label()
627
self.branchnick_label.set_ellipsize(pango.ELLIPSIZE_MIDDLE)
628
self.branchnick_label.set_alignment(0.0, 0.5)
629
224
self.branchnick_label.set_selectable(True)
630
self.table.attach(self.branchnick_label, 1, 2, row, row+1, gtk.EXPAND | gtk.FILL, gtk.FILL)
225
align.add(self.branchnick_label)
226
self.table.attach(align, 1, 2, 3, 4, gtk.EXPAND | gtk.FILL, gtk.FILL)
631
227
self.branchnick_label.show()
230
align = gtk.Alignment(1.0, 0.5)
634
231
label = gtk.Label()
635
label.set_alignment(1.0, 0.5)
636
232
label.set_markup("<b>Timestamp:</b>")
637
self.table.attach(label, 0, 1, row, row+1, gtk.FILL, gtk.FILL)
234
self.table.attach(align, 0, 1, 4, 5, gtk.FILL, gtk.FILL)
238
align = gtk.Alignment(0.0, 0.5)
640
239
self.timestamp = gtk.Label()
641
self.timestamp.set_ellipsize(pango.ELLIPSIZE_END)
642
self.timestamp.set_alignment(0.0, 0.5)
643
240
self.timestamp.set_selectable(True)
644
self.table.attach(self.timestamp, 1, 2, row, row+1, gtk.EXPAND | gtk.FILL, gtk.FILL)
241
align.add(self.timestamp)
242
self.table.attach(align, 1, 2, 4, 5, gtk.EXPAND | gtk.FILL, gtk.FILL)
645
244
self.timestamp.show()
246
align = gtk.Alignment(1.0, 0.5)
648
247
self.tags_label = gtk.Label()
649
self.tags_label.set_alignment(1.0, 0.5)
650
248
self.tags_label.set_markup("<b>Tags:</b>")
651
self.table.attach(self.tags_label, 0, 1, row, row+1, gtk.FILL, gtk.FILL)
249
align.add(self.tags_label)
251
self.table.attach(align, 0, 1, 5, 6, gtk.FILL, gtk.FILL)
652
252
self.tags_label.show()
654
self.tags_list = gtk.Label()
655
self.tags_list.set_ellipsize(pango.ELLIPSIZE_MIDDLE)
656
self.tags_list.set_alignment(0.0, 0.5)
657
self.table.attach(self.tags_list, 1, 2, row, row+1, gtk.EXPAND | gtk.FILL, gtk.FILL)
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)
658
259
self.tags_list.show()
660
self.connect('notify::revision', self._add_tags)
260
self.tags_widgets = []
662
262
return self.table
664
def _create_parents(self):
665
hbox = gtk.HBox(True, 3)
667
self.parents_table = self._create_parents_or_children_table(
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()
669
269
self.parents_widgets = []
670
hbox.pack_start(self.parents_table)
675
def _create_children(self):
676
hbox = gtk.HBox(True, 3)
677
self.children_table = self._create_parents_or_children_table(
679
self.children_widgets = []
680
hbox.pack_start(self.children_table)
684
def _create_parents_or_children_table(self, text):
685
table = gtk.Table(rows=1, columns=2)
686
table.set_row_spacings(3)
687
table.set_col_spacings(6)
690
271
label = gtk.Label()
691
label.set_markup(text)
272
label.set_markup("<b>Parents:</b>")
692
273
align = gtk.Alignment(0.0, 0.5)
694
table.attach(align, 0, 1, 0, 1, gtk.FILL, gtk.FILL)
275
self.parents_table.attach(align, 0, 1, 0, 1, gtk.FILL, gtk.FILL)
279
return self.parents_table
700
281
def _create_message_view(self):
701
msg_buffer = gtk.TextBuffer()
702
self.connect('notify::revision',
703
lambda w, p: msg_buffer.set_text(self._revision.message))
704
window = gtk.ScrolledWindow()
705
window.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
706
window.set_shadow_type(gtk.SHADOW_IN)
707
tv = gtk.TextView(msg_buffer)
708
tv.set_editable(False)
709
tv.set_wrap_mode(gtk.WRAP_WORD)
711
tv.modify_font(pango.FontDescription("Monospace"))
717
def _create_bugs(self):
718
self.bugs_page = BugsTab()
719
self.connect_after('notify::revision', self._update_bugs)
720
self.append_page(self.bugs_page, tab_label=gtk.Label('Bugs'))
722
def _create_file_info_view(self):
723
self.file_info_box = gtk.VBox(False, 6)
724
self.file_info_box.set_border_width(6)
725
self.file_info_buffer = gtk.TextBuffer()
726
window = gtk.ScrolledWindow()
727
window.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
728
window.set_shadow_type(gtk.SHADOW_IN)
729
tv = gtk.TextView(self.file_info_buffer)
730
tv.set_editable(False)
731
tv.set_wrap_mode(gtk.WRAP_WORD)
732
tv.modify_font(pango.FontDescription("Monospace"))
736
self.file_info_box.pack_start(window)
737
self.file_info_box.hide() # Only shown when there are per-file messages
738
self.append_page(self.file_info_box, tab_label=gtk.Label('Per-file'))
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"))