/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: Jelmer Vernooij
  • Date: 2012-07-09 15:23:26 UTC
  • mto: This revision was merged to the branch mainline in revision 794.
  • Revision ID: jelmer@samba.org-20120709152326-dzxb8zoz0btull7n
Remove bzr-notify.

Show diffs side-by-side

added added

removed removed

Lines of Context:
15
15
# along with this program; if not, write to the Free Software
16
16
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
17
17
 
18
 
import pygtk
19
 
pygtk.require("2.0")
20
 
import gtk
21
 
import pango
22
 
import gobject
23
 
import subprocess
 
18
from gi.repository import Gtk
 
19
from gi.repository import Pango
 
20
from gi.repository import GObject
 
21
import webbrowser
 
22
 
 
23
from bzrlib import trace
 
24
from bzrlib.osutils import format_date
 
25
from bzrlib.bencode import bdecode
 
26
from bzrlib.testament import Testament
24
27
 
25
28
from bzrlib.plugins.gtk import icon_path
26
 
from bzrlib.osutils import format_date
27
 
from bzrlib.util.bencode import bdecode
 
29
 
 
30
from bzrlib.plugins.gtk.avatarsbox import AvatarsBox
28
31
 
29
32
try:
30
33
    from bzrlib.plugins.gtk import seahorse
38
41
PAGE_SIGNATURE = 2
39
42
PAGE_BUGS = 3
40
43
 
 
44
 
41
45
def _open_link(widget, uri):
42
 
    subprocess.Popen(['sensible-browser', uri], close_fds=True)
43
 
 
44
 
gtk.link_button_set_uri_hook(_open_link)
45
 
 
46
 
class BugsTab(gtk.VBox):
 
46
    for cmd in ['sensible-browser', 'xdg-open']:
 
47
        if webbrowser._iscommand(cmd):
 
48
            webbrowser._tryorder.insert(0, '%s "%%s"' % cmd)
 
49
    webbrowser.open(uri)
 
50
 
 
51
 
 
52
class BugsTab(Gtk.VBox):
47
53
 
48
54
    def __init__(self):
49
 
        super(BugsTab, self).__init__(False, 6)
50
 
    
51
 
        table = gtk.Table(rows=2, columns=2)
 
55
        super(BugsTab, self).__init__(homogeneous=False, spacing=6)
 
56
 
 
57
        table = Gtk.Table(rows=2, columns=2)
52
58
 
53
59
        table.set_row_spacings(6)
54
60
        table.set_col_spacing(0, 16)
55
61
 
56
 
        image = gtk.Image()
 
62
        image = Gtk.Image()
57
63
        image.set_from_file(icon_path("bug.png"))
58
 
        table.attach(image, 0, 1, 0, 1, gtk.FILL)
 
64
        table.attach(image, 0, 1, 0, 1, Gtk.AttachOptions.FILL)
59
65
 
60
 
        align = gtk.Alignment(0.0, 0.1)
61
 
        self.label = gtk.Label()
 
66
        align = Gtk.Alignment.new(0.0, 0.1, 0, 0)
 
67
        self.label = Gtk.Label()
62
68
        align.add(self.label)
63
 
        table.attach(align, 1, 2, 0, 1, gtk.FILL)
 
69
        table.attach(align, 1, 2, 0, 1, Gtk.AttachOptions.FILL)
64
70
 
65
71
        treeview = self.construct_treeview()
66
 
        table.attach(treeview, 1, 2, 1, 2, gtk.FILL | gtk.EXPAND)
 
72
        table.attach(treeview, 1, 2, 1, 2, Gtk.AttachOptions.FILL | Gtk.AttachOptions.EXPAND)
67
73
 
68
74
        self.set_border_width(6)
69
 
        self.pack_start(table, expand=False)
 
75
        self.pack_start(table, False, True, 0)
70
76
 
71
77
        self.clear()
72
78
        self.show_all()
81
87
                (url, status) = bugline.split(" ")
82
88
                if status == "fixed":
83
89
                    self.add_bug(url, status)
84
 
        
 
90
 
85
91
        if self.num_bugs == 0:
86
92
            return
87
93
        elif self.num_bugs == 1:
94
100
                              "%d %s." % (self.num_bugs, label))
95
101
 
96
102
    def construct_treeview(self):
97
 
        self.bugs = gtk.ListStore(gobject.TYPE_STRING, gobject.TYPE_STRING)
98
 
        self.treeview = gtk.TreeView(self.bugs)
 
