geonetwork-ui 2.6.0-dev.502fa026d → 2.6.0-dev.b306f1194
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/libs/common/domain/src/lib/model/record/metadata.model.mjs +1 -1
- package/esm2022/libs/feature/search/src/lib/favorites/favorite-star/favorite-star.component.mjs +3 -3
- package/esm2022/libs/ui/elements/src/index.mjs +2 -1
- package/esm2022/libs/ui/elements/src/lib/api-card/api-card.component.mjs +27 -6
- package/esm2022/libs/ui/elements/src/lib/download-item/download-item.component.mjs +22 -6
- package/esm2022/libs/ui/elements/src/lib/downloads-list/downloads-list.component.mjs +3 -3
- package/esm2022/libs/ui/elements/src/lib/internal-link-card/internal-link-card.component.mjs +163 -0
- package/esm2022/libs/ui/elements/src/lib/link-card/link-card.component.mjs +29 -7
- package/esm2022/libs/ui/elements/src/lib/metadata-catalog/metadata-catalog.component.mjs +3 -3
- package/esm2022/libs/ui/elements/src/lib/metadata-quality/metadata-quality.component.mjs +3 -3
- package/esm2022/libs/ui/elements/src/lib/record-api-form/record-api-form.component.mjs +3 -2
- package/esm2022/libs/ui/elements/src/lib/ui-elements.module.mjs +10 -4
- package/esm2022/libs/ui/layout/src/lib/carousel/carousel.component.mjs +3 -3
- package/esm2022/libs/ui/widgets/src/lib/progress-bar/progress-bar.component.mjs +3 -3
- package/esm2022/libs/util/shared/src/lib/links/link-utils.mjs +21 -19
- package/esm2022/translations/de.json +7 -0
- package/esm2022/translations/en.json +8 -1
- package/esm2022/translations/es.json +7 -0
- package/esm2022/translations/fr.json +7 -0
- package/esm2022/translations/it.json +7 -1
- package/esm2022/translations/nl.json +7 -0
- package/esm2022/translations/pt.json +7 -0
- package/fesm2022/geonetwork-ui.mjs +315 -55
- package/fesm2022/geonetwork-ui.mjs.map +1 -1
- package/libs/common/domain/src/lib/model/record/metadata.model.d.ts +1 -0
- package/libs/common/domain/src/lib/model/record/metadata.model.d.ts.map +1 -1
- package/libs/ui/elements/src/index.d.ts +1 -0
- package/libs/ui/elements/src/index.d.ts.map +1 -1
- package/libs/ui/elements/src/lib/api-card/api-card.component.d.ts +9 -1
- package/libs/ui/elements/src/lib/api-card/api-card.component.d.ts.map +1 -1
- package/libs/ui/elements/src/lib/download-item/download-item.component.d.ts +8 -1
- package/libs/ui/elements/src/lib/download-item/download-item.component.d.ts.map +1 -1
- package/libs/ui/elements/src/lib/internal-link-card/internal-link-card.component.d.ts +43 -0
- package/libs/ui/elements/src/lib/internal-link-card/internal-link-card.component.d.ts.map +1 -0
- package/libs/ui/elements/src/lib/link-card/link-card.component.d.ts +10 -2
- package/libs/ui/elements/src/lib/link-card/link-card.component.d.ts.map +1 -1
- package/libs/ui/elements/src/lib/record-api-form/record-api-form.component.d.ts.map +1 -1
- package/libs/ui/elements/src/lib/ui-elements.module.d.ts +2 -1
- package/libs/ui/elements/src/lib/ui-elements.module.d.ts.map +1 -1
- package/libs/util/shared/src/lib/links/link-utils.d.ts +16 -16
- package/libs/util/shared/src/lib/links/link-utils.d.ts.map +1 -1
- package/package.json +1 -1
- package/src/libs/common/domain/src/lib/model/record/metadata.model.ts +1 -0
- package/src/libs/feature/search/src/lib/favorites/favorite-star/favorite-star.component.html +1 -1
- package/src/libs/ui/elements/src/index.ts +1 -0
- package/src/libs/ui/elements/src/lib/api-card/api-card.component.html +64 -38
- package/src/libs/ui/elements/src/lib/api-card/api-card.component.ts +26 -2
- package/src/libs/ui/elements/src/lib/download-item/download-item.component.html +17 -17
- package/src/libs/ui/elements/src/lib/download-item/download-item.component.ts +20 -2
- package/src/libs/ui/elements/src/lib/downloads-list/downloads-list.component.html +7 -6
- package/src/libs/ui/elements/src/lib/internal-link-card/internal-link-card.component.css +0 -0
- package/src/libs/ui/elements/src/lib/internal-link-card/internal-link-card.component.html +156 -0
- package/src/libs/ui/elements/src/lib/internal-link-card/internal-link-card.component.ts +190 -0
- package/src/libs/ui/elements/src/lib/link-card/link-card.component.html +27 -29
- package/src/libs/ui/elements/src/lib/link-card/link-card.component.ts +33 -3
- package/src/libs/ui/elements/src/lib/metadata-catalog/metadata-catalog.component.html +1 -1
- package/src/libs/ui/elements/src/lib/metadata-quality/metadata-quality.component.html +4 -2
- package/src/libs/ui/elements/src/lib/record-api-form/record-api-form.component.ts +2 -1
- package/src/libs/ui/elements/src/lib/ui-elements.module.ts +3 -0
- package/src/libs/ui/layout/src/lib/carousel/carousel.component.css +0 -4
- package/src/libs/ui/widgets/src/lib/progress-bar/progress-bar.component.html +2 -2
- package/src/libs/util/shared/src/lib/links/link-utils.ts +20 -18
- package/tailwind.base.css +34 -1
- package/translations/de.json +7 -0
- package/translations/en.json +8 -1
- package/translations/es.json +7 -0
- package/translations/fr.json +7 -0
- package/translations/it.json +7 -1
- package/translations/nl.json +7 -0
- package/translations/pt.json +7 -0
- package/translations/sk.json +7 -0
|
@@ -0,0 +1,163 @@
|
|
|
1
|
+
import { Component, Input, TemplateRef, Output, EventEmitter, ElementRef, } from '@angular/core';
|
|
2
|
+
import { NgClass, NgIf, NgTemplateOutlet } from '@angular/common';
|
|
3
|
+
import { MarkdownParserComponent } from '../markdown-parser/markdown-parser.component';
|
|
4
|
+
import { MetadataQualityComponent } from '../metadata-quality/metadata-quality.component';
|
|
5
|
+
import { ThumbnailComponent } from '../thumbnail/thumbnail.component';
|
|
6
|
+
import { propagateToDocumentOnly, removeWhitespace, stripHtml, } from '../../../../../../libs/util/shared/src';
|
|
7
|
+
import { NgIconComponent, provideIcons, provideNgIconsConfig, } from '@ng-icons/core';
|
|
8
|
+
import { matLocationSearchingOutline, matEmailOutline, matPhoneOutline, matLocationOnOutline, } from '@ng-icons/material-icons/outline';
|
|
9
|
+
import { matCode } from '@ng-icons/material-icons/baseline';
|
|
10
|
+
import { iconoirDatabase, iconoirMap, iconoirInternet } from '@ng-icons/iconoir';
|
|
11
|
+
import { TranslateModule } from '@ngx-translate/core';
|
|
12
|
+
import { marker } from '@biesbjerg/ngx-translate-extract-marker';
|
|
13
|
+
import { fromEvent, Subscription } from 'rxjs';
|
|
14
|
+
import * as i0 from "@angular/core";
|
|
15
|
+
import * as i1 from "@ngx-translate/core";
|
|
16
|
+
marker('record.kind.dataset');
|
|
17
|
+
marker('record.kind.reuse');
|
|
18
|
+
marker('record.kind.service');
|
|
19
|
+
export class InternalLinkCardComponent {
|
|
20
|
+
set size(value) {
|
|
21
|
+
this._size = value;
|
|
22
|
+
this.cardClass = this.sizeClassMap[value] || '';
|
|
23
|
+
this.thumbnailContainerClass = this.thumbnailSizeClassMap[value] || 'hidden';
|
|
24
|
+
}
|
|
25
|
+
get size() {
|
|
26
|
+
return this._size;
|
|
27
|
+
}
|
|
28
|
+
constructor(elementRef) {
|
|
29
|
+
this.elementRef = elementRef;
|
|
30
|
+
this.linkHref = null;
|
|
31
|
+
this.mdSelect = new EventEmitter();
|
|
32
|
+
this.subscription = new Subscription();
|
|
33
|
+
this.cardClass = '';
|
|
34
|
+
this.thumbnailContainerClass = '';
|
|
35
|
+
this._size = 'M';
|
|
36
|
+
this.sizeClassMap = {
|
|
37
|
+
L: 'min-h-[190px] md:w-[992px] py-3 px-3 flex items-start gap-5',
|
|
38
|
+
M: 'min-h-[140px] md:w-[570px] py-3 px-3 flex items-start gap-4',
|
|
39
|
+
S: 'min-h-[220px] md:w-[370px] py-3 px-3 flex gap-4',
|
|
40
|
+
XS: 'min-h-[108px] md:w-[570px] py-3 px-3 flex gap-4',
|
|
41
|
+
};
|
|
42
|
+
this.thumbnailSizeClassMap = {
|
|
43
|
+
L: 'w-[190px] h-[180px] rounded-lg overflow-hidden shrink-0',
|
|
44
|
+
M: 'w-[110px] h-[140px] rounded-lg overflow-hidden shrink-0',
|
|
45
|
+
S: 'hidden',
|
|
46
|
+
XS: 'hidden',
|
|
47
|
+
};
|
|
48
|
+
this.titleClassMap = {
|
|
49
|
+
L: 'text-xl line-clamp-2',
|
|
50
|
+
M: 'text-base line-clamp-2',
|
|
51
|
+
S: 'text-base line-clamp-3',
|
|
52
|
+
XS: 'text-base mt-3 line-clamp-2',
|
|
53
|
+
};
|
|
54
|
+
}
|
|
55
|
+
ngOnInit() {
|
|
56
|
+
this.abstract = removeWhitespace(stripHtml(this.record?.abstract));
|
|
57
|
+
this.subscription.add(fromEvent(this.elementRef.nativeElement, 'click').subscribe((event) => {
|
|
58
|
+
event.preventDefault();
|
|
59
|
+
propagateToDocumentOnly(event);
|
|
60
|
+
this.mdSelect.emit(this.record);
|
|
61
|
+
}));
|
|
62
|
+
}
|
|
63
|
+
get organization() {
|
|
64
|
+
return this.record.ownerOrganization;
|
|
65
|
+
}
|
|
66
|
+
get contacts() {
|
|
67
|
+
return ((this.record.kind === 'dataset'
|
|
68
|
+
? this.record.contactsForResource
|
|
69
|
+
: this.record.contacts) || []);
|
|
70
|
+
}
|
|
71
|
+
getTitleClass() {
|
|
72
|
+
return (this.titleClassMap[this._size] +
|
|
73
|
+
' ' +
|
|
74
|
+
(this.record.ownerOrganization?.name ? '' : 'mt-3') || '');
|
|
75
|
+
}
|
|
76
|
+
openExternalUrl(event, url) {
|
|
77
|
+
event.stopPropagation();
|
|
78
|
+
window.open(url, '_blank');
|
|
79
|
+
}
|
|
80
|
+
openMailto(event, email) {
|
|
81
|
+
event.stopPropagation();
|
|
82
|
+
window.open(`mailto:${email}`, '_blank');
|
|
83
|
+
}
|
|
84
|
+
copyToClipboard(event, text) {
|
|
85
|
+
event.stopPropagation();
|
|
86
|
+
navigator.clipboard.writeText(text);
|
|
87
|
+
}
|
|
88
|
+
get shouldShowThumbnail() {
|
|
89
|
+
return this.size === 'L' || this.size === 'M';
|
|
90
|
+
}
|
|
91
|
+
getKindInfo() {
|
|
92
|
+
if (!this.record?.kind)
|
|
93
|
+
return { text: '', icon: '' };
|
|
94
|
+
switch (this.record.kind.toLowerCase()) {
|
|
95
|
+
case 'dataset':
|
|
96
|
+
return { text: 'record.kind.dataset', icon: 'iconoirDatabase' };
|
|
97
|
+
case 'reuse':
|
|
98
|
+
return { text: 'record.kind.reuse', icon: 'iconoirMap' };
|
|
99
|
+
case 'service':
|
|
100
|
+
return { text: 'record.kind.service', icon: 'matCode' };
|
|
101
|
+
default:
|
|
102
|
+
return { text: '', icon: '' };
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: InternalLinkCardComponent, deps: [{ token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
106
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: InternalLinkCardComponent, isStandalone: true, selector: "gn-ui-internal-link-card", inputs: { record: "record", metadataQualityDisplay: "metadataQualityDisplay", favoriteTemplate: "favoriteTemplate", linkHref: "linkHref", isGeodata: "isGeodata", size: "size" }, outputs: { mdSelect: "mdSelect" }, providers: [
|
|
107
|
+
provideIcons({
|
|
108
|
+
matLocationSearchingOutline,
|
|
109
|
+
matCode,
|
|
110
|
+
iconoirDatabase,
|
|
111
|
+
iconoirMap,
|
|
112
|
+
iconoirInternet,
|
|
113
|
+
matEmailOutline,
|
|
114
|
+
matPhoneOutline,
|
|
115
|
+
matLocationOnOutline,
|
|
116
|
+
}),
|
|
117
|
+
provideNgIconsConfig({
|
|
118
|
+
size: '1.2em',
|
|
119
|
+
}),
|
|
120
|
+
], ngImport: i0, template: "<div\n class=\"rounded-lg group card-shadow cursor-pointer overflow-hidden hover:bg-gray-50\"\n [ngClass]=\"cardClass\"\n>\n <div class=\"flex flex-row justify-between w-full\">\n <div\n *ngIf=\"shouldShowThumbnail\"\n [ngClass]=\"thumbnailContainerClass\"\n class=\"mr-4 flex flex-col\"\n >\n <gn-ui-thumbnail\n class=\"w-full h-full object-cover\"\n [thumbnailUrl]=\"record.overviews?.[0]?.url?.toString() || ''\"\n [fit]=\"'cover'\"\n ></gn-ui-thumbnail>\n </div>\n <div\n class=\"flex flex-col flex-1 relative\"\n [ngClass]=\"{\n 'justify-between': record.ownerOrganization?.name && size !== 'XS',\n }\"\n >\n <div class=\"flex items-center space-x-2\">\n <span\n *ngIf=\"getKindInfo().text\"\n class=\"badge-btn text-white text-xs px-2 py-0.5 font-bold shrink-0 bg-primary leading-tight flex items-center justify-evenly h-6 min-h-6\"\n >\n <ng-icon\n class=\"text-[0.9em] text-white mr-1\"\n [name]=\"getKindInfo().icon\"\n ></ng-icon>\n <span class=\"font-medium text-white text-xs\" translate>\n {{ getKindInfo().text }}\n </span>\n </span>\n <span\n *ngIf=\"isGeodata\"\n class=\"badge-btn text-black text-xs px-2 py-0.5 font-bold shrink-0 bg-primary-white leading-tight flex items-center justify-evenly h-6 min-h-6\"\n [ngClass]=\"size === 'L' ? 'w-[164px]' : 'w-8'\"\n >\n <ng-icon\n class=\"text-[0.9em] text-primary-darkest\"\n name=\"matLocationSearchingOutline\"\n ></ng-icon>\n <ng-container *ngIf=\"size === 'L'\">\n <span\n class=\"font-medium text-primary-darkest text-xs ml-1\"\n translate\n >\n record.metadata.type\n </span>\n </ng-container>\n </span>\n <div class=\"flex items-center\">\n <gn-ui-metadata-quality\n [smaller]=\"true\"\n [metadata]=\"record\"\n [metadataQualityDisplay]=\"metadataQualityDisplay\"\n ></gn-ui-metadata-quality>\n </div>\n <div class=\"absolute top-0 right-0 items-center\">\n <ng-container\n *ngIf=\"size !== 'XS'\"\n [ngTemplateOutlet]=\"favoriteTemplate\"\n [ngTemplateOutletContext]=\"{ $implicit: record }\"\n ></ng-container>\n </div>\n </div>\n <div\n class=\"font-medium text-title group-hover:text-primary overflow-hidden break-words\"\n [ngClass]=\"getTitleClass()\"\n >\n {{ record.title }}\n </div>\n <div\n *ngIf=\"size === 'L'\"\n class=\"mt-1 mb-2 font-normal text-xs text-gray-900 line-clamp-2 overflow-hidden\"\n >\n <gn-ui-markdown-parser\n [textContent]=\"abstract\"\n [whitoutStyles]=\"true\"\n ></gn-ui-markdown-parser>\n </div>\n <div\n *ngIf=\"size !== 'XS' && record.ownerOrganization?.name\"\n class=\"flex items-center justify-evenly bg-gray-50 rounded-lg h-[53px] px-2\"\n >\n <div class=\"flex items-center flex-1 min-w-0\">\n <div\n class=\"w-[45px] h-[45px] rounded-lg overflow-hidden shrink-0 mr-3\"\n >\n <gn-ui-thumbnail\n [thumbnailUrl]=\"\n record.ownerOrganization?.logoUrl?.toString() || ''\n \"\n [fit]=\"'contain'\"\n class=\"w-full h-full rounded-lg\"\n ></gn-ui-thumbnail>\n </div>\n <div *ngIf=\"organization?.name\" class=\"flex-1 w-0 overflow-hidden\">\n <div\n class=\"text-xs text-black font-normal leading-tight truncate\"\n translate\n >\n record.card.metadata.contact\n </div>\n <div class=\"text-xl text-primary-black font-medium truncate\">\n {{ organization.name }}\n </div>\n </div>\n </div>\n <div *ngIf=\"size === 'L'\" class=\"ml-2 flex space-x-2\">\n <div *ngIf=\"organization?.website\" class=\"flex\">\n <button\n [title]=\"organization.website\"\n class=\"w-[40px] h-[32px] flex items-center justify-center rounded-lg border border-[#D4D3D7] px-[8px] py-[4px] hover:bg-primary-lightest\"\n (click)=\"openExternalUrl($event, organization.website)\"\n >\n <ng-icon name=\"iconoirInternet\"></ng-icon>\n </button>\n </div>\n <div *ngIf=\"contacts[0]?.email\" class=\"flex\">\n <button\n [title]=\"contacts[0].email\"\n class=\"w-[40px] h-[32px] flex items-center justify-center rounded-lg border border-[#D4D3D7] px-[8px] py-[4px] hover:bg-primary-lightest\"\n data-cy=\"contact-email\"\n (click)=\"openMailto($event, contacts[0].email)\"\n >\n <ng-icon name=\"matEmailOutline\"></ng-icon>\n </button>\n </div>\n <div *ngIf=\"contacts[0]?.phone\" class=\"flex\">\n <button\n [title]=\"'Copy to clipboard'\"\n class=\"w-[40px] h-[32px] flex items-center justify-center rounded-lg border border-[#D4D3D7] px-[8px] py-[4px] hover:bg-primary-lightest relative group\"\n data-cy=\"contact-phone\"\n (click)=\"copyToClipboard($event, contacts[0].phone)\"\n >\n <ng-icon name=\"matPhoneOutline\"></ng-icon>\n </button>\n </div>\n <div *ngIf=\"contacts[0]?.address\" class=\"flex\">\n <button\n [title]=\"'Copy to clipboard'\"\n class=\"w-[40px] h-[32px] flex items-center justify-center rounded-lg border border-[#D4D3D7] px-[8px] py-[4px] hover:bg-primary-lightest relative group\"\n data-cy=\"contact-phone\"\n (click)=\"copyToClipboard($event, contacts[0].address)\"\n >\n <ng-icon name=\"matLocationOnOutline\"></ng-icon>\n </button>\n </div>\n </div>\n </div>\n </div>\n </div>\n</div>\n", styles: [""], dependencies: [{ kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: ThumbnailComponent, selector: "gn-ui-thumbnail", inputs: ["thumbnailUrl", "fit"], outputs: ["placeholderShown"] }, { kind: "component", type: MetadataQualityComponent, selector: "gn-ui-metadata-quality", inputs: ["metadata", "smaller", "metadataQualityDisplay"] }, { kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: NgIconComponent, selector: "ng-icon", inputs: ["name", "svg", "size", "strokeWidth", "color"] }, { kind: "ngmodule", type: TranslateModule }, { kind: "directive", type: i1.TranslateDirective, selector: "[translate],[ngx-translate]", inputs: ["translate", "translateParams"] }, { kind: "component", type: MarkdownParserComponent, selector: "gn-ui-markdown-parser", inputs: ["textContent", "whitoutStyles"] }] }); }
|
|
121
|
+
}
|
|
122
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: InternalLinkCardComponent, decorators: [{
|
|
123
|
+
type: Component,
|
|
124
|
+
args: [{ selector: 'gn-ui-internal-link-card', standalone: true, imports: [
|
|
125
|
+
NgClass,
|
|
126
|
+
NgIf,
|
|
127
|
+
ThumbnailComponent,
|
|
128
|
+
MetadataQualityComponent,
|
|
129
|
+
NgTemplateOutlet,
|
|
130
|
+
NgIconComponent,
|
|
131
|
+
TranslateModule,
|
|
132
|
+
MarkdownParserComponent,
|
|
133
|
+
], providers: [
|
|
134
|
+
provideIcons({
|
|
135
|
+
matLocationSearchingOutline,
|
|
136
|
+
matCode,
|
|
137
|
+
iconoirDatabase,
|
|
138
|
+
iconoirMap,
|
|
139
|
+
iconoirInternet,
|
|
140
|
+
matEmailOutline,
|
|
141
|
+
matPhoneOutline,
|
|
142
|
+
matLocationOnOutline,
|
|
143
|
+
}),
|
|
144
|
+
provideNgIconsConfig({
|
|
145
|
+
size: '1.2em',
|
|
146
|
+
}),
|
|
147
|
+
], template: "<div\n class=\"rounded-lg group card-shadow cursor-pointer overflow-hidden hover:bg-gray-50\"\n [ngClass]=\"cardClass\"\n>\n <div class=\"flex flex-row justify-between w-full\">\n <div\n *ngIf=\"shouldShowThumbnail\"\n [ngClass]=\"thumbnailContainerClass\"\n class=\"mr-4 flex flex-col\"\n >\n <gn-ui-thumbnail\n class=\"w-full h-full object-cover\"\n [thumbnailUrl]=\"record.overviews?.[0]?.url?.toString() || ''\"\n [fit]=\"'cover'\"\n ></gn-ui-thumbnail>\n </div>\n <div\n class=\"flex flex-col flex-1 relative\"\n [ngClass]=\"{\n 'justify-between': record.ownerOrganization?.name && size !== 'XS',\n }\"\n >\n <div class=\"flex items-center space-x-2\">\n <span\n *ngIf=\"getKindInfo().text\"\n class=\"badge-btn text-white text-xs px-2 py-0.5 font-bold shrink-0 bg-primary leading-tight flex items-center justify-evenly h-6 min-h-6\"\n >\n <ng-icon\n class=\"text-[0.9em] text-white mr-1\"\n [name]=\"getKindInfo().icon\"\n ></ng-icon>\n <span class=\"font-medium text-white text-xs\" translate>\n {{ getKindInfo().text }}\n </span>\n </span>\n <span\n *ngIf=\"isGeodata\"\n class=\"badge-btn text-black text-xs px-2 py-0.5 font-bold shrink-0 bg-primary-white leading-tight flex items-center justify-evenly h-6 min-h-6\"\n [ngClass]=\"size === 'L' ? 'w-[164px]' : 'w-8'\"\n >\n <ng-icon\n class=\"text-[0.9em] text-primary-darkest\"\n name=\"matLocationSearchingOutline\"\n ></ng-icon>\n <ng-container *ngIf=\"size === 'L'\">\n <span\n class=\"font-medium text-primary-darkest text-xs ml-1\"\n translate\n >\n record.metadata.type\n </span>\n </ng-container>\n </span>\n <div class=\"flex items-center\">\n <gn-ui-metadata-quality\n [smaller]=\"true\"\n [metadata]=\"record\"\n [metadataQualityDisplay]=\"metadataQualityDisplay\"\n ></gn-ui-metadata-quality>\n </div>\n <div class=\"absolute top-0 right-0 items-center\">\n <ng-container\n *ngIf=\"size !== 'XS'\"\n [ngTemplateOutlet]=\"favoriteTemplate\"\n [ngTemplateOutletContext]=\"{ $implicit: record }\"\n ></ng-container>\n </div>\n </div>\n <div\n class=\"font-medium text-title group-hover:text-primary overflow-hidden break-words\"\n [ngClass]=\"getTitleClass()\"\n >\n {{ record.title }}\n </div>\n <div\n *ngIf=\"size === 'L'\"\n class=\"mt-1 mb-2 font-normal text-xs text-gray-900 line-clamp-2 overflow-hidden\"\n >\n <gn-ui-markdown-parser\n [textContent]=\"abstract\"\n [whitoutStyles]=\"true\"\n ></gn-ui-markdown-parser>\n </div>\n <div\n *ngIf=\"size !== 'XS' && record.ownerOrganization?.name\"\n class=\"flex items-center justify-evenly bg-gray-50 rounded-lg h-[53px] px-2\"\n >\n <div class=\"flex items-center flex-1 min-w-0\">\n <div\n class=\"w-[45px] h-[45px] rounded-lg overflow-hidden shrink-0 mr-3\"\n >\n <gn-ui-thumbnail\n [thumbnailUrl]=\"\n record.ownerOrganization?.logoUrl?.toString() || ''\n \"\n [fit]=\"'contain'\"\n class=\"w-full h-full rounded-lg\"\n ></gn-ui-thumbnail>\n </div>\n <div *ngIf=\"organization?.name\" class=\"flex-1 w-0 overflow-hidden\">\n <div\n class=\"text-xs text-black font-normal leading-tight truncate\"\n translate\n >\n record.card.metadata.contact\n </div>\n <div class=\"text-xl text-primary-black font-medium truncate\">\n {{ organization.name }}\n </div>\n </div>\n </div>\n <div *ngIf=\"size === 'L'\" class=\"ml-2 flex space-x-2\">\n <div *ngIf=\"organization?.website\" class=\"flex\">\n <button\n [title]=\"organization.website\"\n class=\"w-[40px] h-[32px] flex items-center justify-center rounded-lg border border-[#D4D3D7] px-[8px] py-[4px] hover:bg-primary-lightest\"\n (click)=\"openExternalUrl($event, organization.website)\"\n >\n <ng-icon name=\"iconoirInternet\"></ng-icon>\n </button>\n </div>\n <div *ngIf=\"contacts[0]?.email\" class=\"flex\">\n <button\n [title]=\"contacts[0].email\"\n class=\"w-[40px] h-[32px] flex items-center justify-center rounded-lg border border-[#D4D3D7] px-[8px] py-[4px] hover:bg-primary-lightest\"\n data-cy=\"contact-email\"\n (click)=\"openMailto($event, contacts[0].email)\"\n >\n <ng-icon name=\"matEmailOutline\"></ng-icon>\n </button>\n </div>\n <div *ngIf=\"contacts[0]?.phone\" class=\"flex\">\n <button\n [title]=\"'Copy to clipboard'\"\n class=\"w-[40px] h-[32px] flex items-center justify-center rounded-lg border border-[#D4D3D7] px-[8px] py-[4px] hover:bg-primary-lightest relative group\"\n data-cy=\"contact-phone\"\n (click)=\"copyToClipboard($event, contacts[0].phone)\"\n >\n <ng-icon name=\"matPhoneOutline\"></ng-icon>\n </button>\n </div>\n <div *ngIf=\"contacts[0]?.address\" class=\"flex\">\n <button\n [title]=\"'Copy to clipboard'\"\n class=\"w-[40px] h-[32px] flex items-center justify-center rounded-lg border border-[#D4D3D7] px-[8px] py-[4px] hover:bg-primary-lightest relative group\"\n data-cy=\"contact-phone\"\n (click)=\"copyToClipboard($event, contacts[0].address)\"\n >\n <ng-icon name=\"matLocationOnOutline\"></ng-icon>\n </button>\n </div>\n </div>\n </div>\n </div>\n </div>\n</div>\n" }]
|
|
148
|
+
}], ctorParameters: () => [{ type: i0.ElementRef }], propDecorators: { record: [{
|
|
149
|
+
type: Input
|
|
150
|
+
}], metadataQualityDisplay: [{
|
|
151
|
+
type: Input
|
|
152
|
+
}], favoriteTemplate: [{
|
|
153
|
+
type: Input
|
|
154
|
+
}], linkHref: [{
|
|
155
|
+
type: Input
|
|
156
|
+
}], isGeodata: [{
|
|
157
|
+
type: Input
|
|
158
|
+
}], size: [{
|
|
159
|
+
type: Input
|
|
160
|
+
}], mdSelect: [{
|
|
161
|
+
type: Output
|
|
162
|
+
}] } });
|
|
163
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"internal-link-card.component.js","sourceRoot":"","sources":["../../../../../../../../libs/ui/elements/src/lib/internal-link-card/internal-link-card.component.ts","../../../../../../../../libs/ui/elements/src/lib/internal-link-card/internal-link-card.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  TemplateRef,\n  OnInit,\n  Output,\n  EventEmitter,\n  ElementRef,\n} from '@angular/core'\nimport {\n  CatalogRecord,\n  Organization,\n} from '../../../../../../libs/common/domain/src/lib/model/record'\nimport { NgClass, NgIf, NgTemplateOutlet } from '@angular/common'\nimport { MarkdownParserComponent } from '../markdown-parser/markdown-parser.component'\nimport { MetadataQualityComponent } from '../metadata-quality/metadata-quality.component'\nimport { ThumbnailComponent } from '../thumbnail/thumbnail.component'\nimport {\n  propagateToDocumentOnly,\n  removeWhitespace,\n  stripHtml,\n} from '../../../../../../libs/util/shared/src'\nimport {\n  NgIconComponent,\n  provideIcons,\n  provideNgIconsConfig,\n} from '@ng-icons/core'\nimport {\n  matLocationSearchingOutline,\n  matEmailOutline,\n  matPhoneOutline,\n  matLocationOnOutline,\n} from '@ng-icons/material-icons/outline'\nimport { matCode } from '@ng-icons/material-icons/baseline'\nimport { iconoirDatabase, iconoirMap, iconoirInternet } from '@ng-icons/iconoir'\nimport { TranslateModule } from '@ngx-translate/core'\nimport { marker } from '@biesbjerg/ngx-translate-extract-marker'\nimport { fromEvent, Subscription } from 'rxjs'\n\nmarker('record.kind.dataset')\nmarker('record.kind.reuse')\nmarker('record.kind.service')\n\ntype CardSize = 'L' | 'M' | 'S' | 'XS'\n\n@Component({\n  selector: 'gn-ui-internal-link-card',\n  standalone: true,\n  imports: [\n    NgClass,\n    NgIf,\n    ThumbnailComponent,\n    MetadataQualityComponent,\n    NgTemplateOutlet,\n    NgIconComponent,\n    TranslateModule,\n    MarkdownParserComponent,\n  ],\n  providers: [\n    provideIcons({\n      matLocationSearchingOutline,\n      matCode,\n      iconoirDatabase,\n      iconoirMap,\n      iconoirInternet,\n      matEmailOutline,\n      matPhoneOutline,\n      matLocationOnOutline,\n    }),\n    provideNgIconsConfig({\n      size: '1.2em',\n    }),\n  ],\n  templateUrl: './internal-link-card.component.html',\n  styleUrls: ['./internal-link-card.component.css'],\n})\nexport class InternalLinkCardComponent implements OnInit {\n  @Input() record: CatalogRecord\n  @Input() metadataQualityDisplay: boolean\n  @Input() favoriteTemplate: TemplateRef<{ $implicit: CatalogRecord }>\n  @Input() linkHref: string = null\n  @Input() isGeodata: boolean\n  @Input() set size(value: CardSize) {\n    this._size = value\n    this.cardClass = this.sizeClassMap[value] || ''\n    this.thumbnailContainerClass = this.thumbnailSizeClassMap[value] || 'hidden'\n  }\n  get size(): CardSize {\n    return this._size\n  }\n  @Output() mdSelect = new EventEmitter<CatalogRecord>()\n  subscription = new Subscription()\n\n  abstract: string\n\n  cardClass = ''\n  thumbnailContainerClass = ''\n\n  private _size: CardSize = 'M'\n\n  private readonly sizeClassMap: Record<CardSize, string> = {\n    L: 'min-h-[190px] md:w-[992px] py-3 px-3 flex items-start gap-5',\n    M: 'min-h-[140px] md:w-[570px] py-3 px-3 flex items-start gap-4',\n    S: 'min-h-[220px] md:w-[370px] py-3 px-3 flex gap-4',\n    XS: 'min-h-[108px] md:w-[570px] py-3 px-3 flex gap-4',\n  }\n\n  private readonly thumbnailSizeClassMap: Record<CardSize, string> = {\n    L: 'w-[190px] h-[180px] rounded-lg overflow-hidden shrink-0',\n    M: 'w-[110px] h-[140px] rounded-lg overflow-hidden shrink-0',\n    S: 'hidden',\n    XS: 'hidden',\n  }\n\n  private readonly titleClassMap: Record<CardSize, string> = {\n    L: 'text-xl line-clamp-2',\n    M: 'text-base line-clamp-2',\n    S: 'text-base line-clamp-3',\n    XS: 'text-base mt-3 line-clamp-2',\n  }\n\n  constructor(protected elementRef: ElementRef) {}\n\n  ngOnInit(): void {\n    this.abstract = removeWhitespace(stripHtml(this.record?.abstract))\n    this.subscription.add(\n      fromEvent(this.elementRef.nativeElement, 'click').subscribe(\n        (event: Event) => {\n          event.preventDefault()\n          propagateToDocumentOnly(event)\n          this.mdSelect.emit(this.record)\n        }\n      )\n    )\n  }\n\n  get organization(): Organization {\n    return this.record.ownerOrganization\n  }\n\n  get contacts() {\n    return (\n      (this.record.kind === 'dataset'\n        ? this.record.contactsForResource\n        : this.record.contacts) || []\n    )\n  }\n\n  getTitleClass() {\n    return (\n      this.titleClassMap[this._size] +\n        ' ' +\n        (this.record.ownerOrganization?.name ? '' : 'mt-3') || ''\n    )\n  }\n\n  openExternalUrl(event: Event, url: URL): void {\n    event.stopPropagation()\n    window.open(url, '_blank')\n  }\n\n  openMailto(event: Event, email: string): void {\n    event.stopPropagation()\n    window.open(`mailto:${email}`, '_blank')\n  }\n\n  copyToClipboard(event: Event, text: string): void {\n    event.stopPropagation()\n    navigator.clipboard.writeText(text)\n  }\n\n  get shouldShowThumbnail(): boolean {\n    return this.size === 'L' || this.size === 'M'\n  }\n\n  getKindInfo(): { text: string; icon: string } {\n    if (!this.record?.kind) return { text: '', icon: '' }\n\n    switch (this.record.kind.toLowerCase()) {\n      case 'dataset':\n        return { text: 'record.kind.dataset', icon: 'iconoirDatabase' }\n      case 'reuse':\n        return { text: 'record.kind.reuse', icon: 'iconoirMap' }\n      case 'service':\n        return { text: 'record.kind.service', icon: 'matCode' }\n      default:\n        return { text: '', icon: '' }\n    }\n  }\n}\n","<div\n  class=\"rounded-lg group card-shadow cursor-pointer overflow-hidden hover:bg-gray-50\"\n  [ngClass]=\"cardClass\"\n>\n  <div class=\"flex flex-row justify-between w-full\">\n    <div\n      *ngIf=\"shouldShowThumbnail\"\n      [ngClass]=\"thumbnailContainerClass\"\n      class=\"mr-4 flex flex-col\"\n    >\n      <gn-ui-thumbnail\n        class=\"w-full h-full object-cover\"\n        [thumbnailUrl]=\"record.overviews?.[0]?.url?.toString() || ''\"\n        [fit]=\"'cover'\"\n      ></gn-ui-thumbnail>\n    </div>\n    <div\n      class=\"flex flex-col flex-1 relative\"\n      [ngClass]=\"{\n        'justify-between': record.ownerOrganization?.name && size !== 'XS',\n      }\"\n    >\n      <div class=\"flex items-center space-x-2\">\n        <span\n          *ngIf=\"getKindInfo().text\"\n          class=\"badge-btn text-white text-xs px-2 py-0.5 font-bold shrink-0 bg-primary leading-tight flex items-center justify-evenly h-6 min-h-6\"\n        >\n          <ng-icon\n            class=\"text-[0.9em] text-white mr-1\"\n            [name]=\"getKindInfo().icon\"\n          ></ng-icon>\n          <span class=\"font-medium text-white text-xs\" translate>\n            {{ getKindInfo().text }}\n          </span>\n        </span>\n        <span\n          *ngIf=\"isGeodata\"\n          class=\"badge-btn text-black text-xs px-2 py-0.5 font-bold shrink-0 bg-primary-white leading-tight flex items-center justify-evenly h-6 min-h-6\"\n          [ngClass]=\"size === 'L' ? 'w-[164px]' : 'w-8'\"\n        >\n          <ng-icon\n            class=\"text-[0.9em] text-primary-darkest\"\n            name=\"matLocationSearchingOutline\"\n          ></ng-icon>\n          <ng-container *ngIf=\"size === 'L'\">\n            <span\n              class=\"font-medium text-primary-darkest text-xs ml-1\"\n              translate\n            >\n              record.metadata.type\n            </span>\n          </ng-container>\n        </span>\n        <div class=\"flex items-center\">\n          <gn-ui-metadata-quality\n            [smaller]=\"true\"\n            [metadata]=\"record\"\n            [metadataQualityDisplay]=\"metadataQualityDisplay\"\n          ></gn-ui-metadata-quality>\n        </div>\n        <div class=\"absolute top-0 right-0 items-center\">\n          <ng-container\n            *ngIf=\"size !== 'XS'\"\n            [ngTemplateOutlet]=\"favoriteTemplate\"\n            [ngTemplateOutletContext]=\"{ $implicit: record }\"\n          ></ng-container>\n        </div>\n      </div>\n      <div\n        class=\"font-medium text-title group-hover:text-primary overflow-hidden break-words\"\n        [ngClass]=\"getTitleClass()\"\n      >\n        {{ record.title }}\n      </div>\n      <div\n        *ngIf=\"size === 'L'\"\n        class=\"mt-1 mb-2 font-normal text-xs text-gray-900 line-clamp-2 overflow-hidden\"\n      >\n        <gn-ui-markdown-parser\n          [textContent]=\"abstract\"\n          [whitoutStyles]=\"true\"\n        ></gn-ui-markdown-parser>\n      </div>\n      <div\n        *ngIf=\"size !== 'XS' && record.ownerOrganization?.name\"\n        class=\"flex items-center justify-evenly bg-gray-50 rounded-lg h-[53px] px-2\"\n      >\n        <div class=\"flex items-center flex-1 min-w-0\">\n          <div\n            class=\"w-[45px] h-[45px] rounded-lg overflow-hidden shrink-0 mr-3\"\n          >\n            <gn-ui-thumbnail\n              [thumbnailUrl]=\"\n                record.ownerOrganization?.logoUrl?.toString() || ''\n              \"\n              [fit]=\"'contain'\"\n              class=\"w-full h-full rounded-lg\"\n            ></gn-ui-thumbnail>\n          </div>\n          <div *ngIf=\"organization?.name\" class=\"flex-1 w-0 overflow-hidden\">\n            <div\n              class=\"text-xs text-black font-normal leading-tight truncate\"\n              translate\n            >\n              record.card.metadata.contact\n            </div>\n            <div class=\"text-xl text-primary-black font-medium truncate\">\n              {{ organization.name }}\n            </div>\n          </div>\n        </div>\n        <div *ngIf=\"size === 'L'\" class=\"ml-2 flex space-x-2\">\n          <div *ngIf=\"organization?.website\" class=\"flex\">\n            <button\n              [title]=\"organization.website\"\n              class=\"w-[40px] h-[32px] flex items-center justify-center rounded-lg border border-[#D4D3D7] px-[8px] py-[4px] hover:bg-primary-lightest\"\n              (click)=\"openExternalUrl($event, organization.website)\"\n            >\n              <ng-icon name=\"iconoirInternet\"></ng-icon>\n            </button>\n          </div>\n          <div *ngIf=\"contacts[0]?.email\" class=\"flex\">\n            <button\n              [title]=\"contacts[0].email\"\n              class=\"w-[40px] h-[32px] flex items-center justify-center rounded-lg border border-[#D4D3D7] px-[8px] py-[4px] hover:bg-primary-lightest\"\n              data-cy=\"contact-email\"\n              (click)=\"openMailto($event, contacts[0].email)\"\n            >\n              <ng-icon name=\"matEmailOutline\"></ng-icon>\n            </button>\n          </div>\n          <div *ngIf=\"contacts[0]?.phone\" class=\"flex\">\n            <button\n              [title]=\"'Copy to clipboard'\"\n              class=\"w-[40px] h-[32px] flex items-center justify-center rounded-lg border border-[#D4D3D7] px-[8px] py-[4px] hover:bg-primary-lightest relative group\"\n              data-cy=\"contact-phone\"\n              (click)=\"copyToClipboard($event, contacts[0].phone)\"\n            >\n              <ng-icon name=\"matPhoneOutline\"></ng-icon>\n            </button>\n          </div>\n          <div *ngIf=\"contacts[0]?.address\" class=\"flex\">\n            <button\n              [title]=\"'Copy to clipboard'\"\n              class=\"w-[40px] h-[32px] flex items-center justify-center rounded-lg border border-[#D4D3D7] px-[8px] py-[4px] hover:bg-primary-lightest relative group\"\n              data-cy=\"contact-phone\"\n              (click)=\"copyToClipboard($event, contacts[0].address)\"\n            >\n              <ng-icon name=\"matLocationOnOutline\"></ng-icon>\n            </button>\n          </div>\n        </div>\n      </div>\n    </div>\n  </div>\n</div>\n"]}
|
|
@@ -2,11 +2,27 @@ import { ChangeDetectionStrategy, Component, Input } from '@angular/core';
|
|
|
2
2
|
import { CommonModule } from '@angular/common';
|
|
3
3
|
import { NgIconComponent, provideIcons, provideNgIconsConfig, } from '@ng-icons/core';
|
|
4
4
|
import { matOpenInNew } from '@ng-icons/material-icons/baseline';
|
|
5
|
+
import { getBadgeColor, getFileFormat } from '../../../../../../libs/util/shared/src';
|
|
6
|
+
import { TranslateModule } from '@ngx-translate/core';
|
|
5
7
|
import * as i0 from "@angular/core";
|
|
6
8
|
import * as i1 from "@angular/common";
|
|
9
|
+
import * as i2 from "@ngx-translate/core";
|
|
7
10
|
export class LinkCardComponent {
|
|
8
11
|
constructor() {
|
|
9
|
-
this.
|
|
12
|
+
this.sizeClassMap = {
|
|
13
|
+
L: 'gn-ui-card-l py-2 px-5',
|
|
14
|
+
M: 'gn-ui-card-m py-2 px-5',
|
|
15
|
+
S: 'gn-ui-card-s p-4',
|
|
16
|
+
XS: 'gn-ui-card-xs py-2 px-5',
|
|
17
|
+
};
|
|
18
|
+
this.cardClass = '';
|
|
19
|
+
}
|
|
20
|
+
set size(value) {
|
|
21
|
+
this._size = value;
|
|
22
|
+
this.cardClass = this.sizeClassMap[value];
|
|
23
|
+
}
|
|
24
|
+
get size() {
|
|
25
|
+
return this._size;
|
|
10
26
|
}
|
|
11
27
|
get title() {
|
|
12
28
|
if (this.link.name && this.link.description) {
|
|
@@ -14,25 +30,31 @@ export class LinkCardComponent {
|
|
|
14
30
|
}
|
|
15
31
|
return this.link.name || this.link.description || '';
|
|
16
32
|
}
|
|
33
|
+
getLinkFormat(link) {
|
|
34
|
+
return getFileFormat(link);
|
|
35
|
+
}
|
|
36
|
+
getLinkColor(link) {
|
|
37
|
+
return getBadgeColor(getFileFormat(link));
|
|
38
|
+
}
|
|
17
39
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: LinkCardComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
18
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: LinkCardComponent, isStandalone: true, selector: "gn-ui-link-card", inputs: { link: "link",
|
|
40
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: LinkCardComponent, isStandalone: true, selector: "gn-ui-link-card", inputs: { link: "link", size: "size" }, providers: [
|
|
19
41
|
provideIcons({
|
|
20
42
|
matOpenInNew,
|
|
21
43
|
}),
|
|
22
44
|
provideNgIconsConfig({ size: '1.5em' }),
|
|
23
|
-
], ngImport: i0, template: "<a\n [href]=\"link.url\"\n target=\"_blank\"\n class=\"flex flex-
|
|
45
|
+
], ngImport: i0, template: "<a\n [href]=\"link.url\"\n target=\"_blank\"\n class=\"group flex flex-row justify-between card-shadow cursor-pointer rounded overflow-hidden\"\n [ngClass]=\"cardClass\"\n [title]=\"title\"\n>\n <div class=\"flex flex-col justify-between\">\n <div class=\"gn-ui-card-title\">\n {{ link.description || link.name }}\n </div>\n <div class=\"gn-ui-card-detail\">\n {{ link.name }}\n </div>\n <p *ngIf=\"!link.name && !link.description\" class=\"gn-ui-card-title\">\n {{ link.url }}\n </p>\n <div class=\"pt-1\">\n <span\n class=\"inline-flex items-center justify-center px-2 py-1 text-13 font-medium leading-none text-white rounded transition-opacity opacity-70 group-hover:opacity-100\"\n [style.background-color]=\"getLinkColor(link)\"\n data-cy=\"download-format\"\n >{{\n getLinkFormat(link) || ('downloads.format.unknown' | translate)\n }}</span\n >\n </div>\n </div>\n <div class=\"flex\" [ngClass]=\"size === 'S' ? 'items-end' : 'items-center'\">\n <div class=\"gn-ui-card-icon\">\n <ng-icon\n class=\"inline-block card-icon align-middle\"\n name=\"matOpenInNew\"\n ></ng-icon>\n </div>\n </div>\n</a>\n", styles: [""], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: NgIconComponent, selector: "ng-icon", inputs: ["name", "svg", "size", "strokeWidth", "color"] }, { kind: "ngmodule", type: TranslateModule }, { kind: "pipe", type: i2.TranslatePipe, name: "translate" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
24
46
|
}
|
|
25
47
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: LinkCardComponent, decorators: [{
|
|
26
48
|
type: Component,
|
|
27
|
-
args: [{ selector: 'gn-ui-link-card', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [CommonModule, NgIconComponent], providers: [
|
|
49
|
+
args: [{ selector: 'gn-ui-link-card', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [CommonModule, NgIconComponent, TranslateModule], providers: [
|
|
28
50
|
provideIcons({
|
|
29
51
|
matOpenInNew,
|
|
30
52
|
}),
|
|
31
53
|
provideNgIconsConfig({ size: '1.5em' }),
|
|
32
|
-
], template: "<a\n [href]=\"link.url\"\n target=\"_blank\"\n class=\"flex flex-
|
|
54
|
+
], template: "<a\n [href]=\"link.url\"\n target=\"_blank\"\n class=\"group flex flex-row justify-between card-shadow cursor-pointer rounded overflow-hidden\"\n [ngClass]=\"cardClass\"\n [title]=\"title\"\n>\n <div class=\"flex flex-col justify-between\">\n <div class=\"gn-ui-card-title\">\n {{ link.description || link.name }}\n </div>\n <div class=\"gn-ui-card-detail\">\n {{ link.name }}\n </div>\n <p *ngIf=\"!link.name && !link.description\" class=\"gn-ui-card-title\">\n {{ link.url }}\n </p>\n <div class=\"pt-1\">\n <span\n class=\"inline-flex items-center justify-center px-2 py-1 text-13 font-medium leading-none text-white rounded transition-opacity opacity-70 group-hover:opacity-100\"\n [style.background-color]=\"getLinkColor(link)\"\n data-cy=\"download-format\"\n >{{\n getLinkFormat(link) || ('downloads.format.unknown' | translate)\n }}</span\n >\n </div>\n </div>\n <div class=\"flex\" [ngClass]=\"size === 'S' ? 'items-end' : 'items-center'\">\n <div class=\"gn-ui-card-icon\">\n <ng-icon\n class=\"inline-block card-icon align-middle\"\n name=\"matOpenInNew\"\n ></ng-icon>\n </div>\n </div>\n</a>\n" }]
|
|
33
55
|
}], propDecorators: { link: [{
|
|
34
56
|
type: Input
|
|
35
|
-
}],
|
|
57
|
+
}], size: [{
|
|
36
58
|
type: Input
|
|
37
59
|
}] } });
|
|
38
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
60
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibGluay1jYXJkLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL2xpYnMvdWkvZWxlbWVudHMvc3JjL2xpYi9saW5rLWNhcmQvbGluay1jYXJkLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL2xpYnMvdWkvZWxlbWVudHMvc3JjL2xpYi9saW5rLWNhcmQvbGluay1jYXJkLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSx1QkFBdUIsRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFFLE1BQU0sZUFBZSxDQUFBO0FBTXpFLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQTtBQUM5QyxPQUFPLEVBQ0wsZUFBZSxFQUNmLFlBQVksRUFDWixvQkFBb0IsR0FDckIsTUFBTSxnQkFBZ0IsQ0FBQTtBQUN2QixPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sbUNBQW1DLENBQUE7QUFDaEUsT0FBTyxFQUFFLGFBQWEsRUFBRSxhQUFhLEVBQUUsTUFBTSx3Q0FBd0MsQ0FBQTtBQUNyRixPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0scUJBQXFCLENBQUE7Ozs7QUFpQnJELE1BQU0sT0FBTyxpQkFBaUI7SUFkOUI7UUFpQm1CLGlCQUFZLEdBQTZCO1lBQ3hELENBQUMsRUFBRSx3QkFBd0I7WUFDM0IsQ0FBQyxFQUFFLHdCQUF3QjtZQUMzQixDQUFDLEVBQUUsa0JBQWtCO1lBQ3JCLEVBQUUsRUFBRSx5QkFBeUI7U0FDOUIsQ0FBQTtRQVNELGNBQVMsR0FBRyxFQUFFLENBQUE7S0FnQmY7SUF2QkMsSUFBYSxJQUFJLENBQUMsS0FBZTtRQUMvQixJQUFJLENBQUMsS0FBSyxHQUFHLEtBQUssQ0FBQTtRQUNsQixJQUFJLENBQUMsU0FBUyxHQUFHLElBQUksQ0FBQyxZQUFZLENBQUMsS0FBSyxDQUFDLENBQUE7SUFDM0MsQ0FBQztJQUNELElBQUksSUFBSTtRQUNOLE9BQU8sSUFBSSxDQUFDLEtBQUssQ0FBQTtJQUNuQixDQUFDO0lBR0QsSUFBSSxLQUFLO1FBQ1AsSUFBSSxJQUFJLENBQUMsSUFBSSxDQUFDLElBQUksSUFBSSxJQUFJLENBQUMsSUFBSSxDQUFDLFdBQVcsRUFBRSxDQUFDO1lBQzVDLE9BQU8sR0FBRyxJQUFJLENBQUMsSUFBSSxDQUFDLElBQUksTUFBTSxJQUFJLENBQUMsSUFBSSxDQUFDLFdBQVcsRUFBRSxDQUFBO1FBQ3ZELENBQUM7UUFDRCxPQUFPLElBQUksQ0FBQyxJQUFJLENBQUMsSUFBSSxJQUFJLElBQUksQ0FBQyxJQUFJLENBQUMsV0FBVyxJQUFJLEVBQUUsQ0FBQTtJQUN0RCxDQUFDO0lBRUQsYUFBYSxDQUFDLElBQVM7UUFDckIsT0FBTyxhQUFhLENBQUMsSUFBSSxDQUFDLENBQUE7SUFDNUIsQ0FBQztJQUVELFlBQVksQ0FBQyxJQUFTO1FBQ3BCLE9BQU8sYUFBYSxDQUFDLGFBQWEsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFBO0lBQzNDLENBQUM7K0dBaENVLGlCQUFpQjttR0FBakIsaUJBQWlCLHNHQVBqQjtZQUNULFlBQVksQ0FBQztnQkFDWCxZQUFZO2FBQ2IsQ0FBQztZQUNGLG9CQUFvQixDQUFDLEVBQUUsSUFBSSxFQUFFLE9BQU8sRUFBRSxDQUFDO1NBQ3hDLDBCQzdCSCx3dENBcUNBLHlERGRZLFlBQVksaU9BQUUsZUFBZSw0R0FBRSxlQUFlOzs0RkFRN0MsaUJBQWlCO2tCQWQ3QixTQUFTOytCQUNFLGlCQUFpQixtQkFHVix1QkFBdUIsQ0FBQyxNQUFNLGNBQ25DLElBQUksV0FDUCxDQUFDLFlBQVksRUFBRSxlQUFlLEVBQUUsZUFBZSxDQUFDLGFBQzlDO3dCQUNULFlBQVksQ0FBQzs0QkFDWCxZQUFZO3lCQUNiLENBQUM7d0JBQ0Ysb0JBQW9CLENBQUMsRUFBRSxJQUFJLEVBQUUsT0FBTyxFQUFFLENBQUM7cUJBQ3hDOzhCQUlRLElBQUk7c0JBQVosS0FBSztnQkFRTyxJQUFJO3NCQUFoQixLQUFLIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3ksIENvbXBvbmVudCwgSW5wdXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJ1xuaW1wb3J0IHtcbiAgRGF0YXNldERvd25sb2FkRGlzdHJpYnV0aW9uLFxuICBEYXRhc2V0T25saW5lUmVzb3VyY2UsXG4gIERhdGFzZXRTZXJ2aWNlRGlzdHJpYnV0aW9uLFxufSBmcm9tICcuLi8uLi8uLi8uLi8uLi8uLi9saWJzL2NvbW1vbi9kb21haW4vc3JjL2xpYi9tb2RlbC9yZWNvcmQnXG5pbXBvcnQgeyBDb21tb25Nb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24nXG5pbXBvcnQge1xuICBOZ0ljb25Db21wb25lbnQsXG4gIHByb3ZpZGVJY29ucyxcbiAgcHJvdmlkZU5nSWNvbnNDb25maWcsXG59IGZyb20gJ0BuZy1pY29ucy9jb3JlJ1xuaW1wb3J0IHsgbWF0T3BlbkluTmV3IH0gZnJvbSAnQG5nLWljb25zL21hdGVyaWFsLWljb25zL2Jhc2VsaW5lJ1xuaW1wb3J0IHsgZ2V0QmFkZ2VDb2xvciwgZ2V0RmlsZUZvcm1hdCB9IGZyb20gJy4uLy4uLy4uLy4uLy4uLy4uL2xpYnMvdXRpbC9zaGFyZWQvc3JjJ1xuaW1wb3J0IHsgVHJhbnNsYXRlTW9kdWxlIH0gZnJvbSAnQG5neC10cmFuc2xhdGUvY29yZSdcblxudHlwZSBDYXJkU2l6ZSA9ICdMJyB8ICdNJyB8ICdTJyB8ICdYUydcbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ2duLXVpLWxpbmstY2FyZCcsXG4gIHRlbXBsYXRlVXJsOiAnLi9saW5rLWNhcmQuY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZVVybHM6IFsnLi9saW5rLWNhcmQuY29tcG9uZW50LmNzcyddLFxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaCxcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgaW1wb3J0czogW0NvbW1vbk1vZHVsZSwgTmdJY29uQ29tcG9uZW50LCBUcmFuc2xhdGVNb2R1bGVdLFxuICBwcm92aWRlcnM6IFtcbiAgICBwcm92aWRlSWNvbnMoe1xuICAgICAgbWF0T3BlbkluTmV3LFxuICAgIH0pLFxuICAgIHByb3ZpZGVOZ0ljb25zQ29uZmlnKHsgc2l6ZTogJzEuNWVtJyB9KSxcbiAgXSxcbn0pXG5leHBvcnQgY2xhc3MgTGlua0NhcmRDb21wb25lbnQge1xuICBwcml2YXRlIF9zaXplOiAnTCcgfCAnTScgfCAnUycgfCAnWFMnXG4gIEBJbnB1dCgpIGxpbms6IERhdGFzZXRPbmxpbmVSZXNvdXJjZVxuICBwcml2YXRlIHJlYWRvbmx5IHNpemVDbGFzc01hcDogUmVjb3JkPENhcmRTaXplLCBzdHJpbmc+ID0ge1xuICAgIEw6ICdnbi11aS1jYXJkLWwgcHktMiBweC01JyxcbiAgICBNOiAnZ24tdWktY2FyZC1tIHB5LTIgcHgtNScsXG4gICAgUzogJ2duLXVpLWNhcmQtcyBwLTQnLFxuICAgIFhTOiAnZ24tdWktY2FyZC14cyBweS0yIHB4LTUnLFxuICB9XG5cbiAgQElucHV0KCkgc2V0IHNpemUodmFsdWU6IENhcmRTaXplKSB7XG4gICAgdGhpcy5fc2l6ZSA9IHZhbHVlXG4gICAgdGhpcy5jYXJkQ2xhc3MgPSB0aGlzLnNpemVDbGFzc01hcFt2YWx1ZV1cbiAgfVxuICBnZXQgc2l6ZSgpOiAnTCcgfCAnTScgfCAnUycgfCAnWFMnIHtcbiAgICByZXR1cm4gdGhpcy5fc2l6ZVxuICB9XG4gIGNhcmRDbGFzcyA9ICcnXG5cbiAgZ2V0IHRpdGxlKCkge1xuICAgIGlmICh0aGlzLmxpbmsubmFtZSAmJiB0aGlzLmxpbmsuZGVzY3JpcHRpb24pIHtcbiAgICAgIHJldHVybiBgJHt0aGlzLmxpbmsubmFtZX0gfCAke3RoaXMubGluay5kZXNjcmlwdGlvbn1gXG4gICAgfVxuICAgIHJldHVybiB0aGlzLmxpbmsubmFtZSB8fCB0aGlzLmxpbmsuZGVzY3JpcHRpb24gfHwgJydcbiAgfVxuXG4gIGdldExpbmtGb3JtYXQobGluazogYW55KSB7XG4gICAgcmV0dXJuIGdldEZpbGVGb3JtYXQobGluaylcbiAgfVxuXG4gIGdldExpbmtDb2xvcihsaW5rOiBhbnkpIHtcbiAgICByZXR1cm4gZ2V0QmFkZ2VDb2xvcihnZXRGaWxlRm9ybWF0KGxpbmspKVxuICB9XG59XG4iLCI8YVxuICBbaHJlZl09XCJsaW5rLnVybFwiXG4gIHRhcmdldD1cIl9ibGFua1wiXG4gIGNsYXNzPVwiZ3JvdXAgZmxleCBmbGV4LXJvdyBqdXN0aWZ5LWJldHdlZW4gY2FyZC1zaGFkb3cgY3Vyc29yLXBvaW50ZXIgcm91bmRlZCBvdmVyZmxvdy1oaWRkZW5cIlxuICBbbmdDbGFzc109XCJjYXJkQ2xhc3NcIlxuICBbdGl0bGVdPVwidGl0bGVcIlxuPlxuICA8ZGl2IGNsYXNzPVwiZmxleCBmbGV4LWNvbCBqdXN0aWZ5LWJldHdlZW5cIj5cbiAgICA8ZGl2IGNsYXNzPVwiZ24tdWktY2FyZC10aXRsZVwiPlxuICAgICAge3sgbGluay5kZXNjcmlwdGlvbiB8fCBsaW5rLm5hbWUgfX1cbiAgICA8L2Rpdj5cbiAgICA8ZGl2IGNsYXNzPVwiZ24tdWktY2FyZC1kZXRhaWxcIj5cbiAgICAgIHt7IGxpbmsubmFtZSB9fVxuICAgIDwvZGl2PlxuICAgIDxwICpuZ0lmPVwiIWxpbmsubmFtZSAmJiAhbGluay5kZXNjcmlwdGlvblwiIGNsYXNzPVwiZ24tdWktY2FyZC10aXRsZVwiPlxuICAgICAge3sgbGluay51cmwgfX1cbiAgICA8L3A+XG4gICAgPGRpdiBjbGFzcz1cInB0LTFcIj5cbiAgICAgIDxzcGFuXG4gICAgICAgIGNsYXNzPVwiaW5saW5lLWZsZXggaXRlbXMtY2VudGVyIGp1c3RpZnktY2VudGVyIHB4LTIgcHktMSB0ZXh0LTEzIGZvbnQtbWVkaXVtIGxlYWRpbmctbm9uZSB0ZXh0LXdoaXRlIHJvdW5kZWQgdHJhbnNpdGlvbi1vcGFjaXR5IG9wYWNpdHktNzAgZ3JvdXAtaG92ZXI6b3BhY2l0eS0xMDBcIlxuICAgICAgICBbc3R5bGUuYmFja2dyb3VuZC1jb2xvcl09XCJnZXRMaW5rQ29sb3IobGluaylcIlxuICAgICAgICBkYXRhLWN5PVwiZG93bmxvYWQtZm9ybWF0XCJcbiAgICAgICAgPnt7XG4gICAgICAgICAgZ2V0TGlua0Zvcm1hdChsaW5rKSB8fCAoJ2Rvd25sb2Fkcy5mb3JtYXQudW5rbm93bicgfCB0cmFuc2xhdGUpXG4gICAgICAgIH19PC9zcGFuXG4gICAgICA+XG4gICAgPC9kaXY+XG4gIDwvZGl2PlxuICA8ZGl2IGNsYXNzPVwiZmxleFwiIFtuZ0NsYXNzXT1cInNpemUgPT09ICdTJyA/ICdpdGVtcy1lbmQnIDogJ2l0ZW1zLWNlbnRlcidcIj5cbiAgICA8ZGl2IGNsYXNzPVwiZ24tdWktY2FyZC1pY29uXCI+XG4gICAgICA8bmctaWNvblxuICAgICAgICBjbGFzcz1cImlubGluZS1ibG9jayBjYXJkLWljb24gYWxpZ24tbWlkZGxlXCJcbiAgICAgICAgbmFtZT1cIm1hdE9wZW5Jbk5ld1wiXG4gICAgICA+PC9uZy1pY29uPlxuICAgIDwvZGl2PlxuICA8L2Rpdj5cbjwvYT5cbiJdfQ==
|
|
@@ -4,12 +4,12 @@ import * as i0 from "@angular/core";
|
|
|
4
4
|
import * as i1 from "@ngx-translate/core";
|
|
5
5
|
export class MetadataCatalogComponent {
|
|
6
6
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: MetadataCatalogComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
7
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: MetadataCatalogComponent, isStandalone: true, selector: "gn-ui-metadata-catalog", inputs: { sourceLabel: "sourceLabel" }, ngImport: i0, template: "<div>\n <p class=\"text-gray-
|
|
7
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: MetadataCatalogComponent, isStandalone: true, selector: "gn-ui-metadata-catalog", inputs: { sourceLabel: "sourceLabel" }, ngImport: i0, template: "<div>\n <p class=\"text-gray-900 text-xs mb-3 uppercase\" translate>\n record.metadata.catalog\n </p>\n <p class=\"text-primary font-title text-21 mb-1\">\n {{ sourceLabel }}\n </p>\n</div>\n", styles: [""], dependencies: [{ kind: "ngmodule", type: TranslateModule }, { kind: "directive", type: i1.TranslateDirective, selector: "[translate],[ngx-translate]", inputs: ["translate", "translateParams"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
8
8
|
}
|
|
9
9
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: MetadataCatalogComponent, decorators: [{
|
|
10
10
|
type: Component,
|
|
11
|
-
args: [{ selector: 'gn-ui-metadata-catalog', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [TranslateModule], template: "<div>\n <p class=\"text-gray-
|
|
11
|
+
args: [{ selector: 'gn-ui-metadata-catalog', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [TranslateModule], template: "<div>\n <p class=\"text-gray-900 text-xs mb-3 uppercase\" translate>\n record.metadata.catalog\n </p>\n <p class=\"text-primary font-title text-21 mb-1\">\n {{ sourceLabel }}\n </p>\n</div>\n" }]
|
|
12
12
|
}], propDecorators: { sourceLabel: [{
|
|
13
13
|
type: Input
|
|
14
14
|
}] } });
|
|
15
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
15
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWV0YWRhdGEtY2F0YWxvZy5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9saWJzL3VpL2VsZW1lbnRzL3NyYy9saWIvbWV0YWRhdGEtY2F0YWxvZy9tZXRhZGF0YS1jYXRhbG9nLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL2xpYnMvdWkvZWxlbWVudHMvc3JjL2xpYi9tZXRhZGF0YS1jYXRhbG9nL21ldGFkYXRhLWNhdGFsb2cuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLHVCQUF1QixFQUFFLFNBQVMsRUFBRSxLQUFLLEVBQUUsTUFBTSxlQUFlLENBQUE7QUFDekUsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLHFCQUFxQixDQUFBOzs7QUFVckQsTUFBTSxPQUFPLHdCQUF3QjsrR0FBeEIsd0JBQXdCO21HQUF4Qix3QkFBd0IsMEhDWHJDLDJNQVFBLHlERENZLGVBQWU7OzRGQUVkLHdCQUF3QjtrQkFScEMsU0FBUzsrQkFDRSx3QkFBd0IsbUJBR2pCLHVCQUF1QixDQUFDLE1BQU0sY0FDbkMsSUFBSSxXQUNQLENBQUMsZUFBZSxDQUFDOzhCQUdqQixXQUFXO3NCQUFuQixLQUFLIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3ksIENvbXBvbmVudCwgSW5wdXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJ1xuaW1wb3J0IHsgVHJhbnNsYXRlTW9kdWxlIH0gZnJvbSAnQG5neC10cmFuc2xhdGUvY29yZSdcblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnZ24tdWktbWV0YWRhdGEtY2F0YWxvZycsXG4gIHRlbXBsYXRlVXJsOiAnLi9tZXRhZGF0YS1jYXRhbG9nLmNvbXBvbmVudC5odG1sJyxcbiAgc3R5bGVVcmxzOiBbJy4vbWV0YWRhdGEtY2F0YWxvZy5jb21wb25lbnQuY3NzJ10sXG4gIGNoYW5nZURldGVjdGlvbjogQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3kuT25QdXNoLFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICBpbXBvcnRzOiBbVHJhbnNsYXRlTW9kdWxlXSxcbn0pXG5leHBvcnQgY2xhc3MgTWV0YWRhdGFDYXRhbG9nQ29tcG9uZW50IHtcbiAgQElucHV0KCkgc291cmNlTGFiZWw6IHN0cmluZ1xufVxuIiwiPGRpdj5cbiAgPHAgY2xhc3M9XCJ0ZXh0LWdyYXktOTAwIHRleHQteHMgbWItMyB1cHBlcmNhc2VcIiB0cmFuc2xhdGU+XG4gICAgcmVjb3JkLm1ldGFkYXRhLmNhdGFsb2dcbiAgPC9wPlxuICA8cCBjbGFzcz1cInRleHQtcHJpbWFyeSBmb250LXRpdGxlIHRleHQtMjEgbWItMVwiPlxuICAgIHt7IHNvdXJjZUxhYmVsIH19XG4gIDwvcD5cbjwvZGl2PlxuIl19
|
|
@@ -54,7 +54,7 @@ export class MetadataQualityComponent {
|
|
|
54
54
|
size: '1.2em',
|
|
55
55
|
strokeWidth: '1.5px',
|
|
56
56
|
}),
|
|
57
|
-
], usesOnChanges: true, ngImport: i0, template: "<div *ngIf=\"metadataQualityDisplay\" class=\"
|
|
57
|
+
], usesOnChanges: true, ngImport: i0, template: "<div *ngIf=\"metadataQualityDisplay\" class=\"metadata-quality\">\n <div\n class=\"flex items-center\"\n [class]=\"\n smaller ? 'leading-[8px] min-w-[120px] m-h-[120px]' : 'min-w-[200px]'\n \"\n >\n <gn-ui-progress-bar\n tabindex=\"0\"\n [value]=\"qualityScore\"\n [type]=\"'light'\"\n class=\"flex-grow\"\n ></gn-ui-progress-bar>\n <gn-ui-popover\n [content]=\"popoverItems\"\n theme=\"light-border\"\n [class]=\"smaller ? 'ml-2' : 'ml-2 mt-1'\"\n >\n <ng-icon\n name=\"matInfoOutline\"\n class=\"flex-shrink-0 text-gray-600\"\n ></ng-icon>\n </gn-ui-popover>\n </div>\n</div>\n<ng-template #popoverItems>\n <div class=\"p-2 py-4\">\n <div class=\"mb-4 font-bold\" translate>record.metadata.quality.details</div>\n <gn-ui-metadata-quality-item\n *ngFor=\"let e of items\"\n [name]=\"e.name\"\n [value]=\"e.value\"\n ></gn-ui-metadata-quality-item>\n </div>\n</ng-template>\n", styles: [":host gn-ui-progress-bar{--progress-bar-font-weight: \"normal\"}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: PopoverComponent, selector: "gn-ui-popover", inputs: ["content", "theme"] }, { kind: "component", type: ProgressBarComponent, selector: "gn-ui-progress-bar", inputs: ["value", "type"] }, { kind: "component", type: MetadataQualityItemComponent, selector: "gn-ui-metadata-quality-item", inputs: ["name", "value"] }, { kind: "ngmodule", type: TranslateModule }, { kind: "directive", type: i2.TranslateDirective, selector: "[translate],[ngx-translate]", inputs: ["translate", "translateParams"] }, { kind: "component", type: NgIcon, selector: "ng-icon", inputs: ["name", "svg", "size", "strokeWidth", "color"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
58
58
|
}
|
|
59
59
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: MetadataQualityComponent, decorators: [{
|
|
60
60
|
type: Component,
|
|
@@ -73,7 +73,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
73
73
|
size: '1.2em',
|
|
74
74
|
strokeWidth: '1.5px',
|
|
75
75
|
}),
|
|
76
|
-
], template: "<div *ngIf=\"metadataQualityDisplay\" class=\"
|
|
76
|
+
], template: "<div *ngIf=\"metadataQualityDisplay\" class=\"metadata-quality\">\n <div\n class=\"flex items-center\"\n [class]=\"\n smaller ? 'leading-[8px] min-w-[120px] m-h-[120px]' : 'min-w-[200px]'\n \"\n >\n <gn-ui-progress-bar\n tabindex=\"0\"\n [value]=\"qualityScore\"\n [type]=\"'light'\"\n class=\"flex-grow\"\n ></gn-ui-progress-bar>\n <gn-ui-popover\n [content]=\"popoverItems\"\n theme=\"light-border\"\n [class]=\"smaller ? 'ml-2' : 'ml-2 mt-1'\"\n >\n <ng-icon\n name=\"matInfoOutline\"\n class=\"flex-shrink-0 text-gray-600\"\n ></ng-icon>\n </gn-ui-popover>\n </div>\n</div>\n<ng-template #popoverItems>\n <div class=\"p-2 py-4\">\n <div class=\"mb-4 font-bold\" translate>record.metadata.quality.details</div>\n <gn-ui-metadata-quality-item\n *ngFor=\"let e of items\"\n [name]=\"e.name\"\n [value]=\"e.value\"\n ></gn-ui-metadata-quality-item>\n </div>\n</ng-template>\n", styles: [":host gn-ui-progress-bar{--progress-bar-font-weight: \"normal\"}\n"] }]
|
|
77
77
|
}], propDecorators: { metadata: [{
|
|
78
78
|
type: Input
|
|
79
79
|
}], smaller: [{
|
|
@@ -81,4 +81,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
81
81
|
}], metadataQualityDisplay: [{
|
|
82
82
|
type: Input
|
|
83
83
|
}] } });
|
|
84
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
84
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"metadata-quality.component.js","sourceRoot":"","sources":["../../../../../../../../libs/ui/elements/src/lib/metadata-quality/metadata-quality.component.ts","../../../../../../../../libs/ui/elements/src/lib/metadata-quality/metadata-quality.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component,\n  Input,\n  OnChanges,\n  SimpleChanges,\n} from '@angular/core'\nimport {\n  MetadataQualityItem,\n  MetadataQualityItemComponent,\n} from '../metadata-quality-item/metadata-quality-item.component'\nimport { CatalogRecord } from '../../../../../../libs/common/domain/src/lib/model/record'\nimport {\n  PopoverComponent,\n  ProgressBarComponent,\n} from '../../../../../../libs/ui/widgets/src'\nimport { CommonModule } from '@angular/common'\nimport { TranslateModule } from '@ngx-translate/core'\nimport { NgIcon } from '@ng-icons/core'\nimport { matInfoOutline } from '@ng-icons/material-icons/outline'\nimport { provideIcons, provideNgIconsConfig } from '@ng-icons/core'\n\n@Component({\n  selector: 'gn-ui-metadata-quality',\n  templateUrl: './metadata-quality.component.html',\n  styleUrls: ['./metadata-quality.component.css'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [\n    CommonModule,\n    PopoverComponent,\n    ProgressBarComponent,\n    MetadataQualityItemComponent,\n    TranslateModule,\n    NgIcon,\n  ],\n  providers: [\n    provideIcons({\n      matInfoOutline,\n    }),\n    provideNgIconsConfig({\n      size: '1.2em',\n      strokeWidth: '1.5px',\n    }),\n  ],\n})\nexport class MetadataQualityComponent implements OnChanges {\n  @Input() metadata: Partial<CatalogRecord>\n  @Input() smaller = false\n  @Input() metadataQualityDisplay: boolean\n\n  items: MetadataQualityItem[] = []\n\n  get qualityScore() {\n    const qualityScore = this.metadata?.extras?.qualityScore\n    return typeof qualityScore === 'number'\n      ? qualityScore\n      : this.calculatedQualityScore\n  }\n\n  get calculatedQualityScore(): number {\n    return Math.round(\n      (this.items.filter(({ value }) => value).length * 100) / this.items.length\n    )\n  }\n\n  private add(name: string, value: boolean) {\n    if (this.metadataQualityDisplay?.[name] !== false) {\n      this.items.push({ name, value })\n    }\n  }\n\n  initialize() {\n    const contact = this.metadata?.contacts?.[0]\n    this.items = []\n    this.add('title', !!this.metadata?.title)\n    this.add('description', !!this.metadata?.abstract)\n    this.add('topic', this.metadata?.topics?.length > 0)\n    this.add('keywords', this.metadata?.keywords?.length > 0)\n    this.add('legalConstraints', this.metadata?.legalConstraints?.length > 0)\n    this.add('organisation', !!contact?.organization)\n    this.add('contact', !!contact?.email)\n    this.add('updateFrequency', !!this.metadata?.updateFrequency)\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['metadata'] || changes['metadataQualityDisplay']) {\n      this.initialize()\n    }\n  }\n}\n","<div *ngIf=\"metadataQualityDisplay\" class=\"metadata-quality\">\n  <div\n    class=\"flex items-center\"\n    [class]=\"\n      smaller ? 'leading-[8px] min-w-[120px] m-h-[120px]' : 'min-w-[200px]'\n    \"\n  >\n    <gn-ui-progress-bar\n      tabindex=\"0\"\n      [value]=\"qualityScore\"\n      [type]=\"'light'\"\n      class=\"flex-grow\"\n    ></gn-ui-progress-bar>\n    <gn-ui-popover\n      [content]=\"popoverItems\"\n      theme=\"light-border\"\n      [class]=\"smaller ? 'ml-2' : 'ml-2 mt-1'\"\n    >\n      <ng-icon\n        name=\"matInfoOutline\"\n        class=\"flex-shrink-0 text-gray-600\"\n      ></ng-icon>\n    </gn-ui-popover>\n  </div>\n</div>\n<ng-template #popoverItems>\n  <div class=\"p-2 py-4\">\n    <div class=\"mb-4 font-bold\" translate>record.metadata.quality.details</div>\n    <gn-ui-metadata-quality-item\n      *ngFor=\"let e of items\"\n      [name]=\"e.name\"\n      [value]=\"e.value\"\n    ></gn-ui-metadata-quality-item>\n  </div>\n</ng-template>\n"]}
|
|
@@ -110,7 +110,7 @@ export class RecordApiFormComponent {
|
|
|
110
110
|
maxFeatures: limit !== '-1' ? Number(limit) : undefined,
|
|
111
111
|
limit: limit !== '-1' ? Number(limit) : -1,
|
|
112
112
|
offset: offset !== '' ? Number(offset) : undefined,
|
|
113
|
-
outputCrs: format
|
|
113
|
+
outputCrs: format.toLowerCase().indexOf('json') > -1 ? 'EPSG:4326' : undefined,
|
|
114
114
|
};
|
|
115
115
|
if (this.endpoint instanceof WfsEndpoint) {
|
|
116
116
|
delete options.limit;
|
|
@@ -118,6 +118,7 @@ export class RecordApiFormComponent {
|
|
|
118
118
|
return this.endpoint.getFeatureUrl(this.apiFeatureType, options);
|
|
119
119
|
}
|
|
120
120
|
else {
|
|
121
|
+
delete options.outputCrs;
|
|
121
122
|
return await this.endpoint.getCollectionItemsUrl(this.apiFeatureType, options);
|
|
122
123
|
}
|
|
123
124
|
}
|
|
@@ -136,4 +137,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
136
137
|
}], propDecorators: { apiLink: [{
|
|
137
138
|
type: Input
|
|
138
139
|
}] } });
|
|
139
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"record-api-form.component.js","sourceRoot":"","sources":["../../../../../../../../libs/ui/elements/src/lib/record-api-form/record-api-form.component.ts","../../../../../../../../libs/ui/elements/src/lib/record-api-form/record-api-form.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,KAAK,EAAE,MAAM,eAAe,CAAA;AACzE,OAAO,EAAY,cAAc,EAAE,WAAW,EAAE,MAAM,wBAAwB,CAAA;AAK9E,OAAO,EAAE,gBAAgB,EAAE,MAAM,wCAAwC,CAAA;AACzE,OAAO,EAAE,eAAe,EAAE,aAAa,EAAE,MAAM,EAAE,GAAG,EAAE,SAAS,EAAE,MAAM,MAAM,CAAA;AAC7E,OAAO,EACL,uBAAuB,EAEvB,yBAAyB,EACzB,kBAAkB,GACnB,MAAM,sCAAsC,CAAA;AAC7C,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAA;AAC9C,OAAO,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAA;;;;AAErD,MAAM,cAAc,GAAG;IACrB,MAAM,EAAE,EAAE;IACV,KAAK,EAAE,IAAI;IACX,MAAM,EAAE,kBAAkB;CAC3B,CAAA;AAgBD,MAAM,OAAO,sBAAsB;IAdnC;QA0BE,YAAO,GAAG,IAAI,eAAe,CAAC,cAAc,CAAC,MAAM,CAAC,CAAA;QACpD,WAAM,GAAG,IAAI,eAAe,CAAC,cAAc,CAAC,KAAK,CAAC,CAAA;QAClD,YAAO,GAAG,IAAI,eAAe,CAAC,cAAc,CAAC,MAAM,CAAC,CAAA;QACpD,cAAS,GAAG,IAAI,eAAe,CAC7B,SAAS,CACV,CAAA;QAGD,kBAAa,GAAG,IAAI,CAAA;QAEpB,kBAAa,GAAqB;YAChC,EAAE,KAAK,EAAE,kBAAkB,EAAE,KAAK,EAAE,MAAM,EAAE;SAC7C,CAAA;QAGD,iBAAY,GAAG,aAAa,CAAC;YAC3B,IAAI,CAAC,OAAO;YACZ,IAAI,CAAC,MAAM;YACX,IAAI,CAAC,OAAO;YACZ,mDAAmD;YACnD,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC;SACtD,CAAC,CAAC,IAAI,CACL,SAAS,CAAC,CAAC,CAAC,MAAM,EAAE,KAAK,EAAE,MAAM,CAAC,EAAE,EAAE,CACpC,IAAI,CAAC,mBAAmB,CAAC,MAAM,EAAE,KAAK,EAAE,MAAM,CAAC,CAChD,CACF,CAAA;QAED,oBAAe,GAAG,IAAI,CAAC,MAAM,CAAC,IAAI,CAChC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,KAAK,IAAI,IAAI,KAAK,KAAK,EAAE,CAAC,CAC/C,CAAA;QACD,kBAAa,GAAG,IAAI,CAAC,MAAM,CAAC,IAAI,CAC9B,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC,KAAK,KAAK,IAAI,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAC9C,CAAA;KAiGF;IA5IC,IAAa,OAAO,CAAC,KAAiC;QACpD,IAAI,CAAC,aAAa,GAAG,CAAC,EAAE,KAAK,EAAE,kBAAkB,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,CAAA;QACnE,IAAI,CAAC,qBAAqB,GAAG,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,qBAAqB,CAAC,CAAC,CAAC,SAAS,CAAA;QAC5E,IAAI,CAAC,cAAc,GAAG,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,SAAS,CAAA;QACpD,IAAI,KAAK,EAAE,CAAC;YACV,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC,GAAG,CAAC,IAAI,CAAA;YAChC,IAAI,CAAC,cAAc,EAAE,CAAC,IAAI,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,kBAAkB,EAAE,CAAC,CAAA;QAC7D,CAAC;QACD,IAAI,CAAC,QAAQ,EAAE,CAAA;IACjB,CAAC;IAoCD,SAAS,CAAC,KAAa;QACrB,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;IAC1B,CAAC;IAED,QAAQ,CAAC,KAAa;QACpB,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,KAAK,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,CAAC,CAAA;IAC/C,CAAC;IAED,SAAS,CAAC,KAAuB;QAC/B,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAA;IAClC,CAAC;IAED,QAAQ;QACN,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,CAAA;QACxC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,cAAc,CAAC,KAAK,CAAC,CAAA;QACtC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,CAAA;IAC1C,CAAC;IAED,KAAK,CAAC,kBAAkB;QACtB,IAAI,CAAC,IAAI,CAAC,QAAQ;YAAE,OAAM;QAC1B,MAAM,aAAa,GAAG,CAAC,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC,CAAC,GAAG,CACvD,IAAI,CAAC,oBAAoB,CAC1B,CAAA;QAED,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,aAAa;aACpC,MAAM,CAAC,aAAa,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;aACrC,MAAM,CACL,CAAC,MAAM,EAAE,KAAK,EAAE,IAAI,EAAE,EAAE,CACtB,KAAK,KAAK,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,KAAK,KAAK,MAAM,CAAC,KAAK,CAAC,CAC5D;aACA,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAA;IACnD,CAAC;IAED,oBAAoB,CAAC,QAAkB;QACrC,MAAM,UAAU,GAAG,gBAAgB,CAAC,QAAQ,CAAC,CAAA;QAC7C,OAAO,UAAU;YACf,CAAC,CAAC,EAAE,KAAK,EAAE,UAAU,CAAC,WAAW,EAAE,EAAE,KAAK,EAAE,QAAQ,EAAE;YACtD,CAAC,CAAC,IAAI,CAAA;IACV,CAAC;IAED,KAAK,CAAC,gBAAgB;QACpB,IAAI,CAAC,IAAI,CAAC,QAAQ;YAAE,OAAO,EAAE,CAAA;QAC7B,IAAI,IAAI,CAAC,QAAQ,YAAY,WAAW,EAAE,CAAC;YACzC,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,QAAQ,CAAC,kBAAkB,EAAE,CAAA;YACvD,OAAO,IAAI,CAAC,QAAQ,CAAC,cAAc,EAAE,CAAC,aAAa,CAAA;QACrD,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,MAAM,IAAI,CAAC,QAAQ,CAAC,iBAAiB,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;iBAChE,WAAW,CAAA;QAChB,CAAC;IACH,CAAC;IAED,KAAK,CAAC,cAAc;QAClB,IAAI,CAAC,IAAI,CAAC,UAAU,IAAI,CAAC,IAAI,CAAC,qBAAqB;YAAE,OAAM;QAC3D,IAAI,IAAI,CAAC,qBAAqB,KAAK,KAAK,EAAE,CAAC;YACzC,IAAI,CAAC,QAAQ,GAAG,IAAI,WAAW,CAAC,IAAI,CAAC,UAAU,CAAC,CAAA;YAChD,MAAO,IAAI,CAAC,QAAwB,CAAC,OAAO,EAAE,CAAA;QAChD,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,QAAQ,GAAG,IAAI,cAAc,CAAC,IAAI,CAAC,UAAU,CAAC,CAAA;YACnD,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,QAAQ,CAAC,cAAc,CAAA;YACtD,kFAAkF;YAClF,IAAI,WAAW,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBAC7B,IAAI,CAAC,cAAc,GAAG,WAAW,CAAC,CAAC,CAAC,CAAC,IAAI,CAAA;YAC3C,CAAC;QACH,CAAC;QACD,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAA;IACpC,CAAC;IAED,KAAK,CAAC,mBAAmB,CACvB,MAAc,EACd,KAAa,EACb,MAAc;QAEd,IAAI,CAAC,IAAI,CAAC,UAAU,IAAI,CAAC,IAAI,CAAC,QAAQ,IAAI,CAAC,IAAI,CAAC,cAAc;YAAE,OAAO,EAAE,CAAA;QAEzE,MAAM,OAAO,GAAG;YACd,YAAY,EAAE,MAAM;YACpB,UAAU,EAAE,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,SAAS;YAC/C,WAAW,EAAE,KAAK,KAAK,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,SAAS;YACvD,KAAK,EAAE,KAAK,KAAK,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;YAC1C,MAAM,EAAE,MAAM,KAAK,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,SAAS;YAClD,SAAS,EACP,MAAM,KAAK,CAAC,kBAAkB,IAAI,SAAS,CAAC,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC,CAAC,SAAS;SACzE,CAAA;QAED,IAAI,IAAI,CAAC,QAAQ,YAAY,WAAW,EAAE,CAAC;YACzC,OAAO,OAAO,CAAC,KAAK,CAAA;YACpB,OAAO,CAAC,WAAW,GAAG,KAAK,KAAK,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,SAAS,CAAA;YAChE,OAAO,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC,IAAI,CAAC,cAAc,EAAE,OAAO,CAAC,CAAA;QAClE,CAAC;aAAM,CAAC;YACN,OAAO,MAAM,IAAI,CAAC,QAAQ,CAAC,qBAAqB,CAC9C,IAAI,CAAC,cAAc,EACnB,OAAO,CACR,CAAA;QACH,CAAC;IACH,CAAC;+GA5IU,sBAAsB;mGAAtB,sBAAsB,iHCrCnC,2zGA2FA,qgBD7DI,YAAY,wLACZ,kBAAkB,+JAClB,yBAAyB,qNACzB,uBAAuB,0HACvB,eAAe;;4FAGN,sBAAsB;kBAdlC,SAAS;+BACE,uBAAuB,mBAGhB,uBAAuB,CAAC,MAAM,cACnC,IAAI,WACP;wBACP,YAAY;wBACZ,kBAAkB;wBAClB,yBAAyB;wBACzB,uBAAuB;wBACvB,eAAe;qBAChB;8BAGY,OAAO;sBAAnB,KAAK","sourcesContent":["import { ChangeDetectionStrategy, Component, Input } from '@angular/core'\nimport { MimeType, OgcApiEndpoint, WfsEndpoint } from '@camptocamp/ogc-client'\nimport {\n  DatasetServiceDistribution,\n  ServiceProtocol,\n} from '../../../../../../libs/common/domain/src/lib/model/record'\nimport { mimeTypeToFormat } from '../../../../../../libs/util/shared/src'\nimport { BehaviorSubject, combineLatest, filter, map, switchMap } from 'rxjs'\nimport {\n  CopyTextButtonComponent,\n  DropdownChoice,\n  DropdownSelectorComponent,\n  TextInputComponent,\n} from '../../../../../../libs/ui/inputs/src'\nimport { CommonModule } from '@angular/common'\nimport { TranslateModule } from '@ngx-translate/core'\n\nconst DEFAULT_PARAMS = {\n  OFFSET: '',\n  LIMIT: '-1',\n  FORMAT: 'application/json',\n}\n\n@Component({\n  selector: 'gn-ui-record-api-form',\n  templateUrl: './record-api-form.component.html',\n  styleUrls: ['./record-api-form.component.css'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [\n    CommonModule,\n    TextInputComponent,\n    DropdownSelectorComponent,\n    CopyTextButtonComponent,\n    TranslateModule,\n  ],\n})\nexport class RecordApiFormComponent {\n  @Input() set apiLink(value: DatasetServiceDistribution) {\n    this.outputFormats = [{ value: 'application/json', label: 'JSON' }]\n    this.accessServiceProtocol = value ? value.accessServiceProtocol : undefined\n    this.apiFeatureType = value ? value.name : undefined\n    if (value) {\n      this.apiBaseUrl = value.url.href\n      this.createEndpoint().then(() => this.parseOutputFormats())\n    }\n    this.resetUrl()\n  }\n\n  offset$ = new BehaviorSubject(DEFAULT_PARAMS.OFFSET)\n  limit$ = new BehaviorSubject(DEFAULT_PARAMS.LIMIT)\n  format$ = new BehaviorSubject(DEFAULT_PARAMS.FORMAT)\n  endpoint$ = new BehaviorSubject<WfsEndpoint | OgcApiEndpoint | undefined>(\n    undefined\n  )\n  apiBaseUrl: string\n  apiFeatureType: string\n  supportOffset = true\n  accessServiceProtocol: ServiceProtocol | undefined\n  outputFormats: DropdownChoice[] = [\n    { value: 'application/json', label: 'JSON' },\n  ]\n  endpoint: WfsEndpoint | OgcApiEndpoint | undefined\n\n  apiQueryUrl$ = combineLatest([\n    this.offset$,\n    this.limit$,\n    this.format$,\n    // only compute the url if the endpoint was created\n    this.endpoint$.pipe(filter((endpoint) => !!endpoint)),\n  ]).pipe(\n    switchMap(([offset, limit, format]) =>\n      this.generateApiQueryUrl(offset, limit, format)\n    )\n  )\n\n  noLimitChecked$ = this.limit$.pipe(\n    map((limit) => limit === '-1' || limit === '')\n  )\n  displayLimit$ = this.limit$.pipe(\n    map((limit) => (limit !== '-1' ? limit : ''))\n  )\n\n  setOffset(value: string) {\n    this.offset$.next(value)\n  }\n\n  setLimit(value: string) {\n    this.limit$.next(value === '' ? '-1' : value)\n  }\n\n  setFormat(value: string | unknown) {\n    this.format$.next(String(value))\n  }\n\n  resetUrl() {\n    this.offset$.next(DEFAULT_PARAMS.OFFSET)\n    this.limit$.next(DEFAULT_PARAMS.LIMIT)\n    this.format$.next(DEFAULT_PARAMS.FORMAT)\n  }\n\n  async parseOutputFormats() {\n    if (!this.endpoint) return\n    const outputFormats = (await this.getOutputFormats()).map(\n      this.mimeTypeToFormatName\n    )\n\n    this.outputFormats = this.outputFormats\n      .concat(outputFormats.filter(Boolean))\n      .filter(\n        (format, index, self) =>\n          index === self.findIndex((t) => t.value === format.value)\n      )\n      .sort((a, b) => a.label.localeCompare(b.label))\n  }\n\n  mimeTypeToFormatName(mimeType: MimeType): DropdownChoice | null {\n    const formatName = mimeTypeToFormat(mimeType)\n    return formatName\n      ? { label: formatName.toUpperCase(), value: mimeType }\n      : null\n  }\n\n  async getOutputFormats(): Promise<MimeType[]> {\n    if (!this.endpoint) return []\n    if (this.endpoint instanceof WfsEndpoint) {\n      this.supportOffset = this.endpoint.supportsStartIndex()\n      return this.endpoint.getServiceInfo().outputFormats\n    } else {\n      return (await this.endpoint.getCollectionInfo(this.apiFeatureType))\n        .itemFormats\n    }\n  }\n\n  async createEndpoint() {\n    if (!this.apiBaseUrl || !this.accessServiceProtocol) return\n    if (this.accessServiceProtocol === 'wfs') {\n      this.endpoint = new WfsEndpoint(this.apiBaseUrl)\n      await (this.endpoint as WfsEndpoint).isReady()\n    } else {\n      this.endpoint = new OgcApiEndpoint(this.apiBaseUrl)\n      const collections = await this.endpoint.allCollections\n      // if there's only one collection, use this instead of the name given in the link.\n      if (collections.length === 1) {\n        this.apiFeatureType = collections[0].name\n      }\n    }\n    this.endpoint$.next(this.endpoint)\n  }\n\n  async generateApiQueryUrl(\n    offset: string,\n    limit: string,\n    format: string\n  ): Promise<string> {\n    if (!this.apiBaseUrl || !this.endpoint || !this.apiFeatureType) return ''\n\n    const options = {\n      outputFormat: format,\n      startIndex: offset ? Number(offset) : undefined,\n      maxFeatures: limit !== '-1' ? Number(limit) : undefined,\n      limit: limit !== '-1' ? Number(limit) : -1,\n      offset: offset !== '' ? Number(offset) : undefined,\n      outputCrs:\n        format === ('application/json' || 'geojson') ? 'EPSG:4326' : undefined,\n    }\n\n    if (this.endpoint instanceof WfsEndpoint) {\n      delete options.limit\n      options.maxFeatures = limit !== '-1' ? Number(limit) : undefined\n      return this.endpoint.getFeatureUrl(this.apiFeatureType, options)\n    } else {\n      return await this.endpoint.getCollectionItemsUrl(\n        this.apiFeatureType,\n        options\n      )\n    }\n  }\n}\n","<div class=\"flex flex-col gap-8\">\n  <div class=\"flex flex-col bg-white p-8 ng-star-inserted shadow-xl gap-8\">\n    <div class=\"flex flex-row\">\n      <div class=\"text-[16px] text-black truncate font-title w-11/12\" translate>\n        record.metadata.api.form.create\n      </div>\n      <button\n        (click)=\"resetUrl()\"\n        class=\"bg-primary-opacity-50 inline-flex items-center justify-center px-2 py-1 text-13 font-medium leading-none text-white rounded capitalize text-primary-lightest hover:bg-primary transition-colors\"\n      >\n        <p class=\"text-[13px] uppercase\" translate>\n          record.metadata.api.form.reset\n        </p>\n      </button>\n    </div>\n    <div class=\"flex flex-row flex-wrap justify-between grow gap-5\">\n      <div class=\"flex flex-col gap-3\">\n        <p class=\"text-[14px]\" translate>record.metadata.api.form.limit</p>\n        <div class=\"flex flex-row items-center gap-2\">\n          <gn-ui-text-input\n            class=\"mr-2 w-20\"\n            (valueChange)=\"setLimit($event)\"\n            [value]=\"displayLimit$ | async\"\n            placeholder=\"\"\n          >\n          </gn-ui-text-input>\n          <div class=\"flex items-center\">\n            <input\n              class=\"mr-2 cursor-pointer\"\n              type=\"checkbox\"\n              [checked]=\"noLimitChecked$ | async\"\n              (change)=\"setLimit('-1')\"\n            />\n            <span class=\"text-sm\" translate\n              >record.metadata.api.form.limit.all</span\n            >\n          </div>\n        </div>\n      </div>\n      <div class=\"flex flex-col gap-3 relative\">\n        <p class=\"text-sm\" [class.text-gray-600]=\"!supportOffset\" translate>\n          record.metadata.api.form.offset\n        </p>\n        <div class=\"flex items-center\">\n          <gn-ui-text-input\n            class=\"w-20\"\n            [value]=\"offset$ | async\"\n            [disabled]=\"!supportOffset\"\n            (valueChange)=\"supportOffset ? setOffset($event) : null\"\n            placeholder=\"\"\n          >\n          </gn-ui-text-input>\n          <div\n            *ngIf=\"!supportOffset\"\n            class=\"flex items-center gap-2 text-orange-500 z-10 ml-3\"\n          >\n            <span\n              class=\"material-symbols-outlined\"\n              matTooltip=\"Not supported on this service\"\n            >\n              warning\n            </span>\n          </div>\n        </div>\n      </div>\n      <div class=\"flex flex-col gap-3\">\n        <p class=\"text-sm\" translate>record.metadata.api.form.type</p>\n        <gn-ui-dropdown-selector\n          #dropdown\n          [title]=\"''\"\n          extraBtnClass=\"secondary min-w-full !w-40 !text-black\"\n          [showTitle]=\"false\"\n          class=\"text-black\"\n          [choices]=\"outputFormats\"\n          (selectValue)=\"setFormat($event)\"\n          [selected]=\"format$ | async\"\n        ></gn-ui-dropdown-selector>\n      </div>\n    </div>\n  </div>\n  <div class=\"flex flex-col gap-3 mb-3\">\n    <div class=\"text-sm text-black truncate font-title w-11/12\" translate>\n      record.metadata.api.form.customUrl\n    </div>\n    <div class=\"bg-white rounded-lg\">\n      <gn-ui-copy-text-button\n        [text]=\"apiQueryUrl$ | async\"\n      ></gn-ui-copy-text-button>\n    </div>\n  </div>\n</div>\n"]}
|
|
140
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"record-api-form.component.js","sourceRoot":"","sources":["../../../../../../../../libs/ui/elements/src/lib/record-api-form/record-api-form.component.ts","../../../../../../../../libs/ui/elements/src/lib/record-api-form/record-api-form.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, Input } from '@angular/core'\nimport { MimeType, OgcApiEndpoint, WfsEndpoint } from '@camptocamp/ogc-client'\nimport {\n  DatasetServiceDistribution,\n  ServiceProtocol,\n} from '../../../../../../libs/common/domain/src/lib/model/record'\nimport { mimeTypeToFormat } from '../../../../../../libs/util/shared/src'\nimport { BehaviorSubject, combineLatest, filter, map, switchMap } from 'rxjs'\nimport {\n  CopyTextButtonComponent,\n  DropdownChoice,\n  DropdownSelectorComponent,\n  TextInputComponent,\n} from '../../../../../../libs/ui/inputs/src'\nimport { CommonModule } from '@angular/common'\nimport { TranslateModule } from '@ngx-translate/core'\n\nconst DEFAULT_PARAMS = {\n  OFFSET: '',\n  LIMIT: '-1',\n  FORMAT: 'application/json',\n}\n\n@Component({\n  selector: 'gn-ui-record-api-form',\n  templateUrl: './record-api-form.component.html',\n  styleUrls: ['./record-api-form.component.css'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [\n    CommonModule,\n    TextInputComponent,\n    DropdownSelectorComponent,\n    CopyTextButtonComponent,\n    TranslateModule,\n  ],\n})\nexport class RecordApiFormComponent {\n  @Input() set apiLink(value: DatasetServiceDistribution) {\n    this.outputFormats = [{ value: 'application/json', label: 'JSON' }]\n    this.accessServiceProtocol = value ? value.accessServiceProtocol : undefined\n    this.apiFeatureType = value ? value.name : undefined\n    if (value) {\n      this.apiBaseUrl = value.url.href\n      this.createEndpoint().then(() => this.parseOutputFormats())\n    }\n    this.resetUrl()\n  }\n\n  offset$ = new BehaviorSubject(DEFAULT_PARAMS.OFFSET)\n  limit$ = new BehaviorSubject(DEFAULT_PARAMS.LIMIT)\n  format$ = new BehaviorSubject(DEFAULT_PARAMS.FORMAT)\n  endpoint$ = new BehaviorSubject<WfsEndpoint | OgcApiEndpoint | undefined>(\n    undefined\n  )\n  apiBaseUrl: string\n  apiFeatureType: string\n  supportOffset = true\n  accessServiceProtocol: ServiceProtocol | undefined\n  outputFormats: DropdownChoice[] = [\n    { value: 'application/json', label: 'JSON' },\n  ]\n  endpoint: WfsEndpoint | OgcApiEndpoint | undefined\n\n  apiQueryUrl$ = combineLatest([\n    this.offset$,\n    this.limit$,\n    this.format$,\n    // only compute the url if the endpoint was created\n    this.endpoint$.pipe(filter((endpoint) => !!endpoint)),\n  ]).pipe(\n    switchMap(([offset, limit, format]) =>\n      this.generateApiQueryUrl(offset, limit, format)\n    )\n  )\n\n  noLimitChecked$ = this.limit$.pipe(\n    map((limit) => limit === '-1' || limit === '')\n  )\n  displayLimit$ = this.limit$.pipe(\n    map((limit) => (limit !== '-1' ? limit : ''))\n  )\n\n  setOffset(value: string) {\n    this.offset$.next(value)\n  }\n\n  setLimit(value: string) {\n    this.limit$.next(value === '' ? '-1' : value)\n  }\n\n  setFormat(value: string | unknown) {\n    this.format$.next(String(value))\n  }\n\n  resetUrl() {\n    this.offset$.next(DEFAULT_PARAMS.OFFSET)\n    this.limit$.next(DEFAULT_PARAMS.LIMIT)\n    this.format$.next(DEFAULT_PARAMS.FORMAT)\n  }\n\n  async parseOutputFormats() {\n    if (!this.endpoint) return\n    const outputFormats = (await this.getOutputFormats()).map(\n      this.mimeTypeToFormatName\n    )\n\n    this.outputFormats = this.outputFormats\n      .concat(outputFormats.filter(Boolean))\n      .filter(\n        (format, index, self) =>\n          index === self.findIndex((t) => t.value === format.value)\n      )\n      .sort((a, b) => a.label.localeCompare(b.label))\n  }\n\n  mimeTypeToFormatName(mimeType: MimeType): DropdownChoice | null {\n    const formatName = mimeTypeToFormat(mimeType)\n    return formatName\n      ? { label: formatName.toUpperCase(), value: mimeType }\n      : null\n  }\n\n  async getOutputFormats(): Promise<MimeType[]> {\n    if (!this.endpoint) return []\n    if (this.endpoint instanceof WfsEndpoint) {\n      this.supportOffset = this.endpoint.supportsStartIndex()\n      return this.endpoint.getServiceInfo().outputFormats\n    } else {\n      return (await this.endpoint.getCollectionInfo(this.apiFeatureType))\n        .itemFormats\n    }\n  }\n\n  async createEndpoint() {\n    if (!this.apiBaseUrl || !this.accessServiceProtocol) return\n    if (this.accessServiceProtocol === 'wfs') {\n      this.endpoint = new WfsEndpoint(this.apiBaseUrl)\n      await (this.endpoint as WfsEndpoint).isReady()\n    } else {\n      this.endpoint = new OgcApiEndpoint(this.apiBaseUrl)\n      const collections = await this.endpoint.allCollections\n      // if there's only one collection, use this instead of the name given in the link.\n      if (collections.length === 1) {\n        this.apiFeatureType = collections[0].name\n      }\n    }\n    this.endpoint$.next(this.endpoint)\n  }\n\n  async generateApiQueryUrl(\n    offset: string,\n    limit: string,\n    format: string\n  ): Promise<string> {\n    if (!this.apiBaseUrl || !this.endpoint || !this.apiFeatureType) return ''\n\n    const options = {\n      outputFormat: format,\n      startIndex: offset ? Number(offset) : undefined,\n      maxFeatures: limit !== '-1' ? Number(limit) : undefined,\n      limit: limit !== '-1' ? Number(limit) : -1,\n      offset: offset !== '' ? Number(offset) : undefined,\n      outputCrs:\n        format.toLowerCase().indexOf('json') > -1 ? 'EPSG:4326' : undefined,\n    }\n\n    if (this.endpoint instanceof WfsEndpoint) {\n      delete options.limit\n      options.maxFeatures = limit !== '-1' ? Number(limit) : undefined\n      return this.endpoint.getFeatureUrl(this.apiFeatureType, options)\n    } else {\n      delete options.outputCrs\n      return await this.endpoint.getCollectionItemsUrl(\n        this.apiFeatureType,\n        options\n      )\n    }\n  }\n}\n","<div class=\"flex flex-col gap-8\">\n  <div class=\"flex flex-col bg-white p-8 ng-star-inserted shadow-xl gap-8\">\n    <div class=\"flex flex-row\">\n      <div class=\"text-[16px] text-black truncate font-title w-11/12\" translate>\n        record.metadata.api.form.create\n      </div>\n      <button\n        (click)=\"resetUrl()\"\n        class=\"bg-primary-opacity-50 inline-flex items-center justify-center px-2 py-1 text-13 font-medium leading-none text-white rounded capitalize text-primary-lightest hover:bg-primary transition-colors\"\n      >\n        <p class=\"text-[13px] uppercase\" translate>\n          record.metadata.api.form.reset\n        </p>\n      </button>\n    </div>\n    <div class=\"flex flex-row flex-wrap justify-between grow gap-5\">\n      <div class=\"flex flex-col gap-3\">\n        <p class=\"text-[14px]\" translate>record.metadata.api.form.limit</p>\n        <div class=\"flex flex-row items-center gap-2\">\n          <gn-ui-text-input\n            class=\"mr-2 w-20\"\n            (valueChange)=\"setLimit($event)\"\n            [value]=\"displayLimit$ | async\"\n            placeholder=\"\"\n          >\n          </gn-ui-text-input>\n          <div class=\"flex items-center\">\n            <input\n              class=\"mr-2 cursor-pointer\"\n              type=\"checkbox\"\n              [checked]=\"noLimitChecked$ | async\"\n              (change)=\"setLimit('-1')\"\n            />\n            <span class=\"text-sm\" translate\n              >record.metadata.api.form.limit.all</span\n            >\n          </div>\n        </div>\n      </div>\n      <div class=\"flex flex-col gap-3 relative\">\n        <p class=\"text-sm\" [class.text-gray-600]=\"!supportOffset\" translate>\n          record.metadata.api.form.offset\n        </p>\n        <div class=\"flex items-center\">\n          <gn-ui-text-input\n            class=\"w-20\"\n            [value]=\"offset$ | async\"\n            [disabled]=\"!supportOffset\"\n            (valueChange)=\"supportOffset ? setOffset($event) : null\"\n            placeholder=\"\"\n          >\n          </gn-ui-text-input>\n          <div\n            *ngIf=\"!supportOffset\"\n            class=\"flex items-center gap-2 text-orange-500 z-10 ml-3\"\n          >\n            <span\n              class=\"material-symbols-outlined\"\n              matTooltip=\"Not supported on this service\"\n            >\n              warning\n            </span>\n          </div>\n        </div>\n      </div>\n      <div class=\"flex flex-col gap-3\">\n        <p class=\"text-sm\" translate>record.metadata.api.form.type</p>\n        <gn-ui-dropdown-selector\n          #dropdown\n          [title]=\"''\"\n          extraBtnClass=\"secondary min-w-full !w-40 !text-black\"\n          [showTitle]=\"false\"\n          class=\"text-black\"\n          [choices]=\"outputFormats\"\n          (selectValue)=\"setFormat($event)\"\n          [selected]=\"format$ | async\"\n        ></gn-ui-dropdown-selector>\n      </div>\n    </div>\n  </div>\n  <div class=\"flex flex-col gap-3 mb-3\">\n    <div class=\"text-sm text-black truncate font-title w-11/12\" translate>\n      record.metadata.api.form.customUrl\n    </div>\n    <div class=\"bg-white rounded-lg\">\n      <gn-ui-copy-text-button\n        [text]=\"apiQueryUrl$ | async\"\n      ></gn-ui-copy-text-button>\n    </div>\n  </div>\n</div>\n"]}
|