43
35
It's up to you how to actually draw the nodes and lines (straight,
44
36
curved, kinked, etc.) and to pick the actual colours for each index.
46
assert isinstance(start_revs, list)
47
def update_root_progress(step_number):
48
"""IFF our container received a root progress bar, then update it."""
49
if root_progress is not None:
50
root_progress.update(current=step_number)
39
mainline = branch.revision_history()
40
graph_parents = branch.repository.get_revision_graph(start)
54
41
graph_children = {}
55
update_root_progress(1)
56
progress_bar = ui.ui_factory.nested_progress_bar()
58
progress_bar.update(msg="Arranging tree fragments")
59
for i, (revid, parent_revids) in enumerate(graph.iter_ancestry(start_revs)):
62
if parent_revids is None:
65
if parent_revids == (NULL_REVISION,):
66
graph_parents[revid] = ()
68
graph_parents[revid] = parent_revids
69
for parent in parent_revids:
70
graph_children.setdefault(parent, []).append(revid)
71
graph_children.setdefault(revid, [])
73
progress_bar.finished()
75
update_root_progress(2)
76
progress_bar = ui.ui_factory.nested_progress_bar()
78
progress_bar.update(msg="Removing ghosts", total=len(ghosts))
79
for i, ghost in enumerate(ghosts):
81
progress_bar.update(current=i)
82
for ghost_child in graph_children[ghost]:
83
graph_parents[ghost_child] = [p for p in graph_parents[ghost_child]
86
progress_bar.finished()
87
graph_parents["top:"] = start_revs
89
if len(graph_parents)>0:
90
merge_sorted_revisions = merge_sort(
95
merge_sorted_revisions = ()
98
merge_sorted_revisions = [elem for elem in merge_sorted_revisions \
101
assert merge_sorted_revisions[0][1] == "top:"
102
merge_sorted_revisions = merge_sorted_revisions[1:]
42
for revid in graph_parents.iterkeys():
43
graph_children[revid] = []
45
merge_sorted_revisions = merge_sort(
107
# This will hold an item for each "branch". For a revisions, the revsion
108
# number less the least significant digit is the branch_id, and used as the
109
# key for the dict. Hence revision with the same revsion number less the
110
# least significant digit are considered to be in the same branch line.
111
# e.g.: for revisions 290.12.1 and 290.12.2, the branch_id would be 290.12,
112
# and these two revisions will be in the same branch line. Each value is
113
# a list of rev_indexes in the branch.
118
update_root_progress(3)
119
progress_bar = ui.ui_factory.nested_progress_bar()
121
progress_bar.update(msg="Finding nodes", total=len(merge_sorted_revisions))
122
for (rev_index, (sequence_number,
126
end_of_merge)) in enumerate(merge_sorted_revisions):
128
if rev_index % 25 == 0:
129
progress_bar.update(current=rev_index)
130
if maxnum and rev_index >= maxnum:
132
revid_index[revid] = rev_index
134
parents = graph_parents[revid]
135
linegraph.append([revid,
143
revno_index[revno_sequence] = rev_index
145
branch_id = revno_sequence[0:-1]
148
if branch_id not in branch_lines:
150
branch_lines[branch_id] = branch_line
152
branch_line = branch_lines[branch_id]
154
branch_line.append(rev_index)
156
progress_bar.finished()
159
branch_ids = branch_lines.keys()
161
def branch_id_cmp(x, y):
162
"""Compaire branch_id's first by the number of digits, then reversed
168
return cmp(len_x, len_y)
170
branch_ids.sort(branch_id_cmp)
171
# This will hold a tuple of (child_index, parent_index, col_index) for each
172
# line that needs to be drawn. If col_index is not none, then the line is
173
# drawn along that column, else the the line can be drawn directly between
174
# the child and parent because either the child and parent are in the same
175
# branch line, or the child and parent are 1 row apart.
177
empty_column = [False for i in range(len(graph_parents))]
178
# This will hold a bit map for each cell. If the cell is true, then the
179
# cell allready contains a node or line. This use when deciding what column
180
# to place a branch line or line in, without it overlaping something else.
181
columns = [list(empty_column)]
184
update_root_progress(4)
185
progress_bar = ui.ui_factory.nested_progress_bar()
187
progress_bar.update(msg="Organizing edges", total=len(branch_ids))
188
for i, branch_id in enumerate(branch_ids):
190
progress_bar.update(current=i)
191
branch_line = branch_lines[branch_id]
193
# Find the col_index for the direct parent branch. This will be the
194
# starting point when looking for a free column.
197
if len(branch_id) > 1:
198
parent_revno = branch_id[0:-1]
199
if parent_revno in revno_index:
200
parent_index = revno_index[parent_revno]
201
parent_node = linegraph[parent_index][1]
203
parent_col_index = parent_node[0]
206
col_search_order = _branch_line_col_search_order(columns,
208
color = reduce(lambda x, y: x+y, branch_id, 0)
212
last_rev_index = None
213
for rev_index in branch_line:
215
if broken_line_length and \
216
rev_index - last_rev_index > broken_line_length:
217
line_range.append(last_rev_index+1)
218
line_range.append(rev_index-1)
220
line_range.extend(range(last_rev_index+1, rev_index))
222
line_range.append(rev_index)
223
last_rev_index = rev_index
226
if broken_line_length and \
227
parent_index - last_rev_index > broken_line_length:
228
line_range.append(last_rev_index+1)
230
line_range.extend(range(last_rev_index+1, parent_index))
232
col_index = _find_free_column(columns,
236
node = (col_index, color)
237
for rev_index in branch_line:
238
linegraph[rev_index][1] = node
239
columns[col_index][rev_index] = True
241
for rev_index in branch_line:
246
end_of_merge) = merge_sorted_revisions[rev_index]
248
linegraph[rev_index][4] = graph_children[revid]
249
col_index = linegraph[rev_index][1][0]
251
for parent_revid in graph_parents[revid]:
252
if parent_revid in revid_index:
254
parent_index = revid_index[parent_revid]
255
parent_node = linegraph[parent_index][1]
257
parent_col_index = parent_node[0]
259
parent_col_index = None
261
_line_col_search_order(columns,
265
# If this line is really long, break it.
266
if len(branch_id) > 0 and \
267
broken_line_length and \
268
parent_index - rev_index > broken_line_length:
269
child_line_col_index = \
270
_find_free_column(columns,
274
_mark_column_as_used(columns,
275
child_line_col_index,
278
# Recall _line_col_search_order to reset it back to
281
_line_col_search_order(columns,
284
parent_col_line_index = \
285
_find_free_column(columns,
289
_mark_column_as_used(columns,
290
parent_col_line_index,
292
lines.append((rev_index,
294
(child_line_col_index,
295
parent_col_line_index)))
297
line_col_index = col_index
298
if parent_index - rev_index >1:
299
line_range = range(rev_index + 1, parent_index)
301
_find_free_column(columns,
305
_mark_column_as_used(columns,
308
lines.append((rev_index,
312
progress_bar.finished()
314
update_root_progress(5)
315
progress_bar = ui.ui_factory.nested_progress_bar()
317
progress_bar.update(msg="Pretifying graph", total=len(lines))
318
for i, (child_index, parent_index, line_col_indexes) in enumerate(lines):
320
progress_bar.update(current=i)
321
(child_col_index, child_color) = linegraph[child_index][1]
322
(parent_col_index, parent_color) = linegraph[parent_index][1]
324
if len(line_col_indexes) == 1:
325
if parent_index - child_index == 1:
326
linegraph[child_index][2].append(
331
# line from the child's column to the lines column
332
linegraph[child_index][2].append(
336
# lines down the line's column
337
for line_part_index in range(child_index+1, parent_index-1):
338
linegraph[line_part_index][2].append(
339
(line_col_indexes[0],
342
# line from the line's column to the parent's column
343
linegraph[parent_index-1][2].append(
344
(line_col_indexes[0],
349
# line from the child's column to the lines column
350
linegraph[child_index][2].append(
355
linegraph[child_index+1][2].append(
356
(line_col_indexes[0],
361
linegraph[parent_index-2][2].append(
365
# line from the line's column to the parent's column
366
linegraph[parent_index-1][2].append(
367
(line_col_indexes[1],
371
progress_bar.finished()
372
return (linegraph, revid_index, len(columns))
374
return (linegraph, revid_index, 0)
377
def _branch_line_col_search_order(columns, parent_col_index):
378
for col_index in range(parent_col_index, len(columns)):
380
for col_index in range(parent_col_index-1, -1, -1):
383
def _line_col_search_order(columns, parent_col_index, child_col_index):
384
if parent_col_index is not None:
385
max_index = max(parent_col_index, child_col_index)
386
min_index = min(parent_col_index, child_col_index)
387
for col_index in range(max_index, min_index -1, -1):
390
max_index = child_col_index
391
min_index = child_col_index
392
yield child_col_index
394
while max_index + i < len(columns) or \
396
if max_index + i < len(columns):
398
if min_index - i > -1:
402
def _find_free_column(columns, empty_column, col_search_order, line_range):
403
for col_index in col_search_order:
55
for (rev_index, (sequence_number,
59
end_of_merge)) in enumerate(merge_sorted_revisions):
61
revid_index[revid] = rev_index
63
branch_id = revno_sequence[0:-1]
66
if branch_id not in branch_lines:
67
branch_line = {"branch_id": branch_id,
69
"min_index": rev_index,
71
branch_lines[branch_id] = branch_line
73
branch_line = branch_lines[branch_id]
74
branch_line["rev_indexes"].append(rev_index)
75
if rev_index > branch_line["max_index"]:
76
branch_line["max_index"] = rev_index
78
parents = graph_parents[revid]
79
for parent_revid in parents:
80
graph_children[parent_revid].append(revid)
82
linegraph.append([revid,
89
branch_ids = branch_lines.keys()
90
def branch_id_cmp(x, y):
95
return cmp(len_x, len_y)
97
branch_ids.sort(branch_id_cmp)
99
empty_column = [False for i in range(len(graph_parents))]
100
columns = [list(empty_column)]
103
for branch_id in branch_ids:
104
branch_line = branch_lines[branch_id]
107
if len(branch_id) > 1:
108
parent_branch_id = branch_id[0:-2]
109
parent_col_index = branch_lines[parent_branch_id]["col_index"]
111
branch_line["col_index"] = append_line(columns,
112
(branch_line["min_index"],
113
branch_line["max_index"]),
116
color = reduce(lambda x, y: x+y, branch_id, 0)
117
col_index = branch_line["col_index"]
118
node = (col_index, color)
120
for rev_index in branch_line["rev_indexes"]:
125
end_of_merge) = merge_sorted_revisions[rev_index]
127
linegraph[rev_index][1] = node
128
linegraph[rev_index][4] = graph_children[revid]
130
for parent_revid in graph_parents[revid]:
131
if parent_revid in revid_index:
132
parent_index = revid_index[parent_revid]
133
parent_revno = merge_sorted_revisions[parent_index][3]
134
parent_branch_id = parent_revno[0:-1]
136
if branch_id != parent_branch_id and \
137
parent_index - rev_index > 1:
138
col_index = append_line(columns,
139
(rev_index+1, parent_index-1),
141
branch_line["col_index"])
142
lines.append((rev_index, parent_index, col_index))
144
for (child_index, parent_index, line_col_index) in lines:
145
child_col_index = linegraph[child_index][1][0]
147
parent_node = linegraph[parent_index][1]
148
parent_col_index = parent_node[0]
149
color = parent_node[1]
152
linegraph[child_index][2].append(
156
for line_part_index in range(child_index+1, parent_index-1):
157
linegraph[line_part_index][2].append(
161
linegraph[parent_index-1][2].append(
166
linegraph[child_index][2].append(
170
for line_part_index in range(child_index+1, parent_index):
171
linegraph[line_part_index][2].append(
177
return (linegraph, revid_index)
179
def append_line(columns, line, empty_column, starting_col_index):
180
line_range = range(line[0], line[1]+1)
181
col_order = range(starting_col_index, -1, -1) + \
182
range(starting_col_index+1, len(columns))
183
for col_index in col_order:
404
184
column = columns[col_index]
405
185
has_overlaping_line = False
406
186
for row_index in line_range: