@taiga-ui/addon-table
Version:
A library to display tabled data with filters, search, group actions, etc.
87 lines • 21.4 kB
JavaScript
/// <reference types="@taiga-ui/tsconfig/ng-dev-mode" />
/// <reference types="@taiga-ui/tsconfig/ng-dev-mode" />
import { AsyncPipe, NgIf, NgTemplateOutlet } from '@angular/common';
import { ChangeDetectionStrategy, Component, forwardRef, inject, Input, } from '@angular/core';
import { tuiDefaultSort } from '@taiga-ui/cdk/utils/miscellaneous';
import { TuiIcon } from '@taiga-ui/core/components/icon';
import { TuiTableHead } from '../directives/head.directive';
import { TuiTableResized } from '../directives/resized.directive';
import { TuiTableDirective } from '../directives/table.directive';
import { TUI_TABLE_OPTIONS, TuiSortDirection } from '../table.options';
import * as i0 from "@angular/core";
class TuiTableTh {
constructor() {
this.options = inject(TUI_TABLE_OPTIONS);
this.head = inject(TuiTableHead, {
optional: true,
});
this.width = null;
this.table = inject(forwardRef(() => TuiTableDirective), { optional: true });
this.minWidth = -Infinity;
this.maxWidth = Infinity;
this.sorter = this.head
? (a, b) => tuiDefaultSort(a[this.key], b[this.key])
: null;
this.resizable = this.options.resizable;
this.sticky = this.options.sticky;
this.requiredSort = this.options.requiredSort;
}
get key() {
if (!this.head) {
throw new TuiTableSortKeyException();
}
return this.head.tuiHead;
}
get isCurrent() {
return !!this.sorter && !!this.table && this.sorter === this.table.sorter;
}
get icon() {
if (this.isCurrent) {
return this.table?.direction === TuiSortDirection.Asc
? this.options.sortIcons.asc
: this.options.sortIcons.desc;
}
return this.options.sortIcons.off;
}
updateSorterAndDirection() {
const sorter = this.requiredSort ? this.sorter : null;
this.table?.updateSorterAndDirection(this.isCurrentAndDescDirection ? sorter : this.sorter);
}
onResized(width) {
this.width = Math.min(Math.max(width, this.minWidth), this.maxWidth);
}
get isCurrentAndDescDirection() {
return (this.sorter === this.table?.sorter &&
this.table?.direction === TuiSortDirection.Desc);
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: TuiTableTh, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: TuiTableTh, isStandalone: true, selector: "th[tuiTh]", inputs: { minWidth: "minWidth", maxWidth: "maxWidth", sorter: "sorter", resizable: "resizable", sticky: "sticky", requiredSort: "requiredSort" }, host: { properties: { "style.min-width.px": "width || minWidth", "style.width.px": "width || minWidth", "style.max-width.px": "width || maxWidth", "class._sticky": "sticky" } }, ngImport: i0, template: "<button\n *ngIf=\"sorter && table; else content\"\n type=\"button\"\n class=\"t-sort\"\n [class.t-sort_sorted]=\"isCurrent\"\n (click)=\"updateSorterAndDirection()\"\n>\n <ng-container [ngTemplateOutlet]=\"content\" />\n {{ table && table.change$ | async }}\n <tui-icon\n class=\"t-icon\"\n [icon]=\"icon\"\n />\n</button>\n<ng-template #content>\n <ng-content />\n</ng-template>\n<div\n *ngIf=\"resizable\"\n class=\"t-bar\"\n (tuiResized)=\"onResized($event)\"\n></div>\n", styles: [":host{transition-property:box-shadow;transition-duration:var(--tui-duration, .3s);transition-timing-function:ease-in-out;position:relative;top:0;block-size:var(--tui-height-m);font:var(--tui-font-text-s);text-align:start;font-weight:700;color:var(--tui-text-secondary);background:var(--tui-background-base);cursor:default;padding:0 .75rem;box-sizing:border-box;box-shadow:0 .3125rem #ededed00;border:1px solid var(--tui-border-normal);filter:opacity(1)}@supports (-webkit-hyphens: none){:host{transform:translateZ(0)}}:host:not(:first-child){border-inline-start:none}:host._sticky,:host-context(._stuck) :host._sticky{position:sticky;z-index:30}:host._sticky:first-child,:host-context(._stuck) :host._sticky:first-child{left:0}:host._sticky:after,:host-context(._stuck) :host._sticky:after{transition-property:opacity;transition-duration:var(--tui-duration, .3s);transition-timing-function:ease-in-out;content:\"\";position:absolute;top:0;left:100%;bottom:0;inline-size:.3125rem;pointer-events:none;background:#edededb3;opacity:0}:host-context(._stuck) :host{z-index:20}:host-context(tr:not(:first-child)){border-block-start:none}:host-context(table[data-size=\"l\"]){block-size:var(--tui-height-l);font:var(--tui-font-text-m);font-weight:700;padding:0 1rem}:host-context(table[data-size=\"s\"]){block-size:var(--tui-height-s);font:var(--tui-font-text-s);font-weight:700;padding:0 .5rem}:host-context(thead[tuiThead]){position:sticky}:host-context(table._stuck)._sticky:after{opacity:1}:host-context(thead[tuiThead]._stuck){box-shadow:0 .3125rem #edededb3}:host-context([tuiTheme=\"dark\"])._sticky:after{background:#3c3c3ce6}:host-context([tuiTheme=\"dark\"] thead[tuiThead]._stuck){box-shadow:0 .3125rem #3c3c3ce6}:host-context([tuiTheme=\"dark\"] thead[tuiThead]._stuck):first-child{box-shadow:.0625rem .3125rem #3c3c3ce6}:host-context(table[data-size=\"l\"] thead[tuiThead] tr:nth-child(2)){top:var(--tui-height-l)}:host-context(table[data-size=\"m\"] thead[tuiThead] tr:nth-child(2)){top:var(--tui-height-m)}:host-context(table[data-size=\"s\"] thead[tuiThead] tr:nth-child(2)){top:var(--tui-height-s)}.t-sort{transition-property:color;transition-duration:var(--tui-duration, .3s);transition-timing-function:ease-in-out;-webkit-appearance:none;appearance:none;padding:0;border:0;background:none;line-height:inherit;text-decoration:none;display:inline-flex;vertical-align:top;flex-direction:inherit;align-items:center;outline:none;font:inherit;text-transform:inherit;color:inherit;cursor:pointer}.t-sort_sorted{color:var(--tui-text-primary)}.t-sort:focus-visible{background:var(--tui-service-selection-background)}.t-sort:hover{color:var(--tui-text-primary)}.t-bar{transition-property:opacity;transition-duration:var(--tui-duration, .3s);transition-timing-function:ease-in-out;position:absolute;top:0;bottom:0;right:-1px;inline-size:.1875rem;justify-self:flex-end;border-inline-start:2px solid transparent;background:var(--tui-status-warning);background-clip:content-box;cursor:ew-resize;opacity:0}.t-bar:hover,.t-bar:active{opacity:1}.t-icon{border-width:.25rem}\n"], dependencies: [{ kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: TuiIcon, selector: "tui-icon", inputs: ["icon", "background"] }, { kind: "directive", type: TuiTableResized, selector: "[tuiResized]", outputs: ["tuiResized"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
}
export { TuiTableTh };
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: TuiTableTh, decorators: [{
type: Component,
args: [{ standalone: true, selector: 'th[tuiTh]', imports: [AsyncPipe, NgIf, NgTemplateOutlet, TuiIcon, TuiTableResized], changeDetection: ChangeDetectionStrategy.OnPush, host: {
'[style.min-width.px]': 'width || minWidth',
'[style.width.px]': 'width || minWidth',
'[style.max-width.px]': 'width || maxWidth',
'[class._sticky]': 'sticky',
}, template: "<button\n *ngIf=\"sorter && table; else content\"\n type=\"button\"\n class=\"t-sort\"\n [class.t-sort_sorted]=\"isCurrent\"\n (click)=\"updateSorterAndDirection()\"\n>\n <ng-container [ngTemplateOutlet]=\"content\" />\n {{ table && table.change$ | async }}\n <tui-icon\n class=\"t-icon\"\n [icon]=\"icon\"\n />\n</button>\n<ng-template #content>\n <ng-content />\n</ng-template>\n<div\n *ngIf=\"resizable\"\n class=\"t-bar\"\n (tuiResized)=\"onResized($event)\"\n></div>\n", styles: [":host{transition-property:box-shadow;transition-duration:var(--tui-duration, .3s);transition-timing-function:ease-in-out;position:relative;top:0;block-size:var(--tui-height-m);font:var(--tui-font-text-s);text-align:start;font-weight:700;color:var(--tui-text-secondary);background:var(--tui-background-base);cursor:default;padding:0 .75rem;box-sizing:border-box;box-shadow:0 .3125rem #ededed00;border:1px solid var(--tui-border-normal);filter:opacity(1)}@supports (-webkit-hyphens: none){:host{transform:translateZ(0)}}:host:not(:first-child){border-inline-start:none}:host._sticky,:host-context(._stuck) :host._sticky{position:sticky;z-index:30}:host._sticky:first-child,:host-context(._stuck) :host._sticky:first-child{left:0}:host._sticky:after,:host-context(._stuck) :host._sticky:after{transition-property:opacity;transition-duration:var(--tui-duration, .3s);transition-timing-function:ease-in-out;content:\"\";position:absolute;top:0;left:100%;bottom:0;inline-size:.3125rem;pointer-events:none;background:#edededb3;opacity:0}:host-context(._stuck) :host{z-index:20}:host-context(tr:not(:first-child)){border-block-start:none}:host-context(table[data-size=\"l\"]){block-size:var(--tui-height-l);font:var(--tui-font-text-m);font-weight:700;padding:0 1rem}:host-context(table[data-size=\"s\"]){block-size:var(--tui-height-s);font:var(--tui-font-text-s);font-weight:700;padding:0 .5rem}:host-context(thead[tuiThead]){position:sticky}:host-context(table._stuck)._sticky:after{opacity:1}:host-context(thead[tuiThead]._stuck){box-shadow:0 .3125rem #edededb3}:host-context([tuiTheme=\"dark\"])._sticky:after{background:#3c3c3ce6}:host-context([tuiTheme=\"dark\"] thead[tuiThead]._stuck){box-shadow:0 .3125rem #3c3c3ce6}:host-context([tuiTheme=\"dark\"] thead[tuiThead]._stuck):first-child{box-shadow:.0625rem .3125rem #3c3c3ce6}:host-context(table[data-size=\"l\"] thead[tuiThead] tr:nth-child(2)){top:var(--tui-height-l)}:host-context(table[data-size=\"m\"] thead[tuiThead] tr:nth-child(2)){top:var(--tui-height-m)}:host-context(table[data-size=\"s\"] thead[tuiThead] tr:nth-child(2)){top:var(--tui-height-s)}.t-sort{transition-property:color;transition-duration:var(--tui-duration, .3s);transition-timing-function:ease-in-out;-webkit-appearance:none;appearance:none;padding:0;border:0;background:none;line-height:inherit;text-decoration:none;display:inline-flex;vertical-align:top;flex-direction:inherit;align-items:center;outline:none;font:inherit;text-transform:inherit;color:inherit;cursor:pointer}.t-sort_sorted{color:var(--tui-text-primary)}.t-sort:focus-visible{background:var(--tui-service-selection-background)}.t-sort:hover{color:var(--tui-text-primary)}.t-bar{transition-property:opacity;transition-duration:var(--tui-duration, .3s);transition-timing-function:ease-in-out;position:absolute;top:0;bottom:0;right:-1px;inline-size:.1875rem;justify-self:flex-end;border-inline-start:2px solid transparent;background:var(--tui-status-warning);background-clip:content-box;cursor:ew-resize;opacity:0}.t-bar:hover,.t-bar:active{opacity:1}.t-icon{border-width:.25rem}\n"] }]
}], propDecorators: { minWidth: [{
type: Input
}], maxWidth: [{
type: Input
}], sorter: [{
type: Input
}], resizable: [{
type: Input
}], sticky: [{
type: Input
}], requiredSort: [{
type: Input
}] } });
export class TuiTableSortKeyException extends Error {
constructor() {
super(ngDevMode ? 'Trying to sort with no key' : '');
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"th.component.js","sourceRoot":"","sources":["../../../../../../projects/addon-table/components/table/th/th.component.ts","../../../../../../projects/addon-table/components/table/th/th.template.html"],"names":[],"mappings":"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","sourcesContent":["/// <reference types=\"@taiga-ui/tsconfig/ng-dev-mode\" />\nimport {AsyncPipe, NgIf, NgTemplateOutlet} from '@angular/common';\nimport {\n    ChangeDetectionStrategy,\n    Component,\n    forwardRef,\n    inject,\n    Input,\n} from '@angular/core';\nimport {type TuiComparator} from '@taiga-ui/addon-table/types';\nimport {tuiDefaultSort} from '@taiga-ui/cdk/utils/miscellaneous';\nimport {TuiIcon} from '@taiga-ui/core/components/icon';\n\nimport {TuiTableHead} from '../directives/head.directive';\nimport {TuiTableResized} from '../directives/resized.directive';\nimport {TuiTableDirective} from '../directives/table.directive';\nimport {TUI_TABLE_OPTIONS, TuiSortDirection} from '../table.options';\n\n@Component({\n    standalone: true,\n    selector: 'th[tuiTh]',\n    imports: [AsyncPipe, NgIf, NgTemplateOutlet, TuiIcon, TuiTableResized],\n    templateUrl: './th.template.html',\n    styleUrls: ['./th.style.less'],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n    host: {\n        '[style.min-width.px]': 'width || minWidth',\n        '[style.width.px]': 'width || minWidth',\n        '[style.max-width.px]': 'width || maxWidth',\n        '[class._sticky]': 'sticky',\n    },\n})\nexport class TuiTableTh<T extends Partial<Record<keyof T, unknown>>> {\n    private readonly options = inject(TUI_TABLE_OPTIONS);\n\n    private readonly head = inject<TuiTableHead<T>>(TuiTableHead, {\n        optional: true,\n    });\n\n    protected width: number | null = null;\n\n    protected readonly table = inject<TuiTableDirective<T>>(\n        forwardRef(() => TuiTableDirective),\n        {optional: true},\n    );\n\n    @Input()\n    public minWidth = -Infinity;\n\n    @Input()\n    public maxWidth = Infinity;\n\n    @Input()\n    public sorter: TuiComparator<T> | null = this.head\n        ? (a, b) => tuiDefaultSort(a[this.key], b[this.key])\n        : null;\n\n    @Input()\n    public resizable = this.options.resizable;\n\n    @Input()\n    public sticky = this.options.sticky;\n\n    @Input()\n    public requiredSort = this.options.requiredSort;\n\n    public get key(): keyof T {\n        if (!this.head) {\n            throw new TuiTableSortKeyException();\n        }\n\n        return this.head.tuiHead as keyof T;\n    }\n\n    protected get isCurrent(): boolean {\n        return !!this.sorter && !!this.table && this.sorter === this.table.sorter;\n    }\n\n    protected get icon(): string {\n        if (this.isCurrent) {\n            return this.table?.direction === TuiSortDirection.Asc\n                ? this.options.sortIcons.asc\n                : this.options.sortIcons.desc;\n        }\n\n        return this.options.sortIcons.off;\n    }\n\n    protected updateSorterAndDirection(): void {\n        const sorter = this.requiredSort ? this.sorter : null;\n\n        this.table?.updateSorterAndDirection(\n            this.isCurrentAndDescDirection ? sorter : this.sorter,\n        );\n    }\n\n    protected onResized(width: number): void {\n        this.width = Math.min(Math.max(width, this.minWidth), this.maxWidth);\n    }\n\n    private get isCurrentAndDescDirection(): boolean {\n        return (\n            this.sorter === this.table?.sorter &&\n            this.table?.direction === TuiSortDirection.Desc\n        );\n    }\n}\n\nexport class TuiTableSortKeyException extends Error {\n    constructor() {\n        super(ngDevMode ? 'Trying to sort with no key' : '');\n    }\n}\n","<button\n    *ngIf=\"sorter && table; else content\"\n    type=\"button\"\n    class=\"t-sort\"\n    [class.t-sort_sorted]=\"isCurrent\"\n    (click)=\"updateSorterAndDirection()\"\n>\n    <ng-container [ngTemplateOutlet]=\"content\" />\n    {{ table && table.change$ | async }}\n    <tui-icon\n        class=\"t-icon\"\n        [icon]=\"icon\"\n    />\n</button>\n<ng-template #content>\n    <ng-content />\n</ng-template>\n<div\n    *ngIf=\"resizable\"\n    class=\"t-bar\"\n    (tuiResized)=\"onResized($event)\"\n></div>\n"]}