2022-10-13 08:53:17 +00:00
|
|
|
--!strict
|
2023-07-08 03:01:02 +00:00
|
|
|
local Core = script.Parent.Parent
|
2022-10-13 08:53:17 +00:00
|
|
|
|
|
|
|
local Util = {
|
|
|
|
GlobalTimer = 0,
|
2023-07-08 23:42:28 +00:00
|
|
|
Scale = 1 / 20,
|
2022-10-13 08:53:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
local rayParams = RaycastParams.new()
|
2023-07-08 03:01:02 +00:00
|
|
|
rayParams.RespectCanCollide = true
|
|
|
|
rayParams.IgnoreWater = true
|
2022-10-13 08:53:17 +00:00
|
|
|
|
|
|
|
local SHORT_TO_RAD = (2 * math.pi) / 0x10000
|
|
|
|
local VECTOR3_XZ = Vector3.one - Vector3.yAxis
|
|
|
|
|
2023-07-12 06:11:34 +00:00
|
|
|
local TweenService = game:GetService("TweenService")
|
|
|
|
local fadeOut = TweenInfo.new(0.5)
|
|
|
|
|
2023-07-18 00:19:03 +00:00
|
|
|
local waterPlane = Instance.new("BoxHandleAdornment")
|
|
|
|
waterPlane.Size = Vector3.new(48, 0, 48)
|
|
|
|
waterPlane.Adornee = workspace.Terrain
|
|
|
|
waterPlane.Transparency = 0.5
|
|
|
|
waterPlane.Name = "WaterPlane"
|
|
|
|
|
|
|
|
local focalPlane = waterPlane:Clone()
|
|
|
|
focalPlane.Size = Vector3.new(4, 0, 4)
|
|
|
|
focalPlane.Color3 = Color3.new(1, 0, 1)
|
|
|
|
focalPlane.Name = "FocalPlane"
|
|
|
|
focalPlane.Transparency = 0.1
|
|
|
|
focalPlane.Parent = waterPlane
|
|
|
|
|
2022-10-13 08:53:17 +00:00
|
|
|
local CARDINAL = {
|
|
|
|
-Vector3.xAxis,
|
|
|
|
-Vector3.zAxis,
|
|
|
|
Vector3.xAxis,
|
|
|
|
Vector3.zAxis,
|
|
|
|
}
|
|
|
|
|
2023-07-18 00:19:03 +00:00
|
|
|
local CONSTRUCTORS = {
|
|
|
|
Vector3 = Vector3.new,
|
|
|
|
Vector3int16 = Vector3int16.new,
|
|
|
|
}
|
2022-10-13 08:53:17 +00:00
|
|
|
|
2023-07-18 00:19:03 +00:00
|
|
|
-- stylua: ignore
|
|
|
|
local function vectorModifier(getArgs: (Vector3 | Vector3int16, number) -> (number, number, number)):
|
|
|
|
((vec: Vector3, value: number) -> Vector3) &
|
|
|
|
((vec: Vector3int16, value: number) -> Vector3int16)
|
2022-10-13 08:53:17 +00:00
|
|
|
|
2023-07-18 00:19:03 +00:00
|
|
|
return function (vector, new)
|
|
|
|
local constructor = CONSTRUCTORS[typeof(vector)]
|
|
|
|
return constructor(getArgs(vector, new))
|
|
|
|
end
|
2022-10-13 08:53:17 +00:00
|
|
|
end
|
|
|
|
|
2023-07-18 00:19:03 +00:00
|
|
|
Util.SetX = vectorModifier(function(vector, x)
|
|
|
|
return x, vector.Y, vector.Z
|
|
|
|
end)
|
2022-10-13 08:53:17 +00:00
|
|
|
|
2023-07-18 00:19:03 +00:00
|
|
|
Util.SetY = vectorModifier(function(vector, y)
|
|
|
|
return vector.X, y, vector.Z
|
|
|
|
end)
|
2022-10-13 08:53:17 +00:00
|
|
|
|
2023-07-18 00:19:03 +00:00
|
|
|
Util.SetZ = vectorModifier(function(vector, z)
|
|
|
|
return vector.X, vector.Y, z
|
|
|
|
end)
|
2022-10-13 08:53:17 +00:00
|
|
|
|
|
|
|
function Util.ToRoblox(v: Vector3)
|
|
|
|
return v * Util.Scale
|
|
|
|
end
|
|
|
|
|
|
|
|
function Util.ToSM64(v: Vector3)
|
|
|
|
return v / Util.Scale
|
|
|
|
end
|
|
|
|
|
|
|
|
function Util.ToEulerAngles(v: Vector3int16): Vector3
|
|
|
|
return Vector3.new(v.X, v.Y, v.Z) * SHORT_TO_RAD
|
|
|
|
end
|
|
|
|
|
|
|
|
function Util.ToRotation(v: Vector3int16): CFrame
|
|
|
|
local angle = Util.ToEulerAngles(v)
|
2023-07-08 03:01:02 +00:00
|
|
|
|
2023-07-18 00:19:03 +00:00
|
|
|
-- stylua: ignore
|
2023-07-08 03:01:02 +00:00
|
|
|
local matrix = CFrame.fromAxisAngle(Vector3.yAxis, angle.Y)
|
2023-07-18 00:19:03 +00:00
|
|
|
* CFrame.fromAxisAngle(Vector3.xAxis, -angle.X)
|
|
|
|
* CFrame.fromAxisAngle(Vector3.zAxis, -angle.Z)
|
2022-10-13 08:53:17 +00:00
|
|
|
|
|
|
|
return matrix
|
|
|
|
end
|
|
|
|
|
2023-07-18 00:19:03 +00:00
|
|
|
function Util.DebugWater(waterLevel: number)
|
|
|
|
if script:GetAttribute("Debug") then
|
|
|
|
local robloxLevel = (waterLevel * Util.Scale) + 0.01
|
|
|
|
local focus = workspace.CurrentCamera.Focus
|
|
|
|
|
|
|
|
local x = math.floor(focus.X / 4) * 4
|
|
|
|
local z = math.floor(focus.Z / 4) * 4
|
|
|
|
|
|
|
|
local cf = CFrame.new(x, robloxLevel, z)
|
|
|
|
waterPlane.Parent = script
|
|
|
|
|
|
|
|
focalPlane.CFrame = cf
|
|
|
|
waterPlane.CFrame = cf
|
|
|
|
else
|
|
|
|
waterPlane.Parent = nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2023-07-08 03:01:02 +00:00
|
|
|
function Util.Raycast(pos: Vector3, dir: Vector3, maybeParams: RaycastParams?, worldRoot: WorldRoot?): RaycastResult?
|
2022-10-13 08:53:17 +00:00
|
|
|
local root = worldRoot or workspace
|
2023-07-08 03:01:02 +00:00
|
|
|
local params = maybeParams or rayParams
|
2023-07-12 06:11:34 +00:00
|
|
|
local result = root:Raycast(pos, dir, params)
|
2022-10-13 08:53:17 +00:00
|
|
|
|
|
|
|
if script:GetAttribute("Debug") then
|
|
|
|
local color = Color3.new(result and 0 or 1, result and 1 or 0, 0)
|
|
|
|
|
|
|
|
local line = Instance.new("LineHandleAdornment")
|
|
|
|
line.CFrame = CFrame.new(pos, pos + dir)
|
|
|
|
line.Length = dir.Magnitude
|
|
|
|
line.Thickness = 3
|
|
|
|
line.Color3 = color
|
|
|
|
line.Adornee = workspace.Terrain
|
|
|
|
line.Parent = workspace.Terrain
|
|
|
|
|
2023-07-12 06:11:34 +00:00
|
|
|
local tween = TweenService:Create(line, fadeOut, {
|
|
|
|
Transparency = 1,
|
|
|
|
})
|
|
|
|
|
|
|
|
tween:Play()
|
|
|
|
task.delay(fadeOut.Time, line.Destroy, line)
|
2022-10-13 08:53:17 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
return result
|
|
|
|
end
|
|
|
|
|
2023-07-18 00:19:03 +00:00
|
|
|
-- stylua: ignore
|
|
|
|
function Util.RaycastSM64(pos: Vector3, dir: Vector3, maybeParams: RaycastParams?, worldRoot: WorldRoot?): RaycastResult?
|
|
|
|
local result: RaycastResult? = Util.Raycast(pos * Util.Scale, dir * Util.Scale, maybeParams or rayParams, worldRoot)
|
2022-10-13 08:53:17 +00:00
|
|
|
|
|
|
|
if result then
|
|
|
|
-- Cast back to SM64 unit scale.
|
|
|
|
result = {
|
|
|
|
Normal = result.Normal,
|
|
|
|
Material = result.Material,
|
|
|
|
Instance = result.Instance,
|
|
|
|
Distance = result.Distance / Util.Scale,
|
|
|
|
Position = result.Position / Util.Scale,
|
|
|
|
} :: any
|
|
|
|
end
|
|
|
|
|
|
|
|
return result
|
|
|
|
end
|
|
|
|
|
|
|
|
function Util.FindFloor(pos: Vector3): (number, RaycastResult?)
|
2022-11-15 17:56:31 +00:00
|
|
|
local newPos = pos
|
2023-07-08 03:01:02 +00:00
|
|
|
local height = -11000
|
2022-10-13 08:53:17 +00:00
|
|
|
|
2023-07-09 01:44:44 +00:00
|
|
|
if Core:GetAttribute("TruncateBounds") then
|
2022-11-15 17:56:31 +00:00
|
|
|
local trunc = Vector3int16.new(pos.X, pos.Y, pos.Z)
|
|
|
|
|
|
|
|
if math.abs(trunc.X) >= 0x2000 then
|
2023-07-08 03:01:02 +00:00
|
|
|
return height, nil
|
2022-11-15 17:56:31 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
if math.abs(trunc.Z) >= 0x2000 then
|
2023-07-08 03:01:02 +00:00
|
|
|
return height, nil
|
2022-11-15 17:56:31 +00:00
|
|
|
end
|
2022-10-13 08:53:17 +00:00
|
|
|
|
2022-11-15 17:56:31 +00:00
|
|
|
newPos = Vector3.new(trunc.X, trunc.Y, trunc.Z)
|
2022-10-13 08:53:17 +00:00
|
|
|
end
|
|
|
|
|
2023-07-18 00:19:03 +00:00
|
|
|
local result = Util.RaycastSM64(newPos + (Vector3.yAxis * 100), -Vector3.yAxis * 10000, rayParams)
|
2022-10-13 08:53:17 +00:00
|
|
|
|
|
|
|
if result then
|
2023-07-08 03:01:02 +00:00
|
|
|
height = Util.SignedShort(result.Position.Y)
|
2022-10-13 08:53:17 +00:00
|
|
|
result.Position = Vector3.new(pos.X, height, pos.Z)
|
|
|
|
end
|
2023-07-12 06:11:34 +00:00
|
|
|
|
|
|
|
return height, result
|
2022-10-13 08:53:17 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
function Util.FindCeil(pos: Vector3, height: number?): (number, RaycastResult?)
|
2023-07-12 06:11:34 +00:00
|
|
|
local newHeight = 10000
|
|
|
|
|
|
|
|
if Core:GetAttribute("TruncateBounds") then
|
|
|
|
local trunc = Vector3int16.new(pos.X, pos.Y, pos.Z)
|
|
|
|
|
|
|
|
if math.abs(trunc.X) >= 0x2000 then
|
|
|
|
return newHeight, nil
|
|
|
|
end
|
|
|
|
|
|
|
|
if math.abs(trunc.Z) >= 0x2000 then
|
|
|
|
return newHeight, nil
|
|
|
|
end
|
|
|
|
|
|
|
|
pos = Vector3.new(trunc.X, trunc.Y, trunc.Z)
|
|
|
|
end
|
|
|
|
|
2023-07-08 03:01:02 +00:00
|
|
|
local head = Vector3.new(pos.X, (height or pos.Y) + 80, pos.Z)
|
2023-07-18 00:19:03 +00:00
|
|
|
local result = Util.RaycastSM64(head, Vector3.yAxis * 10000, rayParams)
|
2022-10-13 08:53:17 +00:00
|
|
|
|
|
|
|
if result then
|
2023-07-12 06:11:34 +00:00
|
|
|
newHeight = result.Position.Y
|
2022-10-13 08:53:17 +00:00
|
|
|
end
|
2023-07-12 06:11:34 +00:00
|
|
|
|
|
|
|
return newHeight, result
|
2022-10-13 08:53:17 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
function Util.FindWallCollisions(pos: Vector3, offset: number, radius: number): (Vector3, RaycastResult?)
|
|
|
|
local origin = pos + Vector3.new(0, offset, 0)
|
|
|
|
local lastWall: RaycastResult?
|
|
|
|
local disp = Vector3.zero
|
|
|
|
|
|
|
|
for i, dir in CARDINAL do
|
|
|
|
local contact = Util.RaycastSM64(origin, dir * radius)
|
|
|
|
|
|
|
|
if contact then
|
|
|
|
local normal = contact.Normal
|
|
|
|
|
|
|
|
if math.abs(normal.Y) < 0.01 then
|
|
|
|
local surface = contact.Position
|
2023-07-08 03:01:02 +00:00
|
|
|
local move = (surface - pos) * VECTOR3_XZ
|
|
|
|
local dist = move.Magnitude
|
2022-10-13 08:53:17 +00:00
|
|
|
|
|
|
|
if dist < radius then
|
|
|
|
disp += (contact.Normal * VECTOR3_XZ) * (radius - dist)
|
|
|
|
lastWall = contact
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
return pos + disp, lastWall
|
|
|
|
end
|
|
|
|
|
|
|
|
function Util.SignedShort(x: number)
|
|
|
|
return -0x8000 + math.floor((x + 0x8000) % 0x10000)
|
|
|
|
end
|
|
|
|
|
|
|
|
function Util.SignedInt(x: number)
|
|
|
|
return -0x80000000 + math.floor(x + 0x80000000) % 0x100000000
|
|
|
|
end
|
|
|
|
|
|
|
|
function Util.ApproachFloat(current: number, target: number, inc: number, dec: number?): number
|
|
|
|
if dec == nil then
|
|
|
|
dec = inc
|
|
|
|
end
|
|
|
|
|
|
|
|
assert(dec)
|
|
|
|
|
|
|
|
if current < target then
|
|
|
|
current = math.min(target, current + inc)
|
|
|
|
else
|
|
|
|
current = math.max(target, current - dec)
|
|
|
|
end
|
|
|
|
|
|
|
|
return current
|
|
|
|
end
|
|
|
|
|
|
|
|
function Util.ApproachInt(current: number, target: number, inc: number, dec: number?): number
|
|
|
|
if dec == nil then
|
|
|
|
dec = inc
|
|
|
|
end
|
|
|
|
|
|
|
|
assert(dec)
|
|
|
|
|
|
|
|
if current < target then
|
|
|
|
current = Util.SignedInt(current + inc)
|
|
|
|
current = math.min(target, current)
|
|
|
|
else
|
|
|
|
current = Util.SignedInt(current - dec)
|
|
|
|
current = math.max(target, current)
|
|
|
|
end
|
|
|
|
|
|
|
|
return Util.SignedInt(current)
|
|
|
|
end
|
|
|
|
|
|
|
|
function Util.Sins(short: number): number
|
2023-07-08 03:01:02 +00:00
|
|
|
local value = Util.SignedShort(short)
|
|
|
|
value = math.floor(value / 16) * 16
|
|
|
|
|
|
|
|
return math.sin(value * SHORT_TO_RAD)
|
2022-10-13 08:53:17 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
function Util.Coss(short: number): number
|
2023-07-08 03:01:02 +00:00
|
|
|
local value = Util.SignedShort(short)
|
|
|
|
value = math.floor(value / 16) * 16
|
|
|
|
|
2022-10-13 08:53:17 +00:00
|
|
|
return math.cos(short * SHORT_TO_RAD)
|
|
|
|
end
|
|
|
|
|
|
|
|
local function atan2_lookup(y: number, x: number)
|
2023-07-08 03:01:02 +00:00
|
|
|
local value = math.atan2(y, x) / SHORT_TO_RAD
|
|
|
|
value = math.floor(value / 16) * 16
|
|
|
|
return Util.SignedShort(value)
|
2022-10-13 08:53:17 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
function Util.Atan2s(y: number, x: number): number
|
|
|
|
local ret: number
|
|
|
|
|
|
|
|
if x >= 0 then
|
|
|
|
if y >= 0 then
|
|
|
|
if y >= x then
|
|
|
|
ret = atan2_lookup(x, y)
|
|
|
|
else
|
|
|
|
ret = 0x4000 - atan2_lookup(y, x)
|
|
|
|
end
|
|
|
|
else
|
|
|
|
y = -y
|
|
|
|
|
|
|
|
if y < x then
|
|
|
|
ret = 0x4000 + atan2_lookup(y, x)
|
|
|
|
else
|
|
|
|
ret = 0x8000 - atan2_lookup(x, y)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
else
|
|
|
|
x = -x
|
|
|
|
|
|
|
|
if y < 0 then
|
|
|
|
y = -y
|
|
|
|
|
|
|
|
if y >= x then
|
|
|
|
ret = 0x8000 + atan2_lookup(x, y)
|
|
|
|
else
|
|
|
|
ret = 0xC000 - atan2_lookup(y, x)
|
|
|
|
end
|
|
|
|
else
|
|
|
|
if y < x then
|
|
|
|
ret = 0xC000 + atan2_lookup(y, x)
|
|
|
|
else
|
|
|
|
ret = -atan2_lookup(x, y)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
return Util.SignedShort(ret)
|
|
|
|
end
|
|
|
|
|
2023-07-08 03:01:02 +00:00
|
|
|
return Util
|