/b-gtk/fix-viz

To get this branch, use:
bzr branch http://gegoxaren.bato24.eu/bzr/b-gtk/fix-viz

« back to all changes in this revision

Viewing changes to revisionview.py

  • Committer: Daniel Schierbeck
  • Date: 2008-04-07 20:34:51 UTC
  • mfrom: (450.6.13 bugs)
  • mto: (463.2.1 bug.78765)
  • mto: This revision was merged to the branch mainline in revision 462.
  • Revision ID: daniel.schierbeck@gmail.com-20080407203451-2i6el7jf9t0k9y64
Merged bug page improvements.

Show diffs side-by-side

added added

removed removed

Lines of Context:
20
20
import gtk
21
21
import pango
22
22
import gobject
23
 
import webbrowser
 
23
import subprocess
24
24
 
25
 
from bzrlib import trace
 
25
from bzrlib.plugins.gtk import icon_path
26
26
from bzrlib.osutils import format_date
27
 
try:
28
 
    from bzrlib.bencode import bdecode
29
 
except ImportError:
30
 
    from bzrlib.util.bencode import bdecode
31
 
from bzrlib.testament import Testament
32
 
 
33
 
from bzrlib.plugins.gtk import icon_path
34
 
 
35
 
from bzrlib.plugins.gtk.avatarsbox import AvatarsBox
 
27
from bzrlib.util.bencode import bdecode
36
28
 
37
29
try:
38
30
    from bzrlib.plugins.gtk import seahorse
41
33
else:
42
34
    has_seahorse = True
43
35
 
44
 
PAGE_GENERAL = 0
45
 
PAGE_RELATIONS = 1
46
 
PAGE_SIGNATURE = 2
47
 
PAGE_BUGS = 3
48
 
 
49
 
 
50
36
def _open_link(widget, uri):
51
 
    for cmd in ['sensible-browser', 'xdg-open']:
52
 
        if webbrowser._iscommand(cmd):
53
 
            webbrowser._tryorder.insert(0, '%s "%%s"' % cmd)
54
 
    webbrowser.open(uri)
 
37
    subprocess.Popen(['sensible-browser', uri], close_fds=True)
55
38
 
56
 
if getattr(gtk, 'link_button_set_uri_hook', None) is not None:
57
 
    # Not available before PyGtk-2.10
58
 
    gtk.link_button_set_uri_hook(_open_link)
 
39
gtk.link_button_set_uri_hook(_open_link)
59
40
 
60
41
class BugsTab(gtk.VBox):
61
42
 
62
43
    def __init__(self):
63
44
        super(BugsTab, self).__init__(False, 6)
64
 
 
 
45
    
65
46
        table = gtk.Table(rows=2, columns=2)
66
47
 
67
48
        table.set_row_spacings(6)
95
76
                (url, status) = bugline.split(" ")
96
77
                if status == "fixed":
97
78
                    self.add_bug(url, status)
98
 
 
 
79
        
99
80
        if self.num_bugs == 0:
100
81
            return
101
82
        elif self.num_bugs == 1:
230
211
                                        "This revision has not been signed.")
231
212
 
232
213
    def show_signature(self, crypttext):
233
 
        (cleartext, key) = seahorse.verify(crypttext)
234
 
 
235
 
        assert cleartext is not None
236
 
 
237
 
        inv = self.repository.get_inventory(self.revision.revision_id)
238
 
        expected_testament = Testament(self.revision, inv).as_short_text()
239
 
        if expected_testament != cleartext:
240
 
            self.signature_image.set_from_file(icon_path("sign-bad.png"))
241
 
            self.signature_label.set_markup("<b>Signature does not match repository data</b>\n" +
242
 
                        "The signature plaintext is different from the expected testament plaintext.")
243
 
            return
 
214
        key = seahorse.verify(crypttext)
244
215
 
245
216
        if key and key.is_available():
246
217
            if key.is_trusted():
329
300
        )
330
301
    }
331
302
 
332
 
    def __init__(self, branch=None, repository=None):
 
303
 
 
304
    def __init__(self, branch=None):
333
305
        gtk.Notebook.__init__(self)
334
306
 
335
307
        self._revision = None
336
308
        self._branch = branch
337
 
        if branch is not None:
338
 
            self._repository = branch.repository
339
 
        else:
340
 
            self._repository = repository
341
309
 
342
310
        self._create_general()
343
311
        self._create_relations()
344
 
        # Disabled because testaments aren't verified yet:
345
312
        if has_seahorse:
