2
0
Fork 0
mirror of https://github.com/amix/vimrc synced 2025-01-02 07:08:41 +00:00
vimrc/sources_non_forked/vim-fugitive/autoload/fugitive.vim

6196 lines
206 KiB
VimL
Raw Normal View History

2018-06-14 10:31:12 +00:00
" Location: autoload/fugitive.vim
" Maintainer: Tim Pope <http://tpo.pe/>
if exists('g:autoloaded_fugitive')
finish
endif
let g:autoloaded_fugitive = 1
if !exists('g:fugitive_git_executable')
let g:fugitive_git_executable = 'git'
2019-08-22 15:36:17 +00:00
elseif g:fugitive_git_executable =~# '^\w\+='
let g:fugitive_git_executable = 'env ' . g:fugitive_git_executable
2018-06-14 10:31:12 +00:00
endif
" Section: Utility
function! s:function(name) abort
2019-11-16 15:28:42 +00:00
return function(substitute(a:name,'^s:',matchstr(expand('<sfile>'), '.*\zs<SNR>\d\+_'),''))
2018-06-14 10:31:12 +00:00
endfunction
function! s:sub(str,pat,rep) abort
return substitute(a:str,'\v\C'.a:pat,a:rep,'')
endfunction
function! s:gsub(str,pat,rep) abort
return substitute(a:str,'\v\C'.a:pat,a:rep,'g')
endfunction
2018-08-25 16:13:42 +00:00
function! s:Uniq(list) abort
let i = 0
let seen = {}
while i < len(a:list)
let str = string(a:list[i])
if has_key(seen, str)
call remove(a:list, i)
else
let seen[str] = 1
let i += 1
endif
endwhile
return a:list
endfunction
2018-06-14 10:31:12 +00:00
function! s:winshell() abort
2020-01-29 02:07:36 +00:00
return has('win32') && &shellcmdflag !~# '^-'
2018-06-14 10:31:12 +00:00
endfunction
function! s:shellesc(arg) abort
2019-08-22 15:36:17 +00:00
if type(a:arg) == type([])
return join(map(copy(a:arg), 's:shellesc(v:val)'))
elseif a:arg =~ '^[A-Za-z0-9_/:.-]\+$'
2018-06-14 10:31:12 +00:00
return a:arg
elseif s:winshell()
return '"'.s:gsub(s:gsub(a:arg, '"', '""'), '\%', '"%"').'"'
else
return shellescape(a:arg)
endif
endfunction
2018-09-25 00:40:17 +00:00
let s:fnameescape = " \t\n*?[{`$\\%#'\"|!<"
2018-06-14 10:31:12 +00:00
function! s:fnameescape(file) abort
2019-08-22 15:36:17 +00:00
if type(a:file) == type([])
return join(map(copy(a:file), 's:fnameescape(v:val)'))
elseif exists('*fnameescape')
2018-06-14 10:31:12 +00:00
return fnameescape(a:file)
else
2018-09-25 00:40:17 +00:00
return escape(a:file, s:fnameescape)
2018-06-14 10:31:12 +00:00
endif
endfunction
function! s:throw(string) abort
2019-08-22 15:36:17 +00:00
throw 'fugitive: '.a:string
2018-06-14 10:31:12 +00:00
endfunction
2019-08-22 15:36:17 +00:00
function! s:DirCheck(...) abort
2019-11-16 15:28:42 +00:00
if !empty(a:0 ? s:Dir(a:1) : s:Dir())
return ''
elseif empty(bufname(''))
return 'return ' . string('echoerr "fugitive: working directory does not belong to a Git repository"')
else
return 'return ' . string('echoerr "fugitive: file does not belong to a Git repository"')
2019-08-22 15:36:17 +00:00
endif
2018-06-14 10:31:12 +00:00
endfunction
2019-08-22 15:36:17 +00:00
function! s:Mods(mods, ...) abort
let mods = substitute(a:mods, '\C<mods>', '', '')
let mods = mods =~# '\S$' ? mods . ' ' : mods
if a:0 && mods !~# '\<\%(aboveleft\|belowright\|leftabove\|rightbelow\|topleft\|botright\|tab\)\>'
let mods = a:1 . ' ' . mods
2018-06-14 10:31:12 +00:00
endif
2019-08-22 15:36:17 +00:00
return substitute(mods, '\s\+', ' ', 'g')
2018-06-14 10:31:12 +00:00
endfunction
2019-08-22 15:36:17 +00:00
function! s:Slash(path) abort
if exists('+shellslash')
return tr(a:path, '\', '/')
2018-07-04 10:53:25 +00:00
else
return a:path
endif
endfunction
2018-08-25 16:13:42 +00:00
function! s:Resolve(path) abort
let path = resolve(a:path)
if has('win32')
2019-08-22 15:36:17 +00:00
let path = FugitiveVimPath(fnamemodify(fnamemodify(path, ':h'), ':p') . fnamemodify(path, ':t'))
2018-08-25 16:13:42 +00:00
endif
return path
endfunction
function! s:cpath(path, ...) abort
if exists('+fileignorecase') && &fileignorecase
2019-08-22 15:36:17 +00:00
let path = FugitiveVimPath(tolower(a:path))
2018-08-25 16:13:42 +00:00
else
2019-08-22 15:36:17 +00:00
let path = FugitiveVimPath(a:path)
2018-08-25 16:13:42 +00:00
endif
return a:0 ? path ==# s:cpath(a:1) : path
endfunction
2019-01-08 10:11:54 +00:00
function! s:Cd(...) abort
let cd = exists('*haslocaldir') && haslocaldir() ? 'lcd' : exists(':tcd') && haslocaldir(-1) ? 'tcd' : 'cd'
if !a:0
return cd
endif
let cwd = getcwd()
if s:cpath(cwd, a:1)
return ''
endif
exe cd s:fnameescape(a:1)
return cd . ' ' . s:fnameescape(cwd)
endfunction
2018-06-14 10:31:12 +00:00
let s:executables = {}
function! s:executable(binary) abort
if !has_key(s:executables, a:binary)
let s:executables[a:binary] = executable(a:binary)
endif
return s:executables[a:binary]
endfunction
2020-01-29 02:07:36 +00:00
if !exists('s:temp_scripts')
let s:temp_scripts = {}
endif
function! s:TempScript(...) abort
let body = join(a:000, "\n")
if !has_key(s:temp_scripts, body)
let temp = tempname() . '.sh'
call writefile(['#!/bin/sh'] + a:000, temp)
let s:temp_scripts[body] = temp
endif
return FugitiveGitPath(s:temp_scripts[body])
endfunction
2019-11-16 15:28:42 +00:00
function! s:DoAutocmd(cmd) abort
if v:version >= 704 || (v:version == 703 && has('patch442'))
return 'doautocmd <nomodeline>' . a:cmd
elseif &modelines > 0
return 'try|set modelines=0|doautocmd ' . a:cmd . '|finally|set modelines=' . &modelines . '|endtry'
else
return 'doautocmd ' . a:cmd
endif
endfunction
2019-08-22 15:36:17 +00:00
let s:nowait = v:version >= 704 ? '<nowait>' : ''
function! s:Map(mode, lhs, rhs, ...) abort
for mode in split(a:mode, '\zs')
let flags = (a:0 ? a:1 : '') . (a:rhs =~# '<Plug>' ? '' : '<script>')
let head = a:lhs
let tail = ''
let keys = get(g:, mode.'remap', {})
if type(keys) == type([])
return
2018-08-25 16:13:42 +00:00
endif
2019-08-22 15:36:17 +00:00
while !empty(head)
if has_key(keys, head)
let head = keys[head]
if empty(head)
return
endif
break
endif
let tail = matchstr(head, '<[^<>]*>$\|.$') . tail
let head = substitute(head, '<[^<>]*>$\|.$', '', '')
endwhile
if flags !~# '<unique>' || empty(mapcheck(head.tail, mode))
exe mode.'map <buffer>' s:nowait flags head.tail a:rhs
if a:0 > 1
let b:undo_ftplugin = get(b:, 'undo_ftplugin', 'exe') .
\ '|sil! exe "' . mode . 'unmap <buffer> ' . head.tail . '"'
endif
2018-08-25 16:13:42 +00:00
endif
2019-08-22 15:36:17 +00:00
endfor
endfunction
" Section: Quickfix
function! s:QuickfixGet(nr, ...) abort
if a:nr < 0
return call('getqflist', a:000)
else
return call('getloclist', [a:nr] + a:000)
2018-08-25 16:13:42 +00:00
endif
2018-06-14 10:31:12 +00:00
endfunction
2019-08-22 15:36:17 +00:00
function! s:QuickfixSet(nr, ...) abort
if a:nr < 0
return call('setqflist', a:000)
else
return call('setloclist', [a:nr] + a:000)
endif
endfunction
function! s:QuickfixCreate(nr, opts) abort
if has('patch-7.4.2200')
call s:QuickfixSet(a:nr, [], ' ', a:opts)
else
call s:QuickfixSet(a:nr, [], ' ')
endif
endfunction
2020-01-07 12:45:07 +00:00
function! s:QuickfixStream(nr, event, title, cmd, first, callback, ...) abort
let opts = {'title': a:title, 'context': {'items': []}}
call s:QuickfixCreate(a:nr, opts)
let event = (a:nr < 0 ? 'c' : 'l') . 'fugitive-' . a:event
silent exe s:DoAutocmd('QuickFixCmdPre ' . event)
2019-08-22 15:36:17 +00:00
let winnr = winnr()
exe a:nr < 0 ? 'copen' : 'lopen'
if winnr != winnr()
wincmd p
endif
let buffer = []
let lines = split(s:SystemError(s:shellesc(a:cmd))[0], "\n")
for line in lines
call extend(buffer, call(a:callback, a:000 + [line]))
if len(buffer) >= 20
2020-01-07 12:45:07 +00:00
let contexts = map(copy(buffer), 'get(v:val, "context", {})')
lockvar contexts
call extend(opts.context.items, contexts)
unlet contexts
2019-08-22 15:36:17 +00:00
call s:QuickfixSet(a:nr, remove(buffer, 0, -1), 'a')
redraw
endif
endfor
2020-01-07 12:45:07 +00:00
call extend(buffer, call(a:callback, a:000 + [0]))
call extend(opts.context.items, map(copy(buffer), 'get(v:val, "context", {})'))
lockvar opts.context.items
call s:QuickfixSet(a:nr, buffer, 'a')
2019-08-22 15:36:17 +00:00
2020-01-07 12:45:07 +00:00
silent exe s:DoAutocmd('QuickFixCmdPost ' . event)
2019-08-22 15:36:17 +00:00
if a:first && len(s:QuickfixGet(a:nr))
call s:BlurStatus()
return a:nr < 0 ? 'cfirst' : 'lfirst'
else
return 'exe'
endif
2018-07-30 21:18:16 +00:00
endfunction
2020-01-29 02:07:36 +00:00
let s:common_efm = ''
\ . '%+Egit:%.%#,'
\ . '%+Eusage:%.%#,'
\ . '%+Eerror:%.%#,'
\ . '%+Efatal:%.%#,'
\ . '%-G%.%#%\e[K%.%#,'
\ . '%-G%.%#%\r%.%\+'
function! fugitive#Cwindow() abort
if &buftype == 'quickfix'
cwindow
else
botright cwindow
if &buftype == 'quickfix'
wincmd p
endif
endif
endfunction
2018-08-25 16:13:42 +00:00
" Section: Git
2019-08-22 15:36:17 +00:00
function! s:UserCommandList(...) abort
let git = split(get(g:, 'fugitive_git_command', g:fugitive_git_executable), '\s\+')
let dir = a:0 ? s:Dir(a:1) : ''
if len(dir)
let tree = s:Tree(dir)
if empty(tree)
call add(git, '--git-dir=' . FugitiveGitPath(dir))
elseif len(tree) && s:cpath(tree) !=# s:cpath(getcwd())
if fugitive#GitVersion(1, 8, 5)
call extend(git, ['-C', FugitiveGitPath(tree)])
else
throw 'fugitive: Git 1.8.5 or higher required to change directory'
endif
endif
endif
return git
2018-08-25 16:13:42 +00:00
endfunction
2019-08-22 15:36:17 +00:00
function! s:UserCommand(...) abort
return s:shellesc(call('s:UserCommandList', a:0 ? [a:1] : []) + (a:0 ? a:2 : []))
2018-07-19 12:52:53 +00:00
endfunction
let s:git_versions = {}
2018-06-14 10:31:12 +00:00
function! fugitive#GitVersion(...) abort
if !has_key(s:git_versions, g:fugitive_git_executable)
2019-08-22 15:36:17 +00:00
let s:git_versions[g:fugitive_git_executable] = matchstr(system(g:fugitive_git_executable.' --version'), '\d[^[:space:]]\+')
2018-06-14 10:31:12 +00:00
endif
2019-03-08 11:04:56 +00:00
if !a:0
return s:git_versions[g:fugitive_git_executable]
endif
let components = split(s:git_versions[g:fugitive_git_executable], '\D\+')
if empty(components)
return -1
endif
for i in range(len(a:000))
if a:000[i] > +get(components, i)
return 0
elseif a:000[i] < +get(components, i)
return 1
endif
endfor
return a:000[i] ==# get(components, i)
2018-06-14 10:31:12 +00:00
endfunction
2018-08-25 16:13:42 +00:00
let s:commondirs = {}
function! fugitive#CommonDir(dir) abort
if empty(a:dir)
return ''
endif
if !has_key(s:commondirs, a:dir)
if getfsize(a:dir . '/HEAD') < 10
let s:commondirs[a:dir] = ''
elseif filereadable(a:dir . '/commondir')
2019-08-22 15:36:17 +00:00
let cdir = get(readfile(a:dir . '/commondir', 1), 0, '')
if cdir =~# '^/\|^\a:/'
let s:commondirs[a:dir] = s:Slash(FugitiveVimPath(cdir))
2018-08-25 16:13:42 +00:00
else
2019-08-22 15:36:17 +00:00
let s:commondirs[a:dir] = simplify(a:dir . '/' . cdir)
2018-08-25 16:13:42 +00:00
endif
else
let s:commondirs[a:dir] = a:dir
endif
endif
return s:commondirs[a:dir]
endfunction
2019-03-08 11:04:56 +00:00
function! s:Dir(...) abort
return a:0 ? FugitiveGitDir(a:1) : FugitiveGitDir()
endfunction
2018-07-30 21:18:16 +00:00
function! s:Tree(...) abort
2019-03-08 11:04:56 +00:00
return a:0 ? FugitiveWorkTree(a:1) : FugitiveWorkTree()
2018-07-30 21:18:16 +00:00
endfunction
2019-08-22 15:36:17 +00:00
function! s:HasOpt(args, ...) abort
let args = a:args[0 : index(a:args, '--')]
let opts = copy(a:000)
if type(opts[0]) == type([])
if empty(args) || index(opts[0], args[0]) == -1
return 0
endif
call remove(opts, 0)
endif
for opt in opts
if index(args, opt) != -1
return 1
endif
endfor
endfunction
2018-09-25 00:40:17 +00:00
function! s:PreparePathArgs(cmd, dir, literal) abort
2019-03-08 11:04:56 +00:00
let literal_supported = fugitive#GitVersion(1, 9)
2018-09-25 00:40:17 +00:00
if a:literal && literal_supported
2018-08-25 16:13:42 +00:00
call insert(a:cmd, '--literal-pathspecs')
endif
let split = index(a:cmd, '--')
2019-08-22 15:36:17 +00:00
for i in range(split < 0 ? len(a:cmd) : split)
if type(a:cmd[i]) == type(0)
let a:cmd[i] = fugitive#Path(bufname(a:cmd[i]), './', a:dir)
endif
endfor
2018-09-25 00:40:17 +00:00
if split < 0
2018-08-25 16:13:42 +00:00
return a:cmd
endif
for i in range(split + 1, len(a:cmd) - 1)
2018-09-25 00:40:17 +00:00
if type(a:cmd[i]) == type(0)
let a:cmd[i] = fugitive#Path(bufname(a:cmd[i]), './', a:dir)
elseif a:literal
let a:cmd[i] = fugitive#Path(a:cmd[i], './', a:dir)
elseif !literal_supported
let a:cmd[i] = substitute(a:cmd[i], '^:\%(/\|([^)]*)\)\=:\=', './', '')
endif
2018-08-25 16:13:42 +00:00
endfor
return a:cmd
endfunction
2018-09-25 00:40:17 +00:00
let s:prepare_env = {
\ 'sequence.editor': 'GIT_SEQUENCE_EDITOR',
\ 'core.editor': 'GIT_EDITOR',
\ 'core.askpass': 'GIT_ASKPASS',
\ }
2019-08-22 15:36:17 +00:00
function! fugitive#PrepareDirEnvArgv(...) abort
if a:0 && type(a:1) ==# type([])
2018-09-25 00:40:17 +00:00
let cmd = a:000[1:-1] + a:1
else
let cmd = copy(a:000)
endif
2019-08-22 15:36:17 +00:00
let env = {}
2018-09-25 00:40:17 +00:00
let i = 0
while i < len(cmd)
if cmd[i] =~# '^$\|[\/.]' && cmd[i] !~# '^-'
let dir = remove(cmd, 0)
2019-08-22 15:36:17 +00:00
elseif cmd[i] =~# '^--git-dir='
let dir = remove(cmd, 0)[10:-1]
2018-09-25 00:40:17 +00:00
elseif type(cmd[i]) ==# type(0)
2019-03-08 11:04:56 +00:00
let dir = s:Dir(remove(cmd, i))
2018-09-25 00:40:17 +00:00
elseif cmd[i] ==# '-c' && len(cmd) > i + 1
let key = matchstr(cmd[i+1], '^[^=]*')
if has_key(s:prepare_env, tolower(key)) || key !~# '\.'
let var = get(s:prepare_env, tolower(key), key)
let val = matchstr(cmd[i+1], '=\zs.*')
2019-08-22 15:36:17 +00:00
let env[var] = val
2018-09-25 00:40:17 +00:00
endif
2019-03-08 11:04:56 +00:00
if fugitive#GitVersion(1, 8) && cmd[i+1] =~# '\.'
2018-09-25 00:40:17 +00:00
let i += 2
2019-03-08 11:04:56 +00:00
else
call remove(cmd, i, i + 1)
2018-09-25 00:40:17 +00:00
endif
elseif cmd[i] =~# '^--.*pathspecs$'
let explicit_pathspec_option = 1
2019-03-08 11:04:56 +00:00
if fugitive#GitVersion(1, 9)
2018-09-25 00:40:17 +00:00
let i += 1
2019-03-08 11:04:56 +00:00
else
call remove(cmd, i)
2018-09-25 00:40:17 +00:00
endif
elseif cmd[i] !~# '^-'
break
2018-07-30 21:18:16 +00:00
else
2018-09-25 00:40:17 +00:00
let i += 1
2018-07-30 21:18:16 +00:00
endif
2018-09-25 00:40:17 +00:00
endwhile
if !exists('dir')
2019-03-08 11:04:56 +00:00
let dir = s:Dir()
2018-07-30 21:18:16 +00:00
endif
2018-09-25 00:40:17 +00:00
call s:PreparePathArgs(cmd, dir, !exists('explicit_pathspec_option'))
2019-08-22 15:36:17 +00:00
return [dir, env, cmd]
endfunction
2020-01-29 02:07:36 +00:00
function! s:BuildEnvPrefix(env) abort
let pre = ''
let env = items(a:env)
if empty(env)
return ''
elseif &shellcmdflag =~# '-Command'
return join(map(env, '"$Env:" . v:val[0] . " = ''" . substitute(v:val[1], "''", "''''", "g") . "''; "'), '')
elseif s:winshell()
return join(map(env, '"set " . substitute(join(v:val, "="), "[&|<>^]", "^^^&", "g") . "& "'), '')
else
return 'env ' . s:shellesc(map(env, 'join(v:val, "=")')) . ' '
endif
endfunction
2019-08-22 15:36:17 +00:00
function! s:BuildShell(dir, env, args) abort
let cmd = copy(a:args)
let tree = s:Tree(a:dir)
2020-01-29 02:07:36 +00:00
let pre = s:BuildEnvPrefix(a:env)
2018-09-25 00:40:17 +00:00
if empty(tree) || index(cmd, '--') == len(cmd) - 1
2019-08-22 15:36:17 +00:00
call insert(cmd, '--git-dir=' . FugitiveGitPath(a:dir))
elseif fugitive#GitVersion(1, 8, 5)
call extend(cmd, ['-C', FugitiveGitPath(tree)], 'keep')
2019-03-08 11:04:56 +00:00
else
2019-08-22 15:36:17 +00:00
let pre = 'cd ' . s:shellesc(tree) . (s:winshell() ? '& ' : '; ') . pre
2018-08-25 16:13:42 +00:00
endif
2019-08-22 15:36:17 +00:00
return pre . g:fugitive_git_executable . ' ' . join(map(cmd, 's:shellesc(v:val)'))
endfunction
function! fugitive#Prepare(...) abort
let [dir, env, argv] = call('fugitive#PrepareDirEnvArgv', a:000)
return s:BuildShell(dir, env, argv)
endfunction
function! s:SystemError(cmd, ...) abort
try
if &shellredir ==# '>' && &shell =~# 'sh\|cmd'
let shellredir = &shellredir
if &shell =~# 'csh'
set shellredir=>&
else
set shellredir=>%s\ 2>&1
endif
endif
let out = call('system', [type(a:cmd) ==# type([]) ? fugitive#Prepare(a:cmd) : a:cmd] + a:000)
return [out, v:shell_error]
catch /^Vim\%((\a\+)\)\=:E484:/
let opts = ['shell', 'shellcmdflag', 'shellredir', 'shellquote', 'shellxquote', 'shellxescape', 'shellslash']
call filter(opts, 'exists("+".v:val) && !empty(eval("&".v:val))')
call map(opts, 'v:val."=".eval("&".v:val)')
call s:throw('failed to run `' . a:cmd . '` with ' . join(opts, ' '))
finally
if exists('shellredir')
let &shellredir = shellredir
endif
endtry
endfunction
function! s:ChompError(...) abort
let [out, exec_error] = s:SystemError(call('fugitive#Prepare', a:000))
return [s:sub(out, '\n$', ''), exec_error]
endfunction
function! s:ChompDefault(default, ...) abort
let [out, exec_error] = call('s:ChompError', a:000)
return exec_error ? a:default : out
endfunction
function! s:LinesError(...) abort
let [out, exec_error] = call('s:ChompError', a:000)
return [len(out) && !exec_error ? split(out, "\n", 1) : [], exec_error]
endfunction
function! s:NullError(...) abort
let [out, exec_error] = s:SystemError(call('fugitive#Prepare', a:000))
return [exec_error ? [] : split(out, "\1"), exec_error ? substitute(out, "\n$", "", "") : '', exec_error]
2018-08-25 16:13:42 +00:00
endfunction
2018-09-25 00:40:17 +00:00
function! s:TreeChomp(...) abort
2019-08-22 15:36:17 +00:00
let cmd = call('fugitive#Prepare', a:000)
let [out, exec_error] = s:SystemError(cmd)
let out = s:sub(out, '\n$', '')
if !exec_error
return out
endif
throw 'fugitive: error running `' . cmd . '`: ' . out
endfunction
function! s:EchoExec(...) abort
echo call('s:ChompError', a:000)[0]
call fugitive#ReloadStatus(-1, 1)
return 'checktime'
2018-09-25 00:40:17 +00:00
endfunction
2019-11-16 15:28:42 +00:00
let s:head_cache = {}
2018-08-25 16:13:42 +00:00
function! fugitive#Head(...) abort
2019-03-08 11:04:56 +00:00
let dir = a:0 > 1 ? a:2 : s:Dir()
2019-11-16 15:28:42 +00:00
if empty(dir)
return ''
endif
let file = fugitive#Find('.git/HEAD', dir)
let ftime = getftime(file)
if ftime == -1
2018-08-25 16:13:42 +00:00
return ''
2019-11-16 15:28:42 +00:00
elseif ftime != get(s:head_cache, dir, [-1])[0]
let s:head_cache[dir] = [ftime, readfile(file)[0]]
2018-08-25 16:13:42 +00:00
endif
2019-11-16 15:28:42 +00:00
let head = s:head_cache[dir][1]
2018-08-25 16:13:42 +00:00
if head =~# '^ref: '
return substitute(head, '\C^ref: \%(refs/\%(heads/\|remotes/\|tags/\)\=\)\=', '', '')
2019-08-22 15:36:17 +00:00
elseif head =~# '^\x\{40,\}$'
2018-08-25 16:13:42 +00:00
let len = a:0 ? a:1 : 0
return len < 0 ? head : len ? head[0:len-1] : ''
else
return ''
endif
endfunction
2018-07-30 21:18:16 +00:00
function! fugitive#RevParse(rev, ...) abort
2019-08-22 15:36:17 +00:00
let [hash, exec_error] = s:ChompError([a:0 ? a:1 : s:Dir(), 'rev-parse', '--verify', a:rev, '--'])
if !exec_error && hash =~# '^\x\{40,\}$'
2018-07-30 21:18:16 +00:00
return hash
endif
2019-08-22 15:36:17 +00:00
throw 'fugitive: rev-parse '.a:rev.': '.hash
2018-07-30 21:18:16 +00:00
endfunction
2019-01-08 10:11:54 +00:00
function! s:ConfigTimestamps(dir, dict) abort
let files = ['/etc/gitconfig', '~/.gitconfig',
\ len($XDG_CONFIG_HOME) ? $XDG_CONFIG_HOME . '/git/config' : '~/.config/git/config']
if len(a:dir)
call add(files, fugitive#Find('.git/config', a:dir))
endif
call extend(files, get(a:dict, 'include.path', []))
return join(map(files, 'getftime(expand(v:val))'), ',')
endfunction
let s:config = {}
function! fugitive#Config(...) abort
2019-03-08 11:04:56 +00:00
let dir = s:Dir()
2019-01-08 10:11:54 +00:00
let name = ''
if a:0 >= 2 && type(a:2) == type({})
let name = substitute(a:1, '^[^.]\+\|[^.]\+$', '\L&', 'g')
return len(a:1) ? get(get(a:2, name, []), 0, '') : a:2
elseif a:0 >= 2
let dir = a:2
let name = a:1
elseif a:0 == 1 && type(a:1) == type({})
return a:1
elseif a:0 == 1 && a:1 =~# '^[[:alnum:]-]\+\.'
let name = a:1
elseif a:0 == 1
let dir = a:1
endif
let name = substitute(name, '^[^.]\+\|[^.]\+$', '\L&', 'g')
let key = len(dir) ? dir : '_'
if has_key(s:config, key) && s:config[key][0] ==# s:ConfigTimestamps(dir, s:config[key][1])
let dict = s:config[key][1]
else
let dict = {}
2019-08-22 15:36:17 +00:00
let [lines, message, exec_error] = s:NullError([dir, 'config', '--list', '-z'])
if exec_error
2019-01-08 10:11:54 +00:00
return {}
endif
for line in lines
let key = matchstr(line, "^[^\n]*")
if !has_key(dict, key)
let dict[key] = []
endif
call add(dict[key], strpart(line, len(key) + 1))
endfor
let s:config[dir] = [s:ConfigTimestamps(dir, dict), dict]
lockvar! dict
endif
return len(name) ? get(get(dict, name, []), 0, '') : dict
2018-07-30 21:18:16 +00:00
endfunction
function! s:Remote(dir) abort
let head = FugitiveHead(0, a:dir)
let remote = len(head) ? fugitive#Config('branch.' . head . '.remote') : ''
let i = 10
while remote ==# '.' && i > 0
let head = matchstr(fugitive#Config('branch.' . head . '.merge'), 'refs/heads/\zs.*')
let remote = len(head) ? fugitive#Config('branch.' . head . '.remote') : ''
let i -= 1
endwhile
2018-08-25 16:13:42 +00:00
return remote =~# '^\.\=$' ? 'origin' : remote
2018-07-30 21:18:16 +00:00
endfunction
function! fugitive#RemoteUrl(...) abort
2019-03-08 11:04:56 +00:00
let dir = a:0 > 1 ? a:2 : s:Dir()
2018-07-30 21:18:16 +00:00
let remote = !a:0 || a:1 =~# '^\.\=$' ? s:Remote(dir) : a:1
2019-03-08 11:04:56 +00:00
if !fugitive#GitVersion(2, 7)
2018-07-30 21:18:16 +00:00
return fugitive#Config('remote.' . remote . '.url')
endif
2019-08-22 15:36:17 +00:00
return s:ChompDefault('', [dir, 'remote', 'get-url', remote, '--'])
2018-07-30 21:18:16 +00:00
endfunction
2018-08-25 16:13:42 +00:00
" Section: Repository Object
2018-06-14 10:31:12 +00:00
function! s:add_methods(namespace, method_names) abort
for name in a:method_names
let s:{a:namespace}_prototype[name] = s:function('s:'.a:namespace.'_'.name)
endfor
endfunction
let s:repo_prototype = {}
let s:repos = {}
2018-07-30 21:18:16 +00:00
function! fugitive#repo(...) abort
2019-08-22 15:36:17 +00:00
let dir = a:0 ? s:Dir(a:1) : (len(s:Dir()) ? s:Dir() : FugitiveExtractGitDir(expand('%:p')))
2018-06-14 10:31:12 +00:00
if dir !=# ''
if has_key(s:repos, dir)
let repo = get(s:repos, dir)
else
let repo = {'git_dir': dir}
let s:repos[dir] = repo
endif
2018-08-25 16:13:42 +00:00
return extend(repo, s:repo_prototype, 'keep')
2018-06-14 10:31:12 +00:00
endif
2019-08-22 15:36:17 +00:00
call s:throw('not a Git repository')
2018-06-14 10:31:12 +00:00
endfunction
function! s:repo_dir(...) dict abort
return join([self.git_dir]+a:000,'/')
endfunction
function! s:repo_tree(...) dict abort
2018-07-30 21:18:16 +00:00
let dir = s:Tree(self.git_dir)
2018-06-14 10:31:12 +00:00
if dir ==# ''
call s:throw('no work tree')
else
return join([dir]+a:000,'/')
endif
endfunction
function! s:repo_bare() dict abort
if self.dir() =~# '/\.git$'
return 0
else
2018-07-30 21:18:16 +00:00
return s:Tree(self.git_dir) ==# ''
endif
endfunction
2018-11-01 10:03:42 +00:00
function! s:repo_find(object) dict abort
return fugitive#Find(a:object, self.git_dir)
2018-07-30 21:18:16 +00:00
endfunction
2018-08-25 16:13:42 +00:00
function! s:repo_translate(rev) dict abort
2018-11-01 10:03:42 +00:00
return s:Slash(fugitive#Find(substitute(a:rev, '^/', ':(top)', ''), self.git_dir))
2018-06-14 10:31:12 +00:00
endfunction
function! s:repo_head(...) dict abort
2018-08-25 16:13:42 +00:00
return fugitive#Head(a:0 ? a:1 : 0, self.git_dir)
2018-06-14 10:31:12 +00:00
endfunction
2018-11-01 10:03:42 +00:00
call s:add_methods('repo',['dir','tree','bare','find','translate','head'])
2018-06-14 10:31:12 +00:00
2018-09-25 00:40:17 +00:00
function! s:repo_prepare(...) dict abort
return call('fugitive#Prepare', [self.git_dir] + a:000)
endfunction
2018-06-14 10:31:12 +00:00
function! s:repo_git_command(...) dict abort
2018-07-30 21:18:16 +00:00
let git = s:UserCommand() . ' --git-dir='.s:shellesc(self.git_dir)
2018-06-14 10:31:12 +00:00
return git.join(map(copy(a:000),'" ".s:shellesc(v:val)'),'')
endfunction
function! s:repo_git_chomp(...) dict abort
2018-09-25 00:40:17 +00:00
let git = g:fugitive_git_executable . ' --git-dir='.s:shellesc(self.git_dir)
let output = git . join(map(copy(a:000),'" ".s:shellesc(v:val)'),'')
2019-08-22 15:36:17 +00:00
return s:sub(system(output), '\n$', '')
2018-06-14 10:31:12 +00:00
endfunction
function! s:repo_git_chomp_in_tree(...) dict abort
2019-01-08 10:11:54 +00:00
let cdback = s:Cd(self.tree())
2018-06-14 10:31:12 +00:00
try
2018-07-30 21:18:16 +00:00
return call(self.git_chomp, a:000, self)
2018-06-14 10:31:12 +00:00
finally
2019-01-08 10:11:54 +00:00
execute cdback
2018-06-14 10:31:12 +00:00
endtry
endfunction
function! s:repo_rev_parse(rev) dict abort
2018-07-30 21:18:16 +00:00
return fugitive#RevParse(a:rev, self.git_dir)
2018-06-14 10:31:12 +00:00
endfunction
2018-09-25 00:40:17 +00:00
call s:add_methods('repo',['prepare','git_command','git_chomp','git_chomp_in_tree','rev_parse'])
2018-06-14 10:31:12 +00:00
function! s:repo_superglob(base) dict abort
2019-05-17 14:09:13 +00:00
return map(fugitive#CompleteObject(a:base, self.git_dir), 'substitute(v:val, ''\\\(.\)'', ''\1'', "g")')
2018-06-14 10:31:12 +00:00
endfunction
2018-07-30 21:18:16 +00:00
call s:add_methods('repo',['superglob'])
2018-06-14 10:31:12 +00:00
2018-07-30 21:18:16 +00:00
function! s:repo_config(name) dict abort
return fugitive#Config(a:name, self.git_dir)
endfunction
2018-06-14 10:31:12 +00:00
function! s:repo_user() dict abort
2018-07-30 21:18:16 +00:00
let username = self.config('user.name')
let useremail = self.config('user.email')
2018-06-14 10:31:12 +00:00
return username.' <'.useremail.'>'
endfunction
2018-07-30 21:18:16 +00:00
call s:add_methods('repo',['config', 'user'])
2018-06-14 10:31:12 +00:00
2018-08-25 16:13:42 +00:00
" Section: File API
2018-06-14 10:31:12 +00:00
2018-07-04 10:53:25 +00:00
function! s:DirCommitFile(path) abort
2019-08-22 15:36:17 +00:00
let vals = matchlist(s:Slash(a:path), '\c^fugitive:\%(//\)\=\(.\{-\}\)\%(//\|::\)\(\x\{40,\}\|[0-3]\)\(/.*\)\=$')
2018-07-04 10:53:25 +00:00
if empty(vals)
return ['', '', '']
endif
2018-07-19 12:52:53 +00:00
return vals[1:3]
endfunction
function! s:DirRev(url) abort
let [dir, commit, file] = s:DirCommitFile(a:url)
return [dir, (commit =~# '^.$' ? ':' : '') . commit . substitute(file, '^/', ':', '')]
2018-07-04 10:53:25 +00:00
endfunction
2019-11-16 15:28:42 +00:00
let s:merge_heads = ['MERGE_HEAD', 'REBASE_HEAD', 'CHERRY_PICK_HEAD', 'REVERT_HEAD']
function! s:MergeHead(...) abort
let dir = fugitive#Find('.git/', a:0 ? a:1 : s:Dir())
for head in s:merge_heads
if filereadable(dir . head)
return head
endif
endfor
return ''
endfunction
2018-08-25 16:13:42 +00:00
function! s:Owner(path, ...) abort
2019-03-08 11:04:56 +00:00
let dir = a:0 ? a:1 : s:Dir()
2018-08-25 16:13:42 +00:00
if empty(dir)
return ''
endif
2019-08-22 15:36:17 +00:00
let actualdir = fugitive#Find('.git/', dir)
2018-08-25 16:13:42 +00:00
let [pdir, commit, file] = s:DirCommitFile(a:path)
2018-11-01 10:03:42 +00:00
if s:cpath(dir, pdir)
2019-08-22 15:36:17 +00:00
if commit =~# '^\x\{40,\}$'
2018-11-01 10:03:42 +00:00
return commit
elseif commit ==# '2'
return 'HEAD^{}'
2019-11-16 15:28:42 +00:00
elseif commit ==# '0'
return ''
2018-11-01 10:03:42 +00:00
endif
2019-11-16 15:28:42 +00:00
let merge_head = s:MergeHead()
if empty(merge_head)
2018-11-01 10:03:42 +00:00
return ''
endif
if commit ==# '3'
return merge_head . '^{}'
elseif commit ==# '1'
return s:TreeChomp('merge-base', 'HEAD', merge_head, '--')
endif
2018-08-25 16:13:42 +00:00
endif
let path = fnamemodify(a:path, ':p')
2019-08-22 15:36:17 +00:00
if s:cpath(actualdir, strpart(path, 0, len(actualdir))) && a:path =~# 'HEAD$'
return strpart(path, len(actualdir))
2018-08-25 16:13:42 +00:00
endif
2019-08-22 15:36:17 +00:00
let refs = fugitive#Find('.git/refs', dir)
2018-08-25 16:13:42 +00:00
if s:cpath(refs . '/', path[0 : len(refs)]) && path !~# '[\/]$'
return strpart(path, len(refs) - 4)
endif
return ''
endfunction
2018-07-30 21:18:16 +00:00
function! fugitive#Real(url) abort
2018-08-25 16:13:42 +00:00
if empty(a:url)
return ''
endif
2018-07-04 10:53:25 +00:00
let [dir, commit, file] = s:DirCommitFile(a:url)
if len(dir)
2018-07-30 21:18:16 +00:00
let tree = s:Tree(dir)
2019-08-22 15:36:17 +00:00
return FugitiveVimPath((len(tree) ? tree : dir) . file)
2018-07-30 21:18:16 +00:00
endif
2018-08-25 16:13:42 +00:00
let pre = substitute(matchstr(a:url, '^\a\a\+\ze:'), '^.', '\u&', '')
if len(pre) && pre !=? 'fugitive' && exists('*' . pre . 'Real')
let url = {pre}Real(a:url)
else
let url = fnamemodify(a:url, ':p' . (a:url =~# '[\/]$' ? '' : ':s?[\/]$??'))
2018-07-04 10:53:25 +00:00
endif
2019-08-22 15:36:17 +00:00
return FugitiveVimPath(empty(url) ? a:url : url)
2018-07-04 10:53:25 +00:00
endfunction
2018-07-30 21:18:16 +00:00
function! fugitive#Path(url, ...) abort
2018-09-25 00:40:17 +00:00
if empty(a:url)
return ''
endif
2019-03-08 11:04:56 +00:00
let dir = a:0 > 1 ? a:2 : s:Dir()
2018-09-25 00:40:17 +00:00
let tree = s:Tree(dir)
if !a:0
2018-07-30 21:18:16 +00:00
return fugitive#Real(a:url)
2018-09-25 00:40:17 +00:00
elseif a:1 =~# '\.$'
let path = s:Slash(fugitive#Real(a:url))
let cwd = getcwd()
let lead = ''
while s:cpath(tree . '/', (cwd . '/')[0 : len(tree)])
if s:cpath(cwd . '/', path[0 : len(cwd)])
if strpart(path, len(cwd) + 1) =~# '^\.git\%(/\|$\)'
break
endif
return a:1[0:-2] . (empty(lead) ? './' : lead) . strpart(path, len(cwd) + 1)
endif
let cwd = fnamemodify(cwd, ':h')
let lead .= '../'
endwhile
return a:1[0:-2] . path
2018-07-30 21:18:16 +00:00
endif
2019-03-08 11:04:56 +00:00
let url = a:url
2019-08-22 15:36:17 +00:00
let temp_state = s:TempState(url)
if has_key(temp_state, 'bufnr')
let url = bufname(temp_state.bufnr)
2019-03-08 11:04:56 +00:00
endif
let url = s:Slash(fnamemodify(url, ':p'))
2018-08-25 16:13:42 +00:00
if url =~# '/$' && s:Slash(a:url) !~# '/$'
2018-07-30 21:18:16 +00:00
let url = url[0:-2]
endif
let [argdir, commit, file] = s:DirCommitFile(a:url)
if len(argdir) && s:cpath(argdir) !=# s:cpath(dir)
let file = ''
elseif len(dir) && s:cpath(url[0 : len(dir)]) ==# s:cpath(dir . '/')
let file = '/.git'.url[strlen(dir) : -1]
elseif len(tree) && s:cpath(url[0 : len(tree)]) ==# s:cpath(tree . '/')
let file = url[len(tree) : -1]
2019-01-08 10:11:54 +00:00
elseif s:cpath(url) ==# s:cpath(tree)
2018-07-30 21:18:16 +00:00
let file = '/'
endif
2018-08-25 16:13:42 +00:00
if empty(file) && a:1 =~# '^$\|^[.:]/$'
2019-08-22 15:36:17 +00:00
return FugitiveGitPath(fugitive#Real(a:url))
2018-07-30 21:18:16 +00:00
endif
return substitute(file, '^/', a:1, '')
2018-07-04 10:53:25 +00:00
endfunction
2018-08-25 16:13:42 +00:00
function! s:Relative(...) abort
2019-08-22 15:36:17 +00:00
return fugitive#Path(@%, a:0 ? a:1 : ':(top)', a:0 > 1 ? a:2 : s:Dir())
2018-08-25 16:13:42 +00:00
endfunction
2018-11-01 10:03:42 +00:00
function! fugitive#Find(object, ...) abort
2018-09-25 00:40:17 +00:00
if type(a:object) == type(0)
let name = bufname(a:object)
2019-08-22 15:36:17 +00:00
return FugitiveVimPath(name =~# '^$\|^/\|^\a\+:' ? name : getcwd() . '/' . name)
2018-09-25 00:40:17 +00:00
elseif a:object =~# '^[~$]'
2018-08-25 16:13:42 +00:00
let prefix = matchstr(a:object, '^[~$]\i*')
let owner = expand(prefix)
2019-08-22 15:36:17 +00:00
return FugitiveVimPath((len(owner) ? owner : prefix) . strpart(a:object, len(prefix)))
2018-08-25 16:13:42 +00:00
elseif s:Slash(a:object) =~# '^$\|^/\|^\%(\a\a\+:\).*\%(//\|::\)' . (has('win32') ? '\|^\a:/' : '')
2019-08-22 15:36:17 +00:00
return FugitiveVimPath(a:object)
2018-08-25 16:13:42 +00:00
elseif s:Slash(a:object) =~# '^\.\.\=\%(/\|$\)'
2019-08-22 15:36:17 +00:00
return FugitiveVimPath(simplify(getcwd() . '/' . a:object))
2018-08-25 16:13:42 +00:00
endif
2019-03-08 11:04:56 +00:00
let dir = a:0 ? a:1 : s:Dir()
2018-08-25 16:13:42 +00:00
if empty(dir)
let file = matchstr(a:object, '^\%(:\d:\|[^:]*:\)\zs.*', '', '')
let dir = FugitiveExtractGitDir(file)
if empty(dir)
2019-08-22 15:36:17 +00:00
return fnamemodify(FugitiveVimPath(len(file) ? file : a:object), ':p')
2018-08-25 16:13:42 +00:00
endif
endif
let rev = s:Slash(a:object)
let tree = s:Tree(dir)
let base = len(tree) ? tree : 'fugitive://' . dir . '//0'
if rev ==# '.git'
let f = len(tree) ? tree . '/.git' : dir
elseif rev =~# '^\.git/'
let f = substitute(rev, '^\.git', '', '')
let cdir = fugitive#CommonDir(dir)
if f =~# '^/\.\./\.\.\%(/\|$\)'
let f = simplify(len(tree) ? tree . f[3:-1] : dir . f)
elseif f =~# '^/\.\.\%(/\|$\)'
let f = base . f[3:-1]
elseif cdir !=# dir && (
\ f =~# '^/\%(config\|hooks\|info\|logs/refs\|objects\|refs\|worktrees\)\%(/\|$\)' ||
2019-08-22 15:36:17 +00:00
\ f !~# '^/\%(index$\|index\.lock$\|\w*MSG$\|\w*HEAD$\|logs/\w*HEAD$\|logs$\|rebase-\w\+\)\%(/\|$\)' &&
\ getftime(FugitiveVimPath(dir . f)) < 0 && getftime(FugitiveVimPath(cdir . f)) >= 0)
2018-08-25 16:13:42 +00:00
let f = simplify(cdir . f)
else
let f = simplify(dir . f)
endif
elseif rev ==# ':/'
let f = base
elseif rev =~# '^\.\%(/\|$\)'
let f = base . rev[1:-1]
elseif rev =~# '^::\%(/\|\a\+\:\)'
let f = rev[2:-1]
elseif rev =~# '^::\.\.\=\%(/\|$\)'
let f = simplify(getcwd() . '/' . rev[2:-1])
elseif rev =~# '^::'
let f = base . '/' . rev[2:-1]
elseif rev =~# '^:\%([0-3]:\)\=\.\.\=\%(/\|$\)\|^:[0-3]:\%(/\|\a\+:\)'
let f = rev =~# '^:\%([0-3]:\)\=\.' ? simplify(getcwd() . '/' . matchstr(rev, '\..*')) : rev[3:-1]
if s:cpath(base . '/', (f . '/')[0 : len(base)])
let f = 'fugitive://' . dir . '//' . +matchstr(rev, '^:\zs\d\ze:') . '/' . strpart(f, len(base) + 1)
else
let altdir = FugitiveExtractGitDir(f)
if len(altdir) && !s:cpath(dir, altdir)
2018-11-01 10:03:42 +00:00
return fugitive#Find(a:object, altdir)
2018-08-25 16:13:42 +00:00
endif
endif
elseif rev =~# '^:[0-3]:'
let f = 'fugitive://' . dir . '//' . rev[1] . '/' . rev[3:-1]
elseif rev ==# ':'
if $GIT_INDEX_FILE =~# '/[^/]*index[^/]*\.lock$' && s:cpath(fnamemodify($GIT_INDEX_FILE,':p')[0:strlen(dir)]) ==# s:cpath(dir . '/') && filereadable($GIT_INDEX_FILE)
let f = fnamemodify($GIT_INDEX_FILE, ':p')
else
2019-08-22 15:36:17 +00:00
let f = fugitive#Find('.git/index', dir)
2018-08-25 16:13:42 +00:00
endif
elseif rev =~# '^:(\%(top\|top,literal\|literal,top\|literal\))'
2019-08-22 15:36:17 +00:00
let f = matchstr(rev, ')\zs.*')
if f=~# '^\.\.\=\%(/\|$\)'
let f = simplify(getcwd() . '/' . f)
elseif f !~# '^/\|^\%(\a\a\+:\).*\%(//\|::\)' . (has('win32') ? '\|^\a:/' : '')
let f = base . '/' . f
endif
2018-08-25 16:13:42 +00:00
elseif rev =~# '^:/\@!'
let f = 'fugitive://' . dir . '//0/' . rev[1:-1]
else
if !exists('f')
2020-01-07 12:45:07 +00:00
let commit = substitute(matchstr(rev, '^\%([^:.-]\|\.\.[^/:]\)[^:]*\|^:.*'), '^@\%($\|[~^]\|@{\)\@=', 'HEAD', '')
let file = substitute(matchstr(rev, '^\%([^:.-]\|\.\.[^/:]\)[^:]*\zs:.*'), '^:', '/', '')
2018-08-25 16:13:42 +00:00
if file =~# '^/\.\.\=\%(/\|$\)\|^//\|^/\a\+:'
let file = file =~# '^/\.' ? simplify(getcwd() . file) : file[1:-1]
if s:cpath(base . '/', (file . '/')[0 : len(base)])
let file = '/' . strpart(file, len(base) + 1)
else
let altdir = FugitiveExtractGitDir(file)
if len(altdir) && !s:cpath(dir, altdir)
2018-11-01 10:03:42 +00:00
return fugitive#Find(a:object, altdir)
2018-08-25 16:13:42 +00:00
endif
return file
endif
endif
2019-08-22 15:36:17 +00:00
let commits = split(commit, '\.\.\.-\@!', 1)
if len(commits) == 2
call map(commits, 'empty(v:val) || v:val ==# "@" ? "HEAD" : v:val')
let commit = matchstr(s:ChompDefault('', [dir, 'merge-base'] + commits + ['--']), '\<[0-9a-f]\{40,\}\>')
endif
if commit !~# '^[0-9a-f]\{40,\}$'
2019-12-30 13:28:38 +00:00
let commit = matchstr(s:ChompDefault('', [dir, 'rev-parse', '--verify', commit . (len(file) ? '^{}' : ''), '--']), '\<[0-9a-f]\{40,\}\>')
2018-08-25 16:13:42 +00:00
endif
if len(commit)
let f = 'fugitive://' . dir . '//' . commit . file
else
let f = base . '/' . substitute(rev, '^:/:\=\|^[^:]\+:', '', '')
endif
endif
endif
2019-08-22 15:36:17 +00:00
return FugitiveVimPath(f)
2018-08-25 16:13:42 +00:00
endfunction
function! s:Generate(rev, ...) abort
2019-08-22 15:36:17 +00:00
return fugitive#Find(a:rev, a:0 ? a:1 : s:Dir())
2018-08-25 16:13:42 +00:00
endfunction
2019-08-22 15:36:17 +00:00
function! s:DotRelative(path, ...) abort
let cwd = a:0 ? a:1 : getcwd()
2018-09-25 00:40:17 +00:00
let path = substitute(a:path, '^[~$]\i*', '\=expand(submatch(0))', '')
2019-08-22 15:36:17 +00:00
if len(cwd) && s:cpath(cwd . '/', (path . '/')[0 : len(cwd)])
2018-09-25 00:40:17 +00:00
return '.' . strpart(path, len(cwd))
2018-08-25 16:13:42 +00:00
endif
2018-09-25 00:40:17 +00:00
return a:path
2018-08-25 16:13:42 +00:00
endfunction
function! fugitive#Object(...) abort
2019-03-08 11:04:56 +00:00
let dir = a:0 > 1 ? a:2 : s:Dir()
2018-08-25 16:13:42 +00:00
let [fdir, rev] = s:DirRev(a:0 ? a:1 : @%)
if s:cpath(dir) !=# s:cpath(fdir)
let rev = ''
endif
let tree = s:Tree(dir)
2019-11-16 15:28:42 +00:00
let full = a:0 ? a:1 : @%
let full = fnamemodify(full, ':p' . (s:Slash(full) =~# '/$' ? '' : ':s?/$??'))
2018-08-25 16:13:42 +00:00
if empty(rev) && empty(tree)
2019-11-16 15:28:42 +00:00
return FugitiveGitPath(full)
2018-08-25 16:13:42 +00:00
elseif empty(rev)
2019-11-16 15:28:42 +00:00
let rev = fugitive#Path(full, './', dir)
2019-08-22 15:36:17 +00:00
if rev =~# '^\./.git\%(/\|$\)'
2019-11-16 15:28:42 +00:00
return FugitiveGitPath(full)
2018-08-25 16:13:42 +00:00
endif
endif
if rev !~# '^\.\%(/\|$\)' || s:cpath(getcwd(), tree)
return rev
else
2019-11-16 15:28:42 +00:00
return FugitiveGitPath(tree . rev[1:-1])
2018-08-25 16:13:42 +00:00
endif
endfunction
2018-09-25 00:40:17 +00:00
let s:var = '\%(%\|#<\=\d\+\|##\=\)'
let s:flag = '\%(:[p8~.htre]\|:g\=s\(.\).\{-\}\1.\{-\}\1\)'
let s:expand = '\%(\(' . s:var . '\)\(' . s:flag . '*\)\(:S\)\=\)'
function! s:BufName(var) abort
if a:var ==# '%'
2019-08-22 15:36:17 +00:00
return bufname(get(s:TempState(), 'bufnr', ''))
2018-09-25 00:40:17 +00:00
elseif a:var =~# '^#\d*$'
2019-08-22 15:36:17 +00:00
let nr = get(s:TempState(bufname(+a:var[1:-1])), 'bufnr', '')
2018-09-25 00:40:17 +00:00
return bufname(nr ? nr : +a:var[1:-1])
else
return expand(a:var)
endif
endfunction
2019-08-22 15:36:17 +00:00
function! s:ExpandVarLegacy(str) abort
if get(g:, 'fugitive_legacy_quoting', 1)
return substitute(a:str, '\\\ze[%#!]', '', 'g')
else
return a:str
endif
endfunction
function! s:ExpandVar(other, var, flags, esc, ...) abort
let cwd = a:0 ? a:1 : getcwd()
2018-09-25 00:40:17 +00:00
if a:other =~# '^\'
return a:other[1:-1]
2019-08-22 15:36:17 +00:00
elseif a:other =~# '^'''
return s:ExpandVarLegacy(substitute(a:other[1:-2], "''", "'", "g"))
elseif a:other =~# '^"'
return s:ExpandVarLegacy(substitute(a:other[1:-2], '""', '"', "g"))
2018-09-25 00:40:17 +00:00
elseif a:other =~# '^!'
let buffer = s:BufName(len(a:other) > 1 ? '#'. a:other[1:-1] : '%')
let owner = s:Owner(buffer)
return len(owner) ? owner : '@'
endif
let flags = a:flags
2019-08-22 15:36:17 +00:00
let file = s:DotRelative(fugitive#Real(s:BufName(a:var)), cwd)
2018-09-25 00:40:17 +00:00
while len(flags)
let flag = matchstr(flags, s:flag)
let flags = strpart(flags, len(flag))
if flag ==# ':.'
2019-08-22 15:36:17 +00:00
let file = s:DotRelative(file, cwd)
2018-09-25 00:40:17 +00:00
else
let file = fnamemodify(file, flag)
endif
endwhile
let file = s:Slash(file)
return (len(a:esc) ? shellescape(file) : file)
endfunction
2019-08-22 15:36:17 +00:00
function! s:Expand(rev, ...) abort
2018-08-25 16:13:42 +00:00
if a:rev =~# '^:[0-3]$'
2019-11-16 15:28:42 +00:00
let file = len(expand('%')) ? a:rev . ':%' : '%'
elseif a:rev ==# '>'
let file = '%'
elseif a:rev =~# '^>[~^]'
let file = len(expand('%')) ? '!' . a:rev[1:-1] . ':%' : '%'
2019-08-22 15:36:17 +00:00
elseif a:rev =~# '^>[> ]\@!'
2019-11-16 15:28:42 +00:00
let file = len(expand('%')) ? a:rev[1:-1] . ':%' : '%'
2018-08-25 16:13:42 +00:00
else
let file = a:rev
endif
2018-09-25 00:40:17 +00:00
return substitute(file,
\ '\(\\[' . s:fnameescape . ']\|^\\[>+-]\|!\d*\)\|' . s:expand,
2019-08-22 15:36:17 +00:00
\ '\=s:ExpandVar(submatch(1),submatch(2),submatch(3),"", a:0 ? a:1 : getcwd())', 'g')
2018-09-25 00:40:17 +00:00
endfunction
function! fugitive#Expand(object) abort
return substitute(a:object,
\ '\(\\[' . s:fnameescape . ']\|^\\[>+-]\|!\d*\)\|' . s:expand,
\ '\=s:ExpandVar(submatch(1),submatch(2),submatch(3),submatch(5))', 'g')
2018-08-25 16:13:42 +00:00
endfunction
2019-08-22 15:36:17 +00:00
function! s:ExpandSplit(string, ...) abort
let list = []
let string = a:string
let handle_bar = a:0 && a:1
let dquote = handle_bar ? '"\%([^"]\|""\|\\"\)*"\|' : ''
let cwd = a:0 > 1 ? a:2 : getcwd()
while string =~# '\S'
if handle_bar && string =~# '^\s*|'
return [list, substitute(string, '^\s*', '', '')]
endif
let arg = matchstr(string, '^\s*\%(' . dquote . '''[^'']*''\|\\.\|[^[:space:] ' . (handle_bar ? '|' : '') . ']\)\+')
let string = strpart(string, len(arg))
let arg = substitute(arg, '^\s\+', '', '')
if !exists('seen_separator')
let arg = substitute(arg, '^\%([^:.][^:]*:\|^:\|^:[0-3]:\)\=\zs\.\.\=\%(/.*\)\=$',
\ '\=s:DotRelative(s:Slash(simplify(getcwd() . "/" . submatch(0))), cwd)', '')
endif
let arg = substitute(arg,
\ '\(' . dquote . '''\%(''''\|[^'']\)*''\|\\[' . s:fnameescape . ']\|^\\[>+-]\|!\d*\)\|' . s:expand,
\ '\=s:ExpandVar(submatch(1),submatch(2),submatch(3),submatch(5), cwd)', 'g')
call add(list, arg)
if arg ==# '--'
let seen_separator = 1
endif
endwhile
return handle_bar ? [list, ''] : list
endfunction
function! s:SplitExpand(string, ...) abort
return s:ExpandSplit(a:string, 0, a:0 ? a:1 : getcwd())
endfunction
function! s:SplitExpandChain(string, ...) abort
return s:ExpandSplit(a:string, 1, a:0 ? a:1 : getcwd())
2018-08-25 16:13:42 +00:00
endfunction
2018-07-04 10:53:25 +00:00
let s:trees = {}
let s:indexes = {}
function! s:TreeInfo(dir, commit) abort
if a:commit =~# '^:\=[0-3]$'
let index = get(s:indexes, a:dir, [])
2019-08-22 15:36:17 +00:00
let newftime = getftime(fugitive#Find('.git/index', a:dir))
2018-07-04 10:53:25 +00:00
if get(index, 0, -1) < newftime
2019-08-22 15:36:17 +00:00
let [lines, exec_error] = s:LinesError([a:dir, 'ls-files', '--stage', '--'])
2018-07-04 10:53:25 +00:00
let s:indexes[a:dir] = [newftime, {'0': {}, '1': {}, '2': {}, '3': {}}]
2019-08-22 15:36:17 +00:00
if exec_error
2018-07-04 10:53:25 +00:00
return [{}, -1]
endif
2019-08-22 15:36:17 +00:00
for line in lines
2018-07-04 10:53:25 +00:00
let [info, filename] = split(line, "\t")
let [mode, sha, stage] = split(info, '\s\+')
let s:indexes[a:dir][1][stage][filename] = [newftime, mode, 'blob', sha, -2]
while filename =~# '/'
let filename = substitute(filename, '/[^/]*$', '', '')
let s:indexes[a:dir][1][stage][filename] = [newftime, '040000', 'tree', '', 0]
endwhile
endfor
endif
return [get(s:indexes[a:dir][1], a:commit[-1:-1], {}), newftime]
2019-08-22 15:36:17 +00:00
elseif a:commit =~# '^\x\{40,\}$'
2018-07-04 10:53:25 +00:00
if !has_key(s:trees, a:dir)
2018-08-25 16:13:42 +00:00
let s:trees[a:dir] = {}
endif
if !has_key(s:trees[a:dir], a:commit)
2019-08-22 15:36:17 +00:00
let [ftime, exec_error] = s:ChompError([a:dir, 'log', '-1', '--pretty=format:%ct', a:commit, '--'])
if exec_error
2018-08-25 16:13:42 +00:00
let s:trees[a:dir][a:commit] = [{}, -1]
return s:trees[a:dir][a:commit]
2018-07-04 10:53:25 +00:00
endif
2018-08-25 16:13:42 +00:00
let s:trees[a:dir][a:commit] = [{}, +ftime]
2019-08-22 15:36:17 +00:00
let [lines, exec_error] = s:LinesError([a:dir, 'ls-tree', '-rtl', '--full-name', a:commit, '--'])
if exec_error
2018-08-25 16:13:42 +00:00
return s:trees[a:dir][a:commit]
2018-07-04 10:53:25 +00:00
endif
2019-08-22 15:36:17 +00:00
for line in lines
2018-07-04 10:53:25 +00:00
let [info, filename] = split(line, "\t")
let [mode, type, sha, size] = split(info, '\s\+')
2019-08-22 15:36:17 +00:00
let s:trees[a:dir][a:commit][0][filename] = [+ftime, mode, type, sha, +size, filename]
2018-07-04 10:53:25 +00:00
endfor
endif
2018-08-25 16:13:42 +00:00
return s:trees[a:dir][a:commit]
2018-07-04 10:53:25 +00:00
endif
return [{}, -1]
endfunction
function! s:PathInfo(url) abort
let [dir, commit, file] = s:DirCommitFile(a:url)
if empty(dir) || !get(g:, 'fugitive_file_api', 1)
return [-1, '000000', '', '', -1]
endif
let path = substitute(file[1:-1], '/*$', '', '')
let [tree, ftime] = s:TreeInfo(dir, commit)
let entry = empty(path) ? [ftime, '040000', 'tree', '', -1] : get(tree, path, [])
2019-08-22 15:36:17 +00:00
if empty(entry) || file =~# '/$' && entry[2] !=# 'tree'
2018-07-04 10:53:25 +00:00
return [-1, '000000', '', '', -1]
else
return entry
endif
endfunction
function! fugitive#simplify(url) abort
let [dir, commit, file] = s:DirCommitFile(a:url)
if empty(dir)
return ''
endif
if file =~# '/\.\.\%(/\|$\)'
2018-07-30 21:18:16 +00:00
let tree = s:Tree(dir)
2018-07-04 10:53:25 +00:00
if len(tree)
let path = simplify(tree . file)
if strpart(path . '/', 0, len(tree) + 1) !=# tree . '/'
2019-08-22 15:36:17 +00:00
return FugitiveVimPath(path)
2018-07-04 10:53:25 +00:00
endif
endif
endif
2019-08-22 15:36:17 +00:00
return FugitiveVimPath('fugitive://' . simplify(dir) . '//' . commit . simplify(file))
2018-07-04 10:53:25 +00:00
endfunction
function! fugitive#resolve(url) abort
let url = fugitive#simplify(a:url)
if url =~? '^fugitive:'
return url
else
return resolve(url)
endif
endfunction
function! fugitive#getftime(url) abort
return s:PathInfo(a:url)[0]
endfunction
function! fugitive#getfsize(url) abort
let entry = s:PathInfo(a:url)
if entry[4] == -2 && entry[2] ==# 'blob' && len(entry[3])
let dir = s:DirCommitFile(a:url)[0]
2019-08-22 15:36:17 +00:00
let entry[4] = +s:ChompDefault(-1, [dir, 'cat-file', '-s', entry[3]])
2018-07-04 10:53:25 +00:00
endif
return entry[4]
endfunction
function! fugitive#getftype(url) abort
return get({'tree': 'dir', 'blob': 'file'}, s:PathInfo(a:url)[2], '')
endfunction
function! fugitive#filereadable(url) abort
return s:PathInfo(a:url)[2] ==# 'blob'
endfunction
2018-07-30 21:18:16 +00:00
function! fugitive#filewritable(url) abort
let [dir, commit, file] = s:DirCommitFile(a:url)
2019-08-22 15:36:17 +00:00
if commit !~# '^\d$' || !filewritable(fugitive#Find('.git/index', dir))
2018-07-30 21:18:16 +00:00
return 0
endif
return s:PathInfo(a:url)[2] ==# 'blob' ? 1 : 2
endfunction
2018-07-04 10:53:25 +00:00
function! fugitive#isdirectory(url) abort
return s:PathInfo(a:url)[2] ==# 'tree'
endfunction
2018-07-30 21:18:16 +00:00
function! fugitive#getfperm(url) abort
let [dir, commit, file] = s:DirCommitFile(a:url)
let perm = getfperm(dir)
let fperm = s:PathInfo(a:url)[1]
if fperm ==# '040000'
let fperm = '000755'
endif
if fperm !~# '[15]'
let perm = tr(perm, 'x', '-')
endif
if fperm !~# '[45]$'
let perm = tr(perm, 'rw', '--')
endif
if commit !~# '^\d$'
let perm = tr(perm, 'w', '-')
endif
return perm ==# '---------' ? '' : perm
endfunction
function! fugitive#setfperm(url, perm) abort
let [dir, commit, file] = s:DirCommitFile(a:url)
let entry = s:PathInfo(a:url)
let perm = fugitive#getfperm(a:url)
if commit !~# '^\d$' || entry[2] !=# 'blob' ||
\ substitute(perm, 'x', '-', 'g') !=# substitute(a:perm, 'x', '-', 'g')
return -2
endif
2019-08-22 15:36:17 +00:00
let exec_error = s:SystemError([dir, 'update-index', '--index-info'],
\ (a:perm =~# 'x' ? '000755 ' : '000644 ') . entry[3] . ' ' . commit . "\t" . file[1:-1])[1]
return exec_error ? -1 : 0
2018-07-30 21:18:16 +00:00
endfunction
function! s:TempCmd(out, cmd) abort
try
2019-08-22 15:36:17 +00:00
let cmd = (type(a:cmd) == type([]) ? fugitive#Prepare(a:cmd) : a:cmd)
2018-07-30 21:18:16 +00:00
let redir = ' > ' . a:out
2020-01-29 02:07:36 +00:00
if (s:winshell() || &shellcmdflag ==# '-Command') && !has('nvim')
2018-07-30 21:18:16 +00:00
let cmd_escape_char = &shellxquote == '(' ? '^' : '^^^'
2020-01-29 02:07:36 +00:00
return s:SystemError('cmd /c "' . s:gsub(cmd, '[<>%]', cmd_escape_char . '&') . redir . '"')
2018-07-30 21:18:16 +00:00
elseif &shell =~# 'fish'
2020-01-29 02:07:36 +00:00
return s:SystemError(' begin;' . cmd . redir . ';end ')
2018-07-30 21:18:16 +00:00
else
2020-01-29 02:07:36 +00:00
return s:SystemError(' (' . cmd . redir . ') ')
2018-07-30 21:18:16 +00:00
endif
endtry
endfunction
if !exists('s:blobdirs')
let s:blobdirs = {}
endif
function! s:BlobTemp(url) abort
let [dir, commit, file] = s:DirCommitFile(a:url)
if empty(file)
return ''
endif
if !has_key(s:blobdirs, dir)
2018-08-25 16:13:42 +00:00
let s:blobdirs[dir] = tempname()
2018-07-30 21:18:16 +00:00
endif
2018-08-25 16:13:42 +00:00
let tempfile = s:blobdirs[dir] . '/' . commit . file
2018-07-30 21:18:16 +00:00
let tempparent = fnamemodify(tempfile, ':h')
if !isdirectory(tempparent)
call mkdir(tempparent, 'p')
endif
if commit =~# '^\d$' || !filereadable(tempfile)
let rev = s:DirRev(a:url)[1]
2019-08-22 15:36:17 +00:00
let exec_error = s:TempCmd(tempfile, [dir, 'cat-file', 'blob', rev])[1]
if exec_error
2018-07-30 21:18:16 +00:00
call delete(tempfile)
return ''
endif
endif
2018-08-25 16:13:42 +00:00
return s:Resolve(tempfile)
2018-07-30 21:18:16 +00:00
endfunction
2018-07-04 10:53:25 +00:00
function! fugitive#readfile(url, ...) abort
let entry = s:PathInfo(a:url)
if entry[2] !=# 'blob'
return []
endif
2018-07-30 21:18:16 +00:00
let temp = s:BlobTemp(a:url)
if empty(temp)
return []
2018-07-04 10:53:25 +00:00
endif
2018-07-30 21:18:16 +00:00
return call('readfile', [temp] + a:000)
endfunction
function! fugitive#writefile(lines, url, ...) abort
let url = type(a:url) ==# type('') ? a:url : ''
let [dir, commit, file] = s:DirCommitFile(url)
let entry = s:PathInfo(url)
if commit =~# '^\d$' && entry[2] !=# 'tree'
2018-08-25 16:13:42 +00:00
let temp = tempname()
2018-07-30 21:18:16 +00:00
if a:0 && a:1 =~# 'a' && entry[2] ==# 'blob'
call writefile(fugitive#readfile(url, 'b'), temp, 'b')
endif
call call('writefile', [a:lines, temp] + a:000)
2019-08-22 15:36:17 +00:00
let [hash, exec_error] = s:ChompError([dir, 'hash-object', '-w', temp])
2018-07-30 21:18:16 +00:00
let mode = len(entry[1]) ? entry[1] : '100644'
2019-08-22 15:36:17 +00:00
if !exec_error && hash =~# '^\x\{40,\}$'
let exec_error = s:SystemError([dir, 'update-index', '--index-info'],
\ mode . ' ' . hash . ' ' . commit . "\t" . file[1:-1])[1]
if !exec_error
2018-07-30 21:18:16 +00:00
return 0
endif
2018-07-04 10:53:25 +00:00
endif
endif
2018-07-30 21:18:16 +00:00
return call('writefile', [a:lines, a:url] + a:000)
2018-07-04 10:53:25 +00:00
endfunction
2018-08-25 16:13:42 +00:00
let s:globsubs = {
\ '/**/': '/\%([^./][^/]*/\)*',
\ '/**': '/\%([^./][^/]\+/\)*[^./][^/]*',
\ '**/': '[^/]*\%(/[^./][^/]*\)*',
\ '**': '.*',
\ '/*': '/[^/.][^/]*',
\ '*': '[^/]*',
\ '?': '[^/]'}
2018-07-04 10:53:25 +00:00
function! fugitive#glob(url, ...) abort
let [dirglob, commit, glob] = s:DirCommitFile(a:url)
let append = matchstr(glob, '/*$')
let glob = substitute(glob, '/*$', '', '')
2019-05-17 14:09:13 +00:00
let pattern = '^' . substitute(glob, '/\=\*\*/\=\|/\=\*\|[.?\$]\|^^', '\=get(s:globsubs, submatch(0), "\\" . submatch(0))', 'g')[1:-1] . '$'
2018-07-04 10:53:25 +00:00
let results = []
for dir in dirglob =~# '[*?]' ? split(glob(dirglob), "\n") : [dirglob]
2019-08-22 15:36:17 +00:00
if empty(dir) || !get(g:, 'fugitive_file_api', 1) || !filereadable(fugitive#Find('.git/HEAD', dir))
2018-07-04 10:53:25 +00:00
continue
endif
let files = items(s:TreeInfo(dir, commit)[0])
if len(append)
call filter(files, 'v:val[1][2] ==# "tree"')
endif
call map(files, 'v:val[0]')
call filter(files, 'v:val =~# pattern')
let prepend = 'fugitive://' . dir . '//' . substitute(commit, '^:', '', '') . '/'
call sort(files)
2019-08-22 15:36:17 +00:00
call map(files, 'FugitiveVimPath(prepend . v:val . append)')
2018-07-04 10:53:25 +00:00
call extend(results, files)
endfor
if a:0 > 1 && a:2
return results
else
return join(results, "\n")
endif
endfunction
2018-07-30 21:18:16 +00:00
function! fugitive#delete(url, ...) abort
let [dir, commit, file] = s:DirCommitFile(a:url)
if a:0 && len(a:1) || commit !~# '^\d$'
return -1
endif
let entry = s:PathInfo(a:url)
if entry[2] !=# 'blob'
return -1
endif
2019-08-22 15:36:17 +00:00
let exec_error = s:SystemError([dir, 'update-index', '--index-info'],
\ '000000 0000000000000000000000000000000000000000 ' . commit . "\t" . file[1:-1])[1]
return exec_error ? -1 : 0
2018-07-30 21:18:16 +00:00
endfunction
2018-08-25 16:13:42 +00:00
" Section: Buffer Object
2018-06-14 10:31:12 +00:00
let s:buffer_prototype = {}
2018-07-30 21:18:16 +00:00
function! fugitive#buffer(...) abort
2018-06-14 10:31:12 +00:00
let buffer = {'#': bufnr(a:0 ? a:1 : '%')}
2018-08-25 16:13:42 +00:00
call extend(buffer, s:buffer_prototype, 'keep')
2019-01-08 10:11:54 +00:00
return buffer
2018-06-14 10:31:12 +00:00
endfunction
function! s:buffer_repo() dict abort
2019-08-22 15:36:17 +00:00
return fugitive#repo(self['#'])
2018-06-14 10:31:12 +00:00
endfunction
function! s:buffer_type(...) dict abort
2019-11-16 15:28:42 +00:00
return 'see b:fugitive_type'
2018-06-14 10:31:12 +00:00
endfunction
2019-11-16 15:28:42 +00:00
call s:add_methods('buffer', ['repo', 'type'])
2018-06-14 10:31:12 +00:00
2018-08-25 16:13:42 +00:00
" Section: Completion
2019-08-22 15:36:17 +00:00
function! s:FilterEscape(items, ...) abort
let items = copy(a:items)
2019-11-30 12:06:56 +00:00
call map(items, 's:fnameescape(v:val)')
2019-08-22 15:36:17 +00:00
if a:0 && type(a:1) == type('')
call filter(items, 'strpart(v:val, 0, strlen(a:1)) ==# a:1')
endif
2019-11-30 12:06:56 +00:00
return items
2019-08-22 15:36:17 +00:00
endfunction
2018-08-25 16:13:42 +00:00
function! s:GlobComplete(lead, pattern) abort
2019-08-22 15:36:17 +00:00
if a:lead ==# '/'
return []
elseif v:version >= 704
2018-08-25 16:13:42 +00:00
let results = glob(a:lead . a:pattern, 0, 1)
2018-06-14 10:31:12 +00:00
else
2018-08-25 16:13:42 +00:00
let results = split(glob(a:lead . a:pattern), "\n")
2018-06-14 10:31:12 +00:00
endif
2018-08-25 16:13:42 +00:00
call map(results, 'v:val !~# "/$" && isdirectory(v:val) ? v:val."/" : v:val')
call map(results, 'v:val[ strlen(a:lead) : -1 ]')
return results
2018-06-14 10:31:12 +00:00
endfunction
2019-05-17 14:09:13 +00:00
function! fugitive#CompletePath(base, ...) abort
let dir = a:0 == 1 ? a:1 : a:0 == 3 ? a:3 : s:Dir()
2018-08-25 16:13:42 +00:00
let tree = s:Tree(dir) . '/'
let strip = '^\%(:/:\=\|:(top)\|:(top,literal)\|:(literal,top)\|:(literal)\)'
2018-07-30 21:18:16 +00:00
let base = substitute(a:base, strip, '', '')
2018-08-25 16:13:42 +00:00
if base =~# '^\.git/'
let pattern = s:gsub(base[5:-1], '/', '*&').'*'
let matches = s:GlobComplete(dir . '/', pattern)
let cdir = fugitive#CommonDir(dir)
if len(cdir) && s:cpath(dir) !=# s:cpath(cdir)
call extend(matches, s:GlobComplete(cdir . '/', pattern))
endif
call s:Uniq(matches)
call map(matches, "'.git/' . v:val")
elseif base =~# '^\~/'
let matches = map(s:GlobComplete(expand('~/'), base[2:-1] . '*'), '"~/" . v:val')
elseif a:base =~# '^/\|^\a\+:\|^\.\.\=/\|^:(literal)'
let matches = s:GlobComplete('', base . '*')
elseif len(tree) > 1
let matches = s:GlobComplete(tree, s:gsub(base, '/', '*&').'*')
else
let matches = []
endif
call map(matches, 's:fnameescape(s:Slash(matchstr(a:base, strip) . v:val))')
2018-07-30 21:18:16 +00:00
return matches
2018-06-14 10:31:12 +00:00
endfunction
2019-05-17 14:09:13 +00:00
function! fugitive#PathComplete(...) abort
return call('fugitive#CompletePath', a:000)
endfunction
2019-11-16 15:28:42 +00:00
function! s:CompleteHeads(dir) abort
let dir = fugitive#Find('.git/', a:dir)
return sort(filter(['HEAD', 'FETCH_HEAD', 'ORIG_HEAD'] + s:merge_heads, 'filereadable(dir . v:val)')) +
\ sort(s:LinesError('rev-parse', '--symbolic', '--branches', '--tags', '--remotes')[0])
endfunction
2019-05-17 14:09:13 +00:00
function! fugitive#CompleteObject(base, ...) abort
let dir = a:0 == 1 ? a:1 : a:0 == 3 ? a:3 : s:Dir()
2019-08-22 15:36:17 +00:00
let cwd = getcwd()
2018-08-25 16:13:42 +00:00
let tree = s:Tree(dir) . '/'
let subdir = ''
if len(tree) > 1 && s:cpath(tree, cwd[0 : len(tree) - 1])
let subdir = strpart(cwd, len(tree)) . '/'
endif
if a:base =~# '^\.\=/\|^:(' || a:base !~# ':'
2018-07-30 21:18:16 +00:00
let results = []
2018-08-25 16:13:42 +00:00
if a:base =~# '^refs/'
let results += map(s:GlobComplete(fugitive#CommonDir(dir) . '/', a:base . '*'), 's:Slash(v:val)')
2019-11-30 12:06:56 +00:00
call map(results, 's:fnameescape(v:val)')
2018-08-25 16:13:42 +00:00
elseif a:base !~# '^\.\=/\|^:('
2019-11-16 15:28:42 +00:00
let heads = s:CompleteHeads(dir)
2019-08-22 15:36:17 +00:00
if filereadable(fugitive#Find('.git/refs/stash', dir))
2018-07-30 21:18:16 +00:00
let heads += ["stash"]
2019-08-22 15:36:17 +00:00
let heads += sort(s:LinesError(["stash","list","--pretty=format:%gd"], dir)[0])
2018-07-30 21:18:16 +00:00
endif
2019-11-30 12:06:56 +00:00
let results += s:FilterEscape(heads, a:base)
2018-06-14 10:31:12 +00:00
endif
2018-08-25 16:13:42 +00:00
if !empty(tree)
2019-05-17 14:09:13 +00:00
let results += a:0 == 1 ? fugitive#CompletePath(a:base, dir) : fugitive#CompletePath(a:base)
2018-07-30 21:18:16 +00:00
endif
return results
2018-06-14 10:31:12 +00:00
2018-07-30 21:18:16 +00:00
elseif a:base =~# '^:'
2019-08-22 15:36:17 +00:00
let entries = s:LinesError(['ls-files','--stage'], dir)[0]
2018-08-25 16:13:42 +00:00
if a:base =~# ':\./'
call map(entries, 'substitute(v:val, "\\M\t\\zs" . subdir, "./", "")')
endif
2018-07-30 21:18:16 +00:00
call map(entries,'s:sub(v:val,".*(\\d)\\t(.*)",":\\1:\\2")')
if a:base !~# '^:[0-3]\%(:\|$\)'
call filter(entries,'v:val[1] == "0"')
call map(entries,'v:val[2:-1]')
endif
2018-06-14 10:31:12 +00:00
2018-07-30 21:18:16 +00:00
else
2018-08-25 16:13:42 +00:00
let tree = matchstr(a:base, '.*[:/]')
2019-08-22 15:36:17 +00:00
let entries = s:LinesError(['ls-tree', substitute(tree, ':\zs\./', '\=subdir', '')], dir)[0]
2018-07-30 21:18:16 +00:00
call map(entries,'s:sub(v:val,"^04.*\\zs$","/")')
call map(entries,'tree.s:sub(v:val,".*\t","")')
2018-08-25 16:13:42 +00:00
2018-07-30 21:18:16 +00:00
endif
2019-08-22 15:36:17 +00:00
return s:FilterEscape(entries, a:base)
2018-07-30 21:18:16 +00:00
endfunction
2018-06-14 10:31:12 +00:00
2019-08-22 15:36:17 +00:00
function! s:CompleteSub(subcommand, A, L, P, ...) abort
let pre = strpart(a:L, 0, a:P)
if pre =~# ' -- '
return fugitive#CompletePath(a:A)
elseif a:A =~# '^-' || a:A is# 0
return s:FilterEscape(split(s:ChompDefault('', a:subcommand, '--git-completion-helper'), ' '), a:A)
elseif !a:0
return fugitive#CompleteObject(a:A, s:Dir())
elseif type(a:1) == type(function('tr'))
return call(a:1, [a:A, a:L, a:P])
else
return s:FilterEscape(a:1, a:A)
endif
endfunction
2019-11-16 15:28:42 +00:00
function! s:CompleteRevision(A, L, P, ...) abort
return s:FilterEscape(s:CompleteHeads(s:Dir()), a:A)
2019-08-22 15:36:17 +00:00
endfunction
function! s:CompleteRemote(A, L, P) abort
let remote = matchstr(a:L, '\u\w*[! ] *\zs\S\+\ze ')
if !empty(remote)
let matches = s:LinesError('ls-remote', remote)[0]
call filter(matches, 'v:val =~# "\t" && v:val !~# "{"')
call map(matches, 's:sub(v:val, "^.*\t%(refs/%(heads/|tags/)=)=", "")')
else
let matches = s:LinesError('remote')[0]
endif
return s:FilterEscape(matches, a:A)
2019-05-17 14:09:13 +00:00
endfunction
2018-08-25 16:13:42 +00:00
" Section: Buffer auto-commands
2018-06-14 10:31:12 +00:00
2019-05-17 14:09:13 +00:00
function! s:ReplaceCmd(cmd) abort
2018-08-25 16:13:42 +00:00
let temp = tempname()
2019-08-22 15:36:17 +00:00
let [err, exec_error] = s:TempCmd(temp, a:cmd)
if exec_error
2018-08-25 16:13:42 +00:00
call s:throw((len(err) ? err : filereadable(temp) ? join(readfile(temp), ' ') : 'unknown error running ' . a:cmd))
2018-07-30 21:18:16 +00:00
endif
2020-01-29 02:07:36 +00:00
silent exe 'lockmarks keepalt 0read ++edit' s:fnameescape(temp)
silent keepjumps $delete _
call delete(temp)
if s:cpath(fnamemodify(bufname('$'), ':p'), temp)
silent! execute bufnr('$') . 'bwipeout'
endif
2018-07-30 21:18:16 +00:00
endfunction
2019-01-08 10:11:54 +00:00
function! s:QueryLog(refspec) abort
2020-01-29 02:07:36 +00:00
let lines = s:LinesError(['log', '-n', '256', '--pretty=format:%h%x09%s', a:refspec, '--'])[0]
2019-01-08 10:11:54 +00:00
call map(lines, 'split(v:val, "\t")')
call map(lines, '{"type": "Log", "commit": v:val[0], "subject": v:val[-1]}')
return lines
endfunction
function! s:FormatLog(dict) abort
return a:dict.commit . ' ' . a:dict.subject
endfunction
function! s:FormatRebase(dict) abort
return a:dict.status . ' ' . a:dict.commit . ' ' . a:dict.subject
endfunction
function! s:FormatFile(dict) abort
return a:dict.status . ' ' . a:dict.filename
endfunction
function! s:Format(val) abort
if type(a:val) == type({})
return s:Format{a:val.type}(a:val)
elseif type(a:val) == type([])
return map(copy(a:val), 's:Format(v:val)')
else
return '' . a:val
endif
endfunction
function! s:AddHeader(key, value) abort
if empty(a:value)
return
endif
let before = 1
while !empty(getline(before))
let before += 1
endwhile
call append(before - 1, [a:key . ':' . (len(a:value) ? ' ' . a:value : '')])
if before == 1 && line('$') == 2
2019-11-16 15:28:42 +00:00
silent keepjumps 2delete _
2019-01-08 10:11:54 +00:00
endif
endfunction
function! s:AddSection(label, lines, ...) abort
let note = a:0 ? a:1 : ''
if empty(a:lines) && empty(note)
return
endif
call append(line('$'), ['', a:label . (len(note) ? ': ' . note : ' (' . len(a:lines) . ')')] + s:Format(a:lines))
endfunction
2020-01-29 02:07:36 +00:00
let s:rebase_abbrevs = {
\ 'p': 'pick',
\ 'r': 'reword',
\ 'e': 'edit',
\ 's': 'squash',
\ 'f': 'fixup',
\ 'x': 'exec',
\ 'd': 'drop',
\ 'l': 'label',
\ 't': 'reset',
\ 'm': 'merge',
\ 'b': 'break',
\ }
2018-07-30 21:18:16 +00:00
function! fugitive#BufReadStatus() abort
2018-08-25 16:13:42 +00:00
let amatch = s:Slash(expand('%:p'))
2018-07-30 21:18:16 +00:00
let b:fugitive_type = 'index'
2019-08-22 15:36:17 +00:00
unlet! b:fugitive_reltime
2018-07-30 21:18:16 +00:00
try
2019-01-08 10:11:54 +00:00
silent doautocmd BufReadPre
2020-01-29 02:07:36 +00:00
let config = fugitive#Config()
2018-09-25 00:40:17 +00:00
let cmd = [fnamemodify(amatch, ':h')]
2019-01-08 10:11:54 +00:00
setlocal noro ma nomodeline buftype=nowrite
2019-08-22 15:36:17 +00:00
if s:cpath(fnamemodify($GIT_INDEX_FILE !=# '' ? $GIT_INDEX_FILE : fugitive#Find('.git/index'), ':p')) !=# s:cpath(amatch)
2018-09-25 00:40:17 +00:00
let cmd += ['-c', 'GIT_INDEX_FILE=' . amatch]
2018-07-30 21:18:16 +00:00
endif
2019-01-08 10:11:54 +00:00
2019-11-16 15:28:42 +00:00
let b:fugitive_files = {'Staged': {}, 'Unstaged': {}}
2019-03-08 11:04:56 +00:00
let [staged, unstaged, untracked] = [[], [], []]
2019-11-16 15:28:42 +00:00
let props = {}
if fugitive#GitVersion(2, 11)
let cmd += ['status', '--porcelain=v2', '-bz']
let [output, message, exec_error] = s:NullError(cmd)
if exec_error
throw 'fugitive: ' . message
2019-03-08 11:04:56 +00:00
endif
2019-11-16 15:28:42 +00:00
let i = 0
while i < len(output)
let line = output[i]
let prop = matchlist(line, '# \(\S\+\) \(.*\)')
if len(prop)
let props[prop[1]] = prop[2]
elseif line[0] ==# '?'
call add(untracked, {'type': 'File', 'status': line[0], 'filename': line[2:-1]})
elseif line[0] !=# '#'
if line[0] ==# 'u'
let file = matchstr(line, '^.\{37\} \x\{40,\} \x\{40,\} \x\{40,\} \zs.*$')
else
let file = matchstr(line, '^.\{30\} \x\{40,\} \x\{40,\} \zs.*$')
endif
if line[0] ==# '2'
let i += 1
let file = matchstr(file, ' \zs.*')
let files = output[i] . ' -> ' . file
else
let files = file
endif
let sub = matchstr(line, '^[12u] .. \zs....')
if line[2] !=# '.'
call add(staged, {'type': 'File', 'status': line[2], 'filename': files, 'sub': sub})
endif
if line[3] !=# '.'
call add(unstaged, {'type': 'File', 'status': get({'C':'M','M':'?','U':'?'}, matchstr(sub, 'S\.*\zs[CMU]'), line[3]), 'filename': file, 'sub': sub})
endif
endif
2019-01-08 10:11:54 +00:00
let i += 1
2019-11-16 15:28:42 +00:00
endwhile
let branch = substitute(get(props, 'branch.head', '(unknown)'), '\C^(\%(detached\|unknown\))$', '', '')
if len(branch)
let head = branch
elseif has_key(props, 'branch.oid')
let head = props['branch.oid'][0:10]
else
let head = FugitiveHead(11)
2019-01-08 10:11:54 +00:00
endif
2019-11-16 15:28:42 +00:00
let pull = get(props, 'branch.upstream', '')
else " git < 2.11
let cmd += ['status', '--porcelain', '-bz']
let [output, message, exec_error] = s:NullError(cmd)
if exec_error
throw 'fugitive: ' . message
2019-01-08 10:11:54 +00:00
endif
2019-11-16 15:28:42 +00:00
while get(output, 0, '') =~# '^\l\+:'
call remove(output, 0)
endwhile
let head = matchstr(output[0], '^## \zs\S\+\ze\%($\| \[\)')
let pull = ''
if head =~# '\.\.\.'
let [head, pull] = split(head, '\.\.\.')
let branch = head
elseif head ==# 'HEAD' || empty(head)
let head = FugitiveHead(11)
let branch = ''
else
let branch = head
2019-01-08 10:11:54 +00:00
endif
2019-11-16 15:28:42 +00:00
let i = 0
while i < len(output)
let line = output[i]
let file = line[3:-1]
let files = file
let i += 1
if line[2] !=# ' '
continue
endif
if line[0:1] =~# '[RC]'
let files = output[i] . ' -> ' . file
let i += 1
endif
if line[0] !~# '[ ?!#]'
call add(staged, {'type': 'File', 'status': line[0], 'filename': files, 'sub': ''})
endif
if line[0:1] ==# '??'
call add(untracked, {'type': 'File', 'status': line[1], 'filename': files})
elseif line[1] !~# '[ !#]'
call add(unstaged, {'type': 'File', 'status': line[1], 'filename': file, 'sub': ''})
endif
endwhile
endif
2019-03-08 11:04:56 +00:00
2019-11-30 12:06:56 +00:00
if empty(s:Tree())
let [unstaged, untracked] = [[], []]
endif
2019-03-08 11:04:56 +00:00
for dict in staged
2019-11-16 15:28:42 +00:00
let b:fugitive_files['Staged'][dict.filename] = dict
2019-03-08 11:04:56 +00:00
endfor
for dict in unstaged
2019-11-16 15:28:42 +00:00
let b:fugitive_files['Unstaged'][dict.filename] = dict
2019-03-08 11:04:56 +00:00
endfor
2019-01-08 10:11:54 +00:00
let pull_type = 'Pull'
if len(pull)
let rebase = fugitive#Config('branch.' . branch . '.rebase', config)
if empty(rebase)
let rebase = fugitive#Config('pull.rebase', config)
endif
2020-01-29 02:07:36 +00:00
if rebase =~# '^\%(true\|yes\|on\|1\|interactive\|merges\|preserve\)$'
2019-01-08 10:11:54 +00:00
let pull_type = 'Rebase'
elseif rebase =~# '^\%(false\|no|off\|0\|\)$'
let pull_type = 'Merge'
endif
endif
let push_remote = fugitive#Config('branch.' . branch . '.pushRemote', config)
if empty(push_remote)
let push_remote = fugitive#Config('remote.pushDefault', config)
endif
2020-01-29 02:07:36 +00:00
let fetch_remote = fugitive#Config('branch.' . branch . '.remote', config)
if empty(fetch_remote)
let fetch_remote = 'origin'
endif
if empty(push_remote)
let push_remote = fetch_remote
2019-01-08 10:11:54 +00:00
endif
2020-01-29 02:07:36 +00:00
let push_default = fugitive#Config('push.default')
if empty(push_default)
let push_default = fugitive#GitVersion(2) ? 'simple' : 'matching'
2019-01-08 10:11:54 +00:00
endif
2020-01-29 02:07:36 +00:00
if push_default ==# 'upstream'
let push = pull
2018-07-30 21:18:16 +00:00
else
2020-01-29 02:07:36 +00:00
let push = len(branch) ? (push_remote ==# '.' ? '' : push_remote . '/') . branch : ''
2019-01-08 10:11:54 +00:00
endif
if isdirectory(fugitive#Find('.git/rebase-merge/'))
let rebasing_dir = fugitive#Find('.git/rebase-merge/')
elseif isdirectory(fugitive#Find('.git/rebase-apply/'))
let rebasing_dir = fugitive#Find('.git/rebase-apply/')
endif
let rebasing = []
let rebasing_head = 'detached HEAD'
if exists('rebasing_dir') && filereadable(rebasing_dir . 'git-rebase-todo')
let rebasing_head = substitute(readfile(rebasing_dir . 'head-name')[0], '\C^refs/heads/', '', '')
let len = 11
let lines = readfile(rebasing_dir . 'git-rebase-todo')
for line in lines
let hash = matchstr(line, '^[^a-z].*\s\zs[0-9a-f]\{4,\}\ze\.\.')
if len(hash)
let len = len(hash)
break
endif
endfor
if getfsize(rebasing_dir . 'done') > 0
let done = readfile(rebasing_dir . 'done')
call map(done, 'substitute(v:val, ''^\l\+\>'', "done", "")')
let done[-1] = substitute(done[-1], '^\l\+\>', 'stop', '')
let lines = done + lines
2018-07-30 21:18:16 +00:00
endif
2019-01-08 10:11:54 +00:00
call reverse(lines)
for line in lines
let match = matchlist(line, '^\(\l\+\)\s\+\(\x\{4,\}\)\s\+\(.*\)')
if len(match) && match[1] !~# 'exec\|merge\|label'
call add(rebasing, {'type': 'Rebase', 'status': get(s:rebase_abbrevs, match[1], match[1]), 'commit': strpart(match[2], 0, len), 'subject': match[3]})
endif
endfor
2018-07-30 21:18:16 +00:00
endif
2019-01-08 10:11:54 +00:00
2019-08-22 15:36:17 +00:00
let diff = {'Staged': [], 'Unstaged': []}
if len(staged)
let diff['Staged'] =
\ s:LinesError(['diff', '--color=never', '--no-ext-diff', '--no-prefix', '--cached'])[0]
endif
if len(unstaged)
let diff['Unstaged'] =
\ s:LinesError(['diff', '--color=never', '--no-ext-diff', '--no-prefix'])[0]
endif
let b:fugitive_diff = diff
2019-01-08 10:11:54 +00:00
let expanded = get(b:, 'fugitive_expanded', {'Staged': {}, 'Unstaged': {}})
let b:fugitive_expanded = {'Staged': {}, 'Unstaged': {}}
silent keepjumps %delete_
call s:AddHeader('Head', head)
call s:AddHeader(pull_type, pull)
if push !=# pull
call s:AddHeader('Push', push)
endif
2019-11-30 12:06:56 +00:00
if empty(s:Tree())
call s:AddHeader('Bare', 'yes')
endif
2019-01-08 10:11:54 +00:00
call s:AddSection('Rebasing ' . rebasing_head, rebasing)
2019-08-22 15:36:17 +00:00
call s:AddSection('Untracked', untracked)
2019-01-08 10:11:54 +00:00
call s:AddSection('Unstaged', unstaged)
let unstaged_end = len(unstaged) ? line('$') : 0
call s:AddSection('Staged', staged)
let staged_end = len(staged) ? line('$') : 0
2020-01-29 02:07:36 +00:00
if len(pull) && get(props, 'branch.ab') !~# ' -0$'
call s:AddSection('Unpulled from ' . pull, s:QueryLog(head . '..' . pull))
endif
if len(push) && push !=# pull
call s:AddSection('Unpulled from ' . push, s:QueryLog(head . '..' . push))
endif
if len(pull) && push !=# pull
call s:AddSection('Unpushed to ' . pull, s:QueryLog(pull . '..' . head))
endif
if len(push) && !(push ==# pull && get(props, 'branch.ab') =~# '^+0 ')
call s:AddSection('Unpushed to ' . push, s:QueryLog(push . '..' . head))
endif
2019-01-08 10:11:54 +00:00
2019-03-08 11:04:56 +00:00
setlocal nomodified readonly noswapfile
2019-01-08 10:11:54 +00:00
silent doautocmd BufReadPost
setlocal nomodifiable
2018-07-30 21:18:16 +00:00
if &bufhidden ==# ''
setlocal bufhidden=delete
endif
2019-01-08 10:11:54 +00:00
let b:dispatch = ':Gfetch --all'
2018-07-30 21:18:16 +00:00
call fugitive#MapJumps()
2019-08-22 15:36:17 +00:00
call s:Map('n', '-', ":<C-U>execute <SID>Do('Toggle',0)<CR>", '<silent>')
call s:Map('x', '-', ":<C-U>execute <SID>Do('Toggle',1)<CR>", '<silent>')
call s:Map('n', 's', ":<C-U>execute <SID>Do('Stage',0)<CR>", '<silent>')
call s:Map('x', 's', ":<C-U>execute <SID>Do('Stage',1)<CR>", '<silent>')
call s:Map('n', 'u', ":<C-U>execute <SID>Do('Unstage',0)<CR>", '<silent>')
call s:Map('x', 'u', ":<C-U>execute <SID>Do('Unstage',1)<CR>", '<silent>')
call s:Map('n', 'U', ":exe <SID>EchoExec('reset', '-q')<CR>", '<silent>')
call s:MapMotion('gu', "exe <SID>StageJump(v:count, 'Untracked', 'Unstaged')")
call s:MapMotion('gU', "exe <SID>StageJump(v:count, 'Unstaged', 'Untracked')")
call s:MapMotion('gs', "exe <SID>StageJump(v:count, 'Staged')")
call s:MapMotion('gp', "exe <SID>StageJump(v:count, 'Unpushed')")
call s:MapMotion('gP', "exe <SID>StageJump(v:count, 'Unpulled')")
call s:MapMotion('gr', "exe <SID>StageJump(v:count, 'Rebasing')")
call s:Map('n', 'C', ":<C-U>Gcommit<CR>:echohl WarningMsg<Bar>echo ':Gstatus C is deprecated in favor of cc'<Bar>echohl NONE<CR>", '<silent>')
call s:Map('n', 'a', ":<C-U>execute <SID>Do('Toggle',0)<CR>", '<silent>')
call s:Map('n', 'i', ":<C-U>execute <SID>NextExpandedHunk(v:count1)<CR>", '<silent>')
call s:Map('n', "=", ":<C-U>execute <SID>StageInline('toggle',line('.'),v:count)<CR>", '<silent>')
call s:Map('n', "<", ":<C-U>execute <SID>StageInline('hide', line('.'),v:count)<CR>", '<silent>')
call s:Map('n', ">", ":<C-U>execute <SID>StageInline('show', line('.'),v:count)<CR>", '<silent>')
call s:Map('x', "=", ":<C-U>execute <SID>StageInline('toggle',line(\"'<\"),line(\"'>\")-line(\"'<\")+1)<CR>", '<silent>')
call s:Map('x', "<", ":<C-U>execute <SID>StageInline('hide', line(\"'<\"),line(\"'>\")-line(\"'<\")+1)<CR>", '<silent>')
call s:Map('x', ">", ":<C-U>execute <SID>StageInline('show', line(\"'<\"),line(\"'>\")-line(\"'<\")+1)<CR>", '<silent>')
call s:Map('n', 'D', ":<C-U>execute <SID>StageDiff('Gdiffsplit')<Bar>redraw<Bar>echohl WarningMsg<Bar> echo ':Gstatus D is deprecated in favor of dd'<Bar>echohl NONE<CR>", '<silent>')
call s:Map('n', 'dd', ":<C-U>execute <SID>StageDiff('Gdiffsplit')<CR>", '<silent>')
call s:Map('n', 'dh', ":<C-U>execute <SID>StageDiff('Ghdiffsplit')<CR>", '<silent>')
call s:Map('n', 'ds', ":<C-U>execute <SID>StageDiff('Ghdiffsplit')<CR>", '<silent>')
call s:Map('n', 'dp', ":<C-U>execute <SID>StageDiffEdit()<CR>", '<silent>')
call s:Map('n', 'dv', ":<C-U>execute <SID>StageDiff('Gvdiffsplit')<CR>", '<silent>')
call s:Map('n', 'd?', ":<C-U>help fugitive_d<CR>", '<silent>')
call s:Map('n', 'P', ":<C-U>execute <SID>StagePatch(line('.'),line('.')+v:count1-1)<CR>", '<silent>')
call s:Map('x', 'P', ":<C-U>execute <SID>StagePatch(line(\"'<\"),line(\"'>\"))<CR>", '<silent>')
call s:Map('n', 'p', ":<C-U>if v:count<Bar>silent exe <SID>GF('pedit')<Bar>else<Bar>echoerr 'Use = for inline diff, P for :Git add/reset --patch, 1p for :pedit'<Bar>endif<CR>", '<silent>')
call s:Map('x', 'p', ":<C-U>execute <SID>StagePatch(line(\"'<\"),line(\"'>\"))<CR>", '<silent>')
call s:Map('n', 'I', ":<C-U>execute <SID>StagePatch(line('.'),line('.'))<CR>", '<silent>')
call s:Map('x', 'I', ":<C-U>execute <SID>StagePatch(line(\"'<\"),line(\"'>\"))<CR>", '<silent>')
if empty(mapcheck('q', 'n'))
nnoremap <buffer> <silent> q :<C-U>if bufnr('$') == 1<Bar>quit<Bar>else<Bar>bdelete<Bar>endif<Bar>echohl WarningMsg<Bar>echo ':Gstatus q is deprecated in favor of gq or the built-in <Lt>C-W>q'<Bar>echohl NONE<CR>
endif
call s:Map('n', 'gq', ":<C-U>if bufnr('$') == 1<Bar>quit<Bar>else<Bar>bdelete<Bar>endif<CR>", '<silent>')
call s:Map('n', 'R', ":echohl WarningMsg<Bar>echo 'Reloading is automatic. Use :e to force'<Bar>echohl NONE<CR>", '<silent>')
call s:Map('n', 'g<Bar>', ":<C-U>echoerr 'Changed to X'<CR>", '<silent>')
call s:Map('x', 'g<Bar>', ":<C-U>echoerr 'Changed to X'<CR>", '<silent>')
call s:Map('n', 'X', ":<C-U>execute <SID>StageDelete(line('.'), 0, v:count)<CR>", '<silent>')
call s:Map('x', 'X', ":<C-U>execute <SID>StageDelete(line(\"'<\"), line(\"'>\"), v:count)<CR>", '<silent>')
call s:Map('n', 'gI', ":<C-U>execute <SID>StageIgnore(line('.'), line('.'), v:count)<CR>", '<silent>')
call s:Map('x', 'gI', ":<C-U>execute <SID>StageIgnore(line(\"'<\"), line(\"'>\"), v:count)<CR>", '<silent>')
call s:Map('n', '.', ':<C-U> <C-R>=<SID>StageArgs(0)<CR><Home>')
call s:Map('x', '.', ':<C-U> <C-R>=<SID>StageArgs(1)<CR><Home>')
2019-03-08 11:04:56 +00:00
setlocal filetype=fugitive
2019-01-08 10:11:54 +00:00
for [lnum, section] in [[staged_end, 'Staged'], [unstaged_end, 'Unstaged']]
while len(getline(lnum))
let filename = matchstr(getline(lnum), '^[A-Z?] \zs.*')
if has_key(expanded[section], filename)
call s:StageInline('show', lnum)
endif
let lnum -= 1
endwhile
endfor
2019-08-22 15:36:17 +00:00
let b:fugitive_reltime = reltime()
2020-01-29 02:07:36 +00:00
return 'silent ' . s:DoAutocmd('User FugitiveIndex')
2018-07-30 21:18:16 +00:00
catch /^fugitive:/
2019-08-22 15:36:17 +00:00
return 'echoerr ' . string(v:exception)
2018-07-30 21:18:16 +00:00
endtry
endfunction
function! fugitive#FileReadCmd(...) abort
let amatch = a:0 ? a:1 : expand('<amatch>')
let [dir, rev] = s:DirRev(amatch)
let line = a:0 > 1 ? a:2 : line("'[")
if empty(dir)
return 'noautocmd ' . line . 'read ' . s:fnameescape(amatch)
endif
2019-08-22 15:36:17 +00:00
if rev !~# ':' && s:ChompDefault('', [dir, 'cat-file', '-t', rev]) =~# '^\%(commit\|tag\)$'
let cmd = fugitive#Prepare(dir, 'log', '--pretty=format:%B', '-1', rev, '--')
2018-07-30 21:18:16 +00:00
else
2019-08-22 15:36:17 +00:00
let cmd = fugitive#Prepare(dir, 'cat-file', '-p', rev)
2018-07-30 21:18:16 +00:00
endif
return line . 'read !' . escape(cmd, '!#%')
endfunction
function! fugitive#FileWriteCmd(...) abort
let tmp = tempname()
let amatch = a:0 ? a:1 : expand('<amatch>')
let autype = a:0 > 1 ? 'Buf' : 'File'
if exists('#' . autype . 'WritePre')
2019-11-16 15:28:42 +00:00
execute s:DoAutocmd(autype . 'WritePre ' . s:fnameescape(amatch))
2018-07-30 21:18:16 +00:00
endif
try
let [dir, commit, file] = s:DirCommitFile(amatch)
if commit !~# '^[0-3]$' || !v:cmdbang && (line("'[") != 1 || line("']") != line('$'))
return "noautocmd '[,']write" . (v:cmdbang ? '!' : '') . ' ' . s:fnameescape(amatch)
endif
2019-08-22 15:36:17 +00:00
silent execute "'[,']write !".fugitive#Prepare(dir, 'hash-object', '-w', '--stdin', '--').' > '.tmp
2018-07-30 21:18:16 +00:00
let sha1 = readfile(tmp)[0]
2019-08-22 15:36:17 +00:00
let old_mode = matchstr(s:SystemError([dir, 'ls-files', '--stage', '.' . file])[0], '^\d\+')
2018-08-25 16:13:42 +00:00
if empty(old_mode)
2018-07-30 21:18:16 +00:00
let old_mode = executable(s:Tree(dir) . file) ? '100755' : '100644'
endif
let info = old_mode.' '.sha1.' '.commit."\t".file[1:-1]
2019-08-22 15:36:17 +00:00
let [error, exec_error] = s:SystemError([dir, 'update-index', '--index-info'], info . "\n")
if !exec_error
2018-07-30 21:18:16 +00:00
setlocal nomodified
if exists('#' . autype . 'WritePost')
2019-11-16 15:28:42 +00:00
execute s:DoAutocmd(autype . 'WritePost ' . s:fnameescape(amatch))
2018-07-30 21:18:16 +00:00
endif
return ''
else
return 'echoerr '.string('fugitive: '.error)
endif
finally
call delete(tmp)
endtry
endfunction
function! fugitive#BufReadCmd(...) abort
let amatch = a:0 ? a:1 : expand('<amatch>')
try
let [dir, rev] = s:DirRev(amatch)
if empty(dir)
return 'echo "Invalid Fugitive URL"'
endif
if rev =~# '^:\d$'
let b:fugitive_type = 'stage'
else
2019-08-22 15:36:17 +00:00
let [b:fugitive_type, exec_error] = s:ChompError([dir, 'cat-file', '-t', rev])
if exec_error && rev =~# '^:0'
let sha = s:ChompDefault('', dir, 'write-tree', '--prefix=' . rev[3:-1])
let exec_error = empty(sha)
let b:fugitive_type = exec_error ? '' : 'tree'
2018-07-30 21:18:16 +00:00
endif
2019-08-22 15:36:17 +00:00
if exec_error
2018-09-25 00:40:17 +00:00
let error = b:fugitive_type
2018-07-30 21:18:16 +00:00
unlet b:fugitive_type
2019-08-22 15:36:17 +00:00
setlocal noswapfile
if empty(&bufhidden)
setlocal bufhidden=delete
endif
2018-07-30 21:18:16 +00:00
if rev =~# '^:\d:'
2019-08-22 15:36:17 +00:00
let &l:readonly = !filewritable(fugitive#Find('.git/index', dir))
2019-01-08 10:11:54 +00:00
return 'silent doautocmd BufNewFile'
2018-07-30 21:18:16 +00:00
else
setlocal readonly nomodifiable
2019-08-22 15:36:17 +00:00
return 'silent doautocmd BufNewFile|echo ' . string(error)
2018-07-30 21:18:16 +00:00
endif
elseif b:fugitive_type !~# '^\%(tag\|commit\|tree\|blob\)$'
return "echoerr ".string("fugitive: unrecognized git type '".b:fugitive_type."'")
endif
if !exists('b:fugitive_display_format') && b:fugitive_type != 'blob'
let b:fugitive_display_format = +getbufvar('#','fugitive_display_format')
endif
endif
if b:fugitive_type !=# 'blob'
setlocal nomodeline
endif
setlocal noreadonly modifiable
let pos = getpos('.')
silent keepjumps %delete_
setlocal endofline
try
2020-01-29 02:07:36 +00:00
silent exe s:DoAutocmd('BufReadPre')
2018-07-30 21:18:16 +00:00
if b:fugitive_type ==# 'tree'
let b:fugitive_display_format = b:fugitive_display_format % 2
if b:fugitive_display_format
call s:ReplaceCmd([dir, 'ls-tree', exists('sha') ? sha : rev])
else
if !exists('sha')
2019-08-22 15:36:17 +00:00
let sha = s:TreeChomp(dir, 'rev-parse', '--verify', rev, '--')
2018-07-30 21:18:16 +00:00
endif
call s:ReplaceCmd([dir, 'show', '--no-color', sha])
endif
elseif b:fugitive_type ==# 'tag'
let b:fugitive_display_format = b:fugitive_display_format % 2
if b:fugitive_display_format
call s:ReplaceCmd([dir, 'cat-file', b:fugitive_type, rev])
else
call s:ReplaceCmd([dir, 'cat-file', '-p', rev])
endif
elseif b:fugitive_type ==# 'commit'
let b:fugitive_display_format = b:fugitive_display_format % 2
if b:fugitive_display_format
call s:ReplaceCmd([dir, 'cat-file', b:fugitive_type, rev])
else
2019-11-16 15:28:42 +00:00
call s:ReplaceCmd([dir, 'show', '--no-color', '-m', '--first-parent', '--pretty=format:tree%x20%T%nparent%x20%P%nauthor%x20%an%x20<%ae>%x20%ad%ncommitter%x20%cn%x20<%ce>%x20%cd%nencoding%x20%e%n%n%s%n%n%b', rev])
2020-01-29 02:07:36 +00:00
keepjumps 1
2018-07-30 21:18:16 +00:00
keepjumps call search('^parent ')
if getline('.') ==# 'parent '
silent keepjumps delete_
else
2018-09-25 00:40:17 +00:00
silent exe (exists(':keeppatterns') ? 'keeppatterns' : '') 'keepjumps s/\m\C\%(^parent\)\@<! /\rparent /e' . (&gdefault ? '' : 'g')
2018-07-30 21:18:16 +00:00
endif
keepjumps let lnum = search('^encoding \%(<unknown>\)\=$','W',line('.')+3)
if lnum
silent keepjumps delete_
end
2018-11-01 10:03:42 +00:00
silent exe (exists(':keeppatterns') ? 'keeppatterns' : '') 'keepjumps 1,/^diff --git\|\%$/s/\r$//e'
2018-07-30 21:18:16 +00:00
keepjumps 1
endif
elseif b:fugitive_type ==# 'stage'
call s:ReplaceCmd([dir, 'ls-files', '--stage'])
elseif b:fugitive_type ==# 'blob'
call s:ReplaceCmd([dir, 'cat-file', b:fugitive_type, rev])
endif
finally
keepjumps call setpos('.',pos)
setlocal nomodified noswapfile
2019-01-08 10:11:54 +00:00
let modifiable = rev =~# '^:.:' && b:fugitive_type !=# 'tree'
2019-08-22 15:36:17 +00:00
let &l:readonly = !modifiable || !filewritable(fugitive#Find('.git/index', dir))
if empty(&bufhidden)
2018-07-30 21:18:16 +00:00
setlocal bufhidden=delete
endif
2019-01-08 10:11:54 +00:00
let &l:modifiable = modifiable
2018-07-30 21:18:16 +00:00
if b:fugitive_type !=# 'blob'
setlocal filetype=git foldmethod=syntax
2019-08-22 15:36:17 +00:00
call s:Map('n', 'a', ":<C-U>let b:fugitive_display_format += v:count1<Bar>exe fugitive#BufReadCmd(@%)<CR>", '<silent>')
call s:Map('n', 'i', ":<C-U>let b:fugitive_display_format -= v:count1<Bar>exe fugitive#BufReadCmd(@%)<CR>", '<silent>')
2018-07-30 21:18:16 +00:00
endif
2019-03-08 11:04:56 +00:00
call fugitive#MapJumps()
2018-07-30 21:18:16 +00:00
endtry
2019-01-08 10:11:54 +00:00
setlocal modifiable
2020-01-29 02:07:36 +00:00
let browsex = maparg('<Plug>NetrwBrowseX', 'n')
let remote_check = '\Cnetrw#CheckIfRemote(\%(netrw#GX()\)\=)'
if browsex =~# remote_check
exe 'nnoremap <silent> <buffer> <Plug>NetrwBrowseX' substitute(browsex, remote_check, '0', 'g')
endif
2019-11-16 15:28:42 +00:00
return 'silent ' . s:DoAutocmd('BufReadPost') .
2020-01-29 02:07:36 +00:00
\ (modifiable ? '' : '|setl nomodifiable') . '|silent ' .
\ s:DoAutocmd('User Fugitive' . substitute(b:fugitive_type, '^\l', '\u&', ''))
2018-07-30 21:18:16 +00:00
catch /^fugitive:/
2019-08-22 15:36:17 +00:00
return 'echoerr ' . string(v:exception)
2018-07-30 21:18:16 +00:00
endtry
endfunction
function! fugitive#BufWriteCmd(...) abort
return fugitive#FileWriteCmd(a:0 ? a:1 : expand('<amatch>'), 1)
endfunction
function! fugitive#SourceCmd(...) abort
let amatch = a:0 ? a:1 : expand('<amatch>')
let temp = s:BlobTemp(amatch)
if empty(temp)
return 'noautocmd source ' . s:fnameescape(amatch)
endif
if !exists('g:virtual_scriptnames')
let g:virtual_scriptnames = {}
endif
let g:virtual_scriptnames[temp] = amatch
return 'source ' . s:fnameescape(temp)
endfunction
" Section: Temp files
if !exists('s:temp_files')
let s:temp_files = {}
endif
2019-08-22 15:36:17 +00:00
function! s:TempState(...) abort
return get(s:temp_files, s:cpath(fnamemodify(a:0 ? a:1 : @%, ':p')), {})
endfunction
function! s:TempReadPre(file) abort
if has_key(s:temp_files, s:cpath(a:file))
let dict = s:temp_files[s:cpath(a:file)]
setlocal nomodeline
setlocal bufhidden=delete nobuflisted
setlocal buftype=nowrite
2020-01-29 02:07:36 +00:00
setlocal nomodifiable
2019-08-22 15:36:17 +00:00
if len(dict.dir)
let b:git_dir = dict.dir
call extend(b:, {'fugitive_type': 'temp'}, 'keep')
endif
endif
endfunction
function! s:TempReadPost(file) abort
2018-08-25 16:13:42 +00:00
if has_key(s:temp_files, s:cpath(a:file))
let dict = s:temp_files[s:cpath(a:file)]
if has_key(dict, 'filetype') && dict.filetype !=# &l:filetype
let &l:filetype = dict.filetype
endif
setlocal foldmarker=<<<<<<<,>>>>>>>
2019-08-22 15:36:17 +00:00
if empty(mapcheck('q', 'n'))
nnoremap <buffer> <silent> q :<C-U>bdelete<Bar>echohl WarningMsg<Bar>echo "Temp file q is deprecated in favor of the built-in <Lt>C-W>q"<Bar>echohl NONE<CR>
2018-08-25 16:13:42 +00:00
endif
2019-08-22 15:36:17 +00:00
if !&modifiable
call s:Map('n', 'gq', ":<C-U>bdelete<CR>", '<silent> <unique>')
2019-01-08 10:11:54 +00:00
endif
2018-08-25 16:13:42 +00:00
endif
return ''
endfunction
2018-07-30 21:18:16 +00:00
augroup fugitive_temp
autocmd!
2019-08-22 15:36:17 +00:00
autocmd BufReadPre * exe s:TempReadPre( expand('<amatch>:p'))
autocmd BufReadPost * exe s:TempReadPost(expand('<amatch>:p'))
2018-07-30 21:18:16 +00:00
augroup END
2018-08-25 16:13:42 +00:00
" Section: :Git
2018-07-30 21:18:16 +00:00
2020-01-29 02:07:36 +00:00
function! s:RunJobs() abort
return exists('*job_start') || exists('*jobstart')
endfunction
function! s:RunReceive(state, job, data, ...) abort
call add(a:state.log, a:data)
let data = type(a:data) == type([]) ? join(a:data, "\n") : a:data
if has_key(a:state, 'buffer')
let data = remove(a:state, 'buffer') . data
endif
let escape = "\033]51;[^\007]*"
let a:state.buffer = matchstr(data, escape . "$\\|[\r\n]\\+$")
if len(a:state.buffer)
let data = strpart(data, 0, len(data) - len(a:state.buffer))
endif
let cmd = matchstr(data, escape . "\007")[5:-2]
let data = substitute(data, escape . "\007", '', 'g')
echon substitute(data, "\r\\ze\n", '', 'g')
if cmd =~# '^fugitive:'
let a:state.request = strpart(cmd, 9)
endif
endfunction
function! s:RunSend(job, str) abort
try
if type(a:job) == type(0)
call chansend(a:job, a:str)
else
call ch_sendraw(a:job, a:str)
endif
return len(a:str)
catch /^Vim\%((\a\+)\)\=:E90[06]:/
return 0
endtry
endfunction
if !exists('s:edit_jobs')
let s:edit_jobs = {}
endif
function! s:RunWait(state, job) abort
let finished = 0
try
while get(a:state, 'request', '') !=# 'edit' && (type(a:job) == type(0) ? jobwait([a:job], 1)[0] == -1 : ch_status(a:job) !=# 'closed')
if !exists('*jobwait')
sleep 1m
endif
let peek = getchar(1)
if peek != 0 && !(has('win32') && peek == 128)
let c = getchar()
let c = type(c) == type(0) ? nr2char(c) : c
call s:RunSend(a:job, c)
if !a:state.pty
echon c
endif
endif
endwhile
sleep 1m
echo
if get(a:state, 'request', '') == 'edit'
call remove(a:state, 'request')
let file = readfile(a:state.temp . '.edit')[0]
exe substitute(a:state.mods, '\<tab\>', '-tab', 'g') 'keepalt split' s:fnameescape(file)
set bufhidden=wipe
let s:edit_jobs[bufnr('')] = [a:state, a:job]
endif
let finished = 1
finally
if !finished
try
if a:state.pty
call s:RunSend(a:job, "\<C-C>")
elseif type(a:job) == type(0)
call jobstop(a:job)
else
call job_stop(a:job)
endif
catch /.*/
endtry
endif
endtry
call fugitive#ReloadStatus(a:state.dir, 1)
return ''
endfunction
if !exists('s:resume_queue')
let s:resume_queue = []
endif
function! fugitive#Resume() abort
while len(s:resume_queue)
let [state, job] = remove(s:resume_queue, 0)
call s:RunWait(state, job)
endwhile
endfunction
function! s:RunBufDelete(bufnr) abort
if has_key(s:edit_jobs, a:bufnr) |
if filereadable(s:edit_jobs[a:bufnr][0].temp . '.edit')
call delete(s:edit_jobs[a:bufnr][0].temp . '.edit')
endif
call add(s:resume_queue, remove(s:edit_jobs, a:bufnr))
call feedkeys(":redraw!|call fugitive#Resume()|silent checktime\r", 'n')
endif
endfunction
augroup fugitive_job
autocmd!
autocmd BufDelete * call s:RunBufDelete(expand('<abuf>'))
autocmd VimLeave *
\ for s:jobbuf in keys(s:edit_jobs) |
\ call writefile([], remove(s:edit_jobs, s:jobbuf)[0].temp . '.exit') |
\ endfor
augroup END
let s:disable_colors = []
for s:colortype in ['advice', 'branch', 'diff', 'grep', 'interactive', 'pager', 'push', 'remote', 'showBranch', 'status', 'transport', 'ui']
call extend(s:disable_colors, ['-c', 'color.' . s:colortype . '=false'])
endfor
unlet s:colortype
2019-11-16 15:28:42 +00:00
function! fugitive#Command(line1, line2, range, bang, mods, arg) abort
2019-08-22 15:36:17 +00:00
let dir = s:Dir()
let [args, after] = s:SplitExpandChain(a:arg, s:Tree(dir))
if empty(args)
2019-11-16 15:28:42 +00:00
let cmd = s:StatusCommand(a:line1, a:line2, a:range, a:line2, a:bang, a:mods, '', '', [])
2019-08-22 15:36:17 +00:00
return (empty(cmd) ? 'exe' : cmd) . after
endif
let alias = get(s:Aliases(dir), args[0], '!')
if get(args, 1, '') !=# '--help' && alias !~# '^!\|[\"'']' && !filereadable(s:ExecPath() . '/git-' . args[0])
\ && !(has('win32') && filereadable(s:ExecPath() . '/git-' . args[0] . '.exe'))
call remove(args, 0)
call extend(args, split(alias, '\s\+'), 'keep')
endif
let name = substitute(args[0], '\%(^\|-\)\(\l\)', '\u\1', 'g')
if exists('*s:' . name . 'Subcommand') && get(args, 1, '') !=# '--help'
try
exe s:DirCheck(dir)
2020-01-29 02:07:36 +00:00
let opts = s:{name}Subcommand(a:line1, a:line2, a:range, a:bang, a:mods, args[1:-1])
if type(opts) == type('')
return 'exe ' . string(opts) . after
2020-01-07 12:45:07 +00:00
endif
2019-08-22 15:36:17 +00:00
catch /^fugitive:/
return 'echoerr ' . string(v:exception)
endtry
2020-01-29 02:07:36 +00:00
else
let opts = {}
2018-06-14 10:31:12 +00:00
endif
2020-01-29 02:07:36 +00:00
let args = get(opts, 'args', args)
if a:bang || args[0] =~# '^-p$\|^--paginate$\|diff\%(tool\)\@!\|log\|^show$' ||
2019-08-22 15:36:17 +00:00
\ (args[0] ==# 'stash' && get(args, 1, '') ==# 'show') ||
\ (args[0] ==# 'help' || get(args, 1, '') ==# '--help') && !s:HasOpt(args, '--web')
2020-01-29 02:07:36 +00:00
if args[0] =~# '^-p$\|^--paginate$'
call remove(args, 0)
endif
2019-11-16 15:28:42 +00:00
return s:OpenExec((a:line2 > 0 ? a:line2 : '') . (a:line2 ? 'split' : 'edit'), a:mods, args, dir) . after
2018-07-30 21:18:16 +00:00
endif
2019-08-22 15:36:17 +00:00
if s:HasOpt(args, ['add', 'checkout', 'commit', 'stage', 'stash', 'reset'], '-p', '--patch') ||
2020-01-29 02:07:36 +00:00
\ s:HasOpt(args, ['add', 'clean', 'stage'], '-i', '--interactive')
2019-08-22 15:36:17 +00:00
let mods = substitute(s:Mods(a:mods), '\<tab\>', '-tab', 'g')
2019-11-16 15:28:42 +00:00
let assign = len(dir) ? '|let b:git_dir = ' . string(dir) : ''
2019-08-22 15:36:17 +00:00
if has('nvim')
if &autowrite || &autowriteall | silent! wall | endif
2019-11-16 15:28:42 +00:00
return mods . (a:line2 ? 'split' : 'edit') . ' term://' . s:fnameescape(s:UserCommand(dir, args)) . assign . '|startinsert' . after
2019-08-22 15:36:17 +00:00
elseif has('terminal')
if &autowrite || &autowriteall | silent! wall | endif
2019-11-16 15:28:42 +00:00
return 'exe ' . string(mods . 'terminal ' . (a:line2 ? '' : '++curwin ') . join(map(s:UserCommandList(dir) + args, 's:fnameescape(v:val)'))) . assign . after
2018-06-14 10:31:12 +00:00
endif
2018-07-30 21:18:16 +00:00
endif
2020-01-29 02:07:36 +00:00
let env = get(opts, 'env', {})
if s:RunJobs()
let state = {'dir': dir, 'mods': s:Mods(a:mods), 'temp': tempname(), 'log': []}
let state.pty = get(g:, 'fugitive_pty', has('unix') && (has('patch-8.0.0744') || has('nvim')))
if !state.pty
let args = s:AskPassArgs(dir) + args
endif
let env.FUGITIVE_TEMP = state.temp
let editor = 'sh ' . s:TempScript(
\ '[ -f "$FUGITIVE_TEMP.exit" ] && exit 1',
\ 'echo "$1" > "$FUGITIVE_TEMP.edit"',
\ 'printf "\033]51;fugitive:edit\007"',
\ 'while [ -f "$FUGITIVE_TEMP.edit" -a ! -f "$FUGITIVE_TEMP.exit" ]; do sleep 0.05 2>/dev/null || sleep 1; done',
\ 'exit 0')
call extend(env, {
\ 'NO_COLOR': '1',
\ 'GIT_EDITOR': editor,
\ 'GIT_SEQUENCE_EDITOR': editor,
\ 'GIT_MERGE_AUTOEDIT': '1',
\ 'GIT_PAGER': 'cat',
\ 'PAGER': 'cat'}, 'keep')
let args = ['-c', 'advice.waitingForEditor=false'] + s:disable_colors + args
let argv = s:UserCommandList(dir) + args
if !has('patch-8.0.0902') || has('nvim')
let envlist = map(items(env), 'join(v:val, "=")')
if s:executable('env')
let argv = ['env'] + envlist + argv
elseif has('win32')
let argv = ['cmd.exe', '/c',
\ join(map(envlist, { _, v -> 'set ' . substitute(v, '[&|<>^]', '^^^&', 'g') }) +
\ [s:shellesc(argv)], '& ')]
else
return 'echoerr ' . string('fugitive: "env" command missing')
endif
let env = {}
endif
let state.cmd = argv
let g:_fugitive_last_job = state
if &autowrite || &autowriteall | silent! wall | endif
if exists('*job_start')
let jobopts = {
\ 'mode': 'raw',
\ 'callback': function('s:RunReceive', [state]),
\ }
if state.pty
let jobopts.pty = 1
endif
if len(env)
let jobopts.env = env
endif
let job = job_start(argv, jobopts)
else
let job = jobstart(argv, {
\ 'pty': state.pty,
\ 'env': env,
\ 'TERM': 'dumb',
\ 'on_stdout': function('s:RunReceive', [state]),
\ 'on_stderr': function('s:RunReceive', [state]),
\ })
endif
let state.job = job
call s:RunWait(state, job)
return 'silent checktime' . after
else
if has('gui_running') && !has('win32')
call insert(args, '--no-pager')
endif
if has('nvim')
let env.GIT_TERMINAL_PROMPT = '0'
endif
let pre = s:BuildEnvPrefix(env)
return 'exe ' . string('noautocmd !' . escape(pre . s:UserCommand(dir, args), '!#%')) .
\ '|call fugitive#ReloadStatus(' . string(dir) . ', 1)' .
\ after
2019-08-22 15:36:17 +00:00
endif
2018-07-30 21:18:16 +00:00
endfunction
let s:exec_paths = {}
2019-08-22 15:36:17 +00:00
function! s:ExecPath() abort
2018-07-30 21:18:16 +00:00
if !has_key(s:exec_paths, g:fugitive_git_executable)
let s:exec_paths[g:fugitive_git_executable] = s:sub(system(g:fugitive_git_executable.' --exec-path'),'\n$','')
2018-06-14 10:31:12 +00:00
endif
2019-08-22 15:36:17 +00:00
return s:exec_paths[g:fugitive_git_executable]
endfunction
function! s:Subcommands() abort
let exec_path = s:ExecPath()
2018-07-30 21:18:16 +00:00
return map(split(glob(exec_path.'/git-*'),"\n"),'s:sub(v:val[strlen(exec_path)+5 : -1],"\\.exe$","")')
2018-06-14 10:31:12 +00:00
endfunction
2018-07-30 21:18:16 +00:00
let s:aliases = {}
2019-05-17 14:09:13 +00:00
function! s:Aliases(dir) abort
if !has_key(s:aliases, a:dir)
let s:aliases[a:dir] = {}
2019-08-22 15:36:17 +00:00
let lines = s:NullError([a:dir, 'config', '-z', '--get-regexp', '^alias[.]'])[0]
for line in lines
2019-05-17 14:09:13 +00:00
let s:aliases[a:dir][matchstr(line, '\.\zs.\{-}\ze\n')] = matchstr(line, '\n\zs.*')
2018-07-30 21:18:16 +00:00
endfor
2018-06-14 10:31:12 +00:00
endif
2019-05-17 14:09:13 +00:00
return s:aliases[a:dir]
2018-06-14 10:31:12 +00:00
endfunction
2019-11-16 15:28:42 +00:00
function! fugitive#Complete(lead, ...) abort
2019-05-17 14:09:13 +00:00
let dir = a:0 == 1 ? a:1 : a:0 == 3 ? a:3 : s:Dir()
let pre = a:0 > 1 ? strpart(a:1, 0, a:2) : ''
2019-08-22 15:36:17 +00:00
let subcmd = matchstr(pre, '\u\w*[! ] *\zs[[:alnum:]-]\+\ze ')
if empty(subcmd)
let results = sort(s:Subcommands() + keys(s:Aliases(dir)))
2018-07-30 21:18:16 +00:00
elseif pre =~# ' -- '
2019-05-17 14:09:13 +00:00
return fugitive#CompletePath(a:lead, dir)
2019-08-22 15:36:17 +00:00
elseif a:lead =~# '^-'
let results = split(s:ChompDefault('', dir, subcmd, '--git-completion-helper'), ' ')
2018-06-14 10:31:12 +00:00
else
2019-05-17 14:09:13 +00:00
return fugitive#CompleteObject(a:lead, dir)
2018-06-14 10:31:12 +00:00
endif
2019-08-22 15:36:17 +00:00
return filter(results, 'strpart(v:val, 0, strlen(a:lead)) ==# a:lead')
2018-06-14 10:31:12 +00:00
endfunction
2018-08-25 16:13:42 +00:00
" Section: :Gcd, :Glcd
2018-06-14 10:31:12 +00:00
2019-11-16 15:28:42 +00:00
function! fugitive#CdComplete(A, L, P) abort
2019-05-17 14:09:13 +00:00
return filter(fugitive#CompletePath(a:A), 'v:val =~# "/$"')
2018-06-14 10:31:12 +00:00
endfunction
2019-11-16 15:28:42 +00:00
function! fugitive#Cd(path, ...) abort
2018-08-25 16:13:42 +00:00
let path = substitute(a:path, '^:/:\=\|^:(\%(top\|top,literal\|literal,top\|literal\))', '', '')
2019-11-16 15:28:42 +00:00
if path !~# '^/\|^\a\+:\|^\.\.\=\%(/\|$\)'
let dir = s:Dir()
exe s:DirCheck(dir)
let path = (empty(s:Tree(dir)) ? dir : s:Tree(dir)) . '/' . path
2018-08-25 16:13:42 +00:00
endif
2019-11-16 15:28:42 +00:00
return (a:0 && a:1 ? 'lcd ' : 'cd ') . s:fnameescape(FugitiveVimPath(path))
2018-08-25 16:13:42 +00:00
endfunction
" Section: :Gstatus
2018-06-14 10:31:12 +00:00
2019-08-22 15:36:17 +00:00
function! s:StatusCommand(line1, line2, range, count, bang, mods, reg, arg, args, ...) abort
let dir = a:0 ? a:1 : s:Dir()
exe s:DirCheck(dir)
2018-06-14 10:31:12 +00:00
try
2019-08-22 15:36:17 +00:00
let mods = s:Mods(a:mods, &splitbelow ? 'botright' : 'topleft')
let file = fugitive#Find(':', dir)
2019-03-08 11:04:56 +00:00
let arg = ' +setl\ foldmethod=syntax\ foldlevel=1\|let\ w:fugitive_status=FugitiveGitDir() ' .
\ s:fnameescape(file)
2019-01-08 10:11:54 +00:00
for winnr in range(1, winnr('$'))
if s:cpath(file, fnamemodify(bufname(winbufnr(winnr)), ':p'))
2019-08-22 15:36:17 +00:00
if winnr == winnr()
call s:ReloadStatus()
else
call s:ExpireStatus(dir)
exe winnr . 'wincmd w'
endif
let w:fugitive_status = dir
1
return ''
2019-01-08 10:11:54 +00:00
endif
endfor
if a:count ==# 0
2019-03-08 11:04:56 +00:00
return mods . 'edit' . (a:bang ? '!' : '') . arg
2019-01-08 10:11:54 +00:00
elseif a:bang
2019-03-08 11:04:56 +00:00
return mods . 'pedit' . arg . '|wincmd P'
2019-01-08 10:11:54 +00:00
else
2019-03-08 11:04:56 +00:00
return mods . (a:count > 0 ? a:count : '') . 'split' . arg
2019-01-08 10:11:54 +00:00
endif
2018-06-14 10:31:12 +00:00
catch /^fugitive:/
2019-08-22 15:36:17 +00:00
return 'echoerr ' . string(v:exception)
2018-06-14 10:31:12 +00:00
endtry
return ''
endfunction
2019-08-22 15:36:17 +00:00
function! s:StageJump(offset, section, ...) abort
let line = search('^\%(' . a:section . '\)', 'nw')
if !line && a:0
let line = search('^\%(' . a:1 . '\)', 'nw')
endif
if line
exe line
if a:offset
for i in range(a:offset)
call search(s:file_commit_pattern . '\|^$', 'W')
if empty(getline('.')) && a:0 && getline(line('.') + 1) =~# '^\%(' . a:1 . '\)'
call search(s:file_commit_pattern . '\|^$', 'W')
endif
if empty(getline('.'))
return ''
endif
endfor
call s:StageReveal()
else
call s:StageReveal()
+
endif
endif
return ''
endfunction
2019-01-08 10:11:54 +00:00
function! s:StageSeek(info, fallback) abort
let info = a:info
if empty(info.section)
return a:fallback
endif
let line = search('^' . info.section, 'wn')
if !line
2019-08-22 15:36:17 +00:00
for section in get({'Staged': ['Unstaged', 'Untracked'], 'Unstaged': ['Untracked', 'Staged'], 'Untracked': ['Unstaged', 'Staged']}, info.section, [])
2019-01-08 10:11:54 +00:00
let line = search('^' . section, 'wn')
if line
return line + (info.index > 0 ? 1 : 0)
endif
endfor
return 1
endif
let i = 0
while len(getline(line))
let filename = matchstr(getline(line), '^[A-Z?] \zs.*')
if len(filename) &&
\ ((info.filename[-1:-1] ==# '/' && filename[0 : len(info.filename) - 1] ==# info.filename) ||
\ (filename[-1:-1] ==# '/' && filename ==# info.filename[0 : len(filename) - 1]) ||
\ filename ==# info.filename)
if info.offset < 0
return line
else
if getline(line+1) !~# '^@'
exe s:StageInline('show', line)
endif
if getline(line+1) !~# '^@'
return line
endif
let type = info.sigil ==# '-' ? '-' : '+'
let offset = -1
while offset < info.offset
let line += 1
if getline(line) =~# '^@'
let offset = +matchstr(getline(line), type . '\zs\d\+') - 1
elseif getline(line) =~# '^[ ' . type . ']'
let offset += 1
elseif getline(line) !~# '^[ @\+-]'
return line - 1
endif
endwhile
return line
endif
endif
let commit = matchstr(getline(line), '^\%(\%(\x\x\x\)\@!\l\+\s\+\)\=\zs[0-9a-f]\+')
if len(commit) && commit ==# info.commit
return line
endif
if i ==# info.index
let backup = line
endif
let i += getline(line) !~# '^[ @\+-]'
let line += 1
endwhile
return exists('backup') ? backup : line - 1
endfunction
2019-11-16 15:28:42 +00:00
function! s:DoAutocmdChanged(dir) abort
let dir = a:dir is# -2 ? '' : FugitiveGitDir(a:dir)
if empty(dir) || !exists('#User#FugitiveChanged') || exists('g:fugitive_event')
return ''
endif
try
let g:fugitive_event = dir
exe s:DoAutocmd('User FugitiveChanged')
finally
unlet! g:fugitive_event
" Force statusline reload with the buffer's Git dir
let &ro = &ro
endtry
return ''
endfunction
function! s:ReloadStatusBuffer(...) abort
2019-03-08 11:04:56 +00:00
if get(b:, 'fugitive_type', '') !=# 'index'
return ''
endif
2019-01-08 10:11:54 +00:00
let original_lnum = a:0 ? a:1 : line('.')
let info = s:StageInfo(original_lnum)
call fugitive#BufReadStatus()
exe s:StageSeek(info, original_lnum)
normal! 0
return ''
endfunction
2019-11-16 15:28:42 +00:00
function! s:ReloadStatus(...) abort
call s:ExpireStatus(-1)
call s:ReloadStatusBuffer(a:0 ? a:1 : line('.'))
exe s:DoAutocmdChanged(-1)
return ''
endfunction
2019-08-22 15:36:17 +00:00
let s:last_time = reltime()
if !exists('s:last_times')
let s:last_times = {}
endif
function! s:ExpireStatus(bufnr) abort
if a:bufnr == -2
2019-11-16 15:28:42 +00:00
let s:head_cache = {}
2019-08-22 15:36:17 +00:00
let s:last_time = reltime()
return ''
endif
let dir = s:Dir(a:bufnr)
if len(dir)
let s:last_times[s:cpath(dir)] = reltime()
2019-11-16 15:28:42 +00:00
if has_key(s:head_cache, dir)
call remove(s:head_cache, dir)
endif
2019-08-22 15:36:17 +00:00
endif
return ''
endfunction
function! s:ReloadWinStatus(...) abort
if get(b:, 'fugitive_type', '') !=# 'index' || &modified
2018-06-14 10:31:12 +00:00
return
endif
2019-08-22 15:36:17 +00:00
if !exists('b:fugitive_reltime')
2019-11-16 15:28:42 +00:00
exe s:ReloadStatusBuffer()
2019-08-22 15:36:17 +00:00
return
endif
let t = b:fugitive_reltime
if reltimestr(reltime(s:last_time, t)) =~# '-\|\d\{10\}\.' ||
\ reltimestr(reltime(get(s:last_times, s:cpath(s:Dir()), t), t)) =~# '-\|\d\{10\}\.'
2019-11-16 15:28:42 +00:00
exe s:ReloadStatusBuffer()
2019-08-22 15:36:17 +00:00
endif
endfunction
function! s:ReloadTabStatus(...) abort
let mytab = tabpagenr()
let tab = a:0 ? a:1 : mytab
for winnr in range(1, tabpagewinnr(tab, '$'))
if getbufvar(tabpagebuflist(tab)[winnr-1], 'fugitive_type') ==# 'index'
execute 'tabnext '.tab
if winnr != winnr()
execute winnr.'wincmd w'
let restorewinnr = 1
endif
try
call s:ReloadWinStatus()
finally
if exists('restorewinnr')
unlet restorewinnr
wincmd p
2018-06-14 10:31:12 +00:00
endif
2019-08-22 15:36:17 +00:00
execute 'tabnext '.mytab
endtry
endif
endfor
unlet! t:fugitive_reload_status
endfunction
function! fugitive#ReloadStatus(...) abort
2019-11-16 15:28:42 +00:00
call s:ExpireStatus(a:0 ? a:1 : -1)
if a:0 > 1 ? a:2 : 1
2019-08-22 15:36:17 +00:00
let t = reltime()
let t:fugitive_reload_status = t
for tabnr in exists('*settabvar') ? range(1, tabpagenr('$')) : []
call settabvar(tabnr, 'fugitive_reload_status', t)
2018-06-14 10:31:12 +00:00
endfor
2019-08-22 15:36:17 +00:00
call s:ReloadTabStatus()
2019-11-16 15:28:42 +00:00
exe s:DoAutocmdChanged(a:0 ? a:1 : -1)
2019-08-22 15:36:17 +00:00
else
call s:ReloadWinStatus()
endif
2019-11-16 15:28:42 +00:00
return ''
2018-06-14 10:31:12 +00:00
endfunction
2019-11-16 15:28:42 +00:00
function! fugitive#EfmDir(...) abort
let dir = matchstr(a:0 ? a:1 : &errorformat, '\c,%\\&\%(git\|fugitive\)_\=dir=\zs\%(\\.\|[^,]\)*')
let dir = substitute(dir, '%%', '%', 'g')
let dir = substitute(dir, '\\\ze[\,]', '', 'g')
return dir
2019-03-08 11:04:56 +00:00
endfunction
augroup fugitive_status
autocmd!
2019-08-22 15:36:17 +00:00
autocmd BufWritePost * call fugitive#ReloadStatus(-1, 0)
2019-11-16 15:28:42 +00:00
autocmd ShellCmdPost,ShellFilterPost * nested call fugitive#ReloadStatus(-2, 0)
autocmd BufDelete * nested
\ if getbufvar(+expand('<abuf>'), 'buftype') ==# 'terminal' |
\ if !empty(FugitiveGitDir(+expand('<abuf>'))) |
\ call fugitive#ReloadStatus(+expand('<abuf>'), 1) |
\ else |
\ call fugitive#ReloadStatus(-2, 0) |
\ endif |
\ endif
autocmd QuickFixCmdPost make,lmake,[cl]file,[cl]getfile nested
\ call fugitive#ReloadStatus(fugitive#EfmDir(), 1)
2019-03-08 11:04:56 +00:00
if !has('win32')
2019-08-22 15:36:17 +00:00
autocmd FocusGained * call fugitive#ReloadStatus(-2, 0)
2019-03-08 11:04:56 +00:00
endif
2019-08-22 15:36:17 +00:00
autocmd BufEnter index,index.lock
\ call s:ReloadWinStatus()
autocmd TabEnter *
\ if exists('t:fugitive_reload_status') |
\ call s:ReloadTabStatus() |
\ endif
2019-03-08 11:04:56 +00:00
augroup END
2019-01-08 10:11:54 +00:00
function! s:StageInfo(...) abort
let lnum = a:0 ? a:1 : line('.')
2019-03-08 11:04:56 +00:00
let sigil = matchstr(getline(lnum), '^[ @\+-]')
2019-01-08 10:11:54 +00:00
let offset = -1
2019-03-08 11:04:56 +00:00
if len(sigil)
2019-01-08 10:11:54 +00:00
let type = sigil ==# '-' ? '-' : '+'
while lnum > 0 && getline(lnum) !~# '^@'
if getline(lnum) =~# '^[ '.type.']'
let offset += 1
endif
let lnum -= 1
endwhile
let offset += matchstr(getline(lnum), type.'\zs\d\+')
while getline(lnum) =~# '^[ @\+-]'
let lnum -= 1
endwhile
2018-06-14 10:31:12 +00:00
endif
2019-01-08 10:11:54 +00:00
let slnum = lnum + 1
let section = ''
let index = 0
while len(getline(slnum - 1)) && empty(section)
let slnum -= 1
let section = matchstr(getline(slnum), '^\u\l\+\ze.* (\d\+)$')
if empty(section) && getline(slnum) !~# '^[ @\+-]'
let index += 1
endif
2018-06-14 10:31:12 +00:00
endwhile
2019-08-22 15:36:17 +00:00
let text = matchstr(getline(lnum), '^[A-Z?] \zs.*')
2019-01-08 10:11:54 +00:00
return {'section': section,
\ 'heading': getline(slnum),
\ 'sigil': sigil,
\ 'offset': offset,
2019-08-22 15:36:17 +00:00
\ 'filename': text,
\ 'relative': reverse(split(text, ' -> ')),
\ 'paths': map(reverse(split(text, ' -> ')), 's:Tree() . "/" . v:val'),
2019-01-08 10:11:54 +00:00
\ 'commit': matchstr(getline(lnum), '^\%(\%(\x\x\x\)\@!\l\+\s\+\)\=\zs[0-9a-f]\{4,\}\ze '),
\ 'status': matchstr(getline(lnum), '^[A-Z?]\ze \|^\%(\x\x\x\)\@!\l\+\ze [0-9a-f]'),
2019-11-16 15:28:42 +00:00
\ 'sub': get(get(get(b:fugitive_files, section, {}), text, {}), 'sub', ''),
2019-01-08 10:11:54 +00:00
\ 'index': index}
endfunction
2019-03-08 11:04:56 +00:00
function! s:Selection(arg1, ...) abort
if a:arg1 ==# 'n'
let arg1 = line('.')
2019-08-22 15:36:17 +00:00
let arg2 = -v:count
2019-03-08 11:04:56 +00:00
elseif a:arg1 ==# 'v'
let arg1 = line("'<")
let arg2 = line("'>")
else
let arg1 = a:arg1
let arg2 = a:0 ? a:1 : 0
endif
let first = arg1
if arg2 < 0
2019-11-16 15:28:42 +00:00
let last = first - arg2 - 1
2019-03-08 11:04:56 +00:00
elseif arg2 > 0
let last = arg2
else
let last = first
endif
while getline(first) =~# '^$\|^[A-Z][a-z]'
let first += 1
endwhile
if first > last || &filetype !=# 'fugitive'
return []
endif
let flnum = first
while getline(flnum) =~# '^[ @\+-]'
let flnum -= 1
endwhile
let slnum = flnum + 1
let section = ''
let index = 0
while len(getline(slnum - 1)) && empty(section)
let slnum -= 1
let heading = matchstr(getline(slnum), '^\u\l\+.* (\d\+)$')
if empty(heading) && getline(slnum) !~# '^[ @\+-]'
let index += 1
endif
endwhile
let results = []
let template = {
\ 'heading': heading,
\ 'section': matchstr(heading, '^\u\l\+\ze.* (\d\+)$'),
\ 'filename': '',
2019-08-22 15:36:17 +00:00
\ 'relative': [],
2019-03-08 11:04:56 +00:00
\ 'paths': [],
\ 'commit': '',
\ 'status': '',
\ 'patch': 0,
\ 'index': index}
let line = getline(flnum)
let lnum = first - (arg1 == flnum ? 0 : 1)
let root = s:Tree() . '/'
while lnum <= last
if line =~# '^\u\l\+\ze.* (\d\+)$'
let template.heading = getline(lnum)
let template.section = matchstr(template.heading, '^\u\l\+\ze.* (\d\+)$')
let template.index = 0
elseif line =~# '^[ @\+-]'
let template.index -= 1
if !results[-1].patch
let results[-1].patch = lnum
endif
let results[-1].lnum = lnum
elseif line =~# '^[A-Z?] '
let filename = matchstr(line, '^[A-Z?] \zs.*')
call add(results, extend(deepcopy(template), {
\ 'lnum': lnum,
\ 'filename': filename,
2019-08-22 15:36:17 +00:00
\ 'relative': reverse(split(filename, ' -> ')),
2019-03-08 11:04:56 +00:00
\ 'paths': map(reverse(split(filename, ' -> ')), 'root . v:val'),
\ 'status': matchstr(line, '^[A-Z?]'),
\ }))
elseif line =~# '^\x\x\x\+ '
call add(results, extend({
\ 'lnum': lnum,
\ 'commit': matchstr(line, '^\x\x\x\+'),
\ }, template, 'keep'))
elseif line =~# '^\l\+ \x\x\x\+ '
call add(results, extend({
\ 'lnum': lnum,
\ 'commit': matchstr(line, '^\l\+ \zs\x\x\x\+'),
\ 'status': matchstr(line, '^\l\+'),
\ }, template, 'keep'))
endif
let lnum += 1
let template.index += 1
let line = getline(lnum)
endwhile
if len(results) && results[0].patch && arg2 == 0
while getline(results[0].patch) =~# '^[ \+-]'
let results[0].patch -= 1
endwhile
while getline(results[0].lnum + 1) =~# '^[ \+-]'
let results[0].lnum += 1
endwhile
endif
return results
endfunction
function! s:StageArgs(visual) abort
let commits = []
let paths = []
for record in s:Selection(a:visual ? 'v' : 'n')
if len(record.commit)
call add(commits, record.commit)
endif
call extend(paths, record.paths)
endfor
if s:cpath(s:Tree(), getcwd())
call map(paths, 'fugitive#Path(v:val, "./")')
endif
return join(map(commits + paths, 's:fnameescape(v:val)'), ' ')
endfunction
function! s:Do(action, visual) abort
let line = getline('.')
2019-08-22 15:36:17 +00:00
let reload = 0
2019-11-16 15:28:42 +00:00
if !a:visual && !v:count && line =~# '^[A-Z][a-z]'
2019-03-08 11:04:56 +00:00
let header = matchstr(line, '^\S\+\ze:')
if len(header) && exists('*s:Do' . a:action . header . 'Header')
2019-08-22 15:36:17 +00:00
let reload = s:Do{a:action}{header}Header(matchstr(line, ': \zs.*')) > 0
else
let section = matchstr(line, '^\S\+')
if exists('*s:Do' . a:action . section . 'Heading')
let reload = s:Do{a:action}{section}Heading(line) > 0
endif
2019-03-08 11:04:56 +00:00
endif
2019-08-22 15:36:17 +00:00
return reload ? s:ReloadStatus() : ''
2019-03-08 11:04:56 +00:00
endif
let selection = s:Selection(a:visual ? 'v' : 'n')
if empty(selection)
return ''
endif
call filter(selection, 'v:val.section ==# selection[0].section')
let status = 0
let err = ''
try
for record in selection
if exists('*s:Do' . a:action . record.section)
let status = s:Do{a:action}{record.section}(record)
else
continue
endif
if !status
return ''
endif
let reload = reload || (status > 0)
endfor
if status < 0
execute record.lnum + 1
endif
2019-08-22 15:36:17 +00:00
let success = 1
2019-03-08 11:04:56 +00:00
catch /^fugitive:/
2019-08-22 15:36:17 +00:00
return 'echoerr ' . string(v:exception)
2019-03-08 11:04:56 +00:00
finally
if reload
execute s:ReloadStatus()
endif
2019-08-22 15:36:17 +00:00
if exists('success')
call s:StageReveal()
endif
2019-03-08 11:04:56 +00:00
endtry
return ''
endfunction
2019-11-16 15:28:42 +00:00
function! s:StageReveal() abort
exe 'normal! zv'
let begin = line('.')
2019-01-08 10:11:54 +00:00
if getline(begin) =~# '^@'
2019-03-08 11:04:56 +00:00
let end = begin + 1
2019-01-08 10:11:54 +00:00
while getline(end) =~# '^[ \+-]'
let end += 1
endwhile
2019-08-22 15:36:17 +00:00
elseif getline(begin) =~# '^commit '
let end = begin
while end < line('$') && getline(end + 1) !~# '^commit '
let end += 1
endwhile
elseif getline(begin) =~# s:section_pattern
let end = begin
while len(getline(end + 1))
let end += 1
endwhile
endif
if exists('end')
while line('.') > line('w0') + &scrolloff && end > line('w$')
2019-01-08 10:11:54 +00:00
execute "normal! \<C-E>"
endwhile
2018-06-14 10:31:12 +00:00
endif
endfunction
2019-08-22 15:36:17 +00:00
let s:file_pattern = '^[A-Z?] .\|^diff --'
let s:file_commit_pattern = s:file_pattern . '\|^\%(\l\{3,\} \)\=[0-9a-f]\{4,\} '
let s:item_pattern = s:file_commit_pattern . '\|^@@'
function! s:NextHunk(count) abort
if &filetype ==# 'fugitive' && getline('.') =~# s:file_pattern
exe s:StageInline('show')
endif
for i in range(a:count)
if &filetype ==# 'fugitive'
call search(s:file_pattern . '\|^@', 'W')
if getline('.') =~# s:file_pattern
exe s:StageInline('show')
if getline(line('.') + 1) =~# '^@'
+
endif
endif
else
call search('^@@', 'W')
endif
endfor
call s:StageReveal()
return '.'
endfunction
function! s:PreviousHunk(count) abort
for i in range(a:count)
if &filetype ==# 'fugitive'
let lnum = search(s:file_pattern . '\|^@','Wbn')
call s:StageInline('show', lnum)
call search('^? .\|^@','Wb')
else
call search('^@@', 'Wb')
endif
endfor
call s:StageReveal()
return '.'
endfunction
function! s:NextFile(count) abort
for i in range(a:count)
exe s:StageInline('hide')
if !search(s:file_pattern, 'W')
break
endif
endfor
exe s:StageInline('hide')
return '.'
endfunction
function! s:PreviousFile(count) abort
exe s:StageInline('hide')
for i in range(a:count)
if !search(s:file_pattern, 'Wb')
break
endif
exe s:StageInline('hide')
endfor
return '.'
endfunction
function! s:NextItem(count) abort
for i in range(a:count)
if !search(s:item_pattern, 'W') && getline('.') !~# s:item_pattern
call search('^commit ', 'W')
endif
endfor
call s:StageReveal()
return '.'
endfunction
function! s:PreviousItem(count) abort
2018-06-14 10:31:12 +00:00
for i in range(a:count)
2019-08-22 15:36:17 +00:00
if !search(s:item_pattern, 'Wbe') && getline('.') !~# s:item_pattern
call search('^commit ', 'Wbe')
endif
2018-06-14 10:31:12 +00:00
endfor
2019-01-08 10:11:54 +00:00
call s:StageReveal()
2018-06-14 10:31:12 +00:00
return '.'
endfunction
2019-08-22 15:36:17 +00:00
let s:section_pattern = '^[A-Z][a-z][^:]*$'
let s:section_commit_pattern = s:section_pattern . '\|^commit '
function! s:NextSection(count) abort
let orig = line('.')
if getline('.') !~# '^commit '
-
endif
for i in range(a:count)
if !search(s:section_commit_pattern, 'W')
break
endif
endfor
if getline('.') =~# s:section_commit_pattern
call s:StageReveal()
return getline('.') =~# s:section_pattern ? '+' : ':'
2018-06-14 10:31:12 +00:00
else
2019-08-22 15:36:17 +00:00
return orig
endif
endfunction
function! s:PreviousSection(count) abort
let orig = line('.')
if getline('.') !~# '^commit '
-
endif
for i in range(a:count)
if !search(s:section_commit_pattern . '\|\%^', 'bW')
break
endif
endfor
if getline('.') =~# s:section_commit_pattern || line('.') == 1
2019-01-08 10:11:54 +00:00
call s:StageReveal()
2019-08-22 15:36:17 +00:00
return getline('.') =~# s:section_pattern ? '+' : ':'
else
return orig
endif
endfunction
function! s:NextSectionEnd(count) abort
+
if empty(getline('.'))
+
2018-06-14 10:31:12 +00:00
endif
2019-08-22 15:36:17 +00:00
for i in range(a:count)
if !search(s:section_commit_pattern, 'W')
return '$'
endif
endfor
return search('^.', 'Wb')
endfunction
function! s:PreviousSectionEnd(count) abort
let old = line('.')
for i in range(a:count)
if search(s:section_commit_pattern, 'Wb') <= 1
exe old
if i
break
else
return ''
endif
endif
let old = line('.')
endfor
return search('^.', 'Wb')
endfunction
function! s:PatchSearchExpr(reverse) abort
let line = getline('.')
if col('.') ==# 1 && line =~# '^[+-]'
if line =~# '^[+-]\{3\} '
let pattern = '^[+-]\{3\} ' . substitute(escape(strpart(line, 4), '^$.*[]~\'), '^\w/', '\\w/', '') . '$'
else
let pattern = '^[+-]\s*' . escape(substitute(strpart(line, 1), '^\s*\|\s*$', '', ''), '^$.*[]~\') . '\s*$'
endif
if a:reverse
2019-12-01 19:18:45 +00:00
return '?' . escape(pattern, '/?') . "\<CR>"
2019-08-22 15:36:17 +00:00
else
2019-12-01 19:18:45 +00:00
return '/' . escape(pattern, '/') . "\<CR>"
2019-08-22 15:36:17 +00:00
endif
endif
return a:reverse ? '#' : '*'
2018-06-14 10:31:12 +00:00
endfunction
2019-01-08 10:11:54 +00:00
function! s:StageInline(mode, ...) abort
2019-08-22 15:36:17 +00:00
if &filetype !=# 'fugitive'
return ''
endif
2019-01-08 10:11:54 +00:00
let lnum1 = a:0 ? a:1 : line('.')
let lnum = lnum1 + 1
if a:0 > 1 && a:2 == 0
let info = s:StageInfo(lnum - 1)
2019-08-22 15:36:17 +00:00
if empty(info.paths) && len(info.section)
2019-01-08 10:11:54 +00:00
while len(getline(lnum))
let lnum += 1
endwhile
2018-06-14 10:31:12 +00:00
endif
2019-01-08 10:11:54 +00:00
elseif a:0 > 1
let lnum += a:2 - 1
2018-06-14 10:31:12 +00:00
endif
2019-01-08 10:11:54 +00:00
while lnum > lnum1
let lnum -= 1
while lnum > 0 && getline(lnum) =~# '^[ @\+-]'
let lnum -= 1
endwhile
let info = s:StageInfo(lnum)
if !has_key(b:fugitive_diff, info.section)
continue
endif
if getline(lnum + 1) =~# '^[ @\+-]'
let lnum2 = lnum + 1
while getline(lnum2 + 1) =~# '^[ @\+-]'
let lnum2 += 1
endwhile
if a:mode !=# 'show'
setlocal modifiable noreadonly
exe 'silent keepjumps ' . (lnum + 1) . ',' . lnum2 . 'delete _'
call remove(b:fugitive_expanded[info.section], info.filename)
setlocal nomodifiable readonly nomodified
endif
continue
endif
2019-08-22 15:36:17 +00:00
if !has_key(b:fugitive_diff, info.section) || info.status !~# '^[ADMRU]$' || a:mode ==# 'hide'
2019-01-08 10:11:54 +00:00
continue
endif
let mode = ''
let diff = []
let index = 0
let start = -1
for line in b:fugitive_diff[info.section]
if mode ==# 'await' && line[0] ==# '@'
let mode = 'capture'
endif
if mode !=# 'head' && line !~# '^[ @\+-]'
if len(diff)
break
endif
let start = index
let mode = 'head'
2019-08-22 15:36:17 +00:00
elseif mode ==# 'head' && substitute(line, "\t$", '', '') ==# '--- ' . info.relative[-1]
2019-01-08 10:11:54 +00:00
let mode = 'await'
2019-08-22 15:36:17 +00:00
elseif mode ==# 'head' && substitute(line, "\t$", '', '') ==# '+++ ' . info.relative[0]
2019-01-08 10:11:54 +00:00
let mode = 'await'
elseif mode ==# 'capture'
call add(diff, line)
elseif line[0] ==# '@'
let mode = ''
endif
let index += 1
endfor
if len(diff)
setlocal modifiable noreadonly
silent call append(lnum, diff)
let b:fugitive_expanded[info.section][info.filename] = [start, len(diff)]
setlocal nomodifiable readonly nomodified
endif
endwhile
return lnum
2018-06-14 10:31:12 +00:00
endfunction
2019-08-22 15:36:17 +00:00
function! s:NextExpandedHunk(count) abort
2019-03-08 11:04:56 +00:00
for i in range(a:count)
2019-08-22 15:36:17 +00:00
call s:StageInline('show', line('.'), 1)
call search(s:file_pattern . '\|^@','W')
2019-03-08 11:04:56 +00:00
endfor
return '.'
endfunction
2018-06-14 10:31:12 +00:00
function! s:StageDiff(diff) abort
2019-01-08 10:11:54 +00:00
let lnum = line('.')
let info = s:StageInfo(lnum)
let prefix = info.offset > 0 ? '+' . info.offset : ''
2019-11-16 15:28:42 +00:00
if info.sub =~# '^S'
if info.section ==# 'Staged'
return 'Git! diff --no-ext-diff --submodule=log --cached -- ' . info.paths[0]
elseif info.sub =~# '^SC'
return 'Git! diff --no-ext-diff --submodule=log -- ' . info.paths[0]
else
return 'Git! diff --no-ext-diff --submodule=diff -- ' . info.paths[0]
endif
elseif empty(info.paths) && info.section ==# 'Staged'
2018-06-14 10:31:12 +00:00
return 'Git! diff --no-ext-diff --cached'
2019-08-22 15:36:17 +00:00
elseif empty(info.paths)
2018-06-14 10:31:12 +00:00
return 'Git! diff --no-ext-diff'
2019-08-22 15:36:17 +00:00
elseif len(info.paths) > 1
execute 'Gedit' . prefix s:fnameescape(':0:' . info.paths[0])
return a:diff . '! HEAD:'.s:fnameescape(info.paths[1])
2019-01-08 10:11:54 +00:00
elseif info.section ==# 'Staged' && info.sigil ==# '-'
2019-08-22 15:36:17 +00:00
execute 'Gedit' prefix s:fnameescape(':0:'.info.paths[0])
return a:diff . '! :0:%'
2019-01-08 10:11:54 +00:00
elseif info.section ==# 'Staged'
2019-08-22 15:36:17 +00:00
execute 'Gedit' prefix s:fnameescape(':0:'.info.paths[0])
return a:diff . '! @:%'
2019-01-08 10:11:54 +00:00
elseif info.sigil ==# '-'
2019-08-22 15:36:17 +00:00
execute 'Gedit' prefix s:fnameescape(':0:'.info.paths[0])
return a:diff . '! :(top)%'
2018-06-14 10:31:12 +00:00
else
2019-08-22 15:36:17 +00:00
execute 'Gedit' prefix s:fnameescape(':(top)'.info.paths[0])
return a:diff . '!'
2018-06-14 10:31:12 +00:00
endif
endfunction
function! s:StageDiffEdit() abort
2019-01-08 10:11:54 +00:00
let info = s:StageInfo(line('.'))
2019-08-22 15:36:17 +00:00
let arg = (empty(info.paths) ? s:Tree() : info.paths[0])
2019-01-08 10:11:54 +00:00
if info.section ==# 'Staged'
2019-08-22 15:36:17 +00:00
return 'Git! diff --no-ext-diff --cached '.s:fnameescape(arg)
2019-01-08 10:11:54 +00:00
elseif info.status ==# '?'
2019-08-22 15:36:17 +00:00
call s:TreeChomp('add', '--intent-to-add', '--', arg)
2019-03-08 11:04:56 +00:00
return s:ReloadStatus()
2018-06-14 10:31:12 +00:00
else
2019-08-22 15:36:17 +00:00
return 'Git! diff --no-ext-diff '.s:fnameescape(arg)
2018-06-14 10:31:12 +00:00
endif
endfunction
2019-03-08 11:04:56 +00:00
function! s:StageApply(info, reverse, extra) abort
2019-08-22 15:36:17 +00:00
if a:info.status ==# 'R'
call s:throw('fugitive: patching renamed file not yet supported')
endif
2019-03-08 11:04:56 +00:00
let cmd = ['apply', '-p0', '--recount'] + a:extra
2019-01-08 10:11:54 +00:00
let info = a:info
2019-03-08 11:04:56 +00:00
let start = info.patch
let end = info.lnum
2019-01-08 10:11:54 +00:00
let lines = getline(start, end)
if empty(filter(copy(lines), 'v:val =~# "^[+-]"'))
2019-03-08 11:04:56 +00:00
return -1
2019-01-08 10:11:54 +00:00
endif
while getline(end) =~# '^[-+ ]'
let end += 1
if getline(end) =~# '^[' . (a:reverse ? '+' : '-') . ' ]'
call add(lines, ' ' . getline(end)[1:-1])
endif
endwhile
while start > 0 && getline(start) !~# '^@'
let start -= 1
if getline(start) =~# '^[' . (a:reverse ? '+' : '-') . ' ]'
call insert(lines, ' ' . getline(start)[1:-1])
elseif getline(start) =~# '^@'
call insert(lines, getline(start))
endif
endwhile
2019-08-22 15:36:17 +00:00
if start == 0
throw 'fugitive: cold not find hunk'
elseif getline(start) !~# '^@@ '
throw 'fugitive: cannot apply conflict hunk'
2019-01-08 10:11:54 +00:00
endif
let i = b:fugitive_expanded[info.section][info.filename][0]
let head = []
while get(b:fugitive_diff[info.section], i, '@') !~# '^@'
call add(head, b:fugitive_diff[info.section][i])
let i += 1
endwhile
call extend(lines, head, 'keep')
let temp = tempname()
call writefile(lines, temp)
if a:reverse
call add(cmd, '--reverse')
endif
call extend(cmd, ['--', temp])
2019-08-22 15:36:17 +00:00
let [output, exec_error] = s:ChompError(cmd)
if !exec_error
2019-03-08 11:04:56 +00:00
return 1
endif
call s:throw(output)
2019-01-08 10:11:54 +00:00
endfunction
2019-08-22 15:36:17 +00:00
function! s:StageDelete(lnum1, lnum2, count) abort
let restore = []
let err = ''
try
for info in s:Selection(a:lnum1, a:lnum2)
if empty(info.paths)
continue
endif
2019-11-16 15:28:42 +00:00
let sub = get(get(get(b:fugitive_files, info.section, {}), info.filename, {}), 'sub')
if sub =~# '^S'
if info.status ==# 'A'
continue
endif
if info.section ==# 'Staged'
call s:TreeChomp('reset', '--', info.paths[0])
endif
if info.status =~# '[MD]'
call s:TreeChomp('submodule', 'update', '--', info.paths[0])
call add(restore, ':Git -C ' . info.relative[0] . ' checkout -')
endif
2019-08-22 15:36:17 +00:00
continue
endif
2019-11-16 15:28:42 +00:00
if info.status ==# 'D'
let undo = 'Gremove'
elseif info.paths[0] =~# '/$'
let err .= '|echoerr ' . string('fugitive: will not delete directory ' . string(info.relative[0]))
break
else
let undo = 'Gread ' . s:TreeChomp('hash-object', '-w', '--', info.paths[0])[0:10]
endif
2019-08-22 15:36:17 +00:00
if info.patch
call s:StageApply(info, 1, info.section ==# 'Staged' ? ['--index'] : [])
elseif info.status ==# '?'
call s:TreeChomp('clean', '-f', '--', info.paths[0])
elseif a:count == 2
call s:TreeChomp('checkout', '--ours', '--', info.paths[0])
elseif a:count == 3
call s:TreeChomp('checkout', '--theirs', '--', info.paths[0])
elseif info.status =~# '[ADU]' &&
2019-11-16 15:28:42 +00:00
\ get(b:fugitive_files[info.section ==# 'Staged' ? 'Unstaged' : 'Staged'], info.filename, {'status': ''}).status =~# '[AU]'
2019-08-22 15:36:17 +00:00
call s:TreeChomp('checkout', info.section ==# 'Staged' ? '--ours' : '--theirs', '--', info.paths[0])
elseif info.status ==# 'U'
call s:TreeChomp('rm', '--', info.paths[0])
elseif info.status ==# 'A'
call s:TreeChomp('rm', '-f', '--', info.paths[0])
elseif info.section ==# 'Unstaged'
call s:TreeChomp('checkout', '--', info.paths[0])
else
call s:TreeChomp('checkout', 'HEAD^{}', '--', info.paths[0])
endif
2019-11-16 15:28:42 +00:00
call add(restore, ':Gsplit ' . s:fnameescape(info.relative[0]) . '|' . undo)
2019-08-22 15:36:17 +00:00
endfor
catch /^fugitive:/
2019-11-16 15:28:42 +00:00
let err .= '|echoerr ' . string(v:exception)
2019-08-22 15:36:17 +00:00
endtry
if empty(restore)
return err[1:-1]
2019-01-08 10:11:54 +00:00
endif
2019-08-22 15:36:17 +00:00
exe s:ReloadStatus()
call s:StageReveal()
2019-11-16 15:28:42 +00:00
if len(restore)
return 'checktime|redraw|echomsg ' . string('To restore, ' . join(restore, '|')) . err
else
return 'checktime|redraw' . err
endif
2019-08-22 15:36:17 +00:00
endfunction
function! s:StageIgnore(lnum1, lnum2, count) abort
let paths = []
for info in s:Selection(a:lnum1, a:lnum2)
call extend(paths, info.relative)
endfor
call map(paths, '"/" . v:val')
exe 'Gsplit' (a:count ? '.gitignore' : '.git/info/exclude')
let last = line('$')
if last == 1 && empty(getline(1))
call setline(last, paths)
2019-01-08 10:11:54 +00:00
else
2019-08-22 15:36:17 +00:00
call append(last, paths)
exe last + 1
2019-01-08 10:11:54 +00:00
endif
2019-08-22 15:36:17 +00:00
return ''
2019-01-08 10:11:54 +00:00
endfunction
2019-03-08 11:04:56 +00:00
function! s:DoToggleHeadHeader(value) abort
exe 'edit' s:fnameescape(s:Dir())
call search('\C^index$', 'wc')
endfunction
2019-08-22 15:36:17 +00:00
function! s:DoStageUnpushedHeading(heading) abort
2019-03-08 11:04:56 +00:00
let remote = matchstr(a:heading, 'to \zs[^/]\+\ze/')
if empty(remote)
let remote = '.'
2018-06-14 10:31:12 +00:00
endif
2019-03-08 11:04:56 +00:00
let branch = matchstr(a:heading, 'to \%([^/]\+/\)\=\zs\S\+')
2020-01-29 02:07:36 +00:00
call feedkeys(':Git push ' . remote . ' ' . 'HEAD:' . 'refs/heads/' . branch)
2019-03-08 11:04:56 +00:00
endfunction
2019-08-22 15:36:17 +00:00
function! s:DoToggleUnpushedHeading(heading) abort
return s:DoStageUnpushedHeading(a:heading)
endfunction
function! s:DoStageUnpushed(record) abort
2019-03-08 11:04:56 +00:00
let remote = matchstr(a:record.heading, 'to \zs[^/]\+\ze/')
if empty(remote)
let remote = '.'
endif
let branch = matchstr(a:record.heading, 'to \%([^/]\+/\)\=\zs\S\+')
2020-01-29 02:07:36 +00:00
call feedkeys(':Git push ' . remote . ' ' . a:record.commit . ':' . 'refs/heads/' . branch)
2019-03-08 11:04:56 +00:00
endfunction
2019-08-22 15:36:17 +00:00
function! s:DoToggleUnpushed(record) abort
return s:DoStageUnpushed(a:record)
endfunction
function! s:DoUnstageUnpulledHeading(heading) abort
2020-01-29 02:07:36 +00:00
call feedkeys(':Git rebase')
2019-03-08 11:04:56 +00:00
endfunction
2019-08-22 15:36:17 +00:00
function! s:DoToggleUnpulledHeading(heading) abort
call s:DoUnstageUnpulledHeading(a:heading)
endfunction
function! s:DoUnstageUnpulled(record) abort
2020-01-29 02:07:36 +00:00
call feedkeys(':Git rebase ' . a:record.commit)
2019-03-08 11:04:56 +00:00
endfunction
2019-08-22 15:36:17 +00:00
function! s:DoToggleUnpulled(record) abort
call s:DoUnstageUnpulled(a:record)
endfunction
function! s:DoUnstageUnpushed(record) abort
2020-01-29 02:07:36 +00:00
call feedkeys(':Git rebase --autosquash ' . a:record.commit . '^')
2019-08-22 15:36:17 +00:00
endfunction
2019-03-08 11:04:56 +00:00
function! s:DoToggleStagedHeading(...) abort
call s:TreeChomp('reset', '-q')
return 1
endfunction
function! s:DoUnstageStagedHeading(heading) abort
return s:DoToggleStagedHeading(a:heading)
2019-01-08 10:11:54 +00:00
endfunction
2019-03-08 11:04:56 +00:00
function! s:DoToggleUnstagedHeading(...) abort
call s:TreeChomp('add', '-u')
return 1
endfunction
function! s:DoStageUnstagedHeading(heading) abort
return s:DoToggleUnstagedHeading(a:heading)
endfunction
2019-08-22 15:36:17 +00:00
function! s:DoToggleUntrackedHeading(...) abort
call s:TreeChomp('add', '.')
return 1
endfunction
function! s:DoStageUntrackedHeading(heading) abort
return s:DoToggleUntrackedHeading(a:heading)
endfunction
2019-03-08 11:04:56 +00:00
function! s:DoToggleStaged(record) abort
if a:record.patch
return s:StageApply(a:record, 1, ['--cached'])
2019-01-08 10:11:54 +00:00
else
2019-03-08 11:04:56 +00:00
call s:TreeChomp(['reset', '-q', '--'] + a:record.paths)
return 1
endif
endfunction
function! s:DoUnstageStaged(record) abort
return s:DoToggleStaged(a:record)
endfunction
function! s:DoToggleUnstaged(record) abort
2019-08-22 15:36:17 +00:00
if a:record.patch && a:record.status !=# 'A'
2019-03-08 11:04:56 +00:00
return s:StageApply(a:record, 0, ['--cached'])
else
call s:TreeChomp(['add', '-A', '--'] + a:record.paths)
return 1
endif
endfunction
function! s:DoStageUnstaged(record) abort
return s:DoToggleUnstaged(a:record)
endfunction
function! s:DoUnstageUnstaged(record) abort
if a:record.status ==# 'A'
call s:TreeChomp(['reset', '-q', '--'] + a:record.paths)
return 1
else
return -1
2019-01-08 10:11:54 +00:00
endif
2018-06-14 10:31:12 +00:00
endfunction
2019-08-22 15:36:17 +00:00
function! s:DoToggleUntracked(record) abort
call s:TreeChomp(['add', '--'] + a:record.paths)
return 1
endfunction
function! s:DoStageUntracked(record) abort
return s:DoToggleUntracked(a:record)
endfunction
2018-06-14 10:31:12 +00:00
function! s:StagePatch(lnum1,lnum2) abort
let add = []
let reset = []
2019-08-22 15:36:17 +00:00
let intend = []
2018-06-14 10:31:12 +00:00
for lnum in range(a:lnum1,a:lnum2)
2019-01-08 10:11:54 +00:00
let info = s:StageInfo(lnum)
2019-08-22 15:36:17 +00:00
if empty(info.paths) && info.section ==# 'Staged'
2018-06-14 10:31:12 +00:00
return 'Git reset --patch'
2019-08-22 15:36:17 +00:00
elseif empty(info.paths) && info.section ==# 'Unstaged'
2018-06-14 10:31:12 +00:00
return 'Git add --patch'
2019-08-22 15:36:17 +00:00
elseif empty(info.paths) && info.section ==# 'Untracked'
return 'Git add --interactive'
elseif empty(info.paths)
2018-06-14 10:31:12 +00:00
continue
endif
execute lnum
2019-08-22 15:36:17 +00:00
if info.section ==# 'Staged'
let reset += info.relative
elseif info.section ==# 'Untracked'
let intend += info.paths
2019-01-08 10:11:54 +00:00
elseif info.status !~# '^D'
2019-08-22 15:36:17 +00:00
let add += info.relative
2018-06-14 10:31:12 +00:00
endif
endfor
try
2019-08-22 15:36:17 +00:00
if !empty(intend)
call s:TreeChomp(['add', '--intent-to-add', '--'] + intend)
endif
2018-06-14 10:31:12 +00:00
if !empty(add)
2019-08-22 15:36:17 +00:00
execute "Git add --patch -- ".join(map(add,'s:fnameescape(v:val)'))
2018-06-14 10:31:12 +00:00
endif
if !empty(reset)
2019-08-22 15:36:17 +00:00
execute "Git reset --patch -- ".join(map(reset,'s:fnameescape(v:val)'))
2018-06-14 10:31:12 +00:00
endif
catch /^fugitive:/
2019-08-22 15:36:17 +00:00
return 'echoerr ' . string(v:exception)
2018-06-14 10:31:12 +00:00
endtry
2019-01-08 10:11:54 +00:00
return s:ReloadStatus()
2018-06-14 10:31:12 +00:00
endfunction
2019-08-22 15:36:17 +00:00
" Section: :Gcommit, :Grevert
2018-06-14 10:31:12 +00:00
2019-08-22 15:36:17 +00:00
function! s:CommitInteractive(line1, line2, range, bang, mods, args, patch) abort
let status = s:StatusCommand(a:line1, a:line2, a:range, a:line2, a:bang, a:mods, '', '', [])
let status = len(status) ? status . '|' : ''
if a:patch
return status . 'if search("^Unstaged")|exe "normal >"|exe "+"|endif'
else
return status . 'if search("^Untracked\\|^Unstaged")|exe "+"|endif'
endif
endfunction
2018-06-14 10:31:12 +00:00
2020-01-29 02:07:36 +00:00
function! s:CommitSubcommand(line1, line2, range, bang, mods, args) abort
let argv = copy(a:args)
let i = 0
while get(argv, i, '--') !=# '--'
if argv[i] =~# '^-[apzsneiovq].'
call insert(argv, argv[i][0:1])
let argv[i+1] = '-' . argv[i+1][2:-1]
2018-06-14 10:31:12 +00:00
else
2020-01-29 02:07:36 +00:00
let i += 1
2018-06-14 10:31:12 +00:00
endif
2020-01-29 02:07:36 +00:00
endwhile
if s:HasOpt(argv, '-i', '--interactive')
return s:CommitInteractive(a:line1, a:line2, a:range, a:bang, a:mods, argv, 0)
elseif s:HasOpt(argv, '-p', '--patch')
return s:CommitInteractive(a:line1, a:line2, a:range, a:bang, a:mods, argv, 1)
else
return {}
endif
2018-06-14 10:31:12 +00:00
endfunction
2019-08-22 15:36:17 +00:00
function! s:RevertSubcommand(line1, line2, range, bang, mods, args) abort
2020-01-29 02:07:36 +00:00
return {'args': ['revert', '--edit'] + a:args}
2019-08-22 15:36:17 +00:00
endfunction
2019-11-16 15:28:42 +00:00
function! fugitive#CommitComplete(A, L, P) abort
2018-07-30 21:18:16 +00:00
if a:A =~# '^--fixup=\|^--squash='
2019-08-22 15:36:17 +00:00
let commits = s:LinesError(['log', '--pretty=format:%s', '@{upstream}..'])[0]
let pre = matchstr(a:A, '^--\w*=''\=') . ':/^'
if pre =~# "'"
call map(commits, 'pre . string(tr(v:val, "|\"^$*[]", "......."))[1:-1]')
call filter(commits, 'strpart(v:val, 0, strlen(a:A)) ==# a:A')
return commits
else
return s:FilterEscape(map(commits, 'pre . tr(v:val, "\\ !^$*?[]()''\"`&;<>|#", "....................")'), a:A)
endif
2018-06-14 10:31:12 +00:00
else
2019-08-22 15:36:17 +00:00
return s:CompleteSub('commit', a:A, a:L, a:P, function('fugitive#CompletePath'))
2018-06-14 10:31:12 +00:00
endif
2018-07-30 21:18:16 +00:00
return []
2018-06-14 10:31:12 +00:00
endfunction
2019-11-16 15:28:42 +00:00
function! fugitive#RevertComplete(A, L, P) abort
2019-08-22 15:36:17 +00:00
return s:CompleteSub('revert', a:A, a:L, a:P, function('s:CompleteRevision'))
endfunction
2018-08-25 16:13:42 +00:00
" Section: :Gmerge, :Grebase, :Gpull
2018-06-14 10:31:12 +00:00
2019-11-16 15:28:42 +00:00
function! fugitive#MergeComplete(A, L, P) abort
2019-08-22 15:36:17 +00:00
return s:CompleteSub('merge', a:A, a:L, a:P, function('s:CompleteRevision'))
endfunction
2018-06-14 10:31:12 +00:00
2019-11-16 15:28:42 +00:00
function! fugitive#RebaseComplete(A, L, P) abort
2019-08-22 15:36:17 +00:00
return s:CompleteSub('rebase', a:A, a:L, a:P, function('s:CompleteRevision'))
2018-06-14 10:31:12 +00:00
endfunction
2019-11-16 15:28:42 +00:00
function! fugitive#PullComplete(A, L, P) abort
2019-08-22 15:36:17 +00:00
return s:CompleteSub('pull', a:A, a:L, a:P, function('s:CompleteRemote'))
2018-06-14 10:31:12 +00:00
endfunction
2019-08-22 15:36:17 +00:00
function! s:MergeSubcommand(line1, line2, range, bang, mods, args) abort
2020-01-29 02:07:36 +00:00
let dir = s:Dir()
if empty(a:args) && (
\ filereadable(fugitive#Find('.git/MERGE_MSG', dir)) ||
\ isdirectory(fugitive#Find('.git/rebase-apply', dir)) ||
\ !empty(s:TreeChomp(dir, 'diff-files', '--diff-filter=U')))
return 'echohl WarningMsg|echo ":Git merge for loading conflicts is deprecated in favor of :Git mergetool"|echohl NONE|silent Git' . (a:bang ? '!' : '') . ' mergetool'
endif
return {}
2019-08-22 15:36:17 +00:00
endfunction
function! s:RebaseSubcommand(line1, line2, range, bang, mods, args) abort
2020-01-29 02:07:36 +00:00
if s:HasOpt(a:args, '--autosquash') && !s:HasOpt(a:args, '-i', '--interactive')
return {'env': {'GIT_SEQUENCE_EDITOR': 'true'}, 'args': ['rebase', '--interactive'] + a:args}
endif
return {}
2019-12-12 22:01:41 +00:00
endfunction
2020-01-07 12:45:07 +00:00
" Section: :Git difftool, :Git mergetool
function! s:ToolItems(state, from, to, offsets, text, ...) abort
let items = []
for i in range(len(a:state.diff))
let diff = a:state.diff[i]
let path = (i == len(a:state.diff) - 1) ? a:to : a:from
if empty(path)
return []
endif
let item = {
\ 'valid': a:0 ? a:1 : 1,
\ 'filename': diff.filename . FugitiveVimPath(path),
\ 'lnum': matchstr(get(a:offsets, i), '\d\+'),
\ 'text': a:text}
if len(get(diff, 'module', ''))
let item.module = diff.module . path
endif
call add(items, item)
endfor
let diff = items[0:-2]
let items[-1].context = {'diff': items[0:-2]}
return [items[-1]]
endfunction
function! s:ToolToFrom(str) abort
if a:str =~# ' => '
let str = a:str =~# '{.* => .*}' ? a:str : '{' . a:str . '}'
return [substitute(str, '{.* => \(.*\)}', '\1', ''),
\ substitute(str, '{\(.*\) => .*}', '\1', '')]
else
return [a:str, a:str]
endif
endfunction
function! s:ToolParse(state, line) abort
if type(a:line) !=# type('') || a:state.mode ==# 'hunk' && a:line =~# '^[ +-]'
return []
elseif a:line =~# '^diff '
let a:state.mode = 'diffhead'
let a:state.from = ''
let a:state.to = ''
elseif a:state.mode ==# 'diffhead' && a:line =~# '^--- [^/]'
let a:state.from = a:line[4:-1]
let a:state.to = a:state.from
elseif a:state.mode ==# 'diffhead' && a:line =~# '^+++ [^/]'
let a:state.to = a:line[4:-1]
if empty(get(a:state, 'from', ''))
let a:state.from = a:state.to
endif
elseif a:line[0] ==# '@'
let a:state.mode = 'hunk'
if has_key(a:state, 'from')
let offsets = split(matchstr(a:line, '^@\+ \zs[-+0-9, ]\+\ze @'), ' ')
return s:ToolItems(a:state, a:state.from, a:state.to, offsets, matchstr(a:line, ' @@\+ \zs.*'))
endif
2020-01-29 02:07:36 +00:00
elseif a:line =~# '^\* Unmerged path .'
let file = a:line[16:-1]
return s:ToolItems(a:state, file, file, [], '')
2020-01-07 12:45:07 +00:00
elseif a:line =~# '^[A-Z]\d*\t.\|^:.*\t.'
" --raw, --name-status
let [status; files] = split(a:line, "\t")
return s:ToolItems(a:state, files[0], files[-1], [], a:state.name_only ? '' : status)
elseif a:line =~# '^ \S.* |'
" --stat
let [_, to, changes; __] = matchlist(a:line, '^ \(.\{-\}\) \+|\zs \(.*\)$')
let [to, from] = s:ToolToFrom(to)
return s:ToolItems(a:state, from, to, [], changes)
elseif a:line =~# '^ *\([0-9.]\+%\) .'
" --dirstat
let [_, changes, to; __] = matchlist(a:line, '^ *\([0-9.]\+%\) \(.*\)')
return s:ToolItems(a:state, to, to, [], changes)
elseif a:line =~# '^\(\d\+\|-\)\t\(\d\+\|-\)\t.'
" --numstat
let [_, add, remove, to; __] = matchlist(a:line, '^\(\d\+\|-\)\t\(\d\+\|-\)\t\(.*\)')
let [to, from] = s:ToolToFrom(to)
return s:ToolItems(a:state, from, to, [], add ==# '-' ? 'Binary file' : '+' . add . ' -' . remove, add !=# '-')
elseif a:state.mode !=# 'diffhead' && a:state.mode !=# 'hunk' && len(a:line) || a:line =~# '^git: \|^usage: \|^error: \|^fatal: '
return [{'text': a:line}]
endif
return []
endfunction
function! s:ToolStream(dir, line1, line2, range, bang, mods, args, state, title) abort
let i = 0
let argv = copy(a:args)
let prompt = 1
let state = a:state
while i < len(argv)
let match = matchlist(argv[i], '^\(-[a-zABDFH-KN-RT-Z]\)\ze\(.*\)')
if len(match) && len(match[2])
call insert(argv, match[1])
let argv[i+1] = '-' . match[2]
continue
endif
let arg = argv[i]
if arg =~# '^-t$\|^--tool=\|^--tool-help$\|^--help$'
2020-01-29 02:07:36 +00:00
return {}
2020-01-07 12:45:07 +00:00
elseif arg =~# '^-y$\|^--no-prompt$'
let prompt = 0
call remove(argv, i)
continue
elseif arg ==# '--prompt'
let prompt = 1
call remove(argv, i)
continue
elseif arg =~# '^--\%(no-\)\=\(symlinks\|trust-exit-code\|gui\)$'
call remove(argv, i)
continue
elseif arg ==# '--'
break
endif
let i += 1
endwhile
let a:state.mode = 'init'
let a:state.from = ''
let a:state.to = ''
let exec = s:UserCommandList(a:dir) + ['--no-pager', '-c', 'diff.context=0', 'diff', '--no-ext-diff', '--no-color', '--no-prefix'] + argv
if prompt
return s:QuickfixStream(a:line2, 'difftool', a:title, exec, !a:bang, s:function('s:ToolParse'), a:state)
else
let filename = ''
let cmd = []
let tabnr = tabpagenr() + 1
for line in split(s:SystemError(s:shellesc(exec))[0], "\n")
for item in s:ToolParse(a:state, line)
if len(get(item, 'filename', '')) && item.filename != filename
call add(cmd, 'tabedit ' . s:fnameescape(item.filename))
for i in reverse(range(len(get(item.context, 'diff', []))))
call add(cmd, (i ? 'rightbelow' : 'leftabove') . ' vert Gdiffsplit! ' . s:fnameescape(item.context.diff[i].filename))
endfor
call add(cmd, 'wincmd =')
let filename = item.filename
endif
endfor
endfor
return join(cmd, '|') . (empty(cmd) ? '' : '|' . tabnr . 'tabnext')
endif
endfunction
function! s:MergetoolSubcommand(line1, line2, range, bang, mods, args) abort
let dir = s:Dir()
let i = 0
let argv = copy(a:args)
let prompt = 1
let title = ':Git mergetool' . (len(a:args) ? ' ' . s:fnameescape(a:args) : '')
let cmd = ['diff', '--diff-filter=U']
let state = {'name_only': 0}
let state.diff = [{'prefix': ':2:', 'module': ':2:'}, {'prefix': ':3:', 'module': ':3:'}, {'prefix': ':(top)'}]
call map(state.diff, 'extend(v:val, {"filename": fugitive#Find(v:val.prefix, dir)})')
return s:ToolStream(dir, a:line1, a:line2, a:range, a:bang, a:mods, ['--diff-filter=U'] + a:args, state, title)
endfunction
function! s:DifftoolSubcommand(line1, line2, range, bang, mods, args) abort
let dir = s:Dir()
let i = 0
let argv = copy(a:args)
let commits = []
let cached = 0
let reverse = 1
let prompt = 1
let state = {'name_only': 0}
let merge_base_against = {}
let dash = (index(argv, '--') > i ? ['--'] : [])
while i < len(argv)
let match = matchlist(argv[i], '^\(-[a-zABDFH-KN-RT-Z]\)\ze\(.*\)')
if len(match) && len(match[2])
call insert(argv, match[1])
let argv[i+1] = '-' . match[2]
continue
endif
let arg = argv[i]
if arg ==# '--cached'
let cached = 1
elseif arg ==# '-R'
let reverse = 1
elseif arg ==# '--name-only'
let state.name_only = 1
let argv[0] = '--name-status'
elseif arg ==# '--'
break
elseif arg !~# '^-\|^\.\.\=\%(/\|$\)'
let parsed = s:LinesError(['rev-parse', '--revs-only', substitute(arg, ':.*', '', '')] + dash)[0]
call map(parsed, '{"uninteresting": v:val =~# "^\\^", "prefix": substitute(v:val, "^\\^", "", "") . ":"}')
let merge_base_against = {}
if arg =~# '\.\.\.' && len(parsed) > 2
let display = map(split(arg, '\.\.\.', 1), 'empty(v:val) ? "@" : v:val')
if len(display) == 2
let parsed[0].module = display[1] . ':'
let parsed[1].module = display[0] . ':'
endif
let parsed[2].module = arg . ':'
if empty(commits)
let merge_base_against = parsed[0]
let parsed = [parsed[2]]
endif
elseif arg =~# '\.\.' && len(parsed) == 2
let display = map(split(arg, '\.\.', 1), 'empty(v:val) ? "@" : v:val')
if len(display) == 2
let parsed[0].module = display[0] . ':'
let parsed[1].module = display[1] . ':'
endif
elseif len(parsed) == 1
let parsed[0].module = arg . ':'
endif
call extend(commits, parsed)
endif
let i += 1
endwhile
let title = ':Git difftool' . (len(a:args) ? ' ' . s:fnameescape(a:args) : '')
if len(merge_base_against)
call add(commits, merge_base_against)
endif
let commits = filter(copy(commits), 'v:val.uninteresting') + filter(commits, '!v:val.uninteresting')
if cached
if empty(commits)
call add(commits, {'prefix': '@:', 'module': '@:'})
endif
call add(commits, {'prefix': ':0:', 'module': ':0:'})
elseif len(commits) < 2
call add(commits, {'prefix': ':(top)'})
if len(commits) < 2
call insert(commits, {'prefix': ':0:', 'module': ':0:'})
endif
endif
if reverse
let commits = [commits[-1]] + repeat([commits[0]], len(commits) - 1)
call reverse(commits)
endif
if len(commits) > 2
call add(commits, remove(commits, 0))
endif
call map(commits, 'extend(v:val, {"filename": fugitive#Find(v:val.prefix, dir)})')
let state.diff = commits
return s:ToolStream(dir, a:line1, a:line2, a:range, a:bang, a:mods, argv, state, title)
endfunction
2018-08-25 16:13:42 +00:00
" Section: :Ggrep, :Glog
2018-06-14 10:31:12 +00:00
if !exists('g:fugitive_summary_format')
let g:fugitive_summary_format = '%s'
endif
2019-11-16 15:28:42 +00:00
function! fugitive#GrepComplete(A, L, P) abort
2019-08-22 15:36:17 +00:00
return s:CompleteSub('grep', a:A, a:L, a:P)
endfunction
2019-11-16 15:28:42 +00:00
function! fugitive#LogComplete(A, L, P) abort
2019-08-22 15:36:17 +00:00
return s:CompleteSub('log', a:A, a:L, a:P)
endfunction
function! s:GrepParseLine(prefix, name_only, dir, line) abort
let entry = {'valid': 1}
let match = matchlist(a:line, '^\(.\{-\}\):\(\d\+\):\(\d\+:\)\=\(.*\)$')
if len(match)
let entry.module = match[1]
let entry.lnum = +match[2]
let entry.col = +match[3]
let entry.text = match[4]
elseif a:line =~# '^git: \|^usage: \|^error: \|^fatal: '
return {'text': a:line}
2018-07-30 21:18:16 +00:00
else
2019-08-22 15:36:17 +00:00
let entry.module = matchstr(a:line, '\CBinary file \zs.*\ze matches$')
if len(entry.module)
let entry.text = 'Binary file'
let entry.valid = 0
endif
endif
if empty(entry.module) && a:name_only
let entry.module = a:line
endif
if empty(entry.module)
return {'text': a:line}
endif
if entry.module !~# ':'
let entry.filename = a:prefix . entry.module
else
let entry.filename = fugitive#Find(entry.module, a:dir)
endif
return entry
endfunction
function! s:GrepSubcommand(line1, line2, range, bang, mods, args) abort
let dir = s:Dir()
exe s:DirCheck(dir)
let listnr = a:line1 == 0 ? a:line1 : a:line2
let cmd = ['--no-pager', 'grep', '-n', '--no-color', '--full-name']
if fugitive#GitVersion(2, 19)
call add(cmd, '--column')
endif
let tree = s:Tree(dir)
2020-01-29 02:07:36 +00:00
let args = a:args
2019-08-22 15:36:17 +00:00
let name_only = s:HasOpt(args, '-l', '--files-with-matches', '--name-only', '-L', '--files-without-match')
let title = [listnr < 0 ? ':Ggrep' : ':Glgrep'] + args
if listnr > 0
exe listnr 'wincmd w'
else
call s:BlurStatus()
endif
redraw
call s:QuickfixCreate(listnr, {'title': (listnr < 0 ? ':Ggrep ' : ':Glgrep ') . s:fnameescape(args)})
let tempfile = tempname()
2019-11-16 15:28:42 +00:00
let event = listnr < 0 ? 'grep-fugitive' : 'lgrep-fugitive'
silent exe s:DoAutocmd('QuickFixCmdPre ' . event)
2020-01-29 02:07:36 +00:00
let prefix = FugitiveVimPath(s:HasOpt(args, '--cached') || empty(tree) ?
\ 'fugitive://' . dir . '//0/' :
\ s:cpath(getcwd(), tree) ? '' : tree . '/')
2019-08-22 15:36:17 +00:00
exe '!' . escape(s:UserCommand(dir, cmd + args), '%#!')
\ printf(&shellpipe . (&shellpipe =~# '%s' ? '' : ' %s'), s:shellesc(tempfile))
let list = map(readfile(tempfile), 's:GrepParseLine(prefix, name_only, dir, v:val)')
call s:QuickfixSet(listnr, list, 'a')
2019-11-16 15:28:42 +00:00
silent exe s:DoAutocmd('QuickFixCmdPost ' . event)
2019-08-22 15:36:17 +00:00
if !has('gui_running')
redraw
endif
if !a:bang && !empty(list)
2020-01-29 02:07:36 +00:00
return (listnr < 0 ? 'c' : 'l').'first'
2019-08-22 15:36:17 +00:00
else
2020-01-29 02:07:36 +00:00
return ''
2018-07-30 21:18:16 +00:00
endif
endfunction
2020-01-07 12:45:07 +00:00
let s:log_diff_context = '{"filename": fugitive#Find(v:val . from, a:dir), "lnum": get(offsets, v:key), "module": strpart(v:val, 0, len(a:state.base_module)) . from}'
function! s:LogFlushQueue(state, dir) abort
2019-08-22 15:36:17 +00:00
let queue = remove(a:state, 'queue')
2020-01-07 12:45:07 +00:00
if a:state.child_found && get(a:state, 'ignore_summary')
2019-08-22 15:36:17 +00:00
call remove(queue, 0)
2020-01-07 12:45:07 +00:00
elseif len(queue) && len(a:state.target) && len(get(a:state, 'parents', []))
let from = substitute(a:state.target, '^/', ':', '')
let offsets = []
let queue[0].context.diff = map(copy(a:state.parents), s:log_diff_context)
2019-08-22 15:36:17 +00:00
endif
if len(queue) && queue[-1] ==# {'text': ''}
call remove(queue, -1)
endif
return queue
endfunction
2018-06-14 10:31:12 +00:00
2019-08-22 15:36:17 +00:00
function! s:LogParse(state, dir, line) abort
2020-01-07 12:45:07 +00:00
if a:state.mode ==# 'hunk' && a:line =~# '^[-+ ]'
2019-08-22 15:36:17 +00:00
return []
endif
let list = matchlist(a:line, '^\%(fugitive \(.\{-\}\)\t\|commit \|From \)\=\(\x\{40,\}\)\%( \(.*\)\)\=$')
if len(list)
2020-01-07 12:45:07 +00:00
let queue = s:LogFlushQueue(a:state, a:dir)
let a:state.mode = 'commit'
2019-08-22 15:36:17 +00:00
let a:state.base = 'fugitive://' . a:dir . '//' . list[2]
2020-01-07 12:45:07 +00:00
if len(list[1])
let [a:state.base_module; a:state.parents] = split(list[1], ' ')
else
let a:state.base_module = list[2]
let a:state.parents = []
2019-08-22 15:36:17 +00:00
endif
2020-01-07 12:45:07 +00:00
let a:state.message = list[3]
let a:state.from = ''
let a:state.to = ''
let context = {}
2019-08-22 15:36:17 +00:00
let a:state.queue = [{
\ 'valid': 1,
2020-01-07 12:45:07 +00:00
\ 'context': context,
2019-08-22 15:36:17 +00:00
\ 'filename': a:state.base . a:state.target,
\ 'module': a:state.base_module . substitute(a:state.target, '^/', ':', ''),
\ 'text': a:state.message}]
let a:state.child_found = 0
return queue
elseif type(a:line) == type(0)
2020-01-07 12:45:07 +00:00
return s:LogFlushQueue(a:state, a:dir)
2019-08-22 15:36:17 +00:00
elseif a:line =~# '^diff'
2020-01-07 12:45:07 +00:00
let a:state.mode = 'diffhead'
let a:state.from = ''
let a:state.to = ''
elseif a:state.mode ==# 'diffhead' && a:line =~# '^--- \w/'
let a:state.from = a:line[6:-1]
let a:state.to = a:state.from
elseif a:state.mode ==# 'diffhead' && a:line =~# '^+++ \w/'
let a:state.to = a:line[6:-1]
if empty(get(a:state, 'from', ''))
let a:state.from = a:state.to
endif
elseif a:line =~# '^@@[^@]*+\d' && len(get(a:state, 'to', '')) && has_key(a:state, 'base')
let a:state.mode = 'hunk'
if empty(a:state.target) || a:state.target ==# '/' . a:state.to
if !a:state.child_found && len(a:state.queue) && a:state.queue[-1] ==# {'text': ''}
call remove(a:state.queue, -1)
endif
2019-08-22 15:36:17 +00:00
let a:state.child_found = 1
2020-01-07 12:45:07 +00:00
let offsets = map(split(matchstr(a:line, '^@\+ \zs[-+0-9, ]\+\ze @'), ' '), '+matchstr(v:val, "\\d\\+")')
let context = {}
if len(a:state.parents)
let from = ":" . a:state.from
let context.diff = map(copy(a:state.parents), s:log_diff_context)
endif
2019-08-22 15:36:17 +00:00
call add(a:state.queue, {
\ 'valid': 1,
2020-01-07 12:45:07 +00:00
\ 'context': context,
\ 'filename': FugitiveVimPath(a:state.base . '/' . a:state.to),
\ 'module': a:state.base_module . ':' . a:state.to,
\ 'lnum': offsets[-1],
2019-08-22 15:36:17 +00:00
\ 'text': a:state.message . matchstr(a:line, ' @@\+ .\+')})
endif
elseif a:state.follow &&
\ a:line =~# '^ \%(mode change \d\|\%(create\|delete\) mode \d\|\%(rename\|copy\|rewrite\) .* (\d\+%)$\)'
let rename = matchstr(a:line, '^ rename \zs.* => .*\ze (\d\+%)$')
if len(rename)
let rename = rename =~# '{.* => .*}' ? rename : '{' . rename . '}'
if a:state.target ==# simplify('/' . substitute(rename, '{.* => \(.*\)}', '\1', ''))
let a:state.target = simplify('/' . substitute(rename, '{\(.*\) => .*}', '\1', ''))
2018-06-14 10:31:12 +00:00
endif
endif
2019-08-22 15:36:17 +00:00
if !get(a:state, 'ignore_summary')
call add(a:state.queue, {'text': a:line})
2018-06-14 10:31:12 +00:00
endif
2020-01-07 12:45:07 +00:00
elseif a:state.mode ==# 'commit' || a:state.mode ==# 'init'
2019-08-22 15:36:17 +00:00
call add(a:state.queue, {'text': a:line})
endif
return []
2018-06-14 10:31:12 +00:00
endfunction
2019-11-16 15:28:42 +00:00
function! fugitive#LogCommand(line1, count, range, bang, mods, args, type) abort
2019-08-22 15:36:17 +00:00
let dir = s:Dir()
exe s:DirCheck(dir)
let listnr = a:type =~# '^l' ? 0 : -1
let [args, after] = s:SplitExpandChain(a:args, s:Tree(dir))
let split = index(args, '--')
if split > 0
let paths = args[split : -1]
let args = args[0 : split - 1]
elseif split == 0
let paths = args
let args = []
else
let paths = []
endif
if a:line1 == 0 && a:count
let path = fugitive#Path(bufname(a:count), '/', dir)
2020-01-07 12:45:07 +00:00
let titlepre = ':0,' . a:count
2019-08-22 15:36:17 +00:00
elseif a:count >= 0
let path = fugitive#Path(@%, '/', dir)
2020-01-07 12:45:07 +00:00
let titlepre = a:count == 0 ? ':0,' . bufnr('') : ':'
2019-08-22 15:36:17 +00:00
else
2020-01-07 12:45:07 +00:00
let titlepre = ':'
let path = ''
2019-08-22 15:36:17 +00:00
endif
let range = ''
2020-01-07 12:45:07 +00:00
let extra_args = []
let extra_paths = []
let state = {'mode': 'init', 'child_found': 0, 'queue': [], 'follow': 0}
2019-08-22 15:36:17 +00:00
if path =~# '^/\.git\%(/\|$\)\|^$'
2018-06-14 10:31:12 +00:00
let path = ''
2019-08-22 15:36:17 +00:00
elseif a:line1 == 0
let range = "0," . (a:count ? a:count : bufnr(''))
2020-01-07 12:45:07 +00:00
let extra_paths = ['.' . path]
2019-08-22 15:36:17 +00:00
if (empty(paths) || paths ==# ['--']) && !s:HasOpt(args, '--no-follow')
let state.follow = 1
if !s:HasOpt(args, '--follow')
2020-01-07 12:45:07 +00:00
call insert(extra_args, '--follow')
2019-08-22 15:36:17 +00:00
endif
if !s:HasOpt(args, '--summary')
2020-01-07 12:45:07 +00:00
call insert(extra_args, '--summary')
2019-08-22 15:36:17 +00:00
let state.ignore_summary = 1
endif
endif
elseif a:count > 0
if !s:HasOpt(args, '--merges', '--no-merges')
2020-01-07 12:45:07 +00:00
call insert(extra_args, '--no-merges')
2019-08-22 15:36:17 +00:00
endif
call add(args, '-L' . a:line1 . ',' . a:count . ':' . path[1:-1])
2018-06-14 10:31:12 +00:00
endif
2019-08-22 15:36:17 +00:00
if len(path) && empty(filter(copy(args), 'v:val =~# "^[^-]"'))
let owner = s:Owner(@%, dir)
2018-08-25 16:13:42 +00:00
if len(owner)
2019-08-22 15:36:17 +00:00
call add(args, owner)
2018-06-14 10:31:12 +00:00
endif
endif
2020-01-07 12:45:07 +00:00
if empty(extra_paths)
2019-08-22 15:36:17 +00:00
let path = ''
2018-08-25 16:13:42 +00:00
endif
2019-08-22 15:36:17 +00:00
if s:HasOpt(args, '-g', '--walk-reflogs')
2020-01-07 12:45:07 +00:00
let format = "%gd %P\t%H %gs"
2019-08-22 15:36:17 +00:00
else
2020-01-07 12:45:07 +00:00
let format = "%h %P\t%H " . g:fugitive_summary_format
2018-08-25 16:13:42 +00:00
endif
2019-08-22 15:36:17 +00:00
let cmd = ['--no-pager']
if fugitive#GitVersion(1, 9)
call extend(cmd, ['-c', 'diff.context=0', '-c', 'diff.noprefix=false', 'log'])
else
2019-11-16 15:28:42 +00:00
call extend(cmd, ['log', '-U0', '-s'])
2019-08-22 15:36:17 +00:00
endif
call extend(cmd,
\ ['--no-color', '--no-ext-diff', '--pretty=format:fugitive ' . format] +
2020-01-07 12:45:07 +00:00
\ args + extra_args + paths + extra_paths)
2019-08-22 15:36:17 +00:00
let state.target = path
2020-01-07 12:45:07 +00:00
let title = titlepre . (listnr < 0 ? 'Gclog ' : 'Gllog ') . s:fnameescape(args + paths)
if empty(paths + extra_paths) && empty(a:type) && len(s:Relative('/'))
2019-08-22 15:36:17 +00:00
let after = '|echohl WarningMsg|echo ' . string('Use :0Glog or :0Gclog for old behavior of targeting current file') . '|echohl NONE' . after
endif
2020-01-07 12:45:07 +00:00
return s:QuickfixStream(listnr, 'log', title, s:UserCommandList(dir) + cmd, !a:bang, s:function('s:LogParse'), state, dir) . after
2018-06-14 10:31:12 +00:00
endfunction
2018-08-25 16:13:42 +00:00
" Section: :Gedit, :Gpedit, :Gsplit, :Gvsplit, :Gtabedit, :Gread
2018-06-14 10:31:12 +00:00
function! s:UsableWin(nr) abort
2019-08-22 15:36:17 +00:00
return a:nr && !getwinvar(a:nr, '&previewwindow') && !getwinvar(a:nr, '&winfixwidth') &&
2019-03-08 11:04:56 +00:00
\ (empty(getwinvar(a:nr, 'fugitive_status')) || getbufvar(winbufnr(a:nr), 'fugitive_type') !=# 'index') &&
2019-01-08 10:11:54 +00:00
\ index(['gitrebase', 'gitcommit'], getbufvar(winbufnr(a:nr), '&filetype')) < 0 &&
2018-06-14 10:31:12 +00:00
\ index(['nofile','help','quickfix'], getbufvar(winbufnr(a:nr), '&buftype')) < 0
endfunction
2019-11-16 15:28:42 +00:00
function! s:OpenParse(args, wants_cmd) abort
let opts = []
let cmds = []
2018-07-30 21:18:16 +00:00
let args = copy(a:args)
2019-11-16 15:28:42 +00:00
while !empty(args)
if args[0] =~# '^++'
call add(opts, ' ' . escape(remove(args, 0), ' |"'))
elseif a:wants_cmd && args[0] =~# '^+'
call add(cmds, remove(args, 0)[1:-1])
else
break
endif
2018-07-30 21:18:16 +00:00
endwhile
if len(args)
let file = join(args)
elseif empty(expand('%'))
2019-11-16 15:28:42 +00:00
let file = ''
2018-07-30 21:18:16 +00:00
elseif empty(s:DirCommitFile(@%)[1]) && s:Relative('./') !~# '^\./\.git\>'
2019-11-16 15:28:42 +00:00
let file = '>:0'
2018-07-30 21:18:16 +00:00
else
2019-11-16 15:28:42 +00:00
let file = '>'
2018-07-30 21:18:16 +00:00
endif
2019-11-16 15:28:42 +00:00
let dir = s:Dir()
let efile = s:Expand(file)
let url = fugitive#Find(efile, dir)
if a:wants_cmd && file[0] ==# '>' && efile[0] !=# '>' && get(b:, 'fugitive_type', '') isnot# 'tree' && &filetype !=# 'netrw'
let line = line('.')
if expand('%:p') !=# url
let diffcmd = 'diff'
let from = s:DirRev(@%)[1]
let to = s:DirRev(url)[1]
if empty(from) && empty(to)
let diffcmd = 'diff-files'
let args = ['--', expand('%:p'), url]
elseif empty(to)
let args = [from, '--', url]
elseif empty(from)
let args = [to, '--', expand('%:p')]
let reverse = 1
else
let args = [from, to]
endif
let [res, exec_error] = s:LinesError([dir, diffcmd, '-U0'] + args)
if !exec_error
call filter(res, 'v:val =~# "^@@ "')
call map(res, 'substitute(v:val, ''[-+]\d\+\zs '', ",1 ", "g")')
call map(res, 'matchlist(v:val, ''^@@ -\(\d\+\),\(\d\+\) +\(\d\+\),\(\d\+\) @@'')[1:4]')
if exists('reverse')
call map(res, 'v:val[2:3] + v:val[0:1]')
endif
call filter(res, 'v:val[0] < '.line('.'))
let hunk = get(res, -1, [0,0,0,0])
if hunk[0] + hunk[1] > line('.')
let line = hunk[2] + max([1 - hunk[3], 0])
else
let line = hunk[2] + max([hunk[3], 1]) + line('.') - hunk[0] - max([hunk[1], 1])
endif
endif
endif
call insert(cmds, line)
endif
let pre = join(opts, '')
if len(cmds) > 1
let pre .= ' +' . escape(join(map(cmds, '"exe ".string(v:val)'), '|'), ' |"')
elseif len(cmds)
let pre .= ' +' . escape(cmds[0], ' |"')
endif
return [url, pre]
2018-07-30 21:18:16 +00:00
endfunction
2019-08-22 15:36:17 +00:00
function! s:DiffClose() abort
let mywinnr = winnr()
for winnr in [winnr('#')] + range(winnr('$'),1,-1)
if winnr != mywinnr && getwinvar(winnr,'&diff')
execute winnr.'wincmd w'
close
if winnr('$') > 1
wincmd p
endif
endif
endfor
diffoff!
endfunction
2018-08-25 16:13:42 +00:00
function! s:BlurStatus() abort
2019-01-08 10:11:54 +00:00
if (&previewwindow || exists('w:fugitive_status')) && get(b:,'fugitive_type', '') ==# 'index'
2018-07-30 21:18:16 +00:00
let winnrs = filter([winnr('#')] + range(1, winnr('$')), 's:UsableWin(v:val)')
if len(winnrs)
exe winnrs[0].'wincmd w'
else
2019-03-08 11:04:56 +00:00
belowright new
2018-07-30 21:18:16 +00:00
endif
if &diff
2019-08-22 15:36:17 +00:00
call s:DiffClose()
2018-06-14 10:31:12 +00:00
endif
endif
2018-08-25 16:13:42 +00:00
endfunction
2019-08-22 15:36:17 +00:00
function! s:OpenExec(cmd, mods, args, ...) abort
let dir = a:0 ? s:Dir(a:1) : s:Dir()
let temp = tempname()
let columns = get(g:, 'fugitive_columns', 80)
2020-01-29 02:07:36 +00:00
let env = s:BuildEnvPrefix({'COLUMNS': columns})
2019-08-22 15:36:17 +00:00
silent! execute '!' . escape(env . s:UserCommand(dir, ['--no-pager'] + a:args), '!#%') .
\ (&shell =~# 'csh' ? ' >& ' . temp : ' > ' . temp . ' 2>&1')
redraw!
let temp = s:Resolve(temp)
let first = join(readfile(temp, '', 2), "\n")
if first =~# '\<\([[:upper:][:digit:]_-]\+(\d\+)\).*\1'
let filetype = 'man'
else
let filetype = 'git'
endif
2020-01-29 02:07:36 +00:00
let s:temp_files[s:cpath(temp)] = { 'dir': dir, 'filetype': filetype }
2019-08-22 15:36:17 +00:00
if a:cmd ==# 'edit'
call s:BlurStatus()
endif
silent execute s:Mods(a:mods) . a:cmd temp
call fugitive#ReloadStatus(dir, 1)
return 'echo ' . string(':!' . s:UserCommand(dir, a:args))
endfunction
2018-06-14 10:31:12 +00:00
2019-11-16 15:28:42 +00:00
function! fugitive#Open(cmd, bang, mods, arg, args) abort
2018-06-14 10:31:12 +00:00
if a:bang
2019-08-22 15:36:17 +00:00
return s:OpenExec(a:cmd, a:mods, s:SplitExpand(a:arg, s:Tree()))
2018-06-14 10:31:12 +00:00
endif
2019-08-22 15:36:17 +00:00
let mods = s:Mods(a:mods)
2018-06-14 10:31:12 +00:00
try
2019-11-16 15:28:42 +00:00
let [file, pre] = s:OpenParse(a:args, 1)
2018-06-14 10:31:12 +00:00
catch /^fugitive:/
2019-08-22 15:36:17 +00:00
return 'echoerr ' . string(v:exception)
2018-06-14 10:31:12 +00:00
endtry
2018-08-25 16:13:42 +00:00
if file !~# '^\a\a\+:'
2018-06-14 10:31:12 +00:00
let file = s:sub(file, '/$', '')
endif
2018-08-25 16:13:42 +00:00
if a:cmd ==# 'edit'
call s:BlurStatus()
endif
2019-08-22 15:36:17 +00:00
return mods . a:cmd . pre . ' ' . s:fnameescape(file)
2018-06-14 10:31:12 +00:00
endfunction
2019-11-16 15:28:42 +00:00
function! fugitive#ReadCommand(line1, count, range, bang, mods, arg, args) abort
2019-08-22 15:36:17 +00:00
let mods = s:Mods(a:mods)
let after = a:count
2018-07-30 21:18:16 +00:00
if a:count < 0
let delete = 'silent 1,' . line('$') . 'delete_|'
let after = line('$')
elseif a:range == 2
2019-08-22 15:36:17 +00:00
let delete = 'silent ' . a:line1 . ',' . a:count . 'delete_|'
2018-07-30 21:18:16 +00:00
else
let delete = ''
endif
if a:bang
2019-08-22 15:36:17 +00:00
let dir = s:Dir()
let args = s:SplitExpand(a:arg, s:Tree(dir))
silent execute mods . after . 'read!' escape(s:UserCommand(dir, ['--no-pager'] + args), '!#%')
2018-07-30 21:18:16 +00:00
execute delete . 'diffupdate'
2019-11-16 15:28:42 +00:00
call fugitive#ReloadStatus(dir, 1)
2019-08-22 15:36:17 +00:00
return 'redraw|echo '.string(':!'.s:UserCommand(dir, args))
2018-07-30 21:18:16 +00:00
endif
try
2019-11-16 15:28:42 +00:00
let [file, pre] = s:OpenParse(a:args, 0)
2018-07-30 21:18:16 +00:00
catch /^fugitive:/
2019-08-22 15:36:17 +00:00
return 'echoerr ' . string(v:exception)
2018-07-30 21:18:16 +00:00
endtry
2018-08-25 16:13:42 +00:00
if file =~# '^fugitive:' && after is# 0
2019-08-22 15:36:17 +00:00
return 'exe ' .string(mods . fugitive#FileReadCmd(file, 0, pre)) . '|diffupdate'
2018-08-25 16:13:42 +00:00
endif
2019-01-08 10:11:54 +00:00
if foldlevel(after)
exe after . 'foldopen!'
endif
2019-08-22 15:36:17 +00:00
return mods . after . 'read' . pre . ' ' . s:fnameescape(file) . '|' . delete . 'diffupdate' . (a:count < 0 ? '|' . line('.') : '')
2018-06-14 10:31:12 +00:00
endfunction
2019-11-16 15:28:42 +00:00
function! fugitive#EditComplete(A, L, P) abort
if a:A =~# '^>'
return map(s:FilterEscape(s:CompleteHeads(s:Dir()), a:A[1:-1]), "'>' . v:val")
2018-06-14 10:31:12 +00:00
else
2019-05-17 14:09:13 +00:00
return fugitive#CompleteObject(a:A, a:L, a:P)
2018-06-14 10:31:12 +00:00
endif
endfunction
2019-11-16 15:28:42 +00:00
function! fugitive#ReadComplete(A, L, P) abort
if a:L =~# '^\w\+!'
return fugitive#Complete(a:A, a:L, a:P)
else
return fugitive#EditComplete(a:A, a:L, a:P)
endif
endfunction
2018-06-14 10:31:12 +00:00
2018-08-25 16:13:42 +00:00
" Section: :Gwrite, :Gwq
2018-06-14 10:31:12 +00:00
2019-11-16 15:28:42 +00:00
function! fugitive#WriteCommand(line1, line2, range, bang, mods, arg, args) abort
2018-06-14 10:31:12 +00:00
if exists('b:fugitive_commit_arguments')
return 'write|bdelete'
elseif expand('%:t') == 'COMMIT_EDITMSG' && $GIT_INDEX_FILE != ''
return 'wq'
2018-07-30 21:18:16 +00:00
elseif get(b:, 'fugitive_type', '') ==# 'index'
2018-06-14 10:31:12 +00:00
return 'Gcommit'
2020-01-29 02:07:36 +00:00
elseif &buftype ==# 'nowrite' && getline(4) =~# '^[+-]\{3\} '
return 'echoerr ' . string('fugitive: :Gwrite from :Git diff has been removed in favor of :Git add --edit')
2018-06-14 10:31:12 +00:00
endif
let mytab = tabpagenr()
let mybufnr = bufnr('')
2019-08-22 15:36:17 +00:00
try
let file = len(a:args) ? s:Generate(s:Expand(join(a:args, ' '))) : fugitive#Real(@%)
catch /^fugitive:/
return 'echoerr ' . string(v:exception)
endtry
2018-08-25 16:13:42 +00:00
if empty(file)
2018-06-14 10:31:12 +00:00
return 'echoerr '.string('fugitive: cannot determine file path')
endif
2018-08-25 16:13:42 +00:00
if file =~# '^fugitive:'
2019-05-17 14:09:13 +00:00
return 'write' . (a:bang ? '! ' : ' ') . s:fnameescape(file)
2018-06-14 10:31:12 +00:00
endif
2019-11-16 15:28:42 +00:00
exe s:DirCheck()
2019-08-22 15:36:17 +00:00
let always_permitted = s:cpath(fugitive#Real(@%), file) && empty(s:DirCommitFile(@%)[1])
2019-05-17 14:09:13 +00:00
if !always_permitted && !a:bang && (len(s:TreeChomp('diff', '--name-status', 'HEAD', '--', file)) || len(s:TreeChomp('ls-files', '--others', '--', file)))
2018-06-14 10:31:12 +00:00
let v:errmsg = 'fugitive: file has uncommitted changes (use ! to override)'
return 'echoerr v:errmsg'
endif
let treebufnr = 0
for nr in range(1,bufnr('$'))
if fnamemodify(bufname(nr),':p') ==# file
let treebufnr = nr
endif
endfor
if treebufnr > 0 && treebufnr != bufnr('')
let temp = tempname()
2019-08-22 15:36:17 +00:00
silent execute 'keepalt %write '.temp
2018-06-14 10:31:12 +00:00
for tab in [mytab] + range(1,tabpagenr('$'))
for winnr in range(1,tabpagewinnr(tab,'$'))
if tabpagebuflist(tab)[winnr-1] == treebufnr
execute 'tabnext '.tab
if winnr != winnr()
execute winnr.'wincmd w'
let restorewinnr = 1
endif
try
let lnum = line('.')
let last = line('$')
silent execute '$read '.temp
silent execute '1,'.last.'delete_'
silent write!
silent execute lnum
2019-08-22 15:36:17 +00:00
diffupdate
2018-06-14 10:31:12 +00:00
let did = 1
finally
if exists('restorewinnr')
wincmd p
endif
execute 'tabnext '.mytab
endtry
2019-08-22 15:36:17 +00:00
break
2018-06-14 10:31:12 +00:00
endif
endfor
endfor
if !exists('did')
call writefile(readfile(temp,'b'),file,'b')
endif
else
2018-08-25 16:13:42 +00:00
execute 'write! '.s:fnameescape(file)
2018-06-14 10:31:12 +00:00
endif
2019-05-17 14:09:13 +00:00
if a:bang
2019-08-22 15:36:17 +00:00
let [error, exec_error] = s:ChompError(['add', '--force', '--', file])
2018-06-14 10:31:12 +00:00
else
2019-08-22 15:36:17 +00:00
let [error, exec_error] = s:ChompError(['add', '--', file])
2018-06-14 10:31:12 +00:00
endif
2019-08-22 15:36:17 +00:00
if exec_error
2018-06-14 10:31:12 +00:00
let v:errmsg = 'fugitive: '.error
return 'echoerr v:errmsg'
endif
2018-08-25 16:13:42 +00:00
if s:cpath(fugitive#Real(@%), file) && s:DirCommitFile(@%)[1] =~# '^\d$'
2019-03-08 11:04:56 +00:00
setlocal nomodified
2018-06-14 10:31:12 +00:00
endif
2018-08-25 16:13:42 +00:00
let one = s:Generate(':1:'.file)
let two = s:Generate(':2:'.file)
let three = s:Generate(':3:'.file)
2018-06-14 10:31:12 +00:00
for nr in range(1,bufnr('$'))
let name = fnamemodify(bufname(nr), ':p')
if bufloaded(nr) && !getbufvar(nr,'&modified') && (name ==# one || name ==# two || name ==# three)
execute nr.'bdelete'
endif
endfor
unlet! restorewinnr
2018-08-25 16:13:42 +00:00
let zero = s:Generate(':0:'.file)
2019-11-16 15:28:42 +00:00
silent exe s:DoAutocmd('BufWritePost ' . s:fnameescape(zero))
2018-06-14 10:31:12 +00:00
for tab in range(1,tabpagenr('$'))
for winnr in range(1,tabpagewinnr(tab,'$'))
let bufnr = tabpagebuflist(tab)[winnr-1]
let bufname = fnamemodify(bufname(bufnr), ':p')
if bufname ==# zero && bufnr != mybufnr
execute 'tabnext '.tab
if winnr != winnr()
execute winnr.'wincmd w'
let restorewinnr = 1
endif
try
let lnum = line('.')
let last = line('$')
silent execute '$read '.s:fnameescape(file)
silent execute '1,'.last.'delete_'
silent execute lnum
2019-03-08 11:04:56 +00:00
setlocal nomodified
2018-06-14 10:31:12 +00:00
diffupdate
finally
if exists('restorewinnr')
wincmd p
endif
execute 'tabnext '.mytab
endtry
break
endif
endfor
endfor
2019-11-16 15:28:42 +00:00
call fugitive#ReloadStatus(-1, 1)
2018-06-14 10:31:12 +00:00
return 'checktime'
endfunction
2019-11-16 15:28:42 +00:00
function! fugitive#WqCommand(...) abort
let bang = a:4 ? '!' : ''
2018-06-14 10:31:12 +00:00
if exists('b:fugitive_commit_arguments')
return 'wq'.bang
endif
2019-11-16 15:28:42 +00:00
let result = call('fugitive#WriteCommand', a:000)
2018-06-14 10:31:12 +00:00
if result =~# '^\%(write\|wq\|echoerr\)'
return s:sub(result,'^write','wq')
else
return result.'|quit'.bang
endif
endfunction
2018-08-25 16:13:42 +00:00
" Section: :Gpush, :Gfetch
2018-06-14 10:31:12 +00:00
2019-11-16 15:28:42 +00:00
function! fugitive#PushComplete(A, L, P) abort
2019-08-22 15:36:17 +00:00
return s:CompleteSub('push', a:A, a:L, a:P, function('s:CompleteRemote'))
endfunction
2019-11-16 15:28:42 +00:00
function! fugitive#FetchComplete(A, L, P) abort
2019-08-22 15:36:17 +00:00
return s:CompleteSub('fetch', a:A, a:L, a:P, function('s:CompleteRemote'))
endfunction
function! s:AskPassArgs(dir) abort
if (len($DISPLAY) || len($TERM_PROGRAM) || has('gui_running')) && fugitive#GitVersion(1, 8) &&
2019-12-12 22:01:41 +00:00
\ empty($GIT_ASKPASS) && empty($SSH_ASKPASS) && empty(get(fugitive#Config(a:dir), 'core.askpass', []))
2019-08-22 15:36:17 +00:00
if s:executable(s:ExecPath() . '/git-gui--askpass')
return ['-c', 'core.askPass=' . s:ExecPath() . '/git-gui--askpass']
elseif s:executable('ssh-askpass')
return ['-c', 'core.askPass=ssh-askpass']
endif
endif
return []
endfunction
2018-06-14 10:31:12 +00:00
2019-08-22 15:36:17 +00:00
function! s:Dispatch(bang, cmd, args) abort
let dir = s:Dir()
2018-06-14 10:31:12 +00:00
let [mp, efm, cc] = [&l:mp, &l:efm, get(b:, 'current_compiler', '')]
try
let b:current_compiler = 'git'
2019-11-16 15:28:42 +00:00
let &l:errorformat = s:common_efm .
\ ',%\&git_dir=' . escape(substitute(dir, '%', '%%', 'g'), '\,')
2019-08-22 15:36:17 +00:00
let &l:makeprg = s:UserCommand(dir, s:AskPassArgs(dir) + [a:cmd] + a:args)
2018-06-14 10:31:12 +00:00
if exists(':Make') == 2
2019-03-08 11:04:56 +00:00
Make
2019-08-22 15:36:17 +00:00
return ''
2018-06-14 10:31:12 +00:00
else
2019-08-22 15:36:17 +00:00
if !has('patch-8.1.0334') && has('terminal') && &autowrite
let autowrite_was_set = 1
set noautowrite
silent! wall
endif
2018-06-14 10:31:12 +00:00
silent noautocmd make!
redraw!
2019-11-16 15:28:42 +00:00
return 'call fugitive#Cwindow()|silent ' . s:DoAutocmd('ShellCmdPost')
2018-06-14 10:31:12 +00:00
endif
finally
let [&l:mp, &l:efm, b:current_compiler] = [mp, efm, cc]
if empty(cc) | unlet! b:current_compiler | endif
2019-08-22 15:36:17 +00:00
if exists('autowrite_was_set')
set autowrite
endif
2018-06-14 10:31:12 +00:00
endtry
endfunction
2019-08-22 15:36:17 +00:00
function! s:PushSubcommand(line1, line2, range, bang, mods, args) abort
return s:Dispatch(a:bang ? '!' : '', 'push', a:args)
endfunction
function! s:FetchSubcommand(line1, line2, range, bang, mods, args) abort
return s:Dispatch(a:bang ? '!' : '', 'fetch', a:args)
endfunction
2018-08-25 16:13:42 +00:00
" Section: :Gdiff
2018-06-14 10:31:12 +00:00
augroup fugitive_diff
autocmd!
autocmd BufWinLeave *
\ if s:can_diffoff(+expand('<abuf>')) && s:diff_window_count() == 2 |
2019-03-08 11:04:56 +00:00
\ call s:diffoff_all(s:Dir(+expand('<abuf>'))) |
2018-06-14 10:31:12 +00:00
\ endif
autocmd BufWinEnter *
\ if s:can_diffoff(+expand('<abuf>')) && s:diff_window_count() == 1 |
\ call s:diffoff() |
\ endif
augroup END
function! s:can_diffoff(buf) abort
return getwinvar(bufwinnr(a:buf), '&diff') &&
\ !empty(getwinvar(bufwinnr(a:buf), 'fugitive_diff_restore'))
endfunction
function! fugitive#CanDiffoff(buf) abort
2019-03-08 11:04:56 +00:00
return s:can_diffoff(bufnr(a:buf))
2018-06-14 10:31:12 +00:00
endfunction
function! s:diff_modifier(count) abort
let fdc = matchstr(&diffopt, 'foldcolumn:\zs\d\+')
if &diffopt =~# 'horizontal' && &diffopt !~# 'vertical'
2019-08-22 15:36:17 +00:00
return ''
2018-06-14 10:31:12 +00:00
elseif &diffopt =~# 'vertical'
2019-08-22 15:36:17 +00:00
return 'vertical '
2018-06-14 10:31:12 +00:00
elseif winwidth(0) <= a:count * ((&tw ? &tw : 80) + (empty(fdc) ? 2 : fdc))
2019-08-22 15:36:17 +00:00
return ''
2018-06-14 10:31:12 +00:00
else
2019-08-22 15:36:17 +00:00
return 'vertical '
2018-06-14 10:31:12 +00:00
endif
endfunction
function! s:diff_window_count() abort
let c = 0
for nr in range(1,winnr('$'))
let c += getwinvar(nr,'&diff')
endfor
return c
endfunction
function! s:diff_restore() abort
let restore = 'setlocal nodiff noscrollbind'
\ . ' scrollopt=' . &l:scrollopt
\ . (&l:wrap ? ' wrap' : ' nowrap')
\ . ' foldlevel=999'
\ . ' foldmethod=' . &l:foldmethod
\ . ' foldcolumn=' . &l:foldcolumn
\ . ' foldlevel=' . &l:foldlevel
\ . (&l:foldenable ? ' foldenable' : ' nofoldenable')
if has('cursorbind')
let restore .= (&l:cursorbind ? ' ' : ' no') . 'cursorbind'
endif
return restore
endfunction
function! s:diffthis() abort
if !&diff
let w:fugitive_diff_restore = s:diff_restore()
diffthis
endif
endfunction
function! s:diffoff() abort
if exists('w:fugitive_diff_restore')
execute w:fugitive_diff_restore
unlet w:fugitive_diff_restore
else
diffoff
endif
endfunction
function! s:diffoff_all(dir) abort
let curwin = winnr()
for nr in range(1,winnr('$'))
2019-11-30 12:06:56 +00:00
if getwinvar(nr, '&diff') && !empty(getwinvar(nr, 'fugitive_diff_restore'))
2018-06-14 10:31:12 +00:00
if nr != winnr()
execute nr.'wincmd w'
endif
2019-11-30 12:06:56 +00:00
call s:diffoff()
2018-06-14 10:31:12 +00:00
endif
endfor
execute curwin.'wincmd w'
endfunction
2018-07-30 21:18:16 +00:00
function! s:CompareAge(mine, theirs) abort
2018-06-14 10:31:12 +00:00
let scores = {':0': 1, ':1': 2, ':2': 3, ':': 4, ':3': 5}
2018-07-30 21:18:16 +00:00
let mine = substitute(a:mine, '^:', '', '')
let theirs = substitute(a:theirs, '^:', '', '')
let my_score = get(scores, ':'.mine, 0)
let their_score = get(scores, ':'.theirs, 0)
2018-06-14 10:31:12 +00:00
if my_score || their_score
return my_score < their_score ? -1 : my_score != their_score
2018-07-30 21:18:16 +00:00
elseif mine ==# theirs
2018-06-14 10:31:12 +00:00
return 0
endif
2018-07-30 21:18:16 +00:00
let base = s:TreeChomp('merge-base', mine, theirs)
if base ==# mine
2018-06-14 10:31:12 +00:00
return -1
2018-07-30 21:18:16 +00:00
elseif base ==# theirs
2018-06-14 10:31:12 +00:00
return 1
endif
2018-08-25 16:13:42 +00:00
let my_time = +s:TreeChomp('log', '--max-count=1', '--pretty=format:%at', a:mine, '--')
let their_time = +s:TreeChomp('log', '--max-count=1', '--pretty=format:%at', a:theirs, '--')
2018-06-14 10:31:12 +00:00
return my_time < their_time ? -1 : my_time != their_time
endfunction
2019-08-22 15:36:17 +00:00
function! s:IsConflicted() abort
return len(@%) && !empty(s:ChompDefault('', 'ls-files', '--unmerged', '--', expand('%:p')))
endfunction
2019-11-16 15:28:42 +00:00
function! fugitive#Diffsplit(autodir, keepfocus, mods, arg, args) abort
let args = copy(a:args)
2018-06-14 10:31:12 +00:00
let post = ''
if get(args, 0) =~# '^+'
let post = remove(args, 0)[1:-1]
endif
2019-11-16 15:28:42 +00:00
if exists(':DiffGitCached') && empty(args)
return s:Mods(a:mods) . 'DiffGitCached' . (len(post) ? '|' . post : '')
endif
2018-07-30 21:18:16 +00:00
let commit = s:DirCommitFile(@%)[1]
2019-08-22 15:36:17 +00:00
if a:mods =~# '\<tab\>'
let mods = substitute(a:mods, '\<tab\>', '', 'g')
2019-11-16 15:28:42 +00:00
let pre = 'tab split'
2019-08-22 15:36:17 +00:00
else
let mods = 'keepalt ' . a:mods
2019-11-16 15:28:42 +00:00
let pre = ''
2019-08-22 15:36:17 +00:00
endif
2018-07-30 21:18:16 +00:00
let back = exists('*win_getid') ? 'call win_gotoid(' . win_getid() . ')' : 'wincmd p'
2019-08-22 15:36:17 +00:00
if (empty(args) || args[0] ==# ':') && a:keepfocus
2019-11-16 15:28:42 +00:00
exe s:DirCheck()
2019-08-22 15:36:17 +00:00
if empty(commit) && s:IsConflicted()
let parents = [s:Relative(':2:'), s:Relative(':3:')]
elseif empty(commit)
let parents = [s:Relative(':0:')]
elseif commit =~# '^\d\=$'
let parents = [s:Relative('HEAD:')]
elseif commit =~# '^\x\x\+$'
let parents = s:LinesError(['rev-parse', commit . '^@'])[0]
call map(parents, 's:Relative(v:val . ":")')
endif
endif
try
if exists('parents') && len(parents) > 1
2019-11-16 15:28:42 +00:00
exe pre
2019-08-22 15:36:17 +00:00
let mods = (a:autodir ? s:diff_modifier(len(parents) + 1) : '') . s:Mods(mods, 'leftabove')
let nr = bufnr('')
execute mods 'split' s:fnameescape(s:Generate(parents[0]))
call s:Map('n', 'dp', ':diffput '.nr.'<Bar>diffupdate<CR>', '<silent>')
let nr2 = bufnr('')
call s:diffthis()
exe back
call s:Map('n', 'd2o', ':diffget '.nr2.'<Bar>diffupdate<CR>', '<silent>')
let mods = substitute(mods, '\Cleftabove\|rightbelow\|aboveleft\|belowright', '\=submatch(0) =~# "f" ? "rightbelow" : "leftabove"', '')
for i in range(len(parents)-1, 1, -1)
execute mods 'split' s:fnameescape(s:Generate(parents[i]))
call s:Map('n', 'dp', ':diffput '.nr.'<Bar>diffupdate<CR>', '<silent>')
let nrx = bufnr('')
call s:diffthis()
exe back
call s:Map('n', 'd' . (i + 2) . 'o', ':diffget '.nrx.'<Bar>diffupdate<CR>', '<silent>')
endfor
call s:diffthis()
if len(parents) > 1
wincmd =
endif
2018-06-14 10:31:12 +00:00
return post
2019-08-22 15:36:17 +00:00
elseif len(args)
let arg = join(args, ' ')
if arg ==# ''
return post
2019-11-16 15:28:42 +00:00
elseif arg ==# ':/'
exe s:DirCheck()
2019-08-22 15:36:17 +00:00
let file = s:Relative()
elseif arg ==# ':'
2019-11-16 15:28:42 +00:00
exe s:DirCheck()
2019-08-22 15:36:17 +00:00
let file = s:Relative(':0:')
elseif arg =~# '^:\d$'
2019-11-16 15:28:42 +00:00
exe s:DirCheck()
2019-08-22 15:36:17 +00:00
let file = s:Relative(arg . ':')
2019-11-30 12:06:56 +00:00
elseif arg =~# '^[~^]\d*$'
return 'echoerr ' . string('fugitive: change ' . arg . ' to !' . arg . ' to diff against ancestor')
2019-08-22 15:36:17 +00:00
else
try
let file = arg =~# '^:/.' ? fugitive#RevParse(arg) . s:Relative(':') : s:Expand(arg)
catch /^fugitive:/
return 'echoerr ' . string(v:exception)
endtry
endif
elseif exists('parents') && len(parents)
let file = parents[-1]
elseif len(commit)
2018-08-25 16:13:42 +00:00
let file = s:Relative()
2019-08-22 15:36:17 +00:00
elseif s:IsConflicted()
let file = s:Relative(':1:')
let post = 'echohl WarningMsg|echo "Use :Gdiffsplit! for 3 way diff"|echohl NONE|' . post
2018-06-14 10:31:12 +00:00
else
2019-11-16 15:28:42 +00:00
exe s:DirCheck()
2019-08-22 15:36:17 +00:00
let file = s:Relative(':0:')
2018-06-14 10:31:12 +00:00
endif
2018-07-30 21:18:16 +00:00
let spec = s:Generate(file)
2019-08-22 15:36:17 +00:00
if spec =~# '^fugitive:' && empty(s:DirCommitFile(spec)[2])
let spec = FugitiveVimPath(spec . s:Relative('/'))
2018-06-14 10:31:12 +00:00
endif
2019-11-16 15:28:42 +00:00
exe pre
2019-08-22 15:36:17 +00:00
let restore = s:diff_restore()
2018-06-14 10:31:12 +00:00
let w:fugitive_diff_restore = restore
2019-11-16 15:28:42 +00:00
if len(spec) && s:CompareAge(commit, s:DirCommitFile(spec)[1]) < 0
2019-08-22 15:36:17 +00:00
let mods = s:Mods(mods, 'rightbelow')
2018-06-14 10:31:12 +00:00
else
2019-08-22 15:36:17 +00:00
let mods = s:Mods(mods, 'leftabove')
endif
let mods = (a:autodir ? s:diff_modifier(2) : '') . mods
if &diffopt =~# 'vertical'
let diffopt = &diffopt
set diffopt-=vertical
2018-06-14 10:31:12 +00:00
endif
2019-08-22 15:36:17 +00:00
execute mods 'diffsplit' s:fnameescape(spec)
2018-06-14 10:31:12 +00:00
let &l:readonly = &l:readonly
redraw
let w:fugitive_diff_restore = restore
let winnr = winnr()
if getwinvar('#', '&diff')
2019-08-22 15:36:17 +00:00
if a:keepfocus
exe back
2018-06-14 10:31:12 +00:00
endif
endif
return post
catch /^fugitive:/
2019-08-22 15:36:17 +00:00
return 'echoerr ' . string(v:exception)
finally
if exists('diffopt')
let &diffopt = diffopt
endif
2018-06-14 10:31:12 +00:00
endtry
endfunction
2018-08-25 16:13:42 +00:00
" Section: :Gmove, :Gremove
2018-06-14 10:31:12 +00:00
function! s:Move(force, rename, destination) abort
2019-11-16 15:28:42 +00:00
let dir = s:Dir()
exe s:DirCheck(dir)
if s:DirCommitFile(@%)[1] !~# '^0\=$' || empty(@%)
return 'echoerr ' . string('fugitive: mv not supported for this buffer')
endif
2018-08-25 16:13:42 +00:00
if a:destination =~# '^\.\.\=\%(/\|$\)'
let destination = simplify(getcwd() . '/' . a:destination)
elseif a:destination =~# '^\a\+:\|^/'
let destination = a:destination
elseif a:destination =~# '^:/:\='
2019-11-16 15:28:42 +00:00
let destination = s:Tree(dir) . substitute(a:destination, '^:/:\=', '', '')
2018-08-25 16:13:42 +00:00
elseif a:destination =~# '^:(\%(top\|top,literal\|literal,top\))'
2019-11-16 15:28:42 +00:00
let destination = s:Tree(dir) . matchstr(a:destination, ')\zs.*')
2018-08-25 16:13:42 +00:00
elseif a:destination =~# '^:(literal)'
let destination = simplify(getcwd() . '/' . matchstr(a:destination, ')\zs.*'))
2018-06-14 10:31:12 +00:00
elseif a:rename
2018-08-25 16:13:42 +00:00
let destination = expand('%:p:s?[\/]$??:h') . '/' . a:destination
2018-06-14 10:31:12 +00:00
else
2019-11-16 15:28:42 +00:00
let destination = s:Tree(dir) . '/' . a:destination
2018-06-14 10:31:12 +00:00
endif
2018-08-25 16:13:42 +00:00
let destination = s:Slash(destination)
2018-07-30 21:18:16 +00:00
if isdirectory(@%)
setlocal noswapfile
2018-06-14 10:31:12 +00:00
endif
2019-11-16 15:28:42 +00:00
let [message, exec_error] = s:ChompError(['mv'] + (a:force ? ['-f'] : []) + ['--', expand('%:p'), destination], dir)
2019-08-22 15:36:17 +00:00
if exec_error
2018-06-14 10:31:12 +00:00
let v:errmsg = 'fugitive: '.message
return 'echoerr v:errmsg'
endif
if isdirectory(destination)
let destination = fnamemodify(s:sub(destination,'/$','').'/'.expand('%:t'),':.')
endif
2019-11-16 15:28:42 +00:00
let reload = '|call fugitive#ReloadStatus(' . string(dir) . ', 1)'
2018-07-30 21:18:16 +00:00
if empty(s:DirCommitFile(@%)[1])
2018-06-14 10:31:12 +00:00
if isdirectory(destination)
2019-11-16 15:28:42 +00:00
return 'keepalt edit '.s:fnameescape(destination) . reload
2018-06-14 10:31:12 +00:00
else
2019-11-16 15:28:42 +00:00
return 'keepalt saveas! '.s:fnameescape(destination) . reload
2018-06-14 10:31:12 +00:00
endif
else
2019-11-16 15:28:42 +00:00
return 'file '.s:fnameescape(fugitive#Find(':0:'.destination, dir)) . reload
2018-06-14 10:31:12 +00:00
endif
endfunction
2019-11-16 15:28:42 +00:00
function! fugitive#RenameComplete(A,L,P) abort
2018-08-25 16:13:42 +00:00
if a:A =~# '^[.:]\=/'
2019-05-17 14:09:13 +00:00
return fugitive#CompletePath(a:A)
2018-06-14 10:31:12 +00:00
else
2018-08-25 16:13:42 +00:00
let pre = s:Slash(fnamemodify(expand('%:p:s?[\/]$??'), ':h')) . '/'
2019-05-17 14:09:13 +00:00
return map(fugitive#CompletePath(pre.a:A), 'strpart(v:val, len(pre))')
2018-06-14 10:31:12 +00:00
endif
endfunction
2019-11-16 15:28:42 +00:00
function! fugitive#MoveCommand(line1, line2, range, bang, mods, arg, args) abort
return s:Move(a:bang, 0, a:arg)
endfunction
function! fugitive#RenameCommand(line1, line2, range, bang, mods, arg, args) abort
return s:Move(a:bang, 1, a:arg)
endfunction
2018-06-14 10:31:12 +00:00
function! s:Remove(after, force) abort
2019-11-16 15:28:42 +00:00
let dir = s:Dir()
exe s:DirCheck(dir)
if len(@%) && s:DirCommitFile(@%)[1] ==# ''
2018-06-14 10:31:12 +00:00
let cmd = ['rm']
2018-07-30 21:18:16 +00:00
elseif s:DirCommitFile(@%)[1] ==# '0'
2018-06-14 10:31:12 +00:00
let cmd = ['rm','--cached']
else
2019-11-16 15:28:42 +00:00
return 'echoerr ' . string('fugitive: rm not supported for this buffer')
2018-06-14 10:31:12 +00:00
endif
if a:force
let cmd += ['--force']
endif
2019-11-16 15:28:42 +00:00
let [message, exec_error] = s:ChompError(cmd + ['--', expand('%:p')], dir)
2019-08-22 15:36:17 +00:00
if exec_error
2018-06-14 10:31:12 +00:00
let v:errmsg = 'fugitive: '.s:sub(message,'error:.*\zs\n\(.*-f.*',' (add ! to force)')
return 'echoerr '.string(v:errmsg)
else
2019-11-16 15:28:42 +00:00
return a:after . (a:force ? '!' : ''). '|call fugitive#ReloadStatus(' . string(dir) . ', 1)'
2018-06-14 10:31:12 +00:00
endif
endfunction
2019-11-16 15:28:42 +00:00
function! fugitive#RemoveCommand(line1, line2, range, bang, mods, arg, args) abort
return s:Remove('edit', a:bang)
endfunction
function! fugitive#DeleteCommand(line1, line2, range, bang, mods, arg, args) abort
return s:Remove('bdelete', a:bang)
endfunction
2018-06-14 10:31:12 +00:00
2018-08-25 16:13:42 +00:00
" Section: :Gblame
2018-06-14 10:31:12 +00:00
2018-07-30 21:18:16 +00:00
function! s:Keywordprg() abort
2019-03-08 11:04:56 +00:00
let args = ' --git-dir='.escape(s:Dir(),"\\\"' ")
2018-07-30 21:18:16 +00:00
if has('gui_running') && !has('win32')
return s:UserCommand() . ' --no-pager' . args . ' log -1'
else
return s:UserCommand() . args . ' show'
endif
endfunction
2018-06-14 10:31:12 +00:00
function! s:linechars(pattern) abort
let chars = strlen(s:gsub(matchstr(getline('.'), a:pattern), '.', '.'))
if exists('*synconcealed') && &conceallevel > 1
for col in range(1, chars)
let chars -= synconcealed(line('.'), col)[0]
endfor
endif
return chars
endfunction
2019-08-22 15:36:17 +00:00
function! s:BlameBufnr(...) abort
let state = s:TempState(bufname(a:0 ? a:1 : ''))
if get(state, 'filetype', '') ==# 'fugitiveblame'
return get(state, 'bufnr', -1)
else
return -1
endif
endfunction
function! s:BlameCommitFileLnum(...) abort
let line = a:0 ? a:1 : getline('.')
let state = a:0 ? a:2 : s:TempState()
let commit = matchstr(line, '^\^\=\zs\x\+')
if commit =~# '^0\+$'
let commit = ''
2019-11-16 15:28:42 +00:00
elseif has_key(state, 'blame_reverse_end')
2019-08-22 15:36:17 +00:00
let commit = get(s:LinesError('rev-list', '--ancestry-path', '--reverse', commit . '..' . state.blame_reverse_end)[0], 0, '')
endif
let lnum = +matchstr(line, ' \zs\d\+\ze \%((\| *\d\+)\)')
2019-12-01 19:18:45 +00:00
let path = matchstr(line, '^\^\=[?*]*\x* \+\%(\d\+ \+\d\+ \+\)\=\zs.\{-\}\ze\s*\d\+ \%((\| *\d\+)\)')
2019-08-22 15:36:17 +00:00
if empty(path) && lnum
let path = get(state, 'blame_file', '')
endif
return [commit, path, lnum]
endfunction
function! s:BlameLeave() abort
let bufwinnr = bufwinnr(s:BlameBufnr())
if bufwinnr > 0
let bufnr = bufnr('')
exe bufwinnr . 'wincmd w'
return bufnr . 'bdelete'
endif
return ''
endfunction
function! s:BlameQuit() abort
let cmd = s:BlameLeave()
if empty(cmd)
2018-06-14 10:31:12 +00:00
return 'bdelete'
2019-08-22 15:36:17 +00:00
elseif len(s:DirCommitFile(@%)[1])
return cmd . '|Gedit'
else
return cmd
2018-06-14 10:31:12 +00:00
endif
2019-08-22 15:36:17 +00:00
endfunction
2019-11-16 15:28:42 +00:00
function! fugitive#BlameComplete(A, L, P) abort
2019-08-22 15:36:17 +00:00
return s:CompleteSub('blame', a:A, a:L, a:P)
endfunction
function! s:BlameSubcommand(line1, count, range, bang, mods, args) abort
exe s:DirCheck()
let flags = copy(a:args)
let i = 0
let raw = 0
let commits = []
let files = []
let ranges = []
if a:line1 > 0 && a:count > 0 && a:range != 1
call extend(ranges, ['-L', a:line1 . ',' . a:count])
endif
while i < len(flags)
let match = matchlist(flags[i], '^\(-[a-zABDFH-KN-RT-Z]\)\ze\(.*\)')
if len(match) && len(match[2])
call insert(flags, match[1])
let flags[i+1] = '-' . match[2]
continue
2018-06-14 10:31:12 +00:00
endif
2019-08-22 15:36:17 +00:00
let arg = flags[i]
if arg =~# '^-p$\|^--\%(help\|porcelain\|line-porcelain\|incremental\)$'
let raw = 1
elseif arg ==# '--contents' && i + 1 < len(flags)
call extend(commits, remove(flags, i, i+1))
continue
elseif arg ==# '-L' && i + 1 < len(flags)
call extend(ranges, remove(flags, i, i+1))
continue
elseif arg =~# '^--contents='
call add(commits, remove(flags, i))
continue
elseif arg =~# '^-L.'
call add(ranges, remove(flags, i))
continue
2019-11-16 15:28:42 +00:00
elseif arg =~# '^-[GLS]$\|^--\%(date\|encoding\|contents\|ignore-rev\|ignore-revs-file\)$'
2019-08-22 15:36:17 +00:00
let i += 1
if i == len(flags)
echohl ErrorMsg
echo s:ChompError(['blame', arg])[0]
echohl NONE
return ''
endif
elseif arg ==# '--'
if i + 1 < len(flags)
call extend(files, remove(flags, i + 1, -1))
endif
call remove(flags, i)
break
elseif arg !~# '^-' && (s:HasOpt(flags, '--not') || arg !~# '^\^')
if index(flags, '--') >= 0
call add(commits, remove(flags, i))
continue
endif
if arg =~# '\.\.' && arg !~# '^\.\.\=\%(/\|$\)' && empty(commits)
call add(commits, remove(flags, i))
continue
endif
try
let dcf = s:DirCommitFile(fugitive#Find(arg))
if len(dcf[1]) && empty(dcf[2])
call add(commits, remove(flags, i))
continue
endif
catch /^fugitive:/
endtry
call add(files, remove(flags, i))
continue
2018-06-14 10:31:12 +00:00
endif
2019-08-22 15:36:17 +00:00
let i += 1
endwhile
let file = substitute(get(files, 0, get(s:TempState(), 'blame_file', s:Relative('./'))), '^\.\%(/\|$\)', '', '')
if empty(commits) && len(files) > 1
call add(commits, remove(files, 1))
endif
2019-11-16 15:28:42 +00:00
exe s:BlameLeave()
2019-08-22 15:36:17 +00:00
try
let cmd = ['--no-pager', '-c', 'blame.coloring=none', '-c', 'blame.blankBoundary=false', 'blame', '--show-number']
call extend(cmd, filter(copy(flags), 'v:val !~# "\\v^%(-b|--%(no-)=color-.*|--progress)$"'))
if a:count > 0 && empty(ranges)
let cmd += ['-L', (a:line1 ? a:line1 : line('.')) . ',' . (a:line1 ? a:line1 : line('.'))]
endif
call extend(cmd, ranges)
if len(commits)
let cmd += commits
elseif empty(files) && len(matchstr(s:DirCommitFile(@%)[1], '^\x\x\+$'))
let cmd += [matchstr(s:DirCommitFile(@%)[1], '^\x\x\+$')]
elseif empty(files) && !s:HasOpt(flags, '--reverse')
let cmd += ['--contents', '-']
2018-07-30 21:18:16 +00:00
endif
2019-08-22 15:36:17 +00:00
let basecmd = escape(fugitive#Prepare(cmd) . ' -- ' . s:shellesc(len(files) ? files : file), '!#%')
let tempname = tempname()
let error = tempname . '.err'
let temp = tempname . (raw ? '' : '.fugitiveblame')
if &shell =~# 'csh'
silent! execute '%write !('.basecmd.' > '.temp.') >& '.error
2018-06-14 10:31:12 +00:00
else
2019-08-22 15:36:17 +00:00
silent! execute '%write !'.basecmd.' > '.temp.' 2> '.error
2018-06-14 10:31:12 +00:00
endif
2019-11-16 15:28:42 +00:00
let l:shell_error = v:shell_error
2019-08-22 15:36:17 +00:00
redraw
2019-03-08 11:04:56 +00:00
try
2019-11-16 15:28:42 +00:00
if l:shell_error
2019-08-22 15:36:17 +00:00
let lines = readfile(error)
if empty(lines)
let lines = readfile(temp)
endif
for i in range(len(lines))
if lines[i] =~# '^error: \|^fatal: '
echohl ErrorMsg
echon lines[i]
echohl NONE
break
else
echon lines[i]
endif
if i != len(lines) - 1
echon "\n"
endif
endfor
return ''
endif
2020-01-29 02:07:36 +00:00
let temp_state = {'dir': s:Dir(), 'filetype': (raw ? '' : 'fugitiveblame'), 'blame_flags': flags, 'blame_file': file}
2019-08-22 15:36:17 +00:00
if s:HasOpt(flags, '--reverse')
let temp_state.blame_reverse_end = matchstr(get(commits, 0, ''), '\.\.\zs.*')
2018-06-14 10:31:12 +00:00
endif
2019-08-22 15:36:17 +00:00
if (a:line1 == 0 || a:range == 1) && a:count > 0
let edit = s:Mods(a:mods) . get(['edit', 'split', 'pedit', 'vsplit', 'tabedit'], a:count - (a:line1 ? a:line1 : 1), 'split')
return s:BlameCommit(edit, get(readfile(temp), 0, ''), temp_state)
2018-06-14 10:31:12 +00:00
else
2019-08-22 15:36:17 +00:00
let temp = s:Resolve(temp)
let s:temp_files[s:cpath(temp)] = temp_state
if len(ranges + commits + files) || raw
let mods = s:Mods(a:mods)
if a:count != 0
exe 'silent keepalt' mods 'split' s:fnameescape(temp)
elseif !&modified || a:bang || &bufhidden ==# 'hide' || (empty(&bufhidden) && &hidden)
exe 'silent' mods 'edit' . (a:bang ? '! ' : ' ') . s:fnameescape(temp)
else
return mods . 'edit ' . s:fnameescape(temp)
endif
return ''
endif
if a:mods =~# '\<tab\>'
silent tabedit %
endif
let mods = substitute(a:mods, '\<tab\>', '', 'g')
2018-06-14 10:31:12 +00:00
for winnr in range(winnr('$'),1,-1)
2019-08-22 15:36:17 +00:00
if getwinvar(winnr, '&scrollbind')
call setwinvar(winnr, '&scrollbind', 0)
endif
if exists('+cursorbind') && getwinvar(winnr, '&cursorbind')
2018-06-14 10:31:12 +00:00
call setwinvar(winnr, '&cursorbind', 0)
endif
2019-08-22 15:36:17 +00:00
if s:BlameBufnr(winbufnr(winnr)) > 0
2018-06-14 10:31:12 +00:00
execute winbufnr(winnr).'bdelete'
endif
endfor
let bufnr = bufnr('')
2019-08-22 15:36:17 +00:00
let temp_state.bufnr = bufnr
2018-06-14 10:31:12 +00:00
let restore = 'call setwinvar(bufwinnr('.bufnr.'),"&scrollbind",0)'
if exists('+cursorbind')
let restore .= '|call setwinvar(bufwinnr('.bufnr.'),"&cursorbind",0)'
endif
if &l:wrap
let restore .= '|call setwinvar(bufwinnr('.bufnr.'),"&wrap",1)'
endif
if &l:foldenable
let restore .= '|call setwinvar(bufwinnr('.bufnr.'),"&foldenable",1)'
endif
setlocal scrollbind nowrap nofoldenable
if exists('+cursorbind')
setlocal cursorbind
endif
let top = line('w0') + &scrolloff
let current = line('.')
2019-08-22 15:36:17 +00:00
exe 'silent keepalt' (a:bang ? s:Mods(mods) . 'split' : s:Mods(mods, 'leftabove') . 'vsplit') s:fnameescape(temp)
2018-06-14 10:31:12 +00:00
let w:fugitive_leave = restore
execute top
normal! zt
execute current
if exists('+cursorbind')
setlocal cursorbind
endif
2019-08-22 15:36:17 +00:00
setlocal nonumber scrollbind nowrap foldcolumn=0 nofoldenable winfixwidth
2018-06-14 10:31:12 +00:00
if exists('+relativenumber')
setlocal norelativenumber
endif
2019-11-16 15:28:42 +00:00
if exists('+signcolumn')
setlocal signcolumn=no
endif
2018-06-14 10:31:12 +00:00
execute "vertical resize ".(s:linechars('.\{-\}\ze\s\+\d\+)')+1)
2019-08-22 15:36:17 +00:00
call s:Map('n', 'A', ":<C-u>exe 'vertical resize '.(<SID>linechars('.\\{-\\}\\ze [0-9:/+-][0-9:/+ -]* \\d\\+)')+1+v:count)<CR>", '<silent>')
call s:Map('n', 'C', ":<C-u>exe 'vertical resize '.(<SID>linechars('^\\S\\+')+1+v:count)<CR>", '<silent>')
call s:Map('n', 'D', ":<C-u>exe 'vertical resize '.(<SID>linechars('.\\{-\\}\\ze\\d\\ze\\s\\+\\d\\+)')+1-v:count)<CR>", '<silent>')
2018-06-14 10:31:12 +00:00
redraw
syncbind
endif
endtry
return ''
catch /^fugitive:/
2019-08-22 15:36:17 +00:00
return 'echoerr ' . string(v:exception)
2018-06-14 10:31:12 +00:00
endtry
endfunction
2018-07-30 21:18:16 +00:00
function! s:BlameCommit(cmd, ...) abort
let line = a:0 ? a:1 : getline('.')
2019-08-22 15:36:17 +00:00
let state = a:0 ? a:2 : s:TempState()
let sigil = has_key(state, 'blame_reverse_end') ? '-' : '+'
let mods = (s:BlameBufnr() < 0 ? '' : &splitbelow ? "botright " : "topleft ")
let [commit, path, lnum] = s:BlameCommitFileLnum(line, state)
if empty(commit) && len(path) && has_key(state, 'blame_reverse_end')
let path = (len(state.blame_reverse_end) ? state.blame_reverse_end . ':' : ':(top)') . path
2019-11-16 15:28:42 +00:00
return fugitive#Open(mods . a:cmd, 0, '', '+' . lnum . ' ' . s:fnameescape(path), ['+' . lnum, path])
2019-08-22 15:36:17 +00:00
endif
if commit =~# '^0*$'
return 'echoerr ' . string('fugitive: no commit')
endif
if line =~# '^\^' && !has_key(state, 'blame_reverse_end')
let path = commit . ':' . path
2019-11-16 15:28:42 +00:00
return fugitive#Open(mods . a:cmd, 0, '', '+' . lnum . ' ' . s:fnameescape(path), ['+' . lnum, path])
2019-08-22 15:36:17 +00:00
endif
2019-11-16 15:28:42 +00:00
let cmd = fugitive#Open(mods . a:cmd, 0, '', commit, [commit])
2018-06-14 10:31:12 +00:00
if cmd =~# '^echoerr'
return cmd
endif
execute cmd
2019-08-22 15:36:17 +00:00
if a:cmd ==# 'pedit' || empty(path)
2018-07-30 21:18:16 +00:00
return ''
endif
2018-06-14 10:31:12 +00:00
if search('^diff .* b/\M'.escape(path,'\').'$','W')
call search('^+++')
let head = line('.')
while search('^@@ \|^diff ') && getline('.') =~# '^@@ '
2019-08-22 15:36:17 +00:00
let top = +matchstr(getline('.'),' ' . sigil .'\zs\d\+')
let len = +matchstr(getline('.'),' ' . sigil . '\d\+,\zs\d\+')
2018-06-14 10:31:12 +00:00
if lnum >= top && lnum <= top + len
let offset = lnum - top
if &scrolloff
+
normal! zt
else
normal! zt
+
endif
while offset > 0 && line('.') < line('$')
+
2019-08-22 15:36:17 +00:00
if getline('.') =~# '^[ ' . sigil . ']'
2018-06-14 10:31:12 +00:00
let offset -= 1
endif
endwhile
return 'normal! zv'
endif
endwhile
execute head
normal! zt
endif
return ''
endfunction
2019-08-22 15:36:17 +00:00
function! s:BlameJump(suffix, ...) abort
2019-03-08 11:04:56 +00:00
let suffix = a:suffix
2019-08-22 15:36:17 +00:00
let [commit, path, lnum] = s:BlameCommitFileLnum()
if empty(path)
return 'echoerr ' . string('fugitive: could not determine filename for blame')
endif
if commit =~# '^0*$'
2019-03-08 11:04:56 +00:00
let commit = 'HEAD'
let suffix = ''
2018-06-14 10:31:12 +00:00
endif
let offset = line('.') - line('w0')
2019-08-22 15:36:17 +00:00
let flags = get(s:TempState(), 'blame_flags', [])
if a:0 && a:1
if s:HasOpt(flags, '--reverse')
call remove(flags, '--reverse')
else
call add(flags, '--reverse')
endif
2018-06-14 10:31:12 +00:00
endif
2019-08-22 15:36:17 +00:00
let blame_bufnr = s:BlameBufnr()
if blame_bufnr > 0
let bufnr = bufnr('')
let winnr = bufwinnr(blame_bufnr)
if winnr > 0
exe winnr.'wincmd w'
2019-11-30 12:06:56 +00:00
exe bufnr.'bdelete'
2019-08-22 15:36:17 +00:00
endif
execute 'Gedit' s:fnameescape(commit . suffix . ':' . path)
execute lnum
2018-06-14 10:31:12 +00:00
endif
if exists(':Gblame')
2019-08-22 15:36:17 +00:00
let my_bufnr = bufnr('')
if blame_bufnr < 0
let blame_args = flags + [commit . suffix, '--', path]
let result = s:BlameSubcommand(0, 0, 0, 0, '', blame_args)
else
let blame_args = flags
let result = s:BlameSubcommand(-1, -1, 0, 0, '', blame_args)
endif
if bufnr('') == my_bufnr
return result
endif
execute result
2018-06-14 10:31:12 +00:00
execute lnum
let delta = line('.') - line('w0') - offset
if delta > 0
execute 'normal! '.delta."\<C-E>"
elseif delta < 0
execute 'normal! '.(-delta)."\<C-Y>"
endif
2019-08-22 15:36:17 +00:00
keepjumps syncbind
redraw
echo ':Gblame' s:fnameescape(blame_args)
2018-06-14 10:31:12 +00:00
endif
return ''
endfunction
let s:hash_colors = {}
2019-08-22 15:36:17 +00:00
function! fugitive#BlameSyntax() abort
2018-06-14 10:31:12 +00:00
let conceal = has('conceal') ? ' conceal' : ''
2019-08-22 15:36:17 +00:00
let config = fugitive#Config()
let flags = get(s:TempState(), 'blame_flags', [])
syn match FugitiveblameBlank "^\s\+\s\@=" nextgroup=FugitiveblameAnnotation,FugitiveblameScoreDebug,FugitiveblameOriginalFile,FugitiveblameOriginalLineNumber skipwhite
2019-11-16 15:28:42 +00:00
syn match FugitiveblameHash "\%(^\^\=[?*]*\)\@<=\<\x\{7,\}\>" nextgroup=FugitiveblameAnnotation,FugitiveblameScoreDebug,FugitiveblameOriginalLineNumber,FugitiveblameOriginalFile skipwhite
2019-08-22 15:36:17 +00:00
syn match FugitiveblameUncommitted "\%(^\^\=\)\@<=\<0\{7,\}\>" nextgroup=FugitiveblameAnnotation,FugitiveblameScoreDebug,FugitiveblameOriginalLineNumber,FugitiveblameOriginalFile skipwhite
if get(get(config, 'blame.blankboundary', ['x']), 0, 'x') =~# '^$\|^true$' || s:HasOpt(flags, '-b')
2019-11-16 15:28:42 +00:00
syn match FugitiveblameBoundaryIgnore "^\^[*?]*\x\{7,\}\>" nextgroup=FugitiveblameAnnotation,FugitiveblameScoreDebug,FugitiveblameOriginalLineNumber,FugitiveblameOriginalFile skipwhite
2019-08-22 15:36:17 +00:00
else
syn match FugitiveblameBoundary "^\^"
endif
syn match FugitiveblameScoreDebug " *\d\+\s\+\d\+\s\@=" nextgroup=FugitiveblameAnnotation,FugitiveblameOriginalLineNumber,fugitiveblameOriginalFile contained skipwhite
syn region FugitiveblameAnnotation matchgroup=FugitiveblameDelimiter start="(" end="\%(\s\d\+\)\@<=)" contained keepend oneline
syn match FugitiveblameTime "[0-9:/+-][0-9:/+ -]*[0-9:/+-]\%(\s\+\d\+)\)\@=" contained containedin=FugitiveblameAnnotation
exec 'syn match FugitiveblameLineNumber "\s*\d\+)\@=" contained containedin=FugitiveblameAnnotation' conceal
exec 'syn match FugitiveblameOriginalFile "\s\%(\f\+\D\@<=\|\D\@=\f\+\)\%(\%(\s\+\d\+\)\=\s\%((\|\s*\d\+)\)\)\@=" contained nextgroup=FugitiveblameOriginalLineNumber,FugitiveblameAnnotation skipwhite' (s:HasOpt(flags, '--show-name', '-f') ? '' : conceal)
exec 'syn match FugitiveblameOriginalLineNumber "\s*\d\+\%(\s(\)\@=" contained nextgroup=FugitiveblameAnnotation skipwhite' (s:HasOpt(flags, '--show-number', '-n') ? '' : conceal)
exec 'syn match FugitiveblameOriginalLineNumber "\s*\d\+\%(\s\+\d\+)\)\@=" contained nextgroup=FugitiveblameShort skipwhite' (s:HasOpt(flags, '--show-number', '-n') ? '' : conceal)
2018-06-14 10:31:12 +00:00
syn match FugitiveblameShort " \d\+)" contained contains=FugitiveblameLineNumber
syn match FugitiveblameNotCommittedYet "(\@<=Not Committed Yet\>" contained containedin=FugitiveblameAnnotation
hi def link FugitiveblameBoundary Keyword
hi def link FugitiveblameHash Identifier
2019-08-22 15:36:17 +00:00
hi def link FugitiveblameBoundaryIgnore Ignore
2018-06-14 10:31:12 +00:00
hi def link FugitiveblameUncommitted Ignore
2019-08-22 15:36:17 +00:00
hi def link FugitiveblameScoreDebug Debug
2018-06-14 10:31:12 +00:00
hi def link FugitiveblameTime PreProc
hi def link FugitiveblameLineNumber Number
hi def link FugitiveblameOriginalFile String
hi def link FugitiveblameOriginalLineNumber Float
hi def link FugitiveblameShort FugitiveblameDelimiter
hi def link FugitiveblameDelimiter Delimiter
hi def link FugitiveblameNotCommittedYet Comment
2019-08-22 15:36:17 +00:00
if !get(g:, 'fugitive_dynamic_colors', 1) && !s:HasOpt(flags, '--color-lines') || s:HasOpt(flags, '--no-color-lines')
return
endif
2018-06-14 10:31:12 +00:00
let seen = {}
for lnum in range(1, line('$'))
let hash = matchstr(getline(lnum), '^\^\=\zs\x\{6\}')
if hash ==# '' || hash ==# '000000' || has_key(seen, hash)
continue
endif
let seen[hash] = 1
if &t_Co > 16 && get(g:, 'CSApprox_loaded') && !empty(findfile('autoload/csapprox/per_component.vim', escape(&rtp, ' ')))
\ && empty(get(s:hash_colors, hash))
let [s, r, g, b; __] = map(matchlist(hash, '\(\x\x\)\(\x\x\)\(\x\x\)'), 'str2nr(v:val,16)')
let color = csapprox#per_component#Approximate(r, g, b)
if color == 16 && &background ==# 'dark'
let color = 8
endif
let s:hash_colors[hash] = ' ctermfg='.color
else
let s:hash_colors[hash] = ''
endif
exe 'syn match FugitiveblameHash'.hash.' "\%(^\^\=\)\@<='.hash.'\x\{1,34\}\>" nextgroup=FugitiveblameAnnotation,FugitiveblameOriginalLineNumber,fugitiveblameOriginalFile skipwhite'
endfor
2019-08-22 15:36:17 +00:00
call s:BlameRehighlight()
2018-06-14 10:31:12 +00:00
endfunction
2019-08-22 15:36:17 +00:00
function! s:BlameRehighlight() abort
2018-06-14 10:31:12 +00:00
for [hash, cterm] in items(s:hash_colors)
if !empty(cterm) || has('gui_running') || has('termguicolors') && &termguicolors
exe 'hi FugitiveblameHash'.hash.' guifg=#'.hash.get(s:hash_colors, hash, '')
else
exe 'hi link FugitiveblameHash'.hash.' Identifier'
endif
endfor
endfunction
2019-08-22 15:36:17 +00:00
function! s:BlameFileType() abort
setlocal nomodeline
setlocal foldmethod=manual
if len(s:Dir())
let &l:keywordprg = s:Keywordprg()
endif
let b:undo_ftplugin = 'setl keywordprg= foldmethod<'
if exists('+concealcursor')
setlocal concealcursor=nc conceallevel=2
let b:undo_ftplugin .= ' concealcursor< conceallevel<'
endif
if &modifiable
return ''
endif
2019-11-16 15:28:42 +00:00
call s:Map('n', '<F1>', ':help :Gblame<CR>', '<silent>')
call s:Map('n', 'g?', ':help :Gblame<CR>', '<silent>')
2019-08-22 15:36:17 +00:00
if mapcheck('q', 'n') =~# '^$\|bdelete'
call s:Map('n', 'q', ':exe <SID>BlameQuit()<Bar>echohl WarningMsg<Bar>echo ":Gblame q is deprecated in favor of gq"<Bar>echohl NONE<CR>', '<silent>')
endif
call s:Map('n', 'gq', ':exe <SID>BlameQuit()<CR>', '<silent>')
call s:Map('n', '<2-LeftMouse>', ':<C-U>exe <SID>BlameCommit("exe <SID>BlameLeave()<Bar>edit")<CR>', '<silent>')
call s:Map('n', '<CR>', ':<C-U>exe <SID>BlameCommit("exe <SID>BlameLeave()<Bar>edit")<CR>', '<silent>')
call s:Map('n', '-', ':<C-U>exe <SID>BlameJump("")<CR>', '<silent>')
call s:Map('n', 'P', ':<C-U>exe <SID>BlameJump("^".v:count1)<CR>', '<silent>')
call s:Map('n', '~', ':<C-U>exe <SID>BlameJump("~".v:count1)<CR>', '<silent>')
call s:Map('n', 'i', ':<C-U>exe <SID>BlameCommit("exe <SID>BlameLeave()<Bar>edit")<CR>', '<silent>')
call s:Map('n', 'o', ':<C-U>exe <SID>BlameCommit("split")<CR>', '<silent>')
call s:Map('n', 'O', ':<C-U>exe <SID>BlameCommit("tabedit")<CR>', '<silent>')
call s:Map('n', 'p', ':<C-U>exe <SID>BlameCommit("pedit")<CR>', '<silent>')
endfunction
augroup fugitive_blame
autocmd!
autocmd FileType fugitiveblame call s:BlameFileType()
autocmd ColorScheme,GUIEnter * call s:BlameRehighlight()
autocmd BufWinLeave * execute getwinvar(+bufwinnr(+expand('<abuf>')), 'fugitive_leave')
augroup END
2018-08-25 16:13:42 +00:00
" Section: :Gbrowse
2018-06-14 10:31:12 +00:00
let s:redirects = {}
2019-11-16 15:28:42 +00:00
function! fugitive#BrowseCommand(line1, count, range, bang, mods, arg, args) abort
2019-03-08 11:04:56 +00:00
let dir = s:Dir()
2019-08-22 15:36:17 +00:00
exe s:DirCheck(dir)
2018-06-14 10:31:12 +00:00
try
let validremote = '\.\|\.\=/.*\|[[:alnum:]_-]\+\%(://.\{-\}\)\='
2019-08-22 15:36:17 +00:00
if a:args ==# ['-']
if a:count >= 0
return 'echoerr ' . string('fugitive: ''-'' no longer required to get persistent URL if range given')
else
return 'echoerr ' . string('fugitive: use :0Gbrowse instead of :Gbrowse -')
endif
elseif len(a:args)
2019-05-17 14:09:13 +00:00
let remote = matchstr(join(a:args, ' '),'@\zs\%('.validremote.'\)$')
let rev = substitute(join(a:args, ' '),'@\%('.validremote.'\)$','','')
2018-06-14 10:31:12 +00:00
else
let remote = ''
let rev = ''
endif
if rev ==# ''
2018-08-25 16:13:42 +00:00
let rev = s:DirRev(@%)[1]
endif
if rev =~# '^:\=$'
let expanded = s:Relative()
2018-06-14 10:31:12 +00:00
else
2018-07-30 21:18:16 +00:00
let expanded = s:Expand(rev)
2018-06-14 10:31:12 +00:00
endif
2019-11-16 15:28:42 +00:00
let cdir = FugitiveVimPath(fugitive#CommonDir(dir))
2019-03-08 11:04:56 +00:00
for subdir in ['tags/', 'heads/', 'remotes/']
if expanded !~# '^[./]' && filereadable(cdir . '/refs/' . subdir . expanded)
let expanded = '.git/refs/' . subdir . expanded
2018-08-25 16:13:42 +00:00
endif
endfor
2019-11-16 15:28:42 +00:00
let full = fugitive#Find(expanded, dir)
2018-06-14 10:31:12 +00:00
let commit = ''
2018-07-04 10:53:25 +00:00
if full =~? '^fugitive:'
2019-03-08 11:04:56 +00:00
let [pathdir, commit, path] = s:DirCommitFile(full)
2018-07-04 10:53:25 +00:00
if commit =~# '^:\=\d$'
let commit = ''
endif
2018-06-14 10:31:12 +00:00
if commit =~ '..'
2018-07-30 21:18:16 +00:00
let type = s:TreeChomp('cat-file','-t',commit.s:sub(path,'^/',':'))
2018-06-14 10:31:12 +00:00
let branch = matchstr(expanded, '^[^:]*')
else
let type = 'blob'
endif
let path = path[1:-1]
2019-03-08 11:04:56 +00:00
elseif empty(s:Tree(dir))
let path = '.git/' . full[strlen(dir)+1:-1]
2018-06-14 10:31:12 +00:00
let type = ''
else
2019-11-16 15:28:42 +00:00
let path = fugitive#Path(full, '/')[1:-1]
2018-06-14 10:31:12 +00:00
if path =~# '^\.git/'
let type = ''
2019-11-16 15:28:42 +00:00
elseif isdirectory(full) || empty(path)
2018-06-14 10:31:12 +00:00
let type = 'tree'
else
let type = 'blob'
endif
endif
if type ==# 'tree' && !empty(path)
let path = s:sub(path, '/\=$', '/')
endif
2019-03-08 11:04:56 +00:00
if path =~# '^\.git/.*HEAD$' && filereadable(dir . '/' . path[5:-1])
let body = readfile(dir . '/' . path[5:-1])[0]
2019-08-22 15:36:17 +00:00
if body =~# '^\x\{40,\}$'
2018-06-14 10:31:12 +00:00
let commit = body
let type = 'commit'
let path = ''
elseif body =~# '^ref: refs/'
let path = '.git/' . matchstr(body,'ref: \zs.*')
endif
endif
let merge = ''
if path =~# '^\.git/refs/remotes/.'
if empty(remote)
let remote = matchstr(path, '^\.git/refs/remotes/\zs[^/]\+')
let branch = matchstr(path, '^\.git/refs/remotes/[^/]\+/\zs.\+')
else
let merge = matchstr(path, '^\.git/refs/remotes/[^/]\+/\zs.\+')
let path = '.git/refs/heads/'.merge
endif
elseif path =~# '^\.git/refs/heads/.'
let branch = path[16:-1]
elseif !exists('branch')
2018-07-30 21:18:16 +00:00
let branch = FugitiveHead()
2018-06-14 10:31:12 +00:00
endif
if !empty(branch)
2018-07-30 21:18:16 +00:00
let r = fugitive#Config('branch.'.branch.'.remote')
let m = fugitive#Config('branch.'.branch.'.merge')[11:-1]
2018-06-14 10:31:12 +00:00
if r ==# '.' && !empty(m)
2018-07-30 21:18:16 +00:00
let r2 = fugitive#Config('branch.'.m.'.remote')
2018-06-14 10:31:12 +00:00
if r2 !~# '^\.\=$'
let r = r2
2018-07-30 21:18:16 +00:00
let m = fugitive#Config('branch.'.m.'.merge')[11:-1]
2018-06-14 10:31:12 +00:00
endif
endif
if empty(remote)
let remote = r
endif
if r ==# '.' || r ==# remote
let merge = m
if path =~# '^\.git/refs/heads/.'
let path = '.git/refs/heads/'.merge
endif
endif
endif
2018-08-25 16:13:42 +00:00
let line1 = a:count > 0 ? a:line1 : 0
let line2 = a:count > 0 ? a:count : 0
2018-06-14 10:31:12 +00:00
if empty(commit) && path !~# '^\.git/'
2019-08-22 15:36:17 +00:00
if a:count < 0 && !empty(merge)
2018-06-14 10:31:12 +00:00
let commit = merge
else
2018-08-25 16:13:42 +00:00
let commit = ''
if len(merge)
2018-09-25 00:40:17 +00:00
let owner = s:Owner(@%)
2019-08-22 15:36:17 +00:00
let [commit, exec_error] = s:ChompError(['merge-base', 'refs/remotes/' . remote . '/' . merge, empty(owner) ? 'HEAD' : owner, '--'])
if exec_error
2018-09-25 00:40:17 +00:00
let commit = ''
endif
2019-08-22 15:36:17 +00:00
if a:count > 0 && empty(a:args) && commit =~# '^\x\{40,\}$'
2018-08-25 16:13:42 +00:00
let blame_list = tempname()
call writefile([commit, ''], blame_list, 'b')
let blame_in = tempname()
silent exe '%write' blame_in
2019-08-22 15:36:17 +00:00
let [blame, exec_error] = s:LinesError(['-c', 'blame.coloring=none', 'blame', '--contents', blame_in, '-L', a:line1.','.a:count, '-S', blame_list, '-s', '--show-number', './' . path])
if !exec_error
2018-08-25 16:13:42 +00:00
let blame_regex = '^\^\x\+\s\+\zs\d\+\ze\s'
if get(blame, 0) =~# blame_regex && get(blame, -1) =~# blame_regex
let line1 = +matchstr(blame[0], blame_regex)
let line2 = +matchstr(blame[-1], blame_regex)
else
call s:throw("Can't browse to uncommitted change")
endif
endif
endif
endif
endif
if empty(commit)
2019-08-22 15:36:17 +00:00
let commit = readfile(fugitive#Find('.git/HEAD', dir), '', 1)[0]
2018-06-14 10:31:12 +00:00
endif
2018-08-25 16:13:42 +00:00
let i = 0
while commit =~# '^ref: ' && i < 10
2020-01-07 12:45:07 +00:00
let ref_file = cdir . '/' . commit[5:-1]
if getfsize(ref_file) > 0
let commit = readfile(ref_file, '', 1)[0]
else
let commit = fugitive#RevParse(commit[5:-1], dir)
endif
2018-08-25 16:13:42 +00:00
let i -= 1
endwhile
2018-06-14 10:31:12 +00:00
endif
if empty(remote)
let remote = '.'
endif
2018-07-30 21:18:16 +00:00
let raw = fugitive#RemoteUrl(remote)
if empty(raw)
2018-06-14 10:31:12 +00:00
let raw = remote
endif
if raw =~# '^https\=://' && s:executable('curl')
if !has_key(s:redirects, raw)
let s:redirects[raw] = matchstr(system('curl -I ' .
\ s:shellesc(raw . '/info/refs?service=git-upload-pack')),
\ 'Location: \zs\S\+\ze/info/refs?')
endif
if len(s:redirects[raw])
let raw = s:redirects[raw]
endif
endif
2018-07-30 21:18:16 +00:00
let opts = {
2019-03-08 11:04:56 +00:00
\ 'dir': dir,
\ 'repo': fugitive#repo(dir),
2018-07-30 21:18:16 +00:00
\ 'remote': raw,
\ 'revision': 'No longer provided',
\ 'commit': commit,
\ 'path': path,
\ 'type': type,
2018-08-25 16:13:42 +00:00
\ 'line1': line1,
\ 'line2': line2}
2018-07-30 21:18:16 +00:00
2018-09-25 00:40:17 +00:00
let url = ''
2018-07-30 21:18:16 +00:00
for Handler in get(g:, 'fugitive_browse_handlers', [])
let url = call(Handler, [copy(opts)])
2018-06-14 10:31:12 +00:00
if !empty(url)
break
endif
endfor
2018-07-30 21:18:16 +00:00
if empty(url)
2019-08-22 15:36:17 +00:00
call s:throw("No Gbrowse handler installed for '".raw."'")
2018-06-14 10:31:12 +00:00
endif
let url = s:gsub(url, '[ <>]', '\="%".printf("%02X",char2nr(submatch(0)))')
if a:bang
if has('clipboard')
let @+ = url
endif
return 'echomsg '.string(url)
elseif exists(':Browse') == 2
return 'echomsg '.string(url).'|Browse '.url
else
if !exists('g:loaded_netrw')
runtime! autoload/netrw.vim
endif
if exists('*netrw#BrowseX')
return 'echomsg '.string(url).'|call netrw#BrowseX('.string(url).', 0)'
else
return 'echomsg '.string(url).'|call netrw#NetrwBrowseX('.string(url).', 0)'
endif
endif
catch /^fugitive:/
2019-08-22 15:36:17 +00:00
return 'echoerr ' . string(v:exception)
2018-06-14 10:31:12 +00:00
endtry
endfunction
2018-07-30 21:18:16 +00:00
" Section: Go to file
2018-06-14 10:31:12 +00:00
2018-07-30 21:18:16 +00:00
nnoremap <SID>: :<C-U><C-R>=v:count ? v:count : ''<CR>
function! fugitive#MapCfile(...) abort
exe 'cnoremap <buffer> <expr> <Plug><cfile>' (a:0 ? a:1 : 'fugitive#Cfile()')
let b:undo_ftplugin = get(b:, 'undo_ftplugin', 'exe') . '|sil! exe "cunmap <buffer> <Plug><cfile>"'
if !exists('g:fugitive_no_maps')
2019-08-22 15:36:17 +00:00
call s:Map('n', 'gf', '<SID>:find <Plug><cfile><CR>', '<silent><unique>', 1)
call s:Map('n', '<C-W>f', '<SID>:sfind <Plug><cfile><CR>', '<silent><unique>', 1)
call s:Map('n', '<C-W><C-F>', '<SID>:sfind <Plug><cfile><CR>', '<silent><unique>', 1)
call s:Map('n', '<C-W>gf', '<SID>:tabfind <Plug><cfile><CR>', '<silent><unique>', 1)
call s:Map('c', '<C-R><C-F>', '<Plug><cfile>', '<silent><unique>', 1)
2018-06-14 10:31:12 +00:00
endif
endfunction
2018-07-30 21:18:16 +00:00
function! s:ContainingCommit() abort
2018-08-25 16:13:42 +00:00
let commit = s:Owner(@%)
return empty(commit) ? 'HEAD' : commit
2018-06-14 10:31:12 +00:00
endfunction
2019-08-22 15:36:17 +00:00
function! s:SquashArgument(...) abort
2019-01-08 10:11:54 +00:00
if &filetype == 'fugitive'
2019-08-22 15:36:17 +00:00
let commit = matchstr(getline('.'), '^\%(\%(\x\x\x\)\@!\l\+\s\+\)\=\zs[0-9a-f]\{4,\}\ze ')
elseif has_key(s:temp_files, s:cpath(expand('%:p')))
let commit = matchstr(getline('.'), '\<\x\{4,\}\>')
2019-01-08 10:11:54 +00:00
else
2019-08-22 15:36:17 +00:00
let commit = s:Owner(@%)
2019-01-08 10:11:54 +00:00
endif
2019-08-22 15:36:17 +00:00
return len(commit) && a:0 ? printf(a:1, commit) : commit
endfunction
function! s:RebaseArgument() abort
return s:SquashArgument(' %s^')
2019-01-08 10:11:54 +00:00
endfunction
2018-07-30 21:18:16 +00:00
function! s:NavigateUp(count) abort
2018-08-25 16:13:42 +00:00
let rev = substitute(s:DirRev(@%)[1], '^$', ':', 'g')
2018-07-30 21:18:16 +00:00
let c = a:count
while c
if rev =~# ':.*/.'
let rev = matchstr(rev, '.*\ze/.\+', '')
elseif rev =~# '.:.'
let rev = matchstr(rev, '^.[^:]*:')
elseif rev =~# '^:'
let rev = 'HEAD^{}'
elseif rev =~# ':$'
let rev = rev[0:-2]
2018-06-14 10:31:12 +00:00
else
2018-07-30 21:18:16 +00:00
return rev.'~'.c
2018-06-14 10:31:12 +00:00
endif
2018-07-30 21:18:16 +00:00
let c -= 1
endwhile
return rev
2018-06-14 10:31:12 +00:00
endfunction
2019-08-22 15:36:17 +00:00
function! s:MapMotion(lhs, rhs) abort
call s:Map('n', a:lhs, ":<C-U>" . a:rhs . "<CR>", "<silent>")
call s:Map('o', a:lhs, ":<C-U>" . a:rhs . "<CR>", "<silent>")
call s:Map('x', a:lhs, ":<C-U>exe 'normal! gv'<Bar>" . a:rhs . "<CR>", "<silent>")
endfunction
2018-07-30 21:18:16 +00:00
function! fugitive#MapJumps(...) abort
if !&modifiable
if get(b:, 'fugitive_type', '') ==# 'blob'
2019-08-22 15:36:17 +00:00
let blame_map = 'Gblame<C-R>=v:count ? " --reverse" : ""<CR><CR>'
call s:Map('n', '<2-LeftMouse>', ':<C-U>0,1' . blame_map, '<silent>')
call s:Map('n', '<CR>', ':<C-U>0,1' . blame_map, '<silent>')
call s:Map('n', 'o', ':<C-U>0,2' . blame_map, '<silent>')
call s:Map('n', 'p', ':<C-U>0,3' . blame_map, '<silent>')
call s:Map('n', 'gO', ':<C-U>0,4' . blame_map, '<silent>')
call s:Map('n', 'O', ':<C-U>0,5' . blame_map, '<silent>')
call s:Map('n', 'D', ":<C-U>call <SID>DiffClose()<Bar>Gdiffsplit!<Bar>redraw<Bar>echohl WarningMsg<Bar> echo ':Gstatus D is deprecated in favor of dd'<Bar>echohl NONE<CR>", '<silent>')
call s:Map('n', 'dd', ":<C-U>call <SID>DiffClose()<Bar>Gdiffsplit!<CR>", '<silent>')
call s:Map('n', 'dh', ":<C-U>call <SID>DiffClose()<Bar>Ghdiffsplit!<CR>", '<silent>')
call s:Map('n', 'ds', ":<C-U>call <SID>DiffClose()<Bar>Ghdiffsplit!<CR>", '<silent>')
call s:Map('n', 'dv', ":<C-U>call <SID>DiffClose()<Bar>Gvdiffsplit!<CR>", '<silent>')
call s:Map('n', 'd?', ":<C-U>help fugitive_d<CR>", '<silent>')
2018-06-14 10:31:12 +00:00
else
2019-08-22 15:36:17 +00:00
call s:Map('n', '<2-LeftMouse>', ':<C-U>exe <SID>GF("edit")<CR>', '<silent>')
call s:Map('n', '<CR>', ':<C-U>exe <SID>GF("edit")<CR>', '<silent>')
call s:Map('n', 'o', ':<C-U>exe <SID>GF("split")<CR>', '<silent>')
call s:Map('n', 'gO', ':<C-U>exe <SID>GF("vsplit")<CR>', '<silent>')
call s:Map('n', 'O', ':<C-U>exe <SID>GF("tabedit")<CR>', '<silent>')
call s:Map('n', 'p', ':<C-U>exe <SID>GF("pedit")<CR>', '<silent>')
if !exists('g:fugitive_no_maps')
if exists(':CtrlP') && get(g:, 'ctrl_p_map') =~? '^<c-p>$'
nnoremap <buffer> <silent> <C-P> :<C-U>execute line('.') == 1 ? 'CtrlP ' . fnameescape(<SID>Tree()) : <SID>PreviousItem(v:count1)<CR>
else
nnoremap <buffer> <silent> <C-P> :<C-U>execute <SID>PreviousItem(v:count1)<CR>
endif
nnoremap <buffer> <silent> <C-N> :<C-U>execute <SID>NextItem(v:count1)<CR>
endif
call s:MapMotion('(', 'exe <SID>PreviousItem(v:count1)')
call s:MapMotion(')', 'exe <SID>NextItem(v:count1)')
call s:MapMotion('K', 'exe <SID>PreviousHunk(v:count1)')
call s:MapMotion('J', 'exe <SID>NextHunk(v:count1)')
call s:MapMotion('[c', 'exe <SID>PreviousHunk(v:count1)')
call s:MapMotion(']c', 'exe <SID>NextHunk(v:count1)')
call s:MapMotion('[/', 'exe <SID>PreviousFile(v:count1)')
call s:MapMotion(']/', 'exe <SID>NextFile(v:count1)')
call s:MapMotion('[m', 'exe <SID>PreviousFile(v:count1)')
call s:MapMotion(']m', 'exe <SID>NextFile(v:count1)')
call s:MapMotion('[[', 'exe <SID>PreviousSection(v:count1)')
call s:MapMotion(']]', 'exe <SID>NextSection(v:count1)')
call s:MapMotion('[]', 'exe <SID>PreviousSectionEnd(v:count1)')
call s:MapMotion('][', 'exe <SID>NextSectionEnd(v:count1)')
call s:Map('nxo', '*', '<SID>PatchSearchExpr(0)', '<expr>')
call s:Map('nxo', '#', '<SID>PatchSearchExpr(1)', '<expr>')
endif
call s:Map('n', 'S', ':<C-U>echoerr "Use gO"<CR>', '<silent>')
call s:Map('n', 'dq', ":<C-U>call <SID>DiffClose()<CR>", '<silent>')
call s:Map('n', '-', ":<C-U>exe 'Gedit ' . <SID>fnameescape(<SID>NavigateUp(v:count1))<Bar> if getline(1) =~# '^tree \x\{40,\}$' && empty(getline(2))<Bar>call search('^'.escape(expand('#:t'),'.*[]~\').'/\=$','wc')<Bar>endif<CR>", '<silent>')
call s:Map('n', 'P', ":<C-U>exe 'Gedit ' . <SID>fnameescape(<SID>ContainingCommit().'^'.v:count1.<SID>Relative(':'))<CR>", '<silent>')
call s:Map('n', '~', ":<C-U>exe 'Gedit ' . <SID>fnameescape(<SID>ContainingCommit().'~'.v:count1.<SID>Relative(':'))<CR>", '<silent>')
call s:Map('n', 'C', ":<C-U>exe 'Gedit ' . <SID>fnameescape(<SID>ContainingCommit())<CR>", '<silent>')
call s:Map('n', 'cp', ":<C-U>echoerr 'Use gC'<CR>", '<silent>')
call s:Map('n', 'gC', ":<C-U>exe 'Gpedit ' . <SID>fnameescape(<SID>ContainingCommit())<CR>", '<silent>')
call s:Map('n', 'gc', ":<C-U>exe 'Gpedit ' . <SID>fnameescape(<SID>ContainingCommit())<CR>", '<silent>')
call s:Map('n', 'gi', ":<C-U>exe 'Gsplit' (v:count ? '.gitignore' : '.git/info/exclude')<CR>", '<silent>')
call s:Map('x', 'gi', ":<C-U>exe 'Gsplit' (v:count ? '.gitignore' : '.git/info/exclude')<CR>", '<silent>')
2019-11-16 15:28:42 +00:00
nnoremap <buffer> c<Space> :Git commit<Space>
nnoremap <buffer> c<CR> :Git commit<CR>
2020-01-29 02:07:36 +00:00
nnoremap <buffer> cv<Space> :tab Git commit -v<Space>
nnoremap <buffer> cv<CR> :tab Git commit -v<CR>
2019-03-08 11:04:56 +00:00
nnoremap <buffer> <silent> ca :<C-U>Gcommit --amend<CR>
nnoremap <buffer> <silent> cc :<C-U>Gcommit<CR>
nnoremap <buffer> <silent> ce :<C-U>Gcommit --amend --no-edit<CR>
nnoremap <buffer> <silent> cw :<C-U>Gcommit --amend --only<CR>
2020-01-29 02:07:36 +00:00
nnoremap <buffer> <silent> cva :<C-U>tab Gcommit -v --amend<CR>
nnoremap <buffer> <silent> cvc :<C-U>tab Gcommit -v<CR>
2019-08-22 15:36:17 +00:00
nnoremap <buffer> <silent> cRa :<C-U>Gcommit --reset-author --amend<CR>
nnoremap <buffer> <silent> cRe :<C-U>Gcommit --reset-author --amend --no-edit<CR>
nnoremap <buffer> <silent> cRw :<C-U>Gcommit --reset-author --amend --only<CR>
2019-01-08 10:11:54 +00:00
nnoremap <buffer> cf :<C-U>Gcommit --fixup=<C-R>=<SID>SquashArgument()<CR>
2019-08-22 15:36:17 +00:00
nnoremap <buffer> cF :<C-U><Bar>Grebase --autosquash<C-R>=<SID>RebaseArgument()<CR><Home>Gcommit --fixup=<C-R>=<SID>SquashArgument()<CR>
2019-11-16 15:28:42 +00:00
nnoremap <buffer> cs :<C-U>Gcommit --no-edit --squash=<C-R>=<SID>SquashArgument()<CR>
nnoremap <buffer> cS :<C-U><Bar>Grebase --autosquash<C-R>=<SID>RebaseArgument()<CR><Home>Gcommit --no-edit --squash=<C-R>=<SID>SquashArgument()<CR>
2019-01-08 10:11:54 +00:00
nnoremap <buffer> cA :<C-U>Gcommit --edit --squash=<C-R>=<SID>SquashArgument()<CR>
2019-08-22 15:36:17 +00:00
nnoremap <buffer> <silent> c? :<C-U>help fugitive_c<CR>
2019-11-16 15:28:42 +00:00
nnoremap <buffer> cr<Space> :Git revert<Space>
nnoremap <buffer> cr<CR> :Git revert<CR>
2019-08-22 15:36:17 +00:00
nnoremap <buffer> <silent> crc :<C-U>Grevert <C-R>=<SID>SquashArgument()<CR><CR>
nnoremap <buffer> <silent> crn :<C-U>Grevert --no-commit <C-R>=<SID>SquashArgument()<CR><CR>
nnoremap <buffer> <silent> cr? :help fugitive_cr<CR>
2019-11-16 15:28:42 +00:00
nnoremap <buffer> cm<Space> :Git merge<Space>
nnoremap <buffer> cm<CR> :Git merge<CR>
2020-01-29 02:07:36 +00:00
nnoremap <buffer> cmt :Git mergetool
2019-08-22 15:36:17 +00:00
nnoremap <buffer> <silent> cm? :help fugitive_cm<CR>
2019-11-16 15:28:42 +00:00
nnoremap <buffer> cz<Space> :Git stash<Space>
nnoremap <buffer> cz<CR> :Git stash<CR>
2019-08-22 15:36:17 +00:00
nnoremap <buffer> <silent> cza :<C-U>exe <SID>EchoExec(['stash', 'apply', '--quiet', '--index', 'stash@{' . v:count . '}'])<CR>
nnoremap <buffer> <silent> czA :<C-U>exe <SID>EchoExec(['stash', 'apply', '--quiet', 'stash@{' . v:count . '}'])<CR>
nnoremap <buffer> <silent> czp :<C-U>exe <SID>EchoExec(['stash', 'pop', '--quiet', '--index', 'stash@{' . v:count . '}'])<CR>
nnoremap <buffer> <silent> czP :<C-U>exe <SID>EchoExec(['stash', 'pop', '--quiet', 'stash@{' . v:count . '}'])<CR>
nnoremap <buffer> <silent> czv :<C-U>exe 'Gedit' fugitive#RevParse('stash@{' . v:count . '}')<CR>
nnoremap <buffer> <silent> czw :<C-U>exe <SID>EchoExec(['stash', '--keep-index'] + (v:count > 1 ? ['--all'] : v:count ? ['--include-untracked'] : []))<CR>
nnoremap <buffer> <silent> czz :<C-U>exe <SID>EchoExec(['stash'] + (v:count > 1 ? ['--all'] : v:count ? ['--include-untracked'] : []))<CR>
nnoremap <buffer> <silent> cz? :<C-U>help fugitive_cz<CR>
2019-11-16 15:28:42 +00:00
nnoremap <buffer> co<Space> :Git checkout<Space>
nnoremap <buffer> co<CR> :Git checkout<CR>
2019-08-22 15:36:17 +00:00
nnoremap <buffer> coo :exe <SID>EchoExec(['checkout'] + split(<SID>SquashArgument()) + ['--'])<CR>
nnoremap <buffer> co? :<C-U>help fugitive_co<CR>
2019-11-16 15:28:42 +00:00
nnoremap <buffer> cb<Space> :Git branch<Space>
nnoremap <buffer> cb<CR> :Git branch<CR>
2019-08-22 15:36:17 +00:00
nnoremap <buffer> cb? :<C-U>help fugitive_cb<CR>
2019-11-16 15:28:42 +00:00
nnoremap <buffer> r<Space> :Git rebase<Space>
nnoremap <buffer> r<CR> :Git rebase<CR>
2019-08-22 15:36:17 +00:00
nnoremap <buffer> <silent> ri :<C-U>Grebase --interactive<C-R>=<SID>RebaseArgument()<CR><CR>
nnoremap <buffer> <silent> rf :<C-U>Grebase --autosquash<C-R>=<SID>RebaseArgument()<CR><CR>
2019-01-08 10:11:54 +00:00
nnoremap <buffer> <silent> ru :<C-U>Grebase --interactive @{upstream}<CR>
nnoremap <buffer> <silent> rp :<C-U>Grebase --interactive @{push}<CR>
2019-08-22 15:36:17 +00:00
nnoremap <buffer> <silent> rw :<C-U>Grebase --interactive<C-R>=<SID>RebaseArgument()<CR><Bar>s/^pick/reword/e<CR>
nnoremap <buffer> <silent> rm :<C-U>Grebase --interactive<C-R>=<SID>RebaseArgument()<CR><Bar>s/^pick/edit/e<CR>
nnoremap <buffer> <silent> rd :<C-U>Grebase --interactive<C-R>=<SID>RebaseArgument()<CR><Bar>s/^pick/drop/e<CR>
nnoremap <buffer> <silent> rk :<C-U>Grebase --interactive<C-R>=<SID>RebaseArgument()<CR><Bar>s/^pick/drop/e<CR>
nnoremap <buffer> <silent> rx :<C-U>Grebase --interactive<C-R>=<SID>RebaseArgument()<CR><Bar>s/^pick/drop/e<CR>
2019-01-08 10:11:54 +00:00
nnoremap <buffer> <silent> rr :<C-U>Grebase --continue<CR>
nnoremap <buffer> <silent> rs :<C-U>Grebase --skip<CR>
nnoremap <buffer> <silent> re :<C-U>Grebase --edit-todo<CR>
nnoremap <buffer> <silent> ra :<C-U>Grebase --abort<CR>
2019-08-22 15:36:17 +00:00
nnoremap <buffer> <silent> r? :<C-U>help fugitive_r<CR>
call s:Map('n', '.', ":<C-U> <C-R>=<SID>fnameescape(fugitive#Real(@%))<CR><Home>")
call s:Map('x', '.', ":<C-U> <C-R>=<SID>fnameescape(fugitive#Real(@%))<CR><Home>")
2019-11-16 15:28:42 +00:00
call s:Map('n', 'g?', ":<C-U>help fugitive-map<CR>", '<silent>')
call s:Map('n', '<F1>', ":<C-U>help fugitive-map<CR>", '<silent>')
2018-07-30 21:18:16 +00:00
endif
2018-06-14 10:31:12 +00:00
endfunction
2018-07-30 21:18:16 +00:00
function! s:StatusCfile(...) abort
2019-03-08 11:04:56 +00:00
let tree = s:Tree()
2019-01-08 10:11:54 +00:00
let lead = s:cpath(tree, getcwd()) ? './' : tree . '/'
let info = s:StageInfo()
let line = getline('.')
2019-08-22 15:36:17 +00:00
if len(info.sigil) && len(info.section) && len(info.paths)
2019-01-08 10:11:54 +00:00
if info.section ==# 'Unstaged' && info.sigil !=# '-'
2019-08-22 15:36:17 +00:00
return [lead . info.relative[0], info.offset, 'normal!zv']
2019-01-08 10:11:54 +00:00
elseif info.section ==# 'Staged' && info.sigil ==# '-'
2019-08-22 15:36:17 +00:00
return ['@:' . info.relative[0], info.offset, 'normal!zv']
2019-01-08 10:11:54 +00:00
else
2019-08-22 15:36:17 +00:00
return [':0:' . info.relative[0], info.offset, 'normal!zv']
2019-01-08 10:11:54 +00:00
endif
2019-08-22 15:36:17 +00:00
elseif len(info.paths)
return [lead . info.relative[0]]
2019-01-08 10:11:54 +00:00
elseif len(info.commit)
return [info.commit]
elseif line =~# '^\%(Head\|Merge\|Rebase\|Upstream\|Pull\|Push\): '
return [matchstr(line, ' \zs.*')]
else
return ['']
endif
endfunction
function! fugitive#StatusCfile() abort
let file = s:Generate(s:StatusCfile()[0])
return empty(file) ? fugitive#Cfile() : s:fnameescape(file)
endfunction
function! s:MessageCfile(...) abort
2019-03-08 11:04:56 +00:00
let tree = s:Tree()
2018-08-25 16:13:42 +00:00
let lead = s:cpath(tree, getcwd()) ? './' : tree . '/'
2018-07-30 21:18:16 +00:00
if getline('.') =~# '^.\=\trenamed:.* -> '
2018-08-25 16:13:42 +00:00
return lead . matchstr(getline('.'),' -> \zs.*')
2018-07-30 21:18:16 +00:00
elseif getline('.') =~# '^.\=\t\(\k\| \)\+\p\?: *.'
2018-08-25 16:13:42 +00:00
return lead . matchstr(getline('.'),': *\zs.\{-\}\ze\%( ([^()[:digit:]]\+)\)\=$')
2018-07-30 21:18:16 +00:00
elseif getline('.') =~# '^.\=\t.'
2018-08-25 16:13:42 +00:00
return lead . matchstr(getline('.'),'\t\zs.*')
2018-07-30 21:18:16 +00:00
elseif getline('.') =~# ': needs merge$'
2018-08-25 16:13:42 +00:00
return lead . matchstr(getline('.'),'.*\ze: needs merge$')
2018-07-30 21:18:16 +00:00
elseif getline('.') =~# '^\%(. \)\=Not currently on any branch.$'
return 'HEAD'
elseif getline('.') =~# '^\%(. \)\=On branch '
return 'refs/heads/'.getline('.')[12:]
elseif getline('.') =~# "^\\%(. \\)\=Your branch .*'"
return matchstr(getline('.'),"'\\zs\\S\\+\\ze'")
else
2018-06-14 10:31:12 +00:00
return ''
endif
endfunction
2019-01-08 10:11:54 +00:00
function! fugitive#MessageCfile() abort
let file = s:Generate(s:MessageCfile())
2018-09-25 00:40:17 +00:00
return empty(file) ? fugitive#Cfile() : s:fnameescape(file)
2018-06-14 10:31:12 +00:00
endfunction
function! s:cfile() abort
try
2018-07-30 21:18:16 +00:00
let myhash = s:DirRev(@%)[1]
if len(myhash)
try
let myhash = fugitive#RevParse(myhash)
catch /^fugitive:/
let myhash = ''
endtry
endif
if empty(myhash) && getline(1) =~# '^\%(commit\|tag\) \w'
2018-06-14 10:31:12 +00:00
let myhash = matchstr(getline(1),'^\w\+ \zs\S\+')
endif
2018-07-30 21:18:16 +00:00
let showtree = (getline(1) =~# '^tree ' && getline(2) == "")
2018-06-14 10:31:12 +00:00
2018-07-30 21:18:16 +00:00
let treebase = substitute(s:DirCommitFile(@%)[1], '^\d$', ':&', '') . ':' .
\ s:Relative('') . (s:Relative('') =~# '^$\|/$' ? '' : '/')
2019-08-22 15:36:17 +00:00
if getline('.') =~# '^\d\{6\} \l\{3,8\} \x\{40,\}\t'
2018-07-30 21:18:16 +00:00
return [treebase . s:sub(matchstr(getline('.'),'\t\zs.*'),'/$','')]
elseif showtree
return [treebase . s:sub(getline('.'),'/$','')]
2018-06-14 10:31:12 +00:00
else
let dcmds = []
" Index
2019-08-22 15:36:17 +00:00
if getline('.') =~# '^\d\{6\} \x\{40,\} \d\t'
let ref = matchstr(getline('.'),'\x\{40,\}')
2018-06-14 10:31:12 +00:00
let file = ':'.s:sub(matchstr(getline('.'),'\d\t.*'),'\t',':')
return [file]
endif
if getline('.') =~# '^ref: '
let ref = strpart(getline('.'),5)
2019-08-22 15:36:17 +00:00
elseif getline('.') =~# '^commit \x\{40,\}\>'
let ref = matchstr(getline('.'),'\x\{40,\}')
2018-06-14 10:31:12 +00:00
return [ref]
2019-08-22 15:36:17 +00:00
elseif getline('.') =~# '^parent \x\{40,\}\>'
let ref = matchstr(getline('.'),'\x\{40,\}')
2018-06-14 10:31:12 +00:00
let line = line('.')
let parent = 0
while getline(line) =~# '^parent '
let parent += 1
let line -= 1
endwhile
return [ref]
2019-08-22 15:36:17 +00:00
elseif getline('.') =~# '^tree \x\{40,\}$'
let ref = matchstr(getline('.'),'\x\{40,\}')
2018-07-30 21:18:16 +00:00
if len(myhash) && fugitive#RevParse(myhash.':') ==# ref
2018-06-14 10:31:12 +00:00
let ref = myhash.':'
endif
return [ref]
2019-08-22 15:36:17 +00:00
elseif getline('.') =~# '^object \x\{40,\}$' && getline(line('.')+1) =~ '^type \%(commit\|tree\|blob\)$'
let ref = matchstr(getline('.'),'\x\{40,\}')
2018-06-14 10:31:12 +00:00
let type = matchstr(getline(line('.')+1),'type \zs.*')
elseif getline('.') =~# '^\l\{3,8\} '.myhash.'$'
2018-08-25 16:13:42 +00:00
let ref = s:DirRev(@%)[1]
2018-06-14 10:31:12 +00:00
2019-08-22 15:36:17 +00:00
elseif getline('.') =~# '^\l\{3,8\} \x\{40,\}\>'
let ref = matchstr(getline('.'),'\x\{40,\}')
2018-06-14 10:31:12 +00:00
echoerr "warning: unknown context ".matchstr(getline('.'),'^\l*')
elseif getline('.') =~# '^[+-]\{3\} [abciow12]\=/'
let ref = getline('.')[4:]
2018-07-04 10:53:25 +00:00
elseif getline('.') =~# '^[+-]' && search('^@@ -\d\+\%(,\d\+\)\= +\d\+','bnW')
2018-06-14 10:31:12 +00:00
let type = getline('.')[0]
let lnum = line('.') - 1
let offset = 0
2018-07-04 10:53:25 +00:00
while getline(lnum) !~# '^@@ -\d\+\%(,\d\+\)\= +\d\+'
2018-06-14 10:31:12 +00:00
if getline(lnum) =~# '^[ '.type.']'
let offset += 1
endif
let lnum -= 1
endwhile
let offset += matchstr(getline(lnum), type.'\zs\d\+')
let ref = getline(search('^'.type.'\{3\} [abciow12]/','bnW'))[4:-1]
let dcmds = [offset, 'normal!zv']
elseif getline('.') =~# '^rename from '
let ref = 'a/'.getline('.')[12:]
elseif getline('.') =~# '^rename to '
let ref = 'b/'.getline('.')[10:]
2018-07-04 10:53:25 +00:00
elseif getline('.') =~# '^@@ -\d\+\%(,\d\+\)\= +\d\+'
2018-06-14 10:31:12 +00:00
let diff = getline(search('^diff --git \%([abciow12]/.*\|/dev/null\) \%([abciow12]/.*\|/dev/null\)', 'bcnW'))
let offset = matchstr(getline('.'), '+\zs\d\+')
let dref = matchstr(diff, '\Cdiff --git \zs\%([abciow12]/.*\|/dev/null\)\ze \%([abciow12]/.*\|/dev/null\)')
let ref = matchstr(diff, '\Cdiff --git \%([abciow12]/.*\|/dev/null\) \zs\%([abciow12]/.*\|/dev/null\)')
2019-08-22 15:36:17 +00:00
let dcmd = 'Gdiffsplit! +'.offset
2018-06-14 10:31:12 +00:00
elseif getline('.') =~# '^diff --git \%([abciow12]/.*\|/dev/null\) \%([abciow12]/.*\|/dev/null\)'
let dref = matchstr(getline('.'),'\Cdiff --git \zs\%([abciow12]/.*\|/dev/null\)\ze \%([abciow12]/.*\|/dev/null\)')
let ref = matchstr(getline('.'),'\Cdiff --git \%([abciow12]/.*\|/dev/null\) \zs\%([abciow12]/.*\|/dev/null\)')
2019-08-22 15:36:17 +00:00
let dcmd = 'Gdiffsplit!'
2018-06-14 10:31:12 +00:00
elseif getline('.') =~# '^index ' && getline(line('.')-1) =~# '^diff --git \%([abciow12]/.*\|/dev/null\) \%([abciow12]/.*\|/dev/null\)'
let line = getline(line('.')-1)
let dref = matchstr(line,'\Cdiff --git \zs\%([abciow12]/.*\|/dev/null\)\ze \%([abciow12]/.*\|/dev/null\)')
let ref = matchstr(line,'\Cdiff --git \%([abciow12]/.*\|/dev/null\) \zs\%([abciow12]/.*\|/dev/null\)')
2019-08-22 15:36:17 +00:00
let dcmd = 'Gdiffsplit!'
2018-06-14 10:31:12 +00:00
2019-08-22 15:36:17 +00:00
elseif line('$') == 1 && getline('.') =~ '^\x\{40,\}$'
2018-06-14 10:31:12 +00:00
let ref = getline('.')
2019-08-22 15:36:17 +00:00
elseif expand('<cword>') =~# '^\x\{7,\}\>'
2018-06-14 10:31:12 +00:00
return [expand('<cword>')]
else
let ref = ''
endif
let prefixes = {
\ '1': '',
\ '2': '',
\ 'b': ':0:',
\ 'i': ':0:',
\ 'o': '',
\ 'w': ''}
if len(myhash)
let prefixes.a = myhash.'^:'
let prefixes.b = myhash.':'
endif
let ref = substitute(ref, '^\(\w\)/', '\=get(prefixes, submatch(1), "HEAD:")', '')
if exists('dref')
let dref = substitute(dref, '^\(\w\)/', '\=get(prefixes, submatch(1), "HEAD:")', '')
endif
if ref ==# '/dev/null'
" Empty blob
let ref = 'e69de29bb2d1d6434b8b29ae775ad8c2e48c5391'
endif
if exists('dref')
return [ref, dcmd . ' ' . s:fnameescape(dref)] + dcmds
elseif ref != ""
return [ref] + dcmds
endif
endif
return []
endtry
endfunction
function! s:GF(mode) abort
try
2019-01-08 10:11:54 +00:00
let results = &filetype ==# 'fugitive' ? s:StatusCfile() : &filetype ==# 'gitcommit' ? [s:MessageCfile()] : s:cfile()
2018-06-14 10:31:12 +00:00
catch /^fugitive:/
2019-08-22 15:36:17 +00:00
return 'echoerr ' . string(v:exception)
2018-06-14 10:31:12 +00:00
endtry
2018-07-30 21:18:16 +00:00
if len(results) > 1
return 'G' . a:mode .
2019-08-22 15:36:17 +00:00
\ ' +' . escape(results[1], ' ') . ' ' .
\ s:fnameescape(results[0]) . join(map(results[2:-1], '"|" . v:val'), '')
2019-01-08 10:11:54 +00:00
elseif len(results) && len(results[0])
2018-07-30 21:18:16 +00:00
return 'G' . a:mode . ' ' . s:fnameescape(results[0])
2018-06-14 10:31:12 +00:00
else
return ''
endif
endfunction
function! fugitive#Cfile() abort
let pre = ''
let results = s:cfile()
if empty(results)
let cfile = expand('<cfile>')
if &includeexpr =~# '\<v:fname\>'
sandbox let cfile = eval(substitute(&includeexpr, '\C\<v:fname\>', '\=string(cfile)', 'g'))
endif
return cfile
elseif len(results) > 1
let pre = '+' . join(map(results[1:-1], 'escape(v:val, " ")'), '\|') . ' '
endif
2018-07-30 21:18:16 +00:00
return pre . s:fnameescape(s:Generate(results[0]))
2018-06-14 10:31:12 +00:00
endfunction
" Section: Statusline
function! fugitive#Statusline(...) abort
2019-11-16 15:28:42 +00:00
let dir = s:Dir(bufnr(''))
if empty(dir)
2018-06-14 10:31:12 +00:00
return ''
endif
let status = ''
2018-07-30 21:18:16 +00:00
let commit = s:DirCommitFile(@%)[1]
if len(commit)
2018-09-25 00:40:17 +00:00
let status .= ':' . commit[0:6]
2018-06-14 10:31:12 +00:00
endif
2019-11-16 15:28:42 +00:00
let status .= '('.FugitiveHead(7, dir).')'
2018-07-30 21:18:16 +00:00
return '[Git'.status.']'
2018-06-14 10:31:12 +00:00
endfunction
function! fugitive#statusline(...) abort
return fugitive#Statusline()
endfunction
function! fugitive#head(...) abort
2019-03-08 11:04:56 +00:00
if empty(s:Dir())
2018-06-14 10:31:12 +00:00
return ''
endif
2018-08-25 16:13:42 +00:00
return fugitive#Head(a:0 ? a:1 : 0)
2018-06-14 10:31:12 +00:00
endfunction
" Section: Folding
function! fugitive#Foldtext() abort
if &foldmethod !=# 'syntax'
return foldtext()
endif
let line_foldstart = getline(v:foldstart)
if line_foldstart =~# '^diff '
let [add, remove] = [-1, -1]
let filename = ''
for lnum in range(v:foldstart, v:foldend)
let line = getline(lnum)
if filename ==# '' && line =~# '^[+-]\{3\} [abciow12]/'
let filename = line[6:-1]
endif
if line =~# '^+'
let add += 1
elseif line =~# '^-'
let remove += 1
elseif line =~# '^Binary '
let binary = 1
endif
endfor
if filename ==# ''
let filename = matchstr(line_foldstart, '^diff .\{-\} [abciow12]/\zs.*\ze [abciow12]/')
endif
if filename ==# ''
let filename = line_foldstart[5:-1]
endif
if exists('binary')
return 'Binary: '.filename
else
return (add<10&&remove<100?' ':'') . add . '+ ' . (remove<10&&add<100?' ':'') . remove . '- ' . filename
endif
elseif line_foldstart =~# '^# .*:$'
let lines = getline(v:foldstart, v:foldend)
call filter(lines, 'v:val =~# "^#\t"')
cal map(lines, "s:sub(v:val, '^#\t%(modified: +|renamed: +)=', '')")
cal map(lines, "s:sub(v:val, '^([[:alpha:] ]+): +(.*)', '\\2 (\\1)')")
return line_foldstart.' '.join(lines, ', ')
endif
return foldtext()
endfunction
function! fugitive#foldtext() abort
return fugitive#Foldtext()
endfunction
2018-07-30 21:18:16 +00:00
" Section: Initialization
function! fugitive#Init() abort
2020-01-29 02:07:36 +00:00
throw 'Third party code is using fugitive#Init() which has been removed. Contact the author if you have a reason to still use it'
2018-07-30 21:18:16 +00:00
endfunction
function! fugitive#is_git_dir(path) abort
2020-01-29 02:07:36 +00:00
throw 'Third party code is using fugitive#is_git_dir() which has been removed. Change it to FugitiveIsGitDir()'
2018-07-30 21:18:16 +00:00
endfunction
function! fugitive#extract_git_dir(path) abort
2020-01-29 02:07:36 +00:00
throw 'Third party code is using fugitive#extract_git_dir() which has been removed. Change it to FugitiveExtractGitDir()'
2018-07-30 21:18:16 +00:00
endfunction
function! fugitive#detect(path) abort
2020-01-29 02:07:36 +00:00
throw 'Third party code is using fugitive#detect() which has been removed. Contact the author if you have a reason to still use it'
2018-07-30 21:18:16 +00:00
endfunction
2018-08-25 16:13:42 +00:00
" Section: End