UNPKG

angular-onscreen-material-keyboard

Version:

This package is forked from ngx-material-keyboard with bug fixes and additional features

375 lines 48 kB
import { ChangeDetectionStrategy, Component, EventEmitter, Inject, Input, Output } from '@angular/core'; import { BehaviorSubject } from 'rxjs'; import { MAT_KEYBOARD_DEADKEYS } from '../../configs/keyboard-deadkey.config'; import { KeyboardClassKey } from '../../enums/keyboard-class-key.enum'; export const VALUE_NEWLINE = '\n\r'; export const VALUE_SPACE = ' '; export const VALUE_TAB = '\t'; const REPEAT_TIMEOUT = 500; const REPEAT_INTERVAL = 100; export class MatKeyboardKeyComponent { // Inject dependencies constructor(_deadkeys) { this._deadkeys = _deadkeys; this._deadkeyKeys = []; this._repeatState = false; // true if repeating, false if waiting this.active$ = new BehaviorSubject(false); this.pressed$ = new BehaviorSubject(false); this.genericClick = new EventEmitter(); this.enterClick = new EventEmitter(); this.bkspClick = new EventEmitter(); this.capsClick = new EventEmitter(); this.altClick = new EventEmitter(); this.shiftClick = new EventEmitter(); this.spaceClick = new EventEmitter(); this.tabClick = new EventEmitter(); this.keyClick = new EventEmitter(); } set active(active) { this.active$.next(active); } get active() { return this.active$.getValue(); } set pressed(pressed) { this.pressed$.next(pressed); } get pressed() { return this.pressed$.getValue(); } get lowerKey() { return `${this.key}`.toLowerCase(); } get charCode() { return `${this.key}`.charCodeAt(0); } get isClassKey() { return this.key in KeyboardClassKey; } get isDeadKey() { return this._deadkeyKeys.some((deadKey) => deadKey === `${this.key}`); } get hasIcon() { return this.icon !== undefined && this.icon !== null; } get iconName() { return this.icon.name || ''; } get fontSet() { return this.icon.fontSet || ''; } get fontIcon() { return this.icon.fontIcon || ''; } get svgIcon() { return this.icon.svgIcon || ''; } get cssClass() { const classes = []; if (this.hasIcon) { classes.push('mat-keyboard-key-modifier'); classes.push(`mat-keyboard-key-${this.lowerKey}`); } if (this.isDeadKey) { classes.push('mat-keyboard-key-deadkey'); } return classes.join(' '); } get inputValue() { if (this.control) { return this.control.value; } else if (this.input && this.input.nativeElement && this.input.nativeElement.value) { return this.input.nativeElement.value; } else { return ''; } } set inputValue(inputValue) { if (this.control) { this.control.setValue(inputValue); } else if (this.input && this.input.nativeElement) { this.input.nativeElement.value = inputValue; } } ngOnInit() { // read the deadkeys this._deadkeyKeys = Object.keys(this._deadkeys); } ngOnDestroy() { this.cancelRepeat(); } onClick(event) { // Trigger generic click event this.genericClick.emit(event); // Do not execute keypress if key is currently repeating if (this._repeatState) { return; } // Trigger a global key event. TODO: investigate // this._triggerKeyEvent(); // Manipulate the focused input / textarea value const caret = this.input ? this._getCursorPosition() : 0; let char; switch (this.key) { // this keys have no actions yet // TODO: add deadkeys and modifiers case KeyboardClassKey.Alt: case KeyboardClassKey.AltGr: case KeyboardClassKey.AltLk: this.altClick.emit(event); break; case KeyboardClassKey.Bksp: this.deleteSelectedText(); this.bkspClick.emit(event); break; case KeyboardClassKey.Caps: this.capsClick.emit(event); break; case KeyboardClassKey.Enter: if (this._isTextarea()) { char = VALUE_NEWLINE; } else { this.enterClick.emit(event); // TODO: trigger submit / onSubmit / ngSubmit properly (for the time being this has to be handled by the user himself) // console.log(this.control.ngControl.control.root) // this.input.nativeElement.form.submit(); } break; case KeyboardClassKey.Shift: this.shiftClick.emit(event); break; case KeyboardClassKey.Space: char = VALUE_SPACE; this.spaceClick.emit(event); break; case KeyboardClassKey.Tab: char = VALUE_TAB; this.tabClick.emit(event); break; default: // the key is not mapped or a string char = `${this.key}`; this.keyClick.emit(event); break; } if (char && this.input) { this.replaceSelectedText(char); this._setCursorPosition(caret + 1); } // Dispatch Input Event for Angular to register a change if (this.input && this.input.nativeElement) { setTimeout(() => { this.input.nativeElement.dispatchEvent(new Event('input', { bubbles: true })); }); } } // Handle repeating keys. Keypress logic derived from onClick() onPointerDown() { this.cancelRepeat(); this._repeatState = false; this._repeatTimeoutHandler = setTimeout(() => { // Initialize keypress variables let char; let keyFn; switch (this.key) { // Ignore non-repeating keys case KeyboardClassKey.Alt: case KeyboardClassKey.AltGr: case KeyboardClassKey.AltLk: case KeyboardClassKey.Caps: case KeyboardClassKey.Enter: case KeyboardClassKey.Shift: return; case KeyboardClassKey.Bksp: keyFn = () => { this.deleteSelectedText(); this.bkspClick.emit(); }; break; case KeyboardClassKey.Space: char = VALUE_SPACE; keyFn = () => this.spaceClick.emit(); break; case KeyboardClassKey.Tab: char = VALUE_TAB; keyFn = () => this.tabClick.emit(); break; default: char = `${this.key}`; keyFn = () => this.keyClick.emit(); break; } // Execute repeating keypress this._repeatIntervalHandler = setInterval(() => { const caret = this.input ? this._getCursorPosition() : 0; this._repeatState = true; if (keyFn) { keyFn(); } if (char && this.input) { this.replaceSelectedText(char); this._setCursorPosition(caret + 1); } if (this.input && this.input.nativeElement) { setTimeout(() => this.input.nativeElement.dispatchEvent(new Event('input', { bubbles: true }))); } }, REPEAT_INTERVAL); }, REPEAT_TIMEOUT); } cancelRepeat() { if (this._repeatTimeoutHandler) { clearTimeout(this._repeatTimeoutHandler); this._repeatTimeoutHandler = null; } if (this._repeatIntervalHandler) { clearInterval(this._repeatIntervalHandler); this._repeatIntervalHandler = null; } } deleteSelectedText() { const value = this.inputValue ? this.inputValue.toString() : ''; let caret = this.input ? this._getCursorPosition() : 0; let selectionLength = this._getSelectionLength(); if (selectionLength === 0) { if (caret === 0) { return; } caret--; selectionLength = 1; } const headPart = value.slice(0, caret); const endPart = value.slice(caret + selectionLength); this.inputValue = [headPart, endPart].join(''); this._setCursorPosition(caret); } replaceSelectedText(char) { const value = this.inputValue ? this.inputValue.toString() : ''; const caret = this.input ? this._getCursorPosition() : 0; const selectionLength = this._getSelectionLength(); const headPart = value.slice(0, caret); const endPart = value.slice(caret + selectionLength); this.inputValue = [headPart, char, endPart].join(''); } // TODO: Include for repeating keys as well (if this gets implemented) // private _triggerKeyEvent(): Event { // const keyboardEvent = new KeyboardEvent('keydown'); // // // // keyboardEvent[initMethod]( // // true, // bubbles // // true, // cancelable // // window, // viewArg: should be window // // false, // ctrlKeyArg // // false, // altKeyArg // // false, // shiftKeyArg // // false, // metaKeyArg // // this.charCode, // keyCodeArg : unsigned long - the virtual key code, else 0 // // 0 // charCodeArgs : unsigned long - the Unicode character associated with the depressed key, else 0 // // ); // // // // window.document.dispatchEvent(keyboardEvent); // return keyboardEvent; // } // inspired by: // ref https://stackoverflow.com/a/2897510/1146207 _getCursorPosition() { if (!this.input) { return; } if ('selectionStart' in this.input.nativeElement) { // Standard-compliant browsers return this.input.nativeElement.selectionStart; } else if ('selection' in window.document) { // IE this.input.nativeElement.focus(); const selection = window.document['selection']; const sel = selection.createRange(); const selLen = selection.createRange().text.length; sel.moveStart('character', -this.control.value.length); return sel.text.length - selLen; } } _getSelectionLength() { if (!this.input) { return; } if ('selectionEnd' in this.input.nativeElement) { // Standard-compliant browsers return this.input.nativeElement.selectionEnd - this.input.nativeElement.selectionStart; } if ('selection' in window.document) { // IE this.input.nativeElement.focus(); const selection = window.document['selection']; return selection.createRange().text.length; } } // inspired by: // ref https://stackoverflow.com/a/12518737/1146207 // tslint:disable one-line _setCursorPosition(position) { if (!this.input) { return; } this.inputValue = this.control.value; // ^ this is used to not only get "focus", but // to make sure we don't have it everything -selected- // (it causes an issue in chrome, and having it doesn't hurt any other browser) if ('createTextRange' in this.input.nativeElement) { const range = this.input.nativeElement.createTextRange(); range.move('character', position); range.select(); return true; } else { // (el.selectionStart === 0 added for Firefox bug) if (this.input.nativeElement.selectionStart || this.input.nativeElement.selectionStart === 0) { this.input.nativeElement.focus(); this.input.nativeElement.setSelectionRange(position, position); return true; } // fail city, fortunately this never happens (as far as I've tested) :) else { this.input.nativeElement.focus(); return false; } } } _isTextarea() { return this.input && this.input.nativeElement && this.input.nativeElement.tagName === 'TEXTAREA'; } } MatKeyboardKeyComponent.decorators = [ { type: Component, args: [{ selector: 'mat-keyboard-key', template: "<button mat-raised-button\n class=\"mat-keyboard-key\"\n tabindex=\"-1\"\n [class.mat-keyboard-key-active]=\"active$ | async\"\n [class.mat-keyboard-key-pressed]=\"pressed$ | async\"\n [ngClass]=\"cssClass\"\n (click)=\"onClick($event)\"\n (pointerdown)=\"onPointerDown()\"\n (pointerleave)=\"cancelRepeat()\"\n (pointerup)=\"cancelRepeat()\"\n>\n <mat-icon *ngIf=\"hasIcon; else noIcon\" [fontSet]=\"fontSet\" [fontIcon]=\"fontIcon\" [svgIcon]=\"svgIcon\">{{ iconName }}</mat-icon>\n <ng-template #noIcon>{{ key }}</ng-template>\n</button>\n", changeDetection: ChangeDetectionStrategy.OnPush, preserveWhitespaces: false, styles: ["@charset \"UTF-8\";:host{display:flex;font-family:Roboto,Helvetica Neue,sans-serif;font-size:14px;justify-content:space-between;line-height:20px}.mat-keyboard-key{min-width:0;width:100%}.mat-keyboard-key-active{background-color:#e0e0e0}.mat-keyboard-key-pressed{background-color:#bdbdbd}.mat-keyboard-key-capslock{background-color:#fff}.mat-keyboard-key-capslock:before{background-color:#bdbdbd;border-radius:100%;content:\"\";display:inline-block;height:3px;left:5px;position:absolute;top:5px;transition:.4s cubic-bezier(.25,.8,.25,1);transition-property:background-color,box-shadow;width:3px}.mat-keyboard-key-capslock.mat-keyboard-key-active:before{background-color:#0f0;box-shadow:0 0 \u00A7px #adff2f}:host-context(.dark-theme) .mat-keyboard-key{background-color:#616161;color:#f5f5f5}:host-context(.dark-theme) .mat-keyboard-key-active{background-color:#9e9e9e}:host-context(.dark-theme) .mat-keyboard-key-pressed{background-color:#757575}:host-context(.debug) .mat-keyboard-key-deadkey{background-color:#5f9ea0}:host-context(.debug) .mat-keyboard-key-deadkey.mat-keyboard-key-active{background-color:#6fa8aa}:host-context(.debug) .mat-keyboard-key-deadkey.mat-keyboard-key-pressed{background-color:#7fb1b3}:host-context(.debug) .mat-keyboard-key-modifier{background-color:#7fffd4}:host-context(.debug) .mat-keyboard-key-modifier.mat-keyboard-key-active{background-color:#9fd}:host-context(.debug) .mat-keyboard-key-modifier.mat-keyboard-key-pressed{background-color:#b2ffe5}:host-context(.dark-theme.debug) .mat-keyboard-key-deadkey{background-color:#639}:host-context(.dark-theme.debug) .mat-keyboard-key-deadkey.mat-keyboard-key-active{background-color:#7339ac}:host-context(.dark-theme.debug) .mat-keyboard-key-deadkey.mat-keyboard-key-pressed{background-color:#8040bf}:host-context(.dark-theme.debug) .mat-keyboard-key-modifier{background-color:#9370db}:host-context(.dark-theme.debug) .mat-keyboard-key-modifier.mat-keyboard-key-active{background-color:#a284e0}:host-context(.dark-theme.debug) .mat-keyboard-key-modifier.mat-keyboard-key-pressed{background-color:#b299e5}"] },] } ]; MatKeyboardKeyComponent.ctorParameters = () => [ { type: undefined, decorators: [{ type: Inject, args: [MAT_KEYBOARD_DEADKEYS,] }] } ]; MatKeyboardKeyComponent.propDecorators = { key: [{ type: Input }], icon: [{ type: Input }], active: [{ type: Input }], pressed: [{ type: Input }], input: [{ type: Input }], control: [{ type: Input }], genericClick: [{ type: Output }], enterClick: [{ type: Output }], bkspClick: [{ type: Output }], capsClick: [{ type: Output }], altClick: [{ type: Output }], shiftClick: [{ type: Output }], spaceClick: [{ type: Output }], tabClick: [{ type: Output }], keyClick: [{ type: Output }] }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"keyboard-key.component.js","sourceRoot":"","sources":["../../../../../src/core/src/components/keyboard-key/keyboard-key.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, ElementRef, EventEmitter, Inject, Input, OnDestroy, OnInit, Output } from '@angular/core';\nimport { FormControl } from '@angular/forms';\nimport { BehaviorSubject } from 'rxjs';\nimport { MAT_KEYBOARD_DEADKEYS } from '../../configs/keyboard-deadkey.config';\nimport { KeyboardClassKey } from '../../enums/keyboard-class-key.enum';\nimport { IKeyboardDeadkeys } from '../../interfaces/keyboard-deadkeys.interface';\nimport { IMatIcon } from '../../interfaces/keyboard-icons.interface';\n\nexport const VALUE_NEWLINE = '\\n\\r';\nexport const VALUE_SPACE = ' ';\nexport const VALUE_TAB = '\\t';\nconst REPEAT_TIMEOUT = 500;\nconst REPEAT_INTERVAL = 100;\n\n@Component({\n  selector: 'mat-keyboard-key',\n  templateUrl: './keyboard-key.component.html',\n  styleUrls: ['./keyboard-key.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  preserveWhitespaces: false\n})\nexport class MatKeyboardKeyComponent implements OnInit, OnDestroy {\n\n  private _deadkeyKeys: string[] = [];\n  private _repeatTimeoutHandler: any;\n  private _repeatIntervalHandler: any;\n  private _repeatState: boolean = false; // true if repeating, false if waiting\n\n  active$: BehaviorSubject<boolean> = new BehaviorSubject(false);\n\n  pressed$: BehaviorSubject<boolean> = new BehaviorSubject(false);\n\n  @Input()\n  key: string | KeyboardClassKey;\n\n  @Input()\n  icon: IMatIcon;\n\n  @Input()\n  set active(active: boolean) {\n    this.active$.next(active);\n  }\n\n  get active(): boolean {\n    return this.active$.getValue();\n  }\n\n  @Input()\n  set pressed(pressed: boolean) {\n    this.pressed$.next(pressed);\n  }\n\n  get pressed(): boolean {\n    return this.pressed$.getValue();\n  }\n\n  @Input()\n  input?: ElementRef;\n\n  @Input()\n  control?: FormControl;\n\n  @Output()\n  genericClick = new EventEmitter<MouseEvent>();\n\n  @Output()\n  enterClick = new EventEmitter<MouseEvent>();\n\n  @Output()\n  bkspClick = new EventEmitter<MouseEvent>();\n\n  @Output()\n  capsClick = new EventEmitter<MouseEvent>();\n\n  @Output()\n  altClick = new EventEmitter<MouseEvent>();\n\n  @Output()\n  shiftClick = new EventEmitter<MouseEvent>();\n\n  @Output()\n  spaceClick = new EventEmitter<MouseEvent>();\n\n  @Output()\n  tabClick = new EventEmitter<MouseEvent>();\n\n  @Output()\n  keyClick = new EventEmitter<MouseEvent>();\n\n  get lowerKey(): string {\n    return `${this.key}`.toLowerCase();\n  }\n\n  get charCode(): number {\n    return `${this.key}`.charCodeAt(0);\n  }\n\n  get isClassKey(): boolean {\n    return this.key in KeyboardClassKey;\n  }\n\n  get isDeadKey(): boolean {\n    return this._deadkeyKeys.some((deadKey: string) => deadKey === `${this.key}`);\n  }\n\n  get hasIcon(): boolean {\n    return this.icon !== undefined && this.icon !== null;\n  }\n\n  get iconName(): string {\n    return this.icon.name || '';\n  }\n\n  get fontSet(): string {\n    return this.icon.fontSet || '';\n  }\n\n  get fontIcon(): string {\n    return this.icon.fontIcon || '';\n  }\n\n  get svgIcon(): string {\n    return this.icon.svgIcon || '';\n  }\n\n  get cssClass(): string {\n    const classes = [];\n\n    if (this.hasIcon) {\n      classes.push('mat-keyboard-key-modifier');\n      classes.push(`mat-keyboard-key-${this.lowerKey}`);\n    }\n\n    if (this.isDeadKey) {\n      classes.push('mat-keyboard-key-deadkey');\n    }\n\n    return classes.join(' ');\n  }\n\n  get inputValue(): string {\n    if (this.control) {\n      return this.control.value;\n    } else if (this.input && this.input.nativeElement && this.input.nativeElement.value) {\n      return this.input.nativeElement.value;\n    } else {\n      return '';\n    }\n  }\n\n  set inputValue(inputValue: string) {\n    if (this.control) {\n      this.control.setValue(inputValue);\n    } else if (this.input && this.input.nativeElement) {\n      this.input.nativeElement.value = inputValue;\n    }\n  }\n\n  // Inject dependencies\n  constructor(@Inject(MAT_KEYBOARD_DEADKEYS) private _deadkeys: IKeyboardDeadkeys) { }\n\n  ngOnInit() {\n    // read the deadkeys\n    this._deadkeyKeys = Object.keys(this._deadkeys);\n  }\n\n  ngOnDestroy() {\n    this.cancelRepeat();\n  }\n\n  onClick(event: MouseEvent) {\n    // Trigger generic click event\n    this.genericClick.emit(event);\n\n    // Do not execute keypress if key is currently repeating\n    if (this._repeatState) { return; }\n\n    // Trigger a global key event. TODO: investigate\n    // this._triggerKeyEvent();\n\n    // Manipulate the focused input / textarea value\n    const caret = this.input ? this._getCursorPosition() : 0;\n\n    let char: string;\n    switch (this.key) {\n      // this keys have no actions yet\n      // TODO: add deadkeys and modifiers\n      case KeyboardClassKey.Alt:\n      case KeyboardClassKey.AltGr:\n      case KeyboardClassKey.AltLk:\n        this.altClick.emit(event);\n        break;\n\n      case KeyboardClassKey.Bksp:\n        this.deleteSelectedText();\n        this.bkspClick.emit(event);\n        break;\n\n      case KeyboardClassKey.Caps:\n        this.capsClick.emit(event);\n        break;\n\n      case KeyboardClassKey.Enter:\n        if (this._isTextarea()) {\n          char = VALUE_NEWLINE;\n        } else {\n          this.enterClick.emit(event);\n          // TODO: trigger submit / onSubmit / ngSubmit properly (for the time being this has to be handled by the user himself)\n          // console.log(this.control.ngControl.control.root)\n          // this.input.nativeElement.form.submit();\n        }\n        break;\n\n      case KeyboardClassKey.Shift:\n        this.shiftClick.emit(event);\n        break;\n\n      case KeyboardClassKey.Space:\n        char = VALUE_SPACE;\n        this.spaceClick.emit(event);\n        break;\n\n      case KeyboardClassKey.Tab:\n        char = VALUE_TAB;\n        this.tabClick.emit(event);\n        break;\n\n      default:\n        // the key is not mapped or a string\n        char = `${this.key}`;\n        this.keyClick.emit(event);\n        break;\n    }\n\n    if (char && this.input) {\n      this.replaceSelectedText(char);\n      this._setCursorPosition(caret + 1);\n    }\n\n    // Dispatch Input Event for Angular to register a change\n    if (this.input && this.input.nativeElement) {\n      setTimeout(() => {\n        this.input.nativeElement.dispatchEvent(new Event('input', { bubbles: true }));\n      });\n    }\n  }\n\n  // Handle repeating keys. Keypress logic derived from onClick()\n  onPointerDown() {\n    this.cancelRepeat();\n    this._repeatState = false;\n    this._repeatTimeoutHandler = setTimeout(() => {\n      // Initialize keypress variables\n      let char: string;\n      let keyFn: () => void;\n\n      switch (this.key) {\n        // Ignore non-repeating keys\n        case KeyboardClassKey.Alt:\n        case KeyboardClassKey.AltGr:\n        case KeyboardClassKey.AltLk:\n        case KeyboardClassKey.Caps:\n        case KeyboardClassKey.Enter:\n        case KeyboardClassKey.Shift:\n          return;\n\n        case KeyboardClassKey.Bksp:\n          keyFn = () => {\n            this.deleteSelectedText();\n            this.bkspClick.emit();\n          };\n          break;\n\n        case KeyboardClassKey.Space:\n          char = VALUE_SPACE;\n          keyFn = () => this.spaceClick.emit();\n          break;\n\n        case KeyboardClassKey.Tab:\n          char = VALUE_TAB;\n          keyFn = () => this.tabClick.emit();\n          break;\n\n        default:\n          char = `${this.key}`;\n          keyFn = () => this.keyClick.emit();\n          break;\n      }\n\n      // Execute repeating keypress\n      this._repeatIntervalHandler = setInterval(() => {\n        const caret = this.input ? this._getCursorPosition() : 0;\n        this._repeatState = true;\n\n        if (keyFn) { keyFn(); }\n\n        if (char && this.input) {\n          this.replaceSelectedText(char);\n          this._setCursorPosition(caret + 1);\n        }\n\n        if (this.input && this.input.nativeElement) {\n          setTimeout(() => this.input.nativeElement.dispatchEvent(new Event('input', { bubbles: true })));\n        }\n      }, REPEAT_INTERVAL);\n    }, REPEAT_TIMEOUT);\n  }\n\n  cancelRepeat() {\n    if (this._repeatTimeoutHandler) {\n      clearTimeout(this._repeatTimeoutHandler);\n      this._repeatTimeoutHandler = null;\n    }\n\n    if (this._repeatIntervalHandler) {\n      clearInterval(this._repeatIntervalHandler);\n      this._repeatIntervalHandler = null;\n    }\n  }\n\n  private deleteSelectedText(): void {\n    const value = this.inputValue ? this.inputValue.toString() : '';\n    let caret = this.input ? this._getCursorPosition() : 0;\n    let selectionLength = this._getSelectionLength();\n    if (selectionLength === 0) {\n      if (caret === 0) {\n        return;\n      }\n\n      caret--;\n      selectionLength = 1;\n    }\n\n    const headPart = value.slice(0, caret);\n    const endPart = value.slice(caret + selectionLength);\n\n    this.inputValue = [headPart, endPart].join('');\n    this._setCursorPosition(caret);\n  }\n\n  private replaceSelectedText(char: string): void {\n    const value = this.inputValue ? this.inputValue.toString() : '';\n    const caret = this.input ? this._getCursorPosition() : 0;\n    const selectionLength = this._getSelectionLength();\n    const headPart = value.slice(0, caret);\n    const endPart = value.slice(caret + selectionLength);\n\n    this.inputValue = [headPart, char, endPart].join('');\n  }\n\n  // TODO: Include for repeating keys as well (if this gets implemented)\n  // private _triggerKeyEvent(): Event {\n  //   const keyboardEvent = new KeyboardEvent('keydown');\n  //   //\n  //   // keyboardEvent[initMethod](\n  //   //   true, // bubbles\n  //   //   true, // cancelable\n  //   //   window, // viewArg: should be window\n  //   //   false, // ctrlKeyArg\n  //   //   false, // altKeyArg\n  //   //   false, // shiftKeyArg\n  //   //   false, // metaKeyArg\n  //   //   this.charCode, // keyCodeArg : unsigned long - the virtual key code, else 0\n  //   //   0 // charCodeArgs : unsigned long - the Unicode character associated with the depressed key, else 0\n  //   // );\n  //   //\n  //   // window.document.dispatchEvent(keyboardEvent);\n\n  //   return keyboardEvent;\n  // }\n\n  // inspired by:\n  // ref https://stackoverflow.com/a/2897510/1146207\n  private _getCursorPosition(): number {\n    if (!this.input) {\n      return;\n    }\n\n    if ('selectionStart' in this.input.nativeElement) {\n      // Standard-compliant browsers\n      return this.input.nativeElement.selectionStart;\n    } else if ('selection' in window.document) {\n      // IE\n      this.input.nativeElement.focus();\n      const selection: any = window.document['selection'];\n      const sel = selection.createRange();\n      const selLen = selection.createRange().text.length;\n      sel.moveStart('character', -this.control.value.length);\n\n      return sel.text.length - selLen;\n    }\n  }\n\n  private _getSelectionLength(): number {\n    if (!this.input) {\n      return;\n    }\n\n    if ('selectionEnd' in this.input.nativeElement) {\n      // Standard-compliant browsers\n      return this.input.nativeElement.selectionEnd - this.input.nativeElement.selectionStart;\n    }\n\n    if ('selection' in window.document) {\n      // IE\n      this.input.nativeElement.focus();\n      const selection: any = window.document['selection'];\n      return selection.createRange().text.length;\n    }\n  }\n\n  // inspired by:\n  // ref https://stackoverflow.com/a/12518737/1146207\n  // tslint:disable one-line\n  private _setCursorPosition(position: number): boolean {\n    if (!this.input) {\n      return;\n    }\n\n    this.inputValue = this.control.value;\n    // ^ this is used to not only get \"focus\", but\n    // to make sure we don't have it everything -selected-\n    // (it causes an issue in chrome, and having it doesn't hurt any other browser)\n\n    if ('createTextRange' in this.input.nativeElement) {\n      const range = this.input.nativeElement.createTextRange();\n      range.move('character', position);\n      range.select();\n      return true;\n    } else {\n      // (el.selectionStart === 0 added for Firefox bug)\n      if (this.input.nativeElement.selectionStart || this.input.nativeElement.selectionStart === 0) {\n        this.input.nativeElement.focus();\n        this.input.nativeElement.setSelectionRange(position, position);\n        return true;\n      }\n      // fail city, fortunately this never happens (as far as I've tested) :)\n      else {\n        this.input.nativeElement.focus();\n        return false;\n      }\n    }\n  }\n\n  private _isTextarea(): boolean {\n    return this.input && this.input.nativeElement && this.input.nativeElement.tagName === 'TEXTAREA';\n  }\n\n}\n"]}