]> git.ipfire.org Git - thirdparty/git.git/commitdiff
diff: handle ANSI escape codes in prefix when calculating diffstat width
authorLorenzoPegorari <lorenzo.pegorari2002@gmail.com>
Fri, 27 Feb 2026 21:45:19 +0000 (22:45 +0100)
committerJunio C Hamano <gitster@pobox.com>
Fri, 27 Feb 2026 21:59:22 +0000 (13:59 -0800)
The diffstat width is calculated by taking the terminal width and
incorrectly subtracting the `strlen()` of `line_prefix`, instead of the
actual display width of `line_prefix`, which may contain ANSI escape
codes (e.g., ANSI-colored strings in `log --graph --stat`).

Utilize the display width instead, obtained via `utf8_strnwidth()` with
the flag `skip_ansi`.

Signed-off-by: LorenzoPegorari <lorenzo.pegorari2002@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
diff.c

diff --git a/diff.c b/diff.c
index a1961526c0dab1af182d4f400468bf5617f5175c..9c848ec5db54b2d831826cab6662ff786828153d 100644 (file)
--- a/diff.c
+++ b/diff.c
@@ -2713,7 +2713,9 @@ static void show_stats(struct diffstat_t *data, struct diff_options *options)
        count = i; /* where we can stop scanning in data->files[] */
 
        /*
-        * We have width = stat_width or term_columns() columns total.
+        * We have width = stat_width or term_columns() columns total minus the
+        * length of line_prefix skipping ANSI escape codes to get the display
+        * width (e.g., skip ANSI-colored strings in "log --graph --stat").
         * We want a maximum of min(max_len, stat_name_width) for the name part.
         * We want a maximum of min(max_change, stat_graph_width) for the +- part.
         * We also need 1 for " " and 4 + decimal_width(max_change)
@@ -2740,14 +2742,8 @@ static void show_stats(struct diffstat_t *data, struct diff_options *options)
         * separators and this message, this message will "overflow"
         * making the line longer than the maximum width.
         */
-
-       /*
-        * NEEDSWORK: line_prefix is often used for "log --graph" output
-        * and contains ANSI-colored string.  utf8_strnwidth() should be
-        * used to correctly count the display width instead of strlen().
-        */
        if (options->stat_width == -1)
-               width = term_columns() - strlen(line_prefix);
+               width = term_columns() - utf8_strnwidth(line_prefix, strlen(line_prefix), 1);
        else
                width = options->stat_width ? options->stat_width : 80;
        number_width = decimal_width(max_change) > number_width ?