229 lines
6.8 KiB
Lua
229 lines
6.8 KiB
Lua
local syntax = require "core.syntax"
|
|
|
|
local tokenizer = {}
|
|
|
|
local function push_token(t, type, text)
|
|
local prev_type = t[#t-1]
|
|
local prev_text = t[#t]
|
|
if prev_type and (prev_type == type or prev_text:find("^%s*$")) then
|
|
t[#t-1] = type
|
|
t[#t] = prev_text .. text
|
|
else
|
|
table.insert(t, type)
|
|
table.insert(t, text)
|
|
end
|
|
end
|
|
|
|
|
|
local function push_tokens(t, syn, pattern, full_text, find_results)
|
|
if #find_results > 2 then
|
|
-- We do some manipulation with find_results so that it's arranged
|
|
-- like this:
|
|
-- { start, end, i_1, i_2, i_3, …, i_last }
|
|
-- Each position spans characters from i_n to ((i_n+1) - 1), to form
|
|
-- consecutive spans of text.
|
|
--
|
|
-- If i_1 is not equal to start, start is automatically inserted at
|
|
-- that index.
|
|
if find_results[3] ~= find_results[1] then
|
|
table.insert(find_results, 3, find_results[1])
|
|
end
|
|
-- Copy the ending index to the end of the table, so that an ending index
|
|
-- always follows a starting index after position 3 in the table.
|
|
table.insert(find_results, find_results[2] + 1)
|
|
-- Then, we just iterate over our modified table.
|
|
for i = 3, #find_results - 1 do
|
|
local start = find_results[i]
|
|
local fin = find_results[i + 1] - 1
|
|
local type = pattern.type[i - 2]
|
|
-- ↑ (i - 2) to convert from [3; n] to [1; n]
|
|
local text = full_text:sub(start, fin)
|
|
push_token(t, syn.symbols[text] or type, text)
|
|
end
|
|
else
|
|
local start, fin = find_results[1], find_results[2]
|
|
local text = full_text:sub(start, fin)
|
|
push_token(t, syn.symbols[text] or pattern.type, text)
|
|
end
|
|
end
|
|
|
|
|
|
local function is_escaped(text, idx, esc)
|
|
local byte = esc:byte()
|
|
local count = 0
|
|
for i = idx - 1, 1, -1 do
|
|
if text:byte(i) ~= byte then break end
|
|
count = count + 1
|
|
end
|
|
return count % 2 == 1
|
|
end
|
|
|
|
|
|
local function find_non_escaped(text, pattern, offset, esc)
|
|
while true do
|
|
local s, e = text:find(pattern, offset)
|
|
if not s then break end
|
|
if esc and is_escaped(text, s, esc) then
|
|
offset = e + 1
|
|
else
|
|
return s, e
|
|
end
|
|
end
|
|
end
|
|
|
|
-- State is a 32-bit number that is four separate bytes, illustrating how many
|
|
-- differnet delimiters we have open, and which subsyntaxes we have active.
|
|
-- At most, there are 3 subsyntaxes active at the same time. Beyond that,
|
|
-- does not support further highlighting.
|
|
local function retrieve_syntax_state(incoming_syntax, state)
|
|
local current_syntax, subsyntax_info, current_state, current_level =
|
|
incoming_syntax, nil, state, 0
|
|
if state > 0 and (state > 255 or current_syntax.patterns[state].syntax) then
|
|
-- If we have higher bits, then decode them one at a time, and find which
|
|
-- syntax we're using. Rather than walking the bytes, and calling into
|
|
-- `syntax` each time, we could probably cache this in a single table.
|
|
for i = 0, 2 do
|
|
local target = bit32.extract(state, i*8, 8)
|
|
if target ~= 0 then
|
|
if current_syntax.patterns[target].syntax then
|
|
subsyntax_info = current_syntax.patterns[target]
|
|
current_syntax = type(subsyntax_info.syntax) == "table" and
|
|
subsyntax_info.syntax or syntax.get(subsyntax_info.syntax)
|
|
current_state = 0
|
|
current_level = i+1
|
|
else
|
|
current_state = target
|
|
break
|
|
end
|
|
else
|
|
break
|
|
end
|
|
end
|
|
end
|
|
return current_syntax, subsyntax_info, current_state, current_level
|
|
end
|
|
|
|
function tokenizer.tokenize(incoming_syntax, text, state)
|
|
local res = {}
|
|
local i = 1
|
|
|
|
if #incoming_syntax.patterns == 0 then
|
|
return { "normal", text }
|
|
end
|
|
|
|
state = state or 0
|
|
local current_syntax, subsyntax_info, current_state, current_level =
|
|
retrieve_syntax_state(incoming_syntax, state)
|
|
while i <= #text do
|
|
-- continue trying to match the end pattern of a pair if we have a state set
|
|
if current_state > 0 then
|
|
local p = current_syntax.patterns[current_state]
|
|
local s, e = find_non_escaped(text, p.pattern[2], i, p.pattern[3])
|
|
|
|
local cont = true
|
|
-- If we're in subsyntax mode, always check to see if we end our syntax
|
|
-- first.
|
|
if subsyntax_info then
|
|
local ss, se = find_non_escaped(
|
|
text,
|
|
subsyntax_info.pattern[2],
|
|
i,
|
|
subsyntax_info.pattern[3]
|
|
)
|
|
if ss and (s == nil or ss < s) then
|
|
push_token(res, p.type, text:sub(i, ss - 1))
|
|
i = ss
|
|
cont = false
|
|
end
|
|
end
|
|
if cont then
|
|
if s then
|
|
push_token(res, p.type, text:sub(i, e))
|
|
current_state = 0
|
|
state = bit32.replace(state, 0, current_level*8, 8)
|
|
i = e + 1
|
|
else
|
|
push_token(res, p.type, text:sub(i))
|
|
break
|
|
end
|
|
end
|
|
end
|
|
-- Check for end of syntax.
|
|
if subsyntax_info then
|
|
local s, e = find_non_escaped(
|
|
text,
|
|
"^" .. subsyntax_info.pattern[2],
|
|
i,
|
|
nil
|
|
)
|
|
if s then
|
|
push_token(res, subsyntax_info.type, text:sub(i, e))
|
|
current_level = current_level - 1
|
|
-- Zero out the state above us, as well as our new current state.
|
|
state = bit32.replace(state, 0, current_level*8, 16)
|
|
current_syntax, subsyntax_info, current_state, current_level =
|
|
retrieve_syntax_state(incoming_syntax, state)
|
|
i = e + 1
|
|
end
|
|
end
|
|
|
|
-- find matching pattern
|
|
local matched = false
|
|
for n, p in ipairs(current_syntax.patterns) do
|
|
local pattern = (type(p.pattern) == "table") and p.pattern[1] or p.pattern
|
|
local find_results = { text:find("^" .. pattern, i) }
|
|
local start, fin = find_results[1], find_results[2]
|
|
|
|
if start then
|
|
-- matched pattern; make and add tokens
|
|
push_tokens(res, current_syntax, p, text, find_results)
|
|
|
|
-- update state if this was a start|end pattern pair
|
|
if type(p.pattern) == "table" then
|
|
state = bit32.replace(state, n, current_level*8, 8)
|
|
-- If we've found a new subsyntax, bump our level, and set the
|
|
-- appropriate variables.
|
|
if p.syntax then
|
|
current_level = current_level + 1
|
|
subsyntax_info = p
|
|
current_syntax = type(p.syntax) == "table" and
|
|
p.syntax or syntax.get(p.syntax)
|
|
current_state = 0
|
|
else
|
|
current_state = n
|
|
end
|
|
end
|
|
|
|
-- move cursor past this token
|
|
i = fin + 1
|
|
matched = true
|
|
break
|
|
end
|
|
end
|
|
|
|
-- consume character if we didn't match
|
|
if not matched then
|
|
push_token(res, "normal", text:sub(i, i))
|
|
i = i + 1
|
|
end
|
|
end
|
|
|
|
return res, state
|
|
end
|
|
|
|
|
|
local function iter(t, i)
|
|
i = i + 2
|
|
local type, text = t[i], t[i+1]
|
|
if type then
|
|
return i, type, text
|
|
end
|
|
end
|
|
|
|
function tokenizer.each_token(t)
|
|
return iter, t, -1
|
|
end
|
|
|
|
|
|
return tokenizer
|