blob: 2423fd4469a94fb1371d231aca30b4f4b31147de [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
Doug Kearnsf97f6bb2023-08-27 18:44:09 +020065var ft_visual_basic_content = '\c^\s*\%(Attribute\s\+VB_Name\|Begin\s\+\%(VB\.\|{\%(\x\+-\)\+\x\+}\)\)'
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
Doug Kearnsf97f6bb2023-08-27 18:44:09 +0200149 var line1 = getline(1)
Doug Kearnse06afb72023-08-29 22:21:35 +0200150 if line1 =~ '^#!.*\<\%(rexx\|regina\)\>'
Bram Moolenaar8b5901e2022-06-29 14:39:12 +0100151 setf rexx
Doug Kearnse06afb72023-08-29 22:21:35 +0200152 return
Doug Kearnsf97f6bb2023-08-27 18:44:09 +0200153 elseif line1 == 'VERSION 1.0 CLASS'
Bram Moolenaar8b5901e2022-06-29 14:39:12 +0100154 setf vb
Doug Kearnse06afb72023-08-29 22:21:35 +0200155 return
156 endif
157
158 var nonblank1 = getline(nextnonblank(1))
159 if nonblank1 =~ '^\v%(\%|\\)'
160 setf tex
161 elseif nonblank1 =~ '^\s*\%(/\*\|::\w\)'
162 setf rexx
Bram Moolenaar8b5901e2022-06-29 14:39:12 +0100163 else
164 setf st
165 endif
166enddef
167
Bram Moolenaara2baa732022-02-04 16:09:54 +0000168export def FTlpc()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100169 if exists("g:lpc_syntax_for_c")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000170 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100171 while lnum <= 12
172 if getline(lnum) =~# '^\(//\|inherit\|private\|protected\|nosave\|string\|object\|mapping\|mixed\)'
173 setf lpc
174 return
175 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000176 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100177 endwhile
178 endif
179 setf c
Bram Moolenaara2baa732022-02-04 16:09:54 +0000180enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100181
Bram Moolenaara2baa732022-02-04 16:09:54 +0000182export def FTheader()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100183 if match(getline(1, min([line("$"), 200])), '^@\(interface\|end\|class\)') > -1
184 if exists("g:c_syntax_for_h")
185 setf objc
186 else
187 setf objcpp
188 endif
189 elseif exists("g:c_syntax_for_h")
190 setf c
191 elseif exists("g:ch_syntax_for_h")
192 setf ch
193 else
194 setf cpp
195 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000196enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100197
Bram Moolenaara2baa732022-02-04 16:09:54 +0000198# This function checks if one of the first ten lines start with a '@'. In
199# that case it is probably a change file.
200# If the first line starts with # or ! it's probably a ch file.
201# If a line has "main", "include", "//" or "/*" it's probably ch.
202# Otherwise CHILL is assumed.
203export def FTchange()
204 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100205 while lnum <= 10
206 if getline(lnum)[0] == '@'
207 setf change
208 return
209 endif
210 if lnum == 1 && (getline(1)[0] == '#' || getline(1)[0] == '!')
211 setf ch
212 return
213 endif
214 if getline(lnum) =~ "MODULE"
215 setf chill
216 return
217 endif
218 if getline(lnum) =~ 'main\s*(\|#\s*include\|//'
219 setf ch
220 return
221 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000222 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100223 endwhile
224 setf chill
Bram Moolenaara2baa732022-02-04 16:09:54 +0000225enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100226
Bram Moolenaara2baa732022-02-04 16:09:54 +0000227export def FTent()
228 # This function checks for valid cl syntax in the first five lines.
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100229 # Look for either an opening comment, '#', or a block start, '{'.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000230 # If not found, assume SGML.
231 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100232 while lnum < 6
Bram Moolenaara2baa732022-02-04 16:09:54 +0000233 var line = getline(lnum)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100234 if line =~ '^\s*[#{]'
235 setf cl
236 return
237 elseif line !~ '^\s*$'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000238 # Not a blank line, not a comment, and not a block start,
239 # so doesn't look like valid cl code.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100240 break
241 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000242 lnum += 1
Bram Moolenaarc12dc472022-03-05 13:45:56 +0000243 endwhile
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100244 setf dtd
Bram Moolenaara2baa732022-02-04 16:09:54 +0000245enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100246
Bram Moolenaara2baa732022-02-04 16:09:54 +0000247export def ExCheck()
248 var lines = getline(1, min([line("$"), 100]))
Austin Gatlinf3caeb62021-06-26 12:02:55 +0200249 if exists('g:filetype_euphoria')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000250 exe 'setf ' .. g:filetype_euphoria
Austin Gatlinf3caeb62021-06-26 12:02:55 +0200251 elseif match(lines, '^--\|^ifdef\>\|^include\>') > -1
252 setf euphoria3
253 else
254 setf elixir
255 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000256enddef
Austin Gatlinf3caeb62021-06-26 12:02:55 +0200257
Bram Moolenaara2baa732022-02-04 16:09:54 +0000258export def EuphoriaCheck()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100259 if exists('g:filetype_euphoria')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000260 exe 'setf ' .. g:filetype_euphoria
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100261 else
262 setf euphoria3
263 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000264enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100265
Bram Moolenaara2baa732022-02-04 16:09:54 +0000266export def DtraceCheck()
=?UTF-8?q?Teubel=20Gy=C3=B6rgy?=4d56b972022-02-24 17:59:09 +0000267 if did_filetype()
268 # Filetype was already detected
269 return
270 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000271 var lines = getline(1, min([line("$"), 100]))
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100272 if match(lines, '^module\>\|^import\>') > -1
Bram Moolenaara2baa732022-02-04 16:09:54 +0000273 # D files often start with a module and/or import statement.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100274 setf d
275 elseif match(lines, '^#!\S\+dtrace\|#pragma\s\+D\s\+option\|:\S\{-}:\S\{-}:') > -1
276 setf dtrace
277 else
278 setf d
279 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000280enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100281
Bram Moolenaara2baa732022-02-04 16:09:54 +0000282export def FTe()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100283 if exists('g:filetype_euphoria')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000284 exe 'setf ' .. g:filetype_euphoria
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100285 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000286 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100287 while n < 100 && n <= line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100288 if getline(n) =~ "^\\s*\\(<'\\|'>\\)\\s*$"
289 setf specman
290 return
291 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000292 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100293 endwhile
294 setf eiffel
295 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000296enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100297
Doug Kearns19a3bc32023-08-20 20:51:12 +0200298def IsForth(): bool
299 var first_line = nextnonblank(1)
300
301 # SwiftForth block comment (line is usually filled with '-' or '=') or
302 # OPTIONAL (sometimes precedes the header comment)
303 if getline(first_line) =~? '^\%({\%(\s\|$\)\|OPTIONAL\s\)'
304 return true
305 endif
306
307 var n = first_line
308 while n < 100 && n <= line("$")
309 # Forth comments and colon definitions
310 if getline(n) =~ '^[:(\\] '
311 return true
312 endif
313 n += 1
314 endwhile
315 return false
316enddef
317
318# Distinguish between Forth and Fortran
319export def FTf()
320 if exists("g:filetype_f")
321 exe "setf " .. g:filetype_f
322 elseif IsForth()
323 setf forth
324 else
325 setf fortran
326 endif
327enddef
328
Bram Moolenaara2baa732022-02-04 16:09:54 +0000329export def FTfrm()
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000330 if exists("g:filetype_frm")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000331 exe "setf " .. g:filetype_frm
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000332 return
333 endif
334
Doug Kearnsf97f6bb2023-08-27 18:44:09 +0200335 if getline(1) == "VERSION 5.00"
336 setf vb
337 return
338 endif
339
Bram Moolenaara2baa732022-02-04 16:09:54 +0000340 var lines = getline(1, min([line("$"), 5]))
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000341
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000342 if match(lines, ft_visual_basic_content) > -1
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000343 setf vb
344 else
345 setf form
346 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000347enddef
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000348
Doug Kearns19a3bc32023-08-20 20:51:12 +0200349# Distinguish between Forth and F#
Bram Moolenaara2baa732022-02-04 16:09:54 +0000350export def FTfs()
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000351 if exists("g:filetype_fs")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000352 exe "setf " .. g:filetype_fs
Doug Kearns19a3bc32023-08-20 20:51:12 +0200353 elseif IsForth()
354 setf forth
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000355 else
Johan Kotlinski065088d2023-04-02 20:29:38 +0100356 setf fsharp
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000357 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000358enddef
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000359
Bram Moolenaara2baa732022-02-04 16:09:54 +0000360# Distinguish between HTML, XHTML and Django
361export def FThtml()
362 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100363 while n < 10 && n <= line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100364 if getline(n) =~ '\<DTD\s\+XHTML\s'
365 setf xhtml
366 return
367 endif
368 if getline(n) =~ '{%\s*\(extends\|block\|load\)\>\|{#\s\+'
369 setf htmldjango
370 return
371 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000372 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100373 endwhile
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100374 setf FALLBACK html
Bram Moolenaara2baa732022-02-04 16:09:54 +0000375enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100376
Bram Moolenaara2baa732022-02-04 16:09:54 +0000377# Distinguish between standard IDL and MS-IDL
378export def FTidl()
379 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100380 while n < 50 && n <= line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100381 if getline(n) =~ '^\s*import\s\+"\(unknwn\|objidl\)\.idl"'
382 setf msidl
383 return
384 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000385 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100386 endwhile
387 setf idl
Bram Moolenaara2baa732022-02-04 16:09:54 +0000388enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100389
Bram Moolenaar48c3f4e2022-08-08 15:42:38 +0100390# Distinguish between "default", Prolog and Cproto prototype file.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000391export def ProtoCheck(default: string)
392 # Cproto files have a comment in the first line and a function prototype in
393 # the second line, it always ends in ";". Indent files may also have
394 # comments, thus we can't match comments to see the difference.
395 # IDL files can have a single ';' in the second line, require at least one
396 # chacter before the ';'.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100397 if getline(2) =~ '.;$'
398 setf cpp
399 else
Bram Moolenaar48c3f4e2022-08-08 15:42:38 +0100400 # recognize Prolog by specific text in the first non-empty line
401 # require a blank after the '%' because Perl uses "%list" and "%translate"
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100402 var lnum = getline(nextnonblank(1))
403 if lnum =~ '\<prolog\>' || lnum =~ '^\s*\(%\+\(\s\|$\)\|/\*\)' || lnum =~ ':-'
Bram Moolenaar48c3f4e2022-08-08 15:42:38 +0100404 setf prolog
405 else
406 exe 'setf ' .. default
407 endif
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100408 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000409enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100410
Bram Moolenaara2baa732022-02-04 16:09:54 +0000411export def FTm()
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200412 if exists("g:filetype_m")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000413 exe "setf " .. g:filetype_m
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200414 return
415 endif
416
Bram Moolenaara2baa732022-02-04 16:09:54 +0000417 # excluding end(for|function|if|switch|while) common to Murphi
418 var octave_block_terminators = '\<end\%(_try_catch\|classdef\|enumeration\|events\|methods\|parfor\|properties\)\>'
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200419
Bram Moolenaara2baa732022-02-04 16:09:54 +0000420 var objc_preprocessor = '^\s*#\s*\%(import\|include\|define\|if\|ifn\=def\|undef\|line\|error\|pragma\)\>'
Doug Kearns7329cfa2021-11-26 13:01:41 +0000421
Bram Moolenaara2baa732022-02-04 16:09:54 +0000422 var n = 1
423 var saw_comment = 0 # Whether we've seen a multiline comment leader.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100424 while n < 100
Bram Moolenaara2baa732022-02-04 16:09:54 +0000425 var line = getline(n)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100426 if line =~ '^\s*/\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000427 # /* ... */ is a comment in Objective C and Murphi, so we can't conclude
428 # it's either of them yet, but track this as a hint in case we don't see
429 # anything more definitive.
430 saw_comment = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100431 endif
Doug Kearns7329cfa2021-11-26 13:01:41 +0000432 if line =~ '^\s*//' || line =~ '^\s*@import\>' || line =~ objc_preprocessor
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100433 setf objc
434 return
435 endif
Bram Moolenaarca0627d2021-09-12 17:03:08 +0200436 if line =~ '^\s*\%(#\|%!\)' || line =~ '^\s*unwind_protect\>' ||
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200437 \ line =~ '\%(^\|;\)\s*' .. octave_block_terminators
438 setf octave
439 return
440 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000441 # TODO: could be Matlab or Octave
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100442 if line =~ '^\s*%'
443 setf matlab
444 return
445 endif
446 if line =~ '^\s*(\*'
447 setf mma
448 return
449 endif
450 if line =~ '^\c\s*\(\(type\|var\)\>\|--\)'
451 setf murphi
452 return
453 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000454 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100455 endwhile
456
457 if saw_comment
Bram Moolenaara2baa732022-02-04 16:09:54 +0000458 # We didn't see anything definitive, but this looks like either Objective C
459 # or Murphi based on the comment leader. Assume the former as it is more
460 # common.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100461 setf objc
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100462 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000463 # Default is Matlab
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100464 setf matlab
465 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000466enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100467
Bram Moolenaara2baa732022-02-04 16:09:54 +0000468export def FTmms()
469 var n = 1
Bram Moolenaard7df2792020-01-02 21:34:42 +0100470 while n < 20
Bram Moolenaara2baa732022-02-04 16:09:54 +0000471 var line = getline(n)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100472 if line =~ '^\s*\(%\|//\)' || line =~ '^\*'
473 setf mmix
474 return
475 endif
476 if line =~ '^\s*#'
477 setf make
478 return
479 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000480 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100481 endwhile
482 setf mmix
Bram Moolenaara2baa732022-02-04 16:09:54 +0000483enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100484
Bram Moolenaara2baa732022-02-04 16:09:54 +0000485# This function checks if one of the first five lines start with a dot. In
486# that case it is probably an nroff file: 'filetype' is set and 1 is returned.
487export def FTnroff(): number
488 if getline(1)[0] .. getline(2)[0] .. getline(3)[0]
489 .. getline(4)[0] .. getline(5)[0] =~ '\.'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100490 setf nroff
491 return 1
492 endif
493 return 0
Bram Moolenaara2baa732022-02-04 16:09:54 +0000494enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100495
Bram Moolenaara2baa732022-02-04 16:09:54 +0000496export def FTmm()
497 var n = 1
Bram Moolenaard1caa942020-04-10 22:10:56 +0200498 while n < 20
Bram Moolenaara2baa732022-02-04 16:09:54 +0000499 if getline(n) =~ '^\s*\(#\s*\(include\|import\)\>\|@import\>\|/\*\)'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100500 setf objcpp
501 return
502 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000503 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100504 endwhile
505 setf nroff
Bram Moolenaara2baa732022-02-04 16:09:54 +0000506enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100507
Bram Moolenaarcdbfc6d2022-06-30 16:25:21 +0100508# Returns true if file content looks like LambdaProlog module
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100509def IsLProlog(): bool
Gaetan Lepage4ce1bda2023-05-10 22:01:55 +0100510 # skip apparent comments and blank lines, what looks like
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100511 # LambdaProlog comment may be RAPID header
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100512 var lnum: number = nextnonblank(1)
513 while lnum > 0 && lnum < line('$') && getline(lnum) =~ '^\s*%' # LambdaProlog comment
514 lnum = nextnonblank(lnum + 1)
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100515 endwhile
516 # this pattern must not catch a go.mod file
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100517 return getline(lnum) =~ '\<module\s\+\w\+\s*\.\s*\(%\|$\)'
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100518enddef
519
520# Determine if *.mod is ABB RAPID, LambdaProlog, Modula-2, Modsim III or go.mod
521export def FTmod()
522 if exists("g:filetype_mod")
523 exe "setf " .. g:filetype_mod
Omar El Halabic9fbd252023-05-29 19:59:45 +0100524 elseif expand("<afile>") =~ '\<go.mod$'
525 setf gomod
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100526 elseif IsLProlog()
527 setf lprolog
528 elseif getline(nextnonblank(1)) =~ '\%(\<MODULE\s\+\w\+\s*;\|^\s*(\*\)'
529 setf modula2
530 elseif IsRapid()
531 setf rapid
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100532 else
533 # Nothing recognized, assume modsim3
534 setf modsim3
535 endif
536enddef
537
Bram Moolenaara2baa732022-02-04 16:09:54 +0000538export def FTpl()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100539 if exists("g:filetype_pl")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000540 exe "setf " .. g:filetype_pl
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100541 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000542 # recognize Prolog by specific text in the first non-empty line
543 # require a blank after the '%' because Perl uses "%list" and "%translate"
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100544 var line = getline(nextnonblank(1))
545 if line =~ '\<prolog\>' || line =~ '^\s*\(%\+\(\s\|$\)\|/\*\)' || line =~ ':-'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100546 setf prolog
547 else
548 setf perl
549 endif
550 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000551enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100552
Bram Moolenaara2baa732022-02-04 16:09:54 +0000553export def FTinc()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100554 if exists("g:filetype_inc")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000555 exe "setf " .. g:filetype_inc
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100556 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000557 var lines = getline(1) .. getline(2) .. getline(3)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100558 if lines =~? "perlscript"
559 setf aspperl
560 elseif lines =~ "<%"
561 setf aspvbs
562 elseif lines =~ "<?"
563 setf php
Bram Moolenaara2baa732022-02-04 16:09:54 +0000564 # Pascal supports // comments but they're vary rarely used for file
565 # headers so assume POV-Ray
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000566 elseif lines =~ '^\s*\%({\|(\*\)' || lines =~? ft_pascal_keywords
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100567 setf pascal
Gregory Anders30e212d2022-07-26 21:42:03 +0100568 elseif lines =~# '\<\%(require\|inherit\)\>' || lines =~# '[A-Z][A-Za-z0-9_:${}]*\s\+\%(??\|[?:+]\)\?= '
Gregory Andersfa49eb42022-07-16 17:46:47 +0100569 setf bitbake
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100570 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000571 FTasmsyntax()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100572 if exists("b:asmsyntax")
Gregory Andersfa49eb42022-07-16 17:46:47 +0100573 exe "setf " .. fnameescape(b:asmsyntax)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100574 else
Gregory Andersfa49eb42022-07-16 17:46:47 +0100575 setf pov
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100576 endif
577 endif
578 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000579enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100580
Bram Moolenaara2baa732022-02-04 16:09:54 +0000581export def FTprogress_cweb()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100582 if exists("g:filetype_w")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000583 exe "setf " .. g:filetype_w
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100584 return
585 endif
586 if getline(1) =~ '&ANALYZE' || getline(3) =~ '&GLOBAL-DEFINE'
587 setf progress
588 else
589 setf cweb
590 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000591enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100592
Julien Marrec2e310652023-11-25 15:30:46 +0100593# These include the leading '%' sign
594var ft_swig_keywords = '^\s*%\%(addmethods\|apply\|beginfile\|clear\|constant\|define\|echo\|enddef\|endoffile\|extend\|feature\|fragment\|ignore\|import\|importfile\|include\|includefile\|inline\|insert\|keyword\|module\|name\|namewarn\|native\|newobject\|parms\|pragma\|rename\|template\|typedef\|typemap\|types\|varargs\|warn\)'
595# This is the start/end of a block that is copied literally to the processor file (C/C++)
596var ft_swig_verbatim_block_start = '^\s*%{'
597
598export def FTi()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100599 if exists("g:filetype_i")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000600 exe "setf " .. g:filetype_i
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100601 return
602 endif
Julien Marrec2e310652023-11-25 15:30:46 +0100603 # This function checks for an assembly comment or a SWIG keyword or verbatim block in the first 50 lines.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000604 # If not found, assume Progress.
605 var lnum = 1
Julien Marrec2e310652023-11-25 15:30:46 +0100606 while lnum <= 50 && lnum < line('$')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000607 var line = getline(lnum)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100608 if line =~ '^\s*;' || line =~ '^\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000609 FTasm()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100610 return
Julien Marrec2e310652023-11-25 15:30:46 +0100611 elseif line =~ ft_swig_keywords || line =~ ft_swig_verbatim_block_start
612 setf swig
613 return
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100614 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000615 lnum += 1
Bram Moolenaarc12dc472022-03-05 13:45:56 +0000616 endwhile
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100617 setf progress
Bram Moolenaara2baa732022-02-04 16:09:54 +0000618enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100619
Bram Moolenaara2baa732022-02-04 16:09:54 +0000620var ft_pascal_comments = '^\s*\%({\|(\*\|//\)'
621var ft_pascal_keywords = '^\s*\%(program\|unit\|library\|uses\|begin\|procedure\|function\|const\|type\|var\)\>'
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100622
Bram Moolenaara2baa732022-02-04 16:09:54 +0000623export def FTprogress_pascal()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100624 if exists("g:filetype_p")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000625 exe "setf " .. g:filetype_p
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100626 return
627 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000628 # This function checks for valid Pascal syntax in the first ten lines.
629 # Look for either an opening comment or a program start.
630 # If not found, assume Progress.
631 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100632 while lnum <= 10 && lnum < line('$')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000633 var line = getline(lnum)
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000634 if line =~ ft_pascal_comments || line =~? ft_pascal_keywords
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100635 setf pascal
636 return
637 elseif line !~ '^\s*$' || line =~ '^/\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000638 # Not an empty line: Doesn't look like valid Pascal code.
639 # Or it looks like a Progress /* comment
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100640 break
641 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000642 lnum += 1
Bram Moolenaarc12dc472022-03-05 13:45:56 +0000643 endwhile
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100644 setf progress
Bram Moolenaara2baa732022-02-04 16:09:54 +0000645enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100646
Bram Moolenaara2baa732022-02-04 16:09:54 +0000647export def FTpp()
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100648 if exists("g:filetype_pp")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000649 exe "setf " .. g:filetype_pp
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100650 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000651 var line = getline(nextnonblank(1))
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000652 if line =~ ft_pascal_comments || line =~? ft_pascal_keywords
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100653 setf pascal
654 else
655 setf puppet
656 endif
657 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000658enddef
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100659
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100660# Determine if *.prg is ABB RAPID. Can also be Clipper, FoxPro or eviews
661export def FTprg()
662 if exists("g:filetype_prg")
663 exe "setf " .. g:filetype_prg
664 elseif IsRapid()
665 setf rapid
666 else
667 # Nothing recognized, assume Clipper
668 setf clipper
669 endif
670enddef
671
Bram Moolenaara2baa732022-02-04 16:09:54 +0000672export def FTr()
673 var max = line("$") > 50 ? 50 : line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100674
675 for n in range(1, max)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000676 # Rebol is easy to recognize, check for that first
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100677 if getline(n) =~? '\<REBOL\>'
678 setf rebol
679 return
680 endif
681 endfor
682
683 for n in range(1, max)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000684 # R has # comments
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100685 if getline(n) =~ '^\s*#'
686 setf r
687 return
688 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000689 # Rexx has /* comments */
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100690 if getline(n) =~ '^\s*/\*'
691 setf rexx
692 return
693 endif
694 endfor
695
Bram Moolenaara2baa732022-02-04 16:09:54 +0000696 # Nothing recognized, use user default or assume Rexx
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100697 if exists("g:filetype_r")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000698 exe "setf " .. g:filetype_r
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100699 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000700 # Rexx used to be the default, but R appears to be much more popular.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100701 setf r
702 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000703enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100704
Bram Moolenaara2baa732022-02-04 16:09:54 +0000705export def McSetf()
706 # Rely on the file to start with a comment.
707 # MS message text files use ';', Sendmail files use '#' or 'dnl'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100708 for lnum in range(1, min([line("$"), 20]))
Bram Moolenaara2baa732022-02-04 16:09:54 +0000709 var line = getline(lnum)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100710 if line =~ '^\s*\(#\|dnl\)'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000711 setf m4 # Sendmail .mc file
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100712 return
713 elseif line =~ '^\s*;'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000714 setf msmessages # MS Message text file
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100715 return
716 endif
717 endfor
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100718 setf m4 # Default: Sendmail .mc file
Bram Moolenaara2baa732022-02-04 16:09:54 +0000719enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100720
Bram Moolenaara2baa732022-02-04 16:09:54 +0000721# Called from filetype.vim and scripts.vim.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100722# When "setft" is passed and false then the 'filetype' option is not set.
723export def SetFileTypeSH(name: string, setft = true): string
724 if setft && did_filetype()
Bram Moolenaara2baa732022-02-04 16:09:54 +0000725 # Filetype was already detected
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100726 return ''
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100727 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100728 if setft && expand("<amatch>") =~ g:ft_ignore_pat
729 return ''
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100730 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000731 if name =~ '\<csh\>'
732 # Some .sh scripts contain #!/bin/csh.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100733 return SetFileTypeShell("csh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000734 elseif name =~ '\<tcsh\>'
735 # Some .sh scripts contain #!/bin/tcsh.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100736 return SetFileTypeShell("tcsh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000737 elseif name =~ '\<zsh\>'
738 # Some .sh scripts contain #!/bin/zsh.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100739 return SetFileTypeShell("zsh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000740 elseif name =~ '\<ksh\>'
741 b:is_kornshell = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100742 if exists("b:is_bash")
743 unlet b:is_bash
744 endif
745 if exists("b:is_sh")
746 unlet b:is_sh
747 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000748 elseif exists("g:bash_is_sh") || name =~ '\<bash\>' || name =~ '\<bash2\>'
749 b:is_bash = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100750 if exists("b:is_kornshell")
751 unlet b:is_kornshell
752 endif
753 if exists("b:is_sh")
754 unlet b:is_sh
755 endif
Eisuke Kawashima24482fb2022-11-24 10:58:10 +0000756 elseif name =~ '\<sh\>' || name =~ '\<dash\>'
757 # Ubuntu links "sh" to "dash", thus it is expected to work the same way
Bram Moolenaara2baa732022-02-04 16:09:54 +0000758 b:is_sh = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100759 if exists("b:is_kornshell")
760 unlet b:is_kornshell
761 endif
762 if exists("b:is_bash")
763 unlet b:is_bash
764 endif
765 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100766
767 return SetFileTypeShell("sh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000768enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100769
Bram Moolenaara2baa732022-02-04 16:09:54 +0000770# For shell-like file types, check for an "exec" command hidden in a comment,
771# as used for Tcl.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100772# When "setft" is passed and false then the 'filetype' option is not set.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000773# Also called from scripts.vim, thus can't be local to this script.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100774export def SetFileTypeShell(name: string, setft = true): string
775 if setft && did_filetype()
Bram Moolenaara2baa732022-02-04 16:09:54 +0000776 # Filetype was already detected
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100777 return ''
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100778 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100779 if setft && expand("<amatch>") =~ g:ft_ignore_pat
780 return ''
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100781 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100782
783 var lnum = 2
784 while lnum < 20 && lnum < line("$") && getline(lnum) =~ '^\s*\(#\|$\)'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000785 # Skip empty and comment lines.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100786 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100787 endwhile
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100788 if lnum < line("$") && getline(lnum) =~ '\s*exec\s' && getline(lnum - 1) =~ '^\s*#.*\\$'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000789 # Found an "exec" line after a comment with continuation
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100790 var n = substitute(getline(lnum), '\s*exec\s\+\([^ ]*/\)\=', '', '')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100791 if n =~ '\<tclsh\|\<wish'
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100792 if setft
793 setf tcl
794 endif
795 return 'tcl'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100796 endif
797 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100798
799 if setft
800 exe "setf " .. name
801 endif
802 return name
Bram Moolenaara2baa732022-02-04 16:09:54 +0000803enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100804
Bram Moolenaara2baa732022-02-04 16:09:54 +0000805export def CSH()
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100806 if did_filetype()
Bram Moolenaara2baa732022-02-04 16:09:54 +0000807 # Filetype was already detected
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100808 return
809 endif
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100810 if exists("g:filetype_csh")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000811 SetFileTypeShell(g:filetype_csh)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100812 elseif &shell =~ "tcsh"
Bram Moolenaara2baa732022-02-04 16:09:54 +0000813 SetFileTypeShell("tcsh")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100814 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000815 SetFileTypeShell("csh")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100816 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000817enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100818
Bram Moolenaara2baa732022-02-04 16:09:54 +0000819var ft_rules_udev_rules_pattern = '^\s*\cudev_rules\s*=\s*"\([^"]\{-1,}\)/*".*'
820export def FTRules()
821 var path = expand('<amatch>:p')
Bram Moolenaaraa9675a2020-08-17 21:57:09 +0200822 if path =~ '/\(etc/udev/\%(rules\.d/\)\=.*\.rules\|\%(usr/\)\=lib/udev/\%(rules\.d/\)\=.*\.rules\)$'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100823 setf udevrules
824 return
825 endif
826 if path =~ '^/etc/ufw/'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000827 setf conf # Better than hog
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100828 return
829 endif
830 if path =~ '^/\(etc\|usr/share\)/polkit-1/rules\.d'
831 setf javascript
832 return
833 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000834 var config_lines: list<string>
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100835 try
Bram Moolenaara2baa732022-02-04 16:09:54 +0000836 config_lines = readfile('/etc/udev/udev.conf')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100837 catch /^Vim\%((\a\+)\)\=:E484/
838 setf hog
839 return
840 endtry
Bram Moolenaara2baa732022-02-04 16:09:54 +0000841 var dir = expand('<amatch>:p:h')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100842 for line in config_lines
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000843 if line =~ ft_rules_udev_rules_pattern
844 var udev_rules = substitute(line, ft_rules_udev_rules_pattern, '\1', "")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100845 if dir == udev_rules
846 setf udevrules
847 endif
848 break
849 endif
850 endfor
851 setf hog
Bram Moolenaara2baa732022-02-04 16:09:54 +0000852enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100853
Bram Moolenaara2baa732022-02-04 16:09:54 +0000854export def SQL()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100855 if exists("g:filetype_sql")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000856 exe "setf " .. g:filetype_sql
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100857 else
858 setf sql
859 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000860enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100861
ranjithshegde8cac20e2022-04-13 15:29:21 +0100862# This function checks the first 25 lines of file extension "sc" to resolve
Chris Kipp70ef3f52022-12-14 16:42:15 +0000863# detection between scala and SuperCollider.
864# NOTE: We don't check for 'Class : Method', as this can easily be confused
865# with valid Scala like `val x : Int = 3`. So we instead only rely on
866# checks that can't be confused.
ranjithshegde8cac20e2022-04-13 15:29:21 +0100867export def FTsc()
868 for lnum in range(1, min([line("$"), 25]))
Chris Kipp70ef3f52022-12-14 16:42:15 +0000869 if getline(lnum) =~# 'var\s<\|classvar\s<\|\^this.*\||\w\+|\|+\s\w*\s{\|\*ar\s'
ranjithshegde8cac20e2022-04-13 15:29:21 +0100870 setf supercollider
871 return
872 endif
873 endfor
874 setf scala
875enddef
876
877# This function checks the first line of file extension "scd" to resolve
878# detection between scdoc and SuperCollider
879export def FTscd()
880 if getline(1) =~# '\%^\S\+(\d[0-9A-Za-z]*)\%(\s\+\"[^"]*\"\%(\s\+\"[^"]*\"\)\=\)\=$'
881 setf scdoc
882 else
883 setf supercollider
884 endif
885enddef
886
Bram Moolenaara2baa732022-02-04 16:09:54 +0000887# If the file has an extension of 't' and is in a directory 't' or 'xt' then
888# it is almost certainly a Perl test file.
889# If the first line starts with '#' and contains 'perl' it's probably a Perl
890# file.
891# (Slow test) If a file contains a 'use' statement then it is almost certainly
892# a Perl file.
893export def FTperl(): number
894 var dirname = expand("%:p:h:t")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100895 if expand("%:e") == 't' && (dirname == 't' || dirname == 'xt')
896 setf perl
897 return 1
898 endif
899 if getline(1)[0] == '#' && getline(1) =~ 'perl'
900 setf perl
901 return 1
902 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000903 var save_cursor = getpos('.')
904 call cursor(1, 1)
Bram Moolenaare5b78972022-02-05 19:50:34 +0000905 var has_use = search('^use\s\s*\k', 'c', 30) > 0
Bram Moolenaarf0b03c42017-12-17 17:17:07 +0100906 call setpos('.', save_cursor)
907 if has_use
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100908 setf perl
909 return 1
910 endif
911 return 0
Bram Moolenaara2baa732022-02-04 16:09:54 +0000912enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100913
Bram Moolenaarcdbfc6d2022-06-30 16:25:21 +0100914# LambdaProlog and Standard ML signature files
915export def FTsig()
916 if exists("g:filetype_sig")
917 exe "setf " .. g:filetype_sig
918 return
919 endif
920
921 var lprolog_comment = '^\s*\%(/\*\|%\)'
922 var lprolog_keyword = '^\s*sig\s\+\a'
923 var sml_comment = '^\s*(\*'
924 var sml_keyword = '^\s*\%(signature\|structure\)\s\+\a'
925
926 var line = getline(nextnonblank(1))
927
928 if line =~ lprolog_comment || line =~# lprolog_keyword
929 setf lprolog
930 elseif line =~ sml_comment || line =~# sml_keyword
931 setf sml
932 endif
933enddef
934
Bram Moolenaarbe807d52022-09-01 15:01:25 +0100935# This function checks the first 100 lines of files matching "*.sil" to
936# resolve detection between Swift Intermediate Language and SILE.
937export def FTsil()
938 for lnum in range(1, [line('$'), 100]->min())
939 var line: string = getline(lnum)
940 if line =~ '^\s*[\\%]'
941 setf sile
942 return
943 elseif line =~ '^\s*\S'
944 setf sil
945 return
946 endif
947 endfor
948 # no clue, default to "sil"
949 setf sil
950enddef
951
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100952export def FTsys()
KnoP-01f420ff22022-04-13 20:46:21 +0100953 if exists("g:filetype_sys")
954 exe "setf " .. g:filetype_sys
955 elseif IsRapid()
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100956 setf rapid
957 else
958 setf bat
959 endif
960enddef
961
Bram Moolenaara2baa732022-02-04 16:09:54 +0000962# Choose context, plaintex, or tex (LaTeX) based on these rules:
963# 1. Check the first line of the file for "%&<format>".
964# 2. Check the first 1000 non-comment lines for LaTeX or ConTeXt keywords.
965# 3. Default to "plain" or to g:tex_flavor, can be set in user's vimrc.
966export def FTtex()
967 var firstline = getline(1)
968 var format: string
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100969 if firstline =~ '^%&\s*\a\+'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000970 format = tolower(matchstr(firstline, '\a\+'))
971 format = substitute(format, 'pdf', '', '')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100972 if format == 'tex'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000973 format = 'latex'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100974 elseif format == 'plaintex'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000975 format = 'plain'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100976 endif
977 elseif expand('%') =~ 'tex/context/.*/.*.tex'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000978 format = 'context'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100979 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000980 # Default value, may be changed later:
981 format = exists("g:tex_flavor") ? g:tex_flavor : 'plain'
982 # Save position, go to the top of the file, find first non-comment line.
983 var save_cursor = getpos('.')
984 call cursor(1, 1)
985 var firstNC = search('^\s*[^[:space:]%]', 'c', 1000)
Bram Moolenaare5b78972022-02-05 19:50:34 +0000986 if firstNC > 0
987 # Check the next thousand lines for a LaTeX or ConTeXt keyword.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000988 var lpat = 'documentclass\>\|usepackage\>\|begin{\|newcommand\>\|renewcommand\>'
989 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\>'
990 var kwline = search('^\s*\\\%(' .. lpat .. '\)\|^\s*\\\(' .. cpat .. '\)',
991 'cnp', firstNC + 1000)
992 if kwline == 1 # lpat matched
993 format = 'latex'
994 elseif kwline == 2 # cpat matched
995 format = 'context'
996 endif # If neither matched, keep default set above.
997 # let lline = search('^\s*\\\%(' . lpat . '\)', 'cn', firstNC + 1000)
998 # let cline = search('^\s*\\\%(' . cpat . '\)', 'cn', firstNC + 1000)
999 # if cline > 0
1000 # let format = 'context'
1001 # endif
1002 # if lline > 0 && (cline == 0 || cline > lline)
1003 # let format = 'tex'
1004 # endif
1005 endif # firstNC
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001006 call setpos('.', save_cursor)
Bram Moolenaara2baa732022-02-04 16:09:54 +00001007 endif # firstline =~ '^%&\s*\a\+'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001008
Bram Moolenaara2baa732022-02-04 16:09:54 +00001009 # Translation from formats to file types. TODO: add AMSTeX, RevTex, others?
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001010 if format == 'plain'
1011 setf plaintex
1012 elseif format == 'context'
1013 setf context
Bram Moolenaara2baa732022-02-04 16:09:54 +00001014 else # probably LaTeX
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001015 setf tex
1016 endif
1017 return
Bram Moolenaara2baa732022-02-04 16:09:54 +00001018enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001019
Bram Moolenaara2baa732022-02-04 16:09:54 +00001020export def FTxml()
1021 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +01001022 while n < 100 && n <= line("$")
Bram Moolenaara2baa732022-02-04 16:09:54 +00001023 var line = getline(n)
1024 # DocBook 4 or DocBook 5.
1025 var is_docbook4 = line =~ '<!DOCTYPE.*DocBook'
1026 var is_docbook5 = line =~ ' xmlns="http://docbook.org/ns/docbook"'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001027 if is_docbook4 || is_docbook5
Bram Moolenaara2baa732022-02-04 16:09:54 +00001028 b:docbk_type = "xml"
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001029 if is_docbook5
Bram Moolenaara2baa732022-02-04 16:09:54 +00001030 b:docbk_ver = 5
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001031 else
Bram Moolenaara2baa732022-02-04 16:09:54 +00001032 b:docbk_ver = 4
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001033 endif
1034 setf docbk
1035 return
1036 endif
1037 if line =~ 'xmlns:xbl="http://www.mozilla.org/xbl"'
1038 setf xbl
1039 return
1040 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001041 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001042 endwhile
1043 setf xml
Bram Moolenaara2baa732022-02-04 16:09:54 +00001044enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001045
Bram Moolenaara2baa732022-02-04 16:09:54 +00001046export def FTy()
1047 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +01001048 while n < 100 && n <= line("$")
Bram Moolenaara2baa732022-02-04 16:09:54 +00001049 var line = getline(n)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001050 if line =~ '^\s*%'
1051 setf yacc
1052 return
1053 endif
1054 if getline(n) =~ '^\s*\(#\|class\>\)' && getline(n) !~ '^\s*#\s*include'
1055 setf racc
1056 return
1057 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001058 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001059 endwhile
1060 setf yacc
Bram Moolenaara2baa732022-02-04 16:09:54 +00001061enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001062
Bram Moolenaara2baa732022-02-04 16:09:54 +00001063export def Redif()
1064 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001065 while lnum <= 5 && lnum < line('$')
1066 if getline(lnum) =~ "^\ctemplate-type:"
1067 setf redif
1068 return
1069 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001070 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001071 endwhile
Bram Moolenaara2baa732022-02-04 16:09:54 +00001072enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001073
Bram Moolenaara2baa732022-02-04 16:09:54 +00001074# This function is called for all files under */debian/patches/*, make sure not
1075# to non-dep3patch files, such as README and other text files.
1076export def Dep3patch()
James McCoy647ab4c2021-12-17 20:52:57 +00001077 if expand('%:t') ==# 'series'
1078 return
1079 endif
1080
1081 for ln in getline(1, 100)
1082 if ln =~# '^\%(Description\|Subject\|Origin\|Bug\|Forwarded\|Author\|From\|Reviewed-by\|Acked-by\|Last-Updated\|Applied-Upstream\):'
1083 setf dep3patch
1084 return
1085 elseif ln =~# '^---'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001086 # end of headers found. stop processing
James McCoy647ab4c2021-12-17 20:52:57 +00001087 return
1088 endif
1089 endfor
Bram Moolenaara2baa732022-02-04 16:09:54 +00001090enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001091
Bram Moolenaara2baa732022-02-04 16:09:54 +00001092# This function checks the first 15 lines for appearance of 'FoamFile'
1093# and then 'object' in a following line.
1094# In that case, it's probably an OpenFOAM file
1095export def FTfoam()
1096 var ffile = 0
1097 var lnum = 1
Elwardi2284f6c2022-01-11 18:14:23 +00001098 while lnum <= 15
1099 if getline(lnum) =~# '^FoamFile'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001100 ffile = 1
Elwardi2284f6c2022-01-11 18:14:23 +00001101 elseif ffile == 1 && getline(lnum) =~# '^\s*object'
1102 setf foam
1103 return
1104 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001105 lnum += 1
Elwardi2284f6c2022-01-11 18:14:23 +00001106 endwhile
Bram Moolenaara2baa732022-02-04 16:09:54 +00001107enddef
Elwardi2284f6c2022-01-11 18:14:23 +00001108
Bram Moolenaara2baa732022-02-04 16:09:54 +00001109# Determine if a *.tf file is TF mud client or terraform
1110export def FTtf()
1111 var numberOfLines = line('$')
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001112 for i in range(1, numberOfLines)
Bram Moolenaara2baa732022-02-04 16:09:54 +00001113 var currentLine = trim(getline(i))
1114 var firstCharacter = currentLine[0]
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001115 if firstCharacter !=? ";" && firstCharacter !=? "/" && firstCharacter !=? ""
1116 setf terraform
1117 return
1118 endif
1119 endfor
1120 setf tf
Bram Moolenaara2baa732022-02-04 16:09:54 +00001121enddef
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001122
KnoP-0193c7a452022-04-16 21:14:04 +01001123var ft_krl_header = '\&\w+'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001124# Determine if a *.src file is Kuka Robot Language
1125export def FTsrc()
KnoP-0193c7a452022-04-16 21:14:04 +01001126 var ft_krl_def_or_deffct = '%(global\s+)?def%(fct)?>'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001127 if exists("g:filetype_src")
1128 exe "setf " .. g:filetype_src
KnoP-0193c7a452022-04-16 21:14:04 +01001129 elseif getline(nextnonblank(1)) =~? '\v^\s*%(' .. ft_krl_header .. '|' .. ft_krl_def_or_deffct .. ')'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001130 setf krl
1131 endif
1132enddef
1133
1134# Determine if a *.dat file is Kuka Robot Language
1135export def FTdat()
KnoP-0193c7a452022-04-16 21:14:04 +01001136 var ft_krl_defdat = 'defdat>'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001137 if exists("g:filetype_dat")
1138 exe "setf " .. g:filetype_dat
KnoP-0193c7a452022-04-16 21:14:04 +01001139 elseif getline(nextnonblank(1)) =~? '\v^\s*%(' .. ft_krl_header .. '|' .. ft_krl_defdat .. ')'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001140 setf krl
1141 endif
1142enddef
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001143
Doug Kearns4ac8e792022-10-17 13:32:17 +01001144export def FTlsl()
1145 if exists("g:filetype_lsl")
1146 exe "setf " .. g:filetype_lsl
1147 endif
1148
1149 var line = getline(nextnonblank(1))
1150 if line =~ '^\s*%' || line =~# ':\s*trait\s*$'
1151 setf larch
1152 else
1153 setf lsl
1154 endif
1155enddef
1156
Gaetan Lepage4ce1bda2023-05-10 22:01:55 +01001157export def FTtyp()
1158 if exists("g:filetype_typ")
1159 exe "setf " .. g:filetype_typ
1160 return
1161 endif
1162
1163 # Look for SQL type definition syntax
1164 for line in getline(1, 200)
1165 # SQL type files may define the casing
1166 if line =~ '^CASE\s\==\s\=\(SAME\|LOWER\|UPPER\|OPPOSITE\)$'
1167 setf sql
1168 return
1169 endif
1170
1171 # SQL type files may define some types as follows
1172 if line =~ '^TYPE\s.*$'
1173 setf sql
1174 return
1175 endif
1176 endfor
1177
1178 # Otherwise, affect the typst filetype
1179 setf typst
1180enddef
1181
Turiiya80406c22023-04-22 21:38:47 +01001182# Set the filetype of a *.v file to Verilog, V or Cog based on the first 200
1183# lines.
1184export def FTv()
1185 if did_filetype()
1186 # ":setf" will do nothing, bail out early
1187 return
1188 endif
Christian Brabandt10b4f752024-01-01 19:19:20 +01001189 if exists("g:filetype_v")
1190 exe "setf " .. g:filetype_v
1191 return
1192 endif
Turiiya80406c22023-04-22 21:38:47 +01001193
Christian Brabandt10b4f752024-01-01 19:19:20 +01001194 var in_comment = 0
1195 for lnum in range(1, min([line("$"), 200]))
1196 var line = getline(lnum)
1197 # Skip Verilog and V comments (lines and blocks).
1198 if line =~ '^\s*/\*'
1199 # start comment block
1200 in_comment = 1
1201 endif
1202 if in_comment == 1
1203 if line =~ '\*/'
1204 # end comment block
1205 in_comment = 0
1206 endif
1207 # skip comment-block line
1208 continue
1209 endif
1210 if line =~ '^\s*//'
Turiiya80406c22023-04-22 21:38:47 +01001211 # skip comment line
1212 continue
1213 endif
1214
Christian Brabandt10b4f752024-01-01 19:19:20 +01001215 # Coq: line ends with a '.' followed by an optional variable number of
1216 # spaces or contains the start of a comment, but not inside a Verilog or V
1217 # comment.
1218 # Example: "Definition x := 10. (*".
1219 if (line =~ '\.\s*$' && line !~ '/[/*]') || (line =~ '(\*' && line !~ '/[/*].*(\*')
1220 setf coq
1221 return
1222 endif
1223
Turiiya80406c22023-04-22 21:38:47 +01001224 # Verilog: line ends with ';' followed by an optional variable number of
1225 # spaces and an optional start of a comment.
1226 # Example: " b <= a + 1; // Add 1".
Christian Brabandt10b4f752024-01-01 19:19:20 +01001227 if line =~ ';\s*\(/[/*].*\)\?$'
Turiiya80406c22023-04-22 21:38:47 +01001228 setf verilog
1229 return
1230 endif
Turiiya80406c22023-04-22 21:38:47 +01001231 endfor
1232
1233 # No line matched, fall back to "v".
1234 setf v
1235enddef
1236
Doug Kearnsf97f6bb2023-08-27 18:44:09 +02001237export def FTvba()
1238 if getline(1) =~ '^["#] Vimball Archiver'
1239 setf vim
1240 else
1241 setf vb
1242 endif
1243enddef
1244
Bram Moolenaara2baa732022-02-04 16:09:54 +00001245# Uncomment this line to check for compilation errors early
Bram Moolenaar3e79c972022-02-04 19:48:06 +00001246# defcompile