346
313
            self._create_signature()
347
314
        self._create_file_info_view()
348
315
        self._create_bugs()
349
316
 
350
 
        self.set_current_page(PAGE_GENERAL)
351
 
        self.connect_after('switch-page', self._switch_page_cb)
 
317
        self.set_current_page(0)
352
318
        
353
319
        self._show_callback = None
354
320
        self._clicked_callback = None
404
370
 
405
371
    def _set_revision(self, revision):
406
372
        if revision is None: return
407
 
        
408
 
        self.avatarsbox.reset()
409
 
        
 
373
 
410
374
        self._revision = revision
411
375
        if revision.committer is not None:
412
376
            self.committer.set_text(revision.committer)
413
 
            self.avatarsbox.add(revision.committer, "committer")
414
377
        else:
415
378
            self.committer.set_text("")
416
 
            self.avatarsbox.hide()
417
379
        author = revision.properties.get('author', '')
418
 
        self.avatarsbox.merge(revision.get_apparent_authors(), "author")
419
380
        if author != '':
420
381
            self.author.set_text(author)
421
382
            self.author.show()
428
389
            self.timestamp.set_text(format_date(revision.timestamp,
429
390
                                                revision.timezone))
430
391
        try:
431
 
            self.branchnick.show()
432
 
            self.branchnick_label.show()
433
 
            self.branchnick.set_text(revision.properties['branch-nick'])
 
392
            self.branchnick_label.set_text(revision.properties['branch-nick'])
434
393
        except KeyError:
435
 
            self.branchnick.hide()
436
 
            self.branchnick_label.hide()
 
394
            self.branchnick_label.set_text("")
437
395
 
438
396
        self._add_parents_or_children(revision.parent_ids,
439
397
                                      self.parents_widgets,
440
398
                                      self.parents_table)
441
 
 
 
399
        
442
400
        file_info = revision.properties.get('file-info', None)
443
401
        if file_info is not None:
444
 
            try:
445
 
                file_info = bdecode(file_info.encode('UTF-8'))
446
 
            except ValueError:
447
 
                trace.note('Invalid per-file info for revision:%s, value: %r',
448
 
                           revision.revision_id, file_info)
449
 
                file_info = None
 
402
            file_info = bdecode(file_info.encode('UTF-8'))
450
403
 
451
404
        if file_info:
452
405
            if self._file_id is None:
477
430
        self._add_tags()
478
431
 
479
432
    def _update_signature(self, widget, param):
480
 
        if self.get_current_page() == PAGE_SIGNATURE:
481
 
            self.signature_table.set_revision(self._revision)
 
433
        self.signature_table.set_revision(self._revision)
482
434
 
483
435
    def _update_bugs(self, widget, param):
484
436
        self.bugs_page.set_revision(self._revision)
490
442
                                      self.children_widgets,
491
443
                                      self.children_table)
492
444
 
493
 
    def _switch_page_cb(self, notebook, page, page_num):
494
 
        if page_num == PAGE_SIGNATURE:
495
 
            self.signature_table.set_revision(self._revision)
496
 
 
497
 
 
498
 
 
499
445
    def _show_clicked_cb(self, widget, revid, parentid):
500
446
        """Callback for when the show button for a parent is clicked."""
501
447
        self._show_callback(revid, parentid)
530
476
        table.resize(max(len(revids), 1), 2)
531
477
 
532
478
        for idx, revid in enumerate(revids):
533
 
            align = gtk.Alignment(0.0, 0.0, 1, 1)
 
479
            align = gtk.Alignment(0.0, 0.0)
534
480
            widgets.append(align)
535
481
            table.attach(align, 1, 2, idx, idx + 1,
536
482
                                      gtk.EXPAND | gtk.FILL, gtk.FILL)
553
499
                hbox.pack_start(button, expand=False, fill=True)
554
500
                button.show()
555
501
 
556
 
            button = gtk.Button()
557
 
            revid_label = gtk.Label(str(revid))
558
 
            revid_label.set_ellipsize(pango.ELLIPSIZE_MIDDLE)
559
 
            revid_label.set_alignment(0.0, 0.5)
560
 
            button.add(revid_label)
 
502
            button = gtk.Button(revid)
561
503
            button.connect("clicked",
562
 
                    lambda w, r: self.set_revision(self._repository.get_revision(r)), revid)
 
504
                    lambda w, r: self.set_revision(self._branch.repository.get_revision(r)), revid)
563
505
            button.set_use_underline(False)
