ngx-edu-sharing-metaqs2 0.9.53 → 0.9.54
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/README.md +1 -14
- package/fesm2022/ngx-edu-sharing-metaqs2.mjs +195 -197
- package/fesm2022/ngx-edu-sharing-metaqs2.mjs.map +1 -1
- package/lib/components/collection-issues/collection-issues.component.d.ts +1 -1
- package/lib/components/donut-chart-tooltip/donut-chart-tooltip.component.d.ts +1 -1
- package/lib/components/filter/quality-matrix-filter.component.d.ts +4 -6
- package/lib/components/material-issues/material-issues.component.d.ts +1 -1
- package/lib/components/node-entry/node-entry.component.d.ts +1 -1
- package/lib/components/node-list/node-list.component.d.ts +1 -1
- package/lib/components/quality-matrix/quality_matrix.d.ts +1 -3
- package/lib/counts-with-history/counts-with-history.component.d.ts +1 -1
- package/lib/ng-meta-widgets-lib.module.d.ts +34 -37
- package/lib/node-image-url.pipe.d.ts +1 -1
- package/lib/wrap-observable.pipe.d.ts +1 -1
- package/package.json +6 -8
- package/esm2022/lib/collection-count-history/collection-count-history.component.mjs +0 -131
- package/esm2022/lib/collection-count-history/monthpicker/monthpicker.component.mjs +0 -114
- package/esm2022/lib/components/collection-issues/collection-issues.component.mjs +0 -23
- package/esm2022/lib/components/donut-chart/donut-chart.component.mjs +0 -85
- package/esm2022/lib/components/donut-chart/donut-chart.model.mjs +0 -2
- package/esm2022/lib/components/donut-chart/donut-chart.pipe.mjs +0 -50
- package/esm2022/lib/components/donut-chart-tooltip/donut-chart-tooltip.component.mjs +0 -79
- package/esm2022/lib/components/editorial-link-service/editorial-link.service.mjs +0 -169
- package/esm2022/lib/components/filter/datepicker/datepicker.component.mjs +0 -99
- package/esm2022/lib/components/filter/quality-matrix-filter.component.mjs +0 -47
- package/esm2022/lib/components/loading_indicator/overlay/overlay.service.mjs +0 -41
- package/esm2022/lib/components/loading_indicator/progress-spinner/progress-spinner.component.mjs +0 -65
- package/esm2022/lib/components/material-issues/material-issues.component.mjs +0 -23
- package/esm2022/lib/components/node-entry/node-entry.component.mjs +0 -35
- package/esm2022/lib/components/node-list/node-list.component.mjs +0 -112
- package/esm2022/lib/components/quality-matrix/quality_matrix.mjs +0 -413
- package/esm2022/lib/components/quality-matrix/scroll-marker.directive.mjs +0 -17
- package/esm2022/lib/config-helper.service.mjs +0 -32
- package/esm2022/lib/core/svg-icons.service.mjs +0 -44
- package/esm2022/lib/core/tooltip.service.mjs +0 -146
- package/esm2022/lib/counts-with-history/counts-with-history.component.mjs +0 -203
- package/esm2022/lib/java-api/api/api.mjs +0 -12
- package/esm2022/lib/java-api/api/authProxyController.service.mjs +0 -107
- package/esm2022/lib/java-api/api/collectionAPI.service.mjs +0 -409
- package/esm2022/lib/java-api/api/editorsAPI.service.mjs +0 -157
- package/esm2022/lib/java-api/api/filterAPI.service.mjs +0 -237
- package/esm2022/lib/java-api/api/replicationSourceAPI.service.mjs +0 -230
- package/esm2022/lib/java-api/api.base.service.mjs +0 -66
- package/esm2022/lib/java-api/api.module.mjs +0 -40
- package/esm2022/lib/java-api/configuration.mjs +0 -103
- package/esm2022/lib/java-api/encoder.mjs +0 -19
- package/esm2022/lib/java-api/index.mjs +0 -7
- package/esm2022/lib/java-api/model/collectionWithMissingAttributes.mjs +0 -11
- package/esm2022/lib/java-api/model/count.mjs +0 -11
- package/esm2022/lib/java-api/model/eduCollection.mjs +0 -11
- package/esm2022/lib/java-api/model/filter.mjs +0 -2
- package/esm2022/lib/java-api/model/filterValue.mjs +0 -11
- package/esm2022/lib/java-api/model/materialCountDto.mjs +0 -2
- package/esm2022/lib/java-api/model/materialCountFilter.mjs +0 -11
- package/esm2022/lib/java-api/model/materialWithMissingAttributes.mjs +0 -11
- package/esm2022/lib/java-api/model/matrixRowWithCounts.mjs +0 -2
- package/esm2022/lib/java-api/model/matrixWithCounts.mjs +0 -2
- package/esm2022/lib/java-api/model/missingAttributeResult.mjs +0 -2
- package/esm2022/lib/java-api/model/models.mjs +0 -16
- package/esm2022/lib/java-api/model/qualityMatrix.mjs +0 -2
- package/esm2022/lib/java-api/model/qualityMatrixHeader.mjs +0 -11
- package/esm2022/lib/java-api/model/qualityMatrixReplicationSourceCounts.mjs +0 -11
- package/esm2022/lib/java-api/model/qualityMatrixRow.mjs +0 -2
- package/esm2022/lib/java-api/param.mjs +0 -2
- package/esm2022/lib/java-api/variables.mjs +0 -9
- package/esm2022/lib/meta-api.service.mjs +0 -77
- package/esm2022/lib/ng-meta-widgets-lib.module.mjs +0 -198
- package/esm2022/lib/node-image-url.pipe.mjs +0 -29
- package/esm2022/lib/tree-collection-details/tree-collection-details.component.mjs +0 -87
- package/esm2022/lib/tree-license/tree-license.component.mjs +0 -136
- package/esm2022/lib/tree-search-counts/inline-worker.mjs +0 -102
- package/esm2022/lib/tree-search-counts/tree-search-counts.component.mjs +0 -209
- package/esm2022/lib/wrap-observable.pipe.mjs +0 -21
- package/esm2022/ngx-edu-sharing-metaqs2.mjs +0 -5
- package/esm2022/public-api.mjs +0 -18
- package/esm2022/web-components.mjs +0 -36
|
@@ -1,413 +0,0 @@
|
|
|
1
|
-
import { Component, computed, DestroyRef, ElementRef, HostListener, inject, Input, signal, ViewChild, ViewChildren, } from '@angular/core';
|
|
2
|
-
import { BehaviorSubject, combineLatest, of, Subject } from 'rxjs';
|
|
3
|
-
import { filter, finalize, map, switchMap, take, tap, throttleTime } from 'rxjs/operators';
|
|
4
|
-
import { MetaApiService } from '../../meta-api.service';
|
|
5
|
-
import { FormControl, FormGroup, FormRecord, ReactiveFormsModule } from '@angular/forms';
|
|
6
|
-
import { takeUntilDestroyed, toSignal } from '@angular/core/rxjs-interop';
|
|
7
|
-
import { EditorialLinkService } from '../editorial-link-service/editorial-link.service';
|
|
8
|
-
import { DateTime } from 'luxon';
|
|
9
|
-
import { KeyValuePipe, NgClass, NgForOf, NgIf } from '@angular/common';
|
|
10
|
-
import { Overlay } from '@angular/cdk/overlay';
|
|
11
|
-
import { DonutChartTooltipComponent, transformDonutChartData, } from '../donut-chart-tooltip/donut-chart-tooltip.component';
|
|
12
|
-
import { centerPositionStrategy, TooltipService } from '../../core/tooltip.service';
|
|
13
|
-
import { MatCard, MatCardContent, MatCardHeader, MatCardTitle } from '@angular/material/card';
|
|
14
|
-
import { QualityMatrixFilterComponent } from '../filter/quality-matrix-filter.component';
|
|
15
|
-
import { TranslateModule } from '@ngx-translate/core';
|
|
16
|
-
import { MatIcon } from '@angular/material/icon';
|
|
17
|
-
import { DatepickerComponent } from '../filter/datepicker/datepicker.component';
|
|
18
|
-
import { MatSlideToggle } from '@angular/material/slide-toggle';
|
|
19
|
-
import { ProgressSpinnerComponent } from '../loading_indicator/progress-spinner/progress-spinner.component';
|
|
20
|
-
import { MatCell, MatCellDef, MatColumnDef, MatHeaderCell, MatHeaderCellDef, MatHeaderRow, MatHeaderRowDef, MatRow, MatRowDef, MatTable, } from '@angular/material/table';
|
|
21
|
-
import { MatTooltip } from '@angular/material/tooltip';
|
|
22
|
-
import { MatButton, MatIconButton } from '@angular/material/button';
|
|
23
|
-
import { DonutChartComponent } from '../donut-chart/donut-chart.component';
|
|
24
|
-
import { ScrollMarkerDirective } from './scroll-marker.directive';
|
|
25
|
-
import * as i0 from "@angular/core";
|
|
26
|
-
import * as i1 from "@ngx-translate/core";
|
|
27
|
-
import * as i2 from "@angular/forms";
|
|
28
|
-
const openCloseTimeout = 150;
|
|
29
|
-
export class QualityMatrixComponent {
|
|
30
|
-
constructor() {
|
|
31
|
-
this.ref = inject(ElementRef);
|
|
32
|
-
this.destroyRef = inject(DestroyRef);
|
|
33
|
-
this.metaApi = inject(MetaApiService);
|
|
34
|
-
this.linkService = inject(EditorialLinkService);
|
|
35
|
-
this.overlay = inject(Overlay);
|
|
36
|
-
this.tooltipService = inject(TooltipService);
|
|
37
|
-
this.scrollLeft = signal(0);
|
|
38
|
-
this.scrollWidth = signal(0);
|
|
39
|
-
this.isLeftScrollable = computed(() => this.scrollLeft() > 0);
|
|
40
|
-
this.isRightScrollable = computed(() => {
|
|
41
|
-
const viewport = globalThis.visualViewport;
|
|
42
|
-
if (!viewport) {
|
|
43
|
-
return;
|
|
44
|
-
}
|
|
45
|
-
const viewportWidth = viewport.width;
|
|
46
|
-
return this.scrollWidth() > viewportWidth && this.scrollLeft() + viewportWidth < this.scrollWidth();
|
|
47
|
-
});
|
|
48
|
-
this.filteredColumns = computed(() => {
|
|
49
|
-
const allColumns = this.allColumns();
|
|
50
|
-
const isOnlySourcesWithMaterialEnabled = this.isOnlySourcesWithMaterialEnabled();
|
|
51
|
-
if (isOnlySourcesWithMaterialEnabled) {
|
|
52
|
-
return allColumns.filter((c) => c.total > 0);
|
|
53
|
-
}
|
|
54
|
-
else {
|
|
55
|
-
return allColumns;
|
|
56
|
-
}
|
|
57
|
-
});
|
|
58
|
-
this.datatableColumns = computed(() => {
|
|
59
|
-
return this.filteredColumns()
|
|
60
|
-
.filter((c) => c.level !== 0)
|
|
61
|
-
.map((c) => c.id);
|
|
62
|
-
});
|
|
63
|
-
this.isHistoricalDataEnabledFormField = new FormControl({ value: false, disabled: true });
|
|
64
|
-
this.isHistoricalDataEnabled = toSignal(this.isHistoricalDataEnabledFormField.valueChanges);
|
|
65
|
-
this.isOnlySourcesWithMaterialEnabledFormField = new FormControl({ value: true, disabled: true });
|
|
66
|
-
this.isOnlySourcesWithMaterialEnabled = toSignal(this.isOnlySourcesWithMaterialEnabledFormField.valueChanges);
|
|
67
|
-
this.categoryControls = new FormRecord({});
|
|
68
|
-
this.loadingCount = signal(0);
|
|
69
|
-
this.isLoading = computed(() => this.loadingCount() > 0);
|
|
70
|
-
this.categoryFilterValues = new Map();
|
|
71
|
-
this.refresh$ = new Subject();
|
|
72
|
-
this.recentQualityMatrix$ = new BehaviorSubject({ columns: [], rows: [] });
|
|
73
|
-
this.recentQualityMatrix = toSignal(this.recentQualityMatrix$, {
|
|
74
|
-
initialValue: { columns: [], rows: [] },
|
|
75
|
-
});
|
|
76
|
-
this.pastQualityMatrix$ = new BehaviorSubject({ columns: [], rows: [] });
|
|
77
|
-
this.range = new FormGroup({
|
|
78
|
-
start: new FormControl(),
|
|
79
|
-
end: new FormControl(),
|
|
80
|
-
});
|
|
81
|
-
this.tooltips = new Map();
|
|
82
|
-
this.DateTime = DateTime;
|
|
83
|
-
this.pageTitle = 'Quality Matrix';
|
|
84
|
-
this.allColumns = computed(() => this.recentQualityMatrix()?.columns.toSorted((a, b) => a.label.localeCompare(b.label)));
|
|
85
|
-
this.sourceColumns = computed(() => this.datatableColumns().map((c) => c + '_source'));
|
|
86
|
-
this.currentColumns = computed(() => this.datatableColumns().map((c) => c + '_current'));
|
|
87
|
-
this.pastColumns = computed(() => this.datatableColumns().map((c) => c + '_past'));
|
|
88
|
-
this.allDataColumns = computed(() => {
|
|
89
|
-
const pastColumns = this.pastColumns();
|
|
90
|
-
const currentColumns = this.currentColumns();
|
|
91
|
-
if (this.isHistoricalDataEnabled()) {
|
|
92
|
-
return currentColumns.flatMap((e, i) => [pastColumns[i], e]);
|
|
93
|
-
}
|
|
94
|
-
return currentColumns;
|
|
95
|
-
});
|
|
96
|
-
}
|
|
97
|
-
ngAfterViewChecked() {
|
|
98
|
-
this.onScroll();
|
|
99
|
-
}
|
|
100
|
-
filterIdent(_index, item) {
|
|
101
|
-
return item.key;
|
|
102
|
-
}
|
|
103
|
-
loadData(categories, date) {
|
|
104
|
-
return of(undefined).pipe(tap(() => this.loadingCount.update((it) => it + 1)), switchMap(() => this.metaApi
|
|
105
|
-
.getQualityMatrixWithFiltersV2([
|
|
106
|
-
...this.transformCategoryFilters(categories),
|
|
107
|
-
{
|
|
108
|
-
field: 'asOf',
|
|
109
|
-
values: [
|
|
110
|
-
{
|
|
111
|
-
id: date.toISO({
|
|
112
|
-
includeOffset: false,
|
|
113
|
-
}),
|
|
114
|
-
label: '',
|
|
115
|
-
},
|
|
116
|
-
],
|
|
117
|
-
},
|
|
118
|
-
])
|
|
119
|
-
.pipe(take(1))), finalize(() => this.loadingCount.update((it) => it - 1)));
|
|
120
|
-
}
|
|
121
|
-
transformCategoryFilters(categories) {
|
|
122
|
-
return Object.entries(categories)
|
|
123
|
-
.filter((pair) => !!pair[1]?.length)
|
|
124
|
-
.map(([field, values]) => ({
|
|
125
|
-
field,
|
|
126
|
-
values: values.map((value) => ({ id: value, label: '' })),
|
|
127
|
-
}));
|
|
128
|
-
}
|
|
129
|
-
loadPastData(categories, date) {
|
|
130
|
-
return of(undefined).pipe(switchMap(() => this.loadData(categories, date)));
|
|
131
|
-
}
|
|
132
|
-
loadCurrentData(categories, date) {
|
|
133
|
-
return of(undefined).pipe(switchMap(() => this.loadData(categories, date)));
|
|
134
|
-
}
|
|
135
|
-
openEditLink(source, issue) {
|
|
136
|
-
return this.linkService.openByReplicationsourceAndIssueTypeWithFilters(source, issue, this.categoryControls.value);
|
|
137
|
-
}
|
|
138
|
-
// see https://github.com/angular/components/issues/8361#issuecomment-345804954
|
|
139
|
-
columnIdent(_index, col) {
|
|
140
|
-
return col.id;
|
|
141
|
-
}
|
|
142
|
-
refresh() {
|
|
143
|
-
this.refresh$.next();
|
|
144
|
-
}
|
|
145
|
-
getPastColumn(columnId) {
|
|
146
|
-
return this.pastQualityMatrix$.value?.columns.find((c) => c.id === columnId);
|
|
147
|
-
}
|
|
148
|
-
ngOnInit() {
|
|
149
|
-
this.loadingCount.update((it) => it + 1);
|
|
150
|
-
this.metaApi
|
|
151
|
-
.getCategoryFilters()
|
|
152
|
-
.pipe(take(1), tap((filters) => {
|
|
153
|
-
filters.forEach((filter) => {
|
|
154
|
-
// do not emit the event when setting the control to prevent unnecessary valueChanges events
|
|
155
|
-
this.categoryControls.addControl(filter.field, new FormControl([], { nonNullable: true }), {
|
|
156
|
-
emitEvent: false,
|
|
157
|
-
});
|
|
158
|
-
this.categoryFilterValues.set(filter.field, filter.values.toSorted((a, b) => {
|
|
159
|
-
return a.label.localeCompare(b.label);
|
|
160
|
-
}));
|
|
161
|
-
});
|
|
162
|
-
// emit the current value of the form group once to trigger the valueChanges observable
|
|
163
|
-
this.categoryControls.setValue(this.categoryControls.value, {
|
|
164
|
-
emitEvent: true,
|
|
165
|
-
});
|
|
166
|
-
}), takeUntilDestroyed(this.destroyRef), finalize(() => this.loadingCount.update((it) => it - 1)))
|
|
167
|
-
.subscribe();
|
|
168
|
-
this.loadingCount.update((it) => it + 1);
|
|
169
|
-
this.metaApi
|
|
170
|
-
.getTimerangeFilter()
|
|
171
|
-
.pipe(filter((rangeFilter) => !!rangeFilter), tap((rangeFilter) => {
|
|
172
|
-
const startDate = DateTime.fromISO(rangeFilter.values.find((v) => v.id === 'rangeStart')?.label, {
|
|
173
|
-
zone: 'utc',
|
|
174
|
-
});
|
|
175
|
-
const endDate = DateTime.fromISO(rangeFilter.values.find((v) => v.id === 'rangeEnd')?.label, {
|
|
176
|
-
zone: 'utc',
|
|
177
|
-
});
|
|
178
|
-
this.range.setControl('start', new FormControl(startDate, { nonNullable: true }), {
|
|
179
|
-
emitEvent: false,
|
|
180
|
-
});
|
|
181
|
-
this.range.setControl('end', new FormControl(endDate, { nonNullable: true }), {
|
|
182
|
-
emitEvent: true,
|
|
183
|
-
});
|
|
184
|
-
}), takeUntilDestroyed(this.destroyRef), finalize(() => this.loadingCount.update((it) => it - 1)))
|
|
185
|
-
.subscribe();
|
|
186
|
-
combineLatest([this.categoryControls.valueChanges, this.range.valueChanges, this.refresh$])
|
|
187
|
-
.pipe(filter((data) => !!data[0] && !!data[1].end?.isValid), throttleTime(500), map(([categories, range]) => [categories, range.end]), switchMap(([categories, end]) => this.loadCurrentData(categories, end)), tap((currentQm) => {
|
|
188
|
-
this.recentQualityMatrix$.next(currentQm);
|
|
189
|
-
}), takeUntilDestroyed(this.destroyRef))
|
|
190
|
-
.subscribe();
|
|
191
|
-
let pastDataSubscription;
|
|
192
|
-
const pastData$ = combineLatest([this.categoryControls.valueChanges, this.range.valueChanges, this.refresh$]).pipe(filter((data) => !!data[0] && !!data[1].start?.isValid), throttleTime(500), map(([categories, range]) => [categories, range.start]), switchMap(([categories, start]) => this.loadPastData(categories, start)), tap((pastQm) => {
|
|
193
|
-
this.pastQualityMatrix$.next(pastQm);
|
|
194
|
-
}), takeUntilDestroyed(this.destroyRef));
|
|
195
|
-
this.isHistoricalDataEnabledFormField.valueChanges
|
|
196
|
-
.pipe(tap((isEnabled) => {
|
|
197
|
-
if (isEnabled) {
|
|
198
|
-
pastDataSubscription = pastData$.subscribe();
|
|
199
|
-
this.categoryControls.setValue(this.categoryControls.value, {
|
|
200
|
-
emitEvent: true,
|
|
201
|
-
});
|
|
202
|
-
this.range.setValue(this.range.value, {
|
|
203
|
-
emitEvent: true,
|
|
204
|
-
});
|
|
205
|
-
this.refresh$.next();
|
|
206
|
-
}
|
|
207
|
-
else {
|
|
208
|
-
pastDataSubscription?.unsubscribe();
|
|
209
|
-
}
|
|
210
|
-
}), takeUntilDestroyed(this.destroyRef))
|
|
211
|
-
.subscribe();
|
|
212
|
-
this.refresh$.next();
|
|
213
|
-
this.isOnlySourcesWithMaterialEnabledFormField.setValue(this.isOnlySourcesWithMaterialEnabledFormField.value, {
|
|
214
|
-
emitEvent: true,
|
|
215
|
-
});
|
|
216
|
-
}
|
|
217
|
-
getTrend(row, column) {
|
|
218
|
-
if (this.pastQualityMatrix$.value === null || this.pastQualityMatrix$.value.columns.length === 0) {
|
|
219
|
-
return {};
|
|
220
|
-
}
|
|
221
|
-
const past_row = this.pastQualityMatrix$?.value.rows.find((pr) => pr.meta.id == row.meta.id);
|
|
222
|
-
const past_column = this.pastQualityMatrix$.value.columns.find((c) => c.id == column.id);
|
|
223
|
-
if (!past_row || !past_column) {
|
|
224
|
-
return {};
|
|
225
|
-
}
|
|
226
|
-
const delta = Math.floor((100 * row.counts[column.id].sufficient) / column.total -
|
|
227
|
-
(100 * past_row.counts[column.id].sufficient) / past_column.total);
|
|
228
|
-
const trend = delta === 0 ? 'trending_flat' : delta < 0 ? 'trending_down' : 'trending_up';
|
|
229
|
-
return { delta, trend };
|
|
230
|
-
}
|
|
231
|
-
getDonutSlices(row, column) {
|
|
232
|
-
const sufficientTotal = row.counts[column.id].sufficient;
|
|
233
|
-
const insufficientTotal = row.counts[column.id].insufficient;
|
|
234
|
-
const missingTotal = column.total - sufficientTotal - insufficientTotal;
|
|
235
|
-
let sufficientFillRate = 0;
|
|
236
|
-
let insufficientFillRate = 0;
|
|
237
|
-
//Der Wert berechnet sich aus dem total_count - without_description_count.
|
|
238
|
-
if (column.total !== 0) {
|
|
239
|
-
sufficientFillRate = Math.round((100 * sufficientTotal) / column.total);
|
|
240
|
-
insufficientFillRate = Math.round((100 * insufficientTotal) / column.total);
|
|
241
|
-
}
|
|
242
|
-
const rest = 100 - (sufficientFillRate + insufficientFillRate); // a.k.a. "missing"
|
|
243
|
-
return [
|
|
244
|
-
{
|
|
245
|
-
id: 3,
|
|
246
|
-
percent: rest,
|
|
247
|
-
color: '#D8EDFC',
|
|
248
|
-
label: `${rest} % (${missingTotal})`,
|
|
249
|
-
verboseLabel: `${missingTotal} Metadatens\u00e4tze fehlen`,
|
|
250
|
-
},
|
|
251
|
-
{
|
|
252
|
-
id: 1,
|
|
253
|
-
percent: insufficientFillRate,
|
|
254
|
-
color: '#C20808',
|
|
255
|
-
label: `${insufficientFillRate} % (${insufficientTotal})`,
|
|
256
|
-
verboseLabel: `${insufficientTotal} Metadatens\u00e4tze sind unzureichend ausgef\u00fcllt`,
|
|
257
|
-
},
|
|
258
|
-
{
|
|
259
|
-
id: 2,
|
|
260
|
-
percent: sufficientFillRate,
|
|
261
|
-
color: '#4ABEFF',
|
|
262
|
-
label: `${sufficientFillRate} % (${sufficientTotal})`,
|
|
263
|
-
verboseLabel: `${sufficientTotal} Metadatens\u00e4tze sind gut ausgef\u00fcllt`,
|
|
264
|
-
},
|
|
265
|
-
];
|
|
266
|
-
}
|
|
267
|
-
showTooltip(row, column, cell) {
|
|
268
|
-
if (row.counts[column.id].sufficient + row.counts[column.id].insufficient === 0) {
|
|
269
|
-
return;
|
|
270
|
-
}
|
|
271
|
-
if (this.tooltips.has(cell)) {
|
|
272
|
-
this.tooltips.get(cell).cancelCloseAction();
|
|
273
|
-
return;
|
|
274
|
-
}
|
|
275
|
-
const tooltipRef = this.tooltipService.create(DonutChartTooltipComponent, {
|
|
276
|
-
closeTimeout: openCloseTimeout,
|
|
277
|
-
openTimeout: openCloseTimeout,
|
|
278
|
-
data: {
|
|
279
|
-
label: column.label,
|
|
280
|
-
data: transformDonutChartData(this.getDonutSlices(row, column)),
|
|
281
|
-
},
|
|
282
|
-
overlayConfig: {
|
|
283
|
-
hasBackdrop: false,
|
|
284
|
-
positionStrategy: this.overlay.position().flexibleConnectedTo(cell).withPositions(centerPositionStrategy),
|
|
285
|
-
disposeOnNavigation: true,
|
|
286
|
-
panelClass: 'donut-chart-tooltip',
|
|
287
|
-
},
|
|
288
|
-
onClose: () => {
|
|
289
|
-
this.tooltips.delete(cell);
|
|
290
|
-
},
|
|
291
|
-
});
|
|
292
|
-
tooltipRef.open();
|
|
293
|
-
this.tooltips.set(cell, tooltipRef);
|
|
294
|
-
}
|
|
295
|
-
hideTooltip(cell) {
|
|
296
|
-
this.tooltips.get(cell)?.markForClose();
|
|
297
|
-
}
|
|
298
|
-
findHorizontalScrollContainer(element) {
|
|
299
|
-
if (element.scrollWidth > element.clientWidth && ['auto', 'scroll'].includes(getComputedStyle(element).overflowX)) {
|
|
300
|
-
// find the nearest horizontally scrollable element
|
|
301
|
-
return element;
|
|
302
|
-
}
|
|
303
|
-
else {
|
|
304
|
-
if (!element.parentElement) {
|
|
305
|
-
return document.scrollingElement;
|
|
306
|
-
}
|
|
307
|
-
return this.findHorizontalScrollContainer(element.parentElement);
|
|
308
|
-
}
|
|
309
|
-
}
|
|
310
|
-
scrollToRight() {
|
|
311
|
-
const scrollingElement = this.findHorizontalScrollContainer(this.ref.nativeElement);
|
|
312
|
-
if (!scrollingElement) {
|
|
313
|
-
return;
|
|
314
|
-
}
|
|
315
|
-
const viewport = globalThis.visualViewport;
|
|
316
|
-
if (!viewport) {
|
|
317
|
-
return;
|
|
318
|
-
}
|
|
319
|
-
const viewportWidth = viewport.width;
|
|
320
|
-
const lastVisible = this.scrollMarkers.findLast((marker) => marker.element.nativeElement.getBoundingClientRect().left < viewportWidth);
|
|
321
|
-
if (!lastVisible) {
|
|
322
|
-
return;
|
|
323
|
-
}
|
|
324
|
-
const leftColumnWidth = this.issueTypeHeader?.nativeElement.getBoundingClientRect().width ?? 0;
|
|
325
|
-
scrollingElement.scrollBy({
|
|
326
|
-
left: lastVisible.element.nativeElement.getBoundingClientRect().left - leftColumnWidth,
|
|
327
|
-
behavior: 'smooth',
|
|
328
|
-
});
|
|
329
|
-
}
|
|
330
|
-
scrollToLeft() {
|
|
331
|
-
const scrollingElement = this.findHorizontalScrollContainer(this.ref.nativeElement);
|
|
332
|
-
if (!scrollingElement) {
|
|
333
|
-
return;
|
|
334
|
-
}
|
|
335
|
-
const viewport = globalThis.visualViewport;
|
|
336
|
-
if (!viewport) {
|
|
337
|
-
return;
|
|
338
|
-
}
|
|
339
|
-
const viewportWidth = viewport.width;
|
|
340
|
-
const leftColumnWidth = this.issueTypeHeader?.nativeElement.getBoundingClientRect().width ?? 0;
|
|
341
|
-
const firstVisible = this.scrollMarkers.find((marker) => marker.element.nativeElement.getBoundingClientRect().right > leftColumnWidth);
|
|
342
|
-
if (!firstVisible) {
|
|
343
|
-
return;
|
|
344
|
-
}
|
|
345
|
-
scrollingElement.scrollBy({
|
|
346
|
-
left: firstVisible.element.nativeElement.getBoundingClientRect().right - viewportWidth,
|
|
347
|
-
behavior: 'smooth',
|
|
348
|
-
});
|
|
349
|
-
}
|
|
350
|
-
onScroll() {
|
|
351
|
-
const scrollingElement = this.findHorizontalScrollContainer(this.ref.nativeElement);
|
|
352
|
-
if (!scrollingElement) {
|
|
353
|
-
return;
|
|
354
|
-
}
|
|
355
|
-
this.scrollLeft.set(scrollingElement.scrollLeft);
|
|
356
|
-
this.scrollWidth.set(scrollingElement.scrollWidth);
|
|
357
|
-
}
|
|
358
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: QualityMatrixComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
359
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: QualityMatrixComponent, isStandalone: true, selector: "metaqs2-quality-matrix-v2", inputs: { pageTitle: "pageTitle" }, host: { listeners: { "window:scroll": "onScroll()", "window:resize": "onScroll()" } }, providers: [TooltipService], viewQueries: [{ propertyName: "issueTypeHeader", first: true, predicate: ["issueType"], descendants: true, read: ElementRef, static: true }, { propertyName: "scrollMarkers", predicate: ScrollMarkerDirective, descendants: true }], ngImport: i0, template: "<mat-card appearance=\"raised\">\n <mat-card-header *ngIf=\"pageTitle\">\n <mat-card-title>Qualit\u00E4tsmetrik: {{ pageTitle | translate }}{{isLoading() ? \": Lade neue Daten.\" : \"\"}}</mat-card-title>\n </mat-card-header>\n <mat-card-content class=\"toolbar\">\n <metaqs2-qm-filter *ngFor=\"let filter of categoryControls.controls | keyvalue; trackBy: filterIdent\"\n [options]=\"categoryFilterValues.get(filter.key)\"\n [inputFormControl]=\"filter.value\"\n [label] = \"'filter.' + filter.key | translate\" />\n <div class=\"actionbar\">\n <mat-slide-toggle [disabled]=\"isLoading()\" labelPosition=\"before\" [formControl]=\"isOnlySourcesWithMaterialEnabledFormField\" id=\"onlySourcesWithMaterial\">\n show only sources with material\n </mat-slide-toggle>\n <div style=\"flex: 1 1 auto\"></div>\n <metaqs2-datepicker *ngIf=\"isHistoricalDataEnabled()\" [disabled]=\"isLoading()\" [inputGroup]=\"range\" />\n <mat-slide-toggle [disabled]=\"isLoading()\" labelPosition=\"before\" [formControl]=\"isHistoricalDataEnabledFormField\" id=\"showHistoricalData\">\n show historical data\n </mat-slide-toggle>\n <button\n mat-icon-button\n color=\"primary\"\n [disabled]=\"isLoading()\"\n (click)=\"refresh()\"\n matTooltip=\"Aktualisiere den IST-Stand\"\n matTooltipShowDelay=\"500\"\n >\n <mat-icon>refresh</mat-icon>\n </button>\n <div style=\"flex: 0 0 {{scroller.getBoundingClientRect().width}}px\"></div>\n <div class=\"scroll-controller\" #scroller> <!-- put the whole toolbar in a fixed container (MQS- -->\n <button mat-raised-button (click)=\"scrollToLeft()\" [disabled]=\"!isLeftScrollable()\" class=\"scroll-button\">\n <mat-icon style=\"margin: 0\">chevron_left</mat-icon>\n </button>\n <button mat-raised-button (click)=\"scrollToRight()\" [disabled]=\"!isRightScrollable()\" class=\"scroll-button\">\n <mat-icon style=\"margin: 0\">chevron_right</mat-icon>\n </button>\n </div>\n </div>\n </mat-card-content>\n</mat-card>\n<mat-card>\n <metaqs2-progress-spinner [displayProgressSpinner]=\"isLoading()\"></metaqs2-progress-spinner>\n <table [class.while-loading]=\"isLoading()\" mat-table [dataSource]=\"recentQualityMatrix().rows\" class=\"quality-matrix\">\n <!-- Define columns of table -->\n <!-- Row Header Column -->\n <ng-container matColumnDef=\"row-header\" sticky>\n <th rowspan=\"2\" mat-header-cell *matHeaderCellDef #issueType>Issue Type</th>\n <td\n mat-cell\n *matCellDef=\"let row\"\n matTooltip=\"{{row.meta.alt_label}}\"\n class=\"row-header mat-cell-level-{{row.meta.level + 1}}\"\n >\n {{\"quality_matrix.\" + row.meta.label | translate }}\n </td>\n </ng-container>\n <!-- one column for the source -->\n <ng-container *ngFor=\"let col of filteredColumns(); trackBy:columnIdent\" [matColumnDef]=\"col.id + '_source'\">\n <th [attr.colspan]=\"isHistoricalDataEnabled() ? 2 : 1\" metaqs2ScrollMarker mat-header-cell *matHeaderCellDef matTooltip=\"{{col.altLabel}}\">{{col.label}}</th>\n </ng-container>\n <!-- /source -->\n <!-- current Data Columns -->\n <ng-container *ngFor=\"let column of filteredColumns(); trackBy:columnIdent\" [matColumnDef]=\"column.id + '_current'\">\n <th class=\"recent-data-cell\" mat-header-cell *matHeaderCellDef matTooltip=\"{{column.altLabel}}\">{{ range.controls.end.value?.toLocaleString(DateTime.DATE_SHORT) }} <br>({{column.total}})</th>\n <td class=\"recent-data-cell\" mat-cell *matCellDef=\"let row\" #cell (mouseover)=\"showTooltip(row, column, cell)\" (mouseleave)=\"hideTooltip(cell)\">\n <a *ngIf=\"column.total > 0 else emptyDiv\" class=\"chart-container\" (click)=\"openEditLink( column.label, row.meta.label)\">\n <div>\n <metaqs2-donut-chart [data]=\"getDonutSlices(row, column)\" [borderSize]=\"25\"></metaqs2-donut-chart>\n </div>\n </a>\n <ng-template #emptyDiv>\n <div>-</div>\n </ng-template>\n </td>\n </ng-container>\n <!-- /current Data Columns -->\n <!-- past Data Columns -->\n <ng-container *ngIf=\"isHistoricalDataEnabled()\">\n <ng-container *ngFor=\"let column of filteredColumns(); trackBy:columnIdent\" [matColumnDef]=\"column.id + '_past'\">\n <th class=\"historic-data-cell\" mat-header-cell\n *matHeaderCellDef>{{ range.controls.start.value?.toLocaleString(DateTime.DATE_SHORT) }}\n <br>{{ getPastColumn(column.id)?.total }}\n </th>\n <td class=\"historic-data-cell\" mat-cell *matCellDef=\"let row\">\n <ng-container *ngIf=\"getTrend(row, column) as trend\">\n <span [ngClass]=\"trend.trend\">\n <mat-icon aria-hidden=\"false\" [attr.aria-label]=\"trend.trend\" [fontIcon]=\"trend.trend!\" />\n {{ trend.delta }}\n </span>\n </ng-container>\n\n </td>\n </ng-container>\n </ng-container>\n <!-- /past Data Columns -->\n\n <!-- Generate actual table -->\n <tr mat-header-row *matHeaderRowDef=\"['row-header'].concat(sourceColumns()); sticky: true;\"></tr>\n <tr mat-header-row *matHeaderRowDef=\"allDataColumns(); sticky: true;\" [hidden]=\"!isHistoricalDataEnabled()\"></tr>\n <tr mat-row *matRowDef=\"let row; columns: ['row-header'].concat(allDataColumns())\"></tr>\n </table>\n</mat-card>\n", styles: [".while-loading{filter:blur(2px)}mat-card-actions>mat-slide-toggle{margin-left:20px}.row{flex-grow:1;display:flex;align-items:center}table{width:100%;overflow-y:auto}table .scroll{flex-grow:1;overflow-y:auto}table th{vertical-align:top;padding:.375rem}table th:not(:last-child){border-right:1px solid #e4e4e4}table tr:first-child>th:first-child{text-align:left}table td:first-child{text-align:left}.cell{flex:1}.mat-mdc-cell{padding-top:5px}.mat-mdc-cell,.mat-mdc-header-cell{padding-left:5px;text-align:center}.mat-mdc-cell a.chart-container,.mat-mdc-header-cell a.chart-container{display:flex;justify-content:center}.mat-mdc-cell .trending_up,.mat-mdc-header-cell .trending_up{color:#4abeff}.mat-mdc-cell .trending_down,.mat-mdc-header-cell .trending_down{color:#c20808}.recent-data-cell,.mat-column-row-header{border-right:1px solid #e4e4e4}.mat-header-cell-label{color:#000;padding-left:30px;padding-right:30px;justify-content:left;border-bottom:1px solid #e4e4e4;font-style:normal;font-weight:500;font-size:110%;background:#ddf0fb;text-align:left}.mat-header-cell-label>div{display:flex;font-weight:700;font-style:normal;justify-content:left}.mat-header-cell-small{padding-left:5px;padding-right:5px;justify-content:left;font-style:normal;font-weight:400;font-size:105%;background:#ddf0fb;border-top:1px solid #e4e4e4;border-bottom:1px solid #e4e4e4}.mat-mdc-cell a{cursor:pointer;display:flex;align-items:center;justify-content:left}.mat-mdc-cell a>mat-icon{padding-right:5px;font-size:22px;height:20px;opacity:.75}.mat-mdc-cell a>mat-icon:hover,.mat-mdc-cell a>mat-icon:focus{text-decoration:none}.mat-mdc-cell .number{display:flex;height:47px;justify-content:left;align-items:center;padding-left:15px}.mat-mdc-cell .row-header{display:flex;height:47px;max-height:47px;justify-content:left;align-items:center;padding-left:15px}.mat-mdc-cell .row-header-sum{display:flex;height:47px;max-height:47px;justify-content:left;align-items:center;border-right:5px solid #bdbdbd}.mat-mdc-card{width:100%;box-shadow:none}.mat-mdc-card ::ng-deep mat-card-header{align-items:center}.mat-mdc-card ::ng-deep mat-card-header .mat-card-header-text{flex-grow:1;margin:0}.mat-mdc-card ::ng-deep mat-card-header .mat-card-header-text mat-card-title{padding-top:20px;font-size:120%}.mat-mdc-card ::ng-deep mat-card-header .mat-card-header-text mat-card-title .count{font-size:80%;color:#888}.mat-mdc-card .source-header{font-style:normal;font-weight:400;font-size:20px;height:30px;width:60%}.mat-mdc-card .source-header .robot-explanation{font-size:15px}.mat-mdc-card .scroll{flex-grow:1;overflow-y:auto}.mat-header-cell-scope{color:#545454}.mat-mdc-header-row{height:70px;max-height:70px;min-height:56px}:host{overflow:auto}:host ::ng-deep thead{border:10px solid #ff0000;position:relative;z-index:1}.mat-cell-level-1{padding-left:20px!important}.mat-cell-level-2{padding-left:40px!important}.mat-cell-level-3{padding-left:60px!important}.mat-cell-level-4{padding-left:80px!important}.mat-cell-level-5{padding-left:100px!important}.mat-cell-level-6{padding-left:120px!important}.mat-cell-level-7{padding-left:140px!important}.mat-cell-level-8{padding-left:160px!important}.actionbar{flex:1 1 auto;display:flex;flex-direction:row;justify-content:flex-start;align-items:center;gap:.5rem}.scroll-controller{position:fixed;right:.5rem;z-index:100;display:flex;flex-direction:row;justify-content:space-between;align-items:center;gap:.5rem}.scroll-button{min-width:0;padding-left:.25rem;padding-right:.25rem;text-align:center;height:1.75rem}.toolbar{display:flex;flex-direction:row;justify-content:flex-start;align-items:stretch;gap:.5rem}\n"], dependencies: [{ kind: "component", type: MatCard, selector: "mat-card", inputs: ["appearance"], exportAs: ["matCard"] }, { kind: "component", type: MatCardHeader, selector: "mat-card-header" }, { kind: "directive", type: MatCardTitle, selector: "mat-card-title, [mat-card-title], [matCardTitle]" }, { kind: "directive", type: MatCardContent, selector: "mat-card-content" }, { kind: "component", type: QualityMatrixFilterComponent, selector: "metaqs2-qm-filter", inputs: ["options", "inputFormControl", "label", "multiple"], outputs: ["changedFilters"] }, { kind: "ngmodule", type: TranslateModule }, { kind: "pipe", type: i1.TranslatePipe, name: "translate" }, { kind: "pipe", type: KeyValuePipe, name: "keyvalue" }, { kind: "component", type: MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: DatepickerComponent, selector: "metaqs2-datepicker", inputs: ["disabled", "inputGroup"] }, { kind: "component", type: MatSlideToggle, selector: "mat-slide-toggle", inputs: ["name", "id", "labelPosition", "aria-label", "aria-labelledby", "aria-describedby", "required", "color", "disabled", "disableRipple", "tabIndex", "checked", "hideIcon", "disabledInteractive"], outputs: ["change", "toggleChange"], exportAs: ["matSlideToggle"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "component", type: ProgressSpinnerComponent, selector: "metaqs2-progress-spinner", inputs: ["color", "diameter", "strokeWidth", "backdropEnabled", "positionGloballyCenter", "displayProgressSpinner"] }, { kind: "component", type: MatTable, selector: "mat-table, table[mat-table]", exportAs: ["matTable"] }, { kind: "directive", type: MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "directive", type: MatCell, selector: "mat-cell, td[mat-cell]" }, { kind: "directive", type: MatColumnDef, selector: "[matColumnDef]", inputs: ["matColumnDef"] }, { kind: "directive", type: MatHeaderCell, selector: "mat-header-cell, th[mat-header-cell]" }, { kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "component", type: MatHeaderRow, selector: "mat-header-row, tr[mat-header-row]", exportAs: ["matHeaderRow"] }, { kind: "component", type: MatRow, selector: "mat-row, tr[mat-row]", exportAs: ["matRow"] }, { kind: "directive", type: MatHeaderRowDef, selector: "[matHeaderRowDef]", inputs: ["matHeaderRowDef", "matHeaderRowDefSticky"] }, { kind: "directive", type: MatRowDef, selector: "[matRowDef]", inputs: ["matRowDefColumns", "matRowDefWhen"] }, { kind: "directive", type: MatHeaderCellDef, selector: "[matHeaderCellDef]" }, { kind: "directive", type: NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: MatCellDef, selector: "[matCellDef]" }, { kind: "component", type: MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "component", type: MatIconButton, selector: "button[mat-icon-button]", exportAs: ["matButton"] }, { kind: "component", type: DonutChartComponent, selector: "metaqs2-donut-chart", inputs: ["radius", "viewBox", "borderSize", "strokeWidth", "data"] }, { kind: "directive", type: ScrollMarkerDirective, selector: "[metaqs2ScrollMarker]" }] }); }
|
|
360
|
-
}
|
|
361
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: QualityMatrixComponent, decorators: [{
|
|
362
|
-
type: Component,
|
|
363
|
-
args: [{ selector: 'metaqs2-quality-matrix-v2', providers: [TooltipService], standalone: true, imports: [
|
|
364
|
-
MatCard,
|
|
365
|
-
MatCardHeader,
|
|
366
|
-
MatCardTitle,
|
|
367
|
-
MatCardContent,
|
|
368
|
-
QualityMatrixFilterComponent,
|
|
369
|
-
TranslateModule,
|
|
370
|
-
KeyValuePipe,
|
|
371
|
-
MatIcon,
|
|
372
|
-
DatepickerComponent,
|
|
373
|
-
MatSlideToggle,
|
|
374
|
-
ReactiveFormsModule,
|
|
375
|
-
ProgressSpinnerComponent,
|
|
376
|
-
MatTable,
|
|
377
|
-
MatTooltip,
|
|
378
|
-
MatCell,
|
|
379
|
-
MatColumnDef,
|
|
380
|
-
MatHeaderCell,
|
|
381
|
-
NgClass,
|
|
382
|
-
MatHeaderRow,
|
|
383
|
-
MatRow,
|
|
384
|
-
MatHeaderRowDef,
|
|
385
|
-
MatRowDef,
|
|
386
|
-
MatHeaderCellDef,
|
|
387
|
-
NgForOf,
|
|
388
|
-
NgIf,
|
|
389
|
-
MatCellDef,
|
|
390
|
-
MatButton,
|
|
391
|
-
MatIconButton,
|
|
392
|
-
DonutChartComponent,
|
|
393
|
-
ScrollMarkerDirective,
|
|
394
|
-
], template: "<mat-card appearance=\"raised\">\n <mat-card-header *ngIf=\"pageTitle\">\n <mat-card-title>Qualit\u00E4tsmetrik: {{ pageTitle | translate }}{{isLoading() ? \": Lade neue Daten.\" : \"\"}}</mat-card-title>\n </mat-card-header>\n <mat-card-content class=\"toolbar\">\n <metaqs2-qm-filter *ngFor=\"let filter of categoryControls.controls | keyvalue; trackBy: filterIdent\"\n [options]=\"categoryFilterValues.get(filter.key)\"\n [inputFormControl]=\"filter.value\"\n [label] = \"'filter.' + filter.key | translate\" />\n <div class=\"actionbar\">\n <mat-slide-toggle [disabled]=\"isLoading()\" labelPosition=\"before\" [formControl]=\"isOnlySourcesWithMaterialEnabledFormField\" id=\"onlySourcesWithMaterial\">\n show only sources with material\n </mat-slide-toggle>\n <div style=\"flex: 1 1 auto\"></div>\n <metaqs2-datepicker *ngIf=\"isHistoricalDataEnabled()\" [disabled]=\"isLoading()\" [inputGroup]=\"range\" />\n <mat-slide-toggle [disabled]=\"isLoading()\" labelPosition=\"before\" [formControl]=\"isHistoricalDataEnabledFormField\" id=\"showHistoricalData\">\n show historical data\n </mat-slide-toggle>\n <button\n mat-icon-button\n color=\"primary\"\n [disabled]=\"isLoading()\"\n (click)=\"refresh()\"\n matTooltip=\"Aktualisiere den IST-Stand\"\n matTooltipShowDelay=\"500\"\n >\n <mat-icon>refresh</mat-icon>\n </button>\n <div style=\"flex: 0 0 {{scroller.getBoundingClientRect().width}}px\"></div>\n <div class=\"scroll-controller\" #scroller> <!-- put the whole toolbar in a fixed container (MQS- -->\n <button mat-raised-button (click)=\"scrollToLeft()\" [disabled]=\"!isLeftScrollable()\" class=\"scroll-button\">\n <mat-icon style=\"margin: 0\">chevron_left</mat-icon>\n </button>\n <button mat-raised-button (click)=\"scrollToRight()\" [disabled]=\"!isRightScrollable()\" class=\"scroll-button\">\n <mat-icon style=\"margin: 0\">chevron_right</mat-icon>\n </button>\n </div>\n </div>\n </mat-card-content>\n</mat-card>\n<mat-card>\n <metaqs2-progress-spinner [displayProgressSpinner]=\"isLoading()\"></metaqs2-progress-spinner>\n <table [class.while-loading]=\"isLoading()\" mat-table [dataSource]=\"recentQualityMatrix().rows\" class=\"quality-matrix\">\n <!-- Define columns of table -->\n <!-- Row Header Column -->\n <ng-container matColumnDef=\"row-header\" sticky>\n <th rowspan=\"2\" mat-header-cell *matHeaderCellDef #issueType>Issue Type</th>\n <td\n mat-cell\n *matCellDef=\"let row\"\n matTooltip=\"{{row.meta.alt_label}}\"\n class=\"row-header mat-cell-level-{{row.meta.level + 1}}\"\n >\n {{\"quality_matrix.\" + row.meta.label | translate }}\n </td>\n </ng-container>\n <!-- one column for the source -->\n <ng-container *ngFor=\"let col of filteredColumns(); trackBy:columnIdent\" [matColumnDef]=\"col.id + '_source'\">\n <th [attr.colspan]=\"isHistoricalDataEnabled() ? 2 : 1\" metaqs2ScrollMarker mat-header-cell *matHeaderCellDef matTooltip=\"{{col.altLabel}}\">{{col.label}}</th>\n </ng-container>\n <!-- /source -->\n <!-- current Data Columns -->\n <ng-container *ngFor=\"let column of filteredColumns(); trackBy:columnIdent\" [matColumnDef]=\"column.id + '_current'\">\n <th class=\"recent-data-cell\" mat-header-cell *matHeaderCellDef matTooltip=\"{{column.altLabel}}\">{{ range.controls.end.value?.toLocaleString(DateTime.DATE_SHORT) }} <br>({{column.total}})</th>\n <td class=\"recent-data-cell\" mat-cell *matCellDef=\"let row\" #cell (mouseover)=\"showTooltip(row, column, cell)\" (mouseleave)=\"hideTooltip(cell)\">\n <a *ngIf=\"column.total > 0 else emptyDiv\" class=\"chart-container\" (click)=\"openEditLink( column.label, row.meta.label)\">\n <div>\n <metaqs2-donut-chart [data]=\"getDonutSlices(row, column)\" [borderSize]=\"25\"></metaqs2-donut-chart>\n </div>\n </a>\n <ng-template #emptyDiv>\n <div>-</div>\n </ng-template>\n </td>\n </ng-container>\n <!-- /current Data Columns -->\n <!-- past Data Columns -->\n <ng-container *ngIf=\"isHistoricalDataEnabled()\">\n <ng-container *ngFor=\"let column of filteredColumns(); trackBy:columnIdent\" [matColumnDef]=\"column.id + '_past'\">\n <th class=\"historic-data-cell\" mat-header-cell\n *matHeaderCellDef>{{ range.controls.start.value?.toLocaleString(DateTime.DATE_SHORT) }}\n <br>{{ getPastColumn(column.id)?.total }}\n </th>\n <td class=\"historic-data-cell\" mat-cell *matCellDef=\"let row\">\n <ng-container *ngIf=\"getTrend(row, column) as trend\">\n <span [ngClass]=\"trend.trend\">\n <mat-icon aria-hidden=\"false\" [attr.aria-label]=\"trend.trend\" [fontIcon]=\"trend.trend!\" />\n {{ trend.delta }}\n </span>\n </ng-container>\n\n </td>\n </ng-container>\n </ng-container>\n <!-- /past Data Columns -->\n\n <!-- Generate actual table -->\n <tr mat-header-row *matHeaderRowDef=\"['row-header'].concat(sourceColumns()); sticky: true;\"></tr>\n <tr mat-header-row *matHeaderRowDef=\"allDataColumns(); sticky: true;\" [hidden]=\"!isHistoricalDataEnabled()\"></tr>\n <tr mat-row *matRowDef=\"let row; columns: ['row-header'].concat(allDataColumns())\"></tr>\n </table>\n</mat-card>\n", styles: [".while-loading{filter:blur(2px)}mat-card-actions>mat-slide-toggle{margin-left:20px}.row{flex-grow:1;display:flex;align-items:center}table{width:100%;overflow-y:auto}table .scroll{flex-grow:1;overflow-y:auto}table th{vertical-align:top;padding:.375rem}table th:not(:last-child){border-right:1px solid #e4e4e4}table tr:first-child>th:first-child{text-align:left}table td:first-child{text-align:left}.cell{flex:1}.mat-mdc-cell{padding-top:5px}.mat-mdc-cell,.mat-mdc-header-cell{padding-left:5px;text-align:center}.mat-mdc-cell a.chart-container,.mat-mdc-header-cell a.chart-container{display:flex;justify-content:center}.mat-mdc-cell .trending_up,.mat-mdc-header-cell .trending_up{color:#4abeff}.mat-mdc-cell .trending_down,.mat-mdc-header-cell .trending_down{color:#c20808}.recent-data-cell,.mat-column-row-header{border-right:1px solid #e4e4e4}.mat-header-cell-label{color:#000;padding-left:30px;padding-right:30px;justify-content:left;border-bottom:1px solid #e4e4e4;font-style:normal;font-weight:500;font-size:110%;background:#ddf0fb;text-align:left}.mat-header-cell-label>div{display:flex;font-weight:700;font-style:normal;justify-content:left}.mat-header-cell-small{padding-left:5px;padding-right:5px;justify-content:left;font-style:normal;font-weight:400;font-size:105%;background:#ddf0fb;border-top:1px solid #e4e4e4;border-bottom:1px solid #e4e4e4}.mat-mdc-cell a{cursor:pointer;display:flex;align-items:center;justify-content:left}.mat-mdc-cell a>mat-icon{padding-right:5px;font-size:22px;height:20px;opacity:.75}.mat-mdc-cell a>mat-icon:hover,.mat-mdc-cell a>mat-icon:focus{text-decoration:none}.mat-mdc-cell .number{display:flex;height:47px;justify-content:left;align-items:center;padding-left:15px}.mat-mdc-cell .row-header{display:flex;height:47px;max-height:47px;justify-content:left;align-items:center;padding-left:15px}.mat-mdc-cell .row-header-sum{display:flex;height:47px;max-height:47px;justify-content:left;align-items:center;border-right:5px solid #bdbdbd}.mat-mdc-card{width:100%;box-shadow:none}.mat-mdc-card ::ng-deep mat-card-header{align-items:center}.mat-mdc-card ::ng-deep mat-card-header .mat-card-header-text{flex-grow:1;margin:0}.mat-mdc-card ::ng-deep mat-card-header .mat-card-header-text mat-card-title{padding-top:20px;font-size:120%}.mat-mdc-card ::ng-deep mat-card-header .mat-card-header-text mat-card-title .count{font-size:80%;color:#888}.mat-mdc-card .source-header{font-style:normal;font-weight:400;font-size:20px;height:30px;width:60%}.mat-mdc-card .source-header .robot-explanation{font-size:15px}.mat-mdc-card .scroll{flex-grow:1;overflow-y:auto}.mat-header-cell-scope{color:#545454}.mat-mdc-header-row{height:70px;max-height:70px;min-height:56px}:host{overflow:auto}:host ::ng-deep thead{border:10px solid #ff0000;position:relative;z-index:1}.mat-cell-level-1{padding-left:20px!important}.mat-cell-level-2{padding-left:40px!important}.mat-cell-level-3{padding-left:60px!important}.mat-cell-level-4{padding-left:80px!important}.mat-cell-level-5{padding-left:100px!important}.mat-cell-level-6{padding-left:120px!important}.mat-cell-level-7{padding-left:140px!important}.mat-cell-level-8{padding-left:160px!important}.actionbar{flex:1 1 auto;display:flex;flex-direction:row;justify-content:flex-start;align-items:center;gap:.5rem}.scroll-controller{position:fixed;right:.5rem;z-index:100;display:flex;flex-direction:row;justify-content:space-between;align-items:center;gap:.5rem}.scroll-button{min-width:0;padding-left:.25rem;padding-right:.25rem;text-align:center;height:1.75rem}.toolbar{display:flex;flex-direction:row;justify-content:flex-start;align-items:stretch;gap:.5rem}\n"] }]
|
|
395
|
-
}], propDecorators: { issueTypeHeader: [{
|
|
396
|
-
type: ViewChild,
|
|
397
|
-
args: ['issueType', {
|
|
398
|
-
read: ElementRef,
|
|
399
|
-
static: true,
|
|
400
|
-
}]
|
|
401
|
-
}], scrollMarkers: [{
|
|
402
|
-
type: ViewChildren,
|
|
403
|
-
args: [ScrollMarkerDirective]
|
|
404
|
-
}], pageTitle: [{
|
|
405
|
-
type: Input
|
|
406
|
-
}], onScroll: [{
|
|
407
|
-
type: HostListener,
|
|
408
|
-
args: ['window:scroll']
|
|
409
|
-
}, {
|
|
410
|
-
type: HostListener,
|
|
411
|
-
args: ['window:resize']
|
|
412
|
-
}] } });
|
|
413
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"quality_matrix.js","sourceRoot":"","sources":["../../../../../../projects/ng-meta-widgets-lib/src/lib/components/quality-matrix/quality_matrix.ts","../../../../../../projects/ng-meta-widgets-lib/src/lib/components/quality-matrix/quality-matrix.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewChecked,\n  Component,\n  computed,\n  DestroyRef,\n  ElementRef,\n  HostListener,\n  inject,\n  Input,\n  OnInit,\n  signal,\n  ViewChild,\n  ViewChildren,\n} from '@angular/core';\nimport { BehaviorSubject, combineLatest, Observable, of, Subject, Unsubscribable } from 'rxjs';\nimport { filter, finalize, map, switchMap, take, tap, throttleTime } from 'rxjs/operators';\nimport { MetaApiService } from '../../meta-api.service';\nimport { Filter, FilterValue, QualityMatrix, QualityMatrixHeader, QualityMatrixRow } from '../../java-api';\nimport { FormControl, FormGroup, FormRecord, ReactiveFormsModule } from '@angular/forms';\nimport { takeUntilDestroyed, toSignal } from '@angular/core/rxjs-interop';\nimport { EditorialLinkService } from '../editorial-link-service/editorial-link.service';\nimport { DateTime } from 'luxon';\nimport { DonutSlice } from '../donut-chart/donut-chart.model';\nimport { KeyValue, KeyValuePipe, NgClass, NgForOf, NgIf } from '@angular/common';\nimport { Overlay } from '@angular/cdk/overlay';\nimport {\n  DonutChartTooltipComponent,\n  transformDonutChartData,\n} from '../donut-chart-tooltip/donut-chart-tooltip.component';\nimport { centerPositionStrategy, TooltipRef, TooltipService } from '../../core/tooltip.service';\nimport { MatCard, MatCardContent, MatCardHeader, MatCardTitle } from '@angular/material/card';\nimport { QualityMatrixFilterComponent } from '../filter/quality-matrix-filter.component';\nimport { TranslateModule } from '@ngx-translate/core';\nimport { MatIcon } from '@angular/material/icon';\nimport { DatepickerComponent } from '../filter/datepicker/datepicker.component';\nimport { MatSlideToggle } from '@angular/material/slide-toggle';\nimport { ProgressSpinnerComponent } from '../loading_indicator/progress-spinner/progress-spinner.component';\nimport {\n  MatCell,\n  MatCellDef,\n  MatColumnDef,\n  MatHeaderCell,\n  MatHeaderCellDef,\n  MatHeaderRow,\n  MatHeaderRowDef,\n  MatRow,\n  MatRowDef,\n  MatTable,\n} from '@angular/material/table';\nimport { MatTooltip } from '@angular/material/tooltip';\nimport { MatButton, MatIconButton } from '@angular/material/button';\nimport { DonutChartComponent } from '../donut-chart/donut-chart.component';\nimport { ScrollMarkerDirective } from './scroll-marker.directive';\n\nconst openCloseTimeout = 150;\n\n@Component({\n  selector: 'metaqs2-quality-matrix-v2',\n  providers: [TooltipService],\n  standalone: true,\n  templateUrl: './quality-matrix.html',\n  imports: [\n    MatCard,\n    MatCardHeader,\n    MatCardTitle,\n    MatCardContent,\n    QualityMatrixFilterComponent,\n    TranslateModule,\n    KeyValuePipe,\n    MatIcon,\n    DatepickerComponent,\n    MatSlideToggle,\n    ReactiveFormsModule,\n    ProgressSpinnerComponent,\n    MatTable,\n    MatTooltip,\n    MatCell,\n    MatColumnDef,\n    MatHeaderCell,\n    NgClass,\n    MatHeaderRow,\n    MatRow,\n    MatHeaderRowDef,\n    MatRowDef,\n    MatHeaderCellDef,\n    NgForOf,\n    NgIf,\n    MatCellDef,\n    MatButton,\n    MatIconButton,\n    DonutChartComponent,\n    ScrollMarkerDirective,\n  ],\n  styleUrls: ['./quality_matrix.scss'],\n})\nexport class QualityMatrixComponent implements OnInit, AfterViewChecked {\n  protected readonly ref = inject(ElementRef);\n  protected readonly destroyRef = inject(DestroyRef);\n  protected readonly metaApi = inject(MetaApiService);\n  protected readonly linkService = inject(EditorialLinkService);\n  protected readonly overlay = inject(Overlay);\n  protected readonly tooltipService = inject(TooltipService);\n  private readonly scrollLeft = signal(0);\n  private readonly scrollWidth = signal(0);\n  public readonly isLeftScrollable = computed(() => this.scrollLeft() > 0);\n  public readonly isRightScrollable = computed(() => {\n    const viewport = globalThis.visualViewport;\n    if (!viewport) {\n      return;\n    }\n    const viewportWidth = viewport.width;\n\n    return this.scrollWidth() > viewportWidth && this.scrollLeft() + viewportWidth < this.scrollWidth();\n  });\n  @ViewChild('issueType', {\n    read: ElementRef,\n    static: true,\n  })\n  private readonly issueTypeHeader!: ElementRef<HTMLTableCellElement>;\n  readonly filteredColumns = computed<Array<QualityMatrixHeader>>(() => {\n    const allColumns = this.allColumns();\n    const isOnlySourcesWithMaterialEnabled = this.isOnlySourcesWithMaterialEnabled();\n    if (isOnlySourcesWithMaterialEnabled) {\n      return allColumns.filter((c) => c.total > 0);\n    } else {\n      return allColumns;\n    }\n  });\n  readonly datatableColumns = computed(() => {\n    return this.filteredColumns()\n      .filter((c) => c.level !== 0)\n      .map((c) => c.id);\n  });\n  readonly isHistoricalDataEnabledFormField = new FormControl({ value: false, disabled: true });\n  readonly isHistoricalDataEnabled = toSignal(this.isHistoricalDataEnabledFormField.valueChanges);\n  readonly isOnlySourcesWithMaterialEnabledFormField = new FormControl({ value: true, disabled: true });\n  readonly isOnlySourcesWithMaterialEnabled = toSignal(this.isOnlySourcesWithMaterialEnabledFormField.valueChanges);\n  readonly categoryControls = new FormRecord<FormControl<Array<string>>>({});\n  readonly loadingCount = signal(0);\n  readonly isLoading = computed(() => this.loadingCount() > 0);\n  @ViewChildren(ScrollMarkerDirective)\n  private readonly scrollMarkers!: ReadonlyArray<ScrollMarkerDirective>;\n  readonly categoryFilterValues: Map<string, FilterValue[]> = new Map();\n  protected readonly refresh$ = new Subject<void>();\n  readonly recentQualityMatrix$ = new BehaviorSubject<QualityMatrix>({ columns: [], rows: [] } as QualityMatrix);\n  readonly recentQualityMatrix = toSignal(this.recentQualityMatrix$, {\n    initialValue: { columns: [], rows: [] } as QualityMatrix,\n  });\n  readonly pastQualityMatrix$ = new BehaviorSubject<QualityMatrix | null>({ columns: [], rows: [] } as QualityMatrix);\n  readonly range: FormGroup<{\n    start: FormControl<DateTime<boolean>>;\n    end: FormControl<DateTime<boolean>>;\n  }> = new FormGroup({\n    start: new FormControl(),\n    end: new FormControl(),\n  });\n  private readonly tooltips = new Map<HTMLElement, TooltipRef<DonutChartTooltipComponent>>();\n  protected readonly DateTime = DateTime;\n  @Input()\n  pageTitle: string = 'Quality Matrix';\n\n  readonly allColumns = computed(() =>\n    this.recentQualityMatrix()?.columns.toSorted((a, b) => a.label.localeCompare(b.label))\n  );\n\n  ngAfterViewChecked(): void {\n    this.onScroll();\n  }\n\n  filterIdent(_index: number, item: KeyValue<string, FormControl<string[]>>) {\n    return item.key;\n  }\n\n  sourceColumns = computed(() => this.datatableColumns().map((c) => c + '_source'));\n\n  currentColumns = computed(() => this.datatableColumns().map((c) => c + '_current'));\n\n  pastColumns = computed(() => this.datatableColumns().map((c) => c + '_past'));\n\n  allDataColumns = computed(() => {\n    const pastColumns = this.pastColumns();\n    const currentColumns = this.currentColumns();\n    if (this.isHistoricalDataEnabled()) {\n      return currentColumns.flatMap((e, i) => [pastColumns[i], e]);\n    }\n    return currentColumns;\n  });\n\n  private loadData(categories: { [key: string]: Array<string> }, date: DateTime<true>): Observable<QualityMatrix> {\n    return of(undefined).pipe(\n      tap(() => this.loadingCount.update((it) => it + 1)),\n      switchMap(() =>\n        this.metaApi\n          .getQualityMatrixWithFiltersV2([\n            ...this.transformCategoryFilters(categories),\n            {\n              field: 'asOf',\n              values: [\n                {\n                  id: date.toISO({\n                    includeOffset: false,\n                  }) as string,\n                  label: '',\n                },\n              ],\n            },\n          ])\n          .pipe(take(1))\n      ),\n      finalize(() => this.loadingCount.update((it) => it - 1))\n    );\n  }\n\n  private transformCategoryFilters(categories: { [key: string]: Array<string> }): Filter[] {\n    return Object.entries(categories)\n      .filter((pair): pair is [string, Array<string>] => !!pair[1]?.length)\n      .map(([field, values]) => ({\n        field,\n        values: values.map((value) => ({ id: value, label: '' })),\n      }));\n  }\n\n  private loadPastData(categories: any, date: DateTime<true>) {\n    return of(undefined).pipe(switchMap(() => this.loadData(categories, date)));\n  }\n\n  private loadCurrentData(categories: any, date: DateTime<true>) {\n    return of(undefined).pipe(switchMap(() => this.loadData(categories, date)));\n  }\n\n  openEditLink(source: string, issue: string) {\n    return this.linkService.openByReplicationsourceAndIssueTypeWithFilters(source, issue, this.categoryControls.value);\n  }\n\n  // see https://github.com/angular/components/issues/8361#issuecomment-345804954\n  columnIdent(_index: number, col: QualityMatrixHeader) {\n    return col.id;\n  }\n\n  refresh() {\n    this.refresh$.next();\n  }\n\n  getPastColumn(columnId: string) {\n    return this.pastQualityMatrix$.value?.columns.find((c) => c.id === columnId);\n  }\n\n  public ngOnInit() {\n    this.loadingCount.update((it) => it + 1);\n    this.metaApi\n      .getCategoryFilters()\n      .pipe(\n        take(1),\n        tap((filters) => {\n          filters.forEach((filter) => {\n            // do not emit the event when setting the control to prevent unnecessary valueChanges events\n            this.categoryControls.addControl(filter.field, new FormControl([], { nonNullable: true }), {\n              emitEvent: false,\n            });\n            this.categoryFilterValues.set(\n              filter.field,\n              filter.values.toSorted((a, b) => {\n                return a.label.localeCompare(b.label);\n              })\n            );\n          });\n          // emit the current value of the form group once to trigger the valueChanges observable\n          this.categoryControls.setValue(this.categoryControls.value as { [key: string]: string[] }, {\n            emitEvent: true,\n          });\n        }),\n        takeUntilDestroyed(this.destroyRef),\n        finalize(() => this.loadingCount.update((it) => it - 1))\n      )\n      .subscribe();\n\n    this.loadingCount.update((it) => it + 1);\n    this.metaApi\n      .getTimerangeFilter()\n      .pipe(\n        filter((rangeFilter) => !!rangeFilter),\n        tap((rangeFilter) => {\n          const startDate = DateTime.fromISO(rangeFilter.values.find((v) => v.id === 'rangeStart')?.label!, {\n            zone: 'utc',\n          });\n          const endDate = DateTime.fromISO(rangeFilter.values.find((v) => v.id === 'rangeEnd')?.label!, {\n            zone: 'utc',\n          });\n          this.range.setControl('start', new FormControl<DateTime>(startDate, { nonNullable: true }), {\n            emitEvent: false,\n          });\n          this.range.setControl('end', new FormControl<DateTime>(endDate, { nonNullable: true }), {\n            emitEvent: true,\n          });\n        }),\n        takeUntilDestroyed(this.destroyRef),\n        finalize(() => this.loadingCount.update((it) => it - 1))\n      )\n      .subscribe();\n\n    combineLatest([this.categoryControls.valueChanges, this.range.valueChanges, this.refresh$])\n      .pipe(\n        filter(\n          (data): data is [{ [key: string]: Array<string> }, { end: DateTime<true> }, void] =>\n            !!data[0] && !!data[1].end?.isValid\n        ),\n        throttleTime(500),\n        map(([categories, range]) => [categories, range.end] as const),\n        switchMap(([categories, end]) => this.loadCurrentData(categories, end)),\n        tap((currentQm) => {\n          this.recentQualityMatrix$.next(currentQm);\n        }),\n        takeUntilDestroyed(this.destroyRef)\n      )\n      .subscribe();\n\n    let pastDataSubscription: Unsubscribable | undefined;\n\n    const pastData$ = combineLatest([this.categoryControls.valueChanges, this.range.valueChanges, this.refresh$]).pipe(\n      filter(\n        (data): data is [{ [key: string]: Array<string> }, { start: DateTime<true> }, void] =>\n          !!data[0] && !!data[1].start?.isValid\n      ),\n      throttleTime(500),\n      map(([categories, range]) => [categories, range.start] as const),\n      switchMap(([categories, start]) => this.loadPastData(categories, start)),\n      tap((pastQm) => {\n        this.pastQualityMatrix$.next(pastQm);\n      }),\n      takeUntilDestroyed(this.destroyRef)\n    );\n\n    this.isHistoricalDataEnabledFormField.valueChanges\n      .pipe(\n        tap((isEnabled) => {\n          if (isEnabled) {\n            pastDataSubscription = pastData$.subscribe();\n            this.categoryControls.setValue(this.categoryControls.value as { [key: string]: string[] }, {\n              emitEvent: true,\n            });\n            this.range.setValue(this.range.value as { start: DateTime; end: DateTime }, {\n              emitEvent: true,\n            });\n            this.refresh$.next();\n          } else {\n            pastDataSubscription?.unsubscribe();\n          }\n        }),\n        takeUntilDestroyed(this.destroyRef)\n      )\n      .subscribe();\n\n    this.refresh$.next();\n    this.isOnlySourcesWithMaterialEnabledFormField.setValue(this.isOnlySourcesWithMaterialEnabledFormField.value, {\n      emitEvent: true,\n    });\n  }\n\n  getTrend(row: QualityMatrixRow, column: QualityMatrixHeader): { delta?: number; trend?: string } {\n    if (this.pastQualityMatrix$.value === null || this.pastQualityMatrix$.value.columns.length === 0) {\n      return {};\n    }\n    const past_row = this.pastQualityMatrix$?.value.rows.find((pr) => pr.meta.id == row.meta.id);\n    const past_column = this.pastQualityMatrix$.value.columns.find((c) => c.id == column.id);\n    if (!past_row || !past_column) {\n      return {};\n    }\n\n    const delta = Math.floor(\n      (100 * row.counts[column.id].sufficient) / column.total -\n        (100 * past_row.counts[column.id].sufficient) / past_column.total\n    );\n\n    const trend = delta === 0 ? 'trending_flat' : delta < 0 ? 'trending_down' : 'trending_up';\n    return { delta, trend };\n  }\n\n  getDonutSlices(row: QualityMatrixRow, column: QualityMatrixHeader): DonutSlice[] {\n    const sufficientTotal = row.counts[column.id].sufficient;\n    const insufficientTotal = row.counts[column.id].insufficient;\n    const missingTotal = column.total - sufficientTotal - insufficientTotal;\n    let sufficientFillRate = 0;\n    let insufficientFillRate = 0;\n    //Der Wert berechnet sich aus dem total_count - without_description_count.\n\n    if (column.total !== 0) {\n      sufficientFillRate = Math.round((100 * sufficientTotal) / column.total);\n      insufficientFillRate = Math.round((100 * insufficientTotal) / column.total);\n    }\n    const rest = 100 - (sufficientFillRate + insufficientFillRate); // a.k.a. \"missing\"\n    return [\n      {\n        id: 3,\n        percent: rest,\n        color: '#D8EDFC',\n        label: `${rest} % (${missingTotal})`,\n        verboseLabel: `${missingTotal} Metadatens\\u00e4tze fehlen`,\n      },\n      {\n        id: 1,\n        percent: insufficientFillRate,\n        color: '#C20808',\n        label: `${insufficientFillRate} % (${insufficientTotal})`,\n        verboseLabel: `${insufficientTotal} Metadatens\\u00e4tze sind unzureichend ausgef\\u00fcllt`,\n      },\n      {\n        id: 2,\n        percent: sufficientFillRate,\n        color: '#4ABEFF',\n        label: `${sufficientFillRate} % (${sufficientTotal})`,\n        verboseLabel: `${sufficientTotal} Metadatens\\u00e4tze sind gut ausgef\\u00fcllt`,\n      },\n    ];\n  }\n\n  showTooltip(row: QualityMatrixRow, column: QualityMatrixHeader, cell: HTMLElement) {\n    if (row.counts[column.id].sufficient + row.counts[column.id].insufficient === 0) {\n      return;\n    }\n    if (this.tooltips.has(cell)) {\n      this.tooltips.get(cell)!.cancelCloseAction();\n      return;\n    }\n\n    const tooltipRef = this.tooltipService.create(DonutChartTooltipComponent, {\n      closeTimeout: openCloseTimeout,\n      openTimeout: openCloseTimeout,\n      data: {\n        label: column.label,\n        data: transformDonutChartData(this.getDonutSlices(row, column)),\n      },\n      overlayConfig: {\n        hasBackdrop: false,\n        positionStrategy: this.overlay.position().flexibleConnectedTo(cell).withPositions(centerPositionStrategy),\n        disposeOnNavigation: true,\n        panelClass: 'donut-chart-tooltip',\n      },\n      onClose: () => {\n        this.tooltips.delete(cell);\n      },\n    });\n    tooltipRef.open();\n    this.tooltips.set(cell, tooltipRef);\n  }\n\n  hideTooltip(cell: HTMLElement) {\n    this.tooltips.get(cell)?.markForClose();\n  }\n\n  findHorizontalScrollContainer(element: Element): Element | null {\n    if (element.scrollWidth > element.clientWidth && ['auto', 'scroll'].includes(getComputedStyle(element).overflowX)) {\n      // find the nearest horizontally scrollable element\n      return element;\n    } else {\n      if (!element.parentElement) {\n        return document.scrollingElement;\n      }\n      return this.findHorizontalScrollContainer(element.parentElement);\n    }\n  }\n\n  scrollToRight() {\n    const scrollingElement = this.findHorizontalScrollContainer(this.ref.nativeElement as HTMLElement);\n    if (!scrollingElement) {\n      return;\n    }\n    const viewport = globalThis.visualViewport;\n    if (!viewport) {\n      return;\n    }\n    const viewportWidth = viewport.width;\n\n    const lastVisible = this.scrollMarkers.findLast(\n      (marker) => (marker.element.nativeElement as HTMLElement).getBoundingClientRect().left < viewportWidth\n    );\n\n    if (!lastVisible) {\n      return;\n    }\n\n    const leftColumnWidth = this.issueTypeHeader?.nativeElement.getBoundingClientRect().width ?? 0;\n\n    scrollingElement.scrollBy({\n      left: (lastVisible.element.nativeElement as HTMLElement).getBoundingClientRect().left - leftColumnWidth,\n      behavior: 'smooth',\n    });\n  }\n\n  scrollToLeft() {\n    const scrollingElement = this.findHorizontalScrollContainer(this.ref.nativeElement as HTMLElement);\n    if (!scrollingElement) {\n      return;\n    }\n    const viewport = globalThis.visualViewport;\n    if (!viewport) {\n      return;\n    }\n    const viewportWidth = viewport.width;\n\n    const leftColumnWidth = this.issueTypeHeader?.nativeElement.getBoundingClientRect().width ?? 0;\n\n    const firstVisible = this.scrollMarkers.find(\n      (marker) => (marker.element.nativeElement as HTMLElement).getBoundingClientRect().right > leftColumnWidth\n    );\n\n    if (!firstVisible) {\n      return;\n    }\n\n    scrollingElement.scrollBy({\n      left: (firstVisible.element.nativeElement as HTMLElement).getBoundingClientRect().right - viewportWidth,\n      behavior: 'smooth',\n    });\n  }\n\n  @HostListener('window:scroll')\n  @HostListener('window:resize')\n  onScroll() {\n    const scrollingElement = this.findHorizontalScrollContainer(this.ref.nativeElement as HTMLElement);\n    if (!scrollingElement) {\n      return;\n    }\n    this.scrollLeft.set(scrollingElement.scrollLeft);\n    this.scrollWidth.set(scrollingElement.scrollWidth);\n  }\n}\n","<mat-card appearance=\"raised\">\n  <mat-card-header *ngIf=\"pageTitle\">\n    <mat-card-title>Qualitätsmetrik: {{ pageTitle | translate }}{{isLoading() ? \": Lade neue Daten.\" : \"\"}}</mat-card-title>\n  </mat-card-header>\n  <mat-card-content class=\"toolbar\">\n    <metaqs2-qm-filter *ngFor=\"let filter of categoryControls.controls | keyvalue; trackBy: filterIdent\"\n                       [options]=\"categoryFilterValues.get(filter.key)\"\n                       [inputFormControl]=\"filter.value\"\n                       [label] = \"'filter.' + filter.key | translate\" />\n    <div class=\"actionbar\">\n      <mat-slide-toggle [disabled]=\"isLoading()\" labelPosition=\"before\" [formControl]=\"isOnlySourcesWithMaterialEnabledFormField\" id=\"onlySourcesWithMaterial\">\n        show only sources with material\n      </mat-slide-toggle>\n      <div style=\"flex: 1 1 auto\"></div>\n      <metaqs2-datepicker *ngIf=\"isHistoricalDataEnabled()\" [disabled]=\"isLoading()\" [inputGroup]=\"range\" />\n      <mat-slide-toggle [disabled]=\"isLoading()\" labelPosition=\"before\" [formControl]=\"isHistoricalDataEnabledFormField\" id=\"showHistoricalData\">\n        show historical data\n      </mat-slide-toggle>\n      <button\n        mat-icon-button\n        color=\"primary\"\n        [disabled]=\"isLoading()\"\n        (click)=\"refresh()\"\n        matTooltip=\"Aktualisiere den IST-Stand\"\n        matTooltipShowDelay=\"500\"\n      >\n        <mat-icon>refresh</mat-icon>\n      </button>\n      <div style=\"flex: 0 0 {{scroller.getBoundingClientRect().width}}px\"></div>\n      <div class=\"scroll-controller\" #scroller> <!-- put the whole toolbar in a fixed container (MQS- -->\n        <button mat-raised-button (click)=\"scrollToLeft()\" [disabled]=\"!isLeftScrollable()\" class=\"scroll-button\">\n          <mat-icon style=\"margin: 0\">chevron_left</mat-icon>\n        </button>\n        <button mat-raised-button (click)=\"scrollToRight()\" [disabled]=\"!isRightScrollable()\" class=\"scroll-button\">\n          <mat-icon style=\"margin: 0\">chevron_right</mat-icon>\n        </button>\n      </div>\n    </div>\n  </mat-card-content>\n</mat-card>\n<mat-card>\n  <metaqs2-progress-spinner [displayProgressSpinner]=\"isLoading()\"></metaqs2-progress-spinner>\n  <table [class.while-loading]=\"isLoading()\" mat-table [dataSource]=\"recentQualityMatrix().rows\" class=\"quality-matrix\">\n    <!-- Define columns of table -->\n    <!-- Row Header Column -->\n    <ng-container matColumnDef=\"row-header\" sticky>\n      <th rowspan=\"2\" mat-header-cell *matHeaderCellDef #issueType>Issue Type</th>\n      <td\n        mat-cell\n        *matCellDef=\"let row\"\n        matTooltip=\"{{row.meta.alt_label}}\"\n        class=\"row-header mat-cell-level-{{row.meta.level + 1}}\"\n      >\n        {{\"quality_matrix.\" + row.meta.label | translate }}\n      </td>\n    </ng-container>\n    <!-- one column for the source -->\n    <ng-container *ngFor=\"let col of filteredColumns(); trackBy:columnIdent\" [matColumnDef]=\"col.id + '_source'\">\n      <th [attr.colspan]=\"isHistoricalDataEnabled() ? 2 : 1\" metaqs2ScrollMarker mat-header-cell *matHeaderCellDef matTooltip=\"{{col.altLabel}}\">{{col.label}}</th>\n    </ng-container>\n    <!-- /source -->\n    <!-- current Data Columns -->\n    <ng-container *ngFor=\"let column of filteredColumns(); trackBy:columnIdent\" [matColumnDef]=\"column.id + '_current'\">\n      <th class=\"recent-data-cell\" mat-header-cell *matHeaderCellDef matTooltip=\"{{column.altLabel}}\">{{ range.controls.end.value?.toLocaleString(DateTime.DATE_SHORT) }} <br>({{column.total}})</th>\n      <td class=\"recent-data-cell\" mat-cell *matCellDef=\"let row\" #cell (mouseover)=\"showTooltip(row, column, cell)\" (mouseleave)=\"hideTooltip(cell)\">\n          <a *ngIf=\"column.total > 0 else emptyDiv\" class=\"chart-container\" (click)=\"openEditLink( column.label, row.meta.label)\">\n            <div>\n              <metaqs2-donut-chart [data]=\"getDonutSlices(row, column)\" [borderSize]=\"25\"></metaqs2-donut-chart>\n            </div>\n          </a>\n          <ng-template #emptyDiv>\n            <div>-</div>\n          </ng-template>\n      </td>\n    </ng-container>\n    <!-- /current Data Columns -->\n    <!-- past Data Columns -->\n    <ng-container *ngIf=\"isHistoricalDataEnabled()\">\n      <ng-container *ngFor=\"let column of filteredColumns(); trackBy:columnIdent\" [matColumnDef]=\"column.id + '_past'\">\n        <th class=\"historic-data-cell\" mat-header-cell\n            *matHeaderCellDef>{{ range.controls.start.value?.toLocaleString(DateTime.DATE_SHORT) }}\n          <br>{{ getPastColumn(column.id)?.total }}\n        </th>\n        <td class=\"historic-data-cell\" mat-cell *matCellDef=\"let row\">\n          <ng-container *ngIf=\"getTrend(row, column) as trend\">\n            <span [ngClass]=\"trend.trend\">\n              <mat-icon aria-hidden=\"false\" [attr.aria-label]=\"trend.trend\" [fontIcon]=\"trend.trend!\" />\n              {{ trend.delta }}\n            </span>\n          </ng-container>\n\n        </td>\n      </ng-container>\n    </ng-container>\n    <!-- /past Data Columns -->\n\n      <!-- Generate actual table -->\n    <tr mat-header-row *matHeaderRowDef=\"['row-header'].concat(sourceColumns()); sticky: true;\"></tr>\n    <tr mat-header-row *matHeaderRowDef=\"allDataColumns(); sticky: true;\" [hidden]=\"!isHistoricalDataEnabled()\"></tr>\n    <tr mat-row *matRowDef=\"let row; columns: ['row-header'].concat(allDataColumns())\"></tr>\n  </table>\n</mat-card>\n"]}
|
|
@@ -1,17 +0,0 @@
|
|
|
1
|
-
import { Directive, ElementRef, inject } from '@angular/core';
|
|
2
|
-
import * as i0 from "@angular/core";
|
|
3
|
-
export class ScrollMarkerDirective {
|
|
4
|
-
constructor() {
|
|
5
|
-
this.element = inject(ElementRef);
|
|
6
|
-
}
|
|
7
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScrollMarkerDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
8
|
-
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.13", type: ScrollMarkerDirective, isStandalone: true, selector: "[metaqs2ScrollMarker]", ngImport: i0 }); }
|
|
9
|
-
}
|
|
10
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScrollMarkerDirective, decorators: [{
|
|
11
|
-
type: Directive,
|
|
12
|
-
args: [{
|
|
13
|
-
selector: '[metaqs2ScrollMarker]',
|
|
14
|
-
standalone: true,
|
|
15
|
-
}]
|
|
16
|
-
}] });
|
|
17
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2Nyb2xsLW1hcmtlci5kaXJlY3RpdmUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uZy1tZXRhLXdpZGdldHMtbGliL3NyYy9saWIvY29tcG9uZW50cy9xdWFsaXR5LW1hdHJpeC9zY3JvbGwtbWFya2VyLmRpcmVjdGl2ZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLFVBQVUsRUFBRSxNQUFNLEVBQUUsTUFBTSxlQUFlLENBQUM7O0FBTTlELE1BQU0sT0FBTyxxQkFBcUI7SUFKbEM7UUFLUyxZQUFPLEdBQUcsTUFBTSxDQUFDLFVBQVUsQ0FBQyxDQUFDO0tBQ3JDOytHQUZZLHFCQUFxQjttR0FBckIscUJBQXFCOzs0RkFBckIscUJBQXFCO2tCQUpqQyxTQUFTO21CQUFDO29CQUNULFFBQVEsRUFBRSx1QkFBdUI7b0JBQ2pDLFVBQVUsRUFBRSxJQUFJO2lCQUNqQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IERpcmVjdGl2ZSwgRWxlbWVudFJlZiwgaW5qZWN0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbkBEaXJlY3RpdmUoe1xuICBzZWxlY3RvcjogJ1ttZXRhcXMyU2Nyb2xsTWFya2VyXScsXG4gIHN0YW5kYWxvbmU6IHRydWUsXG59KVxuZXhwb3J0IGNsYXNzIFNjcm9sbE1hcmtlckRpcmVjdGl2ZSB7XG4gIHB1YmxpYyBlbGVtZW50ID0gaW5qZWN0KEVsZW1lbnRSZWYpO1xufVxuIl19
|