/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

Fix regressions in plugins tab.

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.plugins.gtk import icon_path
25
26
from bzrlib.osutils import format_date
26
27
from bzrlib.util.bencode import bdecode
27
 
from bzrlib.testament import Testament
28
 
 
29
 
from bzrlib.plugins.gtk import icon_path
30
 
 
31
 
try:
32
 
    from bzrlib.plugins.gtk import seahorse
33
 
except ImportError:
34
 
    has_seahorse = False
35
 
else:
36
 
    has_seahorse = True
37
 
 
38
 
PAGE_GENERAL = 0
39
 
PAGE_RELATIONS = 1
40
 
PAGE_SIGNATURE = 2
41
 
PAGE_BUGS = 3
42
 
 
43
28
 
44
29
def _open_link(widget, uri):
45
 
    for cmd in ['sensible-browser', 'xdg-open']:
46
 
        if webbrowser._iscommand(cmd):
47
 
            webbrowser._tryorder.insert(0, '%s "%%s"' % cmd)
48
 
    webbrowser.open(uri)
 
30
    subprocess.Popen(['sensible-browser', uri], close_fds=True)
49
31
 
50
32
gtk.link_button_set_uri_hook(_open_link)
51
33
 
52
 
class BugsTab(gtk.VBox):
53
 
 
 
34
class BugsTab(gtk.Table):
54
35
    def __init__(self):
55
 
        super(BugsTab, self).__init__(False, 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.FILL)
65
 
 
66
 
        align = gtk.Alignment(0.0, 0.1)
67
 
        self.label = gtk.Label()
68
 
        align.add(self.label)
69
 
        table.attach(align, 1, 2, 0, 1, gtk.FILL)
70
 
 
71
 
        treeview = self.construct_treeview()
72
 
        table.attach(treeview, 1, 2, 1, 2, gtk.FILL | gtk.EXPAND)
73
 
 
74
 
        self.set_border_width(6)
75
 
        self.pack_start(table, expand=False)
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(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.POLICY_AUTOMATIC, gtk.POLICY_AUTOMATIC)
114
 
        win.set_shadow_type(gtk.SHADOW_IN)
115
 
        win.add(self.treeview)
116
 
 
117
 
        return win
 
36
        super(BugsTab, self).__init__(rows=5, columns=2)
 
37
        self.set_row_spacings(6)
 
38
        self.set_col_spacings(6)
 
39
        self.clear()
118
40
 
119
41
    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.")
 
42
        for c in self.get_children():
 
43
            self.remove(c)
 
44
        self.count = 0
 
45
        self.hide_all() # Only shown when there are bugs
125
46
 
126
47
    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)
 
48
        button = gtk.LinkButton(url, url)
 
49
        self.attach(button, 0, 1, self.count, self.count + 1,
 
50
                              gtk.EXPAND | gtk.FILL, gtk.FILL)
 
51
        status_label = gtk.Label(status)
 
52
        self.attach(status_label, 1, 2, self.count, self.count + 1,
 
53
                              gtk.EXPAND | gtk.FILL, gtk.FILL)
 
54
        self.count += 1
 
55
        self.show_all()
137
56
 
138
57
 
139
58
class SignatureTab(gtk.VBox):
140
 
 
141
 
    def __init__(self, repository):
142
 
        self.key = None
143
 
        self.revision = None
144
 
        self.repository = repository
145
 
 
 
59
    def __init__(self):
 
60
        from gpg import GPGSubprocess
 
61
        self.gpg = GPGSubprocess()
146
62
        super(SignatureTab, self).__init__(False, 6)
147
 
        signature_box = gtk.Table(rows=3, columns=3)
148
 
        signature_box.set_col_spacing(0, 16)
149
 
        signature_box.set_col_spacing(1, 12)
150
 
        signature_box.set_row_spacings(6)
 
63
        signature_box = gtk.Table(rows=1, columns=2)
 
64
        signature_box.set_col_spacing(0, 12)
151
65
 
152
66
        self.signature_image = gtk.Image()
153
67
        signature_box.attach(self.signature_image, 0, 1, 0, 1, gtk.FILL)
154
68
 
