1
0
mirror of https://github.com/SpaceVim/SpaceVim.git synced 2025-02-03 06:50:05 +08:00
SpaceVim/bundle/unite.vim/autoload/unite/util.vim
2020-06-13 14:06:35 +08:00

418 lines
12 KiB
VimL
Vendored

"=============================================================================
" FILE: util.vim
" AUTHOR: Shougo Matsushita <Shougo.Matsu at gmail.com>
" License: MIT license
"=============================================================================
let s:save_cpo = &cpo
set cpo&vim
function! unite#util#get_vital() abort "{{{
if !exists('s:V')
let s:V = vital#unite#new()
endif
return s:V
endfunction"}}}
function! unite#util#get_vital_cache() abort "{{{
if !exists('s:Cache')
let s:Cache = unite#util#get_vital().import('System.Cache.Deprecated')
endif
return s:Cache
endfunction"}}}
function! s:get_prelude() abort "{{{
if !exists('s:Prelude')
let s:Prelude = unite#util#get_vital().import('Prelude')
endif
return s:Prelude
endfunction"}}}
function! s:get_list() abort "{{{
if !exists('s:List')
let s:List = unite#util#get_vital().import('Data.List')
endif
return s:List
endfunction"}}}
function! s:get_string() abort "{{{
if !exists('s:String')
let s:String = unite#util#get_vital().import('Data.String')
endif
return s:String
endfunction"}}}
function! s:get_message() abort "{{{
if !exists('s:Message')
let s:Message = unite#util#get_vital().import('Vim.Message')
endif
return s:Message
endfunction"}}}
function! s:get_system() abort "{{{
if !exists('s:System')
let s:System = unite#util#get_vital().import('System.File')
endif
return s:System
endfunction"}}}
function! s:get_process() abort "{{{
if !exists('s:Process')
let s:Process = unite#util#get_vital().import('Process')
endif
return s:Process
endfunction"}}}
" TODO use vital's
let s:is_windows = has('win16') || has('win32') || has('win64')
function! unite#util#truncate_smart(...) abort
return call(s:get_string().truncate_skipping, a:000)
endfunction
function! unite#util#truncate(...) abort
return call(s:get_string().truncate, a:000)
endfunction
function! unite#util#strchars(...) abort
return call(s:get_string().strchars, a:000)
endfunction
function! unite#util#strwidthpart(...) abort
return call(s:get_string().strwidthpart, a:000)
endfunction
function! unite#util#strwidthpart_reverse(...) abort
return call(s:get_string().strwidthpart_reverse, a:000)
endfunction
function! unite#util#wcswidth(string) abort
return strwidth(a:string)
endfunction
function! unite#util#is_win(...) abort
echoerr 'unite#util#is_win() is deprecated. use unite#util#is_windows() instead.'
return call(s:get_prelude().is_windows, a:000)
endfunction
function! unite#util#is_windows(...) abort
return call(s:get_prelude().is_windows, a:000)
endfunction
function! unite#util#is_mac(...) abort
return call(s:get_prelude().is_mac, a:000)
endfunction
function! unite#util#print_error(msg) abort
let msg = '[unite.vim] ' . a:msg
return call(s:get_message().error, [msg])
endfunction
function! unite#util#smart_execute_command(action, word) abort
execute a:action . ' ' . fnameescape(a:word)
endfunction
function! unite#util#smart_open_command(action, word) abort
call unite#util#smart_execute_command(a:action, a:word)
call unite#remove_previewed_buffer_list(bufnr(a:word))
endfunction
function! unite#util#escape_file_searching(buffer_name) abort
" You should not escape for buflisted() or bufnr()
return a:buffer_name
endfunction
function! unite#util#escape_pattern(...) abort
return call(s:get_string().escape_pattern, a:000)
endfunction
function! unite#util#set_default(var, val, ...) abort "{{{
if !exists(a:var) || type({a:var}) != type(a:val)
if exists(a:var) && type({a:var}) != type(a:val)
call unite#print_error(printf(
\ 'Current %s is wrong type. Ignored your config.', a:var))
endif
let alternate_var = get(a:000, 0, '')
unlet! {a:var}
let {a:var} = exists(alternate_var) ?
\ {alternate_var} : a:val
endif
endfunction"}}}
if unite#util#is_windows()
function! unite#util#substitute_path_separator(...) abort
return call(s:get_prelude().substitute_path_separator, a:000)
endfunction
else
function! unite#util#substitute_path_separator(path) abort
return a:path
endfunction
endif
function! unite#util#path2directory(...) abort
return call(s:get_prelude().path2directory, a:000)
endfunction
function! unite#util#path2project_directory(...) abort
return call(s:get_prelude().path2project_directory, a:000)
endfunction
function! unite#util#has_vimproc(...) abort
return call(s:get_process().has_vimproc, a:000)
endfunction
function! unite#util#has_lua() abort
" Note: Disabled if_lua feature if less than 7.3.885.
" Because if_lua has double free problem.
" Note: Cannot use lua interface in Windows environment if encoding is not utf-8.
" https://github.com/Shougo/unite.vim/issues/466
return has('lua') && (v:version > 703 || v:version == 703 && has('patch885'))
\ && (!unite#util#is_windows() ||
\ &encoding ==# 'utf-8' || &encoding ==# 'latin1')
endfunction
function! unite#util#has_timers() abort
" Vim timers implementation has bug.
" It cannot stop callback handler in the handler.
return has('timers') && (has('nvim') || has('patch-7.4.2304'))
endfunction
function! unite#util#system(...) abort
return call(s:get_process().system, a:000)
endfunction
function! unite#util#system_passwd(...) abort
return call((unite#util#has_vimproc() ?
\ 'vimproc#system_passwd' : 'system'), a:000)
endfunction
function! unite#util#get_last_status(...) abort
return call(s:get_process().get_last_status, a:000)
endfunction
function! unite#util#get_last_errmsg() abort
return unite#util#has_vimproc() ? vimproc#get_last_errmsg() : ''
endfunction
function! unite#util#sort_by(...) abort
return call(s:get_list().sort_by, a:000)
endfunction
function! unite#util#uniq(...) abort
return call(s:get_list().uniq, a:000)
endfunction
function! unite#util#uniq_by(...) abort
return call(s:get_list().uniq_by, a:000)
endfunction
function! unite#util#input(prompt, ...) abort "{{{
let context = unite#get_context()
let prompt = a:prompt
let default = get(a:000, 0, '')
let completion = get(a:000, 1, '')
let source_name = get(a:000, 2, '')
if source_name != ''
let prompt = printf('[%s] %s', source_name, prompt)
endif
let args = [prompt, default]
if completion != ''
call add(args, completion)
endif
return context.unite__is_interactive ? call('input', args) : default
endfunction"}}}
function! unite#util#input_yesno(message) abort "{{{
let yesno = input(a:message . ' [yes/no]: ')
while yesno !~? '^\%(y\%[es]\|n\%[o]\)$'
redraw
if yesno == ''
echo 'Canceled.'
break
endif
" Retry.
call unite#print_error('Invalid input.')
let yesno = input(a:message . ' [yes/no]: ')
endwhile
redraw
return yesno =~? 'y\%[es]'
endfunction"}}}
function! unite#util#input_directory(message) abort "{{{
echo a:message
let dir = unite#util#substitute_path_separator(
\ unite#util#expand(input('', '', 'dir')))
while !isdirectory(dir)
redraw
if dir == ''
echo 'Canceled.'
break
endif
" Retry.
call unite#print_error('Invalid path.')
echo a:message
let dir = unite#util#substitute_path_separator(
\ unite#util#expand(input('', '', 'dir')))
endwhile
return dir
endfunction"}}}
function! unite#util#iconv(...) abort
return call(s:get_process().iconv, a:000)
endfunction
function! unite#util#alternate_buffer() abort "{{{
let unite = unite#get_current_unite()
if s:buflisted(unite.prev_bufnr)
\ && getbufvar(unite.prev_bufnr, '&filetype') !=# "unite"
execute 'buffer' unite.prev_bufnr
keepjumps call winrestview(unite.prev_winsaveview)
return
endif
let listed_buffer_len = len(filter(range(1, bufnr('$')),
\ 's:buflisted(v:val) && getbufvar(v:val, "&filetype") !=# "unite"'))
if listed_buffer_len <= 1
enew
return
endif
let cnt = 0
let pos = 1
let current = 0
while pos <= bufnr('$')
if s:buflisted(pos)
if pos == bufnr('%')
let current = cnt
endif
let cnt += 1
endif
let pos += 1
endwhile
if current > cnt / 2
bprevious
else
bnext
endif
endfunction"}}}
function! unite#util#is_cmdwin() abort "{{{
return bufname('%') ==# '[Command Line]'
endfunction"}}}
function! s:buflisted(bufnr) abort "{{{
return (getbufvar(a:bufnr, '&bufhidden') == '' || buflisted(a:bufnr)) &&
\ (exists('t:tabpagebuffer') ?
\ has_key(t:tabpagebuffer, a:bufnr) && buflisted(a:bufnr) :
\ buflisted(a:bufnr))
endfunction"}}}
function! unite#util#glob(pattern, ...) abort "{{{
let is_force_glob = get(a:000, 0, 1)
if !is_force_glob && (a:pattern =~ '\*$' || a:pattern == '*')
\ && unite#util#has_vimproc() && exists('*vimproc#readdir')
return vimproc#readdir(a:pattern[: -2])
else
" Escape [.
let glob = escape(a:pattern, '?={}[]')
let glob2 = escape(substitute(a:pattern,
\ '[^/]*$', '', '') . '.*', '?={}[]')
return unite#util#uniq(split(unite#util#substitute_path_separator(glob(glob)), '\n')
\ + split(unite#util#substitute_path_separator(glob(glob2)), '\n'))
endif
endfunction"}}}
function! unite#util#command_with_restore_cursor(command) abort "{{{
let pos = getpos('.')
let current = winnr()
execute a:command
let next = winnr()
" Restore cursor.
execute current 'wincmd w'
call setpos('.', pos)
execute next 'wincmd w'
endfunction"}}}
function! unite#util#expand(path) abort "{{{
return s:get_prelude().substitute_path_separator(
\ (a:path =~ '^\~') ? fnamemodify(a:path, ':p') :
\ (a:path =~ '^\$\h\w*') ? substitute(a:path,
\ '^\$\h\w*', '\=eval(submatch(0))', '') :
\ a:path)
endfunction"}}}
function! unite#util#set_default_dictionary_helper(variable, keys, value) abort "{{{
for key in split(a:keys, '\s*,\s*')
if !has_key(a:variable, key)
let a:variable[key] = a:value
endif
endfor
endfunction"}}}
function! unite#util#set_dictionary_helper(variable, keys, value) abort "{{{
for key in split(a:keys, '\s*,\s*')
let a:variable[key] = a:value
endfor
endfunction"}}}
function! unite#util#convert2list(expr) abort "{{{
return type(a:expr) ==# type([]) ? a:expr : [a:expr]
endfunction"}}}
function! unite#util#truncate_wrap(str, max, footer_width, separator) abort "{{{
let width = strwidth(a:str)
if width <= a:max
return unite#util#truncate(a:str, a:max)
elseif &l:wrap
return a:str
endif
let header_width = a:max - strwidth(a:separator) - a:footer_width
return unite#util#strwidthpart(a:str, header_width) . a:separator
\ . unite#util#strwidthpart_reverse(a:str, a:footer_width)
endfunction"}}}
function! unite#util#index_name(list, name) abort "{{{
return index(map(copy(a:list), 'v:val.name'), a:name)
endfunction"}}}
function! unite#util#get_name(list, name, default) abort "{{{
return get(a:list, unite#util#index_name(a:list, a:name), a:default)
endfunction"}}}
function! unite#util#escape_match(str) abort "{{{
return substitute(substitute(escape(a:str, '~\.^$[]'),
\ '\*\@<!\*\*\@!', '[^/]*', 'g'), '\*\*\+', '.*', 'g')
endfunction"}}}
function! unite#util#escape_shell(str) abort "{{{
return '"' . a:str . '"'
endfunction"}}}
function! unite#util#open(path) abort "{{{
return s:get_system().open(a:path)
endfunction"}}}
function! unite#util#move(src, dest) abort "{{{
return s:get_system().move(a:src, a:dest)
endfunction"}}}
function! unite#util#read_lines(source, ...) abort "{{{
let timeout = get(a:000, 0, -1)
if timeout < 0
return a:source.read_lines(-1, timeout)
endif
let lines = []
for _ in range(timeout / 100)
let lines += a:source.read_lines(-1, 100)
endfor
return lines
endfunction"}}}
function! unite#util#is_sudo() abort "{{{
return $SUDO_USER != '' && $USER !=# $SUDO_USER
\ && $HOME !=# expand('~'.$USER)
\ && $HOME ==# expand('~'.$SUDO_USER)
endfunction"}}}
function! unite#util#lcd(dir) abort "{{{
if isdirectory(a:dir)
execute (haslocaldir() ? 'lcd' : 'cd') fnameescape(a:dir)
endif
endfunction"}}}
function! unite#util#redir(cmd) abort "{{{
if exists('*capture')
return capture(a:cmd)
else
let [save_verbose, save_verbosefile] = [&verbose, &verbosefile]
set verbose=0 verbosefile=
redir => res
silent! execute a:cmd
redir END
let [&verbose, &verbosefile] = [save_verbose, save_verbosefile]
return res
endif
endfunction"}}}
let &cpo = s:save_cpo
unlet s:save_cpo