/ZFVimIM

Vim Input Method by pure vim script, support: user word, dynamic word priority, cloud db files

Primary LanguageVim Script

introduction

Input Method by pure vim script, inspired by VimIM

Outstanding features / why another remake:

  • more friendly long sentence match and better predict logic
  • predict from multiple db without switching dbs
  • auto create user word and re-order word priority accorrding to your input history
  • cloud input, auto pull and push your db file from/to Github
  • solve many VimIM's issues:
    • better txt db load performance if has('python')
    • auto disable and re-enable complete engines when using input method
    • sync input method state acrossing buffers

if you like my work, check here for a list of my vim plugins

how to use

  1. requirement:

    • v:version >= 703, older version may work, but not tested
    • (optional) git, for db update
    • (optional) vim8 with job or neovim, and Plug 'ZSaberLv0/ZFVimJob', for async db update
    • (optional) has('python') or has('python3'), for better db load performance
  2. use Vundle or any other plugin manager you like to install

    Plugin 'ZSaberLv0/ZFVimIM'
    Plugin 'YourUserName/YourDbRepo' " repo that contain db files, see `cloud input (minimal recommend config)`
    
    Plugin 'ZSaberLv0/ZFVimJob' " optional, for async db update
    Plugin 'ZSaberLv0/ZFVimGitUtil' " optional, see `g:ZFVimIM_cloudAsync_autoCleanup`
    
  3. use ;; to toggle input method

    default keymaps:

    nnoremap <expr><silent> ;; ZFVimIME_keymap_toggle_n()
    inoremap <expr> ;; ZFVimIME_keymap_toggle_i()
    vnoremap <expr> ;; ZFVimIME_keymap_toggle_v()
    
    nnoremap <expr><silent> ;: ZFVimIME_keymap_next_n()
    inoremap <expr> ;: ZFVimIME_keymap_next_i()
    vnoremap <expr> ;: ZFVimIME_keymap_next_v()
    
    nnoremap <expr><silent> ;, ZFVimIME_keymap_add_n()
    inoremap <expr> ;, ZFVimIME_keymap_add_i()
    xnoremap <expr> ;, ZFVimIME_keymap_add_v()
    
    nnoremap <expr><silent> ;. ZFVimIME_keymap_remove_n()
    inoremap <expr> ;. ZFVimIME_keymap_remove_i()
    xnoremap <expr> ;. ZFVimIME_keymap_remove_v()
    

    you may disable default keymap by let g:ZFVimIM_keymap = 0

  4. during input:

    • scroll page by - or =
    • input and choose word by <space> or 0~9
    • choose head or tail word by [ or ]
  5. your input history would be recorded and automatically push to github, see cloud input below for more info

cloud input

cloud input (minimal recommend config)

  1. fork ZSaberLv0/ZFVimIM_pinyin_base to Plugin 'YourUserName/ZFVimIM_pinyin_base'

    Plugin 'ZSaberLv0/ZFVimIM'
    Plugin 'YourUserName/ZFVimIM_pinyin_base'
    Plugin 'ZSaberLv0/ZFVimIM_openapi'
    Plugin 'ZSaberLv0/ZFVimJob'
    Plugin 'ZSaberLv0/ZFVimGitUtil'
    
  2. supply your git info (make sure it has git push permission)

    let g:ZFVimIM_pinyin_gitUserEmail='YourEmail'
    let g:ZFVimIM_pinyin_gitUserName='YourUserName'
    let g:ZFVimIM_pinyin_gitUserToken='YourGithubAccessToken'
    

cloud input (detail config)

once configured properly, your db changes would be pushed to Github automatically

requirement:

if it's hard to support async mode, you may also:

  • pull and push manually by :call ZFVimIM_download() and :call ZFVimIM_upload()
  • automatically ask you to input git info to push before exit vim, by let g:ZFVimIM_cloudSync_enable=1

of course, you must have push permission for db repo, feel free to fork the default repo (ZSaberLv0/ZFVimIM_pinyin_base), or supply your own db repo

NOTE:

since db files are pretty personal, the default db only contains single word, words would be created during your usage

