blob: 3bf552a1a5c53739f9739faaa6e85b1d955ba78c [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#
5# Maintainer: Bram Moolenaar <Bram@vim.org>
Bram Moolenaarcbaff5e2022-04-08 17:45:08 +01006# Last Change: 2022 Apr 06
Bram Moolenaar851ee6c2017-11-09 20:46:17 +01007
Bram Moolenaara2baa732022-02-04 16:09:54 +00008# These functions are moved here from runtime/filetype.vim to make startup
9# faster.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010010
Bram Moolenaara2baa732022-02-04 16:09:54 +000011export def Check_inp()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010012 if getline(1) =~ '^\*'
13 setf abaqus
14 else
Bram Moolenaara2baa732022-02-04 16:09:54 +000015 var n = 1
16 var nmax = line("$") > 500 ? 500 : line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010017 while n <= nmax
18 if getline(n) =~? "^header surface data"
19 setf trasys
20 break
21 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +000022 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010023 endwhile
24 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +000025enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010026
Bram Moolenaara2baa732022-02-04 16:09:54 +000027# This function checks for the kind of assembly that is wanted by the user, or
28# can be detected from the first five lines of the file.
29export def FTasm()
30 # make sure b:asmsyntax exists
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010031 if !exists("b:asmsyntax")
Bram Moolenaara2baa732022-02-04 16:09:54 +000032 b:asmsyntax = ""
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010033 endif
34
35 if b:asmsyntax == ""
Bram Moolenaara2baa732022-02-04 16:09:54 +000036 FTasmsyntax()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010037 endif
38
Bram Moolenaara2baa732022-02-04 16:09:54 +000039 # if b:asmsyntax still isn't set, default to asmsyntax or GNU
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010040 if b:asmsyntax == ""
41 if exists("g:asmsyntax")
Bram Moolenaara2baa732022-02-04 16:09:54 +000042 b:asmsyntax = g:asmsyntax
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010043 else
Bram Moolenaara2baa732022-02-04 16:09:54 +000044 b:asmsyntax = "asm"
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010045 endif
46 endif
47
Bram Moolenaara2baa732022-02-04 16:09:54 +000048 exe "setf " .. fnameescape(b:asmsyntax)
49enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010050
Bram Moolenaara2baa732022-02-04 16:09:54 +000051export def FTasmsyntax()
52 # see if the file contains any asmsyntax=foo overrides. If so, change
53 # b:asmsyntax appropriately
54 var head = " " .. getline(1) .. " " .. getline(2) .. " "
55 .. getline(3) .. " " .. getline(4) .. " " .. getline(5) .. " "
56 var match = matchstr(head, '\sasmsyntax=\zs[a-zA-Z0-9]\+\ze\s')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010057 if match != ''
Bram Moolenaara2baa732022-02-04 16:09:54 +000058 b:asmsyntax = match
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010059 elseif ((head =~? '\.title') || (head =~? '\.ident') || (head =~? '\.macro') || (head =~? '\.subtitle') || (head =~? '\.library'))
Bram Moolenaara2baa732022-02-04 16:09:54 +000060 b:asmsyntax = "vmasm"
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010061 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +000062enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010063
Bram Moolenaara2baa732022-02-04 16:09:54 +000064var ft_visual_basic_content = '\cVB_Name\|Begin VB\.\(Form\|MDIForm\|UserControl\)'
Doug Kearnsc570e9c2022-01-31 17:09:14 +000065
Bram Moolenaara2baa732022-02-04 16:09:54 +000066# See FTfrm() for Visual Basic form file detection
67export def FTbas()
Bram Moolenaar6517f142022-01-21 14:55:13 +000068 if exists("g:filetype_bas")
Bram Moolenaara2baa732022-02-04 16:09:54 +000069 exe "setf " .. g:filetype_bas
Bram Moolenaar6517f142022-01-21 14:55:13 +000070 return
71 endif
72
Bram Moolenaara2baa732022-02-04 16:09:54 +000073 # most frequent FreeBASIC-specific keywords in distro files
74 var fb_keywords = '\c^\s*\%(extern\|var\|enum\|private\|scope\|union\|byref\|operator\|constructor\|delete\|namespace\|public\|property\|with\|destructor\|using\)\>\%(\s*[:=(]\)\@!'
75 var fb_preproc = '\c^\s*\%(#\a\+\|option\s\+\%(byval\|dynamic\|escape\|\%(no\)\=gosub\|nokeyword\|private\|static\)\>\)'
76 var fb_comment = "^\\s*/'"
77 # OPTION EXPLICIT, without the leading underscore, is common to many dialects
78 var qb64_preproc = '\c^\s*\%($\a\+\|option\s\+\%(_explicit\|_\=explicitarray\)\>\)'
Bram Moolenaar6517f142022-01-21 14:55:13 +000079
Bram Moolenaara2baa732022-02-04 16:09:54 +000080 var lines = getline(1, min([line("$"), 100]))
Bram Moolenaar6517f142022-01-21 14:55:13 +000081
82 if match(lines, fb_preproc) > -1 || match(lines, fb_comment) > -1 || match(lines, fb_keywords) > -1
83 setf freebasic
84 elseif match(lines, qb64_preproc) > -1
85 setf qb64
Bram Moolenaarb2c72352022-02-22 21:17:40 +000086 elseif match(lines, ft_visual_basic_content) > -1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010087 setf vb
88 else
Bram Moolenaar6517f142022-01-21 14:55:13 +000089 setf basic
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010090 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +000091enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010092
Bram Moolenaara2baa732022-02-04 16:09:54 +000093export def FTbtm()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +010094 if exists("g:dosbatch_syntax_for_btm") && g:dosbatch_syntax_for_btm
95 setf dosbatch
96 else
97 setf btm
98 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +000099enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100100
Bram Moolenaara2baa732022-02-04 16:09:54 +0000101export def BindzoneCheck(default = '')
102 if getline(1) .. getline(2) .. getline(3) .. getline(4)
103 =~ '^; <<>> DiG [0-9.]\+.* <<>>\|$ORIGIN\|$TTL\|IN\s\+SOA'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100104 setf bindzone
Bram Moolenaara2baa732022-02-04 16:09:54 +0000105 elseif default != ''
106 exe 'setf ' .. default
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100107 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000108enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100109
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100110# Returns true if file content looks like RAPID
111def IsRapid(sChkExt: string = ""): bool
112 if sChkExt == "cfg"
113 return getline(1) =~? '\v^%(EIO|MMC|MOC|PROC|SIO|SYS):CFG'
114 endif
115 # called from FTmod, FTprg or FTsys
116 return getline(nextnonblank(1)) =~? '\v^\s*%(\%{3}|module\s+\k+\s*%(\(|$))'
117enddef
118
119export def FTcfg()
120 if exists("g:filetype_cfg")
121 exe "setf " .. g:filetype_cfg
122 elseif IsRapid("cfg")
123 setf rapid
124 else
125 setf cfg
126 endif
127enddef
128
Bram Moolenaara2baa732022-02-04 16:09:54 +0000129export def FTlpc()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100130 if exists("g:lpc_syntax_for_c")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000131 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100132 while lnum <= 12
133 if getline(lnum) =~# '^\(//\|inherit\|private\|protected\|nosave\|string\|object\|mapping\|mixed\)'
134 setf lpc
135 return
136 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000137 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100138 endwhile
139 endif
140 setf c
Bram Moolenaara2baa732022-02-04 16:09:54 +0000141enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100142
Bram Moolenaara2baa732022-02-04 16:09:54 +0000143export def FTheader()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100144 if match(getline(1, min([line("$"), 200])), '^@\(interface\|end\|class\)') > -1
145 if exists("g:c_syntax_for_h")
146 setf objc
147 else
148 setf objcpp
149 endif
150 elseif exists("g:c_syntax_for_h")
151 setf c
152 elseif exists("g:ch_syntax_for_h")
153 setf ch
154 else
155 setf cpp
156 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000157enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100158
Bram Moolenaara2baa732022-02-04 16:09:54 +0000159# This function checks if one of the first ten lines start with a '@'. In
160# that case it is probably a change file.
161# If the first line starts with # or ! it's probably a ch file.
162# If a line has "main", "include", "//" or "/*" it's probably ch.
163# Otherwise CHILL is assumed.
164export def FTchange()
165 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100166 while lnum <= 10
167 if getline(lnum)[0] == '@'
168 setf change
169 return
170 endif
171 if lnum == 1 && (getline(1)[0] == '#' || getline(1)[0] == '!')
172 setf ch
173 return
174 endif
175 if getline(lnum) =~ "MODULE"
176 setf chill
177 return
178 endif
179 if getline(lnum) =~ 'main\s*(\|#\s*include\|//'
180 setf ch
181 return
182 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000183 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100184 endwhile
185 setf chill
Bram Moolenaara2baa732022-02-04 16:09:54 +0000186enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100187
Bram Moolenaara2baa732022-02-04 16:09:54 +0000188export def FTent()
189 # This function checks for valid cl syntax in the first five lines.
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100190 # Look for either an opening comment, '#', or a block start, '{'.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000191 # If not found, assume SGML.
192 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100193 while lnum < 6
Bram Moolenaara2baa732022-02-04 16:09:54 +0000194 var line = getline(lnum)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100195 if line =~ '^\s*[#{]'
196 setf cl
197 return
198 elseif line !~ '^\s*$'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000199 # Not a blank line, not a comment, and not a block start,
200 # so doesn't look like valid cl code.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100201 break
202 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000203 lnum += 1
Bram Moolenaarc12dc472022-03-05 13:45:56 +0000204 endwhile
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100205 setf dtd
Bram Moolenaara2baa732022-02-04 16:09:54 +0000206enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100207
Bram Moolenaara2baa732022-02-04 16:09:54 +0000208export def ExCheck()
209 var lines = getline(1, min([line("$"), 100]))
Austin Gatlinf3caeb62021-06-26 12:02:55 +0200210 if exists('g:filetype_euphoria')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000211 exe 'setf ' .. g:filetype_euphoria
Austin Gatlinf3caeb62021-06-26 12:02:55 +0200212 elseif match(lines, '^--\|^ifdef\>\|^include\>') > -1
213 setf euphoria3
214 else
215 setf elixir
216 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000217enddef
Austin Gatlinf3caeb62021-06-26 12:02:55 +0200218
Bram Moolenaara2baa732022-02-04 16:09:54 +0000219export def EuphoriaCheck()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100220 if exists('g:filetype_euphoria')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000221 exe 'setf ' .. g:filetype_euphoria
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100222 else
223 setf euphoria3
224 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000225enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100226
Bram Moolenaara2baa732022-02-04 16:09:54 +0000227export def DtraceCheck()
=?UTF-8?q?Teubel=20Gy=C3=B6rgy?=4d56b972022-02-24 17:59:09 +0000228 if did_filetype()
229 # Filetype was already detected
230 return
231 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000232 var lines = getline(1, min([line("$"), 100]))
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100233 if match(lines, '^module\>\|^import\>') > -1
Bram Moolenaara2baa732022-02-04 16:09:54 +0000234 # D files often start with a module and/or import statement.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100235 setf d
236 elseif match(lines, '^#!\S\+dtrace\|#pragma\s\+D\s\+option\|:\S\{-}:\S\{-}:') > -1
237 setf dtrace
238 else
239 setf d
240 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000241enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100242
Bram Moolenaara2baa732022-02-04 16:09:54 +0000243export def FTe()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100244 if exists('g:filetype_euphoria')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000245 exe 'setf ' .. g:filetype_euphoria
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100246 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000247 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100248 while n < 100 && n <= line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100249 if getline(n) =~ "^\\s*\\(<'\\|'>\\)\\s*$"
250 setf specman
251 return
252 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000253 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100254 endwhile
255 setf eiffel
256 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000257enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100258
Bram Moolenaara2baa732022-02-04 16:09:54 +0000259export def FTfrm()
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000260 if exists("g:filetype_frm")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000261 exe "setf " .. g:filetype_frm
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000262 return
263 endif
264
Bram Moolenaara2baa732022-02-04 16:09:54 +0000265 var lines = getline(1, min([line("$"), 5]))
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000266
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000267 if match(lines, ft_visual_basic_content) > -1
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000268 setf vb
269 else
270 setf form
271 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000272enddef
Doug Kearnsc570e9c2022-01-31 17:09:14 +0000273
Bram Moolenaara2baa732022-02-04 16:09:54 +0000274# Distinguish between Forth and F#.
275# Provided by Doug Kearns.
276export def FTfs()
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000277 if exists("g:filetype_fs")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000278 exe "setf " .. g:filetype_fs
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000279 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000280 var line = getline(nextnonblank(1))
281 # comments and colon definitions
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000282 if line =~ '^\s*\.\=( ' || line =~ '^\s*\\G\= ' || line =~ '^\\$'
283 \ || line =~ '^\s*: \S'
284 setf forth
285 else
Bram Moolenaar53ba95e2021-11-30 13:02:58 +0000286 setf fsharp
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000287 endif
288 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000289enddef
Bram Moolenaar3d14c0f2021-11-27 17:22:07 +0000290
Bram Moolenaara2baa732022-02-04 16:09:54 +0000291# Distinguish between HTML, XHTML and Django
292export def FThtml()
293 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100294 while n < 10 && n <= line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100295 if getline(n) =~ '\<DTD\s\+XHTML\s'
296 setf xhtml
297 return
298 endif
299 if getline(n) =~ '{%\s*\(extends\|block\|load\)\>\|{#\s\+'
300 setf htmldjango
301 return
302 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000303 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100304 endwhile
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100305 setf FALLBACK html
Bram Moolenaara2baa732022-02-04 16:09:54 +0000306enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100307
Bram Moolenaara2baa732022-02-04 16:09:54 +0000308# Distinguish between standard IDL and MS-IDL
309export def FTidl()
310 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100311 while n < 50 && n <= line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100312 if getline(n) =~ '^\s*import\s\+"\(unknwn\|objidl\)\.idl"'
313 setf msidl
314 return
315 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000316 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100317 endwhile
318 setf idl
Bram Moolenaara2baa732022-02-04 16:09:54 +0000319enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100320
Bram Moolenaara2baa732022-02-04 16:09:54 +0000321# Distinguish between "default" and Cproto prototype file. */
322export def ProtoCheck(default: string)
323 # Cproto files have a comment in the first line and a function prototype in
324 # the second line, it always ends in ";". Indent files may also have
325 # comments, thus we can't match comments to see the difference.
326 # IDL files can have a single ';' in the second line, require at least one
327 # chacter before the ';'.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100328 if getline(2) =~ '.;$'
329 setf cpp
330 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000331 exe 'setf ' .. default
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100332 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000333enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100334
Bram Moolenaara2baa732022-02-04 16:09:54 +0000335export def FTm()
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200336 if exists("g:filetype_m")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000337 exe "setf " .. g:filetype_m
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200338 return
339 endif
340
Bram Moolenaara2baa732022-02-04 16:09:54 +0000341 # excluding end(for|function|if|switch|while) common to Murphi
342 var octave_block_terminators = '\<end\%(_try_catch\|classdef\|enumeration\|events\|methods\|parfor\|properties\)\>'
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200343
Bram Moolenaara2baa732022-02-04 16:09:54 +0000344 var objc_preprocessor = '^\s*#\s*\%(import\|include\|define\|if\|ifn\=def\|undef\|line\|error\|pragma\)\>'
Doug Kearns7329cfa2021-11-26 13:01:41 +0000345
Bram Moolenaara2baa732022-02-04 16:09:54 +0000346 var n = 1
347 var saw_comment = 0 # Whether we've seen a multiline comment leader.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100348 while n < 100
Bram Moolenaara2baa732022-02-04 16:09:54 +0000349 var line = getline(n)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100350 if line =~ '^\s*/\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000351 # /* ... */ is a comment in Objective C and Murphi, so we can't conclude
352 # it's either of them yet, but track this as a hint in case we don't see
353 # anything more definitive.
354 saw_comment = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100355 endif
Doug Kearns7329cfa2021-11-26 13:01:41 +0000356 if line =~ '^\s*//' || line =~ '^\s*@import\>' || line =~ objc_preprocessor
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100357 setf objc
358 return
359 endif
Bram Moolenaarca0627d2021-09-12 17:03:08 +0200360 if line =~ '^\s*\%(#\|%!\)' || line =~ '^\s*unwind_protect\>' ||
Bram Moolenaardeba5eb2021-09-03 19:21:36 +0200361 \ line =~ '\%(^\|;\)\s*' .. octave_block_terminators
362 setf octave
363 return
364 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000365 # TODO: could be Matlab or Octave
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100366 if line =~ '^\s*%'
367 setf matlab
368 return
369 endif
370 if line =~ '^\s*(\*'
371 setf mma
372 return
373 endif
374 if line =~ '^\c\s*\(\(type\|var\)\>\|--\)'
375 setf murphi
376 return
377 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000378 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100379 endwhile
380
381 if saw_comment
Bram Moolenaara2baa732022-02-04 16:09:54 +0000382 # We didn't see anything definitive, but this looks like either Objective C
383 # or Murphi based on the comment leader. Assume the former as it is more
384 # common.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100385 setf objc
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100386 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000387 # Default is Matlab
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100388 setf matlab
389 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000390enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100391
Bram Moolenaara2baa732022-02-04 16:09:54 +0000392export def FTmms()
393 var n = 1
Bram Moolenaard7df2792020-01-02 21:34:42 +0100394 while n < 20
Bram Moolenaara2baa732022-02-04 16:09:54 +0000395 var line = getline(n)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100396 if line =~ '^\s*\(%\|//\)' || line =~ '^\*'
397 setf mmix
398 return
399 endif
400 if line =~ '^\s*#'
401 setf make
402 return
403 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000404 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100405 endwhile
406 setf mmix
Bram Moolenaara2baa732022-02-04 16:09:54 +0000407enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100408
Bram Moolenaara2baa732022-02-04 16:09:54 +0000409# This function checks if one of the first five lines start with a dot. In
410# that case it is probably an nroff file: 'filetype' is set and 1 is returned.
411export def FTnroff(): number
412 if getline(1)[0] .. getline(2)[0] .. getline(3)[0]
413 .. getline(4)[0] .. getline(5)[0] =~ '\.'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100414 setf nroff
415 return 1
416 endif
417 return 0
Bram Moolenaara2baa732022-02-04 16:09:54 +0000418enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100419
Bram Moolenaara2baa732022-02-04 16:09:54 +0000420export def FTmm()
421 var n = 1
Bram Moolenaard1caa942020-04-10 22:10:56 +0200422 while n < 20
Bram Moolenaara2baa732022-02-04 16:09:54 +0000423 if getline(n) =~ '^\s*\(#\s*\(include\|import\)\>\|@import\>\|/\*\)'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100424 setf objcpp
425 return
426 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000427 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100428 endwhile
429 setf nroff
Bram Moolenaara2baa732022-02-04 16:09:54 +0000430enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100431
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100432# Returns true if file content looks like LambdaProlog
433def IsLProlog(): bool
434 # skip apparent comments and blank lines, what looks like
435 # LambdaProlog comment may be RAPID header
436 var l: number = nextnonblank(1)
437 while l > 0 && l < line('$') && getline(l) =~ '^\s*%' # LambdaProlog comment
438 l = nextnonblank(l + 1)
439 endwhile
440 # this pattern must not catch a go.mod file
441 return getline(l) =~ '\<module\s\+\w\+\s*\.\s*\(%\|$\)'
442enddef
443
444# Determine if *.mod is ABB RAPID, LambdaProlog, Modula-2, Modsim III or go.mod
445export def FTmod()
446 if exists("g:filetype_mod")
447 exe "setf " .. g:filetype_mod
448 elseif IsLProlog()
449 setf lprolog
450 elseif getline(nextnonblank(1)) =~ '\%(\<MODULE\s\+\w\+\s*;\|^\s*(\*\)'
451 setf modula2
452 elseif IsRapid()
453 setf rapid
454 elseif expand("<afile>") =~ '\<go.mod$'
455 setf gomod
456 else
457 # Nothing recognized, assume modsim3
458 setf modsim3
459 endif
460enddef
461
Bram Moolenaara2baa732022-02-04 16:09:54 +0000462export def FTpl()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100463 if exists("g:filetype_pl")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000464 exe "setf " .. g:filetype_pl
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100465 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000466 # recognize Prolog by specific text in the first non-empty line
467 # require a blank after the '%' because Perl uses "%list" and "%translate"
468 var l = getline(nextnonblank(1))
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100469 if l =~ '\<prolog\>' || l =~ '^\s*\(%\+\(\s\|$\)\|/\*\)' || l =~ ':-'
470 setf prolog
471 else
472 setf perl
473 endif
474 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000475enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100476
Bram Moolenaara2baa732022-02-04 16:09:54 +0000477export def FTinc()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100478 if exists("g:filetype_inc")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000479 exe "setf " .. g:filetype_inc
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100480 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000481 var lines = getline(1) .. getline(2) .. getline(3)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100482 if lines =~? "perlscript"
483 setf aspperl
484 elseif lines =~ "<%"
485 setf aspvbs
486 elseif lines =~ "<?"
487 setf php
Bram Moolenaara2baa732022-02-04 16:09:54 +0000488 # Pascal supports // comments but they're vary rarely used for file
489 # headers so assume POV-Ray
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000490 elseif lines =~ '^\s*\%({\|(\*\)' || lines =~? ft_pascal_keywords
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100491 setf pascal
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100492 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000493 FTasmsyntax()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100494 if exists("b:asmsyntax")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000495 exe "setf " .. fnameescape(b:asmsyntax)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100496 else
497 setf pov
498 endif
499 endif
500 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000501enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100502
Bram Moolenaara2baa732022-02-04 16:09:54 +0000503export def FTprogress_cweb()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100504 if exists("g:filetype_w")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000505 exe "setf " .. g:filetype_w
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100506 return
507 endif
508 if getline(1) =~ '&ANALYZE' || getline(3) =~ '&GLOBAL-DEFINE'
509 setf progress
510 else
511 setf cweb
512 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000513enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100514
Bram Moolenaara2baa732022-02-04 16:09:54 +0000515export def FTprogress_asm()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100516 if exists("g:filetype_i")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000517 exe "setf " .. g:filetype_i
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100518 return
519 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000520 # This function checks for an assembly comment the first ten lines.
521 # If not found, assume Progress.
522 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100523 while lnum <= 10 && lnum < line('$')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000524 var line = getline(lnum)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100525 if line =~ '^\s*;' || line =~ '^\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000526 FTasm()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100527 return
528 elseif line !~ '^\s*$' || line =~ '^/\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000529 # Not an empty line: Doesn't look like valid assembly code.
530 # Or it looks like a Progress /* comment
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100531 break
532 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000533 lnum += 1
Bram Moolenaarc12dc472022-03-05 13:45:56 +0000534 endwhile
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100535 setf progress
Bram Moolenaara2baa732022-02-04 16:09:54 +0000536enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100537
Bram Moolenaara2baa732022-02-04 16:09:54 +0000538var ft_pascal_comments = '^\s*\%({\|(\*\|//\)'
539var ft_pascal_keywords = '^\s*\%(program\|unit\|library\|uses\|begin\|procedure\|function\|const\|type\|var\)\>'
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100540
Bram Moolenaara2baa732022-02-04 16:09:54 +0000541export def FTprogress_pascal()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100542 if exists("g:filetype_p")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000543 exe "setf " .. g:filetype_p
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100544 return
545 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000546 # This function checks for valid Pascal syntax in the first ten lines.
547 # Look for either an opening comment or a program start.
548 # If not found, assume Progress.
549 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100550 while lnum <= 10 && lnum < line('$')
Bram Moolenaara2baa732022-02-04 16:09:54 +0000551 var line = getline(lnum)
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000552 if line =~ ft_pascal_comments || line =~? ft_pascal_keywords
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100553 setf pascal
554 return
555 elseif line !~ '^\s*$' || line =~ '^/\*'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000556 # Not an empty line: Doesn't look like valid Pascal code.
557 # Or it looks like a Progress /* comment
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100558 break
559 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000560 lnum += 1
Bram Moolenaarc12dc472022-03-05 13:45:56 +0000561 endwhile
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100562 setf progress
Bram Moolenaara2baa732022-02-04 16:09:54 +0000563enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100564
Bram Moolenaara2baa732022-02-04 16:09:54 +0000565export def FTpp()
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100566 if exists("g:filetype_pp")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000567 exe "setf " .. g:filetype_pp
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100568 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000569 var line = getline(nextnonblank(1))
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000570 if line =~ ft_pascal_comments || line =~? ft_pascal_keywords
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100571 setf pascal
572 else
573 setf puppet
574 endif
575 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000576enddef
Bram Moolenaara0122dc2021-01-12 17:42:24 +0100577
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100578# Determine if *.prg is ABB RAPID. Can also be Clipper, FoxPro or eviews
579export def FTprg()
580 if exists("g:filetype_prg")
581 exe "setf " .. g:filetype_prg
582 elseif IsRapid()
583 setf rapid
584 else
585 # Nothing recognized, assume Clipper
586 setf clipper
587 endif
588enddef
589
Bram Moolenaara2baa732022-02-04 16:09:54 +0000590export def FTr()
591 var max = line("$") > 50 ? 50 : line("$")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100592
593 for n in range(1, max)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000594 # Rebol is easy to recognize, check for that first
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100595 if getline(n) =~? '\<REBOL\>'
596 setf rebol
597 return
598 endif
599 endfor
600
601 for n in range(1, max)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000602 # R has # comments
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100603 if getline(n) =~ '^\s*#'
604 setf r
605 return
606 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000607 # Rexx has /* comments */
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100608 if getline(n) =~ '^\s*/\*'
609 setf rexx
610 return
611 endif
612 endfor
613
Bram Moolenaara2baa732022-02-04 16:09:54 +0000614 # Nothing recognized, use user default or assume Rexx
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100615 if exists("g:filetype_r")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000616 exe "setf " .. g:filetype_r
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100617 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000618 # Rexx used to be the default, but R appears to be much more popular.
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100619 setf r
620 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000621enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100622
Bram Moolenaara2baa732022-02-04 16:09:54 +0000623export def McSetf()
624 # Rely on the file to start with a comment.
625 # MS message text files use ';', Sendmail files use '#' or 'dnl'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100626 for lnum in range(1, min([line("$"), 20]))
Bram Moolenaara2baa732022-02-04 16:09:54 +0000627 var line = getline(lnum)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100628 if line =~ '^\s*\(#\|dnl\)'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000629 setf m4 # Sendmail .mc file
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100630 return
631 elseif line =~ '^\s*;'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000632 setf msmessages # MS Message text file
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100633 return
634 endif
635 endfor
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100636 setf m4 # Default: Sendmail .mc file
Bram Moolenaara2baa732022-02-04 16:09:54 +0000637enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100638
Bram Moolenaara2baa732022-02-04 16:09:54 +0000639# Called from filetype.vim and scripts.vim.
640export def SetFileTypeSH(name: string)
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100641 if did_filetype()
Bram Moolenaara2baa732022-02-04 16:09:54 +0000642 # Filetype was already detected
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100643 return
644 endif
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100645 if expand("<amatch>") =~ g:ft_ignore_pat
646 return
647 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000648 if name =~ '\<csh\>'
649 # Some .sh scripts contain #!/bin/csh.
650 SetFileTypeShell("csh")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100651 return
Bram Moolenaara2baa732022-02-04 16:09:54 +0000652 elseif name =~ '\<tcsh\>'
653 # Some .sh scripts contain #!/bin/tcsh.
654 SetFileTypeShell("tcsh")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100655 return
Bram Moolenaara2baa732022-02-04 16:09:54 +0000656 elseif name =~ '\<zsh\>'
657 # Some .sh scripts contain #!/bin/zsh.
658 SetFileTypeShell("zsh")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100659 return
Bram Moolenaara2baa732022-02-04 16:09:54 +0000660 elseif name =~ '\<ksh\>'
661 b:is_kornshell = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100662 if exists("b:is_bash")
663 unlet b:is_bash
664 endif
665 if exists("b:is_sh")
666 unlet b:is_sh
667 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000668 elseif exists("g:bash_is_sh") || name =~ '\<bash\>' || name =~ '\<bash2\>'
669 b:is_bash = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100670 if exists("b:is_kornshell")
671 unlet b:is_kornshell
672 endif
673 if exists("b:is_sh")
674 unlet b:is_sh
675 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000676 elseif name =~ '\<sh\>'
677 b:is_sh = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100678 if exists("b:is_kornshell")
679 unlet b:is_kornshell
680 endif
681 if exists("b:is_bash")
682 unlet b:is_bash
683 endif
684 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000685 SetFileTypeShell("sh")
686enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100687
Bram Moolenaara2baa732022-02-04 16:09:54 +0000688# For shell-like file types, check for an "exec" command hidden in a comment,
689# as used for Tcl.
690# Also called from scripts.vim, thus can't be local to this script.
691export def SetFileTypeShell(name: string)
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100692 if did_filetype()
Bram Moolenaara2baa732022-02-04 16:09:54 +0000693 # Filetype was already detected
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100694 return
695 endif
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100696 if expand("<amatch>") =~ g:ft_ignore_pat
697 return
698 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000699 var l = 2
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100700 while l < 20 && l < line("$") && getline(l) =~ '^\s*\(#\|$\)'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000701 # Skip empty and comment lines.
702 l += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100703 endwhile
704 if l < line("$") && getline(l) =~ '\s*exec\s' && getline(l - 1) =~ '^\s*#.*\\$'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000705 # Found an "exec" line after a comment with continuation
706 var n = substitute(getline(l), '\s*exec\s\+\([^ ]*/\)\=', '', '')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100707 if n =~ '\<tclsh\|\<wish'
708 setf tcl
709 return
710 endif
711 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000712 exe "setf " .. name
713enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100714
Bram Moolenaara2baa732022-02-04 16:09:54 +0000715export def CSH()
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100716 if did_filetype()
Bram Moolenaara2baa732022-02-04 16:09:54 +0000717 # Filetype was already detected
Bram Moolenaar147e7d02019-01-18 21:46:47 +0100718 return
719 endif
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100720 if exists("g:filetype_csh")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000721 SetFileTypeShell(g:filetype_csh)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100722 elseif &shell =~ "tcsh"
Bram Moolenaara2baa732022-02-04 16:09:54 +0000723 SetFileTypeShell("tcsh")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100724 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000725 SetFileTypeShell("csh")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100726 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000727enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100728
Bram Moolenaara2baa732022-02-04 16:09:54 +0000729var ft_rules_udev_rules_pattern = '^\s*\cudev_rules\s*=\s*"\([^"]\{-1,}\)/*".*'
730export def FTRules()
731 var path = expand('<amatch>:p')
Bram Moolenaaraa9675a2020-08-17 21:57:09 +0200732 if path =~ '/\(etc/udev/\%(rules\.d/\)\=.*\.rules\|\%(usr/\)\=lib/udev/\%(rules\.d/\)\=.*\.rules\)$'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100733 setf udevrules
734 return
735 endif
736 if path =~ '^/etc/ufw/'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000737 setf conf # Better than hog
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100738 return
739 endif
740 if path =~ '^/\(etc\|usr/share\)/polkit-1/rules\.d'
741 setf javascript
742 return
743 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000744 var config_lines: list<string>
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100745 try
Bram Moolenaara2baa732022-02-04 16:09:54 +0000746 config_lines = readfile('/etc/udev/udev.conf')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100747 catch /^Vim\%((\a\+)\)\=:E484/
748 setf hog
749 return
750 endtry
Bram Moolenaara2baa732022-02-04 16:09:54 +0000751 var dir = expand('<amatch>:p:h')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100752 for line in config_lines
Bram Moolenaarb2c72352022-02-22 21:17:40 +0000753 if line =~ ft_rules_udev_rules_pattern
754 var udev_rules = substitute(line, ft_rules_udev_rules_pattern, '\1', "")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100755 if dir == udev_rules
756 setf udevrules
757 endif
758 break
759 endif
760 endfor
761 setf hog
Bram Moolenaara2baa732022-02-04 16:09:54 +0000762enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100763
Bram Moolenaara2baa732022-02-04 16:09:54 +0000764export def SQL()
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100765 if exists("g:filetype_sql")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000766 exe "setf " .. g:filetype_sql
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100767 else
768 setf sql
769 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000770enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100771
Bram Moolenaara2baa732022-02-04 16:09:54 +0000772# If the file has an extension of 't' and is in a directory 't' or 'xt' then
773# it is almost certainly a Perl test file.
774# If the first line starts with '#' and contains 'perl' it's probably a Perl
775# file.
776# (Slow test) If a file contains a 'use' statement then it is almost certainly
777# a Perl file.
778export def FTperl(): number
779 var dirname = expand("%:p:h:t")
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100780 if expand("%:e") == 't' && (dirname == 't' || dirname == 'xt')
781 setf perl
782 return 1
783 endif
784 if getline(1)[0] == '#' && getline(1) =~ 'perl'
785 setf perl
786 return 1
787 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000788 var save_cursor = getpos('.')
789 call cursor(1, 1)
Bram Moolenaare5b78972022-02-05 19:50:34 +0000790 var has_use = search('^use\s\s*\k', 'c', 30) > 0
Bram Moolenaarf0b03c42017-12-17 17:17:07 +0100791 call setpos('.', save_cursor)
792 if has_use
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100793 setf perl
794 return 1
795 endif
796 return 0
Bram Moolenaara2baa732022-02-04 16:09:54 +0000797enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100798
Bram Moolenaar0bbf09c2022-04-09 15:16:53 +0100799export def FTsys()
800 if IsRapid()
801 setf rapid
802 else
803 setf bat
804 endif
805enddef
806
Bram Moolenaara2baa732022-02-04 16:09:54 +0000807# Choose context, plaintex, or tex (LaTeX) based on these rules:
808# 1. Check the first line of the file for "%&<format>".
809# 2. Check the first 1000 non-comment lines for LaTeX or ConTeXt keywords.
810# 3. Default to "plain" or to g:tex_flavor, can be set in user's vimrc.
811export def FTtex()
812 var firstline = getline(1)
813 var format: string
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100814 if firstline =~ '^%&\s*\a\+'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000815 format = tolower(matchstr(firstline, '\a\+'))
816 format = substitute(format, 'pdf', '', '')
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100817 if format == 'tex'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000818 format = 'latex'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100819 elseif format == 'plaintex'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000820 format = 'plain'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100821 endif
822 elseif expand('%') =~ 'tex/context/.*/.*.tex'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000823 format = 'context'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100824 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000825 # Default value, may be changed later:
826 format = exists("g:tex_flavor") ? g:tex_flavor : 'plain'
827 # Save position, go to the top of the file, find first non-comment line.
828 var save_cursor = getpos('.')
829 call cursor(1, 1)
830 var firstNC = search('^\s*[^[:space:]%]', 'c', 1000)
Bram Moolenaare5b78972022-02-05 19:50:34 +0000831 if firstNC > 0
832 # Check the next thousand lines for a LaTeX or ConTeXt keyword.
Bram Moolenaara2baa732022-02-04 16:09:54 +0000833 var lpat = 'documentclass\>\|usepackage\>\|begin{\|newcommand\>\|renewcommand\>'
834 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\>'
835 var kwline = search('^\s*\\\%(' .. lpat .. '\)\|^\s*\\\(' .. cpat .. '\)',
836 'cnp', firstNC + 1000)
837 if kwline == 1 # lpat matched
838 format = 'latex'
839 elseif kwline == 2 # cpat matched
840 format = 'context'
841 endif # If neither matched, keep default set above.
842 # let lline = search('^\s*\\\%(' . lpat . '\)', 'cn', firstNC + 1000)
843 # let cline = search('^\s*\\\%(' . cpat . '\)', 'cn', firstNC + 1000)
844 # if cline > 0
845 # let format = 'context'
846 # endif
847 # if lline > 0 && (cline == 0 || cline > lline)
848 # let format = 'tex'
849 # endif
850 endif # firstNC
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100851 call setpos('.', save_cursor)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000852 endif # firstline =~ '^%&\s*\a\+'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100853
Bram Moolenaara2baa732022-02-04 16:09:54 +0000854 # Translation from formats to file types. TODO: add AMSTeX, RevTex, others?
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100855 if format == 'plain'
856 setf plaintex
857 elseif format == 'context'
858 setf context
Bram Moolenaara2baa732022-02-04 16:09:54 +0000859 else # probably LaTeX
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100860 setf tex
861 endif
862 return
Bram Moolenaara2baa732022-02-04 16:09:54 +0000863enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100864
Bram Moolenaara2baa732022-02-04 16:09:54 +0000865export def FTxml()
866 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100867 while n < 100 && n <= line("$")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000868 var line = getline(n)
869 # DocBook 4 or DocBook 5.
870 var is_docbook4 = line =~ '<!DOCTYPE.*DocBook'
871 var is_docbook5 = line =~ ' xmlns="http://docbook.org/ns/docbook"'
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100872 if is_docbook4 || is_docbook5
Bram Moolenaara2baa732022-02-04 16:09:54 +0000873 b:docbk_type = "xml"
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100874 if is_docbook5
Bram Moolenaara2baa732022-02-04 16:09:54 +0000875 b:docbk_ver = 5
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100876 else
Bram Moolenaara2baa732022-02-04 16:09:54 +0000877 b:docbk_ver = 4
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100878 endif
879 setf docbk
880 return
881 endif
882 if line =~ 'xmlns:xbl="http://www.mozilla.org/xbl"'
883 setf xbl
884 return
885 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000886 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100887 endwhile
888 setf xml
Bram Moolenaara2baa732022-02-04 16:09:54 +0000889enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100890
Bram Moolenaara2baa732022-02-04 16:09:54 +0000891export def FTy()
892 var n = 1
Bram Moolenaar493fbe42019-03-17 17:16:12 +0100893 while n < 100 && n <= line("$")
Bram Moolenaara2baa732022-02-04 16:09:54 +0000894 var line = getline(n)
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100895 if line =~ '^\s*%'
896 setf yacc
897 return
898 endif
899 if getline(n) =~ '^\s*\(#\|class\>\)' && getline(n) !~ '^\s*#\s*include'
900 setf racc
901 return
902 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000903 n += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100904 endwhile
905 setf yacc
Bram Moolenaara2baa732022-02-04 16:09:54 +0000906enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100907
Bram Moolenaara2baa732022-02-04 16:09:54 +0000908export def Redif()
909 var lnum = 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100910 while lnum <= 5 && lnum < line('$')
911 if getline(lnum) =~ "^\ctemplate-type:"
912 setf redif
913 return
914 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000915 lnum += 1
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100916 endwhile
Bram Moolenaara2baa732022-02-04 16:09:54 +0000917enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100918
Bram Moolenaara2baa732022-02-04 16:09:54 +0000919# This function is called for all files under */debian/patches/*, make sure not
920# to non-dep3patch files, such as README and other text files.
921export def Dep3patch()
James McCoy647ab4c2021-12-17 20:52:57 +0000922 if expand('%:t') ==# 'series'
923 return
924 endif
925
926 for ln in getline(1, 100)
927 if ln =~# '^\%(Description\|Subject\|Origin\|Bug\|Forwarded\|Author\|From\|Reviewed-by\|Acked-by\|Last-Updated\|Applied-Upstream\):'
928 setf dep3patch
929 return
930 elseif ln =~# '^---'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000931 # end of headers found. stop processing
James McCoy647ab4c2021-12-17 20:52:57 +0000932 return
933 endif
934 endfor
Bram Moolenaara2baa732022-02-04 16:09:54 +0000935enddef
Bram Moolenaar851ee6c2017-11-09 20:46:17 +0100936
Bram Moolenaara2baa732022-02-04 16:09:54 +0000937# This function checks the first 15 lines for appearance of 'FoamFile'
938# and then 'object' in a following line.
939# In that case, it's probably an OpenFOAM file
940export def FTfoam()
941 var ffile = 0
942 var lnum = 1
Elwardi2284f6c2022-01-11 18:14:23 +0000943 while lnum <= 15
944 if getline(lnum) =~# '^FoamFile'
Bram Moolenaara2baa732022-02-04 16:09:54 +0000945 ffile = 1
Elwardi2284f6c2022-01-11 18:14:23 +0000946 elseif ffile == 1 && getline(lnum) =~# '^\s*object'
947 setf foam
948 return
949 endif
Bram Moolenaara2baa732022-02-04 16:09:54 +0000950 lnum += 1
Elwardi2284f6c2022-01-11 18:14:23 +0000951 endwhile
Bram Moolenaara2baa732022-02-04 16:09:54 +0000952enddef
Elwardi2284f6c2022-01-11 18:14:23 +0000953
Bram Moolenaara2baa732022-02-04 16:09:54 +0000954# Determine if a *.tf file is TF mud client or terraform
955export def FTtf()
956 var numberOfLines = line('$')
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +0000957 for i in range(1, numberOfLines)
Bram Moolenaara2baa732022-02-04 16:09:54 +0000958 var currentLine = trim(getline(i))
959 var firstCharacter = currentLine[0]
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +0000960 if firstCharacter !=? ";" && firstCharacter !=? "/" && firstCharacter !=? ""
961 setf terraform
962 return
963 endif
964 endfor
965 setf tf
Bram Moolenaara2baa732022-02-04 16:09:54 +0000966enddef
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +0000967
Bram Moolenaar3ad20902022-04-06 18:57:39 +0100968# Determine if a *.src file is Kuka Robot Language
969export def FTsrc()
970 if exists("g:filetype_src")
971 exe "setf " .. g:filetype_src
972 elseif getline(nextnonblank(1)) =~? '^\s*\%(&\w\+\|\%(global\s\+\)\?def\>\)'
973 setf krl
974 endif
975enddef
976
977# Determine if a *.dat file is Kuka Robot Language
978export def FTdat()
979 if exists("g:filetype_dat")
980 exe "setf " .. g:filetype_dat
981 elseif getline(nextnonblank(1)) =~? '^\s*\%(&\w\+\|defdat\>\)'
982 setf krl
983 endif
984enddef
=?UTF-8?q?Dundar=20G=C3=B6c?=bd8168c2022-01-28 14:15:09 +0000985
Bram Moolenaara2baa732022-02-04 16:09:54 +0000986# Uncomment this line to check for compilation errors early
Bram Moolenaar3e79c972022-02-04 19:48:06 +0000987# defcompile