43
36
curved, kinked, etc.) and to pick the actual colours for each index.
39
# We get the mainline so we can pass it to merge_sort to make merge_sort
41
mainline = branch.revision_history()
42
graph_parents = branch.repository.get_revision_graph(start)
48
43
graph_children = {}
49
for (revid, parent_revids) in graph.iter_ancestry(start_revs):
50
if parent_revids is None:
53
if parent_revids == (NULL_REVISION,):
54
graph_parents[revid] = ()
56
graph_parents[revid] = parent_revids
57
for parent in parent_revids:
58
graph_children.setdefault(parent, []).append(revid)
59
graph_children.setdefault(revid, [])
61
for ghost_child in graph_children[ghost]:
62
graph_parents[ghost_child] = [p for p in graph_parents[ghost_child]
64
graph_parents["top:"] = start_revs
66
if len(graph_parents)>0:
67
merge_sorted_revisions = merge_sort(
72
merge_sorted_revisions = ()
75
merge_sorted_revisions = [elem for elem in merge_sorted_revisions \
78
assert merge_sorted_revisions[0][1] == "top:"
79
merge_sorted_revisions = merge_sorted_revisions[1:]
44
for revid in graph_parents.iterkeys():
45
graph_children[revid] = []
47
merge_sorted_revisions = merge_sort(
99
75
end_of_merge)) in enumerate(merge_sorted_revisions):
100
if maxnum and rev_index >= maxnum:
102
77
revid_index[revid] = rev_index
78
revno_index[revno_sequence] = rev_index
80
branch_id = revno_sequence[0:-1]
83
if branch_id not in branch_lines:
84
branch_line = [[], # BL_REV_INDEXES
85
rev_index, # BL_MIN_INDEX
88
branch_lines[branch_id] = branch_line
90
branch_line = branch_lines[branch_id]
92
branch_line[BL_REV_INDEXES].append(rev_index)
93
if rev_index > branch_line[BL_MAX_INDEX]:
94
branch_line[BL_MAX_INDEX] = rev_index
104
96
parents = graph_parents[revid]
97
for parent_revid in parents:
98
graph_children[parent_revid].append(revid)
105
100
linegraph.append([revid,
113
revno_index[revno_sequence] = rev_index
115
branch_id = revno_sequence[0:-1]
118
if branch_id not in branch_lines:
120
branch_lines[branch_id] = branch_line
122
branch_line = branch_lines[branch_id]
124
branch_line.append(rev_index)
127
branch_ids = branch_lines.keys()
129
def branch_id_cmp(x, y):
130
"""Compaire branch_id's first by the number of digits, then reversed
136
return cmp(len_x, len_y)
138
branch_ids.sort(branch_id_cmp)
139
# This will hold a tuple of (child_index, parent_index, col_index) for each
140
# line that needs to be drawn. If col_index is not none, then the line is
141
# drawn along that column, else the the line can be drawn directly between
142
# the child and parent because either the child and parent are in the same
143
# branch line, or the child and parent are 1 row apart.
145
empty_column = [False for i in range(len(graph_parents))]
146
# This will hold a bit map for each cell. If the cell is true, then the
147
# cell allready contains a node or line. This use when deciding what column
148
# to place a branch line or line in, without it overlaping something else.
149
columns = [list(empty_column)]
152
for branch_id in branch_ids:
153
branch_line = branch_lines[branch_id]
155
# Find the col_index for the direct parent branch. This will be the
156
# starting point when looking for a free column.
159
if len(branch_id) > 1:
160
parent_revno = branch_id[0:-1]
161
if parent_revno in revno_index:
162
parent_index = revno_index[parent_revno]
163
parent_node = linegraph[parent_index][1]
165
parent_col_index = parent_node[0]
107
branch_ids = branch_lines.keys()
109
def branch_id_cmp(x, y):
110
"""Compaire branch_id's first by the number of digits, then reversed
116
return cmp(len_x, len_y)
118
branch_ids.sort(branch_id_cmp)
119
# This will hold a tuple of (child_index, parent_index, col_index) for each
120
# line that needs to be drawn. If col_index is not none, then the line is
121
# drawn along that column, else the the line can be drawn directly between
122
# the child and parent because either the child and parent are in the same
123
# branch line, or the child and parent are 1 row apart.
125
empty_column = [False for i in range(len(graph_parents))]
126
# This will hold a bit map for each cell. If the cell is true, then the
127
# cell allready contains a node or line. This use when deciding what column
128
# to place a branch line or line in, without it overlaping something else.
129
columns = [list(empty_column)]
132
for branch_id in branch_ids:
133
branch_line = branch_lines[branch_id]
135
# Find the col_index for the direct parent branch. This will be the
136
# starting point when looking for a free column.
138
if len(branch_id) > 1:
139
parent_branch_id = branch_id[0:-2]
140
parent_col_index = branch_lines[parent_branch_id][BL_COL_INDEX]
141
parent_revno = branch_id[0:-1]
142
if parent_revno in revno_index:
143
parent_index = revno_index[parent_revno]
144
branch_line[BL_MAX_INDEX] = parent_index - 1
146
col_search_order = _branch_line_col_search_order(columns,
148
branch_line[BL_COL_INDEX] = _append_line(columns,
149
(branch_line[BL_MIN_INDEX],
150
branch_line[BL_MAX_INDEX]),
153
color = reduce(lambda x, y: x+y, branch_id, 0)
154
col_index = branch_line[BL_COL_INDEX]
155
node = (col_index, color)
157
for rev_index in branch_line[BL_REV_INDEXES]:
162
end_of_merge) = merge_sorted_revisions[rev_index]
164
linegraph[rev_index][1] = node
165
linegraph[rev_index][4] = graph_children[revid]
167
for parent_revid in graph_parents[revid]:
168
if parent_revid in revid_index:
169
parent_index = revid_index[parent_revid]
170
parent_revno = merge_sorted_revisions[parent_index][3]
171
parent_branch_id = parent_revno[0:-1]
173
is_direct_parent_line = False
174
if len(branch_id) > 1:
175
if parent_revno == branch_id[0:-1]:
176
is_direct_parent_line = True
168
col_search_order = _branch_line_col_search_order(columns,
170
color = reduce(lambda x, y: x+y, branch_id, 0)
174
last_rev_index = None
175
for rev_index in branch_line:
177
if broken_line_length and \
178
rev_index - last_rev_index > broken_line_length:
179
line_range.append(last_rev_index+1)
180
line_range.append(rev_index-1)
182
line_range.extend(range(last_rev_index+1, rev_index))
184
line_range.append(rev_index)
185
last_rev_index = rev_index
188
if broken_line_length and \
189
parent_index - last_rev_index > broken_line_length:
190
line_range.append(last_rev_index+1)
192
line_range.extend(range(last_rev_index+1, parent_index))
194
col_index = _find_free_column(columns,
198
node = (col_index, color)
199
for rev_index in branch_line:
200
linegraph[rev_index][1] = node
201
columns[col_index][rev_index] = True
203
for rev_index in branch_line:
208
end_of_merge) = merge_sorted_revisions[rev_index]
210
linegraph[rev_index][4] = graph_children[revid]
211
col_index = linegraph[rev_index][1][0]
213
for parent_revid in graph_parents[revid]:
214
if parent_revid in revid_index:
216
parent_index = revid_index[parent_revid]
178
# A line only needs it's own column if it is going from
179
# one branch line to another, it's not the line to the
180
# direct parent, and if it is longer than one row.
181
if branch_id != parent_branch_id and \
182
parent_index - rev_index > 1 and \
183
not is_direct_parent_line:
217
184
parent_node = linegraph[parent_index][1]
219
186
parent_col_index = parent_node[0]
222
189
col_search_order = \
223
190
_line_col_search_order(columns,
224
191
parent_col_index,
227
# If this line is really long, break it.
228
if len(branch_id) > 0 and \
229
broken_line_length and \
230
parent_index - rev_index > broken_line_length:
231
child_line_col_index = \
232
_find_free_column(columns,
236
_mark_column_as_used(columns,
237
child_line_col_index,
240
# Recall _line_col_search_order to reset it back to
243
_line_col_search_order(columns,
246
parent_col_line_index = \
247
_find_free_column(columns,
251
_mark_column_as_used(columns,
252
parent_col_line_index,
254
lines.append((rev_index,
256
(child_line_col_index,
257
parent_col_line_index)))
259
line_col_index = col_index
260
if parent_index - rev_index >1:
261
line_range = range(rev_index + 1, parent_index)
263
_find_free_column(columns,
267
_mark_column_as_used(columns,
270
lines.append((rev_index,
274
for (child_index, parent_index, line_col_indexes) in lines:
275
(child_col_index, child_color) = linegraph[child_index][1]
276
(parent_col_index, parent_color) = linegraph[parent_index][1]
278
if len(line_col_indexes) == 1:
279
if parent_index - child_index == 1:
280
linegraph[child_index][2].append(
285
# line from the child's column to the lines column
286
linegraph[child_index][2].append(
290
# lines down the line's column
291
for line_part_index in range(child_index+1, parent_index-1):
292
linegraph[line_part_index][2].append(
293
(line_col_indexes[0],
296
# line from the line's column to the parent's column
297
linegraph[parent_index-1][2].append(
298
(line_col_indexes[0],
303
# line from the child's column to the lines column
304
linegraph[child_index][2].append(
309
linegraph[child_index+1][2].append(
310
(line_col_indexes[0],
315
linegraph[parent_index-2][2].append(
319
# line from the line's column to the parent's column
320
linegraph[parent_index-1][2].append(
321
(line_col_indexes[1],
324
return (linegraph, revid_index, len(columns))
326
return (linegraph, revid_index, 0)
192
branch_line[BL_COL_INDEX])
193
col_index = _append_line(columns,
194
(rev_index+1, parent_index-1),
197
lines.append((rev_index, parent_index, col_index))
199
for (child_index, parent_index, line_col_index) in lines:
200
child_col_index = linegraph[child_index][1][0]
202
parent_node = linegraph[parent_index][1]
203
parent_col_index = parent_node[0]
204
color = parent_node[1]
207
# line from the child's column to the lines column
208
linegraph[child_index][2].append(
212
# lines down the line's column
213
for line_part_index in range(child_index+1, parent_index-1):
214
linegraph[line_part_index][2].append(
218
# line from the line's column to the parent's column
219
linegraph[parent_index-1][2].append(
224
# lines down the child's column
225
for line_part_index in range(child_index, parent_index-1):
226
linegraph[line_part_index][2].append(
230
# line from the child's column to the parent's column
231
linegraph[parent_index-1][2].append(
236
return (linegraph, revid_index)
329
238
def _branch_line_col_search_order(columns, parent_col_index):
330
for col_index in range(parent_col_index, len(columns)):
332
for col_index in range(parent_col_index-1, -1, -1):
239
return range(parent_col_index, len(columns)) + \
240
range(parent_col_index-1, -1, -1)
335
242
def _line_col_search_order(columns, parent_col_index, child_col_index):
243
dest_col_indexes = []
336
244
if parent_col_index is not None:
337
max_index = max(parent_col_index, child_col_index)
338
min_index = min(parent_col_index, child_col_index)
339
for col_index in range(max_index, min_index -1, -1):
245
dest_col_indexes.append(parent_col_index)
342
max_index = child_col_index
343
min_index = child_col_index
344
yield child_col_index
247
dest_col_indexes.append(child_col_index)
248
dest_col_indexes.append(child_col_index)
249
dest_col_indexes.sort()
250
col_search_order = range(dest_col_indexes[1], dest_col_indexes[0] -1, -1)
346
while max_index + i < len(columns) or \
348
if max_index + i < len(columns):
350
if min_index - i > -1:
252
while dest_col_indexes[1] + i < len(columns) or \
253
dest_col_indexes[0] - i > -1:
254
if dest_col_indexes[1] + i < len(columns):
255
col_search_order.append(dest_col_indexes[1] + i)
256
if dest_col_indexes[0] - i > -1:
257
col_search_order.append(dest_col_indexes[0] - i)
259
return col_search_order
354
def _find_free_column(columns, empty_column, col_search_order, line_range):
261
def _append_line(columns, line, empty_column, col_search_order):
262
line_range = range(line[0], line[1]+1)
355
264
for col_index in col_search_order:
356
265
column = columns[col_index]
357
266
has_overlaping_line = False