-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathlanguages.lua
177 lines (141 loc) · 3.77 KB
/
languages.lua
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
local core = require 'core'
local command = require 'core.command'
local common = require 'core.common'
local config = require 'plugins.evergreen.config'
local util = require 'plugins.evergreen.util'
local ts = require 'libraries.tree_sitter'
local M = {
defs = {},
langCache = {},
queryCache = {
highlights = {},
},
}
function M.addDef(defOptions)
local def = {}
assert(defOptions.name, 'Name is required for language definition')
assert(not M.defs[defOptions.name], 'Duplicate language name')
assert(defOptions.path, 'Path is required for language definition')
def.name = defOptions.name
def.files = defOptions.files
local path = common.home_expand(defOptions.path)
if defOptions.files and #defOptions.files > 0 then
def.soFile = util.joinPath {
path,
defOptions.soFile and
defOptions.soFile:gsub('{SOEXT}', config.soExt) or
'parser' .. config.soExt
}
end
def.queryFiles = {}
def.queryFiles.highlights = util.joinPath {
path,
defOptions.queryFiles.highlights or 'queries/highlights.scm'
}
M.defs[#M.defs + 1] = def
M.defs[def.name] = def
end
function M.findDef(filename)
local bestScore = 0
local bestDef
for i = #M.defs, 1, -1 do
local def = M.defs[i]
if not def.files then goto continue end
for _, pattern in ipairs(def.files) do
local s, e = filename:find(pattern)
if not s then goto continue end
local score = e - s
if score > bestScore then
bestScore = score
bestDef = def
end
::continue::
end
::continue::
end
return bestDef
end
function M.getLang(def)
local lang = M.langCache[def.name]
if lang then
return lang
end
local ok, result = pcall(ts.Language.load, def.soFile, def.name)
if not ok then
core.error('Error loading language ' .. def.name .. ':\n' .. result)
return nil
end
M.langCache[def.name] = result
core.log('Loaded language ' .. def.name)
return result
end
function M.getQuery(def, queryType)
local query = M.queryCache[queryType][def.name]
if query then
return query
end
local f = io.open(def.queryFiles[queryType])
if not f then
core.error('Error loading ' .. def.name .. ' ' .. queryType .. ' query')
return nil
end
local builder = { '; EVERGREEN: BEGIN ' .. def.name .. '\n' }
while true do
local head = f:read '*l'
if not head:match '%s*;' then
break
end
local names = head:match '%s*;+%s*inherits%s*:%s*([%l_,]*)'
if names then
for name in names:gmatch '[%l_]+' do
local inheritDef = M.defs[name]
if not inheritDef then
core.warn(
'Could not find language %s to inherit queries from. \z
Syntax highlighting may be incomplete.',
name
)
goto continue
end
builder[#builder + 1] = '; EVERGREEN: INHERIT ' .. name .. '\n'
builder[#builder + 1] = M.getQuery(M.defs[name], queryType)
::continue::
end
end
end
f:seek('set', 0)
builder[#builder + 1] = f:read '*a'
f:close()
builder[#builder + 1] = '; EVERGREEN: END ' .. def.name .. '\n'
query = table.concat(builder)
M.queryCache[queryType][def.name] = query
core.log('Loaded ' .. def.name .. ' ' .. queryType .. ' query')
return query
end
local queryRecents = {}
command.add(nil, {
['evergreen:view-highlights-query'] = function()
core.command_view:enter('View highlights query for language', {
submit = function(name)
local def = M.defs[name]
if not def then
core.error('No such langauge %s', name)
return
end
local doc = core.open_doc('highlights.scm')
core.root_view:open_doc(doc)
doc:insert(1, 1, M.getQuery(def, 'highlights'))
doc.new_file = false
doc:clean()
end,
suggest = function(name)
local names = {}
for _, def in ipairs(M.defs) do
names[#names + 1] = def.name
end
return common.fuzzy_match_with_recents(names, queryRecents, name)
end,
})
end,
})
return M