1
0
mirror of https://github.com/ComfyFactory/ComfyFactorio.git synced 2025-01-10 00:43:27 +02:00
ComfyFactorio/utils/list_utils.lua

126 lines
2.9 KiB
Lua
Raw Normal View History

2021-03-24 21:14:55 +02:00
--luacheck: ignore
2018-09-19 06:51:25 +02:00
local function assert_argument_valid(a, arg_type)
2021-03-24 17:46:00 +02:00
arg_type = arg_type or 'table'
if type(a) ~= arg_type then
error("bad argument #1 to '" .. debug.getinfo(2, 'n').name .. "' (table expected, got " .. type(a) .. ')', 3)
end
2018-09-19 06:51:25 +02:00
end
table.remove_element = function(t, element)
2021-03-24 17:46:00 +02:00
assert_argument_valid(t)
for k, v in pairs(t) do
if v == element then
table.remove(t, k)
break
end
2018-09-19 06:51:25 +02:00
end
end
2021-03-24 17:46:00 +02:00
table.add_all = function(t1, t2)
assert_argument_valid(t1)
assert_argument_valid(t2)
for k, v in pairs(t2) do
if tonumber(k) then
table.insert(t1, v)
else
t1[k] = v
end
2018-09-19 06:51:25 +02:00
end
end
table.size = function(t)
2021-03-24 17:46:00 +02:00
assert_argument_valid(t)
local size = 0
for _, _ in pairs(t) do
size = size + 1
end
return size
2018-09-19 06:51:25 +02:00
end
table.index_of = function(t, e)
2021-03-24 17:46:00 +02:00
assert_argument_valid(t)
local i = 1
for _, v in pairs(t) do
if v == e then
return i
end
i = i + 1
2018-09-19 06:51:25 +02:00
end
2021-03-24 17:46:00 +02:00
return -1
2018-09-19 06:51:25 +02:00
end
table.contains = function(t, e)
2021-03-24 17:46:00 +02:00
assert_argument_valid(t)
return table.index_of(t, e) > -1
2018-09-19 06:51:25 +02:00
end
2021-03-24 17:46:00 +02:00
table.set = function(t, index, element)
assert_argument_valid(t)
assert_argument_valid(index, 'number')
local i = 1
for k, v in pairs(t) do
if i == index then
t[k] = element
return nil
end
i = i + 1
2018-09-19 06:51:25 +02:00
end
2021-03-24 17:46:00 +02:00
error('Index out of bounds', 2)
2018-09-19 06:51:25 +02:00
end
2021-03-24 17:46:00 +02:00
table.get = function(t, index)
assert_argument_valid(t)
assert_argument_valid(index, 'number')
local i = 1
for k, v in pairs(t) do
if i == index then
return t[k]
end
i = i + 1
2018-09-19 06:51:25 +02:00
end
2021-03-24 17:46:00 +02:00
error('Index out of bounds', 2)
2018-09-19 06:51:25 +02:00
end
--[[
2021-03-24 17:46:00 +02:00
Returns the index where t[index] == target.
If there is no such index, returns a negative vaule such that bit32.bnot(value) is
2018-09-19 06:51:25 +02:00
the index that the vaule should be inserted to keep the list ordered.
2021-03-24 17:46:00 +02:00
t must be a list in ascending order for the return value to be valid.
2018-09-19 06:51:25 +02:00
Usage example:
local t = {1,3,5,7,9}
local x = 5
2021-03-24 17:46:00 +02:00
local index = table.binary_search(t, x)
if index < 0 then
game.print("value not found, smallest index where t[index] > x is: " .. bit32.bnot(index))
2018-09-19 06:51:25 +02:00
else
game.print("value found at index: " .. index)
2021-03-24 17:46:00 +02:00
end
2018-09-19 06:51:25 +02:00
]]
table.binary_search = function(t, target)
2021-03-24 17:46:00 +02:00
--For some reason bit32.bnot doesn't return negative numbers so I'm using ~x = -1 - x instead.
2018-09-19 06:51:25 +02:00
assert_argument_valid(t)
2021-03-24 17:46:00 +02:00
assert_argument_valid(target, 'number')
2018-09-19 06:51:25 +02:00
local lower = 1
local upper = #t
2021-03-24 17:46:00 +02:00
2018-09-19 06:51:25 +02:00
if upper == 0 then
2021-03-24 17:46:00 +02:00
return -2 -- ~1
2018-09-19 06:51:25 +02:00
end
2021-03-24 17:46:00 +02:00
repeat
local mid = math.floor((lower + upper) / 2)
local value = t[mid]
if value == target then
return mid
elseif value < target then
lower = mid + 1
else
upper = mid - 1
end
2018-09-19 06:51:25 +02:00
until lower > upper
2021-03-24 17:46:00 +02:00
2018-09-19 06:51:25 +02:00
return -1 - lower -- ~lower
2021-03-24 17:46:00 +02:00
end