blob: a740305226c07ed98cb2020c4d5e35a4a1a2a50f [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>
Christian Brabandt99181202025-01-25 14:54:28 +01006# Last Change: 2025 Jan 25
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()
Wu, Zhenyu4f73c072025-01-08 20:20:06 +010035 # tiasm uses `* commment`
36 if join(getline(1, 10), "\n") =~ '\%(\%(^\|\n\)\*\|Texas Instruments Incorporated\)'
37 setf tiasm
38 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +000039 # make sure b:asmsyntax exists
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010040 if !exists("b:asmsyntax")
Bram Moolenaara2baa732022-02-04 16:09:54 +000041 b:asmsyntax = ""
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010042 endif
43
44 if b:asmsyntax == ""
Bram Moolenaara2baa732022-02-04 16:09:54 +000045 FTasmsyntax()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010046 endif
47
Bram Moolenaara2baa732022-02-04 16:09:54 +000048 # if b:asmsyntax still isn't set, default to asmsyntax or GNU
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010049 if b:asmsyntax == ""
50 if exists("g:asmsyntax")
Bram Moolenaara2baa732022-02-04 16:09:54 +000051 b:asmsyntax = g:asmsyntax
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010052 else
Bram Moolenaara2baa732022-02-04 16:09:54 +000053 b:asmsyntax = "asm"
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010054 endif
55 endif
56
Bram Moolenaara2baa732022-02-04 16:09:54 +000057 exe "setf " .. fnameescape(b:asmsyntax)
58enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010059
Bram Moolenaara2baa732022-02-04 16:09:54 +000060export def FTasmsyntax()
61 # see if the file contains any asmsyntax=foo overrides. If so, change
62 # b:asmsyntax appropriately
63 var head = " " .. getline(1) .. " " .. getline(2) .. " "
64 .. getline(3) .. " " .. getline(4) .. " " .. getline(5) .. " "
65 var match = matchstr(head, '\sasmsyntax=\zs[a-zA-Z0-9]\+\ze\s')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010066 if match != ''
Bram Moolenaara2baa732022-02-04 16:09:54 +000067 b:asmsyntax = match
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010068 elseif ((head =~? '\.title') || (head =~? '\.ident') || (head =~? '\.macro') || (head =~? '\.subtitle') || (head =~? '\.library'))
Bram Moolenaara2baa732022-02-04 16:09:54 +000069 b:asmsyntax = "vmasm"
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010070 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +000071enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010072
Doug Kearnsf97f6bb2023-08-27 18:44:09 +020073var ft_visual_basic_content = '\c^\s*\%(Attribute\s\+VB_Name\|Begin\s\+\%(VB\.\|{\%(\x\+-\)\+\x\+}\)\)'
Doug Kearnsc570e9c2022-01-31 17:09:14 +000074
Bram Moolenaara2baa732022-02-04 16:09:54 +000075# See FTfrm() for Visual Basic form file detection
76export def FTbas()
Bram Moolenaar6517f142022-01-21 14:55:13 +000077 if exists("g:filetype_bas")
Bram Moolenaara2baa732022-02-04 16:09:54 +000078 exe "setf " .. g:filetype_bas
Bram Moolenaar6517f142022-01-21 14:55:13 +000079 return
80 endif
81
Bram Moolenaara2baa732022-02-04 16:09:54 +000082 # most frequent FreeBASIC-specific keywords in distro files
83 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 +010084 var fb_preproc = '\c^\s*\%(' ..
85 # preprocessor
86 '#\s*\a\+\|' ..
87 # compiler option
88 'option\s\+\%(byval\|dynamic\|escape\|\%(no\)\=gosub\|nokeyword\|private\|static\)\>\|' ..
89 # metacommand
90 '\%(''\|rem\)\s*\$lang\>\|' ..
91 # default datatype
92 'def\%(byte\|longint\|short\|ubyte\|uint\|ulongint\|ushort\)\>' ..
93 '\)'
Bram Moolenaara2baa732022-02-04 16:09:54 +000094 var fb_comment = "^\\s*/'"
Bram Moolenaar8b5901e2022-06-29 14:39:12 +010095
Bram Moolenaara2baa732022-02-04 16:09:54 +000096 # OPTION EXPLICIT, without the leading underscore, is common to many dialects
97 var qb64_preproc = '\c^\s*\%($\a\+\|option\s\+\%(_explicit\|_\=explicitarray\)\>\)'
Bram Moolenaar6517f142022-01-21 14:55:13 +000098
Bram Moolenaar8b5901e2022-06-29 14:39:12 +010099 for lnum in range(1, min([line("$"), 100]))
100 var line = getline(lnum)
101 if line =~ ft_visual_basic_content
102 setf vb
103 return
104 elseif line =~ fb_preproc || line =~ fb_comment || line =~ fb_keywords
105 setf freebasic
106 return
107 elseif line =~ qb64_preproc
108 setf qb64
109 return
110 endif
111 endfor
112 setf basic
Bram Moolenaara2baa732022-02-04 16:09:54 +0000113enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100114
Bram Moolenaara2baa732022-02-04 16:09:54 +0000115export def FTbtm()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100116 if exists("g:dosbatch_syntax_for_btm") && g:dosbatch_syntax_for_btm
117 setf dosbatch
118 else
119 setf btm
120 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000121enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100122
Bram Moolenaara2baa732022-02-04 16:09:54 +0000123export def BindzoneCheck(default = '')
124 if getline(1) .. getline(2) .. getline(3) .. getline(4)
125 =~ '^; <<>> DiG [0-9.]\+.* <<>>\|$ORIGIN\|$TTL\|IN\s\+SOA'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100126 setf bindzone
Bram Moolenaara2baa732022-02-04 16:09:54 +0000127 elseif default != ''
128 exe 'setf ' .. default
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100129 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000130enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100131
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100132# Returns true if file content looks like RAPID
133def IsRapid(sChkExt: string = ""): bool
134 if sChkExt == "cfg"
135 return getline(1) =~? '\v^%(EIO|MMC|MOC|PROC|SIO|SYS):CFG'
136 endif
137 # called from FTmod, FTprg or FTsys
138 return getline(nextnonblank(1)) =~? '\v^\s*%(\%{3}|module\s+\k+\s*%(\(|$))'
139enddef
140
141export def FTcfg()
142 if exists("g:filetype_cfg")
143 exe "setf " .. g:filetype_cfg
144 elseif IsRapid("cfg")
145 setf rapid
146 else
147 setf cfg
148 endif
149enddef
150
Wu, Zhenyue2c27ca2024-11-19 20:55:25 +0100151export def FTcl()
152 if join(getline(1, 4), '') =~ '/\*'
153 setf opencl
154 else
155 setf lisp
156 endif
157enddef
158
Bram Moolenaar8b5901e2022-06-29 14:39:12 +0100159export def FTcls()
160 if exists("g:filetype_cls")
161 exe "setf " .. g:filetype_cls
162 return
163 endif
164
Doug Kearnsf97f6bb2023-08-27 18:44:09 +0200165 var line1 = getline(1)
Doug Kearnse06afb72023-08-29 22:21:35 +0200166 if line1 =~ '^#!.*\<\%(rexx\|regina\)\>'
Bram Moolenaar8b5901e2022-06-29 14:39:12 +0100167 setf rexx
Doug Kearnse06afb72023-08-29 22:21:35 +0200168 return
Doug Kearnsf97f6bb2023-08-27 18:44:09 +0200169 elseif line1 == 'VERSION 1.0 CLASS'
Bram Moolenaar8b5901e2022-06-29 14:39:12 +0100170 setf vb
Doug Kearnse06afb72023-08-29 22:21:35 +0200171 return
172 endif
173
174 var nonblank1 = getline(nextnonblank(1))
175 if nonblank1 =~ '^\v%(\%|\\)'
176 setf tex
177 elseif nonblank1 =~ '^\s*\%(/\*\|::\w\)'
178 setf rexx
Bram Moolenaar8b5901e2022-06-29 14:39:12 +0100179 else
180 setf st
181 endif
182enddef
183
Wu, Zhenyubc32bbd2024-11-14 22:55:36 +0100184export def FTll()
185 if getline(1) =~ ';\|\<source_filename\>\|\<target\>'
186 setf llvm
187 else
188 setf lifelines
189 endif
190enddef
191
Bram Moolenaara2baa732022-02-04 16:09:54 +0000192export def FTlpc()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100193 if exists("g:lpc_syntax_for_c")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000194 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100195 while lnum <= 12
196 if getline(lnum) =~# '^\(//\|inherit\|private\|protected\|nosave\|string\|object\|mapping\|mixed\)'
197 setf lpc
198 return
199 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000200 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100201 endwhile
202 endif
203 setf c
Bram Moolenaara2baa732022-02-04 16:09:54 +0000204enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100205
Bram Moolenaara2baa732022-02-04 16:09:54 +0000206export def FTheader()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100207 if match(getline(1, min([line("$"), 200])), '^@\(interface\|end\|class\)') > -1
208 if exists("g:c_syntax_for_h")
209 setf objc
210 else
211 setf objcpp
212 endif
213 elseif exists("g:c_syntax_for_h")
214 setf c
215 elseif exists("g:ch_syntax_for_h")
216 setf ch
217 else
218 setf cpp
219 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000220enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100221
Bram Moolenaara2baa732022-02-04 16:09:54 +0000222# This function checks if one of the first ten lines start with a '@'. In
223# that case it is probably a change file.
224# If the first line starts with # or ! it's probably a ch file.
225# If a line has "main", "include", "//" or "/*" it's probably ch.
226# Otherwise CHILL is assumed.
227export def FTchange()
228 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100229 while lnum <= 10
230 if getline(lnum)[0] == '@'
231 setf change
232 return
233 endif
234 if lnum == 1 && (getline(1)[0] == '#' || getline(1)[0] == '!')
235 setf ch
236 return
237 endif
238 if getline(lnum) =~ "MODULE"
239 setf chill
240 return
241 endif
242 if getline(lnum) =~ 'main\s*(\|#\s*include\|//'
243 setf ch
244 return
245 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000246 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100247 endwhile
248 setf chill
Bram Moolenaara2baa732022-02-04 16:09:54 +0000249enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100250
Bram Moolenaara2baa732022-02-04 16:09:54 +0000251export def FTent()
252 # This function checks for valid cl syntax in the first five lines.
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100253 # Look for either an opening comment, '#', or a block start, '{'.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000254 # If not found, assume SGML.
255 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100256 while lnum < 6
Bram Moolenaara2baa732022-02-04 16:09:54 +0000257 var line = getline(lnum)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100258 if line =~ '^\s*[#{]'
259 setf cl
260 return
261 elseif line !~ '^\s*$'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000262 # Not a blank line, not a comment, and not a block start,
263 # so doesn't look like valid cl code.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100264 break
265 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000266 lnum += 1
Bram Moolenaarc12dc472022-03-05 13:45:56 +0000267 endwhile
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100268 setf dtd
Bram Moolenaara2baa732022-02-04 16:09:54 +0000269enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100270
Bram Moolenaara2baa732022-02-04 16:09:54 +0000271export def ExCheck()
272 var lines = getline(1, min([line("$"), 100]))
Austin Gatlinf3caeb62021-06-26 12:02:55 +0200273 if exists('g:filetype_euphoria')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000274 exe 'setf ' .. g:filetype_euphoria
Austin Gatlinf3caeb62021-06-26 12:02:55 +0200275 elseif match(lines, '^--\|^ifdef\>\|^include\>') > -1
276 setf euphoria3
277 else
278 setf elixir
279 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000280enddef
Austin Gatlinf3caeb62021-06-26 12:02:55 +0200281
Bram Moolenaara2baa732022-02-04 16:09:54 +0000282export def EuphoriaCheck()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100283 if exists('g:filetype_euphoria')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000284 exe 'setf ' .. g:filetype_euphoria
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100285 else
286 setf euphoria3
287 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000288enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100289
Bram Moolenaara2baa732022-02-04 16:09:54 +0000290export def DtraceCheck()
=?UTF-8?q?Teubel=20Gy=C3=B6rgy?=4d56b972022-02-24 17:59:09 +0000291 if did_filetype()
292 # Filetype was already detected
293 return
294 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000295 var lines = getline(1, min([line("$"), 100]))
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100296 if match(lines, '^module\>\|^import\>') > -1
Bram Moolenaara2baa732022-02-04 16:09:54 +0000297 # D files often start with a module and/or import statement.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100298 setf d
299 elseif match(lines, '^#!\S\+dtrace\|#pragma\s\+D\s\+option\|:\S\{-}:\S\{-}:') > -1
300 setf dtrace
301 else
302 setf d
303 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000304enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100305
Doug Kearns68a89472024-01-05 17:59:04 +0100306export def FTdef()
Wu, Zhenyu61ee8332024-04-09 22:09:30 +0200307 # LaTeX def files are usually generated by docstrip, which will output '%%' in first line
308 if getline(1) =~ '%%'
309 setf tex
310 endif
Doug Kearns68a89472024-01-05 17:59:04 +0100311 if get(g:, "filetype_def", "") == "modula2" || IsModula2()
312 SetFiletypeModula2()
313 return
314 endif
315
316 if exists("g:filetype_def")
317 exe "setf " .. g:filetype_def
318 else
319 setf def
320 endif
321enddef
322
Bram Moolenaara2baa732022-02-04 16:09:54 +0000323export def FTe()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100324 if exists('g:filetype_euphoria')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000325 exe 'setf ' .. g:filetype_euphoria
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100326 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000327 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100328 while n < 100 && n <= line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100329 if getline(n) =~ "^\\s*\\(<'\\|'>\\)\\s*$"
330 setf specman
331 return
332 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000333 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100334 endwhile
335 setf eiffel
336 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000337enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100338
Doug Kearns19a3bc32023-08-20 20:51:12 +0200339def IsForth(): bool
340 var first_line = nextnonblank(1)
341
342 # SwiftForth block comment (line is usually filled with '-' or '=') or
343 # OPTIONAL (sometimes precedes the header comment)
344 if getline(first_line) =~? '^\%({\%(\s\|$\)\|OPTIONAL\s\)'
345 return true
346 endif
347
348 var n = first_line
349 while n < 100 && n <= line("$")
350 # Forth comments and colon definitions
351 if getline(n) =~ '^[:(\\] '
352 return true
353 endif
354 n += 1
355 endwhile
356 return false
357enddef
358
359# Distinguish between Forth and Fortran
360export def FTf()
361 if exists("g:filetype_f")
362 exe "setf " .. g:filetype_f
363 elseif IsForth()
364 setf forth
365 else
366 setf fortran
367 endif
368enddef
369
Bram Moolenaara2baa732022-02-04 16:09:54 +0000370export def FTfrm()
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000371 if exists("g:filetype_frm")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000372 exe "setf " .. g:filetype_frm
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000373 return
374 endif
375
Doug Kearnsf97f6bb2023-08-27 18:44:09 +0200376 if getline(1) == "VERSION 5.00"
377 setf vb
378 return
379 endif
380
Bram Moolenaara2baa732022-02-04 16:09:54 +0000381 var lines = getline(1, min([line("$"), 5]))
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000382
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000383 if match(lines, ft_visual_basic_content) > -1
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000384 setf vb
385 else
386 setf form
387 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000388enddef
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000389
Doug Kearns19a3bc32023-08-20 20:51:12 +0200390# Distinguish between Forth and F#
Bram Moolenaara2baa732022-02-04 16:09:54 +0000391export def FTfs()
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000392 if exists("g:filetype_fs")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000393 exe "setf " .. g:filetype_fs
Doug Kearns19a3bc32023-08-20 20:51:12 +0200394 elseif IsForth()
395 setf forth
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000396 else
Johan Kotlinski065088d2023-04-02 20:29:38 +0100397 setf fsharp
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000398 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000399enddef
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000400
Amelia Clarke35dfe582024-05-24 08:05:00 +0200401# Recursively search for Hare source files in a directory and any
402# subdirectories, up to a given depth.
403def IsHareModule(dir: string, depth: number): bool
404 if depth <= 0
405 return !empty(glob(dir .. '/*.ha'))
406 endif
407
408 return reduce(sort(glob(dir .. '/*', true, true),
409 (a, b) => isdirectory(a) - isdirectory(b)),
410 (acc, n) => acc
411 || n =~ '\.ha$'
412 || isdirectory(n)
413 && IsHareModule(n, depth - 1),
414 false)
415enddef
416
417# Determine if a README file exists within a Hare module and should be given the
418# Haredoc filetype.
419export def FTharedoc()
420 if exists('g:filetype_haredoc')
421 if IsHareModule('<afile>:h', get(g:, 'haredoc_search_depth', 1))
422 setf haredoc
423 endif
424 endif
425enddef
426
Dennis van den Berg1ad194c2024-07-09 19:25:33 +0200427# Distinguish between HTML, XHTML, Django and Angular
Bram Moolenaara2baa732022-02-04 16:09:54 +0000428export def FThtml()
429 var n = 1
Dennis van den Berg1ad194c2024-07-09 19:25:33 +0200430
431 # Test if the filename follows the Angular component template convention
Christian Brabandtc03f6312024-07-10 19:23:39 +0200432 # Disabled for the reasons mentioned here: #13594
433 # if expand('%:t') =~ '^.*\.component\.html$'
434 # setf htmlangular
435 # return
436 # endif
Dennis van den Berg1ad194c2024-07-09 19:25:33 +0200437
Afiq Nazrie9718ed72024-06-18 19:59:53 +0200438 while n < 40 && n <= line("$")
Dennis van den Berg1ad194c2024-07-09 19:25:33 +0200439 # Check for Angular
Christian Brabandt668e9f22025-01-11 09:19:12 +0100440 if getline(n) =~ '@\(if\|for\|defer\|switch\)\|\*\(ngIf\|ngFor\|ngSwitch\|ngTemplateOutlet\)\|ng-template\|ng-content'
Dennis van den Berg1ad194c2024-07-09 19:25:33 +0200441 setf htmlangular
442 return
443 endif
Dennis van den Berg1ad194c2024-07-09 19:25:33 +0200444 # Check for XHTML
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100445 if getline(n) =~ '\<DTD\s\+XHTML\s'
446 setf xhtml
447 return
448 endif
Dennis van den Berg1ad194c2024-07-09 19:25:33 +0200449 # Check for Django
Afiq Nazrie9718ed72024-06-18 19:59:53 +0200450 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 +0100451 setf htmldjango
452 return
453 endif
EliSaudere57c9a12024-07-28 21:28:11 +0200454 # Check for SuperHTML
455 if getline(n) =~ '<extend\|<super>'
456 setf superhtml
457 return
458 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000459 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100460 endwhile
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100461 setf FALLBACK html
Bram Moolenaara2baa732022-02-04 16:09:54 +0000462enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100463
Bram Moolenaara2baa732022-02-04 16:09:54 +0000464# Distinguish between standard IDL and MS-IDL
465export def FTidl()
466 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100467 while n < 50 && n <= line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100468 if getline(n) =~ '^\s*import\s\+"\(unknwn\|objidl\)\.idl"'
469 setf msidl
470 return
471 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000472 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100473 endwhile
474 setf idl
Bram Moolenaara2baa732022-02-04 16:09:54 +0000475enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100476
Wu, Zhenyu887a38c2024-05-09 20:35:13 +0200477# Distinguish between "default", Prolog, zsh module's C and Cproto prototype file.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000478export def ProtoCheck(default: string)
Wu, Zhenyu887a38c2024-05-09 20:35:13 +0200479 # zsh modules use '#include "*.pro"'
480 # https://github.com/zsh-users/zsh/blob/63f086d167960a27ecdbcb762179e2c2bf8a29f5/Src/Modules/example.c#L31
481 if getline(1) =~ '/* Generated automatically */'
482 setf c
Bram Moolenaara2baa732022-02-04 16:09:54 +0000483 # Cproto files have a comment in the first line and a function prototype in
484 # the second line, it always ends in ";". Indent files may also have
485 # comments, thus we can't match comments to see the difference.
486 # IDL files can have a single ';' in the second line, require at least one
487 # chacter before the ';'.
Wu, Zhenyu887a38c2024-05-09 20:35:13 +0200488 elseif getline(2) =~ '.;$'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100489 setf cpp
490 else
Bram Moolenaar48c3f4e2022-08-08 15:42:38 +0100491 # recognize Prolog by specific text in the first non-empty line
492 # require a blank after the '%' because Perl uses "%list" and "%translate"
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100493 var lnum = getline(nextnonblank(1))
igna_martinoli37853b72024-07-18 21:34:36 +0200494 if lnum =~ '\<prolog\>' || lnum =~ prolog_pattern
Bram Moolenaar48c3f4e2022-08-08 15:42:38 +0100495 setf prolog
496 else
497 exe 'setf ' .. default
498 endif
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100499 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000500enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100501
Bram Moolenaara2baa732022-02-04 16:09:54 +0000502export def FTm()
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200503 if exists("g:filetype_m")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000504 exe "setf " .. g:filetype_m
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200505 return
506 endif
507
Bram Moolenaara2baa732022-02-04 16:09:54 +0000508 # excluding end(for|function|if|switch|while) common to Murphi
509 var octave_block_terminators = '\<end\%(_try_catch\|classdef\|enumeration\|events\|methods\|parfor\|properties\)\>'
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200510
Bram Moolenaara2baa732022-02-04 16:09:54 +0000511 var objc_preprocessor = '^\s*#\s*\%(import\|include\|define\|if\|ifn\=def\|undef\|line\|error\|pragma\)\>'
Doug Kearns7329cfa2021-11-26 13:01:41 +0000512
Bram Moolenaara2baa732022-02-04 16:09:54 +0000513 var n = 1
514 var saw_comment = 0 # Whether we've seen a multiline comment leader.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100515 while n < 100
Bram Moolenaara2baa732022-02-04 16:09:54 +0000516 var line = getline(n)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100517 if line =~ '^\s*/\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000518 # /* ... */ is a comment in Objective C and Murphi, so we can't conclude
519 # it's either of them yet, but track this as a hint in case we don't see
520 # anything more definitive.
521 saw_comment = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100522 endif
Doug Kearns7329cfa2021-11-26 13:01:41 +0000523 if line =~ '^\s*//' || line =~ '^\s*@import\>' || line =~ objc_preprocessor
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100524 setf objc
525 return
526 endif
Bram Moolenaarca0627d2021-09-12 17:03:08 +0200527 if line =~ '^\s*\%(#\|%!\)' || line =~ '^\s*unwind_protect\>' ||
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200528 \ line =~ '\%(^\|;\)\s*' .. octave_block_terminators
529 setf octave
530 return
531 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000532 # TODO: could be Matlab or Octave
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100533 if line =~ '^\s*%'
534 setf matlab
535 return
536 endif
537 if line =~ '^\s*(\*'
538 setf mma
539 return
540 endif
541 if line =~ '^\c\s*\(\(type\|var\)\>\|--\)'
542 setf murphi
543 return
544 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000545 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100546 endwhile
547
548 if saw_comment
Bram Moolenaara2baa732022-02-04 16:09:54 +0000549 # We didn't see anything definitive, but this looks like either Objective C
550 # or Murphi based on the comment leader. Assume the former as it is more
551 # common.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100552 setf objc
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100553 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000554 # Default is Matlab
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100555 setf matlab
556 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000557enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100558
Ken Takataeb4b9032024-07-25 21:07:13 +0200559export def FTmake()
560 # Check if it is a Microsoft Makefile
561 unlet! b:make_microsoft
562 var n = 1
563 while n < 1000 && n <= line('$')
564 var line = getline(n)
565 if line =~? '^\s*!\s*\(ifn\=\(def\)\=\|include\|message\|error\)\>'
566 b:make_microsoft = 1
567 break
568 elseif line =~ '^ *ifn\=\(eq\|def\)\>' || line =~ '^ *[-s]\=include\s'
569 break
570 elseif line =~ '^ *\w\+\s*[!?:+]='
571 break
572 endif
573 n += 1
574 endwhile
575 setf make
576enddef
577
Bram Moolenaara2baa732022-02-04 16:09:54 +0000578export def FTmms()
579 var n = 1
Bram Moolenaard7df2792020-01-02 21:34:42 +0100580 while n < 20
Bram Moolenaara2baa732022-02-04 16:09:54 +0000581 var line = getline(n)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100582 if line =~ '^\s*\(%\|//\)' || line =~ '^\*'
583 setf mmix
584 return
585 endif
586 if line =~ '^\s*#'
587 setf make
588 return
589 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000590 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100591 endwhile
592 setf mmix
Bram Moolenaara2baa732022-02-04 16:09:54 +0000593enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100594
Bram Moolenaara2baa732022-02-04 16:09:54 +0000595# This function checks if one of the first five lines start with a dot. In
596# that case it is probably an nroff file: 'filetype' is set and 1 is returned.
597export def FTnroff(): number
598 if getline(1)[0] .. getline(2)[0] .. getline(3)[0]
599 .. getline(4)[0] .. getline(5)[0] =~ '\.'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100600 setf nroff
601 return 1
602 endif
603 return 0
Bram Moolenaara2baa732022-02-04 16:09:54 +0000604enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100605
Bram Moolenaara2baa732022-02-04 16:09:54 +0000606export def FTmm()
607 var n = 1
Bram Moolenaard1caa942020-04-10 22:10:56 +0200608 while n < 20
Bram Moolenaara2baa732022-02-04 16:09:54 +0000609 if getline(n) =~ '^\s*\(#\s*\(include\|import\)\>\|@import\>\|/\*\)'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100610 setf objcpp
611 return
612 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000613 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100614 endwhile
615 setf nroff
Bram Moolenaara2baa732022-02-04 16:09:54 +0000616enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100617
Bram Moolenaarcdbfc6d2022-06-30 16:25:21 +0100618# Returns true if file content looks like LambdaProlog module
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100619def IsLProlog(): bool
Gaetan Lepage4ce1bda2023-05-10 22:01:55 +0100620 # skip apparent comments and blank lines, what looks like
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100621 # LambdaProlog comment may be RAPID header
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100622 var lnum: number = nextnonblank(1)
623 while lnum > 0 && lnum < line('$') && getline(lnum) =~ '^\s*%' # LambdaProlog comment
624 lnum = nextnonblank(lnum + 1)
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100625 endwhile
626 # this pattern must not catch a go.mod file
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100627 return getline(lnum) =~ '\<module\s\+\w\+\s*\.\s*\(%\|$\)'
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100628enddef
629
Doug Kearns68a89472024-01-05 17:59:04 +0100630def IsModula2(): bool
dkearnsef387c02024-02-20 06:58:30 +1100631 return getline(nextnonblank(1)) =~ '\<MODULE\s\+\w\+\s*\%(\[.*]\s*\)\=;\|^\s*(\*'
Doug Kearns68a89472024-01-05 17:59:04 +0100632enddef
633
634def SetFiletypeModula2()
635 const KNOWN_DIALECTS = ["iso", "pim", "r10"]
636 const KNOWN_EXTENSIONS = ["gm2"]
637 const LINE_COUNT = 200
638 const TAG = '(\*!m2\(\w\+\)\%(+\(\w\+\)\)\=\*)'
639
640 var dialect = get(g:, "modula2_default_dialect", "pim")
641 var extension = get(g:, "modula2_default_extension", "")
642
643 var matches = []
644
645 # ignore unknown dialects or badly formatted tags
646 for lnum in range(1, min([line("$"), LINE_COUNT]))
647 matches = matchlist(getline(lnum), TAG)
648 if !empty(matches)
649 if index(KNOWN_DIALECTS, matches[1]) >= 0
650 dialect = matches[1]
651 endif
652 if index(KNOWN_EXTENSIONS, matches[2]) >= 0
653 extension = matches[2]
654 endif
655 break
656 endif
657 endfor
658
659 modula2#SetDialect(dialect, extension)
660
661 setf modula2
662enddef
663
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100664# Determine if *.mod is ABB RAPID, LambdaProlog, Modula-2, Modsim III or go.mod
665export def FTmod()
Doug Kearns68a89472024-01-05 17:59:04 +0100666 if get(g:, "filetype_mod", "") == "modula2" || IsModula2()
667 SetFiletypeModula2()
668 return
669 endif
670
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100671 if exists("g:filetype_mod")
672 exe "setf " .. g:filetype_mod
Omar El Halabic9fbd252023-05-29 19:59:45 +0100673 elseif expand("<afile>") =~ '\<go.mod$'
674 setf gomod
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100675 elseif IsLProlog()
676 setf lprolog
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100677 elseif IsRapid()
678 setf rapid
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100679 else
680 # Nothing recognized, assume modsim3
681 setf modsim3
682 endif
683enddef
684
Bram Moolenaara2baa732022-02-04 16:09:54 +0000685export def FTpl()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100686 if exists("g:filetype_pl")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000687 exe "setf " .. g:filetype_pl
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100688 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000689 # recognize Prolog by specific text in the first non-empty line
690 # require a blank after the '%' because Perl uses "%list" and "%translate"
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100691 var line = getline(nextnonblank(1))
igna_martinoli37853b72024-07-18 21:34:36 +0200692 if line =~ '\<prolog\>' || line =~ prolog_pattern
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100693 setf prolog
694 else
695 setf perl
696 endif
697 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000698enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100699
Bram Moolenaara2baa732022-02-04 16:09:54 +0000700export def FTinc()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100701 if exists("g:filetype_inc")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000702 exe "setf " .. g:filetype_inc
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100703 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000704 var lines = getline(1) .. getline(2) .. getline(3)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100705 if lines =~? "perlscript"
706 setf aspperl
707 elseif lines =~ "<%"
708 setf aspvbs
709 elseif lines =~ "<?"
710 setf php
Bram Moolenaara2baa732022-02-04 16:09:54 +0000711 # Pascal supports // comments but they're vary rarely used for file
712 # headers so assume POV-Ray
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000713 elseif lines =~ '^\s*\%({\|(\*\)' || lines =~? ft_pascal_keywords
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100714 setf pascal
Gregory Anders30e212d2022-07-26 21:42:03 +0100715 elseif lines =~# '\<\%(require\|inherit\)\>' || lines =~# '[A-Z][A-Za-z0-9_:${}]*\s\+\%(??\|[?:+]\)\?= '
Gregory Andersfa49eb42022-07-16 17:46:47 +0100716 setf bitbake
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100717 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000718 FTasmsyntax()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100719 if exists("b:asmsyntax")
Gregory Andersfa49eb42022-07-16 17:46:47 +0100720 exe "setf " .. fnameescape(b:asmsyntax)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100721 else
Gregory Andersfa49eb42022-07-16 17:46:47 +0100722 setf pov
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100723 endif
724 endif
725 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000726enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100727
Bram Moolenaara2baa732022-02-04 16:09:54 +0000728export def FTprogress_cweb()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100729 if exists("g:filetype_w")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000730 exe "setf " .. g:filetype_w
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100731 return
732 endif
733 if getline(1) =~ '&ANALYZE' || getline(3) =~ '&GLOBAL-DEFINE'
734 setf progress
735 else
736 setf cweb
737 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000738enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100739
Julien Marrec2e310652023-11-25 15:30:46 +0100740# These include the leading '%' sign
741var 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\)'
742# This is the start/end of a block that is copied literally to the processor file (C/C++)
743var ft_swig_verbatim_block_start = '^\s*%{'
744
745export def FTi()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100746 if exists("g:filetype_i")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000747 exe "setf " .. g:filetype_i
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100748 return
749 endif
Julien Marrec2e310652023-11-25 15:30:46 +0100750 # 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 +0000751 # If not found, assume Progress.
752 var lnum = 1
Julien Marrec2e310652023-11-25 15:30:46 +0100753 while lnum <= 50 && lnum < line('$')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000754 var line = getline(lnum)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100755 if line =~ '^\s*;' || line =~ '^\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000756 FTasm()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100757 return
Julien Marrec2e310652023-11-25 15:30:46 +0100758 elseif line =~ ft_swig_keywords || line =~ ft_swig_verbatim_block_start
759 setf swig
760 return
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100761 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000762 lnum += 1
Bram Moolenaarc12dc472022-03-05 13:45:56 +0000763 endwhile
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100764 setf progress
Bram Moolenaara2baa732022-02-04 16:09:54 +0000765enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100766
Bram Moolenaara2baa732022-02-04 16:09:54 +0000767var ft_pascal_comments = '^\s*\%({\|(\*\|//\)'
768var ft_pascal_keywords = '^\s*\%(program\|unit\|library\|uses\|begin\|procedure\|function\|const\|type\|var\)\>'
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100769
Bram Moolenaara2baa732022-02-04 16:09:54 +0000770export def FTprogress_pascal()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100771 if exists("g:filetype_p")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000772 exe "setf " .. g:filetype_p
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100773 return
774 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000775 # This function checks for valid Pascal syntax in the first ten lines.
776 # Look for either an opening comment or a program start.
777 # If not found, assume Progress.
778 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100779 while lnum <= 10 && lnum < line('$')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000780 var line = getline(lnum)
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000781 if line =~ ft_pascal_comments || line =~? ft_pascal_keywords
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100782 setf pascal
783 return
784 elseif line !~ '^\s*$' || line =~ '^/\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000785 # Not an empty line: Doesn't look like valid Pascal code.
786 # Or it looks like a Progress /* comment
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100787 break
788 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000789 lnum += 1
Bram Moolenaarc12dc472022-03-05 13:45:56 +0000790 endwhile
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100791 setf progress
Bram Moolenaara2baa732022-02-04 16:09:54 +0000792enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100793
Bram Moolenaara2baa732022-02-04 16:09:54 +0000794export def FTpp()
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100795 if exists("g:filetype_pp")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000796 exe "setf " .. g:filetype_pp
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100797 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000798 var line = getline(nextnonblank(1))
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000799 if line =~ ft_pascal_comments || line =~? ft_pascal_keywords
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100800 setf pascal
801 else
802 setf puppet
803 endif
804 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000805enddef
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100806
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100807# Determine if *.prg is ABB RAPID. Can also be Clipper, FoxPro or eviews
808export def FTprg()
809 if exists("g:filetype_prg")
810 exe "setf " .. g:filetype_prg
811 elseif IsRapid()
812 setf rapid
813 else
814 # Nothing recognized, assume Clipper
815 setf clipper
816 endif
817enddef
818
Bram Moolenaara2baa732022-02-04 16:09:54 +0000819export def FTr()
820 var max = line("$") > 50 ? 50 : line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100821
822 for n in range(1, max)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000823 # Rebol is easy to recognize, check for that first
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100824 if getline(n) =~? '\<REBOL\>'
825 setf rebol
826 return
827 endif
828 endfor
829
830 for n in range(1, max)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000831 # R has # comments
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100832 if getline(n) =~ '^\s*#'
833 setf r
834 return
835 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000836 # Rexx has /* comments */
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100837 if getline(n) =~ '^\s*/\*'
838 setf rexx
839 return
840 endif
841 endfor
842
Bram Moolenaara2baa732022-02-04 16:09:54 +0000843 # Nothing recognized, use user default or assume Rexx
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100844 if exists("g:filetype_r")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000845 exe "setf " .. g:filetype_r
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100846 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000847 # Rexx used to be the default, but R appears to be much more popular.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100848 setf r
849 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000850enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100851
Bram Moolenaara2baa732022-02-04 16:09:54 +0000852export def McSetf()
853 # Rely on the file to start with a comment.
854 # MS message text files use ';', Sendmail files use '#' or 'dnl'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100855 for lnum in range(1, min([line("$"), 20]))
Bram Moolenaara2baa732022-02-04 16:09:54 +0000856 var line = getline(lnum)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100857 if line =~ '^\s*\(#\|dnl\)'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000858 setf m4 # Sendmail .mc file
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100859 return
860 elseif line =~ '^\s*;'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000861 setf msmessages # MS Message text file
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100862 return
863 endif
864 endfor
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100865 setf m4 # Default: Sendmail .mc file
Bram Moolenaara2baa732022-02-04 16:09:54 +0000866enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100867
Bram Moolenaara2baa732022-02-04 16:09:54 +0000868# Called from filetype.vim and scripts.vim.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100869# When "setft" is passed and false then the 'filetype' option is not set.
870export def SetFileTypeSH(name: string, setft = true): string
871 if setft && did_filetype()
Bram Moolenaara2baa732022-02-04 16:09:54 +0000872 # Filetype was already detected
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100873 return ''
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100874 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100875 if setft && expand("<amatch>") =~ g:ft_ignore_pat
876 return ''
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100877 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000878 if name =~ '\<csh\>'
879 # Some .sh scripts contain #!/bin/csh.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100880 return SetFileTypeShell("csh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000881 elseif name =~ '\<tcsh\>'
882 # Some .sh scripts contain #!/bin/tcsh.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100883 return SetFileTypeShell("tcsh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000884 elseif name =~ '\<zsh\>'
885 # Some .sh scripts contain #!/bin/zsh.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100886 return SetFileTypeShell("zsh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000887 elseif name =~ '\<ksh\>'
888 b:is_kornshell = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100889 if exists("b:is_bash")
890 unlet b:is_bash
891 endif
892 if exists("b:is_sh")
893 unlet b:is_sh
894 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000895 elseif exists("g:bash_is_sh") || name =~ '\<bash\>' || name =~ '\<bash2\>'
896 b:is_bash = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100897 if exists("b:is_kornshell")
898 unlet b:is_kornshell
899 endif
900 if exists("b:is_sh")
901 unlet b:is_sh
902 endif
Eisuke Kawashima24482fb2022-11-24 10:58:10 +0000903 elseif name =~ '\<sh\>' || name =~ '\<dash\>'
904 # Ubuntu links "sh" to "dash", thus it is expected to work the same way
Bram Moolenaara2baa732022-02-04 16:09:54 +0000905 b:is_sh = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100906 if exists("b:is_kornshell")
907 unlet b:is_kornshell
908 endif
909 if exists("b:is_bash")
910 unlet b:is_bash
911 endif
912 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100913
914 return SetFileTypeShell("sh", setft)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000915enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100916
Bram Moolenaara2baa732022-02-04 16:09:54 +0000917# For shell-like file types, check for an "exec" command hidden in a comment,
918# as used for Tcl.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100919# When "setft" is passed and false then the 'filetype' option is not set.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000920# Also called from scripts.vim, thus can't be local to this script.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100921export def SetFileTypeShell(name: string, setft = true): string
922 if setft && did_filetype()
Bram Moolenaara2baa732022-02-04 16:09:54 +0000923 # Filetype was already detected
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100924 return ''
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100925 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100926 if setft && expand("<amatch>") =~ g:ft_ignore_pat
927 return ''
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100928 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100929
930 var lnum = 2
931 while lnum < 20 && lnum < line("$") && getline(lnum) =~ '^\s*\(#\|$\)'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000932 # Skip empty and comment lines.
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100933 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100934 endwhile
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100935 if lnum < line("$") && getline(lnum) =~ '\s*exec\s' && getline(lnum - 1) =~ '^\s*#.*\\$'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000936 # Found an "exec" line after a comment with continuation
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100937 var n = substitute(getline(lnum), '\s*exec\s\+\([^ ]*/\)\=', '', '')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100938 if n =~ '\<tclsh\|\<wish'
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100939 if setft
940 setf tcl
941 endif
942 return 'tcl'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100943 endif
944 endif
Bram Moolenaarf07d1a72023-06-09 21:01:47 +0100945
946 if setft
947 exe "setf " .. name
948 endif
949 return name
Bram Moolenaara2baa732022-02-04 16:09:54 +0000950enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100951
Bram Moolenaara2baa732022-02-04 16:09:54 +0000952export def CSH()
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100953 if did_filetype()
Bram Moolenaara2baa732022-02-04 16:09:54 +0000954 # Filetype was already detected
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100955 return
956 endif
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100957 if exists("g:filetype_csh")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000958 SetFileTypeShell(g:filetype_csh)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100959 elseif &shell =~ "tcsh"
Bram Moolenaara2baa732022-02-04 16:09:54 +0000960 SetFileTypeShell("tcsh")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100961 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000962 SetFileTypeShell("csh")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100963 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000964enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100965
Bram Moolenaara2baa732022-02-04 16:09:54 +0000966var ft_rules_udev_rules_pattern = '^\s*\cudev_rules\s*=\s*"\([^"]\{-1,}\)/*".*'
967export def FTRules()
968 var path = expand('<amatch>:p')
Bram Moolenaaraa9675a2020-08-17 21:57:09 +0200969 if path =~ '/\(etc/udev/\%(rules\.d/\)\=.*\.rules\|\%(usr/\)\=lib/udev/\%(rules\.d/\)\=.*\.rules\)$'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100970 setf udevrules
971 return
972 endif
973 if path =~ '^/etc/ufw/'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000974 setf conf # Better than hog
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100975 return
976 endif
977 if path =~ '^/\(etc\|usr/share\)/polkit-1/rules\.d'
978 setf javascript
979 return
980 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000981 var config_lines: list<string>
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100982 try
Bram Moolenaara2baa732022-02-04 16:09:54 +0000983 config_lines = readfile('/etc/udev/udev.conf')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100984 catch /^Vim\%((\a\+)\)\=:E484/
985 setf hog
986 return
987 endtry
Bram Moolenaara2baa732022-02-04 16:09:54 +0000988 var dir = expand('<amatch>:p:h')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100989 for line in config_lines
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000990 if line =~ ft_rules_udev_rules_pattern
991 var udev_rules = substitute(line, ft_rules_udev_rules_pattern, '\1', "")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100992 if dir == udev_rules
993 setf udevrules
994 endif
995 break
996 endif
997 endfor
998 setf hog
Bram Moolenaara2baa732022-02-04 16:09:54 +0000999enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001000
Bram Moolenaara2baa732022-02-04 16:09:54 +00001001export def SQL()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001002 if exists("g:filetype_sql")
Bram Moolenaara2baa732022-02-04 16:09:54 +00001003 exe "setf " .. g:filetype_sql
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001004 else
1005 setf sql
1006 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001007enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001008
Wu, Zhenyu4f73c072025-01-08 20:20:06 +01001009export def FTsa()
1010 if join(getline(1, 4), "\n") =~# '\%(^\|\n\);'
1011 setf tiasm
1012 return
1013 endif
1014 setf sather
1015enddef
1016
ranjithshegde8cac20e2022-04-13 15:29:21 +01001017# This function checks the first 25 lines of file extension "sc" to resolve
Chris Kipp70ef3f52022-12-14 16:42:15 +00001018# detection between scala and SuperCollider.
1019# NOTE: We don't check for 'Class : Method', as this can easily be confused
1020# with valid Scala like `val x : Int = 3`. So we instead only rely on
1021# checks that can't be confused.
ranjithshegde8cac20e2022-04-13 15:29:21 +01001022export def FTsc()
1023 for lnum in range(1, min([line("$"), 25]))
Chris Kipp70ef3f52022-12-14 16:42:15 +00001024 if getline(lnum) =~# 'var\s<\|classvar\s<\|\^this.*\||\w\+|\|+\s\w*\s{\|\*ar\s'
ranjithshegde8cac20e2022-04-13 15:29:21 +01001025 setf supercollider
1026 return
1027 endif
1028 endfor
1029 setf scala
1030enddef
1031
1032# This function checks the first line of file extension "scd" to resolve
1033# detection between scdoc and SuperCollider
1034export def FTscd()
1035 if getline(1) =~# '\%^\S\+(\d[0-9A-Za-z]*)\%(\s\+\"[^"]*\"\%(\s\+\"[^"]*\"\)\=\)\=$'
1036 setf scdoc
1037 else
1038 setf supercollider
1039 endif
1040enddef
1041
Bram Moolenaara2baa732022-02-04 16:09:54 +00001042# If the file has an extension of 't' and is in a directory 't' or 'xt' then
1043# it is almost certainly a Perl test file.
1044# If the first line starts with '#' and contains 'perl' it's probably a Perl
1045# file.
1046# (Slow test) If a file contains a 'use' statement then it is almost certainly
1047# a Perl file.
1048export def FTperl(): number
1049 var dirname = expand("%:p:h:t")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001050 if expand("%:e") == 't' && (dirname == 't' || dirname == 'xt')
1051 setf perl
1052 return 1
1053 endif
1054 if getline(1)[0] == '#' && getline(1) =~ 'perl'
1055 setf perl
1056 return 1
1057 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001058 var save_cursor = getpos('.')
1059 call cursor(1, 1)
Bram Moolenaare5b78972022-02-05 19:50:34 +00001060 var has_use = search('^use\s\s*\k', 'c', 30) > 0
Bram Moolenaarf0b03c42017-12-17 17:17:07 +01001061 call setpos('.', save_cursor)
1062 if has_use
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001063 setf perl
1064 return 1
1065 endif
1066 return 0
Bram Moolenaara2baa732022-02-04 16:09:54 +00001067enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001068
Bram Moolenaarcdbfc6d2022-06-30 16:25:21 +01001069# LambdaProlog and Standard ML signature files
1070export def FTsig()
1071 if exists("g:filetype_sig")
1072 exe "setf " .. g:filetype_sig
1073 return
1074 endif
1075
1076 var lprolog_comment = '^\s*\%(/\*\|%\)'
1077 var lprolog_keyword = '^\s*sig\s\+\a'
1078 var sml_comment = '^\s*(\*'
1079 var sml_keyword = '^\s*\%(signature\|structure\)\s\+\a'
1080
1081 var line = getline(nextnonblank(1))
1082
1083 if line =~ lprolog_comment || line =~# lprolog_keyword
1084 setf lprolog
1085 elseif line =~ sml_comment || line =~# sml_keyword
1086 setf sml
1087 endif
1088enddef
1089
Bram Moolenaarbe807d52022-09-01 15:01:25 +01001090# This function checks the first 100 lines of files matching "*.sil" to
1091# resolve detection between Swift Intermediate Language and SILE.
1092export def FTsil()
1093 for lnum in range(1, [line('$'), 100]->min())
1094 var line: string = getline(lnum)
1095 if line =~ '^\s*[\\%]'
1096 setf sile
1097 return
1098 elseif line =~ '^\s*\S'
1099 setf sil
1100 return
1101 endif
1102 endfor
1103 # no clue, default to "sil"
1104 setf sil
1105enddef
1106
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +01001107export def FTsys()
KnoP-01f420ff22022-04-13 20:46:21 +01001108 if exists("g:filetype_sys")
1109 exe "setf " .. g:filetype_sys
1110 elseif IsRapid()
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +01001111 setf rapid
1112 else
1113 setf bat
1114 endif
1115enddef
1116
Bram Moolenaara2baa732022-02-04 16:09:54 +00001117# Choose context, plaintex, or tex (LaTeX) based on these rules:
1118# 1. Check the first line of the file for "%&<format>".
1119# 2. Check the first 1000 non-comment lines for LaTeX or ConTeXt keywords.
1120# 3. Default to "plain" or to g:tex_flavor, can be set in user's vimrc.
1121export def FTtex()
1122 var firstline = getline(1)
1123 var format: string
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001124 if firstline =~ '^%&\s*\a\+'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001125 format = tolower(matchstr(firstline, '\a\+'))
1126 format = substitute(format, 'pdf', '', '')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001127 if format == 'tex'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001128 format = 'latex'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001129 elseif format == 'plaintex'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001130 format = 'plain'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001131 endif
1132 elseif expand('%') =~ 'tex/context/.*/.*.tex'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001133 format = 'context'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001134 else
Bram Moolenaara2baa732022-02-04 16:09:54 +00001135 # Default value, may be changed later:
1136 format = exists("g:tex_flavor") ? g:tex_flavor : 'plain'
1137 # Save position, go to the top of the file, find first non-comment line.
1138 var save_cursor = getpos('.')
1139 call cursor(1, 1)
1140 var firstNC = search('^\s*[^[:space:]%]', 'c', 1000)
Bram Moolenaare5b78972022-02-05 19:50:34 +00001141 if firstNC > 0
1142 # Check the next thousand lines for a LaTeX or ConTeXt keyword.
Bram Moolenaara2baa732022-02-04 16:09:54 +00001143 var lpat = 'documentclass\>\|usepackage\>\|begin{\|newcommand\>\|renewcommand\>'
1144 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\>'
1145 var kwline = search('^\s*\\\%(' .. lpat .. '\)\|^\s*\\\(' .. cpat .. '\)',
1146 'cnp', firstNC + 1000)
1147 if kwline == 1 # lpat matched
1148 format = 'latex'
1149 elseif kwline == 2 # cpat matched
1150 format = 'context'
1151 endif # If neither matched, keep default set above.
1152 # let lline = search('^\s*\\\%(' . lpat . '\)', 'cn', firstNC + 1000)
1153 # let cline = search('^\s*\\\%(' . cpat . '\)', 'cn', firstNC + 1000)
1154 # if cline > 0
1155 # let format = 'context'
1156 # endif
1157 # if lline > 0 && (cline == 0 || cline > lline)
1158 # let format = 'tex'
1159 # endif
1160 endif # firstNC
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001161 call setpos('.', save_cursor)
Bram Moolenaara2baa732022-02-04 16:09:54 +00001162 endif # firstline =~ '^%&\s*\a\+'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001163
Bram Moolenaara2baa732022-02-04 16:09:54 +00001164 # Translation from formats to file types. TODO: add AMSTeX, RevTex, others?
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001165 if format == 'plain'
1166 setf plaintex
1167 elseif format == 'context'
1168 setf context
Bram Moolenaara2baa732022-02-04 16:09:54 +00001169 else # probably LaTeX
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001170 setf tex
1171 endif
1172 return
Bram Moolenaara2baa732022-02-04 16:09:54 +00001173enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001174
Bram Moolenaara2baa732022-02-04 16:09:54 +00001175export def FTxml()
1176 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +01001177 while n < 100 && n <= line("$")
Bram Moolenaara2baa732022-02-04 16:09:54 +00001178 var line = getline(n)
1179 # DocBook 4 or DocBook 5.
1180 var is_docbook4 = line =~ '<!DOCTYPE.*DocBook'
1181 var is_docbook5 = line =~ ' xmlns="http://docbook.org/ns/docbook"'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001182 if is_docbook4 || is_docbook5
Bram Moolenaara2baa732022-02-04 16:09:54 +00001183 b:docbk_type = "xml"
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001184 if is_docbook5
Bram Moolenaara2baa732022-02-04 16:09:54 +00001185 b:docbk_ver = 5
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001186 else
Bram Moolenaara2baa732022-02-04 16:09:54 +00001187 b:docbk_ver = 4
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001188 endif
1189 setf docbk
1190 return
1191 endif
1192 if line =~ 'xmlns:xbl="http://www.mozilla.org/xbl"'
1193 setf xbl
1194 return
1195 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001196 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001197 endwhile
1198 setf xml
Bram Moolenaara2baa732022-02-04 16:09:54 +00001199enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001200
Bram Moolenaara2baa732022-02-04 16:09:54 +00001201export def FTy()
1202 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +01001203 while n < 100 && n <= line("$")
Bram Moolenaara2baa732022-02-04 16:09:54 +00001204 var line = getline(n)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001205 if line =~ '^\s*%'
1206 setf yacc
1207 return
1208 endif
1209 if getline(n) =~ '^\s*\(#\|class\>\)' && getline(n) !~ '^\s*#\s*include'
1210 setf racc
1211 return
1212 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001213 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001214 endwhile
1215 setf yacc
Bram Moolenaara2baa732022-02-04 16:09:54 +00001216enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001217
Bram Moolenaara2baa732022-02-04 16:09:54 +00001218export def Redif()
1219 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001220 while lnum <= 5 && lnum < line('$')
1221 if getline(lnum) =~ "^\ctemplate-type:"
1222 setf redif
1223 return
1224 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001225 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001226 endwhile
Bram Moolenaara2baa732022-02-04 16:09:54 +00001227enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001228
Bram Moolenaara2baa732022-02-04 16:09:54 +00001229# This function is called for all files under */debian/patches/*, make sure not
1230# to non-dep3patch files, such as README and other text files.
1231export def Dep3patch()
James McCoy647ab4c2021-12-17 20:52:57 +00001232 if expand('%:t') ==# 'series'
1233 return
1234 endif
1235
1236 for ln in getline(1, 100)
1237 if ln =~# '^\%(Description\|Subject\|Origin\|Bug\|Forwarded\|Author\|From\|Reviewed-by\|Acked-by\|Last-Updated\|Applied-Upstream\):'
1238 setf dep3patch
1239 return
1240 elseif ln =~# '^---'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001241 # end of headers found. stop processing
James McCoy647ab4c2021-12-17 20:52:57 +00001242 return
1243 endif
1244 endfor
Bram Moolenaara2baa732022-02-04 16:09:54 +00001245enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01001246
Bram Moolenaara2baa732022-02-04 16:09:54 +00001247# This function checks the first 15 lines for appearance of 'FoamFile'
1248# and then 'object' in a following line.
1249# In that case, it's probably an OpenFOAM file
1250export def FTfoam()
1251 var ffile = 0
1252 var lnum = 1
Elwardi2284f6c2022-01-11 18:14:23 +00001253 while lnum <= 15
1254 if getline(lnum) =~# '^FoamFile'
Bram Moolenaara2baa732022-02-04 16:09:54 +00001255 ffile = 1
Elwardi2284f6c2022-01-11 18:14:23 +00001256 elseif ffile == 1 && getline(lnum) =~# '^\s*object'
1257 setf foam
1258 return
1259 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +00001260 lnum += 1
Elwardi2284f6c2022-01-11 18:14:23 +00001261 endwhile
Bram Moolenaara2baa732022-02-04 16:09:54 +00001262enddef
Elwardi2284f6c2022-01-11 18:14:23 +00001263
Bram Moolenaara2baa732022-02-04 16:09:54 +00001264# Determine if a *.tf file is TF mud client or terraform
1265export def FTtf()
1266 var numberOfLines = line('$')
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001267 for i in range(1, numberOfLines)
Bram Moolenaara2baa732022-02-04 16:09:54 +00001268 var currentLine = trim(getline(i))
1269 var firstCharacter = currentLine[0]
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001270 if firstCharacter !=? ";" && firstCharacter !=? "/" && firstCharacter !=? ""
1271 setf terraform
1272 return
1273 endif
1274 endfor
1275 setf tf
Bram Moolenaara2baa732022-02-04 16:09:54 +00001276enddef
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001277
KnoP-0193c7a452022-04-16 21:14:04 +01001278var ft_krl_header = '\&\w+'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001279# Determine if a *.src file is Kuka Robot Language
1280export def FTsrc()
KnoP-0193c7a452022-04-16 21:14:04 +01001281 var ft_krl_def_or_deffct = '%(global\s+)?def%(fct)?>'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001282 if exists("g:filetype_src")
1283 exe "setf " .. g:filetype_src
KnoP-0193c7a452022-04-16 21:14:04 +01001284 elseif getline(nextnonblank(1)) =~? '\v^\s*%(' .. ft_krl_header .. '|' .. ft_krl_def_or_deffct .. ')'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001285 setf krl
1286 endif
1287enddef
1288
1289# Determine if a *.dat file is Kuka Robot Language
1290export def FTdat()
KnoP-0193c7a452022-04-16 21:14:04 +01001291 var ft_krl_defdat = 'defdat>'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001292 if exists("g:filetype_dat")
1293 exe "setf " .. g:filetype_dat
KnoP-0193c7a452022-04-16 21:14:04 +01001294 elseif getline(nextnonblank(1)) =~? '\v^\s*%(' .. ft_krl_header .. '|' .. ft_krl_defdat .. ')'
Bram Moolenaar3ad20902022-04-06 18:57:39 +01001295 setf krl
1296 endif
1297enddef
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +00001298
Doug Kearns4ac8e792022-10-17 13:32:17 +01001299export def FTlsl()
1300 if exists("g:filetype_lsl")
1301 exe "setf " .. g:filetype_lsl
1302 endif
1303
1304 var line = getline(nextnonblank(1))
1305 if line =~ '^\s*%' || line =~# ':\s*trait\s*$'
1306 setf larch
1307 else
1308 setf lsl
1309 endif
1310enddef
1311
Gaetan Lepage4ce1bda2023-05-10 22:01:55 +01001312export def FTtyp()
1313 if exists("g:filetype_typ")
1314 exe "setf " .. g:filetype_typ
1315 return
1316 endif
1317
1318 # Look for SQL type definition syntax
1319 for line in getline(1, 200)
1320 # SQL type files may define the casing
1321 if line =~ '^CASE\s\==\s\=\(SAME\|LOWER\|UPPER\|OPPOSITE\)$'
1322 setf sql
1323 return
1324 endif
1325
1326 # SQL type files may define some types as follows
1327 if line =~ '^TYPE\s.*$'
1328 setf sql
1329 return
1330 endif
1331 endfor
1332
1333 # Otherwise, affect the typst filetype
1334 setf typst
1335enddef
1336
PowerUser64aa61b8a2024-06-19 20:32:11 +02001337# Detect Microsoft Developer Studio Project files (Makefile) or Faust DSP
1338# files.
1339export def FTdsp()
1340 if exists("g:filetype_dsp")
1341 exe "setf " .. g:filetype_dsp
1342 return
1343 endif
1344
1345 # Test the filename
1346 if expand('%:t') =~ '^[mM]akefile.*$'
1347 setf make
1348 return
1349 endif
1350
1351 # Test the file contents
1352 for line in getline(1, 200)
1353 # Chech for comment style
1354 if line =~ '^#.*'
1355 setf make
1356 return
1357 endif
1358
1359 # Check for common lines
1360 if line =~ '^.*Microsoft Developer Studio Project File.*$'
1361 setf make
1362 return
1363 endif
1364
1365 if line =~ '^!MESSAGE This is not a valid makefile\..+$'
1366 setf make
1367 return
1368 endif
1369
1370 # Check for keywords
1371 if line =~ '^!(IF,ELSEIF,ENDIF).*$'
1372 setf make
1373 return
1374 endif
1375
1376 # Check for common assignments
1377 if line =~ '^SOURCE=.*$'
1378 setf make
1379 return
1380 endif
1381 endfor
1382
1383 # Otherwise, assume we have a Faust file
1384 setf faust
1385enddef
1386
Christian Brabandt99181202025-01-25 14:54:28 +01001387# Set the filetype of a *.v file to Verilog, V or Cog based on the first 500
Turiiya80406c22023-04-22 21:38:47 +01001388# lines.
1389export def FTv()
1390 if did_filetype()
1391 # ":setf" will do nothing, bail out early
1392 return
1393 endif
Christian Brabandt10b4f752024-01-01 19:19:20 +01001394 if exists("g:filetype_v")
1395 exe "setf " .. g:filetype_v
1396 return
1397 endif
Turiiya80406c22023-04-22 21:38:47 +01001398
Christian Brabandt10b4f752024-01-01 19:19:20 +01001399 var in_comment = 0
Christian Brabandt99181202025-01-25 14:54:28 +01001400 for lnum in range(1, min([line("$"), 500]))
Christian Brabandt10b4f752024-01-01 19:19:20 +01001401 var line = getline(lnum)
1402 # Skip Verilog and V comments (lines and blocks).
1403 if line =~ '^\s*/\*'
1404 # start comment block
1405 in_comment = 1
1406 endif
1407 if in_comment == 1
1408 if line =~ '\*/'
1409 # end comment block
1410 in_comment = 0
1411 endif
1412 # skip comment-block line
1413 continue
1414 endif
1415 if line =~ '^\s*//'
Turiiya80406c22023-04-22 21:38:47 +01001416 # skip comment line
1417 continue
1418 endif
1419
Christian Brabandt10b4f752024-01-01 19:19:20 +01001420 # Coq: line ends with a '.' followed by an optional variable number of
1421 # spaces or contains the start of a comment, but not inside a Verilog or V
1422 # comment.
1423 # Example: "Definition x := 10. (*".
1424 if (line =~ '\.\s*$' && line !~ '/[/*]') || (line =~ '(\*' && line !~ '/[/*].*(\*')
1425 setf coq
1426 return
1427 endif
1428
Turiiya80406c22023-04-22 21:38:47 +01001429 # Verilog: line ends with ';' followed by an optional variable number of
1430 # spaces and an optional start of a comment.
1431 # Example: " b <= a + 1; // Add 1".
Christian Brabandtfb49e3c2025-01-25 16:18:51 +01001432 # Alternatively: a module is defined: " module MyModule ( input )"
1433 if line =~ ';\s*\(/[/*].*\)\?$' || line =~ '\C^\s*module\s\+\w\+\s*('
Turiiya80406c22023-04-22 21:38:47 +01001434 setf verilog
1435 return
1436 endif
Turiiya80406c22023-04-22 21:38:47 +01001437 endfor
1438
1439 # No line matched, fall back to "v".
1440 setf v
1441enddef
1442
Doug Kearnsf97f6bb2023-08-27 18:44:09 +02001443export def FTvba()
1444 if getline(1) =~ '^["#] Vimball Archiver'
1445 setf vim
1446 else
1447 setf vb
1448 endif
1449enddef
1450
Colin Caine4b3fab12024-04-18 23:53:02 +02001451export def Detect_UCI_statements(): bool
1452 # Match a config or package statement at the start of the line.
1453 const config_or_package_statement = '^\s*\(\(c\|config\)\|\(p\|package\)\)\s\+\S'
1454 # Match a line that is either all blank or blank followed by a comment
1455 const comment_or_blank = '^\s*\(#.*\)\?$'
1456
1457 # Return true iff the file has a config or package statement near the
1458 # top of the file and all preceding lines were comments or blank.
1459 return getline(1) =~# config_or_package_statement
1460 \ || getline(1) =~# comment_or_blank
1461 \ && ( getline(2) =~# config_or_package_statement
1462 \ || getline(2) =~# comment_or_blank
1463 \ && getline(3) =~# config_or_package_statement
1464 \ )
1465enddef
1466
Bram Moolenaara2baa732022-02-04 16:09:54 +00001467# Uncomment this line to check for compilation errors early
dkearnsef387c02024-02-20 06:58:30 +11001468# defcompile