blob: 9f53f684604c8bb0ce2c4f6c642c0b6ef04ba0f1 [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#
Christian Brabandte978b452023-08-13 10:33:05 +02005# Maintainer: The Vim Project <https://github.com/vim/vim>
6# Last Change: 2023 Aug 10
7# Former Maintainer: Bram Moolenaar <Bram@vim.org>
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01008
Bram Moolenaara2baa732022-02-04 16:09:54 +00009# These functions are moved here from runtime/filetype.vim to make startup
10# faster.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010011
Bram Moolenaara2baa732022-02-04 16:09:54 +000012export def Check_inp()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010013 if getline(1) =~ '^\*'
14 setf abaqus
15 else
Bram Moolenaara2baa732022-02-04 16:09:54 +000016 var n = 1
17 var nmax = line("$") > 500 ? 500 : line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010018 while n <= nmax
19 if getline(n) =~? "^header surface data"
20 setf trasys
21 break
22 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +000023 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010024 endwhile
25 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +000026enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010027
Bram Moolenaara2baa732022-02-04 16:09:54 +000028# This function checks for the kind of assembly that is wanted by the user, or
29# can be detected from the first five lines of the file.
30export def FTasm()
31 # make sure b:asmsyntax exists
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010032 if !exists("b:asmsyntax")
Bram Moolenaara2baa732022-02-04 16:09:54 +000033 b:asmsyntax = ""
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010034 endif
35
36 if b:asmsyntax == ""
Bram Moolenaara2baa732022-02-04 16:09:54 +000037 FTasmsyntax()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010038 endif
39
Bram Moolenaara2baa732022-02-04 16:09:54 +000040 # if b:asmsyntax still isn't set, default to asmsyntax or GNU
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010041 if b:asmsyntax == ""
42 if exists("g:asmsyntax")
Bram Moolenaara2baa732022-02-04 16:09:54 +000043 b:asmsyntax = g:asmsyntax
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010044 else
Bram Moolenaara2baa732022-02-04 16:09:54 +000045 b:asmsyntax = "asm"
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010046 endif
47 endif
48
Bram Moolenaara2baa732022-02-04 16:09:54 +000049 exe "setf " .. fnameescape(b:asmsyntax)
50enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010051
Bram Moolenaara2baa732022-02-04 16:09:54 +000052export def FTasmsyntax()
53 # see if the file contains any asmsyntax=foo overrides. If so, change
54 # b:asmsyntax appropriately
55 var head = " " .. getline(1) .. " " .. getline(2) .. " "
56 .. getline(3) .. " " .. getline(4) .. " " .. getline(5) .. " "
57 var match = matchstr(head, '\sasmsyntax=\zs[a-zA-Z0-9]\+\ze\s')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010058 if match != ''
Bram Moolenaara2baa732022-02-04 16:09:54 +000059 b:asmsyntax = match
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010060 elseif ((head =~? '\.title') || (head =~? '\.ident') || (head =~? '\.macro') || (head =~? '\.subtitle') || (head =~? '\.library'))
Bram Moolenaara2baa732022-02-04 16:09:54 +000061 b:asmsyntax = "vmasm"
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010062 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +000063enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010064
Bram Moolenaara2baa732022-02-04 16:09:54 +000065var ft_visual_basic_content = '\cVB_Name\|Begin VB\.\(Form\|MDIForm\|UserControl\)'
Doug Kearnsc570e9c2022-01-31 17:09:14 +000066
Bram Moolenaara2baa732022-02-04 16:09:54 +000067# See FTfrm() for Visual Basic form file detection
68export def FTbas()
Bram Moolenaar6517f142022-01-21 14:55:13 +000069 if exists("g:filetype_bas")
Bram Moolenaara2baa732022-02-04 16:09:54 +000070 exe "setf " .. g:filetype_bas
Bram Moolenaar6517f142022-01-21 14:55:13 +000071 return
72 endif
73
Bram Moolenaara2baa732022-02-04 16:09:54 +000074 # most frequent FreeBASIC-specific keywords in distro files
75 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 +010076 var fb_preproc = '\c^\s*\%(' ..
77 # preprocessor
78 '#\s*\a\+\|' ..
79 # compiler option
80 'option\s\+\%(byval\|dynamic\|escape\|\%(no\)\=gosub\|nokeyword\|private\|static\)\>\|' ..
81 # metacommand
82 '\%(''\|rem\)\s*\$lang\>\|' ..
83 # default datatype
84 'def\%(byte\|longint\|short\|ubyte\|uint\|ulongint\|ushort\)\>' ..
85 '\)'
Bram Moolenaara2baa732022-02-04 16:09:54 +000086 var fb_comment = "^\\s*/'"
Bram Moolenaar8b5901e2022-06-29 14:39:12 +010087
Bram Moolenaara2baa732022-02-04 16:09:54 +000088 # OPTION EXPLICIT, without the leading underscore, is common to many dialects
89 var qb64_preproc = '\c^\s*\%($\a\+\|option\s\+\%(_explicit\|_\=explicitarray\)\>\)'
Bram Moolenaar6517f142022-01-21 14:55:13 +000090
Bram Moolenaar8b5901e2022-06-29 14:39:12 +010091 for lnum in range(1, min([line("$"), 100]))
92 var line = getline(lnum)
93 if line =~ ft_visual_basic_content
94 setf vb
95 return
96 elseif line =~ fb_preproc || line =~ fb_comment || line =~ fb_keywords
97 setf freebasic
98 return
99 elseif line =~ qb64_preproc
100 setf qb64
101 return
102 endif
103 endfor
104 setf basic
Bram Moolenaara2baa732022-02-04 16:09:54 +0000105enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100106
Bram Moolenaara2baa732022-02-04 16:09:54 +0000107export def FTbtm()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100108 if exists("g:dosbatch_syntax_for_btm") && g:dosbatch_syntax_for_btm
109 setf dosbatch
110 else
111 setf btm
112 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000113enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100114
Bram Moolenaara2baa732022-02-04 16:09:54 +0000115export def BindzoneCheck(default = '')
116 if getline(1) .. getline(2) .. getline(3) .. getline(4)
117 =~ '^; <<>> DiG [0-9.]\+.* <<>>\|$ORIGIN\|$TTL\|IN\s\+SOA'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100118 setf bindzone
Bram Moolenaara2baa732022-02-04 16:09:54 +0000119 elseif default != ''
120 exe 'setf ' .. default
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100121 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000122enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100123
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100124# Returns true if file content looks like RAPID
125def IsRapid(sChkExt: string = ""): bool
126 if sChkExt == "cfg"
127 return getline(1) =~? '\v^%(EIO|MMC|MOC|PROC|SIO|SYS):CFG'
128 endif
129 # called from FTmod, FTprg or FTsys
130 return getline(nextnonblank(1)) =~? '\v^\s*%(\%{3}|module\s+\k+\s*%(\(|$))'
131enddef
132
133export def FTcfg()
134 if exists("g:filetype_cfg")
135 exe "setf " .. g:filetype_cfg
136 elseif IsRapid("cfg")
137 setf rapid
138 else
139 setf cfg
140 endif
141enddef
142
Bram Moolenaar8b5901e2022-06-29 14:39:12 +0100143export def FTcls()
144 if exists("g:filetype_cls")
145 exe "setf " .. g:filetype_cls
146 return
147 endif
148
Martin Tournoijbd053f82022-10-16 12:49:12 +0100149 if getline(1) =~ '^\v%(\%|\\)'
Bram Moolenaar8b5901e2022-06-29 14:39:12 +0100150 setf tex
151 elseif getline(1)[0] == '#' && getline(1) =~ 'rexx'
152 setf rexx
153 elseif getline(1) == 'VERSION 1.0 CLASS'
154 setf vb
155 else
156 setf st
157 endif
158enddef
159
Bram Moolenaara2baa732022-02-04 16:09:54 +0000160export def FTlpc()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100161 if exists("g:lpc_syntax_for_c")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000162 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100163 while lnum <= 12
164 if getline(lnum) =~# '^\(//\|inherit\|private\|protected\|nosave\|string\|object\|mapping\|mixed\)'
165 setf lpc
166 return
167 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000168 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100169 endwhile
170 endif
171 setf c
Bram Moolenaara2baa732022-02-04 16:09:54 +0000172enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100173
Bram Moolenaara2baa732022-02-04 16:09:54 +0000174export def FTheader()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100175 if match(getline(1, min([line("$"), 200])), '^@\(interface\|end\|class\)') > -1
176 if exists("g:c_syntax_for_h")
177 setf objc
178 else
179 setf objcpp
180 endif
181 elseif exists("g:c_syntax_for_h")
182 setf c
183 elseif exists("g:ch_syntax_for_h")
184 setf ch
185 else
186 setf cpp
187 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000188enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100189
Bram Moolenaara2baa732022-02-04 16:09:54 +0000190# This function checks if one of the first ten lines start with a '@'. In
191# that case it is probably a change file.
192# If the first line starts with # or ! it's probably a ch file.
193# If a line has "main", "include", "//" or "/*" it's probably ch.
194# Otherwise CHILL is assumed.
195export def FTchange()
196 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100197 while lnum <= 10
198 if getline(lnum)[0] == '@'
199 setf change
200 return
201 endif
202 if lnum == 1 && (getline(1)[0] == '#' || getline(1)[0] == '!')
203 setf ch
204 return
205 endif
206 if getline(lnum) =~ "MODULE"
207 setf chill
208 return
209 endif
210 if getline(lnum) =~ 'main\s*(\|#\s*include\|//'
211 setf ch
212 return
213 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000214 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100215 endwhile
216 setf chill
Bram Moolenaara2baa732022-02-04 16:09:54 +0000217enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100218
Bram Moolenaara2baa732022-02-04 16:09:54 +0000219export def FTent()
220 # This function checks for valid cl syntax in the first five lines.
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100221 # Look for either an opening comment, '#', or a block start, '{'.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000222 # If not found, assume SGML.
223 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100224 while lnum < 6
Bram Moolenaara2baa732022-02-04 16:09:54 +0000225 var line = getline(lnum)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100226 if line =~ '^\s*[#{]'
227 setf cl
228 return
229 elseif line !~ '^\s*$'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000230 # Not a blank line, not a comment, and not a block start,
231 # so doesn't look like valid cl code.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100232 break
233 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000234 lnum += 1
Bram Moolenaarc12dc472022-03-05 13:45:56 +0000235 endwhile
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100236 setf dtd
Bram Moolenaara2baa732022-02-04 16:09:54 +0000237enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100238
Bram Moolenaara2baa732022-02-04 16:09:54 +0000239export def ExCheck()
240 var lines = getline(1, min([line("$"), 100]))
Austin Gatlinf3caeb62021-06-26 12:02:55 +0200241 if exists('g:filetype_euphoria')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000242 exe 'setf ' .. g:filetype_euphoria
Austin Gatlinf3caeb62021-06-26 12:02:55 +0200243 elseif match(lines, '^--\|^ifdef\>\|^include\>') > -1
244 setf euphoria3
245 else
246 setf elixir
247 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000248enddef
Austin Gatlinf3caeb62021-06-26 12:02:55 +0200249
Bram Moolenaara2baa732022-02-04 16:09:54 +0000250export def EuphoriaCheck()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100251 if exists('g:filetype_euphoria')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000252 exe 'setf ' .. g:filetype_euphoria
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100253 else
254 setf euphoria3
255 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000256enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100257
Bram Moolenaara2baa732022-02-04 16:09:54 +0000258export def DtraceCheck()
=?UTF-8?q?Teubel=20Gy=C3=B6rgy?=4d56b972022-02-24 17:59:09 +0000259 if did_filetype()
260 # Filetype was already detected
261 return
262 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000263 var lines = getline(1, min([line("$"), 100]))
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100264 if match(lines, '^module\>\|^import\>') > -1
Bram Moolenaara2baa732022-02-04 16:09:54 +0000265 # D files often start with a module and/or import statement.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100266 setf d
267 elseif match(lines, '^#!\S\+dtrace\|#pragma\s\+D\s\+option\|:\S\{-}:\S\{-}:') > -1
268 setf dtrace
269 else
270 setf d
271 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000272enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100273
Bram Moolenaara2baa732022-02-04 16:09:54 +0000274export def FTe()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100275 if exists('g:filetype_euphoria')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000276 exe 'setf ' .. g:filetype_euphoria
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100277 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000278 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100279 while n < 100 && n <= line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100280 if getline(n) =~ "^\\s*\\(<'\\|'>\\)\\s*$"
281 setf specman
282 return
283 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000284 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100285 endwhile
286 setf eiffel
287 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000288enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100289
Bram Moolenaara2baa732022-02-04 16:09:54 +0000290export def FTfrm()
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000291 if exists("g:filetype_frm")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000292 exe "setf " .. g:filetype_frm
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000293 return
294 endif
295
Bram Moolenaara2baa732022-02-04 16:09:54 +0000296 var lines = getline(1, min([line("$"), 5]))
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000297
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000298 if match(lines, ft_visual_basic_content) > -1
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000299 setf vb
300 else
301 setf form
302 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000303enddef
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000304
Bram Moolenaara2baa732022-02-04 16:09:54 +0000305# Distinguish between Forth and F#.
306# Provided by Doug Kearns.
307export def FTfs()
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000308 if exists("g:filetype_fs")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000309 exe "setf " .. g:filetype_fs
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000310 else
Johan Kotlinski065088d2023-04-02 20:29:38 +0100311 var n = 1
312 while n < 100 && n <= line("$")
313 # Forth comments and colon definitions
314 if getline(n) =~ "^[:(\\\\] "
315 setf forth
316 return
317 endif
318 n += 1
319 endwhile
320 setf fsharp
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000321 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000322enddef
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000323
Bram Moolenaara2baa732022-02-04 16:09:54 +0000324# Distinguish between HTML, XHTML and Django
325export def FThtml()
326 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100327 while n < 10 && n <= line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100328 if getline(n) =~ '\<DTD\s\+XHTML\s'
329 setf xhtml
330 return
331 endif
332 if getline(n) =~ '{%\s*\(extends\|block\|load\)\>\|{#\s\+'
333 setf htmldjango
334 return
335 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000336 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100337 endwhile
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100338 setf FALLBACK html
Bram Moolenaara2baa732022-02-04 16:09:54 +0000339enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100340
Bram Moolenaara2baa732022-02-04 16:09:54 +0000341# Distinguish between standard IDL and MS-IDL
342export def FTidl()
343 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100344 while n < 50 && n <= line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100345 if getline(n) =~ '^\s*import\s\+"\(unknwn\|objidl\)\.idl"'
346 setf msidl
347 return
348 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000349 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100350 endwhile
351 setf idl
Bram Moolenaara2baa732022-02-04 16:09:54 +0000352enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100353
Bram Moolenaar48c3f4e2022-08-08 15:42:38 +0100354# Distinguish between "default", Prolog and Cproto prototype file.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000355export def ProtoCheck(default: string)
356 # Cproto files have a comment in the first line and a function prototype in
357 # the second line, it always ends in ";". Indent files may also have
358 # comments, thus we can't match comments to see the difference.
359 # IDL files can have a single ';' in the second line, require at least one
360 # chacter before the ';'.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100361 if getline(2) =~ '.;$'
362 setf cpp
363 else
Bram Moolenaar48c3f4e2022-08-08 15:42:38 +0100364 # recognize Prolog by specific text in the first non-empty line
365 # require a blank after the '%' because Perl uses "%list" and "%translate"
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100366 var lnum = getline(nextnonblank(1))
367 if lnum =~ '\<prolog\>' || lnum =~ '^\s*\(%\+\(\s\|$\)\|/\*\)' || lnum =~ ':-'
Bram Moolenaar48c3f4e2022-08-08 15:42:38 +0100368 setf prolog
369 else
370 exe 'setf ' .. default
371 endif
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100372 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000373enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100374
Bram Moolenaara2baa732022-02-04 16:09:54 +0000375export def FTm()
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200376 if exists("g:filetype_m")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000377 exe "setf " .. g:filetype_m
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200378 return
379 endif
380
Bram Moolenaara2baa732022-02-04 16:09:54 +0000381 # excluding end(for|function|if|switch|while) common to Murphi
382 var octave_block_terminators = '\<end\%(_try_catch\|classdef\|enumeration\|events\|methods\|parfor\|properties\)\>'
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200383
Bram Moolenaara2baa732022-02-04 16:09:54 +0000384 var objc_preprocessor = '^\s*#\s*\%(import\|include\|define\|if\|ifn\=def\|undef\|line\|error\|pragma\)\>'
Doug Kearns7329cfa2021-11-26 13:01:41 +0000385
Bram Moolenaara2baa732022-02-04 16:09:54 +0000386 var n = 1
387 var saw_comment = 0 # Whether we've seen a multiline comment leader.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100388 while n < 100
Bram Moolenaara2baa732022-02-04 16:09:54 +0000389 var line = getline(n)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100390 if line =~ '^\s*/\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000391 # /* ... */ is a comment in Objective C and Murphi, so we can't conclude
392 # it's either of them yet, but track this as a hint in case we don't see
393 # anything more definitive.
394 saw_comment = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100395 endif
Doug Kearns7329cfa2021-11-26 13:01:41 +0000396 if line =~ '^\s*//' || line =~ '^\s*@import\>' || line =~ objc_preprocessor
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100397 setf objc
398 return
399 endif
Bram Moolenaarca0627d2021-09-12 17:03:08 +0200400 if line =~ '^\s*\%(#\|%!\)' || line =~ '^\s*unwind_protect\>' ||
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200401 \ line =~ '\%(^\|;\)\s*' .. octave_block_terminators
402 setf octave
403 return
404 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000405 # TODO: could be Matlab or Octave
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100406 if line =~ '^\s*%'
407 setf matlab
408 return
409 endif
410 if line =~ '^\s*(\*'
411 setf mma
412 return
413 endif
414 if line =~ '^\c\s*\(\(type\|var\)\>\|--\)'
415 setf murphi
416 return
417 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000418 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100419 endwhile
420
421 if saw_comment
Bram Moolenaara2baa732022-02-04 16:09:54 +0000422 # We didn't see anything definitive, but this looks like either Objective C
423 # or Murphi based on the comment leader. Assume the former as it is more
424 # common.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100425 setf objc
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100426 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000427 # Default is Matlab
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100428 setf matlab
429 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000430enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100431
Bram Moolenaara2baa732022-02-04 16:09:54 +0000432export def FTmms()
433 var n = 1
Bram Moolenaard7df2792020-01-02 21:34:42 +0100434 while n < 20
Bram Moolenaara2baa732022-02-04 16:09:54 +0000435 var line = getline(n)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100436 if line =~ '^\s*\(%\|//\)' || line =~ '^\*'
437 setf mmix
438 return
439 endif
440 if line =~ '^\s*#'
441 setf make
442 return
443 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000444 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100445 endwhile
446 setf mmix
Bram Moolenaara2baa732022-02-04 16:09:54 +0000447enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100448
Bram Moolenaara2baa732022-02-04 16:09:54 +0000449# This function checks if one of the first five lines start with a dot. In
450# that case it is probably an nroff file: 'filetype' is set and 1 is returned.
451export def FTnroff(): number
452 if getline(1)[0] .. getline(2)[0] .. getline(3)[0]
453 .. getline(4)[0] .. getline(5)[0] =~ '\.'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100454 setf nroff
455 return 1
456 endif
457 return 0
Bram Moolenaara2baa732022-02-04 16:09:54 +0000458enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100459
Bram Moolenaara2baa732022-02-04 16:09:54 +0000460export def FTmm()
461 var n = 1
Bram Moolenaard1caa942020-04-10 22:10:56 +0200462 while n < 20
Bram Moolenaara2baa732022-02-04 16:09:54 +0000463 if getline(n) =~ '^\s*\(#\s*\(include\|import\)\>\|@import\>\|/\*\)'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100464 setf objcpp
465 return
466 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000467 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100468 endwhile
469 setf nroff
Bram Moolenaara2baa732022-02-04 16:09:54 +0000470enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100471
Bram Moolenaarcdbfc6d2022-06-30 16:25:21 +0100472# Returns true if file content looks like LambdaProlog module
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100473def IsLProlog(): bool
Gaetan Lepage4ce1bda2023-05-10 22:01:55 +0100474 # skip apparent comments and blank lines, what looks like
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100475 # LambdaProlog comment may be RAPID header
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100476 var lnum: number = nextnonblank(1)
477 while lnum > 0 && lnum < line('$') && getline(lnum) =~ '^\s*%' # LambdaProlog comment
478 lnum = nextnonblank(lnum + 1)
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100479 endwhile
480 # this pattern must not catch a go.mod file
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100481 return getline(lnum) =~ '\<module\s\+\w\+\s*\.\s*\(%\|$\)'
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100482enddef
483
484# Determine if *.mod is ABB RAPID, LambdaProlog, Modula-2, Modsim III or go.mod
485export def FTmod()
486 if exists("g:filetype_mod")
487 exe "setf " .. g:filetype_mod
Omar El Halabic9fbd252023-05-29 19:59:45 +0100488 elseif expand("<afile>") =~ '\<go.mod$'
489 setf gomod
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100490 elseif IsLProlog()
491 setf lprolog
492 elseif getline(nextnonblank(1)) =~ '\%(\<MODULE\s\+\w\+\s*;\|^\s*(\*\)'
493 setf modula2
494 elseif IsRapid()
495 setf rapid
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100496 else
497 # Nothing recognized, assume modsim3
498 setf modsim3
499 endif
500enddef
501
Bram Moolenaara2baa732022-02-04 16:09:54 +0000502export def FTpl()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100503 if exists("g:filetype_pl")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000504 exe "setf " .. g:filetype_pl
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100505 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000506 # recognize Prolog by specific text in the first non-empty line
507 # require a blank after the '%' because Perl uses "%list" and "%translate"
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100508 var line = getline(nextnonblank(1))
509 if line =~ '\<prolog\>' || line =~ '^\s*\(%\+\(\s\|$\)\|/\*\)' || line =~ ':-'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100510 setf prolog
511 else
512 setf perl
513 endif
514 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000515enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100516
Bram Moolenaara2baa732022-02-04 16:09:54 +0000517export def FTinc()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100518 if exists("g:filetype_inc")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000519 exe "setf " .. g:filetype_inc
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100520 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000521 var lines = getline(1) .. getline(2) .. getline(3)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100522 if lines =~? "perlscript"
523 setf aspperl
524 elseif lines =~ "<%"
525 setf aspvbs
526 elseif lines =~ "<?"
527 setf php
Bram Moolenaara2baa732022-02-04 16:09:54 +0000528 # Pascal supports // comments but they're vary rarely used for file
529 # headers so assume POV-Ray
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000530 elseif lines =~ '^\s*\%({\|(\*\)' || lines =~? ft_pascal_keywords
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100531 setf pascal
Gregory Anders30e212d2022-07-26 21:42:03 +0100532 elseif lines =~# '\<\%(require\|inherit\)\>' || lines =~# '[A-Z][A-Za-z0-9_:${}]*\s\+\%(??\|[?:+]\)\?= '
Gregory Andersfa49eb42022-07-16 17:46:47 +0100533 setf bitbake
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100534 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000535 FTasmsyntax()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100536 if exists("b:asmsyntax")
Gregory Andersfa49eb42022-07-16 17:46:47 +0100537 exe "setf " .. fnameescape(b:asmsyntax)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100538 else
Gregory Andersfa49eb42022-07-16 17:46:47 +0100539 setf pov
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100540 endif
541 endif
542 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000543enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100544
Bram Moolenaara2baa732022-02-04 16:09:54 +0000545export def FTprogress_cweb()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100546 if exists("g:filetype_w")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000547 exe "setf " .. g:filetype_w
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100548 return
549 endif
550 if getline(1) =~ '&ANALYZE' || getline(3) =~ '&GLOBAL-DEFINE'
551 setf progress
552 else
553 setf cweb
554 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000555enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100556
Bram Moolenaara2baa732022-02-04 16:09:54 +0000557export def FTprogress_asm()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100558 if exists("g:filetype_i")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000559 exe "setf " .. g:filetype_i
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100560 return
561 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000562 # This function checks for an assembly comment the first ten lines.
563 # If not found, assume Progress.
564 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100565 while lnum <= 10 && lnum < line('$')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000566 var line = getline(lnum)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100567 if line =~ '^\s*;' || line =~ '^\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000568 FTasm()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100569 return
570 elseif line !~ '^\s*$' || line =~ '^/\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000571 # Not an empty line: Doesn't look like valid assembly code.
572 # Or it looks like a Progress /* comment
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100573 break
574 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000575 lnum += 1
Bram Moolenaarc12dc472022-03-05 13:45:56 +0000576 endwhile
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100577 setf progress
Bram Moolenaara2baa732022-02-04 16:09:54 +0000578enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100579
Bram Moolenaara2baa732022-02-04 16:09:54 +0000580var ft_pascal_comments = '^\s*\%({\|(\*\|//\)'
581var ft_pascal_keywords = '^\s*\%(program\|unit\|library\|uses\|begin\|procedure\|function\|const\|type\|var\)\>'
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100582
Bram Moolenaara2baa732022-02-04 16:09:54 +0000583export def FTprogress_pascal()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100584 if exists("g:filetype_p")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000585 exe "setf " .. g:filetype_p
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100586 return
587 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000588 # This function checks for valid Pascal syntax in the first ten lines.
589 # Look for either an opening comment or a program start.
590 # If not found, assume Progress.
591 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100592 while lnum <= 10 && lnum < line('$')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000593 var line = getline(lnum)
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000594 if line =~ ft_pascal_comments || line =~? ft_pascal_keywords
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100595 setf pascal
596 return
597 elseif line !~ '^\s*$' || line =~ '^/\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000598 # Not an empty line: Doesn't look like valid Pascal code.
599 # Or it looks like a Progress /* comment
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100600 break
601 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000602 lnum += 1
Bram Moolenaarc12dc472022-03-05 13:45:56 +0000603 endwhile
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100604 setf progress
Bram Moolenaara2baa732022-02-04 16:09:54 +0000605enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100606
Bram Moolenaara2baa732022-02-04 16:09:54 +0000607export def FTpp()
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100608 if exists("g:filetype_pp")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000609 exe "setf " .. g:filetype_pp
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100610 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000611 var line = getline(nextnonblank(1))
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000612 if line =~ ft_pascal_comments || line =~? ft_pascal_keywords
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100613 setf pascal
614 else
615 setf puppet
616 endif
617 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000618enddef
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100619
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100620# Determine if *.prg is ABB RAPID. Can also be Clipper, FoxPro or eviews
621export def FTprg()
622 if exists("g:filetype_prg")
623 exe "setf " .. g:filetype_prg
624 elseif IsRapid()
625 setf rapid
626 else
627 # Nothing recognized, assume Clipper
628 setf clipper
629 endif
630enddef
631
Bram Moolenaara2baa732022-02-04 16:09:54 +0000632export def FTr()
633 var max = line("$") > 50 ? 50 : line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100634
635 for n in range(1, max)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000636 # Rebol is easy to recognize, check for that first
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100637 if getline(n) =~? '\<REBOL\>'
638 setf rebol
639 return
640 endif
641 endfor
642
643 for n in range(1, max)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000644 # R has # comments
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100645 if getline(n) =~ '^\s*#'
646 setf r
647 return
648 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000649 # Rexx has /* comments */
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100650 if getline(n) =~ '^\s*/\*'
651 setf rexx
652 return
653 endif
654 endfor
655
Bram Moolenaara2baa732022-02-04 16:09:54 +0000656 # Nothing recognized, use user default or assume Rexx
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100657 if exists("g:filetype_r")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000658 exe "setf " .. g:filetype_r
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100659 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000660 # Rexx used to be the default, but R appears to be much more popular.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100661 setf r
662 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000663enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100664
Bram Moolenaara2baa732022-02-04 16:09:54 +0000665export def McSetf()
666 # Rely on the file to start with a comment.
667 # MS message text files use ';', Sendmail files use '#' or 'dnl'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100668 for lnum in range(1, min([line("$"), 20]))
Bram Moolenaara2baa732022-02-04 16:09:54 +0000669 var line = getline(lnum)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100670 if line =~ '^\s*\(#\|dnl\)'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000671 setf m4 # Sendmail .mc file
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100672 return
673 elseif line =~ '^\s*;'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000674 setf msmessages # MS Message text file
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100675 return
676 endif
677 endfor
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100678 setf m4 # Default: Sendmail .mc file
Bram Moolenaara2baa732022-02-04 16:09:54 +0000679enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100680
Bram Moolenaara2baa732022-02-04 16:09:54 +0000681# Called from filetype.vim and scripts.vim.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100682# When "setft" is passed and false then the 'filetype' option is not set.
683export def SetFileTypeSH(name: string, setft = true): string
684 if setft && did_filetype()
Bram Moolenaara2baa732022-02-04 16:09:54 +0000685 # Filetype was already detected
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100686 return ''
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100687 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100688 if setft && expand("<amatch>") =~ g:ft_ignore_pat
689 return ''
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100690 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000691 if name =~ '\<csh\>'
692 # Some .sh scripts contain #!/bin/csh.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100693 return SetFileTypeShell("csh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000694 elseif name =~ '\<tcsh\>'
695 # Some .sh scripts contain #!/bin/tcsh.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100696 return SetFileTypeShell("tcsh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000697 elseif name =~ '\<zsh\>'
698 # Some .sh scripts contain #!/bin/zsh.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100699 return SetFileTypeShell("zsh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000700 elseif name =~ '\<ksh\>'
701 b:is_kornshell = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100702 if exists("b:is_bash")
703 unlet b:is_bash
704 endif
705 if exists("b:is_sh")
706 unlet b:is_sh
707 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000708 elseif exists("g:bash_is_sh") || name =~ '\<bash\>' || name =~ '\<bash2\>'
709 b:is_bash = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100710 if exists("b:is_kornshell")
711 unlet b:is_kornshell
712 endif
713 if exists("b:is_sh")
714 unlet b:is_sh
715 endif
Eisuke Kawashima24482fb2022-11-24 10:58:10 +0000716 elseif name =~ '\<sh\>' || name =~ '\<dash\>'
717 # Ubuntu links "sh" to "dash", thus it is expected to work the same way
Bram Moolenaara2baa732022-02-04 16:09:54 +0000718 b:is_sh = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100719 if exists("b:is_kornshell")
720 unlet b:is_kornshell
721 endif
722 if exists("b:is_bash")
723 unlet b:is_bash
724 endif
725 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100726
727 return SetFileTypeShell("sh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000728enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100729
Bram Moolenaara2baa732022-02-04 16:09:54 +0000730# For shell-like file types, check for an "exec" command hidden in a comment,
731# as used for Tcl.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100732# When "setft" is passed and false then the 'filetype' option is not set.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000733# Also called from scripts.vim, thus can't be local to this script.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100734export def SetFileTypeShell(name: string, setft = true): string
735 if setft && did_filetype()
Bram Moolenaara2baa732022-02-04 16:09:54 +0000736 # Filetype was already detected
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100737 return ''
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100738 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100739 if setft && expand("<amatch>") =~ g:ft_ignore_pat
740 return ''
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100741 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100742
743 var lnum = 2
744 while lnum < 20 && lnum < line("$") && getline(lnum) =~ '^\s*\(#\|$\)'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000745 # Skip empty and comment lines.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100746 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100747 endwhile
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100748 if lnum < line("$") && getline(lnum) =~ '\s*exec\s' && getline(lnum - 1) =~ '^\s*#.*\\$'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000749 # Found an "exec" line after a comment with continuation
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100750 var n = substitute(getline(lnum), '\s*exec\s\+\([^ ]*/\)\=', '', '')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100751 if n =~ '\<tclsh\|\<wish'
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100752 if setft
753 setf tcl
754 endif
755 return 'tcl'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100756 endif
757 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100758
759 if setft
760 exe "setf " .. name
761 endif
762 return name
Bram Moolenaara2baa732022-02-04 16:09:54 +0000763enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100764
Bram Moolenaara2baa732022-02-04 16:09:54 +0000765export def CSH()
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100766 if did_filetype()
Bram Moolenaara2baa732022-02-04 16:09:54 +0000767 # Filetype was already detected
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100768 return
769 endif
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100770 if exists("g:filetype_csh")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000771 SetFileTypeShell(g:filetype_csh)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100772 elseif &shell =~ "tcsh"
Bram Moolenaara2baa732022-02-04 16:09:54 +0000773 SetFileTypeShell("tcsh")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100774 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000775 SetFileTypeShell("csh")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100776 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000777enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100778
Bram Moolenaara2baa732022-02-04 16:09:54 +0000779var ft_rules_udev_rules_pattern = '^\s*\cudev_rules\s*=\s*"\([^"]\{-1,}\)/*".*'
780export def FTRules()
781 var path = expand('<amatch>:p')
Bram Moolenaaraa9675a2020-08-17 21:57:09 +0200782 if path =~ '/\(etc/udev/\%(rules\.d/\)\=.*\.rules\|\%(usr/\)\=lib/udev/\%(rules\.d/\)\=.*\.rules\)$'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100783 setf udevrules
784 return
785 endif
786 if path =~ '^/etc/ufw/'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000787 setf conf # Better than hog
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100788 return
789 endif
790 if path =~ '^/\(etc\|usr/share\)/polkit-1/rules\.d'
791 setf javascript
792 return
793 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000794 var config_lines: list<string>
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100795 try
Bram Moolenaara2baa732022-02-04 16:09:54 +0000796 config_lines = readfile('/etc/udev/udev.conf')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100797 catch /^Vim\%((\a\+)\)\=:E484/
798 setf hog
799 return
800 endtry
Bram Moolenaara2baa732022-02-04 16:09:54 +0000801 var dir = expand('<amatch>:p:h')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100802 for line in config_lines
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000803 if line =~ ft_rules_udev_rules_pattern
804 var udev_rules = substitute(line, ft_rules_udev_rules_pattern, '\1', "")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100805 if dir == udev_rules
806 setf udevrules
807 endif
808 break
809 endif
810 endfor
811 setf hog
Bram Moolenaara2baa732022-02-04 16:09:54 +0000812enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100813
Bram Moolenaara2baa732022-02-04 16:09:54 +0000814export def SQL()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100815 if exists("g:filetype_sql")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000816 exe "setf " .. g:filetype_sql
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100817 else
818 setf sql
819 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000820enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100821
ranjithshegde8cac20e2022-04-13 15:29:21 +0100822# This function checks the first 25 lines of file extension "sc" to resolve
Chris Kipp70ef3f52022-12-14 16:42:15 +0000823# detection between scala and SuperCollider.
824# NOTE: We don't check for 'Class : Method', as this can easily be confused
825# with valid Scala like `val x : Int = 3`. So we instead only rely on
826# checks that can't be confused.
ranjithshegde8cac20e2022-04-13 15:29:21 +0100827export def FTsc()
828 for lnum in range(1, min([line("$"), 25]))
Chris Kipp70ef3f52022-12-14 16:42:15 +0000829 if getline(lnum) =~# 'var\s<\|classvar\s<\|\^this.*\||\w\+|\|+\s\w*\s{\|\*ar\s'
ranjithshegde8cac20e2022-04-13 15:29:21 +0100830 setf supercollider
831 return
832 endif
833 endfor
834 setf scala
835enddef
836
837# This function checks the first line of file extension "scd" to resolve
838# detection between scdoc and SuperCollider
839export def FTscd()
840 if getline(1) =~# '\%^\S\+(\d[0-9A-Za-z]*)\%(\s\+\"[^"]*\"\%(\s\+\"[^"]*\"\)\=\)\=$'
841 setf scdoc
842 else
843 setf supercollider
844 endif
845enddef
846
Bram Moolenaara2baa732022-02-04 16:09:54 +0000847# If the file has an extension of 't' and is in a directory 't' or 'xt' then
848# it is almost certainly a Perl test file.
849# If the first line starts with '#' and contains 'perl' it's probably a Perl
850# file.
851# (Slow test) If a file contains a 'use' statement then it is almost certainly
852# a Perl file.
853export def FTperl(): number
854 var dirname = expand("%:p:h:t")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100855 if expand("%:e") == 't' && (dirname == 't' || dirname == 'xt')
856 setf perl
857 return 1
858 endif
859 if getline(1)[0] == '#' && getline(1) =~ 'perl'
860 setf perl
861 return 1
862 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000863 var save_cursor = getpos('.')
864 call cursor(1, 1)
Bram Moolenaare5b78972022-02-05 19:50:34 +0000865 var has_use = search('^use\s\s*\k', 'c', 30) > 0
Bram Moolenaarf0b03c42017-12-17 17:17:07 +0100866 call setpos('.', save_cursor)
867 if has_use
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100868 setf perl
869 return 1
870 endif
871 return 0
Bram Moolenaara2baa732022-02-04 16:09:54 +0000872enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100873
Bram Moolenaarcdbfc6d2022-06-30 16:25:21 +0100874# LambdaProlog and Standard ML signature files
875export def FTsig()
876 if exists("g:filetype_sig")
877 exe "setf " .. g:filetype_sig
878 return
879 endif
880
881 var lprolog_comment = '^\s*\%(/\*\|%\)'
882 var lprolog_keyword = '^\s*sig\s\+\a'
883 var sml_comment = '^\s*(\*'
884 var sml_keyword = '^\s*\%(signature\|structure\)\s\+\a'
885
886 var line = getline(nextnonblank(1))
887
888 if line =~ lprolog_comment || line =~# lprolog_keyword
889 setf lprolog
890 elseif line =~ sml_comment || line =~# sml_keyword
891 setf sml
892 endif
893enddef
894
Bram Moolenaarbe807d52022-09-01 15:01:25 +0100895# This function checks the first 100 lines of files matching "*.sil" to
896# resolve detection between Swift Intermediate Language and SILE.
897export def FTsil()
898 for lnum in range(1, [line('$'), 100]->min())
899 var line: string = getline(lnum)
900 if line =~ '^\s*[\\%]'
901 setf sile
902 return
903 elseif line =~ '^\s*\S'
904 setf sil
905 return
906 endif
907 endfor
908 # no clue, default to "sil"
909 setf sil
910enddef
911
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100912export def FTsys()
KnoP-01f420ff22022-04-13 20:46:21 +0100913 if exists("g:filetype_sys")
914 exe "setf " .. g:filetype_sys
915 elseif IsRapid()
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100916 setf rapid
917 else
918 setf bat
919 endif
920enddef
921
Bram Moolenaara2baa732022-02-04 16:09:54 +0000922# Choose context, plaintex, or tex (LaTeX) based on these rules:
923# 1. Check the first line of the file for "%&<format>".
924# 2. Check the first 1000 non-comment lines for LaTeX or ConTeXt keywords.
925# 3. Default to "plain" or to g:tex_flavor, can be set in user's vimrc.
926export def FTtex()
927 var firstline = getline(1)
928 var format: string
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100929 if firstline =~ '^%&\s*\a\+'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000930 format = tolower(matchstr(firstline, '\a\+'))
931 format = substitute(format, 'pdf', '', '')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100932 if format == 'tex'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000933 format = 'latex'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100934 elseif format == 'plaintex'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000935 format = 'plain'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100936 endif
937 elseif expand('%') =~ 'tex/context/.*/.*.tex'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000938 format = 'context'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100939 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000940 # Default value, may be changed later:
941 format = exists("g:tex_flavor") ? g:tex_flavor : 'plain'
942 # Save position, go to the top of the file, find first non-comment line.
943 var save_cursor = getpos('.')
944 call cursor(1, 1)
945 var firstNC = search('^\s*[^[:space:]%]', 'c', 1000)
Bram Moolenaare5b78972022-02-05 19:50:34 +0000946 if firstNC > 0
947 # Check the next thousand lines for a LaTeX or ConTeXt keyword.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000948 var lpat = 'documentclass\>\|usepackage\>\|begin{\|newcommand\>\|renewcommand\>'
949 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\>'
950 var kwline = search('^\s*\\\%(' .. lpat .. '\)\|^\s*\\\(' .. cpat .. '\)',
951 'cnp', firstNC + 1000)
952 if kwline == 1 # lpat matched
953 format = 'latex'
954 elseif kwline == 2 # cpat matched
955 format = 'context'
956 endif # If neither matched, keep default set above.
957 # let lline = search('^\s*\\\%(' . lpat . '\)', 'cn', firstNC + 1000)
958 # let cline = search('^\s*\\\%(' . cpat . '\)', 'cn', firstNC + 1000)
959 # if cline > 0
960 # let format = 'context'
961 # endif
962 # if lline > 0 && (cline == 0 || cline > lline)
963 # let format = 'tex'
964 # endif
965 endif # firstNC
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100966 call setpos('.', save_cursor)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000967 endif # firstline =~ '^%&\s*\a\+'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100968
Bram Moolenaara2baa732022-02-04 16:09:54 +0000969 # Translation from formats to file types. TODO: add AMSTeX, RevTex, others?
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100970 if format == 'plain'
971 setf plaintex
972 elseif format == 'context'
973 setf context
Bram Moolenaara2baa732022-02-04 16:09:54 +0000974 else # probably LaTeX
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100975 setf tex
976 endif
977 return
Bram Moolenaara2baa732022-02-04 16:09:54 +0000978enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100979
Bram Moolenaara2baa732022-02-04 16:09:54 +0000980export def FTxml()
981 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100982 while n < 100 && n <= line("$")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000983 var line = getline(n)
984 # DocBook 4 or DocBook 5.
985 var is_docbook4 = line =~ '<!DOCTYPE.*DocBook'
986 var is_docbook5 = line =~ ' xmlns="http://docbook.org/ns/docbook"'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100987 if is_docbook4 || is_docbook5
Bram Moolenaara2baa732022-02-04 16:09:54 +0000988 b:docbk_type = "xml"
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100989 if is_docbook5
Bram Moolenaara2baa732022-02-04 16:09:54 +0000990 b:docbk_ver = 5
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100991 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000992 b:docbk_ver = 4
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100993 endif
994 setf docbk
995 return
996 endif
997 if line =~ 'xmlns:xbl="http://www.mozilla.org/xbl"'
998 setf xbl
999 return
1000 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001001 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001002 endwhile
1003 setf xml
Bram Moolenaara2baa732022-02-04 16:09:54 +00001004enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001005
Bram Moolenaara2baa732022-02-04 16:09:54 +00001006export def FTy()
1007 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +01001008 while n < 100 && n <= line("$")
Bram Moolenaara2baa732022-02-04 16:09:54 +00001009 var line = getline(n)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001010 if line =~ '^\s*%'
1011 setf yacc
1012 return
1013 endif
1014 if getline(n) =~ '^\s*\(#\|class\>\)' && getline(n) !~ '^\s*#\s*include'
1015 setf racc
1016 return
1017 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001018 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001019 endwhile
1020 setf yacc
Bram Moolenaara2baa732022-02-04 16:09:54 +00001021enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001022
Bram Moolenaara2baa732022-02-04 16:09:54 +00001023export def Redif()
1024 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001025 while lnum <= 5 && lnum < line('$')
1026 if getline(lnum) =~ "^\ctemplate-type:"
1027 setf redif
1028 return
1029 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001030 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001031 endwhile
Bram Moolenaara2baa732022-02-04 16:09:54 +00001032enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001033
Bram Moolenaara2baa732022-02-04 16:09:54 +00001034# This function is called for all files under */debian/patches/*, make sure not
1035# to non-dep3patch files, such as README and other text files.
1036export def Dep3patch()
James McCoy647ab4c2021-12-17 20:52:57 +00001037 if expand('%:t') ==# 'series'
1038 return
1039 endif
1040
1041 for ln in getline(1, 100)
1042 if ln =~# '^\%(Description\|Subject\|Origin\|Bug\|Forwarded\|Author\|From\|Reviewed-by\|Acked-by\|Last-Updated\|Applied-Upstream\):'
1043 setf dep3patch
1044 return
1045 elseif ln =~# '^---'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001046 # end of headers found. stop processing
James McCoy647ab4c2021-12-17 20:52:57 +00001047 return
1048 endif
1049 endfor
Bram Moolenaara2baa732022-02-04 16:09:54 +00001050enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001051
Bram Moolenaara2baa732022-02-04 16:09:54 +00001052# This function checks the first 15 lines for appearance of 'FoamFile'
1053# and then 'object' in a following line.
1054# In that case, it's probably an OpenFOAM file
1055export def FTfoam()
1056 var ffile = 0
1057 var lnum = 1
Elwardi2284f6c2022-01-11 18:14:23 +00001058 while lnum <= 15
1059 if getline(lnum) =~# '^FoamFile'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001060 ffile = 1
Elwardi2284f6c2022-01-11 18:14:23 +00001061 elseif ffile == 1 && getline(lnum) =~# '^\s*object'
1062 setf foam
1063 return
1064 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001065 lnum += 1
Elwardi2284f6c2022-01-11 18:14:23 +00001066 endwhile
Bram Moolenaara2baa732022-02-04 16:09:54 +00001067enddef
Elwardi2284f6c2022-01-11 18:14:23 +00001068
Bram Moolenaara2baa732022-02-04 16:09:54 +00001069# Determine if a *.tf file is TF mud client or terraform
1070export def FTtf()
1071 var numberOfLines = line('$')
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001072 for i in range(1, numberOfLines)
Bram Moolenaara2baa732022-02-04 16:09:54 +00001073 var currentLine = trim(getline(i))
1074 var firstCharacter = currentLine[0]
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001075 if firstCharacter !=? ";" && firstCharacter !=? "/" && firstCharacter !=? ""
1076 setf terraform
1077 return
1078 endif
1079 endfor
1080 setf tf
Bram Moolenaara2baa732022-02-04 16:09:54 +00001081enddef
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001082
KnoP-0193c7a452022-04-16 21:14:04 +01001083var ft_krl_header = '\&\w+'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001084# Determine if a *.src file is Kuka Robot Language
1085export def FTsrc()
KnoP-0193c7a452022-04-16 21:14:04 +01001086 var ft_krl_def_or_deffct = '%(global\s+)?def%(fct)?>'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001087 if exists("g:filetype_src")
1088 exe "setf " .. g:filetype_src
KnoP-0193c7a452022-04-16 21:14:04 +01001089 elseif getline(nextnonblank(1)) =~? '\v^\s*%(' .. ft_krl_header .. '|' .. ft_krl_def_or_deffct .. ')'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001090 setf krl
1091 endif
1092enddef
1093
1094# Determine if a *.dat file is Kuka Robot Language
1095export def FTdat()
KnoP-0193c7a452022-04-16 21:14:04 +01001096 var ft_krl_defdat = 'defdat>'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001097 if exists("g:filetype_dat")
1098 exe "setf " .. g:filetype_dat
KnoP-0193c7a452022-04-16 21:14:04 +01001099 elseif getline(nextnonblank(1)) =~? '\v^\s*%(' .. ft_krl_header .. '|' .. ft_krl_defdat .. ')'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001100 setf krl
1101 endif
1102enddef
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001103
Doug Kearns4ac8e792022-10-17 13:32:17 +01001104export def FTlsl()
1105 if exists("g:filetype_lsl")
1106 exe "setf " .. g:filetype_lsl
1107 endif
1108
1109 var line = getline(nextnonblank(1))
1110 if line =~ '^\s*%' || line =~# ':\s*trait\s*$'
1111 setf larch
1112 else
1113 setf lsl
1114 endif
1115enddef
1116
Gaetan Lepage4ce1bda2023-05-10 22:01:55 +01001117export def FTtyp()
1118 if exists("g:filetype_typ")
1119 exe "setf " .. g:filetype_typ
1120 return
1121 endif
1122
1123 # Look for SQL type definition syntax
1124 for line in getline(1, 200)
1125 # SQL type files may define the casing
1126 if line =~ '^CASE\s\==\s\=\(SAME\|LOWER\|UPPER\|OPPOSITE\)$'
1127 setf sql
1128 return
1129 endif
1130
1131 # SQL type files may define some types as follows
1132 if line =~ '^TYPE\s.*$'
1133 setf sql
1134 return
1135 endif
1136 endfor
1137
1138 # Otherwise, affect the typst filetype
1139 setf typst
1140enddef
1141
Turiiya80406c22023-04-22 21:38:47 +01001142# Set the filetype of a *.v file to Verilog, V or Cog based on the first 200
1143# lines.
1144export def FTv()
1145 if did_filetype()
1146 # ":setf" will do nothing, bail out early
1147 return
1148 endif
1149
1150 for line in getline(1, 200)
1151 if line[0] =~ '^\s*/'
1152 # skip comment line
1153 continue
1154 endif
1155
1156 # Verilog: line ends with ';' followed by an optional variable number of
1157 # spaces and an optional start of a comment.
1158 # Example: " b <= a + 1; // Add 1".
1159 if line =~ ';\(\s*\)\?\(/.*\)\?$'
1160 setf verilog
1161 return
1162 endif
1163
1164 # Coq: line ends with a '.' followed by an optional variable number of
1165 # spaces and an optional start of a comment.
1166 # Example: "Definition x := 10. (*".
1167 if line =~ '\.\(\s*\)\?\((\*.*\)\?$'
1168 setf coq
1169 return
1170 endif
1171 endfor
1172
1173 # No line matched, fall back to "v".
1174 setf v
1175enddef
1176
Bram Moolenaara2baa732022-02-04 16:09:54 +00001177# Uncomment this line to check for compilation errors early
Bram Moolenaar3e79c972022-02-04 19:48:06 +00001178# defcompile