if you prefer other choices, see db samples:

NOTE:

your db repo may contain many commits after long time usage, which may cause a huge .git dir, it's recommended to clean up it occasionally, by:

  • delete and re-create the repo
  • if you have push --force permission, use ZSaberLv0/ZFVimGitUtil's :ZFGitHardRemoveAllHistory to remove all history commits, or use g:ZFVimIM_cloudAsync_autoCleanup for short

detailed

configs

  • let g:ZFVimIM_autoAddWordLen=3*4

    when you choose word and the word's byte length less than this value, we would add the word to db file automatically (ignored when g:ZFVimIM_autoAddWordChecker is set)

  • let g:ZFVimIM_autoAddWordChecker=[]

    list of function to check whether need to add user word

    function! MyChecker(userWord)
        let needAdd = ...
        return needAdd
    endfunction
    let g:ZFVimIM_autoAddWordChecker=[function('MyChecker')]
    

    when any of checker returned 0, we won't add user word

  • let g:ZFVimIM_symbolMap = {}

    used to transform unicode symbols during input

    it's empty by default, typical config for Chinese:

    let g:ZFVimIM_symbolMap = {
                \   ' ' : [''],
                \   '`' : ['·'],
                \   '!' : ['!'],
                \   '$' : ['¥'],
                \   '^' : ['……'],
                \   '-' : [''],
                \   '_' : ['——'],
                \   '(' : ['('],
                \   ')' : [')'],
                \   '[' : ['【'],
                \   ']' : ['】'],
                \   '<' : ['《'],
                \   '>' : ['》'],
                \   '\' : ['、'],
                \   '/' : ['、'],
                \   ';' : [';'],
                \   ':' : [':'],
                \   ',' : [','],
                \   '.' : ['。'],
                \   '?' : ['?'],
                \   "'" : ['‘', '’'],
                \   '"' : ['“', '”'],
                \   '0' : [''],
                \   '1' : [''],
                \   '2' : [''],
                \   '3' : [''],
                \   '4' : [''],
                \   '5' : [''],
                \   '6' : [''],
                \   '7' : [''],
                \   '8' : [''],
                \   '9' : [''],
                \ }
    

    note, if you want to change this setting at runtime, you should use call ZFVimIME_stop() | call ZFVimIME_start() to restart to take effect, or, add autocmd to ZFVimIM_event_OnStart to setup this value

  • let g:ZFVimIM_cachePath=$HOME.'/.vim_cache/ZFVimIM'

    cache path for temp files

  • let g:ZFVimIM_cloudAsync_outputTo={...}

    for async cloud input, output log to where (see ZFJobOutput), default:

    let g:ZFVimIM_cloudAsync_outputTo = {
                \   'outputType' : 'statusline',
                \   'outputId' : 'ZFVimIM_cloud_async',
                \ }
    
  • let g:ZFVimIM_cloudAsync_autoCleanup=30

    for async cloud input only, we would try to remove all history commits if:

    • g:ZFVimIM_cloudAsync_autoCleanup greater than 0
    • your git rev-list --count HEAD exceeds g:ZFVimIM_cloudAsync_autoCleanup
    • have ZSaberLv0/ZFVimGitUtil installed

    NOTE:

    • this require you have git push --force permission, if not, please disable this feature, otherwise your commits may lost occasionally (each time when commits exceeds g:ZFVimIM_cloudAsync_autoCleanup)
  • let g:ZFVimIM_cloudAsync_autoInit=1

    for async cloud input only, when on, we would load db when VimEnter, to reduce the time you first ZFVimIME_start()

