1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193
|
vim9script
# Vim indent file
# Language: VisualBasic (ft=vb) / Basic (ft=basic) / SaxBasic (ft=vb)
# Author: Johannes Zellner <johannes@zellner.org>
# Maintainer: Michael Soyka (mssr953@gmail.com)
# Contributors: Doug Kearns (dougkearns@gmail.com)
# Last Change: Fri, 18 Jun 2004 07:22:42 CEST
# Small update 2010 Jul 28 by Maxim Kim
# 2022/12/15: add support for multiline statements.
# 2022/12/21: move VbGetIndent from global to script-local scope
# 2022/12/26: recognize "Type" keyword
# 2023/07/13: correct/extend line continuation pattern (Doug Kearns)
# 2023/07/14: add more keywords; various optimizations (Doug Kearns)
# 2023/07/20: convert to Vim9 script
# 2023/07/23: improve detection of preproc directives (Doug Kearns)
if exists("b:did_indent")
finish
endif
b:did_indent = v:true
setlocal autoindent
setlocal indentexpr=VbGetIndent()
setlocal indentkeys&
setlocal indentkeys+==~else,=~elseif,=~end,=~wend,=~case,=~next,=~select,=~loop
b:undo_indent = "setlocal autoindent< indentexpr< indentkeys<"
# Only define the function once.
if exists("*VbGetIndent")
finish
endif
# These regular expressions identify statement labels and preprocessor
# directives.
#
const RE_LABEL: string = '^\s*\k\+:\s*$'
const RE_PREPROC: string =
'^\s*#\%(const\|if\|elseif\|else\|end\|region\|enable\|disable\)\>'
# Microsoft documentation states that line continuation is indicated by a
# two-character sequence at end-of-line: a space character followed by an
# underscore. Nonetheless, it has been reported that additional
# whitespace after the underscore is also allowed. We will support both.
# However, VB 16.0 also permits a comment after the underscore which,
# for simplicity, we do not support.
#
const RE_LINE_CONTINUATION: string = '\s_\s*$'
# The following regular expressions are used to increase the indent
# after statements that open a new scope.
#
const RE_INCR_INDENT_1: string =
'^\s*\%(begin\|select\|case\|default\|if\|else\|elseif\|do\|for\|while\|with\)\>'
const RE_INCR_INDENT_2: string =
'^\s*\%(\%(private\|public\|friend\)\s\+\)\=\%(static\s\+\)\=\%(function\|sub\|property\)\>'
const RE_INCR_INDENT_3: string =
'^\s*\%(\%(private\|public\)\s\+\)\=\%(enum\|type\)\>'
def VbGetIndent(): number
var this_lnum: number = v:lnum
var this_line: string = getline(this_lnum)
var this_indent: number = 0
# labels and preprocessor statements get zero indent immediately
if (this_line =~? RE_LABEL) || (this_line =~? RE_PREPROC)
return this_indent
endif
# Get the current value of 'shiftwidth'
const SHIFTWIDTH: number = shiftwidth()
# Find a non-blank line above the current line.
# Skip over labels and preprocessor directives.
var lnum: number = this_lnum
var previous_line: string
while lnum > 0
lnum = prevnonblank(lnum - 1)
previous_line = getline(lnum)
if (previous_line !~? RE_LABEL) || (previous_line !~? RE_PREPROC)
break
endif
endwhile
# Hit the start of the file, use zero indent.
if lnum == 0
return this_indent
endif
# Variable "previous_line" now contains the text in buffer line "lnum".
# Multi-line statements have the underscore character at end-of-line:
#
# object.method(arguments, _
# arguments, _
# arguments)
#
# and require extra logic to determine the correct indentation.
#
# Case 1: Line "lnum" is the first line of a multiline statement.
# Line "lnum" will have a trailing underscore character
# but the preceding non-blank line does not.
# Line "this_lnum" will be indented relative to "lnum".
#
# Case 2: Line "lnum" is the last line of a multiline statement.
# Line "lnum" will not have a trailing underscore character
# but the preceding non-blank line will.
# Line "this_lnum" will have the same indentation as the starting
# line of the multiline statement.
#
# Case 3: Line "lnum" is neither the first nor last line.
# Lines "lnum" and "lnum-1" will have a trailing underscore
# character.
# Line "this_lnum" will have the same indentation as the preceding
# line.
#
# No matter which case it is, the starting line of the statement must be
# found. It will be assumed that multiline statements cannot have
# intermingled comments, statement labels, preprocessor directives or
# blank lines.
#
var lnum_is_continued: bool = (previous_line =~? RE_LINE_CONTINUATION)
var before_lnum: number
var before_previous_line: string
if lnum > 1
before_lnum = prevnonblank(lnum - 1)
before_previous_line = getline(before_lnum)
else
before_lnum = 0
before_previous_line = ""
endif
if before_previous_line !~? RE_LINE_CONTINUATION
# Variable "previous_line" contains the start of a statement.
#
this_indent = indent(lnum)
if lnum_is_continued
this_indent += SHIFTWIDTH
endif
elseif ! lnum_is_continued
# Line "lnum" contains the last line of a multiline statement.
# Need to find where this multiline statement begins
#
while before_lnum > 0
before_lnum -= 1
if getline(before_lnum) !~? RE_LINE_CONTINUATION
before_lnum += 1
break
endif
endwhile
if before_lnum == 0
before_lnum = 1
endif
previous_line = getline(before_lnum)
this_indent = indent(before_lnum)
else
# Line "lnum" is not the first or last line of a multiline statement.
#
this_indent = indent(lnum)
endif
# Increment indent
if (previous_line =~? RE_INCR_INDENT_1) ||
(previous_line =~? RE_INCR_INDENT_2) ||
(previous_line =~? RE_INCR_INDENT_3)
this_indent += SHIFTWIDTH
endif
# Decrement indent
if this_line =~? '^\s*end\s\+select\>'
if previous_line !~? '^\s*select\>'
this_indent -= 2 * SHIFTWIDTH
else
# this case is for an empty 'select' -- 'end select'
# (w/o any case statements) like:
#
# select case readwrite
# end select
this_indent -= SHIFTWIDTH
endif
elseif this_line =~? '^\s*\%(end\|else\|elseif\|until\|loop\|next\|wend\)\>'
this_indent -= SHIFTWIDTH
elseif this_line =~? '^\s*\%(case\|default\)\>'
if previous_line !~? '^\s*select\>'
this_indent -= SHIFTWIDTH
endif
endif
return this_indent
enddef
# vim:sw=4
|