Skip to content

Commit e1c639e

Browse files
authored
Merge pull request DSpace#2719 from 4Science/main_CST-12825
ROR Integration - Identifier Visualization
2 parents c75e6e2 + 678d089 commit e1c639e

12 files changed

Lines changed: 272 additions & 11 deletions

File tree

src/app/entity-groups/research-entities/item-pages/org-unit/org-unit.component.html

Lines changed: 12 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,18 @@
3232
</ds-generic-item-page-field>
3333
</div>
3434
<div class="col-xs-12 col-md-7">
35+
<ds-item-page-img-field
36+
[fields]="['organization.identifier.ror']"
37+
[img]="{
38+
URI: './assets/images/ror-icon.svg',
39+
alt: 'item.page.image.alt.ROR',
40+
heightVar: '--ds-item-page-img-field-ror-inline-height'
41+
}"
42+
[item]="object"
43+
[label]="'orgunit.page.ror'"
44+
[urlRegex]="'(.*)ror.org'"
45+
>
46+
</ds-item-page-img-field>
3547
<ds-related-items
3648
[parentItem]="object"
3749
[relationType]="'isPublicationOfOrgUnit'"

src/app/item-page/field-components/metadata-values/metadata-values.component.html

Lines changed: 13 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -3,8 +3,8 @@
33
<!--
44
Choose a template. Priority: markdown, link, browse link.
55
-->
6-
<ng-container *ngTemplateOutlet="(renderMarkdown ? markdown : (hasLink(mdValue) ? link : (hasBrowseDefinition() ? browselink : simple)));
7-
context: {value: mdValue.value}">
6+
<ng-container *ngTemplateOutlet="(renderMarkdown ? markdown : (hasLink(mdValue) ? (img != null ? linkImg : link) : (hasBrowseDefinition() ? browselink : simple)));
7+
context: {value: mdValue.value, img}">
88
</ng-container>
99
<span class="separator" *ngIf="!last" [innerHTML]="separator"></span>
1010
</ng-container>
@@ -23,6 +23,17 @@
2323
</a>
2424
</ng-template>
2525

26+
<!-- Render value as a link with icon -->
27+
<ng-template #linkImg let-img="img" let-value="value">
28+
<a [href]="value" class="link-anchor dont-break-out ds-simple-metadata-link" target="_blank">
29+
<img class="link-logo"
30+
[alt]="img.alt | translate"
31+
[style.height]="'var(' + img.heightVar + ', --ds-item-page-img-field-default-inline-height)'"
32+
[src]="img.URI"/>
33+
{{value}}
34+
</a>
35+
</ng-template>
36+
2637
<!-- Render simple value in a span -->
2738
<ng-template #simple let-value="value">
2839
<span class="dont-break-out preserve-line-breaks">{{value}}</span>

src/app/item-page/field-components/metadata-values/metadata-values.component.ts

Lines changed: 6 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ import { APP_CONFIG, AppConfig } from '../../../../config/app-config.interface';
44
import { BrowseDefinition } from '../../../core/shared/browse-definition.model';
55
import { hasValue } from '../../../shared/empty.util';
66
import { VALUE_LIST_BROWSE_DEFINITION } from '../../../core/shared/value-list-browse-definition.resource-type';
7+
import { ImageField } from '../../simple/field-components/specific-field/item-page-field.component';
78

89
/**
910
* This component renders the configured 'values' into the ds-metadata-field-wrapper component.
@@ -55,6 +56,11 @@ export class MetadataValuesComponent implements OnChanges {
5556

5657
@Input() browseDefinition?: BrowseDefinition;
5758

59+
/**
60+
* Optional {@code ImageField} reference that represents an image to be displayed inline.
61+
*/
62+
@Input() img?: ImageField;
63+
5864
ngOnChanges(changes: SimpleChanges): void {
5965
this.renderMarkdown = !!this.appConfig.markdown.enabled && this.enableMarkdown;
6066
}

src/app/item-page/item-shared.module.ts

