@lumeer/pivot 0.0.13 → 0.2.0
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/fesm2022/lumeer-pivot.mjs +56 -43
- package/fesm2022/lumeer-pivot.mjs.map +1 -1
- package/index.d.ts +332 -5
- package/lumeer-pivot-0.2.0.tgz +0 -0
- package/package.json +4 -6
- package/esm2022/lib/directives/lmr-templates.directive.mjs +0 -27
- package/esm2022/lib/lmr-pivot-table.component.mjs +0 -147
- package/esm2022/lib/lmr-pivot-table.module.mjs +0 -63
- package/esm2022/lib/lmr-simple-pivot-table.component.mjs +0 -101
- package/esm2022/lib/pipes/cell-has-value.pipe.mjs +0 -35
- package/esm2022/lib/pipes/contrast-color.pipe.mjs +0 -41
- package/esm2022/lib/pipes/filter-visible-cells.pipe.mjs +0 -35
- package/esm2022/lib/pipes/is-cell-expandable.pipe.mjs +0 -35
- package/esm2022/lib/pipes/is-cell-expanded.pipe.mjs +0 -37
- package/esm2022/lib/pipes/is-cell-rows-expandable.pipe.mjs +0 -35
- package/esm2022/lib/pipes/pivot-data-empty.pipe.mjs +0 -38
- package/esm2022/lib/util/lmr-pivot-config.mjs +0 -17
- package/esm2022/lib/util/lmr-pivot-constants.mjs +0 -12
- package/esm2022/lib/util/lmr-pivot-data.mjs +0 -2
- package/esm2022/lib/util/lmr-pivot-state.mjs +0 -156
- package/esm2022/lib/util/lmr-pivot-table.mjs +0 -2
- package/esm2022/lib/util/lmr-simple-pivot-config.mjs +0 -2
- package/esm2022/lib/util/pivot-data-converter.mjs +0 -507
- package/esm2022/lib/util/pivot-table-converter.mjs +0 -1017
- package/esm2022/lib/util/pivot-util.mjs +0 -75
- package/esm2022/lumeer-pivot.mjs +0 -5
- package/esm2022/public-api.mjs +0 -13
- package/lib/directives/lmr-templates.directive.d.ts +0 -14
- package/lib/lmr-pivot-table.component.d.ts +0 -49
- package/lib/lmr-pivot-table.module.d.ts +0 -18
- package/lib/lmr-simple-pivot-table.component.d.ts +0 -38
- package/lib/pipes/cell-has-value.pipe.d.ts +0 -8
- package/lib/pipes/contrast-color.pipe.d.ts +0 -11
- package/lib/pipes/filter-visible-cells.pipe.d.ts +0 -9
- package/lib/pipes/is-cell-expandable.pipe.d.ts +0 -8
- package/lib/pipes/is-cell-expanded.pipe.d.ts +0 -9
- package/lib/pipes/is-cell-rows-expandable.pipe.d.ts +0 -8
- package/lib/pipes/pivot-data-empty.pipe.d.ts +0 -8
- package/lib/util/lmr-pivot-config.d.ts +0 -84
- package/lib/util/lmr-pivot-constants.d.ts +0 -11
- package/lib/util/lmr-pivot-data.d.ts +0 -50
- package/lib/util/lmr-pivot-state.d.ts +0 -14
- package/lib/util/lmr-pivot-table.d.ts +0 -25
- package/lib/util/lmr-simple-pivot-config.d.ts +0 -9
- package/lib/util/pivot-data-converter.d.ts +0 -46
- package/lib/util/pivot-table-converter.d.ts +0 -74
- package/lib/util/pivot-util.d.ts +0 -12
- package/lumeer-pivot-0.0.13.tgz +0 -0
- package/public-api.d.ts +0 -9
package/index.d.ts
CHANGED
|
@@ -1,5 +1,332 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
1
|
+
import * as i0 from '@angular/core';
|
|
2
|
+
import { OnInit, OnChanges, TemplateRef, EventEmitter, SimpleChanges, PipeTransform } from '@angular/core';
|
|
3
|
+
import { Constraint, DataResource, DataAggregationType, ConstraintData, QueryStem, QueryAttribute, Collection, DocumentsAndLinksData, LinkType, Query, Attribute, LanguageTag } from '@lumeer/data-filters';
|
|
4
|
+
import { Observable, BehaviorSubject } from 'rxjs';
|
|
5
|
+
import * as i11 from '@angular/common';
|
|
6
|
+
import * as i12 from '@angular/cdk/scrolling';
|
|
7
|
+
|
|
8
|
+
interface LmrPivotData {
|
|
9
|
+
data: LmrPivotStemData[];
|
|
10
|
+
constraintData?: ConstraintData;
|
|
11
|
+
mergeTables?: boolean;
|
|
12
|
+
ableToMerge?: boolean;
|
|
13
|
+
}
|
|
14
|
+
interface LmrPivotStemData {
|
|
15
|
+
columnHeaders: LmrPivotDataHeader[];
|
|
16
|
+
columnHeaderAttributes: LmrPivotHeaderAttribute[];
|
|
17
|
+
rowHeaders: LmrPivotDataHeader[];
|
|
18
|
+
rowHeaderAttributes: LmrPivotHeaderAttribute[];
|
|
19
|
+
valueTitles: string[];
|
|
20
|
+
values: any[][];
|
|
21
|
+
dataResources: DataResource[][][];
|
|
22
|
+
valuesConstraints?: Constraint[];
|
|
23
|
+
valueTypes?: LmrPivotValueType[];
|
|
24
|
+
valueAggregations?: DataAggregationType[];
|
|
25
|
+
rowsConfig: LmrPivotDimensionConfig[];
|
|
26
|
+
columnsConfig: LmrPivotDimensionConfig[];
|
|
27
|
+
hasAdditionalColumnLevel?: boolean;
|
|
28
|
+
}
|
|
29
|
+
interface LmrPivotDimensionConfig {
|
|
30
|
+
showSums?: boolean;
|
|
31
|
+
sticky?: boolean;
|
|
32
|
+
sort?: LmrPivotSort;
|
|
33
|
+
expressions?: LmrPivotExpression[];
|
|
34
|
+
}
|
|
35
|
+
interface LmrPivotHeaderAttribute {
|
|
36
|
+
title: string;
|
|
37
|
+
color: string;
|
|
38
|
+
}
|
|
39
|
+
interface LmrPivotDataHeader {
|
|
40
|
+
title: string;
|
|
41
|
+
children?: LmrPivotDataHeader[];
|
|
42
|
+
targetIndex?: number;
|
|
43
|
+
color: string;
|
|
44
|
+
isValueHeader: boolean;
|
|
45
|
+
constraint?: Constraint;
|
|
46
|
+
attributeName?: string;
|
|
47
|
+
expressions?: LmrPivotDataHeaderExpression[];
|
|
48
|
+
}
|
|
49
|
+
interface LmrPivotDataHeaderExpression extends LmrPivotExpression {
|
|
50
|
+
firstHeaderIndex?: number;
|
|
51
|
+
operands: LmrPivotDataHeaderOperand[];
|
|
52
|
+
}
|
|
53
|
+
type LmrPivotDataHeaderOperand = LmrPivotHeaderOperand & {
|
|
54
|
+
headers: LmrPivotDataHeader[];
|
|
55
|
+
} | LmrPivotValueOperand | LmrPivotDataHeaderExpression;
|
|
56
|
+
|
|
57
|
+
interface LmrPivotConfig {
|
|
58
|
+
version?: LmrPivotConfigVersion;
|
|
59
|
+
stemsConfigs: LmrPivotStemConfig[];
|
|
60
|
+
mergeTables?: boolean;
|
|
61
|
+
}
|
|
62
|
+
interface LmrPivotTransform {
|
|
63
|
+
checkValidConstraintOverride?: (c1: Constraint, c2: Constraint) => Constraint;
|
|
64
|
+
formatAggregation?: (type: DataAggregationType) => string;
|
|
65
|
+
formatSummaryHeader?: (header: LmrPivotDataHeader, level: number) => {
|
|
66
|
+
title?: string;
|
|
67
|
+
summary: string;
|
|
68
|
+
};
|
|
69
|
+
formatRowHeader?: (title: string, level: number) => string;
|
|
70
|
+
formatColumnHeader?: (title: string, level: number) => string;
|
|
71
|
+
}
|
|
72
|
+
interface LmrPivotStemConfig {
|
|
73
|
+
stem?: QueryStem;
|
|
74
|
+
rowAttributes: LmrPivotRowAttribute[];
|
|
75
|
+
columnAttributes: LmrPivotColumnAttribute[];
|
|
76
|
+
valueAttributes: LmrPivotValueAttribute[];
|
|
77
|
+
}
|
|
78
|
+
declare enum LmrPivotConfigVersion {
|
|
79
|
+
V1 = "1"
|
|
80
|
+
}
|
|
81
|
+
interface LmrPivotAttribute extends QueryAttribute {
|
|
82
|
+
}
|
|
83
|
+
interface LmrPivotRowColumnAttribute extends LmrPivotAttribute {
|
|
84
|
+
showSums?: boolean;
|
|
85
|
+
sticky?: boolean;
|
|
86
|
+
sort?: LmrPivotSort;
|
|
87
|
+
expressions?: LmrPivotExpression[];
|
|
88
|
+
}
|
|
89
|
+
interface LmrPivotRowAttribute extends LmrPivotRowColumnAttribute {
|
|
90
|
+
showHeader?: boolean;
|
|
91
|
+
}
|
|
92
|
+
interface LmrPivotColumnAttribute extends LmrPivotRowColumnAttribute {
|
|
93
|
+
}
|
|
94
|
+
interface LmrPivotSortValue {
|
|
95
|
+
title: string;
|
|
96
|
+
isSummary?: boolean;
|
|
97
|
+
}
|
|
98
|
+
interface LmrPivotSortList {
|
|
99
|
+
valueTitle: string;
|
|
100
|
+
values: LmrPivotSortValue[];
|
|
101
|
+
}
|
|
102
|
+
interface LmrPivotSort {
|
|
103
|
+
attribute?: LmrPivotAttribute;
|
|
104
|
+
list?: LmrPivotSortList;
|
|
105
|
+
asc: boolean;
|
|
106
|
+
}
|
|
107
|
+
declare enum LmrPivotValueType {
|
|
108
|
+
Default = "default",
|
|
109
|
+
ColumnPercentage = "column",
|
|
110
|
+
RowPercentage = "row",
|
|
111
|
+
AllPercentage = "all"
|
|
112
|
+
}
|
|
113
|
+
interface LmrPivotValueAttribute extends LmrPivotAttribute {
|
|
114
|
+
aggregation: DataAggregationType;
|
|
115
|
+
valueType?: LmrPivotValueType;
|
|
116
|
+
}
|
|
117
|
+
interface LmrPivotExpression {
|
|
118
|
+
operation: LmrPivotExpressionOperation;
|
|
119
|
+
operands: LmrPivotOperand[];
|
|
120
|
+
title: string;
|
|
121
|
+
type: 'expression';
|
|
122
|
+
position: LmrPivotPosition;
|
|
123
|
+
expandable?: boolean;
|
|
124
|
+
}
|
|
125
|
+
declare enum LmrPivotPosition {
|
|
126
|
+
BeforeHeader = "beforeHeader",
|
|
127
|
+
StickToEnd = "stickToEnd"
|
|
128
|
+
}
|
|
129
|
+
type LmrPivotExpressionOperation = 'add' | 'subtract' | 'multiply' | 'divide';
|
|
130
|
+
interface LmrPivotHeaderOperand {
|
|
131
|
+
type: 'header';
|
|
132
|
+
value: string;
|
|
133
|
+
}
|
|
134
|
+
interface LmrPivotValueOperand {
|
|
135
|
+
type: 'value';
|
|
136
|
+
value: number;
|
|
137
|
+
}
|
|
138
|
+
type LmrPivotOperand = LmrPivotHeaderOperand | LmrPivotValueOperand | LmrPivotExpression;
|
|
139
|
+
|
|
140
|
+
interface LmrPivotTable {
|
|
141
|
+
cells: LmrPivotTableCell[][];
|
|
142
|
+
}
|
|
143
|
+
interface LmrPivotTableCell {
|
|
144
|
+
value: any;
|
|
145
|
+
dataResources?: DataResource[];
|
|
146
|
+
constraint?: Constraint;
|
|
147
|
+
summary?: string;
|
|
148
|
+
rowSpan: number;
|
|
149
|
+
colSpan: number;
|
|
150
|
+
cssClass: string;
|
|
151
|
+
isValue?: boolean;
|
|
152
|
+
isHeader?: boolean;
|
|
153
|
+
isAttributeHeader?: boolean;
|
|
154
|
+
isSummary?: boolean;
|
|
155
|
+
background?: string;
|
|
156
|
+
label?: string;
|
|
157
|
+
stickyTop?: boolean;
|
|
158
|
+
stickyStart?: boolean;
|
|
159
|
+
rowIndexes?: number[];
|
|
160
|
+
childIndexes?: number[];
|
|
161
|
+
originalRowIndex?: number;
|
|
162
|
+
expandable?: boolean;
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
interface LmrPivotTableState {
|
|
166
|
+
cells?: LmrPivotTableCellState[][];
|
|
167
|
+
}
|
|
168
|
+
interface LmrPivotTableCellState {
|
|
169
|
+
collapsed?: boolean;
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
declare class LmrPivotTableComponent implements OnInit, OnChanges {
|
|
173
|
+
collections: Collection[];
|
|
174
|
+
data: DocumentsAndLinksData;
|
|
175
|
+
linkTypes: LinkType[];
|
|
176
|
+
query: Query;
|
|
177
|
+
constraintData: ConstraintData;
|
|
178
|
+
config: LmrPivotConfig;
|
|
179
|
+
transform: LmrPivotTransform;
|
|
180
|
+
emptyTablesTemplateInput: TemplateRef<any>;
|
|
181
|
+
tableCellTemplateInput: TemplateRef<any>;
|
|
182
|
+
initiallyCollapsed: boolean;
|
|
183
|
+
cellClick: EventEmitter<{
|
|
184
|
+
cell: LmrPivotTableCell;
|
|
185
|
+
tableIndex: number;
|
|
186
|
+
rowIndex: number;
|
|
187
|
+
columnIndex: number;
|
|
188
|
+
}>;
|
|
189
|
+
pivotDataChange: EventEmitter<LmrPivotData>;
|
|
190
|
+
pivotTablesChange: EventEmitter<LmrPivotTable[]>;
|
|
191
|
+
emptyTablesTemplate: TemplateRef<any>;
|
|
192
|
+
tableCellTemplate: TemplateRef<any>;
|
|
193
|
+
private readonly pivotTransformer;
|
|
194
|
+
private readonly pivotTableConverter;
|
|
195
|
+
readonly stickyColumnWidth = 150;
|
|
196
|
+
readonly stickyColumnHeight = 40;
|
|
197
|
+
private dataSubject$;
|
|
198
|
+
private currentTables;
|
|
199
|
+
pivotData$: Observable<LmrPivotData>;
|
|
200
|
+
pivotTables$: Observable<LmrPivotTable[]>;
|
|
201
|
+
pivotStates$: BehaviorSubject<LmrPivotTableState[]>;
|
|
202
|
+
ngOnInit(): void;
|
|
203
|
+
private onPivotDataChange;
|
|
204
|
+
private onPivotTablesChange;
|
|
205
|
+
private handleData;
|
|
206
|
+
ngOnChanges(changes: SimpleChanges): void;
|
|
207
|
+
private resetState;
|
|
208
|
+
onCellClick(cell: LmrPivotTableCell, row: LmrPivotTableCell[], tableIndex: number, rowIndex: number, columnIndex: number): void;
|
|
209
|
+
private setState;
|
|
210
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<LmrPivotTableComponent, never>;
|
|
211
|
+
static ɵcmp: i0.ɵɵComponentDeclaration<LmrPivotTableComponent, "lmr-pivot-table", never, { "collections": { "alias": "collections"; "required": false; }; "data": { "alias": "data"; "required": false; }; "linkTypes": { "alias": "linkTypes"; "required": false; }; "query": { "alias": "query"; "required": false; }; "constraintData": { "alias": "constraintData"; "required": false; }; "config": { "alias": "config"; "required": false; }; "transform": { "alias": "transform"; "required": false; }; "emptyTablesTemplateInput": { "alias": "emptyTablesTemplateInput"; "required": false; }; "tableCellTemplateInput": { "alias": "tableCellTemplateInput"; "required": false; }; "initiallyCollapsed": { "alias": "initiallyCollapsed"; "required": false; }; }, { "cellClick": "cellClick"; "pivotDataChange": "pivotDataChange"; "pivotTablesChange": "pivotTablesChange"; }, ["emptyTablesTemplate", "tableCellTemplate"], never, false, never>;
|
|
212
|
+
}
|
|
213
|
+
|
|
214
|
+
interface LmrSimplePivotConfig {
|
|
215
|
+
rowAttributes: LmrSimplePivotRowAttribute[];
|
|
216
|
+
columnAttributes: LmrSimplePivotColumnAttribute[];
|
|
217
|
+
valueAttributes: LmrSimplePivotValueAttribute[];
|
|
218
|
+
}
|
|
219
|
+
type LmrSimplePivotRowAttribute = Omit<LmrPivotRowAttribute, 'resourceType' | 'resourceId' | 'resourceIndex'>;
|
|
220
|
+
type LmrSimplePivotColumnAttribute = Omit<LmrPivotColumnAttribute, 'resourceType' | 'resourceId' | 'resourceIndex'>;
|
|
221
|
+
type LmrSimplePivotValueAttribute = Omit<LmrPivotValueAttribute, 'resourceType' | 'resourceId' | 'resourceIndex'>;
|
|
222
|
+
|
|
223
|
+
declare class LmrSimplePivotTableComponent implements OnChanges {
|
|
224
|
+
rows: Record<string, any>[];
|
|
225
|
+
attributes: Attribute[];
|
|
226
|
+
color: string;
|
|
227
|
+
config: LmrSimplePivotConfig;
|
|
228
|
+
transform: LmrPivotTransform;
|
|
229
|
+
locale: LanguageTag;
|
|
230
|
+
initiallyCollapsed: boolean;
|
|
231
|
+
cellClick: EventEmitter<{
|
|
232
|
+
cell: LmrPivotTableCell;
|
|
233
|
+
tableIndex: number;
|
|
234
|
+
rowIndex: number;
|
|
235
|
+
columnIndex: number;
|
|
236
|
+
}>;
|
|
237
|
+
pivotDataChange: EventEmitter<LmrPivotData>;
|
|
238
|
+
pivotTablesChange: EventEmitter<LmrPivotTable[]>;
|
|
239
|
+
emptyTablesTemplate: TemplateRef<any>;
|
|
240
|
+
tableCellTemplate: TemplateRef<any>;
|
|
241
|
+
readonly collectionId: string;
|
|
242
|
+
readonly query: Query;
|
|
243
|
+
collection: Collection;
|
|
244
|
+
pivotConfig: LmrPivotConfig;
|
|
245
|
+
data: DocumentsAndLinksData;
|
|
246
|
+
constraintData: ConstraintData;
|
|
247
|
+
ngOnChanges(changes: SimpleChanges): void;
|
|
248
|
+
private createCollection;
|
|
249
|
+
private createConfig;
|
|
250
|
+
private createRows;
|
|
251
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<LmrSimplePivotTableComponent, never>;
|
|
252
|
+
static ɵcmp: i0.ɵɵComponentDeclaration<LmrSimplePivotTableComponent, "lmr-simple-pivot-table", never, { "rows": { "alias": "rows"; "required": false; }; "attributes": { "alias": "attributes"; "required": false; }; "color": { "alias": "color"; "required": false; }; "config": { "alias": "config"; "required": false; }; "transform": { "alias": "transform"; "required": false; }; "locale": { "alias": "locale"; "required": false; }; "initiallyCollapsed": { "alias": "initiallyCollapsed"; "required": false; }; }, { "cellClick": "cellClick"; "pivotDataChange": "pivotDataChange"; "pivotTablesChange": "pivotTablesChange"; }, ["emptyTablesTemplate", "tableCellTemplate"], never, false, never>;
|
|
253
|
+
}
|
|
254
|
+
|
|
255
|
+
declare class PivotDataEmptyPipe implements PipeTransform {
|
|
256
|
+
transform(value: LmrPivotData): boolean;
|
|
257
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<PivotDataEmptyPipe, never>;
|
|
258
|
+
static ɵpipe: i0.ɵɵPipeDeclaration<PivotDataEmptyPipe, "pivotDataEmpty", false>;
|
|
259
|
+
}
|
|
260
|
+
|
|
261
|
+
declare class CellHasValuePipe implements PipeTransform {
|
|
262
|
+
transform(cell: LmrPivotTableCell): boolean;
|
|
263
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<CellHasValuePipe, never>;
|
|
264
|
+
static ɵpipe: i0.ɵɵPipeDeclaration<CellHasValuePipe, "cellHasValue", false>;
|
|
265
|
+
}
|
|
266
|
+
|
|
267
|
+
declare class IsCellExpandablePipe implements PipeTransform {
|
|
268
|
+
transform(cell: LmrPivotTableCell): boolean;
|
|
269
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<IsCellExpandablePipe, never>;
|
|
270
|
+
static ɵpipe: i0.ɵɵPipeDeclaration<IsCellExpandablePipe, "isCellExpandable", false>;
|
|
271
|
+
}
|
|
272
|
+
|
|
273
|
+
declare class IsCellExpandedPipe implements PipeTransform {
|
|
274
|
+
transform(cell: LmrPivotTableCell, columnIndex: number, state: LmrPivotTableState): boolean;
|
|
275
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<IsCellExpandedPipe, never>;
|
|
276
|
+
static ɵpipe: i0.ɵɵPipeDeclaration<IsCellExpandedPipe, "isCellExpanded", false>;
|
|
277
|
+
}
|
|
278
|
+
|
|
279
|
+
declare class IsCellRowsExpandablePipe implements PipeTransform {
|
|
280
|
+
transform(cell: LmrPivotTableCell): boolean;
|
|
281
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<IsCellRowsExpandablePipe, never>;
|
|
282
|
+
static ɵpipe: i0.ɵɵPipeDeclaration<IsCellRowsExpandablePipe, "isCellRowsExpandable", false>;
|
|
283
|
+
}
|
|
284
|
+
|
|
285
|
+
declare class FilterVisibleCellsPipe implements PipeTransform {
|
|
286
|
+
transform(table: LmrPivotTable, state: LmrPivotTableState): LmrPivotTableCell[][];
|
|
287
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<FilterVisibleCellsPipe, never>;
|
|
288
|
+
static ɵpipe: i0.ɵɵPipeDeclaration<FilterVisibleCellsPipe, "filterVisibleCells", false>;
|
|
289
|
+
}
|
|
290
|
+
|
|
291
|
+
declare class ContrastColorPipe implements PipeTransform {
|
|
292
|
+
transform(color: string, returnCodes?: {
|
|
293
|
+
dark: string;
|
|
294
|
+
light: string;
|
|
295
|
+
}, opacity?: number): string;
|
|
296
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<ContrastColorPipe, never>;
|
|
297
|
+
static ɵpipe: i0.ɵɵPipeDeclaration<ContrastColorPipe, "contrastColor", false>;
|
|
298
|
+
static ɵprov: i0.ɵɵInjectableDeclaration<ContrastColorPipe>;
|
|
299
|
+
}
|
|
300
|
+
|
|
301
|
+
declare class LmrEmptyTablesTemplateDirective {
|
|
302
|
+
template: TemplateRef<any>;
|
|
303
|
+
constructor(template: TemplateRef<any>);
|
|
304
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<LmrEmptyTablesTemplateDirective, never>;
|
|
305
|
+
static ɵdir: i0.ɵɵDirectiveDeclaration<LmrEmptyTablesTemplateDirective, "[lmr-empty-tables-tmp]", never, {}, {}, never, never, false, never>;
|
|
306
|
+
}
|
|
307
|
+
declare class LmrTableCellTemplateDirective {
|
|
308
|
+
template: TemplateRef<any>;
|
|
309
|
+
constructor(template: TemplateRef<any>);
|
|
310
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<LmrTableCellTemplateDirective, never>;
|
|
311
|
+
static ɵdir: i0.ɵɵDirectiveDeclaration<LmrTableCellTemplateDirective, "[lmr-table-cell-tmp]", never, {}, {}, never, never, false, never>;
|
|
312
|
+
}
|
|
313
|
+
|
|
314
|
+
declare class LmrPivotTableModule {
|
|
315
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<LmrPivotTableModule, never>;
|
|
316
|
+
static ɵmod: i0.ɵɵNgModuleDeclaration<LmrPivotTableModule, [typeof LmrPivotTableComponent, typeof LmrSimplePivotTableComponent, typeof PivotDataEmptyPipe, typeof CellHasValuePipe, typeof IsCellExpandablePipe, typeof IsCellExpandedPipe, typeof IsCellRowsExpandablePipe, typeof FilterVisibleCellsPipe, typeof ContrastColorPipe, typeof LmrEmptyTablesTemplateDirective, typeof LmrTableCellTemplateDirective], [typeof i11.CommonModule, typeof i12.ScrollingModule], [typeof LmrPivotTableComponent, typeof LmrSimplePivotTableComponent, typeof LmrEmptyTablesTemplateDirective, typeof LmrTableCellTemplateDirective]>;
|
|
317
|
+
static ɵinj: i0.ɵɵInjectorDeclaration<LmrPivotTableModule>;
|
|
318
|
+
}
|
|
319
|
+
|
|
320
|
+
declare function pivotAttributesAreSame(a1: LmrPivotAttribute, a2: LmrPivotAttribute): boolean;
|
|
321
|
+
declare function isPivotConfigChanged(viewConfig: LmrPivotConfig, currentConfig: LmrPivotConfig): boolean;
|
|
322
|
+
declare function createDefaultPivotConfig(query: Query): LmrPivotConfig;
|
|
323
|
+
declare function createDefaultPivotStemConfig(stem?: QueryStem): LmrPivotStemConfig;
|
|
324
|
+
declare function pivotConfigIsEmpty(config: LmrPivotConfig): boolean;
|
|
325
|
+
declare function pivotStemConfigIsEmpty(config: LmrPivotStemConfig): boolean;
|
|
326
|
+
declare function contrastColor(color: string, returnCodes?: {
|
|
327
|
+
dark: string;
|
|
328
|
+
light: string;
|
|
329
|
+
}, opacity?: number): string;
|
|
330
|
+
|
|
331
|
+
export { LmrEmptyTablesTemplateDirective, LmrPivotConfigVersion, LmrPivotPosition, LmrPivotTableComponent, LmrPivotTableModule, LmrPivotValueType, LmrSimplePivotTableComponent, LmrTableCellTemplateDirective, contrastColor, createDefaultPivotConfig, createDefaultPivotStemConfig, isPivotConfigChanged, pivotAttributesAreSame, pivotConfigIsEmpty, pivotStemConfigIsEmpty };
|
|
332
|
+
export type { LmrPivotAttribute, LmrPivotColumnAttribute, LmrPivotConfig, LmrPivotData, LmrPivotDataHeader, LmrPivotDataHeaderExpression, LmrPivotDataHeaderOperand, LmrPivotDimensionConfig, LmrPivotExpression, LmrPivotExpressionOperation, LmrPivotHeaderAttribute, LmrPivotHeaderOperand, LmrPivotOperand, LmrPivotRowAttribute, LmrPivotRowColumnAttribute, LmrPivotSort, LmrPivotSortList, LmrPivotSortValue, LmrPivotStemConfig, LmrPivotStemData, LmrPivotTable, LmrPivotTableCell, LmrPivotTransform, LmrPivotValueAttribute, LmrPivotValueOperand, LmrSimplePivotColumnAttribute, LmrSimplePivotConfig, LmrSimplePivotRowAttribute, LmrSimplePivotValueAttribute };
|
|
Binary file
|
package/package.json
CHANGED
|
@@ -1,10 +1,10 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@lumeer/pivot",
|
|
3
|
-
"version": "0.0
|
|
3
|
+
"version": "0.2.0",
|
|
4
4
|
"peerDependencies": {
|
|
5
|
-
"@angular/cdk": "^
|
|
6
|
-
"@angular/common": "^
|
|
7
|
-
"@angular/core": "^
|
|
5
|
+
"@angular/cdk": "^20.0.0",
|
|
6
|
+
"@angular/common": "^20.0.0",
|
|
7
|
+
"@angular/core": "^20.0.0",
|
|
8
8
|
"@lumeer/data-filters": "^0.8.6",
|
|
9
9
|
"@lumeer/utils": "^0.1.3"
|
|
10
10
|
},
|
|
@@ -20,8 +20,6 @@
|
|
|
20
20
|
},
|
|
21
21
|
".": {
|
|
22
22
|
"types": "./index.d.ts",
|
|
23
|
-
"esm2022": "./esm2022/lumeer-pivot.mjs",
|
|
24
|
-
"esm": "./esm2022/lumeer-pivot.mjs",
|
|
25
23
|
"default": "./fesm2022/lumeer-pivot.mjs"
|
|
26
24
|
}
|
|
27
25
|
}
|
|
@@ -1,27 +0,0 @@
|
|
|
1
|
-
import { Directive } from '@angular/core';
|
|
2
|
-
import * as i0 from "@angular/core";
|
|
3
|
-
export class LmrEmptyTablesTemplateDirective {
|
|
4
|
-
template;
|
|
5
|
-
constructor(template) {
|
|
6
|
-
this.template = template;
|
|
7
|
-
}
|
|
8
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: LmrEmptyTablesTemplateDirective, deps: [{ token: i0.TemplateRef }], target: i0.ɵɵFactoryTarget.Directive });
|
|
9
|
-
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "17.3.12", type: LmrEmptyTablesTemplateDirective, selector: "[lmr-empty-tables-tmp]", ngImport: i0 });
|
|
10
|
-
}
|
|
11
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: LmrEmptyTablesTemplateDirective, decorators: [{
|
|
12
|
-
type: Directive,
|
|
13
|
-
args: [{ selector: '[lmr-empty-tables-tmp]' }]
|
|
14
|
-
}], ctorParameters: () => [{ type: i0.TemplateRef }] });
|
|
15
|
-
export class LmrTableCellTemplateDirective {
|
|
16
|
-
template;
|
|
17
|
-
constructor(template) {
|
|
18
|
-
this.template = template;
|
|
19
|
-
}
|
|
20
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: LmrTableCellTemplateDirective, deps: [{ token: i0.TemplateRef }], target: i0.ɵɵFactoryTarget.Directive });
|
|
21
|
-
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "17.3.12", type: LmrTableCellTemplateDirective, selector: "[lmr-table-cell-tmp]", ngImport: i0 });
|
|
22
|
-
}
|
|
23
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: LmrTableCellTemplateDirective, decorators: [{
|
|
24
|
-
type: Directive,
|
|
25
|
-
args: [{ selector: '[lmr-table-cell-tmp]' }]
|
|
26
|
-
}], ctorParameters: () => [{ type: i0.TemplateRef }] });
|
|
27
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG1yLXRlbXBsYXRlcy5kaXJlY3RpdmUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9sbXItcGl2b3QtdGFibGUvc3JjL2xpYi9kaXJlY3RpdmVzL2xtci10ZW1wbGF0ZXMuZGlyZWN0aXZlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBQyxTQUFTLEVBQWMsTUFBTSxlQUFlLENBQUM7O0FBR3JELE1BQU0sT0FBTywrQkFBK0I7SUFDdkI7SUFBbkIsWUFBbUIsUUFBMEI7UUFBMUIsYUFBUSxHQUFSLFFBQVEsQ0FBa0I7SUFBSSxDQUFDO3dHQUR2QywrQkFBK0I7NEZBQS9CLCtCQUErQjs7NEZBQS9CLCtCQUErQjtrQkFEM0MsU0FBUzttQkFBQyxFQUFFLFFBQVEsRUFBRSx3QkFBd0IsRUFBRTs7QUFNakQsTUFBTSxPQUFPLDZCQUE2QjtJQUNyQjtJQUFuQixZQUFtQixRQUEwQjtRQUExQixhQUFRLEdBQVIsUUFBUSxDQUFrQjtJQUFJLENBQUM7d0dBRHZDLDZCQUE2Qjs0RkFBN0IsNkJBQTZCOzs0RkFBN0IsNkJBQTZCO2tCQUR6QyxTQUFTO21CQUFDLEVBQUUsUUFBUSxFQUFFLHNCQUFzQixFQUFFIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtEaXJlY3RpdmUsIFRlbXBsYXRlUmVmfSBmcm9tICdAYW5ndWxhci9jb3JlJztcblxuQERpcmVjdGl2ZSh7IHNlbGVjdG9yOiAnW2xtci1lbXB0eS10YWJsZXMtdG1wXScgfSlcbmV4cG9ydCBjbGFzcyBMbXJFbXB0eVRhYmxlc1RlbXBsYXRlRGlyZWN0aXZlIHtcbiAgY29uc3RydWN0b3IocHVibGljIHRlbXBsYXRlOiBUZW1wbGF0ZVJlZjxhbnk+KSB7IH1cbn1cblxuQERpcmVjdGl2ZSh7IHNlbGVjdG9yOiAnW2xtci10YWJsZS1jZWxsLXRtcF0nIH0pXG5leHBvcnQgY2xhc3MgTG1yVGFibGVDZWxsVGVtcGxhdGVEaXJlY3RpdmUge1xuICBjb25zdHJ1Y3RvcihwdWJsaWMgdGVtcGxhdGU6IFRlbXBsYXRlUmVmPGFueT4pIHsgfVxufVxuIl19
|
|
@@ -1,147 +0,0 @@
|
|
|
1
|
-
import { ChangeDetectionStrategy, Component, ContentChild, EventEmitter, Input, Output, TemplateRef } from '@angular/core';
|
|
2
|
-
import { PivotDataConverter } from './util/pivot-data-converter';
|
|
3
|
-
import { asyncScheduler, BehaviorSubject, filter, map, tap, throttleTime } from 'rxjs';
|
|
4
|
-
import { PivotTableConverter } from './util/pivot-table-converter';
|
|
5
|
-
import { LmrEmptyTablesTemplateDirective, LmrTableCellTemplateDirective } from './directives/lmr-templates.directive';
|
|
6
|
-
import { collapseAllCells, isCellExpandable, toggleExpanded } from './util/lmr-pivot-state';
|
|
7
|
-
import { isPivotConfigChanged } from './util/pivot-util';
|
|
8
|
-
import * as i0 from "@angular/core";
|
|
9
|
-
import * as i1 from "@angular/common";
|
|
10
|
-
import * as i2 from "./pipes/pivot-data-empty.pipe";
|
|
11
|
-
import * as i3 from "./pipes/cell-has-value.pipe";
|
|
12
|
-
import * as i4 from "./pipes/is-cell-expandable.pipe";
|
|
13
|
-
import * as i5 from "./pipes/is-cell-expanded.pipe";
|
|
14
|
-
import * as i6 from "./pipes/is-cell-rows-expandable.pipe";
|
|
15
|
-
import * as i7 from "./pipes/filter-visible-cells.pipe";
|
|
16
|
-
import * as i8 from "./pipes/contrast-color.pipe";
|
|
17
|
-
export class LmrPivotTableComponent {
|
|
18
|
-
collections;
|
|
19
|
-
data;
|
|
20
|
-
linkTypes;
|
|
21
|
-
query;
|
|
22
|
-
constraintData;
|
|
23
|
-
config;
|
|
24
|
-
transform;
|
|
25
|
-
emptyTablesTemplateInput;
|
|
26
|
-
tableCellTemplateInput;
|
|
27
|
-
initiallyCollapsed;
|
|
28
|
-
cellClick = new EventEmitter();
|
|
29
|
-
pivotDataChange = new EventEmitter();
|
|
30
|
-
pivotTablesChange = new EventEmitter();
|
|
31
|
-
emptyTablesTemplate;
|
|
32
|
-
tableCellTemplate;
|
|
33
|
-
pivotTransformer = new PivotDataConverter();
|
|
34
|
-
pivotTableConverter = new PivotTableConverter();
|
|
35
|
-
stickyColumnWidth = 150;
|
|
36
|
-
stickyColumnHeight = 40;
|
|
37
|
-
dataSubject$ = new BehaviorSubject(null);
|
|
38
|
-
currentTables;
|
|
39
|
-
pivotData$;
|
|
40
|
-
pivotTables$;
|
|
41
|
-
pivotStates$ = new BehaviorSubject([]);
|
|
42
|
-
ngOnInit() {
|
|
43
|
-
const observable = this.dataSubject$.pipe(filter(data => !!data));
|
|
44
|
-
this.pivotData$ = observable.pipe(throttleTime(200, asyncScheduler, { trailing: true, leading: true }), map(data => this.handleData(data)), tap(data => this.onPivotDataChange(data)));
|
|
45
|
-
this.pivotTables$ = this.pivotData$.pipe(map(data => this.pivotTableConverter.createTables(data, this.transform)), tap(tables => this.onPivotTablesChange(tables)));
|
|
46
|
-
}
|
|
47
|
-
onPivotDataChange(data) {
|
|
48
|
-
this.pivotDataChange.emit(data);
|
|
49
|
-
}
|
|
50
|
-
onPivotTablesChange(tables) {
|
|
51
|
-
if (this.initiallyCollapsed && tablesAreVeryDifferent(this.currentTables, tables)) {
|
|
52
|
-
this.pivotStates$.next(tables.map(table => collapseAllCells(table)));
|
|
53
|
-
}
|
|
54
|
-
this.currentTables = tables;
|
|
55
|
-
this.pivotTablesChange.emit(tables);
|
|
56
|
-
}
|
|
57
|
-
handleData(data) {
|
|
58
|
-
return this.pivotTransformer.createData(data.config, data.transform, data.collections, data.linkTypes, data.data, data.query, data.constraintData);
|
|
59
|
-
}
|
|
60
|
-
ngOnChanges(changes) {
|
|
61
|
-
if (shouldResetState(changes)) {
|
|
62
|
-
this.resetState();
|
|
63
|
-
}
|
|
64
|
-
this.dataSubject$.next({
|
|
65
|
-
config: this.config,
|
|
66
|
-
transform: this.transform,
|
|
67
|
-
collections: this.collections,
|
|
68
|
-
linkTypes: this.linkTypes,
|
|
69
|
-
data: this.data,
|
|
70
|
-
query: this.query,
|
|
71
|
-
constraintData: this.constraintData,
|
|
72
|
-
});
|
|
73
|
-
}
|
|
74
|
-
resetState() {
|
|
75
|
-
// this.pivotStates$.next([]);
|
|
76
|
-
}
|
|
77
|
-
onCellClick(cell, row, tableIndex, rowIndex, columnIndex) {
|
|
78
|
-
if (isCellExpandable(cell)) {
|
|
79
|
-
const oldState = this.pivotStates$.value[tableIndex];
|
|
80
|
-
const newState = toggleExpanded(cell, columnIndex, oldState);
|
|
81
|
-
this.setState(tableIndex, newState);
|
|
82
|
-
}
|
|
83
|
-
else if (cell?.isHeader || cell?.isValue) {
|
|
84
|
-
const headerCell = cell.isHeader ? cell : row.find(c => c.isHeader);
|
|
85
|
-
this.cellClick.emit({ cell, tableIndex, rowIndex: headerCell?.originalRowIndex || rowIndex, columnIndex });
|
|
86
|
-
}
|
|
87
|
-
}
|
|
88
|
-
setState(index, state) {
|
|
89
|
-
const statesCopy = [...(this.pivotStates$.value || [])];
|
|
90
|
-
statesCopy.splice(index, 1, state);
|
|
91
|
-
this.pivotStates$.next(statesCopy);
|
|
92
|
-
}
|
|
93
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: LmrPivotTableComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
94
|
-
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: LmrPivotTableComponent, selector: "lmr-pivot-table", inputs: { collections: "collections", data: "data", linkTypes: "linkTypes", query: "query", constraintData: "constraintData", config: "config", transform: "transform", emptyTablesTemplateInput: "emptyTablesTemplateInput", tableCellTemplateInput: "tableCellTemplateInput", initiallyCollapsed: "initiallyCollapsed" }, outputs: { cellClick: "cellClick", pivotDataChange: "pivotDataChange", pivotTablesChange: "pivotTablesChange" }, queries: [{ propertyName: "emptyTablesTemplate", first: true, predicate: LmrEmptyTablesTemplateDirective, descendants: true, read: TemplateRef }, { propertyName: "tableCellTemplate", first: true, predicate: LmrTableCellTemplateDirective, descendants: true, read: TemplateRef }], usesOnChanges: true, ngImport: i0, template: "<ng-container *ngIf=\"{pivotTables: pivotTables$ | async, pivotStates: pivotStates$ | async, pivotData: pivotData$ | async} as data\">\n <ng-container *ngIf=\"data.pivotTables?.length && !(data.pivotData | pivotDataEmpty)\">\n <table *ngFor=\"let pivotTable of data.pivotTables; let first; let tableIndex = index\"\n class=\"table table-without-padding table-borderless table-md\"\n [class.mt-4]=\"!first\">\n <tr *ngFor=\"let rowCells of pivotTable | filterVisibleCells:data.pivotStates[tableIndex]; let rowIndex = index\">\n\n <ng-container *ngFor=\"let cell of rowCells; let cellIndex = index\">\n <td *ngIf=\"cell && {hasValue: cell | cellHasValue} as cellData\"\n class=\"cell position-relative {{cell.constraint ? (cell.constraint.type | lowercase) : ''}} text-truncate\"\n [class.expandable]=\"cell | isCellExpandable\"\n [class.expandable-rows]=\"cell | isCellRowsExpandable\"\n [class.expanded]=\"cell | isCellExpanded:cellIndex:data.pivotStates[tableIndex]\"\n [class.sticky-start]=\"cell.stickyStart\"\n [class.sticky-top]=\"cell.stickyTop\"\n [rowSpan]=\"cell.rowSpan\"\n [colSpan]=\"cell.colSpan\"\n [style.top.px]=\"cell.stickyTop ? (rowIndex * stickyColumnHeight) : undefined\"\n [style.left.px]=\"cell.stickyStart ? (cellIndex * stickyColumnWidth) : undefined\"\n [style.width.px]=\"cell.stickyStart ? (cell.colSpan * stickyColumnWidth) : undefined\"\n [style.max-width.px]=\"cell.stickyStart ? (cell.colSpan * stickyColumnWidth) : 300\"\n [style.min-width.px]=\"cell.stickyStart ? (cell.colSpan * stickyColumnWidth) : undefined\"\n [ngClass]=\"cell.cssClass\"\n [style.background]=\"cell.background\"\n [style.color]=\"cell.background && (cell.background | contrastColor)\"\n (click)=\"onCellClick(cell, rowCells, tableIndex, rowIndex, cellIndex)\">\n <ng-container *ngIf=\"cell.summary\">\n <ng-container *ngIf=\"cellData.hasValue\">\n <div class=\"d-flex align-items-center h-100\">\n <span class=\"summary me-2\">{{cell.summary}}</span>\n <ng-template #defaultTableCellTemplate>\n <span class=\"flex-grow-1 h-100 text-truncate d-inline-block\">\n {{ cell.value }}\n </span>\n </ng-template>\n\n <ng-template\n [ngTemplateOutlet]=\"tableCellTemplate || tableCellTemplateInput || defaultTableCellTemplate\"\n [ngTemplateOutletContext]=\"{ value: cell.value, cell: cell }\">\n </ng-template>\n </div>\n </ng-container>\n <ng-container *ngIf=\"!cellData.hasValue\">\n <span class=\"d-block\" style=\"overflow: hidden; white-space: normal\">{{cell.summary}}</span>\n </ng-container>\n </ng-container>\n <ng-container *ngIf=\"!cell.summary\">\n <ng-container *ngIf=\"cellData.hasValue\">\n <ng-template #defaultTableCellTemplate>\n <span class=\"d-block\" style=\"overflow: hidden; white-space: normal; cursor: pointer\">\n {{ cell.value }}\n </span>\n </ng-template>\n\n <ng-template\n [ngTemplateOutlet]=\"tableCellTemplate || tableCellTemplateInput || defaultTableCellTemplate\"\n [ngTemplateOutletContext]=\"{ value: cell.value, cell: cell }\">\n </ng-template>\n </ng-container>\n <ng-container *ngIf=\"!cellData.hasValue\"> </ng-container>\n </ng-container>\n </td>\n </ng-container>\n </tr>\n </table>\n </ng-container>\n\n <ng-container *ngIf=\"!data.pivotTables?.length || (data.pivotData | pivotDataEmpty)\">\n <ng-template #defaultEmptyTablesTemplate>\n <div> </div>\n </ng-template>\n\n <ng-template\n [ngTemplateOutlet]=\"emptyTablesTemplate || emptyTablesTemplateInput || defaultEmptyTablesTemplate\"\n [ngTemplateOutletContext]=\"{ }\">\n </ng-template>\n </ng-container>\n</ng-container>\n", styles: [".table{border-spacing:0;border-collapse:separate}.pivot-data-cell,.pivot-data-group-cell,.pivot-column-header-cell,.pivot-column-group-header-cell{text-align:right}.pivot-column-header-cell,.pivot-row-attribute-header-cell,.pivot-row-header-cell,.pivot-data-group-cell,.pivot-row-group-header-cell,.pivot-column-group-header-cell{font-weight:700;border:1px #f8f9fa solid}.pivot-row-header-cell.sticky-start,.pivot-row-group-header-cell.sticky-start,.pivot-empty-cell.sticky-start{position:sticky!important;width:150px;min-width:150px;max-width:150px!important;z-index:9}.pivot-row-header-cell.expandable,.pivot-row-group-header-cell.expandable{position:absolute;cursor:pointer;padding-right:15px}.pivot-row-header-cell.expandable:after,.pivot-row-group-header-cell.expandable:after{content:\"\";position:absolute;top:50%;right:5px;width:0;height:0;border-top:5px solid transparent;border-bottom:5px solid transparent;border-left:5px solid #253746;transform:translateY(-50%)}.pivot-row-header-cell.expandable.expanded.expandable-rows:after,.pivot-row-header-cell.expandable.expanded:after,.pivot-row-group-header-cell.expandable.expanded.expandable-rows:after,.pivot-row-group-header-cell.expandable.expanded:after{border-left:5px solid transparent;border-right:5px solid transparent;border-top:5px solid #253746;border-bottom:0}.pivot-column-header-cell.sticky-top,.pivot-row-attribute-header-cell.sticky-top,.pivot-column-group-header-cell.sticky-top,.pivot-empty-cell.sticky-top{position:sticky!important;z-index:10}.pivot-row-attribute-header-cell.sticky-top,.pivot-row-attribute-header-cell.sticky-start{position:sticky!important;z-index:11}.pivot-empty-cell{background:#fff;border:1px white solid}.pivot-empty-cell.sticky-top{z-index:11}.cell.color{padding:0!important;height:30px}.cell.color .summary{padding-left:.5rem}\n"], dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }, { kind: "pipe", type: i1.LowerCasePipe, name: "lowercase" }, { kind: "pipe", type: i2.PivotDataEmptyPipe, name: "pivotDataEmpty" }, { kind: "pipe", type: i3.CellHasValuePipe, name: "cellHasValue" }, { kind: "pipe", type: i4.IsCellExpandablePipe, name: "isCellExpandable" }, { kind: "pipe", type: i5.IsCellExpandedPipe, name: "isCellExpanded" }, { kind: "pipe", type: i6.IsCellRowsExpandablePipe, name: "isCellRowsExpandable" }, { kind: "pipe", type: i7.FilterVisibleCellsPipe, name: "filterVisibleCells" }, { kind: "pipe", type: i8.ContrastColorPipe, name: "contrastColor" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
95
|
-
}
|
|
96
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: LmrPivotTableComponent, decorators: [{
|
|
97
|
-
type: Component,
|
|
98
|
-
args: [{ selector: 'lmr-pivot-table', changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container *ngIf=\"{pivotTables: pivotTables$ | async, pivotStates: pivotStates$ | async, pivotData: pivotData$ | async} as data\">\n <ng-container *ngIf=\"data.pivotTables?.length && !(data.pivotData | pivotDataEmpty)\">\n <table *ngFor=\"let pivotTable of data.pivotTables; let first; let tableIndex = index\"\n class=\"table table-without-padding table-borderless table-md\"\n [class.mt-4]=\"!first\">\n <tr *ngFor=\"let rowCells of pivotTable | filterVisibleCells:data.pivotStates[tableIndex]; let rowIndex = index\">\n\n <ng-container *ngFor=\"let cell of rowCells; let cellIndex = index\">\n <td *ngIf=\"cell && {hasValue: cell | cellHasValue} as cellData\"\n class=\"cell position-relative {{cell.constraint ? (cell.constraint.type | lowercase) : ''}} text-truncate\"\n [class.expandable]=\"cell | isCellExpandable\"\n [class.expandable-rows]=\"cell | isCellRowsExpandable\"\n [class.expanded]=\"cell | isCellExpanded:cellIndex:data.pivotStates[tableIndex]\"\n [class.sticky-start]=\"cell.stickyStart\"\n [class.sticky-top]=\"cell.stickyTop\"\n [rowSpan]=\"cell.rowSpan\"\n [colSpan]=\"cell.colSpan\"\n [style.top.px]=\"cell.stickyTop ? (rowIndex * stickyColumnHeight) : undefined\"\n [style.left.px]=\"cell.stickyStart ? (cellIndex * stickyColumnWidth) : undefined\"\n [style.width.px]=\"cell.stickyStart ? (cell.colSpan * stickyColumnWidth) : undefined\"\n [style.max-width.px]=\"cell.stickyStart ? (cell.colSpan * stickyColumnWidth) : 300\"\n [style.min-width.px]=\"cell.stickyStart ? (cell.colSpan * stickyColumnWidth) : undefined\"\n [ngClass]=\"cell.cssClass\"\n [style.background]=\"cell.background\"\n [style.color]=\"cell.background && (cell.background | contrastColor)\"\n (click)=\"onCellClick(cell, rowCells, tableIndex, rowIndex, cellIndex)\">\n <ng-container *ngIf=\"cell.summary\">\n <ng-container *ngIf=\"cellData.hasValue\">\n <div class=\"d-flex align-items-center h-100\">\n <span class=\"summary me-2\">{{cell.summary}}</span>\n <ng-template #defaultTableCellTemplate>\n <span class=\"flex-grow-1 h-100 text-truncate d-inline-block\">\n {{ cell.value }}\n </span>\n </ng-template>\n\n <ng-template\n [ngTemplateOutlet]=\"tableCellTemplate || tableCellTemplateInput || defaultTableCellTemplate\"\n [ngTemplateOutletContext]=\"{ value: cell.value, cell: cell }\">\n </ng-template>\n </div>\n </ng-container>\n <ng-container *ngIf=\"!cellData.hasValue\">\n <span class=\"d-block\" style=\"overflow: hidden; white-space: normal\">{{cell.summary}}</span>\n </ng-container>\n </ng-container>\n <ng-container *ngIf=\"!cell.summary\">\n <ng-container *ngIf=\"cellData.hasValue\">\n <ng-template #defaultTableCellTemplate>\n <span class=\"d-block\" style=\"overflow: hidden; white-space: normal; cursor: pointer\">\n {{ cell.value }}\n </span>\n </ng-template>\n\n <ng-template\n [ngTemplateOutlet]=\"tableCellTemplate || tableCellTemplateInput || defaultTableCellTemplate\"\n [ngTemplateOutletContext]=\"{ value: cell.value, cell: cell }\">\n </ng-template>\n </ng-container>\n <ng-container *ngIf=\"!cellData.hasValue\"> </ng-container>\n </ng-container>\n </td>\n </ng-container>\n </tr>\n </table>\n </ng-container>\n\n <ng-container *ngIf=\"!data.pivotTables?.length || (data.pivotData | pivotDataEmpty)\">\n <ng-template #defaultEmptyTablesTemplate>\n <div> </div>\n </ng-template>\n\n <ng-template\n [ngTemplateOutlet]=\"emptyTablesTemplate || emptyTablesTemplateInput || defaultEmptyTablesTemplate\"\n [ngTemplateOutletContext]=\"{ }\">\n </ng-template>\n </ng-container>\n</ng-container>\n", styles: [".table{border-spacing:0;border-collapse:separate}.pivot-data-cell,.pivot-data-group-cell,.pivot-column-header-cell,.pivot-column-group-header-cell{text-align:right}.pivot-column-header-cell,.pivot-row-attribute-header-cell,.pivot-row-header-cell,.pivot-data-group-cell,.pivot-row-group-header-cell,.pivot-column-group-header-cell{font-weight:700;border:1px #f8f9fa solid}.pivot-row-header-cell.sticky-start,.pivot-row-group-header-cell.sticky-start,.pivot-empty-cell.sticky-start{position:sticky!important;width:150px;min-width:150px;max-width:150px!important;z-index:9}.pivot-row-header-cell.expandable,.pivot-row-group-header-cell.expandable{position:absolute;cursor:pointer;padding-right:15px}.pivot-row-header-cell.expandable:after,.pivot-row-group-header-cell.expandable:after{content:\"\";position:absolute;top:50%;right:5px;width:0;height:0;border-top:5px solid transparent;border-bottom:5px solid transparent;border-left:5px solid #253746;transform:translateY(-50%)}.pivot-row-header-cell.expandable.expanded.expandable-rows:after,.pivot-row-header-cell.expandable.expanded:after,.pivot-row-group-header-cell.expandable.expanded.expandable-rows:after,.pivot-row-group-header-cell.expandable.expanded:after{border-left:5px solid transparent;border-right:5px solid transparent;border-top:5px solid #253746;border-bottom:0}.pivot-column-header-cell.sticky-top,.pivot-row-attribute-header-cell.sticky-top,.pivot-column-group-header-cell.sticky-top,.pivot-empty-cell.sticky-top{position:sticky!important;z-index:10}.pivot-row-attribute-header-cell.sticky-top,.pivot-row-attribute-header-cell.sticky-start{position:sticky!important;z-index:11}.pivot-empty-cell{background:#fff;border:1px white solid}.pivot-empty-cell.sticky-top{z-index:11}.cell.color{padding:0!important;height:30px}.cell.color .summary{padding-left:.5rem}\n"] }]
|
|
99
|
-
}], propDecorators: { collections: [{
|
|
100
|
-
type: Input
|
|
101
|
-
}], data: [{
|
|
102
|
-
type: Input
|
|
103
|
-
}], linkTypes: [{
|
|
104
|
-
type: Input
|
|
105
|
-
}], query: [{
|
|
106
|
-
type: Input
|
|
107
|
-
}], constraintData: [{
|
|
108
|
-
type: Input
|
|
109
|
-
}], config: [{
|
|
110
|
-
type: Input
|
|
111
|
-
}], transform: [{
|
|
112
|
-
type: Input
|
|
113
|
-
}], emptyTablesTemplateInput: [{
|
|
114
|
-
type: Input
|
|
115
|
-
}], tableCellTemplateInput: [{
|
|
116
|
-
type: Input
|
|
117
|
-
}], initiallyCollapsed: [{
|
|
118
|
-
type: Input
|
|
119
|
-
}], cellClick: [{
|
|
120
|
-
type: Output
|
|
121
|
-
}], pivotDataChange: [{
|
|
122
|
-
type: Output
|
|
123
|
-
}], pivotTablesChange: [{
|
|
124
|
-
type: Output
|
|
125
|
-
}], emptyTablesTemplate: [{
|
|
126
|
-
type: ContentChild,
|
|
127
|
-
args: [LmrEmptyTablesTemplateDirective, { read: TemplateRef }]
|
|
128
|
-
}], tableCellTemplate: [{
|
|
129
|
-
type: ContentChild,
|
|
130
|
-
args: [LmrTableCellTemplateDirective, { read: TemplateRef }]
|
|
131
|
-
}] } });
|
|
132
|
-
function shouldResetState(changes) {
|
|
133
|
-
if (changes['config']) {
|
|
134
|
-
const previousValue = changes['config'].previousValue;
|
|
135
|
-
const currentValue = changes['config'].currentValue;
|
|
136
|
-
return isPivotConfigChanged(previousValue, currentValue);
|
|
137
|
-
}
|
|
138
|
-
return false;
|
|
139
|
-
}
|
|
140
|
-
function tablesAreVeryDifferent(t1, t2) {
|
|
141
|
-
if ((t1 || []).length !== (t2 || []).length) {
|
|
142
|
-
return true;
|
|
143
|
-
}
|
|
144
|
-
// row numbers are different
|
|
145
|
-
return (t1 || []).some((t, index) => t.cells?.length !== t2[index].cells?.length);
|
|
146
|
-
}
|
|
147
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lmr-pivot-table.component.js","sourceRoot":"","sources":["../../../../projects/lmr-pivot-table/src/lib/lmr-pivot-table.component.ts","../../../../projects/lmr-pivot-table/src/lib/lmr-pivot-table.component.html"],"names":[],"mappings":"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","sourcesContent":["import {ChangeDetectionStrategy, Component, ContentChild, EventEmitter, Input, OnChanges, OnInit, Output, SimpleChanges, TemplateRef} from '@angular/core';\nimport {LmrPivotConfig, LmrPivotTransform} from './util/lmr-pivot-config';\nimport {Collection, ConstraintData, DocumentsAndLinksData, LinkType, Query} from '@lumeer/data-filters';\nimport {PivotDataConverter} from './util/pivot-data-converter';\nimport {LmrPivotData} from './util/lmr-pivot-data';\nimport {asyncScheduler, BehaviorSubject, filter, map, Observable, tap, throttleTime} from 'rxjs';\nimport {LmrPivotTable, LmrPivotTableCell} from './util/lmr-pivot-table';\nimport {PivotTableConverter} from './util/pivot-table-converter';\nimport {LmrEmptyTablesTemplateDirective, LmrTableCellTemplateDirective} from './directives/lmr-templates.directive';\nimport {collapseAllCells, isCellExpandable, LmrPivotTableState, toggleExpanded} from './util/lmr-pivot-state';\nimport {isPivotConfigChanged} from './util/pivot-util';\n\ninterface Data {\n  collections: Collection[];\n  linkTypes: LinkType[];\n  data: DocumentsAndLinksData;\n  query: Query;\n  constraintData: ConstraintData;\n  config: LmrPivotConfig;\n  transform: LmrPivotTransform;\n}\n\n@Component({\n  selector: 'lmr-pivot-table',\n  templateUrl: 'lmr-pivot-table.component.html',\n  styleUrls: ['./lmr-pivot-table.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class LmrPivotTableComponent implements OnInit, OnChanges {\n\n  @Input()\n  public collections: Collection[];\n\n  @Input()\n  public data: DocumentsAndLinksData;\n\n  @Input()\n  public linkTypes: LinkType[];\n\n  @Input()\n  public query: Query;\n\n  @Input()\n  public constraintData: ConstraintData;\n\n  @Input()\n  public config: LmrPivotConfig;\n\n  @Input()\n  public transform: LmrPivotTransform;\n\n  @Input()\n  public emptyTablesTemplateInput: TemplateRef<any>;\n\n  @Input()\n  public tableCellTemplateInput: TemplateRef<any>;\n\n  @Input()\n  public initiallyCollapsed: boolean;\n\n  @Output()\n  public cellClick = new EventEmitter<{cell: LmrPivotTableCell; tableIndex: number; rowIndex: number; columnIndex: number }>();\n\n  @Output()\n  public pivotDataChange = new EventEmitter<LmrPivotData>();\n\n  @Output()\n  public pivotTablesChange = new EventEmitter<LmrPivotTable[]>();\n\n  @ContentChild(LmrEmptyTablesTemplateDirective, {read: TemplateRef}) emptyTablesTemplate: TemplateRef<any>;\n  @ContentChild(LmrTableCellTemplateDirective, {read: TemplateRef}) tableCellTemplate: TemplateRef<any>;\n\n  private readonly pivotTransformer = new PivotDataConverter();\n  private readonly pivotTableConverter: PivotTableConverter = new PivotTableConverter();\n  public readonly stickyColumnWidth = 150;\n  public readonly stickyColumnHeight = 40;\n\n  private dataSubject$ = new BehaviorSubject<Data>(null);\n  private currentTables: LmrPivotTable[];\n\n  public pivotData$: Observable<LmrPivotData>;\n  public pivotTables$: Observable<LmrPivotTable[]>;\n  public pivotStates$ = new BehaviorSubject<LmrPivotTableState[]>([]);\n\n  public ngOnInit() {\n    const observable = this.dataSubject$.pipe(filter(data => !!data));\n\n    this.pivotData$ = observable.pipe(\n      throttleTime(200, asyncScheduler, {trailing: true, leading: true}),\n      map(data => this.handleData(data)),\n      tap(data => this.onPivotDataChange(data)),\n    );\n\n    this.pivotTables$ = this.pivotData$.pipe(\n      map(data => this.pivotTableConverter.createTables(data, this.transform)),\n      tap(tables => this.onPivotTablesChange(tables))\n    );\n  }\n\n  private onPivotDataChange(data: LmrPivotData) {\n    this.pivotDataChange.emit(data);\n  }\n\n  private onPivotTablesChange(tables: LmrPivotTable[]) {\n    if (this.initiallyCollapsed && tablesAreVeryDifferent(this.currentTables, tables)) {\n      this.pivotStates$.next(tables.map(table => collapseAllCells(table)));\n    }\n\n    this.currentTables = tables;\n    this.pivotTablesChange.emit(tables);\n  }\n\n  private handleData(data: Data): LmrPivotData {\n    return this.pivotTransformer.createData(\n      data.config,\n      data.transform,\n      data.collections,\n      data.linkTypes,\n      data.data,\n      data.query,\n      data.constraintData\n    );\n  }\n\n  public ngOnChanges(changes: SimpleChanges) {\n    if (shouldResetState(changes)) {\n      this.resetState();\n    }\n    this.dataSubject$.next({\n      config: this.config,\n      transform: this.transform,\n      collections: this.collections,\n      linkTypes: this.linkTypes,\n      data: this.data,\n      query: this.query,\n      constraintData: this.constraintData,\n    });\n  }\n\n  private resetState() {\n    // this.pivotStates$.next([]);\n  }\n\n  public onCellClick(cell: LmrPivotTableCell, row: LmrPivotTableCell[], tableIndex: number, rowIndex: number, columnIndex: number) {\n    if (isCellExpandable(cell)) {\n      const oldState = this.pivotStates$.value[tableIndex]\n      const newState = toggleExpanded(cell, columnIndex, oldState)\n      this.setState(tableIndex, newState)\n    } else if (cell?.isHeader || cell?.isValue) {\n      const headerCell = cell.isHeader ? cell : row.find(c => c.isHeader)\n      this.cellClick.emit({cell, tableIndex, rowIndex: headerCell?.originalRowIndex || rowIndex, columnIndex})\n    }\n  }\n\n  private setState(index: number, state: LmrPivotTableState) {\n    const statesCopy = [...(this.pivotStates$.value || [])];\n    statesCopy.splice(index, 1, state)\n    this.pivotStates$.next(statesCopy)\n  }\n}\n\nfunction shouldResetState(changes: SimpleChanges): boolean {\n  if (changes['config']) {\n    const previousValue = changes['config'].previousValue as LmrPivotConfig\n    const currentValue = changes['config'].currentValue as LmrPivotConfig\n    return isPivotConfigChanged(previousValue, currentValue)\n  }\n  return false;\n}\n\nfunction tablesAreVeryDifferent(t1: LmrPivotTable[], t2: LmrPivotTable[]): boolean {\n  if ((t1 || []).length !== (t2 || []).length) {\n    return true\n  }\n  // row numbers are different\n  return (t1 || []).some((t, index) => t.cells?.length !== t2[index].cells?.length);\n}\n","<ng-container *ngIf=\"{pivotTables: pivotTables$ | async, pivotStates: pivotStates$ | async, pivotData: pivotData$ | async} as data\">\n  <ng-container *ngIf=\"data.pivotTables?.length && !(data.pivotData | pivotDataEmpty)\">\n    <table *ngFor=\"let pivotTable of data.pivotTables; let first; let tableIndex = index\"\n           class=\"table table-without-padding table-borderless table-md\"\n           [class.mt-4]=\"!first\">\n      <tr *ngFor=\"let rowCells of pivotTable | filterVisibleCells:data.pivotStates[tableIndex]; let rowIndex = index\">\n\n        <ng-container *ngFor=\"let cell of rowCells; let cellIndex = index\">\n          <td *ngIf=\"cell && {hasValue: cell | cellHasValue} as cellData\"\n              class=\"cell position-relative {{cell.constraint ? (cell.constraint.type | lowercase) : ''}} text-truncate\"\n              [class.expandable]=\"cell | isCellExpandable\"\n              [class.expandable-rows]=\"cell | isCellRowsExpandable\"\n              [class.expanded]=\"cell | isCellExpanded:cellIndex:data.pivotStates[tableIndex]\"\n              [class.sticky-start]=\"cell.stickyStart\"\n              [class.sticky-top]=\"cell.stickyTop\"\n              [rowSpan]=\"cell.rowSpan\"\n              [colSpan]=\"cell.colSpan\"\n              [style.top.px]=\"cell.stickyTop ? (rowIndex * stickyColumnHeight) : undefined\"\n              [style.left.px]=\"cell.stickyStart ? (cellIndex * stickyColumnWidth) : undefined\"\n              [style.width.px]=\"cell.stickyStart ? (cell.colSpan * stickyColumnWidth) : undefined\"\n              [style.max-width.px]=\"cell.stickyStart ? (cell.colSpan * stickyColumnWidth) : 300\"\n              [style.min-width.px]=\"cell.stickyStart ? (cell.colSpan * stickyColumnWidth) : undefined\"\n              [ngClass]=\"cell.cssClass\"\n              [style.background]=\"cell.background\"\n              [style.color]=\"cell.background && (cell.background | contrastColor)\"\n              (click)=\"onCellClick(cell, rowCells, tableIndex, rowIndex, cellIndex)\">\n            <ng-container *ngIf=\"cell.summary\">\n              <ng-container *ngIf=\"cellData.hasValue\">\n                <div class=\"d-flex align-items-center h-100\">\n                  <span class=\"summary me-2\">{{cell.summary}}</span>\n                  <ng-template #defaultTableCellTemplate>\n                    <span class=\"flex-grow-1 h-100 text-truncate d-inline-block\">\n                      {{ cell.value }}\n                    </span>\n                  </ng-template>\n\n                  <ng-template\n                    [ngTemplateOutlet]=\"tableCellTemplate || tableCellTemplateInput || defaultTableCellTemplate\"\n                    [ngTemplateOutletContext]=\"{ value: cell.value, cell: cell }\">\n                  </ng-template>\n                </div>\n              </ng-container>\n              <ng-container *ngIf=\"!cellData.hasValue\">\n                <span class=\"d-block\" style=\"overflow: hidden; white-space: normal\">{{cell.summary}}</span>\n              </ng-container>\n            </ng-container>\n            <ng-container *ngIf=\"!cell.summary\">\n              <ng-container *ngIf=\"cellData.hasValue\">\n                <ng-template #defaultTableCellTemplate>\n                  <span class=\"d-block\" style=\"overflow: hidden; white-space: normal; cursor: pointer\">\n                    {{ cell.value }}\n                  </span>\n                </ng-template>\n\n                <ng-template\n                  [ngTemplateOutlet]=\"tableCellTemplate || tableCellTemplateInput || defaultTableCellTemplate\"\n                  [ngTemplateOutletContext]=\"{ value: cell.value, cell: cell }\">\n                </ng-template>\n              </ng-container>\n              <ng-container *ngIf=\"!cellData.hasValue\">&nbsp;</ng-container>\n            </ng-container>\n          </td>\n        </ng-container>\n      </tr>\n    </table>\n  </ng-container>\n\n  <ng-container *ngIf=\"!data.pivotTables?.length || (data.pivotData | pivotDataEmpty)\">\n    <ng-template #defaultEmptyTablesTemplate>\n      <div>&nbsp;</div>\n    </ng-template>\n\n    <ng-template\n      [ngTemplateOutlet]=\"emptyTablesTemplate || emptyTablesTemplateInput || defaultEmptyTablesTemplate\"\n      [ngTemplateOutletContext]=\"{  }\">\n    </ng-template>\n  </ng-container>\n</ng-container>\n"]}
|