static const char **column_colors;
static unsigned short column_colors_max;
+static unsigned int max_lanes = 15;
+
static void parse_graph_colors_config(struct strvec *colors, const char *string)
{
const char *end, *start;
struct strbuf prefix_buf;
};
+static inline int graph_needs_truncation(int lane)
+{
+ return lane >= max_lanes;
+}
+
static const char *diff_output_prefix_callback(struct diff_options *opt, void *data)
{
struct git_graph *graph = data;
{
struct commit_list *parent;
int max_new_columns;
- int i, seen_this, is_commit_in_columns;
+ int i, seen_this, is_commit_in_columns, max;
/*
* Swap graph->columns with graph->new_columns
}
}
+ /*
+ * Cap to the hard-coded limit.
+ * Allow commits from merges to align to the merged lane.
+ */
+ max = max_lanes * 2 + 2;
+ if (graph->width > max)
+ graph->width = max;
+
/*
* Shrink mapping_size to be the minimum necessary
*/
* Output a padding row, that leaves all branch lines unchanged
*/
for (i = 0; i < graph->num_new_columns; i++) {
+ if (graph_needs_truncation(i))
+ break;
graph_line_write_column(line, &graph->new_columns[i], '|');
graph_line_addch(line, ' ');
}
seen_this = 1;
graph_line_write_column(line, col, '|');
graph_line_addchars(line, ' ', graph->expansion_row);
+ } else if (seen_this && graph_needs_truncation(i)) {
+ break;
} else if (seen_this && (graph->expansion_row == 0)) {
/*
* This is the first line of the pre-commit output.
col = &graph->new_columns[j];
graph_line_write_column(line, col, '-');
+
+ /*
+ * Commit is at commit_index, each iteration move one lane to
+ * the right from the commit.
+ */
+ if (graph_needs_truncation(graph->commit_index + 1 + i))
+ break;
+
graph_line_write_column(line, col, (i == dashed_parents - 1) ? '.' : '-');
}
seen_this = 1;
graph_output_commit_char(graph, line);
+ if (graph_needs_truncation(i)) {
+ graph_line_addch(line, ' ');
+ break;
+ }
+
if (graph->num_parents > 2)
graph_draw_octopus_merge(graph, line);
+ } else if (graph_needs_truncation(i)) {
+ seen_this = 1;
+ break;
} else if (seen_this && (graph->edges_added > 1)) {
graph_line_write_column(line, col, '\\');
} else if (seen_this && (graph->edges_added == 1)) {
/*
* Update graph->state
+ *
+ * If the commit is a merge and the first parent is in a visible lane,
+ * then the GRAPH_POST_MERGE is needed to draw the merge lane.
+ *
+ * If the commit is over the truncation limit, but the first parent is on
+ * a visible lane, then we still need the merge lane but truncated.
+ *
+ * If both commit and first parent are over the truncation limit, then
+ * there's no need to draw the merge lane because it would work as a
+ * padding lane.
*/
- if (graph->num_parents > 1)
- graph_update_state(graph, GRAPH_POST_MERGE);
- else if (graph_is_mapping_correct(graph))
+ if (graph->num_parents > 1) {
+ if (!graph_needs_truncation(graph->commit_index)) {
+ graph_update_state(graph, GRAPH_POST_MERGE);
+ } else {
+ struct commit_list *p = first_interesting_parent(graph);
+ int lane;
+
+ /*
+ * graph->num_parents are found using first_interesting_parent
+ * and next_interesting_parent so it can't be a scenario
+ * where num_parents > 1 and there are no interesting parents
+ */
+ if (!p)
+ BUG("num_parents > 1 but no interesting parent");
+
+ lane = graph_find_new_column_by_commit(graph, p->item);
+
+ if (!graph_needs_truncation(lane))
+ graph_update_state(graph, GRAPH_POST_MERGE);
+ else if (graph_is_mapping_correct(graph))
+ graph_update_state(graph, GRAPH_PADDING);
+ else
+ graph_update_state(graph, GRAPH_COLLAPSING);
+ }
+ } else if (graph_is_mapping_correct(graph)) {
graph_update_state(graph, GRAPH_PADDING);
- else
+ } else {
graph_update_state(graph, GRAPH_COLLAPSING);
+ }
}
static const char merge_chars[] = {'/', '|', '\\'};
int par_column;
int idx = graph->merge_layout;
char c;
+ int truncated = 0;
seen_this = 1;
for (j = 0; j < graph->num_parents; j++) {
c = merge_chars[idx];
graph_line_write_column(line, &graph->new_columns[par_column], c);
+
+ /*
+ * j counts parents, it needs to be halved to be
+ * comparable with i. Don't truncate if there are
+ * no more lanes to print (end of the lane)
+ */
+ if (graph_needs_truncation(j / 2 + i) &&
+ j / 2 + i <= graph->num_columns) {
+ if ((j + i * 2) % 2 != 0)
+ graph_line_addch(line, ' ');
+ truncated = 1;
+ break;
+ }
+
if (idx == 2) {
- if (graph->edges_added > 0 || j < graph->num_parents - 1)
+ /*
+ * Check if the next lane needs truncation
+ * to avoid having the padding doubled
+ */
+ if (graph_needs_truncation((j + 1) / 2 + i) &&
+ j < graph->num_parents - 1) {
+ truncated = 1;
+ break;
+ } else if (graph->edges_added > 0 || j < graph->num_parents - 1)
graph_line_addch(line, ' ');
} else {
idx++;
}
parents = next_interesting_parent(graph, parents);
}
+ if (truncated)
+ break;
if (graph->edges_added == 0)
graph_line_addch(line, ' ');
-
+ } else if (graph_needs_truncation(i)) {
+ break;
} else if (seen_this) {
if (graph->edges_added > 0)
graph_line_write_column(line, col, '\\');
else
graph_line_write_column(line, col, '|');
- graph_line_addch(line, ' ');
+ /*
+ * If it's between two lanes and next would be truncated,
+ * don't add space padding.
+ */
+ if (!graph_needs_truncation(i + 1))
+ graph_line_addch(line, ' ');
} else {
graph_line_write_column(line, col, '|');
if (graph->merge_layout != 0 || i != graph->commit_index - 1) {
short used_horizontal = 0;
int horizontal_edge = -1;
int horizontal_edge_target = -1;
+ int truncated = 0;
/*
* Swap the mapping and old_mapping arrays
*/
for (i = 0; i < graph->mapping_size; i++) {
int target = graph->mapping[i];
- if (target < 0)
- graph_line_addch(line, ' ');
- else if (target * 2 == i)
- graph_line_write_column(line, &graph->new_columns[target], '|');
- else if (target == horizontal_edge_target &&
- i != horizontal_edge - 1) {
- /*
- * Set the mappings for all but the
- * first segment to -1 so that they
- * won't continue into the next line.
- */
- if (i != (target * 2)+3)
- graph->mapping[i] = -1;
- used_horizontal = 1;
- graph_line_write_column(line, &graph->new_columns[target], '_');
+
+ if (!truncated && graph_needs_truncation(i / 2)) {
+ truncated = 1;
+ }
+
+ if (target < 0) {
+ if (!truncated)
+ graph_line_addch(line, ' ');
+ } else if (target * 2 == i) {
+ if (!truncated)
+ graph_line_write_column(line, &graph->new_columns[target], '|');
+ } else if (target == horizontal_edge_target &&
+ i != horizontal_edge - 1) {
+ /*
+ * Set the mappings for all but the
+ * first segment to -1 so that they
+ * won't continue into the next line.
+ */
+ if (i != (target * 2)+3)
+ graph->mapping[i] = -1;
+ used_horizontal = 1;
+ if (!truncated)
+ graph_line_write_column(line, &graph->new_columns[target], '_');
} else {
if (used_horizontal && i < horizontal_edge)
graph->mapping[i] = -1;
- graph_line_write_column(line, &graph->new_columns[target], '/');
-
+ if (!truncated)
+ graph_line_write_column(line, &graph->new_columns[target], '/');
}
}
for (i = 0; i < graph->num_columns; i++) {
struct column *col = &graph->columns[i];
+ if (graph_needs_truncation(i))
+ break;
+
graph_line_write_column(&line, col, '|');
if (col->commit == graph->commit && graph->num_parents > 2) {