blob: 173588393792560e2a5c1de6c9bde6bdaa187c29 [file] [log] [blame]
Bram Moolenaara2baa732022-02-04 16:09:54 +00001vim9script
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01002
Bram Moolenaara2baa732022-02-04 16:09:54 +00003# Vim functions for file type detection
4#
5# Maintainer: Bram Moolenaar <Bram@vim.org>
Bram Moolenaarf07d1a72023-06-09 21:01:47 +01006# Last Change: 2023 Jun 09
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01007
Bram Moolenaara2baa732022-02-04 16:09:54 +00008# These functions are moved here from runtime/filetype.vim to make startup
9# faster.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010010
Bram Moolenaara2baa732022-02-04 16:09:54 +000011export def Check_inp()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010012 if getline(1) =~ '^\*'
13 setf abaqus
14 else
Bram Moolenaara2baa732022-02-04 16:09:54 +000015 var n = 1
16 var nmax = line("$") > 500 ? 500 : line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010017 while n <= nmax
18 if getline(n) =~? "^header surface data"
19 setf trasys
20 break
21 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +000022 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010023 endwhile
24 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +000025enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010026
Bram Moolenaara2baa732022-02-04 16:09:54 +000027# This function checks for the kind of assembly that is wanted by the user, or
28# can be detected from the first five lines of the file.
29export def FTasm()
30 # make sure b:asmsyntax exists
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010031 if !exists("b:asmsyntax")
Bram Moolenaara2baa732022-02-04 16:09:54 +000032 b:asmsyntax = ""
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010033 endif
34
35 if b:asmsyntax == ""
Bram Moolenaara2baa732022-02-04 16:09:54 +000036 FTasmsyntax()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010037 endif
38
Bram Moolenaara2baa732022-02-04 16:09:54 +000039 # if b:asmsyntax still isn't set, default to asmsyntax or GNU
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010040 if b:asmsyntax == ""
41 if exists("g:asmsyntax")
Bram Moolenaara2baa732022-02-04 16:09:54 +000042 b:asmsyntax = g:asmsyntax
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010043 else
Bram Moolenaara2baa732022-02-04 16:09:54 +000044 b:asmsyntax = "asm"
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010045 endif
46 endif
47
Bram Moolenaara2baa732022-02-04 16:09:54 +000048 exe "setf " .. fnameescape(b:asmsyntax)
49enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010050
Bram Moolenaara2baa732022-02-04 16:09:54 +000051export def FTasmsyntax()
52 # see if the file contains any asmsyntax=foo overrides. If so, change
53 # b:asmsyntax appropriately
54 var head = " " .. getline(1) .. " " .. getline(2) .. " "
55 .. getline(3) .. " " .. getline(4) .. " " .. getline(5) .. " "
56 var match = matchstr(head, '\sasmsyntax=\zs[a-zA-Z0-9]\+\ze\s')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010057 if match != ''
Bram Moolenaara2baa732022-02-04 16:09:54 +000058 b:asmsyntax = match
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010059 elseif ((head =~? '\.title') || (head =~? '\.ident') || (head =~? '\.macro') || (head =~? '\.subtitle') || (head =~? '\.library'))
Bram Moolenaara2baa732022-02-04 16:09:54 +000060 b:asmsyntax = "vmasm"
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010061 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +000062enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010063
Bram Moolenaara2baa732022-02-04 16:09:54 +000064var ft_visual_basic_content = '\cVB_Name\|Begin VB\.\(Form\|MDIForm\|UserControl\)'
Doug Kearnsc570e9c2022-01-31 17:09:14 +000065
Bram Moolenaara2baa732022-02-04 16:09:54 +000066# See FTfrm() for Visual Basic form file detection
67export def FTbas()
Bram Moolenaar6517f142022-01-21 14:55:13 +000068 if exists("g:filetype_bas")
Bram Moolenaara2baa732022-02-04 16:09:54 +000069 exe "setf " .. g:filetype_bas
Bram Moolenaar6517f142022-01-21 14:55:13 +000070 return
71 endif
72
Bram Moolenaara2baa732022-02-04 16:09:54 +000073 # most frequent FreeBASIC-specific keywords in distro files
74 var fb_keywords = '\c^\s*\%(extern\|var\|enum\|private\|scope\|union\|byref\|operator\|constructor\|delete\|namespace\|public\|property\|with\|destructor\|using\)\>\%(\s*[:=(]\)\@!'
Bram Moolenaar8b5901e2022-06-29 14:39:12 +010075 var fb_preproc = '\c^\s*\%(' ..
76 # preprocessor
77 '#\s*\a\+\|' ..
78 # compiler option
79 'option\s\+\%(byval\|dynamic\|escape\|\%(no\)\=gosub\|nokeyword\|private\|static\)\>\|' ..
80 # metacommand
81 '\%(''\|rem\)\s*\$lang\>\|' ..
82 # default datatype
83 'def\%(byte\|longint\|short\|ubyte\|uint\|ulongint\|ushort\)\>' ..
84 '\)'
Bram Moolenaara2baa732022-02-04 16:09:54 +000085 var fb_comment = "^\\s*/'"
Bram Moolenaar8b5901e2022-06-29 14:39:12 +010086
Bram Moolenaara2baa732022-02-04 16:09:54 +000087 # OPTION EXPLICIT, without the leading underscore, is common to many dialects
88 var qb64_preproc = '\c^\s*\%($\a\+\|option\s\+\%(_explicit\|_\=explicitarray\)\>\)'
Bram Moolenaar6517f142022-01-21 14:55:13 +000089
Bram Moolenaar8b5901e2022-06-29 14:39:12 +010090 for lnum in range(1, min([line("$"), 100]))
91 var line = getline(lnum)
92 if line =~ ft_visual_basic_content
93 setf vb
94 return
95 elseif line =~ fb_preproc || line =~ fb_comment || line =~ fb_keywords
96 setf freebasic
97 return
98 elseif line =~ qb64_preproc
99 setf qb64
100 return
101 endif
102 endfor
103 setf basic
Bram Moolenaara2baa732022-02-04 16:09:54 +0000104enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100105
Bram Moolenaara2baa732022-02-04 16:09:54 +0000106export def FTbtm()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100107 if exists("g:dosbatch_syntax_for_btm") && g:dosbatch_syntax_for_btm
108 setf dosbatch
109 else
110 setf btm
111 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000112enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100113
Bram Moolenaara2baa732022-02-04 16:09:54 +0000114export def BindzoneCheck(default = '')
115 if getline(1) .. getline(2) .. getline(3) .. getline(4)
116 =~ '^; <<>> DiG [0-9.]\+.* <<>>\|$ORIGIN\|$TTL\|IN\s\+SOA'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100117 setf bindzone
Bram Moolenaara2baa732022-02-04 16:09:54 +0000118 elseif default != ''
119 exe 'setf ' .. default
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100120 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000121enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100122
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100123# Returns true if file content looks like RAPID
124def IsRapid(sChkExt: string = ""): bool
125 if sChkExt == "cfg"
126 return getline(1) =~? '\v^%(EIO|MMC|MOC|PROC|SIO|SYS):CFG'
127 endif
128 # called from FTmod, FTprg or FTsys
129 return getline(nextnonblank(1)) =~? '\v^\s*%(\%{3}|module\s+\k+\s*%(\(|$))'
130enddef
131
132export def FTcfg()
133 if exists("g:filetype_cfg")
134 exe "setf " .. g:filetype_cfg
135 elseif IsRapid("cfg")
136 setf rapid
137 else
138 setf cfg
139 endif
140enddef
141
Bram Moolenaar8b5901e2022-06-29 14:39:12 +0100142export def FTcls()
143 if exists("g:filetype_cls")
144 exe "setf " .. g:filetype_cls
145 return
146 endif
147
Martin Tournoijbd053f82022-10-16 12:49:12 +0100148 if getline(1) =~ '^\v%(\%|\\)'
Bram Moolenaar8b5901e2022-06-29 14:39:12 +0100149 setf tex
150 elseif getline(1)[0] == '#' && getline(1) =~ 'rexx'
151 setf rexx
152 elseif getline(1) == 'VERSION 1.0 CLASS'
153 setf vb
154 else
155 setf st
156 endif
157enddef
158
Bram Moolenaara2baa732022-02-04 16:09:54 +0000159export def FTlpc()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100160 if exists("g:lpc_syntax_for_c")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000161 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100162 while lnum <= 12
163 if getline(lnum) =~# '^\(//\|inherit\|private\|protected\|nosave\|string\|object\|mapping\|mixed\)'
164 setf lpc
165 return
166 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000167 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100168 endwhile
169 endif
170 setf c
Bram Moolenaara2baa732022-02-04 16:09:54 +0000171enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100172
Bram Moolenaara2baa732022-02-04 16:09:54 +0000173export def FTheader()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100174 if match(getline(1, min([line("$"), 200])), '^@\(interface\|end\|class\)') > -1
175 if exists("g:c_syntax_for_h")
176 setf objc
177 else
178 setf objcpp
179 endif
180 elseif exists("g:c_syntax_for_h")
181 setf c
182 elseif exists("g:ch_syntax_for_h")
183 setf ch
184 else
185 setf cpp
186 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000187enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100188
Bram Moolenaara2baa732022-02-04 16:09:54 +0000189# This function checks if one of the first ten lines start with a '@'. In
190# that case it is probably a change file.
191# If the first line starts with # or ! it's probably a ch file.
192# If a line has "main", "include", "//" or "/*" it's probably ch.
193# Otherwise CHILL is assumed.
194export def FTchange()
195 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100196 while lnum <= 10
197 if getline(lnum)[0] == '@'
198 setf change
199 return
200 endif
201 if lnum == 1 && (getline(1)[0] == '#' || getline(1)[0] == '!')
202 setf ch
203 return
204 endif
205 if getline(lnum) =~ "MODULE"
206 setf chill
207 return
208 endif
209 if getline(lnum) =~ 'main\s*(\|#\s*include\|//'
210 setf ch
211 return
212 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000213 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100214 endwhile
215 setf chill
Bram Moolenaara2baa732022-02-04 16:09:54 +0000216enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100217
Bram Moolenaara2baa732022-02-04 16:09:54 +0000218export def FTent()
219 # This function checks for valid cl syntax in the first five lines.
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100220 # Look for either an opening comment, '#', or a block start, '{'.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000221 # If not found, assume SGML.
222 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100223 while lnum < 6
Bram Moolenaara2baa732022-02-04 16:09:54 +0000224 var line = getline(lnum)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100225 if line =~ '^\s*[#{]'
226 setf cl
227 return
228 elseif line !~ '^\s*$'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000229 # Not a blank line, not a comment, and not a block start,
230 # so doesn't look like valid cl code.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100231 break
232 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000233 lnum += 1
Bram Moolenaarc12dc472022-03-05 13:45:56 +0000234 endwhile
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100235 setf dtd
Bram Moolenaara2baa732022-02-04 16:09:54 +0000236enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100237
Bram Moolenaara2baa732022-02-04 16:09:54 +0000238export def ExCheck()
239 var lines = getline(1, min([line("$"), 100]))
Austin Gatlinf3caeb62021-06-26 12:02:55 +0200240 if exists('g:filetype_euphoria')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000241 exe 'setf ' .. g:filetype_euphoria
Austin Gatlinf3caeb62021-06-26 12:02:55 +0200242 elseif match(lines, '^--\|^ifdef\>\|^include\>') > -1
243 setf euphoria3
244 else
245 setf elixir
246 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000247enddef
Austin Gatlinf3caeb62021-06-26 12:02:55 +0200248
Bram Moolenaara2baa732022-02-04 16:09:54 +0000249export def EuphoriaCheck()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100250 if exists('g:filetype_euphoria')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000251 exe 'setf ' .. g:filetype_euphoria
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100252 else
253 setf euphoria3
254 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000255enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100256
Bram Moolenaara2baa732022-02-04 16:09:54 +0000257export def DtraceCheck()
=?UTF-8?q?Teubel=20Gy=C3=B6rgy?=4d56b972022-02-24 17:59:09 +0000258 if did_filetype()
259 # Filetype was already detected
260 return
261 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000262 var lines = getline(1, min([line("$"), 100]))
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100263 if match(lines, '^module\>\|^import\>') > -1
Bram Moolenaara2baa732022-02-04 16:09:54 +0000264 # D files often start with a module and/or import statement.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100265 setf d
266 elseif match(lines, '^#!\S\+dtrace\|#pragma\s\+D\s\+option\|:\S\{-}:\S\{-}:') > -1
267 setf dtrace
268 else
269 setf d
270 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000271enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100272
Bram Moolenaara2baa732022-02-04 16:09:54 +0000273export def FTe()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100274 if exists('g:filetype_euphoria')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000275 exe 'setf ' .. g:filetype_euphoria
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100276 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000277 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100278 while n < 100 && n <= line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100279 if getline(n) =~ "^\\s*\\(<'\\|'>\\)\\s*$"
280 setf specman
281 return
282 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000283 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100284 endwhile
285 setf eiffel
286 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000287enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100288
Bram Moolenaara2baa732022-02-04 16:09:54 +0000289export def FTfrm()
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000290 if exists("g:filetype_frm")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000291 exe "setf " .. g:filetype_frm
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000292 return
293 endif
294
Bram Moolenaara2baa732022-02-04 16:09:54 +0000295 var lines = getline(1, min([line("$"), 5]))
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000296
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000297 if match(lines, ft_visual_basic_content) > -1
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000298 setf vb
299 else
300 setf form
301 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000302enddef
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000303
Bram Moolenaara2baa732022-02-04 16:09:54 +0000304# Distinguish between Forth and F#.
305# Provided by Doug Kearns.
306export def FTfs()
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000307 if exists("g:filetype_fs")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000308 exe "setf " .. g:filetype_fs
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000309 else
Johan Kotlinski065088d2023-04-02 20:29:38 +0100310 var n = 1
311 while n < 100 && n <= line("$")
312 # Forth comments and colon definitions
313 if getline(n) =~ "^[:(\\\\] "
314 setf forth
315 return
316 endif
317 n += 1
318 endwhile
319 setf fsharp
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000320 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000321enddef
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000322
Bram Moolenaara2baa732022-02-04 16:09:54 +0000323# Distinguish between HTML, XHTML and Django
324export def FThtml()
325 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100326 while n < 10 && n <= line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100327 if getline(n) =~ '\<DTD\s\+XHTML\s'
328 setf xhtml
329 return
330 endif
331 if getline(n) =~ '{%\s*\(extends\|block\|load\)\>\|{#\s\+'
332 setf htmldjango
333 return
334 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000335 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100336 endwhile
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100337 setf FALLBACK html
Bram Moolenaara2baa732022-02-04 16:09:54 +0000338enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100339
Bram Moolenaara2baa732022-02-04 16:09:54 +0000340# Distinguish between standard IDL and MS-IDL
341export def FTidl()
342 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100343 while n < 50 && n <= line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100344 if getline(n) =~ '^\s*import\s\+"\(unknwn\|objidl\)\.idl"'
345 setf msidl
346 return
347 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000348 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100349 endwhile
350 setf idl
Bram Moolenaara2baa732022-02-04 16:09:54 +0000351enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100352
Bram Moolenaar48c3f4e2022-08-08 15:42:38 +0100353# Distinguish between "default", Prolog and Cproto prototype file.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000354export def ProtoCheck(default: string)
355 # Cproto files have a comment in the first line and a function prototype in
356 # the second line, it always ends in ";". Indent files may also have
357 # comments, thus we can't match comments to see the difference.
358 # IDL files can have a single ';' in the second line, require at least one
359 # chacter before the ';'.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100360 if getline(2) =~ '.;$'
361 setf cpp
362 else
Bram Moolenaar48c3f4e2022-08-08 15:42:38 +0100363 # recognize Prolog by specific text in the first non-empty line
364 # require a blank after the '%' because Perl uses "%list" and "%translate"
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100365 var lnum = getline(nextnonblank(1))
366 if lnum =~ '\<prolog\>' || lnum =~ '^\s*\(%\+\(\s\|$\)\|/\*\)' || lnum =~ ':-'
Bram Moolenaar48c3f4e2022-08-08 15:42:38 +0100367 setf prolog
368 else
369 exe 'setf ' .. default
370 endif
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100371 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000372enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100373
Bram Moolenaara2baa732022-02-04 16:09:54 +0000374export def FTm()
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200375 if exists("g:filetype_m")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000376 exe "setf " .. g:filetype_m
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200377 return
378 endif
379
Bram Moolenaara2baa732022-02-04 16:09:54 +0000380 # excluding end(for|function|if|switch|while) common to Murphi
381 var octave_block_terminators = '\<end\%(_try_catch\|classdef\|enumeration\|events\|methods\|parfor\|properties\)\>'
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200382
Bram Moolenaara2baa732022-02-04 16:09:54 +0000383 var objc_preprocessor = '^\s*#\s*\%(import\|include\|define\|if\|ifn\=def\|undef\|line\|error\|pragma\)\>'
Doug Kearns7329cfa2021-11-26 13:01:41 +0000384
Bram Moolenaara2baa732022-02-04 16:09:54 +0000385 var n = 1
386 var saw_comment = 0 # Whether we've seen a multiline comment leader.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100387 while n < 100
Bram Moolenaara2baa732022-02-04 16:09:54 +0000388 var line = getline(n)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100389 if line =~ '^\s*/\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000390 # /* ... */ is a comment in Objective C and Murphi, so we can't conclude
391 # it's either of them yet, but track this as a hint in case we don't see
392 # anything more definitive.
393 saw_comment = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100394 endif
Doug Kearns7329cfa2021-11-26 13:01:41 +0000395 if line =~ '^\s*//' || line =~ '^\s*@import\>' || line =~ objc_preprocessor
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100396 setf objc
397 return
398 endif
Bram Moolenaarca0627d2021-09-12 17:03:08 +0200399 if line =~ '^\s*\%(#\|%!\)' || line =~ '^\s*unwind_protect\>' ||
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200400 \ line =~ '\%(^\|;\)\s*' .. octave_block_terminators
401 setf octave
402 return
403 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000404 # TODO: could be Matlab or Octave
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100405 if line =~ '^\s*%'
406 setf matlab
407 return
408 endif
409 if line =~ '^\s*(\*'
410 setf mma
411 return
412 endif
413 if line =~ '^\c\s*\(\(type\|var\)\>\|--\)'
414 setf murphi
415 return
416 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000417 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100418 endwhile
419
420 if saw_comment
Bram Moolenaara2baa732022-02-04 16:09:54 +0000421 # We didn't see anything definitive, but this looks like either Objective C
422 # or Murphi based on the comment leader. Assume the former as it is more
423 # common.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100424 setf objc
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100425 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000426 # Default is Matlab
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100427 setf matlab
428 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000429enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100430
Bram Moolenaara2baa732022-02-04 16:09:54 +0000431export def FTmms()
432 var n = 1
Bram Moolenaard7df2792020-01-02 21:34:42 +0100433 while n < 20
Bram Moolenaara2baa732022-02-04 16:09:54 +0000434 var line = getline(n)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100435 if line =~ '^\s*\(%\|//\)' || line =~ '^\*'
436 setf mmix
437 return
438 endif
439 if line =~ '^\s*#'
440 setf make
441 return
442 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000443 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100444 endwhile
445 setf mmix
Bram Moolenaara2baa732022-02-04 16:09:54 +0000446enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100447
Bram Moolenaara2baa732022-02-04 16:09:54 +0000448# This function checks if one of the first five lines start with a dot. In
449# that case it is probably an nroff file: 'filetype' is set and 1 is returned.
450export def FTnroff(): number
451 if getline(1)[0] .. getline(2)[0] .. getline(3)[0]
452 .. getline(4)[0] .. getline(5)[0] =~ '\.'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100453 setf nroff
454 return 1
455 endif
456 return 0
Bram Moolenaara2baa732022-02-04 16:09:54 +0000457enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100458
Bram Moolenaara2baa732022-02-04 16:09:54 +0000459export def FTmm()
460 var n = 1
Bram Moolenaard1caa942020-04-10 22:10:56 +0200461 while n < 20
Bram Moolenaara2baa732022-02-04 16:09:54 +0000462 if getline(n) =~ '^\s*\(#\s*\(include\|import\)\>\|@import\>\|/\*\)'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100463 setf objcpp
464 return
465 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000466 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100467 endwhile
468 setf nroff
Bram Moolenaara2baa732022-02-04 16:09:54 +0000469enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100470
Bram Moolenaarcdbfc6d2022-06-30 16:25:21 +0100471# Returns true if file content looks like LambdaProlog module
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100472def IsLProlog(): bool
Gaetan Lepage4ce1bda2023-05-10 22:01:55 +0100473 # skip apparent comments and blank lines, what looks like
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100474 # LambdaProlog comment may be RAPID header
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100475 var lnum: number = nextnonblank(1)
476 while lnum > 0 && lnum < line('$') && getline(lnum) =~ '^\s*%' # LambdaProlog comment
477 lnum = nextnonblank(lnum + 1)
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100478 endwhile
479 # this pattern must not catch a go.mod file
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100480 return getline(lnum) =~ '\<module\s\+\w\+\s*\.\s*\(%\|$\)'
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100481enddef
482
483# Determine if *.mod is ABB RAPID, LambdaProlog, Modula-2, Modsim III or go.mod
484export def FTmod()
485 if exists("g:filetype_mod")
486 exe "setf " .. g:filetype_mod
Omar El Halabic9fbd252023-05-29 19:59:45 +0100487 elseif expand("<afile>") =~ '\<go.mod$'
488 setf gomod
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100489 elseif IsLProlog()
490 setf lprolog
491 elseif getline(nextnonblank(1)) =~ '\%(\<MODULE\s\+\w\+\s*;\|^\s*(\*\)'
492 setf modula2
493 elseif IsRapid()
494 setf rapid
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100495 else
496 # Nothing recognized, assume modsim3
497 setf modsim3
498 endif
499enddef
500
Bram Moolenaara2baa732022-02-04 16:09:54 +0000501export def FTpl()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100502 if exists("g:filetype_pl")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000503 exe "setf " .. g:filetype_pl
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100504 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000505 # recognize Prolog by specific text in the first non-empty line
506 # require a blank after the '%' because Perl uses "%list" and "%translate"
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100507 var line = getline(nextnonblank(1))
508 if line =~ '\<prolog\>' || line =~ '^\s*\(%\+\(\s\|$\)\|/\*\)' || line =~ ':-'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100509 setf prolog
510 else
511 setf perl
512 endif
513 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000514enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100515
Bram Moolenaara2baa732022-02-04 16:09:54 +0000516export def FTinc()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100517 if exists("g:filetype_inc")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000518 exe "setf " .. g:filetype_inc
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100519 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000520 var lines = getline(1) .. getline(2) .. getline(3)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100521 if lines =~? "perlscript"
522 setf aspperl
523 elseif lines =~ "<%"
524 setf aspvbs
525 elseif lines =~ "<?"
526 setf php
Bram Moolenaara2baa732022-02-04 16:09:54 +0000527 # Pascal supports // comments but they're vary rarely used for file
528 # headers so assume POV-Ray
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000529 elseif lines =~ '^\s*\%({\|(\*\)' || lines =~? ft_pascal_keywords
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100530 setf pascal
Gregory Anders30e212d2022-07-26 21:42:03 +0100531 elseif lines =~# '\<\%(require\|inherit\)\>' || lines =~# '[A-Z][A-Za-z0-9_:${}]*\s\+\%(??\|[?:+]\)\?= '
Gregory Andersfa49eb42022-07-16 17:46:47 +0100532 setf bitbake
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100533 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000534 FTasmsyntax()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100535 if exists("b:asmsyntax")
Gregory Andersfa49eb42022-07-16 17:46:47 +0100536 exe "setf " .. fnameescape(b:asmsyntax)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100537 else
Gregory Andersfa49eb42022-07-16 17:46:47 +0100538 setf pov
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100539 endif
540 endif
541 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000542enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100543
Bram Moolenaara2baa732022-02-04 16:09:54 +0000544export def FTprogress_cweb()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100545 if exists("g:filetype_w")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000546 exe "setf " .. g:filetype_w
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100547 return
548 endif
549 if getline(1) =~ '&ANALYZE' || getline(3) =~ '&GLOBAL-DEFINE'
550 setf progress
551 else
552 setf cweb
553 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000554enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100555
Bram Moolenaara2baa732022-02-04 16:09:54 +0000556export def FTprogress_asm()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100557 if exists("g:filetype_i")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000558 exe "setf " .. g:filetype_i
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100559 return
560 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000561 # This function checks for an assembly comment the first ten lines.
562 # If not found, assume Progress.
563 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100564 while lnum <= 10 && lnum < line('$')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000565 var line = getline(lnum)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100566 if line =~ '^\s*;' || line =~ '^\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000567 FTasm()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100568 return
569 elseif line !~ '^\s*$' || line =~ '^/\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000570 # Not an empty line: Doesn't look like valid assembly code.
571 # Or it looks like a Progress /* comment
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100572 break
573 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000574 lnum += 1
Bram Moolenaarc12dc472022-03-05 13:45:56 +0000575 endwhile
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100576 setf progress
Bram Moolenaara2baa732022-02-04 16:09:54 +0000577enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100578
Bram Moolenaara2baa732022-02-04 16:09:54 +0000579var ft_pascal_comments = '^\s*\%({\|(\*\|//\)'
580var ft_pascal_keywords = '^\s*\%(program\|unit\|library\|uses\|begin\|procedure\|function\|const\|type\|var\)\>'
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100581
Bram Moolenaara2baa732022-02-04 16:09:54 +0000582export def FTprogress_pascal()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100583 if exists("g:filetype_p")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000584 exe "setf " .. g:filetype_p
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100585 return
586 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000587 # This function checks for valid Pascal syntax in the first ten lines.
588 # Look for either an opening comment or a program start.
589 # If not found, assume Progress.
590 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100591 while lnum <= 10 && lnum < line('$')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000592 var line = getline(lnum)
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000593 if line =~ ft_pascal_comments || line =~? ft_pascal_keywords
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100594 setf pascal
595 return
596 elseif line !~ '^\s*$' || line =~ '^/\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000597 # Not an empty line: Doesn't look like valid Pascal code.
598 # Or it looks like a Progress /* comment
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100599 break
600 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000601 lnum += 1
Bram Moolenaarc12dc472022-03-05 13:45:56 +0000602 endwhile
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100603 setf progress
Bram Moolenaara2baa732022-02-04 16:09:54 +0000604enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100605
Bram Moolenaara2baa732022-02-04 16:09:54 +0000606export def FTpp()
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100607 if exists("g:filetype_pp")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000608 exe "setf " .. g:filetype_pp
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100609 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000610 var line = getline(nextnonblank(1))
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000611 if line =~ ft_pascal_comments || line =~? ft_pascal_keywords
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100612 setf pascal
613 else
614 setf puppet
615 endif
616 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000617enddef
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100618
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100619# Determine if *.prg is ABB RAPID. Can also be Clipper, FoxPro or eviews
620export def FTprg()
621 if exists("g:filetype_prg")
622 exe "setf " .. g:filetype_prg
623 elseif IsRapid()
624 setf rapid
625 else
626 # Nothing recognized, assume Clipper
627 setf clipper
628 endif
629enddef
630
Bram Moolenaara2baa732022-02-04 16:09:54 +0000631export def FTr()
632 var max = line("$") > 50 ? 50 : line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100633
634 for n in range(1, max)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000635 # Rebol is easy to recognize, check for that first
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100636 if getline(n) =~? '\<REBOL\>'
637 setf rebol
638 return
639 endif
640 endfor
641
642 for n in range(1, max)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000643 # R has # comments
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100644 if getline(n) =~ '^\s*#'
645 setf r
646 return
647 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000648 # Rexx has /* comments */
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100649 if getline(n) =~ '^\s*/\*'
650 setf rexx
651 return
652 endif
653 endfor
654
Bram Moolenaara2baa732022-02-04 16:09:54 +0000655 # Nothing recognized, use user default or assume Rexx
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100656 if exists("g:filetype_r")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000657 exe "setf " .. g:filetype_r
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100658 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000659 # Rexx used to be the default, but R appears to be much more popular.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100660 setf r
661 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000662enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100663
Bram Moolenaara2baa732022-02-04 16:09:54 +0000664export def McSetf()
665 # Rely on the file to start with a comment.
666 # MS message text files use ';', Sendmail files use '#' or 'dnl'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100667 for lnum in range(1, min([line("$"), 20]))
Bram Moolenaara2baa732022-02-04 16:09:54 +0000668 var line = getline(lnum)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100669 if line =~ '^\s*\(#\|dnl\)'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000670 setf m4 # Sendmail .mc file
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100671 return
672 elseif line =~ '^\s*;'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000673 setf msmessages # MS Message text file
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100674 return
675 endif
676 endfor
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100677 setf m4 # Default: Sendmail .mc file
Bram Moolenaara2baa732022-02-04 16:09:54 +0000678enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100679
Bram Moolenaara2baa732022-02-04 16:09:54 +0000680# Called from filetype.vim and scripts.vim.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100681# When "setft" is passed and false then the 'filetype' option is not set.
682export def SetFileTypeSH(name: string, setft = true): string
683 if setft && did_filetype()
Bram Moolenaara2baa732022-02-04 16:09:54 +0000684 # Filetype was already detected
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100685 return ''
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100686 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100687 if setft && expand("<amatch>") =~ g:ft_ignore_pat
688 return ''
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100689 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000690 if name =~ '\<csh\>'
691 # Some .sh scripts contain #!/bin/csh.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100692 return SetFileTypeShell("csh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000693 elseif name =~ '\<tcsh\>'
694 # Some .sh scripts contain #!/bin/tcsh.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100695 return SetFileTypeShell("tcsh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000696 elseif name =~ '\<zsh\>'
697 # Some .sh scripts contain #!/bin/zsh.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100698 return SetFileTypeShell("zsh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000699 elseif name =~ '\<ksh\>'
700 b:is_kornshell = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100701 if exists("b:is_bash")
702 unlet b:is_bash
703 endif
704 if exists("b:is_sh")
705 unlet b:is_sh
706 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000707 elseif exists("g:bash_is_sh") || name =~ '\<bash\>' || name =~ '\<bash2\>'
708 b:is_bash = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100709 if exists("b:is_kornshell")
710 unlet b:is_kornshell
711 endif
712 if exists("b:is_sh")
713 unlet b:is_sh
714 endif
Eisuke Kawashima24482fb2022-11-24 10:58:10 +0000715 elseif name =~ '\<sh\>' || name =~ '\<dash\>'
716 # Ubuntu links "sh" to "dash", thus it is expected to work the same way
Bram Moolenaara2baa732022-02-04 16:09:54 +0000717 b:is_sh = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100718 if exists("b:is_kornshell")
719 unlet b:is_kornshell
720 endif
721 if exists("b:is_bash")
722 unlet b:is_bash
723 endif
724 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100725
726 return SetFileTypeShell("sh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000727enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100728
Bram Moolenaara2baa732022-02-04 16:09:54 +0000729# For shell-like file types, check for an "exec" command hidden in a comment,
730# as used for Tcl.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100731# When "setft" is passed and false then the 'filetype' option is not set.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000732# Also called from scripts.vim, thus can't be local to this script.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100733export def SetFileTypeShell(name: string, setft = true): string
734 if setft && did_filetype()
Bram Moolenaara2baa732022-02-04 16:09:54 +0000735 # Filetype was already detected
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100736 return ''
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100737 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100738 if setft && expand("<amatch>") =~ g:ft_ignore_pat
739 return ''
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100740 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100741
742 var lnum = 2
743 while lnum < 20 && lnum < line("$") && getline(lnum) =~ '^\s*\(#\|$\)'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000744 # Skip empty and comment lines.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100745 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100746 endwhile
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100747 if lnum < line("$") && getline(lnum) =~ '\s*exec\s' && getline(lnum - 1) =~ '^\s*#.*\\$'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000748 # Found an "exec" line after a comment with continuation
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100749 var n = substitute(getline(lnum), '\s*exec\s\+\([^ ]*/\)\=', '', '')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100750 if n =~ '\<tclsh\|\<wish'
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100751 if setft
752 setf tcl
753 endif
754 return 'tcl'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100755 endif
756 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100757
758 if setft
759 exe "setf " .. name
760 endif
761 return name
Bram Moolenaara2baa732022-02-04 16:09:54 +0000762enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100763
Bram Moolenaara2baa732022-02-04 16:09:54 +0000764export def CSH()
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100765 if did_filetype()
Bram Moolenaara2baa732022-02-04 16:09:54 +0000766 # Filetype was already detected
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100767 return
768 endif
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100769 if exists("g:filetype_csh")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000770 SetFileTypeShell(g:filetype_csh)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100771 elseif &shell =~ "tcsh"
Bram Moolenaara2baa732022-02-04 16:09:54 +0000772 SetFileTypeShell("tcsh")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100773 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000774 SetFileTypeShell("csh")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100775 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000776enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100777
Bram Moolenaara2baa732022-02-04 16:09:54 +0000778var ft_rules_udev_rules_pattern = '^\s*\cudev_rules\s*=\s*"\([^"]\{-1,}\)/*".*'
779export def FTRules()
780 var path = expand('<amatch>:p')
Bram Moolenaaraa9675a2020-08-17 21:57:09 +0200781 if path =~ '/\(etc/udev/\%(rules\.d/\)\=.*\.rules\|\%(usr/\)\=lib/udev/\%(rules\.d/\)\=.*\.rules\)$'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100782 setf udevrules
783 return
784 endif
785 if path =~ '^/etc/ufw/'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000786 setf conf # Better than hog
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100787 return
788 endif
789 if path =~ '^/\(etc\|usr/share\)/polkit-1/rules\.d'
790 setf javascript
791 return
792 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000793 var config_lines: list<string>
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100794 try
Bram Moolenaara2baa732022-02-04 16:09:54 +0000795 config_lines = readfile('/etc/udev/udev.conf')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100796 catch /^Vim\%((\a\+)\)\=:E484/
797 setf hog
798 return
799 endtry
Bram Moolenaara2baa732022-02-04 16:09:54 +0000800 var dir = expand('<amatch>:p:h')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100801 for line in config_lines
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000802 if line =~ ft_rules_udev_rules_pattern
803 var udev_rules = substitute(line, ft_rules_udev_rules_pattern, '\1', "")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100804 if dir == udev_rules
805 setf udevrules
806 endif
807 break
808 endif
809 endfor
810 setf hog
Bram Moolenaara2baa732022-02-04 16:09:54 +0000811enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100812
Bram Moolenaara2baa732022-02-04 16:09:54 +0000813export def SQL()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100814 if exists("g:filetype_sql")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000815 exe "setf " .. g:filetype_sql
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100816 else
817 setf sql
818 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000819enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100820
ranjithshegde8cac20e2022-04-13 15:29:21 +0100821# This function checks the first 25 lines of file extension "sc" to resolve
Chris Kipp70ef3f52022-12-14 16:42:15 +0000822# detection between scala and SuperCollider.
823# NOTE: We don't check for 'Class : Method', as this can easily be confused
824# with valid Scala like `val x : Int = 3`. So we instead only rely on
825# checks that can't be confused.
ranjithshegde8cac20e2022-04-13 15:29:21 +0100826export def FTsc()
827 for lnum in range(1, min([line("$"), 25]))
Chris Kipp70ef3f52022-12-14 16:42:15 +0000828 if getline(lnum) =~# 'var\s<\|classvar\s<\|\^this.*\||\w\+|\|+\s\w*\s{\|\*ar\s'
ranjithshegde8cac20e2022-04-13 15:29:21 +0100829 setf supercollider
830 return
831 endif
832 endfor
833 setf scala
834enddef
835
836# This function checks the first line of file extension "scd" to resolve
837# detection between scdoc and SuperCollider
838export def FTscd()
839 if getline(1) =~# '\%^\S\+(\d[0-9A-Za-z]*)\%(\s\+\"[^"]*\"\%(\s\+\"[^"]*\"\)\=\)\=$'
840 setf scdoc
841 else
842 setf supercollider
843 endif
844enddef
845
Bram Moolenaara2baa732022-02-04 16:09:54 +0000846# If the file has an extension of 't' and is in a directory 't' or 'xt' then
847# it is almost certainly a Perl test file.
848# If the first line starts with '#' and contains 'perl' it's probably a Perl
849# file.
850# (Slow test) If a file contains a 'use' statement then it is almost certainly
851# a Perl file.
852export def FTperl(): number
853 var dirname = expand("%:p:h:t")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100854 if expand("%:e") == 't' && (dirname == 't' || dirname == 'xt')
855 setf perl
856 return 1
857 endif
858 if getline(1)[0] == '#' && getline(1) =~ 'perl'
859 setf perl
860 return 1
861 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000862 var save_cursor = getpos('.')
863 call cursor(1, 1)
Bram Moolenaare5b78972022-02-05 19:50:34 +0000864 var has_use = search('^use\s\s*\k', 'c', 30) > 0
Bram Moolenaarf0b03c42017-12-17 17:17:07 +0100865 call setpos('.', save_cursor)
866 if has_use
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100867 setf perl
868 return 1
869 endif
870 return 0
Bram Moolenaara2baa732022-02-04 16:09:54 +0000871enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100872
Bram Moolenaarcdbfc6d2022-06-30 16:25:21 +0100873# LambdaProlog and Standard ML signature files
874export def FTsig()
875 if exists("g:filetype_sig")
876 exe "setf " .. g:filetype_sig
877 return
878 endif
879
880 var lprolog_comment = '^\s*\%(/\*\|%\)'
881 var lprolog_keyword = '^\s*sig\s\+\a'
882 var sml_comment = '^\s*(\*'
883 var sml_keyword = '^\s*\%(signature\|structure\)\s\+\a'
884
885 var line = getline(nextnonblank(1))
886
887 if line =~ lprolog_comment || line =~# lprolog_keyword
888 setf lprolog
889 elseif line =~ sml_comment || line =~# sml_keyword
890 setf sml
891 endif
892enddef
893
Bram Moolenaarbe807d52022-09-01 15:01:25 +0100894# This function checks the first 100 lines of files matching "*.sil" to
895# resolve detection between Swift Intermediate Language and SILE.
896export def FTsil()
897 for lnum in range(1, [line('$'), 100]->min())
898 var line: string = getline(lnum)
899 if line =~ '^\s*[\\%]'
900 setf sile
901 return
902 elseif line =~ '^\s*\S'
903 setf sil
904 return
905 endif
906 endfor
907 # no clue, default to "sil"
908 setf sil
909enddef
910
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100911export def FTsys()
KnoP-01f420ff22022-04-13 20:46:21 +0100912 if exists("g:filetype_sys")
913 exe "setf " .. g:filetype_sys
914 elseif IsRapid()
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100915 setf rapid
916 else
917 setf bat
918 endif
919enddef
920
Bram Moolenaara2baa732022-02-04 16:09:54 +0000921# Choose context, plaintex, or tex (LaTeX) based on these rules:
922# 1. Check the first line of the file for "%&<format>".
923# 2. Check the first 1000 non-comment lines for LaTeX or ConTeXt keywords.
924# 3. Default to "plain" or to g:tex_flavor, can be set in user's vimrc.
925export def FTtex()
926 var firstline = getline(1)
927 var format: string
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100928 if firstline =~ '^%&\s*\a\+'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000929 format = tolower(matchstr(firstline, '\a\+'))
930 format = substitute(format, 'pdf', '', '')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100931 if format == 'tex'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000932 format = 'latex'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100933 elseif format == 'plaintex'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000934 format = 'plain'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100935 endif
936 elseif expand('%') =~ 'tex/context/.*/.*.tex'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000937 format = 'context'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100938 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000939 # Default value, may be changed later:
940 format = exists("g:tex_flavor") ? g:tex_flavor : 'plain'
941 # Save position, go to the top of the file, find first non-comment line.
942 var save_cursor = getpos('.')
943 call cursor(1, 1)
944 var firstNC = search('^\s*[^[:space:]%]', 'c', 1000)
Bram Moolenaare5b78972022-02-05 19:50:34 +0000945 if firstNC > 0
946 # Check the next thousand lines for a LaTeX or ConTeXt keyword.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000947 var lpat = 'documentclass\>\|usepackage\>\|begin{\|newcommand\>\|renewcommand\>'
948 var cpat = 'start\a\+\|setup\a\+\|usemodule\|enablemode\|enableregime\|setvariables\|useencoding\|usesymbols\|stelle\a\+\|verwende\a\+\|stel\a\+\|gebruik\a\+\|usa\a\+\|imposta\a\+\|regle\a\+\|utilisemodule\>'
949 var kwline = search('^\s*\\\%(' .. lpat .. '\)\|^\s*\\\(' .. cpat .. '\)',
950 'cnp', firstNC + 1000)
951 if kwline == 1 # lpat matched
952 format = 'latex'
953 elseif kwline == 2 # cpat matched
954 format = 'context'
955 endif # If neither matched, keep default set above.
956 # let lline = search('^\s*\\\%(' . lpat . '\)', 'cn', firstNC + 1000)
957 # let cline = search('^\s*\\\%(' . cpat . '\)', 'cn', firstNC + 1000)
958 # if cline > 0
959 # let format = 'context'
960 # endif
961 # if lline > 0 && (cline == 0 || cline > lline)
962 # let format = 'tex'
963 # endif
964 endif # firstNC
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100965 call setpos('.', save_cursor)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000966 endif # firstline =~ '^%&\s*\a\+'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100967
Bram Moolenaara2baa732022-02-04 16:09:54 +0000968 # Translation from formats to file types. TODO: add AMSTeX, RevTex, others?
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100969 if format == 'plain'
970 setf plaintex
971 elseif format == 'context'
972 setf context
Bram Moolenaara2baa732022-02-04 16:09:54 +0000973 else # probably LaTeX
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100974 setf tex
975 endif
976 return
Bram Moolenaara2baa732022-02-04 16:09:54 +0000977enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100978
Bram Moolenaara2baa732022-02-04 16:09:54 +0000979export def FTxml()
980 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100981 while n < 100 && n <= line("$")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000982 var line = getline(n)
983 # DocBook 4 or DocBook 5.
984 var is_docbook4 = line =~ '<!DOCTYPE.*DocBook'
985 var is_docbook5 = line =~ ' xmlns="http://docbook.org/ns/docbook"'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100986 if is_docbook4 || is_docbook5
Bram Moolenaara2baa732022-02-04 16:09:54 +0000987 b:docbk_type = "xml"
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100988 if is_docbook5
Bram Moolenaara2baa732022-02-04 16:09:54 +0000989 b:docbk_ver = 5
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100990 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000991 b:docbk_ver = 4
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100992 endif
993 setf docbk
994 return
995 endif
996 if line =~ 'xmlns:xbl="http://www.mozilla.org/xbl"'
997 setf xbl
998 return
999 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001000 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001001 endwhile
1002 setf xml
Bram Moolenaara2baa732022-02-04 16:09:54 +00001003enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001004
Bram Moolenaara2baa732022-02-04 16:09:54 +00001005export def FTy()
1006 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +01001007 while n < 100 && n <= line("$")
Bram Moolenaara2baa732022-02-04 16:09:54 +00001008 var line = getline(n)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001009 if line =~ '^\s*%'
1010 setf yacc
1011 return
1012 endif
1013 if getline(n) =~ '^\s*\(#\|class\>\)' && getline(n) !~ '^\s*#\s*include'
1014 setf racc
1015 return
1016 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001017 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001018 endwhile
1019 setf yacc
Bram Moolenaara2baa732022-02-04 16:09:54 +00001020enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001021
Bram Moolenaara2baa732022-02-04 16:09:54 +00001022export def Redif()
1023 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001024 while lnum <= 5 && lnum < line('$')
1025 if getline(lnum) =~ "^\ctemplate-type:"
1026 setf redif
1027 return
1028 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001029 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001030 endwhile
Bram Moolenaara2baa732022-02-04 16:09:54 +00001031enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001032
Bram Moolenaara2baa732022-02-04 16:09:54 +00001033# This function is called for all files under */debian/patches/*, make sure not
1034# to non-dep3patch files, such as README and other text files.
1035export def Dep3patch()
James McCoy647ab4c2021-12-17 20:52:57 +00001036 if expand('%:t') ==# 'series'
1037 return
1038 endif
1039
1040 for ln in getline(1, 100)
1041 if ln =~# '^\%(Description\|Subject\|Origin\|Bug\|Forwarded\|Author\|From\|Reviewed-by\|Acked-by\|Last-Updated\|Applied-Upstream\):'
1042 setf dep3patch
1043 return
1044 elseif ln =~# '^---'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001045 # end of headers found. stop processing
James McCoy647ab4c2021-12-17 20:52:57 +00001046 return
1047 endif
1048 endfor
Bram Moolenaara2baa732022-02-04 16:09:54 +00001049enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001050
Bram Moolenaara2baa732022-02-04 16:09:54 +00001051# This function checks the first 15 lines for appearance of 'FoamFile'
1052# and then 'object' in a following line.
1053# In that case, it's probably an OpenFOAM file
1054export def FTfoam()
1055 var ffile = 0
1056 var lnum = 1
Elwardi2284f6c2022-01-11 18:14:23 +00001057 while lnum <= 15
1058 if getline(lnum) =~# '^FoamFile'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001059 ffile = 1
Elwardi2284f6c2022-01-11 18:14:23 +00001060 elseif ffile == 1 && getline(lnum) =~# '^\s*object'
1061 setf foam
1062 return
1063 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001064 lnum += 1
Elwardi2284f6c2022-01-11 18:14:23 +00001065 endwhile
Bram Moolenaara2baa732022-02-04 16:09:54 +00001066enddef
Elwardi2284f6c2022-01-11 18:14:23 +00001067
Bram Moolenaara2baa732022-02-04 16:09:54 +00001068# Determine if a *.tf file is TF mud client or terraform
1069export def FTtf()
1070 var numberOfLines = line('$')
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001071 for i in range(1, numberOfLines)
Bram Moolenaara2baa732022-02-04 16:09:54 +00001072 var currentLine = trim(getline(i))
1073 var firstCharacter = currentLine[0]
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001074 if firstCharacter !=? ";" && firstCharacter !=? "/" && firstCharacter !=? ""
1075 setf terraform
1076 return
1077 endif
1078 endfor
1079 setf tf
Bram Moolenaara2baa732022-02-04 16:09:54 +00001080enddef
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001081
KnoP-0193c7a452022-04-16 21:14:04 +01001082var ft_krl_header = '\&\w+'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001083# Determine if a *.src file is Kuka Robot Language
1084export def FTsrc()
KnoP-0193c7a452022-04-16 21:14:04 +01001085 var ft_krl_def_or_deffct = '%(global\s+)?def%(fct)?>'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001086 if exists("g:filetype_src")
1087 exe "setf " .. g:filetype_src
KnoP-0193c7a452022-04-16 21:14:04 +01001088 elseif getline(nextnonblank(1)) =~? '\v^\s*%(' .. ft_krl_header .. '|' .. ft_krl_def_or_deffct .. ')'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001089 setf krl
1090 endif
1091enddef
1092
1093# Determine if a *.dat file is Kuka Robot Language
1094export def FTdat()
KnoP-0193c7a452022-04-16 21:14:04 +01001095 var ft_krl_defdat = 'defdat>'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001096 if exists("g:filetype_dat")
1097 exe "setf " .. g:filetype_dat
KnoP-0193c7a452022-04-16 21:14:04 +01001098 elseif getline(nextnonblank(1)) =~? '\v^\s*%(' .. ft_krl_header .. '|' .. ft_krl_defdat .. ')'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001099 setf krl
1100 endif
1101enddef
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001102
Doug Kearns4ac8e792022-10-17 13:32:17 +01001103export def FTlsl()
1104 if exists("g:filetype_lsl")
1105 exe "setf " .. g:filetype_lsl
1106 endif
1107
1108 var line = getline(nextnonblank(1))
1109 if line =~ '^\s*%' || line =~# ':\s*trait\s*$'
1110 setf larch
1111 else
1112 setf lsl
1113 endif
1114enddef
1115
Gaetan Lepage4ce1bda2023-05-10 22:01:55 +01001116export def FTtyp()
1117 if exists("g:filetype_typ")
1118 exe "setf " .. g:filetype_typ
1119 return
1120 endif
1121
1122 # Look for SQL type definition syntax
1123 for line in getline(1, 200)
1124 # SQL type files may define the casing
1125 if line =~ '^CASE\s\==\s\=\(SAME\|LOWER\|UPPER\|OPPOSITE\)$'
1126 setf sql
1127 return
1128 endif
1129
1130 # SQL type files may define some types as follows
1131 if line =~ '^TYPE\s.*$'
1132 setf sql
1133 return
1134 endif
1135 endfor
1136
1137 # Otherwise, affect the typst filetype
1138 setf typst
1139enddef
1140
Turiiya80406c22023-04-22 21:38:47 +01001141# Set the filetype of a *.v file to Verilog, V or Cog based on the first 200
1142# lines.
1143export def FTv()
1144 if did_filetype()
1145 # ":setf" will do nothing, bail out early
1146 return
1147 endif
1148
1149 for line in getline(1, 200)
1150 if line[0] =~ '^\s*/'
1151 # skip comment line
1152 continue
1153 endif
1154
1155 # Verilog: line ends with ';' followed by an optional variable number of
1156 # spaces and an optional start of a comment.
1157 # Example: " b <= a + 1; // Add 1".
1158 if line =~ ';\(\s*\)\?\(/.*\)\?$'
1159 setf verilog
1160 return
1161 endif
1162
1163 # Coq: line ends with a '.' followed by an optional variable number of
1164 # spaces and an optional start of a comment.
1165 # Example: "Definition x := 10. (*".
1166 if line =~ '\.\(\s*\)\?\((\*.*\)\?$'
1167 setf coq
1168 return
1169 endif
1170 endfor
1171
1172 # No line matched, fall back to "v".
1173 setf v
1174enddef
1175
Bram Moolenaara2baa732022-02-04 16:09:54 +00001176# Uncomment this line to check for compilation errors early
Bram Moolenaar3e79c972022-02-04 19:48:06 +00001177# defcompile