Files
dspace-angular/src/app/shared/object-collection/shared/tabulatable-objects/tabulatable-objects-loader.component.ts
FrancescoMolinaro b6245d3c2d improve responsivness
2024-01-09 12:11:53 +01:00

207 lines
6.1 KiB
TypeScript

import {
ChangeDetectorRef,
Component,
ComponentRef,
EventEmitter,
Input,
OnChanges,
OnDestroy,
OnInit,
Output,
SimpleChanges,
ViewChild
} from '@angular/core';
import { ListableObject } from '../listable-object.model';
import { ViewMode } from '../../../../core/shared/view-mode.model';
import { Context } from '../../../../core/shared/context.model';
import { CollectionElementLinkType } from '../../collection-element-link.type';
import { combineLatest, Observable, of as observableOf, Subscription } from 'rxjs';
import { ThemeService } from '../../../theme-support/theme.service';
import { hasNoValue, hasValue, isNotEmpty } from '../../../empty.util';
import { take } from 'rxjs/operators';
import { GenericConstructor } from '../../../../core/shared/generic-constructor';
import { TabulatableObjectsDirective } from './tabulatable-objects.directive';
import { PaginatedList } from '../../../../core/data/paginated-list.model';
import { getTabulatableObjectsComponent } from './tabulatable-objects.decorator';
@Component({
selector: 'ds-tabulatable-objects-loader',
templateUrl: './tabulatable-objects-loader.component.html'
})
/**
* Component for determining what component to use depending on the item's entity type (dspace.entity.type)
*/
export class TabulatableObjectsLoaderComponent implements OnInit, OnChanges, OnDestroy {
/**
* The items to determine the component for
*/
@Input() objects: PaginatedList<ListableObject>;
/**
* The context of listable object
*/
@Input() context: Context;
/**
* The type of link used to render the links inside the listable object
*/
@Input() linkType: CollectionElementLinkType;
/**
* The identifier of the list this element resides in
*/
@Input() listID: string;
/**
* Whether to show the badge label or not
*/
@Input() showLabel = true;
/**
* Whether to show the thumbnail preview
*/
@Input() showThumbnails;
/**
* The value to display for this element
*/
@Input() value: string;
/**
* Directive hook used to place the dynamic child component
*/
@ViewChild(TabulatableObjectsDirective, { static: true }) tabulatableObjectsDirective: TabulatableObjectsDirective;
/**
* Emit when the listable object has been reloaded.
*/
@Output() contentChange = new EventEmitter<PaginatedList<ListableObject>>();
/**
* Array to track all subscriptions and unsubscribe them onDestroy
* @type {Array}
*/
protected subs: Subscription[] = [];
/**
* The reference to the dynamic component
*/
protected compRef: ComponentRef<Component>;
/**
* The view mode used to identify the components
*/
protected viewMode: ViewMode = ViewMode.Table;
/**
* The list of input and output names for the dynamic component
*/
protected inAndOutputNames: string[] = [
'objects',
'linkType',
'listID',
'showLabel',
'showThumbnails',
'context',
'viewMode',
'value',
'hideBadges',
'contentChange',
];
constructor(private cdr: ChangeDetectorRef, private themeService: ThemeService) {
}
/**
* Setup the dynamic child component
*/
ngOnInit(): void {
this.instantiateComponent(this.objects);
}
/**
* Whenever the inputs change, update the inputs of the dynamic component
*/
ngOnChanges(changes: SimpleChanges): void {
if (hasNoValue(this.compRef)) {
// sometimes the component has not been initialized yet, so it first needs to be initialized
// before being called again
this.instantiateComponent(this.objects, changes);
} else {
// if an input or output has changed
if (this.inAndOutputNames.some((name: any) => hasValue(changes[name]))) {
this.connectInputsAndOutputs();
if (this.compRef?.instance && 'ngOnChanges' in this.compRef.instance) {
(this.compRef.instance as any).ngOnChanges(changes);
}
}
}
}
ngOnDestroy() {
this.subs
.filter((subscription) => hasValue(subscription))
.forEach((subscription) => subscription.unsubscribe());
}
private instantiateComponent(objects: PaginatedList<ListableObject>, changes?: SimpleChanges): void {
// objects need to have same render type so we access just the first in the page
const component = this.getComponent(objects?.page[0]?.getRenderTypes(), this.viewMode, this.context);
const viewContainerRef = this.tabulatableObjectsDirective.viewContainerRef;
viewContainerRef.clear();
this.compRef = viewContainerRef.createComponent(
component, {
index: 0,
injector: undefined
}
);
if (hasValue(changes)) {
this.ngOnChanges(changes);
} else {
this.connectInputsAndOutputs();
}
if ((this.compRef.instance as any).reloadedObject) {
combineLatest([
observableOf(changes),
(this.compRef.instance as any).reloadedObject.pipe(take(1)) as Observable<PaginatedList<ListableObject>>,
]).subscribe(([simpleChanges, reloadedObjects]: [SimpleChanges, PaginatedList<ListableObject>]) => {
if (reloadedObjects) {
this.compRef.destroy();
this.objects = reloadedObjects;
this.instantiateComponent(reloadedObjects, simpleChanges);
this.cdr.detectChanges();
this.contentChange.emit(reloadedObjects);
}
});
}
}
/**
* Fetch the component depending on the item's entity type, view mode and context
* @returns {GenericConstructor<Component>}
*/
getComponent(renderTypes: (string | GenericConstructor<ListableObject>)[],
viewMode: ViewMode,
context: Context): GenericConstructor<Component> {
return getTabulatableObjectsComponent(renderTypes, viewMode, context, this.themeService.getThemeName());
}
/**
* Connect the in and outputs of this component to the dynamic component,
* to ensure they're in sync
*/
protected connectInputsAndOutputs(): void {
if (isNotEmpty(this.inAndOutputNames) && hasValue(this.compRef) && hasValue(this.compRef.instance)) {
this.inAndOutputNames.filter((name: any) => this[name] !== undefined).forEach((name: any) => {
this.compRef.instance[name] = this[name];
});
}
}
}