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
42
def _open_link(widget, uri):
43
for cmd in ['sensible-browser', 'xdg-open']:
44
if webbrowser._iscommand(cmd):
45
webbrowser._tryorder.insert(0, '%s "%%s"' % cmd)
48
gtk.link_button_set_uri_hook(_open_link)
50
class BugsTab(gtk.VBox):
53
super(BugsTab, self).__init__(False, 6)
55
table = gtk.Table(rows=2, columns=2)
57
table.set_row_spacings(6)
58
table.set_col_spacing(0, 16)
61
image.set_from_file(icon_path("bug.png"))
62
table.attach(image, 0, 1, 0, 1, gtk.FILL)
64
align = gtk.Alignment(0.0, 0.1)
65
self.label = gtk.Label()
67
table.attach(align, 1, 2, 0, 1, gtk.FILL)
69
treeview = self.construct_treeview()
70
table.attach(treeview, 1, 2, 1, 2, gtk.FILL | gtk.EXPAND)
72
self.set_border_width(6)
73
self.pack_start(table, expand=False)
78
def set_revision(self, revision):
83
bugs_text = revision.properties.get('bugs', '')
84
for bugline in bugs_text.splitlines():
85
(url, status) = bugline.split(" ")
87
self.add_bug(url, status)
89
if self.num_bugs == 0:
91
elif self.num_bugs == 1:
96
self.label.set_markup("<b>Bugs fixed</b>\n" +
97
"This revision claims to fix " +
98
"%d %s." % (self.num_bugs, label))
100
def construct_treeview(self):
101
self.bugs = gtk.ListStore(gobject.TYPE_STRING, gobject.TYPE_STRING)
102
self.treeview = gtk.TreeView(self.bugs)
103
self.treeview.set_headers_visible(False)
105
uri_column = gtk.TreeViewColumn('Bug URI', gtk.CellRendererText(), text=0)
106
self.treeview.append_column(uri_column)
108
self.treeview.connect('row-activated', self.on_row_activated)
110
win = gtk.ScrolledWindow()
111
win.set_policy(gtk.POLICY_AUTOMATIC, gtk.POLICY_AUTOMATIC)
112
win.set_shadow_type(gtk.SHADOW_IN)
113
win.add(self.treeview)
120
self.set_sensitive(False)
121
self.label.set_markup("<b>No bugs fixed</b>\n" +
122
"This revision does not claim to fix any bugs.")
124
def add_bug(self, url, status):
126
self.bugs.append([url, status])
127
self.set_sensitive(True)
129
def get_num_bugs(self):
132
def on_row_activated(self, treeview, path, column):
133
uri = self.bugs.get_value(self.bugs.get_iter(path), 0)
134
_open_link(self, uri)
137
class SignatureTab(gtk.VBox):
139
def __init__(self, repository):
142
self.repository = repository
144
super(SignatureTab, self).__init__(False, 6)
145
signature_box = gtk.Table(rows=3, columns=3)
146
signature_box.set_col_spacing(0, 16)
147
signature_box.set_col_spacing(1, 12)
148
signature_box.set_row_spacings(6)
150
self.signature_image = gtk.Image()
151
signature_box.attach(self.signature_image, 0, 1, 0, 1, gtk.FILL)
153
align = gtk.Alignment(0.0, 0.1)
154
self.signature_label = gtk.Label()
155
align.add(self.signature_label)
156
signature_box.attach(align, 1, 3, 0, 1, gtk.FILL)
158
align = gtk.Alignment(0.0, 0.5)
159
self.signature_key_id_label = gtk.Label()
160
self.signature_key_id_label.set_markup("<b>Key Id:</b>")
161
align.add(self.signature_key_id_label)
162
signature_box.attach(align, 1, 2, 1, 2, gtk.FILL, gtk.FILL)
164
align = gtk.Alignment(0.0, 0.5)
165
self.signature_key_id = gtk.Label()
166
self.signature_key_id.set_selectable(True)
167
align.add(self.signature_key_id)
168
signature_box.attach(align, 2, 3, 1, 2, gtk.EXPAND | gtk.FILL, gtk.FILL)
170
align = gtk.Alignment(0.0, 0.5)
171
self.signature_fingerprint_label = gtk.Label()
172
self.signature_fingerprint_label.set_markup("<b>Fingerprint:</b>")
173
align.add(self.signature_fingerprint_label)
174
signature_box.attach(align, 1, 2, 2, 3, gtk.FILL, gtk.FILL)
176
align = gtk.Alignment(0.0, 0.5)
177
self.signature_fingerprint = gtk.Label()
178
self.signature_fingerprint.set_selectable(True)
179
align.add(self.signature_fingerprint)
180
signature_box.attach(align, 2, 3, 2, 3, gtk.EXPAND | gtk.FILL, gtk.FILL)
182
align = gtk.Alignment(0.0, 0.5)
183
self.signature_trust_label = gtk.Label()
184
self.signature_trust_label.set_markup("<b>Trust:</b>")
185
align.add(self.signature_trust_label)
186
signature_box.attach(align, 1, 2, 3, 4, gtk.FILL, gtk.FILL)
188
align = gtk.Alignment(0.0, 0.5)
189
self.signature_trust = gtk.Label()
190
self.signature_trust.set_selectable(True)
191
align.add(self.signature_trust)
192
signature_box.attach(align, 2, 3, 3, 4, gtk.EXPAND | gtk.FILL, gtk.FILL)
194
self.set_border_width(6)
195
self.pack_start(signature_box, expand=False)
198
def set_revision(self, revision):
199
self.revision = revision
200
revid = revision.revision_id
202
if self.repository.has_signature_for_revision_id(revid):
203
crypttext = self.repository.get_signature_text(revid)
204
self.show_signature(crypttext)
206
self.show_no_signature()
208
def show_no_signature(self):
209
self.signature_key_id_label.hide()
210
self.signature_key_id.set_text("")
212
self.signature_fingerprint_label.hide()
213
self.signature_fingerprint.set_text("")
215
self.signature_trust_label.hide()
216
self.signature_trust.set_text("")
218
self.signature_image.set_from_file(icon_path("sign-unknown.png"))
219
self.signature_label.set_markup("<b>Authenticity unknown</b>\n" +
220
"This revision has not been signed.")
222
def show_signature(self, crypttext):
223
(cleartext, key) = seahorse.verify(crypttext)
225
if key and key.is_available():
227
if key.get_display_name() == self.revision.committer:
228
self.signature_image.set_from_file(icon_path("sign-ok.png"))
229
self.signature_label.set_markup("<b>Authenticity confirmed</b>\n" +
230
"This revision has been signed with " +
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
"Revision committer is not the same as signer.")
237
self.signature_image.set_from_file(icon_path("sign-bad.png"))
238
self.signature_label.set_markup("<b>Authenticity cannot be confirmed</b>\n" +
239
"This revision has been signed, but the " +
240
"key is not trusted.")
242
self.show_no_signature()
243
self.signature_image.set_from_file(icon_path("sign-bad.png"))
244
self.signature_label.set_markup("<b>Authenticity cannot be confirmed</b>\n" +
245
"Signature key not available.")
248
trust = key.get_trust()
250
if trust <= seahorse.TRUST_NEVER:
251
trust_text = 'never trusted'
252
elif trust == seahorse.TRUST_UNKNOWN:
253
trust_text = 'not trusted'
254
elif trust == seahorse.TRUST_MARGINAL:
255
trust_text = 'marginally trusted'
256
elif trust == seahorse.TRUST_FULL:
257
trust_text = 'fully trusted'
258
elif trust == seahorse.TRUST_ULTIMATE:
259
trust_text = 'ultimately trusted'
261
self.signature_key_id_label.show()
262
self.signature_key_id.set_text(key.get_id())
264
fingerprint = key.get_fingerprint()
265
if fingerprint == "":
266
fingerprint = '<span foreground="dim grey">N/A</span>'
268
self.signature_fingerprint_label.show()
269
self.signature_fingerprint.set_markup(fingerprint)
271
self.signature_trust_label.show()
272
self.signature_trust.set_text('This key is ' + trust_text)
275
class RevisionView(gtk.Notebook):
27
276
""" Custom widget for commit log details.
29
278
A variety of bzr tools may need to implement such a thing. This is a
33
def __init__(self, revision=None, scroll=True, tags=[],
35
super(LogView, self).__init__()
37
self.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
284
gobject.TYPE_PYOBJECT,
286
'The branch holding the revision being displayed',
287
gobject.PARAM_CONSTRUCT_ONLY | gobject.PARAM_WRITABLE
291
gobject.TYPE_PYOBJECT,
293
'The revision being displayed',
294
gobject.PARAM_READWRITE
298
gobject.TYPE_PYOBJECT,
301
gobject.PARAM_READWRITE
305
gobject.TYPE_PYOBJECT,
308
gobject.PARAM_READWRITE
312
def __init__(self, branch=None, repository=None):
313
gtk.Notebook.__init__(self)
315
self._revision = None
316
self._branch = branch
317
if branch is not None:
318
self._repository = branch.repository
39
self.set_policy(gtk.POLICY_NEVER, gtk.POLICY_NEVER)
40
self.set_shadow_type(gtk.SHADOW_NONE)
41
self.show_children = show_children
320
self._repository = repository
322
self._create_general()
323
self._create_relations()
324
# Disabled because testaments aren't verified yet:
326
self._create_signature()
327
self._create_file_info_view()
330
self.set_current_page(PAGE_GENERAL)
331
self.connect_after('switch-page', self._switch_page_cb)
43
333
self._show_callback = None
44
self._go_callback = None
45
334
self._clicked_callback = None
47
if revision is not None:
48
self.set_revision(revision, tags=tags)
336
self._revision = None
337
self._branch = branch
341
self.set_file_id(None)
343
def do_get_property(self, property):
344
if property.name == 'branch':
346
elif property.name == 'revision':
347
return self._revision
348
elif property.name == 'children':
349
return self._children
350
elif property.name == 'file-id':
353
raise AttributeError, 'unknown property %s' % property.name
355
def do_set_property(self, property, value):
356
if property.name == 'branch':
358
elif property.name == 'revision':
359
self._set_revision(value)
360
elif property.name == 'children':
361
self.set_children(value)
362
elif property.name == 'file-id':
363
self._file_id = value
365
raise AttributeError, 'unknown property %s' % property.name
50
367
def set_show_callback(self, callback):
51
368
self._show_callback = callback
53
def set_go_callback(self, callback):
54
self._go_callback = callback
56
def set_revision(self, revision, tags=[], children=[]):
370
def set_file_id(self, file_id):
371
"""Set a specific file id that we want to track.
373
This just effects the display of a per-file commit message.
374
If it is set to None, then all commit messages will be shown.
376
self.set_property('file-id', file_id)
378
def set_revision(self, revision):
379
if revision != self._revision:
380
self.set_property('revision', revision)
382
def get_revision(self):
383
return self.get_property('revision')
385
def _set_revision(self, revision):
386
if revision is None: return
57
388
self._revision = revision
58
self.revision_id.set_text(revision.revision_id)
59
389
if revision.committer is not None:
60
390
self.committer.set_text(revision.committer)