diff --git a/projects/demo/src/app/app.routes.ts b/projects/demo/src/app/app.routes.ts index 1e6187a15..fedaa0314 100644 --- a/projects/demo/src/app/app.routes.ts +++ b/projects/demo/src/app/app.routes.ts @@ -219,6 +219,13 @@ export const appRoutes: Routes = [ title: 'Supported types', }, }, + { + path: DemoPath.RealWorldForm, + loadComponent: () => import('../pages/documentation/real-world-form'), + data: { + title: 'Maskito in Real World Form', + }, + }, { path: DemoPath.Changelog, loadComponent: () => diff --git a/projects/demo/src/app/constants/demo-path.ts b/projects/demo/src/app/constants/demo-path.ts index f67da32ca..b43ca7b67 100644 --- a/projects/demo/src/app/constants/demo-path.ts +++ b/projects/demo/src/app/constants/demo-path.ts @@ -26,6 +26,7 @@ export const DemoPath = { Placeholder: 'recipes/placeholder', BrowserSupport: 'browser-support', SupportedInputTypes: 'supported-input-types', + RealWorldForm: 'real-world-form', Changelog: 'changelog', Stackblitz: 'stackblitz', } as const; diff --git a/projects/demo/src/pages/documentation/real-world-form/index.html b/projects/demo/src/pages/documentation/real-world-form/index.html new file mode 100644 index 000000000..58158b700 --- /dev/null +++ b/projects/demo/src/pages/documentation/real-world-form/index.html @@ -0,0 +1,174 @@ +
diff --git a/projects/demo/src/pages/documentation/real-world-form/index.less b/projects/demo/src/pages/documentation/real-world-form/index.less new file mode 100644 index 000000000..71220e209 --- /dev/null +++ b/projects/demo/src/pages/documentation/real-world-form/index.less @@ -0,0 +1,16 @@ +:host { + display: flex; + justify-content: center; + align-items: center; + padding-top: 2rem; +} + +form { + width: 80%; + max-width: 40rem; +} + +.password-icon { + pointer-events: all; + cursor: pointer; +} diff --git a/projects/demo/src/pages/documentation/real-world-form/index.ts b/projects/demo/src/pages/documentation/real-world-form/index.ts new file mode 100644 index 000000000..ac224d6a4 --- /dev/null +++ b/projects/demo/src/pages/documentation/real-world-form/index.ts @@ -0,0 +1,111 @@ +import {ChangeDetectionStrategy, Component, inject} from '@angular/core'; +import {FormControl, FormGroup, ReactiveFormsModule} from '@angular/forms'; +import {MaskitoDirective} from '@maskito/angular'; +import type {MaskitoOptions} from '@maskito/core'; +import { + maskitoAddOnFocusPlugin, + maskitoDateOptionsGenerator, + maskitoNumberOptionsGenerator, + maskitoRemoveOnBlurPlugin, +} from '@maskito/kit'; +import {maskitoGetCountryFromNumber, maskitoPhoneOptionsGenerator} from '@maskito/phone'; +import {TUI_IS_APPLE} from '@taiga-ui/cdk'; +import { + TuiButtonModule, + TuiFlagPipeModule, + TuiSvgModule, + TuiTextfieldControllerModule, +} from '@taiga-ui/core'; +import {TuiInputModule, TuiTextareaModule} from '@taiga-ui/kit'; +import metadata from 'libphonenumber-js/min/metadata'; + +const MONEY_AMOUNT_MASK = maskitoNumberOptionsGenerator({ + min: 0, + prefix: '$ ', + precision: 2, +}); + +const ONLY_LATIN_LETTERS_RE = /^[a-z]+$/i; + +@Component({ + standalone: true, + selector: 'real-world-form', + imports: [ + ReactiveFormsModule, + TuiInputModule, + TuiButtonModule, + TuiTextfieldControllerModule, + TuiTextareaModule, + MaskitoDirective, + TuiFlagPipeModule, + TuiSvgModule, + ], + templateUrl: './index.html', + styleUrls: ['./index.less'], + changeDetection: ChangeDetectionStrategy.OnPush, +}) +export default class RealWorldForm { + private readonly isApple = inject(TUI_IS_APPLE); + + protected readonly form = new FormGroup({ + name: new FormControl(''), + surname: new FormControl(''), + phone: new FormControl(''), + password: new FormControl(''), + repeatedPassword: new FormControl(''), + transactionDate: new FormControl(''), + transactionAmount: new FormControl(''), + address: new FormControl(''), + }); + + protected nameMask: MaskitoOptions = { + mask: ONLY_LATIN_LETTERS_RE, + }; + + protected surnameMask: MaskitoOptions = { + mask: ONLY_LATIN_LETTERS_RE, + postprocessors: [ + ({value, selection}) => ({selection, value: value.toUpperCase()}), + ], + }; + + protected readonly phoneMask = maskitoPhoneOptionsGenerator({ + metadata, + strict: false, + }); + + protected passwordMask: MaskitoOptions = { + mask: /^\d*[a-z]?\d*$/i, + }; + + protected readonly transactionDateMask = maskitoDateOptionsGenerator({ + mode: 'dd/mm/yyyy', + }); + + protected readonly transactionAmountMask: MaskitoOptions = { + ...MONEY_AMOUNT_MASK, + plugins: [ + ...MONEY_AMOUNT_MASK.plugins, + maskitoAddOnFocusPlugin('$ '), + maskitoRemoveOnBlurPlugin('$ '), + ], + }; + + protected readonly addressMask: MaskitoOptions = { + mask: /^[a-z1-9\s.,/]+$/i, + }; + + protected showPassword = false; + + protected get countryIsoCode(): string { + return maskitoGetCountryFromNumber(this.form.value.phone || '', metadata) || ''; + } + + protected get phoneTextfieldPattern(): string { + return this.isApple ? '+[0-9-]{1,20}' : ''; + } + + protected log(something: any): void { + console.info(something); + } +} diff --git a/projects/demo/src/pages/pages.ts b/projects/demo/src/pages/pages.ts index 588903f94..855578ba5 100644 --- a/projects/demo/src/pages/pages.ts +++ b/projects/demo/src/pages/pages.ts @@ -167,6 +167,12 @@ export const DEMO_PAGES: TuiDocPages = [ keywords: 'input, type, text, password, search, tel, url, email, number, date, month', }, + { + section: 'Other', + title: 'Maskito in Real World Form', + route: DemoPath.RealWorldForm, + keywords: 'browser, autofill, showcase, in, action, demo', + }, { section: 'Other', title: 'Changelog',