ngx-edu-sharing-metaqs2 0.9.27 → 0.9.29
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/esm2022/lib/components/collection-issues/collection-issues.component.mjs +3 -3
- package/esm2022/lib/components/material-issues/material-issues.component.mjs +3 -3
- package/esm2022/lib/components/node-list/node-list.component.mjs +15 -8
- package/esm2022/lib/tree-search-counts/inline-worker.mjs +7 -8
- package/esm2022/lib/tree-search-counts/tree-search-counts.component.mjs +14 -6
- package/fesm2022/ngx-edu-sharing-metaqs2.mjs +35 -22
- package/fesm2022/ngx-edu-sharing-metaqs2.mjs.map +1 -1
- package/lib/components/node-list/node-list.component.d.ts +7 -3
- package/lib/tree-search-counts/tree-search-counts.component.d.ts +2 -1
- package/package.json +1 -1
|
@@ -14,10 +14,10 @@ export class CollectionIssuesComponent {
|
|
|
14
14
|
this.issues = await this.filterAPIService.getCollectionIssueFieldNames().toPromise();
|
|
15
15
|
}
|
|
16
16
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CollectionIssuesComponent, deps: [{ token: i1.FilterAPIService }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
17
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: CollectionIssuesComponent, selector: "metaqs2-collection-issues", ngImport: i0, template: "<metaqs2-node-list\n *ngFor=\"let mode of issues\"\n [
|
|
17
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: CollectionIssuesComponent, selector: "metaqs2-collection-issues", ngImport: i0, template: "<metaqs2-node-list\n *ngFor=\"let mode of issues\"\n [collectionId]=\"collectionId\"\n [mode]=\"mode\"\n type=\"collection\"\n></metaqs2-node-list>\n", styles: [":host{padding:0 30px;display:flex;gap:30px;flex-wrap:wrap}:host>metaqs2-node-list{width:400px}\n"], dependencies: [{ kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "component", type: i3.NodeListComponent, selector: "metaqs2-node-list", inputs: ["mode", "type", "collectionId"], outputs: ["countChanged"] }] }); }
|
|
18
18
|
}
|
|
19
19
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CollectionIssuesComponent, decorators: [{
|
|
20
20
|
type: Component,
|
|
21
|
-
args: [{ selector: 'metaqs2-collection-issues', template: "<metaqs2-node-list\n *ngFor=\"let mode of issues\"\n [
|
|
21
|
+
args: [{ selector: 'metaqs2-collection-issues', template: "<metaqs2-node-list\n *ngFor=\"let mode of issues\"\n [collectionId]=\"collectionId\"\n [mode]=\"mode\"\n type=\"collection\"\n></metaqs2-node-list>\n", styles: [":host{padding:0 30px;display:flex;gap:30px;flex-wrap:wrap}:host>metaqs2-node-list{width:400px}\n"] }]
|
|
22
22
|
}], ctorParameters: () => [{ type: i1.FilterAPIService }] });
|
|
23
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
23
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29sbGVjdGlvbi1pc3N1ZXMuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmctbWV0YS13aWRnZXRzLWxpYi9zcmMvbGliL2NvbXBvbmVudHMvY29sbGVjdGlvbi1pc3N1ZXMvY29sbGVjdGlvbi1pc3N1ZXMuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmctbWV0YS13aWRnZXRzLWxpYi9zcmMvbGliL2NvbXBvbmVudHMvY29sbGVjdGlvbi1pc3N1ZXMvY29sbGVjdGlvbi1pc3N1ZXMuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBVSxNQUFNLGVBQWUsQ0FBQzs7Ozs7QUFRbEQsTUFBTSxPQUFPLHlCQUF5QjtJQUtwQyxZQUFvQixnQkFBa0M7UUFBbEMscUJBQWdCLEdBQWhCLGdCQUFnQixDQUFrQjtRQUp0RCxXQUFNLEdBQWEsRUFBRSxDQUFDO1FBQ3RCLHVFQUF1RTtRQUN2RSxpQkFBWSxHQUFXLHNDQUFzQyxDQUFDO0lBRUwsQ0FBQztJQUUxRCxLQUFLLENBQUMsUUFBUTtRQUNaLElBQUksQ0FBQyxNQUFNLEdBQUcsTUFBTSxJQUFJLENBQUMsZ0JBQWdCLENBQUMsNEJBQTRCLEVBQUUsQ0FBQyxTQUFTLEVBQUUsQ0FBQztJQUN2RixDQUFDOytHQVRVLHlCQUF5QjttR0FBekIseUJBQXlCLGlFQ1J0QywySkFNQTs7NEZERWEseUJBQXlCO2tCQUxyQyxTQUFTOytCQUNFLDJCQUEyQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbXBvbmVudCwgT25Jbml0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBGaWx0ZXJBUElTZXJ2aWNlIH0gZnJvbSAnLi4vLi4vamF2YS1hcGknO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdtZXRhcXMyLWNvbGxlY3Rpb24taXNzdWVzJyxcbiAgdGVtcGxhdGVVcmw6ICcuL2NvbGxlY3Rpb24taXNzdWVzLmNvbXBvbmVudC5odG1sJyxcbiAgc3R5bGVVcmxzOiBbJy4vY29sbGVjdGlvbi1pc3N1ZXMuY29tcG9uZW50LnNjc3MnXSxcbn0pXG5leHBvcnQgY2xhc3MgQ29sbGVjdGlvbklzc3Vlc0NvbXBvbmVudCBpbXBsZW1lbnRzIE9uSW5pdCB7XG4gIGlzc3Vlczogc3RyaW5nW10gPSBbXTtcbiAgLy8gQFRPRE86IElucHV0IGlzIGJyb2tlbiwgbWF5YmUgY2F1c2Ugb2Ygd2l0aENvbXBvbmVudElucHV0QmluZGluZygpPyFcbiAgY29sbGVjdGlvbklkOiBzdHJpbmcgPSAnOTRmMjJjOWItMGQzYS00YzFjLTg5ODctNGM4ZTgzZjNhOTJlJztcblxuICBjb25zdHJ1Y3Rvcihwcml2YXRlIGZpbHRlckFQSVNlcnZpY2U6IEZpbHRlckFQSVNlcnZpY2UpIHt9XG5cbiAgYXN5bmMgbmdPbkluaXQoKSB7XG4gICAgdGhpcy5pc3N1ZXMgPSBhd2FpdCB0aGlzLmZpbHRlckFQSVNlcnZpY2UuZ2V0Q29sbGVjdGlvbklzc3VlRmllbGROYW1lcygpLnRvUHJvbWlzZSgpO1xuICB9XG59XG4iLCI8bWV0YXFzMi1ub2RlLWxpc3RcbiAgKm5nRm9yPVwibGV0IG1vZGUgb2YgaXNzdWVzXCJcbiAgW2NvbGxlY3Rpb25JZF09XCJjb2xsZWN0aW9uSWRcIlxuICBbbW9kZV09XCJtb2RlXCJcbiAgdHlwZT1cImNvbGxlY3Rpb25cIlxuPjwvbWV0YXFzMi1ub2RlLWxpc3Q+XG4iXX0=
|
|
@@ -14,10 +14,10 @@ export class MaterialIssuesComponent {
|
|
|
14
14
|
this.issues = await this.filterAPIService.getMaterialIssueFieldNames().toPromise();
|
|
15
15
|
}
|
|
16
16
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: MaterialIssuesComponent, deps: [{ token: i1.FilterAPIService }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
17
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: MaterialIssuesComponent, selector: "metaqs2-material-issues", ngImport: i0, template: "<metaqs2-node-list\n *ngFor=\"let mode of issues\"\n [
|
|
17
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: MaterialIssuesComponent, selector: "metaqs2-material-issues", ngImport: i0, template: "<metaqs2-node-list\n *ngFor=\"let mode of issues\"\n [collectionId]=\"collectionId\"\n [mode]=\"mode\"\n type=\"material\"\n></metaqs2-node-list>\n", styles: [":host{padding:0 30px;display:flex;gap:30px;flex-wrap:wrap}:host>metaqs2-node-list{width:400px}\n"], dependencies: [{ kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "component", type: i3.NodeListComponent, selector: "metaqs2-node-list", inputs: ["mode", "type", "collectionId"], outputs: ["countChanged"] }] }); }
|
|
18
18
|
}
|
|
19
19
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: MaterialIssuesComponent, decorators: [{
|
|
20
20
|
type: Component,
|
|
21
|
-
args: [{ selector: 'metaqs2-material-issues', template: "<metaqs2-node-list\n *ngFor=\"let mode of issues\"\n [
|
|
21
|
+
args: [{ selector: 'metaqs2-material-issues', template: "<metaqs2-node-list\n *ngFor=\"let mode of issues\"\n [collectionId]=\"collectionId\"\n [mode]=\"mode\"\n type=\"material\"\n></metaqs2-node-list>\n", styles: [":host{padding:0 30px;display:flex;gap:30px;flex-wrap:wrap}:host>metaqs2-node-list{width:400px}\n"] }]
|
|
22
22
|
}], ctorParameters: () => [{ type: i1.FilterAPIService }] });
|
|
23
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
23
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWF0ZXJpYWwtaXNzdWVzLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25nLW1ldGEtd2lkZ2V0cy1saWIvc3JjL2xpYi9jb21wb25lbnRzL21hdGVyaWFsLWlzc3Vlcy9tYXRlcmlhbC1pc3N1ZXMuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmctbWV0YS13aWRnZXRzLWxpYi9zcmMvbGliL2NvbXBvbmVudHMvbWF0ZXJpYWwtaXNzdWVzL21hdGVyaWFsLWlzc3Vlcy5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFVLE1BQU0sZUFBZSxDQUFDOzs7OztBQVFsRCxNQUFNLE9BQU8sdUJBQXVCO0lBS2xDLFlBQW9CLGdCQUFrQztRQUFsQyxxQkFBZ0IsR0FBaEIsZ0JBQWdCLENBQWtCO1FBSnRELFdBQU0sR0FBYSxFQUFFLENBQUM7UUFDdEIsdUVBQXVFO1FBQ3ZFLGlCQUFZLEdBQVcsc0NBQXNDLENBQUM7SUFFTCxDQUFDO0lBRTFELEtBQUssQ0FBQyxRQUFRO1FBQ1osSUFBSSxDQUFDLE1BQU0sR0FBRyxNQUFNLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQywwQkFBMEIsRUFBRSxDQUFDLFNBQVMsRUFBRSxDQUFDO0lBQ3JGLENBQUM7K0dBVFUsdUJBQXVCO21HQUF2Qix1QkFBdUIsK0RDUnBDLHlKQU1BOzs0RkRFYSx1QkFBdUI7a0JBTG5DLFNBQVM7K0JBQ0UseUJBQXlCIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tcG9uZW50LCBPbkluaXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IEZpbHRlckFQSVNlcnZpY2UgfSBmcm9tICcuLi8uLi9qYXZhLWFwaSc7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ21ldGFxczItbWF0ZXJpYWwtaXNzdWVzJyxcbiAgdGVtcGxhdGVVcmw6ICcuL21hdGVyaWFsLWlzc3Vlcy5jb21wb25lbnQuaHRtbCcsXG4gIHN0eWxlVXJsczogWycuL21hdGVyaWFsLWlzc3Vlcy5jb21wb25lbnQuc2NzcyddLFxufSlcbmV4cG9ydCBjbGFzcyBNYXRlcmlhbElzc3Vlc0NvbXBvbmVudCBpbXBsZW1lbnRzIE9uSW5pdCB7XG4gIGlzc3Vlczogc3RyaW5nW10gPSBbXTtcbiAgLy8gQFRPRE86IElucHV0IGlzIGJyb2tlbiwgbWF5YmUgY2F1c2Ugb2Ygd2l0aENvbXBvbmVudElucHV0QmluZGluZygpPyFcbiAgY29sbGVjdGlvbklkOiBzdHJpbmcgPSAnOTRmMjJjOWItMGQzYS00YzFjLTg5ODctNGM4ZTgzZjNhOTJlJztcblxuICBjb25zdHJ1Y3Rvcihwcml2YXRlIGZpbHRlckFQSVNlcnZpY2U6IEZpbHRlckFQSVNlcnZpY2UpIHt9XG5cbiAgYXN5bmMgbmdPbkluaXQoKSB7XG4gICAgdGhpcy5pc3N1ZXMgPSBhd2FpdCB0aGlzLmZpbHRlckFQSVNlcnZpY2UuZ2V0TWF0ZXJpYWxJc3N1ZUZpZWxkTmFtZXMoKS50b1Byb21pc2UoKTtcbiAgfVxufVxuIiwiPG1ldGFxczItbm9kZS1saXN0XG4gICpuZ0Zvcj1cImxldCBtb2RlIG9mIGlzc3Vlc1wiXG4gIFtjb2xsZWN0aW9uSWRdPVwiY29sbGVjdGlvbklkXCJcbiAgW21vZGVdPVwibW9kZVwiXG4gIHR5cGU9XCJtYXRlcmlhbFwiXG4+PC9tZXRhcXMyLW5vZGUtbGlzdD5cbiJdfQ==
|
|
@@ -1,6 +1,6 @@
|
|
|
1
|
-
import { Component, inject, Input } from '@angular/core';
|
|
1
|
+
import { Component, EventEmitter, inject, Input, Output } from '@angular/core';
|
|
2
2
|
import { BehaviorSubject, Subject } from 'rxjs';
|
|
3
|
-
import { map, shareReplay } from 'rxjs/operators';
|
|
3
|
+
import { map, shareReplay, takeUntil } from 'rxjs/operators';
|
|
4
4
|
import { wrapResponse } from '../../wrap-observable.pipe';
|
|
5
5
|
import { ConfigHelperService } from '../../config-helper.service';
|
|
6
6
|
import * as i0 from "@angular/core";
|
|
@@ -18,6 +18,10 @@ export class NodeListComponent {
|
|
|
18
18
|
constructor(collectionAPIService) {
|
|
19
19
|
this.collectionAPIService = collectionAPIService;
|
|
20
20
|
this.env = inject(ConfigHelperService);
|
|
21
|
+
/**
|
|
22
|
+
* outputs the current count of materials in this view
|
|
23
|
+
*/
|
|
24
|
+
this.countChanged = new EventEmitter();
|
|
21
25
|
/**
|
|
22
26
|
* default max count of nodes that should be shown
|
|
23
27
|
*/
|
|
@@ -30,9 +34,10 @@ export class NodeListComponent {
|
|
|
30
34
|
* should all nodes be shown?
|
|
31
35
|
*/
|
|
32
36
|
this.showAll = false;
|
|
37
|
+
this.data.pipe(takeUntil(this.destroyed$)).subscribe((d) => this.countChanged.next(d?.length || 0));
|
|
33
38
|
}
|
|
34
39
|
async ngOnChanges(changes) {
|
|
35
|
-
if (this.
|
|
40
|
+
if (this.collectionId) {
|
|
36
41
|
await this.refresh();
|
|
37
42
|
}
|
|
38
43
|
}
|
|
@@ -43,10 +48,10 @@ export class NodeListComponent {
|
|
|
43
48
|
async refresh() {
|
|
44
49
|
let getByMissingAttribute = () => {
|
|
45
50
|
if (this.type === 'material') {
|
|
46
|
-
return this.collectionAPIService.getMaterialsWithMissingAttribute(this.
|
|
51
|
+
return this.collectionAPIService.getMaterialsWithMissingAttribute(this.collectionId, this.mode);
|
|
47
52
|
}
|
|
48
53
|
else {
|
|
49
|
-
return this.collectionAPIService.getCollectionsWithMissingAttribute(this.
|
|
54
|
+
return this.collectionAPIService.getCollectionsWithMissingAttribute(this.collectionId, this.mode);
|
|
50
55
|
}
|
|
51
56
|
};
|
|
52
57
|
// reset all data before switching
|
|
@@ -86,7 +91,7 @@ export class NodeListComponent {
|
|
|
86
91
|
// this.data = this.rawData?.filter((d) => (d as MaterialCounts).materials_count <= (this.count ?? 0));
|
|
87
92
|
}
|
|
88
93
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: NodeListComponent, deps: [{ token: i1.CollectionAPIService }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
89
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: NodeListComponent, selector: "metaqs2-node-list", inputs: { mode: "mode", type: "type",
|
|
94
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: NodeListComponent, selector: "metaqs2-node-list", inputs: { mode: "mode", type: "type", collectionId: "collectionId" }, outputs: { countChanged: "countChanged" }, usesOnChanges: true, ngImport: i0, template: "<div class=\"wrapper\">\n <mat-card *ngIf=\"mode\">\n <mat-card-header>\n <mat-card-title>\n {{ 'issues.' + mode | translate }}\n <span class=\"count\" *ngIf=\"(data | async) as d\">({{ d.length || 0 }})</span>\n </mat-card-title>\n <button mat-icon-button color=\"primary\" [disabled]=\"!(data | async)\" (click)=\"refresh()\">\n <mat-icon>refresh</mat-icon>\n </button>\n </mat-card-header>\n <!-- @TODO -->\n <div *ngIf=\"type === 'collection' && mode === 'TODO_count'\" class=\"count-slider\">\n <label>Sammlungen mit {{ count }} oder weniger Inhalten</label>\n <mat-slider\n [showTickMarks]=\"true\"\n step=\"1\"\n [ngModel]=\"count\"\n (ngModelChange)=\"count = $event; filterCount()\"\n min=\"0\"\n max=\"10\"\n ></mat-slider>\n </div>\n <ng-container *ngIf=\"wrappedData$ | async as wrappedData\">\n <mat-spinner *ngIf=\"wrappedData.state === 'loading'\" diameter=\"40\" color=\"primary\"></mat-spinner>\n <div class=\"error\" *ngIf=\"wrappedData.state === 'error'\">\n <p class=\"error-text\">Etwas ist schief gelaufen.</p>\n </div>\n <div class=\"scroll\" *ngIf=\"wrappedData.state === 'success'\">\n <ng-container *ngIf=\"(data | async) as d\">\n <metaqs2-node-entry\n *ngFor=\"let node of DEFAULT_LIMIT < d.length && !showAll ? d.slice(0, DEFAULT_LIMIT) : d\"\n [node]=\"node\"\n (edit)=\"editNode(node)\"\n ></metaqs2-node-entry>\n <div *ngIf=\"!d.length\">\n <div class=\"all-done\">\n <p class=\"all-done-emoji\">\uD83E\uDD73</p>\n <p class=\"all-done-text\">Keine Elemente ausstehend</p>\n </div>\n </div>\n <button *ngIf=\"DEFAULT_LIMIT < d.length && !showAll\" (click)=\"showAll = true\" color=\"primary\" mat-button>\n Alle Elemente anzeigen <mat-icon>keyboard_arrow_down</mat-icon>\n </button>\n </ng-container>\n </div>\n </ng-container>\n </mat-card>\n</div>\n", styles: [".wrapper{width:100%;height:100%}.wrapper mat-card{display:flex;flex-direction:column;width:100%;height:100%;padding:0;box-shadow:none}.wrapper mat-card ::ng-deep mat-card-header{align-items:center}.wrapper mat-card ::ng-deep mat-card-header .mat-card-header-text{flex-grow:1;margin:0}.wrapper mat-card ::ng-deep mat-card-header .mat-card-header-text mat-card-title{padding-top:20px;font-size:120%}.wrapper mat-card ::ng-deep mat-card-header .mat-card-header-text mat-card-title .count{font-size:80%;color:#888}.wrapper mat-card .count-slider{display:flex;flex-direction:column;width:100%}.wrapper mat-card .count-slider>label{padding:0 20px;font-size:90%}.wrapper mat-card .count-slider>mat-slider{width:100%}.wrapper mat-card mat-spinner{flex-grow:1;display:flex;align-self:center;justify-self:center;margin-top:20%}.wrapper mat-card .scroll{flex-grow:1;overflow-y:auto}.wrapper mat-card .scroll>button{display:flex;margin:auto}.wrapper mat-card .scroll .all-done{display:flex;flex-direction:column;align-items:center}.wrapper mat-card .scroll .all-done .all-done-emoji{font-size:72px;margin-bottom:20px}.wrapper mat-card .scroll .all-done .all-done-text{text-align:center;font-size:1.3em;color:var(--font-light)}.wrapper mat-card .error{border:#d00 solid 2px;background-color:#d005;display:flex;flex-direction:column;align-items:center}.wrapper mat-card .error .error-text{font-size:1.3em}\n"], dependencies: [{ kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i3.MatCard, selector: "mat-card", inputs: ["appearance"], exportAs: ["matCard"] }, { kind: "component", type: i3.MatCardHeader, selector: "mat-card-header" }, { kind: "directive", type: i3.MatCardTitle, selector: "mat-card-title, [mat-card-title], [matCardTitle]" }, { kind: "component", type: i4.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: i5.MatSlider, selector: "mat-slider", inputs: ["disabled", "discrete", "showTickMarks", "min", "color", "disableRipple", "max", "step", "displayWith"], exportAs: ["matSlider"] }, { kind: "component", type: i6.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "component", type: i6.MatIconButton, selector: "button[mat-icon-button]", exportAs: ["matButton"] }, { kind: "component", type: i7.MatProgressSpinner, selector: "mat-progress-spinner, mat-spinner", inputs: ["color", "mode", "value", "diameter", "strokeWidth"], exportAs: ["matProgressSpinner"] }, { kind: "directive", type: i8.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i8.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i9.NodeEntryComponent, selector: "metaqs2-node-entry", inputs: ["node"], outputs: ["edit"] }, { kind: "pipe", type: i2.AsyncPipe, name: "async" }, { kind: "pipe", type: i10.TranslatePipe, name: "translate" }] }); }
|
|
90
95
|
}
|
|
91
96
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: NodeListComponent, decorators: [{
|
|
92
97
|
type: Component,
|
|
@@ -95,7 +100,9 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
95
100
|
type: Input
|
|
96
101
|
}], type: [{
|
|
97
102
|
type: Input
|
|
98
|
-
}],
|
|
103
|
+
}], collectionId: [{
|
|
99
104
|
type: Input
|
|
105
|
+
}], countChanged: [{
|
|
106
|
+
type: Output
|
|
100
107
|
}] } });
|
|
101
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node-list.component.js","sourceRoot":"","sources":["../../../../../../projects/ng-meta-widgets-lib/src/lib/components/node-list/node-list.component.ts","../../../../../../projects/ng-meta-widgets-lib/src/lib/components/node-list/node-list.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, inject, Input, OnChanges, OnDestroy, SimpleChanges } from '@angular/core';\nimport { BehaviorSubject, Observable, Subject } from 'rxjs';\nimport { map, shareReplay } from 'rxjs/operators';\nimport { CollectionAPIService, CollectionWithMissingAttributes, MaterialWithMissingAttributes } from '../../java-api';\nimport { WrappedResponse, wrapResponse } from '../../wrap-observable.pipe';\nimport { ConfigHelperService } from '../../config-helper.service';\n\n@Component({\n  selector: 'metaqs2-node-list',\n  templateUrl: './node-list.component.html',\n  styleUrls: ['./node-list.component.scss'],\n})\nexport class NodeListComponent implements OnChanges, OnDestroy {\n  private env = inject(ConfigHelperService);\n  @Input() mode: string;\n  @Input() type: 'material' | 'collection';\n  @Input() parentId: string;\n\n  /**\n   * default max count of nodes that should be shown\n   */\n  readonly DEFAULT_LIMIT = 25;\n  wrappedData$: Observable<WrappedResponse<(MaterialWithMissingAttributes | CollectionWithMissingAttributes)[]>>;\n  data = new BehaviorSubject<(MaterialWithMissingAttributes | CollectionWithMissingAttributes)[] | undefined>(\n    undefined\n  );\n  // rawData: (MaterialWithMissingAttributes | CollectionWithMissingAttributes)[] | undefined;\n  count: number | null = 0;\n\n  private destroyed$ = new Subject<void>();\n  /**\n   * should all nodes be shown?\n   */\n  showAll = false;\n\n  constructor(private collectionAPIService: CollectionAPIService) {}\n\n  async ngOnChanges(changes: SimpleChanges) {\n    if (this.parentId) {\n      await this.refresh();\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.destroyed$.next();\n    this.destroyed$.complete();\n  }\n\n  async refresh() {\n    let getByMissingAttribute = (): Observable<(MaterialWithMissingAttributes | CollectionWithMissingAttributes)[]> => {\n      if (this.type === 'material') {\n        return this.collectionAPIService.getMaterialsWithMissingAttribute(this.parentId, this.mode);\n      } else {\n        return this.collectionAPIService.getCollectionsWithMissingAttribute(this.parentId, this.mode);\n      }\n    };\n\n    // reset all data before switching\n    this.data.next(undefined);\n    //this.rawData = undefined;\n    this.wrappedData$ = getByMissingAttribute().pipe(wrapResponse(), shareReplay(1));\n    this.data.next(\n      await this.wrappedData$\n        .pipe(map((wrappedData) => (wrappedData.state === 'success' ? wrappedData.data : undefined)))\n        .toPromise()\n    );\n    //this.rawData = this.data.value?.slice();\n\n    /**\n     * @TODO: Count mode\n     */\n    /*\n        if (?) {\n          this.filterCount();\n        }\n         */\n  }\n\n  editNode(node: MaterialWithMissingAttributes | CollectionWithMissingAttributes) {\n    let action: string;\n    if (this.mode === 'without_license') {\n      action = 'OPTIONS.LICENSE';\n    } else {\n      action = 'OPTIONS.EDIT';\n    }\n\n    const id = node.node_id!!;\n    const win = window.open(\n      this.env.eduSharingPath +\n        '/components/editorial-desk?mode=audit&viewType=single&ids=' +\n        encodeURIComponent(id) +\n        '&action=' +\n        action\n    );\n  }\n\n  filterCount() {\n    // @TODO\n    // this.data = this.rawData?.filter((d) => (d as MaterialCounts).materials_count <= (this.count ?? 0));\n  }\n}\n","<div class=\"wrapper\">\n  <mat-card *ngIf=\"mode\">\n    <mat-card-header>\n      <mat-card-title>\n        {{ 'issues.' + mode  | translate }}\n        <span class=\"count\" *ngIf=\"(data | async) as d\">({{ d.length || 0 }})</span>\n      </mat-card-title>\n      <button mat-icon-button color=\"primary\" [disabled]=\"!(data | async)\" (click)=\"refresh()\">\n        <mat-icon>refresh</mat-icon>\n      </button>\n    </mat-card-header>\n    <!-- @TODO -->\n    <div *ngIf=\"type === 'collection' && mode === 'TODO_count'\" class=\"count-slider\">\n      <label>Sammlungen mit {{ count }} oder weniger Inhalten</label>\n      <mat-slider\n        [showTickMarks]=\"true\"\n        step=\"1\"\n        [ngModel]=\"count\"\n        (ngModelChange)=\"count = $event; filterCount()\"\n        min=\"0\"\n        max=\"10\"\n      ></mat-slider>\n    </div>\n    <ng-container *ngIf=\"wrappedData$ | async as wrappedData\">\n      <mat-spinner *ngIf=\"wrappedData.state === 'loading'\" diameter=\"40\" color=\"primary\"></mat-spinner>\n      <div class=\"error\" *ngIf=\"wrappedData.state === 'error'\">\n        <p class=\"error-text\">Etwas ist schief gelaufen.</p>\n      </div>\n      <div class=\"scroll\" *ngIf=\"wrappedData.state === 'success'\">\n        <ng-container *ngIf=\"(data | async) as d\">\n          <metaqs2-node-entry\n            *ngFor=\"let node of DEFAULT_LIMIT < d.length && !showAll ? d.slice(0, DEFAULT_LIMIT) : d\"\n            [node]=\"node\"\n            (edit)=\"editNode(node)\"\n          ></metaqs2-node-entry>\n          <div *ngIf=\"!d.length\">\n            <div class=\"all-done\">\n              <p class=\"all-done-emoji\">🥳</p>\n              <p class=\"all-done-text\">Keine Elemente ausstehend</p>\n            </div>\n          </div>\n          <button *ngIf=\"DEFAULT_LIMIT < d.length && !showAll\" (click)=\"showAll = true\" color=\"primary\" mat-button>\n            Alle Elemente anzeigen <mat-icon>keyboard_arrow_down</mat-icon>\n          </button>\n        </ng-container>\n      </div>\n    </ng-container>\n  </mat-card>\n</div>\n"]}
|
|
108
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node-list.component.js","sourceRoot":"","sources":["../../../../../../projects/ng-meta-widgets-lib/src/lib/components/node-list/node-list.component.ts","../../../../../../projects/ng-meta-widgets-lib/src/lib/components/node-list/node-list.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,MAAM,EAAE,KAAK,EAAwB,MAAM,EAAiB,MAAM,eAAe,CAAC;AACpH,OAAO,EAAE,eAAe,EAAc,OAAO,EAAE,MAAM,MAAM,CAAC;AAC5D,OAAO,EAAE,GAAG,EAAE,WAAW,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AAE7D,OAAO,EAAmB,YAAY,EAAE,MAAM,4BAA4B,CAAC;AAC3E,OAAO,EAAE,mBAAmB,EAAE,MAAM,6BAA6B,CAAC;;;;;;;;;;;;AAOlE,MAAM,OAAO,iBAAiB;IA2B5B,YAAoB,oBAA0C;QAA1C,yBAAoB,GAApB,oBAAoB,CAAsB;QA1BtD,QAAG,GAAG,MAAM,CAAC,mBAAmB,CAAC,CAAC;QAI1C;;WAEG;QACO,iBAAY,GAAG,IAAI,YAAY,EAAU,CAAC;QAEpD;;WAEG;QACM,kBAAa,GAAG,EAAE,CAAC;QAE5B,SAAI,GAAG,IAAI,eAAe,CACxB,SAAS,CACV,CAAC;QACF,4FAA4F;QAC5F,UAAK,GAAkB,CAAC,CAAC;QAEjB,eAAU,GAAG,IAAI,OAAO,EAAQ,CAAC;QACzC;;WAEG;QACH,YAAO,GAAG,KAAK,CAAC;QAGd,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC,EAAE,MAAM,IAAI,CAAC,CAAC,CAAC,CAAC;IACtG,CAAC;IAED,KAAK,CAAC,WAAW,CAAC,OAAsB;QACtC,IAAI,IAAI,CAAC,YAAY,EAAE,CAAC;YACtB,MAAM,IAAI,CAAC,OAAO,EAAE,CAAC;QACvB,CAAC;IACH,CAAC;IAED,WAAW;QACT,IAAI,CAAC,UAAU,CAAC,IAAI,EAAE,CAAC;QACvB,IAAI,CAAC,UAAU,CAAC,QAAQ,EAAE,CAAC;IAC7B,CAAC;IAED,KAAK,CAAC,OAAO;QACX,IAAI,qBAAqB,GAAG,GAAoF,EAAE;YAChH,IAAI,IAAI,CAAC,IAAI,KAAK,UAAU,EAAE,CAAC;gBAC7B,OAAO,IAAI,CAAC,oBAAoB,CAAC,gCAAgC,CAAC,IAAI,CAAC,YAAY,EAAE,IAAI,CAAC,IAAI,CAAC,CAAC;YAClG,CAAC;iBAAM,CAAC;gBACN,OAAO,IAAI,CAAC,oBAAoB,CAAC,kCAAkC,CAAC,IAAI,CAAC,YAAY,EAAE,IAAI,CAAC,IAAI,CAAC,CAAC;YACpG,CAAC;QACH,CAAC,CAAC;QAEF,kCAAkC;QAClC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QAC1B,2BAA2B;QAC3B,IAAI,CAAC,YAAY,GAAG,qBAAqB,EAAE,CAAC,IAAI,CAAC,YAAY,EAAE,EAAE,WAAW,CAAC,CAAC,CAAC,CAAC,CAAC;QACjF,IAAI,CAAC,IAAI,CAAC,IAAI,CACZ,MAAM,IAAI,CAAC,YAAY;aACpB,IAAI,CAAC,GAAG,CAAC,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC,WAAW,CAAC,KAAK,KAAK,SAAS,CAAC,CAAC,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC;aAC5F,SAAS,EAAE,CACf,CAAC;QACF,0CAA0C;QAE1C;;WAEG;QACH;;;;eAIO;IACT,CAAC;IAED,QAAQ,CAAC,IAAqE;QAC5E,IAAI,MAAc,CAAC;QACnB,IAAI,IAAI,CAAC,IAAI,KAAK,iBAAiB,EAAE,CAAC;YACpC,MAAM,GAAG,iBAAiB,CAAC;QAC7B,CAAC;aAAM,CAAC;YACN,MAAM,GAAG,cAAc,CAAC;QAC1B,CAAC;QAED,MAAM,EAAE,GAAG,IAAI,CAAC,OAAS,CAAC;QAC1B,MAAM,GAAG,GAAG,MAAM,CAAC,IAAI,CACrB,IAAI,CAAC,GAAG,CAAC,cAAc;YACrB,4DAA4D;YAC5D,kBAAkB,CAAC,EAAE,CAAC;YACtB,UAAU;YACV,MAAM,CACT,CAAC;IACJ,CAAC;IAED,WAAW;QACT,QAAQ;QACR,uGAAuG;IACzG,CAAC;+GA7FU,iBAAiB;mGAAjB,iBAAiB,+LCZ9B,kjEAiDA;;4FDrCa,iBAAiB;kBAL7B,SAAS;+BACE,mBAAmB;yFAMpB,IAAI;sBAAZ,KAAK;gBACG,IAAI;sBAAZ,KAAK;gBACG,YAAY;sBAApB,KAAK;gBAII,YAAY;sBAArB,MAAM","sourcesContent":["import { Component, EventEmitter, inject, Input, OnChanges, OnDestroy, Output, SimpleChanges } from '@angular/core';\nimport { BehaviorSubject, Observable, Subject } from 'rxjs';\nimport { map, shareReplay, takeUntil } from 'rxjs/operators';\nimport { CollectionAPIService, CollectionWithMissingAttributes, MaterialWithMissingAttributes } from '../../java-api';\nimport { WrappedResponse, wrapResponse } from '../../wrap-observable.pipe';\nimport { ConfigHelperService } from '../../config-helper.service';\n\n@Component({\n  selector: 'metaqs2-node-list',\n  templateUrl: './node-list.component.html',\n  styleUrls: ['./node-list.component.scss'],\n})\nexport class NodeListComponent implements OnChanges, OnDestroy {\n  private env = inject(ConfigHelperService);\n  @Input() mode: string;\n  @Input() type: 'material' | 'collection';\n  @Input() collectionId: string;\n  /**\n   * outputs the current count of materials in this view\n   */\n  @Output() countChanged = new EventEmitter<number>();\n\n  /**\n   * default max count of nodes that should be shown\n   */\n  readonly DEFAULT_LIMIT = 25;\n  wrappedData$: Observable<WrappedResponse<(MaterialWithMissingAttributes | CollectionWithMissingAttributes)[]>>;\n  data = new BehaviorSubject<(MaterialWithMissingAttributes | CollectionWithMissingAttributes)[] | undefined>(\n    undefined\n  );\n  // rawData: (MaterialWithMissingAttributes | CollectionWithMissingAttributes)[] | undefined;\n  count: number | null = 0;\n\n  private destroyed$ = new Subject<void>();\n  /**\n   * should all nodes be shown?\n   */\n  showAll = false;\n\n  constructor(private collectionAPIService: CollectionAPIService) {\n    this.data.pipe(takeUntil(this.destroyed$)).subscribe((d) => this.countChanged.next(d?.length || 0));\n  }\n\n  async ngOnChanges(changes: SimpleChanges) {\n    if (this.collectionId) {\n      await this.refresh();\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.destroyed$.next();\n    this.destroyed$.complete();\n  }\n\n  async refresh() {\n    let getByMissingAttribute = (): Observable<(MaterialWithMissingAttributes | CollectionWithMissingAttributes)[]> => {\n      if (this.type === 'material') {\n        return this.collectionAPIService.getMaterialsWithMissingAttribute(this.collectionId, this.mode);\n      } else {\n        return this.collectionAPIService.getCollectionsWithMissingAttribute(this.collectionId, this.mode);\n      }\n    };\n\n    // reset all data before switching\n    this.data.next(undefined);\n    //this.rawData = undefined;\n    this.wrappedData$ = getByMissingAttribute().pipe(wrapResponse(), shareReplay(1));\n    this.data.next(\n      await this.wrappedData$\n        .pipe(map((wrappedData) => (wrappedData.state === 'success' ? wrappedData.data : undefined)))\n        .toPromise()\n    );\n    //this.rawData = this.data.value?.slice();\n\n    /**\n     * @TODO: Count mode\n     */\n    /*\n        if (?) {\n          this.filterCount();\n        }\n         */\n  }\n\n  editNode(node: MaterialWithMissingAttributes | CollectionWithMissingAttributes) {\n    let action: string;\n    if (this.mode === 'without_license') {\n      action = 'OPTIONS.LICENSE';\n    } else {\n      action = 'OPTIONS.EDIT';\n    }\n\n    const id = node.node_id!!;\n    const win = window.open(\n      this.env.eduSharingPath +\n        '/components/editorial-desk?mode=audit&viewType=single&ids=' +\n        encodeURIComponent(id) +\n        '&action=' +\n        action\n    );\n  }\n\n  filterCount() {\n    // @TODO\n    // this.data = this.rawData?.filter((d) => (d as MaterialCounts).materials_count <= (this.count ?? 0));\n  }\n}\n","<div class=\"wrapper\">\n  <mat-card *ngIf=\"mode\">\n    <mat-card-header>\n      <mat-card-title>\n        {{ 'issues.' + mode  | translate }}\n        <span class=\"count\" *ngIf=\"(data | async) as d\">({{ d.length || 0 }})</span>\n      </mat-card-title>\n      <button mat-icon-button color=\"primary\" [disabled]=\"!(data | async)\" (click)=\"refresh()\">\n        <mat-icon>refresh</mat-icon>\n      </button>\n    </mat-card-header>\n    <!-- @TODO -->\n    <div *ngIf=\"type === 'collection' && mode === 'TODO_count'\" class=\"count-slider\">\n      <label>Sammlungen mit {{ count }} oder weniger Inhalten</label>\n      <mat-slider\n        [showTickMarks]=\"true\"\n        step=\"1\"\n        [ngModel]=\"count\"\n        (ngModelChange)=\"count = $event; filterCount()\"\n        min=\"0\"\n        max=\"10\"\n      ></mat-slider>\n    </div>\n    <ng-container *ngIf=\"wrappedData$ | async as wrappedData\">\n      <mat-spinner *ngIf=\"wrappedData.state === 'loading'\" diameter=\"40\" color=\"primary\"></mat-spinner>\n      <div class=\"error\" *ngIf=\"wrappedData.state === 'error'\">\n        <p class=\"error-text\">Etwas ist schief gelaufen.</p>\n      </div>\n      <div class=\"scroll\" *ngIf=\"wrappedData.state === 'success'\">\n        <ng-container *ngIf=\"(data | async) as d\">\n          <metaqs2-node-entry\n            *ngFor=\"let node of DEFAULT_LIMIT < d.length && !showAll ? d.slice(0, DEFAULT_LIMIT) : d\"\n            [node]=\"node\"\n            (edit)=\"editNode(node)\"\n          ></metaqs2-node-entry>\n          <div *ngIf=\"!d.length\">\n            <div class=\"all-done\">\n              <p class=\"all-done-emoji\">🥳</p>\n              <p class=\"all-done-text\">Keine Elemente ausstehend</p>\n            </div>\n          </div>\n          <button *ngIf=\"DEFAULT_LIMIT < d.length && !showAll\" (click)=\"showAll = true\" color=\"primary\" mat-button>\n            Alle Elemente anzeigen <mat-icon>keyboard_arrow_down</mat-icon>\n          </button>\n        </ng-container>\n      </div>\n    </ng-container>\n  </mat-card>\n</div>\n"]}
|
|
@@ -20,7 +20,7 @@ export class InlineWorker {
|
|
|
20
20
|
// Perform the fetch request
|
|
21
21
|
const { basePath, collectionId, searchTerm, oerOnly } = data;
|
|
22
22
|
searchForTypeAndCollection(basePath, searchTerm, oerOnly)
|
|
23
|
-
.then((response) => (response.ok ? response : Promise.reject(response)))
|
|
23
|
+
.then((response) => (response.ok ? response : Promise.reject(response)), (error) => Promise.reject(error))
|
|
24
24
|
.then((response) => response.json())
|
|
25
25
|
.then((data) => {
|
|
26
26
|
resolve({ collectionId, data });
|
|
@@ -46,12 +46,11 @@ export class InlineWorker {
|
|
|
46
46
|
});
|
|
47
47
|
}
|
|
48
48
|
const searchForTypeAndCollection = (basePath, searchTerm, oerOnly = false) => {
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
return fetch(url, { credentials: 'include' });
|
|
49
|
+
//strip trailing slash from basePath, if present
|
|
50
|
+
const searchUrl = new URL(basePath.replace(/\/$/, '') + '/collections/v2/material/search');
|
|
51
|
+
searchUrl.searchParams.set('term', searchTerm);
|
|
52
|
+
searchUrl.searchParams.set('oerOnly', oerOnly ? 'true' : 'false');
|
|
53
|
+
return fetch(searchUrl, { credentials: 'include' });
|
|
55
54
|
};
|
|
56
55
|
self.onmessage = (evt) => {
|
|
57
56
|
enqueueRequest(evt)
|
|
@@ -100,4 +99,4 @@ export class InlineWorker {
|
|
|
100
99
|
}
|
|
101
100
|
}
|
|
102
101
|
}
|
|
103
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"inline-worker.js","sourceRoot":"","sources":["../../../../../projects/ng-meta-widgets-lib/src/lib/tree-search-counts/inline-worker.ts"],"names":[],"mappings":"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","sourcesContent":["import { Observable, Subject } from 'rxjs';\nexport interface RequestData {\n  basePath: string;\n  collectionId: string;\n  searchTerm: string;\n  oerOnly: boolean;\n}\n\nexport class InlineWorker {\n  private readonly worker: Worker;\n  private onMessage = new Subject<MessageEvent>();\n  private onError = new Subject<ErrorEvent>();\n\n  private func = () => {\n    // START OF WORKER THREAD CODE\n    console.debug('Start worker thread, wait for postMessage: ');\n\n    const maxConcurrentRequests = 10; // Limit the number of concurrent fetch requests\n    let activeRequests = 0;\n    let requestQueue: { data: RequestData; resolve: any; reject: any }[] = [];\n\n    // Function to process the next request in the queue\n    function processQueue() {\n      if (activeRequests >= maxConcurrentRequests || requestQueue.length === 0) {\n        return; // Exit if max requests are active or queue is empty\n      }\n      // @ts-ignore\n      const { data, resolve, reject } = requestQueue.shift();\n      activeRequests++;\n\n      // Perform the fetch request\n      const { basePath, collectionId, searchTerm, oerOnly } = data;\n      searchForTypeAndCollection(basePath, searchTerm, oerOnly)\n        .then((response) => (response.ok ? response : Promise.reject(response)))\n        .then((response) => response.json())\n        .then((data) => {\n          resolve({ collectionId, data });\n        })\n        .catch((error) => {\n          reject(error); // Handle the error\n        })\n        .finally(() => {\n          activeRequests--; // Decrement active request count\n          processQueue(); // Process the next request in the queue\n        });\n    }\n\n    // Add a new fetch request to the queue\n    function enqueueRequest(event: MessageEvent) {\n      return new Promise((resolve, reject) => {\n        try {\n          requestQueue.push({ data: event.data, resolve, reject });\n          processQueue(); // Start processing the queue\n        } catch (error) {\n          console.error('Error enqueuing request:', error);\n        }\n      });\n    }\n\n    const searchForTypeAndCollection = (basePath: string, searchTerm: string, oerOnly: boolean = false) => {\n      const urlParams = new URLSearchParams({\n        term: searchTerm,\n        oerOnly: oerOnly ? 'true' : 'false',\n      });\n      const url = basePath + '/collections/v2/material/search?' + urlParams;\n      return fetch(url, { credentials: 'include' });\n    };\n\n    self.onmessage = (evt: MessageEvent) => {\n      enqueueRequest(evt)\n        .then((data) => {\n          postMessage(data);\n        })\n        .catch((error) => {\n          postMessage({ success: false, error: error.message });\n        });\n    };\n    // END OF WORKER THREAD CODE\n  };\n  private basePath: any;\n\n  constructor({ basePath }: { basePath: string }) {\n    this.basePath = basePath;\n    const WORKER_ENABLED = !!Worker;\n\n    if (WORKER_ENABLED) {\n      const functionBody = this.func\n        .toString()\n        .replace(/^[^{]*{\\s*/, '')\n        .replace(/\\s*}[^}]*$/, '');\n\n      this.worker = new Worker(URL.createObjectURL(new Blob([functionBody], { type: 'text/javascript' })));\n\n      this.worker.onmessage = (data) => {\n        this.onMessage.next(data);\n      };\n\n      this.worker.onerror = (data) => {\n        console.error('workers error: ', data);\n        this.onError.next(data);\n      };\n    } else {\n      throw new Error('WebWorker is not enabled');\n    }\n  }\n\n  postMessage(data: Partial<RequestData>) {\n    this.worker.postMessage({ basePath: this.basePath, ...data });\n  }\n\n  onmessage(): Observable<MessageEvent> {\n    return this.onMessage.asObservable();\n  }\n\n  onerror(): Observable<ErrorEvent> {\n    return this.onError.asObservable();\n  }\n  //TODO: implement graceful shutdown and abort outstanding requests\n  terminate() {\n    if (this.worker) {\n      this.worker.terminate();\n    }\n  }\n}\n"]}
|
|
102
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"inline-worker.js","sourceRoot":"","sources":["../../../../../projects/ng-meta-widgets-lib/src/lib/tree-search-counts/inline-worker.ts"],"names":[],"mappings":"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","sourcesContent":["import { Observable, Subject } from 'rxjs';\nexport interface RequestData {\n  basePath: string;\n  collectionId: string;\n  searchTerm: string;\n  oerOnly: boolean;\n}\n\nexport class InlineWorker {\n  private readonly worker: Worker;\n  private onMessage = new Subject<MessageEvent>();\n  private onError = new Subject<ErrorEvent>();\n\n  private func = () => {\n    // START OF WORKER THREAD CODE\n    console.debug('Start worker thread, wait for postMessage: ');\n\n    const maxConcurrentRequests = 10; // Limit the number of concurrent fetch requests\n    let activeRequests = 0;\n    let requestQueue: { data: RequestData; resolve: any; reject: any }[] = [];\n\n    // Function to process the next request in the queue\n    function processQueue() {\n      if (activeRequests >= maxConcurrentRequests || requestQueue.length === 0) {\n        return; // Exit if max requests are active or queue is empty\n      }\n      // @ts-ignore\n      const { data, resolve, reject } = requestQueue.shift();\n      activeRequests++;\n\n      // Perform the fetch request\n      const { basePath, collectionId, searchTerm, oerOnly } = data;\n      searchForTypeAndCollection(basePath, searchTerm, oerOnly)\n        .then(\n          (response) => (response.ok ? response : Promise.reject(response)),\n          (error) => Promise.reject(error)\n        )\n        .then((response) => response.json())\n        .then((data) => {\n          resolve({ collectionId, data });\n        })\n        .catch((error) => {\n          reject(error); // Handle the error\n        })\n        .finally(() => {\n          activeRequests--; // Decrement active request count\n          processQueue(); // Process the next request in the queue\n        });\n    }\n\n    // Add a new fetch request to the queue\n    function enqueueRequest(event: MessageEvent) {\n      return new Promise((resolve, reject) => {\n        try {\n          requestQueue.push({ data: event.data, resolve, reject });\n          processQueue(); // Start processing the queue\n        } catch (error) {\n          console.error('Error enqueuing request:', error);\n        }\n      });\n    }\n\n    const searchForTypeAndCollection = (basePath: string, searchTerm: string, oerOnly: boolean = false) => {\n      //strip trailing slash from basePath, if present\n      const searchUrl = new URL(basePath.replace(/\\/$/, '') + '/collections/v2/material/search');\n      searchUrl.searchParams.set('term', searchTerm);\n      searchUrl.searchParams.set('oerOnly', oerOnly ? 'true' : 'false');\n      return fetch(searchUrl, { credentials: 'include' });\n    };\n\n    self.onmessage = (evt: MessageEvent) => {\n      enqueueRequest(evt)\n        .then((data) => {\n          postMessage(data);\n        })\n        .catch((error) => {\n          postMessage({ success: false, error: error.message });\n        });\n    };\n    // END OF WORKER THREAD CODE\n  };\n  private basePath: any;\n\n  constructor({ basePath }: { basePath: string }) {\n    this.basePath = basePath;\n    const WORKER_ENABLED = !!Worker;\n\n    if (WORKER_ENABLED) {\n      const functionBody = this.func\n        .toString()\n        .replace(/^[^{]*{\\s*/, '')\n        .replace(/\\s*}[^}]*$/, '');\n\n      this.worker = new Worker(URL.createObjectURL(new Blob([functionBody], { type: 'text/javascript' })));\n\n      this.worker.onmessage = (data) => {\n        this.onMessage.next(data);\n      };\n\n      this.worker.onerror = (data) => {\n        console.error('workers error: ', data);\n        this.onError.next(data);\n      };\n    } else {\n      throw new Error('WebWorker is not enabled');\n    }\n  }\n\n  postMessage(data: Partial<RequestData>) {\n    this.worker.postMessage({ basePath: this.basePath, ...data });\n  }\n\n  onmessage(): Observable<MessageEvent> {\n    return this.onMessage.asObservable();\n  }\n\n  onerror(): Observable<ErrorEvent> {\n    return this.onError.asObservable();\n  }\n  //TODO: implement graceful shutdown and abort outstanding requests\n  terminate() {\n    if (this.worker) {\n      this.worker.terminate();\n    }\n  }\n}\n"]}
|
|
@@ -1,8 +1,9 @@
|
|
|
1
|
-
import { ChangeDetectionStrategy, Component, Input, signal } from '@angular/core';
|
|
1
|
+
import { ChangeDetectionStrategy, Component, Inject, Input, signal, } from '@angular/core';
|
|
2
2
|
import { BehaviorSubject } from 'rxjs';
|
|
3
3
|
import { take } from 'rxjs/operators';
|
|
4
4
|
import { FormControl } from '@angular/forms';
|
|
5
5
|
import { InlineWorker } from './inline-worker';
|
|
6
|
+
import { DOCUMENT } from '@angular/common';
|
|
6
7
|
import * as i0 from "@angular/core";
|
|
7
8
|
import * as i1 from "../meta-api.service";
|
|
8
9
|
import * as i2 from "../config-helper.service";
|
|
@@ -15,10 +16,11 @@ import * as i8 from "@angular/forms";
|
|
|
15
16
|
import * as i9 from "@angular/material/tooltip";
|
|
16
17
|
import * as i10 from "../components/filter/quality-matrix-filter.component";
|
|
17
18
|
export class TreeSearchCountsComponent {
|
|
18
|
-
constructor(metaApi, env, changeDetector) {
|
|
19
|
+
constructor(metaApi, env, changeDetector, document) {
|
|
19
20
|
this.metaApi = metaApi;
|
|
20
21
|
this.env = env;
|
|
21
22
|
this.changeDetector = changeDetector;
|
|
23
|
+
this.document = document;
|
|
22
24
|
/**
|
|
23
25
|
* default max count of columns that should be shown
|
|
24
26
|
*/
|
|
@@ -48,7 +50,10 @@ export class TreeSearchCountsComponent {
|
|
|
48
50
|
if (this.worker) {
|
|
49
51
|
this.worker.terminate();
|
|
50
52
|
}
|
|
51
|
-
|
|
53
|
+
// applications that use us, may provide a relative URL for apiPath.
|
|
54
|
+
// because the Worker runs under a ::blob url, we have to rebase the URL to the search API to the Document, we run in.
|
|
55
|
+
const workerUrl = new URL(this.env.apiPath, this.document.baseURI).toString();
|
|
56
|
+
this.worker = new InlineWorker({ basePath: workerUrl });
|
|
52
57
|
this.worker.onmessage().subscribe((event) => {
|
|
53
58
|
// one message per row
|
|
54
59
|
let total = 0;
|
|
@@ -186,13 +191,16 @@ export class TreeSearchCountsComponent {
|
|
|
186
191
|
get alternatingDataColumns() {
|
|
187
192
|
return this.collectionColumns.flatMap((e, i) => [e, this.searchColumns[i]]);
|
|
188
193
|
}
|
|
189
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TreeSearchCountsComponent, deps: [{ token: i1.MetaApiService }, { token: i2.ConfigHelperService }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
194
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TreeSearchCountsComponent, deps: [{ token: i1.MetaApiService }, { token: i2.ConfigHelperService }, { token: i0.ChangeDetectorRef }, { token: DOCUMENT }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
190
195
|
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: TreeSearchCountsComponent, selector: "metaqs2-tree-search-counts", inputs: { pageTitle: "pageTitle" }, ngImport: i0, template: "<mat-card appearance=\"raised\">\n<mat-card-header *ngIf=\"pageTitle\">\n <mat-card-title>\n Material in Sammlungen{{isLoading() ? \": Lade neue Daten.\" : \"\"}}</mat-card-title\n >\n</mat-card-header>\n<mat-card-content>\n<!-- consider to put the filter in the table header to avoid that it is scrolled out of view-->\n <metaqs2-qm-filter\n aria-disabled=\"true\"\n [label]=\"'Sammlung'\"\n [options]=\"collections$ | async\"\n [multiple]=\"false\"\n [inputFormControl]=\"selectedCollection\"\n >\n </metaqs2-qm-filter>\n\n</mat-card-content>\n</mat-card>\n<mat-card>\n <table [ngClass]=\"{'while-loading': isLoading()}\" mat-table [dataSource]=\"materialCounts.rows\" *ngIf=\"columns\" class=\"quality-matrix\">\n <!-- Define columns of table -->\n <!-- Label Column -->\n <ng-container matColumnDef=\"label-col\" sticky>\n <th rowspan=\"2\" mat-header-cell *matHeaderCellDef>Sammlung\n <div >\n <mat-slide-toggle labelPosition=\"before\" class=\"toggle\" [formControl]=\"showOERonly\" >\n zeige nur OER\n </mat-slide-toggle>\n </div>\n <div>\n <mat-slide-toggle labelPosition=\"before\" class=\"toggle\" [formControl]=\"showAllColumns\" >\n zeige alle Typen\n </mat-slide-toggle>\n </div>\n </th>\n <td\n mat-cell\n *matCellDef=\"let row\"\n matTooltip=\"{{row.meta.alt_label}}\"\n class=\"label-col {{'mat-cell-level-' + (row.meta.level+1)}}\"\n >\n <a (click)=\"openCollection(row.meta.id)\">\n <mat-icon [svgIcon]=\"row.meta.level<=1?'layers':'child'\" />\n {{ row.meta.label }}\n </a>\n </td>\n </ng-container>\n <!-- one column for the MaterialType spanning the collection and search columns-->\n <ng-container *ngFor=\"let col of filteredColumns; trackBy:columnIdent\" [matColumnDef]=\"col.id + '_type'\">\n <th [attr.colspan]=\"2\" mat-header-cell *matHeaderCellDef class='mat-cell-level-\"{{col.level}}\"' matTooltip=\"{{col.label}}\" >{{col.label}}</th>\n </ng-container>\n <!-- /source -->\n <!-- collection Data Columns -->\n <ng-container *ngFor=\"let column of filteredColumns; trackBy:columnIdent\" [matColumnDef]=\"column.id + COLLECTION_POSTFIX\">\n <th mat-header-cell *matHeaderCellDef matTooltip=\"{{column.label}}\" >in Sammlung</th>\n <td [class.noMaterial]=\"!row.counts[column.id]?.sufficient || row.counts[column.id]?.sufficient <= 1\"\n [class.fewMaterials]=\"1 < row.counts[column.id]?.sufficient && row.counts[column.id]?.sufficient <= 3\"\n class=\"collection-data-cell\"\n mat-cell *matCellDef=\"let row\">\n <a (click)=\"showCollectionItems(row.meta.id, row.meta.label, column.id)\">\n {{ row.counts[column.id]?.sufficient || '0' }}\n </a>\n </td>\n </ng-container>\n <!-- /collection Data Columns -->\n <!-- search Data Columns -->\n <ng-container *ngFor=\"let column of filteredColumns let index = index; trackBy:columnIdent\" [matColumnDef]=\"column.id + SEARCH_POSTFIX\">\n <th mat-header-cell *matHeaderCellDef>Suche</th>\n <td class=\"search-data-cell\" mat-cell *matCellDef=\"let row\">\n <a (click)=\"searchInEditor(column.id, row.meta.label)\">\n <ng-container *ngIf=\"(( searchCounts | async)?.has(column.id + '_' + row.meta.id) ) else zeroOrLoading\">\n {{ (searchCounts | async)?.get(column.id + '_' + row.meta.id) }}\n </ng-container>\n <ng-template #zeroOrLoading>\n <ng-container *ngIf=\"rowsLoaded.has(row.meta.id) else loadingBlock\"> 0 </ng-container>\n </ng-template>\n <ng-template #loadingBlock>\n loading\u2026\n </ng-template>\n </a>\n </td>\n </ng-container>\n <!-- /search Data Columns -->\n <!-- Generate actual table -->\n <tr mat-header-row *matHeaderRowDef=\"['label-col'].concat( typeColumns ); sticky: true;\"></tr>\n <tr mat-header-row *matHeaderRowDef=\"alternatingDataColumns; sticky: true;\"></tr>\n <tr mat-row *matRowDef=\"let row; columns: ['label-col'].concat(alternatingDataColumns)\"></tr>\n </table>\n</mat-card>", styles: ["tr:nth-child(2n){background-color:#e4e4e4}tr:nth-child(2n)>.mat-column-label-col,tr:nth-child(2n) .search-data-cell{border-right-color:#fff}.search-data-cell,.mat-mdc-header-cell,.mat-column-label-col{border-right:1px solid #e4e4e4}.search-data-cell mat-icon,.mat-mdc-header-cell mat-icon,.mat-column-label-col mat-icon{color:var(--mat-table-row-item-label-text-color);height:16px;padding-right:5px;font-size:16px}.mat-mdc-header-cell,.mat-mdc-cell{padding-left:5px;text-align:center}.mat-mdc-header-cell a,.mat-mdc-cell a{cursor:pointer;text-decoration:underline}.mat-mdc-header-cell a:hover,.mat-mdc-cell a:hover{text-decoration:underline}.label-col a{text-decoration:none;align-items:center;display:flex}.mat-column-totals_collection.noMaterial{background-color:#fad6da}.mat-column-totals_collection.fewMaterials{background-color:#fff1d6}.mat-cell-level-1{padding-left:20px!important;text-align:left}.mat-cell-level-2{padding-left:40px!important;text-align:left}.mat-cell-level-3{padding-left:60px!important;text-align:left}.mat-cell-level-4{padding-left:80px!important;text-align:left}.mat-cell-level-5{padding-left:100px!important;text-align:left}.mat-cell-level-6{padding-left:120px!important;text-align:left}.mat-cell-level-7{padding-left:140px!important;text-align:left}.mat-cell-level-8{padding-left:160px!important;text-align:left}\n"], dependencies: [{ kind: "directive", type: i3.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i4.MatCard, selector: "mat-card", inputs: ["appearance"], exportAs: ["matCard"] }, { kind: "directive", type: i4.MatCardContent, selector: "mat-card-content" }, { kind: "component", type: i4.MatCardHeader, selector: "mat-card-header" }, { kind: "directive", type: i4.MatCardTitle, selector: "mat-card-title, [mat-card-title], [matCardTitle]" }, { kind: "component", type: i5.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: i6.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: "component", type: i7.MatTable, selector: "mat-table, table[mat-table]", exportAs: ["matTable"] }, { kind: "directive", type: i7.MatHeaderCellDef, selector: "[matHeaderCellDef]" }, { kind: "directive", type: i7.MatHeaderRowDef, selector: "[matHeaderRowDef]", inputs: ["matHeaderRowDef", "matHeaderRowDefSticky"] }, { kind: "directive", type: i7.MatColumnDef, selector: "[matColumnDef]", inputs: ["matColumnDef"] }, { kind: "directive", type: i7.MatCellDef, selector: "[matCellDef]" }, { kind: "directive", type: i7.MatRowDef, selector: "[matRowDef]", inputs: ["matRowDefColumns", "matRowDefWhen"] }, { kind: "directive", type: i7.MatHeaderCell, selector: "mat-header-cell, th[mat-header-cell]" }, { kind: "directive", type: i7.MatCell, selector: "mat-cell, td[mat-cell]" }, { kind: "component", type: i7.MatHeaderRow, selector: "mat-header-row, tr[mat-header-row]", exportAs: ["matHeaderRow"] }, { kind: "component", type: i7.MatRow, selector: "mat-row, tr[mat-row]", exportAs: ["matRow"] }, { kind: "directive", type: i8.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i8.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i9.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "component", type: i10.QualityMatrixFilterComponent, selector: "metaqs2-qm-filter", inputs: ["options", "inputFormControl", "label", "multiple"], outputs: ["changedFilters"] }, { kind: "pipe", type: i3.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
191
196
|
}
|
|
192
197
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TreeSearchCountsComponent, decorators: [{
|
|
193
198
|
type: Component,
|
|
194
199
|
args: [{ selector: 'metaqs2-tree-search-counts', changeDetection: ChangeDetectionStrategy.OnPush, template: "<mat-card appearance=\"raised\">\n<mat-card-header *ngIf=\"pageTitle\">\n <mat-card-title>\n Material in Sammlungen{{isLoading() ? \": Lade neue Daten.\" : \"\"}}</mat-card-title\n >\n</mat-card-header>\n<mat-card-content>\n<!-- consider to put the filter in the table header to avoid that it is scrolled out of view-->\n <metaqs2-qm-filter\n aria-disabled=\"true\"\n [label]=\"'Sammlung'\"\n [options]=\"collections$ | async\"\n [multiple]=\"false\"\n [inputFormControl]=\"selectedCollection\"\n >\n </metaqs2-qm-filter>\n\n</mat-card-content>\n</mat-card>\n<mat-card>\n <table [ngClass]=\"{'while-loading': isLoading()}\" mat-table [dataSource]=\"materialCounts.rows\" *ngIf=\"columns\" class=\"quality-matrix\">\n <!-- Define columns of table -->\n <!-- Label Column -->\n <ng-container matColumnDef=\"label-col\" sticky>\n <th rowspan=\"2\" mat-header-cell *matHeaderCellDef>Sammlung\n <div >\n <mat-slide-toggle labelPosition=\"before\" class=\"toggle\" [formControl]=\"showOERonly\" >\n zeige nur OER\n </mat-slide-toggle>\n </div>\n <div>\n <mat-slide-toggle labelPosition=\"before\" class=\"toggle\" [formControl]=\"showAllColumns\" >\n zeige alle Typen\n </mat-slide-toggle>\n </div>\n </th>\n <td\n mat-cell\n *matCellDef=\"let row\"\n matTooltip=\"{{row.meta.alt_label}}\"\n class=\"label-col {{'mat-cell-level-' + (row.meta.level+1)}}\"\n >\n <a (click)=\"openCollection(row.meta.id)\">\n <mat-icon [svgIcon]=\"row.meta.level<=1?'layers':'child'\" />\n {{ row.meta.label }}\n </a>\n </td>\n </ng-container>\n <!-- one column for the MaterialType spanning the collection and search columns-->\n <ng-container *ngFor=\"let col of filteredColumns; trackBy:columnIdent\" [matColumnDef]=\"col.id + '_type'\">\n <th [attr.colspan]=\"2\" mat-header-cell *matHeaderCellDef class='mat-cell-level-\"{{col.level}}\"' matTooltip=\"{{col.label}}\" >{{col.label}}</th>\n </ng-container>\n <!-- /source -->\n <!-- collection Data Columns -->\n <ng-container *ngFor=\"let column of filteredColumns; trackBy:columnIdent\" [matColumnDef]=\"column.id + COLLECTION_POSTFIX\">\n <th mat-header-cell *matHeaderCellDef matTooltip=\"{{column.label}}\" >in Sammlung</th>\n <td [class.noMaterial]=\"!row.counts[column.id]?.sufficient || row.counts[column.id]?.sufficient <= 1\"\n [class.fewMaterials]=\"1 < row.counts[column.id]?.sufficient && row.counts[column.id]?.sufficient <= 3\"\n class=\"collection-data-cell\"\n mat-cell *matCellDef=\"let row\">\n <a (click)=\"showCollectionItems(row.meta.id, row.meta.label, column.id)\">\n {{ row.counts[column.id]?.sufficient || '0' }}\n </a>\n </td>\n </ng-container>\n <!-- /collection Data Columns -->\n <!-- search Data Columns -->\n <ng-container *ngFor=\"let column of filteredColumns let index = index; trackBy:columnIdent\" [matColumnDef]=\"column.id + SEARCH_POSTFIX\">\n <th mat-header-cell *matHeaderCellDef>Suche</th>\n <td class=\"search-data-cell\" mat-cell *matCellDef=\"let row\">\n <a (click)=\"searchInEditor(column.id, row.meta.label)\">\n <ng-container *ngIf=\"(( searchCounts | async)?.has(column.id + '_' + row.meta.id) ) else zeroOrLoading\">\n {{ (searchCounts | async)?.get(column.id + '_' + row.meta.id) }}\n </ng-container>\n <ng-template #zeroOrLoading>\n <ng-container *ngIf=\"rowsLoaded.has(row.meta.id) else loadingBlock\"> 0 </ng-container>\n </ng-template>\n <ng-template #loadingBlock>\n loading\u2026\n </ng-template>\n </a>\n </td>\n </ng-container>\n <!-- /search Data Columns -->\n <!-- Generate actual table -->\n <tr mat-header-row *matHeaderRowDef=\"['label-col'].concat( typeColumns ); sticky: true;\"></tr>\n <tr mat-header-row *matHeaderRowDef=\"alternatingDataColumns; sticky: true;\"></tr>\n <tr mat-row *matRowDef=\"let row; columns: ['label-col'].concat(alternatingDataColumns)\"></tr>\n </table>\n</mat-card>", styles: ["tr:nth-child(2n){background-color:#e4e4e4}tr:nth-child(2n)>.mat-column-label-col,tr:nth-child(2n) .search-data-cell{border-right-color:#fff}.search-data-cell,.mat-mdc-header-cell,.mat-column-label-col{border-right:1px solid #e4e4e4}.search-data-cell mat-icon,.mat-mdc-header-cell mat-icon,.mat-column-label-col mat-icon{color:var(--mat-table-row-item-label-text-color);height:16px;padding-right:5px;font-size:16px}.mat-mdc-header-cell,.mat-mdc-cell{padding-left:5px;text-align:center}.mat-mdc-header-cell a,.mat-mdc-cell a{cursor:pointer;text-decoration:underline}.mat-mdc-header-cell a:hover,.mat-mdc-cell a:hover{text-decoration:underline}.label-col a{text-decoration:none;align-items:center;display:flex}.mat-column-totals_collection.noMaterial{background-color:#fad6da}.mat-column-totals_collection.fewMaterials{background-color:#fff1d6}.mat-cell-level-1{padding-left:20px!important;text-align:left}.mat-cell-level-2{padding-left:40px!important;text-align:left}.mat-cell-level-3{padding-left:60px!important;text-align:left}.mat-cell-level-4{padding-left:80px!important;text-align:left}.mat-cell-level-5{padding-left:100px!important;text-align:left}.mat-cell-level-6{padding-left:120px!important;text-align:left}.mat-cell-level-7{padding-left:140px!important;text-align:left}.mat-cell-level-8{padding-left:160px!important;text-align:left}\n"] }]
|
|
195
|
-
}], ctorParameters: () => [{ type: i1.MetaApiService }, { type: i2.ConfigHelperService }, { type: i0.ChangeDetectorRef }
|
|
200
|
+
}], ctorParameters: () => [{ type: i1.MetaApiService }, { type: i2.ConfigHelperService }, { type: i0.ChangeDetectorRef }, { type: Document, decorators: [{
|
|
201
|
+
type: Inject,
|
|
202
|
+
args: [DOCUMENT]
|
|
203
|
+
}] }], propDecorators: { pageTitle: [{
|
|
196
204
|
type: Input
|
|
197
205
|
}] } });
|
|
198
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tree-search-counts.component.js","sourceRoot":"","sources":["../../../../../projects/ng-meta-widgets-lib/src/lib/tree-search-counts/tree-search-counts.component.ts","../../../../../projects/ng-meta-widgets-lib/src/lib/tree-search-counts/tree-search-counts.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, ChangeDetectorRef, Component, Input, OnDestroy, OnInit, signal } from '@angular/core';\nimport { BehaviorSubject } from 'rxjs';\nimport { take } from 'rxjs/operators';\nimport { MetaApiService } from '../meta-api.service';\nimport { ConfigHelperService } from '../config-helper.service';\nimport { FilterValue, QualityMatrix } from '../java-api';\nimport { FormControl } from '@angular/forms';\nimport { InlineWorker } from './inline-worker';\n\nexport interface DataColumn {\n  id: string;\n  icon: string;\n  label: string;\n  level?: number;\n}\n\n@Component({\n  selector: 'metaqs2-tree-search-counts',\n  templateUrl: './tree-search-counts.component.html',\n  styleUrls: ['./tree-search-counts.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class TreeSearchCountsComponent implements OnInit, OnDestroy {\n  /**\n   * default max count of columns that should be shown\n   */\n  readonly DEFAULT_COLUMN_LIMIT = 6;\n  readonly COLLECTION_POSTFIX = '_collection';\n  readonly SEARCH_POSTFIX = '_search';\n  readonly collections$: BehaviorSubject<FilterValue[]> = new BehaviorSubject<FilterValue[]>([]);\n  readonly searchCounts = new BehaviorSubject<Map<string, number>>(new Map<string, number>());\n  readonly showOERonly = new FormControl({ value: true, disabled: false }, { nonNullable: true });\n  readonly showAllColumns = new FormControl({ value: false, disabled: false }, { nonNullable: true });\n  readonly selectedCollection = new FormControl<string>('', { nonNullable: true });\n\n  // convert this in an Obeservable\n  protected materialCounts: QualityMatrix;\n  public columns: DataColumn[];\n  protected rowsLoaded: Map<string, boolean> = new Map<string, boolean>();\n\n  isLoading = signal(true);\n  private worker: InlineWorker;\n\n  @Input()\n  protected pageTitle: string;\n\n  constructor(\n    private readonly metaApi: MetaApiService,\n    private env: ConfigHelperService,\n    private readonly changeDetector: ChangeDetectorRef\n  ) {}\n\n  ngOnInit(): void {\n    this.initCollectionsFilter();\n    this.initOerFilter();\n  }\n\n  ngOnDestroy(): void {\n    if (this.worker) {\n      this.worker.terminate();\n    }\n  }\n\n  private letTheWorkerSearch(): void {\n    this.searchCounts.next(new Map<string, number>());\n    this.rowsLoaded.clear();\n    if (this.worker) {\n      this.worker.terminate();\n    }\n    this.worker = new InlineWorker({ basePath: this.env.apiPath });\n    this.worker.onmessage().subscribe((event) => {\n      // one message per row\n      let total = 0;\n\n      Object.entries(event.data.data).forEach(([key, value]) => {\n        // key is the material type, value is the count\n        const columnKey = key + '_' + event.data.collectionId;\n        total += value as number;\n        this.searchCounts.value.set(columnKey, value as number);\n      });\n      this.searchCounts.value.set('totals_' + event.data.collectionId, total);\n      this.searchCounts.next(this.searchCounts.value);\n      this.rowsLoaded.set(event.data.collectionId, true);\n    });\n\n    this.worker.onerror().subscribe((data) => {\n      console.error('in main thread', data);\n    });\n\n    this.materialCounts.rows.forEach((row) => {\n      this.worker.postMessage({\n        collectionId: row.meta.id,\n        searchTerm: row.meta.label,\n        oerOnly: this.showOERonly.value,\n      });\n    });\n  }\n\n  private initOerFilter(): void {\n    this.showOERonly.valueChanges.subscribe(() => {\n      const collectionId = this.extractCollectionIdFromVocabUrl(this.selectedCollection.value);\n      this.loadDataForCollection(collectionId);\n    });\n  }\n\n  private initCollectionsFilter(): void {\n    this.selectedCollection.valueChanges.subscribe((collectionIdValue) => {\n      const id = this.extractCollectionIdFromVocabUrl(collectionIdValue);\n      this.loadDataForCollection(id);\n    });\n    this.metaApi\n      .getCollectionsFilter()\n      .pipe(take(1))\n      .subscribe((collections) => {\n        const sortedCollections = collections.sort((a, b) => a.label.localeCompare(b.label));\n        this.collections$.next(sortedCollections);\n        this.selectedCollection.setValue(sortedCollections[0].id);\n      });\n  }\n\n  private async loadDataForCollection(collectionId: string): Promise<void> {\n    this.isLoading.set(true);\n    this.materialCounts = await this.metaApi\n      .getMaterialCountMatrixPerCollection(collectionId, this.showOERonly.value)\n      .toPromise();\n    //TODO: try to avoid to recalculate the columns on every change\n    this.columns = this.materialCounts.columns.map((c) => {\n      return {\n        id: c.id,\n        icon: 'total',\n        label: c.label,\n        level: c.level,\n      };\n    });\n\n    this.letTheWorkerSearch();\n\n    this.isLoading.set(false);\n    this.changeDetector.detectChanges();\n  }\n\n  private extractCollectionIdFromVocabUrl(url: string): string {\n    return url.substring(url.lastIndexOf('/') + 1);\n  }\n\n  openCollection(id: string) {\n    const collectionId = this.extractCollectionIdFromVocabUrl(id);\n    window.open(\n      this.env.eduSharingPath +\n        '/components/editorial-desk/?mode=render&viewType=single&ids=' +\n        encodeURIComponent(collectionId)\n    );\n  }\n\n  searchInEditor(materialType: string, searchTerm: string) {\n    const filters: any = {};\n    if (materialType !== 'totals') {\n      filters['ccm:oeh_lrt_aggregated'] = [materialType];\n    }\n    if (this.showOERonly.value) {\n      filters['virtual:editorial_license'] = ['oer'];\n    }\n    const url =\n      this.env.eduSharingPath +\n      '/components/editorial-desk?mode=audit_metaqs&' +\n      'q=' +\n      encodeURIComponent(searchTerm) +\n      '&' +\n      'title=' +\n      encodeURIComponent('MetaQS - Suchergebnisse für \"' + searchTerm + '\"') +\n      '&' +\n      'filters=' +\n      encodeURIComponent(JSON.stringify(filters)) +\n      '&' +\n      'panel=contents&tab=4&';\n\n    const ref = window.open(url);\n  }\n\n  showCollectionItems(collectionId: string, collectionName: string, materialType: string) {\n    const id = this.extractCollectionIdFromVocabUrl(collectionId);\n    const filters: any = {\n      //\"virtual:collection_id_primary\": [id], //works often, but not always\n      'virtual:metaqs_collection_id': [id],\n    };\n    if (materialType !== 'totals') {\n      filters['ccm:oeh_lrt_aggregated'] = [materialType];\n    }\n    if (this.showOERonly.value) {\n      filters['virtual:editorial_license'] = ['oer'];\n    }\n    window.open(\n      this.env.eduSharingPath +\n        '/components/editorial-desk?mode=audit_metaqs&' +\n        'title=' +\n        encodeURIComponent('MetaQS - Sammlungsinhalte für \"' + collectionName + '\"') +\n        '&' +\n        'filters=' +\n        encodeURIComponent(JSON.stringify(filters)) +\n        '&' +\n        'panel=contents&tab=4&'\n    );\n  }\n\n  columnIdent(_index: number, col: { id: string }) {\n    return col.id;\n  }\n\n  get filteredColumns() {\n    // return either all or just a given amount of columns\n    return this.showAllColumns.value ? this.columns : this.columns.slice(0, this.DEFAULT_COLUMN_LIMIT);\n  }\n\n  get columnNames() {\n    return this.filteredColumns.map((c) => c.id);\n  }\n\n  get typeColumns(): string[] {\n    return this.columnNames.map((c) => c + '_type');\n  }\n\n  get collectionColumns(): string[] {\n    return this.columnNames.map((c) => c + this.COLLECTION_POSTFIX);\n  }\n\n  get searchColumns(): string[] {\n    return this.columnNames.map((c) => c + this.SEARCH_POSTFIX);\n  }\n\n  get alternatingDataColumns(): string[] {\n    return this.collectionColumns.flatMap((e, i) => [e, this.searchColumns[i]]);\n  }\n}\n","<mat-card appearance=\"raised\">\n<mat-card-header *ngIf=\"pageTitle\">\n  <mat-card-title>\n    Material in Sammlungen{{isLoading() ? \": Lade neue Daten.\" : \"\"}}</mat-card-title\n  >\n</mat-card-header>\n<mat-card-content>\n<!-- consider to put the filter in the table header to avoid that it is scrolled out of view-->\n    <metaqs2-qm-filter\n      aria-disabled=\"true\"\n      [label]=\"'Sammlung'\"\n      [options]=\"collections$ | async\"\n      [multiple]=\"false\"\n      [inputFormControl]=\"selectedCollection\"\n    >\n    </metaqs2-qm-filter>\n\n</mat-card-content>\n</mat-card>\n<mat-card>\n  <table [ngClass]=\"{'while-loading': isLoading()}\" mat-table [dataSource]=\"materialCounts.rows\" *ngIf=\"columns\" class=\"quality-matrix\">\n    <!-- Define columns of table -->\n    <!-- Label Column -->\n    <ng-container matColumnDef=\"label-col\" sticky>\n      <th rowspan=\"2\" mat-header-cell *matHeaderCellDef>Sammlung\n      <div >\n        <mat-slide-toggle labelPosition=\"before\" class=\"toggle\" [formControl]=\"showOERonly\" >\n          zeige nur OER\n        </mat-slide-toggle>\n      </div>\n        <div>\n        <mat-slide-toggle labelPosition=\"before\" class=\"toggle\" [formControl]=\"showAllColumns\" >\n          zeige alle Typen\n        </mat-slide-toggle>\n        </div>\n      </th>\n      <td\n        mat-cell\n        *matCellDef=\"let row\"\n        matTooltip=\"{{row.meta.alt_label}}\"\n        class=\"label-col {{'mat-cell-level-' + (row.meta.level+1)}}\"\n      >\n        <a (click)=\"openCollection(row.meta.id)\">\n          <mat-icon [svgIcon]=\"row.meta.level<=1?'layers':'child'\" />\n          {{ row.meta.label }}\n        </a>\n      </td>\n    </ng-container>\n    <!-- one column for the MaterialType spanning the collection and search columns-->\n    <ng-container *ngFor=\"let col of filteredColumns; trackBy:columnIdent\" [matColumnDef]=\"col.id + '_type'\">\n      <th [attr.colspan]=\"2\" mat-header-cell *matHeaderCellDef class='mat-cell-level-\"{{col.level}}\"' matTooltip=\"{{col.label}}\" >{{col.label}}</th>\n    </ng-container>\n    <!-- /source -->\n    <!-- collection Data Columns -->\n    <ng-container *ngFor=\"let column of filteredColumns; trackBy:columnIdent\" [matColumnDef]=\"column.id + COLLECTION_POSTFIX\">\n      <th mat-header-cell *matHeaderCellDef matTooltip=\"{{column.label}}\" >in Sammlung</th>\n      <td [class.noMaterial]=\"!row.counts[column.id]?.sufficient || row.counts[column.id]?.sufficient <= 1\"\n          [class.fewMaterials]=\"1 < row.counts[column.id]?.sufficient && row.counts[column.id]?.sufficient <= 3\"\n          class=\"collection-data-cell\"\n          mat-cell *matCellDef=\"let row\">\n            <a (click)=\"showCollectionItems(row.meta.id, row.meta.label, column.id)\">\n              {{ row.counts[column.id]?.sufficient || '0' }}\n            </a>\n      </td>\n    </ng-container>\n    <!-- /collection Data Columns -->\n    <!-- search Data Columns -->\n    <ng-container *ngFor=\"let column of filteredColumns let index = index; trackBy:columnIdent\" [matColumnDef]=\"column.id + SEARCH_POSTFIX\">\n      <th mat-header-cell *matHeaderCellDef>Suche</th>\n      <td class=\"search-data-cell\" mat-cell *matCellDef=\"let row\">\n        <a (click)=\"searchInEditor(column.id, row.meta.label)\">\n          <ng-container *ngIf=\"(( searchCounts | async)?.has(column.id + '_' + row.meta.id) ) else zeroOrLoading\">\n            {{ (searchCounts | async)?.get(column.id + '_' + row.meta.id) }}\n          </ng-container>\n          <ng-template #zeroOrLoading>\n            <ng-container *ngIf=\"rowsLoaded.has(row.meta.id) else loadingBlock\"> 0 </ng-container>\n          </ng-template>\n          <ng-template #loadingBlock>\n            loading…\n          </ng-template>\n        </a>\n      </td>\n    </ng-container>\n    <!-- /search Data Columns -->\n      <!-- Generate actual table -->\n    <tr mat-header-row *matHeaderRowDef=\"['label-col'].concat( typeColumns ); sticky: true;\"></tr>\n    <tr mat-header-row *matHeaderRowDef=\"alternatingDataColumns; sticky: true;\"></tr>\n    <tr mat-row *matRowDef=\"let row; columns: ['label-col'].concat(alternatingDataColumns)\"></tr>\n  </table>\n</mat-card>"]}
|
|
206
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tree-search-counts.component.js","sourceRoot":"","sources":["../../../../../projects/ng-meta-widgets-lib/src/lib/tree-search-counts/tree-search-counts.component.ts","../../../../../projects/ng-meta-widgets-lib/src/lib/tree-search-counts/tree-search-counts.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  Inject,\n  Input,\n  OnDestroy,\n  OnInit,\n  signal,\n} from '@angular/core';\nimport { BehaviorSubject } from 'rxjs';\nimport { take } from 'rxjs/operators';\nimport { MetaApiService } from '../meta-api.service';\nimport { ConfigHelperService } from '../config-helper.service';\nimport { FilterValue, QualityMatrix } from '../java-api';\nimport { FormControl } from '@angular/forms';\nimport { InlineWorker } from './inline-worker';\nimport { DOCUMENT } from '@angular/common';\n\nexport interface DataColumn {\n  id: string;\n  icon: string;\n  label: string;\n  level?: number;\n}\n\n@Component({\n  selector: 'metaqs2-tree-search-counts',\n  templateUrl: './tree-search-counts.component.html',\n  styleUrls: ['./tree-search-counts.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class TreeSearchCountsComponent implements OnInit, OnDestroy {\n  /**\n   * default max count of columns that should be shown\n   */\n  readonly DEFAULT_COLUMN_LIMIT = 6;\n  readonly COLLECTION_POSTFIX = '_collection';\n  readonly SEARCH_POSTFIX = '_search';\n  readonly collections$: BehaviorSubject<FilterValue[]> = new BehaviorSubject<FilterValue[]>([]);\n  readonly searchCounts = new BehaviorSubject<Map<string, number>>(new Map<string, number>());\n  readonly showOERonly = new FormControl({ value: true, disabled: false }, { nonNullable: true });\n  readonly showAllColumns = new FormControl({ value: false, disabled: false }, { nonNullable: true });\n  readonly selectedCollection = new FormControl<string>('', { nonNullable: true });\n\n  // convert this in an Obeservable\n  protected materialCounts: QualityMatrix;\n  public columns: DataColumn[];\n  protected rowsLoaded: Map<string, boolean> = new Map<string, boolean>();\n\n  isLoading = signal(true);\n  private worker: InlineWorker;\n\n  @Input()\n  protected pageTitle: string;\n\n  constructor(\n    private readonly metaApi: MetaApiService,\n    private env: ConfigHelperService,\n    private readonly changeDetector: ChangeDetectorRef,\n    @Inject(DOCUMENT)\n    private readonly document: Document\n  ) {}\n\n  ngOnInit(): void {\n    this.initCollectionsFilter();\n    this.initOerFilter();\n  }\n\n  ngOnDestroy(): void {\n    if (this.worker) {\n      this.worker.terminate();\n    }\n  }\n\n  private letTheWorkerSearch(): void {\n    this.searchCounts.next(new Map<string, number>());\n    this.rowsLoaded.clear();\n    if (this.worker) {\n      this.worker.terminate();\n    }\n    // applications that use us, may provide a relative URL for apiPath.\n    // because the Worker runs under a ::blob url, we have to rebase the URL to the search API to the Document, we run in.\n    const workerUrl = new URL(this.env.apiPath, this.document.baseURI).toString();\n    this.worker = new InlineWorker({ basePath: workerUrl });\n    this.worker.onmessage().subscribe((event) => {\n      // one message per row\n      let total = 0;\n\n      Object.entries(event.data.data).forEach(([key, value]) => {\n        // key is the material type, value is the count\n        const columnKey = key + '_' + event.data.collectionId;\n        total += value as number;\n        this.searchCounts.value.set(columnKey, value as number);\n      });\n      this.searchCounts.value.set('totals_' + event.data.collectionId, total);\n      this.searchCounts.next(this.searchCounts.value);\n      this.rowsLoaded.set(event.data.collectionId, true);\n    });\n\n    this.worker.onerror().subscribe((data) => {\n      console.error('in main thread', data);\n    });\n\n    this.materialCounts.rows.forEach((row) => {\n      this.worker.postMessage({\n        collectionId: row.meta.id,\n        searchTerm: row.meta.label,\n        oerOnly: this.showOERonly.value,\n      });\n    });\n  }\n\n  private initOerFilter(): void {\n    this.showOERonly.valueChanges.subscribe(() => {\n      const collectionId = this.extractCollectionIdFromVocabUrl(this.selectedCollection.value);\n      this.loadDataForCollection(collectionId);\n    });\n  }\n\n  private initCollectionsFilter(): void {\n    this.selectedCollection.valueChanges.subscribe((collectionIdValue) => {\n      const id = this.extractCollectionIdFromVocabUrl(collectionIdValue);\n      this.loadDataForCollection(id);\n    });\n    this.metaApi\n      .getCollectionsFilter()\n      .pipe(take(1))\n      .subscribe((collections) => {\n        const sortedCollections = collections.sort((a, b) => a.label.localeCompare(b.label));\n        this.collections$.next(sortedCollections);\n        this.selectedCollection.setValue(sortedCollections[0].id);\n      });\n  }\n\n  private async loadDataForCollection(collectionId: string): Promise<void> {\n    this.isLoading.set(true);\n    this.materialCounts = await this.metaApi\n      .getMaterialCountMatrixPerCollection(collectionId, this.showOERonly.value)\n      .toPromise();\n    //TODO: try to avoid to recalculate the columns on every change\n    this.columns = this.materialCounts.columns.map((c) => {\n      return {\n        id: c.id,\n        icon: 'total',\n        label: c.label,\n        level: c.level,\n      };\n    });\n\n    this.letTheWorkerSearch();\n\n    this.isLoading.set(false);\n    this.changeDetector.detectChanges();\n  }\n\n  private extractCollectionIdFromVocabUrl(url: string): string {\n    return url.substring(url.lastIndexOf('/') + 1);\n  }\n\n  openCollection(id: string) {\n    const collectionId = this.extractCollectionIdFromVocabUrl(id);\n    window.open(\n      this.env.eduSharingPath +\n        '/components/editorial-desk/?mode=render&viewType=single&ids=' +\n        encodeURIComponent(collectionId)\n    );\n  }\n\n  searchInEditor(materialType: string, searchTerm: string) {\n    const filters: any = {};\n    if (materialType !== 'totals') {\n      filters['ccm:oeh_lrt_aggregated'] = [materialType];\n    }\n    if (this.showOERonly.value) {\n      filters['virtual:editorial_license'] = ['oer'];\n    }\n    const url =\n      this.env.eduSharingPath +\n      '/components/editorial-desk?mode=audit_metaqs&' +\n      'q=' +\n      encodeURIComponent(searchTerm) +\n      '&' +\n      'title=' +\n      encodeURIComponent('MetaQS - Suchergebnisse für \"' + searchTerm + '\"') +\n      '&' +\n      'filters=' +\n      encodeURIComponent(JSON.stringify(filters)) +\n      '&' +\n      'panel=contents&tab=4&';\n\n    const ref = window.open(url);\n  }\n\n  showCollectionItems(collectionId: string, collectionName: string, materialType: string) {\n    const id = this.extractCollectionIdFromVocabUrl(collectionId);\n    const filters: any = {\n      //\"virtual:collection_id_primary\": [id], //works often, but not always\n      'virtual:metaqs_collection_id': [id],\n    };\n    if (materialType !== 'totals') {\n      filters['ccm:oeh_lrt_aggregated'] = [materialType];\n    }\n    if (this.showOERonly.value) {\n      filters['virtual:editorial_license'] = ['oer'];\n    }\n    window.open(\n      this.env.eduSharingPath +\n        '/components/editorial-desk?mode=audit_metaqs&' +\n        'title=' +\n        encodeURIComponent('MetaQS - Sammlungsinhalte für \"' + collectionName + '\"') +\n        '&' +\n        'filters=' +\n        encodeURIComponent(JSON.stringify(filters)) +\n        '&' +\n        'panel=contents&tab=4&'\n    );\n  }\n\n  columnIdent(_index: number, col: { id: string }) {\n    return col.id;\n  }\n\n  get filteredColumns() {\n    // return either all or just a given amount of columns\n    return this.showAllColumns.value ? this.columns : this.columns.slice(0, this.DEFAULT_COLUMN_LIMIT);\n  }\n\n  get columnNames() {\n    return this.filteredColumns.map((c) => c.id);\n  }\n\n  get typeColumns(): string[] {\n    return this.columnNames.map((c) => c + '_type');\n  }\n\n  get collectionColumns(): string[] {\n    return this.columnNames.map((c) => c + this.COLLECTION_POSTFIX);\n  }\n\n  get searchColumns(): string[] {\n    return this.columnNames.map((c) => c + this.SEARCH_POSTFIX);\n  }\n\n  get alternatingDataColumns(): string[] {\n    return this.collectionColumns.flatMap((e, i) => [e, this.searchColumns[i]]);\n  }\n}\n","<mat-card appearance=\"raised\">\n<mat-card-header *ngIf=\"pageTitle\">\n  <mat-card-title>\n    Material in Sammlungen{{isLoading() ? \": Lade neue Daten.\" : \"\"}}</mat-card-title\n  >\n</mat-card-header>\n<mat-card-content>\n<!-- consider to put the filter in the table header to avoid that it is scrolled out of view-->\n    <metaqs2-qm-filter\n      aria-disabled=\"true\"\n      [label]=\"'Sammlung'\"\n      [options]=\"collections$ | async\"\n      [multiple]=\"false\"\n      [inputFormControl]=\"selectedCollection\"\n    >\n    </metaqs2-qm-filter>\n\n</mat-card-content>\n</mat-card>\n<mat-card>\n  <table [ngClass]=\"{'while-loading': isLoading()}\" mat-table [dataSource]=\"materialCounts.rows\" *ngIf=\"columns\" class=\"quality-matrix\">\n    <!-- Define columns of table -->\n    <!-- Label Column -->\n    <ng-container matColumnDef=\"label-col\" sticky>\n      <th rowspan=\"2\" mat-header-cell *matHeaderCellDef>Sammlung\n      <div >\n        <mat-slide-toggle labelPosition=\"before\" class=\"toggle\" [formControl]=\"showOERonly\" >\n          zeige nur OER\n        </mat-slide-toggle>\n      </div>\n        <div>\n        <mat-slide-toggle labelPosition=\"before\" class=\"toggle\" [formControl]=\"showAllColumns\" >\n          zeige alle Typen\n        </mat-slide-toggle>\n        </div>\n      </th>\n      <td\n        mat-cell\n        *matCellDef=\"let row\"\n        matTooltip=\"{{row.meta.alt_label}}\"\n        class=\"label-col {{'mat-cell-level-' + (row.meta.level+1)}}\"\n      >\n        <a (click)=\"openCollection(row.meta.id)\">\n          <mat-icon [svgIcon]=\"row.meta.level<=1?'layers':'child'\" />\n          {{ row.meta.label }}\n        </a>\n      </td>\n    </ng-container>\n    <!-- one column for the MaterialType spanning the collection and search columns-->\n    <ng-container *ngFor=\"let col of filteredColumns; trackBy:columnIdent\" [matColumnDef]=\"col.id + '_type'\">\n      <th [attr.colspan]=\"2\" mat-header-cell *matHeaderCellDef class='mat-cell-level-\"{{col.level}}\"' matTooltip=\"{{col.label}}\" >{{col.label}}</th>\n    </ng-container>\n    <!-- /source -->\n    <!-- collection Data Columns -->\n    <ng-container *ngFor=\"let column of filteredColumns; trackBy:columnIdent\" [matColumnDef]=\"column.id + COLLECTION_POSTFIX\">\n      <th mat-header-cell *matHeaderCellDef matTooltip=\"{{column.label}}\" >in Sammlung</th>\n      <td [class.noMaterial]=\"!row.counts[column.id]?.sufficient || row.counts[column.id]?.sufficient <= 1\"\n          [class.fewMaterials]=\"1 < row.counts[column.id]?.sufficient && row.counts[column.id]?.sufficient <= 3\"\n          class=\"collection-data-cell\"\n          mat-cell *matCellDef=\"let row\">\n            <a (click)=\"showCollectionItems(row.meta.id, row.meta.label, column.id)\">\n              {{ row.counts[column.id]?.sufficient || '0' }}\n            </a>\n      </td>\n    </ng-container>\n    <!-- /collection Data Columns -->\n    <!-- search Data Columns -->\n    <ng-container *ngFor=\"let column of filteredColumns let index = index; trackBy:columnIdent\" [matColumnDef]=\"column.id + SEARCH_POSTFIX\">\n      <th mat-header-cell *matHeaderCellDef>Suche</th>\n      <td class=\"search-data-cell\" mat-cell *matCellDef=\"let row\">\n        <a (click)=\"searchInEditor(column.id, row.meta.label)\">\n          <ng-container *ngIf=\"(( searchCounts | async)?.has(column.id + '_' + row.meta.id) ) else zeroOrLoading\">\n            {{ (searchCounts | async)?.get(column.id + '_' + row.meta.id) }}\n          </ng-container>\n          <ng-template #zeroOrLoading>\n            <ng-container *ngIf=\"rowsLoaded.has(row.meta.id) else loadingBlock\"> 0 </ng-container>\n          </ng-template>\n          <ng-template #loadingBlock>\n            loading…\n          </ng-template>\n        </a>\n      </td>\n    </ng-container>\n    <!-- /search Data Columns -->\n      <!-- Generate actual table -->\n    <tr mat-header-row *matHeaderRowDef=\"['label-col'].concat( typeColumns ); sticky: true;\"></tr>\n    <tr mat-header-row *matHeaderRowDef=\"alternatingDataColumns; sticky: true;\"></tr>\n    <tr mat-row *matRowDef=\"let row; columns: ['label-col'].concat(alternatingDataColumns)\"></tr>\n  </table>\n</mat-card>"]}
|