blob: 3c7d85cec652f05ec60516e2f9c67ada2d5d2a52 [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
Bram Moolenaara2baa732022-02-04 16:09:54 +000012export def Check_inp()
Wu, Zhenyu61ee8332024-04-09 22:09:30 +020013 if getline(1) =~ '%%'
14 setf tex
15 elseif getline(1) =~ '^\*'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010016 setf abaqus
17 else
Bram Moolenaara2baa732022-02-04 16:09:54 +000018 var n = 1
19 var nmax = line("$") > 500 ? 500 : line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010020 while n <= nmax
21 if getline(n) =~? "^header surface data"
22 setf trasys
23 break
24 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +000025 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010026 endwhile
27 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +000028enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010029
Bram Moolenaara2baa732022-02-04 16:09:54 +000030# This function checks for the kind of assembly that is wanted by the user, or
31# can be detected from the first five lines of the file.
32export def FTasm()
33 # make sure b:asmsyntax exists
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010034 if !exists("b:asmsyntax")
Bram Moolenaara2baa732022-02-04 16:09:54 +000035 b:asmsyntax = ""
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010036 endif
37
38 if b:asmsyntax == ""
Bram Moolenaara2baa732022-02-04 16:09:54 +000039 FTasmsyntax()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010040 endif
41
Bram Moolenaara2baa732022-02-04 16:09:54 +000042 # if b:asmsyntax still isn't set, default to asmsyntax or GNU
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010043 if b:asmsyntax == ""
44 if exists("g:asmsyntax")
Bram Moolenaara2baa732022-02-04 16:09:54 +000045 b:asmsyntax = g:asmsyntax
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010046 else
Bram Moolenaara2baa732022-02-04 16:09:54 +000047 b:asmsyntax = "asm"
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010048 endif
49 endif
50
Bram Moolenaara2baa732022-02-04 16:09:54 +000051 exe "setf " .. fnameescape(b:asmsyntax)
52enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010053
Bram Moolenaara2baa732022-02-04 16:09:54 +000054export def FTasmsyntax()
55 # see if the file contains any asmsyntax=foo overrides. If so, change
56 # b:asmsyntax appropriately
57 var head = " " .. getline(1) .. " " .. getline(2) .. " "
58 .. getline(3) .. " " .. getline(4) .. " " .. getline(5) .. " "
59 var match = matchstr(head, '\sasmsyntax=\zs[a-zA-Z0-9]\+\ze\s')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010060 if match != ''
Bram Moolenaara2baa732022-02-04 16:09:54 +000061 b:asmsyntax = match
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010062 elseif ((head =~? '\.title') || (head =~? '\.ident') || (head =~? '\.macro') || (head =~? '\.subtitle') || (head =~? '\.library'))
Bram Moolenaara2baa732022-02-04 16:09:54 +000063 b:asmsyntax = "vmasm"
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010064 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +000065enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010066
Doug Kearnsf97f6bb2023-08-27 18:44:09 +020067var ft_visual_basic_content = '\c^\s*\%(Attribute\s\+VB_Name\|Begin\s\+\%(VB\.\|{\%(\x\+-\)\+\x\+}\)\)'
Doug Kearnsc570e9c2022-01-31 17:09:14 +000068
Bram Moolenaara2baa732022-02-04 16:09:54 +000069# See FTfrm() for Visual Basic form file detection
70export def FTbas()
Bram Moolenaar6517f142022-01-21 14:55:13 +000071 if exists("g:filetype_bas")
Bram Moolenaara2baa732022-02-04 16:09:54 +000072 exe "setf " .. g:filetype_bas
Bram Moolenaar6517f142022-01-21 14:55:13 +000073 return
74 endif
75
Bram Moolenaara2baa732022-02-04 16:09:54 +000076 # most frequent FreeBASIC-specific keywords in distro files
77 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 +010078 var fb_preproc = '\c^\s*\%(' ..
79 # preprocessor
80 '#\s*\a\+\|' ..
81 # compiler option
82 'option\s\+\%(byval\|dynamic\|escape\|\%(no\)\=gosub\|nokeyword\|private\|static\)\>\|' ..
83 # metacommand
84 '\%(''\|rem\)\s*\$lang\>\|' ..
85 # default datatype
86 'def\%(byte\|longint\|short\|ubyte\|uint\|ulongint\|ushort\)\>' ..
87 '\)'
Bram Moolenaara2baa732022-02-04 16:09:54 +000088 var fb_comment = "^\\s*/'"
Bram Moolenaar8b5901e2022-06-29 14:39:12 +010089
Bram Moolenaara2baa732022-02-04 16:09:54 +000090 # OPTION EXPLICIT, without the leading underscore, is common to many dialects
91 var qb64_preproc = '\c^\s*\%($\a\+\|option\s\+\%(_explicit\|_\=explicitarray\)\>\)'
Bram Moolenaar6517f142022-01-21 14:55:13 +000092
Bram Moolenaar8b5901e2022-06-29 14:39:12 +010093 for lnum in range(1, min([line("$"), 100]))
94 var line = getline(lnum)
95 if line =~ ft_visual_basic_content
96 setf vb
97 return
98 elseif line =~ fb_preproc || line =~ fb_comment || line =~ fb_keywords
99 setf freebasic
100 return
101 elseif line =~ qb64_preproc
102 setf qb64
103 return
104 endif
105 endfor
106 setf basic
Bram Moolenaara2baa732022-02-04 16:09:54 +0000107enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100108
Bram Moolenaara2baa732022-02-04 16:09:54 +0000109export def FTbtm()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100110 if exists("g:dosbatch_syntax_for_btm") && g:dosbatch_syntax_for_btm
111 setf dosbatch
112 else
113 setf btm
114 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000115enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100116
Bram Moolenaara2baa732022-02-04 16:09:54 +0000117export def BindzoneCheck(default = '')
118 if getline(1) .. getline(2) .. getline(3) .. getline(4)
119 =~ '^; <<>> DiG [0-9.]\+.* <<>>\|$ORIGIN\|$TTL\|IN\s\+SOA'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100120 setf bindzone
Bram Moolenaara2baa732022-02-04 16:09:54 +0000121 elseif default != ''
122 exe 'setf ' .. default
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100123 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000124enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100125
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100126# Returns true if file content looks like RAPID
127def IsRapid(sChkExt: string = ""): bool
128 if sChkExt == "cfg"
129 return getline(1) =~? '\v^%(EIO|MMC|MOC|PROC|SIO|SYS):CFG'
130 endif
131 # called from FTmod, FTprg or FTsys
132 return getline(nextnonblank(1)) =~? '\v^\s*%(\%{3}|module\s+\k+\s*%(\(|$))'
133enddef
134
135export def FTcfg()
136 if exists("g:filetype_cfg")
137 exe "setf " .. g:filetype_cfg
138 elseif IsRapid("cfg")
139 setf rapid
140 else
141 setf cfg
142 endif
143enddef
144
Bram Moolenaar8b5901e2022-06-29 14:39:12 +0100145export def FTcls()
146 if exists("g:filetype_cls")
147 exe "setf " .. g:filetype_cls
148 return
149 endif
150
Doug Kearnsf97f6bb2023-08-27 18:44:09 +0200151 var line1 = getline(1)
Doug Kearnse06afb72023-08-29 22:21:35 +0200152 if line1 =~ '^#!.*\<\%(rexx\|regina\)\>'
Bram Moolenaar8b5901e2022-06-29 14:39:12 +0100153 setf rexx
Doug Kearnse06afb72023-08-29 22:21:35 +0200154 return
Doug Kearnsf97f6bb2023-08-27 18:44:09 +0200155 elseif line1 == 'VERSION 1.0 CLASS'
Bram Moolenaar8b5901e2022-06-29 14:39:12 +0100156 setf vb
Doug Kearnse06afb72023-08-29 22:21:35 +0200157 return
158 endif
159
160 var nonblank1 = getline(nextnonblank(1))
161 if nonblank1 =~ '^\v%(\%|\\)'
162 setf tex
163 elseif nonblank1 =~ '^\s*\%(/\*\|::\w\)'
164 setf rexx
Bram Moolenaar8b5901e2022-06-29 14:39:12 +0100165 else
166 setf st
167 endif
168enddef
169
Bram Moolenaara2baa732022-02-04 16:09:54 +0000170export def FTlpc()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100171 if exists("g:lpc_syntax_for_c")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000172 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100173 while lnum <= 12
174 if getline(lnum) =~# '^\(//\|inherit\|private\|protected\|nosave\|string\|object\|mapping\|mixed\)'
175 setf lpc
176 return
177 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000178 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100179 endwhile
180 endif
181 setf c
Bram Moolenaara2baa732022-02-04 16:09:54 +0000182enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100183
Bram Moolenaara2baa732022-02-04 16:09:54 +0000184export def FTheader()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100185 if match(getline(1, min([line("$"), 200])), '^@\(interface\|end\|class\)') > -1
186 if exists("g:c_syntax_for_h")
187 setf objc
188 else
189 setf objcpp
190 endif
191 elseif exists("g:c_syntax_for_h")
192 setf c
193 elseif exists("g:ch_syntax_for_h")
194 setf ch
195 else
196 setf cpp
197 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000198enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100199
Bram Moolenaara2baa732022-02-04 16:09:54 +0000200# This function checks if one of the first ten lines start with a '@'. In
201# that case it is probably a change file.
202# If the first line starts with # or ! it's probably a ch file.
203# If a line has "main", "include", "//" or "/*" it's probably ch.
204# Otherwise CHILL is assumed.
205export def FTchange()
206 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100207 while lnum <= 10
208 if getline(lnum)[0] == '@'
209 setf change
210 return
211 endif
212 if lnum == 1 && (getline(1)[0] == '#' || getline(1)[0] == '!')
213 setf ch
214 return
215 endif
216 if getline(lnum) =~ "MODULE"
217 setf chill
218 return
219 endif
220 if getline(lnum) =~ 'main\s*(\|#\s*include\|//'
221 setf ch
222 return
223 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000224 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100225 endwhile
226 setf chill
Bram Moolenaara2baa732022-02-04 16:09:54 +0000227enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100228
Bram Moolenaara2baa732022-02-04 16:09:54 +0000229export def FTent()
230 # This function checks for valid cl syntax in the first five lines.
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100231 # Look for either an opening comment, '#', or a block start, '{'.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000232 # If not found, assume SGML.
233 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100234 while lnum < 6
Bram Moolenaara2baa732022-02-04 16:09:54 +0000235 var line = getline(lnum)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100236 if line =~ '^\s*[#{]'
237 setf cl
238 return
239 elseif line !~ '^\s*$'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000240 # Not a blank line, not a comment, and not a block start,
241 # so doesn't look like valid cl code.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100242 break
243 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000244 lnum += 1
Bram Moolenaarc12dc472022-03-05 13:45:56 +0000245 endwhile
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100246 setf dtd
Bram Moolenaara2baa732022-02-04 16:09:54 +0000247enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100248
Bram Moolenaara2baa732022-02-04 16:09:54 +0000249export def ExCheck()
250 var lines = getline(1, min([line("$"), 100]))
Austin Gatlinf3caeb62021-06-26 12:02:55 +0200251 if exists('g:filetype_euphoria')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000252 exe 'setf ' .. g:filetype_euphoria
Austin Gatlinf3caeb62021-06-26 12:02:55 +0200253 elseif match(lines, '^--\|^ifdef\>\|^include\>') > -1
254 setf euphoria3
255 else
256 setf elixir
257 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000258enddef
Austin Gatlinf3caeb62021-06-26 12:02:55 +0200259
Bram Moolenaara2baa732022-02-04 16:09:54 +0000260export def EuphoriaCheck()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100261 if exists('g:filetype_euphoria')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000262 exe 'setf ' .. g:filetype_euphoria
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100263 else
264 setf euphoria3
265 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000266enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100267
Bram Moolenaara2baa732022-02-04 16:09:54 +0000268export def DtraceCheck()
=?UTF-8?q?Teubel=20Gy=C3=B6rgy?=4d56b972022-02-24 17:59:09 +0000269 if did_filetype()
270 # Filetype was already detected
271 return
272 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000273 var lines = getline(1, min([line("$"), 100]))
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100274 if match(lines, '^module\>\|^import\>') > -1
Bram Moolenaara2baa732022-02-04 16:09:54 +0000275 # D files often start with a module and/or import statement.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100276 setf d
277 elseif match(lines, '^#!\S\+dtrace\|#pragma\s\+D\s\+option\|:\S\{-}:\S\{-}:') > -1
278 setf dtrace
279 else
280 setf d
281 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000282enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100283
Doug Kearns68a89472024-01-05 17:59:04 +0100284export def FTdef()
Wu, Zhenyu61ee8332024-04-09 22:09:30 +0200285 # LaTeX def files are usually generated by docstrip, which will output '%%' in first line
286 if getline(1) =~ '%%'
287 setf tex
288 endif
Doug Kearns68a89472024-01-05 17:59:04 +0100289 if get(g:, "filetype_def", "") == "modula2" || IsModula2()
290 SetFiletypeModula2()
291 return
292 endif
293
294 if exists("g:filetype_def")
295 exe "setf " .. g:filetype_def
296 else
297 setf def
298 endif
299enddef
300
Bram Moolenaara2baa732022-02-04 16:09:54 +0000301export def FTe()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100302 if exists('g:filetype_euphoria')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000303 exe 'setf ' .. g:filetype_euphoria
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100304 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000305 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100306 while n < 100 && n <= line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100307 if getline(n) =~ "^\\s*\\(<'\\|'>\\)\\s*$"
308 setf specman
309 return
310 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000311 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100312 endwhile
313 setf eiffel
314 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000315enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100316
Doug Kearns19a3bc32023-08-20 20:51:12 +0200317def IsForth(): bool
318 var first_line = nextnonblank(1)
319
320 # SwiftForth block comment (line is usually filled with '-' or '=') or
321 # OPTIONAL (sometimes precedes the header comment)
322 if getline(first_line) =~? '^\%({\%(\s\|$\)\|OPTIONAL\s\)'
323 return true
324 endif
325
326 var n = first_line
327 while n < 100 && n <= line("$")
328 # Forth comments and colon definitions
329 if getline(n) =~ '^[:(\\] '
330 return true
331 endif
332 n += 1
333 endwhile
334 return false
335enddef
336
337# Distinguish between Forth and Fortran
338export def FTf()
339 if exists("g:filetype_f")
340 exe "setf " .. g:filetype_f
341 elseif IsForth()
342 setf forth
343 else
344 setf fortran
345 endif
346enddef
347
Bram Moolenaara2baa732022-02-04 16:09:54 +0000348export def FTfrm()
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000349 if exists("g:filetype_frm")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000350 exe "setf " .. g:filetype_frm
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000351 return
352 endif
353
Doug Kearnsf97f6bb2023-08-27 18:44:09 +0200354 if getline(1) == "VERSION 5.00"
355 setf vb
356 return
357 endif
358
Bram Moolenaara2baa732022-02-04 16:09:54 +0000359 var lines = getline(1, min([line("$"), 5]))
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000360
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000361 if match(lines, ft_visual_basic_content) > -1
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000362 setf vb
363 else
364 setf form
365 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000366enddef
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000367
Doug Kearns19a3bc32023-08-20 20:51:12 +0200368# Distinguish between Forth and F#
Bram Moolenaara2baa732022-02-04 16:09:54 +0000369export def FTfs()
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000370 if exists("g:filetype_fs")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000371 exe "setf " .. g:filetype_fs
Doug Kearns19a3bc32023-08-20 20:51:12 +0200372 elseif IsForth()
373 setf forth
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000374 else
Johan Kotlinski065088d2023-04-02 20:29:38 +0100375 setf fsharp
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000376 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000377enddef
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000378
Amelia Clarke35dfe582024-05-24 08:05:00 +0200379# Recursively search for Hare source files in a directory and any
380# subdirectories, up to a given depth.
381def IsHareModule(dir: string, depth: number): bool
382 if depth <= 0
383 return !empty(glob(dir .. '/*.ha'))
384 endif
385
386 return reduce(sort(glob(dir .. '/*', true, true),
387 (a, b) => isdirectory(a) - isdirectory(b)),
388 (acc, n) => acc
389 || n =~ '\.ha$'
390 || isdirectory(n)
391 && IsHareModule(n, depth - 1),
392 false)
393enddef
394
395# Determine if a README file exists within a Hare module and should be given the
396# Haredoc filetype.
397export def FTharedoc()
398 if exists('g:filetype_haredoc')
399 if IsHareModule('<afile>:h', get(g:, 'haredoc_search_depth', 1))
400 setf haredoc
401 endif
402 endif
403enddef
404
Dennis van den Berg1ad194c2024-07-09 19:25:33 +0200405# Distinguish between HTML, XHTML, Django and Angular
Bram Moolenaara2baa732022-02-04 16:09:54 +0000406export def FThtml()
407 var n = 1
Dennis van den Berg1ad194c2024-07-09 19:25:33 +0200408
409 # Test if the filename follows the Angular component template convention
Christian Brabandtc03f6312024-07-10 19:23:39 +0200410 # Disabled for the reasons mentioned here: #13594
411 # if expand('%:t') =~ '^.*\.component\.html$'
412 # setf htmlangular
413 # return
414 # endif
Dennis van den Berg1ad194c2024-07-09 19:25:33 +0200415
Afiq Nazrie9718ed72024-06-18 19:59:53 +0200416 while n < 40 && n <= line("$")
Dennis van den Berg1ad194c2024-07-09 19:25:33 +0200417 # Check for Angular
418 if getline(n) =~ '@\(if\|for\|defer\|switch\)\|\*\(ngIf\|ngFor\|ngSwitch\|ngTemplateOutlet\)\|ng-template\|ng-content\|{{.*}}'
419 setf htmlangular
420 return
421 endif
422
423 # Check for XHTML
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100424 if getline(n) =~ '\<DTD\s\+XHTML\s'
425 setf xhtml
426 return
427 endif
Dennis van den Berg1ad194c2024-07-09 19:25:33 +0200428 # Check for Django
Afiq Nazrie9718ed72024-06-18 19:59:53 +0200429 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 +0100430 setf htmldjango
431 return
432 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000433 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100434 endwhile
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100435 setf FALLBACK html
Bram Moolenaara2baa732022-02-04 16:09:54 +0000436enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100437
Bram Moolenaara2baa732022-02-04 16:09:54 +0000438# Distinguish between standard IDL and MS-IDL
439export def FTidl()
440 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100441 while n < 50 && n <= line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100442 if getline(n) =~ '^\s*import\s\+"\(unknwn\|objidl\)\.idl"'
443 setf msidl
444 return
445 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000446 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100447 endwhile
448 setf idl
Bram Moolenaara2baa732022-02-04 16:09:54 +0000449enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100450
Wu, Zhenyu887a38c2024-05-09 20:35:13 +0200451# Distinguish between "default", Prolog, zsh module's C and Cproto prototype file.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000452export def ProtoCheck(default: string)
Wu, Zhenyu887a38c2024-05-09 20:35:13 +0200453 # zsh modules use '#include "*.pro"'
454 # https://github.com/zsh-users/zsh/blob/63f086d167960a27ecdbcb762179e2c2bf8a29f5/Src/Modules/example.c#L31
455 if getline(1) =~ '/* Generated automatically */'
456 setf c
Bram Moolenaara2baa732022-02-04 16:09:54 +0000457 # Cproto files have a comment in the first line and a function prototype in
458 # the second line, it always ends in ";". Indent files may also have
459 # comments, thus we can't match comments to see the difference.
460 # IDL files can have a single ';' in the second line, require at least one
461 # chacter before the ';'.
Wu, Zhenyu887a38c2024-05-09 20:35:13 +0200462 elseif getline(2) =~ '.;$'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100463 setf cpp
464 else
Bram Moolenaar48c3f4e2022-08-08 15:42:38 +0100465 # recognize Prolog by specific text in the first non-empty line
466 # require a blank after the '%' because Perl uses "%list" and "%translate"
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100467 var lnum = getline(nextnonblank(1))
468 if lnum =~ '\<prolog\>' || lnum =~ '^\s*\(%\+\(\s\|$\)\|/\*\)' || lnum =~ ':-'
Bram Moolenaar48c3f4e2022-08-08 15:42:38 +0100469 setf prolog
470 else
471 exe 'setf ' .. default
472 endif
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100473 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000474enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100475
Bram Moolenaara2baa732022-02-04 16:09:54 +0000476export def FTm()
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200477 if exists("g:filetype_m")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000478 exe "setf " .. g:filetype_m
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200479 return
480 endif
481
Bram Moolenaara2baa732022-02-04 16:09:54 +0000482 # excluding end(for|function|if|switch|while) common to Murphi
483 var octave_block_terminators = '\<end\%(_try_catch\|classdef\|enumeration\|events\|methods\|parfor\|properties\)\>'
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200484
Bram Moolenaara2baa732022-02-04 16:09:54 +0000485 var objc_preprocessor = '^\s*#\s*\%(import\|include\|define\|if\|ifn\=def\|undef\|line\|error\|pragma\)\>'
Doug Kearns7329cfa2021-11-26 13:01:41 +0000486
Bram Moolenaara2baa732022-02-04 16:09:54 +0000487 var n = 1
488 var saw_comment = 0 # Whether we've seen a multiline comment leader.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100489 while n < 100
Bram Moolenaara2baa732022-02-04 16:09:54 +0000490 var line = getline(n)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100491 if line =~ '^\s*/\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000492 # /* ... */ is a comment in Objective C and Murphi, so we can't conclude
493 # it's either of them yet, but track this as a hint in case we don't see
494 # anything more definitive.
495 saw_comment = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100496 endif
Doug Kearns7329cfa2021-11-26 13:01:41 +0000497 if line =~ '^\s*//' || line =~ '^\s*@import\>' || line =~ objc_preprocessor
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100498 setf objc
499 return
500 endif
Bram Moolenaarca0627d2021-09-12 17:03:08 +0200501 if line =~ '^\s*\%(#\|%!\)' || line =~ '^\s*unwind_protect\>' ||
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200502 \ line =~ '\%(^\|;\)\s*' .. octave_block_terminators
503 setf octave
504 return
505 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000506 # TODO: could be Matlab or Octave
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100507 if line =~ '^\s*%'
508 setf matlab
509 return
510 endif
511 if line =~ '^\s*(\*'
512 setf mma
513 return
514 endif
515 if line =~ '^\c\s*\(\(type\|var\)\>\|--\)'
516 setf murphi
517 return
518 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000519 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100520 endwhile
521
522 if saw_comment
Bram Moolenaara2baa732022-02-04 16:09:54 +0000523 # We didn't see anything definitive, but this looks like either Objective C
524 # or Murphi based on the comment leader. Assume the former as it is more
525 # common.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100526 setf objc
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100527 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000528 # Default is Matlab
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100529 setf matlab
530 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000531enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100532
Bram Moolenaara2baa732022-02-04 16:09:54 +0000533export def FTmms()
534 var n = 1
Bram Moolenaard7df2792020-01-02 21:34:42 +0100535 while n < 20
Bram Moolenaara2baa732022-02-04 16:09:54 +0000536 var line = getline(n)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100537 if line =~ '^\s*\(%\|//\)' || line =~ '^\*'
538 setf mmix
539 return
540 endif
541 if line =~ '^\s*#'
542 setf make
543 return
544 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000545 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100546 endwhile
547 setf mmix
Bram Moolenaara2baa732022-02-04 16:09:54 +0000548enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100549
Bram Moolenaara2baa732022-02-04 16:09:54 +0000550# This function checks if one of the first five lines start with a dot. In
551# that case it is probably an nroff file: 'filetype' is set and 1 is returned.
552export def FTnroff(): number
553 if getline(1)[0] .. getline(2)[0] .. getline(3)[0]
554 .. getline(4)[0] .. getline(5)[0] =~ '\.'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100555 setf nroff
556 return 1
557 endif
558 return 0
Bram Moolenaara2baa732022-02-04 16:09:54 +0000559enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100560
Bram Moolenaara2baa732022-02-04 16:09:54 +0000561export def FTmm()
562 var n = 1
Bram Moolenaard1caa942020-04-10 22:10:56 +0200563 while n < 20
Bram Moolenaara2baa732022-02-04 16:09:54 +0000564 if getline(n) =~ '^\s*\(#\s*\(include\|import\)\>\|@import\>\|/\*\)'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100565 setf objcpp
566 return
567 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000568 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100569 endwhile
570 setf nroff
Bram Moolenaara2baa732022-02-04 16:09:54 +0000571enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100572
Bram Moolenaarcdbfc6d2022-06-30 16:25:21 +0100573# Returns true if file content looks like LambdaProlog module
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100574def IsLProlog(): bool
Gaetan Lepage4ce1bda2023-05-10 22:01:55 +0100575 # skip apparent comments and blank lines, what looks like
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100576 # LambdaProlog comment may be RAPID header
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100577 var lnum: number = nextnonblank(1)
578 while lnum > 0 && lnum < line('$') && getline(lnum) =~ '^\s*%' # LambdaProlog comment
579 lnum = nextnonblank(lnum + 1)
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100580 endwhile
581 # this pattern must not catch a go.mod file
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100582 return getline(lnum) =~ '\<module\s\+\w\+\s*\.\s*\(%\|$\)'
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100583enddef
584
Doug Kearns68a89472024-01-05 17:59:04 +0100585def IsModula2(): bool
dkearnsef387c02024-02-20 06:58:30 +1100586 return getline(nextnonblank(1)) =~ '\<MODULE\s\+\w\+\s*\%(\[.*]\s*\)\=;\|^\s*(\*'
Doug Kearns68a89472024-01-05 17:59:04 +0100587enddef
588
589def SetFiletypeModula2()
590 const KNOWN_DIALECTS = ["iso", "pim", "r10"]
591 const KNOWN_EXTENSIONS = ["gm2"]
592 const LINE_COUNT = 200
593 const TAG = '(\*!m2\(\w\+\)\%(+\(\w\+\)\)\=\*)'
594
595 var dialect = get(g:, "modula2_default_dialect", "pim")
596 var extension = get(g:, "modula2_default_extension", "")
597
598 var matches = []
599
600 # ignore unknown dialects or badly formatted tags
601 for lnum in range(1, min([line("$"), LINE_COUNT]))
602 matches = matchlist(getline(lnum), TAG)
603 if !empty(matches)
604 if index(KNOWN_DIALECTS, matches[1]) >= 0
605 dialect = matches[1]
606 endif
607 if index(KNOWN_EXTENSIONS, matches[2]) >= 0
608 extension = matches[2]
609 endif
610 break
611 endif
612 endfor
613
614 modula2#SetDialect(dialect, extension)
615
616 setf modula2
617enddef
618
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100619# Determine if *.mod is ABB RAPID, LambdaProlog, Modula-2, Modsim III or go.mod
620export def FTmod()
Doug Kearns68a89472024-01-05 17:59:04 +0100621 if get(g:, "filetype_mod", "") == "modula2" || IsModula2()
622 SetFiletypeModula2()
623 return
624 endif
625
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100626 if exists("g:filetype_mod")
627 exe "setf " .. g:filetype_mod
Omar El Halabic9fbd252023-05-29 19:59:45 +0100628 elseif expand("<afile>") =~ '\<go.mod$'
629 setf gomod
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100630 elseif IsLProlog()
631 setf lprolog
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100632 elseif IsRapid()
633 setf rapid
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100634 else
635 # Nothing recognized, assume modsim3
636 setf modsim3
637 endif
638enddef
639
Bram Moolenaara2baa732022-02-04 16:09:54 +0000640export def FTpl()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100641 if exists("g:filetype_pl")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000642 exe "setf " .. g:filetype_pl
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100643 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000644 # recognize Prolog by specific text in the first non-empty line
645 # require a blank after the '%' because Perl uses "%list" and "%translate"
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100646 var line = getline(nextnonblank(1))
647 if line =~ '\<prolog\>' || line =~ '^\s*\(%\+\(\s\|$\)\|/\*\)' || line =~ ':-'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100648 setf prolog
649 else
650 setf perl
651 endif
652 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000653enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100654
Bram Moolenaara2baa732022-02-04 16:09:54 +0000655export def FTinc()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100656 if exists("g:filetype_inc")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000657 exe "setf " .. g:filetype_inc
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100658 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000659 var lines = getline(1) .. getline(2) .. getline(3)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100660 if lines =~? "perlscript"
661 setf aspperl
662 elseif lines =~ "<%"
663 setf aspvbs
664 elseif lines =~ "<?"
665 setf php
Bram Moolenaara2baa732022-02-04 16:09:54 +0000666 # Pascal supports // comments but they're vary rarely used for file
667 # headers so assume POV-Ray
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000668 elseif lines =~ '^\s*\%({\|(\*\)' || lines =~? ft_pascal_keywords
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100669 setf pascal
Gregory Anders30e212d2022-07-26 21:42:03 +0100670 elseif lines =~# '\<\%(require\|inherit\)\>' || lines =~# '[A-Z][A-Za-z0-9_:${}]*\s\+\%(??\|[?:+]\)\?= '
Gregory Andersfa49eb42022-07-16 17:46:47 +0100671 setf bitbake
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100672 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000673 FTasmsyntax()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100674 if exists("b:asmsyntax")
Gregory Andersfa49eb42022-07-16 17:46:47 +0100675 exe "setf " .. fnameescape(b:asmsyntax)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100676 else
Gregory Andersfa49eb42022-07-16 17:46:47 +0100677 setf pov
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100678 endif
679 endif
680 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000681enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100682
Bram Moolenaara2baa732022-02-04 16:09:54 +0000683export def FTprogress_cweb()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100684 if exists("g:filetype_w")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000685 exe "setf " .. g:filetype_w
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100686 return
687 endif
688 if getline(1) =~ '&ANALYZE' || getline(3) =~ '&GLOBAL-DEFINE'
689 setf progress
690 else
691 setf cweb
692 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000693enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100694
Julien Marrec2e310652023-11-25 15:30:46 +0100695# These include the leading '%' sign
696var 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\)'
697# This is the start/end of a block that is copied literally to the processor file (C/C++)
698var ft_swig_verbatim_block_start = '^\s*%{'
699
700export def FTi()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100701 if exists("g:filetype_i")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000702 exe "setf " .. g:filetype_i
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100703 return
704 endif
Julien Marrec2e310652023-11-25 15:30:46 +0100705 # 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 +0000706 # If not found, assume Progress.
707 var lnum = 1
Julien Marrec2e310652023-11-25 15:30:46 +0100708 while lnum <= 50 && lnum < line('$')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000709 var line = getline(lnum)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100710 if line =~ '^\s*;' || line =~ '^\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000711 FTasm()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100712 return
Julien Marrec2e310652023-11-25 15:30:46 +0100713 elseif line =~ ft_swig_keywords || line =~ ft_swig_verbatim_block_start
714 setf swig
715 return
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100716 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000717 lnum += 1
Bram Moolenaarc12dc472022-03-05 13:45:56 +0000718 endwhile
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100719 setf progress
Bram Moolenaara2baa732022-02-04 16:09:54 +0000720enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100721
Bram Moolenaara2baa732022-02-04 16:09:54 +0000722var ft_pascal_comments = '^\s*\%({\|(\*\|//\)'
723var ft_pascal_keywords = '^\s*\%(program\|unit\|library\|uses\|begin\|procedure\|function\|const\|type\|var\)\>'
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100724
Bram Moolenaara2baa732022-02-04 16:09:54 +0000725export def FTprogress_pascal()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100726 if exists("g:filetype_p")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000727 exe "setf " .. g:filetype_p
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100728 return
729 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000730 # This function checks for valid Pascal syntax in the first ten lines.
731 # Look for either an opening comment or a program start.
732 # If not found, assume Progress.
733 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100734 while lnum <= 10 && lnum < line('$')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000735 var line = getline(lnum)
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000736 if line =~ ft_pascal_comments || line =~? ft_pascal_keywords
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100737 setf pascal
738 return
739 elseif line !~ '^\s*$' || line =~ '^/\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000740 # Not an empty line: Doesn't look like valid Pascal code.
741 # Or it looks like a Progress /* comment
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100742 break
743 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000744 lnum += 1
Bram Moolenaarc12dc472022-03-05 13:45:56 +0000745 endwhile
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100746 setf progress
Bram Moolenaara2baa732022-02-04 16:09:54 +0000747enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100748
Bram Moolenaara2baa732022-02-04 16:09:54 +0000749export def FTpp()
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100750 if exists("g:filetype_pp")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000751 exe "setf " .. g:filetype_pp
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100752 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000753 var line = getline(nextnonblank(1))
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000754 if line =~ ft_pascal_comments || line =~? ft_pascal_keywords
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100755 setf pascal
756 else
757 setf puppet
758 endif
759 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000760enddef
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100761
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100762# Determine if *.prg is ABB RAPID. Can also be Clipper, FoxPro or eviews
763export def FTprg()
764 if exists("g:filetype_prg")
765 exe "setf " .. g:filetype_prg
766 elseif IsRapid()
767 setf rapid
768 else
769 # Nothing recognized, assume Clipper
770 setf clipper
771 endif
772enddef
773
Bram Moolenaara2baa732022-02-04 16:09:54 +0000774export def FTr()
775 var max = line("$") > 50 ? 50 : line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100776
777 for n in range(1, max)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000778 # Rebol is easy to recognize, check for that first
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100779 if getline(n) =~? '\<REBOL\>'
780 setf rebol
781 return
782 endif
783 endfor
784
785 for n in range(1, max)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000786 # R has # comments
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100787 if getline(n) =~ '^\s*#'
788 setf r
789 return
790 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000791 # Rexx has /* comments */
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100792 if getline(n) =~ '^\s*/\*'
793 setf rexx
794 return
795 endif
796 endfor
797
Bram Moolenaara2baa732022-02-04 16:09:54 +0000798 # Nothing recognized, use user default or assume Rexx
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100799 if exists("g:filetype_r")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000800 exe "setf " .. g:filetype_r
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100801 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000802 # Rexx used to be the default, but R appears to be much more popular.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100803 setf r
804 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000805enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100806
Bram Moolenaara2baa732022-02-04 16:09:54 +0000807export def McSetf()
808 # Rely on the file to start with a comment.
809 # MS message text files use ';', Sendmail files use '#' or 'dnl'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100810 for lnum in range(1, min([line("$"), 20]))
Bram Moolenaara2baa732022-02-04 16:09:54 +0000811 var line = getline(lnum)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100812 if line =~ '^\s*\(#\|dnl\)'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000813 setf m4 # Sendmail .mc file
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100814 return
815 elseif line =~ '^\s*;'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000816 setf msmessages # MS Message text file
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100817 return
818 endif
819 endfor
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100820 setf m4 # Default: Sendmail .mc file
Bram Moolenaara2baa732022-02-04 16:09:54 +0000821enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100822
Bram Moolenaara2baa732022-02-04 16:09:54 +0000823# Called from filetype.vim and scripts.vim.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100824# When "setft" is passed and false then the 'filetype' option is not set.
825export def SetFileTypeSH(name: string, setft = true): string
826 if setft && did_filetype()
Bram Moolenaara2baa732022-02-04 16:09:54 +0000827 # Filetype was already detected
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100828 return ''
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100829 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100830 if setft && expand("<amatch>") =~ g:ft_ignore_pat
831 return ''
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100832 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000833 if name =~ '\<csh\>'
834 # Some .sh scripts contain #!/bin/csh.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100835 return SetFileTypeShell("csh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000836 elseif name =~ '\<tcsh\>'
837 # Some .sh scripts contain #!/bin/tcsh.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100838 return SetFileTypeShell("tcsh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000839 elseif name =~ '\<zsh\>'
840 # Some .sh scripts contain #!/bin/zsh.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100841 return SetFileTypeShell("zsh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000842 elseif name =~ '\<ksh\>'
843 b:is_kornshell = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100844 if exists("b:is_bash")
845 unlet b:is_bash
846 endif
847 if exists("b:is_sh")
848 unlet b:is_sh
849 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000850 elseif exists("g:bash_is_sh") || name =~ '\<bash\>' || name =~ '\<bash2\>'
851 b:is_bash = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100852 if exists("b:is_kornshell")
853 unlet b:is_kornshell
854 endif
855 if exists("b:is_sh")
856 unlet b:is_sh
857 endif
Eisuke Kawashima24482fb2022-11-24 10:58:10 +0000858 elseif name =~ '\<sh\>' || name =~ '\<dash\>'
859 # Ubuntu links "sh" to "dash", thus it is expected to work the same way
Bram Moolenaara2baa732022-02-04 16:09:54 +0000860 b:is_sh = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100861 if exists("b:is_kornshell")
862 unlet b:is_kornshell
863 endif
864 if exists("b:is_bash")
865 unlet b:is_bash
866 endif
867 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100868
869 return SetFileTypeShell("sh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000870enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100871
Bram Moolenaara2baa732022-02-04 16:09:54 +0000872# For shell-like file types, check for an "exec" command hidden in a comment,
873# as used for Tcl.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100874# When "setft" is passed and false then the 'filetype' option is not set.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000875# Also called from scripts.vim, thus can't be local to this script.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100876export def SetFileTypeShell(name: string, setft = true): string
877 if setft && did_filetype()
Bram Moolenaara2baa732022-02-04 16:09:54 +0000878 # Filetype was already detected
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100879 return ''
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100880 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100881 if setft && expand("<amatch>") =~ g:ft_ignore_pat
882 return ''
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100883 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100884
885 var lnum = 2
886 while lnum < 20 && lnum < line("$") && getline(lnum) =~ '^\s*\(#\|$\)'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000887 # Skip empty and comment lines.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100888 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100889 endwhile
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100890 if lnum < line("$") && getline(lnum) =~ '\s*exec\s' && getline(lnum - 1) =~ '^\s*#.*\\$'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000891 # Found an "exec" line after a comment with continuation
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100892 var n = substitute(getline(lnum), '\s*exec\s\+\([^ ]*/\)\=', '', '')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100893 if n =~ '\<tclsh\|\<wish'
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100894 if setft
895 setf tcl
896 endif
897 return 'tcl'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100898 endif
899 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100900
901 if setft
902 exe "setf " .. name
903 endif
904 return name
Bram Moolenaara2baa732022-02-04 16:09:54 +0000905enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100906
Bram Moolenaara2baa732022-02-04 16:09:54 +0000907export def CSH()
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100908 if did_filetype()
Bram Moolenaara2baa732022-02-04 16:09:54 +0000909 # Filetype was already detected
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100910 return
911 endif
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100912 if exists("g:filetype_csh")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000913 SetFileTypeShell(g:filetype_csh)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100914 elseif &shell =~ "tcsh"
Bram Moolenaara2baa732022-02-04 16:09:54 +0000915 SetFileTypeShell("tcsh")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100916 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000917 SetFileTypeShell("csh")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100918 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000919enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100920
Bram Moolenaara2baa732022-02-04 16:09:54 +0000921var ft_rules_udev_rules_pattern = '^\s*\cudev_rules\s*=\s*"\([^"]\{-1,}\)/*".*'
922export def FTRules()
923 var path = expand('<amatch>:p')
Bram Moolenaaraa9675a2020-08-17 21:57:09 +0200924 if path =~ '/\(etc/udev/\%(rules\.d/\)\=.*\.rules\|\%(usr/\)\=lib/udev/\%(rules\.d/\)\=.*\.rules\)$'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100925 setf udevrules
926 return
927 endif
928 if path =~ '^/etc/ufw/'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000929 setf conf # Better than hog
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100930 return
931 endif
932 if path =~ '^/\(etc\|usr/share\)/polkit-1/rules\.d'
933 setf javascript
934 return
935 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000936 var config_lines: list<string>
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100937 try
Bram Moolenaara2baa732022-02-04 16:09:54 +0000938 config_lines = readfile('/etc/udev/udev.conf')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100939 catch /^Vim\%((\a\+)\)\=:E484/
940 setf hog
941 return
942 endtry
Bram Moolenaara2baa732022-02-04 16:09:54 +0000943 var dir = expand('<amatch>:p:h')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100944 for line in config_lines
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000945 if line =~ ft_rules_udev_rules_pattern
946 var udev_rules = substitute(line, ft_rules_udev_rules_pattern, '\1', "")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100947 if dir == udev_rules
948 setf udevrules
949 endif
950 break
951 endif
952 endfor
953 setf hog
Bram Moolenaara2baa732022-02-04 16:09:54 +0000954enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100955
Bram Moolenaara2baa732022-02-04 16:09:54 +0000956export def SQL()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100957 if exists("g:filetype_sql")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000958 exe "setf " .. g:filetype_sql
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100959 else
960 setf sql
961 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000962enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100963
ranjithshegde8cac20e2022-04-13 15:29:21 +0100964# This function checks the first 25 lines of file extension "sc" to resolve
Chris Kipp70ef3f52022-12-14 16:42:15 +0000965# detection between scala and SuperCollider.
966# NOTE: We don't check for 'Class : Method', as this can easily be confused
967# with valid Scala like `val x : Int = 3`. So we instead only rely on
968# checks that can't be confused.
ranjithshegde8cac20e2022-04-13 15:29:21 +0100969export def FTsc()
970 for lnum in range(1, min([line("$"), 25]))
Chris Kipp70ef3f52022-12-14 16:42:15 +0000971 if getline(lnum) =~# 'var\s<\|classvar\s<\|\^this.*\||\w\+|\|+\s\w*\s{\|\*ar\s'
ranjithshegde8cac20e2022-04-13 15:29:21 +0100972 setf supercollider
973 return
974 endif
975 endfor
976 setf scala
977enddef
978
979# This function checks the first line of file extension "scd" to resolve
980# detection between scdoc and SuperCollider
981export def FTscd()
982 if getline(1) =~# '\%^\S\+(\d[0-9A-Za-z]*)\%(\s\+\"[^"]*\"\%(\s\+\"[^"]*\"\)\=\)\=$'
983 setf scdoc
984 else
985 setf supercollider
986 endif
987enddef
988
Bram Moolenaara2baa732022-02-04 16:09:54 +0000989# If the file has an extension of 't' and is in a directory 't' or 'xt' then
990# it is almost certainly a Perl test file.
991# If the first line starts with '#' and contains 'perl' it's probably a Perl
992# file.
993# (Slow test) If a file contains a 'use' statement then it is almost certainly
994# a Perl file.
995export def FTperl(): number
996 var dirname = expand("%:p:h:t")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100997 if expand("%:e") == 't' && (dirname == 't' || dirname == 'xt')
998 setf perl
999 return 1
1000 endif
1001 if getline(1)[0] == '#' && getline(1) =~ 'perl'
1002 setf perl
1003 return 1
1004 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001005 var save_cursor = getpos('.')
1006 call cursor(1, 1)
Bram Moolenaare5b78972022-02-05 19:50:34 +00001007 var has_use = search('^use\s\s*\k', 'c', 30) > 0
Bram Moolenaarf0b03c42017-12-17 17:17:07 +01001008 call setpos('.', save_cursor)
1009 if has_use
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001010 setf perl
1011 return 1
1012 endif
1013 return 0
Bram Moolenaara2baa732022-02-04 16:09:54 +00001014enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001015
Bram Moolenaarcdbfc6d2022-06-30 16:25:21 +01001016# LambdaProlog and Standard ML signature files
1017export def FTsig()
1018 if exists("g:filetype_sig")
1019 exe "setf " .. g:filetype_sig
1020 return
1021 endif
1022
1023 var lprolog_comment = '^\s*\%(/\*\|%\)'
1024 var lprolog_keyword = '^\s*sig\s\+\a'
1025 var sml_comment = '^\s*(\*'
1026 var sml_keyword = '^\s*\%(signature\|structure\)\s\+\a'
1027
1028 var line = getline(nextnonblank(1))
1029
1030 if line =~ lprolog_comment || line =~# lprolog_keyword
1031 setf lprolog
1032 elseif line =~ sml_comment || line =~# sml_keyword
1033 setf sml
1034 endif
1035enddef
1036
Bram Moolenaarbe807d52022-09-01 15:01:25 +01001037# This function checks the first 100 lines of files matching "*.sil" to
1038# resolve detection between Swift Intermediate Language and SILE.
1039export def FTsil()
1040 for lnum in range(1, [line('$'), 100]->min())
1041 var line: string = getline(lnum)
1042 if line =~ '^\s*[\\%]'
1043 setf sile
1044 return
1045 elseif line =~ '^\s*\S'
1046 setf sil
1047 return
1048 endif
1049 endfor
1050 # no clue, default to "sil"
1051 setf sil
1052enddef
1053
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +01001054export def FTsys()
KnoP-01f420ff22022-04-13 20:46:21 +01001055 if exists("g:filetype_sys")
1056 exe "setf " .. g:filetype_sys
1057 elseif IsRapid()
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +01001058 setf rapid
1059 else
1060 setf bat
1061 endif
1062enddef
1063
Bram Moolenaara2baa732022-02-04 16:09:54 +00001064# Choose context, plaintex, or tex (LaTeX) based on these rules:
1065# 1. Check the first line of the file for "%&<format>".
1066# 2. Check the first 1000 non-comment lines for LaTeX or ConTeXt keywords.
1067# 3. Default to "plain" or to g:tex_flavor, can be set in user's vimrc.
1068export def FTtex()
1069 var firstline = getline(1)
1070 var format: string
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001071 if firstline =~ '^%&\s*\a\+'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001072 format = tolower(matchstr(firstline, '\a\+'))
1073 format = substitute(format, 'pdf', '', '')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001074 if format == 'tex'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001075 format = 'latex'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001076 elseif format == 'plaintex'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001077 format = 'plain'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001078 endif
1079 elseif expand('%') =~ 'tex/context/.*/.*.tex'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001080 format = 'context'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001081 else
Bram Moolenaara2baa732022-02-04 16:09:54 +00001082 # Default value, may be changed later:
1083 format = exists("g:tex_flavor") ? g:tex_flavor : 'plain'
1084 # Save position, go to the top of the file, find first non-comment line.
1085 var save_cursor = getpos('.')
1086 call cursor(1, 1)
1087 var firstNC = search('^\s*[^[:space:]%]', 'c', 1000)
Bram Moolenaare5b78972022-02-05 19:50:34 +00001088 if firstNC > 0
1089 # Check the next thousand lines for a LaTeX or ConTeXt keyword.
Bram Moolenaara2baa732022-02-04 16:09:54 +00001090 var lpat = 'documentclass\>\|usepackage\>\|begin{\|newcommand\>\|renewcommand\>'
1091 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\>'
1092 var kwline = search('^\s*\\\%(' .. lpat .. '\)\|^\s*\\\(' .. cpat .. '\)',
1093 'cnp', firstNC + 1000)
1094 if kwline == 1 # lpat matched
1095 format = 'latex'
1096 elseif kwline == 2 # cpat matched
1097 format = 'context'
1098 endif # If neither matched, keep default set above.
1099 # let lline = search('^\s*\\\%(' . lpat . '\)', 'cn', firstNC + 1000)
1100 # let cline = search('^\s*\\\%(' . cpat . '\)', 'cn', firstNC + 1000)
1101 # if cline > 0
1102 # let format = 'context'
1103 # endif
1104 # if lline > 0 && (cline == 0 || cline > lline)
1105 # let format = 'tex'
1106 # endif
1107 endif # firstNC
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001108 call setpos('.', save_cursor)
Bram Moolenaara2baa732022-02-04 16:09:54 +00001109 endif # firstline =~ '^%&\s*\a\+'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001110
Bram Moolenaara2baa732022-02-04 16:09:54 +00001111 # Translation from formats to file types. TODO: add AMSTeX, RevTex, others?
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001112 if format == 'plain'
1113 setf plaintex
1114 elseif format == 'context'
1115 setf context
Bram Moolenaara2baa732022-02-04 16:09:54 +00001116 else # probably LaTeX
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001117 setf tex
1118 endif
1119 return
Bram Moolenaara2baa732022-02-04 16:09:54 +00001120enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001121
Bram Moolenaara2baa732022-02-04 16:09:54 +00001122export def FTxml()
1123 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +01001124 while n < 100 && n <= line("$")
Bram Moolenaara2baa732022-02-04 16:09:54 +00001125 var line = getline(n)
1126 # DocBook 4 or DocBook 5.
1127 var is_docbook4 = line =~ '<!DOCTYPE.*DocBook'
1128 var is_docbook5 = line =~ ' xmlns="http://docbook.org/ns/docbook"'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001129 if is_docbook4 || is_docbook5
Bram Moolenaara2baa732022-02-04 16:09:54 +00001130 b:docbk_type = "xml"
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001131 if is_docbook5
Bram Moolenaara2baa732022-02-04 16:09:54 +00001132 b:docbk_ver = 5
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001133 else
Bram Moolenaara2baa732022-02-04 16:09:54 +00001134 b:docbk_ver = 4
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001135 endif
1136 setf docbk
1137 return
1138 endif
1139 if line =~ 'xmlns:xbl="http://www.mozilla.org/xbl"'
1140 setf xbl
1141 return
1142 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001143 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001144 endwhile
1145 setf xml
Bram Moolenaara2baa732022-02-04 16:09:54 +00001146enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001147
Bram Moolenaara2baa732022-02-04 16:09:54 +00001148export def FTy()
1149 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +01001150 while n < 100 && n <= line("$")
Bram Moolenaara2baa732022-02-04 16:09:54 +00001151 var line = getline(n)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001152 if line =~ '^\s*%'
1153 setf yacc
1154 return
1155 endif
1156 if getline(n) =~ '^\s*\(#\|class\>\)' && getline(n) !~ '^\s*#\s*include'
1157 setf racc
1158 return
1159 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001160 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001161 endwhile
1162 setf yacc
Bram Moolenaara2baa732022-02-04 16:09:54 +00001163enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001164
Bram Moolenaara2baa732022-02-04 16:09:54 +00001165export def Redif()
1166 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001167 while lnum <= 5 && lnum < line('$')
1168 if getline(lnum) =~ "^\ctemplate-type:"
1169 setf redif
1170 return
1171 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001172 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001173 endwhile
Bram Moolenaara2baa732022-02-04 16:09:54 +00001174enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001175
Bram Moolenaara2baa732022-02-04 16:09:54 +00001176# This function is called for all files under */debian/patches/*, make sure not
1177# to non-dep3patch files, such as README and other text files.
1178export def Dep3patch()
James McCoy647ab4c2021-12-17 20:52:57 +00001179 if expand('%:t') ==# 'series'
1180 return
1181 endif
1182
1183 for ln in getline(1, 100)
1184 if ln =~# '^\%(Description\|Subject\|Origin\|Bug\|Forwarded\|Author\|From\|Reviewed-by\|Acked-by\|Last-Updated\|Applied-Upstream\):'
1185 setf dep3patch
1186 return
1187 elseif ln =~# '^---'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001188 # end of headers found. stop processing
James McCoy647ab4c2021-12-17 20:52:57 +00001189 return
1190 endif
1191 endfor
Bram Moolenaara2baa732022-02-04 16:09:54 +00001192enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001193
Bram Moolenaara2baa732022-02-04 16:09:54 +00001194# This function checks the first 15 lines for appearance of 'FoamFile'
1195# and then 'object' in a following line.
1196# In that case, it's probably an OpenFOAM file
1197export def FTfoam()
1198 var ffile = 0
1199 var lnum = 1
Elwardi2284f6c2022-01-11 18:14:23 +00001200 while lnum <= 15
1201 if getline(lnum) =~# '^FoamFile'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001202 ffile = 1
Elwardi2284f6c2022-01-11 18:14:23 +00001203 elseif ffile == 1 && getline(lnum) =~# '^\s*object'
1204 setf foam
1205 return
1206 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001207 lnum += 1
Elwardi2284f6c2022-01-11 18:14:23 +00001208 endwhile
Bram Moolenaara2baa732022-02-04 16:09:54 +00001209enddef
Elwardi2284f6c2022-01-11 18:14:23 +00001210
Bram Moolenaara2baa732022-02-04 16:09:54 +00001211# Determine if a *.tf file is TF mud client or terraform
1212export def FTtf()
1213 var numberOfLines = line('$')
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001214 for i in range(1, numberOfLines)
Bram Moolenaara2baa732022-02-04 16:09:54 +00001215 var currentLine = trim(getline(i))
1216 var firstCharacter = currentLine[0]
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001217 if firstCharacter !=? ";" && firstCharacter !=? "/" && firstCharacter !=? ""
1218 setf terraform
1219 return
1220 endif
1221 endfor
1222 setf tf
Bram Moolenaara2baa732022-02-04 16:09:54 +00001223enddef
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001224
KnoP-0193c7a452022-04-16 21:14:04 +01001225var ft_krl_header = '\&\w+'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001226# Determine if a *.src file is Kuka Robot Language
1227export def FTsrc()
KnoP-0193c7a452022-04-16 21:14:04 +01001228 var ft_krl_def_or_deffct = '%(global\s+)?def%(fct)?>'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001229 if exists("g:filetype_src")
1230 exe "setf " .. g:filetype_src
KnoP-0193c7a452022-04-16 21:14:04 +01001231 elseif getline(nextnonblank(1)) =~? '\v^\s*%(' .. ft_krl_header .. '|' .. ft_krl_def_or_deffct .. ')'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001232 setf krl
1233 endif
1234enddef
1235
1236# Determine if a *.dat file is Kuka Robot Language
1237export def FTdat()
KnoP-0193c7a452022-04-16 21:14:04 +01001238 var ft_krl_defdat = 'defdat>'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001239 if exists("g:filetype_dat")
1240 exe "setf " .. g:filetype_dat
KnoP-0193c7a452022-04-16 21:14:04 +01001241 elseif getline(nextnonblank(1)) =~? '\v^\s*%(' .. ft_krl_header .. '|' .. ft_krl_defdat .. ')'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001242 setf krl
1243 endif
1244enddef
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001245
Doug Kearns4ac8e792022-10-17 13:32:17 +01001246export def FTlsl()
1247 if exists("g:filetype_lsl")
1248 exe "setf " .. g:filetype_lsl
1249 endif
1250
1251 var line = getline(nextnonblank(1))
1252 if line =~ '^\s*%' || line =~# ':\s*trait\s*$'
1253 setf larch
1254 else
1255 setf lsl
1256 endif
1257enddef
1258
Gaetan Lepage4ce1bda2023-05-10 22:01:55 +01001259export def FTtyp()
1260 if exists("g:filetype_typ")
1261 exe "setf " .. g:filetype_typ
1262 return
1263 endif
1264
1265 # Look for SQL type definition syntax
1266 for line in getline(1, 200)
1267 # SQL type files may define the casing
1268 if line =~ '^CASE\s\==\s\=\(SAME\|LOWER\|UPPER\|OPPOSITE\)$'
1269 setf sql
1270 return
1271 endif
1272
1273 # SQL type files may define some types as follows
1274 if line =~ '^TYPE\s.*$'
1275 setf sql
1276 return
1277 endif
1278 endfor
1279
1280 # Otherwise, affect the typst filetype
1281 setf typst
1282enddef
1283
PowerUser64aa61b8a2024-06-19 20:32:11 +02001284# Detect Microsoft Developer Studio Project files (Makefile) or Faust DSP
1285# files.
1286export def FTdsp()
1287 if exists("g:filetype_dsp")
1288 exe "setf " .. g:filetype_dsp
1289 return
1290 endif
1291
1292 # Test the filename
1293 if expand('%:t') =~ '^[mM]akefile.*$'
1294 setf make
1295 return
1296 endif
1297
1298 # Test the file contents
1299 for line in getline(1, 200)
1300 # Chech for comment style
1301 if line =~ '^#.*'
1302 setf make
1303 return
1304 endif
1305
1306 # Check for common lines
1307 if line =~ '^.*Microsoft Developer Studio Project File.*$'
1308 setf make
1309 return
1310 endif
1311
1312 if line =~ '^!MESSAGE This is not a valid makefile\..+$'
1313 setf make
1314 return
1315 endif
1316
1317 # Check for keywords
1318 if line =~ '^!(IF,ELSEIF,ENDIF).*$'
1319 setf make
1320 return
1321 endif
1322
1323 # Check for common assignments
1324 if line =~ '^SOURCE=.*$'
1325 setf make
1326 return
1327 endif
1328 endfor
1329
1330 # Otherwise, assume we have a Faust file
1331 setf faust
1332enddef
1333
Turiiya80406c22023-04-22 21:38:47 +01001334# Set the filetype of a *.v file to Verilog, V or Cog based on the first 200
1335# lines.
1336export def FTv()
1337 if did_filetype()
1338 # ":setf" will do nothing, bail out early
1339 return
1340 endif
Christian Brabandt10b4f752024-01-01 19:19:20 +01001341 if exists("g:filetype_v")
1342 exe "setf " .. g:filetype_v
1343 return
1344 endif
Turiiya80406c22023-04-22 21:38:47 +01001345
Christian Brabandt10b4f752024-01-01 19:19:20 +01001346 var in_comment = 0
1347 for lnum in range(1, min([line("$"), 200]))
1348 var line = getline(lnum)
1349 # Skip Verilog and V comments (lines and blocks).
1350 if line =~ '^\s*/\*'
1351 # start comment block
1352 in_comment = 1
1353 endif
1354 if in_comment == 1
1355 if line =~ '\*/'
1356 # end comment block
1357 in_comment = 0
1358 endif
1359 # skip comment-block line
1360 continue
1361 endif
1362 if line =~ '^\s*//'
Turiiya80406c22023-04-22 21:38:47 +01001363 # skip comment line
1364 continue
1365 endif
1366
Christian Brabandt10b4f752024-01-01 19:19:20 +01001367 # Coq: line ends with a '.' followed by an optional variable number of
1368 # spaces or contains the start of a comment, but not inside a Verilog or V
1369 # comment.
1370 # Example: "Definition x := 10. (*".
1371 if (line =~ '\.\s*$' && line !~ '/[/*]') || (line =~ '(\*' && line !~ '/[/*].*(\*')
1372 setf coq
1373 return
1374 endif
1375
Turiiya80406c22023-04-22 21:38:47 +01001376 # Verilog: line ends with ';' followed by an optional variable number of
1377 # spaces and an optional start of a comment.
1378 # Example: " b <= a + 1; // Add 1".
Christian Brabandt10b4f752024-01-01 19:19:20 +01001379 if line =~ ';\s*\(/[/*].*\)\?$'
Turiiya80406c22023-04-22 21:38:47 +01001380 setf verilog
1381 return
1382 endif
Turiiya80406c22023-04-22 21:38:47 +01001383 endfor
1384
1385 # No line matched, fall back to "v".
1386 setf v
1387enddef
1388
Doug Kearnsf97f6bb2023-08-27 18:44:09 +02001389export def FTvba()
1390 if getline(1) =~ '^["#] Vimball Archiver'
1391 setf vim
1392 else
1393 setf vb
1394 endif
1395enddef
1396
Colin Caine4b3fab12024-04-18 23:53:02 +02001397export def Detect_UCI_statements(): bool
1398 # Match a config or package statement at the start of the line.
1399 const config_or_package_statement = '^\s*\(\(c\|config\)\|\(p\|package\)\)\s\+\S'
1400 # Match a line that is either all blank or blank followed by a comment
1401 const comment_or_blank = '^\s*\(#.*\)\?$'
1402
1403 # Return true iff the file has a config or package statement near the
1404 # top of the file and all preceding lines were comments or blank.
1405 return getline(1) =~# config_or_package_statement
1406 \ || getline(1) =~# comment_or_blank
1407 \ && ( getline(2) =~# config_or_package_statement
1408 \ || getline(2) =~# comment_or_blank
1409 \ && getline(3) =~# config_or_package_statement
1410 \ )
1411enddef
1412
Bram Moolenaara2baa732022-02-04 16:09:54 +00001413# Uncomment this line to check for compilation errors early
dkearnsef387c02024-02-20 06:58:30 +11001414# defcompile