@nova-ui/bits
Version:
SolarWinds Nova Framework
273 lines • 49.3 kB
JavaScript
// © 2022 SolarWinds Worldwide, LLC. All rights reserved.
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to
// deal in the Software without restriction, including without limitation the
// rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
// sell copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
import { Component, ElementRef, EventEmitter, forwardRef, Input, Output, ViewChild, ViewEncapsulation, } from "@angular/core";
import { NG_VALIDATORS, NG_VALUE_ACCESSOR, } from "@angular/forms";
import isNumber from "lodash/isNumber";
import isUndefined from "lodash/isUndefined";
import round from "lodash/round";
import { KEYBOARD_CODE } from "../../../constants/keycode.constants";
import { regexpValidation } from "../../../constants/regex.constants";
import { NuiFormFieldControl } from "../../form-field/public-api";
import * as i0 from "@angular/core";
import * as i1 from "@angular/common";
import * as i2 from "../../button/button.component";
import * as i3 from "@angular/forms";
// <example-url>./../examples/index.html#/textbox/textbox-number</example-url>
export class TextboxNumberComponent {
constructor() {
/**
* Value that is exposed through ControlValueAccessor
*/
this.value = null;
/**
* The option to disable the textboxNumber.
*/
this.disabled = false;
/**
* Value used as a placeholder for the text box.
*/
this.placeholder = "";
/**
* The option to make the textboxNumber read only.
*/
this.readonly = false;
/**
* Step by which the value are increased/decreased
* when clicking on up/down buttons
*/
this.step = 1;
/**
* The decimal precision to use for rounding each step increase/decrease
* when clicking the up/down buttons
*/
this.stepPrecision = 10;
/**
* Input to apply error state styles
*/
this.isInErrorState = false; // TODO: do we need to hook this up?
/**
* Input to set aria label text
*/
this.ariaLabel = "Textbox number input";
/**
* Event fired when textBoxNumber is focused out.
*/
this.blurred = new EventEmitter();
this.valueChange = new EventEmitter();
this.isRepeatable = true;
}
static createRangeValidator(min, max) {
const rangeValidator = (c) => {
if (!c.value && c.value !== 0) {
return null;
}
if ((!isUndefined(max) && c.value > max) ||
(!isUndefined(min) && c.value < min)) {
return {
range: {
given: c.value,
max,
min,
},
};
}
return null;
};
return rangeValidator;
}
ngOnChanges(changes) {
if (changes["minValue"] || changes["maxValue"]) {
this.validatorFn = TextboxNumberComponent.createRangeValidator(this.minValue, this.maxValue);
}
}
onBlurEventEmit() {
this.onTouched();
this.blurred.emit(this.value);
}
addNumber(valueChange) {
if (isUndefined(this.value) || isNaN(this.value)) {
this.value = 0;
}
// Explicitly converting current value to number because it can also be a string, and cause issues, like NUI-5599
const newValue = this.clampToRange(Number(this.value) + valueChange);
this.onValueChange(round(newValue, this.stepPrecision));
}
onValueChange(value) {
this.value = value;
setTimeout(() => {
this.onChange(this.value);
this.valueChange.emit(this.value);
this.onTouched();
});
}
onChange(value) { }
onTouched() { }
writeValue(value) {
this.value = value;
}
registerOnChange(fn) {
this.onChange = fn;
}
registerOnTouched(fn) {
this.onTouched = fn;
}
setDisabledState(isDisabled) {
this.disabled = isDisabled;
}
compareMin() {
return this.value <= this.minValue;
}
compareMax() {
return this.value >= this.maxValue;
}
validate(c) {
this.formControl = c;
return ((this.validatorFn ? this.validatorFn(c) : null) ||
this.nativeValidator());
}
hasError() {
return ((this.formControl?.touched || this.formControl?.dirty) &&
!this.formControl?.valid);
}
nativeValidator() {
if (this.input.nativeElement.validity.valid) {
return null;
}
return { format: "Invalid input" };
}
clampToRange(value) {
if (!isNumber(value)) {
return value;
}
if (value > this.maxValue) {
return this.maxValue;
}
if (value < this.minValue) {
return this.minValue;
}
return value;
}
// We need to restrict user input because safari does not prevent non numerical input in type 'number'
preventNonNumericalPaste(event) {
const pastedString = event.clipboardData?.getData("text/plain");
if (!pastedString) {
throw new Error("Unable to parse clipboardData");
}
this.preventDefaultEventBehavior(event, pastedString, regexpValidation.multiCharNumeric);
}
preventNonNumericalInput(event) {
const inputString = event.key;
if (!this.isMetaKey(event)) {
this.preventDefaultEventBehavior(event, inputString, regexpValidation.singleCharNumeric);
}
}
preventDefaultEventBehavior(event, inputString, regexp) {
if (!inputString?.match(regexp)) {
event.preventDefault();
}
}
isMetaKey(event) {
return (event.ctrlKey ||
event.metaKey ||
event.code === KEYBOARD_CODE.BACKSPACE ||
event.code === KEYBOARD_CODE.DELETE ||
event.code === KEYBOARD_CODE.TAB ||
event.code === KEYBOARD_CODE.ARROW_DOWN ||
event.code === KEYBOARD_CODE.ARROW_LEFT ||
event.code === KEYBOARD_CODE.ARROW_RIGHT ||
event.code === KEYBOARD_CODE.ARROW_UP);
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TextboxNumberComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: TextboxNumberComponent, selector: "nui-textbox-number", inputs: { value: "value", customBoxWidth: "customBoxWidth", disabled: "disabled", name: "name", placeholder: "placeholder", readonly: "readonly", step: "step", stepPrecision: "stepPrecision", isInErrorState: "isInErrorState", ariaLabel: "ariaLabel", minValue: "minValue", maxValue: "maxValue" }, outputs: { blurred: "blurred", valueChange: "valueChange" }, host: { attributes: { "role": "spinbutton" }, properties: { "attr.aria-label": "ariaLabel", "attr.aria-valuemin": "minValue || null", "attr.aria-valuemax": "maxValue || null", "attr.aria-valuenow": "value || 0" } }, providers: [
{
provide: NG_VALUE_ACCESSOR,
useExisting: forwardRef(() => TextboxNumberComponent),
multi: true,
},
{
provide: NG_VALIDATORS,
useExisting: forwardRef(() => TextboxNumberComponent),
multi: true,
},
{
provide: NuiFormFieldControl,
useExisting: forwardRef(() => TextboxNumberComponent),
multi: true,
},
], viewQueries: [{ propertyName: "input", first: true, predicate: ["numberInput"], descendants: true, static: true }], usesOnChanges: true, ngImport: i0, template: "<div\n class=\"nui-textbox nui-textbox-number\"\n [class.nui-textbox--disabled]=\"disabled\"\n [class.has-error]=\"hasError() || isInErrorState\"\n>\n <div\n class=\"has-feedback\"\n [ngStyle]=\"{ maxWidth: customBoxWidth, minWidth: customBoxWidth }\"\n >\n <input\n class=\"form-control input-control nui-textbox__input\"\n #numberInput\n step=\"any\"\n [disabled]=\"disabled\"\n type=\"number\"\n [ngModel]=\"value\"\n (ngModelChange)=\"onValueChange($event)\"\n [ngModelOptions]=\"{ standalone: true }\"\n [min]=\"minValue\"\n [max]=\"maxValue\"\n [attr.name]=\"name\"\n (blur)=\"onBlurEventEmit()\"\n (keydown)=\"preventNonNumericalInput($event)\"\n (paste)=\"preventNonNumericalPaste($event)\"\n [placeholder]=\"placeholder\"\n [readonly]=\"readonly\"\n aria-label=\"Textbox number input\"\n />\n </div>\n <span class=\"nui-textbox-number__buttons\">\n <span class=\"nui-textbox-number__updown nui-textbox-number__up-button\">\n <button\n nui-button\n type=\"button\"\n displayStyle=\"action\"\n icon=\"caret-up\"\n [isRepeat]=\"isRepeatable\"\n [disabled]=\"readonly || disabled || compareMax()\"\n (click)=\"addNumber(+step)\"\n aria-label=\"Up\"\n ></button>\n </span>\n <span\n class=\"nui-textbox-number__updown nui-textbox-number__updown-divider\"\n ></span>\n <span\n class=\"nui-textbox-number__updown nui-textbox-number__down-button\"\n >\n <button\n nui-button\n type=\"button\"\n displayStyle=\"action\"\n icon=\"caret-down\"\n [isRepeat]=\"isRepeatable\"\n [disabled]=\"readonly || disabled || compareMin()\"\n (click)=\"addNumber(-step)\"\n aria-label=\"Down\"\n ></button>\n </span>\n </span>\n</div>\n", styles: [".nui .nui-textbox--disabled .has-feedback .input-control,.nui .nui-textbox--disabled .has-feedback .input-control:hover{color:var(--nui-color-text-disabled,rgba(17, 17, 17, .3))}.nui .nui-textbox--bold .nui-textbox__input{line-height:18px;font-size:13px;font-weight:600;color:#111;color:var(--nui-color-text-default,#111)}.nui .nui-textbox__input{min-height:30px}.nui .nui-textbox .has-feedback{position:relative}.nui .nui-textbox .has-feedback .input-control{padding:5px 10px;overflow-y:hidden}.nui .nui-textbox .has-feedback .input-control.nui-textbox__input--multiline{padding-top:5px}.nui .nui-textbox .has-feedback .input-control:focus{outline:none;z-index:500;box-shadow:0 0 0 2px #00c4d280}.nui .nui .nui-textbox--centered .has-feedback .input-control{padding:0;text-align:center}.nui .nui-textbox .input-control{resize:vertical}.nui .nui-textbox .input-control[hidden]{display:none!important}.nui .nui-textbox.has-error .help-block,.nui .nui-textbox.has-error .control-label,.nui .nui-textbox.has-error .radio,.nui .nui-textbox.has-error .checkbox,.nui .nui-textbox.has-error .radio-inline,.nui .nui-textbox.has-error .checkbox-inline,.nui .nui-textbox.has-error.nui-radio label,.nui .nui-textbox.has-error.checkbox label,.nui .nui-textbox.has-error.radio-inline label,.nui .nui-textbox.has-error.checkbox-inline label{color:var(--nui-color-text-default,#111)}.nui .nui-textbox.has-error .nui-validation>div{padding:5px 0;color:var(--nui-color-text-default,#111)}.nui .nui-textbox.has-error .input-group-addon{color:var(--nui-color-text-default,#111);border-color:var(--nui-color-line-critical,#dd2c00);background-color:var(--nui-color-bg-content,#fff)}.nui .nui-textbox.has-error .form-control-feedback{color:var(--nui-color-text-default,#111)}.nui .nui-textbox .form-control-feedback{top:12px;margin:0 7px 0 0;position:absolute;right:0;z-index:2;display:block;width:30px;height:30px;line-height:30px;text-align:center;pointer-events:none}.nui .nui-textbox .form-control-feedback.nui-textbox__busy{top:0;margin:0 1px 0 0}.nui .nui-textbox .form-control-feedback svg{position:relative;bottom:3px}.nui .textarea-scrollable .has-feedback .input-control{overflow-y:auto}\n", ".nui .nui-textbox input[type=number]::-webkit-inner-spin-button{display:none}.nui .nui-textbox input[type=number]{-moz-appearance:textfield}.nui .nui-textbox--disabled .nui-textbox-number__buttons,.nui .nui-textbox--disabled .nui-textbox-number__updown-divider{border-color:var(--nui-color-disabled-light,#e9e9e9)}.nui .nui-textbox-number{display:flex}.nui .nui-textbox-number__buttons{display:inline-flex;height:30px;border:solid 1px;border-color:var(--nui-color-line-default,#d9d9d9);border-top-right-radius:3px;border-bottom-right-radius:3px;border-left:none}.nui .nui-textbox-number__updown{display:block;line-height:1;height:30px;top:1px;text-align:center}.nui .nui-textbox-number__updown button.nui-button{padding-left:5px;padding-top:7px;width:30px;height:28px;min-height:28px}.nui .nui-textbox-number__updown button.nui-button:focus{outline:none}.nui .nui-textbox-number__updown-divider{z-index:1;height:28px;top:1px;right:30px;width:0;border-right:solid 1px;border-right-color:var(--nui-color-line-default,#d9d9d9)}.nui .nui-textbox-number__up-button{right:1px}.nui .nui-textbox-number__down-button{right:30px}.nui .nui-textbox-number .has-feedback{flex:1}.nui .nui-textbox-number .input-control{border-top-right-radius:0;border-bottom-right-radius:0}.nui .nui-textbox-number input[type=number]~nui-icon>.form-control-feedback{margin:0 65px 0 0}.nui .nui-textbox-number input[type=number]~nui-icon>.form-control-feedback.nui-textbox__busy{top:0;margin:0 59px 0 0}\n"], dependencies: [{ kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "component", type: i2.ButtonComponent, selector: "[nui-button]", inputs: ["displayStyle", "icon", "iconColor", "iconRight", "isBusy", "isEmpty", "ariaLabel", "isRepeat", "size"] }, { kind: "directive", type: i3.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i3.NumberValueAccessor, selector: "input[type=number][formControlName],input[type=number][formControl],input[type=number][ngModel]" }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.MinValidator, selector: "input[type=number][min][formControlName],input[type=number][min][formControl],input[type=number][min][ngModel]", inputs: ["min"] }, { kind: "directive", type: i3.MaxValidator, selector: "input[type=number][max][formControlName],input[type=number][max][formControl],input[type=number][max][ngModel]", inputs: ["max"] }, { kind: "directive", type: i3.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }], encapsulation: i0.ViewEncapsulation.None }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TextboxNumberComponent, decorators: [{
type: Component,
args: [{ selector: "nui-textbox-number", providers: [
{
provide: NG_VALUE_ACCESSOR,
useExisting: forwardRef(() => TextboxNumberComponent),
multi: true,
},
{
provide: NG_VALIDATORS,
useExisting: forwardRef(() => TextboxNumberComponent),
multi: true,
},
{
provide: NuiFormFieldControl,
useExisting: forwardRef(() => TextboxNumberComponent),
multi: true,
},
], encapsulation: ViewEncapsulation.None, host: {
role: "spinbutton",
"[attr.aria-label]": "ariaLabel",
"[attr.aria-valuemin]": "minValue || null",
"[attr.aria-valuemax]": "maxValue || null",
"[attr.aria-valuenow]": "value || 0",
}, template: "<div\n class=\"nui-textbox nui-textbox-number\"\n [class.nui-textbox--disabled]=\"disabled\"\n [class.has-error]=\"hasError() || isInErrorState\"\n>\n <div\n class=\"has-feedback\"\n [ngStyle]=\"{ maxWidth: customBoxWidth, minWidth: customBoxWidth }\"\n >\n <input\n class=\"form-control input-control nui-textbox__input\"\n #numberInput\n step=\"any\"\n [disabled]=\"disabled\"\n type=\"number\"\n [ngModel]=\"value\"\n (ngModelChange)=\"onValueChange($event)\"\n [ngModelOptions]=\"{ standalone: true }\"\n [min]=\"minValue\"\n [max]=\"maxValue\"\n [attr.name]=\"name\"\n (blur)=\"onBlurEventEmit()\"\n (keydown)=\"preventNonNumericalInput($event)\"\n (paste)=\"preventNonNumericalPaste($event)\"\n [placeholder]=\"placeholder\"\n [readonly]=\"readonly\"\n aria-label=\"Textbox number input\"\n />\n </div>\n <span class=\"nui-textbox-number__buttons\">\n <span class=\"nui-textbox-number__updown nui-textbox-number__up-button\">\n <button\n nui-button\n type=\"button\"\n displayStyle=\"action\"\n icon=\"caret-up\"\n [isRepeat]=\"isRepeatable\"\n [disabled]=\"readonly || disabled || compareMax()\"\n (click)=\"addNumber(+step)\"\n aria-label=\"Up\"\n ></button>\n </span>\n <span\n class=\"nui-textbox-number__updown nui-textbox-number__updown-divider\"\n ></span>\n <span\n class=\"nui-textbox-number__updown nui-textbox-number__down-button\"\n >\n <button\n nui-button\n type=\"button\"\n displayStyle=\"action\"\n icon=\"caret-down\"\n [isRepeat]=\"isRepeatable\"\n [disabled]=\"readonly || disabled || compareMin()\"\n (click)=\"addNumber(-step)\"\n aria-label=\"Down\"\n ></button>\n </span>\n </span>\n</div>\n", styles: [".nui .nui-textbox--disabled .has-feedback .input-control,.nui .nui-textbox--disabled .has-feedback .input-control:hover{color:var(--nui-color-text-disabled,rgba(17, 17, 17, .3))}.nui .nui-textbox--bold .nui-textbox__input{line-height:18px;font-size:13px;font-weight:600;color:#111;color:var(--nui-color-text-default,#111)}.nui .nui-textbox__input{min-height:30px}.nui .nui-textbox .has-feedback{position:relative}.nui .nui-textbox .has-feedback .input-control{padding:5px 10px;overflow-y:hidden}.nui .nui-textbox .has-feedback .input-control.nui-textbox__input--multiline{padding-top:5px}.nui .nui-textbox .has-feedback .input-control:focus{outline:none;z-index:500;box-shadow:0 0 0 2px #00c4d280}.nui .nui .nui-textbox--centered .has-feedback .input-control{padding:0;text-align:center}.nui .nui-textbox .input-control{resize:vertical}.nui .nui-textbox .input-control[hidden]{display:none!important}.nui .nui-textbox.has-error .help-block,.nui .nui-textbox.has-error .control-label,.nui .nui-textbox.has-error .radio,.nui .nui-textbox.has-error .checkbox,.nui .nui-textbox.has-error .radio-inline,.nui .nui-textbox.has-error .checkbox-inline,.nui .nui-textbox.has-error.nui-radio label,.nui .nui-textbox.has-error.checkbox label,.nui .nui-textbox.has-error.radio-inline label,.nui .nui-textbox.has-error.checkbox-inline label{color:var(--nui-color-text-default,#111)}.nui .nui-textbox.has-error .nui-validation>div{padding:5px 0;color:var(--nui-color-text-default,#111)}.nui .nui-textbox.has-error .input-group-addon{color:var(--nui-color-text-default,#111);border-color:var(--nui-color-line-critical,#dd2c00);background-color:var(--nui-color-bg-content,#fff)}.nui .nui-textbox.has-error .form-control-feedback{color:var(--nui-color-text-default,#111)}.nui .nui-textbox .form-control-feedback{top:12px;margin:0 7px 0 0;position:absolute;right:0;z-index:2;display:block;width:30px;height:30px;line-height:30px;text-align:center;pointer-events:none}.nui .nui-textbox .form-control-feedback.nui-textbox__busy{top:0;margin:0 1px 0 0}.nui .nui-textbox .form-control-feedback svg{position:relative;bottom:3px}.nui .textarea-scrollable .has-feedback .input-control{overflow-y:auto}\n", ".nui .nui-textbox input[type=number]::-webkit-inner-spin-button{display:none}.nui .nui-textbox input[type=number]{-moz-appearance:textfield}.nui .nui-textbox--disabled .nui-textbox-number__buttons,.nui .nui-textbox--disabled .nui-textbox-number__updown-divider{border-color:var(--nui-color-disabled-light,#e9e9e9)}.nui .nui-textbox-number{display:flex}.nui .nui-textbox-number__buttons{display:inline-flex;height:30px;border:solid 1px;border-color:var(--nui-color-line-default,#d9d9d9);border-top-right-radius:3px;border-bottom-right-radius:3px;border-left:none}.nui .nui-textbox-number__updown{display:block;line-height:1;height:30px;top:1px;text-align:center}.nui .nui-textbox-number__updown button.nui-button{padding-left:5px;padding-top:7px;width:30px;height:28px;min-height:28px}.nui .nui-textbox-number__updown button.nui-button:focus{outline:none}.nui .nui-textbox-number__updown-divider{z-index:1;height:28px;top:1px;right:30px;width:0;border-right:solid 1px;border-right-color:var(--nui-color-line-default,#d9d9d9)}.nui .nui-textbox-number__up-button{right:1px}.nui .nui-textbox-number__down-button{right:30px}.nui .nui-textbox-number .has-feedback{flex:1}.nui .nui-textbox-number .input-control{border-top-right-radius:0;border-bottom-right-radius:0}.nui .nui-textbox-number input[type=number]~nui-icon>.form-control-feedback{margin:0 65px 0 0}.nui .nui-textbox-number input[type=number]~nui-icon>.form-control-feedback.nui-textbox__busy{top:0;margin:0 59px 0 0}\n"] }]
}], propDecorators: { value: [{
type: Input
}], customBoxWidth: [{
type: Input
}], disabled: [{
type: Input
}], name: [{
type: Input
}], placeholder: [{
type: Input
}], readonly: [{
type: Input
}], step: [{
type: Input
}], stepPrecision: [{
type: Input
}], isInErrorState: [{
type: Input
}], ariaLabel: [{
type: Input
}], minValue: [{
type: Input
}], maxValue: [{
type: Input
}], blurred: [{
type: Output
}], valueChange: [{
type: Output
}], input: [{
type: ViewChild,
args: ["numberInput", { static: true }]
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"textbox-number.component.js","sourceRoot":"","sources":["../../../../../src/lib/textbox/textbox-number/textbox-number.component.ts","../../../../../src/lib/textbox/textbox-number/textbox-number.component.html"],"names":[],"mappings":"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","sourcesContent":["// © 2022 SolarWinds Worldwide, LLC. All rights reserved.\n//\n// Permission is hereby granted, free of charge, to any person obtaining a copy\n//  of this software and associated documentation files (the \"Software\"), to\n//  deal in the Software without restriction, including without limitation the\n//  rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\n//  sell copies of the Software, and to permit persons to whom the Software is\n//  furnished to do so, subject to the following conditions:\n//\n// The above copyright notice and this permission notice shall be included in\n//  all copies or substantial portions of the Software.\n//\n// THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n//  IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n//  FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n//  AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n//  LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n//  OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN\n//  THE SOFTWARE.\n\nimport {\n    Component,\n    ElementRef,\n    EventEmitter,\n    forwardRef,\n    Input,\n    OnChanges,\n    Output,\n    SimpleChanges,\n    ViewChild,\n    ViewEncapsulation,\n} from \"@angular/core\";\nimport {\n    ControlValueAccessor,\n    FormControl,\n    NG_VALIDATORS,\n    NG_VALUE_ACCESSOR,\n    ValidationErrors,\n} from \"@angular/forms\";\nimport isNumber from \"lodash/isNumber\";\nimport isUndefined from \"lodash/isUndefined\";\nimport round from \"lodash/round\";\n\nimport { KEYBOARD_CODE } from \"../../../constants/keycode.constants\";\nimport { regexpValidation } from \"../../../constants/regex.constants\";\nimport { NuiFormFieldControl } from \"../../form-field/public-api\";\n\n// <example-url>./../examples/index.html#/textbox/textbox-number</example-url>\n\n@Component({\n    selector: \"nui-textbox-number\",\n    templateUrl: \"./textbox-number.component.html\",\n    providers: [\n        {\n            provide: NG_VALUE_ACCESSOR,\n            useExisting: forwardRef(() => TextboxNumberComponent),\n            multi: true,\n        },\n        {\n            provide: NG_VALIDATORS,\n            useExisting: forwardRef(() => TextboxNumberComponent),\n            multi: true,\n        },\n        {\n            provide: NuiFormFieldControl,\n            useExisting: forwardRef(() => TextboxNumberComponent),\n            multi: true,\n        },\n    ],\n    styleUrls: [\"../textbox.component.less\", \"./textbox-number.component.less\"],\n    encapsulation: ViewEncapsulation.None,\n    host: {\n        role: \"spinbutton\",\n        \"[attr.aria-label]\": \"ariaLabel\",\n        \"[attr.aria-valuemin]\": \"minValue || null\",\n        \"[attr.aria-valuemax]\": \"maxValue || null\",\n        \"[attr.aria-valuenow]\": \"value || 0\",\n    },\n})\nexport class TextboxNumberComponent\n    implements ControlValueAccessor, NuiFormFieldControl, OnChanges\n{\n    private static createRangeValidator(\n        min: number,\n        max: number\n    ): (c: FormControl) => ValidationErrors | null {\n        const rangeValidator = (c: FormControl) => {\n            if (!c.value && c.value !== 0) {\n                return null;\n            }\n\n            if (\n                (!isUndefined(max) && c.value > max) ||\n                (!isUndefined(min) && c.value < min)\n            ) {\n                return {\n                    range: {\n                        given: c.value,\n                        max,\n                        min,\n                    },\n                };\n            }\n\n            return null;\n        };\n\n        return rangeValidator;\n    }\n\n    /**\n     * Value that is exposed through ControlValueAccessor\n     */\n    @Input() public value: any = null;\n\n    /**\n     * Use to set a custom width for the input field.\n     */\n    @Input() public customBoxWidth: string;\n    /**\n     * The option to disable the textboxNumber.\n     */\n    @Input() public disabled = false;\n    /**\n     * Name of the element.\n     */\n    @Input() public name: string;\n    /**\n     * Value used as a placeholder for the text box.\n     */\n    @Input() public placeholder = \"\";\n    /**\n     * The option to make the textboxNumber read only.\n     */\n    @Input() public readonly = false;\n\n    /**\n     * Step by which the value are increased/decreased\n     * when clicking on up/down buttons\n     */\n    @Input() step: number = 1;\n\n    /**\n     * The decimal precision to use for rounding each step increase/decrease\n     * when clicking the up/down buttons\n     */\n    @Input() stepPrecision: number = 10;\n\n    /**\n     * Input to apply error state styles\n     */\n    @Input() public isInErrorState: boolean = false; // TODO: do we need to hook this up?\n\n    /**\n     * Input to set aria label text\n     */\n    @Input() public ariaLabel: string = \"Textbox number input\";\n\n    /**\n     * Minimum value of textBoxNumber component\n     */\n    @Input() public minValue: number;\n    /**\n     * Maximum value of textBoxNumber component\n     */\n    @Input() public maxValue: number;\n    /**\n     * Event fired when textBoxNumber is focused out.\n     */\n    @Output() blurred: EventEmitter<any> = new EventEmitter<any>();\n\n    @Output() valueChange: EventEmitter<number> = new EventEmitter<number>();\n\n    @ViewChild(\"numberInput\", { static: true })\n    private input: ElementRef<HTMLFieldSetElement>;\n\n    public formControl: FormControl;\n\n    public isRepeatable = true;\n\n    private validatorFn: (c: FormControl) => ValidationErrors | null;\n\n    public ngOnChanges(changes: SimpleChanges): void {\n        if (changes[\"minValue\"] || changes[\"maxValue\"]) {\n            this.validatorFn = TextboxNumberComponent.createRangeValidator(\n                this.minValue,\n                this.maxValue\n            );\n        }\n    }\n\n    public onBlurEventEmit(): void {\n        this.onTouched();\n        this.blurred.emit(this.value);\n    }\n\n    public addNumber(valueChange: number): void {\n        if (isUndefined(this.value) || isNaN(this.value)) {\n            this.value = 0;\n        }\n        // Explicitly converting current value to number because it can also be a string, and cause issues, like NUI-5599\n        const newValue = this.clampToRange(Number(this.value) + valueChange);\n\n        this.onValueChange(round(newValue, this.stepPrecision));\n    }\n\n    public onValueChange(value: any): void {\n        this.value = value;\n\n        setTimeout(() => {\n            this.onChange(this.value);\n            this.valueChange.emit(this.value);\n\n            this.onTouched();\n        });\n    }\n\n    public onChange(value: any): void {}\n\n    public onTouched(): void {}\n\n    public writeValue(value: any): void {\n        this.value = value;\n    }\n\n    public registerOnChange(fn: (value: any) => void): void {\n        this.onChange = fn;\n    }\n\n    public registerOnTouched(fn: () => void): void {\n        this.onTouched = fn;\n    }\n\n    public setDisabledState(isDisabled: boolean): void {\n        this.disabled = isDisabled;\n    }\n\n    public compareMin(): boolean {\n        return this.value <= this.minValue;\n    }\n\n    public compareMax(): boolean {\n        return this.value >= this.maxValue;\n    }\n\n    public validate(c: FormControl): ValidationErrors | null {\n        this.formControl = c;\n        return (\n            (this.validatorFn ? this.validatorFn(c) : null) ||\n            this.nativeValidator()\n        );\n    }\n\n    public hasError(): boolean {\n        return (\n            (this.formControl?.touched || this.formControl?.dirty) &&\n            !this.formControl?.valid\n        );\n    }\n\n    private nativeValidator(): ValidationErrors | null {\n        if (this.input.nativeElement.validity.valid) {\n            return null;\n        }\n\n        return { format: \"Invalid input\" };\n    }\n\n    private clampToRange(value: number): number {\n        if (!isNumber(value)) {\n            return value;\n        }\n        if (value > this.maxValue) {\n            return this.maxValue;\n        }\n        if (value < this.minValue) {\n            return this.minValue;\n        }\n        return value;\n    }\n\n    // We need to restrict user input because safari does not prevent non numerical input in type 'number'\n    public preventNonNumericalPaste(event: ClipboardEvent): void {\n        const pastedString: string | undefined =\n            event.clipboardData?.getData(\"text/plain\");\n        if (!pastedString) {\n            throw new Error(\"Unable to parse clipboardData\");\n        }\n        this.preventDefaultEventBehavior(\n            event,\n            pastedString,\n            regexpValidation.multiCharNumeric\n        );\n    }\n\n    public preventNonNumericalInput(event: KeyboardEvent): void {\n        const inputString: string = event.key;\n        if (!this.isMetaKey(event)) {\n            this.preventDefaultEventBehavior(\n                event,\n                inputString,\n                regexpValidation.singleCharNumeric\n            );\n        }\n    }\n\n    private preventDefaultEventBehavior(\n        event: KeyboardEvent | ClipboardEvent,\n        inputString: string,\n        regexp: RegExp\n    ): void {\n        if (!inputString?.match(regexp)) {\n            event.preventDefault();\n        }\n    }\n\n    private isMetaKey(event: KeyboardEvent): boolean {\n        return (\n            event.ctrlKey ||\n            event.metaKey ||\n            event.code === KEYBOARD_CODE.BACKSPACE ||\n            event.code === KEYBOARD_CODE.DELETE ||\n            event.code === KEYBOARD_CODE.TAB ||\n            event.code === KEYBOARD_CODE.ARROW_DOWN ||\n            event.code === KEYBOARD_CODE.ARROW_LEFT ||\n            event.code === KEYBOARD_CODE.ARROW_RIGHT ||\n            event.code === KEYBOARD_CODE.ARROW_UP\n        );\n    }\n}\n","<div\n    class=\"nui-textbox nui-textbox-number\"\n    [class.nui-textbox--disabled]=\"disabled\"\n    [class.has-error]=\"hasError() || isInErrorState\"\n>\n    <div\n        class=\"has-feedback\"\n        [ngStyle]=\"{ maxWidth: customBoxWidth, minWidth: customBoxWidth }\"\n    >\n        <input\n            class=\"form-control input-control nui-textbox__input\"\n            #numberInput\n            step=\"any\"\n            [disabled]=\"disabled\"\n            type=\"number\"\n            [ngModel]=\"value\"\n            (ngModelChange)=\"onValueChange($event)\"\n            [ngModelOptions]=\"{ standalone: true }\"\n            [min]=\"minValue\"\n            [max]=\"maxValue\"\n            [attr.name]=\"name\"\n            (blur)=\"onBlurEventEmit()\"\n            (keydown)=\"preventNonNumericalInput($event)\"\n            (paste)=\"preventNonNumericalPaste($event)\"\n            [placeholder]=\"placeholder\"\n            [readonly]=\"readonly\"\n            aria-label=\"Textbox number input\"\n        />\n    </div>\n    <span class=\"nui-textbox-number__buttons\">\n        <span class=\"nui-textbox-number__updown nui-textbox-number__up-button\">\n            <button\n                nui-button\n                type=\"button\"\n                displayStyle=\"action\"\n                icon=\"caret-up\"\n                [isRepeat]=\"isRepeatable\"\n                [disabled]=\"readonly || disabled || compareMax()\"\n                (click)=\"addNumber(+step)\"\n                aria-label=\"Up\"\n            ></button>\n        </span>\n        <span\n            class=\"nui-textbox-number__updown nui-textbox-number__updown-divider\"\n        ></span>\n        <span\n            class=\"nui-textbox-number__updown nui-textbox-number__down-button\"\n        >\n            <button\n                nui-button\n                type=\"button\"\n                displayStyle=\"action\"\n                icon=\"caret-down\"\n                [isRepeat]=\"isRepeatable\"\n                [disabled]=\"readonly || disabled || compareMin()\"\n                (click)=\"addNumber(-step)\"\n                aria-label=\"Down\"\n            ></button>\n        </span>\n    </span>\n</div>\n"]}