1
0
Fork 0
mirror of https://github.com/Reuh/anselme.git synced 2025-10-27 16:49:31 +00:00
anselme/anselme/stdlib/number.lua

152 lines
6.2 KiB
Lua

--- # Arithmetic and math functions
--
-- Comparaison operators are designed to be chained:
-- ```
-- 1 < 2 < 3
-- // is parsed as
-- (1 < 2) < 3
-- // (1 < 2) returns 2, 2 < 3 returns 3 which is true
-- ```
-- @titlelevel 3
local ast = require("anselme.ast")
local Boolean, Number = ast.Boolean, ast.Number
return {
--- Pi.
{ "pi", Number:new(math.pi) },
{
--- Returns `b` if `a` < `b`, false otherwise.
"_<_", "(a::is number, b::is number)",
function(state, a, b)
if a.number < b.number then return b
else return Boolean:new(false)
end
end
},
--- Returns false.
{ "_<_", "(a::is false, b::is number)", function(state, a, b) return Boolean:new(false) end },
{
--- Returns `b` if `a` <= `b`, false otherwise.
"_<=_", "(a::is number, b::is number)",
function(state, a, b)
if a.number <= b.number then return b
else return Boolean:new(false)
end
end
},
--- Returns false.
{ "_<=_", "(a::is false, b::is number)", function(state, a, b) return Boolean:new(false) end },
{
--- Returns `b` if `a` > `b`, false otherwise.
"_>_", "(a::is number, b::is number)",
function(state, a, b)
if a.number > b.number then return b
else return Boolean:new(false)
end
end
},
--- Returns false.
{ "_>_", "(a::is false, b::is number)", function(state, a, b) return Boolean:new(false) end },
{
--- Returns `b` if `a` >= `b`, false otherwise.
"_>=_", "(a::is number, b::is number)",
function(state, a, b)
if a.number >= b.number then return b
else return Boolean:new(false)
end
end
},
--- Returns false.
{ "_>=_", "(a::is false, b::is number)", function(state, a, b) return Boolean:new(false) end },
--- Returns `a` + `b`.
{ "_+_", "(a::is number, b::is number)", function(state, a, b) return Number:new(a.number + b.number) end },
--- Returns `a` - `b`.
{ "_-_", "(a::is number, b::is number)", function(state, a, b) return Number:new(a.number - b.number) end },
--- Returns `a` * `b`.
{ "_*_", "(a::is number, b::is number)", function(state, a, b) return Number:new(a.number * b.number) end },
--- Returns `a` / `b`.
{ "_/_", "(a::is number, b::is number)", function(state, a, b) return Number:new(a.number / b.number) end },
{
--- Returns the integer division of `a` by `b`.
"div", "(a::is number, b::is number)", function(state, a, b)
local r = a.number / b.number
if r < 0 then
return Number:new(math.ceil(r))
else
return Number:new(math.floor(r))
end
end
},
--- Returns the modulo of `a` by `b`.
{ "_%_", "(a::is number, b::is number)", function(state, a, b) return Number:new(a.number % b.number) end },
--- Returns `a` to the power of `b`.
{ "_^_", "(a::is number, b::is number)", function(state, a, b) return Number:new(a.number ^ b.number) end },
--- Returns the negative of `a`.
{ "-_", "(a::is number)", function(state, a) return Number:new(-a.number) end },
--- Returns `a`.
{ "+_", "(a::is number)", function(state, a) return a end },
--- Returns a random integer number with uniform distribution in [`min`, `max`].
{ "rand", "(min::is number, max::is number)", function(state, min, max) return Number:new(math.random(min.number, max.number)) end },
--- Returns a random integer number with uniform distribution in [1, `max`].
{ "rand", "(max::is number)", function(state, max) return Number:new(math.random(max.number)) end },
--- Returns a random float number with uniform distribution in [0,1).
{ "rand", "()", function(state) return Number:new(math.random()) end },
--- Returns the largest integral value less than or equal to `x`.
{ "floor", "(x::is number)", function(state, x) return Number:new(math.floor(x.number)) end },
--- Returns the smallest integral value greater than or equal to `x`.
{ "ceil", "(x::is number)", function(state, x) return Number:new(math.ceil(x.number)) end },
{
--- Returns `x` rounded to the nearest integer.
-- If `increment` > 1, rounds to the nearest float with `increment` decimals.
"round", "(x::is number, increment=1)",
function(state, x, increment)
local n = x.number / increment.number
if n >= 0 then
return Number:new(math.floor(n + 0.5) * increment.number)
else
return Number:new(math.ceil(n - 0.5) * increment.number)
end
end
},
--- Returns the square root of `x`.
{ "sqrt", "(x::is number)", function(state, x) return Number:new(math.sqrt(x.number)) end },
--- Returns the absolute value of `x`.
{ "abs", "(x::is number)", function(state, x) return Number:new(math.abs(x.number)) end },
--- Returns the exponential of `x`.
{ "exp", "(x::is number)", function(state, x) return Number:new(math.exp(x.number)) end },
--- Returns the natural logarithm of `x`.
{ "log", "(x::is number)", function(state, x) return Number:new(math.log(x.number)) end },
--- Returns the logarithm in base `base` of `x`.
{ "log", "(x::is number, base::is number)", function(state, x, base) return Number:new(math.log(x.number, base.number)) end },
--- Convert `x` from radian to degrees.
{ "deg", "(x::is number)", function(state, x) return Number:new(math.deg(x.number)) end },
--- Convert `x` from degrees to radians.
{ "rad", "(x::is number)", function(state, x) return Number:new(math.rad(x.number)) end },
--- ## Trigonometric functions
--
-- All triginometric functions take and return angles in radians.
--- Returns the cosinus of `x`.
{ "cos", "(x::is number)", function(state, x) return Number:new(math.cos(x.number)) end },
--- Returns the sinus of `x`.
{ "sin", "(x::is number)", function(state, x) return Number:new(math.sin(x.number)) end },
--- Returns the tagent of `x`.
{ "tan", "(x::is number)", function(state, x) return Number:new(math.tan(x.number)) end },
--- Returns the arc cosinus of `x`.
{ "acos", "(x::is number)", function(state, x) return Number:new(math.acos(x.number)) end },
--- Returns the arc sinus of `x`.
{ "asin", "(x::is number)", function(state, x) return Number:new(math.asin(x.number)) end },
--- Returns the arc tangent of `x`.
{ "atan", "(x::is number)", function(state, x) return Number:new(math.atan(x.number)) end },
--- Returns the arc tangent of `x` / `y`, taking the signs of both arguments into account to find the correct quandrant (see [atan2](https://en.wikipedia.org/wiki/Atan2)).
{ "atan", "(y::is number, x::is number)", function(state, y, x) return Number:new((math.atan2 or math.atan)(y.number, x.number)) end },
}