1
0
Fork 0
mirror of https://github.com/vim/vim synced 2025-03-17 23:37:08 +01:00
vim/runtime/indent/r.vim

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

524 lines
14 KiB
VimL
Raw Normal View History

2011-09-14 17:55:08 +02:00
" Vim indent file
" Language: R
" Maintainer: This runtime file is looking for a new maintainer.
" Former Maintainer: Jakson Alves de Aquino <jalvesaq@gmail.com>
" Former Repository: https://github.com/jalvesaq/R-Vim-runtime
" Last Change: 2023 Oct 08 10:45AM
" 2024 Feb 19 by Vim Project (announce adoption)
2011-09-14 17:55:08 +02:00
" Only load this indent file when no other was loaded.
2013-02-20 21:26:00 +01:00
if exists("b:did_indent")
2015-06-19 13:27:23 +02:00
finish
2011-09-14 17:55:08 +02:00
endif
2013-02-20 21:26:00 +01:00
let b:did_indent = 1
2011-09-14 17:55:08 +02:00
setlocal indentkeys=0{,0},:,!^F,o,O,e
setlocal indentexpr=GetRIndent()
setlocal autoindent
2011-09-14 17:55:08 +02:00
2023-02-27 15:49:53 +00:00
let b:undo_indent = "setl inde< indk<"
2011-09-14 17:55:08 +02:00
" Only define the function once.
if exists("*GetRIndent")
2015-06-19 13:27:23 +02:00
finish
2011-09-14 17:55:08 +02:00
endif
2018-08-28 22:58:02 +02:00
let s:cpo_save = &cpo
set cpo&vim
2011-09-14 17:55:08 +02:00
" Options to make the indentation more similar to Emacs/ESS:
2018-08-28 22:58:02 +02:00
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',
2023-02-27 15:49:53 +00:00
\ '\(&\||\|+\|-\|\*\|/\|=\|\~\|%\|->\||>\)\s*$')
2011-09-14 17:55:08 +02:00
function s:RDelete_quotes(line)
2015-06-19 13:27:23 +02:00
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
elseif 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
2015-06-19 13:27:23 +02:00
if a:line[i] == "'"
let i += 1
endif
elseif 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
2015-06-19 13:27:23 +02:00
endif
endif
if i == llen
break
endif
let line1 = line1 . a:line[i]
let j += 1
let i += 1
endwhile
return line1
2011-09-14 17:55:08 +02:00
endfunction
" Convert foo(bar()) int foo()
function s:RDelete_parens(line)
2015-06-19 13:27:23 +02:00
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
2011-09-14 17:55:08 +02:00
let i += 1
2015-06-19 13:27:23 +02:00
if a:line[i] == ')'
let nop -= 1
elseif a:line[i] == '('
let nop += 1
2015-06-19 13:27:23 +02:00
endif
endwhile
let line1 = line1 . a:line[i]
endif
let i += 1
endwhile
return line1
2011-09-14 17:55:08 +02:00
endfunction
2023-04-22 22:40:14 +01:00
function s:Get_paren_balance(line, o, c)
2015-06-19 13:27:23 +02:00
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
2011-09-14 17:55:08 +02:00
endfunction
2023-04-22 22:40:14 +01:00
function s:Get_matching_brace(linenr, o, c, delbrace)
2015-06-19 13:27:23 +02:00
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
2011-09-14 17:55:08 +02:00
endfunction
" This function is buggy because there 'if's without 'else'
" It must be rewritten relying more on indentation
2023-04-22 22:40:14 +01:00
function s:Get_matching_if(linenr, delif)
2015-06-19 13:27:23 +02:00
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
2011-09-14 17:55:08 +02:00
endif
2015-06-19 13:27:23 +02:00
let i -= 1
let line = SanitizeRLine(getline(i))
endwhile
if ifhere
return ifhere
else
return a:linenr
endif
2011-09-14 17:55:08 +02:00
endfunction
2023-04-22 22:40:14 +01:00
function s:Get_last_paren_idx(line, o, c, pb)
2015-06-19 13:27:23 +02:00
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
elseif line[idx] == a:c
let blc += 1
2015-06-19 13:27:23 +02:00
endif
let idx += 1
endwhile
return theidx + 1
2011-09-14 17:55:08 +02:00
endfunction
" Get previous relevant line. Search back until getting a line that isn't
" comment or blank
function s:Get_prev_line(lineno)
2015-06-19 13:27:23 +02:00
let lnum = a:lineno - 1
let data = getline( lnum )
while lnum > 0 && (data =~ '^\s*#' || data =~ '^\s*$')
let lnum = lnum - 1
2011-09-14 17:55:08 +02:00
let data = getline( lnum )
2015-06-19 13:27:23 +02:00
endwhile
return lnum
2011-09-14 17:55:08 +02:00
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)
2015-06-19 13:27:23 +02:00
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
2011-09-14 17:55:08 +02:00
endfunction
function GetRIndent()
2015-06-19 13:27:23 +02:00
let clnum = line(".") " current line
2011-09-14 17:55:08 +02:00
2015-06-19 13:27:23 +02:00
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
2011-09-14 17:55:08 +02:00
endif
2015-06-19 13:27:23 +02:00
endif
let cline = SanitizeRLine(cline)
2018-08-28 22:58:02 +02:00
if cline =~ '^\s*}'
2015-06-19 13:27:23 +02:00
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
2011-09-14 17:55:08 +02:00
endif
2015-06-19 13:27:23 +02:00
endif
2011-09-14 17:55:08 +02:00
2018-08-28 22:58:02 +02:00
if cline =~ '^\s*)$'
let indline = s:Get_matching_brace(clnum, '(', ')', 1)
return indent(indline)
endif
2015-06-19 13:27:23 +02:00
" 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
2011-09-14 17:55:08 +02:00
2015-06-19 13:27:23 +02:00
let line = SanitizeRLine(getline(lnum))
2011-09-14 17:55:08 +02:00
2015-06-19 13:27:23 +02:00
if &filetype == "rhelp"
if cline =~ '^\\dontshow{' || cline =~ '^\\dontrun{' || cline =~ '^\\donttest{' || cline =~ '^\\testonly{'
return 0
2011-09-14 17:55:08 +02:00
endif
2015-06-19 13:27:23 +02:00
if line =~ '^\\examples{' || line =~ '^\\usage{' || line =~ '^\\dontshow{' || line =~ '^\\dontrun{' || line =~ '^\\donttest{' || line =~ '^\\testonly{'
return 0
2011-09-14 17:55:08 +02:00
endif
2015-06-19 13:27:23 +02:00
endif
if &filetype == "rnoweb" && line =~ "^<<.*>>="
return 0
endif
2016-03-12 12:57:59 +01:00
if cline =~ '^\s*{' && s:Get_paren_balance(cline, '{', '}') > 0
2015-06-19 13:27:23 +02:00
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
2017-04-09 20:11:58 +02:00
if nline =~ '^\s*function\s*(' && indent(nlnum) == shiftwidth()
2015-06-19 13:27:23 +02:00
return 0
endif
2011-09-14 17:55:08 +02:00
endif
2015-06-19 13:27:23 +02:00
if s:Get_paren_balance(line, "(", ")") == 0
return indent(lnum)
2011-09-14 17:55:08 +02:00
endif
2015-06-19 13:27:23 +02:00
endif
2011-09-14 17:55:08 +02:00
2015-06-19 13:27:23 +02:00
" line is an incomplete command:
2016-03-12 12:57:59 +01:00
if line =~ '\<\(if\|while\|for\|function\)\s*()$' || line =~ '\<else$' || line =~ '<-$' || line =~ '->$'
2017-04-09 20:11:58 +02:00
return indent(lnum) + shiftwidth()
2015-06-19 13:27:23 +02:00
endif
2011-09-14 17:55:08 +02:00
2015-06-19 13:27:23 +02:00
" Deal with () and []
2011-09-14 17:55:08 +02:00
2015-06-19 13:27:23 +02:00
let pb = s:Get_paren_balance(line, '(', ')')
2011-09-14 17:55:08 +02:00
2015-06-19 13:27:23 +02:00
if line =~ '^\s*{$' || line =~ '(\s*{' || (pb == 0 && (line =~ '{$' || line =~ '(\s*{$'))
2017-04-09 20:11:58 +02:00
return indent(lnum) + shiftwidth()
2015-06-19 13:27:23 +02:00
endif
2011-09-14 17:55:08 +02:00
2015-06-19 13:27:23 +02:00
let s:curtabstop = repeat(' ', &tabstop)
2011-09-14 17:55:08 +02:00
2015-06-19 13:27:23 +02:00
if g:r_indent_align_args == 1
if pb > 0 && line =~ '{$'
2017-04-09 20:11:58 +02:00
return s:Get_last_paren_idx(line, '(', ')', pb) + shiftwidth()
2011-09-14 17:55:08 +02:00
endif
2015-06-19 13:27:23 +02:00
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
2011-09-14 17:55:08 +02:00
endif
2015-06-19 13:27:23 +02:00
if pb < 0 && line =~ '.*[,&|\-\*+<>]$'
if line =~ '.*[\-\*+>]$'
let is_op = v:true
else
let is_op = v:false
endif
2015-06-19 13:27:23 +02:00
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
elseif line[ind] == '('
let pb += 1
if is_op && pb == 0
return indent(lnum)
2015-06-19 13:27:23 +02:00
endif
endif
if pb == 1
return ind + 1
endif
let ind -= 1
2011-09-14 17:55:08 +02:00
endwhile
2015-06-19 13:27:23 +02:00
let lnum -= 1
endwhile
return 0
endif
2011-09-14 17:55:08 +02:00
2015-06-19 13:27:23 +02:00
if bb > 0
let ind = s:Get_last_paren_idx(getline(lnum), '[', ']', bb)
return ind
2011-09-14 17:55:08 +02:00
endif
2015-06-19 13:27:23 +02:00
endif
2011-09-14 17:55:08 +02:00
2015-06-19 13:27:23 +02:00
let post_block = 0
2016-03-12 12:57:59 +01:00
if line =~ '}$' && s:Get_paren_balance(line, '{', '}') < 0
2015-06-19 13:27:23 +02:00
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
2023-02-27 15:49:53 +00:00
if substitute(line, '#.*', '', '') =~ g:r_indent_op_pattern && s:Get_paren_balance(line, "(", ")") == 0
if substitute(oline, '#.*', '', '') =~ g:r_indent_op_pattern && s:Get_paren_balance(line, "(", ")") == 0
2015-06-19 13:27:23 +02:00
return indent(lnum)
else
2017-04-09 20:11:58 +02:00
return indent(lnum) + shiftwidth()
2015-06-19 13:27:23 +02:00
endif
elseif substitute(oline, '#.*', '', '') =~ g:r_indent_op_pattern && s:Get_paren_balance(line, "(", ")") == 0
return indent(lnum) - shiftwidth()
2015-06-19 13:27:23 +02:00
endif
2023-02-27 15:49:53 +00:00
elseif substitute(line, '#.*', '', '') =~ g:r_indent_op_pattern && s:Get_paren_balance(line, "(", ")") == 0
return indent(lnum) + shiftwidth()
2015-06-19 13:27:23 +02:00
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
2017-04-09 20:11:58 +02:00
return indent(lnum) + shiftwidth()
2011-09-14 17:55:08 +02:00
endif
2015-06-19 13:27:23 +02:00
" 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 =~ '<-$'
2017-04-09 20:11:58 +02:00
return indent(lnum) + shiftwidth()
2015-06-19 13:27:23 +02:00
endif
2011-09-14 17:55:08 +02:00
endif
2015-06-19 13:27:23 +02:00
endif
2011-09-14 17:55:08 +02:00
2015-06-19 13:27:23 +02:00
if cline =~ '^\s*else'
if line =~ '<-\s*if\s*()'
2017-04-09 20:11:58 +02:00
return indent(lnum) + shiftwidth()
elseif line =~ '\<if\s*()'
return indent(lnum)
2015-06-19 13:27:23 +02:00
else
return indent(lnum) - shiftwidth()
2015-06-19 13:27:23 +02:00
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)
2011-09-14 17:55:08 +02:00
let pline = SanitizeRLine(getline(plnum))
2015-06-19 13:27:23 +02:00
endif
endif
2011-09-14 17:55:08 +02:00
2015-06-19 13:27:23 +02:00
if pline =~ 'else$'
let ppost_else = 1
let plnum = s:Get_matching_if(plnum, 0)
let pline = SanitizeRLine(getline(plnum))
endif
2011-09-14 17:55:08 +02:00
2015-06-19 13:27:23 +02:00
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
2011-09-14 17:55:08 +02:00
2015-06-19 13:27:23 +02:00
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)
2011-09-14 17:55:08 +02:00
endif
2015-06-19 13:27:23 +02:00
endif
2011-09-14 17:55:08 +02:00
2015-06-19 13:27:23 +02:00
let ind = indent(lnum)
2011-09-14 17:55:08 +02:00
2015-06-19 13:27:23 +02:00
if g:r_indent_align_args == 0 && pb != 0
2017-04-09 20:11:58 +02:00
let ind += pb * shiftwidth()
2015-06-19 13:27:23 +02:00
return ind
endif
2011-09-14 17:55:08 +02:00
2015-06-19 13:27:23 +02:00
if g:r_indent_align_args == 0 && bb != 0
2017-04-09 20:11:58 +02:00
let ind += bb * shiftwidth()
2015-06-19 13:27:23 +02:00
return ind
endif
2011-10-20 22:22:38 +02:00
2016-03-12 12:57:59 +01:00
if plnum > 0
let pind = indent(plnum)
else
let pind = 0
endif
2017-04-09 20:11:58 +02:00
if ind == pind || (ind == (pind + shiftwidth()) && pline =~ '{$' && ppost_else == 0)
2015-06-19 13:27:23 +02:00
return ind
endif
let pline = getline(plnum)
let pbb = s:Get_paren_balance(pline, '[', ']')
2011-09-14 17:55:08 +02:00
2015-06-19 13:27:23 +02:00
while pind < ind && plnum > 0 && ppb == 0 && pbb == 0
let ind = pind
let plnum = s:Get_prev_line(plnum)
2011-10-20 22:22:38 +02:00
let pline = getline(plnum)
2015-06-19 13:27:23 +02:00
let ppb = s:Get_paren_balance(pline, '(', ')')
2011-10-20 22:22:38 +02:00
let pbb = s:Get_paren_balance(pline, '[', ']')
2015-06-19 13:27:23 +02:00
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, '[', ']')
2011-09-14 17:55:08 +02:00
endwhile
2015-06-19 13:27:23 +02:00
let pind = indent(plnum)
2017-04-09 20:11:58 +02:00
if ind == (pind + shiftwidth()) && pline =~ '{$'
2015-06-19 13:27:23 +02:00
return ind
endif
endwhile
2011-09-14 17:55:08 +02:00
2015-06-19 13:27:23 +02:00
return ind
2011-09-14 17:55:08 +02:00
endfunction
2018-08-28 22:58:02 +02:00
let &cpo = s:cpo_save
unlet s:cpo_save
2015-06-19 13:27:23 +02:00
" vim: sw=2