blob: 9d0f2ee7afaa2f4e8a197ccbfb8875ff870f3db1 [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#
Doug Kearns68a89472024-01-05 17:59:04 +01005# Maintainer: The Vim Project <https://github.com/vim/vim>
6# Last Change: 2024 Jan 05
Christian Brabandte978b452023-08-13 10:33:05 +02007# 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
Doug Kearns68a89472024-01-05 17:59:04 +0100282export def FTdef()
283 if get(g:, "filetype_def", "") == "modula2" || IsModula2()
284 SetFiletypeModula2()
285 return
286 endif
287
288 if exists("g:filetype_def")
289 exe "setf " .. g:filetype_def
290 else
291 setf def
292 endif
293enddef
294
Bram Moolenaara2baa732022-02-04 16:09:54 +0000295export def FTe()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100296 if exists('g:filetype_euphoria')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000297 exe 'setf ' .. g:filetype_euphoria
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100298 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000299 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100300 while n < 100 && n <= line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100301 if getline(n) =~ "^\\s*\\(<'\\|'>\\)\\s*$"
302 setf specman
303 return
304 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000305 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100306 endwhile
307 setf eiffel
308 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000309enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100310
Doug Kearns19a3bc32023-08-20 20:51:12 +0200311def IsForth(): bool
312 var first_line = nextnonblank(1)
313
314 # SwiftForth block comment (line is usually filled with '-' or '=') or
315 # OPTIONAL (sometimes precedes the header comment)
316 if getline(first_line) =~? '^\%({\%(\s\|$\)\|OPTIONAL\s\)'
317 return true
318 endif
319
320 var n = first_line
321 while n < 100 && n <= line("$")
322 # Forth comments and colon definitions
323 if getline(n) =~ '^[:(\\] '
324 return true
325 endif
326 n += 1
327 endwhile
328 return false
329enddef
330
331# Distinguish between Forth and Fortran
332export def FTf()
333 if exists("g:filetype_f")
334 exe "setf " .. g:filetype_f
335 elseif IsForth()
336 setf forth
337 else
338 setf fortran
339 endif
340enddef
341
Bram Moolenaara2baa732022-02-04 16:09:54 +0000342export def FTfrm()
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000343 if exists("g:filetype_frm")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000344 exe "setf " .. g:filetype_frm
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000345 return
346 endif
347
Doug Kearnsf97f6bb2023-08-27 18:44:09 +0200348 if getline(1) == "VERSION 5.00"
349 setf vb
350 return
351 endif
352
Bram Moolenaara2baa732022-02-04 16:09:54 +0000353 var lines = getline(1, min([line("$"), 5]))
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000354
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000355 if match(lines, ft_visual_basic_content) > -1
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000356 setf vb
357 else
358 setf form
359 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000360enddef
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000361
Doug Kearns19a3bc32023-08-20 20:51:12 +0200362# Distinguish between Forth and F#
Bram Moolenaara2baa732022-02-04 16:09:54 +0000363export def FTfs()
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000364 if exists("g:filetype_fs")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000365 exe "setf " .. g:filetype_fs
Doug Kearns19a3bc32023-08-20 20:51:12 +0200366 elseif IsForth()
367 setf forth
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000368 else
Johan Kotlinski065088d2023-04-02 20:29:38 +0100369 setf fsharp
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000370 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000371enddef
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000372
Bram Moolenaara2baa732022-02-04 16:09:54 +0000373# Distinguish between HTML, XHTML and Django
374export def FThtml()
375 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100376 while n < 10 && n <= line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100377 if getline(n) =~ '\<DTD\s\+XHTML\s'
378 setf xhtml
379 return
380 endif
381 if getline(n) =~ '{%\s*\(extends\|block\|load\)\>\|{#\s\+'
382 setf htmldjango
383 return
384 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000385 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100386 endwhile
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100387 setf FALLBACK html
Bram Moolenaara2baa732022-02-04 16:09:54 +0000388enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100389
Bram Moolenaara2baa732022-02-04 16:09:54 +0000390# Distinguish between standard IDL and MS-IDL
391export def FTidl()
392 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100393 while n < 50 && n <= line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100394 if getline(n) =~ '^\s*import\s\+"\(unknwn\|objidl\)\.idl"'
395 setf msidl
396 return
397 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000398 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100399 endwhile
400 setf idl
Bram Moolenaara2baa732022-02-04 16:09:54 +0000401enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100402
Bram Moolenaar48c3f4e2022-08-08 15:42:38 +0100403# Distinguish between "default", Prolog and Cproto prototype file.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000404export def ProtoCheck(default: string)
405 # Cproto files have a comment in the first line and a function prototype in
406 # the second line, it always ends in ";". Indent files may also have
407 # comments, thus we can't match comments to see the difference.
408 # IDL files can have a single ';' in the second line, require at least one
409 # chacter before the ';'.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100410 if getline(2) =~ '.;$'
411 setf cpp
412 else
Bram Moolenaar48c3f4e2022-08-08 15:42:38 +0100413 # recognize Prolog by specific text in the first non-empty line
414 # require a blank after the '%' because Perl uses "%list" and "%translate"
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100415 var lnum = getline(nextnonblank(1))
416 if lnum =~ '\<prolog\>' || lnum =~ '^\s*\(%\+\(\s\|$\)\|/\*\)' || lnum =~ ':-'
Bram Moolenaar48c3f4e2022-08-08 15:42:38 +0100417 setf prolog
418 else
419 exe 'setf ' .. default
420 endif
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100421 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000422enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100423
Bram Moolenaara2baa732022-02-04 16:09:54 +0000424export def FTm()
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200425 if exists("g:filetype_m")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000426 exe "setf " .. g:filetype_m
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200427 return
428 endif
429
Bram Moolenaara2baa732022-02-04 16:09:54 +0000430 # excluding end(for|function|if|switch|while) common to Murphi
431 var octave_block_terminators = '\<end\%(_try_catch\|classdef\|enumeration\|events\|methods\|parfor\|properties\)\>'
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200432
Bram Moolenaara2baa732022-02-04 16:09:54 +0000433 var objc_preprocessor = '^\s*#\s*\%(import\|include\|define\|if\|ifn\=def\|undef\|line\|error\|pragma\)\>'
Doug Kearns7329cfa2021-11-26 13:01:41 +0000434
Bram Moolenaara2baa732022-02-04 16:09:54 +0000435 var n = 1
436 var saw_comment = 0 # Whether we've seen a multiline comment leader.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100437 while n < 100
Bram Moolenaara2baa732022-02-04 16:09:54 +0000438 var line = getline(n)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100439 if line =~ '^\s*/\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000440 # /* ... */ is a comment in Objective C and Murphi, so we can't conclude
441 # it's either of them yet, but track this as a hint in case we don't see
442 # anything more definitive.
443 saw_comment = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100444 endif
Doug Kearns7329cfa2021-11-26 13:01:41 +0000445 if line =~ '^\s*//' || line =~ '^\s*@import\>' || line =~ objc_preprocessor
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100446 setf objc
447 return
448 endif
Bram Moolenaarca0627d2021-09-12 17:03:08 +0200449 if line =~ '^\s*\%(#\|%!\)' || line =~ '^\s*unwind_protect\>' ||
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200450 \ line =~ '\%(^\|;\)\s*' .. octave_block_terminators
451 setf octave
452 return
453 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000454 # TODO: could be Matlab or Octave
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100455 if line =~ '^\s*%'
456 setf matlab
457 return
458 endif
459 if line =~ '^\s*(\*'
460 setf mma
461 return
462 endif
463 if line =~ '^\c\s*\(\(type\|var\)\>\|--\)'
464 setf murphi
465 return
466 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000467 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100468 endwhile
469
470 if saw_comment
Bram Moolenaara2baa732022-02-04 16:09:54 +0000471 # We didn't see anything definitive, but this looks like either Objective C
472 # or Murphi based on the comment leader. Assume the former as it is more
473 # common.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100474 setf objc
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100475 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000476 # Default is Matlab
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100477 setf matlab
478 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000479enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100480
Bram Moolenaara2baa732022-02-04 16:09:54 +0000481export def FTmms()
482 var n = 1
Bram Moolenaard7df2792020-01-02 21:34:42 +0100483 while n < 20
Bram Moolenaara2baa732022-02-04 16:09:54 +0000484 var line = getline(n)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100485 if line =~ '^\s*\(%\|//\)' || line =~ '^\*'
486 setf mmix
487 return
488 endif
489 if line =~ '^\s*#'
490 setf make
491 return
492 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000493 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100494 endwhile
495 setf mmix
Bram Moolenaara2baa732022-02-04 16:09:54 +0000496enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100497
Bram Moolenaara2baa732022-02-04 16:09:54 +0000498# This function checks if one of the first five lines start with a dot. In
499# that case it is probably an nroff file: 'filetype' is set and 1 is returned.
500export def FTnroff(): number
501 if getline(1)[0] .. getline(2)[0] .. getline(3)[0]
502 .. getline(4)[0] .. getline(5)[0] =~ '\.'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100503 setf nroff
504 return 1
505 endif
506 return 0
Bram Moolenaara2baa732022-02-04 16:09:54 +0000507enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100508
Bram Moolenaara2baa732022-02-04 16:09:54 +0000509export def FTmm()
510 var n = 1
Bram Moolenaard1caa942020-04-10 22:10:56 +0200511 while n < 20
Bram Moolenaara2baa732022-02-04 16:09:54 +0000512 if getline(n) =~ '^\s*\(#\s*\(include\|import\)\>\|@import\>\|/\*\)'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100513 setf objcpp
514 return
515 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000516 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100517 endwhile
518 setf nroff
Bram Moolenaara2baa732022-02-04 16:09:54 +0000519enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100520
Bram Moolenaarcdbfc6d2022-06-30 16:25:21 +0100521# Returns true if file content looks like LambdaProlog module
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100522def IsLProlog(): bool
Gaetan Lepage4ce1bda2023-05-10 22:01:55 +0100523 # skip apparent comments and blank lines, what looks like
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100524 # LambdaProlog comment may be RAPID header
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100525 var lnum: number = nextnonblank(1)
526 while lnum > 0 && lnum < line('$') && getline(lnum) =~ '^\s*%' # LambdaProlog comment
527 lnum = nextnonblank(lnum + 1)
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100528 endwhile
529 # this pattern must not catch a go.mod file
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100530 return getline(lnum) =~ '\<module\s\+\w\+\s*\.\s*\(%\|$\)'
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100531enddef
532
Doug Kearns68a89472024-01-05 17:59:04 +0100533def IsModula2(): bool
534 return getline(nextnonblank(1)) =~ '\<MODULE\s\+\w\+\s*;\|^\s*(\*'
535enddef
536
537def SetFiletypeModula2()
538 const KNOWN_DIALECTS = ["iso", "pim", "r10"]
539 const KNOWN_EXTENSIONS = ["gm2"]
540 const LINE_COUNT = 200
541 const TAG = '(\*!m2\(\w\+\)\%(+\(\w\+\)\)\=\*)'
542
543 var dialect = get(g:, "modula2_default_dialect", "pim")
544 var extension = get(g:, "modula2_default_extension", "")
545
546 var matches = []
547
548 # ignore unknown dialects or badly formatted tags
549 for lnum in range(1, min([line("$"), LINE_COUNT]))
550 matches = matchlist(getline(lnum), TAG)
551 if !empty(matches)
552 if index(KNOWN_DIALECTS, matches[1]) >= 0
553 dialect = matches[1]
554 endif
555 if index(KNOWN_EXTENSIONS, matches[2]) >= 0
556 extension = matches[2]
557 endif
558 break
559 endif
560 endfor
561
562 modula2#SetDialect(dialect, extension)
563
564 setf modula2
565enddef
566
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100567# Determine if *.mod is ABB RAPID, LambdaProlog, Modula-2, Modsim III or go.mod
568export def FTmod()
Doug Kearns68a89472024-01-05 17:59:04 +0100569 if get(g:, "filetype_mod", "") == "modula2" || IsModula2()
570 SetFiletypeModula2()
571 return
572 endif
573
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100574 if exists("g:filetype_mod")
575 exe "setf " .. g:filetype_mod
Omar El Halabic9fbd252023-05-29 19:59:45 +0100576 elseif expand("<afile>") =~ '\<go.mod$'
577 setf gomod
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100578 elseif IsLProlog()
579 setf lprolog
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100580 elseif IsRapid()
581 setf rapid
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100582 else
583 # Nothing recognized, assume modsim3
584 setf modsim3
585 endif
586enddef
587
Bram Moolenaara2baa732022-02-04 16:09:54 +0000588export def FTpl()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100589 if exists("g:filetype_pl")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000590 exe "setf " .. g:filetype_pl
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100591 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000592 # recognize Prolog by specific text in the first non-empty line
593 # require a blank after the '%' because Perl uses "%list" and "%translate"
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100594 var line = getline(nextnonblank(1))
595 if line =~ '\<prolog\>' || line =~ '^\s*\(%\+\(\s\|$\)\|/\*\)' || line =~ ':-'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100596 setf prolog
597 else
598 setf perl
599 endif
600 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000601enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100602
Bram Moolenaara2baa732022-02-04 16:09:54 +0000603export def FTinc()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100604 if exists("g:filetype_inc")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000605 exe "setf " .. g:filetype_inc
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100606 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000607 var lines = getline(1) .. getline(2) .. getline(3)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100608 if lines =~? "perlscript"
609 setf aspperl
610 elseif lines =~ "<%"
611 setf aspvbs
612 elseif lines =~ "<?"
613 setf php
Bram Moolenaara2baa732022-02-04 16:09:54 +0000614 # Pascal supports // comments but they're vary rarely used for file
615 # headers so assume POV-Ray
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000616 elseif lines =~ '^\s*\%({\|(\*\)' || lines =~? ft_pascal_keywords
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100617 setf pascal
Gregory Anders30e212d2022-07-26 21:42:03 +0100618 elseif lines =~# '\<\%(require\|inherit\)\>' || lines =~# '[A-Z][A-Za-z0-9_:${}]*\s\+\%(??\|[?:+]\)\?= '
Gregory Andersfa49eb42022-07-16 17:46:47 +0100619 setf bitbake
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100620 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000621 FTasmsyntax()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100622 if exists("b:asmsyntax")
Gregory Andersfa49eb42022-07-16 17:46:47 +0100623 exe "setf " .. fnameescape(b:asmsyntax)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100624 else
Gregory Andersfa49eb42022-07-16 17:46:47 +0100625 setf pov
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100626 endif
627 endif
628 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000629enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100630
Bram Moolenaara2baa732022-02-04 16:09:54 +0000631export def FTprogress_cweb()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100632 if exists("g:filetype_w")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000633 exe "setf " .. g:filetype_w
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100634 return
635 endif
636 if getline(1) =~ '&ANALYZE' || getline(3) =~ '&GLOBAL-DEFINE'
637 setf progress
638 else
639 setf cweb
640 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000641enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100642
Julien Marrec2e310652023-11-25 15:30:46 +0100643# These include the leading '%' sign
644var 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\)'
645# This is the start/end of a block that is copied literally to the processor file (C/C++)
646var ft_swig_verbatim_block_start = '^\s*%{'
647
648export def FTi()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100649 if exists("g:filetype_i")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000650 exe "setf " .. g:filetype_i
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100651 return
652 endif
Julien Marrec2e310652023-11-25 15:30:46 +0100653 # 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 +0000654 # If not found, assume Progress.
655 var lnum = 1
Julien Marrec2e310652023-11-25 15:30:46 +0100656 while lnum <= 50 && lnum < line('$')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000657 var line = getline(lnum)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100658 if line =~ '^\s*;' || line =~ '^\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000659 FTasm()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100660 return
Julien Marrec2e310652023-11-25 15:30:46 +0100661 elseif line =~ ft_swig_keywords || line =~ ft_swig_verbatim_block_start
662 setf swig
663 return
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100664 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000665 lnum += 1
Bram Moolenaarc12dc472022-03-05 13:45:56 +0000666 endwhile
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100667 setf progress
Bram Moolenaara2baa732022-02-04 16:09:54 +0000668enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100669
Bram Moolenaara2baa732022-02-04 16:09:54 +0000670var ft_pascal_comments = '^\s*\%({\|(\*\|//\)'
671var ft_pascal_keywords = '^\s*\%(program\|unit\|library\|uses\|begin\|procedure\|function\|const\|type\|var\)\>'
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100672
Bram Moolenaara2baa732022-02-04 16:09:54 +0000673export def FTprogress_pascal()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100674 if exists("g:filetype_p")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000675 exe "setf " .. g:filetype_p
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100676 return
677 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000678 # This function checks for valid Pascal syntax in the first ten lines.
679 # Look for either an opening comment or a program start.
680 # If not found, assume Progress.
681 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100682 while lnum <= 10 && lnum < line('$')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000683 var line = getline(lnum)
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000684 if line =~ ft_pascal_comments || line =~? ft_pascal_keywords
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100685 setf pascal
686 return
687 elseif line !~ '^\s*$' || line =~ '^/\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000688 # Not an empty line: Doesn't look like valid Pascal code.
689 # Or it looks like a Progress /* comment
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100690 break
691 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000692 lnum += 1
Bram Moolenaarc12dc472022-03-05 13:45:56 +0000693 endwhile
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100694 setf progress
Bram Moolenaara2baa732022-02-04 16:09:54 +0000695enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100696
Bram Moolenaara2baa732022-02-04 16:09:54 +0000697export def FTpp()
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100698 if exists("g:filetype_pp")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000699 exe "setf " .. g:filetype_pp
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100700 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000701 var line = getline(nextnonblank(1))
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000702 if line =~ ft_pascal_comments || line =~? ft_pascal_keywords
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100703 setf pascal
704 else
705 setf puppet
706 endif
707 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000708enddef
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100709
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100710# Determine if *.prg is ABB RAPID. Can also be Clipper, FoxPro or eviews
711export def FTprg()
712 if exists("g:filetype_prg")
713 exe "setf " .. g:filetype_prg
714 elseif IsRapid()
715 setf rapid
716 else
717 # Nothing recognized, assume Clipper
718 setf clipper
719 endif
720enddef
721
Bram Moolenaara2baa732022-02-04 16:09:54 +0000722export def FTr()
723 var max = line("$") > 50 ? 50 : line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100724
725 for n in range(1, max)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000726 # Rebol is easy to recognize, check for that first
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100727 if getline(n) =~? '\<REBOL\>'
728 setf rebol
729 return
730 endif
731 endfor
732
733 for n in range(1, max)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000734 # R has # comments
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100735 if getline(n) =~ '^\s*#'
736 setf r
737 return
738 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000739 # Rexx has /* comments */
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100740 if getline(n) =~ '^\s*/\*'
741 setf rexx
742 return
743 endif
744 endfor
745
Bram Moolenaara2baa732022-02-04 16:09:54 +0000746 # Nothing recognized, use user default or assume Rexx
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100747 if exists("g:filetype_r")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000748 exe "setf " .. g:filetype_r
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100749 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000750 # Rexx used to be the default, but R appears to be much more popular.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100751 setf r
752 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000753enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100754
Bram Moolenaara2baa732022-02-04 16:09:54 +0000755export def McSetf()
756 # Rely on the file to start with a comment.
757 # MS message text files use ';', Sendmail files use '#' or 'dnl'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100758 for lnum in range(1, min([line("$"), 20]))
Bram Moolenaara2baa732022-02-04 16:09:54 +0000759 var line = getline(lnum)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100760 if line =~ '^\s*\(#\|dnl\)'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000761 setf m4 # Sendmail .mc file
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100762 return
763 elseif line =~ '^\s*;'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000764 setf msmessages # MS Message text file
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100765 return
766 endif
767 endfor
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100768 setf m4 # Default: Sendmail .mc file
Bram Moolenaara2baa732022-02-04 16:09:54 +0000769enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100770
Bram Moolenaara2baa732022-02-04 16:09:54 +0000771# Called from filetype.vim and scripts.vim.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100772# When "setft" is passed and false then the 'filetype' option is not set.
773export def SetFileTypeSH(name: string, setft = true): string
774 if setft && did_filetype()
Bram Moolenaara2baa732022-02-04 16:09:54 +0000775 # Filetype was already detected
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100776 return ''
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100777 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100778 if setft && expand("<amatch>") =~ g:ft_ignore_pat
779 return ''
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100780 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000781 if name =~ '\<csh\>'
782 # Some .sh scripts contain #!/bin/csh.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100783 return SetFileTypeShell("csh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000784 elseif name =~ '\<tcsh\>'
785 # Some .sh scripts contain #!/bin/tcsh.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100786 return SetFileTypeShell("tcsh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000787 elseif name =~ '\<zsh\>'
788 # Some .sh scripts contain #!/bin/zsh.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100789 return SetFileTypeShell("zsh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000790 elseif name =~ '\<ksh\>'
791 b:is_kornshell = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100792 if exists("b:is_bash")
793 unlet b:is_bash
794 endif
795 if exists("b:is_sh")
796 unlet b:is_sh
797 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000798 elseif exists("g:bash_is_sh") || name =~ '\<bash\>' || name =~ '\<bash2\>'
799 b:is_bash = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100800 if exists("b:is_kornshell")
801 unlet b:is_kornshell
802 endif
803 if exists("b:is_sh")
804 unlet b:is_sh
805 endif
Eisuke Kawashima24482fb2022-11-24 10:58:10 +0000806 elseif name =~ '\<sh\>' || name =~ '\<dash\>'
807 # Ubuntu links "sh" to "dash", thus it is expected to work the same way
Bram Moolenaara2baa732022-02-04 16:09:54 +0000808 b:is_sh = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100809 if exists("b:is_kornshell")
810 unlet b:is_kornshell
811 endif
812 if exists("b:is_bash")
813 unlet b:is_bash
814 endif
815 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100816
817 return SetFileTypeShell("sh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000818enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100819
Bram Moolenaara2baa732022-02-04 16:09:54 +0000820# For shell-like file types, check for an "exec" command hidden in a comment,
821# as used for Tcl.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100822# When "setft" is passed and false then the 'filetype' option is not set.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000823# Also called from scripts.vim, thus can't be local to this script.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100824export def SetFileTypeShell(name: string, setft = true): string
825 if setft && did_filetype()
Bram Moolenaara2baa732022-02-04 16:09:54 +0000826 # Filetype was already detected
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100827 return ''
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100828 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100829 if setft && expand("<amatch>") =~ g:ft_ignore_pat
830 return ''
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100831 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100832
833 var lnum = 2
834 while lnum < 20 && lnum < line("$") && getline(lnum) =~ '^\s*\(#\|$\)'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000835 # Skip empty and comment lines.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100836 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100837 endwhile
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100838 if lnum < line("$") && getline(lnum) =~ '\s*exec\s' && getline(lnum - 1) =~ '^\s*#.*\\$'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000839 # Found an "exec" line after a comment with continuation
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100840 var n = substitute(getline(lnum), '\s*exec\s\+\([^ ]*/\)\=', '', '')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100841 if n =~ '\<tclsh\|\<wish'
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100842 if setft
843 setf tcl
844 endif
845 return 'tcl'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100846 endif
847 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100848
849 if setft
850 exe "setf " .. name
851 endif
852 return name
Bram Moolenaara2baa732022-02-04 16:09:54 +0000853enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100854
Bram Moolenaara2baa732022-02-04 16:09:54 +0000855export def CSH()
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100856 if did_filetype()
Bram Moolenaara2baa732022-02-04 16:09:54 +0000857 # Filetype was already detected
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100858 return
859 endif
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100860 if exists("g:filetype_csh")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000861 SetFileTypeShell(g:filetype_csh)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100862 elseif &shell =~ "tcsh"
Bram Moolenaara2baa732022-02-04 16:09:54 +0000863 SetFileTypeShell("tcsh")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100864 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000865 SetFileTypeShell("csh")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100866 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000867enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100868
Bram Moolenaara2baa732022-02-04 16:09:54 +0000869var ft_rules_udev_rules_pattern = '^\s*\cudev_rules\s*=\s*"\([^"]\{-1,}\)/*".*'
870export def FTRules()
871 var path = expand('<amatch>:p')
Bram Moolenaaraa9675a2020-08-17 21:57:09 +0200872 if path =~ '/\(etc/udev/\%(rules\.d/\)\=.*\.rules\|\%(usr/\)\=lib/udev/\%(rules\.d/\)\=.*\.rules\)$'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100873 setf udevrules
874 return
875 endif
876 if path =~ '^/etc/ufw/'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000877 setf conf # Better than hog
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100878 return
879 endif
880 if path =~ '^/\(etc\|usr/share\)/polkit-1/rules\.d'
881 setf javascript
882 return
883 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000884 var config_lines: list<string>
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100885 try
Bram Moolenaara2baa732022-02-04 16:09:54 +0000886 config_lines = readfile('/etc/udev/udev.conf')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100887 catch /^Vim\%((\a\+)\)\=:E484/
888 setf hog
889 return
890 endtry
Bram Moolenaara2baa732022-02-04 16:09:54 +0000891 var dir = expand('<amatch>:p:h')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100892 for line in config_lines
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000893 if line =~ ft_rules_udev_rules_pattern
894 var udev_rules = substitute(line, ft_rules_udev_rules_pattern, '\1', "")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100895 if dir == udev_rules
896 setf udevrules
897 endif
898 break
899 endif
900 endfor
901 setf hog
Bram Moolenaara2baa732022-02-04 16:09:54 +0000902enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100903
Bram Moolenaara2baa732022-02-04 16:09:54 +0000904export def SQL()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100905 if exists("g:filetype_sql")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000906 exe "setf " .. g:filetype_sql
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100907 else
908 setf sql
909 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000910enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100911
ranjithshegde8cac20e2022-04-13 15:29:21 +0100912# This function checks the first 25 lines of file extension "sc" to resolve
Chris Kipp70ef3f52022-12-14 16:42:15 +0000913# detection between scala and SuperCollider.
914# NOTE: We don't check for 'Class : Method', as this can easily be confused
915# with valid Scala like `val x : Int = 3`. So we instead only rely on
916# checks that can't be confused.
ranjithshegde8cac20e2022-04-13 15:29:21 +0100917export def FTsc()
918 for lnum in range(1, min([line("$"), 25]))
Chris Kipp70ef3f52022-12-14 16:42:15 +0000919 if getline(lnum) =~# 'var\s<\|classvar\s<\|\^this.*\||\w\+|\|+\s\w*\s{\|\*ar\s'
ranjithshegde8cac20e2022-04-13 15:29:21 +0100920 setf supercollider
921 return
922 endif
923 endfor
924 setf scala
925enddef
926
927# This function checks the first line of file extension "scd" to resolve
928# detection between scdoc and SuperCollider
929export def FTscd()
930 if getline(1) =~# '\%^\S\+(\d[0-9A-Za-z]*)\%(\s\+\"[^"]*\"\%(\s\+\"[^"]*\"\)\=\)\=$'
931 setf scdoc
932 else
933 setf supercollider
934 endif
935enddef
936
Bram Moolenaara2baa732022-02-04 16:09:54 +0000937# If the file has an extension of 't' and is in a directory 't' or 'xt' then
938# it is almost certainly a Perl test file.
939# If the first line starts with '#' and contains 'perl' it's probably a Perl
940# file.
941# (Slow test) If a file contains a 'use' statement then it is almost certainly
942# a Perl file.
943export def FTperl(): number
944 var dirname = expand("%:p:h:t")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100945 if expand("%:e") == 't' && (dirname == 't' || dirname == 'xt')
946 setf perl
947 return 1
948 endif
949 if getline(1)[0] == '#' && getline(1) =~ 'perl'
950 setf perl
951 return 1
952 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000953 var save_cursor = getpos('.')
954 call cursor(1, 1)
Bram Moolenaare5b78972022-02-05 19:50:34 +0000955 var has_use = search('^use\s\s*\k', 'c', 30) > 0
Bram Moolenaarf0b03c42017-12-17 17:17:07 +0100956 call setpos('.', save_cursor)
957 if has_use
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100958 setf perl
959 return 1
960 endif
961 return 0
Bram Moolenaara2baa732022-02-04 16:09:54 +0000962enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100963
Bram Moolenaarcdbfc6d2022-06-30 16:25:21 +0100964# LambdaProlog and Standard ML signature files
965export def FTsig()
966 if exists("g:filetype_sig")
967 exe "setf " .. g:filetype_sig
968 return
969 endif
970
971 var lprolog_comment = '^\s*\%(/\*\|%\)'
972 var lprolog_keyword = '^\s*sig\s\+\a'
973 var sml_comment = '^\s*(\*'
974 var sml_keyword = '^\s*\%(signature\|structure\)\s\+\a'
975
976 var line = getline(nextnonblank(1))
977
978 if line =~ lprolog_comment || line =~# lprolog_keyword
979 setf lprolog
980 elseif line =~ sml_comment || line =~# sml_keyword
981 setf sml
982 endif
983enddef
984
Bram Moolenaarbe807d52022-09-01 15:01:25 +0100985# This function checks the first 100 lines of files matching "*.sil" to
986# resolve detection between Swift Intermediate Language and SILE.
987export def FTsil()
988 for lnum in range(1, [line('$'), 100]->min())
989 var line: string = getline(lnum)
990 if line =~ '^\s*[\\%]'
991 setf sile
992 return
993 elseif line =~ '^\s*\S'
994 setf sil
995 return
996 endif
997 endfor
998 # no clue, default to "sil"
999 setf sil
1000enddef
1001
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +01001002export def FTsys()
KnoP-01f420ff22022-04-13 20:46:21 +01001003 if exists("g:filetype_sys")
1004 exe "setf " .. g:filetype_sys
1005 elseif IsRapid()
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +01001006 setf rapid
1007 else
1008 setf bat
1009 endif
1010enddef
1011
Bram Moolenaara2baa732022-02-04 16:09:54 +00001012# Choose context, plaintex, or tex (LaTeX) based on these rules:
1013# 1. Check the first line of the file for "%&<format>".
1014# 2. Check the first 1000 non-comment lines for LaTeX or ConTeXt keywords.
1015# 3. Default to "plain" or to g:tex_flavor, can be set in user's vimrc.
1016export def FTtex()
1017 var firstline = getline(1)
1018 var format: string
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001019 if firstline =~ '^%&\s*\a\+'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001020 format = tolower(matchstr(firstline, '\a\+'))
1021 format = substitute(format, 'pdf', '', '')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001022 if format == 'tex'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001023 format = 'latex'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001024 elseif format == 'plaintex'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001025 format = 'plain'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001026 endif
1027 elseif expand('%') =~ 'tex/context/.*/.*.tex'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001028 format = 'context'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001029 else
Bram Moolenaara2baa732022-02-04 16:09:54 +00001030 # Default value, may be changed later:
1031 format = exists("g:tex_flavor") ? g:tex_flavor : 'plain'
1032 # Save position, go to the top of the file, find first non-comment line.
1033 var save_cursor = getpos('.')
1034 call cursor(1, 1)
1035 var firstNC = search('^\s*[^[:space:]%]', 'c', 1000)
Bram Moolenaare5b78972022-02-05 19:50:34 +00001036 if firstNC > 0
1037 # Check the next thousand lines for a LaTeX or ConTeXt keyword.
Bram Moolenaara2baa732022-02-04 16:09:54 +00001038 var lpat = 'documentclass\>\|usepackage\>\|begin{\|newcommand\>\|renewcommand\>'
1039 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\>'
1040 var kwline = search('^\s*\\\%(' .. lpat .. '\)\|^\s*\\\(' .. cpat .. '\)',
1041 'cnp', firstNC + 1000)
1042 if kwline == 1 # lpat matched
1043 format = 'latex'
1044 elseif kwline == 2 # cpat matched
1045 format = 'context'
1046 endif # If neither matched, keep default set above.
1047 # let lline = search('^\s*\\\%(' . lpat . '\)', 'cn', firstNC + 1000)
1048 # let cline = search('^\s*\\\%(' . cpat . '\)', 'cn', firstNC + 1000)
1049 # if cline > 0
1050 # let format = 'context'
1051 # endif
1052 # if lline > 0 && (cline == 0 || cline > lline)
1053 # let format = 'tex'
1054 # endif
1055 endif # firstNC
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001056 call setpos('.', save_cursor)
Bram Moolenaara2baa732022-02-04 16:09:54 +00001057 endif # firstline =~ '^%&\s*\a\+'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001058
Bram Moolenaara2baa732022-02-04 16:09:54 +00001059 # Translation from formats to file types. TODO: add AMSTeX, RevTex, others?
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001060 if format == 'plain'
1061 setf plaintex
1062 elseif format == 'context'
1063 setf context
Bram Moolenaara2baa732022-02-04 16:09:54 +00001064 else # probably LaTeX
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001065 setf tex
1066 endif
1067 return
Bram Moolenaara2baa732022-02-04 16:09:54 +00001068enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001069
Bram Moolenaara2baa732022-02-04 16:09:54 +00001070export def FTxml()
1071 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +01001072 while n < 100 && n <= line("$")
Bram Moolenaara2baa732022-02-04 16:09:54 +00001073 var line = getline(n)
1074 # DocBook 4 or DocBook 5.
1075 var is_docbook4 = line =~ '<!DOCTYPE.*DocBook'
1076 var is_docbook5 = line =~ ' xmlns="http://docbook.org/ns/docbook"'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001077 if is_docbook4 || is_docbook5
Bram Moolenaara2baa732022-02-04 16:09:54 +00001078 b:docbk_type = "xml"
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001079 if is_docbook5
Bram Moolenaara2baa732022-02-04 16:09:54 +00001080 b:docbk_ver = 5
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001081 else
Bram Moolenaara2baa732022-02-04 16:09:54 +00001082 b:docbk_ver = 4
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001083 endif
1084 setf docbk
1085 return
1086 endif
1087 if line =~ 'xmlns:xbl="http://www.mozilla.org/xbl"'
1088 setf xbl
1089 return
1090 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001091 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001092 endwhile
1093 setf xml
Bram Moolenaara2baa732022-02-04 16:09:54 +00001094enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001095
Bram Moolenaara2baa732022-02-04 16:09:54 +00001096export def FTy()
1097 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +01001098 while n < 100 && n <= line("$")
Bram Moolenaara2baa732022-02-04 16:09:54 +00001099 var line = getline(n)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001100 if line =~ '^\s*%'
1101 setf yacc
1102 return
1103 endif
1104 if getline(n) =~ '^\s*\(#\|class\>\)' && getline(n) !~ '^\s*#\s*include'
1105 setf racc
1106 return
1107 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001108 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001109 endwhile
1110 setf yacc
Bram Moolenaara2baa732022-02-04 16:09:54 +00001111enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001112
Bram Moolenaara2baa732022-02-04 16:09:54 +00001113export def Redif()
1114 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001115 while lnum <= 5 && lnum < line('$')
1116 if getline(lnum) =~ "^\ctemplate-type:"
1117 setf redif
1118 return
1119 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001120 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001121 endwhile
Bram Moolenaara2baa732022-02-04 16:09:54 +00001122enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001123
Bram Moolenaara2baa732022-02-04 16:09:54 +00001124# This function is called for all files under */debian/patches/*, make sure not
1125# to non-dep3patch files, such as README and other text files.
1126export def Dep3patch()
James McCoy647ab4c2021-12-17 20:52:57 +00001127 if expand('%:t') ==# 'series'
1128 return
1129 endif
1130
1131 for ln in getline(1, 100)
1132 if ln =~# '^\%(Description\|Subject\|Origin\|Bug\|Forwarded\|Author\|From\|Reviewed-by\|Acked-by\|Last-Updated\|Applied-Upstream\):'
1133 setf dep3patch
1134 return
1135 elseif ln =~# '^---'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001136 # end of headers found. stop processing
James McCoy647ab4c2021-12-17 20:52:57 +00001137 return
1138 endif
1139 endfor
Bram Moolenaara2baa732022-02-04 16:09:54 +00001140enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001141
Bram Moolenaara2baa732022-02-04 16:09:54 +00001142# This function checks the first 15 lines for appearance of 'FoamFile'
1143# and then 'object' in a following line.
1144# In that case, it's probably an OpenFOAM file
1145export def FTfoam()
1146 var ffile = 0
1147 var lnum = 1
Elwardi2284f6c2022-01-11 18:14:23 +00001148 while lnum <= 15
1149 if getline(lnum) =~# '^FoamFile'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001150 ffile = 1
Elwardi2284f6c2022-01-11 18:14:23 +00001151 elseif ffile == 1 && getline(lnum) =~# '^\s*object'
1152 setf foam
1153 return
1154 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001155 lnum += 1
Elwardi2284f6c2022-01-11 18:14:23 +00001156 endwhile
Bram Moolenaara2baa732022-02-04 16:09:54 +00001157enddef
Elwardi2284f6c2022-01-11 18:14:23 +00001158
Bram Moolenaara2baa732022-02-04 16:09:54 +00001159# Determine if a *.tf file is TF mud client or terraform
1160export def FTtf()
1161 var numberOfLines = line('$')
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001162 for i in range(1, numberOfLines)
Bram Moolenaara2baa732022-02-04 16:09:54 +00001163 var currentLine = trim(getline(i))
1164 var firstCharacter = currentLine[0]
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001165 if firstCharacter !=? ";" && firstCharacter !=? "/" && firstCharacter !=? ""
1166 setf terraform
1167 return
1168 endif
1169 endfor
1170 setf tf
Bram Moolenaara2baa732022-02-04 16:09:54 +00001171enddef
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001172
KnoP-0193c7a452022-04-16 21:14:04 +01001173var ft_krl_header = '\&\w+'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001174# Determine if a *.src file is Kuka Robot Language
1175export def FTsrc()
KnoP-0193c7a452022-04-16 21:14:04 +01001176 var ft_krl_def_or_deffct = '%(global\s+)?def%(fct)?>'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001177 if exists("g:filetype_src")
1178 exe "setf " .. g:filetype_src
KnoP-0193c7a452022-04-16 21:14:04 +01001179 elseif getline(nextnonblank(1)) =~? '\v^\s*%(' .. ft_krl_header .. '|' .. ft_krl_def_or_deffct .. ')'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001180 setf krl
1181 endif
1182enddef
1183
1184# Determine if a *.dat file is Kuka Robot Language
1185export def FTdat()
KnoP-0193c7a452022-04-16 21:14:04 +01001186 var ft_krl_defdat = 'defdat>'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001187 if exists("g:filetype_dat")
1188 exe "setf " .. g:filetype_dat
KnoP-0193c7a452022-04-16 21:14:04 +01001189 elseif getline(nextnonblank(1)) =~? '\v^\s*%(' .. ft_krl_header .. '|' .. ft_krl_defdat .. ')'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001190 setf krl
1191 endif
1192enddef
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001193
Doug Kearns4ac8e792022-10-17 13:32:17 +01001194export def FTlsl()
1195 if exists("g:filetype_lsl")
1196 exe "setf " .. g:filetype_lsl
1197 endif
1198
1199 var line = getline(nextnonblank(1))
1200 if line =~ '^\s*%' || line =~# ':\s*trait\s*$'
1201 setf larch
1202 else
1203 setf lsl
1204 endif
1205enddef
1206
Gaetan Lepage4ce1bda2023-05-10 22:01:55 +01001207export def FTtyp()
1208 if exists("g:filetype_typ")
1209 exe "setf " .. g:filetype_typ
1210 return
1211 endif
1212
1213 # Look for SQL type definition syntax
1214 for line in getline(1, 200)
1215 # SQL type files may define the casing
1216 if line =~ '^CASE\s\==\s\=\(SAME\|LOWER\|UPPER\|OPPOSITE\)$'
1217 setf sql
1218 return
1219 endif
1220
1221 # SQL type files may define some types as follows
1222 if line =~ '^TYPE\s.*$'
1223 setf sql
1224 return
1225 endif
1226 endfor
1227
1228 # Otherwise, affect the typst filetype
1229 setf typst
1230enddef
1231
Turiiya80406c22023-04-22 21:38:47 +01001232# Set the filetype of a *.v file to Verilog, V or Cog based on the first 200
1233# lines.
1234export def FTv()
1235 if did_filetype()
1236 # ":setf" will do nothing, bail out early
1237 return
1238 endif
Christian Brabandt10b4f752024-01-01 19:19:20 +01001239 if exists("g:filetype_v")
1240 exe "setf " .. g:filetype_v
1241 return
1242 endif
Turiiya80406c22023-04-22 21:38:47 +01001243
Christian Brabandt10b4f752024-01-01 19:19:20 +01001244 var in_comment = 0
1245 for lnum in range(1, min([line("$"), 200]))
1246 var line = getline(lnum)
1247 # Skip Verilog and V comments (lines and blocks).
1248 if line =~ '^\s*/\*'
1249 # start comment block
1250 in_comment = 1
1251 endif
1252 if in_comment == 1
1253 if line =~ '\*/'
1254 # end comment block
1255 in_comment = 0
1256 endif
1257 # skip comment-block line
1258 continue
1259 endif
1260 if line =~ '^\s*//'
Turiiya80406c22023-04-22 21:38:47 +01001261 # skip comment line
1262 continue
1263 endif
1264
Christian Brabandt10b4f752024-01-01 19:19:20 +01001265 # Coq: line ends with a '.' followed by an optional variable number of
1266 # spaces or contains the start of a comment, but not inside a Verilog or V
1267 # comment.
1268 # Example: "Definition x := 10. (*".
1269 if (line =~ '\.\s*$' && line !~ '/[/*]') || (line =~ '(\*' && line !~ '/[/*].*(\*')
1270 setf coq
1271 return
1272 endif
1273
Turiiya80406c22023-04-22 21:38:47 +01001274 # Verilog: line ends with ';' followed by an optional variable number of
1275 # spaces and an optional start of a comment.
1276 # Example: " b <= a + 1; // Add 1".
Christian Brabandt10b4f752024-01-01 19:19:20 +01001277 if line =~ ';\s*\(/[/*].*\)\?$'
Turiiya80406c22023-04-22 21:38:47 +01001278 setf verilog
1279 return
1280 endif
Turiiya80406c22023-04-22 21:38:47 +01001281 endfor
1282
1283 # No line matched, fall back to "v".
1284 setf v
1285enddef
1286
Doug Kearnsf97f6bb2023-08-27 18:44:09 +02001287export def FTvba()
1288 if getline(1) =~ '^["#] Vimball Archiver'
1289 setf vim
1290 else
1291 setf vb
1292 endif
1293enddef
1294
Bram Moolenaara2baa732022-02-04 16:09:54 +00001295# Uncomment this line to check for compilation errors early
Doug Kearns68a89472024-01-05 17:59:04 +01001296defcompile