1
0
mirror of https://github.com/ComfyFactory/ComfyFactorio.git synced 2025-01-24 03:47:58 +02:00
cogito cffb655300 pp: common: make create_bounding_box_by_points more generic
This function was limited only to objects that had either
'bounding_box' or 'position'  datafield. This changes broades this
limitation.
2020-02-01 10:08:36 +01:00

500 lines
12 KiB
Lua

local public = {}
public.init = function()
if global.this == nil then
global.this = {}
end
end
--[[
rand_range - Return random integer within the range.
@param start - Start range.
@param stop - Stop range.
--]]
public.rand_range = function(start, stop)
if not global.this.rng then
global.this.rng = game.create_random_generator()
end
return global.this.rng(start, stop)
end
--[[
for_bounding_box - Execute function per every position within bb.
@param surf - LuaSurface, that will be given into func.
@param bb - BoundingBox
@param func - User supplied callback that will be executed.
--]]
public.for_bounding_box = function(surf, bb, func)
for x = bb.left_top.x, bb.right_bottom.x do
for y = bb.left_top.y, bb.right_bottom.y do
func(surf, x, y)
end
end
end
--[[
get_axis - Extract axis value from any point format.
@param point - Table with or without explicity axis members.
@param axis - Single character string describing the axis.
--]]
public.get_axis = function(point, axis)
if point.target then
return public.get_axis(point.target, axis)
end
if point.position then
return public.get_axis(point.position, axis)
end
if point[axis] then
return point[axis]
end
if #point ~= 2 then
log("get_axis: invalid point format")
return nil
end
if axis == "x" then
return point[1]
end
return point[2]
end
--[[
get_close_random_position - Gets randomized close position to origin,
@param origin - Position that will be taken as relative
point for calculation.
@param radius - Radius space.
--]]
public.get_close_random_position = function(origin, radius)
local x = public.get_axis(origin, "x")
local y = public.get_axis(origin, "y")
x = public.rand_range(x - radius, x + radius)
y = public.rand_range(y - radius, y + radius)
return { x = x, y = y }
end
--[[
get_distance - Returns distance in tiles between 2 points.
@param a - Position, first point.
@param b - Position, second point.
--]]
public.get_distance = function(a, b)
local h = (public.get_axis(a, "x") - public.get_axis(b, "x")) ^ 2
local v = (public.get_axis(a, "y") - public.get_axis(b, "y")) ^ 2
return math.sqrt(h + v)
end
--[[
point_in_bounding_box - Check whatever point is within bb.
@param point - Position
@param bb - BoundingBox
--]]
public.point_in_bounding_box = function(point, bb)
local x = public.get_axis(point, "x")
local y = public.get_axis(point, "y")
if bb.left_top.x <= x and bb.right_bottom.x >= x and
bb.left_top.y <= y and bb.right_bottom.y >= y then
return true
end
return false
end
public.direction_lookup = {
[-1] = {
[1] = defines.direction.southwest,
[0] = defines.direction.west,
[-1] = defines.direction.northwest,
},
[0] = {
[1] = defines.direction.south,
[-1] = defines.direction.north
},
[1] = {
[1] = defines.direction.southeast,
[0] = defines.direction.east,
[-1] = defines.direction.northeast,
},
}
--[[
get_readable_direction - Return readable direction from point a to b.
@param a - Position A
@param b - Position B
--]]
public.get_readable_direction = function(a, b)
local a_x = public.get_axis(a, "x")
local a_y = public.get_axis(a, "y")
local b_x = public.get_axis(b, "x")
local b_y = public.get_axis(b, "y")
local h, v
if a_x < b_x then
h = 1
elseif a_x > b_x then
h = -1
else
h = 0
end
if a_y < b_y then
v = 1
elseif a_y > b_y then
v = -1
else
v = 0
end
local mapping = {
[defines.direction.southwest] = "south-west",
[defines.direction.west] = "west",
[defines.direction.northwest] = "north-west",
[defines.direction.south] = "south",
[defines.direction.north] = "north",
[defines.direction.southeast] = "south-east",
[defines.direction.east] = "east",
[defines.direction.northeast] = "north-east",
}
return mapping[public.direction_lookup[h][v]]
end
--[[
create_bounding_box_by_points - Construct a BoundingBox using points
from any array of objects such as bounding boxes.
@param objects - Array of objects.
--]]
public.create_bounding_box_by_points = function(objects)
local box = {
left_top = {
x = public.get_axis(objects[1], "x"),
y = public.get_axis(objects[1], "y")
},
right_bottom = {
x = public.get_axis(objects[1], "x"),
y = public.get_axis(objects[1], "y")
}
}
for i = 2, #objects do
local object = objects[i]
if object.bounding_box then
local bb = object.bounding_box
if box.left_top.x > bb.left_top.x then
box.left_top.x = bb.left_top.x
end
if box.right_bottom.x < bb.right_bottom.x then
box.right_bottom.x = bb.right_bottom.x
end
if box.left_top.y > bb.left_top.y then
box.left_top.y = bb.left_top.y
end
if box.right_bottom.y < bb.right_bottom.y then
box.right_bottom.y = bb.right_bottom.y
end
else
local x = public.get_axis(object, "x")
local y = public.get_axis(object, "y")
if box.left_top.x > x then
box.left_top.x = x
elseif box.right_bottom.x < x then
box.right_bottom.x = x
end
if box.left_top.y > y then
box.left_top.y = y
elseif box.right_bottom.y < y then
box.right_bottom.y = y
end
end
end
box.left_top.x = box.left_top.x - 1
box.left_top.y = box.left_top.y - 1
box.right_bottom.x = box.right_bottom.x + 1
box.right_bottom.y = box.right_bottom.y + 1
return box
end
--[[
merge_bounding_boxes - Merge array of BoundingBox objects into a single
object.
@param bbs - Array of BoundingBox objects.
--]]
public.merge_bounding_boxes = function(bbs)
if bbs == nil then
log("common.merge_bounding_boxes: bbs is nil")
return
end
if #bbs <= 0 then
log("common.merge_bounding_boxes: bbs is empty")
return
end
local box = {
left_top = {
x = bbs[1].left_top.x,
y = bbs[1].left_top.y,
},
right_bottom = {
x = bbs[1].right_bottom.x,
y = bbs[1].right_bottom.y,
}
}
for i = 2, #bbs do
local bb = bbs[i]
if box.left_top.x > bb.left_top.x then
box.left_top.x = bb.left_top.x
end
if box.right_bottom.x < bb.right_bottom.x then
box.right_bottom.x = bb.right_bottom.x
end
if box.left_top.y > bb.left_top.y then
box.left_top.y = bb.left_top.y
end
if box.right_bottom.y < bb.right_bottom.y then
box.right_bottom.y = bb.right_bottom.y
end
end
return box
end
--[[
get_time - Return strigified time of a tick.
@param ticks - Just a ticks.
--]]
public.get_time = function(ticks)
local seconds = math.floor((ticks / 60) % 60)
local minutes = math.floor((ticks / 60 / 60) % 60)
local hours = math.floor(ticks / 60 / 60 / 60)
local time
if hours > 0 then
time = string.format("%02d:%01d:%02d", hours, minutes, seconds)
elseif minutes > 0 then
time = string.format("%02d:%02d", minutes, seconds)
else
time = string.format("00:%02d", seconds)
end
return time
end
--[[
polygon_insert - Append vertex in clockwise order.
@param vertex - Point to insert,
@param vertices - Tables of vertices.
--]]
public.polygon_append_vertex = function(vertices, vertex)
table.insert(vertices, vertex)
local x_avg, y_avg = 0, 0
for _, v in pairs(vertices) do
x_avg = x_avg + public.get_axis(v, "x")
y_avg = y_avg + public.get_axis(v, "y")
end
x_avg = x_avg / #vertices
y_avg = y_avg / #vertices
local delta_x, delta_y, rad1, rad2
for i = 1, #vertices, 1 do
for j = 1, #vertices - i do
local v = vertices[j]
delta_x = public.get_axis(v, "x") - x_avg
delta_y = public.get_axis(v, "y") - y_avg
rad1 = ((math.atan2(delta_x, delta_y) * (180 / 3.14)) + 360) % 360
v = vertices[j + 1]
delta_x = public.get_axis(v, "x") - x_avg
delta_y = public.get_axis(v, "y") - y_avg
rad2 = ((math.atan2(delta_x, delta_y) * (180 / 3.14)) + 360) % 360
if rad1 > rad2 then
vertices[j], vertices[j + 1] = vertices[j + 1], vertices[j]
end
end
end
end
--[[
positions_equal - Checks if given positions are equal.
@param a - Position a
@param b - Position b
--]]
public.positions_equal = function(a, b)
local p1 = public.get_axis(a, "x")
local p2 = public.get_axis(b, "x")
if p1 ~= p2 then
return false
end
p1 = public.get_axis(a, "y")
p2 = public.get_axis(b, "y")
if p1 ~= p2 then
return false
end
return true
end
local function rev(array, index)
if index == nil then
index = 0
end
index = #array - index
return array[index]
end
--[[
deepcopy - Makes a deep copy of an object.
@param orig - Object to copy.
--]]
public.deepcopy = function(orig)
local orig_type = type(orig)
local copy
if orig_type == 'table' then
copy = {}
for orig_key, orig_value in next, orig, nil do
copy[public.deepcopy(orig_key)] = public.deepcopy(orig_value)
end
setmetatable(copy, public.deepcopy(getmetatable(orig)))
else -- number, string, boolean, etc
copy = orig
end
return copy
end
local function convex_hull_turn(a, b, c)
local x1, x2, x3, y1, y2, y3
x1 = public.get_axis(a, "x")
x2 = public.get_axis(b, "x")
y1 = public.get_axis(a, "y")
y2 = public.get_axis(b, "y")
if c then
x3 = public.get_axis(c, "x")
y3 = public.get_axis(c, "y")
return (x2 - x1) * (y3 - y1) - (y2 - y1) * (x3 - x1)
end
return (x1 * y2) - (y1 * x2)
end
--[[
convex_hull - Generate convex hull out of given vertices.
@param vertices - Table of positions.
--]]
public.get_convex_hull = function(_vertices)
if #_vertices == 0 then
return {}
end
local vertices = public.deepcopy(_vertices)
-- Get the lowest point
local v, y1, y2, x1, x2, lowest_index
local lowest = vertices[1]
for i = 2, #vertices do
v = vertices[i]
y1 = public.get_axis(v, "y")
y2 = public.get_axis(lowest, "y")
if y1 < y2 then
lowest = v
lowest_index = i
elseif y1 == y2 then
x1 = public.get_axis(v, "x")
x2 = public.get_axis(lowest, "x")
if x1 < x2 then
lowest = v
lowest_index = i
end
end
end
table.remove(vertices, lowest_index)
x1 = public.get_axis(lowest, "x")
y1 = public.get_axis(lowest, "y")
-- Sort by angle to horizontal axis.
local rad1, rad2, dist1, dist2
local i, j = 1, 1
while i <= #vertices do
while j <= #vertices - i do
v = vertices[j]
x2 = public.get_axis(v, "x")
y2 = public.get_axis(v, "y")
rad1 = (math.atan2(y2 - y1, x2 - x1) * (180/3.14) + 320) % 360
v = vertices[j + 1]
x2 = public.get_axis(v, "x")
y2 = public.get_axis(v, "y")
rad2 = (math.atan2(y2 - y1, x2 - x1) * (180/3.14) + 320) % 360
if rad1 > rad2 then
vertices[j + 1], vertices[j] = vertices[j], vertices[j + 1]
elseif rad1 == rad2 then
dist1 = public.get_distance(lowest, vertices[j])
dist2 = public.get_distance(lowest, vertices[j + 1])
if dist1 > dist2 then
table.remove(vertices, j + 1)
else
table.remove(vertices, j)
end
end
j = j + 1
end
i = i + 1
end
if #vertices <= 3 then
return {}
end
-- Traverse points.
local stack = {
vertices[1],
vertices[2],
vertices[3],
}
local point
for i = 4, #vertices do
point = vertices[i]
while #stack > 1
and convex_hull_turn(point, rev(stack, 1), rev(stack)) >= 0 do
table.remove(stack)
end
table.insert(stack, point)
end
table.insert(stack, lowest)
return stack
end
return public