@radix-ng/primitives 0.18.2 → 0.20.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/avatar/index.d.ts +1 -2
- package/avatar/src/avatar-fallback.directive.d.ts +11 -23
- package/avatar/src/avatar-image.directive.d.ts +10 -14
- package/avatar/src/avatar-root.directive.d.ts +5 -15
- package/checkbox/index.d.ts +11 -0
- package/checkbox/src/checkbox-button.directive.d.ts +1 -1
- package/checkbox/src/checkbox-indicator.directive.d.ts +1 -1
- package/checkbox/src/checkbox-input.directive.d.ts +1 -1
- package/checkbox/src/checkbox.directive.d.ts +1 -1
- package/compodoc/documentation.json +9385 -4858
- package/core/index.d.ts +2 -0
- package/core/src/control-value-accessor/index.d.ts +75 -0
- package/core/src/create-inject-context/assert-injector.d.ts +51 -0
- package/core/src/create-inject-context/index.d.ts +68 -0
- package/core/src/types.d.ts +23 -0
- package/esm2022/avatar/index.mjs +1 -1
- package/esm2022/avatar/src/avatar-fallback.directive.mjs +38 -40
- package/esm2022/avatar/src/avatar-image.directive.mjs +25 -26
- package/esm2022/avatar/src/avatar-root.directive.mjs +13 -25
- package/esm2022/checkbox/index.mjs +31 -1
- package/esm2022/checkbox/src/checkbox-button.directive.mjs +3 -3
- package/esm2022/checkbox/src/checkbox-indicator.directive.mjs +3 -3
- package/esm2022/checkbox/src/checkbox-input.directive.mjs +3 -3
- package/esm2022/checkbox/src/checkbox.directive.mjs +3 -3
- package/esm2022/core/index.mjs +3 -1
- package/esm2022/core/src/control-value-accessor/index.mjs +103 -0
- package/esm2022/core/src/create-inject-context/assert-injector.mjs +15 -0
- package/esm2022/core/src/create-inject-context/index.mjs +116 -0
- package/esm2022/core/src/types.mjs +2 -0
- package/esm2022/popover/index.mjs +41 -0
- package/esm2022/popover/radix-ng-primitives-popover.mjs +5 -0
- package/esm2022/popover/src/popover-arrow.directive.mjs +112 -0
- package/esm2022/popover/src/popover-arrow.token.mjs +3 -0
- package/esm2022/popover/src/popover-close.directive.mjs +37 -0
- package/esm2022/popover/src/popover-content.directive.mjs +227 -0
- package/esm2022/popover/src/popover-root.directive.mjs +142 -0
- package/esm2022/popover/src/popover-root.inject.mjs +7 -0
- package/esm2022/popover/src/popover-root.token.mjs +3 -0
- package/esm2022/popover/src/popover-trigger.directive.mjs +42 -0
- package/esm2022/popover/src/popover.constants.mjs +90 -0
- package/esm2022/popover/src/popover.types.mjs +14 -0
- package/esm2022/popover/src/popover.utils.mjs +115 -0
- package/esm2022/radio/index.mjs +2 -1
- package/esm2022/radio/src/radio-item-input.directive.mjs +37 -0
- package/esm2022/radio/src/radio-item.directive.mjs +55 -39
- package/esm2022/radio/src/radio-root.directive.mjs +30 -120
- package/esm2022/radio/src/radio-tokens.mjs +1 -1
- package/esm2022/roving-focus/index.mjs +3 -0
- package/esm2022/roving-focus/radix-ng-primitives-roving-focus.mjs +5 -0
- package/esm2022/roving-focus/src/roving-focus-group.directive.mjs +138 -0
- package/esm2022/roving-focus/src/roving-focus-item.directive.mjs +133 -0
- package/esm2022/roving-focus/src/utils.mjs +47 -0
- package/esm2022/select/src/select-item.directive.mjs +8 -2
- package/esm2022/toggle/src/toggle-input.directive.mjs +4 -3
- package/fesm2022/radix-ng-primitives-avatar.mjs +70 -85
- package/fesm2022/radix-ng-primitives-avatar.mjs.map +1 -1
- package/fesm2022/radix-ng-primitives-checkbox.mjs +35 -10
- package/fesm2022/radix-ng-primitives-checkbox.mjs.map +1 -1
- package/fesm2022/radix-ng-primitives-core.mjs +230 -3
- package/fesm2022/radix-ng-primitives-core.mjs.map +1 -1
- package/fesm2022/radix-ng-primitives-popover.mjs +796 -0
- package/fesm2022/radix-ng-primitives-popover.mjs.map +1 -0
- package/fesm2022/radix-ng-primitives-radio.mjs +145 -186
- package/fesm2022/radix-ng-primitives-radio.mjs.map +1 -1
- package/fesm2022/radix-ng-primitives-roving-focus.mjs +320 -0
- package/fesm2022/radix-ng-primitives-roving-focus.mjs.map +1 -0
- package/fesm2022/radix-ng-primitives-select.mjs +7 -1
- package/fesm2022/radix-ng-primitives-select.mjs.map +1 -1
- package/fesm2022/radix-ng-primitives-toggle.mjs +3 -2
- package/fesm2022/radix-ng-primitives-toggle.mjs.map +1 -1
- package/package.json +18 -6
- package/popover/README.md +3 -0
- package/popover/index.d.ts +17 -0
- package/popover/src/popover-arrow.directive.d.ts +37 -0
- package/popover/src/popover-arrow.token.d.ts +3 -0
- package/popover/src/popover-close.directive.d.ts +15 -0
- package/popover/src/popover-content.directive.d.ts +84 -0
- package/popover/src/popover-root.directive.d.ts +58 -0
- package/popover/src/popover-root.inject.d.ts +2 -0
- package/popover/src/popover-root.token.d.ts +3 -0
- package/popover/src/popover-trigger.directive.d.ts +18 -0
- package/popover/src/popover.constants.d.ts +8 -0
- package/popover/src/popover.types.d.ts +34 -0
- package/popover/src/popover.utils.d.ts +12 -0
- package/radio/index.d.ts +1 -0
- package/radio/src/radio-item-input.directive.d.ts +12 -0
- package/radio/src/radio-item.directive.d.ts +23 -14
- package/radio/src/radio-root.directive.d.ts +19 -34
- package/radio/src/radio-tokens.d.ts +6 -4
- package/roving-focus/README.md +3 -0
- package/roving-focus/index.d.ts +3 -0
- package/roving-focus/src/roving-focus-group.directive.d.ts +50 -0
- package/roving-focus/src/roving-focus-item.directive.d.ts +50 -0
- package/roving-focus/src/utils.d.ts +19 -0
- package/select/src/select-item.directive.d.ts +7 -1
- package/toggle/src/toggle-input.directive.d.ts +1 -1
@@ -1,27 +1,21 @@
|
|
1
|
-
import {
|
2
|
-
import { DOWN_ARROW, ENTER, LEFT_ARROW, RIGHT_ARROW, SPACE, TAB, UP_ARROW } from '@angular/cdk/keycodes';
|
3
|
-
import { booleanAttribute, ContentChildren, Directive, EventEmitter, Input, Output, QueryList } from '@angular/core';
|
1
|
+
import { booleanAttribute, computed, Directive, input, Input, model, output, signal } from '@angular/core';
|
4
2
|
import { NG_VALUE_ACCESSOR } from '@angular/forms';
|
5
|
-
import {
|
6
|
-
import { RdxRadioItemDirective } from './radio-item.directive';
|
3
|
+
import { RdxRovingFocusGroupDirective } from '@radix-ng/primitives/roving-focus';
|
7
4
|
import { RDX_RADIO_GROUP } from './radio-tokens';
|
8
5
|
import * as i0 from "@angular/core";
|
6
|
+
import * as i1 from "@radix-ng/primitives/roving-focus";
|
9
7
|
export class RdxRadioGroupDirective {
|
10
8
|
constructor() {
|
11
|
-
this.
|
12
|
-
this.disabled = false;
|
13
|
-
|
14
|
-
|
15
|
-
* Horizontal radio buttons can sometimes be challenging to scan and localize.
|
16
|
-
* The horizontal arrangement of radio buttons may also lead to difficulties in determining which
|
17
|
-
* label corresponds to which button: whether the label is above or below the button.
|
18
|
-
* @default 'vertical'
|
19
|
-
*/
|
20
|
-
this._orientation = 'vertical';
|
9
|
+
this.value = model(null);
|
10
|
+
this.disabled = input(false, { transform: booleanAttribute });
|
11
|
+
this.required = input(false, { transform: booleanAttribute });
|
12
|
+
this.orientation = input();
|
21
13
|
/**
|
22
14
|
* Event handler called when the value changes.
|
23
15
|
*/
|
24
|
-
this.onValueChange =
|
16
|
+
this.onValueChange = output();
|
17
|
+
this.disable = signal(this.disabled());
|
18
|
+
this.disableState = computed(() => this.disable() || this.disabled());
|
25
19
|
/**
|
26
20
|
* The callback function to call when the value of the radio group changes.
|
27
21
|
*/
|
@@ -30,131 +24,60 @@ export class RdxRadioGroupDirective {
|
|
30
24
|
};
|
31
25
|
/**
|
32
26
|
* The callback function to call when the radio group is touched.
|
27
|
+
* @ignore
|
33
28
|
*/
|
34
29
|
this.onTouched = () => {
|
35
30
|
/* Empty */
|
36
31
|
};
|
37
32
|
}
|
38
|
-
ngAfterContentInit() {
|
39
|
-
this.focusKeyManager = new FocusKeyManager(this.radioItems).withWrap().withVerticalOrientation();
|
40
|
-
this.radioItems.changes.pipe(takeUntil(this.destroy$)).subscribe(() => {
|
41
|
-
this.updateActiveItem();
|
42
|
-
});
|
43
|
-
this.updateActiveItem(false);
|
44
|
-
}
|
45
|
-
ngOnDestroy() {
|
46
|
-
this.destroy$.next();
|
47
|
-
this.destroy$.complete();
|
48
|
-
}
|
49
33
|
/**
|
50
34
|
* Select a radio item.
|
51
35
|
* @param value The value of the radio item to select.
|
36
|
+
* @ignore
|
52
37
|
*/
|
53
38
|
select(value) {
|
54
|
-
this.value
|
39
|
+
this.value.set(value);
|
55
40
|
this.onValueChange.emit(value);
|
56
41
|
this.onChange?.(value);
|
57
|
-
this.updateActiveItem();
|
58
42
|
this.onTouched();
|
59
43
|
}
|
60
44
|
/**
|
61
45
|
* Update the value of the radio group.
|
62
46
|
* @param value The new value of the radio group.
|
63
|
-
* @
|
47
|
+
* @ignore
|
64
48
|
*/
|
65
49
|
writeValue(value) {
|
66
|
-
this.value
|
67
|
-
if (this.radioItems) {
|
68
|
-
this.updateActiveItem(false);
|
69
|
-
}
|
50
|
+
this.value.set(value);
|
70
51
|
}
|
71
52
|
/**
|
72
53
|
* Register a callback function to call when the value of the radio group changes.
|
73
54
|
* @param fn The callback function to call when the value of the radio group changes.
|
74
|
-
* @
|
55
|
+
* @ignore
|
75
56
|
*/
|
76
57
|
registerOnChange(fn) {
|
77
58
|
this.onChange = fn;
|
78
59
|
}
|
60
|
+
/** @ignore */
|
79
61
|
registerOnTouched(fn) {
|
80
62
|
this.onTouched = fn;
|
81
63
|
}
|
82
64
|
/**
|
83
65
|
* Set the disabled state of the radio group.
|
84
66
|
* @param isDisabled Whether the radio group is disabled.
|
85
|
-
* @
|
67
|
+
* @ignore
|
86
68
|
*/
|
87
69
|
setDisabledState(isDisabled) {
|
88
|
-
this.
|
89
|
-
}
|
90
|
-
/**
|
91
|
-
* When focus leaves the radio group.
|
92
|
-
*/
|
93
|
-
onFocusin(event) {
|
94
|
-
const target = event.target;
|
95
|
-
const radioItem = this.radioItems.find((item) => item.element.nativeElement === target);
|
96
|
-
if (radioItem) {
|
97
|
-
this.focusKeyManager.setActiveItem(radioItem);
|
98
|
-
}
|
70
|
+
this.disable.set(isDisabled);
|
99
71
|
}
|
100
|
-
onKeydown(
|
101
|
-
if (this.
|
72
|
+
onKeydown() {
|
73
|
+
if (this.disableState())
|
102
74
|
return;
|
103
|
-
switch (event.keyCode) {
|
104
|
-
case ENTER:
|
105
|
-
case SPACE:
|
106
|
-
event.preventDefault();
|
107
|
-
this.selectFocusedItem();
|
108
|
-
break;
|
109
|
-
case DOWN_ARROW:
|
110
|
-
case RIGHT_ARROW:
|
111
|
-
event.preventDefault();
|
112
|
-
this.focusKeyManager.setNextItemActive();
|
113
|
-
this.selectFocusedItem();
|
114
|
-
break;
|
115
|
-
case UP_ARROW:
|
116
|
-
case LEFT_ARROW:
|
117
|
-
event.preventDefault();
|
118
|
-
this.focusKeyManager.setPreviousItemActive();
|
119
|
-
this.selectFocusedItem();
|
120
|
-
break;
|
121
|
-
case TAB:
|
122
|
-
this.tabNavigation(event);
|
123
|
-
break;
|
124
|
-
default:
|
125
|
-
this.focusKeyManager.onKeydown(event);
|
126
|
-
}
|
127
|
-
}
|
128
|
-
selectFocusedItem() {
|
129
|
-
const focusedItem = this.focusKeyManager.activeItem;
|
130
|
-
if (focusedItem) {
|
131
|
-
this.select(focusedItem.value);
|
132
|
-
}
|
133
|
-
}
|
134
|
-
updateActiveItem(setFocus = true) {
|
135
|
-
const activeItem = this.radioItems.find((item) => item.value === this.value);
|
136
|
-
if (activeItem) {
|
137
|
-
this.focusKeyManager.setActiveItem(activeItem);
|
138
|
-
}
|
139
|
-
else if (this.radioItems.length > 0 && setFocus) {
|
140
|
-
this.focusKeyManager.setFirstItemActive();
|
141
|
-
}
|
142
|
-
}
|
143
|
-
tabNavigation(event) {
|
144
|
-
event.preventDefault();
|
145
|
-
const checkedItem = this.radioItems.find((item) => item.checked);
|
146
|
-
if (checkedItem) {
|
147
|
-
checkedItem.focus();
|
148
|
-
}
|
149
|
-
else if (this.radioItems.first) {
|
150
|
-
this.radioItems.first.focus();
|
151
|
-
}
|
152
75
|
}
|
153
76
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.11", ngImport: i0, type: RdxRadioGroupDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
|
154
|
-
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "
|
77
|
+
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "17.1.0", version: "18.2.11", type: RdxRadioGroupDirective, isStandalone: true, selector: "[rdxRadioRoot]", inputs: { value: { classPropertyName: "value", publicName: "value", isSignal: true, isRequired: false, transformFunction: null }, disabled: { classPropertyName: "disabled", publicName: "disabled", isSignal: true, isRequired: false, transformFunction: null }, defaultValue: { classPropertyName: "defaultValue", publicName: "defaultValue", isSignal: false, isRequired: false, transformFunction: null }, required: { classPropertyName: "required", publicName: "required", isSignal: true, isRequired: false, transformFunction: null }, orientation: { classPropertyName: "orientation", publicName: "orientation", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { value: "valueChange", onValueChange: "onValueChange" }, host: { attributes: { "role": "radiogroup" }, listeners: { "keydown": "onKeydown()" }, properties: { "attr.aria-orientation": "orientation()", "attr.aria-required": "required()", "attr.data-disabled": "disableState() ? \"\" : null" } }, providers: [
|
155
78
|
{ provide: RDX_RADIO_GROUP, useExisting: RdxRadioGroupDirective },
|
156
79
|
{ provide: NG_VALUE_ACCESSOR, useExisting: RdxRadioGroupDirective, multi: true }
|
157
|
-
],
|
80
|
+
], exportAs: ["rdxRadioRoot"], hostDirectives: [{ directive: i1.RdxRovingFocusGroupDirective, inputs: ["dir", "dir", "orientation", "orientation", "loop", "loop"] }], ngImport: i0 }); }
|
158
81
|
}
|
159
82
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.11", ngImport: i0, type: RdxRadioGroupDirective, decorators: [{
|
160
83
|
type: Directive,
|
@@ -166,29 +89,16 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.11", ngImpo
|
|
166
89
|
{ provide: RDX_RADIO_GROUP, useExisting: RdxRadioGroupDirective },
|
167
90
|
{ provide: NG_VALUE_ACCESSOR, useExisting: RdxRadioGroupDirective, multi: true }
|
168
91
|
],
|
92
|
+
hostDirectives: [{ directive: RdxRovingFocusGroupDirective, inputs: ['dir', 'orientation', 'loop'] }],
|
169
93
|
host: {
|
170
94
|
role: 'radiogroup',
|
171
|
-
'[attr.aria-orientation]': '
|
172
|
-
'[attr.
|
173
|
-
'[attr.
|
174
|
-
'
|
175
|
-
'(keydown)': 'onKeydown($event)',
|
176
|
-
'(focusin)': 'onFocusin($event)'
|
95
|
+
'[attr.aria-orientation]': 'orientation()',
|
96
|
+
'[attr.aria-required]': 'required()',
|
97
|
+
'[attr.data-disabled]': 'disableState() ? "" : null',
|
98
|
+
'(keydown)': 'onKeydown()'
|
177
99
|
}
|
178
100
|
}]
|
179
|
-
}], propDecorators: {
|
180
|
-
type: ContentChildren,
|
181
|
-
args: [RdxRadioItemDirective, { descendants: true }]
|
182
|
-
}], value: [{
|
183
|
-
type: Input
|
184
|
-
}], disabled: [{
|
185
|
-
type: Input,
|
186
|
-
args: [{ transform: booleanAttribute }]
|
187
|
-
}], dir: [{
|
188
|
-
type: Input
|
189
|
-
}], defaultValue: [{
|
101
|
+
}], propDecorators: { defaultValue: [{
|
190
102
|
type: Input
|
191
|
-
}], onValueChange: [{
|
192
|
-
type: Output
|
193
103
|
}] } });
|
194
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"radio-root.directive.js","sourceRoot":"","sources":["../../../../../packages/primitives/radio/src/radio-root.directive.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAC;AACpD,OAAO,EAAE,UAAU,EAAE,KAAK,EAAE,UAAU,EAAE,WAAW,EAAE,KAAK,EAAE,GAAG,EAAE,QAAQ,EAAE,MAAM,uBAAuB,CAAC;AACzG,OAAO,EAEH,gBAAgB,EAChB,eAAe,EACf,SAAS,EACT,YAAY,EACZ,KAAK,EAEL,MAAM,EACN,SAAS,EACZ,MAAM,eAAe,CAAC;AACvB,OAAO,EAAwB,iBAAiB,EAAE,MAAM,gBAAgB,CAAC;AACzE,OAAO,EAAE,OAAO,EAAE,SAAS,EAAE,MAAM,MAAM,CAAC;AAC1C,OAAO,EAAE,qBAAqB,EAAE,MAAM,wBAAwB,CAAC;AAC/D,OAAO,EAAwC,eAAe,EAAE,MAAM,gBAAgB,CAAC;;AAoBvF,MAAM,OAAO,sBAAsB;IAlBnC;QAuBY,aAAQ,GAAG,IAAI,OAAO,EAAQ,CAAC;QAKC,aAAQ,GAAG,KAAK,CAAC;QAMzD;;;;;;WAMG;QACM,iBAAY,GAAG,UAAU,CAAC;QAEnC;;WAEG;QACgB,kBAAa,GAAG,IAAI,YAAY,EAAU,CAAC;QAE9D;;WAEG;QACK,aAAQ,GAA4B,GAAG,EAAE;YAC7C,WAAW;QACf,CAAC,CAAC;QAEF;;WAEG;QACH,cAAS,GAAe,GAAG,EAAE;YACzB,WAAW;QACf,CAAC,CAAC;KAgIL;IA9HG,kBAAkB;QACd,IAAI,CAAC,eAAe,GAAG,IAAI,eAAe,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,QAAQ,EAAE,CAAC,uBAAuB,EAAE,CAAC;QAEjG,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,SAAS,CAAC,GAAG,EAAE;YAClE,IAAI,CAAC,gBAAgB,EAAE,CAAC;QAC5B,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC;IACjC,CAAC;IAED,WAAW;QACP,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;QACrB,IAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC;IAC7B,CAAC;IAED;;;OAGG;IACH,MAAM,CAAC,KAAa;QAChB,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;QACnB,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC/B,IAAI,CAAC,QAAQ,EAAE,CAAC,KAAK,CAAC,CAAC;QACvB,IAAI,CAAC,gBAAgB,EAAE,CAAC;QACxB,IAAI,CAAC,SAAS,EAAE,CAAC;IACrB,CAAC;IAED;;;;OAIG;IACH,UAAU,CAAC,KAAa;QACpB,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;QACnB,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;YAClB,IAAI,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC;QACjC,CAAC;IACL,CAAC;IAED;;;;OAIG;IACH,gBAAgB,CAAC,EAA2B;QACxC,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;IACvB,CAAC;IAED,iBAAiB,CAAC,EAAc;QAC5B,IAAI,CAAC,SAAS,GAAG,EAAE,CAAC;IACxB,CAAC;IAED;;;;OAIG;IACH,gBAAgB,CAAC,UAAmB;QAChC,IAAI,CAAC,QAAQ,GAAG,UAAU,CAAC;IAC/B,CAAC;IAED;;OAEG;IACO,SAAS,CAAC,KAAiB;QACjC,MAAM,MAAM,GAAG,KAAK,CAAC,MAAqB,CAAC;QAC3C,MAAM,SAAS,GAAG,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,aAAa,KAAK,MAAM,CAAC,CAAC;QACxF,IAAI,SAAS,EAAE,CAAC;YACZ,IAAI,CAAC,eAAe,CAAC,aAAa,CAAC,SAAS,CAAC,CAAC;QAClD,CAAC;IACL,CAAC;IAES,SAAS,CAAC,KAAoB;QACpC,IAAI,IAAI,CAAC,QAAQ;YAAE,OAAO;QAE1B,QAAQ,KAAK,CAAC,OAAO,EAAE,CAAC;YACpB,KAAK,KAAK,CAAC;YACX,KAAK,KAAK;gBACN,KAAK,CAAC,cAAc,EAAE,CAAC;gBACvB,IAAI,CAAC,iBAAiB,EAAE,CAAC;gBACzB,MAAM;YACV,KAAK,UAAU,CAAC;YAChB,KAAK,WAAW;gBACZ,KAAK,CAAC,cAAc,EAAE,CAAC;gBACvB,IAAI,CAAC,eAAe,CAAC,iBAAiB,EAAE,CAAC;gBACzC,IAAI,CAAC,iBAAiB,EAAE,CAAC;gBACzB,MAAM;YACV,KAAK,QAAQ,CAAC;YACd,KAAK,UAAU;gBACX,KAAK,CAAC,cAAc,EAAE,CAAC;gBACvB,IAAI,CAAC,eAAe,CAAC,qBAAqB,EAAE,CAAC;gBAC7C,IAAI,CAAC,iBAAiB,EAAE,CAAC;gBACzB,MAAM;YACV,KAAK,GAAG;gBACJ,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC;gBAC1B,MAAM;YACV;gBACI,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;QAC9C,CAAC;IACL,CAAC;IAEO,iBAAiB;QACrB,MAAM,WAAW,GAAG,IAAI,CAAC,eAAe,CAAC,UAAU,CAAC;QACpD,IAAI,WAAW,EAAE,CAAC;YACd,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;QACnC,CAAC;IACL,CAAC;IAEO,gBAAgB,CAAC,QAAQ,GAAG,IAAI;QACpC,MAAM,UAAU,GAAG,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,KAAK,KAAK,IAAI,CAAC,KAAK,CAAC,CAAC;QAC7E,IAAI,UAAU,EAAE,CAAC;YACb,IAAI,CAAC,eAAe,CAAC,aAAa,CAAC,UAAU,CAAC,CAAC;QACnD,CAAC;aAAM,IAAI,IAAI,CAAC,UAAU,CAAC,MAAM,GAAG,CAAC,IAAI,QAAQ,EAAE,CAAC;YAChD,IAAI,CAAC,eAAe,CAAC,kBAAkB,EAAE,CAAC;QAC9C,CAAC;IACL,CAAC;IAEO,aAAa,CAAC,KAAoB;QACtC,KAAK,CAAC,cAAc,EAAE,CAAC;QACvB,MAAM,WAAW,GAAG,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QACjE,IAAI,WAAW,EAAE,CAAC;YACd,WAAW,CAAC,KAAK,EAAE,CAAC;QACxB,CAAC;aAAM,IAAI,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE,CAAC;YAC/B,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,KAAK,EAAE,CAAC;QAClC,CAAC;IACL,CAAC;+GAzKQ,sBAAsB;mGAAtB,sBAAsB,+GAUX,gBAAgB,yXAxBzB;YACP,EAAE,OAAO,EAAE,eAAe,EAAE,WAAW,EAAE,sBAAsB,EAAE;YACjE,EAAE,OAAO,EAAE,iBAAiB,EAAE,WAAW,EAAE,sBAAsB,EAAE,KAAK,EAAE,IAAI,EAAE;SACnF,qDAcgB,qBAAqB;;4FAH7B,sBAAsB;kBAlBlC,SAAS;mBAAC;oBACP,QAAQ,EAAE,gBAAgB;oBAC1B,QAAQ,EAAE,cAAc;oBACxB,UAAU,EAAE,IAAI;oBAChB,SAAS,EAAE;wBACP,EAAE,OAAO,EAAE,eAAe,EAAE,WAAW,wBAAwB,EAAE;wBACjE,EAAE,OAAO,EAAE,iBAAiB,EAAE,WAAW,wBAAwB,EAAE,KAAK,EAAE,IAAI,EAAE;qBACnF;oBACD,IAAI,EAAE;wBACF,IAAI,EAAE,YAAY;wBAClB,yBAAyB,EAAE,cAAc;wBACzC,sBAAsB,EAAE,sBAAsB;wBAC9C,iBAAiB,EAAE,IAAI;wBACvB,YAAY,EAAE,KAAK;wBACnB,WAAW,EAAE,mBAAmB;wBAChC,WAAW,EAAE,mBAAmB;qBACnC;iBACJ;8BAIkE,UAAU;sBAAxE,eAAe;uBAAC,qBAAqB,EAAE,EAAE,WAAW,EAAE,IAAI,EAAE;gBAKpD,KAAK;sBAAb,KAAK;gBAEkC,QAAQ;sBAA/C,KAAK;uBAAC,EAAE,SAAS,EAAE,gBAAgB,EAAE;gBAE7B,GAAG;sBAAX,KAAK;gBAEG,YAAY;sBAApB,KAAK;gBAca,aAAa;sBAA/B,MAAM","sourcesContent":["import { FocusKeyManager } from '@angular/cdk/a11y';\nimport { DOWN_ARROW, ENTER, LEFT_ARROW, RIGHT_ARROW, SPACE, TAB, UP_ARROW } from '@angular/cdk/keycodes';\nimport {\n    AfterContentInit,\n    booleanAttribute,\n    ContentChildren,\n    Directive,\n    EventEmitter,\n    Input,\n    OnDestroy,\n    Output,\n    QueryList\n} from '@angular/core';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\nimport { Subject, takeUntil } from 'rxjs';\nimport { RdxRadioItemDirective } from './radio-item.directive';\nimport { RadioGroupDirective, RadioGroupProps, RDX_RADIO_GROUP } from './radio-tokens';\n\n@Directive({\n    selector: '[rdxRadioRoot]',\n    exportAs: 'rdxRadioRoot',\n    standalone: true,\n    providers: [\n        { provide: RDX_RADIO_GROUP, useExisting: RdxRadioGroupDirective },\n        { provide: NG_VALUE_ACCESSOR, useExisting: RdxRadioGroupDirective, multi: true }\n    ],\n    host: {\n        role: 'radiogroup',\n        '[attr.aria-orientation]': '_orientation',\n        '[attr.data-disabled]': 'disabled ? \"\" : null',\n        '[attr.tabindex]': '-1',\n        '[attr.dir]': 'dir',\n        '(keydown)': 'onKeydown($event)',\n        '(focusin)': 'onFocusin($event)'\n    }\n})\nexport class RdxRadioGroupDirective\n    implements RadioGroupProps, RadioGroupDirective, ControlValueAccessor, AfterContentInit, OnDestroy\n{\n    @ContentChildren(RdxRadioItemDirective, { descendants: true }) radioItems!: QueryList<RdxRadioItemDirective>;\n    private focusKeyManager!: FocusKeyManager<RdxRadioItemDirective>;\n    private destroy$ = new Subject<void>();\n\n    name?: string | undefined;\n    @Input() value?: string;\n\n    @Input({ transform: booleanAttribute }) disabled = false;\n\n    @Input() dir?: string;\n\n    @Input() defaultValue?: string;\n\n    /**\n     * The orientation of the radio group only vertical.\n     * Horizontal radio buttons can sometimes be challenging to scan and localize.\n     * The horizontal arrangement of radio buttons may also lead to difficulties in determining which\n     * label corresponds to which button: whether the label is above or below the button.\n     * @default 'vertical'\n     */\n    readonly _orientation = 'vertical';\n\n    /**\n     * Event handler called when the value changes.\n     */\n    @Output() readonly onValueChange = new EventEmitter<string>();\n\n    /**\n     * The callback function to call when the value of the radio group changes.\n     */\n    private onChange: (value: string) => void = () => {\n        /* Empty */\n    };\n\n    /**\n     * The callback function to call when the radio group is touched.\n     */\n    onTouched: () => void = () => {\n        /* Empty */\n    };\n\n    ngAfterContentInit() {\n        this.focusKeyManager = new FocusKeyManager(this.radioItems).withWrap().withVerticalOrientation();\n\n        this.radioItems.changes.pipe(takeUntil(this.destroy$)).subscribe(() => {\n            this.updateActiveItem();\n        });\n\n        this.updateActiveItem(false);\n    }\n\n    ngOnDestroy() {\n        this.destroy$.next();\n        this.destroy$.complete();\n    }\n\n    /**\n     * Select a radio item.\n     * @param value The value of the radio item to select.\n     */\n    select(value: string): void {\n        this.value = value;\n        this.onValueChange.emit(value);\n        this.onChange?.(value);\n        this.updateActiveItem();\n        this.onTouched();\n    }\n\n    /**\n     * Update the value of the radio group.\n     * @param value The new value of the radio group.\n     * @internal\n     */\n    writeValue(value: string): void {\n        this.value = value;\n        if (this.radioItems) {\n            this.updateActiveItem(false);\n        }\n    }\n\n    /**\n     * Register a callback function to call when the value of the radio group changes.\n     * @param fn The callback function to call when the value of the radio group changes.\n     * @internal\n     */\n    registerOnChange(fn: (value: string) => void): void {\n        this.onChange = fn;\n    }\n\n    registerOnTouched(fn: () => void): void {\n        this.onTouched = fn;\n    }\n\n    /**\n     * Set the disabled state of the radio group.\n     * @param isDisabled Whether the radio group is disabled.\n     * @internal\n     */\n    setDisabledState(isDisabled: boolean): void {\n        this.disabled = isDisabled;\n    }\n\n    /**\n     * When focus leaves the radio group.\n     */\n    protected onFocusin(event: FocusEvent): void {\n        const target = event.target as HTMLElement;\n        const radioItem = this.radioItems.find((item) => item.element.nativeElement === target);\n        if (radioItem) {\n            this.focusKeyManager.setActiveItem(radioItem);\n        }\n    }\n\n    protected onKeydown(event: KeyboardEvent): void {\n        if (this.disabled) return;\n\n        switch (event.keyCode) {\n            case ENTER:\n            case SPACE:\n                event.preventDefault();\n                this.selectFocusedItem();\n                break;\n            case DOWN_ARROW:\n            case RIGHT_ARROW:\n                event.preventDefault();\n                this.focusKeyManager.setNextItemActive();\n                this.selectFocusedItem();\n                break;\n            case UP_ARROW:\n            case LEFT_ARROW:\n                event.preventDefault();\n                this.focusKeyManager.setPreviousItemActive();\n                this.selectFocusedItem();\n                break;\n            case TAB:\n                this.tabNavigation(event);\n                break;\n            default:\n                this.focusKeyManager.onKeydown(event);\n        }\n    }\n\n    private selectFocusedItem(): void {\n        const focusedItem = this.focusKeyManager.activeItem;\n        if (focusedItem) {\n            this.select(focusedItem.value);\n        }\n    }\n\n    private updateActiveItem(setFocus = true): void {\n        const activeItem = this.radioItems.find((item) => item.value === this.value);\n        if (activeItem) {\n            this.focusKeyManager.setActiveItem(activeItem);\n        } else if (this.radioItems.length > 0 && setFocus) {\n            this.focusKeyManager.setFirstItemActive();\n        }\n    }\n\n    private tabNavigation(event: KeyboardEvent): void {\n        event.preventDefault();\n        const checkedItem = this.radioItems.find((item) => item.checked);\n        if (checkedItem) {\n            checkedItem.focus();\n        } else if (this.radioItems.first) {\n            this.radioItems.first.focus();\n        }\n    }\n}\n"]}
|
104
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmFkaW8tcm9vdC5kaXJlY3RpdmUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9wcmltaXRpdmVzL3JhZGlvL3NyYy9yYWRpby1yb290LmRpcmVjdGl2ZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFDQSxPQUFPLEVBQUUsZ0JBQWdCLEVBQUUsUUFBUSxFQUFFLFNBQVMsRUFBRSxLQUFLLEVBQUUsS0FBSyxFQUFFLEtBQUssRUFBRSxNQUFNLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQzNHLE9BQU8sRUFBd0IsaUJBQWlCLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUN6RSxPQUFPLEVBQWUsNEJBQTRCLEVBQUUsTUFBTSxtQ0FBbUMsQ0FBQztBQUM5RixPQUFPLEVBQXdDLGVBQWUsRUFBRSxNQUFNLGdCQUFnQixDQUFDOzs7QUFtQnZGLE1BQU0sT0FBTyxzQkFBc0I7SUFqQm5DO1FBa0JhLFVBQUssR0FBRyxLQUFLLENBQWdCLElBQUksQ0FBQyxDQUFDO1FBRW5DLGFBQVEsR0FBRyxLQUFLLENBQXdCLEtBQUssRUFBRSxFQUFFLFNBQVMsRUFBRSxnQkFBZ0IsRUFBRSxDQUFDLENBQUM7UUFJaEYsYUFBUSxHQUFHLEtBQUssQ0FBd0IsS0FBSyxFQUFFLEVBQUUsU0FBUyxFQUFFLGdCQUFnQixFQUFFLENBQUMsQ0FBQztRQUVoRixnQkFBVyxHQUFHLEtBQUssRUFBZSxDQUFDO1FBRTVDOztXQUVHO1FBQ00sa0JBQWEsR0FBRyxNQUFNLEVBQVUsQ0FBQztRQUV6QixZQUFPLEdBQUcsTUFBTSxDQUFVLElBQUksQ0FBQyxRQUFRLEVBQUUsQ0FBQyxDQUFDO1FBQ25ELGlCQUFZLEdBQUcsUUFBUSxDQUFDLEdBQUcsRUFBRSxDQUFDLElBQUksQ0FBQyxPQUFPLEVBQUUsSUFBSSxJQUFJLENBQUMsUUFBUSxFQUFFLENBQUMsQ0FBQztRQUUxRTs7V0FFRztRQUNLLGFBQVEsR0FBNEIsR0FBRyxFQUFFO1lBQzdDLFdBQVc7UUFDZixDQUFDLENBQUM7UUFFRjs7O1dBR0c7UUFDSCxjQUFTLEdBQWUsR0FBRyxFQUFFO1lBQ3pCLFdBQVc7UUFDZixDQUFDLENBQUM7S0FpREw7SUEvQ0c7Ozs7T0FJRztJQUNILE1BQU0sQ0FBQyxLQUFhO1FBQ2hCLElBQUksQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQ3RCLElBQUksQ0FBQyxhQUFhLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQy9CLElBQUksQ0FBQyxRQUFRLEVBQUUsQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUN2QixJQUFJLENBQUMsU0FBUyxFQUFFLENBQUM7SUFDckIsQ0FBQztJQUVEOzs7O09BSUc7SUFDSCxVQUFVLENBQUMsS0FBYTtRQUNwQixJQUFJLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxLQUFLLENBQUMsQ0FBQztJQUMxQixDQUFDO0lBRUQ7Ozs7T0FJRztJQUNILGdCQUFnQixDQUFDLEVBQTJCO1FBQ3hDLElBQUksQ0FBQyxRQUFRLEdBQUcsRUFBRSxDQUFDO0lBQ3ZCLENBQUM7SUFFRCxjQUFjO0lBQ2QsaUJBQWlCLENBQUMsRUFBYztRQUM1QixJQUFJLENBQUMsU0FBUyxHQUFHLEVBQUUsQ0FBQztJQUN4QixDQUFDO0lBRUQ7Ozs7T0FJRztJQUNILGdCQUFnQixDQUFDLFVBQW1CO1FBQ2hDLElBQUksQ0FBQyxPQUFPLENBQUMsR0FBRyxDQUFDLFVBQVUsQ0FBQyxDQUFDO0lBQ2pDLENBQUM7SUFFUyxTQUFTO1FBQ2YsSUFBSSxJQUFJLENBQUMsWUFBWSxFQUFFO1lBQUUsT0FBTztJQUNwQyxDQUFDOytHQWhGUSxzQkFBc0I7bUdBQXRCLHNCQUFzQiwyZ0NBYnBCO1lBQ1AsRUFBRSxPQUFPLEVBQUUsZUFBZSxFQUFFLFdBQVcsRUFBRSxzQkFBc0IsRUFBRTtZQUNqRSxFQUFFLE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxXQUFXLEVBQUUsc0JBQXNCLEVBQUUsS0FBSyxFQUFFLElBQUksRUFBRTtTQUNuRjs7NEZBVVEsc0JBQXNCO2tCQWpCbEMsU0FBUzttQkFBQztvQkFDUCxRQUFRLEVBQUUsZ0JBQWdCO29CQUMxQixRQUFRLEVBQUUsY0FBYztvQkFDeEIsVUFBVSxFQUFFLElBQUk7b0JBQ2hCLFNBQVMsRUFBRTt3QkFDUCxFQUFFLE9BQU8sRUFBRSxlQUFlLEVBQUUsV0FBVyx3QkFBd0IsRUFBRTt3QkFDakUsRUFBRSxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsV0FBVyx3QkFBd0IsRUFBRSxLQUFLLEVBQUUsSUFBSSxFQUFFO3FCQUNuRjtvQkFDRCxjQUFjLEVBQUUsQ0FBQyxFQUFFLFNBQVMsRUFBRSw0QkFBNEIsRUFBRSxNQUFNLEVBQUUsQ0FBQyxLQUFLLEVBQUUsYUFBYSxFQUFFLE1BQU0sQ0FBQyxFQUFFLENBQUM7b0JBQ3JHLElBQUksRUFBRTt3QkFDRixJQUFJLEVBQUUsWUFBWTt3QkFDbEIseUJBQXlCLEVBQUUsZUFBZTt3QkFDMUMsc0JBQXNCLEVBQUUsWUFBWTt3QkFDcEMsc0JBQXNCLEVBQUUsNEJBQTRCO3dCQUNwRCxXQUFXLEVBQUUsYUFBYTtxQkFDN0I7aUJBQ0o7OEJBTVksWUFBWTtzQkFBcEIsS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEJvb2xlYW5JbnB1dCB9IGZyb20gJ0Bhbmd1bGFyL2Nkay9jb2VyY2lvbic7XG5pbXBvcnQgeyBib29sZWFuQXR0cmlidXRlLCBjb21wdXRlZCwgRGlyZWN0aXZlLCBpbnB1dCwgSW5wdXQsIG1vZGVsLCBvdXRwdXQsIHNpZ25hbCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ29udHJvbFZhbHVlQWNjZXNzb3IsIE5HX1ZBTFVFX0FDQ0VTU09SIH0gZnJvbSAnQGFuZ3VsYXIvZm9ybXMnO1xuaW1wb3J0IHsgT3JpZW50YXRpb24sIFJkeFJvdmluZ0ZvY3VzR3JvdXBEaXJlY3RpdmUgfSBmcm9tICdAcmFkaXgtbmcvcHJpbWl0aXZlcy9yb3ZpbmctZm9jdXMnO1xuaW1wb3J0IHsgUmFkaW9Hcm91cERpcmVjdGl2ZSwgUmFkaW9Hcm91cFByb3BzLCBSRFhfUkFESU9fR1JPVVAgfSBmcm9tICcuL3JhZGlvLXRva2Vucyc7XG5cbkBEaXJlY3RpdmUoe1xuICAgIHNlbGVjdG9yOiAnW3JkeFJhZGlvUm9vdF0nLFxuICAgIGV4cG9ydEFzOiAncmR4UmFkaW9Sb290JyxcbiAgICBzdGFuZGFsb25lOiB0cnVlLFxuICAgIHByb3ZpZGVyczogW1xuICAgICAgICB7IHByb3ZpZGU6IFJEWF9SQURJT19HUk9VUCwgdXNlRXhpc3Rpbmc6IFJkeFJhZGlvR3JvdXBEaXJlY3RpdmUgfSxcbiAgICAgICAgeyBwcm92aWRlOiBOR19WQUxVRV9BQ0NFU1NPUiwgdXNlRXhpc3Rpbmc6IFJkeFJhZGlvR3JvdXBEaXJlY3RpdmUsIG11bHRpOiB0cnVlIH1cbiAgICBdLFxuICAgIGhvc3REaXJlY3RpdmVzOiBbeyBkaXJlY3RpdmU6IFJkeFJvdmluZ0ZvY3VzR3JvdXBEaXJlY3RpdmUsIGlucHV0czogWydkaXInLCAnb3JpZW50YXRpb24nLCAnbG9vcCddIH1dLFxuICAgIGhvc3Q6IHtcbiAgICAgICAgcm9sZTogJ3JhZGlvZ3JvdXAnLFxuICAgICAgICAnW2F0dHIuYXJpYS1vcmllbnRhdGlvbl0nOiAnb3JpZW50YXRpb24oKScsXG4gICAgICAgICdbYXR0ci5hcmlhLXJlcXVpcmVkXSc6ICdyZXF1aXJlZCgpJyxcbiAgICAgICAgJ1thdHRyLmRhdGEtZGlzYWJsZWRdJzogJ2Rpc2FibGVTdGF0ZSgpID8gXCJcIiA6IG51bGwnLFxuICAgICAgICAnKGtleWRvd24pJzogJ29uS2V5ZG93bigpJ1xuICAgIH1cbn0pXG5leHBvcnQgY2xhc3MgUmR4UmFkaW9Hcm91cERpcmVjdGl2ZSBpbXBsZW1lbnRzIFJhZGlvR3JvdXBQcm9wcywgUmFkaW9Hcm91cERpcmVjdGl2ZSwgQ29udHJvbFZhbHVlQWNjZXNzb3Ige1xuICAgIHJlYWRvbmx5IHZhbHVlID0gbW9kZWw8c3RyaW5nIHwgbnVsbD4obnVsbCk7XG5cbiAgICByZWFkb25seSBkaXNhYmxlZCA9IGlucHV0PGJvb2xlYW4sIEJvb2xlYW5JbnB1dD4oZmFsc2UsIHsgdHJhbnNmb3JtOiBib29sZWFuQXR0cmlidXRlIH0pO1xuXG4gICAgQElucHV0KCkgZGVmYXVsdFZhbHVlPzogc3RyaW5nO1xuXG4gICAgcmVhZG9ubHkgcmVxdWlyZWQgPSBpbnB1dDxib29sZWFuLCBCb29sZWFuSW5wdXQ+KGZhbHNlLCB7IHRyYW5zZm9ybTogYm9vbGVhbkF0dHJpYnV0ZSB9KTtcblxuICAgIHJlYWRvbmx5IG9yaWVudGF0aW9uID0gaW5wdXQ8T3JpZW50YXRpb24+KCk7XG5cbiAgICAvKipcbiAgICAgKiBFdmVudCBoYW5kbGVyIGNhbGxlZCB3aGVuIHRoZSB2YWx1ZSBjaGFuZ2VzLlxuICAgICAqL1xuICAgIHJlYWRvbmx5IG9uVmFsdWVDaGFuZ2UgPSBvdXRwdXQ8c3RyaW5nPigpO1xuXG4gICAgcHJpdmF0ZSByZWFkb25seSBkaXNhYmxlID0gc2lnbmFsPGJvb2xlYW4+KHRoaXMuZGlzYWJsZWQoKSk7XG4gICAgcmVhZG9ubHkgZGlzYWJsZVN0YXRlID0gY29tcHV0ZWQoKCkgPT4gdGhpcy5kaXNhYmxlKCkgfHwgdGhpcy5kaXNhYmxlZCgpKTtcblxuICAgIC8qKlxuICAgICAqIFRoZSBjYWxsYmFjayBmdW5jdGlvbiB0byBjYWxsIHdoZW4gdGhlIHZhbHVlIG9mIHRoZSByYWRpbyBncm91cCBjaGFuZ2VzLlxuICAgICAqL1xuICAgIHByaXZhdGUgb25DaGFuZ2U6ICh2YWx1ZTogc3RyaW5nKSA9PiB2b2lkID0gKCkgPT4ge1xuICAgICAgICAvKiBFbXB0eSAqL1xuICAgIH07XG5cbiAgICAvKipcbiAgICAgKiBUaGUgY2FsbGJhY2sgZnVuY3Rpb24gdG8gY2FsbCB3aGVuIHRoZSByYWRpbyBncm91cCBpcyB0b3VjaGVkLlxuICAgICAqIEBpZ25vcmVcbiAgICAgKi9cbiAgICBvblRvdWNoZWQ6ICgpID0+IHZvaWQgPSAoKSA9PiB7XG4gICAgICAgIC8qIEVtcHR5ICovXG4gICAgfTtcblxuICAgIC8qKlxuICAgICAqIFNlbGVjdCBhIHJhZGlvIGl0ZW0uXG4gICAgICogQHBhcmFtIHZhbHVlIFRoZSB2YWx1ZSBvZiB0aGUgcmFkaW8gaXRlbSB0byBzZWxlY3QuXG4gICAgICogQGlnbm9yZVxuICAgICAqL1xuICAgIHNlbGVjdCh2YWx1ZTogc3RyaW5nKTogdm9pZCB7XG4gICAgICAgIHRoaXMudmFsdWUuc2V0KHZhbHVlKTtcbiAgICAgICAgdGhpcy5vblZhbHVlQ2hhbmdlLmVtaXQodmFsdWUpO1xuICAgICAgICB0aGlzLm9uQ2hhbmdlPy4odmFsdWUpO1xuICAgICAgICB0aGlzLm9uVG91Y2hlZCgpO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIFVwZGF0ZSB0aGUgdmFsdWUgb2YgdGhlIHJhZGlvIGdyb3VwLlxuICAgICAqIEBwYXJhbSB2YWx1ZSBUaGUgbmV3IHZhbHVlIG9mIHRoZSByYWRpbyBncm91cC5cbiAgICAgKiBAaWdub3JlXG4gICAgICovXG4gICAgd3JpdGVWYWx1ZSh2YWx1ZTogc3RyaW5nKTogdm9pZCB7XG4gICAgICAgIHRoaXMudmFsdWUuc2V0KHZhbHVlKTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBSZWdpc3RlciBhIGNhbGxiYWNrIGZ1bmN0aW9uIHRvIGNhbGwgd2hlbiB0aGUgdmFsdWUgb2YgdGhlIHJhZGlvIGdyb3VwIGNoYW5nZXMuXG4gICAgICogQHBhcmFtIGZuIFRoZSBjYWxsYmFjayBmdW5jdGlvbiB0byBjYWxsIHdoZW4gdGhlIHZhbHVlIG9mIHRoZSByYWRpbyBncm91cCBjaGFuZ2VzLlxuICAgICAqIEBpZ25vcmVcbiAgICAgKi9cbiAgICByZWdpc3Rlck9uQ2hhbmdlKGZuOiAodmFsdWU6IHN0cmluZykgPT4gdm9pZCk6IHZvaWQge1xuICAgICAgICB0aGlzLm9uQ2hhbmdlID0gZm47XG4gICAgfVxuXG4gICAgLyoqIEBpZ25vcmUgKi9cbiAgICByZWdpc3Rlck9uVG91Y2hlZChmbjogKCkgPT4gdm9pZCk6IHZvaWQge1xuICAgICAgICB0aGlzLm9uVG91Y2hlZCA9IGZuO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIFNldCB0aGUgZGlzYWJsZWQgc3RhdGUgb2YgdGhlIHJhZGlvIGdyb3VwLlxuICAgICAqIEBwYXJhbSBpc0Rpc2FibGVkIFdoZXRoZXIgdGhlIHJhZGlvIGdyb3VwIGlzIGRpc2FibGVkLlxuICAgICAqIEBpZ25vcmVcbiAgICAgKi9cbiAgICBzZXREaXNhYmxlZFN0YXRlKGlzRGlzYWJsZWQ6IGJvb2xlYW4pOiB2b2lkIHtcbiAgICAgICAgdGhpcy5kaXNhYmxlLnNldChpc0Rpc2FibGVkKTtcbiAgICB9XG5cbiAgICBwcm90ZWN0ZWQgb25LZXlkb3duKCk6IHZvaWQge1xuICAgICAgICBpZiAodGhpcy5kaXNhYmxlU3RhdGUoKSkgcmV0dXJuO1xuICAgIH1cbn1cbiJdfQ==
|
@@ -1,3 +1,3 @@
|
|
1
1
|
import { InjectionToken } from '@angular/core';
|
2
2
|
export const RDX_RADIO_GROUP = new InjectionToken('RdxRadioGroup');
|
3
|
-
//# sourceMappingURL=data:application/json;base64,
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmFkaW8tdG9rZW5zLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvcHJpbWl0aXZlcy9yYWRpby9zcmMvcmFkaW8tdG9rZW5zLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUNBLE9BQU8sRUFBRSxjQUFjLEVBQWlELE1BQU0sZUFBZSxDQUFDO0FBZ0I5RixNQUFNLENBQUMsTUFBTSxlQUFlLEdBQUcsSUFBSSxjQUFjLENBQXNCLGVBQWUsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQm9vbGVhbklucHV0IH0gZnJvbSAnQGFuZ3VsYXIvY2RrL2NvZXJjaW9uJztcbmltcG9ydCB7IEluamVjdGlvblRva2VuLCBJbnB1dFNpZ25hbFdpdGhUcmFuc2Zvcm0sIE1vZGVsU2lnbmFsLCBTaWduYWwgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcblxuZXhwb3J0IGludGVyZmFjZSBSYWRpb0dyb3VwUHJvcHMge1xuICAgIG5hbWU/OiBzdHJpbmc7XG4gICAgZGlzYWJsZWQ/OiBJbnB1dFNpZ25hbFdpdGhUcmFuc2Zvcm08Ym9vbGVhbiwgQm9vbGVhbklucHV0PjtcbiAgICBkZWZhdWx0VmFsdWU/OiBzdHJpbmc7XG4gICAgdmFsdWU6IE1vZGVsU2lnbmFsPHN0cmluZyB8IG51bGw+O1xuICAgIGRpc2FibGVTdGF0ZTogU2lnbmFsPGJvb2xlYW4+O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFJhZGlvR3JvdXBEaXJlY3RpdmUgZXh0ZW5kcyBSYWRpb0dyb3VwUHJvcHMge1xuICAgIHNlbGVjdCh2YWx1ZTogc3RyaW5nIHwgbnVsbCk6IHZvaWQ7XG5cbiAgICBvblRvdWNoZWQoKTogdm9pZDtcbn1cblxuZXhwb3J0IGNvbnN0IFJEWF9SQURJT19HUk9VUCA9IG5ldyBJbmplY3Rpb25Ub2tlbjxSYWRpb0dyb3VwRGlyZWN0aXZlPignUmR4UmFkaW9Hcm91cCcpO1xuIl19
|
@@ -0,0 +1,3 @@
|
|
1
|
+
export * from './src/roving-focus-group.directive';
|
2
|
+
export * from './src/roving-focus-item.directive';
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wYWNrYWdlcy9wcmltaXRpdmVzL3JvdmluZy1mb2N1cy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLG9DQUFvQyxDQUFDO0FBQ25ELGNBQWMsbUNBQW1DLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL3NyYy9yb3ZpbmctZm9jdXMtZ3JvdXAuZGlyZWN0aXZlJztcbmV4cG9ydCAqIGZyb20gJy4vc3JjL3JvdmluZy1mb2N1cy1pdGVtLmRpcmVjdGl2ZSc7XG5cbmV4cG9ydCB0eXBlIHsgRGlyZWN0aW9uLCBPcmllbnRhdGlvbiB9IGZyb20gJy4vc3JjL3V0aWxzJztcbiJdfQ==
|
@@ -0,0 +1,5 @@
|
|
1
|
+
/**
|
2
|
+
* Generated bundle index. Do not edit.
|
3
|
+
*/
|
4
|
+
export * from './index';
|
5
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmFkaXgtbmctcHJpbWl0aXZlcy1yb3ZpbmctZm9jdXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wYWNrYWdlcy9wcmltaXRpdmVzL3JvdmluZy1mb2N1cy9yYWRpeC1uZy1wcmltaXRpdmVzLXJvdmluZy1mb2N1cy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUVILGNBQWMsU0FBUyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBHZW5lcmF0ZWQgYnVuZGxlIGluZGV4LiBEbyBub3QgZWRpdC5cbiAqL1xuXG5leHBvcnQgKiBmcm9tICcuL2luZGV4JztcbiJdfQ==
|
@@ -0,0 +1,138 @@
|
|
1
|
+
import { booleanAttribute, Directive, ElementRef, EventEmitter, inject, Input, NgZone, Output, signal } from '@angular/core';
|
2
|
+
import { ENTRY_FOCUS, EVENT_OPTIONS, focusFirst } from './utils';
|
3
|
+
import * as i0 from "@angular/core";
|
4
|
+
export class RdxRovingFocusGroupDirective {
|
5
|
+
constructor() {
|
6
|
+
this.ngZone = inject(NgZone);
|
7
|
+
this.elementRef = inject(ElementRef);
|
8
|
+
this.dir = 'ltr';
|
9
|
+
this.loop = true;
|
10
|
+
this.preventScrollOnEntryFocus = false;
|
11
|
+
this.entryFocus = new EventEmitter();
|
12
|
+
this.currentTabStopIdChange = new EventEmitter();
|
13
|
+
/** @ignore */
|
14
|
+
this.currentTabStopId = signal(null);
|
15
|
+
/** @ignore */
|
16
|
+
this.focusableItems = signal([]);
|
17
|
+
this.isClickFocus = signal(false);
|
18
|
+
this.isTabbingBackOut = signal(false);
|
19
|
+
this.focusableItemsCount = signal(0);
|
20
|
+
}
|
21
|
+
/** @ignore */
|
22
|
+
get dataOrientation() {
|
23
|
+
return this.orientation || 'horizontal';
|
24
|
+
}
|
25
|
+
/** @ignore */
|
26
|
+
get tabIndex() {
|
27
|
+
return this.isTabbingBackOut() || this.getFocusableItemsCount() === 0 ? -1 : 0;
|
28
|
+
}
|
29
|
+
/** @ignore */
|
30
|
+
handleBlur() {
|
31
|
+
this.isTabbingBackOut.set(false);
|
32
|
+
}
|
33
|
+
/** @ignore */
|
34
|
+
handleMouseUp() {
|
35
|
+
// reset `isClickFocus` after 1 tick because handleFocus might not triggered due to focused element
|
36
|
+
this.ngZone.runOutsideAngular(() => {
|
37
|
+
// eslint-disable-next-line promise/catch-or-return,promise/always-return
|
38
|
+
Promise.resolve().then(() => {
|
39
|
+
this.ngZone.run(() => {
|
40
|
+
this.isClickFocus.set(false);
|
41
|
+
});
|
42
|
+
});
|
43
|
+
});
|
44
|
+
}
|
45
|
+
/** @ignore */
|
46
|
+
handleFocus(event) {
|
47
|
+
// We normally wouldn't need this check, because we already check
|
48
|
+
// that the focus is on the current target and not bubbling to it.
|
49
|
+
// We do this because Safari doesn't focus buttons when clicked, and
|
50
|
+
// instead, the wrapper will get focused and not through a bubbling event.
|
51
|
+
const isKeyboardFocus = !this.isClickFocus();
|
52
|
+
if (event.currentTarget === this.elementRef.nativeElement &&
|
53
|
+
event.target === event.currentTarget &&
|
54
|
+
isKeyboardFocus &&
|
55
|
+
!this.isTabbingBackOut()) {
|
56
|
+
const entryFocusEvent = new CustomEvent(ENTRY_FOCUS, EVENT_OPTIONS);
|
57
|
+
this.elementRef.nativeElement.dispatchEvent(entryFocusEvent);
|
58
|
+
this.entryFocus.emit(entryFocusEvent);
|
59
|
+
if (!entryFocusEvent.defaultPrevented) {
|
60
|
+
const items = this.focusableItems().filter((item) => item.dataset['disabled'] !== '');
|
61
|
+
const activeItem = items.find((item) => item.getAttribute('data-active') === 'true');
|
62
|
+
const currentItem = items.find((item) => item.id === this.currentTabStopId());
|
63
|
+
const candidateItems = [activeItem, currentItem, ...items].filter(Boolean);
|
64
|
+
focusFirst(candidateItems, this.preventScrollOnEntryFocus);
|
65
|
+
}
|
66
|
+
}
|
67
|
+
this.isClickFocus.set(false);
|
68
|
+
}
|
69
|
+
/** @ignore */
|
70
|
+
handleMouseDown() {
|
71
|
+
this.isClickFocus.set(true);
|
72
|
+
}
|
73
|
+
/** @ignore */
|
74
|
+
onItemFocus(tabStopId) {
|
75
|
+
this.currentTabStopId.set(tabStopId);
|
76
|
+
this.currentTabStopIdChange.emit(tabStopId);
|
77
|
+
}
|
78
|
+
/** @ignore */
|
79
|
+
onItemShiftTab() {
|
80
|
+
this.isTabbingBackOut.set(true);
|
81
|
+
}
|
82
|
+
/** @ignore */
|
83
|
+
onFocusableItemAdd() {
|
84
|
+
this.focusableItemsCount.update((count) => count + 1);
|
85
|
+
}
|
86
|
+
/** @ignore */
|
87
|
+
onFocusableItemRemove() {
|
88
|
+
this.focusableItemsCount.update((count) => Math.max(0, count - 1));
|
89
|
+
}
|
90
|
+
/** @ignore */
|
91
|
+
registerItem(item) {
|
92
|
+
const currentItems = this.focusableItems();
|
93
|
+
this.focusableItems.set([...currentItems, item]);
|
94
|
+
}
|
95
|
+
/** @ignore */
|
96
|
+
unregisterItem(item) {
|
97
|
+
const currentItems = this.focusableItems();
|
98
|
+
this.focusableItems.set(currentItems.filter((el) => el !== item));
|
99
|
+
}
|
100
|
+
/** @ignore */
|
101
|
+
getFocusableItemsCount() {
|
102
|
+
return this.focusableItemsCount();
|
103
|
+
}
|
104
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.11", ngImport: i0, type: RdxRovingFocusGroupDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
|
105
|
+
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "16.1.0", version: "18.2.11", type: RdxRovingFocusGroupDirective, isStandalone: true, selector: "[rdxRovingFocusGroup]", inputs: { orientation: "orientation", dir: "dir", loop: ["loop", "loop", booleanAttribute], preventScrollOnEntryFocus: ["preventScrollOnEntryFocus", "preventScrollOnEntryFocus", booleanAttribute] }, outputs: { entryFocus: "entryFocus", currentTabStopIdChange: "currentTabStopIdChange" }, host: { listeners: { "focus": "handleFocus($event)", "blur": "handleBlur()", "mouseup": "handleMouseUp()", "mousedown": "handleMouseDown()" }, properties: { "attr.data-orientation": "dataOrientation", "attr.tabindex": "tabIndex", "attr.dir": "dir" }, styleAttribute: "outline: none;" }, ngImport: i0 }); }
|
106
|
+
}
|
107
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.11", ngImport: i0, type: RdxRovingFocusGroupDirective, decorators: [{
|
108
|
+
type: Directive,
|
109
|
+
args: [{
|
110
|
+
selector: '[rdxRovingFocusGroup]',
|
111
|
+
standalone: true,
|
112
|
+
host: {
|
113
|
+
'[attr.data-orientation]': 'dataOrientation',
|
114
|
+
'[attr.tabindex]': 'tabIndex',
|
115
|
+
'[attr.dir]': 'dir',
|
116
|
+
'(focus)': 'handleFocus($event)',
|
117
|
+
'(blur)': 'handleBlur()',
|
118
|
+
'(mouseup)': 'handleMouseUp()',
|
119
|
+
'(mousedown)': 'handleMouseDown()',
|
120
|
+
style: 'outline: none;'
|
121
|
+
}
|
122
|
+
}]
|
123
|
+
}], propDecorators: { orientation: [{
|
124
|
+
type: Input
|
125
|
+
}], dir: [{
|
126
|
+
type: Input
|
127
|
+
}], loop: [{
|
128
|
+
type: Input,
|
129
|
+
args: [{ transform: booleanAttribute }]
|
130
|
+
}], preventScrollOnEntryFocus: [{
|
131
|
+
type: Input,
|
132
|
+
args: [{ transform: booleanAttribute }]
|
133
|
+
}], entryFocus: [{
|
134
|
+
type: Output
|
135
|
+
}], currentTabStopIdChange: [{
|
136
|
+
type: Output
|
137
|
+
}] } });
|
138
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"roving-focus-group.directive.js","sourceRoot":"","sources":["../../../../../packages/primitives/roving-focus/src/roving-focus-group.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    booleanAttribute,\n    Directive,\n    ElementRef,\n    EventEmitter,\n    inject,\n    Input,\n    NgZone,\n    Output,\n    signal\n} from '@angular/core';\nimport { Direction, ENTRY_FOCUS, EVENT_OPTIONS, focusFirst, Orientation } from './utils';\n\n@Directive({\n    selector: '[rdxRovingFocusGroup]',\n    standalone: true,\n    host: {\n        '[attr.data-orientation]': 'dataOrientation',\n        '[attr.tabindex]': 'tabIndex',\n        '[attr.dir]': 'dir',\n        '(focus)': 'handleFocus($event)',\n        '(blur)': 'handleBlur()',\n        '(mouseup)': 'handleMouseUp()',\n        '(mousedown)': 'handleMouseDown()',\n        style: 'outline: none;'\n    }\n})\nexport class RdxRovingFocusGroupDirective {\n    private readonly ngZone = inject(NgZone);\n    private readonly elementRef = inject(ElementRef);\n\n    @Input() orientation: Orientation | undefined;\n    @Input() dir: Direction = 'ltr';\n    @Input({ transform: booleanAttribute }) loop: boolean = true;\n    @Input({ transform: booleanAttribute }) preventScrollOnEntryFocus: boolean = false;\n\n    @Output() entryFocus = new EventEmitter<Event>();\n    @Output() currentTabStopIdChange = new EventEmitter<string | null>();\n\n    /** @ignore */\n    readonly currentTabStopId = signal<string | null>(null);\n\n    /** @ignore */\n    readonly focusableItems = signal<HTMLElement[]>([]);\n\n    private readonly isClickFocus = signal(false);\n    private readonly isTabbingBackOut = signal(false);\n    private readonly focusableItemsCount = signal(0);\n\n    /** @ignore */\n    get dataOrientation() {\n        return this.orientation || 'horizontal';\n    }\n\n    /** @ignore */\n    get tabIndex() {\n        return this.isTabbingBackOut() || this.getFocusableItemsCount() === 0 ? -1 : 0;\n    }\n\n    /** @ignore */\n    handleBlur() {\n        this.isTabbingBackOut.set(false);\n    }\n\n    /** @ignore */\n    handleMouseUp() {\n        // reset `isClickFocus` after 1 tick because handleFocus might not triggered due to focused element\n        this.ngZone.runOutsideAngular(() => {\n            // eslint-disable-next-line promise/catch-or-return,promise/always-return\n            Promise.resolve().then(() => {\n                this.ngZone.run(() => {\n                    this.isClickFocus.set(false);\n                });\n            });\n        });\n    }\n\n    /** @ignore */\n    handleFocus(event: FocusEvent) {\n        // We normally wouldn't need this check, because we already check\n        // that the focus is on the current target and not bubbling to it.\n        // We do this because Safari doesn't focus buttons when clicked, and\n        // instead, the wrapper will get focused and not through a bubbling event.\n        const isKeyboardFocus = !this.isClickFocus();\n\n        if (\n            event.currentTarget === this.elementRef.nativeElement &&\n            event.target === event.currentTarget &&\n            isKeyboardFocus &&\n            !this.isTabbingBackOut()\n        ) {\n            const entryFocusEvent = new CustomEvent(ENTRY_FOCUS, EVENT_OPTIONS);\n            this.elementRef.nativeElement.dispatchEvent(entryFocusEvent);\n            this.entryFocus.emit(entryFocusEvent);\n\n            if (!entryFocusEvent.defaultPrevented) {\n                const items = this.focusableItems().filter((item) => item.dataset['disabled'] !== '');\n                const activeItem = items.find((item) => item.getAttribute('data-active') === 'true');\n                const currentItem = items.find((item) => item.id === this.currentTabStopId());\n                const candidateItems = [activeItem, currentItem, ...items].filter(Boolean) as HTMLElement[];\n\n                focusFirst(candidateItems, this.preventScrollOnEntryFocus);\n            }\n        }\n        this.isClickFocus.set(false);\n    }\n\n    /** @ignore */\n    handleMouseDown() {\n        this.isClickFocus.set(true);\n    }\n\n    /** @ignore */\n    onItemFocus(tabStopId: string) {\n        this.currentTabStopId.set(tabStopId);\n        this.currentTabStopIdChange.emit(tabStopId);\n    }\n\n    /** @ignore */\n    onItemShiftTab() {\n        this.isTabbingBackOut.set(true);\n    }\n\n    /** @ignore */\n    onFocusableItemAdd() {\n        this.focusableItemsCount.update((count) => count + 1);\n    }\n\n    /** @ignore */\n    onFocusableItemRemove() {\n        this.focusableItemsCount.update((count) => Math.max(0, count - 1));\n    }\n\n    /** @ignore */\n    registerItem(item: HTMLElement) {\n        const currentItems = this.focusableItems();\n        this.focusableItems.set([...currentItems, item]);\n    }\n\n    /** @ignore */\n    unregisterItem(item: HTMLElement) {\n        const currentItems = this.focusableItems();\n        this.focusableItems.set(currentItems.filter((el) => el !== item));\n    }\n\n    /** @ignore */\n    getFocusableItemsCount() {\n        return this.focusableItemsCount();\n    }\n}\n"]}
|
@@ -0,0 +1,133 @@
|
|
1
|
+
import { booleanAttribute, computed, Directive, ElementRef, inject, Input, NgZone } from '@angular/core';
|
2
|
+
import { RdxRovingFocusGroupDirective } from './roving-focus-group.directive';
|
3
|
+
import { focusFirst, generateId, getFocusIntent, wrapArray } from './utils';
|
4
|
+
import * as i0 from "@angular/core";
|
5
|
+
export class RdxRovingFocusItemDirective {
|
6
|
+
constructor() {
|
7
|
+
this.elementRef = inject(ElementRef);
|
8
|
+
this.ngZone = inject(NgZone);
|
9
|
+
this.parent = inject(RdxRovingFocusGroupDirective);
|
10
|
+
this.focusable = true;
|
11
|
+
this.active = true;
|
12
|
+
this.allowShiftKey = false;
|
13
|
+
this.id = computed(() => this.tabStopId || generateId());
|
14
|
+
/** @ignore */
|
15
|
+
this.isCurrentTabStop = computed(() => this.parent.currentTabStopId() === this.id());
|
16
|
+
}
|
17
|
+
/**
|
18
|
+
* Lifecycle hook triggered on initialization.
|
19
|
+
* Registers the element with the parent roving focus group if it is focusable.
|
20
|
+
* @ignore
|
21
|
+
*/
|
22
|
+
ngOnInit() {
|
23
|
+
if (this.focusable) {
|
24
|
+
this.parent.registerItem(this.elementRef.nativeElement);
|
25
|
+
this.parent.onFocusableItemAdd();
|
26
|
+
}
|
27
|
+
}
|
28
|
+
/**
|
29
|
+
* Lifecycle hook triggered on destruction.
|
30
|
+
* Unregisters the element from the parent roving focus group if it is focusable.
|
31
|
+
* @ignore
|
32
|
+
*/
|
33
|
+
ngOnDestroy() {
|
34
|
+
if (this.focusable) {
|
35
|
+
this.parent.unregisterItem(this.elementRef.nativeElement);
|
36
|
+
this.parent.onFocusableItemRemove();
|
37
|
+
}
|
38
|
+
}
|
39
|
+
/**
|
40
|
+
* Determines the `tabIndex` of the element.
|
41
|
+
* Returns `0` if the element is the current tab stop; otherwise, returns `-1`.
|
42
|
+
* @ignore
|
43
|
+
*/
|
44
|
+
get tabIndex() {
|
45
|
+
return this.isCurrentTabStop() ? 0 : -1;
|
46
|
+
}
|
47
|
+
/** @ignore */
|
48
|
+
handleMouseDown(event) {
|
49
|
+
if (!this.focusable) {
|
50
|
+
// We prevent focusing non-focusable items on `mousedown`.
|
51
|
+
// Even though the item has tabIndex={-1}, that only means take it out of the tab order.
|
52
|
+
event.preventDefault();
|
53
|
+
}
|
54
|
+
else {
|
55
|
+
// Safari doesn't focus a button when clicked so we run our logic on mousedown also
|
56
|
+
this.parent.onItemFocus(this.id());
|
57
|
+
}
|
58
|
+
}
|
59
|
+
/** @ignore */
|
60
|
+
onFocus() {
|
61
|
+
this.parent.onItemFocus(this.id());
|
62
|
+
}
|
63
|
+
/**
|
64
|
+
* Handles the `keydown` event for keyboard navigation within the roving focus group.
|
65
|
+
* Supports navigation based on orientation and direction, and focuses appropriate elements.
|
66
|
+
*
|
67
|
+
* @param event The `KeyboardEvent` object.
|
68
|
+
* @ignore
|
69
|
+
*/
|
70
|
+
handleKeydown(event) {
|
71
|
+
if (event.key === 'Tab' && event.shiftKey) {
|
72
|
+
this.parent.onItemShiftTab();
|
73
|
+
return;
|
74
|
+
}
|
75
|
+
if (event.target !== this.elementRef.nativeElement)
|
76
|
+
return;
|
77
|
+
const focusIntent = getFocusIntent(event, this.parent.orientation, this.parent.dir);
|
78
|
+
if (focusIntent !== undefined) {
|
79
|
+
if (event.metaKey || event.ctrlKey || event.altKey || (this.allowShiftKey ? false : event.shiftKey)) {
|
80
|
+
return;
|
81
|
+
}
|
82
|
+
event.preventDefault();
|
83
|
+
let candidateNodes = this.parent.focusableItems().filter((item) => item.dataset['disabled'] !== '');
|
84
|
+
if (focusIntent === 'last') {
|
85
|
+
candidateNodes.reverse();
|
86
|
+
}
|
87
|
+
else if (focusIntent === 'prev' || focusIntent === 'next') {
|
88
|
+
if (focusIntent === 'prev')
|
89
|
+
candidateNodes.reverse();
|
90
|
+
const currentIndex = candidateNodes.indexOf(this.elementRef.nativeElement);
|
91
|
+
candidateNodes = this.parent.loop
|
92
|
+
? wrapArray(candidateNodes, currentIndex + 1)
|
93
|
+
: candidateNodes.slice(currentIndex + 1);
|
94
|
+
}
|
95
|
+
this.ngZone.runOutsideAngular(() => {
|
96
|
+
// eslint-disable-next-line promise/always-return,promise/catch-or-return
|
97
|
+
Promise.resolve().then(() => {
|
98
|
+
focusFirst(candidateNodes, false, this.elementRef.nativeElement);
|
99
|
+
});
|
100
|
+
});
|
101
|
+
}
|
102
|
+
}
|
103
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.11", ngImport: i0, type: RdxRovingFocusItemDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
|
104
|
+
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "16.1.0", version: "18.2.11", type: RdxRovingFocusItemDirective, isStandalone: true, selector: "[rdxRovingFocusItem]", inputs: { focusable: ["focusable", "focusable", booleanAttribute], active: ["active", "active", booleanAttribute], tabStopId: "tabStopId", allowShiftKey: ["allowShiftKey", "allowShiftKey", booleanAttribute] }, host: { listeners: { "mousedown": "handleMouseDown($event)", "keydown": "handleKeydown($event)", "focus": "onFocus()" }, properties: { "attr.tabindex": "tabIndex", "attr.data-orientation": "parent.orientation", "attr.data-active": "active", "attr.data-disabled": "!focusable ? \"\" : undefined" } }, ngImport: i0 }); }
|
105
|
+
}
|
106
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.11", ngImport: i0, type: RdxRovingFocusItemDirective, decorators: [{
|
107
|
+
type: Directive,
|
108
|
+
args: [{
|
109
|
+
selector: '[rdxRovingFocusItem]',
|
110
|
+
standalone: true,
|
111
|
+
host: {
|
112
|
+
'[attr.tabindex]': 'tabIndex',
|
113
|
+
'[attr.data-orientation]': 'parent.orientation',
|
114
|
+
'[attr.data-active]': 'active',
|
115
|
+
'[attr.data-disabled]': '!focusable ? "" : undefined',
|
116
|
+
'(mousedown)': 'handleMouseDown($event)',
|
117
|
+
'(keydown)': 'handleKeydown($event)',
|
118
|
+
'(focus)': 'onFocus()'
|
119
|
+
}
|
120
|
+
}]
|
121
|
+
}], propDecorators: { focusable: [{
|
122
|
+
type: Input,
|
123
|
+
args: [{ transform: booleanAttribute }]
|
124
|
+
}], active: [{
|
125
|
+
type: Input,
|
126
|
+
args: [{ transform: booleanAttribute }]
|
127
|
+
}], tabStopId: [{
|
128
|
+
type: Input
|
129
|
+
}], allowShiftKey: [{
|
130
|
+
type: Input,
|
131
|
+
args: [{ transform: booleanAttribute }]
|
132
|
+
}] } });
|
133
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"roving-focus-item.directive.js","sourceRoot":"","sources":["../../../../../packages/primitives/roving-focus/src/roving-focus-item.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    booleanAttribute,\n    computed,\n    Directive,\n    ElementRef,\n    inject,\n    Input,\n    NgZone,\n    OnDestroy,\n    OnInit\n} from '@angular/core';\nimport { RdxRovingFocusGroupDirective } from './roving-focus-group.directive';\nimport { focusFirst, generateId, getFocusIntent, wrapArray } from './utils';\n\n@Directive({\n    selector: '[rdxRovingFocusItem]',\n    standalone: true,\n    host: {\n        '[attr.tabindex]': 'tabIndex',\n        '[attr.data-orientation]': 'parent.orientation',\n        '[attr.data-active]': 'active',\n        '[attr.data-disabled]': '!focusable ? \"\" : undefined',\n        '(mousedown)': 'handleMouseDown($event)',\n        '(keydown)': 'handleKeydown($event)',\n        '(focus)': 'onFocus()'\n    }\n})\nexport class RdxRovingFocusItemDirective implements OnInit, OnDestroy {\n    private readonly elementRef = inject(ElementRef);\n    private readonly ngZone = inject(NgZone);\n    protected readonly parent = inject(RdxRovingFocusGroupDirective);\n\n    @Input({ transform: booleanAttribute }) focusable: boolean = true;\n    @Input({ transform: booleanAttribute }) active: boolean = true;\n    @Input() tabStopId: string;\n    @Input({ transform: booleanAttribute }) allowShiftKey: boolean = false;\n\n    private readonly id = computed(() => this.tabStopId || generateId());\n\n    /** @ignore */\n    readonly isCurrentTabStop = computed(() => this.parent.currentTabStopId() === this.id());\n\n    /**\n     * Lifecycle hook triggered on initialization.\n     * Registers the element with the parent roving focus group if it is focusable.\n     * @ignore\n     */\n    ngOnInit() {\n        if (this.focusable) {\n            this.parent.registerItem(this.elementRef.nativeElement);\n            this.parent.onFocusableItemAdd();\n        }\n    }\n\n    /**\n     * Lifecycle hook triggered on destruction.\n     * Unregisters the element from the parent roving focus group if it is focusable.\n     * @ignore\n     */\n    ngOnDestroy() {\n        if (this.focusable) {\n            this.parent.unregisterItem(this.elementRef.nativeElement);\n            this.parent.onFocusableItemRemove();\n        }\n    }\n\n    /**\n     * Determines the `tabIndex` of the element.\n     * Returns `0` if the element is the current tab stop; otherwise, returns `-1`.\n     * @ignore\n     */\n    get tabIndex() {\n        return this.isCurrentTabStop() ? 0 : -1;\n    }\n\n    /** @ignore */\n    handleMouseDown(event: MouseEvent) {\n        if (!this.focusable) {\n            // We prevent focusing non-focusable items on `mousedown`.\n            // Even though the item has tabIndex={-1}, that only means take it out of the tab order.\n            event.preventDefault();\n        } else {\n            // Safari doesn't focus a button when clicked so we run our logic on mousedown also\n            this.parent.onItemFocus(this.id());\n        }\n    }\n\n    /** @ignore */\n    onFocus() {\n        this.parent.onItemFocus(this.id());\n    }\n\n    /**\n     * Handles the `keydown` event for keyboard navigation within the roving focus group.\n     * Supports navigation based on orientation and direction, and focuses appropriate elements.\n     *\n     * @param event The `KeyboardEvent` object.\n     * @ignore\n     */\n    handleKeydown(event: KeyboardEvent) {\n        if (event.key === 'Tab' && event.shiftKey) {\n            this.parent.onItemShiftTab();\n            return;\n        }\n\n        if (event.target !== this.elementRef.nativeElement) return;\n\n        const focusIntent = getFocusIntent(event, this.parent.orientation, this.parent.dir);\n\n        if (focusIntent !== undefined) {\n            if (event.metaKey || event.ctrlKey || event.altKey || (this.allowShiftKey ? false : event.shiftKey)) {\n                return;\n            }\n\n            event.preventDefault();\n\n            let candidateNodes = this.parent.focusableItems().filter((item) => item.dataset['disabled'] !== '');\n\n            if (focusIntent === 'last') {\n                candidateNodes.reverse();\n            } else if (focusIntent === 'prev' || focusIntent === 'next') {\n                if (focusIntent === 'prev') candidateNodes.reverse();\n                const currentIndex = candidateNodes.indexOf(this.elementRef.nativeElement);\n\n                candidateNodes = this.parent.loop\n                    ? wrapArray(candidateNodes, currentIndex + 1)\n                    : candidateNodes.slice(currentIndex + 1);\n            }\n\n            this.ngZone.runOutsideAngular(() => {\n                // eslint-disable-next-line promise/always-return,promise/catch-or-return\n                Promise.resolve().then(() => {\n                    focusFirst(candidateNodes, false, this.elementRef.nativeElement);\n                });\n            });\n        }\n    }\n}\n"]}
|