hook global BufCreate .*\.(diff|patch) %{ set-option buffer filetype diff } hook global WinSetOption filetype=diff %{ require-module diff map buffer normal :diff-jump } hook -group diff-highlight global WinSetOption filetype=diff %{ add-highlighter window/diff ref diff hook -once -always window WinSetOption filetype=.* %{ remove-highlighter window/diff } } provide-module diff %§ add-highlighter shared/diff group add-highlighter shared/diff/ regex "^\+[^\n]*\n" 0:green,default add-highlighter shared/diff/ regex "^-[^\n]*\n" 0:red,default add-highlighter shared/diff/ regex "^@@[^\n]*@@" 0:cyan,default # If any trailing whitespace was introduced in diff, show it with red background add-highlighter shared/diff/ regex "^\+[^\n]*?(\h+)\n" 1:default,red define-command diff-jump -params .. -docstring %{ diff-jump [] []: edit the diff's source file at the cursor position. Paths are resolved relative to , or the current working directory if unspecified. Switches: - jump to the old file instead of the new file - strip leading directory components, like -p in patch(1). Defaults to 1 if there is a 'diff' line (as printed by 'diff -r'), or 0 otherwise. } %{ evaluate-commands -draft -save-regs c %{ # Save the column because we will move the cursor. set-register c %val{cursor_column} # If there is a "diff" line, we don't need to look further back. try %{ execute-keys %{^(?:> )*diff\bx} } catch %{ # A single file diff won't have a diff line. Start parsing from # the buffer start, so we can tell if +++/--- lines are headers # or content. execute-keys Gk } diff-parse BEGIN %{ my $seen_ddash = 0; foreach (@ARGV) { if ($seen_ddash or !m{^-}) { $directory = $_; } elsif ($_ eq "-") { $version = "-", $other_version = "+"; } elsif (m{^-(\d+)$}) { $strip = $1; } elsif ($_ eq "--") { $seen_ddash = 1; } else { fail "unknown option: $_"; } } } END %exp{ my $file_column; if (not defined $file_line) { $file_line = ""; $file_column = ""; } else { my $diff_column = %reg{c}; $file_column = $diff_column - 1; # Account for [ +-] diff prefix. # If the cursor was on a hunk header, go to the section header if possible. if ($diff_line_text =~ m{^(@@ -\d+(?:,\d+)? \+\d+(?:,\d+) @@ )([^\n]*)}) { my $hunk_header_prefix = $1; my $hunk_header_from_userdiff = $2; open FILE, "<", $file or fail "failed to open file: $!: $file"; my @lines = ; for (my $i = $file_line - 1; $i >= 0 and $i < scalar @lines; $i--) { if ($lines[$i] !~ m{\Q$hunk_header_from_userdiff}) { next; } $file_line = $i + 1; # Re-add 1 because the @@ line does not have a [ +-] diff prefix. $file_column = $diff_column + 1 - length $hunk_header_prefix; last; } } } printf "set-register c %%s $file_line $file_column\n", quote($file); } -- %arg{@} evaluate-commands -client %val{client} %{ evaluate-commands -try-client %opt{jumpclient} %{ edit -existing -- %reg{c} } } } } complete-command diff-jump file define-command -hidden diff-parse -params 2.. %{ evaluate-commands -save-regs ae| %{ set-register a %arg{@} set-register e nop set-register | %{ eval set -- "$kak_quoted_reg_a" perl "${kak_runtime}/rc/filetype/diff-parse.pl" "$@" >"$kak_command_fifo" } execute-keys %reg{e} } } § define-command \ -docstring %{diff-select-file: Select surrounding patch file} \ -params 0 \ diff-select-file %{ evaluate-commands -itersel -save-regs 'ose/' %{ try %{ execute-keys '"oZgl^diff ;"sZ' 'Ge"eZ' try %{ execute-keys '"sz?\n(?=diff )"e' } execute-keys '"ez' } catch %{ execute-keys '"oz' fail 'Not in a diff file' } } } define-command \ -docstring %{diff-select-hunk: Select surrounding patch hunk} \ -params 0 \ diff-select-hunk %{ evaluate-commands -itersel -save-regs 'ose/' %{ try %{ execute-keys '"oZgl^@@ ;"sZ' 'Ge"eZ' try %{ execute-keys '"sz?\n(?=diff )"e' } try %{ execute-keys '"sz?\n(?=@@ )"e' } execute-keys '"ez' } catch %{ execute-keys '"oz' fail 'Not in a diff hunk' } } }