Skip to content

Commit 26bfc58

Browse files
[DSpace#2719][CST-12825] Refactors item-page-img-field component
1 parent ca3749b commit 26bfc58

7 files changed

Lines changed: 164 additions & 9 deletions

File tree

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

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,7 @@
2525

2626
<!-- Render value as a link with icon -->
2727
<ng-template #linkImg let-img="img" let-value="value">
28-
<a class="link-anchor" [href]="value" class="dont-break-out ds-simple-metadata-link" target="_blank">
28+
<a [href]="value" class="link-anchor dont-break-out ds-simple-metadata-link" target="_blank">
2929
<img class="link-logo"
3030
[alt]="img.alt | translate"
3131
[style.height]="'var(' + img.heightVar + ', --ds-item-page-img-field-default-inline-height)'"

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

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -4,7 +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/img/item-page-img-field.component';
7+
import { ImageField } from '../../simple/field-components/specific-field/item-page-field.component';
88

99
/**
1010
* This component renders the configured 'values' into the ds-metadata-field-wrapper component.
Lines changed: 84 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,84 @@
1+
import { ComponentFixture, TestBed } from '@angular/core/testing';
2+
3+
import { ImageField, 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+
16+
let component: ItemPageImgFieldComponent;
17+
let fixture: ComponentFixture<ItemPageImgFieldComponent>;
18+
19+
const mockField = 'organization.identifier.ror';
20+
const mockValue = 'http://ror.org/awesome-identifier';
21+
const mockLabel = 'ROR label';
22+
const mockUrlRegex = '(.*)ror.org';
23+
const mockImg = {
24+
URI: './assets/images/ror-icon.svg',
25+
alt: 'item.page.image.alt.ROR',
26+
heightVar: '--ds-item-page-img-field-ror-inline-height'
27+
} as ImageField;
28+
29+
describe('ItemPageImgFieldComponent', () => {
30+
31+
beforeEach(async () => {
32+
await TestBed.configureTestingModule({
33+
imports: [TranslateModule.forRoot({
34+
loader: {
35+
provide: TranslateLoader,
36+
useClass: TranslateLoaderMock
37+
}
38+
})],
39+
providers: [
40+
{ provide: APP_CONFIG, useValue: environment },
41+
{ provide: BrowseDefinitionDataService, useValue: BrowseDefinitionDataServiceStub }
42+
],
43+
declarations: [ItemPageImgFieldComponent, GenericItemPageFieldComponent, MetadataValuesComponent],
44+
schemas: [NO_ERRORS_SCHEMA]
45+
})
46+
.overrideComponent(GenericItemPageFieldComponent, {
47+
set: { changeDetection: ChangeDetectionStrategy.Default }
48+
})
49+
.compileComponents();
50+
51+
fixture = TestBed.createComponent(ItemPageImgFieldComponent);
52+
component = fixture.componentInstance;
53+
component.item = mockItemWithMetadataFieldsAndValue([mockField], mockValue);
54+
component.fields = [mockField];
55+
component.label = mockLabel;
56+
component.urlRegex = mockUrlRegex;
57+
component.img = mockImg;
58+
fixture.detectChanges();
59+
});
60+
61+
it('should create', () => {
62+
expect(component).toBeTruthy();
63+
});
64+
65+
it('should display display img tag', () => {
66+
const image = fixture.debugElement.query(By.css('img.link-logo'));
67+
expect(image).not.toBeNull();
68+
});
69+
70+
it('should have right attributes', () => {
71+
const image = fixture.debugElement.query(By.css('img.link-logo'));
72+
expect(image.attributes.src).toEqual(mockImg.URI);
73+
expect(image.attributes.alt).toEqual(mockImg.alt);
74+
75+
const imageEl = image.nativeElement;
76+
expect(imageEl.style.height).toContain(mockImg.heightVar);
77+
});
78+
79+
it('should have the right value', () => {
80+
const imageAnchor = fixture.debugElement.query(By.css('a.link-anchor'));
81+
const anchorEl = imageAnchor.nativeElement;
82+
expect(anchorEl.innerHTML).toContain(mockValue);
83+
});
84+
});
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>)

src/assets/i18n/en.json5

Lines changed: 7 additions & 7 deletions
Original file line numberDiff line numberDiff line change
@@ -2564,19 +2564,19 @@
25642564

25652565
"item.preview.oaire.fundingStream": "Funding Stream:",
25662566

2567-
"item.preview.oairecerif.identifier.url" : "URL",
2567+
"item.preview.oairecerif.identifier.url": "URL",
25682568

2569-
"item.preview.organization.address.addressCountry" : "Country",
2569+
"item.preview.organization.address.addressCountry": "Country",
25702570

2571-
"item.preview.organization.foundingDate" : "Founding Date",
2571+
"item.preview.organization.foundingDate": "Founding Date",
25722572

2573-
"item.preview.organization.identifier.crossrefid" : "CrossRef ID",
2573+
"item.preview.organization.identifier.crossrefid": "CrossRef ID",
25742574

2575-
"item.preview.organization.identifier.isni" : "ISNI",
2575+
"item.preview.organization.identifier.isni": "ISNI",
25762576

2577-
"item.preview.organization.identifier.ror" : "ROR ID",
2577+
"item.preview.organization.identifier.ror": "ROR ID",
25782578

2579-
"item.preview.organization.legalName" : "Legal Name",
2579+
"item.preview.organization.legalName": "Legal Name",
25802580

25812581
"item.select.confirm": "Confirm selected",
25822582

0 commit comments

Comments
 (0)