19
19
pygtk.require("2.0")
25
from bzrlib.plugins.gtk import icon_path
23
26
from bzrlib.osutils import format_date
26
class LogView(gtk.ScrolledWindow):
27
from bzrlib.util.bencode import bdecode
30
from bzrlib.plugins.gtk import seahorse
41
def _open_link(widget, uri):
42
subprocess.Popen(['sensible-browser', uri], close_fds=True)
44
gtk.link_button_set_uri_hook(_open_link)
46
class BugsTab(gtk.VBox):
49
super(BugsTab, self).__init__(False, 6)
51
table = gtk.Table(rows=2, columns=2)
53
table.set_row_spacings(6)
54
table.set_col_spacing(0, 16)
57
image.set_from_file(icon_path("bug.png"))
58
table.attach(image, 0, 1, 0, 1, gtk.FILL)
60
align = gtk.Alignment(0.0, 0.1)
61
self.label = gtk.Label()
63
table.attach(align, 1, 2, 0, 1, gtk.FILL)
65
treeview = self.construct_treeview()
66
table.attach(treeview, 1, 2, 1, 2, gtk.FILL | gtk.EXPAND)
68
self.set_border_width(6)
69
self.pack_start(table, expand=False)
74
def set_revision(self, revision):
79
bugs_text = revision.properties.get('bugs', '')
80
for bugline in bugs_text.splitlines():
81
(url, status) = bugline.split(" ")
83
self.add_bug(url, status)
85
if self.num_bugs == 0:
87
elif self.num_bugs == 1:
92
self.label.set_markup("<b>Bugs fixed</b>\n" +
93
"This revision claims to fix " +
94
"%d %s." % (self.num_bugs, label))
96
def construct_treeview(self):
97
self.bugs = gtk.ListStore(gobject.TYPE_STRING, gobject.TYPE_STRING)
98
self.treeview = gtk.TreeView(self.bugs)
99
self.treeview.set_headers_visible(False)
101
uri_column = gtk.TreeViewColumn('Bug URI', gtk.CellRendererText(), text=0)
102
self.treeview.append_column(uri_column)
104
self.treeview.connect('row-activated', self.on_row_activated)
106
win = gtk.ScrolledWindow()
107
win.set_policy(gtk.POLICY_AUTOMATIC, gtk.POLICY_AUTOMATIC)
108
win.set_shadow_type(gtk.SHADOW_IN)
109
win.add(self.treeview)
116
self.set_sensitive(False)
117
self.label.set_markup("<b>No bugs fixed</b>\n" +
118
"This revision does not claim to fix any bugs.")
120
def add_bug(self, url, status):
122
self.bugs.append([url, status])
123
self.set_sensitive(True)
125
def get_num_bugs(self):
128
def on_row_activated(self, treeview, path, column):
129
uri = self.bugs.get_value(self.bugs.get_iter(path), 0)
130
_open_link(self, uri)
133
class SignatureTab(gtk.VBox):
135
def __init__(self, repository):
138
self.repository = repository
140
super(SignatureTab, self).__init__(False, 6)
141
signature_box = gtk.Table(rows=3, columns=3)
142
signature_box.set_col_spacing(0, 16)
143
signature_box.set_col_spacing(1, 12)
144
signature_box.set_row_spacings(6)
146
self.signature_image = gtk.Image()
147
signature_box.attach(self.signature_image, 0, 1, 0, 1, gtk.FILL)
149
align = gtk.Alignment(0.0, 0.1)
150
self.signature_label = gtk.Label()
151
align.add(self.signature_label)
152
signature_box.attach(align, 1, 3, 0, 1, gtk.FILL)
154
align = gtk.Alignment(0.0, 0.5)
155
self.signature_key_id_label = gtk.Label()
156
self.signature_key_id_label.set_markup("<b>Key Id:</b>")
157
align.add(self.signature_key_id_label)
158
signature_box.attach(align, 1, 2, 1, 2, gtk.FILL, gtk.FILL)
160
align = gtk.Alignment(0.0, 0.5)
161
self.signature_key_id = gtk.Label()
162
self.signature_key_id.set_selectable(True)
163
align.add(self.signature_key_id)
164
signature_box.attach(align, 2, 3, 1, 2, gtk.EXPAND | gtk.FILL, gtk.FILL)
166
align = gtk.Alignment(0.0, 0.5)
167
self.signature_fingerprint_label = gtk.Label()
168
self.signature_fingerprint_label.set_markup("<b>Fingerprint:</b>")
169
align.add(self.signature_fingerprint_label)
170
signature_box.attach(align, 1, 2, 2, 3, gtk.FILL, gtk.FILL)
172
align = gtk.Alignment(0.0, 0.5)
173
self.signature_fingerprint = gtk.Label()
174
self.signature_fingerprint.set_selectable(True)
175
align.add(self.signature_fingerprint)
176
signature_box.attach(align, 2, 3, 2, 3, gtk.EXPAND | gtk.FILL, gtk.FILL)
178
align = gtk.Alignment(0.0, 0.5)
179
self.signature_trust_label = gtk.Label()
180
self.signature_trust_label.set_markup("<b>Trust:</b>")
181
align.add(self.signature_trust_label)
182
signature_box.attach(align, 1, 2, 3, 4, gtk.FILL, gtk.FILL)
184
align = gtk.Alignment(0.0, 0.5)
185
self.signature_trust = gtk.Label()
186
self.signature_trust.set_selectable(True)
187
align.add(self.signature_trust)
188
signature_box.attach(align, 2, 3, 3, 4, gtk.EXPAND | gtk.FILL, gtk.FILL)
190
self.set_border_width(6)
191
self.pack_start(signature_box, expand=False)
194
def set_revision(self, revision):
195
self.revision = revision
196
revid = revision.revision_id
198
if self.repository.has_signature_for_revision_id(revid):
199
crypttext = self.repository.get_signature_text(revid)
200
self.show_signature(crypttext)
202
self.show_no_signature()
204
def show_no_signature(self):
205
self.signature_key_id_label.hide()
206
self.signature_key_id.set_text("")
208
self.signature_fingerprint_label.hide()
209
self.signature_fingerprint.set_text("")
211
self.signature_trust_label.hide()
212
self.signature_trust.set_text("")
214
self.signature_image.set_from_file(icon_path("sign-unknown.png"))
215
self.signature_label.set_markup("<b>Authenticity unknown</b>\n" +
216
"This revision has not been signed.")
218
def show_signature(self, crypttext):
219
key = seahorse.verify(crypttext)
221
if key and key.is_available():
223
if key.get_display_name() == self.revision.committer:
224
self.signature_image.set_from_file(icon_path("sign-ok.png"))
225
self.signature_label.set_markup("<b>Authenticity confirmed</b>\n" +
226
"This revision has been signed with " +
229
self.signature_image.set_from_file(icon_path("sign-bad.png"))
230
self.signature_label.set_markup("<b>Authenticity cannot be confirmed</b>\n" +
231
"Revision committer is not the same as signer.")
233
self.signature_image.set_from_file(icon_path("sign-bad.png"))
234
self.signature_label.set_markup("<b>Authenticity cannot be confirmed</b>\n" +
235
"This revision has been signed, but the " +
236
"key is not trusted.")
238
self.show_no_signature()
239
self.signature_image.set_from_file(icon_path("sign-bad.png"))
240
self.signature_label.set_markup("<b>Authenticity cannot be confirmed</b>\n" +
241
"Signature key not available.")
244
trust = key.get_trust()
246
if trust <= seahorse.TRUST_NEVER:
247
trust_text = 'never trusted'
248
elif trust == seahorse.TRUST_UNKNOWN:
249
trust_text = 'not trusted'
250
elif trust == seahorse.TRUST_MARGINAL:
251
trust_text = 'marginally trusted'
252
elif trust == seahorse.TRUST_FULL:
253
trust_text = 'fully trusted'
254
elif trust == seahorse.TRUST_ULTIMATE:
255
trust_text = 'ultimately trusted'
257
self.signature_key_id_label.show()
258
self.signature_key_id.set_text(key.get_id())
260
fingerprint = key.get_fingerprint()
261
if fingerprint == "":
262
fingerprint = '<span foreground="dim grey">N/A</span>'
264
self.signature_fingerprint_label.show()
265
self.signature_fingerprint.set_markup(fingerprint)
267
self.signature_trust_label.show()
268
self.signature_trust.set_text('This key is ' + trust_text)
271
class RevisionView(gtk.Notebook):
27
272
""" Custom widget for commit log details.
29
274
A variety of bzr tools may need to implement such a thing. This is a
33
def __init__(self, revision=None, scroll=True, tags=None):
34
super(LogView, self).__init__()
36
self.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
280
gobject.TYPE_PYOBJECT,
282
'The branch holding the revision being displayed',
283
gobject.PARAM_CONSTRUCT_ONLY | gobject.PARAM_WRITABLE
287
gobject.TYPE_PYOBJECT,
289
'The revision being displayed',
290
gobject.PARAM_READWRITE
294
gobject.TYPE_PYOBJECT,
297
gobject.PARAM_READWRITE
301
gobject.TYPE_PYOBJECT,
304
gobject.PARAM_READWRITE
308
def __init__(self, branch=None, repository=None):
309
gtk.Notebook.__init__(self)
311
self._revision = None
312
self._branch = branch
313
if branch is not None:
314
self._repository = branch.repository
38
self.set_policy(gtk.POLICY_NEVER, gtk.POLICY_NEVER)
39
self.set_shadow_type(gtk.SHADOW_NONE)
316
self._repository = repository
318
self._create_general()
319
self._create_relations()
321
self._create_signature()
322
self._create_file_info_view()
325
self.set_current_page(PAGE_GENERAL)
326
self.connect_after('switch-page', self._switch_page_cb)
41
328
self._show_callback = None
42
self._go_callback = None
43
329
self._clicked_callback = None
45
if revision is not None:
46
self.set_revision(revision, tags=tags)
331
self._revision = None
332
self._branch = branch
336
self.set_file_id(None)
338
def do_get_property(self, property):
339
if property.name == 'branch':
341
elif property.name == 'revision':
342
return self._revision
343
elif property.name == 'children':
344
return self._children
345
elif property.name == 'file-id':
348
raise AttributeError, 'unknown property %s' % property.name
350
def do_set_property(self, property, value):
351
if property.name == 'branch':
353
elif property.name == 'revision':
354
self._set_revision(value)
355
elif property.name == 'children':
356
self.set_children(value)
357
elif property.name == 'file-id':
358
self._file_id = value
360
raise AttributeError, 'unknown property %s' % property.name
48
362
def set_show_callback(self, callback):
49
363
self._show_callback = callback
51
def set_go_callback(self, callback):
52
self._go_callback = callback
54
def set_revision(self, revision, tags=None):
365
def set_file_id(self, file_id):
366
"""Set a specific file id that we want to track.
368
This just effects the display of a per-file commit message.
369
If it is set to None, then all commit messages will be shown.
371
self.set_property('file-id', file_id)
373
def set_revision(self, revision):
374
if revision != self._revision:
375
self.set_property('revision', revision)
377
def get_revision(self):
378
return self.get_property('revision')
380
def _set_revision(self, revision):
381
if revision is None: return
55
383
self._revision = revision
56
self.revision_id.set_text(revision.revision_id)
57
384
if revision.committer is not None:
58
385
self.committer.set_text(revision.committer)
60
387
self.committer.set_text("")
388
author = revision.properties.get('author', '')
390
self.author.set_text(author)
392
self.author_label.show()
395
self.author_label.hide()
61
397
if revision.timestamp is not None:
62
398
self.timestamp.set_text(format_date(revision.timestamp,
63
399
revision.timezone))
64
self.message_buffer.set_text(revision.message)
66
401
self.branchnick_label.set_text(revision.properties['branch-nick'])
68
403
self.branchnick_label.set_text("")
70
self._add_parents(revision.parent_ids)
405
self._add_parents_or_children(revision.parent_ids,
406
self.parents_widgets,
409
file_info = revision.properties.get('file-info', None)
410
if file_info is not None:
411
file_info = bdecode(file_info.encode('UTF-8'))
414
if self._file_id is None:
417
text.append('%(path)s\n%(message)s' % fi)
418
self.file_info_buffer.set_text('\n'.join(text))
419
self.file_info_box.show()
423
if fi['file_id'] == self._file_id:
424
text.append(fi['message'])
426
self.file_info_buffer.set_text('\n'.join(text))
427
self.file_info_box.show()
429
self.file_info_box.hide()
431
self.file_info_box.hide()
433
def update_tags(self):
434
if self._branch is not None and self._branch.supports_tags():
435
self._tagdict = self._branch.tags.get_reverse_tag_dict()
441
def _update_signature(self, widget, param):
442
if self.get_current_page() == PAGE_SIGNATURE:
443
self.signature_table.set_revision(self._revision)
445
def _update_bugs(self, widget, param):
446
self.bugs_page.set_revision(self._revision)
447
label = self.get_tab_label(self.bugs_page)
448
label.set_sensitive(self.bugs_page.get_num_bugs() != 0)
450
def set_children(self, children):
451
self._add_parents_or_children(children,
452
self.children_widgets,
455
def _switch_page_cb(self, notebook, page, page_num):
456
if page_num == PAGE_SIGNATURE:
457
self.signature_table.set_revision(self._revision)
73
461
def _show_clicked_cb(self, widget, revid, parentid):
74
462
"""Callback for when the show button for a parent is clicked."""
222
637
self.tags_label.set_markup("<b>Tags:</b>")
223
638
align.add(self.tags_label)
225
self.table.attach(align, 0, 1, 4, 5, gtk.FILL, gtk.FILL)
640
self.table.attach(align, 0, 1, 5, 6, gtk.FILL, gtk.FILL)
226
641
self.tags_label.show()
228
643
align = gtk.Alignment(0.0, 0.5)
229
self.tags_list = gtk.VBox()
644
self.tags_list = gtk.Label()
230
645
align.add(self.tags_list)
231
self.table.attach(align, 1, 2, 4, 5, gtk.EXPAND | gtk.FILL, gtk.FILL)
646
self.table.attach(align, 1, 2, 5, 6, gtk.EXPAND | gtk.FILL, gtk.FILL)
233
648
self.tags_list.show()
234
self.tags_widgets = []
650
self.connect('notify::revision', self._add_tags)
236
652
return self.table
238
def _create_parents_table(self):
239
self.parents_table = gtk.Table(rows=1, columns=2)
240
self.parents_table.set_row_spacings(3)
241
self.parents_table.set_col_spacings(6)
242
self.parents_table.show()
654
def _create_parents(self):
655
hbox = gtk.HBox(True, 3)
657
self.parents_table = self._create_parents_or_children_table(
243
659
self.parents_widgets = []
660
hbox.pack_start(self.parents_table)
665
def _create_children(self):
666
hbox = gtk.HBox(True, 3)
667
self.children_table = self._create_parents_or_children_table(
669
self.children_widgets = []
670
hbox.pack_start(self.children_table)
674
def _create_parents_or_children_table(self, text):
675
table = gtk.Table(rows=1, columns=2)
676
table.set_row_spacings(3)
677
table.set_col_spacings(6)
245
680
label = gtk.Label()
246
label.set_markup("<b>Parents:</b>")
681
label.set_markup(text)
247
682
align = gtk.Alignment(0.0, 0.5)
249
self.parents_table.attach(align, 0, 1, 0, 1, gtk.FILL, gtk.FILL)
684
table.attach(align, 0, 1, 0, 1, gtk.FILL, gtk.FILL)
253
return self.parents_table
255
690
def _create_message_view(self):
256
self.message_buffer = gtk.TextBuffer()
257
tv = gtk.TextView(self.message_buffer)
258
tv.set_editable(False)
259
tv.set_wrap_mode(gtk.WRAP_WORD)
260
tv.modify_font(pango.FontDescription("Monospace"))
691
msg_buffer = gtk.TextBuffer()
692
self.connect('notify::revision',
693
lambda w, p: msg_buffer.set_text(self._revision.message))
694
window = gtk.ScrolledWindow()
695
window.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
696
window.set_shadow_type(gtk.SHADOW_IN)
697
tv = gtk.TextView(msg_buffer)
698
tv.set_editable(False)
699
tv.set_wrap_mode(gtk.WRAP_WORD)
701
tv.modify_font(pango.FontDescription("Monospace"))
707
def _create_bugs(self):
708
self.bugs_page = BugsTab()
709
self.connect_after('notify::revision', self._update_bugs)
710
self.append_page(self.bugs_page, tab_label=gtk.Label('Bugs'))
712
def _create_file_info_view(self):
713
self.file_info_box = gtk.VBox(False, 6)
714
self.file_info_box.set_border_width(6)
715
self.file_info_buffer = gtk.TextBuffer()
716
window = gtk.ScrolledWindow()
717
window.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
718
window.set_shadow_type(gtk.SHADOW_IN)
719
tv = gtk.TextView(self.file_info_buffer)
720
tv.set_editable(False)
721
tv.set_wrap_mode(gtk.WRAP_WORD)
722
tv.modify_font(pango.FontDescription("Monospace"))
726
self.file_info_box.pack_start(window)
727
self.file_info_box.hide() # Only shown when there are per-file messages
728
self.append_page(self.file_info_box, tab_label=gtk.Label('Per-file'))