From 3dc8498fd2c6bdfd93a46811203f545970bda065 Mon Sep 17 00:00:00 2001
From: haya14busa <haya14busa@gmail.com>
Date: Fri, 24 Nov 2017 12:38:13 +0000
Subject: [PATCH 01/13] git mv autoload/vimlparser.vim
 autoload/vital/__vimlparser__/VimlParser.vim

---
 autoload/{vimlparser.vim => vital/__vimlparser__/VimlParser.vim} | 0
 1 file changed, 0 insertions(+), 0 deletions(-)
 rename autoload/{vimlparser.vim => vital/__vimlparser__/VimlParser.vim} (100%)

diff --git a/autoload/vimlparser.vim b/autoload/vital/__vimlparser__/VimlParser.vim
similarity index 100%
rename from autoload/vimlparser.vim
rename to autoload/vital/__vimlparser__/VimlParser.vim

From cd27075b7b4eebc6ec8d3efc28207baf57b4f919 Mon Sep 17 00:00:00 2001
From: haya14busa <haya14busa@gmail.com>
Date: Fri, 24 Nov 2017 12:41:48 +0000
Subject: [PATCH 02/13] make VimlParser vital module

---
 autoload/vimlparser.vim                      |  26 ++
 autoload/vital.vim                           |  12 +
 autoload/vital/__vimlparser__/VimlParser.vim |  22 +-
 autoload/vital/_vimlparser.vim               |   9 +
 autoload/vital/vimlparser.vim                | 328 +++++++++++++++++++
 autoload/vital/vimlparser.vital              |   3 +
 6 files changed, 379 insertions(+), 21 deletions(-)
 create mode 100644 autoload/vimlparser.vim
 create mode 100644 autoload/vital.vim
 create mode 100644 autoload/vital/_vimlparser.vim
 create mode 100644 autoload/vital/vimlparser.vim
 create mode 100644 autoload/vital/vimlparser.vital

diff --git a/autoload/vimlparser.vim b/autoload/vimlparser.vim
new file mode 100644
index 00000000..315b5725
--- /dev/null
+++ b/autoload/vimlparser.vim
@@ -0,0 +1,26 @@
+let s:VimLParser = vital#vimlparser#import('VimlParser')
+
+function! vimlparser#import()
+  return s:VimLParser.import()
+endfunction
+
+" @brief Read input as VimScript and return stringified AST.
+" @param input Input filename or string of VimScript.
+" @return Stringified AST.
+function! vimlparser#test(input, ...)
+  try
+    if a:0 > 0
+      let l:neovim = a:1
+    else
+      let l:neovim = 0
+    endif
+    let i = type(a:input) == 1 && filereadable(a:input) ? readfile(a:input) : split(a:input, "\n")
+    let r = s:StringReader.new(i)
+    let p = s:VimLParser.new(l:neovim)
+    let c = s:Compiler.new()
+    echo join(c.compile(p.parse(r)), "\n")
+  catch
+    echoerr substitute(v:throwpoint, '\.\.\zs\d\+', '\=s:numtoname(submatch(0))', 'g') . "\n" . v:exception
+  endtry
+endfunction
+
diff --git a/autoload/vital.vim b/autoload/vital.vim
new file mode 100644
index 00000000..f1ba849a
--- /dev/null
+++ b/autoload/vital.vim
@@ -0,0 +1,12 @@
+function! vital#of(name) abort
+  let files = globpath(&runtimepath, 'autoload/vital/' . a:name . '.vital', 1)
+  let file = split(files, "\n")
+  if empty(file)
+    throw 'vital: version file not found: ' . a:name
+  endif
+  let ver = readfile(file[0], 'b')
+  if empty(ver)
+    throw 'vital: invalid version file: ' . a:name
+  endif
+  return vital#_{substitute(ver[0], '\W', '', 'g')}#new()
+endfunction
diff --git a/autoload/vital/__vimlparser__/VimlParser.vim b/autoload/vital/__vimlparser__/VimlParser.vim
index 40a633bb..277f8325 100644
--- a/autoload/vital/__vimlparser__/VimlParser.vim
+++ b/autoload/vital/__vimlparser__/VimlParser.vim
@@ -4,30 +4,10 @@
 "
 " License: This file is placed in the public domain.
 
-function! vimlparser#import()
+function! s:import() abort
   return s:
 endfunction
 
-" @brief Read input as VimScript and return stringified AST.
-" @param input Input filename or string of VimScript.
-" @return Stringified AST.
-function! vimlparser#test(input, ...)
-  try
-    if a:0 > 0
-      let l:neovim = a:1
-    else
-      let l:neovim = 0
-    endif
-    let i = type(a:input) == 1 && filereadable(a:input) ? readfile(a:input) : split(a:input, "\n")
-    let r = s:StringReader.new(i)
-    let p = s:VimLParser.new(l:neovim)
-    let c = s:Compiler.new()
-    echo join(c.compile(p.parse(r)), "\n")
-  catch
-    echoerr substitute(v:throwpoint, '\.\.\zs\d\+', '\=s:numtoname(submatch(0))', 'g') . "\n" . v:exception
-  endtry
-endfunction
-
 function! s:numtoname(num)
   let sig = printf("function('%s')", a:num)
   for k in keys(s:)
