diff options
Diffstat (limited to 'ui-log.c')
-rw-r--r-- | ui-log.c | 131 |
1 files changed, 120 insertions, 11 deletions
@@ -12,7 +12,7 @@ | |||
12 | #include "ui-shared.h" | 12 | #include "ui-shared.h" |
13 | #include "argv-array.h" | 13 | #include "argv-array.h" |
14 | 14 | ||
15 | static int files, add_lines, rem_lines; | 15 | static int files, add_lines, rem_lines, lines_counted; |
16 | 16 | ||
17 | /* | 17 | /* |
18 | * The list of available column colors in the commit graph. | 18 | * The list of available column colors in the commit graph. |
@@ -67,7 +67,7 @@ void show_commit_decorations(struct commit *commit) | |||
67 | strncpy(buf, deco->name + 11, sizeof(buf) - 1); | 67 | strncpy(buf, deco->name + 11, sizeof(buf) - 1); |
68 | cgit_log_link(buf, NULL, "branch-deco", buf, NULL, | 68 | cgit_log_link(buf, NULL, "branch-deco", buf, NULL, |
69 | ctx.qry.vpath, 0, NULL, NULL, | 69 | ctx.qry.vpath, 0, NULL, NULL, |
70 | ctx.qry.showmsg); | 70 | ctx.qry.showmsg, 0); |
71 | } | 71 | } |
72 | else if (starts_with(deco->name, "tag: refs/tags/")) { | 72 | else if (starts_with(deco->name, "tag: refs/tags/")) { |
73 | strncpy(buf, deco->name + 15, sizeof(buf) - 1); | 73 | strncpy(buf, deco->name + 15, sizeof(buf) - 1); |
@@ -84,7 +84,7 @@ void show_commit_decorations(struct commit *commit) | |||
84 | cgit_log_link(buf, NULL, "remote-deco", NULL, | 84 | cgit_log_link(buf, NULL, "remote-deco", NULL, |
85 | sha1_to_hex(commit->object.sha1), | 85 | sha1_to_hex(commit->object.sha1), |
86 | ctx.qry.vpath, 0, NULL, NULL, | 86 | ctx.qry.vpath, 0, NULL, NULL, |
87 | ctx.qry.showmsg); | 87 | ctx.qry.showmsg, 0); |
88 | } | 88 | } |
89 | else { | 89 | else { |
90 | strncpy(buf, deco->name, sizeof(buf) - 1); | 90 | strncpy(buf, deco->name, sizeof(buf) - 1); |
@@ -98,6 +98,74 @@ next: | |||
98 | html("</span>"); | 98 | html("</span>"); |
99 | } | 99 | } |
100 | 100 | ||
101 | static void handle_rename(struct diff_filepair *pair) | ||
102 | { | ||
103 | /* | ||
104 | * After we have seen a rename, we generate links to the previous | ||
105 | * name of the file so that commit & diff views get fed the path | ||
106 | * that is correct for the commit they are showing, avoiding the | ||
107 | * need to walk the entire history leading back to every commit we | ||
108 | * show in order detect renames. | ||
109 | */ | ||
110 | if (0 != strcmp(ctx.qry.vpath, pair->two->path)) { | ||
111 | free(ctx.qry.vpath); | ||
112 | ctx.qry.vpath = xstrdup(pair->two->path); | ||
113 | } | ||
114 | inspect_files(pair); | ||
115 | } | ||
116 | |||
117 | static int show_commit(struct commit *commit, struct rev_info *revs) | ||
118 | { | ||
119 | struct commit_list *parents = commit->parents; | ||
120 | struct commit *parent; | ||
121 | int found = 0, saved_fmt; | ||
122 | unsigned saved_flags = revs->diffopt.flags; | ||
123 | |||
124 | |||
125 | /* Always show if we're not in "follow" mode with a single file. */ | ||
126 | if (!ctx.qry.follow) | ||
127 | return 1; | ||
128 | |||
129 | /* | ||
130 | * In "follow" mode, we don't show merges. This is consistent with | ||
131 | * "git log --follow -- <file>". | ||
132 | */ | ||
133 | if (parents && parents->next) | ||
134 | return 0; | ||
135 | |||
136 | /* | ||
137 | * If this is the root commit, do what rev_info tells us. | ||
138 | */ | ||
139 | if (!parents) | ||
140 | return revs->show_root_diff; | ||
141 | |||
142 | /* When we get here we have precisely one parent. */ | ||
143 | parent = parents->item; | ||
144 | parse_commit(parent); | ||
145 | |||
146 | files = 0; | ||
147 | add_lines = 0; | ||
148 | rem_lines = 0; | ||
149 | |||
150 | DIFF_OPT_SET(&revs->diffopt, RECURSIVE); | ||
151 | diff_tree_sha1(parent->tree->object.sha1, | ||
152 | commit->tree->object.sha1, | ||
153 | "", &revs->diffopt); | ||
154 | diffcore_std(&revs->diffopt); | ||
155 | |||
156 | found = !diff_queue_is_empty(); | ||
157 | saved_fmt = revs->diffopt.output_format; | ||
158 | revs->diffopt.output_format = DIFF_FORMAT_CALLBACK; | ||
159 | revs->diffopt.format_callback = cgit_diff_tree_cb; | ||
160 | revs->diffopt.format_callback_data = handle_rename; | ||
161 | diff_flush(&revs->diffopt); | ||
162 | revs->diffopt.output_format = saved_fmt; | ||
163 | revs->diffopt.flags = saved_flags; | ||
164 | |||
165 | lines_counted = 1; | ||
166 | return found; | ||
167 | } | ||
168 | |||
101 | static void print_commit(struct commit *commit, struct rev_info *revs) | 169 | static void print_commit(struct commit *commit, struct rev_info *revs) |
102 | { | 170 | { |
103 | struct commitinfo *info; | 171 | struct commitinfo *info; |
@@ -177,7 +245,8 @@ static void print_commit(struct commit *commit, struct rev_info *revs) | |||
177 | cgit_print_age(commit->date, TM_WEEK * 2, FMT_SHORTDATE); | 245 | cgit_print_age(commit->date, TM_WEEK * 2, FMT_SHORTDATE); |
178 | } | 246 | } |
179 | 247 | ||
180 | if (ctx.repo->enable_log_filecount || ctx.repo->enable_log_linecount) { | 248 | if (!lines_counted && (ctx.repo->enable_log_filecount || |
249 | ctx.repo->enable_log_linecount)) { | ||
181 | files = 0; | 250 | files = 0; |
182 | add_lines = 0; | 251 | add_lines = 0; |
183 | rem_lines = 0; | 252 | rem_lines = 0; |
@@ -325,7 +394,17 @@ void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *pattern | |||
325 | } | 394 | } |
326 | } | 395 | } |
327 | } | 396 | } |
328 | if (commit_graph) { | 397 | |
398 | if (!path || !ctx.cfg.enable_follow_links) { | ||
399 | /* | ||
400 | * If we don't have a path, "follow" is a no-op so make sure | ||
401 | * the variable is set to false to avoid needing to check | ||
402 | * both this and whether we have a path everywhere. | ||
403 | */ | ||
404 | ctx.qry.follow = 0; | ||
405 | } | ||
406 | |||
407 | if (commit_graph && !ctx.qry.follow) { | ||
329 | argv_array_push(&rev_argv, "--graph"); | 408 | argv_array_push(&rev_argv, "--graph"); |
330 | argv_array_push(&rev_argv, "--color"); | 409 | argv_array_push(&rev_argv, "--color"); |
331 | graph_set_column_colors(column_colors_html, | 410 | graph_set_column_colors(column_colors_html, |
@@ -337,6 +416,8 @@ void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *pattern | |||
337 | else if (commit_sort == 2) | 416 | else if (commit_sort == 2) |
338 | argv_array_push(&rev_argv, "--topo-order"); | 417 | argv_array_push(&rev_argv, "--topo-order"); |
339 | 418 | ||
419 | if (path && ctx.qry.follow) | ||
420 | argv_array_push(&rev_argv, "--follow"); | ||
340 | argv_array_push(&rev_argv, "--"); | 421 | argv_array_push(&rev_argv, "--"); |
341 | if (path) | 422 | if (path) |
342 | argv_array_push(&rev_argv, path); | 423 | argv_array_push(&rev_argv, path); |
@@ -347,10 +428,17 @@ void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *pattern | |||
347 | rev.verbose_header = 1; | 428 | rev.verbose_header = 1; |
348 | rev.show_root_diff = 0; | 429 | rev.show_root_diff = 0; |
349 | rev.ignore_missing = 1; | 430 | rev.ignore_missing = 1; |
431 | rev.simplify_history = 1; | ||
350 | setup_revisions(rev_argv.argc, rev_argv.argv, &rev, NULL); | 432 | setup_revisions(rev_argv.argc, rev_argv.argv, &rev, NULL); |
351 | load_ref_decorations(DECORATE_FULL_REFS); | 433 | load_ref_decorations(DECORATE_FULL_REFS); |
352 | rev.show_decorations = 1; | 434 | rev.show_decorations = 1; |
353 | rev.grep_filter.regflags |= REG_ICASE; | 435 | rev.grep_filter.regflags |= REG_ICASE; |
436 | |||
437 | rev.diffopt.detect_rename = 1; | ||
438 | rev.diffopt.rename_limit = ctx.cfg.renamelimit; | ||
439 | if (ctx.qry.ignorews) | ||
440 | DIFF_XDL_SET(&rev.diffopt, IGNORE_WHITESPACE); | ||
441 | |||
354 | compile_grep_patterns(&rev.grep_filter); | 442 | compile_grep_patterns(&rev.grep_filter); |
355 | prepare_revision_walk(&rev); | 443 | prepare_revision_walk(&rev); |
356 | 444 | ||
@@ -368,11 +456,12 @@ void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *pattern | |||
368 | cgit_log_link(ctx.qry.showmsg ? "Collapse" : "Expand", NULL, | 456 | cgit_log_link(ctx.qry.showmsg ? "Collapse" : "Expand", NULL, |
369 | NULL, ctx.qry.head, ctx.qry.sha1, | 457 | NULL, ctx.qry.head, ctx.qry.sha1, |
370 | ctx.qry.vpath, ctx.qry.ofs, ctx.qry.grep, | 458 | ctx.qry.vpath, ctx.qry.ofs, ctx.qry.grep, |
371 | ctx.qry.search, ctx.qry.showmsg ? 0 : 1); | 459 | ctx.qry.search, ctx.qry.showmsg ? 0 : 1, |
460 | ctx.qry.follow); | ||
372 | html(")"); | 461 | html(")"); |
373 | } | 462 | } |
374 | html("</th><th class='left'>Author</th>"); | 463 | html("</th><th class='left'>Author</th>"); |
375 | if (commit_graph) | 464 | if (rev.graph) |
376 | html("<th class='left'>Age</th>"); | 465 | html("<th class='left'>Age</th>"); |
377 | if (ctx.repo->enable_log_filecount) { | 466 | if (ctx.repo->enable_log_filecount) { |
378 | html("<th class='left'>Files</th>"); | 467 | html("<th class='left'>Files</th>"); |
@@ -388,13 +477,30 @@ void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *pattern | |||
388 | ofs = 0; | 477 | ofs = 0; |
389 | 478 | ||
390 | for (i = 0; i < ofs && (commit = get_revision(&rev)) != NULL; i++) { | 479 | for (i = 0; i < ofs && (commit = get_revision(&rev)) != NULL; i++) { |
480 | if (show_commit(commit, &rev)) | ||
481 | i++; | ||
391 | free_commit_buffer(commit); | 482 | free_commit_buffer(commit); |
392 | free_commit_list(commit->parents); | 483 | free_commit_list(commit->parents); |
393 | commit->parents = NULL; | 484 | commit->parents = NULL; |
394 | } | 485 | } |
395 | 486 | ||
396 | for (i = 0; i < cnt && (commit = get_revision(&rev)) != NULL; i++) { | 487 | for (i = 0; i < cnt && (commit = get_revision(&rev)) != NULL; i++) { |
397 | print_commit(commit, &rev); | 488 | /* |
489 | * In "follow" mode, we must count the files and lines the | ||
490 | * first time we invoke diff on a given commit, and we need | ||
491 | * to do that to see if the commit touches the path we care | ||
492 | * about, so we do it in show_commit. Hence we must clear | ||
493 | * lines_counted here. | ||
494 | * | ||
495 | * This has the side effect of avoiding running diff twice | ||
496 | * when we are both following renames and showing file | ||
497 | * and/or line counts. | ||
498 | */ | ||
499 | lines_counted = 0; | ||
500 | if (show_commit(commit, &rev)) { | ||
501 | i++; | ||
502 | print_commit(commit, &rev); | ||
503 | } | ||
398 | free_commit_buffer(commit); | 504 | free_commit_buffer(commit); |
399 | free_commit_list(commit->parents); | 505 | free_commit_list(commit->parents); |
400 | commit->parents = NULL; | 506 | commit->parents = NULL; |
@@ -406,7 +512,8 @@ void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *pattern | |||
406 | cgit_log_link("[prev]", NULL, NULL, ctx.qry.head, | 512 | cgit_log_link("[prev]", NULL, NULL, ctx.qry.head, |
407 | ctx.qry.sha1, ctx.qry.vpath, | 513 | ctx.qry.sha1, ctx.qry.vpath, |
408 | ofs - cnt, ctx.qry.grep, | 514 | ofs - cnt, ctx.qry.grep, |
409 | ctx.qry.search, ctx.qry.showmsg); | 515 | ctx.qry.search, ctx.qry.showmsg, |
516 | ctx.qry.follow); | ||
410 | html("</li>"); | 517 | html("</li>"); |
411 | } | 518 | } |
412 | if ((commit = get_revision(&rev)) != NULL) { | 519 | if ((commit = get_revision(&rev)) != NULL) { |
@@ -414,14 +521,16 @@ void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *pattern | |||
414 | cgit_log_link("[next]", NULL, NULL, ctx.qry.head, | 521 | cgit_log_link("[next]", NULL, NULL, ctx.qry.head, |
415 | ctx.qry.sha1, ctx.qry.vpath, | 522 | ctx.qry.sha1, ctx.qry.vpath, |
416 | ofs + cnt, ctx.qry.grep, | 523 | ofs + cnt, ctx.qry.grep, |
417 | ctx.qry.search, ctx.qry.showmsg); | 524 | ctx.qry.search, ctx.qry.showmsg, |
525 | ctx.qry.follow); | ||
418 | html("</li>"); | 526 | html("</li>"); |
419 | } | 527 | } |
420 | html("</ul>"); | 528 | html("</ul>"); |
421 | } else if ((commit = get_revision(&rev)) != NULL) { | 529 | } else if ((commit = get_revision(&rev)) != NULL) { |
422 | htmlf("<tr class='nohover'><td colspan='%d'>", columns); | 530 | htmlf("<tr class='nohover'><td colspan='%d'>", columns); |
423 | cgit_log_link("[...]", NULL, NULL, ctx.qry.head, NULL, | 531 | cgit_log_link("[...]", NULL, NULL, ctx.qry.head, NULL, |
424 | ctx.qry.vpath, 0, NULL, NULL, ctx.qry.showmsg); | 532 | ctx.qry.vpath, 0, NULL, NULL, ctx.qry.showmsg, |
533 | ctx.qry.follow); | ||
425 | html("</td></tr>\n"); | 534 | html("</td></tr>\n"); |
426 | } | 535 | } |
427 | 536 | ||