155
 
        align = gtk.Alignment(0.0, 0.1)
156
69
        self.signature_label = gtk.Label()
157
 
        align.add(self.signature_label)
158
 
        signature_box.attach(align, 1, 3, 0, 1, gtk.FILL)
159
 
 
160
 
        align = gtk.Alignment(0.0, 0.5)
161
 
        self.signature_key_id_label = gtk.Label()
162
 
        self.signature_key_id_label.set_markup("<b>Key Id:</b>")
163
 
        align.add(self.signature_key_id_label)
164
 
        signature_box.attach(align, 1, 2, 1, 2, gtk.FILL, gtk.FILL)
 
70
        signature_box.attach(self.signature_label, 1, 2, 0, 1, gtk.FILL)
 
71
 
 
72
        signature_info = gtk.Table(rows=1, columns=2)
 
73
        signature_info.set_row_spacings(6)
 
74
        signature_info.set_col_spacings(6)
 
75
 
 
76
        align = gtk.Alignment(1.0, 0.5)
 
77
        label = gtk.Label()
 
78
        label.set_markup("<b>Key Id:</b>")
 
79
        align.add(label)
 
80
        signature_info.attach(align, 0, 1, 0, 1, gtk.FILL, gtk.FILL)
165
81
 
166
82
        align = gtk.Alignment(0.0, 0.5)
167
83
        self.signature_key_id = gtk.Label()
168
84
        self.signature_key_id.set_selectable(True)
169
85
        align.add(self.signature_key_id)
170
 
        signature_box.attach(align, 2, 3, 1, 2, gtk.EXPAND | gtk.FILL, gtk.FILL)
171
 
 
172
 
        align = gtk.Alignment(0.0, 0.5)
173
 
        self.signature_fingerprint_label = gtk.Label()
174
 
        self.signature_fingerprint_label.set_markup("<b>Fingerprint:</b>")
175
 
        align.add(self.signature_fingerprint_label)
176
 
        signature_box.attach(align, 1, 2, 2, 3, gtk.FILL, gtk.FILL)
177
 
 
178
 
        align = gtk.Alignment(0.0, 0.5)
179
 
        self.signature_fingerprint = gtk.Label()
180
 
        self.signature_fingerprint.set_selectable(True)
181
 
        align.add(self.signature_fingerprint)
182
 
        signature_box.attach(align, 2, 3, 2, 3, gtk.EXPAND | gtk.FILL, gtk.FILL)
183
 
 
184
 
        align = gtk.Alignment(0.0, 0.5)
185
 
        self.signature_trust_label = gtk.Label()
186
 
        self.signature_trust_label.set_markup("<b>Trust:</b>")
187
 
        align.add(self.signature_trust_label)
188
 
        signature_box.attach(align, 1, 2, 3, 4, gtk.FILL, gtk.FILL)
189
 
 
190
 
        align = gtk.Alignment(0.0, 0.5)
191
 
        self.signature_trust = gtk.Label()
192
 
        self.signature_trust.set_selectable(True)
193
 
        align.add(self.signature_trust)
