/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-03 20:30:15 UTC
  • mto: (450.1.18 trunk)
  • mto: This revision was merged to the branch mainline in revision 458.
  • Revision ID: daniel.schierbeck@gmail.com-20080403203015-ch8ck8gyjjhhiggf
Made the crypt code only use one DBus call to get key fields.

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
 
from gi.repository import Gtk
19
 
from gi.repository import Pango
20
 
from gi.repository import GObject
21
 
import webbrowser
 
18
import pygtk
 
19
pygtk.require("2.0")
 
20
import gtk
 
21
import pango
 
22
import gobject
 
23
import subprocess
22
24
 
23
 
from bzrlib import trace
 
25
from bzrlib.plugins.gtk import icon_path
24
26
from bzrlib.osutils import format_date
25
 
from bzrlib.bencode import bdecode
26
 
from bzrlib.testament import Testament
27
 
 
28
 
from bzrlib.plugins.gtk import icon_path
29
 
 
30
 
from bzrlib.plugins.gtk.avatarsbox import AvatarsBox
 
27
from bzrlib.util.bencode import bdecode
31
28
 
32
29
try:
33
30
    from bzrlib.plugins.gtk import seahorse
36
33
else:
37
34
    has_seahorse = True
38
35
 
39
 
PAGE_GENERAL = 0
40
 
PAGE_RELATIONS = 1
41
 
PAGE_SIGNATURE = 2
42
 
PAGE_BUGS = 3
43
 
 
44
 
 
45
36
def _open_link(widget, uri):
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):
 
37
    subprocess.Popen(['sensible-browser', uri], close_fds=True)
 
38
 
 
39
gtk.link_button_set_uri_hook(_open_link)
 
40
 
 
41
class BugsTab(gtk.Table):
53
42
 
54
43
    def __init__(self):
55
 
        super(BugsTab, self).__init__(homogeneous=False, spacing=6)
56
 
 
57
 
        table = Gtk.Table(rows=2, columns=2)
58
 
 
59
 
        table.set_row_spacings(6)
60
 
        table.set_col_spacing(0, 16)
61
 
 
62
 
        image = Gtk.Image()
63
 
        image.set_from_file(icon_path("bug.png"))
64
 
        table.attach(image, 0, 1, 0, 1, Gtk.AttachOptions.FILL)
65
 
 
66
 
        align = Gtk.Alignment.new(0.0, 0.1, 0, 0)
67
 
        self.label = Gtk.Label()
68
 
        align.add(self.label)
69
 
        table.attach(align, 1, 2, 0, 1, Gtk.AttachOptions.FILL)
70
 
 
71
 
        treeview = self.construct_treeview()
72
 
        table.attach(treeview, 1, 2, 1, 2, Gtk.AttachOptions.FILL | Gtk.AttachOptions.EXPAND)
73
 
 
74
 
        self.set_border_width(6)
75
 
        self.pack_start(table, False, True, 0)
76
 
 
77
 
        self.clear()
78
 
        self.show_all()
79
 
 
80
 
    def set_revision(self, revision):
81
 
        if revision is None:
82
 
            return
83
 
 
84
 
        self.clear()
85
 
        bugs_text = revision.properties.get('bugs', '')
86
 
        for bugline in bugs_text.splitlines():
87
 
                (url, status) = bugline.split(" ")
88
 
                if status == "fixed":
89
 
                    self.add_bug(url, status)
90
 
 
91
 
        if self.num_bugs == 0:
92
 
            return
93
 
        elif self.num_bugs == 1:
94
 
            label = "bug"
95
 
        else:
96
 
            label = "bugs"
97
 
 
98
 
        self.label.set_markup("<b>Bugs fixed</b>\n" +
99
 
                              "This revision claims to fix " +
100
 
                              "%d %s." % (self.num_bugs, label))
101
 
 
102
 
    def construct_treeview(self):
103
 
        self.bugs = Gtk.ListStore(GObject.TYPE_STRING, GObject.TYPE_STRING)
104
 
        self.treeview = Gtk.TreeView(model=self.bugs)
105
 
        self.treeview.set_headers_visible(False)
106
 
 
107
 
        uri_column = Gtk.TreeViewColumn('Bug URI', Gtk.CellRendererText(), text=0)
108
 
        self.treeview.append_column(uri_column)
