@taiga-ui/kit
Version:
Taiga UI Angular main components kit
111 lines • 19.6 kB
JavaScript
import { AsyncPipe } from '@angular/common';
import { ChangeDetectionStrategy, ChangeDetectorRef, Component, ElementRef, inject, Input, NgZone, Output, signal, ViewChild, } from '@angular/core';
import { takeUntilDestroyed, toSignal } from '@angular/core/rxjs-interop';
import { WaResizeObserver } from '@ng-web-apis/resize-observer';
import { tuiTypedFromEvent, tuiZonefree } from '@taiga-ui/cdk/observables';
import { tuiInjectElement, tuiIsCurrentTarget } from '@taiga-ui/cdk/utils/dom';
import { TUI_HINT_COMPONENT, TuiHint, TuiHintDirective, } from '@taiga-ui/core/directives/hint';
import { PolymorpheusOutlet, PolymorpheusTemplate } from '@taiga-ui/polymorpheus';
import { BehaviorSubject, distinctUntilChanged, filter, map, of, pairwise, startWith, Subject, switchMap, timer, } from 'rxjs';
import { TUI_LINE_CLAMP_OPTIONS } from './line-clamp.options';
import { TuiLineClampBox } from './line-clamp-box.component';
import { TuiLineClampPositionDirective } from './line-clamp-position.directive';
import * as i0 from "@angular/core";
import * as i1 from "@taiga-ui/core/directives/hint";
class TuiLineClamp {
constructor() {
this.options = inject(TUI_LINE_CLAMP_OPTIONS);
this.el = tuiInjectElement();
this.cd = inject(ChangeDetectorRef);
this.zone = inject(NgZone);
this.linesLimit$ = new BehaviorSubject(1);
this.isOverflown$ = new Subject();
this.initialized = signal(false);
this.maxHeight = signal(0);
this.height = signal(0);
this.$ = timer(0)
.pipe(tuiZonefree(this.zone), takeUntilDestroyed())
.subscribe(() => this.initialized.set(true));
this.lineClamp = toSignal(this.linesLimit$.pipe(startWith(1), pairwise(), switchMap(([prev, next]) => next >= prev
? of(next)
: tuiTypedFromEvent(this.el, 'transitionend').pipe(filter(tuiIsCurrentTarget), map(() => next))), takeUntilDestroyed()), { initialValue: 0 });
this.lineHeight = 24;
this.overflownChange = this.isOverflown$.pipe(distinctUntilChanged());
}
set linesLimit(linesLimit) {
this.linesLimit$.next(linesLimit);
}
ngDoCheck() {
this.update();
this.isOverflown$.next(this.overflown);
}
ngAfterViewInit() {
this.initialized.set(true);
}
get overflown() {
if (!this.outlet) {
return false;
}
const { scrollHeight, scrollWidth } = this.outlet.nativeElement;
const { clientHeight, clientWidth } = this.el;
// 4px buffer for IE/Edge incorrectly rounding scrollHeight
return scrollHeight - clientHeight > 4 || scrollWidth - clientWidth > 0;
}
get computedContent() {
return this.options.showHint && this.overflown ? this.content : '';
}
updateView() {
this.cd.detectChanges();
}
update() {
if (this.outlet) {
this.height.set(this.outlet.nativeElement.scrollHeight + 4);
}
if (this.initialized()) {
this.maxHeight.set(this.lineHeight * this.linesLimit$.value);
}
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: TuiLineClamp, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: TuiLineClamp, isStandalone: true, selector: "tui-line-clamp", inputs: { lineHeight: "lineHeight", content: "content", linesLimit: "linesLimit" }, outputs: { overflownChange: "overflownChange" }, host: { listeners: { "transitionend": "updateView()", "mouseenter": "updateView()", "resize": "updateView()" }, properties: { "style.height.px": "height()", "style.max-height.px": "maxHeight()", "class._initialized": "initialized()" } }, providers: [
{
provide: TUI_HINT_COMPONENT,
useValue: TuiLineClampBox,
},
], viewQueries: [{ propertyName: "outlet", first: true, predicate: TuiHintDirective, descendants: true, read: ElementRef }], ngImport: i0, template: "<div\n tuiLineClampPosition\n class=\"t-wrapper\"\n [style.-webkit-line-clamp]=\"lineClamp()\"\n [style.word-break]=\"lineClamp() > 1 ? 'break-word' : 'break-all'\"\n [tuiHint]=\"computedContent\"\n>\n <ng-container *polymorpheusOutlet=\"content as text\">\n {{ text }}\n </ng-container>\n</div>\n", styles: [":host{position:relative;display:block;overflow:hidden}:host._initialized{transition-property:max-height;transition-duration:var(--tui-duration, .3s);transition-timing-function:ease-in-out}.t-wrapper{display:-webkit-box;-webkit-box-orient:vertical;overflow:hidden;overflow-wrap:anywhere}\n"], dependencies: [{ kind: "directive", type: PolymorpheusOutlet, selector: "[polymorpheusOutlet]", inputs: ["polymorpheusOutlet", "polymorpheusOutletContext"] }, { kind: "directive", type: i1.TuiHintDirective, selector: "[tuiHint]:not(ng-container):not(ng-template)", inputs: ["tuiHintContext", "tuiHintAppearance", "tuiHint"] }, { kind: "directive", type: TuiLineClampPositionDirective, selector: "[tuiLineClampPosition]" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
}
export { TuiLineClamp };
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: TuiLineClamp, decorators: [{
type: Component,
args: [{ standalone: true, selector: 'tui-line-clamp', imports: [
AsyncPipe,
PolymorpheusOutlet,
PolymorpheusTemplate,
TuiHint,
TuiLineClampPositionDirective,
WaResizeObserver,
], changeDetection: ChangeDetectionStrategy.OnPush, providers: [
{
provide: TUI_HINT_COMPONENT,
useValue: TuiLineClampBox,
},
], host: {
'[style.height.px]': 'height()',
'[style.max-height.px]': 'maxHeight()',
'[class._initialized]': 'initialized()',
'(transitionend)': 'updateView()',
'(mouseenter)': 'updateView()',
'(resize)': 'updateView()',
}, template: "<div\n tuiLineClampPosition\n class=\"t-wrapper\"\n [style.-webkit-line-clamp]=\"lineClamp()\"\n [style.word-break]=\"lineClamp() > 1 ? 'break-word' : 'break-all'\"\n [tuiHint]=\"computedContent\"\n>\n <ng-container *polymorpheusOutlet=\"content as text\">\n {{ text }}\n </ng-container>\n</div>\n", styles: [":host{position:relative;display:block;overflow:hidden}:host._initialized{transition-property:max-height;transition-duration:var(--tui-duration, .3s);transition-timing-function:ease-in-out}.t-wrapper{display:-webkit-box;-webkit-box-orient:vertical;overflow:hidden;overflow-wrap:anywhere}\n"] }]
}], propDecorators: { outlet: [{
type: ViewChild,
args: [TuiHintDirective, { read: ElementRef }]
}], lineHeight: [{
type: Input
}], content: [{
type: Input
}], overflownChange: [{
type: Output
}], linesLimit: [{
type: Input
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"line-clamp.component.js","sourceRoot":"","sources":["../../../../../projects/kit/components/line-clamp/line-clamp.component.ts","../../../../../projects/kit/components/line-clamp/line-clamp.template.html"],"names":[],"mappings":"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","sourcesContent":["import {AsyncPipe} from '@angular/common';\nimport type {AfterViewInit, DoCheck} from '@angular/core';\nimport {\n    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    ElementRef,\n    inject,\n    Input,\n    NgZone,\n    Output,\n    signal,\n    ViewChild,\n} from '@angular/core';\nimport {takeUntilDestroyed, toSignal} from '@angular/core/rxjs-interop';\nimport {WaResizeObserver} from '@ng-web-apis/resize-observer';\nimport {tuiTypedFromEvent, tuiZonefree} from '@taiga-ui/cdk/observables';\nimport {tuiInjectElement, tuiIsCurrentTarget} from '@taiga-ui/cdk/utils/dom';\nimport {\n    TUI_HINT_COMPONENT,\n    TuiHint,\n    TuiHintDirective,\n} from '@taiga-ui/core/directives/hint';\nimport type {PolymorpheusContent} from '@taiga-ui/polymorpheus';\nimport {PolymorpheusOutlet, PolymorpheusTemplate} from '@taiga-ui/polymorpheus';\nimport type {Observable} from 'rxjs';\nimport {\n    BehaviorSubject,\n    distinctUntilChanged,\n    filter,\n    map,\n    of,\n    pairwise,\n    startWith,\n    Subject,\n    switchMap,\n    timer,\n} from 'rxjs';\n\nimport {TUI_LINE_CLAMP_OPTIONS} from './line-clamp.options';\nimport {TuiLineClampBox} from './line-clamp-box.component';\nimport {TuiLineClampPositionDirective} from './line-clamp-position.directive';\n\n@Component({\n    standalone: true,\n    selector: 'tui-line-clamp',\n    imports: [\n        AsyncPipe,\n        PolymorpheusOutlet,\n        PolymorpheusTemplate,\n        TuiHint,\n        TuiLineClampPositionDirective,\n        WaResizeObserver,\n    ],\n    templateUrl: './line-clamp.template.html',\n    styleUrls: ['./line-clamp.style.less'],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n    providers: [\n        {\n            provide: TUI_HINT_COMPONENT,\n            useValue: TuiLineClampBox,\n        },\n    ],\n    host: {\n        '[style.height.px]': 'height()',\n        '[style.max-height.px]': 'maxHeight()',\n        '[class._initialized]': 'initialized()',\n        '(transitionend)': 'updateView()',\n        '(mouseenter)': 'updateView()',\n        '(resize)': 'updateView()',\n    },\n})\nexport class TuiLineClamp implements DoCheck, AfterViewInit {\n    @ViewChild(TuiHintDirective, {read: ElementRef})\n    private readonly outlet?: ElementRef<HTMLElement>;\n\n    private readonly options = inject(TUI_LINE_CLAMP_OPTIONS);\n    private readonly el = tuiInjectElement();\n    private readonly cd = inject(ChangeDetectorRef);\n    private readonly zone: NgZone = inject(NgZone);\n    private readonly linesLimit$ = new BehaviorSubject(1);\n    private readonly isOverflown$ = new Subject<boolean>();\n    protected initialized = signal(false);\n    protected maxHeight = signal(0);\n    protected height = signal(0);\n\n    protected readonly $ = timer(0)\n        .pipe(tuiZonefree(this.zone), takeUntilDestroyed())\n        .subscribe(() => this.initialized.set(true));\n\n    protected lineClamp = toSignal(\n        this.linesLimit$.pipe(\n            startWith(1),\n            pairwise(),\n            switchMap(([prev, next]) =>\n                next >= prev\n                    ? of(next)\n                    : tuiTypedFromEvent(this.el, 'transitionend').pipe(\n                          filter(tuiIsCurrentTarget),\n                          map(() => next),\n                      ),\n            ),\n            takeUntilDestroyed(),\n        ),\n        {initialValue: 0},\n    );\n\n    @Input()\n    public lineHeight = 24;\n\n    @Input()\n    public content: PolymorpheusContent;\n\n    @Output()\n    public readonly overflownChange: Observable<boolean> =\n        this.isOverflown$.pipe(distinctUntilChanged());\n\n    @Input()\n    public set linesLimit(linesLimit: number) {\n        this.linesLimit$.next(linesLimit);\n    }\n\n    public ngDoCheck(): void {\n        this.update();\n        this.isOverflown$.next(this.overflown);\n    }\n\n    public ngAfterViewInit(): void {\n        this.initialized.set(true);\n    }\n\n    protected get overflown(): boolean {\n        if (!this.outlet) {\n            return false;\n        }\n\n        const {scrollHeight, scrollWidth} = this.outlet.nativeElement;\n        const {clientHeight, clientWidth} = this.el;\n\n        // 4px buffer for IE/Edge incorrectly rounding scrollHeight\n        return scrollHeight - clientHeight > 4 || scrollWidth - clientWidth > 0;\n    }\n\n    protected get computedContent(): PolymorpheusContent {\n        return this.options.showHint && this.overflown ? this.content : '';\n    }\n\n    protected updateView(): void {\n        this.cd.detectChanges();\n    }\n\n    private update(): void {\n        if (this.outlet) {\n            this.height.set(this.outlet.nativeElement.scrollHeight + 4);\n        }\n\n        if (this.initialized()) {\n            this.maxHeight.set(this.lineHeight * this.linesLimit$.value);\n        }\n    }\n}\n","<div\n    tuiLineClampPosition\n    class=\"t-wrapper\"\n    [style.-webkit-line-clamp]=\"lineClamp()\"\n    [style.word-break]=\"lineClamp() > 1 ? 'break-word' : 'break-all'\"\n    [tuiHint]=\"computedContent\"\n>\n    <ng-container *polymorpheusOutlet=\"content as text\">\n        {{ text }}\n    </ng-container>\n</div>\n"]}