let s:self_version = expand(':t:r') let s:self_file = expand('') let s:base_dir = expand(':h') let s:loaded = {} let s:cache_module_path = {} let s:cache_sid = {} let s:_unify_path_cache = {} function! s:plugin_name() abort let info_file = get(split(glob(s:base_dir . '/*.vital', 1), "\n"), 0, '') return fnamemodify(info_file, ':t:r') endfunction function! s:vital_files() abort if !exists('s:vital_files') let s:vital_files = \ map( \ s:plugin_name() ==# 'vital' \ ? s:_global_vital_files() \ : s:_self_vital_files(), \ 'fnamemodify(v:val, ":p:gs?[\\\\/]?/?")') endif return copy(s:vital_files) endfunction function! s:import(name, ...) abort 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 = s:_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 function! s:load(...) dict abort 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, s:_import(name)) endif unlet arg endfor return self endfunction function! s:unload() abort let s:loaded = {} let s:cache_sid = {} let s:cache_module_path = {} unlet! s:vital_files endfunction function! s:exists(name) abort return s:_get_module_path(a:name) !=# '' endfunction function! s:search(pattern) abort let paths = s:_extract_files(a:pattern, s:vital_files()) let modules = sort(map(paths, 's:_file2module(v:val)')) return s:_uniq(modules) endfunction function! s:_import(name) abort if type(a:name) == type(0) return s:_build_module(a:name) endif let path = s:_get_module_path(a:name) if path ==# '' throw 'vital: module not found: ' . a:name endif let sid = s:_get_sid_by_script(path) if !sid try execute 'source' fnameescape(path) catch /^Vim\%((\a\+)\)\?:E484/ throw 'vital: module not found: ' . a:name catch /^Vim\%((\a\+)\)\?:E127/ " Ignore. endtry let sid = s:_get_sid_by_script(path) endif return s:_build_module(sid) endfunction function! s:_get_module_path(name) abort let key = a:name . '_' if has_key(s:cache_module_path, key) return s:cache_module_path[key] endif if s:_is_absolute_path(a:name) && filereadable(a:name) return a:name endif if a:name =~# '\v^\u\w*%(\.\u\w*)*$' let paths = s:_extract_files(a:name, s:vital_files()) else throw 'vital: Invalid module name: ' . a:name endif call filter(paths, 'filereadable(expand(v:val, 1))') let path = get(paths, 0, '') let s:cache_module_path[key] = path return path endfunction function! s:_get_sid_by_script(path) abort if has_key(s:cache_sid, a:path) return s:cache_sid[a:path] endif let path = s:_unify_path(a:path) let p = 'stridx(v:val, s:self_version) > 0 || stridx(v:val, "__latest__") > 0' for line in filter(split(s:_redir('scriptnames'), "\n"), p) let list = matchlist(line, '^\s*\(\d\+\):\s\+\(.\+\)\s*$') if !empty(list) && s:_unify_path(list[2]) ==# path let s:cache_sid[a:path] = list[1] - 0 return s:cache_sid[a:path] endif endfor return 0 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 if filereadable(expand(':r') . '.VIM') " 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 function! s:_self_vital_files() abort let base = s: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 " Copy from System.Filepath if has('win16') || has('win32') || has('win64') function! s:_is_absolute_path(path) abort return a:path =~? '^[a-z]:[/\\]' endfunction else function! s:_is_absolute_path(path) abort return a:path[0] ==# '/' endfunction endif function! s:_build_module(sid) abort if has_key(s:loaded, a:sid) return copy(s:loaded[a:sid]) endif let functions = s:_get_functions(a:sid) let prefix = '' . a:sid . '_' let module = {} for func in functions let module[func] = function(prefix . func) endfor 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_debug() to avoid cyclic " dependences but remove the cache if module._vital_loaded() fails. let s:loaded[a:sid] = get(g:, 'vital_debug', 0) ? module : export_module if has_key(module, '_vital_loaded') try let V = vital#{s:self_version}#new() call module._vital_loaded(V) catch unlet s:loaded[a:sid] throw 'vital: fail to call ._vital_loaded(): ' . v:exception endtry endif return copy(s:loaded[a:sid]) endfunction if exists('+regexpengine') function! s:_get_functions(sid) abort let funcs = s:_redir(printf("function /\\%%#=2^\%d_", a:sid)) let map_pat = '' . a:sid . '_\zs\w\+' return map(split(funcs, "\n"), 'matchstr(v:val, map_pat)') endfunction else function! s:_get_functions(sid) abort let prefix = '' . a:sid . '_' let funcs = s:_redir('function') let filter_pat = '^\s*function ' . prefix let map_pat = prefix . '\zs\w\+' return map(filter(split(funcs, "\n"), \ 'stridx(v:val, prefix) > 0 && v:val =~# filter_pat'), \ 'matchstr(v:val, map_pat)') endfunction endif 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) let i -= 2 else let i -= 1 endif endwhile return a:list endfunction endif 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 function! vital#{s:self_version}#new() abort let sid = s:_get_sid_by_script(s:self_file) return s:_build_module(sid) endfunction