vimrc/sources_non_forked/vim-gitgutter/autoload/gitgutter/diff.vim

418 lines
14 KiB
VimL
Raw Normal View History

2020-04-26 01:56:16 +00:00
scriptencoding utf8
2018-03-31 14:56:26 +00:00
let s:nomodeline = (v:version > 703 || (v:version == 703 && has('patch442'))) ? '<nomodeline>' : ''
let s:hunk_re = '^@@ -\(\d\+\),\?\(\d*\) +\(\d\+\),\?\(\d*\) @@'
2020-12-04 21:15:32 +00:00
let s:temp_from = tempname()
let s:temp_buffer = tempname()
2018-12-17 11:28:27 +00:00
let s:counter = 0
2018-11-01 10:03:42 +00:00
" Returns a diff of the buffer against the index or the working tree.
"
" After running the diff we pass it through grep where available to reduce
" subsequent processing by the plugin. If grep is not available the plugin
" does the filtering instead.
"
" When diffing against the index:
"
2018-03-31 14:56:26 +00:00
" The buffer contents is not the same as the file on disk so we need to pass
" two instances of the file to git-diff:
"
" git diff myfileA myfileB
"
" where myfileA comes from
"
2018-03-31 14:56:26 +00:00
" git show :myfile > myfileA
"
2018-03-31 14:56:26 +00:00
" and myfileB is the buffer contents.
"
2018-03-31 14:56:26 +00:00
" Regarding line endings:
"
2018-03-31 14:56:26 +00:00
" git-show does not convert line endings.
" git-diff FILE FILE does convert line endings for the given files.
"
2018-03-31 14:56:26 +00:00
" If a file has CRLF line endings and git's core.autocrlf is true,
" the file in git's object store will have LF line endings. Writing
" it out via git-show will produce a file with LF line endings.
"
2018-03-31 14:56:26 +00:00
" If this last file is one of the files passed to git-diff, git-diff will
" convert its line endings to CRLF before diffing -- which is what we want --
" but also by default output a warning on stderr.
"
2018-03-31 14:56:26 +00:00
" warning: LF will be replace by CRLF in <temp file>.
" The file will have its original line endings in your working directory.
"
2018-03-31 14:56:26 +00:00
" When running the diff asynchronously, the warning message triggers the stderr
" callbacks which assume the overall command has failed and reset all the
" signs. As this is not what we want, and we can safely ignore the warning,
" we turn it off by passing the '-c "core.safecrlf=false"' argument to
" git-diff.
"
2018-03-31 14:56:26 +00:00
" When writing the temporary files we preserve the original file's extension
" so that repos using .gitattributes to control EOL conversion continue to
" convert correctly.
2018-11-01 10:03:42 +00:00
"
" Arguments:
"
" bufnr - the number of the buffer to be diffed
" from - 'index' or 'working_tree'; what the buffer is diffed against
" preserve_full_diff - truthy to return the full diff or falsey to return only
" the hunk headers (@@ -x,y +m,n @@); only possible if
" grep is available.
function! gitgutter#diff#run_diff(bufnr, from, preserve_full_diff) abort
2019-08-22 15:36:17 +00:00
if gitgutter#utility#repo_path(a:bufnr, 0) == -1
2021-05-05 08:25:00 +00:00
throw 'gitgutter path not set'
2019-08-22 15:36:17 +00:00
endif
2018-03-31 14:56:26 +00:00
if gitgutter#utility#repo_path(a:bufnr, 0) == -2
throw 'gitgutter not tracked'
endif
2022-08-08 13:45:56 +00:00
if gitgutter#utility#repo_path(a:bufnr, 0) == -3
throw 'gitgutter assume unchanged'
endif
" Wrap compound commands in parentheses to make Windows happy.
" bash doesn't mind the parentheses.
let cmd = '('
2018-11-01 10:03:42 +00:00
" Append buffer number to temp filenames to avoid race conditions between
" writing and reading the files when asynchronously processing multiple
" buffers.
2018-03-31 14:56:26 +00:00
" Without the buffer number, buff_file would have a race between the
" second gitgutter#process_buffer() writing the file (synchronously, below)
" and the first gitgutter#process_buffer()'s async job reading it (with
" git-diff).
2020-12-04 21:15:32 +00:00
let buff_file = s:temp_buffer.'.'.a:bufnr
2018-03-31 14:56:26 +00:00
2018-12-17 11:28:27 +00:00
" Add a counter to avoid a similar race with two quick writes of the same buffer.
" Use a modulus greater than a maximum reasonable number of visible buffers.
let s:counter = (s:counter + 1) % 20
let buff_file .= '.'.s:counter
2018-03-31 14:56:26 +00:00
let extension = gitgutter#utility#extension(a:bufnr)
if !empty(extension)
let buff_file .= '.'.extension
endif
2018-03-31 14:56:26 +00:00
" Write buffer to temporary file.
" Note: this is synchronous.
call s:write_buffer(a:bufnr, buff_file)
2018-11-01 10:03:42 +00:00
if a:from ==# 'index'
" Without the buffer number, from_file would have a race in the shell
" between the second process writing it (with git-show) and the first
" reading it (with git-diff).
2020-12-04 21:15:32 +00:00
let from_file = s:temp_from.'.'.a:bufnr
2018-11-01 10:03:42 +00:00
2018-12-17 11:28:27 +00:00
" Add a counter to avoid a similar race with two quick writes of the same buffer.
let from_file .= '.'.s:counter
2018-11-01 10:03:42 +00:00
if !empty(extension)
let from_file .= '.'.extension
endif
" Write file from index to temporary file.
2024-01-07 15:14:20 +00:00
let index_name = gitgutter#utility#get_diff_base(a:bufnr).':'.gitgutter#utility#base_path(a:bufnr)
2024-10-06 08:25:50 +00:00
let cmd .= gitgutter#git(a:bufnr).' --no-pager show --textconv '.index_name.' > '.from_file.' || exit 0) && ('
2018-11-01 10:03:42 +00:00
elseif a:from ==# 'working_tree'
let from_file = gitgutter#utility#repo_path(a:bufnr, 1)
endif
" Call git-diff.
2024-10-06 08:25:50 +00:00
let cmd .= gitgutter#git(a:bufnr).' --no-pager'
2024-01-07 15:14:20 +00:00
if gitgutter#utility#git_supports_command_line_config_override()
let cmd .= ' -c "diff.autorefreshindex=0"'
2017-04-01 11:22:06 +00:00
let cmd .= ' -c "diff.noprefix=false"'
2018-03-31 14:56:26 +00:00
let cmd .= ' -c "core.safecrlf=false"'
endif
2018-11-01 10:03:42 +00:00
let cmd .= ' diff --no-ext-diff --no-color -U0 '.g:gitgutter_diff_args.' -- '.from_file.' '.buff_file
2018-03-31 14:56:26 +00:00
" Pipe git-diff output into grep.
if !a:preserve_full_diff && !empty(g:gitgutter_grep)
let cmd .= ' | '.g:gitgutter_grep.' '.gitgutter#utility#shellescape('^@@ ')
endif
2018-03-31 14:56:26 +00:00
" grep exits with 1 when no matches are found; git-diff exits with 1 when
" differences are found. However we want to treat non-matches and
" differences as non-erroneous behaviour; so we OR the command with one
" which always exits with success (0).
let cmd .= ' || exit 0'
let cmd .= ')'
2018-03-31 14:56:26 +00:00
if g:gitgutter_async && gitgutter#async#available()
call gitgutter#async#execute(cmd, a:bufnr, {
\ 'out': function('gitgutter#diff#handler'),
\ 'err': function('gitgutter#hunk#reset'),
\ })
return 'async'
else
2024-01-07 15:14:20 +00:00
let [diff, error_code] = gitgutter#utility#system(cmd)
2024-01-07 15:14:20 +00:00
if error_code
2018-03-31 14:56:26 +00:00
call gitgutter#debug#log(diff)
throw 'gitgutter diff failed'
endif
return diff
endif
endfunction
2018-03-31 14:56:26 +00:00
function! gitgutter#diff#handler(bufnr, diff) abort
call gitgutter#debug#log(a:diff)
2018-07-04 10:53:25 +00:00
if !bufexists(a:bufnr)
return
endif
2018-03-31 14:56:26 +00:00
call gitgutter#hunk#set_hunks(a:bufnr, gitgutter#diff#parse_diff(a:diff))
let modified_lines = gitgutter#diff#process_hunks(a:bufnr, gitgutter#hunk#hunks(a:bufnr))
let signs_count = len(modified_lines)
2020-05-10 14:24:38 +00:00
if g:gitgutter_max_signs != -1 && signs_count > g:gitgutter_max_signs
2018-03-31 14:56:26 +00:00
call gitgutter#utility#warn_once(a:bufnr, printf(
\ 'exceeded maximum number of signs (%d > %d, configured by g:gitgutter_max_signs).',
\ signs_count, g:gitgutter_max_signs), 'max_signs')
call gitgutter#sign#clear_signs(a:bufnr)
else
2019-11-16 15:28:42 +00:00
if g:gitgutter_signs || g:gitgutter_highlight_lines || g:gitgutter_highlight_linenrs
2018-03-31 14:56:26 +00:00
call gitgutter#sign#update_signs(a:bufnr, modified_lines)
endif
endif
call s:save_last_seen_change(a:bufnr)
if exists('#User#GitGutter')
let g:gitgutter_hook_context = {'bufnr': a:bufnr}
execute 'doautocmd' s:nomodeline 'User GitGutter'
unlet g:gitgutter_hook_context
endif
endfunction
function! gitgutter#diff#parse_diff(diff) abort
let hunks = []
for line in split(a:diff, '\n')
let hunk_info = gitgutter#diff#parse_hunk(line)
if len(hunk_info) == 4
call add(hunks, hunk_info)
endif
endfor
return hunks
endfunction
function! gitgutter#diff#parse_hunk(line) abort
let matches = matchlist(a:line, s:hunk_re)
if len(matches) > 0
let from_line = str2nr(matches[1])
let from_count = (matches[2] == '') ? 1 : str2nr(matches[2])
let to_line = str2nr(matches[3])
let to_count = (matches[4] == '') ? 1 : str2nr(matches[4])
return [from_line, from_count, to_line, to_count]
else
return []
end
endfunction
2018-03-31 14:56:26 +00:00
" This function is public so it may be used by other plugins
" e.g. vim-signature.
function! gitgutter#diff#process_hunks(bufnr, hunks) abort
let modified_lines = []
for hunk in a:hunks
2018-03-31 14:56:26 +00:00
call extend(modified_lines, s:process_hunk(a:bufnr, hunk))
endfor
return modified_lines
endfunction
" Returns [ [<line_number (number)>, <name (string)>], ...]
2018-03-31 14:56:26 +00:00
function! s:process_hunk(bufnr, hunk) abort
let modifications = []
let from_line = a:hunk[0]
let from_count = a:hunk[1]
let to_line = a:hunk[2]
let to_count = a:hunk[3]
2018-03-31 14:56:26 +00:00
if s:is_added(from_count, to_count)
call s:process_added(modifications, from_count, to_count, to_line)
call gitgutter#hunk#increment_lines_added(a:bufnr, to_count)
2018-03-31 14:56:26 +00:00
elseif s:is_removed(from_count, to_count)
call s:process_removed(modifications, from_count, to_count, to_line)
call gitgutter#hunk#increment_lines_removed(a:bufnr, from_count)
2018-03-31 14:56:26 +00:00
elseif s:is_modified(from_count, to_count)
call s:process_modified(modifications, from_count, to_count, to_line)
call gitgutter#hunk#increment_lines_modified(a:bufnr, to_count)
2018-03-31 14:56:26 +00:00
elseif s:is_modified_and_added(from_count, to_count)
call s:process_modified_and_added(modifications, from_count, to_count, to_line)
call gitgutter#hunk#increment_lines_added(a:bufnr, to_count - from_count)
call gitgutter#hunk#increment_lines_modified(a:bufnr, from_count)
2018-03-31 14:56:26 +00:00
elseif s:is_modified_and_removed(from_count, to_count)
call s:process_modified_and_removed(modifications, from_count, to_count, to_line)
call gitgutter#hunk#increment_lines_modified(a:bufnr, to_count)
call gitgutter#hunk#increment_lines_removed(a:bufnr, from_count - to_count)
endif
return modifications
endfunction
2018-03-31 14:56:26 +00:00
function! s:is_added(from_count, to_count) abort
return a:from_count == 0 && a:to_count > 0
endfunction
2018-03-31 14:56:26 +00:00
function! s:is_removed(from_count, to_count) abort
return a:from_count > 0 && a:to_count == 0
endfunction
2018-03-31 14:56:26 +00:00
function! s:is_modified(from_count, to_count) abort
return a:from_count > 0 && a:to_count > 0 && a:from_count == a:to_count
endfunction
2018-03-31 14:56:26 +00:00
function! s:is_modified_and_added(from_count, to_count) abort
return a:from_count > 0 && a:to_count > 0 && a:from_count < a:to_count
endfunction
2018-03-31 14:56:26 +00:00
function! s:is_modified_and_removed(from_count, to_count) abort
return a:from_count > 0 && a:to_count > 0 && a:from_count > a:to_count
endfunction
2018-03-31 14:56:26 +00:00
function! s:process_added(modifications, from_count, to_count, to_line) abort
let offset = 0
while offset < a:to_count
let line_number = a:to_line + offset
call add(a:modifications, [line_number, 'added'])
let offset += 1
endwhile
endfunction
2018-03-31 14:56:26 +00:00
function! s:process_removed(modifications, from_count, to_count, to_line) abort
if a:to_line == 0
call add(a:modifications, [1, 'removed_first_line'])
else
call add(a:modifications, [a:to_line, 'removed'])
endif
endfunction
2018-03-31 14:56:26 +00:00
function! s:process_modified(modifications, from_count, to_count, to_line) abort
let offset = 0
while offset < a:to_count
let line_number = a:to_line + offset
call add(a:modifications, [line_number, 'modified'])
let offset += 1
endwhile
endfunction
2018-03-31 14:56:26 +00:00
function! s:process_modified_and_added(modifications, from_count, to_count, to_line) abort
let offset = 0
while offset < a:from_count
let line_number = a:to_line + offset
call add(a:modifications, [line_number, 'modified'])
let offset += 1
endwhile
while offset < a:to_count
let line_number = a:to_line + offset
call add(a:modifications, [line_number, 'added'])
let offset += 1
endwhile
endfunction
2018-03-31 14:56:26 +00:00
function! s:process_modified_and_removed(modifications, from_count, to_count, to_line) abort
let offset = 0
while offset < a:to_count
let line_number = a:to_line + offset
call add(a:modifications, [line_number, 'modified'])
let offset += 1
endwhile
let a:modifications[-1] = [a:to_line + offset - 1, 'modified_removed']
endfunction
2018-03-31 14:56:26 +00:00
" Returns a diff for the current hunk.
2018-11-01 10:03:42 +00:00
" Assumes there is only 1 current hunk unless the optional argument is given,
" in which case the cursor is in two hunks and the argument specifies the one
" to choose.
"
" Optional argument: 0 (to use the first hunk) or 1 (to use the second).
function! gitgutter#diff#hunk_diff(bufnr, full_diff, ...)
let modified_diff = []
2018-11-01 10:03:42 +00:00
let hunk_index = 0
2018-03-31 14:56:26 +00:00
let keep_line = 1
" Don't keepempty when splitting because the diff we want may not be the
" final one. Instead add trailing NL at end of function.
for line in split(a:full_diff, '\n')
let hunk_info = gitgutter#diff#parse_hunk(line)
if len(hunk_info) == 4 " start of new hunk
let keep_line = gitgutter#hunk#cursor_in_hunk(hunk_info)
2018-11-01 10:03:42 +00:00
if a:0 && hunk_index != a:1
let keep_line = 0
endif
let hunk_index += 1
endif
if keep_line
call add(modified_diff, line)
endif
endfor
2018-03-31 14:56:26 +00:00
return join(modified_diff, "\n")."\n"
endfunction
2018-03-31 14:56:26 +00:00
function! s:write_buffer(bufnr, file)
let bufcontents = getbufline(a:bufnr, 1, '$')
2018-12-17 11:28:27 +00:00
if bufcontents == [''] && line2byte(1) == -1
" Special case: completely empty buffer.
" A nearly empty buffer of only a newline has line2byte(1) == 1.
call writefile([], a:file)
return
endif
2018-03-31 14:56:26 +00:00
if getbufvar(a:bufnr, '&fileformat') ==# 'dos'
2023-07-15 10:43:27 +00:00
if getbufvar(a:bufnr, '&endofline')
call map(bufcontents, 'v:val."\r"')
else
for i in range(len(bufcontents) - 1)
let bufcontents[i] = bufcontents[i] . "\r"
endfor
endif
2018-03-31 14:56:26 +00:00
endif
2019-11-16 15:28:42 +00:00
if getbufvar(a:bufnr, '&endofline')
call add(bufcontents, '')
endif
2018-03-31 14:56:26 +00:00
let fenc = getbufvar(a:bufnr, '&fileencoding')
if fenc !=# &encoding
call map(bufcontents, 'iconv(v:val, &encoding, "'.fenc.'")')
endif
if getbufvar(a:bufnr, '&bomb')
let bufcontents[0]=''.bufcontents[0]
endif
2018-03-31 14:56:26 +00:00
2020-12-04 21:15:32 +00:00
" The file we are writing to is a temporary file. Sometimes the parent
" directory is deleted outside Vim but, because Vim caches the directory
" name at startup and does not check for its existence subsequently, Vim
" does not realise. This causes E482 errors.
try
call writefile(bufcontents, a:file, 'b')
catch /E482/
call mkdir(fnamemodify(a:file, ':h'), '', '0700')
call writefile(bufcontents, a:file, 'b')
endtry
endfunction
2018-03-31 14:56:26 +00:00
function! s:save_last_seen_change(bufnr) abort
call gitgutter#utility#setbufvar(a:bufnr, 'tick', getbufvar(a:bufnr, 'changedtick'))
endfunction