It's common to optionally prefix commands like 'packadd!' with 'silent!', but that wasn't recognized by these patterns. This change adds 'silent' support to the 'packadd', 'runtime', and 'colorscheme' command patterns. closes: #18361 Signed-off-by: Jon Parise <jon@indelible.org> Signed-off-by: Christian Brabandt <cb@256bit.org>
237 lines
6.8 KiB
VimL
237 lines
6.8 KiB
VimL
vim9script
|
|
|
|
# Language: Vim9 script
|
|
# Contributers: @lacygoill
|
|
# Shane-XB-Qian
|
|
# Andrew Radev
|
|
# Last Change: 2025 Sep 21
|
|
#
|
|
# Vim Script to handle jumping to the targets of several types of Vim commands
|
|
# (:import, :packadd, :runtime, :colorscheme), and to autoloaded functions of
|
|
# the style <path>#<function_name>.
|
|
#
|
|
# see runtime/ftplugin/vim.vim
|
|
|
|
# Interface {{{1
|
|
export def Find(editcmd: string) #{{{2
|
|
var curline: string = getline('.')
|
|
|
|
if curline =~ '^\s*\%(:\s*\)\=\%(sil\%[ent]!\=\s\+\)\=packadd!\=\s'
|
|
HandlePackaddLine(editcmd, curline)
|
|
return
|
|
endif
|
|
|
|
if curline =~ '^\s*\%(:\s*\)\=\%(sil\%[ent]!\=\s\+\)\=ru\%[ntime]!\='
|
|
HandleRuntimeLine(editcmd, curline, expand('<cfile>'))
|
|
return
|
|
endif
|
|
|
|
if curline =~ '^\s*\%(:\s*\)\=\%(sil\%[ent]!\=\s\+\)\=colo\%[rscheme]\s'
|
|
HandleColoLine(editcmd, curline)
|
|
return
|
|
endif
|
|
|
|
if curline =~ '^\s*\%(:\s*\)\=import\s'
|
|
HandleImportLine(editcmd, curline)
|
|
return
|
|
endif
|
|
|
|
var curfunc = FindCurfunc()
|
|
|
|
if stridx(curfunc, '#') >= 0
|
|
var parts = split(curfunc, '#')
|
|
var path = $"autoload/{join(parts[0 : -2], '/')}.vim"
|
|
var resolved_path = globpath(&runtimepath, path)
|
|
|
|
if resolved_path != ''
|
|
var function_pattern: string = $'^\s*\%(:\s*\)\=fun\%[ction]!\=\s\+\zs{curfunc}('
|
|
resolved_path->Open(editcmd, function_pattern)
|
|
endif
|
|
return
|
|
endif
|
|
|
|
try
|
|
execute 'normal! ' .. editcmd
|
|
catch
|
|
Error(v:exception)
|
|
endtry
|
|
enddef
|
|
#}}}1
|
|
# Core {{{1
|
|
def HandlePackaddLine(editcmd: string, curline: string) #{{{2
|
|
var pat: string = '\s*\%(:\s*\)\=packadd!\=\s\+\zs\S\+\>\ze'
|
|
var plugin: string = curline
|
|
->matchstr(pat)
|
|
->substitute('^vim-\|\.vim$', '', 'g')
|
|
|
|
if plugin == ''
|
|
Fallback(editcmd)
|
|
else
|
|
var files: list<string> = getcompletion($'plugin/{plugin}', 'runtime')
|
|
->map((_, fname: string) => fname->findfile(&rtp)->fnamemodify(':p'))
|
|
->filter((_, path: string): bool => filereadable(path))
|
|
if empty(files)
|
|
echo 'Could not find any plugin file for ' .. string(plugin)
|
|
return
|
|
endif
|
|
files->Open(editcmd)
|
|
endif
|
|
enddef
|
|
|
|
def HandleRuntimeLine(editcmd: string, curline: string, cfile: string) #{{{2
|
|
var fname: string
|
|
var where_pat: string = '\%(START\|OPT\|PACK\|ALL\)'
|
|
|
|
if cfile == 'runtime' || cfile =~# $'^{where_pat}$'
|
|
# then the cursor was not on one of the filenames, jump to the first file:
|
|
var fname_pat: string = $'\s*\%(:\s*\)\=ru\%[ntime]\%(!\s*\|\s\+\)\%({where_pat}\s\+\)\=\zs\S\+\>\ze'
|
|
fname = curline->matchstr(fname_pat)
|
|
else
|
|
fname = cfile
|
|
endif
|
|
|
|
if fname == ''
|
|
Fallback(editcmd)
|
|
else
|
|
var file: string = fname
|
|
->findfile(&rtp)
|
|
->fnamemodify(':p')
|
|
if file == '' || !filereadable(file)
|
|
echo 'Could not be found in the runtimepath: ' .. string(fname)
|
|
return
|
|
endif
|
|
file->Open(editcmd)
|
|
endif
|
|
enddef
|
|
|
|
def HandleColoLine(editcmd: string, curline: string) #{{{2
|
|
var pat: string = '\s*\%(:\s*\)\=colo\%[rscheme]\s\+\zs\S\+\>\ze'
|
|
var colo: string = curline->matchstr(pat)
|
|
|
|
if colo == ''
|
|
Fallback(editcmd)
|
|
else
|
|
var files: list<string> = getcompletion($'colors/{colo}', 'runtime')
|
|
->map((_, fname: string) => fname->findfile(&rtp)->fnamemodify(':p'))
|
|
->filter((_, path: string): bool => filereadable(path))
|
|
if empty(files)
|
|
echo 'Could not find any colorscheme file for ' .. string(colo)
|
|
return
|
|
endif
|
|
files->Open(editcmd)
|
|
endif
|
|
enddef
|
|
|
|
def HandleImportLine(editcmd: string, curline: string) #{{{2
|
|
var fname: string
|
|
var import_cmd: string = '^\s*\%(:\s*\)\=import\s\+\%(autoload\s\+\)\='
|
|
var import_alias: string = '\%(\s\+as\s\+\w\+\)\=$'
|
|
var import_string: string = import_cmd .. '\([''"]\)\zs.*\ze\1' .. import_alias
|
|
var import_expr: string = import_cmd .. '\zs.*\ze' .. import_alias
|
|
# the script is referred to by its name in a quoted string
|
|
if curline =~ import_string
|
|
fname = curline->matchstr(import_string)
|
|
# the script is referred to by an expression
|
|
elseif curline =~ import_expr
|
|
try
|
|
sandbox fname = curline
|
|
->matchstr(import_expr)
|
|
->eval()
|
|
catch
|
|
Error(v:exception)
|
|
return
|
|
endtry
|
|
endif
|
|
|
|
var filepath: string
|
|
if fname->isabsolutepath()
|
|
filepath = fname
|
|
elseif fname[0] == '.'
|
|
filepath = (expand('%:h') .. '/' .. fname)->simplify()
|
|
else
|
|
var subdir: string = curline =~ '^\s*import\s\+autoload\>' ? 'autoload' : 'import'
|
|
# Matching patterns in `'wildignore'` can be slow.
|
|
# Let's set `{nosuf}` to `true` to avoid `globpath()` to be slow.
|
|
filepath = globpath(&runtimepath, subdir .. '/' .. fname, true, true)
|
|
->get(0, '')
|
|
endif
|
|
|
|
if !filepath->filereadable()
|
|
printf('E447: Can''t find file "%s" in path', fname)
|
|
->Error()
|
|
return
|
|
endif
|
|
|
|
var how_to_split: string = {
|
|
gF: 'edit',
|
|
"\<C-W>F": 'split',
|
|
"\<C-W>gF": 'tab split',
|
|
}[editcmd]
|
|
execute how_to_split .. ' ' .. filepath
|
|
enddef
|
|
|
|
def Open(target: any, editcmd: string, search_pattern: string = '') #{{{2
|
|
var split: string = editcmd[0] == 'g' ? 'edit' : editcmd[1] == 'g' ? 'tabedit' : 'split'
|
|
var fname: string
|
|
var cmd: string
|
|
|
|
if target->typename() == 'list<string>'
|
|
if target->empty()
|
|
return
|
|
endif
|
|
fname = target[0]
|
|
else
|
|
if target->typename() != 'string'
|
|
return
|
|
endif
|
|
fname = target
|
|
endif
|
|
|
|
if search_pattern != ''
|
|
var escaped_pattern = escape(search_pattern, '\#'' ')
|
|
cmd = $'+silent\ call\ search(''{escaped_pattern}'')'
|
|
endif
|
|
|
|
execute $'{split} {cmd} {fname}'
|
|
|
|
# If there are several files to open, put them into an arglist.
|
|
if target->typename() == 'list<string>'
|
|
&& target->len() > 1
|
|
var arglist: list<string> = target
|
|
->copy()
|
|
->map((_, f: string) => f->fnameescape())
|
|
execute $'arglocal {arglist->join()}'
|
|
endif
|
|
enddef
|
|
#}}}1
|
|
# Util {{{1
|
|
def Error(msg: string) #{{{2
|
|
echohl ErrorMsg
|
|
echomsg msg
|
|
echohl NONE
|
|
enddef
|
|
|
|
def Fallback(editcmd: string) #{{{2
|
|
try
|
|
execute 'normal! ' .. editcmd .. 'zv'
|
|
catch
|
|
Error(v:exception)
|
|
endtry
|
|
enddef
|
|
|
|
def FindCurfunc(): string #{{{2
|
|
var curfunc = ''
|
|
var saved_iskeyword = &iskeyword
|
|
|
|
try
|
|
set iskeyword+=#
|
|
curfunc = expand('<cword>')
|
|
finally
|
|
&iskeyword = saved_iskeyword
|
|
endtry
|
|
|
|
return curfunc
|
|
enddef
|
|
|
|
# vim: sw=4 et
|