UNPKG

ng-otp-input

Version:

A fully customizable, one-time password input component for the web built with Angular.

384 lines 28.7 kB
/** * @fileoverview added by tsickle * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc */ import { Component, Input, Output, EventEmitter } from '@angular/core'; import { FormGroup, FormControl } from '@angular/forms'; import { KeysPipe } from '../../pipes/keys.pipe'; import { Config } from '../../models/config'; var NgOtpInputComponent = /** @class */ (function () { function NgOtpInputComponent(keysPipe) { this.keysPipe = keysPipe; this.config = { length: 4 }; // tslint:disable-next-line: no-output-on-prefix this.onInputChange = new EventEmitter(); this.inputControls = new Array(this.config.length); this.componentKey = Math.random() .toString(36) .substring(2) + new Date().getTime().toString(36); } /** * @return {?} */ NgOtpInputComponent.prototype.ngOnInit = /** * @return {?} */ function () { this.otpForm = new FormGroup({}); for (var index = 0; index < this.config.length; index++) { this.otpForm.addControl(this.getControlName(index), new FormControl()); } this.inputType = this.getInputType(); }; /** * @return {?} */ NgOtpInputComponent.prototype.ngAfterViewInit = /** * @return {?} */ function () { var _this = this; if (!this.config.disableAutoFocus) { /** @type {?} */ var containerItem = document.getElementById("c_" + this.componentKey); if (containerItem) { containerItem.addEventListener('paste', (/** * @param {?} evt * @return {?} */ function (evt) { return _this.handlePaste(evt); })); /** @type {?} */ var ele = containerItem.getElementsByClassName('otp-input')[0]; if (ele && ele.focus) { ele.focus(); } } } }; /** * @private * @param {?} idx * @return {?} */ NgOtpInputComponent.prototype.getControlName = /** * @private * @param {?} idx * @return {?} */ function (idx) { return "ctrl_" + idx; }; /** * @param {?} event * @return {?} */ NgOtpInputComponent.prototype.ifLeftArrow = /** * @param {?} event * @return {?} */ function (event) { return this.ifKeyCode(event, 37); }; /** * @param {?} event * @return {?} */ NgOtpInputComponent.prototype.ifRightArrow = /** * @param {?} event * @return {?} */ function (event) { return this.ifKeyCode(event, 39); }; /** * @param {?} event * @return {?} */ NgOtpInputComponent.prototype.ifBackspaceOrDelete = /** * @param {?} event * @return {?} */ function (event) { return (event.key === 'Backspace' || event.key === 'Delete' || this.ifKeyCode(event, 8) || this.ifKeyCode(event, 46)); }; /** * @param {?} event * @param {?} targetCode * @return {?} */ NgOtpInputComponent.prototype.ifKeyCode = /** * @param {?} event * @param {?} targetCode * @return {?} */ function (event, targetCode) { /** @type {?} */ var key = event.keyCode || event.charCode; // tslint:disable-next-line: triple-equals return key == targetCode ? true : false; }; /** * @param {?} $event * @return {?} */ NgOtpInputComponent.prototype.onKeyDown = /** * @param {?} $event * @return {?} */ function ($event) { /** @type {?} */ var isSpace = this.ifKeyCode($event, 32); if (isSpace) { // prevent space return false; } }; /** * @param {?} $event * @param {?} inputIdx * @return {?} */ NgOtpInputComponent.prototype.onKeyUp = /** * @param {?} $event * @param {?} inputIdx * @return {?} */ function ($event, inputIdx) { /** @type {?} */ var nextInputId = this.appendKey("otp_" + (inputIdx + 1)); /** @type {?} */ var prevInputId = this.appendKey("otp_" + (inputIdx - 1)); if (this.ifRightArrow($event)) { this.setSelected(nextInputId); return; } if (this.ifLeftArrow($event)) { this.setSelected(prevInputId); return; } /** @type {?} */ var isBackspace = this.ifBackspaceOrDelete($event); if (isBackspace && !$event.target.value) { this.setSelected(prevInputId); this.rebuildValue(); return; } if (!$event.target.value) { return; } if (this.ifValidEntry($event)) { this.setSelected(nextInputId); } this.rebuildValue(); }; /** * @param {?} id * @return {?} */ NgOtpInputComponent.prototype.appendKey = /** * @param {?} id * @return {?} */ function (id) { return id + "_" + this.componentKey; }; /** * @param {?} eleId * @return {?} */ NgOtpInputComponent.prototype.setSelected = /** * @param {?} eleId * @return {?} */ function (eleId) { this.focusTo(eleId); /** @type {?} */ var ele = document.getElementById(eleId); if (ele && ele.setSelectionRange) { setTimeout((/** * @return {?} */ function () { ele.setSelectionRange(0, 1); }), 0); } }; /** * @param {?} event * @return {?} */ NgOtpInputComponent.prototype.ifValidEntry = /** * @param {?} event * @return {?} */ function (event) { /** @type {?} */ var inp = String.fromCharCode(event.keyCode); /** @type {?} */ var isMobile = /iPhone|iPad|iPod|Android/i.test(navigator.userAgent); return (isMobile || /[a-zA-Z0-9-_]/.test(inp) || (this.config.allowKeyCodes && this.config.allowKeyCodes.includes(event.keyCode)) || (event.keyCode >= 96 && event.keyCode <= 105)); }; /** * @param {?} eleId * @return {?} */ NgOtpInputComponent.prototype.focusTo = /** * @param {?} eleId * @return {?} */ function (eleId) { /** @type {?} */ var ele = document.getElementById(eleId); if (ele) { ele.focus(); } }; // method to set component value // method to set component value /** * @param {?} value * @return {?} */ NgOtpInputComponent.prototype.setValue = // method to set component value /** * @param {?} value * @return {?} */ function (value) { var _this = this; if (this.config.allowNumbersOnly && isNaN(value)) { return; } this.otpForm.reset(); if (!value) { this.rebuildValue(); return; } value = value.toString().replace(/\s/g, ''); // remove whitespace Array.from(value).forEach((/** * @param {?} c * @param {?} idx * @return {?} */ function (c, idx) { if (_this.otpForm.get(_this.getControlName(idx))) { _this.otpForm.get(_this.getControlName(idx)).setValue(c); } })); if (!this.config.disableAutoFocus) { /** @type {?} */ var containerItem = document.getElementById("c_" + this.componentKey); /** @type {?} */ var indexOfElementToFocus = value.length < this.config.length ? value.length : (this.config.length - 1); /** @type {?} */ var ele = containerItem.getElementsByClassName('otp-input')[indexOfElementToFocus]; if (ele && ele.focus) { ele.focus(); } } this.rebuildValue(); }; /** * @return {?} */ NgOtpInputComponent.prototype.rebuildValue = /** * @return {?} */ function () { var _this = this; /** @type {?} */ var val = ''; this.keysPipe.transform(this.otpForm.controls).forEach((/** * @param {?} k * @return {?} */ function (k) { if (_this.otpForm.controls[k].value) { val += _this.otpForm.controls[k].value; } })); this.onInputChange.emit(val); }; /** * @return {?} */ NgOtpInputComponent.prototype.getInputType = /** * @return {?} */ function () { return this.config.isPasswordInput ? 'password' : this.config.allowNumbersOnly ? 'tel' : 'text'; }; /** * @param {?} e * @return {?} */ NgOtpInputComponent.prototype.handlePaste = /** * @param {?} e * @return {?} */ function (e) { // Get pasted data via clipboard API /** @type {?} */ var clipboardData = e.clipboardData || window['clipboardData']; if (clipboardData) { /** @type {?} */ var pastedData = clipboardData.getData('Text'); } // Stop data actually being pasted into div e.stopPropagation(); e.preventDefault(); if (!pastedData) { return; } this.setValue(pastedData); }; NgOtpInputComponent.decorators = [ { type: Component, args: [{ // tslint:disable-next-line: component-selector selector: 'ng-otp-input', template: "<div class=\"wrapper {{config.containerClass}}\" id=\"c_{{componentKey}}\" *ngIf=\"otpForm?.controls\"\r\n [ngStyle]=\"config.containerStyles\">\r\n <input [pattern]=\"config.allowNumbersOnly ? '\\\\d*' : ''\" [type]=\"inputType\" numberOnly [placeholder]=\"config?.placeholder || ''\"\r\n [disabledNumberOnly]=\"!config.allowNumbersOnly\" [ngStyle]=\"config.inputStyles\" maxlength=\"1\"\r\n class=\"otp-input {{config.inputClass}}\" autocomplete=\"off\" *ngFor=\"let item of otpForm?.controls | keys;let i=index\"\r\n [formControl]=\"otpForm.controls[item]\" id=\"otp_{{i}}_{{componentKey}}\" (keydown)=\"onKeyDown($event)\"\r\n (keyup)=\"onKeyUp($event,i)\">\r\n</div>", styles: [".otp-input{width:50px;height:50px;border-radius:4px;border:1px solid #c5c5c5;text-align:center;font-size:32px}.wrapper .otp-input:not(:last-child){margin-right:8px}@media screen and (max-width:767px){.otp-input{width:40px;font-size:24px;height:40px}}@media screen and (max-width:420px){.otp-input{width:30px;font-size:18px;height:30px}}"] }] } ]; /** @nocollapse */ NgOtpInputComponent.ctorParameters = function () { return [ { type: KeysPipe } ]; }; NgOtpInputComponent.propDecorators = { config: [{ type: Input }], onInputChange: [{ type: Output }] }; return NgOtpInputComponent; }()); export { NgOtpInputComponent }; if (false) { /** @type {?} */ NgOtpInputComponent.prototype.config; /** @type {?} */ NgOtpInputComponent.prototype.onInputChange; /** @type {?} */ NgOtpInputComponent.prototype.otpForm; /** @type {?} */ NgOtpInputComponent.prototype.inputControls; /** @type {?} */ NgOtpInputComponent.prototype.componentKey; /** @type {?} */ NgOtpInputComponent.prototype.inputType; /** * @type {?} * @private */ NgOtpInputComponent.prototype.keysPipe; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ng-otp-input.component.js","sourceRoot":"ng://ng-otp-input/","sources":["lib/components/ng-otp-input/ng-otp-input.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  Component,\r\n  OnInit,\r\n  Input,\r\n  Output,\r\n  EventEmitter,\r\n  AfterViewInit\r\n} from '@angular/core';\r\nimport { FormGroup, FormControl } from '@angular/forms';\r\nimport { KeysPipe } from '../../pipes/keys.pipe';\r\nimport { Config } from '../../models/config';\r\n@Component({\r\n  // tslint:disable-next-line: component-selector\r\n  selector: 'ng-otp-input',\r\n  templateUrl: './ng-otp-input.component.html',\r\n  styleUrls: ['./ng-otp-input.component.scss']\r\n})\r\nexport class NgOtpInputComponent implements OnInit, AfterViewInit {\r\n  @Input() config: Config = { length: 4 };\r\n  // tslint:disable-next-line: no-output-on-prefix\r\n  @Output() onInputChange = new EventEmitter<string>();\r\n  otpForm: FormGroup;\r\n  inputControls: FormControl[] = new Array(this.config.length);\r\n  componentKey =\r\n    Math.random()\r\n      .toString(36)\r\n      .substring(2) + new Date().getTime().toString(36);\r\n  inputType: string;\r\n  constructor(private keysPipe: KeysPipe) {}\r\n\r\n  ngOnInit() {\r\n    this.otpForm = new FormGroup({});\r\n    for (let index = 0; index < this.config.length; index++) {\r\n      this.otpForm.addControl(this.getControlName(index), new FormControl());\r\n    }\r\n    this.inputType = this.getInputType();\r\n    \r\n  }\r\n  ngAfterViewInit(): void {\r\n    if (!this.config.disableAutoFocus) {\r\n      const containerItem = document.getElementById(`c_${this.componentKey}`);\r\n      if (containerItem) {\r\n        containerItem.addEventListener('paste', (evt) => this.handlePaste(evt));\r\n        const ele: any = containerItem.getElementsByClassName('otp-input')[0];\r\n        if (ele && ele.focus) {\r\n          ele.focus();\r\n        }\r\n      }\r\n    }\r\n  }\r\n  private getControlName(idx) {\r\n    return `ctrl_${idx}`;\r\n  }\r\n\r\n  ifLeftArrow(event) {\r\n    return this.ifKeyCode(event, 37);\r\n  }\r\n\r\n\r\n  ifRightArrow(event) {\r\n    return this.ifKeyCode(event, 39);\r\n  }\r\n\r\n  ifBackspaceOrDelete(event) {\r\n    return (\r\n      event.key === 'Backspace' ||\r\n      event.key === 'Delete' ||\r\n      this.ifKeyCode(event, 8) ||\r\n      this.ifKeyCode(event, 46)\r\n    );\r\n  }\r\n\r\n  ifKeyCode(event, targetCode) {\r\n    const key = event.keyCode || event.charCode;\r\n    // tslint:disable-next-line: triple-equals\r\n    return key == targetCode ? true : false;\r\n  }\r\n  onKeyDown($event) {\r\n    var isSpace=this.ifKeyCode($event,32)\r\n    if (isSpace) {// prevent space\r\n    return false;\r\n    }\r\n  }\r\n\r\n  onKeyUp($event, inputIdx) {\r\n    const nextInputId = this.appendKey(`otp_${inputIdx + 1}`);\r\n    const prevInputId = this.appendKey(`otp_${inputIdx - 1}`);\r\n    if (this.ifRightArrow($event)) {\r\n      this.setSelected(nextInputId);\r\n      return;\r\n    }\r\n    if (this.ifLeftArrow($event)) {\r\n      this.setSelected(prevInputId);\r\n      return;\r\n    }\r\n    const isBackspace = this.ifBackspaceOrDelete($event);\r\n    if (isBackspace && !$event.target.value) {\r\n      this.setSelected(prevInputId);\r\n      this.rebuildValue();\r\n      return;\r\n    }\r\n    if (!$event.target.value) {\r\n      return;\r\n    }\r\n    if (this.ifValidEntry($event)) {\r\n      this.setSelected(nextInputId);\r\n    }\r\n    this.rebuildValue();\r\n  }\r\n\r\n  appendKey(id) {\r\n    return `${id}_${this.componentKey}`;\r\n  }\r\n\r\n  setSelected(eleId) {\r\n    this.focusTo(eleId);\r\n    const ele: any = document.getElementById(eleId);\r\n    if (ele && ele.setSelectionRange) {\r\n      setTimeout(() => {\r\n        ele.setSelectionRange(0, 1);\r\n      }, 0);\r\n    }\r\n  }\r\n\r\n  ifValidEntry(event) {\r\n    const inp = String.fromCharCode(event.keyCode);\r\n    const isMobile = /iPhone|iPad|iPod|Android/i.test(navigator.userAgent);\r\n    return (\r\n      isMobile ||\r\n      /[a-zA-Z0-9-_]/.test(inp) ||\r\n      (this.config.allowKeyCodes &&\r\n        this.config.allowKeyCodes.includes(event.keyCode)) ||\r\n      (event.keyCode >= 96 && event.keyCode <= 105)\r\n    );\r\n  }\r\n\r\n  focusTo(eleId) {\r\n    const ele: any = document.getElementById(eleId);\r\n    if (ele) {\r\n      ele.focus();\r\n    }\r\n  }\r\n\r\n  // method to set component value\r\n  setValue(value: any) {\r\n    if (this.config.allowNumbersOnly && isNaN(value)) {\r\n        return;\r\n    }\r\n    this.otpForm.reset();\r\n     if (!value) {\r\n       this.rebuildValue();\r\n       return;\r\n     }\r\n     value = value.toString().replace(/\\s/g, ''); // remove whitespace\r\n     Array.from(value).forEach((c, idx) => {\r\n          if (this.otpForm.get(this.getControlName(idx))) {\r\n            this.otpForm.get(this.getControlName(idx)).setValue(c);\r\n          }\r\n     });\r\n     if (!this.config.disableAutoFocus) {\r\n      const containerItem = document.getElementById(`c_${this.componentKey}`);\r\n      var indexOfElementToFocus = value.length < this.config.length ? value.length : (this.config.length - 1);\r\n      let ele : any = containerItem.getElementsByClassName('otp-input')[indexOfElementToFocus];\r\n      if (ele && ele.focus) {\r\n        ele.focus();\r\n      }\r\n     }\r\n     this.rebuildValue();\r\n  }\r\n\r\n\r\n  rebuildValue() {\r\n    let val = '';\r\n    this.keysPipe.transform(this.otpForm.controls).forEach(k => {\r\n      if (this.otpForm.controls[k].value) {\r\n        val += this.otpForm.controls[k].value;\r\n      }\r\n    });\r\n    this.onInputChange.emit(val);\r\n  }\r\n  getInputType():string{\r\n    return this.config.isPasswordInput \r\n      ? 'password' \r\n      : this.config.allowNumbersOnly \r\n        ? 'tel'\r\n        : 'text';\r\n  }\r\n  handlePaste(e) {\r\n    // Get pasted data via clipboard API\r\n    let clipboardData = e.clipboardData || window['clipboardData'];\r\n    if(clipboardData){\r\n     var pastedData =clipboardData.getData('Text');\r\n    }\r\n    // Stop data actually being pasted into div\r\n    e.stopPropagation();\r\n    e.preventDefault();\r\n    if (!pastedData) {\r\n      return;\r\n    }\r\n    this.setValue(pastedData);\r\n  }\r\n}\r\n"]}