forked from Simnation/Main
146 lines
No EOL
4.2 KiB
Lua
146 lines
No EOL
4.2 KiB
Lua
Math = Math or {}
|
|
|
|
function Math.Clamp(value, min, max)
|
|
return math.min(math.max(value, min), max)
|
|
end
|
|
|
|
function Math.Remap(value, min, max, newMin, newMax)
|
|
return newMin + (value - min) / (max - min) * (newMax - newMin)
|
|
end
|
|
|
|
function Math.PointInRadius(radius)
|
|
local angle = math.rad(math.random(0, 360))
|
|
return vector2(radius * math.cos(angle), radius * math.sin(angle))
|
|
end
|
|
|
|
function Math.Normalize(value, min, max)
|
|
if max == min then return 0 end -- Avoid division by zero
|
|
return (value - min) / (max - min)
|
|
end
|
|
|
|
function Math.Normalize2D(x, y)
|
|
if type(x) == "vector2" then
|
|
x, y = x.x, x.y
|
|
end
|
|
local length = math.sqrt(x*x + y*y)
|
|
return length ~= 0 and vector2(x / length, y / length) or vector2(0, 0)
|
|
end
|
|
|
|
function Math.Normalize3D(x, y, z)
|
|
if type(x) == "vector3" then
|
|
x, y, z = x.x, x.y, x.z
|
|
end
|
|
local length = math.sqrt(x*x + y*y + z*z)
|
|
return length ~= 0 and vector3(x / length, y / length, z / length) or vector3(0, 0, 0)
|
|
end
|
|
|
|
function Math.Normalize4D(x, y, z, w)
|
|
if type(x) == "vector4" then
|
|
x, y, z, w = x.x, x.y, x.z, x.w
|
|
end
|
|
local length = math.sqrt(x*x + y*y + z*z + w*w)
|
|
return length ~= 0 and vector4(x / length, y / length, z / length, w / length) or vector4(0, 0, 0, 0)
|
|
end
|
|
|
|
function Math.DirectionToTarget(fromV3, toV3)
|
|
return Math.Normalize3D(toV3.x - fromV3.x, toV3.y - fromV3.y, toV3.z - fromV3.z)
|
|
end
|
|
|
|
function Deg2Rad(deg)
|
|
return deg * math.pi / 180.0
|
|
end
|
|
|
|
function RotVector(pos, rot)
|
|
local pitch = Deg2Rad(rot.x)
|
|
local roll = Deg2Rad(rot.y)
|
|
local yaw = Deg2Rad(rot.z)
|
|
|
|
local cosY = math.cos(yaw)
|
|
local sinY = math.sin(yaw)
|
|
local cosP = math.cos(pitch)
|
|
local sinP = math.sin(pitch)
|
|
local cosR = math.cos(roll)
|
|
local sinR = math.sin(roll)
|
|
|
|
local m11 = cosY * cosR + sinY * sinP * sinR
|
|
local m12 = sinR * cosP
|
|
local m13 = -sinY * cosR + cosY * sinP * sinR
|
|
|
|
local m21 = -cosY * sinR + sinY * sinP * cosR
|
|
local m22 = cosR * cosP
|
|
local m23 = sinR * sinY + cosY * sinP * cosR
|
|
|
|
local m31 = sinY * cosP
|
|
local m32 = -sinP
|
|
local m33 = cosY * cosP
|
|
|
|
return vector3(pos.x * m11 + pos.y * m21 + pos.z * m31, pos.x * m12 + pos.y * m22 + pos.z * m32, pos.x * m13 + pos.y * m23 + pos.z * m33)
|
|
end
|
|
|
|
|
|
function Math.GetOffsetFromMatrix(position, rotation, offset)
|
|
local rotated = RotVector(offset, rotation)
|
|
print("Rotated: " .. tostring(rotated))
|
|
return position + rotated
|
|
end
|
|
|
|
function Math.InBoundary(pos, boundary)
|
|
if not boundary then return true end
|
|
|
|
local x, y, z = table.unpack(pos)
|
|
|
|
-- Handle legacy min/max boundary format for backwards compatibility
|
|
if boundary.min and boundary.max then
|
|
local minX, minY, minZ = table.unpack(boundary.min)
|
|
local maxX, maxY, maxZ = table.unpack(boundary.max)
|
|
return x >= minX and x <= maxX and y >= minY and y <= maxY and z >= minZ and z <= maxZ
|
|
end
|
|
|
|
-- Handle list of points (polygon boundary)
|
|
if boundary.points and #boundary.points > 0 then
|
|
local points = boundary.points
|
|
local minZ = boundary.minZ or -math.huge
|
|
local maxZ = boundary.maxZ or math.huge
|
|
|
|
-- Check Z bounds first
|
|
if z < minZ or z > maxZ then
|
|
return false
|
|
end
|
|
|
|
-- Point-in-polygon test using ray casting algorithm (improved version)
|
|
local inside = false
|
|
local n = #points
|
|
|
|
for i = 1, n do
|
|
local j = i == n and 1 or i + 1 -- Next point (wrap around)
|
|
|
|
local xi, yi = points[i].x or points[i][1], points[i].y or points[i][2]
|
|
local xj, yj = points[j].x or points[j][1], points[j].y or points[j][2]
|
|
|
|
-- Ensure xi, yi, xj, yj are numbers
|
|
if not (xi and yi and xj and yj) then
|
|
goto continue
|
|
end
|
|
|
|
-- Ray casting test
|
|
if ((yi > y) ~= (yj > y)) then
|
|
-- Calculate intersection point
|
|
local intersect = (xj - xi) * (y - yi) / (yj - yi) + xi
|
|
if x < intersect then
|
|
inside = not inside
|
|
end
|
|
end
|
|
|
|
::continue::
|
|
end
|
|
|
|
return inside
|
|
end
|
|
|
|
-- Fallback to true if boundary format is not recognized
|
|
return true
|
|
end
|
|
|
|
exports('Math', Math)
|
|
|
|
return Math |