@danske/sapphire-angular 2.2.0 → 2.3.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/esm2020/lib/radio/public_api.mjs +3 -1
- package/esm2020/lib/radio/src/radio-group.component.mjs +22 -159
- package/esm2020/lib/radio/src/radio.component.mjs +8 -267
- package/esm2020/lib/radio/src/radio.module.mjs +40 -6
- package/esm2020/lib/radio/src/segmented/segmented-radio-group.component.mjs +130 -0
- package/esm2020/lib/radio/src/segmented/segmented-radio.component.mjs +25 -0
- package/esm2020/lib/radio/src/shared/radio-base.mjs +276 -0
- package/esm2020/lib/radio/src/shared/radio-group-base.mjs +166 -0
- package/esm2020/lib/segmented-tabs/src/segmented-tabs.component.mjs +19 -12
- package/esm2020/lib/table/public_api.mjs +2 -1
- package/esm2020/lib/table/src/cdk-virtual-scroll-viewport-fix.directive.mjs +61 -0
- package/esm2020/lib/table/src/table.module.mjs +7 -2
- package/fesm2015/danske-sapphire-angular.mjs +358 -77
- package/fesm2015/danske-sapphire-angular.mjs.map +1 -1
- package/fesm2020/danske-sapphire-angular.mjs +354 -76
- package/fesm2020/danske-sapphire-angular.mjs.map +1 -1
- package/lib/radio/public_api.d.ts +2 -0
- package/lib/radio/src/radio-group.component.d.ts +5 -60
- package/lib/radio/src/radio.component.d.ts +4 -96
- package/lib/radio/src/radio.module.d.ts +8 -5
- package/lib/radio/src/segmented/segmented-radio-group.component.d.ts +40 -0
- package/lib/radio/src/segmented/segmented-radio.component.d.ts +8 -0
- package/lib/radio/src/shared/radio-base.d.ts +103 -0
- package/lib/radio/src/shared/radio-group-base.d.ts +63 -0
- package/lib/segmented-tabs/src/segmented-tabs.component.d.ts +3 -2
- package/lib/table/public_api.d.ts +1 -0
- package/lib/table/src/cdk-virtual-scroll-viewport-fix.directive.d.ts +25 -0
- package/lib/table/src/table.module.d.ts +3 -2
- package/package.json +2 -2
|
@@ -0,0 +1,166 @@
|
|
|
1
|
+
import { __decorate } from "tslib";
|
|
2
|
+
import { ContentChildren, Directive, EventEmitter, forwardRef, Input, Optional, Output, QueryList, } from '@angular/core';
|
|
3
|
+
import { RadioComponent } from '../radio.component';
|
|
4
|
+
import { coerceBooleanProperty } from '@angular/cdk/coercion';
|
|
5
|
+
import { AutoId } from '../../../common/auto-id.decorator';
|
|
6
|
+
import { SapphireRadioChange } from './radio-base';
|
|
7
|
+
import * as i0 from "@angular/core";
|
|
8
|
+
import * as i1 from "../../../field/src/field.component";
|
|
9
|
+
export class RadioGroupBase {
|
|
10
|
+
/** Name of the radio button group. All radio buttons inside this group will use this name. */
|
|
11
|
+
get name() {
|
|
12
|
+
return this._name;
|
|
13
|
+
}
|
|
14
|
+
set name(value) {
|
|
15
|
+
this._name = value;
|
|
16
|
+
this.updateRadioButtonNames();
|
|
17
|
+
}
|
|
18
|
+
get value() {
|
|
19
|
+
return this._value;
|
|
20
|
+
}
|
|
21
|
+
set value(newValue) {
|
|
22
|
+
if (this._value !== newValue) {
|
|
23
|
+
this._value = newValue;
|
|
24
|
+
this.updateSelectedRadioFromValue();
|
|
25
|
+
this._checkSelectedRadioButton();
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
/**
|
|
29
|
+
* The currently selected radio button. If set to a new radio button, the radio group value
|
|
30
|
+
* will be updated to match the new selected button. */
|
|
31
|
+
get selected() {
|
|
32
|
+
return this._selected;
|
|
33
|
+
}
|
|
34
|
+
set selected(selected) {
|
|
35
|
+
this._selected = selected;
|
|
36
|
+
this.value = selected ? selected.value : null;
|
|
37
|
+
this._checkSelectedRadioButton();
|
|
38
|
+
}
|
|
39
|
+
/** Whether the radio group is disabled */
|
|
40
|
+
get disabled() {
|
|
41
|
+
return this._disabled;
|
|
42
|
+
}
|
|
43
|
+
set disabled(value) {
|
|
44
|
+
this._disabled = coerceBooleanProperty(value);
|
|
45
|
+
}
|
|
46
|
+
/** Whether the radio group is readonly */
|
|
47
|
+
get readonly() {
|
|
48
|
+
return this._readonly;
|
|
49
|
+
}
|
|
50
|
+
set readonly(value) {
|
|
51
|
+
this._readonly = coerceBooleanProperty(value);
|
|
52
|
+
}
|
|
53
|
+
/** Whether the radio group is required */
|
|
54
|
+
get required() {
|
|
55
|
+
return this._required;
|
|
56
|
+
}
|
|
57
|
+
set required(value) {
|
|
58
|
+
this._required = coerceBooleanProperty(value);
|
|
59
|
+
}
|
|
60
|
+
constructor(changeDetectorRef, _field) {
|
|
61
|
+
this.changeDetectorRef = changeDetectorRef;
|
|
62
|
+
this._field = _field;
|
|
63
|
+
this.uniqueId = '';
|
|
64
|
+
this.orientation = 'vertical';
|
|
65
|
+
this._name = `sapphire-radio-group-${this.uniqueId}`;
|
|
66
|
+
/**
|
|
67
|
+
* Event emitted when the group value changes.
|
|
68
|
+
* Change events are only emitted when the value changes due to user interaction with
|
|
69
|
+
* the radio button (the same behavior as `<input type-"radio">`).
|
|
70
|
+
*/
|
|
71
|
+
this.change = new EventEmitter();
|
|
72
|
+
this._value = '';
|
|
73
|
+
this._selected = null;
|
|
74
|
+
this._disabled = false;
|
|
75
|
+
this._readonly = false;
|
|
76
|
+
this._required = false;
|
|
77
|
+
/** Child radio buttons. */
|
|
78
|
+
this.radios = new QueryList();
|
|
79
|
+
this.onTouched = () => { };
|
|
80
|
+
this._controlValueAccessorChangeFn = () => { };
|
|
81
|
+
}
|
|
82
|
+
_checkSelectedRadioButton() {
|
|
83
|
+
if (this._selected && !this._selected.checked) {
|
|
84
|
+
this._selected.checked = true;
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
/** Dispatch change event with current selection and group value. */
|
|
88
|
+
_emitChangeEvent() {
|
|
89
|
+
this.change.emit(new SapphireRadioChange(this._selected, this._value));
|
|
90
|
+
}
|
|
91
|
+
// Implemented as part of ControlValueAccessor.
|
|
92
|
+
writeValue(value) {
|
|
93
|
+
this.value = value;
|
|
94
|
+
}
|
|
95
|
+
// Implemented as part of ControlValueAccessor.
|
|
96
|
+
registerOnChange(fn) {
|
|
97
|
+
this._controlValueAccessorChangeFn = fn;
|
|
98
|
+
}
|
|
99
|
+
// Implemented as part of ControlValueAccessor.
|
|
100
|
+
registerOnTouched(fn) {
|
|
101
|
+
this.onTouched = fn;
|
|
102
|
+
}
|
|
103
|
+
// Implemented as part of ControlValueAccessor.
|
|
104
|
+
setDisabledState(disabled) {
|
|
105
|
+
this.disabled = disabled;
|
|
106
|
+
this.changeDetectorRef.markForCheck();
|
|
107
|
+
}
|
|
108
|
+
// Implementing FieldControl interface
|
|
109
|
+
isDisabled() {
|
|
110
|
+
return this.disabled;
|
|
111
|
+
}
|
|
112
|
+
/** Updates the `selected` radio button from the internal _value state. */
|
|
113
|
+
updateSelectedRadioFromValue() {
|
|
114
|
+
// If the value already matches the selected radio, do nothing.
|
|
115
|
+
const alreadySelected = this._selected !== null && this._selected.value === this._value;
|
|
116
|
+
if (this.radios && !alreadySelected) {
|
|
117
|
+
this._selected = null;
|
|
118
|
+
this.radios.forEach((radio) => {
|
|
119
|
+
radio.checked = this.value === radio.value;
|
|
120
|
+
if (radio.checked) {
|
|
121
|
+
this._selected = radio;
|
|
122
|
+
}
|
|
123
|
+
});
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
updateRadioButtonNames() {
|
|
127
|
+
if (this.radios) {
|
|
128
|
+
this.radios.forEach((radio) => {
|
|
129
|
+
radio.name = this.name;
|
|
130
|
+
});
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
RadioGroupBase.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.0", ngImport: i0, type: RadioGroupBase, deps: [{ token: i0.ChangeDetectorRef }, { token: i1.FieldComponent, optional: true }], target: i0.ɵɵFactoryTarget.Directive });
|
|
135
|
+
RadioGroupBase.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "15.2.0", type: RadioGroupBase, isStandalone: true, inputs: { orientation: "orientation", name: "name", value: "value", selected: "selected", disabled: "disabled", readonly: "readonly", required: "required" }, outputs: { change: "change" }, queries: [{ propertyName: "radios", predicate: i0.forwardRef(function () { return RadioComponent; }), descendants: true }], ngImport: i0 });
|
|
136
|
+
__decorate([
|
|
137
|
+
AutoId()
|
|
138
|
+
], RadioGroupBase.prototype, "uniqueId", void 0);
|
|
139
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.0", ngImport: i0, type: RadioGroupBase, decorators: [{
|
|
140
|
+
type: Directive,
|
|
141
|
+
args: [{
|
|
142
|
+
standalone: true,
|
|
143
|
+
}]
|
|
144
|
+
}], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }, { type: i1.FieldComponent, decorators: [{
|
|
145
|
+
type: Optional
|
|
146
|
+
}] }]; }, propDecorators: { uniqueId: [], orientation: [{
|
|
147
|
+
type: Input
|
|
148
|
+
}], name: [{
|
|
149
|
+
type: Input
|
|
150
|
+
}], change: [{
|
|
151
|
+
type: Output
|
|
152
|
+
}], value: [{
|
|
153
|
+
type: Input
|
|
154
|
+
}], selected: [{
|
|
155
|
+
type: Input
|
|
156
|
+
}], disabled: [{
|
|
157
|
+
type: Input
|
|
158
|
+
}], readonly: [{
|
|
159
|
+
type: Input
|
|
160
|
+
}], required: [{
|
|
161
|
+
type: Input
|
|
162
|
+
}], radios: [{
|
|
163
|
+
type: ContentChildren,
|
|
164
|
+
args: [forwardRef(() => RadioComponent), { descendants: true }]
|
|
165
|
+
}] } });
|
|
166
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"radio-group-base.js","sourceRoot":"","sources":["../../../../../../src/lib/radio/src/shared/radio-group-base.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectorRef,\n  ContentChildren,\n  Directive,\n  EventEmitter,\n  forwardRef,\n  Input,\n  Optional,\n  Output,\n  QueryList,\n} from '@angular/core';\nimport { RadioComponent } from '../radio.component';\nimport { ControlValueAccessor } from '@angular/forms';\nimport { BooleanInput, coerceBooleanProperty } from '@angular/cdk/coercion';\nimport { AutoId } from '../../../common/auto-id.decorator';\nimport { FieldControl } from '../../../field/src/field-control';\nimport { SapphireRadioChange } from './radio-base';\nimport { FieldComponent } from '../../../field/src/field.component';\n\n@Directive({\n  standalone: true,\n})\nexport abstract class RadioGroupBase\n  implements ControlValueAccessor, FieldControl\n{\n  @AutoId()\n  public uniqueId = '';\n\n  @Input() orientation: 'vertical' | 'horizontal' = 'vertical';\n\n  /** Name of the radio button group. All radio buttons inside this group will use this name. */\n  @Input()\n  get name(): string {\n    return this._name;\n  }\n  set name(value: string) {\n    this._name = value;\n    this.updateRadioButtonNames();\n  }\n  private _name: string = `sapphire-radio-group-${this.uniqueId}`;\n\n  /**\n   * Event emitted when the group value changes.\n   * Change events are only emitted when the value changes due to user interaction with\n   * the radio button (the same behavior as `<input type-\"radio\">`).\n   */\n  @Output() readonly change: EventEmitter<SapphireRadioChange> =\n    new EventEmitter<SapphireRadioChange>();\n\n  @Input()\n  get value(): any {\n    return this._value;\n  }\n  set value(newValue: any) {\n    if (this._value !== newValue) {\n      this._value = newValue;\n\n      this.updateSelectedRadioFromValue();\n      this._checkSelectedRadioButton();\n    }\n  }\n  private _value: any = '';\n\n  /**\n   * The currently selected radio button. If set to a new radio button, the radio group value\n   * will be updated to match the new selected button. */\n  @Input()\n  get selected() {\n    return this._selected;\n  }\n  set selected(selected: RadioComponent | null) {\n    this._selected = selected;\n    this.value = selected ? selected.value : null;\n    this._checkSelectedRadioButton();\n  }\n  private _selected: RadioComponent | null = null;\n\n  /** Whether the radio group is disabled */\n  @Input()\n  get disabled(): boolean {\n    return this._disabled;\n  }\n  set disabled(value: BooleanInput) {\n    this._disabled = coerceBooleanProperty(value);\n  }\n\n  private _disabled: boolean = false;\n\n  /** Whether the radio group is readonly */\n  @Input()\n  get readonly(): boolean {\n    return this._readonly;\n  }\n  set readonly(value: BooleanInput) {\n    this._readonly = coerceBooleanProperty(value);\n  }\n\n  private _readonly: boolean = false;\n\n  /** Whether the radio group is required */\n  @Input()\n  get required(): boolean {\n    return this._required;\n  }\n  set required(value: BooleanInput) {\n    this._required = coerceBooleanProperty(value);\n  }\n\n  private _required: boolean = false;\n\n  /** Child radio buttons. */\n  @ContentChildren(forwardRef(() => RadioComponent), { descendants: true })\n  private radios: QueryList<RadioComponent> = new QueryList<RadioComponent>();\n\n  onTouched: () => any = () => {};\n  _controlValueAccessorChangeFn: (value: any) => void = () => {};\n\n  constructor(\n    private changeDetectorRef: ChangeDetectorRef,\n    @Optional() public _field: FieldComponent\n  ) {}\n\n  _checkSelectedRadioButton() {\n    if (this._selected && !this._selected.checked) {\n      this._selected.checked = true;\n    }\n  }\n\n  /** Dispatch change event with current selection and group value. */\n  _emitChangeEvent(): void {\n    this.change.emit(new SapphireRadioChange(this._selected!, this._value));\n  }\n\n  // Implemented as part of ControlValueAccessor.\n  writeValue(value: any) {\n    this.value = value;\n  }\n\n  // Implemented as part of ControlValueAccessor.\n  registerOnChange(fn: (value: any) => void) {\n    this._controlValueAccessorChangeFn = fn;\n  }\n\n  // Implemented as part of ControlValueAccessor.\n  registerOnTouched(fn: any) {\n    this.onTouched = fn;\n  }\n\n  // Implemented as part of ControlValueAccessor.\n  setDisabledState(disabled: boolean) {\n    this.disabled = disabled;\n    this.changeDetectorRef.markForCheck();\n  }\n\n  // Implementing FieldControl interface\n  isDisabled(): boolean {\n    return this.disabled;\n  }\n\n  /** Updates the `selected` radio button from the internal _value state. */\n  private updateSelectedRadioFromValue(): void {\n    // If the value already matches the selected radio, do nothing.\n    const alreadySelected =\n      this._selected !== null && this._selected.value === this._value;\n\n    if (this.radios && !alreadySelected) {\n      this._selected = null;\n      this.radios.forEach((radio) => {\n        radio.checked = this.value === radio.value;\n        if (radio.checked) {\n          this._selected = radio;\n        }\n      });\n    }\n  }\n\n  private updateRadioButtonNames(): void {\n    if (this.radios) {\n      this.radios.forEach((radio) => {\n        radio.name = this.name;\n      });\n    }\n  }\n}\n"]}
|
|
@@ -32,7 +32,7 @@ export class SegmentedTabsComponent {
|
|
|
32
32
|
this.selectedIndexChange = new EventEmitter();
|
|
33
33
|
this.tabs = new QueryList();
|
|
34
34
|
this._tabIds = [];
|
|
35
|
-
this.
|
|
35
|
+
this._gliderStyles = { left: '0px', width: '0px' };
|
|
36
36
|
this._animationEnabled = false;
|
|
37
37
|
this._focusVisible = false;
|
|
38
38
|
this.skipAnimationFrame();
|
|
@@ -54,17 +54,17 @@ export class SegmentedTabsComponent {
|
|
|
54
54
|
// changed after it was checked" angular error.
|
|
55
55
|
//
|
|
56
56
|
// https://angular.io/errors/NG0100
|
|
57
|
-
this.
|
|
57
|
+
this.setGliderStyles();
|
|
58
58
|
this.changeDetector.detectChanges();
|
|
59
59
|
this.resizeObserver = new ResizeObserver(() => {
|
|
60
60
|
// zonejs doesn't monkey-patch resize observer (yet)
|
|
61
61
|
this.zone.run(() => {
|
|
62
|
-
this.
|
|
62
|
+
this.setGliderStyles();
|
|
63
63
|
});
|
|
64
64
|
});
|
|
65
65
|
this.resizeObserver.observe(this.segmentedControl.nativeElement);
|
|
66
66
|
this.selectedIndexChange.subscribe(() => {
|
|
67
|
-
this.
|
|
67
|
+
this.setGliderStyles();
|
|
68
68
|
});
|
|
69
69
|
}
|
|
70
70
|
ngOnDestroy() {
|
|
@@ -72,7 +72,7 @@ export class SegmentedTabsComponent {
|
|
|
72
72
|
this.selectedIndexChange.unsubscribe();
|
|
73
73
|
}
|
|
74
74
|
ngAfterViewChecked() {
|
|
75
|
-
this.
|
|
75
|
+
this.setGliderStyles();
|
|
76
76
|
this.changeDetector.detectChanges();
|
|
77
77
|
}
|
|
78
78
|
getFocusedTab() {
|
|
@@ -142,11 +142,18 @@ export class SegmentedTabsComponent {
|
|
|
142
142
|
return;
|
|
143
143
|
}
|
|
144
144
|
}
|
|
145
|
-
|
|
145
|
+
setGliderStyles() {
|
|
146
|
+
const { offsetLeft, clientWidth } = this.getSelectedTabPosition();
|
|
147
|
+
this._gliderStyles = {
|
|
148
|
+
left: `${offsetLeft}px`,
|
|
149
|
+
width: `${clientWidth}px`,
|
|
150
|
+
};
|
|
151
|
+
}
|
|
152
|
+
getSelectedTabPosition() {
|
|
146
153
|
const tab = this.tabs.toArray()[this.selectedIndex]?._button?.nativeElement;
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
154
|
+
return {
|
|
155
|
+
offsetLeft: tab?.offsetLeft || 0,
|
|
156
|
+
clientWidth: tab?.clientWidth || 0,
|
|
150
157
|
};
|
|
151
158
|
}
|
|
152
159
|
onTabsChange() {
|
|
@@ -197,7 +204,7 @@ export class SegmentedTabsComponent {
|
|
|
197
204
|
}
|
|
198
205
|
}
|
|
199
206
|
SegmentedTabsComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.0", ngImport: i0, type: SegmentedTabsComponent, deps: [{ token: i0.NgZone }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
200
|
-
SegmentedTabsComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.0", type: SegmentedTabsComponent, selector: "sp-segmented-tabs", inputs: { size: "size", align: "align", keyboardActivation: "keyboardActivation", disabled: "disabled", selectedIndex: "selectedIndex" }, outputs: { selectedIndexChange: "selectedIndexChange" }, host: { properties: { "attr.align": "null" } }, providers: [ViewEncapsulationProvider], queries: [{ propertyName: "tabs", predicate: i0.forwardRef(function () { return SegmentedTabComponent; }) }], viewQueries: [{ propertyName: "segmentedControl", first: true, predicate: ["segmentedControl"], descendants: true }], exportAs: ["spSegmentedTabs"], ngImport: i0, template: "<div\n #segmentedControl\n class=\"sapphire-segmented-control\"\n [class.sapphire-segmented-control--lg]=\"size === 'lg'\"\n [class.sapphire-segmented-control--sm]=\"size === 'sm'\"\n [class.focus-visible]=\"_focusVisible\"\n [class.sapphire-segmented-control--manual-keyboard-activation]=\"\n keyboardActivation === 'manual'\n \"\n [ngClass]=\"\n align === 'stretch'\n ? 'sapphire-segmented-control--stretch'\n : 'sapphire-segmented-control--align-' + align\n \"\n>\n <div\n class=\"sapphire-segmented-control__button-container\"\n role=\"tablist\"\n aria-orientation=\"horizontal\"\n cdkMonitorSubtreeFocus\n (cdkFocusChange)=\"_onFocusChange($event)\"\n (keydown)=\"_onKeyDown($event)\"\n >\n <ng-content select=\"sp-segmented-tab\"></ng-content>\n <span\n [style.left]=\"
|
|
207
|
+
SegmentedTabsComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.0", type: SegmentedTabsComponent, selector: "sp-segmented-tabs", inputs: { size: "size", align: "align", keyboardActivation: "keyboardActivation", disabled: "disabled", selectedIndex: "selectedIndex" }, outputs: { selectedIndexChange: "selectedIndexChange" }, host: { properties: { "attr.align": "null" } }, providers: [ViewEncapsulationProvider], queries: [{ propertyName: "tabs", predicate: i0.forwardRef(function () { return SegmentedTabComponent; }) }], viewQueries: [{ propertyName: "segmentedControl", first: true, predicate: ["segmentedControl"], descendants: true }], exportAs: ["spSegmentedTabs"], ngImport: i0, template: "<div\n #segmentedControl\n class=\"sapphire-segmented-control\"\n [class.sapphire-segmented-control--lg]=\"size === 'lg'\"\n [class.sapphire-segmented-control--sm]=\"size === 'sm'\"\n [class.focus-visible]=\"_focusVisible\"\n [class.sapphire-segmented-control--manual-keyboard-activation]=\"\n keyboardActivation === 'manual'\n \"\n [ngClass]=\"\n align === 'stretch'\n ? 'sapphire-segmented-control--stretch'\n : 'sapphire-segmented-control--align-' + align\n \"\n>\n <div\n class=\"sapphire-segmented-control__button-container\"\n role=\"tablist\"\n aria-orientation=\"horizontal\"\n cdkMonitorSubtreeFocus\n (cdkFocusChange)=\"_onFocusChange($event)\"\n (keydown)=\"_onKeyDown($event)\"\n >\n <ng-content select=\"sp-segmented-tab\"></ng-content>\n <span\n [style.left]=\"_gliderStyles.left\"\n [style.width]=\"_gliderStyles.width\"\n [class.sapphire-segmented-control__glider--with-transition]=\"\n _animationEnabled\n \"\n class=\"sapphire-segmented-control__glider\"\n role=\"none\"\n ></span>\n </div>\n</div>\n<ng-container *ngTemplateOutlet=\"_getTabContentTemplate()\"></ng-container>\n", styles: [".sapphire-segmented-control{display:flex;height:var(--sapphire-semantic-size-height-control-md)}.sapphire-segmented-control--stretch .sapphire-segmented-control__button-container{width:100%}.sapphire-segmented-control--align-center{justify-content:center}.sapphire-segmented-control--align-right{justify-content:flex-end}.sapphire-segmented-control--align-left{justify-content:flex-start}.sapphire-segmented-control__button-container{display:inline-flex;position:relative;height:100%;background-color:var(--sapphire-semantic-color-background-control-special-secondary-default);border-radius:var(--sapphire-semantic-size-height-control-lg);padding:var(--sapphire-global-size-static-5);box-sizing:border-box}.sapphire-segmented-control__glider{position:absolute;top:var(--sapphire-global-size-static-5);left:0;height:calc(100% - var(--sapphire-global-size-static-5) * 2);display:block;background-color:var(--sapphire-semantic-color-background-control-special-primary-default);box-shadow:var(--sapphire-global-shadow-sm);box-sizing:border-box;z-index:1;border-radius:var(--sapphire-semantic-size-height-control-lg)}.sapphire-segmented-control__glider--with-transition{transition:width var(--sapphire-semantic-time-motion-quick) ease-in-out,left var(--sapphire-semantic-time-motion-quick) ease-in-out}.sapphire-segmented-control__button{font-family:var(--sapphire-semantic-font-name-default);font-weight:var(--sapphire-semantic-font-weight-default-semibold);font-size:var(--sapphire-semantic-size-font-control-md);-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;flex:1 1 auto;box-sizing:content-box;margin:0;border:0;height:100%;min-width:var(--sapphire-global-size-generic-120);padding:0 var(--sapphire-semantic-size-spacing-50);background-color:transparent;color:var(--sapphire-semantic-color-content-default-primary);cursor:pointer;outline:none;z-index:2;display:flex;align-items:center;justify-content:center;position:relative;transition:color var(--sapphire-semantic-time-motion-quick) ease-in-out}.sapphire-segmented-control__button--active{color:var(--sapphire-semantic-color-content-link-primary-default)}.sapphire-segmented-control__button--disabled{opacity:var(--sapphire-semantic-opacity-disabled);cursor:not-allowed}.sapphire-segmented-control.focus-visible:focus-within .sapphire-segmented-control__glider{outline:var(--sapphire-semantic-size-focus-ring) solid var(--sapphire-semantic-color-focus-ring)}.sapphire-segmented-control--manual-keyboard-activation.focus-visible .sapphire-segmented-control__glider{transition:none}.sapphire-segmented-control--manual-keyboard-activation.focus-visible .sapphire-segmented-control__button:focus{outline:var(--sapphire-semantic-size-focus-ring) solid var(--sapphire-semantic-color-focus-ring);border-radius:var(--sapphire-semantic-size-height-control-lg)}.sapphire-segmented-control.focus-visible:focus-within.sapphire-segmented-control--manual-keyboard-activation .sapphire-segmented-control__glider{outline:none}.sapphire-segmented-control--lg{height:var(--sapphire-semantic-size-height-control-lg)}.sapphire-segmented-control--lg .sapphire-segmented-control__button{font-size:var(--sapphire-semantic-size-font-control-default)}.sapphire-segmented-control--sm{height:var(--sapphire-semantic-size-height-control-sm)}.sapphire-segmented-control--sm .sapphire-segmented-control__button{font-size:var(--sapphire-semantic-size-font-control-sm)}.sapphire-segmented-control__radio-input{margin:0;overflow:visible;position:absolute;top:0;left:0;height:100%;width:100%;opacity:.0001;z-index:1;cursor:pointer}.sapphire-segmented-control__button--disabled .sapphire-segmented-control__radio-input{cursor:not-allowed}\n"], dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i2.CdkMonitorFocus, selector: "[cdkMonitorElementFocus], [cdkMonitorSubtreeFocus]", outputs: ["cdkFocusChange"], exportAs: ["cdkMonitorFocus"] }] });
|
|
201
208
|
__decorate([
|
|
202
209
|
CoerceBoolean
|
|
203
210
|
], SegmentedTabsComponent.prototype, "disabled", void 0);
|
|
@@ -205,7 +212,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.0", ngImpor
|
|
|
205
212
|
type: Component,
|
|
206
213
|
args: [{ selector: 'sp-segmented-tabs', providers: [ViewEncapsulationProvider], exportAs: 'spSegmentedTabs', host: {
|
|
207
214
|
'[attr.align]': 'null',
|
|
208
|
-
}, template: "<div\n #segmentedControl\n class=\"sapphire-segmented-control\"\n [class.sapphire-segmented-control--lg]=\"size === 'lg'\"\n [class.sapphire-segmented-control--sm]=\"size === 'sm'\"\n [class.focus-visible]=\"_focusVisible\"\n [class.sapphire-segmented-control--manual-keyboard-activation]=\"\n keyboardActivation === 'manual'\n \"\n [ngClass]=\"\n align === 'stretch'\n ? 'sapphire-segmented-control--stretch'\n : 'sapphire-segmented-control--align-' + align\n \"\n>\n <div\n class=\"sapphire-segmented-control__button-container\"\n role=\"tablist\"\n aria-orientation=\"horizontal\"\n cdkMonitorSubtreeFocus\n (cdkFocusChange)=\"_onFocusChange($event)\"\n (keydown)=\"_onKeyDown($event)\"\n >\n <ng-content select=\"sp-segmented-tab\"></ng-content>\n <span\n [style.left]=\"
|
|
215
|
+
}, template: "<div\n #segmentedControl\n class=\"sapphire-segmented-control\"\n [class.sapphire-segmented-control--lg]=\"size === 'lg'\"\n [class.sapphire-segmented-control--sm]=\"size === 'sm'\"\n [class.focus-visible]=\"_focusVisible\"\n [class.sapphire-segmented-control--manual-keyboard-activation]=\"\n keyboardActivation === 'manual'\n \"\n [ngClass]=\"\n align === 'stretch'\n ? 'sapphire-segmented-control--stretch'\n : 'sapphire-segmented-control--align-' + align\n \"\n>\n <div\n class=\"sapphire-segmented-control__button-container\"\n role=\"tablist\"\n aria-orientation=\"horizontal\"\n cdkMonitorSubtreeFocus\n (cdkFocusChange)=\"_onFocusChange($event)\"\n (keydown)=\"_onKeyDown($event)\"\n >\n <ng-content select=\"sp-segmented-tab\"></ng-content>\n <span\n [style.left]=\"_gliderStyles.left\"\n [style.width]=\"_gliderStyles.width\"\n [class.sapphire-segmented-control__glider--with-transition]=\"\n _animationEnabled\n \"\n class=\"sapphire-segmented-control__glider\"\n role=\"none\"\n ></span>\n </div>\n</div>\n<ng-container *ngTemplateOutlet=\"_getTabContentTemplate()\"></ng-container>\n", styles: [".sapphire-segmented-control{display:flex;height:var(--sapphire-semantic-size-height-control-md)}.sapphire-segmented-control--stretch .sapphire-segmented-control__button-container{width:100%}.sapphire-segmented-control--align-center{justify-content:center}.sapphire-segmented-control--align-right{justify-content:flex-end}.sapphire-segmented-control--align-left{justify-content:flex-start}.sapphire-segmented-control__button-container{display:inline-flex;position:relative;height:100%;background-color:var(--sapphire-semantic-color-background-control-special-secondary-default);border-radius:var(--sapphire-semantic-size-height-control-lg);padding:var(--sapphire-global-size-static-5);box-sizing:border-box}.sapphire-segmented-control__glider{position:absolute;top:var(--sapphire-global-size-static-5);left:0;height:calc(100% - var(--sapphire-global-size-static-5) * 2);display:block;background-color:var(--sapphire-semantic-color-background-control-special-primary-default);box-shadow:var(--sapphire-global-shadow-sm);box-sizing:border-box;z-index:1;border-radius:var(--sapphire-semantic-size-height-control-lg)}.sapphire-segmented-control__glider--with-transition{transition:width var(--sapphire-semantic-time-motion-quick) ease-in-out,left var(--sapphire-semantic-time-motion-quick) ease-in-out}.sapphire-segmented-control__button{font-family:var(--sapphire-semantic-font-name-default);font-weight:var(--sapphire-semantic-font-weight-default-semibold);font-size:var(--sapphire-semantic-size-font-control-md);-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;flex:1 1 auto;box-sizing:content-box;margin:0;border:0;height:100%;min-width:var(--sapphire-global-size-generic-120);padding:0 var(--sapphire-semantic-size-spacing-50);background-color:transparent;color:var(--sapphire-semantic-color-content-default-primary);cursor:pointer;outline:none;z-index:2;display:flex;align-items:center;justify-content:center;position:relative;transition:color var(--sapphire-semantic-time-motion-quick) ease-in-out}.sapphire-segmented-control__button--active{color:var(--sapphire-semantic-color-content-link-primary-default)}.sapphire-segmented-control__button--disabled{opacity:var(--sapphire-semantic-opacity-disabled);cursor:not-allowed}.sapphire-segmented-control.focus-visible:focus-within .sapphire-segmented-control__glider{outline:var(--sapphire-semantic-size-focus-ring) solid var(--sapphire-semantic-color-focus-ring)}.sapphire-segmented-control--manual-keyboard-activation.focus-visible .sapphire-segmented-control__glider{transition:none}.sapphire-segmented-control--manual-keyboard-activation.focus-visible .sapphire-segmented-control__button:focus{outline:var(--sapphire-semantic-size-focus-ring) solid var(--sapphire-semantic-color-focus-ring);border-radius:var(--sapphire-semantic-size-height-control-lg)}.sapphire-segmented-control.focus-visible:focus-within.sapphire-segmented-control--manual-keyboard-activation .sapphire-segmented-control__glider{outline:none}.sapphire-segmented-control--lg{height:var(--sapphire-semantic-size-height-control-lg)}.sapphire-segmented-control--lg .sapphire-segmented-control__button{font-size:var(--sapphire-semantic-size-font-control-default)}.sapphire-segmented-control--sm{height:var(--sapphire-semantic-size-height-control-sm)}.sapphire-segmented-control--sm .sapphire-segmented-control__button{font-size:var(--sapphire-semantic-size-font-control-sm)}.sapphire-segmented-control__radio-input{margin:0;overflow:visible;position:absolute;top:0;left:0;height:100%;width:100%;opacity:.0001;z-index:1;cursor:pointer}.sapphire-segmented-control__button--disabled .sapphire-segmented-control__radio-input{cursor:not-allowed}\n"] }]
|
|
209
216
|
}], ctorParameters: function () { return [{ type: i0.NgZone }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { segmentedControl: [{
|
|
210
217
|
type: ViewChild,
|
|
211
218
|
args: ['segmentedControl']
|
|
@@ -225,4 +232,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.0", ngImpor
|
|
|
225
232
|
type: ContentChildren,
|
|
226
233
|
args: [forwardRef(() => SegmentedTabComponent)]
|
|
227
234
|
}] } });
|
|
228
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"segmented-tabs.component.js","sourceRoot":"","sources":["../../../../../src/lib/segmented-tabs/src/segmented-tabs.component.ts","../../../../../src/lib/segmented-tabs/src/segmented-tabs.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterContentInit,\n  AfterViewChecked,\n  AfterViewInit,\n  ChangeDetectorRef,\n  Component,\n  ContentChildren,\n  ElementRef,\n  EventEmitter,\n  forwardRef,\n  Input,\n  NgZone,\n  OnDestroy,\n  Output,\n  QueryList,\n  ViewChild,\n} from '@angular/core';\nimport {\n  BooleanInput,\n  coerceNumberProperty,\n  NumberInput,\n} from '@angular/cdk/coercion';\nimport { FocusOrigin } from '@angular/cdk/a11y';\n\nimport { CoerceBoolean } from '../../common/coerce-boolean.decorator';\nimport { SegmentedTabComponent } from './segmented-tab.component';\nimport { ViewEncapsulationProvider } from '../../common/sapphire-view-encapsulation';\n\n@Component({\n  selector: 'sp-segmented-tabs',\n  templateUrl: './segmented-tabs.component.html',\n  styleUrls: ['./segmented-tabs.component.scss'],\n  providers: [ViewEncapsulationProvider],\n  exportAs: 'spSegmentedTabs',\n  host: {\n    '[attr.align]': 'null',\n  },\n})\nexport class SegmentedTabsComponent\n  implements AfterContentInit, AfterViewInit, OnDestroy, AfterViewChecked\n{\n  @ViewChild('segmentedControl') segmentedControl!: ElementRef;\n\n  /** Size of tabs */\n  @Input()\n  size: 'sm' | 'md' | 'lg' = 'md';\n\n  /**\n   * Alignment of tabs' buttons group.\n   * @default 'left'\n   */\n  @Input()\n  align?: 'left' | 'center' | 'right' | 'stretch' = 'left';\n\n  /** Whether tabs are activated automatically on focus or manually. */\n  @Input()\n  keyboardActivation?: 'manual' | 'automatic' = 'automatic';\n\n  /** Whether the entire component should be disabled */\n  @Input()\n  @CoerceBoolean\n  disabled: BooleanInput;\n\n  /** Index of the selected tab */\n  @Input()\n  get selectedIndex(): number {\n    return this._selectedIndex || 0;\n  }\n  set selectedIndex(value: NumberInput) {\n    this._selectedIndex = coerceNumberProperty(value);\n    this._focusedIndex = this._selectedIndex;\n  }\n  private _selectedIndex?: number;\n  private _focusedIndex: number = 0;\n\n  @Output() readonly selectedIndexChange: EventEmitter<number> =\n    new EventEmitter<number>();\n\n  @ContentChildren(forwardRef(() => SegmentedTabComponent))\n  readonly tabs: QueryList<SegmentedTabComponent> = new QueryList();\n  private _tabIds: string[] = [];\n\n  private resizeObserver: ResizeObserver | undefined;\n  _activeTabPosition = { left: '0px', width: '0px' };\n  _animationEnabled = false;\n  _focusVisible = false;\n\n  constructor(private zone: NgZone, private changeDetector: ChangeDetectorRef) {\n    this.skipAnimationFrame();\n  }\n\n  ngAfterContentInit(): void {\n    this._tabIds = this.tabs.map((tab) => tab.id);\n    this.tabs.changes.subscribe(() => {\n      this.onTabsChange();\n    });\n    if (this._selectedIndex === undefined) {\n      this.selectedIndex = this.findFocusableIndex(0, 'right') || 0;\n      this.selectedIndexChange.emit(this.selectedIndex);\n    }\n  }\n\n  ngAfterViewInit(): void {\n    // The value of \"_getTabContentTemplate\", which is bound in the\n    // template, changes after the view has been checked. This\n    // \"detectChanges()\" avoids the \"NG0100: Expression has\n    // changed after it was checked\" angular error.\n    //\n    // https://angular.io/errors/NG0100\n    this.setActiveTabPosition();\n    this.changeDetector.detectChanges();\n    this.resizeObserver = new ResizeObserver(() => {\n      // zonejs doesn't monkey-patch resize observer (yet)\n      this.zone.run(() => {\n        this.setActiveTabPosition();\n      });\n    });\n\n    this.resizeObserver.observe(this.segmentedControl.nativeElement);\n    this.selectedIndexChange.subscribe(() => {\n      this.setActiveTabPosition();\n    });\n  }\n\n  ngOnDestroy(): void {\n    this.resizeObserver?.unobserve(this.segmentedControl.nativeElement);\n    this.selectedIndexChange.unsubscribe();\n  }\n\n  ngAfterViewChecked(): void {\n    this.setActiveTabPosition();\n    this.changeDetector.detectChanges();\n  }\n\n  getFocusedTab() {\n    return this.tabs.toArray()[this._focusedIndex];\n  }\n\n  selectTab(selectedTab: SegmentedTabComponent) {\n    this.selectedIndex = this.tabs\n      .toArray()\n      .findIndex((tab) => tab === selectedTab);\n    this.selectedIndexChange.emit(this.selectedIndex);\n  }\n\n  isSelected(tab: SegmentedTabComponent): boolean {\n    return this.tabs?.toArray().indexOf(tab) === this.selectedIndex;\n  }\n\n  _getTabContentTemplate() {\n    return this.tabs.toArray()[this.selectedIndex]?._contentTemplate || null;\n  }\n\n  _onFocusChange(origin: FocusOrigin) {\n    this._focusVisible = origin === 'keyboard';\n    if (!origin) {\n      this._focusedIndex = this.selectedIndex;\n    }\n  }\n\n  _onKeyDown(event: KeyboardEvent) {\n    const focusNextValidTab = (\n      targetIndex: number,\n      direction: 'left' | 'right'\n    ) => {\n      if (this.tabs.toArray().every((tab) => tab.disabled)) return;\n      const index = this.findFocusableIndex(targetIndex, direction);\n      if (index === null) return;\n      const button = this.tabs.toArray()[index]._button?.nativeElement;\n      if (!button) return;\n      button.focus();\n      if (this.keyboardActivation === 'automatic') {\n        this.selectedIndex = index;\n        this.selectedIndexChange.emit(this.selectedIndex);\n      } else {\n        this._focusedIndex = index;\n      }\n    };\n    switch (event.key) {\n      case 'ArrowLeft': {\n        event.stopPropagation();\n        focusNextValidTab(this._focusedIndex - 1, 'left');\n        break;\n      }\n      case 'ArrowRight': {\n        event.stopPropagation();\n        focusNextValidTab(this._focusedIndex + 1, 'right');\n        break;\n      }\n      case 'Home': {\n        event.stopPropagation();\n        event.preventDefault();\n        focusNextValidTab(0, 'right');\n        break;\n      }\n      case 'End': {\n        event.stopPropagation();\n        event.preventDefault();\n        focusNextValidTab(this.tabs.length - 1, 'left');\n        break;\n      }\n      default:\n        return;\n    }\n  }\n\n  private setActiveTabPosition() {\n    const tab = this.tabs.toArray()[this.selectedIndex]?._button?.nativeElement;\n    this._activeTabPosition = {\n      left: `${tab?.offsetLeft || 0}px`,\n      width: `${tab?.clientWidth || 0}px`,\n    };\n  }\n\n  private onTabsChange() {\n    // Ensure the current tab doesn't change if tabs change\n    // and tab with currently selected id still exists\n    const previousTabId = this._tabIds[this.selectedIndex];\n    this._tabIds = this.tabs.map((tab) => tab.id);\n    if (this._tabIds.includes(previousTabId)) {\n      this.selectedIndex = this._tabIds.findIndex(\n        (tabId) => tabId === previousTabId\n      );\n    } else {\n      const maxIndex = this.tabs.length - 1;\n      if (this.selectedIndex > maxIndex) {\n        this.selectedIndex = maxIndex;\n        this.selectedIndexChange.emit(this.selectedIndex);\n      }\n    }\n    this.skipAnimationFrame();\n  }\n\n  private skipAnimationFrame() {\n    // We want to avoid animation on first render and whenever tab content changes\n    this._animationEnabled = false;\n    requestAnimationFrame(() => {\n      this._animationEnabled = true;\n    });\n  }\n\n  private findFocusableIndex(\n    index: number,\n    direction: 'left' | 'right'\n  ): number | null {\n    if (this.tabs.toArray().every((tab) => tab.disabled)) return null;\n    const constrainIndex = (index: number) => {\n      if (index >= this.tabs.length) {\n        return 0;\n      } else if (index < 0) {\n        return this.tabs.length - 1;\n      } else {\n        return index;\n      }\n    };\n    const constrainedIndex = constrainIndex(index);\n    if (this.tabs.toArray()[constrainedIndex].disabled) {\n      return this.findFocusableIndex(\n        constrainedIndex + (direction === 'left' ? -1 : 1),\n        direction\n      );\n    } else {\n      return constrainedIndex;\n    }\n  }\n}\n","<div\n  #segmentedControl\n  class=\"sapphire-segmented-control\"\n  [class.sapphire-segmented-control--lg]=\"size === 'lg'\"\n  [class.sapphire-segmented-control--sm]=\"size === 'sm'\"\n  [class.focus-visible]=\"_focusVisible\"\n  [class.sapphire-segmented-control--manual-keyboard-activation]=\"\n    keyboardActivation === 'manual'\n  \"\n  [ngClass]=\"\n    align === 'stretch'\n      ? 'sapphire-segmented-control--stretch'\n      : 'sapphire-segmented-control--align-' + align\n  \"\n>\n  <div\n    class=\"sapphire-segmented-control__button-container\"\n    role=\"tablist\"\n    aria-orientation=\"horizontal\"\n    cdkMonitorSubtreeFocus\n    (cdkFocusChange)=\"_onFocusChange($event)\"\n    (keydown)=\"_onKeyDown($event)\"\n  >\n    <ng-content select=\"sp-segmented-tab\"></ng-content>\n    <span\n      [style.left]=\"_activeTabPosition.left\"\n      [style.width]=\"_activeTabPosition.width\"\n      [class.sapphire-segmented-control__glider--with-transition]=\"\n        _animationEnabled\n      \"\n      class=\"sapphire-segmented-control__glider\"\n      role=\"none\"\n    ></span>\n  </div>\n</div>\n<ng-container *ngTemplateOutlet=\"_getTabContentTemplate()\"></ng-container>\n"]}
|
|
235
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"segmented-tabs.component.js","sourceRoot":"","sources":["../../../../../src/lib/segmented-tabs/src/segmented-tabs.component.ts","../../../../../src/lib/segmented-tabs/src/segmented-tabs.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterContentInit,\n  AfterViewChecked,\n  AfterViewInit,\n  ChangeDetectorRef,\n  Component,\n  ContentChildren,\n  ElementRef,\n  EventEmitter,\n  forwardRef,\n  Input,\n  NgZone,\n  OnDestroy,\n  Output,\n  QueryList,\n  ViewChild,\n} from '@angular/core';\nimport {\n  BooleanInput,\n  coerceNumberProperty,\n  NumberInput,\n} from '@angular/cdk/coercion';\nimport { FocusOrigin } from '@angular/cdk/a11y';\n\nimport { CoerceBoolean } from '../../common/coerce-boolean.decorator';\nimport { SegmentedTabComponent } from './segmented-tab.component';\nimport { ViewEncapsulationProvider } from '../../common/sapphire-view-encapsulation';\n\n@Component({\n  selector: 'sp-segmented-tabs',\n  templateUrl: './segmented-tabs.component.html',\n  styleUrls: ['./segmented-tabs.component.scss'],\n  providers: [ViewEncapsulationProvider],\n  exportAs: 'spSegmentedTabs',\n  host: {\n    '[attr.align]': 'null',\n  },\n})\nexport class SegmentedTabsComponent\n  implements AfterContentInit, AfterViewInit, OnDestroy, AfterViewChecked\n{\n  @ViewChild('segmentedControl') segmentedControl!: ElementRef;\n\n  /** Size of tabs */\n  @Input()\n  size: 'sm' | 'md' | 'lg' = 'md';\n\n  /**\n   * Alignment of tabs' buttons group.\n   * @default 'left'\n   */\n  @Input()\n  align?: 'left' | 'center' | 'right' | 'stretch' = 'left';\n\n  /** Whether tabs are activated automatically on focus or manually. */\n  @Input()\n  keyboardActivation?: 'manual' | 'automatic' = 'automatic';\n\n  /** Whether the entire component should be disabled */\n  @Input()\n  @CoerceBoolean\n  disabled: BooleanInput;\n\n  /** Index of the selected tab */\n  @Input()\n  get selectedIndex(): number {\n    return this._selectedIndex || 0;\n  }\n  set selectedIndex(value: NumberInput) {\n    this._selectedIndex = coerceNumberProperty(value);\n    this._focusedIndex = this._selectedIndex;\n  }\n  private _selectedIndex?: number;\n  private _focusedIndex: number = 0;\n\n  @Output() readonly selectedIndexChange: EventEmitter<number> =\n    new EventEmitter<number>();\n\n  @ContentChildren(forwardRef(() => SegmentedTabComponent))\n  readonly tabs: QueryList<SegmentedTabComponent> = new QueryList();\n  private _tabIds: string[] = [];\n\n  private resizeObserver: ResizeObserver | undefined;\n  _gliderStyles = { left: '0px', width: '0px' };\n  _animationEnabled = false;\n  _focusVisible = false;\n\n  constructor(private zone: NgZone, private changeDetector: ChangeDetectorRef) {\n    this.skipAnimationFrame();\n  }\n\n  ngAfterContentInit(): void {\n    this._tabIds = this.tabs.map((tab) => tab.id);\n    this.tabs.changes.subscribe(() => {\n      this.onTabsChange();\n    });\n    if (this._selectedIndex === undefined) {\n      this.selectedIndex = this.findFocusableIndex(0, 'right') || 0;\n      this.selectedIndexChange.emit(this.selectedIndex);\n    }\n  }\n\n  ngAfterViewInit(): void {\n    // The value of \"_getTabContentTemplate\", which is bound in the\n    // template, changes after the view has been checked. This\n    // \"detectChanges()\" avoids the \"NG0100: Expression has\n    // changed after it was checked\" angular error.\n    //\n    // https://angular.io/errors/NG0100\n    this.setGliderStyles();\n    this.changeDetector.detectChanges();\n    this.resizeObserver = new ResizeObserver(() => {\n      // zonejs doesn't monkey-patch resize observer (yet)\n      this.zone.run(() => {\n        this.setGliderStyles();\n      });\n    });\n\n    this.resizeObserver.observe(this.segmentedControl.nativeElement);\n    this.selectedIndexChange.subscribe(() => {\n      this.setGliderStyles();\n    });\n  }\n\n  ngOnDestroy(): void {\n    this.resizeObserver?.unobserve(this.segmentedControl.nativeElement);\n    this.selectedIndexChange.unsubscribe();\n  }\n\n  ngAfterViewChecked(): void {\n    this.setGliderStyles();\n    this.changeDetector.detectChanges();\n  }\n\n  getFocusedTab() {\n    return this.tabs.toArray()[this._focusedIndex];\n  }\n\n  selectTab(selectedTab: SegmentedTabComponent) {\n    this.selectedIndex = this.tabs\n      .toArray()\n      .findIndex((tab) => tab === selectedTab);\n    this.selectedIndexChange.emit(this.selectedIndex);\n  }\n\n  isSelected(tab: SegmentedTabComponent): boolean {\n    return this.tabs?.toArray().indexOf(tab) === this.selectedIndex;\n  }\n\n  _getTabContentTemplate() {\n    return this.tabs.toArray()[this.selectedIndex]?._contentTemplate || null;\n  }\n\n  _onFocusChange(origin: FocusOrigin) {\n    this._focusVisible = origin === 'keyboard';\n    if (!origin) {\n      this._focusedIndex = this.selectedIndex;\n    }\n  }\n\n  _onKeyDown(event: KeyboardEvent) {\n    const focusNextValidTab = (\n      targetIndex: number,\n      direction: 'left' | 'right'\n    ) => {\n      if (this.tabs.toArray().every((tab) => tab.disabled)) return;\n      const index = this.findFocusableIndex(targetIndex, direction);\n      if (index === null) return;\n      const button = this.tabs.toArray()[index]._button?.nativeElement;\n      if (!button) return;\n      button.focus();\n      if (this.keyboardActivation === 'automatic') {\n        this.selectedIndex = index;\n        this.selectedIndexChange.emit(this.selectedIndex);\n      } else {\n        this._focusedIndex = index;\n      }\n    };\n    switch (event.key) {\n      case 'ArrowLeft': {\n        event.stopPropagation();\n        focusNextValidTab(this._focusedIndex - 1, 'left');\n        break;\n      }\n      case 'ArrowRight': {\n        event.stopPropagation();\n        focusNextValidTab(this._focusedIndex + 1, 'right');\n        break;\n      }\n      case 'Home': {\n        event.stopPropagation();\n        event.preventDefault();\n        focusNextValidTab(0, 'right');\n        break;\n      }\n      case 'End': {\n        event.stopPropagation();\n        event.preventDefault();\n        focusNextValidTab(this.tabs.length - 1, 'left');\n        break;\n      }\n      default:\n        return;\n    }\n  }\n\n  private setGliderStyles() {\n    const { offsetLeft, clientWidth } = this.getSelectedTabPosition();\n    this._gliderStyles = {\n      left: `${offsetLeft}px`,\n      width: `${clientWidth}px`,\n    };\n  }\n\n  private getSelectedTabPosition() {\n    const tab = this.tabs.toArray()[this.selectedIndex]?._button?.nativeElement;\n    return {\n      offsetLeft: tab?.offsetLeft || 0,\n      clientWidth: tab?.clientWidth || 0,\n    };\n  }\n\n  private onTabsChange() {\n    // Ensure the current tab doesn't change if tabs change\n    // and tab with currently selected id still exists\n    const previousTabId = this._tabIds[this.selectedIndex];\n    this._tabIds = this.tabs.map((tab) => tab.id);\n    if (this._tabIds.includes(previousTabId)) {\n      this.selectedIndex = this._tabIds.findIndex(\n        (tabId) => tabId === previousTabId\n      );\n    } else {\n      const maxIndex = this.tabs.length - 1;\n      if (this.selectedIndex > maxIndex) {\n        this.selectedIndex = maxIndex;\n        this.selectedIndexChange.emit(this.selectedIndex);\n      }\n    }\n    this.skipAnimationFrame();\n  }\n\n  private skipAnimationFrame() {\n    // We want to avoid animation on first render and whenever tab content changes\n    this._animationEnabled = false;\n    requestAnimationFrame(() => {\n      this._animationEnabled = true;\n    });\n  }\n\n  private findFocusableIndex(\n    index: number,\n    direction: 'left' | 'right'\n  ): number | null {\n    if (this.tabs.toArray().every((tab) => tab.disabled)) return null;\n    const constrainIndex = (index: number) => {\n      if (index >= this.tabs.length) {\n        return 0;\n      } else if (index < 0) {\n        return this.tabs.length - 1;\n      } else {\n        return index;\n      }\n    };\n    const constrainedIndex = constrainIndex(index);\n    if (this.tabs.toArray()[constrainedIndex].disabled) {\n      return this.findFocusableIndex(\n        constrainedIndex + (direction === 'left' ? -1 : 1),\n        direction\n      );\n    } else {\n      return constrainedIndex;\n    }\n  }\n}\n","<div\n  #segmentedControl\n  class=\"sapphire-segmented-control\"\n  [class.sapphire-segmented-control--lg]=\"size === 'lg'\"\n  [class.sapphire-segmented-control--sm]=\"size === 'sm'\"\n  [class.focus-visible]=\"_focusVisible\"\n  [class.sapphire-segmented-control--manual-keyboard-activation]=\"\n    keyboardActivation === 'manual'\n  \"\n  [ngClass]=\"\n    align === 'stretch'\n      ? 'sapphire-segmented-control--stretch'\n      : 'sapphire-segmented-control--align-' + align\n  \"\n>\n  <div\n    class=\"sapphire-segmented-control__button-container\"\n    role=\"tablist\"\n    aria-orientation=\"horizontal\"\n    cdkMonitorSubtreeFocus\n    (cdkFocusChange)=\"_onFocusChange($event)\"\n    (keydown)=\"_onKeyDown($event)\"\n  >\n    <ng-content select=\"sp-segmented-tab\"></ng-content>\n    <span\n      [style.left]=\"_gliderStyles.left\"\n      [style.width]=\"_gliderStyles.width\"\n      [class.sapphire-segmented-control__glider--with-transition]=\"\n        _animationEnabled\n      \"\n      class=\"sapphire-segmented-control__glider\"\n      role=\"none\"\n    ></span>\n  </div>\n</div>\n<ng-container *ngTemplateOutlet=\"_getTabContentTemplate()\"></ng-container>\n"]}
|
|
@@ -13,4 +13,5 @@ export * from './src/table-head.directive';
|
|
|
13
13
|
export * from './src/table-sort.directive';
|
|
14
14
|
export * from './src/table-sort-header.directive';
|
|
15
15
|
export * from './src/table.module';
|
|
16
|
-
|
|
16
|
+
export { CdkVirtualScrollViewportFixDirective } from './src/cdk-virtual-scroll-viewport-fix.directive';
|
|
17
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljX2FwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9saWIvdGFibGUvcHVibGljX2FwaS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSw0RUFBNEU7QUFDNUUsNkVBQTZFO0FBQzdFLHNEQUFzRDtBQUN0RCxXQUFXO0FBQ1gsY0FBYyx1QkFBdUIsQ0FBQztBQUN0QyxjQUFjLHVCQUF1QixDQUFDO0FBQ3RDLGNBQWMsMkJBQTJCLENBQUM7QUFDMUMsY0FBYyw0QkFBNEIsQ0FBQztBQUMzQyxjQUFjLDhCQUE4QixDQUFDO0FBQzdDLGNBQWMsNEJBQTRCLENBQUM7QUFDM0MsY0FBYyxpQ0FBaUMsQ0FBQztBQUNoRCxjQUFjLDRCQUE0QixDQUFDO0FBQzNDLGNBQWMsNEJBQTRCLENBQUM7QUFDM0MsY0FBYyxtQ0FBbUMsQ0FBQztBQUNsRCxjQUFjLG9CQUFvQixDQUFDO0FBQ25DLE9BQU8sRUFBRSxvQ0FBb0MsRUFBRSxNQUFNLGlEQUFpRCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLy8gV2UgaGF2ZSBzZXBhcmF0ZSBwdWJsaWNfYXBpLnRzIGZpbGUgZm9yIGVhY2ggYW5ndWxhciBtb2R1bGUsIHRvIGhhdmUgdGhlbVxuLy8gZXhwb3NlZCBhcyBzZXBhcmF0ZSBlbnRyeSBwb2ludHMuIFRoZSBidWlsZCBpcyBzdGlsbCBub3QgZW1pdHRpbmcgc2VwYXJhdGVcbi8vIGVudHJ5IHBvaW50cywgd2hpY2ggd2lsbCBiZSBhZGRyZXNzZWQgaW4gdGhpcyB0YXNrOlxuLy8gV0NDSi04MzhcbmV4cG9ydCAqIGZyb20gJy4vc3JjL3RhYmxlLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL3NyYy90YWJsZS5kaXJlY3RpdmUnO1xuZXhwb3J0ICogZnJvbSAnLi9zcmMvdGFibGUtcm93LmRpcmVjdGl2ZSc7XG5leHBvcnQgKiBmcm9tICcuL3NyYy90YWJsZS1jZWxsLmRpcmVjdGl2ZSc7XG5leHBvcnQgKiBmcm9tICcuL3NyYy90YWJsZS1mb290ZXIuZGlyZWN0aXZlJztcbmV4cG9ydCAqIGZyb20gJy4vc3JjL3RhYmxlLWJvZHkuZGlyZWN0aXZlJztcbmV4cG9ydCAqIGZyb20gJy4vc3JjL3RhYmxlLWhlYWQtY2VsbC5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9zcmMvdGFibGUtaGVhZC5kaXJlY3RpdmUnO1xuZXhwb3J0ICogZnJvbSAnLi9zcmMvdGFibGUtc29ydC5kaXJlY3RpdmUnO1xuZXhwb3J0ICogZnJvbSAnLi9zcmMvdGFibGUtc29ydC1oZWFkZXIuZGlyZWN0aXZlJztcbmV4cG9ydCAqIGZyb20gJy4vc3JjL3RhYmxlLm1vZHVsZSc7XG5leHBvcnQgeyBDZGtWaXJ0dWFsU2Nyb2xsVmlld3BvcnRGaXhEaXJlY3RpdmUgfSBmcm9tICcuL3NyYy9jZGstdmlydHVhbC1zY3JvbGwtdmlld3BvcnQtZml4LmRpcmVjdGl2ZSc7XG4iXX0=
|
|
@@ -0,0 +1,61 @@
|
|
|
1
|
+
import { Directive } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
import * as i1 from "@angular/cdk/scrolling";
|
|
4
|
+
/**
|
|
5
|
+
* A patch to fix the issue of sticky header position getting affected
|
|
6
|
+
* by transform, which is due to transform making the element
|
|
7
|
+
* a [containing block](https://developer.mozilla.org/en-US/docs/Web/CSS/Containing_block).
|
|
8
|
+
*
|
|
9
|
+
* It intercepts style assignments on the content wrapper, and converts
|
|
10
|
+
* transform: translateY(value) to top: value, to avoid making the content
|
|
11
|
+
* wrapper a containing block for sticky positioned children.
|
|
12
|
+
*
|
|
13
|
+
* @see
|
|
14
|
+
* https://github.com/angular/components/issues/14833
|
|
15
|
+
* https://github.com/angular/components/issues/18240
|
|
16
|
+
* https://github.com/angular/components/issues/21576
|
|
17
|
+
* https://github.com/angular/components/issues/28584
|
|
18
|
+
*/
|
|
19
|
+
export class CdkVirtualScrollViewportFixDirective {
|
|
20
|
+
constructor(cdkViewport) {
|
|
21
|
+
this.cdkViewport = cdkViewport;
|
|
22
|
+
}
|
|
23
|
+
ngOnInit() {
|
|
24
|
+
const contentWrapper = this.cdkViewport._contentWrapper.nativeElement;
|
|
25
|
+
/**
|
|
26
|
+
* Proxy to set `top: {value}px` instead of `transform: translateY({value}px)`
|
|
27
|
+
* whenever `style.transform` is set.
|
|
28
|
+
*/
|
|
29
|
+
const styleProxy = new Proxy(contentWrapper.style, {
|
|
30
|
+
set: (target, prop, value) => {
|
|
31
|
+
const top = prop === 'transform'
|
|
32
|
+
? value?.match(/translateY\((\d+)px\)/)?.[1]
|
|
33
|
+
: undefined;
|
|
34
|
+
if (top) {
|
|
35
|
+
target['top'] = `${top}px`;
|
|
36
|
+
return true;
|
|
37
|
+
}
|
|
38
|
+
target[prop] = value;
|
|
39
|
+
return true;
|
|
40
|
+
},
|
|
41
|
+
});
|
|
42
|
+
// Not directly setting style proxy to contentWrapper.style, to limit
|
|
43
|
+
// it to the styles set by CdkVirtualScrollViewport.
|
|
44
|
+
// i.e. only setting `top` instead of `transform`, when set by cdk viewport.
|
|
45
|
+
this.cdkViewport._contentWrapper.nativeElement = new Proxy(contentWrapper, {
|
|
46
|
+
get: (target, prop) => {
|
|
47
|
+
return prop === 'style' ? styleProxy : target[prop];
|
|
48
|
+
},
|
|
49
|
+
});
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
CdkVirtualScrollViewportFixDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.0", ngImport: i0, type: CdkVirtualScrollViewportFixDirective, deps: [{ token: i1.CdkVirtualScrollViewport }], target: i0.ɵɵFactoryTarget.Directive });
|
|
53
|
+
CdkVirtualScrollViewportFixDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "15.2.0", type: CdkVirtualScrollViewportFixDirective, isStandalone: true, selector: "cdk-virtual-scroll-viewport", ngImport: i0 });
|
|
54
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.0", ngImport: i0, type: CdkVirtualScrollViewportFixDirective, decorators: [{
|
|
55
|
+
type: Directive,
|
|
56
|
+
args: [{
|
|
57
|
+
selector: 'cdk-virtual-scroll-viewport',
|
|
58
|
+
standalone: true,
|
|
59
|
+
}]
|
|
60
|
+
}], ctorParameters: function () { return [{ type: i1.CdkVirtualScrollViewport }]; } });
|
|
61
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2RrLXZpcnR1YWwtc2Nyb2xsLXZpZXdwb3J0LWZpeC5kaXJlY3RpdmUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9zcmMvbGliL3RhYmxlL3NyYy9jZGstdmlydHVhbC1zY3JvbGwtdmlld3BvcnQtZml4LmRpcmVjdGl2ZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFVLE1BQU0sZUFBZSxDQUFDOzs7QUFHbEQ7Ozs7Ozs7Ozs7Ozs7O0dBY0c7QUFLSCxNQUFNLE9BQU8sb0NBQW9DO0lBQy9DLFlBQW9CLFdBQXFDO1FBQXJDLGdCQUFXLEdBQVgsV0FBVyxDQUEwQjtJQUFHLENBQUM7SUFFN0QsUUFBUTtRQUNOLE1BQU0sY0FBYyxHQUFHLElBQUksQ0FBQyxXQUFXLENBQUMsZUFBZSxDQUFDLGFBQWEsQ0FBQztRQUN0RTs7O1dBR0c7UUFDSCxNQUFNLFVBQVUsR0FBRyxJQUFJLEtBQUssQ0FBQyxjQUFjLENBQUMsS0FBSyxFQUFFO1lBQ2pELEdBQUcsRUFBRSxDQUFDLE1BQU0sRUFBRSxJQUFJLEVBQUUsS0FBSyxFQUFFLEVBQUU7Z0JBQzNCLE1BQU0sR0FBRyxHQUNQLElBQUksS0FBSyxXQUFXO29CQUNsQixDQUFDLENBQUMsS0FBSyxFQUFFLEtBQUssQ0FBQyx1QkFBdUIsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDO29CQUM1QyxDQUFDLENBQUMsU0FBUyxDQUFDO2dCQUNoQixJQUFJLEdBQUcsRUFBRTtvQkFDUCxNQUFNLENBQUMsS0FBSyxDQUFDLEdBQUcsR0FBRyxHQUFHLElBQUksQ0FBQztvQkFDM0IsT0FBTyxJQUFJLENBQUM7aUJBQ2I7Z0JBQ0EsTUFBYyxDQUFDLElBQUksQ0FBQyxHQUFHLEtBQUssQ0FBQztnQkFDOUIsT0FBTyxJQUFJLENBQUM7WUFDZCxDQUFDO1NBQ0YsQ0FBQyxDQUFDO1FBRUgscUVBQXFFO1FBQ3JFLG9EQUFvRDtRQUNwRCw0RUFBNEU7UUFDNUUsSUFBSSxDQUFDLFdBQVcsQ0FBQyxlQUFlLENBQUMsYUFBYSxHQUFHLElBQUksS0FBSyxDQUFDLGNBQWMsRUFBRTtZQUN6RSxHQUFHLEVBQUUsQ0FBQyxNQUFNLEVBQUUsSUFBSSxFQUFFLEVBQUU7Z0JBQ3BCLE9BQU8sSUFBSSxLQUFLLE9BQU8sQ0FBQyxDQUFDLENBQUMsVUFBVSxDQUFDLENBQUMsQ0FBRSxNQUFjLENBQUMsSUFBSSxDQUFDLENBQUM7WUFDL0QsQ0FBQztTQUNGLENBQUMsQ0FBQztJQUNMLENBQUM7O2lJQWhDVSxvQ0FBb0M7cUhBQXBDLG9DQUFvQzsyRkFBcEMsb0NBQW9DO2tCQUpoRCxTQUFTO21CQUFDO29CQUNULFFBQVEsRUFBRSw2QkFBNkI7b0JBQ3ZDLFVBQVUsRUFBRSxJQUFJO2lCQUNqQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IERpcmVjdGl2ZSwgT25Jbml0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBDZGtWaXJ0dWFsU2Nyb2xsVmlld3BvcnQgfSBmcm9tICdAYW5ndWxhci9jZGsvc2Nyb2xsaW5nJztcblxuLyoqXG4gKiBBIHBhdGNoIHRvIGZpeCB0aGUgaXNzdWUgb2Ygc3RpY2t5IGhlYWRlciBwb3NpdGlvbiBnZXR0aW5nIGFmZmVjdGVkXG4gKiBieSB0cmFuc2Zvcm0sIHdoaWNoIGlzIGR1ZSB0byB0cmFuc2Zvcm0gbWFraW5nIHRoZSBlbGVtZW50XG4gKiBhIFtjb250YWluaW5nIGJsb2NrXShodHRwczovL2RldmVsb3Blci5tb3ppbGxhLm9yZy9lbi1VUy9kb2NzL1dlYi9DU1MvQ29udGFpbmluZ19ibG9jaykuXG4gKlxuICogSXQgaW50ZXJjZXB0cyBzdHlsZSBhc3NpZ25tZW50cyBvbiB0aGUgY29udGVudCB3cmFwcGVyLCBhbmQgY29udmVydHNcbiAqIHRyYW5zZm9ybTogdHJhbnNsYXRlWSh2YWx1ZSkgdG8gdG9wOiB2YWx1ZSwgdG8gYXZvaWQgbWFraW5nIHRoZSBjb250ZW50XG4gKiB3cmFwcGVyIGEgY29udGFpbmluZyBibG9jayBmb3Igc3RpY2t5IHBvc2l0aW9uZWQgY2hpbGRyZW4uXG4gKlxuICogQHNlZVxuICogaHR0cHM6Ly9naXRodWIuY29tL2FuZ3VsYXIvY29tcG9uZW50cy9pc3N1ZXMvMTQ4MzNcbiAqIGh0dHBzOi8vZ2l0aHViLmNvbS9hbmd1bGFyL2NvbXBvbmVudHMvaXNzdWVzLzE4MjQwXG4gKiBodHRwczovL2dpdGh1Yi5jb20vYW5ndWxhci9jb21wb25lbnRzL2lzc3Vlcy8yMTU3NlxuICogaHR0cHM6Ly9naXRodWIuY29tL2FuZ3VsYXIvY29tcG9uZW50cy9pc3N1ZXMvMjg1ODRcbiAqL1xuQERpcmVjdGl2ZSh7XG4gIHNlbGVjdG9yOiAnY2RrLXZpcnR1YWwtc2Nyb2xsLXZpZXdwb3J0JyxcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbn0pXG5leHBvcnQgY2xhc3MgQ2RrVmlydHVhbFNjcm9sbFZpZXdwb3J0Rml4RGlyZWN0aXZlIGltcGxlbWVudHMgT25Jbml0IHtcbiAgY29uc3RydWN0b3IocHJpdmF0ZSBjZGtWaWV3cG9ydDogQ2RrVmlydHVhbFNjcm9sbFZpZXdwb3J0KSB7fVxuXG4gIG5nT25Jbml0KCkge1xuICAgIGNvbnN0IGNvbnRlbnRXcmFwcGVyID0gdGhpcy5jZGtWaWV3cG9ydC5fY29udGVudFdyYXBwZXIubmF0aXZlRWxlbWVudDtcbiAgICAvKipcbiAgICAgKiBQcm94eSB0byBzZXQgYHRvcDoge3ZhbHVlfXB4YCBpbnN0ZWFkIG9mIGB0cmFuc2Zvcm06IHRyYW5zbGF0ZVkoe3ZhbHVlfXB4KWBcbiAgICAgKiB3aGVuZXZlciBgc3R5bGUudHJhbnNmb3JtYCBpcyBzZXQuXG4gICAgICovXG4gICAgY29uc3Qgc3R5bGVQcm94eSA9IG5ldyBQcm94eShjb250ZW50V3JhcHBlci5zdHlsZSwge1xuICAgICAgc2V0OiAodGFyZ2V0LCBwcm9wLCB2YWx1ZSkgPT4ge1xuICAgICAgICBjb25zdCB0b3A6IHN0cmluZyB8IHVuZGVmaW5lZCA9XG4gICAgICAgICAgcHJvcCA9PT0gJ3RyYW5zZm9ybSdcbiAgICAgICAgICAgID8gdmFsdWU/Lm1hdGNoKC90cmFuc2xhdGVZXFwoKFxcZCspcHhcXCkvKT8uWzFdXG4gICAgICAgICAgICA6IHVuZGVmaW5lZDtcbiAgICAgICAgaWYgKHRvcCkge1xuICAgICAgICAgIHRhcmdldFsndG9wJ10gPSBgJHt0b3B9cHhgO1xuICAgICAgICAgIHJldHVybiB0cnVlO1xuICAgICAgICB9XG4gICAgICAgICh0YXJnZXQgYXMgYW55KVtwcm9wXSA9IHZhbHVlO1xuICAgICAgICByZXR1cm4gdHJ1ZTtcbiAgICAgIH0sXG4gICAgfSk7XG5cbiAgICAvLyBOb3QgZGlyZWN0bHkgc2V0dGluZyBzdHlsZSBwcm94eSB0byBjb250ZW50V3JhcHBlci5zdHlsZSwgdG8gbGltaXRcbiAgICAvLyBpdCB0byB0aGUgc3R5bGVzIHNldCBieSBDZGtWaXJ0dWFsU2Nyb2xsVmlld3BvcnQuXG4gICAgLy8gaS5lLiBvbmx5IHNldHRpbmcgYHRvcGAgaW5zdGVhZCBvZiBgdHJhbnNmb3JtYCwgd2hlbiBzZXQgYnkgY2RrIHZpZXdwb3J0LlxuICAgIHRoaXMuY2RrVmlld3BvcnQuX2NvbnRlbnRXcmFwcGVyLm5hdGl2ZUVsZW1lbnQgPSBuZXcgUHJveHkoY29udGVudFdyYXBwZXIsIHtcbiAgICAgIGdldDogKHRhcmdldCwgcHJvcCkgPT4ge1xuICAgICAgICByZXR1cm4gcHJvcCA9PT0gJ3N0eWxlJyA/IHN0eWxlUHJveHkgOiAodGFyZ2V0IGFzIGFueSlbcHJvcF07XG4gICAgICB9LFxuICAgIH0pO1xuICB9XG59XG4iXX0=
|
|
@@ -13,6 +13,7 @@ import { TruncatedWithTooltipDirective } from '../../tooltip/public_api';
|
|
|
13
13
|
import { UseComponentStyles } from '../../common/sapphire-view-encapsulation';
|
|
14
14
|
import { TableDirective } from './table.directive';
|
|
15
15
|
import { TableFooterDirective } from './table-footer.directive';
|
|
16
|
+
import { CdkVirtualScrollViewportFixDirective } from './cdk-virtual-scroll-viewport-fix.directive';
|
|
16
17
|
import * as i0 from "@angular/core";
|
|
17
18
|
export class SapphireTableModule {
|
|
18
19
|
}
|
|
@@ -29,6 +30,7 @@ SapphireTableModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", ver
|
|
|
29
30
|
TableSortHeaderDirective], imports: [CommonModule,
|
|
30
31
|
SapphireIconModule,
|
|
31
32
|
TruncatedWithTooltipDirective,
|
|
33
|
+
CdkVirtualScrollViewportFixDirective,
|
|
32
34
|
UseComponentStyles], exports: [TableComponent,
|
|
33
35
|
TableDirective,
|
|
34
36
|
TableBodyDirective,
|
|
@@ -38,7 +40,8 @@ SapphireTableModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", ver
|
|
|
38
40
|
TableHeadCellComponent,
|
|
39
41
|
TableCellDirective,
|
|
40
42
|
TableSortDirective,
|
|
41
|
-
TableSortHeaderDirective
|
|
43
|
+
TableSortHeaderDirective,
|
|
44
|
+
CdkVirtualScrollViewportFixDirective] });
|
|
42
45
|
SapphireTableModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "15.2.0", ngImport: i0, type: SapphireTableModule, imports: [CommonModule,
|
|
43
46
|
SapphireIconModule] });
|
|
44
47
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.0", ngImport: i0, type: SapphireTableModule, decorators: [{
|
|
@@ -60,6 +63,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.0", ngImpor
|
|
|
60
63
|
CommonModule,
|
|
61
64
|
SapphireIconModule,
|
|
62
65
|
TruncatedWithTooltipDirective,
|
|
66
|
+
CdkVirtualScrollViewportFixDirective,
|
|
63
67
|
UseComponentStyles,
|
|
64
68
|
],
|
|
65
69
|
exports: [
|
|
@@ -73,7 +77,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.0", ngImpor
|
|
|
73
77
|
TableCellDirective,
|
|
74
78
|
TableSortDirective,
|
|
75
79
|
TableSortHeaderDirective,
|
|
80
|
+
CdkVirtualScrollViewportFixDirective,
|
|
76
81
|
],
|
|
77
82
|
}]
|
|
78
83
|
}] });
|
|
79
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
84
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGFibGUubW9kdWxlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi90YWJsZS9zcmMvdGFibGUubW9kdWxlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDekMsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSxtQkFBbUIsQ0FBQztBQUNuRCxPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQztBQUM1RCxPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQztBQUM1RCxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSx1QkFBdUIsQ0FBQztBQUMxRCxPQUFPLEVBQUUsc0JBQXNCLEVBQUUsTUFBTSw2QkFBNkIsQ0FBQztBQUNyRSxPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQztBQUM1RCxPQUFPLEVBQUUsd0JBQXdCLEVBQUUsTUFBTSwrQkFBK0IsQ0FBQztBQUN6RSxPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQztBQUM1RCxPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSx1QkFBdUIsQ0FBQztBQUMzRCxPQUFPLEVBQUUsNkJBQTZCLEVBQUUsTUFBTSwwQkFBMEIsQ0FBQztBQUN6RSxPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSwwQ0FBMEMsQ0FBQztBQUM5RSxPQUFPLEVBQUUsY0FBYyxFQUFFLE1BQU0sbUJBQW1CLENBQUM7QUFDbkQsT0FBTyxFQUFFLG9CQUFvQixFQUFFLE1BQU0sMEJBQTBCLENBQUM7QUFDaEUsT0FBTyxFQUFFLG9DQUFvQyxFQUFFLE1BQU0sNkNBQTZDLENBQUM7O0FBb0NuRyxNQUFNLE9BQU8sbUJBQW1COztnSEFBbkIsbUJBQW1CO2lIQUFuQixtQkFBbUIsaUJBaEM1QixjQUFjO1FBQ2QsY0FBYztRQUNkLGtCQUFrQjtRQUNsQixvQkFBb0I7UUFDcEIsa0JBQWtCO1FBQ2xCLGlCQUFpQjtRQUNqQixzQkFBc0I7UUFDdEIsa0JBQWtCO1FBQ2xCLGtCQUFrQjtRQUNsQix3QkFBd0IsYUFHeEIsWUFBWTtRQUNaLGtCQUFrQjtRQUNsQiw2QkFBNkI7UUFDN0Isb0NBQW9DO1FBQ3BDLGtCQUFrQixhQUdsQixjQUFjO1FBQ2QsY0FBYztRQUNkLGtCQUFrQjtRQUNsQixvQkFBb0I7UUFDcEIsa0JBQWtCO1FBQ2xCLGlCQUFpQjtRQUNqQixzQkFBc0I7UUFDdEIsa0JBQWtCO1FBQ2xCLGtCQUFrQjtRQUNsQix3QkFBd0I7UUFDeEIsb0NBQW9DO2lIQUczQixtQkFBbUIsWUFwQjVCLFlBQVk7UUFDWixrQkFBa0I7MkZBbUJULG1CQUFtQjtrQkFsQy9CLFFBQVE7bUJBQUM7b0JBQ1IsWUFBWSxFQUFFO3dCQUNaLGNBQWM7d0JBQ2QsY0FBYzt3QkFDZCxrQkFBa0I7d0JBQ2xCLG9CQUFvQjt3QkFDcEIsa0JBQWtCO3dCQUNsQixpQkFBaUI7d0JBQ2pCLHNCQUFzQjt3QkFDdEIsa0JBQWtCO3dCQUNsQixrQkFBa0I7d0JBQ2xCLHdCQUF3QjtxQkFDekI7b0JBQ0QsT0FBTyxFQUFFO3dCQUNQLFlBQVk7d0JBQ1osa0JBQWtCO3dCQUNsQiw2QkFBNkI7d0JBQzdCLG9DQUFvQzt3QkFDcEMsa0JBQWtCO3FCQUNuQjtvQkFDRCxPQUFPLEVBQUU7d0JBQ1AsY0FBYzt3QkFDZCxjQUFjO3dCQUNkLGtCQUFrQjt3QkFDbEIsb0JBQW9CO3dCQUNwQixrQkFBa0I7d0JBQ2xCLGlCQUFpQjt3QkFDakIsc0JBQXNCO3dCQUN0QixrQkFBa0I7d0JBQ2xCLGtCQUFrQjt3QkFDbEIsd0JBQXdCO3dCQUN4QixvQ0FBb0M7cUJBQ3JDO2lCQUNGIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgTmdNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQgeyBUYWJsZUNvbXBvbmVudCB9IGZyb20gJy4vdGFibGUuY29tcG9uZW50JztcbmltcG9ydCB7IFRhYmxlQm9keURpcmVjdGl2ZSB9IGZyb20gJy4vdGFibGUtYm9keS5kaXJlY3RpdmUnO1xuaW1wb3J0IHsgVGFibGVIZWFkRGlyZWN0aXZlIH0gZnJvbSAnLi90YWJsZS1oZWFkLmRpcmVjdGl2ZSc7XG5pbXBvcnQgeyBUYWJsZVJvd0RpcmVjdGl2ZSB9IGZyb20gJy4vdGFibGUtcm93LmRpcmVjdGl2ZSc7XG5pbXBvcnQgeyBUYWJsZUhlYWRDZWxsQ29tcG9uZW50IH0gZnJvbSAnLi90YWJsZS1oZWFkLWNlbGwuY29tcG9uZW50JztcbmltcG9ydCB7IFRhYmxlQ2VsbERpcmVjdGl2ZSB9IGZyb20gJy4vdGFibGUtY2VsbC5kaXJlY3RpdmUnO1xuaW1wb3J0IHsgVGFibGVTb3J0SGVhZGVyRGlyZWN0aXZlIH0gZnJvbSAnLi90YWJsZS1zb3J0LWhlYWRlci5kaXJlY3RpdmUnO1xuaW1wb3J0IHsgVGFibGVTb3J0RGlyZWN0aXZlIH0gZnJvbSAnLi90YWJsZS1zb3J0LmRpcmVjdGl2ZSc7XG5pbXBvcnQgeyBTYXBwaGlyZUljb25Nb2R1bGUgfSBmcm9tICcuLi8uLi9pY29uL3B1YmxpY19hcGknO1xuaW1wb3J0IHsgVHJ1bmNhdGVkV2l0aFRvb2x0aXBEaXJlY3RpdmUgfSBmcm9tICcuLi8uLi90b29sdGlwL3B1YmxpY19hcGknO1xuaW1wb3J0IHsgVXNlQ29tcG9uZW50U3R5bGVzIH0gZnJvbSAnLi4vLi4vY29tbW9uL3NhcHBoaXJlLXZpZXctZW5jYXBzdWxhdGlvbic7XG5pbXBvcnQgeyBUYWJsZURpcmVjdGl2ZSB9IGZyb20gJy4vdGFibGUuZGlyZWN0aXZlJztcbmltcG9ydCB7IFRhYmxlRm9vdGVyRGlyZWN0aXZlIH0gZnJvbSAnLi90YWJsZS1mb290ZXIuZGlyZWN0aXZlJztcbmltcG9ydCB7IENka1ZpcnR1YWxTY3JvbGxWaWV3cG9ydEZpeERpcmVjdGl2ZSB9IGZyb20gJy4vY2RrLXZpcnR1YWwtc2Nyb2xsLXZpZXdwb3J0LWZpeC5kaXJlY3RpdmUnO1xuXG5ATmdNb2R1bGUoe1xuICBkZWNsYXJhdGlvbnM6IFtcbiAgICBUYWJsZUNvbXBvbmVudCxcbiAgICBUYWJsZURpcmVjdGl2ZSxcbiAgICBUYWJsZUJvZHlEaXJlY3RpdmUsXG4gICAgVGFibGVGb290ZXJEaXJlY3RpdmUsXG4gICAgVGFibGVIZWFkRGlyZWN0aXZlLFxuICAgIFRhYmxlUm93RGlyZWN0aXZlLFxuICAgIFRhYmxlSGVhZENlbGxDb21wb25lbnQsXG4gICAgVGFibGVDZWxsRGlyZWN0aXZlLFxuICAgIFRhYmxlU29ydERpcmVjdGl2ZSxcbiAgICBUYWJsZVNvcnRIZWFkZXJEaXJlY3RpdmUsXG4gIF0sXG4gIGltcG9ydHM6IFtcbiAgICBDb21tb25Nb2R1bGUsXG4gICAgU2FwcGhpcmVJY29uTW9kdWxlLFxuICAgIFRydW5jYXRlZFdpdGhUb29sdGlwRGlyZWN0aXZlLFxuICAgIENka1ZpcnR1YWxTY3JvbGxWaWV3cG9ydEZpeERpcmVjdGl2ZSxcbiAgICBVc2VDb21wb25lbnRTdHlsZXMsXG4gIF0sXG4gIGV4cG9ydHM6IFtcbiAgICBUYWJsZUNvbXBvbmVudCxcbiAgICBUYWJsZURpcmVjdGl2ZSxcbiAgICBUYWJsZUJvZHlEaXJlY3RpdmUsXG4gICAgVGFibGVGb290ZXJEaXJlY3RpdmUsXG4gICAgVGFibGVIZWFkRGlyZWN0aXZlLFxuICAgIFRhYmxlUm93RGlyZWN0aXZlLFxuICAgIFRhYmxlSGVhZENlbGxDb21wb25lbnQsXG4gICAgVGFibGVDZWxsRGlyZWN0aXZlLFxuICAgIFRhYmxlU29ydERpcmVjdGl2ZSxcbiAgICBUYWJsZVNvcnRIZWFkZXJEaXJlY3RpdmUsXG4gICAgQ2RrVmlydHVhbFNjcm9sbFZpZXdwb3J0Rml4RGlyZWN0aXZlLFxuICBdLFxufSlcbmV4cG9ydCBjbGFzcyBTYXBwaGlyZVRhYmxlTW9kdWxlIHt9XG4iXX0=
|