Lines changed: 26 additions & 9 deletions
Original file line numberDiff line numberDiff line change
@@ -1,21 +1,36 @@
1-
import { RelatedEntitiesSearchComponent } from './simple/related-entities/related-entities-search/related-entities-search.component';
1+
import {
2+
RelatedEntitiesSearchComponent
3+
} from './simple/related-entities/related-entities-search/related-entities-search.component';
24
import { NgModule } from '@angular/core';
3-
import { CommonModule } from '@angular/common';
5+
import { CommonModule, NgOptimizedImage } from '@angular/common';
46
import { SearchModule } from '../shared/search/search.module';
57
import { SharedModule } from '../shared/shared.module';
68
import { TranslateModule } from '@ngx-translate/core';
79
import { DYNAMIC_FORM_CONTROL_MAP_FN } from '@ng-dynamic-forms/core';
8-
import { dsDynamicFormControlMapFn } from '../shared/form/builder/ds-dynamic-form-ui/ds-dynamic-form-control-container.component';
9-
import { TabbedRelatedEntitiesSearchComponent } from './simple/related-entities/tabbed-related-entities-search/tabbed-related-entities-search.component';
10-
import { ItemVersionsDeleteModalComponent } from './versions/item-versions-delete-modal/item-versions-delete-modal.component';
11-
import { ItemVersionsSummaryModalComponent } from './versions/item-versions-summary-modal/item-versions-summary-modal.component';
10+
import {
11+
dsDynamicFormControlMapFn
12+
} from '../shared/form/builder/ds-dynamic-form-ui/ds-dynamic-form-control-container.component';
13+
import {
14+
TabbedRelatedEntitiesSearchComponent
15+
} from './simple/related-entities/tabbed-related-entities-search/tabbed-related-entities-search.component';
16+
import {
17+
ItemVersionsDeleteModalComponent
18+
} from './versions/item-versions-delete-modal/item-versions-delete-modal.component';
19+
import {
20+
ItemVersionsSummaryModalComponent
21+
} from './versions/item-versions-summary-modal/item-versions-summary-modal.component';
1222
import { MetadataValuesComponent } from './field-components/metadata-values/metadata-values.component';
13-
import { GenericItemPageFieldComponent } from './simple/field-components/specific-field/generic/generic-item-page-field.component';
14-
import { MetadataRepresentationListComponent } from './simple/metadata-representation-list/metadata-representation-list.component';
23+
import {
24+
GenericItemPageFieldComponent
25+
} from './simple/field-components/specific-field/generic/generic-item-page-field.component';
26+
import {
27+
MetadataRepresentationListComponent
28+
} from './simple/metadata-representation-list/metadata-representation-list.component';
1529
import { RelatedItemsComponent } from './simple/related-items/related-items-component';
1630
import {
1731
ThemedMetadataRepresentationListComponent
1832
} from './simple/metadata-representation-list/themed-metadata-representation-list.component';
33+
import { ItemPageImgFieldComponent } from './simple/field-components/specific-field/img/item-page-img-field.component';
1934