194
 
        signature_box.attach(align, 2, 3, 3, 4, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
86
        signature_info.attach(align, 1, 2, 0, 1, gtk.EXPAND | gtk.FILL, gtk.FILL)
195
87
 
196
88
        self.set_border_width(6)
197
89
        self.pack_start(signature_box, expand=False)
 
90
        self.pack_start(signature_info, expand=False)
198
91
        self.show_all()
199
92
 
200
 
    def set_revision(self, revision):
201
 
        self.revision = revision
202
 
        revid = revision.revision_id
203
 
 
204
 
        if self.repository.has_signature_for_revision_id(revid):
205
 
            crypttext = self.repository.get_signature_text(revid)
206
 
            self.show_signature(crypttext)
207
 
        else:
208
 
            self.show_no_signature()
209
 
 
210
93
    def show_no_signature(self):
211
 
        self.signature_key_id_label.hide()
212
94
        self.signature_key_id.set_text("")
213
 
 
214
 
        self.signature_fingerprint_label.hide()
215
 
        self.signature_fingerprint.set_text("")
216
 
 
217
 
        self.signature_trust_label.hide()
218
 
        self.signature_trust.set_text("")
219
 
 
220
95
        self.signature_image.set_from_file(icon_path("sign-unknown.png"))
221
 
        self.signature_label.set_markup("<b>Authenticity unknown</b>\n" +
222
 
                                        "This revision has not been signed.")
223
 
 
224
 
    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
236
 
 
237
 
        if key and key.is_available():
238
 
            if key.is_trusted():
239
 
                if key.get_display_name() == self.revision.committer:
240
 
                    self.signature_image.set_from_file(icon_path("sign-ok.png"))
241
 
                    self.signature_label.set_markup("<b>Authenticity confirmed</b>\n" +
242
 
                                                    "This revision has been signed with " +
243
 
                                                    "a trusted key.")
244
 
                else:
245
 
                    self.signature_image.set_from_file(icon_path("sign-bad.png"))
246
 
                    self.signature_label.set_markup("<b>Authenticity cannot be confirmed</b>\n" +
247
 
                                                    "Revision committer is not the same as signer.")
248
 
            else:
249
 
                self.signature_image.set_from_file(icon_path("sign-bad.png"))
250
 
                self.signature_label.set_markup("<b>Authenticity cannot be confirmed</b>\n" +
251
 
                                                "This revision has been signed, but the " +
252
 
                                                "key is not trusted.")
 
96
        self.signature_label.set_text("This revision has not been signed.")
 
97
 
 
98
    def show_signature(self, text):
 
99
        signature = self.gpg.verify(text)
 
100
 
 
101
        if signature.key_id is not None:
 
102
            self.signature_key_id.set_text(signature.key_id)
 
103
 
 
104
        if signature.is_valid():
 
105
            self.signature_image.set_from_file(icon_path("sign-ok.png"))
 
106
            self.signature_label.set_text("This revision has been signed.")
253
107
        else:
254
 
            self.show_no_signature()
255
108
            self.signature_image.set_from_file(icon_path("sign-bad.png"))
256
 
            self.signature_label.set_markup("<b>Authenticity cannot be confirmed</b>\n" +
257
 
                                            "Signature key not available.")
258
 
            return
259
 
 
260
 
        trust = key.get_trust()
261
 
 
262
 
        if trust <= seahorse.TRUST_NEVER:
263
 
            trust_text = 'never trusted'
264
 
        elif trust == seahorse.TRUST_UNKNOWN:
265
 
            trust_text = 'not trusted'
266
 
        elif trust == seahorse.TRUST_MARGINAL:
267
 
            trust_text = 'marginally trusted'
268
 
        elif trust == seahorse.TRUST_FULL:
269
 
            trust_text = 'fully trusted'
270
 
        elif trust == seahorse.TRUST_ULTIMATE:
271
 
            trust_text = 'ultimately trusted'
272
 
 
273
 
        self.signature_key_id_label.show()
274
 
        self.signature_key_id.set_text(key.get_id())
275
 
 
276
 
        fingerprint = key.get_fingerprint()
277
 
        if fingerprint == "":
278
 
            fingerprint = '<span foreground="dim grey">N/A</span>'
279
 
 
280
 
        self.signature_fingerprint_label.show()
281
 
        self.signature_fingerprint.set_markup(fingerprint)
282
 
 
283
 
        self.signature_trust_label.show()
284
 
        self.signature_trust.set_text('This key is ' + trust_text)
 
109
            self.signature_label.set_text("This revision has been signed, " + 
 
110
                    "but the authenticity of the signature cannot be verified.")
285
111
 
286
112
 
287
113
class RevisionView(gtk.Notebook):
321
147
        )
322
148
    }
323
149
 
324
 
    def __init__(self, branch=None, repository=None):
 
150
 
 
151
    def __init__(self, branch=None):
325
152
        gtk.Notebook.__init__(self)
326
153
 
327
 
        self._revision = None
328
 
        self._branch = branch
329
 
        if branch is not None:
330
 
            self._repository = branch.repository
331
 
        else:
332
 
            self._repository = repository
333
 
 
334
154
        self._create_general()
335
155
        self._create_relations()
336
 
        # Disabled because testaments aren't verified yet:
