@tiba-spark/client-shared-lib 25.3.0-97 → 25.3.0-98
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/esm2022/libraries/directives/dynamic-ellipsis-tooltip.directive.mjs +149 -0
- package/esm2022/libraries/directives/index.mjs +2 -1
- package/fesm2022/tiba-spark-client-shared-lib.mjs +260 -116
- package/fesm2022/tiba-spark-client-shared-lib.mjs.map +1 -1
- package/libraries/directives/dynamic-ellipsis-tooltip.directive.d.ts +30 -0
- package/libraries/directives/dynamic-ellipsis-tooltip.directive.d.ts.map +1 -0
- package/libraries/directives/index.d.ts +1 -0
- package/libraries/directives/index.d.ts.map +1 -1
- package/package.json +1 -1
|
@@ -0,0 +1,149 @@
|
|
|
1
|
+
import { Directive, HostListener, Input } from '@angular/core';
|
|
2
|
+
import { MatTooltipModule } from '@angular/material/tooltip';
|
|
3
|
+
import { findParentElement } from '../utils/element.util';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
import * as i1 from "@angular/material/tooltip";
|
|
6
|
+
export class DynamicEllipsisTooltipDirective {
|
|
7
|
+
constructor(matTooltip, elementRef, renderer) {
|
|
8
|
+
this.matTooltip = matTooltip;
|
|
9
|
+
this.elementRef = elementRef;
|
|
10
|
+
this.renderer = renderer;
|
|
11
|
+
this.originalPlaceholder = '';
|
|
12
|
+
this.truncatedPlaceholder = '';
|
|
13
|
+
}
|
|
14
|
+
ngAfterViewInit() {
|
|
15
|
+
if (this.querySelector?.length > 0) {
|
|
16
|
+
this.element = this.elementRef.nativeElement.querySelector(this.querySelector);
|
|
17
|
+
}
|
|
18
|
+
else if (this.findParentElementClass?.length > 0) {
|
|
19
|
+
this.element = findParentElement(this.elementRef, this.findParentElementClass);
|
|
20
|
+
}
|
|
21
|
+
else {
|
|
22
|
+
this.element = this.elementRef.nativeElement;
|
|
23
|
+
}
|
|
24
|
+
this.originalPlaceholder = this.element['placeholder'];
|
|
25
|
+
if (this.originalPlaceholder?.length > 0) {
|
|
26
|
+
this.setupListeners();
|
|
27
|
+
this.setupResizeObserver();
|
|
28
|
+
this.calculateTruncatedPlaceholder();
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
ngOnDestroy() {
|
|
32
|
+
if (this.resizeObserver) {
|
|
33
|
+
this.resizeObserver.disconnect();
|
|
34
|
+
}
|
|
35
|
+
if (this.focusListener) {
|
|
36
|
+
this.element.removeEventListener('focus', this.focusListener);
|
|
37
|
+
}
|
|
38
|
+
if (this.blurListener) {
|
|
39
|
+
this.element.removeEventListener('blur', this.blurListener);
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
setupListeners() {
|
|
43
|
+
const input = this.element;
|
|
44
|
+
this.focusListener = () => {
|
|
45
|
+
// On focus, use truncated placeholder if overflow exists
|
|
46
|
+
if (this.isPlaceholderOverflowing()) {
|
|
47
|
+
this.renderer.setAttribute(input, 'placeholder', this.truncatedPlaceholder);
|
|
48
|
+
}
|
|
49
|
+
};
|
|
50
|
+
this.blurListener = () => {
|
|
51
|
+
this.renderer.setAttribute(input, 'placeholder', this.truncatedPlaceholder);
|
|
52
|
+
};
|
|
53
|
+
input.addEventListener('focus', this.focusListener);
|
|
54
|
+
input.addEventListener('blur', this.blurListener);
|
|
55
|
+
}
|
|
56
|
+
calculateTruncatedPlaceholder() {
|
|
57
|
+
if (!this.originalPlaceholder)
|
|
58
|
+
return;
|
|
59
|
+
const input = this.element;
|
|
60
|
+
const maxWidth = this.getAvailableWidth();
|
|
61
|
+
if (this.getTextWidth(this.originalPlaceholder) <= maxWidth) {
|
|
62
|
+
this.truncatedPlaceholder = this.originalPlaceholder;
|
|
63
|
+
this.renderer.setAttribute(input, 'placeholder', this.truncatedPlaceholder);
|
|
64
|
+
return;
|
|
65
|
+
}
|
|
66
|
+
// Binary search for optimal length
|
|
67
|
+
let left = 0;
|
|
68
|
+
let right = this.originalPlaceholder.length;
|
|
69
|
+
let bestFit = '';
|
|
70
|
+
while (left <= right) {
|
|
71
|
+
const mid = Math.floor((left + right) / 2);
|
|
72
|
+
const testText = this.originalPlaceholder.substring(0, mid) + '...';
|
|
73
|
+
const textWidth = this.getTextWidth(testText);
|
|
74
|
+
if (textWidth <= maxWidth) {
|
|
75
|
+
bestFit = testText;
|
|
76
|
+
left = mid + 1;
|
|
77
|
+
}
|
|
78
|
+
else {
|
|
79
|
+
right = mid - 1;
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
this.truncatedPlaceholder = bestFit || this.originalPlaceholder.substring(0, 10) + '...';
|
|
83
|
+
// Apply the truncated placeholder immediately
|
|
84
|
+
this.renderer.setAttribute(input, 'placeholder', this.truncatedPlaceholder);
|
|
85
|
+
}
|
|
86
|
+
getAvailableWidth() {
|
|
87
|
+
const input = this.element;
|
|
88
|
+
const computedStyle = window.getComputedStyle(input);
|
|
89
|
+
const paddingLeft = parseFloat(computedStyle.paddingLeft) || 0;
|
|
90
|
+
const paddingRight = parseFloat(computedStyle.paddingRight) || 0;
|
|
91
|
+
return input.clientWidth - paddingLeft - paddingRight;
|
|
92
|
+
}
|
|
93
|
+
getTextWidth(text) {
|
|
94
|
+
const input = this.element;
|
|
95
|
+
const computedStyle = window.getComputedStyle(input);
|
|
96
|
+
const canvas = document.createElement('canvas');
|
|
97
|
+
const context = canvas.getContext('2d');
|
|
98
|
+
context.font = `${computedStyle.fontSize} ${computedStyle.fontFamily}`;
|
|
99
|
+
return context.measureText(text).width;
|
|
100
|
+
}
|
|
101
|
+
isPlaceholderOverflowing() {
|
|
102
|
+
const textWidth = this.getTextWidth(this.originalPlaceholder);
|
|
103
|
+
const availableWidth = this.getAvailableWidth();
|
|
104
|
+
return textWidth > availableWidth;
|
|
105
|
+
}
|
|
106
|
+
setupResizeObserver() {
|
|
107
|
+
if ('ResizeObserver' in window) {
|
|
108
|
+
this.resizeObserver = new ResizeObserver(() => {
|
|
109
|
+
this.calculateTruncatedPlaceholder();
|
|
110
|
+
});
|
|
111
|
+
this.resizeObserver.observe(this.element);
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
onMouseOver() {
|
|
115
|
+
if (this.element) {
|
|
116
|
+
this.checkPlaceholderOverflow();
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
checkPlaceholderOverflow() {
|
|
120
|
+
if (this.originalPlaceholder?.length > 0) {
|
|
121
|
+
this.matTooltip.disabled = !this.isPlaceholderOverflowing();
|
|
122
|
+
}
|
|
123
|
+
else {
|
|
124
|
+
this.matTooltip.disabled = this.element.scrollWidth <= this.element.clientWidth;
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DynamicEllipsisTooltipDirective, deps: [{ token: i1.MatTooltip }, { token: i0.ElementRef }, { token: i0.Renderer2 }], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
128
|
+
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.13", type: DynamicEllipsisTooltipDirective, isStandalone: true, selector: "[matTooltip][querySelector]", inputs: { querySelector: "querySelector", findParentElementClass: "findParentElementClass" }, host: { listeners: { "mouseover": "onMouseOver()" } }, providers: [
|
|
129
|
+
MatTooltipModule,
|
|
130
|
+
], ngImport: i0 }); }
|
|
131
|
+
}
|
|
132
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DynamicEllipsisTooltipDirective, decorators: [{
|
|
133
|
+
type: Directive,
|
|
134
|
+
args: [{
|
|
135
|
+
selector: '[matTooltip][querySelector]',
|
|
136
|
+
standalone: true,
|
|
137
|
+
providers: [
|
|
138
|
+
MatTooltipModule,
|
|
139
|
+
]
|
|
140
|
+
}]
|
|
141
|
+
}], ctorParameters: () => [{ type: i1.MatTooltip }, { type: i0.ElementRef }, { type: i0.Renderer2 }], propDecorators: { querySelector: [{
|
|
142
|
+
type: Input
|
|
143
|
+
}], findParentElementClass: [{
|
|
144
|
+
type: Input
|
|
145
|
+
}], onMouseOver: [{
|
|
146
|
+
type: HostListener,
|
|
147
|
+
args: ['mouseover']
|
|
148
|
+
}] } });
|
|
149
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dynamic-ellipsis-tooltip.directive.js","sourceRoot":"","sources":["../../../../../projects/client-shared-lib/src/libraries/directives/dynamic-ellipsis-tooltip.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, Directive, ElementRef, HostListener, Input, OnDestroy, Renderer2 } from '@angular/core';\nimport { MatTooltip, MatTooltipModule } from '@angular/material/tooltip';\nimport { findParentElement } from '../utils/element.util';\n\n@Directive({\n  selector: '[matTooltip][querySelector]',\n  standalone: true,\n  providers: [\n    MatTooltipModule,\n  ]\n})\nexport class DynamicEllipsisTooltipDirective implements AfterViewInit, OnDestroy {\n\n  @Input() querySelector: string;\n  @Input() findParentElementClass: string;\n\n  private element: HTMLElement;\n\n  private originalPlaceholder: string = '';\n  private truncatedPlaceholder: string = '';\n  private resizeObserver?: ResizeObserver;\n  private focusListener?: () => void;\n  private blurListener?: () => void;\n\n  constructor(private matTooltip: MatTooltip, private elementRef: ElementRef<HTMLElement>, private renderer: Renderer2) { }\n\n  public ngAfterViewInit(): void {\n    if (this.querySelector?.length > 0) {\n      this.element = this.elementRef.nativeElement.querySelector(this.querySelector);\n    } else if (this.findParentElementClass?.length > 0) {\n      this.element = findParentElement(this.elementRef, this.findParentElementClass);\n    } else {\n      this.element = this.elementRef.nativeElement;\n    }\n\n    this.originalPlaceholder = this.element['placeholder'];\n    if (this.originalPlaceholder?.length > 0) {\n      this.setupListeners();\n      this.setupResizeObserver();\n      this.calculateTruncatedPlaceholder();\n    }\n  }\n\n  ngOnDestroy() {\n    if (this.resizeObserver) {\n      this.resizeObserver.disconnect();\n    }\n    if (this.focusListener) {\n      this.element.removeEventListener('focus', this.focusListener);\n    }\n    if (this.blurListener) {\n      this.element.removeEventListener('blur', this.blurListener);\n    }\n  }\n\n  private setupListeners() {\n    const input = this.element;\n\n    this.focusListener = () => {\n      // On focus, use truncated placeholder if overflow exists\n      if (this.isPlaceholderOverflowing()) {\n        this.renderer.setAttribute(input, 'placeholder', this.truncatedPlaceholder);\n      }\n    };\n\n    this.blurListener = () => {\n      this.renderer.setAttribute(input, 'placeholder', this.truncatedPlaceholder);\n    };\n\n    input.addEventListener('focus', this.focusListener);\n    input.addEventListener('blur', this.blurListener);\n  }\n\n  private calculateTruncatedPlaceholder() {\n    if (!this.originalPlaceholder) return;\n\n    const input = this.element;\n    const maxWidth = this.getAvailableWidth();\n\n    if (this.getTextWidth(this.originalPlaceholder) <= maxWidth) {\n      this.truncatedPlaceholder = this.originalPlaceholder;\n      this.renderer.setAttribute(input, 'placeholder', this.truncatedPlaceholder);\n      return;\n    }\n\n    // Binary search for optimal length\n    let left = 0;\n    let right = this.originalPlaceholder.length;\n    let bestFit = '';\n\n    while (left <= right) {\n      const mid = Math.floor((left + right) / 2);\n      const testText = this.originalPlaceholder.substring(0, mid) + '...';\n      const textWidth = this.getTextWidth(testText);\n\n      if (textWidth <= maxWidth) {\n        bestFit = testText;\n        left = mid + 1;\n      } else {\n        right = mid - 1;\n      }\n    }\n\n    this.truncatedPlaceholder = bestFit || this.originalPlaceholder.substring(0, 10) + '...';\n\n    // Apply the truncated placeholder immediately\n    this.renderer.setAttribute(input, 'placeholder', this.truncatedPlaceholder);\n  }\n\n  private getAvailableWidth(): number {\n    const input = this.element;\n    const computedStyle = window.getComputedStyle(input);\n\n    const paddingLeft = parseFloat(computedStyle.paddingLeft) || 0;\n    const paddingRight = parseFloat(computedStyle.paddingRight) || 0;\n\n    return input.clientWidth - paddingLeft - paddingRight;\n  }\n\n  private getTextWidth(text: string): number {\n    const input = this.element;\n    const computedStyle = window.getComputedStyle(input);\n\n    const canvas = document.createElement('canvas');\n    const context = canvas.getContext('2d')!;\n    context.font = `${computedStyle.fontSize} ${computedStyle.fontFamily}`;\n\n    return context.measureText(text).width;\n  }\n\n  private isPlaceholderOverflowing(): boolean {\n    const textWidth = this.getTextWidth(this.originalPlaceholder);\n    const availableWidth = this.getAvailableWidth();\n    return textWidth > availableWidth;\n  }\n\n  private setupResizeObserver() {\n    if ('ResizeObserver' in window) {\n      this.resizeObserver = new ResizeObserver(() => {\n        this.calculateTruncatedPlaceholder();\n      });\n      this.resizeObserver.observe(this.element);\n    }\n  }\n\n  @HostListener('mouseover') onMouseOver() {\n    if (this.element) {\n      this.checkPlaceholderOverflow();\n    }\n  }\n\n  checkPlaceholderOverflow() {\n    if (this.originalPlaceholder?.length > 0) {\n      this.matTooltip.disabled = !this.isPlaceholderOverflowing();\n    } else {\n      this.matTooltip.disabled = this.element.scrollWidth <= this.element.clientWidth;\n    }\n  }\n}\n"]}
|
|
@@ -3,6 +3,7 @@ export * from './shared-directives.module';
|
|
|
3
3
|
// Directives
|
|
4
4
|
export * from './autofocus.directive';
|
|
5
5
|
export * from './container-ref.directive';
|
|
6
|
+
export * from './dynamic-ellipsis-tooltip.directive';
|
|
6
7
|
export * from './ellipsis-multiline.directive';
|
|
7
8
|
export * from './formatted-input.directive';
|
|
8
9
|
export * from './is-ellipsis.directive';
|
|
@@ -12,4 +13,4 @@ export * from './offscreen.directive';
|
|
|
12
13
|
export * from './stop-propagation.directive';
|
|
13
14
|
export * from './tb-type-number.directive';
|
|
14
15
|
export * from './var.directive';
|
|
15
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
16
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtc2hhcmVkLWxpYi9zcmMvbGlicmFyaWVzL2RpcmVjdGl2ZXMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsU0FBUztBQUNULGNBQWMsNEJBQTRCLENBQUM7QUFFM0MsYUFBYTtBQUNiLGNBQWMsdUJBQXVCLENBQUM7QUFDdEMsY0FBYywyQkFBMkIsQ0FBQztBQUMxQyxjQUFjLHNDQUFzQyxDQUFDO0FBQ3JELGNBQWMsZ0NBQWdDLENBQUM7QUFDL0MsY0FBYyw2QkFBNkIsQ0FBQztBQUM1QyxjQUFjLHlCQUF5QixDQUFDO0FBQ3hDLGNBQWMsNEJBQTRCLENBQUM7QUFDM0MsY0FBYyxxQ0FBcUMsQ0FBQztBQUNwRCxjQUFjLHVCQUF1QixDQUFDO0FBQ3RDLGNBQWMsOEJBQThCLENBQUM7QUFDN0MsY0FBYyw0QkFBNEIsQ0FBQztBQUMzQyxjQUFjLGlCQUFpQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLy8gTW9kdWxlXG5leHBvcnQgKiBmcm9tICcuL3NoYXJlZC1kaXJlY3RpdmVzLm1vZHVsZSc7XG5cbi8vIERpcmVjdGl2ZXNcbmV4cG9ydCAqIGZyb20gJy4vYXV0b2ZvY3VzLmRpcmVjdGl2ZSc7XG5leHBvcnQgKiBmcm9tICcuL2NvbnRhaW5lci1yZWYuZGlyZWN0aXZlJztcbmV4cG9ydCAqIGZyb20gJy4vZHluYW1pYy1lbGxpcHNpcy10b29sdGlwLmRpcmVjdGl2ZSc7XG5leHBvcnQgKiBmcm9tICcuL2VsbGlwc2lzLW11bHRpbGluZS5kaXJlY3RpdmUnO1xuZXhwb3J0ICogZnJvbSAnLi9mb3JtYXR0ZWQtaW5wdXQuZGlyZWN0aXZlJztcbmV4cG9ydCAqIGZyb20gJy4vaXMtZWxsaXBzaXMuZGlyZWN0aXZlJztcbmV4cG9ydCAqIGZyb20gJy4vbWF0LWhpbnQtZXJyb3IuZGlyZWN0aXZlJztcbmV4cG9ydCAqIGZyb20gJy4vbmF0aXZlLWVsZW1lbnQtaW5qZWN0b3IuZGlyZWN0aXZlJztcbmV4cG9ydCAqIGZyb20gJy4vb2Zmc2NyZWVuLmRpcmVjdGl2ZSc7XG5leHBvcnQgKiBmcm9tICcuL3N0b3AtcHJvcGFnYXRpb24uZGlyZWN0aXZlJztcbmV4cG9ydCAqIGZyb20gJy4vdGItdHlwZS1udW1iZXIuZGlyZWN0aXZlJztcbmV4cG9ydCAqIGZyb20gJy4vdmFyLmRpcmVjdGl2ZSc7XG4iXX0=
|