diff --git a/dist/js/bundle.js b/dist/js/bundle.js index 348f7b0..fea99b2 100644 --- a/dist/js/bundle.js +++ b/dist/js/bundle.js @@ -1,2 +1,2 @@ -!function e(t,n,i){function o(l,s){if(!n[l]){if(!t[l]){var u="function"==typeof require&&require;if(!s&&u)return u(l,!0);if(a)return a(l,!0);var c=new Error("Cannot find module '"+l+"'");throw c.code="MODULE_NOT_FOUND",c}var r=n[l]={exports:{}};t[l][0].call(r.exports,(function(e){return o(t[l][1][e]||e)}),r,r.exports,e,t,n,i)}return n[l].exports}for(var a="function"==typeof require&&require,l=0;l0&&void 0!==arguments[0]?arguments[0]:1,t=this.value,n=this.selectionStartPosition,i=t.slice(0,n);if(""!==i){var o=t.slice(n),a=i.length-e,l=i.slice(0,a)+o,s=n-e;this.value=l,this.selectionPosition=s,this.inputEventHandler()}}},{key:"replaceValue",value:function(e){this.removeValue(),this.insertValue(e),this.inputEventHandler()}},{key:"clear",value:function(){this.value="",this.focus()}},{key:"validation",value:function(){var e=this.selectionStartPosition,t=this.value.replace(/[^0-9+*#]/g,""),n=e+t.length-this.value.length;this.value=t,this.selectionPosition=n}},{key:"inputEventHandler",value:function(){this.validation(),""===this.value?(console.log("value empty"),this.config.onValueEmpty()):(console.log("value non empty"),this.config.onValueNonEmpty())}},{key:"skeleton",get:function(){var e=document.createElement("input");return e.id=this.id,e.className="input-element",e.name="number",e.type="text",e.autofocus=!0,e.inputMode="none",e.autocomplete="off",e.addEventListener("input",this.inputEventHandler.bind(this)),e}},{key:"querySelector",get:function(){return document.getElementById(this.id)}},{key:"id",get:function(){return"input-"+this.config.namespace}},{key:"selectionStartPosition",get:function(){return this.makeSureFocused(),this.querySelector.selectionStart}},{key:"selectionEndPosition",get:function(){return this.makeSureFocused(),this.querySelector.selectionEnd}},{key:"selectionPosition",set:function(e){this.makeSureFocused(),this.querySelector.selectionStart=e,this.querySelector.selectionEnd=e}},{key:"focused",get:function(){return document.activeElement===this.querySelector}},{key:"value",get:function(){return this.querySelector.value},set:function(e){this.querySelector.value=e}}]),e}();n.default=o},{}],3:[function(e,t,n){"use strict";Object.defineProperty(n,"__esModule",{value:!0});n.default=[{namespace:"one",ariaLabel:"One",title:"1"},{namespace:"two",ariaLabel:"Two",title:"2",subtitle:"ABC"},{namespace:"three",ariaLabel:"Three",title:"3",subtitle:"DEF"},{namespace:"four",ariaLabel:"Four",title:"4",subtitle:"GHI"},{namespace:"five",ariaLabel:"Five",title:"5",subtitle:"JKL"},{namespace:"six",ariaLabel:"Six",title:"6",subtitle:"MNO"},{namespace:"seven",ariaLabel:"Seven",title:"7",subtitle:"PQRS"},{namespace:"eight",ariaLabel:"Eight",title:"8",subtitle:"TUV"},{namespace:"nine",ariaLabel:"Nine",title:"9",subtitle:"WXYZ"},{namespace:"star",ariaLabel:"Star",title:"*"},{namespace:"zero",ariaLabel:"Zero",title:"0",subtitle:"+"},{namespace:"hash",ariaLabel:"Hash",title:"#"}]},{}],4:[function(e,t,n){"use strict";var i=function(){function e(e,t){for(var n=0;n0&&void 0!==arguments[0]?arguments[0]:1,t=this.value,n=this.selectionStartPosition,i=t.slice(0,n);if(""!==i){var o=t.slice(n),a=i.length-e,l=i.slice(0,a)+o,s=n-e;this.value=l,this.selectionPosition=s,this.inputEventHandler()}}},{key:"replaceValue",value:function(e){this.removeValue(),this.insertValue(e),this.inputEventHandler()}},{key:"clear",value:function(){this.value="",this.focus()}},{key:"validation",value:function(){var e=this.value,t=this.selectionStartPosition,n=e.replace(/[^0-9+*#]/g,""),i=t+n.length-e.length;this.value=n,this.selectionPosition=i}},{key:"inputEventHandler",value:function(){this.validation(),""===this.value?(console.log("value empty"),this.config.onValueEmpty()):(console.log("value non empty"),this.config.onValueNonEmpty())}},{key:"skeleton",get:function(){var e=document.createElement("input");return e.id=this.id,e.className="input-element",e.name="number",e.type="text",e.autofocus=!0,e.inputMode="none",e.autocomplete="off",e.addEventListener("input",this.inputEventHandler.bind(this)),e}},{key:"querySelector",get:function(){return document.getElementById(this.id)}},{key:"id",get:function(){return"input-"+this.config.namespace}},{key:"selectionStartPosition",get:function(){return this.makeSureFocused(),this.querySelector.selectionStart}},{key:"selectionEndPosition",get:function(){return this.makeSureFocused(),this.querySelector.selectionEnd}},{key:"selectionPosition",set:function(e){this.makeSureFocused(),this.querySelector.selectionStart=e,this.querySelector.selectionEnd=e}},{key:"focused",get:function(){return document.activeElement===this.querySelector}},{key:"value",get:function(){return this.querySelector.value},set:function(e){this.querySelector.value=e}}]),e}();n.default=o},{}],3:[function(e,t,n){"use strict";Object.defineProperty(n,"__esModule",{value:!0});n.default=[{namespace:"one",ariaLabel:"One",title:"1"},{namespace:"two",ariaLabel:"Two",title:"2",subtitle:"ABC"},{namespace:"three",ariaLabel:"Three",title:"3",subtitle:"DEF"},{namespace:"four",ariaLabel:"Four",title:"4",subtitle:"GHI"},{namespace:"five",ariaLabel:"Five",title:"5",subtitle:"JKL"},{namespace:"six",ariaLabel:"Six",title:"6",subtitle:"MNO"},{namespace:"seven",ariaLabel:"Seven",title:"7",subtitle:"PQRS"},{namespace:"eight",ariaLabel:"Eight",title:"8",subtitle:"TUV"},{namespace:"nine",ariaLabel:"Nine",title:"9",subtitle:"WXYZ"},{namespace:"star",ariaLabel:"Star",title:"*"},{namespace:"zero",ariaLabel:"Zero",title:"0",subtitle:"+"},{namespace:"hash",ariaLabel:"Hash",title:"#"}]},{}],4:[function(e,t,n){"use strict";var i=function(){function e(e,t){for(var n=0;n {\r\n this.config.onLongPress!(this.config.subtitle!);\r\n },\r\n onPressStart: () => {\r\n this.config.onClick(this.config.title);\r\n },\r\n onLongPressCancel: this.config.onLongPressCancel,\r\n });\r\n }\r\n\r\n /**\r\n *\r\n * Apply click event listener on button element\r\n *\r\n * @param button\r\n *\r\n */\r\n addClickEventListener(button: HTMLButtonElement) {\r\n button.addEventListener('click', () => {\r\n this.config.onClick(this.config.title);\r\n });\r\n }\r\n\r\n /**\r\n *\r\n * If onLongPress callback is exist in config apply long press event\r\n * Else apply click event listener on button\r\n *\r\n * @param button\r\n *\r\n */\r\n configureButtonEvents(button: HTMLButtonElement) {\r\n if (this.config.onLongPress !== undefined) {\r\n // apply long press event\r\n this.applyLongPressEvent(button);\r\n } else {\r\n // add click event listener\r\n this.addClickEventListener(button);\r\n }\r\n }\r\n\r\n /**\r\n *\r\n * Dialpad button tile element\r\n *\r\n */\r\n get titleElement(): HTMLHeadingElement {\r\n const title = document.createElement('h1');\r\n title.classList.add('dialpad-btn__title');\r\n title.innerText = this.config.title;\r\n return title;\r\n }\r\n\r\n /**\r\n *\r\n * Dialpad button subtile\r\n *\r\n */\r\n get subtitleElement(): HTMLParagraphElement {\r\n const subtitle = document.createElement('p');\r\n subtitle.classList.add('dialpad-btn__subtitle');\r\n subtitle.innerText = this.config.subtitle ?? '';\r\n return subtitle;\r\n }\r\n\r\n /**\r\n *\r\n * Unique id for dialpad button\r\n *\r\n */\r\n get id() {\r\n return `dialpad-btn-${this.config.namespace}`;\r\n }\r\n\r\n /**\r\n *\r\n * Dialpad button query selector for dom manipulation\r\n *\r\n */\r\n get querySelector(): HTMLButtonElement {\r\n return document.getElementById(this.id)! as HTMLButtonElement;\r\n }\r\n\r\n /**\r\n *\r\n * Appends dialpad button to a specified parent element.\r\n *\r\n * @param parentElement\r\n *\r\n */\r\n build(parentElement: HTMLElement) {\r\n parentElement.appendChild(this.skeleton);\r\n }\r\n}\r\n","import { InputElementConfig } from './types';\r\n\r\n/**\r\n *\r\n * Input Element\r\n *\r\n */\r\nexport default class InputElement {\r\n // input element config\r\n private config: InputElementConfig;\r\n\r\n /**\r\n *\r\n * construct InputElement instance\r\n *\r\n */\r\n constructor(config: InputElementConfig) {\r\n this.config = config;\r\n }\r\n\r\n /**\r\n *\r\n * input element skeleton\r\n *\r\n */\r\n private get skeleton(): HTMLInputElement {\r\n const input = document.createElement('input');\r\n input.id = this.id;\r\n input.className = 'input-element';\r\n input.name = 'number';\r\n input.type = 'text';\r\n input.autofocus = true;\r\n input.inputMode = 'none';\r\n input.autocomplete = 'off';\r\n\r\n // add input event listener\r\n input.addEventListener('input', this.inputEventHandler.bind(this));\r\n\r\n return input;\r\n }\r\n\r\n /**\r\n *\r\n * Input element query selector for dom manipulation\r\n *\r\n */\r\n get querySelector(): HTMLInputElement {\r\n return document.getElementById(this.id) as HTMLInputElement;\r\n }\r\n\r\n /**\r\n *\r\n * Unique id for input element\r\n *\r\n * @returns {string}\r\n *\r\n */\r\n get id(): string {\r\n return `input-${this.config.namespace}`;\r\n }\r\n\r\n /**\r\n *\r\n * Appends input element to a specified parent element.\r\n *\r\n * @param parentElement\r\n *\r\n */\r\n build(parentElement: HTMLElement) {\r\n parentElement.appendChild(this.skeleton);\r\n }\r\n\r\n /**\r\n *\r\n * Make sure input element is focused\r\n *\r\n */\r\n makeSureFocused() {\r\n if (!this.focused) {\r\n this.focus();\r\n }\r\n }\r\n\r\n /**\r\n *\r\n * Get input element selection start position\r\n *\r\n */\r\n get selectionStartPosition(): number {\r\n // make sure the input is focused\r\n this.makeSureFocused();\r\n return this.querySelector.selectionStart!;\r\n }\r\n\r\n /**\r\n *\r\n * Get input element caret end position\r\n *\r\n */\r\n get selectionEndPosition(): number {\r\n // make sure the input is focused\r\n this.makeSureFocused();\r\n return this.querySelector.selectionEnd!;\r\n }\r\n\r\n /**\r\n *\r\n * Set input element selection start and end positions\r\n *\r\n * @param position - desired caret position index value\r\n *\r\n */\r\n set selectionPosition(position: number) {\r\n // make sure the input is focused\r\n this.makeSureFocused();\r\n\r\n // set selection\r\n this.querySelector.selectionStart = position;\r\n this.querySelector.selectionEnd = position;\r\n }\r\n\r\n /**\r\n *\r\n * To check whether the input element is focused or not\r\n *\r\n */\r\n get focused(): boolean {\r\n return document.activeElement === this.querySelector;\r\n }\r\n\r\n /**\r\n *\r\n * To get input element value\r\n *\r\n */\r\n get value(): string {\r\n return this.querySelector.value;\r\n }\r\n\r\n /**\r\n *\r\n * To set input element value\r\n *\r\n * @param value - The value to set\r\n *\r\n */\r\n set value(value: string) {\r\n this.querySelector.value = value;\r\n }\r\n\r\n /**\r\n *\r\n * Focus input element\r\n *\r\n */\r\n focus() {\r\n this.querySelector.focus();\r\n }\r\n\r\n /**\r\n *\r\n * Insert a value at the caret position\r\n *\r\n * @param value - The value to be inserted\r\n *\r\n */\r\n // eslint-disable-next-line class-methods-use-this\r\n insertValue(value: string) {\r\n // capture current state\r\n const currentValue = this.value;\r\n const { selectionStartPosition, selectionEndPosition } = this;\r\n\r\n // prepare updated state\r\n let updatedValue;\r\n\r\n if (selectionStartPosition !== selectionEndPosition) {\r\n // if input selection start value is not equal to selection end position\r\n // it means input value is selected, in this case remove selected value\r\n // and insert new value at the selection start (caret) position\r\n updatedValue =\r\n currentValue.slice(0, selectionStartPosition) +\r\n value +\r\n currentValue.slice(selectionEndPosition);\r\n } else {\r\n // else input selection start value is same as selection end value\r\n // insert new value at selection start (caret) position\r\n updatedValue =\r\n currentValue.slice(0, selectionStartPosition) +\r\n value +\r\n currentValue.slice(selectionStartPosition);\r\n }\r\n const updatedCaretPosition = selectionStartPosition + value.length;\r\n\r\n // update state\r\n this.value = updatedValue;\r\n this.selectionPosition = updatedCaretPosition;\r\n }\r\n\r\n /**\r\n *\r\n * Remove characters from the value starting at the caret position\r\n *\r\n * @param count - The number of characters, default value is 1\r\n *\r\n */\r\n removeValue(count: number = 1) {\r\n // capture current state\r\n const currentValue = this.value;\r\n const currentCaretPosition = this.selectionStartPosition;\r\n\r\n // prepare updated state\r\n const beforeCaretValue = currentValue.slice(0, currentCaretPosition);\r\n\r\n // if before caret value is empty do nothing\r\n if (beforeCaretValue !== '') {\r\n const afterCaretValue = currentValue.slice(currentCaretPosition);\r\n const endIndex = beforeCaretValue.length - count;\r\n const updatedValue =\r\n beforeCaretValue.slice(0, endIndex) + afterCaretValue;\r\n const updatedCaretPosition = currentCaretPosition - count;\r\n\r\n // update state\r\n this.value = updatedValue;\r\n this.selectionPosition = updatedCaretPosition;\r\n\r\n // check empty or not\r\n this.inputEventHandler();\r\n }\r\n }\r\n\r\n /**\r\n *\r\n * @param value - The value to be replaced current last value\r\n *\r\n */\r\n replaceValue(value: string) {\r\n this.removeValue();\r\n this.insertValue(value);\r\n this.inputEventHandler();\r\n }\r\n\r\n /**\r\n *\r\n * Clear the value of input element\r\n *\r\n */\r\n clear() {\r\n this.value = '';\r\n this.focus();\r\n }\r\n\r\n /**\r\n *\r\n * Remove unwanted symbols and allow only digits, +, *, and #\r\n *\r\n */\r\n validation() {\r\n // prepare updated state\r\n const caretPositionBeforeRemoveUnwantedChars = this.selectionStartPosition;\r\n const updatedValueAfterRemoveUnwantedChars = this.value.replace(\r\n /[^0-9+*#]/g,\r\n ''\r\n );\r\n const updatedCaretPosition =\r\n caretPositionBeforeRemoveUnwantedChars +\r\n updatedValueAfterRemoveUnwantedChars.length -\r\n this.value.length;\r\n\r\n // update state\r\n this.value = updatedValueAfterRemoveUnwantedChars;\r\n this.selectionPosition = updatedCaretPosition;\r\n }\r\n\r\n /**\r\n *\r\n * `input` event handler\r\n *\r\n */\r\n private inputEventHandler() {\r\n // Remove unwanted symbols and allow only digits, +, *, and #\r\n this.validation();\r\n\r\n // check whether the value is empty or not, and act accordingly.\r\n if (this.value === '') {\r\n // eslint-disable-next-line no-console\r\n console.log('value empty');\r\n this.config.onValueEmpty();\r\n } else {\r\n // eslint-disable-next-line no-console\r\n console.log('value non empty');\r\n this.config.onValueNonEmpty();\r\n }\r\n }\r\n}\r\n","import { KeypadButtonData } from './types';\n\n/**\n *\n * Keypad buttons data\n *\n */\nconst KEYPAD_BUTTONS_DATA: Array = [\n {\n namespace: 'one',\n ariaLabel: 'One',\n title: '1',\n },\n {\n namespace: 'two',\n ariaLabel: 'Two',\n title: '2',\n subtitle: 'ABC',\n },\n {\n namespace: 'three',\n ariaLabel: 'Three',\n title: '3',\n subtitle: 'DEF',\n },\n {\n namespace: 'four',\n ariaLabel: 'Four',\n title: '4',\n subtitle: 'GHI',\n },\n {\n namespace: 'five',\n ariaLabel: 'Five',\n title: '5',\n subtitle: 'JKL',\n },\n {\n namespace: 'six',\n ariaLabel: 'Six',\n title: '6',\n subtitle: 'MNO',\n },\n {\n namespace: 'seven',\n ariaLabel: 'Seven',\n title: '7',\n subtitle: 'PQRS',\n },\n {\n namespace: 'eight',\n ariaLabel: 'Eight',\n title: '8',\n subtitle: 'TUV',\n },\n {\n namespace: 'nine',\n ariaLabel: 'Nine',\n title: '9',\n subtitle: 'WXYZ',\n },\n {\n namespace: 'star',\n ariaLabel: 'Star',\n title: '*',\n },\n {\n namespace: 'zero',\n ariaLabel: 'Zero',\n title: '0',\n subtitle: '+',\n },\n {\n namespace: 'hash',\n ariaLabel: 'Hash',\n title: '#',\n },\n];\n\nexport default KEYPAD_BUTTONS_DATA;\n","import DialpadButton from '../components/buttons/buttons';\nimport LongPressEvent from '../utilities/longPress';\nimport KEYPAD_BUTTONS_DATA from './data';\nimport { KeypadButtonData, KeypadConfig } from './types';\n\n/**\n *\n * Keypad\n *\n */\nexport default class Keypad {\n // keypad config\n private config: KeypadConfig;\n\n /**\n *\n * construct Keypad instance\n *\n */\n constructor(config: KeypadConfig) {\n this.config = config;\n }\n\n /**\n *\n * Keypad skeleton\n *\n */\n private get skeleton(): HTMLDivElement {\n const keypad = document.createElement('div');\n keypad.id = this.id;\n keypad.classList.add('keypad');\n\n // append digits buttons\n KEYPAD_BUTTONS_DATA.forEach((config: KeypadButtonData) => {\n // create button instance\n const btn = new DialpadButton({\n namespace: config.namespace,\n ariaLabel: config.ariaLabel,\n title: config.title,\n subtitle: config.subtitle,\n onClick: this.config.onKeypadBtnClick,\n onLongPress:\n config.title === '0' ? this.config.onZeroBtnLongPress : undefined,\n onLongPressCancel:\n config.title === '0'\n ? this.config.onZeroBtnLongPressCancel\n : undefined,\n });\n\n // append button\n btn.build(keypad);\n });\n\n // append dummy element\n keypad.appendChild(document.createElement('span'));\n\n // append call button\n keypad.appendChild(this.callButton);\n\n // append backspace button\n keypad.appendChild(this.backspaceButton);\n\n return keypad;\n }\n\n // eslint-disable-next-line class-methods-use-this\n getMaterialIcon(iconName: string) {\n const iconElement = document.createElement('span');\n iconElement.className = 'material-symbols-outlined';\n iconElement.innerText = iconName;\n return iconElement;\n }\n\n /**\n *\n * Keypad call button\n *\n */\n private get callButton(): HTMLElement {\n const callBtn = document.createElement('button');\n callBtn.classList.add('keypad__call-btn');\n callBtn.setAttribute('aria-label', 'call button');\n\n // append call icon\n callBtn.appendChild(this.getMaterialIcon('call'));\n\n // add click event listener\n callBtn.addEventListener('click', this.config.onCallBtnClick);\n\n return callBtn;\n }\n\n /**\n *\n * Keypad backspace button\n *\n */\n private get backspaceButton(): HTMLElement {\n const backspaceBtn = document.createElement('button');\n backspaceBtn.classList.add('keypad__backspace-btn');\n backspaceBtn.setAttribute('aria-label', 'Backspace button');\n backspaceBtn.disabled = true;\n\n // append backspace icon\n backspaceBtn.appendChild(this.getMaterialIcon('backspace'));\n\n // add click event listener\n backspaceBtn.addEventListener('click', this.config.onBackspaceBtnClick);\n\n // apply long press event\n LongPressEvent.apply({\n target: backspaceBtn,\n onLongPressCallback: this.config.onBackspaceBtnLongPress,\n });\n\n return backspaceBtn;\n }\n\n /**\n *\n * To get backspace button element for dom manipulations\n *\n */\n get backspaceButtonElement(): HTMLButtonElement {\n return this.querySelector.querySelector(\n '.keypad__backspace-btn'\n ) as HTMLButtonElement;\n }\n\n /**\n *\n * Unique id for keypad\n *\n */\n get id() {\n return `keypad-${this.config.namespace}`;\n }\n\n /**\n *\n * Keypad query selector for dom manipulation\n *\n */\n get querySelector(): HTMLDivElement {\n return document.getElementById(this.id)! as HTMLDivElement;\n }\n\n /**\n *\n * Appends keypad to a specified parent element.\n *\n * @param parentElement\n *\n */\n build(parentElement: HTMLElement) {\n parentElement.appendChild(this.skeleton);\n }\n\n /**\n *\n * Enable backspace button\n *\n */\n enableBackspaceButton() {\n this.backspaceButtonElement.disabled = false;\n }\n\n /**\n *\n * Disable backspace button\n *\n */\n disableBackspaceButton() {\n this.backspaceButtonElement.disabled = true;\n }\n}\n","import Dialpad from './pages/dialpad';\n\nwindow.onload = () => {\n // create dialpad instance\n const dialpad = new Dialpad({ namespace: 'demo' });\n\n // build dialpad\n dialpad.build(document.body);\n};\n","import InputElement from '../components/forms/inputs';\r\nimport Keypad from '../layout/keypad';\r\nimport DialpadConfig from './types';\r\n\r\n/**\r\n *\r\n * Dialpad Page\r\n *\r\n */\r\nexport default class Dialpad {\r\n // dialpad configuration\r\n private config: DialpadConfig;\r\n\r\n // input field instance\r\n private inputField!: InputElement;\r\n\r\n // keypad instance\r\n private keypad!: Keypad;\r\n\r\n // recent call number\r\n private recentCallOnNumber?: string;\r\n\r\n /**\r\n *\r\n * construct Dialpad instance\r\n *\r\n */\r\n constructor(config: DialpadConfig) {\r\n this.config = config;\r\n }\r\n\r\n /**\r\n *\r\n * page skeleton\r\n *\r\n * @returns {HTMLDivElement}\r\n *\r\n */\r\n private skeleton(): HTMLElement {\r\n const dialpad = document.createElement('section');\r\n dialpad.id = this.id;\r\n dialpad.className = 'dialpad';\r\n\r\n // build input field\r\n this.inputFieldLayout.build(dialpad);\r\n\r\n // build keypad layout\r\n this.keypadLayout.build(dialpad);\r\n\r\n // append copyright text\r\n dialpad.appendChild(this.copyrightText);\r\n\r\n return dialpad;\r\n }\r\n\r\n /**\r\n *\r\n * To get Input Element\r\n *\r\n */\r\n private get inputFieldLayout(): InputElement {\r\n this.inputField = new InputElement({\r\n namespace: this.config.namespace,\r\n onValueEmpty: () => {\r\n // eslint-disable-next-line no-console\r\n console.log('disable backspace button');\r\n this.keypad.disableBackspaceButton();\r\n },\r\n onValueNonEmpty: () => {\r\n // eslint-disable-next-line no-console\r\n console.log('enable backspace button');\r\n this.keypad.enableBackspaceButton();\r\n },\r\n });\r\n return this.inputField;\r\n }\r\n\r\n /**\r\n *\r\n * To get Keypad Layout\r\n *\r\n */\r\n private get keypadLayout() {\r\n this.keypad = new Keypad({\r\n namespace: this.config.namespace,\r\n onKeypadBtnClick: (value: string) => {\r\n // eslint-disable-next-line no-console\r\n console.log('clicked on button', value);\r\n\r\n // insert value\r\n this.inputField.insertValue(value);\r\n this.keypad.enableBackspaceButton();\r\n },\r\n onZeroBtnLongPress: (value: string) => {\r\n // eslint-disable-next-line no-console\r\n console.log('long pressed on zero button');\r\n\r\n // insert zero button subtitle value `+`\r\n this.inputField.replaceValue(value);\r\n },\r\n onZeroBtnLongPressCancel: () => {\r\n // focus input field\r\n this.inputField.focus();\r\n },\r\n onCallBtnClick: () => {\r\n // eslint-disable-next-line no-console\r\n console.log('clicked on call button');\r\n\r\n // check whether try to call on recent number\r\n if (\r\n this.inputField.value === '' &&\r\n this.recentCallOnNumber !== undefined\r\n ) {\r\n // eslint-disable-next-line no-console\r\n console.log('fill last call on number in input field value');\r\n this.inputField.value = this.recentCallOnNumber;\r\n this.inputField.focus();\r\n this.keypad.enableBackspaceButton();\r\n }\r\n // if value is not empty (or add check for valid phone number)\r\n else if (this.inputField.value !== '') {\r\n // update last call on number\r\n this.recentCallOnNumber = this.inputField.value;\r\n\r\n // eslint-disable-next-line no-console\r\n console.log('placing call on ', this.inputField.value);\r\n }\r\n },\r\n onBackspaceBtnClick: () => {\r\n // eslint-disable-next-line no-console\r\n console.log('clicked on clear button');\r\n this.inputField.removeValue();\r\n },\r\n onBackspaceBtnLongPress: () => {\r\n // eslint-disable-next-line no-console\r\n console.log('long press on backspace button');\r\n\r\n // clear input value and hide backspace button\r\n this.inputField.value = '';\r\n this.inputField.focus();\r\n this.keypad.disableBackspaceButton();\r\n },\r\n });\r\n\r\n return this.keypad;\r\n }\r\n\r\n /**\r\n *\r\n * append copyright text element\r\n *\r\n */\r\n // eslint-disable-next-line class-methods-use-this\r\n get copyrightText(): HTMLParagraphElement {\r\n // create copyright text\r\n const copyrightTextElement = document.createElement('p');\r\n copyrightTextElement.className = 'copyright-text';\r\n copyrightTextElement.innerText = `Made by • Satyam Seth Ⓒ ${new Date().getFullYear()}`;\r\n\r\n return copyrightTextElement;\r\n }\r\n\r\n /**\r\n *\r\n * Unique id for page\r\n *\r\n * @returns {string}\r\n *\r\n */\r\n get id(): string {\r\n return `dialpad-${this.config.namespace}`;\r\n }\r\n\r\n /**\r\n *\r\n * querySelect for app\r\n *\r\n * @returns {HTMLElement}\r\n *\r\n */\r\n get element(): HTMLElement {\r\n return document.getElementById(this.id)!;\r\n }\r\n\r\n /**\r\n *\r\n * Remove dialpad skeleton from dom\r\n *\r\n */\r\n destroy(): void {\r\n this.element.remove();\r\n }\r\n\r\n /**\r\n *\r\n * Append dialpad skeleton into parent element\r\n *\r\n * @param parentElement {HTMLElement}\r\n */\r\n build(parentElement: HTMLElement) {\r\n parentElement.appendChild(this.skeleton());\r\n }\r\n}\r\n","import LongPressEventConfig from './types';\n\n/**\n *\n * LongPress event\n *\n */\nexport default class LongPressEvent {\n // target html element\n private config: LongPressEventConfig;\n\n // state to detect press and hold\n private isHeld: boolean = false;\n\n // setTimeout Id\n private activeHoldTimeoutId: number | null = null;\n\n // long press timeout in milliseconds\n private longPressTimeout = 500;\n\n /**\n *\n * construct LongPressEvent instance and add event listener\n *\n */\n constructor(config: LongPressEventConfig) {\n this.config = config;\n\n // start timer\n this.config.target.addEventListener(\n 'mousedown',\n this.onHoldStart.bind(this)\n );\n this.config.target.addEventListener(\n 'touchstart',\n this.onHoldStart.bind(this)\n );\n\n // stop timer\n this.config.target.addEventListener('mouseup', this.onHoldEnd.bind(this));\n this.config.target.addEventListener('mouseout', this.onHoldEnd.bind(this));\n this.config.target.addEventListener(\n 'mouseleave',\n this.onHoldEnd.bind(this)\n );\n this.config.target.addEventListener('touchend', this.onHoldEnd.bind(this));\n this.config.target.addEventListener(\n 'touchcancel',\n this.onHoldEnd.bind(this)\n );\n }\n\n /**\n *\n * start set timeout timer on long press event for callback execution\n *\n */\n private onHoldStart() {\n // eslint-disable-next-line no-console\n console.log('Start Pressing');\n this.isHeld = true;\n\n // call on press start handler\n if (this.config.onPressStart) {\n this.config.onPressStart();\n }\n\n this.activeHoldTimeoutId = window.setTimeout(() => {\n if (this.isHeld === true) {\n // eslint-disable-next-line no-console\n console.log('long press detected');\n this.config.onLongPressCallback();\n }\n }, this.longPressTimeout);\n }\n\n /**\n *\n * clear set timeout timer on long press event\n *\n */\n private onHoldEnd() {\n // eslint-disable-next-line no-console\n console.log('Stop Pressing');\n this.isHeld = false;\n\n // call on press start handler\n if (this.config.onLongPressCancel) {\n this.config.onLongPressCancel();\n }\n\n if (this.activeHoldTimeoutId !== null) {\n window.clearTimeout(this.activeHoldTimeoutId);\n this.activeHoldTimeoutId = null;\n }\n }\n\n /**\n *\n * Apply long press event without using `new` keyword\n *\n */\n static apply(config: LongPressEventConfig) {\n // eslint-disable-next-line no-new\n new LongPressEvent(config);\n }\n}\n\n// Example how to use\n// window.onload = () => {\n// // const demoButton = document.querySelector('button');\n// // Using new keyword\n// // new ClickAndHold(demoButton, () => {\n// // console.log('Long Press');\n// // alert('Long Press Detected');\n// // });\n// // // Without using new keyword\n// // ClickAndHold.apply(demoButton, () => {\n// // console.log('Long Press');\n// // alert('Long Press Detected');\n// // });\n// };\n"]} \ No newline at end of file +{"version":3,"sources":["node_modules/browser-pack/_prelude.js","bundle.js","src/ts/components/buttons/buttons.ts","src/ts/components/forms/inputs.ts","src/ts/layout/data.ts","src/ts/layout/keypad.ts","src/ts/main.ts","src/ts/pages/dialpad.ts","src/ts/utilities/longPress.ts"],"names":["r","e","n","t","o","i","f","c","require","u","a","Error","code","p","exports","call","length","module","_createClass","defineProperties","target","props","descriptor","enumerable","configurable","writable","Object","defineProperty","key","Constructor","protoProps","staticProps","prototype","__importDefault","mod","__esModule","default","value","longPress_1","DialpadButton","config","instance","TypeError","_classCallCheck","this","undefined","subtitle","onLongPress","button","_this","apply","onLongPressCallback","onPressStart","onClick","title","onLongPressCancel","_this2","addEventListener","applyLongPressEvent","addClickEventListener","parentElement","appendChild","skeleton","get","document","createElement","id","classList","add","setAttribute","ariaLabel","titleElement","subtitleElement","configureButtonEvents","innerText","_a","namespace","getElementById","InputElement","focused","focus","querySelector","currentValue","selectionStartPosition","selectionEndPosition","updatedValue","slice","updatedCaretPosition","selectionPosition","count","arguments","currentCaretPosition","beforeCaretValue","afterCaretValue","endIndex","inputEventHandler","removeValue","insertValue","caretPositionBeforeRemoveUnwantedChars","updatedValueAfterRemoveUnwantedChars","replace","validation","console","log","onValueEmpty","onValueNonEmpty","input","className","name","type","autofocus","inputMode","autocomplete","bind","makeSureFocused","selectionStart","selectionEnd","set","position","activeElement","buttons_1","data_1","Keypad","iconName","iconElement","backspaceButtonElement","disabled","keypad","forEach","onKeypadBtnClick","onZeroBtnLongPress","onZeroBtnLongPressCancel","build","callButton","backspaceButton","callBtn","getMaterialIcon","onCallBtnClick","backspaceBtn","onBackspaceBtnClick","onBackspaceBtnLongPress","dialpad_1","window","onload","body","inputs_1","keypad_1","Dialpad","dialpad","inputFieldLayout","keypadLayout","copyrightText","element","remove","inputField","disableBackspaceButton","enableBackspaceButton","replaceValue","recentCallOnNumber","copyrightTextElement","Date","getFullYear","LongPressEvent","isHeld","activeHoldTimeoutId","longPressTimeout","onHoldStart","onHoldEnd","setTimeout","clearTimeout"],"mappings":"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","file":"../bundle.js","sourcesContent":["(function(){function r(e,n,t){function o(i,f){if(!n[i]){if(!e[i]){var c=\"function\"==typeof require&&require;if(!f&&c)return c(i,!0);if(u)return u(i,!0);var a=new Error(\"Cannot find module '\"+i+\"'\");throw a.code=\"MODULE_NOT_FOUND\",a}var p=n[i]={exports:{}};e[i][0].call(p.exports,function(r){var n=e[i][1][r];return o(n||r)},p,p.exports,r,e,n,t)}return n[i].exports}for(var u=\"function\"==typeof require&&require,i=0;i {\n this.config.onLongPress!(this.config.subtitle!);\n },\n onPressStart: () => {\n this.config.onClick(this.config.title);\n },\n onLongPressCancel: this.config.onLongPressCancel,\n });\n }\n\n /**\n *\n * Apply click event listener on button element\n *\n * @param button\n *\n */\n addClickEventListener(button: HTMLButtonElement) {\n button.addEventListener('click', () => {\n this.config.onClick(this.config.title);\n });\n }\n\n /**\n *\n * If onLongPress callback is exist in config apply long press event\n * Else apply click event listener on button\n *\n * @param button\n *\n */\n configureButtonEvents(button: HTMLButtonElement) {\n if (this.config.onLongPress !== undefined) {\n // apply long press event\n this.applyLongPressEvent(button);\n } else {\n // add click event listener\n this.addClickEventListener(button);\n }\n }\n\n /**\n *\n * Dialpad button tile element\n *\n */\n get titleElement(): HTMLHeadingElement {\n const title = document.createElement('h1');\n title.classList.add('dialpad-btn__title');\n title.innerText = this.config.title;\n return title;\n }\n\n /**\n *\n * Dialpad button subtile\n *\n */\n get subtitleElement(): HTMLParagraphElement {\n const subtitle = document.createElement('p');\n subtitle.classList.add('dialpad-btn__subtitle');\n subtitle.innerText = this.config.subtitle ?? '';\n return subtitle;\n }\n\n /**\n *\n * Unique id for dialpad button\n *\n */\n get id() {\n return `dialpad-btn-${this.config.namespace}`;\n }\n\n /**\n *\n * Dialpad button query selector for dom manipulation\n *\n */\n get querySelector(): HTMLButtonElement {\n return document.getElementById(this.id)! as HTMLButtonElement;\n }\n\n /**\n *\n * Appends dialpad button to a specified parent element.\n *\n * @param parentElement\n *\n */\n build(parentElement: HTMLElement) {\n parentElement.appendChild(this.skeleton);\n }\n}\n","import { InputElementConfig } from './types';\n\n/**\n *\n * Input Element\n *\n */\nexport default class InputElement {\n // input element config\n private config: InputElementConfig;\n\n /**\n *\n * construct InputElement instance\n *\n */\n constructor(config: InputElementConfig) {\n this.config = config;\n }\n\n /**\n *\n * input element skeleton\n *\n */\n get skeleton(): HTMLInputElement {\n const input = document.createElement('input');\n input.id = this.id;\n input.className = 'input-element';\n input.name = 'number';\n input.type = 'text';\n input.autofocus = true;\n input.inputMode = 'none';\n input.autocomplete = 'off';\n\n // add input event listener\n input.addEventListener('input', this.inputEventHandler.bind(this));\n\n return input;\n }\n\n /**\n *\n * Input element query selector for dom manipulation\n *\n */\n get querySelector(): HTMLInputElement {\n return document.getElementById(this.id) as HTMLInputElement;\n }\n\n /**\n *\n * Unique id for input element\n *\n * @returns {string}\n *\n */\n get id(): string {\n return `input-${this.config.namespace}`;\n }\n\n /**\n *\n * Appends input element to a specified parent element.\n *\n * @param parentElement\n *\n */\n build(parentElement: HTMLElement) {\n parentElement.appendChild(this.skeleton);\n }\n\n /**\n *\n * Make sure input element is focused\n *\n */\n makeSureFocused() {\n if (!this.focused) {\n this.focus();\n }\n }\n\n /**\n *\n * Get input element selection start position\n *\n */\n get selectionStartPosition(): number {\n // make sure the input is focused\n this.makeSureFocused();\n return this.querySelector.selectionStart!;\n }\n\n /**\n *\n * Get input element caret end position\n *\n */\n get selectionEndPosition(): number {\n // make sure the input is focused\n this.makeSureFocused();\n return this.querySelector.selectionEnd!;\n }\n\n /**\n *\n * Set input element selection start and end positions\n *\n * @param position - desired caret position index value\n *\n */\n set selectionPosition(position: number) {\n // make sure the input is focused\n this.makeSureFocused();\n\n // set selection\n this.querySelector.selectionStart = position;\n this.querySelector.selectionEnd = position;\n }\n\n /**\n *\n * Focus input element\n *\n */\n focus() {\n this.querySelector.focus();\n }\n\n /**\n *\n * To check whether the input element is focused or not\n *\n */\n get focused(): boolean {\n return document.activeElement === this.querySelector;\n }\n\n /**\n *\n * To get input element value\n *\n */\n get value(): string {\n return this.querySelector.value;\n }\n\n /**\n *\n * To set input element value\n *\n * @param value - The value to set\n *\n */\n set value(value: string) {\n this.querySelector.value = value;\n }\n\n /**\n *\n * Insert a value at the caret position\n *\n * @param value - The value to be inserted\n *\n */\n // eslint-disable-next-line class-methods-use-this\n insertValue(value: string) {\n // capture current state\n const currentValue = this.value;\n const { selectionStartPosition, selectionEndPosition } = this;\n\n // prepare updated state\n let updatedValue;\n\n if (selectionStartPosition !== selectionEndPosition) {\n // if input selection start value is not equal to selection end position\n // it means input value is selected, in this case remove selected value\n // and insert new value at the selection start (caret) position\n updatedValue =\n currentValue.slice(0, selectionStartPosition) +\n value +\n currentValue.slice(selectionEndPosition);\n } else {\n // else input selection start value is same as selection end value\n // insert new value at selection start (caret) position\n updatedValue =\n currentValue.slice(0, selectionStartPosition) +\n value +\n currentValue.slice(selectionStartPosition);\n }\n const updatedCaretPosition = selectionStartPosition + value.length;\n\n // update state\n this.value = updatedValue;\n this.selectionPosition = updatedCaretPosition;\n }\n\n /**\n *\n * Remove characters from the value starting at the caret position\n *\n * @param count - The number of characters, default value is 1\n *\n */\n removeValue(count: number = 1) {\n // capture current state\n const currentValue = this.value;\n const currentCaretPosition = this.selectionStartPosition;\n\n // prepare updated state\n const beforeCaretValue = currentValue.slice(0, currentCaretPosition);\n\n // if before caret value is empty do nothing\n if (beforeCaretValue !== '') {\n const afterCaretValue = currentValue.slice(currentCaretPosition);\n const endIndex = beforeCaretValue.length - count;\n const updatedValue =\n beforeCaretValue.slice(0, endIndex) + afterCaretValue;\n const updatedCaretPosition = currentCaretPosition - count;\n\n // update state\n this.value = updatedValue;\n this.selectionPosition = updatedCaretPosition;\n\n // check empty or not\n this.inputEventHandler();\n }\n }\n\n /**\n *\n * @param value - The value to be replaced current last value\n *\n */\n replaceValue(value: string) {\n this.removeValue();\n this.insertValue(value);\n this.inputEventHandler();\n }\n\n /**\n *\n * Clear the value of input element\n *\n */\n clear() {\n this.value = '';\n this.focus();\n }\n\n /**\n *\n * Remove unwanted symbols and allow only digits, +, *, and #\n *\n */\n validation() {\n // prepare updated state\n const { value } = this;\n const caretPositionBeforeRemoveUnwantedChars = this.selectionStartPosition;\n const updatedValueAfterRemoveUnwantedChars = value.replace(\n /[^0-9+*#]/g,\n ''\n );\n const updatedCaretPosition =\n caretPositionBeforeRemoveUnwantedChars +\n updatedValueAfterRemoveUnwantedChars.length -\n value.length;\n\n // update state\n this.value = updatedValueAfterRemoveUnwantedChars;\n this.selectionPosition = updatedCaretPosition;\n }\n\n /**\n *\n * `input` event handler\n *\n */\n inputEventHandler() {\n // Remove unwanted symbols and allow only digits, +, *, and #\n this.validation();\n\n // check whether the value is empty or not, and act accordingly.\n if (this.value === '') {\n // eslint-disable-next-line no-console\n console.log('value empty');\n this.config.onValueEmpty();\n } else {\n // eslint-disable-next-line no-console\n console.log('value non empty');\n this.config.onValueNonEmpty();\n }\n }\n}\n","import { KeypadButtonData } from './types';\n\n/**\n *\n * Keypad buttons data\n *\n */\nconst KEYPAD_BUTTONS_DATA: Array = [\n {\n namespace: 'one',\n ariaLabel: 'One',\n title: '1',\n },\n {\n namespace: 'two',\n ariaLabel: 'Two',\n title: '2',\n subtitle: 'ABC',\n },\n {\n namespace: 'three',\n ariaLabel: 'Three',\n title: '3',\n subtitle: 'DEF',\n },\n {\n namespace: 'four',\n ariaLabel: 'Four',\n title: '4',\n subtitle: 'GHI',\n },\n {\n namespace: 'five',\n ariaLabel: 'Five',\n title: '5',\n subtitle: 'JKL',\n },\n {\n namespace: 'six',\n ariaLabel: 'Six',\n title: '6',\n subtitle: 'MNO',\n },\n {\n namespace: 'seven',\n ariaLabel: 'Seven',\n title: '7',\n subtitle: 'PQRS',\n },\n {\n namespace: 'eight',\n ariaLabel: 'Eight',\n title: '8',\n subtitle: 'TUV',\n },\n {\n namespace: 'nine',\n ariaLabel: 'Nine',\n title: '9',\n subtitle: 'WXYZ',\n },\n {\n namespace: 'star',\n ariaLabel: 'Star',\n title: '*',\n },\n {\n namespace: 'zero',\n ariaLabel: 'Zero',\n title: '0',\n subtitle: '+',\n },\n {\n namespace: 'hash',\n ariaLabel: 'Hash',\n title: '#',\n },\n];\n\nexport default KEYPAD_BUTTONS_DATA;\n","import DialpadButton from '../components/buttons/buttons';\nimport LongPressEvent from '../utilities/longPress';\nimport KEYPAD_BUTTONS_DATA from './data';\nimport { KeypadButtonData, KeypadConfig } from './types';\n\n/**\n *\n * Keypad\n *\n */\nexport default class Keypad {\n // keypad config\n private config: KeypadConfig;\n\n /**\n *\n * construct Keypad instance\n *\n */\n constructor(config: KeypadConfig) {\n this.config = config;\n }\n\n /**\n *\n * Keypad skeleton\n *\n */\n private get skeleton(): HTMLDivElement {\n const keypad = document.createElement('div');\n keypad.id = this.id;\n keypad.classList.add('keypad');\n\n // append digits buttons\n KEYPAD_BUTTONS_DATA.forEach((config: KeypadButtonData) => {\n // create button instance\n const btn = new DialpadButton({\n namespace: config.namespace,\n ariaLabel: config.ariaLabel,\n title: config.title,\n subtitle: config.subtitle,\n onClick: this.config.onKeypadBtnClick,\n onLongPress:\n config.title === '0' ? this.config.onZeroBtnLongPress : undefined,\n onLongPressCancel:\n config.title === '0'\n ? this.config.onZeroBtnLongPressCancel\n : undefined,\n });\n\n // append button\n btn.build(keypad);\n });\n\n // append dummy element\n keypad.appendChild(document.createElement('span'));\n\n // append call button\n keypad.appendChild(this.callButton);\n\n // append backspace button\n keypad.appendChild(this.backspaceButton);\n\n return keypad;\n }\n\n // eslint-disable-next-line class-methods-use-this\n getMaterialIcon(iconName: string) {\n const iconElement = document.createElement('span');\n iconElement.className = 'material-symbols-outlined';\n iconElement.innerText = iconName;\n return iconElement;\n }\n\n /**\n *\n * Keypad call button\n *\n */\n private get callButton(): HTMLElement {\n const callBtn = document.createElement('button');\n callBtn.classList.add('keypad__call-btn');\n callBtn.setAttribute('aria-label', 'call button');\n\n // append call icon\n callBtn.appendChild(this.getMaterialIcon('call'));\n\n // add click event listener\n callBtn.addEventListener('click', this.config.onCallBtnClick);\n\n return callBtn;\n }\n\n /**\n *\n * Keypad backspace button\n *\n */\n private get backspaceButton(): HTMLElement {\n const backspaceBtn = document.createElement('button');\n backspaceBtn.classList.add('keypad__backspace-btn');\n backspaceBtn.setAttribute('aria-label', 'Backspace button');\n backspaceBtn.disabled = true;\n\n // append backspace icon\n backspaceBtn.appendChild(this.getMaterialIcon('backspace'));\n\n // add click event listener\n backspaceBtn.addEventListener('click', this.config.onBackspaceBtnClick);\n\n // apply long press event\n LongPressEvent.apply({\n target: backspaceBtn,\n onLongPressCallback: this.config.onBackspaceBtnLongPress,\n });\n\n return backspaceBtn;\n }\n\n /**\n *\n * To get backspace button element for dom manipulations\n *\n */\n get backspaceButtonElement(): HTMLButtonElement {\n return this.querySelector.querySelector(\n '.keypad__backspace-btn'\n ) as HTMLButtonElement;\n }\n\n /**\n *\n * Unique id for keypad\n *\n */\n get id() {\n return `keypad-${this.config.namespace}`;\n }\n\n /**\n *\n * Keypad query selector for dom manipulation\n *\n */\n get querySelector(): HTMLDivElement {\n return document.getElementById(this.id)! as HTMLDivElement;\n }\n\n /**\n *\n * Appends keypad to a specified parent element.\n *\n * @param parentElement\n *\n */\n build(parentElement: HTMLElement) {\n parentElement.appendChild(this.skeleton);\n }\n\n /**\n *\n * Enable backspace button\n *\n */\n enableBackspaceButton() {\n this.backspaceButtonElement.disabled = false;\n }\n\n /**\n *\n * Disable backspace button\n *\n */\n disableBackspaceButton() {\n this.backspaceButtonElement.disabled = true;\n }\n}\n","import Dialpad from './pages/dialpad';\n\nwindow.onload = () => {\n // create dialpad instance\n const dialpad = new Dialpad({ namespace: 'demo' });\n\n // build dialpad\n dialpad.build(document.body);\n};\n","import InputElement from '../components/forms/inputs';\r\nimport Keypad from '../layout/keypad';\r\nimport DialpadConfig from './types';\r\n\r\n/**\r\n *\r\n * Dialpad Page\r\n *\r\n */\r\nexport default class Dialpad {\r\n // dialpad configuration\r\n private config: DialpadConfig;\r\n\r\n // input field instance\r\n private inputField!: InputElement;\r\n\r\n // keypad instance\r\n private keypad!: Keypad;\r\n\r\n // recent call number\r\n private recentCallOnNumber?: string;\r\n\r\n /**\r\n *\r\n * construct Dialpad instance\r\n *\r\n */\r\n constructor(config: DialpadConfig) {\r\n this.config = config;\r\n }\r\n\r\n /**\r\n *\r\n * page skeleton\r\n *\r\n * @returns {HTMLDivElement}\r\n *\r\n */\r\n private skeleton(): HTMLElement {\r\n const dialpad = document.createElement('section');\r\n dialpad.id = this.id;\r\n dialpad.className = 'dialpad';\r\n\r\n // build input field\r\n this.inputFieldLayout.build(dialpad);\r\n\r\n // build keypad layout\r\n this.keypadLayout.build(dialpad);\r\n\r\n // append copyright text\r\n dialpad.appendChild(this.copyrightText);\r\n\r\n return dialpad;\r\n }\r\n\r\n /**\r\n *\r\n * To get Input Element\r\n *\r\n */\r\n private get inputFieldLayout(): InputElement {\r\n this.inputField = new InputElement({\r\n namespace: this.config.namespace,\r\n onValueEmpty: () => {\r\n // eslint-disable-next-line no-console\r\n console.log('disable backspace button');\r\n this.keypad.disableBackspaceButton();\r\n },\r\n onValueNonEmpty: () => {\r\n // eslint-disable-next-line no-console\r\n console.log('enable backspace button');\r\n this.keypad.enableBackspaceButton();\r\n },\r\n });\r\n return this.inputField;\r\n }\r\n\r\n /**\r\n *\r\n * To get Keypad Layout\r\n *\r\n */\r\n private get keypadLayout() {\r\n this.keypad = new Keypad({\r\n namespace: this.config.namespace,\r\n onKeypadBtnClick: (value: string) => {\r\n // eslint-disable-next-line no-console\r\n console.log('clicked on button', value);\r\n\r\n // insert value\r\n this.inputField.insertValue(value);\r\n this.keypad.enableBackspaceButton();\r\n },\r\n onZeroBtnLongPress: (value: string) => {\r\n // eslint-disable-next-line no-console\r\n console.log('long pressed on zero button');\r\n\r\n // insert zero button subtitle value `+`\r\n this.inputField.replaceValue(value);\r\n },\r\n onZeroBtnLongPressCancel: () => {\r\n // focus input field\r\n this.inputField.focus();\r\n },\r\n onCallBtnClick: () => {\r\n // eslint-disable-next-line no-console\r\n console.log('clicked on call button');\r\n\r\n // check whether try to call on recent number\r\n if (\r\n this.inputField.value === '' &&\r\n this.recentCallOnNumber !== undefined\r\n ) {\r\n // eslint-disable-next-line no-console\r\n console.log('fill last call on number in input field value');\r\n this.inputField.value = this.recentCallOnNumber;\r\n this.inputField.focus();\r\n this.keypad.enableBackspaceButton();\r\n }\r\n // if value is not empty (or add check for valid phone number)\r\n else if (this.inputField.value !== '') {\r\n // update last call on number\r\n this.recentCallOnNumber = this.inputField.value;\r\n\r\n // eslint-disable-next-line no-console\r\n console.log('placing call on ', this.inputField.value);\r\n }\r\n },\r\n onBackspaceBtnClick: () => {\r\n // eslint-disable-next-line no-console\r\n console.log('clicked on clear button');\r\n this.inputField.removeValue();\r\n },\r\n onBackspaceBtnLongPress: () => {\r\n // eslint-disable-next-line no-console\r\n console.log('long press on backspace button');\r\n\r\n // clear input value and hide backspace button\r\n this.inputField.value = '';\r\n this.inputField.focus();\r\n this.keypad.disableBackspaceButton();\r\n },\r\n });\r\n\r\n return this.keypad;\r\n }\r\n\r\n /**\r\n *\r\n * append copyright text element\r\n *\r\n */\r\n // eslint-disable-next-line class-methods-use-this\r\n get copyrightText(): HTMLParagraphElement {\r\n // create copyright text\r\n const copyrightTextElement = document.createElement('p');\r\n copyrightTextElement.className = 'copyright-text';\r\n copyrightTextElement.innerText = `Made by • Satyam Seth Ⓒ ${new Date().getFullYear()}`;\r\n\r\n return copyrightTextElement;\r\n }\r\n\r\n /**\r\n *\r\n * Unique id for page\r\n *\r\n * @returns {string}\r\n *\r\n */\r\n get id(): string {\r\n return `dialpad-${this.config.namespace}`;\r\n }\r\n\r\n /**\r\n *\r\n * querySelect for app\r\n *\r\n * @returns {HTMLElement}\r\n *\r\n */\r\n get element(): HTMLElement {\r\n return document.getElementById(this.id)!;\r\n }\r\n\r\n /**\r\n *\r\n * Remove dialpad skeleton from dom\r\n *\r\n */\r\n destroy(): void {\r\n this.element.remove();\r\n }\r\n\r\n /**\r\n *\r\n * Append dialpad skeleton into parent element\r\n *\r\n * @param parentElement {HTMLElement}\r\n */\r\n build(parentElement: HTMLElement) {\r\n parentElement.appendChild(this.skeleton());\r\n }\r\n}\r\n","import LongPressEventConfig from './types';\n\n/**\n *\n * LongPress event\n *\n */\nexport default class LongPressEvent {\n // target html element\n private config: LongPressEventConfig;\n\n // state to detect press and hold\n private isHeld: boolean = false;\n\n // setTimeout Id\n private activeHoldTimeoutId: number | null = null;\n\n // long press timeout in milliseconds\n private longPressTimeout = 500;\n\n /**\n *\n * construct LongPressEvent instance and add event listener\n *\n */\n constructor(config: LongPressEventConfig) {\n this.config = config;\n\n // start timer\n this.config.target.addEventListener(\n 'mousedown',\n this.onHoldStart.bind(this)\n );\n this.config.target.addEventListener(\n 'touchstart',\n this.onHoldStart.bind(this)\n );\n\n // stop timer\n this.config.target.addEventListener('mouseup', this.onHoldEnd.bind(this));\n this.config.target.addEventListener('mouseout', this.onHoldEnd.bind(this));\n this.config.target.addEventListener(\n 'mouseleave',\n this.onHoldEnd.bind(this)\n );\n this.config.target.addEventListener('touchend', this.onHoldEnd.bind(this));\n this.config.target.addEventListener(\n 'touchcancel',\n this.onHoldEnd.bind(this)\n );\n }\n\n /**\n *\n * start set timeout timer on long press event for callback execution\n *\n */\n private onHoldStart() {\n // eslint-disable-next-line no-console\n console.log('Start Pressing');\n this.isHeld = true;\n\n // call on press start handler\n if (this.config.onPressStart) {\n this.config.onPressStart();\n }\n\n this.activeHoldTimeoutId = window.setTimeout(() => {\n if (this.isHeld === true) {\n // eslint-disable-next-line no-console\n console.log('long press detected');\n this.config.onLongPressCallback();\n }\n }, this.longPressTimeout);\n }\n\n /**\n *\n * clear set timeout timer on long press event\n *\n */\n private onHoldEnd() {\n // eslint-disable-next-line no-console\n console.log('Stop Pressing');\n this.isHeld = false;\n\n // call on press start handler\n if (this.config.onLongPressCancel) {\n this.config.onLongPressCancel();\n }\n\n if (this.activeHoldTimeoutId !== null) {\n window.clearTimeout(this.activeHoldTimeoutId);\n this.activeHoldTimeoutId = null;\n }\n }\n\n /**\n *\n * Apply long press event without using `new` keyword\n *\n */\n static apply(config: LongPressEventConfig) {\n // eslint-disable-next-line no-new\n new LongPressEvent(config);\n }\n}\n\n// Example how to use\n// window.onload = () => {\n// // const demoButton = document.querySelector('button');\n// // Using new keyword\n// // new ClickAndHold(demoButton, () => {\n// // console.log('Long Press');\n// // alert('Long Press Detected');\n// // });\n// // // Without using new keyword\n// // ClickAndHold.apply(demoButton, () => {\n// // console.log('Long Press');\n// // alert('Long Press Detected');\n// // });\n// };\n"]} \ No newline at end of file diff --git a/src/ts/components/forms/inputs.ts b/src/ts/components/forms/inputs.ts index 9a4241d..df511d3 100644 --- a/src/ts/components/forms/inputs.ts +++ b/src/ts/components/forms/inputs.ts @@ -23,7 +23,7 @@ export default class InputElement { * input element skeleton * */ - private get skeleton(): HTMLInputElement { + get skeleton(): HTMLInputElement { const input = document.createElement('input'); input.id = this.id; input.className = 'input-element'; @@ -119,6 +119,15 @@ export default class InputElement { this.querySelector.selectionEnd = position; } + /** + * + * Focus input element + * + */ + focus() { + this.querySelector.focus(); + } + /** * * To check whether the input element is focused or not @@ -148,15 +157,6 @@ export default class InputElement { this.querySelector.value = value; } - /** - * - * Focus input element - * - */ - focus() { - this.querySelector.focus(); - } - /** * * Insert a value at the caret position @@ -256,15 +256,16 @@ export default class InputElement { */ validation() { // prepare updated state + const { value } = this; const caretPositionBeforeRemoveUnwantedChars = this.selectionStartPosition; - const updatedValueAfterRemoveUnwantedChars = this.value.replace( + const updatedValueAfterRemoveUnwantedChars = value.replace( /[^0-9+*#]/g, '' ); const updatedCaretPosition = caretPositionBeforeRemoveUnwantedChars + updatedValueAfterRemoveUnwantedChars.length - - this.value.length; + value.length; // update state this.value = updatedValueAfterRemoveUnwantedChars; @@ -276,7 +277,7 @@ export default class InputElement { * `input` event handler * */ - private inputEventHandler() { + inputEventHandler() { // Remove unwanted symbols and allow only digits, +, *, and # this.validation(); diff --git a/tests/components/buttons.ts b/tests/components/buttons.ts index 1e6d8af..666bca7 100644 --- a/tests/components/buttons.ts +++ b/tests/components/buttons.ts @@ -324,7 +324,7 @@ describe('Test Dialpad Button', () => { const button = new DialpadButton(validConfig); // Create spy for skeleton getter - const skeletonGetterSpy = sinon.spy(button, 'skeleton', ['get']); + const skeletonSpy = sinon.spy(button, 'skeleton', ['get']); // Create spy for appendChild method const appendChildSpy = sinon.spy(document.body, 'appendChild'); @@ -333,7 +333,7 @@ describe('Test Dialpad Button', () => { button.build(document.body); // Assert that the skeleton getter was accessed - expect(skeletonGetterSpy.get.calledOnce).to.be.true; + expect(skeletonSpy.get.calledOnce).to.be.true; // Assert that the parentElement appendChild was called once with the correct argument expect(appendChildSpy.calledOnceWith(sinon.match.instanceOf(HTMLElement))) diff --git a/tests/components/inputs.ts b/tests/components/inputs.ts new file mode 100644 index 0000000..9329c4e --- /dev/null +++ b/tests/components/inputs.ts @@ -0,0 +1,627 @@ +/* eslint-disable no-console */ +/* eslint-disable no-unused-expressions */ +import { expect } from 'chai'; +import { afterEach, beforeEach, describe, it } from 'mocha'; +import sinon from 'sinon'; +import InputElement from '../../src/ts/components/forms/inputs'; +import { InputElementConfig } from '../../src/ts/components/forms/types'; + +const jsdom = require('jsdom-global'); + +describe('Test Input Element', () => { + let config: InputElementConfig; + let cleanup: any; + + beforeEach(() => { + cleanup = jsdom(); + + config = { + namespace: 'test-namespace', + onValueEmpty: () => { + console.log('value empty'); + }, + onValueNonEmpty: () => { + console.log('value non empty'); + }, + }; + }); + + afterEach(() => { + // cleanup jsdom + cleanup(); + + // Restore the spies + sinon.restore(); + }); + + it('should able to create object for valid config', () => { + // Create InputElement instance + const input = new InputElement(config); + + // Assert that input is instance of InputElement + expect(input).to.be.instanceOf(InputElement); + }); + + it('should has correct id', () => { + // Create InputElement instance + const input = new InputElement(config); + + // Assert that id is correct + expect(input.id).to.equal(`input-test-namespace`); + }); + + it('querySelector should retrieve button HTMLElement', () => { + // Create InputElement + const input = new InputElement(config); + + // Expected input id + const inputId = `input-${config.namespace}`; + + // Create spy for document getElementById + const getElementByIdSpy = sinon.spy(document, 'getElementById'); + + // Call the build method + input.build(document.body); + + // Call querySelector + const result = input.querySelector; + + // Assert that getElementById call with expected id + expect(getElementByIdSpy.calledOnceWith(inputId)).to.be.true; + + // Assert that the result is an HTMLInputElement + expect(result).to.be.an.instanceOf(HTMLInputElement); + + // Assert that result HTMLElement has expected id + expect(result.id).to.equal(inputId); + }); + + it('build should append skeleton to parentElement', () => { + // Create InputElement instance + const input = new InputElement(config); + + // Create spy for skeleton getter + const skeletonSpy = sinon.spy(input, 'skeleton', ['get']); + + // Create spy for appendChild method + const appendChildSpy = sinon.spy(document.body, 'appendChild'); + + // Call the build method + input.build(document.body); + + // Assert that the skeleton getter was accessed + expect(skeletonSpy.get.calledOnce).to.be.true; + + // Assert that the parentElement appendChild was called once with the correct argument + expect(appendChildSpy.calledOnceWith(sinon.match.instanceOf(HTMLElement))) + .to.be.true; + + // Assert that the parentElement now contains the button skeleton + expect(input.querySelector).to.exist; + }); + + it('skeleton should return correct html element', () => { + // Create InputElement instance + const input = new InputElement(config); + + // Create stub for inputEventHandler + const inputEventHandlerStub = sinon.stub(input, 'inputEventHandler'); + + // Access skeleton + const { skeleton } = input; + + // Assert that the skeleton is an HTMLElement + expect(skeleton).to.be.instanceOf(HTMLInputElement); + + // Assert that the skeleton has the correct tag name + expect(skeleton.tagName).to.be.equal('INPUT'); + + // Assert that the skeleton has the correct class name + expect(skeleton.className).to.be.equal('input-element'); + + // Assert that the skeleton has the correct name attribute + expect(skeleton.getAttribute('name')).to.be.equal('number'); + + // Assert that the skeleton has the correct type attribute + expect(skeleton.getAttribute('type')).to.be.equal('text'); + + // Assert that the skeleton has the correct autofocus attribute + expect(skeleton.autofocus).to.be.true; + + // Assert that the skeleton has the correct inputMode attribute + expect(skeleton.getAttribute('inputMode')).to.be.equal('none'); + + // Assert that the skeleton has the correct autocomplete attribute + expect(skeleton.getAttribute('autocomplete')).to.be.equal('off'); + + // Simulate a click event on the skeleton + const inputEvent = new Event('input'); + skeleton.dispatchEvent(inputEvent); + + // Assert that the inputEventHandlerStub called once + expect(inputEventHandlerStub.calledOnce).to.be.true; + }); + + it('makeSureFocused should focus the element when not already focused', () => { + // Create InputElement instance + const input = new InputElement(config); + + // Create spy for focus + const focusSpy = sinon.spy(input, 'focus'); + + // build InputElement + input.build(document.body); + + // Call makeSureFocused to focus the element + input.makeSureFocused(); + + // Assert that focusSpy is called once + expect(focusSpy.calledOnce).to.be.true; + + // Call makeSureFocused + input.makeSureFocused(); + + // Assert that focusSpy is not called twice + // but called once because the input element is already focused. + expect(focusSpy.calledOnce).to.be.true; + }); + + it('selectionStartPosition should return the correct selection start position', () => { + // Create InputElement instance + const input = new InputElement(config); + + // Create spy for makeSureFocused + const makeSureFocusedSpy = sinon.spy(input, 'makeSureFocused'); + + // build InputElement + input.build(document.body); + + // Call selectionStartPosition and assert the value + expect(input.selectionStartPosition).to.equal(0); + + // Assert that makeSureFocusedSpy call once + expect(makeSureFocusedSpy.calledOnce).to.be.true; + }); + + it('selectionEndPosition should return the correct selection end position', () => { + // Create InputElement instance + const input = new InputElement(config); + + // Create spy for makeSureFocused + const makeSureFocusedSpy = sinon.spy(input, 'makeSureFocused'); + + // build InputElement + input.build(document.body); + + // Call selectionEndPosition and assert the value + expect(input.selectionEndPosition).to.equal(0); + + // Assert that makeSureFocusedSpy call once + expect(makeSureFocusedSpy.calledOnce).to.be.true; + }); + + it('selectionPosition should set the correct selection position', () => { + // Create InputElement instance + const input = new InputElement(config); + + // Create spy for makeSureFocused + const makeSureFocusedSpy = sinon.spy(input, 'makeSureFocused'); + + // build InputElement + input.build(document.body); + + // Set input field value + input.querySelector.value = 'hello'; + + // Call selectionPosition to set the selection position + input.selectionPosition = 2; + + // Assert that makeSureFocusedSpy call once + expect(makeSureFocusedSpy.calledOnce).to.be.true; + + // Assert the selection start position is correctly set + expect(input.selectionStartPosition).to.equal(2); + + // Assert the selection end position is correctly set + expect(input.selectionEndPosition).to.equal(2); + }); + + it('focused should return correct focus status', () => { + // Create InputElement instance + const input = new InputElement(config); + + // build InputElement + input.build(document.body); + + // Focus input element + input.querySelector.focus(); + + // Assert that the focused returns true + expect(input.focused).to.be.true; + + // Blur input element + input.querySelector.blur(); + + // Assert that the focused returns false + expect(input.focused).to.be.false; + }); + + it('focus should focus input element', () => { + // Create InputElement instance + const input = new InputElement(config); + + // build InputElement + input.build(document.body); + + // Assert InputElement is not focused before call focus + expect(input.focused).to.be.false; + + // Call focus + input.focus(); + + // Assert InputElement is focused after call focus + expect(input.focused).to.be.true; + }); + + it('value should return correct value', () => { + // Create InputElement instance + const input = new InputElement(config); + + // build InputElement + input.build(document.body); + + // Set input element value + input.querySelector.value = 'hello'; + + // Assert that the focused returns false + expect(input.value).to.be.equal('hello'); + }); + + it('value should set correct value', () => { + // Create InputElement instance + const input = new InputElement(config); + + // build InputElement + input.build(document.body); + + // Set input element value + input.value = 'hello'; + + // Assert that the focused returns false + expect(input.querySelector.value).to.be.equal('hello'); + }); + + it('replaceValue should working correctly', () => { + // Create InputElement instance + const input = new InputElement(config); + + // Create spy for removeValue + const removeValueSpy = sinon.spy(input, 'removeValue'); + + // Create spy for insertValue + const insertValueSpy = sinon.spy(input, 'insertValue'); + + // Create spy for inputEventHandler + const inputEventHandlerSpy = sinon.spy(input, 'inputEventHandler'); + + // build InputElement + input.build(document.body); + + // Set input element value + input.querySelector.value = 'hello'; + + // Call replaceValue + input.replaceValue('bye'); + + // Assert that the removeValueSpy called once + expect(removeValueSpy.calledOnce).to.be.true; + + // Assert that the insertValueSpy called once expected string value + expect(insertValueSpy.calledOnceWithExactly('bye')).to.be.true; + + // Assert that the inputEventHandlerSpy called twice + // (first time for remove value and second time for replaceValue) + expect(inputEventHandlerSpy.calledTwice).to.be.true; + }); + + it('clear should set the value to an empty string', () => { + // Create InputElement instance + const input = new InputElement(config); + + // Create spy for focus + const focusSpy = sinon.spy(input, 'focus'); + + // build InputElement + input.build(document.body); + + // Set input element value + input.querySelector.value = 'hello'; + + // Call clear method + input.clear(); + + // Assert input value is empty + expect(input.querySelector.value).to.equal(''); + + // Assert that the focusSpy called once + expect(focusSpy.calledOnce).to.be.true; + }); + + it('validation should update correct value and selection position after removing unwanted characters', () => { + // Create InputElement instance + const input = new InputElement(config); + + // build InputElement + input.build(document.body); + + // Set input element value + input.querySelector.value = '123abc*456#'; + + // Create spy for value getter and setter + const valueSpy = sinon.spy(input, 'value', ['get', 'set']); + + // Create spy for selectionStartPosition getter + const selectionStartPositionSpy = sinon.spy( + input, + 'selectionStartPosition', + ['get'] + ); + + // Create spy for selectionStartPosition setter + const selectionPositionSpy = sinon.spy(input, 'selectionPosition', ['set']); + + // Call the validation method + input.validation(); + + // Assert that the focusSpy called once + expect(valueSpy.get.calledOnce).to.be.true; + + // Assert that the selectionStartPositionSpy called once + expect(selectionStartPositionSpy.get.calledOnce).to.be.true; + + // Assert that the value is updated correctly + expect(valueSpy.set.calledOnceWith('123*456#')).to.be.true; + + // Assert that the selection position is updated correctly - 13 + 8 -13 = 8 + expect(selectionPositionSpy.set.calledOnceWith('123*456#'.length)).to.be + .true; + }); + + it('inputEventHandler should call validation and trigger callbacks based on value', () => { + // Create spy for onValueEmpty + const onValueEmptySpy = sinon.spy(); + + // Create spy for onValueNonEmptySpy + const onValueNonEmptySpy = sinon.spy(); + + // Create InputElement instance + const input = new InputElement({ + namespace: 'test-namespace', + onValueEmpty: onValueEmptySpy, + onValueNonEmpty: onValueNonEmptySpy, + }); + + // Create spy for validation + const validationSpy = sinon.spy(input, 'validation'); + + // build InputElement + input.build(document.body); + + // Call inputEventHandler method + input.inputEventHandler(); + + // Assert that validationSpy is called + expect(validationSpy.calledOnce).to.be.true; + + // Assert onValueEmptySpy is called + expect(onValueEmptySpy.calledOnce).to.be.true; + + // Assert onValueNonEmptySpy is not called + expect(onValueNonEmptySpy.calledOnce).to.be.false; + + // Reset spies + validationSpy.resetHistory(); + onValueEmptySpy.resetHistory(); + onValueNonEmptySpy.resetHistory(); + + // Set input element value + input.querySelector.value = '1234'; + + // Call inputEventHandler method + input.inputEventHandler(); + + // Assert that validationSpy is called + expect(validationSpy.calledOnce).to.be.true; + + // Assert onValueEmptySpy is not called + expect(onValueEmptySpy.calledOnce).to.be.false; + + // Assert onValueNonEmptySpy is called + expect(onValueNonEmptySpy.calledOnce).to.be.true; + }); + + it('removeValue should remove characters from the value starting at the caret position', () => { + // Create InputElement instance + const input = new InputElement(config); + + // Create spy for value getter and setter + const valueSpy = sinon.spy(input, 'value', ['get', 'set']); + + // Create spy for selectionStartPosition getter + const selectionStartPositionSpy = sinon.spy( + input, + 'selectionStartPosition', + ['get'] + ); + + // Create spy for selectionPosition setter + const selectionPositionSpy = sinon.spy(input, 'selectionPosition', ['set']); + + // Create spy for inputEventHandler + const inputEventHandlerSpy = sinon.spy(input, 'inputEventHandler'); + + // build InputElement + input.build(document.body); + + // Call removeValue method + input.removeValue(); + + // Assert that the focusSpy called once + expect(valueSpy.get.calledOnce).to.be.true; + + // Assert that the selectionStartPositionSpy called once + expect(selectionStartPositionSpy.get.calledOnce).to.be.true; + + // Assert that the valueSpy setter not called + expect(valueSpy.set.called).to.be.false; + + // Assert that the selectionPositionSpy setter not called + expect(selectionPositionSpy.set.called).to.be.false; + + // Assert that the inputEventHandlerSpy not called + expect(inputEventHandlerSpy.called).to.be.false; + + // Reset spies + valueSpy.get.resetHistory(); + valueSpy.set.resetHistory(); + selectionStartPositionSpy.get.resetHistory(); + selectionPositionSpy.set.resetHistory(); + inputEventHandlerSpy.resetHistory(); + + // Set input element value + input.querySelector.value = '123456789'; + + // Set selection start position + input.querySelector.selectionStart = 6; + + // Call removeValue method + input.removeValue(2); + + // Assert that the valueSpy getter called thrice + expect(valueSpy.get.calledThrice).to.be.true; + + // Assert that the selectionStartPositionSpy called twice + expect(selectionStartPositionSpy.get.calledTwice).to.be.true; + + // Assert that the valueSpy setter first time called twice + expect(valueSpy.set.calledTwice).to.be.true; + + // Assert that the valueSpy setter first time called with correct args + expect(valueSpy.set.getCall(0).calledWithExactly('1234789')).to.be.true; + + // Assert that the selectionPositionSpy setter called twice + expect(selectionPositionSpy.set.calledTwice).to.be.true; + + // Assert that the selectionPositionSpy setter first time called with correct args + expect(selectionPositionSpy.set.getCall(0).calledWithExactly(4)).to.be.true; + + // Assert that the inputEventHandlerSpy called once + expect(inputEventHandlerSpy.calledOnce).to.be.true; + }); + + it('insertValue should insert a value at the caret position', () => { + // Create InputElement instance + const input = new InputElement(config); + + // Create spy for value getter and setter + const valueSpy = sinon.spy(input, 'value', ['get', 'set']); + + // Create spy for selectionStartPosition getter and setter + const selectionStartPositionSpy = sinon.spy( + input, + 'selectionStartPosition', + ['get'] + ); + + // Create spy for selectionEndPosition getter and setter + const selectionEndPositionSpy = sinon.spy(input, 'selectionEndPosition', [ + 'get', + ]); + + // Create spy for selectionPosition setter + const selectionPositionSpy = sinon.spy(input, 'selectionPosition', ['set']); + + // build InputElement + input.build(document.body); + + // Set input element value + input.querySelector.value = '123789'; + + // Set selection start position + input.querySelector.selectionStart = 3; + + // Set selection end position + input.querySelector.selectionEnd = 3; + + // Call insertValue method - Insert '456' at the caret position + input.insertValue('456'); + + // Assert that the valueSpy getter called once + expect(valueSpy.get.calledOnce).to.be.true; + + // Assert that the selectionStartPositionSpy called once + expect(selectionStartPositionSpy.get.calledOnce).to.be.true; + + // Assert that the selectionEndPositionSpy called once + expect(selectionEndPositionSpy.get.calledOnce).to.be.true; + + // Assert that the valueSpy setter called once with correct args + expect(valueSpy.set.calledOnceWithExactly('123456789')).to.be.true; + + // Assert that the selectionPositionSpy setter called once with correct args + expect(selectionPositionSpy.set.calledOnceWithExactly(6)).to.be.true; + }); + + it('insertValue should replace selected value with the new value', () => { + // Create InputElement instance + const input = new InputElement(config); + + // Create spy for value getter and setter + const valueSpy = sinon.spy(input, 'value', ['get', 'set']); + + // Create spy for selectionStartPosition getter and setter + const selectionStartPositionSpy = sinon.spy( + input, + 'selectionStartPosition', + ['get'] + ); + + // Create spy for selectionEndPosition getter and setter + const selectionEndPositionSpy = sinon.spy(input, 'selectionEndPosition', [ + 'get', + ]); + + // Create spy for selectionPosition setter + const selectionPositionSpy = sinon.spy(input, 'selectionPosition', ['set']); + + // build InputElement + input.build(document.body); + + // Set input element value + input.querySelector.value = '123789'; + + // Set selection start position + input.querySelector.selectionStart = 3; + + // Set selection end position + input.querySelector.selectionEnd = 6; + + // Call insertValue method - Insert '456' at the caret position + input.insertValue('456'); + + // Assert that the valueSpy getter called once + expect(valueSpy.get.calledOnce).to.be.true; + + // Assert that the selectionStartPositionSpy called once + expect(selectionStartPositionSpy.get.calledOnce).to.be.true; + + // Assert that the selectionEndPositionSpy called once + expect(selectionEndPositionSpy.get.calledOnce).to.be.true; + + // Assert that the valueSpy setter called once with correct args + expect(valueSpy.set.calledOnceWithExactly('123456')).to.be.true; + + // Assert that the selectionPositionSpy setter called once with correct args + expect(selectionPositionSpy.set.calledOnceWithExactly(6)).to.be.true; + }); +});