1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
|
-module(gleam@set).
-compile([no_auto_import, nowarn_unused_vars, nowarn_unused_function]).
-export([new/0, size/1, insert/2, contains/2, delete/2, to_list/1, from_list/1, fold/3, filter/2, drop/2, take/2, union/2, intersection/2]).
-export_type([set/1]).
-opaque set(DJZ) :: {set, gleam@dict:dict(DJZ, list(nil))}.
-spec new() -> set(any()).
new() ->
{set, gleam@dict:new()}.
-spec size(set(any())) -> integer().
size(Set) ->
gleam@dict:size(erlang:element(2, Set)).
-spec insert(set(DKF), DKF) -> set(DKF).
insert(Set, Member) ->
{set, gleam@dict:insert(erlang:element(2, Set), Member, [])}.
-spec contains(set(DKI), DKI) -> boolean().
contains(Set, Member) ->
_pipe = erlang:element(2, Set),
_pipe@1 = gleam@dict:get(_pipe, Member),
gleam@result:is_ok(_pipe@1).
-spec delete(set(DKK), DKK) -> set(DKK).
delete(Set, Member) ->
{set, gleam@dict:delete(erlang:element(2, Set), Member)}.
-spec to_list(set(DKN)) -> list(DKN).
to_list(Set) ->
gleam@dict:keys(erlang:element(2, Set)).
-spec from_list(list(DKQ)) -> set(DKQ).
from_list(Members) ->
Map = gleam@list:fold(
Members,
gleam@dict:new(),
fun(M, K) -> gleam@dict:insert(M, K, []) end
),
{set, Map}.
-spec fold(set(DKT), DKV, fun((DKV, DKT) -> DKV)) -> DKV.
fold(Set, Initial, Reducer) ->
gleam@dict:fold(
erlang:element(2, Set),
Initial,
fun(A, K, _) -> Reducer(A, K) end
).
-spec filter(set(DKW), fun((DKW) -> boolean())) -> set(DKW).
filter(Set, Predicate) ->
{set,
gleam@dict:filter(erlang:element(2, Set), fun(M, _) -> Predicate(M) end)}.
-spec drop(set(DKZ), list(DKZ)) -> set(DKZ).
drop(Set, Disallowed) ->
gleam@list:fold(Disallowed, Set, fun delete/2).
-spec take(set(DLD), list(DLD)) -> set(DLD).
take(Set, Desired) ->
{set, gleam@dict:take(erlang:element(2, Set), Desired)}.
-spec order(set(DLH), set(DLH)) -> {set(DLH), set(DLH)}.
order(First, Second) ->
case gleam@dict:size(erlang:element(2, First)) > gleam@dict:size(
erlang:element(2, Second)
) of
true ->
{First, Second};
false ->
{Second, First}
end.
-spec union(set(DLM), set(DLM)) -> set(DLM).
union(First, Second) ->
{Larger, Smaller} = order(First, Second),
fold(Smaller, Larger, fun insert/2).
-spec intersection(set(DLQ), set(DLQ)) -> set(DLQ).
intersection(First, Second) ->
{Larger, Smaller} = order(First, Second),
take(Larger, to_list(Smaller)).
|