Merge branch 'jn/gitweb-side-by-side-diff'
* jn/gitweb-side-by-side-diff: gitweb: Add navigation to select side-by-side diff gitweb: Use href(-replay=>1,...) for formats links in "commitdiff" t9500: Add basic sanity tests for side-by-side diff in gitweb t9500: Add test for handling incomplete lines in diff by gitweb gitweb: Give side-by-side diff extra CSS styling gitweb: Add a feature to show side-by-side diff gitweb: Extract formatting of diff chunk header gitweb: Refactor diff body line classification
This commit is contained in:
commit
6fa625a6b7
@ -759,6 +759,7 @@ our @cgi_param_mapping = (
|
|||||||
extra_options => "opt",
|
extra_options => "opt",
|
||||||
search_use_regexp => "sr",
|
search_use_regexp => "sr",
|
||||||
ctag => "by_tag",
|
ctag => "by_tag",
|
||||||
|
diff_style => "ds",
|
||||||
# this must be last entry (for manipulation from JavaScript)
|
# this must be last entry (for manipulation from JavaScript)
|
||||||
javascript => "js"
|
javascript => "js"
|
||||||
);
|
);
|
||||||
@ -2225,93 +2226,119 @@ sub format_diff_cc_simplified {
|
|||||||
return $result;
|
return $result;
|
||||||
}
|
}
|
||||||
|
|
||||||
# format patch (diff) line (not to be used for diff headers)
|
sub diff_line_class {
|
||||||
sub format_diff_line {
|
my ($line, $from, $to) = @_;
|
||||||
|
|
||||||
|
# ordinary diff
|
||||||
|
my $num_sign = 1;
|
||||||
|
# combined diff
|
||||||
|
if ($from && $to && ref($from->{'href'}) eq "ARRAY") {
|
||||||
|
$num_sign = scalar @{$from->{'href'}};
|
||||||
|
}
|
||||||
|
|
||||||
|
my @diff_line_classifier = (
|
||||||
|
{ regexp => qr/^\@\@{$num_sign} /, class => "chunk_header"},
|
||||||
|
{ regexp => qr/^\\/, class => "incomplete" },
|
||||||
|
{ regexp => qr/^ {$num_sign}/, class => "ctx" },
|
||||||
|
# classifier for context must come before classifier add/rem,
|
||||||
|
# or we would have to use more complicated regexp, for example
|
||||||
|
# qr/(?= {0,$m}\+)[+ ]{$num_sign}/, where $m = $num_sign - 1;
|
||||||
|
{ regexp => qr/^[+ ]{$num_sign}/, class => "add" },
|
||||||
|
{ regexp => qr/^[- ]{$num_sign}/, class => "rem" },
|
||||||
|
);
|
||||||
|
for my $clsfy (@diff_line_classifier) {
|
||||||
|
return $clsfy->{'class'}
|
||||||
|
if ($line =~ $clsfy->{'regexp'});
|
||||||
|
}
|
||||||
|
|
||||||
|
# fallback
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
# assumes that $from and $to are defined and correctly filled,
|
||||||
|
# and that $line holds a line of chunk header for unified diff
|
||||||
|
sub format_unidiff_chunk_header {
|
||||||
|
my ($line, $from, $to) = @_;
|
||||||
|
|
||||||
|
my ($from_text, $from_start, $from_lines, $to_text, $to_start, $to_lines, $section) =
|
||||||
|
$line =~ m/^\@{2} (-(\d+)(?:,(\d+))?) (\+(\d+)(?:,(\d+))?) \@{2}(.*)$/;
|
||||||
|
|
||||||
|
$from_lines = 0 unless defined $from_lines;
|
||||||
|
$to_lines = 0 unless defined $to_lines;
|
||||||
|
|
||||||
|
if ($from->{'href'}) {
|
||||||
|
$from_text = $cgi->a({-href=>"$from->{'href'}#l$from_start",
|
||||||
|
-class=>"list"}, $from_text);
|
||||||
|
}
|
||||||
|
if ($to->{'href'}) {
|
||||||
|
$to_text = $cgi->a({-href=>"$to->{'href'}#l$to_start",
|
||||||
|
-class=>"list"}, $to_text);
|
||||||
|
}
|
||||||
|
$line = "<span class=\"chunk_info\">@@ $from_text $to_text @@</span>" .
|
||||||
|
"<span class=\"section\">" . esc_html($section, -nbsp=>1) . "</span>";
|
||||||
|
return $line;
|
||||||
|
}
|
||||||
|
|
||||||
|
# assumes that $from and $to are defined and correctly filled,
|
||||||
|
# and that $line holds a line of chunk header for combined diff
|
||||||
|
sub format_cc_diff_chunk_header {
|
||||||
|
my ($line, $from, $to) = @_;
|
||||||
|
|
||||||
|
my ($prefix, $ranges, $section) = $line =~ m/^(\@+) (.*?) \@+(.*)$/;
|
||||||
|
my (@from_text, @from_start, @from_nlines, $to_text, $to_start, $to_nlines);
|
||||||
|
|
||||||
|
@from_text = split(' ', $ranges);
|
||||||
|
for (my $i = 0; $i < @from_text; ++$i) {
|
||||||
|
($from_start[$i], $from_nlines[$i]) =
|
||||||
|
(split(',', substr($from_text[$i], 1)), 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
$to_text = pop @from_text;
|
||||||
|
$to_start = pop @from_start;
|
||||||
|
$to_nlines = pop @from_nlines;
|
||||||
|
|
||||||
|
$line = "<span class=\"chunk_info\">$prefix ";
|
||||||
|
for (my $i = 0; $i < @from_text; ++$i) {
|
||||||
|
if ($from->{'href'}[$i]) {
|
||||||
|
$line .= $cgi->a({-href=>"$from->{'href'}[$i]#l$from_start[$i]",
|
||||||
|
-class=>"list"}, $from_text[$i]);
|
||||||
|
} else {
|
||||||
|
$line .= $from_text[$i];
|
||||||
|
}
|
||||||
|
$line .= " ";
|
||||||
|
}
|
||||||
|
if ($to->{'href'}) {
|
||||||
|
$line .= $cgi->a({-href=>"$to->{'href'}#l$to_start",
|
||||||
|
-class=>"list"}, $to_text);
|
||||||
|
} else {
|
||||||
|
$line .= $to_text;
|
||||||
|
}
|
||||||
|
$line .= " $prefix</span>" .
|
||||||
|
"<span class=\"section\">" . esc_html($section, -nbsp=>1) . "</span>";
|
||||||
|
return $line;
|
||||||
|
}
|
||||||
|
|
||||||
|
# process patch (diff) line (not to be used for diff headers),
|
||||||
|
# returning class and HTML-formatted (but not wrapped) line
|
||||||
|
sub process_diff_line {
|
||||||
my $line = shift;
|
my $line = shift;
|
||||||
my ($from, $to) = @_;
|
my ($from, $to) = @_;
|
||||||
my $diff_class = "";
|
|
||||||
|
my $diff_class = diff_line_class($line, $from, $to);
|
||||||
|
|
||||||
chomp $line;
|
chomp $line;
|
||||||
|
|
||||||
if ($from && $to && ref($from->{'href'}) eq "ARRAY") {
|
|
||||||
# combined diff
|
|
||||||
my $prefix = substr($line, 0, scalar @{$from->{'href'}});
|
|
||||||
if ($line =~ m/^\@{3}/) {
|
|
||||||
$diff_class = " chunk_header";
|
|
||||||
} elsif ($line =~ m/^\\/) {
|
|
||||||
$diff_class = " incomplete";
|
|
||||||
} elsif ($prefix =~ tr/+/+/) {
|
|
||||||
$diff_class = " add";
|
|
||||||
} elsif ($prefix =~ tr/-/-/) {
|
|
||||||
$diff_class = " rem";
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
# assume ordinary diff
|
|
||||||
my $char = substr($line, 0, 1);
|
|
||||||
if ($char eq '+') {
|
|
||||||
$diff_class = " add";
|
|
||||||
} elsif ($char eq '-') {
|
|
||||||
$diff_class = " rem";
|
|
||||||
} elsif ($char eq '@') {
|
|
||||||
$diff_class = " chunk_header";
|
|
||||||
} elsif ($char eq "\\") {
|
|
||||||
$diff_class = " incomplete";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
$line = untabify($line);
|
$line = untabify($line);
|
||||||
|
|
||||||
if ($from && $to && $line =~ m/^\@{2} /) {
|
if ($from && $to && $line =~ m/^\@{2} /) {
|
||||||
my ($from_text, $from_start, $from_lines, $to_text, $to_start, $to_lines, $section) =
|
$line = format_unidiff_chunk_header($line, $from, $to);
|
||||||
$line =~ m/^\@{2} (-(\d+)(?:,(\d+))?) (\+(\d+)(?:,(\d+))?) \@{2}(.*)$/;
|
return $diff_class, $line;
|
||||||
|
|
||||||
$from_lines = 0 unless defined $from_lines;
|
|
||||||
$to_lines = 0 unless defined $to_lines;
|
|
||||||
|
|
||||||
if ($from->{'href'}) {
|
|
||||||
$from_text = $cgi->a({-href=>"$from->{'href'}#l$from_start",
|
|
||||||
-class=>"list"}, $from_text);
|
|
||||||
}
|
|
||||||
if ($to->{'href'}) {
|
|
||||||
$to_text = $cgi->a({-href=>"$to->{'href'}#l$to_start",
|
|
||||||
-class=>"list"}, $to_text);
|
|
||||||
}
|
|
||||||
$line = "<span class=\"chunk_info\">@@ $from_text $to_text @@</span>" .
|
|
||||||
"<span class=\"section\">" . esc_html($section, -nbsp=>1) . "</span>";
|
|
||||||
return "<div class=\"diff$diff_class\">$line</div>\n";
|
|
||||||
} elsif ($from && $to && $line =~ m/^\@{3}/) {
|
} elsif ($from && $to && $line =~ m/^\@{3}/) {
|
||||||
my ($prefix, $ranges, $section) = $line =~ m/^(\@+) (.*?) \@+(.*)$/;
|
$line = format_cc_diff_chunk_header($line, $from, $to);
|
||||||
my (@from_text, @from_start, @from_nlines, $to_text, $to_start, $to_nlines);
|
return $diff_class, $line;
|
||||||
|
|
||||||
@from_text = split(' ', $ranges);
|
|
||||||
for (my $i = 0; $i < @from_text; ++$i) {
|
|
||||||
($from_start[$i], $from_nlines[$i]) =
|
|
||||||
(split(',', substr($from_text[$i], 1)), 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
$to_text = pop @from_text;
|
|
||||||
$to_start = pop @from_start;
|
|
||||||
$to_nlines = pop @from_nlines;
|
|
||||||
|
|
||||||
$line = "<span class=\"chunk_info\">$prefix ";
|
|
||||||
for (my $i = 0; $i < @from_text; ++$i) {
|
|
||||||
if ($from->{'href'}[$i]) {
|
|
||||||
$line .= $cgi->a({-href=>"$from->{'href'}[$i]#l$from_start[$i]",
|
|
||||||
-class=>"list"}, $from_text[$i]);
|
|
||||||
} else {
|
|
||||||
$line .= $from_text[$i];
|
|
||||||
}
|
|
||||||
$line .= " ";
|
|
||||||
}
|
|
||||||
if ($to->{'href'}) {
|
|
||||||
$line .= $cgi->a({-href=>"$to->{'href'}#l$to_start",
|
|
||||||
-class=>"list"}, $to_text);
|
|
||||||
} else {
|
|
||||||
$line .= $to_text;
|
|
||||||
}
|
|
||||||
$line .= " $prefix</span>" .
|
|
||||||
"<span class=\"section\">" . esc_html($section, -nbsp=>1) . "</span>";
|
|
||||||
return "<div class=\"diff$diff_class\">$line</div>\n";
|
|
||||||
}
|
}
|
||||||
return "<div class=\"diff$diff_class\">" . esc_html($line, -nbsp=>1) . "</div>\n";
|
return $diff_class, esc_html($line, -nbsp=>1);
|
||||||
}
|
}
|
||||||
|
|
||||||
# Generates undef or something like "_snapshot_" or "snapshot (_tbz2_ _zip_)",
|
# Generates undef or something like "_snapshot_" or "snapshot (_tbz2_ _zip_)",
|
||||||
@ -4833,8 +4860,97 @@ sub git_difftree_body {
|
|||||||
print "</table>\n";
|
print "</table>\n";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
sub print_sidebyside_diff_chunk {
|
||||||
|
my @chunk = @_;
|
||||||
|
my (@ctx, @rem, @add);
|
||||||
|
|
||||||
|
return unless @chunk;
|
||||||
|
|
||||||
|
# incomplete last line might be among removed or added lines,
|
||||||
|
# or both, or among context lines: find which
|
||||||
|
for (my $i = 1; $i < @chunk; $i++) {
|
||||||
|
if ($chunk[$i][0] eq 'incomplete') {
|
||||||
|
$chunk[$i][0] = $chunk[$i-1][0];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
# guardian
|
||||||
|
push @chunk, ["", ""];
|
||||||
|
|
||||||
|
foreach my $line_info (@chunk) {
|
||||||
|
my ($class, $line) = @$line_info;
|
||||||
|
|
||||||
|
# print chunk headers
|
||||||
|
if ($class && $class eq 'chunk_header') {
|
||||||
|
print $line;
|
||||||
|
next;
|
||||||
|
}
|
||||||
|
|
||||||
|
## print from accumulator when type of class of lines change
|
||||||
|
# empty contents block on start rem/add block, or end of chunk
|
||||||
|
if (@ctx && (!$class || $class eq 'rem' || $class eq 'add')) {
|
||||||
|
print join '',
|
||||||
|
'<div class="chunk_block ctx">',
|
||||||
|
'<div class="old">',
|
||||||
|
@ctx,
|
||||||
|
'</div>',
|
||||||
|
'<div class="new">',
|
||||||
|
@ctx,
|
||||||
|
'</div>',
|
||||||
|
'</div>';
|
||||||
|
@ctx = ();
|
||||||
|
}
|
||||||
|
# empty add/rem block on start context block, or end of chunk
|
||||||
|
if ((@rem || @add) && (!$class || $class eq 'ctx')) {
|
||||||
|
if (!@add) {
|
||||||
|
# pure removal
|
||||||
|
print join '',
|
||||||
|
'<div class="chunk_block rem">',
|
||||||
|
'<div class="old">',
|
||||||
|
@rem,
|
||||||
|
'</div>',
|
||||||
|
'</div>';
|
||||||
|
} elsif (!@rem) {
|
||||||
|
# pure addition
|
||||||
|
print join '',
|
||||||
|
'<div class="chunk_block add">',
|
||||||
|
'<div class="new">',
|
||||||
|
@add,
|
||||||
|
'</div>',
|
||||||
|
'</div>';
|
||||||
|
} else {
|
||||||
|
# assume that it is change
|
||||||
|
print join '',
|
||||||
|
'<div class="chunk_block chg">',
|
||||||
|
'<div class="old">',
|
||||||
|
@rem,
|
||||||
|
'</div>',
|
||||||
|
'<div class="new">',
|
||||||
|
@add,
|
||||||
|
'</div>',
|
||||||
|
'</div>';
|
||||||
|
}
|
||||||
|
@rem = @add = ();
|
||||||
|
}
|
||||||
|
|
||||||
|
## adding lines to accumulator
|
||||||
|
# guardian value
|
||||||
|
last unless $line;
|
||||||
|
# rem, add or change
|
||||||
|
if ($class eq 'rem') {
|
||||||
|
push @rem, $line;
|
||||||
|
} elsif ($class eq 'add') {
|
||||||
|
push @add, $line;
|
||||||
|
}
|
||||||
|
# context line
|
||||||
|
if ($class eq 'ctx') {
|
||||||
|
push @ctx, $line;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
sub git_patchset_body {
|
sub git_patchset_body {
|
||||||
my ($fd, $difftree, $hash, @hash_parents) = @_;
|
my ($fd, $diff_style, $difftree, $hash, @hash_parents) = @_;
|
||||||
my ($hash_parent) = $hash_parents[0];
|
my ($hash_parent) = $hash_parents[0];
|
||||||
|
|
||||||
my $is_combined = (@hash_parents > 1);
|
my $is_combined = (@hash_parents > 1);
|
||||||
@ -4844,6 +4960,7 @@ sub git_patchset_body {
|
|||||||
my $diffinfo;
|
my $diffinfo;
|
||||||
my $to_name;
|
my $to_name;
|
||||||
my (%from, %to);
|
my (%from, %to);
|
||||||
|
my @chunk; # for side-by-side diff
|
||||||
|
|
||||||
print "<div class=\"patchset\">\n";
|
print "<div class=\"patchset\">\n";
|
||||||
|
|
||||||
@ -4950,10 +5067,29 @@ sub git_patchset_body {
|
|||||||
|
|
||||||
next PATCH if ($patch_line =~ m/^diff /);
|
next PATCH if ($patch_line =~ m/^diff /);
|
||||||
|
|
||||||
print format_diff_line($patch_line, \%from, \%to);
|
my ($class, $line) = process_diff_line($patch_line, \%from, \%to);
|
||||||
|
my $diff_classes = "diff";
|
||||||
|
$diff_classes .= " $class" if ($class);
|
||||||
|
$line = "<div class=\"$diff_classes\">$line</div>\n";
|
||||||
|
|
||||||
|
if ($diff_style eq 'sidebyside' && !$is_combined) {
|
||||||
|
if ($class eq 'chunk_header') {
|
||||||
|
print_sidebyside_diff_chunk(@chunk);
|
||||||
|
@chunk = ( [ $class, $line ] );
|
||||||
|
} else {
|
||||||
|
push @chunk, [ $class, $line ];
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
# default 'inline' style and unknown styles
|
||||||
|
print $line;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
} continue {
|
} continue {
|
||||||
|
if (@chunk) {
|
||||||
|
print_sidebyside_diff_chunk(@chunk);
|
||||||
|
@chunk = ();
|
||||||
|
}
|
||||||
print "</div>\n"; # class="patch"
|
print "</div>\n"; # class="patch"
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -6949,6 +7085,7 @@ sub git_object {
|
|||||||
|
|
||||||
sub git_blobdiff {
|
sub git_blobdiff {
|
||||||
my $format = shift || 'html';
|
my $format = shift || 'html';
|
||||||
|
my $diff_style = $input_params{'diff_style'} || 'inline';
|
||||||
|
|
||||||
my $fd;
|
my $fd;
|
||||||
my @difftree;
|
my @difftree;
|
||||||
@ -7027,6 +7164,7 @@ sub git_blobdiff {
|
|||||||
my $formats_nav =
|
my $formats_nav =
|
||||||
$cgi->a({-href => href(action=>"blobdiff_plain", -replay=>1)},
|
$cgi->a({-href => href(action=>"blobdiff_plain", -replay=>1)},
|
||||||
"raw");
|
"raw");
|
||||||
|
$formats_nav .= diff_style_nav($diff_style);
|
||||||
git_header_html(undef, $expires);
|
git_header_html(undef, $expires);
|
||||||
if (defined $hash_base && (my %co = parse_commit($hash_base))) {
|
if (defined $hash_base && (my %co = parse_commit($hash_base))) {
|
||||||
git_print_page_nav('','', $hash_base,$co{'tree'},$hash_base, $formats_nav);
|
git_print_page_nav('','', $hash_base,$co{'tree'},$hash_base, $formats_nav);
|
||||||
@ -7058,7 +7196,8 @@ sub git_blobdiff {
|
|||||||
if ($format eq 'html') {
|
if ($format eq 'html') {
|
||||||
print "<div class=\"page_body\">\n";
|
print "<div class=\"page_body\">\n";
|
||||||
|
|
||||||
git_patchset_body($fd, [ \%diffinfo ], $hash_base, $hash_parent_base);
|
git_patchset_body($fd, $diff_style,
|
||||||
|
[ \%diffinfo ], $hash_base, $hash_parent_base);
|
||||||
close $fd;
|
close $fd;
|
||||||
|
|
||||||
print "</div>\n"; # class="page_body"
|
print "</div>\n"; # class="page_body"
|
||||||
@ -7083,9 +7222,31 @@ sub git_blobdiff_plain {
|
|||||||
git_blobdiff('plain');
|
git_blobdiff('plain');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# assumes that it is added as later part of already existing navigation,
|
||||||
|
# so it returns "| foo | bar" rather than just "foo | bar"
|
||||||
|
sub diff_style_nav {
|
||||||
|
my ($diff_style, $is_combined) = @_;
|
||||||
|
$diff_style ||= 'inline';
|
||||||
|
|
||||||
|
return "" if ($is_combined);
|
||||||
|
|
||||||
|
my @styles = (inline => 'inline', 'sidebyside' => 'side by side');
|
||||||
|
my %styles = @styles;
|
||||||
|
@styles =
|
||||||
|
@styles[ map { $_ * 2 } 0..$#styles/2 ];
|
||||||
|
|
||||||
|
return join '',
|
||||||
|
map { " | ".$_ }
|
||||||
|
map {
|
||||||
|
$_ eq $diff_style ? $styles{$_} :
|
||||||
|
$cgi->a({-href => href(-replay=>1, diff_style => $_)}, $styles{$_})
|
||||||
|
} @styles;
|
||||||
|
}
|
||||||
|
|
||||||
sub git_commitdiff {
|
sub git_commitdiff {
|
||||||
my %params = @_;
|
my %params = @_;
|
||||||
my $format = $params{-format} || 'html';
|
my $format = $params{-format} || 'html';
|
||||||
|
my $diff_style = $input_params{'diff_style'} || 'inline';
|
||||||
|
|
||||||
my ($patch_max) = gitweb_get_feature('patches');
|
my ($patch_max) = gitweb_get_feature('patches');
|
||||||
if ($format eq 'patch') {
|
if ($format eq 'patch') {
|
||||||
@ -7111,6 +7272,7 @@ sub git_commitdiff {
|
|||||||
$cgi->a({-href => href(action=>"patch", -replay=>1)},
|
$cgi->a({-href => href(action=>"patch", -replay=>1)},
|
||||||
"patch");
|
"patch");
|
||||||
}
|
}
|
||||||
|
$formats_nav .= diff_style_nav($diff_style, @{$co{'parents'}} > 1);
|
||||||
|
|
||||||
if (defined $hash_parent &&
|
if (defined $hash_parent &&
|
||||||
$hash_parent ne '-c' && $hash_parent ne '--cc') {
|
$hash_parent ne '-c' && $hash_parent ne '--cc') {
|
||||||
@ -7128,8 +7290,8 @@ sub git_commitdiff {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
$formats_nav .= ': ' .
|
$formats_nav .= ': ' .
|
||||||
$cgi->a({-href => href(action=>"commitdiff",
|
$cgi->a({-href => href(-replay=>1,
|
||||||
hash=>$hash_parent)},
|
hash=>$hash_parent, hash_base=>undef)},
|
||||||
esc_html($hash_parent_short)) .
|
esc_html($hash_parent_short)) .
|
||||||
')';
|
')';
|
||||||
} elsif (!$co{'parent'}) {
|
} elsif (!$co{'parent'}) {
|
||||||
@ -7139,28 +7301,28 @@ sub git_commitdiff {
|
|||||||
# single parent commit
|
# single parent commit
|
||||||
$formats_nav .=
|
$formats_nav .=
|
||||||
' (parent: ' .
|
' (parent: ' .
|
||||||
$cgi->a({-href => href(action=>"commitdiff",
|
$cgi->a({-href => href(-replay=>1,
|
||||||
hash=>$co{'parent'})},
|
hash=>$co{'parent'}, hash_base=>undef)},
|
||||||
esc_html(substr($co{'parent'}, 0, 7))) .
|
esc_html(substr($co{'parent'}, 0, 7))) .
|
||||||
')';
|
')';
|
||||||
} else {
|
} else {
|
||||||
# merge commit
|
# merge commit
|
||||||
if ($hash_parent eq '--cc') {
|
if ($hash_parent eq '--cc') {
|
||||||
$formats_nav .= ' | ' .
|
$formats_nav .= ' | ' .
|
||||||
$cgi->a({-href => href(action=>"commitdiff",
|
$cgi->a({-href => href(-replay=>1,
|
||||||
hash=>$hash, hash_parent=>'-c')},
|
hash=>$hash, hash_parent=>'-c')},
|
||||||
'combined');
|
'combined');
|
||||||
} else { # $hash_parent eq '-c'
|
} else { # $hash_parent eq '-c'
|
||||||
$formats_nav .= ' | ' .
|
$formats_nav .= ' | ' .
|
||||||
$cgi->a({-href => href(action=>"commitdiff",
|
$cgi->a({-href => href(-replay=>1,
|
||||||
hash=>$hash, hash_parent=>'--cc')},
|
hash=>$hash, hash_parent=>'--cc')},
|
||||||
'compact');
|
'compact');
|
||||||
}
|
}
|
||||||
$formats_nav .=
|
$formats_nav .=
|
||||||
' (merge: ' .
|
' (merge: ' .
|
||||||
join(' ', map {
|
join(' ', map {
|
||||||
$cgi->a({-href => href(action=>"commitdiff",
|
$cgi->a({-href => href(-replay=>1,
|
||||||
hash=>$_)},
|
hash=>$_, hash_base=>undef)},
|
||||||
esc_html(substr($_, 0, 7)));
|
esc_html(substr($_, 0, 7)));
|
||||||
} @{$co{'parents'}} ) .
|
} @{$co{'parents'}} ) .
|
||||||
')';
|
')';
|
||||||
@ -7289,7 +7451,8 @@ sub git_commitdiff {
|
|||||||
$use_parents ? @{$co{'parents'}} : $hash_parent);
|
$use_parents ? @{$co{'parents'}} : $hash_parent);
|
||||||
print "<br/>\n";
|
print "<br/>\n";
|
||||||
|
|
||||||
git_patchset_body($fd, \@difftree, $hash,
|
git_patchset_body($fd, $diff_style,
|
||||||
|
\@difftree, $hash,
|
||||||
$use_parents ? @{$co{'parents'}} : $hash_parent);
|
$use_parents ? @{$co{'parents'}} : $hash_parent);
|
||||||
close $fd;
|
close $fd;
|
||||||
print "</div>\n"; # class="page_body"
|
print "</div>\n"; # class="page_body"
|
||||||
|
@ -475,6 +475,36 @@ div.diff.nodifferences {
|
|||||||
color: #600000;
|
color: #600000;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* side-by-side diff */
|
||||||
|
div.chunk_block {
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
div.chunk_block div.old {
|
||||||
|
float: left;
|
||||||
|
width: 50%;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
div.chunk_block div.new {
|
||||||
|
margin-left: 50%;
|
||||||
|
width: 50%;
|
||||||
|
}
|
||||||
|
|
||||||
|
div.chunk_block.rem div.old div.diff.rem {
|
||||||
|
background-color: #fff5f5;
|
||||||
|
}
|
||||||
|
div.chunk_block.add div.new div.diff.add {
|
||||||
|
background-color: #f8fff8;
|
||||||
|
}
|
||||||
|
div.chunk_block.chg div div.diff {
|
||||||
|
background-color: #fffff0;
|
||||||
|
}
|
||||||
|
div.chunk_block.ctx div div.diff.ctx {
|
||||||
|
color: #404040;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
div.index_include {
|
div.index_include {
|
||||||
border: solid #d9d8d1;
|
border: solid #d9d8d1;
|
||||||
border-width: 0px 0px 1px;
|
border-width: 0px 0px 1px;
|
||||||
|
@ -273,6 +273,53 @@ test_expect_success \
|
|||||||
'commitdiff(2): directory becomes symlink' \
|
'commitdiff(2): directory becomes symlink' \
|
||||||
'gitweb_run "p=.git;a=commitdiff;hp=foo-becomes-a-directory;h=foo-symlinked-to-bar"'
|
'gitweb_run "p=.git;a=commitdiff;hp=foo-becomes-a-directory;h=foo-symlinked-to-bar"'
|
||||||
|
|
||||||
|
# ----------------------------------------------------------------------
|
||||||
|
# commitdiff testing (incomplete lines)
|
||||||
|
|
||||||
|
test_expect_success 'setup incomplete lines' '
|
||||||
|
cat >file<<-\EOF &&
|
||||||
|
Dominus regit me,
|
||||||
|
et nihil mihi deerit.
|
||||||
|
In loco pascuae ibi me collocavit,
|
||||||
|
super aquam refectionis educavit me;
|
||||||
|
animam meam convertit,
|
||||||
|
deduxit me super semitas jusitiae,
|
||||||
|
propter nomen suum.
|
||||||
|
CHANGE_ME
|
||||||
|
EOF
|
||||||
|
git commit -a -m "Preparing for incomplete lines" &&
|
||||||
|
echo "incomplete" | tr -d "\\012" >>file &&
|
||||||
|
git commit -a -m "Add incomplete line" &&
|
||||||
|
git tag incomplete_lines_add &&
|
||||||
|
sed -e s/CHANGE_ME/change_me/ <file >file+ &&
|
||||||
|
mv -f file+ file &&
|
||||||
|
git commit -a -m "Incomplete context line" &&
|
||||||
|
git tag incomplete_lines_ctx &&
|
||||||
|
echo "Dominus regit me," >file &&
|
||||||
|
echo "incomplete line" | tr -d "\\012" >>file &&
|
||||||
|
git commit -a -m "Change incomplete line" &&
|
||||||
|
git tag incomplete_lines_chg
|
||||||
|
echo "Dominus regit me," >file &&
|
||||||
|
git commit -a -m "Remove incomplete line" &&
|
||||||
|
git tag incomplete_lines_rem
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'commitdiff(1): addition of incomplete line' '
|
||||||
|
gitweb_run "p=.git;a=commitdiff;h=incomplete_lines_add"
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'commitdiff(1): incomplete line as context line' '
|
||||||
|
gitweb_run "p=.git;a=commitdiff;h=incomplete_lines_ctx"
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'commitdiff(1): change incomplete line' '
|
||||||
|
gitweb_run "p=.git;a=commitdiff;h=incomplete_lines_chg"
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'commitdiff(1): removal of incomplete line' '
|
||||||
|
gitweb_run "p=.git;a=commitdiff;h=incomplete_lines_rem"
|
||||||
|
'
|
||||||
|
|
||||||
# ----------------------------------------------------------------------
|
# ----------------------------------------------------------------------
|
||||||
# commit, commitdiff: merge, large
|
# commit, commitdiff: merge, large
|
||||||
test_expect_success \
|
test_expect_success \
|
||||||
@ -282,7 +329,8 @@ test_expect_success \
|
|||||||
git add b &&
|
git add b &&
|
||||||
git commit -a -m "On branch" &&
|
git commit -a -m "On branch" &&
|
||||||
git checkout master &&
|
git checkout master &&
|
||||||
git pull . b'
|
git pull . b &&
|
||||||
|
git tag merge_commit'
|
||||||
|
|
||||||
test_expect_success \
|
test_expect_success \
|
||||||
'commit(0): merge commit' \
|
'commit(0): merge commit' \
|
||||||
@ -331,6 +379,29 @@ test_expect_success \
|
|||||||
'commitdiff(1): large commit' \
|
'commitdiff(1): large commit' \
|
||||||
'gitweb_run "p=.git;a=commitdiff;h=b"'
|
'gitweb_run "p=.git;a=commitdiff;h=b"'
|
||||||
|
|
||||||
|
# ----------------------------------------------------------------------
|
||||||
|
# side-by-side diff
|
||||||
|
|
||||||
|
test_expect_success 'side-by-side: addition of incomplete line' '
|
||||||
|
gitweb_run "p=.git;a=commitdiff;h=incomplete_lines_add;ds=sidebyside"
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'side-by-side: incomplete line as context line' '
|
||||||
|
gitweb_run "p=.git;a=commitdiff;h=incomplete_lines_ctx;ds=sidebyside"
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'side-by-side: changed incomplete line' '
|
||||||
|
gitweb_run "p=.git;a=commitdiff;h=incomplete_lines_chg;ds=sidebyside"
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'side-by-side: removal of incomplete line' '
|
||||||
|
gitweb_run "p=.git;a=commitdiff;h=incomplete_lines_rem;ds=sidebyside"
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'side-by-side: merge commit' '
|
||||||
|
gitweb_run "p=.git;a=commitdiff;h=merge_commit;ds=sidebyside"
|
||||||
|
'
|
||||||
|
|
||||||
# ----------------------------------------------------------------------
|
# ----------------------------------------------------------------------
|
||||||
# tags testing
|
# tags testing
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user