@hestia-earth/ui-components 0.27.0 → 0.27.2
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/common/blank-node-state-notice/blank-node-state-notice.component.d.ts +6 -5
- package/common/data-table/data-table.component.d.ts +10 -12
- package/common/dropdown/dropdown.component.d.ts +5 -5
- package/common/navigation-menu/navigation-menu.component.d.ts +6 -6
- package/common/social-tags/social-tags.component.d.ts +13 -5
- package/esm2022/common/blank-node-state-notice/blank-node-state-notice.component.mjs +9 -16
- package/esm2022/common/data-table/data-table.component.mjs +15 -27
- package/esm2022/common/dropdown/dropdown.component.mjs +8 -15
- package/esm2022/common/navigation-menu/navigation-menu.component.mjs +15 -23
- package/esm2022/common/social-tags/social-tags.component.mjs +21 -18
- package/esm2022/node/node-csv-export-confirm/node-csv-export-confirm.component.mjs +3 -3
- package/esm2022/node/node-csv-select-headers/node-csv-select-headers.component.mjs +4 -4
- package/fesm2022/hestia-earth-ui-components.mjs +63 -95
- package/fesm2022/hestia-earth-ui-components.mjs.map +1 -1
- package/node/node-csv-select-headers/node-csv-select-headers.component.d.ts +1 -1
- package/package.json +1 -1
|
@@ -1,35 +1,38 @@
|
|
|
1
|
-
import { ChangeDetectionStrategy, Component, HostBinding,
|
|
1
|
+
import { ChangeDetectionStrategy, Component, HostBinding, computed, effect, inject, input } from '@angular/core';
|
|
2
2
|
import { Meta } from '@angular/platform-browser';
|
|
3
3
|
import * as i0 from "@angular/core";
|
|
4
4
|
export class SocialTagsComponent {
|
|
5
5
|
constructor() {
|
|
6
6
|
this.meta = inject(Meta);
|
|
7
|
-
this.config = {};
|
|
7
|
+
this.config = input({});
|
|
8
8
|
this.classes = 'is-hidden';
|
|
9
|
-
|
|
10
|
-
ngOnInit() {
|
|
11
|
-
this.meta.addTag({ charset: 'UTF-8' });
|
|
12
|
-
const config = {
|
|
9
|
+
this.configs = computed(() => ({
|
|
13
10
|
'og:url': window.location.href.split('?')[0],
|
|
14
11
|
...this.config
|
|
15
|
-
};
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
this.meta.updateTag({ name
|
|
20
|
-
|
|
12
|
+
}));
|
|
13
|
+
this.meta.addTag({ charset: 'UTF-8' });
|
|
14
|
+
effect(() => {
|
|
15
|
+
Object.entries(this.configs()).map(([name, content]) => {
|
|
16
|
+
this.meta.updateTag({ name, content });
|
|
17
|
+
if (name === 'og:description') {
|
|
18
|
+
this.meta.updateTag({ name: 'description', content });
|
|
19
|
+
}
|
|
20
|
+
});
|
|
21
21
|
});
|
|
22
22
|
}
|
|
23
23
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.8", ngImport: i0, type: SocialTagsComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
24
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "
|
|
24
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "17.3.8", type: SocialTagsComponent, isStandalone: true, selector: "he-social-tags", inputs: { config: { classPropertyName: "config", publicName: "config", isSignal: true, isRequired: false, transformFunction: null } }, host: { properties: { "class": "this.classes" } }, ngImport: i0, template: '', isInline: true, changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
25
25
|
}
|
|
26
26
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.8", ngImport: i0, type: SocialTagsComponent, decorators: [{
|
|
27
27
|
type: Component,
|
|
28
|
-
args: [{
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
28
|
+
args: [{
|
|
29
|
+
selector: 'he-social-tags',
|
|
30
|
+
template: '',
|
|
31
|
+
changeDetection: ChangeDetectionStrategy.OnPush,
|
|
32
|
+
standalone: true
|
|
33
|
+
}]
|
|
34
|
+
}], ctorParameters: () => [], propDecorators: { classes: [{
|
|
32
35
|
type: HostBinding,
|
|
33
36
|
args: ['class']
|
|
34
37
|
}] } });
|
|
35
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
38
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic29jaWFsLXRhZ3MuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL2NvbW1vbi9zb2NpYWwtdGFncy9zb2NpYWwtdGFncy5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLHVCQUF1QixFQUFFLFNBQVMsRUFBRSxXQUFXLEVBQUUsUUFBUSxFQUFFLE1BQU0sRUFBRSxNQUFNLEVBQUUsS0FBSyxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ2pILE9BQU8sRUFBRSxJQUFJLEVBQUUsTUFBTSwyQkFBMkIsQ0FBQzs7QUFnQmpELE1BQU0sT0FBTyxtQkFBbUI7SUFhOUI7UUFaUSxTQUFJLEdBQUcsTUFBTSxDQUFDLElBQUksQ0FBQyxDQUFDO1FBRWxCLFdBQU0sR0FBRyxLQUFLLENBQW9CLEVBQUUsQ0FBQyxDQUFDO1FBR3pDLFlBQU8sR0FBRyxXQUFXLENBQUM7UUFFckIsWUFBTyxHQUFHLFFBQVEsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxDQUFDO1lBQ2hDLFFBQVEsRUFBRSxNQUFNLENBQUMsUUFBUSxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDO1lBQzVDLEdBQUcsSUFBSSxDQUFDLE1BQU07U0FDZixDQUFDLENBQUMsQ0FBQztRQUdGLElBQUksQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLEVBQUUsT0FBTyxFQUFFLE9BQU8sRUFBRSxDQUFDLENBQUM7UUFFdkMsTUFBTSxDQUFDLEdBQUcsRUFBRTtZQUNWLE1BQU0sQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLE9BQU8sRUFBRSxDQUFDLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxJQUFJLEVBQUUsT0FBTyxDQUFDLEVBQUUsRUFBRTtnQkFDckQsSUFBSSxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsRUFBRSxJQUFJLEVBQUUsT0FBTyxFQUFFLENBQUMsQ0FBQztnQkFDdkMsSUFBSSxJQUFJLEtBQUssZ0JBQWdCLEVBQUUsQ0FBQztvQkFDOUIsSUFBSSxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsRUFBRSxJQUFJLEVBQUUsYUFBYSxFQUFFLE9BQU8sRUFBRSxDQUFDLENBQUM7Z0JBQ3hELENBQUM7WUFDSCxDQUFDLENBQUMsQ0FBQztRQUNMLENBQUMsQ0FBQyxDQUFDO0lBQ0wsQ0FBQzs4R0F4QlUsbUJBQW1CO2tHQUFuQixtQkFBbUIsb1FBSnBCLEVBQUU7OzJGQUlELG1CQUFtQjtrQkFOL0IsU0FBUzttQkFBQztvQkFDVCxRQUFRLEVBQUUsZ0JBQWdCO29CQUMxQixRQUFRLEVBQUUsRUFBRTtvQkFDWixlQUFlLEVBQUUsdUJBQXVCLENBQUMsTUFBTTtvQkFDL0MsVUFBVSxFQUFFLElBQUk7aUJBQ2pCO3dEQU9RLE9BQU87c0JBRGIsV0FBVzt1QkFBQyxPQUFPIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3ksIENvbXBvbmVudCwgSG9zdEJpbmRpbmcsIGNvbXB1dGVkLCBlZmZlY3QsIGluamVjdCwgaW5wdXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IE1ldGEgfSBmcm9tICdAYW5ndWxhci9wbGF0Zm9ybS1icm93c2VyJztcblxuaW50ZXJmYWNlIElTb2NpYWxUYWdzQ29uZmlnIHtcbiAgJ29nOnR5cGUnPzogc3RyaW5nO1xuICAnb2c6dXJsJz86IHN0cmluZztcbiAgJ29nOnRpdGxlJz86IHN0cmluZztcbiAgJ29nOmRlc2NyaXB0aW9uJz86IHN0cmluZztcbiAgJ29nOmltYWdlJz86IHN0cmluZztcbn1cblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnaGUtc29jaWFsLXRhZ3MnLFxuICB0ZW1wbGF0ZTogJycsXG4gIGNoYW5nZURldGVjdGlvbjogQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3kuT25QdXNoLFxuICBzdGFuZGFsb25lOiB0cnVlXG59KVxuZXhwb3J0IGNsYXNzIFNvY2lhbFRhZ3NDb21wb25lbnQge1xuICBwcml2YXRlIG1ldGEgPSBpbmplY3QoTWV0YSk7XG5cbiAgcHJvdGVjdGVkIGNvbmZpZyA9IGlucHV0PElTb2NpYWxUYWdzQ29uZmlnPih7fSk7XG5cbiAgQEhvc3RCaW5kaW5nKCdjbGFzcycpXG4gIHB1YmxpYyBjbGFzc2VzID0gJ2lzLWhpZGRlbic7XG5cbiAgcHJpdmF0ZSBjb25maWdzID0gY29tcHV0ZWQoKCkgPT4gKHtcbiAgICAnb2c6dXJsJzogd2luZG93LmxvY2F0aW9uLmhyZWYuc3BsaXQoJz8nKVswXSxcbiAgICAuLi50aGlzLmNvbmZpZ1xuICB9KSk7XG5cbiAgY29uc3RydWN0b3IoKSB7XG4gICAgdGhpcy5tZXRhLmFkZFRhZyh7IGNoYXJzZXQ6ICdVVEYtOCcgfSk7XG5cbiAgICBlZmZlY3QoKCkgPT4ge1xuICAgICAgT2JqZWN0LmVudHJpZXModGhpcy5jb25maWdzKCkpLm1hcCgoW25hbWUsIGNvbnRlbnRdKSA9PiB7XG4gICAgICAgIHRoaXMubWV0YS51cGRhdGVUYWcoeyBuYW1lLCBjb250ZW50IH0pO1xuICAgICAgICBpZiAobmFtZSA9PT0gJ29nOmRlc2NyaXB0aW9uJykge1xuICAgICAgICAgIHRoaXMubWV0YS51cGRhdGVUYWcoeyBuYW1lOiAnZGVzY3JpcHRpb24nLCBjb250ZW50IH0pO1xuICAgICAgICB9XG4gICAgICB9KTtcbiAgICB9KTtcbiAgfVxufVxuIl19
|
|
@@ -32,10 +32,10 @@ export class NodeCsvExportConfirmComponent {
|
|
|
32
32
|
this.downloadFilename = computed(() => fileToExt(this.filename(), this.extension()));
|
|
33
33
|
}
|
|
34
34
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.8", ngImport: i0, type: NodeCsvExportConfirmComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
35
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.8", type: NodeCsvExportConfirmComponent, isStandalone: true, selector: "he-node-csv-export-confirm", inputs: { nodes: { classPropertyName: "nodes", publicName: "nodes", isSignal: true, isRequired: false, transformFunction: null }, filename: { classPropertyName: "filename", publicName: "filename", isSignal: true, isRequired: false, transformFunction: null }, headerKeys: { classPropertyName: "headerKeys", publicName: "headerKeys", isSignal: true, isRequired: false, transformFunction: null }, extension: { classPropertyName: "extension", publicName: "extension", isSignal: true, isRequired: false, transformFunction: null }, isUpload: { classPropertyName: "isUpload", publicName: "isUpload", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { closed: "closed" }, ngImport: i0, template: "<div class=\"modal is-active\">\n <div class=\"modal-background\"></div>\n <div class=\"modal-card\">\n <header class=\"modal-card-head\">\n <p class=\"modal-card-title\">Export as CSV</p>\n <button class=\"delete\" aria-label=\"close\" (click)=\"closed.emit(true)\"></button>\n </header>\n <section class=\"modal-card-body\">\n @if (isUpload()) {\n <div class=\"notification is-info\" role=\"alert\">\n <span>After Download, you can edit and</span>\n <a class=\"px-1\" routerLink=\"../\">upload the CSV file</a>\n <span>to submit your content on the Hestia platform, and your draft will remain unchanged.</span>\n <p>\n <span>\n Alternatively, you can import the CSV file right back by clicking on the \"Import from CSV\" button and\n selecting the exported file.\n </span>\n </p>\n </div>\n
|
|
35
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.8", type: NodeCsvExportConfirmComponent, isStandalone: true, selector: "he-node-csv-export-confirm", inputs: { nodes: { classPropertyName: "nodes", publicName: "nodes", isSignal: true, isRequired: false, transformFunction: null }, filename: { classPropertyName: "filename", publicName: "filename", isSignal: true, isRequired: false, transformFunction: null }, headerKeys: { classPropertyName: "headerKeys", publicName: "headerKeys", isSignal: true, isRequired: false, transformFunction: null }, extension: { classPropertyName: "extension", publicName: "extension", isSignal: true, isRequired: false, transformFunction: null }, isUpload: { classPropertyName: "isUpload", publicName: "isUpload", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { closed: "closed" }, ngImport: i0, template: "<div class=\"modal is-active\">\n <div class=\"modal-background\"></div>\n <div class=\"modal-card\">\n <header class=\"modal-card-head\">\n <p class=\"modal-card-title\">Export as CSV</p>\n <button class=\"delete\" aria-label=\"close\" (click)=\"closed.emit(true)\"></button>\n </header>\n <section class=\"modal-card-body\">\n @if (isUpload()) {\n <div class=\"notification is-info\" role=\"alert\">\n <span>After Download, you can edit and</span>\n <a class=\"px-1\" routerLink=\"../\">upload the CSV file</a>\n <span>to submit your content on the Hestia platform, and your draft will remain unchanged.</span>\n <p>\n <span>\n Alternatively, you can import the CSV file right back by clicking on the \"Import from CSV\" button and\n selecting the exported file.\n </span>\n </p>\n </div>\n\n } @else if (includedNodes().length > 1) {\n <p class=\"mb-2\">\n <b>{{ includedNodes().length }}</b>\n <span class=\"px-1\">Nodes will be included in your download.</span>\n <a (click)=\"showIncludeNodes.set(!showIncludeNodes())\">\n @if (showIncludeNodes()) {\n Hide list\n } @else {\n Show list\n }\n </a>\n </p>\n @if (showIncludeNodes()) {\n <he-data-table maxHeight=\"400\">\n <table class=\"table is-fullwidth is-striped is-mb-0\">\n <thead>\n <tr>\n <th class=\"width-auto has-border-right\">\n <span>Type</span>\n </th>\n <th>\n <span>Name</span>\n </th>\n </tr>\n </thead>\n <tbody>\n @for (node of includedNodes(); track node) {\n <tr>\n <td class=\"width-auto has-border-right\">\n {{ node.node['@type'] }}\n </td>\n <td>\n <he-node-link [node]=\"node.node\" [showExternalLink]=\"true\">\n <span>{{ node.node.name }}</span>\n </he-node-link>\n </td>\n </tr>\n }\n </tbody>\n </table>\n </he-data-table>\n }\n }\n\n <he-node-csv-select-headers\n [class.is-hidden]=\"isUpload()\"\n [csv]=\"csvData()\"\n [keys]=\"headerKeys()\"\n [includeDefaultCSV]=\"isUpload()\"\n (headersChanged)=\"headers.set($event)\"></he-node-csv-select-headers>\n </section>\n <footer class=\"modal-card-foot\">\n <a\n class=\"button is-primary\"\n target=\"_blank\"\n [attr.disabled]=\"csvContent() ? null : true\"\n [href]=\"csvContent()\"\n [attr.download]=\"csvContent() ? downloadFilename() : null\"\n (click)=\"closed.emit(true)\">\n <fa-icon class=\"mr-2\" [icon]=\"faDownload\"></fa-icon>\n <span>Download CSV</span>\n </a>\n <button class=\"button is-ghost\" (click)=\"closed.emit(true)\">\n <span>Cancel</span>\n </button>\n </footer>\n </div>\n</div>\n", styles: [""], dependencies: [{ kind: "directive", type: RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "info", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }, { kind: "component", type: DataTableComponent, selector: "he-data-table", inputs: ["minHeight", "maxHeight", "small"] }, { kind: "component", type: NodeLinkComponent, selector: "he-node-link", inputs: ["node", "showExternalLink", "linkClass"] }, { kind: "component", type: NodeCsvSelectHeadersComponent, selector: "he-node-csv-select-headers", inputs: ["csv", "keys", "includeDefaultCSV"], outputs: ["headersChanged"] }, { kind: "component", type: FaIconComponent, selector: "fa-icon", inputs: ["icon", "title", "animation", "spin", "pulse", "mask", "styles", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "classes", "transform", "a11yRole"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
36
36
|
}
|
|
37
37
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.8", ngImport: i0, type: NodeCsvExportConfirmComponent, decorators: [{
|
|
38
38
|
type: Component,
|
|
39
|
-
args: [{ selector: 'he-node-csv-export-confirm', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [RouterLink, DataTableComponent, NodeLinkComponent, NodeCsvSelectHeadersComponent, FaIconComponent], template: "<div class=\"modal is-active\">\n <div class=\"modal-background\"></div>\n <div class=\"modal-card\">\n <header class=\"modal-card-head\">\n <p class=\"modal-card-title\">Export as CSV</p>\n <button class=\"delete\" aria-label=\"close\" (click)=\"closed.emit(true)\"></button>\n </header>\n <section class=\"modal-card-body\">\n @if (isUpload()) {\n <div class=\"notification is-info\" role=\"alert\">\n <span>After Download, you can edit and</span>\n <a class=\"px-1\" routerLink=\"../\">upload the CSV file</a>\n <span>to submit your content on the Hestia platform, and your draft will remain unchanged.</span>\n <p>\n <span>\n Alternatively, you can import the CSV file right back by clicking on the \"Import from CSV\" button and\n selecting the exported file.\n </span>\n </p>\n </div>\n
|
|
39
|
+
args: [{ selector: 'he-node-csv-export-confirm', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [RouterLink, DataTableComponent, NodeLinkComponent, NodeCsvSelectHeadersComponent, FaIconComponent], template: "<div class=\"modal is-active\">\n <div class=\"modal-background\"></div>\n <div class=\"modal-card\">\n <header class=\"modal-card-head\">\n <p class=\"modal-card-title\">Export as CSV</p>\n <button class=\"delete\" aria-label=\"close\" (click)=\"closed.emit(true)\"></button>\n </header>\n <section class=\"modal-card-body\">\n @if (isUpload()) {\n <div class=\"notification is-info\" role=\"alert\">\n <span>After Download, you can edit and</span>\n <a class=\"px-1\" routerLink=\"../\">upload the CSV file</a>\n <span>to submit your content on the Hestia platform, and your draft will remain unchanged.</span>\n <p>\n <span>\n Alternatively, you can import the CSV file right back by clicking on the \"Import from CSV\" button and\n selecting the exported file.\n </span>\n </p>\n </div>\n\n } @else if (includedNodes().length > 1) {\n <p class=\"mb-2\">\n <b>{{ includedNodes().length }}</b>\n <span class=\"px-1\">Nodes will be included in your download.</span>\n <a (click)=\"showIncludeNodes.set(!showIncludeNodes())\">\n @if (showIncludeNodes()) {\n Hide list\n } @else {\n Show list\n }\n </a>\n </p>\n @if (showIncludeNodes()) {\n <he-data-table maxHeight=\"400\">\n <table class=\"table is-fullwidth is-striped is-mb-0\">\n <thead>\n <tr>\n <th class=\"width-auto has-border-right\">\n <span>Type</span>\n </th>\n <th>\n <span>Name</span>\n </th>\n </tr>\n </thead>\n <tbody>\n @for (node of includedNodes(); track node) {\n <tr>\n <td class=\"width-auto has-border-right\">\n {{ node.node['@type'] }}\n </td>\n <td>\n <he-node-link [node]=\"node.node\" [showExternalLink]=\"true\">\n <span>{{ node.node.name }}</span>\n </he-node-link>\n </td>\n </tr>\n }\n </tbody>\n </table>\n </he-data-table>\n }\n }\n\n <he-node-csv-select-headers\n [class.is-hidden]=\"isUpload()\"\n [csv]=\"csvData()\"\n [keys]=\"headerKeys()\"\n [includeDefaultCSV]=\"isUpload()\"\n (headersChanged)=\"headers.set($event)\"></he-node-csv-select-headers>\n </section>\n <footer class=\"modal-card-foot\">\n <a\n class=\"button is-primary\"\n target=\"_blank\"\n [attr.disabled]=\"csvContent() ? null : true\"\n [href]=\"csvContent()\"\n [attr.download]=\"csvContent() ? downloadFilename() : null\"\n (click)=\"closed.emit(true)\">\n <fa-icon class=\"mr-2\" [icon]=\"faDownload\"></fa-icon>\n <span>Download CSV</span>\n </a>\n <button class=\"button is-ghost\" (click)=\"closed.emit(true)\">\n <span>Cancel</span>\n </button>\n </footer>\n </div>\n</div>\n" }]
|
|
40
40
|
}] });
|
|
41
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node-csv-export-confirm.component.js","sourceRoot":"","sources":["../../../../src/node/node-csv-export-confirm/node-csv-export-confirm.component.ts","../../../../src/node/node-csv-export-confirm/node-csv-export-confirm.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, computed, inject, input, output, signal } from '@angular/core';\nimport { RouterLink } from '@angular/router';\nimport { DomSanitizer } from '@angular/platform-browser';\nimport { fileToExt, SupportedExtensions } from '@hestia-earth/api';\nimport { JSONLD } from '@hestia-earth/schema';\nimport { toCsv } from '@hestia-earth/schema-convert';\nimport { FaIconComponent } from '@fortawesome/angular-fontawesome';\nimport { faDownload } from '@fortawesome/free-solid-svg-icons';\n\nimport { NodeCsvSelectHeadersComponent } from '../node-csv-select-headers/node-csv-select-headers.component';\nimport { NodeLinkComponent } from '../node-link/node-link.component';\nimport { DataTableComponent } from '../../common/data-table/data-table.component';\n\ninterface INodeIncluded {\n  node: JSONLD<any>;\n  included: boolean;\n}\n\n@Component({\n  selector: 'he-node-csv-export-confirm',\n  templateUrl: './node-csv-export-confirm.component.html',\n  styleUrls: ['./node-csv-export-confirm.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [RouterLink, DataTableComponent, NodeLinkComponent, NodeCsvSelectHeadersComponent, FaIconComponent]\n})\nexport class NodeCsvExportConfirmComponent {\n  private readonly domSanitizer = inject(DomSanitizer);\n\n  protected readonly faDownload = faDownload;\n\n  protected headers = signal<string[]>([]);\n\n  protected nodes = input<JSONLD<any>[]>([]);\n  protected filename = input<string>('download');\n  protected headerKeys = input<string[]>([]);\n  protected extension = input(SupportedExtensions.csv);\n  protected isUpload = input(true);\n\n  protected closed = output<boolean>();\n\n  protected showIncludeNodes = signal(false);\n  protected includedNodes = computed(() =>\n    (this.nodes() ?? []).map(node => ({ node, included: true }) as INodeIncluded)\n  );\n  private included = computed(() =>\n    this.includedNodes()\n      .filter(({ included }) => included)\n      .map(({ node }) => node)\n  );\n\n  protected csvData = computed(() => toCsv(this.nodes() || [], { includeExising: !this.isUpload() }));\n  protected csvContent = computed(() =>\n    this.headers().length\n      ? this.domSanitizer.bypassSecurityTrustResourceUrl(\n          `data:text/html;charset=utf-8,${encodeURIComponent(\n            toCsv(this.included(), { includeExising: !this.isUpload(), selectedHeaders: this.headers() })\n          )}`\n        )\n      : undefined\n  );\n\n  protected downloadFilename = computed(() => fileToExt(this.filename(), this.extension()));\n}\n","<div class=\"modal is-active\">\n  <div class=\"modal-background\"></div>\n  <div class=\"modal-card\">\n    <header class=\"modal-card-head\">\n      <p class=\"modal-card-title\">Export as CSV</p>\n      <button class=\"delete\" aria-label=\"close\" (click)=\"closed.emit(true)\"></button>\n    </header>\n    <section class=\"modal-card-body\">\n      @if (isUpload()) {\n        <div class=\"notification is-info\" role=\"alert\">\n          <span>After Download, you can edit and</span>\n          <a class=\"px-1\" routerLink=\"../\">upload the CSV file</a>\n          <span>to submit your content on the Hestia platform, and your draft will remain unchanged.</span>\n          <p>\n            <span>\n              Alternatively, you can import the CSV file right back by clicking on the \"Import from CSV\" button and\n              selecting the exported file.\n            </span>\n          </p>\n        </div>\n      }\n\n      @if (!isUpload() && includedNodes().length > 1) {\n        <p class=\"mb-2\">\n          <b>{{ includedNodes().length }}</b>\n          <span class=\"px-1\">Nodes will be included in your download.</span>\n          <a (click)=\"showIncludeNodes.set(!showIncludeNodes())\">\n            @if (showIncludeNodes()) {\n              Hide list\n            } @else {\n              Show list\n            }\n          </a>\n        </p>\n        @if (showIncludeNodes()) {\n          <he-data-table maxHeight=\"400\">\n            <table class=\"table is-fullwidth is-striped is-mb-0\">\n              <thead>\n                <tr>\n                  <th class=\"width-auto has-border-right\">\n                    <span>Type</span>\n                  </th>\n                  <th>\n                    <span>Name</span>\n                  </th>\n                </tr>\n              </thead>\n              <tbody>\n                @for (node of includedNodes(); track node) {\n                  <tr>\n                    <td class=\"width-auto has-border-right\">\n                      {{ node.node['@type'] }}\n                    </td>\n                    <td>\n                      <he-node-link [node]=\"node.node\" [showExternalLink]=\"true\">\n                        <span>{{ node.node.name }}</span>\n                      </he-node-link>\n                    </td>\n                  </tr>\n                }\n              </tbody>\n            </table>\n          </he-data-table>\n        }\n      }\n\n      <he-node-csv-select-headers\n        [class.is-hidden]=\"isUpload()\"\n        [csv]=\"csvData()\"\n        [keys]=\"headerKeys()\"\n        [includeDefaultCSV]=\"isUpload()\"\n        (headersChanged)=\"headers.set($event)\"></he-node-csv-select-headers>\n    </section>\n    <footer class=\"modal-card-foot\">\n      <a\n        class=\"button is-primary\"\n        target=\"_blank\"\n        [attr.disabled]=\"csvContent() ? null : true\"\n        [href]=\"csvContent()\"\n        [attr.download]=\"csvContent() ? downloadFilename() : null\"\n        (click)=\"closed.emit(true)\">\n        <fa-icon class=\"mr-2\" [icon]=\"faDownload\"></fa-icon>\n        <span>Download CSV</span>\n      </a>\n      <button class=\"button is-ghost\" (click)=\"closed.emit(true)\">\n        <span>Cancel</span>\n      </button>\n    </footer>\n  </div>\n</div>\n"]}
|
|
41
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node-csv-export-confirm.component.js","sourceRoot":"","sources":["../../../../src/node/node-csv-export-confirm/node-csv-export-confirm.component.ts","../../../../src/node/node-csv-export-confirm/node-csv-export-confirm.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, computed, inject, input, output, signal } from '@angular/core';\nimport { RouterLink } from '@angular/router';\nimport { DomSanitizer } from '@angular/platform-browser';\nimport { fileToExt, SupportedExtensions } from '@hestia-earth/api';\nimport { JSONLD } from '@hestia-earth/schema';\nimport { toCsv } from '@hestia-earth/schema-convert';\nimport { FaIconComponent } from '@fortawesome/angular-fontawesome';\nimport { faDownload } from '@fortawesome/free-solid-svg-icons';\n\nimport { NodeCsvSelectHeadersComponent } from '../node-csv-select-headers/node-csv-select-headers.component';\nimport { NodeLinkComponent } from '../node-link/node-link.component';\nimport { DataTableComponent } from '../../common/data-table/data-table.component';\n\ninterface INodeIncluded {\n  node: JSONLD<any>;\n  included: boolean;\n}\n\n@Component({\n  selector: 'he-node-csv-export-confirm',\n  templateUrl: './node-csv-export-confirm.component.html',\n  styleUrls: ['./node-csv-export-confirm.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [RouterLink, DataTableComponent, NodeLinkComponent, NodeCsvSelectHeadersComponent, FaIconComponent]\n})\nexport class NodeCsvExportConfirmComponent {\n  private readonly domSanitizer = inject(DomSanitizer);\n\n  protected readonly faDownload = faDownload;\n\n  protected headers = signal<string[]>([]);\n\n  protected nodes = input<JSONLD<any>[]>([]);\n  protected filename = input<string>('download');\n  protected headerKeys = input<string[]>([]);\n  protected extension = input(SupportedExtensions.csv);\n  protected isUpload = input(true);\n\n  protected closed = output<boolean>();\n\n  protected showIncludeNodes = signal(false);\n  protected includedNodes = computed(() =>\n    (this.nodes() ?? []).map(node => ({ node, included: true }) as INodeIncluded)\n  );\n  private included = computed(() =>\n    this.includedNodes()\n      .filter(({ included }) => included)\n      .map(({ node }) => node)\n  );\n\n  protected csvData = computed(() => toCsv(this.nodes() || [], { includeExising: !this.isUpload() }));\n  protected csvContent = computed(() =>\n    this.headers().length\n      ? this.domSanitizer.bypassSecurityTrustResourceUrl(\n          `data:text/html;charset=utf-8,${encodeURIComponent(\n            toCsv(this.included(), { includeExising: !this.isUpload(), selectedHeaders: this.headers() })\n          )}`\n        )\n      : undefined\n  );\n\n  protected downloadFilename = computed(() => fileToExt(this.filename(), this.extension()));\n}\n","<div class=\"modal is-active\">\n  <div class=\"modal-background\"></div>\n  <div class=\"modal-card\">\n    <header class=\"modal-card-head\">\n      <p class=\"modal-card-title\">Export as CSV</p>\n      <button class=\"delete\" aria-label=\"close\" (click)=\"closed.emit(true)\"></button>\n    </header>\n    <section class=\"modal-card-body\">\n      @if (isUpload()) {\n        <div class=\"notification is-info\" role=\"alert\">\n          <span>After Download, you can edit and</span>\n          <a class=\"px-1\" routerLink=\"../\">upload the CSV file</a>\n          <span>to submit your content on the Hestia platform, and your draft will remain unchanged.</span>\n          <p>\n            <span>\n              Alternatively, you can import the CSV file right back by clicking on the \"Import from CSV\" button and\n              selecting the exported file.\n            </span>\n          </p>\n        </div>\n\n      } @else if (includedNodes().length > 1) {\n        <p class=\"mb-2\">\n          <b>{{ includedNodes().length }}</b>\n          <span class=\"px-1\">Nodes will be included in your download.</span>\n          <a (click)=\"showIncludeNodes.set(!showIncludeNodes())\">\n            @if (showIncludeNodes()) {\n              Hide list\n            } @else {\n              Show list\n            }\n          </a>\n        </p>\n        @if (showIncludeNodes()) {\n          <he-data-table maxHeight=\"400\">\n            <table class=\"table is-fullwidth is-striped is-mb-0\">\n              <thead>\n                <tr>\n                  <th class=\"width-auto has-border-right\">\n                    <span>Type</span>\n                  </th>\n                  <th>\n                    <span>Name</span>\n                  </th>\n                </tr>\n              </thead>\n              <tbody>\n                @for (node of includedNodes(); track node) {\n                  <tr>\n                    <td class=\"width-auto has-border-right\">\n                      {{ node.node['@type'] }}\n                    </td>\n                    <td>\n                      <he-node-link [node]=\"node.node\" [showExternalLink]=\"true\">\n                        <span>{{ node.node.name }}</span>\n                      </he-node-link>\n                    </td>\n                  </tr>\n                }\n              </tbody>\n            </table>\n          </he-data-table>\n        }\n      }\n\n      <he-node-csv-select-headers\n        [class.is-hidden]=\"isUpload()\"\n        [csv]=\"csvData()\"\n        [keys]=\"headerKeys()\"\n        [includeDefaultCSV]=\"isUpload()\"\n        (headersChanged)=\"headers.set($event)\"></he-node-csv-select-headers>\n    </section>\n    <footer class=\"modal-card-foot\">\n      <a\n        class=\"button is-primary\"\n        target=\"_blank\"\n        [attr.disabled]=\"csvContent() ? null : true\"\n        [href]=\"csvContent()\"\n        [attr.download]=\"csvContent() ? downloadFilename() : null\"\n        (click)=\"closed.emit(true)\">\n        <fa-icon class=\"mr-2\" [icon]=\"faDownload\"></fa-icon>\n        <span>Download CSV</span>\n      </a>\n      <button class=\"button is-ghost\" (click)=\"closed.emit(true)\">\n        <span>Cancel</span>\n      </button>\n    </footer>\n  </div>\n</div>\n"]}
|
|
@@ -40,7 +40,7 @@ export class NodeCsvSelectHeadersComponent {
|
|
|
40
40
|
this.faAngleLeft = faAngleLeft;
|
|
41
41
|
this.csv = input.required();
|
|
42
42
|
this.csv$ = toObservable(this.csv);
|
|
43
|
-
this.keys = input
|
|
43
|
+
this.keys = input([]);
|
|
44
44
|
this.keys$ = toObservable(this.keys);
|
|
45
45
|
this.includeDefaultCSV = input(false);
|
|
46
46
|
this.headersChanged = output();
|
|
@@ -58,7 +58,7 @@ export class NodeCsvSelectHeadersComponent {
|
|
|
58
58
|
this.headers = signal({});
|
|
59
59
|
this.groups = computed(() => Object.values(this.headers()));
|
|
60
60
|
combineLatest([this.csv$, this.keys$, this.schemas$])
|
|
61
|
-
.pipe(filter(([csv,
|
|
61
|
+
.pipe(filter(([csv, _keys, schemas]) => [!isUndefined(csv), !isUndefined(schemas)].every(Boolean)), tap(() => this.loading.set(true)), mergeMap(([csv, keys, schemas]) => forkJoin({
|
|
62
62
|
headers: this.schemaService.parseHeaders$(csv),
|
|
63
63
|
keys: of(keys),
|
|
64
64
|
schemas: of(schemas)
|
|
@@ -149,7 +149,7 @@ export class NodeCsvSelectHeadersComponent {
|
|
|
149
149
|
this.headers.set(headers);
|
|
150
150
|
}
|
|
151
151
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.8", ngImport: i0, type: NodeCsvSelectHeadersComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
152
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.8", type: NodeCsvSelectHeadersComponent, isStandalone: true, selector: "he-node-csv-select-headers", inputs: { csv: { classPropertyName: "csv", publicName: "csv", isSignal: true, isRequired: true, transformFunction: null }, keys: { classPropertyName: "keys", publicName: "keys", isSignal: true, isRequired:
|
|
152
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.8", type: NodeCsvSelectHeadersComponent, isStandalone: true, selector: "he-node-csv-select-headers", inputs: { csv: { classPropertyName: "csv", publicName: "csv", isSignal: true, isRequired: true, transformFunction: null }, keys: { classPropertyName: "keys", publicName: "keys", isSignal: true, isRequired: false, transformFunction: null }, includeDefaultCSV: { classPropertyName: "includeDefaultCSV", publicName: "includeDefaultCSV", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { headersChanged: "headersChanged" }, ngImport: i0, template: "<p class=\"mb-2\">Please select which columns you would like to include:</p>\n\n<p class=\"my-2 is-size-7\">\n <i>You can drag and drop the headers to sort them as they would appear in the CSV file.</i>\n</p>\n\n<div class=\"columns toggle-all mx-4 pb-1 mb-0\">\n <div class=\"column\">\n <label class=\"checkbox ml-1\">\n <input type=\"checkbox\" class=\"selector\" [ngModel]=\"allSelected()\" (change)=\"selectAll($event.target.checked)\" />\n <span class=\"ml-2\">Toggle All</span>\n </label>\n </div>\n\n <div class=\"column is-narrow has-text-right\">\n <div ngbDropdown container=\"body\" placement=\"bottom-end\">\n <button ngbDropdownToggle class=\"button is-small\" type=\"button\" id=\"select-menu\">\n <span>Advanced Filters</span>\n <span class=\"icon is-small\">\n <fa-icon [icon]=\"faFilter\" aria-hidden=\"true\"></fa-icon>\n </span>\n </button>\n <div ngbDropdownMenu aria-labelledby=\"select-menu\">\n <div class=\"dropdown-content\">\n <a ngbDropdownItem>\n <label class=\"checkbox ml-1\">\n <input\n type=\"checkbox\"\n class=\"selector\"\n [ngModel]=\"showNonIncluded()\"\n (change)=\"showNonIncluded.set($event.target.checked)\" />\n <span class=\"ml-2\">\n Include\n <code>internal</code>\n fields\n </span>\n </label>\n </a>\n <div class=\"dropdown-item\">\n <p>Toggle Term Fields</p>\n </div>\n @for (field of termFields; track field) {\n <div ngbDropdownItem>\n <label class=\"checkbox ml-1\">\n <input\n type=\"checkbox\"\n class=\"selector\"\n [checked]=\"isTermFieldSelect(field)\"\n (change)=\"toggleTermField($event.target.checked, field)\" />\n <span class=\"ml-2\">{{ field }}</span>\n </label>\n </div>\n }\n </div>\n </div>\n </div>\n </div>\n</div>\n\n@if (loading()) {\n <div class=\"has-text-center py-3\">\n <fa-icon [icon]=\"faSpinner\" [pulse]=\"true\" size=\"lg\"></fa-icon>\n </div>\n} @else {\n <div class=\"drag-container\">\n @for (group of groups(); track group.key) {\n <div class=\"card\">\n <header class=\"card-header\">\n <div class=\"card-header-title\">\n <label class=\"checkbox ml-1\">\n <input\n type=\"checkbox\"\n class=\"selector\"\n [indeterminate]=\"group.partialSelected\"\n [ngModel]=\"group.selected\"\n (change)=\"updateGroup($event.target.checked, group)\" />\n <span class=\"ml-2\">{{ group.key }}</span>\n </label>\n </div>\n <span\n class=\"card-header-icon has-text-secondary\"\n aria-label=\"open / close\"\n (click)=\"group.open = !group.open\"\n pointer>\n <span class=\"icon\">\n <fa-icon [icon]=\"faAngleDown\" [class.is-hidden]=\"!group.open\"></fa-icon>\n <fa-icon [icon]=\"faAngleLeft\" [class.is-hidden]=\"group.open\"></fa-icon>\n </span>\n </span>\n </header>\n <div class=\"card-content p-3\" [class.is-hidden]=\"!group.open\">\n <ul cdkDropList (cdkDropListDropped)=\"dropHeader($event, group)\">\n @for (header of group.headers; track header.key) {\n @if (header.included || showNonIncluded()) {\n <li cdkDrag>\n <label class=\"checkbox ml-2\">\n <input\n type=\"checkbox\"\n class=\"selector\"\n [ngModel]=\"header.selected\"\n (change)=\"updateHeader($event.target.checked, group, header)\" />\n <span class=\"ml-2\">{{ header.key }}</span>\n </label>\n </li>\n }\n }\n </ul>\n </div>\n </div>\n }\n </div>\n}\n", styles: [".toggle-all{border-bottom:1px solid #4a4a4a}.drag-container,.table-container{max-height:45vh;overflow-y:auto}.cdk-drag-preview{border-radius:3px;box-shadow:0 5px 5px -3px #0003,0 8px 10px 1px #00000024,0 3px 14px 2px #0000001f;list-style:none}.cdk-drag-placeholder{opacity:0}.cdk-drag-animating{transition:transform .25s cubic-bezier(0,0,.2,1)}\n"], dependencies: [{ kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i1.CheckboxControlValueAccessor, selector: "input[type=checkbox][formControlName],input[type=checkbox][formControl],input[type=checkbox][ngModel]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "directive", type: NgbDropdown, selector: "[ngbDropdown]", inputs: ["autoClose", "dropdownClass", "open", "placement", "popperOptions", "container", "display"], outputs: ["openChange"], exportAs: ["ngbDropdown"] }, { kind: "directive", type: NgbDropdownToggle, selector: "[ngbDropdownToggle]" }, { kind: "component", type: FaIconComponent, selector: "fa-icon", inputs: ["icon", "title", "animation", "spin", "pulse", "mask", "styles", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "classes", "transform", "a11yRole"] }, { kind: "directive", type: NgbDropdownMenu, selector: "[ngbDropdownMenu]" }, { kind: "directive", type: NgbDropdownItem, selector: "[ngbDropdownItem]", inputs: ["tabindex", "disabled"] }, { kind: "directive", type: CdkDropList, selector: "[cdkDropList], cdk-drop-list", inputs: ["cdkDropListConnectedTo", "cdkDropListData", "cdkDropListOrientation", "id", "cdkDropListLockAxis", "cdkDropListDisabled", "cdkDropListSortingDisabled", "cdkDropListEnterPredicate", "cdkDropListSortPredicate", "cdkDropListAutoScrollDisabled", "cdkDropListAutoScrollStep"], outputs: ["cdkDropListDropped", "cdkDropListEntered", "cdkDropListExited", "cdkDropListSorted"], exportAs: ["cdkDropList"] }, { kind: "directive", type: CdkDrag, selector: "[cdkDrag]", inputs: ["cdkDragData", "cdkDragLockAxis", "cdkDragRootElement", "cdkDragBoundary", "cdkDragStartDelay", "cdkDragFreeDragPosition", "cdkDragDisabled", "cdkDragConstrainPosition", "cdkDragPreviewClass", "cdkDragPreviewContainer"], outputs: ["cdkDragStarted", "cdkDragReleased", "cdkDragEnded", "cdkDragEntered", "cdkDragExited", "cdkDragDropped", "cdkDragMoved"], exportAs: ["cdkDrag"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
153
153
|
}
|
|
154
154
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.8", ngImport: i0, type: NodeCsvSelectHeadersComponent, decorators: [{
|
|
155
155
|
type: Component,
|
|
@@ -164,4 +164,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.8", ngImpor
|
|
|
164
164
|
CdkDrag
|
|
165
165
|
], template: "<p class=\"mb-2\">Please select which columns you would like to include:</p>\n\n<p class=\"my-2 is-size-7\">\n <i>You can drag and drop the headers to sort them as they would appear in the CSV file.</i>\n</p>\n\n<div class=\"columns toggle-all mx-4 pb-1 mb-0\">\n <div class=\"column\">\n <label class=\"checkbox ml-1\">\n <input type=\"checkbox\" class=\"selector\" [ngModel]=\"allSelected()\" (change)=\"selectAll($event.target.checked)\" />\n <span class=\"ml-2\">Toggle All</span>\n </label>\n </div>\n\n <div class=\"column is-narrow has-text-right\">\n <div ngbDropdown container=\"body\" placement=\"bottom-end\">\n <button ngbDropdownToggle class=\"button is-small\" type=\"button\" id=\"select-menu\">\n <span>Advanced Filters</span>\n <span class=\"icon is-small\">\n <fa-icon [icon]=\"faFilter\" aria-hidden=\"true\"></fa-icon>\n </span>\n </button>\n <div ngbDropdownMenu aria-labelledby=\"select-menu\">\n <div class=\"dropdown-content\">\n <a ngbDropdownItem>\n <label class=\"checkbox ml-1\">\n <input\n type=\"checkbox\"\n class=\"selector\"\n [ngModel]=\"showNonIncluded()\"\n (change)=\"showNonIncluded.set($event.target.checked)\" />\n <span class=\"ml-2\">\n Include\n <code>internal</code>\n fields\n </span>\n </label>\n </a>\n <div class=\"dropdown-item\">\n <p>Toggle Term Fields</p>\n </div>\n @for (field of termFields; track field) {\n <div ngbDropdownItem>\n <label class=\"checkbox ml-1\">\n <input\n type=\"checkbox\"\n class=\"selector\"\n [checked]=\"isTermFieldSelect(field)\"\n (change)=\"toggleTermField($event.target.checked, field)\" />\n <span class=\"ml-2\">{{ field }}</span>\n </label>\n </div>\n }\n </div>\n </div>\n </div>\n </div>\n</div>\n\n@if (loading()) {\n <div class=\"has-text-center py-3\">\n <fa-icon [icon]=\"faSpinner\" [pulse]=\"true\" size=\"lg\"></fa-icon>\n </div>\n} @else {\n <div class=\"drag-container\">\n @for (group of groups(); track group.key) {\n <div class=\"card\">\n <header class=\"card-header\">\n <div class=\"card-header-title\">\n <label class=\"checkbox ml-1\">\n <input\n type=\"checkbox\"\n class=\"selector\"\n [indeterminate]=\"group.partialSelected\"\n [ngModel]=\"group.selected\"\n (change)=\"updateGroup($event.target.checked, group)\" />\n <span class=\"ml-2\">{{ group.key }}</span>\n </label>\n </div>\n <span\n class=\"card-header-icon has-text-secondary\"\n aria-label=\"open / close\"\n (click)=\"group.open = !group.open\"\n pointer>\n <span class=\"icon\">\n <fa-icon [icon]=\"faAngleDown\" [class.is-hidden]=\"!group.open\"></fa-icon>\n <fa-icon [icon]=\"faAngleLeft\" [class.is-hidden]=\"group.open\"></fa-icon>\n </span>\n </span>\n </header>\n <div class=\"card-content p-3\" [class.is-hidden]=\"!group.open\">\n <ul cdkDropList (cdkDropListDropped)=\"dropHeader($event, group)\">\n @for (header of group.headers; track header.key) {\n @if (header.included || showNonIncluded()) {\n <li cdkDrag>\n <label class=\"checkbox ml-2\">\n <input\n type=\"checkbox\"\n class=\"selector\"\n [ngModel]=\"header.selected\"\n (change)=\"updateHeader($event.target.checked, group, header)\" />\n <span class=\"ml-2\">{{ header.key }}</span>\n </label>\n </li>\n }\n }\n </ul>\n </div>\n </div>\n }\n </div>\n}\n", styles: [".toggle-all{border-bottom:1px solid #4a4a4a}.drag-container,.table-container{max-height:45vh;overflow-y:auto}.cdk-drag-preview{border-radius:3px;box-shadow:0 5px 5px -3px #0003,0 8px 10px 1px #00000024,0 3px 14px 2px #0000001f;list-style:none}.cdk-drag-placeholder{opacity:0}.cdk-drag-animating{transition:transform .25s cubic-bezier(0,0,.2,1)}\n"] }]
|
|
166
166
|
}], ctorParameters: () => [] });
|
|
167
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node-csv-select-headers.component.js","sourceRoot":"","sources":["../../../../src/node/node-csv-select-headers/node-csv-select-headers.component.ts","../../../../src/node/node-csv-select-headers/node-csv-select-headers.component.html"],"names":[],"mappings":"AAAA,OAAO,EACL,uBAAuB,EACvB,SAAS,EACT,UAAU,EACV,QAAQ,EACR,MAAM,EACN,MAAM,EACN,KAAK,EACL,MAAM,EACN,MAAM,EACP,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,kBAAkB,EAAE,YAAY,EAAE,MAAM,4BAA4B,CAAC;AAC9E,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAC7C,OAAO,EAAe,eAAe,EAAE,WAAW,EAAE,OAAO,EAAE,MAAM,wBAAwB,CAAC;AAC5F,OAAO,EAAE,oBAAoB,EAAE,aAAa,EAAE,MAAM,wCAAwC,CAAC;AAC7F,OAAO,EAAE,eAAe,EAAE,MAAM,kCAAkC,CAAC;AACnE,OAAO,EAAE,WAAW,EAAE,iBAAiB,EAAE,eAAe,EAAE,eAAe,EAAE,MAAM,4BAA4B,CAAC;AAC9G,OAAO,EAAE,MAAM,EAAE,QAAQ,EAAE,GAAG,EAAE,MAAM,gBAAgB,CAAC;AACvD,OAAO,EAAE,WAAW,EAAE,MAAM,qBAAqB,CAAC;AAClD,OAAO,EAAE,aAAa,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,MAAM,CAAC;AAEnD,OAAO,EAAE,eAAe,EAAE,MAAM,6BAA6B,CAAC;AAC9D,OAAO,EAAE,WAAW,EAAE,WAAW,EAAE,QAAQ,EAAE,SAAS,EAAE,MAAM,mCAAmC,CAAC;;;AAuBlG,MAAM,WAAW,GAAG,CAAC,MAAc,EAAE,EAAE;IACrC,MAAM,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IAChC,OAAO,KAAK,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;AAClE,CAAC,CAAC;AAEF,MAAM,UAAU,GAAG,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,EAAE,UAAU,CAAC,CAAC;AAExD,MAAM,WAAW,GAAG,CAAC,KAAa,EAAE,EAAE,GAAG,EAAW,EAAE,EAAE,CACtD,CAAC,MAAM,EAAE,aAAa,EAAE,SAAS,EAAE,SAAS,EAAE,WAAW,EAAE,KAAK,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAC9E,GAAG,CAAC,QAAQ,CAAC,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CACxC,CAAC;AAEJ,MAAM,0BAA0B,GAAG,CAAC,EAAE,OAAO,EAAE,GAAG,KAAK,EAAU,EAAE,EAAE,CAAC,CAAC;IACrE,GAAG,KAAK;IACR,OAAO,EAAE,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,EAAE,GAAG,MAAM,EAAE,QAAQ,EAAE,KAAK,CAAC,QAAQ,EAAE,CAAC,CAAC;CAC1E,CAAC,CAAC;AAEH,MAAM,mBAAmB,GAAG,CAAC,KAAa,EAAE,EAAE;IAC5C,MAAM,WAAW,GAAG,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,QAAQ,EAAE,EAAE,EAAE,CAAC,QAAQ,CAAC,CAAC;IACpE,OAAO;QACL,GAAG,KAAK;QACR,QAAQ,EAAE,WAAW;QACrB,eAAe,EAAE,CAAC,WAAW,IAAI,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,EAAE,QAAQ,EAAE,EAAE,EAAE,CAAC,QAAQ,CAAC;KAChF,CAAC;AACJ,CAAC,CAAC;AAmBF,MAAM,OAAO,6BAA6B;IAoCxC;QAnCQ,eAAU,GAAG,MAAM,CAAC,UAAU,CAAC,CAAC;QAChC,kBAAa,GAAG,MAAM,CAAC,eAAe,CAAC,CAAC;QAE7B,cAAS,GAAG,SAAS,CAAC;QACtB,aAAQ,GAAG,QAAQ,CAAC;QACpB,gBAAW,GAAG,WAAW,CAAC;QAC1B,gBAAW,GAAG,WAAW,CAAC;QAEnC,QAAG,GAAG,KAAK,CAAC,QAAQ,EAAU,CAAC;QACjC,SAAI,GAAG,YAAY,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAC5B,SAAI,GAAG,KAAK,CAAC,QAAQ,EAAY,CAAC;QACpC,UAAK,GAAG,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC9B,sBAAiB,GAAG,KAAK,CAAC,KAAK,CAAC,CAAC;QAEjC,mBAAc,GAAG,MAAM,EAAY,CAAC;QAEpC,YAAO,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC;QACvB,eAAU,GAAG,UAAU,CAAC;QACxB,uBAAkB,GAAG,MAAM,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC;QACrC,oBAAe,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;QAEhC,gBAAW,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,GAAG,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC;QACvG,oBAAe,GAAG,QAAQ,CAAC,GAAG,EAAE,CACtC,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC;aAC1B,OAAO,CAAC,CAAC,EAAE,OAAO,EAAE,EAAE,EAAE,CAAC,OAAO,CAAC;aACjC,MAAM,CAAC,CAAC,EAAE,QAAQ,EAAE,QAAQ,EAAE,EAAE,EAAE,CAAC,QAAQ,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,eAAe,EAAE,CAAC,CAAC;aACpF,GAAG,CAAC,CAAC,EAAE,GAAG,EAAE,EAAE,EAAE,CAAC,GAAG,CAAC,CACzB,CAAC;QAEM,aAAQ,GAAG,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC;QAEjF,eAAU,GAAG,MAAM,CAAW,EAAE,CAAC,CAAC;QAChC,YAAO,GAAG,MAAM,CAAkB,EAAE,CAAC,CAAC;QACtC,WAAM,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC;QAG/D,aAAa,CAAC,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,KAAK,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;aAClD,IAAI,CACH,MAAM,CAAC,CAAC,CAAC,GAAG,EAAE,IAAI,EAAE,OAAO,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,CAAC,GAAG,CAAC,EAAE,IAAI,EAAE,MAAM,GAAG,CAAC,EAAE,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,EAC7G,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,EACjC,QAAQ,CAAC,CAAC,CAAC,GAAG,EAAE,IAAI,EAAE,OAAO,CAAC,EAAE,EAAE,CAChC,QAAQ,CAAC;YACP,OAAO,EAAE,IAAI,CAAC,aAAa,CAAC,aAAa,CAAC,GAAG,CAAC;YAC9C,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC;YACd,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC;SACrB,CAAC,CACH,EACD,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,CACnC;aACA,SAAS,CAAC,CAAC,EAAE,OAAO,EAAE,IAAI,EAAE,OAAO,EAAE,EAAE,EAAE;YACxC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;YAE7B,MAAM,UAAU,GAAG,aAAa,CAAC,OAAO,CAAC,CAAC;YAC1C,MAAM,iBAAiB,GAAG,IAAI,CAAC,iBAAiB,EAAE,CAAC,CAAC,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC,oBAAoB,CAAC,OAAO,CAAC,CAAC;YAChG,MAAM,UAAU,GAAG,CAAC,GAAW,EAAE,EAAE,CACjC,CAAC,CAAC,IAAI,CAAC,MAAM,IAAI,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,iBAAiB,CAAC,GAAG,CAAC,CAAC;YAEhF,IAAI,CAAC,OAAO,CAAC,GAAG,CACd,OAAO,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,SAAS,EAAE,EAAE;gBACjC,MAAM,QAAQ,GAAG,WAAW,CAAC,SAAS,CAAC,CAAC;gBACxC,IAAI,CAAC,QAAQ,CAAC,GAAG,IAAI,CAAC,QAAQ,CAAC,IAAI;oBACjC,GAAG,EAAE,QAAQ;oBACb,OAAO,EAAE,EAAE;oBACX,QAAQ,EAAE,KAAK;oBACf,eAAe,EAAE,KAAK;oBACtB,IAAI,EAAE,IAAI;iBACX,CAAC;gBACF,MAAM,KAAK,GAAG,IAAI,CAAC,QAAQ,CAAC,CAAC;gBAC7B,MAAM,WAAW,GAAG;oBAClB,GAAG,EAAE,SAAS;oBACd,QAAQ,EAAE,UAAU,CAAC,SAAS,CAAC;oBAC/B,QAAQ,EAAE,UAAU,CAAC,SAAS,CAAC;iBAChC,CAAC;gBACF,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;gBAChC,IAAI,CAAC,QAAQ,CAAC,GAAG,mBAAmB,CAAC,KAAK,CAAC,CAAC;gBAC5C,OAAO,IAAI,CAAC;YACd,CAAC,EAAE,EAAqB,CAAC,CAC1B,CAAC;QACJ,CAAC,CAAC,CAAC;QAEL,MAAM,CAAC,GAAG,EAAE;YACV,IAAI,IAAI,CAAC,eAAe,EAAE,EAAE,MAAM,EAAE,CAAC;gBACnC,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,eAAe,EAAE,CAAC,CAAC;YACnD,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAEO,kBAAkB,CAAC,KAAa;QACtC,MAAM,OAAO,GAAG,EAAE,GAAG,IAAI,CAAC,OAAO,EAAE,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE,KAAK,EAAE,CAAC;QAC1D,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;IAC5B,CAAC;IAES,UAAU,CAAC,KAA6B,EAAE,KAAa;QAC/D,MAAM,OAAO,GAAG,KAAK,CAAC,OAAO,CAAC,KAAK,EAAE,CAAC;QACtC,eAAe,CAAC,OAAO,EAAE,KAAK,CAAC,aAAa,EAAE,KAAK,CAAC,YAAY,CAAC,CAAC;QAClE,MAAM,YAAY,GAAG,EAAE,GAAG,KAAK,EAAE,OAAO,EAAE,CAAC;QAC3C,IAAI,CAAC,kBAAkB,CAAC,YAAY,CAAC,CAAC;IACxC,CAAC;IAES,SAAS,CAAC,WAAoB;QACtC,yBAAyB;QACzB,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;QAExC,kBAAkB;QAClB,MAAM,OAAO,GAAG,MAAM,CAAC,WAAW,CAChC,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,KAAK,CAAC,EAAE,EAAE,CAAC;YACnD,GAAG;YACH,0BAA0B,CAAC;gBACzB,GAAG,KAAK;gBACR,QAAQ,EAAE,WAAW;aACZ,CAAC;SACb,CAAC,CACgB,CAAC;QACrB,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;IAC5B,CAAC;IAES,WAAW,CAAC,QAAiB,EAAE,KAAa;QACpD,MAAM,YAAY,GAAG,0BAA0B,CAAC,EAAE,GAAG,KAAK,EAAE,QAAQ,EAAE,CAAC,CAAC;QACxE,IAAI,CAAC,kBAAkB,CAAC,YAAY,CAAC,CAAC;IACxC,CAAC;IAES,YAAY,CAAC,QAAiB,EAAE,KAAa,EAAE,MAAe;QACtE,MAAM,YAAY,GAAG,mBAAmB,CAAC;YACvC,GAAG,KAAK;YACR,OAAO,EAAE,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,QAAQ,EAAE,CAAC,CAAC,GAAG,KAAK,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC;SACpG,CAAC,CAAC;QACH,IAAI,CAAC,kBAAkB,CAAC,YAAY,CAAC,CAAC;IACxC,CAAC;IAED,cAAc;IAEJ,iBAAiB,CAAC,KAAa;QACvC,OAAO,IAAI,CAAC,kBAAkB,EAAE,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;IACnD,CAAC;IAES,eAAe,CAAC,QAAiB,EAAE,KAAa;QACxD,oBAAoB;QACpB,MAAM,kBAAkB,GAAG,IAAI,CAAC,kBAAkB,EAAE,CAAC;QACrD,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,kBAAkB,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,KAAK,KAAK,IAAI,QAAQ,CAAC,CAAC,CAAC;QAE7F,MAAM,OAAO,GAAG,MAAM,CAAC,WAAW,CAChC,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,KAAK,CAAC,EAAE,EAAE,CAAC;YACnD,GAAG;YACH,mBAAmB,CAAC;gBAClB,GAAG,KAAK;gBACR,OAAO,EAAE,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;oBACpC,GAAG,MAAM;oBACT,QAAQ,EAAE,WAAW,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,MAAM,CAAC,QAAQ;iBAClE,CAAC,CAAC;aACM,CAAC;SACb,CAAC,CACgB,CAAC;QACrB,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;IAC5B,CAAC;8GA1JU,6BAA6B;kGAA7B,6BAA6B,khBCxF1C,6nIAgHA,mZDlCI,WAAW,ghBACX,WAAW,oNACX,iBAAiB,gEACjB,eAAe,kPACf,eAAe,8DACf,eAAe,gGACf,WAAW,+dACX,OAAO;;2FAGE,6BAA6B;kBAjBzC,SAAS;+BACE,4BAA4B,mBAGrB,uBAAuB,CAAC,MAAM,cACnC,IAAI,WACP;wBACP,WAAW;wBACX,WAAW;wBACX,iBAAiB;wBACjB,eAAe;wBACf,eAAe;wBACf,eAAe;wBACf,WAAW;wBACX,OAAO;qBACR","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component,\n  DestroyRef,\n  computed,\n  effect,\n  inject,\n  input,\n  output,\n  signal\n} from '@angular/core';\nimport { takeUntilDestroyed, toObservable } from '@angular/core/rxjs-interop';\nimport { FormsModule } from '@angular/forms';\nimport { CdkDragDrop, moveItemInArray, CdkDropList, CdkDrag } from '@angular/cdk/drag-drop';\nimport { isDefaultCSVSelected, isCSVIncluded } from '@hestia-earth/json-schema/schema-utils';\nimport { FaIconComponent } from '@fortawesome/angular-fontawesome';\nimport { NgbDropdown, NgbDropdownToggle, NgbDropdownMenu, NgbDropdownItem } from '@ng-bootstrap/ng-bootstrap';\nimport { filter, mergeMap, tap } from 'rxjs/operators';\nimport { isUndefined } from '@hestia-earth/utils';\nimport { combineLatest, forkJoin, of } from 'rxjs';\n\nimport { HeSchemaService } from '../../schema/schema.service';\nimport { faAngleDown, faAngleLeft, faFilter, faSpinner } from '@fortawesome/free-solid-svg-icons';\n\ninterface IHeader {\n  key: string;\n  selected: boolean;\n  /**\n   * `false` if it is an `internal` field.\n   */\n  included: boolean;\n}\n\ninterface IGroup {\n  key: string;\n  selected: boolean;\n  partialSelected: boolean;\n  open: boolean;\n  headers: IHeader[];\n}\n\ninterface IGroupedHeaders {\n  [group: string]: IGroup;\n}\n\nconst headerGroup = (header: string) => {\n  const parts = header.split('.');\n  return parts.length === 2 ? '' : [parts[0], parts[1]].join('.');\n};\n\nconst termFields = ['@id', 'name', 'units', 'termType'];\n\nconst isTermField = (field: string, { key }: IHeader) =>\n  ['term', 'methodModel', 'country', 'product', 'operation', 'key'].some(parent =>\n    key.endsWith([parent, field].join('.'))\n  );\n\nconst updateGroupHeadersSelected = ({ headers, ...group }: IGroup) => ({\n  ...group,\n  headers: headers.map(header => ({ ...header, selected: group.selected }))\n});\n\nconst updateGroupSelected = (group: IGroup) => {\n  const allSelected = group.headers.every(({ selected }) => selected);\n  return {\n    ...group,\n    selected: allSelected,\n    partialSelected: !allSelected && group.headers.some(({ selected }) => selected)\n  };\n};\n\n@Component({\n  selector: 'he-node-csv-select-headers',\n  templateUrl: './node-csv-select-headers.component.html',\n  styleUrls: ['./node-csv-select-headers.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [\n    FormsModule,\n    NgbDropdown,\n    NgbDropdownToggle,\n    FaIconComponent,\n    NgbDropdownMenu,\n    NgbDropdownItem,\n    CdkDropList,\n    CdkDrag\n  ]\n})\nexport class NodeCsvSelectHeadersComponent {\n  private destroyRef = inject(DestroyRef);\n  private schemaService = inject(HeSchemaService);\n\n  protected readonly faSpinner = faSpinner;\n  protected readonly faFilter = faFilter;\n  protected readonly faAngleDown = faAngleDown;\n  protected readonly faAngleLeft = faAngleLeft;\n\n  protected csv = input.required<string>();\n  private csv$ = toObservable(this.csv);\n  protected keys = input.required<string[]>();\n  private keys$ = toObservable(this.keys);\n  protected includeDefaultCSV = input(false);\n\n  protected headersChanged = output<string[]>();\n\n  protected loading = signal(true);\n  protected termFields = termFields;\n  protected selectedTermFields = signal(['@id']);\n  protected showNonIncluded = signal(false);\n\n  protected allSelected = computed(() => Object.keys(this.headers()).every(key => this.headers()[key].selected));\n  private selectedHeaders = computed(() =>\n    Object.values(this.headers())\n      .flatMap(({ headers }) => headers)\n      .filter(({ selected, included }) => selected && (included || this.showNonIncluded()))\n      .map(({ key }) => key)\n  );\n\n  private schemas$ = this.schemaService.schemas$.pipe(takeUntilDestroyed(this.destroyRef));\n\n  private csvHeaders = signal<string[]>([]);\n  protected headers = signal<IGroupedHeaders>({});\n  protected groups = computed(() => Object.values(this.headers()));\n\n  constructor() {\n    combineLatest([this.csv$, this.keys$, this.schemas$])\n      .pipe(\n        filter(([csv, keys, schemas]) => [!isUndefined(csv), keys?.length > 0, !isUndefined(schemas)].every(Boolean)),\n        tap(() => this.loading.set(true)),\n        mergeMap(([csv, keys, schemas]) =>\n          forkJoin({\n            headers: this.schemaService.parseHeaders$(csv),\n            keys: of(keys),\n            schemas: of(schemas)\n          })\n        ),\n        tap(() => this.loading.set(false))\n      )\n      .subscribe(({ headers, keys, schemas }) => {\n        this.csvHeaders.set(headers);\n\n        const isIncluded = isCSVIncluded(schemas);\n        const isDefaultSelected = this.includeDefaultCSV() ? () => true : isDefaultCSVSelected(schemas);\n        const isSelected = (key: string) =>\n          (!keys.length || keys.some(v => key.startsWith(v))) && isDefaultSelected(key);\n\n        this.headers.set(\n          headers.reduce((prev, headerKey) => {\n            const groupKey = headerGroup(headerKey);\n            prev[groupKey] = prev[groupKey] || {\n              key: groupKey,\n              headers: [],\n              selected: false,\n              partialSelected: false,\n              open: true\n            };\n            const group = prev[groupKey];\n            const groupHeader = {\n              key: headerKey,\n              selected: isSelected(headerKey),\n              included: isIncluded(headerKey)\n            };\n            group.headers.push(groupHeader);\n            prev[groupKey] = updateGroupSelected(group);\n            return prev;\n          }, {} as IGroupedHeaders)\n        );\n      });\n\n    effect(() => {\n      if (this.selectedHeaders()?.length) {\n        this.headersChanged.emit(this.selectedHeaders());\n      }\n    });\n  }\n\n  private updateHeadersGroup(group: IGroup) {\n    const headers = { ...this.headers(), [group.key]: group };\n    this.headers.set(headers);\n  }\n\n  protected dropHeader(event: CdkDragDrop<IHeader[]>, group: IGroup) {\n    const headers = group.headers.slice();\n    moveItemInArray(headers, event.previousIndex, event.currentIndex);\n    const updatedGroup = { ...group, headers };\n    this.updateHeadersGroup(updatedGroup);\n  }\n\n  protected selectAll(allSelected: boolean) {\n    // select all term fields\n    this.selectedTermFields.set(termFields);\n\n    // select all keys\n    const headers = Object.fromEntries(\n      Object.entries(this.headers()).map(([key, group]) => [\n        key,\n        updateGroupHeadersSelected({\n          ...group,\n          selected: allSelected\n        } as IGroup)\n      ])\n    ) as IGroupedHeaders;\n    this.headers.set(headers);\n  }\n\n  protected updateGroup(selected: boolean, group: IGroup) {\n    const updatedGroup = updateGroupHeadersSelected({ ...group, selected });\n    this.updateHeadersGroup(updatedGroup);\n  }\n\n  protected updateHeader(selected: boolean, group: IGroup, header: IHeader) {\n    const updatedGroup = updateGroupSelected({\n      ...group,\n      headers: group.headers.map(v => ({ ...v, selected: v.key === header.key ? selected : v.selected }))\n    });\n    this.updateHeadersGroup(updatedGroup);\n  }\n\n  // Term fields\n\n  protected isTermFieldSelect(field: string) {\n    return this.selectedTermFields().includes(field);\n  }\n\n  protected toggleTermField(selected: boolean, field: string) {\n    // update term field\n    const selectedTermFields = this.selectedTermFields();\n    this.selectedTermFields.set(selectedTermFields.filter(value => value !== field || selected));\n\n    const headers = Object.fromEntries(\n      Object.entries(this.headers()).map(([key, group]) => [\n        key,\n        updateGroupSelected({\n          ...group,\n          headers: group.headers.map(header => ({\n            ...header,\n            selected: isTermField(field, header) ? selected : header.selected\n          }))\n        } as IGroup)\n      ])\n    ) as IGroupedHeaders;\n    this.headers.set(headers);\n  }\n}\n","<p class=\"mb-2\">Please select which columns you would like to include:</p>\n\n<p class=\"my-2 is-size-7\">\n  <i>You can drag and drop the headers to sort them as they would appear in the CSV file.</i>\n</p>\n\n<div class=\"columns toggle-all mx-4 pb-1 mb-0\">\n  <div class=\"column\">\n    <label class=\"checkbox ml-1\">\n      <input type=\"checkbox\" class=\"selector\" [ngModel]=\"allSelected()\" (change)=\"selectAll($event.target.checked)\" />\n      <span class=\"ml-2\">Toggle All</span>\n    </label>\n  </div>\n\n  <div class=\"column is-narrow has-text-right\">\n    <div ngbDropdown container=\"body\" placement=\"bottom-end\">\n      <button ngbDropdownToggle class=\"button is-small\" type=\"button\" id=\"select-menu\">\n        <span>Advanced Filters</span>\n        <span class=\"icon is-small\">\n          <fa-icon [icon]=\"faFilter\" aria-hidden=\"true\"></fa-icon>\n        </span>\n      </button>\n      <div ngbDropdownMenu aria-labelledby=\"select-menu\">\n        <div class=\"dropdown-content\">\n          <a ngbDropdownItem>\n            <label class=\"checkbox ml-1\">\n              <input\n                type=\"checkbox\"\n                class=\"selector\"\n                [ngModel]=\"showNonIncluded()\"\n                (change)=\"showNonIncluded.set($event.target.checked)\" />\n              <span class=\"ml-2\">\n                Include\n                <code>internal</code>\n                fields\n              </span>\n            </label>\n          </a>\n          <div class=\"dropdown-item\">\n            <p>Toggle Term Fields</p>\n          </div>\n          @for (field of termFields; track field) {\n            <div ngbDropdownItem>\n              <label class=\"checkbox ml-1\">\n                <input\n                  type=\"checkbox\"\n                  class=\"selector\"\n                  [checked]=\"isTermFieldSelect(field)\"\n                  (change)=\"toggleTermField($event.target.checked, field)\" />\n                <span class=\"ml-2\">{{ field }}</span>\n              </label>\n            </div>\n          }\n        </div>\n      </div>\n    </div>\n  </div>\n</div>\n\n@if (loading()) {\n  <div class=\"has-text-center py-3\">\n    <fa-icon [icon]=\"faSpinner\" [pulse]=\"true\" size=\"lg\"></fa-icon>\n  </div>\n} @else {\n  <div class=\"drag-container\">\n    @for (group of groups(); track group.key) {\n      <div class=\"card\">\n        <header class=\"card-header\">\n          <div class=\"card-header-title\">\n            <label class=\"checkbox ml-1\">\n              <input\n                type=\"checkbox\"\n                class=\"selector\"\n                [indeterminate]=\"group.partialSelected\"\n                [ngModel]=\"group.selected\"\n                (change)=\"updateGroup($event.target.checked, group)\" />\n              <span class=\"ml-2\">{{ group.key }}</span>\n            </label>\n          </div>\n          <span\n            class=\"card-header-icon has-text-secondary\"\n            aria-label=\"open / close\"\n            (click)=\"group.open = !group.open\"\n            pointer>\n            <span class=\"icon\">\n              <fa-icon [icon]=\"faAngleDown\" [class.is-hidden]=\"!group.open\"></fa-icon>\n              <fa-icon [icon]=\"faAngleLeft\" [class.is-hidden]=\"group.open\"></fa-icon>\n            </span>\n          </span>\n        </header>\n        <div class=\"card-content p-3\" [class.is-hidden]=\"!group.open\">\n          <ul cdkDropList (cdkDropListDropped)=\"dropHeader($event, group)\">\n            @for (header of group.headers; track header.key) {\n              @if (header.included || showNonIncluded()) {\n                <li cdkDrag>\n                  <label class=\"checkbox ml-2\">\n                    <input\n                      type=\"checkbox\"\n                      class=\"selector\"\n                      [ngModel]=\"header.selected\"\n                      (change)=\"updateHeader($event.target.checked, group, header)\" />\n                    <span class=\"ml-2\">{{ header.key }}</span>\n                  </label>\n                </li>\n              }\n            }\n          </ul>\n        </div>\n      </div>\n    }\n  </div>\n}\n"]}
|
|
167
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node-csv-select-headers.component.js","sourceRoot":"","sources":["../../../../src/node/node-csv-select-headers/node-csv-select-headers.component.ts","../../../../src/node/node-csv-select-headers/node-csv-select-headers.component.html"],"names":[],"mappings":"AAAA,OAAO,EACL,uBAAuB,EACvB,SAAS,EACT,UAAU,EACV,QAAQ,EACR,MAAM,EACN,MAAM,EACN,KAAK,EACL,MAAM,EACN,MAAM,EACP,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,kBAAkB,EAAE,YAAY,EAAE,MAAM,4BAA4B,CAAC;AAC9E,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAC7C,OAAO,EAAe,eAAe,EAAE,WAAW,EAAE,OAAO,EAAE,MAAM,wBAAwB,CAAC;AAC5F,OAAO,EAAE,oBAAoB,EAAE,aAAa,EAAE,MAAM,wCAAwC,CAAC;AAC7F,OAAO,EAAE,eAAe,EAAE,MAAM,kCAAkC,CAAC;AACnE,OAAO,EAAE,WAAW,EAAE,iBAAiB,EAAE,eAAe,EAAE,eAAe,EAAE,MAAM,4BAA4B,CAAC;AAC9G,OAAO,EAAE,MAAM,EAAE,QAAQ,EAAE,GAAG,EAAE,MAAM,gBAAgB,CAAC;AACvD,OAAO,EAAE,WAAW,EAAE,MAAM,qBAAqB,CAAC;AAClD,OAAO,EAAE,aAAa,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,MAAM,CAAC;AAEnD,OAAO,EAAE,eAAe,EAAE,MAAM,6BAA6B,CAAC;AAC9D,OAAO,EAAE,WAAW,EAAE,WAAW,EAAE,QAAQ,EAAE,SAAS,EAAE,MAAM,mCAAmC,CAAC;;;AAuBlG,MAAM,WAAW,GAAG,CAAC,MAAc,EAAE,EAAE;IACrC,MAAM,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IAChC,OAAO,KAAK,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;AAClE,CAAC,CAAC;AAEF,MAAM,UAAU,GAAG,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,EAAE,UAAU,CAAC,CAAC;AAExD,MAAM,WAAW,GAAG,CAAC,KAAa,EAAE,EAAE,GAAG,EAAW,EAAE,EAAE,CACtD,CAAC,MAAM,EAAE,aAAa,EAAE,SAAS,EAAE,SAAS,EAAE,WAAW,EAAE,KAAK,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAC9E,GAAG,CAAC,QAAQ,CAAC,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CACxC,CAAC;AAEJ,MAAM,0BAA0B,GAAG,CAAC,EAAE,OAAO,EAAE,GAAG,KAAK,EAAU,EAAE,EAAE,CAAC,CAAC;IACrE,GAAG,KAAK;IACR,OAAO,EAAE,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,EAAE,GAAG,MAAM,EAAE,QAAQ,EAAE,KAAK,CAAC,QAAQ,EAAE,CAAC,CAAC;CAC1E,CAAC,CAAC;AAEH,MAAM,mBAAmB,GAAG,CAAC,KAAa,EAAE,EAAE;IAC5C,MAAM,WAAW,GAAG,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,QAAQ,EAAE,EAAE,EAAE,CAAC,QAAQ,CAAC,CAAC;IACpE,OAAO;QACL,GAAG,KAAK;QACR,QAAQ,EAAE,WAAW;QACrB,eAAe,EAAE,CAAC,WAAW,IAAI,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,EAAE,QAAQ,EAAE,EAAE,EAAE,CAAC,QAAQ,CAAC;KAChF,CAAC;AACJ,CAAC,CAAC;AAmBF,MAAM,OAAO,6BAA6B;IAoCxC;QAnCQ,eAAU,GAAG,MAAM,CAAC,UAAU,CAAC,CAAC;QAChC,kBAAa,GAAG,MAAM,CAAC,eAAe,CAAC,CAAC;QAE7B,cAAS,GAAG,SAAS,CAAC;QACtB,aAAQ,GAAG,QAAQ,CAAC;QACpB,gBAAW,GAAG,WAAW,CAAC;QAC1B,gBAAW,GAAG,WAAW,CAAC;QAEnC,QAAG,GAAG,KAAK,CAAC,QAAQ,EAAU,CAAC;QACjC,SAAI,GAAG,YAAY,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAC5B,SAAI,GAAG,KAAK,CAAW,EAAE,CAAC,CAAC;QAC7B,UAAK,GAAG,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC9B,sBAAiB,GAAG,KAAK,CAAC,KAAK,CAAC,CAAC;QAEjC,mBAAc,GAAG,MAAM,EAAY,CAAC;QAEpC,YAAO,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC;QACvB,eAAU,GAAG,UAAU,CAAC;QACxB,uBAAkB,GAAG,MAAM,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC;QACrC,oBAAe,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;QAEhC,gBAAW,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,GAAG,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC;QACvG,oBAAe,GAAG,QAAQ,CAAC,GAAG,EAAE,CACtC,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC;aAC1B,OAAO,CAAC,CAAC,EAAE,OAAO,EAAE,EAAE,EAAE,CAAC,OAAO,CAAC;aACjC,MAAM,CAAC,CAAC,EAAE,QAAQ,EAAE,QAAQ,EAAE,EAAE,EAAE,CAAC,QAAQ,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,eAAe,EAAE,CAAC,CAAC;aACpF,GAAG,CAAC,CAAC,EAAE,GAAG,EAAE,EAAE,EAAE,CAAC,GAAG,CAAC,CACzB,CAAC;QAEM,aAAQ,GAAG,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC;QAEjF,eAAU,GAAG,MAAM,CAAW,EAAE,CAAC,CAAC;QAChC,YAAO,GAAG,MAAM,CAAkB,EAAE,CAAC,CAAC;QACtC,WAAM,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC;QAG/D,aAAa,CAAC,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,KAAK,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;aAClD,IAAI,CACH,MAAM,CAAC,CAAC,CAAC,GAAG,EAAE,KAAK,EAAE,OAAO,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,CAAC,GAAG,CAAC,EAAE,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,EAC5F,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,EACjC,QAAQ,CAAC,CAAC,CAAC,GAAG,EAAE,IAAI,EAAE,OAAO,CAAC,EAAE,EAAE,CAChC,QAAQ,CAAC;YACP,OAAO,EAAE,IAAI,CAAC,aAAa,CAAC,aAAa,CAAC,GAAG,CAAC;YAC9C,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC;YACd,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC;SACrB,CAAC,CACH,EACD,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,CACnC;aACA,SAAS,CAAC,CAAC,EAAE,OAAO,EAAE,IAAI,EAAE,OAAO,EAAE,EAAE,EAAE;YACxC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;YAE7B,MAAM,UAAU,GAAG,aAAa,CAAC,OAAO,CAAC,CAAC;YAC1C,MAAM,iBAAiB,GAAG,IAAI,CAAC,iBAAiB,EAAE,CAAC,CAAC,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC,oBAAoB,CAAC,OAAO,CAAC,CAAC;YAChG,MAAM,UAAU,GAAG,CAAC,GAAW,EAAE,EAAE,CACjC,CAAC,CAAC,IAAI,CAAC,MAAM,IAAI,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,iBAAiB,CAAC,GAAG,CAAC,CAAC;YAEhF,IAAI,CAAC,OAAO,CAAC,GAAG,CACd,OAAO,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,SAAS,EAAE,EAAE;gBACjC,MAAM,QAAQ,GAAG,WAAW,CAAC,SAAS,CAAC,CAAC;gBACxC,IAAI,CAAC,QAAQ,CAAC,GAAG,IAAI,CAAC,QAAQ,CAAC,IAAI;oBACjC,GAAG,EAAE,QAAQ;oBACb,OAAO,EAAE,EAAE;oBACX,QAAQ,EAAE,KAAK;oBACf,eAAe,EAAE,KAAK;oBACtB,IAAI,EAAE,IAAI;iBACX,CAAC;gBACF,MAAM,KAAK,GAAG,IAAI,CAAC,QAAQ,CAAC,CAAC;gBAC7B,MAAM,WAAW,GAAG;oBAClB,GAAG,EAAE,SAAS;oBACd,QAAQ,EAAE,UAAU,CAAC,SAAS,CAAC;oBAC/B,QAAQ,EAAE,UAAU,CAAC,SAAS,CAAC;iBAChC,CAAC;gBACF,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;gBAChC,IAAI,CAAC,QAAQ,CAAC,GAAG,mBAAmB,CAAC,KAAK,CAAC,CAAC;gBAC5C,OAAO,IAAI,CAAC;YACd,CAAC,EAAE,EAAqB,CAAC,CAC1B,CAAC;QACJ,CAAC,CAAC,CAAC;QAEL,MAAM,CAAC,GAAG,EAAE;YACV,IAAI,IAAI,CAAC,eAAe,EAAE,EAAE,MAAM,EAAE,CAAC;gBACnC,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,eAAe,EAAE,CAAC,CAAC;YACnD,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAEO,kBAAkB,CAAC,KAAa;QACtC,MAAM,OAAO,GAAG,EAAE,GAAG,IAAI,CAAC,OAAO,EAAE,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE,KAAK,EAAE,CAAC;QAC1D,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;IAC5B,CAAC;IAES,UAAU,CAAC,KAA6B,EAAE,KAAa;QAC/D,MAAM,OAAO,GAAG,KAAK,CAAC,OAAO,CAAC,KAAK,EAAE,CAAC;QACtC,eAAe,CAAC,OAAO,EAAE,KAAK,CAAC,aAAa,EAAE,KAAK,CAAC,YAAY,CAAC,CAAC;QAClE,MAAM,YAAY,GAAG,EAAE,GAAG,KAAK,EAAE,OAAO,EAAE,CAAC;QAC3C,IAAI,CAAC,kBAAkB,CAAC,YAAY,CAAC,CAAC;IACxC,CAAC;IAES,SAAS,CAAC,WAAoB;QACtC,yBAAyB;QACzB,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;QAExC,kBAAkB;QAClB,MAAM,OAAO,GAAG,MAAM,CAAC,WAAW,CAChC,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,KAAK,CAAC,EAAE,EAAE,CAAC;YACnD,GAAG;YACH,0BAA0B,CAAC;gBACzB,GAAG,KAAK;gBACR,QAAQ,EAAE,WAAW;aACZ,CAAC;SACb,CAAC,CACgB,CAAC;QACrB,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;IAC5B,CAAC;IAES,WAAW,CAAC,QAAiB,EAAE,KAAa;QACpD,MAAM,YAAY,GAAG,0BAA0B,CAAC,EAAE,GAAG,KAAK,EAAE,QAAQ,EAAE,CAAC,CAAC;QACxE,IAAI,CAAC,kBAAkB,CAAC,YAAY,CAAC,CAAC;IACxC,CAAC;IAES,YAAY,CAAC,QAAiB,EAAE,KAAa,EAAE,MAAe;QACtE,MAAM,YAAY,GAAG,mBAAmB,CAAC;YACvC,GAAG,KAAK;YACR,OAAO,EAAE,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,QAAQ,EAAE,CAAC,CAAC,GAAG,KAAK,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC;SACpG,CAAC,CAAC;QACH,IAAI,CAAC,kBAAkB,CAAC,YAAY,CAAC,CAAC;IACxC,CAAC;IAED,cAAc;IAEJ,iBAAiB,CAAC,KAAa;QACvC,OAAO,IAAI,CAAC,kBAAkB,EAAE,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;IACnD,CAAC;IAES,eAAe,CAAC,QAAiB,EAAE,KAAa;QACxD,oBAAoB;QACpB,MAAM,kBAAkB,GAAG,IAAI,CAAC,kBAAkB,EAAE,CAAC;QACrD,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,kBAAkB,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,KAAK,KAAK,IAAI,QAAQ,CAAC,CAAC,CAAC;QAE7F,MAAM,OAAO,GAAG,MAAM,CAAC,WAAW,CAChC,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,KAAK,CAAC,EAAE,EAAE,CAAC;YACnD,GAAG;YACH,mBAAmB,CAAC;gBAClB,GAAG,KAAK;gBACR,OAAO,EAAE,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;oBACpC,GAAG,MAAM;oBACT,QAAQ,EAAE,WAAW,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,MAAM,CAAC,QAAQ;iBAClE,CAAC,CAAC;aACM,CAAC;SACb,CAAC,CACgB,CAAC;QACrB,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;IAC5B,CAAC;8GA1JU,6BAA6B;kGAA7B,6BAA6B,mhBCxF1C,6nIAgHA,mZDlCI,WAAW,ghBACX,WAAW,oNACX,iBAAiB,gEACjB,eAAe,kPACf,eAAe,8DACf,eAAe,gGACf,WAAW,+dACX,OAAO;;2FAGE,6BAA6B;kBAjBzC,SAAS;+BACE,4BAA4B,mBAGrB,uBAAuB,CAAC,MAAM,cACnC,IAAI,WACP;wBACP,WAAW;wBACX,WAAW;wBACX,iBAAiB;wBACjB,eAAe;wBACf,eAAe;wBACf,eAAe;wBACf,WAAW;wBACX,OAAO;qBACR","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component,\n  DestroyRef,\n  computed,\n  effect,\n  inject,\n  input,\n  output,\n  signal\n} from '@angular/core';\nimport { takeUntilDestroyed, toObservable } from '@angular/core/rxjs-interop';\nimport { FormsModule } from '@angular/forms';\nimport { CdkDragDrop, moveItemInArray, CdkDropList, CdkDrag } from '@angular/cdk/drag-drop';\nimport { isDefaultCSVSelected, isCSVIncluded } from '@hestia-earth/json-schema/schema-utils';\nimport { FaIconComponent } from '@fortawesome/angular-fontawesome';\nimport { NgbDropdown, NgbDropdownToggle, NgbDropdownMenu, NgbDropdownItem } from '@ng-bootstrap/ng-bootstrap';\nimport { filter, mergeMap, tap } from 'rxjs/operators';\nimport { isUndefined } from '@hestia-earth/utils';\nimport { combineLatest, forkJoin, of } from 'rxjs';\n\nimport { HeSchemaService } from '../../schema/schema.service';\nimport { faAngleDown, faAngleLeft, faFilter, faSpinner } from '@fortawesome/free-solid-svg-icons';\n\ninterface IHeader {\n  key: string;\n  selected: boolean;\n  /**\n   * `false` if it is an `internal` field.\n   */\n  included: boolean;\n}\n\ninterface IGroup {\n  key: string;\n  selected: boolean;\n  partialSelected: boolean;\n  open: boolean;\n  headers: IHeader[];\n}\n\ninterface IGroupedHeaders {\n  [group: string]: IGroup;\n}\n\nconst headerGroup = (header: string) => {\n  const parts = header.split('.');\n  return parts.length === 2 ? '' : [parts[0], parts[1]].join('.');\n};\n\nconst termFields = ['@id', 'name', 'units', 'termType'];\n\nconst isTermField = (field: string, { key }: IHeader) =>\n  ['term', 'methodModel', 'country', 'product', 'operation', 'key'].some(parent =>\n    key.endsWith([parent, field].join('.'))\n  );\n\nconst updateGroupHeadersSelected = ({ headers, ...group }: IGroup) => ({\n  ...group,\n  headers: headers.map(header => ({ ...header, selected: group.selected }))\n});\n\nconst updateGroupSelected = (group: IGroup) => {\n  const allSelected = group.headers.every(({ selected }) => selected);\n  return {\n    ...group,\n    selected: allSelected,\n    partialSelected: !allSelected && group.headers.some(({ selected }) => selected)\n  };\n};\n\n@Component({\n  selector: 'he-node-csv-select-headers',\n  templateUrl: './node-csv-select-headers.component.html',\n  styleUrls: ['./node-csv-select-headers.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [\n    FormsModule,\n    NgbDropdown,\n    NgbDropdownToggle,\n    FaIconComponent,\n    NgbDropdownMenu,\n    NgbDropdownItem,\n    CdkDropList,\n    CdkDrag\n  ]\n})\nexport class NodeCsvSelectHeadersComponent {\n  private destroyRef = inject(DestroyRef);\n  private schemaService = inject(HeSchemaService);\n\n  protected readonly faSpinner = faSpinner;\n  protected readonly faFilter = faFilter;\n  protected readonly faAngleDown = faAngleDown;\n  protected readonly faAngleLeft = faAngleLeft;\n\n  protected csv = input.required<string>();\n  private csv$ = toObservable(this.csv);\n  protected keys = input<string[]>([]);\n  private keys$ = toObservable(this.keys);\n  protected includeDefaultCSV = input(false);\n\n  protected headersChanged = output<string[]>();\n\n  protected loading = signal(true);\n  protected termFields = termFields;\n  protected selectedTermFields = signal(['@id']);\n  protected showNonIncluded = signal(false);\n\n  protected allSelected = computed(() => Object.keys(this.headers()).every(key => this.headers()[key].selected));\n  private selectedHeaders = computed(() =>\n    Object.values(this.headers())\n      .flatMap(({ headers }) => headers)\n      .filter(({ selected, included }) => selected && (included || this.showNonIncluded()))\n      .map(({ key }) => key)\n  );\n\n  private schemas$ = this.schemaService.schemas$.pipe(takeUntilDestroyed(this.destroyRef));\n\n  private csvHeaders = signal<string[]>([]);\n  protected headers = signal<IGroupedHeaders>({});\n  protected groups = computed(() => Object.values(this.headers()));\n\n  constructor() {\n    combineLatest([this.csv$, this.keys$, this.schemas$])\n      .pipe(\n        filter(([csv, _keys, schemas]) => [!isUndefined(csv), !isUndefined(schemas)].every(Boolean)),\n        tap(() => this.loading.set(true)),\n        mergeMap(([csv, keys, schemas]) =>\n          forkJoin({\n            headers: this.schemaService.parseHeaders$(csv),\n            keys: of(keys),\n            schemas: of(schemas)\n          })\n        ),\n        tap(() => this.loading.set(false))\n      )\n      .subscribe(({ headers, keys, schemas }) => {\n        this.csvHeaders.set(headers);\n\n        const isIncluded = isCSVIncluded(schemas);\n        const isDefaultSelected = this.includeDefaultCSV() ? () => true : isDefaultCSVSelected(schemas);\n        const isSelected = (key: string) =>\n          (!keys.length || keys.some(v => key.startsWith(v))) && isDefaultSelected(key);\n\n        this.headers.set(\n          headers.reduce((prev, headerKey) => {\n            const groupKey = headerGroup(headerKey);\n            prev[groupKey] = prev[groupKey] || {\n              key: groupKey,\n              headers: [],\n              selected: false,\n              partialSelected: false,\n              open: true\n            };\n            const group = prev[groupKey];\n            const groupHeader = {\n              key: headerKey,\n              selected: isSelected(headerKey),\n              included: isIncluded(headerKey)\n            };\n            group.headers.push(groupHeader);\n            prev[groupKey] = updateGroupSelected(group);\n            return prev;\n          }, {} as IGroupedHeaders)\n        );\n      });\n\n    effect(() => {\n      if (this.selectedHeaders()?.length) {\n        this.headersChanged.emit(this.selectedHeaders());\n      }\n    });\n  }\n\n  private updateHeadersGroup(group: IGroup) {\n    const headers = { ...this.headers(), [group.key]: group };\n    this.headers.set(headers);\n  }\n\n  protected dropHeader(event: CdkDragDrop<IHeader[]>, group: IGroup) {\n    const headers = group.headers.slice();\n    moveItemInArray(headers, event.previousIndex, event.currentIndex);\n    const updatedGroup = { ...group, headers };\n    this.updateHeadersGroup(updatedGroup);\n  }\n\n  protected selectAll(allSelected: boolean) {\n    // select all term fields\n    this.selectedTermFields.set(termFields);\n\n    // select all keys\n    const headers = Object.fromEntries(\n      Object.entries(this.headers()).map(([key, group]) => [\n        key,\n        updateGroupHeadersSelected({\n          ...group,\n          selected: allSelected\n        } as IGroup)\n      ])\n    ) as IGroupedHeaders;\n    this.headers.set(headers);\n  }\n\n  protected updateGroup(selected: boolean, group: IGroup) {\n    const updatedGroup = updateGroupHeadersSelected({ ...group, selected });\n    this.updateHeadersGroup(updatedGroup);\n  }\n\n  protected updateHeader(selected: boolean, group: IGroup, header: IHeader) {\n    const updatedGroup = updateGroupSelected({\n      ...group,\n      headers: group.headers.map(v => ({ ...v, selected: v.key === header.key ? selected : v.selected }))\n    });\n    this.updateHeadersGroup(updatedGroup);\n  }\n\n  // Term fields\n\n  protected isTermFieldSelect(field: string) {\n    return this.selectedTermFields().includes(field);\n  }\n\n  protected toggleTermField(selected: boolean, field: string) {\n    // update term field\n    const selectedTermFields = this.selectedTermFields();\n    this.selectedTermFields.set(selectedTermFields.filter(value => value !== field || selected));\n\n    const headers = Object.fromEntries(\n      Object.entries(this.headers()).map(([key, group]) => [\n        key,\n        updateGroupSelected({\n          ...group,\n          headers: group.headers.map(header => ({\n            ...header,\n            selected: isTermField(field, header) ? selected : header.selected\n          }))\n        } as IGroup)\n      ])\n    ) as IGroupedHeaders;\n    this.headers.set(headers);\n  }\n}\n","<p class=\"mb-2\">Please select which columns you would like to include:</p>\n\n<p class=\"my-2 is-size-7\">\n  <i>You can drag and drop the headers to sort them as they would appear in the CSV file.</i>\n</p>\n\n<div class=\"columns toggle-all mx-4 pb-1 mb-0\">\n  <div class=\"column\">\n    <label class=\"checkbox ml-1\">\n      <input type=\"checkbox\" class=\"selector\" [ngModel]=\"allSelected()\" (change)=\"selectAll($event.target.checked)\" />\n      <span class=\"ml-2\">Toggle All</span>\n    </label>\n  </div>\n\n  <div class=\"column is-narrow has-text-right\">\n    <div ngbDropdown container=\"body\" placement=\"bottom-end\">\n      <button ngbDropdownToggle class=\"button is-small\" type=\"button\" id=\"select-menu\">\n        <span>Advanced Filters</span>\n        <span class=\"icon is-small\">\n          <fa-icon [icon]=\"faFilter\" aria-hidden=\"true\"></fa-icon>\n        </span>\n      </button>\n      <div ngbDropdownMenu aria-labelledby=\"select-menu\">\n        <div class=\"dropdown-content\">\n          <a ngbDropdownItem>\n            <label class=\"checkbox ml-1\">\n              <input\n                type=\"checkbox\"\n                class=\"selector\"\n                [ngModel]=\"showNonIncluded()\"\n                (change)=\"showNonIncluded.set($event.target.checked)\" />\n              <span class=\"ml-2\">\n                Include\n                <code>internal</code>\n                fields\n              </span>\n            </label>\n          </a>\n          <div class=\"dropdown-item\">\n            <p>Toggle Term Fields</p>\n          </div>\n          @for (field of termFields; track field) {\n            <div ngbDropdownItem>\n              <label class=\"checkbox ml-1\">\n                <input\n                  type=\"checkbox\"\n                  class=\"selector\"\n                  [checked]=\"isTermFieldSelect(field)\"\n                  (change)=\"toggleTermField($event.target.checked, field)\" />\n                <span class=\"ml-2\">{{ field }}</span>\n              </label>\n            </div>\n          }\n        </div>\n      </div>\n    </div>\n  </div>\n</div>\n\n@if (loading()) {\n  <div class=\"has-text-center py-3\">\n    <fa-icon [icon]=\"faSpinner\" [pulse]=\"true\" size=\"lg\"></fa-icon>\n  </div>\n} @else {\n  <div class=\"drag-container\">\n    @for (group of groups(); track group.key) {\n      <div class=\"card\">\n        <header class=\"card-header\">\n          <div class=\"card-header-title\">\n            <label class=\"checkbox ml-1\">\n              <input\n                type=\"checkbox\"\n                class=\"selector\"\n                [indeterminate]=\"group.partialSelected\"\n                [ngModel]=\"group.selected\"\n                (change)=\"updateGroup($event.target.checked, group)\" />\n              <span class=\"ml-2\">{{ group.key }}</span>\n            </label>\n          </div>\n          <span\n            class=\"card-header-icon has-text-secondary\"\n            aria-label=\"open / close\"\n            (click)=\"group.open = !group.open\"\n            pointer>\n            <span class=\"icon\">\n              <fa-icon [icon]=\"faAngleDown\" [class.is-hidden]=\"!group.open\"></fa-icon>\n              <fa-icon [icon]=\"faAngleLeft\" [class.is-hidden]=\"group.open\"></fa-icon>\n            </span>\n          </span>\n        </header>\n        <div class=\"card-content p-3\" [class.is-hidden]=\"!group.open\">\n          <ul cdkDropList (cdkDropListDropped)=\"dropHeader($event, group)\">\n            @for (header of group.headers; track header.key) {\n              @if (header.included || showNonIncluded()) {\n                <li cdkDrag>\n                  <label class=\"checkbox ml-2\">\n                    <input\n                      type=\"checkbox\"\n                      class=\"selector\"\n                      [ngModel]=\"header.selected\"\n                      (change)=\"updateHeader($event.target.checked, group, header)\" />\n                    <span class=\"ml-2\">{{ header.key }}</span>\n                  </label>\n                </li>\n              }\n            }\n          </ul>\n        </div>\n      </div>\n    }\n  </div>\n}\n"]}
|