summaryrefslogtreecommitdiff
path: root/autoload/tools/patch.kak
blob: a481ff41263e8e3fa7ff3789bffcf4547070e333 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
define-command patch -params .. -docstring %{
    patch [<arguments>]: apply selections in diff to a file

    Given some selections within a unified diff, apply the changed lines in
    each selection by piping them to "patch <arguments> 1>&2"
    (or "<arguments> 1>&2" if <arguments> starts with a non-option argument).
    If successful, the in-buffer diff will be updated to reflect the applied
    changes.
    For selections that contain no newline, the entire enclosing diff hunk
    is applied (unless the cursor is inside a diff header, in which case
    the entire diff is applied).
    To revert changes, <arguments> must contain "--reverse" or "-R".
} %{
    evaluate-commands -draft -itersel -save-regs aes|^ %{
        try %{
            execute-keys <a-k>\n<ret>
        } catch %{
            # The selection contains no newline.
            execute-keys -save-regs '' Z
            execute-keys <a-l><semicolon><a-?>^diff<ret>
            try %{
                execute-keys <a-k>^@@<ret>
                # If the cursor is in a diff hunk, stage the entire hunk.
                execute-keys z
                execute-keys /.*?(?:(?=\n@@)|(?=\ndiff)|(?=\n\n)|\z)<ret>x<semicolon><a-?>^@@<ret>
            } catch %{
                # If the cursor is in a diff header, stage the entire diff.
                execute-keys <a-semicolon>?.*?(?:(?=\ndiff)|(?=\n\n)|\z)<ret>
            }
        }
        # We want to apply only the selected lines. Remember them.
        execute-keys <a-:>
        set-register s %val{selection_desc}
        # Select forward until the end of the last hunk.
        execute-keys H?.*?(?:(?=\n@@)|(?=\ndiff)|(?=\n\n)|\z)<ret>x
        # Select backward to the beginning of the first hunk's diff header.
        execute-keys <a-semicolon><a-L><a-?>^diff<ret>
        # Move cursor to the beginning so we know the diff's offset within the buffer.
        execute-keys <a-:><a-semicolon>
        set-register a %arg{@}
        set-register e nop
        set-register | %{
            # The selected range to apply.
            IFS=' .,' read min_line _ max_line _ <<-EOF
            $kak_reg_s
	EOF
            min_line=$((min_line - kak_cursor_line + 1))
            max_line=$((max_line - kak_cursor_line + 1))

            # Since registers are never empty, we get an empty arg even if
            # there were no args. This does no harm because we pass it to
            # a shell where it expands to nothing.
            eval set -- "$kak_quoted_reg_a"

            perl "${kak_runtime}"/rc/tools/patch-range.pl $min_line $max_line "$@" ||
                echo >$kak_command_fifo "set-register e fail 'patch: failed to apply selections, see *debug* buffer'"
        }
        execute-keys |<ret>
        %reg{e}
    }
}

provide-module patch %§§