added mat currency input field
This commit is contained in:
parent
6b76e5b905
commit
782500c009
@ -4,6 +4,9 @@
|
||||
"newProjectRoot": "projects",
|
||||
"projects": {
|
||||
"material-mentions": {
|
||||
"i18n": {
|
||||
"sourceLocale": "de"
|
||||
},
|
||||
"projectType": "application",
|
||||
"schematics": {
|
||||
"@schematics/angular:component": {
|
||||
@ -101,4 +104,4 @@
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
6
src/app/currency-input/currency-input.component.html
Normal file
6
src/app/currency-input/currency-input.component.html
Normal file
@ -0,0 +1,6 @@
|
||||
<div>
|
||||
<input #input [value]="formattedValue()" [readonly]="disabled" [attr.placeholder]="placeholder"
|
||||
[attr.aria-describedby]="userAriaDescribedBy" [attr.required]="required" (input)="onUserInput($event)"
|
||||
(keydown)="onInputKeyDown($event)" (keypress)="onInputKeyPress($event)" (paste)="onPaste($event)"
|
||||
(focus)="onFocusIn($event)" (blur)="onFocusOut($event)" />
|
||||
</div>
|
16
src/app/currency-input/currency-input.component.scss
Normal file
16
src/app/currency-input/currency-input.component.scss
Normal file
@ -0,0 +1,16 @@
|
||||
div {
|
||||
display: flex;
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
input {
|
||||
flex-grow: 1;
|
||||
|
||||
border: none;
|
||||
background: none;
|
||||
padding: 0;
|
||||
outline: none;
|
||||
font: inherit;
|
||||
text-align: right;
|
||||
color: currentColor;
|
||||
}
|
23
src/app/currency-input/currency-input.component.spec.ts
Normal file
23
src/app/currency-input/currency-input.component.spec.ts
Normal file
@ -0,0 +1,23 @@
|
||||
import { ComponentFixture, TestBed } from '@angular/core/testing';
|
||||
|
||||
import { CurrencyInputComponent } from './currency-input.component';
|
||||
|
||||
describe('CurrencyInputComponent', () => {
|
||||
let component: CurrencyInputComponent;
|
||||
let fixture: ComponentFixture<CurrencyInputComponent>;
|
||||
|
||||
beforeEach(async () => {
|
||||
await TestBed.configureTestingModule({
|
||||
imports: [CurrencyInputComponent]
|
||||
})
|
||||
.compileComponents();
|
||||
|
||||
fixture = TestBed.createComponent(CurrencyInputComponent);
|
||||
component = fixture.componentInstance;
|
||||
fixture.detectChanges();
|
||||
});
|
||||
|
||||
it('should create', () => {
|
||||
expect(component).toBeTruthy();
|
||||
});
|
||||
});
|
873
src/app/currency-input/currency-input.component.ts
Normal file
873
src/app/currency-input/currency-input.component.ts
Normal file
@ -0,0 +1,873 @@
|
||||
import { BooleanInput, coerceBooleanProperty } from '@angular/cdk/coercion';
|
||||
import { Component, ElementRef, HostBinding, Inject, Input, LOCALE_ID, Optional, Self, ViewChild } from '@angular/core';
|
||||
import { ControlValueAccessor, NgControl, Validators } from '@angular/forms';
|
||||
import { MatFormFieldControl } from '@angular/material/form-field';
|
||||
import { Subject } from 'rxjs';
|
||||
|
||||
interface FormatterOptions {
|
||||
useGrouping: boolean,
|
||||
minimumFractionDigits: number,
|
||||
maximumFractionDigits: number
|
||||
}
|
||||
|
||||
@Component({
|
||||
selector: 'app-currency-input',
|
||||
imports: [],
|
||||
providers: [{ provide: MatFormFieldControl, useExisting: CurrencyInputComponent }],
|
||||
templateUrl: './currency-input.component.html',
|
||||
styleUrl: './currency-input.component.scss'
|
||||
})
|
||||
export class CurrencyInputComponent implements MatFormFieldControl<number>, ControlValueAccessor {
|
||||
|
||||
constructor(@Optional() @Self() public ngControl: NgControl, @Inject(LOCALE_ID) public locale: string) {
|
||||
// Replace the provider from above with this.
|
||||
if (this.ngControl != null) {
|
||||
// Setting the value accessor directly (instead of using
|
||||
// the providers) to avoid running into a circular import.
|
||||
this.ngControl.valueAccessor = this;
|
||||
}
|
||||
|
||||
this.constructParsers();
|
||||
}
|
||||
|
||||
ngAfterViewInit(): void {
|
||||
if (this.ngControl)
|
||||
this.required = this.ngControl.control?.hasValidator(Validators.required);
|
||||
// if (this.commentText !== '') {
|
||||
// this._commentInputRef.nativeElement.innerHTML = this.commentText;
|
||||
// this.onChange(this.commentText);
|
||||
// this.stateChanges.next()
|
||||
// }
|
||||
}
|
||||
|
||||
ngOnDestroy() { this.stateChanges.complete(); }
|
||||
|
||||
// ==================================================
|
||||
// => Everything related to custom form field control
|
||||
// ==================================================
|
||||
static nextId = 0;
|
||||
@HostBinding() id = `currency-input-${CurrencyInputComponent.nextId++}`;
|
||||
@HostBinding('class.floating')
|
||||
|
||||
@Input('aria-describedby') userAriaDescribedBy: string = '';
|
||||
@Input()
|
||||
get value(): number | null {
|
||||
return this.inputValue;
|
||||
}
|
||||
set value(value: number | null) {
|
||||
this.inputValue = value ?? 0;
|
||||
}
|
||||
@Input()
|
||||
get placeholder() {
|
||||
return this._placeholder;
|
||||
}
|
||||
set placeholder(plh) {
|
||||
this._placeholder = plh;
|
||||
this.stateChanges.next();
|
||||
}
|
||||
get shouldLabelFloat() {
|
||||
return this.focused || !this.empty;
|
||||
}
|
||||
@Input()
|
||||
get required(): boolean { return this._required; }
|
||||
set required(req: BooleanInput) {
|
||||
this._required = coerceBooleanProperty(req);
|
||||
this.stateChanges.next();
|
||||
}
|
||||
@Input()
|
||||
get disabled(): boolean { return this._disabled; }
|
||||
set disabled(value: BooleanInput) {
|
||||
this._disabled = coerceBooleanProperty(value);
|
||||
this.stateChanges.next();
|
||||
}
|
||||
|
||||
inputValue: number | null = null;
|
||||
controlType = 'currency-input';
|
||||
stateChanges = new Subject<void>();
|
||||
private _placeholder: string = '';
|
||||
private _required = false;
|
||||
private _disabled = false;
|
||||
focused = false;
|
||||
touched = false;
|
||||
|
||||
onChange = (value: number | null) => { };
|
||||
onTouched = () => { };
|
||||
|
||||
writeValue(value: number): void {
|
||||
this.inputValue = value;
|
||||
// throw new Error('Method not implemented.');
|
||||
}
|
||||
registerOnChange(onChange: any): void {
|
||||
this.onChange = onChange;
|
||||
// throw new Error('Method not implemented.');
|
||||
}
|
||||
registerOnTouched(onTouched: any): void {
|
||||
this.onTouched = onTouched;
|
||||
// throw new Error('Method not implemented.');
|
||||
}
|
||||
setDisabledState?(isDisabled: boolean): void {
|
||||
this.disabled = isDisabled;
|
||||
}
|
||||
|
||||
onFocusIn(event: FocusEvent) {
|
||||
if (!this.focused) {
|
||||
this.focused = true;
|
||||
this.stateChanges.next();
|
||||
}
|
||||
}
|
||||
|
||||
onFocusOut(event: FocusEvent) {
|
||||
// if (!this._commentInputRef.nativeElement.contains(event.relatedTarget as Element)) {
|
||||
// this.cleanEmptyElements(this._commentInputRef.nativeElement);
|
||||
// this.commentText = this._commentInputRef.nativeElement.innerHTML;
|
||||
// this.touched = true;
|
||||
// this.focused = false;
|
||||
// this.onTouched();
|
||||
// this.onChange(this.commentText);
|
||||
// this.stateChanges.next();
|
||||
// }
|
||||
this.touched = true;
|
||||
this.focused = false;
|
||||
this.onTouched();
|
||||
this.onChange(this.inputValue);
|
||||
this.stateChanges.next();
|
||||
}
|
||||
|
||||
setDescribedByIds(ids: string[]) {
|
||||
const controlElement = this.input?.nativeElement;
|
||||
controlElement?.setAttribute('aria-describedby', ids.join(' '));
|
||||
}
|
||||
|
||||
onContainerClick(event: MouseEvent) {
|
||||
this.input.nativeElement.focus();
|
||||
}
|
||||
|
||||
get empty(): boolean {
|
||||
return this.inputValue === null;
|
||||
}
|
||||
|
||||
get errorState(): boolean {
|
||||
return this.touched && this.required && this.inputValue === null;
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
// ==================================================
|
||||
// => Everything related to custom form field control
|
||||
// ==================================================
|
||||
|
||||
@ViewChild('input') input!: ElementRef<HTMLInputElement>;
|
||||
|
||||
isSpecialChar: boolean | null | undefined = undefined;
|
||||
lastValue: string | null | undefined = undefined;
|
||||
min: number = 0;
|
||||
groupChar: string = '';
|
||||
maxFractionDigits = 2;
|
||||
maxLength?: number;
|
||||
|
||||
_numeral: any;
|
||||
_decimal: any;
|
||||
_decimalChar: string = '';
|
||||
_group: any;
|
||||
_minusSign: any;
|
||||
_index: number | any;
|
||||
|
||||
get allowMinusSign(): boolean {
|
||||
return this.min == null || this.min < 0;
|
||||
}
|
||||
|
||||
get formatterOptions(): FormatterOptions {
|
||||
return {
|
||||
useGrouping: true,
|
||||
minimumFractionDigits: 2,
|
||||
maximumFractionDigits: 2
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
constructParsers() {
|
||||
const numerals = [...new Intl.NumberFormat(this.locale, { useGrouping: false }).format(9876543210)].reverse();
|
||||
const index = new Map(numerals.map((d, i) => [d, i]));
|
||||
this._numeral = new RegExp(`[${numerals.join('')}]`, 'g');
|
||||
this._group = this.getGroupingExpression();
|
||||
this._minusSign = this.getMinusSignExpression();
|
||||
this._decimal = this.getDecimalExpression();
|
||||
this._decimalChar = this.getDecimalChar();
|
||||
this._index = (d: any) => index.get(d);
|
||||
}
|
||||
|
||||
getDecimalExpression(): RegExp {
|
||||
const decimalChar = this.getDecimalChar();
|
||||
return new RegExp(`[${decimalChar}]`, 'g');
|
||||
}
|
||||
|
||||
getDecimalChar(): string {
|
||||
const formatter = new Intl.NumberFormat(this.locale, { useGrouping: false });
|
||||
return formatter
|
||||
.format(1.1)
|
||||
.trim()
|
||||
.replace(this._numeral, '');
|
||||
}
|
||||
|
||||
getGroupingExpression(): RegExp {
|
||||
const formatter = new Intl.NumberFormat(this.locale, { useGrouping: true });
|
||||
this.groupChar = formatter.format(1000000).trim().replace(this._numeral, '').charAt(0);
|
||||
return new RegExp(`[${this.groupChar}]`, 'g');
|
||||
}
|
||||
|
||||
getMinusSignExpression(): RegExp {
|
||||
const formatter = new Intl.NumberFormat(this.locale, { useGrouping: false });
|
||||
return new RegExp(`[${formatter.format(-1).trim().replace(this._numeral, '')}]`, 'g');
|
||||
}
|
||||
|
||||
isMinusSign(char: string) {
|
||||
if (this._minusSign.test(char) || char === '-') {
|
||||
this._minusSign.lastIndex = 0;
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
isDecimalSign(char: string) {
|
||||
if (this._decimal.test(char)) {
|
||||
this._decimal.lastIndex = 0;
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
parseValue(text: any) {
|
||||
let filteredText = text
|
||||
.trim()
|
||||
.replace(/\s/g, '')
|
||||
.replace(this._group, '')
|
||||
.replace(this._minusSign, '-')
|
||||
.replace(this._decimal, '.')
|
||||
.replace(this._numeral, this._index);
|
||||
|
||||
if (filteredText) {
|
||||
if (filteredText === '-')
|
||||
// Minus sign
|
||||
return filteredText;
|
||||
|
||||
let parsedValue = +filteredText;
|
||||
return isNaN(parsedValue) ? null : parsedValue;
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
getCharIndexes(val: string) {
|
||||
const decimalCharIndex = val.search(this._decimal);
|
||||
this._decimal.lastIndex = 0;
|
||||
const minusCharIndex = val.search(this._minusSign);
|
||||
this._minusSign.lastIndex = 0;
|
||||
|
||||
return { decimalCharIndex, minusCharIndex };
|
||||
}
|
||||
|
||||
|
||||
|
||||
onUserInput(event: Event) {
|
||||
if (this.disabled) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (this.isSpecialChar) {
|
||||
(event.target as HTMLInputElement).value = this.lastValue as string;
|
||||
}
|
||||
this.isSpecialChar = false;
|
||||
}
|
||||
|
||||
onInputKeyDown(event: KeyboardEvent) {
|
||||
if (this.disabled) {
|
||||
return;
|
||||
}
|
||||
|
||||
this.lastValue = (event.target as HTMLInputElement).value;
|
||||
if ((event as KeyboardEvent).shiftKey || (event as KeyboardEvent).altKey || event.metaKey || event.ctrlKey) {
|
||||
this.isSpecialChar = true;
|
||||
return;
|
||||
}
|
||||
|
||||
let selectionStart = (event.target as HTMLInputElement).selectionStart as number;
|
||||
let selectionEnd = (event.target as HTMLInputElement).selectionEnd as number;
|
||||
let inputValue = (event.target as HTMLInputElement).value as string;
|
||||
let newValueStr = null;
|
||||
|
||||
if (event.altKey) {
|
||||
event.preventDefault();
|
||||
}
|
||||
|
||||
switch (event.key) {
|
||||
case 'ArrowUp':
|
||||
event.preventDefault();
|
||||
break;
|
||||
|
||||
case 'ArrowDown':
|
||||
event.preventDefault();
|
||||
break;
|
||||
|
||||
case 'ArrowLeft':
|
||||
for (let index = selectionStart; index <= inputValue.length; index++) {
|
||||
const previousCharIndex = index === 0 ? 0 : index - 1;
|
||||
if (this.isNumeralChar(inputValue.charAt(previousCharIndex))) {
|
||||
this.input.nativeElement.setSelectionRange(index, index);
|
||||
break;
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
case 'ArrowRight':
|
||||
for (let index = selectionEnd; index >= 0; index--) {
|
||||
if (this.isNumeralChar(inputValue.charAt(index))) {
|
||||
this.input.nativeElement.setSelectionRange(index, index);
|
||||
break;
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
case 'Tab':
|
||||
case 'Enter':
|
||||
newValueStr = this.validateValue(this.parseValue(this.input.nativeElement.value));
|
||||
this.input.nativeElement.value = this.formatValue(newValueStr);
|
||||
this.input.nativeElement.setAttribute('aria-valuenow', newValueStr?.toString() ?? '');
|
||||
this.updateModel(event, newValueStr);
|
||||
break;
|
||||
|
||||
case 'Backspace': {
|
||||
event.preventDefault();
|
||||
|
||||
if (selectionStart === selectionEnd) {
|
||||
// if ((selectionStart == 1 && this.prefix) || (selectionStart == inputValue.length && this.suffix)) {
|
||||
// break;
|
||||
// }
|
||||
|
||||
const deleteChar = inputValue.charAt(selectionStart - 1);
|
||||
const { decimalCharIndex, decimalCharIndexWithoutPrefix } = this.getDecimalCharIndexes(inputValue);
|
||||
|
||||
if (this.isNumeralChar(deleteChar)) {
|
||||
const decimalLength = this.getDecimalLength(inputValue);
|
||||
|
||||
if (this._group.test(deleteChar)) {
|
||||
this._group.lastIndex = 0;
|
||||
newValueStr = inputValue.slice(0, selectionStart - 2) + inputValue.slice(selectionStart - 1);
|
||||
} else if (this._decimal.test(deleteChar)) {
|
||||
this._decimal.lastIndex = 0;
|
||||
|
||||
if (decimalLength) {
|
||||
this.input?.nativeElement.setSelectionRange(selectionStart - 1, selectionStart - 1);
|
||||
} else {
|
||||
newValueStr = inputValue.slice(0, selectionStart - 1) + inputValue.slice(selectionStart);
|
||||
}
|
||||
} else if (decimalCharIndex > 0 && selectionStart > decimalCharIndex) {
|
||||
const insertedText = this.isDecimalMode() && (this.formatterOptions.minimumFractionDigits || 0) < <number>decimalLength ? '' : '0';
|
||||
newValueStr = inputValue.slice(0, selectionStart - 1) + insertedText + inputValue.slice(selectionStart);
|
||||
} else if (decimalCharIndexWithoutPrefix === 1) {
|
||||
newValueStr = inputValue.slice(0, selectionStart - 1) + '0' + inputValue.slice(selectionStart);
|
||||
newValueStr = (this.parseValue(newValueStr) as number) > 0 ? newValueStr : '';
|
||||
} else {
|
||||
newValueStr = inputValue.slice(0, selectionStart - 1) + inputValue.slice(selectionStart);
|
||||
}
|
||||
}
|
||||
|
||||
this.updateValue(event, newValueStr, null, 'delete-single');
|
||||
} else {
|
||||
newValueStr = this.deleteRange(inputValue, selectionStart, selectionEnd);
|
||||
this.updateValue(event, newValueStr, null, 'delete-range');
|
||||
}
|
||||
|
||||
break;
|
||||
}
|
||||
|
||||
case 'Delete':
|
||||
event.preventDefault();
|
||||
|
||||
if (selectionStart === selectionEnd) {
|
||||
// if ((selectionStart == 0 && this.prefix) || (selectionStart == inputValue.length - 1 && this.suffix)) {
|
||||
// break;
|
||||
// }
|
||||
const deleteChar = inputValue.charAt(selectionStart);
|
||||
const { decimalCharIndex, decimalCharIndexWithoutPrefix } = this.getDecimalCharIndexes(inputValue);
|
||||
|
||||
if (this.isNumeralChar(deleteChar)) {
|
||||
const decimalLength = this.getDecimalLength(inputValue);
|
||||
|
||||
if (this._group.test(deleteChar)) {
|
||||
this._group.lastIndex = 0;
|
||||
newValueStr = inputValue.slice(0, selectionStart) + inputValue.slice(selectionStart + 2);
|
||||
} else if (this._decimal.test(deleteChar)) {
|
||||
this._decimal.lastIndex = 0;
|
||||
|
||||
if (decimalLength) {
|
||||
this.input?.nativeElement.setSelectionRange(selectionStart + 1, selectionStart + 1);
|
||||
} else {
|
||||
newValueStr = inputValue.slice(0, selectionStart) + inputValue.slice(selectionStart + 1);
|
||||
}
|
||||
} else if (decimalCharIndex > 0 && selectionStart > decimalCharIndex) {
|
||||
const insertedText = this.isDecimalMode() && (this.formatterOptions.minimumFractionDigits || 0) < <number>decimalLength ? '' : '0';
|
||||
newValueStr = inputValue.slice(0, selectionStart) + insertedText + inputValue.slice(selectionStart + 1);
|
||||
} else if (decimalCharIndexWithoutPrefix === 1) {
|
||||
newValueStr = inputValue.slice(0, selectionStart) + '0' + inputValue.slice(selectionStart + 1);
|
||||
newValueStr = (this.parseValue(newValueStr) as number) > 0 ? newValueStr : '';
|
||||
} else {
|
||||
newValueStr = inputValue.slice(0, selectionStart) + inputValue.slice(selectionStart + 1);
|
||||
}
|
||||
}
|
||||
|
||||
this.updateValue(event, newValueStr as string, null, 'delete-back-single');
|
||||
} else {
|
||||
newValueStr = this.deleteRange(inputValue, selectionStart, selectionEnd);
|
||||
this.updateValue(event, newValueStr, null, 'delete-range');
|
||||
}
|
||||
break;
|
||||
|
||||
case 'Home':
|
||||
if (this.min) {
|
||||
this.updateModel(event, this.min);
|
||||
event.preventDefault();
|
||||
}
|
||||
break;
|
||||
|
||||
case 'End':
|
||||
// if (this.max) {
|
||||
// this.updateModel(event, this.max);
|
||||
// event.preventDefault();
|
||||
// }
|
||||
break;
|
||||
|
||||
default:
|
||||
break;
|
||||
}
|
||||
|
||||
// this.onKeyDown.emit(event);
|
||||
}
|
||||
|
||||
|
||||
onInputKeyPress(event: KeyboardEvent) {
|
||||
if (this.disabled) {
|
||||
return;
|
||||
}
|
||||
|
||||
let code = event.which || event.keyCode;
|
||||
let char = String.fromCharCode(code);
|
||||
let isDecimalSign = this.isDecimalSign(char);
|
||||
const isMinusSign = this.isMinusSign(char);
|
||||
|
||||
if (code != 13) {
|
||||
event.preventDefault();
|
||||
}
|
||||
if (!isDecimalSign && event.code === 'NumpadDecimal') {
|
||||
isDecimalSign = true;
|
||||
char = this._decimalChar;
|
||||
code = char.charCodeAt(0);
|
||||
}
|
||||
const { value, selectionStart, selectionEnd } = this.input.nativeElement;
|
||||
const newValue = this.parseValue(value + char);
|
||||
const newValueStr = newValue != null ? newValue.toString() : '';
|
||||
const selectedValue = value.substring(selectionStart ?? 0, selectionEnd ?? 0);
|
||||
const selectedValueParsed = this.parseValue(selectedValue);
|
||||
const selectedValueStr = selectedValueParsed != null ? selectedValueParsed.toString() : '';
|
||||
|
||||
if (selectionStart !== selectionEnd && selectedValueStr.length > 0) {
|
||||
this.insert(event, char, { isDecimalSign, isMinusSign });
|
||||
return;
|
||||
}
|
||||
|
||||
if (this.maxLength && newValueStr.length > this.maxLength) {
|
||||
return;
|
||||
}
|
||||
|
||||
if ((48 <= code && code <= 57) || isMinusSign || isDecimalSign) {
|
||||
this.insert(event, char, { isDecimalSign, isMinusSign });
|
||||
}
|
||||
}
|
||||
|
||||
insert(event: Event, text: string, sign = { isDecimalSign: false, isMinusSign: false }) {
|
||||
const minusCharIndexOnText = text.search(this._minusSign);
|
||||
this._minusSign.lastIndex = 0;
|
||||
if (!this.allowMinusSign && minusCharIndexOnText !== -1) {
|
||||
return;
|
||||
}
|
||||
|
||||
let selectionStart = this.input?.nativeElement.selectionStart ?? 0;
|
||||
let selectionEnd = this.input?.nativeElement.selectionEnd ?? 0;
|
||||
let inputValue = this.input?.nativeElement.value.trim();
|
||||
const { decimalCharIndex, minusCharIndex } = this.getCharIndexes(inputValue);
|
||||
let newValueStr;
|
||||
|
||||
if (sign.isMinusSign) {
|
||||
if (selectionStart === 0) {
|
||||
newValueStr = inputValue;
|
||||
if (minusCharIndex === -1 || selectionEnd !== 0) {
|
||||
newValueStr = this.insertText(inputValue, text, 0, selectionEnd);
|
||||
}
|
||||
|
||||
this.updateValue(event, newValueStr, text, 'insert');
|
||||
}
|
||||
} else if (sign.isDecimalSign) {
|
||||
if (decimalCharIndex > 0 && selectionStart === decimalCharIndex) {
|
||||
this.updateValue(event, inputValue, text, 'insert');
|
||||
} else if (decimalCharIndex > selectionStart && decimalCharIndex < selectionEnd) {
|
||||
newValueStr = this.insertText(inputValue, text, selectionStart, selectionEnd);
|
||||
this.updateValue(event, newValueStr, text, 'insert');
|
||||
} else if (decimalCharIndex === -1 && this.maxFractionDigits) {
|
||||
newValueStr = this.insertText(inputValue, text, selectionStart, selectionEnd);
|
||||
this.updateValue(event, newValueStr, text, 'insert');
|
||||
}
|
||||
} else {
|
||||
const maxFractionDigits = this.formatterOptions.maximumFractionDigits; // this.numberFormat.resolvedOptions().maximumFractionDigits;
|
||||
const operation = selectionStart !== selectionEnd ? 'range-insert' : 'insert';
|
||||
|
||||
if (decimalCharIndex > 0 && selectionStart > decimalCharIndex) {
|
||||
if (selectionStart + text.length - (decimalCharIndex + 1) <= maxFractionDigits) {
|
||||
// const charIndex = currencyCharIndex >= selectionStart ? currencyCharIndex - 1 : suffixCharIndex >= selectionStart ? suffixCharIndex : inputValue.length;
|
||||
const charIndex = inputValue.length;
|
||||
newValueStr = inputValue.slice(0, selectionStart) + text + inputValue.slice(selectionStart + text.length, charIndex) + inputValue.slice(charIndex);
|
||||
this.updateValue(event, newValueStr, text, operation);
|
||||
}
|
||||
} else {
|
||||
newValueStr = this.insertText(inputValue, text, selectionStart, selectionEnd);
|
||||
this.updateValue(event, newValueStr, text, operation);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
insertText(value: string, text: string, start: number, end: number): string {
|
||||
let textSplit = text === '.' ? text : text.split('.');
|
||||
|
||||
if (textSplit.length === 2) {
|
||||
const decimalCharIndex = value.slice(start, end).search(this._decimal);
|
||||
this._decimal.lastIndex = 0;
|
||||
return decimalCharIndex > 0 ? value.slice(0, start) + this.formatValue(text) + value.slice(end) : value || this.formatValue(text);
|
||||
} else if (end - start === value.length) {
|
||||
return this.formatValue(text);
|
||||
} else if (start === 0) {
|
||||
return text + value.slice(end);
|
||||
} else if (end === value.length) {
|
||||
return value.slice(0, start) + text;
|
||||
} else {
|
||||
return value.slice(0, start) + text + value.slice(end);
|
||||
}
|
||||
}
|
||||
|
||||
deleteRange(value: string, start: number, end: number) {
|
||||
let newValueStr;
|
||||
|
||||
if (end - start === value.length) newValueStr = '';
|
||||
else if (start === 0) newValueStr = value.slice(end);
|
||||
else if (end === value.length) newValueStr = value.slice(0, start);
|
||||
else newValueStr = value.slice(0, start) + value.slice(end);
|
||||
|
||||
return newValueStr;
|
||||
}
|
||||
|
||||
updateValue(event: Event, valueStr?: null | string, insertedValueStr?: null | string, operation?: null | string) {
|
||||
let currentValue = this.input?.nativeElement.value;
|
||||
let newValue = null;
|
||||
|
||||
if (valueStr != null) {
|
||||
newValue = this.parseValue(valueStr);
|
||||
// newValue = !newValue && !this.allowEmpty ? 0 : newValue;
|
||||
this.updateInput(newValue, insertedValueStr, operation, valueStr);
|
||||
|
||||
this.handleOnInput(event, currentValue, newValue);
|
||||
}
|
||||
}
|
||||
|
||||
updateInput(value: any, insertedValueStr?: | null | string, operation?: | null | string, valueStr?: | null | string) {
|
||||
insertedValueStr = insertedValueStr || '';
|
||||
|
||||
let inputValue = this.input?.nativeElement.value;
|
||||
let newValue = this.formatValue(value);
|
||||
let currentLength = inputValue.length;
|
||||
|
||||
if (newValue !== valueStr) {
|
||||
newValue = this.concatValues(newValue, valueStr as string);
|
||||
}
|
||||
|
||||
if (currentLength === 0) {
|
||||
this.input.nativeElement.value = newValue;
|
||||
this.input.nativeElement.setSelectionRange(0, 0);
|
||||
const index = this.initCursor();
|
||||
const selectionEnd = index + insertedValueStr.length;
|
||||
this.input.nativeElement.setSelectionRange(selectionEnd, selectionEnd);
|
||||
} else {
|
||||
let selectionStart = this.input.nativeElement.selectionStart ?? 0;
|
||||
let selectionEnd = this.input.nativeElement.selectionEnd ?? 0;
|
||||
|
||||
if (this.maxLength && newValue.length > this.maxLength) {
|
||||
newValue = newValue.slice(0, this.maxLength);
|
||||
selectionStart = Math.min(selectionStart, this.maxLength);
|
||||
selectionEnd = Math.min(selectionEnd, this.maxLength);
|
||||
}
|
||||
|
||||
if (this.maxLength && this.maxLength < newValue.length) {
|
||||
return;
|
||||
}
|
||||
|
||||
this.input.nativeElement.value = newValue;
|
||||
let newLength = newValue.length;
|
||||
|
||||
if (operation === 'range-insert') {
|
||||
const startValue = this.parseValue((inputValue || '').slice(0, selectionStart));
|
||||
const startValueStr = startValue !== null ? startValue.toString() : '';
|
||||
const startExpr = startValueStr.split('').join(`(${this.groupChar})?`);
|
||||
const sRegex = new RegExp(startExpr, 'g');
|
||||
sRegex.test(newValue);
|
||||
|
||||
const tExpr = insertedValueStr.split('').join(`(${this.groupChar})?`);
|
||||
const tRegex = new RegExp(tExpr, 'g');
|
||||
tRegex.test(newValue.slice(sRegex.lastIndex));
|
||||
|
||||
selectionEnd = sRegex.lastIndex + tRegex.lastIndex;
|
||||
this.input.nativeElement.setSelectionRange(selectionEnd, selectionEnd);
|
||||
} else if (newLength === currentLength) {
|
||||
if (operation === 'insert' || operation === 'delete-back-single') this.input.nativeElement.setSelectionRange(selectionEnd + 1, selectionEnd + 1);
|
||||
else if (operation === 'delete-single') this.input.nativeElement.setSelectionRange(selectionEnd - 1, selectionEnd - 1);
|
||||
else if (operation === 'delete-range' || operation === 'spin') this.input.nativeElement.setSelectionRange(selectionEnd, selectionEnd);
|
||||
} else if (operation === 'delete-back-single') {
|
||||
let prevChar = inputValue.charAt(selectionEnd - 1);
|
||||
let nextChar = inputValue.charAt(selectionEnd);
|
||||
let diff = currentLength - newLength;
|
||||
let isGroupChar = this._group.test(nextChar);
|
||||
|
||||
if (isGroupChar && diff === 1) {
|
||||
selectionEnd += 1;
|
||||
} else if (!isGroupChar && this.isNumeralChar(prevChar)) {
|
||||
selectionEnd += -1 * diff + 1;
|
||||
}
|
||||
|
||||
this._group.lastIndex = 0;
|
||||
this.input.nativeElement.setSelectionRange(selectionEnd, selectionEnd);
|
||||
} else if (inputValue === '-' && operation === 'insert') {
|
||||
this.input.nativeElement.setSelectionRange(0, 0);
|
||||
const index = this.initCursor();
|
||||
const selectionEnd = index + insertedValueStr.length + 1;
|
||||
this.input.nativeElement.setSelectionRange(selectionEnd, selectionEnd);
|
||||
} else {
|
||||
selectionEnd = selectionEnd + (newLength - currentLength);
|
||||
this.input.nativeElement.setSelectionRange(selectionEnd, selectionEnd);
|
||||
}
|
||||
}
|
||||
|
||||
this.input.nativeElement.setAttribute('aria-valuenow', value);
|
||||
}
|
||||
|
||||
initCursor() {
|
||||
let selectionStart = this.input?.nativeElement.selectionStart ?? 0;
|
||||
let selectionEnd = this.input?.nativeElement.selectionEnd ?? 0;
|
||||
let inputValue = this.input?.nativeElement.value;
|
||||
let valueLength = inputValue.length;
|
||||
let index = null;
|
||||
|
||||
// remove prefix
|
||||
let prefixLength = 0;
|
||||
// let prefixLength = (this.prefixChar || '').length;
|
||||
// inputValue = inputValue.replace(this._prefix, '');
|
||||
|
||||
// Will allow selecting whole prefix. But not a part of it.
|
||||
// Negative values will trigger clauses after this to fix the cursor position.
|
||||
if (selectionStart === selectionEnd || selectionStart !== 0 || selectionEnd < prefixLength) {
|
||||
selectionStart -= prefixLength;
|
||||
}
|
||||
|
||||
let char = inputValue.charAt(selectionStart);
|
||||
if (this.isNumeralChar(char)) {
|
||||
return selectionStart + prefixLength;
|
||||
}
|
||||
|
||||
//left
|
||||
let i = selectionStart - 1;
|
||||
while (i >= 0) {
|
||||
char = inputValue.charAt(i);
|
||||
if (this.isNumeralChar(char)) {
|
||||
index = i + prefixLength;
|
||||
break;
|
||||
} else {
|
||||
i--;
|
||||
}
|
||||
}
|
||||
|
||||
if (index !== null) {
|
||||
this.input?.nativeElement.setSelectionRange(index + 1, index + 1);
|
||||
} else {
|
||||
i = selectionStart;
|
||||
while (i < valueLength) {
|
||||
char = inputValue.charAt(i);
|
||||
if (this.isNumeralChar(char)) {
|
||||
index = i + prefixLength;
|
||||
break;
|
||||
} else {
|
||||
i++;
|
||||
}
|
||||
}
|
||||
|
||||
if (index !== null) {
|
||||
this.input?.nativeElement.setSelectionRange(index, index);
|
||||
}
|
||||
}
|
||||
|
||||
return index || 0;
|
||||
}
|
||||
|
||||
handleOnInput(event: Event, currentValue: string, newValue: any) {
|
||||
if (this.isValueChanged(currentValue, newValue)) {
|
||||
(this.input as ElementRef).nativeElement.value = this.formatValue(newValue);
|
||||
this.input?.nativeElement.setAttribute('aria-valuenow', newValue);
|
||||
this.updateModel(event, newValue);
|
||||
// this.onInput.emit({ originalEvent: event, value: newValue, formattedValue: currentValue });
|
||||
}
|
||||
}
|
||||
|
||||
concatValues(val1: string, val2: string) {
|
||||
if (val1 && val2) {
|
||||
let decimalCharIndex = val2.search(this._decimal);
|
||||
this._decimal.lastIndex = 0;
|
||||
|
||||
return decimalCharIndex !== -1 ? val1.split(this._decimal)[0] + val2.slice(decimalCharIndex) : val1;
|
||||
|
||||
}
|
||||
return val1;
|
||||
}
|
||||
|
||||
getDecimalLength(value: string) {
|
||||
if (value) {
|
||||
const valueSplit = value.split(this._decimal);
|
||||
|
||||
if (valueSplit.length === 2) {
|
||||
return valueSplit[1]
|
||||
.trim()
|
||||
.replace(/\s/g, '')
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
onPaste(event: ClipboardEvent) {
|
||||
if (!this.disabled && !this.disabled) {
|
||||
event.preventDefault();
|
||||
let data = (event.clipboardData || (document as any).defaultView['clipboardData']).getData('Text');
|
||||
if (data) {
|
||||
if (this.maxLength) {
|
||||
data = data.toString().substring(0, this.maxLength);
|
||||
}
|
||||
|
||||
let filteredData = this.parseValue(data);
|
||||
if (filteredData != null) {
|
||||
this.insert(event, filteredData.toString());
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
isValueChanged(currentValue: string, newValue: string) {
|
||||
if (newValue === null && currentValue !== null) {
|
||||
return true;
|
||||
}
|
||||
|
||||
if (newValue != null) {
|
||||
let parsedCurrentValue = typeof currentValue === 'string' ? this.parseValue(currentValue) : currentValue;
|
||||
return newValue !== parsedCurrentValue;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
isNumeralChar(char: string) {
|
||||
if (char.length === 1 && (this._numeral.test(char) || this._decimal.test(char) || this._group.test(char) || this._minusSign.test(char))) {
|
||||
this.resetRegex();
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
isDecimalMode(): boolean {
|
||||
return true;
|
||||
}
|
||||
|
||||
getDecimalCharIndexes(val: string) {
|
||||
let decimalCharIndex = val.search(this._decimal);
|
||||
this._decimal.lastIndex = 0;
|
||||
|
||||
const filteredVal = val
|
||||
.trim()
|
||||
.replace(/\s/g, '')
|
||||
const decimalCharIndexWithoutPrefix = filteredVal.search(this._decimal);
|
||||
this._decimal.lastIndex = 0;
|
||||
|
||||
return { decimalCharIndex, decimalCharIndexWithoutPrefix };
|
||||
}
|
||||
|
||||
validateValue(value: number | string) {
|
||||
if (value === '-' || value == null) {
|
||||
return null;
|
||||
}
|
||||
|
||||
if (this.min != null && (value as number) < this.min) {
|
||||
return this.min;
|
||||
}
|
||||
|
||||
// if (this.max != null && (value as number) > this.max) {
|
||||
// return this.max;
|
||||
// }
|
||||
|
||||
return value;
|
||||
}
|
||||
|
||||
resetRegex() {
|
||||
this._numeral.lastIndex = 0;
|
||||
this._decimal.lastIndex = 0;
|
||||
this._group.lastIndex = 0;
|
||||
this._minusSign.lastIndex = 0;
|
||||
}
|
||||
|
||||
formatValue(value: any) {
|
||||
if (value != null) {
|
||||
if (value === '-') {
|
||||
// Minus sign
|
||||
return value;
|
||||
}
|
||||
|
||||
// if (this.format) {
|
||||
let formatter = new Intl.NumberFormat(this.locale, this.formatterOptions);
|
||||
let formattedValue = formatter.format(value);
|
||||
|
||||
return formattedValue;
|
||||
// }
|
||||
|
||||
return value.toString();
|
||||
}
|
||||
|
||||
return '';
|
||||
}
|
||||
|
||||
|
||||
formattedValue() {
|
||||
const val = !this.value && !this.required ? 0 : this.value;
|
||||
return this.formatValue(val);
|
||||
}
|
||||
|
||||
updateModel(event: Event, value: any) {
|
||||
const isBlurUpdateOnMode = this.ngControl?.control?.updateOn === 'blur';
|
||||
|
||||
if (this.inputValue !== value) {
|
||||
this.inputValue = value;
|
||||
if (!(isBlurUpdateOnMode && this.focused)) {
|
||||
// this.onModelChange(value);
|
||||
this.stateChanges.next();
|
||||
}
|
||||
} else if (isBlurUpdateOnMode) {
|
||||
// this.onModelChange(value);
|
||||
this.stateChanges.next();
|
||||
}
|
||||
// this.onModelTouched();
|
||||
this.onTouched();
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue
Block a user