-sub flush_diff ($$$) {
- my ($dst, $ctx, $linkify) = @_;
- my $diff = $ctx->{-diff};
- my $spfx = $ctx->{-spfx};
- my $state = DSTATE_INIT;
- my $dctx = { Q => '' }; # {}, keys: oid_a, oid_b, path_a, path_b
-
- foreach my $s (@$diff) {
- if ($s =~ /^---$/) {
- to_state($dst, $state, DSTATE_STAT);
- $$dst .= $s;
- } elsif ($s =~ /^ / || ($s =~ /^$/ && $state >= DSTATE_CTX)) {
- # works for common cases, but not weird/long filenames
- if ($state == DSTATE_STAT &&
- $s =~ /^ (.+)( +\| .*\z)/s) {
- anchor0($dst, $ctx, $linkify, $1, $2) and next;
- } elsif ($state2class[$state]) {
- to_state($dst, $state, DSTATE_CTX);
- }
- $$dst .= $linkify->to_html($s);
- } elsif ($s =~ /^-- $/) { # email signature begins
- $state == DSTATE_INIT or
- to_state($dst, $state, DSTATE_INIT);
- $$dst .= $s;
- } elsif ($s =~ m!^diff --git ($PATH_A) ($PATH_B)$!o) {
- my ($pa, $pb) = ($1, $2);
- if ($state != DSTATE_HEAD) {
- to_state($dst, $state, DSTATE_HEAD);
- }
- $pa = (split('/', git_unquote($pa), 2))[1];
- $pb = (split('/', git_unquote($pb), 2))[1];
- $dctx = {
- Q => "?b=".uri_escape_utf8($pb, UNSAFE),
- };
- if ($pa ne $pb) {
- $dctx->{Q} .= '&a='.
- uri_escape_utf8($pa, UNSAFE);
- }
- anchor1($dst, $ctx, $linkify, $pb, $s) and next;
- $$dst .= $linkify->to_html($s);
- } elsif ($s =~ s/^(index $OID_NULL\.\.)($OID_BLOB)\b//o) {
- $$dst .= $1 . oid($dctx, $spfx, $2);
- $dctx = { Q => '' };
- $$dst .= $linkify->to_html($s) ;
- } elsif ($s =~ s/^index ($OID_BLOB)(\.\.$OID_NULL)\b//o) {
- $$dst .= 'index ' . oid($dctx, $spfx, $1) . $2;
- $dctx = { Q => '' };
- $$dst .= $linkify->to_html($s);
- } elsif ($s =~ /^index ($OID_BLOB)\.\.($OID_BLOB)/o) {
- $dctx->{oid_a} = $1;
- $dctx->{oid_b} = $2;
- $$dst .= $linkify->to_html($s);
- } elsif ($s =~ s/^@@ (\S+) (\S+) @@//) {
- $$dst .= '</span>' if $state2class[$state];
- $$dst .= qq(<span\nclass="hunk">);
- $$dst .= diff_hunk($dctx, $spfx, $1, $2);
- $$dst .= '</span>';
- $state = DSTATE_CTX;
- $$dst .= $linkify->to_html($s);
- } elsif ($s =~ m!^--- (?:$PATH_A)!o ||
- $s =~ m!^\+{3} (?:$PATH_B)!o) {
- # color only (no oid link) if missing dctx->{oid_*}
- $state <= DSTATE_STAT and
- to_state($dst, $state, DSTATE_HEAD);
- $$dst .= $linkify->to_html($s);
- } elsif ($s =~ /^\+/) {
- if ($state != DSTATE_ADD && $state > DSTATE_STAT) {
- to_state($dst, $state, DSTATE_ADD);
+sub diff_before_or_after ($$) {
+ my ($ctx, $x) = @_;
+ my $linkify = $ctx->{-linkify};
+ my $dst = $ctx->{obuf};
+ my $anchors = exists($ctx->{-anchors}) ? 1 : 0;
+ for my $y (split(/(^---\n)/sm, $$x)) {
+ if ($y =~ /\A---\n\z/s) {
+ $$dst .= "---\n"; # all HTML is "\r\n" => "\n"
+ $anchors |= 2;
+ } elsif ($anchors == 3 && $y =~ /^ [0-9]+ files? changed, /sm) {
+ # ok, looks like a diffstat, go line-by-line:
+ for my $l (split(/^/m, $y)) {
+ if ($l =~ /^ (.+)( +\| .*\z)/s) {
+ anchor0($dst, $ctx, $1, $2) and next;
+ }
+ $$dst .= $linkify->to_html($l);