564
 
            hbox.pack_start(button, expand=True, fill=True)
565
 
            button.show_all()
 
506
            hbox.pack_start(button, expand=False, fill=True)
 
507
            button.show()
566
508
 
567
509
    def _create_general(self):
568
510
        vbox = gtk.VBox(False, 6)
581
523
        vbox.show()
582
524
 
583
525
    def _create_signature(self):
584
 
        self.signature_table = SignatureTab(self._repository)
 
526
        self.signature_table = SignatureTab(self._branch.repository)
585
527
        self.append_page(self.signature_table, tab_label=gtk.Label('Signature'))
586
528
        self.connect_after('notify::revision', self._update_signature)
587
529
 
588
530
    def _create_headers(self):
589
 
        self.avatarsbox = AvatarsBox()
590
 
        
591
531
        self.table = gtk.Table(rows=5, columns=2)
592
532
        self.table.set_row_spacings(6)
593
533
        self.table.set_col_spacings(6)
594
534
        self.table.show()
595
 
        
596
 
        self.avatarsbox.pack_start(self.table)
597
 
        self.avatarsbox.show()
598
 
 
599
 
        row = 0
600
 
 
 
535
 
 
536
        align = gtk.Alignment(1.0, 0.5)
601
537
        label = gtk.Label()
602
 
        label.set_alignment(1.0, 0.5)
603
538
        label.set_markup("<b>Revision Id:</b>")
604
 
        self.table.attach(label, 0, 1, row, row+1, gtk.FILL, gtk.FILL)
 
539
        align.add(label)
 
540
        self.table.attach(align, 0, 1, 0, 1, gtk.FILL, gtk.FILL)
 
541
        align.show()
605
542
        label.show()
606
543
 
 
544
        align = gtk.Alignment(0.0, 0.5)
607
545
        revision_id = gtk.Label()
608
 
        revision_id.set_ellipsize(pango.ELLIPSIZE_MIDDLE)
609
 
        revision_id.set_alignment(0.0, 0.5)
610
546
        revision_id.set_selectable(True)
611
547
        self.connect('notify::revision', 
612
548
                lambda w, p: revision_id.set_text(self._revision.revision_id))
