Module:Array: Difference between revisions

From Ato Wiki
starcitizen>Alistair3149
(Remove logger dependency)
starcitizen>Alistair3149
No edit summary
Line 1: Line 1:
-- Imported from: https://runescape.wiki/w/Module:Array
-- Imported from: https://runescape.wiki/w/Module:Array


-- <nowiki>
local libraryUtil = require('libraryUtil')
local libraryUtil = require('libraryUtil')
local checkType = libraryUtil.checkType
local checkType = libraryUtil.checkType
local checkTypeMulti = libraryUtil.checkTypeMulti
local checkTypeMulti = libraryUtil.checkTypeMulti
local arr = {}


setmetatable(arr, {
---@class Array
    __call = function (_, array)
---@operator call(any[]): Array
        return arr.new(array)
---@operator concat(any[]): Array
    end
---@operator concat(number|string|function): string
---@operator unm: Array
---@operator add(number|number[]|Array): Array
---@operator sub(number|number[]|Array): Array
---@operator mul(number|number[]|Array): Array
---@operator div(number|number[]|Array): Array
---@operator pow(number|number[]|Array): Array
local Array = {
pop = table.remove
}
Array.__index = Array
 
setmetatable(Array, {
__index = table,
__call = function (_, arr)
return Array.new(arr)
end
})
})


function arr.__index(t, k)
-- function Array.__tostring(arr)
    if type(k) == 'table' then
-- -- local dumpObject = require('Module:Logger').dumpObject
        local res = arr.new()
-- require 'log'
        for i = 1, #t do
-- local dumpObject = dumpObject
            res[i] = t[k[i]]
-- local mt = getmetatable(arr)
        end
-- setmetatable(arr, nil)
        return res
-- local str = dumpObject(arr, {clean=true, collapseLimit=100})
    else
-- setmetatable(arr, mt)
        return arr[k]
-- return str
    end
-- end
 
function Array.__concat(lhs, rhs)
if type(lhs) == 'table' and type(rhs) == 'table' then
local res = {}
for i = 1, #lhs do
res[i] = lhs[i]
end
local l = #lhs
for i = 1, #rhs do
res[i + l] = rhs[i]
end
return setmetatable(res, getmetatable(lhs) or getmetatable(rhs))
else
return tostring(lhs) .. tostring(rhs)
end
end
 
function Array.__unm(arr)
return Array.map(arr, function(x) return -x end)
end
 
---@param lhs number|number[]|Array
---@param rhs number|number[]|Array
---@param funName string
---@param opName string
---@param fun fun(lhs: number, rhs: number): number
---@return Array
local function mathTemplate(lhs, rhs, funName, opName, fun)
checkTypeMulti('Module:Array.' .. funName, 1, lhs, {'number', 'table'})
checkTypeMulti('Module:Array.' .. funName, 2, rhs, {'number', 'table'})
local res = {}
 
