UNPKG

ngx-otp-code

Version:

Customizable OTP input component for Angular with Web OTP support.

181 lines 23.4 kB
import { NgClass, NgFor } from '@angular/common'; import { Component, EventEmitter, forwardRef, Input, Output, ViewChildren } from '@angular/core'; import { NG_VALUE_ACCESSOR } from '@angular/forms'; import * as i0 from "@angular/core"; export class NgxOtpCodeComponent { get length() { return this.config.length ?? 6; } get inputType() { return this.config.inputType ?? 'text'; } get placeholder() { return this.config.placeholder ?? ''; } get autoFocus() { return this.config.autoFocus ?? true; } get isAlpha() { return this.config.isAlpha ?? false; } get mask() { return this.config.mask ?? false; } get useWebOtp() { return this.config.useWebOtp ?? false; } get inputClass() { return this.config.inputClass ?? ''; } constructor(renderer) { this.renderer = renderer; this.config = {}; this.codeFilled = new EventEmitter(); this.error = new EventEmitter(); this.arrayValue = []; this.value = ''; this.nameItem = 'otp-'; this.onChange = () => { }; this.onTouched = () => { }; } ngAfterViewInit() { if (this.autoFocus) { this.focusInput(0); } if (this.useWebOtp && 'OTPCredential' in window) { this.initWebOtp(); } } initWebOtp() { const ac = new AbortController(); navigator.credentials.get({ otp: { transport: ['sms'] }, signal: ac.signal }).then((otp) => { if (otp?.code) { this.setOtpValue(otp.code); } }).catch((err) => { console.log(err); }); } onKey(event, index) { const input = event.target; const key = event.key; if (key === 'Backspace') { this.arrayValue[index] = ''; input.value = ''; if (index > 0) { this.focusInput(index - 1); } this.updateValue(); event.preventDefault(); return; } // Allow only a single char per box if (key.length === 1) { // Check validity based on input type if (this.inputType === 'number' && !/^\d$/.test(key)) { event.preventDefault(); return; } if (this.isAlpha && !/^[A-Za-z0-9]$/.test(key)) { event.preventDefault(); return; } // Assign value this.arrayValue[index] = key; input.value = key; // Move to next input if not last if (index < this.length - 1) { this.focusInput(index + 1); } else { input.blur(); // Optionally blur if finished } this.updateValue(); event.preventDefault(); } else { // Ignore other keys (arrows, etc.) event.preventDefault(); } } onPaste(event) { event.preventDefault(); const pasted = event.clipboardData?.getData('text') || ''; if (this.inputType === 'number' && /\D/.test(pasted)) { this.error.emit('El código solo debe contener números.'); return; } if (pasted.length < this.length) { this.error.emit(`La cantidad de caracteres no puede ser menor a ${this.length}.`); return; } this.setOtpValue(pasted); } setOtpValue(value) { this.arrayValue = value.split('').slice(0, this.length); this.otpInputs?.forEach((ref, i) => { ref.nativeElement.value = this.arrayValue[i] || ''; }); this.updateValue(); } focusInput(index) { const el = this.otpInputs?.get(index); if (el) { el.nativeElement.focus(); el.nativeElement.select(); } } updateValue() { this.value = this.arrayValue.join(''); this.onChange(this.value); if (this.value.length === this.length) { this.codeFilled.emit(this.value); } } writeValue(value) { this.value = value || ''; this.arrayValue = this.value.split(''); setTimeout(() => { this.otpInputs?.forEach((ref, i) => { ref.nativeElement.value = this.arrayValue[i] || ''; }); }); } registerOnChange(fn) { this.onChange = fn; } registerOnTouched(fn) { this.onTouched = fn; } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: NgxOtpCodeComponent, deps: [{ token: i0.Renderer2 }], target: i0.ɵɵFactoryTarget.Component }); } static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: NgxOtpCodeComponent, isStandalone: true, selector: "ngx-otp-code", inputs: { config: "config" }, outputs: { codeFilled: "codeFilled", error: "error" }, providers: [ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => NgxOtpCodeComponent), multi: true } ], viewQueries: [{ propertyName: "otpInputs", predicate: ["otpInput"], descendants: true }], ngImport: i0, template: "<div class=\"otp-container\" [ngClass]=\"config?.containerClass\">\n <input\n *ngFor=\"let item of [].constructor(length); let i = index\"\n #otpInput\n [type]=\"config?.mask ? 'password' : config?.inputType || 'text'\"\n [attr.placeholder]=\"config?.placeholder\"\n maxlength=\"1\"\n (keydown)=\"onKey($event, i)\"\n (paste)=\"onPaste($event)\"\n autocomplete=\"one-time-code\"\n class=\"otp-input\"\n [ngClass]=\"inputClass\"\n [attr.aria-label]=\"'C\u00F3digo ' + (i + 1)\"\n />\n</div>\n", styles: [".otp-container{display:flex;gap:.5rem;justify-content:center;align-items:center}.otp-container input::-webkit-outer-spin-button,.otp-container input::-webkit-inner-spin-button{-webkit-appearance:none;margin:0}.otp-container input[type=number]{-moz-appearance:textfield}.otp-container .otp-input{width:3rem;height:3rem;text-align:center;font-size:1.5rem;border:1px solid #ccc;border-radius:4px}.otp-container .otp-input:focus{border-color:#007bff;outline:none}\n"], dependencies: [{ kind: "directive", type: NgFor, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }] }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: NgxOtpCodeComponent, decorators: [{ type: Component, args: [{ selector: 'ngx-otp-code', providers: [ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => NgxOtpCodeComponent), multi: true } ], standalone: true, imports: [NgFor, NgClass], template: "<div class=\"otp-container\" [ngClass]=\"config?.containerClass\">\n <input\n *ngFor=\"let item of [].constructor(length); let i = index\"\n #otpInput\n [type]=\"config?.mask ? 'password' : config?.inputType || 'text'\"\n [attr.placeholder]=\"config?.placeholder\"\n maxlength=\"1\"\n (keydown)=\"onKey($event, i)\"\n (paste)=\"onPaste($event)\"\n autocomplete=\"one-time-code\"\n class=\"otp-input\"\n [ngClass]=\"inputClass\"\n [attr.aria-label]=\"'C\u00F3digo ' + (i + 1)\"\n />\n</div>\n", styles: [".otp-container{display:flex;gap:.5rem;justify-content:center;align-items:center}.otp-container input::-webkit-outer-spin-button,.otp-container input::-webkit-inner-spin-button{-webkit-appearance:none;margin:0}.otp-container input[type=number]{-moz-appearance:textfield}.otp-container .otp-input{width:3rem;height:3rem;text-align:center;font-size:1.5rem;border:1px solid #ccc;border-radius:4px}.otp-container .otp-input:focus{border-color:#007bff;outline:none}\n"] }] }], ctorParameters: () => [{ type: i0.Renderer2 }], propDecorators: { config: [{ type: Input }], codeFilled: [{ type: Output }], error: [{ type: Output }], otpInputs: [{ type: ViewChildren, args: ['otpInput'] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ngx-otp-code.component.js","sourceRoot":"","sources":["../../../../projects/ngx-otp/src/lib/ngx-otp-code.component.ts","../../../../projects/ngx-otp/src/lib/ngx-otp-code.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,MAAM,iBAAiB,CAAC;AACjD,OAAO,EAAiB,SAAS,EAAc,YAAY,EAAE,UAAU,EAAE,KAAK,EAAE,MAAM,EAAwB,YAAY,EAAE,MAAM,eAAe,CAAC;AAClJ,OAAO,EAAwB,iBAAiB,EAAE,MAAM,gBAAgB,CAAC;;AAiBzE,MAAM,OAAO,mBAAmB;IAe9B,IAAI,MAAM;QACR,OAAO,IAAI,CAAC,MAAM,CAAC,MAAM,IAAI,CAAC,CAAC;IACjC,CAAC;IAED,IAAI,SAAS;QACX,OAAO,IAAI,CAAC,MAAM,CAAC,SAAS,IAAI,MAAM,CAAC;IACzC,CAAC;IAED,IAAI,WAAW;QACb,OAAO,IAAI,CAAC,MAAM,CAAC,WAAW,IAAI,EAAE,CAAC;IACvC,CAAC;IAED,IAAI,SAAS;QACX,OAAO,IAAI,CAAC,MAAM,CAAC,SAAS,IAAI,IAAI,CAAC;IACvC,CAAC;IAED,IAAI,OAAO;QACT,OAAO,IAAI,CAAC,MAAM,CAAC,OAAO,IAAI,KAAK,CAAC;IACtC,CAAC;IAED,IAAI,IAAI;QACN,OAAO,IAAI,CAAC,MAAM,CAAC,IAAI,IAAI,KAAK,CAAC;IACnC,CAAC;IAED,IAAI,SAAS;QACX,OAAO,IAAI,CAAC,MAAM,CAAC,SAAS,IAAI,KAAK,CAAC;IACxC,CAAC;IAED,IAAI,UAAU;QACZ,OAAO,IAAI,CAAC,MAAM,CAAC,UAAU,IAAI,EAAE,CAAC;IACtC,CAAC;IAED,YAAoB,QAAmB;QAAnB,aAAQ,GAAR,QAAQ,CAAW;QA9C9B,WAAM,GAAiB,EAAE,CAAC;QAEzB,eAAU,GAAG,IAAI,YAAY,EAAU,CAAC;QACxC,UAAK,GAAG,IAAI,YAAY,EAAU,CAAC;QAI7C,eAAU,GAAa,EAAE,CAAC;QAC1B,UAAK,GAAW,EAAE,CAAC;QACnB,aAAQ,GAAG,MAAM,CAAC;QAEV,aAAQ,GAAQ,GAAG,EAAE,GAAE,CAAC,CAAC;QACzB,cAAS,GAAQ,GAAG,EAAE,GAAE,CAAC,CAAC;IAkCQ,CAAC;IAE3C,eAAe;QACb,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;YACnB,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;QACrB,CAAC;QAED,IAAI,IAAI,CAAC,SAAS,IAAI,eAAe,IAAI,MAAM,EAAE,CAAC;YAChD,IAAI,CAAC,UAAU,EAAE,CAAC;QACpB,CAAC;IACH,CAAC;IAED,UAAU;QACR,MAAM,EAAE,GAAG,IAAI,eAAe,EAAE,CAAC;QAChC,SAAiB,CAAC,WAAW,CAAC,GAAG,CAAC;YACjC,GAAG,EAAE,EAAE,SAAS,EAAE,CAAC,KAAK,CAAC,EAAE;YAC3B,MAAM,EAAE,EAAE,CAAC,MAAM;SAClB,CAAC,CAAC,IAAI,CAAC,CAAC,GAAQ,EAAE,EAAE;YACnB,IAAI,GAAG,EAAE,IAAI,EAAE,CAAC;gBACd,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;YAC7B,CAAC;QACH,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,GAAO,EAAE,EAAE;YACnB,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;QACnB,CAAC,CAAC,CAAC;IACL,CAAC;IAED,KAAK,CAAC,KAAoB,EAAE,KAAa;QACvC,MAAM,KAAK,GAAG,KAAK,CAAC,MAA0B,CAAC;QAC/C,MAAM,GAAG,GAAG,KAAK,CAAC,GAAG,CAAC;QAEtB,IAAI,GAAG,KAAK,WAAW,EAAE,CAAC;YACxB,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,GAAG,EAAE,CAAC;YAC5B,KAAK,CAAC,KAAK,GAAG,EAAE,CAAC;YACjB,IAAI,KAAK,GAAG,CAAC,EAAE,CAAC;gBACd,IAAI,CAAC,UAAU,CAAC,KAAK,GAAG,CAAC,CAAC,CAAC;YAC7B,CAAC;YACD,IAAI,CAAC,WAAW,EAAE,CAAC;YACnB,KAAK,CAAC,cAAc,EAAE,CAAC;YACvB,OAAO;QACT,CAAC;QAED,mCAAmC;QACnC,IAAI,GAAG,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACrB,qCAAqC;YACrC,IAAI,IAAI,CAAC,SAAS,KAAK,QAAQ,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC;gBACrD,KAAK,CAAC,cAAc,EAAE,CAAC;gBACvB,OAAO;YACT,CAAC;YAED,IAAI,IAAI,CAAC,OAAO,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC;gBAC/C,KAAK,CAAC,cAAc,EAAE,CAAC;gBACvB,OAAO;YACT,CAAC;YAED,eAAe;YACf,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,GAAG,GAAG,CAAC;YAC7B,KAAK,CAAC,KAAK,GAAG,GAAG,CAAC;YAElB,iCAAiC;YACjC,IAAI,KAAK,GAAG,IAAI,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBAC5B,IAAI,CAAC,UAAU,CAAC,KAAK,GAAG,CAAC,CAAC,CAAC;YAC7B,CAAC;iBAAM,CAAC;gBACN,KAAK,CAAC,IAAI,EAAE,CAAC,CAAC,8BAA8B;YAC9C,CAAC;YAED,IAAI,CAAC,WAAW,EAAE,CAAC;YACnB,KAAK,CAAC,cAAc,EAAE,CAAC;QACzB,CAAC;aAAM,CAAC;YACN,mCAAmC;YACnC,KAAK,CAAC,cAAc,EAAE,CAAC;QACzB,CAAC;IACH,CAAC;IAGD,OAAO,CAAC,KAAqB;QAC3B,KAAK,CAAC,cAAc,EAAE,CAAC;QACvB,MAAM,MAAM,GAAG,KAAK,CAAC,aAAa,EAAE,OAAO,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC;QAE1D,IAAI,IAAI,CAAC,SAAS,KAAK,QAAQ,IAAI,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC;YACrD,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,uCAAuC,CAAC,CAAC;YACzD,OAAO;QACT,CAAC;QAED,IAAI,MAAM,CAAC,MAAM,GAAG,IAAI,CAAC,MAAM,EAAE,CAAC;YAChC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,kDAAkD,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC;YAClF,OAAO;QACT,CAAC;QAED,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;IAC3B,CAAC;IAEO,WAAW,CAAC,KAAa;QAC/B,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,IAAI,CAAC,MAAM,CAAC,CAAC;QACxD,IAAI,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE;YACjC,GAAG,CAAC,aAAa,CAAC,KAAK,GAAG,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;QACrD,CAAC,CAAC,CAAC;QACH,IAAI,CAAC,WAAW,EAAE,CAAC;IACrB,CAAC;IAEO,UAAU,CAAC,KAAa;QAC9B,MAAM,EAAE,GAAG,IAAI,CAAC,SAAS,EAAE,GAAG,CAAC,KAAK,CAAC,CAAC;QACtC,IAAI,EAAE,EAAE,CAAC;YACP,EAAE,CAAC,aAAa,CAAC,KAAK,EAAE,CAAC;YACzB,EAAE,CAAC,aAAa,CAAC,MAAM,EAAE,CAAC;QAC5B,CAAC;IACH,CAAC;IAEO,WAAW;QACjB,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACtC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC1B,IAAI,IAAI,CAAC,KAAK,CAAC,MAAM,KAAK,IAAI,CAAC,MAAM,EAAE,CAAC;YACtC,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACnC,CAAC;IACH,CAAC;IAED,UAAU,CAAC,KAAU;QACnB,IAAI,CAAC,KAAK,GAAG,KAAK,IAAI,EAAE,CAAC;QACzB,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;QACvC,UAAU,CAAC,GAAG,EAAE;YACd,IAAI,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE;gBACjC,GAAG,CAAC,aAAa,CAAC,KAAK,GAAG,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;YACrD,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;IAED,gBAAgB,CAAC,EAAO;QACtB,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;IACrB,CAAC;IAED,iBAAiB,CAAC,EAAO;QACvB,IAAI,CAAC,SAAS,GAAG,EAAE,CAAC;IACtB,CAAC;+GAlLU,mBAAmB;mGAAnB,mBAAmB,gJAVnB;YACT;gBACE,OAAO,EAAE,iBAAiB;gBAC1B,WAAW,EAAE,UAAU,CAAC,GAAG,EAAE,CAAC,mBAAmB,CAAC;gBAClD,KAAK,EAAE,IAAI;aACZ;SACF,oHCfH,ghBAeA,ugBDEY,KAAK,mHAAC,OAAO;;4FAEZ,mBAAmB;kBAd/B,SAAS;+BACE,cAAc,aAGb;wBACT;4BACE,OAAO,EAAE,iBAAiB;4BAC1B,WAAW,EAAE,UAAU,CAAC,GAAG,EAAE,oBAAoB,CAAC;4BAClD,KAAK,EAAE,IAAI;yBACZ;qBACF,cACW,IAAI,WACP,CAAC,KAAK,EAAC,OAAO,CAAC;8EAGf,MAAM;sBAAd,KAAK;gBAEI,UAAU;sBAAnB,MAAM;gBACG,KAAK;sBAAd,MAAM;gBAEmB,SAAS;sBAAlC,YAAY;uBAAC,UAAU","sourcesContent":["import { NgClass, NgFor } from '@angular/common';\nimport { AfterViewInit, Component, ElementRef, EventEmitter, forwardRef, Input, Output, QueryList, Renderer2, ViewChildren } from '@angular/core';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\nimport { NgxOtpConfig } from './ngx-otp-code.interface';\n\n@Component({\n  selector: 'ngx-otp-code',\n  templateUrl:'./ngx-otp-code.component.html',\n  styleUrl: './ngx-otp-code.component.scss',\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => NgxOtpCodeComponent),\n      multi: true\n    }\n  ],\n  standalone: true,\n  imports: [NgFor,NgClass]\n})\nexport class NgxOtpCodeComponent implements ControlValueAccessor, AfterViewInit {\n  @Input() config: NgxOtpConfig = {};\n\n  @Output() codeFilled = new EventEmitter<string>();\n  @Output() error = new EventEmitter<string>();\n\n  @ViewChildren('otpInput') otpInputs!: QueryList<ElementRef<HTMLInputElement>>;\n\n  arrayValue: string[] = [];\n  value: string = '';\n  nameItem = 'otp-';\n  \n  private onChange: any = () => {};\n  private onTouched: any = () => {};\n\n  get length(): number {\n    return this.config.length ?? 6;\n  }\n\n  get inputType(): 'text' | 'number' {\n    return this.config.inputType ?? 'text';\n  }\n\n  get placeholder(): string {\n    return this.config.placeholder ?? '';\n  }\n\n  get autoFocus(): boolean {\n    return this.config.autoFocus ?? true;\n  }\n\n  get isAlpha(): boolean {\n    return this.config.isAlpha ?? false;\n  }\n\n  get mask(): boolean {\n    return this.config.mask ?? false;\n  }\n\n  get useWebOtp(): boolean {\n    return this.config.useWebOtp ?? false;\n  }\n\n  get inputClass(): string {\n    return this.config.inputClass ?? '';\n  }\n\n  constructor(private renderer: Renderer2) {}\n\n  ngAfterViewInit(): void {\n    if (this.autoFocus) {\n      this.focusInput(0);\n    }\n\n    if (this.useWebOtp && 'OTPCredential' in window) {\n      this.initWebOtp();\n    }\n  }\n\n  initWebOtp() {\n    const ac = new AbortController();\n    (navigator as any).credentials.get({\n      otp: { transport: ['sms'] },\n      signal: ac.signal\n    }).then((otp: any) => {\n      if (otp?.code) {\n        this.setOtpValue(otp.code);\n      }\n    }).catch((err:any) => {\n      console.log(err);\n    });\n  }\n\n  onKey(event: KeyboardEvent, index: number) {\n    const input = event.target as HTMLInputElement;\n    const key = event.key;\n  \n    if (key === 'Backspace') {\n      this.arrayValue[index] = '';\n      input.value = '';\n      if (index > 0) {\n        this.focusInput(index - 1);\n      }\n      this.updateValue();\n      event.preventDefault();\n      return;\n    }\n  \n    // Allow only a single char per box\n    if (key.length === 1) {\n      // Check validity based on input type\n      if (this.inputType === 'number' && !/^\\d$/.test(key)) {\n        event.preventDefault();\n        return;\n      }\n  \n      if (this.isAlpha && !/^[A-Za-z0-9]$/.test(key)) {\n        event.preventDefault();\n        return;\n      }\n  \n      // Assign value\n      this.arrayValue[index] = key;\n      input.value = key;\n  \n      // Move to next input if not last\n      if (index < this.length - 1) {\n        this.focusInput(index + 1);\n      } else {\n        input.blur(); // Optionally blur if finished\n      }\n  \n      this.updateValue();\n      event.preventDefault();\n    } else {\n      // Ignore other keys (arrows, etc.)\n      event.preventDefault();\n    }\n  }\n  \n\n  onPaste(event: ClipboardEvent) {\n    event.preventDefault();\n    const pasted = event.clipboardData?.getData('text') || '';\n\n    if (this.inputType === 'number' && /\\D/.test(pasted)) {\n      this.error.emit('El código solo debe contener números.');\n      return;\n    }\n\n    if (pasted.length < this.length) {\n      this.error.emit(`La cantidad de caracteres no puede ser menor a ${this.length}.`);\n      return;\n    }\n\n    this.setOtpValue(pasted);\n  }\n\n  private setOtpValue(value: string) {\n    this.arrayValue = value.split('').slice(0, this.length);\n    this.otpInputs?.forEach((ref, i) => {\n      ref.nativeElement.value = this.arrayValue[i] || '';\n    });\n    this.updateValue();\n  }\n\n  private focusInput(index: number) {\n    const el = this.otpInputs?.get(index);\n    if (el) {\n      el.nativeElement.focus();\n      el.nativeElement.select();\n    }\n  }\n\n  private updateValue() {\n    this.value = this.arrayValue.join('');\n    this.onChange(this.value);\n    if (this.value.length === this.length) {\n      this.codeFilled.emit(this.value);\n    }\n  }\n\n  writeValue(value: any): void {\n    this.value = value || '';\n    this.arrayValue = this.value.split('');\n    setTimeout(() => {\n      this.otpInputs?.forEach((ref, i) => {\n        ref.nativeElement.value = this.arrayValue[i] || '';\n      });\n    });\n  }\n\n  registerOnChange(fn: any): void {\n    this.onChange = fn;\n  }\n\n  registerOnTouched(fn: any): void {\n    this.onTouched = fn;\n  }\n}\n","<div class=\"otp-container\" [ngClass]=\"config?.containerClass\">\n  <input\n    *ngFor=\"let item of [].constructor(length); let i = index\"\n    #otpInput\n    [type]=\"config?.mask ? 'password' : config?.inputType || 'text'\"\n    [attr.placeholder]=\"config?.placeholder\"\n    maxlength=\"1\"\n    (keydown)=\"onKey($event, i)\"\n    (paste)=\"onPaste($event)\"\n    autocomplete=\"one-time-code\"\n    class=\"otp-input\"\n    [ngClass]=\"inputClass\"\n    [attr.aria-label]=\"'Código ' + (i + 1)\"\n  />\n</div>\n"]}