functions

  • ZFVimIME_start() ZFVimIME_stop() ZFVimIME_toggle() ZFVimIME_next()

    start or stop, must called during Insert Mode, as <c-r>=ZFVimIME_start()<cr>

  • :IMAdd word key or ZFVimIM_wordAdd(word, key)

    manually add word

  • :IMRemove word [key] or ZFVimIM_wordRemove(word [, key])

    manually remove word

  • :IMReorder word [key] or ZFVimIM_wordReorder(word [, key])

    manually reorder word priority, by reducing it's input history count to a proper value

  • ZFVimIM_complete(key [, option])

    • option

      {
        'sentence' : '0/1',
        'crossDb' : 'maxNum, default to g:ZFVimIM_crossDbLimit',
        'predict' : 'maxNum, default to g:ZFVimIM_predictLimit',
        'match' : 'maxNum, default to -1',
        'db' : {...}, // which db to use, empty for current
      }
      
    • return

      [
        {
          'dbId' : 'match from which db',
          'len' : 'match count in key',
          'key' : 'matched full key',
          'word' : 'matched word',
          'type' : 'type of completion: sentence/match/predict',
          'sentenceList' : [ // (optional) for sentence type only, list of word that complete as sentence
            {
              'key' : '',
              'word' : '',
            },
          ],
        },
        ...
      ]
      

    note, you may supply your own function named ZFVimIM_complete to override the default one, and use ZFVimIM_completeDefault(key, option) to achieve custom IME complete

functions (for db repo)

  • ZFVimIM_dbInit(option)

    to register a db, option:

    {
      'name' : '(required) name of your db',
      'priority' : '(optional) 100 by default, smaller value means higher priority',
      'switchable' : '(optional) 1 by default, when off, won't be enabled by ZFVimIME_keymap_next_n() series',
      'dbCallback' : '(optional) func(key, option), see ZFVimIM_complete',
                     // when dbCallback supplied, words would be fetched from this callback instead
      'menuLabel' : '(optional) string or function(item), when not empty, show label after key hint',
      'implData' : { // extra data for impl
      },
    }
    

    return db object which would stored in g:ZFVimIM_db

  • ZFVimIM_cloudRegister(cloudOption)

    register cloud info, when registered, we would try to pull/push from/to remote repo

    cloudOption:

    {
      'mode' : '(optional) git/local',
      'cloudInitMode' : '(optional) forceAsync/forceSync/preferAsync/preferSync',
      'dbId' : '(required) dbId generated by ZFVimIM_dbInit()'
      'repoPath' : '(required) git/local repo path',
      'dbFile' : '(required) db file path relative to repoPath, must start with /',
      'dbCountFile' : '(optional) db count file path relative to repoPath, must start with /',
      'gitUserEmail' : '(optional) git user email',
      'gitUserName' : '(optional) git user name',
      'gitUserToken' : '(optional) git access token or password',
    }
    

Make your own db

  1. supply your db file with this format:

    a 啊 阿
    a 锕
    ai 爱 唉
    ohayou お早う おはようございます
    tang _(:з」∠)_
    haha ^\ ^
    

    key can be a-z, word can be any string (if word contain space, you may escape it by \ )

    save it as utf-8 encoding

  2. format the db file to ensure it's valid

    call ZFVimIM_dbNormalize('/path/to/dbFile')
    

    this may take a long time, but for only once

  3. put the db file to your git repo, according to the db samples below

db samples

  • ZSaberLv0/ZFVimIM_openapi : pinyin repo using thirdparty's openapi, recommended to install as default, and it shows the way to achieve complex async db logic
  • ZSaberLv0/ZFVimIM_pinyin_base : base pinyin repo that only contain single word, recommended if you care about performance or want to create personal user word during usage
  • ZSaberLv0/ZFVimIM_wubi_base : wubi converted from ywvim, I'm not familiar with wubi, just put it here in case you want to test
  • ZSaberLv0/ZFVimIM_pinyin : pinyin repo which I personally used, update frequently
  • ZSaberLv0/ZFVimIM_pinyin_huge : huge pinyin repo that contains many words, it's converted from other IME and haven't been daily used, which may contain many useless words, I put it here just in case you prefer huge db or want to test huge db's performance

