diff options
author | Adam Stankiewicz <sheerun@sher.pl> | 2019-09-06 14:32:07 +0200 |
---|---|---|
committer | Adam Stankiewicz <sheerun@sher.pl> | 2019-09-06 14:32:07 +0200 |
commit | 84ec4eedcdd2892249b5369f91a6dd1d12fef2fc (patch) | |
tree | 6c9806851123656af2b71f6c6f5d89649442909c /indent/tsx.vim | |
parent | 66b769328c4511b2273f01c70de971c41f6964dd (diff) | |
download | vim-polyglot-4.0.0.tar.gz vim-polyglot-4.0.0.zip |
Switch typescript provider, closes #428v4.0.0
Diffstat (limited to 'indent/tsx.vim')
-rw-r--r-- | indent/tsx.vim | 114 |
1 files changed, 114 insertions, 0 deletions
diff --git a/indent/tsx.vim b/indent/tsx.vim new file mode 100644 index 00000000..906ee222 --- /dev/null +++ b/indent/tsx.vim @@ -0,0 +1,114 @@ +if exists('g:polyglot_disabled') && index(g:polyglot_disabled, 'typescript') != -1 + finish +endif + +" Save the current JavaScript indentexpr. +let b:tsx_ts_indentexpr = &indentexpr + +" Prologue; load in XML indentation. +if exists('b:did_indent') + let s:did_indent=b:did_indent + unlet b:did_indent +endif +exe 'runtime! indent/xml.vim' +if exists('s:did_indent') + let b:did_indent=s:did_indent +endif + +setlocal indentexpr=GetTsxIndent() + +" JS indentkeys +setlocal indentkeys=0{,0},0),0],0\,,!^F,o,O,e +" XML indentkeys +setlocal indentkeys+=*<Return>,<>>,<<>,/ + +" Multiline end tag regex (line beginning with '>' or '/>') +let s:endtag = '^\s*\/\?>\s*;\=' +let s:startexp = '[\{\(]\s*$' + +" Get all syntax types at the beginning of a given line. +fu! SynSOL(lnum) + return map(synstack(a:lnum, 1), 'synIDattr(v:val, "name")') +endfu + +" Get all syntax types at the end of a given line. +fu! SynEOL(lnum) + let lnum = prevnonblank(a:lnum) + let col = strlen(getline(lnum)) + return map(synstack(lnum, col), 'synIDattr(v:val, "name")') +endfu + +" Check if a syntax attribute is XMLish. +fu! SynAttrXMLish(synattr) + return a:synattr =~ "^xml" || a:synattr =~ "^tsx" +endfu + +" Check if a synstack is XMLish (i.e., has an XMLish last attribute). +fu! SynXMLish(syns) + return SynAttrXMLish(get(a:syns, -1)) +endfu + +" Check if a synstack denotes the end of a TSX block. +fu! SynTSXBlockEnd(syns) + return get(a:syns, -1) =~ '\%(ts\|typescript\)Braces' && + \ SynAttrXMLish(get(a:syns, -2)) +endfu + +" Determine how many tsxRegions deep a synstack is. +fu! SynTSXDepth(syns) + return len(filter(copy(a:syns), 'v:val ==# "tsxRegion"')) +endfu + +" Check whether `cursyn' continues the same tsxRegion as `prevsyn'. +fu! SynTSXContinues(cursyn, prevsyn) + let curdepth = SynTSXDepth(a:cursyn) + let prevdepth = SynTSXDepth(a:prevsyn) + + " In most places, we expect the nesting depths to be the same between any + " two consecutive positions within a tsxRegion (e.g., between a parent and + " child node, between two TSX attributes, etc.). The exception is between + " sibling nodes, where after a completed element (with depth N), we return + " to the parent's nesting (depth N - 1). This case is easily detected, + " since it is the only time when the top syntax element in the synstack is + " tsxRegion---specifically, the tsxRegion corresponding to the parent. + return prevdepth == curdepth || + \ (prevdepth == curdepth + 1 && get(a:cursyn, -1) ==# 'tsxRegion') +endfu + +" Cleverly mix JS and XML indentation. +fu! GetTsxIndent() + let cursyn = SynSOL(v:lnum) + let prevsyn = SynEOL(v:lnum - 1) + + " Use XML indenting iff: + " - the syntax at the end of the previous line was either TSX or was the + " closing brace of a jsBlock whose parent syntax was TSX; and + " - the current line continues the same tsxRegion as the previous line. + if (SynXMLish(prevsyn) || SynTSXBlockEnd(prevsyn)) && + \ SynTSXContinues(cursyn, prevsyn) + let ind = XmlIndentGet(v:lnum, 0) + let l:line = getline(v:lnum) + let l:pline = getline(v:lnum - 1) + + " Align '/>' and '>' with '<' for multiline tags. + " Align end of expression ')' or '}'. + if l:line =~? s:endtag + let ind = ind - shiftwidth() + endif + + " Then correct the indentation of any TSX following '/>' or '>'. + " Align start of expression '(' or '{' + if l:pline =~? s:endtag || l:pline =~? s:startexp + let ind = ind + shiftwidth() + endif + else + if len(b:tsx_ts_indentexpr) + " Invoke the base TS package's custom indenter + let ind = eval(b:tsx_ts_indentexpr) + else + let ind = cindent(v:lnum) + endif + endif + + return ind +endfu |