diff --git a/.github/workflows/publish.yml b/.github/workflows/publish.yml
index 0add2295..6a223369 100644
--- a/.github/workflows/publish.yml
+++ b/.github/workflows/publish.yml
@@ -8,6 +8,8 @@ on:
jobs:
build:
runs-on: ubuntu-latest
+ outputs:
+ version: ${{ steps.get_version.outputs.version }}
steps:
- uses: actions/checkout@v4
with:
@@ -34,7 +36,7 @@ jobs:
id: get_version
run: |
VERSION=$(node -p "require('./dist/ngx-mask-lib/package.json').version")
- echo "version=$VERSION" >> $GITHUB_OUTPUT
+ echo "version=$VERSION" >> $GITHUB_OUTPUT
slack_notification:
needs:
diff --git a/.github/workflows/quality-check.yml b/.github/workflows/quality-check.yml
index 97b5d7bc..f423ddfe 100644
--- a/.github/workflows/quality-check.yml
+++ b/.github/workflows/quality-check.yml
@@ -21,4 +21,4 @@ jobs:
- name: Check quality
run: |
bun i
- bash .github/workflows/scripts/quality.sh
\ No newline at end of file
+ bash .github/workflows/scripts/quality.sh
diff --git a/CHANGELOG.md b/CHANGELOG.md
index 6667c655..0a18fc93 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -1,3 +1,13 @@
+# 18.0.3(2024-11-05)
+
+### Fix
+
+- Fix ([#1372](https://github.com/JsDaddy/ngx-mask/issues/1372))
+- Fix ([#1441](https://github.com/JsDaddy/ngx-mask/issues/1441))
+- Fix ([#1442](https://github.com/JsDaddy/ngx-mask/issues/1442))
+- Fix ([#1440](https://github.com/JsDaddy/ngx-mask/issues/1440))
+- Fix ([#1409](https://github.com/JsDaddy/ngx-mask/issues/1409))
+
# 18.0.2(2024-11-01)
### Fix
diff --git a/package.json b/package.json
index 6b9529d7..8a20bd64 100644
--- a/package.json
+++ b/package.json
@@ -1,6 +1,6 @@
{
"name": "ngx-mask",
- "version": "18.0.2",
+ "version": "18.0.3",
"description": "Awesome ngx mask",
"license": "MIT",
"engines": {
diff --git a/projects/ngx-mask-lib/package.json b/projects/ngx-mask-lib/package.json
index a018296e..fcdab315 100644
--- a/projects/ngx-mask-lib/package.json
+++ b/projects/ngx-mask-lib/package.json
@@ -1,6 +1,6 @@
{
"name": "ngx-mask",
- "version": "18.0.2",
+ "version": "18.0.3",
"description": "awesome ngx mask",
"keywords": [
"ng2-mask",
diff --git a/projects/ngx-mask-lib/src/lib/ngx-mask-expression.enum.ts b/projects/ngx-mask-lib/src/lib/ngx-mask-expression.enum.ts
index c9c2916b..3aadf5ab 100644
--- a/projects/ngx-mask-lib/src/lib/ngx-mask-expression.enum.ts
+++ b/projects/ngx-mask-lib/src/lib/ngx-mask-expression.enum.ts
@@ -12,6 +12,7 @@ export const enum MaskExpression {
HOURS_HOUR = 'Hh',
SECONDS = 's0',
HOURS_MINUTES_SECONDS = 'Hh:m0:s0',
+ EMAIL_MASK = 'A*@A*.A*',
HOURS_MINUTES = 'Hh:m0',
MINUTES_SECONDS = 'm0:s0',
DAYS_MONTHS_YEARS = 'd0/M0/0000',
diff --git a/projects/ngx-mask-lib/src/lib/ngx-mask.directive.ts b/projects/ngx-mask-lib/src/lib/ngx-mask.directive.ts
index 4c8b77ac..4ab0e2a0 100644
--- a/projects/ngx-mask-lib/src/lib/ngx-mask.directive.ts
+++ b/projects/ngx-mask-lib/src/lib/ngx-mask.directive.ts
@@ -280,7 +280,15 @@ export class NgxMaskDirective implements ControlValueAccessor, OnChanges, Valida
if (timeMasks.includes(this._maskValue)) {
return this._validateTime(processedValue);
}
+ if (this._maskValue === MaskExpression.EMAIL_MASK) {
+ const emailPattern = /^[^@]+@[^@]+\.[^@]+$/;
+ if (!emailPattern.test(processedValue) && processedValue) {
+ return this._createValidationError(processedValue);
+ } else {
+ return null;
+ }
+ }
if (processedValue && processedValue.length >= 1) {
let counterOfOpt = 0;
@@ -461,7 +469,11 @@ export class NgxMaskDirective implements ControlValueAccessor, OnChanges, Valida
const selStart = Number(this._maskService.selStart) - prefixLength;
const selEnd = Number(this._maskService.selEnd) - prefixLength;
- if (this._code === MaskExpression.BACKSPACE) {
+ const backspaceOrDelete =
+ this._code === MaskExpression.BACKSPACE ||
+ this._code === MaskExpression.DELETE;
+
+ if (backspaceOrDelete) {
if (!selectRangeBackspace) {
if (this._maskService.selStart === prefixLength) {
this._maskService.actualValue = `${this.prefix}${this._maskService.maskIsShown.slice(0, selEnd)}${this._inputValue.split(this.prefix).join('')}`;
@@ -505,8 +517,9 @@ export class NgxMaskDirective implements ControlValueAccessor, OnChanges, Valida
this._maskService.actualValue = `${part1}${this._maskService.placeHolderCharacter}${part2}`;
}
}
+ position = this._code === MaskExpression.DELETE ? position + 1 : position;
}
- if (this._code !== MaskExpression.BACKSPACE) {
+ if (!backspaceOrDelete) {
if (!checkSymbols && !checkSpecialCharacter && selectRangeBackspace) {
position = Number(el.selectionStart) - 1;
} else if (
@@ -996,6 +1009,7 @@ export class NgxMaskDirective implements ControlValueAccessor, OnChanges, Valida
if (typeof this.inputTransformFn !== 'function') {
this._maskService.writingValue = true;
}
+
this._maskService.formElementProperty = [
'value',
this._maskService.applyMask(inputValue, this._maskService.maskExpression),
diff --git a/projects/ngx-mask-lib/src/lib/ngx-mask.service.ts b/projects/ngx-mask-lib/src/lib/ngx-mask.service.ts
index 777e43bb..1de853eb 100644
--- a/projects/ngx-mask-lib/src/lib/ngx-mask.service.ts
+++ b/projects/ngx-mask-lib/src/lib/ngx-mask.service.ts
@@ -206,6 +206,7 @@ export class NgxMaskService extends NgxMaskApplierService {
this._emitValue =
this._previousValue !== this._currentValue ||
this.maskChanged ||
+ this.writingValue ||
(this._previousValue === this._currentValue && justPasted);
}
@@ -215,6 +216,7 @@ export class NgxMaskService extends NgxMaskApplierService {
? requestAnimationFrame(() => this.formControlResult(result))
: this.formControlResult(result)
: '';
+
if (!this.showMaskTyped || (this.showMaskTyped && this.hiddenInput)) {
if (this.hiddenInput) {
if (backspaced) {
@@ -530,6 +532,10 @@ export class NgxMaskService extends NgxMaskApplierService {
* @param inputValue the current form input value
*/
private formControlResult(inputValue: string): void {
+ if (this.writingValue && !inputValue) {
+ this.onChange('');
+ return;
+ }
if (this.writingValue || (!this.triggerOnMaskChange && this.maskChanged)) {
// eslint-disable-next-line no-unused-expressions,@typescript-eslint/no-unused-expressions
this.triggerOnMaskChange && this.maskChanged
@@ -583,9 +589,11 @@ export class NgxMaskService extends NgxMaskApplierService {
) {
return value;
}
- if (String(value).length > 16 && this.separatorLimit.length > 14) {
+
+ if (String(value).length > 14 && this.maskExpression.startsWith(MaskExpression.SEPARATOR)) {
return String(value);
}
+
const num = Number(value);
if (this.maskExpression.startsWith(MaskExpression.SEPARATOR) && Number.isNaN(num)) {
const val = String(value).replace(',', '.');
@@ -686,7 +694,7 @@ export class NgxMaskService extends NgxMaskApplierService {
if (processedResult === this.decimalMarker) {
return null;
}
- if (this.separatorLimit.length > 14) {
+ if (separatorValue.length > 14) {
return String(separatorValue);
}
return this._checkPrecision(this.maskExpression, separatorValue);
diff --git a/projects/ngx-mask-lib/src/test/basic-logic.spec.ts b/projects/ngx-mask-lib/src/test/basic-logic.spec.ts
index b072e0af..24cc97f6 100644
--- a/projects/ngx-mask-lib/src/test/basic-logic.spec.ts
+++ b/projects/ngx-mask-lib/src/test/basic-logic.spec.ts
@@ -969,4 +969,56 @@ describe('Directive: Mask', () => {
expect(component.form.dirty).toBe(false);
});
+
+ it('mask sepator.2 after setValue should be dont dirty', () => {
+ component.mask = 'separator.0';
+ component.form.setValue('2002');
+
+ expect(component.form.dirty).toBe(false);
+ });
+
+ it('should return empty string in formControl mask SSS-SSS-SSS', () => {
+ component.mask = 'SSS-SSS-SSS';
+ component.form.setValue('978-1-93624-386-0');
+ const debugElement: DebugElement = fixture.debugElement.query(By.css('input'));
+ const inputTarget: HTMLInputElement = debugElement.nativeElement as HTMLInputElement;
+ spyOnProperty(document, 'activeElement').and.returnValue(inputTarget);
+ fixture.detectChanges();
+
+ expect(inputTarget.value).toBe('');
+ });
+
+ it('should return empty string in formControl mask AAA-AAA-AAA', () => {
+ component.mask = 'AAA-AAA-AAA';
+ component.form.setValue('978-123-936');
+ const debugElement: DebugElement = fixture.debugElement.query(By.css('input'));
+ const inputTarget: HTMLInputElement = debugElement.nativeElement as HTMLInputElement;
+ spyOnProperty(document, 'activeElement').and.returnValue(inputTarget);
+ fixture.detectChanges();
+
+ expect(inputTarget.value).toBe('');
+ });
+
+ it('should return empty string in formControl mask (000) 000-000', () => {
+ component.mask = '(000) 000-000';
+ component.form.setValue('978-123-936');
+ const debugElement: DebugElement = fixture.debugElement.query(By.css('input'));
+ const inputTarget: HTMLInputElement = debugElement.nativeElement as HTMLInputElement;
+ spyOnProperty(document, 'activeElement').and.returnValue(inputTarget);
+ fixture.detectChanges();
+
+ expect(inputTarget.value).toBe('');
+ });
+
+ it('should return empty string in formControl mask (000) 000-000 with prefix +7', () => {
+ component.mask = '(000) 000-000';
+ component.prefix = '+7 ';
+ component.form.setValue('978-123-936');
+ const debugElement: DebugElement = fixture.debugElement.query(By.css('input'));
+ const inputTarget: HTMLInputElement = debugElement.nativeElement as HTMLInputElement;
+ spyOnProperty(document, 'activeElement').and.returnValue(inputTarget);
+ fixture.detectChanges();
+
+ expect(inputTarget.value).toBe('');
+ });
});
diff --git a/projects/ngx-mask-lib/src/test/cursor.cy-spec.ts b/projects/ngx-mask-lib/src/test/cursor.cy-spec.ts
index e433c013..a83b43b1 100644
--- a/projects/ngx-mask-lib/src/test/cursor.cy-spec.ts
+++ b/projects/ngx-mask-lib/src/test/cursor.cy-spec.ts
@@ -176,108 +176,6 @@ describe('Test Date Hh:m0', () => {
.clear();
});
- it('Mask separator.2 check cursor with value 100.0', () => {
- cy.mount(CypressTestMaskComponent, {
- componentProperties: {
- mask: 'separator.2',
- decimalMarker: '.',
- thousandSeparator: ',',
- },
- imports: [CypressTestMaskModule],
- });
- cy.get('#masked')
- .type('1000')
- .type('{leftArrow}')
- .type('.')
- .should('have.value', '100.0')
- .should('have.prop', 'selectionStart', 4);
- });
-
- it('Mask separator.2 check cursor with value 1.00', () => {
- cy.mount(CypressTestMaskComponent, {
- componentProperties: {
- mask: 'separator.2',
- decimalMarker: '.',
- thousandSeparator: ',',
- },
- imports: [CypressTestMaskModule],
- });
- cy.get('#masked')
- .type('1000')
- .type('{leftArrow}'.repeat(3))
- .type('.')
- .should('have.value', '1.00')
- .should('have.prop', 'selectionStart', 2);
- });
-
- it('Mask separator.2 check cursor with value 123456789.20', () => {
- cy.mount(CypressTestMaskComponent, {
- componentProperties: {
- mask: 'separator.2',
- decimalMarker: '.',
- thousandSeparator: ',',
- },
- imports: [CypressTestMaskModule],
- });
- cy.get('#masked')
- .type('123456789.20')
- .type('{leftArrow}'.repeat(4))
- .type('.')
- .should('have.value', '12,345,678.9')
- .should('have.prop', 'selectionStart', 11);
- });
-
- it('Mask separator.2 check cursor with value 100.0', () => {
- cy.mount(CypressTestMaskComponent, {
- componentProperties: {
- mask: 'separator.2',
- decimalMarker: ',',
- thousandSeparator: '.',
- },
- imports: [CypressTestMaskModule],
- });
- cy.get('#masked')
- .type('1000')
- .type('{leftArrow}')
- .type(',')
- .should('have.value', '100,0')
- .should('have.prop', 'selectionStart', 4);
- });
-
- it('Mask separator.2 check cursor with value 1.00', () => {
- cy.mount(CypressTestMaskComponent, {
- componentProperties: {
- mask: 'separator.2',
- decimalMarker: ',',
- thousandSeparator: '.',
- },
- imports: [CypressTestMaskModule],
- });
- cy.get('#masked')
- .type('1000')
- .type('{leftArrow}'.repeat(3))
- .type(',')
- .should('have.value', '1,00')
- .should('have.prop', 'selectionStart', 2);
- });
-
- it('Mask separator.2 check cursor with value 123456789.20', () => {
- cy.mount(CypressTestMaskComponent, {
- componentProperties: {
- mask: 'separator.2',
- decimalMarker: ',',
- thousandSeparator: '.',
- },
- imports: [CypressTestMaskModule],
- });
- cy.get('#masked')
- .type('123456789,20')
- .type('{leftArrow}'.repeat(4))
- .type(',')
- .should('have.value', '12.345.678,9')
- .should('have.prop', 'selectionStart', 11);
- });
-
it('Mask d0/M0/0000 should set cursor on right position', () => {
cy.mount(CypressTestMaskComponent, {
componentProperties: {
@@ -381,95 +279,4 @@ describe('Test Date Hh:m0', () => {
cy.get('#masked').type('111').should('have.value', '(11) 1');
cy.get('#masked').type('{backspace}').should('have.prop', 'selectionStart', 4);
});
-
- it('when decimalMarker doenst set should have right position cursor thousandSeparator = .', () => {
- cy.mount(CypressTestMaskComponent, {
- componentProperties: {
- mask: 'separator.2',
- thousandSeparator: '.',
- },
- imports: [CypressTestMaskModule],
- });
-
- cy.get('#masked')
- .type('12345678,00')
-
- .should('have.value', '12.345.678,00')
- .type('{leftArrow}'.repeat(3))
- .type('{backspace}'.repeat(3))
- .should('have.value', '12.345,00');
- });
-
- it('when decimalMarker doenst set should have right position cursor thousandSeparator = ,', () => {
- cy.mount(CypressTestMaskComponent, {
- componentProperties: {
- mask: 'separator.2',
- thousandSeparator: ',',
- },
- imports: [CypressTestMaskModule],
- });
-
- cy.get('#masked')
- .type('12345678.00')
-
- .should('have.value', '12,345,678.00')
- .type('{leftArrow}'.repeat(3))
- .type('{backspace}'.repeat(3))
- .should('have.value', '12,345.00');
- });
-
- it('should place cursor after backspace with separatorLimit = 10 in correct position', () => {
- cy.mount(CypressTestMaskComponent, {
- componentProperties: {
- mask: 'separator.2',
- separatorLimit: '10',
- },
- imports: [CypressTestMaskModule],
- });
-
- cy.get('#masked')
- .type('12.10')
- .should('have.value', '12.10')
- .type('{leftArrow}'.repeat(2))
- .type('{backspace}')
- .should('have.value', '12')
- .should('have.prop', 'selectionStart', 2);
- });
-
- it('should place cursor after backspace with separatorLimit = 100 in correct position', () => {
- cy.mount(CypressTestMaskComponent, {
- componentProperties: {
- mask: 'separator.2',
- separatorLimit: '100',
- },
- imports: [CypressTestMaskModule],
- });
-
- cy.get('#masked')
- .type('123.10')
- .should('have.value', '123.10')
- .type('{leftArrow}'.repeat(2))
- .type('{backspace}')
- .should('have.value', '123')
- .should('have.prop', 'selectionStart', 3);
- });
-
- it('should place cursor after backspace with separatorLimit = 1000 in correct position', () => {
- cy.mount(CypressTestMaskComponent, {
- componentProperties: {
- mask: 'separator.2',
- thousandSeparator: ',',
- separatorLimit: '1000',
- },
- imports: [CypressTestMaskModule],
- });
-
- cy.get('#masked')
- .type('1234.10')
- .should('have.value', '1,234.10')
- .type('{leftArrow}'.repeat(2))
- .type('{backspace}')
- .should('have.value', '1,234')
- .should('have.prop', 'selectionStart', 5);
- });
});
diff --git a/projects/ngx-mask-lib/src/test/delete.cy-spec.ts b/projects/ngx-mask-lib/src/test/delete.cy-spec.ts
index c711933f..2b3a12b1 100644
--- a/projects/ngx-mask-lib/src/test/delete.cy-spec.ts
+++ b/projects/ngx-mask-lib/src/test/delete.cy-spec.ts
@@ -289,42 +289,6 @@ describe('Directive: Mask (Delete)', () => {
.should('have.value', '+32 __ ___ __ __');
});
- it('should backspace with separator and prefix', () => {
- cy.mount(CypressTestMaskComponent, {
- componentProperties: {
- mask: 'separator.2',
- thousandSeparator: ',',
- prefix: '$ ',
- },
- imports: [CypressTestMaskModule],
- });
-
- cy.get('#masked')
- .type('1234567890')
- .should('have.value', '$ 1,234,567,890')
- .type('{leftArrow}'.repeat(3))
- .type('{backspace}')
- .should('have.prop', 'selectionStart', 11);
- });
-
- it('should backspace with separator and prefix', () => {
- cy.mount(CypressTestMaskComponent, {
- componentProperties: {
- mask: 'separator.2',
- thousandSeparator: '.',
- prefix: '$ ',
- },
- imports: [CypressTestMaskModule],
- });
-
- cy.get('#masked')
- .type('1234567890')
- .should('have.value', '$ 1.234.567.890')
- .type('{leftArrow}'.repeat(3))
- .type('{backspace}')
- .should('have.prop', 'selectionStart', 11);
- });
-
it('should backspace with showMaskTyped and leadZeroDateTime', () => {
cy.mount(CypressTestMaskComponent, {
componentProperties: {
@@ -376,233 +340,6 @@ describe('Directive: Mask (Delete)', () => {
.should('have.value', '__:__:____');
});
- it('should correct work after backspace separator.6 decimalMarker . thousandSeparator ,', () => {
- cy.mount(CypressTestMaskComponent, {
- componentProperties: {
- mask: 'separator.6',
- decimalMarker: '.',
- thousandSeparator: ',',
- },
- imports: [CypressTestMaskModule],
- });
-
- cy.get('#masked')
- .type('0.000001')
- .should('have.value', '0.000001')
- .type('{leftArrow}'.repeat(2))
- .type('{backspace}')
- .should('have.value', '0.00001')
- .type('{leftArrow}'.repeat(2))
- .type('{backspace}')
- .should('have.value', '0.0001')
- .type('{backspace}')
- .should('have.value', '1');
- });
-
- it('should correct work after backspace separator.2 decimalMarker . thousandSeparator ,', () => {
- cy.mount(CypressTestMaskComponent, {
- componentProperties: {
- mask: 'separator.2',
- decimalMarker: '.',
- thousandSeparator: ',',
- },
- imports: [CypressTestMaskModule],
- });
-
- cy.get('#masked')
- .type('0.01')
- .should('have.value', '0.01')
- .type('{leftArrow}')
- .type('{backspace}')
- .should('have.value', '0.1')
- .type('{leftArrow}')
- .type('{backspace}')
- .should('have.value', '1');
- });
-
- it('should correct work after backspace separator.2 decimalMarker . thousandSeparator , allowNegative', () => {
- cy.mount(CypressTestMaskComponent, {
- componentProperties: {
- mask: 'separator.2',
- decimalMarker: '.',
- thousandSeparator: ',',
- allowNegativeNumbers: true,
- },
- imports: [CypressTestMaskModule],
- });
-
- cy.get('#masked')
- .type('-0.01')
- .should('have.value', '-0.01')
- .type('{leftArrow}')
- .type('{backspace}')
- .should('have.value', '-0.1')
- .type('{leftArrow}')
- .type('{backspace}')
- .should('have.value', '-1');
- });
-
- it('should correct work after backspace separator.3 decimalMarker . thousandSeparator , allowNegative', () => {
- cy.mount(CypressTestMaskComponent, {
- componentProperties: {
- mask: 'separator.3',
- decimalMarker: '.',
- thousandSeparator: ',',
- allowNegativeNumbers: true,
- },
- imports: [CypressTestMaskModule],
- });
-
- cy.get('#masked')
- .type('-0.014')
- .should('have.value', '-0.014')
- .type('{leftArrow}'.repeat(2))
- .type('{backspace}')
- .should('have.value', '-0.14')
- .type('{leftArrow}'.repeat(2))
- .type('{backspace}')
- .should('have.value', '-14');
- });
-
- it('should correct work after backspace separator.3 leadZero allowNegative', () => {
- cy.mount(CypressTestMaskComponent, {
- componentProperties: {
- mask: 'separator.3',
- allowNegativeNumbers: true,
- leadZero: true,
- },
- imports: [CypressTestMaskModule],
- });
-
- cy.get('#masked')
- .type('-0.1')
- .should('have.value', '-0.1')
- .type('{leftArrow}'.repeat(2))
- .type('{backspace}')
- .should('have.value', '-1')
- .type('{backspace}')
- .should('have.value', '-');
- });
-
- it('should correct work after backspace separator', () => {
- cy.mount(CypressTestMaskComponent, {
- componentProperties: {
- mask: 'separator',
- },
- imports: [CypressTestMaskModule],
- });
-
- cy.get('#masked')
- .type('0.33')
- .should('have.value', '0.33')
- .type('{leftArrow}'.repeat(2))
- .type('{backspace}')
- .should('have.value', '33');
- });
-
- it('should correct work after backspace separator leadZero', () => {
- cy.mount(CypressTestMaskComponent, {
- componentProperties: {
- mask: 'separator',
- leadZero: true,
- },
- imports: [CypressTestMaskModule],
- });
-
- cy.get('#masked')
- .type('0.33')
- .should('have.value', '0.33')
- .type('{leftArrow}'.repeat(2))
- .type('{backspace}')
- .should('have.value', '33');
- });
-
- it('should correct work after backspace separator allowNegativeNumbers', () => {
- cy.mount(CypressTestMaskComponent, {
- componentProperties: {
- mask: 'separator',
- allowNegativeNumbers: true,
- },
- imports: [CypressTestMaskModule],
- });
-
- cy.get('#masked')
- .type('-0.33')
- .should('have.value', '-0.33')
- .type('{leftArrow}'.repeat(2))
- .type('{backspace}')
- .should('have.value', '-33');
- });
-
- it('should correct work after backspace separator leadZero', () => {
- cy.mount(CypressTestMaskComponent, {
- componentProperties: {
- mask: 'separator',
- leadZero: true,
- allowNegativeNumbers: true,
- },
- imports: [CypressTestMaskModule],
- });
-
- cy.get('#masked')
- .type('-0.33')
- .should('have.value', '-0.33')
- .type('{leftArrow}'.repeat(2))
- .type('{backspace}')
- .should('have.value', '-33');
- });
-
- it('should correct work after backspace separator.2 when first digit .', () => {
- cy.mount(CypressTestMaskComponent, {
- componentProperties: {
- mask: 'separator.2',
- thousandSeparator: '.',
- },
- imports: [CypressTestMaskModule],
- });
-
- cy.get('#masked')
- .type('50004')
- .should('have.value', '50.004')
- .type('{leftArrow}'.repeat(5))
- .type('{backspace}')
- .should('have.value', '4');
- });
-
- it('should correct work after backspace separator.2 when first digit ,', () => {
- cy.mount(CypressTestMaskComponent, {
- componentProperties: {
- mask: 'separator.2',
- thousandSeparator: ',',
- },
- imports: [CypressTestMaskModule],
- });
-
- cy.get('#masked')
- .type('50004')
- .should('have.value', '50,004')
- .type('{leftArrow}'.repeat(5))
- .type('{backspace}')
- .should('have.value', '4');
- });
-
- it('should correct work after backspace separator.2 when first digit whitespace', () => {
- cy.mount(CypressTestMaskComponent, {
- componentProperties: {
- mask: 'separator.2',
- thousandSeparator: ' ',
- },
- imports: [CypressTestMaskModule],
- });
-
- cy.get('#masked')
- .type('50004')
- .should('have.value', '50 004')
- .type('{leftArrow}'.repeat(5))
- .type('{backspace}')
- .should('have.value', '4');
- });
-
it('should backspace with mask Hh:m0', () => {
cy.mount(CypressTestMaskComponent, {
componentProperties: {
diff --git a/projects/ngx-mask-lib/src/test/keep-character-position.cy-spec.ts b/projects/ngx-mask-lib/src/test/keep-character-position.cy-spec.ts
index 00448085..bfba7384 100644
--- a/projects/ngx-mask-lib/src/test/keep-character-position.cy-spec.ts
+++ b/projects/ngx-mask-lib/src/test/keep-character-position.cy-spec.ts
@@ -207,4 +207,47 @@ describe('Directive: Mask (Delete)', () => {
.type('{backspace}'.repeat(2))
.should('have.value', '12/__/567 test');
});
+
+ it('should delete character from del', () => {
+ cy.mount(CypressTestMaskComponent, {
+ componentProperties: {
+ mask: '000-000-000',
+ keepCharacterPositions: true,
+ showMaskTyped: true,
+ },
+ imports: [CypressTestMaskModule],
+ });
+
+ cy.get('#masked')
+ .type('123456789')
+ .type('{leftArrow}'.repeat(11))
+ .type('{del}'.repeat(11))
+ .should('have.value', '___-___-___');
+
+ cy.get('#masked').clear();
+ cy.get('#masked')
+ .type('123456789')
+ .type('{leftArrow}'.repeat(4))
+ .type('{del}')
+ .should('have.value', '123-456-789')
+ .should('have.prop', 'selectionStart', 8);
+ });
+
+ it('should delete character from del', () => {
+ cy.mount(CypressTestMaskComponent, {
+ componentProperties: {
+ mask: '0000 0000 0000 0000',
+ keepCharacterPositions: true,
+ showMaskTyped: true,
+ },
+ imports: [CypressTestMaskModule],
+ });
+
+ cy.get('#masked')
+ .type('1234567891011121')
+ .type('{leftArrow}'.repeat(5))
+ .type('{del}')
+ .should('have.value', '1234 5678 9101 1121')
+ .should('have.prop', 'selectionStart', 15);
+ });
});
diff --git a/projects/ngx-mask-lib/src/test/separator.cy-spec.ts b/projects/ngx-mask-lib/src/test/separator.cy-spec.ts
new file mode 100644
index 00000000..d7176ed8
--- /dev/null
+++ b/projects/ngx-mask-lib/src/test/separator.cy-spec.ts
@@ -0,0 +1,460 @@
+import { CypressTestMaskModule } from './utils/cypress-test.module';
+import { CypressTestMaskComponent } from './utils/cypress-test-component.component';
+
+describe('Test Date Hh:m0', () => {
+ it('Mask separator.2 check cursor with value 100.0', () => {
+ cy.mount(CypressTestMaskComponent, {
+ componentProperties: {
+ mask: 'separator.2',
+ decimalMarker: '.',
+ thousandSeparator: ',',
+ },
+ imports: [CypressTestMaskModule],
+ });
+ cy.get('#masked')
+ .type('1000')
+ .type('{leftArrow}')
+ .type('.')
+ .should('have.value', '100.0')
+ .should('have.prop', 'selectionStart', 4);
+ });
+
+ it('Mask separator.2 check cursor with value 1.00', () => {
+ cy.mount(CypressTestMaskComponent, {
+ componentProperties: {
+ mask: 'separator.2',
+ decimalMarker: '.',
+ thousandSeparator: ',',
+ },
+ imports: [CypressTestMaskModule],
+ });
+ cy.get('#masked')
+ .type('1000')
+ .type('{leftArrow}'.repeat(3))
+ .type('.')
+ .should('have.value', '1.00')
+ .should('have.prop', 'selectionStart', 2);
+ });
+
+ it('Mask separator.2 check cursor with value 123456789.20', () => {
+ cy.mount(CypressTestMaskComponent, {
+ componentProperties: {
+ mask: 'separator.2',
+ decimalMarker: '.',
+ thousandSeparator: ',',
+ },
+ imports: [CypressTestMaskModule],
+ });
+ cy.get('#masked')
+ .type('123456789.20')
+ .type('{leftArrow}'.repeat(4))
+ .type('.')
+ .should('have.value', '12,345,678.9')
+ .should('have.prop', 'selectionStart', 11);
+ });
+
+ it('Mask separator.2 check cursor with value 100.0', () => {
+ cy.mount(CypressTestMaskComponent, {
+ componentProperties: {
+ mask: 'separator.2',
+ decimalMarker: ',',
+ thousandSeparator: '.',
+ },
+ imports: [CypressTestMaskModule],
+ });
+ cy.get('#masked')
+ .type('1000')
+ .type('{leftArrow}')
+ .type(',')
+ .should('have.value', '100,0')
+ .should('have.prop', 'selectionStart', 4);
+ });
+
+ it('Mask separator.2 check cursor with value 1.00', () => {
+ cy.mount(CypressTestMaskComponent, {
+ componentProperties: {
+ mask: 'separator.2',
+ decimalMarker: ',',
+ thousandSeparator: '.',
+ },
+ imports: [CypressTestMaskModule],
+ });
+ cy.get('#masked')
+ .type('1000')
+ .type('{leftArrow}'.repeat(3))
+ .type(',')
+ .should('have.value', '1,00')
+ .should('have.prop', 'selectionStart', 2);
+ });
+
+ it('Mask separator.2 check cursor with value 123456789.20', () => {
+ cy.mount(CypressTestMaskComponent, {
+ componentProperties: {
+ mask: 'separator.2',
+ decimalMarker: ',',
+ thousandSeparator: '.',
+ },
+ imports: [CypressTestMaskModule],
+ });
+ cy.get('#masked')
+ .type('123456789,20')
+ .type('{leftArrow}'.repeat(4))
+ .type(',')
+ .should('have.value', '12.345.678,9')
+ .should('have.prop', 'selectionStart', 11);
+ });
+
+ it('when decimalMarker doenst set should have right position cursor thousandSeparator = .', () => {
+ cy.mount(CypressTestMaskComponent, {
+ componentProperties: {
+ mask: 'separator.2',
+ thousandSeparator: '.',
+ },
+ imports: [CypressTestMaskModule],
+ });
+
+ cy.get('#masked')
+ .type('12345678,00')
+
+ .should('have.value', '12.345.678,00')
+ .type('{leftArrow}'.repeat(3))
+ .type('{backspace}'.repeat(3))
+ .should('have.value', '12.345,00');
+ });
+
+ it('when decimalMarker doenst set should have right position cursor thousandSeparator = ,', () => {
+ cy.mount(CypressTestMaskComponent, {
+ componentProperties: {
+ mask: 'separator.2',
+ thousandSeparator: ',',
+ },
+ imports: [CypressTestMaskModule],
+ });
+
+ cy.get('#masked')
+ .type('12345678.00')
+
+ .should('have.value', '12,345,678.00')
+ .type('{leftArrow}'.repeat(3))
+ .type('{backspace}'.repeat(3))
+ .should('have.value', '12,345.00');
+ });
+
+ it('should place cursor after backspace with separatorLimit = 10 in correct position', () => {
+ cy.mount(CypressTestMaskComponent, {
+ componentProperties: {
+ mask: 'separator.2',
+ separatorLimit: '10',
+ },
+ imports: [CypressTestMaskModule],
+ });
+
+ cy.get('#masked')
+ .type('12.10')
+ .should('have.value', '12.10')
+ .type('{leftArrow}'.repeat(2))
+ .type('{backspace}')
+ .should('have.value', '12')
+ .should('have.prop', 'selectionStart', 2);
+ });
+
+ it('should place cursor after backspace with separatorLimit = 100 in correct position', () => {
+ cy.mount(CypressTestMaskComponent, {
+ componentProperties: {
+ mask: 'separator.2',
+ separatorLimit: '100',
+ },
+ imports: [CypressTestMaskModule],
+ });
+
+ cy.get('#masked')
+ .type('123.10')
+ .should('have.value', '123.10')
+ .type('{leftArrow}'.repeat(2))
+ .type('{backspace}')
+ .should('have.value', '123')
+ .should('have.prop', 'selectionStart', 3);
+ });
+
+ it('should place cursor after backspace with separatorLimit = 1000 in correct position', () => {
+ cy.mount(CypressTestMaskComponent, {
+ componentProperties: {
+ mask: 'separator.2',
+ thousandSeparator: ',',
+ separatorLimit: '1000',
+ },
+ imports: [CypressTestMaskModule],
+ });
+
+ cy.get('#masked')
+ .type('1234.10')
+ .should('have.value', '1,234.10')
+ .type('{leftArrow}'.repeat(2))
+ .type('{backspace}')
+ .should('have.value', '1,234')
+ .should('have.prop', 'selectionStart', 5);
+ });
+
+ it('should backspace with separator and prefix', () => {
+ cy.mount(CypressTestMaskComponent, {
+ componentProperties: {
+ mask: 'separator.2',
+ thousandSeparator: ',',
+ prefix: '$ ',
+ },
+ imports: [CypressTestMaskModule],
+ });
+
+ cy.get('#masked')
+ .type('1234567890')
+ .should('have.value', '$ 1,234,567,890')
+ .type('{leftArrow}'.repeat(3))
+ .type('{backspace}')
+ .should('have.prop', 'selectionStart', 11);
+ });
+
+ it('should backspace with separator and prefix', () => {
+ cy.mount(CypressTestMaskComponent, {
+ componentProperties: {
+ mask: 'separator.2',
+ thousandSeparator: '.',
+ prefix: '$ ',
+ },
+ imports: [CypressTestMaskModule],
+ });
+
+ cy.get('#masked')
+ .type('1234567890')
+ .should('have.value', '$ 1.234.567.890')
+ .type('{leftArrow}'.repeat(3))
+ .type('{backspace}')
+ .should('have.prop', 'selectionStart', 11);
+ });
+
+ it('should correct work after backspace separator.6 decimalMarker . thousandSeparator ,', () => {
+ cy.mount(CypressTestMaskComponent, {
+ componentProperties: {
+ mask: 'separator.6',
+ decimalMarker: '.',
+ thousandSeparator: ',',
+ },
+ imports: [CypressTestMaskModule],
+ });
+
+ cy.get('#masked')
+ .type('0.000001')
+ .should('have.value', '0.000001')
+ .type('{leftArrow}'.repeat(2))
+ .type('{backspace}')
+ .should('have.value', '0.00001')
+ .type('{leftArrow}'.repeat(2))
+ .type('{backspace}')
+ .should('have.value', '0.0001')
+ .type('{backspace}')
+ .should('have.value', '1');
+ });
+
+ it('should correct work after backspace separator.2 decimalMarker . thousandSeparator ,', () => {
+ cy.mount(CypressTestMaskComponent, {
+ componentProperties: {
+ mask: 'separator.2',
+ decimalMarker: '.',
+ thousandSeparator: ',',
+ },
+ imports: [CypressTestMaskModule],
+ });
+
+ cy.get('#masked')
+ .type('0.01')
+ .should('have.value', '0.01')
+ .type('{leftArrow}')
+ .type('{backspace}')
+ .should('have.value', '0.1')
+ .type('{leftArrow}')
+ .type('{backspace}')
+ .should('have.value', '1');
+ });
+
+ it('should correct work after backspace separator.2 decimalMarker . thousandSeparator , allowNegative', () => {
+ cy.mount(CypressTestMaskComponent, {
+ componentProperties: {
+ mask: 'separator.2',
+ decimalMarker: '.',
+ thousandSeparator: ',',
+ allowNegativeNumbers: true,
+ },
+ imports: [CypressTestMaskModule],
+ });
+
+ cy.get('#masked')
+ .type('-0.01')
+ .should('have.value', '-0.01')
+ .type('{leftArrow}')
+ .type('{backspace}')
+ .should('have.value', '-0.1')
+ .type('{leftArrow}')
+ .type('{backspace}')
+ .should('have.value', '-1');
+ });
+
+ it('should correct work after backspace separator.3 decimalMarker . thousandSeparator , allowNegative', () => {
+ cy.mount(CypressTestMaskComponent, {
+ componentProperties: {
+ mask: 'separator.3',
+ decimalMarker: '.',
+ thousandSeparator: ',',
+ allowNegativeNumbers: true,
+ },
+ imports: [CypressTestMaskModule],
+ });
+
+ cy.get('#masked')
+ .type('-0.014')
+ .should('have.value', '-0.014')
+ .type('{leftArrow}'.repeat(2))
+ .type('{backspace}')
+ .should('have.value', '-0.14')
+ .type('{leftArrow}'.repeat(2))
+ .type('{backspace}')
+ .should('have.value', '-14');
+ });
+
+ it('should correct work after backspace separator.3 leadZero allowNegative', () => {
+ cy.mount(CypressTestMaskComponent, {
+ componentProperties: {
+ mask: 'separator.3',
+ allowNegativeNumbers: true,
+ leadZero: true,
+ },
+ imports: [CypressTestMaskModule],
+ });
+
+ cy.get('#masked')
+ .type('-0.1')
+ .should('have.value', '-0.1')
+ .type('{leftArrow}'.repeat(2))
+ .type('{backspace}')
+ .should('have.value', '-1')
+ .type('{backspace}')
+ .should('have.value', '-');
+ });
+
+ it('should correct work after backspace separator', () => {
+ cy.mount(CypressTestMaskComponent, {
+ componentProperties: {
+ mask: 'separator',
+ },
+ imports: [CypressTestMaskModule],
+ });
+
+ cy.get('#masked')
+ .type('0.33')
+ .should('have.value', '0.33')
+ .type('{leftArrow}'.repeat(2))
+ .type('{backspace}')
+ .should('have.value', '33');
+ });
+
+ it('should correct work after backspace separator leadZero', () => {
+ cy.mount(CypressTestMaskComponent, {
+ componentProperties: {
+ mask: 'separator',
+ leadZero: true,
+ },
+ imports: [CypressTestMaskModule],
+ });
+
+ cy.get('#masked')
+ .type('0.33')
+ .should('have.value', '0.33')
+ .type('{leftArrow}'.repeat(2))
+ .type('{backspace}')
+ .should('have.value', '33');
+ });
+
+ it('should correct work after backspace separator allowNegativeNumbers', () => {
+ cy.mount(CypressTestMaskComponent, {
+ componentProperties: {
+ mask: 'separator',
+ allowNegativeNumbers: true,
+ },
+ imports: [CypressTestMaskModule],
+ });
+
+ cy.get('#masked')
+ .type('-0.33')
+ .should('have.value', '-0.33')
+ .type('{leftArrow}'.repeat(2))
+ .type('{backspace}')
+ .should('have.value', '-33');
+ });
+
+ it('should correct work after backspace separator leadZero', () => {
+ cy.mount(CypressTestMaskComponent, {
+ componentProperties: {
+ mask: 'separator',
+ leadZero: true,
+ allowNegativeNumbers: true,
+ },
+ imports: [CypressTestMaskModule],
+ });
+
+ cy.get('#masked')
+ .type('-0.33')
+ .should('have.value', '-0.33')
+ .type('{leftArrow}'.repeat(2))
+ .type('{backspace}')
+ .should('have.value', '-33');
+ });
+
+ it('should correct work after backspace separator.2 when first digit .', () => {
+ cy.mount(CypressTestMaskComponent, {
+ componentProperties: {
+ mask: 'separator.2',
+ thousandSeparator: '.',
+ },
+ imports: [CypressTestMaskModule],
+ });
+
+ cy.get('#masked')
+ .type('50004')
+ .should('have.value', '50.004')
+ .type('{leftArrow}'.repeat(5))
+ .type('{backspace}')
+ .should('have.value', '4');
+ });
+
+ it('should correct work after backspace separator.2 when first digit ,', () => {
+ cy.mount(CypressTestMaskComponent, {
+ componentProperties: {
+ mask: 'separator.2',
+ thousandSeparator: ',',
+ },
+ imports: [CypressTestMaskModule],
+ });
+
+ cy.get('#masked')
+ .type('50004')
+ .should('have.value', '50,004')
+ .type('{leftArrow}'.repeat(5))
+ .type('{backspace}')
+ .should('have.value', '4');
+ });
+
+ it('should correct work after backspace separator.2 when first digit whitespace', () => {
+ cy.mount(CypressTestMaskComponent, {
+ componentProperties: {
+ mask: 'separator.2',
+ thousandSeparator: ' ',
+ },
+ imports: [CypressTestMaskModule],
+ });
+
+ cy.get('#masked')
+ .type('50004')
+ .should('have.value', '50 004')
+ .type('{leftArrow}'.repeat(5))
+ .type('{backspace}')
+ .should('have.value', '4');
+ });
+});
diff --git a/projects/ngx-mask-lib/src/test/separator.spec.ts b/projects/ngx-mask-lib/src/test/separator.spec.ts
index 970bd8dd..93529b7a 100644
--- a/projects/ngx-mask-lib/src/test/separator.spec.ts
+++ b/projects/ngx-mask-lib/src/test/separator.spec.ts
@@ -1835,4 +1835,42 @@ describe('Separator: Mask', () => {
fixture.detectChanges();
expect(inputTarget.value).toBe('10.1000000000');
}));
+
+ it('should support big numbers with separator', () => {
+ component.mask = 'separator';
+
+ equal('12345678910111215', '12 345 678 910 111 215', fixture);
+ expect(component.form.value).toBe('12345678910111215');
+ equal('12345678910111215.9999', '12 345 678 910 111 215.9999', fixture);
+ expect(component.form.value).toBe('12345678910111215.9999');
+ });
+
+ it('should support big numbers with separator 2', () => {
+ component.mask = 'separator.2';
+
+ equal('12345678910111215', '12 345 678 910 111 215', fixture);
+ expect(component.form.value).toBe('12345678910111215');
+ equal('12345678910111215.9999', '12 345 678 910 111 215.99', fixture);
+ expect(component.form.value).toBe('12345678910111215.99');
+ });
+
+ it('should support big numbers with separator 2 thousand =.', () => {
+ component.mask = 'separator.2';
+ component.thousandSeparator = '.';
+
+ equal('12345678910111215', '12.345.678.910.111.215', fixture);
+ expect(component.form.value).toBe('12345678910111215');
+ equal('12345678910111215,99', '12.345.678.910.111.215,99', fixture);
+ expect(component.form.value).toBe('12345678910111215.99');
+ });
+
+ it('should support big numbers with separator 2 thousand =,', () => {
+ component.mask = 'separator.2';
+ component.thousandSeparator = ',';
+
+ equal('12345678910111215', '12,345,678,910,111,215', fixture);
+ expect(component.form.value).toBe('12345678910111215');
+ equal('12345678910111215.9999', '12,345,678,910,111,215.99', fixture);
+ expect(component.form.value).toBe('12345678910111215.99');
+ });
});
diff --git a/projects/ngx-mask-lib/src/test/validation.spec.ts b/projects/ngx-mask-lib/src/test/validation.spec.ts
index cc65a35e..6ce0fdd9 100644
--- a/projects/ngx-mask-lib/src/test/validation.spec.ts
+++ b/projects/ngx-mask-lib/src/test/validation.spec.ts
@@ -87,6 +87,23 @@ export class TestValidatorDropSpecialCharacters {
public dropSpecialCharacters = [' '];
}
+@Component({
+ selector: 'jsdaddy-open-source-test',
+ template: `
+
+ `,
+})
+// eslint-disable-next-line @angular-eslint/component-class-suffix
+export class TestValidatorEmailMask {
+ public form: FormControl = new FormControl('', Validators.required);
+ public mask = 'A*@A*.A*';
+ public dropSpecialCharacters = false;
+}
+
describe('Directive: Mask (Validation)', () => {
describe('Global validation true, validation attribute on input not specified', () => {
let fixture: ComponentFixture;
@@ -422,4 +439,46 @@ describe('Directive: Mask (Validation)', () => {
expect(component.form.valid).toBe(true);
});
});
+
+ describe('Global validation true, email mask', () => {
+ let fixture: ComponentFixture;
+ let component: TestValidatorEmailMask;
+
+ beforeEach(() => {
+ TestBed.configureTestingModule({
+ declarations: [TestValidatorEmailMask],
+ imports: [ReactiveFormsModule, NgxMaskDirective],
+ providers: [provideNgxMask({ validation: true })],
+ });
+ fixture = TestBed.createComponent(TestValidatorEmailMask);
+ component = fixture.componentInstance;
+ fixture.detectChanges();
+ });
+
+ it('email Mask should validated correct', () => {
+ component.mask = 'A*@A*.A*';
+ component.dropSpecialCharacters = false;
+
+ equal('validate', 'validate', fixture);
+ expect(component.form.valid).toBe(false);
+
+ equal('validate@', 'validate@', fixture);
+ expect(component.form.valid).toBe(false);
+
+ equal('validate@some', 'validate@some', fixture);
+ expect(component.form.valid).toBe(false);
+
+ equal('validate@some.', 'validate@some.', fixture);
+ expect(component.form.valid).toBe(false);
+
+ equal('validate@some.e', 'validate@some.e', fixture);
+ expect(component.form.valid).toBe(true);
+
+ equal('validate@some.eu', 'validate@some.eu', fixture);
+ expect(component.form.valid).toBe(true);
+
+ equal('validate@some.com', 'validate@some.com', fixture);
+ expect(component.form.valid).toBe(true);
+ });
+ });
});
diff --git a/src/assets/content/common-cases.ts b/src/assets/content/common-cases.ts
index 70d807f5..b775d193 100644
--- a/src/assets/content/common-cases.ts
+++ b/src/assets/content/common-cases.ts
@@ -63,6 +63,13 @@ export const ComDocs: ComDoc[] = [
id: 8,
anchor: 'email-mask',
},
+ {
+ header: 'Email mask with validation',
+ text: '',
+ code: ``,
+ id: 8,
+ anchor: 'email-mask',
+ },
{
header: 'Allow negative numbers to mask',
text: 'You can allow negative numbers',
@@ -142,8 +149,18 @@ export const ComExamples: TExample[] = [
{
_placeholder: 'Valid email',
_mask: 'A*@A*.SSS',
+ _validation: true,
+ _dropSpecialCharacters: false,
control: { form: new UntypedFormControl(''), model: '' },
},
+ {
+ _placeholder: 'Valid email',
+ _validation: true,
+ _dropSpecialCharacters: false,
+ _mask: 'A*@A*.A*',
+ control: { form: new UntypedFormControl(''), model: '' },
+ },
+
{
_placeholder: 'allowNegativeNumbers mask',
_allowNegativeNumbers: true,