103
        self.bugs = Gtk.ListStore(GObject.TYPE_STRING, GObject.TYPE_STRING)
 
104
        self.treeview = Gtk.TreeView(model=self.bugs)
99
105
        self.treeview.set_headers_visible(False)
100
106
 
101
 
        uri_column = gtk.TreeViewColumn('Bug URI', gtk.CellRendererText(), text=0)
 
107
        uri_column = Gtk.TreeViewColumn('Bug URI', Gtk.CellRendererText(), text=0)
102
108
        self.treeview.append_column(uri_column)
103
109
 
104
110
        self.treeview.connect('row-activated', self.on_row_activated)
105
111
 
106
 
        win = gtk.ScrolledWindow()
107
 
        win.set_policy(gtk.POLICY_AUTOMATIC, gtk.POLICY_AUTOMATIC)
108
 
        win.set_shadow_type(gtk.SHADOW_IN)
 
112
        win = Gtk.ScrolledWindow()
 
113
        win.set_policy(Gtk.PolicyType.AUTOMATIC, Gtk.PolicyType.AUTOMATIC)
 
114
        win.set_shadow_type(Gtk.ShadowType.IN)
109
115
        win.add(self.treeview)
110
116
 
111
117
        return win
130
136
        _open_link(self, uri)
131
137
 
132
138
 
133
 
class SignatureTab(gtk.VBox):
 
139
class SignatureTab(Gtk.VBox):
134
140
 
135
141
    def __init__(self, repository):
136
142
        self.key = None
137
143
        self.revision = None
138
144
        self.repository = repository
139
145
 
140
 
        super(SignatureTab, self).__init__(False, 6)
141
 
        signature_box = gtk.Table(rows=3, columns=3)
 
146
        super(SignatureTab, self).__init__(homogeneous=False, spacing=6)
 
147
        signature_box = Gtk.Table(rows=3, columns=3)
142
148
        signature_box.set_col_spacing(0, 16)
143
149
        signature_box.set_col_spacing(1, 12)
144
150
        signature_box.set_row_spacings(6)
145
151
 
146
 
        self.signature_image = gtk.Image()
147
 
        signature_box.attach(self.signature_image, 0, 1, 0, 1, gtk.FILL)
 
152
        self.signature_image = Gtk.Image()
 
153
        signature_box.attach(self.signature_image, 0, 1, 0, 1, Gtk.AttachOptions.FILL)
148
154
 
149
 
        align = gtk.Alignment(0.0, 0.1)
150
 
        self.signature_label = gtk.Label()
 
155
        align = Gtk.Alignment.new(0.0, 0.1, 0.0, 0.0)
 
156
        self.signature_label = Gtk.Label()
151
157
        align.add(self.signature_label)
152
 
        signature_box.attach(align, 1, 3, 0, 1, gtk.FILL)
 
158
        signature_box.attach(align, 1, 3, 0, 1, Gtk.AttachOptions.FILL)
153
159
 
154
 
        align = gtk.Alignment(0.0, 0.5)
155
 
        self.signature_key_id_label = gtk.Label()
 
160
        align = Gtk.Alignment.new(0.0, 0.5, 0.0, 0.0)
 
161
        self.signature_key_id_label = Gtk.Label()
156
162
        self.signature_key_id_label.set_markup("<b>Key Id:</b>")
157
163
        align.add(self.signature_key_id_label)
158
 
        signature_box.attach(align, 1, 2, 1, 2, gtk.FILL, gtk.FILL)
 
