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
29
def _open_link(widget, uri):
30
subprocess.Popen(['sensible-browser', uri], close_fds=True)
32
gtk.link_button_set_uri_hook(_open_link)
34
class BugsTab(gtk.Table):
36
super(BugsTab, self).__init__(rows=5, columns=2)
37
self.set_row_spacings(6)
38
self.set_col_spacings(6)
42
for c in self.get_children():
45
self.hide_all() # Only shown when there are bugs
47
def add_bug(self, url, status):
48
button = gtk.LinkButton(url, url)
49
self.attach(button, 0, 1, self.count, self.count + 1,
50
gtk.EXPAND | gtk.FILL, gtk.FILL)
51
status_label = gtk.Label(status)
52
self.attach(status_label, 1, 2, self.count, self.count + 1,
53
gtk.EXPAND | gtk.FILL, gtk.FILL)
58
class SignatureTab(gtk.VBox):
60
from gpg import GPGSubprocess
61
self.gpg = GPGSubprocess()
62
super(SignatureTab, self).__init__(False, 6)
63
signature_box = gtk.Table(rows=1, columns=2)
64
signature_box.set_col_spacing(0, 12)
66
self.signature_image = gtk.Image()
67
signature_box.attach(self.signature_image, 0, 1, 0, 1, gtk.FILL)
69
self.signature_label = gtk.Label()
70
signature_box.attach(self.signature_label, 1, 2, 0, 1, gtk.FILL)
72
signature_info = gtk.Table(rows=1, columns=2)
73
signature_info.set_row_spacings(6)
74
signature_info.set_col_spacings(6)
76
align = gtk.Alignment(1.0, 0.5)
78
label.set_markup("<b>Key Id:</b>")
80
signature_info.attach(align, 0, 1, 0, 1, gtk.FILL, gtk.FILL)
82
align = gtk.Alignment(0.0, 0.5)
83
self.signature_key_id = gtk.Label()
84
self.signature_key_id.set_selectable(True)
85
align.add(self.signature_key_id)
86
signature_info.attach(align, 1, 2, 0, 1, gtk.EXPAND | gtk.FILL, gtk.FILL)
88
self.set_border_width(6)
89
self.pack_start(signature_box, expand=False)
90
self.pack_start(signature_info, expand=False)
93
def show_no_signature(self):
94
self.signature_key_id.set_text("")
95
self.signature_image.set_from_file(icon_path("sign-unknown.png"))
96
self.signature_label.set_text("This revision has not been signed.")
98
def show_signature(self, text):
99
signature = self.gpg.verify(text)
101
if signature.key_id is not None:
102
self.signature_key_id.set_text(signature.key_id)
104
if signature.is_valid():
105
self.signature_image.set_from_file(icon_path("sign-ok.png"))
106
self.signature_label.set_text("This revision has been signed.")
108
self.signature_image.set_from_file(icon_path("sign-bad.png"))
109
self.signature_label.set_text("This revision has been signed, " +
110
"but the authenticity of the signature cannot be verified.")
113
class RevisionView(gtk.Notebook):
27
114
""" Custom widget for commit log details.
29
116
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)
39
self.set_policy(gtk.POLICY_NEVER, gtk.POLICY_NEVER)
40
self.set_shadow_type(gtk.SHADOW_NONE)
41
self.show_children = show_children
122
gobject.TYPE_PYOBJECT,
124
'The branch holding the revision being displayed',
125
gobject.PARAM_CONSTRUCT_ONLY | gobject.PARAM_WRITABLE
129
gobject.TYPE_PYOBJECT,
131
'The revision being displayed',
132
gobject.PARAM_READWRITE
136
gobject.TYPE_PYOBJECT,
139
gobject.PARAM_READWRITE
143
gobject.TYPE_PYOBJECT,
146
gobject.PARAM_READWRITE
151
def __init__(self, branch=None):
152
gtk.Notebook.__init__(self)
154
self._create_general()
155
self._create_relations()
156
self._create_signature()
157
self._create_file_info_view()
160
self.set_current_page(0)
43
162
self._show_callback = None
44
self._go_callback = None
45
163
self._clicked_callback = None
47
if revision is not None:
48
self.set_revision(revision, tags=tags)
165
self._revision = None
166
self._branch = branch
170
self.set_file_id(None)
172
def do_get_property(self, property):
173
if property.name == 'branch':
175
elif property.name == 'revision':
176
return self._revision
177
elif property.name == 'children':
178
return self._children
179
elif property.name == 'file-id':
182
raise AttributeError, 'unknown property %s' % property.name
184
def do_set_property(self, property, value):
185
if property.name == 'branch':
187
elif property.name == 'revision':
188
self._set_revision(value)
189
elif property.name == 'children':
190
self.set_children(value)
191
elif property.name == 'file-id':
192
self._file_id = value
194
raise AttributeError, 'unknown property %s' % property.name
50
196
def set_show_callback(self, callback):
51
197
self._show_callback = callback
53
def set_go_callback(self, callback):
54
self._go_callback = callback
56
def set_revision(self, revision, tags=[], children=[]):
199
def set_file_id(self, file_id):
200
"""Set a specific file id that we want to track.
202
This just effects the display of a per-file commit message.
203
If it is set to None, then all commit messages will be shown.
205
self.set_property('file-id', file_id)
207
def set_revision(self, revision):
208
if revision != self._revision:
209
self.set_property('revision', revision)
211
def get_revision(self):
212
return self.get_property('revision')
214
def _set_revision(self, revision):
215
if revision is None: return
57
217
self._revision = revision
58
self.revision_id.set_text(revision.revision_id)
59
218
if revision.committer is not None:
60
219
self.committer.set_text(revision.committer)
82
240
self.parents_widgets,
83
241
self.parents_table)
85
if self.show_children:
86
self._add_parents_or_children(children,
87
self.children_widgets,
243
file_info = revision.properties.get('file-info', None)
244
if file_info is not None:
245
file_info = bdecode(file_info.encode('UTF-8'))
248
if self._file_id is None:
251
text.append('%(path)s\n%(message)s' % fi)
252
self.file_info_buffer.set_text('\n'.join(text))
253
self.file_info_box.show()
257
if fi['file_id'] == self._file_id:
258
text.append(fi['message'])
260
self.file_info_buffer.set_text('\n'.join(text))
261
self.file_info_box.show()
263
self.file_info_box.hide()
265
self.file_info_box.hide()
267
self.bugs_table.clear()
268
bugs_text = revision.properties.get('bugs', None)
270
for bugline in bugs_text.splitlines():
271
(url, status) = bugline.split(" ")
272
self.bugs_table.add_bug(url, status)
274
def update_tags(self):
275
if self._branch is not None and self._branch.supports_tags():
276
self._tagdict = self._branch.tags.get_reverse_tag_dict()
282
def _update_signature(self, widget, param):
283
revid = self._revision.revision_id
285
if self._branch.repository.has_signature_for_revision_id(revid):
286
signature_text = self._branch.repository.get_signature_text(revid)
287
self.signature_table.show_signature(signature_text)
289
self.signature_table.show_no_signature()
291
def set_children(self, children):
292
self._add_parents_or_children(children,
293
self.children_widgets,
92
296
def _show_clicked_cb(self, widget, revid, parentid):
93
297
"""Callback for when the show button for a parent is clicked."""
147
350
hbox.pack_start(button, expand=False, fill=True)
150
if self._go_callback is not None:
151
button = gtk.Button(revid)
152
button.connect("clicked", self._go_clicked_cb, revid)
154
button = gtk.Label(revid)
353
button = gtk.Button(revid)
354
button.connect("clicked",
355
lambda w, r: self.set_revision(self._branch.repository.get_revision(r)), revid)
155
356
button.set_use_underline(False)
156
357
hbox.pack_start(button, expand=False, fill=True)
360
def _create_general(self):
160
361
vbox = gtk.VBox(False, 6)
161
362
vbox.set_border_width(6)
162
363
vbox.pack_start(self._create_headers(), expand=False, fill=True)
364
vbox.pack_start(self._create_message_view())
365
self.append_page(vbox, tab_label=gtk.Label("General"))
368
def _create_relations(self):
369
vbox = gtk.VBox(False, 6)
370
vbox.set_border_width(6)
163
371
vbox.pack_start(self._create_parents(), expand=False, fill=True)
164
if self.show_children:
165
vbox.pack_start(self._create_children(), expand=False, fill=True)
166
vbox.pack_start(self._create_message_view())
167
self.add_with_viewport(vbox)
372
vbox.pack_start(self._create_children(), expand=False, fill=True)
373
self.append_page(vbox, tab_label=gtk.Label("Relations"))
376
def _create_signature(self):
378
self.signature_table = SignatureTab()
379
except ValueError: # No GPG found
380
self.signature_table = None
382
self.append_page(self.signature_table, tab_label=gtk.Label('Signature'))
383
self.connect_after('notify::revision', self._update_signature)
170
385
def _create_headers(self):
171
386
self.table = gtk.Table(rows=5, columns=2)
172
387
self.table.set_row_spacings(6)
314
529
def _create_message_view(self):
315
self.message_buffer = gtk.TextBuffer()
316
tv = gtk.TextView(self.message_buffer)
317
tv.set_editable(False)
318
tv.set_wrap_mode(gtk.WRAP_WORD)
319
tv.modify_font(pango.FontDescription("Monospace"))
530
msg_buffer = gtk.TextBuffer()
531
self.connect('notify::revision',
532
lambda w, p: msg_buffer.set_text(self._revision.message))
533
window = gtk.ScrolledWindow()
534
window.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
535
window.set_shadow_type(gtk.SHADOW_IN)
536
tv = gtk.TextView(msg_buffer)
537
tv.set_editable(False)
538
tv.set_wrap_mode(gtk.WRAP_WORD)
540
tv.modify_font(pango.FontDescription("Monospace"))
546
def _create_bugs(self):
547
self.bugs_table = BugsTab()
548
self.append_page(self.bugs_table, tab_label=gtk.Label('Bugs'))
550
def _create_file_info_view(self):
551
self.file_info_box = gtk.VBox(False, 6)
552
self.file_info_box.set_border_width(6)
553
self.file_info_buffer = gtk.TextBuffer()
554
window = gtk.ScrolledWindow()
555
window.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
556
window.set_shadow_type(gtk.SHADOW_IN)
557
tv = gtk.TextView(self.file_info_buffer)
558
tv.set_editable(False)
559
tv.set_wrap_mode(gtk.WRAP_WORD)
560
tv.modify_font(pango.FontDescription("Monospace"))
564
self.file_info_box.pack_start(window)
565
self.file_info_box.hide() # Only shown when there are per-file messages
566
self.append_page(self.file_info_box, tab_label=gtk.Label('Per-file'))