FAQ

  • Q: How to use in Command-line (search or command) ?

    A: ZFVimIM can be used inside command-line-window, you may:

    • (in normal mode) use q: or q/ to enter command-line-window

    • (when entering command) use these keymaps:

      function! ZF_Setting_cmdEdit()
          let cmdtype = getcmdtype()
          if cmdtype != ':' && cmdtype != '/'
              return ''
          endif
          call feedkeys("\<c-c>q" . cmdtype . 'k0' . (getcmdpos() - 1) . 'li', 'nt')
          return ''
      endfunction
      cnoremap <silent><expr> ;; ZF_Setting_cmdEdit()
      
  • Q: external db source?

    A: the ZSaberLv0/ZFVimIM_openapi is a good example, which achieves:

    • using external source to supply db contents
    • async mode
  • Q: lazy db load?

    A: you may manually use these methods to achieve lazy load:

    • register:
      • ZFVimIM_dbInit(...) : register a empty db that can be toggle by ZFVimIME_keymap_toggle_n() or ZFVimIME_keymap_next_n()
    • db load:
      • ZFVimIM_cloudRegister(...) : (recommended) register cloud setting, and would load db content when called
      • ZFVimIM_dbLoad(...) : to load actual db content, can be called separately for split db, new data would be merged to old data
      • g:ZFVimIM_db : (not recommended) manually modify internal db data
  • Q: apply changes / input history to local files only?

    A: use mode='local' option when ZFVimIM_cloudRegister(...), and all changes would be stored to local file only, example:

      ```
      let db = ZFVimIM_dbInit({
                  \   'name' : 'YourDb',
                  \ })
      call ZFVimIM_cloudRegister({
                  \   'mode' : 'local',
                  \   'dbId' : db['dbId'],
                  \   'repoPath' : '/path/to/repo',
                  \   'dbFile' : '/YourDbFile',
                  \   'dbCountFile' : '/YourDbCountFile',
                  \ })
      ```
    
  • Q: strange complete popup?

    A: we use omnifunc to achieve IM popup, which would conflict with most of complete engines, by default, we would automatically disable complete engines when IM started, if your other plugins conflict with IM, you may disable it manually (see this)

    also, if any strange behaviors occurred, :verbose set omnifunc? to check whether it's changed by other plugins

  • Q: meet some weird problem, how to check log?

    A: use :IMCloudLog to check first, if not enough:

    1. put this in your vimrc: let g:ZFJobVerboseLogEnable = 1
    2. restart vim and reproduce your problem
    3. write log file by: :call writefile(g:ZFJobVerboseLog, 'log.txt')

    WARNING : the verbose log may contain your git access token or password, please verify before posting the log file to public

known issue

  • too slow

    check first: has('python') and ZFVimJob is installed and available, without them, the pure vim script fallback is always very slow (about 2 seconds for 200KB db file)

    if your db file is very large, it's slow to save and load db even if has('python'), because reading and processing large files also takes a long time

    this plugin is designed lightweight that can fallback to pure vimscript, so, there's no plan to completely move db data to python side (further more, async complete popup would break :lmap logic, and require features like LSP plugins, no plan to achieve this too)

    if you want to benchmark:

    1. let g:ZFVimIM_DEBUG_profile = 1
    2. input freely
    3. call ZFVimIM_DEBUG_profileInfo() to check which step consumed most time

    PS: you may want to check ZSaberLv0/ZFVimIM_openapi for how to use external tool to supply db contents

  • use with LSP plugins

    it's possible, but it's a better design to make a external executable for LSP plugins, not some vimscript like this plugin, so, no plan on this

    if you really want to hack, there's two idea:

    • use ZFVimIM_complete() to get word completion, and supply things like omnifunc for LSP plugins
    • use python or other tools to parse db files and supply LSP plugins
  • can not use in input()

    unfortunately, I've no idea how to make lmap work in input(), and there's no plan to make complex cmap to achieve this

    of course, if you have better solution, PR is always welcomed

  • slow in cygwin's vim

    we need a lot of cygpath call to ensure paths can be recognized by both cygwin shell and Windows external command (/usr/bin/git and C:/Program Files/Git/bin/git for example), the typical and main reason is C:/Program Files/Git/bin/git can not read cygwin's path (/cygdrive/c/file)

    since cygwin's vim is not common use case, and solving this problem require a lot of tedious conversion in job script, I would prefer keep it simple stupid

    if you are sure all of your external commands is inside cygwin shell, you may let g:ZFVimIM_disableCygpath=1 to disable the cygpath call (the required commands are git and python/python3)