109
 
 
110
 
        self.treeview.connect('row-activated', self.on_row_activated)
111
 
 
112
 
        win = Gtk.ScrolledWindow()
113
 
        win.set_policy(Gtk.PolicyType.AUTOMATIC, Gtk.PolicyType.AUTOMATIC)
114
 
        win.set_shadow_type(Gtk.ShadowType.IN)
115
 
        win.add(self.treeview)
116
 
 
117
 
        return win
 
44
        super(BugsTab, self).__init__(rows=5, columns=2)
 
45
        self.set_row_spacings(6)
 
46
        self.set_col_spacings(6)
 
47
        self.clear()
118
48
 
119
49
    def clear(self):
120
 
        self.num_bugs = 0
121
 
        self.bugs.clear()
122
 
        self.set_sensitive(False)
123
 
        self.label.set_markup("<b>No bugs fixed</b>\n" +
124
 
                              "This revision does not claim to fix any bugs.")
 
50
        for c in self.get_children():
 
51
            self.remove(c)
 
52
        self.count = 0
 
53
        self.hide_all() # Only shown when there are bugs
125
54
 
126
55
    def add_bug(self, url, status):
127
 
        self.num_bugs += 1
128
 
        self.bugs.append([url, status])
129
 
        self.set_sensitive(True)
130
 
 
131
 
    def get_num_bugs(self):
132
 
        return self.num_bugs
133
 
 
134
 
    def on_row_activated(self, treeview, path, column):
135
 
        uri = self.bugs.get_value(self.bugs.get_iter(path), 0)
136
 
        _open_link(self, uri)
137
 
 
138
 
 
139
 
class SignatureTab(Gtk.VBox):
 
56
        button = gtk.LinkButton(url, url)
 
57
        self.attach(button, 0, 1, self.count, self.count + 1,
 
58
                              gtk.EXPAND | gtk.FILL, gtk.FILL)
 
59
        status_label = gtk.Label(status)
 
60
        self.attach(status_label, 1, 2, self.count, self.count + 1,
 
61
                              gtk.EXPAND | gtk.FILL, gtk.FILL)
 
62
        self.count += 1
 
63
        self.show_all()
 
64
 
 
65
 
 
66
class SignatureTab(gtk.VBox):
140
67
 
141
68
    def __init__(self, repository):
142
69
        self.key = None
143
70
        self.revision = None
144
71
        self.repository = repository
145
72
 
146
 
        super(SignatureTab, self).__init__(homogeneous=False, spacing=6)
147
 
        signature_box = Gtk.Table(rows=3, columns=3)
 
73
        super(SignatureTab, self).__init__(False, 6)
 
74
        signature_box = gtk.Table(rows=3, columns=3)
148
75
        signature_box.set_col_spacing(0, 16)
149
76
        signature_box.set_col_spacing(1, 12)
150
77
        signature_box.set_row_spacings(6)
151
78
 
152
 
        self.signature_image = Gtk.Image()
153
 
        signature_box.attach(self.signature_image, 0, 1, 0, 1, Gtk.AttachOptions.FILL)
 
79
        self.signature_image = gtk.Image()
 
80
        signature_box.attach(self.signature_image, 0, 1, 0, 1, gtk.FILL)
154
81
 
155
 
        align = Gtk.Alignment.new(0.0, 0.1, 0.0, 0.0)
156
 
        self.signature_label = Gtk.Label()
 
82
        align = gtk.Alignment(0.0, 0.1)
 
83
        self.signature_label = gtk.Label()
157
84
        align.add(self.signature_label)
158
 
        signature_box.attach(align, 1, 3, 0, 1, Gtk.AttachOptions.FILL)
 
85
        signature_box.attach(align, 1, 3, 0, 1, gtk.FILL)
159
86
 
160
 
        align = Gtk.Alignment.new(0.0, 0.5, 0.0, 0.0)
161
 
        self.signature_key_id_label = Gtk.Label()
 
87
        align = gtk.Alignment(0.0, 0.5)
 
88
        self.signature_key_id_label = gtk.Label()
162
89
        self.signature_key_id_label.set_markup("<b>Key Id:</b>")
163
90
        align.add(self.signature_key_id_label)
