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
        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=[]):
 
34
 
        super(LogView, self).__init__()
 
36
 
            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
 
38
 
            self.set_policy(gtk.POLICY_NEVER, gtk.POLICY_NEVER)
 
39
 
        self.set_shadow_type(gtk.SHADOW_NONE)
 
 
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)
 
41
333
        self._show_callback = None
 
42
 
        self._go_callback = None
 
43
334
        self._clicked_callback = None
 
45
 
        if revision is not None:
 
46
 
            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
 
48
367
    def set_show_callback(self, callback):
 
49
368
        self._show_callback = callback
 
51
 
    def set_go_callback(self, callback):
 
52
 
        self._go_callback = callback
 
54
 
    def set_revision(self, revision, tags=[]):
 
 
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
 
55
388
        self._revision = revision
 
56
 
        self.revision_id.set_text(revision.revision_id)
 
57
389
        if revision.committer is not None:
 
58
390
            self.committer.set_text(revision.committer)
 
60
392
            self.committer.set_text("")
 
 
393
        author = revision.properties.get('author', '')
 
 
395
            self.author.set_text(author)
 
 
397
            self.author_label.show()
 
 
400
            self.author_label.hide()
 
61
402
        if revision.timestamp is not None:
 
62
403
            self.timestamp.set_text(format_date(revision.timestamp,
 
63
404
                                                revision.timezone))
 
64
 
        self.message_buffer.set_text(revision.message)
 
66
406
            self.branchnick_label.set_text(revision.properties['branch-nick'])
 
68
408
            self.branchnick_label.set_text("")
 
70
 
        self._add_parents(revision.parent_ids)
 
 
