ng-otp-input
Version:
A fully customizable, one-time password input component for the web built with Angular.
384 lines • 28.7 kB
JavaScript
/**
* @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"]}