52
50
def construct(self):
53
51
"""Construct the window contents."""
52
vbox = gtk.VBox(spacing=0)
55
vbox.pack_start(self.construct_navigation(), expand=False, fill=True)
54
57
paned = gtk.VPaned()
55
58
paned.pack1(self.construct_top(), resize=True, shrink=False)
56
paned.pack2(self.construct_bottom(), resize=True, shrink=True)
59
paned.pack2(self.construct_bottom(), resize=False, shrink=True)
61
vbox.pack_start(paned, expand=True, fill=True)
62
vbox.set_focus_child(paned)
60
66
def construct_top(self):
61
67
"""Construct the top-half of the window."""
62
vbox = gtk.VBox(spacing=6)
63
vbox.set_border_width(12)
66
68
scrollwin = gtk.ScrolledWindow()
67
69
scrollwin.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
68
70
scrollwin.set_shadow_type(gtk.SHADOW_IN)
69
#scrollwin.set_border_width(12)
70
vbox.pack_start(scrollwin, expand=True, fill=True)
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)
77
80
scrollwin.add(self.treeview)
78
81
self.treeview.show()
80
83
cell = CellRendererGraph()
81
84
column = gtk.TreeViewColumn()
82
column.set_resizable(False)
85
column.set_resizable(True)
83
86
column.pack_start(cell, expand=False)
84
87
column.add_attribute(cell, "node", 1)
85
88
column.add_attribute(cell, "in-lines", 2)
112
115
column.add_attribute(cell, "text", 6)
113
116
self.treeview.append_column(column)
115
hbox = gtk.HBox(False, spacing=6)
116
vbox.pack_start(hbox, expand=False, fill=False)
120
def construct_navigation(self):
121
"""Construct the navigation buttons."""
123
frame.set_shadow_type(gtk.SHADOW_OUT)
126
hbox = gtk.HBox(spacing=12)
119
130
self.back_button = gtk.Button(stock=gtk.STOCK_GO_BACK)
131
self.back_button.set_relief(gtk.RELIEF_NONE)
120
132
self.back_button.add_accelerator("clicked", self.accel_group, ord('['),
122
134
self.back_button.connect("clicked", self._back_clicked_cb)
124
136
self.back_button.show()
126
138
self.fwd_button = gtk.Button(stock=gtk.STOCK_GO_FORWARD)
139
self.fwd_button.set_relief(gtk.RELIEF_NONE)
127
140
self.fwd_button.add_accelerator("clicked", self.accel_group, ord(']'),
129
142
self.fwd_button.connect("clicked", self._fwd_clicked_cb)
130
143
hbox.pack_start(self.fwd_button, expand=False, fill=True)
131
144
self.fwd_button.show()
135
148
def construct_bottom(self):
136
149
"""Construct the bottom half of the window."""
137
label = gtk.Label("test")
142
def set_branch(self, branch, start):
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))
155
self.logview.set_show_callback(self._show_clicked_cb)
156
self.logview.set_go_callback(self._go_clicked_cb)
159
def set_branch(self, branch, start, maxnum):
143
160
"""Set the branch and start position for this window.
145
162
Creates a new TreeModel and populates it with information about
146
163
the new branch before updating the window title and model of the
149
168
# [ revision, node, last_lines, lines, message, committer, timestamp ]
150
169
self.model = gtk.ListStore(gobject.TYPE_PYOBJECT,
151
170
gobject.TYPE_PYOBJECT,
159
(revids, self.revisions, colours, self.children) \
160
= distances(branch, start)
161
for revision, node, lines in graph(revids, self.revisions, colours):
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.
162
185
message = revision.message.split("\n")[0]
163
186
if revision.committer is not None:
164
187
timestamp = format_date(revision.timestamp, revision.timezone)
168
self.model.append([ revision, node, last_lines, lines,
169
message, revision.committer, timestamp ])
190
self.model.append([revision, node, last_lines, lines,
191
message, revision.committer, timestamp])
170
192
self.index[revision] = index
173
193
last_lines = lines
194
if maxnum is not None and index > maxnum:
175
self.set_title(os.path.basename(branch.base) + " - bzrk")
197
self.set_title(branch.nick + " - bzrk")
176
198
self.treeview.set_model(self.model)
178
200
def _treeview_cursor_cb(self, *args):
180
202
(path, col) = self.treeview.get_cursor()
181
203
revision = self.model[path][0]
183
self.back_button.set_sensitive(len(self.children[revision]) > 0)
184
self.fwd_button.set_sensitive(len(revision.parent_ids) > 0)
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)
186
209
def _back_clicked_cb(self, *args):
187
210
"""Callback for when the back button is clicked."""
188
211
(path, col) = self.treeview.get_cursor()
189
212
revision = self.model[path][0]
213
if not len(self.parent_ids[revision]):
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])
222
next = self.revisions[self.parent_ids[revision][0]]
223
self.treeview.set_cursor(self.index[next])
224
self.treeview.grab_focus()
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]
190
230
if not len(self.children[revision]):
198
238
prev = list(self.children[revision])[0]
199
239
self.treeview.set_cursor(self.index[prev])
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):
240
self.treeview.grab_focus()
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()
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,
253
description = revid + " - " + branch.nick
254
window.set_diff(description, rev_tree, parent_tree)
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()
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()],
268
menu.popup(None, None, None, event.button, event.get_time())
270
def selected_revision(self, path):
271
return self.model[path][0]
273
def selected_revisions(self):
274
return [self.selected_revision(path) for path in \
275
self.treeview.get_selection().get_selected_rows()[1]]
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
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])
214
next = self.revisions[revision.parent_ids[0]]
215
self.treeview.set_cursor(self.index[next])
284
parent_id = self.parent_ids[revision][0]
285
self.show_diff(self.branch, revision.revision_id, parent_id)
286
self.treeview.grab_focus()