337
 
        if has_seahorse:
338
 
            self._create_signature()
 
156
        self._create_signature()
339
157
        self._create_file_info_view()
340
158
        self._create_bugs()
341
159
 
342
 
        self.set_current_page(PAGE_GENERAL)
343
 
        self.connect_after('switch-page', self._switch_page_cb)
 
160
        self.set_current_page(0)
344
161
        
345
162
        self._show_callback = None
346
163
        self._clicked_callback = None
447
264
        else:
448
265
            self.file_info_box.hide()
449
266
 
 
267
        self.bugs_table.clear()
 
268
        bugs_text = revision.properties.get('bugs', None)
 
269
        if bugs_text:
 
270
            for bugline in bugs_text.splitlines():
 
271
                (url, status) = bugline.split(" ")
 
272
                self.bugs_table.add_bug(url, status)
 
273
 
450
274
    def update_tags(self):
451
275
        if self._branch is not None and self._branch.supports_tags():
452
276
            self._tagdict = self._branch.tags.get_reverse_tag_dict()
456
280
        self._add_tags()
457
281
 
458
282
    def _update_signature(self, widget, param):
459
 
        if self.get_current_page() == PAGE_SIGNATURE:
460
 
            self.signature_table.set_revision(self._revision)
 
283
        revid = self._revision.revision_id
461
284
 
462
 
    def _update_bugs(self, widget, param):
463
 
        self.bugs_page.set_revision(self._revision)
464
 
        label = self.get_tab_label(self.bugs_page)
465
 
        label.set_sensitive(self.bugs_page.get_num_bugs() != 0)
 
285
        if self._branch.repository.has_signature_for_revision_id(revid):
 
286
            signature_text = self._branch.repository.get_signature_text(revid)
 
287
            self.signature_table.show_signature(signature_text)
 
288
        else:
 
289
            self.signature_table.show_no_signature()
466
290
 
467
291
    def set_children(self, children):
468
292
        self._add_parents_or_children(children,
469
293
                                      self.children_widgets,
470
294
                                      self.children_table)
471
295
 
472
 
    def _switch_page_cb(self, notebook, page, page_num):
473
 
        if page_num == PAGE_SIGNATURE:
474
 
            self.signature_table.set_revision(self._revision)
475
 
 
476
 
 
477
 
 
478
296
    def _show_clicked_cb(self, widget, revid, parentid):
479
297
        """Callback for when the show button for a parent is clicked."""
480
298
        self._show_callback(revid, parentid)
509
327
        table.resize(max(len(revids), 1), 2)
510
328
 
511
329
        for idx, revid in enumerate(revids):
512
 
            align = gtk.Alignment(0.0, 0.0, 1, 1)
 
330
            align = gtk.Alignment(0.0, 0.0)
513
331
            widgets.append(align)
514
332
            table.attach(align, 1, 2, idx, idx + 1,
515
333
                                      gtk.EXPAND | gtk.FILL, gtk.FILL)
532
350
                hbox.pack_start(button, expand=False, fill=True)
533
351
                button.show()
534
352
 
535
 
            button = gtk.Button()
536
 
            revid_label = gtk.Label(str(revid))
537
 
            revid_label.set_ellipsize(pango.ELLIPSIZE_MIDDLE)
538
 
            revid_label.set_alignment(0.0, 0.5)
539
 
            button.add(revid_label)
 
353
            button = gtk.Button(revid)
540
354
            button.connect("clicked",
541
 
                    lambda w, r: self.set_revision(self._repository.get_revision(r)), revid)
 
355
                    lambda w, r: self.set_revision(self._branch.repository.get_revision(r)), revid)
542
356
            button.set_use_underline(False)
543
 
            hbox.pack_start(button, expand=True, fill=True)
544
 
            button.show_all()
 
357
            hbox.pack_start(button, expand=False, fill=True)
 
358
            button.show()
545
359
 
546
360
    def _create_general(self):
547
361
        vbox = gtk.VBox(False, 6)
560
374
        vbox.show()
561
375
 
562
376
    def _create_signature(self):
563
 
        self.signature_table = SignatureTab(self._repository)
 
377
        try:
 