613
 
        self.table.attach(revision_id, 1, 2, row, row+1, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
549
        align.add(revision_id)
 
550
        self.table.attach(align, 1, 2, 0, 1, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
551
        align.show()
614
552
        revision_id.show()
615
553
 
616
 
        row += 1
 
554
        align = gtk.Alignment(1.0, 0.5)
617
555
        self.author_label = gtk.Label()
618
 
        self.author_label.set_alignment(1.0, 0.5)
619
556
        self.author_label.set_markup("<b>Author:</b>")
620
 
        self.table.attach(self.author_label, 0, 1, row, row+1, gtk.FILL, gtk.FILL)
 
557
        align.add(self.author_label)
 
558
        self.table.attach(align, 0, 1, 1, 2, gtk.FILL, gtk.FILL)
 
559
        align.show()
621
560
        self.author_label.show()
622
561
 
 
562
        align = gtk.Alignment(0.0, 0.5)
623
563
        self.author = gtk.Label()
624
 
        self.author.set_ellipsize(pango.ELLIPSIZE_END)
625
 
        self.author.set_alignment(0.0, 0.5)
626
564
        self.author.set_selectable(True)
627
 
        self.table.attach(self.author, 1, 2, row, row+1, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
565
        align.add(self.author)
 
566
        self.table.attach(align, 1, 2, 1, 2, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
567
        align.show()
628
568
        self.author.show()
629
569
        self.author.hide()
630
570
 
631
 
        row += 1
 
571
        align = gtk.Alignment(1.0, 0.5)
632
572
        label = gtk.Label()
633
 
        label.set_alignment(1.0, 0.5)
634
573
        label.set_markup("<b>Committer:</b>")
635
 
        self.table.attach(label, 0, 1, row, row+1, gtk.FILL, gtk.FILL)
 
574
        align.add(label)
 
575
        self.table.attach(align, 0, 1, 2, 3, gtk.FILL, gtk.FILL)
 
576
        align.show()
636
577
        label.show()
637
578
 
 
579
        align = gtk.Alignment(0.0, 0.5)
638
580
        self.committer = gtk.Label()
639
 
        self.committer.set_ellipsize(pango.ELLIPSIZE_END)
640
 
        self.committer.set_alignment(0.0, 0.5)
641
581
        self.committer.set_selectable(True)
642
 
        self.table.attach(self.committer, 1, 2, row, row+1, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
582
        align.add(self.committer)
 
583
        self.table.attach(align, 1, 2, 2, 3, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
584
        align.show()
643
585
        self.committer.show()
644
586
 
645
 
        row += 1
 
587
        align = gtk.Alignment(0.0, 0.5)
 
588
        label = gtk.Label()
 
589
        label.set_markup("<b>Branch nick:</b>")
 
590
        align.add(label)
 
591
        self.table.attach(align, 0, 1, 3, 4, gtk.FILL, gtk.FILL)
 
592
        label.show()
 
593
        align.show()
 
594
 
 
595
        align = gtk.Alignment(0.0, 0.5)
646
596
        self.branchnick_label = gtk.Label()
647
 
        self.branchnick_label.set_alignment(1.0, 0.5)
648
 
        self.branchnick_label.set_markup("<b>Branch nick:</b>")
649
 
        self.table.attach(self.branchnick_label, 0, 1, row, row+1, gtk.FILL, gtk.FILL)
 
597
        self.branchnick_label.set_selectable(True)
 
598
        align.add(self.branchnick_label)
 
599
        self.table.attach(align, 1, 2, 3, 4, gtk.EXPAND | gtk.FILL, gtk.FILL)
650
600
        self.branchnick_label.show()
651
 
 
652
 
        self.branchnick = gtk.Label()
653
 
        self.branchnick.set_ellipsize(pango.ELLIPSIZE_MIDDLE)
654
 
        self.branchnick.set_alignment(0.0, 0.5)
655
 
        self.branchnick.set_selectable(True)
656
 
        self.table.attach(self.branchnick, 1, 2, row, row+1, gtk.EXPAND | gtk.FILL, gtk.FILL)
657
 
        self.branchnick.show()
658
 
 
659
 
        row += 1
 
601
        align.show()
 
602
 
 
603
        align = gtk.Alignment(1.0, 0.5)
660
604
        label = gtk.Label()
661
 
        label.set_alignment(1.0, 0.5)
662
605
        label.set_markup("<b>Timestamp:</b>")
663
 
        self.table.attach(label, 0, 1, row, row+1, gtk.FILL, gtk.FILL)
 
606
        align.add(label)
 
607
        self.table.attach(align, 0, 1, 4, 5, gtk.FILL, gtk.FILL)
 
608
        align.show()
664
609
        label.show()
665
610
 
 
611
        align = gtk.Alignment(0.0, 0.5)
666
612
        self.timestamp = gtk.Label()
667
 
        self.timestamp.set_ellipsize(pango.ELLIPSIZE_END)
668
 
        self.timestamp.set_alignment(0.0, 0.5)
669
613
        self.timestamp.set_selectable(True)
670
 
        self.table.attach(self.timestamp, 1, 2, row, row+1, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
614
        align.add(self.timestamp)
 
615
        self.table.attach(align, 1, 2, 4, 5, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
616
        align.show()
671
617
        self.timestamp.show()
672
618
 
673
 
        row += 1
 
619
        align = gtk.Alignment(1.0, 0.5)
674
620
        self.tags_label = gtk.Label()
675
 
        self.tags_label.set_alignment(1.0, 0.5)
676
621
        self.tags_label.set_markup("<b>Tags:</b>")
677
 
        self.table.attach(self.tags_label, 0, 1, row, row+1, gtk.FILL, gtk.FILL)
 
622
        align.add(self.tags_label)
 
623
        align.show()
 
624
        self.table.attach(align, 0, 1, 5, 6, gtk.FILL, gtk.FILL)
678
625
        self.tags_label.show()
679
626
 
 
627
        align = gtk.Alignment(0.0, 0.5)
680
628
        self.tags_list = gtk.Label()
681
 
        self.tags_list.set_ellipsize(pango.ELLIPSIZE_MIDDLE)
682
 
        self.tags_list.set_alignment(0.0, 0.5)
683
 
        self.table.attach(self.tags_list, 1, 2, row, row+1, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
629
        align.add(self.tags_list)
 
630
        self.table.attach(align, 1, 2, 5, 6, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
631
        align.show()
684
632
        self.tags_list.show()
685
633
 
686
634
        self.connect('notify::revision', self._add_tags)
687
635
 
688
 
        self.avatarsbox.show()
689
 
        return self.avatarsbox
 
636
        return self.table
690
637
    
691
638
    def _create_parents(self):
692
639
        hbox = gtk.HBox(True, 3)