From 1714ebdd1b4e3d5c182b64aa6a3684f23a16be32 Mon Sep 17 00:00:00 2001 From: Olivia Guyot Date: Mon, 25 Mar 2024 11:55:58 +0100 Subject: [PATCH 01/14] feat(dh): move keywords back below abstract, improve badges --- .../metadata-info.component.html | 26 ++++++++++--------- .../src/lib/badge/badge.component.html | 4 ++- 2 files changed, 17 insertions(+), 13 deletions(-) diff --git a/libs/ui/elements/src/lib/metadata-info/metadata-info.component.html b/libs/ui/elements/src/lib/metadata-info/metadata-info.component.html index 8977999eb4..8369e1aa7b 100644 --- a/libs/ui/elements/src/lib/metadata-info/metadata-info.component.html +++ b/libs/ui/elements/src/lib/metadata-info/metadata-info.component.html @@ -7,6 +7,20 @@ > +
+

+ record.metadata.keywords +

+
+ {{ keyword.label }} +
+
@@ -234,17 +248,5 @@
> -
-

record.metadata.keywords

-
- {{ keyword.label }} -
-
diff --git a/libs/ui/widgets/src/lib/badge/badge.component.html b/libs/ui/widgets/src/lib/badge/badge.component.html index a204b9c40d..8c439407a0 100644 --- a/libs/ui/widgets/src/lib/badge/badge.component.html +++ b/libs/ui/widgets/src/lib/badge/badge.component.html @@ -1,7 +1,9 @@
From cc83a29a49ba4dc9dbbb7d338485ac9f67845f88 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 8 Apr 2024 15:14:37 +0000 Subject: [PATCH 02/14] chore(deps-dev): bump axios from 0.27.2 to 1.6.8 Bumps [axios](https://github.com/axios/axios) from 0.27.2 to 1.6.8. - [Release notes](https://github.com/axios/axios/releases) - [Changelog](https://github.com/axios/axios/blob/v1.x/CHANGELOG.md) - [Commits](https://github.com/axios/axios/compare/v0.27.2...v1.6.8) --- updated-dependencies: - dependency-name: axios dependency-type: indirect ... Signed-off-by: dependabot[bot] --- package-lock.json | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/package-lock.json b/package-lock.json index 3f8ddd5125..9e4aca1a0e 100644 --- a/package-lock.json +++ b/package-lock.json @@ -14578,11 +14578,11 @@ "devOptional": true }, "node_modules/axios": { - "version": "1.6.0", - "resolved": "https://registry.npmjs.org/axios/-/axios-1.6.0.tgz", - "integrity": "sha512-EZ1DYihju9pwVB+jg67ogm+Tmqc6JmhamRN6I4Zt8DfZu5lbcQGw3ozH9lFejSJgs/ibaef3A9PMXPLeefFGJg==", + "version": "1.6.8", + "resolved": "https://registry.npmjs.org/axios/-/axios-1.6.8.tgz", + "integrity": "sha512-v/ZHtJDU39mDpyBoFVkETcd/uNdxrWRrg3bKpOKzXFA6Bvqopts6ALSMU3y6ijYxbw2B+wPrIv46egTzJXCLGQ==", "dependencies": { - "follow-redirects": "^1.15.0", + "follow-redirects": "^1.15.6", "form-data": "^4.0.0", "proxy-from-env": "^1.1.0" } From 72e6273d8b3a3cc59e81f4104b76f8cc966ed0a6 Mon Sep 17 00:00:00 2001 From: Tobias Kohr Date: Mon, 8 Apr 2024 17:22:54 +0200 Subject: [PATCH 03/14] feat(router): allow router.config to be used in extended router services also using npm package --- libs/feature/router/src/lib/default/index.ts | 1 + libs/feature/router/src/lib/default/router.service.ts | 2 +- 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/libs/feature/router/src/lib/default/index.ts b/libs/feature/router/src/lib/default/index.ts index 6b87a6da5a..bfd9dedce7 100644 --- a/libs/feature/router/src/lib/default/index.ts +++ b/libs/feature/router/src/lib/default/index.ts @@ -2,6 +2,7 @@ export * from './router.module' export * from './constants' export * from './state/router.facade' export * from './router.service' +export * from './router.config' export * from './state/router.effects' export * from './state/router.facade' export * from './container/search-router.container.directive' diff --git a/libs/feature/router/src/lib/default/router.service.ts b/libs/feature/router/src/lib/default/router.service.ts index 39d41a9ac3..6020fcc566 100644 --- a/libs/feature/router/src/lib/default/router.service.ts +++ b/libs/feature/router/src/lib/default/router.service.ts @@ -8,7 +8,7 @@ import { ROUTER_CONFIG, RouterConfigModel } from './router.config' }) export class RouterService { constructor( - @Inject(ROUTER_CONFIG) private routerConfig: RouterConfigModel, + @Inject(ROUTER_CONFIG) protected routerConfig: RouterConfigModel, private router: Router ) {} From 816b02eba075ff5706f86dcc2052dad98558b23f Mon Sep 17 00:00:00 2001 From: Olivia Guyot Date: Tue, 9 Apr 2024 10:21:35 +0200 Subject: [PATCH 04/14] e2e(datahub): wait more precisely for favorite to be registered Before that the filtering on favorites was sometimes done too early --- apps/datahub-e2e/src/e2e/home.cy.ts | 7 ++++++- .../favorites/favorite-star/favorite-star.component.html | 6 ++++-- .../favorite-star/favorite-star.component.spec.ts | 6 +++--- 3 files changed, 13 insertions(+), 6 deletions(-) diff --git a/apps/datahub-e2e/src/e2e/home.cy.ts b/apps/datahub-e2e/src/e2e/home.cy.ts index e8465065f5..feee999e30 100644 --- a/apps/datahub-e2e/src/e2e/home.cy.ts +++ b/apps/datahub-e2e/src/e2e/home.cy.ts @@ -96,7 +96,12 @@ describe('home', () => { .invoke('text') .as('favoriteTitle') cy.get('@favoriteItem').find('gn-ui-favorite-star button').click() - cy.wait(100) + + // wait for the favorite count to change before filtering + cy.get('@favoriteItem') + .find('[data-test=favorite-count]') + .invoke('text') + .should('eq', '1') // show my favorites only cy.get('datahub-header-badge-button[label$=favorites] button').click({ diff --git a/libs/feature/search/src/lib/favorites/favorite-star/favorite-star.component.html b/libs/feature/search/src/lib/favorites/favorite-star/favorite-star.component.html index ad071d7bd2..78c5fb604d 100644 --- a/libs/feature/search/src/lib/favorites/favorite-star/favorite-star.component.html +++ b/libs/feature/search/src/lib/favorites/favorite-star/favorite-star.component.html @@ -1,6 +1,7 @@
{{ favoriteCount }} @@ -10,7 +11,8 @@ [disabled]="loading || (isAnonymous$ | async)" > diff --git a/libs/feature/search/src/lib/favorites/favorite-star/favorite-star.component.spec.ts b/libs/feature/search/src/lib/favorites/favorite-star/favorite-star.component.spec.ts index b2d3272a07..7c01e1502f 100644 --- a/libs/feature/search/src/lib/favorites/favorite-star/favorite-star.component.spec.ts +++ b/libs/feature/search/src/lib/favorites/favorite-star/favorite-star.component.spec.ts @@ -88,7 +88,7 @@ describe('FavoriteStarComponent', () => { }) it('shows the amount of favorites on the record', () => { favoriteCountHTMLEl = fixture.debugElement.query( - By.css('.favorite-count') + By.css('[data-test=favorite-count]') ).nativeElement expect(favoriteCountHTMLEl).toBeTruthy() expect(favoriteCountHTMLEl.textContent).toEqual( @@ -103,7 +103,7 @@ describe('FavoriteStarComponent', () => { }) it('does not show the amount of favorites on the record', () => { const favoriteCountEl = fixture.debugElement.query( - By.css('.favorite-count') + By.css('[data-test=favorite-count]') ) expect(favoriteCountEl).toBeFalsy() }) @@ -182,7 +182,7 @@ describe('FavoriteStarComponent', () => { } fixture.detectChanges() favoriteCountHTMLEl = fixture.debugElement.query( - By.css('.favorite-count') + By.css('[data-test=favorite-count]') ).nativeElement }) describe('When my record is part of the updates', () => { From 4a3257fb475afe171673d448bb0a21a34db71692 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Laure-H=C3=A9l=C3=A8ne=20Bruneton?= Date: Tue, 5 Mar 2024 15:19:29 +0100 Subject: [PATCH 05/14] feat(ui-elements): switch thumbnail to standalone --- libs/ui/elements/src/lib/thumbnail/thumbnail.component.spec.ts | 3 +-- libs/ui/elements/src/lib/thumbnail/thumbnail.component.ts | 3 +++ libs/ui/elements/src/lib/ui-elements.module.ts | 2 +- 3 files changed, 5 insertions(+), 3 deletions(-) diff --git a/libs/ui/elements/src/lib/thumbnail/thumbnail.component.spec.ts b/libs/ui/elements/src/lib/thumbnail/thumbnail.component.spec.ts index 598a4fc5a4..10bfdf5fff 100644 --- a/libs/ui/elements/src/lib/thumbnail/thumbnail.component.spec.ts +++ b/libs/ui/elements/src/lib/thumbnail/thumbnail.component.spec.ts @@ -16,8 +16,7 @@ describe('ThumbnailComponent', () => { beforeEach(async () => { await TestBed.configureTestingModule({ - imports: [UtilSharedModule], - declarations: [ThumbnailComponent], + imports: [ThumbnailComponent, UtilSharedModule], }) .overrideComponent(ThumbnailComponent, { set: { changeDetection: ChangeDetectionStrategy.Default }, diff --git a/libs/ui/elements/src/lib/thumbnail/thumbnail.component.ts b/libs/ui/elements/src/lib/thumbnail/thumbnail.component.ts index 239dc4ca52..9d1e292980 100644 --- a/libs/ui/elements/src/lib/thumbnail/thumbnail.component.ts +++ b/libs/ui/elements/src/lib/thumbnail/thumbnail.component.ts @@ -1,3 +1,4 @@ +import { CommonModule } from '@angular/common' import { ChangeDetectionStrategy, Component, @@ -32,6 +33,8 @@ type FitOptions = 'cover' | 'contain' | 'scale-down' selector: 'gn-ui-thumbnail', templateUrl: './thumbnail.component.html', changeDetection: ChangeDetectionStrategy.OnPush, + standalone: true, + imports: [CommonModule], }) export class ThumbnailComponent implements OnInit, OnChanges { @Input() thumbnailUrl: string | string[] diff --git a/libs/ui/elements/src/lib/ui-elements.module.ts b/libs/ui/elements/src/lib/ui-elements.module.ts index 7a6feab03f..d2f7f9d2ab 100644 --- a/libs/ui/elements/src/lib/ui-elements.module.ts +++ b/libs/ui/elements/src/lib/ui-elements.module.ts @@ -46,6 +46,7 @@ import { ImageOverlayPreviewComponent } from './image-overlay-preview/image-over FormsModule, NgOptimizedImage, MarkdownParserComponent, + ThumbnailComponent, ], declarations: [ MetadataInfoComponent, @@ -61,7 +62,6 @@ import { ImageOverlayPreviewComponent } from './image-overlay-preview/image-over MetadataQualityItemComponent, SearchResultsErrorComponent, PaginationComponent, - ThumbnailComponent, AvatarComponent, UserPreviewComponent, GnUiLinkifyDirective, From ad242372a7011e0295719b5f96ed548c82d221a3 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Laure-H=C3=A9l=C3=A8ne=20Bruneton?= Date: Tue, 5 Mar 2024 15:20:59 +0100 Subject: [PATCH 06/14] refactor(ui-elements): better thumbnail types --- libs/ui/elements/src/lib/thumbnail/thumbnail.component.ts | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/libs/ui/elements/src/lib/thumbnail/thumbnail.component.ts b/libs/ui/elements/src/lib/thumbnail/thumbnail.component.ts index 9d1e292980..4e27b9f4db 100644 --- a/libs/ui/elements/src/lib/thumbnail/thumbnail.component.ts +++ b/libs/ui/elements/src/lib/thumbnail/thumbnail.component.ts @@ -19,16 +19,15 @@ export const THUMBNAIL_PLACEHOLDER = new InjectionToken( 'thumbnail-placeholder' ) +type FitOptions = 'cover' | 'contain' | 'scale-down' type ThumbnailImageObject = { url: string - fit?: 'cover' | 'contain' | 'scale-down' + fit?: FitOptions } const DEFAULT_PLACEHOLDER = '' -type FitOptions = 'cover' | 'contain' | 'scale-down' - @Component({ selector: 'gn-ui-thumbnail', templateUrl: './thumbnail.component.html', From d677121074927ffac71129497854ec3d2c957bd6 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Laure-H=C3=A9l=C3=A8ne=20Bruneton?= Date: Tue, 5 Mar 2024 15:21:49 +0100 Subject: [PATCH 07/14] feat(ui-elements): thumbnail stories --- .../thumbnail/thumbnail.component.stories.ts | 22 +++++++++++++++++++ 1 file changed, 22 insertions(+) create mode 100644 libs/ui/elements/src/lib/thumbnail/thumbnail.component.stories.ts diff --git a/libs/ui/elements/src/lib/thumbnail/thumbnail.component.stories.ts b/libs/ui/elements/src/lib/thumbnail/thumbnail.component.stories.ts new file mode 100644 index 0000000000..14e4f60552 --- /dev/null +++ b/libs/ui/elements/src/lib/thumbnail/thumbnail.component.stories.ts @@ -0,0 +1,22 @@ +import { + componentWrapperDecorator, + Meta, + moduleMetadata, + StoryObj, +} from '@storybook/angular' +import { ThumbnailComponent } from './thumbnail.component' + +export default { + title: 'Elements/ThumbnailComponent', + component: ThumbnailComponent, + decorators: [ + moduleMetadata({ + imports: [ThumbnailComponent], + }), + componentWrapperDecorator( + (story) => `
${story}
` + ), + ], +} as Meta + +export const Primary: StoryObj = {} From 9452e06f7ef653e83ab1a926a30e56e27dc58331 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Laure-H=C3=A9l=C3=A8ne=20Bruneton?= Date: Wed, 6 Mar 2024 11:09:42 +0100 Subject: [PATCH 08/14] feat(utils): add functions to downgrade image to max size --- .../shared/src/lib/utils/bytes-convert.ts | 3 + .../util/shared/src/lib/utils/image-resize.ts | 72 +++++++++++++++++++ libs/util/shared/src/lib/utils/index.ts | 12 ++-- 3 files changed, 82 insertions(+), 5 deletions(-) create mode 100644 libs/util/shared/src/lib/utils/bytes-convert.ts create mode 100644 libs/util/shared/src/lib/utils/image-resize.ts diff --git a/libs/util/shared/src/lib/utils/bytes-convert.ts b/libs/util/shared/src/lib/utils/bytes-convert.ts new file mode 100644 index 0000000000..a49f1494a7 --- /dev/null +++ b/libs/util/shared/src/lib/utils/bytes-convert.ts @@ -0,0 +1,3 @@ +export function megabytesToBytes(megabytes) { + return megabytes * 1048576 +} diff --git a/libs/util/shared/src/lib/utils/image-resize.ts b/libs/util/shared/src/lib/utils/image-resize.ts new file mode 100644 index 0000000000..7737d7ae97 --- /dev/null +++ b/libs/util/shared/src/lib/utils/image-resize.ts @@ -0,0 +1,72 @@ +export function downsizeImage( + blob: Blob, + maxWidth: number, + maxHeight: number +): Promise { + return new Promise((resolve, reject) => { + const image = new Image() + image.src = URL.createObjectURL(blob) + image.onload = () => { + let width = image.width + let height = image.height + + if (width > maxWidth || height > maxHeight) { + if (width > height) { + height = height * (maxWidth / width) + width = maxWidth + } else { + width = width * (maxHeight / height) + height = maxHeight + } + } + + const canvas = document.createElement('canvas') + canvas.width = width + canvas.height = height + + const context = canvas.getContext('2d') + context.drawImage(image, 0, 0, width, height) + + canvas.toBlob(resolve, blob.type) + } + image.onerror = reject + }) +} + +export function downgradeImage( + blob: Blob, + maxSizeBytes: number +): Promise { + return new Promise((resolve, reject) => { + const image = new Image() + image.src = URL.createObjectURL(blob) + image.onload = () => { + const width = image.width + const height = image.height + let quality = 1.0 + + const canvas = document.createElement('canvas') + canvas.width = width + canvas.height = height + + const context = canvas.getContext('2d') + context.drawImage(image, 0, 0, width, height) + + const compressAndResolveBlob = (blobToCompress: Blob) => { + if (blobToCompress.size <= maxSizeBytes) { + resolve(blobToCompress) + } else { + quality -= 0.1 + if (quality >= 0) { + canvas.toBlob(compressAndResolveBlob, blob.type, quality) + } else { + reject('Unable to compress image below max size') + } + } + } + + canvas.toBlob(compressAndResolveBlob, blob.type, quality) + } + image.onerror = reject + }) +} diff --git a/libs/util/shared/src/lib/utils/index.ts b/libs/util/shared/src/lib/utils/index.ts index f8850de052..d3bfe45bdb 100644 --- a/libs/util/shared/src/lib/utils/index.ts +++ b/libs/util/shared/src/lib/utils/index.ts @@ -1,9 +1,11 @@ +export * from './bytes-convert' +export * from './event' +export * from './fuzzy-filter' +export * from './geojson' +export * from './image-resize' export * from './parse' -export * from './strip-html' export * from './remove-whitespace' -export * from './geojson' export * from './sort-by' -export * from './url' -export * from './event' -export * from './fuzzy-filter' +export * from './strip-html' export * from './temporal-extent-union' +export * from './url' From d9eb6d4e6c9cef1d3778900d4b3b9f5199c222fb Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Laure-H=C3=A9l=C3=A8ne=20Bruneton?= Date: Fri, 15 Mar 2024 13:43:49 +0100 Subject: [PATCH 09/14] feat(ui-inputs): add files drop directive --- .../lib/files-drop/files-drop.directive.ts | 45 +++++++++++++++++++ 1 file changed, 45 insertions(+) create mode 100644 libs/ui/inputs/src/lib/files-drop/files-drop.directive.ts diff --git a/libs/ui/inputs/src/lib/files-drop/files-drop.directive.ts b/libs/ui/inputs/src/lib/files-drop/files-drop.directive.ts new file mode 100644 index 0000000000..a70e1b4d9b --- /dev/null +++ b/libs/ui/inputs/src/lib/files-drop/files-drop.directive.ts @@ -0,0 +1,45 @@ +import { Directive, HostListener, Output, EventEmitter } from '@angular/core' + +@Directive({ + selector: '[gnUiFilesDrop]', + standalone: true, +}) +export class FilesDropDirective { + @Output() dragFilesOver: EventEmitter = new EventEmitter() + @Output() dropFiles: EventEmitter = new EventEmitter() + + dragEnterCounter = 0 + + @HostListener('dragenter', ['$event']) + _onDragEnter(event: DragEvent) { + event.preventDefault() + this.dragEnterCounter++ + this.dragFilesOver.emit(true) + } + + @HostListener('dragover', ['$event']) + _onDragOver(event: DragEvent) { + event.preventDefault() + } + + @HostListener('dragleave', ['$event']) + _onDragLeave(event: DragEvent) { + event.preventDefault() + this.dragEnterCounter = Math.max(0, this.dragEnterCounter - 1) + if (this.dragEnterCounter === 0) { + this.dragFilesOver.emit(false) + } + } + + @HostListener('drop', ['$event']) + _onDrop(event: DragEvent) { + event.preventDefault() + this.dragEnterCounter = 0 + this.dragFilesOver.emit(false) + + const files = Array.from(event.dataTransfer.files) + if (files.length > 0) { + this.dropFiles.emit(files) + } + } +} From d89ac1d7055b57dc40a7bc1f86ba6455d51a93db Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Laure-H=C3=A9l=C3=A8ne=20Bruneton?= Date: Tue, 5 Mar 2024 15:22:44 +0100 Subject: [PATCH 10/14] feat(ui-inputs): add image input component --- .../lib/image-input/image-input.component.css | 0 .../image-input/image-input.component.html | 146 +++++++++++++ .../image-input/image-input.component.spec.ts | 159 +++++++++++++++ .../image-input.component.stories.ts | 126 ++++++++++++ .../lib/image-input/image-input.component.ts | 192 ++++++++++++++++++ libs/ui/inputs/src/lib/ui-inputs.module.ts | 3 + 6 files changed, 626 insertions(+) create mode 100644 libs/ui/inputs/src/lib/image-input/image-input.component.css create mode 100644 libs/ui/inputs/src/lib/image-input/image-input.component.html create mode 100644 libs/ui/inputs/src/lib/image-input/image-input.component.spec.ts create mode 100644 libs/ui/inputs/src/lib/image-input/image-input.component.stories.ts create mode 100644 libs/ui/inputs/src/lib/image-input/image-input.component.ts diff --git a/libs/ui/inputs/src/lib/image-input/image-input.component.css b/libs/ui/inputs/src/lib/image-input/image-input.component.css new file mode 100644 index 0000000000..e69de29bb2 diff --git a/libs/ui/inputs/src/lib/image-input/image-input.component.html b/libs/ui/inputs/src/lib/image-input/image-input.component.html new file mode 100644 index 0000000000..2accf0801b --- /dev/null +++ b/libs/ui/inputs/src/lib/image-input/image-input.component.html @@ -0,0 +1,146 @@ + + + + +
+
+ + + delete + +
+ +
+ + delete + Supprimer + + + add + Texte alternatif + +
+
+
+ + +
+ +
+ + link + Saisir une URL + +
+
+
+
+
+ link + + + arrow_upward + +
+
+
+
+
diff --git a/libs/ui/inputs/src/lib/image-input/image-input.component.spec.ts b/libs/ui/inputs/src/lib/image-input/image-input.component.spec.ts new file mode 100644 index 0000000000..4a470b9d73 --- /dev/null +++ b/libs/ui/inputs/src/lib/image-input/image-input.component.spec.ts @@ -0,0 +1,159 @@ +import { ComponentFixture, TestBed, waitForAsync } from '@angular/core/testing' + +import { HttpHeaders } from '@angular/common/http' +import { + HttpClientTestingModule, + HttpTestingController, +} from '@angular/common/http/testing' +import { TranslateModule } from '@ngx-translate/core' +import { ImageInputComponent } from './image-input.component' + +describe('ImageInputComponent', () => { + let component: ImageInputComponent + let fixture: ComponentFixture + let httpTestingController: HttpTestingController + + beforeEach(async () => { + await TestBed.configureTestingModule({ + imports: [ + ImageInputComponent, + HttpClientTestingModule, + TranslateModule.forRoot(), + ], + }).compileComponents() + httpTestingController = TestBed.inject(HttpTestingController) + }) + + beforeEach(() => { + fixture = TestBed.createComponent(ImageInputComponent) + component = fixture.componentInstance + fixture.detectChanges() + }) + + it('should create', () => { + expect(component).toBeTruthy() + }) + + describe('file', () => { + it('should filter only image type files', () => { + const someNonImageFile = new File([], 'someNonImageFile', { + type: 'text/plain', + }) + const someImageFile = new File([], 'someImageFile', { type: 'image/png' }) + const result = component.filterTypeImage([ + someNonImageFile, + someImageFile, + ]) + expect(result).toEqual([someImageFile]) + }) + }) + + describe('url', () => { + const testMaxSizeMB = 1 + const testUrl = 'http://test.com/image.png' + + beforeEach(() => { + component.maxSizeMB = testMaxSizeMB + component.urlInputValue = testUrl + }) + + it('should emit the downloaded file on nominal case', waitForAsync(() => { + jest.spyOn(component.fileChange, 'emit') + + component.downloadUrl() + + const reqHead = httpTestingController.expectOne(testUrl) + expect(reqHead.request.method).toEqual('HEAD') + + const responseHeaders = new HttpHeaders() + .set('content-type', 'image/png') + .set('content-length', '1048575') + reqHead.flush(null, { + headers: responseHeaders, + status: 200, + statusText: 'OK', + }) + + setTimeout(() => { + const reqGet = httpTestingController.expectOne(testUrl) + expect(reqGet.request.method).toEqual('GET') + + reqGet.flush(new Blob()) + + expect(component.fileChange.emit).toHaveBeenCalled() + + httpTestingController.verify() + }, 0) + })) + + it('should not download the file when content-type is not image', waitForAsync(() => { + component.downloadUrl() + + const reqHead = httpTestingController.expectOne(testUrl) + expect(reqHead.request.method).toEqual('HEAD') + + const responseHeaders = new HttpHeaders() + .set('content-type', 'text/plain') + .set('content-length', '1048575') + reqHead.flush(null, { + headers: responseHeaders, + status: 200, + statusText: 'OK', + }) + + httpTestingController.verify() + })) + + it('should not download the file when content-length is above limit', waitForAsync(() => { + component.downloadUrl() + + const reqHead = httpTestingController.expectOne(testUrl) + expect(reqHead.request.method).toEqual('HEAD') + + const responseHeaders = new HttpHeaders() + .set('content-type', 'image/png') + .set('content-length', '1048577') + reqHead.flush(null, { + headers: responseHeaders, + status: 200, + statusText: 'OK', + }) + + httpTestingController.verify() + })) + + it('should emit the file URL when encountering a download error', waitForAsync(() => { + jest.spyOn(component.urlChange, 'emit') + + component.downloadUrl() + + const reqHead = httpTestingController.expectOne(testUrl) + expect(reqHead.request.method).toEqual('HEAD') + + const responseHeaders = new HttpHeaders() + .set('content-type', 'image/png') + .set('content-length', '1048575') + reqHead.flush(null, { + headers: responseHeaders, + status: 200, + statusText: 'OK', + }) + + setTimeout(() => { + const reqGet = httpTestingController.expectOne(testUrl) + expect(reqGet.request.method).toEqual('GET') + + const testError = new ProgressEvent('error', { + lengthComputable: false, + loaded: 0, + total: 0, + }) + reqGet.error(testError) + + expect(component.urlChange.emit).toHaveBeenCalled() + + httpTestingController.verify() + }, 0) + })) + }) +}) diff --git a/libs/ui/inputs/src/lib/image-input/image-input.component.stories.ts b/libs/ui/inputs/src/lib/image-input/image-input.component.stories.ts new file mode 100644 index 0000000000..8fe42384d2 --- /dev/null +++ b/libs/ui/inputs/src/lib/image-input/image-input.component.stories.ts @@ -0,0 +1,126 @@ +import { + applicationConfig, + Meta, + moduleMetadata, + StoryObj, +} from '@storybook/angular' +import { ImageInputComponent } from './image-input.component' +import { importProvidersFrom } from '@angular/core' +import { HttpClientModule } from '@angular/common/http' + +export default { + title: 'Inputs/ImageInputComponent', + component: ImageInputComponent, + decorators: [ + applicationConfig({ + providers: [importProvidersFrom(HttpClientModule)], + }), + moduleMetadata({ + imports: [ImageInputComponent], + }), + ], +} as Meta + +export const WithoutImage: StoryObj = { + args: { + maxSizeMB: 5, + }, + render: (args) => ({ + props: args, + template: ` +
+ + +
`, + }), +} + +export const WithImage: StoryObj = { + args: { + maxSizeMB: 5, + previewUrl: + '', + altText: 'Some alternative text', + }, + render: (args) => ({ + props: args, + template: ` +
+ +
`, + }), +} + +export const WithBrokenImage: StoryObj = { + args: { + maxSizeMB: 5, + previewUrl: 'https://broken/url', + altText: 'Some alternative text', + }, + render: (args) => ({ + props: args, + template: ` +
+ +
`, + }), +} + +export const UploadProgress5: StoryObj = { + args: { + maxSizeMB: 5, + uploadProgress: 5, + }, + render: (args) => ({ + props: args, + template: ` +
+ +
`, + }), +} + +export const UploadProgress95: StoryObj = { + args: { + maxSizeMB: 5, + uploadProgress: 95, + }, + render: (args) => ({ + props: args, + template: ` +
+ +
`, + }), +} + +export const UploadError: StoryObj = { + args: { + maxSizeMB: 5, + uploadError: true, + }, + render: (args) => ({ + props: args, + template: ` +
+ +
`, + }), +} diff --git a/libs/ui/inputs/src/lib/image-input/image-input.component.ts b/libs/ui/inputs/src/lib/image-input/image-input.component.ts new file mode 100644 index 0000000000..aee44be7c5 --- /dev/null +++ b/libs/ui/inputs/src/lib/image-input/image-input.component.ts @@ -0,0 +1,192 @@ +import { CommonModule } from '@angular/common' +import { HttpClient } from '@angular/common/http' +import { + ChangeDetectionStrategy, + ChangeDetectorRef, + Component, + ElementRef, + EventEmitter, + Input, + Output, + ViewChild, +} from '@angular/core' +import { MatIconModule } from '@angular/material/icon' +import { downgradeImage, megabytesToBytes } from '@geonetwork-ui/util/shared' +import { ButtonComponent } from '../button/button.component' +import { FilesDropDirective } from '../files-drop/files-drop.directive' +import { MatProgressSpinnerModule } from '@angular/material/progress-spinner' +import { firstValueFrom } from 'rxjs' + +@Component({ + selector: 'gn-ui-image-input', + templateUrl: './image-input.component.html', + styleUrls: ['./image-input.component.css'], + changeDetection: ChangeDetectionStrategy.OnPush, + standalone: true, + imports: [ + CommonModule, + ButtonComponent, + MatIconModule, + FilesDropDirective, + MatProgressSpinnerModule, + ], +}) +export class ImageInputComponent { + @Input() maxSizeMB: number + @Input() previewUrl?: string + @Input() altText?: string + @Input() uploadProgress?: number + @Input() uploadError?: boolean + @Output() fileChange: EventEmitter = new EventEmitter() + @Output() urlChange: EventEmitter = new EventEmitter() + @Output() uploadCancel: EventEmitter = new EventEmitter() + @Output() delete: EventEmitter = new EventEmitter() + @Output() altTextChange: EventEmitter = new EventEmitter() + + dragFilesOver = false + showUrlInput = false + downloadError = false + showAltTextInput = false + + urlInputValue?: string + lastUploadType?: 'file' | 'url' + lastUploadContent?: string | File + + constructor(private http: HttpClient, private cd: ChangeDetectorRef) {} + + getPrimaryText() { + if (this.uploadError) { + return "L'image n'a pas pu être chargée" + } + if (this.uploadProgress) { + return 'Chargement en cours...' + } + return 'Sélectionner une image' + } + + getSecondaryText() { + if (this.uploadError) { + return 'Réessayer' + } + if (this.uploadProgress) { + return 'Annuler' + } + return 'ou la glisser ici' + } + + handleDragFilesOver(dragFilesOver: boolean) { + if (!this.showUrlInput) { + this.dragFilesOver = dragFilesOver + this.cd.markForCheck() + } + } + + handleDropFiles(files: File[]) { + if (!this.showUrlInput) { + const validFiles = this.filterTypeImage(files) + if (validFiles.length > 0) { + this.resizeAndEmit(validFiles[0]) + } + } + } + + handleFileInput(event: Event) { + const inputFiles = Array.from((event.target as HTMLInputElement).files) + const validFiles = this.filterTypeImage(inputFiles) + if (validFiles.length > 0) { + this.resizeAndEmit(validFiles[0]) + } + } + + displayUrlInput() { + this.uploadCancel.emit() + this.showUrlInput = true + } + + handleUrlChange(event: Event) { + this.downloadError = false + this.urlInputValue = (event.target as HTMLInputElement).value + } + + async downloadUrl() { + const name = this.urlInputValue.split('/').pop() + + try { + const response = await firstValueFrom( + this.http.head(this.urlInputValue, { observe: 'response' }) + ) + if ( + response.headers.get('content-type')?.startsWith('image/') && + parseInt(response.headers.get('content-length')) < + megabytesToBytes(this.maxSizeMB) + ) { + this.http.get(this.urlInputValue, { responseType: 'blob' }).subscribe({ + next: (blob) => { + this.cd.markForCheck() + const file = new File([blob], name) + this.fileChange.emit(file) + }, + error: () => { + this.downloadError = true + this.cd.markForCheck() + this.urlChange.emit(this.urlInputValue) + }, + }) + } + } catch { + this.downloadError = true + this.cd.markForCheck() + return + } + } + + handleSecondaryTextClick() { + if (this.uploadError) { + this.handleRetry() + } else if (this.uploadProgress) { + this.handleCancel() + } + } + + handleCancel() { + this.uploadCancel.emit() + } + + handleRetry() { + switch (this.lastUploadType) { + case 'file': + this.fileChange.emit(this.lastUploadContent as File) + break + case 'url': + this.urlChange.emit(this.lastUploadContent as string) + break + } + } + + handleDelete() { + this.delete.emit() + } + + toggleAltTextInput() { + this.showAltTextInput = !this.showAltTextInput + } + + handleAltTextChange(event: Event) { + const input = event.target as HTMLInputElement + this.altTextChange.emit(input.value) + } + + private filterTypeImage(files: File[]) { + return files.filter((file) => { + return file.type.startsWith('image/') + }) + } + + private resizeAndEmit(imageToResize: File) { + const maxSizeBytes = megabytesToBytes(this.maxSizeMB) + downgradeImage(imageToResize, maxSizeBytes).then((resizedImage) => { + const fileToEmit = new File([resizedImage], imageToResize.name) + this.fileChange.emit(fileToEmit) + }) + } +} diff --git a/libs/ui/inputs/src/lib/ui-inputs.module.ts b/libs/ui/inputs/src/lib/ui-inputs.module.ts index 92b7b25c07..aba5bcbd30 100644 --- a/libs/ui/inputs/src/lib/ui-inputs.module.ts +++ b/libs/ui/inputs/src/lib/ui-inputs.module.ts @@ -42,6 +42,7 @@ import { MatInputModule } from '@angular/material/input' import { MatDatepickerModule } from '@angular/material/datepicker' import { MatNativeDateModule } from '@angular/material/core' import { EditableLabelDirective } from './editable-label/editable-label.directive' +import { ImageInputComponent } from './image-input/image-input.component' @NgModule({ declarations: [ @@ -89,6 +90,7 @@ import { EditableLabelDirective } from './editable-label/editable-label.directiv EditableLabelDirective, TextAreaComponent, ButtonComponent, + ImageInputComponent, ], exports: [ DropdownSelectorComponent, @@ -109,6 +111,7 @@ import { EditableLabelDirective } from './editable-label/editable-label.directiv SearchInputComponent, DateRangePickerComponent, EditableLabelDirective, + ImageInputComponent, ], }) export class UiInputsModule {} From 01743be9ca9cc0ce264752a2611dc26f1eeb9be5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Laure-H=C3=A9l=C3=A8ne=20Bruneton?= Date: Thu, 14 Mar 2024 11:33:26 +0100 Subject: [PATCH 11/14] feat(GN4): change attachments POST to multi-part --- .../gn4/src/openapi/.openapi-generator/FILES | 1 - .../src/openapi/api/records.api.service.ts | 55 +++++++++++++++---- .../openapi/model/inlineObject3.api.model.ts | 18 ------ .../gn4/src/openapi/model/models.ts | 1 - libs/data-access/gn4/src/spec.yaml | 2 +- 5 files changed, 44 insertions(+), 33 deletions(-) delete mode 100644 libs/data-access/gn4/src/openapi/model/inlineObject3.api.model.ts diff --git a/libs/data-access/gn4/src/openapi/.openapi-generator/FILES b/libs/data-access/gn4/src/openapi/.openapi-generator/FILES index 1f64e54901..1f2ab7c87a 100644 --- a/libs/data-access/gn4/src/openapi/.openapi-generator/FILES +++ b/libs/data-access/gn4/src/openapi/.openapi-generator/FILES @@ -74,7 +74,6 @@ model/iProcessingReport.api.model.ts model/iSODate.api.model.ts model/infoReport.api.model.ts model/inlineObject1.api.model.ts -model/inlineObject3.api.model.ts model/inlineObject4.api.model.ts model/isoLanguage.api.model.ts model/jSONObject.api.model.ts diff --git a/libs/data-access/gn4/src/openapi/api/records.api.service.ts b/libs/data-access/gn4/src/openapi/api/records.api.service.ts index 5273152d6c..04cb7f290c 100644 --- a/libs/data-access/gn4/src/openapi/api/records.api.service.ts +++ b/libs/data-access/gn4/src/openapi/api/records.api.service.ts @@ -30,7 +30,6 @@ import { import { ExtentDtoApiModel } from '../model/models' import { FeatureResponseApiModel } from '../model/models' import { IProcessingReportApiModel } from '../model/models' -import { InlineObject3ApiModel } from '../model/models' import { MetadataBatchApproveParameterApiModel } from '../model/models' import { MetadataBatchSubmitParameterApiModel } from '../model/models' import { MetadataCategoryApiModel } from '../model/models' @@ -75,6 +74,20 @@ export class RecordsApiService { this.encoder = this.configuration.encoder || new CustomHttpParameterCodec() } + /** + * @param consumes string[] mime-types + * @return true: consumes contains 'multipart/form-data', false: otherwise + */ + private canConsumeForm(consumes: string[]): boolean { + const form = 'multipart/form-data' + for (const consume of consumes) { + if (form === consume) { + return true + } + } + return false + } + private addToHttpParams( httpParams: HttpParams, value: any, @@ -8290,44 +8303,44 @@ export class RecordsApiService { /** * Create a new resource for a given metadata * @param metadataUuid The metadata UUID + * @param file The file to upload * @param visibility The sharing policy * @param approved Use approved version or not - * @param inlineObject3ApiModel * @param observe set whether or not to return the data Observable as the body, response or events. defaults to returning the body. * @param reportProgress flag to report request and response progress. */ public putResource( metadataUuid: string, + file: Blob, visibility?: 'public' | 'private', approved?: boolean, - inlineObject3ApiModel?: InlineObject3ApiModel, observe?: 'body', reportProgress?: boolean, options?: { httpHeaderAccept?: 'application/json' } ): Observable public putResource( metadataUuid: string, + file: Blob, visibility?: 'public' | 'private', approved?: boolean, - inlineObject3ApiModel?: InlineObject3ApiModel, observe?: 'response', reportProgress?: boolean, options?: { httpHeaderAccept?: 'application/json' } ): Observable> public putResource( metadataUuid: string, + file: Blob, visibility?: 'public' | 'private', approved?: boolean, - inlineObject3ApiModel?: InlineObject3ApiModel, observe?: 'events', reportProgress?: boolean, options?: { httpHeaderAccept?: 'application/json' } ): Observable> public putResource( metadataUuid: string, + file: Blob, visibility?: 'public' | 'private', approved?: boolean, - inlineObject3ApiModel?: InlineObject3ApiModel, observe: any = 'body', reportProgress: boolean = false, options?: { httpHeaderAccept?: 'application/json' } @@ -8337,6 +8350,11 @@ export class RecordsApiService { 'Required parameter metadataUuid was null or undefined when calling putResource.' ) } + if (file === null || file === undefined) { + throw new Error( + 'Required parameter file was null or undefined when calling putResource.' + ) + } let queryParameters = new HttpParams({ encoder: this.encoder }) if (visibility !== undefined && visibility !== null) { @@ -8369,11 +8387,24 @@ export class RecordsApiService { } // to determine the Content-Type header - const consumes: string[] = ['application/json'] - const httpContentTypeSelected: string | undefined = - this.configuration.selectHeaderContentType(consumes) - if (httpContentTypeSelected !== undefined) { - headers = headers.set('Content-Type', httpContentTypeSelected) + const consumes: string[] = ['multipart/form-data'] + + const canConsumeForm = this.canConsumeForm(consumes) + + let formParams: { append(param: string, value: any): any } + let useForm = false + let convertFormParamsToString = false + // use FormData to transmit files using content-type "multipart/form-data" + // see https://stackoverflow.com/questions/4007969/application-x-www-form-urlencoded-or-multipart-form-data + useForm = canConsumeForm + if (useForm) { + formParams = new FormData() + } else { + formParams = new HttpParams({ encoder: this.encoder }) + } + + if (file !== undefined) { + formParams = (formParams.append('file', file) as any) || formParams } let responseType_: 'text' | 'json' = 'json' @@ -8388,7 +8419,7 @@ export class RecordsApiService { `${this.configuration.basePath}/records/${encodeURIComponent( String(metadataUuid) )}/attachments`, - inlineObject3ApiModel, + convertFormParamsToString ? formParams.toString() : formParams, { params: queryParameters, responseType: responseType_, diff --git a/libs/data-access/gn4/src/openapi/model/inlineObject3.api.model.ts b/libs/data-access/gn4/src/openapi/model/inlineObject3.api.model.ts deleted file mode 100644 index f776d3ee1b..0000000000 --- a/libs/data-access/gn4/src/openapi/model/inlineObject3.api.model.ts +++ /dev/null @@ -1,18 +0,0 @@ -/** - * GeoNetwork 4.2.7 OpenAPI Documentation - * This is the description of the GeoNetwork OpenAPI. Use this API to manage your catalog. - * - * The version of the OpenAPI document: 4.2.7 - * Contact: geonetwork-users@lists.sourceforge.net - * - * NOTE: This class is auto generated by OpenAPI Generator (https://openapi-generator.tech). - * https://openapi-generator.tech - * Do not edit the class manually. - */ - -export interface InlineObject3ApiModel { - /** - * The file to upload - */ - file: Blob -} diff --git a/libs/data-access/gn4/src/openapi/model/models.ts b/libs/data-access/gn4/src/openapi/model/models.ts index a26e92c1db..9aa45f7830 100644 --- a/libs/data-access/gn4/src/openapi/model/models.ts +++ b/libs/data-access/gn4/src/openapi/model/models.ts @@ -33,7 +33,6 @@ export * from './iProcessingReport.api.model' export * from './iSODate.api.model' export * from './infoReport.api.model' export * from './inlineObject1.api.model' -export * from './inlineObject3.api.model' export * from './inlineObject4.api.model' export * from './isoLanguage.api.model' export * from './jSONObject.api.model' diff --git a/libs/data-access/gn4/src/spec.yaml b/libs/data-access/gn4/src/spec.yaml index 0731f208b2..93ce3a5b5c 100644 --- a/libs/data-access/gn4/src/spec.yaml +++ b/libs/data-access/gn4/src/spec.yaml @@ -7390,7 +7390,7 @@ paths: example: true requestBody: content: - application/json: + multipart/form-data: schema: required: - file From e748d8c86a532cbff61e9db636459f117dcd003b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Laure-H=C3=A9l=C3=A8ne=20Bruneton?= Date: Fri, 8 Mar 2024 17:23:33 +0100 Subject: [PATCH 12/14] feat(editor): add overview upload component --- .../overview-upload.component.css | 0 .../overview-upload.component.html | 8 ++ .../overview-upload.component.spec.ts | 91 +++++++++++++++++++ .../overview-upload.component.ts | 70 ++++++++++++++ 4 files changed, 169 insertions(+) create mode 100644 libs/feature/editor/src/lib/components/overview-upload/overview-upload.component.css create mode 100644 libs/feature/editor/src/lib/components/overview-upload/overview-upload.component.html create mode 100644 libs/feature/editor/src/lib/components/overview-upload/overview-upload.component.spec.ts create mode 100644 libs/feature/editor/src/lib/components/overview-upload/overview-upload.component.ts diff --git a/libs/feature/editor/src/lib/components/overview-upload/overview-upload.component.css b/libs/feature/editor/src/lib/components/overview-upload/overview-upload.component.css new file mode 100644 index 0000000000..e69de29bb2 diff --git a/libs/feature/editor/src/lib/components/overview-upload/overview-upload.component.html b/libs/feature/editor/src/lib/components/overview-upload/overview-upload.component.html new file mode 100644 index 0000000000..97d6aea100 --- /dev/null +++ b/libs/feature/editor/src/lib/components/overview-upload/overview-upload.component.html @@ -0,0 +1,8 @@ + diff --git a/libs/feature/editor/src/lib/components/overview-upload/overview-upload.component.spec.ts b/libs/feature/editor/src/lib/components/overview-upload/overview-upload.component.spec.ts new file mode 100644 index 0000000000..25b6441795 --- /dev/null +++ b/libs/feature/editor/src/lib/components/overview-upload/overview-upload.component.spec.ts @@ -0,0 +1,91 @@ +import { HttpClientTestingModule } from '@angular/common/http/testing' +import { ComponentFixture, TestBed } from '@angular/core/testing' +import { RecordsApiService } from '@geonetwork-ui/data-access/gn4' +import { TranslateModule } from '@ngx-translate/core' +import { of } from 'rxjs' +import { OverviewUploadComponent } from './overview-upload.component' + +class RecordsApiServiceMock { + getAllResources = jest.fn(() => + of([{ filename: 'filenameGet', url: 'urlGet' }]) + ) + putResource = jest.fn(() => of({ filename: 'filenamePut', url: 'urlPut' })) + putResourceFromURL = jest.fn(() => + of({ filename: 'filenamePutUrl', url: 'urlPutUrl' }) + ) + delResource = jest.fn(() => of(void 0)) +} + +const metadataUuid = '8505d991-e38f-4704-a47a-e7d335dfbef5' + +describe('OverviewUploadComponent', () => { + let component: OverviewUploadComponent + let fixture: ComponentFixture + let recordsApiService: RecordsApiService + + beforeEach(async () => { + await TestBed.configureTestingModule({ + imports: [ + OverviewUploadComponent, + HttpClientTestingModule, + TranslateModule.forRoot(), + ], + providers: [ + { + provide: RecordsApiService, + useClass: RecordsApiServiceMock, + }, + ], + }).compileComponents() + recordsApiService = TestBed.inject(RecordsApiService) + + fixture = TestBed.createComponent(OverviewUploadComponent) + component = fixture.componentInstance + component.metadataUuid = metadataUuid + fixture.detectChanges() + }) + + it('should create', () => { + expect(component).toBeTruthy() + }) + + it('should get all resources corresponding to the metadata UUID on init', () => { + expect(recordsApiService.getAllResources).toHaveBeenCalledWith(metadataUuid) + expect(component.resourceFileName).toEqual('filenameGet') + expect(component.resourceUrl).toEqual('urlGet') + }) + + it('should put the file resource on file change', () => { + const someFile = new File([], 'someFile') + component.handleFileChange(someFile) + expect(recordsApiService.putResource).toHaveBeenCalledWith( + metadataUuid, + someFile, + 'public' + ) + expect(component.resourceFileName).toEqual('filenamePut') + expect(component.resourceUrl).toEqual('urlPut') + }) + + it('should put the resource from URL on URL change', () => { + component.handleUrlChange('someUrl') + expect(recordsApiService.putResourceFromURL).toHaveBeenCalledWith( + metadataUuid, + 'someUrl', + 'public' + ) + expect(component.resourceFileName).toEqual('filenamePutUrl') + expect(component.resourceUrl).toEqual('urlPutUrl') + }) + + it('should delete the resource corresponding to the metadata UUID on delete', () => { + component.resourceFileName = 'filenameDelete' + component.handleDelete() + expect(recordsApiService.delResource).toHaveBeenCalledWith( + metadataUuid, + 'filenameDelete' + ) + expect(component.resourceFileName).toBeNull() + expect(component.resourceUrl).toBeNull() + }) +}) diff --git a/libs/feature/editor/src/lib/components/overview-upload/overview-upload.component.ts b/libs/feature/editor/src/lib/components/overview-upload/overview-upload.component.ts new file mode 100644 index 0000000000..4e437f3064 --- /dev/null +++ b/libs/feature/editor/src/lib/components/overview-upload/overview-upload.component.ts @@ -0,0 +1,70 @@ +import { + ChangeDetectionStrategy, + ChangeDetectorRef, + Component, + Input, + OnInit, +} from '@angular/core' +import { CommonModule } from '@angular/common' +import { RecordsApiService } from '@geonetwork-ui/data-access/gn4' +import { UiInputsModule } from '@geonetwork-ui/ui/inputs' + +@Component({ + selector: 'gn-ui-overview-upload', + standalone: true, + imports: [CommonModule, UiInputsModule], + templateUrl: './overview-upload.component.html', + styleUrls: ['./overview-upload.component.css'], + changeDetection: ChangeDetectionStrategy.OnPush, +}) +export class OverviewUploadComponent implements OnInit { + @Input() metadataUuid: string + + resourceFileName: string + resourceUrl: string + + constructor( + private recordsApiService: RecordsApiService, + private cd: ChangeDetectorRef + ) {} + + ngOnInit(): void { + this.recordsApiService + .getAllResources(this.metadataUuid) + .subscribe((resources) => { + this.resourceFileName = resources[0]?.filename + this.resourceUrl = resources[0]?.url + this.cd.markForCheck() + }) + } + + handleFileChange(file: File) { + this.recordsApiService + .putResource(this.metadataUuid, file, 'public') + .subscribe((resource) => { + this.resourceFileName = resource.filename + this.resourceUrl = resource.url + this.cd.markForCheck() + }) + } + + handleUrlChange(url: string) { + this.recordsApiService + .putResourceFromURL(this.metadataUuid, url, 'public') + .subscribe((resource) => { + this.resourceFileName = resource.filename + this.resourceUrl = resource.url + this.cd.markForCheck() + }) + } + + handleDelete() { + this.recordsApiService + .delResource(this.metadataUuid, this.resourceFileName) + .subscribe(() => { + this.resourceFileName = null + this.resourceUrl = null + this.cd.markForCheck() + }) + } +} From 1a868f802b6184929dbb1b18a5a6693c7fdabe19 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Laure-H=C3=A9l=C3=A8ne=20Bruneton?= Date: Fri, 5 Apr 2024 14:31:07 +0200 Subject: [PATCH 13/14] chore: i18n --- .../image-input/image-input.component.html | 12 +++++------ .../image-input.component.stories.ts | 15 ++++++++++--- .../lib/image-input/image-input.component.ts | 21 ++++++++++--------- translations/de.json | 10 +++++++++ translations/en.json | 10 +++++++++ translations/es.json | 10 +++++++++ translations/fr.json | 10 +++++++++ translations/it.json | 10 +++++++++ translations/nl.json | 10 +++++++++ translations/pt.json | 10 +++++++++ translations/sk.json | 10 +++++++++ 11 files changed, 109 insertions(+), 19 deletions(-) diff --git a/libs/ui/inputs/src/lib/image-input/image-input.component.html b/libs/ui/inputs/src/lib/image-input/image-input.component.html index 2accf0801b..ceeb82456a 100644 --- a/libs/ui/inputs/src/lib/image-input/image-input.component.html +++ b/libs/ui/inputs/src/lib/image-input/image-input.component.html @@ -23,7 +23,7 @@ *ngIf="showAltTextInput" type="text" class="py-3 px-2 border-2 border-gray-300 rounded-lg text-sm font-medium" - placeholder="Texte alternatif de l'image" + [placeholder]="'input.image.altTextPlaceholder' | translate" [value]="altText" (change)="handleAltTextChange($event)" /> @@ -33,7 +33,7 @@ (buttonClick)="handleDelete()" > delete - Supprimer + {{ 'input.image.delete' | translate }} add - Texte alternatif + {{ 'input.image.displayAltTextInput' | translate }}
@@ -86,7 +86,7 @@ >
-

{{ getPrimaryText() }}

+

{{ getPrimaryText() | translate }}

- {{ getSecondaryText() }} + {{ getSecondaryText() | translate }}

link - Saisir une URL + {{ 'input.image.displayUrlInput' | translate }}
diff --git a/libs/ui/inputs/src/lib/image-input/image-input.component.stories.ts b/libs/ui/inputs/src/lib/image-input/image-input.component.stories.ts index 8fe42384d2..342c655fa4 100644 --- a/libs/ui/inputs/src/lib/image-input/image-input.component.stories.ts +++ b/libs/ui/inputs/src/lib/image-input/image-input.component.stories.ts @@ -1,3 +1,10 @@ +import { HttpClientModule } from '@angular/common/http' +import { importProvidersFrom } from '@angular/core' +import { + TRANSLATE_DEFAULT_CONFIG, + UtilI18nModule, +} from '@geonetwork-ui/util/i18n' +import { TranslateModule } from '@ngx-translate/core' import { applicationConfig, Meta, @@ -5,8 +12,6 @@ import { StoryObj, } from '@storybook/angular' import { ImageInputComponent } from './image-input.component' -import { importProvidersFrom } from '@angular/core' -import { HttpClientModule } from '@angular/common/http' export default { title: 'Inputs/ImageInputComponent', @@ -16,7 +21,11 @@ export default { providers: [importProvidersFrom(HttpClientModule)], }), moduleMetadata({ - imports: [ImageInputComponent], + imports: [ + ImageInputComponent, + UtilI18nModule, + TranslateModule.forRoot(TRANSLATE_DEFAULT_CONFIG), + ], }), ], } as Meta diff --git a/libs/ui/inputs/src/lib/image-input/image-input.component.ts b/libs/ui/inputs/src/lib/image-input/image-input.component.ts index aee44be7c5..87af56458f 100644 --- a/libs/ui/inputs/src/lib/image-input/image-input.component.ts +++ b/libs/ui/inputs/src/lib/image-input/image-input.component.ts @@ -4,18 +4,18 @@ import { ChangeDetectionStrategy, ChangeDetectorRef, Component, - ElementRef, EventEmitter, Input, Output, - ViewChild, } from '@angular/core' import { MatIconModule } from '@angular/material/icon' +import { MatProgressSpinnerModule } from '@angular/material/progress-spinner' import { downgradeImage, megabytesToBytes } from '@geonetwork-ui/util/shared' +import { firstValueFrom } from 'rxjs' import { ButtonComponent } from '../button/button.component' import { FilesDropDirective } from '../files-drop/files-drop.directive' -import { MatProgressSpinnerModule } from '@angular/material/progress-spinner' -import { firstValueFrom } from 'rxjs' +import { TranslateModule } from '@ngx-translate/core' +import { marker } from '@biesbjerg/ngx-translate-extract-marker' @Component({ selector: 'gn-ui-image-input', @@ -29,6 +29,7 @@ import { firstValueFrom } from 'rxjs' MatIconModule, FilesDropDirective, MatProgressSpinnerModule, + TranslateModule, ], }) export class ImageInputComponent { @@ -56,22 +57,22 @@ export class ImageInputComponent { getPrimaryText() { if (this.uploadError) { - return "L'image n'a pas pu être chargée" + return marker('input.image.uploadErrorLabel') } if (this.uploadProgress) { - return 'Chargement en cours...' + return marker('input.image.uploadProgressLabel') } - return 'Sélectionner une image' + return marker('input.image.selectFileLabel') } getSecondaryText() { if (this.uploadError) { - return 'Réessayer' + return marker('input.image.uploadErrorRetry') } if (this.uploadProgress) { - return 'Annuler' + return marker('input.image.uploadProgressCancel') } - return 'ou la glisser ici' + return marker('input.image.dropFileLabel') } handleDragFilesOver(dragFilesOver: boolean) { diff --git a/translations/de.json b/translations/de.json index f8324d3d2e..ca8ada8354 100644 --- a/translations/de.json +++ b/translations/de.json @@ -159,6 +159,16 @@ "facets.block.title.tag.default": "Stichwort", "facets.block.title.th_regions_tree.default": "Regionen", "favorite.not.authenticated.tooltip": "
Anmelden, um auf diese Funktion zuzugreifen
", + "input.image.altTextPlaceholder": "", + "input.image.delete": "", + "input.image.displayAltTextInput": "", + "input.image.displayUrlInput": "", + "input.image.dropFileLabel": "", + "input.image.selectFileLabel": "", + "input.image.uploadErrorLabel": "", + "input.image.uploadErrorRetry": "", + "input.image.uploadProgressCancel": "", + "input.image.uploadProgressLabel": "", "language.ca": "Katalanisch", "language.cs": "Tschechisch", "language.de": "Deutsch", diff --git a/translations/en.json b/translations/en.json index dd45df888c..4ed6558167 100644 --- a/translations/en.json +++ b/translations/en.json @@ -159,6 +159,16 @@ "facets.block.title.tag.default": "Tag", "facets.block.title.th_regions_tree.default": "Regions", "favorite.not.authenticated.tooltip": "
Login to access this feature
", + "input.image.altTextPlaceholder": "Image alternate text", + "input.image.delete": "Delete", + "input.image.displayAltTextInput": "Alternate text", + "input.image.displayUrlInput": "Enter a URL", + "input.image.dropFileLabel": "or drop it here", + "input.image.selectFileLabel": "Select an image", + "input.image.uploadErrorLabel": "The image could not be uploaded", + "input.image.uploadErrorRetry": "Retry", + "input.image.uploadProgressCancel": "Cancel", + "input.image.uploadProgressLabel": "Upload in progress...", "language.ca": "Catalan", "language.cs": "Czech", "language.de": "German", diff --git a/translations/es.json b/translations/es.json index 9f82e0ab23..3187be45c1 100644 --- a/translations/es.json +++ b/translations/es.json @@ -159,6 +159,16 @@ "facets.block.title.tag.default": "", "facets.block.title.th_regions_tree.default": "", "favorite.not.authenticated.tooltip": "", + "input.image.altTextPlaceholder": "", + "input.image.delete": "", + "input.image.displayAltTextInput": "", + "input.image.displayUrlInput": "", + "input.image.dropFileLabel": "", + "input.image.selectFileLabel": "", + "input.image.uploadErrorLabel": "", + "input.image.uploadErrorRetry": "", + "input.image.uploadProgressCancel": "", + "input.image.uploadProgressLabel": "", "language.ca": "Catalán", "language.cs": "Checo", "language.de": "Alemán", diff --git a/translations/fr.json b/translations/fr.json index 799605a027..a8fb60e052 100644 --- a/translations/fr.json +++ b/translations/fr.json @@ -159,6 +159,16 @@ "facets.block.title.tag.default": "Tag", "facets.block.title.th_regions_tree.default": "Régions", "favorite.not.authenticated.tooltip": "
Connectez-vous pour avoir accès à cette fonctionnalité
", + "input.image.altTextPlaceholder": "Texte alternatif de l'image", + "input.image.delete": "Supprimer", + "input.image.displayAltTextInput": "Texte alternatif", + "input.image.displayUrlInput": "Saisir une URL", + "input.image.dropFileLabel": "ou la glisser ici", + "input.image.selectFileLabel": "Sélectionner une image", + "input.image.uploadErrorLabel": "L'image n'a pas pu être chargée", + "input.image.uploadErrorRetry": "Réessayer", + "input.image.uploadProgressCancel": "Annuler", + "input.image.uploadProgressLabel": "Chargement en cours...", "language.ca": "Catalan", "language.cs": "Tchèque", "language.de": "Allemand", diff --git a/translations/it.json b/translations/it.json index 9b03fabaca..38d5c8c2e4 100644 --- a/translations/it.json +++ b/translations/it.json @@ -159,6 +159,16 @@ "facets.block.title.tag.default": "Tag", "facets.block.title.th_regions_tree.default": "Regioni", "favorite.not.authenticated.tooltip": "
Login per accedere a questa funzionalità
", + "input.image.altTextPlaceholder": "", + "input.image.delete": "", + "input.image.displayAltTextInput": "", + "input.image.displayUrlInput": "", + "input.image.dropFileLabel": "", + "input.image.selectFileLabel": "", + "input.image.uploadErrorLabel": "", + "input.image.uploadErrorRetry": "", + "input.image.uploadProgressCancel": "", + "input.image.uploadProgressLabel": "", "language.ca": "Catalano", "language.cs": "Ceco", "language.de": "Tedesco", diff --git a/translations/nl.json b/translations/nl.json index 64ee837518..36cccca0d0 100644 --- a/translations/nl.json +++ b/translations/nl.json @@ -159,6 +159,16 @@ "facets.block.title.tag.default": "", "facets.block.title.th_regions_tree.default": "", "favorite.not.authenticated.tooltip": "", + "input.image.altTextPlaceholder": "", + "input.image.delete": "", + "input.image.displayAltTextInput": "", + "input.image.displayUrlInput": "", + "input.image.dropFileLabel": "", + "input.image.selectFileLabel": "", + "input.image.uploadErrorLabel": "", + "input.image.uploadErrorRetry": "", + "input.image.uploadProgressCancel": "", + "input.image.uploadProgressLabel": "", "language.ca": "Catalaans", "language.cs": "Tsjechisch", "language.de": "Duits", diff --git a/translations/pt.json b/translations/pt.json index 86fe64024b..078c076012 100644 --- a/translations/pt.json +++ b/translations/pt.json @@ -159,6 +159,16 @@ "facets.block.title.tag.default": "", "facets.block.title.th_regions_tree.default": "", "favorite.not.authenticated.tooltip": "", + "input.image.altTextPlaceholder": "", + "input.image.delete": "", + "input.image.displayAltTextInput": "", + "input.image.displayUrlInput": "", + "input.image.dropFileLabel": "", + "input.image.selectFileLabel": "", + "input.image.uploadErrorLabel": "", + "input.image.uploadErrorRetry": "", + "input.image.uploadProgressCancel": "", + "input.image.uploadProgressLabel": "", "language.ca": "Catalão", "language.cs": "Tcheco", "language.de": "Alemão", diff --git a/translations/sk.json b/translations/sk.json index db63a71d01..2492181281 100644 --- a/translations/sk.json +++ b/translations/sk.json @@ -159,6 +159,16 @@ "facets.block.title.tag.default": "Štítok", "facets.block.title.th_regions_tree.default": "Regióny", "favorite.not.authenticated.tooltip": "
Prihlásiť sa pre prístup k tejto funkcii
", + "input.image.altTextPlaceholder": "", + "input.image.delete": "", + "input.image.displayAltTextInput": "", + "input.image.displayUrlInput": "", + "input.image.dropFileLabel": "", + "input.image.selectFileLabel": "", + "input.image.uploadErrorLabel": "", + "input.image.uploadErrorRetry": "", + "input.image.uploadProgressCancel": "", + "input.image.uploadProgressLabel": "", "language.ca": "Catalánsky", "language.cs": "Čeština", "language.de": "Nemecky", From 2bd7075caaa93ba577f108f87da7f483a5d18336 Mon Sep 17 00:00:00 2001 From: Olivia Guyot Date: Tue, 9 Apr 2024 17:09:39 +0200 Subject: [PATCH 14/14] fix(ui): assign a default style to a gn-ui-button in case non given --- libs/ui/inputs/src/lib/button/button.component.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/libs/ui/inputs/src/lib/button/button.component.ts b/libs/ui/inputs/src/lib/button/button.component.ts index af989df803..a2339085dc 100644 --- a/libs/ui/inputs/src/lib/button/button.component.ts +++ b/libs/ui/inputs/src/lib/button/button.component.ts @@ -15,7 +15,7 @@ import { propagateToDocumentOnly } from '@geonetwork-ui/util/shared' standalone: true, }) export class ButtonComponent { - private btnClass: string + private btnClass = 'gn-ui-btn-default' @Input() set type( value: 'primary' | 'secondary' | 'default' | 'outline' | 'light'