378
            self.signature_table = SignatureTab()
 
379
        except ValueError: # No GPG found
 
380
            self.signature_table = None
 
381
            return
564
382
        self.append_page(self.signature_table, tab_label=gtk.Label('Signature'))
565
383
        self.connect_after('notify::revision', self._update_signature)
566
384
 
570
388
        self.table.set_col_spacings(6)
571
389
        self.table.show()
572
390
 
573
 
        row = 0
574
 
 
 
391
        align = gtk.Alignment(1.0, 0.5)
575
392
        label = gtk.Label()
576
 
        label.set_alignment(1.0, 0.5)
577
393
        label.set_markup("<b>Revision Id:</b>")
578
 
        self.table.attach(label, 0, 1, row, row+1, gtk.FILL, gtk.FILL)
 
394
        align.add(label)
 
395
        self.table.attach(align, 0, 1, 0, 1, gtk.FILL, gtk.FILL)
 
396
        align.show()
579
397
        label.show()
580
398
 
 
399
        align = gtk.Alignment(0.0, 0.5)
581
400
        revision_id = gtk.Label()
582
 
        revision_id.set_ellipsize(pango.ELLIPSIZE_MIDDLE)
583
 
        revision_id.set_alignment(0.0, 0.5)
584
401
        revision_id.set_selectable(True)
585
402
        self.connect('notify::revision', 
586
403
                lambda w, p: revision_id.set_text(self._revision.revision_id))
