Signed-off-by: Nguyễn Thái Ngọc Duy <pclouds@xxxxxxxxx> --- I got mad after tracing two consecutive red history lines in `git log --graph --oneline` back to their merge points, far far away. Yeah probably should fire up tig, or gitk or something. This may sound like a good thing to add, but I don't know how good it is compared to the good old 16 color palette, yet as I haven't tried it for long since it's just written. BTW anyone interested in bringing this type [1] of --graph to git? I tried the unicode box characters, but the vertical lines do not join with diagonal ones, making the graph a bit ugly (even though it's still better than ascii version) [1] https://github.com/magit/magit/issues/495#issuecomment-17480757 Documentation/rev-list-options.txt | 5 ++++- graph.c | 31 ++++++++++++++++++++++++++++++- graph.h | 8 +++++++- revision.c | 4 ++++ 4 files changed, 45 insertions(+), 3 deletions(-) diff --git a/Documentation/rev-list-options.txt b/Documentation/rev-list-options.txt index 5da7cf5..0a0c2f3 100644 --- a/Documentation/rev-list-options.txt +++ b/Documentation/rev-list-options.txt @@ -825,7 +825,7 @@ you would get an output like this: -xxxxxxx... 1st on a ----------------------------------------------------------------------- ---graph:: +--graph[=<options>]:: Draw a text-based graphical representation of the commit history on the left hand side of the output. This may cause extra lines to be printed in between commits, in order for the graph history @@ -836,6 +836,9 @@ This enables parent rewriting, see 'History Simplification' below. + This implies the `--topo-order` option by default, but the `--date-order` option may also be specified. ++ +The only supported option is `256colors` which allows more than 16 +colors for drawing the commit history. --show-linear-break[=<barrier>]:: When --graph is not used, all history branches are flattened diff --git a/graph.c b/graph.c index d4e8519..75375a1 100644 --- a/graph.c +++ b/graph.c @@ -78,6 +78,7 @@ static void graph_show_line_prefix(const struct diff_options *diffopt) static const char **column_colors; static unsigned short column_colors_max; +static int column_colors_step; void graph_set_column_colors(const char **colors, unsigned short colors_max) { @@ -234,10 +235,24 @@ void graph_setup_line_prefix(struct diff_options *diffopt) } -struct git_graph *graph_init(struct rev_info *opt) +struct git_graph *graph_init_with_options(struct rev_info *opt, const char *arg) { struct git_graph *graph = xmalloc(sizeof(struct git_graph)); + if (arg && !strcmp(arg, "256colors")) { + int i, start = 17, stop = 232; + column_colors_max = stop - start; + column_colors = + xmalloc((column_colors_max + 1) * sizeof(*column_colors)); + for (i = start; i < stop; i++) { + struct strbuf sb = STRBUF_INIT; + strbuf_addf(&sb, "\033[38;5;%dm", i); + column_colors[i - start] = strbuf_detach(&sb, NULL); + } + column_colors[column_colors_max] = xstrdup(GIT_COLOR_RESET); + /* ignore the closet 16 colors on either side for the next line */ + column_colors_step = 16; + } if (!column_colors) graph_set_column_colors(column_colors_ansi, column_colors_ansi_max); @@ -382,6 +397,20 @@ static unsigned short graph_get_current_column_color(const struct git_graph *gra */ static void graph_increment_column_color(struct git_graph *graph) { + if (column_colors_step) { + static int random_initialized; + int v; + + if (!random_initialized) { + srand((unsigned int)getpid()); + random_initialized = 1; + } + v = rand() % (column_colors_max - column_colors_step * 2); + graph->default_column_color += column_colors_step + v; + graph->default_column_color %= column_colors_max; + return; + } + graph->default_column_color = (graph->default_column_color + 1) % column_colors_max; } diff --git a/graph.h b/graph.h index af62339..8069aa4 100644 --- a/graph.h +++ b/graph.h @@ -40,7 +40,13 @@ void graph_set_column_colors(const char **colors, unsigned short colors_max); /* * Create a new struct git_graph. */ -struct git_graph *graph_init(struct rev_info *opt); +struct git_graph *graph_init_with_options(struct rev_info *opt, const char *arg); + +static inline struct git_graph *graph_init(struct rev_info *opt) +{ + return graph_init_with_options(opt, NULL); +} + /* * Update a git_graph with a new commit. diff --git a/revision.c b/revision.c index b37dbec..07bea54 100644 --- a/revision.c +++ b/revision.c @@ -1933,6 +1933,10 @@ static int handle_revision_opt(struct rev_info *revs, int argc, const char **arg revs->topo_order = 1; revs->rewrite_parents = 1; revs->graph = graph_init(revs); + } else if (skip_prefix(arg, "--graph=", &arg)) { + revs->topo_order = 1; + revs->rewrite_parents = 1; + revs->graph = graph_init_with_options(revs, arg); } else if (!strcmp(arg, "--root")) { revs->show_root_diff = 1; } else if (!strcmp(arg, "--no-commit-id")) { -- 2.8.2.524.g6ff3d78