2019-02-15 20:17:30 -08:00
|
|
|
if mathUtilsG then
|
|
|
|
return mathUtilsG
|
|
|
|
end
|
2016-10-14 17:00:18 -07:00
|
|
|
local mathUtils = {}
|
|
|
|
|
|
|
|
-- imports
|
|
|
|
|
|
|
|
local constants = require("Constants")
|
|
|
|
|
|
|
|
-- constants
|
|
|
|
|
|
|
|
local TICKS_A_MINUTE = constants.TICKS_A_MINUTE
|
|
|
|
|
|
|
|
-- imported functions
|
|
|
|
|
2016-11-03 16:51:35 -07:00
|
|
|
local mSqrt = math.sqrt
|
|
|
|
local mLog10 = math.log10
|
2016-10-14 17:00:18 -07:00
|
|
|
|
2017-06-30 21:36:23 -07:00
|
|
|
local mRandom = math.random
|
2018-01-13 16:12:09 -08:00
|
|
|
local mFloor = math.floor
|
2018-01-25 21:13:47 -08:00
|
|
|
local mAbs = math.abs
|
2017-06-30 21:36:23 -07:00
|
|
|
|
2017-06-12 20:16:43 -07:00
|
|
|
-- module code
|
|
|
|
|
2018-01-20 23:42:47 -08:00
|
|
|
function mathUtils.roundToFloor(number, multiple)
|
|
|
|
return mFloor(number / multiple) * multiple
|
|
|
|
end
|
|
|
|
|
2016-10-14 17:00:18 -07:00
|
|
|
function mathUtils.roundToNearest(number, multiple)
|
|
|
|
local num = number + (multiple * 0.5)
|
|
|
|
return num - (num % multiple)
|
|
|
|
end
|
|
|
|
|
|
|
|
function mathUtils.randomTickEvent(tick, low, high)
|
2019-04-24 23:13:22 -07:00
|
|
|
local range = high - low
|
|
|
|
local minutesToTick = (range * mRandom()) + low
|
2019-11-29 16:49:22 -08:00
|
|
|
return tick + mathUtils.roundToNearest(TICKS_A_MINUTE * minutesToTick, 1)
|
|
|
|
end
|
|
|
|
|
|
|
|
function mathUtils.distort(xorRandom, num, stdDev, min, max)
|
|
|
|
local min = min or num * 0.85
|
|
|
|
local max = max or num * 1.30
|
|
|
|
local sd = stdDev or 0.30
|
|
|
|
if (num < 0) then
|
|
|
|
local t = min
|
|
|
|
min = max
|
|
|
|
max = t
|
|
|
|
end
|
|
|
|
return mathUtils.roundToNearest(mathUtils.gaussianRandomRangeRG(num, num * sd, min, max, xorRandom), 0.01)
|
2016-10-14 17:00:18 -07:00
|
|
|
end
|
|
|
|
|
2018-02-14 00:28:42 -08:00
|
|
|
function mathUtils.linearInterpolation(percent, min, max)
|
|
|
|
return ((max - min) * percent) + min
|
|
|
|
end
|
|
|
|
|
2018-01-13 16:12:09 -08:00
|
|
|
function mathUtils.xorRandom(state)
|
|
|
|
local xor = bit32.bxor
|
|
|
|
local lshift = bit32.lshift
|
|
|
|
local rshift = bit32.rshift
|
2019-02-15 20:17:30 -08:00
|
|
|
|
2018-01-13 16:12:09 -08:00
|
|
|
state = state + 21594771
|
|
|
|
|
|
|
|
return function()
|
2019-10-19 14:04:38 -07:00
|
|
|
state = xor(state, lshift(state, 13))
|
|
|
|
state = xor(state, rshift(state, 17))
|
|
|
|
state = xor(state, lshift(state, 5))
|
|
|
|
state = state % 2147483647
|
|
|
|
return state * 4.65661287525e-10
|
2018-01-13 16:12:09 -08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-02-18 16:21:18 -08:00
|
|
|
function mathUtils.linearInterpolation(percent, min, max)
|
|
|
|
return ((max - min) * percent) + min
|
|
|
|
end
|
|
|
|
|
2016-11-03 16:51:35 -07:00
|
|
|
--[[
|
|
|
|
Used for gaussian random numbers
|
|
|
|
--]]
|
2018-01-13 16:12:09 -08:00
|
|
|
function mathUtils.gaussianRandom(mean, std_dev)
|
|
|
|
-- marsagliaPolarMethod
|
2019-02-15 20:17:30 -08:00
|
|
|
local iid1
|
|
|
|
local iid2
|
|
|
|
local q
|
2016-11-03 16:51:35 -07:00
|
|
|
repeat
|
2019-10-19 14:04:38 -07:00
|
|
|
iid1 = 2 * mRandom() + -1
|
|
|
|
iid2 = 2 * mRandom() + -1
|
|
|
|
q = (iid1 * iid1) + (iid2 * iid2)
|
2016-11-03 16:51:35 -07:00
|
|
|
until (q ~= 0) and (q < 1)
|
|
|
|
local s = mSqrt((-2 * mLog10(q)) / q)
|
2018-01-13 16:12:09 -08:00
|
|
|
local v = iid1 * s
|
2019-02-15 20:17:30 -08:00
|
|
|
|
2018-01-13 16:12:09 -08:00
|
|
|
return mean + (v * std_dev)
|
2016-11-03 16:51:35 -07:00
|
|
|
end
|
|
|
|
|
2018-01-13 16:12:09 -08:00
|
|
|
function mathUtils.gaussianRandomRange(mean, std_dev, min, max)
|
2019-08-16 18:52:16 -07:00
|
|
|
if (min >= max) then
|
2019-10-19 14:04:38 -07:00
|
|
|
return min
|
2018-01-13 16:12:09 -08:00
|
|
|
end
|
|
|
|
local r
|
|
|
|
repeat
|
2019-10-19 14:04:38 -07:00
|
|
|
local iid1
|
|
|
|
local iid2
|
|
|
|
local q
|
|
|
|
repeat
|
|
|
|
iid1 = 2 * mRandom() + -1
|
|
|
|
iid2 = 2 * mRandom() + -1
|
|
|
|
q = (iid1 * iid1) + (iid2 * iid2)
|
|
|
|
until (q ~= 0) and (q < 1)
|
|
|
|
local s = mSqrt((-2 * mLog10(q)) / q)
|
|
|
|
local v = iid1 * s
|
|
|
|
|
|
|
|
r = mean + (v * std_dev)
|
2018-01-13 16:12:09 -08:00
|
|
|
until (r >= min) and (r <= max)
|
|
|
|
return r
|
2016-11-03 16:51:35 -07:00
|
|
|
end
|
|
|
|
|
2019-02-15 20:17:30 -08:00
|
|
|
function mathUtils.gaussianRandomRG(mean, std_dev, rg)
|
2018-01-13 16:12:09 -08:00
|
|
|
-- marsagliaPolarMethod
|
2019-02-15 20:17:30 -08:00
|
|
|
local iid1
|
|
|
|
local iid2
|
|
|
|
local q
|
2016-11-03 16:51:35 -07:00
|
|
|
repeat
|
2019-10-19 14:04:38 -07:00
|
|
|
iid1 = 2 * rg() + -1
|
|
|
|
iid2 = 2 * rg() + -1
|
|
|
|
q = (iid1 * iid1) + (iid2 * iid2)
|
2018-01-13 16:12:09 -08:00
|
|
|
until (q ~= 0) and (q < 1)
|
|
|
|
local s = mSqrt((-2 * mLog10(q)) / q)
|
|
|
|
local v = iid1 * s
|
2019-02-15 20:17:30 -08:00
|
|
|
|
2018-01-13 16:12:09 -08:00
|
|
|
return mean + (v * std_dev)
|
|
|
|
end
|
|
|
|
|
|
|
|
function mathUtils.gaussianRandomRangeRG(mean, std_dev, min, max, rg)
|
|
|
|
local r
|
2019-08-16 18:52:16 -07:00
|
|
|
if (min >= max) then
|
2019-10-19 14:04:38 -07:00
|
|
|
return min
|
2018-01-13 16:12:09 -08:00
|
|
|
end
|
|
|
|
repeat
|
2019-10-19 14:04:38 -07:00
|
|
|
local iid1
|
|
|
|
local iid2
|
|
|
|
local q
|
|
|
|
repeat
|
|
|
|
iid1 = 2 * rg() + -1
|
|
|
|
iid2 = 2 * rg() + -1
|
|
|
|
q = (iid1 * iid1) + (iid2 * iid2)
|
|
|
|
until (q ~= 0) and (q < 1)
|
|
|
|
local s = mSqrt((-2 * mLog10(q)) / q)
|
|
|
|
local v = iid1 * s
|
|
|
|
r = mean + (v * std_dev)
|
2018-01-13 16:12:09 -08:00
|
|
|
until (r >= min) and (r <= max)
|
|
|
|
return r
|
2016-11-03 16:51:35 -07:00
|
|
|
end
|
|
|
|
|
2017-06-12 20:16:43 -07:00
|
|
|
function mathUtils.euclideanDistanceNamed(p1, p2)
|
|
|
|
local xs = p1.x - p2.x
|
|
|
|
local ys = p1.y - p2.y
|
|
|
|
return ((xs * xs) + (ys * ys)) ^ 0.5
|
|
|
|
end
|
|
|
|
|
|
|
|
function mathUtils.euclideanDistancePoints(x1, y1, x2, y2)
|
|
|
|
local xs = x1 - x2
|
|
|
|
local ys = y1 - y2
|
|
|
|
return ((xs * xs) + (ys * ys)) ^ 0.5
|
|
|
|
end
|
|
|
|
|
2018-01-25 21:13:47 -08:00
|
|
|
function mathUtils.mahattenDistancePoints(x1, y1, x2, y2)
|
|
|
|
local xs = x1 - x2
|
|
|
|
local ys = y1 - y2
|
|
|
|
return mAbs(xs + ys)
|
|
|
|
end
|
|
|
|
|
2017-06-12 20:16:43 -07:00
|
|
|
function mathUtils.euclideanDistanceArray(p1, p2)
|
|
|
|
local xs = p1[1] - p2[1]
|
|
|
|
local ys = p1[2] - p2[2]
|
|
|
|
return ((xs * xs) + (ys * ys)) ^ 0.5
|
|
|
|
end
|
|
|
|
|
2019-10-20 17:53:16 -07:00
|
|
|
function mathUtils.distortPosition(position, size)
|
2017-06-12 20:16:43 -07:00
|
|
|
local xDistort = mathUtils.gaussianRandomRange(1, 0.5, 0, 2) - 1
|
|
|
|
local yDistort = mathUtils.gaussianRandomRange(1, 0.5, 0, 2) - 1
|
2019-10-20 17:53:16 -07:00
|
|
|
position.x = position.x + (xDistort * size)
|
|
|
|
position.y = position.y + (yDistort * size)
|
2019-04-07 22:22:02 -07:00
|
|
|
return position
|
2017-06-12 20:16:43 -07:00
|
|
|
end
|
|
|
|
|
2019-02-15 20:17:30 -08:00
|
|
|
mathUtilsG = mathUtils
|
2016-10-14 17:00:18 -07:00
|
|
|
return mathUtils
|