1

294 lines
7.8 KiB
Lua
Raw Normal View History

2024-12-30 20:31:50 -08:00
local Cursor = require('argonaut.cursor')
local Options = require('argonaut.options')
local Pairing = require('argonaut.pairing')
local Param = require('argonaut.param')
local Range = {}
Range.__index = Range
local function find_closest_pair(pairing)
local stop_cursor = pairing:find_closest(true)
if stop_cursor then
local start_cursor = pairing:find_closest(false)
if start_cursor then
return Range.new(start_cursor, stop_cursor, pairing)
end
end
end
local function find_closest()
local ranges = {}
for _, brace in pairs({'(', '[', '{', '<'}) do
local pairing = Pairing.from_brace(brace)
local range = find_closest_pair(pairing)
if range then
table.insert(ranges, range)
end
end
if #ranges > 0 then
table.sort(ranges)
2025-01-19 10:35:27 -08:00
return ranges[1]
2024-12-30 20:31:50 -08:00
end
end
2025-01-18 20:17:56 -08:00
local function find_at_cursor(start_cursor, parent_range)
2024-12-30 20:31:50 -08:00
local pairing = Pairing.from_brace(start_cursor:get_value())
if pairing then
local cursor_current = Cursor:get_current()
start_cursor:set_current()
2025-01-18 20:17:56 -08:00
2024-12-30 20:31:50 -08:00
local stop_cursor = pairing:find_closest(true)
cursor_current:set_current()
2025-01-18 20:17:56 -08:00
if not stop_cursor then
return
end
if parent_range and not parent_range:contains(stop_cursor) then
return
2024-12-30 20:31:50 -08:00
end
2025-01-18 20:17:56 -08:00
2025-01-19 10:35:27 -08:00
return Range.new(start_cursor, stop_cursor, pairing)
2024-12-30 20:31:50 -08:00
end
end
function Range.new(start_cursor, stop_cursor, pairing)
local range = {
start_cursor = start_cursor,
stop_cursor = stop_cursor,
pairing = pairing,
params = {},
}
2025-01-19 10:35:27 -08:00
setmetatable(range, Range)
if range:parse() then
return range
end
2024-12-30 20:31:50 -08:00
end
function Range:parse()
local param = nil
local param_index = 1
2025-01-01 18:51:30 -08:00
local append_param_value = function(cell)
2024-12-30 20:31:50 -08:00
if not param then
param = Param.new(self, param_index)
param_index = param_index + 1
end
2025-01-01 18:51:30 -08:00
param:append(cell)
2024-12-30 20:31:50 -08:00
end
local append_param = function()
if param and not param:is_empty() then
2025-01-18 20:07:08 -08:00
table.insert(self.params, param)
2024-12-30 20:31:50 -08:00
end
param = nil
end
2025-01-01 18:51:30 -08:00
local cursor = self.start_cursor:get_next()
2024-12-30 20:31:50 -08:00
while cursor ~= self.stop_cursor do
local value = cursor:get_value()
2025-01-01 18:51:30 -08:00
if cursor:is_string() or cursor:is_comment() then
append_param_value(Param.new_cursor_cell(cursor))
2024-12-30 20:31:50 -08:00
else
2025-01-18 20:17:56 -08:00
local range = find_at_cursor(cursor, self)
2024-12-30 20:31:50 -08:00
if range then
2025-01-01 18:51:30 -08:00
append_param_value(Param.new_range_cell(range))
2024-12-30 20:31:50 -08:00
cursor = range.stop_cursor
elseif value == ',' then
append_param()
else
2025-01-01 18:51:30 -08:00
append_param_value(Param.new_cursor_cell(cursor))
2024-12-30 20:31:50 -08:00
end
end
2025-01-01 18:51:30 -08:00
cursor = cursor:get_next()
2024-12-30 20:31:50 -08:00
end
append_param()
2025-01-19 10:35:27 -08:00
return self:validate()
end
function Range:validate()
for _, param in ipairs(self.params) do
if not param:validate() then
return false
end
end
return true
2024-12-30 20:31:50 -08:00
end
function Range:hit_test(cursor)
if self:contains(cursor) then
2025-01-18 19:25:38 -08:00
local frame = {
type = 'range',
pairing = self.pairing.open .. self.pairing.close,
param_count = #self.params,
param_index = 0,
}
2025-01-19 09:49:45 -08:00
if cursor == self.start_cursor and self.params[1]:is_after_cursor(cursor) then
cursor = cursor:get_next()
elseif cursor == self.stop_cursor and self.params[#self.params]:is_before_cursor(cursor) then
cursor = cursor:get_previous()
end
for i = 2, #self.params do
local current_param = self.params[i]
local previous_param = self.params[i - 1]
if previous_param:is_before_cursor(cursor) and current_param:is_after_cursor(cursor) then
if not previous_param:hit_test(cursor) and not current_param:hit_test(cursor) then
cursor = cursor:get_previous()
break
end
end
end
2024-12-30 20:31:50 -08:00
for i, param in pairs(self.params) do
2025-01-18 18:24:01 -08:00
local trace = param:hit_test(cursor)
if trace then
2025-01-18 19:25:38 -08:00
frame.param_index = i
table.insert(trace, 1, frame)
2025-01-18 18:24:01 -08:00
return trace
2024-12-30 20:31:50 -08:00
end
end
end
end
function Range:hit_search(trace, depth)
2025-01-18 18:24:01 -08:00
local frame = trace[depth]
2025-01-18 19:25:38 -08:00
assert(frame.type == 'range')
assert(frame.pairing == self.pairing.open .. self.pairing.close)
assert(frame.param_count == #self.params)
assert(frame.param_index <= #self.params)
2025-01-18 19:11:04 -08:00
2025-01-19 09:49:45 -08:00
if depth == #trace then
return self
2024-12-30 20:31:50 -08:00
else
2025-01-18 19:25:38 -08:00
return self.params[frame.param_index]:hit_search(trace, depth + 1)
2024-12-30 20:31:50 -08:00
end
end
function Range:write(builder, wrapped)
if wrapped == nil then
wrapped = self:is_wrapped()
end
if wrapped then
builder:write(self.pairing.open)
if #self.params > 0 then
builder:endline()
builder:push_indent()
for i, param in ipairs(self.params) do
2024-12-31 10:20:43 -08:00
local is_first_param = i == 1
2024-12-30 20:31:50 -08:00
local is_last_param = i == #self.params
2024-12-31 10:20:43 -08:00
if self:query_option('comma_prefix') then
if is_first_param then
if self:query_option('comma_prefix_indent') then
builder:write(' ')
end
else
builder:write(', ')
end
param:write(builder)
else
param:write(builder)
if not is_last_param or self:query_option('comma_last') then
builder:write(',')
end
2024-12-30 20:31:50 -08:00
end
2024-12-31 10:20:43 -08:00
2024-12-30 20:31:50 -08:00
if not is_last_param then
builder:endline()
end
end
if self:query_option('brace_last_wrap') then
builder:endline()
builder:pop_indent()
end
end
if not self:query_option('brace_last_wrap') then
if self:query_option('brace_last_indent') then
builder:indent()
end
if self:query_option('brace_pad') then
builder:write(' ')
end
end
builder:write(self.pairing.close)
else
builder:write(self.pairing.open)
if self:query_option('brace_pad') then
builder:write(' ')
end
for i, param in ipairs( self.params ) do
param:write(builder)
local is_last_param = i == #self.params
if not is_last_param then
builder:write(', ')
end
end
if self:query_option('brace_pad') then
builder:write(' ')
end
builder:write(self.pairing.close)
end
end
function Range:contains(cursor)
return cursor >= self.start_cursor and cursor <= self.stop_cursor
end
function Range:query_option(name)
return Options.query(name, self.pairing)
end
function Range:is_wrapped()
2024-12-31 10:05:01 -08:00
for _, param in ipairs(self.params) do
if param:is_wrapped() then
return true
end
end
return false
2024-12-30 20:31:50 -08:00
end
function Range.__lt(range_1, range_2)
local cursor = Cursor:get_current()
local row_diff1 = range_1.start_cursor.row - cursor.row
local col_diff1 = range_1.start_cursor.col - cursor.col
local row_diff2 = range_2.start_cursor.row - cursor.row
local col_diff2 = range_2.start_cursor.col - cursor.col
if row_diff1 < row_diff2 then
return false
elseif row_diff1 > row_diff2 then
return true
elseif col_diff1 < col_diff2 then
return false
elseif col_diff1 > col_diff2 then
return true
else
return true
end
end
return {
find_closest = find_closest,
find_at_cursor = find_at_cursor,
}