
"dict" to "List" and "Dict", or in Nomsu's environment, "_List" and "_Dict", removed uuid.lua and replaced it with core/id.nom for handling IDs.
105 lines
3.6 KiB
Plaintext
105 lines
3.6 KiB
Plaintext
-- This file contains container classes, i.e. Lists, Dicts, and Sets
|
|
|
|
{:insert,:remove,:concat} = table
|
|
{:repr, :stringify, :equivalent, :nth_to_last, :size} = require 'utils'
|
|
|
|
local List, Dict
|
|
|
|
-- List and Dict classes to provide basic equality/tostring functionality for the tables
|
|
-- used in Nomsu. This way, they retain a notion of whether they were originally lists or dicts.
|
|
_list_mt =
|
|
__eq:equivalent
|
|
-- Could consider adding a __newindex to enforce list-ness, but would hurt performance
|
|
__tostring: =>
|
|
"["..concat([repr(b) for b in *@], ", ").."]"
|
|
__lt: (other)=>
|
|
assert type(@) == 'table' and type(other) == 'table', "Incompatible types for comparison"
|
|
for i=1,math.max(#@, #other)
|
|
if not @[i] and other[i] then return true
|
|
elseif @[i] and not other[i] then return false
|
|
elseif @[i] < other[i] then return true
|
|
elseif @[i] > other[i] then return false
|
|
return false
|
|
__le: (other)=>
|
|
assert type(@) == 'table' and type(other) == 'table', "Incompatible types for comparison"
|
|
for i=1,math.max(#@, #other)
|
|
if not @[i] and other[i] then return true
|
|
elseif @[i] and not other[i] then return false
|
|
elseif @[i] < other[i] then return true
|
|
elseif @[i] > other[i] then return false
|
|
return true
|
|
__add: (other)=>
|
|
ret = List[x for x in *@]
|
|
for x in *other
|
|
insert(ret, x)
|
|
return ret
|
|
__index:
|
|
add_1: insert, append_1: insert
|
|
add_1_at_index_2: (t,x,i)-> insert(t,i,x)
|
|
at_index_1_add_2: insert
|
|
pop: remove, remove_last: remove, remove_index_1: remove
|
|
last: (=> @[#@]), first: (=> @[1])
|
|
_1_st_to_last:nth_to_last, _1_nd_to_last:nth_to_last
|
|
_1_rd_to_last:nth_to_last, _1_th_to_last:nth_to_last
|
|
has_1: (item)=>
|
|
for x in *@
|
|
if x == item
|
|
return true
|
|
return false
|
|
index_of_1: (item)=>
|
|
for i,x in ipairs @
|
|
if x == item
|
|
return i
|
|
return nil
|
|
-- TODO: remove this safety check to get better performance?
|
|
__newindex: (k,v)=>
|
|
assert type(k) == 'number', "List indices must be numbers"
|
|
rawset(@, k, v)
|
|
|
|
List = (t)-> setmetatable(t, _list_mt)
|
|
|
|
walk_items = (i)=>
|
|
i = i + 1
|
|
k, v = next(@table, @key)
|
|
if k != nil
|
|
@key = k
|
|
return i, Dict{key:k, value:v}
|
|
|
|
_dict_mt =
|
|
__eq:equivalent
|
|
__len:size
|
|
__tostring: =>
|
|
"{"..concat(["#{repr(k)}: #{repr(v)}" for k,v in pairs @], ", ").."}"
|
|
__ipairs: => walk_items, {table:@, key:nil}, 0
|
|
__band: (other)=>
|
|
Dict{k,v for k,v in pairs(@) when other[k] != nil}
|
|
__bor: (other)=>
|
|
ret = {k,v for k,v in pairs(@)}
|
|
for k,v in pairs(other)
|
|
if ret[k] == nil then ret[k] = v
|
|
return Dict(ret)
|
|
__bxor: (other)=>
|
|
ret = {k,v for k,v in pairs(@)}
|
|
for k,v in pairs(other)
|
|
if ret[k] == nil then ret[k] = v
|
|
else ret[k] = nil
|
|
return Dict(ret)
|
|
__add: (other)=>
|
|
ret = {k,v for k,v in pairs(@)}
|
|
for k,v in pairs(other)
|
|
if ret[k] == nil then ret[k] = v
|
|
else ret[k] += v
|
|
return Dict(ret)
|
|
__sub: (other)=>
|
|
ret = {k,v for k,v in pairs(@)}
|
|
for k,v in pairs(other)
|
|
if ret[k] == nil then ret[k] = -v
|
|
else ret[k] -= v
|
|
return Dict(ret)
|
|
Dict = (t)-> setmetatable(t, _dict_mt)
|
|
|
|
for i,entry in ipairs(Dict({x:99}))
|
|
assert(i == 1 and entry.key == "x" and entry.value == 99, "ipairs compatibility issue")
|
|
|
|
return {:List, :Dict}
|