" Copyright 2013 LuoChen (luochen1990@gmail.com). Licensed under the Apache License 2.0. if exists('s:loaded') | finish | endif | let s:loaded = 1 fun s:trim(s) return substitute(a:s, '\v^\s*(.{-})\s*$', '\1', '') endfun fun s:concat(strs) return join(filter(a:strs, "v:val !~ '^[ ]*$'"), ',') endfun fun s:resolve_parenthesis_with(init_state, p) let [paren, contained, containedin, contains_prefix, contains, op] = a:init_state let p = (type(a:p) == type([])) ? ((len(a:p) == 3) ? printf('start=#%s# step=%s end=#%s#', a:p[0], op, a:p[-1]) : printf('start=#%s# end=#%s#', a:p[0], a:p[-1])) : a:p "NOTE: preprocess the old style parentheses config let ls = split(p, '\v%(%(start|step|end)\=(.)%(\1@!.)*\1[^ ]*|\w+%(\=[^ ]*)?) ?\zs', 0) for s in ls let [k, v] = [matchstr(s, '^[^=]\+\ze\(=\|$\)'), matchstr(s, '^[^=]\+=\zs.*')] if k == 'step' let op = s:trim(v) elseif k == 'contains_prefix' let contains_prefix = s:trim(v) elseif k == 'contains' let contains = s:concat([contains, s:trim(v)]) elseif k == 'containedin' let containedin = s:concat([containedin, s:trim(v)]) elseif k == 'contained' let contained = 1 else let paren .= s endif endfor let rst = [paren, contained, containedin, contains_prefix, contains, op] "echom json_encode(rst) return rst endfun fun s:resolve_parenthesis_from_config(config) return s:resolve_parenthesis_with(['', 0, '', a:config.contains_prefix, '', a:config.operators], a:config.parentheses_options) endfun fun s:synID(prefix, group, lv, id) return a:prefix.'_lv'.a:lv.'_'.a:group.a:id endfun fun s:synGroupID(prefix, group, lv) return a:prefix.a:group.'_lv'.a:lv endfun fun rainbow#syn(config) let conf = a:config let prefix = conf.syn_name_prefix let cycle = conf.cycle let glob_paran_opts = s:resolve_parenthesis_from_config(conf) let b:rainbow_loaded = cycle for id in range(len(conf.parentheses)) let [paren, contained, containedin, contains_prefix, contains, op] = s:resolve_parenthesis_with(glob_paran_opts, conf.parentheses[id]) for lv in range(cycle) let lv2 = ((lv + cycle - 1) % cycle) let [rid, pid, gid2] = [s:synID(prefix, 'r', lv, id), s:synID(prefix, 'p', lv, id), s:synGroupID(prefix, 'Regions', lv2)] if len(op) > 2 exe 'syn match '.s:synID(prefix, 'o', lv, id).' '.op.' containedin='.s:synID(prefix, 'r', lv, id).' contained' endif let real_contained = (lv == 0)? (contained? 'contained ' : '') : 'contained ' let real_containedin = (lv == 0)? s:concat([containedin, '@'.gid2]) : '@'.gid2 let real_contains = s:concat([contains_prefix, contains]) exe 'syn region '.rid.' matchgroup='.pid.' '.real_contained.'containedin='.real_containedin.' contains='.real_contains.' '.paren endfor endfor for lv in range(cycle) exe 'syn cluster '.s:synGroupID(prefix, 'Regions', lv).' contains='.join(map(range(len(conf.parentheses)), 's:synID(prefix, "r", lv, v:val)'), ',') exe 'syn cluster '.s:synGroupID(prefix, 'Parentheses', lv).' contains='.join(map(range(len(conf.parentheses)), 's:synID(prefix, "p", lv, v:val)'), ',') exe 'syn cluster '.s:synGroupID(prefix, 'Operators', lv).' contains='.join(map(range(len(conf.parentheses)), 's:synID(prefix, "o", lv, v:val)'), ',') endfor exe 'syn cluster '.prefix.'Regions contains='.join(map(range(cycle), '"@".s:synGroupID(prefix, "Regions", v:val)'), ',') exe 'syn cluster '.prefix.'Parentheses contains='.join(map(range(cycle), '"@".s:synGroupID(prefix, "Parentheses", v:val)'), ',') exe 'syn cluster '.prefix.'Operators contains='.join(map(range(cycle), '"@".s:synGroupID(prefix, "Operators", v:val)'), ',') if has_key(conf, 'after') | for cmd in conf.after | exe cmd | endfor | endif endfun fun rainbow#syn_clear(config) let conf = a:config let prefix = conf.syn_name_prefix for id in range(len(conf.parentheses)) for lv in range(conf.cycle) let [rid, oid] = [s:synID(prefix, 'r', lv, id), s:synID(prefix, 'o', lv, id)] exe 'syn clear '.rid exe 'syn clear '.oid endfor endfor endfun fun rainbow#hi(config) let conf = a:config let prefix = conf.syn_name_prefix for id in range(len(conf.parentheses)) for lv in range(conf.cycle) let [pid, oid] = [s:synID(prefix, 'p', lv, id), s:synID(prefix, 'o', lv, id)] let ctermfg = conf.ctermfgs[lv % len(conf.ctermfgs)] let guifg = conf.guifgs[lv % len(conf.guifgs)] let cterm = conf.cterms[lv % len(conf.cterms)] let gui = conf.guis[lv % len(conf.guis)] let hi_style = 'ctermfg='.ctermfg.' guifg='.guifg.(len(cterm) > 0 ? ' cterm='.cterm : '').(len(gui) > 0 ? ' gui='.gui : '') exe 'hi '.pid.' '.hi_style exe 'hi '.oid.' '.hi_style endfor endfor endfun fun rainbow#hi_clear(config) let conf = a:config let prefix = conf.syn_name_prefix for id in range(len(conf.parentheses)) for lv in range(conf.cycle) let [pid, oid] = [s:synID(prefix, 'p', lv, id), s:synID(prefix, 'o', lv, id)] exe 'hi clear '.pid exe 'hi clear '.oid endfor endfor endfun