blob: ce7f44fa65ac8cb0202e5c907d0708f4e8650270 [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
Wu, Zhenyue2c27ca2024-11-19 20:55:25 +0100147export def FTcl()
148 if join(getline(1, 4), '') =~ '/\*'
149 setf opencl
150 else
151 setf lisp
152 endif
153enddef
154
Bram Moolenaar8b5901e2022-06-29 14:39:12 +0100155export def FTcls()
156 if exists("g:filetype_cls")
157 exe "setf " .. g:filetype_cls
158 return
159 endif
160
Doug Kearnsf97f6bb2023-08-27 18:44:09 +0200161 var line1 = getline(1)
Doug Kearnse06afb72023-08-29 22:21:35 +0200162 if line1 =~ '^#!.*\<\%(rexx\|regina\)\>'
Bram Moolenaar8b5901e2022-06-29 14:39:12 +0100163 setf rexx
Doug Kearnse06afb72023-08-29 22:21:35 +0200164 return
Doug Kearnsf97f6bb2023-08-27 18:44:09 +0200165 elseif line1 == 'VERSION 1.0 CLASS'
Bram Moolenaar8b5901e2022-06-29 14:39:12 +0100166 setf vb
Doug Kearnse06afb72023-08-29 22:21:35 +0200167 return
168 endif
169
170 var nonblank1 = getline(nextnonblank(1))
171 if nonblank1 =~ '^\v%(\%|\\)'
172 setf tex
173 elseif nonblank1 =~ '^\s*\%(/\*\|::\w\)'
174 setf rexx
Bram Moolenaar8b5901e2022-06-29 14:39:12 +0100175 else
176 setf st
177 endif
178enddef
179
Wu, Zhenyubc32bbd2024-11-14 22:55:36 +0100180export def FTll()
181 if getline(1) =~ ';\|\<source_filename\>\|\<target\>'
182 setf llvm
183 else
184 setf lifelines
185 endif
186enddef
187
Bram Moolenaara2baa732022-02-04 16:09:54 +0000188export def FTlpc()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100189 if exists("g:lpc_syntax_for_c")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000190 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100191 while lnum <= 12
192 if getline(lnum) =~# '^\(//\|inherit\|private\|protected\|nosave\|string\|object\|mapping\|mixed\)'
193 setf lpc
194 return
195 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000196 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100197 endwhile
198 endif
199 setf c
Bram Moolenaara2baa732022-02-04 16:09:54 +0000200enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100201
Bram Moolenaara2baa732022-02-04 16:09:54 +0000202export def FTheader()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100203 if match(getline(1, min([line("$"), 200])), '^@\(interface\|end\|class\)') > -1
204 if exists("g:c_syntax_for_h")
205 setf objc
206 else
207 setf objcpp
208 endif
209 elseif exists("g:c_syntax_for_h")
210 setf c
211 elseif exists("g:ch_syntax_for_h")
212 setf ch
213 else
214 setf cpp
215 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000216enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100217
Bram Moolenaara2baa732022-02-04 16:09:54 +0000218# This function checks if one of the first ten lines start with a '@'. In
219# that case it is probably a change file.
220# If the first line starts with # or ! it's probably a ch file.
221# If a line has "main", "include", "//" or "/*" it's probably ch.
222# Otherwise CHILL is assumed.
223export def FTchange()
224 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100225 while lnum <= 10
226 if getline(lnum)[0] == '@'
227 setf change
228 return
229 endif
230 if lnum == 1 && (getline(1)[0] == '#' || getline(1)[0] == '!')
231 setf ch
232 return
233 endif
234 if getline(lnum) =~ "MODULE"
235 setf chill
236 return
237 endif
238 if getline(lnum) =~ 'main\s*(\|#\s*include\|//'
239 setf ch
240 return
241 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000242 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100243 endwhile
244 setf chill
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 FTent()
248 # This function checks for valid cl syntax in the first five lines.
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100249 # Look for either an opening comment, '#', or a block start, '{'.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000250 # If not found, assume SGML.
251 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100252 while lnum < 6
Bram Moolenaara2baa732022-02-04 16:09:54 +0000253 var line = getline(lnum)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100254 if line =~ '^\s*[#{]'
255 setf cl
256 return
257 elseif line !~ '^\s*$'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000258 # Not a blank line, not a comment, and not a block start,
259 # so doesn't look like valid cl code.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100260 break
261 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000262 lnum += 1
Bram Moolenaarc12dc472022-03-05 13:45:56 +0000263 endwhile
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100264 setf dtd
Bram Moolenaara2baa732022-02-04 16:09:54 +0000265enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100266
Bram Moolenaara2baa732022-02-04 16:09:54 +0000267export def ExCheck()
268 var lines = getline(1, min([line("$"), 100]))
Austin Gatlinf3caeb62021-06-26 12:02:55 +0200269 if exists('g:filetype_euphoria')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000270 exe 'setf ' .. g:filetype_euphoria
Austin Gatlinf3caeb62021-06-26 12:02:55 +0200271 elseif match(lines, '^--\|^ifdef\>\|^include\>') > -1
272 setf euphoria3
273 else
274 setf elixir
275 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000276enddef
Austin Gatlinf3caeb62021-06-26 12:02:55 +0200277
Bram Moolenaara2baa732022-02-04 16:09:54 +0000278export def EuphoriaCheck()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100279 if exists('g:filetype_euphoria')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000280 exe 'setf ' .. g:filetype_euphoria
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100281 else
282 setf euphoria3
283 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000284enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100285
Bram Moolenaara2baa732022-02-04 16:09:54 +0000286export def DtraceCheck()
=?UTF-8?q?Teubel=20Gy=C3=B6rgy?=4d56b972022-02-24 17:59:09 +0000287 if did_filetype()
288 # Filetype was already detected
289 return
290 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000291 var lines = getline(1, min([line("$"), 100]))
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100292 if match(lines, '^module\>\|^import\>') > -1
Bram Moolenaara2baa732022-02-04 16:09:54 +0000293 # D files often start with a module and/or import statement.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100294 setf d
295 elseif match(lines, '^#!\S\+dtrace\|#pragma\s\+D\s\+option\|:\S\{-}:\S\{-}:') > -1
296 setf dtrace
297 else
298 setf d
299 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000300enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100301
Doug Kearns68a89472024-01-05 17:59:04 +0100302export def FTdef()
Wu, Zhenyu61ee8332024-04-09 22:09:30 +0200303 # LaTeX def files are usually generated by docstrip, which will output '%%' in first line
304 if getline(1) =~ '%%'
305 setf tex
306 endif
Doug Kearns68a89472024-01-05 17:59:04 +0100307 if get(g:, "filetype_def", "") == "modula2" || IsModula2()
308 SetFiletypeModula2()
309 return
310 endif
311
312 if exists("g:filetype_def")
313 exe "setf " .. g:filetype_def
314 else
315 setf def
316 endif
317enddef
318
Bram Moolenaara2baa732022-02-04 16:09:54 +0000319export def FTe()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100320 if exists('g:filetype_euphoria')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000321 exe 'setf ' .. g:filetype_euphoria
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100322 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000323 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100324 while n < 100 && n <= line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100325 if getline(n) =~ "^\\s*\\(<'\\|'>\\)\\s*$"
326 setf specman
327 return
328 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000329 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100330 endwhile
331 setf eiffel
332 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000333enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100334
Doug Kearns19a3bc32023-08-20 20:51:12 +0200335def IsForth(): bool
336 var first_line = nextnonblank(1)
337
338 # SwiftForth block comment (line is usually filled with '-' or '=') or
339 # OPTIONAL (sometimes precedes the header comment)
340 if getline(first_line) =~? '^\%({\%(\s\|$\)\|OPTIONAL\s\)'
341 return true
342 endif
343
344 var n = first_line
345 while n < 100 && n <= line("$")
346 # Forth comments and colon definitions
347 if getline(n) =~ '^[:(\\] '
348 return true
349 endif
350 n += 1
351 endwhile
352 return false
353enddef
354
355# Distinguish between Forth and Fortran
356export def FTf()
357 if exists("g:filetype_f")
358 exe "setf " .. g:filetype_f
359 elseif IsForth()
360 setf forth
361 else
362 setf fortran
363 endif
364enddef
365
Bram Moolenaara2baa732022-02-04 16:09:54 +0000366export def FTfrm()
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000367 if exists("g:filetype_frm")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000368 exe "setf " .. g:filetype_frm
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000369 return
370 endif
371
Doug Kearnsf97f6bb2023-08-27 18:44:09 +0200372 if getline(1) == "VERSION 5.00"
373 setf vb
374 return
375 endif
376
Bram Moolenaara2baa732022-02-04 16:09:54 +0000377 var lines = getline(1, min([line("$"), 5]))
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000378
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000379 if match(lines, ft_visual_basic_content) > -1
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000380 setf vb
381 else
382 setf form
383 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000384enddef
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000385
Doug Kearns19a3bc32023-08-20 20:51:12 +0200386# Distinguish between Forth and F#
Bram Moolenaara2baa732022-02-04 16:09:54 +0000387export def FTfs()
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000388 if exists("g:filetype_fs")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000389 exe "setf " .. g:filetype_fs
Doug Kearns19a3bc32023-08-20 20:51:12 +0200390 elseif IsForth()
391 setf forth
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000392 else
Johan Kotlinski065088d2023-04-02 20:29:38 +0100393 setf fsharp
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000394 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000395enddef
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000396
Amelia Clarke35dfe582024-05-24 08:05:00 +0200397# Recursively search for Hare source files in a directory and any
398# subdirectories, up to a given depth.
399def IsHareModule(dir: string, depth: number): bool
400 if depth <= 0
401 return !empty(glob(dir .. '/*.ha'))
402 endif
403
404 return reduce(sort(glob(dir .. '/*', true, true),
405 (a, b) => isdirectory(a) - isdirectory(b)),
406 (acc, n) => acc
407 || n =~ '\.ha$'
408 || isdirectory(n)
409 && IsHareModule(n, depth - 1),
410 false)
411enddef
412
413# Determine if a README file exists within a Hare module and should be given the
414# Haredoc filetype.
415export def FTharedoc()
416 if exists('g:filetype_haredoc')
417 if IsHareModule('<afile>:h', get(g:, 'haredoc_search_depth', 1))
418 setf haredoc
419 endif
420 endif
421enddef
422
Dennis van den Berg1ad194c2024-07-09 19:25:33 +0200423# Distinguish between HTML, XHTML, Django and Angular
Bram Moolenaara2baa732022-02-04 16:09:54 +0000424export def FThtml()
425 var n = 1
Dennis van den Berg1ad194c2024-07-09 19:25:33 +0200426
427 # Test if the filename follows the Angular component template convention
Christian Brabandtc03f6312024-07-10 19:23:39 +0200428 # Disabled for the reasons mentioned here: #13594
429 # if expand('%:t') =~ '^.*\.component\.html$'
430 # setf htmlangular
431 # return
432 # endif
Dennis van den Berg1ad194c2024-07-09 19:25:33 +0200433
Afiq Nazrie9718ed72024-06-18 19:59:53 +0200434 while n < 40 && n <= line("$")
Dennis van den Berg1ad194c2024-07-09 19:25:33 +0200435 # Check for Angular
436 if getline(n) =~ '@\(if\|for\|defer\|switch\)\|\*\(ngIf\|ngFor\|ngSwitch\|ngTemplateOutlet\)\|ng-template\|ng-content\|{{.*}}'
437 setf htmlangular
438 return
439 endif
Dennis van den Berg1ad194c2024-07-09 19:25:33 +0200440 # Check for XHTML
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100441 if getline(n) =~ '\<DTD\s\+XHTML\s'
442 setf xhtml
443 return
444 endif
Dennis van den Berg1ad194c2024-07-09 19:25:33 +0200445 # Check for Django
Afiq Nazrie9718ed72024-06-18 19:59:53 +0200446 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 +0100447 setf htmldjango
448 return
449 endif
EliSaudere57c9a12024-07-28 21:28:11 +0200450 # Check for SuperHTML
451 if getline(n) =~ '<extend\|<super>'
452 setf superhtml
453 return
454 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000455 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100456 endwhile
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100457 setf FALLBACK html
Bram Moolenaara2baa732022-02-04 16:09:54 +0000458enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100459
Bram Moolenaara2baa732022-02-04 16:09:54 +0000460# Distinguish between standard IDL and MS-IDL
461export def FTidl()
462 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100463 while n < 50 && n <= line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100464 if getline(n) =~ '^\s*import\s\+"\(unknwn\|objidl\)\.idl"'
465 setf msidl
466 return
467 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000468 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100469 endwhile
470 setf idl
Bram Moolenaara2baa732022-02-04 16:09:54 +0000471enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100472
Wu, Zhenyu887a38c2024-05-09 20:35:13 +0200473# Distinguish between "default", Prolog, zsh module's C and Cproto prototype file.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000474export def ProtoCheck(default: string)
Wu, Zhenyu887a38c2024-05-09 20:35:13 +0200475 # zsh modules use '#include "*.pro"'
476 # https://github.com/zsh-users/zsh/blob/63f086d167960a27ecdbcb762179e2c2bf8a29f5/Src/Modules/example.c#L31
477 if getline(1) =~ '/* Generated automatically */'
478 setf c
Bram Moolenaara2baa732022-02-04 16:09:54 +0000479 # Cproto files have a comment in the first line and a function prototype in
480 # the second line, it always ends in ";". Indent files may also have
481 # comments, thus we can't match comments to see the difference.
482 # IDL files can have a single ';' in the second line, require at least one
483 # chacter before the ';'.
Wu, Zhenyu887a38c2024-05-09 20:35:13 +0200484 elseif getline(2) =~ '.;$'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100485 setf cpp
486 else
Bram Moolenaar48c3f4e2022-08-08 15:42:38 +0100487 # recognize Prolog by specific text in the first non-empty line
488 # require a blank after the '%' because Perl uses "%list" and "%translate"
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100489 var lnum = getline(nextnonblank(1))
igna_martinoli37853b72024-07-18 21:34:36 +0200490 if lnum =~ '\<prolog\>' || lnum =~ prolog_pattern
Bram Moolenaar48c3f4e2022-08-08 15:42:38 +0100491 setf prolog
492 else
493 exe 'setf ' .. default
494 endif
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100495 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000496enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100497
Bram Moolenaara2baa732022-02-04 16:09:54 +0000498export def FTm()
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200499 if exists("g:filetype_m")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000500 exe "setf " .. g:filetype_m
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200501 return
502 endif
503
Bram Moolenaara2baa732022-02-04 16:09:54 +0000504 # excluding end(for|function|if|switch|while) common to Murphi
505 var octave_block_terminators = '\<end\%(_try_catch\|classdef\|enumeration\|events\|methods\|parfor\|properties\)\>'
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200506
Bram Moolenaara2baa732022-02-04 16:09:54 +0000507 var objc_preprocessor = '^\s*#\s*\%(import\|include\|define\|if\|ifn\=def\|undef\|line\|error\|pragma\)\>'
Doug Kearns7329cfa2021-11-26 13:01:41 +0000508
Bram Moolenaara2baa732022-02-04 16:09:54 +0000509 var n = 1
510 var saw_comment = 0 # Whether we've seen a multiline comment leader.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100511 while n < 100
Bram Moolenaara2baa732022-02-04 16:09:54 +0000512 var line = getline(n)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100513 if line =~ '^\s*/\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000514 # /* ... */ is a comment in Objective C and Murphi, so we can't conclude
515 # it's either of them yet, but track this as a hint in case we don't see
516 # anything more definitive.
517 saw_comment = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100518 endif
Doug Kearns7329cfa2021-11-26 13:01:41 +0000519 if line =~ '^\s*//' || line =~ '^\s*@import\>' || line =~ objc_preprocessor
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100520 setf objc
521 return
522 endif
Bram Moolenaarca0627d2021-09-12 17:03:08 +0200523 if line =~ '^\s*\%(#\|%!\)' || line =~ '^\s*unwind_protect\>' ||
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200524 \ line =~ '\%(^\|;\)\s*' .. octave_block_terminators
525 setf octave
526 return
527 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000528 # TODO: could be Matlab or Octave
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100529 if line =~ '^\s*%'
530 setf matlab
531 return
532 endif
533 if line =~ '^\s*(\*'
534 setf mma
535 return
536 endif
537 if line =~ '^\c\s*\(\(type\|var\)\>\|--\)'
538 setf murphi
539 return
540 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000541 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100542 endwhile
543
544 if saw_comment
Bram Moolenaara2baa732022-02-04 16:09:54 +0000545 # We didn't see anything definitive, but this looks like either Objective C
546 # or Murphi based on the comment leader. Assume the former as it is more
547 # common.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100548 setf objc
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100549 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000550 # Default is Matlab
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100551 setf matlab
552 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000553enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100554
Ken Takataeb4b9032024-07-25 21:07:13 +0200555export def FTmake()
556 # Check if it is a Microsoft Makefile
557 unlet! b:make_microsoft
558 var n = 1
559 while n < 1000 && n <= line('$')
560 var line = getline(n)
561 if line =~? '^\s*!\s*\(ifn\=\(def\)\=\|include\|message\|error\)\>'
562 b:make_microsoft = 1
563 break
564 elseif line =~ '^ *ifn\=\(eq\|def\)\>' || line =~ '^ *[-s]\=include\s'
565 break
566 elseif line =~ '^ *\w\+\s*[!?:+]='
567 break
568 endif
569 n += 1
570 endwhile
571 setf make
572enddef
573
Bram Moolenaara2baa732022-02-04 16:09:54 +0000574export def FTmms()
575 var n = 1
Bram Moolenaard7df2792020-01-02 21:34:42 +0100576 while n < 20
Bram Moolenaara2baa732022-02-04 16:09:54 +0000577 var line = getline(n)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100578 if line =~ '^\s*\(%\|//\)' || line =~ '^\*'
579 setf mmix
580 return
581 endif
582 if line =~ '^\s*#'
583 setf make
584 return
585 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000586 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100587 endwhile
588 setf mmix
Bram Moolenaara2baa732022-02-04 16:09:54 +0000589enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100590
Bram Moolenaara2baa732022-02-04 16:09:54 +0000591# This function checks if one of the first five lines start with a dot. In
592# that case it is probably an nroff file: 'filetype' is set and 1 is returned.
593export def FTnroff(): number
594 if getline(1)[0] .. getline(2)[0] .. getline(3)[0]
595 .. getline(4)[0] .. getline(5)[0] =~ '\.'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100596 setf nroff
597 return 1
598 endif
599 return 0
Bram Moolenaara2baa732022-02-04 16:09:54 +0000600enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100601
Bram Moolenaara2baa732022-02-04 16:09:54 +0000602export def FTmm()
603 var n = 1
Bram Moolenaard1caa942020-04-10 22:10:56 +0200604 while n < 20
Bram Moolenaara2baa732022-02-04 16:09:54 +0000605 if getline(n) =~ '^\s*\(#\s*\(include\|import\)\>\|@import\>\|/\*\)'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100606 setf objcpp
607 return
608 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000609 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100610 endwhile
611 setf nroff
Bram Moolenaara2baa732022-02-04 16:09:54 +0000612enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100613
Bram Moolenaarcdbfc6d2022-06-30 16:25:21 +0100614# Returns true if file content looks like LambdaProlog module
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100615def IsLProlog(): bool
Gaetan Lepage4ce1bda2023-05-10 22:01:55 +0100616 # skip apparent comments and blank lines, what looks like
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100617 # LambdaProlog comment may be RAPID header
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100618 var lnum: number = nextnonblank(1)
619 while lnum > 0 && lnum < line('$') && getline(lnum) =~ '^\s*%' # LambdaProlog comment
620 lnum = nextnonblank(lnum + 1)
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100621 endwhile
622 # this pattern must not catch a go.mod file
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100623 return getline(lnum) =~ '\<module\s\+\w\+\s*\.\s*\(%\|$\)'
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100624enddef
625
Doug Kearns68a89472024-01-05 17:59:04 +0100626def IsModula2(): bool
dkearnsef387c02024-02-20 06:58:30 +1100627 return getline(nextnonblank(1)) =~ '\<MODULE\s\+\w\+\s*\%(\[.*]\s*\)\=;\|^\s*(\*'
Doug Kearns68a89472024-01-05 17:59:04 +0100628enddef
629
630def SetFiletypeModula2()
631 const KNOWN_DIALECTS = ["iso", "pim", "r10"]
632 const KNOWN_EXTENSIONS = ["gm2"]
633 const LINE_COUNT = 200
634 const TAG = '(\*!m2\(\w\+\)\%(+\(\w\+\)\)\=\*)'
635
636 var dialect = get(g:, "modula2_default_dialect", "pim")
637 var extension = get(g:, "modula2_default_extension", "")
638
639 var matches = []
640
641 # ignore unknown dialects or badly formatted tags
642 for lnum in range(1, min([line("$"), LINE_COUNT]))
643 matches = matchlist(getline(lnum), TAG)
644 if !empty(matches)
645 if index(KNOWN_DIALECTS, matches[1]) >= 0
646 dialect = matches[1]
647 endif
648 if index(KNOWN_EXTENSIONS, matches[2]) >= 0
649 extension = matches[2]
650 endif
651 break
652 endif
653 endfor
654
655 modula2#SetDialect(dialect, extension)
656
657 setf modula2
658enddef
659
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100660# Determine if *.mod is ABB RAPID, LambdaProlog, Modula-2, Modsim III or go.mod
661export def FTmod()
Doug Kearns68a89472024-01-05 17:59:04 +0100662 if get(g:, "filetype_mod", "") == "modula2" || IsModula2()
663 SetFiletypeModula2()
664 return
665 endif
666
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100667 if exists("g:filetype_mod")
668 exe "setf " .. g:filetype_mod
Omar El Halabic9fbd252023-05-29 19:59:45 +0100669 elseif expand("<afile>") =~ '\<go.mod$'
670 setf gomod
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100671 elseif IsLProlog()
672 setf lprolog
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100673 elseif IsRapid()
674 setf rapid
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100675 else
676 # Nothing recognized, assume modsim3
677 setf modsim3
678 endif
679enddef
680
Bram Moolenaara2baa732022-02-04 16:09:54 +0000681export def FTpl()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100682 if exists("g:filetype_pl")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000683 exe "setf " .. g:filetype_pl
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100684 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000685 # recognize Prolog by specific text in the first non-empty line
686 # require a blank after the '%' because Perl uses "%list" and "%translate"
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100687 var line = getline(nextnonblank(1))
igna_martinoli37853b72024-07-18 21:34:36 +0200688 if line =~ '\<prolog\>' || line =~ prolog_pattern
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100689 setf prolog
690 else
691 setf perl
692 endif
693 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000694enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100695
Bram Moolenaara2baa732022-02-04 16:09:54 +0000696export def FTinc()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100697 if exists("g:filetype_inc")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000698 exe "setf " .. g:filetype_inc
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100699 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000700 var lines = getline(1) .. getline(2) .. getline(3)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100701 if lines =~? "perlscript"
702 setf aspperl
703 elseif lines =~ "<%"
704 setf aspvbs
705 elseif lines =~ "<?"
706 setf php
Bram Moolenaara2baa732022-02-04 16:09:54 +0000707 # Pascal supports // comments but they're vary rarely used for file
708 # headers so assume POV-Ray
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000709 elseif lines =~ '^\s*\%({\|(\*\)' || lines =~? ft_pascal_keywords
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100710 setf pascal
Gregory Anders30e212d2022-07-26 21:42:03 +0100711 elseif lines =~# '\<\%(require\|inherit\)\>' || lines =~# '[A-Z][A-Za-z0-9_:${}]*\s\+\%(??\|[?:+]\)\?= '
Gregory Andersfa49eb42022-07-16 17:46:47 +0100712 setf bitbake
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100713 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000714 FTasmsyntax()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100715 if exists("b:asmsyntax")
Gregory Andersfa49eb42022-07-16 17:46:47 +0100716 exe "setf " .. fnameescape(b:asmsyntax)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100717 else
Gregory Andersfa49eb42022-07-16 17:46:47 +0100718 setf pov
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100719 endif
720 endif
721 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000722enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100723
Bram Moolenaara2baa732022-02-04 16:09:54 +0000724export def FTprogress_cweb()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100725 if exists("g:filetype_w")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000726 exe "setf " .. g:filetype_w
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100727 return
728 endif
729 if getline(1) =~ '&ANALYZE' || getline(3) =~ '&GLOBAL-DEFINE'
730 setf progress
731 else
732 setf cweb
733 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000734enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100735
Julien Marrec2e310652023-11-25 15:30:46 +0100736# These include the leading '%' sign
737var 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\)'
738# This is the start/end of a block that is copied literally to the processor file (C/C++)
739var ft_swig_verbatim_block_start = '^\s*%{'
740
741export def FTi()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100742 if exists("g:filetype_i")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000743 exe "setf " .. g:filetype_i
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100744 return
745 endif
Julien Marrec2e310652023-11-25 15:30:46 +0100746 # 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 +0000747 # If not found, assume Progress.
748 var lnum = 1
Julien Marrec2e310652023-11-25 15:30:46 +0100749 while lnum <= 50 && lnum < line('$')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000750 var line = getline(lnum)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100751 if line =~ '^\s*;' || line =~ '^\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000752 FTasm()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100753 return
Julien Marrec2e310652023-11-25 15:30:46 +0100754 elseif line =~ ft_swig_keywords || line =~ ft_swig_verbatim_block_start
755 setf swig
756 return
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100757 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000758 lnum += 1
Bram Moolenaarc12dc472022-03-05 13:45:56 +0000759 endwhile
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100760 setf progress
Bram Moolenaara2baa732022-02-04 16:09:54 +0000761enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100762
Bram Moolenaara2baa732022-02-04 16:09:54 +0000763var ft_pascal_comments = '^\s*\%({\|(\*\|//\)'
764var ft_pascal_keywords = '^\s*\%(program\|unit\|library\|uses\|begin\|procedure\|function\|const\|type\|var\)\>'
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100765
Bram Moolenaara2baa732022-02-04 16:09:54 +0000766export def FTprogress_pascal()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100767 if exists("g:filetype_p")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000768 exe "setf " .. g:filetype_p
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100769 return
770 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000771 # This function checks for valid Pascal syntax in the first ten lines.
772 # Look for either an opening comment or a program start.
773 # If not found, assume Progress.
774 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100775 while lnum <= 10 && lnum < line('$')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000776 var line = getline(lnum)
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000777 if line =~ ft_pascal_comments || line =~? ft_pascal_keywords
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100778 setf pascal
779 return
780 elseif line !~ '^\s*$' || line =~ '^/\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000781 # Not an empty line: Doesn't look like valid Pascal code.
782 # Or it looks like a Progress /* comment
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100783 break
784 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000785 lnum += 1
Bram Moolenaarc12dc472022-03-05 13:45:56 +0000786 endwhile
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100787 setf progress
Bram Moolenaara2baa732022-02-04 16:09:54 +0000788enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100789
Bram Moolenaara2baa732022-02-04 16:09:54 +0000790export def FTpp()
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100791 if exists("g:filetype_pp")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000792 exe "setf " .. g:filetype_pp
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100793 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000794 var line = getline(nextnonblank(1))
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000795 if line =~ ft_pascal_comments || line =~? ft_pascal_keywords
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100796 setf pascal
797 else
798 setf puppet
799 endif
800 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000801enddef
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100802
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100803# Determine if *.prg is ABB RAPID. Can also be Clipper, FoxPro or eviews
804export def FTprg()
805 if exists("g:filetype_prg")
806 exe "setf " .. g:filetype_prg
807 elseif IsRapid()
808 setf rapid
809 else
810 # Nothing recognized, assume Clipper
811 setf clipper
812 endif
813enddef
814
Bram Moolenaara2baa732022-02-04 16:09:54 +0000815export def FTr()
816 var max = line("$") > 50 ? 50 : line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100817
818 for n in range(1, max)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000819 # Rebol is easy to recognize, check for that first
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100820 if getline(n) =~? '\<REBOL\>'
821 setf rebol
822 return
823 endif
824 endfor
825
826 for n in range(1, max)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000827 # R has # comments
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100828 if getline(n) =~ '^\s*#'
829 setf r
830 return
831 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000832 # Rexx has /* comments */
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100833 if getline(n) =~ '^\s*/\*'
834 setf rexx
835 return
836 endif
837 endfor
838
Bram Moolenaara2baa732022-02-04 16:09:54 +0000839 # Nothing recognized, use user default or assume Rexx
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100840 if exists("g:filetype_r")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000841 exe "setf " .. g:filetype_r
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100842 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000843 # Rexx used to be the default, but R appears to be much more popular.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100844 setf r
845 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000846enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100847
Bram Moolenaara2baa732022-02-04 16:09:54 +0000848export def McSetf()
849 # Rely on the file to start with a comment.
850 # MS message text files use ';', Sendmail files use '#' or 'dnl'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100851 for lnum in range(1, min([line("$"), 20]))
Bram Moolenaara2baa732022-02-04 16:09:54 +0000852 var line = getline(lnum)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100853 if line =~ '^\s*\(#\|dnl\)'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000854 setf m4 # Sendmail .mc file
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100855 return
856 elseif line =~ '^\s*;'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000857 setf msmessages # MS Message text file
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100858 return
859 endif
860 endfor
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100861 setf m4 # Default: Sendmail .mc file
Bram Moolenaara2baa732022-02-04 16:09:54 +0000862enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100863
Bram Moolenaara2baa732022-02-04 16:09:54 +0000864# Called from filetype.vim and scripts.vim.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100865# When "setft" is passed and false then the 'filetype' option is not set.
866export def SetFileTypeSH(name: string, setft = true): string
867 if setft && did_filetype()
Bram Moolenaara2baa732022-02-04 16:09:54 +0000868 # Filetype was already detected
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100869 return ''
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100870 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100871 if setft && expand("<amatch>") =~ g:ft_ignore_pat
872 return ''
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100873 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000874 if name =~ '\<csh\>'
875 # Some .sh scripts contain #!/bin/csh.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100876 return SetFileTypeShell("csh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000877 elseif name =~ '\<tcsh\>'
878 # Some .sh scripts contain #!/bin/tcsh.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100879 return SetFileTypeShell("tcsh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000880 elseif name =~ '\<zsh\>'
881 # Some .sh scripts contain #!/bin/zsh.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100882 return SetFileTypeShell("zsh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000883 elseif name =~ '\<ksh\>'
884 b:is_kornshell = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100885 if exists("b:is_bash")
886 unlet b:is_bash
887 endif
888 if exists("b:is_sh")
889 unlet b:is_sh
890 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000891 elseif exists("g:bash_is_sh") || name =~ '\<bash\>' || name =~ '\<bash2\>'
892 b:is_bash = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100893 if exists("b:is_kornshell")
894 unlet b:is_kornshell
895 endif
896 if exists("b:is_sh")
897 unlet b:is_sh
898 endif
Luca Saccarolab9b762c2024-12-27 16:08:14 +0100899 return SetFileTypeShell("bash", setft)
Eisuke Kawashima24482fb2022-11-24 10:58:10 +0000900 elseif name =~ '\<sh\>' || name =~ '\<dash\>'
901 # Ubuntu links "sh" to "dash", thus it is expected to work the same way
Bram Moolenaara2baa732022-02-04 16:09:54 +0000902 b:is_sh = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100903 if exists("b:is_kornshell")
904 unlet b:is_kornshell
905 endif
906 if exists("b:is_bash")
907 unlet b:is_bash
908 endif
909 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100910
911 return SetFileTypeShell("sh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000912enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100913
Bram Moolenaara2baa732022-02-04 16:09:54 +0000914# For shell-like file types, check for an "exec" command hidden in a comment,
915# as used for Tcl.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100916# When "setft" is passed and false then the 'filetype' option is not set.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000917# Also called from scripts.vim, thus can't be local to this script.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100918export def SetFileTypeShell(name: string, setft = true): string
919 if setft && did_filetype()
Bram Moolenaara2baa732022-02-04 16:09:54 +0000920 # Filetype was already detected
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100921 return ''
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100922 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100923 if setft && expand("<amatch>") =~ g:ft_ignore_pat
924 return ''
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100925 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100926
927 var lnum = 2
928 while lnum < 20 && lnum < line("$") && getline(lnum) =~ '^\s*\(#\|$\)'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000929 # Skip empty and comment lines.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100930 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100931 endwhile
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100932 if lnum < line("$") && getline(lnum) =~ '\s*exec\s' && getline(lnum - 1) =~ '^\s*#.*\\$'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000933 # Found an "exec" line after a comment with continuation
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100934 var n = substitute(getline(lnum), '\s*exec\s\+\([^ ]*/\)\=', '', '')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100935 if n =~ '\<tclsh\|\<wish'
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100936 if setft
937 setf tcl
938 endif
939 return 'tcl'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100940 endif
941 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100942
943 if setft
944 exe "setf " .. name
945 endif
946 return name
Bram Moolenaara2baa732022-02-04 16:09:54 +0000947enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100948
Bram Moolenaara2baa732022-02-04 16:09:54 +0000949export def CSH()
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100950 if did_filetype()
Bram Moolenaara2baa732022-02-04 16:09:54 +0000951 # Filetype was already detected
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100952 return
953 endif
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100954 if exists("g:filetype_csh")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000955 SetFileTypeShell(g:filetype_csh)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100956 elseif &shell =~ "tcsh"
Bram Moolenaara2baa732022-02-04 16:09:54 +0000957 SetFileTypeShell("tcsh")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100958 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000959 SetFileTypeShell("csh")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100960 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000961enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100962
Bram Moolenaara2baa732022-02-04 16:09:54 +0000963var ft_rules_udev_rules_pattern = '^\s*\cudev_rules\s*=\s*"\([^"]\{-1,}\)/*".*'
964export def FTRules()
965 var path = expand('<amatch>:p')
Bram Moolenaaraa9675a2020-08-17 21:57:09 +0200966 if path =~ '/\(etc/udev/\%(rules\.d/\)\=.*\.rules\|\%(usr/\)\=lib/udev/\%(rules\.d/\)\=.*\.rules\)$'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100967 setf udevrules
968 return
969 endif
970 if path =~ '^/etc/ufw/'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000971 setf conf # Better than hog
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100972 return
973 endif
974 if path =~ '^/\(etc\|usr/share\)/polkit-1/rules\.d'
975 setf javascript
976 return
977 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000978 var config_lines: list<string>
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100979 try
Bram Moolenaara2baa732022-02-04 16:09:54 +0000980 config_lines = readfile('/etc/udev/udev.conf')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100981 catch /^Vim\%((\a\+)\)\=:E484/
982 setf hog
983 return
984 endtry
Bram Moolenaara2baa732022-02-04 16:09:54 +0000985 var dir = expand('<amatch>:p:h')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100986 for line in config_lines
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000987 if line =~ ft_rules_udev_rules_pattern
988 var udev_rules = substitute(line, ft_rules_udev_rules_pattern, '\1', "")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100989 if dir == udev_rules
990 setf udevrules
991 endif
992 break
993 endif
994 endfor
995 setf hog
Bram Moolenaara2baa732022-02-04 16:09:54 +0000996enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100997
Bram Moolenaara2baa732022-02-04 16:09:54 +0000998export def SQL()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100999 if exists("g:filetype_sql")
Bram Moolenaara2baa732022-02-04 16:09:54 +00001000 exe "setf " .. g:filetype_sql
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001001 else
1002 setf sql
1003 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001004enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001005
ranjithshegde8cac20e2022-04-13 15:29:21 +01001006# This function checks the first 25 lines of file extension "sc" to resolve
Chris Kipp70ef3f52022-12-14 16:42:15 +00001007# detection between scala and SuperCollider.
1008# NOTE: We don't check for 'Class : Method', as this can easily be confused
1009# with valid Scala like `val x : Int = 3`. So we instead only rely on
1010# checks that can't be confused.
ranjithshegde8cac20e2022-04-13 15:29:21 +01001011export def FTsc()
1012 for lnum in range(1, min([line("$"), 25]))
Chris Kipp70ef3f52022-12-14 16:42:15 +00001013 if getline(lnum) =~# 'var\s<\|classvar\s<\|\^this.*\||\w\+|\|+\s\w*\s{\|\*ar\s'
ranjithshegde8cac20e2022-04-13 15:29:21 +01001014 setf supercollider
1015 return
1016 endif
1017 endfor
1018 setf scala
1019enddef
1020
1021# This function checks the first line of file extension "scd" to resolve
1022# detection between scdoc and SuperCollider
1023export def FTscd()
1024 if getline(1) =~# '\%^\S\+(\d[0-9A-Za-z]*)\%(\s\+\"[^"]*\"\%(\s\+\"[^"]*\"\)\=\)\=$'
1025 setf scdoc
1026 else
1027 setf supercollider
1028 endif
1029enddef
1030
Bram Moolenaara2baa732022-02-04 16:09:54 +00001031# If the file has an extension of 't' and is in a directory 't' or 'xt' then
1032# it is almost certainly a Perl test file.
1033# If the first line starts with '#' and contains 'perl' it's probably a Perl
1034# file.
1035# (Slow test) If a file contains a 'use' statement then it is almost certainly
1036# a Perl file.
1037export def FTperl(): number
1038 var dirname = expand("%:p:h:t")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001039 if expand("%:e") == 't' && (dirname == 't' || dirname == 'xt')
1040 setf perl
1041 return 1
1042 endif
1043 if getline(1)[0] == '#' && getline(1) =~ 'perl'
1044 setf perl
1045 return 1
1046 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001047 var save_cursor = getpos('.')
1048 call cursor(1, 1)
Bram Moolenaare5b78972022-02-05 19:50:34 +00001049 var has_use = search('^use\s\s*\k', 'c', 30) > 0
Bram Moolenaarf0b03c42017-12-17 17:17:07 +01001050 call setpos('.', save_cursor)
1051 if has_use
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001052 setf perl
1053 return 1
1054 endif
1055 return 0
Bram Moolenaara2baa732022-02-04 16:09:54 +00001056enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001057
Bram Moolenaarcdbfc6d2022-06-30 16:25:21 +01001058# LambdaProlog and Standard ML signature files
1059export def FTsig()
1060 if exists("g:filetype_sig")
1061 exe "setf " .. g:filetype_sig
1062 return
1063 endif
1064
1065 var lprolog_comment = '^\s*\%(/\*\|%\)'
1066 var lprolog_keyword = '^\s*sig\s\+\a'
1067 var sml_comment = '^\s*(\*'
1068 var sml_keyword = '^\s*\%(signature\|structure\)\s\+\a'
1069
1070 var line = getline(nextnonblank(1))
1071
1072 if line =~ lprolog_comment || line =~# lprolog_keyword
1073 setf lprolog
1074 elseif line =~ sml_comment || line =~# sml_keyword
1075 setf sml
1076 endif
1077enddef
1078
Bram Moolenaarbe807d52022-09-01 15:01:25 +01001079# This function checks the first 100 lines of files matching "*.sil" to
1080# resolve detection between Swift Intermediate Language and SILE.
1081export def FTsil()
1082 for lnum in range(1, [line('$'), 100]->min())
1083 var line: string = getline(lnum)
1084 if line =~ '^\s*[\\%]'
1085 setf sile
1086 return
1087 elseif line =~ '^\s*\S'
1088 setf sil
1089 return
1090 endif
1091 endfor
1092 # no clue, default to "sil"
1093 setf sil
1094enddef
1095
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +01001096export def FTsys()
KnoP-01f420ff22022-04-13 20:46:21 +01001097 if exists("g:filetype_sys")
1098 exe "setf " .. g:filetype_sys
1099 elseif IsRapid()
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +01001100 setf rapid
1101 else
1102 setf bat
1103 endif
1104enddef
1105
Bram Moolenaara2baa732022-02-04 16:09:54 +00001106# Choose context, plaintex, or tex (LaTeX) based on these rules:
1107# 1. Check the first line of the file for "%&<format>".
1108# 2. Check the first 1000 non-comment lines for LaTeX or ConTeXt keywords.
1109# 3. Default to "plain" or to g:tex_flavor, can be set in user's vimrc.
1110export def FTtex()
1111 var firstline = getline(1)
1112 var format: string
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001113 if firstline =~ '^%&\s*\a\+'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001114 format = tolower(matchstr(firstline, '\a\+'))
1115 format = substitute(format, 'pdf', '', '')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001116 if format == 'tex'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001117 format = 'latex'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001118 elseif format == 'plaintex'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001119 format = 'plain'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001120 endif
1121 elseif expand('%') =~ 'tex/context/.*/.*.tex'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001122 format = 'context'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001123 else
Bram Moolenaara2baa732022-02-04 16:09:54 +00001124 # Default value, may be changed later:
1125 format = exists("g:tex_flavor") ? g:tex_flavor : 'plain'
1126 # Save position, go to the top of the file, find first non-comment line.
1127 var save_cursor = getpos('.')
1128 call cursor(1, 1)
1129 var firstNC = search('^\s*[^[:space:]%]', 'c', 1000)
Bram Moolenaare5b78972022-02-05 19:50:34 +00001130 if firstNC > 0
1131 # Check the next thousand lines for a LaTeX or ConTeXt keyword.
Bram Moolenaara2baa732022-02-04 16:09:54 +00001132 var lpat = 'documentclass\>\|usepackage\>\|begin{\|newcommand\>\|renewcommand\>'
1133 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\>'
1134 var kwline = search('^\s*\\\%(' .. lpat .. '\)\|^\s*\\\(' .. cpat .. '\)',
1135 'cnp', firstNC + 1000)
1136 if kwline == 1 # lpat matched
1137 format = 'latex'
1138 elseif kwline == 2 # cpat matched
1139 format = 'context'
1140 endif # If neither matched, keep default set above.
1141 # let lline = search('^\s*\\\%(' . lpat . '\)', 'cn', firstNC + 1000)
1142 # let cline = search('^\s*\\\%(' . cpat . '\)', 'cn', firstNC + 1000)
1143 # if cline > 0
1144 # let format = 'context'
1145 # endif
1146 # if lline > 0 && (cline == 0 || cline > lline)
1147 # let format = 'tex'
1148 # endif
1149 endif # firstNC
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001150 call setpos('.', save_cursor)
Bram Moolenaara2baa732022-02-04 16:09:54 +00001151 endif # firstline =~ '^%&\s*\a\+'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001152
Bram Moolenaara2baa732022-02-04 16:09:54 +00001153 # Translation from formats to file types. TODO: add AMSTeX, RevTex, others?
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001154 if format == 'plain'
1155 setf plaintex
1156 elseif format == 'context'
1157 setf context
Bram Moolenaara2baa732022-02-04 16:09:54 +00001158 else # probably LaTeX
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001159 setf tex
1160 endif
1161 return
Bram Moolenaara2baa732022-02-04 16:09:54 +00001162enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001163
Bram Moolenaara2baa732022-02-04 16:09:54 +00001164export def FTxml()
1165 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +01001166 while n < 100 && n <= line("$")
Bram Moolenaara2baa732022-02-04 16:09:54 +00001167 var line = getline(n)
1168 # DocBook 4 or DocBook 5.
1169 var is_docbook4 = line =~ '<!DOCTYPE.*DocBook'
1170 var is_docbook5 = line =~ ' xmlns="http://docbook.org/ns/docbook"'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001171 if is_docbook4 || is_docbook5
Bram Moolenaara2baa732022-02-04 16:09:54 +00001172 b:docbk_type = "xml"
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001173 if is_docbook5
Bram Moolenaara2baa732022-02-04 16:09:54 +00001174 b:docbk_ver = 5
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001175 else
Bram Moolenaara2baa732022-02-04 16:09:54 +00001176 b:docbk_ver = 4
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001177 endif
1178 setf docbk
1179 return
1180 endif
1181 if line =~ 'xmlns:xbl="http://www.mozilla.org/xbl"'
1182 setf xbl
1183 return
1184 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001185 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001186 endwhile
1187 setf xml
Bram Moolenaara2baa732022-02-04 16:09:54 +00001188enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001189
Bram Moolenaara2baa732022-02-04 16:09:54 +00001190export def FTy()
1191 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +01001192 while n < 100 && n <= line("$")
Bram Moolenaara2baa732022-02-04 16:09:54 +00001193 var line = getline(n)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001194 if line =~ '^\s*%'
1195 setf yacc
1196 return
1197 endif
1198 if getline(n) =~ '^\s*\(#\|class\>\)' && getline(n) !~ '^\s*#\s*include'
1199 setf racc
1200 return
1201 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001202 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001203 endwhile
1204 setf yacc
Bram Moolenaara2baa732022-02-04 16:09:54 +00001205enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001206
Bram Moolenaara2baa732022-02-04 16:09:54 +00001207export def Redif()
1208 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001209 while lnum <= 5 && lnum < line('$')
1210 if getline(lnum) =~ "^\ctemplate-type:"
1211 setf redif
1212 return
1213 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001214 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001215 endwhile
Bram Moolenaara2baa732022-02-04 16:09:54 +00001216enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001217
Bram Moolenaara2baa732022-02-04 16:09:54 +00001218# This function is called for all files under */debian/patches/*, make sure not
1219# to non-dep3patch files, such as README and other text files.
1220export def Dep3patch()
James McCoy647ab4c2021-12-17 20:52:57 +00001221 if expand('%:t') ==# 'series'
1222 return
1223 endif
1224
1225 for ln in getline(1, 100)
1226 if ln =~# '^\%(Description\|Subject\|Origin\|Bug\|Forwarded\|Author\|From\|Reviewed-by\|Acked-by\|Last-Updated\|Applied-Upstream\):'
1227 setf dep3patch
1228 return
1229 elseif ln =~# '^---'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001230 # end of headers found. stop processing
James McCoy647ab4c2021-12-17 20:52:57 +00001231 return
1232 endif
1233 endfor
Bram Moolenaara2baa732022-02-04 16:09:54 +00001234enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001235
Bram Moolenaara2baa732022-02-04 16:09:54 +00001236# This function checks the first 15 lines for appearance of 'FoamFile'
1237# and then 'object' in a following line.
1238# In that case, it's probably an OpenFOAM file
1239export def FTfoam()
1240 var ffile = 0
1241 var lnum = 1
Elwardi2284f6c2022-01-11 18:14:23 +00001242 while lnum <= 15
1243 if getline(lnum) =~# '^FoamFile'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001244 ffile = 1
Elwardi2284f6c2022-01-11 18:14:23 +00001245 elseif ffile == 1 && getline(lnum) =~# '^\s*object'
1246 setf foam
1247 return
1248 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001249 lnum += 1
Elwardi2284f6c2022-01-11 18:14:23 +00001250 endwhile
Bram Moolenaara2baa732022-02-04 16:09:54 +00001251enddef
Elwardi2284f6c2022-01-11 18:14:23 +00001252
Bram Moolenaara2baa732022-02-04 16:09:54 +00001253# Determine if a *.tf file is TF mud client or terraform
1254export def FTtf()
1255 var numberOfLines = line('$')
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001256 for i in range(1, numberOfLines)
Bram Moolenaara2baa732022-02-04 16:09:54 +00001257 var currentLine = trim(getline(i))
1258 var firstCharacter = currentLine[0]
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001259 if firstCharacter !=? ";" && firstCharacter !=? "/" && firstCharacter !=? ""
1260 setf terraform
1261 return
1262 endif
1263 endfor
1264 setf tf
Bram Moolenaara2baa732022-02-04 16:09:54 +00001265enddef
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001266
KnoP-0193c7a452022-04-16 21:14:04 +01001267var ft_krl_header = '\&\w+'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001268# Determine if a *.src file is Kuka Robot Language
1269export def FTsrc()
KnoP-0193c7a452022-04-16 21:14:04 +01001270 var ft_krl_def_or_deffct = '%(global\s+)?def%(fct)?>'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001271 if exists("g:filetype_src")
1272 exe "setf " .. g:filetype_src
KnoP-0193c7a452022-04-16 21:14:04 +01001273 elseif getline(nextnonblank(1)) =~? '\v^\s*%(' .. ft_krl_header .. '|' .. ft_krl_def_or_deffct .. ')'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001274 setf krl
1275 endif
1276enddef
1277
1278# Determine if a *.dat file is Kuka Robot Language
1279export def FTdat()
KnoP-0193c7a452022-04-16 21:14:04 +01001280 var ft_krl_defdat = 'defdat>'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001281 if exists("g:filetype_dat")
1282 exe "setf " .. g:filetype_dat
KnoP-0193c7a452022-04-16 21:14:04 +01001283 elseif getline(nextnonblank(1)) =~? '\v^\s*%(' .. ft_krl_header .. '|' .. ft_krl_defdat .. ')'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001284 setf krl
1285 endif
1286enddef
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001287
Doug Kearns4ac8e792022-10-17 13:32:17 +01001288export def FTlsl()
1289 if exists("g:filetype_lsl")
1290 exe "setf " .. g:filetype_lsl
1291 endif
1292
1293 var line = getline(nextnonblank(1))
1294 if line =~ '^\s*%' || line =~# ':\s*trait\s*$'
1295 setf larch
1296 else
1297 setf lsl
1298 endif
1299enddef
1300
Gaetan Lepage4ce1bda2023-05-10 22:01:55 +01001301export def FTtyp()
1302 if exists("g:filetype_typ")
1303 exe "setf " .. g:filetype_typ
1304 return
1305 endif
1306
1307 # Look for SQL type definition syntax
1308 for line in getline(1, 200)
1309 # SQL type files may define the casing
1310 if line =~ '^CASE\s\==\s\=\(SAME\|LOWER\|UPPER\|OPPOSITE\)$'
1311 setf sql
1312 return
1313 endif
1314
1315 # SQL type files may define some types as follows
1316 if line =~ '^TYPE\s.*$'
1317 setf sql
1318 return
1319 endif
1320 endfor
1321
1322 # Otherwise, affect the typst filetype
1323 setf typst
1324enddef
1325
PowerUser64aa61b8a2024-06-19 20:32:11 +02001326# Detect Microsoft Developer Studio Project files (Makefile) or Faust DSP
1327# files.
1328export def FTdsp()
1329 if exists("g:filetype_dsp")
1330 exe "setf " .. g:filetype_dsp
1331 return
1332 endif
1333
1334 # Test the filename
1335 if expand('%:t') =~ '^[mM]akefile.*$'
1336 setf make
1337 return
1338 endif
1339
1340 # Test the file contents
1341 for line in getline(1, 200)
1342 # Chech for comment style
1343 if line =~ '^#.*'
1344 setf make
1345 return
1346 endif
1347
1348 # Check for common lines
1349 if line =~ '^.*Microsoft Developer Studio Project File.*$'
1350 setf make
1351 return
1352 endif
1353
1354 if line =~ '^!MESSAGE This is not a valid makefile\..+$'
1355 setf make
1356 return
1357 endif
1358
1359 # Check for keywords
1360 if line =~ '^!(IF,ELSEIF,ENDIF).*$'
1361 setf make
1362 return
1363 endif
1364
1365 # Check for common assignments
1366 if line =~ '^SOURCE=.*$'
1367 setf make
1368 return
1369 endif
1370 endfor
1371
1372 # Otherwise, assume we have a Faust file
1373 setf faust
1374enddef
1375
Turiiya80406c22023-04-22 21:38:47 +01001376# Set the filetype of a *.v file to Verilog, V or Cog based on the first 200
1377# lines.
1378export def FTv()
1379 if did_filetype()
1380 # ":setf" will do nothing, bail out early
1381 return
1382 endif
Christian Brabandt10b4f752024-01-01 19:19:20 +01001383 if exists("g:filetype_v")
1384 exe "setf " .. g:filetype_v
1385 return
1386 endif
Turiiya80406c22023-04-22 21:38:47 +01001387
Christian Brabandt10b4f752024-01-01 19:19:20 +01001388 var in_comment = 0
1389 for lnum in range(1, min([line("$"), 200]))
1390 var line = getline(lnum)
1391 # Skip Verilog and V comments (lines and blocks).
1392 if line =~ '^\s*/\*'
1393 # start comment block
1394 in_comment = 1
1395 endif
1396 if in_comment == 1
1397 if line =~ '\*/'
1398 # end comment block
1399 in_comment = 0
1400 endif
1401 # skip comment-block line
1402 continue
1403 endif
1404 if line =~ '^\s*//'
Turiiya80406c22023-04-22 21:38:47 +01001405 # skip comment line
1406 continue
1407 endif
1408
Christian Brabandt10b4f752024-01-01 19:19:20 +01001409 # Coq: line ends with a '.' followed by an optional variable number of
1410 # spaces or contains the start of a comment, but not inside a Verilog or V
1411 # comment.
1412 # Example: "Definition x := 10. (*".
1413 if (line =~ '\.\s*$' && line !~ '/[/*]') || (line =~ '(\*' && line !~ '/[/*].*(\*')
1414 setf coq
1415 return
1416 endif
1417
Turiiya80406c22023-04-22 21:38:47 +01001418 # Verilog: line ends with ';' followed by an optional variable number of
1419 # spaces and an optional start of a comment.
1420 # Example: " b <= a + 1; // Add 1".
Christian Brabandt10b4f752024-01-01 19:19:20 +01001421 if line =~ ';\s*\(/[/*].*\)\?$'
Turiiya80406c22023-04-22 21:38:47 +01001422 setf verilog
1423 return
1424 endif
Turiiya80406c22023-04-22 21:38:47 +01001425 endfor
1426
1427 # No line matched, fall back to "v".
1428 setf v
1429enddef
1430
Doug Kearnsf97f6bb2023-08-27 18:44:09 +02001431export def FTvba()
1432 if getline(1) =~ '^["#] Vimball Archiver'
1433 setf vim
1434 else
1435 setf vb
1436 endif
1437enddef
1438
Colin Caine4b3fab12024-04-18 23:53:02 +02001439export def Detect_UCI_statements(): bool
1440 # Match a config or package statement at the start of the line.
1441 const config_or_package_statement = '^\s*\(\(c\|config\)\|\(p\|package\)\)\s\+\S'
1442 # Match a line that is either all blank or blank followed by a comment
1443 const comment_or_blank = '^\s*\(#.*\)\?$'
1444
1445 # Return true iff the file has a config or package statement near the
1446 # top of the file and all preceding lines were comments or blank.
1447 return getline(1) =~# config_or_package_statement
1448 \ || getline(1) =~# comment_or_blank
1449 \ && ( getline(2) =~# config_or_package_statement
1450 \ || getline(2) =~# comment_or_blank
1451 \ && getline(3) =~# config_or_package_statement
1452 \ )
1453enddef
1454
Bram Moolenaara2baa732022-02-04 16:09:54 +00001455# Uncomment this line to check for compilation errors early
dkearnsef387c02024-02-20 06:58:30 +11001456# defcompile