@leanix/components 0.4.211 → 0.4.213
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.
@@ -108,7 +108,7 @@ export class EllipsisComponent {
|
|
108
108
|
return offsetWidth < scrollWidth;
|
109
109
|
}
|
110
110
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.0", ngImport: i0, type: EllipsisComponent, deps: [{ token: LX_ELLIPSIS_DEBOUNCE_ON_RESIZE }, { token: i0.ChangeDetectorRef }, { token: i0.ElementRef }, { token: i1.ResizeObserverService }, { token: i2.TranslateService }], target: i0.ɵɵFactoryTarget.Component }); }
|
111
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.1.0", type: EllipsisComponent, isStandalone: true, selector: "lx-ellipsis", inputs: { content: "content", escapeHtmlInContent: "escapeHtmlInContent" }, viewQueries: [{ propertyName: "contentSpanEl", first: true, predicate: ["contentEl"], descendants: true }, { propertyName: "showMoreButtonEl", first: true, predicate: ["showMoreButton"], descendants: true, read: ElementRef }], ngImport: i0, template: "<span\n #contentEl\n [class.showMore]=\"isShowingMore$ | async\"\n class=\"content truncate lx-margin-right\"\n [innerHtml]=\"sanitizedContent$ | async\"\n></span>\n<button *ngIf=\"showButton$ | async\" (click)=\"onShowMoreToggle()\" lx-button #showMoreButton size=\"auto\" mode=\"link\" color=\"primary\">\n {{ showMoreButtonLabel$ | async }}\n</button>\n", styles: [":host{display:block}.content{display:inline-block;word-break:break-word}.
|
111
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.1.0", type: EllipsisComponent, isStandalone: true, selector: "lx-ellipsis", inputs: { content: "content", escapeHtmlInContent: "escapeHtmlInContent" }, viewQueries: [{ propertyName: "contentSpanEl", first: true, predicate: ["contentEl"], descendants: true }, { propertyName: "showMoreButtonEl", first: true, predicate: ["showMoreButton"], descendants: true, read: ElementRef }], ngImport: i0, template: "<span\n #contentEl\n [class.showMore]=\"isShowingMore$ | async\"\n class=\"content truncate lx-margin-right\"\n [innerHtml]=\"sanitizedContent$ | async\"\n></span>\n<button *ngIf=\"showButton$ | async\" (click)=\"onShowMoreToggle()\" lx-button #showMoreButton size=\"auto\" mode=\"link\" color=\"primary\">\n {{ showMoreButtonLabel$ | async }}\n</button>\n", styles: [":host{display:block}.content{display:inline-block;word-break:break-word}.truncate:not(.showMore){width:calc(100% - 140px);white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.content.showMore+button{margin-top:4px;display:block}button{white-space:nowrap;vertical-align:top}\n"], dependencies: [{ kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: ButtonComponent, selector: "button[lx-button]", inputs: ["size", "color", "mode", "pressed", "selected", "square", "circle", "disabled", "showSpinner"] }, { kind: "pipe", type: AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
112
112
|
}
|
113
113
|
__decorate([
|
114
114
|
Observe('contentSpanEl')
|
@@ -121,7 +121,7 @@ __decorate([
|
|
121
121
|
], EllipsisComponent.prototype, "content$", void 0);
|
122
122
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.0", ngImport: i0, type: EllipsisComponent, decorators: [{
|
123
123
|
type: Component,
|
124
|
-
args: [{ selector: 'lx-ellipsis', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [NgIf, ButtonComponent, AsyncPipe], template: "<span\n #contentEl\n [class.showMore]=\"isShowingMore$ | async\"\n class=\"content truncate lx-margin-right\"\n [innerHtml]=\"sanitizedContent$ | async\"\n></span>\n<button *ngIf=\"showButton$ | async\" (click)=\"onShowMoreToggle()\" lx-button #showMoreButton size=\"auto\" mode=\"link\" color=\"primary\">\n {{ showMoreButtonLabel$ | async }}\n</button>\n", styles: [":host{display:block}.content{display:inline-block;word-break:break-word}.
|
124
|
+
args: [{ selector: 'lx-ellipsis', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [NgIf, ButtonComponent, AsyncPipe], template: "<span\n #contentEl\n [class.showMore]=\"isShowingMore$ | async\"\n class=\"content truncate lx-margin-right\"\n [innerHtml]=\"sanitizedContent$ | async\"\n></span>\n<button *ngIf=\"showButton$ | async\" (click)=\"onShowMoreToggle()\" lx-button #showMoreButton size=\"auto\" mode=\"link\" color=\"primary\">\n {{ showMoreButtonLabel$ | async }}\n</button>\n", styles: [":host{display:block}.content{display:inline-block;word-break:break-word}.truncate:not(.showMore){width:calc(100% - 140px);white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.content.showMore+button{margin-top:4px;display:block}button{white-space:nowrap;vertical-align:top}\n"] }]
|
125
125
|
}], ctorParameters: () => [{ type: undefined, decorators: [{
|
126
126
|
type: Inject,
|
127
127
|
args: [LX_ELLIPSIS_DEBOUNCE_ON_RESIZE]
|
@@ -136,4 +136,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.0", ngImpor
|
|
136
136
|
type: ViewChild,
|
137
137
|
args: ['showMoreButton', { read: ElementRef }]
|
138
138
|
}], content$: [] } });
|
139
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ellipsis.component.js","sourceRoot":"","sources":["../../../../../../../../libs/components/src/lib/core-ui/components/ellipsis/ellipsis.component.ts","../../../../../../../../libs/components/src/lib/core-ui/components/ellipsis/ellipsis.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AsyncPipe, NgIf } from '@angular/common';\nimport {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  Inject,\n  InjectionToken,\n  Input,\n  OnDestroy,\n  OnInit,\n  ViewChild\n} from '@angular/core';\nimport { TranslateService } from '@ngx-translate/core';\nimport { escape } from 'lodash/fp';\nimport { BehaviorSubject, combineLatest, concat, merge, Observable, Subject } from 'rxjs';\nimport {\n  debounceTime,\n  distinctUntilChanged,\n  filter,\n  map,\n  pairwise,\n  skip,\n  startWith,\n  switchMap,\n  take,\n  takeUntil,\n  withLatestFrom\n} from 'rxjs/operators';\nimport { Observe } from '../../../shared/observe';\nimport { ResizeObserverService } from '../../services/resize-observer.service';\nimport { ButtonComponent } from '../button/button.component';\n\nexport const LX_ELLIPSIS_DEBOUNCE_ON_RESIZE = new InjectionToken<number>('LX_ELLIPSIS_DEBOUNCE_ON_RESIZE', {\n  providedIn: 'root',\n  factory: () => EllipsisComponent.DEFAULT_RESIZE_DEBOUNCE_MS\n});\n\n/**\n * You can set a custom max-width CSS property on your lx-ellipsis host element\n * if you want its content to never exceed a specific width,\n * e.g. <lx-ellipsis style=\"max-width: 300px\" content=\"Hello World[..]\"></lx-ellipsis>.\n */\n@Component({\n  selector: 'lx-ellipsis',\n  templateUrl: 'ellipsis.component.html',\n  styleUrls: [`ellipsis.component.scss`],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [NgIf, ButtonComponent, AsyncPipe]\n})\nexport class EllipsisComponent implements OnInit, OnDestroy {\n  static DEFAULT_RESIZE_DEBOUNCE_MS = 500;\n\n  @Input() content: string = '';\n  /**\n   * Only set this to false if the content is not a user provided string\n   * or if you sanitize the provided content yourself.\n   */\n  @Input() escapeHtmlInContent = true;\n\n  @Observe('contentSpanEl') private contentSpanEl$!: Observable<ElementRef<HTMLSpanElement>>;\n  @ViewChild('contentEl') contentSpanEl!: ElementRef<HTMLSpanElement>;\n  @Observe('showMoreButtonEl') private showMoreButtonEl$!: Observable<ElementRef<HTMLButtonElement>>;\n  @ViewChild('showMoreButton', { read: ElementRef }) showMoreButtonEl!: ElementRef<HTMLButtonElement>;\n\n  isShowingMore$ = new BehaviorSubject(false);\n\n  showButton$!: Observable<boolean>;\n  showMoreButtonLabel$!: Observable<string>;\n\n  sanitizedContent$!: Observable<string>;\n  @Observe('content')\n  private content$!: Observable<string>;\n\n  private destroyed$ = new Subject<void>();\n\n  constructor(\n    @Inject(LX_ELLIPSIS_DEBOUNCE_ON_RESIZE)\n    private debounceMsAfterResize: number,\n    private cdRef: ChangeDetectorRef,\n    private hostEl: ElementRef,\n    private resizeObserverService: ResizeObserverService,\n    private translateService: TranslateService\n  ) {}\n\n  ngOnInit(): void {\n    this.sanitizedContent$ = this.content$.pipe(\n      map((content) => {\n        if (this.escapeHtmlInContent && typeof content === 'string') {\n          return escape(content);\n        }\n        return content;\n      })\n    );\n\n    this.showMoreButtonLabel$ = this.isShowingMore$.pipe(\n      switchMap((isShowingMore) => {\n        const translationKey = isShowingMore ? 'common.showLess' : 'common.showMore';\n        return this.translateService.get(translationKey);\n      })\n    );\n    const newWidthOnResize$ = new Subject<number>();\n    const newHeightOnResize$ = new Subject<number>();\n    this.resizeObserverService.observe(this.hostEl.nativeElement, (resizedElement) => {\n      newWidthOnResize$.next(resizedElement.contentRect.width);\n      newHeightOnResize$.next(resizedElement.contentRect.height);\n    });\n    const containerWidthChangedSignificantlyAfterResize$ = newWidthOnResize$.pipe(\n      startWith(0),\n      pairwise(),\n      filter(([previousWidth, newWidth]) => Math.abs(newWidth - previousWidth) > 1),\n      map(([, newWidth]) => newWidth)\n    );\n    const userTriggeredTriggeredShowLess$ = this.isShowingMore$.pipe(filter((isShowingMore) => !isShowingMore));\n\n    const reevaluateIfContentIsOverflowing$ = combineLatest([\n      this.contentSpanEl$,\n      userTriggeredTriggeredShowLess$,\n      containerWidthChangedSignificantlyAfterResize$\n    ]);\n\n    const isContentOverflowing$ = merge(\n      concat(\n        reevaluateIfContentIsOverflowing$.pipe(take(1)), // debounce all but the first output\n        reevaluateIfContentIsOverflowing$.pipe(debounceTime(this.debounceMsAfterResize))\n      ),\n      this.content$.pipe(skip(1)) // immediately reevaluate on subsequent content changes\n    ).pipe(\n      withLatestFrom(this.contentSpanEl$),\n      map(([, contentSpanRef]) => this.isContentOverflowing(contentSpanRef))\n    );\n\n    const userTriggeredTriggeredShowMore$ = this.isShowingMore$.pipe(filter((isShowingMore) => isShowingMore));\n\n    const buttonHeight$ = this.showMoreButtonEl$.pipe(\n      filter((showMoreButtonEl) => !!showMoreButtonEl),\n      map((showMoreButtonEl) => showMoreButtonEl.nativeElement.offsetHeight),\n      distinctUntilChanged()\n    );\n    const userIncreasedBrowserWindowSizeToThePointOfNoTruncationNecessary$ = combineLatest([\n      buttonHeight$,\n      newHeightOnResize$.pipe(distinctUntilChanged())\n    ]).pipe(\n      withLatestFrom(this.isShowingMore$),\n      filter(([, isShowingMore]) => isShowingMore),\n      map(([[buttonHeight, newHeight]]) => {\n        const heightOfOneLine = buttonHeight;\n        const showLessButtonMarginTop = 4;\n        const heightOfSpanInShowMoreMode = newHeight - buttonHeight - showLessButtonMarginTop;\n        const thresholdToDetectContentInSingleLine = 4;\n        const differenceBetweenContentSpanAndButtonHeight = Math.abs(heightOfOneLine - heightOfSpanInShowMoreMode);\n        const isSpanContentDisplayedInOneLineAgain = differenceBetweenContentSpanAndButtonHeight < thresholdToDetectContentInSingleLine;\n        return isSpanContentDisplayedInOneLineAgain;\n      })\n    );\n    const contentFitsInOneLineAgainWhileShowMoreIsEnabled$ = userIncreasedBrowserWindowSizeToThePointOfNoTruncationNecessary$.pipe(\n      startWith(false),\n      pairwise(),\n      filter(([previousTextIsNowInOneLineAgain]) => !previousTextIsNowInOneLineAgain),\n      map(([, textIsNowInOneLineAgain]) => !textIsNowInOneLineAgain)\n    );\n    const contentIsOverflowingAndShowMoreIsNotEnabled$ = isContentOverflowing$.pipe(\n      withLatestFrom(this.isShowingMore$),\n      filter(([, isShowingMore]) => !isShowingMore),\n      map(([isContentLongerThanContainerWidth]) => {\n        return isContentLongerThanContainerWidth;\n      })\n    );\n    this.showButton$ = merge(\n      userTriggeredTriggeredShowMore$,\n      contentFitsInOneLineAgainWhileShowMoreIsEnabled$,\n      contentIsOverflowingAndShowMoreIsNotEnabled$\n    );\n\n    // As long as no parent component is listening on resize events,\n    // the ChangeDetectorRef.markForCheck() call done by the async pipe\n    // will not result in a change detection cycle in this component when its size changes.\n    // This is the least amount of ChangeDetectorRef.detectChanges() calls\n    // I was able to come up with. The Angular profiler shows acceptable numbers of change detection.\n    this.detectChangesWhenObservableEmits(isContentOverflowing$);\n    this.detectChangesWhenObservableEmits(this.showButton$);\n    this.detectChangesWhenObservableEmits(this.content$);\n    this.detectChangesWhenObservableEmits(contentIsOverflowingAndShowMoreIsNotEnabled$);\n    this.detectChangesWhenObservableEmits(isContentOverflowing$);\n  }\n\n  onShowMoreToggle() {\n    this.isShowingMore$.next(!this.isShowingMore$.getValue());\n    this.cdRef.detectChanges();\n  }\n\n  ngOnDestroy(): void {\n    this.resizeObserverService.unobserve(this.hostEl.nativeElement);\n  }\n\n  private detectChangesWhenObservableEmits(observable$: Observable<any>) {\n    observable$.pipe(takeUntil(this.destroyed$)).subscribe(() => this.cdRef.detectChanges());\n  }\n\n  private isContentOverflowing(contentSpanElRef: ElementRef<HTMLSpanElement>) {\n    const scrollWidth = contentSpanElRef.nativeElement.scrollWidth;\n    const offsetWidth = contentSpanElRef.nativeElement.clientWidth;\n    return offsetWidth < scrollWidth;\n  }\n}\n","<span\n  #contentEl\n  [class.showMore]=\"isShowingMore$ | async\"\n  class=\"content truncate lx-margin-right\"\n  [innerHtml]=\"sanitizedContent$ | async\"\n></span>\n<button *ngIf=\"showButton$ | async\" (click)=\"onShowMoreToggle()\" lx-button #showMoreButton size=\"auto\" mode=\"link\" color=\"primary\">\n  {{ showMoreButtonLabel$ | async }}\n</button>\n"]}
|
139
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ellipsis.component.js","sourceRoot":"","sources":["../../../../../../../../libs/components/src/lib/core-ui/components/ellipsis/ellipsis.component.ts","../../../../../../../../libs/components/src/lib/core-ui/components/ellipsis/ellipsis.component.html"],"names":[],"mappings":";AAAA,OAAO,EAAE,SAAS,EAAE,IAAI,EAAE,MAAM,iBAAiB,CAAC;AAClD,OAAO,EACL,uBAAuB,EAEvB,SAAS,EACT,UAAU,EACV,MAAM,EACN,cAAc,EACd,KAAK,EAGL,SAAS,EACV,MAAM,eAAe,CAAC;AAEvB,OAAO,EAAE,MAAM,EAAE,MAAM,WAAW,CAAC;AACnC,OAAO,EAAE,eAAe,EAAE,aAAa,EAAE,MAAM,EAAE,KAAK,EAAc,OAAO,EAAE,MAAM,MAAM,CAAC;AAC1F,OAAO,EACL,YAAY,EACZ,oBAAoB,EACpB,MAAM,EACN,GAAG,EACH,QAAQ,EACR,IAAI,EACJ,SAAS,EACT,SAAS,EACT,IAAI,EACJ,SAAS,EACT,cAAc,EACf,MAAM,gBAAgB,CAAC;AACxB,OAAO,EAAE,OAAO,EAAE,MAAM,yBAAyB,CAAC;AAElD,OAAO,EAAE,eAAe,EAAE,MAAM,4BAA4B,CAAC;;;;AAE7D,MAAM,CAAC,MAAM,8BAA8B,GAAG,IAAI,cAAc,CAAS,gCAAgC,EAAE;IACzG,UAAU,EAAE,MAAM;IAClB,OAAO,EAAE,GAAG,EAAE,CAAC,iBAAiB,CAAC,0BAA0B;CAC5D,CAAC,CAAC;AAEH;;;;GAIG;AASH,MAAM,OAAO,iBAAiB;aACrB,+BAA0B,GAAG,GAAG,AAAN,CAAO;IAyBxC,YAEU,qBAA6B,EAC7B,KAAwB,EACxB,MAAkB,EAClB,qBAA4C,EAC5C,gBAAkC;QAJlC,0BAAqB,GAArB,qBAAqB,CAAQ;QAC7B,UAAK,GAAL,KAAK,CAAmB;QACxB,WAAM,GAAN,MAAM,CAAY;QAClB,0BAAqB,GAArB,qBAAqB,CAAuB;QAC5C,qBAAgB,GAAhB,gBAAgB,CAAkB;QA7BnC,YAAO,GAAW,EAAE,CAAC;QAC9B;;;WAGG;QACM,wBAAmB,GAAG,IAAI,CAAC;QAOpC,mBAAc,GAAG,IAAI,eAAe,CAAC,KAAK,CAAC,CAAC;QASpC,eAAU,GAAG,IAAI,OAAO,EAAQ,CAAC;IAStC,CAAC;IAEJ,QAAQ;QACN,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,QAAQ,CAAC,IAAI,CACzC,GAAG,CAAC,CAAC,OAAO,EAAE,EAAE;YACd,IAAI,IAAI,CAAC,mBAAmB,IAAI,OAAO,OAAO,KAAK,QAAQ,EAAE;gBAC3D,OAAO,MAAM,CAAC,OAAO,CAAC,CAAC;aACxB;YACD,OAAO,OAAO,CAAC;QACjB,CAAC,CAAC,CACH,CAAC;QAEF,IAAI,CAAC,oBAAoB,GAAG,IAAI,CAAC,cAAc,CAAC,IAAI,CAClD,SAAS,CAAC,CAAC,aAAa,EAAE,EAAE;YAC1B,MAAM,cAAc,GAAG,aAAa,CAAC,CAAC,CAAC,iBAAiB,CAAC,CAAC,CAAC,iBAAiB,CAAC;YAC7E,OAAO,IAAI,CAAC,gBAAgB,CAAC,GAAG,CAAC,cAAc,CAAC,CAAC;QACnD,CAAC,CAAC,CACH,CAAC;QACF,MAAM,iBAAiB,GAAG,IAAI,OAAO,EAAU,CAAC;QAChD,MAAM,kBAAkB,GAAG,IAAI,OAAO,EAAU,CAAC;QACjD,IAAI,CAAC,qBAAqB,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,aAAa,EAAE,CAAC,cAAc,EAAE,EAAE;YAC/E,iBAAiB,CAAC,IAAI,CAAC,cAAc,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;YACzD,kBAAkB,CAAC,IAAI,CAAC,cAAc,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;QAC7D,CAAC,CAAC,CAAC;QACH,MAAM,8CAA8C,GAAG,iBAAiB,CAAC,IAAI,CAC3E,SAAS,CAAC,CAAC,CAAC,EACZ,QAAQ,EAAE,EACV,MAAM,CAAC,CAAC,CAAC,aAAa,EAAE,QAAQ,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,GAAG,CAAC,QAAQ,GAAG,aAAa,CAAC,GAAG,CAAC,CAAC,EAC7E,GAAG,CAAC,CAAC,CAAC,EAAE,QAAQ,CAAC,EAAE,EAAE,CAAC,QAAQ,CAAC,CAChC,CAAC;QACF,MAAM,+BAA+B,GAAG,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,aAAa,EAAE,EAAE,CAAC,CAAC,aAAa,CAAC,CAAC,CAAC;QAE5G,MAAM,iCAAiC,GAAG,aAAa,CAAC;YACtD,IAAI,CAAC,cAAc;YACnB,+BAA+B;YAC/B,8CAA8C;SAC/C,CAAC,CAAC;QAEH,MAAM,qBAAqB,GAAG,KAAK,CACjC,MAAM,CACJ,iCAAiC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,EAAE,oCAAoC;QACrF,iCAAiC,CAAC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CACjF,EACD,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,uDAAuD;SACpF,CAAC,IAAI,CACJ,cAAc,CAAC,IAAI,CAAC,cAAc,CAAC,EACnC,GAAG,CAAC,CAAC,CAAC,EAAE,cAAc,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,oBAAoB,CAAC,cAAc,CAAC,CAAC,CACvE,CAAC;QAEF,MAAM,+BAA+B,GAAG,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,aAAa,EAAE,EAAE,CAAC,aAAa,CAAC,CAAC,CAAC;QAE3G,MAAM,aAAa,GAAG,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAC/C,MAAM,CAAC,CAAC,gBAAgB,EAAE,EAAE,CAAC,CAAC,CAAC,gBAAgB,CAAC,EAChD,GAAG,CAAC,CAAC,gBAAgB,EAAE,EAAE,CAAC,gBAAgB,CAAC,aAAa,CAAC,YAAY,CAAC,EACtE,oBAAoB,EAAE,CACvB,CAAC;QACF,MAAM,gEAAgE,GAAG,aAAa,CAAC;YACrF,aAAa;YACb,kBAAkB,CAAC,IAAI,CAAC,oBAAoB,EAAE,CAAC;SAChD,CAAC,CAAC,IAAI,CACL,cAAc,CAAC,IAAI,CAAC,cAAc,CAAC,EACnC,MAAM,CAAC,CAAC,CAAC,EAAE,aAAa,CAAC,EAAE,EAAE,CAAC,aAAa,CAAC,EAC5C,GAAG,CAAC,CAAC,CAAC,CAAC,YAAY,EAAE,SAAS,CAAC,CAAC,EAAE,EAAE;YAClC,MAAM,eAAe,GAAG,YAAY,CAAC;YACrC,MAAM,uBAAuB,GAAG,CAAC,CAAC;YAClC,MAAM,0BAA0B,GAAG,SAAS,GAAG,YAAY,GAAG,uBAAuB,CAAC;YACtF,MAAM,oCAAoC,GAAG,CAAC,CAAC;YAC/C,MAAM,2CAA2C,GAAG,IAAI,CAAC,GAAG,CAAC,eAAe,GAAG,0BAA0B,CAAC,CAAC;YAC3G,MAAM,oCAAoC,GAAG,2CAA2C,GAAG,oCAAoC,CAAC;YAChI,OAAO,oCAAoC,CAAC;QAC9C,CAAC,CAAC,CACH,CAAC;QACF,MAAM,gDAAgD,GAAG,gEAAgE,CAAC,IAAI,CAC5H,SAAS,CAAC,KAAK,CAAC,EAChB,QAAQ,EAAE,EACV,MAAM,CAAC,CAAC,CAAC,+BAA+B,CAAC,EAAE,EAAE,CAAC,CAAC,+BAA+B,CAAC,EAC/E,GAAG,CAAC,CAAC,CAAC,EAAE,uBAAuB,CAAC,EAAE,EAAE,CAAC,CAAC,uBAAuB,CAAC,CAC/D,CAAC;QACF,MAAM,4CAA4C,GAAG,qBAAqB,CAAC,IAAI,CAC7E,cAAc,CAAC,IAAI,CAAC,cAAc,CAAC,EACnC,MAAM,CAAC,CAAC,CAAC,EAAE,aAAa,CAAC,EAAE,EAAE,CAAC,CAAC,aAAa,CAAC,EAC7C,GAAG,CAAC,CAAC,CAAC,iCAAiC,CAAC,EAAE,EAAE;YAC1C,OAAO,iCAAiC,CAAC;QAC3C,CAAC,CAAC,CACH,CAAC;QACF,IAAI,CAAC,WAAW,GAAG,KAAK,CACtB,+BAA+B,EAC/B,gDAAgD,EAChD,4CAA4C,CAC7C,CAAC;QAEF,gEAAgE;QAChE,mEAAmE;QACnE,uFAAuF;QACvF,sEAAsE;QACtE,iGAAiG;QACjG,IAAI,CAAC,gCAAgC,CAAC,qBAAqB,CAAC,CAAC;QAC7D,IAAI,CAAC,gCAAgC,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QACxD,IAAI,CAAC,gCAAgC,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QACrD,IAAI,CAAC,gCAAgC,CAAC,4CAA4C,CAAC,CAAC;QACpF,IAAI,CAAC,gCAAgC,CAAC,qBAAqB,CAAC,CAAC;IAC/D,CAAC;IAED,gBAAgB;QACd,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,cAAc,CAAC,QAAQ,EAAE,CAAC,CAAC;QAC1D,IAAI,CAAC,KAAK,CAAC,aAAa,EAAE,CAAC;IAC7B,CAAC;IAED,WAAW;QACT,IAAI,CAAC,qBAAqB,CAAC,SAAS,CAAC,IAAI,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC;IAClE,CAAC;IAEO,gCAAgC,CAAC,WAA4B;QACnE,WAAW,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,SAAS,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,aAAa,EAAE,CAAC,CAAC;IAC3F,CAAC;IAEO,oBAAoB,CAAC,gBAA6C;QACxE,MAAM,WAAW,GAAG,gBAAgB,CAAC,aAAa,CAAC,WAAW,CAAC;QAC/D,MAAM,WAAW,GAAG,gBAAgB,CAAC,aAAa,CAAC,WAAW,CAAC;QAC/D,OAAO,WAAW,GAAG,WAAW,CAAC;IACnC,CAAC;8GAzJU,iBAAiB,kBA2BlB,8BAA8B;kGA3B7B,iBAAiB,+UAaS,UAAU,6BChEjD,2WASA,mVDwCY,IAAI,6FAAE,eAAe,kKAAE,SAAS;;AAYR;IAAjC,OAAO,CAAC,eAAe,CAAC;yDAAkE;AAEtD;IAApC,OAAO,CAAC,kBAAkB,CAAC;4DAAuE;AAU3F;IADP,OAAO,CAAC,SAAS,CAAC;mDACmB;2FAtB3B,iBAAiB;kBAR7B,SAAS;+BACE,aAAa,mBAGN,uBAAuB,CAAC,MAAM,cACnC,IAAI,WACP,CAAC,IAAI,EAAE,eAAe,EAAE,SAAS,CAAC;;0BA6BxC,MAAM;2BAAC,8BAA8B;qKAxB/B,OAAO;sBAAf,KAAK;gBAKG,mBAAmB;sBAA3B,KAAK;gBAE4B,cAAc,MACxB,aAAa;sBAApC,SAAS;uBAAC,WAAW;gBACe,iBAAiB,MACH,gBAAgB;sBAAlE,SAAS;uBAAC,gBAAgB,EAAE,EAAE,IAAI,EAAE,UAAU,EAAE;gBASzC,QAAQ","sourcesContent":["import { AsyncPipe, NgIf } from '@angular/common';\nimport {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  Inject,\n  InjectionToken,\n  Input,\n  OnDestroy,\n  OnInit,\n  ViewChild\n} from '@angular/core';\nimport { TranslateService } from '@ngx-translate/core';\nimport { escape } from 'lodash/fp';\nimport { BehaviorSubject, combineLatest, concat, merge, Observable, Subject } from 'rxjs';\nimport {\n  debounceTime,\n  distinctUntilChanged,\n  filter,\n  map,\n  pairwise,\n  skip,\n  startWith,\n  switchMap,\n  take,\n  takeUntil,\n  withLatestFrom\n} from 'rxjs/operators';\nimport { Observe } from '../../../shared/observe';\nimport { ResizeObserverService } from '../../services/resize-observer.service';\nimport { ButtonComponent } from '../button/button.component';\n\nexport const LX_ELLIPSIS_DEBOUNCE_ON_RESIZE = new InjectionToken<number>('LX_ELLIPSIS_DEBOUNCE_ON_RESIZE', {\n  providedIn: 'root',\n  factory: () => EllipsisComponent.DEFAULT_RESIZE_DEBOUNCE_MS\n});\n\n/**\n * You can set a custom max-width CSS property on your lx-ellipsis host element\n * if you want its content to never exceed a specific width,\n * e.g. <lx-ellipsis style=\"max-width: 300px\" content=\"Hello World[..]\"></lx-ellipsis>.\n */\n@Component({\n  selector: 'lx-ellipsis',\n  templateUrl: 'ellipsis.component.html',\n  styleUrls: [`ellipsis.component.scss`],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [NgIf, ButtonComponent, AsyncPipe]\n})\nexport class EllipsisComponent implements OnInit, OnDestroy {\n  static DEFAULT_RESIZE_DEBOUNCE_MS = 500;\n\n  @Input() content: string = '';\n  /**\n   * Only set this to false if the content is not a user provided string\n   * or if you sanitize the provided content yourself.\n   */\n  @Input() escapeHtmlInContent = true;\n\n  @Observe('contentSpanEl') private contentSpanEl$!: Observable<ElementRef<HTMLSpanElement>>;\n  @ViewChild('contentEl') contentSpanEl!: ElementRef<HTMLSpanElement>;\n  @Observe('showMoreButtonEl') private showMoreButtonEl$!: Observable<ElementRef<HTMLButtonElement>>;\n  @ViewChild('showMoreButton', { read: ElementRef }) showMoreButtonEl!: ElementRef<HTMLButtonElement>;\n\n  isShowingMore$ = new BehaviorSubject(false);\n\n  showButton$!: Observable<boolean>;\n  showMoreButtonLabel$!: Observable<string>;\n\n  sanitizedContent$!: Observable<string>;\n  @Observe('content')\n  private content$!: Observable<string>;\n\n  private destroyed$ = new Subject<void>();\n\n  constructor(\n    @Inject(LX_ELLIPSIS_DEBOUNCE_ON_RESIZE)\n    private debounceMsAfterResize: number,\n    private cdRef: ChangeDetectorRef,\n    private hostEl: ElementRef,\n    private resizeObserverService: ResizeObserverService,\n    private translateService: TranslateService\n  ) {}\n\n  ngOnInit(): void {\n    this.sanitizedContent$ = this.content$.pipe(\n      map((content) => {\n        if (this.escapeHtmlInContent && typeof content === 'string') {\n          return escape(content);\n        }\n        return content;\n      })\n    );\n\n    this.showMoreButtonLabel$ = this.isShowingMore$.pipe(\n      switchMap((isShowingMore) => {\n        const translationKey = isShowingMore ? 'common.showLess' : 'common.showMore';\n        return this.translateService.get(translationKey);\n      })\n    );\n    const newWidthOnResize$ = new Subject<number>();\n    const newHeightOnResize$ = new Subject<number>();\n    this.resizeObserverService.observe(this.hostEl.nativeElement, (resizedElement) => {\n      newWidthOnResize$.next(resizedElement.contentRect.width);\n      newHeightOnResize$.next(resizedElement.contentRect.height);\n    });\n    const containerWidthChangedSignificantlyAfterResize$ = newWidthOnResize$.pipe(\n      startWith(0),\n      pairwise(),\n      filter(([previousWidth, newWidth]) => Math.abs(newWidth - previousWidth) > 1),\n      map(([, newWidth]) => newWidth)\n    );\n    const userTriggeredTriggeredShowLess$ = this.isShowingMore$.pipe(filter((isShowingMore) => !isShowingMore));\n\n    const reevaluateIfContentIsOverflowing$ = combineLatest([\n      this.contentSpanEl$,\n      userTriggeredTriggeredShowLess$,\n      containerWidthChangedSignificantlyAfterResize$\n    ]);\n\n    const isContentOverflowing$ = merge(\n      concat(\n        reevaluateIfContentIsOverflowing$.pipe(take(1)), // debounce all but the first output\n        reevaluateIfContentIsOverflowing$.pipe(debounceTime(this.debounceMsAfterResize))\n      ),\n      this.content$.pipe(skip(1)) // immediately reevaluate on subsequent content changes\n    ).pipe(\n      withLatestFrom(this.contentSpanEl$),\n      map(([, contentSpanRef]) => this.isContentOverflowing(contentSpanRef))\n    );\n\n    const userTriggeredTriggeredShowMore$ = this.isShowingMore$.pipe(filter((isShowingMore) => isShowingMore));\n\n    const buttonHeight$ = this.showMoreButtonEl$.pipe(\n      filter((showMoreButtonEl) => !!showMoreButtonEl),\n      map((showMoreButtonEl) => showMoreButtonEl.nativeElement.offsetHeight),\n      distinctUntilChanged()\n    );\n    const userIncreasedBrowserWindowSizeToThePointOfNoTruncationNecessary$ = combineLatest([\n      buttonHeight$,\n      newHeightOnResize$.pipe(distinctUntilChanged())\n    ]).pipe(\n      withLatestFrom(this.isShowingMore$),\n      filter(([, isShowingMore]) => isShowingMore),\n      map(([[buttonHeight, newHeight]]) => {\n        const heightOfOneLine = buttonHeight;\n        const showLessButtonMarginTop = 4;\n        const heightOfSpanInShowMoreMode = newHeight - buttonHeight - showLessButtonMarginTop;\n        const thresholdToDetectContentInSingleLine = 4;\n        const differenceBetweenContentSpanAndButtonHeight = Math.abs(heightOfOneLine - heightOfSpanInShowMoreMode);\n        const isSpanContentDisplayedInOneLineAgain = differenceBetweenContentSpanAndButtonHeight < thresholdToDetectContentInSingleLine;\n        return isSpanContentDisplayedInOneLineAgain;\n      })\n    );\n    const contentFitsInOneLineAgainWhileShowMoreIsEnabled$ = userIncreasedBrowserWindowSizeToThePointOfNoTruncationNecessary$.pipe(\n      startWith(false),\n      pairwise(),\n      filter(([previousTextIsNowInOneLineAgain]) => !previousTextIsNowInOneLineAgain),\n      map(([, textIsNowInOneLineAgain]) => !textIsNowInOneLineAgain)\n    );\n    const contentIsOverflowingAndShowMoreIsNotEnabled$ = isContentOverflowing$.pipe(\n      withLatestFrom(this.isShowingMore$),\n      filter(([, isShowingMore]) => !isShowingMore),\n      map(([isContentLongerThanContainerWidth]) => {\n        return isContentLongerThanContainerWidth;\n      })\n    );\n    this.showButton$ = merge(\n      userTriggeredTriggeredShowMore$,\n      contentFitsInOneLineAgainWhileShowMoreIsEnabled$,\n      contentIsOverflowingAndShowMoreIsNotEnabled$\n    );\n\n    // As long as no parent component is listening on resize events,\n    // the ChangeDetectorRef.markForCheck() call done by the async pipe\n    // will not result in a change detection cycle in this component when its size changes.\n    // This is the least amount of ChangeDetectorRef.detectChanges() calls\n    // I was able to come up with. The Angular profiler shows acceptable numbers of change detection.\n    this.detectChangesWhenObservableEmits(isContentOverflowing$);\n    this.detectChangesWhenObservableEmits(this.showButton$);\n    this.detectChangesWhenObservableEmits(this.content$);\n    this.detectChangesWhenObservableEmits(contentIsOverflowingAndShowMoreIsNotEnabled$);\n    this.detectChangesWhenObservableEmits(isContentOverflowing$);\n  }\n\n  onShowMoreToggle() {\n    this.isShowingMore$.next(!this.isShowingMore$.getValue());\n    this.cdRef.detectChanges();\n  }\n\n  ngOnDestroy(): void {\n    this.resizeObserverService.unobserve(this.hostEl.nativeElement);\n  }\n\n  private detectChangesWhenObservableEmits(observable$: Observable<any>) {\n    observable$.pipe(takeUntil(this.destroyed$)).subscribe(() => this.cdRef.detectChanges());\n  }\n\n  private isContentOverflowing(contentSpanElRef: ElementRef<HTMLSpanElement>) {\n    const scrollWidth = contentSpanElRef.nativeElement.scrollWidth;\n    const offsetWidth = contentSpanElRef.nativeElement.clientWidth;\n    return offsetWidth < scrollWidth;\n  }\n}\n","<span\n  #contentEl\n  [class.showMore]=\"isShowingMore$ | async\"\n  class=\"content truncate lx-margin-right\"\n  [innerHtml]=\"sanitizedContent$ | async\"\n></span>\n<button *ngIf=\"showButton$ | async\" (click)=\"onShowMoreToggle()\" lx-button #showMoreButton size=\"auto\" mode=\"link\" color=\"primary\">\n  {{ showMoreButtonLabel$ | async }}\n</button>\n"]}
|
@@ -8,11 +8,28 @@ import Color from 'color';
|
|
8
8
|
*/
|
9
9
|
export function getContrastColor(colorHex) {
|
10
10
|
try {
|
11
|
-
|
12
|
-
|
13
|
-
const
|
14
|
-
const
|
15
|
-
|
11
|
+
colorHex = colorHex || '#000';
|
12
|
+
const backgroundColor = Color(colorHex);
|
13
|
+
const foregroundColor = Color(backgroundColor.isDark() ? '#FFFFFF' : '#000000');
|
14
|
+
const backgroundColorLuminance = backgroundColor.luminosity();
|
15
|
+
const foregroundColorLuminance = foregroundColor.luminosity();
|
16
|
+
/**
|
17
|
+
* Use the contrast ratio formula to determine the most eligible text color for the given background color.
|
18
|
+
* It is possible tha the background color is dark based on the RGB value, but has opacity, which makes it light.
|
19
|
+
* In such cases, the luminance value is used to determine the color. The solution is based on the following article:
|
20
|
+
* https://dev.to/alvaromontoro/building-your-own-color-contrast-checker-4j7o
|
21
|
+
*/
|
22
|
+
const contrastRatio = backgroundColorLuminance > foregroundColorLuminance
|
23
|
+
? (foregroundColorLuminance + 0.05) / (backgroundColorLuminance + 0.05)
|
24
|
+
: (backgroundColorLuminance + 0.05) / (foregroundColorLuminance + 0.05);
|
25
|
+
/**
|
26
|
+
* Check contrast ratio for WCAG 2.0 level AA compliance level for small text. If the
|
27
|
+
* contrast ratio is less than 1 / 4.5, return the inverted foreground color.
|
28
|
+
*/
|
29
|
+
if (contrastRatio >= 1 / 4.5) {
|
30
|
+
return foregroundColor.negate().hex();
|
31
|
+
}
|
32
|
+
return foregroundColor.hex();
|
16
33
|
}
|
17
34
|
catch {
|
18
35
|
return '#FFFFFF';
|
@@ -32,4 +49,4 @@ export function shorthandHexHandle(hex) {
|
|
32
49
|
export function isValidHexColor(color) {
|
33
50
|
return /^#[0-9A-F]{6}$/i.test(color);
|
34
51
|
}
|
35
|
-
//# sourceMappingURL=data:application/json;base64,
|
52
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29yZS1jc3MuaGVscGVycy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL2xpYnMvY29tcG9uZW50cy9zcmMvbGliL2NvcmUtdWkvZnVuY3Rpb25zL2NvcmUtY3NzLmhlbHBlcnMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxLQUFLLE1BQU0sT0FBTyxDQUFDO0FBRTFCOzs7Ozs7R0FNRztBQUNILE1BQU0sVUFBVSxnQkFBZ0IsQ0FBQyxRQUFpQjtJQUNoRCxJQUFJO1FBQ0YsUUFBUSxHQUFHLFFBQVEsSUFBSSxNQUFNLENBQUM7UUFFOUIsTUFBTSxlQUFlLEdBQUcsS0FBSyxDQUFDLFFBQVEsQ0FBQyxDQUFDO1FBQ3hDLE1BQU0sZUFBZSxHQUFHLEtBQUssQ0FBQyxlQUFlLENBQUMsTUFBTSxFQUFFLENBQUMsQ0FBQyxDQUFDLFNBQVMsQ0FBQyxDQUFDLENBQUMsU0FBUyxDQUFDLENBQUM7UUFFaEYsTUFBTSx3QkFBd0IsR0FBRyxlQUFlLENBQUMsVUFBVSxFQUFFLENBQUM7UUFDOUQsTUFBTSx3QkFBd0IsR0FBRyxlQUFlLENBQUMsVUFBVSxFQUFFLENBQUM7UUFFOUQ7Ozs7O1dBS0c7UUFDSCxNQUFNLGFBQWEsR0FDakIsd0JBQXdCLEdBQUcsd0JBQXdCO1lBQ2pELENBQUMsQ0FBQyxDQUFDLHdCQUF3QixHQUFHLElBQUksQ0FBQyxHQUFHLENBQUMsd0JBQXdCLEdBQUcsSUFBSSxDQUFDO1lBQ3ZFLENBQUMsQ0FBQyxDQUFDLHdCQUF3QixHQUFHLElBQUksQ0FBQyxHQUFHLENBQUMsd0JBQXdCLEdBQUcsSUFBSSxDQUFDLENBQUM7UUFFNUU7OztXQUdHO1FBQ0gsSUFBSSxhQUFhLElBQUksQ0FBQyxHQUFHLEdBQUcsRUFBRTtZQUM1QixPQUFPLGVBQWUsQ0FBQyxNQUFNLEVBQUUsQ0FBQyxHQUFHLEVBQUUsQ0FBQztTQUN2QztRQUVELE9BQU8sZUFBZSxDQUFDLEdBQUcsRUFBRSxDQUFDO0tBQzlCO0lBQUMsTUFBTTtRQUNOLE9BQU8sU0FBUyxDQUFDO0tBQ2xCO0FBQ0gsQ0FBQztBQUVELE1BQU0sVUFBVSxrQkFBa0IsQ0FBQyxHQUFXO0lBQzVDLE1BQU0sY0FBYyxHQUFHLG1DQUFtQyxDQUFDO0lBQzNELE1BQU0sU0FBUyxHQUFHLEdBQUcsQ0FBQyxLQUFLLENBQUMsY0FBYyxDQUFDLENBQUM7SUFDNUMsSUFBSSxTQUFTLEVBQUU7UUFDYixNQUFNLFlBQVksR0FBRyxTQUFTLENBQUMsQ0FBQyxDQUFFLEdBQUcsU0FBUyxDQUFDLENBQUMsQ0FBQyxHQUFHLFNBQVMsQ0FBQyxDQUFDLENBQUMsR0FBRyxTQUFTLENBQUMsQ0FBQyxDQUFDLEdBQUcsU0FBUyxDQUFDLENBQUMsQ0FBQyxHQUFHLFNBQVMsQ0FBQyxDQUFDLENBQUMsR0FBRyxTQUFTLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDN0gsT0FBTyxZQUFZLENBQUM7S0FDckI7U0FBTTtRQUNMLE9BQU8sR0FBRyxDQUFDO0tBQ1o7QUFDSCxDQUFDO0FBRUQsTUFBTSxVQUFVLGVBQWUsQ0FBQyxLQUFhO0lBQzNDLE9BQU8saUJBQWlCLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDO0FBQ3ZDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgQ29sb3IgZnJvbSAnY29sb3InO1xuXG4vKipcbiAqIENvbXB1dGUgdGhlIG1vc3QgZWxpZ2libGUgdGV4dCBjb2xvciBmb3IgYSBnaXZlbiBiYWNrZ3JvdW5kIGNvbG9yIChibGFjayBvciB3aGl0ZSksIGRlcGVuZGluZyBvbiB0aGUgbHVtaW5hbmNlIG9mIHRoZVxuICogYmFja2dyb3VuZCBjb2xvci4gSW4gY2FzZSB0aGUgcHJvdmlkZWQgY29sb3IgaXMgdW5kZWZpbmVkIG9yIGludmFsaWQsIHdoaXRlICgjRkZGRkZGKSBpcyByZXR1cm5lZC5cbiAqXG4gKiBAcGFyYW0gY29sb3JIZXggQ29sb3Igc3RyaW5nIGluIGhleGFkZWNpbWFsIGVuY29kaW5nLlxuICogQHJldHVybnMgRXF1aXZhbGVudCBjb250cmFzdCBjb2xvciBpbiBoZXhhZGVjaW1hbCBlbmNvZGluZy5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGdldENvbnRyYXN0Q29sb3IoY29sb3JIZXg/OiBzdHJpbmcpOiBzdHJpbmcge1xuICB0cnkge1xuICAgIGNvbG9ySGV4ID0gY29sb3JIZXggfHwgJyMwMDAnO1xuXG4gICAgY29uc3QgYmFja2dyb3VuZENvbG9yID0gQ29sb3IoY29sb3JIZXgpO1xuICAgIGNvbnN0IGZvcmVncm91bmRDb2xvciA9IENvbG9yKGJhY2tncm91bmRDb2xvci5pc0RhcmsoKSA/ICcjRkZGRkZGJyA6ICcjMDAwMDAwJyk7XG5cbiAgICBjb25zdCBiYWNrZ3JvdW5kQ29sb3JMdW1pbmFuY2UgPSBiYWNrZ3JvdW5kQ29sb3IubHVtaW5vc2l0eSgpO1xuICAgIGNvbnN0IGZvcmVncm91bmRDb2xvckx1bWluYW5jZSA9IGZvcmVncm91bmRDb2xvci5sdW1pbm9zaXR5KCk7XG5cbiAgICAvKipcbiAgICAgKiBVc2UgdGhlIGNvbnRyYXN0IHJhdGlvIGZvcm11bGEgdG8gZGV0ZXJtaW5lIHRoZSBtb3N0IGVsaWdpYmxlIHRleHQgY29sb3IgZm9yIHRoZSBnaXZlbiBiYWNrZ3JvdW5kIGNvbG9yLlxuICAgICAqIEl0IGlzIHBvc3NpYmxlIHRoYSB0aGUgYmFja2dyb3VuZCBjb2xvciBpcyBkYXJrIGJhc2VkIG9uIHRoZSBSR0IgdmFsdWUsIGJ1dCBoYXMgb3BhY2l0eSwgd2hpY2ggbWFrZXMgaXQgbGlnaHQuXG4gICAgICogSW4gc3VjaCBjYXNlcywgdGhlIGx1bWluYW5jZSB2YWx1ZSBpcyB1c2VkIHRvIGRldGVybWluZSB0aGUgY29sb3IuIFRoZSBzb2x1dGlvbiBpcyBiYXNlZCBvbiB0aGUgZm9sbG93aW5nIGFydGljbGU6XG4gICAgICogaHR0cHM6Ly9kZXYudG8vYWx2YXJvbW9udG9yby9idWlsZGluZy15b3VyLW93bi1jb2xvci1jb250cmFzdC1jaGVja2VyLTRqN29cbiAgICAgKi9cbiAgICBjb25zdCBjb250cmFzdFJhdGlvID1cbiAgICAgIGJhY2tncm91bmRDb2xvckx1bWluYW5jZSA+IGZvcmVncm91bmRDb2xvckx1bWluYW5jZVxuICAgICAgICA/IChmb3JlZ3JvdW5kQ29sb3JMdW1pbmFuY2UgKyAwLjA1KSAvIChiYWNrZ3JvdW5kQ29sb3JMdW1pbmFuY2UgKyAwLjA1KVxuICAgICAgICA6IChiYWNrZ3JvdW5kQ29sb3JMdW1pbmFuY2UgKyAwLjA1KSAvIChmb3JlZ3JvdW5kQ29sb3JMdW1pbmFuY2UgKyAwLjA1KTtcblxuICAgIC8qKlxuICAgICAqIENoZWNrIGNvbnRyYXN0IHJhdGlvIGZvciBXQ0FHIDIuMCBsZXZlbCBBQSBjb21wbGlhbmNlIGxldmVsIGZvciBzbWFsbCB0ZXh0LiBJZiB0aGVcbiAgICAgKiBjb250cmFzdCByYXRpbyBpcyBsZXNzIHRoYW4gMSAvIDQuNSwgcmV0dXJuIHRoZSBpbnZlcnRlZCBmb3JlZ3JvdW5kIGNvbG9yLlxuICAgICAqL1xuICAgIGlmIChjb250cmFzdFJhdGlvID49IDEgLyA0LjUpIHtcbiAgICAgIHJldHVybiBmb3JlZ3JvdW5kQ29sb3IubmVnYXRlKCkuaGV4KCk7XG4gICAgfVxuXG4gICAgcmV0dXJuIGZvcmVncm91bmRDb2xvci5oZXgoKTtcbiAgfSBjYXRjaCB7XG4gICAgcmV0dXJuICcjRkZGRkZGJztcbiAgfVxufVxuXG5leHBvcnQgZnVuY3Rpb24gc2hvcnRoYW5kSGV4SGFuZGxlKGhleDogc3RyaW5nKTogc3RyaW5nIHtcbiAgY29uc3Qgc2hvcnRoYW5kUmVnZXggPSAvXigjKShbYS1mXFxkXSkoW2EtZlxcZF0pKFthLWZcXGRdKSQvaTtcbiAgY29uc3Qgc2hvcnRoYW5kID0gaGV4Lm1hdGNoKHNob3J0aGFuZFJlZ2V4KTtcbiAgaWYgKHNob3J0aGFuZCkge1xuICAgIGNvbnN0IGNvbnZlcnRlZEhleCA9IHNob3J0aGFuZFsxXSEgKyBzaG9ydGhhbmRbMl0gKyBzaG9ydGhhbmRbMl0gKyBzaG9ydGhhbmRbM10gKyBzaG9ydGhhbmRbM10gKyBzaG9ydGhhbmRbNF0gKyBzaG9ydGhhbmRbNF07XG4gICAgcmV0dXJuIGNvbnZlcnRlZEhleDtcbiAgfSBlbHNlIHtcbiAgICByZXR1cm4gaGV4O1xuICB9XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBpc1ZhbGlkSGV4Q29sb3IoY29sb3I6IHN0cmluZykge1xuICByZXR1cm4gL14jWzAtOUEtRl17Nn0kL2kudGVzdChjb2xvcik7XG59XG4iXX0=
|
@@ -768,7 +768,7 @@ class EllipsisComponent {
|
|
768
768
|
return offsetWidth < scrollWidth;
|
769
769
|
}
|
770
770
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.0", ngImport: i0, type: EllipsisComponent, deps: [{ token: LX_ELLIPSIS_DEBOUNCE_ON_RESIZE }, { token: i0.ChangeDetectorRef }, { token: i0.ElementRef }, { token: ResizeObserverService }, { token: i1$2.TranslateService }], target: i0.ɵɵFactoryTarget.Component }); }
|
771
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.1.0", type: EllipsisComponent, isStandalone: true, selector: "lx-ellipsis", inputs: { content: "content", escapeHtmlInContent: "escapeHtmlInContent" }, viewQueries: [{ propertyName: "contentSpanEl", first: true, predicate: ["contentEl"], descendants: true }, { propertyName: "showMoreButtonEl", first: true, predicate: ["showMoreButton"], descendants: true, read: ElementRef }], ngImport: i0, template: "<span\n #contentEl\n [class.showMore]=\"isShowingMore$ | async\"\n class=\"content truncate lx-margin-right\"\n [innerHtml]=\"sanitizedContent$ | async\"\n></span>\n<button *ngIf=\"showButton$ | async\" (click)=\"onShowMoreToggle()\" lx-button #showMoreButton size=\"auto\" mode=\"link\" color=\"primary\">\n {{ showMoreButtonLabel$ | async }}\n</button>\n", styles: [":host{display:block}.content{display:inline-block;word-break:break-word}.
|
771
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.1.0", type: EllipsisComponent, isStandalone: true, selector: "lx-ellipsis", inputs: { content: "content", escapeHtmlInContent: "escapeHtmlInContent" }, viewQueries: [{ propertyName: "contentSpanEl", first: true, predicate: ["contentEl"], descendants: true }, { propertyName: "showMoreButtonEl", first: true, predicate: ["showMoreButton"], descendants: true, read: ElementRef }], ngImport: i0, template: "<span\n #contentEl\n [class.showMore]=\"isShowingMore$ | async\"\n class=\"content truncate lx-margin-right\"\n [innerHtml]=\"sanitizedContent$ | async\"\n></span>\n<button *ngIf=\"showButton$ | async\" (click)=\"onShowMoreToggle()\" lx-button #showMoreButton size=\"auto\" mode=\"link\" color=\"primary\">\n {{ showMoreButtonLabel$ | async }}\n</button>\n", styles: [":host{display:block}.content{display:inline-block;word-break:break-word}.truncate:not(.showMore){width:calc(100% - 140px);white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.content.showMore+button{margin-top:4px;display:block}button{white-space:nowrap;vertical-align:top}\n"], dependencies: [{ kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: ButtonComponent, selector: "button[lx-button]", inputs: ["size", "color", "mode", "pressed", "selected", "square", "circle", "disabled", "showSpinner"] }, { kind: "pipe", type: AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
772
772
|
}
|
773
773
|
__decorate([
|
774
774
|
Observe('contentSpanEl')
|
@@ -781,7 +781,7 @@ __decorate([
|
|
781
781
|
], EllipsisComponent.prototype, "content$", void 0);
|
782
782
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.0", ngImport: i0, type: EllipsisComponent, decorators: [{
|
783
783
|
type: Component,
|
784
|
-
args: [{ selector: 'lx-ellipsis', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [NgIf, ButtonComponent, AsyncPipe], template: "<span\n #contentEl\n [class.showMore]=\"isShowingMore$ | async\"\n class=\"content truncate lx-margin-right\"\n [innerHtml]=\"sanitizedContent$ | async\"\n></span>\n<button *ngIf=\"showButton$ | async\" (click)=\"onShowMoreToggle()\" lx-button #showMoreButton size=\"auto\" mode=\"link\" color=\"primary\">\n {{ showMoreButtonLabel$ | async }}\n</button>\n", styles: [":host{display:block}.content{display:inline-block;word-break:break-word}.
|
784
|
+
args: [{ selector: 'lx-ellipsis', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [NgIf, ButtonComponent, AsyncPipe], template: "<span\n #contentEl\n [class.showMore]=\"isShowingMore$ | async\"\n class=\"content truncate lx-margin-right\"\n [innerHtml]=\"sanitizedContent$ | async\"\n></span>\n<button *ngIf=\"showButton$ | async\" (click)=\"onShowMoreToggle()\" lx-button #showMoreButton size=\"auto\" mode=\"link\" color=\"primary\">\n {{ showMoreButtonLabel$ | async }}\n</button>\n", styles: [":host{display:block}.content{display:inline-block;word-break:break-word}.truncate:not(.showMore){width:calc(100% - 140px);white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.content.showMore+button{margin-top:4px;display:block}button{white-space:nowrap;vertical-align:top}\n"] }]
|
785
785
|
}], ctorParameters: () => [{ type: undefined, decorators: [{
|
786
786
|
type: Inject,
|
787
787
|
args: [LX_ELLIPSIS_DEBOUNCE_ON_RESIZE]
|
@@ -1411,11 +1411,28 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.0", ngImpor
|
|
1411
1411
|
*/
|
1412
1412
|
function getContrastColor(colorHex) {
|
1413
1413
|
try {
|
1414
|
-
|
1415
|
-
|
1416
|
-
const
|
1417
|
-
const
|
1418
|
-
|
1414
|
+
colorHex = colorHex || '#000';
|
1415
|
+
const backgroundColor = Color(colorHex);
|
1416
|
+
const foregroundColor = Color(backgroundColor.isDark() ? '#FFFFFF' : '#000000');
|
1417
|
+
const backgroundColorLuminance = backgroundColor.luminosity();
|
1418
|
+
const foregroundColorLuminance = foregroundColor.luminosity();
|
1419
|
+
/**
|
1420
|
+
* Use the contrast ratio formula to determine the most eligible text color for the given background color.
|
1421
|
+
* It is possible tha the background color is dark based on the RGB value, but has opacity, which makes it light.
|
1422
|
+
* In such cases, the luminance value is used to determine the color. The solution is based on the following article:
|
1423
|
+
* https://dev.to/alvaromontoro/building-your-own-color-contrast-checker-4j7o
|
1424
|
+
*/
|
1425
|
+
const contrastRatio = backgroundColorLuminance > foregroundColorLuminance
|
1426
|
+
? (foregroundColorLuminance + 0.05) / (backgroundColorLuminance + 0.05)
|
1427
|
+
: (backgroundColorLuminance + 0.05) / (foregroundColorLuminance + 0.05);
|
1428
|
+
/**
|
1429
|
+
* Check contrast ratio for WCAG 2.0 level AA compliance level for small text. If the
|
1430
|
+
* contrast ratio is less than 1 / 4.5, return the inverted foreground color.
|
1431
|
+
*/
|
1432
|
+
if (contrastRatio >= 1 / 4.5) {
|
1433
|
+
return foregroundColor.negate().hex();
|
1434
|
+
}
|
1435
|
+
return foregroundColor.hex();
|
1419
1436
|
}
|
1420
1437
|
catch {
|
1421
1438
|
return '#FFFFFF';
|