53
50
def construct(self):
54
51
"""Construct the window contents."""
52
vbox = gtk.VBox(spacing=0)
55
vbox.pack_start(self.construct_navigation(), expand=False, fill=True)
55
57
paned = gtk.VPaned()
56
58
paned.pack1(self.construct_top(), resize=True, shrink=False)
57
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)
61
66
def construct_top(self):
62
67
"""Construct the top-half of the window."""
63
vbox = gtk.VBox(spacing=6)
64
vbox.set_border_width(12)
67
68
scrollwin = gtk.ScrolledWindow()
68
69
scrollwin.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
69
70
scrollwin.set_shadow_type(gtk.SHADOW_IN)
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.repository, 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(revision.parent_ids) > 0)
205
self.back_button.set_sensitive(len(self.parent_ids[revision]) > 0)
184
206
self.fwd_button.set_sensitive(len(self.children[revision]) > 0)
208
if self.branch.supports_tags():
209
tagdict = self.branch.tags.get_reverse_tag_dict()
210
if tagdict.has_key(revision.revision_id):
211
tags = tagdict[revision.revision_id]
212
self.logview.set_revision(revision, tags)
186
214
def _back_clicked_cb(self, *args):
187
215
"""Callback for when the back button is clicked."""
188
216
(path, col) = self.treeview.get_cursor()
189
217
revision = self.model[path][0]
190
if not len(revision.parent_ids):
218
if not len(self.parent_ids[revision]):
193
for parent_id in revision.parent_ids:
221
for parent_id in self.parent_ids[revision]:
194
222
parent = self.revisions[parent_id]
195
223
if same_branch(revision, parent):
196
224
self.treeview.set_cursor(self.index[parent])
199
next = self.revisions[revision.parent_ids[0]]
227
next = self.revisions[self.parent_ids[revision][0]]
200
228
self.treeview.set_cursor(self.index[next])
229
self.treeview.grab_focus()
202
231
def _fwd_clicked_cb(self, *args):
203
232
"""Callback for when the forward button is clicked."""
214
243
prev = list(self.children[revision])[0]
215
244
self.treeview.set_cursor(self.index[prev])
245
self.treeview.grab_focus()
247
def _go_clicked_cb(self, revid):
248
"""Callback for when the go button for a parent is clicked."""
249
self.treeview.set_cursor(self.index[self.revisions[revid]])
250
self.treeview.grab_focus()
252
def show_diff(self, branch, revid, parentid):
253
"""Open a new window to show a diff between the given revisions."""
254
from bzrlib.plugins.gtk.diff import DiffWindow
255
window = DiffWindow()
256
(parent_tree, rev_tree) = branch.repository.revision_trees([parentid,
258
description = revid + " - " + branch.nick
259
window.set_diff(description, rev_tree, parent_tree)
262
def _show_clicked_cb(self, revid, parentid):
263
"""Callback for when the show button for a parent is clicked."""
264
self.show_diff(self.branch, revid, parentid)
265
self.treeview.grab_focus()
267
def _treeview_row_mouseclick(self, widget, event):
268
from bzrlib.plugins.gtk.revisionmenu import RevisionPopupMenu
269
if event.button == 3:
270
menu = RevisionPopupMenu(self.branch.repository,
271
[x.revision_id for x in self.selected_revisions()],
273
menu.popup(None, None, None, event.button, event.get_time())
275
def selected_revision(self, path):
276
return self.model[path][0]
278
def selected_revisions(self):
279
return [self.selected_revision(path) for path in \
280
self.treeview.get_selection().get_selected_rows()[1]]
282
def _treeview_row_activated_cb(self, widget, path, col):
283
# TODO: more than one parent
284
"""Callback for when a treeview row gets activated."""
285
revision = self.selected_revision(path)
286
if len(self.parent_ids[revision]) == 0:
287
# Ignore revisions without parent
289
parent_id = self.parent_ids[revision][0]
290
self.show_diff(self.branch, revision.revision_id, parent_id)
291
self.treeview.grab_focus()