/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 branchwin.py

  • Committer: Scott James Remnant
  • Date: 2005-10-17 04:26:00 UTC
  • Revision ID: scott@netsplit.com-20051017042600-b3a3265abfffdf53
Split the display in two with a pane, we'll use the bottom half to show
information about the current revision.  Add a Back and Forward button
which figure out which revision is logically the next of previous and
moves the cursor to it.  Handle the cursor-changed event to enable/disable
the buttons (and soon update the bottom pane).

Further split up graph.py so we can stash the internal lists to do the
above; also it may allow us in future to produce partial graphs.

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
#!/usr/bin/python
1
2
# -*- coding: UTF-8 -*-
2
3
"""Branch window.
3
4
 
9
10
__author__    = "Scott James Remnant <scott@ubuntu.com>"
10
11
 
11
12
 
 
13
import os
 
14
 
12
15
import gtk
13
16
import gobject
14
17
import pango
33
36
 
34
37
        # Use three-quarters of the screen by default
35
38
        screen = self.get_screen()
36
 
        monitor = screen.get_monitor_geometry(0)
37
 
        width = int(monitor.width * 0.75)
38
 
        height = int(monitor.height * 0.75)
 
39
        width = int(screen.get_width() * 0.75)
 
40
        height = int(screen.get_height() * 0.75)
39
41
        self.set_default_size(width, height)
40
42
 
41
43
        # FIXME AndyFitz!
49
51
 
50
52
    def construct(self):
51
53
        """Construct the window contents."""
52
 
        vbox = gtk.VBox(spacing=0)
53
 
        self.add(vbox)
54
 
 
55
 
        vbox.pack_start(self.construct_navigation(), expand=False, fill=True)
56
 
 
57
54
        paned = gtk.VPaned()
58
55
        paned.pack1(self.construct_top(), resize=True, shrink=False)
59
 
        paned.pack2(self.construct_bottom(), resize=False, shrink=True)
 
56
        paned.pack2(self.construct_bottom(), resize=True, shrink=True)
 
57
        self.add(paned)
60
58
        paned.show()
61
 
        vbox.pack_start(paned, expand=True, fill=True)
62
 
        vbox.set_focus_child(paned)
63
 
 
64
 
        vbox.show()
65
59
 
66
60
    def construct_top(self):
67
61
        """Construct the top-half of the window."""
 
62
        vbox = gtk.VBox(spacing=6)
 
63
        vbox.set_border_width(12)
 
64
        vbox.show()
 
65
 
68
66
        scrollwin = gtk.ScrolledWindow()
69
67
        scrollwin.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
70
68
        scrollwin.set_shadow_type(gtk.SHADOW_IN)
 
69
        #scrollwin.set_border_width(12)
 
70
        vbox.pack_start(scrollwin, expand=True, fill=True)
71
71
        scrollwin.show()
72
72
 
73
73
        self.treeview = gtk.TreeView()
74
74
        self.treeview.set_rules_hint(True)
75
75
        self.treeview.set_search_column(4)
76
76
        self.treeview.connect("cursor-changed", self._treeview_cursor_cb)
77
 
        self.treeview.connect("row-activated", self._treeview_row_activated_cb)
78
 
        self.treeview.connect("button-release-event", 
79
 
                self._treeview_row_mouseclick)
80
77
        scrollwin.add(self.treeview)
81
78
        self.treeview.show()
82
79
 
83
80
        cell = CellRendererGraph()
84
81
        column = gtk.TreeViewColumn()
85
 
        column.set_resizable(True)
 
82
        column.set_resizable(False)
86
83
        column.pack_start(cell, expand=False)
87
84
        column.add_attribute(cell, "node", 1)
88
85
        column.add_attribute(cell, "in-lines", 2)
115
112
        column.add_attribute(cell, "text", 6)
116
113
        self.treeview.append_column(column)
117
114
 
118
 
        return scrollwin
119
 
 
120
 
    def construct_navigation(self):
121
 
        """Construct the navigation buttons."""
122
 
        frame = gtk.Frame()
123
 
        frame.set_shadow_type(gtk.SHADOW_OUT)
124
 
        frame.show()
125
 
        
126
 
        hbox = gtk.HBox(spacing=12)
127
 
        frame.add(hbox)
 
115
        hbox = gtk.HBox(False, spacing=6)
 
116
        vbox.pack_start(hbox, expand=False, fill=False)
128
117
        hbox.show()
129
118
 
130
119
        self.back_button = gtk.Button(stock=gtk.STOCK_GO_BACK)
131
 
        self.back_button.set_relief(gtk.RELIEF_NONE)
132
120
        self.back_button.add_accelerator("clicked", self.accel_group, ord('['),
133
121
                                         0, 0)
134
122
        self.back_button.connect("clicked", self._back_clicked_cb)
136
124
        self.back_button.show()
137
125
 
138
126
        self.fwd_button = gtk.Button(stock=gtk.STOCK_GO_FORWARD)
139
 
        self.fwd_button.set_relief(gtk.RELIEF_NONE)
140
127
        self.fwd_button.add_accelerator("clicked", self.accel_group, ord(']'),
141
128
                                        0, 0)
142
129
        self.fwd_button.connect("clicked", self._fwd_clicked_cb)
143
130
        hbox.pack_start(self.fwd_button, expand=False, fill=True)
144
131
        self.fwd_button.show()
145
132
 
146
 
        return frame
 
133
        return vbox
147
134
 
148
135
    def construct_bottom(self):
149
136
        """Construct the bottom half of the window."""
150
 
        from bzrlib.plugins.gtk.logview import LogView
151
 
        self.logview = LogView()
152
 
        (width, height) = self.get_size()
153
 
        self.logview.set_size_request(width, int(height / 2.5))
154
 
        self.logview.show()
155
 
        self.logview.set_show_callback(self._show_clicked_cb)
156
 
        self.logview.set_go_callback(self._go_clicked_cb)
157
 
        return self.logview
158
 
 
159
 
    def set_branch(self, branch, start, maxnum):
 
137
        label = gtk.Label("test")
 
138
        label.show()
 
139
 
 
140
        return label
 
141
 
 
142
    def set_branch(self, branch, start):
160
143
        """Set the branch and start position for this window.
