123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523 |
- " Vim indent file
- " Language: R
- " Author: Jakson Alves de Aquino <jalvesaq@gmail.com>
- " Homepage: https://github.com/jalvesaq/R-Vim-runtime
- " Last Change: Sun Aug 19, 2018 09:13PM
- " Only load this indent file when no other was loaded.
- if exists("b:did_indent")
- finish
- endif
- let b:did_indent = 1
- setlocal indentkeys=0{,0},:,!^F,o,O,e
- setlocal indentexpr=GetRIndent()
- " Only define the function once.
- if exists("*GetRIndent")
- finish
- endif
- let s:cpo_save = &cpo
- set cpo&vim
- " Options to make the indentation more similar to Emacs/ESS:
- let g:r_indent_align_args = get(g:, 'r_indent_align_args', 1)
- let g:r_indent_ess_comments = get(g:, 'r_indent_ess_comments', 0)
- let g:r_indent_comment_column = get(g:, 'r_indent_comment_column', 40)
- let g:r_indent_ess_compatible = get(g:, 'r_indent_ess_compatible', 0)
- let g:r_indent_op_pattern = get(g:, 'r_indent_op_pattern',
- \ '\(&\||\|+\|-\|\*\|/\|=\|\~\|%\|->\)\s*$')
- function s:RDelete_quotes(line)
- let i = 0
- let j = 0
- let line1 = ""
- let llen = strlen(a:line)
- while i < llen
- if a:line[i] == '"'
- let i += 1
- let line1 = line1 . 's'
- while !(a:line[i] == '"' && ((i > 1 && a:line[i-1] == '\' && a:line[i-2] == '\') || a:line[i-1] != '\')) && i < llen
- let i += 1
- endwhile
- if a:line[i] == '"'
- let i += 1
- endif
- else
- if a:line[i] == "'"
- let i += 1
- let line1 = line1 . 's'
- while !(a:line[i] == "'" && ((i > 1 && a:line[i-1] == '\' && a:line[i-2] == '\') || a:line[i-1] != '\')) && i < llen
- let i += 1
- endwhile
- if a:line[i] == "'"
- let i += 1
- endif
- else
- if a:line[i] == "`"
- let i += 1
- let line1 = line1 . 's'
- while a:line[i] != "`" && i < llen
- let i += 1
- endwhile
- if a:line[i] == "`"
- let i += 1
- endif
- endif
- endif
- endif
- if i == llen
- break
- endif
- let line1 = line1 . a:line[i]
- let j += 1
- let i += 1
- endwhile
- return line1
- endfunction
- " Convert foo(bar()) int foo()
- function s:RDelete_parens(line)
- if s:Get_paren_balance(a:line, "(", ")") != 0
- return a:line
- endif
- let i = 0
- let j = 0
- let line1 = ""
- let llen = strlen(a:line)
- while i < llen
- let line1 = line1 . a:line[i]
- if a:line[i] == '('
- let nop = 1
- while nop > 0 && i < llen
- let i += 1
- if a:line[i] == ')'
- let nop -= 1
- else
- if a:line[i] == '('
- let nop += 1
- endif
- endif
- endwhile
- let line1 = line1 . a:line[i]
- endif
- let i += 1
- endwhile
- return line1
- endfunction
- function! s:Get_paren_balance(line, o, c)
- let line2 = substitute(a:line, a:o, "", "g")
- let openp = strlen(a:line) - strlen(line2)
- let line3 = substitute(line2, a:c, "", "g")
- let closep = strlen(line2) - strlen(line3)
- return openp - closep
- endfunction
- function! s:Get_matching_brace(linenr, o, c, delbrace)
- let line = SanitizeRLine(getline(a:linenr))
- if a:delbrace == 1
- let line = substitute(line, '{$', "", "")
- endif
- let pb = s:Get_paren_balance(line, a:o, a:c)
- let i = a:linenr
- while pb != 0 && i > 1
- let i -= 1
- let pb += s:Get_paren_balance(SanitizeRLine(getline(i)), a:o, a:c)
- endwhile
- return i
- endfunction
- " This function is buggy because there 'if's without 'else'
- " It must be rewritten relying more on indentation
- function! s:Get_matching_if(linenr, delif)
- let line = SanitizeRLine(getline(a:linenr))
- if a:delif
- let line = substitute(line, "if", "", "g")
- endif
- let elsenr = 0
- let i = a:linenr
- let ifhere = 0
- while i > 0
- let line2 = substitute(line, '\<else\>', "xxx", "g")
- let elsenr += strlen(line) - strlen(line2)
- if line =~ '.*\s*if\s*()' || line =~ '.*\s*if\s*()'
- let elsenr -= 1
- if elsenr == 0
- let ifhere = i
- break
- endif
- endif
- let i -= 1
- let line = SanitizeRLine(getline(i))
- endwhile
- if ifhere
- return ifhere
- else
- return a:linenr
- endif
- endfunction
- function! s:Get_last_paren_idx(line, o, c, pb)
- let blc = a:pb
- let line = substitute(a:line, '\t', s:curtabstop, "g")
- let theidx = -1
- let llen = strlen(line)
- let idx = 0
- while idx < llen
- if line[idx] == a:o
- let blc -= 1
- if blc == 0
- let theidx = idx
- endif
- else
- if line[idx] == a:c
- let blc += 1
- endif
- endif
- let idx += 1
- endwhile
- return theidx + 1
- endfunction
- " Get previous relevant line. Search back until getting a line that isn't
- " comment or blank
- function s:Get_prev_line(lineno)
- let lnum = a:lineno - 1
- let data = getline( lnum )
- while lnum > 0 && (data =~ '^\s*#' || data =~ '^\s*$')
- let lnum = lnum - 1
- let data = getline( lnum )
- endwhile
- return lnum
- endfunction
- " This function is also used by r-plugin/common_global.vim
- " Delete from '#' to the end of the line, unless the '#' is inside a string.
- function SanitizeRLine(line)
- let newline = s:RDelete_quotes(a:line)
- let newline = s:RDelete_parens(newline)
- let newline = substitute(newline, '#.*', "", "")
- let newline = substitute(newline, '\s*$', "", "")
- if &filetype == "rhelp" && newline =~ '^\\method{.*}{.*}(.*'
- let newline = substitute(newline, '^\\method{\(.*\)}{.*}', '\1', "")
- endif
- return newline
- endfunction
- function GetRIndent()
- let clnum = line(".") " current line
- let cline = getline(clnum)
- if cline =~ '^\s*#'
- if g:r_indent_ess_comments == 1
- if cline =~ '^\s*###'
- return 0
- endif
- if cline !~ '^\s*##'
- return g:r_indent_comment_column
- endif
- endif
- endif
- let cline = SanitizeRLine(cline)
- if cline =~ '^\s*}'
- let indline = s:Get_matching_brace(clnum, '{', '}', 1)
- if indline > 0 && indline != clnum
- let iline = SanitizeRLine(getline(indline))
- if s:Get_paren_balance(iline, "(", ")") == 0 || iline =~ '(\s*{$'
- return indent(indline)
- else
- let indline = s:Get_matching_brace(indline, '(', ')', 1)
- return indent(indline)
- endif
- endif
- endif
- if cline =~ '^\s*)$'
- let indline = s:Get_matching_brace(clnum, '(', ')', 1)
- return indent(indline)
- endif
- " Find the first non blank line above the current line
- let lnum = s:Get_prev_line(clnum)
- " Hit the start of the file, use zero indent.
- if lnum == 0
- return 0
- endif
- let line = SanitizeRLine(getline(lnum))
- if &filetype == "rhelp"
- if cline =~ '^\\dontshow{' || cline =~ '^\\dontrun{' || cline =~ '^\\donttest{' || cline =~ '^\\testonly{'
- return 0
- endif
- if line =~ '^\\examples{' || line =~ '^\\usage{' || line =~ '^\\dontshow{' || line =~ '^\\dontrun{' || line =~ '^\\donttest{' || line =~ '^\\testonly{'
- return 0
- endif
- endif
- if &filetype == "rnoweb" && line =~ "^<<.*>>="
- return 0
- endif
- if cline =~ '^\s*{' && s:Get_paren_balance(cline, '{', '}') > 0
- if g:r_indent_ess_compatible && line =~ ')$'
- let nlnum = lnum
- let nline = line
- while s:Get_paren_balance(nline, '(', ')') < 0
- let nlnum = s:Get_prev_line(nlnum)
- let nline = SanitizeRLine(getline(nlnum)) . nline
- endwhile
- if nline =~ '^\s*function\s*(' && indent(nlnum) == shiftwidth()
- return 0
- endif
- endif
- if s:Get_paren_balance(line, "(", ")") == 0
- return indent(lnum)
- endif
- endif
- " line is an incomplete command:
- if line =~ '\<\(if\|while\|for\|function\)\s*()$' || line =~ '\<else$' || line =~ '<-$' || line =~ '->$'
- return indent(lnum) + shiftwidth()
- endif
- " Deal with () and []
- let pb = s:Get_paren_balance(line, '(', ')')
- if line =~ '^\s*{$' || line =~ '(\s*{' || (pb == 0 && (line =~ '{$' || line =~ '(\s*{$'))
- return indent(lnum) + shiftwidth()
- endif
- let s:curtabstop = repeat(' ', &tabstop)
- if g:r_indent_align_args == 1
- if pb > 0 && line =~ '{$'
- return s:Get_last_paren_idx(line, '(', ')', pb) + shiftwidth()
- endif
- let bb = s:Get_paren_balance(line, '[', ']')
- if pb > 0
- if &filetype == "rhelp"
- let ind = s:Get_last_paren_idx(line, '(', ')', pb)
- else
- let ind = s:Get_last_paren_idx(getline(lnum), '(', ')', pb)
- endif
- return ind
- endif
- if pb < 0 && line =~ '.*[,&|\-\*+<>]$'
- let lnum = s:Get_prev_line(lnum)
- while pb < 1 && lnum > 0
- let line = SanitizeRLine(getline(lnum))
- let line = substitute(line, '\t', s:curtabstop, "g")
- let ind = strlen(line)
- while ind > 0
- if line[ind] == ')'
- let pb -= 1
- else
- if line[ind] == '('
- let pb += 1
- endif
- endif
- if pb == 1
- return ind + 1
- endif
- let ind -= 1
- endwhile
- let lnum -= 1
- endwhile
- return 0
- endif
- if bb > 0
- let ind = s:Get_last_paren_idx(getline(lnum), '[', ']', bb)
- return ind
- endif
- endif
- let post_block = 0
- if line =~ '}$' && s:Get_paren_balance(line, '{', '}') < 0
- let lnum = s:Get_matching_brace(lnum, '{', '}', 0)
- let line = SanitizeRLine(getline(lnum))
- if lnum > 0 && line =~ '^\s*{'
- let lnum = s:Get_prev_line(lnum)
- let line = SanitizeRLine(getline(lnum))
- endif
- let pb = s:Get_paren_balance(line, '(', ')')
- let post_block = 1
- endif
- " Indent after operator pattern
- let olnum = s:Get_prev_line(lnum)
- let oline = getline(olnum)
- if olnum > 0
- if line =~ g:r_indent_op_pattern && s:Get_paren_balance(line, "(", ")") == 0
- if oline =~ g:r_indent_op_pattern && s:Get_paren_balance(line, "(", ")") == 0
- return indent(lnum)
- else
- return indent(lnum) + shiftwidth()
- endif
- else
- if oline =~ g:r_indent_op_pattern && s:Get_paren_balance(line, "(", ")") == 0
- return indent(lnum) - shiftwidth()
- endif
- endif
- endif
- let post_fun = 0
- if pb < 0 && line !~ ')\s*[,&|\-\*+<>]$'
- let post_fun = 1
- while pb < 0 && lnum > 0
- let lnum -= 1
- let linepiece = SanitizeRLine(getline(lnum))
- let pb += s:Get_paren_balance(linepiece, "(", ")")
- let line = linepiece . line
- endwhile
- if line =~ '{$' && post_block == 0
- return indent(lnum) + shiftwidth()
- endif
- " Now we can do some tests again
- if cline =~ '^\s*{'
- return indent(lnum)
- endif
- if post_block == 0
- let newl = SanitizeRLine(line)
- if newl =~ '\<\(if\|while\|for\|function\)\s*()$' || newl =~ '\<else$' || newl =~ '<-$'
- return indent(lnum) + shiftwidth()
- endif
- endif
- endif
- if cline =~ '^\s*else'
- if line =~ '<-\s*if\s*()'
- return indent(lnum) + shiftwidth()
- else
- if line =~ '\<if\s*()'
- return indent(lnum)
- else
- return indent(lnum) - shiftwidth()
- endif
- endif
- endif
- let bb = s:Get_paren_balance(line, '[', ']')
- if bb < 0 && line =~ '.*]'
- while bb < 0 && lnum > 0
- let lnum -= 1
- let linepiece = SanitizeRLine(getline(lnum))
- let bb += s:Get_paren_balance(linepiece, "[", "]")
- let line = linepiece . line
- endwhile
- let line = s:RDelete_parens(line)
- endif
- let plnum = s:Get_prev_line(lnum)
- let ppost_else = 0
- if plnum > 0
- let pline = SanitizeRLine(getline(plnum))
- let ppost_block = 0
- if pline =~ '}$'
- let ppost_block = 1
- let plnum = s:Get_matching_brace(plnum, '{', '}', 0)
- let pline = SanitizeRLine(getline(plnum))
- if pline =~ '^\s*{$' && plnum > 0
- let plnum = s:Get_prev_line(plnum)
- let pline = SanitizeRLine(getline(plnum))
- endif
- endif
- if pline =~ 'else$'
- let ppost_else = 1
- let plnum = s:Get_matching_if(plnum, 0)
- let pline = SanitizeRLine(getline(plnum))
- endif
- if pline =~ '^\s*else\s*if\s*('
- let pplnum = s:Get_prev_line(plnum)
- let ppline = SanitizeRLine(getline(pplnum))
- while ppline =~ '^\s*else\s*if\s*(' || ppline =~ '^\s*if\s*()\s*\S$'
- let plnum = pplnum
- let pline = ppline
- let pplnum = s:Get_prev_line(plnum)
- let ppline = SanitizeRLine(getline(pplnum))
- endwhile
- while ppline =~ '\<\(if\|while\|for\|function\)\s*()$' || ppline =~ '\<else$' || ppline =~ '<-$'
- let plnum = pplnum
- let pline = ppline
- let pplnum = s:Get_prev_line(plnum)
- let ppline = SanitizeRLine(getline(pplnum))
- endwhile
- endif
- let ppb = s:Get_paren_balance(pline, '(', ')')
- if ppb < 0 && (pline =~ ')\s*{$' || pline =~ ')$')
- while ppb < 0 && plnum > 0
- let plnum -= 1
- let linepiece = SanitizeRLine(getline(plnum))
- let ppb += s:Get_paren_balance(linepiece, "(", ")")
- let pline = linepiece . pline
- endwhile
- let pline = s:RDelete_parens(pline)
- endif
- endif
- let ind = indent(lnum)
- if g:r_indent_align_args == 0 && pb != 0
- let ind += pb * shiftwidth()
- return ind
- endif
- if g:r_indent_align_args == 0 && bb != 0
- let ind += bb * shiftwidth()
- return ind
- endif
- if plnum > 0
- let pind = indent(plnum)
- else
- let pind = 0
- endif
- if ind == pind || (ind == (pind + shiftwidth()) && pline =~ '{$' && ppost_else == 0)
- return ind
- endif
- let pline = getline(plnum)
- let pbb = s:Get_paren_balance(pline, '[', ']')
- while pind < ind && plnum > 0 && ppb == 0 && pbb == 0
- let ind = pind
- let plnum = s:Get_prev_line(plnum)
- let pline = getline(plnum)
- let ppb = s:Get_paren_balance(pline, '(', ')')
- let pbb = s:Get_paren_balance(pline, '[', ']')
- while pline =~ '^\s*else'
- let plnum = s:Get_matching_if(plnum, 1)
- let pline = getline(plnum)
- let ppb = s:Get_paren_balance(pline, '(', ')')
- let pbb = s:Get_paren_balance(pline, '[', ']')
- endwhile
- let pind = indent(plnum)
- if ind == (pind + shiftwidth()) && pline =~ '{$'
- return ind
- endif
- endwhile
- return ind
- endfunction
- let &cpo = s:cpo_save
- unlet s:cpo_save
- " vim: sw=2
|