2035
const ENTRY_COMPONENTS = [
2136
ItemVersionsDeleteModalComponent,
@@ -32,6 +47,7 @@ const COMPONENTS = [
3247
MetadataRepresentationListComponent,
3348
ThemedMetadataRepresentationListComponent,
3449
RelatedItemsComponent,
50+
ItemPageImgFieldComponent,
3551
];
3652

3753
@NgModule({
@@ -42,7 +58,8 @@ const COMPONENTS = [
4258
CommonModule,
4359
SearchModule,
4460
SharedModule,
45-
TranslateModule
61+
TranslateModule,
62+
NgOptimizedImage
4663
],
4764
exports: [
4865
...COMPONENTS
Lines changed: 85 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,85 @@
1+
import { ComponentFixture, TestBed } from '@angular/core/testing';
2+
3+
import { ItemPageImgFieldComponent } from './item-page-img-field.component';
4+
import { TranslateLoader, TranslateModule } from '@ngx-translate/core';
5+
import { TranslateLoaderMock } from '../../../../../shared/testing/translate-loader.mock';
6+
import { APP_CONFIG } from '../../../../../../config/app-config.interface';
7+
import { environment } from '../../../../../../environments/environment';
8+
import { BrowseDefinitionDataService } from '../../../../../core/browse/browse-definition-data.service';
9+
import { BrowseDefinitionDataServiceStub } from '../../../../../shared/testing/browse-definition-data-service.stub';
10+
import { GenericItemPageFieldComponent } from '../generic/generic-item-page-field.component';
11+
import { MetadataValuesComponent } from '../../../../field-components/metadata-values/metadata-values.component';
12+
import { ChangeDetectionStrategy, NO_ERRORS_SCHEMA } from '@angular/core';
13+
import { mockItemWithMetadataFieldsAndValue } from '../item-page-field.component.spec';
14+
import { By } from '@angular/platform-browser';
15+
import { ImageField } from '../item-page-field.component';
16+
17+
let component: ItemPageImgFieldComponent;
18+
let fixture: ComponentFixture<ItemPageImgFieldComponent>;
19+
20+
const mockField = 'organization.identifier.ror';
21+
const mockValue = 'http://ror.org/awesome-identifier';
22+
const mockLabel = 'ROR label';
23+
const mockUrlRegex = '(.*)ror.org';
24+
const mockImg = {
25+
URI: './assets/images/ror-icon.svg',
26+
alt: 'item.page.image.alt.ROR',
27+
heightVar: '--ds-item-page-img-field-ror-inline-height'
28+
} as ImageField;
29+
30+
describe('ItemPageImgFieldComponent', () => {
31+
32+
beforeEach(async () => {
33+
await TestBed.configureTestingModule({
34+
imports: [TranslateModule.forRoot({
35+
loader: {
36+
provide: TranslateLoader,
37+
useClass: TranslateLoaderMock
38+
}
39+
})],
40+
providers: [
41+
{ provide: APP_CONFIG, useValue: environment },
42+
{ provide: BrowseDefinitionDataService, useValue: BrowseDefinitionDataServiceStub }
43+
],
44+
declarations: [ItemPageImgFieldComponent, GenericItemPageFieldComponent, MetadataValuesComponent],
45+
schemas: [NO_ERRORS_SCHEMA]
46+
})
47+
.overrideComponent(GenericItemPageFieldComponent, {
48+
set: { changeDetection: ChangeDetectionStrategy.Default }
49+
})
50+
.compileComponents();
51+
52+
fixture = TestBed.createComponent(ItemPageImgFieldComponent);
53+
component = fixture.componentInstance;
54+
component.item = mockItemWithMetadataFieldsAndValue([mockField], mockValue);
55+
component.fields = [mockField];
56+
component.label = mockLabel;
57+
component.urlRegex = mockUrlRegex;
58+
component.img = mockImg;
59+
fixture.detectChanges();
60+
});
61+
62+
it('should create', () => {
63+
expect(component).toBeTruthy();
64+
});
65+
66+
it('should display display img tag', () => {
67+
const image = fixture.debugElement.query(By.css('img.link-logo'));
68+
expect(image).not.toBeNull();
69+
});
70+
71+
it('should have right attributes', () => {
72+
const image = fixture.debugElement.query(By.css('img.link-logo'));
73+
expect(image.attributes.src).toEqual(mockImg.URI);
74+
expect(image.attributes.alt).toEqual(mockImg.alt);
75+
76+
const imageEl = image.nativeElement;
77+
expect(imageEl.style.height).toContain(mockImg.heightVar);
78+
});
79+
80+
it('should have the right value', () => {
81+
const imageAnchor = fixture.debugElement.query(By.css('a.link-anchor'));
82+
const anchorEl = imageAnchor.nativeElement;
83+
expect(anchorEl.innerHTML).toContain(mockValue);
84+
});
85+
});
Lines changed: 46 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,46 @@
1+
import { Component, Input } from '@angular/core';
2+
import { ImageField, ItemPageFieldComponent } from '../item-page-field.component';
3+
import { Item } from '../../../../../core/shared/item.model';
4+
5+
@Component({
6+
selector: 'ds-item-page-img-field',
7+
templateUrl: '../item-page-field.component.html'
8+
})
9+
/**
10+
* Component that renders an inline image for a given field.
11+
* This component uses a given {@code ImageField} configuration to correctly render the img.
12+
*/
13+
export class ItemPageImgFieldComponent extends ItemPageFieldComponent {
14+
15+
/**
16+
* The item to display metadata for
17+
*/
18+
@Input() item: Item;
19+
20+
/**
21+
* Separator string between multiple values of the metadata fields defined
22+
* @type {string}
23+
*/
24+
@Input() separator: string;
25+
26+
/**
27+
* Fields (schema.element.qualifier) used to render their values.
28+
*/
29+
@Input() fields: string[];
30+
31+
/**
32+
* Label i18n key for the rendered metadata
33+
*/
34+
@Input() label: string;
35+
36+
/**
37+
* Image Configuration
38+
*/
39+
@Input() img: ImageField;
40+
41+
/**
42+
* Whether any valid HTTP(S) URL should be rendered as a link
43+
*/
44+
@Input() urlRegex?: string;
45+
46+
}

src/app/item-page/simple/field-components/specific-field/item-page-field.component.html

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -6,5 +6,6 @@
66
[enableMarkdown]="enableMarkdown"
77
[urlRegex]="urlRegex"
88
[browseDefinition]="browseDefinition|async"
9+
[img]="img"
910
></ds-metadata-values>
1011
</div>

src/app/item-page/simple/field-components/specific-field/item-page-field.component.ts

Lines changed: 24 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,25 @@ import { BrowseDefinition } from '../../../../core/shared/browse-definition.mode
66
import { BrowseDefinitionDataService } from '../../../../core/browse/browse-definition-data.service';
77
import { getRemoteDataPayload } from '../../../../core/shared/operators';
88

9+
/**
10+
* Interface that encapsulate Image configuration for this component.
11+
*/
12+
export interface ImageField {
13+
/**
14+
* URI that is used to retrieve the image.
15+
*/
16+
URI: string;
17+
/**
18+
* i18n Key that represents the alt text to display
19+
*/
20+
alt: string;
21+
/**
22+
* CSS variable that contains the height of the inline image.
23+
*/
24+
heightVar: string;
25+
}
26+
27+
928
/**
1029
* This component can be used to represent metadata on a simple item page.
1130
* It expects one input parameter of type Item to which the metadata belongs.
@@ -51,6 +70,11 @@ export class ItemPageFieldComponent {
5170
*/
5271
urlRegex?: string;
5372

73+
/**
74+
* Image Configuration
75+
*/
76+
img: ImageField;
77+
5478
/**
5579
* Return browse definition that matches any field used in this component if it is configured as a browse
5680
* link in dspace.cfg (webui.browse.link.<n>)

0 commit comments

Comments
 (0)