@ethlete/core 4.0.3 → 4.2.0
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/esm2022/lib/components/structured-data/structured-data.component.mjs +4 -4
- package/esm2022/lib/directives/animatable/animatable.directive.mjs +4 -4
- package/esm2022/lib/directives/animated-if/animated-if.directive.mjs +4 -4
- package/esm2022/lib/directives/animated-lifecycle/animated-lifecycle.directive.mjs +4 -4
- package/esm2022/lib/directives/animated-overlay/animated-overlay.directive.mjs +4 -4
- package/esm2022/lib/directives/click-outside/click-outside.directive.mjs +4 -4
- package/esm2022/lib/directives/cursor-drag-scroll/cursor-drag-scroll.directive.mjs +4 -4
- package/esm2022/lib/directives/debug/debug.directive.mjs +4 -4
- package/esm2022/lib/directives/delayable/delayable.directive.mjs +4 -4
- package/esm2022/lib/directives/is-active-element/is-active-element.directive.mjs +4 -4
- package/esm2022/lib/directives/is-element/is-element.directive.mjs +4 -4
- package/esm2022/lib/directives/let/let.directive.mjs +4 -4
- package/esm2022/lib/directives/observe-content/observe-content.directive.mjs +4 -4
- package/esm2022/lib/directives/observe-resize/observe-resize.directive.mjs +4 -4
- package/esm2022/lib/directives/observe-scroll-state/observe-scroll-state.directive.mjs +4 -4
- package/esm2022/lib/directives/observe-visibility/observe-visibility.directive.mjs +5 -5
- package/esm2022/lib/directives/repeat/repeat.directive.mjs +5 -5
- package/esm2022/lib/directives/root-boundary/root-boundary.directive.mjs +4 -4
- package/esm2022/lib/directives/scroll-observer-first-element/scroll-observer-first-element.directive.mjs +4 -4
- package/esm2022/lib/directives/scroll-observer-ignore-target/scroll-observer-ignore-target.directive.mjs +4 -4
- package/esm2022/lib/directives/scroll-observer-last-element/scroll-observer-last-element.directive.mjs +4 -4
- package/esm2022/lib/directives/seo/seo.directive.mjs +4 -4
- package/esm2022/lib/pipes/infer-mime-type/infer-mime-type.pipe.mjs +4 -4
- package/esm2022/lib/pipes/normalize-game-result-type/normalize-game-result-type.pipe.mjs +4 -4
- package/esm2022/lib/pipes/normalize-match-participants/normalize-match-participants.pipe.mjs +4 -4
- package/esm2022/lib/pipes/normalize-match-score/normalize-match-score.pipe.mjs +4 -4
- package/esm2022/lib/pipes/normalize-match-state/normalize-match-state.pipe.mjs +4 -4
- package/esm2022/lib/pipes/normalize-match-type/normalize-match-type.pipe.mjs +4 -4
- package/esm2022/lib/pipes/to-array/to-array.pipe.mjs +4 -4
- package/esm2022/lib/services/click-observer.service.mjs +7 -7
- package/esm2022/lib/services/content-observer.service.mjs +7 -7
- package/esm2022/lib/services/focus-visible.service.mjs +5 -5
- package/esm2022/lib/services/intersection-observer.service.mjs +7 -7
- package/esm2022/lib/services/resize-observer.service.mjs +7 -7
- package/esm2022/lib/services/router-state.service.mjs +5 -5
- package/esm2022/lib/services/viewport.service.mjs +5 -5
- package/esm2022/lib/utils/signal.utils.mjs +203 -20
- package/fesm2022/ethlete-core.mjs +328 -146
- package/fesm2022/ethlete-core.mjs.map +1 -1
- package/lib/utils/signal.utils.d.ts +43 -2
- package/package.json +11 -11
|
@@ -1,19 +1,47 @@
|
|
|
1
1
|
import { coerceElement } from '@angular/cdk/coercion';
|
|
2
|
-
import { ElementRef, computed, effect, inject, isSignal, signal } from '@angular/core';
|
|
3
|
-
import { toSignal } from '@angular/core/rxjs-interop';
|
|
4
|
-
import { Observable, map } from 'rxjs';
|
|
2
|
+
import { DestroyRef, ElementRef, NgZone, QueryList, afterNextRender, computed, effect, inject, isDevMode, isSignal, signal, } from '@angular/core';
|
|
3
|
+
import { toObservable, toSignal } from '@angular/core/rxjs-interop';
|
|
4
|
+
import { Observable, map, of, pairwise, startWith, switchMap } from 'rxjs';
|
|
5
|
+
import { isElementVisible } from './scrollable.utils';
|
|
6
|
+
function isElementSignal(el) {
|
|
7
|
+
if (isSignal(el)) {
|
|
8
|
+
const val = el();
|
|
9
|
+
return typeof val === 'object' && val !== null && 'currentElement' in val && 'previousElement' in val;
|
|
10
|
+
}
|
|
11
|
+
return false;
|
|
12
|
+
}
|
|
5
13
|
const buildElementSignal = (el) => {
|
|
14
|
+
if (el === null || el === undefined) {
|
|
15
|
+
return signal({ currentElement: null, previousElement: null });
|
|
16
|
+
}
|
|
17
|
+
if (isElementSignal(el)) {
|
|
18
|
+
return el;
|
|
19
|
+
}
|
|
6
20
|
let mElSignal = null;
|
|
21
|
+
const switchElement = () => switchMap((elOrRef) => {
|
|
22
|
+
if (elOrRef instanceof QueryList) {
|
|
23
|
+
return elOrRef.changes.pipe(startWith(elOrRef), map(() => (elOrRef.first ? coerceElement(elOrRef.first) : null)));
|
|
24
|
+
}
|
|
25
|
+
else {
|
|
26
|
+
return of(coerceElement(elOrRef));
|
|
27
|
+
}
|
|
28
|
+
});
|
|
7
29
|
if (el instanceof Observable) {
|
|
8
|
-
mElSignal = toSignal(el.pipe(
|
|
30
|
+
mElSignal = toSignal(el.pipe(switchElement()), { initialValue: null });
|
|
9
31
|
}
|
|
10
32
|
else if (isSignal(el)) {
|
|
11
|
-
mElSignal =
|
|
33
|
+
mElSignal = toSignal(toObservable(el).pipe(switchElement()));
|
|
34
|
+
}
|
|
35
|
+
else if (el instanceof QueryList) {
|
|
36
|
+
mElSignal = toSignal(el.changes.pipe(startWith(el), map(() => (el.first ? coerceElement(el.first) : null))));
|
|
12
37
|
}
|
|
13
38
|
else {
|
|
14
39
|
mElSignal = signal(coerceElement(el));
|
|
15
40
|
}
|
|
16
|
-
return mElSignal
|
|
41
|
+
return toSignal(toObservable(mElSignal).pipe(startWith(null), pairwise(), map(([previousElement, currentElement]) => ({
|
|
42
|
+
currentElement: currentElement ?? null,
|
|
43
|
+
previousElement: previousElement ?? null,
|
|
44
|
+
}))), { initialValue: { currentElement: null, previousElement: null } });
|
|
17
45
|
};
|
|
18
46
|
export const buildSignalEffects = (config) => {
|
|
19
47
|
const { map, eachItemFn, cleanupFn } = config;
|
|
@@ -38,60 +66,215 @@ export const buildSignalEffects = (config) => {
|
|
|
38
66
|
};
|
|
39
67
|
return { remove, has };
|
|
40
68
|
};
|
|
69
|
+
export const signalIsRendered = () => {
|
|
70
|
+
const isRendered = signal(false);
|
|
71
|
+
afterNextRender(() => isRendered.set(true));
|
|
72
|
+
return isRendered.asReadonly();
|
|
73
|
+
};
|
|
41
74
|
export const signalClasses = (el, classMap) => {
|
|
42
|
-
const
|
|
75
|
+
const elements = buildElementSignal(el);
|
|
43
76
|
return buildSignalEffects({
|
|
44
77
|
map: classMap,
|
|
45
78
|
eachItemFn: ({ key, value }) => {
|
|
46
79
|
if (value) {
|
|
47
|
-
|
|
80
|
+
elements().currentElement?.classList.add(key);
|
|
48
81
|
}
|
|
49
82
|
else {
|
|
50
|
-
|
|
83
|
+
elements().currentElement?.classList.remove(key);
|
|
51
84
|
}
|
|
52
85
|
},
|
|
53
|
-
cleanupFn: ({ key }) =>
|
|
86
|
+
cleanupFn: ({ key }) => elements().currentElement?.classList.remove(key),
|
|
54
87
|
});
|
|
55
88
|
};
|
|
56
89
|
export const signalHostClasses = (classMap) => signalClasses(inject(ElementRef), classMap);
|
|
57
90
|
const ALWAYS_TRUE_ATTRIBUTE_KEYS = ['disabled', 'readonly', 'required', 'checked', 'selected'];
|
|
58
91
|
export const signalAttributes = (el, attributeMap) => {
|
|
59
|
-
const
|
|
92
|
+
const elements = buildElementSignal(el);
|
|
60
93
|
return buildSignalEffects({
|
|
61
94
|
map: attributeMap,
|
|
62
95
|
eachItemFn: ({ key, value }) => {
|
|
63
96
|
const valueString = `${value}`;
|
|
64
97
|
if (ALWAYS_TRUE_ATTRIBUTE_KEYS.includes(key)) {
|
|
65
98
|
if (value) {
|
|
66
|
-
|
|
99
|
+
elements().currentElement?.setAttribute(key, '');
|
|
67
100
|
}
|
|
68
101
|
else {
|
|
69
|
-
|
|
102
|
+
elements().currentElement?.removeAttribute(key);
|
|
70
103
|
}
|
|
71
104
|
}
|
|
72
105
|
else {
|
|
73
106
|
if (value === null || value === undefined) {
|
|
74
|
-
|
|
107
|
+
elements().currentElement?.removeAttribute(key);
|
|
75
108
|
}
|
|
76
109
|
else {
|
|
77
|
-
|
|
110
|
+
elements().currentElement?.setAttribute(key, valueString);
|
|
78
111
|
}
|
|
79
112
|
}
|
|
80
113
|
},
|
|
81
|
-
cleanupFn: ({ key }) =>
|
|
114
|
+
cleanupFn: ({ key }) => elements().currentElement?.removeAttribute(key),
|
|
82
115
|
});
|
|
83
116
|
};
|
|
84
117
|
export const signalHostAttributes = (attributeMap) => signalAttributes(inject(ElementRef), attributeMap);
|
|
85
118
|
export const signalStyles = (el, styleMap) => {
|
|
86
|
-
const
|
|
119
|
+
const elements = buildElementSignal(el);
|
|
87
120
|
return buildSignalEffects({
|
|
88
121
|
map: styleMap,
|
|
89
122
|
eachItemFn: ({ key, value }) => {
|
|
90
123
|
const valueString = `${value}`;
|
|
91
|
-
|
|
124
|
+
elements().currentElement?.style.setProperty(key, valueString);
|
|
92
125
|
},
|
|
93
|
-
cleanupFn: ({ key }) =>
|
|
126
|
+
cleanupFn: ({ key }) => elements().currentElement?.style.removeProperty(key),
|
|
94
127
|
});
|
|
95
128
|
};
|
|
96
129
|
export const signalHostStyles = (styleMap) => signalStyles(inject(ElementRef), styleMap);
|
|
97
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"signal.utils.js","sourceRoot":"","sources":["../../../../../../libs/core/src/lib/utils/signal.utils.ts"],"names":[],"mappings":"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","sourcesContent":["import { coerceElement } from '@angular/cdk/coercion';\nimport { EffectRef, ElementRef, Signal, computed, effect, inject, isSignal, signal } from '@angular/core';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { Observable, map } from 'rxjs';\n\ntype SignalElementBindingType =\n  | HTMLElement\n  | ElementRef<HTMLElement>\n  | Observable<HTMLElement | ElementRef<HTMLElement> | null | undefined>\n  | Signal<HTMLElement | ElementRef<HTMLElement> | null | undefined>;\n\nconst buildElementSignal = (el: SignalElementBindingType) => {\n  let mElSignal: Signal<HTMLElement | null | undefined> | null = null;\n\n  if (el instanceof Observable) {\n    mElSignal = toSignal(el.pipe(map((elOrRef) => coerceElement(elOrRef))), { initialValue: null });\n  } else if (isSignal(el)) {\n    mElSignal = computed(() => coerceElement(el()));\n  } else {\n    mElSignal = signal(coerceElement(el));\n  }\n\n  return mElSignal as Signal<HTMLElement | null | undefined>;\n};\n\nexport const buildSignalEffects = <T extends Record<string, Signal<unknown>>>(config: {\n  map: T;\n  eachItemFn: (pair: { key: string; value: unknown }) => void;\n  cleanupFn: (pair: { key: string; value: unknown }) => void;\n}) => {\n  const { map, eachItemFn, cleanupFn } = config;\n\n  const effectRefMap: Record<string, EffectRef> = {};\n\n  for (const [tokenString, signal] of Object.entries(map)) {\n    const tokenArray = tokenString.split(' ').filter((token) => !!token);\n\n    for (const token of tokenArray) {\n      const ref = effect(() => {\n        const value = signal();\n        eachItemFn({ key: token, value });\n      });\n\n      effectRefMap[token] = ref;\n    }\n  }\n\n  const has = (token: string) => token in effectRefMap;\n\n  const remove = (...tokens: string[]) => {\n    for (const tokenString of tokens) {\n      effectRefMap[tokenString]?.destroy();\n\n      cleanupFn({ key: tokenString, value: map[tokenString]?.() });\n\n      delete effectRefMap[tokenString];\n    }\n  };\n\n  return { remove, has };\n};\n\nexport const signalClasses = <T extends Record<string, Signal<unknown>>>(el: SignalElementBindingType, classMap: T) => {\n  const element = buildElementSignal(el);\n\n  return buildSignalEffects({\n    map: classMap,\n    eachItemFn: ({ key, value }) => {\n      if (value) {\n        element()?.classList.add(key);\n      } else {\n        element()?.classList.remove(key);\n      }\n    },\n    cleanupFn: ({ key }) => element()?.classList.remove(key),\n  });\n};\n\nexport const signalHostClasses = <T extends Record<string, Signal<unknown>>>(classMap: T) =>\n  signalClasses(inject(ElementRef), classMap);\n\nconst ALWAYS_TRUE_ATTRIBUTE_KEYS = ['disabled', 'readonly', 'required', 'checked', 'selected'];\n\nexport const signalAttributes = <T extends Record<string, Signal<unknown>>>(\n  el: SignalElementBindingType,\n  attributeMap: T,\n) => {\n  const element = buildElementSignal(el);\n\n  return buildSignalEffects({\n    map: attributeMap,\n    eachItemFn: ({ key, value }) => {\n      const valueString = `${value}`;\n\n      if (ALWAYS_TRUE_ATTRIBUTE_KEYS.includes(key)) {\n        if (value) {\n          element()?.setAttribute(key, '');\n        } else {\n          element()?.removeAttribute(key);\n        }\n      } else {\n        if (value === null || value === undefined) {\n          element()?.removeAttribute(key);\n        } else {\n          element()?.setAttribute(key, valueString);\n        }\n      }\n    },\n    cleanupFn: ({ key }) => element!()?.removeAttribute(key),\n  });\n};\n\nexport const signalHostAttributes = <T extends Record<string, Signal<unknown>>>(attributeMap: T) =>\n  signalAttributes(inject(ElementRef), attributeMap);\n\nexport const signalStyles = <T extends Record<string, Signal<unknown>>>(el: SignalElementBindingType, styleMap: T) => {\n  const element = buildElementSignal(el);\n\n  return buildSignalEffects({\n    map: styleMap,\n    eachItemFn: ({ key, value }) => {\n      const valueString = `${value}`;\n\n      element()?.style.setProperty(key, valueString);\n    },\n    cleanupFn: ({ key }) => element()?.style.removeProperty(key),\n  });\n};\n\nexport const signalHostStyles = <T extends Record<string, Signal<unknown>>>(styleMap: T) =>\n  signalStyles(inject(ElementRef), styleMap);\n"]}
|
|
130
|
+
export const signalElementDimensions = (el) => {
|
|
131
|
+
const destroyRef = inject(DestroyRef);
|
|
132
|
+
const elements = buildElementSignal(el);
|
|
133
|
+
const zone = inject(NgZone);
|
|
134
|
+
const isRendered = signalIsRendered();
|
|
135
|
+
const initialValue = () => ({
|
|
136
|
+
rect: elements().currentElement?.getBoundingClientRect() ?? null,
|
|
137
|
+
borderBoxSize: null,
|
|
138
|
+
contentBoxSize: null,
|
|
139
|
+
devicePixelContentBoxSize: null,
|
|
140
|
+
});
|
|
141
|
+
const elementDimensionsSignal = signal(initialValue());
|
|
142
|
+
const observer = new ResizeObserver((e) => {
|
|
143
|
+
if (!isRendered())
|
|
144
|
+
return;
|
|
145
|
+
const entry = e[0];
|
|
146
|
+
if (entry) {
|
|
147
|
+
const devicePixelContentBoxSize = entry.devicePixelContentBoxSize?.[0] ?? null;
|
|
148
|
+
const borderBoxSize = entry.borderBoxSize?.[0] ?? null;
|
|
149
|
+
const contentBoxSize = entry.contentBoxSize?.[0] ?? null;
|
|
150
|
+
zone.run(() => elementDimensionsSignal.set({
|
|
151
|
+
rect: entry.contentRect,
|
|
152
|
+
borderBoxSize: borderBoxSize
|
|
153
|
+
? { inlineSize: borderBoxSize.inlineSize, blockSize: borderBoxSize.blockSize }
|
|
154
|
+
: null,
|
|
155
|
+
contentBoxSize: contentBoxSize
|
|
156
|
+
? { inlineSize: contentBoxSize.inlineSize, blockSize: contentBoxSize.blockSize }
|
|
157
|
+
: null,
|
|
158
|
+
devicePixelContentBoxSize: devicePixelContentBoxSize
|
|
159
|
+
? { inlineSize: devicePixelContentBoxSize.inlineSize, blockSize: devicePixelContentBoxSize.blockSize }
|
|
160
|
+
: null,
|
|
161
|
+
}));
|
|
162
|
+
}
|
|
163
|
+
});
|
|
164
|
+
effect(() => {
|
|
165
|
+
const els = elements();
|
|
166
|
+
elementDimensionsSignal.set(initialValue());
|
|
167
|
+
if (els.previousElement) {
|
|
168
|
+
observer.disconnect();
|
|
169
|
+
}
|
|
170
|
+
if (els.currentElement) {
|
|
171
|
+
const computedDisplay = getComputedStyle(els.currentElement).display;
|
|
172
|
+
const currentElIsAngularComponent = els.currentElement?.tagName.toLowerCase().includes('-');
|
|
173
|
+
if (computedDisplay === 'inline' && isDevMode() && currentElIsAngularComponent) {
|
|
174
|
+
console.error(`Element <${els.currentElement?.tagName.toLowerCase()}> is an Angular component and has a display of 'inline'. Inline elements cannot be observed for dimensions. Please change it to 'block' or something else.`);
|
|
175
|
+
}
|
|
176
|
+
observer.observe(els.currentElement);
|
|
177
|
+
}
|
|
178
|
+
}, { allowSignalWrites: true });
|
|
179
|
+
destroyRef.onDestroy(() => observer.disconnect());
|
|
180
|
+
return elementDimensionsSignal.asReadonly();
|
|
181
|
+
};
|
|
182
|
+
export const signalHostElementDimensions = () => signalElementDimensions(inject(ElementRef));
|
|
183
|
+
export const signalElementMutations = (el, options) => {
|
|
184
|
+
const destroyRef = inject(DestroyRef);
|
|
185
|
+
const elements = buildElementSignal(el);
|
|
186
|
+
const zone = inject(NgZone);
|
|
187
|
+
const isRendered = signalIsRendered();
|
|
188
|
+
const elementMutationsSignal = signal(null);
|
|
189
|
+
const observer = new MutationObserver((e) => {
|
|
190
|
+
if (!isRendered())
|
|
191
|
+
return;
|
|
192
|
+
const entry = e[0];
|
|
193
|
+
if (entry) {
|
|
194
|
+
zone.run(() => elementMutationsSignal.set(entry));
|
|
195
|
+
}
|
|
196
|
+
});
|
|
197
|
+
effect(() => {
|
|
198
|
+
const els = elements();
|
|
199
|
+
elementMutationsSignal.set(null);
|
|
200
|
+
if (els.previousElement) {
|
|
201
|
+
observer.disconnect();
|
|
202
|
+
}
|
|
203
|
+
if (els.currentElement) {
|
|
204
|
+
observer.observe(els.currentElement, options);
|
|
205
|
+
}
|
|
206
|
+
}, { allowSignalWrites: true });
|
|
207
|
+
destroyRef.onDestroy(() => observer.disconnect());
|
|
208
|
+
return elementMutationsSignal.asReadonly();
|
|
209
|
+
};
|
|
210
|
+
export const signalHostElementMutations = (options) => signalElementMutations(inject(ElementRef), options);
|
|
211
|
+
export const signalElementScrollState = (el) => {
|
|
212
|
+
const elements = buildElementSignal(el);
|
|
213
|
+
const elementDimensions = signalElementDimensions(elements);
|
|
214
|
+
const elementMutations = signalElementMutations(elements, { childList: true, subtree: true });
|
|
215
|
+
const isRendered = signalIsRendered();
|
|
216
|
+
return computed(() => {
|
|
217
|
+
const element = elements().currentElement;
|
|
218
|
+
const dimensions = elementDimensions();
|
|
219
|
+
const notScrollable = () => ({
|
|
220
|
+
canScroll: false,
|
|
221
|
+
canScrollHorizontally: false,
|
|
222
|
+
canScrollVertically: false,
|
|
223
|
+
scrollWidth: element?.scrollWidth ?? null,
|
|
224
|
+
scrollHeight: element?.scrollHeight ?? null,
|
|
225
|
+
elementDimensions: dimensions,
|
|
226
|
+
});
|
|
227
|
+
// We are not interested what the mutation is, just that there is one.
|
|
228
|
+
// Changes to the DOM can affect the scroll state of the element.
|
|
229
|
+
elementMutations();
|
|
230
|
+
if (!element || !dimensions.rect || !isRendered())
|
|
231
|
+
return notScrollable();
|
|
232
|
+
const { scrollWidth, scrollHeight } = element;
|
|
233
|
+
const { width, height } = dimensions.rect;
|
|
234
|
+
const canScrollHorizontally = scrollWidth > width;
|
|
235
|
+
const canScrollVertically = scrollHeight > height;
|
|
236
|
+
return {
|
|
237
|
+
canScroll: canScrollHorizontally || canScrollVertically,
|
|
238
|
+
canScrollHorizontally,
|
|
239
|
+
canScrollVertically,
|
|
240
|
+
scrollWidth,
|
|
241
|
+
scrollHeight,
|
|
242
|
+
elementDimensions: dimensions,
|
|
243
|
+
};
|
|
244
|
+
});
|
|
245
|
+
};
|
|
246
|
+
export const signalHostElementScrollState = () => signalElementScrollState(inject(ElementRef));
|
|
247
|
+
export const signalElementIntersection = (el, container, options) => {
|
|
248
|
+
const destroyRef = inject(DestroyRef);
|
|
249
|
+
const elements = buildElementSignal(el);
|
|
250
|
+
const containerElements = buildElementSignal(container);
|
|
251
|
+
const zone = inject(NgZone);
|
|
252
|
+
const isRendered = signalIsRendered();
|
|
253
|
+
const initialValue = () => ({
|
|
254
|
+
isIntersecting: isElementVisible({ element: elements().currentElement, container: containerElements().currentElement })?.inline ??
|
|
255
|
+
false,
|
|
256
|
+
});
|
|
257
|
+
const elementIntersectionSignal = signal(initialValue());
|
|
258
|
+
const observer = new IntersectionObserver((e) => {
|
|
259
|
+
if (!isRendered())
|
|
260
|
+
return;
|
|
261
|
+
const entry = e[0];
|
|
262
|
+
if (entry) {
|
|
263
|
+
zone.run(() => elementIntersectionSignal.set(entry));
|
|
264
|
+
}
|
|
265
|
+
}, options);
|
|
266
|
+
effect(() => {
|
|
267
|
+
const els = elements();
|
|
268
|
+
elementIntersectionSignal.set(initialValue());
|
|
269
|
+
if (els.previousElement) {
|
|
270
|
+
observer.disconnect();
|
|
271
|
+
}
|
|
272
|
+
if (els.currentElement) {
|
|
273
|
+
observer.observe(els.currentElement);
|
|
274
|
+
}
|
|
275
|
+
}, { allowSignalWrites: true });
|
|
276
|
+
destroyRef.onDestroy(() => observer.disconnect());
|
|
277
|
+
return elementIntersectionSignal.asReadonly();
|
|
278
|
+
};
|
|
279
|
+
export const signalHostElementIntersection = (container, options) => signalElementIntersection(inject(ElementRef), container, options);
|
|
280
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"signal.utils.js","sourceRoot":"","sources":["../../../../../../libs/core/src/lib/utils/signal.utils.ts"],"names":[],"mappings":"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","sourcesContent":["import { coerceElement } from '@angular/cdk/coercion';\nimport {\n  DestroyRef,\n  EffectRef,\n  ElementRef,\n  NgZone,\n  QueryList,\n  Signal,\n  afterNextRender,\n  computed,\n  effect,\n  inject,\n  isDevMode,\n  isSignal,\n  signal,\n} from '@angular/core';\nimport { toObservable, toSignal } from '@angular/core/rxjs-interop';\nimport { Observable, map, of, pairwise, startWith, switchMap } from 'rxjs';\nimport { isElementVisible } from './scrollable.utils';\n\ntype SignalElementBindingComplexType =\n  | HTMLElement\n  | ElementRef<HTMLElement>\n  | QueryList<ElementRef<HTMLElement> | HTMLElement>\n  | null\n  | undefined;\n\ntype SignalElementBindingType =\n  | HTMLElement\n  | ElementRef<HTMLElement>\n  | Observable<SignalElementBindingComplexType>\n  | Signal<SignalElementBindingComplexType>\n  | QueryList<ElementRef<HTMLElement> | HTMLElement>\n  | ElementSignal;\n\ntype ElementSignal = Signal<{\n  currentElement: HTMLElement | null;\n  previousElement: HTMLElement | null;\n}>;\n\nfunction isElementSignal(el: unknown): el is ElementSignal {\n  if (isSignal(el)) {\n    const val = el();\n    return typeof val === 'object' && val !== null && 'currentElement' in val && 'previousElement' in val;\n  }\n\n  return false;\n}\n\nconst buildElementSignal = (el: SignalElementBindingType | null | undefined): ElementSignal => {\n  if (el === null || el === undefined) {\n    return signal({ currentElement: null, previousElement: null });\n  }\n\n  if (isElementSignal(el)) {\n    return el;\n  }\n\n  let mElSignal: Signal<HTMLElement | null | undefined> | null = null;\n\n  const switchElement = () =>\n    switchMap((elOrRef) => {\n      if (elOrRef instanceof QueryList) {\n        return elOrRef.changes.pipe(\n          startWith(elOrRef),\n          map(() => (elOrRef.first ? coerceElement(elOrRef.first) : null)),\n        );\n      } else {\n        return of(coerceElement(elOrRef));\n      }\n    });\n\n  if (el instanceof Observable) {\n    mElSignal = toSignal(el.pipe(switchElement()), { initialValue: null });\n  } else if (isSignal(el)) {\n    mElSignal = toSignal(toObservable(el).pipe(switchElement()));\n  } else if (el instanceof QueryList) {\n    mElSignal = toSignal(\n      el.changes.pipe(\n        startWith(el),\n        map(() => (el.first ? coerceElement(el.first) : null)),\n      ),\n    );\n  } else {\n    mElSignal = signal(coerceElement(el));\n  }\n\n  return toSignal(\n    toObservable(mElSignal).pipe(\n      startWith(null),\n      pairwise(),\n      map(([previousElement, currentElement]) => ({\n        currentElement: currentElement ?? null,\n        previousElement: previousElement ?? null,\n      })),\n    ),\n    { initialValue: { currentElement: null, previousElement: null } },\n  );\n};\n\nexport const buildSignalEffects = <T extends Record<string, Signal<unknown>>>(config: {\n  map: T;\n  eachItemFn: (pair: { key: string; value: unknown }) => void;\n  cleanupFn: (pair: { key: string; value: unknown }) => void;\n}) => {\n  const { map, eachItemFn, cleanupFn } = config;\n\n  const effectRefMap: Record<string, EffectRef> = {};\n\n  for (const [tokenString, signal] of Object.entries(map)) {\n    const tokenArray = tokenString.split(' ').filter((token) => !!token);\n\n    for (const token of tokenArray) {\n      const ref = effect(() => {\n        const value = signal();\n        eachItemFn({ key: token, value });\n      });\n\n      effectRefMap[token] = ref;\n    }\n  }\n\n  const has = (token: string) => token in effectRefMap;\n\n  const remove = (...tokens: string[]) => {\n    for (const tokenString of tokens) {\n      effectRefMap[tokenString]?.destroy();\n\n      cleanupFn({ key: tokenString, value: map[tokenString]?.() });\n\n      delete effectRefMap[tokenString];\n    }\n  };\n\n  return { remove, has };\n};\n\nexport const signalIsRendered = () => {\n  const isRendered = signal(false);\n\n  afterNextRender(() => isRendered.set(true));\n\n  return isRendered.asReadonly();\n};\n\nexport const signalClasses = <T extends Record<string, Signal<unknown>>>(el: SignalElementBindingType, classMap: T) => {\n  const elements = buildElementSignal(el);\n\n  return buildSignalEffects({\n    map: classMap,\n    eachItemFn: ({ key, value }) => {\n      if (value) {\n        elements().currentElement?.classList.add(key);\n      } else {\n        elements().currentElement?.classList.remove(key);\n      }\n    },\n    cleanupFn: ({ key }) => elements().currentElement?.classList.remove(key),\n  });\n};\n\nexport const signalHostClasses = <T extends Record<string, Signal<unknown>>>(classMap: T) =>\n  signalClasses(inject(ElementRef), classMap);\n\nconst ALWAYS_TRUE_ATTRIBUTE_KEYS = ['disabled', 'readonly', 'required', 'checked', 'selected'];\n\nexport const signalAttributes = <T extends Record<string, Signal<unknown>>>(\n  el: SignalElementBindingType,\n  attributeMap: T,\n) => {\n  const elements = buildElementSignal(el);\n\n  return buildSignalEffects({\n    map: attributeMap,\n    eachItemFn: ({ key, value }) => {\n      const valueString = `${value}`;\n\n      if (ALWAYS_TRUE_ATTRIBUTE_KEYS.includes(key)) {\n        if (value) {\n          elements().currentElement?.setAttribute(key, '');\n        } else {\n          elements().currentElement?.removeAttribute(key);\n        }\n      } else {\n        if (value === null || value === undefined) {\n          elements().currentElement?.removeAttribute(key);\n        } else {\n          elements().currentElement?.setAttribute(key, valueString);\n        }\n      }\n    },\n    cleanupFn: ({ key }) => elements().currentElement?.removeAttribute(key),\n  });\n};\n\nexport const signalHostAttributes = <T extends Record<string, Signal<unknown>>>(attributeMap: T) =>\n  signalAttributes(inject(ElementRef), attributeMap);\n\nexport const signalStyles = <T extends Record<string, Signal<unknown>>>(el: SignalElementBindingType, styleMap: T) => {\n  const elements = buildElementSignal(el);\n\n  return buildSignalEffects({\n    map: styleMap,\n    eachItemFn: ({ key, value }) => {\n      const valueString = `${value}`;\n\n      elements().currentElement?.style.setProperty(key, valueString);\n    },\n    cleanupFn: ({ key }) => elements().currentElement?.style.removeProperty(key),\n  });\n};\n\nexport const signalHostStyles = <T extends Record<string, Signal<unknown>>>(styleMap: T) =>\n  signalStyles(inject(ElementRef), styleMap);\n\nexport interface LogicalSize {\n  inlineSize: number;\n  blockSize: number;\n}\n\nexport interface ElementDimensions {\n  rect: DOMRectReadOnly | null;\n  borderBoxSize: LogicalSize | null;\n  contentBoxSize: LogicalSize | null;\n  devicePixelContentBoxSize: LogicalSize | null;\n}\n\nexport const signalElementDimensions = (el: SignalElementBindingType) => {\n  const destroyRef = inject(DestroyRef);\n  const elements = buildElementSignal(el);\n  const zone = inject(NgZone);\n  const isRendered = signalIsRendered();\n\n  const initialValue = () => ({\n    rect: elements().currentElement?.getBoundingClientRect() ?? null,\n    borderBoxSize: null,\n    contentBoxSize: null,\n    devicePixelContentBoxSize: null,\n  });\n\n  const elementDimensionsSignal = signal<ElementDimensions>(initialValue());\n\n  const observer = new ResizeObserver((e) => {\n    if (!isRendered()) return;\n\n    const entry = e[0];\n\n    if (entry) {\n      const devicePixelContentBoxSize = entry.devicePixelContentBoxSize?.[0] ?? null;\n      const borderBoxSize = entry.borderBoxSize?.[0] ?? null;\n      const contentBoxSize = entry.contentBoxSize?.[0] ?? null;\n\n      zone.run(() =>\n        elementDimensionsSignal.set({\n          rect: entry.contentRect,\n          borderBoxSize: borderBoxSize\n            ? { inlineSize: borderBoxSize.inlineSize, blockSize: borderBoxSize.blockSize }\n            : null,\n          contentBoxSize: contentBoxSize\n            ? { inlineSize: contentBoxSize.inlineSize, blockSize: contentBoxSize.blockSize }\n            : null,\n          devicePixelContentBoxSize: devicePixelContentBoxSize\n            ? { inlineSize: devicePixelContentBoxSize.inlineSize, blockSize: devicePixelContentBoxSize.blockSize }\n            : null,\n        }),\n      );\n    }\n  });\n\n  effect(\n    () => {\n      const els = elements();\n\n      elementDimensionsSignal.set(initialValue());\n\n      if (els.previousElement) {\n        observer.disconnect();\n      }\n\n      if (els.currentElement) {\n        const computedDisplay = getComputedStyle(els.currentElement).display;\n        const currentElIsAngularComponent = els.currentElement?.tagName.toLowerCase().includes('-');\n\n        if (computedDisplay === 'inline' && isDevMode() && currentElIsAngularComponent) {\n          console.error(\n            `Element <${els.currentElement?.tagName.toLowerCase()}> is an Angular component and has a display of 'inline'. Inline elements cannot be observed for dimensions. Please change it to 'block' or something else.`,\n          );\n        }\n\n        observer.observe(els.currentElement);\n      }\n    },\n    { allowSignalWrites: true },\n  );\n\n  destroyRef.onDestroy(() => observer.disconnect());\n\n  return elementDimensionsSignal.asReadonly();\n};\n\nexport const signalHostElementDimensions = () => signalElementDimensions(inject(ElementRef));\n\nexport const signalElementMutations = (el: SignalElementBindingType, options?: MutationObserverInit) => {\n  const destroyRef = inject(DestroyRef);\n  const elements = buildElementSignal(el);\n  const zone = inject(NgZone);\n  const isRendered = signalIsRendered();\n\n  const elementMutationsSignal = signal<MutationRecord | null>(null);\n\n  const observer = new MutationObserver((e) => {\n    if (!isRendered()) return;\n\n    const entry = e[0];\n\n    if (entry) {\n      zone.run(() => elementMutationsSignal.set(entry));\n    }\n  });\n\n  effect(\n    () => {\n      const els = elements();\n\n      elementMutationsSignal.set(null);\n\n      if (els.previousElement) {\n        observer.disconnect();\n      }\n\n      if (els.currentElement) {\n        observer.observe(els.currentElement, options);\n      }\n    },\n    { allowSignalWrites: true },\n  );\n\n  destroyRef.onDestroy(() => observer.disconnect());\n\n  return elementMutationsSignal.asReadonly();\n};\n\nexport const signalHostElementMutations = (options?: MutationObserverInit) =>\n  signalElementMutations(inject(ElementRef), options);\n\nexport const signalElementScrollState = (el: SignalElementBindingType) => {\n  const elements = buildElementSignal(el);\n  const elementDimensions = signalElementDimensions(elements);\n  const elementMutations = signalElementMutations(elements, { childList: true, subtree: true });\n  const isRendered = signalIsRendered();\n\n  return computed(() => {\n    const element = elements().currentElement;\n    const dimensions = elementDimensions();\n\n    const notScrollable = () => ({\n      canScroll: false,\n      canScrollHorizontally: false,\n      canScrollVertically: false,\n      scrollWidth: element?.scrollWidth ?? null,\n      scrollHeight: element?.scrollHeight ?? null,\n      elementDimensions: dimensions,\n    });\n\n    // We are not interested what the mutation is, just that there is one.\n    // Changes to the DOM can affect the scroll state of the element.\n    elementMutations();\n\n    if (!element || !dimensions.rect || !isRendered()) return notScrollable();\n\n    const { scrollWidth, scrollHeight } = element;\n    const { width, height } = dimensions.rect;\n\n    const canScrollHorizontally = scrollWidth > width;\n    const canScrollVertically = scrollHeight > height;\n\n    return {\n      canScroll: canScrollHorizontally || canScrollVertically,\n      canScrollHorizontally,\n      canScrollVertically,\n      scrollWidth,\n      scrollHeight,\n      elementDimensions: dimensions,\n    };\n  });\n};\n\nexport const signalHostElementScrollState = () => signalElementScrollState(inject(ElementRef));\n\nexport interface ElementIntersection {\n  isIntersecting: boolean;\n}\n\nexport const signalElementIntersection = (\n  el: SignalElementBindingType,\n  container?: SignalElementBindingType,\n  options?: IntersectionObserverInit,\n) => {\n  const destroyRef = inject(DestroyRef);\n  const elements = buildElementSignal(el);\n  const containerElements = buildElementSignal(container);\n  const zone = inject(NgZone);\n  const isRendered = signalIsRendered();\n\n  const initialValue = () => ({\n    isIntersecting:\n      isElementVisible({ element: elements().currentElement, container: containerElements().currentElement })?.inline ??\n      false,\n  });\n\n  const elementIntersectionSignal = signal<ElementIntersection>(initialValue());\n\n  const observer = new IntersectionObserver((e) => {\n    if (!isRendered()) return;\n\n    const entry = e[0];\n\n    if (entry) {\n      zone.run(() => elementIntersectionSignal.set(entry));\n    }\n  }, options);\n\n  effect(\n    () => {\n      const els = elements();\n\n      elementIntersectionSignal.set(initialValue());\n\n      if (els.previousElement) {\n        observer.disconnect();\n      }\n\n      if (els.currentElement) {\n        observer.observe(els.currentElement);\n      }\n    },\n    { allowSignalWrites: true },\n  );\n\n  destroyRef.onDestroy(() => observer.disconnect());\n\n  return elementIntersectionSignal.asReadonly();\n};\n\nexport const signalHostElementIntersection = (\n  container?: SignalElementBindingType,\n  options?: IntersectionObserverInit,\n) => signalElementIntersection(inject(ElementRef), container, options);\n"]}
|