if type(lhs) == 'number' then
for i = 1, #rhs do
res[i] = fun(lhs, rhs[i])
end
elseif type(rhs) == 'number' then
for i = 1, #lhs do
res[i] = fun(lhs[i], rhs)
end
else
assert(#lhs == #rhs, string.format('Elementwise %s failed because arrays have different sizes (left: %d, right: %d)', opName, #lhs, #rhs))
for i = 1, #lhs do
res[i] = fun(lhs[i], rhs[i])
end
end
 
return setmetatable(res, getmetatable(lhs) or getmetatable(rhs))
end
end


function arr.__tostring(array)
function Array.__add(lhs, rhs)
    setmetatable(array, nil)
return mathTemplate(lhs, rhs, '__add', 'addition', function(x, y) return x + y end)
    local str = mw.logObject( array )
    setmetatable(array, arr)
    return str
end
end


function arr.__concat(lhs, rhs)
function Array.__sub(lhs, rhs)
    if type(lhs) == 'table' and type(rhs) == 'table' then
return mathTemplate(lhs, rhs, '__sub', 'substraction', function(x, y) return x - y end)
        local res = setmetatable({}, getmetatable(lhs) or getmetatable(rhs))
        for i = 1, #lhs do
            res[i] = lhs[i]
        end
        local l = #lhs
        for i = 1, #rhs do
            res[i + l] = rhs[i]
        end
        return res
    else
        return tostring(lhs) .. tostring(rhs)
    end
end
end


function arr.__unm(array)
function Array.__mul(lhs, rhs)
    return arr.map(array, function(x) return -x end)
return mathTemplate(lhs, rhs, '__mul', 'multiplication', function(x, y) return x * y end)
end
end


local function mathTemplate(lhs, rhs, funName, fun)
function Array.__div(lhs, rhs)
    checkTypeMulti('Module:Array.' .. funName, 1, lhs, {'number', 'table'})
return mathTemplate(lhs, rhs, '__div', 'division', function(x, y) return x / y end)
    checkTypeMulti('Module:Array.' .. funName, 2, rhs, {'number', 'table'})
end
    local res = setmetatable({}, getmetatable(lhs) or getmetatable(rhs))


    if type(lhs) == 'number' then
function Array.__pow(lhs, rhs)
        for i = 1, #rhs do
return mathTemplate(lhs, rhs, '__pow', 'exponentiation', function(x, y) return x ^ y end)
            res[i] = fun(lhs, rhs[i])
end
        end
    elseif type(rhs) == 'number' then
        for i = 1, #lhs do
            res[i] = fun(lhs[i], rhs)
        end
    else
        assert(#lhs == #rhs, string.format('Tables are not equal length (lhs=%d, rhs=%d)', #lhs, #rhs))
        for i = 1, #lhs do
            res[i] = fun(lhs[i], rhs[i])
        end
    end


    return res
function Array.__eq(lhs, rhs)
if #lhs ~= #rhs then
return false
end
for i = 1, #lhs do
if lhs[i] ~= rhs[i] then
return false
end
end
return true
end
end


function arr.__add(lhs, rhs)
---Behaviour depends on the value of `fn`:
    return mathTemplate(lhs, rhs, '__add', function(x, y) return x + y end)
---* `nil` - Checks that the array doesn't contain any **false** elements.
---* `fun(elem: any, i?: integer): boolean` - Returns **true** if `fn` returns **true** for every element.
---* `number` | `table` | `boolean` - Checks that all elements in `arr` are equal to this value.
---@param arr any[]
---@param fn? any
---@return boolean
function Array.all(arr, fn)
checkType('Module:Array.all', 1, arr, 'table')
if fn == nil then fn = function(item) return item end end
if type(fn) ~= 'function' then
local val = fn
fn = function(item) return item == val end
end
local i = 1
while arr[i] ~= nil do
---@diagnostic disable-next-line: redundant-parameter
if not fn(arr[i], i) then
return false
end
i = i + 1
end
return true
end
end


function arr.__sub(lhs, rhs)
---Behaviour depends on the value of `fn`:
    return mathTemplate(lhs, rhs, '__sub', function(x, y) return x - y end)
---* `nil` - Checks that the array contains at least one non **false** element.
---* `fun(elem: any, i?: integer): boolean` - Returns **true** if `fn` returns **true** for at least one element.
---* `number` | `table` | `boolean` - Checks that `arr` contains this value.
---@param arr any[]
---@param fn? any
---@return boolean
function Array.any(arr, fn)
checkType('Module:Array.any', 1, arr, 'table')
if fn == nil then fn = function(item) return item end end
if type(fn) ~= 'function' then
local val = fn
fn = function(item) return item == val end
end
local i = 1
while arr[i] ~= nil do
---@diagnostic disable-next-line: redundant-parameter
if fn(arr[i], i) then
return true
end
i = i + 1
end
return false
end
end


function arr.__mul(lhs, rhs)
---Recursively removes all metatables.
    return mathTemplate(lhs, rhs, '__mul', function(x, y) return x * y end)
---@param arr any[]
---@return any[]
function Array.clean(arr)
checkType('Module:Array.clean', 1, arr, 'table')
for i = 1, #arr do
if type(arr[i]) == 'table' then
Array.clean(arr[i])
end
end
setmetatable(arr, nil)
return arr
end
end


function arr.__div(lhs, rhs)
---Make a copy of the input table. Preserves metatables.
    return mathTemplate(lhs, rhs, '__div', function(x, y) return x / y end)
---@generic T: any[]
---@param arr T
---@param deep? boolean # Recursively clone subtables if **true**.
---@return T
function Array.clone(arr, deep)
checkType('Module:Array.clone', 1, arr, 'table')
checkType('Module:Array.clone', 2, deep, 'boolean', true)
local res = {}
for i = 1, #arr do
if deep == true and type(arr[i]) == 'table' then
res[i] = Array.clone(arr[i], true)
else
res[i] = arr[i]
end
end
return setmetatable(res, getmetatable(arr))
end
end


function arr.__pow(lhs, rhs)
---Check if `arr` contains `val`.
    return mathTemplate(lhs, rhs, '__pow', function(x, y) return x ^ y end)
---@param arr any[]
---@param val any
---@return boolean
function Array.contains(arr, val)
checkType('Module:Array.contains', 1, arr, 'table')
for i = 1, #arr do
if arr[i] == val then
return true
end
end
return false
end
end


function arr.__lt(lhs, rhs)
---Check if `arr` contains any of the values in the table `t`.
    for i = 1, math.min(#lhs, #rhs) do
---@param arr any[]
        if lhs[i] >= rhs[i] then
---@param t any[]
            return false
---@return boolean
        end
function Array.containsAny(arr, t)
    end
checkType('Module:Array.containsAny', 1, arr, 'table')
    return true
checkType('Module:Array.containsAny', 2, t, 'table')
local lookupTbl = {}
for i = 1, #t do
lookupTbl[t[i]] = true
end
for i = 1, #arr do
if lookupTbl[arr[i]] then
return true
end
end
return false
end
end


function arr.__le(lhs, rhs)
---Check if `arr` contains all values in the table `t`.
    for i = 1, math.min(#lhs, #rhs) do
---@param arr any[]
        if lhs[i] > rhs[i] then
---@param t any[]
            return false
---@return boolean
        end
function Array.containsAll(arr, t)
    end
checkType('Module:Array.containsAll', 1, arr, 'table')
    return true
checkType('Module:Array.containsAll', 2, t, 'table')
local lookupTbl = {}
local l = #t
local trueCount = 0
for i = 1, l do
lookupTbl[t[i]] = false
end
for i = 1, #arr do
if lookupTbl[arr[i]] == false then
lookupTbl[arr[i]] = true
trueCount = trueCount + 1
end
if trueCount == l then
return true
end
end
return false
end
end


function arr.__eq(lhs, rhs)
---Convolute two number arrays.
     if #lhs ~= #rhs then
---@generic T: number[]
         return false
---@param x T
    end
---@param y T
    for i = 1, #lhs do
---@return T
        if lhs[i] ~= rhs[i] then
function Array.convolve(x, y)
            return false
checkType('Module:Array.convolve', 1, x, 'table')
checkType('Module:Array.convolve', 2, y, 'table')
local z = {}
     local xLen, yLen = #x, #y
    for j = 1, (xLen + yLen - 1) do
         local sum = 0
        for k = math.max(1, j - yLen + 1), math.min(xLen, j) do
            sum = sum + x[k] * y[j-k+1]
         end
         end
        z[j] = sum
     end
     end
     return true
     return setmetatable(z, getmetatable(x) or getmetatable(y))
end
end


function arr.all(array, fn)
---Remove **nil** values from `arr` while preserving order.
    checkType('Module:Array.all', 1, array, 'table')
---@generic T: any[]
    if fn == nil then fn = function(item) return item end end
---@param arr T
    if type(fn) ~= 'function' then
---@return T
        local val = fn
function Array.condenseSparse(arr)
        fn = function(item) return item == val end
checkType('Module:Array.condenseSparse', 1, arr, 'table')
    end
local keys = {}
    local i = 1
local res = {}
    while array[i] ~= nil do
local l = 0
        if not fn(array[i], i) then
for k in pairs(arr) do
            return false
l = l + 1
        end
keys[l] = k
        i = i + 1
end
    end
table.sort(keys)
    return true
for i = 1, l do
res[i] = arr[keys[i]]
end
return setmetatable(res, getmetatable(arr))
end
end


function arr.any(array, fn)
---Behaviour depends on value of `val`:
    checkType('Module:Array.any', 1, array, 'table')
---* `nil` - Counts the number of non **false** elements.
    if fn == nil then fn = function(item) return item end end
---* `fun(elem: any): boolean` - Count the number of times the function returned **true**.
    if type(fn) ~= 'function' then
---* `boolean` | `number` | `table` - Counts the number of times this value occurs in `arr`.
        local val = fn
---@param arr any[]
        fn = function(item) return item == val end
---@param val? any
    end
---@return integer
    local i = 1
function Array.count(arr, val)
    while array[i] ~= nil do
checkType('Module:Array.count', 1, arr, 'table')
        if fn(array[i], i) then
if val == nil then val = function(item) return item end end
            return true
if type(val) ~= 'function' then
        end
local _val = val
        i = i + 1
val = function(item) return item == _val end
    end
end
    return false
local count = 0
for i = 1, #arr do
if val(arr[i]) then
count = count + 1
end
end
return count
end
end


function arr.clean(array)
---Differentiate the array
    checkType('Module:Array.clean', 1, array, 'table')
---@generic T: number[]
    for i = 1, #array do
---@param arr T
        if type(array[i]) == 'table' then
---@param order number? # Oder of the differentiation. Default is 1.
            arr.clean(array[i])
---@return T # Length is `#arr - order`
        end
function Array.diff(arr, order)
    end
checkType('Module:Array.diff', 1, arr, 'table')
    setmetatable(array, nil)
checkType('Module:Array.diff', 2, order, 'number', true)
    return array
local res = {}
for i = 1, #arr - 1 do
res[i] = arr[i+1] - arr[i]
end
if order and order > 1 then
return Array.diff(res, order - 1)
end
return setmetatable(res, getmetatable(arr))
end
end


function arr.contains(array, elem, useElemTableContent)
---Loops over `arr` and passes each element as the first argument to `fn`. This function returns nothing.
    checkType('Module:Array.contains', 1, array, 'table')
---@param arr any[]
    if type(elem) == 'table' and useElemTableContent ~= false then
---@param fn fun(elem: any, i?: integer)
        local elemMap = {}
function Array.each(arr, fn)
        local isFound = {}
checkType('Module:Array.each', 1, arr, 'table')
        arr.each(elem, function(x, i) elemMap[x] = i; isFound[i] = false end)
checkType('Module:Array.each', 2, fn, 'function')
        for i = 1, #array do
local i = 1
            local j = elemMap[array[i]]
while arr[i] ~= nil do
            if j then
fn(arr[i], i)
                isFound[j] = true
i = i + 1
            end
end
        end
        return arr.all(isFound, true)
    else
        return arr.any(array, function(item) return item == elem end)
    end
end
end


function arr.count(array, fn)
---Makes a copy of `arr` with only elements for which `fn` returned **true**.
    checkType('Module:Array.count', 1, array, 'table')
---@generic T: any[]
    if fn == nil then fn = function(item) return item end end
---@param arr T
    if type(fn) ~= 'function' then
---@param fn fun(elem: any, i?: integer): boolean
        local val = fn
---@return T
        fn = function(item) return item == val end
function Array.filter(arr, fn)
    end
checkType('Module:Array.filter', 1, arr, 'table')
    local count = 0
checkType('Module:Array.filter', 2, fn, 'function')
    for i = 1, #array do
local r = {}
        if fn(array[i]) then
local len = 0
            count = count + 1
local i = 1
        end
while arr[i] ~= nil do
    end
if fn(arr[i], i) then
    return count
len = len + 1
r[len] = arr[i]
end
i = i + 1
end
return setmetatable(r, getmetatable(arr))
end
end


function arr.diff(array, order)
---Find the first elements for which `fn` returns **true**.
    checkType('Module:Array.diff', 1, array, 'table')
---@param arr any[]
    checkType('Module:Array.diff', 2, order, 'number', true)
---@param fn any # A value to look for or a function of the form `fun(elem: any, i?: integer): boolean`.
    local res = setmetatable({}, getmetatable(array))
---@param default? any # Value to return if no element passes the test.
    for i = 1, #array - 1 do
---@return any? elem # The first element that passed the test.
        res[i] = array[i+1] - array[i]
---@return integer? i # The index of the item that passed the test.
    end
function Array.find(arr, fn, default)
    if order and order > 1 then
checkType('Module:Array.find', 1, arr, 'table')
        return arr.diff(res, order - 1)
checkTypeMulti('Module:Array.find_index', 2, fn, {'function', 'table', 'number', 'boolean'})
    end
if type(fn) ~= 'function' then
    return res
local _val = fn
fn = function(item) return item == _val end
end
local i = 1
while arr[i] ~= nil do
---@diagnostic disable-next-line: redundant-parameter
if fn(arr[i], i) then
return arr[i], i
end
i = i + 1
end
return default, nil
end
end


function arr.each(array, fn)
---Find the index of `val`.
    checkType('Module:Array.each', 1, array, 'table')
---@param arr any[]
    checkType('Module:Array.each', 2, fn, 'function')
---@param val any # A value to look for or a function of the form `fun(elem: any, i?: integer): boolean`.
    local i = 1
---@param default? any # Value to return if no element passes the test.
    while array[i] ~= nil do
---@return integer?
        fn(array[i], i)
function Array.find_index(arr, val, default)
        i = i + 1
checkType('Module:Array.find_index', 1, arr, 'table')
    end
checkTypeMulti('Module:Array.find_index', 2, val, {'function', 'table', 'number', 'boolean'})
if type(val) ~= 'function' then
local _val = val
val = function(item) return item == _val end
end
local i = 1
while arr[i] ~= nil do
---@diagnostic disable-next-line: redundant-parameter
if val(arr[i], i) then
return i
end
i = i + 1
end
return default
end
end


function arr.filter(array, fn)
---Extracts a subset of `arr`.
    checkType('Module:Array.filter', 1, array, 'table')
---@generic T: any[]
    if fn == nil then fn = function(item) return item end end
---@param arr T
    if type(fn) ~= 'function' then
---@param indexes integer|integer[] # Indexes of the elements.
        local val = fn
---@return T
        fn = function(item) return item == val end
function Array.get(arr, indexes)
    end
checkType('Module:Array.set', 1, arr, 'table')
    local r = setmetatable({}, getmetatable(array))
checkTypeMulti('Module:Array.set', 2, indexes, {'table', 'number'})
    local len = 0
if type(indexes) == 'number' then
    local i = 1
indexes = {indexes}
    while array[i] ~= nil do
end
        if fn(array[i], i) then
local res = {}
            len = len + 1
for i = 1, #indexes do
            r[len] = array[i]
res[i] = arr[indexes[i]]
        end
end
        i = i + 1
return setmetatable(res, getmetatable(arr))
    end
    return r
end
end


function arr.find(array, fn, default)
---Integrates the array. Effectively does $\left\{\sum^{n}_{start}{arr[n]} \,\Bigg|\, n \in [start, stop]\right\}$.
    checkType('Module:Array.find', 1, array, 'table')
---@generic T: number[]
    checkTypeMulti('Module:Array.find_index', 2, fn, {'function', 'table', 'number', 'boolean'})
---@param arr T # number[]
    if type(fn) ~= 'function' then
---@param start? integer # Index where to start the summation. Defaults to 1.
        local val = fn
---@param stop? integer # Index where to stop the summation. Defaults to #arr.
        fn = function(item) return item == val end
---@return T
    end
function Array.int(arr, start, stop)
    local i = 1
checkType('Module:Array.int', 1, arr, 'table')
    while array[i] ~= nil do
checkType('Module:Array.int', 2, start, 'number', true)
        if fn(array[i], i) then
checkType('Module:Array.int', 3, stop, 'number', true)
            return array[i], i
local res = {}
        end
start = start or 1
        i = i + 1
stop = stop or #arr
    end
res[1] = arr[start]
    return default
for i = 1, stop - start do
res[i+1] = res[i] + arr[start + i]
end
return setmetatable(res, getmetatable(arr))
end
end


function arr.find_index(array, fn, default)
---Returns an array with elements that are present in both tables.
    checkType('Module:Array.find_index', 1, array, 'table')
---@generic T: any[]
    checkTypeMulti('Module:Array.find_index', 2, fn, {'function', 'table', 'number', 'boolean'})
---@param arr1 T
    if type(fn) ~= 'function' then
---@param arr2 T
        local val = fn
---@return T
        fn = function(item) return item == val end
function Array.intersect(arr1, arr2)
    end
checkType('Module:Array.intersect', 1, arr1, 'table')
    local i = 1
checkType('Module:Array.intersect', 2, arr2, 'table')
    while array[i] ~= nil do
local arr2Elements = {}
        if fn(array[i], i) then
local res = {}
            return i
local len = 0
        end
Array.each(arr2, function(item) arr2Elements[item] = true end)
        i = i + 1
Array.each(arr1, function(item)
    end
if arr2Elements[item] then
    return default
len = len + 1
res[len] = item
end
end)
return setmetatable(res, getmetatable(arr1) or getmetatable(arr2))
end
end


function arr.newIncrementor(start, step)
---Checks if the two inputs have at least one element in common.
    checkType('Module:Array.newIncrementor', 1, start, 'number', true)
---@param arr1 any[]
    checkType('Module:Array.newIncrementor', 2, step, 'number', true)
---@param arr2 any[]
    step = step or 1
---@return boolean
    local n = (start or 1) - step
function Array.intersects(arr1, arr2)
    local obj = {}
checkType('Module:Array.intersects', 1, arr1, 'table')
    return setmetatable(obj, {
checkType('Module:Array.intersects', 2, arr2, 'table')
        __call = function() n = n + step return n end,
local small = {}
        __tostring = function() return n end,
local large
        __index = function() return n end,
if #arr1 <= #arr2 then
        __newindex = function(self, k, v)
Array.each(arr1, function(item) small[item] = true end)
            if k == 'step' and type(v) == 'number' then
large = arr2
                step = v
else
            elseif type(v) == 'number' then
Array.each(arr2, function(item) small[item] = true end)
                n = v
large = arr1
            end
end
        end,
return Array.any(large, function(item) return small[item] end)
        __concat = function(x, y) return tostring(x) .. tostring(y) end
    })
end
end


function arr.int(array, start, stop)
---Inserts values into `arr`.
    checkType('Module:Array.int', 1, array, 'table')
---@generic T: any[]
    checkType('Module:Array.int', 2, start, 'number', true)
---@param arr T
    checkType('Module:Array.int', 3, stop, 'number', true)
---@param val any # If `val` is an array and `unpackVal` is **true** then the individual elements of `val` are inserted.
    local res = setmetatable({}, getmetatable(array))
---@param index? integer # Location to start the insertion. Default is at the end of `arr`.
    start = start or 1
---@param unpackVal? boolean # Default is **false**.
    stop = stop or #array
---@return T
    res[1] = array[start]
---@overload fun(arr: T, val: any, unpackVal: boolean): T
    for i = 1, stop - start do
function Array.insert(arr, val, index, unpackVal)
        res[i+1] = res[i] + array[start + i]
checkType('Module:Array.insert', 1, arr, 'table')
    end
checkTypeMulti('Module:Array.insert', 3, index, {'number', 'boolean', 'nil'})
    return res
checkType('Module:Array.insert', 4, unpackVal, 'boolean', true)
if type(index) == 'boolean'  then
unpackVal, index = index, nil
end
local len = #arr
index = index or (len + 1)
local mt = getmetatable(arr)
setmetatable(arr, nil)
 
if type(val) == 'table' and unpackVal then
local len2 = #val
for i = 0, len - index do
arr[len + len2 - i] = arr[len - i]
end
for i = 0, len2 - 1 do
arr[index + i] = val[i + 1]
end
else
table.insert(arr, index, val)
end
 
return setmetatable(arr, mt)
end
end


function arr.intersect(array1, array2)
---Returns the last element of `arr`.
    checkType('Module:Array.intersect', 1, array1, 'table')
---@param arr any[]
    checkType('Module:Array.intersect', 2, array2, 'table')
---@param offset? integer
    local array2Elements = {}
---@return any
    local res = setmetatable({}, getmetatable(array1) or getmetatable(array2))
function Array.last(arr, offset)
    local len = 0
checkType('Module:Array.last', 1, arr, 'table')
    arr.each(array2, function(item) array2Elements[item] = true end)
checkType('Module:Array.last', 2, offset, 'number', true)
    arr.each(array1, function(item)
return arr[#arr + offset]
        if array2Elements[item] then
            len = len + 1
            res[len] = item
        end
    end)
    return res
end
end


function arr.intersects(array1, array2)
---Returns a new table were each element of `arr` is modified by `fn`.
    checkType('Module:Array.intersects', 1, array1, 'table')
---@generic T: any[]
    checkType('Module:Array.intersects', 2, array2, 'table')
---@param arr T
    local small = {}
---@param fn fun(elem: any, i?: integer): any # First argument is the current element, the second argument is the index of the current element.
    local large
---@return T
    if #array1 <= #array2 then
function Array.map(arr, fn)
        arr.each(array1, function(item) small[item] = true end)
checkType('Module:Array.map', 1, arr, 'table')
        large = array2
checkType('Module:Array.map', 2, fn, 'function')
    else
local len = 0
        arr.each(array2, function(item) small[item] = true end)
local r = {}
        large = array1
local i = 1
    end
while arr[i] ~= nil do
    return arr.any(large, function(item) return small[item] end)
local tmp = fn(arr[i], i)
if tmp ~= nil then
len = len + 1
r[len] = tmp
end
i = i + 1
end
return setmetatable(r, getmetatable(arr))
end
end


function arr.insert(array, val, index, unpackVal)
---Find the element for which `fn` returned the largest value.
    checkType('Module:Array.insert', 1, array, 'table')
---@param arr any[]
    checkType('Module:Array.insert', 3, index, 'number', true)
---@param fn fun(elem: any): any # The returned value needs to be comparable using the `<` operator.
    checkType('Module:Array.insert', 4, unpackVal, 'boolean', true)
---@return any elem # The element with the largest `fn` value.
    local len = #array
---@return integer i # The index of this element.
    index = index or (len + 1)
function Array.max_by(arr, fn)
checkType('Module:Array.max_by', 1, arr, 'table')
checkType('Module:Array.max_by', 2, fn, 'function')
return unpack(Array.reduce(arr, function(new, old, i)
local y = fn(new)
return y > old[2] and {new, y, i} or old
end, {nil, -math.huge}))
end


    if type(val) == 'table' and unpackVal ~= false then
---Find the largest value in the array.
        local len2 = #val
---@param arr any[] # The values need to be comparable using the `<` operator.
        for i = 0, len - index do
---@return any elem
            array[len + len2 - i] = array[len - i]
---@return integer i # The index of the largest value.
        end
function Array.max(arr)
        for i = 0, len2 - 1 do
checkType('Module:Array.max', 1, arr, 'table')
            array[index + i] = val[i + 1]
local val, _, i = Array.max_by(arr, function(x) return x end)
        end
return val, i
    else
end
        table.insert(array, index, val)
    end


    return array
---Find the smallest value in the array.
---@param arr any[] # The values need to be comparable using the `<` operator.
---@return any elem
---@return integer i # The index of the smallest value.
function Array.min(arr)
checkType('Module:Array.min', 1, arr, 'table')
local val, _, i = Array.max_by(arr, function(x) return -x end)
return val, i
end
end


function arr.map(array, fn)
---Turn the input table into an Array. This makes it possible to use the colon `:` operator to access the Array methods.
    checkType('Module:Array.map', 1, array, 'table')
---
    checkType('Module:Array.map', 2, fn, 'function')
---It also enables the use of math operators with the array.
    local len = 0
---```
    local r = setmetatable({}, getmetatable(array))
---local x = arr.new{ 1, 2, 3 }
    local i = 1
---local y = arr{ 4, 5, 6 } -- Alternative notation
    while array[i] ~= nil do
---
        local tmp = fn(array[i], i)
---print( -x ) --> { -1, -2, -3 }
        if tmp ~= nil then
---print( x + 2 ) --> { 3, 4, 5 }
            len = len + 1
---print( x - 2 ) --> { -1, 0, 1 }
            r[len] = tmp
---print( x * 2 ) --> { 2, 4, 6 }
        end
---print( x / 2 ) --> { 0.5, 1, 1.5 }
        i = i + 1
---print( x ^ 2 ) --> { 1, 4, 9 }
    end
---
    return r
---print( x + y ) --> { 5, 7, 9 }
---print( x .. y ) --> { 1, 2, 3, 4, 5, 6 }
---print( (x .. y):reject{3, 4, 5} ) --> { 1, 2, 6 }
---print( x:sum() ) --> 6
---
---print( x:update( {1, 3}, y:get{2, 3} * 2 ) ) --> { 10, 2, 12 }
---```
---@param arr? any[]
---@return Array
function Array.new(arr)
local obj = arr or {}
for _, v in pairs(obj) do
if type(v) == 'table' then
Array.new(v)
end
end
 
if getmetatable(obj) == nil then
setmetatable(obj, Array)
end
 
return obj
end
end


function arr.max_by(array, fn)
---Creates an object that returns a value that is `step` higher than the previous value each time it gets called.
    checkType('Module:Array.max_by', 1, array, 'table')
---
    checkType('Module:Array.max_by', 2, fn, 'function')
---The stored value can be read without incrementing by reading the `val` field.
    return unpack(arr.reduce(array, function(new, old, i)
---
        local y = fn(new)
---A new stored value can be set through the `val` field.
        return y > old[2] and {new, y, i} or old
---
    end, {nil, -math.huge}))
---A new step size can be set through the `step` field.
---```
---local inc = arr.newIncrementor(10, 5)
---print( inc() ) --> 10
---print( inc() ) --> 15
---print( inc.val ) --> 15
---inc.val = 100
---inc.step = 20
---print( inc.val ) --> 100
---print( inc() ) --> 120
---```
---@param start? number # Default is 1.
---@param step? number # Default is 1.
---@return Incrementor
function Array.newIncrementor(start, step)
checkType('Module:Array.newIncrementor', 1, start, 'number', true)
checkType('Module:Array.newIncrementor', 2, step, 'number', true)
step = step or 1
local n = (start or 1) - step
---@class Incrementor
local obj = {}
return setmetatable(obj, {
__call = function() n = n + step return n end,
__tostring = function() return n end,
__index = function() return n end,
__newindex = function(self, k, v)
if k == 'step' and type(v) == 'number' then
step = v
elseif type(v) == 'number' then
n = v
end
end,
__concat = function(x, y) return tostring(x) .. tostring(y) end
})
end
end


function arr.max(array)
---Returns a range of numbers.
    checkType('Module:Array.max', 1, array, 'table')
---@param start number # Start value inclusive.
    local val, _, i = arr.max_by(array, function(x) return x end)
---@param stop number # Stop value inclusive for integers, exclusive for floats.
    return val, i
---@param step? number # Default is 1.
---@return Array
---@overload fun(stop: number): Array
function Array.range(start, stop, step)
checkType('Module:Array.range', 1, start, 'number')
checkType('Module:Array.range', 2, stop, 'number', true)
checkType('Module:Array.range', 3, step, 'number', true)
local arr = {}
local len = 0
if not stop then
stop = start
start = 1
end
for i = start, stop, step or 1 do
len = len + 1
arr[len] = i
end
return setmetatable(arr, Array)
end
end


function arr.min(array)
---Condenses the array into a single value.
    checkType('Module:Array.min', 1, array, 'table')
---
    local val, _, i = arr.max_by(array, function(x) return -x end)
---For each element `fn` is called with the current element, the current accumulator, and the current element index. The returned value of `fn` becomes the accumulator for the next element.
    return val, i
---
---If no `accumulator` value is given at the start then the first element off `arr` becomes the accumulator and the iteration starts from the second element.
---```
---local t = { 1, 2, 3, 4 }
---local sum = arr.reduce( t, function(elem, acc) return acc + elem end ) -- sum == 10
---```
---@param arr any[]
---@param fn fun(elem: any, acc: any, i?: integer): any # The result of this function becomes the `acc` for the next element.
---@param accumulator? any
---@return any # This is the last accumulator value.
function Array.reduce(arr, fn, accumulator)
checkType('Module:Array.reduce', 1, arr, 'table')
checkType('Module:Array.reduce', 2, fn, 'function')
local acc = accumulator
local i = 1
if acc == nil then
acc = arr[1]
i = 2
end
while arr[i] ~= nil do
acc = fn(arr[i], acc, i)
i = i + 1
end
return acc
end
end


function arr.new(array)
---Make a copy off `arr` with certain values removed.
    array = array or {}
---
    for _, v in pairs(array) do
---Behaviour for different values of `val`:
        if type(v) == 'table' then
---* `boolean` | `number` - Remove values equal to this.
            arr.new(v)
---* `table` - Remove all values in this table.
        end
---* `fun(elem: any, i?: integer): boolean` - Remove elements for which the functions returns **true**.
    end
---@generic T: any[]
---@param arr T
---@param val table|function|number|boolean
---@return T
function Array.reject(arr, val)
checkType('Module:Array.reject', 1, arr, 'table')
checkTypeMulti('Module:Array.reject', 2, val, {'function', 'table', 'number', 'boolean'})
if type(val) ~= 'function' and type(val) ~= 'table' then
val = {val}
end
local r = {}
local len = 0
if type(val) == 'function' then
local i = 1
while arr[i] ~= nil do
if not val(arr[i], i) then
len = len + 1
r[len] = arr[i]
end
i = i + 1
end
else
local rejectMap = {}
Array.each(val --[[@as any[] ]], function(item) rejectMap[item] = true end)
local i = 1
while arr[i] ~= nil do
if not rejectMap[arr[i]] then
len = len + 1
r[len] = arr[i]
end
i = i + 1
end
end
return setmetatable(r, getmetatable(arr))
end


    if getmetatable(array) == nil then
---Returns an Array with `val` repeated `n` times.
        setmetatable(array, arr)
---@param val any
    end
---@param n integer
---@return Array
function Array.rep(val, n)
checkType('Module:Array.rep', 2, n, 'number')
local r = {}
for i = 1, n do
r[i] = val
end
return setmetatable(r, Array)
end


    return array
---Condenses the array into a single value while saving every accumulator value.
---
---For each element `fn` is called with the current element, the current accumulator, and the current element index. The returned value of `fn` becomes the accumulator for the next element.
---
---If no `accumulator` value is given at the start then the first element off `arr` becomes the accumulator and the iteration starts from the second element.
---```
---local t = { 1, 2, 3, 4 }
---local x = arr.scan( t, function(elem, acc) return acc + elem end ) -- x = { 1, 3, 6, 10 }
---```
---@generic T: any[]
---@param arr T
---@param fn fun(elem: any, acc: any, i?: integer): any # Returned value becomes the accumulator for the next element.
---@param accumulator? any
---@return T
function Array.scan(arr, fn, accumulator)
checkType('Module:Array.scan', 1, arr, 'table')
checkType('Module:Array.scan', 2, fn, 'function')
local acc = accumulator
local r = {}
local i = 1
while arr[i] ~= nil do
if i == 1 and not accumulator then
acc = arr[i]
else
acc = fn(arr[i], acc, i)
end
r[i] = acc
i = i + 1
end
return setmetatable(r, getmetatable(arr))
end
end


function arr.range(start, stop, step)
---Update a range of index with a range of values.
    checkType('Module:Array.range', 1, start, 'number')
---
    checkType('Module:Array.range', 2, stop, 'number', true)
---If if only one value is given but multiple indexes than that value is set for all those indexes.
    checkType('Module:Array.range', 3, step, 'number', true)
---
    local array = setmetatable({}, arr)
---If `values` is a table then it must of the same length as `indexes`.
    local len = 0
---@generic T: any[]
    if not stop then
---@param arr T
        stop = start
---@param indexes integer|integer[]
        start = 1
---@param values any|any[]
    end
---@return T
    for i = start, stop, step or 1 do
function Array.set(arr, indexes, values)
        len = len + 1
checkType('Module:Array.set', 1, arr, 'table')
        array[len] = i
checkTypeMulti('Module:Array.set', 2, indexes, {'table', 'number'})
    end
local mt = getmetatable(arr)
    return array
setmetatable(arr, nil)
if type(indexes) == 'number' then
indexes = {indexes}
end
if type(values) == 'table' then
assert(#indexes == #values, string.format("Module:Array.set: 'indexes' and 'values' arrays are not equal length (#indexes = %d, #values = %d)", #indexes, #values))
for i = 1, #indexes do
arr[indexes[i]] = values[i]
end
else
for i = 1, #indexes do
arr[indexes[i]] = values
end
end
return setmetatable(arr, mt)
end
end


function arr.reduce(array, fn, accumulator)
---Extract a subtable from `arr`.
    checkType('Module:Array.reduce', 1, array, 'table')
---@generic T: any[]
    checkType('Module:Array.reduce', 2, fn, 'function')
---@param arr T
    local acc = accumulator
---@param start integer # Start index. Use negative values to count form the end of the array.
    local i = 1
---@param stop integer # Stop index. Use negative values to count form the end of the array.
    if acc == nil then
---@return T
        acc = array[1]
---@overload fun(arr: T, stop: integer): T
        i = 2
function Array.slice(arr, start, stop)
    end
checkType('Module:Array.slice', 1, arr, 'table')
    while array[i] ~= nil do
checkType('Module:Array.slice', 2, start, 'number', true)
        acc = fn(array[i], acc, i)
checkType('Module:Array.slice', 3, stop, 'number', true)
        i = i + 1
start = start or #arr
    end
if start < 0 then
    return acc
start = #arr + start
end
if stop == nil then
stop = start
start = 1
end
if stop < 0 then
stop = #arr + stop
end
local r = {}
local len = 0
for i = start, stop do
len = len + 1
r[len] = arr[i]
end
return setmetatable(r, getmetatable(arr))
end
end


function arr.reject(array, fn)
---Split `arr` into two arrays.
    checkType('Module:Array.reject', 1, array, 'table')
---@generic T: any[]
    checkTypeMulti('Module:Array.reject', 2, fn, {'function', 'table', 'number', 'boolean'})
---@param arr T
    if fn == nil then fn = function(item) return item end end
---@param index integer # Index to split on.
    if type(fn) ~= 'function' and type(fn) ~= 'table' then
---@return T x # [1, index]
        fn = {fn}
---@return T y # [index + 1, #arr]
    end
function Array.split(arr, index)
    local r = setmetatable({}, getmetatable(array))
checkType('Module:Array.split', 1, arr, 'table')
    local len = 0
checkType('Module:Array.split', 2, index, 'number')
    if type(fn) == 'function' then
local x = {}
        local i = 1
local y = {}
        while array[i] ~= nil do
for i = 1, #arr do
            if not fn(array[i], i) then
table.insert(i <= index and x or y, arr[i])
                len = len + 1
end
                r[len] = array[i]
return setmetatable(x, getmetatable(arr)), setmetatable(y, getmetatable(arr))
            end
            i = i + 1
        end
    else
        local rejectMap = {}
        arr.each(fn, function(item) rejectMap[item] = true end)
        local i = 1
        while array[i] ~= nil do
            if not rejectMap[array[i]] then
                len = len + 1
                r[len] = array[i]
            end
            i = i + 1
        end
    end
    return r
end
end


function arr.rep(val, n)
---Returns the sum of all elements of `arr`.
    checkType('Module:Array.rep', 2, n, 'number')
---@param arr number[]
    local r = setmetatable({}, arr)
---@return number
    for i = 1, n do
function Array.sum(arr)
        r[i] = val
checkType('Module:Array.sum', 1, arr, 'table')
    end
local res = 0
    return r
for i = 1, #arr do
res = res + arr[i]
end
return res
end
end


function arr.scan(array, fn, accumulator)
---Extract a subtable from `arr`.
    checkType('Module:Array.scan', 1, array, 'table')
---@generic T: any[]
    checkType('Module:Array.scan', 2, fn, 'function')
---@param arr T
    local acc = accumulator
---@param count integer # Length of the subtable.
    local r = setmetatable({}, getmetatable(array))
---@param start? integer # Start index. Default is 1.
    local i = 1
---@return T
    while array[i] ~= nil do
function Array.take(arr, count, start)
        if i == 1 and not accumulator then
checkType('Module:Array.take', 1, arr, 'table')
            acc = array[i]
checkType('Module:Array.take', 2, count, 'number')
        else
checkType('Module:Array.take', 3, start, 'number', true)
            acc = fn(array[i], acc)
local x = {}
        end
start = start or 1
        r[i] = acc
for i = start, math.min(#arr, count + start - 1) do
        i = i + 1
table.insert(x, arr[i])
    end
end
    return r
return setmetatable(x, getmetatable(arr))
end
end


function arr.slice(array, start, finish)
---Extract a subtable from `arr`.
    checkType('Module:Array.slice', 1, array, 'table')
---```
    checkType('Module:Array.slice', 2, start, 'number', true)
---local t = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 }
    checkType('Module:Array.slice', 3, finish, 'number', true)
---local x = arr.take_every( t, 2 )      --> x = { 1, 3, 5, 7, 9 }
    start = start or 1
---local x = arr.take_every( t, 2, 3 )    --> x = { 1, 3, 5 }
    finish = finish or #array
---local x = arr.take_every( t, 2, 3, 2 ) --> x = { 2, 4, 6 }
    if start < 0 and finish == nil then
--- ```
        finish = #array + start
---@generic T: any[]
        start = 1
---@param arr T
    elseif start < 0 then
---@param n integer # Step size.
        start = #array + start
---@param start? integer # Start index.
    end
---@param count? integer # Max amount of elements to get.
    if finish < 0 then
---@return T
        finish = #array + finish
function Array.take_every(arr, n, start, count)
    end
checkType('Module:Array.take_every', 1, arr, 'table')
    local r = setmetatable({}, getmetatable(array))
checkType('Module:Array.take_every', 2, n, 'number')
    local len = 0
checkType('Module:Array.take_every', 3, start, 'number', true)
    for i = start, finish do
checkType('Module:Array.take_every', 4, count, 'number', true)
        len = len + 1
count = count or #arr
        r[len] = array[i]
local r = {}
    end
local len = 0
    return r
local i = start or 1
while arr[i] ~= nil and len < count do
len = len + 1
r[len] = arr[i]
i = i + n
end
return setmetatable(r, getmetatable(arr))
end
end


function arr.split(array, count)
---Return a new table with all duplicates removed.
    checkType('Module:Array.split', 1, array, 'table')
---@generic T: any[]
    checkType('Module:Array.split', 2, count, 'number')
---@param arr T
    local x = setmetatable({}, getmetatable(array))
---@param fn? fun(elem: any): any # Function to generate an id for each element. The result will then contain elements that generated unique ids.
    local y = setmetatable({}, getmetatable(array))
---@return T
    for i = 1, #array do
function Array.unique(arr, fn)
        table.insert(i <= count and x or y, array[i])
checkType('Module:Array.unique', 1, arr, 'table')
    end
checkType('Module:Array.unique', 2, fn, 'function', true)
    return x, y
fn = fn or function(item) return item end
local r = {}
local len = 0
local hash = {}
local i = 1
while arr[i] ~= nil do
local id = fn(arr[i])
if not hash[id] then
len = len + 1
r[len] = arr[i]
hash[id] = true
end
i = i + 1
end
return setmetatable(r, getmetatable(arr))
end
end


function arr.sum(array)
---Combine elements with the same index from multiple arrays.
    checkType('Module:Array.sum', 1, array, 'table')
---```
    local res = 0
---local x = {1, 2, 3}
    for i = 1, #array do
---local y = {4, 5, 6, 7}
        res = res + array[i]
---local z = arr.zip( x, y ) --> z = { { 1, 4 }, { 2, 5 }, { 3, 6 }, { 7 } }
    end
---```
    return res
---@param ... any[]
---@return Array
function Array.zip(...)
local arrs = { ... }
checkType('Module:Array.zip', 1, arrs[1], 'table')
local r = {}
local _, longest = Array.max_by(arrs, function(arr) return #arr end)
for i = 1, longest do
local q = {}
for j = 1, #arrs do
table.insert(q, arrs[j][i])
end
table.insert(r, setmetatable(q, Array))
end
return setmetatable(r, Array)
end
end


function arr.take(array, count, offset)
-- Range indexing has a performance impact so this is placed in a separate subclass
    checkType('Module:Array.take', 1, array, 'table')
Array.RI_mt = {}
    checkType('Module:Array.take', 2, count, 'number')
for k, v in pairs(Array) do
    checkType('Module:Array.take', 3, offset, 'number', true)
Array.RI_mt[k] = v
    local x = setmetatable({}, getmetatable(array))
    for i = offset or 1, #array do
        if i <= count then
            table.insert(x, array[i])
        end
    end
    return x
end
end


function arr.take_every(array, n, offset)
function Array.RI_mt.__index(t, k)
    checkType('Module:Array.take_every', 1, array, 'table')
if type(k) == 'table' then
    checkType('Module:Array.take_every', 2, n, 'number')
local res = {}
    checkType('Module:Array.take_every', 3, offset, 'number', true)
for i = 1, #k do
    local r = setmetatable({}, getmetatable(array))
res[i] = t[k[i]]
    local len = 0
end
    local i = offset or 1
return setmetatable(res, Array)
    while array[i] ~= nil do
else
        len = len + 1
return Array[k]
        r[len] = array[i]
end
        i = i + n
    end
    return r
end
end


function arr.unique(array, fn)
function Array.RI_mt.__newindex(t, k, v)
    checkType('Module:Array.unique', 1, array, 'table')
if type(k) == 'table' then
    checkType('Module:Array.unique', 2, fn, 'function', true)
if type(v) == 'table' then
    fn = fn or function(item) return item end
for i = 1, #k do
    local r = setmetatable({}, getmetatable(array))
t[k[i]] = v[i]
    local len = 0
end
    local hash = {}
else
    local i = 1
for i = 1, #k do
    while array[i] ~= nil do
t[k[i]] = v
        local id = fn(array[i])
end
        if not hash[id] then
end
            len = len + 1
else
            r[len] = array[i]
rawset(t, k, v)
            hash[id] = true
end
        end
        i = i + 1
    end
    return r
end
end


function arr.update(array, indexes, values)
---Enable range indexing on the input array.
    checkType('Module:Array.update', 1, array, 'table')
---
    checkTypeMulti('Module:Array.update', 2, indexes, {'table', 'number'})
---This has a performance impact on reads and writes to the table.
    if type(indexes) == 'number' then
---```
        indexes = {indexes}
---local t = arr{10, 11, 12, 13, 14, 15}:ri()
    end
---print( t[{2, 3}] ) --> { 11, 12 }
    if type(values) == 'table' then
---```
        assert(#indexes == #values, 'Values array must be of equal length as index array')
---@param arr any[]
        for i = 1, #indexes do
---@param recursive? boolean # Default is false.
            array[indexes[i]] = values[i]
---@return Array
        end
function Array.ri(arr, recursive)
    else
checkType('Module:Array.ri', 1, arr, 'table')
        for i = 1, #indexes do
checkType('Module:Array.ri', 2, recursive, 'boolean', true)
            array[indexes[i]] = values
arr = arr or {}
        end
if recursive then
    end
for _, v in pairs(arr) do
    return array
if type(v) == 'table' then
Array.ri(v, true)
end
end
end
 
if getmetatable(arr) == nil or getmetatable(arr) == Array then
setmetatable(arr, Array.RI_mt)
end
 
return arr
end
end


function arr.zip(...)
---Globally enable range indexing on all Array objects by default.
    local arrays = { ... }
---@param set boolean
    checkType('Module:Array.zip', 1, arrays[1], 'table')
function Array.allwaysAllowRangeIndexing(set)
    local r = setmetatable({}, getmetatable(arrays[1]))
checkType('Module:Array.allwaysAllowRangeIndexing', 1, set, 'boolean')
    local _, longest = arr.max_by(arrays, function(array) return #array end)
if set then
    for i = 1, longest do
Array.__index = Array.RI_mt.__index
        local q = {}
Array.__newindex = Array.RI_mt.__newindex
        for j = 1, #arrays do
else
            table.insert(q, arrays[j][i])
Array.__index = Array
        end
Array.__newindex = nil
        table.insert(r, q)
end
    end
    return r
end
end


return arr
return Array
-- </nowiki>

Revision as of 12:23, 24 June 2023

Documentation for this module may be created at Module:Array/doc

-- Imported from: https://runescape.wiki/w/Module:Array

local libraryUtil = require('libraryUtil')
local checkType = libraryUtil.checkType
local checkTypeMulti = libraryUtil.checkTypeMulti

---@class Array
---@operator call(any[]): Array
---@operator concat(any[]): Array
---@operator concat(number|string|function): string
---@operator unm: Array
---@operator add(number|number[]|Array): Array
---@operator sub(number|number[]|Array): Array
---@operator mul(number|number[]|Array): Array
---@operator div(number|number[]|Array): Array
---@operator pow(number|number[]|Array): Array
local Array = {
	pop = table.remove
}
Array.__index = Array

setmetatable(Array, {
	__index = table,
	__call = function (_, arr)
		return Array.new(arr)
	end
})

-- function Array.__tostring(arr)
-- 	-- local dumpObject = require('Module:Logger').dumpObject
-- 	require 'log'
-- 	local dumpObject = dumpObject
-- 	local mt = getmetatable(arr)
-- 	setmetatable(arr, nil)
-- 	local str = dumpObject(arr, {clean=true, collapseLimit=100})
-- 	setmetatable(arr, mt)
-- 	return str
-- end

function Array.__concat(lhs, rhs)
	if type(lhs) == 'table' and type(rhs) == 'table' then
		local res = {}
		for i = 1, #lhs do
			res[i] = lhs[i]
		end
		local l = #lhs
		for i = 1, #rhs do
			res[i + l] = rhs[i]
		end
		return setmetatable(res, getmetatable(lhs) or getmetatable(rhs))
	else
		return tostring(lhs) .. tostring(rhs)
	end
end

function Array.__unm(arr)
	return Array.map(arr, function(x) return -x end)
end

---@param lhs number|number[]|Array
---@param rhs number|number[]|Array
---@param funName string
---@param opName string
---@param fun fun(lhs: number, rhs: number): number
---@return Array
local function mathTemplate(lhs, rhs, funName, opName, fun)
	checkTypeMulti('Module:Array.' .. funName, 1, lhs, {'number', 'table'})
	checkTypeMulti('Module:Array.' .. funName, 2, rhs, {'number', 'table'})
	local res = {}

	if type(lhs) == 'number' then
		for i = 1, #rhs do
			res[i] = fun(lhs, rhs[i])
		end
	elseif type(rhs) == 'number' then
		for i = 1, #lhs do
			res[i] = fun(lhs[i], rhs)
		end
	else
		assert(#lhs == #rhs, string.format('Elementwise %s failed because arrays have different sizes (left: %d, right: %d)', opName, #lhs, #rhs))
		for i = 1, #lhs do
			res[i] = fun(lhs[i], rhs[i])
		end
	end

	return setmetatable(res, getmetatable(lhs) or getmetatable(rhs))
end

function Array.__add(lhs, rhs)
	return mathTemplate(lhs, rhs, '__add', 'addition', function(x, y) return x + y end)
end

function Array.__sub(lhs, rhs)
	return mathTemplate(lhs, rhs, '__sub', 'substraction', function(x, y) return x - y end)
end

function Array.__mul(lhs, rhs)
	return mathTemplate(lhs, rhs, '__mul', 'multiplication', function(x, y) return x * y end)
end

function Array.__div(lhs, rhs)
	return mathTemplate(lhs, rhs, '__div', 'division', function(x, y) return x / y end)
end

function Array.__pow(lhs, rhs)
	return mathTemplate(lhs, rhs, '__pow', 'exponentiation', function(x, y) return x ^ y end)
end

function Array.__eq(lhs, rhs)
	if #lhs ~= #rhs then
		return false
	end
	for i = 1, #lhs do
		if lhs[i] ~= rhs[i] then
			return false
		end
	end
	return true
end

---Behaviour depends on the value of `fn`:
---* `nil` - Checks that the array doesn't contain any **false** elements.
---* `fun(elem: any, i?: integer): boolean` - Returns **true** if `fn` returns **true** for every element.
---* `number` | `table` | `boolean` - Checks that all elements in `arr` are equal to this value.
---@param arr any[]
---@param fn? any
---@return boolean
function Array.all(arr, fn)
	checkType('Module:Array.all', 1, arr, 'table')
	if fn == nil then fn = function(item) return item end end
	if type(fn) ~= 'function' then
		local val = fn
		fn = function(item) return item == val end
	end
	local i = 1
	while arr[i] ~= nil do
		---@diagnostic disable-next-line: redundant-parameter
		if not fn(arr[i], i) then
			return false
		end
		i = i + 1
	end
	return true
end

---Behaviour depends on the value of `fn`:
---* `nil` - Checks that the array contains at least one non **false** element.
---* `fun(elem: any, i?: integer): boolean` - Returns **true** if `fn` returns **true** for at least one element.
---* `number` | `table` | `boolean` - Checks that `arr` contains this value.
---@param arr any[]
---@param fn? any
---@return boolean
function Array.any(arr, fn)
	checkType('Module:Array.any', 1, arr, 'table')
	if fn == nil then fn = function(item) return item end end
	if type(fn) ~= 'function' then
		local val = fn
		fn = function(item) return item == val end
	end
	local i = 1
	while arr[i] ~= nil do
		---@diagnostic disable-next-line: redundant-parameter
		if fn(arr[i], i) then
			return true
		end
		i = i + 1
	end
	return false
end

---Recursively removes all metatables.
---@param arr any[]
---@return any[]
function Array.clean(arr)
	checkType('Module:Array.clean', 1, arr, 'table')
	for i = 1, #arr do
		if type(arr[i]) == 'table' then
			Array.clean(arr[i])
		end
	end
	setmetatable(arr, nil)
	return arr
end

---Make a copy of the input table. Preserves metatables.
---@generic T: any[]
---@param arr T
---@param deep? boolean # Recursively clone subtables if **true**.
---@return T
function Array.clone(arr, deep)
	checkType('Module:Array.clone', 1, arr, 'table')
	checkType('Module:Array.clone', 2, deep, 'boolean', true)
	local res = {}
	for i = 1, #arr do
		if deep == true and type(arr[i]) == 'table' then
			res[i] = Array.clone(arr[i], true)
		else
			res[i] = arr[i]
		end
	end
	return setmetatable(res, getmetatable(arr))
end

---Check if `arr` contains `val`.
---@param arr any[]
---@param val any
---@return boolean
function Array.contains(arr, val)
	checkType('Module:Array.contains', 1, arr, 'table')
	for i = 1, #arr do
		if arr[i] == val then
			return true
		end
	end
	return false
end

---Check if `arr` contains any of the values in the table `t`.
---@param arr any[]
---@param t any[]
---@return boolean
function Array.containsAny(arr, t)
	checkType('Module:Array.containsAny', 1, arr, 'table')
	checkType('Module:Array.containsAny', 2, t, 'table')
	local lookupTbl = {}
	for i = 1, #t do
		lookupTbl[t[i]] = true
	end
	for i = 1, #arr do
		if lookupTbl[arr[i]] then
			return true
		end
	end
	return false
end

---Check if `arr` contains all values in the table `t`.
---@param arr any[]
---@param t any[]
---@return boolean
function Array.containsAll(arr, t)
	checkType('Module:Array.containsAll', 1, arr, 'table')
	checkType('Module:Array.containsAll', 2, t, 'table')
	local lookupTbl = {}
	local l = #t
	local trueCount = 0
	for i = 1, l do
		lookupTbl[t[i]] = false
	end
	for i = 1, #arr do
		if lookupTbl[arr[i]] == false then
			lookupTbl[arr[i]] = true
			trueCount = trueCount + 1
		end
		if trueCount == l then
			return true
		end
	end
	return false
end

---Convolute two number arrays.
---@generic T: number[]
---@param x T
---@param y T
---@return T
function Array.convolve(x, y)
	checkType('Module:Array.convolve', 1, x, 'table')
	checkType('Module:Array.convolve', 2, y, 'table')
	local z = {}
    local xLen, yLen = #x, #y
    for j = 1, (xLen + yLen - 1) do
        local sum = 0
        for k = math.max(1, j - yLen + 1), math.min(xLen, j) do
            sum = sum + x[k] * y[j-k+1]
        end
        z[j] = sum
    end
    return setmetatable(z, getmetatable(x) or getmetatable(y))
end

---Remove **nil** values from `arr` while preserving order.
---@generic T: any[]
---@param arr T
---@return T
function Array.condenseSparse(arr)
	checkType('Module:Array.condenseSparse', 1, arr, 'table')
	local keys = {}
	local res = {}
	local l = 0
	for k in pairs(arr) do
		l = l + 1
		keys[l] = k
	end
	table.sort(keys)
	for i =  1, l do
		res[i] = arr[keys[i]]
	end
	return setmetatable(res, getmetatable(arr))
end

---Behaviour depends on value of `val`:
---* `nil` - Counts the number of non **false** elements.
---* `fun(elem: any): boolean` - Count the number of times the function returned **true**.
---* `boolean` | `number` | `table` - Counts the number of times this value occurs in `arr`.
---@param arr any[]
---@param val? any
---@return integer
function Array.count(arr, val)
	checkType('Module:Array.count', 1, arr, 'table')
	if val == nil then val = function(item) return item end end
	if type(val) ~= 'function' then
		local _val = val
		val = function(item) return item == _val end
	end
	local count = 0
	for i = 1, #arr do
		if val(arr[i]) then
			count = count + 1
		end
	end
	return count
end

---Differentiate the array
---@generic T: number[]
---@param arr T
---@param order number? # Oder of the differentiation. Default is 1.
---@return T # Length is `#arr - order`
function Array.diff(arr, order)
	checkType('Module:Array.diff', 1, arr, 'table')
	checkType('Module:Array.diff', 2, order, 'number', true)
	local res = {}
	for i = 1, #arr - 1 do
		res[i] = arr[i+1] - arr[i]
	end
	if order and order > 1 then
		return Array.diff(res, order - 1)
	end
	return setmetatable(res, getmetatable(arr))
end

---Loops over `arr` and passes each element as the first argument to `fn`. This function returns nothing.
---@param arr any[]
---@param fn fun(elem: any, i?: integer)
function Array.each(arr, fn)
	checkType('Module:Array.each', 1, arr, 'table')
	checkType('Module:Array.each', 2, fn, 'function')
	local i = 1
	while arr[i] ~= nil do
		fn(arr[i], i)
		i = i + 1
	end
end

---Makes a copy of `arr` with only elements for which `fn` returned **true**.
---@generic T: any[]
---@param arr T
---@param fn fun(elem: any, i?: integer): boolean
---@return T
function Array.filter(arr, fn)
	checkType('Module:Array.filter', 1, arr, 'table')
	checkType('Module:Array.filter', 2, fn, 'function')
	local r = {}
	local len = 0
	local i = 1
	while arr[i] ~= nil do
		if fn(arr[i], i) then
			len = len + 1
			r[len] = arr[i]
		end
		i = i + 1
	end
	return setmetatable(r, getmetatable(arr))
end

---Find the first elements for which `fn` returns **true**.
---@param arr any[]
---@param fn any # A value to look for or a function of the form `fun(elem: any, i?: integer): boolean`.
---@param default? any # Value to return if no element passes the test.
---@return any? elem # The first element that passed the test.
---@return integer? i # The index of the item that passed the test.
function Array.find(arr, fn, default)
	checkType('Module:Array.find', 1, arr, 'table')
	checkTypeMulti('Module:Array.find_index', 2, fn, {'function', 'table', 'number', 'boolean'})
	if type(fn) ~= 'function' then
		local _val = fn
		fn = function(item) return item == _val end
	end
	local i = 1
	while arr[i] ~= nil do
		---@diagnostic disable-next-line: redundant-parameter
		if fn(arr[i], i) then
			return arr[i], i
		end
		i = i + 1
	end
	return default, nil
end

---Find the index of `val`.
---@param arr any[]
---@param val any # A value to look for or a function of the form `fun(elem: any, i?: integer): boolean`.
---@param default? any # Value to return if no element passes the test.
---@return integer?
function Array.find_index(arr, val, default)
	checkType('Module:Array.find_index', 1, arr, 'table')
	checkTypeMulti('Module:Array.find_index', 2, val, {'function', 'table', 'number', 'boolean'})
	if type(val) ~= 'function' then
		local _val = val
		val = function(item) return item == _val end
	end
	local i = 1
	while arr[i] ~= nil do
		---@diagnostic disable-next-line: redundant-parameter
		if val(arr[i], i) then
			return i
		end
		i = i + 1
	end
	return default
end

---Extracts a subset of `arr`.
---@generic T: any[]
---@param arr T
---@param indexes integer|integer[] # Indexes of the elements.
---@return T
function Array.get(arr, indexes)
	checkType('Module:Array.set', 1, arr, 'table')
	checkTypeMulti('Module:Array.set', 2, indexes, {'table', 'number'})
	if type(indexes) == 'number' then
		indexes = {indexes}
	end
	local res = {}
	for i = 1, #indexes do
		 res[i] = arr[indexes[i]]
	end
	return setmetatable(res, getmetatable(arr))
end

---Integrates the array. Effectively does $\left\{\sum^{n}_{start}{arr[n]} \,\Bigg|\, n \in [start, stop]\right\}$.
---@generic T: number[]
---@param arr T # number[]
---@param start? integer # Index where to start the summation. Defaults to 1.
---@param stop? integer # Index where to stop the summation. Defaults to #arr.
---@return T
function Array.int(arr, start, stop)
	checkType('Module:Array.int', 1, arr, 'table')
	checkType('Module:Array.int', 2, start, 'number', true)
	checkType('Module:Array.int', 3, stop, 'number', true)
	local res = {}
	start = start or 1
	stop = stop or #arr
	res[1] = arr[start]
	for i = 1, stop - start do
		res[i+1] = res[i] + arr[start + i]
	end
	return setmetatable(res, getmetatable(arr))
end

---Returns an array with elements that are present in both tables.
---@generic T: any[]
---@param arr1 T
---@param arr2 T
---@return T
function Array.intersect(arr1, arr2)
	checkType('Module:Array.intersect', 1, arr1, 'table')
	checkType('Module:Array.intersect', 2, arr2, 'table')
	local arr2Elements = {}
	local res = {}
	local len = 0
	Array.each(arr2, function(item) arr2Elements[item] = true end)
	Array.each(arr1, function(item)
		if arr2Elements[item] then
			len = len + 1
			res[len] = item
		end
	end)
	return setmetatable(res, getmetatable(arr1) or getmetatable(arr2))
end

---Checks if the two inputs have at least one element in common.
---@param arr1 any[]
---@param arr2 any[]
---@return boolean
function Array.intersects(arr1, arr2)
	checkType('Module:Array.intersects', 1, arr1, 'table')
	checkType('Module:Array.intersects', 2, arr2, 'table')
	local small = {}
	local large
	if #arr1 <= #arr2 then
		Array.each(arr1, function(item) small[item] = true end)
		large = arr2
	else
		Array.each(arr2, function(item) small[item] = true end)
		large = arr1
	end
	return Array.any(large, function(item) return small[item] end)
end

---Inserts values into `arr`.
---@generic T: any[]
---@param arr T
---@param val any # If `val` is an array and `unpackVal` is **true** then the individual elements of `val` are inserted.
---@param index? integer # Location to start the insertion. Default is at the end of `arr`.
---@param unpackVal? boolean # Default is **false**.
---@return T
---@overload fun(arr: T, val: any, unpackVal: boolean): T
function Array.insert(arr, val, index, unpackVal)
	checkType('Module:Array.insert', 1, arr, 'table')
	checkTypeMulti('Module:Array.insert', 3, index, {'number', 'boolean', 'nil'})
	checkType('Module:Array.insert', 4, unpackVal, 'boolean', true)
	if type(index) == 'boolean'  then
		unpackVal, index = index, nil
	end
	local len = #arr
	index = index or (len + 1)
	local mt = getmetatable(arr)
	setmetatable(arr, nil)

	if type(val) == 'table' and unpackVal then
		local len2 = #val
		for i = 0, len - index do
			arr[len + len2 - i] = arr[len - i]
		end
		for i = 0, len2 - 1 do
			arr[index + i] = val[i + 1]
		end
	else
		table.insert(arr, index, val)
	end

	return setmetatable(arr, mt)
end

---Returns the last element of `arr`.
---@param arr any[]
---@param offset? integer
---@return any
function Array.last(arr, offset)
	checkType('Module:Array.last', 1, arr, 'table')
	checkType('Module:Array.last', 2, offset, 'number', true)
	return arr[#arr + offset]
end

---Returns a new table were each element of `arr` is modified by `fn`.
---@generic T: any[]
---@param arr T
---@param fn fun(elem: any, i?: integer): any # First argument is the current element, the second argument is the index of the current element.
---@return T
function Array.map(arr, fn)
	checkType('Module:Array.map', 1, arr, 'table')
	checkType('Module:Array.map', 2, fn, 'function')
	local len = 0
	local r = {}
	local i = 1
	while arr[i] ~= nil do
		local tmp = fn(arr[i], i)
		if tmp ~= nil then
			len = len + 1
			r[len] = tmp
		end
		i = i + 1
	end
	return setmetatable(r, getmetatable(arr))
end

---Find the element for which `fn` returned the largest value.
---@param arr any[]
---@param fn fun(elem: any): any # The returned value needs to be comparable using the `<` operator.
---@return any elem # The element with the largest `fn` value.
---@return integer i # The index of this element.
function Array.max_by(arr, fn)
	checkType('Module:Array.max_by', 1, arr, 'table')
	checkType('Module:Array.max_by', 2, fn, 'function')
	return unpack(Array.reduce(arr, function(new, old, i)
		local y = fn(new)
		return y > old[2] and {new, y, i} or old
	end, {nil, -math.huge}))
end

---Find the largest value in the array.
---@param arr any[] # The values need to be comparable using the `<` operator.
---@return any elem
---@return integer i # The index of the largest value.
function Array.max(arr)
	checkType('Module:Array.max', 1, arr, 'table')
	local val, _, i = Array.max_by(arr, function(x) return x end)
	return val, i
end

---Find the smallest value in the array.
---@param arr any[] # The values need to be comparable using the `<` operator.
---@return any elem
---@return integer i # The index of the smallest value.
function Array.min(arr)
	checkType('Module:Array.min', 1, arr, 'table')
	local val, _, i = Array.max_by(arr, function(x) return -x end)
	return val, i
end

---Turn the input table into an Array. This makes it possible to use the colon `:` operator to access the Array methods.
---
---It also enables the use of math operators with the array.
---```
---local x = arr.new{ 1, 2, 3 }
---local y = arr{ 4, 5, 6 } -- Alternative notation
---
---print( -x ) --> { -1, -2, -3 }
---print( x + 2 ) --> { 3, 4, 5 }
---print( x - 2 ) --> { -1, 0, 1 }
---print( x * 2 ) --> { 2, 4, 6 }
---print( x / 2 ) --> { 0.5, 1, 1.5 }
---print( x ^ 2 ) --> { 1, 4, 9 }
---
---print( x + y ) --> { 5, 7, 9 }
---print( x .. y ) --> { 1, 2, 3, 4, 5, 6 }
---print( (x .. y):reject{3, 4, 5} ) --> { 1, 2, 6 }
---print( x:sum() ) --> 6
---
---print( x:update( {1, 3}, y:get{2, 3} * 2 ) ) --> { 10, 2, 12 }
---```
---@param arr? any[]
---@return Array
function Array.new(arr)
	local obj = arr or {}
	for _, v in pairs(obj) do
		if type(v) == 'table' then
			Array.new(v)
		end
	end

	if getmetatable(obj) == nil then
		setmetatable(obj, Array)
	end

	return obj
end

---Creates an object that returns a value that is `step` higher than the previous value each time it gets called.
---
---The stored value can be read without incrementing by reading the `val` field.
---
---A new stored value can be set through the `val` field.
---
---A new step size can be set through the `step` field.
---```
---local inc = arr.newIncrementor(10, 5)
---print( inc() ) --> 10
---print( inc() ) --> 15
---print( inc.val ) --> 15
---inc.val = 100
---inc.step = 20
---print( inc.val ) --> 100
---print( inc() ) --> 120
---```
---@param start? number # Default is 1.
---@param step? number # Default is 1.
---@return Incrementor
function Array.newIncrementor(start, step)
	checkType('Module:Array.newIncrementor', 1, start, 'number', true)
	checkType('Module:Array.newIncrementor', 2, step, 'number', true)
	step = step or 1
	local n = (start or 1) - step
	---@class Incrementor
	local obj = {}
	return setmetatable(obj, {
		__call = function() n = n + step return n end,
		__tostring = function() return n end,
		__index = function() return n end,
		__newindex = function(self, k, v)
			if k == 'step' and type(v) == 'number' then
				step = v
			elseif type(v) == 'number' then
				n = v
			end
		end,
		__concat = function(x, y) return tostring(x) .. tostring(y) end
	})
end

---Returns a range of numbers.
---@param start number # Start value inclusive.
---@param stop number # Stop value inclusive for integers, exclusive for floats.
---@param step? number # Default is 1.
---@return Array
---@overload fun(stop: number): Array
function Array.range(start, stop, step)
	checkType('Module:Array.range', 1, start, 'number')
	checkType('Module:Array.range', 2, stop, 'number', true)
	checkType('Module:Array.range', 3, step, 'number', true)
	local arr = {}
	local len = 0
	if not stop then
		stop = start
		start = 1
	end
	for i = start, stop, step or 1 do
		len = len + 1
		arr[len] = i
	end
	return setmetatable(arr, Array)
end

---Condenses the array into a single value.
---
---For each element `fn` is called with the current element, the current accumulator, and the current element index. The returned value of `fn` becomes the accumulator for the next element.
---
---If no `accumulator` value is given at the start then the first element off `arr` becomes the accumulator and the iteration starts from the second element.
---```
---local t = { 1, 2, 3, 4 }
---local sum = arr.reduce( t, function(elem, acc) return acc + elem end ) -- sum == 10
---```
---@param arr any[]
---@param fn fun(elem: any, acc: any, i?: integer): any # The result of this function becomes the `acc` for the next element.
---@param accumulator? any
---@return any # This is the last accumulator value.
function Array.reduce(arr, fn, accumulator)
	checkType('Module:Array.reduce', 1, arr, 'table')
	checkType('Module:Array.reduce', 2, fn, 'function')
	local acc = accumulator
	local i = 1
	if acc == nil then
		acc = arr[1]
		i = 2
	end
	while arr[i] ~= nil do
		acc = fn(arr[i], acc, i)
		i = i + 1
	end
	return acc
end

---Make a copy off `arr` with certain values removed.
---
---Behaviour for different values of `val`:
---* `boolean` | `number` - Remove values equal to this.
---* `table` - Remove all values in this table.
---* `fun(elem: any, i?: integer): boolean` - Remove elements for which the functions returns **true**.
---@generic T: any[]
---@param arr T
---@param val table|function|number|boolean
---@return T
function Array.reject(arr, val)
	checkType('Module:Array.reject', 1, arr, 'table')
	checkTypeMulti('Module:Array.reject', 2, val, {'function', 'table', 'number', 'boolean'})
	if type(val) ~= 'function' and type(val) ~= 'table' then
		val = {val}
	end
	local r = {}
	local len = 0
	if type(val) == 'function' then
		local i = 1
		while arr[i] ~= nil do
			if not val(arr[i], i) then
				len = len + 1
				r[len] = arr[i]
			end
			i = i + 1
		end
	else
		local rejectMap = {}
		Array.each(val --[[@as any[] ]], function(item) rejectMap[item] = true end)
		local i = 1
		while arr[i] ~= nil do
			if not rejectMap[arr[i]] then
				len = len + 1
				r[len] = arr[i]
			end
			i = i + 1
		end
	end
	return setmetatable(r, getmetatable(arr))
end

---Returns an Array with `val` repeated `n` times.
---@param val any
---@param n integer
---@return Array
function Array.rep(val, n)
	checkType('Module:Array.rep', 2, n, 'number')
	local r = {}
	for i = 1, n do
		r[i] = val
	end
	return setmetatable(r, Array)
end

---Condenses the array into a single value while saving every accumulator value.
---
---For each element `fn` is called with the current element, the current accumulator, and the current element index. The returned value of `fn` becomes the accumulator for the next element.
---
---If no `accumulator` value is given at the start then the first element off `arr` becomes the accumulator and the iteration starts from the second element.
---```
---local t = { 1, 2, 3, 4 }
---local x = arr.scan( t, function(elem, acc) return acc + elem end ) -- x = { 1, 3, 6, 10 }
---```
---@generic T: any[]
---@param arr T
---@param fn fun(elem: any, acc: any, i?: integer): any # Returned value becomes the accumulator for the next element.
---@param accumulator? any
---@return T
function Array.scan(arr, fn, accumulator)
	checkType('Module:Array.scan', 1, arr, 'table')
	checkType('Module:Array.scan', 2, fn, 'function')
	local acc = accumulator
	local r = {}
	local i = 1
	while arr[i] ~= nil do
		if i == 1 and not accumulator then
			acc = arr[i]
		else
			acc = fn(arr[i], acc, i)
		end
		r[i] = acc
		i = i + 1
	end
	return setmetatable(r, getmetatable(arr))
end

---Update a range of index with a range of values.
---
---If if only one value is given but multiple indexes than that value is set for all those indexes.
---
---If `values` is a table then it must of the same length as `indexes`.
---@generic T: any[]
---@param arr T
---@param indexes integer|integer[]
---@param values any|any[]
---@return T
function Array.set(arr, indexes, values)
	checkType('Module:Array.set', 1, arr, 'table')
	checkTypeMulti('Module:Array.set', 2, indexes, {'table', 'number'})
	local mt = getmetatable(arr)
	setmetatable(arr, nil)
	if type(indexes) == 'number' then
		indexes = {indexes}
	end
	if type(values) == 'table' then
		assert(#indexes == #values, string.format("Module:Array.set: 'indexes' and 'values' arrays are not equal length (#indexes = %d, #values = %d)", #indexes, #values))
		for i = 1, #indexes do
			arr[indexes[i]] = values[i]
		end
	else
		for i = 1, #indexes do
			arr[indexes[i]] = values
		end
	end
	return setmetatable(arr, mt)
end

---Extract a subtable from `arr`.
---@generic T: any[]
---@param arr T
---@param start integer # Start index. Use negative values to count form the end of the array.
---@param stop integer # Stop index. Use negative values to count form the end of the array.
---@return T
---@overload fun(arr: T, stop: integer): T
function Array.slice(arr, start, stop)
	checkType('Module:Array.slice', 1, arr, 'table')
	checkType('Module:Array.slice', 2, start, 'number', true)
	checkType('Module:Array.slice', 3, stop, 'number', true)
	start = start or #arr
	if start < 0 then
		start = #arr + start
	end
	if stop == nil then
		stop = start
		start = 1
	end
	if stop < 0 then
		stop = #arr + stop
	end
	local r = {}
	local len = 0
	for i = start, stop do
		len = len + 1
		r[len] = arr[i]
	end
	return setmetatable(r, getmetatable(arr))
end

---Split `arr` into two arrays.
---@generic T: any[]
---@param arr T
---@param index integer # Index to split on.
---@return T x # [1, index]
---@return T y # [index + 1, #arr]
function Array.split(arr, index)
	checkType('Module:Array.split', 1, arr, 'table')
	checkType('Module:Array.split', 2, index, 'number')
	local x = {}
	local y = {}
	for i = 1, #arr do
		table.insert(i <= index and x or y, arr[i])
	end
	return setmetatable(x, getmetatable(arr)), setmetatable(y, getmetatable(arr))
end

---Returns the sum of all elements of `arr`.
---@param arr number[]
---@return number
function Array.sum(arr)
	checkType('Module:Array.sum', 1, arr, 'table')
	local res = 0
	for i = 1, #arr do
		res = res + arr[i]
	end
	return res
end

---Extract a subtable from `arr`.
---@generic T: any[]
---@param arr T
---@param count integer # Length of the subtable.
---@param start? integer # Start index. Default is 1.
---@return T
function Array.take(arr, count, start)
	checkType('Module:Array.take', 1, arr, 'table')
	checkType('Module:Array.take', 2, count, 'number')
	checkType('Module:Array.take', 3, start, 'number', true)
	local x = {}
	start = start or 1
	for i = start, math.min(#arr, count + start - 1) do
		table.insert(x, arr[i])
	end
	return setmetatable(x, getmetatable(arr))
end

---Extract a subtable from `arr`.
---```
---local t = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 }
---local x = arr.take_every( t, 2 )       --> x = { 1, 3, 5, 7, 9 }
---local x = arr.take_every( t, 2, 3 )    --> x = { 1, 3, 5 }
---local x = arr.take_every( t, 2, 3, 2 ) --> x = { 2, 4, 6 }
--- ```
---@generic T: any[]
---@param arr T
---@param n integer # Step size.
---@param start? integer # Start index.
---@param count? integer # Max amount of elements to get.
---@return T
function Array.take_every(arr, n, start, count)
	checkType('Module:Array.take_every', 1, arr, 'table')
	checkType('Module:Array.take_every', 2, n, 'number')
	checkType('Module:Array.take_every', 3, start, 'number', true)
	checkType('Module:Array.take_every', 4, count, 'number', true)
	count = count or #arr
	local r = {}
	local len = 0
	local i = start or 1
	while arr[i] ~= nil and len < count do
		len = len + 1
		r[len] = arr[i]
		i = i + n
	end
	return setmetatable(r, getmetatable(arr))
end

---Return a new table with all duplicates removed.
---@generic T: any[]
---@param arr T
---@param fn? fun(elem: any): any # Function to generate an id for each element. The result will then contain elements that generated unique ids.
---@return T
function Array.unique(arr, fn)
	checkType('Module:Array.unique', 1, arr, 'table')
	checkType('Module:Array.unique', 2, fn, 'function', true)
	fn = fn or function(item) return item end
	local r = {}
	local len = 0
	local hash = {}
	local i = 1
	while arr[i] ~= nil do
		local id = fn(arr[i])
		if not hash[id] then
			len = len + 1
			r[len] = arr[i]
			hash[id] = true
		end
		i = i + 1
	end
	return setmetatable(r, getmetatable(arr))
end

---Combine elements with the same index from multiple arrays.
---```
---local x = {1, 2, 3}
---local y = {4, 5, 6, 7}
---local z = arr.zip( x, y ) --> z = { { 1, 4 }, { 2, 5 }, { 3, 6 }, { 7 } }
---```
---@param ... any[]
---@return Array
function Array.zip(...)
	local arrs = { ... }
	checkType('Module:Array.zip', 1, arrs[1], 'table')
	local r = {}
	local _, longest = Array.max_by(arrs, function(arr) return #arr end)
	for i = 1, longest do
		local q = {}
		for j = 1, #arrs do
			table.insert(q, arrs[j][i])
		end
		table.insert(r, setmetatable(q, Array))
	end
	return setmetatable(r, Array)
end

-- Range indexing has a performance impact so this is placed in a separate subclass
Array.RI_mt = {}
for k, v in pairs(Array) do
	Array.RI_mt[k] = v
end

function Array.RI_mt.__index(t, k)
	if type(k) == 'table' then
		local res = {}
		for i = 1, #k do
			res[i] = t[k[i]]
		end
		return setmetatable(res, Array)
	else
		return Array[k]
	end
end

function Array.RI_mt.__newindex(t, k, v)
	if type(k) == 'table' then
		if type(v) == 'table' then
			for i = 1, #k do
				t[k[i]] = v[i]
			end
		else
			for i = 1, #k do
				t[k[i]] = v
			end
		end
	else
		rawset(t, k, v)
	end
end

---Enable range indexing on the input array.
---
---This has a performance impact on reads and writes to the table.
---```
---local t = arr{10, 11, 12, 13, 14, 15}:ri()
---print( t[{2, 3}] ) --> { 11, 12 }
---```
---@param arr any[]
---@param recursive? boolean # Default is false.
---@return Array
function Array.ri(arr, recursive)
	checkType('Module:Array.ri', 1, arr, 'table')
	checkType('Module:Array.ri', 2, recursive, 'boolean', true)
	arr = arr or {}
	if recursive then
		for _, v in pairs(arr) do
			if type(v) == 'table' then
				Array.ri(v, true)
			end
		end
	end

	if getmetatable(arr) == nil or getmetatable(arr) == Array then
		setmetatable(arr, Array.RI_mt)
	end

	return arr
end

---Globally enable range indexing on all Array objects by default.
---@param set boolean
function Array.allwaysAllowRangeIndexing(set)
	checkType('Module:Array.allwaysAllowRangeIndexing', 1, set, 'boolean')
	if set then
		Array.__index = Array.RI_mt.__index
		Array.__newindex = Array.RI_mt.__newindex
	else
		Array.__index = Array
		Array.__newindex = nil
	end
end

return Array