164
        signature_box.attach(align, 1, 2, 1, 2, Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
159
165
 
160
 
        align = gtk.Alignment(0.0, 0.5)
161
 
        self.signature_key_id = gtk.Label()
 
166
        align = Gtk.Alignment.new(0.0, 0.5, 0.0, 0.0)
 
167
        self.signature_key_id = Gtk.Label()
162
168
        self.signature_key_id.set_selectable(True)
163
169
        align.add(self.signature_key_id)
164
 
        signature_box.attach(align, 2, 3, 1, 2, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
170
        signature_box.attach(align, 2, 3, 1, 2, Gtk.AttachOptions.EXPAND | Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
165
171
 
166
 
        align = gtk.Alignment(0.0, 0.5)
167
 
        self.signature_fingerprint_label = gtk.Label()
 
172
        align = Gtk.Alignment.new(0.0, 0.5, 0.0, 0.0)
 
173
        self.signature_fingerprint_label = Gtk.Label()
168
174
        self.signature_fingerprint_label.set_markup("<b>Fingerprint:</b>")
169
175
        align.add(self.signature_fingerprint_label)
170
 
        signature_box.attach(align, 1, 2, 2, 3, gtk.FILL, gtk.FILL)
 
176
        signature_box.attach(align, 1, 2, 2, 3, Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
171
177
 
172
 
        align = gtk.Alignment(0.0, 0.5)
173
 
        self.signature_fingerprint = gtk.Label()
 
178
        align = Gtk.Alignment.new(0.0, 0.5, 0.0, 0.0)
 
179
        self.signature_fingerprint = Gtk.Label()
174
180
        self.signature_fingerprint.set_selectable(True)
175
181
        align.add(self.signature_fingerprint)
176
 
        signature_box.attach(align, 2, 3, 2, 3, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
182
        signature_box.attach(align, 2, 3, 2, 3, Gtk.AttachOptions.EXPAND | Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
177
183
 
178
 
        align = gtk.Alignment(0.0, 0.5)
179
 
        self.signature_trust_label = gtk.Label()
 
184
        align = Gtk.Alignment.new(0.0, 0.5, 0.0, 0.0)
 
185
        self.signature_trust_label = Gtk.Label()
180
186
        self.signature_trust_label.set_markup("<b>Trust:</b>")
181
187
        align.add(self.signature_trust_label)
182
 
        signature_box.attach(align, 1, 2, 3, 4, gtk.FILL, gtk.FILL)
 
188
        signature_box.attach(align, 1, 2, 3, 4, Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
183
189
 
184
 
        align = gtk.Alignment(0.0, 0.5)
185
 
        self.signature_trust = gtk.Label()
 
190
        align = Gtk.Alignment.new(0.0, 0.5, 0.0, 0.0)
 
191
        self.signature_trust = Gtk.Label()
186
192
        self.signature_trust.set_selectable(True)
187
193
        align.add(self.signature_trust)
188
 
        signature_box.attach(align, 2, 3, 3, 4, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
194
        signature_box.attach(align, 2, 3, 3, 4, Gtk.AttachOptions.EXPAND | Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
189
195
 
190
196
        self.set_border_width(6)
191
 
        self.pack_start(signature_box, expand=False)
 
197
        self.pack_start(signature_box, False, True, 0)
192
198
        self.show_all()
193
199
 
194
200
    def set_revision(self, revision):
216
222
                                        "This revision has not been signed.")
217
223
 
218
224
    def show_signature(self, crypttext):
219
 
        key = seahorse.verify(crypttext)
 
225
        (cleartext, key) = seahorse.verify(crypttext)
 
226
 
 
227
        assert cleartext is not None
 
228
 
 
229
        inv = self.repository.get_inventory(self.revision.revision_id)
 
230
        expected_testament = Testament(self.revision, inv).as_short_text()
 
231
        if expected_testament != cleartext:
 
232
            self.signature_image.set_from_file(icon_path("sign-bad.png"))
 
233
            self.signature_label.set_markup("<b>Signature does not match repository data</b>\n" +
 
234
                        "The signature plaintext is different from the expected testament plaintext.")
 
235
            return
220
236
 
221
237
        if key and key.is_available():
222
238
            if key.is_trusted():
268
284
        self.signature_trust.set_text('This key is ' + trust_text)
269
285
 
270
286
 
271
 
class RevisionView(gtk.Notebook):
 
287
class RevisionView(Gtk.Notebook):
272
288
    """ Custom widget for commit log details.
273
289
 
274
290
    A variety of bzr tools may need to implement such a thing. This is a
277
293
 
278
294
    __gproperties__ = {
279
295
        'branch': (
280
 
            gobject.TYPE_PYOBJECT,
 
296
            GObject.TYPE_PYOBJECT,
281
297
            'Branch',
282
298
            'The branch holding the revision being displayed',
283
 
            gobject.PARAM_CONSTRUCT_ONLY | gobject.PARAM_WRITABLE
 
299
            GObject.PARAM_CONSTRUCT_ONLY | GObject.PARAM_WRITABLE
284
300
        ),
285
301
 
286
302
        'revision': (
287
 
            gobject.TYPE_PYOBJECT,
 
303
            GObject.TYPE_PYOBJECT,
288
304
            'Revision',
289
305
            'The revision being displayed',
290
 
            gobject.PARAM_READWRITE
 
306
            GObject.PARAM_READWRITE
291
307
        ),
292
308
 
293
309
        'children': (
294
 
            gobject.TYPE_PYOBJECT,
 
310
            GObject.TYPE_PYOBJECT,
295
311
            'Children',
296
312
            'Child revisions',
297
 
            gobject.PARAM_READWRITE
 
313
            GObject.PARAM_READWRITE
298
314
        ),
299
315
 
300
316
        'file-id': (
301
 
            gobject.TYPE_PYOBJECT,
 
317
            GObject.TYPE_PYOBJECT,
302
318
            'File Id',
303
319
            'The file id',
304
 
            gobject.PARAM_READWRITE
 
320
            GObject.PARAM_READWRITE
305
321
        )
306
322
    }
307
323
 
308
324
    def __init__(self, branch=None, repository=None):
309
 
        gtk.Notebook.__init__(self)
 
325
        super(RevisionView, self).__init__()
310
326
 
311
327
        self._revision = None
312
328
        self._branch = branch
314
330
            self._repository = branch.repository
315
331
        else:
316
332
            self._repository = repository
 
333
        self.signature_table = None
317
334
 
318
335
        self._create_general()
319
336
        self._create_relations()
320
337
        # Disabled because testaments aren't verified yet:
321
 
        # if has_seahorse:
322
 
        #    self._create_signature()
 
338
        if has_seahorse:
 
339
            self._create_signature()
323
340
        self._create_file_info_view()
324
341
        self._create_bugs()
325
342
 
380
397
 
381
398
    def _set_revision(self, revision):
382
399
        if revision is None: return
383
 
 
 
400
        
 
401
        self.avatarsbox.reset()
 
402
        
384
403
        self._revision = revision
385
404
        if revision.committer is not None:
386
405
            self.committer.set_text(revision.committer)
 
406
            self.avatarsbox.add(revision.committer, "committer")
387
407
        else:
388
408
            self.committer.set_text("")
 
409
            self.avatarsbox.hide()
389
410
        author = revision.properties.get('author', '')
 
411
        self.avatarsbox.merge(revision.get_apparent_authors(), "author")
390
412
        if author != '':
391
413
            self.author.set_text(author)
392
414
            self.author.show()
399
421
            self.timestamp.set_text(format_date(revision.timestamp,
400
422
                                                revision.timezone))
401
423
        try:
402
 
            self.branchnick_label.set_text(revision.properties['branch-nick'])
 
424
            self.branchnick.show()
 
425
            self.branchnick_label.show()
 
426
            self.branchnick.set_text(revision.properties['branch-nick'])
403
427
        except KeyError:
404
 
            self.branchnick_label.set_text("")
 
428
            self.branchnick.hide()
 
429
            self.branchnick_label.hide()
405
430
 
406
431
        self._add_parents_or_children(revision.parent_ids,
407
432
                                      self.parents_widgets,
408
433
                                      self.parents_table)
409
 
        
 
434
 
410
435
        file_info = revision.properties.get('file-info', None)
411
436
        if file_info is not None:
412
 
            file_info = bdecode(file_info.encode('UTF-8'))
 
437
            try:
 
438
                file_info = bdecode(file_info.encode('UTF-8'))
 
439
            except ValueError:
 
440
                trace.note('Invalid per-file info for revision:%s, value: %r',
 
441
                           revision.revision_id, file_info)
 
442
                file_info = None
413
443
 
414
444
        if file_info:
415
445
            if self._file_id is None:
440
470
        self._add_tags()
441
471
 
442
472
    def _update_signature(self, widget, param):
 
473
        if not has_seahorse:
 
474
            return
443
475
        if self.get_current_page() == PAGE_SIGNATURE:
444
476
            self.signature_table.set_revision(self._revision)
445
477
 
454
486
                                      self.children_table)
455
487
 
456
488
    def _switch_page_cb(self, notebook, page, page_num):
 
489
        if not has_seahorse:
 
490
            return
457
491
        if page_num == PAGE_SIGNATURE:
458
492
            self.signature_table.set_revision(self._revision)
459
493
 
493
527
        table.resize(max(len(revids), 1), 2)
494
528
 
495
529
        for idx, revid in enumerate(revids):
496
 
            align = gtk.Alignment(0.0, 0.0)
 
530
            align = Gtk.Alignment.new(0.0, 0.0, 1, 1)
497
531
            widgets.append(align)
498
532
            table.attach(align, 1, 2, idx, idx + 1,
499
 
                                      gtk.EXPAND | gtk.FILL, gtk.FILL)
 
533
                                      Gtk.AttachOptions.EXPAND | Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
500
534
            align.show()
501
535
 
502
 
            hbox = gtk.HBox(False, spacing=6)
 
536
            hbox = Gtk.HBox(homogeneous=False, spacing=6)
503
537
            align.add(hbox)
504
538
            hbox.show()
505
539
 
506
 
            image = gtk.Image()
 
540
            image = Gtk.Image()
507
541
            image.set_from_stock(
508
 
                gtk.STOCK_FIND, gtk.ICON_SIZE_SMALL_TOOLBAR)
 
542
                Gtk.STOCK_FIND, Gtk.IconSize.SMALL_TOOLBAR)
509
543
            image.show()
510
544
 
511
545
            if self._show_callback is not None:
512
 
                button = gtk.Button()
 
546
                button = Gtk.Button()
513
547
                button.add(image)
514
548
                button.connect("clicked", self._show_clicked_cb,
515
549
                               self._revision.revision_id, revid)
516
 
                hbox.pack_start(button, expand=False, fill=True)
 
550
                hbox.pack_start(button, False, True, 0)
517
551
                button.show()
518
552
 
519
 
            button = gtk.Button(revid)
 
553
            button = Gtk.Button()
 
554
            revid_label = Gtk.Label(label=str(revid))
 
555
            revid_label.set_ellipsize(Pango.EllipsizeMode.MIDDLE)
 
556
            revid_label.set_alignment(0.0, 0.5)
 
557
            button.add(revid_label)
520
558
            button.connect("clicked",
521
 
                    lambda w, r: self.set_revision(self._repository.get_revision(r)), revid)
 
559
                    lambda w, r: self.set_revision(
 
560
                        self._repository.get_revision(r)), revid)
522
561
            button.set_use_underline(False)
523
 
            hbox.pack_start(button, expand=False, fill=True)
524
 
            button.show()
 
562
            hbox.pack_start(button, True, True, 0)
 
563
            button.show_all()
525
564
 
526
565
    def _create_general(self):
527
 
        vbox = gtk.VBox(False, 6)
 
566
        vbox = Gtk.VBox(homogeneous=False, spacing=6)
528
567
        vbox.set_border_width(6)
529
 
        vbox.pack_start(self._create_headers(), expand=False, fill=True)
530
 
        vbox.pack_start(self._create_message_view())
531
 
        self.append_page(vbox, tab_label=gtk.Label("General"))
 
568
        vbox.pack_start(self._create_headers(), False, True, 0)
 
569
        vbox.pack_start(self._create_message_view(), True, True, 0)
 
570
        self.append_page(vbox, Gtk.Label(label="General"))
532
571
        vbox.show()
533
572
 
534
573
    def _create_relations(self):
535
 
        vbox = gtk.VBox(False, 6)
 
574
        vbox = Gtk.VBox(homogeneous=False, spacing=6)
536
575
        vbox.set_border_width(6)
537
 
        vbox.pack_start(self._create_parents(), expand=False, fill=True)
538
 
        vbox.pack_start(self._create_children(), expand=False, fill=True)
539
 
        self.append_page(vbox, tab_label=gtk.Label("Relations"))
 
576
        vbox.pack_start(self._create_parents(), False, True, 0)
 
577
        vbox.pack_start(self._create_children(), False, True, 0)
 
578
        self.append_page(vbox, Gtk.Label(label="Relations"))
540
579
        vbox.show()
541
580
 
542
581
    def _create_signature(self):
543
582
        self.signature_table = SignatureTab(self._repository)
544
 
        self.append_page(self.signature_table, tab_label=gtk.Label('Signature'))
 
583
        self.append_page(
 
584
            self.signature_table, Gtk.Label(label='Signature'))
545
585
        self.connect_after('notify::revision', self._update_signature)
546
586
 
547
587
    def _create_headers(self):
548
 
        self.table = gtk.Table(rows=5, columns=2)
 
588
        self.avatarsbox = AvatarsBox()
 
589
        
 
590
        self.table = Gtk.Table(rows=5, columns=2)
549
591
        self.table.set_row_spacings(6)
550
592
        self.table.set_col_spacings(6)
551
593
        self.table.show()
552
 
 
553
 
        align = gtk.Alignment(1.0, 0.5)
554
 
        label = gtk.Label()
 
594
        
 
595
        self.avatarsbox.pack_start(self.table, True, True, 0)
 
596
        self.avatarsbox.show()
 
597
 
 
598
        row = 0
 
599
 
 
600
        label = Gtk.Label()
 
601
        label.set_alignment(1.0, 0.5)
555
602
        label.set_markup("<b>Revision Id:</b>")
556
 
        align.add(label)
557
 
        self.table.attach(align, 0, 1, 0, 1, gtk.FILL, gtk.FILL)
558
 
        align.show()
 
603
        self.table.attach(label, 0, 1, row, row+1, Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
559
604
        label.show()
560
605
 
561
 
        align = gtk.Alignment(0.0, 0.5)
562
 
        revision_id = gtk.Label()
 
606
        revision_id = Gtk.Label()
 
607
        revision_id.set_ellipsize(Pango.EllipsizeMode.MIDDLE)
 
608
        revision_id.set_alignment(0.0, 0.5)
563
609
        revision_id.set_selectable(True)
564
610
        self.connect('notify::revision', 
565
611
                lambda w, p: revision_id.set_text(self._revision.revision_id))
566
 
        align.add(revision_id)
567
 
        self.table.attach(align, 1, 2, 0, 1, gtk.EXPAND | gtk.FILL, gtk.FILL)
568
 
        align.show()
 
612
        self.table.attach(revision_id, 1, 2, row, row+1, Gtk.AttachOptions.EXPAND | Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
569
613
        revision_id.show()
570
614
 
571
 
        align = gtk.Alignment(1.0, 0.5)
572
 
        self.author_label = gtk.Label()
 
615
        row += 1
 
616
        self.author_label = Gtk.Label()
 
617
        self.author_label.set_alignment(1.0, 0.5)
573
618
        self.author_label.set_markup("<b>Author:</b>")
574
 
        align.add(self.author_label)
575
 
        self.table.attach(align, 0, 1, 1, 2, gtk.FILL, gtk.FILL)
576
 
        align.show()
 
619
        self.table.attach(self.author_label, 0, 1, row, row+1, Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
577
620
        self.author_label.show()
578
621
 
579
 
        align = gtk.Alignment(0.0, 0.5)
580
 
        self.author = gtk.Label()
 
622
        self.author = Gtk.Label()
 
623
        self.author.set_ellipsize(Pango.EllipsizeMode.END)
 
624
        self.author.set_alignment(0.0, 0.5)
581
625
        self.author.set_selectable(True)
582
 
        align.add(self.author)
583
 
        self.table.attach(align, 1, 2, 1, 2, gtk.EXPAND | gtk.FILL, gtk.FILL)
584
 
        align.show()
 
626
        self.table.attach(self.author, 1, 2, row, row+1, Gtk.AttachOptions.EXPAND | Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
585
627
        self.author.show()
586
628
        self.author.hide()
587
629
 
588
 
        align = gtk.Alignment(1.0, 0.5)
589
 
        label = gtk.Label()
 
630
        row += 1
 
631
        label = Gtk.Label()
 
632
        label.set_alignment(1.0, 0.5)
590
633
        label.set_markup("<b>Committer:</b>")
591
 
        align.add(label)
592
 
        self.table.attach(align, 0, 1, 2, 3, gtk.FILL, gtk.FILL)
593
 
        align.show()
 
634
        self.table.attach(label, 0, 1, row, row+1, Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
594
635
        label.show()
595
636
 
596
 
        align = gtk.Alignment(0.0, 0.5)
597
 
        self.committer = gtk.Label()
 
637
        self.committer = Gtk.Label()
 
638
        self.committer.set_ellipsize(Pango.EllipsizeMode.END)
 
639
        self.committer.set_alignment(0.0, 0.5)
598
640
        self.committer.set_selectable(True)
599
 
        align.add(self.committer)
600
 
        self.table.attach(align, 1, 2, 2, 3, gtk.EXPAND | gtk.FILL, gtk.FILL)
601
 
        align.show()
 
641
        self.table.attach(self.committer, 1, 2, row, row+1, Gtk.AttachOptions.EXPAND | Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
602
642
        self.committer.show()
603
643
 
604
 
        align = gtk.Alignment(0.0, 0.5)
605
 
        label = gtk.Label()
606
 
        label.set_markup("<b>Branch nick:</b>")
607
 
        align.add(label)
608
 
        self.table.attach(align, 0, 1, 3, 4, gtk.FILL, gtk.FILL)
609
 
        label.show()
610
 
        align.show()
611
 
 
612
 
        align = gtk.Alignment(0.0, 0.5)
613
 
        self.branchnick_label = gtk.Label()
614
 
        self.branchnick_label.set_selectable(True)
615
 
        align.add(self.branchnick_label)
616
 
        self.table.attach(align, 1, 2, 3, 4, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
644
        row += 1
 
645
        self.branchnick_label = Gtk.Label()
 
646
        self.branchnick_label.set_alignment(1.0, 0.5)
 
647
        self.branchnick_label.set_markup("<b>Branch nick:</b>")
 
648
        self.table.attach(self.branchnick_label, 0, 1, row, row+1, Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
617
649
        self.branchnick_label.show()
618
 
        align.show()
619
 
 
620
 
        align = gtk.Alignment(1.0, 0.5)
621
 
        label = gtk.Label()
 
650
 
 
651
        self.branchnick = Gtk.Label()
 
652
        self.branchnick.set_ellipsize(Pango.EllipsizeMode.MIDDLE)
 
653
        self.branchnick.set_alignment(0.0, 0.5)
 
654
        self.branchnick.set_selectable(True)
 
655
        self.table.attach(self.branchnick, 1, 2, row, row+1, Gtk.AttachOptions.EXPAND | Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
 
656
        self.branchnick.show()
 
657
 
 
658
        row += 1
 
659
        label = Gtk.Label()
 
660
        label.set_alignment(1.0, 0.5)
622
661
        label.set_markup("<b>Timestamp:</b>")
623
 
        align.add(label)
624
 
        self.table.attach(align, 0, 1, 4, 5, gtk.FILL, gtk.FILL)
625
 
        align.show()
 
662
        self.table.attach(label, 0, 1, row, row+1, Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
626
663
        label.show()
627
664
 
628
 
        align = gtk.Alignment(0.0, 0.5)
629
 
        self.timestamp = gtk.Label()
 
665
        self.timestamp = Gtk.Label()
 
666
        self.timestamp.set_ellipsize(Pango.EllipsizeMode.END)
 
667
        self.timestamp.set_alignment(0.0, 0.5)
630
668
        self.timestamp.set_selectable(True)
631
 
        align.add(self.timestamp)
632
 
        self.table.attach(align, 1, 2, 4, 5, gtk.EXPAND | gtk.FILL, gtk.FILL)
633
 
        align.show()
 
669
        self.table.attach(self.timestamp, 1, 2, row, row+1, Gtk.AttachOptions.EXPAND | Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
634
670
        self.timestamp.show()
635
671
 
636
 
        align = gtk.Alignment(1.0, 0.5)
637
 
        self.tags_label = gtk.Label()
 
672
        row += 1
 
673
        self.tags_label = Gtk.Label()
 
674
        self.tags_label.set_alignment(1.0, 0.5)
638
675
        self.tags_label.set_markup("<b>Tags:</b>")
639
 
        align.add(self.tags_label)
640
 
        align.show()
641
 
        self.table.attach(align, 0, 1, 5, 6, gtk.FILL, gtk.FILL)
 
676
        self.table.attach(self.tags_label, 0, 1, row, row+1, Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
642
677
        self.tags_label.show()
643
678
 
644
 
        align = gtk.Alignment(0.0, 0.5)
645
 
        self.tags_list = gtk.Label()
646
 
        align.add(self.tags_list)
647
 
        self.table.attach(align, 1, 2, 5, 6, gtk.EXPAND | gtk.FILL, gtk.FILL)
648
 
        align.show()
 
679
        self.tags_list = Gtk.Label()
 
680
        self.tags_list.set_ellipsize(Pango.EllipsizeMode.MIDDLE)
 
681
        self.tags_list.set_alignment(0.0, 0.5)
 
682
        self.table.attach(self.tags_list, 1, 2, row, row+1, Gtk.AttachOptions.EXPAND | Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
649
683
        self.tags_list.show()
650
684
 
651
685
        self.connect('notify::revision', self._add_tags)
652
686
 
653
 
        return self.table
 
687
        return self.avatarsbox
654
688
    
655
689
    def _create_parents(self):
656
 
        hbox = gtk.HBox(True, 3)
 
690
        hbox = Gtk.HBox(homogeneous=True, spacing=3)
657
691
        
658
692
        self.parents_table = self._create_parents_or_children_table(
659
693
            "<b>Parents:</b>")
660
694
        self.parents_widgets = []
661
 
        hbox.pack_start(self.parents_table)
 
695
        hbox.pack_start(self.parents_table, True, True, 0)
662
696
 
663
697
        hbox.show()
664
698
        return hbox
665
699
 
666
700
    def _create_children(self):
667
 
        hbox = gtk.HBox(True, 3)
 
701
        hbox = Gtk.HBox(homogeneous=True, spacing=3)
668
702
        self.children_table = self._create_parents_or_children_table(
669
703
            "<b>Children:</b>")
670
704
        self.children_widgets = []
671
 
        hbox.pack_start(self.children_table)
 
705
        hbox.pack_start(self.children_table, True, True, 0)
672
706
        hbox.show()
673
707
        return hbox
674
708
        
675
709
    def _create_parents_or_children_table(self, text):
676
 
        table = gtk.Table(rows=1, columns=2)
 
710
        table = Gtk.Table(rows=1, columns=2)
677
711
        table.set_row_spacings(3)
678
712
        table.set_col_spacings(6)
679
713
        table.show()
680
714
 
681
 
        label = gtk.Label()
 
715
        label = Gtk.Label()
682
716
        label.set_markup(text)
683
 
        align = gtk.Alignment(0.0, 0.5)
 
717
        align = Gtk.Alignment.new(0.0, 0.5, 0, 0)
684
718
        align.add(label)
685
 
        table.attach(align, 0, 1, 0, 1, gtk.FILL, gtk.FILL)
 
719
        table.attach(align, 0, 1, 0, 1, Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
686
720
        label.show()
687
721
        align.show()
688
722
 
689
723
        return table
690
724
 
691
725
    def _create_message_view(self):
692
 
        msg_buffer = gtk.TextBuffer()
 
726
        msg_buffer = Gtk.TextBuffer()
693
727
        self.connect('notify::revision',
694
728
                lambda w, p: msg_buffer.set_text(self._revision.message))
695
 
        window = gtk.ScrolledWindow()
696
 
        window.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
697
 
        window.set_shadow_type(gtk.SHADOW_IN)
698
 
        tv = gtk.TextView(msg_buffer)
 
729
        window = Gtk.ScrolledWindow()
 
730
        window.set_policy(Gtk.PolicyType.NEVER, Gtk.PolicyType.AUTOMATIC)
 
731
        window.set_shadow_type(Gtk.ShadowType.IN)
 
732
        tv = Gtk.TextView(buffer=msg_buffer)
699
733
        tv.set_editable(False)
700
 
        tv.set_wrap_mode(gtk.WRAP_WORD)
 
734
        tv.set_wrap_mode(Gtk.WrapMode.WORD)
701
735
 
702
 
        tv.modify_font(pango.FontDescription("Monospace"))
 
736
        tv.modify_font(Pango.FontDescription("Monospace"))
703
737
        tv.show()
704
738
        window.add(tv)
705
739
        window.show()
708
742
    def _create_bugs(self):
709
743
        self.bugs_page = BugsTab()
710
744
        self.connect_after('notify::revision', self._update_bugs) 
711
 
        self.append_page(self.bugs_page, tab_label=gtk.Label('Bugs'))
 
745
        self.append_page(self.bugs_page, Gtk.Label(label='Bugs'))
712
746
 
713
747
    def _create_file_info_view(self):
714
 
        self.file_info_box = gtk.VBox(False, 6)
 
748
        self.file_info_box = Gtk.VBox(homogeneous=False, spacing=6)
715
749
        self.file_info_box.set_border_width(6)
716
 
        self.file_info_buffer = gtk.TextBuffer()
717
 
        window = gtk.ScrolledWindow()
718
 
        window.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
719
 
        window.set_shadow_type(gtk.SHADOW_IN)
720
 
        tv = gtk.TextView(self.file_info_buffer)
 
750
        self.file_info_buffer = Gtk.TextBuffer()
 
751
        window = Gtk.ScrolledWindow()
 
752
        window.set_policy(Gtk.PolicyType.NEVER, Gtk.PolicyType.AUTOMATIC)
 
753
        window.set_shadow_type(Gtk.ShadowType.IN)
 
754
        tv = Gtk.TextView(buffer=self.file_info_buffer)
721
755
        tv.set_editable(False)
722
 
        tv.set_wrap_mode(gtk.WRAP_WORD)
723
 
        tv.modify_font(pango.FontDescription("Monospace"))
 
756
        tv.set_wrap_mode(Gtk.WrapMode.WORD)
 
757
        tv.modify_font(Pango.FontDescription("Monospace"))
724
758
        tv.show()
725
759
        window.add(tv)
726
760
        window.show()
727
 
        self.file_info_box.pack_start(window)
 
761
        self.file_info_box.pack_start(window, True, True, 0)
728
762
        self.file_info_box.hide() # Only shown when there are per-file messages
729
 
        self.append_page(self.file_info_box, tab_label=gtk.Label('Per-file'))
 
763
        self.append_page(self.file_info_box, Gtk.Label(label='Per-file'))
730
764