410
        self._add_parents_or_children(revision.parent_ids,
 
 
411
                                      self.parents_widgets,
 
 
414
        file_info = revision.properties.get('file-info', None)
 
 
415
        if file_info is not None:
 
 
416
            file_info = bdecode(file_info.encode('UTF-8'))
 
 
419
            if self._file_id is None:
 
 
422
                    text.append('%(path)s\n%(message)s' % fi)
 
 
423
                self.file_info_buffer.set_text('\n'.join(text))
 
 
424
                self.file_info_box.show()
 
 
428
                    if fi['file_id'] == self._file_id:
 
 
429
                        text.append(fi['message'])
 
 
431
                    self.file_info_buffer.set_text('\n'.join(text))
 
 
432
                    self.file_info_box.show()
 
 
434
                    self.file_info_box.hide()
 
 
436
            self.file_info_box.hide()
 
 
438
    def update_tags(self):
 
 
439
        if self._branch is not None and self._branch.supports_tags():
 
 
440
            self._tagdict = self._branch.tags.get_reverse_tag_dict()
 
 
446
    def _update_signature(self, widget, param):
 
 
447
        if self.get_current_page() == PAGE_SIGNATURE:
 
 
448
            self.signature_table.set_revision(self._revision)
 
 
450
    def _update_bugs(self, widget, param):
 
 
451
        self.bugs_page.set_revision(self._revision)
 
 
452
        label = self.get_tab_label(self.bugs_page)
 
 
453
        label.set_sensitive(self.bugs_page.get_num_bugs() != 0)
 
 
455
    def set_children(self, children):
 
 
456
        self._add_parents_or_children(children,
 
 
457
                                      self.children_widgets,
 
 
460
    def _switch_page_cb(self, notebook, page, page_num):
 
 
461
        if page_num == PAGE_SIGNATURE:
 
 
462
            self.signature_table.set_revision(self._revision)
 
73
466
    def _show_clicked_cb(self, widget, revid, parentid):
 
74
467
        """Callback for when the show button for a parent is clicked."""
 
 
222
642
        self.tags_label.set_markup("<b>Tags:</b>")
 
223
643
        align.add(self.tags_label)
 
225
 
        self.table.attach(align, 0, 1, 4, 5, gtk.FILL, gtk.FILL)
 
 
645
        self.table.attach(align, 0, 1, 5, 6, gtk.FILL, gtk.FILL)
 
226
646
        self.tags_label.show()
 
228
648
        align = gtk.Alignment(0.0, 0.5)
 
229
 
        self.tags_list = gtk.VBox()
 
 
649
        self.tags_list = gtk.Label()
 
230
650
        align.add(self.tags_list)
 
231
 
        self.table.attach(align, 1, 2, 4, 5, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
 
651
        self.table.attach(align, 1, 2, 5, 6, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
233
653
        self.tags_list.show()
 
234
 
        self.tags_widgets = []
 
 
655
        self.connect('notify::revision', self._add_tags)
 
236
657
        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()
 
 
659
    def _create_parents(self):
 
 
660
        hbox = gtk.HBox(True, 3)
 
 
662
        self.parents_table = self._create_parents_or_children_table(
 
243
664
        self.parents_widgets = []
 
 
665
        hbox.pack_start(self.parents_table)
 
 
670
    def _create_children(self):
 
 
671
        hbox = gtk.HBox(True, 3)
 
 
672
        self.children_table = self._create_parents_or_children_table(
 
 
674
        self.children_widgets = []
 
 
675
        hbox.pack_start(self.children_table)
 
 
679
    def _create_parents_or_children_table(self, text):
 
 
680
        table = gtk.Table(rows=1, columns=2)
 
 
681
        table.set_row_spacings(3)
 
 
682
        table.set_col_spacings(6)
 
245
685
        label = gtk.Label()
 
246
 
        label.set_markup("<b>Parents:</b>")
 
 
686
        label.set_markup(text)
 
247
687
        align = gtk.Alignment(0.0, 0.5)
 
249
 
        self.parents_table.attach(align, 0, 1, 0, 1, gtk.FILL, gtk.FILL)
 
 
689
        table.attach(align, 0, 1, 0, 1, gtk.FILL, gtk.FILL)
 
253
 
        return self.parents_table
 
255
695
    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"))
 
 
696
        msg_buffer = gtk.TextBuffer()
 
 
697
        self.connect('notify::revision',
 
 
698
                lambda w, p: msg_buffer.set_text(self._revision.message))
 
 
699
        window = gtk.ScrolledWindow()
 
 
700
        window.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
 
 
701
        window.set_shadow_type(gtk.SHADOW_IN)
 
 
702
        tv = gtk.TextView(msg_buffer)
 
 
703
        tv.set_editable(False)
 
 
704
        tv.set_wrap_mode(gtk.WRAP_WORD)
 
 
706
        tv.modify_font(pango.FontDescription("Monospace"))
 
 
712
    def _create_bugs(self):
 
 
713
        self.bugs_page = BugsTab()
 
 
714
        self.connect_after('notify::revision', self._update_bugs) 
 
 
715
        self.append_page(self.bugs_page, tab_label=gtk.Label('Bugs'))
 
 
717
    def _create_file_info_view(self):
 
 
718
        self.file_info_box = gtk.VBox(False, 6)
 
 
719
        self.file_info_box.set_border_width(6)
 
 
720
        self.file_info_buffer = gtk.TextBuffer()
 
 
721
        window = gtk.ScrolledWindow()
 
 
722
        window.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
 
 
723
        window.set_shadow_type(gtk.SHADOW_IN)
 
 
724
        tv = gtk.TextView(self.file_info_buffer)
 
 
725
        tv.set_editable(False)
 
 
726
        tv.set_wrap_mode(gtk.WRAP_WORD)
 
 
727
        tv.modify_font(pango.FontDescription("Monospace"))
 
 
731
        self.file_info_box.pack_start(window)
 
 
732
        self.file_info_box.hide() # Only shown when there are per-file messages
 
 
733
        self.append_page(self.file_info_box, tab_label=gtk.Label('Per-file'))