systelab-virtual-keyboard 0.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md ADDED
@@ -0,0 +1,37 @@
1
+ # SystelabVirtualKeyboard
2
+
3
+ Systelab Virtual Keyboard is a on-screen keyboard focused on touch devices.
4
+
5
+ ## Using the directive
6
+
7
+ Before use the directive you have to import the module into your application.
8
+
9
+ ```typescript
10
+ NgModule({
11
+ imports: [
12
+ ...,
13
+ SystelabVirtualKeyboardModule,
14
+ ]
15
+ })
16
+ ```
17
+
18
+ Then the virtual keyboard can be enabled for inputs or textareas. To enable it the attribute *vkEnabled* can be used into the desire HTML element. It can be binding dynamically in the same way *[vkEnabled]="vkEnabledVariable"*
19
+
20
+ ```html
21
+ <input class="input" type="number" placeholder="Numeric input default layout" vkEnabled vkFixedBottom [vkConfig]="vkConfig">
22
+ ```
23
+
24
+ ## Directive options
25
+
26
+ The default behaviour can be overrided trought some attributes:
27
+
28
+ * vkEnabled: enable the virtual keyboard for the element
29
+ * vkFixedBottom: fix the virtual keyboard to the bottom of the page
30
+ * vkConfig: configuration object
31
+
32
+ ### Configuration object
33
+
34
+ The configuration params are describe into the interface *SystelabVirtualKeyboardConfig*
35
+
36
+ * layout: usually the virtual keyboard select the layout between *default* or *numeric* depending on the input type. But with the config object you can override this behaviour and force the desired layout. The available layouts are described in the enum *SystelabVirtualKeyboardLayouts*
37
+
@@ -0,0 +1,32 @@
1
+ /*
2
+ * Copyright (c) 2020 - 2024 - Instrumentation Laboratory Company and Systelab Technologies, SA. All rights reserved.
3
+ * NOTICE: All information contained herein is and remains the property of Instrumentation Laboratory Company and its
4
+ * affiliates, if any. The intellectual and technical concepts contained herein are proprietary to Instrumentation
5
+ * Laboratory Company and its affiliates and may be covered by U.S. and foreign patents and patent applications, and/or
6
+ * are protected by trade secret or copyright law. Dissemination of this information or reproduction of this material is
7
+ * strictly forbidden unless prior written permission is Instrumentation Laboratory Company.
8
+ */
9
+ export var SystelabVirtualKeyboardButton;
10
+ (function (SystelabVirtualKeyboardButton) {
11
+ SystelabVirtualKeyboardButton["Done"] = "{done}";
12
+ SystelabVirtualKeyboardButton["Enter"] = "{enter}";
13
+ SystelabVirtualKeyboardButton["Shift"] = "{shift}";
14
+ SystelabVirtualKeyboardButton["Lock"] = "{lock}";
15
+ SystelabVirtualKeyboardButton["Backspace"] = "{bksp}";
16
+ SystelabVirtualKeyboardButton["Language"] = "{language}";
17
+ SystelabVirtualKeyboardButton["Space"] = "{space}";
18
+ SystelabVirtualKeyboardButton["Tab"] = "{tab}";
19
+ })(SystelabVirtualKeyboardButton || (SystelabVirtualKeyboardButton = {}));
20
+ export var SystelabVirtualKeyboardLayouts;
21
+ (function (SystelabVirtualKeyboardLayouts) {
22
+ SystelabVirtualKeyboardLayouts["default"] = "default";
23
+ SystelabVirtualKeyboardLayouts["shift"] = "shift";
24
+ SystelabVirtualKeyboardLayouts["numeric"] = "numeric";
25
+ })(SystelabVirtualKeyboardLayouts || (SystelabVirtualKeyboardLayouts = {}));
26
+ export var SystelabVirtualKeyboardInputTypes;
27
+ (function (SystelabVirtualKeyboardInputTypes) {
28
+ SystelabVirtualKeyboardInputTypes["text"] = "text";
29
+ SystelabVirtualKeyboardInputTypes["number"] = "number";
30
+ SystelabVirtualKeyboardInputTypes["password"] = "password";
31
+ })(SystelabVirtualKeyboardInputTypes || (SystelabVirtualKeyboardInputTypes = {}));
32
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uc3RhbnRzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vcHJvamVjdHMvc3lzdGVsYWItdmlydHVhbC1rZXlib2FyZC9zcmMvbGliL2NvbnN0YW50cy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7OztHQU9HO0FBRUgsTUFBTSxDQUFOLElBQVksNkJBU1g7QUFURCxXQUFZLDZCQUE2QjtJQUN2QyxnREFBZSxDQUFBO0lBQ2Ysa0RBQWlCLENBQUE7SUFDakIsa0RBQWlCLENBQUE7SUFDakIsZ0RBQWUsQ0FBQTtJQUNmLHFEQUFvQixDQUFBO0lBQ3BCLHdEQUF1QixDQUFBO0lBQ3ZCLGtEQUFpQixDQUFBO0lBQ2pCLDhDQUFhLENBQUE7QUFDZixDQUFDLEVBVFcsNkJBQTZCLEtBQTdCLDZCQUE2QixRQVN4QztBQUVELE1BQU0sQ0FBTixJQUFZLDhCQUlYO0FBSkQsV0FBWSw4QkFBOEI7SUFDeEMscURBQW1CLENBQUE7SUFDbkIsaURBQWUsQ0FBQTtJQUNmLHFEQUFtQixDQUFBO0FBQ3JCLENBQUMsRUFKVyw4QkFBOEIsS0FBOUIsOEJBQThCLFFBSXpDO0FBRUQsTUFBTSxDQUFOLElBQVksaUNBSVg7QUFKRCxXQUFZLGlDQUFpQztJQUMzQyxrREFBYSxDQUFBO0lBQ2Isc0RBQWlCLENBQUE7SUFDakIsMERBQXFCLENBQUE7QUFDdkIsQ0FBQyxFQUpXLGlDQUFpQyxLQUFqQyxpQ0FBaUMsUUFJNUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKlxuICogQ29weXJpZ2h0IChjKSAyMDIwIC0gMjAyNCAtIEluc3RydW1lbnRhdGlvbiBMYWJvcmF0b3J5IENvbXBhbnkgYW5kIFN5c3RlbGFiIFRlY2hub2xvZ2llcywgU0EuIEFsbCByaWdodHMgcmVzZXJ2ZWQuXG4gKiBOT1RJQ0U6ICBBbGwgaW5mb3JtYXRpb24gY29udGFpbmVkIGhlcmVpbiBpcyBhbmQgcmVtYWlucyB0aGUgcHJvcGVydHkgb2YgSW5zdHJ1bWVudGF0aW9uIExhYm9yYXRvcnkgQ29tcGFueSBhbmQgaXRzXG4gKiBhZmZpbGlhdGVzLCBpZiBhbnkuICBUaGUgaW50ZWxsZWN0dWFsIGFuZCB0ZWNobmljYWwgY29uY2VwdHMgY29udGFpbmVkIGhlcmVpbiBhcmUgcHJvcHJpZXRhcnkgdG8gSW5zdHJ1bWVudGF0aW9uXG4gKiBMYWJvcmF0b3J5IENvbXBhbnkgYW5kIGl0cyBhZmZpbGlhdGVzIGFuZCBtYXkgYmUgY292ZXJlZCBieSBVLlMuIGFuZCBmb3JlaWduIHBhdGVudHMgYW5kIHBhdGVudCBhcHBsaWNhdGlvbnMsIGFuZC9vclxuICogYXJlIHByb3RlY3RlZCBieSB0cmFkZSBzZWNyZXQgb3IgY29weXJpZ2h0IGxhdy4gRGlzc2VtaW5hdGlvbiBvZiB0aGlzIGluZm9ybWF0aW9uIG9yIHJlcHJvZHVjdGlvbiBvZiB0aGlzIG1hdGVyaWFsIGlzXG4gKiBzdHJpY3RseSBmb3JiaWRkZW4gdW5sZXNzIHByaW9yIHdyaXR0ZW4gcGVybWlzc2lvbiBpcyBJbnN0cnVtZW50YXRpb24gTGFib3JhdG9yeSBDb21wYW55LlxuICovXG5cbmV4cG9ydCBlbnVtIFN5c3RlbGFiVmlydHVhbEtleWJvYXJkQnV0dG9uIHtcbiAgRG9uZSA9ICd7ZG9uZX0nLFxuICBFbnRlciA9ICd7ZW50ZXJ9JyxcbiAgU2hpZnQgPSAne3NoaWZ0fScsXG4gIExvY2sgPSAne2xvY2t9JyxcbiAgQmFja3NwYWNlID0gJ3tia3NwfScsXG4gIExhbmd1YWdlID0gJ3tsYW5ndWFnZX0nLFxuICBTcGFjZSA9ICd7c3BhY2V9JyxcbiAgVGFiID0gJ3t0YWJ9Jyxcbn1cblxuZXhwb3J0IGVudW0gU3lzdGVsYWJWaXJ0dWFsS2V5Ym9hcmRMYXlvdXRzIHtcbiAgZGVmYXVsdCA9ICdkZWZhdWx0JyxcbiAgc2hpZnQgPSAnc2hpZnQnLFxuICBudW1lcmljID0gJ251bWVyaWMnLFxufVxuXG5leHBvcnQgZW51bSBTeXN0ZWxhYlZpcnR1YWxLZXlib2FyZElucHV0VHlwZXMge1xuICB0ZXh0ID0gJ3RleHQnLFxuICBudW1iZXIgPSAnbnVtYmVyJyxcbiAgcGFzc3dvcmQgPSAncGFzc3dvcmQnLFxufVxuIl19
@@ -0,0 +1,123 @@
1
+ import { Injectable } from '@angular/core';
2
+ import { ComponentPortal } from '@angular/cdk/portal';
3
+ import { SystelabVirtualKeyboardLayouts } from './constants';
4
+ import { SystelabVirtualKeyboardComponent } from './systelab-virtual-keyboard.component';
5
+ import * as i0 from "@angular/core";
6
+ import * as i1 from "@angular/cdk/overlay";
7
+ export class SystelabVirtualKeyboardOverlayService {
8
+ constructor(overlay) {
9
+ this.overlay = overlay;
10
+ }
11
+ isCreated() {
12
+ return !!this.overlayRef;
13
+ }
14
+ isOpen() {
15
+ return this.open;
16
+ }
17
+ create(inputOrigin, fixedBottom, layout = SystelabVirtualKeyboardLayouts.default) {
18
+ this.layout = layout;
19
+ this.overlayRef = this.overlay.create({
20
+ hasBackdrop: false,
21
+ scrollStrategy: this.overlay.scrollStrategies.reposition(),
22
+ disposeOnNavigation: true,
23
+ });
24
+ this.overlayRef.addPanelClass('virtual-keyboard-overlay-pane');
25
+ if (fixedBottom) {
26
+ this.overlayRef.addPanelClass('virtual-keyboard-fixed-bottom');
27
+ }
28
+ this.updatePositionStrategy(inputOrigin, fixedBottom);
29
+ this.updateSize();
30
+ this.open = true;
31
+ return this.overlayRef.attach(new ComponentPortal(SystelabVirtualKeyboardComponent));
32
+ }
33
+ destroy() {
34
+ if (this.overlayRef) {
35
+ this.overlayRef.dispose();
36
+ }
37
+ this.overlayRef = null;
38
+ this.open = false;
39
+ }
40
+ hasAttached() {
41
+ return this.overlayRef?.hasAttached();
42
+ }
43
+ updatePositionStrategy(inputOrigin, fixedBottom) {
44
+ this.overlayRef.updatePositionStrategy(this.getPositionStrategy(inputOrigin, fixedBottom));
45
+ }
46
+ updateSize() {
47
+ this.overlayRef.updateSize(this.getOverlaySize());
48
+ }
49
+ getPositionStrategy(inputOrigin, fixedBottom) {
50
+ if (fixedBottom) {
51
+ return this.overlay.position().global().centerHorizontally().bottom('0');
52
+ }
53
+ const pointWithDimensions = this.computePositionStrategyOrigin(inputOrigin);
54
+ return this.overlay
55
+ .position()
56
+ .flexibleConnectedTo(pointWithDimensions)
57
+ .withFlexibleDimensions(false)
58
+ .withLockedPosition(true)
59
+ .withPush(false)
60
+ .withPositions([
61
+ {
62
+ originX: 'start',
63
+ originY: 'bottom',
64
+ overlayX: 'start',
65
+ overlayY: 'top',
66
+ },
67
+ {
68
+ originX: 'start',
69
+ originY: 'top',
70
+ overlayX: 'start',
71
+ overlayY: 'bottom',
72
+ },
73
+ {
74
+ originX: 'end',
75
+ originY: 'bottom',
76
+ overlayX: 'end',
77
+ overlayY: 'top',
78
+ },
79
+ {
80
+ originX: 'end',
81
+ originY: 'top',
82
+ overlayX: 'end',
83
+ overlayY: 'bottom',
84
+ },
85
+ ]);
86
+ }
87
+ getOverlaySize() {
88
+ const overlayWidth = this.layout === SystelabVirtualKeyboardLayouts.numeric ? '400px' : '1200px';
89
+ return {
90
+ width: overlayWidth,
91
+ maxWidth: overlayWidth,
92
+ minWidth: overlayWidth,
93
+ };
94
+ }
95
+ computePositionStrategyOrigin(inputOrigin) {
96
+ const overlayOffsetX = this.computeOverlayOffsetX(inputOrigin);
97
+ const { width: overlayWidthString } = this.getOverlaySize();
98
+ const overlayWidth = Number(overlayWidthString.replace('px', ''));
99
+ const { x, y, width, height } = inputOrigin.getBoundingClientRect();
100
+ return {
101
+ width,
102
+ height,
103
+ x: width < overlayWidth ? x - overlayOffsetX : x + overlayOffsetX,
104
+ y,
105
+ };
106
+ }
107
+ computeOverlayOffsetX(inputOrigin) {
108
+ const { width: overlayWidthString } = this.getOverlaySize();
109
+ const overlayWidth = Number(overlayWidthString.replace('px', ''));
110
+ const inputWidth = inputOrigin.getBoundingClientRect().width;
111
+ const extraWidth = overlayWidth - inputWidth;
112
+ return Math.abs(extraWidth) / 2;
113
+ }
114
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.9", ngImport: i0, type: SystelabVirtualKeyboardOverlayService, deps: [{ token: i1.Overlay }], target: i0.ɵɵFactoryTarget.Injectable }); }
115
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.0.9", ngImport: i0, type: SystelabVirtualKeyboardOverlayService, providedIn: 'root' }); }
116
+ }
117
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.9", ngImport: i0, type: SystelabVirtualKeyboardOverlayService, decorators: [{
118
+ type: Injectable,
119
+ args: [{
120
+ providedIn: 'root',
121
+ }]
122
+ }], ctorParameters: () => [{ type: i1.Overlay }] });
123
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"systelab-virtual-keyboard-overlay.service.js","sourceRoot":"","sources":["../../../../projects/systelab-virtual-keyboard/src/lib/systelab-virtual-keyboard-overlay.service.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2020 - 2024 - Instrumentation Laboratory Company and Systelab Technologies, SA. All rights reserved.\n * NOTICE:  All information contained herein is and remains the property of Instrumentation Laboratory Company and its\n * affiliates, if any.  The intellectual and technical concepts contained herein are proprietary to Instrumentation\n * Laboratory Company and its affiliates and may be covered by U.S. and foreign patents and patent applications, and/or\n * are protected by trade secret or copyright law. Dissemination of this information or reproduction of this material is\n * strictly forbidden unless prior written permission is Instrumentation Laboratory Company.\n */\n\nimport { Overlay, OverlayRef, OverlaySizeConfig, PositionStrategy } from '@angular/cdk/overlay';\nimport { ComponentRef, Injectable } from '@angular/core';\nimport { ComponentPortal } from '@angular/cdk/portal';\nimport { SystelabVirtualKeyboardLayouts } from './constants';\nimport { SystelabVirtualKeyboardComponent } from './systelab-virtual-keyboard.component';\n\ninterface PositionStrategyOrigin {\n  x: number;\n  y: number;\n  width?: number;\n  height?: number;\n}\n\n@Injectable({\n  providedIn: 'root',\n})\nexport class SystelabVirtualKeyboardOverlayService {\n  private overlayRef!: OverlayRef;\n  private open: boolean;\n  private layout: SystelabVirtualKeyboardLayouts;\n\n  constructor(private readonly overlay: Overlay) {}\n\n  public isCreated(): boolean {\n    return !!this.overlayRef;\n  }\n\n  public isOpen(): boolean {\n    return this.open;\n  }\n\n  public create(\n    inputOrigin: HTMLInputElement,\n    fixedBottom: boolean,\n    layout: SystelabVirtualKeyboardLayouts = SystelabVirtualKeyboardLayouts.default,\n  ): ComponentRef<SystelabVirtualKeyboardComponent> {\n    this.layout = layout;\n    this.overlayRef = this.overlay.create({\n      hasBackdrop: false,\n      scrollStrategy: this.overlay.scrollStrategies.reposition(),\n      disposeOnNavigation: true,\n    });\n    this.overlayRef.addPanelClass('virtual-keyboard-overlay-pane');\n    if (fixedBottom) {\n      this.overlayRef.addPanelClass('virtual-keyboard-fixed-bottom');\n    }\n\n    this.updatePositionStrategy(inputOrigin, fixedBottom);\n    this.updateSize();\n\n    this.open = true;\n    return this.overlayRef.attach(new ComponentPortal(SystelabVirtualKeyboardComponent));\n  }\n\n  public destroy(): void {\n    if (this.overlayRef) {\n      this.overlayRef.dispose();\n    }\n    this.overlayRef = null;\n    this.open = false;\n  }\n\n  public hasAttached(): boolean {\n    return this.overlayRef?.hasAttached();\n  }\n\n  private updatePositionStrategy(inputOrigin: HTMLInputElement, fixedBottom: boolean): void {\n    this.overlayRef.updatePositionStrategy(this.getPositionStrategy(inputOrigin, fixedBottom));\n  }\n\n  private updateSize(): void {\n    this.overlayRef.updateSize(this.getOverlaySize());\n  }\n\n  private getPositionStrategy(inputOrigin: HTMLInputElement, fixedBottom: boolean): PositionStrategy {\n    if (fixedBottom) {\n      return this.overlay.position().global().centerHorizontally().bottom('0');\n    }\n\n    const pointWithDimensions: PositionStrategyOrigin = this.computePositionStrategyOrigin(inputOrigin);\n\n    return this.overlay\n      .position()\n      .flexibleConnectedTo(pointWithDimensions)\n      .withFlexibleDimensions(false)\n      .withLockedPosition(true)\n      .withPush(false)\n      .withPositions([\n        {\n          originX: 'start',\n          originY: 'bottom',\n          overlayX: 'start',\n          overlayY: 'top',\n        },\n        {\n          originX: 'start',\n          originY: 'top',\n          overlayX: 'start',\n          overlayY: 'bottom',\n        },\n        {\n          originX: 'end',\n          originY: 'bottom',\n          overlayX: 'end',\n          overlayY: 'top',\n        },\n        {\n          originX: 'end',\n          originY: 'top',\n          overlayX: 'end',\n          overlayY: 'bottom',\n        },\n      ]);\n  }\n\n  private getOverlaySize(): OverlaySizeConfig {\n    const overlayWidth = this.layout === SystelabVirtualKeyboardLayouts.numeric ? '400px' : '1200px';\n\n    return {\n      width: overlayWidth,\n      maxWidth: overlayWidth,\n      minWidth: overlayWidth,\n    };\n  }\n\n  private computePositionStrategyOrigin(inputOrigin: HTMLInputElement): PositionStrategyOrigin {\n    const overlayOffsetX = this.computeOverlayOffsetX(inputOrigin);\n    const { width: overlayWidthString } = this.getOverlaySize();\n    const overlayWidth = Number((overlayWidthString as string).replace('px', ''));\n    const {x, y, width, height} = inputOrigin.getBoundingClientRect();\n    return {\n      width,\n      height,\n      x: width < overlayWidth ? x - overlayOffsetX : x + overlayOffsetX,\n      y,\n    };\n  }\n\n  private computeOverlayOffsetX(inputOrigin: HTMLInputElement): number {\n    const { width: overlayWidthString } = this.getOverlaySize();\n    const overlayWidth = Number((overlayWidthString as string).replace('px', ''));\n    const inputWidth = inputOrigin.getBoundingClientRect().width;\n\n    const extraWidth = overlayWidth - inputWidth;\n\n    return Math.abs(extraWidth) / 2;\n  }\n}\n"]}
@@ -0,0 +1,352 @@
1
+ import { Component, EventEmitter, HostListener, Output } from '@angular/core';
2
+ import SimpleKeyboard from 'simple-keyboard';
3
+ import { SystelabVirtualKeyboardButton, SystelabVirtualKeyboardInputTypes, SystelabVirtualKeyboardLayouts } from './constants';
4
+ import * as i0 from "@angular/core";
5
+ export class SystelabVirtualKeyboardComponent {
6
+ handleKeyUp(event) {
7
+ if (event.isTrusted) {
8
+ this.caretEventHandler(event);
9
+ }
10
+ }
11
+ handleMouseUp(event) {
12
+ this.caretEventHandler(event);
13
+ }
14
+ handleSelect(event) {
15
+ this.caretEventHandler(event);
16
+ }
17
+ handleSelectionChange(event) {
18
+ this.caretEventHandler(event);
19
+ }
20
+ get maxLength() {
21
+ return this.activeInputElement?.maxLength ?? -1;
22
+ }
23
+ get isTextarea() {
24
+ return this.activeInputElement?.type === 'textarea';
25
+ }
26
+ constructor(elementRef) {
27
+ this.elementRef = elementRef;
28
+ this.debug = false;
29
+ this.selectedLayout = SystelabVirtualKeyboardLayouts.default;
30
+ this.caretPosition = null;
31
+ this.caretPositionEnd = null;
32
+ this.closePanel = new EventEmitter();
33
+ }
34
+ ngAfterViewInit() {
35
+ const layout = {
36
+ [SystelabVirtualKeyboardLayouts.default]: [
37
+ '` 1 2 3 4 5 6 7 8 9 0 - = {bksp}',
38
+ '{tab} q w e r t y u i o p [ ] \\',
39
+ "{lock} a s d f g h j k l ; ' {enter}",
40
+ '{shift} z x c v b n m , . / {shift}',
41
+ '{space}',
42
+ ],
43
+ [SystelabVirtualKeyboardLayouts.shift]: [
44
+ '~ ! @ # $ % ^ &amp; * ( ) _ + {bksp}',
45
+ '{tab} Q W E R T Y U I O P { } |',
46
+ '{lock} A S D F G H J K L : " {enter}',
47
+ '{shift} Z X C V B N M &lt; &gt; ? {shift}',
48
+ '{space}',
49
+ ],
50
+ [SystelabVirtualKeyboardLayouts.numeric]: ['7 8 9', '4 5 6', '1 2 3', '0 {bksp}'],
51
+ };
52
+ this.keyboard = new SimpleKeyboard('.simple-keyboard', {
53
+ onKeyPress: (button) => this.handleKeyPress(button),
54
+ mergeDisplay: true,
55
+ display: {
56
+ [SystelabVirtualKeyboardButton.Backspace]: 'delete',
57
+ },
58
+ buttonTheme: [
59
+ {
60
+ class: 'virtual-keyboard-delete-button',
61
+ buttons: `${SystelabVirtualKeyboardButton.Backspace}`,
62
+ },
63
+ ],
64
+ layout,
65
+ });
66
+ this.setLayout(this.selectedLayout);
67
+ if (this.debug) {
68
+ console.log('Layout: ', layout);
69
+ }
70
+ }
71
+ setActiveInput(input) {
72
+ this.activeInputElement = input;
73
+ if (this.debug) {
74
+ const inputType = this.activeInputElement?.type;
75
+ console.log('Layout:', `${inputType}_${this.selectedLayout}`);
76
+ }
77
+ let selectionStart;
78
+ let selectionEnd;
79
+ if (this.isInputNumeric(input)) {
80
+ selectionStart = this.activeInputElement.value.toString().length;
81
+ selectionEnd = this.activeInputElement.value.toString().length;
82
+ }
83
+ else {
84
+ selectionStart = this.activeInputElement.selectionStart;
85
+ selectionEnd = this.activeInputElement.selectionEnd;
86
+ }
87
+ this.setCaretPosition(selectionStart, selectionEnd);
88
+ if (this.debug) {
89
+ console.log('Caret start at:', this.caretPosition, this.caretPositionEnd);
90
+ }
91
+ this.focusActiveInput();
92
+ }
93
+ setLayout(layout) {
94
+ this.selectedLayout = layout;
95
+ if (this.keyboard) {
96
+ this.keyboard.setOptions({
97
+ layoutName: layout,
98
+ });
99
+ }
100
+ }
101
+ isInputAlphabetic(activeInputElement) {
102
+ const inputType = activeInputElement?.type;
103
+ return inputType && [SystelabVirtualKeyboardInputTypes.text, SystelabVirtualKeyboardInputTypes.password].some((i) => i === inputType);
104
+ }
105
+ isInputNumeric(activeInputElement) {
106
+ const inputType = activeInputElement?.type;
107
+ return inputType && [SystelabVirtualKeyboardInputTypes.number].some((i) => i === inputType);
108
+ }
109
+ handleKeyPress(button, e) {
110
+ if (this.debug) {
111
+ console.log('Key press:', button);
112
+ }
113
+ if (button === SystelabVirtualKeyboardButton.Shift || button === SystelabVirtualKeyboardButton.Lock) {
114
+ this.toggleShift();
115
+ }
116
+ else if (button === SystelabVirtualKeyboardButton.Done) {
117
+ this.closePanel.emit();
118
+ return;
119
+ }
120
+ if (!this.isAcceptedNonStandardButton(button) && !this.isStandardButton(button)) {
121
+ return;
122
+ }
123
+ const output = this.handleButtonOutput(button);
124
+ if (this.activeInputElement) {
125
+ this.activeInputElement.value = output;
126
+ if (this.debug) {
127
+ console.log('Caret at:', this.caretPosition, this.caretPositionEnd, 'Button', e);
128
+ }
129
+ }
130
+ this.dispatchEvents(button);
131
+ }
132
+ handleButtonOutput(button) {
133
+ const commonParams = this.getCommonParams();
134
+ let output = this.activeInputElement?.value || '';
135
+ if (!this.isStandardButton(button)) {
136
+ if (button === SystelabVirtualKeyboardButton.Backspace) {
137
+ output = this.removeAt(output, ...commonParams);
138
+ }
139
+ else if (button === SystelabVirtualKeyboardButton.Space) {
140
+ output = this.addStringAt(output, ' ', ...commonParams);
141
+ }
142
+ else if (button === SystelabVirtualKeyboardButton.Tab) {
143
+ output = this.addStringAt(output, '\t', ...commonParams);
144
+ }
145
+ else if (button === SystelabVirtualKeyboardButton.Enter) {
146
+ if (this.isTextarea) {
147
+ output = this.addStringAt(output, '\n', ...commonParams);
148
+ }
149
+ }
150
+ else {
151
+ return;
152
+ }
153
+ }
154
+ else {
155
+ output = this.addStringAt(output, button, ...commonParams);
156
+ }
157
+ return output;
158
+ }
159
+ getCommonParams() {
160
+ return [this.caretPosition || 0, this.caretPositionEnd || 0, true];
161
+ }
162
+ isAcceptedNonStandardButton(button) {
163
+ return [
164
+ SystelabVirtualKeyboardButton.Backspace.toString(),
165
+ SystelabVirtualKeyboardButton.Space.toString(),
166
+ SystelabVirtualKeyboardButton.Tab.toString(),
167
+ SystelabVirtualKeyboardButton.Enter.toString(),
168
+ ].includes(button);
169
+ }
170
+ dispatchEvents(button) {
171
+ const { key, code } = this.convertFromButtonToCode(button);
172
+ const eventInit = {
173
+ bubbles: true,
174
+ cancelable: true,
175
+ shiftKey: this.selectedLayout === SystelabVirtualKeyboardLayouts.shift,
176
+ key: key,
177
+ code: code,
178
+ location: 0,
179
+ };
180
+ // Simulate all needed events on base element
181
+ this.activeInputElement?.dispatchEvent(new KeyboardEvent('keydown', eventInit));
182
+ this.activeInputElement?.dispatchEvent(new KeyboardEvent('keypress', eventInit));
183
+ this.activeInputElement?.dispatchEvent(new Event('input', { bubbles: true }));
184
+ this.activeInputElement?.dispatchEvent(new KeyboardEvent('keyup', eventInit));
185
+ // And set focus to input
186
+ this.focusActiveInput();
187
+ }
188
+ /*
189
+ * AUXILIARY METHODS
190
+ * */
191
+ convertFromButtonToCode(button) {
192
+ let key;
193
+ let code;
194
+ if (button.includes('{') && button.includes('}')) {
195
+ // Capitalize name
196
+ key = button.slice(1, button.length - 1).toLowerCase();
197
+ key = key.charAt(0).toUpperCase() + key.slice(1);
198
+ code = key;
199
+ // Fix to standard key code
200
+ if (code.toLowerCase() === SystelabVirtualKeyboardButton.Backspace.slice(1, SystelabVirtualKeyboardButton.Backspace.length - 1).toLowerCase()) {
201
+ code = 'Backspace';
202
+ }
203
+ }
204
+ else {
205
+ key = button;
206
+ code = Number.isInteger(Number(button)) ? `Digit${button}` : `Key${button.toUpperCase()}`;
207
+ }
208
+ return { key, code };
209
+ }
210
+ focusActiveInput() {
211
+ this.activeInputElement?.focus();
212
+ if (!this.isInputNumeric(this.activeInputElement)) {
213
+ this.activeInputElement?.setSelectionRange(this.caretPosition, this.caretPositionEnd);
214
+ }
215
+ }
216
+ toggleShift() {
217
+ let currentLayout = this.keyboard.options.layoutName;
218
+ let selectedLayout = currentLayout === SystelabVirtualKeyboardLayouts.default ? SystelabVirtualKeyboardLayouts.shift : SystelabVirtualKeyboardLayouts.default;
219
+ this.setLayout(selectedLayout);
220
+ }
221
+ isStandardButton(button) {
222
+ return button && !(button[0] === '{' && button[button.length - 1] === '}');
223
+ }
224
+ /*
225
+ * OUTPUT STRING METHODS
226
+ * */
227
+ removeAt(source, position = source.length, positionEnd = source.length, moveCaret = false) {
228
+ if (position === 0 && positionEnd === 0) {
229
+ return source;
230
+ }
231
+ let output;
232
+ if (position === positionEnd) {
233
+ if (position && position >= 0) {
234
+ output = source.slice(0, position - 1) + source.slice(position);
235
+ if (moveCaret) {
236
+ this.updateCaretPosition(1, true);
237
+ }
238
+ }
239
+ else {
240
+ output = source.slice(0, -1);
241
+ if (moveCaret) {
242
+ this.updateCaretPosition(1, true);
243
+ }
244
+ }
245
+ }
246
+ else {
247
+ output = source.slice(0, position) + source.slice(positionEnd);
248
+ if (moveCaret) {
249
+ this.setCaretPosition(position);
250
+ }
251
+ }
252
+ return output;
253
+ }
254
+ addStringAt(source, str, position = source.length, positionEnd = source.length, moveCaret = false) {
255
+ if (this.maxLength !== -1 && source.length >= this.maxLength) {
256
+ return source;
257
+ }
258
+ let output;
259
+ if (!position && position !== 0) {
260
+ output = source + str;
261
+ }
262
+ else {
263
+ output = [source.slice(0, position), str, source.slice(positionEnd)].join('');
264
+ if (moveCaret) {
265
+ this.updateCaretPosition(str.length, false);
266
+ }
267
+ }
268
+ return output;
269
+ }
270
+ /*
271
+ * CARET METHODS
272
+ * */
273
+ caretEventHandler(event) {
274
+ let targetTagName = '';
275
+ if (event.target.tagName) {
276
+ targetTagName = event.target.tagName.toLowerCase();
277
+ }
278
+ const isTextInput = targetTagName === 'textarea' ||
279
+ (targetTagName === 'input' && ['text', 'search', 'email', 'password', 'url', 'tel'].includes(event.target.type));
280
+ const isKeyboard = event.target === this.elementRef.nativeElement || (event.target && this.elementRef.nativeElement.contains(event.target));
281
+ if (isTextInput && this.activeInputElement == event.target) {
282
+ /**
283
+ * Tracks current cursor position
284
+ * As keys are pressed, text will be added/removed at that position within the input.
285
+ */
286
+ this.setCaretPosition(event.target.selectionStart, event.target.selectionEnd);
287
+ if (this.debug) {
288
+ console.log('Caret at:', this.caretPosition, this.caretPositionEnd, event && event.target.tagName.toLowerCase(), event);
289
+ }
290
+ }
291
+ else if (event.type === 'pointerup' && this.activeInputElement === document.activeElement) {
292
+ return;
293
+ }
294
+ else if (!isKeyboard && event?.type !== 'selectionchange') {
295
+ /**
296
+ * we must ensure caretPosition doesn't persist once reactivated.
297
+ */
298
+ this.setCaretPosition(null);
299
+ if (this.debug) {
300
+ console.log(`Caret position reset due to "${event?.type}" event`, event);
301
+ }
302
+ }
303
+ }
304
+ updateCaretPosition(length, minus = false) {
305
+ const newCaretPos = this.computeNewCaretPosition(length, minus);
306
+ this.setCaretPosition(newCaretPos);
307
+ // Scroll to bottom
308
+ setTimeout(() => {
309
+ this.activeInputElement?.scrollTo({
310
+ top: this.activeInputElement.scrollHeight,
311
+ });
312
+ });
313
+ }
314
+ computeNewCaretPosition(length, minus = false) {
315
+ let caretPosition = this.caretPosition;
316
+ if (caretPosition != null) {
317
+ if (minus) {
318
+ if (caretPosition > 0)
319
+ caretPosition = caretPosition - length;
320
+ }
321
+ else {
322
+ caretPosition = caretPosition + length;
323
+ }
324
+ }
325
+ return caretPosition;
326
+ }
327
+ setCaretPosition(position, endPosition = position) {
328
+ this.caretPosition = position;
329
+ this.caretPositionEnd = endPosition;
330
+ }
331
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.9", ngImport: i0, type: SystelabVirtualKeyboardComponent, deps: [{ token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component }); }
332
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.0.9", type: SystelabVirtualKeyboardComponent, isStandalone: true, selector: "systelab-virtual-keyboard.component", outputs: { closePanel: "closePanel" }, host: { listeners: { "window:keyup": "handleKeyUp($event)", "window:pointerup": "handleMouseUp($event)", "window:select": "handleSelect($event)", "window:selectionchange": "handleSelectionChange($event)" } }, ngImport: i0, template: "<!--\n ~ Copyright (c) 2020 - 2024 - Instrumentation Laboratory Company and Systelab Technologies, SA. All rights reserved.\n ~ NOTICE: All information contained herein is and remains the property of Instrumentation Laboratory Company and its\n ~ affiliates, if any. The intellectual and technical concepts contained herein are proprietary to Instrumentation\n ~ Laboratory Company and its affiliates and may be covered by U.S. and foreign patents and patent applications, and/or\n ~ are protected by trade secret or copyright law. Dissemination of this information or reproduction of this material is\n ~ strictly forbidden unless prior written permission is Instrumentation Laboratory Company.\n -->\n\n<div class=\"simple-keyboard\"></div>\n", styles: [":host{width:100%}\n"] }); }
333
+ }
334
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.9", ngImport: i0, type: SystelabVirtualKeyboardComponent, decorators: [{
335
+ type: Component,
336
+ args: [{ selector: 'systelab-virtual-keyboard.component', standalone: true, imports: [], template: "<!--\n ~ Copyright (c) 2020 - 2024 - Instrumentation Laboratory Company and Systelab Technologies, SA. All rights reserved.\n ~ NOTICE: All information contained herein is and remains the property of Instrumentation Laboratory Company and its\n ~ affiliates, if any. The intellectual and technical concepts contained herein are proprietary to Instrumentation\n ~ Laboratory Company and its affiliates and may be covered by U.S. and foreign patents and patent applications, and/or\n ~ are protected by trade secret or copyright law. Dissemination of this information or reproduction of this material is\n ~ strictly forbidden unless prior written permission is Instrumentation Laboratory Company.\n -->\n\n<div class=\"simple-keyboard\"></div>\n", styles: [":host{width:100%}\n"] }]
337
+ }], ctorParameters: () => [{ type: i0.ElementRef }], propDecorators: { handleKeyUp: [{
338
+ type: HostListener,
339
+ args: ['window:keyup', ['$event']]
340
+ }], handleMouseUp: [{
341
+ type: HostListener,
342
+ args: ['window:pointerup', ['$event']]
343
+ }], handleSelect: [{
344
+ type: HostListener,
345
+ args: ['window:select', ['$event']]
346
+ }], handleSelectionChange: [{
347
+ type: HostListener,
348
+ args: ['window:selectionchange', ['$event']]
349
+ }], closePanel: [{
350
+ type: Output
351
+ }] } });
352
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"systelab-virtual-keyboard.component.js","sourceRoot":"","sources":["../../../../projects/systelab-virtual-keyboard/src/lib/systelab-virtual-keyboard.component.ts","../../../../projects/systelab-virtual-keyboard/src/lib/systelab-virtual-keyboard.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, Component, ElementRef, EventEmitter, HostListener, Output } from '@angular/core';\nimport SimpleKeyboard from 'simple-keyboard';\nimport {\n  SystelabVirtualKeyboardButton,\n  SystelabVirtualKeyboardInputTypes,\n  SystelabVirtualKeyboardLayouts\n} from './constants';\n\n@Component({\n  selector: 'systelab-virtual-keyboard.component',\n  standalone: true,\n  imports: [],\n  templateUrl: './systelab-virtual-keyboard.component.html',\n  styleUrl: 'systelab-virtual-keyboard.component.scss',\n})\nexport class SystelabVirtualKeyboardComponent implements AfterViewInit {\n  @HostListener('window:keyup', ['$event'])\n  handleKeyUp(event: KeyboardEvent): void {\n    if (event.isTrusted) {\n      this.caretEventHandler(event);\n    }\n  }\n\n  @HostListener('window:pointerup', ['$event'])\n  handleMouseUp(event: PointerEvent): void {\n    this.caretEventHandler(event);\n  }\n\n  @HostListener('window:select', ['$event'])\n  handleSelect(event: Event): void {\n    this.caretEventHandler(event);\n  }\n\n  @HostListener('window:selectionchange', ['$event'])\n  handleSelectionChange(event: Event): void {\n    this.caretEventHandler(event);\n  }\n\n  get maxLength(): number {\n    return this.activeInputElement?.maxLength ?? -1;\n  }\n\n  get isTextarea(): boolean {\n    return this.activeInputElement?.type === 'textarea';\n  }\n\n  public debug = false;\n  private selectedLayout: SystelabVirtualKeyboardLayouts = SystelabVirtualKeyboardLayouts.default;\n  private keyboard: SimpleKeyboard;\n  private caretPosition: number | null = null;\n  private caretPositionEnd: number | null = null;\n  private activeInputElement!: HTMLInputElement | HTMLTextAreaElement | null;\n\n  @Output() closePanel = new EventEmitter<void>();\n\n  constructor(private elementRef: ElementRef<HTMLInputElement>) {}\n\n  ngAfterViewInit() {\n    const layout = {\n      [SystelabVirtualKeyboardLayouts.default]: [\n        '` 1 2 3 4 5 6 7 8 9 0 - = {bksp}',\n        '{tab} q w e r t y u i o p [ ] \\\\',\n        \"{lock} a s d f g h j k l ; ' {enter}\",\n        '{shift} z x c v b n m , . / {shift}',\n        '{space}',\n      ],\n      [SystelabVirtualKeyboardLayouts.shift]: [\n        '~ ! @ # $ % ^ &amp; * ( ) _ + {bksp}',\n        '{tab} Q W E R T Y U I O P { } |',\n        '{lock} A S D F G H J K L : \" {enter}',\n        '{shift} Z X C V B N M &lt; &gt; ? {shift}',\n        '{space}',\n      ],\n      [SystelabVirtualKeyboardLayouts.numeric]: ['7 8 9', '4 5 6', '1 2 3', '0 {bksp}'],\n    };\n    this.keyboard = new SimpleKeyboard('.simple-keyboard', {\n      onKeyPress: (button) => this.handleKeyPress(button),\n      mergeDisplay: true,\n      display: {\n        [SystelabVirtualKeyboardButton.Backspace]: 'delete',\n      },\n      buttonTheme: [\n        {\n          class: 'virtual-keyboard-delete-button',\n          buttons: `${SystelabVirtualKeyboardButton.Backspace}`,\n        },\n      ],\n      layout,\n    });\n    this.setLayout(this.selectedLayout);\n    if (this.debug) {\n      console.log('Layout: ', layout);\n    }\n  }\n\n  public setActiveInput(input: HTMLInputElement | HTMLTextAreaElement): void {\n    this.activeInputElement = input;\n\n    if (this.debug) {\n      const inputType = this.activeInputElement?.type;\n      console.log('Layout:', `${inputType}_${this.selectedLayout}`);\n    }\n\n    let selectionStart: number;\n    let selectionEnd: number;\n    if (this.isInputNumeric(input)) {\n      selectionStart = this.activeInputElement.value.toString().length;\n      selectionEnd = this.activeInputElement.value.toString().length;\n    } else {\n      selectionStart = this.activeInputElement.selectionStart;\n      selectionEnd = this.activeInputElement.selectionEnd;\n    }\n\n    this.setCaretPosition(selectionStart, selectionEnd);\n\n    if (this.debug) {\n      console.log('Caret start at:', this.caretPosition, this.caretPositionEnd);\n    }\n    this.focusActiveInput();\n  }\n\n  public setLayout(layout: SystelabVirtualKeyboardLayouts): void {\n    this.selectedLayout = layout;\n    if (this.keyboard) {\n      this.keyboard.setOptions({\n        layoutName: layout,\n      });\n    }\n  }\n\n  private isInputAlphabetic(activeInputElement: HTMLInputElement | HTMLTextAreaElement): boolean {\n    const inputType = activeInputElement?.type;\n    return inputType && [SystelabVirtualKeyboardInputTypes.text, SystelabVirtualKeyboardInputTypes.password].some((i) => i === inputType);\n  }\n\n  private isInputNumeric(activeInputElement: HTMLInputElement | HTMLTextAreaElement): boolean {\n    const inputType = activeInputElement?.type;\n    return inputType && [SystelabVirtualKeyboardInputTypes.number].some((i) => i === inputType);\n  }\n\n  private handleKeyPress(button: string, e?: Event): void {\n    if (this.debug) {\n      console.log('Key press:', button);\n    }\n\n    if (button === SystelabVirtualKeyboardButton.Shift || button === SystelabVirtualKeyboardButton.Lock) {\n      this.toggleShift();\n    } else if (button === SystelabVirtualKeyboardButton.Done) {\n      this.closePanel.emit();\n      return;\n    }\n\n    if (!this.isAcceptedNonStandardButton(button) && !this.isStandardButton(button)) {\n      return;\n    }\n\n    const output = this.handleButtonOutput(button);\n\n    if (this.activeInputElement) {\n      this.activeInputElement.value = output;\n\n      if (this.debug) {\n        console.log('Caret at:', this.caretPosition, this.caretPositionEnd, 'Button', e);\n      }\n    }\n\n    this.dispatchEvents(button);\n  }\n\n  private handleButtonOutput(button: string): string {\n    const commonParams: [number, number, boolean] = this.getCommonParams();\n    let output = this.activeInputElement?.value || '';\n    if (!this.isStandardButton(button)) {\n      if (button === SystelabVirtualKeyboardButton.Backspace) {\n        output = this.removeAt(output, ...commonParams);\n      } else if (button === SystelabVirtualKeyboardButton.Space) {\n        output = this.addStringAt(output, ' ', ...commonParams);\n      } else if (button === SystelabVirtualKeyboardButton.Tab) {\n        output = this.addStringAt(output, '\\t', ...commonParams);\n      } else if (button === SystelabVirtualKeyboardButton.Enter) {\n        if (this.isTextarea) {\n          output = this.addStringAt(output, '\\n', ...commonParams);\n        }\n      } else {\n        return;\n      }\n    } else {\n      output = this.addStringAt(output, button, ...commonParams);\n    }\n\n    return output;\n  }\n\n  private getCommonParams(): [number, number, boolean] {\n    return [this.caretPosition || 0, this.caretPositionEnd || 0, true];\n  }\n\n  private isAcceptedNonStandardButton(button: string): boolean {\n    return [\n      SystelabVirtualKeyboardButton.Backspace.toString(),\n      SystelabVirtualKeyboardButton.Space.toString(),\n      SystelabVirtualKeyboardButton.Tab.toString(),\n      SystelabVirtualKeyboardButton.Enter.toString(),\n    ].includes(button);\n  }\n\n  private dispatchEvents(button: string) {\n    const { key, code } = this.convertFromButtonToCode(button);\n\n    const eventInit: KeyboardEventInit = {\n      bubbles: true,\n      cancelable: true,\n      shiftKey: this.selectedLayout === SystelabVirtualKeyboardLayouts.shift,\n      key: key,\n      code: code,\n      location: 0,\n    };\n\n    // Simulate all needed events on base element\n    this.activeInputElement?.dispatchEvent(new KeyboardEvent('keydown', eventInit));\n    this.activeInputElement?.dispatchEvent(new KeyboardEvent('keypress', eventInit));\n    this.activeInputElement?.dispatchEvent(new Event('input', { bubbles: true }));\n    this.activeInputElement?.dispatchEvent(new KeyboardEvent('keyup', eventInit));\n\n    // And set focus to input\n    this.focusActiveInput();\n  }\n\n  /*\n   * AUXILIARY METHODS\n   * */\n  private convertFromButtonToCode(button: string): { key: string; code: string } {\n    let key: string;\n    let code: string;\n    if (button.includes('{') && button.includes('}')) {\n      // Capitalize name\n      key = button.slice(1, button.length - 1).toLowerCase();\n      key = key.charAt(0).toUpperCase() + key.slice(1);\n      code = key;\n\n      // Fix to standard key code\n      if (code.toLowerCase() === SystelabVirtualKeyboardButton.Backspace.slice(1, SystelabVirtualKeyboardButton.Backspace.length - 1).toLowerCase()) {\n        code = 'Backspace';\n      }\n    } else {\n      key = button;\n      code = Number.isInteger(Number(button)) ? `Digit${button}` : `Key${button.toUpperCase()}`;\n    }\n\n    return { key, code };\n  }\n\n  private focusActiveInput(): void {\n    this.activeInputElement?.focus();\n    if (!this.isInputNumeric(this.activeInputElement)) {\n      this.activeInputElement?.setSelectionRange(this.caretPosition, this.caretPositionEnd);\n    }\n  }\n\n  private toggleShift() {\n    let currentLayout = this.keyboard.options.layoutName;\n    let selectedLayout: SystelabVirtualKeyboardLayouts =\n      currentLayout === SystelabVirtualKeyboardLayouts.default ? SystelabVirtualKeyboardLayouts.shift : SystelabVirtualKeyboardLayouts.default;\n\n    this.setLayout(selectedLayout);\n  }\n\n  private isStandardButton(button: string) {\n    return button && !(button[0] === '{' && button[button.length - 1] === '}');\n  }\n\n  /*\n   * OUTPUT STRING METHODS\n   * */\n  private removeAt(source: string, position = source.length, positionEnd = source.length, moveCaret = false): string {\n    if (position === 0 && positionEnd === 0) {\n      return source;\n    }\n\n    let output: string;\n\n    if (position === positionEnd) {\n      if (position && position >= 0) {\n        output = source.slice(0, position - 1) + source.slice(position);\n        if (moveCaret) {\n          this.updateCaretPosition(1, true);\n        }\n      } else {\n        output = source.slice(0, -1);\n        if (moveCaret) {\n          this.updateCaretPosition(1, true);\n        }\n      }\n    } else {\n      output = source.slice(0, position) + source.slice(positionEnd);\n      if (moveCaret) {\n        this.setCaretPosition(position);\n      }\n    }\n\n    return output;\n  }\n\n  private addStringAt(source: string, str: string, position = source.length, positionEnd = source.length, moveCaret = false): string {\n    if (this.maxLength !== -1 && source.length >= this.maxLength) {\n      return source;\n    }\n\n    let output: string;\n\n    if (!position && position !== 0) {\n      output = source + str;\n    } else {\n      output = [source.slice(0, position), str, source.slice(positionEnd)].join('');\n      if (moveCaret) {\n        this.updateCaretPosition(str.length, false);\n      }\n    }\n\n    return output;\n  }\n\n  /*\n   * CARET METHODS\n   * */\n  private caretEventHandler(event: any) {\n    let targetTagName = '';\n    if (event.target.tagName) {\n      targetTagName = event.target.tagName.toLowerCase();\n    }\n\n    const isTextInput =\n      targetTagName === 'textarea' ||\n      (targetTagName === 'input' && ['text', 'search', 'email', 'password', 'url', 'tel'].includes(event.target.type));\n\n    const isKeyboard =\n      event.target === this.elementRef.nativeElement || (event.target && this.elementRef.nativeElement.contains(event.target));\n\n    if (isTextInput && this.activeInputElement == event.target) {\n      /**\n       * Tracks current cursor position\n       * As keys are pressed, text will be added/removed at that position within the input.\n       */\n      this.setCaretPosition(event.target.selectionStart, event.target.selectionEnd);\n\n      if (this.debug) {\n        console.log('Caret at:', this.caretPosition, this.caretPositionEnd, event && event.target.tagName.toLowerCase(), event);\n      }\n    } else if (event.type === 'pointerup' && this.activeInputElement === document.activeElement) {\n      return;\n    } else if (!isKeyboard && event?.type !== 'selectionchange') {\n      /**\n       * we must ensure caretPosition doesn't persist once reactivated.\n       */\n      this.setCaretPosition(null);\n\n      if (this.debug) {\n        console.log(`Caret position reset due to \"${event?.type}\" event`, event);\n      }\n    }\n  }\n\n  private updateCaretPosition(length: number, minus = false) {\n    const newCaretPos = this.computeNewCaretPosition(length, minus);\n    this.setCaretPosition(newCaretPos);\n    // Scroll to bottom\n    setTimeout(() => {\n      this.activeInputElement?.scrollTo({\n        top: this.activeInputElement.scrollHeight,\n      } as ScrollToOptions);\n    });\n  }\n\n  private computeNewCaretPosition(length: number, minus = false) {\n    let caretPosition = this.caretPosition;\n\n    if (caretPosition != null) {\n      if (minus) {\n        if (caretPosition > 0) caretPosition = caretPosition - length;\n      } else {\n        caretPosition = caretPosition + length;\n      }\n    }\n    return caretPosition;\n  }\n\n  private setCaretPosition(position: number | null, endPosition = position): void {\n    this.caretPosition = position;\n    this.caretPositionEnd = endPosition;\n  }\n}\n","<!--\n  ~ Copyright (c) 2020 - 2024 - Instrumentation Laboratory Company and Systelab Technologies, SA. All rights reserved.\n  ~ NOTICE:  All information contained herein is and remains the property of Instrumentation Laboratory Company and its\n  ~ affiliates, if any.  The intellectual and technical concepts contained herein are proprietary to Instrumentation\n  ~ Laboratory Company and its affiliates and may be covered by U.S. and foreign patents and patent applications, and/or\n  ~ are protected by trade secret or copyright law. Dissemination of this information or reproduction of this material is\n  ~ strictly forbidden unless prior written permission is Instrumentation Laboratory Company.\n  -->\n\n<div class=\"simple-keyboard\"></div>\n"]}
@@ -0,0 +1,3 @@
1
+ import { InjectionToken } from '@angular/core';
2
+ export const VIRTUAL_KEYBOARD_CONFIG = new InjectionToken('VIRTUAL_KEYBOARD_CONFIG');
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3lzdGVsYWItdmlydHVhbC1rZXlib2FyZC5jb25maWcuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9zeXN0ZWxhYi12aXJ0dWFsLWtleWJvYXJkL3NyYy9saWIvc3lzdGVsYWItdmlydHVhbC1rZXlib2FyZC5jb25maWcudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUcvQyxNQUFNLENBQUMsTUFBTSx1QkFBdUIsR0FBRyxJQUFJLGNBQWMsQ0FBZ0MseUJBQXlCLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEluamVjdGlvblRva2VuIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBTeXN0ZWxhYlZpcnR1YWxLZXlib2FyZExheW91dHMgfSBmcm9tICcuL2NvbnN0YW50cyc7XG5cbmV4cG9ydCBjb25zdCBWSVJUVUFMX0tFWUJPQVJEX0NPTkZJRyA9IG5ldyBJbmplY3Rpb25Ub2tlbjxTeXN0ZWxhYlZpcnR1YWxLZXlib2FyZENvbmZpZz4oJ1ZJUlRVQUxfS0VZQk9BUkRfQ09ORklHJyk7XG5cbmV4cG9ydCBpbnRlcmZhY2UgU3lzdGVsYWJWaXJ0dWFsS2V5Ym9hcmRDb25maWcge1xuICBsYXlvdXQ/OiBTeXN0ZWxhYlZpcnR1YWxLZXlib2FyZExheW91dHM7XG59XG4iXX0=