587
 
        self.table.attach(revision_id, 1, 2, row, row+1, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
404
        align.add(revision_id)
 
405
        self.table.attach(align, 1, 2, 0, 1, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
406
        align.show()
588
407
        revision_id.show()
589
408
 
590
 
        row += 1
 
409
        align = gtk.Alignment(1.0, 0.5)
591
410
        self.author_label = gtk.Label()
592
 
        self.author_label.set_alignment(1.0, 0.5)
593
411
        self.author_label.set_markup("<b>Author:</b>")
594
 
        self.table.attach(self.author_label, 0, 1, row, row+1, gtk.FILL, gtk.FILL)
 
412
        align.add(self.author_label)
 
413
        self.table.attach(align, 0, 1, 1, 2, gtk.FILL, gtk.FILL)
 
414
        align.show()
595
415
        self.author_label.show()
596
416
 
 
417
        align = gtk.Alignment(0.0, 0.5)
597
418
        self.author = gtk.Label()
598
 
        self.author.set_ellipsize(pango.ELLIPSIZE_END)
599
 
        self.author.set_alignment(0.0, 0.5)
600
419
        self.author.set_selectable(True)
601
 
        self.table.attach(self.author, 1, 2, row, row+1, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
420
        align.add(self.author)
 
421
        self.table.attach(align, 1, 2, 1, 2, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
422
        align.show()
602
423
        self.author.show()
603
424
        self.author.hide()
604
425
 
605
 
        row += 1
 
426
        align = gtk.Alignment(1.0, 0.5)
606
427
        label = gtk.Label()
607
 
        label.set_alignment(1.0, 0.5)
608
428
        label.set_markup("<b>Committer:</b>")
609
 
        self.table.attach(label, 0, 1, row, row+1, gtk.FILL, gtk.FILL)
 
429
        align.add(label)
 
430
        self.table.attach(align, 0, 1, 2, 3, gtk.FILL, gtk.FILL)
 
431
        align.show()
610
432
        label.show()
611
433
 
 
434
        align = gtk.Alignment(0.0, 0.5)
612
435
        self.committer = gtk.Label()
613
 
        self.committer.set_ellipsize(pango.ELLIPSIZE_END)
614
 
        self.committer.set_alignment(0.0, 0.5)
615
436
        self.committer.set_selectable(True)
616
 
        self.table.attach(self.committer, 1, 2, row, row+1, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
437
        align.add(self.committer)
 
438
        self.table.attach(align, 1, 2, 2, 3, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
439
        align.show()
617
440
        self.committer.show()
618
441
 
619
 
        row += 1
 
442
        align = gtk.Alignment(0.0, 0.5)
620
443
        label = gtk.Label()
621
 
        label.set_alignment(1.0, 0.5)
622
444
        label.set_markup("<b>Branch nick:</b>")
623
 
        self.table.attach(label, 0, 1, row, row+1, gtk.FILL, gtk.FILL)
 
445
        align.add(label)
 
446
        self.table.attach(align, 0, 1, 3, 4, gtk.FILL, gtk.FILL)
624
447
        label.show()
 
448
        align.show()
625
449
 
 
450
        align = gtk.Alignment(0.0, 0.5)
626
451
        self.branchnick_label = gtk.Label()
627
 
        self.branchnick_label.set_ellipsize(pango.ELLIPSIZE_MIDDLE)
628
 
        self.branchnick_label.set_alignment(0.0, 0.5)
629
452
        self.branchnick_label.set_selectable(True)
630
 
        self.table.attach(self.branchnick_label, 1, 2, row, row+1, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
453
        align.add(self.branchnick_label)
 
454
        self.table.attach(align, 1, 2, 3, 4, gtk.EXPAND | gtk.FILL, gtk.FILL)
631
455
        self.branchnick_label.show()
 
456
        align.show()
632
457
 
633
 
        row += 1
 
458
        align = gtk.Alignment(1.0, 0.5)
634
459
        label = gtk.Label()
635
 
        label.set_alignment(1.0, 0.5)
636
460
        label.set_markup("<b>Timestamp:</b>")
637
 
        self.table.attach(label, 0, 1, row, row+1, gtk.FILL, gtk.FILL)
 
461
        align.add(label)
 
462
        self.table.attach(align, 0, 1, 4, 5, gtk.FILL, gtk.FILL)
 
463
        align.show()
638
464
        label.show()
639
465
 
 
466
        align = gtk.Alignment(0.0, 0.5)
640
467
        self.timestamp = gtk.Label()
641
 
        self.timestamp.set_ellipsize(pango.ELLIPSIZE_END)
642
 
        self.timestamp.set_alignment(0.0, 0.5)
643
468
        self.timestamp.set_selectable(True)
644
 
        self.table.attach(self.timestamp, 1, 2, row, row+1, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
469
        align.add(self.timestamp)
 
470
        self.table.attach(align, 1, 2, 4, 5, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
471
        align.show()
645
472
        self.timestamp.show()
646
473
 
647
 
        row += 1
 
474
        align = gtk.Alignment(1.0, 0.5)
648
475
        self.tags_label = gtk.Label()
649
 
        self.tags_label.set_alignment(1.0, 0.5)
650
476
        self.tags_label.set_markup("<b>Tags:</b>")
651
 
        self.table.attach(self.tags_label, 0, 1, row, row+1, gtk.FILL, gtk.FILL)
 
477
        align.add(self.tags_label)
 
478
        align.show()
 
479
        self.table.attach(align, 0, 1, 5, 6, gtk.FILL, gtk.FILL)
652
480
        self.tags_label.show()
653
481
 
 
482
        align = gtk.Alignment(0.0, 0.5)
654
483
        self.tags_list = gtk.Label()
655
 
        self.tags_list.set_ellipsize(pango.ELLIPSIZE_MIDDLE)
656
 
        self.tags_list.set_alignment(0.0, 0.5)
657
 
        self.table.attach(self.tags_list, 1, 2, row, row+1, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
484
        align.add(self.tags_list)
 
485
        self.table.attach(align, 1, 2, 5, 6, gtk.EXPAND | gtk.FILL, gtk.FILL)
 
486
        align.show()
658
487
        self.tags_list.show()
659
488
 
660
489
        self.connect('notify::revision', self._add_tags)
715
544
        return window
716
545
 
717
546
    def _create_bugs(self):
718
 
        self.bugs_page = BugsTab()
719
 
        self.connect_after('notify::revision', self._update_bugs) 
720
 
        self.append_page(self.bugs_page, tab_label=gtk.Label('Bugs'))
 
547
        self.bugs_table = BugsTab()
 
548
        self.append_page(self.bugs_table, tab_label=gtk.Label('Bugs'))
721
549
 
722
550
    def _create_file_info_view(self):
723
551
        self.file_info_box = gtk.VBox(False, 6)