@fireflysemantics/slice 14.0.7

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.
@@ -0,0 +1,213 @@
1
+ import { ESTORE_CONFIG_DEFAULT } from "./AbstractStore";
2
+ import { fromEvent, of } from 'rxjs';
3
+ import { switchMap, pairwise, debounceTime, distinctUntilChanged, map } 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 Object.assign({}, 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
+ //# 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 } 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: E, 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}"]}
@@ -0,0 +1,10 @@
1
+ /*
2
+ * Public API Surface of slice
3
+ */
4
+ export * from './lib/models/';
5
+ export * from './lib/AbstractStore';
6
+ export * from './lib/EStore';
7
+ export * from './lib/OStore';
8
+ export * from './lib/Slice';
9
+ export * from './lib/utilities';
10
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3Byb2plY3RzL3NsaWNlL3NyYy9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBQ0gsY0FBYyxlQUFlLENBQUE7QUFDN0IsY0FBYyxxQkFBcUIsQ0FBQTtBQUNuQyxjQUFjLGNBQWMsQ0FBQTtBQUM1QixjQUFjLGNBQWMsQ0FBQTtBQUM1QixjQUFjLGFBQWEsQ0FBQTtBQUMzQixjQUFjLGlCQUFpQixDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiLypcbiAqIFB1YmxpYyBBUEkgU3VyZmFjZSBvZiBzbGljZVxuICovXG5leHBvcnQgKiBmcm9tICcuL2xpYi9tb2RlbHMvJ1xuZXhwb3J0ICogZnJvbSAnLi9saWIvQWJzdHJhY3RTdG9yZSdcbmV4cG9ydCAqIGZyb20gJy4vbGliL0VTdG9yZSdcbmV4cG9ydCAqIGZyb20gJy4vbGliL09TdG9yZSdcbmV4cG9ydCAqIGZyb20gJy4vbGliL1NsaWNlJ1xuZXhwb3J0ICogZnJvbSAnLi9saWIvdXRpbGl0aWVzJ1xuIl19