161
144
 
162
145
        Creates a new TreeModel and populates it with information about
163
146
        the new branch before updating the window title and model of the
164
147
        treeview itself.
165
148
        """
166
 
        self.branch = branch
167
 
 
168
149
        # [ revision, node, last_lines, lines, message, committer, timestamp ]
169
150
        self.model = gtk.ListStore(gobject.TYPE_PYOBJECT,
170
151
                                   gobject.TYPE_PYOBJECT,
175
156
        index = 0
176
157
 
177
158
        last_lines = []
178
 
        (self.revisions, colours, self.children, self.parent_ids,
179
 
            merge_sorted) = distances(branch, start)
180
 
        for (index, (revision, node, lines)) in enumerate(graph(
181
 
                self.revisions, colours, merge_sorted)):
182
 
            # FIXME: at this point we should be able to show the graph order
183
 
            # and lines with no message or commit data - and then incrementally
184
 
            # fill the timestamp, committer etc data as desired.
 
159
        (revids, self.revisions, colours, self.children) \
 
160
                 = distances(branch, start)
 
161
        for revision, node, lines in graph(revids, self.revisions, colours):
185
162
            message = revision.message.split("\n")[0]
186
163
            if revision.committer is not None:
187
164
                timestamp = format_date(revision.timestamp, revision.timezone)
188
165
            else:
189
166
                timestamp = None
190
 
            self.model.append([revision, node, last_lines, lines,
191
 
                               message, revision.committer, timestamp])
 
167
 
 
168
            self.model.append([ revision, node, last_lines, lines,
 
169
                                message, revision.committer, timestamp ])
192
170
            self.index[revision] = index
 
171
            index += 1
 
172
 
193
173
            last_lines = lines
194
 
            if maxnum is not None and index > maxnum:
195
 
                break
196
174
 
197
 
        self.set_title(branch.nick + " - bzrk")
 
175
        self.set_title(os.path.basename(branch.base) + " - bzrk")
198
176
        self.treeview.set_model(self.model)
199
177
 
200
178
    def _treeview_cursor_cb(self, *args):
202
180
        (path, col) = self.treeview.get_cursor()
203
181
        revision = self.model[path][0]
204
182
 
205
 
        self.back_button.set_sensitive(len(self.parent_ids[revision]) > 0)
206
 
        self.fwd_button.set_sensitive(len(self.children[revision]) > 0)
207
 
        self.logview.set_revision(revision)
 
183
        self.back_button.set_sensitive(len(self.children[revision]) > 0)
 
184
        self.fwd_button.set_sensitive(len(revision.parent_ids) > 0)
208
185
 
209
186
    def _back_clicked_cb(self, *args):
210
187
        """Callback for when the back button is clicked."""
211
188
        (path, col) = self.treeview.get_cursor()
212
189
        revision = self.model[path][0]
213
 
        if not len(self.parent_ids[revision]):
214
 
            return
215
 
 
216
 
        for parent_id in self.parent_ids[revision]:
217
 
            parent = self.revisions[parent_id]
218
 
            if same_branch(revision, parent):
219
 
                self.treeview.set_cursor(self.index[parent])
220
 
                break
221
 
        else:
222
 
            next = self.revisions[self.parent_ids[revision][0]]
223
 
            self.treeview.set_cursor(self.index[next])
224
 
        self.treeview.grab_focus()
225
 
 
226
 
    def _fwd_clicked_cb(self, *args):
227
 
        """Callback for when the forward button is clicked."""
228
 
        (path, col) = self.treeview.get_cursor()
229
 
        revision = self.model[path][0]
230
190
        if not len(self.children[revision]):
231
191
            return
232
192
 
237
197
        else:
238
198
            prev = list(self.children[revision])[0]
239
199
            self.treeview.set_cursor(self.index[prev])
240
 
        self.treeview.grab_focus()
241
 
 
242
 
    def _go_clicked_cb(self, revid):
243
 
        """Callback for when the go button for a parent is clicked."""
244
 
        self.treeview.set_cursor(self.index[self.revisions[revid]])
245
 
        self.treeview.grab_focus()
246
 
 
247
 
    def show_diff(self, branch, revid, parentid):
248
 
        """Open a new window to show a diff between the given revisions."""
249
 
        from bzrlib.plugins.gtk.diff import DiffWindow
250
 
        window = DiffWindow()
251
 
        (parent_tree, rev_tree) = branch.repository.revision_trees([parentid, 
252
 
                                                                   revid])
253
 
        description = revid + " - " + branch.nick
254
 
        window.set_diff(description, rev_tree, parent_tree)
255
 
        window.show()
256
 
 
257
 
    def _show_clicked_cb(self, revid, parentid):
258
 
        """Callback for when the show button for a parent is clicked."""
259
 
        self.show_diff(self.branch, revid, parentid)
260
 
        self.treeview.grab_focus()
261
 
 
262
 
    def _treeview_row_mouseclick(self, widget, event):
263
 
        from bzrlib.plugins.gtk.revisionmenu import RevisionPopupMenu
264
 
        if event.button == 3:
265
 
            menu = RevisionPopupMenu(self.branch.repository, 
266
 
                [x.revision_id for x in self.selected_revisions()],
267
 
                self.branch)
268
 
            menu.popup(None, None, None, event.button, event.get_time())
269
 
 
270
 
    def selected_revision(self, path):
271
 
        return self.model[path][0]
272
 
 
273
 
    def selected_revisions(self):
274
 
        return [self.selected_revision(path) for path in \
275
 
                self.treeview.get_selection().get_selected_rows()[1]]
276
 
 
277
 
    def _treeview_row_activated_cb(self, widget, path, col):
278
 
        # TODO: more than one parent
279
 
        """Callback for when a treeview row gets activated."""
280
 
        revision = self.selected_revision(path)
281
 
        if len(self.parent_ids[revision]) == 0:
282
 
            # Ignore revisions without parent
 
200
 
 
201
    def _fwd_clicked_cb(self, *args):
 
202
        """Callback for when the forward button is clicked."""
 
203
        (path, col) = self.treeview.get_cursor()
 
204
        revision = self.model[path][0]
 
205
        if not len(revision.parent_ids):
283
206
            return
284
 
        parent_id = self.parent_ids[revision][0]
285
 
        self.show_diff(self.branch, revision.revision_id, parent_id)
286
 
        self.treeview.grab_focus()
 
207
 
 
208
        for parent_id in revision.parent_ids:
 
209
            parent = self.revisions[parent_id]
 
210
            if same_branch(revision, parent):
 
211
                self.treeview.set_cursor(self.index[parent])
 
212
                break
 
213
        else:
 
214
            next = self.revisions[revision.parent_ids[0]]
 
215
            self.treeview.set_cursor(self.index[next])
 
216