164
 
        signature_box.attach(align, 1, 2, 1, 2, Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
 
91
        signature_box.attach(align, 1, 2, 1, 2, gtk.FILL, gtk.FILL)
165
92
 
166
 
        align = Gtk.Alignment.new(0.0, 0.5, 0.0, 0.0)
167
 
        self.signature_key_id = Gtk.Label()
 
93
        align = gtk.Alignment(0.0, 0.5)
 
94
        self.signature_key_id = gtk.Label()
168
95
        self.signature_key_id.set_selectable(True)
169
96
        align.add(self.signature_key_id)
170
 
        signature_box.attach(align, 2, 3, 1, 2, Gtk.AttachOptions.EXPAND | Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
 
97
        signature_box.attach(align, 2, 3, 1, 2, gtk.EXPAND | gtk.FILL, gtk.FILL)
171
98
 
172
 
        align = Gtk.Alignment.new(0.0, 0.5, 0.0, 0.0)
173
 
        self.signature_fingerprint_label = Gtk.Label()
 
99
        align = gtk.Alignment(0.0, 0.5)
 
100
        self.signature_fingerprint_label = gtk.Label()
174
101
        self.signature_fingerprint_label.set_markup("<b>Fingerprint:</b>")
175
102
        align.add(self.signature_fingerprint_label)
176
 
        signature_box.attach(align, 1, 2, 2, 3, Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
 
103
        signature_box.attach(align, 1, 2, 2, 3, gtk.FILL, gtk.FILL)
177
104
 
178
 
        align = Gtk.Alignment.new(0.0, 0.5, 0.0, 0.0)
179
 
        self.signature_fingerprint = Gtk.Label()
 
105
        align = gtk.Alignment(0.0, 0.5)
 
106
        self.signature_fingerprint = gtk.Label()
180
107
        self.signature_fingerprint.set_selectable(True)
181
108
        align.add(self.signature_fingerprint)
182
 
        signature_box.attach(align, 2, 3, 2, 3, Gtk.AttachOptions.EXPAND | Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
 
109
        signature_box.attach(align, 2, 3, 2, 3, gtk.EXPAND | gtk.FILL, gtk.FILL)
183
110
 
184
 
        align = Gtk.Alignment.new(0.0, 0.5, 0.0, 0.0)
185
 
        self.signature_trust_label = Gtk.Label()
 
111
        align = gtk.Alignment(0.0, 0.5)
 
112
        self.signature_trust_label = gtk.Label()
186
113
        self.signature_trust_label.set_markup("<b>Trust:</b>")
187
114
        align.add(self.signature_trust_label)
188
 
        signature_box.attach(align, 1, 2, 3, 4, Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
 
115
        signature_box.attach(align, 1, 2, 3, 4, gtk.FILL, gtk.FILL)
189
116
 
190
 
        align = Gtk.Alignment.new(0.0, 0.5, 0.0, 0.0)
191
 
        self.signature_trust = Gtk.Label()
 
117
        align = gtk.Alignment(0.0, 0.5)
 
118
        self.signature_trust = gtk.Label()
192
119
        self.signature_trust.set_selectable(True)
193
120
        align.add(self.signature_trust)
194
 
        signature_box.attach(align, 2, 3, 3, 4, Gtk.AttachOptions.EXPAND | Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
 
121
        signature_box.attach(align, 2, 3, 3, 4, gtk.EXPAND | gtk.FILL, gtk.FILL)
195
122
 
196
123
        self.set_border_width(6)
197
 
        self.pack_start(signature_box, False, True, 0)
 
124
        self.pack_start(signature_box, expand=False)
198
125
        self.show_all()
199
126
 
200
127
    def set_revision(self, revision):
222
149
                                        "This revision has not been signed.")
223
150
 
224
151
    def show_signature(self, 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
 
152
        key = seahorse.verify(crypttext)
236
153
 
237
154
        if key and key.is_available():
238
155
            if key.is_trusted():
284
201
        self.signature_trust.set_text('This key is ' + trust_text)
285
202
 
286
203
 
287
 
class RevisionView(Gtk.Notebook):
 
204
class RevisionView(gtk.Notebook):
288
205
    """ Custom widget for commit log details.
289
206
 
290
207
    A variety of bzr tools may need to implement such a thing. This is a
293
210
 
294
211
    __gproperties__ = {
295
212
        'branch': (
296
 
            GObject.TYPE_PYOBJECT,
 
213
            gobject.TYPE_PYOBJECT,
297
214
            'Branch',
298
215
            'The branch holding the revision being displayed',
299
 
            GObject.PARAM_CONSTRUCT_ONLY | GObject.PARAM_WRITABLE
 
216
            gobject.PARAM_CONSTRUCT_ONLY | gobject.PARAM_WRITABLE
300
217
        ),
301
218
 
302
219
        'revision': (
303
 
            GObject.TYPE_PYOBJECT,
 
220
            gobject.TYPE_PYOBJECT,
304
221
            'Revision',
305
222
            'The revision being displayed',
306
 
            GObject.PARAM_READWRITE
 
223
            gobject.PARAM_READWRITE
307
224
        ),
308
225
 
309
226
        'children': (
310
 
            GObject.TYPE_PYOBJECT,
 
227
            gobject.TYPE_PYOBJECT,
311
228
            'Children',
312
229
            'Child revisions',
313
 
            GObject.PARAM_READWRITE
 
230
            gobject.PARAM_READWRITE
314
231
        ),
315
232
 
316
233
        'file-id': (
317
 
            GObject.TYPE_PYOBJECT,
 
234
            gobject.TYPE_PYOBJECT,
318
235
            'File Id',
319
236
            'The file id',
320
 
            GObject.PARAM_READWRITE
 
237
            gobject.PARAM_READWRITE
321
238
        )
322
239
    }
323
240
 
324
 
    def __init__(self, branch=None, repository=None):
325
 
        super(RevisionView, self).__init__()
 
241
 
 
242
    def __init__(self, branch=None):
 
243
        gtk.Notebook.__init__(self)
326
244
 
327
245
        self._revision = None
328
246
        self._branch = branch
329
 
        if branch is not None:
330
 
            self._repository = branch.repository
331
 
        else:
332
 
            self._repository = repository
333
 
        self.signature_table = None
334
247
 
335
248
        self._create_general()
336
249
        self._create_relations()
337
 
        # Disabled because testaments aren't verified yet:
338
250
        if has_seahorse:
339
251
            self._create_signature()
340
252
        self._create_file_info_view()
341
253
        self._create_bugs()
342
254
 
343
 
        self.set_current_page(PAGE_GENERAL)
344
 
        self.connect_after('switch-page', self._switch_page_cb)
 
255
        self.set_current_page(0)
345
256
        
346
257
        self._show_callback = None
347
258
        self._clicked_callback = None
397
308
 
398
309
    def _set_revision(self, revision):
399
310
        if revision is None: return
400
 
        
401
 
        self.avatarsbox.reset()
402
 
        
 
311
 
403
312
        self._revision = revision
404
313
        if revision.committer is not None:
405
314
            self.committer.set_text(revision.committer)
406
 
            self.avatarsbox.add(revision.committer, "committer")
407
315
        else:
408
316
            self.committer.set_text("")
409
 
            self.avatarsbox.hide()
410
317
        author = revision.properties.get('author', '')
411
 
        self.avatarsbox.merge(revision.get_apparent_authors(), "author")
412
318
        if author != '':
413
319
            self.author.set_text(author)
414
320
            self.author.show()
421
327
            self.timestamp.set_text(format_date(revision.timestamp,
422
328
                                                revision.timezone))
423
329
        try:
424
 
            self.branchnick.show()
425
 
            self.branchnick_label.show()
426
 
            self.branchnick.set_text(revision.properties['branch-nick'])
 
330
            self.branchnick_label.set_text(revision.properties['branch-nick'])
427
331
        except KeyError:
428
 
            self.branchnick.hide()
429
 
            self.branchnick_label.hide()
 
332
            self.branchnick_label.set_text("")
430
333
 
431
334
        self._add_parents_or_children(revision.parent_ids,
432
335
                                      self.parents_widgets,
433
336
                                      self.parents_table)
434
 
 
 
337
        
435
338
        file_info = revision.properties.get('file-info', None)
436
339
        if file_info is not None:
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
 
340
            file_info = bdecode(file_info.encode('UTF-8'))
443
341
 
444
342
        if file_info:
445
343
            if self._file_id is None:
461
359
        else:
462
360
            self.file_info_box.hide()
463
361
 
 
362
        self.bugs_table.clear()
 
363
        bugs_text = revision.properties.get('bugs', None)
 
364
        if bugs_text:
 
365
            for bugline in bugs_text.splitlines():
 
366
                (url, status) = bugline.split(" ")
 
367
                self.bugs_table.add_bug(url, status)
 
368
 
464
369
    def update_tags(self):
465
370
        if self._branch is not None and self._branch.supports_tags():
466
371
            self._tagdict = self._branch.tags.get_reverse_tag_dict()
470
375
        self._add_tags()
471
376
 
472
377
    def _update_signature(self, widget, param):
473
 
        if not has_seahorse:
474
 
            return
475
 
        if self.get_current_page() == PAGE_SIGNATURE:
476
 
            self.signature_table.set_revision(self._revision)
477
 
 
478
 
    def _update_bugs(self, widget, param):
479
 
        self.bugs_page.set_revision(self._revision)
480
 
        label = self.get_tab_label(self.bugs_page)
481
 
        label.set_sensitive(self.bugs_page.get_num_bugs() != 0)
 
378
        self.signature_table.set_revision(self._revision)
482
379
 
483
380
    def set_children(self, children):
484
381
        self._add_parents_or_children(children,
485
382
                                      self.children_widgets,
486
383
                                      self.children_table)
487
384
 
488
 
    def _switch_page_cb(self, notebook, page, page_num):
489
 
        if not has_seahorse:
490
 
            return
491
 
        if page_num == PAGE_SIGNATURE:
492
 
            self.signature_table.set_revision(self._revision)
493
 
 
494
 
 
495
 
 
496
385
    def _show_clicked_cb(self, widget, revid, parentid):
497
386
        """Callback for when the show button for a parent is clicked."""
498
387
        self._show_callback(revid, parentid)
527
416
        table.resize(max(len(revids), 1), 2)
528
417
 
529
418
        for idx, revid in enumerate(revids):
530
 
            align = Gtk.Alignment.new(0.0, 0.0, 1, 1)
 
419
            align = gtk.Alignment(0.0, 0.0)
531
420
            widgets.append(align)
532
421
            table.attach(align, 1, 2, idx, idx + 1,
533
 
                                      Gtk.AttachOptions.EXPAND | Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
 
422
                                      gtk.EXPAND | gtk.FILL, gtk.FILL)
534
423
            align.show()
535
424
 
536
 
            hbox = Gtk.HBox(homogeneous=False, spacing=6)
 
425
            hbox = gtk.HBox(False, spacing=6)
537
426
            align.add(hbox)
538
427
            hbox.show()
539
428
 
540
 
            image = Gtk.Image()
 
429
            image = gtk.Image()
541
430
            image.set_from_stock(
542
 
                Gtk.STOCK_FIND, Gtk.IconSize.SMALL_TOOLBAR)
 
431
                gtk.STOCK_FIND, gtk.ICON_SIZE_SMALL_TOOLBAR)
543
432
            image.show()
544
433
 
545
434
            if self._show_callback is not None:
546
 
                button = Gtk.Button()
 
435
                button = gtk.Button()
547
436
                button.add(image)
548
437
                button.connect("clicked", self._show_clicked_cb,
549
438
                               self._revision.revision_id, revid)
550
 
                hbox.pack_start(button, False, True, 0)
 
439
                hbox.pack_start(button, expand=False, fill=True)
551
440
                button.show()
552
441
 
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)
 
442
            button = gtk.Button(revid)
558
443
            button.connect("clicked",
559
 
                    lambda w, r: self.set_revision(
560
 
                        self._repository.get_revision(r)), revid)
 
444
                    lambda w, r: self.set_revision(self._branch.repository.get_revision(r)), revid)
561
445
            button.set_use_underline(False)
562
 
            hbox.pack_start(button, True, True, 0)
563
 
            button.show_all()
 
446
            hbox.pack_start(button, expand=False, fill=True)
 
447
            button.show()
564
448
 
565
449
    def _create_general(self):
566
 
        vbox = Gtk.VBox(homogeneous=False, spacing=6)
 
450
        vbox = gtk.VBox(False, 6)
567
451
        vbox.set_border_width(6)
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"))
 
452
        vbox.pack_start(self._create_headers(), expand=False, fill=True)
 
453
        vbox.pack_start(self._create_message_view())
 
454
        self.append_page(vbox, tab_label=gtk.Label("General"))
571
455
        vbox.show()
572
456
 
573
457
    def _create_relations(self):
574
 
        vbox = Gtk.VBox(homogeneous=False, spacing=6)
 
458
        vbox = gtk.VBox(False, 6)
575
459
        vbox.set_border_width(6)
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"))
 
460
        vbox.pack_start(self._create_parents(), expand=False, fill=True)
 
461
        vbox.pack_start(self._create_children(), expand=False, fill=True)
 
462
        self.append_page(vbox, tab_label=gtk.Label("Relations"))
579
463
        vbox.show()
580
464
 
581
465
    def _create_signature(self):
582
 
        self.signature_table = SignatureTab(self._repository)
583
 
        self.append_page(
584
 
            self.signature_table, Gtk.Label(label='Signature'))
 
466
        self.signature_table = SignatureTab(self._branch.repository)
 
467
        self.append_page(self.signature_table, tab_label=gtk.Label('Signature'))
585
468
        self.connect_after('notify::revision', self._update_signature)
586
469
 
587
470
    def _create_headers(self):
588
 
        self.avatarsbox = AvatarsBox()
589
 
        
590
 
        self.table = Gtk.Table(rows=5, columns=2)
 
471
        self.table = gtk.Table(rows=5, columns=2)
591
472
        self.table.set_row_spacings(6)
592
473
        self.table.set_col_spacings(6)
593
474
        self.table.show()
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)
 
475
 
 
476
        align = gtk.Alignment(1.0, 0.5)
 
477
        label = gtk.Label()
602
478
        label.set_markup("<b>Revision Id:</b>")
603
 
        self.table.attach(label, 0, 1, row, row+1, Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
 
479
        align.add(label)
 
480
        self.table.attach(align, 0, 1, 0, 1, gtk.FILL, gtk.FILL)
 
481
        align.show()
604
482
        label.show()
605
483
 
606
 
        revision_id = Gtk.Label()
607
 
        revision_id.set_ellipsize(Pango.EllipsizeMode.MIDDLE)
608
 
        revision_id.set_alignment(0.0, 0.5)
 
484
        align = gtk.Alignment(0.0, 0.5)
 
485
        revision_id = gtk.Label()
609
486
        revision_id.set_selectable(True)
610
487
        self.connect('notify::revision', 
611
488
                lambda w, p: revision_id.set_text(self._revision.revision_id))
612
 
        self.table.attach(revision_id, 1, 2, row, row+1, Gtk.AttachOptions.EXPAND | Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
 
489
        align.add(revision_id)
 
490
        self.table.attach(align, 1, 2, 0, 1, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
491
        align.show()
613
492
        revision_id.show()
614
493
 
615
 
        row += 1
616
 
        self.author_label = Gtk.Label()
617
 
        self.author_label.set_alignment(1.0, 0.5)
 
494
        align = gtk.Alignment(1.0, 0.5)
 
495
        self.author_label = gtk.Label()
618
496
        self.author_label.set_markup("<b>Author:</b>")
619
 
        self.table.attach(self.author_label, 0, 1, row, row+1, Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
 
497
        align.add(self.author_label)
 
498
        self.table.attach(align, 0, 1, 1, 2, gtk.FILL, gtk.FILL)
 
499
        align.show()
620
500
        self.author_label.show()
621
501
 
622
 
        self.author = Gtk.Label()
623
 
        self.author.set_ellipsize(Pango.EllipsizeMode.END)
624
 
        self.author.set_alignment(0.0, 0.5)
 
502
        align = gtk.Alignment(0.0, 0.5)
 
503
        self.author = gtk.Label()
625
504
        self.author.set_selectable(True)
626
 
        self.table.attach(self.author, 1, 2, row, row+1, Gtk.AttachOptions.EXPAND | Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
 
505
        align.add(self.author)
 
506
        self.table.attach(align, 1, 2, 1, 2, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
507
        align.show()
627
508
        self.author.show()
628
509
        self.author.hide()
629
510
 
630
 
        row += 1
631
 
        label = Gtk.Label()
632
 
        label.set_alignment(1.0, 0.5)
 
511
        align = gtk.Alignment(1.0, 0.5)
 
512
        label = gtk.Label()
633
513
        label.set_markup("<b>Committer:</b>")
634
 
        self.table.attach(label, 0, 1, row, row+1, Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
 
514
        align.add(label)
 
515
        self.table.attach(align, 0, 1, 2, 3, gtk.FILL, gtk.FILL)
 
516
        align.show()
635
517
        label.show()
636
518
 
637
 
        self.committer = Gtk.Label()
638
 
        self.committer.set_ellipsize(Pango.EllipsizeMode.END)
639
 
        self.committer.set_alignment(0.0, 0.5)
 
519
        align = gtk.Alignment(0.0, 0.5)
 
520
        self.committer = gtk.Label()
640
521
        self.committer.set_selectable(True)
641
 
        self.table.attach(self.committer, 1, 2, row, row+1, Gtk.AttachOptions.EXPAND | Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
 
522
        align.add(self.committer)
 
523
        self.table.attach(align, 1, 2, 2, 3, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
524
        align.show()
642
525
        self.committer.show()
643
526
 
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)
 
527
        align = gtk.Alignment(0.0, 0.5)
 
528
        label = gtk.Label()
 
529
        label.set_markup("<b>Branch nick:</b>")
 
530
        align.add(label)
 
531
        self.table.attach(align, 0, 1, 3, 4, gtk.FILL, gtk.FILL)
 
532
        label.show()
 
533
        align.show()
 
534
 
 
535
        align = gtk.Alignment(0.0, 0.5)
 
536
        self.branchnick_label = gtk.Label()
 
537
        self.branchnick_label.set_selectable(True)
 
538
        align.add(self.branchnick_label)
 
539
        self.table.attach(align, 1, 2, 3, 4, gtk.EXPAND | gtk.FILL, gtk.FILL)
649
540
        self.branchnick_label.show()
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)
 
541
        align.show()
 
542
 
 
543
        align = gtk.Alignment(1.0, 0.5)
 
544
        label = gtk.Label()
661
545
        label.set_markup("<b>Timestamp:</b>")
662
 
        self.table.attach(label, 0, 1, row, row+1, Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
 
546
        align.add(label)
 
547
        self.table.attach(align, 0, 1, 4, 5, gtk.FILL, gtk.FILL)
 
548
        align.show()
663
549
        label.show()
664
550
 
665
 
        self.timestamp = Gtk.Label()
666
 
        self.timestamp.set_ellipsize(Pango.EllipsizeMode.END)
667
 
        self.timestamp.set_alignment(0.0, 0.5)
 
551
        align = gtk.Alignment(0.0, 0.5)
 
552
        self.timestamp = gtk.Label()
668
553
        self.timestamp.set_selectable(True)
669
 
        self.table.attach(self.timestamp, 1, 2, row, row+1, Gtk.AttachOptions.EXPAND | Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
 
554
        align.add(self.timestamp)
 
555
        self.table.attach(align, 1, 2, 4, 5, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
556
        align.show()
670
557
        self.timestamp.show()
671
558
 
672
 
        row += 1
673
 
        self.tags_label = Gtk.Label()
674
 
        self.tags_label.set_alignment(1.0, 0.5)
 
559
        align = gtk.Alignment(1.0, 0.5)
 
560
        self.tags_label = gtk.Label()
675
561
        self.tags_label.set_markup("<b>Tags:</b>")
676
 
        self.table.attach(self.tags_label, 0, 1, row, row+1, Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
 
562
        align.add(self.tags_label)
 
563
        align.show()
 
564
        self.table.attach(align, 0, 1, 5, 6, gtk.FILL, gtk.FILL)
677
565
        self.tags_label.show()
678
566
 
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)
 
567
        align = gtk.Alignment(0.0, 0.5)
 
568
        self.tags_list = gtk.Label()
 
569
        align.add(self.tags_list)
 
570
        self.table.attach(align, 1, 2, 5, 6, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
571
        align.show()
683
572
        self.tags_list.show()
684
573
 
685
574
        self.connect('notify::revision', self._add_tags)
686
575
 
687
 
        return self.avatarsbox
 
576
        return self.table
688
577
    
689
578
    def _create_parents(self):
690
 
        hbox = Gtk.HBox(homogeneous=True, spacing=3)
 
579
        hbox = gtk.HBox(True, 3)
691
580
        
692
581
        self.parents_table = self._create_parents_or_children_table(
693
582
            "<b>Parents:</b>")
694
583
        self.parents_widgets = []
695
 
        hbox.pack_start(self.parents_table, True, True, 0)
 
584
        hbox.pack_start(self.parents_table)
696
585
 
697
586
        hbox.show()
698
587
        return hbox
699
588
 
700
589
    def _create_children(self):
701
 
        hbox = Gtk.HBox(homogeneous=True, spacing=3)
 
590
        hbox = gtk.HBox(True, 3)
702
591
        self.children_table = self._create_parents_or_children_table(
703
592
            "<b>Children:</b>")
704
593
        self.children_widgets = []
705
 
        hbox.pack_start(self.children_table, True, True, 0)
 
594
        hbox.pack_start(self.children_table)
706
595
        hbox.show()
707
596
        return hbox
708
597
        
709
598
    def _create_parents_or_children_table(self, text):
710
 
        table = Gtk.Table(rows=1, columns=2)
 
599
        table = gtk.Table(rows=1, columns=2)
711
600
        table.set_row_spacings(3)
712
601
        table.set_col_spacings(6)
713
602
        table.show()
714
603
 
715
 
        label = Gtk.Label()
 
604
        label = gtk.Label()
716
605
        label.set_markup(text)
717
 
        align = Gtk.Alignment.new(0.0, 0.5, 0, 0)
 
606
        align = gtk.Alignment(0.0, 0.5)
718
607
        align.add(label)
719
 
        table.attach(align, 0, 1, 0, 1, Gtk.AttachOptions.FILL, Gtk.AttachOptions.FILL)
 
608
        table.attach(align, 0, 1, 0, 1, gtk.FILL, gtk.FILL)
720
609
        label.show()
721
610
        align.show()
722
611
 
723
612
        return table
724
613
 
725
614
    def _create_message_view(self):
726
 
        msg_buffer = Gtk.TextBuffer()
 
615
        msg_buffer = gtk.TextBuffer()
727
616
        self.connect('notify::revision',
728
617
                lambda w, p: msg_buffer.set_text(self._revision.message))
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)
 
618
        window = gtk.ScrolledWindow()
 
619
        window.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
 
620
        window.set_shadow_type(gtk.SHADOW_IN)
 
621
        tv = gtk.TextView(msg_buffer)
733
622
        tv.set_editable(False)
734
 
        tv.set_wrap_mode(Gtk.WrapMode.WORD)
 
623
        tv.set_wrap_mode(gtk.WRAP_WORD)
735
624
 
736
 
        tv.modify_font(Pango.FontDescription("Monospace"))
 
625
        tv.modify_font(pango.FontDescription("Monospace"))
737
626
        tv.show()
738
627
        window.add(tv)
739
628
        window.show()
740
629
        return window
741
630
 
742
631
    def _create_bugs(self):
743
 
        self.bugs_page = BugsTab()
744
 
        self.connect_after('notify::revision', self._update_bugs) 
745
 
        self.append_page(self.bugs_page, Gtk.Label(label='Bugs'))
 
632
        self.bugs_table = BugsTab()
 
633
        self.append_page(self.bugs_table, tab_label=gtk.Label('Bugs'))
746
634
 
747
635
    def _create_file_info_view(self):
748
 
        self.file_info_box = Gtk.VBox(homogeneous=False, spacing=6)
 
636
        self.file_info_box = gtk.VBox(False, 6)
749
637
        self.file_info_box.set_border_width(6)
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)
 
638
        self.file_info_buffer = gtk.TextBuffer()
 
639
        window = gtk.ScrolledWindow()
 
640
        window.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
 
641
        window.set_shadow_type(gtk.SHADOW_IN)
 
642
        tv = gtk.TextView(self.file_info_buffer)
755
643
        tv.set_editable(False)
756
 
        tv.set_wrap_mode(Gtk.WrapMode.WORD)
757
 
        tv.modify_font(Pango.FontDescription("Monospace"))
 
644
        tv.set_wrap_mode(gtk.WRAP_WORD)
 
645
        tv.modify_font(pango.FontDescription("Monospace"))
758
646
        tv.show()
759
647
        window.add(tv)
760
648
        window.show()
761
 
        self.file_info_box.pack_start(window, True, True, 0)
 
649
        self.file_info_box.pack_start(window)
762
650
        self.file_info_box.hide() # Only shown when there are per-file messages
763
 
        self.append_page(self.file_info_box, Gtk.Label(label='Per-file'))
 
651
        self.append_page(self.file_info_box, tab_label=gtk.Label('Per-file'))
764
652