let s:plugin_name = expand(':t:r') let s:vital_base_dir = expand(':h') let s:project_root = expand(':h:h:h') let s:has_latest_module = isdirectory(expand(':h') . '/__latest__') let s:is_vital_vim = s:plugin_name is# '_latest__' let s:loaded = {} let s:cache_sid = {} " function() wrapper if v:version > 703 || v:version == 703 && has('patch1170') function! s:_function(fstr) abort return function(a:fstr) endfunction else function! s:_SID() abort return matchstr(expand(''), '\zs\d\+\ze__SID$') endfunction let s:_s = '' . s:_SID() . '_' function! s:_function(fstr) abort return function(substitute(a:fstr, 's:', s:_s, 'g')) endfunction endif function! vital#{s:plugin_name}#of() abort return s:new(s:plugin_name) endfunction let s:Vital = {} function! s:new(plugin_name) abort let base = deepcopy(s:Vital) let base.plugin_name = a:plugin_name return base endfunction function! s:vital_files() abort if !exists('s:vital_files') let s:vital_files = map( \ s:is_vital_vim ? s:_global_vital_files() : s:_self_vital_files(), \ 'fnamemodify(v:val, ":p:gs?[\\\\/]?/?")') endif return copy(s:vital_files) endfunction let s:Vital.vital_files = s:_function('s:vital_files') function! s:import(name, ...) abort dict let target = {} let functions = [] for a in a:000 if type(a) == type({}) let target = a elseif type(a) == type([]) let functions = a endif unlet a endfor let module = self._import(a:name) if empty(functions) call extend(target, module, 'keep') else for f in functions if has_key(module, f) && !has_key(target, f) let target[f] = module[f] endif endfor endif return target endfunction let s:Vital.import = s:_function('s:import') function! s:load(...) abort dict for arg in a:000 let [name; as] = type(arg) == type([]) ? arg[: 1] : [arg, arg] let target = split(join(as, ''), '\W\+') let dict = self let dict_type = type({}) while !empty(target) let ns = remove(target, 0) if !has_key(dict, ns) let dict[ns] = {} endif if type(dict[ns]) == dict_type let dict = dict[ns] else unlet dict break endif endwhile if exists('dict') call extend(dict, self._import(name)) endif unlet arg endfor return self endfunction let s:Vital.load = s:_function('s:load') function! s:unload() abort dict let s:loaded = {} let s:cache_sid = {} unlet! s:vital_files endfunction let s:Vital.unload = s:_function('s:unload') function! s:exists(name) abort dict let b = exists(printf('*vital#_%s#%s#import', self.plugin_name, substitute(a:name, '\.', '#', 'g'))) if b return b endif let name_path = substitute(a:name, '\.', '/', 'g') let path = printf('%s/_%s/%s.vim', s:vital_base_dir, self.plugin_name, name_path) let b = filereadable(path) if b return b endif let path = printf('%s/_%s/%s.vim', s:vital_base_dir, '_latest__', name_path) let b = filereadable(path) endfunction let s:Vital.exists = s:_function('s:exists') function! s:search(pattern) abort dict let paths = s:_extract_files(a:pattern, self.vital_files()) let modules = sort(map(paths, 's:_file2module(v:val)')) return s:_uniq(modules) endfunction let s:Vital.search = s:_function('s:search') function! s:_self_vital_files() abort let base = s:vital_base_dir . '/*/**/*.vim' return split(glob(base, 1), "\n") endfunction function! s:_global_vital_files() abort let pattern = 'autoload/vital/__latest__/**/*.vim' return split(globpath(&runtimepath, pattern, 1), "\n") endfunction function! s:_extract_files(pattern, files) abort let tr = {'.': '/', '*': '[^/]*', '**': '.*'} let target = substitute(a:pattern, '\.\|\*\*\?', '\=tr[submatch(0)]', 'g') let regexp = printf('autoload/vital/[^/]\+/%s.vim$', target) return filter(a:files, 'v:val =~# regexp') endfunction function! s:_file2module(file) abort let filename = fnamemodify(a:file, ':p:gs?[\\/]?/?') let tail = matchstr(filename, 'autoload/vital/_\w\+/\zs.*\ze\.vim$') return join(split(tail, '[\\/]\+'), '.') endfunction " @param {string} name e.g. Data.List function! s:_import(name) abort dict if has_key(s:loaded, a:name) return copy(s:loaded[a:name]) endif let module = self._get_module(a:name) if has_key(module, '_vital_created') call module._vital_created(module) endif let export_module = filter(copy(module), 'v:key =~# "^\\a"') " Cache module before calling module.vital_loaded() to avoid cyclic " dependences but remove the cache if module._vital_loaded() fails. " let s:loaded[a:name] = export_module let s:loaded[a:name] = export_module if has_key(module, '_vital_loaded') try call module._vital_loaded(vital#{s:plugin_name}#of()) catch unlet s:loaded[a:name] throw 'vital: fail to call ._vital_loaded(): ' . v:exception endtry endif return copy(s:loaded[a:name]) endfunction let s:Vital._import = s:_function('s:_import') " s:_get_module() returns module object wihch has all script local functions. function! s:_get_module(name) abort dict try let module = vital#_{self.plugin_name}#{substitute(a:name, '\.', '#', 'g')}#import() catch /E117: Unknown function:/ if !s:has_latest_module throw 'vital: module not found: ' . a:name endif " Retry to support loading self modules. let module = s:_get_latest_module(a:name) endtry return module endfunction let s:Vital._get_module = s:_function('s:_get_module') function! s:_get_latest_module(name) abort return s:sid2sfuncs(s:_module_sid(a:name)) endfunction function! s:_module_sid(name) abort let module_rel_path = 'autoload/vital/__latest__/' . substitute(a:name, '\.', '/', 'g') . '.vim' let module_full_path = s:_unify_path(get(split(globpath(s:_module_sid_base_dir(), module_rel_path, 0), "\n"), 0, '')) if !filereadable(module_full_path) throw 'vital: module not found: ' . a:name endif let p = substitute(module_rel_path, '/', '[/\\\\]\\+', 'g') let sid = s:_sid(module_full_path, p) if !sid call s:_source(module_full_path) let sid = s:_sid(module_full_path, p) if !sid throw 'vital: cannot get from path' endif endif return sid endfunction function! s:_module_sid_base_dir() abort return s:is_vital_vim ? &rtp : s:project_root endfunction function! s:_source(path) abort execute 'source' fnameescape(a:path) endfunction " @vimlint(EVL102, 1, l:_) " @vimlint(EVL102, 1, l:__) function! s:_sid(fullpath, filter_pattern) abort if has_key(s:cache_sid, a:fullpath) return s:cache_sid[a:fullpath] endif for line in filter(split(s:_redir(':scriptnames'), "\n"), 'v:val =~# a:filter_pattern') let [_, sid, path; __] = matchlist(line, '^\s*\(\d\+\):\s\+\(.\+\)\s*$') if s:_unify_path(path) is# a:fullpath let s:cache_sid[a:fullpath] = sid return s:cache_sid[a:fullpath] endif endfor return 0 endfunction function! s:_redir(cmd) abort 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 endfunction if filereadable(expand(':r') . '.VIM') let s:_unify_path_cache = {} " resolve() is slow, so we cache results. " Note: On windows, vim can't expand path names from 8.3 formats. " So if getting full path via and $HOME was set as 8.3 format, " vital load duplicated scripts. Below's :~ avoid this issue. function! s:_unify_path(path) abort if has_key(s:_unify_path_cache, a:path) return s:_unify_path_cache[a:path] endif let value = tolower(fnamemodify(resolve(fnamemodify( \ a:path, ':p')), ':~:gs?[\\/]?/?')) let s:_unify_path_cache[a:path] = value return value endfunction else function! s:_unify_path(path) abort return resolve(fnamemodify(a:path, ':p:gs?[\\/]?/?')) endfunction endif " copied and modified from Vim.ScriptLocal let s:SNR = join(map(range(len("\")), '"[\\x" . printf("%0x", char2nr("\"[v:val])) . "]"'), '') function! s:sid2sfuncs(sid) abort let fs = split(s:_redir(printf(':function /^%s%s_', s:SNR, a:sid)), "\n") let r = {} let pattern = printf('\m^function\s%d_\zs\w\{-}\ze(', a:sid) for fname in map(fs, 'matchstr(v:val, pattern)') let r[fname] = function(s:_sfuncname(a:sid, fname)) endfor return r endfunction "" Return funcname of script local functions with SID function! s:_sfuncname(sid, funcname) abort return printf('%s_%s', a:sid, a:funcname) endfunction if exists('*uniq') function! s:_uniq(list) abort return uniq(a:list) endfunction else function! s:_uniq(list) abort let i = len(a:list) - 1 while 0 < i if a:list[i] ==# a:list[i - 1] call remove(a:list, i) endif let i -= 1 endwhile return a:list endfunction endif