@fireflysemantics/slice 15.0.18 → 15.0.24
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +1 -0
- package/esm2020/lib/AbstractStore.mjs +296 -0
- package/{esm2015/lib/EStore.js → esm2020/lib/EStore.mjs} +10 -10
- package/esm2020/lib/Slice.mjs +222 -0
- package/esm2020/lib/utilities.mjs +226 -0
- package/{esm2015/public-api.js → esm2020/public-api.mjs} +1 -1
- package/fesm2015/{fireflysemantics-slice.js → fireflysemantics-slice.mjs} +35 -22
- package/fesm2015/fireflysemantics-slice.mjs.map +1 -0
- package/fesm2020/fireflysemantics-slice.mjs +1494 -0
- package/fesm2020/fireflysemantics-slice.mjs.map +1 -0
- package/lib/models/Predicate.d.ts +1 -1
- package/lib/models/scrollPosition.d.ts +1 -1
- package/lib/utilities.d.ts +12 -1
- package/package.json +22 -9
- package/bundles/fireflysemantics-slice.umd.js +0 -1891
- package/bundles/fireflysemantics-slice.umd.js.map +0 -1
- package/esm2015/lib/AbstractStore.js +0 -296
- package/esm2015/lib/Slice.js +0 -222
- package/esm2015/lib/utilities.js +0 -213
- package/fesm2015/fireflysemantics-slice.js.map +0 -1
- /package/{esm2015/fireflysemantics-slice.js → esm2020/fireflysemantics-slice.mjs} +0 -0
- /package/{esm2015/lib/OStore.js → esm2020/lib/OStore.mjs} +0 -0
- /package/{esm2015/lib/models/ActionTypes.js → esm2020/lib/models/ActionTypes.mjs} +0 -0
- /package/{esm2015/lib/models/Delta.js → esm2020/lib/models/Delta.mjs} +0 -0
- /package/{esm2015/lib/models/Entity.js → esm2020/lib/models/Entity.mjs} +0 -0
- /package/{esm2015/lib/models/Predicate.js → esm2020/lib/models/Predicate.mjs} +0 -0
- /package/{esm2015/lib/models/StoreConfig.js → esm2020/lib/models/StoreConfig.mjs} +0 -0
- /package/{esm2015/lib/models/constants.js → esm2020/lib/models/constants.mjs} +0 -0
- /package/{esm2015/lib/models/index.js → esm2020/lib/models/index.mjs} +0 -0
- /package/{esm2015/lib/models/scrollPosition.js → esm2020/lib/models/scrollPosition.mjs} +0 -0
- /package/{fireflysemantics-slice.d.ts → index.d.ts} +0 -0
@@ -0,0 +1,222 @@
|
|
1
|
+
import { AbstractStore } from "./AbstractStore";
|
2
|
+
const { isArray } = Array;
|
3
|
+
export class Slice extends AbstractStore {
|
4
|
+
/**
|
5
|
+
* perform initial notification to all observers,
|
6
|
+
* such that operations like {@link combineLatest}{}
|
7
|
+
* will execute at least once.
|
8
|
+
*
|
9
|
+
* @param label The slice label
|
10
|
+
* @param predicate The slice predicate
|
11
|
+
* @param eStore The EStore instance containing the elements considered for slicing
|
12
|
+
*
|
13
|
+
* @example
|
14
|
+
<pre>
|
15
|
+
//Empty slice
|
16
|
+
new Slice<Todo>(Todo.COMPLETE, todo=>!todo.complete);
|
17
|
+
|
18
|
+
//Initialized slice
|
19
|
+
let todos = [new Todo(false, "You complete me!"),
|
20
|
+
new Todo(true, "You completed me!")];
|
21
|
+
new Slice<Todo>(Todo.COMPLETE, todo=>!todo.complete, todos);
|
22
|
+
</pre>
|
23
|
+
*/
|
24
|
+
constructor(label, predicate, eStore) {
|
25
|
+
super();
|
26
|
+
this.label = label;
|
27
|
+
this.predicate = predicate;
|
28
|
+
this.eStore = eStore;
|
29
|
+
/* The slice element entries */
|
30
|
+
this.entries = new Map();
|
31
|
+
const entities = eStore.allSnapshot();
|
32
|
+
this.config = eStore.config;
|
33
|
+
let passed = this.test(predicate, entities);
|
34
|
+
const delta = { type: "Initialize" /* ActionTypes.INTIALIZE */, entries: passed };
|
35
|
+
this.post(passed);
|
36
|
+
this.notifyDelta.next(delta);
|
37
|
+
}
|
38
|
+
/**
|
39
|
+
* Add the element if it satisfies the predicate
|
40
|
+
* and notify subscribers that an element was added.
|
41
|
+
*
|
42
|
+
* @param e The element to be considered for slicing
|
43
|
+
*/
|
44
|
+
post(e) {
|
45
|
+
if (isArray(e)) {
|
46
|
+
this.postA(e);
|
47
|
+
}
|
48
|
+
else {
|
49
|
+
if (this.predicate(e)) {
|
50
|
+
const id = e[this.config.guidKey];
|
51
|
+
this.entries.set(id, e);
|
52
|
+
const delta = { type: "Post" /* ActionTypes.POST */, entries: [e] };
|
53
|
+
this.notifyAll([...Array.from(this.entries.values())], delta);
|
54
|
+
}
|
55
|
+
}
|
56
|
+
}
|
57
|
+
/**
|
58
|
+
* Add the elements if they satisfy the predicate
|
59
|
+
* and notify subscribers that elements were added.
|
60
|
+
*
|
61
|
+
* @param e The element to be considered for slicing
|
62
|
+
*/
|
63
|
+
postN(...e) {
|
64
|
+
this.postA(e);
|
65
|
+
}
|
66
|
+
/**
|
67
|
+
* Add the elements if they satisfy the predicate
|
68
|
+
* and notify subscribers that elements were added.
|
69
|
+
*
|
70
|
+
* @param e The element to be considered for slicing
|
71
|
+
*/
|
72
|
+
postA(e) {
|
73
|
+
const d = [];
|
74
|
+
e.forEach(e => {
|
75
|
+
if (this.predicate(e)) {
|
76
|
+
const id = e[this.config.guidKey];
|
77
|
+
this.entries.set(id, e);
|
78
|
+
d.push(e);
|
79
|
+
}
|
80
|
+
});
|
81
|
+
const delta = { type: "Post" /* ActionTypes.POST */, entries: d };
|
82
|
+
this.notifyAll([...Array.from(this.entries.values())], delta);
|
83
|
+
}
|
84
|
+
/**
|
85
|
+
* Delete an element from the slice.
|
86
|
+
*
|
87
|
+
* @param e The element to be deleted if it satisfies the predicate
|
88
|
+
*/
|
89
|
+
delete(e) {
|
90
|
+
if (isArray(e)) {
|
91
|
+
this.deleteA(e);
|
92
|
+
}
|
93
|
+
else {
|
94
|
+
if (this.predicate(e)) {
|
95
|
+
const id = e[this.config.guidKey];
|
96
|
+
this.entries.delete(id);
|
97
|
+
const delta = { type: "Delete" /* ActionTypes.DELETE */, entries: [e] };
|
98
|
+
this.notifyAll(Array.from(this.entries.values()), delta);
|
99
|
+
}
|
100
|
+
}
|
101
|
+
}
|
102
|
+
/**
|
103
|
+
* @param e The elements to be deleted if it satisfies the predicate
|
104
|
+
*/
|
105
|
+
deleteN(...e) {
|
106
|
+
this.deleteA(e);
|
107
|
+
}
|
108
|
+
/**
|
109
|
+
* @param e The elements to be deleted if they satisfy the predicate
|
110
|
+
*/
|
111
|
+
deleteA(e) {
|
112
|
+
const d = [];
|
113
|
+
e.forEach(e => {
|
114
|
+
if (this.predicate(e)) {
|
115
|
+
const id = e[this.config.guidKey];
|
116
|
+
d.push(this.entries.get(id));
|
117
|
+
this.entries.delete(id);
|
118
|
+
}
|
119
|
+
});
|
120
|
+
const delta = { type: "Delete" /* ActionTypes.DELETE */, entries: d };
|
121
|
+
this.notifyAll([...Array.from(this.entries.values())], delta);
|
122
|
+
}
|
123
|
+
/**
|
124
|
+
* Update the slice when an Entity instance mutates.
|
125
|
+
*
|
126
|
+
* @param e The element to be added or deleted depending on predicate reevaluation
|
127
|
+
*/
|
128
|
+
put(e) {
|
129
|
+
if (isArray(e)) {
|
130
|
+
this.putA(e);
|
131
|
+
}
|
132
|
+
else {
|
133
|
+
const id = e[this.config.guidKey];
|
134
|
+
if (this.entries.get(id)) {
|
135
|
+
if (!this.predicate(e)) {
|
136
|
+
//Note that this is a ActionTypes.DELETE because we are removing the
|
137
|
+
//entity from the slice.
|
138
|
+
const delta = { type: "Delete" /* ActionTypes.DELETE */, entries: [e] };
|
139
|
+
this.entries.delete(id);
|
140
|
+
this.notifyAll([...Array.from(this.entries.values())], delta);
|
141
|
+
}
|
142
|
+
}
|
143
|
+
else if (this.predicate(e)) {
|
144
|
+
this.entries.set(id, e);
|
145
|
+
const delta = { type: "Put" /* ActionTypes.PUT */, entries: [e] };
|
146
|
+
this.notifyAll([...Array.from(this.entries.values())], delta);
|
147
|
+
}
|
148
|
+
}
|
149
|
+
}
|
150
|
+
/**
|
151
|
+
* Update the slice with mutated Entity instances.
|
152
|
+
*
|
153
|
+
* @param e The elements to be deleted if it satisfies the predicate
|
154
|
+
*/
|
155
|
+
putN(...e) {
|
156
|
+
this.putA(e);
|
157
|
+
}
|
158
|
+
/**
|
159
|
+
* @param e The elements to be put
|
160
|
+
*/
|
161
|
+
putA(e) {
|
162
|
+
const d = []; //instances to delete
|
163
|
+
const u = []; //instances to update
|
164
|
+
e.forEach(e => {
|
165
|
+
const id = e[this.config.guidKey];
|
166
|
+
if (this.entries.get(id)) {
|
167
|
+
if (!this.predicate(e)) {
|
168
|
+
d.push(this.entries.get(id));
|
169
|
+
}
|
170
|
+
}
|
171
|
+
else if (this.predicate(e)) {
|
172
|
+
u.push(e);
|
173
|
+
}
|
174
|
+
});
|
175
|
+
if (d.length > 0) {
|
176
|
+
d.forEach(e => {
|
177
|
+
this.entries.delete(e[this.config.guidKey]);
|
178
|
+
});
|
179
|
+
const delta = { type: "Delete" /* ActionTypes.DELETE */, entries: d };
|
180
|
+
this.notifyAll([...Array.from(this.entries.values())], delta);
|
181
|
+
}
|
182
|
+
if (u.length > 0) {
|
183
|
+
u.forEach(e => {
|
184
|
+
this.entries.set(e[this.config.guidKey], e);
|
185
|
+
});
|
186
|
+
const delta = { type: "Put" /* ActionTypes.PUT */, entries: u };
|
187
|
+
this.notifyAll([...Array.from(this.entries.values())], delta);
|
188
|
+
}
|
189
|
+
}
|
190
|
+
/**
|
191
|
+
* Resets the slice to empty.
|
192
|
+
*/
|
193
|
+
reset() {
|
194
|
+
let delta = {
|
195
|
+
type: "Reset" /* ActionTypes.RESET */,
|
196
|
+
entries: [...Array.from(this.entries.values())]
|
197
|
+
};
|
198
|
+
this.notifyAll([], delta);
|
199
|
+
this.entries = new Map();
|
200
|
+
}
|
201
|
+
/**
|
202
|
+
* Utility method that applies the predicate to an array
|
203
|
+
* of entities and return the ones that pass the test.
|
204
|
+
*
|
205
|
+
* Used to create an initial set of values
|
206
|
+
* that should be part of the `Slice`.
|
207
|
+
*
|
208
|
+
* @param p
|
209
|
+
* @param e
|
210
|
+
* @return The the array of entities that pass the predicate test.
|
211
|
+
*/
|
212
|
+
test(p, e) {
|
213
|
+
let v = [];
|
214
|
+
e.forEach((e) => {
|
215
|
+
if (p(e)) {
|
216
|
+
v.push(e);
|
217
|
+
}
|
218
|
+
});
|
219
|
+
return v;
|
220
|
+
}
|
221
|
+
}
|
222
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Slice.js","sourceRoot":"","sources":["../../../../projects/slice/src/lib/Slice.ts"],"names":[],"mappings":"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","sourcesContent":["import { Delta, ActionTypes, Predicate } from \"./models\"\nimport { AbstractStore } from \"./AbstractStore\"\nimport { EStore } from \"./EStore\";\n\nconst { isArray } = Array\n\nexport class Slice<E> extends AbstractStore<E> {\n\n\n    /* The slice element entries */\n    public override entries: Map<string, E> = new Map();\n\n    /**\n     * perform initial notification to all observers,\n     * such that operations like {@link combineLatest}{}\n     * will execute at least once.\n     * \n     * @param label The slice label\n     * @param predicate The slice predicate\n     * @param eStore The EStore instance containing the elements considered for slicing\n     * \n     * @example \n       <pre>\n       //Empty slice\n       new Slice<Todo>(Todo.COMPLETE, todo=>!todo.complete);\n  \n       //Initialized slice\n       let todos = [new Todo(false, \"You complete me!\"), \n                    new Todo(true, \"You completed me!\")];\n       new Slice<Todo>(Todo.COMPLETE, todo=>!todo.complete, todos);\n       </pre>\n     */\n    constructor(\n        public label: string,\n        public predicate: Predicate<E>,\n        public eStore: EStore<E>) {\n        super();\n        const entities: E[] = eStore.allSnapshot()\n        this.config = eStore.config\n        let passed: E[] = this.test(predicate, entities);\n        const delta: Delta<E> = { type: ActionTypes.INTIALIZE, entries: passed };\n        this.post(passed);\n        this.notifyDelta.next(delta)\n    }\n\n    /**\n     * Add the element if it satisfies the predicate\n     * and notify subscribers that an element was added.\n     *\n     * @param e The element to be considered for slicing\n     */\n    post(e: E | E[]) {\n        if (isArray(e)) {\n            this.postA(e)\n        }\n        else {\n            if (this.predicate(e)) {\n                const id = (<any>e)[this.config.guidKey];\n                this.entries.set(id, e);\n                const delta: Delta<E> = { type: ActionTypes.POST, entries: [e] };\n                this.notifyAll([...Array.from(this.entries.values())], delta);\n            }\n        }\n    }\n\n    /**\n     * Add the elements if they satisfy the predicate\n     * and notify subscribers that elements were added.\n     *\n     * @param e The element to be considered for slicing\n     */\n    postN(...e: E[]) {\n        this.postA(e);\n    }\n\n    /**\n     * Add the elements if they satisfy the predicate\n     * and notify subscribers that elements were added.\n     *\n     * @param e The element to be considered for slicing\n     */\n    postA(e: E[]) {\n        const d: E[] = [];\n        e.forEach(e => {\n            if (this.predicate(e)) {\n                const id = (<any>e)[this.config.guidKey];\n                this.entries.set(id, e);\n                d.push(e);\n            }\n        });\n        const delta: Delta<E> = { type: ActionTypes.POST, entries: d };\n        this.notifyAll([...Array.from(this.entries.values())], delta);\n    }\n\n    /**\n     * Delete an element from the slice.\n     *\n     * @param e The element to be deleted if it satisfies the predicate\n     */\n    delete(e: E | E[]) {\n        if (isArray(e)) {\n            this.deleteA(e)\n        }\n        else {\n            if (this.predicate(e)) {\n                const id = (<any>e)[this.config.guidKey]\n                this.entries.delete(id)\n                const delta: Delta<E> = { type: ActionTypes.DELETE, entries: [e] }\n                this.notifyAll(Array.from(this.entries.values()), delta)\n            }\n        }\n    }\n\n    /**\n     * @param e The elements to be deleted if it satisfies the predicate\n     */\n    deleteN(...e: E[]) {\n        this.deleteA(e);\n    }\n\n    /**\n     * @param e The elements to be deleted if they satisfy the predicate\n     */\n    deleteA(e: E[]) {\n        const d: E[] = []\n        e.forEach(e => {\n            if (this.predicate(e)) {\n                const id = (<any>e)[this.config.guidKey]\n                d.push(this.entries.get(id)!)\n                this.entries.delete(id)\n            }\n        });\n        const delta: Delta<E> = { type: ActionTypes.DELETE, entries: d };\n        this.notifyAll([...Array.from(this.entries.values())], delta);\n    }\n\n    /**\n     * Update the slice when an Entity instance mutates.\n     *\n     * @param e The element to be added or deleted depending on predicate reevaluation\n     */\n    put(e: E | E[]) {\n        if (isArray(e)) {\n            this.putA(e)\n        }\n        else {\n            const id = (<any>e)[this.config.guidKey];\n            if (this.entries.get(id)) {\n                if (!this.predicate(e)) {\n                    //Note that this is a ActionTypes.DELETE because we are removing the\n                    //entity from the slice.\n                    const delta: Delta<E> = { type: ActionTypes.DELETE, entries: [e] };\n                    this.entries.delete(id);\n                    this.notifyAll([...Array.from(this.entries.values())], delta);\n                }\n            } else if (this.predicate(e)) {\n                this.entries.set(id, e);\n                const delta: Delta<E> = { type: ActionTypes.PUT, entries: [e] };\n                this.notifyAll([...Array.from(this.entries.values())], delta);\n            }    \n        }\n    }\n\n    /**\n     * Update the slice with mutated Entity instances.\n     *\n     * @param e The elements to be deleted if it satisfies the predicate\n     */\n    putN(...e: E[]) {\n        this.putA(e);\n    }\n\n    /**\n     * @param e The elements to be put\n     */\n    putA(e: E[]) {\n        const d: E[] = []; //instances to delete\n        const u: E[] = []; //instances to update\n        e.forEach(e => {\n            const id = (<any>e)[this.config.guidKey];\n            if (this.entries.get(id)) {\n                if (!this.predicate(e)) {\n                    d.push(this.entries.get(id)!);\n                }\n            } else if (this.predicate(e)) {\n                u.push(e);\n            }\n        });\n        if (d.length > 0) {\n            d.forEach(e => {\n                this.entries.delete((<any>e)[this.config.guidKey])\n            });\n            const delta: Delta<E> = { type: ActionTypes.DELETE, entries: d };\n            this.notifyAll([...Array.from(this.entries.values())], delta);\n        }\n        if (u.length > 0) {\n            u.forEach(e => {\n                this.entries.set((<any>e)[this.config.guidKey], e);\n            });\n            const delta: Delta<E> = { type: ActionTypes.PUT, entries: u };\n            this.notifyAll([...Array.from(this.entries.values())], delta);\n        }\n    }\n\n    /**\n     * Resets the slice to empty.\n     */\n    reset() {\n        let delta: Delta<E> = {\n            type: ActionTypes.RESET,\n            entries: [...Array.from(this.entries.values())]\n        };\n        this.notifyAll([], delta);\n        this.entries = new Map();\n    }\n\n    /**\n     * Utility method that applies the predicate to an array\n     * of entities and return the ones that pass the test.\n     *\n     * Used to create an initial set of values\n     * that should be part of the `Slice`.\n     *\n     * @param p\n     * @param e\n     * @return The the array of entities that pass the predicate test.\n     */\n    public test(p: Predicate<E>, e: E[]): E[] {\n        let v: E[] = [];\n        e.forEach((e: E) => {\n            if (p(e)) {\n                v.push(e);\n            }\n        });\n        return v;\n    }\n}\n"]}
|
@@ -0,0 +1,226 @@
|
|
1
|
+
import { ESTORE_CONFIG_DEFAULT } from "./AbstractStore";
|
2
|
+
import { fromEvent, of } from 'rxjs';
|
3
|
+
import { switchMap, pairwise, debounceTime, distinctUntilChanged, map, filter } from 'rxjs/operators';
|
4
|
+
import { nanoid } from "nanoid";
|
5
|
+
/**
|
6
|
+
* Returns all the entities are distinct by the
|
7
|
+
* `property` value argument.
|
8
|
+
*
|
9
|
+
* Note that the implementation uses a `Map<string, E>` to
|
10
|
+
* index the entities by key. Therefore the more recent occurences
|
11
|
+
* matching a key instance will overwrite the previous ones.
|
12
|
+
*
|
13
|
+
* @param property The name of the property to check for distinct values by.
|
14
|
+
* @param entities The entities in the array.
|
15
|
+
*
|
16
|
+
* @example
|
17
|
+
```
|
18
|
+
let todos: Todo[] = [
|
19
|
+
{ id: 1, title: "Lets do it!" },
|
20
|
+
{ id: 1, title: "Lets do it again!" },
|
21
|
+
{ id: 2, title: "All done!" }
|
22
|
+
];
|
23
|
+
|
24
|
+
let todos2: Todo[] = [
|
25
|
+
{ id: 1, title: "Lets do it!" },
|
26
|
+
{ id: 2, title: "All done!" }
|
27
|
+
];
|
28
|
+
|
29
|
+
expect(distinct(todos, "id").length).toEqual(2);
|
30
|
+
expect(distinct(todos2, "id").length).toEqual(2);
|
31
|
+
|
32
|
+
```
|
33
|
+
*/
|
34
|
+
export function distinct(entities, property) {
|
35
|
+
const entitiesByProperty = new Map(entities.map(e => [e[property], e]));
|
36
|
+
return Array.from(entitiesByProperty.values());
|
37
|
+
}
|
38
|
+
/**
|
39
|
+
* Returns true if all the entities are distinct by the
|
40
|
+
* `property` value argument.
|
41
|
+
*
|
42
|
+
* @param property The name of the property to check for distinct values by.
|
43
|
+
* @param entities The entities in the array.
|
44
|
+
*
|
45
|
+
* @example
|
46
|
+
*
|
47
|
+
```
|
48
|
+
let todos: Todo[] = [
|
49
|
+
{ id: 1, title: "Lets do it!" },
|
50
|
+
{ id: 1, title: "Lets do it again!" },
|
51
|
+
{ id: 2, title: "All done!" }
|
52
|
+
];
|
53
|
+
|
54
|
+
let todos2: Todo[] = [
|
55
|
+
{ id: 1, title: "Lets do it!" },
|
56
|
+
{ id: 2, title: "All done!" }
|
57
|
+
];
|
58
|
+
|
59
|
+
expect(unique(todos, "id")).toBeFalsy();
|
60
|
+
expect(unique(todos2, "id")).toBeTruthy();
|
61
|
+
```
|
62
|
+
*/
|
63
|
+
export function unique(entities, property) {
|
64
|
+
return entities.length == distinct(entities, property).length ? true : false;
|
65
|
+
}
|
66
|
+
/**
|
67
|
+
* Create a global ID
|
68
|
+
* @return The global id.
|
69
|
+
*
|
70
|
+
* @example
|
71
|
+
* let e.guid = GUID();
|
72
|
+
*/
|
73
|
+
export function GUID() {
|
74
|
+
return nanoid();
|
75
|
+
}
|
76
|
+
/**
|
77
|
+
* Set the global identfication property on the instance.
|
78
|
+
*
|
79
|
+
* @param e Entity we want to set the global identifier on.
|
80
|
+
* @param gid The name of the `gid` property. If not specified it defaults to `ESTORE_CONFIG_DEFAULT.guidKey`.
|
81
|
+
*/
|
82
|
+
export function attachGUID(e, gid) {
|
83
|
+
const guidKey = gid ? gid : ESTORE_CONFIG_DEFAULT.guidKey;
|
84
|
+
let id = nanoid();
|
85
|
+
e[guidKey] = id;
|
86
|
+
return id;
|
87
|
+
}
|
88
|
+
/**
|
89
|
+
* Set the global identfication property on the instance.
|
90
|
+
*
|
91
|
+
* @param e[] Entity array we want to set the global identifiers on.
|
92
|
+
* @param gid The name of the `gid` property. If not specified it defaults to `gid`.
|
93
|
+
*/
|
94
|
+
export function attachGUIDs(e, gid) {
|
95
|
+
e.forEach(e => {
|
96
|
+
attachGUID(e, gid);
|
97
|
+
});
|
98
|
+
}
|
99
|
+
/**
|
100
|
+
* Create a shallow copy of the argument.
|
101
|
+
* @param o The object to copy
|
102
|
+
*/
|
103
|
+
export function shallowCopy(o) {
|
104
|
+
return { ...o };
|
105
|
+
}
|
106
|
+
/**
|
107
|
+
* Create a deep copy of the argument.
|
108
|
+
* @param o The object to copy
|
109
|
+
*/
|
110
|
+
export function deepCopy(o) {
|
111
|
+
return JSON.parse(JSON.stringify(o));
|
112
|
+
}
|
113
|
+
/**
|
114
|
+
* Gets the current active value from the `active`
|
115
|
+
* Map.
|
116
|
+
*
|
117
|
+
* This is used for the scenario where we are managing
|
118
|
+
* a single active instance. For example
|
119
|
+
* when selecting a book from a collection of books.
|
120
|
+
*
|
121
|
+
* The selected `Book` instance becomes the active value.
|
122
|
+
*
|
123
|
+
* @example
|
124
|
+
* const book:Book = getActiveValue(bookStore.active);
|
125
|
+
* @param m
|
126
|
+
*/
|
127
|
+
export function getActiveValue(m) {
|
128
|
+
if (m.size) {
|
129
|
+
return m.entries().next().value[1];
|
130
|
+
}
|
131
|
+
return null;
|
132
|
+
}
|
133
|
+
/**
|
134
|
+
* The method can be used to exclude keys from an instance
|
135
|
+
* of type `E`.
|
136
|
+
*
|
137
|
+
* We can use this to exclude values when searching an object.
|
138
|
+
*
|
139
|
+
* @param entity An instance of type E
|
140
|
+
* @param exclude The keys to exclude
|
141
|
+
*
|
142
|
+
* @example
|
143
|
+
* todo = { id: '1', description: 'Do it!' }
|
144
|
+
* let keys = excludeKeys<Todo>(todo, ['id]);
|
145
|
+
* // keys = ['description']
|
146
|
+
*/
|
147
|
+
export function excludeKeys(entity, exclude) {
|
148
|
+
const keys = Object.keys(entity);
|
149
|
+
return keys.filter((key) => {
|
150
|
+
return exclude.indexOf(key) < 0;
|
151
|
+
});
|
152
|
+
}
|
153
|
+
/**
|
154
|
+
*
|
155
|
+
* @param entities The entity to search
|
156
|
+
* @param exclude Keys to exclude from each entity
|
157
|
+
*
|
158
|
+
* @return E[] Array of entities with properties containing the search term.
|
159
|
+
*/
|
160
|
+
export function search(query = '', entities, exclude = []) {
|
161
|
+
const { isArray } = Array;
|
162
|
+
query = query.toLowerCase();
|
163
|
+
return entities.filter(function (e) {
|
164
|
+
//Do the keys calculation on each instance e:E
|
165
|
+
//because an instance can have optional parameters,
|
166
|
+
//and thus we have to check each instance, not just
|
167
|
+
//the first one in the array.
|
168
|
+
const keys = excludeKeys(e, exclude);
|
169
|
+
return keys.some((key) => {
|
170
|
+
const value = e[key];
|
171
|
+
if (!value) {
|
172
|
+
return false;
|
173
|
+
}
|
174
|
+
if (isArray(value)) {
|
175
|
+
return value.some(v => {
|
176
|
+
return String(v).toLowerCase().includes(query);
|
177
|
+
});
|
178
|
+
}
|
179
|
+
else {
|
180
|
+
return String(value).toLowerCase().includes(query);
|
181
|
+
}
|
182
|
+
});
|
183
|
+
});
|
184
|
+
}
|
185
|
+
/**
|
186
|
+
* @param scrollable The element being scrolled
|
187
|
+
* @param debounceMS The number of milliseconds to debounce scroll events
|
188
|
+
* @param sp The function returning the scroll position coordinates.
|
189
|
+
* @return A boolean valued observable indicating whether the element is scrolling up or down
|
190
|
+
*/
|
191
|
+
export function scrollingUp(scrollable, debounceMS, sp) {
|
192
|
+
return fromEvent(scrollable, 'scroll').pipe(debounceTime(debounceMS), distinctUntilChanged(), map(v => sp()), pairwise(), switchMap(p => {
|
193
|
+
const y1 = p[0][1];
|
194
|
+
const y2 = p[1][1];
|
195
|
+
return y1 - y2 > 0 ? of(false) : of(true);
|
196
|
+
}));
|
197
|
+
}
|
198
|
+
/**
|
199
|
+
* Filters the entities properties to the set contained in the
|
200
|
+
* `keys` array.
|
201
|
+
*
|
202
|
+
* @param keys The array of keys that the entity be limited to
|
203
|
+
* @param entity The entity to map
|
204
|
+
* @return An entity instance that has only the keys provided in the keys array
|
205
|
+
*/
|
206
|
+
export function mapEntity(keys, entity) {
|
207
|
+
const result = {};
|
208
|
+
keys.forEach(k => {
|
209
|
+
result[k] = entity[k];
|
210
|
+
});
|
211
|
+
return result;
|
212
|
+
}
|
213
|
+
/**
|
214
|
+
* Returns an `Observable<E>` instance that
|
215
|
+
* filters for arguments where the property
|
216
|
+
* value matches the provided value.
|
217
|
+
*
|
218
|
+
* @param value The value targeted
|
219
|
+
* @param propertyName The name of the property to contain the value
|
220
|
+
* @param obs The Slice Object Store Observable
|
221
|
+
* @returns Observable<E>
|
222
|
+
*/
|
223
|
+
export function onFilteredEvent(value, propertyName, obs) {
|
224
|
+
return obs.pipe(filter((e) => !!(e && e[propertyName] === value)));
|
225
|
+
}
|
226
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utilities.js","sourceRoot":"","sources":["../../../../projects/slice/src/lib/utilities.ts"],"names":[],"mappings":"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","sourcesContent":["import { ESTORE_CONFIG_DEFAULT } from \"./AbstractStore\";\nimport { Observable, fromEvent, of } from 'rxjs'\nimport { switchMap, pairwise, debounceTime, distinctUntilChanged, map, filter } from 'rxjs/operators'\nimport { nanoid} from \"nanoid\"\nimport { scrollPosition } from \"./models/scrollPosition\";\n\n/**\n * Returns all the entities are distinct by the \n * `property` value argument.  \n * \n * Note that the implementation uses a `Map<string, E>` to\n * index the entities by key.  Therefore the more recent occurences \n * matching a key instance will overwrite the previous ones.\n * \n * @param property The name of the property to check for distinct values by.\n * @param entities The entities in the array.\n * \n * @example\n ```\n  let todos: Todo[] = [\n    { id: 1, title: \"Lets do it!\" },\n    { id: 1, title: \"Lets do it again!\" },\n    { id: 2, title: \"All done!\" }\n  ];\n\n  let todos2: Todo[] = [\n    { id: 1, title: \"Lets do it!\" },\n    { id: 2, title: \"All done!\" }\n  ];\n\n  expect(distinct(todos, \"id\").length).toEqual(2);\n  expect(distinct(todos2, \"id\").length).toEqual(2);\n\n ```\n */\nexport function distinct<E, K extends keyof E>(entities: E[], property: K): E[] {\n  const entitiesByProperty = new Map(entities.map(e => [e[property], e] as [E[K], E]));\n  return Array.from(entitiesByProperty.values());\n}\n\n/**\n * Returns true if all the entities are distinct by the \n * `property` value argument.\n * \n * @param property The name of the property to check for distinct values by.\n * @param entities The entities in the array.\n * \n * @example\n * \n ```\n  let todos: Todo[] = [\n    { id: 1, title: \"Lets do it!\" },\n    { id: 1, title: \"Lets do it again!\" },\n    { id: 2, title: \"All done!\" }\n  ];\n\n  let todos2: Todo[] = [\n    { id: 1, title: \"Lets do it!\" },\n    { id: 2, title: \"All done!\" }\n  ];\n\n  expect(unique(todos, \"id\")).toBeFalsy();\n  expect(unique(todos2, \"id\")).toBeTruthy();\n ```\n */\nexport function unique<E>(entities: E[], property: keyof E):boolean {\n  return entities.length == distinct(entities, property).length ? true : false;\n}\n\n/**\n * Create a global ID\n * @return The global id.\n * \n * @example\n * let e.guid = GUID();\n */\nexport function GUID() {\n  return nanoid();\n}\n\n/**\n * Set the global identfication property on the instance.\n * \n * @param e Entity we want to set the global identifier on.\n * @param gid The name of the `gid` property.  If not specified it defaults to `ESTORE_CONFIG_DEFAULT.guidKey`.\n */\nexport function attachGUID<E>(e: E, gid?: string): string {\n  const guidKey = gid ? gid : ESTORE_CONFIG_DEFAULT.guidKey\n  let id: string = nanoid();\n  (<any>e)[guidKey] = id\n  return id\n}\n\n/**\n * Set the global identfication property on the instance.\n * \n * @param e[] Entity array we want to set the global identifiers on.\n * @param gid The name of the `gid` property.  If not specified it defaults to `gid`.\n */\nexport function attachGUIDs<E>(e: E[], gid?: string) {\n  e.forEach(e => {\n    attachGUID(e, gid);\n  });\n}\n\n/**\n * Create a shallow copy of the argument.\n * @param o The object to copy\n */\nexport function shallowCopy<E>(o: E) {\n  return { ...o };\n}\n\n/**\n * Create a deep copy of the argument.\n * @param o The object to copy\n */\nexport function deepCopy<E>(o: E) {\n  return JSON.parse(JSON.stringify(o));\n}\n\n/**\n * Gets the current active value from the `active`\n * Map.  \n * \n * This is used for the scenario where we are managing\n * a single active instance.  For example \n * when selecting a book from a collection of books.  \n * \n * The selected `Book` instance becomes the active value.\n * \n * @example\n * const book:Book = getActiveValue(bookStore.active);\n * @param m \n */\nexport function getActiveValue<E>(m: Map<any, E>) {\n  if (m.size) {\n    return m.entries().next().value[1];\n  }\n  return null;\n}\n\n/**\n * The method can be used to exclude keys from an instance\n * of type `E`.  \n * \n * We can use this to exclude values when searching an object.\n * \n * @param entity An instance of type E\n * @param exclude The keys to exclude\n * \n * @example\n * todo = { id: '1', description: 'Do it!' }\n * let keys = excludeKeys<Todo>(todo, ['id]);\n * // keys = ['description']\n */\nexport function excludeKeys<E>(entity: any, exclude: string[]) {\n  const keys: string[] = Object.keys(entity);\n  return keys.filter((key) => {\n    return exclude.indexOf(key) < 0;\n  });\n}\n\n/**\n * \n * @param entities The entity to search\n * @param exclude Keys to exclude from each entity\n * \n * @return E[] Array of entities with properties containing the search term.\n */\nexport function search<E>(query: string = '', entities: E[], exclude: string[] = []): E[] {\n  const { isArray } = Array\n\n  query = query.toLowerCase();\n\n\n  return entities.filter(function (e: E) {\n    //Do the keys calculation on each instance e:E\n    //because an instance can have optional parameters,\n    //and thus we have to check each instance, not just\n    //the first one in the array.\n    const keys = excludeKeys(e, exclude)\n    return keys.some( (key) => {\n      const value = (e as any)[key];\n      if (!value) {\n        return false;\n      }\n      if (isArray(value)) {\n        return value.some(v => {\n          return String(v).toLowerCase().includes(query);\n        });\n      }\n      else {\n        return String(value).toLowerCase().includes(query);\n      }\n    })\n  });\n}\n\n/**\n * @param scrollable The element being scrolled\n * @param debounceMS The number of milliseconds to debounce scroll events\n * @param sp The function returning the scroll position coordinates.\n * @return A boolean valued observable indicating whether the element is scrolling up or down\n */\nexport function scrollingUp(\n  scrollable: any, \n  debounceMS: number, \n  sp: scrollPosition): Observable<boolean> {\n  return fromEvent(scrollable, 'scroll').pipe(\n    debounceTime(debounceMS), \n    distinctUntilChanged(), \n    map(v => sp()), \n    pairwise(), \n    switchMap(p => {\n    const y1 = p[0][1]\n    const y2 = p[1][1]\n    return y1 - y2 > 0 ? of(false) : of(true)\n  }))\n}\n\n/**\n * Filters the entities properties to the set contained in the \n * `keys` array.\n *  \n * @param keys The array of keys that the entity be limited to\n * @param entity The entity to map\n * @return An entity instance that has only the keys provided in the keys array \n */\nexport function mapEntity(keys:string[], entity:any) {\n  const result:any = {}\n  keys.forEach(k=>{\n    result[k] = entity[k]\n  })\n  return result\n}\n\n/**\n * Returns an `Observable<E>` instance that \n * filters for arguments where the property \n * value matches the provided value.\n * \n * @param value The value targeted\n * @param propertyName The name of the property to contain the value\n * @param obs The Slice Object Store Observable\n * @returns Observable<E>\n */\n export function onFilteredEvent<E>(\n  value: any,\n  propertyName: string,\n  obs: Observable<E>\n): Observable<E> {\n  return obs.pipe(filter((e:any) => !!(e && e[propertyName] === value)));\n}"]}
|
@@ -7,4 +7,4 @@ export * from './lib/EStore';
|
|
7
7
|
export * from './lib/OStore';
|
8
8
|
export * from './lib/Slice';
|
9
9
|
export * from './lib/utilities';
|
10
|
-
//# sourceMappingURL=data:application/json;base64,
|
10
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3Byb2plY3RzL3NsaWNlL3NyYy9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsY0FBYyxlQUFlLENBQUE7QUFDN0IsY0FBYyxxQkFBcUIsQ0FBQTtBQUNuQyxjQUFjLGNBQWMsQ0FBQTtBQUM1QixjQUFjLGNBQWMsQ0FBQTtBQUM1QixjQUFjLGFBQWEsQ0FBQTtBQUMzQixjQUFjLGlCQUFpQixDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiLypcbiAqIFB1YmxpYyBBUEkgU3VyZmFjZSBvZiBzbGljZVxuICovXG5cbmV4cG9ydCAqIGZyb20gJy4vbGliL21vZGVscy8nXG5leHBvcnQgKiBmcm9tICcuL2xpYi9BYnN0cmFjdFN0b3JlJ1xuZXhwb3J0ICogZnJvbSAnLi9saWIvRVN0b3JlJ1xuZXhwb3J0ICogZnJvbSAnLi9saWIvT1N0b3JlJ1xuZXhwb3J0ICogZnJvbSAnLi9saWIvU2xpY2UnXG5leHBvcnQgKiBmcm9tICcuL2xpYi91dGlsaXRpZXMnXG4iXX0=
|