@ethlete/core 4.23.8 → 4.24.1
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/CHANGELOG.md +12 -0
- package/fesm2022/ethlete-core.mjs +131 -139
- package/fesm2022/ethlete-core.mjs.map +1 -1
- package/lib/types/angular.types.d.ts +0 -1
- package/lib/utils/signal.utils.d.ts +5 -0
- package/package.json +7 -9
- package/esm2022/ethlete-core.mjs +0 -5
- package/esm2022/index.mjs +0 -38
- package/esm2022/lib/components/structured-data/index.mjs +0 -2
- package/esm2022/lib/components/structured-data/structured-data.component.mjs +0 -37
- package/esm2022/lib/constants/index.mjs +0 -2
- package/esm2022/lib/constants/viewport.constants.mjs +0 -17
- package/esm2022/lib/decorators/memo/index.mjs +0 -2
- package/esm2022/lib/decorators/memo/memo.decorator.mjs +0 -25
- package/esm2022/lib/directives/animatable/animatable.directive.mjs +0 -121
- package/esm2022/lib/directives/animatable/index.mjs +0 -2
- package/esm2022/lib/directives/animated-if/animated-if.directive.mjs +0 -54
- package/esm2022/lib/directives/animated-if/index.mjs +0 -2
- package/esm2022/lib/directives/animated-lifecycle/animated-lifecycle.directive.mjs +0 -136
- package/esm2022/lib/directives/animated-lifecycle/index.mjs +0 -2
- package/esm2022/lib/directives/animated-overlay/animated-overlay.directive.mjs +0 -344
- package/esm2022/lib/directives/animated-overlay/index.mjs +0 -2
- package/esm2022/lib/directives/click-outside/click-outside.directive.mjs +0 -37
- package/esm2022/lib/directives/click-outside/index.mjs +0 -2
- package/esm2022/lib/directives/cursor-drag-scroll/cursor-drag-scroll.constants.mjs +0 -3
- package/esm2022/lib/directives/cursor-drag-scroll/cursor-drag-scroll.directive.mjs +0 -170
- package/esm2022/lib/directives/cursor-drag-scroll/index.mjs +0 -3
- package/esm2022/lib/directives/debug/debug.directive.mjs +0 -44
- package/esm2022/lib/directives/debug/index.mjs +0 -2
- package/esm2022/lib/directives/delayable/delayable.directive.mjs +0 -55
- package/esm2022/lib/directives/delayable/index.mjs +0 -2
- package/esm2022/lib/directives/is-active-element/index.mjs +0 -2
- package/esm2022/lib/directives/is-active-element/is-active-element.directive.mjs +0 -33
- package/esm2022/lib/directives/is-element/index.mjs +0 -2
- package/esm2022/lib/directives/is-element/is-element.directive.mjs +0 -33
- package/esm2022/lib/directives/let/index.mjs +0 -3
- package/esm2022/lib/directives/let/let.directive.mjs +0 -40
- package/esm2022/lib/directives/let/let.types.mjs +0 -9
- package/esm2022/lib/directives/observe-content/index.mjs +0 -2
- package/esm2022/lib/directives/observe-content/observe-content.directive.mjs +0 -72
- package/esm2022/lib/directives/observe-resize/index.mjs +0 -2
- package/esm2022/lib/directives/observe-resize/observe-resize.directive.mjs +0 -72
- package/esm2022/lib/directives/observe-scroll-state/index.mjs +0 -4
- package/esm2022/lib/directives/observe-scroll-state/observe-scroll-state.constants.mjs +0 -3
- package/esm2022/lib/directives/observe-scroll-state/observe-scroll-state.directive.mjs +0 -180
- package/esm2022/lib/directives/observe-scroll-state/observe-scroll-state.types.mjs +0 -2
- package/esm2022/lib/directives/observe-visibility/index.mjs +0 -2
- package/esm2022/lib/directives/observe-visibility/observe-visibility.directive.mjs +0 -80
- package/esm2022/lib/directives/repeat/index.mjs +0 -2
- package/esm2022/lib/directives/repeat/repeat.directive.mjs +0 -35
- package/esm2022/lib/directives/root-boundary/index.mjs +0 -2
- package/esm2022/lib/directives/root-boundary/root-boundary.directive.mjs +0 -36
- package/esm2022/lib/directives/scroll-observer-first-element/index.mjs +0 -2
- package/esm2022/lib/directives/scroll-observer-first-element/scroll-observer-first-element.directive.mjs +0 -24
- package/esm2022/lib/directives/scroll-observer-ignore-target/index.mjs +0 -2
- package/esm2022/lib/directives/scroll-observer-ignore-target/scroll-observer-ignore-target.directive.mjs +0 -18
- package/esm2022/lib/directives/scroll-observer-last-element/index.mjs +0 -2
- package/esm2022/lib/directives/scroll-observer-last-element/scroll-observer-last-element.directive.mjs +0 -27
- package/esm2022/lib/directives/seo/index.mjs +0 -5
- package/esm2022/lib/directives/seo/seo.directive.constants.mjs +0 -3
- package/esm2022/lib/directives/seo/seo.directive.mjs +0 -166
- package/esm2022/lib/directives/seo/seo.directive.types.mjs +0 -2
- package/esm2022/lib/directives/seo/seo.directive.utils.mjs +0 -7
- package/esm2022/lib/pipes/infer-mime-type/index.mjs +0 -3
- package/esm2022/lib/pipes/infer-mime-type/infer-mime-type.pipe.mjs +0 -18
- package/esm2022/lib/pipes/infer-mime-type/infer-mime-type.util.mjs +0 -196
- package/esm2022/lib/pipes/normalize-game-result-type/index.mjs +0 -4
- package/esm2022/lib/pipes/normalize-game-result-type/normalize-game-result-type.pipe.mjs +0 -15
- package/esm2022/lib/pipes/normalize-game-result-type/normalize-game-result-type.types.mjs +0 -2
- package/esm2022/lib/pipes/normalize-game-result-type/normalize-game-result-type.util.mjs +0 -34
- package/esm2022/lib/pipes/normalize-match-participants/index.mjs +0 -4
- package/esm2022/lib/pipes/normalize-match-participants/normalize-match-participants.pipe.mjs +0 -15
- package/esm2022/lib/pipes/normalize-match-participants/normalize-match-participants.types.mjs +0 -2
- package/esm2022/lib/pipes/normalize-match-participants/normalize-match-participants.util.mjs +0 -47
- package/esm2022/lib/pipes/normalize-match-score/index.mjs +0 -4
- package/esm2022/lib/pipes/normalize-match-score/normalize-match-score.pipe.mjs +0 -15
- package/esm2022/lib/pipes/normalize-match-score/normalize-match-score.types.mjs +0 -2
- package/esm2022/lib/pipes/normalize-match-score/normalize-match-score.util.mjs +0 -126
- package/esm2022/lib/pipes/normalize-match-state/index.mjs +0 -4
- package/esm2022/lib/pipes/normalize-match-state/normalize-match-state.constants.mjs +0 -2
- package/esm2022/lib/pipes/normalize-match-state/normalize-match-state.pipe.mjs +0 -15
- package/esm2022/lib/pipes/normalize-match-state/normalize-match-state.util.mjs +0 -19
- package/esm2022/lib/pipes/normalize-match-type/index.mjs +0 -3
- package/esm2022/lib/pipes/normalize-match-type/normalize-match-type.pipe.mjs +0 -15
- package/esm2022/lib/pipes/normalize-match-type/normalize-match-type.util.mjs +0 -38
- package/esm2022/lib/pipes/to-array/index.mjs +0 -3
- package/esm2022/lib/pipes/to-array/to-array.pipe.mjs +0 -15
- package/esm2022/lib/pipes/to-array/to-array.util.mjs +0 -5
- package/esm2022/lib/props/create-prop-handlers.mjs +0 -17
- package/esm2022/lib/props/create-props.mjs +0 -53
- package/esm2022/lib/props/create-setup.mjs +0 -19
- package/esm2022/lib/props/dependency-stash.mjs +0 -20
- package/esm2022/lib/props/element-dictionary.mjs +0 -44
- package/esm2022/lib/props/index.mjs +0 -9
- package/esm2022/lib/props/props-binding.mjs +0 -53
- package/esm2022/lib/props/props.directive.mjs +0 -35
- package/esm2022/lib/props/template-input.mjs +0 -41
- package/esm2022/lib/services/click-observer.service.mjs +0 -73
- package/esm2022/lib/services/content-observer.service.mjs +0 -79
- package/esm2022/lib/services/focus-visible.service.mjs +0 -34
- package/esm2022/lib/services/index.mjs +0 -9
- package/esm2022/lib/services/intersection-observer.service.mjs +0 -75
- package/esm2022/lib/services/resize-observer.service.mjs +0 -75
- package/esm2022/lib/services/router-state.service.mjs +0 -205
- package/esm2022/lib/services/viewport.service.mjs +0 -244
- package/esm2022/lib/services/viewport.types.mjs +0 -2
- package/esm2022/lib/types/angular.types.mjs +0 -15
- package/esm2022/lib/types/i18n.types.mjs +0 -2
- package/esm2022/lib/types/index.mjs +0 -5
- package/esm2022/lib/types/value.types.mjs +0 -2
- package/esm2022/lib/types/viewport.types.mjs +0 -2
- package/esm2022/lib/utils/active-selection-model.utils.mjs +0 -91
- package/esm2022/lib/utils/angular.utils.mjs +0 -7
- package/esm2022/lib/utils/animation.utils.mjs +0 -113
- package/esm2022/lib/utils/clamp.util.mjs +0 -4
- package/esm2022/lib/utils/clone.util.mjs +0 -78
- package/esm2022/lib/utils/component-id.utils.mjs +0 -7
- package/esm2022/lib/utils/cookie.util.mjs +0 -53
- package/esm2022/lib/utils/destroy.utils.mjs +0 -13
- package/esm2022/lib/utils/equal.util.mjs +0 -118
- package/esm2022/lib/utils/form.utils.mjs +0 -63
- package/esm2022/lib/utils/index.mjs +0 -27
- package/esm2022/lib/utils/inject.utils.mjs +0 -4
- package/esm2022/lib/utils/intersection-observer.utils.mjs +0 -65
- package/esm2022/lib/utils/key-press-manager.utils.mjs +0 -30
- package/esm2022/lib/utils/media-query-observable.util.mjs +0 -18
- package/esm2022/lib/utils/mutation-observable.util.mjs +0 -42
- package/esm2022/lib/utils/object.utils.mjs +0 -30
- package/esm2022/lib/utils/reactive-binding.util.mjs +0 -124
- package/esm2022/lib/utils/resize-observable.util.mjs +0 -12
- package/esm2022/lib/utils/round.utils.mjs +0 -5
- package/esm2022/lib/utils/runtime-error.utils.mjs +0 -31
- package/esm2022/lib/utils/scrollable.utils.mjs +0 -157
- package/esm2022/lib/utils/selection-model.utils.mjs +0 -376
- package/esm2022/lib/utils/signal.utils.mjs +0 -1004
- package/esm2022/lib/utils/smart-block-scroll-strategy.utils.mjs +0 -96
- package/esm2022/lib/utils/value.utils.mjs +0 -22
- package/esm2022/lib/utils/viewport.util.mjs +0 -5
- package/esm2022/lib/validators/at-least-one-required.validator.mjs +0 -20
- package/esm2022/lib/validators/index.mjs +0 -15
- package/esm2022/lib/validators/is-array-not-empty.validator.mjs +0 -9
- package/esm2022/lib/validators/is-email.validator.mjs +0 -10
- package/esm2022/lib/validators/must-match.validator.mjs +0 -19
|
@@ -1,157 +0,0 @@
|
|
|
1
|
-
import { clamp } from './clamp.util';
|
|
2
|
-
export const elementCanScroll = (element, direction) => {
|
|
3
|
-
const { scrollHeight, clientHeight, scrollWidth, clientWidth } = element;
|
|
4
|
-
if (direction === 'x') {
|
|
5
|
-
return scrollWidth > clientWidth;
|
|
6
|
-
}
|
|
7
|
-
else if (direction === 'y') {
|
|
8
|
-
return scrollHeight > clientHeight;
|
|
9
|
-
}
|
|
10
|
-
return scrollHeight > clientHeight || scrollWidth > clientWidth;
|
|
11
|
-
};
|
|
12
|
-
export const isElementVisible = (options) => {
|
|
13
|
-
let { container } = options;
|
|
14
|
-
const { element } = options;
|
|
15
|
-
if (!element || container === null) {
|
|
16
|
-
return null;
|
|
17
|
-
}
|
|
18
|
-
container ||= document.documentElement;
|
|
19
|
-
const canScroll = elementCanScroll(container);
|
|
20
|
-
if (!canScroll) {
|
|
21
|
-
return {
|
|
22
|
-
inline: true,
|
|
23
|
-
block: true,
|
|
24
|
-
blockIntersection: 1,
|
|
25
|
-
inlineIntersection: 1,
|
|
26
|
-
intersectionRatio: 1,
|
|
27
|
-
isIntersecting: true,
|
|
28
|
-
element,
|
|
29
|
-
};
|
|
30
|
-
}
|
|
31
|
-
const elementRect = options.elementRect || element.getBoundingClientRect();
|
|
32
|
-
const containerRect = options.containerRect || container.getBoundingClientRect();
|
|
33
|
-
const elementInlineStart = elementRect.left;
|
|
34
|
-
const elementBlockStart = elementRect.top;
|
|
35
|
-
const containerInlineStart = containerRect.left;
|
|
36
|
-
const containerBlockStart = containerRect.top;
|
|
37
|
-
const elWith = elementRect.width || 1;
|
|
38
|
-
const elHeight = elementRect.height || 1;
|
|
39
|
-
const elementInlineEnd = elementInlineStart + elWith;
|
|
40
|
-
const elementBlockEnd = elementBlockStart + elHeight;
|
|
41
|
-
const containerInlineEnd = containerInlineStart + containerRect.width;
|
|
42
|
-
const containerBlockEnd = containerBlockStart + containerRect.height;
|
|
43
|
-
const isElementInlineVisible = elementInlineStart >= containerInlineStart && elementInlineEnd <= containerInlineEnd;
|
|
44
|
-
const isElementBlockVisible = elementBlockStart >= containerBlockStart && elementBlockEnd <= containerBlockEnd;
|
|
45
|
-
const inlineIntersection = Math.min(elementInlineEnd, containerInlineEnd) - Math.max(elementInlineStart, containerInlineStart);
|
|
46
|
-
const blockIntersection = Math.min(elementBlockEnd, containerBlockEnd) - Math.max(elementBlockStart, containerBlockStart);
|
|
47
|
-
const inlineIntersectionPercentage = clamp(inlineIntersection / elWith, 0, 1);
|
|
48
|
-
const blockIntersectionPercentage = clamp(blockIntersection / elHeight, 0, 1);
|
|
49
|
-
return {
|
|
50
|
-
inline: isElementInlineVisible,
|
|
51
|
-
block: isElementBlockVisible,
|
|
52
|
-
inlineIntersection: inlineIntersectionPercentage,
|
|
53
|
-
blockIntersection: blockIntersectionPercentage,
|
|
54
|
-
isIntersecting: isElementInlineVisible && isElementBlockVisible,
|
|
55
|
-
element,
|
|
56
|
-
// Round the intersection ratio to the nearest 0.01 to avoid floating point errors and system scaling issues.
|
|
57
|
-
intersectionRatio: Math.round(Math.min(inlineIntersectionPercentage, blockIntersectionPercentage) * 100) / 100,
|
|
58
|
-
};
|
|
59
|
-
};
|
|
60
|
-
export const getElementScrollCoordinates = (options) => {
|
|
61
|
-
let { container } = options;
|
|
62
|
-
const { element, direction, behavior = 'smooth', origin = 'nearest', scrollBlockMargin = 0, scrollInlineMargin = 0, } = options;
|
|
63
|
-
if (!element || container === null) {
|
|
64
|
-
return {
|
|
65
|
-
behavior,
|
|
66
|
-
left: undefined,
|
|
67
|
-
top: undefined,
|
|
68
|
-
};
|
|
69
|
-
}
|
|
70
|
-
container ||= document.documentElement;
|
|
71
|
-
const canScroll = elementCanScroll(container);
|
|
72
|
-
if (!canScroll) {
|
|
73
|
-
return {
|
|
74
|
-
behavior,
|
|
75
|
-
left: undefined,
|
|
76
|
-
top: undefined,
|
|
77
|
-
};
|
|
78
|
-
}
|
|
79
|
-
const elementRect = element.getBoundingClientRect();
|
|
80
|
-
const containerRect = container.getBoundingClientRect();
|
|
81
|
-
const { scrollLeft, scrollTop } = container;
|
|
82
|
-
const { width: elementWidth, height: elementHeight, left: elementLeft, top: elementTop } = elementRect;
|
|
83
|
-
const { width: containerWidth, height: containerHeight, left: containerLeft, top: containerTop } = containerRect;
|
|
84
|
-
const shouldScrollLeft = direction === 'inline' || direction === 'both' || !direction;
|
|
85
|
-
const shouldScrollTop = direction === 'block' || direction === 'both' || !direction;
|
|
86
|
-
let scrollLeftTo = scrollLeft;
|
|
87
|
-
let scrollTopTo = scrollTop;
|
|
88
|
-
const scrollToElementStart = () => {
|
|
89
|
-
const relativeTop = elementTop - containerTop;
|
|
90
|
-
const relativeLeft = elementLeft - containerLeft;
|
|
91
|
-
const amountToScrollTop = relativeTop;
|
|
92
|
-
const amountToScrollLeft = relativeLeft;
|
|
93
|
-
const scrollTopPosition = scrollTop + amountToScrollTop;
|
|
94
|
-
const scrollLeftPosition = scrollLeft + amountToScrollLeft;
|
|
95
|
-
scrollLeftTo = scrollLeftPosition - scrollInlineMargin;
|
|
96
|
-
scrollTopTo = scrollTopPosition - scrollBlockMargin;
|
|
97
|
-
};
|
|
98
|
-
const scrollToElementEnd = () => {
|
|
99
|
-
const relativeTop = elementTop - containerTop;
|
|
100
|
-
const relativeLeft = elementLeft - containerLeft;
|
|
101
|
-
const amountToScrollTop = relativeTop - containerHeight + elementHeight;
|
|
102
|
-
const amountToScrollLeft = relativeLeft - containerWidth + elementWidth;
|
|
103
|
-
const scrollTopPosition = scrollTop + amountToScrollTop;
|
|
104
|
-
const scrollLeftPosition = scrollLeft + amountToScrollLeft;
|
|
105
|
-
scrollLeftTo = scrollLeftPosition + scrollInlineMargin;
|
|
106
|
-
scrollTopTo = scrollTopPosition + scrollBlockMargin;
|
|
107
|
-
};
|
|
108
|
-
const scrollToElementCenter = () => {
|
|
109
|
-
const relativeTop = elementTop - containerTop;
|
|
110
|
-
const relativeLeft = elementLeft - containerLeft;
|
|
111
|
-
const amountToScrollTop = relativeTop - containerHeight / 2 + elementHeight / 2;
|
|
112
|
-
const amountToScrollLeft = relativeLeft - containerWidth / 2 + elementWidth / 2;
|
|
113
|
-
const scrollTopPosition = scrollTop + amountToScrollTop;
|
|
114
|
-
const scrollLeftPosition = scrollLeft + amountToScrollLeft;
|
|
115
|
-
scrollLeftTo = scrollLeftPosition;
|
|
116
|
-
scrollTopTo = scrollTopPosition;
|
|
117
|
-
};
|
|
118
|
-
const scrollToElementNearest = () => {
|
|
119
|
-
const isAbove = elementRect.bottom < containerRect.top;
|
|
120
|
-
const isPartialAbove = elementRect.top < containerRect.top && elementRect.bottom > containerRect.top;
|
|
121
|
-
const isBelow = elementRect.top > containerRect.bottom;
|
|
122
|
-
const isPartialBelow = elementRect.top < containerRect.bottom && elementRect.bottom > containerRect.bottom;
|
|
123
|
-
const isLeft = elementRect.right < containerRect.left;
|
|
124
|
-
const isPartialLeft = elementRect.left < containerRect.left && elementRect.right > containerRect.left;
|
|
125
|
-
const isRight = elementRect.left > containerRect.right;
|
|
126
|
-
const isPartialRight = elementRect.left < containerRect.right && elementRect.right > containerRect.right;
|
|
127
|
-
if (isAbove || isPartialAbove || isLeft || isPartialLeft) {
|
|
128
|
-
scrollToElementStart();
|
|
129
|
-
}
|
|
130
|
-
else if (isBelow || isPartialBelow || isRight || isPartialRight) {
|
|
131
|
-
scrollToElementEnd();
|
|
132
|
-
}
|
|
133
|
-
};
|
|
134
|
-
switch (origin) {
|
|
135
|
-
case 'start':
|
|
136
|
-
scrollToElementStart();
|
|
137
|
-
break;
|
|
138
|
-
case 'end':
|
|
139
|
-
scrollToElementEnd();
|
|
140
|
-
break;
|
|
141
|
-
case 'center':
|
|
142
|
-
scrollToElementCenter();
|
|
143
|
-
break;
|
|
144
|
-
case 'nearest':
|
|
145
|
-
scrollToElementNearest();
|
|
146
|
-
break;
|
|
147
|
-
}
|
|
148
|
-
return {
|
|
149
|
-
behavior,
|
|
150
|
-
left: shouldScrollLeft ? scrollLeftTo : undefined,
|
|
151
|
-
top: shouldScrollTop ? scrollTopTo : undefined,
|
|
152
|
-
};
|
|
153
|
-
};
|
|
154
|
-
export const scrollToElement = (options) => {
|
|
155
|
-
options.container?.scrollTo(getElementScrollCoordinates(options));
|
|
156
|
-
};
|
|
157
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"scrollable.utils.js","sourceRoot":"","sources":["../../../../../../libs/core/src/lib/utils/scrollable.utils.ts"],"names":[],"mappings":"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","sourcesContent":["import { clamp } from './clamp.util';\n\nexport const elementCanScroll = (element: HTMLElement, direction?: 'x' | 'y') => {\n  const { scrollHeight, clientHeight, scrollWidth, clientWidth } = element;\n\n  if (direction === 'x') {\n    return scrollWidth > clientWidth;\n  } else if (direction === 'y') {\n    return scrollHeight > clientHeight;\n  }\n\n  return scrollHeight > clientHeight || scrollWidth > clientWidth;\n};\n\nexport interface IsElementVisibleOptions {\n  /**\n   * The element to check if it is visible inside a container.\n   */\n  element?: HTMLElement | null;\n\n  /**\n   * The container to check if the element is visible inside.\n   * @default document.documentElement\n   */\n  container?: HTMLElement | null;\n\n  /**\n   * The container's rect to check if the element is visible inside. Can be supplied to reduce the amount of DOM reads.\n   * @default container.getBoundingClientRect()\n   */\n  containerRect?: DOMRect | null;\n\n  /**\n   * The element's rect. Can be supplied to reduce the amount of DOM reads.\n   * @default container.getBoundingClientRect()\n   */\n  elementRect?: DOMRect | null;\n}\n\nexport interface CurrentElementVisibility {\n  /**\n   * Whether the element is visible in the inline direction.\n   */\n  inline: boolean;\n\n  /**\n   * Whether the element is visible in the block direction.\n   */\n  block: boolean;\n\n  /**\n   * The percentage of the element that is visible in the inline direction.\n   */\n  inlineIntersection: number;\n\n  /**\n   * The percentage of the element that is visible in the block direction.\n   */\n  blockIntersection: number;\n\n  /**\n   * Whether the element is intersecting the container.\n   */\n  isIntersecting: boolean;\n\n  /**\n   * The ratio of the element that is intersecting the container.\n   */\n  intersectionRatio: number;\n\n  /**\n   * The element that is being checked for visibility.\n   */\n  element: HTMLElement;\n}\n\nexport const isElementVisible = (options: IsElementVisibleOptions): CurrentElementVisibility | null => {\n  let { container } = options;\n  const { element } = options;\n\n  if (!element || container === null) {\n    return null;\n  }\n\n  container ||= document.documentElement;\n\n  const canScroll = elementCanScroll(container);\n\n  if (!canScroll) {\n    return {\n      inline: true,\n      block: true,\n      blockIntersection: 1,\n      inlineIntersection: 1,\n      intersectionRatio: 1,\n      isIntersecting: true,\n      element,\n    };\n  }\n\n  const elementRect = options.elementRect || element.getBoundingClientRect();\n  const containerRect = options.containerRect || container.getBoundingClientRect();\n\n  const elementInlineStart = elementRect.left;\n  const elementBlockStart = elementRect.top;\n\n  const containerInlineStart = containerRect.left;\n  const containerBlockStart = containerRect.top;\n\n  const elWith = elementRect.width || 1;\n  const elHeight = elementRect.height || 1;\n\n  const elementInlineEnd = elementInlineStart + elWith;\n  const elementBlockEnd = elementBlockStart + elHeight;\n\n  const containerInlineEnd = containerInlineStart + containerRect.width;\n  const containerBlockEnd = containerBlockStart + containerRect.height;\n\n  const isElementInlineVisible = elementInlineStart >= containerInlineStart && elementInlineEnd <= containerInlineEnd;\n  const isElementBlockVisible = elementBlockStart >= containerBlockStart && elementBlockEnd <= containerBlockEnd;\n\n  const inlineIntersection =\n    Math.min(elementInlineEnd, containerInlineEnd) - Math.max(elementInlineStart, containerInlineStart);\n  const blockIntersection =\n    Math.min(elementBlockEnd, containerBlockEnd) - Math.max(elementBlockStart, containerBlockStart);\n\n  const inlineIntersectionPercentage = clamp(inlineIntersection / elWith, 0, 1);\n  const blockIntersectionPercentage = clamp(blockIntersection / elHeight, 0, 1);\n\n  return {\n    inline: isElementInlineVisible,\n    block: isElementBlockVisible,\n    inlineIntersection: inlineIntersectionPercentage,\n    blockIntersection: blockIntersectionPercentage,\n    isIntersecting: isElementInlineVisible && isElementBlockVisible,\n    element,\n\n    // Round the intersection ratio to the nearest 0.01 to avoid floating point errors and system scaling issues.\n    intersectionRatio: Math.round(Math.min(inlineIntersectionPercentage, blockIntersectionPercentage) * 100) / 100,\n  };\n};\n\nexport const getElementScrollCoordinates = (options: ScrollToElementOptions): ScrollToOptions => {\n  let { container } = options;\n  const {\n    element,\n    direction,\n    behavior = 'smooth',\n    origin = 'nearest',\n    scrollBlockMargin = 0,\n    scrollInlineMargin = 0,\n  } = options;\n\n  if (!element || container === null) {\n    return {\n      behavior,\n      left: undefined,\n      top: undefined,\n    };\n  }\n\n  container ||= document.documentElement;\n\n  const canScroll = elementCanScroll(container);\n\n  if (!canScroll) {\n    return {\n      behavior,\n      left: undefined,\n      top: undefined,\n    };\n  }\n\n  const elementRect = element.getBoundingClientRect();\n  const containerRect = container.getBoundingClientRect();\n\n  const { scrollLeft, scrollTop } = container;\n  const { width: elementWidth, height: elementHeight, left: elementLeft, top: elementTop } = elementRect;\n  const { width: containerWidth, height: containerHeight, left: containerLeft, top: containerTop } = containerRect;\n\n  const shouldScrollLeft = direction === 'inline' || direction === 'both' || !direction;\n  const shouldScrollTop = direction === 'block' || direction === 'both' || !direction;\n\n  let scrollLeftTo = scrollLeft;\n  let scrollTopTo = scrollTop;\n\n  const scrollToElementStart = () => {\n    const relativeTop = elementTop - containerTop;\n    const relativeLeft = elementLeft - containerLeft;\n\n    const amountToScrollTop = relativeTop;\n    const amountToScrollLeft = relativeLeft;\n\n    const scrollTopPosition = scrollTop + amountToScrollTop;\n    const scrollLeftPosition = scrollLeft + amountToScrollLeft;\n\n    scrollLeftTo = scrollLeftPosition - scrollInlineMargin;\n    scrollTopTo = scrollTopPosition - scrollBlockMargin;\n  };\n\n  const scrollToElementEnd = () => {\n    const relativeTop = elementTop - containerTop;\n    const relativeLeft = elementLeft - containerLeft;\n\n    const amountToScrollTop = relativeTop - containerHeight + elementHeight;\n    const amountToScrollLeft = relativeLeft - containerWidth + elementWidth;\n\n    const scrollTopPosition = scrollTop + amountToScrollTop;\n    const scrollLeftPosition = scrollLeft + amountToScrollLeft;\n\n    scrollLeftTo = scrollLeftPosition + scrollInlineMargin;\n    scrollTopTo = scrollTopPosition + scrollBlockMargin;\n  };\n\n  const scrollToElementCenter = () => {\n    const relativeTop = elementTop - containerTop;\n    const relativeLeft = elementLeft - containerLeft;\n\n    const amountToScrollTop = relativeTop - containerHeight / 2 + elementHeight / 2;\n    const amountToScrollLeft = relativeLeft - containerWidth / 2 + elementWidth / 2;\n\n    const scrollTopPosition = scrollTop + amountToScrollTop;\n    const scrollLeftPosition = scrollLeft + amountToScrollLeft;\n\n    scrollLeftTo = scrollLeftPosition;\n    scrollTopTo = scrollTopPosition;\n  };\n\n  const scrollToElementNearest = () => {\n    const isAbove = elementRect.bottom < containerRect.top;\n    const isPartialAbove = elementRect.top < containerRect.top && elementRect.bottom > containerRect.top;\n\n    const isBelow = elementRect.top > containerRect.bottom;\n    const isPartialBelow = elementRect.top < containerRect.bottom && elementRect.bottom > containerRect.bottom;\n\n    const isLeft = elementRect.right < containerRect.left;\n    const isPartialLeft = elementRect.left < containerRect.left && elementRect.right > containerRect.left;\n\n    const isRight = elementRect.left > containerRect.right;\n    const isPartialRight = elementRect.left < containerRect.right && elementRect.right > containerRect.right;\n\n    if (isAbove || isPartialAbove || isLeft || isPartialLeft) {\n      scrollToElementStart();\n    } else if (isBelow || isPartialBelow || isRight || isPartialRight) {\n      scrollToElementEnd();\n    }\n  };\n\n  switch (origin) {\n    case 'start':\n      scrollToElementStart();\n      break;\n    case 'end':\n      scrollToElementEnd();\n      break;\n    case 'center':\n      scrollToElementCenter();\n      break;\n    case 'nearest':\n      scrollToElementNearest();\n      break;\n  }\n\n  return {\n    behavior,\n    left: shouldScrollLeft ? scrollLeftTo : undefined,\n    top: shouldScrollTop ? scrollTopTo : undefined,\n  };\n};\n\nexport interface ScrollToElementOptions {\n  /**\n   * The element to scroll to.\n   */\n  element?: HTMLElement | null;\n\n  /**\n   * The scroll container to scroll to the element in.\n   * @default document.documentElement\n   */\n  container?: HTMLElement | null;\n\n  /**\n   * The direction to scroll in.\n   * @default 'both'\n   */\n  direction?: 'inline' | 'block' | 'both';\n\n  /**\n   * The origin of the element to scroll to.\n   * @default 'nearest'\n   */\n  origin?: 'start' | 'end' | 'center' | 'nearest';\n\n  /**\n   * The scroll behavior.\n   * @default 'smooth'\n   */\n  behavior?: ScrollBehavior;\n\n  /**\n   * The scroll inline-margin\n   * @default 0\n   */\n  scrollInlineMargin?: number;\n\n  /**\n   * The scroll block-margin\n   * @default 0\n   */\n  scrollBlockMargin?: number;\n}\n\nexport const scrollToElement = (options: ScrollToElementOptions) => {\n  options.container?.scrollTo(getElementScrollCoordinates(options));\n};\n"]}
|
|
@@ -1,376 +0,0 @@
|
|
|
1
|
-
import { assertInInjectionContext } from '@angular/core';
|
|
2
|
-
import { takeUntilDestroyed, toSignal } from '@angular/core/rxjs-interop';
|
|
3
|
-
import { BehaviorSubject, combineLatest, filter, map, shareReplay, startWith, take, takeUntil, tap, } from 'rxjs';
|
|
4
|
-
import { switchQueryListChanges } from './angular.utils';
|
|
5
|
-
import { createDestroy } from './destroy.utils';
|
|
6
|
-
import { getObjectProperty, isObject } from './object.utils';
|
|
7
|
-
export class SelectionModel {
|
|
8
|
-
get selection$() {
|
|
9
|
-
return this._selection$.asObservable();
|
|
10
|
-
}
|
|
11
|
-
get selection() {
|
|
12
|
-
return this._selection$.value;
|
|
13
|
-
}
|
|
14
|
-
get options$() {
|
|
15
|
-
return this._options$.asObservable();
|
|
16
|
-
}
|
|
17
|
-
get options() {
|
|
18
|
-
return this._options$.value;
|
|
19
|
-
}
|
|
20
|
-
get valueBinding$() {
|
|
21
|
-
return this._valueBinding$.asObservable();
|
|
22
|
-
}
|
|
23
|
-
get valueBinding() {
|
|
24
|
-
return this._valueBinding$.value;
|
|
25
|
-
}
|
|
26
|
-
get keyBinding$() {
|
|
27
|
-
return combineLatest([this._keyBinding$, this.valueBinding$]).pipe(map(([keyBinding, valueBinding]) => keyBinding || valueBinding));
|
|
28
|
-
}
|
|
29
|
-
get keyBinding() {
|
|
30
|
-
return this._keyBinding$.value || this.valueBinding;
|
|
31
|
-
}
|
|
32
|
-
get labelBinding$() {
|
|
33
|
-
return this._labelBinding$.asObservable();
|
|
34
|
-
}
|
|
35
|
-
get labelBinding() {
|
|
36
|
-
return this._labelBinding$.value;
|
|
37
|
-
}
|
|
38
|
-
get disabledBinding$() {
|
|
39
|
-
return this._disabledBinding$.asObservable();
|
|
40
|
-
}
|
|
41
|
-
get disabledBinding() {
|
|
42
|
-
return this._disabledBinding$.value;
|
|
43
|
-
}
|
|
44
|
-
get allowMultiple$() {
|
|
45
|
-
return this._allowMultiple$.asObservable();
|
|
46
|
-
}
|
|
47
|
-
get allowMultiple() {
|
|
48
|
-
return this._allowMultiple$.value;
|
|
49
|
-
}
|
|
50
|
-
get filter$() {
|
|
51
|
-
return this._filter$.asObservable();
|
|
52
|
-
}
|
|
53
|
-
get filter() {
|
|
54
|
-
return this._filter$.value;
|
|
55
|
-
}
|
|
56
|
-
constructor() {
|
|
57
|
-
this._destroy$ = createDestroy();
|
|
58
|
-
this._lastSelectionSetSubscription = null;
|
|
59
|
-
this._selection$ = new BehaviorSubject([]);
|
|
60
|
-
this._options$ = new BehaviorSubject([]);
|
|
61
|
-
this.optionsSignal = toSignal(this.options$);
|
|
62
|
-
this._valueBinding$ = new BehaviorSubject(null);
|
|
63
|
-
this._keyBinding$ = new BehaviorSubject(null);
|
|
64
|
-
this._labelBinding$ = new BehaviorSubject(null);
|
|
65
|
-
this._disabledBinding$ = new BehaviorSubject(null);
|
|
66
|
-
this._allowMultiple$ = new BehaviorSubject(false);
|
|
67
|
-
this._filter$ = new BehaviorSubject('');
|
|
68
|
-
this.value$ = combineLatest([this.selection$, this.valueBinding$, this.allowMultiple$]).pipe(map(([selection, valueBinding, allowMultiple]) => {
|
|
69
|
-
if (allowMultiple) {
|
|
70
|
-
return selection.map((option) => this.execFnOrGetOptionProperty(option, valueBinding));
|
|
71
|
-
}
|
|
72
|
-
const [option] = selection;
|
|
73
|
-
if (!option)
|
|
74
|
-
return null;
|
|
75
|
-
return this.execFnOrGetOptionProperty(option, valueBinding);
|
|
76
|
-
}));
|
|
77
|
-
this.filteredOptions$ = combineLatest([this.options$, this.filter$, this.labelBinding$]).pipe(map(([options, filter]) => this.getFilteredOptions(filter, options)), shareReplay({ bufferSize: 1, refCount: true }));
|
|
78
|
-
this._optionsAndSelection$ = combineLatest([this._options$, this._selection$]);
|
|
79
|
-
this.trackByOptionKey = (option) => this.getKey(option);
|
|
80
|
-
this.allowMultiple$
|
|
81
|
-
.pipe(tap(() => {
|
|
82
|
-
if (this.allowMultiple)
|
|
83
|
-
return;
|
|
84
|
-
const [option] = this.selection;
|
|
85
|
-
if (!option)
|
|
86
|
-
return;
|
|
87
|
-
this.setSelection([option]);
|
|
88
|
-
}), takeUntilDestroyed())
|
|
89
|
-
.subscribe();
|
|
90
|
-
}
|
|
91
|
-
setSelection(selection) {
|
|
92
|
-
if (!Array.isArray(selection)) {
|
|
93
|
-
selection = [selection];
|
|
94
|
-
}
|
|
95
|
-
this._selection$.next(selection);
|
|
96
|
-
return this;
|
|
97
|
-
}
|
|
98
|
-
setSelectionFromValue(value) {
|
|
99
|
-
if (Array.isArray(value)) {
|
|
100
|
-
const selection = value.map((v) => this.getOptionByValue(v)).filter((v) => v !== undefined);
|
|
101
|
-
this.setSelection(selection);
|
|
102
|
-
}
|
|
103
|
-
else {
|
|
104
|
-
const selection = this.getOptionByValue(value);
|
|
105
|
-
if (selection) {
|
|
106
|
-
this.setSelection(selection);
|
|
107
|
-
}
|
|
108
|
-
else {
|
|
109
|
-
this.clearSelectedOptions();
|
|
110
|
-
}
|
|
111
|
-
}
|
|
112
|
-
}
|
|
113
|
-
setSelectionFromValue$(value) {
|
|
114
|
-
this._lastSelectionSetSubscription?.unsubscribe();
|
|
115
|
-
this._lastSelectionSetSubscription = this.options$
|
|
116
|
-
.pipe(takeUntil(this._destroy$), filter((o) => !!o.length), tap(() => this.setSelectionFromValue(value)), take(1))
|
|
117
|
-
.subscribe();
|
|
118
|
-
}
|
|
119
|
-
setValueBinding(fnOrPropertyPath) {
|
|
120
|
-
this._valueBinding$.next(fnOrPropertyPath);
|
|
121
|
-
return this;
|
|
122
|
-
}
|
|
123
|
-
setKeyBinding(fnOrPropertyPath) {
|
|
124
|
-
this._keyBinding$.next(fnOrPropertyPath);
|
|
125
|
-
return this;
|
|
126
|
-
}
|
|
127
|
-
setLabelBinding(fnOrPropertyPath) {
|
|
128
|
-
this._labelBinding$.next(fnOrPropertyPath);
|
|
129
|
-
return this;
|
|
130
|
-
}
|
|
131
|
-
setDisabledBinding(fnOrPropertyPath) {
|
|
132
|
-
this._disabledBinding$.next(fnOrPropertyPath);
|
|
133
|
-
return this;
|
|
134
|
-
}
|
|
135
|
-
setOptions(options) {
|
|
136
|
-
this._options$.next(options);
|
|
137
|
-
return this;
|
|
138
|
-
}
|
|
139
|
-
setOptionsFromQueryList(queryList) {
|
|
140
|
-
assertInInjectionContext(this.setOptionsFromQueryList);
|
|
141
|
-
queryList.changes
|
|
142
|
-
.pipe(startWith(queryList), tap((list) => this.setOptions(list.toArray())), takeUntilDestroyed())
|
|
143
|
-
.subscribe();
|
|
144
|
-
return this;
|
|
145
|
-
}
|
|
146
|
-
setOptionsFromQueryList$(queryList$) {
|
|
147
|
-
assertInInjectionContext(this.setOptionsFromQueryList$);
|
|
148
|
-
queryList$
|
|
149
|
-
.pipe(switchQueryListChanges(), tap((list) => this.setOptions(list?.toArray() || [])), takeUntilDestroyed())
|
|
150
|
-
.subscribe();
|
|
151
|
-
return this;
|
|
152
|
-
}
|
|
153
|
-
setAllowMultiple(allowMultiple) {
|
|
154
|
-
this._allowMultiple$.next(allowMultiple);
|
|
155
|
-
return this;
|
|
156
|
-
}
|
|
157
|
-
setFilter(filter) {
|
|
158
|
-
const sanitizedFilter = filter?.trim() || '';
|
|
159
|
-
this._filter$.next(sanitizedFilter);
|
|
160
|
-
}
|
|
161
|
-
reset() {
|
|
162
|
-
this._selection$.next([]);
|
|
163
|
-
this._options$.next([]);
|
|
164
|
-
this._valueBinding$.next(null);
|
|
165
|
-
this._keyBinding$.next(null);
|
|
166
|
-
this._disabledBinding$.next(null);
|
|
167
|
-
this._labelBinding$.next(null);
|
|
168
|
-
this._allowMultiple$.next(false);
|
|
169
|
-
}
|
|
170
|
-
getOptionByValue$(value) {
|
|
171
|
-
return this._optionsAndSelection$.pipe(map(() => this.getOptionByValue(value)));
|
|
172
|
-
}
|
|
173
|
-
getOptionByLabel$(label) {
|
|
174
|
-
return this._optionsAndSelection$.pipe(map(() => this.getOptionByLabel(label)));
|
|
175
|
-
}
|
|
176
|
-
getOptionByKey$(key) {
|
|
177
|
-
return this._optionsAndSelection$.pipe(map(() => this.getOptionByKey(key)));
|
|
178
|
-
}
|
|
179
|
-
isSelected$(option) {
|
|
180
|
-
return this._selection$.pipe(map(() => this.isSelected(option)));
|
|
181
|
-
}
|
|
182
|
-
isDisabled$(option) {
|
|
183
|
-
return this._optionsAndSelection$.pipe(map(() => this.isDisabled(option)));
|
|
184
|
-
}
|
|
185
|
-
getLabel$(option) {
|
|
186
|
-
return this.labelBinding$.pipe(map((labelBinding) => this.execFnOrGetOptionProperty(option, labelBinding)));
|
|
187
|
-
}
|
|
188
|
-
getValue$(option) {
|
|
189
|
-
return this.valueBinding$.pipe(map((valueBinding) => this.execFnOrGetOptionProperty(option, valueBinding)));
|
|
190
|
-
}
|
|
191
|
-
getKey$(option) {
|
|
192
|
-
return this.keyBinding$.pipe(map((keyBinding) => this.execFnOrGetOptionProperty(option, keyBinding)));
|
|
193
|
-
}
|
|
194
|
-
getOptionByValue(value) {
|
|
195
|
-
return this.getOption(value, this.valueBinding);
|
|
196
|
-
}
|
|
197
|
-
getOptionByLabel(label) {
|
|
198
|
-
return this.getOption(label, this.labelBinding);
|
|
199
|
-
}
|
|
200
|
-
getOptionByKey(key) {
|
|
201
|
-
return this.getOption(key, this.keyBinding);
|
|
202
|
-
}
|
|
203
|
-
getOption(value, propertyPath = null, options = [...this.options, ...this.selection]) {
|
|
204
|
-
if (!propertyPath) {
|
|
205
|
-
return options.find((option) => option === value);
|
|
206
|
-
}
|
|
207
|
-
return options.find((option) => {
|
|
208
|
-
if (!isObject(option))
|
|
209
|
-
return false;
|
|
210
|
-
return this.execFnOrGetOptionProperty(option, propertyPath) === value;
|
|
211
|
-
});
|
|
212
|
-
}
|
|
213
|
-
getOptionByIndex(index, options = this.getFilteredOptions()) {
|
|
214
|
-
return options[index] || null;
|
|
215
|
-
}
|
|
216
|
-
getOptionByOffset(offset, index, config = { clamp: true }) {
|
|
217
|
-
const { loop, clamp, skipDisabled, options = this.getFilteredOptions() } = config;
|
|
218
|
-
const newIndex = index + offset;
|
|
219
|
-
const remainingOffset = newIndex * -1;
|
|
220
|
-
let optionResult = null;
|
|
221
|
-
if (newIndex < 0) {
|
|
222
|
-
if (loop) {
|
|
223
|
-
optionResult = this.getOptionByOffset(remainingOffset, options.length - 1, config);
|
|
224
|
-
}
|
|
225
|
-
else if (clamp) {
|
|
226
|
-
optionResult = this.getFirstOption();
|
|
227
|
-
}
|
|
228
|
-
else {
|
|
229
|
-
optionResult = null;
|
|
230
|
-
}
|
|
231
|
-
}
|
|
232
|
-
else if (newIndex >= options.length) {
|
|
233
|
-
if (loop) {
|
|
234
|
-
optionResult = this.getOptionByOffset(remainingOffset, 0, config);
|
|
235
|
-
}
|
|
236
|
-
else if (clamp) {
|
|
237
|
-
optionResult = this.getLastOption();
|
|
238
|
-
}
|
|
239
|
-
else {
|
|
240
|
-
optionResult = null;
|
|
241
|
-
}
|
|
242
|
-
}
|
|
243
|
-
else {
|
|
244
|
-
optionResult = this.getOptionByIndex(newIndex);
|
|
245
|
-
}
|
|
246
|
-
if (optionResult && skipDisabled && this.isDisabled(optionResult)) {
|
|
247
|
-
return this.getOptionByOffset(offset, newIndex, config);
|
|
248
|
-
}
|
|
249
|
-
return optionResult;
|
|
250
|
-
}
|
|
251
|
-
getFirstOption(options = this.getFilteredOptions()) {
|
|
252
|
-
return this.getOptionByIndex(0, options);
|
|
253
|
-
}
|
|
254
|
-
getLastOption(options = this.getFilteredOptions()) {
|
|
255
|
-
return this.getOptionByIndex(options.length - 1, options);
|
|
256
|
-
}
|
|
257
|
-
getOptionIndex(option, options = this.getFilteredOptions()) {
|
|
258
|
-
const key = this.getKey(option);
|
|
259
|
-
const index = options.findIndex((o) => this.getKey(o) === key);
|
|
260
|
-
return index === -1 ? null : index;
|
|
261
|
-
}
|
|
262
|
-
getNonMultipleSelectedOption() {
|
|
263
|
-
if (this.allowMultiple)
|
|
264
|
-
return null;
|
|
265
|
-
return this.selection[0] || null;
|
|
266
|
-
}
|
|
267
|
-
getNonMultipleSelectedOptionIndex() {
|
|
268
|
-
if (this.allowMultiple)
|
|
269
|
-
return null;
|
|
270
|
-
const opt = this.getNonMultipleSelectedOption();
|
|
271
|
-
if (!opt)
|
|
272
|
-
return null;
|
|
273
|
-
return this.getOptionIndex(opt);
|
|
274
|
-
}
|
|
275
|
-
getNonMultipleOptionByOffsetFromSelected(offset, config = { clamp: true }) {
|
|
276
|
-
if (this.allowMultiple)
|
|
277
|
-
return null;
|
|
278
|
-
const index = this.getNonMultipleSelectedOptionIndex();
|
|
279
|
-
if (index === null)
|
|
280
|
-
return null;
|
|
281
|
-
return this.getOptionByOffset(index, offset, config);
|
|
282
|
-
}
|
|
283
|
-
getLabel(option) {
|
|
284
|
-
return this.execFnOrGetOptionProperty(option, this.labelBinding);
|
|
285
|
-
}
|
|
286
|
-
getValue(option) {
|
|
287
|
-
return this.execFnOrGetOptionProperty(option, this.valueBinding);
|
|
288
|
-
}
|
|
289
|
-
getKey(option) {
|
|
290
|
-
return this.execFnOrGetOptionProperty(option, this.keyBinding);
|
|
291
|
-
}
|
|
292
|
-
getDisabled(option) {
|
|
293
|
-
return this.execFnOrGetOptionProperty(option, this.disabledBinding);
|
|
294
|
-
}
|
|
295
|
-
execFnOrGetOptionProperty(option, fnOrPropertyPath) {
|
|
296
|
-
if (!fnOrPropertyPath || !isObject(option))
|
|
297
|
-
return option;
|
|
298
|
-
if (typeof fnOrPropertyPath === 'function') {
|
|
299
|
-
return fnOrPropertyPath(option);
|
|
300
|
-
}
|
|
301
|
-
return getObjectProperty(option, fnOrPropertyPath);
|
|
302
|
-
}
|
|
303
|
-
execFnOrGetOptionPropertyNullable(option, fnOrPropertyPath) {
|
|
304
|
-
if (!fnOrPropertyPath || !isObject(option))
|
|
305
|
-
return null;
|
|
306
|
-
return this.execFnOrGetOptionProperty(option, fnOrPropertyPath);
|
|
307
|
-
}
|
|
308
|
-
getOptionProperty(option, propertyPath) {
|
|
309
|
-
if (!propertyPath || !isObject(option))
|
|
310
|
-
return option;
|
|
311
|
-
return getObjectProperty(option, propertyPath);
|
|
312
|
-
}
|
|
313
|
-
isSelected(option) {
|
|
314
|
-
const optionKey = this.execFnOrGetOptionProperty(option, this.keyBinding);
|
|
315
|
-
return this.getOption(optionKey, this.keyBinding, this.selection) !== undefined;
|
|
316
|
-
}
|
|
317
|
-
isDisabled(option) {
|
|
318
|
-
return !!this.execFnOrGetOptionPropertyNullable(option, this.disabledBinding);
|
|
319
|
-
}
|
|
320
|
-
getFilteredOptions(filter = this.filter, options = this.options) {
|
|
321
|
-
if (!filter)
|
|
322
|
-
return options;
|
|
323
|
-
const splitFilter = filter.split(' ').filter((f) => f);
|
|
324
|
-
return options.filter((option) => {
|
|
325
|
-
const label = this.getLabel(option);
|
|
326
|
-
if (!label || typeof label !== 'string')
|
|
327
|
-
return false;
|
|
328
|
-
const splitLabel = label.split(' ').filter((l) => l);
|
|
329
|
-
return splitFilter.every((f) => splitLabel.some((l) => l.toLowerCase().includes(f.toLowerCase())));
|
|
330
|
-
});
|
|
331
|
-
}
|
|
332
|
-
addSelectedOption(option) {
|
|
333
|
-
if (this.allowMultiple) {
|
|
334
|
-
if (this.isSelected(option))
|
|
335
|
-
return;
|
|
336
|
-
this._selection$.next([...this.selection, option]);
|
|
337
|
-
}
|
|
338
|
-
else {
|
|
339
|
-
this._selection$.next([option]);
|
|
340
|
-
}
|
|
341
|
-
}
|
|
342
|
-
removeSelectedOption(option) {
|
|
343
|
-
const optionKey = this.execFnOrGetOptionProperty(option, this.keyBinding);
|
|
344
|
-
if (!this.isSelected(option))
|
|
345
|
-
return;
|
|
346
|
-
this._selection$.next(this.selection.filter((selectedOption) => {
|
|
347
|
-
const key = this.execFnOrGetOptionProperty(selectedOption, this.keyBinding);
|
|
348
|
-
return key !== optionKey;
|
|
349
|
-
}));
|
|
350
|
-
}
|
|
351
|
-
clearSelectedOptions() {
|
|
352
|
-
this._selection$.next([]);
|
|
353
|
-
}
|
|
354
|
-
toggleSelectedOption(option) {
|
|
355
|
-
if (this.isSelected(option)) {
|
|
356
|
-
this.removeSelectedOption(option);
|
|
357
|
-
}
|
|
358
|
-
else {
|
|
359
|
-
this.addSelectedOption(option);
|
|
360
|
-
}
|
|
361
|
-
}
|
|
362
|
-
// FIXME: Toggle should respect disabled binding if it exists
|
|
363
|
-
toggleAllSelectedOptions() {
|
|
364
|
-
const filteredOptions = this.getFilteredOptions();
|
|
365
|
-
const unselectedOptions = filteredOptions.filter((option) => !this.isSelected(option));
|
|
366
|
-
if (unselectedOptions.length) {
|
|
367
|
-
const selection = unselectedOptions.filter((o) => !this.selection.some((s) => this.getKey(s) === this.getKey(o)));
|
|
368
|
-
this._selection$.next([...this.selection, ...selection]);
|
|
369
|
-
}
|
|
370
|
-
else {
|
|
371
|
-
const selection = this.selection.filter((s) => !filteredOptions.some((o) => this.getKey(o) === this.getKey(s)));
|
|
372
|
-
this._selection$.next(selection);
|
|
373
|
-
}
|
|
374
|
-
}
|
|
375
|
-
}
|
|
376
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"selection-model.utils.js","sourceRoot":"","sources":["../../../../../../libs/core/src/lib/utils/selection-model.utils.ts"],"names":[],"mappings":"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","sourcesContent":["import { assertInInjectionContext } from '@angular/core';\nimport { takeUntilDestroyed, toSignal } from '@angular/core/rxjs-interop';\nimport {\n  BehaviorSubject,\n  Observable,\n  Subscription,\n  combineLatest,\n  filter,\n  map,\n  shareReplay,\n  startWith,\n  take,\n  takeUntil,\n  tap,\n} from 'rxjs';\nimport { TypedQueryList } from '../types';\nimport { switchQueryListChanges } from './angular.utils';\nimport { createDestroy } from './destroy.utils';\nimport { getObjectProperty, isObject } from './object.utils';\n\nexport type SelectionModelTypes = string | number | Record<string, unknown> | unknown;\nexport type SelectionModelPropertyPath = string;\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nexport type SelectionModelOptionValueFn<T extends SelectionModelTypes = any> = (option: T) => unknown;\n\n/**\n * You can use a property path or a function to get the value of an option. The function should be as **lightweight as possible** as it will be called **a lot**.\n *\n * @example\n * // Property path\n * \"id\" // option.id\n * \"user.name\" // option.user.name\n *\n * // Function\n * (option) => option.id // option.id\n * (option) => option.user.name // option.user.name\n */\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nexport type SelectionModelBinding<T extends SelectionModelTypes = any> =\n  | SelectionModelPropertyPath\n  | SelectionModelOptionValueFn<T>;\n\nexport class SelectionModel<T extends SelectionModelTypes = unknown> {\n  private readonly _destroy$ = createDestroy();\n  private _lastSelectionSetSubscription: Subscription | null = null;\n\n  get selection$() {\n    return this._selection$.asObservable();\n  }\n  get selection() {\n    return this._selection$.value;\n  }\n  private readonly _selection$ = new BehaviorSubject<T[]>([]);\n\n  get options$() {\n    return this._options$.asObservable();\n  }\n  get options() {\n    return this._options$.value;\n  }\n  private readonly _options$ = new BehaviorSubject<T[]>([]);\n  readonly optionsSignal = toSignal(this.options$);\n\n  get valueBinding$() {\n    return this._valueBinding$.asObservable();\n  }\n  get valueBinding() {\n    return this._valueBinding$.value;\n  }\n  private readonly _valueBinding$ = new BehaviorSubject<SelectionModelBinding<T> | null>(null);\n\n  get keyBinding$() {\n    return combineLatest([this._keyBinding$, this.valueBinding$]).pipe(\n      map(([keyBinding, valueBinding]) => keyBinding || valueBinding),\n    );\n  }\n  get keyBinding() {\n    return this._keyBinding$.value || this.valueBinding;\n  }\n  private readonly _keyBinding$ = new BehaviorSubject<SelectionModelBinding<T> | null>(null);\n\n  get labelBinding$() {\n    return this._labelBinding$.asObservable();\n  }\n  get labelBinding() {\n    return this._labelBinding$.value;\n  }\n  private readonly _labelBinding$ = new BehaviorSubject<SelectionModelBinding<T> | null>(null);\n\n  get disabledBinding$() {\n    return this._disabledBinding$.asObservable();\n  }\n  get disabledBinding() {\n    return this._disabledBinding$.value;\n  }\n  private readonly _disabledBinding$ = new BehaviorSubject<SelectionModelBinding<T> | null>(null);\n\n  get allowMultiple$() {\n    return this._allowMultiple$.asObservable();\n  }\n  get allowMultiple() {\n    return this._allowMultiple$.value;\n  }\n  private readonly _allowMultiple$ = new BehaviorSubject<boolean>(false);\n\n  get filter$() {\n    return this._filter$.asObservable();\n  }\n  get filter() {\n    return this._filter$.value;\n  }\n  private readonly _filter$ = new BehaviorSubject<string>('');\n\n  readonly value$ = combineLatest([this.selection$, this.valueBinding$, this.allowMultiple$]).pipe(\n    map(([selection, valueBinding, allowMultiple]) => {\n      if (allowMultiple) {\n        return selection.map((option) => this.execFnOrGetOptionProperty(option, valueBinding));\n      }\n\n      const [option] = selection;\n\n      if (!option) return null;\n\n      return this.execFnOrGetOptionProperty(option, valueBinding);\n    }),\n  );\n\n  readonly filteredOptions$ = combineLatest([this.options$, this.filter$, this.labelBinding$]).pipe(\n    map(([options, filter]) => this.getFilteredOptions(filter, options)),\n    shareReplay({ bufferSize: 1, refCount: true }),\n  );\n\n  private readonly _optionsAndSelection$ = combineLatest([this._options$, this._selection$]);\n\n  constructor() {\n    this.allowMultiple$\n      .pipe(\n        tap(() => {\n          if (this.allowMultiple) return;\n\n          const [option] = this.selection;\n\n          if (!option) return;\n\n          this.setSelection([option]);\n        }),\n        takeUntilDestroyed(),\n      )\n      .subscribe();\n  }\n\n  setSelection(selection: T | T[]) {\n    if (!Array.isArray(selection)) {\n      selection = [selection];\n    }\n\n    this._selection$.next(selection);\n\n    return this;\n  }\n\n  setSelectionFromValue(value: unknown) {\n    if (Array.isArray(value)) {\n      const selection = value.map((v) => this.getOptionByValue(v)).filter((v): v is T => v !== undefined);\n\n      this.setSelection(selection);\n    } else {\n      const selection = this.getOptionByValue(value);\n\n      if (selection) {\n        this.setSelection(selection);\n      } else {\n        this.clearSelectedOptions();\n      }\n    }\n  }\n\n  setSelectionFromValue$(value: unknown) {\n    this._lastSelectionSetSubscription?.unsubscribe();\n\n    this._lastSelectionSetSubscription = this.options$\n      .pipe(\n        takeUntil(this._destroy$),\n        filter((o) => !!o.length),\n        tap(() => this.setSelectionFromValue(value)),\n        take(1),\n      )\n      .subscribe();\n  }\n\n  setValueBinding(fnOrPropertyPath: SelectionModelBinding<T> | null) {\n    this._valueBinding$.next(fnOrPropertyPath);\n\n    return this;\n  }\n\n  setKeyBinding(fnOrPropertyPath: SelectionModelBinding<T> | null) {\n    this._keyBinding$.next(fnOrPropertyPath);\n\n    return this;\n  }\n\n  setLabelBinding(fnOrPropertyPath: SelectionModelBinding<T> | null) {\n    this._labelBinding$.next(fnOrPropertyPath);\n\n    return this;\n  }\n\n  setDisabledBinding(fnOrPropertyPath: SelectionModelBinding<T> | null) {\n    this._disabledBinding$.next(fnOrPropertyPath);\n\n    return this;\n  }\n\n  setOptions(options: T[]) {\n    this._options$.next(options);\n\n    return this;\n  }\n\n  setOptionsFromQueryList(queryList: TypedQueryList<T>) {\n    assertInInjectionContext(this.setOptionsFromQueryList);\n\n    queryList.changes\n      .pipe(\n        startWith(queryList),\n        tap((list) => this.setOptions(list.toArray())),\n        takeUntilDestroyed(),\n      )\n      .subscribe();\n\n    return this;\n  }\n\n  setOptionsFromQueryList$(queryList$: Observable<TypedQueryList<T> | null | undefined>) {\n    assertInInjectionContext(this.setOptionsFromQueryList$);\n\n    queryList$\n      .pipe(\n        switchQueryListChanges(),\n        tap((list) => this.setOptions(list?.toArray() || [])),\n        takeUntilDestroyed(),\n      )\n      .subscribe();\n\n    return this;\n  }\n\n  setAllowMultiple(allowMultiple: boolean) {\n    this._allowMultiple$.next(allowMultiple);\n\n    return this;\n  }\n\n  setFilter(filter: string | null) {\n    const sanitizedFilter = filter?.trim() || '';\n\n    this._filter$.next(sanitizedFilter);\n  }\n\n  reset() {\n    this._selection$.next([]);\n    this._options$.next([]);\n    this._valueBinding$.next(null);\n    this._keyBinding$.next(null);\n    this._disabledBinding$.next(null);\n    this._labelBinding$.next(null);\n    this._allowMultiple$.next(false);\n  }\n\n  trackByOptionKey = (option: T) => this.getKey(option);\n\n  getOptionByValue$(value: unknown) {\n    return this._optionsAndSelection$.pipe(map(() => this.getOptionByValue(value)));\n  }\n\n  getOptionByLabel$(label: string) {\n    return this._optionsAndSelection$.pipe(map(() => this.getOptionByLabel(label)));\n  }\n\n  getOptionByKey$(key: string) {\n    return this._optionsAndSelection$.pipe(map(() => this.getOptionByKey(key)));\n  }\n\n  isSelected$(option: T) {\n    return this._selection$.pipe(map(() => this.isSelected(option)));\n  }\n\n  isDisabled$(option: T) {\n    return this._optionsAndSelection$.pipe(map(() => this.isDisabled(option)));\n  }\n\n  getLabel$(option: T) {\n    return this.labelBinding$.pipe(map((labelBinding) => this.execFnOrGetOptionProperty(option, labelBinding)));\n  }\n\n  getValue$(option: T) {\n    return this.valueBinding$.pipe(map((valueBinding) => this.execFnOrGetOptionProperty(option, valueBinding)));\n  }\n\n  getKey$(option: T) {\n    return this.keyBinding$.pipe(map((keyBinding) => this.execFnOrGetOptionProperty(option, keyBinding)));\n  }\n\n  getOptionByValue(value: unknown) {\n    return this.getOption(value, this.valueBinding);\n  }\n\n  getOptionByLabel(label: string) {\n    return this.getOption(label, this.labelBinding);\n  }\n\n  getOptionByKey(key: string) {\n    return this.getOption(key, this.keyBinding);\n  }\n\n  getOption(\n    value: unknown,\n    propertyPath: SelectionModelBinding<T> | null = null,\n    options = [...this.options, ...this.selection],\n  ) {\n    if (!propertyPath) {\n      return options.find((option) => option === value);\n    }\n\n    return options.find((option) => {\n      if (!isObject(option)) return false;\n\n      return this.execFnOrGetOptionProperty(option, propertyPath) === value;\n    });\n  }\n\n  getOptionByIndex(index: number, options = this.getFilteredOptions()): T | null {\n    return options[index] || null;\n  }\n\n  getOptionByOffset(\n    offset: number,\n    index: number,\n    config: { loop?: boolean; clamp?: boolean; skipDisabled?: boolean; options?: T[] } = { clamp: true },\n  ): T | null {\n    const { loop, clamp, skipDisabled, options = this.getFilteredOptions() } = config;\n\n    const newIndex = index + offset;\n    const remainingOffset = newIndex * -1;\n\n    let optionResult: T | null = null;\n\n    if (newIndex < 0) {\n      if (loop) {\n        optionResult = this.getOptionByOffset(remainingOffset, options.length - 1, config);\n      } else if (clamp) {\n        optionResult = this.getFirstOption();\n      } else {\n        optionResult = null;\n      }\n    } else if (newIndex >= options.length) {\n      if (loop) {\n        optionResult = this.getOptionByOffset(remainingOffset, 0, config);\n      } else if (clamp) {\n        optionResult = this.getLastOption();\n      } else {\n        optionResult = null;\n      }\n    } else {\n      optionResult = this.getOptionByIndex(newIndex);\n    }\n\n    if (optionResult && skipDisabled && this.isDisabled(optionResult)) {\n      return this.getOptionByOffset(offset, newIndex, config);\n    }\n\n    return optionResult;\n  }\n\n  getFirstOption(options = this.getFilteredOptions()): T | null {\n    return this.getOptionByIndex(0, options);\n  }\n\n  getLastOption(options = this.getFilteredOptions()): T | null {\n    return this.getOptionByIndex(options.length - 1, options);\n  }\n\n  getOptionIndex(option: T, options = this.getFilteredOptions()) {\n    const key = this.getKey(option);\n    const index = options.findIndex((o) => this.getKey(o) === key);\n\n    return index === -1 ? null : index;\n  }\n\n  getNonMultipleSelectedOption(): T | null {\n    if (this.allowMultiple) return null;\n\n    return this.selection[0] || null;\n  }\n\n  getNonMultipleSelectedOptionIndex() {\n    if (this.allowMultiple) return null;\n\n    const opt = this.getNonMultipleSelectedOption();\n\n    if (!opt) return null;\n\n    return this.getOptionIndex(opt);\n  }\n\n  getNonMultipleOptionByOffsetFromSelected(\n    offset: number,\n    config: { loop?: boolean; clamp?: boolean } = { clamp: true },\n  ) {\n    if (this.allowMultiple) return null;\n\n    const index = this.getNonMultipleSelectedOptionIndex();\n\n    if (index === null) return null;\n\n    return this.getOptionByOffset(index, offset, config);\n  }\n\n  getLabel(option: T) {\n    return this.execFnOrGetOptionProperty(option, this.labelBinding);\n  }\n\n  getValue(option: T) {\n    return this.execFnOrGetOptionProperty(option, this.valueBinding);\n  }\n\n  getKey(option: T) {\n    return this.execFnOrGetOptionProperty(option, this.keyBinding);\n  }\n\n  getDisabled(option: T) {\n    return this.execFnOrGetOptionProperty(option, this.disabledBinding);\n  }\n\n  execFnOrGetOptionProperty(option: T, fnOrPropertyPath: SelectionModelBinding<T> | null) {\n    if (!fnOrPropertyPath || !isObject(option)) return option;\n\n    if (typeof fnOrPropertyPath === 'function') {\n      return fnOrPropertyPath(option);\n    }\n\n    return getObjectProperty(option, fnOrPropertyPath);\n  }\n\n  execFnOrGetOptionPropertyNullable(option: T, fnOrPropertyPath: SelectionModelBinding<T> | null) {\n    if (!fnOrPropertyPath || !isObject(option)) return null;\n\n    return this.execFnOrGetOptionProperty(option, fnOrPropertyPath);\n  }\n\n  getOptionProperty(option: T, propertyPath: string | null) {\n    if (!propertyPath || !isObject(option)) return option;\n\n    return getObjectProperty(option, propertyPath);\n  }\n\n  isSelected(option: T) {\n    const optionKey = this.execFnOrGetOptionProperty(option, this.keyBinding);\n\n    return this.getOption(optionKey, this.keyBinding, this.selection) !== undefined;\n  }\n\n  isDisabled(option: T) {\n    return !!this.execFnOrGetOptionPropertyNullable(option, this.disabledBinding);\n  }\n\n  getFilteredOptions(filter = this.filter, options = this.options) {\n    if (!filter) return options;\n\n    const splitFilter = filter.split(' ').filter((f) => f);\n\n    return options.filter((option) => {\n      const label = this.getLabel(option);\n\n      if (!label || typeof label !== 'string') return false;\n\n      const splitLabel = label.split(' ').filter((l) => l);\n\n      return splitFilter.every((f) => splitLabel.some((l) => l.toLowerCase().includes(f.toLowerCase())));\n    });\n  }\n\n  addSelectedOption(option: T) {\n    if (this.allowMultiple) {\n      if (this.isSelected(option)) return;\n\n      this._selection$.next([...this.selection, option]);\n    } else {\n      this._selection$.next([option]);\n    }\n  }\n\n  removeSelectedOption(option: T) {\n    const optionKey = this.execFnOrGetOptionProperty(option, this.keyBinding);\n\n    if (!this.isSelected(option)) return;\n\n    this._selection$.next(\n      this.selection.filter((selectedOption) => {\n        const key = this.execFnOrGetOptionProperty(selectedOption, this.keyBinding);\n\n        return key !== optionKey;\n      }),\n    );\n  }\n\n  clearSelectedOptions() {\n    this._selection$.next([]);\n  }\n\n  toggleSelectedOption(option: T) {\n    if (this.isSelected(option)) {\n      this.removeSelectedOption(option);\n    } else {\n      this.addSelectedOption(option);\n    }\n  }\n\n  // FIXME: Toggle should respect disabled binding if it exists\n  toggleAllSelectedOptions() {\n    const filteredOptions = this.getFilteredOptions();\n    const unselectedOptions = filteredOptions.filter((option) => !this.isSelected(option));\n\n    if (unselectedOptions.length) {\n      const selection = unselectedOptions.filter((o) => !this.selection.some((s) => this.getKey(s) === this.getKey(o)));\n\n      this._selection$.next([...this.selection, ...selection]);\n    } else {\n      const selection = this.selection.filter((s) => !filteredOptions.some((o) => this.getKey(o) === this.getKey(s)));\n\n      this._selection$.next(selection);\n    }\n  }\n}\n"]}
|