@rivet-health/design-system 7.0.0 → 7.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2020/lib/input/alphabetic-field/alphabetic-field.component.mjs +48 -0
- package/esm2020/lib/input/currency-field/currency-field.component.mjs +22 -0
- package/esm2020/lib/input/number-field/number-field.component.mjs +95 -0
- package/esm2020/lib/input/percentage-field/percentage-field.component.mjs +19 -0
- package/esm2020/lib/input/single-select/single-select.component.mjs +41 -3
- package/esm2020/lib/input/text-field/text-field.component.mjs +100 -0
- package/esm2020/lib/riv.module.mjs +28 -3
- package/esm2020/public-api.mjs +6 -1
- package/fesm2015/rivet-health-design-system.mjs +396 -81
- package/fesm2015/rivet-health-design-system.mjs.map +1 -1
- package/fesm2020/rivet-health-design-system.mjs +401 -81
- package/fesm2020/rivet-health-design-system.mjs.map +1 -1
- package/lib/input/alphabetic-field/alphabetic-field.component.d.ts +16 -0
- package/lib/input/currency-field/currency-field.component.d.ts +8 -0
- package/lib/input/number-field/number-field.component.d.ts +28 -0
- package/lib/input/percentage-field/percentage-field.component.d.ts +7 -0
- package/lib/input/single-select/single-select.component.d.ts +6 -1
- package/lib/input/text-field/text-field.component.d.ts +31 -0
- package/lib/riv.module.d.ts +55 -50
- package/package.json +1 -1
- package/public-api.d.ts +5 -0
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output, } from '@angular/core';
|
|
2
|
+
import { TextFieldComponent } from '../text-field/text-field.component';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "../input-label/input-label.component";
|
|
5
|
+
export class AlphabeticFieldComponent extends TextFieldComponent {
|
|
6
|
+
constructor() {
|
|
7
|
+
super(...arguments);
|
|
8
|
+
this.valueChange = new EventEmitter();
|
|
9
|
+
}
|
|
10
|
+
set value(v) {
|
|
11
|
+
if (AlphabeticFieldComponent.clean(v, this.casing) ===
|
|
12
|
+
AlphabeticFieldComponent.clean(this._text, this.casing))
|
|
13
|
+
return;
|
|
14
|
+
this._text = AlphabeticFieldComponent.format(v, this.casing);
|
|
15
|
+
}
|
|
16
|
+
set text(t) {
|
|
17
|
+
this._text = t;
|
|
18
|
+
this.valueChange.emit(AlphabeticFieldComponent.clean(t, this.casing));
|
|
19
|
+
}
|
|
20
|
+
get text() {
|
|
21
|
+
return this._text;
|
|
22
|
+
}
|
|
23
|
+
blur() {
|
|
24
|
+
this._text = AlphabeticFieldComponent.format(this._text, this.casing);
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
AlphabeticFieldComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: AlphabeticFieldComponent, deps: null, target: i0.ɵɵFactoryTarget.Component });
|
|
28
|
+
AlphabeticFieldComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: AlphabeticFieldComponent, selector: "riv-alphabetic-field", inputs: { value: "value" }, outputs: { valueChange: "valueChange" }, usesInheritance: true, ngImport: i0, template: "<riv-input-label\n [label]=\"label\"\n [help]=\"help\"\n [required]=\"required\"\n [labelActionText]=\"labelActionText\"\n (labelAction)=\"labelAction.emit($event)\"\n>\n <input\n #input\n [placeholder]=\"placeholder\"\n [type]=\"type\"\n [required]=\"required\"\n [disabled]=\"disabled\"\n [readonly]=\"readonly\"\n [value]=\"text\"\n (input)=\"text = input.value\"\n [attr.maxlength]=\"maxLength\"\n [attr.minlength]=\"minLength\"\n [class.warning]=\"state === 'warning'\"\n [class.error]=\"state === 'error'\"\n autocomplete=\"off\"\n autocorrect=\"off\"\n autocapitalize=\"off\"\n spellcheck=\"false\"\n (blur)=\"blur()\"\n />\n</riv-input-label>\n", styles: ["input{width:100%;border:var(--border-width) solid var(--border-light);border-radius:var(--border-radius-small);font:var(--input-medium);color:var(--type-light-high-contrast);padding:var(--size-small)}input::placeholder{color:var(--type-light-disabled)}input:disabled{color:var(--type-light-disabled);background-color:var(--surface-light-1)}input:read-only{background-color:var(--surface-light-1)}input:focus{outline:none;border:var(--border-width) solid var(--purp-60)}input.warning{border-color:var(--surface-dark-caution)}input.error{border-color:var(--surface-dark-danger)}\n"], dependencies: [{ kind: "component", type: i1.InputLabelComponent, selector: "riv-input-label", inputs: ["label", "help", "required", "labelActionText"], outputs: ["labelAction"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
29
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: AlphabeticFieldComponent, decorators: [{
|
|
30
|
+
type: Component,
|
|
31
|
+
args: [{ selector: 'riv-alphabetic-field', changeDetection: ChangeDetectionStrategy.OnPush, template: "<riv-input-label\n [label]=\"label\"\n [help]=\"help\"\n [required]=\"required\"\n [labelActionText]=\"labelActionText\"\n (labelAction)=\"labelAction.emit($event)\"\n>\n <input\n #input\n [placeholder]=\"placeholder\"\n [type]=\"type\"\n [required]=\"required\"\n [disabled]=\"disabled\"\n [readonly]=\"readonly\"\n [value]=\"text\"\n (input)=\"text = input.value\"\n [attr.maxlength]=\"maxLength\"\n [attr.minlength]=\"minLength\"\n [class.warning]=\"state === 'warning'\"\n [class.error]=\"state === 'error'\"\n autocomplete=\"off\"\n autocorrect=\"off\"\n autocapitalize=\"off\"\n spellcheck=\"false\"\n (blur)=\"blur()\"\n />\n</riv-input-label>\n", styles: ["input{width:100%;border:var(--border-width) solid var(--border-light);border-radius:var(--border-radius-small);font:var(--input-medium);color:var(--type-light-high-contrast);padding:var(--size-small)}input::placeholder{color:var(--type-light-disabled)}input:disabled{color:var(--type-light-disabled);background-color:var(--surface-light-1)}input:read-only{background-color:var(--surface-light-1)}input:focus{outline:none;border:var(--border-width) solid var(--purp-60)}input.warning{border-color:var(--surface-dark-caution)}input.error{border-color:var(--surface-dark-danger)}\n"] }]
|
|
32
|
+
}], propDecorators: { value: [{
|
|
33
|
+
type: Input
|
|
34
|
+
}], valueChange: [{
|
|
35
|
+
type: Output
|
|
36
|
+
}] } });
|
|
37
|
+
(function (AlphabeticFieldComponent) {
|
|
38
|
+
function clean(v, casing) {
|
|
39
|
+
const cleaned = v.replaceAll(/[^A-Za-z]/g, '');
|
|
40
|
+
return TextFieldComponent.clean(cleaned, casing);
|
|
41
|
+
}
|
|
42
|
+
AlphabeticFieldComponent.clean = clean;
|
|
43
|
+
function format(v, casing) {
|
|
44
|
+
return clean(v, casing);
|
|
45
|
+
}
|
|
46
|
+
AlphabeticFieldComponent.format = format;
|
|
47
|
+
})(AlphabeticFieldComponent || (AlphabeticFieldComponent = {}));
|
|
48
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYWxwaGFiZXRpYy1maWVsZC5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9yaXYvc3JjL2xpYi9pbnB1dC9hbHBoYWJldGljLWZpZWxkL2FscGhhYmV0aWMtZmllbGQuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvcml2L3NyYy9saWIvaW5wdXQvdGV4dC1maWVsZC90ZXh0LWZpZWxkLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFDTCx1QkFBdUIsRUFDdkIsU0FBUyxFQUNULFlBQVksRUFDWixLQUFLLEVBQ0wsTUFBTSxHQUNQLE1BQU0sZUFBZSxDQUFDO0FBQ3ZCLE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxNQUFNLG9DQUFvQyxDQUFDOzs7QUFReEUsTUFBTSxPQUFPLHdCQUF5QixTQUFRLGtCQUFrQjtJQU5oRTs7UUFrQlcsZ0JBQVcsR0FBRyxJQUFJLFlBQVksRUFBVSxDQUFDO0tBYW5EO0lBeEJDLElBQ2EsS0FBSyxDQUFDLENBQVM7UUFDMUIsSUFDRSx3QkFBd0IsQ0FBQyxLQUFLLENBQUMsQ0FBQyxFQUFFLElBQUksQ0FBQyxNQUFNLENBQUM7WUFDOUMsd0JBQXdCLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxLQUFLLEVBQUUsSUFBSSxDQUFDLE1BQU0sQ0FBQztZQUV2RCxPQUFPO1FBQ1QsSUFBSSxDQUFDLEtBQUssR0FBRyx3QkFBd0IsQ0FBQyxNQUFNLENBQUMsQ0FBQyxFQUFFLElBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQztJQUMvRCxDQUFDO0lBS0QsSUFBYSxJQUFJLENBQUMsQ0FBUztRQUN6QixJQUFJLENBQUMsS0FBSyxHQUFHLENBQUMsQ0FBQztRQUNmLElBQUksQ0FBQyxXQUFXLENBQUMsSUFBSSxDQUFDLHdCQUF3QixDQUFDLEtBQUssQ0FBQyxDQUFDLEVBQUUsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUM7SUFDeEUsQ0FBQztJQUNELElBQWEsSUFBSTtRQUNmLE9BQU8sSUFBSSxDQUFDLEtBQUssQ0FBQztJQUNwQixDQUFDO0lBRVEsSUFBSTtRQUNYLElBQUksQ0FBQyxLQUFLLEdBQUcsd0JBQXdCLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxLQUFLLEVBQUUsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDO0lBQ3hFLENBQUM7O3FIQXhCVSx3QkFBd0I7eUdBQXhCLHdCQUF3Qix3SkNmckMsd3NCQTJCQTsyRkRaYSx3QkFBd0I7a0JBTnBDLFNBQVM7K0JBQ0Usc0JBQXNCLG1CQUdmLHVCQUF1QixDQUFDLE1BQU07OEJBSWxDLEtBQUs7c0JBRGpCLEtBQUs7Z0JBV0csV0FBVztzQkFEbkIsTUFBTTs7QUFnQlQsV0FBaUIsd0JBQXdCO0lBQ3ZDLFNBQWdCLEtBQUssQ0FBQyxDQUFTLEVBQUUsTUFBaUM7UUFDaEUsTUFBTSxPQUFPLEdBQUcsQ0FBQyxDQUFDLFVBQVUsQ0FBQyxZQUFZLEVBQUUsRUFBRSxDQUFDLENBQUM7UUFDL0MsT0FBTyxrQkFBa0IsQ0FBQyxLQUFLLENBQUMsT0FBTyxFQUFFLE1BQU0sQ0FBQyxDQUFDO0lBQ25ELENBQUM7SUFIZSw4QkFBSyxRQUdwQixDQUFBO0lBRUQsU0FBZ0IsTUFBTSxDQUFDLENBQVMsRUFBRSxNQUFpQztRQUNqRSxPQUFPLEtBQUssQ0FBQyxDQUFDLEVBQUUsTUFBTSxDQUFDLENBQUM7SUFDMUIsQ0FBQztJQUZlLCtCQUFNLFNBRXJCLENBQUE7QUFDSCxDQUFDLEVBVGdCLHdCQUF3QixLQUF4Qix3QkFBd0IsUUFTeEMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge1xuICBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneSxcbiAgQ29tcG9uZW50LFxuICBFdmVudEVtaXR0ZXIsXG4gIElucHV0LFxuICBPdXRwdXQsXG59IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgVGV4dEZpZWxkQ29tcG9uZW50IH0gZnJvbSAnLi4vdGV4dC1maWVsZC90ZXh0LWZpZWxkLmNvbXBvbmVudCc7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ3Jpdi1hbHBoYWJldGljLWZpZWxkJyxcbiAgdGVtcGxhdGVVcmw6ICcuLi90ZXh0LWZpZWxkL3RleHQtZmllbGQuY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZVVybHM6IFsnLi4vdGV4dC1maWVsZC90ZXh0LWZpZWxkLmNvbXBvbmVudC5jc3MnXSxcbiAgY2hhbmdlRGV0ZWN0aW9uOiBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneS5PblB1c2gsXG59KVxuZXhwb3J0IGNsYXNzIEFscGhhYmV0aWNGaWVsZENvbXBvbmVudCBleHRlbmRzIFRleHRGaWVsZENvbXBvbmVudCB7XG4gIEBJbnB1dCgpXG4gIG92ZXJyaWRlIHNldCB2YWx1ZSh2OiBzdHJpbmcpIHtcbiAgICBpZiAoXG4gICAgICBBbHBoYWJldGljRmllbGRDb21wb25lbnQuY2xlYW4odiwgdGhpcy5jYXNpbmcpID09PVxuICAgICAgQWxwaGFiZXRpY0ZpZWxkQ29tcG9uZW50LmNsZWFuKHRoaXMuX3RleHQsIHRoaXMuY2FzaW5nKVxuICAgIClcbiAgICAgIHJldHVybjtcbiAgICB0aGlzLl90ZXh0ID0gQWxwaGFiZXRpY0ZpZWxkQ29tcG9uZW50LmZvcm1hdCh2LCB0aGlzLmNhc2luZyk7XG4gIH1cblxuICBAT3V0cHV0KClcbiAgb3ZlcnJpZGUgdmFsdWVDaGFuZ2UgPSBuZXcgRXZlbnRFbWl0dGVyPHN0cmluZz4oKTtcblxuICBvdmVycmlkZSBzZXQgdGV4dCh0OiBzdHJpbmcpIHtcbiAgICB0aGlzLl90ZXh0ID0gdDtcbiAgICB0aGlzLnZhbHVlQ2hhbmdlLmVtaXQoQWxwaGFiZXRpY0ZpZWxkQ29tcG9uZW50LmNsZWFuKHQsIHRoaXMuY2FzaW5nKSk7XG4gIH1cbiAgb3ZlcnJpZGUgZ2V0IHRleHQoKTogc3RyaW5nIHtcbiAgICByZXR1cm4gdGhpcy5fdGV4dDtcbiAgfVxuXG4gIG92ZXJyaWRlIGJsdXIoKSB7XG4gICAgdGhpcy5fdGV4dCA9IEFscGhhYmV0aWNGaWVsZENvbXBvbmVudC5mb3JtYXQodGhpcy5fdGV4dCwgdGhpcy5jYXNpbmcpO1xuICB9XG59XG5cbmV4cG9ydCBuYW1lc3BhY2UgQWxwaGFiZXRpY0ZpZWxkQ29tcG9uZW50IHtcbiAgZXhwb3J0IGZ1bmN0aW9uIGNsZWFuKHY6IHN0cmluZywgY2FzaW5nOiBUZXh0RmllbGRDb21wb25lbnQuQ2FzaW5nKTogc3RyaW5nIHtcbiAgICBjb25zdCBjbGVhbmVkID0gdi5yZXBsYWNlQWxsKC9bXkEtWmEtel0vZywgJycpO1xuICAgIHJldHVybiBUZXh0RmllbGRDb21wb25lbnQuY2xlYW4oY2xlYW5lZCwgY2FzaW5nKTtcbiAgfVxuXG4gIGV4cG9ydCBmdW5jdGlvbiBmb3JtYXQodjogc3RyaW5nLCBjYXNpbmc6IFRleHRGaWVsZENvbXBvbmVudC5DYXNpbmcpOiBzdHJpbmcge1xuICAgIHJldHVybiBjbGVhbih2LCBjYXNpbmcpO1xuICB9XG59XG4iLCI8cml2LWlucHV0LWxhYmVsXG4gIFtsYWJlbF09XCJsYWJlbFwiXG4gIFtoZWxwXT1cImhlbHBcIlxuICBbcmVxdWlyZWRdPVwicmVxdWlyZWRcIlxuICBbbGFiZWxBY3Rpb25UZXh0XT1cImxhYmVsQWN0aW9uVGV4dFwiXG4gIChsYWJlbEFjdGlvbik9XCJsYWJlbEFjdGlvbi5lbWl0KCRldmVudClcIlxuPlxuICA8aW5wdXRcbiAgICAjaW5wdXRcbiAgICBbcGxhY2Vob2xkZXJdPVwicGxhY2Vob2xkZXJcIlxuICAgIFt0eXBlXT1cInR5cGVcIlxuICAgIFtyZXF1aXJlZF09XCJyZXF1aXJlZFwiXG4gICAgW2Rpc2FibGVkXT1cImRpc2FibGVkXCJcbiAgICBbcmVhZG9ubHldPVwicmVhZG9ubHlcIlxuICAgIFt2YWx1ZV09XCJ0ZXh0XCJcbiAgICAoaW5wdXQpPVwidGV4dCA9IGlucHV0LnZhbHVlXCJcbiAgICBbYXR0ci5tYXhsZW5ndGhdPVwibWF4TGVuZ3RoXCJcbiAgICBbYXR0ci5taW5sZW5ndGhdPVwibWluTGVuZ3RoXCJcbiAgICBbY2xhc3Mud2FybmluZ109XCJzdGF0ZSA9PT0gJ3dhcm5pbmcnXCJcbiAgICBbY2xhc3MuZXJyb3JdPVwic3RhdGUgPT09ICdlcnJvcidcIlxuICAgIGF1dG9jb21wbGV0ZT1cIm9mZlwiXG4gICAgYXV0b2NvcnJlY3Q9XCJvZmZcIlxuICAgIGF1dG9jYXBpdGFsaXplPVwib2ZmXCJcbiAgICBzcGVsbGNoZWNrPVwiZmFsc2VcIlxuICAgIChibHVyKT1cImJsdXIoKVwiXG4gIC8+XG48L3Jpdi1pbnB1dC1sYWJlbD5cbiJdfQ==
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, Input } from '@angular/core';
|
|
2
|
+
import { NumberFieldComponent } from '../number-field/number-field.component';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "../input-label/input-label.component";
|
|
5
|
+
export class CurrencyFieldComponent extends NumberFieldComponent {
|
|
6
|
+
constructor() {
|
|
7
|
+
super(...arguments);
|
|
8
|
+
this.placeholder = '0.00';
|
|
9
|
+
this.maximumFractionDigits = 2;
|
|
10
|
+
}
|
|
11
|
+
}
|
|
12
|
+
CurrencyFieldComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: CurrencyFieldComponent, deps: null, target: i0.ɵɵFactoryTarget.Component });
|
|
13
|
+
CurrencyFieldComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: CurrencyFieldComponent, selector: "riv-currency-field", inputs: { placeholder: "placeholder", maximumFractionDigits: "maximumFractionDigits" }, usesInheritance: true, ngImport: i0, template: "<riv-input-label\n [label]=\"label\"\n [help]=\"help\"\n [required]=\"required\"\n [labelActionText]=\"labelActionText\"\n (labelAction)=\"labelAction.emit($event)\"\n>\n <span class=\"prefix\">$</span>\n <input\n #input\n [placeholder]=\"placeholder\"\n type=\"text\"\n [required]=\"required\"\n [disabled]=\"disabled\"\n [readonly]=\"readonly\"\n [value]=\"text\"\n (input)=\"text = input.value\"\n [class.warning]=\"state === 'warning'\"\n [class.error]=\"state === 'error'\"\n [style.textAlign]=\"align\"\n autocomplete=\"off\"\n autocorrect=\"off\"\n autocapitalize=\"off\"\n spellcheck=\"false\"\n (blur)=\"blur()\"\n />\n</riv-input-label>\n", styles: [":host{position:relative}.prefix{position:absolute;font:var(--input-medium);padding:var(--size-small)}input{width:100%;border:var(--border-width) solid var(--border-light);border-radius:var(--border-radius-small);font:var(--input-medium);color:var(--type-light-high-contrast);padding:var(--size-small) var(--size-small) var(--size-small) var(--size-xlarge)}input::placeholder{color:var(--type-light-disabled)}input:disabled{color:var(--type-light-disabled);background-color:var(--surface-light-1)}input:read-only{background-color:var(--surface-light-1)}input:focus{outline:none;border:var(--border-width) solid var(--purp-60)}input.warning{border-color:var(--surface-dark-caution)}input.error{border-color:var(--surface-dark-danger)}\n"], dependencies: [{ kind: "component", type: i1.InputLabelComponent, selector: "riv-input-label", inputs: ["label", "help", "required", "labelActionText"], outputs: ["labelAction"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
14
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: CurrencyFieldComponent, decorators: [{
|
|
15
|
+
type: Component,
|
|
16
|
+
args: [{ selector: 'riv-currency-field', changeDetection: ChangeDetectionStrategy.OnPush, template: "<riv-input-label\n [label]=\"label\"\n [help]=\"help\"\n [required]=\"required\"\n [labelActionText]=\"labelActionText\"\n (labelAction)=\"labelAction.emit($event)\"\n>\n <span class=\"prefix\">$</span>\n <input\n #input\n [placeholder]=\"placeholder\"\n type=\"text\"\n [required]=\"required\"\n [disabled]=\"disabled\"\n [readonly]=\"readonly\"\n [value]=\"text\"\n (input)=\"text = input.value\"\n [class.warning]=\"state === 'warning'\"\n [class.error]=\"state === 'error'\"\n [style.textAlign]=\"align\"\n autocomplete=\"off\"\n autocorrect=\"off\"\n autocapitalize=\"off\"\n spellcheck=\"false\"\n (blur)=\"blur()\"\n />\n</riv-input-label>\n", styles: [":host{position:relative}.prefix{position:absolute;font:var(--input-medium);padding:var(--size-small)}input{width:100%;border:var(--border-width) solid var(--border-light);border-radius:var(--border-radius-small);font:var(--input-medium);color:var(--type-light-high-contrast);padding:var(--size-small) var(--size-small) var(--size-small) var(--size-xlarge)}input::placeholder{color:var(--type-light-disabled)}input:disabled{color:var(--type-light-disabled);background-color:var(--surface-light-1)}input:read-only{background-color:var(--surface-light-1)}input:focus{outline:none;border:var(--border-width) solid var(--purp-60)}input.warning{border-color:var(--surface-dark-caution)}input.error{border-color:var(--surface-dark-danger)}\n"] }]
|
|
17
|
+
}], propDecorators: { placeholder: [{
|
|
18
|
+
type: Input
|
|
19
|
+
}], maximumFractionDigits: [{
|
|
20
|
+
type: Input
|
|
21
|
+
}] } });
|
|
22
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY3VycmVuY3ktZmllbGQuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvcml2L3NyYy9saWIvaW5wdXQvY3VycmVuY3ktZmllbGQvY3VycmVuY3ktZmllbGQuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvcml2L3NyYy9saWIvaW5wdXQvY3VycmVuY3ktZmllbGQvY3VycmVuY3ktZmllbGQuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLHVCQUF1QixFQUFFLFNBQVMsRUFBRSxLQUFLLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDMUUsT0FBTyxFQUFFLG9CQUFvQixFQUFFLE1BQU0sd0NBQXdDLENBQUM7OztBQVE5RSxNQUFNLE9BQU8sc0JBQXVCLFNBQVEsb0JBQW9CO0lBTmhFOztRQVFXLGdCQUFXLEdBQVcsTUFBTSxDQUFDO1FBRzdCLDBCQUFxQixHQUFXLENBQUMsQ0FBQztLQUM1Qzs7bUhBTlksc0JBQXNCO3VHQUF0QixzQkFBc0IseUtDVG5DLGtzQkEyQkE7MkZEbEJhLHNCQUFzQjtrQkFObEMsU0FBUzsrQkFDRSxvQkFBb0IsbUJBR2IsdUJBQXVCLENBQUMsTUFBTTs4QkFJdEMsV0FBVztzQkFEbkIsS0FBSztnQkFJRyxxQkFBcUI7c0JBRDdCLEtBQUsiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneSwgQ29tcG9uZW50LCBJbnB1dCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgTnVtYmVyRmllbGRDb21wb25lbnQgfSBmcm9tICcuLi9udW1iZXItZmllbGQvbnVtYmVyLWZpZWxkLmNvbXBvbmVudCc7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ3Jpdi1jdXJyZW5jeS1maWVsZCcsXG4gIHRlbXBsYXRlVXJsOiAnLi9jdXJyZW5jeS1maWVsZC5jb21wb25lbnQuaHRtbCcsXG4gIHN0eWxlVXJsczogWycuL2N1cnJlbmN5LWZpZWxkLmNvbXBvbmVudC5jc3MnXSxcbiAgY2hhbmdlRGV0ZWN0aW9uOiBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneS5PblB1c2gsXG59KVxuZXhwb3J0IGNsYXNzIEN1cnJlbmN5RmllbGRDb21wb25lbnQgZXh0ZW5kcyBOdW1iZXJGaWVsZENvbXBvbmVudCB7XG4gIEBJbnB1dCgpXG4gIG92ZXJyaWRlIHBsYWNlaG9sZGVyOiBzdHJpbmcgPSAnMC4wMCc7XG5cbiAgQElucHV0KClcbiAgb3ZlcnJpZGUgbWF4aW11bUZyYWN0aW9uRGlnaXRzOiBudW1iZXIgPSAyO1xufVxuIiwiPHJpdi1pbnB1dC1sYWJlbFxuICBbbGFiZWxdPVwibGFiZWxcIlxuICBbaGVscF09XCJoZWxwXCJcbiAgW3JlcXVpcmVkXT1cInJlcXVpcmVkXCJcbiAgW2xhYmVsQWN0aW9uVGV4dF09XCJsYWJlbEFjdGlvblRleHRcIlxuICAobGFiZWxBY3Rpb24pPVwibGFiZWxBY3Rpb24uZW1pdCgkZXZlbnQpXCJcbj5cbiAgPHNwYW4gY2xhc3M9XCJwcmVmaXhcIj4kPC9zcGFuPlxuICA8aW5wdXRcbiAgICAjaW5wdXRcbiAgICBbcGxhY2Vob2xkZXJdPVwicGxhY2Vob2xkZXJcIlxuICAgIHR5cGU9XCJ0ZXh0XCJcbiAgICBbcmVxdWlyZWRdPVwicmVxdWlyZWRcIlxuICAgIFtkaXNhYmxlZF09XCJkaXNhYmxlZFwiXG4gICAgW3JlYWRvbmx5XT1cInJlYWRvbmx5XCJcbiAgICBbdmFsdWVdPVwidGV4dFwiXG4gICAgKGlucHV0KT1cInRleHQgPSBpbnB1dC52YWx1ZVwiXG4gICAgW2NsYXNzLndhcm5pbmddPVwic3RhdGUgPT09ICd3YXJuaW5nJ1wiXG4gICAgW2NsYXNzLmVycm9yXT1cInN0YXRlID09PSAnZXJyb3InXCJcbiAgICBbc3R5bGUudGV4dEFsaWduXT1cImFsaWduXCJcbiAgICBhdXRvY29tcGxldGU9XCJvZmZcIlxuICAgIGF1dG9jb3JyZWN0PVwib2ZmXCJcbiAgICBhdXRvY2FwaXRhbGl6ZT1cIm9mZlwiXG4gICAgc3BlbGxjaGVjaz1cImZhbHNlXCJcbiAgICAoYmx1cik9XCJibHVyKClcIlxuICAvPlxuPC9yaXYtaW5wdXQtbGFiZWw+XG4iXX0=
|
|
@@ -0,0 +1,95 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output, } from '@angular/core';
|
|
2
|
+
import { InputLabelComponent } from '../input-label/input-label.component';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "../input-label/input-label.component";
|
|
5
|
+
export class NumberFieldComponent extends InputLabelComponent {
|
|
6
|
+
constructor() {
|
|
7
|
+
super(...arguments);
|
|
8
|
+
this.disabled = false;
|
|
9
|
+
this.readonly = false;
|
|
10
|
+
this.placeholder = '';
|
|
11
|
+
this.state = 'valid';
|
|
12
|
+
this.maximumFractionDigits = 0;
|
|
13
|
+
this.min = -Infinity;
|
|
14
|
+
this.max = Infinity;
|
|
15
|
+
this.align = 'left';
|
|
16
|
+
this.valueChange = new EventEmitter();
|
|
17
|
+
this._text = '';
|
|
18
|
+
}
|
|
19
|
+
set value(v) {
|
|
20
|
+
if (NumberFieldComponent.clean(v, this.maximumFractionDigits, this.min, this.max) ===
|
|
21
|
+
NumberFieldComponent.clean(this._text, this.maximumFractionDigits, this.min, this.max))
|
|
22
|
+
return;
|
|
23
|
+
this._text = NumberFieldComponent.format(v, this.maximumFractionDigits, this.min, this.max);
|
|
24
|
+
}
|
|
25
|
+
set text(t) {
|
|
26
|
+
this._text = t;
|
|
27
|
+
this.valueChange.emit(NumberFieldComponent.clean(t, this.maximumFractionDigits, this.min, this.max));
|
|
28
|
+
}
|
|
29
|
+
get text() {
|
|
30
|
+
return this._text;
|
|
31
|
+
}
|
|
32
|
+
blur() {
|
|
33
|
+
this._text = NumberFieldComponent.format(this._text, this.maximumFractionDigits, this.min, this.max);
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
NumberFieldComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: NumberFieldComponent, deps: null, target: i0.ɵɵFactoryTarget.Component });
|
|
37
|
+
NumberFieldComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: NumberFieldComponent, selector: "riv-number-field", inputs: { disabled: "disabled", readonly: "readonly", placeholder: "placeholder", state: "state", maximumFractionDigits: "maximumFractionDigits", min: "min", max: "max", align: "align", value: "value" }, outputs: { valueChange: "valueChange" }, usesInheritance: true, ngImport: i0, template: "<riv-input-label\n [label]=\"label\"\n [help]=\"help\"\n [required]=\"required\"\n [labelActionText]=\"labelActionText\"\n (labelAction)=\"labelAction.emit($event)\"\n>\n <input\n #input\n [placeholder]=\"placeholder\"\n type=\"text\"\n [required]=\"required\"\n [disabled]=\"disabled\"\n [readonly]=\"readonly\"\n [value]=\"text\"\n (input)=\"text = input.value\"\n [class.warning]=\"state === 'warning'\"\n [class.error]=\"state === 'error'\"\n [style.textAlign]=\"align\"\n autocomplete=\"off\"\n autocorrect=\"off\"\n autocapitalize=\"off\"\n spellcheck=\"false\"\n (blur)=\"blur()\"\n />\n</riv-input-label>\n", styles: ["input{width:100%;border:var(--border-width) solid var(--border-light);border-radius:var(--border-radius-small);font:var(--input-medium);color:var(--type-light-high-contrast);padding:var(--size-small)}input::placeholder{color:var(--type-light-disabled)}input:disabled{color:var(--type-light-disabled);background-color:var(--surface-light-1)}input:read-only{background-color:var(--surface-light-1)}input:focus{outline:none;border:var(--border-width) solid var(--purp-60)}input.warning{border-color:var(--surface-dark-caution)}input.error{border-color:var(--surface-dark-danger)}\n"], dependencies: [{ kind: "component", type: i1.InputLabelComponent, selector: "riv-input-label", inputs: ["label", "help", "required", "labelActionText"], outputs: ["labelAction"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
38
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: NumberFieldComponent, decorators: [{
|
|
39
|
+
type: Component,
|
|
40
|
+
args: [{ selector: 'riv-number-field', changeDetection: ChangeDetectionStrategy.OnPush, template: "<riv-input-label\n [label]=\"label\"\n [help]=\"help\"\n [required]=\"required\"\n [labelActionText]=\"labelActionText\"\n (labelAction)=\"labelAction.emit($event)\"\n>\n <input\n #input\n [placeholder]=\"placeholder\"\n type=\"text\"\n [required]=\"required\"\n [disabled]=\"disabled\"\n [readonly]=\"readonly\"\n [value]=\"text\"\n (input)=\"text = input.value\"\n [class.warning]=\"state === 'warning'\"\n [class.error]=\"state === 'error'\"\n [style.textAlign]=\"align\"\n autocomplete=\"off\"\n autocorrect=\"off\"\n autocapitalize=\"off\"\n spellcheck=\"false\"\n (blur)=\"blur()\"\n />\n</riv-input-label>\n", styles: ["input{width:100%;border:var(--border-width) solid var(--border-light);border-radius:var(--border-radius-small);font:var(--input-medium);color:var(--type-light-high-contrast);padding:var(--size-small)}input::placeholder{color:var(--type-light-disabled)}input:disabled{color:var(--type-light-disabled);background-color:var(--surface-light-1)}input:read-only{background-color:var(--surface-light-1)}input:focus{outline:none;border:var(--border-width) solid var(--purp-60)}input.warning{border-color:var(--surface-dark-caution)}input.error{border-color:var(--surface-dark-danger)}\n"] }]
|
|
41
|
+
}], propDecorators: { disabled: [{
|
|
42
|
+
type: Input
|
|
43
|
+
}], readonly: [{
|
|
44
|
+
type: Input
|
|
45
|
+
}], placeholder: [{
|
|
46
|
+
type: Input
|
|
47
|
+
}], state: [{
|
|
48
|
+
type: Input
|
|
49
|
+
}], maximumFractionDigits: [{
|
|
50
|
+
type: Input
|
|
51
|
+
}], min: [{
|
|
52
|
+
type: Input
|
|
53
|
+
}], max: [{
|
|
54
|
+
type: Input
|
|
55
|
+
}], align: [{
|
|
56
|
+
type: Input
|
|
57
|
+
}], value: [{
|
|
58
|
+
type: Input
|
|
59
|
+
}], valueChange: [{
|
|
60
|
+
type: Output
|
|
61
|
+
}] } });
|
|
62
|
+
(function (NumberFieldComponent) {
|
|
63
|
+
NumberFieldComponent.Alignments = ['left', 'center', 'right'];
|
|
64
|
+
function clean(v, maximumFractionDigits, min, max) {
|
|
65
|
+
function digitsOnly(str) {
|
|
66
|
+
return str.replace(/\D/g, '');
|
|
67
|
+
}
|
|
68
|
+
if (!v)
|
|
69
|
+
return null;
|
|
70
|
+
const decimalPointIdx = v.indexOf('.');
|
|
71
|
+
const signIdx = v.indexOf('-');
|
|
72
|
+
const cleanedString = (signIdx > -1 ? '-' : '') +
|
|
73
|
+
(decimalPointIdx > -1
|
|
74
|
+
? digitsOnly(v.substring(0, decimalPointIdx)) +
|
|
75
|
+
'.' +
|
|
76
|
+
digitsOnly(v.substring(decimalPointIdx + 1))
|
|
77
|
+
: digitsOnly(v));
|
|
78
|
+
if (cleanedString === '.' ||
|
|
79
|
+
cleanedString === '-' ||
|
|
80
|
+
cleanedString === '-.') {
|
|
81
|
+
return null;
|
|
82
|
+
}
|
|
83
|
+
const cleaned = Number(Number(cleanedString).toFixed(maximumFractionDigits));
|
|
84
|
+
return Math.max(min, Math.min(max, cleaned));
|
|
85
|
+
}
|
|
86
|
+
NumberFieldComponent.clean = clean;
|
|
87
|
+
function format(v, maximumFractionDigits, min, max) {
|
|
88
|
+
const cleaned = clean(v, maximumFractionDigits, min, max);
|
|
89
|
+
if (cleaned === null)
|
|
90
|
+
return '';
|
|
91
|
+
return new Intl.NumberFormat('en-US', { maximumFractionDigits }).format(cleaned);
|
|
92
|
+
}
|
|
93
|
+
NumberFieldComponent.format = format;
|
|
94
|
+
})(NumberFieldComponent || (NumberFieldComponent = {}));
|
|
95
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"number-field.component.js","sourceRoot":"","sources":["../../../../../../projects/riv/src/lib/input/number-field/number-field.component.ts","../../../../../../projects/riv/src/lib/input/number-field/number-field.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component,\n  EventEmitter,\n  Input,\n  Output,\n} from '@angular/core';\nimport { InputLabelComponent } from '../input-label/input-label.component';\nimport { TextFieldComponent } from '../text-field/text-field.component';\n\n@Component({\n  selector: 'riv-number-field',\n  templateUrl: './number-field.component.html',\n  styleUrls: ['./number-field.component.css'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class NumberFieldComponent extends InputLabelComponent {\n  @Input()\n  disabled: boolean = false;\n\n  @Input()\n  readonly: boolean = false;\n\n  @Input()\n  placeholder: string = '';\n\n  @Input()\n  state: TextFieldComponent.State = 'valid';\n\n  @Input()\n  maximumFractionDigits: number = 0;\n\n  @Input()\n  min: number = -Infinity;\n\n  @Input()\n  max: number = Infinity;\n\n  @Input()\n  align: NumberFieldComponent.Alignment = 'left';\n\n  @Input()\n  set value(v: string | null) {\n    if (\n      NumberFieldComponent.clean(\n        v,\n        this.maximumFractionDigits,\n        this.min,\n        this.max,\n      ) ===\n      NumberFieldComponent.clean(\n        this._text,\n        this.maximumFractionDigits,\n        this.min,\n        this.max,\n      )\n    )\n      return;\n    this._text = NumberFieldComponent.format(\n      v,\n      this.maximumFractionDigits,\n      this.min,\n      this.max,\n    );\n  }\n\n  @Output()\n  valueChange = new EventEmitter<number | null>();\n\n  protected _text: string = '';\n\n  set text(t: string) {\n    this._text = t;\n    this.valueChange.emit(\n      NumberFieldComponent.clean(\n        t,\n        this.maximumFractionDigits,\n        this.min,\n        this.max,\n      ),\n    );\n  }\n  get text(): string {\n    return this._text;\n  }\n\n  blur() {\n    this._text = NumberFieldComponent.format(\n      this._text,\n      this.maximumFractionDigits,\n      this.min,\n      this.max,\n    );\n  }\n}\n\nexport namespace NumberFieldComponent {\n  export const Alignments = ['left', 'center', 'right'] as const;\n  export type Alignment = (typeof Alignments)[number];\n\n  export function clean(\n    v: string | null,\n    maximumFractionDigits: number,\n    min: number,\n    max: number,\n  ): number | null {\n    function digitsOnly(str: string): string {\n      return str.replace(/\\D/g, '');\n    }\n\n    if (!v) return null;\n    const decimalPointIdx = v.indexOf('.');\n    const signIdx = v.indexOf('-');\n    const cleanedString: string =\n      (signIdx > -1 ? '-' : '') +\n      (decimalPointIdx > -1\n        ? digitsOnly(v.substring(0, decimalPointIdx)) +\n          '.' +\n          digitsOnly(v.substring(decimalPointIdx + 1))\n        : digitsOnly(v));\n    if (\n      cleanedString === '.' ||\n      cleanedString === '-' ||\n      cleanedString === '-.'\n    ) {\n      return null;\n    }\n    const cleaned = Number(\n      Number(cleanedString).toFixed(maximumFractionDigits),\n    );\n    return Math.max(min, Math.min(max, cleaned));\n  }\n\n  export function format(\n    v: string | null,\n    maximumFractionDigits: number,\n    min: number,\n    max: number,\n  ): string {\n    const cleaned = clean(v, maximumFractionDigits, min, max);\n    if (cleaned === null) return '';\n    return new Intl.NumberFormat('en-US', { maximumFractionDigits }).format(\n      cleaned,\n    );\n  }\n}\n","<riv-input-label\n  [label]=\"label\"\n  [help]=\"help\"\n  [required]=\"required\"\n  [labelActionText]=\"labelActionText\"\n  (labelAction)=\"labelAction.emit($event)\"\n>\n  <input\n    #input\n    [placeholder]=\"placeholder\"\n    type=\"text\"\n    [required]=\"required\"\n    [disabled]=\"disabled\"\n    [readonly]=\"readonly\"\n    [value]=\"text\"\n    (input)=\"text = input.value\"\n    [class.warning]=\"state === 'warning'\"\n    [class.error]=\"state === 'error'\"\n    [style.textAlign]=\"align\"\n    autocomplete=\"off\"\n    autocorrect=\"off\"\n    autocapitalize=\"off\"\n    spellcheck=\"false\"\n    (blur)=\"blur()\"\n  />\n</riv-input-label>\n"]}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, Input } from '@angular/core';
|
|
2
|
+
import { NumberFieldComponent } from '../number-field/number-field.component';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "../input-label/input-label.component";
|
|
5
|
+
export class PercentageFieldComponent extends NumberFieldComponent {
|
|
6
|
+
constructor() {
|
|
7
|
+
super(...arguments);
|
|
8
|
+
this.align = 'right';
|
|
9
|
+
}
|
|
10
|
+
}
|
|
11
|
+
PercentageFieldComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: PercentageFieldComponent, deps: null, target: i0.ɵɵFactoryTarget.Component });
|
|
12
|
+
PercentageFieldComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: PercentageFieldComponent, selector: "riv-percentage-field", inputs: { align: "align" }, usesInheritance: true, ngImport: i0, template: "<riv-input-label\n [label]=\"label\"\n [help]=\"help\"\n [required]=\"required\"\n [labelActionText]=\"labelActionText\"\n (labelAction)=\"labelAction.emit($event)\"\n>\n <input\n #input\n [placeholder]=\"placeholder\"\n type=\"text\"\n [required]=\"required\"\n [disabled]=\"disabled\"\n [readonly]=\"readonly\"\n [value]=\"text\"\n (input)=\"text = input.value\"\n [class.warning]=\"state === 'warning'\"\n [class.error]=\"state === 'error'\"\n [style.textAlign]=\"align\"\n autocomplete=\"off\"\n autocorrect=\"off\"\n autocapitalize=\"off\"\n spellcheck=\"false\"\n (blur)=\"blur()\"\n />\n <span class=\"suffix\">%</span>\n</riv-input-label>\n", styles: [":host{position:relative}.suffix{position:absolute;right:0;font:var(--input-medium);padding:var(--size-small)}input{width:100%;border:var(--border-width) solid var(--border-light);border-radius:var(--border-radius-small);font:var(--input-medium);color:var(--type-light-high-contrast);padding:var(--size-small) var(--size-xlarge) var(--size-small) var(--size-small)}input::placeholder{color:var(--type-light-disabled)}input:disabled{color:var(--type-light-disabled);background-color:var(--surface-light-1)}input:read-only{background-color:var(--surface-light-1)}input:focus{outline:none;border:var(--border-width) solid var(--purp-60)}input.warning{border-color:var(--surface-dark-caution)}input.error{border-color:var(--surface-dark-danger)}\n"], dependencies: [{ kind: "component", type: i1.InputLabelComponent, selector: "riv-input-label", inputs: ["label", "help", "required", "labelActionText"], outputs: ["labelAction"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
13
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: PercentageFieldComponent, decorators: [{
|
|
14
|
+
type: Component,
|
|
15
|
+
args: [{ selector: 'riv-percentage-field', changeDetection: ChangeDetectionStrategy.OnPush, template: "<riv-input-label\n [label]=\"label\"\n [help]=\"help\"\n [required]=\"required\"\n [labelActionText]=\"labelActionText\"\n (labelAction)=\"labelAction.emit($event)\"\n>\n <input\n #input\n [placeholder]=\"placeholder\"\n type=\"text\"\n [required]=\"required\"\n [disabled]=\"disabled\"\n [readonly]=\"readonly\"\n [value]=\"text\"\n (input)=\"text = input.value\"\n [class.warning]=\"state === 'warning'\"\n [class.error]=\"state === 'error'\"\n [style.textAlign]=\"align\"\n autocomplete=\"off\"\n autocorrect=\"off\"\n autocapitalize=\"off\"\n spellcheck=\"false\"\n (blur)=\"blur()\"\n />\n <span class=\"suffix\">%</span>\n</riv-input-label>\n", styles: [":host{position:relative}.suffix{position:absolute;right:0;font:var(--input-medium);padding:var(--size-small)}input{width:100%;border:var(--border-width) solid var(--border-light);border-radius:var(--border-radius-small);font:var(--input-medium);color:var(--type-light-high-contrast);padding:var(--size-small) var(--size-xlarge) var(--size-small) var(--size-small)}input::placeholder{color:var(--type-light-disabled)}input:disabled{color:var(--type-light-disabled);background-color:var(--surface-light-1)}input:read-only{background-color:var(--surface-light-1)}input:focus{outline:none;border:var(--border-width) solid var(--purp-60)}input.warning{border-color:var(--surface-dark-caution)}input.error{border-color:var(--surface-dark-danger)}\n"] }]
|
|
16
|
+
}], propDecorators: { align: [{
|
|
17
|
+
type: Input
|
|
18
|
+
}] } });
|
|
19
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGVyY2VudGFnZS1maWVsZC5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9yaXYvc3JjL2xpYi9pbnB1dC9wZXJjZW50YWdlLWZpZWxkL3BlcmNlbnRhZ2UtZmllbGQuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvcml2L3NyYy9saWIvaW5wdXQvcGVyY2VudGFnZS1maWVsZC9wZXJjZW50YWdlLWZpZWxkLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSx1QkFBdUIsRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQzFFLE9BQU8sRUFBRSxvQkFBb0IsRUFBRSxNQUFNLHdDQUF3QyxDQUFDOzs7QUFROUUsTUFBTSxPQUFPLHdCQUF5QixTQUFRLG9CQUFvQjtJQU5sRTs7UUFRVyxVQUFLLEdBQW1DLE9BQU8sQ0FBQztLQUMxRDs7cUhBSFksd0JBQXdCO3lHQUF4Qix3QkFBd0IsK0dDVHJDLGtzQkEyQkE7MkZEbEJhLHdCQUF3QjtrQkFOcEMsU0FBUzsrQkFDRSxzQkFBc0IsbUJBR2YsdUJBQXVCLENBQUMsTUFBTTs4QkFJdEMsS0FBSztzQkFEYixLQUFLIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3ksIENvbXBvbmVudCwgSW5wdXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IE51bWJlckZpZWxkQ29tcG9uZW50IH0gZnJvbSAnLi4vbnVtYmVyLWZpZWxkL251bWJlci1maWVsZC5jb21wb25lbnQnO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdyaXYtcGVyY2VudGFnZS1maWVsZCcsXG4gIHRlbXBsYXRlVXJsOiAnLi9wZXJjZW50YWdlLWZpZWxkLmNvbXBvbmVudC5odG1sJyxcbiAgc3R5bGVVcmxzOiBbJy4vcGVyY2VudGFnZS1maWVsZC5jb21wb25lbnQuY3NzJ10sXG4gIGNoYW5nZURldGVjdGlvbjogQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3kuT25QdXNoLFxufSlcbmV4cG9ydCBjbGFzcyBQZXJjZW50YWdlRmllbGRDb21wb25lbnQgZXh0ZW5kcyBOdW1iZXJGaWVsZENvbXBvbmVudCB7XG4gIEBJbnB1dCgpXG4gIG92ZXJyaWRlIGFsaWduOiBOdW1iZXJGaWVsZENvbXBvbmVudC5BbGlnbm1lbnQgPSAncmlnaHQnO1xufVxuIiwiPHJpdi1pbnB1dC1sYWJlbFxuICBbbGFiZWxdPVwibGFiZWxcIlxuICBbaGVscF09XCJoZWxwXCJcbiAgW3JlcXVpcmVkXT1cInJlcXVpcmVkXCJcbiAgW2xhYmVsQWN0aW9uVGV4dF09XCJsYWJlbEFjdGlvblRleHRcIlxuICAobGFiZWxBY3Rpb24pPVwibGFiZWxBY3Rpb24uZW1pdCgkZXZlbnQpXCJcbj5cbiAgPGlucHV0XG4gICAgI2lucHV0XG4gICAgW3BsYWNlaG9sZGVyXT1cInBsYWNlaG9sZGVyXCJcbiAgICB0eXBlPVwidGV4dFwiXG4gICAgW3JlcXVpcmVkXT1cInJlcXVpcmVkXCJcbiAgICBbZGlzYWJsZWRdPVwiZGlzYWJsZWRcIlxuICAgIFtyZWFkb25seV09XCJyZWFkb25seVwiXG4gICAgW3ZhbHVlXT1cInRleHRcIlxuICAgIChpbnB1dCk9XCJ0ZXh0ID0gaW5wdXQudmFsdWVcIlxuICAgIFtjbGFzcy53YXJuaW5nXT1cInN0YXRlID09PSAnd2FybmluZydcIlxuICAgIFtjbGFzcy5lcnJvcl09XCJzdGF0ZSA9PT0gJ2Vycm9yJ1wiXG4gICAgW3N0eWxlLnRleHRBbGlnbl09XCJhbGlnblwiXG4gICAgYXV0b2NvbXBsZXRlPVwib2ZmXCJcbiAgICBhdXRvY29ycmVjdD1cIm9mZlwiXG4gICAgYXV0b2NhcGl0YWxpemU9XCJvZmZcIlxuICAgIHNwZWxsY2hlY2s9XCJmYWxzZVwiXG4gICAgKGJsdXIpPVwiYmx1cigpXCJcbiAgLz5cbiAgPHNwYW4gY2xhc3M9XCJzdWZmaXhcIj4lPC9zcGFuPlxuPC9yaXYtaW5wdXQtbGFiZWw+XG4iXX0=
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output, ViewChild, } from '@angular/core';
|
|
2
2
|
import Fuse from 'fuse.js';
|
|
3
|
+
import { debounceTime, from, map, pairwise, startWith, switchMap, } from 'rxjs';
|
|
3
4
|
import { InputLabelComponent } from '../input-label/input-label.component';
|
|
4
5
|
import * as i0 from "@angular/core";
|
|
5
6
|
import * as i1 from "@angular/common";
|
|
@@ -9,6 +10,7 @@ import * as i4 from "../../visualization/highlight/highlight.component";
|
|
|
9
10
|
import * as i5 from "../../icon/icon.component";
|
|
10
11
|
import * as i6 from "../input-label/input-label.component";
|
|
11
12
|
import * as i7 from "../../load/loading-cover/loading-cover.component";
|
|
13
|
+
import * as i8 from "../../visualization/zero-state/zero-state.component";
|
|
12
14
|
export class SingleSelectComponent extends InputLabelComponent {
|
|
13
15
|
constructor() {
|
|
14
16
|
super(...arguments);
|
|
@@ -64,10 +66,10 @@ export class SingleSelectComponent extends InputLabelComponent {
|
|
|
64
66
|
}
|
|
65
67
|
}
|
|
66
68
|
SingleSelectComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: SingleSelectComponent, deps: null, target: i0.ɵɵFactoryTarget.Component });
|
|
67
|
-
SingleSelectComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: SingleSelectComponent, selector: "riv-single-select", inputs: { groups: "groups", selectedOption: "selectedOption", filterabilityOptions: "filterabilityOptions", loading: "loading", locked: "locked", maxCalloutHeight: "maxCalloutHeight", noOptionsMessage: "noOptionsMessage", nodeTemplate: "nodeTemplate", triggerTemplate: "triggerTemplate", placeholder: "placeholder", disabled: "disabled" }, outputs: { filterQueryChange: "filterQueryChange", selectedOptionChange: "selectedOptionChange" }, viewQueries: [{ propertyName: "customTriggerButton", first: true, predicate: ["customTriggerButton"], descendants: true }, { propertyName: "standardTriggerButton", first: true, predicate: ["standardTriggerButton"], descendants: true }], usesInheritance: true, ngImport: i0, template: "<ng-container *ngIf=\"triggerTemplate; else standardTrigger\">\n <button\n #customTriggerButton\n (click)=\"allowedOpen()\"\n [disabled]=\"disabled || locked\"\n type=\"button\"\n >\n <ng-container\n [ngTemplateOutlet]=\"triggerTemplate\"\n [ngTemplateOutletContext]=\"{ selectedOption }\"\n ></ng-container>\n </button>\n</ng-container>\n<ng-template #standardTrigger>\n <riv-input-label\n [label]=\"label\"\n [help]=\"help\"\n [required]=\"required\"\n [labelActionText]=\"labelActionText\"\n (labelAction)=\"labelAction.emit($event)\"\n >\n <button\n #standardTriggerButton\n class=\"trigger\"\n (click)=\"allowedOpen()\"\n [disabled]=\"disabled || locked\"\n type=\"button\"\n >\n <ng-container *ngIf=\"selectedOption; else placeholderValue\">\n <span class=\"value\">{{ selectedOption.title }}</span>\n </ng-container>\n <ng-template #placeholderValue>\n <span class=\"value placeholder\">{{ placeholder }}</span>\n </ng-template>\n <span class=\"chevron\">\n <riv-icon *ngIf=\"!locked\" [name]=\"'ChevronDown'\" [size]=\"16\"></riv-icon>\n <riv-icon *ngIf=\"locked\" [name]=\"'Lock'\" [size]=\"16\"></riv-icon>\n </span>\n </button>\n </riv-input-label>\n</ng-template>\n\n<riv-callout\n *ngIf=\"open\"\n [anchor]=\"getTrigger()\"\n [theme]=\"'light'\"\n [showCaret]=\"false\"\n [allowedPositions]=\"[\n 'top-left',\n 'top-center',\n 'top-right',\n 'bottom-right',\n 'bottom-center',\n 'bottom-left'\n ]\"\n (close)=\"open = false\"\n>\n <input\n *ngIf=\"filterabilityOptions.enabled\"\n #filter\n class=\"filter\"\n [placeholder]=\"getFilterPlaceholder()\"\n [value]=\"filterQuery\"\n (input)=\"filterQuery = filter.value; filterQueryChange.emit(filterQuery)\"\n [rivFocusOnInit]=\"true\"\n />\n <div class=\"options\" [style.max-height]=\"maxCalloutHeight\">\n <riv-loading-cover [loading]=\"loading\">\n <ng-container *ngFor=\"let group of groups; trackBy: trackByHeader\">\n <ng-container *ngIf=\"group.header\">\n <span class=\"group-header\">\n {{ group.header }}\n </span>\n </ng-container>\n <ng-container *ngIf=\"group.options; let nodes\">\n <ng-container *ngIf=\"nodes.length > 0; else empty\">\n <ng-container *ngFor=\"let node of nodes; trackBy: trackByOption\">\n <ng-container *ngIf=\"nodeTemplate; else standardTemplate\">\n <button\n class=\"custom-single-select-node\"\n [disabled]=\"node.disabled\"\n (click)=\"selectOption(node)\"\n type=\"button\"\n >\n <ng-container\n [ngTemplateOutlet]=\"nodeTemplate\"\n [ngTemplateOutletContext]=\"{ node }\"\n ></ng-container>\n </button>\n </ng-container>\n <ng-template #standardTemplate>\n <button\n class=\"single-select-node\"\n [class.selected]=\"node?.id === selectedOption?.id\"\n [class.disabled]=\"node?.disabled\"\n [disabled]=\"node?.disabled\"\n (click)=\"selectOption(node)\"\n type=\"button\"\n >\n <riv-icon\n [name]=\"'Check'\"\n *ngIf=\"node?.id === selectedOption?.id\"\n [size]=\"16\"\n ></riv-icon>\n <span class=\"label\">\n <span class=\"label-title\">\n <riv-highlight\n [text]=\"node?.title || ''\"\n [indices]=\"node?.titleHighlightIndices || []\"\n ></riv-highlight>\n </span>\n <span *ngIf=\"node?.subtitle\" class=\"label-subtitle\">\n <riv-highlight\n [text]=\"node?.subtitle || ''\"\n [indices]=\"node?.subtitleHighlightIndices || []\"\n ></riv-highlight>\n </span>\n </span>\n </button>\n </ng-template>\n </ng-container>\n </ng-container>\n <ng-template #empty>\n <div class=\"empty\">\n {{ noOptionsMessage }}\n </div>\n </ng-template>\n </ng-container>\n </ng-container>\n </riv-loading-cover>\n </div>\n <div class=\"footer\">\n <ng-content select=\"[footer]\"
|
|
69
|
+
SingleSelectComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: SingleSelectComponent, selector: "riv-single-select", inputs: { groups: "groups", selectedOption: "selectedOption", filterabilityOptions: "filterabilityOptions", loading: "loading", locked: "locked", maxCalloutHeight: "maxCalloutHeight", noOptionsMessage: "noOptionsMessage", nodeTemplate: "nodeTemplate", triggerTemplate: "triggerTemplate", placeholder: "placeholder", disabled: "disabled" }, outputs: { filterQueryChange: "filterQueryChange", selectedOptionChange: "selectedOptionChange" }, viewQueries: [{ propertyName: "customTriggerButton", first: true, predicate: ["customTriggerButton"], descendants: true }, { propertyName: "standardTriggerButton", first: true, predicate: ["standardTriggerButton"], descendants: true }], usesInheritance: true, ngImport: i0, template: "<ng-container *ngIf=\"triggerTemplate; else standardTrigger\">\n <button\n #customTriggerButton\n (click)=\"allowedOpen()\"\n [disabled]=\"disabled || locked\"\n type=\"button\"\n >\n <ng-container\n [ngTemplateOutlet]=\"triggerTemplate\"\n [ngTemplateOutletContext]=\"{ selectedOption }\"\n ></ng-container>\n </button>\n</ng-container>\n<ng-template #standardTrigger>\n <riv-input-label\n [label]=\"label\"\n [help]=\"help\"\n [required]=\"required\"\n [labelActionText]=\"labelActionText\"\n (labelAction)=\"labelAction.emit($event)\"\n >\n <button\n #standardTriggerButton\n class=\"trigger\"\n (click)=\"allowedOpen()\"\n [disabled]=\"disabled || locked\"\n type=\"button\"\n >\n <ng-container *ngIf=\"selectedOption; else placeholderValue\">\n <span class=\"value\">{{ selectedOption.title }}</span>\n </ng-container>\n <ng-template #placeholderValue>\n <span class=\"value placeholder\">{{ placeholder }}</span>\n </ng-template>\n <span class=\"chevron\">\n <riv-icon *ngIf=\"!locked\" [name]=\"'ChevronDown'\" [size]=\"16\"></riv-icon>\n <riv-icon *ngIf=\"locked\" [name]=\"'Lock'\" [size]=\"16\"></riv-icon>\n </span>\n </button>\n </riv-input-label>\n</ng-template>\n\n<riv-callout\n *ngIf=\"open\"\n [anchor]=\"getTrigger()\"\n [theme]=\"'light'\"\n [showCaret]=\"false\"\n [allowedPositions]=\"[\n 'top-left',\n 'top-center',\n 'top-right',\n 'bottom-right',\n 'bottom-center',\n 'bottom-left'\n ]\"\n (close)=\"open = false\"\n>\n <input\n *ngIf=\"filterabilityOptions.enabled\"\n #filter\n class=\"filter\"\n [placeholder]=\"getFilterPlaceholder()\"\n [value]=\"filterQuery\"\n (input)=\"filterQuery = filter.value; filterQueryChange.emit(filterQuery)\"\n [rivFocusOnInit]=\"true\"\n />\n <div class=\"options\" [style.max-height]=\"maxCalloutHeight\">\n <riv-loading-cover [loading]=\"loading\">\n <riv-zero-state\n *ngIf=\"!groups.length\"\n [message]=\"noOptionsMessage\"\n ></riv-zero-state>\n <ng-container *ngFor=\"let group of groups; trackBy: trackByHeader\">\n <ng-container *ngIf=\"group.header\">\n <span class=\"group-header\">\n {{ group.header }}\n </span>\n </ng-container>\n <ng-container *ngIf=\"group.options; let nodes\">\n <ng-container *ngIf=\"nodes.length > 0; else empty\">\n <ng-container *ngFor=\"let node of nodes; trackBy: trackByOption\">\n <ng-container *ngIf=\"nodeTemplate; else standardTemplate\">\n <button\n class=\"custom-single-select-node\"\n [disabled]=\"node.disabled\"\n (click)=\"selectOption(node)\"\n type=\"button\"\n >\n <ng-container\n [ngTemplateOutlet]=\"nodeTemplate\"\n [ngTemplateOutletContext]=\"{ node }\"\n ></ng-container>\n </button>\n </ng-container>\n <ng-template #standardTemplate>\n <button\n class=\"single-select-node\"\n [class.selected]=\"node?.id === selectedOption?.id\"\n [class.disabled]=\"node?.disabled\"\n [disabled]=\"node?.disabled\"\n (click)=\"selectOption(node)\"\n type=\"button\"\n >\n <riv-icon\n [name]=\"'Check'\"\n *ngIf=\"node?.id === selectedOption?.id\"\n [size]=\"16\"\n ></riv-icon>\n <span class=\"label\">\n <span class=\"label-title\">\n <riv-highlight\n [text]=\"node?.title || ''\"\n [indices]=\"node?.titleHighlightIndices || []\"\n ></riv-highlight>\n </span>\n <span *ngIf=\"node?.subtitle\" class=\"label-subtitle\">\n <riv-highlight\n [text]=\"node?.subtitle || ''\"\n [indices]=\"node?.subtitleHighlightIndices || []\"\n ></riv-highlight>\n </span>\n </span>\n </button>\n </ng-template>\n </ng-container>\n </ng-container>\n <ng-template #empty>\n <div class=\"empty\">\n {{ noOptionsMessage }}\n </div>\n </ng-template>\n </ng-container>\n </ng-container>\n <ng-content select=\"[postOptionList]\"></ng-content>\n </riv-loading-cover>\n </div>\n <div class=\"footer\">\n <ng-content select=\"[footer]\"></ng-content>\n </div>\n</riv-callout>\n", styles: [".trigger{width:100%;border:var(--border-width) solid var(--border-light);border-radius:var(--border-radius-small);display:flex;gap:var(--size-small);background-color:var(--surface-light-0)}.trigger:focus{outline:none;border:var(--border-width) solid var(--purp-60)}.trigger:disabled{color:var(--type-light-disabled);background-color:var(--surface-light-1)}.value{font-size:var(--type-2-font-size);line-height:var(--type-2-line-height-0);color:var(--type-light-high-contrast);padding:var(--size-small);flex-grow:1;text-align:left;overflow:hidden;text-overflow:ellipsis;white-space:pre}.value.placeholder{color:var(--type-light-disabled)}.chevron{display:flex;justify-content:center;align-items:center;padding:var(--size-small);background-color:var(--surface-light-2);border-left:var(--border-width) solid var(--border-light)}.filter{width:100%;outline:none;border:none;border-bottom:var(--border-width) solid var(--border-light);font-size:var(--type-2-font-size);line-height:var(--type-2-line-height-0);color:var(--type-light-high-contrast);padding:var(--size-small)}.filter::placeholder{color:var(--type-light-disabled)}.options{max-width:calc(var(--base-grid-size) * 150);overflow-y:auto}.single-select-node{display:flex;align-items:center;overflow:hidden;flex-grow:1;text-align:left;padding:var(--size-xsmall) var(--size-xsmall) var(--size-xsmall) calc(var(--base-grid-size) * 6);width:100%}.single-select-node.selected{padding-left:var(--size-small)}.single-select-node:hover{background-color:var(--surface-light-2)}.single-select-node.disabled{cursor:default;background-color:var(--surface-light-0)}.single-select-node.disabled .label,.single-select-node.disabled .label-subtitle{color:var(--type-light-disabled)}.single-select-node riv-icon{flex-shrink:0}.single-select-node .label{font-size:calc(var(--base-grid-size) * 4);line-height:calc(var(--base-grid-size) * 6);overflow:hidden;text-overflow:ellipsis;white-space:pre;flex-grow:1;padding:0 calc(var(--base-grid-size) * 2);min-width:calc(var(--base-grid-size) * 16)}.single-select-node .label-subtitle{padding-left:var(--base-grid-size);color:var(--type-light-low-contrast)}.custom-single-select-node{display:block;width:100%;text-align:left}.empty{padding:var(--size-medium);text-align:center;color:var(--type-light-disabled)}.footer:not(:empty){border-top:var(--border-width) solid var(--border-light-blend);background-color:var(--surface-light-1);border-bottom-left-radius:var(--border-radius-medium);border-bottom-right-radius:var(--border-radius-medium)}.group-header{display:flex;padding:8px 8px 4px;align-items:flex-start;gap:10px;align-self:stretch;font:var(--title-01)}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i2.CalloutComponent, selector: "riv-callout", inputs: ["anchor", "isModal", "preferredPosition", "allowedPositions", "fallbackDirection", "showCaret", "theme"], outputs: ["close"] }, { kind: "directive", type: i3.FocusOnInitDirective, selector: "[rivFocusOnInit]", inputs: ["rivFocusOnInit"] }, { kind: "component", type: i4.HighlightComponent, selector: "riv-highlight", inputs: ["text", "indices"] }, { kind: "component", type: i5.IconComponent, selector: "riv-icon", inputs: ["name", "size", "customSize", "strokeWidth"] }, { kind: "component", type: i6.InputLabelComponent, selector: "riv-input-label", inputs: ["label", "help", "required", "labelActionText"], outputs: ["labelAction"] }, { kind: "component", type: i7.LoadingCoverComponent, selector: "riv-loading-cover", inputs: ["loading", "loadingSize", "errorMessage"] }, { kind: "component", type: i8.ZeroStateComponent, selector: "riv-zero-state", inputs: ["message"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
68
70
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: SingleSelectComponent, decorators: [{
|
|
69
71
|
type: Component,
|
|
70
|
-
args: [{ selector: 'riv-single-select', changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container *ngIf=\"triggerTemplate; else standardTrigger\">\n <button\n #customTriggerButton\n (click)=\"allowedOpen()\"\n [disabled]=\"disabled || locked\"\n type=\"button\"\n >\n <ng-container\n [ngTemplateOutlet]=\"triggerTemplate\"\n [ngTemplateOutletContext]=\"{ selectedOption }\"\n ></ng-container>\n </button>\n</ng-container>\n<ng-template #standardTrigger>\n <riv-input-label\n [label]=\"label\"\n [help]=\"help\"\n [required]=\"required\"\n [labelActionText]=\"labelActionText\"\n (labelAction)=\"labelAction.emit($event)\"\n >\n <button\n #standardTriggerButton\n class=\"trigger\"\n (click)=\"allowedOpen()\"\n [disabled]=\"disabled || locked\"\n type=\"button\"\n >\n <ng-container *ngIf=\"selectedOption; else placeholderValue\">\n <span class=\"value\">{{ selectedOption.title }}</span>\n </ng-container>\n <ng-template #placeholderValue>\n <span class=\"value placeholder\">{{ placeholder }}</span>\n </ng-template>\n <span class=\"chevron\">\n <riv-icon *ngIf=\"!locked\" [name]=\"'ChevronDown'\" [size]=\"16\"></riv-icon>\n <riv-icon *ngIf=\"locked\" [name]=\"'Lock'\" [size]=\"16\"></riv-icon>\n </span>\n </button>\n </riv-input-label>\n</ng-template>\n\n<riv-callout\n *ngIf=\"open\"\n [anchor]=\"getTrigger()\"\n [theme]=\"'light'\"\n [showCaret]=\"false\"\n [allowedPositions]=\"[\n 'top-left',\n 'top-center',\n 'top-right',\n 'bottom-right',\n 'bottom-center',\n 'bottom-left'\n ]\"\n (close)=\"open = false\"\n>\n <input\n *ngIf=\"filterabilityOptions.enabled\"\n #filter\n class=\"filter\"\n [placeholder]=\"getFilterPlaceholder()\"\n [value]=\"filterQuery\"\n (input)=\"filterQuery = filter.value; filterQueryChange.emit(filterQuery)\"\n [rivFocusOnInit]=\"true\"\n />\n <div class=\"options\" [style.max-height]=\"maxCalloutHeight\">\n <riv-loading-cover [loading]=\"loading\">\n <ng-container *ngFor=\"let group of groups; trackBy: trackByHeader\">\n <ng-container *ngIf=\"group.header\">\n <span class=\"group-header\">\n {{ group.header }}\n </span>\n </ng-container>\n <ng-container *ngIf=\"group.options; let nodes\">\n <ng-container *ngIf=\"nodes.length > 0; else empty\">\n <ng-container *ngFor=\"let node of nodes; trackBy: trackByOption\">\n <ng-container *ngIf=\"nodeTemplate; else standardTemplate\">\n <button\n class=\"custom-single-select-node\"\n [disabled]=\"node.disabled\"\n (click)=\"selectOption(node)\"\n type=\"button\"\n >\n <ng-container\n [ngTemplateOutlet]=\"nodeTemplate\"\n [ngTemplateOutletContext]=\"{ node }\"\n ></ng-container>\n </button>\n </ng-container>\n <ng-template #standardTemplate>\n <button\n class=\"single-select-node\"\n [class.selected]=\"node?.id === selectedOption?.id\"\n [class.disabled]=\"node?.disabled\"\n [disabled]=\"node?.disabled\"\n (click)=\"selectOption(node)\"\n type=\"button\"\n >\n <riv-icon\n [name]=\"'Check'\"\n *ngIf=\"node?.id === selectedOption?.id\"\n [size]=\"16\"\n ></riv-icon>\n <span class=\"label\">\n <span class=\"label-title\">\n <riv-highlight\n [text]=\"node?.title || ''\"\n [indices]=\"node?.titleHighlightIndices || []\"\n ></riv-highlight>\n </span>\n <span *ngIf=\"node?.subtitle\" class=\"label-subtitle\">\n <riv-highlight\n [text]=\"node?.subtitle || ''\"\n [indices]=\"node?.subtitleHighlightIndices || []\"\n ></riv-highlight>\n </span>\n </span>\n </button>\n </ng-template>\n </ng-container>\n </ng-container>\n <ng-template #empty>\n <div class=\"empty\">\n {{ noOptionsMessage }}\n </div>\n </ng-template>\n </ng-container>\n </ng-container>\n </riv-loading-cover>\n </div>\n <div class=\"footer\">\n <ng-content select=\"[footer]\"
|
|
72
|
+
args: [{ selector: 'riv-single-select', changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container *ngIf=\"triggerTemplate; else standardTrigger\">\n <button\n #customTriggerButton\n (click)=\"allowedOpen()\"\n [disabled]=\"disabled || locked\"\n type=\"button\"\n >\n <ng-container\n [ngTemplateOutlet]=\"triggerTemplate\"\n [ngTemplateOutletContext]=\"{ selectedOption }\"\n ></ng-container>\n </button>\n</ng-container>\n<ng-template #standardTrigger>\n <riv-input-label\n [label]=\"label\"\n [help]=\"help\"\n [required]=\"required\"\n [labelActionText]=\"labelActionText\"\n (labelAction)=\"labelAction.emit($event)\"\n >\n <button\n #standardTriggerButton\n class=\"trigger\"\n (click)=\"allowedOpen()\"\n [disabled]=\"disabled || locked\"\n type=\"button\"\n >\n <ng-container *ngIf=\"selectedOption; else placeholderValue\">\n <span class=\"value\">{{ selectedOption.title }}</span>\n </ng-container>\n <ng-template #placeholderValue>\n <span class=\"value placeholder\">{{ placeholder }}</span>\n </ng-template>\n <span class=\"chevron\">\n <riv-icon *ngIf=\"!locked\" [name]=\"'ChevronDown'\" [size]=\"16\"></riv-icon>\n <riv-icon *ngIf=\"locked\" [name]=\"'Lock'\" [size]=\"16\"></riv-icon>\n </span>\n </button>\n </riv-input-label>\n</ng-template>\n\n<riv-callout\n *ngIf=\"open\"\n [anchor]=\"getTrigger()\"\n [theme]=\"'light'\"\n [showCaret]=\"false\"\n [allowedPositions]=\"[\n 'top-left',\n 'top-center',\n 'top-right',\n 'bottom-right',\n 'bottom-center',\n 'bottom-left'\n ]\"\n (close)=\"open = false\"\n>\n <input\n *ngIf=\"filterabilityOptions.enabled\"\n #filter\n class=\"filter\"\n [placeholder]=\"getFilterPlaceholder()\"\n [value]=\"filterQuery\"\n (input)=\"filterQuery = filter.value; filterQueryChange.emit(filterQuery)\"\n [rivFocusOnInit]=\"true\"\n />\n <div class=\"options\" [style.max-height]=\"maxCalloutHeight\">\n <riv-loading-cover [loading]=\"loading\">\n <riv-zero-state\n *ngIf=\"!groups.length\"\n [message]=\"noOptionsMessage\"\n ></riv-zero-state>\n <ng-container *ngFor=\"let group of groups; trackBy: trackByHeader\">\n <ng-container *ngIf=\"group.header\">\n <span class=\"group-header\">\n {{ group.header }}\n </span>\n </ng-container>\n <ng-container *ngIf=\"group.options; let nodes\">\n <ng-container *ngIf=\"nodes.length > 0; else empty\">\n <ng-container *ngFor=\"let node of nodes; trackBy: trackByOption\">\n <ng-container *ngIf=\"nodeTemplate; else standardTemplate\">\n <button\n class=\"custom-single-select-node\"\n [disabled]=\"node.disabled\"\n (click)=\"selectOption(node)\"\n type=\"button\"\n >\n <ng-container\n [ngTemplateOutlet]=\"nodeTemplate\"\n [ngTemplateOutletContext]=\"{ node }\"\n ></ng-container>\n </button>\n </ng-container>\n <ng-template #standardTemplate>\n <button\n class=\"single-select-node\"\n [class.selected]=\"node?.id === selectedOption?.id\"\n [class.disabled]=\"node?.disabled\"\n [disabled]=\"node?.disabled\"\n (click)=\"selectOption(node)\"\n type=\"button\"\n >\n <riv-icon\n [name]=\"'Check'\"\n *ngIf=\"node?.id === selectedOption?.id\"\n [size]=\"16\"\n ></riv-icon>\n <span class=\"label\">\n <span class=\"label-title\">\n <riv-highlight\n [text]=\"node?.title || ''\"\n [indices]=\"node?.titleHighlightIndices || []\"\n ></riv-highlight>\n </span>\n <span *ngIf=\"node?.subtitle\" class=\"label-subtitle\">\n <riv-highlight\n [text]=\"node?.subtitle || ''\"\n [indices]=\"node?.subtitleHighlightIndices || []\"\n ></riv-highlight>\n </span>\n </span>\n </button>\n </ng-template>\n </ng-container>\n </ng-container>\n <ng-template #empty>\n <div class=\"empty\">\n {{ noOptionsMessage }}\n </div>\n </ng-template>\n </ng-container>\n </ng-container>\n <ng-content select=\"[postOptionList]\"></ng-content>\n </riv-loading-cover>\n </div>\n <div class=\"footer\">\n <ng-content select=\"[footer]\"></ng-content>\n </div>\n</riv-callout>\n", styles: [".trigger{width:100%;border:var(--border-width) solid var(--border-light);border-radius:var(--border-radius-small);display:flex;gap:var(--size-small);background-color:var(--surface-light-0)}.trigger:focus{outline:none;border:var(--border-width) solid var(--purp-60)}.trigger:disabled{color:var(--type-light-disabled);background-color:var(--surface-light-1)}.value{font-size:var(--type-2-font-size);line-height:var(--type-2-line-height-0);color:var(--type-light-high-contrast);padding:var(--size-small);flex-grow:1;text-align:left;overflow:hidden;text-overflow:ellipsis;white-space:pre}.value.placeholder{color:var(--type-light-disabled)}.chevron{display:flex;justify-content:center;align-items:center;padding:var(--size-small);background-color:var(--surface-light-2);border-left:var(--border-width) solid var(--border-light)}.filter{width:100%;outline:none;border:none;border-bottom:var(--border-width) solid var(--border-light);font-size:var(--type-2-font-size);line-height:var(--type-2-line-height-0);color:var(--type-light-high-contrast);padding:var(--size-small)}.filter::placeholder{color:var(--type-light-disabled)}.options{max-width:calc(var(--base-grid-size) * 150);overflow-y:auto}.single-select-node{display:flex;align-items:center;overflow:hidden;flex-grow:1;text-align:left;padding:var(--size-xsmall) var(--size-xsmall) var(--size-xsmall) calc(var(--base-grid-size) * 6);width:100%}.single-select-node.selected{padding-left:var(--size-small)}.single-select-node:hover{background-color:var(--surface-light-2)}.single-select-node.disabled{cursor:default;background-color:var(--surface-light-0)}.single-select-node.disabled .label,.single-select-node.disabled .label-subtitle{color:var(--type-light-disabled)}.single-select-node riv-icon{flex-shrink:0}.single-select-node .label{font-size:calc(var(--base-grid-size) * 4);line-height:calc(var(--base-grid-size) * 6);overflow:hidden;text-overflow:ellipsis;white-space:pre;flex-grow:1;padding:0 calc(var(--base-grid-size) * 2);min-width:calc(var(--base-grid-size) * 16)}.single-select-node .label-subtitle{padding-left:var(--base-grid-size);color:var(--type-light-low-contrast)}.custom-single-select-node{display:block;width:100%;text-align:left}.empty{padding:var(--size-medium);text-align:center;color:var(--type-light-disabled)}.footer:not(:empty){border-top:var(--border-width) solid var(--border-light-blend);background-color:var(--surface-light-1);border-bottom-left-radius:var(--border-radius-medium);border-bottom-right-radius:var(--border-radius-medium)}.group-header{display:flex;padding:8px 8px 4px;align-items:flex-start;gap:10px;align-self:stretch;font:var(--title-01)}\n"] }]
|
|
71
73
|
}], propDecorators: { groups: [{
|
|
72
74
|
type: Input
|
|
73
75
|
}], selectedOption: [{
|
|
@@ -135,5 +137,41 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
|
|
|
135
137
|
};
|
|
136
138
|
}
|
|
137
139
|
SingleSelectComponent.getFilterSync = getFilterSync;
|
|
140
|
+
function getFilterAsync(query$, fetcher, debounceInput = 300) {
|
|
141
|
+
return query$.pipe(debounceTime(debounceInput), switchMap(query => from(fetcher(query)).pipe(map(groups => {
|
|
142
|
+
if (!query)
|
|
143
|
+
return { loading: false, groups };
|
|
144
|
+
const fuseOptions = {
|
|
145
|
+
keys: ['title', 'subtitle'],
|
|
146
|
+
includeMatches: true,
|
|
147
|
+
shouldSort: true,
|
|
148
|
+
threshold: 0.6,
|
|
149
|
+
};
|
|
150
|
+
const filteredGroups = groups
|
|
151
|
+
.map(group => {
|
|
152
|
+
const fuse = new Fuse(group.options, fuseOptions);
|
|
153
|
+
const matches = fuse.search(query);
|
|
154
|
+
return {
|
|
155
|
+
...group,
|
|
156
|
+
options: matches.map(match => ({
|
|
157
|
+
...match.item,
|
|
158
|
+
titleHighlightIndices: [
|
|
159
|
+
...(match.matches?.find(result => result.key === 'title')
|
|
160
|
+
?.indices || []),
|
|
161
|
+
],
|
|
162
|
+
subtitleHighlightIndices: [
|
|
163
|
+
...(match.matches?.find(result => result.key === 'subtitle')?.indices || []),
|
|
164
|
+
],
|
|
165
|
+
})),
|
|
166
|
+
};
|
|
167
|
+
})
|
|
168
|
+
.filter(group => group.options.length > 0); // Remove empty groups
|
|
169
|
+
return { loading: false, groups: filteredGroups };
|
|
170
|
+
}), startWith({ loading: true, groups: [] }))), pairwise(), map(([previous, next]) => ({
|
|
171
|
+
loading: next.loading,
|
|
172
|
+
groups: next.loading ? previous.groups : next.groups,
|
|
173
|
+
})), startWith({ loading: false, groups: [] }));
|
|
174
|
+
}
|
|
175
|
+
SingleSelectComponent.getFilterAsync = getFilterAsync;
|
|
138
176
|
})(SingleSelectComponent || (SingleSelectComponent = {}));
|
|
139
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"single-select.component.js","sourceRoot":"","sources":["../../../../../../projects/riv/src/lib/input/single-select/single-select.component.ts","../../../../../../projects/riv/src/lib/input/single-select/single-select.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component,\n  ElementRef,\n  EventEmitter,\n  Input,\n  Output,\n  TemplateRef,\n  ViewChild,\n} from '@angular/core';\nimport Fuse from 'fuse.js';\nimport { HighlightComponent } from '../../visualization/highlight/highlight.component';\nimport { InputLabelComponent } from '../input-label/input-label.component';\n\n@Component({\n  selector: 'riv-single-select',\n  templateUrl: './single-select.component.html',\n  styleUrls: ['./single-select.component.css'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class SingleSelectComponent<\n  T extends SingleSelectComponent.OptionGroup<U>,\n  U extends SingleSelectComponent.Option,\n> extends InputLabelComponent {\n  @Input()\n  groups: T[] = [];\n\n  @Input()\n  selectedOption: U | null = null;\n\n  @Input()\n  filterabilityOptions: SingleSelectComponent.FilterabilityOptions = {\n    enabled: false,\n  };\n\n  @Input()\n  loading: boolean = false;\n\n  @Input()\n  locked: boolean = false;\n\n  @Input()\n  maxCalloutHeight: string = '50vh';\n\n  @Input()\n  noOptionsMessage: string = 'No available options';\n\n  @Input()\n  nodeTemplate?: TemplateRef<unknown>;\n\n  @Input()\n  triggerTemplate?: TemplateRef<unknown>;\n\n  @Input()\n  placeholder: string = 'Select...';\n\n  @Input()\n  disabled: boolean = false;\n\n  filterQuery: string = '';\n\n  @Output()\n  filterQueryChange = new EventEmitter<string>();\n\n  @Output()\n  selectedOptionChange = new EventEmitter<U>();\n\n  @ViewChild('customTriggerButton') customTriggerButton?: ElementRef;\n  @ViewChild('standardTriggerButton') standardTriggerButton?: ElementRef;\n\n  getTrigger(): Element | null {\n    return (\n      this.customTriggerButton?.nativeElement ??\n      this.standardTriggerButton?.nativeElement\n    );\n  }\n\n  getFilterPlaceholder(): string {\n    if (\n      this.filterabilityOptions.enabled &&\n      this.filterabilityOptions.placeholder\n    ) {\n      return this.filterabilityOptions.placeholder;\n    }\n    return 'Filter options...';\n  }\n\n  allowedOpen(): void {\n    if (!this.disabled && !this.locked) {\n      this.open = true;\n    }\n  }\n\n  public close(): void {\n    this.open = false;\n  }\n\n  selectOption(newlySelectedOption: U) {\n    this.selectedOptionChange.emit(newlySelectedOption);\n    this.open = false;\n\n    // Clear filter after an option is selected.\n    if (this.filterabilityOptions.enabled) {\n      this.filterQuery = '';\n      this.filterQueryChange.emit(this.filterQuery);\n    }\n  }\n\n  trackByHeader(i: number, group: SingleSelectComponent.OptionGroup<U>) {\n    return `${i}${group.header}`;\n  }\n\n  trackByOption(_: number, option: U) {\n    return option.id;\n  }\n\n  open: boolean = false;\n}\n\nexport namespace SingleSelectComponent {\n  export type Option = {\n    id: string;\n    title: string;\n    subtitle?: string;\n    disabled?: boolean;\n    titleHighlightIndices?: HighlightComponent.HighlightIndices[];\n    subtitleHighlightIndices?: HighlightComponent.HighlightIndices[];\n  };\n\n  export type OptionGroup<O extends Option> = {\n    header?: string;\n    options: O[];\n  };\n\n  export type FilterabilityOptions =\n    | {\n        enabled: false;\n      }\n    | {\n        enabled: true;\n        placeholder?: string;\n      };\n\n  export function getFilterSync<T extends OptionGroup<U>, U extends Option>(\n    groups: T[],\n    exact: boolean = false,\n  ): (query: string) => T[] {\n    return function filter(query: string) {\n      if (!query) return groups;\n\n      const fuseOptions = {\n        keys: ['title', 'subtitle'],\n        includeMatches: true,\n        shouldSort: true,\n        threshold: exact ? 0.0 : 0.6,\n      };\n\n      return groups\n        .map(group => {\n          const fuse = new Fuse(group.options, fuseOptions);\n          const matches = fuse.search(query);\n\n          return {\n            ...group,\n            options: matches.map(match => ({\n              ...match.item,\n              titleHighlightIndices: [\n                ...(match.matches?.find(result => result.key === 'title')\n                  ?.indices || []),\n              ],\n              subtitleHighlightIndices: [\n                ...(match.matches?.find(result => result.key === 'subtitle')\n                  ?.indices || []),\n              ],\n            })),\n          };\n        })\n        .filter(group => group.options.length > 0); // Remove empty groups\n    };\n  }\n}\n","<ng-container *ngIf=\"triggerTemplate; else standardTrigger\">\n  <button\n    #customTriggerButton\n    (click)=\"allowedOpen()\"\n    [disabled]=\"disabled || locked\"\n    type=\"button\"\n  >\n    <ng-container\n      [ngTemplateOutlet]=\"triggerTemplate\"\n      [ngTemplateOutletContext]=\"{ selectedOption }\"\n    ></ng-container>\n  </button>\n</ng-container>\n<ng-template #standardTrigger>\n  <riv-input-label\n    [label]=\"label\"\n    [help]=\"help\"\n    [required]=\"required\"\n    [labelActionText]=\"labelActionText\"\n    (labelAction)=\"labelAction.emit($event)\"\n  >\n    <button\n      #standardTriggerButton\n      class=\"trigger\"\n      (click)=\"allowedOpen()\"\n      [disabled]=\"disabled || locked\"\n      type=\"button\"\n    >\n      <ng-container *ngIf=\"selectedOption; else placeholderValue\">\n        <span class=\"value\">{{ selectedOption.title }}</span>\n      </ng-container>\n      <ng-template #placeholderValue>\n        <span class=\"value placeholder\">{{ placeholder }}</span>\n      </ng-template>\n      <span class=\"chevron\">\n        <riv-icon *ngIf=\"!locked\" [name]=\"'ChevronDown'\" [size]=\"16\"></riv-icon>\n        <riv-icon *ngIf=\"locked\" [name]=\"'Lock'\" [size]=\"16\"></riv-icon>\n      </span>\n    </button>\n  </riv-input-label>\n</ng-template>\n\n<riv-callout\n  *ngIf=\"open\"\n  [anchor]=\"getTrigger()\"\n  [theme]=\"'light'\"\n  [showCaret]=\"false\"\n  [allowedPositions]=\"[\n    'top-left',\n    'top-center',\n    'top-right',\n    'bottom-right',\n    'bottom-center',\n    'bottom-left'\n  ]\"\n  (close)=\"open = false\"\n>\n  <input\n    *ngIf=\"filterabilityOptions.enabled\"\n    #filter\n    class=\"filter\"\n    [placeholder]=\"getFilterPlaceholder()\"\n    [value]=\"filterQuery\"\n    (input)=\"filterQuery = filter.value; filterQueryChange.emit(filterQuery)\"\n    [rivFocusOnInit]=\"true\"\n  />\n  <div class=\"options\" [style.max-height]=\"maxCalloutHeight\">\n    <riv-loading-cover [loading]=\"loading\">\n      <ng-container *ngFor=\"let group of groups; trackBy: trackByHeader\">\n        <ng-container *ngIf=\"group.header\">\n          <span class=\"group-header\">\n            {{ group.header }}\n          </span>\n        </ng-container>\n        <ng-container *ngIf=\"group.options; let nodes\">\n          <ng-container *ngIf=\"nodes.length > 0; else empty\">\n            <ng-container *ngFor=\"let node of nodes; trackBy: trackByOption\">\n              <ng-container *ngIf=\"nodeTemplate; else standardTemplate\">\n                <button\n                  class=\"custom-single-select-node\"\n                  [disabled]=\"node.disabled\"\n                  (click)=\"selectOption(node)\"\n                  type=\"button\"\n                >\n                  <ng-container\n                    [ngTemplateOutlet]=\"nodeTemplate\"\n                    [ngTemplateOutletContext]=\"{ node }\"\n                  ></ng-container>\n                </button>\n              </ng-container>\n              <ng-template #standardTemplate>\n                <button\n                  class=\"single-select-node\"\n                  [class.selected]=\"node?.id === selectedOption?.id\"\n                  [class.disabled]=\"node?.disabled\"\n                  [disabled]=\"node?.disabled\"\n                  (click)=\"selectOption(node)\"\n                  type=\"button\"\n                >\n                  <riv-icon\n                    [name]=\"'Check'\"\n                    *ngIf=\"node?.id === selectedOption?.id\"\n                    [size]=\"16\"\n                  ></riv-icon>\n                  <span class=\"label\">\n                    <span class=\"label-title\">\n                      <riv-highlight\n                        [text]=\"node?.title || ''\"\n                        [indices]=\"node?.titleHighlightIndices || []\"\n                      ></riv-highlight>\n                    </span>\n                    <span *ngIf=\"node?.subtitle\" class=\"label-subtitle\">\n                      <riv-highlight\n                        [text]=\"node?.subtitle || ''\"\n                        [indices]=\"node?.subtitleHighlightIndices || []\"\n                      ></riv-highlight>\n                    </span>\n                  </span>\n                </button>\n              </ng-template>\n            </ng-container>\n          </ng-container>\n          <ng-template #empty>\n            <div class=\"empty\">\n              {{ noOptionsMessage }}\n            </div>\n          </ng-template>\n        </ng-container>\n      </ng-container>\n    </riv-loading-cover>\n  </div>\n  <div class=\"footer\">\n    <ng-content select=\"[footer]\"> </ng-content>\n  </div>\n</riv-callout>\n"]}
|
|
177
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"single-select.component.js","sourceRoot":"","sources":["../../../../../../projects/riv/src/lib/input/single-select/single-select.component.ts","../../../../../../projects/riv/src/lib/input/single-select/single-select.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component,\n  ElementRef,\n  EventEmitter,\n  Input,\n  Output,\n  TemplateRef,\n  ViewChild,\n} from '@angular/core';\nimport Fuse from 'fuse.js';\nimport {\n  debounceTime,\n  from,\n  map,\n  Observable,\n  pairwise,\n  startWith,\n  switchMap,\n} from 'rxjs';\nimport { HighlightComponent } from '../../visualization/highlight/highlight.component';\nimport { InputLabelComponent } from '../input-label/input-label.component';\n\n@Component({\n  selector: 'riv-single-select',\n  templateUrl: './single-select.component.html',\n  styleUrls: ['./single-select.component.css'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class SingleSelectComponent<\n  T extends SingleSelectComponent.OptionGroup<U>,\n  U extends SingleSelectComponent.Option,\n> extends InputLabelComponent {\n  @Input()\n  groups: T[] = [];\n\n  @Input()\n  selectedOption: U | null = null;\n\n  @Input()\n  filterabilityOptions: SingleSelectComponent.FilterabilityOptions = {\n    enabled: false,\n  };\n\n  @Input()\n  loading: boolean = false;\n\n  @Input()\n  locked: boolean = false;\n\n  @Input()\n  maxCalloutHeight: string = '50vh';\n\n  @Input()\n  noOptionsMessage: string = 'No available options';\n\n  @Input()\n  nodeTemplate?: TemplateRef<unknown>;\n\n  @Input()\n  triggerTemplate?: TemplateRef<unknown>;\n\n  @Input()\n  placeholder: string = 'Select...';\n\n  @Input()\n  disabled: boolean = false;\n\n  filterQuery: string = '';\n\n  @Output()\n  filterQueryChange = new EventEmitter<string>();\n\n  @Output()\n  selectedOptionChange = new EventEmitter<U>();\n\n  @ViewChild('customTriggerButton') customTriggerButton?: ElementRef;\n  @ViewChild('standardTriggerButton') standardTriggerButton?: ElementRef;\n\n  getTrigger(): Element | null {\n    return (\n      this.customTriggerButton?.nativeElement ??\n      this.standardTriggerButton?.nativeElement\n    );\n  }\n\n  getFilterPlaceholder(): string {\n    if (\n      this.filterabilityOptions.enabled &&\n      this.filterabilityOptions.placeholder\n    ) {\n      return this.filterabilityOptions.placeholder;\n    }\n    return 'Filter options...';\n  }\n\n  allowedOpen(): void {\n    if (!this.disabled && !this.locked) {\n      this.open = true;\n    }\n  }\n\n  public close(): void {\n    this.open = false;\n  }\n\n  selectOption(newlySelectedOption: U) {\n    this.selectedOptionChange.emit(newlySelectedOption);\n    this.open = false;\n\n    // Clear filter after an option is selected.\n    if (this.filterabilityOptions.enabled) {\n      this.filterQuery = '';\n      this.filterQueryChange.emit(this.filterQuery);\n    }\n  }\n\n  trackByHeader(i: number, group: SingleSelectComponent.OptionGroup<U>) {\n    return `${i}${group.header}`;\n  }\n\n  trackByOption(_: number, option: U) {\n    return option.id;\n  }\n\n  open: boolean = false;\n}\n\nexport namespace SingleSelectComponent {\n  export type Option = {\n    id: string;\n    title: string;\n    subtitle?: string;\n    disabled?: boolean;\n    titleHighlightIndices?: HighlightComponent.HighlightIndices[];\n    subtitleHighlightIndices?: HighlightComponent.HighlightIndices[];\n  };\n\n  export type OptionGroup<O extends Option> = {\n    header?: string;\n    options: O[];\n  };\n\n  export type FilterabilityOptions =\n    | {\n        enabled: false;\n      }\n    | {\n        enabled: true;\n        placeholder?: string;\n      };\n\n  export function getFilterSync<T extends OptionGroup<U>, U extends Option>(\n    groups: T[],\n    exact: boolean = false,\n  ): (query: string) => T[] {\n    return function filter(query: string) {\n      if (!query) return groups;\n\n      const fuseOptions = {\n        keys: ['title', 'subtitle'],\n        includeMatches: true,\n        shouldSort: true,\n        threshold: exact ? 0.0 : 0.6,\n      };\n\n      return groups\n        .map(group => {\n          const fuse = new Fuse(group.options, fuseOptions);\n          const matches = fuse.search(query);\n\n          return {\n            ...group,\n            options: matches.map(match => ({\n              ...match.item,\n              titleHighlightIndices: [\n                ...(match.matches?.find(result => result.key === 'title')\n                  ?.indices || []),\n              ],\n              subtitleHighlightIndices: [\n                ...(match.matches?.find(result => result.key === 'subtitle')\n                  ?.indices || []),\n              ],\n            })),\n          };\n        })\n        .filter(group => group.options.length > 0); // Remove empty groups\n    };\n  }\n\n  export function getFilterAsync<T extends OptionGroup<U>, U extends Option>(\n    query$: Observable<string>,\n    fetcher: (query: string) => Promise<T[]>,\n    debounceInput: number = 300,\n  ): Observable<{\n    loading: boolean;\n    groups: T[];\n  }> {\n    return query$.pipe(\n      debounceTime(debounceInput),\n      switchMap(query =>\n        from(fetcher(query)).pipe(\n          map(groups => {\n            if (!query) return { loading: false, groups };\n\n            const fuseOptions = {\n              keys: ['title', 'subtitle'],\n              includeMatches: true,\n              shouldSort: true,\n              threshold: 0.6,\n            };\n\n            const filteredGroups = groups\n              .map(group => {\n                const fuse = new Fuse(group.options, fuseOptions);\n                const matches = fuse.search(query);\n\n                return {\n                  ...group,\n                  options: matches.map(match => ({\n                    ...match.item,\n                    titleHighlightIndices: [\n                      ...(match.matches?.find(result => result.key === 'title')\n                        ?.indices || []),\n                    ],\n                    subtitleHighlightIndices: [\n                      ...(match.matches?.find(\n                        result => result.key === 'subtitle',\n                      )?.indices || []),\n                    ],\n                  })),\n                };\n              })\n              .filter(group => group.options.length > 0); // Remove empty groups\n\n            return { loading: false, groups: filteredGroups };\n          }),\n          startWith({ loading: true, groups: [] }),\n        ),\n      ),\n      pairwise(),\n      map(([previous, next]) => ({\n        loading: next.loading,\n        groups: next.loading ? previous.groups : next.groups,\n      })),\n      startWith({ loading: false, groups: [] }),\n    );\n  }\n}\n","<ng-container *ngIf=\"triggerTemplate; else standardTrigger\">\n  <button\n    #customTriggerButton\n    (click)=\"allowedOpen()\"\n    [disabled]=\"disabled || locked\"\n    type=\"button\"\n  >\n    <ng-container\n      [ngTemplateOutlet]=\"triggerTemplate\"\n      [ngTemplateOutletContext]=\"{ selectedOption }\"\n    ></ng-container>\n  </button>\n</ng-container>\n<ng-template #standardTrigger>\n  <riv-input-label\n    [label]=\"label\"\n    [help]=\"help\"\n    [required]=\"required\"\n    [labelActionText]=\"labelActionText\"\n    (labelAction)=\"labelAction.emit($event)\"\n  >\n    <button\n      #standardTriggerButton\n      class=\"trigger\"\n      (click)=\"allowedOpen()\"\n      [disabled]=\"disabled || locked\"\n      type=\"button\"\n    >\n      <ng-container *ngIf=\"selectedOption; else placeholderValue\">\n        <span class=\"value\">{{ selectedOption.title }}</span>\n      </ng-container>\n      <ng-template #placeholderValue>\n        <span class=\"value placeholder\">{{ placeholder }}</span>\n      </ng-template>\n      <span class=\"chevron\">\n        <riv-icon *ngIf=\"!locked\" [name]=\"'ChevronDown'\" [size]=\"16\"></riv-icon>\n        <riv-icon *ngIf=\"locked\" [name]=\"'Lock'\" [size]=\"16\"></riv-icon>\n      </span>\n    </button>\n  </riv-input-label>\n</ng-template>\n\n<riv-callout\n  *ngIf=\"open\"\n  [anchor]=\"getTrigger()\"\n  [theme]=\"'light'\"\n  [showCaret]=\"false\"\n  [allowedPositions]=\"[\n    'top-left',\n    'top-center',\n    'top-right',\n    'bottom-right',\n    'bottom-center',\n    'bottom-left'\n  ]\"\n  (close)=\"open = false\"\n>\n  <input\n    *ngIf=\"filterabilityOptions.enabled\"\n    #filter\n    class=\"filter\"\n    [placeholder]=\"getFilterPlaceholder()\"\n    [value]=\"filterQuery\"\n    (input)=\"filterQuery = filter.value; filterQueryChange.emit(filterQuery)\"\n    [rivFocusOnInit]=\"true\"\n  />\n  <div class=\"options\" [style.max-height]=\"maxCalloutHeight\">\n    <riv-loading-cover [loading]=\"loading\">\n      <riv-zero-state\n        *ngIf=\"!groups.length\"\n        [message]=\"noOptionsMessage\"\n      ></riv-zero-state>\n      <ng-container *ngFor=\"let group of groups; trackBy: trackByHeader\">\n        <ng-container *ngIf=\"group.header\">\n          <span class=\"group-header\">\n            {{ group.header }}\n          </span>\n        </ng-container>\n        <ng-container *ngIf=\"group.options; let nodes\">\n          <ng-container *ngIf=\"nodes.length > 0; else empty\">\n            <ng-container *ngFor=\"let node of nodes; trackBy: trackByOption\">\n              <ng-container *ngIf=\"nodeTemplate; else standardTemplate\">\n                <button\n                  class=\"custom-single-select-node\"\n                  [disabled]=\"node.disabled\"\n                  (click)=\"selectOption(node)\"\n                  type=\"button\"\n                >\n                  <ng-container\n                    [ngTemplateOutlet]=\"nodeTemplate\"\n                    [ngTemplateOutletContext]=\"{ node }\"\n                  ></ng-container>\n                </button>\n              </ng-container>\n              <ng-template #standardTemplate>\n                <button\n                  class=\"single-select-node\"\n                  [class.selected]=\"node?.id === selectedOption?.id\"\n                  [class.disabled]=\"node?.disabled\"\n                  [disabled]=\"node?.disabled\"\n                  (click)=\"selectOption(node)\"\n                  type=\"button\"\n                >\n                  <riv-icon\n                    [name]=\"'Check'\"\n                    *ngIf=\"node?.id === selectedOption?.id\"\n                    [size]=\"16\"\n                  ></riv-icon>\n                  <span class=\"label\">\n                    <span class=\"label-title\">\n                      <riv-highlight\n                        [text]=\"node?.title || ''\"\n                        [indices]=\"node?.titleHighlightIndices || []\"\n                      ></riv-highlight>\n                    </span>\n                    <span *ngIf=\"node?.subtitle\" class=\"label-subtitle\">\n                      <riv-highlight\n                        [text]=\"node?.subtitle || ''\"\n                        [indices]=\"node?.subtitleHighlightIndices || []\"\n                      ></riv-highlight>\n                    </span>\n                  </span>\n                </button>\n              </ng-template>\n            </ng-container>\n          </ng-container>\n          <ng-template #empty>\n            <div class=\"empty\">\n              {{ noOptionsMessage }}\n            </div>\n          </ng-template>\n        </ng-container>\n      </ng-container>\n      <ng-content select=\"[postOptionList]\"></ng-content>\n    </riv-loading-cover>\n  </div>\n  <div class=\"footer\">\n    <ng-content select=\"[footer]\"></ng-content>\n  </div>\n</riv-callout>\n"]}
|