blob: e53cdacbe34c88ada5a6efc27ba5ddb1effb9203 [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>
Amelia Clarke35dfe582024-05-24 08:05:00 +02006# Last Change: 2024 May 23
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
igna_martinoli37853b72024-07-18 21:34:36 +020012var prolog_pattern = '^\s*\(:-\|%\+\(\s\|$\)\|\/\*\)\|\.\s*$'
13
Bram Moolenaara2baa732022-02-04 16:09:54 +000014export def Check_inp()
Wu, Zhenyu61ee8332024-04-09 22:09:30 +020015 if getline(1) =~ '%%'
16 setf tex
17 elseif getline(1) =~ '^\*'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010018 setf abaqus
19 else
Bram Moolenaara2baa732022-02-04 16:09:54 +000020 var n = 1
21 var nmax = line("$") > 500 ? 500 : line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010022 while n <= nmax
23 if getline(n) =~? "^header surface data"
24 setf trasys
25 break
26 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +000027 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010028 endwhile
29 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +000030enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010031
Bram Moolenaara2baa732022-02-04 16:09:54 +000032# This function checks for the kind of assembly that is wanted by the user, or
33# can be detected from the first five lines of the file.
34export def FTasm()
35 # make sure b:asmsyntax exists
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010036 if !exists("b:asmsyntax")
Bram Moolenaara2baa732022-02-04 16:09:54 +000037 b:asmsyntax = ""
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010038 endif
39
40 if b:asmsyntax == ""
Bram Moolenaara2baa732022-02-04 16:09:54 +000041 FTasmsyntax()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010042 endif
43
Bram Moolenaara2baa732022-02-04 16:09:54 +000044 # if b:asmsyntax still isn't set, default to asmsyntax or GNU
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010045 if b:asmsyntax == ""
46 if exists("g:asmsyntax")
Bram Moolenaara2baa732022-02-04 16:09:54 +000047 b:asmsyntax = g:asmsyntax
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010048 else
Bram Moolenaara2baa732022-02-04 16:09:54 +000049 b:asmsyntax = "asm"
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010050 endif
51 endif
52
Bram Moolenaara2baa732022-02-04 16:09:54 +000053 exe "setf " .. fnameescape(b:asmsyntax)
54enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010055
Bram Moolenaara2baa732022-02-04 16:09:54 +000056export def FTasmsyntax()
57 # see if the file contains any asmsyntax=foo overrides. If so, change
58 # b:asmsyntax appropriately
59 var head = " " .. getline(1) .. " " .. getline(2) .. " "
60 .. getline(3) .. " " .. getline(4) .. " " .. getline(5) .. " "
61 var match = matchstr(head, '\sasmsyntax=\zs[a-zA-Z0-9]\+\ze\s')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010062 if match != ''
Bram Moolenaara2baa732022-02-04 16:09:54 +000063 b:asmsyntax = match
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010064 elseif ((head =~? '\.title') || (head =~? '\.ident') || (head =~? '\.macro') || (head =~? '\.subtitle') || (head =~? '\.library'))
Bram Moolenaara2baa732022-02-04 16:09:54 +000065 b:asmsyntax = "vmasm"
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010066 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +000067enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010068
Doug Kearnsf97f6bb2023-08-27 18:44:09 +020069var ft_visual_basic_content = '\c^\s*\%(Attribute\s\+VB_Name\|Begin\s\+\%(VB\.\|{\%(\x\+-\)\+\x\+}\)\)'
Doug Kearnsc570e9c2022-01-31 17:09:14 +000070
Bram Moolenaara2baa732022-02-04 16:09:54 +000071# See FTfrm() for Visual Basic form file detection
72export def FTbas()
Bram Moolenaar6517f142022-01-21 14:55:13 +000073 if exists("g:filetype_bas")
Bram Moolenaara2baa732022-02-04 16:09:54 +000074 exe "setf " .. g:filetype_bas
Bram Moolenaar6517f142022-01-21 14:55:13 +000075 return
76 endif
77
Bram Moolenaara2baa732022-02-04 16:09:54 +000078 # most frequent FreeBASIC-specific keywords in distro files
79 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 +010080 var fb_preproc = '\c^\s*\%(' ..
81 # preprocessor
82 '#\s*\a\+\|' ..
83 # compiler option
84 'option\s\+\%(byval\|dynamic\|escape\|\%(no\)\=gosub\|nokeyword\|private\|static\)\>\|' ..
85 # metacommand
86 '\%(''\|rem\)\s*\$lang\>\|' ..
87 # default datatype
88 'def\%(byte\|longint\|short\|ubyte\|uint\|ulongint\|ushort\)\>' ..
89 '\)'
Bram Moolenaara2baa732022-02-04 16:09:54 +000090 var fb_comment = "^\\s*/'"
Bram Moolenaar8b5901e2022-06-29 14:39:12 +010091
Bram Moolenaara2baa732022-02-04 16:09:54 +000092 # OPTION EXPLICIT, without the leading underscore, is common to many dialects
93 var qb64_preproc = '\c^\s*\%($\a\+\|option\s\+\%(_explicit\|_\=explicitarray\)\>\)'
Bram Moolenaar6517f142022-01-21 14:55:13 +000094
Bram Moolenaar8b5901e2022-06-29 14:39:12 +010095 for lnum in range(1, min([line("$"), 100]))
96 var line = getline(lnum)
97 if line =~ ft_visual_basic_content
98 setf vb
99 return
100 elseif line =~ fb_preproc || line =~ fb_comment || line =~ fb_keywords
101 setf freebasic
102 return
103 elseif line =~ qb64_preproc
104 setf qb64
105 return
106 endif
107 endfor
108 setf basic
Bram Moolenaara2baa732022-02-04 16:09:54 +0000109enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100110
Bram Moolenaara2baa732022-02-04 16:09:54 +0000111export def FTbtm()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100112 if exists("g:dosbatch_syntax_for_btm") && g:dosbatch_syntax_for_btm
113 setf dosbatch
114 else
115 setf btm
116 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000117enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100118
Bram Moolenaara2baa732022-02-04 16:09:54 +0000119export def BindzoneCheck(default = '')
120 if getline(1) .. getline(2) .. getline(3) .. getline(4)
121 =~ '^; <<>> DiG [0-9.]\+.* <<>>\|$ORIGIN\|$TTL\|IN\s\+SOA'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100122 setf bindzone
Bram Moolenaara2baa732022-02-04 16:09:54 +0000123 elseif default != ''
124 exe 'setf ' .. default
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100125 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000126enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100127
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100128# Returns true if file content looks like RAPID
129def IsRapid(sChkExt: string = ""): bool
130 if sChkExt == "cfg"
131 return getline(1) =~? '\v^%(EIO|MMC|MOC|PROC|SIO|SYS):CFG'
132 endif
133 # called from FTmod, FTprg or FTsys
134 return getline(nextnonblank(1)) =~? '\v^\s*%(\%{3}|module\s+\k+\s*%(\(|$))'
135enddef
136
137export def FTcfg()
138 if exists("g:filetype_cfg")
139 exe "setf " .. g:filetype_cfg
140 elseif IsRapid("cfg")
141 setf rapid
142 else
143 setf cfg
144 endif
145enddef
146
Bram Moolenaar8b5901e2022-06-29 14:39:12 +0100147export def FTcls()
148 if exists("g:filetype_cls")
149 exe "setf " .. g:filetype_cls
150 return
151 endif
152
Doug Kearnsf97f6bb2023-08-27 18:44:09 +0200153 var line1 = getline(1)
Doug Kearnse06afb72023-08-29 22:21:35 +0200154 if line1 =~ '^#!.*\<\%(rexx\|regina\)\>'
Bram Moolenaar8b5901e2022-06-29 14:39:12 +0100155 setf rexx
Doug Kearnse06afb72023-08-29 22:21:35 +0200156 return
Doug Kearnsf97f6bb2023-08-27 18:44:09 +0200157 elseif line1 == 'VERSION 1.0 CLASS'
Bram Moolenaar8b5901e2022-06-29 14:39:12 +0100158 setf vb
Doug Kearnse06afb72023-08-29 22:21:35 +0200159 return
160 endif
161
162 var nonblank1 = getline(nextnonblank(1))
163 if nonblank1 =~ '^\v%(\%|\\)'
164 setf tex
165 elseif nonblank1 =~ '^\s*\%(/\*\|::\w\)'
166 setf rexx
Bram Moolenaar8b5901e2022-06-29 14:39:12 +0100167 else
168 setf st
169 endif
170enddef
171
Bram Moolenaara2baa732022-02-04 16:09:54 +0000172export def FTlpc()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100173 if exists("g:lpc_syntax_for_c")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000174 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100175 while lnum <= 12
176 if getline(lnum) =~# '^\(//\|inherit\|private\|protected\|nosave\|string\|object\|mapping\|mixed\)'
177 setf lpc
178 return
179 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000180 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100181 endwhile
182 endif
183 setf c
Bram Moolenaara2baa732022-02-04 16:09:54 +0000184enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100185
Bram Moolenaara2baa732022-02-04 16:09:54 +0000186export def FTheader()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100187 if match(getline(1, min([line("$"), 200])), '^@\(interface\|end\|class\)') > -1
188 if exists("g:c_syntax_for_h")
189 setf objc
190 else
191 setf objcpp
192 endif
193 elseif exists("g:c_syntax_for_h")
194 setf c
195 elseif exists("g:ch_syntax_for_h")
196 setf ch
197 else
198 setf cpp
199 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000200enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100201
Bram Moolenaara2baa732022-02-04 16:09:54 +0000202# This function checks if one of the first ten lines start with a '@'. In
203# that case it is probably a change file.
204# If the first line starts with # or ! it's probably a ch file.
205# If a line has "main", "include", "//" or "/*" it's probably ch.
206# Otherwise CHILL is assumed.
207export def FTchange()
208 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100209 while lnum <= 10
210 if getline(lnum)[0] == '@'
211 setf change
212 return
213 endif
214 if lnum == 1 && (getline(1)[0] == '#' || getline(1)[0] == '!')
215 setf ch
216 return
217 endif
218 if getline(lnum) =~ "MODULE"
219 setf chill
220 return
221 endif
222 if getline(lnum) =~ 'main\s*(\|#\s*include\|//'
223 setf ch
224 return
225 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000226 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100227 endwhile
228 setf chill
Bram Moolenaara2baa732022-02-04 16:09:54 +0000229enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100230
Bram Moolenaara2baa732022-02-04 16:09:54 +0000231export def FTent()
232 # This function checks for valid cl syntax in the first five lines.
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100233 # Look for either an opening comment, '#', or a block start, '{'.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000234 # If not found, assume SGML.
235 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100236 while lnum < 6
Bram Moolenaara2baa732022-02-04 16:09:54 +0000237 var line = getline(lnum)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100238 if line =~ '^\s*[#{]'
239 setf cl
240 return
241 elseif line !~ '^\s*$'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000242 # Not a blank line, not a comment, and not a block start,
243 # so doesn't look like valid cl code.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100244 break
245 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000246 lnum += 1
Bram Moolenaarc12dc472022-03-05 13:45:56 +0000247 endwhile
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100248 setf dtd
Bram Moolenaara2baa732022-02-04 16:09:54 +0000249enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100250
Bram Moolenaara2baa732022-02-04 16:09:54 +0000251export def ExCheck()
252 var lines = getline(1, min([line("$"), 100]))
Austin Gatlinf3caeb62021-06-26 12:02:55 +0200253 if exists('g:filetype_euphoria')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000254 exe 'setf ' .. g:filetype_euphoria
Austin Gatlinf3caeb62021-06-26 12:02:55 +0200255 elseif match(lines, '^--\|^ifdef\>\|^include\>') > -1
256 setf euphoria3
257 else
258 setf elixir
259 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000260enddef
Austin Gatlinf3caeb62021-06-26 12:02:55 +0200261
Bram Moolenaara2baa732022-02-04 16:09:54 +0000262export def EuphoriaCheck()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100263 if exists('g:filetype_euphoria')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000264 exe 'setf ' .. g:filetype_euphoria
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100265 else
266 setf euphoria3
267 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000268enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100269
Bram Moolenaara2baa732022-02-04 16:09:54 +0000270export def DtraceCheck()
=?UTF-8?q?Teubel=20Gy=C3=B6rgy?=4d56b972022-02-24 17:59:09 +0000271 if did_filetype()
272 # Filetype was already detected
273 return
274 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000275 var lines = getline(1, min([line("$"), 100]))
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100276 if match(lines, '^module\>\|^import\>') > -1
Bram Moolenaara2baa732022-02-04 16:09:54 +0000277 # D files often start with a module and/or import statement.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100278 setf d
279 elseif match(lines, '^#!\S\+dtrace\|#pragma\s\+D\s\+option\|:\S\{-}:\S\{-}:') > -1
280 setf dtrace
281 else
282 setf d
283 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000284enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100285
Doug Kearns68a89472024-01-05 17:59:04 +0100286export def FTdef()
Wu, Zhenyu61ee8332024-04-09 22:09:30 +0200287 # LaTeX def files are usually generated by docstrip, which will output '%%' in first line
288 if getline(1) =~ '%%'
289 setf tex
290 endif
Doug Kearns68a89472024-01-05 17:59:04 +0100291 if get(g:, "filetype_def", "") == "modula2" || IsModula2()
292 SetFiletypeModula2()
293 return
294 endif
295
296 if exists("g:filetype_def")
297 exe "setf " .. g:filetype_def
298 else
299 setf def
300 endif
301enddef
302
Bram Moolenaara2baa732022-02-04 16:09:54 +0000303export def FTe()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100304 if exists('g:filetype_euphoria')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000305 exe 'setf ' .. g:filetype_euphoria
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100306 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000307 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100308 while n < 100 && n <= line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100309 if getline(n) =~ "^\\s*\\(<'\\|'>\\)\\s*$"
310 setf specman
311 return
312 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000313 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100314 endwhile
315 setf eiffel
316 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000317enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100318
Doug Kearns19a3bc32023-08-20 20:51:12 +0200319def IsForth(): bool
320 var first_line = nextnonblank(1)
321
322 # SwiftForth block comment (line is usually filled with '-' or '=') or
323 # OPTIONAL (sometimes precedes the header comment)
324 if getline(first_line) =~? '^\%({\%(\s\|$\)\|OPTIONAL\s\)'
325 return true
326 endif
327
328 var n = first_line
329 while n < 100 && n <= line("$")
330 # Forth comments and colon definitions
331 if getline(n) =~ '^[:(\\] '
332 return true
333 endif
334 n += 1
335 endwhile
336 return false
337enddef
338
339# Distinguish between Forth and Fortran
340export def FTf()
341 if exists("g:filetype_f")
342 exe "setf " .. g:filetype_f
343 elseif IsForth()
344 setf forth
345 else
346 setf fortran
347 endif
348enddef
349
Bram Moolenaara2baa732022-02-04 16:09:54 +0000350export def FTfrm()
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000351 if exists("g:filetype_frm")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000352 exe "setf " .. g:filetype_frm
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000353 return
354 endif
355
Doug Kearnsf97f6bb2023-08-27 18:44:09 +0200356 if getline(1) == "VERSION 5.00"
357 setf vb
358 return
359 endif
360
Bram Moolenaara2baa732022-02-04 16:09:54 +0000361 var lines = getline(1, min([line("$"), 5]))
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000362
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000363 if match(lines, ft_visual_basic_content) > -1
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000364 setf vb
365 else
366 setf form
367 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000368enddef
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000369
Doug Kearns19a3bc32023-08-20 20:51:12 +0200370# Distinguish between Forth and F#
Bram Moolenaara2baa732022-02-04 16:09:54 +0000371export def FTfs()
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000372 if exists("g:filetype_fs")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000373 exe "setf " .. g:filetype_fs
Doug Kearns19a3bc32023-08-20 20:51:12 +0200374 elseif IsForth()
375 setf forth
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000376 else
Johan Kotlinski065088d2023-04-02 20:29:38 +0100377 setf fsharp
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000378 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000379enddef
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000380
Amelia Clarke35dfe582024-05-24 08:05:00 +0200381# Recursively search for Hare source files in a directory and any
382# subdirectories, up to a given depth.
383def IsHareModule(dir: string, depth: number): bool
384 if depth <= 0
385 return !empty(glob(dir .. '/*.ha'))
386 endif
387
388 return reduce(sort(glob(dir .. '/*', true, true),
389 (a, b) => isdirectory(a) - isdirectory(b)),
390 (acc, n) => acc
391 || n =~ '\.ha$'
392 || isdirectory(n)
393 && IsHareModule(n, depth - 1),
394 false)
395enddef
396
397# Determine if a README file exists within a Hare module and should be given the
398# Haredoc filetype.
399export def FTharedoc()
400 if exists('g:filetype_haredoc')
401 if IsHareModule('<afile>:h', get(g:, 'haredoc_search_depth', 1))
402 setf haredoc
403 endif
404 endif
405enddef
406
Dennis van den Berg1ad194c2024-07-09 19:25:33 +0200407# Distinguish between HTML, XHTML, Django and Angular
Bram Moolenaara2baa732022-02-04 16:09:54 +0000408export def FThtml()
409 var n = 1
Dennis van den Berg1ad194c2024-07-09 19:25:33 +0200410
411 # Test if the filename follows the Angular component template convention
Christian Brabandtc03f6312024-07-10 19:23:39 +0200412 # Disabled for the reasons mentioned here: #13594
413 # if expand('%:t') =~ '^.*\.component\.html$'
414 # setf htmlangular
415 # return
416 # endif
Dennis van den Berg1ad194c2024-07-09 19:25:33 +0200417
Afiq Nazrie9718ed72024-06-18 19:59:53 +0200418 while n < 40 && n <= line("$")
Dennis van den Berg1ad194c2024-07-09 19:25:33 +0200419 # Check for Angular
420 if getline(n) =~ '@\(if\|for\|defer\|switch\)\|\*\(ngIf\|ngFor\|ngSwitch\|ngTemplateOutlet\)\|ng-template\|ng-content\|{{.*}}'
421 setf htmlangular
422 return
423 endif
424
425 # Check for XHTML
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100426 if getline(n) =~ '\<DTD\s\+XHTML\s'
427 setf xhtml
428 return
429 endif
Dennis van den Berg1ad194c2024-07-09 19:25:33 +0200430 # Check for Django
Afiq Nazrie9718ed72024-06-18 19:59:53 +0200431 if getline(n) =~ '{%\s*\(autoescape\|block\|comment\|csrf_token\|cycle\|debug\|extends\|filter\|firstof\|for\|if\|ifchanged\|include\|load\|lorem\|now\|query_string\|regroup\|resetcycle\|spaceless\|templatetag\|url\|verbatim\|widthratio\|with\)\>\|{#\s\+'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100432 setf htmldjango
433 return
434 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000435 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100436 endwhile
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100437 setf FALLBACK html
Bram Moolenaara2baa732022-02-04 16:09:54 +0000438enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100439
Bram Moolenaara2baa732022-02-04 16:09:54 +0000440# Distinguish between standard IDL and MS-IDL
441export def FTidl()
442 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100443 while n < 50 && n <= line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100444 if getline(n) =~ '^\s*import\s\+"\(unknwn\|objidl\)\.idl"'
445 setf msidl
446 return
447 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000448 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100449 endwhile
450 setf idl
Bram Moolenaara2baa732022-02-04 16:09:54 +0000451enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100452
Wu, Zhenyu887a38c2024-05-09 20:35:13 +0200453# Distinguish between "default", Prolog, zsh module's C and Cproto prototype file.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000454export def ProtoCheck(default: string)
Wu, Zhenyu887a38c2024-05-09 20:35:13 +0200455 # zsh modules use '#include "*.pro"'
456 # https://github.com/zsh-users/zsh/blob/63f086d167960a27ecdbcb762179e2c2bf8a29f5/Src/Modules/example.c#L31
457 if getline(1) =~ '/* Generated automatically */'
458 setf c
Bram Moolenaara2baa732022-02-04 16:09:54 +0000459 # Cproto files have a comment in the first line and a function prototype in
460 # the second line, it always ends in ";". Indent files may also have
461 # comments, thus we can't match comments to see the difference.
462 # IDL files can have a single ';' in the second line, require at least one
463 # chacter before the ';'.
Wu, Zhenyu887a38c2024-05-09 20:35:13 +0200464 elseif getline(2) =~ '.;$'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100465 setf cpp
466 else
Bram Moolenaar48c3f4e2022-08-08 15:42:38 +0100467 # recognize Prolog by specific text in the first non-empty line
468 # require a blank after the '%' because Perl uses "%list" and "%translate"
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100469 var lnum = getline(nextnonblank(1))
igna_martinoli37853b72024-07-18 21:34:36 +0200470 if lnum =~ '\<prolog\>' || lnum =~ prolog_pattern
Bram Moolenaar48c3f4e2022-08-08 15:42:38 +0100471 setf prolog
472 else
473 exe 'setf ' .. default
474 endif
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100475 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000476enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100477
Bram Moolenaara2baa732022-02-04 16:09:54 +0000478export def FTm()
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200479 if exists("g:filetype_m")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000480 exe "setf " .. g:filetype_m
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200481 return
482 endif
483
Bram Moolenaara2baa732022-02-04 16:09:54 +0000484 # excluding end(for|function|if|switch|while) common to Murphi
485 var octave_block_terminators = '\<end\%(_try_catch\|classdef\|enumeration\|events\|methods\|parfor\|properties\)\>'
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200486
Bram Moolenaara2baa732022-02-04 16:09:54 +0000487 var objc_preprocessor = '^\s*#\s*\%(import\|include\|define\|if\|ifn\=def\|undef\|line\|error\|pragma\)\>'
Doug Kearns7329cfa2021-11-26 13:01:41 +0000488
Bram Moolenaara2baa732022-02-04 16:09:54 +0000489 var n = 1
490 var saw_comment = 0 # Whether we've seen a multiline comment leader.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100491 while n < 100
Bram Moolenaara2baa732022-02-04 16:09:54 +0000492 var line = getline(n)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100493 if line =~ '^\s*/\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000494 # /* ... */ is a comment in Objective C and Murphi, so we can't conclude
495 # it's either of them yet, but track this as a hint in case we don't see
496 # anything more definitive.
497 saw_comment = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100498 endif
Doug Kearns7329cfa2021-11-26 13:01:41 +0000499 if line =~ '^\s*//' || line =~ '^\s*@import\>' || line =~ objc_preprocessor
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100500 setf objc
501 return
502 endif
Bram Moolenaarca0627d2021-09-12 17:03:08 +0200503 if line =~ '^\s*\%(#\|%!\)' || line =~ '^\s*unwind_protect\>' ||
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200504 \ line =~ '\%(^\|;\)\s*' .. octave_block_terminators
505 setf octave
506 return
507 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000508 # TODO: could be Matlab or Octave
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100509 if line =~ '^\s*%'
510 setf matlab
511 return
512 endif
513 if line =~ '^\s*(\*'
514 setf mma
515 return
516 endif
517 if line =~ '^\c\s*\(\(type\|var\)\>\|--\)'
518 setf murphi
519 return
520 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000521 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100522 endwhile
523
524 if saw_comment
Bram Moolenaara2baa732022-02-04 16:09:54 +0000525 # We didn't see anything definitive, but this looks like either Objective C
526 # or Murphi based on the comment leader. Assume the former as it is more
527 # common.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100528 setf objc
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100529 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000530 # Default is Matlab
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100531 setf matlab
532 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000533enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100534
Ken Takataeb4b9032024-07-25 21:07:13 +0200535export def FTmake()
536 # Check if it is a Microsoft Makefile
537 unlet! b:make_microsoft
538 var n = 1
539 while n < 1000 && n <= line('$')
540 var line = getline(n)
541 if line =~? '^\s*!\s*\(ifn\=\(def\)\=\|include\|message\|error\)\>'
542 b:make_microsoft = 1
543 break
544 elseif line =~ '^ *ifn\=\(eq\|def\)\>' || line =~ '^ *[-s]\=include\s'
545 break
546 elseif line =~ '^ *\w\+\s*[!?:+]='
547 break
548 endif
549 n += 1
550 endwhile
551 setf make
552enddef
553
Bram Moolenaara2baa732022-02-04 16:09:54 +0000554export def FTmms()
555 var n = 1
Bram Moolenaard7df2792020-01-02 21:34:42 +0100556 while n < 20
Bram Moolenaara2baa732022-02-04 16:09:54 +0000557 var line = getline(n)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100558 if line =~ '^\s*\(%\|//\)' || line =~ '^\*'
559 setf mmix
560 return
561 endif
562 if line =~ '^\s*#'
563 setf make
564 return
565 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000566 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100567 endwhile
568 setf mmix
Bram Moolenaara2baa732022-02-04 16:09:54 +0000569enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100570
Bram Moolenaara2baa732022-02-04 16:09:54 +0000571# This function checks if one of the first five lines start with a dot. In
572# that case it is probably an nroff file: 'filetype' is set and 1 is returned.
573export def FTnroff(): number
574 if getline(1)[0] .. getline(2)[0] .. getline(3)[0]
575 .. getline(4)[0] .. getline(5)[0] =~ '\.'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100576 setf nroff
577 return 1
578 endif
579 return 0
Bram Moolenaara2baa732022-02-04 16:09:54 +0000580enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100581
Bram Moolenaara2baa732022-02-04 16:09:54 +0000582export def FTmm()
583 var n = 1
Bram Moolenaard1caa942020-04-10 22:10:56 +0200584 while n < 20
Bram Moolenaara2baa732022-02-04 16:09:54 +0000585 if getline(n) =~ '^\s*\(#\s*\(include\|import\)\>\|@import\>\|/\*\)'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100586 setf objcpp
587 return
588 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000589 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100590 endwhile
591 setf nroff
Bram Moolenaara2baa732022-02-04 16:09:54 +0000592enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100593
Bram Moolenaarcdbfc6d2022-06-30 16:25:21 +0100594# Returns true if file content looks like LambdaProlog module
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100595def IsLProlog(): bool
Gaetan Lepage4ce1bda2023-05-10 22:01:55 +0100596 # skip apparent comments and blank lines, what looks like
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100597 # LambdaProlog comment may be RAPID header
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100598 var lnum: number = nextnonblank(1)
599 while lnum > 0 && lnum < line('$') && getline(lnum) =~ '^\s*%' # LambdaProlog comment
600 lnum = nextnonblank(lnum + 1)
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100601 endwhile
602 # this pattern must not catch a go.mod file
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100603 return getline(lnum) =~ '\<module\s\+\w\+\s*\.\s*\(%\|$\)'
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100604enddef
605
Doug Kearns68a89472024-01-05 17:59:04 +0100606def IsModula2(): bool
dkearnsef387c02024-02-20 06:58:30 +1100607 return getline(nextnonblank(1)) =~ '\<MODULE\s\+\w\+\s*\%(\[.*]\s*\)\=;\|^\s*(\*'
Doug Kearns68a89472024-01-05 17:59:04 +0100608enddef
609
610def SetFiletypeModula2()
611 const KNOWN_DIALECTS = ["iso", "pim", "r10"]
612 const KNOWN_EXTENSIONS = ["gm2"]
613 const LINE_COUNT = 200
614 const TAG = '(\*!m2\(\w\+\)\%(+\(\w\+\)\)\=\*)'
615
616 var dialect = get(g:, "modula2_default_dialect", "pim")
617 var extension = get(g:, "modula2_default_extension", "")
618
619 var matches = []
620
621 # ignore unknown dialects or badly formatted tags
622 for lnum in range(1, min([line("$"), LINE_COUNT]))
623 matches = matchlist(getline(lnum), TAG)
624 if !empty(matches)
625 if index(KNOWN_DIALECTS, matches[1]) >= 0
626 dialect = matches[1]
627 endif
628 if index(KNOWN_EXTENSIONS, matches[2]) >= 0
629 extension = matches[2]
630 endif
631 break
632 endif
633 endfor
634
635 modula2#SetDialect(dialect, extension)
636
637 setf modula2
638enddef
639
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100640# Determine if *.mod is ABB RAPID, LambdaProlog, Modula-2, Modsim III or go.mod
641export def FTmod()
Doug Kearns68a89472024-01-05 17:59:04 +0100642 if get(g:, "filetype_mod", "") == "modula2" || IsModula2()
643 SetFiletypeModula2()
644 return
645 endif
646
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100647 if exists("g:filetype_mod")
648 exe "setf " .. g:filetype_mod
Omar El Halabic9fbd252023-05-29 19:59:45 +0100649 elseif expand("<afile>") =~ '\<go.mod$'
650 setf gomod
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100651 elseif IsLProlog()
652 setf lprolog
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100653 elseif IsRapid()
654 setf rapid
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100655 else
656 # Nothing recognized, assume modsim3
657 setf modsim3
658 endif
659enddef
660
Bram Moolenaara2baa732022-02-04 16:09:54 +0000661export def FTpl()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100662 if exists("g:filetype_pl")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000663 exe "setf " .. g:filetype_pl
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100664 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000665 # recognize Prolog by specific text in the first non-empty line
666 # require a blank after the '%' because Perl uses "%list" and "%translate"
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100667 var line = getline(nextnonblank(1))
igna_martinoli37853b72024-07-18 21:34:36 +0200668 if line =~ '\<prolog\>' || line =~ prolog_pattern
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100669 setf prolog
670 else
671 setf perl
672 endif
673 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000674enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100675
Bram Moolenaara2baa732022-02-04 16:09:54 +0000676export def FTinc()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100677 if exists("g:filetype_inc")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000678 exe "setf " .. g:filetype_inc
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100679 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000680 var lines = getline(1) .. getline(2) .. getline(3)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100681 if lines =~? "perlscript"
682 setf aspperl
683 elseif lines =~ "<%"
684 setf aspvbs
685 elseif lines =~ "<?"
686 setf php
Bram Moolenaara2baa732022-02-04 16:09:54 +0000687 # Pascal supports // comments but they're vary rarely used for file
688 # headers so assume POV-Ray
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000689 elseif lines =~ '^\s*\%({\|(\*\)' || lines =~? ft_pascal_keywords
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100690 setf pascal
Gregory Anders30e212d2022-07-26 21:42:03 +0100691 elseif lines =~# '\<\%(require\|inherit\)\>' || lines =~# '[A-Z][A-Za-z0-9_:${}]*\s\+\%(??\|[?:+]\)\?= '
Gregory Andersfa49eb42022-07-16 17:46:47 +0100692 setf bitbake
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100693 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000694 FTasmsyntax()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100695 if exists("b:asmsyntax")
Gregory Andersfa49eb42022-07-16 17:46:47 +0100696 exe "setf " .. fnameescape(b:asmsyntax)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100697 else
Gregory Andersfa49eb42022-07-16 17:46:47 +0100698 setf pov
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100699 endif
700 endif
701 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000702enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100703
Bram Moolenaara2baa732022-02-04 16:09:54 +0000704export def FTprogress_cweb()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100705 if exists("g:filetype_w")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000706 exe "setf " .. g:filetype_w
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100707 return
708 endif
709 if getline(1) =~ '&ANALYZE' || getline(3) =~ '&GLOBAL-DEFINE'
710 setf progress
711 else
712 setf cweb
713 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000714enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100715
Julien Marrec2e310652023-11-25 15:30:46 +0100716# These include the leading '%' sign
717var 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\)'
718# This is the start/end of a block that is copied literally to the processor file (C/C++)
719var ft_swig_verbatim_block_start = '^\s*%{'
720
721export def FTi()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100722 if exists("g:filetype_i")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000723 exe "setf " .. g:filetype_i
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100724 return
725 endif
Julien Marrec2e310652023-11-25 15:30:46 +0100726 # 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 +0000727 # If not found, assume Progress.
728 var lnum = 1
Julien Marrec2e310652023-11-25 15:30:46 +0100729 while lnum <= 50 && lnum < line('$')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000730 var line = getline(lnum)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100731 if line =~ '^\s*;' || line =~ '^\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000732 FTasm()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100733 return
Julien Marrec2e310652023-11-25 15:30:46 +0100734 elseif line =~ ft_swig_keywords || line =~ ft_swig_verbatim_block_start
735 setf swig
736 return
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100737 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000738 lnum += 1
Bram Moolenaarc12dc472022-03-05 13:45:56 +0000739 endwhile
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100740 setf progress
Bram Moolenaara2baa732022-02-04 16:09:54 +0000741enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100742
Bram Moolenaara2baa732022-02-04 16:09:54 +0000743var ft_pascal_comments = '^\s*\%({\|(\*\|//\)'
744var ft_pascal_keywords = '^\s*\%(program\|unit\|library\|uses\|begin\|procedure\|function\|const\|type\|var\)\>'
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100745
Bram Moolenaara2baa732022-02-04 16:09:54 +0000746export def FTprogress_pascal()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100747 if exists("g:filetype_p")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000748 exe "setf " .. g:filetype_p
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100749 return
750 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000751 # This function checks for valid Pascal syntax in the first ten lines.
752 # Look for either an opening comment or a program start.
753 # If not found, assume Progress.
754 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100755 while lnum <= 10 && lnum < line('$')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000756 var line = getline(lnum)
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000757 if line =~ ft_pascal_comments || line =~? ft_pascal_keywords
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100758 setf pascal
759 return
760 elseif line !~ '^\s*$' || line =~ '^/\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000761 # Not an empty line: Doesn't look like valid Pascal code.
762 # Or it looks like a Progress /* comment
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100763 break
764 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000765 lnum += 1
Bram Moolenaarc12dc472022-03-05 13:45:56 +0000766 endwhile
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100767 setf progress
Bram Moolenaara2baa732022-02-04 16:09:54 +0000768enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100769
Bram Moolenaara2baa732022-02-04 16:09:54 +0000770export def FTpp()
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100771 if exists("g:filetype_pp")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000772 exe "setf " .. g:filetype_pp
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100773 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000774 var line = getline(nextnonblank(1))
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000775 if line =~ ft_pascal_comments || line =~? ft_pascal_keywords
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100776 setf pascal
777 else
778 setf puppet
779 endif
780 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000781enddef
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100782
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100783# Determine if *.prg is ABB RAPID. Can also be Clipper, FoxPro or eviews
784export def FTprg()
785 if exists("g:filetype_prg")
786 exe "setf " .. g:filetype_prg
787 elseif IsRapid()
788 setf rapid
789 else
790 # Nothing recognized, assume Clipper
791 setf clipper
792 endif
793enddef
794
Bram Moolenaara2baa732022-02-04 16:09:54 +0000795export def FTr()
796 var max = line("$") > 50 ? 50 : line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100797
798 for n in range(1, max)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000799 # Rebol is easy to recognize, check for that first
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100800 if getline(n) =~? '\<REBOL\>'
801 setf rebol
802 return
803 endif
804 endfor
805
806 for n in range(1, max)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000807 # R has # comments
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100808 if getline(n) =~ '^\s*#'
809 setf r
810 return
811 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000812 # Rexx has /* comments */
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100813 if getline(n) =~ '^\s*/\*'
814 setf rexx
815 return
816 endif
817 endfor
818
Bram Moolenaara2baa732022-02-04 16:09:54 +0000819 # Nothing recognized, use user default or assume Rexx
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100820 if exists("g:filetype_r")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000821 exe "setf " .. g:filetype_r
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100822 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000823 # Rexx used to be the default, but R appears to be much more popular.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100824 setf r
825 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000826enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100827
Bram Moolenaara2baa732022-02-04 16:09:54 +0000828export def McSetf()
829 # Rely on the file to start with a comment.
830 # MS message text files use ';', Sendmail files use '#' or 'dnl'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100831 for lnum in range(1, min([line("$"), 20]))
Bram Moolenaara2baa732022-02-04 16:09:54 +0000832 var line = getline(lnum)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100833 if line =~ '^\s*\(#\|dnl\)'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000834 setf m4 # Sendmail .mc file
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100835 return
836 elseif line =~ '^\s*;'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000837 setf msmessages # MS Message text file
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100838 return
839 endif
840 endfor
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100841 setf m4 # Default: Sendmail .mc file
Bram Moolenaara2baa732022-02-04 16:09:54 +0000842enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100843
Bram Moolenaara2baa732022-02-04 16:09:54 +0000844# Called from filetype.vim and scripts.vim.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100845# When "setft" is passed and false then the 'filetype' option is not set.
846export def SetFileTypeSH(name: string, setft = true): string
847 if setft && did_filetype()
Bram Moolenaara2baa732022-02-04 16:09:54 +0000848 # Filetype was already detected
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100849 return ''
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100850 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100851 if setft && expand("<amatch>") =~ g:ft_ignore_pat
852 return ''
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100853 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000854 if name =~ '\<csh\>'
855 # Some .sh scripts contain #!/bin/csh.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100856 return SetFileTypeShell("csh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000857 elseif name =~ '\<tcsh\>'
858 # Some .sh scripts contain #!/bin/tcsh.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100859 return SetFileTypeShell("tcsh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000860 elseif name =~ '\<zsh\>'
861 # Some .sh scripts contain #!/bin/zsh.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100862 return SetFileTypeShell("zsh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000863 elseif name =~ '\<ksh\>'
864 b:is_kornshell = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100865 if exists("b:is_bash")
866 unlet b:is_bash
867 endif
868 if exists("b:is_sh")
869 unlet b:is_sh
870 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000871 elseif exists("g:bash_is_sh") || name =~ '\<bash\>' || name =~ '\<bash2\>'
872 b:is_bash = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100873 if exists("b:is_kornshell")
874 unlet b:is_kornshell
875 endif
876 if exists("b:is_sh")
877 unlet b:is_sh
878 endif
Eisuke Kawashima24482fb2022-11-24 10:58:10 +0000879 elseif name =~ '\<sh\>' || name =~ '\<dash\>'
880 # Ubuntu links "sh" to "dash", thus it is expected to work the same way
Bram Moolenaara2baa732022-02-04 16:09:54 +0000881 b:is_sh = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100882 if exists("b:is_kornshell")
883 unlet b:is_kornshell
884 endif
885 if exists("b:is_bash")
886 unlet b:is_bash
887 endif
888 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100889
890 return SetFileTypeShell("sh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000891enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100892
Bram Moolenaara2baa732022-02-04 16:09:54 +0000893# For shell-like file types, check for an "exec" command hidden in a comment,
894# as used for Tcl.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100895# When "setft" is passed and false then the 'filetype' option is not set.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000896# Also called from scripts.vim, thus can't be local to this script.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100897export def SetFileTypeShell(name: string, setft = true): string
898 if setft && did_filetype()
Bram Moolenaara2baa732022-02-04 16:09:54 +0000899 # Filetype was already detected
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100900 return ''
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100901 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100902 if setft && expand("<amatch>") =~ g:ft_ignore_pat
903 return ''
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100904 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100905
906 var lnum = 2
907 while lnum < 20 && lnum < line("$") && getline(lnum) =~ '^\s*\(#\|$\)'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000908 # Skip empty and comment lines.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100909 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100910 endwhile
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100911 if lnum < line("$") && getline(lnum) =~ '\s*exec\s' && getline(lnum - 1) =~ '^\s*#.*\\$'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000912 # Found an "exec" line after a comment with continuation
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100913 var n = substitute(getline(lnum), '\s*exec\s\+\([^ ]*/\)\=', '', '')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100914 if n =~ '\<tclsh\|\<wish'
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100915 if setft
916 setf tcl
917 endif
918 return 'tcl'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100919 endif
920 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100921
922 if setft
923 exe "setf " .. name
924 endif
925 return name
Bram Moolenaara2baa732022-02-04 16:09:54 +0000926enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100927
Bram Moolenaara2baa732022-02-04 16:09:54 +0000928export def CSH()
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100929 if did_filetype()
Bram Moolenaara2baa732022-02-04 16:09:54 +0000930 # Filetype was already detected
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100931 return
932 endif
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100933 if exists("g:filetype_csh")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000934 SetFileTypeShell(g:filetype_csh)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100935 elseif &shell =~ "tcsh"
Bram Moolenaara2baa732022-02-04 16:09:54 +0000936 SetFileTypeShell("tcsh")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100937 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000938 SetFileTypeShell("csh")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100939 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000940enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100941
Bram Moolenaara2baa732022-02-04 16:09:54 +0000942var ft_rules_udev_rules_pattern = '^\s*\cudev_rules\s*=\s*"\([^"]\{-1,}\)/*".*'
943export def FTRules()
944 var path = expand('<amatch>:p')
Bram Moolenaaraa9675a2020-08-17 21:57:09 +0200945 if path =~ '/\(etc/udev/\%(rules\.d/\)\=.*\.rules\|\%(usr/\)\=lib/udev/\%(rules\.d/\)\=.*\.rules\)$'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100946 setf udevrules
947 return
948 endif
949 if path =~ '^/etc/ufw/'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000950 setf conf # Better than hog
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100951 return
952 endif
953 if path =~ '^/\(etc\|usr/share\)/polkit-1/rules\.d'
954 setf javascript
955 return
956 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000957 var config_lines: list<string>
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100958 try
Bram Moolenaara2baa732022-02-04 16:09:54 +0000959 config_lines = readfile('/etc/udev/udev.conf')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100960 catch /^Vim\%((\a\+)\)\=:E484/
961 setf hog
962 return
963 endtry
Bram Moolenaara2baa732022-02-04 16:09:54 +0000964 var dir = expand('<amatch>:p:h')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100965 for line in config_lines
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000966 if line =~ ft_rules_udev_rules_pattern
967 var udev_rules = substitute(line, ft_rules_udev_rules_pattern, '\1', "")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100968 if dir == udev_rules
969 setf udevrules
970 endif
971 break
972 endif
973 endfor
974 setf hog
Bram Moolenaara2baa732022-02-04 16:09:54 +0000975enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100976
Bram Moolenaara2baa732022-02-04 16:09:54 +0000977export def SQL()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100978 if exists("g:filetype_sql")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000979 exe "setf " .. g:filetype_sql
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100980 else
981 setf sql
982 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000983enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100984
ranjithshegde8cac20e2022-04-13 15:29:21 +0100985# This function checks the first 25 lines of file extension "sc" to resolve
Chris Kipp70ef3f52022-12-14 16:42:15 +0000986# detection between scala and SuperCollider.
987# NOTE: We don't check for 'Class : Method', as this can easily be confused
988# with valid Scala like `val x : Int = 3`. So we instead only rely on
989# checks that can't be confused.
ranjithshegde8cac20e2022-04-13 15:29:21 +0100990export def FTsc()
991 for lnum in range(1, min([line("$"), 25]))
Chris Kipp70ef3f52022-12-14 16:42:15 +0000992 if getline(lnum) =~# 'var\s<\|classvar\s<\|\^this.*\||\w\+|\|+\s\w*\s{\|\*ar\s'
ranjithshegde8cac20e2022-04-13 15:29:21 +0100993 setf supercollider
994 return
995 endif
996 endfor
997 setf scala
998enddef
999
1000# This function checks the first line of file extension "scd" to resolve
1001# detection between scdoc and SuperCollider
1002export def FTscd()
1003 if getline(1) =~# '\%^\S\+(\d[0-9A-Za-z]*)\%(\s\+\"[^"]*\"\%(\s\+\"[^"]*\"\)\=\)\=$'
1004 setf scdoc
1005 else
1006 setf supercollider
1007 endif
1008enddef
1009
Bram Moolenaara2baa732022-02-04 16:09:54 +00001010# If the file has an extension of 't' and is in a directory 't' or 'xt' then
1011# it is almost certainly a Perl test file.
1012# If the first line starts with '#' and contains 'perl' it's probably a Perl
1013# file.
1014# (Slow test) If a file contains a 'use' statement then it is almost certainly
1015# a Perl file.
1016export def FTperl(): number
1017 var dirname = expand("%:p:h:t")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001018 if expand("%:e") == 't' && (dirname == 't' || dirname == 'xt')
1019 setf perl
1020 return 1
1021 endif
1022 if getline(1)[0] == '#' && getline(1) =~ 'perl'
1023 setf perl
1024 return 1
1025 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001026 var save_cursor = getpos('.')
1027 call cursor(1, 1)
Bram Moolenaare5b78972022-02-05 19:50:34 +00001028 var has_use = search('^use\s\s*\k', 'c', 30) > 0
Bram Moolenaarf0b03c42017-12-17 17:17:07 +01001029 call setpos('.', save_cursor)
1030 if has_use
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001031 setf perl
1032 return 1
1033 endif
1034 return 0
Bram Moolenaara2baa732022-02-04 16:09:54 +00001035enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001036
Bram Moolenaarcdbfc6d2022-06-30 16:25:21 +01001037# LambdaProlog and Standard ML signature files
1038export def FTsig()
1039 if exists("g:filetype_sig")
1040 exe "setf " .. g:filetype_sig
1041 return
1042 endif
1043
1044 var lprolog_comment = '^\s*\%(/\*\|%\)'
1045 var lprolog_keyword = '^\s*sig\s\+\a'
1046 var sml_comment = '^\s*(\*'
1047 var sml_keyword = '^\s*\%(signature\|structure\)\s\+\a'
1048
1049 var line = getline(nextnonblank(1))
1050
1051 if line =~ lprolog_comment || line =~# lprolog_keyword
1052 setf lprolog
1053 elseif line =~ sml_comment || line =~# sml_keyword
1054 setf sml
1055 endif
1056enddef
1057
Bram Moolenaarbe807d52022-09-01 15:01:25 +01001058# This function checks the first 100 lines of files matching "*.sil" to
1059# resolve detection between Swift Intermediate Language and SILE.
1060export def FTsil()
1061 for lnum in range(1, [line('$'), 100]->min())
1062 var line: string = getline(lnum)
1063 if line =~ '^\s*[\\%]'
1064 setf sile
1065 return
1066 elseif line =~ '^\s*\S'
1067 setf sil
1068 return
1069 endif
1070 endfor
1071 # no clue, default to "sil"
1072 setf sil
1073enddef
1074
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +01001075export def FTsys()
KnoP-01f420ff22022-04-13 20:46:21 +01001076 if exists("g:filetype_sys")
1077 exe "setf " .. g:filetype_sys
1078 elseif IsRapid()
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +01001079 setf rapid
1080 else
1081 setf bat
1082 endif
1083enddef
1084
Bram Moolenaara2baa732022-02-04 16:09:54 +00001085# Choose context, plaintex, or tex (LaTeX) based on these rules:
1086# 1. Check the first line of the file for "%&<format>".
1087# 2. Check the first 1000 non-comment lines for LaTeX or ConTeXt keywords.
1088# 3. Default to "plain" or to g:tex_flavor, can be set in user's vimrc.
1089export def FTtex()
1090 var firstline = getline(1)
1091 var format: string
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001092 if firstline =~ '^%&\s*\a\+'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001093 format = tolower(matchstr(firstline, '\a\+'))
1094 format = substitute(format, 'pdf', '', '')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001095 if format == 'tex'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001096 format = 'latex'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001097 elseif format == 'plaintex'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001098 format = 'plain'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001099 endif
1100 elseif expand('%') =~ 'tex/context/.*/.*.tex'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001101 format = 'context'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001102 else
Bram Moolenaara2baa732022-02-04 16:09:54 +00001103 # Default value, may be changed later:
1104 format = exists("g:tex_flavor") ? g:tex_flavor : 'plain'
1105 # Save position, go to the top of the file, find first non-comment line.
1106 var save_cursor = getpos('.')
1107 call cursor(1, 1)
1108 var firstNC = search('^\s*[^[:space:]%]', 'c', 1000)
Bram Moolenaare5b78972022-02-05 19:50:34 +00001109 if firstNC > 0
1110 # Check the next thousand lines for a LaTeX or ConTeXt keyword.
Bram Moolenaara2baa732022-02-04 16:09:54 +00001111 var lpat = 'documentclass\>\|usepackage\>\|begin{\|newcommand\>\|renewcommand\>'
1112 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\>'
1113 var kwline = search('^\s*\\\%(' .. lpat .. '\)\|^\s*\\\(' .. cpat .. '\)',
1114 'cnp', firstNC + 1000)
1115 if kwline == 1 # lpat matched
1116 format = 'latex'
1117 elseif kwline == 2 # cpat matched
1118 format = 'context'
1119 endif # If neither matched, keep default set above.
1120 # let lline = search('^\s*\\\%(' . lpat . '\)', 'cn', firstNC + 1000)
1121 # let cline = search('^\s*\\\%(' . cpat . '\)', 'cn', firstNC + 1000)
1122 # if cline > 0
1123 # let format = 'context'
1124 # endif
1125 # if lline > 0 && (cline == 0 || cline > lline)
1126 # let format = 'tex'
1127 # endif
1128 endif # firstNC
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001129 call setpos('.', save_cursor)
Bram Moolenaara2baa732022-02-04 16:09:54 +00001130 endif # firstline =~ '^%&\s*\a\+'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001131
Bram Moolenaara2baa732022-02-04 16:09:54 +00001132 # Translation from formats to file types. TODO: add AMSTeX, RevTex, others?
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001133 if format == 'plain'
1134 setf plaintex
1135 elseif format == 'context'
1136 setf context
Bram Moolenaara2baa732022-02-04 16:09:54 +00001137 else # probably LaTeX
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001138 setf tex
1139 endif
1140 return
Bram Moolenaara2baa732022-02-04 16:09:54 +00001141enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001142
Bram Moolenaara2baa732022-02-04 16:09:54 +00001143export def FTxml()
1144 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +01001145 while n < 100 && n <= line("$")
Bram Moolenaara2baa732022-02-04 16:09:54 +00001146 var line = getline(n)
1147 # DocBook 4 or DocBook 5.
1148 var is_docbook4 = line =~ '<!DOCTYPE.*DocBook'
1149 var is_docbook5 = line =~ ' xmlns="http://docbook.org/ns/docbook"'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001150 if is_docbook4 || is_docbook5
Bram Moolenaara2baa732022-02-04 16:09:54 +00001151 b:docbk_type = "xml"
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001152 if is_docbook5
Bram Moolenaara2baa732022-02-04 16:09:54 +00001153 b:docbk_ver = 5
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001154 else
Bram Moolenaara2baa732022-02-04 16:09:54 +00001155 b:docbk_ver = 4
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001156 endif
1157 setf docbk
1158 return
1159 endif
1160 if line =~ 'xmlns:xbl="http://www.mozilla.org/xbl"'
1161 setf xbl
1162 return
1163 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001164 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001165 endwhile
1166 setf xml
Bram Moolenaara2baa732022-02-04 16:09:54 +00001167enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001168
Bram Moolenaara2baa732022-02-04 16:09:54 +00001169export def FTy()
1170 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +01001171 while n < 100 && n <= line("$")
Bram Moolenaara2baa732022-02-04 16:09:54 +00001172 var line = getline(n)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001173 if line =~ '^\s*%'
1174 setf yacc
1175 return
1176 endif
1177 if getline(n) =~ '^\s*\(#\|class\>\)' && getline(n) !~ '^\s*#\s*include'
1178 setf racc
1179 return
1180 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001181 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001182 endwhile
1183 setf yacc
Bram Moolenaara2baa732022-02-04 16:09:54 +00001184enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001185
Bram Moolenaara2baa732022-02-04 16:09:54 +00001186export def Redif()
1187 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001188 while lnum <= 5 && lnum < line('$')
1189 if getline(lnum) =~ "^\ctemplate-type:"
1190 setf redif
1191 return
1192 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001193 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001194 endwhile
Bram Moolenaara2baa732022-02-04 16:09:54 +00001195enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001196
Bram Moolenaara2baa732022-02-04 16:09:54 +00001197# This function is called for all files under */debian/patches/*, make sure not
1198# to non-dep3patch files, such as README and other text files.
1199export def Dep3patch()
James McCoy647ab4c2021-12-17 20:52:57 +00001200 if expand('%:t') ==# 'series'
1201 return
1202 endif
1203
1204 for ln in getline(1, 100)
1205 if ln =~# '^\%(Description\|Subject\|Origin\|Bug\|Forwarded\|Author\|From\|Reviewed-by\|Acked-by\|Last-Updated\|Applied-Upstream\):'
1206 setf dep3patch
1207 return
1208 elseif ln =~# '^---'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001209 # end of headers found. stop processing
James McCoy647ab4c2021-12-17 20:52:57 +00001210 return
1211 endif
1212 endfor
Bram Moolenaara2baa732022-02-04 16:09:54 +00001213enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001214
Bram Moolenaara2baa732022-02-04 16:09:54 +00001215# This function checks the first 15 lines for appearance of 'FoamFile'
1216# and then 'object' in a following line.
1217# In that case, it's probably an OpenFOAM file
1218export def FTfoam()
1219 var ffile = 0
1220 var lnum = 1
Elwardi2284f6c2022-01-11 18:14:23 +00001221 while lnum <= 15
1222 if getline(lnum) =~# '^FoamFile'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001223 ffile = 1
Elwardi2284f6c2022-01-11 18:14:23 +00001224 elseif ffile == 1 && getline(lnum) =~# '^\s*object'
1225 setf foam
1226 return
1227 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001228 lnum += 1
Elwardi2284f6c2022-01-11 18:14:23 +00001229 endwhile
Bram Moolenaara2baa732022-02-04 16:09:54 +00001230enddef
Elwardi2284f6c2022-01-11 18:14:23 +00001231
Bram Moolenaara2baa732022-02-04 16:09:54 +00001232# Determine if a *.tf file is TF mud client or terraform
1233export def FTtf()
1234 var numberOfLines = line('$')
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001235 for i in range(1, numberOfLines)
Bram Moolenaara2baa732022-02-04 16:09:54 +00001236 var currentLine = trim(getline(i))
1237 var firstCharacter = currentLine[0]
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001238 if firstCharacter !=? ";" && firstCharacter !=? "/" && firstCharacter !=? ""
1239 setf terraform
1240 return
1241 endif
1242 endfor
1243 setf tf
Bram Moolenaara2baa732022-02-04 16:09:54 +00001244enddef
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001245
KnoP-0193c7a452022-04-16 21:14:04 +01001246var ft_krl_header = '\&\w+'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001247# Determine if a *.src file is Kuka Robot Language
1248export def FTsrc()
KnoP-0193c7a452022-04-16 21:14:04 +01001249 var ft_krl_def_or_deffct = '%(global\s+)?def%(fct)?>'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001250 if exists("g:filetype_src")
1251 exe "setf " .. g:filetype_src
KnoP-0193c7a452022-04-16 21:14:04 +01001252 elseif getline(nextnonblank(1)) =~? '\v^\s*%(' .. ft_krl_header .. '|' .. ft_krl_def_or_deffct .. ')'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001253 setf krl
1254 endif
1255enddef
1256
1257# Determine if a *.dat file is Kuka Robot Language
1258export def FTdat()
KnoP-0193c7a452022-04-16 21:14:04 +01001259 var ft_krl_defdat = 'defdat>'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001260 if exists("g:filetype_dat")
1261 exe "setf " .. g:filetype_dat
KnoP-0193c7a452022-04-16 21:14:04 +01001262 elseif getline(nextnonblank(1)) =~? '\v^\s*%(' .. ft_krl_header .. '|' .. ft_krl_defdat .. ')'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001263 setf krl
1264 endif
1265enddef
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001266
Doug Kearns4ac8e792022-10-17 13:32:17 +01001267export def FTlsl()
1268 if exists("g:filetype_lsl")
1269 exe "setf " .. g:filetype_lsl
1270 endif
1271
1272 var line = getline(nextnonblank(1))
1273 if line =~ '^\s*%' || line =~# ':\s*trait\s*$'
1274 setf larch
1275 else
1276 setf lsl
1277 endif
1278enddef
1279
Gaetan Lepage4ce1bda2023-05-10 22:01:55 +01001280export def FTtyp()
1281 if exists("g:filetype_typ")
1282 exe "setf " .. g:filetype_typ
1283 return
1284 endif
1285
1286 # Look for SQL type definition syntax
1287 for line in getline(1, 200)
1288 # SQL type files may define the casing
1289 if line =~ '^CASE\s\==\s\=\(SAME\|LOWER\|UPPER\|OPPOSITE\)$'
1290 setf sql
1291 return
1292 endif
1293
1294 # SQL type files may define some types as follows
1295 if line =~ '^TYPE\s.*$'
1296 setf sql
1297 return
1298 endif
1299 endfor
1300
1301 # Otherwise, affect the typst filetype
1302 setf typst
1303enddef
1304
PowerUser64aa61b8a2024-06-19 20:32:11 +02001305# Detect Microsoft Developer Studio Project files (Makefile) or Faust DSP
1306# files.
1307export def FTdsp()
1308 if exists("g:filetype_dsp")
1309 exe "setf " .. g:filetype_dsp
1310 return
1311 endif
1312
1313 # Test the filename
1314 if expand('%:t') =~ '^[mM]akefile.*$'
1315 setf make
1316 return
1317 endif
1318
1319 # Test the file contents
1320 for line in getline(1, 200)
1321 # Chech for comment style
1322 if line =~ '^#.*'
1323 setf make
1324 return
1325 endif
1326
1327 # Check for common lines
1328 if line =~ '^.*Microsoft Developer Studio Project File.*$'
1329 setf make
1330 return
1331 endif
1332
1333 if line =~ '^!MESSAGE This is not a valid makefile\..+$'
1334 setf make
1335 return
1336 endif
1337
1338 # Check for keywords
1339 if line =~ '^!(IF,ELSEIF,ENDIF).*$'
1340 setf make
1341 return
1342 endif
1343
1344 # Check for common assignments
1345 if line =~ '^SOURCE=.*$'
1346 setf make
1347 return
1348 endif
1349 endfor
1350
1351 # Otherwise, assume we have a Faust file
1352 setf faust
1353enddef
1354
Turiiya80406c22023-04-22 21:38:47 +01001355# Set the filetype of a *.v file to Verilog, V or Cog based on the first 200
1356# lines.
1357export def FTv()
1358 if did_filetype()
1359 # ":setf" will do nothing, bail out early
1360 return
1361 endif
Christian Brabandt10b4f752024-01-01 19:19:20 +01001362 if exists("g:filetype_v")
1363 exe "setf " .. g:filetype_v
1364 return
1365 endif
Turiiya80406c22023-04-22 21:38:47 +01001366
Christian Brabandt10b4f752024-01-01 19:19:20 +01001367 var in_comment = 0
1368 for lnum in range(1, min([line("$"), 200]))
1369 var line = getline(lnum)
1370 # Skip Verilog and V comments (lines and blocks).
1371 if line =~ '^\s*/\*'
1372 # start comment block
1373 in_comment = 1
1374 endif
1375 if in_comment == 1
1376 if line =~ '\*/'
1377 # end comment block
1378 in_comment = 0
1379 endif
1380 # skip comment-block line
1381 continue
1382 endif
1383 if line =~ '^\s*//'
Turiiya80406c22023-04-22 21:38:47 +01001384 # skip comment line
1385 continue
1386 endif
1387
Christian Brabandt10b4f752024-01-01 19:19:20 +01001388 # Coq: line ends with a '.' followed by an optional variable number of
1389 # spaces or contains the start of a comment, but not inside a Verilog or V
1390 # comment.
1391 # Example: "Definition x := 10. (*".
1392 if (line =~ '\.\s*$' && line !~ '/[/*]') || (line =~ '(\*' && line !~ '/[/*].*(\*')
1393 setf coq
1394 return
1395 endif
1396
Turiiya80406c22023-04-22 21:38:47 +01001397 # Verilog: line ends with ';' followed by an optional variable number of
1398 # spaces and an optional start of a comment.
1399 # Example: " b <= a + 1; // Add 1".
Christian Brabandt10b4f752024-01-01 19:19:20 +01001400 if line =~ ';\s*\(/[/*].*\)\?$'
Turiiya80406c22023-04-22 21:38:47 +01001401 setf verilog
1402 return
1403 endif
Turiiya80406c22023-04-22 21:38:47 +01001404 endfor
1405
1406 # No line matched, fall back to "v".
1407 setf v
1408enddef
1409
Doug Kearnsf97f6bb2023-08-27 18:44:09 +02001410export def FTvba()
1411 if getline(1) =~ '^["#] Vimball Archiver'
1412 setf vim
1413 else
1414 setf vb
1415 endif
1416enddef
1417
Colin Caine4b3fab12024-04-18 23:53:02 +02001418export def Detect_UCI_statements(): bool
1419 # Match a config or package statement at the start of the line.
1420 const config_or_package_statement = '^\s*\(\(c\|config\)\|\(p\|package\)\)\s\+\S'
1421 # Match a line that is either all blank or blank followed by a comment
1422 const comment_or_blank = '^\s*\(#.*\)\?$'
1423
1424 # Return true iff the file has a config or package statement near the
1425 # top of the file and all preceding lines were comments or blank.
1426 return getline(1) =~# config_or_package_statement
1427 \ || getline(1) =~# comment_or_blank
1428 \ && ( getline(2) =~# config_or_package_statement
1429 \ || getline(2) =~# comment_or_blank
1430 \ && getline(3) =~# config_or_package_statement
1431 \ )
1432enddef
1433
Bram Moolenaara2baa732022-02-04 16:09:54 +00001434# Uncomment this line to check for compilation errors early
dkearnsef387c02024-02-20 06:58:30 +11001435# defcompile