409 lines
12 KiB
VimL
409 lines
12 KiB
VimL
" Author: w0rp <devw0rp@gmail.com>
|
|
" Description: Contains miscellaneous functions
|
|
|
|
" A wrapper function for mode() so we can test calls for it.
|
|
function! ale#util#Mode(...) abort
|
|
return call('mode', a:000)
|
|
endfunction
|
|
|
|
" A wrapper function for feedkeys so we can test calls for it.
|
|
function! ale#util#FeedKeys(...) abort
|
|
return call('feedkeys', a:000)
|
|
endfunction
|
|
|
|
" Show a message in as small a window as possible.
|
|
"
|
|
" Vim 8 does not support echoing long messages from asynchronous callbacks,
|
|
" but NeoVim does. Small messages can be echoed in Vim 8, and larger messages
|
|
" have to be shown in preview windows.
|
|
function! ale#util#ShowMessage(string) abort
|
|
" We have to assume the user is using a monospace font.
|
|
if has('nvim') || (a:string !~? "\n" && len(a:string) < &columns)
|
|
execute 'echo a:string'
|
|
else
|
|
call ale#preview#Show(split(a:string, "\n"))
|
|
endif
|
|
endfunction
|
|
|
|
" A wrapper function for execute, so we can test executing some commands.
|
|
function! ale#util#Execute(expr) abort
|
|
execute a:expr
|
|
endfunction
|
|
|
|
if !exists('g:ale#util#nul_file')
|
|
" A null file for sending output to nothing.
|
|
let g:ale#util#nul_file = '/dev/null'
|
|
|
|
if has('win32')
|
|
let g:ale#util#nul_file = 'nul'
|
|
endif
|
|
endif
|
|
|
|
" Return the number of lines for a given buffer.
|
|
function! ale#util#GetLineCount(buffer) abort
|
|
return len(getbufline(a:buffer, 1, '$'))
|
|
endfunction
|
|
|
|
function! ale#util#GetFunction(string_or_ref) abort
|
|
if type(a:string_or_ref) == type('')
|
|
return function(a:string_or_ref)
|
|
endif
|
|
|
|
return a:string_or_ref
|
|
endfunction
|
|
|
|
function! ale#util#Open(filename, line, column, options) abort
|
|
if get(a:options, 'open_in_tab', 0)
|
|
call ale#util#Execute('tabedit ' . fnameescape(a:filename))
|
|
else
|
|
call ale#util#Execute('edit ' . fnameescape(a:filename))
|
|
endif
|
|
|
|
call cursor(a:line, a:column)
|
|
endfunction
|
|
|
|
let g:ale#util#error_priority = 5
|
|
let g:ale#util#warning_priority = 4
|
|
let g:ale#util#info_priority = 3
|
|
let g:ale#util#style_error_priority = 2
|
|
let g:ale#util#style_warning_priority = 1
|
|
|
|
function! ale#util#GetItemPriority(item) abort
|
|
if a:item.type is# 'I'
|
|
return g:ale#util#info_priority
|
|
endif
|
|
|
|
if a:item.type is# 'W'
|
|
if get(a:item, 'sub_type', '') is# 'style'
|
|
return g:ale#util#style_warning_priority
|
|
endif
|
|
|
|
return g:ale#util#warning_priority
|
|
endif
|
|
|
|
if get(a:item, 'sub_type', '') is# 'style'
|
|
return g:ale#util#style_error_priority
|
|
endif
|
|
|
|
return g:ale#util#error_priority
|
|
endfunction
|
|
|
|
" Compare two loclist items for ALE, sorted by their buffers, filenames, and
|
|
" line numbers and column numbers.
|
|
function! ale#util#LocItemCompare(left, right) abort
|
|
if a:left.bufnr < a:right.bufnr
|
|
return -1
|
|
endif
|
|
|
|
if a:left.bufnr > a:right.bufnr
|
|
return 1
|
|
endif
|
|
|
|
if a:left.bufnr == -1
|
|
if a:left.filename < a:right.filename
|
|
return -1
|
|
endif
|
|
|
|
if a:left.filename > a:right.filename
|
|
return 1
|
|
endif
|
|
endif
|
|
|
|
if a:left.lnum < a:right.lnum
|
|
return -1
|
|
endif
|
|
|
|
if a:left.lnum > a:right.lnum
|
|
return 1
|
|
endif
|
|
|
|
if a:left.col < a:right.col
|
|
return -1
|
|
endif
|
|
|
|
if a:left.col > a:right.col
|
|
return 1
|
|
endif
|
|
|
|
" When either of the items lacks a problem type, then the two items should
|
|
" be considered equal. This is important for loclist jumping.
|
|
if !has_key(a:left, 'type') || !has_key(a:right, 'type')
|
|
return 0
|
|
endif
|
|
|
|
let l:left_priority = ale#util#GetItemPriority(a:left)
|
|
let l:right_priority = ale#util#GetItemPriority(a:right)
|
|
|
|
if l:left_priority < l:right_priority
|
|
return -1
|
|
endif
|
|
|
|
if l:left_priority > l:right_priority
|
|
return 1
|
|
endif
|
|
|
|
return 0
|
|
endfunction
|
|
|
|
" Compare two loclist items, including the text for the items.
|
|
"
|
|
" This function can be used for de-duplicating lists.
|
|
function! ale#util#LocItemCompareWithText(left, right) abort
|
|
let l:cmp_value = ale#util#LocItemCompare(a:left, a:right)
|
|
|
|
if l:cmp_value
|
|
return l:cmp_value
|
|
endif
|
|
|
|
if a:left.text < a:right.text
|
|
return -1
|
|
endif
|
|
|
|
if a:left.text > a:right.text
|
|
return 1
|
|
endif
|
|
|
|
return 0
|
|
endfunction
|
|
|
|
" This function will perform a binary search and a small sequential search
|
|
" on the list to find the last problem in the buffer and line which is
|
|
" on or before the column. The index of the problem will be returned.
|
|
"
|
|
" -1 will be returned if nothing can be found.
|
|
function! ale#util#BinarySearch(loclist, buffer, line, column) abort
|
|
let l:min = 0
|
|
let l:max = len(a:loclist) - 1
|
|
|
|
while 1
|
|
if l:max < l:min
|
|
return -1
|
|
endif
|
|
|
|
let l:mid = (l:min + l:max) / 2
|
|
let l:item = a:loclist[l:mid]
|
|
|
|
" Binary search for equal buffers, equal lines, then near columns.
|
|
if l:item.bufnr < a:buffer
|
|
let l:min = l:mid + 1
|
|
elseif l:item.bufnr > a:buffer
|
|
let l:max = l:mid - 1
|
|
elseif l:item.lnum < a:line
|
|
let l:min = l:mid + 1
|
|
elseif l:item.lnum > a:line
|
|
let l:max = l:mid - 1
|
|
else
|
|
" This part is a small sequential search.
|
|
let l:index = l:mid
|
|
|
|
" Search backwards to find the first problem on the line.
|
|
while l:index > 0
|
|
\&& a:loclist[l:index - 1].bufnr == a:buffer
|
|
\&& a:loclist[l:index - 1].lnum == a:line
|
|
let l:index -= 1
|
|
endwhile
|
|
|
|
" Find the last problem on or before this column.
|
|
while l:index < l:max
|
|
\&& a:loclist[l:index + 1].bufnr == a:buffer
|
|
\&& a:loclist[l:index + 1].lnum == a:line
|
|
\&& a:loclist[l:index + 1].col <= a:column
|
|
let l:index += 1
|
|
endwhile
|
|
|
|
" Scan forwards to find the last item on the column for the item
|
|
" we found, which will have the most serious problem.
|
|
let l:item_column = a:loclist[l:index].col
|
|
|
|
while l:index < l:max
|
|
\&& a:loclist[l:index + 1].bufnr == a:buffer
|
|
\&& a:loclist[l:index + 1].lnum == a:line
|
|
\&& a:loclist[l:index + 1].col == l:item_column
|
|
let l:index += 1
|
|
endwhile
|
|
|
|
return l:index
|
|
endif
|
|
endwhile
|
|
endfunction
|
|
|
|
" A function for testing if a function is running inside a sandbox.
|
|
" See :help sandbox
|
|
function! ale#util#InSandbox() abort
|
|
try
|
|
function! s:SandboxCheck() abort
|
|
endfunction
|
|
catch /^Vim\%((\a\+)\)\=:E48/
|
|
" E48 is the sandbox error.
|
|
return 1
|
|
endtry
|
|
|
|
return 0
|
|
endfunction
|
|
|
|
" Get the number of milliseconds since some vague, but consistent, point in
|
|
" the past.
|
|
"
|
|
" This function can be used for timing execution, etc.
|
|
"
|
|
" The time will be returned as a Number.
|
|
function! ale#util#ClockMilliseconds() abort
|
|
return float2nr(reltimefloat(reltime()) * 1000)
|
|
endfunction
|
|
|
|
" Given a single line, or a List of lines, and a single pattern, or a List
|
|
" of patterns, return all of the matches for the lines(s) from the given
|
|
" patterns, using matchlist().
|
|
"
|
|
" Only the first pattern which matches a line will be returned.
|
|
function! ale#util#GetMatches(lines, patterns) abort
|
|
let l:matches = []
|
|
let l:lines = type(a:lines) == type([]) ? a:lines : [a:lines]
|
|
let l:patterns = type(a:patterns) == type([]) ? a:patterns : [a:patterns]
|
|
|
|
for l:line in l:lines
|
|
for l:pattern in l:patterns
|
|
let l:match = matchlist(l:line, l:pattern)
|
|
|
|
if !empty(l:match)
|
|
call add(l:matches, l:match)
|
|
break
|
|
endif
|
|
endfor
|
|
endfor
|
|
|
|
return l:matches
|
|
endfunction
|
|
|
|
function! s:LoadArgCount(function) abort
|
|
let l:Function = a:function
|
|
|
|
redir => l:output
|
|
silent! function Function
|
|
redir END
|
|
|
|
if !exists('l:output')
|
|
return 0
|
|
endif
|
|
|
|
let l:match = matchstr(split(l:output, "\n")[0], '\v\([^)]+\)')[1:-2]
|
|
let l:arg_list = filter(split(l:match, ', '), 'v:val isnot# ''...''')
|
|
|
|
return len(l:arg_list)
|
|
endfunction
|
|
|
|
" Given the name of a function, a Funcref, or a lambda, return the number
|
|
" of named arguments for a function.
|
|
function! ale#util#FunctionArgCount(function) abort
|
|
let l:Function = ale#util#GetFunction(a:function)
|
|
let l:count = s:LoadArgCount(l:Function)
|
|
|
|
" If we failed to get the count, forcibly load the autoload file, if the
|
|
" function is an autoload function. autoload functions aren't normally
|
|
" defined until they are called.
|
|
if l:count == 0
|
|
let l:function_name = matchlist(string(l:Function), 'function([''"]\(.\+\)[''"])')[1]
|
|
|
|
if l:function_name =~# '#'
|
|
execute 'runtime autoload/' . join(split(l:function_name, '#')[:-2], '/') . '.vim'
|
|
let l:count = s:LoadArgCount(l:Function)
|
|
endif
|
|
endif
|
|
|
|
return l:count
|
|
endfunction
|
|
|
|
" Escape a string so the characters in it will be safe for use inside of PCRE
|
|
" or RE2 regular expressions without characters having special meanings.
|
|
function! ale#util#EscapePCRE(unsafe_string) abort
|
|
return substitute(a:unsafe_string, '\([\-\[\]{}()*+?.^$|]\)', '\\\1', 'g')
|
|
endfunction
|
|
|
|
" Escape a string so that it can be used as a literal string inside an evaled
|
|
" vim command.
|
|
function! ale#util#EscapeVim(unsafe_string) abort
|
|
return "'" . substitute(a:unsafe_string, "'", "''", 'g') . "'"
|
|
endfunction
|
|
|
|
|
|
" Given a String or a List of String values, try and decode the string(s)
|
|
" as a JSON value which can be decoded with json_decode. If the JSON string
|
|
" is invalid, the default argument value will be returned instead.
|
|
"
|
|
" This function is useful in code where the data can't be trusted to be valid
|
|
" JSON, and where throwing exceptions is mostly just irritating.
|
|
function! ale#util#FuzzyJSONDecode(data, default) abort
|
|
if empty(a:data)
|
|
return a:default
|
|
endif
|
|
|
|
let l:str = type(a:data) == type('') ? a:data : join(a:data, '')
|
|
|
|
try
|
|
let l:result = json_decode(l:str)
|
|
|
|
" Vim 8 only uses the value v:none for decoding blank strings.
|
|
if !has('nvim') && l:result is v:none
|
|
return a:default
|
|
endif
|
|
|
|
return l:result
|
|
catch /E474/
|
|
return a:default
|
|
endtry
|
|
endfunction
|
|
|
|
" Write a file, including carriage return characters for DOS files.
|
|
"
|
|
" The buffer number is required for determining the fileformat setting for
|
|
" the buffer.
|
|
function! ale#util#Writefile(buffer, lines, filename) abort
|
|
let l:corrected_lines = getbufvar(a:buffer, '&fileformat') is# 'dos'
|
|
\ ? map(copy(a:lines), 'v:val . "\r"')
|
|
\ : a:lines
|
|
|
|
call writefile(l:corrected_lines, a:filename) " no-custom-checks
|
|
endfunction
|
|
|
|
if !exists('s:patial_timers')
|
|
let s:partial_timers = {}
|
|
endif
|
|
|
|
function! s:ApplyPartialTimer(timer_id) abort
|
|
if has_key(s:partial_timers, a:timer_id)
|
|
let [l:Callback, l:args] = remove(s:partial_timers, a:timer_id)
|
|
call call(l:Callback, [a:timer_id] + l:args)
|
|
endif
|
|
endfunction
|
|
|
|
" Given a delay, a callback, a List of arguments, start a timer with
|
|
" timer_start() and call the callback provided with [timer_id] + args.
|
|
"
|
|
" The timer must not be stopped with timer_stop().
|
|
" Use ale#util#StopPartialTimer() instead, which can stop any timer, and will
|
|
" clear any arguments saved for executing callbacks later.
|
|
function! ale#util#StartPartialTimer(delay, callback, args) abort
|
|
let l:timer_id = timer_start(a:delay, function('s:ApplyPartialTimer'))
|
|
let s:partial_timers[l:timer_id] = [a:callback, a:args]
|
|
|
|
return l:timer_id
|
|
endfunction
|
|
|
|
function! ale#util#StopPartialTimer(timer_id) abort
|
|
call timer_stop(a:timer_id)
|
|
|
|
if has_key(s:partial_timers, a:timer_id)
|
|
call remove(s:partial_timers, a:timer_id)
|
|
endif
|
|
endfunction
|
|
|
|
" Given a possibly multi-byte string and a 1-based character position on a
|
|
" line, return the 1-based byte position on that line.
|
|
function! ale#util#Col(str, chr) abort
|
|
if a:chr < 2
|
|
return a:chr
|
|
endif
|
|
|
|
return strlen(join(split(a:str, '\zs')[0:a:chr - 2], '')) + 1
|
|
endfunction
|