2016-01-14 21:38:38 -05:00
|
|
|
" MIT License. Copyright (c) 2013-2016 Bailey Ling.
|
2013-08-18 14:34:02 -04:00
|
|
|
" vim: et ts=2 sts=2 sw=2
|
|
|
|
|
2016-09-23 20:16:30 -04:00
|
|
|
scriptencoding utf-8
|
|
|
|
|
2013-08-18 14:34:02 -04:00
|
|
|
let s:has_fugitive = exists('*fugitive#head')
|
|
|
|
let s:has_lawrencium = exists('*lawrencium#statusline')
|
2013-11-12 09:45:04 -05:00
|
|
|
let s:has_vcscommand = get(g:, 'airline#extensions#branch#use_vcscommand', 0) && exists('*VCSCommandGetStatusLine')
|
2016-09-28 12:03:13 -04:00
|
|
|
let s:has_async = v:version >= 800 && has('job')
|
|
|
|
let s:git_cmd = 'git status --porcelain -- '
|
|
|
|
let s:hg_cmd = 'hg status -u -- '
|
2013-08-05 23:07:01 -04:00
|
|
|
|
2013-11-07 18:36:59 -05:00
|
|
|
if !s:has_fugitive && !s:has_lawrencium && !s:has_vcscommand
|
2013-09-10 11:37:25 -04:00
|
|
|
finish
|
|
|
|
endif
|
|
|
|
|
2016-02-02 16:45:47 -05:00
|
|
|
let s:git_dirs = {}
|
|
|
|
let s:untracked_git = {}
|
|
|
|
let s:untracked_hg = {}
|
|
|
|
|
2015-02-27 22:04:13 -05:00
|
|
|
let s:head_format = get(g:, 'airline#extensions#branch#format', 0)
|
|
|
|
if s:head_format == 1
|
|
|
|
function! s:format_name(name)
|
|
|
|
return fnamemodify(a:name, ':t')
|
|
|
|
endfunction
|
2015-11-19 05:03:54 -05:00
|
|
|
elseif s:head_format == 2
|
|
|
|
function! s:format_name(name)
|
|
|
|
return pathshorten(a:name)
|
|
|
|
endfunction
|
2015-02-27 22:04:13 -05:00
|
|
|
elseif type(s:head_format) == type('')
|
|
|
|
function! s:format_name(name)
|
|
|
|
return call(s:head_format, [a:name])
|
|
|
|
endfunction
|
|
|
|
else
|
|
|
|
function! s:format_name(name)
|
|
|
|
return a:name
|
|
|
|
endfunction
|
|
|
|
endif
|
|
|
|
|
2014-04-02 20:54:43 -04:00
|
|
|
function! s:get_git_branch(path)
|
2016-01-29 07:11:14 -05:00
|
|
|
if !s:has_fugitive
|
|
|
|
return ''
|
2014-04-02 20:54:43 -04:00
|
|
|
endif
|
|
|
|
|
2016-01-29 07:11:14 -05:00
|
|
|
let name = fugitive#head(7)
|
|
|
|
if empty(name)
|
|
|
|
if has_key(s:git_dirs, a:path)
|
|
|
|
return s:git_dirs[a:path]
|
|
|
|
endif
|
|
|
|
|
|
|
|
let dir = fugitive#extract_git_dir(a:path)
|
|
|
|
if empty(dir)
|
2014-04-02 20:54:43 -04:00
|
|
|
let name = ''
|
2016-01-29 07:11:14 -05:00
|
|
|
else
|
|
|
|
try
|
|
|
|
let line = join(readfile(dir . '/HEAD'))
|
|
|
|
if strpart(line, 0, 16) == 'ref: refs/heads/'
|
|
|
|
let name = strpart(line, 16)
|
|
|
|
else
|
|
|
|
" raw commit hash
|
|
|
|
let name = strpart(line, 0, 7)
|
|
|
|
endif
|
|
|
|
catch
|
|
|
|
let name = ''
|
|
|
|
endtry
|
|
|
|
endif
|
2014-04-02 20:54:43 -04:00
|
|
|
endif
|
|
|
|
|
|
|
|
let s:git_dirs[a:path] = name
|
|
|
|
return name
|
|
|
|
endfunction
|
|
|
|
|
2016-02-02 16:45:47 -05:00
|
|
|
function! s:get_git_untracked(file)
|
|
|
|
if empty(a:file)
|
2016-09-28 12:03:13 -04:00
|
|
|
return
|
2016-02-02 16:45:47 -05:00
|
|
|
endif
|
2016-09-28 12:03:13 -04:00
|
|
|
if !has_key(s:untracked_git, a:file)
|
|
|
|
if s:has_async
|
|
|
|
call s:DoAsync(s:git_cmd, a:file)
|
|
|
|
else
|
|
|
|
let output = system(s:git_cmd. shellescape(a:file))
|
|
|
|
if output[0:1] is# '??' && output[3:-2] is? a:file
|
|
|
|
let untracked = get(g:, 'airline#extensions#branch#notexists', g:airline_symbols.notexists)
|
|
|
|
endif
|
|
|
|
let s:untracked_git[a:file] = untracked
|
2016-02-05 17:07:47 -05:00
|
|
|
endif
|
2016-02-02 16:45:47 -05:00
|
|
|
endif
|
|
|
|
endfunction
|
|
|
|
|
|
|
|
function! s:get_hg_untracked(file)
|
2016-09-28 12:03:13 -04:00
|
|
|
if !s:has_lawrencium && empty(a:file)
|
|
|
|
return
|
|
|
|
endif
|
|
|
|
" delete cache when unlet b:airline head?
|
|
|
|
if !has_key(s:untracked_hg, a:file)
|
|
|
|
if s:has_async
|
|
|
|
call s:DoAsync(s:hg_cmd, a:file)
|
2016-02-02 16:45:47 -05:00
|
|
|
else
|
2016-09-28 12:03:13 -04:00
|
|
|
let untracked = (system(s:hg_cmd. shellescape(a:file))[0] is# '?' ?
|
2016-02-02 16:45:47 -05:00
|
|
|
\ get(g:, 'airline#extensions#branch#notexists', g:airline_symbols.notexists) : '')
|
|
|
|
let s:untracked_hg[a:file] = untracked
|
|
|
|
endif
|
|
|
|
endif
|
|
|
|
endfunction
|
|
|
|
|
2016-01-29 07:11:14 -05:00
|
|
|
function! s:get_hg_branch()
|
|
|
|
if s:has_lawrencium
|
2016-02-16 11:48:20 -05:00
|
|
|
return lawrencium#statusline()
|
2016-01-29 07:11:14 -05:00
|
|
|
endif
|
|
|
|
return ''
|
|
|
|
endfunction
|
|
|
|
|
2016-09-28 12:03:13 -04:00
|
|
|
if s:has_async
|
|
|
|
let s:jobs = {}
|
|
|
|
|
|
|
|
function! s:on_stdout(channel, msg) dict abort
|
|
|
|
let self.buf .= a:msg
|
|
|
|
endfunction
|
|
|
|
|
|
|
|
function! s:on_exit(channel) dict abort
|
|
|
|
let untracked = get(g:, 'airline#extensions#branch#notexists', g:airline_symbols.notexists)
|
|
|
|
if empty(self.buf)
|
|
|
|
let s:untracked_{self.cmd}[self.file] = ''
|
|
|
|
else
|
|
|
|
let s:untracked_{self.cmd}[self.file] = untracked
|
|
|
|
endif
|
|
|
|
endfunction
|
|
|
|
|
|
|
|
function! s:DoAsync(cmd, file)
|
|
|
|
if (has('win32') || has('win64')) && &shell =~ 'cmd'
|
|
|
|
let cmd = a:cmd. shellescape(a:file)
|
|
|
|
else
|
|
|
|
let cmd = ['sh', '-c', a:cmd. shellescape(a:file)]
|
|
|
|
endif
|
|
|
|
let cmdstring = split(a:cmd)[0]
|
|
|
|
|
|
|
|
let options = {'cmd': cmdstring, 'buf': '', 'file': a:file}
|
|
|
|
if has_key(s:jobs, a:file)
|
|
|
|
if job_status(get(s:jobs, a:file)) == 'run'
|
|
|
|
return
|
|
|
|
else
|
|
|
|
call job_stop(get(s:jobs, a:file))
|
|
|
|
call remove(s:jobs, a:file)
|
|
|
|
endif
|
|
|
|
endif
|
|
|
|
let id = job_start(cmd, {
|
|
|
|
\ 'err_io': 'out',
|
|
|
|
\ 'out_cb': function('s:on_stdout', options),
|
|
|
|
\ 'close_cb': function('s:on_exit', options)})
|
|
|
|
let s:jobs[a:file] = id
|
|
|
|
endfu
|
|
|
|
endif
|
|
|
|
|
2013-12-03 00:32:54 -05:00
|
|
|
function! airline#extensions#branch#head()
|
2014-04-02 20:54:43 -04:00
|
|
|
if exists('b:airline_head') && !empty(b:airline_head)
|
2014-03-24 14:01:31 -04:00
|
|
|
return b:airline_head
|
|
|
|
endif
|
|
|
|
|
|
|
|
let b:airline_head = ''
|
2016-01-29 07:11:14 -05:00
|
|
|
let l:heads = {}
|
|
|
|
let l:vcs_priority = get(g:, "airline#extensions#branch#vcs_priority", ["git", "mercurial"])
|
2014-08-15 16:07:18 -04:00
|
|
|
let found_fugitive_head = 0
|
2013-08-18 13:22:35 -04:00
|
|
|
|
2016-07-01 05:47:13 -04:00
|
|
|
if exists("*fnamemodify")
|
|
|
|
let l:git_head = s:get_git_branch(fnamemodify(resolve(@%), ":p:h"))
|
|
|
|
else
|
|
|
|
let l:git_head = s:get_git_branch(expand("%:p:h"))
|
|
|
|
endif
|
2016-01-29 07:11:14 -05:00
|
|
|
let l:hg_head = s:get_hg_branch()
|
2013-08-18 14:34:02 -04:00
|
|
|
|
2016-09-28 12:03:13 -04:00
|
|
|
let l:file = expand("%:p")
|
2016-01-29 07:11:14 -05:00
|
|
|
if !empty(l:git_head)
|
|
|
|
let found_fugitive_head = 1
|
|
|
|
let l:heads.git = (!empty(l:hg_head) ? "git:" : '') . s:format_name(l:git_head)
|
2016-09-28 12:03:13 -04:00
|
|
|
call s:get_git_untracked(l:file)
|
|
|
|
let l:heads.git .= get(s:untracked_git, l:file, '')
|
2013-08-18 13:22:35 -04:00
|
|
|
endif
|
|
|
|
|
2016-01-29 07:11:14 -05:00
|
|
|
if !empty(l:hg_head)
|
|
|
|
let l:heads.mercurial = (!empty(l:git_head) ? "hg:" : '') . s:format_name(l:hg_head)
|
2016-09-28 12:03:13 -04:00
|
|
|
call s:get_hg_untracked(l:file)
|
|
|
|
let l:heads.mercurial.= get(s:untracked_hg, l:file, '')
|
2013-08-18 13:22:35 -04:00
|
|
|
endif
|
|
|
|
|
2016-01-29 07:11:14 -05:00
|
|
|
if empty(l:heads)
|
2013-11-07 18:36:59 -05:00
|
|
|
if s:has_vcscommand
|
|
|
|
call VCSCommandEnableBufferSetup()
|
|
|
|
if exists('b:VCSCommandBufferInfo')
|
2016-01-29 07:11:14 -05:00
|
|
|
let b:airline_head = s:format_name(get(b:VCSCommandBufferInfo, 0, ''))
|
2013-11-07 18:36:59 -05:00
|
|
|
endif
|
|
|
|
endif
|
2016-01-29 07:11:14 -05:00
|
|
|
else
|
2016-05-22 15:29:00 -04:00
|
|
|
let b:airline_head = get(b:, 'airline_head', '')
|
2016-01-29 07:11:14 -05:00
|
|
|
for vcs in l:vcs_priority
|
|
|
|
if has_key(l:heads, vcs)
|
|
|
|
if !empty(b:airline_head)
|
|
|
|
let b:airline_head = b:airline_head . " | "
|
|
|
|
endif
|
|
|
|
let b:airline_head = b:airline_head . l:heads[vcs]
|
|
|
|
endif
|
|
|
|
endfor
|
2013-11-07 18:36:59 -05:00
|
|
|
endif
|
|
|
|
|
2014-05-21 13:41:53 -04:00
|
|
|
if exists("g:airline#extensions#branch#displayed_head_limit")
|
|
|
|
let w:displayed_head_limit = g:airline#extensions#branch#displayed_head_limit
|
|
|
|
if len(b:airline_head) > w:displayed_head_limit - 1
|
2016-02-25 05:30:02 -05:00
|
|
|
let b:airline_head = b:airline_head[0:(w:displayed_head_limit - 1)].(&encoding ==? 'utf-8' ? '…' : '.')
|
2014-05-21 13:41:53 -04:00
|
|
|
endif
|
|
|
|
endif
|
|
|
|
|
2016-01-29 07:11:14 -05:00
|
|
|
if empty(b:airline_head) || !found_fugitive_head && !s:check_in_path()
|
|
|
|
let b:airline_head = ''
|
|
|
|
endif
|
2016-07-03 14:44:05 -04:00
|
|
|
let minwidth = empty(get(b:, 'airline_hunks', '')) ? 14 : 7
|
|
|
|
let b:airline_head = airline#util#shorten(b:airline_head, 120, minwidth)
|
2014-03-24 14:01:31 -04:00
|
|
|
return b:airline_head
|
2013-12-03 00:32:54 -05:00
|
|
|
endfunction
|
|
|
|
|
|
|
|
function! airline#extensions#branch#get_head()
|
|
|
|
let head = airline#extensions#branch#head()
|
2016-01-28 09:54:14 -05:00
|
|
|
let empty_message = get(g:, 'airline#extensions#branch#empty_message', '')
|
2014-01-15 23:31:07 -05:00
|
|
|
let symbol = get(g:, 'airline#extensions#branch#symbol', g:airline_symbols.branch)
|
2013-12-03 00:32:54 -05:00
|
|
|
return empty(head)
|
2014-01-15 23:31:07 -05:00
|
|
|
\ ? empty_message
|
|
|
|
\ : printf('%s%s', empty(symbol) ? '' : symbol.(g:airline_symbols.space), head)
|
2013-08-06 00:42:59 -04:00
|
|
|
endfunction
|
|
|
|
|
2013-09-08 10:03:49 -04:00
|
|
|
function! s:check_in_path()
|
|
|
|
if !exists('b:airline_branch_path')
|
|
|
|
let root = get(b:, 'git_dir', get(b:, 'mercurial_dir', ''))
|
2014-02-25 16:36:36 -05:00
|
|
|
let bufferpath = resolve(fnamemodify(expand('%'), ':p'))
|
2013-10-01 21:36:24 -04:00
|
|
|
|
2013-10-02 10:18:33 -04:00
|
|
|
if !filereadable(root) "not a file
|
|
|
|
" if .git is a directory, it's the old submodule format
|
|
|
|
if match(root, '\.git$') >= 0
|
|
|
|
let root = expand(fnamemodify(root, ':h'))
|
|
|
|
else
|
|
|
|
" else it's the newer format, and we need to guesstimate
|
|
|
|
let pattern = '\.git\(\\\|\/\)modules\(\\\|\/\)'
|
|
|
|
if match(root, pattern) >= 0
|
|
|
|
let root = substitute(root, pattern, '', '')
|
|
|
|
endif
|
2015-01-03 19:38:17 -05:00
|
|
|
endif
|
2013-10-01 21:36:24 -04:00
|
|
|
endif
|
2013-10-02 10:18:33 -04:00
|
|
|
|
2013-09-08 10:03:49 -04:00
|
|
|
let b:airline_file_in_root = stridx(bufferpath, root) > -1
|
|
|
|
endif
|
|
|
|
return b:airline_file_in_root
|
|
|
|
endfunction
|
|
|
|
|
2016-09-28 12:03:13 -04:00
|
|
|
function! s:reset_untracked_cache()
|
2016-02-02 16:45:47 -05:00
|
|
|
if exists("s:untracked_git")
|
|
|
|
let s:untracked_git={}
|
|
|
|
endif
|
|
|
|
if exists("s:untracked_hg")
|
|
|
|
let s:untracked_hg={}
|
|
|
|
endif
|
|
|
|
endfunction
|
|
|
|
|
2013-08-05 23:07:01 -04:00
|
|
|
function! airline#extensions#branch#init(ext)
|
2013-08-30 17:51:10 -04:00
|
|
|
call airline#parts#define_function('branch', 'airline#extensions#branch#get_head')
|
2013-09-08 10:03:49 -04:00
|
|
|
|
|
|
|
autocmd BufReadPost * unlet! b:airline_file_in_root
|
2014-04-02 20:54:43 -04:00
|
|
|
autocmd CursorHold,ShellCmdPost,CmdwinLeave * unlet! b:airline_head
|
2016-01-29 09:29:40 -05:00
|
|
|
autocmd User AirlineBeforeRefresh unlet! b:airline_head
|
2016-09-28 12:03:13 -04:00
|
|
|
autocmd ShellCmdPost,BufWritePost * call s:reset_untracked_cache()
|
2013-08-05 23:07:01 -04:00
|
|
|
endfunction
|