diff --git a/autoload/vital/_vimlparser.vim b/autoload/vital/_vimlparser.vim
new file mode 100644
index 00000000..55104952
--- /dev/null
+++ b/autoload/vital/_vimlparser.vim
@@ -0,0 +1,9 @@
+let s:_plugin_name = expand('<sfile>:t:r')
+
+function! vital#{s:_plugin_name}#new() abort
+  return vital#{s:_plugin_name[1:]}#new()
+endfunction
+
+function! vital#{s:_plugin_name}#function(funcname) abort
+  silent! return function(a:funcname)
+endfunction
diff --git a/autoload/vital/vimlparser.vim b/autoload/vital/vimlparser.vim
new file mode 100644
index 00000000..b622a97e
--- /dev/null
+++ b/autoload/vital/vimlparser.vim
@@ -0,0 +1,328 @@
+let s:plugin_name = expand('<sfile>:t:r')
+let s:vital_base_dir = expand('<sfile>:h')
+let s:project_root = expand('<sfile>:h:h:h')
+let s:is_vital_vim = s:plugin_name is# 'vital'
+
+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('<sfile>'), '<SNR>\zs\d\+\ze__SID$')
+  endfunction
+  let s:_s = '<SNR>' . s:_SID() . '_'
+  function! s:_function(fstr) abort
+    return function(substitute(a:fstr, 's:', s:_s, 'g'))
+  endfunction
+endif
+
+function! vital#{s:plugin_name}#new() abort
+  return s:new(s:plugin_name)
+endfunction
+
+function! vital#{s:plugin_name}#import(...) abort
+  if !exists('s:V')
+    let s:V = s:new(s:plugin_name)
+  endif
+  return call(s:V.import, a:000, s:V)
+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
+  if a:name !~# '\v^\u\w*%(\.\u\w*)*$'
+    throw 'vital: Invalid module name: ' . a:name
+  endif
+  return s:_module_path(a:name) isnot# ''
+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:plugin_name() abort dict
+  return self._plugin_name
+endfunction
+let s:Vital.plugin_name = s:_function('s:plugin_name')
+
+function! s:_self_vital_files() abort
+  let builtin = printf('%s/__%s__/', s:vital_base_dir, s:plugin_name)
+  let installed = printf('%s/_%s/', s:vital_base_dir, s:plugin_name)
+  let base = builtin . ',' . installed
+  return split(globpath(base, '**/*.vim', 1), "\n")
+endfunction
+
+function! s:_global_vital_files() abort
+  let pattern = 'autoload/vital/__*__/**/*.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}#new())
+    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
+  let funcname = s:_import_func_name(self.plugin_name(), a:name)
+  try
+    return call(funcname, [])
+  catch /^Vim\%((\a\+)\)\?:E117/
+    return s:_get_builtin_module(a:name)
+  endtry
+endfunction
+
+function! s:_get_builtin_module(name) abort
+ return s:sid2sfuncs(s:_module_sid(a:name))
+endfunction
+
+if s:is_vital_vim
+  " For vital.vim, we can use s:_get_builtin_module directly
+  let s:Vital._get_module = s:_function('s:_get_builtin_module')
+else
+  let s:Vital._get_module = s:_function('s:_get_module')
+endif
+
+function! s:_import_func_name(plugin_name, module_name) abort
+  return printf('vital#_%s#%s#import', a:plugin_name, s:_dot_to_sharp(a:module_name))
+endfunction
+
+function! s:_module_sid(name) abort
+  let path = s:_module_path(a:name)
+  if !filereadable(path)
+    throw 'vital: module not found: ' . a:name
+  endif
+  let vital_dir = s:is_vital_vim ? '__\w\+__' : printf('_\{1,2}%s\%%(__\)\?', s:plugin_name)
+  let base = join([vital_dir, ''], '[/\\]\+')
+  let p = base . substitute('' . a:name, '\.', '[/\\\\]\\+', 'g')
+  let sid = s:_sid(path, p)
+  if !sid
+    call s:_source(path)
+    let sid = s:_sid(path, p)
+    if !sid
+      throw printf('vital: cannot get <SID> from path: %s', path)
+    endif
+  endif
+  return sid
+endfunction
+
+function! s:_module_path(name) abort
+  return get(s:_extract_files(a:name, s:vital_files()), 0, '')
+endfunction
+
+function! s:_module_sid_base_dir() abort
+  return s:is_vital_vim ? &rtp : s:project_root
+endfunction
+
+function! s:_dot_to_sharp(name) abort
+  return substitute(a:name, '\.', '#', 'g')
+endfunction
+
+function! s:_source(path) abort
+  execute 'source' fnameescape(a:path)
+endfunction
+
+" @vimlint(EVL102, 1, l:_)
+" @vimlint(EVL102, 1, l:__)
+function! s:_sid(path, filter_pattern) abort
+  let unified_path = s:_unify_path(a:path)
+  if has_key(s:cache_sid, unified_path)
+    return s:cache_sid[unified_path]
+  endif
+  for line in filter(split(s:_execute(':scriptnames'), "\n"), 'v:val =~# a:filter_pattern')
+    let [_, sid, path; __] = matchlist(line, '^\s*\(\d\+\):\s\+\(.\+\)\s*$')
+    if s:_unify_path(path) is# unified_path
+      let s:cache_sid[unified_path] = sid
+      return s:cache_sid[unified_path]
+    endif
+  endfor
+  return 0
+endfunction
+
+" A bug of execute() is fixed in Vim 8.0.0264
+if has('patch-8.0.0264')
+  let s:_execute = function('execute')
+else
+  function! s:_execute(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
+endif
+
+if filereadable(expand('<sfile>:r') . '.VIM') " is case-insensitive or not
+  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 <sfile> 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("\<SNR>")), '"[\\x" . printf("%0x", char2nr("\<SNR>"[v:val])) . "]"'), '')
+function! s:sid2sfuncs(sid) abort
+  let fs = split(s:_execute(printf(':function /^%s%s_', s:SNR, a:sid)), "\n")
+  let r = {}
+  let pattern = printf('\m^function\s<SNR>%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('<SNR>%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
diff --git a/autoload/vital/vimlparser.vital b/autoload/vital/vimlparser.vital
new file mode 100644
index 00000000..d8f2701e
--- /dev/null
+++ b/autoload/vital/vimlparser.vital
@@ -0,0 +1,3 @@
+vimlparser
+982e0bf3a5c181c0fa9d37bab784412f941f6b80
+

From 01518952ad2244fc0df3cf56bd5b30e98acf90d5 Mon Sep 17 00:00:00 2001
From: haya14busa <haya14busa@gmail.com>
Date: Fri, 24 Nov 2017 12:46:49 +0000
Subject: [PATCH 03/13] fix vimlparser#test()

---
 autoload/vimlparser.vim | 3 ++-
 1 file changed, 2 insertions(+), 1 deletion(-)

diff --git a/autoload/vimlparser.vim b/autoload/vimlparser.vim
index 315b5725..d8aee269 100644
--- a/autoload/vimlparser.vim
+++ b/autoload/vimlparser.vim
@@ -1,5 +1,7 @@
 let s:VimLParser = vital#vimlparser#import('VimlParser')
 
+call extend(s:, s:VimLParser.import())
+
 function! vimlparser#import()
   return s:VimLParser.import()
 endfunction
@@ -23,4 +25,3 @@ function! vimlparser#test(input, ...)
     echoerr substitute(v:throwpoint, '\.\.\zs\d\+', '\=s:numtoname(submatch(0))', 'g') . "\n" . v:exception
   endtry
 endfunction
-

From 64653fba14ce67881535354ca8f9c059e9b42bbc Mon Sep 17 00:00:00 2001
From: haya14busa <haya14busa@gmail.com>
Date: Fri, 24 Nov 2017 12:54:54 +0000
Subject: [PATCH 04/13] Add comment for vital module

---
 autoload/vimlparser.vim | 8 ++++++++
 1 file changed, 8 insertions(+)

diff --git a/autoload/vimlparser.vim b/autoload/vimlparser.vim
index d8aee269..bd02a8f0 100644
--- a/autoload/vimlparser.vim
+++ b/autoload/vimlparser.vim
@@ -1,7 +1,15 @@
+" The main code was moved to autoload/vital/__vimlparser__/VimlParser.vim to
+" make vim-vimlparser.vim vital-module.
+"
+" See https://github.com/vim-jp/vital.vim for vital module.
+
 let s:VimLParser = vital#vimlparser#import('VimlParser')
 
 call extend(s:, s:VimLParser.import())
 
+" To Vim plugin developer who want to depend on vim-vimlparser:
+" Please use vimlparser as vital-module instead of this autoload function.
+" We do not ensure that future changes are backward compatible.
 function! vimlparser#import()
   return s:VimLParser.import()
 endfunction

From bb0dbdbcce3d37d0d29875ffb0cf2d0cc3a662a2 Mon Sep 17 00:00:00 2001
From: haya14busa <haya14busa@gmail.com>
Date: Fri, 24 Nov 2017 13:40:58 +0000
Subject: [PATCH 05/13] fix extend broke s:VimlParser

---
 autoload/vimlparser.vim | 6 ++----
 1 file changed, 2 insertions(+), 4 deletions(-)

diff --git a/autoload/vimlparser.vim b/autoload/vimlparser.vim
index bd02a8f0..9c0929ad 100644
--- a/autoload/vimlparser.vim
+++ b/autoload/vimlparser.vim
@@ -3,15 +3,13 @@
 "
 " See https://github.com/vim-jp/vital.vim for vital module.
 
-let s:VimLParser = vital#vimlparser#import('VimlParser')
-
-call extend(s:, s:VimLParser.import())
+let s:VimLParser = vital#vimlparser#import('VimlParser').import()
 
 " To Vim plugin developer who want to depend on vim-vimlparser:
 " Please use vimlparser as vital-module instead of this autoload function.
 " We do not ensure that future changes are backward compatible.
 function! vimlparser#import()
-  return s:VimLParser.import()
+  return s:VimLParser
 endfunction
 
 " @brief Read input as VimScript and return stringified AST.

From 02306efc7bb4926539197a91fa396bbc45c85d83 Mon Sep 17 00:00:00 2001
From: haya14busa <haya14busa@gmail.com>
Date: Fri, 24 Nov 2017 14:00:25 +0000
Subject: [PATCH 06/13] fix vimlparser path

---
 Makefile          | 4 ++--
 js/jscompiler.vim | 4 ++--
 py/pycompiler.vim | 4 ++--
 3 files changed, 6 insertions(+), 6 deletions(-)

diff --git a/Makefile b/Makefile
index 1676b4d9..7e56a851 100644
--- a/Makefile
+++ b/Makefile
@@ -2,10 +2,10 @@ COMPILED_FILES:=js/vimlparser.js py/vimlparser.py
 
 all: $(COMPILED_FILES)
 
-js/vimlparser.js: autoload/vimlparser.vim js/jscompiler.vim js/vimlfunc.js
+js/vimlparser.js: autoload/vital/__vimlparser__/VimlParser.vim js/jscompiler.vim js/vimlfunc.js
 	scripts/jscompile.sh $< $@
 
-py/vimlparser.py: autoload/vimlparser.vim py/pycompiler.vim py/vimlfunc.py
+py/vimlparser.py: autoload/vital/__vimlparser__/VimlParser.vim py/pycompiler.vim py/vimlfunc.py
 	scripts/pycompile.sh $< $@
 
 clean_compiled:
diff --git a/js/jscompiler.vim b/js/jscompiler.vim
index da90892b..29d91a5c 100644
--- a/js/jscompiler.vim
+++ b/js/jscompiler.vim
@@ -885,7 +885,7 @@ endfunction
 
 function! s:parse_args() abort
   let v = [
-  \  fnamemodify(s:script_dir . '/../autoload/vimlparser.vim', ':p'),
+  \  fnamemodify(s:script_dir . '/../autoload/vital/__vimlparser__/VimlParser.vim', ':p'),
   \  fnamemodify(s:script_dir . '/vimlparser.js', ':p')
   \]
   let args = argv()[1:]
@@ -894,7 +894,7 @@ function! s:parse_args() abort
     if len(args) != 2
       throw 'invalid argument: ' . string(args)
     endif
-	let v = args
+    let v = args
   endif
   return v
 endfunction:
diff --git a/py/pycompiler.vim b/py/pycompiler.vim
index 8e387d99..a0e2da9b 100644
--- a/py/pycompiler.vim
+++ b/py/pycompiler.vim
@@ -857,7 +857,7 @@ endfunction
 
 function! s:parse_args() abort
   let v = [
-  \  fnamemodify(s:script_dir . '/../autoload/vimlparser.vim', ':p'),
+  \  fnamemodify(s:script_dir . '/../autoload/vital/__vimlparser__/VimlParser.vim', ':p'),
   \  fnamemodify(s:script_dir . '/vimlparser.py', ':p')
   \]
   let args = argv()[1:]
@@ -866,7 +866,7 @@ function! s:parse_args() abort
     if len(args) != 2
       throw 'invalid argument: ' . string(args)
     endif
-	let v = args
+    let v = args
   endif
   return v
 endfunction:

From 1bab11e0b1ad108245dd9fc6ff9b41db7f8d6ede Mon Sep 17 00:00:00 2001
From: haya14busa <haya14busa@gmail.com>
Date: Fri, 24 Nov 2017 14:28:01 +0000
Subject: [PATCH 07/13] fix s:numtoname

---
 autoload/vimlparser.vim                      | 33 ++++++++++++++++++--
 autoload/vital/__vimlparser__/VimlParser.vim | 14 ---------
 2 files changed, 31 insertions(+), 16 deletions(-)

diff --git a/autoload/vimlparser.vim b/autoload/vimlparser.vim
index 9c0929ad..c5035dcb 100644
--- a/autoload/vimlparser.vim
+++ b/autoload/vimlparser.vim
@@ -3,13 +3,13 @@
 "
 " See https://github.com/vim-jp/vital.vim for vital module.
 
-let s:VimLParser = vital#vimlparser#import('VimlParser').import()
+call extend(s:, vital#vimlparser#import('VimlParser').import())
 
 " To Vim plugin developer who want to depend on vim-vimlparser:
 " Please use vimlparser as vital-module instead of this autoload function.
 " We do not ensure that future changes are backward compatible.
 function! vimlparser#import()
-  return s:VimLParser
+  return s:
 endfunction
 
 " @brief Read input as VimScript and return stringified AST.
@@ -31,3 +31,32 @@ function! vimlparser#test(input, ...)
     echoerr substitute(v:throwpoint, '\.\.\zs\d\+', '\=s:numtoname(submatch(0))', 'g') . "\n" . v:exception
   endtry
 endfunction
+
+if has('patch-7.4.1842')
+  function! s:numtoname(num)
+    for k in keys(s:)
+      if type(s:[k]) == type({})
+        for name in keys(s:[k])
+          if type(s:[k][name]) == type(function('tr')) && get(s:[k][name], 'name') == a:num
+            return printf('%s.%s', k, name)
+          endif
+        endfor
+      endif
+    endfor
+    return a:num
+  endfunction
+else
+  function! s:numtoname(num)
+    let sig = printf("function('%s')", a:num)
+    for k in keys(s:)
+      if type(s:[k]) == type({})
+        for name in keys(s:[k])
+          if type(s:[k][name]) == type(function('tr')) && string(s:[k][name]) == sig
+            return printf('%s.%s', k, name)
+          endif
+        endfor
+      endif
+    endfor
+    return a:num
+  endfunction
+endif
diff --git a/autoload/vital/__vimlparser__/VimlParser.vim b/autoload/vital/__vimlparser__/VimlParser.vim
index 277f8325..a3429d81 100644
--- a/autoload/vital/__vimlparser__/VimlParser.vim
+++ b/autoload/vital/__vimlparser__/VimlParser.vim
@@ -8,20 +8,6 @@ function! s:import() abort
   return s:
 endfunction
 
-function! s:numtoname(num)
-  let sig = printf("function('%s')", a:num)
-  for k in keys(s:)
-    if type(s:[k]) == type({})
-      for name in keys(s:[k])
-        if type(s:[k][name]) == type(function('tr')) && string(s:[k][name]) == sig
-          return printf('%s.%s', k, name)
-        endif
-      endfor
-    endif
-  endfor
-  return a:num
-endfunction
-
 let s:NIL = []
 let s:TRUE = 1
 let s:FALSE = 0

From ff7fecb8493987f6dbccb46b8ace45afd3c07ddc Mon Sep 17 00:00:00 2001
From: haya14busa <haya14busa@gmail.com>
Date: Fri, 24 Nov 2017 14:38:11 +0000
Subject: [PATCH 08/13] make

---
 js/vimlparser.js | 72 ++++++++++++++++++++++++------------------------
 1 file changed, 36 insertions(+), 36 deletions(-)

diff --git a/js/vimlparser.js b/js/vimlparser.js
index 58a4e15e..9b6b6410 100644
--- a/js/vimlparser.js
+++ b/js/vimlparser.js
@@ -620,9 +620,9 @@ VimLParser.prototype.pop_context = function() {
 
 VimLParser.prototype.find_context = function(type) {
     var i = 0;
-    var __c3 = this.context;
-    for (var __i3 = 0; __i3 < __c3.length; ++__i3) {
-        var node = __c3[__i3];
+    var __c1 = this.context;
+    for (var __i1 = 0; __i1 < __c1.length; ++__i1) {
+        var node = __c1[__i1];
         if (node.type == type) {
             return i;
         }
@@ -1200,9 +1200,9 @@ VimLParser.prototype.find_command = function() {
         return this.find_command_cache[name];
     }
     var cmd = NIL;
-    var __c4 = this.builtin_commands;
-    for (var __i4 = 0; __i4 < __c4.length; ++__i4) {
-        var x = __c4[__i4];
+    var __c2 = this.builtin_commands;
+    for (var __i2 = 0; __i2 < __c2.length; ++__i2) {
+        var x = __c2[__i2];
         if (viml_stridx(x.name, name) == 0 && viml_len(name) >= x.minlen) {
             delete cmd;
             var cmd = x;
@@ -1210,18 +1210,18 @@ VimLParser.prototype.find_command = function() {
         }
     }
     if (this.neovim) {
-        var __c5 = this.neovim_additional_commands;
-        for (var __i5 = 0; __i5 < __c5.length; ++__i5) {
-            var x = __c5[__i5];
+        var __c3 = this.neovim_additional_commands;
+        for (var __i3 = 0; __i3 < __c3.length; ++__i3) {
+            var x = __c3[__i3];
             if (viml_stridx(x.name, name) == 0 && viml_len(name) >= x.minlen) {
                 delete cmd;
                 var cmd = x;
                 break;
             }
         }
-        var __c6 = this.neovim_removed_commands;
-        for (var __i6 = 0; __i6 < __c6.length; ++__i6) {
-            var x = __c6[__i6];
+        var __c4 = this.neovim_removed_commands;
+        for (var __i4 = 0; __i4 < __c4.length; ++__i4) {
+            var x = __c4[__i4];
             if (viml_stridx(x.name, name) == 0 && viml_len(name) >= x.minlen) {
                 delete cmd;
                 var cmd = NIL;
@@ -3666,9 +3666,9 @@ StringReader.prototype.__init__ = function(lines) {
     var offset = 0;
     while (lnum < viml_len(lines)) {
         var col = 0;
-        var __c7 = viml_split(lines[lnum], "\\zs");
-        for (var __i7 = 0; __i7 < __c7.length; ++__i7) {
-            var c = __c7[__i7];
+        var __c5 = viml_split(lines[lnum], "\\zs");
+        for (var __i5 = 0; __i5 < __c5.length; ++__i5) {
+            var c = __c5[__i5];
             viml_add(this.buf, c);
             viml_add(this.pos, [lnum + 1, col + 1, offset]);
             col += viml_len(c);
@@ -3677,9 +3677,9 @@ StringReader.prototype.__init__ = function(lines) {
         while (lnum + 1 < viml_len(lines) && viml_eqregh(lines[lnum + 1], "^\\s*\\\\")) {
             var skip = TRUE;
             var col = 0;
-            var __c8 = viml_split(lines[lnum + 1], "\\zs");
-            for (var __i8 = 0; __i8 < __c8.length; ++__i8) {
-                var c = __c8[__i8];
+            var __c6 = viml_split(lines[lnum + 1], "\\zs");
+            for (var __i6 = 0; __i6 < __c6.length; ++__i6) {
+                var c = __c6[__i6];
                 if (skip) {
                     if (c == "\\") {
                         var skip = FALSE;
@@ -3781,9 +3781,9 @@ StringReader.prototype.readline = function() {
 
 StringReader.prototype.getstr = function(begin, end) {
     var r = "";
-    var __c9 = viml_range(begin.i, end.i - 1);
-    for (var __i9 = 0; __i9 < __c9.length; ++__i9) {
-        var i = __c9[__i9];
+    var __c7 = viml_range(begin.i, end.i - 1);
+    for (var __i7 = 0; __i7 < __c7.length; ++__i7) {
+        var i = __c7[__i7];
         if (i >= viml_len(this.buf)) {
             break;
         }
@@ -4211,9 +4211,9 @@ Compiler.prototype.compile = function(node) {
 }
 
 Compiler.prototype.compile_body = function(body) {
-    var __c10 = body;
-    for (var __i10 = 0; __i10 < __c10.length; ++__i10) {
-        var node = __c10[__i10];
+    var __c8 = body;
+    for (var __i8 = 0; __i8 < __c8.length; ++__i8) {
+        var node = __c8[__i8];
         this.compile(node);
     }
 }
@@ -4312,9 +4312,9 @@ Compiler.prototype.compile_if = function(node) {
     this.incindent("  ");
     this.compile_body(node.body);
     this.decindent();
-    var __c11 = node.elseif;
-    for (var __i11 = 0; __i11 < __c11.length; ++__i11) {
-        var enode = __c11[__i11];
+    var __c9 = node.elseif;
+    for (var __i9 = 0; __i9 < __c9.length; ++__i9) {
+        var enode = __c9[__i9];
         this.out(" elseif %s", this.compile(enode.cond));
         this.incindent("  ");
         this.compile_body(enode.body);
@@ -4371,9 +4371,9 @@ Compiler.prototype.compile_try = function(node) {
     this.out("(try");
     this.incindent("  ");
     this.compile_body(node.body);
-    var __c12 = node.catch;
-    for (var __i12 = 0; __i12 < __c12.length; ++__i12) {
-        var cnode = __c12[__i12];
+    var __c10 = node.catch;
+    for (var __i10 = 0; __i10 < __c10.length; ++__i10) {
+        var cnode = __c10[__i10];
         if (cnode.pattern !== NIL) {
             this.decindent();
             this.out(" catch /%s/", cnode.pattern);
@@ -5363,9 +5363,9 @@ RegexpParser.prototype.get_token_sq_char_class = function() {
         var r = this.reader.read_alpha();
         if (this.reader.p(0) == ":" && this.reader.p(1) == "]") {
             this.reader.seek_cur(2);
-            var __c13 = class_names;
-            for (var __i13 = 0; __i13 < __c13.length; ++__i13) {
-                var name = __c13[__i13];
+            var __c11 = class_names;
+            for (var __i11 = 0; __i11 < __c11.length; ++__i11) {
+                var name = __c11[__i11];
                 if (r == name) {
                     return "[:" + name + ":]";
                 }
@@ -5498,9 +5498,9 @@ RegexpParser.prototype.getoctchrs = function() {
 
 RegexpParser.prototype.gethexchrs = function(n) {
     var r = "";
-    var __c14 = viml_range(n);
-    for (var __i14 = 0; __i14 < __c14.length; ++__i14) {
-        var i = __c14[__i14];
+    var __c12 = viml_range(n);
+    for (var __i12 = 0; __i12 < __c12.length; ++__i12) {
+        var i = __c12[__i12];
         var c = this.reader.peek();
         if (!isxdigit(c)) {
             break;

From b2fb2c94cc8e3bc9ec3fc94d5df0fbc0af543f46 Mon Sep 17 00:00:00 2001
From: haya14busa <haya14busa@gmail.com>
Date: Sat, 25 Nov 2017 10:30:53 +0000
Subject: [PATCH 09/13] :visual to make exit code 0

---
 js/jscompiler.vim | 9 +++++++++
 py/pycompiler.vim | 9 +++++++++
 2 files changed, 18 insertions(+)

diff --git a/js/jscompiler.vim b/js/jscompiler.vim
index 29d91a5c..ff805cbc 100644
--- a/js/jscompiler.vim
+++ b/js/jscompiler.vim
@@ -907,6 +907,15 @@ function! s:main() abort
     call writefile([v:exception], has('win32') ? 'conout$' : '/dev/stderr')
     cquit
   endtry
+  if mode(1) ==# 'ce'
+    " This :visual is needed to make exit code 0 for old vim.
+    " Exit code is set to 1, Vim exit from Ex mode after an error
+    " output is performed even if the error is caught by try-catch.
+    "
+    " This problem is fixed at Vim v.8.0.0184.
+    " https://github.com/vim/vim/releases/tag/v8.0.0184
+    visual
+  endif
 endfunction
 
 call s:main()
diff --git a/py/pycompiler.vim b/py/pycompiler.vim
index a0e2da9b..0b46761d 100644
--- a/py/pycompiler.vim
+++ b/py/pycompiler.vim
@@ -879,6 +879,15 @@ function! s:main() abort
     call writefile([v:exception], has('win32') ? 'conout$' : '/dev/stderr')
     cquit
   endtry
+  if mode(1) ==# 'ce'
+    " This :visual is needed to make exit code 0 for old vim.
+    " Exit code is set to 1, Vim exit from Ex mode after an error
+    " output is performed even if the error is caught by try-catch.
+    "
+    " This problem is fixed at Vim v.8.0.0184.
+    " https://github.com/vim/vim/releases/tag/v8.0.0184
+    visual
+  endif
 endfunction
 
 call s:main()

From d4278fdbbe7567f18b79b18510a490878b7bdebd Mon Sep 17 00:00:00 2001
From: haya14busa <haya14busa@gmail.com>
Date: Sat, 25 Nov 2017 10:45:51 +0000
Subject: [PATCH 10/13] profile only main file

---
 test/vimrc | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/test/vimrc b/test/vimrc
index e98989a5..4604974e 100644
--- a/test/vimrc
+++ b/test/vimrc
@@ -2,5 +2,5 @@ let &rtp .= ',' . getcwd()
 
 if $TEST_PROFILE != ''
   execute 'profile' 'start' $TEST_PROFILE
-  profile! file ./autoload/*
+  profile! file ./autoload/vital/__vimlparser__/*
 endif

From 048634f44c83a92508181907e06a564c8eaa7691 Mon Sep 17 00:00:00 2001
From: haya14busa <haya14busa@gmail.com>
Date: Sat, 25 Nov 2017 12:08:42 +0000
Subject: [PATCH 11/13] update scripts/update_builtin_commands.vim

---
 scripts/update_builtin_commands.vim | 4 ++--
 1 file changed, 2 insertions(+), 2 deletions(-)

diff --git a/scripts/update_builtin_commands.vim b/scripts/update_builtin_commands.vim
index 213bbc3a..2bec4e40 100644
--- a/scripts/update_builtin_commands.vim
+++ b/scripts/update_builtin_commands.vim
@@ -129,9 +129,9 @@ function! g:VimLParserNewCmds(ex_cmds_h) abort
   let new_cmds = s:gen_new_builtin(vimlparser#import().VimLParser.builtin_commands, latest)
   let generated_text = s:gen_viml(new_cmds)
   if generated_text == ''
-    verbose echo 's:VimLParser.builtin_commands in autoload/vimlparser.vim is up-to-date.'
+    verbose echo 's:VimLParser.builtin_commands is up-to-date.'
   else
-    verbose echo "Append following lines to s:VimLParser.builtin_commands in autoload/vimlparser.vim\n"
+    verbose echo "Append following lines to s:VimLParser.builtin_commands\n"
     verbose echo generated_text
   endif
 endfunction

From 190ad40fbf1b02bfb97e4f8b2f45224fe857720f Mon Sep 17 00:00:00 2001
From: haya14busa <haya14busa@gmail.com>
Date: Sat, 25 Nov 2017 12:10:40 +0000
Subject: [PATCH 12/13] s/VimlParser/VimLParser/

---
 Makefile                                                      | 4 ++--
 autoload/vimlparser.vim                                       | 4 ++--
 .../vital/__vimlparser__/{VimlParser.vim => VimLParser.vim}   | 0
 js/jscompiler.vim                                             | 2 +-
 py/pycompiler.vim                                             | 2 +-
 5 files changed, 6 insertions(+), 6 deletions(-)
 rename autoload/vital/__vimlparser__/{VimlParser.vim => VimLParser.vim} (100%)

diff --git a/Makefile b/Makefile
index 30c7a364..57aa651f 100644
--- a/Makefile
+++ b/Makefile
@@ -2,10 +2,10 @@ COMPILED_FILES:=js/vimlparser.js py/vimlparser.py
 
 all: $(COMPILED_FILES)
 
-js/vimlparser.js: autoload/vital/__vimlparser__/VimlParser.vim js/jscompiler.vim js/vimlfunc.js
+js/vimlparser.js: autoload/vital/__vimlparser__/VimLParser.vim js/jscompiler.vim js/vimlfunc.js
 	scripts/jscompile.sh $< $@
 
-py/vimlparser.py: autoload/vital/__vimlparser__/VimlParser.vim py/pycompiler.vim py/vimlfunc.py
+py/vimlparser.py: autoload/vital/__vimlparser__/VimLParser.vim py/pycompiler.vim py/vimlfunc.py
 	scripts/pycompile.sh $< $@
 
 clean_compiled:
diff --git a/autoload/vimlparser.vim b/autoload/vimlparser.vim
index c5035dcb..2f61271a 100644
--- a/autoload/vimlparser.vim
+++ b/autoload/vimlparser.vim
@@ -1,9 +1,9 @@
-" The main code was moved to autoload/vital/__vimlparser__/VimlParser.vim to
+" The main code was moved to autoload/vital/__vimlparser__/VimLParser.vim to
 " make vim-vimlparser.vim vital-module.
 "
 " See https://github.com/vim-jp/vital.vim for vital module.
 
-call extend(s:, vital#vimlparser#import('VimlParser').import())
+call extend(s:, vital#vimlparser#import('VimLParser').import())
 
 " To Vim plugin developer who want to depend on vim-vimlparser:
 " Please use vimlparser as vital-module instead of this autoload function.
diff --git a/autoload/vital/__vimlparser__/VimlParser.vim b/autoload/vital/__vimlparser__/VimLParser.vim
similarity index 100%
rename from autoload/vital/__vimlparser__/VimlParser.vim
rename to autoload/vital/__vimlparser__/VimLParser.vim
diff --git a/js/jscompiler.vim b/js/jscompiler.vim
index ff805cbc..86521aad 100644
--- a/js/jscompiler.vim
+++ b/js/jscompiler.vim
@@ -885,7 +885,7 @@ endfunction
 
 function! s:parse_args() abort
   let v = [
-  \  fnamemodify(s:script_dir . '/../autoload/vital/__vimlparser__/VimlParser.vim', ':p'),
+  \  fnamemodify(s:script_dir . '/../autoload/vital/__vimlparser__/VimLParser.vim', ':p'),
   \  fnamemodify(s:script_dir . '/vimlparser.js', ':p')
   \]
   let args = argv()[1:]
diff --git a/py/pycompiler.vim b/py/pycompiler.vim
index 0b46761d..475f5425 100644
--- a/py/pycompiler.vim
+++ b/py/pycompiler.vim
@@ -857,7 +857,7 @@ endfunction
 
 function! s:parse_args() abort
   let v = [
-  \  fnamemodify(s:script_dir . '/../autoload/vital/__vimlparser__/VimlParser.vim', ':p'),
+  \  fnamemodify(s:script_dir . '/../autoload/vital/__vimlparser__/VimLParser.vim', ':p'),
   \  fnamemodify(s:script_dir . '/vimlparser.py', ':p')
   \]
   let args = argv()[1:]

From 67cead6b35cfbdc4ee375b1dd5fa2bde1121b50b Mon Sep 17 00:00:00 2001
From: haya14busa <haya14busa@gmail.com>
Date: Sun, 26 Nov 2017 01:03:18 +0000
Subject: [PATCH 13/13] Add comments about public API as vital-module

---
 README.mkd                                   |  2 +-
 autoload/vital/__vimlparser__/VimLParser.vim | 29 ++++++++++++++++++--
 2 files changed, 28 insertions(+), 3 deletions(-)

diff --git a/README.mkd b/README.mkd
index 9745d3a0..29b9ccac 100644
--- a/README.mkd
+++ b/README.mkd
@@ -14,7 +14,7 @@ This parser provide same feature for following languages.
 
 * Vim script
 * Python
-* Javascript 
+* JavaScript
 
 ## Example
 
diff --git a/autoload/vital/__vimlparser__/VimLParser.vim b/autoload/vital/__vimlparser__/VimLParser.vim
index 8dc74c1d..80df8a04 100644
--- a/autoload/vital/__vimlparser__/VimLParser.vim
+++ b/autoload/vital/__vimlparser__/VimLParser.vim
@@ -4,6 +4,31 @@
 "
 " License: This file is placed in the public domain.
 
+" s:import is only public API and you can access this script-local function via
+" vital.vim. See https://github.com/vim-jp/vital.vim for more detail.
+"
+" Historically, vimlparser.vim is not vital module and there are other exported
+" functions as vital-module, but they are not expected to be used.
+" The one reason we didn't make them as private function (add `_` prefix to
+" script local function) is that this file will be translated into other
+" languages like Python and JavaScript. Also, vimlparser developers have no
+" need to learn vital.vim convention and develop vimlparser same as before as
+" much as possible.
+"
+" Example:
+" 	call extend(s:, vital#vital#import('VimLParser').import())
+"
+" 	function! s:run() abort
+" 	  let code = [
+" 	  \  'let s:message = printf("hello %d", 1+(2*3))'
+" 	  \ ]
+" 	  let r = s:StringReader.new(code)
+" 	  let p = s:VimLParser.new()
+" 	  let c = s:Compiler.new()
+" 	  echo join(c.compile(p.parse(r)), "\n")
+" 	endfunction
+"
+" 	call s:run()
 function! s:import() abort
   return s:
 endfunction
@@ -906,7 +931,7 @@ function! s:VimLParser.find_command()
     endif
   endfor
 
-  if self.neovim  
+  if self.neovim
     for x in self.neovim_additional_commands
       if stridx(x.name, name) == 0 && len(name) >= x.minlen
         unlet cmd
@@ -923,7 +948,7 @@ function! s:VimLParser.find_command()
       endif
     endfor
   endif
-  
+
   " FIXME: user defined command
   if (cmd is s:NIL || cmd.name ==# 'Print') && name =~# '^[A-Z]'
     let name .= self.reader.read_alnum()