mirror of
https://github.com/DSpace/dspace-angular.git
synced 2025-10-07 10:04:11 +00:00
Merge branch 'master' into language-header
This commit is contained in:
@@ -4,6 +4,7 @@ import { NgModule } from '@angular/core';
|
|||||||
import { MetadataSchemaComponent } from './metadata-schema/metadata-schema.component';
|
import { MetadataSchemaComponent } from './metadata-schema/metadata-schema.component';
|
||||||
import { URLCombiner } from '../../core/url-combiner/url-combiner';
|
import { URLCombiner } from '../../core/url-combiner/url-combiner';
|
||||||
import { getRegistriesModulePath } from '../admin-routing.module';
|
import { getRegistriesModulePath } from '../admin-routing.module';
|
||||||
|
import { I18nBreadcrumbResolver } from '../../core/breadcrumbs/i18n-breadcrumb.resolver';
|
||||||
|
|
||||||
const BITSTREAMFORMATS_MODULE_PATH = 'bitstream-formats';
|
const BITSTREAMFORMATS_MODULE_PATH = 'bitstream-formats';
|
||||||
|
|
||||||
@@ -14,16 +15,28 @@ export function getBitstreamFormatsModulePath() {
|
|||||||
@NgModule({
|
@NgModule({
|
||||||
imports: [
|
imports: [
|
||||||
RouterModule.forChild([
|
RouterModule.forChild([
|
||||||
{path: 'metadata', component: MetadataRegistryComponent, data: {title: 'admin.registries.metadata.title'}},
|
|
||||||
{
|
{
|
||||||
path: 'metadata/:schemaName',
|
path: 'metadata',
|
||||||
|
resolve: { breadcrumb: I18nBreadcrumbResolver },
|
||||||
|
data: {title: 'admin.registries.metadata.title', breadcrumbKey: 'admin.registries.metadata'},
|
||||||
|
children: [
|
||||||
|
{
|
||||||
|
path: '',
|
||||||
|
component: MetadataRegistryComponent
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: ':schemaName',
|
||||||
|
resolve: { breadcrumb: I18nBreadcrumbResolver },
|
||||||
component: MetadataSchemaComponent,
|
component: MetadataSchemaComponent,
|
||||||
data: {title: 'admin.registries.schema.title'}
|
data: {title: 'admin.registries.schema.title', breadcrumbKey: 'admin.registries.schema'}
|
||||||
|
}
|
||||||
|
]
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
path: BITSTREAMFORMATS_MODULE_PATH,
|
path: BITSTREAMFORMATS_MODULE_PATH,
|
||||||
|
resolve: { breadcrumb: I18nBreadcrumbResolver },
|
||||||
loadChildren: './bitstream-formats/bitstream-formats.module#BitstreamFormatsModule',
|
loadChildren: './bitstream-formats/bitstream-formats.module#BitstreamFormatsModule',
|
||||||
data: {title: 'admin.registries.bitstream-formats.title'}
|
data: {title: 'admin.registries.bitstream-formats.title', breadcrumbKey: 'admin.registries.bitstream-formats'}
|
||||||
},
|
},
|
||||||
])
|
])
|
||||||
]
|
]
|
||||||
|
@@ -4,6 +4,7 @@ import { BitstreamFormatsResolver } from './bitstream-formats.resolver';
|
|||||||
import { EditBitstreamFormatComponent } from './edit-bitstream-format/edit-bitstream-format.component';
|
import { EditBitstreamFormatComponent } from './edit-bitstream-format/edit-bitstream-format.component';
|
||||||
import { BitstreamFormatsComponent } from './bitstream-formats.component';
|
import { BitstreamFormatsComponent } from './bitstream-formats.component';
|
||||||
import { AddBitstreamFormatComponent } from './add-bitstream-format/add-bitstream-format.component';
|
import { AddBitstreamFormatComponent } from './add-bitstream-format/add-bitstream-format.component';
|
||||||
|
import { I18nBreadcrumbResolver } from '../../../core/breadcrumbs/i18n-breadcrumb.resolver';
|
||||||
|
|
||||||
const BITSTREAMFORMAT_EDIT_PATH = ':id/edit';
|
const BITSTREAMFORMAT_EDIT_PATH = ':id/edit';
|
||||||
const BITSTREAMFORMAT_ADD_PATH = 'add';
|
const BITSTREAMFORMAT_ADD_PATH = 'add';
|
||||||
@@ -17,14 +18,18 @@ const BITSTREAMFORMAT_ADD_PATH = 'add';
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
path: BITSTREAMFORMAT_ADD_PATH,
|
path: BITSTREAMFORMAT_ADD_PATH,
|
||||||
|
resolve: { breadcrumb: I18nBreadcrumbResolver },
|
||||||
component: AddBitstreamFormatComponent,
|
component: AddBitstreamFormatComponent,
|
||||||
|
data: {breadcrumbKey: 'admin.registries.bitstream-formats.create'}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
path: BITSTREAMFORMAT_EDIT_PATH,
|
path: BITSTREAMFORMAT_EDIT_PATH,
|
||||||
component: EditBitstreamFormatComponent,
|
component: EditBitstreamFormatComponent,
|
||||||
resolve: {
|
resolve: {
|
||||||
bitstreamFormat: BitstreamFormatsResolver
|
bitstreamFormat: BitstreamFormatsResolver,
|
||||||
}
|
breadcrumb: I18nBreadcrumbResolver
|
||||||
|
},
|
||||||
|
data: {breadcrumbKey: 'admin.registries.bitstream-formats.edit'}
|
||||||
},
|
},
|
||||||
])
|
])
|
||||||
],
|
],
|
||||||
|
@@ -11,7 +11,6 @@
|
|||||||
<ds-pagination
|
<ds-pagination
|
||||||
*ngIf="(metadataSchemas | async)?.payload?.totalElements > 0"
|
*ngIf="(metadataSchemas | async)?.payload?.totalElements > 0"
|
||||||
[paginationOptions]="config"
|
[paginationOptions]="config"
|
||||||
[pageInfoState]="(metadataSchemas | async)?.payload"
|
|
||||||
[collectionSize]="(metadataSchemas | async)?.payload?.totalElements"
|
[collectionSize]="(metadataSchemas | async)?.payload?.totalElements"
|
||||||
[hideGear]="true"
|
[hideGear]="true"
|
||||||
[hidePagerWhenSinglePage]="true"
|
[hidePagerWhenSinglePage]="true"
|
||||||
|
@@ -4,7 +4,7 @@ import { Observable, combineLatest as observableCombineLatest } from 'rxjs';
|
|||||||
import { RemoteData } from '../../../core/data/remote-data';
|
import { RemoteData } from '../../../core/data/remote-data';
|
||||||
import { PaginatedList } from '../../../core/data/paginated-list';
|
import { PaginatedList } from '../../../core/data/paginated-list';
|
||||||
import { PaginationComponentOptions } from '../../../shared/pagination/pagination-component-options.model';
|
import { PaginationComponentOptions } from '../../../shared/pagination/pagination-component-options.model';
|
||||||
import { map, take } from 'rxjs/operators';
|
import { filter, map, switchMap, take } from 'rxjs/operators';
|
||||||
import { hasValue } from '../../../shared/empty.util';
|
import { hasValue } from '../../../shared/empty.util';
|
||||||
import { RestResponse } from '../../../core/cache/response.models';
|
import { RestResponse } from '../../../core/cache/response.models';
|
||||||
import { zip } from 'rxjs/internal/observable/zip';
|
import { zip } from 'rxjs/internal/observable/zip';
|
||||||
@@ -12,6 +12,8 @@ import { NotificationsService } from '../../../shared/notifications/notification
|
|||||||
import { Route, Router } from '@angular/router';
|
import { Route, Router } from '@angular/router';
|
||||||
import { TranslateService } from '@ngx-translate/core';
|
import { TranslateService } from '@ngx-translate/core';
|
||||||
import { MetadataSchema } from '../../../core/metadata/metadata-schema.model';
|
import { MetadataSchema } from '../../../core/metadata/metadata-schema.model';
|
||||||
|
import { toFindListOptions } from '../../../shared/pagination/pagination.utils';
|
||||||
|
import { BehaviorSubject } from 'rxjs/internal/BehaviorSubject';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'ds-metadata-registry',
|
selector: 'ds-metadata-registry',
|
||||||
@@ -37,6 +39,11 @@ export class MetadataRegistryComponent {
|
|||||||
pageSize: 25
|
pageSize: 25
|
||||||
});
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Whether or not the list of MetadataSchemas needs an update
|
||||||
|
*/
|
||||||
|
needsUpdate$: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(true);
|
||||||
|
|
||||||
constructor(private registryService: RegistryService,
|
constructor(private registryService: RegistryService,
|
||||||
private notificationsService: NotificationsService,
|
private notificationsService: NotificationsService,
|
||||||
private router: Router,
|
private router: Router,
|
||||||
@@ -50,14 +57,17 @@ export class MetadataRegistryComponent {
|
|||||||
*/
|
*/
|
||||||
onPageChange(event) {
|
onPageChange(event) {
|
||||||
this.config.currentPage = event;
|
this.config.currentPage = event;
|
||||||
this.updateSchemas();
|
this.forceUpdateSchemas();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Update the list of schemas by fetching it from the rest api or cache
|
* Update the list of schemas by fetching it from the rest api or cache
|
||||||
*/
|
*/
|
||||||
private updateSchemas() {
|
private updateSchemas() {
|
||||||
this.metadataSchemas = this.registryService.getMetadataSchemas(this.config);
|
this.metadataSchemas = this.needsUpdate$.pipe(
|
||||||
|
filter((update) => update === true),
|
||||||
|
switchMap(() => this.registryService.getMetadataSchemas(toFindListOptions(this.config)))
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -65,8 +75,7 @@ export class MetadataRegistryComponent {
|
|||||||
* a new REST call
|
* a new REST call
|
||||||
*/
|
*/
|
||||||
public forceUpdateSchemas() {
|
public forceUpdateSchemas() {
|
||||||
this.registryService.clearMetadataSchemaRequests().subscribe();
|
this.needsUpdate$.next(true);
|
||||||
this.updateSchemas();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -125,6 +134,7 @@ export class MetadataRegistryComponent {
|
|||||||
* Delete all the selected metadata schemas
|
* Delete all the selected metadata schemas
|
||||||
*/
|
*/
|
||||||
deleteSchemas() {
|
deleteSchemas() {
|
||||||
|
this.registryService.clearMetadataSchemaRequests().subscribe();
|
||||||
this.registryService.getSelectedMetadataSchemas().pipe(take(1)).subscribe(
|
this.registryService.getSelectedMetadataSchemas().pipe(take(1)).subscribe(
|
||||||
(schemas) => {
|
(schemas) => {
|
||||||
const tasks$ = [];
|
const tasks$ = [];
|
||||||
|
@@ -21,7 +21,8 @@ describe('MetadataSchemaFormComponent', () => {
|
|||||||
const registryServiceStub = {
|
const registryServiceStub = {
|
||||||
getActiveMetadataSchema: () => observableOf(undefined),
|
getActiveMetadataSchema: () => observableOf(undefined),
|
||||||
createOrUpdateMetadataSchema: (schema: MetadataSchema) => observableOf(schema),
|
createOrUpdateMetadataSchema: (schema: MetadataSchema) => observableOf(schema),
|
||||||
cancelEditMetadataSchema: () => {}
|
cancelEditMetadataSchema: () => {},
|
||||||
|
clearMetadataSchemaRequests: () => observableOf(undefined)
|
||||||
};
|
};
|
||||||
const formBuilderServiceStub = {
|
const formBuilderServiceStub = {
|
||||||
createFormGroup: () => {
|
createFormGroup: () => {
|
||||||
|
@@ -128,6 +128,7 @@ export class MetadataSchemaFormComponent implements OnInit, OnDestroy {
|
|||||||
* Emit the updated/created schema using the EventEmitter submitForm
|
* Emit the updated/created schema using the EventEmitter submitForm
|
||||||
*/
|
*/
|
||||||
onSubmit() {
|
onSubmit() {
|
||||||
|
this.registryService.clearMetadataSchemaRequests().subscribe();
|
||||||
this.registryService.getActiveMetadataSchema().pipe(take(1)).subscribe(
|
this.registryService.getActiveMetadataSchema().pipe(take(1)).subscribe(
|
||||||
(schema) => {
|
(schema) => {
|
||||||
const values = {
|
const values = {
|
||||||
@@ -139,7 +140,7 @@ export class MetadataSchemaFormComponent implements OnInit, OnDestroy {
|
|||||||
this.submitForm.emit(newSchema);
|
this.submitForm.emit(newSchema);
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
this.registryService.createOrUpdateMetadataSchema(Object.assign(new MetadataSchema(), {
|
this.registryService.createOrUpdateMetadataSchema(Object.assign(new MetadataSchema(), schema, {
|
||||||
id: schema.id,
|
id: schema.id,
|
||||||
prefix: (values.prefix ? values.prefix : schema.prefix),
|
prefix: (values.prefix ? values.prefix : schema.prefix),
|
||||||
namespace: (values.namespace ? values.namespace : schema.namespace)
|
namespace: (values.namespace ? values.namespace : schema.namespace)
|
||||||
@@ -148,6 +149,7 @@ export class MetadataSchemaFormComponent implements OnInit, OnDestroy {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
this.clearFields();
|
this.clearFields();
|
||||||
|
this.registryService.cancelEditMetadataSchema();
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@@ -30,6 +30,7 @@ describe('MetadataFieldFormComponent', () => {
|
|||||||
createOrUpdateMetadataField: (field: MetadataField) => observableOf(field),
|
createOrUpdateMetadataField: (field: MetadataField) => observableOf(field),
|
||||||
cancelEditMetadataField: () => {},
|
cancelEditMetadataField: () => {},
|
||||||
cancelEditMetadataSchema: () => {},
|
cancelEditMetadataSchema: () => {},
|
||||||
|
clearMetadataFieldRequests: () => observableOf(undefined)
|
||||||
};
|
};
|
||||||
const formBuilderServiceStub = {
|
const formBuilderServiceStub = {
|
||||||
createFormGroup: () => {
|
createFormGroup: () => {
|
||||||
|
@@ -153,6 +153,7 @@ export class MetadataFieldFormComponent implements OnInit, OnDestroy {
|
|||||||
* Emit the updated/created field using the EventEmitter submitForm
|
* Emit the updated/created field using the EventEmitter submitForm
|
||||||
*/
|
*/
|
||||||
onSubmit() {
|
onSubmit() {
|
||||||
|
this.registryService.clearMetadataFieldRequests().subscribe();
|
||||||
this.registryService.getActiveMetadataField().pipe(take(1)).subscribe(
|
this.registryService.getActiveMetadataField().pipe(take(1)).subscribe(
|
||||||
(field) => {
|
(field) => {
|
||||||
const values = {
|
const values = {
|
||||||
@@ -166,7 +167,7 @@ export class MetadataFieldFormComponent implements OnInit, OnDestroy {
|
|||||||
this.submitForm.emit(newField);
|
this.submitForm.emit(newField);
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
this.registryService.createOrUpdateMetadataField(Object.assign(new MetadataField(), {
|
this.registryService.createOrUpdateMetadataField(Object.assign(new MetadataField(), field, {
|
||||||
id: field.id,
|
id: field.id,
|
||||||
schema: this.metadataSchema,
|
schema: this.metadataSchema,
|
||||||
element: (values.element ? values.element : field.element),
|
element: (values.element ? values.element : field.element),
|
||||||
@@ -177,6 +178,7 @@ export class MetadataFieldFormComponent implements OnInit, OnDestroy {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
this.clearFields();
|
this.clearFields();
|
||||||
|
this.registryService.cancelEditMetadataField();
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@@ -1,22 +1,23 @@
|
|||||||
<div class="container">
|
<div class="container">
|
||||||
<div class="metadata-schema row">
|
<div class="metadata-schema row">
|
||||||
<div class="col-12">
|
<div class="col-12" *ngVar="(metadataSchema$ | async) as schema">
|
||||||
|
|
||||||
<h2 id="header" class="border-bottom pb-2">{{'admin.registries.schema.head' | translate}}: "{{(metadataSchema | async)?.payload?.prefix}}"</h2>
|
<h2 id="header" class="border-bottom pb-2">{{'admin.registries.schema.head' | translate}}: "{{schema?.prefix}}"</h2>
|
||||||
|
|
||||||
<p id="description" class="pb-2">{{'admin.registries.schema.description' | translate:namespace }}</p>
|
<p id="description" class="pb-2">{{'admin.registries.schema.description' | translate:{ namespace: schema?.namespace } }}</p>
|
||||||
|
|
||||||
<ds-metadata-field-form
|
<ds-metadata-field-form
|
||||||
[metadataSchema]="(metadataSchema | async)?.payload"
|
[metadataSchema]="schema"
|
||||||
(submitForm)="forceUpdateFields()"></ds-metadata-field-form>
|
(submitForm)="forceUpdateFields()"></ds-metadata-field-form>
|
||||||
|
|
||||||
<h3>{{'admin.registries.schema.fields.head' | translate}}</h3>
|
<h3>{{'admin.registries.schema.fields.head' | translate}}</h3>
|
||||||
|
|
||||||
|
<ng-container *ngVar="(metadataFields$ | async)?.payload as fields">
|
||||||
<ds-pagination
|
<ds-pagination
|
||||||
*ngIf="(metadataFields | async)?.payload?.totalElements > 0"
|
*ngIf="fields?.totalElements > 0"
|
||||||
[paginationOptions]="config"
|
[paginationOptions]="config"
|
||||||
[pageInfoState]="(metadataFields | async)?.payload"
|
[pageInfoState]="fields"
|
||||||
[collectionSize]="(metadataFields | async)?.payload?.totalElements"
|
[collectionSize]="fields?.totalElements"
|
||||||
[hideGear]="false"
|
[hideGear]="false"
|
||||||
[hidePagerWhenSinglePage]="true"
|
[hidePagerWhenSinglePage]="true"
|
||||||
(pageChange)="onPageChange($event)">
|
(pageChange)="onPageChange($event)">
|
||||||
@@ -30,7 +31,7 @@
|
|||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
<tr *ngFor="let field of (metadataFields | async)?.payload?.page"
|
<tr *ngFor="let field of fields?.page"
|
||||||
[ngClass]="{'table-primary' : isActive(field) | async}">
|
[ngClass]="{'table-primary' : isActive(field) | async}">
|
||||||
<td>
|
<td>
|
||||||
<label>
|
<label>
|
||||||
@@ -39,7 +40,7 @@
|
|||||||
(change)="selectMetadataField(field, $event)">
|
(change)="selectMetadataField(field, $event)">
|
||||||
</label>
|
</label>
|
||||||
</td>
|
</td>
|
||||||
<td class="selectable-row" (click)="editField(field)">{{(metadataSchema | async)?.payload?.prefix}}.{{field.element}}<label *ngIf="field.qualifier">.</label>{{field.qualifier}}</td>
|
<td class="selectable-row" (click)="editField(field)">{{schema?.prefix}}.{{field.element}}<label *ngIf="field.qualifier">.</label>{{field.qualifier}}</td>
|
||||||
<td class="selectable-row" (click)="editField(field)">{{field.scopeNote}}</td>
|
<td class="selectable-row" (click)="editField(field)">{{field.scopeNote}}</td>
|
||||||
</tr>
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
@@ -47,14 +48,15 @@
|
|||||||
</div>
|
</div>
|
||||||
</ds-pagination>
|
</ds-pagination>
|
||||||
|
|
||||||
<div *ngIf="(metadataFields | async)?.payload?.totalElements == 0" class="alert alert-info w-100 mb-2" role="alert">
|
<div *ngIf="fields?.totalElements == 0" class="alert alert-info w-100 mb-2" role="alert">
|
||||||
{{'admin.registries.schema.fields.no-items' | translate}}
|
{{'admin.registries.schema.fields.no-items' | translate}}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
<button [routerLink]="['/admin/registries/metadata']" class="btn btn-primary">{{'admin.registries.schema.return' | translate}}</button>
|
<button [routerLink]="['/admin/registries/metadata']" class="btn btn-primary">{{'admin.registries.schema.return' | translate}}</button>
|
||||||
<button *ngIf="(metadataFields | async)?.payload?.page?.length > 0" type="submit" class="btn btn-danger float-right" (click)="deleteFields()">{{'admin.registries.schema.fields.table.delete' | translate}}</button>
|
<button *ngIf="fields?.page?.length > 0" type="submit" class="btn btn-danger float-right" (click)="deleteFields()">{{'admin.registries.schema.fields.table.delete' | translate}}</button>
|
||||||
</div>
|
</div>
|
||||||
|
</ng-container>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@@ -22,6 +22,7 @@ import { RestResponse } from '../../../core/cache/response.models';
|
|||||||
import { MetadataSchema } from '../../../core/metadata/metadata-schema.model';
|
import { MetadataSchema } from '../../../core/metadata/metadata-schema.model';
|
||||||
import { MetadataField } from '../../../core/metadata/metadata-field.model';
|
import { MetadataField } from '../../../core/metadata/metadata-field.model';
|
||||||
import { createSuccessfulRemoteDataObject$ } from '../../../shared/remote-data.utils';
|
import { createSuccessfulRemoteDataObject$ } from '../../../shared/remote-data.utils';
|
||||||
|
import { VarDirective } from '../../../shared/utils/var.directive';
|
||||||
|
|
||||||
describe('MetadataSchemaComponent', () => {
|
describe('MetadataSchemaComponent', () => {
|
||||||
let comp: MetadataSchemaComponent;
|
let comp: MetadataSchemaComponent;
|
||||||
@@ -124,7 +125,7 @@ describe('MetadataSchemaComponent', () => {
|
|||||||
beforeEach(async(() => {
|
beforeEach(async(() => {
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
imports: [CommonModule, RouterTestingModule.withRoutes([]), TranslateModule.forRoot(), NgbModule],
|
imports: [CommonModule, RouterTestingModule.withRoutes([]), TranslateModule.forRoot(), NgbModule],
|
||||||
declarations: [MetadataSchemaComponent, PaginationComponent, EnumKeysPipe],
|
declarations: [MetadataSchemaComponent, PaginationComponent, EnumKeysPipe, VarDirective],
|
||||||
providers: [
|
providers: [
|
||||||
{ provide: RegistryService, useValue: registryServiceStub },
|
{ provide: RegistryService, useValue: registryServiceStub },
|
||||||
{ provide: ActivatedRoute, useValue: activatedRouteStub },
|
{ provide: ActivatedRoute, useValue: activatedRouteStub },
|
||||||
|
@@ -5,7 +5,7 @@ import { Observable, combineLatest as observableCombineLatest } from 'rxjs';
|
|||||||
import { RemoteData } from '../../../core/data/remote-data';
|
import { RemoteData } from '../../../core/data/remote-data';
|
||||||
import { PaginatedList } from '../../../core/data/paginated-list';
|
import { PaginatedList } from '../../../core/data/paginated-list';
|
||||||
import { PaginationComponentOptions } from '../../../shared/pagination/pagination-component-options.model';
|
import { PaginationComponentOptions } from '../../../shared/pagination/pagination-component-options.model';
|
||||||
import { map, take } from 'rxjs/operators';
|
import { map, switchMap, take } from 'rxjs/operators';
|
||||||
import { hasValue } from '../../../shared/empty.util';
|
import { hasValue } from '../../../shared/empty.util';
|
||||||
import { RestResponse } from '../../../core/cache/response.models';
|
import { RestResponse } from '../../../core/cache/response.models';
|
||||||
import { zip } from 'rxjs/internal/observable/zip';
|
import { zip } from 'rxjs/internal/observable/zip';
|
||||||
@@ -13,6 +13,10 @@ import { NotificationsService } from '../../../shared/notifications/notification
|
|||||||
import { TranslateService } from '@ngx-translate/core';
|
import { TranslateService } from '@ngx-translate/core';
|
||||||
import { MetadataField } from '../../../core/metadata/metadata-field.model';
|
import { MetadataField } from '../../../core/metadata/metadata-field.model';
|
||||||
import { MetadataSchema } from '../../../core/metadata/metadata-schema.model';
|
import { MetadataSchema } from '../../../core/metadata/metadata-schema.model';
|
||||||
|
import { getFirstSucceededRemoteDataPayload } from '../../../core/shared/operators';
|
||||||
|
import { toFindListOptions } from '../../../shared/pagination/pagination.utils';
|
||||||
|
import { BehaviorSubject } from 'rxjs/internal/BehaviorSubject';
|
||||||
|
import { combineLatest } from 'rxjs/internal/observable/combineLatest';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'ds-metadata-schema',
|
selector: 'ds-metadata-schema',
|
||||||
@@ -24,21 +28,15 @@ import { MetadataSchema } from '../../../core/metadata/metadata-schema.model';
|
|||||||
* The admin can create, edit or delete metadata fields here.
|
* The admin can create, edit or delete metadata fields here.
|
||||||
*/
|
*/
|
||||||
export class MetadataSchemaComponent implements OnInit {
|
export class MetadataSchemaComponent implements OnInit {
|
||||||
|
|
||||||
/**
|
|
||||||
* The namespace of the metadata schema
|
|
||||||
*/
|
|
||||||
namespace;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The metadata schema
|
* The metadata schema
|
||||||
*/
|
*/
|
||||||
metadataSchema: Observable<RemoteData<MetadataSchema>>;
|
metadataSchema$: Observable<MetadataSchema>;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A list of all the fields attached to this metadata schema
|
* A list of all the fields attached to this metadata schema
|
||||||
*/
|
*/
|
||||||
metadataFields: Observable<RemoteData<PaginatedList<MetadataField>>>;
|
metadataFields$: Observable<RemoteData<PaginatedList<MetadataField>>>;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Pagination config used to display the list of metadata fields
|
* Pagination config used to display the list of metadata fields
|
||||||
@@ -49,6 +47,11 @@ export class MetadataSchemaComponent implements OnInit {
|
|||||||
pageSizeOptions: [25, 50, 100, 200]
|
pageSizeOptions: [25, 50, 100, 200]
|
||||||
});
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Whether or not the list of MetadataFields needs an update
|
||||||
|
*/
|
||||||
|
needsUpdate$: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(true);
|
||||||
|
|
||||||
constructor(private registryService: RegistryService,
|
constructor(private registryService: RegistryService,
|
||||||
private route: ActivatedRoute,
|
private route: ActivatedRoute,
|
||||||
private notificationsService: NotificationsService,
|
private notificationsService: NotificationsService,
|
||||||
@@ -68,7 +71,7 @@ export class MetadataSchemaComponent implements OnInit {
|
|||||||
* @param params
|
* @param params
|
||||||
*/
|
*/
|
||||||
initialize(params) {
|
initialize(params) {
|
||||||
this.metadataSchema = this.registryService.getMetadataSchemaByName(params.schemaName);
|
this.metadataSchema$ = this.registryService.getMetadataSchemaByName(params.schemaName).pipe(getFirstSucceededRemoteDataPayload());
|
||||||
this.updateFields();
|
this.updateFields();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -78,18 +81,20 @@ export class MetadataSchemaComponent implements OnInit {
|
|||||||
*/
|
*/
|
||||||
onPageChange(event) {
|
onPageChange(event) {
|
||||||
this.config.currentPage = event;
|
this.config.currentPage = event;
|
||||||
this.updateFields();
|
this.forceUpdateFields();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Update the list of fields by fetching it from the rest api or cache
|
* Update the list of fields by fetching it from the rest api or cache
|
||||||
*/
|
*/
|
||||||
private updateFields() {
|
private updateFields() {
|
||||||
this.metadataSchema.subscribe((schemaData) => {
|
this.metadataFields$ = combineLatest(this.metadataSchema$, this.needsUpdate$).pipe(
|
||||||
const schema = schemaData.payload;
|
switchMap(([schema, update]: [MetadataSchema, boolean]) => {
|
||||||
this.metadataFields = this.registryService.getMetadataFieldsBySchema(schema, this.config);
|
if (update) {
|
||||||
this.namespace = {namespace: schemaData.payload.namespace};
|
return this.registryService.getMetadataFieldsBySchema(schema, toFindListOptions(this.config));
|
||||||
});
|
}
|
||||||
|
})
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -97,8 +102,7 @@ export class MetadataSchemaComponent implements OnInit {
|
|||||||
* a new REST call
|
* a new REST call
|
||||||
*/
|
*/
|
||||||
public forceUpdateFields() {
|
public forceUpdateFields() {
|
||||||
this.registryService.clearMetadataFieldRequests().subscribe();
|
this.needsUpdate$.next(true);
|
||||||
this.updateFields();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -157,6 +161,7 @@ export class MetadataSchemaComponent implements OnInit {
|
|||||||
* Delete all the selected metadata fields
|
* Delete all the selected metadata fields
|
||||||
*/
|
*/
|
||||||
deleteFields() {
|
deleteFields() {
|
||||||
|
this.registryService.clearMetadataFieldRequests().subscribe();
|
||||||
this.registryService.getSelectedMetadataFields().pipe(take(1)).subscribe(
|
this.registryService.getSelectedMetadataFields().pipe(take(1)).subscribe(
|
||||||
(fields) => {
|
(fields) => {
|
||||||
const tasks$ = [];
|
const tasks$ = [];
|
||||||
|
@@ -32,6 +32,7 @@ import { ItemVersionHistoryComponent } from './item-version-history/item-version
|
|||||||
import { ItemAuthorizationsComponent } from './item-authorizations/item-authorizations.component';
|
import { ItemAuthorizationsComponent } from './item-authorizations/item-authorizations.component';
|
||||||
import { ResourcePolicyEditComponent } from '../../shared/resource-policies/edit/resource-policy-edit.component';
|
import { ResourcePolicyEditComponent } from '../../shared/resource-policies/edit/resource-policy-edit.component';
|
||||||
import { ResourcePolicyCreateComponent } from '../../shared/resource-policies/create/resource-policy-create.component';
|
import { ResourcePolicyCreateComponent } from '../../shared/resource-policies/create/resource-policy-create.component';
|
||||||
|
import { ObjectValuesPipe } from '../../shared/utils/object-values-pipe';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Module that contains all components related to the Edit Item page administrator functionality
|
* Module that contains all components related to the Edit Item page administrator functionality
|
||||||
@@ -75,7 +76,8 @@ import { ResourcePolicyCreateComponent } from '../../shared/resource-policies/cr
|
|||||||
ResourcePolicyCreateComponent,
|
ResourcePolicyCreateComponent,
|
||||||
],
|
],
|
||||||
providers: [
|
providers: [
|
||||||
BundleDataService
|
BundleDataService,
|
||||||
|
ObjectValuesPipe
|
||||||
]
|
]
|
||||||
})
|
})
|
||||||
export class EditItemPageModule {
|
export class EditItemPageModule {
|
||||||
|
@@ -36,7 +36,8 @@
|
|||||||
<ds-item-edit-bitstream-bundle *ngFor="let bundle of bundles"
|
<ds-item-edit-bitstream-bundle *ngFor="let bundle of bundles"
|
||||||
[bundle]="bundle"
|
[bundle]="bundle"
|
||||||
[item]="item"
|
[item]="item"
|
||||||
[columnSizes]="columnSizes">
|
[columnSizes]="columnSizes"
|
||||||
|
(dropObject)="dropBitstream(bundle, $event)">
|
||||||
</ds-item-edit-bitstream-bundle>
|
</ds-item-edit-bitstream-bundle>
|
||||||
</div>
|
</div>
|
||||||
<div *ngIf="bundles?.length === 0"
|
<div *ngIf="bundles?.length === 0"
|
||||||
|
@@ -188,8 +188,21 @@ describe('ItemBitstreamsComponent', () => {
|
|||||||
it('should not call deleteAndReturnResponse on the bitstreamService for the unmarked field', () => {
|
it('should not call deleteAndReturnResponse on the bitstreamService for the unmarked field', () => {
|
||||||
expect(bitstreamService.deleteAndReturnResponse).not.toHaveBeenCalledWith(bitstream1.id);
|
expect(bitstreamService.deleteAndReturnResponse).not.toHaveBeenCalledWith(bitstream1.id);
|
||||||
});
|
});
|
||||||
|
});
|
||||||
|
|
||||||
it('should send out a patch for the move operations', () => {
|
describe('when dropBitstream is called', () => {
|
||||||
|
const event = {
|
||||||
|
fromIndex: 0,
|
||||||
|
toIndex: 50,
|
||||||
|
// tslint:disable-next-line:no-empty
|
||||||
|
finish: () => {}
|
||||||
|
};
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
comp.dropBitstream(bundle, event);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should send out a patch for the move operation', () => {
|
||||||
expect(bundleService.patch).toHaveBeenCalled();
|
expect(bundleService.patch).toHaveBeenCalled();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
import { ChangeDetectorRef, Component, Inject, OnDestroy } from '@angular/core';
|
import { ChangeDetectorRef, Component, NgZone, OnDestroy } from '@angular/core';
|
||||||
import { AbstractItemUpdateComponent } from '../abstract-item-update/abstract-item-update.component';
|
import { AbstractItemUpdateComponent } from '../abstract-item-update/abstract-item-update.component';
|
||||||
import { filter, map, switchMap, take, tap } from 'rxjs/operators';
|
import { filter, map, switchMap, take } from 'rxjs/operators';
|
||||||
import { Observable } from 'rxjs/internal/Observable';
|
import { Observable } from 'rxjs/internal/Observable';
|
||||||
import { Subscription } from 'rxjs/internal/Subscription';
|
import { Subscription } from 'rxjs/internal/Subscription';
|
||||||
import { ItemDataService } from '../../../core/data/item-data.service';
|
import { ItemDataService } from '../../../core/data/item-data.service';
|
||||||
@@ -9,8 +9,8 @@ import { ActivatedRoute, Router } from '@angular/router';
|
|||||||
import { NotificationsService } from '../../../shared/notifications/notifications.service';
|
import { NotificationsService } from '../../../shared/notifications/notifications.service';
|
||||||
import { TranslateService } from '@ngx-translate/core';
|
import { TranslateService } from '@ngx-translate/core';
|
||||||
import { BitstreamDataService } from '../../../core/data/bitstream-data.service';
|
import { BitstreamDataService } from '../../../core/data/bitstream-data.service';
|
||||||
import { hasValue, isNotEmpty, isNotEmptyOperator } from '../../../shared/empty.util';
|
import { hasValue, isNotEmpty } from '../../../shared/empty.util';
|
||||||
import { zip as observableZip, combineLatest as observableCombineLatest, of as observableOf } from 'rxjs';
|
import { zip as observableZip, of as observableOf } from 'rxjs';
|
||||||
import { ErrorResponse, RestResponse } from '../../../core/cache/response.models';
|
import { ErrorResponse, RestResponse } from '../../../core/cache/response.models';
|
||||||
import { ObjectCacheService } from '../../../core/cache/object-cache.service';
|
import { ObjectCacheService } from '../../../core/cache/object-cache.service';
|
||||||
import { RequestService } from '../../../core/data/request.service';
|
import { RequestService } from '../../../core/data/request.service';
|
||||||
@@ -22,8 +22,6 @@ import { Bundle } from '../../../core/shared/bundle.model';
|
|||||||
import { FieldUpdate, FieldUpdates } from '../../../core/data/object-updates/object-updates.reducer';
|
import { FieldUpdate, FieldUpdates } from '../../../core/data/object-updates/object-updates.reducer';
|
||||||
import { Bitstream } from '../../../core/shared/bitstream.model';
|
import { Bitstream } from '../../../core/shared/bitstream.model';
|
||||||
import { FieldChangeType } from '../../../core/data/object-updates/object-updates.actions';
|
import { FieldChangeType } from '../../../core/data/object-updates/object-updates.actions';
|
||||||
import { Operation } from 'fast-json-patch';
|
|
||||||
import { MoveOperation } from 'fast-json-patch/lib/core';
|
|
||||||
import { BundleDataService } from '../../../core/data/bundle-data.service';
|
import { BundleDataService } from '../../../core/data/bundle-data.service';
|
||||||
import { PaginatedSearchOptions } from '../../../shared/search/paginated-search-options.model';
|
import { PaginatedSearchOptions } from '../../../shared/search/paginated-search-options.model';
|
||||||
import { ResponsiveColumnSizes } from '../../../shared/responsive-table-sizes/responsive-column-sizes';
|
import { ResponsiveColumnSizes } from '../../../shared/responsive-table-sizes/responsive-column-sizes';
|
||||||
@@ -90,7 +88,8 @@ export class ItemBitstreamsComponent extends AbstractItemUpdateComponent impleme
|
|||||||
public objectCache: ObjectCacheService,
|
public objectCache: ObjectCacheService,
|
||||||
public requestService: RequestService,
|
public requestService: RequestService,
|
||||||
public cdRef: ChangeDetectorRef,
|
public cdRef: ChangeDetectorRef,
|
||||||
public bundleService: BundleDataService
|
public bundleService: BundleDataService,
|
||||||
|
public zone: NgZone
|
||||||
) {
|
) {
|
||||||
super(itemService, objectUpdatesService, router, notificationsService, translateService, route);
|
super(itemService, objectUpdatesService, router, notificationsService, translateService, route);
|
||||||
}
|
}
|
||||||
@@ -143,7 +142,6 @@ export class ItemBitstreamsComponent extends AbstractItemUpdateComponent impleme
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Submit the current changes
|
* Submit the current changes
|
||||||
* Bitstreams that were dragged around send out a patch request with move operations to the rest API
|
|
||||||
* Bitstreams marked as deleted send out a delete request to the rest API
|
* Bitstreams marked as deleted send out a delete request to the rest API
|
||||||
* Display notifications and reset the current item/updates
|
* Display notifications and reset the current item/updates
|
||||||
*/
|
*/
|
||||||
@@ -151,32 +149,6 @@ export class ItemBitstreamsComponent extends AbstractItemUpdateComponent impleme
|
|||||||
this.submitting = true;
|
this.submitting = true;
|
||||||
const bundlesOnce$ = this.bundles$.pipe(take(1));
|
const bundlesOnce$ = this.bundles$.pipe(take(1));
|
||||||
|
|
||||||
// Fetch all move operations for each bundle
|
|
||||||
const moveOperations$ = bundlesOnce$.pipe(
|
|
||||||
switchMap((bundles: Bundle[]) => observableZip(...bundles.map((bundle: Bundle) =>
|
|
||||||
this.objectUpdatesService.getMoveOperations(bundle.self).pipe(
|
|
||||||
take(1),
|
|
||||||
map((operations: MoveOperation[]) => [...operations.map((operation: MoveOperation) => Object.assign(operation, {
|
|
||||||
from: `/_links/bitstreams${operation.from}/href`,
|
|
||||||
path: `/_links/bitstreams${operation.path}/href`
|
|
||||||
}))])
|
|
||||||
)
|
|
||||||
)))
|
|
||||||
);
|
|
||||||
|
|
||||||
// Send out an immediate patch request for each bundle
|
|
||||||
const patchResponses$ = observableCombineLatest(bundlesOnce$, moveOperations$).pipe(
|
|
||||||
switchMap(([bundles, moveOperationList]: [Bundle[], Operation[][]]) =>
|
|
||||||
observableZip(...bundles.map((bundle: Bundle, index: number) => {
|
|
||||||
if (isNotEmpty(moveOperationList[index])) {
|
|
||||||
return this.bundleService.patch(bundle, moveOperationList[index]);
|
|
||||||
} else {
|
|
||||||
return observableOf(undefined);
|
|
||||||
}
|
|
||||||
}))
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
// Fetch all removed bitstreams from the object update service
|
// Fetch all removed bitstreams from the object update service
|
||||||
const removedBitstreams$ = bundlesOnce$.pipe(
|
const removedBitstreams$ = bundlesOnce$.pipe(
|
||||||
switchMap((bundles: Bundle[]) => observableZip(
|
switchMap((bundles: Bundle[]) => observableZip(
|
||||||
@@ -201,19 +173,42 @@ export class ItemBitstreamsComponent extends AbstractItemUpdateComponent impleme
|
|||||||
);
|
);
|
||||||
|
|
||||||
// Perform the setup actions from above in order and display notifications
|
// Perform the setup actions from above in order and display notifications
|
||||||
patchResponses$.pipe(
|
removedResponses$.pipe(take(1)).subscribe((responses: RestResponse[]) => {
|
||||||
switchMap((responses: RestResponse[]) => {
|
|
||||||
this.displayNotifications('item.edit.bitstreams.notifications.move', responses);
|
|
||||||
return removedResponses$
|
|
||||||
}),
|
|
||||||
take(1)
|
|
||||||
).subscribe((responses: RestResponse[]) => {
|
|
||||||
this.displayNotifications('item.edit.bitstreams.notifications.remove', responses);
|
this.displayNotifications('item.edit.bitstreams.notifications.remove', responses);
|
||||||
this.reset();
|
this.reset();
|
||||||
this.submitting = false;
|
this.submitting = false;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A bitstream was dropped in a new location. Send out a Move Patch request to the REST API, display notifications,
|
||||||
|
* refresh the bundle's cache (so the lists can properly reload) and call the event's callback function (which will
|
||||||
|
* navigate the user to the correct page)
|
||||||
|
* @param bundle The bundle to send patch requests to
|
||||||
|
* @param event The event containing the index the bitstream came from and was dropped to
|
||||||
|
*/
|
||||||
|
dropBitstream(bundle: Bundle, event: any) {
|
||||||
|
this.zone.runOutsideAngular(() => {
|
||||||
|
if (hasValue(event) && hasValue(event.fromIndex) && hasValue(event.toIndex) && hasValue(event.finish)) {
|
||||||
|
const moveOperation = Object.assign({
|
||||||
|
op: 'move',
|
||||||
|
from: `/_links/bitstreams/${event.fromIndex}/href`,
|
||||||
|
path: `/_links/bitstreams/${event.toIndex}/href`
|
||||||
|
});
|
||||||
|
this.bundleService.patch(bundle, [moveOperation]).pipe(take(1)).subscribe((response: RestResponse) => {
|
||||||
|
this.zone.run(() => {
|
||||||
|
this.displayNotifications('item.edit.bitstreams.notifications.move', [response]);
|
||||||
|
// Remove all cached requests from this bundle and call the event's callback when the requests are cleared
|
||||||
|
this.requestService.removeByHrefSubstring(bundle.self).pipe(
|
||||||
|
filter((isCached) => isCached),
|
||||||
|
take(1)
|
||||||
|
).subscribe(() => event.finish());
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Display notifications
|
* Display notifications
|
||||||
* - Error notification for each failed response with their message
|
* - Error notification for each failed response with their message
|
||||||
|
@@ -17,5 +17,5 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<ds-paginated-drag-and-drop-bitstream-list [bundle]="bundle" [columnSizes]="columnSizes"></ds-paginated-drag-and-drop-bitstream-list>
|
<ds-paginated-drag-and-drop-bitstream-list [bundle]="bundle" [columnSizes]="columnSizes" (dropObject)="dropObject.emit($event)"></ds-paginated-drag-and-drop-bitstream-list>
|
||||||
</ng-template>
|
</ng-template>
|
||||||
|
@@ -1,4 +1,4 @@
|
|||||||
import { Component, Input, OnInit, ViewChild, ViewContainerRef } from '@angular/core';
|
import { Component, EventEmitter, Input, OnInit, Output, ViewChild, ViewContainerRef } from '@angular/core';
|
||||||
import { Bundle } from '../../../../core/shared/bundle.model';
|
import { Bundle } from '../../../../core/shared/bundle.model';
|
||||||
import { Item } from '../../../../core/shared/item.model';
|
import { Item } from '../../../../core/shared/item.model';
|
||||||
import { ResponsiveColumnSizes } from '../../../../shared/responsive-table-sizes/responsive-column-sizes';
|
import { ResponsiveColumnSizes } from '../../../../shared/responsive-table-sizes/responsive-column-sizes';
|
||||||
@@ -36,6 +36,13 @@ export class ItemEditBitstreamBundleComponent implements OnInit {
|
|||||||
*/
|
*/
|
||||||
@Input() columnSizes: ResponsiveTableSizes;
|
@Input() columnSizes: ResponsiveTableSizes;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Send an event when the user drops an object on the pagination
|
||||||
|
* The event contains details about the index the object came from and is dropped to (across the entirety of the list,
|
||||||
|
* not just within a single page)
|
||||||
|
*/
|
||||||
|
@Output() dropObject: EventEmitter<any> = new EventEmitter<any>();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The bootstrap sizes used for the Bundle Name column
|
* The bootstrap sizes used for the Bundle Name column
|
||||||
* This column stretches over the first 3 columns and thus is a combination of their sizes processed in ngOnInit
|
* This column stretches over the first 3 columns and thus is a combination of their sizes processed in ngOnInit
|
||||||
|
@@ -7,18 +7,20 @@
|
|||||||
[collectionSize]="(objectsRD$ | async)?.payload?.totalElements"
|
[collectionSize]="(objectsRD$ | async)?.payload?.totalElements"
|
||||||
[disableRouteParameterUpdate]="true"
|
[disableRouteParameterUpdate]="true"
|
||||||
(pageChange)="switchPage($event)">
|
(pageChange)="switchPage($event)">
|
||||||
|
<ng-container *ngIf="!(loading$ | async)">
|
||||||
<div [id]="bundle.id" class="bundle-bitstreams-list"
|
<div [id]="bundle.id" class="bundle-bitstreams-list"
|
||||||
[ngClass]="{'mb-3': (objectsRD$ | async)?.payload?.totalElements > pageSize}"
|
[ngClass]="{'mb-3': (objectsRD$ | async)?.payload?.totalElements > pageSize}"
|
||||||
*ngVar="((updates$ | async) | dsObjectValues) as updateValues" cdkDropList (cdkDropListDropped)="drop($event)">
|
*ngVar="(updates$ | async) as updates" cdkDropList (cdkDropListDropped)="drop($event)">
|
||||||
<div class="row bitstream-row" *ngFor="let updateValue of updateValues" cdkDrag
|
<ng-container *ngIf="updates">
|
||||||
[id]="updateValue.field.uuid"
|
<div class="row bitstream-row" *ngFor="let uuid of customOrder" cdkDrag
|
||||||
|
[id]="uuid"
|
||||||
[ngClass]="{
|
[ngClass]="{
|
||||||
'table-warning': updateValue.changeType === 0,
|
'table-warning': updates[uuid].changeType === 0,
|
||||||
'table-danger': updateValue.changeType === 2,
|
'table-danger': updates[uuid].changeType === 2,
|
||||||
'table-success': updateValue.changeType === 1,
|
'table-success': updates[uuid].changeType === 1,
|
||||||
'bg-white': updateValue.changeType === undefined
|
'bg-white': updates[uuid].changeType === undefined
|
||||||
}">
|
}">
|
||||||
<ds-item-edit-bitstream [fieldUpdate]="updateValue"
|
<ds-item-edit-bitstream [fieldUpdate]="updates[uuid]"
|
||||||
[bundleUrl]="bundle.self"
|
[bundleUrl]="bundle.self"
|
||||||
[columnSizes]="columnSizes">
|
[columnSizes]="columnSizes">
|
||||||
<div class="d-flex align-items-center" slot="drag-handle" cdkDragHandle>
|
<div class="d-flex align-items-center" slot="drag-handle" cdkDragHandle>
|
||||||
@@ -26,5 +28,8 @@
|
|||||||
</div>
|
</div>
|
||||||
</ds-item-edit-bitstream>
|
</ds-item-edit-bitstream>
|
||||||
</div>
|
</div>
|
||||||
|
</ng-container>
|
||||||
</div>
|
</div>
|
||||||
|
</ng-container>
|
||||||
|
<ds-loading *ngIf="(loading$ | async)" [message]="'loading.bitstreams' | translate"></ds-loading>
|
||||||
</ds-pagination>
|
</ds-pagination>
|
||||||
|
@@ -16,12 +16,15 @@ import { ResponsiveTableSizes } from '../../../../../shared/responsive-table-siz
|
|||||||
import { ResponsiveColumnSizes } from '../../../../../shared/responsive-table-sizes/responsive-column-sizes';
|
import { ResponsiveColumnSizes } from '../../../../../shared/responsive-table-sizes/responsive-column-sizes';
|
||||||
import { createSuccessfulRemoteDataObject$ } from '../../../../../shared/remote-data.utils';
|
import { createSuccessfulRemoteDataObject$ } from '../../../../../shared/remote-data.utils';
|
||||||
import { createPaginatedList } from '../../../../../shared/testing/utils.test';
|
import { createPaginatedList } from '../../../../../shared/testing/utils.test';
|
||||||
|
import { RequestService } from '../../../../../core/data/request.service';
|
||||||
|
|
||||||
describe('PaginatedDragAndDropBitstreamListComponent', () => {
|
describe('PaginatedDragAndDropBitstreamListComponent', () => {
|
||||||
let comp: PaginatedDragAndDropBitstreamListComponent;
|
let comp: PaginatedDragAndDropBitstreamListComponent;
|
||||||
let fixture: ComponentFixture<PaginatedDragAndDropBitstreamListComponent>;
|
let fixture: ComponentFixture<PaginatedDragAndDropBitstreamListComponent>;
|
||||||
let objectUpdatesService: ObjectUpdatesService;
|
let objectUpdatesService: ObjectUpdatesService;
|
||||||
let bundleService: BundleDataService;
|
let bundleService: BundleDataService;
|
||||||
|
let objectValuesPipe: ObjectValuesPipe;
|
||||||
|
let requestService: RequestService;
|
||||||
|
|
||||||
const columnSizes = new ResponsiveTableSizes([
|
const columnSizes = new ResponsiveTableSizes([
|
||||||
new ResponsiveColumnSizes(2, 2, 3, 4, 4),
|
new ResponsiveColumnSizes(2, 2, 3, 4, 4),
|
||||||
@@ -97,15 +100,24 @@ describe('PaginatedDragAndDropBitstreamListComponent', () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
bundleService = jasmine.createSpyObj('bundleService', {
|
bundleService = jasmine.createSpyObj('bundleService', {
|
||||||
getBitstreams: createSuccessfulRemoteDataObject$(createPaginatedList([bitstream1, bitstream2]))
|
getBitstreams: createSuccessfulRemoteDataObject$(createPaginatedList([bitstream1, bitstream2])),
|
||||||
|
getBitstreamsEndpoint: observableOf('')
|
||||||
|
});
|
||||||
|
|
||||||
|
objectValuesPipe = new ObjectValuesPipe();
|
||||||
|
|
||||||
|
requestService = jasmine.createSpyObj('requestService', {
|
||||||
|
hasByHrefObservable: observableOf(true)
|
||||||
});
|
});
|
||||||
|
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
imports: [TranslateModule.forRoot()],
|
imports: [TranslateModule.forRoot()],
|
||||||
declarations: [PaginatedDragAndDropBitstreamListComponent, VarDirective, ObjectValuesPipe],
|
declarations: [PaginatedDragAndDropBitstreamListComponent, VarDirective],
|
||||||
providers: [
|
providers: [
|
||||||
{ provide: ObjectUpdatesService, useValue: objectUpdatesService },
|
{ provide: ObjectUpdatesService, useValue: objectUpdatesService },
|
||||||
{ provide: BundleDataService, useValue: bundleService }
|
{ provide: BundleDataService, useValue: bundleService },
|
||||||
|
{ provide: ObjectValuesPipe, useValue: objectValuesPipe },
|
||||||
|
{ provide: RequestService, useValue: requestService }
|
||||||
], schemas: [
|
], schemas: [
|
||||||
NO_ERRORS_SCHEMA
|
NO_ERRORS_SCHEMA
|
||||||
]
|
]
|
||||||
|
@@ -8,6 +8,8 @@ import { switchMap } from 'rxjs/operators';
|
|||||||
import { PaginatedSearchOptions } from '../../../../../shared/search/paginated-search-options.model';
|
import { PaginatedSearchOptions } from '../../../../../shared/search/paginated-search-options.model';
|
||||||
import { ResponsiveTableSizes } from '../../../../../shared/responsive-table-sizes/responsive-table-sizes';
|
import { ResponsiveTableSizes } from '../../../../../shared/responsive-table-sizes/responsive-table-sizes';
|
||||||
import { followLink } from '../../../../../shared/utils/follow-link-config.model';
|
import { followLink } from '../../../../../shared/utils/follow-link-config.model';
|
||||||
|
import { ObjectValuesPipe } from '../../../../../shared/utils/object-values-pipe';
|
||||||
|
import { RequestService } from '../../../../../core/data/request.service';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'ds-paginated-drag-and-drop-bitstream-list',
|
selector: 'ds-paginated-drag-and-drop-bitstream-list',
|
||||||
@@ -33,8 +35,10 @@ export class PaginatedDragAndDropBitstreamListComponent extends AbstractPaginate
|
|||||||
|
|
||||||
constructor(protected objectUpdatesService: ObjectUpdatesService,
|
constructor(protected objectUpdatesService: ObjectUpdatesService,
|
||||||
protected elRef: ElementRef,
|
protected elRef: ElementRef,
|
||||||
protected bundleService: BundleDataService) {
|
protected objectValuesPipe: ObjectValuesPipe,
|
||||||
super(objectUpdatesService, elRef);
|
protected bundleService: BundleDataService,
|
||||||
|
protected requestService: RequestService) {
|
||||||
|
super(objectUpdatesService, elRef, objectValuesPipe);
|
||||||
}
|
}
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
@@ -46,12 +50,18 @@ export class PaginatedDragAndDropBitstreamListComponent extends AbstractPaginate
|
|||||||
*/
|
*/
|
||||||
initializeObjectsRD(): void {
|
initializeObjectsRD(): void {
|
||||||
this.objectsRD$ = this.currentPage$.pipe(
|
this.objectsRD$ = this.currentPage$.pipe(
|
||||||
switchMap((page: number) => this.bundleService.getBitstreams(
|
switchMap((page: number) => {
|
||||||
|
const paginatedOptions = new PaginatedSearchOptions({pagination: Object.assign({}, this.options, { currentPage: page })});
|
||||||
|
return this.bundleService.getBitstreamsEndpoint(this.bundle.id, paginatedOptions).pipe(
|
||||||
|
switchMap((href) => this.requestService.hasByHrefObservable(href)),
|
||||||
|
switchMap(() => this.bundleService.getBitstreams(
|
||||||
this.bundle.id,
|
this.bundle.id,
|
||||||
new PaginatedSearchOptions({pagination: Object.assign({}, this.options, { currentPage: page })}),
|
paginatedOptions,
|
||||||
followLink('format')
|
followLink('format')
|
||||||
))
|
))
|
||||||
);
|
);
|
||||||
|
})
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@@ -7,9 +7,9 @@
|
|||||||
|
|
||||||
</div>
|
</div>
|
||||||
<div class="add">
|
<div class="add">
|
||||||
<a class="btn btn-lg btn-primary mt-1 ml-2" [routerLink]="['/submit']" role="button">
|
<button class="btn btn-lg btn-primary mt-1 ml-2" (click)="openDialog()" role="button">
|
||||||
<i class="fa fa-plus-circle" aria-hidden="true"></i> {{'mydspace.new-submission' | translate}}
|
<i class="fa fa-plus-circle" aria-hidden="true"></i> {{'mydspace.new-submission' | translate}}
|
||||||
</a>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
import { ChangeDetectorRef, Component, NO_ERRORS_SCHEMA } from '@angular/core';
|
import { ChangeDetectorRef, Component, NO_ERRORS_SCHEMA } from '@angular/core';
|
||||||
import { async, ComponentFixture, inject, TestBed } from '@angular/core/testing';
|
import { async, ComponentFixture, inject, TestBed, tick, fakeAsync } from '@angular/core/testing';
|
||||||
import { RouterTestingModule } from '@angular/router/testing';
|
import { RouterTestingModule } from '@angular/router/testing';
|
||||||
|
|
||||||
import { Store } from '@ngrx/store';
|
import { Store } from '@ngrx/store';
|
||||||
@@ -21,6 +21,8 @@ import { NotificationsServiceStub } from '../../shared/testing/notifications-ser
|
|||||||
import { SharedModule } from '../../shared/shared.module';
|
import { SharedModule } from '../../shared/shared.module';
|
||||||
import { getMockScrollToService } from '../../shared/mocks/scroll-to-service.mock';
|
import { getMockScrollToService } from '../../shared/mocks/scroll-to-service.mock';
|
||||||
import { UploaderService } from '../../shared/uploader/uploader.service';
|
import { UploaderService } from '../../shared/uploader/uploader.service';
|
||||||
|
import { By } from '@angular/platform-browser';
|
||||||
|
import { NgbModal } from '@ng-bootstrap/ng-bootstrap';
|
||||||
|
|
||||||
describe('MyDSpaceNewSubmissionComponent test', () => {
|
describe('MyDSpaceNewSubmissionComponent test', () => {
|
||||||
|
|
||||||
@@ -54,6 +56,11 @@ describe('MyDSpaceNewSubmissionComponent test', () => {
|
|||||||
{ provide: ScrollToService, useValue: getMockScrollToService() },
|
{ provide: ScrollToService, useValue: getMockScrollToService() },
|
||||||
{ provide: Store, useValue: store },
|
{ provide: Store, useValue: store },
|
||||||
{ provide: TranslateService, useValue: translateService },
|
{ provide: TranslateService, useValue: translateService },
|
||||||
|
{
|
||||||
|
provide: NgbModal, useValue: {
|
||||||
|
open: () => {/*comment*/}
|
||||||
|
}
|
||||||
|
},
|
||||||
ChangeDetectorRef,
|
ChangeDetectorRef,
|
||||||
MyDSpaceNewSubmissionComponent,
|
MyDSpaceNewSubmissionComponent,
|
||||||
UploaderService
|
UploaderService
|
||||||
@@ -86,6 +93,25 @@ describe('MyDSpaceNewSubmissionComponent test', () => {
|
|||||||
}));
|
}));
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('', () => {
|
||||||
|
let fixture: ComponentFixture<MyDSpaceNewSubmissionComponent>;
|
||||||
|
let comp: MyDSpaceNewSubmissionComponent;
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
fixture = TestBed.createComponent(MyDSpaceNewSubmissionComponent);
|
||||||
|
comp = fixture.componentInstance;
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should call app.openDialog', () => {
|
||||||
|
spyOn(comp, 'openDialog');
|
||||||
|
const submissionButton = fixture.debugElement.query(By.css('button.btn-primary'));
|
||||||
|
submissionButton.triggerEventHandler('click', {
|
||||||
|
preventDefault: () => {/**/
|
||||||
|
}
|
||||||
|
});
|
||||||
|
expect(comp.openDialog).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// declare a test component
|
// declare a test component
|
||||||
|
@@ -15,6 +15,9 @@ import { HALEndpointService } from '../../core/shared/hal-endpoint.service';
|
|||||||
import { NotificationType } from '../../shared/notifications/models/notification-type';
|
import { NotificationType } from '../../shared/notifications/models/notification-type';
|
||||||
import { hasValue } from '../../shared/empty.util';
|
import { hasValue } from '../../shared/empty.util';
|
||||||
import { SearchResult } from '../../shared/search/search-result.model';
|
import { SearchResult } from '../../shared/search/search-result.model';
|
||||||
|
import { Router } from '@angular/router';
|
||||||
|
import { NgbModal } from '@ng-bootstrap/ng-bootstrap';
|
||||||
|
import { CreateItemParentSelectorComponent } from 'src/app/shared/dso-selector/modal-wrappers/create-item-parent-selector/create-item-parent-selector.component';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This component represents the whole mydspace page header
|
* This component represents the whole mydspace page header
|
||||||
@@ -55,7 +58,9 @@ export class MyDSpaceNewSubmissionComponent implements OnDestroy, OnInit {
|
|||||||
private halService: HALEndpointService,
|
private halService: HALEndpointService,
|
||||||
private notificationsService: NotificationsService,
|
private notificationsService: NotificationsService,
|
||||||
private store: Store<SubmissionState>,
|
private store: Store<SubmissionState>,
|
||||||
private translate: TranslateService) {
|
private translate: TranslateService,
|
||||||
|
private router: Router,
|
||||||
|
private modalService: NgbModal) {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -105,6 +110,14 @@ export class MyDSpaceNewSubmissionComponent implements OnDestroy, OnInit {
|
|||||||
this.notificationsService.error(null, this.translate.get('mydspace.upload.upload-failed'));
|
this.notificationsService.error(null, this.translate.get('mydspace.upload.upload-failed'));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Method called on clicking the button "New Submition", It opens a dialog for
|
||||||
|
* select a collection.
|
||||||
|
*/
|
||||||
|
openDialog() {
|
||||||
|
this.modalService.open(CreateItemParentSelectorComponent);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Unsubscribe from the subscription
|
* Unsubscribe from the subscription
|
||||||
*/
|
*/
|
||||||
|
@@ -14,7 +14,7 @@ describe('I18nBreadcrumbResolver', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('should resolve the breadcrumb config', () => {
|
it('should resolve the breadcrumb config', () => {
|
||||||
const resolvedConfig = resolver.resolve({ data: { breadcrumbKey: i18nKey }, url: [path] } as any, {} as any);
|
const resolvedConfig = resolver.resolve({ data: { breadcrumbKey: i18nKey }, url: [path], pathFromRoot: [{ url: [path] }] } as any, {} as any);
|
||||||
const expectedConfig = { provider: i18nBreadcrumbService, key: i18nKey, url: path };
|
const expectedConfig = { provider: i18nBreadcrumbService, key: i18nKey, url: path };
|
||||||
expect(resolvedConfig).toEqual(expectedConfig);
|
expect(resolvedConfig).toEqual(expectedConfig);
|
||||||
});
|
});
|
||||||
|
@@ -25,7 +25,17 @@ export class I18nBreadcrumbResolver implements Resolve<BreadcrumbConfig<string>>
|
|||||||
if (hasNoValue(key)) {
|
if (hasNoValue(key)) {
|
||||||
throw new Error('You provided an i18nBreadcrumbResolver for url \"' + route.url + '\" but no breadcrumbKey in the route\'s data')
|
throw new Error('You provided an i18nBreadcrumbResolver for url \"' + route.url + '\" but no breadcrumbKey in the route\'s data')
|
||||||
}
|
}
|
||||||
const fullPath = route.url.join('');
|
const fullPath = this.getResolvedUrl(route);
|
||||||
return { provider: this.breadcrumbService, key: key, url: fullPath };
|
return { provider: this.breadcrumbService, key: key, url: fullPath };
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Resolve the full URL of an ActivatedRouteSnapshot
|
||||||
|
* @param route
|
||||||
|
*/
|
||||||
|
getResolvedUrl(route: ActivatedRouteSnapshot): string {
|
||||||
|
return route.pathFromRoot
|
||||||
|
.map((v) => v.url.map((segment) => segment.toString()).join('/'))
|
||||||
|
.join('/');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
45
src/app/core/cache/response.models.ts
vendored
45
src/app/core/cache/response.models.ts
vendored
@@ -6,9 +6,6 @@ import { ConfigObject } from '../config/models/config.model';
|
|||||||
import { FacetValue } from '../../shared/search/facet-value.model';
|
import { FacetValue } from '../../shared/search/facet-value.model';
|
||||||
import { SearchFilterConfig } from '../../shared/search/search-filter-config.model';
|
import { SearchFilterConfig } from '../../shared/search/search-filter-config.model';
|
||||||
import { IntegrationModel } from '../integration/models/integration.model';
|
import { IntegrationModel } from '../integration/models/integration.model';
|
||||||
import { RegistryMetadataschemasResponse } from '../registry/registry-metadataschemas-response.model';
|
|
||||||
import { RegistryMetadatafieldsResponse } from '../registry/registry-metadatafields-response.model';
|
|
||||||
import { RegistryBitstreamformatsResponse } from '../registry/registry-bitstreamformats-response.model';
|
|
||||||
import { PaginatedList } from '../data/paginated-list';
|
import { PaginatedList } from '../data/paginated-list';
|
||||||
import { SubmissionObject } from '../submission/models/submission-object.model';
|
import { SubmissionObject } from '../submission/models/submission-object.model';
|
||||||
import { DSpaceObject } from '../shared/dspace-object.model';
|
import { DSpaceObject } from '../shared/dspace-object.model';
|
||||||
@@ -40,48 +37,6 @@ export class DSOSuccessResponse extends RestResponse {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* A successful response containing a list of MetadataSchemas wrapped in a RegistryMetadataschemasResponse
|
|
||||||
*/
|
|
||||||
export class RegistryMetadataschemasSuccessResponse extends RestResponse {
|
|
||||||
constructor(
|
|
||||||
public metadataschemasResponse: RegistryMetadataschemasResponse,
|
|
||||||
public statusCode: number,
|
|
||||||
public statusText: string,
|
|
||||||
public pageInfo?: PageInfo
|
|
||||||
) {
|
|
||||||
super(true, statusCode, statusText);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A successful response containing a list of MetadataFields wrapped in a RegistryMetadatafieldsResponse
|
|
||||||
*/
|
|
||||||
export class RegistryMetadatafieldsSuccessResponse extends RestResponse {
|
|
||||||
constructor(
|
|
||||||
public metadatafieldsResponse: RegistryMetadatafieldsResponse,
|
|
||||||
public statusCode: number,
|
|
||||||
public statusText: string,
|
|
||||||
public pageInfo?: PageInfo
|
|
||||||
) {
|
|
||||||
super(true, statusCode, statusText);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A successful response containing a list of BitstreamFormats wrapped in a RegistryBitstreamformatsResponse
|
|
||||||
*/
|
|
||||||
export class RegistryBitstreamformatsSuccessResponse extends RestResponse {
|
|
||||||
constructor(
|
|
||||||
public bitstreamformatsResponse: RegistryBitstreamformatsResponse,
|
|
||||||
public statusCode: number,
|
|
||||||
public statusText: string,
|
|
||||||
public pageInfo?: PageInfo
|
|
||||||
) {
|
|
||||||
super(true, statusCode, statusText);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A successful response containing exactly one MetadataSchema
|
* A successful response containing exactly one MetadataSchema
|
||||||
*/
|
*/
|
||||||
|
@@ -69,13 +69,8 @@ import { ItemDataService } from './data/item-data.service';
|
|||||||
import { LicenseDataService } from './data/license-data.service';
|
import { LicenseDataService } from './data/license-data.service';
|
||||||
import { LookupRelationService } from './data/lookup-relation.service';
|
import { LookupRelationService } from './data/lookup-relation.service';
|
||||||
import { MappedCollectionsReponseParsingService } from './data/mapped-collections-reponse-parsing.service';
|
import { MappedCollectionsReponseParsingService } from './data/mapped-collections-reponse-parsing.service';
|
||||||
import { MetadatafieldParsingService } from './data/metadatafield-parsing.service';
|
|
||||||
import { MetadataschemaParsingService } from './data/metadataschema-parsing.service';
|
|
||||||
import { MyDSpaceResponseParsingService } from './data/mydspace-response-parsing.service';
|
import { MyDSpaceResponseParsingService } from './data/mydspace-response-parsing.service';
|
||||||
import { ObjectUpdatesService } from './data/object-updates/object-updates.service';
|
import { ObjectUpdatesService } from './data/object-updates/object-updates.service';
|
||||||
import { RegistryBitstreamformatsResponseParsingService } from './data/registry-bitstreamformats-response-parsing.service';
|
|
||||||
import { RegistryMetadatafieldsResponseParsingService } from './data/registry-metadatafields-response-parsing.service';
|
|
||||||
import { RegistryMetadataschemasResponseParsingService } from './data/registry-metadataschemas-response-parsing.service';
|
|
||||||
import { RelationshipTypeService } from './data/relationship-type.service';
|
import { RelationshipTypeService } from './data/relationship-type.service';
|
||||||
import { RelationshipService } from './data/relationship.service';
|
import { RelationshipService } from './data/relationship.service';
|
||||||
import { ResourcePolicyService } from './resource-policy/resource-policy.service';
|
import { ResourcePolicyService } from './resource-policy/resource-policy.service';
|
||||||
@@ -146,6 +141,8 @@ import { VersionHistory } from './shared/version-history.model';
|
|||||||
import { WorkflowActionDataService } from './data/workflow-action-data.service';
|
import { WorkflowActionDataService } from './data/workflow-action-data.service';
|
||||||
import { WorkflowAction } from './tasks/models/workflow-action-object.model';
|
import { WorkflowAction } from './tasks/models/workflow-action-object.model';
|
||||||
import { LocaleInterceptor } from './locale/locale.interceptor';
|
import { LocaleInterceptor } from './locale/locale.interceptor';
|
||||||
|
import { MetadataSchemaDataService } from './data/metadata-schema-data.service';
|
||||||
|
import { MetadataFieldDataService } from './data/metadata-field-data.service';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* When not in production, endpoint responses can be mocked for testing purposes
|
* When not in production, endpoint responses can be mocked for testing purposes
|
||||||
@@ -202,9 +199,6 @@ const PROVIDERS = [
|
|||||||
FacetValueResponseParsingService,
|
FacetValueResponseParsingService,
|
||||||
FacetValueMapResponseParsingService,
|
FacetValueMapResponseParsingService,
|
||||||
FacetConfigResponseParsingService,
|
FacetConfigResponseParsingService,
|
||||||
RegistryMetadataschemasResponseParsingService,
|
|
||||||
RegistryMetadatafieldsResponseParsingService,
|
|
||||||
RegistryBitstreamformatsResponseParsingService,
|
|
||||||
MappedCollectionsReponseParsingService,
|
MappedCollectionsReponseParsingService,
|
||||||
DebugResponseParsingService,
|
DebugResponseParsingService,
|
||||||
SearchResponseParsingService,
|
SearchResponseParsingService,
|
||||||
@@ -224,8 +218,6 @@ const PROVIDERS = [
|
|||||||
JsonPatchOperationsBuilder,
|
JsonPatchOperationsBuilder,
|
||||||
AuthorityService,
|
AuthorityService,
|
||||||
IntegrationResponseParsingService,
|
IntegrationResponseParsingService,
|
||||||
MetadataschemaParsingService,
|
|
||||||
MetadatafieldParsingService,
|
|
||||||
UploaderService,
|
UploaderService,
|
||||||
UUIDService,
|
UUIDService,
|
||||||
NotificationsService,
|
NotificationsService,
|
||||||
@@ -265,6 +257,8 @@ const PROVIDERS = [
|
|||||||
LicenseDataService,
|
LicenseDataService,
|
||||||
ItemTypeDataService,
|
ItemTypeDataService,
|
||||||
WorkflowActionDataService,
|
WorkflowActionDataService,
|
||||||
|
MetadataSchemaDataService,
|
||||||
|
MetadataFieldDataService,
|
||||||
// register AuthInterceptor as HttpInterceptor
|
// register AuthInterceptor as HttpInterceptor
|
||||||
{
|
{
|
||||||
provide: HTTP_INTERCEPTORS,
|
provide: HTTP_INTERCEPTORS,
|
||||||
|
@@ -88,10 +88,12 @@ export class BundleDataService extends DataService<Bundle> {
|
|||||||
/**
|
/**
|
||||||
* Get the bitstreams endpoint for a bundle
|
* Get the bitstreams endpoint for a bundle
|
||||||
* @param bundleId
|
* @param bundleId
|
||||||
|
* @param searchOptions
|
||||||
*/
|
*/
|
||||||
getBitstreamsEndpoint(bundleId: string): Observable<string> {
|
getBitstreamsEndpoint(bundleId: string, searchOptions?: PaginatedSearchOptions): Observable<string> {
|
||||||
return this.getBrowseEndpoint().pipe(
|
return this.getBrowseEndpoint().pipe(
|
||||||
switchMap((href: string) => this.halService.getEndpoint(this.bitstreamsEndpoint, `${href}/${bundleId}`))
|
switchMap((href: string) => this.halService.getEndpoint(this.bitstreamsEndpoint, `${href}/${bundleId}`)),
|
||||||
|
map((href) => searchOptions ? searchOptions.toRestUrl(href) : href)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -102,9 +104,8 @@ export class BundleDataService extends DataService<Bundle> {
|
|||||||
* @param linksToFollow The {@link FollowLinkConfig}s for the request
|
* @param linksToFollow The {@link FollowLinkConfig}s for the request
|
||||||
*/
|
*/
|
||||||
getBitstreams(bundleId: string, searchOptions?: PaginatedSearchOptions, ...linksToFollow: Array<FollowLinkConfig<Bitstream>>): Observable<RemoteData<PaginatedList<Bitstream>>> {
|
getBitstreams(bundleId: string, searchOptions?: PaginatedSearchOptions, ...linksToFollow: Array<FollowLinkConfig<Bitstream>>): Observable<RemoteData<PaginatedList<Bitstream>>> {
|
||||||
const hrefObs = this.getBitstreamsEndpoint(bundleId).pipe(
|
const hrefObs = this.getBitstreamsEndpoint(bundleId, searchOptions);
|
||||||
map((href) => searchOptions ? searchOptions.toRestUrl(href) : href)
|
|
||||||
);
|
|
||||||
hrefObs.pipe(
|
hrefObs.pipe(
|
||||||
take(1)
|
take(1)
|
||||||
).subscribe((href) => {
|
).subscribe((href) => {
|
||||||
|
@@ -13,13 +13,19 @@ import { RequestEntry } from './request.reducer';
|
|||||||
import { ErrorResponse, RestResponse } from '../cache/response.models';
|
import { ErrorResponse, RestResponse } from '../cache/response.models';
|
||||||
import { ObjectCacheService } from '../cache/object-cache.service';
|
import { ObjectCacheService } from '../cache/object-cache.service';
|
||||||
import { RemoteDataBuildService } from '../cache/builders/remote-data-build.service';
|
import { RemoteDataBuildService } from '../cache/builders/remote-data-build.service';
|
||||||
|
import { Collection } from '../shared/collection.model';
|
||||||
|
import { PageInfo } from '../shared/page-info.model';
|
||||||
|
import { PaginatedList } from './paginated-list';
|
||||||
|
import { createSuccessfulRemoteDataObject } from 'src/app/shared/remote-data.utils';
|
||||||
|
import { hot, getTestScheduler, cold } from 'jasmine-marbles';
|
||||||
|
import { TestScheduler } from 'rxjs/testing';
|
||||||
|
|
||||||
const url = 'fake-url';
|
const url = 'fake-url';
|
||||||
const collectionId = 'fake-collection-id';
|
const collectionId = 'fake-collection-id';
|
||||||
|
|
||||||
describe('CollectionDataService', () => {
|
describe('CollectionDataService', () => {
|
||||||
let service: CollectionDataService;
|
let service: CollectionDataService;
|
||||||
|
let scheduler: TestScheduler;
|
||||||
let requestService: RequestService;
|
let requestService: RequestService;
|
||||||
let translate: TranslateService;
|
let translate: TranslateService;
|
||||||
let notificationsService: any;
|
let notificationsService: any;
|
||||||
@@ -27,6 +33,44 @@ describe('CollectionDataService', () => {
|
|||||||
let objectCache: ObjectCacheService;
|
let objectCache: ObjectCacheService;
|
||||||
let halService: any;
|
let halService: any;
|
||||||
|
|
||||||
|
const mockCollection1: Collection = Object.assign(new Collection(), {
|
||||||
|
id: 'test-collection-1-1',
|
||||||
|
name: 'test-collection-1',
|
||||||
|
_links: {
|
||||||
|
self: {
|
||||||
|
href: 'https://rest.api/collections/test-collection-1-1'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const mockCollection2: Collection = Object.assign(new Collection(), {
|
||||||
|
id: 'test-collection-2-2',
|
||||||
|
name: 'test-collection-2',
|
||||||
|
_links: {
|
||||||
|
self: {
|
||||||
|
href: 'https://rest.api/collections/test-collection-2-2'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const mockCollection3: Collection = Object.assign(new Collection(), {
|
||||||
|
id: 'test-collection-3-3',
|
||||||
|
name: 'test-collection-3',
|
||||||
|
_links: {
|
||||||
|
self: {
|
||||||
|
href: 'https://rest.api/collections/test-collection-3-3'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const queryString = 'test-string';
|
||||||
|
const communityId = '8b3c613a-5a4b-438b-9686-be1d5b4a1c5a';
|
||||||
|
|
||||||
|
const pageInfo = new PageInfo();
|
||||||
|
const array = [mockCollection1, mockCollection2, mockCollection3];
|
||||||
|
const paginatedList = new PaginatedList(pageInfo, array);
|
||||||
|
const paginatedListRD = createSuccessfulRemoteDataObject(paginatedList);
|
||||||
|
|
||||||
describe('when the requests are successful', () => {
|
describe('when the requests are successful', () => {
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
createService();
|
createService();
|
||||||
@@ -74,6 +118,43 @@ describe('CollectionDataService', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('when calling getAuthorizedCollection', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
scheduler = getTestScheduler();
|
||||||
|
spyOn(service, 'getAuthorizedCollection').and.callThrough();
|
||||||
|
spyOn(service, 'getAuthorizedCollectionByCommunity').and.callThrough();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should proxy the call to getAuthorizedCollection', () => {
|
||||||
|
scheduler.schedule(() => service.getAuthorizedCollection(queryString));
|
||||||
|
scheduler.flush();
|
||||||
|
|
||||||
|
expect(service.getAuthorizedCollection).toHaveBeenCalledWith(queryString);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return a RemoteData<PaginatedList<Colletion>> for the getAuthorizedCollection', () => {
|
||||||
|
const result = service.getAuthorizedCollection(queryString)
|
||||||
|
const expected = cold('a|', {
|
||||||
|
a: paginatedListRD
|
||||||
|
});
|
||||||
|
expect(result).toBeObservable(expected);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should proxy the call to getAuthorizedCollectionByCommunity', () => {
|
||||||
|
scheduler.schedule(() => service.getAuthorizedCollectionByCommunity(communityId, queryString));
|
||||||
|
scheduler.flush();
|
||||||
|
|
||||||
|
expect(service.getAuthorizedCollectionByCommunity).toHaveBeenCalledWith(communityId, queryString);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return a RemoteData<PaginatedList<Colletion>> for the getAuthorizedCollectionByCommunity', () => {
|
||||||
|
const result = service.getAuthorizedCollectionByCommunity(communityId, queryString)
|
||||||
|
const expected = cold('a|', {
|
||||||
|
a: paginatedListRD
|
||||||
|
});
|
||||||
|
expect(result).toBeObservable(expected);
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('when the requests are unsuccessful', () => {
|
describe('when the requests are unsuccessful', () => {
|
||||||
@@ -117,7 +198,9 @@ describe('CollectionDataService', () => {
|
|||||||
function createService(requestEntry$?) {
|
function createService(requestEntry$?) {
|
||||||
requestService = getMockRequestService(requestEntry$);
|
requestService = getMockRequestService(requestEntry$);
|
||||||
rdbService = jasmine.createSpyObj('rdbService', {
|
rdbService = jasmine.createSpyObj('rdbService', {
|
||||||
buildList: jasmine.createSpy('buildList')
|
buildList: hot('a|', {
|
||||||
|
a: paginatedListRD
|
||||||
|
})
|
||||||
});
|
});
|
||||||
objectCache = jasmine.createSpyObj('objectCache', {
|
objectCache = jasmine.createSpyObj('objectCache', {
|
||||||
remove: jasmine.createSpy('remove')
|
remove: jasmine.createSpy('remove')
|
||||||
|
@@ -72,14 +72,18 @@ export class CollectionDataService extends ComColDataService<Collection> {
|
|||||||
/**
|
/**
|
||||||
* Get all collections the user is authorized to submit to
|
* Get all collections the user is authorized to submit to
|
||||||
*
|
*
|
||||||
|
* @param query limit the returned collection to those with metadata values matching the query terms.
|
||||||
* @param options The [[FindListOptions]] object
|
* @param options The [[FindListOptions]] object
|
||||||
* @return Observable<RemoteData<PaginatedList<Collection>>>
|
* @return Observable<RemoteData<PaginatedList<Collection>>>
|
||||||
* collection list
|
* collection list
|
||||||
*/
|
*/
|
||||||
getAuthorizedCollection(options: FindListOptions = {}): Observable<RemoteData<PaginatedList<Collection>>> {
|
getAuthorizedCollection(query: string, options: FindListOptions = {}, ...linksToFollow: Array<FollowLinkConfig<Collection>>): Observable<RemoteData<PaginatedList<Collection>>> {
|
||||||
const searchHref = 'findAuthorized';
|
const searchHref = 'findSubmitAuthorized';
|
||||||
|
options = Object.assign({}, options, {
|
||||||
|
searchParams: [new RequestParam('query', query)]
|
||||||
|
});
|
||||||
|
|
||||||
return this.searchBy(searchHref, options).pipe(
|
return this.searchBy(searchHref, options, ...linksToFollow).pipe(
|
||||||
filter((collections: RemoteData<PaginatedList<Collection>>) => !collections.isResponsePending));
|
filter((collections: RemoteData<PaginatedList<Collection>>) => !collections.isResponsePending));
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -87,14 +91,18 @@ export class CollectionDataService extends ComColDataService<Collection> {
|
|||||||
* Get all collections the user is authorized to submit to, by community
|
* Get all collections the user is authorized to submit to, by community
|
||||||
*
|
*
|
||||||
* @param communityId The community id
|
* @param communityId The community id
|
||||||
|
* @param query limit the returned collection to those with metadata values matching the query terms.
|
||||||
* @param options The [[FindListOptions]] object
|
* @param options The [[FindListOptions]] object
|
||||||
* @return Observable<RemoteData<PaginatedList<Collection>>>
|
* @return Observable<RemoteData<PaginatedList<Collection>>>
|
||||||
* collection list
|
* collection list
|
||||||
*/
|
*/
|
||||||
getAuthorizedCollectionByCommunity(communityId: string, options: FindListOptions = {}): Observable<RemoteData<PaginatedList<Collection>>> {
|
getAuthorizedCollectionByCommunity(communityId: string, query: string, options: FindListOptions = {}): Observable<RemoteData<PaginatedList<Collection>>> {
|
||||||
const searchHref = 'findAuthorizedByCommunity';
|
const searchHref = 'findSubmitAuthorizedByCommunity';
|
||||||
options = Object.assign({}, options, {
|
options = Object.assign({}, options, {
|
||||||
searchParams: [new RequestParam('uuid', communityId)]
|
searchParams: [
|
||||||
|
new RequestParam('uuid', communityId),
|
||||||
|
new RequestParam('query', query)
|
||||||
|
]
|
||||||
});
|
});
|
||||||
|
|
||||||
return this.searchBy(searchHref, options).pipe(
|
return this.searchBy(searchHref, options).pipe(
|
||||||
@@ -108,7 +116,7 @@ export class CollectionDataService extends ComColDataService<Collection> {
|
|||||||
* true if the user has at least one collection to submit to
|
* true if the user has at least one collection to submit to
|
||||||
*/
|
*/
|
||||||
hasAuthorizedCollection(): Observable<boolean> {
|
hasAuthorizedCollection(): Observable<boolean> {
|
||||||
const searchHref = 'findAuthorized';
|
const searchHref = 'findSubmitAuthorized';
|
||||||
const options = new FindListOptions();
|
const options = new FindListOptions();
|
||||||
options.elementsPerPage = 1;
|
options.elementsPerPage = 1;
|
||||||
|
|
||||||
|
@@ -45,11 +45,12 @@ import {
|
|||||||
FindListOptions,
|
FindListOptions,
|
||||||
FindListRequest,
|
FindListRequest,
|
||||||
GetRequest,
|
GetRequest,
|
||||||
PatchRequest
|
PatchRequest, PutRequest
|
||||||
} from './request.models';
|
} from './request.models';
|
||||||
import { RequestEntry } from './request.reducer';
|
import { RequestEntry } from './request.reducer';
|
||||||
import { RequestService } from './request.service';
|
import { RequestService } from './request.service';
|
||||||
import { RestRequestMethod } from './rest-request-method';
|
import { RestRequestMethod } from './rest-request-method';
|
||||||
|
import { GenericConstructor } from '../shared/generic-constructor';
|
||||||
|
|
||||||
export abstract class DataService<T extends CacheableObject> {
|
export abstract class DataService<T extends CacheableObject> {
|
||||||
protected abstract requestService: RequestService;
|
protected abstract requestService: RequestService;
|
||||||
@@ -343,7 +344,9 @@ export abstract class DataService<T extends CacheableObject> {
|
|||||||
tap((href: string) => {
|
tap((href: string) => {
|
||||||
this.requestService.removeByHrefSubstring(href);
|
this.requestService.removeByHrefSubstring(href);
|
||||||
const request = new FindListRequest(this.requestService.generateRequestId(), href, options);
|
const request = new FindListRequest(this.requestService.generateRequestId(), href, options);
|
||||||
request.responseMsToLive = 10 * 1000;
|
if (hasValue(this.responseMsToLive)) {
|
||||||
|
request.responseMsToLive = this.responseMsToLive;
|
||||||
|
}
|
||||||
|
|
||||||
this.requestService.configure(request);
|
this.requestService.configure(request);
|
||||||
}
|
}
|
||||||
@@ -381,6 +384,28 @@ export abstract class DataService<T extends CacheableObject> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Send a PUT request for the specified object
|
||||||
|
*
|
||||||
|
* @param object The object to send a put request for.
|
||||||
|
*/
|
||||||
|
put(object: T): Observable<RemoteData<T>> {
|
||||||
|
const requestId = this.requestService.generateRequestId();
|
||||||
|
const serializedObject = new DSpaceSerializer(object.constructor as GenericConstructor<{}>).serialize(object);
|
||||||
|
const request = new PutRequest(requestId, object._links.self.href, serializedObject);
|
||||||
|
|
||||||
|
if (hasValue(this.responseMsToLive)) {
|
||||||
|
request.responseMsToLive = this.responseMsToLive;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.requestService.configure(request);
|
||||||
|
|
||||||
|
return this.requestService.getByUUID(requestId).pipe(
|
||||||
|
find((re: RequestEntry) => hasValue(re) && re.completed),
|
||||||
|
switchMap(() => this.findByHref(object._links.self.href))
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Add a new patch to the object cache
|
* Add a new patch to the object cache
|
||||||
* The patch is derived from the differences between the given object and its version in the object cache
|
* The patch is derived from the differences between the given object and its version in the object cache
|
||||||
|
114
src/app/core/data/metadata-field-data.service.spec.ts
Normal file
114
src/app/core/data/metadata-field-data.service.spec.ts
Normal file
@@ -0,0 +1,114 @@
|
|||||||
|
import { RequestService } from './request.service';
|
||||||
|
import { HALEndpointService } from '../shared/hal-endpoint.service';
|
||||||
|
import { NotificationsService } from '../../shared/notifications/notifications.service';
|
||||||
|
import { of as observableOf } from 'rxjs/internal/observable/of';
|
||||||
|
import { RestResponse } from '../cache/response.models';
|
||||||
|
import { HALEndpointServiceStub } from '../../shared/testing/hal-endpoint-service.stub';
|
||||||
|
import { CreateRequest, FindListOptions, PutRequest } from './request.models';
|
||||||
|
import { MetadataFieldDataService } from './metadata-field-data.service';
|
||||||
|
import { MetadataField } from '../metadata/metadata-field.model';
|
||||||
|
import { MetadataSchema } from '../metadata/metadata-schema.model';
|
||||||
|
import { RemoteDataBuildService } from '../cache/builders/remote-data-build.service';
|
||||||
|
import { createSuccessfulRemoteDataObject$ } from '../../shared/remote-data.utils';
|
||||||
|
import { RequestParam } from '../cache/models/request-param.model';
|
||||||
|
|
||||||
|
describe('MetadataFieldDataService', () => {
|
||||||
|
let metadataFieldService: MetadataFieldDataService;
|
||||||
|
let requestService: RequestService;
|
||||||
|
let halService: HALEndpointService;
|
||||||
|
let notificationsService: NotificationsService;
|
||||||
|
let schema: MetadataSchema;
|
||||||
|
let rdbService: RemoteDataBuildService;
|
||||||
|
|
||||||
|
const endpoint = 'api/metadatafield/endpoint';
|
||||||
|
|
||||||
|
function init() {
|
||||||
|
schema = Object.assign(new MetadataSchema(), {
|
||||||
|
prefix: 'dc',
|
||||||
|
namespace: 'namespace',
|
||||||
|
_links: {
|
||||||
|
self: { href: 'selflink' }
|
||||||
|
}
|
||||||
|
});
|
||||||
|
requestService = jasmine.createSpyObj('requestService', {
|
||||||
|
generateRequestId: '34cfed7c-f597-49ef-9cbe-ea351f0023c2',
|
||||||
|
configure: {},
|
||||||
|
getByUUID: observableOf({ response: new RestResponse(true, 200, 'OK') }),
|
||||||
|
removeByHrefSubstring: {}
|
||||||
|
});
|
||||||
|
halService = Object.assign(new HALEndpointServiceStub(endpoint));
|
||||||
|
notificationsService = jasmine.createSpyObj('notificationsService', {
|
||||||
|
error: {}
|
||||||
|
});
|
||||||
|
rdbService = jasmine.createSpyObj('rdbService', {
|
||||||
|
buildSingle: createSuccessfulRemoteDataObject$(undefined)
|
||||||
|
});
|
||||||
|
metadataFieldService = new MetadataFieldDataService(requestService, rdbService, undefined, halService, undefined, undefined, undefined, notificationsService);
|
||||||
|
}
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
init();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('findBySchema', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
spyOn(metadataFieldService, 'searchBy');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should call searchBy with the correct arguments', () => {
|
||||||
|
metadataFieldService.findBySchema(schema);
|
||||||
|
const expectedOptions = Object.assign(new FindListOptions(), {
|
||||||
|
searchParams: [new RequestParam('schema', schema.prefix)]
|
||||||
|
});
|
||||||
|
expect(metadataFieldService.searchBy).toHaveBeenCalledWith('bySchema', expectedOptions);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('createOrUpdateMetadataField', () => {
|
||||||
|
let field: MetadataField;
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
field = Object.assign(new MetadataField(), {
|
||||||
|
element: 'identifier',
|
||||||
|
qualifier: undefined,
|
||||||
|
schema: schema,
|
||||||
|
_links: {
|
||||||
|
self: { href: 'selflink' }
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('called with a new metadata field', () => {
|
||||||
|
it('should send a CreateRequest', (done) => {
|
||||||
|
metadataFieldService.createOrUpdateMetadataField(field).subscribe(() => {
|
||||||
|
expect(requestService.configure).toHaveBeenCalledWith(jasmine.any(CreateRequest));
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('called with an existing metadata field', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
field = Object.assign(field, {
|
||||||
|
id: 'id-of-existing-field'
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should send a PutRequest', (done) => {
|
||||||
|
metadataFieldService.createOrUpdateMetadataField(field).subscribe(() => {
|
||||||
|
expect(requestService.configure).toHaveBeenCalledWith(jasmine.any(PutRequest));
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('clearRequests', () => {
|
||||||
|
it('should remove requests on the data service\'s endpoint', (done) => {
|
||||||
|
metadataFieldService.clearRequests().subscribe(() => {
|
||||||
|
expect(requestService.removeByHrefSubstring).toHaveBeenCalledWith(`${endpoint}/${(metadataFieldService as any).linkPath}`);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
89
src/app/core/data/metadata-field-data.service.ts
Normal file
89
src/app/core/data/metadata-field-data.service.ts
Normal file
@@ -0,0 +1,89 @@
|
|||||||
|
import { Injectable } from '@angular/core';
|
||||||
|
import { dataService } from '../cache/builders/build-decorators';
|
||||||
|
import { DataService } from './data.service';
|
||||||
|
import { RequestService } from './request.service';
|
||||||
|
import { RemoteDataBuildService } from '../cache/builders/remote-data-build.service';
|
||||||
|
import { Store } from '@ngrx/store';
|
||||||
|
import { CoreState } from '../core.reducers';
|
||||||
|
import { HALEndpointService } from '../shared/hal-endpoint.service';
|
||||||
|
import { ObjectCacheService } from '../cache/object-cache.service';
|
||||||
|
import { DefaultChangeAnalyzer } from './default-change-analyzer.service';
|
||||||
|
import { HttpClient } from '@angular/common/http';
|
||||||
|
import { NotificationsService } from '../../shared/notifications/notifications.service';
|
||||||
|
import { METADATA_FIELD } from '../metadata/metadata-field.resource-type';
|
||||||
|
import { MetadataField } from '../metadata/metadata-field.model';
|
||||||
|
import { MetadataSchema } from '../metadata/metadata-schema.model';
|
||||||
|
import { FindListOptions, FindListRequest } from './request.models';
|
||||||
|
import { FollowLinkConfig } from '../../shared/utils/follow-link-config.model';
|
||||||
|
import { Observable } from 'rxjs/internal/Observable';
|
||||||
|
import { hasValue } from '../../shared/empty.util';
|
||||||
|
import { find, skipWhile, switchMap, tap } from 'rxjs/operators';
|
||||||
|
import { RemoteData } from './remote-data';
|
||||||
|
import { RequestParam } from '../cache/models/request-param.model';
|
||||||
|
import { PaginatedList } from './paginated-list';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A service responsible for fetching/sending data from/to the REST API on the metadatafields endpoint
|
||||||
|
*/
|
||||||
|
@Injectable()
|
||||||
|
@dataService(METADATA_FIELD)
|
||||||
|
export class MetadataFieldDataService extends DataService<MetadataField> {
|
||||||
|
protected linkPath = 'metadatafields';
|
||||||
|
protected searchBySchemaLinkPath = 'bySchema';
|
||||||
|
|
||||||
|
constructor(
|
||||||
|
protected requestService: RequestService,
|
||||||
|
protected rdbService: RemoteDataBuildService,
|
||||||
|
protected store: Store<CoreState>,
|
||||||
|
protected halService: HALEndpointService,
|
||||||
|
protected objectCache: ObjectCacheService,
|
||||||
|
protected comparator: DefaultChangeAnalyzer<MetadataField>,
|
||||||
|
protected http: HttpClient,
|
||||||
|
protected notificationsService: NotificationsService) {
|
||||||
|
super();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Find metadata fields belonging to a metadata schema
|
||||||
|
* @param schema The metadata schema to list fields for
|
||||||
|
* @param options The options info used to retrieve the fields
|
||||||
|
* @param linksToFollow List of {@link FollowLinkConfig} that indicate which {@link HALLink}s should be automatically resolved
|
||||||
|
*/
|
||||||
|
findBySchema(schema: MetadataSchema, options: FindListOptions = {}, ...linksToFollow: Array<FollowLinkConfig<MetadataField>>) {
|
||||||
|
const optionsWithSchema = Object.assign(new FindListOptions(), options, {
|
||||||
|
searchParams: [new RequestParam('schema', schema.prefix)]
|
||||||
|
});
|
||||||
|
return this.searchBy(this.searchBySchemaLinkPath, optionsWithSchema, ...linksToFollow);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create or Update a MetadataField
|
||||||
|
* If the MetadataField contains an id, it is assumed the field already exists and is updated instead
|
||||||
|
* Since creating or updating is nearly identical, the only real difference is the request (and slight difference in endpoint):
|
||||||
|
* - On creation, a CreateRequest is used
|
||||||
|
* - On update, a PutRequest is used
|
||||||
|
* @param field The MetadataField to create or update
|
||||||
|
*/
|
||||||
|
createOrUpdateMetadataField(field: MetadataField): Observable<RemoteData<MetadataField>> {
|
||||||
|
const isUpdate = hasValue(field.id);
|
||||||
|
|
||||||
|
if (isUpdate) {
|
||||||
|
return this.put(field);
|
||||||
|
} else {
|
||||||
|
return this.create(field, new RequestParam('schemaId', field.schema.id));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Clear all metadata field requests
|
||||||
|
* Used for refreshing lists after adding/updating/removing a metadata field from a metadata schema
|
||||||
|
*/
|
||||||
|
clearRequests(): Observable<string> {
|
||||||
|
return this.getBrowseEndpoint().pipe(
|
||||||
|
tap((href: string) => {
|
||||||
|
this.requestService.removeByHrefSubstring(href);
|
||||||
|
})
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
89
src/app/core/data/metadata-schema-data.service.spec.ts
Normal file
89
src/app/core/data/metadata-schema-data.service.spec.ts
Normal file
@@ -0,0 +1,89 @@
|
|||||||
|
import { RequestService } from './request.service';
|
||||||
|
import { HALEndpointService } from '../shared/hal-endpoint.service';
|
||||||
|
import { NotificationsService } from '../../shared/notifications/notifications.service';
|
||||||
|
import { MetadataSchemaDataService } from './metadata-schema-data.service';
|
||||||
|
import { of as observableOf } from 'rxjs';
|
||||||
|
import { RestResponse } from '../cache/response.models';
|
||||||
|
import { HALEndpointServiceStub } from '../../shared/testing/hal-endpoint-service.stub';
|
||||||
|
import { MetadataSchema } from '../metadata/metadata-schema.model';
|
||||||
|
import { CreateRequest, PutRequest } from './request.models';
|
||||||
|
import { RemoteDataBuildService } from '../cache/builders/remote-data-build.service';
|
||||||
|
import { createSuccessfulRemoteDataObject$ } from '../../shared/remote-data.utils';
|
||||||
|
|
||||||
|
describe('MetadataSchemaDataService', () => {
|
||||||
|
let metadataSchemaService: MetadataSchemaDataService;
|
||||||
|
let requestService: RequestService;
|
||||||
|
let halService: HALEndpointService;
|
||||||
|
let notificationsService: NotificationsService;
|
||||||
|
let rdbService: RemoteDataBuildService;
|
||||||
|
|
||||||
|
const endpoint = 'api/metadataschema/endpoint';
|
||||||
|
|
||||||
|
function init() {
|
||||||
|
requestService = jasmine.createSpyObj('requestService', {
|
||||||
|
generateRequestId: '34cfed7c-f597-49ef-9cbe-ea351f0023c2',
|
||||||
|
configure: {},
|
||||||
|
getByUUID: observableOf({ response: new RestResponse(true, 200, 'OK') }),
|
||||||
|
removeByHrefSubstring: {}
|
||||||
|
});
|
||||||
|
halService = Object.assign(new HALEndpointServiceStub(endpoint));
|
||||||
|
notificationsService = jasmine.createSpyObj('notificationsService', {
|
||||||
|
error: {}
|
||||||
|
});
|
||||||
|
rdbService = jasmine.createSpyObj('rdbService', {
|
||||||
|
buildSingle: createSuccessfulRemoteDataObject$(undefined)
|
||||||
|
});
|
||||||
|
metadataSchemaService = new MetadataSchemaDataService(requestService, rdbService, undefined, halService, undefined, undefined, undefined, notificationsService);
|
||||||
|
}
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
init();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('createOrUpdateMetadataSchema', () => {
|
||||||
|
let schema: MetadataSchema;
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
schema = Object.assign(new MetadataSchema(), {
|
||||||
|
prefix: 'dc',
|
||||||
|
namespace: 'namespace',
|
||||||
|
_links: {
|
||||||
|
self: { href: 'selflink' }
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('called with a new metadata schema', () => {
|
||||||
|
it('should send a CreateRequest', (done) => {
|
||||||
|
metadataSchemaService.createOrUpdateMetadataSchema(schema).subscribe(() => {
|
||||||
|
expect(requestService.configure).toHaveBeenCalledWith(jasmine.any(CreateRequest));
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('called with an existing metadata schema', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
schema = Object.assign(schema, {
|
||||||
|
id: 'id-of-existing-schema'
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should send a PutRequest', (done) => {
|
||||||
|
metadataSchemaService.createOrUpdateMetadataSchema(schema).subscribe(() => {
|
||||||
|
expect(requestService.configure).toHaveBeenCalledWith(jasmine.any(PutRequest));
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('clearRequests', () => {
|
||||||
|
it('should remove requests on the data service\'s endpoint', (done) => {
|
||||||
|
metadataSchemaService.clearRequests().subscribe(() => {
|
||||||
|
expect(requestService.removeByHrefSubstring).toHaveBeenCalledWith(`${endpoint}/${(metadataSchemaService as any).linkPath}`);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
@@ -9,37 +9,21 @@ import { CoreState } from '../core.reducers';
|
|||||||
import { MetadataSchema } from '../metadata/metadata-schema.model';
|
import { MetadataSchema } from '../metadata/metadata-schema.model';
|
||||||
import { METADATA_SCHEMA } from '../metadata/metadata-schema.resource-type';
|
import { METADATA_SCHEMA } from '../metadata/metadata-schema.resource-type';
|
||||||
import { HALEndpointService } from '../shared/hal-endpoint.service';
|
import { HALEndpointService } from '../shared/hal-endpoint.service';
|
||||||
import { ChangeAnalyzer } from './change-analyzer';
|
|
||||||
|
|
||||||
import { DataService } from './data.service';
|
import { DataService } from './data.service';
|
||||||
import { DefaultChangeAnalyzer } from './default-change-analyzer.service';
|
import { DefaultChangeAnalyzer } from './default-change-analyzer.service';
|
||||||
import { RequestService } from './request.service';
|
import { RequestService } from './request.service';
|
||||||
|
import { Observable } from 'rxjs/internal/Observable';
|
||||||
/* tslint:disable:max-classes-per-file */
|
import { hasValue } from '../../shared/empty.util';
|
||||||
class DataServiceImpl extends DataService<MetadataSchema> {
|
import { tap } from 'rxjs/operators';
|
||||||
protected linkPath = 'metadataschemas';
|
import { RemoteData } from './remote-data';
|
||||||
|
|
||||||
constructor(
|
|
||||||
protected requestService: RequestService,
|
|
||||||
protected rdbService: RemoteDataBuildService,
|
|
||||||
protected store: Store<CoreState>,
|
|
||||||
protected objectCache: ObjectCacheService,
|
|
||||||
protected halService: HALEndpointService,
|
|
||||||
protected notificationsService: NotificationsService,
|
|
||||||
protected http: HttpClient,
|
|
||||||
protected comparator: ChangeAnalyzer<MetadataSchema>) {
|
|
||||||
super();
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A service responsible for fetching/sending data from/to the REST API on the metadataschemas endpoint
|
* A service responsible for fetching/sending data from/to the REST API on the metadataschemas endpoint
|
||||||
*/
|
*/
|
||||||
@Injectable()
|
@Injectable()
|
||||||
@dataService(METADATA_SCHEMA)
|
@dataService(METADATA_SCHEMA)
|
||||||
export class MetadataSchemaDataService {
|
export class MetadataSchemaDataService extends DataService<MetadataSchema> {
|
||||||
private dataService: DataServiceImpl;
|
protected linkPath = 'metadataschemas';
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
protected requestService: RequestService,
|
protected requestService: RequestService,
|
||||||
@@ -50,6 +34,35 @@ export class MetadataSchemaDataService {
|
|||||||
protected comparator: DefaultChangeAnalyzer<MetadataSchema>,
|
protected comparator: DefaultChangeAnalyzer<MetadataSchema>,
|
||||||
protected http: HttpClient,
|
protected http: HttpClient,
|
||||||
protected notificationsService: NotificationsService) {
|
protected notificationsService: NotificationsService) {
|
||||||
this.dataService = new DataServiceImpl(requestService, rdbService, null, objectCache, halService, notificationsService, http, comparator);
|
super();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create or Update a MetadataSchema
|
||||||
|
* If the MetadataSchema contains an id, it is assumed the schema already exists and is updated instead
|
||||||
|
* Since creating or updating is nearly identical, the only real difference is the request (and slight difference in endpoint):
|
||||||
|
* - On creation, a CreateRequest is used
|
||||||
|
* - On update, a PutRequest is used
|
||||||
|
* @param schema The MetadataSchema to create or update
|
||||||
|
*/
|
||||||
|
createOrUpdateMetadataSchema(schema: MetadataSchema): Observable<RemoteData<MetadataSchema>> {
|
||||||
|
const isUpdate = hasValue(schema.id);
|
||||||
|
|
||||||
|
if (isUpdate) {
|
||||||
|
return this.put(schema);
|
||||||
|
} else {
|
||||||
|
return this.create(schema);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Clear all metadata schema requests
|
||||||
|
* Used for refreshing lists after adding/updating/removing a metadata schema in the registry
|
||||||
|
*/
|
||||||
|
clearRequests(): Observable<string> {
|
||||||
|
return this.getBrowseEndpoint().pipe(
|
||||||
|
tap((href: string) => this.requestService.removeByHrefSubstring(href))
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@@ -1,22 +0,0 @@
|
|||||||
import { Injectable } from '@angular/core';
|
|
||||||
import { MetadatafieldSuccessResponse, RestResponse } from '../cache/response.models';
|
|
||||||
import { DSpaceRESTV2Response } from '../dspace-rest-v2/dspace-rest-v2-response.model';
|
|
||||||
import { DSpaceSerializer } from '../dspace-rest-v2/dspace.serializer';
|
|
||||||
import { MetadataField } from '../metadata/metadata-field.model';
|
|
||||||
import { ResponseParsingService } from './parsing.service';
|
|
||||||
import { RestRequest } from './request.models';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A service responsible for parsing DSpaceRESTV2Response data related to a single MetadataField to a valid RestResponse
|
|
||||||
*/
|
|
||||||
@Injectable()
|
|
||||||
export class MetadatafieldParsingService implements ResponseParsingService {
|
|
||||||
|
|
||||||
parse(request: RestRequest, data: DSpaceRESTV2Response): RestResponse {
|
|
||||||
const payload = data.payload;
|
|
||||||
|
|
||||||
const deserialized = new DSpaceSerializer(MetadataField).deserialize(payload);
|
|
||||||
return new MetadatafieldSuccessResponse(deserialized, data.statusCode, data.statusText);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
@@ -1,19 +0,0 @@
|
|||||||
import { Injectable } from '@angular/core';
|
|
||||||
import { MetadataschemaSuccessResponse, RestResponse } from '../cache/response.models';
|
|
||||||
import { DSpaceRESTV2Response } from '../dspace-rest-v2/dspace-rest-v2-response.model';
|
|
||||||
import { DSpaceSerializer } from '../dspace-rest-v2/dspace.serializer';
|
|
||||||
import { MetadataSchema } from '../metadata/metadata-schema.model';
|
|
||||||
import { ResponseParsingService } from './parsing.service';
|
|
||||||
import { RestRequest } from './request.models';
|
|
||||||
|
|
||||||
@Injectable()
|
|
||||||
export class MetadataschemaParsingService implements ResponseParsingService {
|
|
||||||
|
|
||||||
parse(request: RestRequest, data: DSpaceRESTV2Response): RestResponse {
|
|
||||||
const payload = data.payload;
|
|
||||||
|
|
||||||
const deserialized = new DSpaceSerializer(MetadataSchema).deserialize(payload);
|
|
||||||
return new MetadataschemaSuccessResponse(deserialized, data.statusCode, data.statusText);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
@@ -8,7 +8,6 @@ import {INotification} from '../../../shared/notifications/models/notification.m
|
|||||||
*/
|
*/
|
||||||
export const ObjectUpdatesActionTypes = {
|
export const ObjectUpdatesActionTypes = {
|
||||||
INITIALIZE_FIELDS: type('dspace/core/cache/object-updates/INITIALIZE_FIELDS'),
|
INITIALIZE_FIELDS: type('dspace/core/cache/object-updates/INITIALIZE_FIELDS'),
|
||||||
ADD_PAGE_TO_CUSTOM_ORDER: type('dspace/core/cache/object-updates/ADD_PAGE_TO_CUSTOM_ORDER'),
|
|
||||||
SET_EDITABLE_FIELD: type('dspace/core/cache/object-updates/SET_EDITABLE_FIELD'),
|
SET_EDITABLE_FIELD: type('dspace/core/cache/object-updates/SET_EDITABLE_FIELD'),
|
||||||
SET_VALID_FIELD: type('dspace/core/cache/object-updates/SET_VALID_FIELD'),
|
SET_VALID_FIELD: type('dspace/core/cache/object-updates/SET_VALID_FIELD'),
|
||||||
ADD_FIELD: type('dspace/core/cache/object-updates/ADD_FIELD'),
|
ADD_FIELD: type('dspace/core/cache/object-updates/ADD_FIELD'),
|
||||||
@@ -17,8 +16,7 @@ export const ObjectUpdatesActionTypes = {
|
|||||||
REINSTATE: type('dspace/core/cache/object-updates/REINSTATE'),
|
REINSTATE: type('dspace/core/cache/object-updates/REINSTATE'),
|
||||||
REMOVE: type('dspace/core/cache/object-updates/REMOVE'),
|
REMOVE: type('dspace/core/cache/object-updates/REMOVE'),
|
||||||
REMOVE_ALL: type('dspace/core/cache/object-updates/REMOVE_ALL'),
|
REMOVE_ALL: type('dspace/core/cache/object-updates/REMOVE_ALL'),
|
||||||
REMOVE_FIELD: type('dspace/core/cache/object-updates/REMOVE_FIELD'),
|
REMOVE_FIELD: type('dspace/core/cache/object-updates/REMOVE_FIELD')
|
||||||
MOVE: type('dspace/core/cache/object-updates/MOVE'),
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/* tslint:disable:max-classes-per-file */
|
/* tslint:disable:max-classes-per-file */
|
||||||
@@ -29,8 +27,7 @@ export const ObjectUpdatesActionTypes = {
|
|||||||
export enum FieldChangeType {
|
export enum FieldChangeType {
|
||||||
UPDATE = 0,
|
UPDATE = 0,
|
||||||
ADD = 1,
|
ADD = 1,
|
||||||
REMOVE = 2,
|
REMOVE = 2
|
||||||
MOVE = 3
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -41,10 +38,7 @@ export class InitializeFieldsAction implements Action {
|
|||||||
payload: {
|
payload: {
|
||||||
url: string,
|
url: string,
|
||||||
fields: Identifiable[],
|
fields: Identifiable[],
|
||||||
lastModified: Date,
|
lastModified: Date
|
||||||
order: string[],
|
|
||||||
pageSize: number,
|
|
||||||
page: number
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -61,42 +55,9 @@ export class InitializeFieldsAction implements Action {
|
|||||||
constructor(
|
constructor(
|
||||||
url: string,
|
url: string,
|
||||||
fields: Identifiable[],
|
fields: Identifiable[],
|
||||||
lastModified: Date,
|
lastModified: Date
|
||||||
order: string[] = [],
|
|
||||||
pageSize: number = 9999,
|
|
||||||
page: number = 0
|
|
||||||
) {
|
) {
|
||||||
this.payload = { url, fields, lastModified, order, pageSize, page };
|
this.payload = { url, fields, lastModified };
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* An ngrx action to initialize a new page's fields in the ObjectUpdates state
|
|
||||||
*/
|
|
||||||
export class AddPageToCustomOrderAction implements Action {
|
|
||||||
type = ObjectUpdatesActionTypes.ADD_PAGE_TO_CUSTOM_ORDER;
|
|
||||||
payload: {
|
|
||||||
url: string,
|
|
||||||
fields: Identifiable[],
|
|
||||||
order: string[],
|
|
||||||
page: number
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a new AddPageToCustomOrderAction
|
|
||||||
*
|
|
||||||
* @param url The unique url of the page for which the fields are being added
|
|
||||||
* @param fields The identifiable fields of which the updates are kept track of
|
|
||||||
* @param order A custom order to keep track of objects moving around
|
|
||||||
* @param page The page to populate in the custom order
|
|
||||||
*/
|
|
||||||
constructor(
|
|
||||||
url: string,
|
|
||||||
fields: Identifiable[],
|
|
||||||
order: string[] = [],
|
|
||||||
page: number = 0
|
|
||||||
) {
|
|
||||||
this.payload = { url, fields, order, page };
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -320,43 +281,6 @@ export class RemoveFieldUpdateAction implements Action {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* An ngrx action to remove a single field update in the ObjectUpdates state for a certain page url and field uuid
|
|
||||||
*/
|
|
||||||
export class MoveFieldUpdateAction implements Action {
|
|
||||||
type = ObjectUpdatesActionTypes.MOVE;
|
|
||||||
payload: {
|
|
||||||
url: string,
|
|
||||||
from: number,
|
|
||||||
to: number,
|
|
||||||
fromPage: number,
|
|
||||||
toPage: number,
|
|
||||||
field?: Identifiable
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a new RemoveObjectUpdatesAction
|
|
||||||
*
|
|
||||||
* @param url
|
|
||||||
* the unique url of the page for which a field's change should be removed
|
|
||||||
* @param from The index of the object to move
|
|
||||||
* @param to The index to move the object to
|
|
||||||
* @param fromPage The page to move the object from
|
|
||||||
* @param toPage The page to move the object to
|
|
||||||
* @param field Optional field to add to the fieldUpdates list (useful when we want to track updates across multiple pages)
|
|
||||||
*/
|
|
||||||
constructor(
|
|
||||||
url: string,
|
|
||||||
from: number,
|
|
||||||
to: number,
|
|
||||||
fromPage: number,
|
|
||||||
toPage: number,
|
|
||||||
field?: Identifiable
|
|
||||||
) {
|
|
||||||
this.payload = { url, from, to, fromPage, toPage, field };
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* tslint:enable:max-classes-per-file */
|
/* tslint:enable:max-classes-per-file */
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -369,8 +293,6 @@ export type ObjectUpdatesAction
|
|||||||
| ReinstateObjectUpdatesAction
|
| ReinstateObjectUpdatesAction
|
||||||
| RemoveObjectUpdatesAction
|
| RemoveObjectUpdatesAction
|
||||||
| RemoveFieldUpdateAction
|
| RemoveFieldUpdateAction
|
||||||
| MoveFieldUpdateAction
|
|
||||||
| AddPageToCustomOrderAction
|
|
||||||
| RemoveAllObjectUpdatesAction
|
| RemoveAllObjectUpdatesAction
|
||||||
| SelectVirtualMetadataAction
|
| SelectVirtualMetadataAction
|
||||||
| SetEditableFieldUpdateAction
|
| SetEditableFieldUpdateAction
|
||||||
|
@@ -1,9 +1,9 @@
|
|||||||
import * as deepFreeze from 'deep-freeze';
|
import * as deepFreeze from 'deep-freeze';
|
||||||
import {
|
import {
|
||||||
AddFieldUpdateAction, AddPageToCustomOrderAction,
|
AddFieldUpdateAction,
|
||||||
DiscardObjectUpdatesAction,
|
DiscardObjectUpdatesAction,
|
||||||
FieldChangeType,
|
FieldChangeType,
|
||||||
InitializeFieldsAction, MoveFieldUpdateAction,
|
InitializeFieldsAction,
|
||||||
ReinstateObjectUpdatesAction, RemoveAllObjectUpdatesAction,
|
ReinstateObjectUpdatesAction, RemoveAllObjectUpdatesAction,
|
||||||
RemoveFieldUpdateAction, RemoveObjectUpdatesAction, SelectVirtualMetadataAction,
|
RemoveFieldUpdateAction, RemoveObjectUpdatesAction, SelectVirtualMetadataAction,
|
||||||
SetEditableFieldUpdateAction, SetValidFieldUpdateAction
|
SetEditableFieldUpdateAction, SetValidFieldUpdateAction
|
||||||
@@ -85,16 +85,6 @@ describe('objectUpdatesReducer', () => {
|
|||||||
virtualMetadataSources: {
|
virtualMetadataSources: {
|
||||||
[relationship.uuid]: {[identifiable1.uuid]: true}
|
[relationship.uuid]: {[identifiable1.uuid]: true}
|
||||||
},
|
},
|
||||||
customOrder: {
|
|
||||||
initialOrderPages: [
|
|
||||||
{ order: [identifiable1.uuid, identifiable2.uuid, identifiable3.uuid] }
|
|
||||||
],
|
|
||||||
newOrderPages: [
|
|
||||||
{ order: [identifiable1.uuid, identifiable2.uuid, identifiable3.uuid] }
|
|
||||||
],
|
|
||||||
pageSize: 10,
|
|
||||||
changed: false
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -121,16 +111,6 @@ describe('objectUpdatesReducer', () => {
|
|||||||
virtualMetadataSources: {
|
virtualMetadataSources: {
|
||||||
[relationship.uuid]: {[identifiable1.uuid]: true}
|
[relationship.uuid]: {[identifiable1.uuid]: true}
|
||||||
},
|
},
|
||||||
customOrder: {
|
|
||||||
initialOrderPages: [
|
|
||||||
{ order: [identifiable1.uuid, identifiable2.uuid, identifiable3.uuid] }
|
|
||||||
],
|
|
||||||
newOrderPages: [
|
|
||||||
{ order: [identifiable1.uuid, identifiable2.uuid, identifiable3.uuid] }
|
|
||||||
],
|
|
||||||
pageSize: 10,
|
|
||||||
changed: false
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
[url + OBJECT_UPDATES_TRASH_PATH]: {
|
[url + OBJECT_UPDATES_TRASH_PATH]: {
|
||||||
fieldStates: {
|
fieldStates: {
|
||||||
@@ -165,16 +145,6 @@ describe('objectUpdatesReducer', () => {
|
|||||||
virtualMetadataSources: {
|
virtualMetadataSources: {
|
||||||
[relationship.uuid]: {[identifiable1.uuid]: true}
|
[relationship.uuid]: {[identifiable1.uuid]: true}
|
||||||
},
|
},
|
||||||
customOrder: {
|
|
||||||
initialOrderPages: [
|
|
||||||
{ order: [identifiable1.uuid, identifiable2.uuid, identifiable3.uuid] }
|
|
||||||
],
|
|
||||||
newOrderPages: [
|
|
||||||
{ order: [identifiable1.uuid, identifiable2.uuid, identifiable3.uuid] }
|
|
||||||
],
|
|
||||||
pageSize: 10,
|
|
||||||
changed: false
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -243,7 +213,7 @@ describe('objectUpdatesReducer', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('should initialize all fields when the INITIALIZE action is dispatched, based on the payload', () => {
|
it('should initialize all fields when the INITIALIZE action is dispatched, based on the payload', () => {
|
||||||
const action = new InitializeFieldsAction(url, [identifiable1, identifiable3], modDate, [identifiable1.uuid, identifiable3.uuid], 10, 0);
|
const action = new InitializeFieldsAction(url, [identifiable1, identifiable3], modDate);
|
||||||
|
|
||||||
const expectedState = {
|
const expectedState = {
|
||||||
[url]: {
|
[url]: {
|
||||||
@@ -261,17 +231,7 @@ describe('objectUpdatesReducer', () => {
|
|||||||
},
|
},
|
||||||
fieldUpdates: {},
|
fieldUpdates: {},
|
||||||
virtualMetadataSources: {},
|
virtualMetadataSources: {},
|
||||||
lastModified: modDate,
|
lastModified: modDate
|
||||||
customOrder: {
|
|
||||||
initialOrderPages: [
|
|
||||||
{ order: [identifiable1.uuid, identifiable3.uuid] }
|
|
||||||
],
|
|
||||||
newOrderPages: [
|
|
||||||
{ order: [identifiable1.uuid, identifiable3.uuid] }
|
|
||||||
],
|
|
||||||
pageSize: 10,
|
|
||||||
changed: false
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
const newState = objectUpdatesReducer(testState, action);
|
const newState = objectUpdatesReducer(testState, action);
|
||||||
@@ -337,30 +297,4 @@ describe('objectUpdatesReducer', () => {
|
|||||||
const newState = objectUpdatesReducer(testState, action);
|
const newState = objectUpdatesReducer(testState, action);
|
||||||
expect(newState[url].fieldUpdates[uuid]).toBeUndefined();
|
expect(newState[url].fieldUpdates[uuid]).toBeUndefined();
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should move the custom order from the state when the MOVE action is dispatched', () => {
|
|
||||||
const action = new MoveFieldUpdateAction(url, 0, 1, 0, 0);
|
|
||||||
|
|
||||||
const newState = objectUpdatesReducer(testState, action);
|
|
||||||
expect(newState[url].customOrder.newOrderPages[0].order[0]).toEqual(testState[url].customOrder.newOrderPages[0].order[1]);
|
|
||||||
expect(newState[url].customOrder.newOrderPages[0].order[1]).toEqual(testState[url].customOrder.newOrderPages[0].order[0]);
|
|
||||||
expect(newState[url].customOrder.changed).toEqual(true);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should add a new page to the custom order and add empty pages in between when the ADD_PAGE_TO_CUSTOM_ORDER action is dispatched', () => {
|
|
||||||
const identifiable4 = {
|
|
||||||
uuid: 'a23eae5a-7857-4ef9-8e52-989436ad2955',
|
|
||||||
key: 'dc.description.abstract',
|
|
||||||
language: null,
|
|
||||||
value: 'Extra value'
|
|
||||||
};
|
|
||||||
const action = new AddPageToCustomOrderAction(url, [identifiable4], [identifiable4.uuid], 2);
|
|
||||||
|
|
||||||
const newState = objectUpdatesReducer(testState, action);
|
|
||||||
// Confirm the page in between the two pages (index 1) has been filled with 10 (page size) undefined values
|
|
||||||
expect(newState[url].customOrder.newOrderPages[1].order.length).toEqual(10);
|
|
||||||
expect(newState[url].customOrder.newOrderPages[1].order[0]).toBeUndefined();
|
|
||||||
// Verify the new page is correct
|
|
||||||
expect(newState[url].customOrder.newOrderPages[2].order[0]).toEqual(identifiable4.uuid);
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
@@ -1,8 +1,8 @@
|
|||||||
import {
|
import {
|
||||||
AddFieldUpdateAction, AddPageToCustomOrderAction,
|
AddFieldUpdateAction,
|
||||||
DiscardObjectUpdatesAction,
|
DiscardObjectUpdatesAction,
|
||||||
FieldChangeType,
|
FieldChangeType,
|
||||||
InitializeFieldsAction, MoveFieldUpdateAction,
|
InitializeFieldsAction,
|
||||||
ObjectUpdatesAction,
|
ObjectUpdatesAction,
|
||||||
ObjectUpdatesActionTypes,
|
ObjectUpdatesActionTypes,
|
||||||
ReinstateObjectUpdatesAction,
|
ReinstateObjectUpdatesAction,
|
||||||
@@ -12,9 +12,7 @@ import {
|
|||||||
SetValidFieldUpdateAction,
|
SetValidFieldUpdateAction,
|
||||||
SelectVirtualMetadataAction,
|
SelectVirtualMetadataAction,
|
||||||
} from './object-updates.actions';
|
} from './object-updates.actions';
|
||||||
import { hasNoValue, hasValue, isEmpty, isNotEmpty } from '../../../shared/empty.util';
|
import { hasNoValue, hasValue } from '../../../shared/empty.util';
|
||||||
import { moveItemInArray, transferArrayItem } from '@angular/cdk/drag-drop';
|
|
||||||
import { from } from 'rxjs/internal/observable/from';
|
|
||||||
import {Relationship} from '../../shared/item-relationships/relationship.model';
|
import {Relationship} from '../../shared/item-relationships/relationship.model';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -83,20 +81,6 @@ export interface DeleteRelationship extends Relationship {
|
|||||||
keepRightVirtualMetadata: boolean,
|
keepRightVirtualMetadata: boolean,
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* A custom order given to the list of objects
|
|
||||||
*/
|
|
||||||
export interface CustomOrder {
|
|
||||||
initialOrderPages: OrderPage[],
|
|
||||||
newOrderPages: OrderPage[],
|
|
||||||
pageSize: number;
|
|
||||||
changed: boolean
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface OrderPage {
|
|
||||||
order: string[]
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The updated state of a single page
|
* The updated state of a single page
|
||||||
*/
|
*/
|
||||||
@@ -105,7 +89,6 @@ export interface ObjectUpdatesEntry {
|
|||||||
fieldUpdates: FieldUpdates;
|
fieldUpdates: FieldUpdates;
|
||||||
virtualMetadataSources: VirtualMetadataSources;
|
virtualMetadataSources: VirtualMetadataSources;
|
||||||
lastModified: Date;
|
lastModified: Date;
|
||||||
customOrder: CustomOrder
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -138,9 +121,6 @@ export function objectUpdatesReducer(state = initialState, action: ObjectUpdates
|
|||||||
case ObjectUpdatesActionTypes.INITIALIZE_FIELDS: {
|
case ObjectUpdatesActionTypes.INITIALIZE_FIELDS: {
|
||||||
return initializeFieldsUpdate(state, action as InitializeFieldsAction);
|
return initializeFieldsUpdate(state, action as InitializeFieldsAction);
|
||||||
}
|
}
|
||||||
case ObjectUpdatesActionTypes.ADD_PAGE_TO_CUSTOM_ORDER: {
|
|
||||||
return addPageToCustomOrder(state, action as AddPageToCustomOrderAction);
|
|
||||||
}
|
|
||||||
case ObjectUpdatesActionTypes.ADD_FIELD: {
|
case ObjectUpdatesActionTypes.ADD_FIELD: {
|
||||||
return addFieldUpdate(state, action as AddFieldUpdateAction);
|
return addFieldUpdate(state, action as AddFieldUpdateAction);
|
||||||
}
|
}
|
||||||
@@ -168,9 +148,6 @@ export function objectUpdatesReducer(state = initialState, action: ObjectUpdates
|
|||||||
case ObjectUpdatesActionTypes.SET_VALID_FIELD: {
|
case ObjectUpdatesActionTypes.SET_VALID_FIELD: {
|
||||||
return setValidFieldUpdate(state, action as SetValidFieldUpdateAction);
|
return setValidFieldUpdate(state, action as SetValidFieldUpdateAction);
|
||||||
}
|
}
|
||||||
case ObjectUpdatesActionTypes.MOVE: {
|
|
||||||
return moveFieldUpdate(state, action as MoveFieldUpdateAction);
|
|
||||||
}
|
|
||||||
default: {
|
default: {
|
||||||
return state;
|
return state;
|
||||||
}
|
}
|
||||||
@@ -186,50 +163,18 @@ function initializeFieldsUpdate(state: any, action: InitializeFieldsAction) {
|
|||||||
const url: string = action.payload.url;
|
const url: string = action.payload.url;
|
||||||
const fields: Identifiable[] = action.payload.fields;
|
const fields: Identifiable[] = action.payload.fields;
|
||||||
const lastModifiedServer: Date = action.payload.lastModified;
|
const lastModifiedServer: Date = action.payload.lastModified;
|
||||||
const order = action.payload.order;
|
|
||||||
const pageSize = action.payload.pageSize;
|
|
||||||
const page = action.payload.page;
|
|
||||||
const fieldStates = createInitialFieldStates(fields);
|
const fieldStates = createInitialFieldStates(fields);
|
||||||
const initialOrderPages = addOrderToPages([], order, pageSize, page);
|
|
||||||
const newPageState = Object.assign(
|
const newPageState = Object.assign(
|
||||||
{},
|
{},
|
||||||
state[url],
|
state[url],
|
||||||
{ fieldStates: fieldStates },
|
{ fieldStates: fieldStates },
|
||||||
{ fieldUpdates: {} },
|
{ fieldUpdates: {} },
|
||||||
{ virtualMetadataSources: {} },
|
{ virtualMetadataSources: {} },
|
||||||
{ lastModified: lastModifiedServer },
|
{ lastModified: lastModifiedServer }
|
||||||
{ customOrder: {
|
|
||||||
initialOrderPages: initialOrderPages,
|
|
||||||
newOrderPages: initialOrderPages,
|
|
||||||
pageSize: pageSize,
|
|
||||||
changed: false }
|
|
||||||
}
|
|
||||||
);
|
);
|
||||||
return Object.assign({}, state, { [url]: newPageState });
|
return Object.assign({}, state, { [url]: newPageState });
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Add a page of objects to the state of a specific url and update a specific page of the custom order
|
|
||||||
* @param state The current state
|
|
||||||
* @param action The action to perform on the current state
|
|
||||||
*/
|
|
||||||
function addPageToCustomOrder(state: any, action: AddPageToCustomOrderAction) {
|
|
||||||
const url: string = action.payload.url;
|
|
||||||
const fields: Identifiable[] = action.payload.fields;
|
|
||||||
const fieldStates = createInitialFieldStates(fields);
|
|
||||||
const order = action.payload.order;
|
|
||||||
const page = action.payload.page;
|
|
||||||
const pageState: ObjectUpdatesEntry = state[url] || {};
|
|
||||||
const newPageState = Object.assign({}, pageState, {
|
|
||||||
fieldStates: Object.assign({}, pageState.fieldStates, fieldStates),
|
|
||||||
customOrder: Object.assign({}, pageState.customOrder, {
|
|
||||||
newOrderPages: addOrderToPages(pageState.customOrder.newOrderPages, order, pageState.customOrder.pageSize, page),
|
|
||||||
initialOrderPages: addOrderToPages(pageState.customOrder.initialOrderPages, order, pageState.customOrder.pageSize, page)
|
|
||||||
})
|
|
||||||
});
|
|
||||||
return Object.assign({}, state, { [url]: newPageState });
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Add a new update for a specific field to the store
|
* Add a new update for a specific field to the store
|
||||||
* @param state The current state
|
* @param state The current state
|
||||||
@@ -338,19 +283,9 @@ function discardObjectUpdatesFor(url: string, state: any) {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
const newCustomOrder = Object.assign({}, pageState.customOrder);
|
|
||||||
if (pageState.customOrder.changed) {
|
|
||||||
const initialOrder = pageState.customOrder.initialOrderPages;
|
|
||||||
if (isNotEmpty(initialOrder)) {
|
|
||||||
newCustomOrder.newOrderPages = initialOrder;
|
|
||||||
newCustomOrder.changed = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const discardedPageState = Object.assign({}, pageState, {
|
const discardedPageState = Object.assign({}, pageState, {
|
||||||
fieldUpdates: {},
|
fieldUpdates: {},
|
||||||
fieldStates: newFieldStates,
|
fieldStates: newFieldStates
|
||||||
customOrder: newCustomOrder
|
|
||||||
});
|
});
|
||||||
return Object.assign({}, state, { [url]: discardedPageState }, { [url + OBJECT_UPDATES_TRASH_PATH]: pageState });
|
return Object.assign({}, state, { [url]: discardedPageState }, { [url + OBJECT_UPDATES_TRASH_PATH]: pageState });
|
||||||
}
|
}
|
||||||
@@ -504,121 +439,3 @@ function createInitialFieldStates(fields: Identifiable[]) {
|
|||||||
uuids.forEach((uuid: string) => fieldStates[uuid] = initialFieldState);
|
uuids.forEach((uuid: string) => fieldStates[uuid] = initialFieldState);
|
||||||
return fieldStates;
|
return fieldStates;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Method to add a list of objects to an existing FieldStates object
|
|
||||||
* @param fieldStates FieldStates to add states to
|
|
||||||
* @param fields Identifiable objects The list of objects to add to the FieldStates
|
|
||||||
*/
|
|
||||||
function addFieldStates(fieldStates: FieldStates, fields: Identifiable[]) {
|
|
||||||
const uuids = fields.map((field: Identifiable) => field.uuid);
|
|
||||||
uuids.forEach((uuid: string) => fieldStates[uuid] = initialFieldState);
|
|
||||||
return fieldStates;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Move an object within the custom order of a page state
|
|
||||||
* @param state The current state
|
|
||||||
* @param action The move action to perform
|
|
||||||
*/
|
|
||||||
function moveFieldUpdate(state: any, action: MoveFieldUpdateAction) {
|
|
||||||
const url = action.payload.url;
|
|
||||||
const fromIndex = action.payload.from;
|
|
||||||
const toIndex = action.payload.to;
|
|
||||||
const fromPage = action.payload.fromPage;
|
|
||||||
const toPage = action.payload.toPage;
|
|
||||||
const field = action.payload.field;
|
|
||||||
|
|
||||||
const pageState: ObjectUpdatesEntry = state[url];
|
|
||||||
const initialOrderPages = pageState.customOrder.initialOrderPages;
|
|
||||||
const customOrderPages = [...pageState.customOrder.newOrderPages];
|
|
||||||
|
|
||||||
// Create a copy of the custom orders for the from- and to-pages
|
|
||||||
const fromPageOrder = [...customOrderPages[fromPage].order];
|
|
||||||
const toPageOrder = [...customOrderPages[toPage].order];
|
|
||||||
if (fromPage === toPage) {
|
|
||||||
if (isNotEmpty(customOrderPages[fromPage]) && isNotEmpty(customOrderPages[fromPage].order[fromIndex]) && isNotEmpty(customOrderPages[fromPage].order[toIndex])) {
|
|
||||||
// Move an item from one index to another within the same page
|
|
||||||
moveItemInArray(fromPageOrder, fromIndex, toIndex);
|
|
||||||
// Update the custom order for this page
|
|
||||||
customOrderPages[fromPage] = { order: fromPageOrder };
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (isNotEmpty(customOrderPages[fromPage]) && hasValue(customOrderPages[toPage]) && isNotEmpty(customOrderPages[fromPage].order[fromIndex])) {
|
|
||||||
// Move an item from one index of one page to an index in another page
|
|
||||||
transferArrayItem(fromPageOrder, toPageOrder, fromIndex, toIndex);
|
|
||||||
// Update the custom order for both pages
|
|
||||||
customOrderPages[fromPage] = { order: fromPageOrder };
|
|
||||||
customOrderPages[toPage] = { order: toPageOrder };
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create a field update if it doesn't exist for this field yet
|
|
||||||
let fieldUpdate = {};
|
|
||||||
if (hasValue(field)) {
|
|
||||||
fieldUpdate = pageState.fieldUpdates[field.uuid];
|
|
||||||
if (hasNoValue(fieldUpdate)) {
|
|
||||||
fieldUpdate = { field: field, changeType: undefined }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Update the store's state with new values and return
|
|
||||||
return Object.assign({}, state, { [url]: Object.assign({}, pageState, {
|
|
||||||
fieldUpdates: Object.assign({}, pageState.fieldUpdates, hasValue(field) ? { [field.uuid]: fieldUpdate } : {}),
|
|
||||||
customOrder: Object.assign({}, pageState.customOrder, { newOrderPages: customOrderPages, changed: checkForOrderChanges(initialOrderPages, customOrderPages) })
|
|
||||||
})})
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Compare two lists of OrderPage objects and return whether there's at least one change in the order of objects within
|
|
||||||
* @param initialOrderPages The initial list of OrderPages
|
|
||||||
* @param customOrderPages The changed list of OrderPages
|
|
||||||
*/
|
|
||||||
function checkForOrderChanges(initialOrderPages: OrderPage[], customOrderPages: OrderPage[]) {
|
|
||||||
let changed = false;
|
|
||||||
initialOrderPages.forEach((orderPage: OrderPage, page: number) => {
|
|
||||||
if (isNotEmpty(orderPage) && isNotEmpty(orderPage.order) && isNotEmpty(customOrderPages[page]) && isNotEmpty(customOrderPages[page].order)) {
|
|
||||||
orderPage.order.forEach((id: string, index: number) => {
|
|
||||||
if (id !== customOrderPages[page].order[index]) {
|
|
||||||
changed = true;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
if (changed) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
return changed;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Initialize a custom order page by providing the list of all pages, a list of UUIDs, pageSize and the page to populate
|
|
||||||
* @param initialPages The initial list of OrderPage objects
|
|
||||||
* @param order The list of UUIDs to create a page for
|
|
||||||
* @param pageSize The pageSize used to populate empty spacer pages
|
|
||||||
* @param page The index of the page to add
|
|
||||||
*/
|
|
||||||
function addOrderToPages(initialPages: OrderPage[], order: string[], pageSize: number, page: number): OrderPage[] {
|
|
||||||
const result = [...initialPages];
|
|
||||||
const orderPage: OrderPage = { order: order };
|
|
||||||
if (page < result.length) {
|
|
||||||
// The page we're trying to add already exists in the list. Overwrite it.
|
|
||||||
result[page] = orderPage;
|
|
||||||
} else if (page === result.length) {
|
|
||||||
// The page we're trying to add is the next page in the list, add it.
|
|
||||||
result.push(orderPage);
|
|
||||||
} else {
|
|
||||||
// The page we're trying to add is at least one page ahead of the list, fill the list with empty pages before adding the page.
|
|
||||||
const emptyOrder = [];
|
|
||||||
for (let i = 0; i < pageSize; i++) {
|
|
||||||
emptyOrder.push(undefined);
|
|
||||||
}
|
|
||||||
const emptyOrderPage: OrderPage = { order: emptyOrder };
|
|
||||||
for (let i = result.length; i < page; i++) {
|
|
||||||
result.push(emptyOrderPage);
|
|
||||||
}
|
|
||||||
result.push(orderPage);
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
@@ -2,7 +2,6 @@ import { Store } from '@ngrx/store';
|
|||||||
import { CoreState } from '../../core.reducers';
|
import { CoreState } from '../../core.reducers';
|
||||||
import { ObjectUpdatesService } from './object-updates.service';
|
import { ObjectUpdatesService } from './object-updates.service';
|
||||||
import {
|
import {
|
||||||
AddPageToCustomOrderAction,
|
|
||||||
DiscardObjectUpdatesAction,
|
DiscardObjectUpdatesAction,
|
||||||
FieldChangeType,
|
FieldChangeType,
|
||||||
InitializeFieldsAction, ReinstateObjectUpdatesAction, RemoveFieldUpdateAction, SelectVirtualMetadataAction,
|
InitializeFieldsAction, ReinstateObjectUpdatesAction, RemoveFieldUpdateAction, SelectVirtualMetadataAction,
|
||||||
@@ -13,8 +12,6 @@ import { Notification } from '../../../shared/notifications/models/notification.
|
|||||||
import { NotificationType } from '../../../shared/notifications/models/notification-type';
|
import { NotificationType } from '../../../shared/notifications/models/notification-type';
|
||||||
import { OBJECT_UPDATES_TRASH_PATH } from './object-updates.reducer';
|
import { OBJECT_UPDATES_TRASH_PATH } from './object-updates.reducer';
|
||||||
import {Relationship} from '../../shared/item-relationships/relationship.model';
|
import {Relationship} from '../../shared/item-relationships/relationship.model';
|
||||||
import { MoveOperation } from 'fast-json-patch/lib/core';
|
|
||||||
import { ArrayMoveChangeAnalyzer } from '../array-move-change-analyzer.service';
|
|
||||||
|
|
||||||
describe('ObjectUpdatesService', () => {
|
describe('ObjectUpdatesService', () => {
|
||||||
let service: ObjectUpdatesService;
|
let service: ObjectUpdatesService;
|
||||||
@@ -47,7 +44,7 @@ describe('ObjectUpdatesService', () => {
|
|||||||
};
|
};
|
||||||
store = new Store<CoreState>(undefined, undefined, undefined);
|
store = new Store<CoreState>(undefined, undefined, undefined);
|
||||||
spyOn(store, 'dispatch');
|
spyOn(store, 'dispatch');
|
||||||
service = new ObjectUpdatesService(store, new ArrayMoveChangeAnalyzer<string>());
|
service = new ObjectUpdatesService(store);
|
||||||
|
|
||||||
spyOn(service as any, 'getObjectEntry').and.returnValue(observableOf(objectEntry));
|
spyOn(service as any, 'getObjectEntry').and.returnValue(observableOf(objectEntry));
|
||||||
spyOn(service as any, 'getFieldState').and.callFake((uuid) => {
|
spyOn(service as any, 'getFieldState').and.callFake((uuid) => {
|
||||||
@@ -63,25 +60,6 @@ describe('ObjectUpdatesService', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('initializeWithCustomOrder', () => {
|
|
||||||
const pageSize = 20;
|
|
||||||
const page = 0;
|
|
||||||
|
|
||||||
it('should dispatch an INITIALIZE action with the correct URL, initial identifiables, last modified , custom order, page size and page', () => {
|
|
||||||
service.initializeWithCustomOrder(url, identifiables, modDate, pageSize, page);
|
|
||||||
expect(store.dispatch).toHaveBeenCalledWith(new InitializeFieldsAction(url, identifiables, modDate, identifiables.map((identifiable) => identifiable.uuid), pageSize, page));
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('addPageToCustomOrder', () => {
|
|
||||||
const page = 2;
|
|
||||||
|
|
||||||
it('should dispatch an ADD_PAGE_TO_CUSTOM_ORDER action with the correct URL, identifiables, custom order and page number to add', () => {
|
|
||||||
service.addPageToCustomOrder(url, identifiables, page);
|
|
||||||
expect(store.dispatch).toHaveBeenCalledWith(new AddPageToCustomOrderAction(url, identifiables, identifiables.map((identifiable) => identifiable.uuid), page));
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('getFieldUpdates', () => {
|
describe('getFieldUpdates', () => {
|
||||||
it('should return the list of all fields, including their update if there is one', () => {
|
it('should return the list of all fields, including their update if there is one', () => {
|
||||||
const result$ = service.getFieldUpdates(url, identifiables);
|
const result$ = service.getFieldUpdates(url, identifiables);
|
||||||
@@ -116,49 +94,6 @@ describe('ObjectUpdatesService', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('getFieldUpdatesByCustomOrder', () => {
|
|
||||||
beforeEach(() => {
|
|
||||||
const fieldStates = {
|
|
||||||
[identifiable1.uuid]: { editable: false, isNew: false, isValid: true },
|
|
||||||
[identifiable2.uuid]: { editable: true, isNew: false, isValid: false },
|
|
||||||
[identifiable3.uuid]: { editable: true, isNew: true, isValid: true },
|
|
||||||
};
|
|
||||||
|
|
||||||
const customOrder = {
|
|
||||||
initialOrderPages: [{
|
|
||||||
order: [identifiable1.uuid, identifiable2.uuid, identifiable3.uuid]
|
|
||||||
}],
|
|
||||||
newOrderPages: [{
|
|
||||||
order: [identifiable2.uuid, identifiable3.uuid, identifiable1.uuid]
|
|
||||||
}],
|
|
||||||
pageSize: 20,
|
|
||||||
changed: true
|
|
||||||
};
|
|
||||||
|
|
||||||
const objectEntry = {
|
|
||||||
fieldStates, fieldUpdates, lastModified: modDate, virtualMetadataSources: {}, customOrder
|
|
||||||
};
|
|
||||||
|
|
||||||
(service as any).getObjectEntry.and.returnValue(observableOf(objectEntry))
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should return the list of all fields, including their update if there is one, ordered by their custom order', (done) => {
|
|
||||||
const result$ = service.getFieldUpdatesByCustomOrder(url, identifiables);
|
|
||||||
expect((service as any).getObjectEntry).toHaveBeenCalledWith(url);
|
|
||||||
|
|
||||||
const expectedResult = {
|
|
||||||
[identifiable2.uuid]: { field: identifiable2, changeType: undefined },
|
|
||||||
[identifiable3.uuid]: { field: identifiable3, changeType: FieldChangeType.ADD },
|
|
||||||
[identifiable1.uuid]: { field: identifiable1Updated, changeType: FieldChangeType.UPDATE }
|
|
||||||
};
|
|
||||||
|
|
||||||
result$.subscribe((result) => {
|
|
||||||
expect(result).toEqual(expectedResult);
|
|
||||||
done();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('isEditable', () => {
|
describe('isEditable', () => {
|
||||||
it('should return false if this identifiable is currently not editable in the store', () => {
|
it('should return false if this identifiable is currently not editable in the store', () => {
|
||||||
const result$ = service.isEditable(url, identifiable1.uuid);
|
const result$ = service.isEditable(url, identifiable1.uuid);
|
||||||
@@ -274,11 +209,7 @@ describe('ObjectUpdatesService', () => {
|
|||||||
});
|
});
|
||||||
describe('when updates are emtpy', () => {
|
describe('when updates are emtpy', () => {
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
(service as any).getObjectEntry.and.returnValue(observableOf({
|
(service as any).getObjectEntry.and.returnValue(observableOf({}))
|
||||||
customOrder: {
|
|
||||||
changed: false
|
|
||||||
}
|
|
||||||
}))
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should return false when there are no updates', () => {
|
it('should return false when there are no updates', () => {
|
||||||
@@ -346,44 +277,4 @@ describe('ObjectUpdatesService', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('getMoveOperations', () => {
|
|
||||||
beforeEach(() => {
|
|
||||||
const fieldStates = {
|
|
||||||
[identifiable1.uuid]: { editable: false, isNew: false, isValid: true },
|
|
||||||
[identifiable2.uuid]: { editable: true, isNew: false, isValid: false },
|
|
||||||
[identifiable3.uuid]: { editable: true, isNew: true, isValid: true },
|
|
||||||
};
|
|
||||||
|
|
||||||
const customOrder = {
|
|
||||||
initialOrderPages: [{
|
|
||||||
order: [identifiable1.uuid, identifiable2.uuid, identifiable3.uuid]
|
|
||||||
}],
|
|
||||||
newOrderPages: [{
|
|
||||||
order: [identifiable2.uuid, identifiable3.uuid, identifiable1.uuid]
|
|
||||||
}],
|
|
||||||
pageSize: 20,
|
|
||||||
changed: true
|
|
||||||
};
|
|
||||||
|
|
||||||
const objectEntry = {
|
|
||||||
fieldStates, fieldUpdates, lastModified: modDate, virtualMetadataSources: {}, customOrder
|
|
||||||
};
|
|
||||||
|
|
||||||
(service as any).getObjectEntry.and.returnValue(observableOf(objectEntry))
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should return the expected move operations', (done) => {
|
|
||||||
const result$ = service.getMoveOperations(url);
|
|
||||||
|
|
||||||
const expectedResult = [
|
|
||||||
{ op: 'move', from: '/0', path: '/2' }
|
|
||||||
] as MoveOperation[];
|
|
||||||
|
|
||||||
result$.subscribe((result) => {
|
|
||||||
expect(result).toEqual(expectedResult);
|
|
||||||
done();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
});
|
});
|
||||||
|
@@ -8,16 +8,15 @@ import {
|
|||||||
Identifiable,
|
Identifiable,
|
||||||
OBJECT_UPDATES_TRASH_PATH,
|
OBJECT_UPDATES_TRASH_PATH,
|
||||||
ObjectUpdatesEntry,
|
ObjectUpdatesEntry,
|
||||||
ObjectUpdatesState, OrderPage,
|
ObjectUpdatesState,
|
||||||
VirtualMetadataSource
|
VirtualMetadataSource
|
||||||
} from './object-updates.reducer';
|
} from './object-updates.reducer';
|
||||||
import { Observable } from 'rxjs';
|
import { Observable } from 'rxjs';
|
||||||
import {
|
import {
|
||||||
AddFieldUpdateAction, AddPageToCustomOrderAction,
|
AddFieldUpdateAction,
|
||||||
DiscardObjectUpdatesAction,
|
DiscardObjectUpdatesAction,
|
||||||
FieldChangeType,
|
FieldChangeType,
|
||||||
InitializeFieldsAction,
|
InitializeFieldsAction,
|
||||||
MoveFieldUpdateAction,
|
|
||||||
ReinstateObjectUpdatesAction,
|
ReinstateObjectUpdatesAction,
|
||||||
RemoveFieldUpdateAction,
|
RemoveFieldUpdateAction,
|
||||||
SelectVirtualMetadataAction,
|
SelectVirtualMetadataAction,
|
||||||
@@ -25,11 +24,8 @@ import {
|
|||||||
SetValidFieldUpdateAction
|
SetValidFieldUpdateAction
|
||||||
} from './object-updates.actions';
|
} from './object-updates.actions';
|
||||||
import { distinctUntilChanged, filter, map, switchMap } from 'rxjs/operators';
|
import { distinctUntilChanged, filter, map, switchMap } from 'rxjs/operators';
|
||||||
import { hasNoValue, hasValue, isEmpty, isNotEmpty } from '../../../shared/empty.util';
|
import { hasNoValue, hasValue, isEmpty, isNotEmpty, isNotEmptyOperator } from '../../../shared/empty.util';
|
||||||
import { INotification } from '../../../shared/notifications/models/notification.model';
|
import { INotification } from '../../../shared/notifications/models/notification.model';
|
||||||
import { ArrayMoveChangeAnalyzer } from '../array-move-change-analyzer.service';
|
|
||||||
import { MoveOperation } from 'fast-json-patch/lib/core';
|
|
||||||
import { flatten } from '@angular/compiler';
|
|
||||||
|
|
||||||
function objectUpdatesStateSelector(): MemoizedSelector<CoreState, ObjectUpdatesState> {
|
function objectUpdatesStateSelector(): MemoizedSelector<CoreState, ObjectUpdatesState> {
|
||||||
return createSelector(coreSelector, (state: CoreState) => state['cache/object-updates']);
|
return createSelector(coreSelector, (state: CoreState) => state['cache/object-updates']);
|
||||||
@@ -52,9 +48,7 @@ function virtualMetadataSourceSelector(url: string, source: string): MemoizedSel
|
|||||||
*/
|
*/
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class ObjectUpdatesService {
|
export class ObjectUpdatesService {
|
||||||
constructor(private store: Store<CoreState>,
|
constructor(private store: Store<CoreState>) {
|
||||||
private comparator: ArrayMoveChangeAnalyzer<string>) {
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -67,28 +61,6 @@ export class ObjectUpdatesService {
|
|||||||
this.store.dispatch(new InitializeFieldsAction(url, fields, lastModified));
|
this.store.dispatch(new InitializeFieldsAction(url, fields, lastModified));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Method to dispatch an InitializeFieldsAction to the store and keeping track of the order objects are stored
|
|
||||||
* @param url The page's URL for which the changes are being mapped
|
|
||||||
* @param fields The initial fields for the page's object
|
|
||||||
* @param lastModified The date the object was last modified
|
|
||||||
* @param pageSize The page size to use for adding pages to the custom order
|
|
||||||
* @param page The first page to populate the custom order with
|
|
||||||
*/
|
|
||||||
initializeWithCustomOrder(url, fields: Identifiable[], lastModified: Date, pageSize = 9999, page = 0): void {
|
|
||||||
this.store.dispatch(new InitializeFieldsAction(url, fields, lastModified, fields.map((field) => field.uuid), pageSize, page));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Method to dispatch an AddPageToCustomOrderAction, adding a new page to an already existing custom order tracking
|
|
||||||
* @param url The URL for which the changes are being mapped
|
|
||||||
* @param fields The fields to add a new page for
|
|
||||||
* @param page The page number (starting from index 0)
|
|
||||||
*/
|
|
||||||
addPageToCustomOrder(url, fields: Identifiable[], page: number): void {
|
|
||||||
this.store.dispatch(new AddPageToCustomOrderAction(url, fields, fields.map((field) => field.uuid), page));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Method to dispatch an AddFieldUpdateAction to the store
|
* Method to dispatch an AddFieldUpdateAction to the store
|
||||||
* @param url The page's URL for which the changes are saved
|
* @param url The page's URL for which the changes are saved
|
||||||
@@ -153,7 +125,7 @@ export class ObjectUpdatesService {
|
|||||||
*/
|
*/
|
||||||
getFieldUpdatesExclusive(url: string, initialFields: Identifiable[]): Observable<FieldUpdates> {
|
getFieldUpdatesExclusive(url: string, initialFields: Identifiable[]): Observable<FieldUpdates> {
|
||||||
const objectUpdates = this.getObjectEntry(url);
|
const objectUpdates = this.getObjectEntry(url);
|
||||||
return objectUpdates.pipe(map((objectEntry) => {
|
return objectUpdates.pipe(isNotEmptyOperator(), map((objectEntry) => {
|
||||||
const fieldUpdates: FieldUpdates = {};
|
const fieldUpdates: FieldUpdates = {};
|
||||||
for (const object of initialFields) {
|
for (const object of initialFields) {
|
||||||
let fieldUpdate = objectEntry.fieldUpdates[object.uuid];
|
let fieldUpdate = objectEntry.fieldUpdates[object.uuid];
|
||||||
@@ -166,31 +138,6 @@ export class ObjectUpdatesService {
|
|||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Method that combines the state's updates with the initial values (when there's no update),
|
|
||||||
* sorted by their custom order to create a FieldUpdates object
|
|
||||||
* @param url The URL of the page for which the FieldUpdates should be requested
|
|
||||||
* @param initialFields The initial values of the fields
|
|
||||||
* @param page The page to retrieve
|
|
||||||
*/
|
|
||||||
getFieldUpdatesByCustomOrder(url: string, initialFields: Identifiable[], page = 0): Observable<FieldUpdates> {
|
|
||||||
const objectUpdates = this.getObjectEntry(url);
|
|
||||||
return objectUpdates.pipe(map((objectEntry) => {
|
|
||||||
const fieldUpdates: FieldUpdates = {};
|
|
||||||
if (hasValue(objectEntry) && hasValue(objectEntry.customOrder) && isNotEmpty(objectEntry.customOrder.newOrderPages) && page < objectEntry.customOrder.newOrderPages.length) {
|
|
||||||
for (const uuid of objectEntry.customOrder.newOrderPages[page].order) {
|
|
||||||
let fieldUpdate = objectEntry.fieldUpdates[uuid];
|
|
||||||
if (isEmpty(fieldUpdate)) {
|
|
||||||
const identifiable = initialFields.find((object: Identifiable) => object.uuid === uuid);
|
|
||||||
fieldUpdate = {field: identifiable, changeType: undefined};
|
|
||||||
}
|
|
||||||
fieldUpdates[uuid] = fieldUpdate;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return fieldUpdates;
|
|
||||||
}))
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Method to check if a specific field is currently editable in the store
|
* Method to check if a specific field is currently editable in the store
|
||||||
* @param url The URL of the page on which the field resides
|
* @param url The URL of the page on which the field resides
|
||||||
@@ -260,19 +207,6 @@ export class ObjectUpdatesService {
|
|||||||
this.saveFieldUpdate(url, field, FieldChangeType.UPDATE);
|
this.saveFieldUpdate(url, field, FieldChangeType.UPDATE);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Dispatches a MoveFieldUpdateAction
|
|
||||||
* @param url The page's URL for which the changes are saved
|
|
||||||
* @param from The index of the object to move
|
|
||||||
* @param to The index to move the object to
|
|
||||||
* @param fromPage The page to move the object from
|
|
||||||
* @param toPage The page to move the object to
|
|
||||||
* @param field Optional field to add to the fieldUpdates list (useful if we want to track updates across multiple pages)
|
|
||||||
*/
|
|
||||||
saveMoveFieldUpdate(url: string, from: number, to: number, fromPage = 0, toPage = 0, field?: Identifiable) {
|
|
||||||
this.store.dispatch(new MoveFieldUpdateAction(url, from, to, fromPage, toPage, field));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Check whether the virtual metadata of a given item is selected to be saved as real metadata
|
* Check whether the virtual metadata of a given item is selected to be saved as real metadata
|
||||||
* @param url The URL of the page on which the field resides
|
* @param url The URL of the page on which the field resides
|
||||||
@@ -387,7 +321,7 @@ export class ObjectUpdatesService {
|
|||||||
* @param url The page's url to check for in the store
|
* @param url The page's url to check for in the store
|
||||||
*/
|
*/
|
||||||
hasUpdates(url: string): Observable<boolean> {
|
hasUpdates(url: string): Observable<boolean> {
|
||||||
return this.getObjectEntry(url).pipe(map((objectEntry) => hasValue(objectEntry) && (isNotEmpty(objectEntry.fieldUpdates) || objectEntry.customOrder.changed)));
|
return this.getObjectEntry(url).pipe(map((objectEntry) => hasValue(objectEntry) && isNotEmpty(objectEntry.fieldUpdates)));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -405,19 +339,4 @@ export class ObjectUpdatesService {
|
|||||||
getLastModified(url: string): Observable<Date> {
|
getLastModified(url: string): Observable<Date> {
|
||||||
return this.getObjectEntry(url).pipe(map((entry: ObjectUpdatesEntry) => entry.lastModified));
|
return this.getObjectEntry(url).pipe(map((entry: ObjectUpdatesEntry) => entry.lastModified));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Get move operations based on the custom order
|
|
||||||
* @param url The page's url
|
|
||||||
*/
|
|
||||||
getMoveOperations(url: string): Observable<MoveOperation[]> {
|
|
||||||
return this.getObjectEntry(url).pipe(
|
|
||||||
map((objectEntry) => objectEntry.customOrder),
|
|
||||||
map((customOrder) => this.comparator.diff(
|
|
||||||
flatten(customOrder.initialOrderPages.map((orderPage: OrderPage) => orderPage.order)),
|
|
||||||
flatten(customOrder.newOrderPages.map((orderPage: OrderPage) => orderPage.order)))
|
|
||||||
)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@@ -1,41 +0,0 @@
|
|||||||
import { PageInfo } from '../shared/page-info.model';
|
|
||||||
import { DSOResponseParsingService } from './dso-response-parsing.service';
|
|
||||||
import { DSpaceRESTV2Response } from '../dspace-rest-v2/dspace-rest-v2-response.model';
|
|
||||||
import {
|
|
||||||
RegistryBitstreamformatsSuccessResponse
|
|
||||||
} from '../cache/response.models';
|
|
||||||
import { RegistryBitstreamformatsResponseParsingService } from './registry-bitstreamformats-response-parsing.service';
|
|
||||||
|
|
||||||
describe('RegistryBitstreamformatsResponseParsingService', () => {
|
|
||||||
let service: RegistryBitstreamformatsResponseParsingService;
|
|
||||||
|
|
||||||
const mockDSOParser = Object.assign({
|
|
||||||
processPageInfo: () => new PageInfo()
|
|
||||||
}) as DSOResponseParsingService;
|
|
||||||
|
|
||||||
const data = Object.assign({
|
|
||||||
payload: {
|
|
||||||
_embedded: {
|
|
||||||
bitstreamformats: [
|
|
||||||
{
|
|
||||||
uuid: 'uuid-1',
|
|
||||||
description: 'a description'
|
|
||||||
},
|
|
||||||
{
|
|
||||||
uuid: 'uuid-2',
|
|
||||||
description: 'another description'
|
|
||||||
},
|
|
||||||
]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}) as DSpaceRESTV2Response;
|
|
||||||
|
|
||||||
beforeEach(() => {
|
|
||||||
service = new RegistryBitstreamformatsResponseParsingService(mockDSOParser);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should parse the data correctly', () => {
|
|
||||||
const response = service.parse(null, data);
|
|
||||||
expect(response.constructor).toBe(RegistryBitstreamformatsSuccessResponse);
|
|
||||||
});
|
|
||||||
});
|
|
@@ -1,25 +0,0 @@
|
|||||||
import { Injectable } from '@angular/core';
|
|
||||||
import { RegistryBitstreamformatsSuccessResponse, RestResponse } from '../cache/response.models';
|
|
||||||
import { DSpaceRESTV2Response } from '../dspace-rest-v2/dspace-rest-v2-response.model';
|
|
||||||
import { DSpaceSerializer } from '../dspace-rest-v2/dspace.serializer';
|
|
||||||
import { RegistryBitstreamformatsResponse } from '../registry/registry-bitstreamformats-response.model';
|
|
||||||
import { DSOResponseParsingService } from './dso-response-parsing.service';
|
|
||||||
import { ResponseParsingService } from './parsing.service';
|
|
||||||
import { RestRequest } from './request.models';
|
|
||||||
|
|
||||||
@Injectable()
|
|
||||||
export class RegistryBitstreamformatsResponseParsingService implements ResponseParsingService {
|
|
||||||
constructor(private dsoParser: DSOResponseParsingService) {
|
|
||||||
}
|
|
||||||
|
|
||||||
parse(request: RestRequest, data: DSpaceRESTV2Response): RestResponse {
|
|
||||||
const payload = data.payload;
|
|
||||||
|
|
||||||
const bitstreamformats = payload._embedded.bitstreamformats;
|
|
||||||
payload.bitstreamformats = bitstreamformats;
|
|
||||||
|
|
||||||
const deserialized = new DSpaceSerializer(RegistryBitstreamformatsResponse).deserialize(payload);
|
|
||||||
return new RegistryBitstreamformatsSuccessResponse(deserialized, data.statusCode, data.statusText, this.dsoParser.processPageInfo(data.payload.page));
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
@@ -1,68 +0,0 @@
|
|||||||
import { PageInfo } from '../shared/page-info.model';
|
|
||||||
import { DSOResponseParsingService } from './dso-response-parsing.service';
|
|
||||||
import { DSpaceRESTV2Response } from '../dspace-rest-v2/dspace-rest-v2-response.model';
|
|
||||||
import {
|
|
||||||
RegistryMetadatafieldsSuccessResponse
|
|
||||||
} from '../cache/response.models';
|
|
||||||
import { RegistryMetadatafieldsResponseParsingService } from './registry-metadatafields-response-parsing.service';
|
|
||||||
|
|
||||||
describe('RegistryMetadatafieldsResponseParsingService', () => {
|
|
||||||
let service: RegistryMetadatafieldsResponseParsingService;
|
|
||||||
|
|
||||||
const mockDSOParser = Object.assign({
|
|
||||||
processPageInfo: () => new PageInfo()
|
|
||||||
}) as DSOResponseParsingService;
|
|
||||||
|
|
||||||
const data = Object.assign({
|
|
||||||
payload: {
|
|
||||||
_embedded: {
|
|
||||||
metadatafields: [
|
|
||||||
{
|
|
||||||
id: 1,
|
|
||||||
element: 'element',
|
|
||||||
qualifier: 'qualifier',
|
|
||||||
scopeNote: 'a scope note',
|
|
||||||
_embedded: {
|
|
||||||
schema: {
|
|
||||||
id: 1,
|
|
||||||
prefix: 'test',
|
|
||||||
namespace: 'test namespace'
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
id: 2,
|
|
||||||
element: 'secondelement',
|
|
||||||
qualifier: 'secondqualifier',
|
|
||||||
scopeNote: 'a second scope note',
|
|
||||||
_embedded: {
|
|
||||||
schema: {
|
|
||||||
id: 1,
|
|
||||||
prefix: 'test',
|
|
||||||
namespace: 'test namespace'
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}) as DSpaceRESTV2Response;
|
|
||||||
|
|
||||||
const emptyData = Object.assign({
|
|
||||||
payload: {}
|
|
||||||
}) as DSpaceRESTV2Response;
|
|
||||||
|
|
||||||
beforeEach(() => {
|
|
||||||
service = new RegistryMetadatafieldsResponseParsingService(mockDSOParser);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should parse the data correctly', () => {
|
|
||||||
const response = service.parse(null, data);
|
|
||||||
expect(response.constructor).toBe(RegistryMetadatafieldsSuccessResponse);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should not produce an error and parse the data correctly when the data is empty', () => {
|
|
||||||
const response = service.parse(null, emptyData);
|
|
||||||
expect(response.constructor).toBe(RegistryMetadatafieldsSuccessResponse);
|
|
||||||
});
|
|
||||||
});
|
|
@@ -1,34 +0,0 @@
|
|||||||
import { Injectable } from '@angular/core';
|
|
||||||
import { hasValue } from '../../shared/empty.util';
|
|
||||||
import { RegistryMetadatafieldsSuccessResponse, RestResponse } from '../cache/response.models';
|
|
||||||
import { DSpaceRESTV2Response } from '../dspace-rest-v2/dspace-rest-v2-response.model';
|
|
||||||
import { DSpaceSerializer } from '../dspace-rest-v2/dspace.serializer';
|
|
||||||
import { RegistryMetadatafieldsResponse } from '../registry/registry-metadatafields-response.model';
|
|
||||||
import { DSOResponseParsingService } from './dso-response-parsing.service';
|
|
||||||
import { ResponseParsingService } from './parsing.service';
|
|
||||||
import { RestRequest } from './request.models';
|
|
||||||
|
|
||||||
@Injectable()
|
|
||||||
export class RegistryMetadatafieldsResponseParsingService implements ResponseParsingService {
|
|
||||||
constructor(private dsoParser: DSOResponseParsingService) {
|
|
||||||
}
|
|
||||||
|
|
||||||
parse(request: RestRequest, data: DSpaceRESTV2Response): RestResponse {
|
|
||||||
const payload = data.payload;
|
|
||||||
|
|
||||||
let metadatafields = [];
|
|
||||||
|
|
||||||
if (hasValue(payload._embedded)) {
|
|
||||||
metadatafields = payload._embedded.metadatafields;
|
|
||||||
metadatafields.forEach((field) => {
|
|
||||||
field.schema = field._embedded.schema;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
payload.metadatafields = metadatafields;
|
|
||||||
|
|
||||||
const deserialized = new DSpaceSerializer(RegistryMetadatafieldsResponse).deserialize(payload);
|
|
||||||
return new RegistryMetadatafieldsSuccessResponse(deserialized, data.statusCode, data.statusText, this.dsoParser.processPageInfo(data.payload));
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
@@ -1,50 +0,0 @@
|
|||||||
import { RegistryMetadataschemasResponseParsingService } from './registry-metadataschemas-response-parsing.service';
|
|
||||||
import { PageInfo } from '../shared/page-info.model';
|
|
||||||
import { DSOResponseParsingService } from './dso-response-parsing.service';
|
|
||||||
import { DSpaceRESTV2Response } from '../dspace-rest-v2/dspace-rest-v2-response.model';
|
|
||||||
import { RegistryMetadataschemasSuccessResponse } from '../cache/response.models';
|
|
||||||
|
|
||||||
describe('RegistryMetadataschemasResponseParsingService', () => {
|
|
||||||
let service: RegistryMetadataschemasResponseParsingService;
|
|
||||||
|
|
||||||
const mockDSOParser = Object.assign({
|
|
||||||
processPageInfo: () => new PageInfo()
|
|
||||||
}) as DSOResponseParsingService;
|
|
||||||
|
|
||||||
const data = Object.assign({
|
|
||||||
payload: {
|
|
||||||
_embedded: {
|
|
||||||
metadataschemas: [
|
|
||||||
{
|
|
||||||
id: 1,
|
|
||||||
prefix: 'test',
|
|
||||||
namespace: 'test namespace'
|
|
||||||
},
|
|
||||||
{
|
|
||||||
id: 2,
|
|
||||||
prefix: 'second',
|
|
||||||
namespace: 'second test namespace'
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}) as DSpaceRESTV2Response;
|
|
||||||
|
|
||||||
const emptyData = Object.assign({
|
|
||||||
payload: {}
|
|
||||||
}) as DSpaceRESTV2Response;
|
|
||||||
|
|
||||||
beforeEach(() => {
|
|
||||||
service = new RegistryMetadataschemasResponseParsingService(mockDSOParser);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should parse the data correctly', () => {
|
|
||||||
const response = service.parse(null, data);
|
|
||||||
expect(response.constructor).toBe(RegistryMetadataschemasSuccessResponse);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should not produce an error and parse the data correctly when the data is empty', () => {
|
|
||||||
const response = service.parse(null, emptyData);
|
|
||||||
expect(response.constructor).toBe(RegistryMetadataschemasSuccessResponse);
|
|
||||||
});
|
|
||||||
});
|
|
@@ -1,29 +0,0 @@
|
|||||||
import { Injectable } from '@angular/core';
|
|
||||||
import { hasValue } from '../../shared/empty.util';
|
|
||||||
import { RegistryMetadataschemasSuccessResponse, RestResponse } from '../cache/response.models';
|
|
||||||
import { DSpaceRESTV2Response } from '../dspace-rest-v2/dspace-rest-v2-response.model';
|
|
||||||
import { DSpaceSerializer } from '../dspace-rest-v2/dspace.serializer';
|
|
||||||
import { RegistryMetadataschemasResponse } from '../registry/registry-metadataschemas-response.model';
|
|
||||||
import { DSOResponseParsingService } from './dso-response-parsing.service';
|
|
||||||
import { ResponseParsingService } from './parsing.service';
|
|
||||||
import { RestRequest } from './request.models';
|
|
||||||
|
|
||||||
@Injectable()
|
|
||||||
export class RegistryMetadataschemasResponseParsingService implements ResponseParsingService {
|
|
||||||
constructor(private dsoParser: DSOResponseParsingService) {
|
|
||||||
}
|
|
||||||
|
|
||||||
parse(request: RestRequest, data: DSpaceRESTV2Response): RestResponse {
|
|
||||||
const payload = data.payload;
|
|
||||||
|
|
||||||
let metadataschemas = [];
|
|
||||||
if (hasValue(payload._embedded)) {
|
|
||||||
metadataschemas = payload._embedded.metadataschemas;
|
|
||||||
}
|
|
||||||
payload.metadataschemas = metadataschemas;
|
|
||||||
|
|
||||||
const deserialized = new DSpaceSerializer(RegistryMetadataschemasResponse).deserialize(payload);
|
|
||||||
return new RegistryMetadataschemasSuccessResponse(deserialized, data.statusCode, data.statusText, this.dsoParser.processPageInfo(data.payload));
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
@@ -14,8 +14,6 @@ import { RestRequestMethod } from './rest-request-method';
|
|||||||
import { RequestParam } from '../cache/models/request-param.model';
|
import { RequestParam } from '../cache/models/request-param.model';
|
||||||
import { EpersonResponseParsingService } from '../eperson/eperson-response-parsing.service';
|
import { EpersonResponseParsingService } from '../eperson/eperson-response-parsing.service';
|
||||||
import { BrowseItemsResponseParsingService } from './browse-items-response-parsing-service';
|
import { BrowseItemsResponseParsingService } from './browse-items-response-parsing-service';
|
||||||
import { MetadataschemaParsingService } from './metadataschema-parsing.service';
|
|
||||||
import { MetadatafieldParsingService } from './metadatafield-parsing.service';
|
|
||||||
import { URLCombiner } from '../url-combiner/url-combiner';
|
import { URLCombiner } from '../url-combiner/url-combiner';
|
||||||
import { TaskResponseParsingService } from '../tasks/task-response-parsing.service';
|
import { TaskResponseParsingService } from '../tasks/task-response-parsing.service';
|
||||||
import { ContentSourceResponseParsingService } from './content-source-response-parsing.service';
|
import { ContentSourceResponseParsingService } from './content-source-response-parsing.service';
|
||||||
@@ -251,58 +249,6 @@ export class IntegrationRequest extends GetRequest {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Request to create a MetadataSchema
|
|
||||||
*/
|
|
||||||
export class CreateMetadataSchemaRequest extends PostRequest {
|
|
||||||
constructor(uuid: string, href: string, public body?: any, public options?: HttpOptions) {
|
|
||||||
super(uuid, href, body, options);
|
|
||||||
}
|
|
||||||
|
|
||||||
getResponseParser(): GenericConstructor<ResponseParsingService> {
|
|
||||||
return MetadataschemaParsingService;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Request to update a MetadataSchema
|
|
||||||
*/
|
|
||||||
export class UpdateMetadataSchemaRequest extends PutRequest {
|
|
||||||
constructor(uuid: string, href: string, public body?: any, public options?: HttpOptions) {
|
|
||||||
super(uuid, href, body, options);
|
|
||||||
}
|
|
||||||
|
|
||||||
getResponseParser(): GenericConstructor<ResponseParsingService> {
|
|
||||||
return MetadataschemaParsingService;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Request to create a MetadataField
|
|
||||||
*/
|
|
||||||
export class CreateMetadataFieldRequest extends PostRequest {
|
|
||||||
constructor(uuid: string, href: string, public body?: any, public options?: HttpOptions) {
|
|
||||||
super(uuid, href, body, options);
|
|
||||||
}
|
|
||||||
|
|
||||||
getResponseParser(): GenericConstructor<ResponseParsingService> {
|
|
||||||
return MetadatafieldParsingService;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Request to update a MetadataField
|
|
||||||
*/
|
|
||||||
export class UpdateMetadataFieldRequest extends PutRequest {
|
|
||||||
constructor(uuid: string, href: string, public body?: any, public options?: HttpOptions) {
|
|
||||||
super(uuid, href, body, options);
|
|
||||||
}
|
|
||||||
|
|
||||||
getResponseParser(): GenericConstructor<ResponseParsingService> {
|
|
||||||
return MetadatafieldParsingService;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Class representing a submission HTTP GET request object
|
* Class representing a submission HTTP GET request object
|
||||||
*/
|
*/
|
||||||
|
@@ -201,8 +201,9 @@ export class RequestService {
|
|||||||
* Remove all request cache providing (part of) the href
|
* Remove all request cache providing (part of) the href
|
||||||
* This also includes href-to-uuid index cache
|
* This also includes href-to-uuid index cache
|
||||||
* @param href A substring of the request(s) href
|
* @param href A substring of the request(s) href
|
||||||
|
* @return Returns an observable emitting whether or not the cache is removed
|
||||||
*/
|
*/
|
||||||
removeByHrefSubstring(href: string) {
|
removeByHrefSubstring(href: string): Observable<boolean> {
|
||||||
this.store.pipe(
|
this.store.pipe(
|
||||||
select(uuidsFromHrefSubstringSelector(requestIndexSelector, href)),
|
select(uuidsFromHrefSubstringSelector(requestIndexSelector, href)),
|
||||||
take(1)
|
take(1)
|
||||||
@@ -213,6 +214,11 @@ export class RequestService {
|
|||||||
});
|
});
|
||||||
this.requestsOnTheirWayToTheStore = this.requestsOnTheirWayToTheStore.filter((reqHref: string) => reqHref.indexOf(href) < 0);
|
this.requestsOnTheirWayToTheStore = this.requestsOnTheirWayToTheStore.filter((reqHref: string) => reqHref.indexOf(href) < 0);
|
||||||
this.indexStore.dispatch(new RemoveFromIndexBySubstringAction(IndexName.REQUEST, href));
|
this.indexStore.dispatch(new RemoveFromIndexBySubstringAction(IndexName.REQUEST, href));
|
||||||
|
|
||||||
|
return this.store.pipe(
|
||||||
|
select(uuidsFromHrefSubstringSelector(requestIndexSelector, href)),
|
||||||
|
map((uuids) => isEmpty(uuids))
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@@ -1,24 +0,0 @@
|
|||||||
import { autoserialize, deserialize } from 'cerialize';
|
|
||||||
import { BITSTREAM_FORMAT } from '../shared/bitstream-format.resource-type';
|
|
||||||
import { HALLink } from '../shared/hal-link.model';
|
|
||||||
import { PageInfo } from '../shared/page-info.model';
|
|
||||||
import { BitstreamFormat } from '../shared/bitstream-format.model';
|
|
||||||
import { link } from '../cache/builders/build-decorators';
|
|
||||||
|
|
||||||
export class RegistryBitstreamformatsResponse {
|
|
||||||
@autoserialize
|
|
||||||
page: PageInfo;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The {@link HALLink}s for this RegistryBitstreamformatsResponse
|
|
||||||
*/
|
|
||||||
@deserialize
|
|
||||||
_links: {
|
|
||||||
self: HALLink;
|
|
||||||
bitstreamformats: HALLink;
|
|
||||||
};
|
|
||||||
|
|
||||||
@link(BITSTREAM_FORMAT)
|
|
||||||
bitstreamformats?: BitstreamFormat[];
|
|
||||||
|
|
||||||
}
|
|
@@ -1,46 +0,0 @@
|
|||||||
import { autoserialize, deserialize } from 'cerialize';
|
|
||||||
import { typedObject } from '../cache/builders/build-decorators';
|
|
||||||
import { MetadataField } from '../metadata/metadata-field.model';
|
|
||||||
import { METADATA_FIELD } from '../metadata/metadata-field.resource-type';
|
|
||||||
import { HALLink } from '../shared/hal-link.model';
|
|
||||||
import { PageInfo } from '../shared/page-info.model';
|
|
||||||
import { ResourceType } from '../shared/resource-type';
|
|
||||||
import { excludeFromEquals } from '../utilities/equals.decorators';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Class that represents a response with a registry's metadata fields
|
|
||||||
*/
|
|
||||||
@typedObject
|
|
||||||
export class RegistryMetadatafieldsResponse {
|
|
||||||
static type = METADATA_FIELD;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The object type
|
|
||||||
*/
|
|
||||||
@excludeFromEquals
|
|
||||||
@autoserialize
|
|
||||||
type: ResourceType;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* List of metadata fields in the response
|
|
||||||
*/
|
|
||||||
@deserialize
|
|
||||||
metadatafields: MetadataField[];
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Page info of this response
|
|
||||||
*/
|
|
||||||
@autoserialize
|
|
||||||
page: PageInfo;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The REST link to this response
|
|
||||||
*/
|
|
||||||
@autoserialize
|
|
||||||
self: string;
|
|
||||||
|
|
||||||
@deserialize
|
|
||||||
_links: {
|
|
||||||
self: HALLink,
|
|
||||||
}
|
|
||||||
}
|
|
@@ -1,14 +0,0 @@
|
|||||||
import { PageInfo } from '../shared/page-info.model';
|
|
||||||
import { autoserialize, deserialize } from 'cerialize';
|
|
||||||
import { MetadataSchema } from '../metadata/metadata-schema.model';
|
|
||||||
|
|
||||||
export class RegistryMetadataschemasResponse {
|
|
||||||
@deserialize
|
|
||||||
metadataschemas: MetadataSchema[];
|
|
||||||
|
|
||||||
@autoserialize
|
|
||||||
page: PageInfo;
|
|
||||||
|
|
||||||
@autoserialize
|
|
||||||
self: string;
|
|
||||||
}
|
|
@@ -3,8 +3,7 @@ import { Component } from '@angular/core';
|
|||||||
import { TestBed } from '@angular/core/testing';
|
import { TestBed } from '@angular/core/testing';
|
||||||
import { Store, StoreModule } from '@ngrx/store';
|
import { Store, StoreModule } from '@ngrx/store';
|
||||||
import { TranslateModule } from '@ngx-translate/core';
|
import { TranslateModule } from '@ngx-translate/core';
|
||||||
import { combineLatest as observableCombineLatest, Observable, of as observableOf } from 'rxjs';
|
import { Observable, of as observableOf } from 'rxjs';
|
||||||
import { map } from 'rxjs/operators';
|
|
||||||
import {
|
import {
|
||||||
MetadataRegistryCancelFieldAction,
|
MetadataRegistryCancelFieldAction,
|
||||||
MetadataRegistryCancelSchemaAction,
|
MetadataRegistryCancelSchemaAction,
|
||||||
@@ -17,30 +16,20 @@ import {
|
|||||||
MetadataRegistrySelectFieldAction,
|
MetadataRegistrySelectFieldAction,
|
||||||
MetadataRegistrySelectSchemaAction
|
MetadataRegistrySelectSchemaAction
|
||||||
} from '../../+admin/admin-registries/metadata-registry/metadata-registry.actions';
|
} from '../../+admin/admin-registries/metadata-registry/metadata-registry.actions';
|
||||||
import { getMockRequestService } from '../../shared/mocks/request.service.mock';
|
|
||||||
import { NotificationsService } from '../../shared/notifications/notifications.service';
|
import { NotificationsService } from '../../shared/notifications/notifications.service';
|
||||||
import { PaginationComponentOptions } from '../../shared/pagination/pagination-component-options.model';
|
|
||||||
import { StoreMock } from '../../shared/testing/store.mock';
|
import { StoreMock } from '../../shared/testing/store.mock';
|
||||||
import { NotificationsServiceStub } from '../../shared/testing/notifications-service.stub';
|
import { NotificationsServiceStub } from '../../shared/testing/notifications-service.stub';
|
||||||
import { createSuccessfulRemoteDataObject$ } from '../../shared/remote-data.utils';
|
|
||||||
import { RemoteDataBuildService } from '../cache/builders/remote-data-build.service';
|
|
||||||
|
|
||||||
import {
|
import { RestResponse } from '../cache/response.models';
|
||||||
RegistryMetadatafieldsSuccessResponse,
|
|
||||||
RegistryMetadataschemasSuccessResponse,
|
|
||||||
RestResponse
|
|
||||||
} from '../cache/response.models';
|
|
||||||
import { RemoteData } from '../data/remote-data';
|
|
||||||
import { RequestEntry } from '../data/request.reducer';
|
|
||||||
import { RequestService } from '../data/request.service';
|
|
||||||
import { MetadataField } from '../metadata/metadata-field.model';
|
import { MetadataField } from '../metadata/metadata-field.model';
|
||||||
import { MetadataSchema } from '../metadata/metadata-schema.model';
|
import { MetadataSchema } from '../metadata/metadata-schema.model';
|
||||||
import { HALEndpointService } from '../shared/hal-endpoint.service';
|
|
||||||
import { PageInfo } from '../shared/page-info.model';
|
|
||||||
import { RegistryMetadatafieldsResponse } from './registry-metadatafields-response.model';
|
|
||||||
import { RegistryMetadataschemasResponse } from './registry-metadataschemas-response.model';
|
|
||||||
import { RegistryService } from './registry.service';
|
import { RegistryService } from './registry.service';
|
||||||
import { storeModuleConfig } from '../../app.reducer';
|
import { storeModuleConfig } from '../../app.reducer';
|
||||||
|
import { FindListOptions } from '../data/request.models';
|
||||||
|
import { MetadataSchemaDataService } from '../data/metadata-schema-data.service';
|
||||||
|
import { MetadataFieldDataService } from '../data/metadata-field-data.service';
|
||||||
|
import { createSuccessfulRemoteDataObject$ } from '../../shared/remote-data.utils';
|
||||||
|
import { createPaginatedList } from '../../shared/testing/utils.test';
|
||||||
|
|
||||||
@Component({ template: '' })
|
@Component({ template: '' })
|
||||||
class DummyComponent {
|
class DummyComponent {
|
||||||
@@ -49,12 +38,20 @@ class DummyComponent {
|
|||||||
describe('RegistryService', () => {
|
describe('RegistryService', () => {
|
||||||
let registryService: RegistryService;
|
let registryService: RegistryService;
|
||||||
let mockStore;
|
let mockStore;
|
||||||
const pagination: PaginationComponentOptions = Object.assign(new PaginationComponentOptions(), {
|
let metadataSchemaService: MetadataSchemaDataService;
|
||||||
id: 'registry-service-spec-pagination',
|
let metadataFieldService: MetadataFieldDataService;
|
||||||
pageSize: 20
|
|
||||||
|
let options: FindListOptions;
|
||||||
|
let mockSchemasList: MetadataSchema[];
|
||||||
|
let mockFieldsList: MetadataField[];
|
||||||
|
|
||||||
|
function init() {
|
||||||
|
options = Object.assign(new FindListOptions(), {
|
||||||
|
currentPage: 1,
|
||||||
|
elementsPerPage: 20
|
||||||
});
|
});
|
||||||
|
|
||||||
const mockSchemasList = [
|
mockSchemasList = [
|
||||||
Object.assign(new MetadataSchema(), {
|
Object.assign(new MetadataSchema(), {
|
||||||
id: 1,
|
id: 1,
|
||||||
_links: {
|
_links: {
|
||||||
@@ -74,7 +71,8 @@ describe('RegistryService', () => {
|
|||||||
type: MetadataSchema.type
|
type: MetadataSchema.type
|
||||||
})
|
})
|
||||||
];
|
];
|
||||||
const mockFieldsList = [
|
|
||||||
|
mockFieldsList = [
|
||||||
Object.assign(new MetadataField(),
|
Object.assign(new MetadataField(),
|
||||||
{
|
{
|
||||||
id: 1,
|
id: 1,
|
||||||
@@ -125,135 +123,84 @@ describe('RegistryService', () => {
|
|||||||
})
|
})
|
||||||
];
|
];
|
||||||
|
|
||||||
const pageInfo = new PageInfo();
|
metadataSchemaService = jasmine.createSpyObj('metadataSchemaService', {
|
||||||
pageInfo.elementsPerPage = 20;
|
findAll: createSuccessfulRemoteDataObject$(createPaginatedList(mockSchemasList)),
|
||||||
pageInfo.currentPage = 1;
|
findById: createSuccessfulRemoteDataObject$(mockSchemasList[0]),
|
||||||
|
createOrUpdateMetadataSchema: createSuccessfulRemoteDataObject$(mockSchemasList[0]),
|
||||||
|
deleteAndReturnResponse: observableOf(new RestResponse(true, 200, 'OK')),
|
||||||
|
clearRequests: observableOf('href')
|
||||||
|
});
|
||||||
|
|
||||||
const endpoint = 'path';
|
metadataFieldService = jasmine.createSpyObj('metadataFieldService', {
|
||||||
const endpointWithParams = `${endpoint}?size=${pageInfo.elementsPerPage}&page=${pageInfo.currentPage - 1}`;
|
findAll: createSuccessfulRemoteDataObject$(createPaginatedList(mockFieldsList)),
|
||||||
const fieldEndpointWithParams = `${endpoint}?schema=${mockSchemasList[0].prefix}&size=${pageInfo.elementsPerPage}&page=${pageInfo.currentPage - 1}`;
|
findById: createSuccessfulRemoteDataObject$(mockFieldsList[0]),
|
||||||
|
createOrUpdateMetadataField: createSuccessfulRemoteDataObject$(mockFieldsList[0]),
|
||||||
const halServiceStub = {
|
deleteAndReturnResponse: observableOf(new RestResponse(true, 200, 'OK')),
|
||||||
getEndpoint: (link: string) => observableOf(endpoint)
|
clearRequests: observableOf('href')
|
||||||
};
|
});
|
||||||
|
|
||||||
const rdbStub = {
|
|
||||||
toRemoteDataObservable: (requestEntryObs: Observable<RequestEntry>, payloadObs: Observable<any>) => {
|
|
||||||
return observableCombineLatest(requestEntryObs,
|
|
||||||
payloadObs).pipe(map(([req, pay]) => {
|
|
||||||
return { req, pay };
|
|
||||||
})
|
|
||||||
);
|
|
||||||
},
|
|
||||||
aggregate: (input: Array<Observable<RemoteData<any>>>): Observable<RemoteData<any[]>> => {
|
|
||||||
return createSuccessfulRemoteDataObject$([]);
|
|
||||||
}
|
}
|
||||||
};
|
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
|
init();
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
imports: [CommonModule, StoreModule.forRoot({}, storeModuleConfig), TranslateModule.forRoot()],
|
imports: [CommonModule, StoreModule.forRoot({}, storeModuleConfig), TranslateModule.forRoot()],
|
||||||
declarations: [
|
declarations: [
|
||||||
DummyComponent
|
DummyComponent
|
||||||
],
|
],
|
||||||
providers: [
|
providers: [
|
||||||
{ provide: RequestService, useValue: getMockRequestService() },
|
|
||||||
{ provide: RemoteDataBuildService, useValue: rdbStub },
|
|
||||||
{ provide: HALEndpointService, useValue: halServiceStub },
|
|
||||||
{ provide: Store, useClass: StoreMock },
|
{ provide: Store, useClass: StoreMock },
|
||||||
{ provide: NotificationsService, useValue: new NotificationsServiceStub() },
|
{ provide: NotificationsService, useValue: new NotificationsServiceStub() },
|
||||||
|
{ provide: MetadataSchemaDataService, useValue: metadataSchemaService },
|
||||||
|
{ provide: MetadataFieldDataService, useValue: metadataFieldService },
|
||||||
RegistryService
|
RegistryService
|
||||||
]
|
]
|
||||||
});
|
});
|
||||||
registryService = TestBed.get(RegistryService);
|
registryService = TestBed.get(RegistryService);
|
||||||
mockStore = TestBed.get(Store);
|
mockStore = TestBed.get(Store);
|
||||||
spyOn((registryService as any).halService, 'getEndpoint').and.returnValue(observableOf(endpoint));
|
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('when requesting metadataschemas', () => {
|
describe('when requesting metadataschemas', () => {
|
||||||
const queryResponse = Object.assign(new RegistryMetadataschemasResponse(), {
|
let result;
|
||||||
metadataschemas: mockSchemasList,
|
|
||||||
page: pageInfo
|
|
||||||
});
|
|
||||||
const response = new RegistryMetadataschemasSuccessResponse(queryResponse, 200, 'OK', pageInfo);
|
|
||||||
const responseEntry = Object.assign(new RequestEntry(), { response: response });
|
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
(registryService as any).requestService.getByHref.and.returnValue(observableOf(responseEntry));
|
result = registryService.getMetadataSchemas(options);
|
||||||
/* tslint:disable:no-empty */
|
|
||||||
registryService.getMetadataSchemas(pagination).subscribe((value) => {
|
|
||||||
});
|
|
||||||
/* tslint:enable:no-empty */
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should call getEndpoint on the halService', () => {
|
it('should call metadataSchemaService.findAll', (done) => {
|
||||||
expect((registryService as any).halService.getEndpoint).toHaveBeenCalled();
|
result.subscribe(() => {
|
||||||
|
expect(metadataSchemaService.findAll).toHaveBeenCalled();
|
||||||
|
done();
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should send out the request on the request service', () => {
|
|
||||||
expect((registryService as any).requestService.configure).toHaveBeenCalled();
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should call getByHref on the request service with the correct request url', () => {
|
|
||||||
expect((registryService as any).requestService.getByHref).toHaveBeenCalledWith(endpointWithParams);
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('when requesting metadataschema by name', () => {
|
describe('when requesting metadataschema by name', () => {
|
||||||
const queryResponse = Object.assign(new RegistryMetadataschemasResponse(), {
|
let result;
|
||||||
metadataschemas: mockSchemasList,
|
|
||||||
page: pageInfo
|
|
||||||
});
|
|
||||||
const response = new RegistryMetadataschemasSuccessResponse(queryResponse, 200, 'OK', pageInfo);
|
|
||||||
const responseEntry = Object.assign(new RequestEntry(), { response: response });
|
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
(registryService as any).requestService.getByHref.and.returnValue(observableOf(responseEntry));
|
result = registryService.getMetadataSchemaByName(mockSchemasList[0].prefix);
|
||||||
/* tslint:disable:no-empty */
|
|
||||||
registryService.getMetadataSchemaByName(mockSchemasList[0].prefix).subscribe((value) => {
|
|
||||||
});
|
|
||||||
/* tslint:enable:no-empty */
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should call getEndpoint on the halService', () => {
|
it('should call metadataSchemaService.findById with the correct ID', (done) => {
|
||||||
expect((registryService as any).halService.getEndpoint).toHaveBeenCalled();
|
result.subscribe(() => {
|
||||||
|
expect(metadataSchemaService.findById).toHaveBeenCalledWith(`${mockSchemasList[0].id}`);
|
||||||
|
done();
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should send out the request on the request service', () => {
|
|
||||||
expect((registryService as any).requestService.configure).toHaveBeenCalled();
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should call getByHref on the request service with the correct request url', () => {
|
|
||||||
expect((registryService as any).requestService.getByHref.calls.argsFor(0)[0]).toContain(endpoint);
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('when requesting metadatafields', () => {
|
describe('when requesting metadatafields', () => {
|
||||||
const queryResponse = Object.assign(new RegistryMetadatafieldsResponse(), {
|
let result;
|
||||||
metadatafields: mockFieldsList,
|
|
||||||
page: pageInfo
|
|
||||||
});
|
|
||||||
const response = new RegistryMetadatafieldsSuccessResponse(queryResponse, 200, 'OK', pageInfo);
|
|
||||||
const responseEntry = Object.assign(new RequestEntry(), { response: response });
|
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
(registryService as any).requestService.getByHref.and.returnValue(observableOf(responseEntry));
|
result = registryService.getAllMetadataFields();
|
||||||
/* tslint:disable:no-empty */
|
|
||||||
registryService.getMetadataFieldsBySchema(mockSchemasList[0], pagination).subscribe((value) => {
|
|
||||||
});
|
|
||||||
/* tslint:enable:no-empty */
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should call getEndpoint on the halService', () => {
|
it('should call metadataFieldService.findAll', (done) => {
|
||||||
expect((registryService as any).halService.getEndpoint).toHaveBeenCalled();
|
result.subscribe(() => {
|
||||||
|
expect(metadataFieldService.findAll).toHaveBeenCalled();
|
||||||
|
done();
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should send out the request on the request service', () => {
|
|
||||||
expect((registryService as any).requestService.configure).toHaveBeenCalled();
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should call getByHref on the request service with the correct request url', () => {
|
|
||||||
expect((registryService as any).requestService.getByHref).toHaveBeenCalledWith(fieldEndpointWithParams);
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -370,9 +317,10 @@ describe('RegistryService', () => {
|
|||||||
result = registryService.createOrUpdateMetadataSchema(mockSchemasList[0]);
|
result = registryService.createOrUpdateMetadataSchema(mockSchemasList[0]);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should return the created/updated metadata schema', () => {
|
it('should return the created/updated metadata schema', (done) => {
|
||||||
result.subscribe((schema: MetadataSchema) => {
|
result.subscribe((schema: MetadataSchema) => {
|
||||||
expect(schema).toEqual(mockSchemasList[0]);
|
expect(schema).toEqual(mockSchemasList[0]);
|
||||||
|
done();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -384,9 +332,10 @@ describe('RegistryService', () => {
|
|||||||
result = registryService.createOrUpdateMetadataField(mockFieldsList[0]);
|
result = registryService.createOrUpdateMetadataField(mockFieldsList[0]);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should return the created/updated metadata field', () => {
|
it('should return the created/updated metadata field', (done) => {
|
||||||
result.subscribe((field: MetadataField) => {
|
result.subscribe((field: MetadataField) => {
|
||||||
expect(field).toEqual(mockFieldsList[0]);
|
expect(field).toEqual(mockFieldsList[0]);
|
||||||
|
done();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -425,7 +374,7 @@ describe('RegistryService', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('should remove the requests related to metadata schemas from cache', () => {
|
it('should remove the requests related to metadata schemas from cache', () => {
|
||||||
expect((registryService as any).requestService.removeByHrefSubstring).toHaveBeenCalled();
|
expect(metadataSchemaService.clearRequests).toHaveBeenCalled();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -435,7 +384,7 @@ describe('RegistryService', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('should remove the requests related to metadata fields from cache', () => {
|
it('should remove the requests related to metadata fields from cache', () => {
|
||||||
expect((registryService as any).requestService.removeByHrefSubstring).toHaveBeenCalled();
|
expect(metadataFieldService.clearRequests).toHaveBeenCalled();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@@ -2,37 +2,18 @@ import { combineLatest as observableCombineLatest, Observable } from 'rxjs';
|
|||||||
import { Injectable } from '@angular/core';
|
import { Injectable } from '@angular/core';
|
||||||
import { RemoteData } from '../data/remote-data';
|
import { RemoteData } from '../data/remote-data';
|
||||||
import { PaginatedList } from '../data/paginated-list';
|
import { PaginatedList } from '../data/paginated-list';
|
||||||
import { DSpaceSerializer } from '../dspace-rest-v2/dspace.serializer';
|
|
||||||
import { PageInfo } from '../shared/page-info.model';
|
import { PageInfo } from '../shared/page-info.model';
|
||||||
import {
|
import { FindListOptions } from '../data/request.models';
|
||||||
CreateMetadataFieldRequest,
|
|
||||||
CreateMetadataSchemaRequest,
|
|
||||||
DeleteRequest,
|
|
||||||
GetRequest,
|
|
||||||
RestRequest,
|
|
||||||
UpdateMetadataFieldRequest,
|
|
||||||
UpdateMetadataSchemaRequest
|
|
||||||
} from '../data/request.models';
|
|
||||||
import { GenericConstructor } from '../shared/generic-constructor';
|
|
||||||
import { ResponseParsingService } from '../data/parsing.service';
|
|
||||||
import { RegistryMetadataschemasResponseParsingService } from '../data/registry-metadataschemas-response-parsing.service';
|
|
||||||
import { RemoteDataBuildService } from '../cache/builders/remote-data-build.service';
|
import { RemoteDataBuildService } from '../cache/builders/remote-data-build.service';
|
||||||
import { RequestService } from '../data/request.service';
|
import { RequestService } from '../data/request.service';
|
||||||
import { RegistryMetadataschemasResponse } from './registry-metadataschemas-response.model';
|
|
||||||
import {
|
import {
|
||||||
MetadatafieldSuccessResponse,
|
MetadatafieldSuccessResponse,
|
||||||
MetadataschemaSuccessResponse,
|
MetadataschemaSuccessResponse,
|
||||||
RegistryMetadatafieldsSuccessResponse,
|
|
||||||
RegistryMetadataschemasSuccessResponse,
|
|
||||||
RestResponse
|
RestResponse
|
||||||
} from '../cache/response.models';
|
} from '../cache/response.models';
|
||||||
import { HALEndpointService } from '../shared/hal-endpoint.service';
|
import { HALEndpointService } from '../shared/hal-endpoint.service';
|
||||||
import { RegistryMetadatafieldsResponseParsingService } from '../data/registry-metadatafields-response-parsing.service';
|
import { hasNoValue, hasValue, hasValueOperator, isNotEmpty, isNotEmptyOperator } from '../../shared/empty.util';
|
||||||
import { RegistryMetadatafieldsResponse } from './registry-metadatafields-response.model';
|
import { getAllSucceededRemoteDataPayload, getFirstSucceededRemoteDataPayload } from '../shared/operators';
|
||||||
import { hasNoValue, hasValue, isNotEmpty, isNotEmptyOperator } from '../../shared/empty.util';
|
|
||||||
import { URLCombiner } from '../url-combiner/url-combiner';
|
|
||||||
import { PaginationComponentOptions } from '../../shared/pagination/pagination-component-options.model';
|
|
||||||
import { configureRequest, getResponseFromEntry } from '../shared/operators';
|
|
||||||
import { createSelector, select, Store } from '@ngrx/store';
|
import { createSelector, select, Store } from '@ngrx/store';
|
||||||
import { AppState } from '../../app.reducer';
|
import { AppState } from '../../app.reducer';
|
||||||
import { MetadataRegistryState } from '../../+admin/admin-registries/metadata-registry/metadata-registry.reducers';
|
import { MetadataRegistryState } from '../../+admin/admin-registries/metadata-registry/metadata-registry.reducers';
|
||||||
@@ -48,15 +29,14 @@ import {
|
|||||||
MetadataRegistrySelectFieldAction,
|
MetadataRegistrySelectFieldAction,
|
||||||
MetadataRegistrySelectSchemaAction
|
MetadataRegistrySelectSchemaAction
|
||||||
} from '../../+admin/admin-registries/metadata-registry/metadata-registry.actions';
|
} from '../../+admin/admin-registries/metadata-registry/metadata-registry.actions';
|
||||||
import { distinctUntilChanged, flatMap, map, take, tap } from 'rxjs/operators';
|
import { flatMap, map, tap } from 'rxjs/operators';
|
||||||
import { NotificationsService } from '../../shared/notifications/notifications.service';
|
import { NotificationsService } from '../../shared/notifications/notifications.service';
|
||||||
import { NotificationOptions } from '../../shared/notifications/models/notification-options.model';
|
|
||||||
import { HttpOptions } from '../dspace-rest-v2/dspace-rest-v2.service';
|
|
||||||
import { HttpHeaders } from '@angular/common/http';
|
|
||||||
import { TranslateService } from '@ngx-translate/core';
|
import { TranslateService } from '@ngx-translate/core';
|
||||||
import { MetadataSchema } from '../metadata/metadata-schema.model';
|
import { MetadataSchema } from '../metadata/metadata-schema.model';
|
||||||
import { MetadataField } from '../metadata/metadata-field.model';
|
import { MetadataField } from '../metadata/metadata-field.model';
|
||||||
import { getClassForType } from '../cache/builders/build-decorators';
|
import { MetadataSchemaDataService } from '../data/metadata-schema-data.service';
|
||||||
|
import { MetadataFieldDataService } from '../data/metadata-field-data.service';
|
||||||
|
import { FollowLinkConfig } from '../../shared/utils/follow-link-config.model';
|
||||||
|
|
||||||
const metadataRegistryStateSelector = (state: AppState) => state.metadataRegistry;
|
const metadataRegistryStateSelector = (state: AppState) => state.metadataRegistry;
|
||||||
const editMetadataSchemaSelector = createSelector(metadataRegistryStateSelector, (metadataState: MetadataRegistryState) => metadataState.editSchema);
|
const editMetadataSchemaSelector = createSelector(metadataRegistryStateSelector, (metadataState: MetadataRegistryState) => metadataState.editSchema);
|
||||||
@@ -70,221 +50,64 @@ const selectedMetadataFieldsSelector = createSelector(metadataRegistryStateSelec
|
|||||||
@Injectable()
|
@Injectable()
|
||||||
export class RegistryService {
|
export class RegistryService {
|
||||||
|
|
||||||
private metadataSchemasPath = 'metadataschemas';
|
constructor(private store: Store<AppState>,
|
||||||
private metadataFieldsPath = 'metadatafields';
|
|
||||||
|
|
||||||
// private bitstreamFormatsPath = 'bitstreamformats';
|
|
||||||
|
|
||||||
constructor(protected requestService: RequestService,
|
|
||||||
private rdb: RemoteDataBuildService,
|
|
||||||
private halService: HALEndpointService,
|
|
||||||
private store: Store<AppState>,
|
|
||||||
private notificationsService: NotificationsService,
|
private notificationsService: NotificationsService,
|
||||||
private translateService: TranslateService) {
|
private translateService: TranslateService,
|
||||||
|
private metadataSchemaService: MetadataSchemaDataService,
|
||||||
|
private metadataFieldService: MetadataFieldDataService) {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Retrieves all metadata schemas
|
* Retrieves all metadata schemas
|
||||||
* @param pagination The pagination info used to retrieve the schemas
|
* @param options The options used to retrieve the schemas
|
||||||
|
* @param linksToFollow List of {@link FollowLinkConfig} that indicate which {@link HALLink}s should be automatically resolved
|
||||||
*/
|
*/
|
||||||
public getMetadataSchemas(pagination: PaginationComponentOptions): Observable<RemoteData<PaginatedList<MetadataSchema>>> {
|
public getMetadataSchemas(options: FindListOptions = {}, ...linksToFollow: Array<FollowLinkConfig<MetadataSchema>>): Observable<RemoteData<PaginatedList<MetadataSchema>>> {
|
||||||
const requestObs = this.getMetadataSchemasRequestObs(pagination);
|
return this.metadataSchemaService.findAll(options, ...linksToFollow);
|
||||||
|
|
||||||
const requestEntryObs = requestObs.pipe(
|
|
||||||
flatMap((request: RestRequest) => this.requestService.getByHref(request.href))
|
|
||||||
);
|
|
||||||
|
|
||||||
const rmrObs: Observable<RegistryMetadataschemasResponse> = requestEntryObs.pipe(
|
|
||||||
getResponseFromEntry(),
|
|
||||||
map((response: RegistryMetadataschemasSuccessResponse) => response.metadataschemasResponse)
|
|
||||||
);
|
|
||||||
|
|
||||||
const metadataschemasObs: Observable<MetadataSchema[]> = rmrObs.pipe(
|
|
||||||
map((rmr: RegistryMetadataschemasResponse) => rmr.metadataschemas)
|
|
||||||
);
|
|
||||||
|
|
||||||
const pageInfoObs: Observable<PageInfo> = requestEntryObs.pipe(
|
|
||||||
getResponseFromEntry(),
|
|
||||||
map((response: RegistryMetadataschemasSuccessResponse) => response.pageInfo)
|
|
||||||
);
|
|
||||||
|
|
||||||
const payloadObs = observableCombineLatest(metadataschemasObs, pageInfoObs).pipe(
|
|
||||||
map(([metadataschemas, pageInfo]) => {
|
|
||||||
return new PaginatedList(pageInfo, metadataschemas);
|
|
||||||
})
|
|
||||||
);
|
|
||||||
|
|
||||||
return this.rdb.toRemoteDataObservable(requestEntryObs, payloadObs);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Retrieves a metadata schema by its name
|
* Retrieves a metadata schema by its name
|
||||||
* @param schemaName The name of the schema to find
|
* @param schemaName The name of the schema to find
|
||||||
|
* @param linksToFollow List of {@link FollowLinkConfig} that indicate which {@link HALLink}s should be automatically resolved
|
||||||
*/
|
*/
|
||||||
public getMetadataSchemaByName(schemaName: string): Observable<RemoteData<MetadataSchema>> {
|
public getMetadataSchemaByName(schemaName: string, ...linksToFollow: Array<FollowLinkConfig<MetadataSchema>>): Observable<RemoteData<MetadataSchema>> {
|
||||||
// Temporary pagination to get ALL metadataschemas until there's a rest api endpoint for fetching a specific schema
|
// Temporary options to get ALL metadataschemas until there's a rest api endpoint for fetching a specific schema
|
||||||
const pagination: PaginationComponentOptions = Object.assign(new PaginationComponentOptions(), {
|
const options: FindListOptions = Object.assign(new FindListOptions(), {
|
||||||
id: 'all-metadatafields-pagination',
|
elementsPerPage: 10000
|
||||||
pageSize: 10000
|
|
||||||
});
|
});
|
||||||
const requestObs = this.getMetadataSchemasRequestObs(pagination);
|
return this.getMetadataSchemas(options).pipe(
|
||||||
|
getFirstSucceededRemoteDataPayload(),
|
||||||
const requestEntryObs = requestObs.pipe(
|
map((schemas: PaginatedList<MetadataSchema>) => schemas.page),
|
||||||
flatMap((request: RestRequest) => this.requestService.getByHref(request.href))
|
isNotEmptyOperator(),
|
||||||
|
map((schemas: MetadataSchema[]) => schemas.filter((schema) => schema.prefix === schemaName)[0]),
|
||||||
|
flatMap((schema: MetadataSchema) => this.metadataSchemaService.findById(`${schema.id}`, ...linksToFollow))
|
||||||
);
|
);
|
||||||
|
|
||||||
const rmrObs: Observable<RegistryMetadataschemasResponse> = requestEntryObs.pipe(
|
|
||||||
getResponseFromEntry(),
|
|
||||||
map((response: RegistryMetadataschemasSuccessResponse) => response.metadataschemasResponse)
|
|
||||||
);
|
|
||||||
|
|
||||||
const metadataschemaObs: Observable<MetadataSchema> = rmrObs.pipe(
|
|
||||||
map((rmr: RegistryMetadataschemasResponse) => rmr.metadataschemas),
|
|
||||||
map((metadataSchemas: MetadataSchema[]) => metadataSchemas.filter((value) => value.prefix === schemaName)[0])
|
|
||||||
);
|
|
||||||
|
|
||||||
return this.rdb.toRemoteDataObservable(requestEntryObs, metadataschemaObs);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* retrieves all metadata fields that belong to a certain metadata schema
|
* retrieves all metadata fields that belong to a certain metadata schema
|
||||||
* @param schema The schema to filter by
|
* @param schema The schema to filter by
|
||||||
* @param pagination The pagination info used to retrieve the fields
|
* @param options The options info used to retrieve the fields
|
||||||
|
* @param linksToFollow List of {@link FollowLinkConfig} that indicate which {@link HALLink}s should be automatically resolved
|
||||||
*/
|
*/
|
||||||
public getMetadataFieldsBySchema(schema: MetadataSchema, pagination: PaginationComponentOptions): Observable<RemoteData<PaginatedList<MetadataField>>> {
|
public getMetadataFieldsBySchema(schema: MetadataSchema, options: FindListOptions = {}, ...linksToFollow: Array<FollowLinkConfig<MetadataField>>): Observable<RemoteData<PaginatedList<MetadataField>>> {
|
||||||
const requestObs = this.getMetadataFieldsBySchemaRequestObs(pagination, schema);
|
return this.metadataFieldService.findBySchema(schema, options, ...linksToFollow);
|
||||||
|
|
||||||
const requestEntryObs = requestObs.pipe(
|
|
||||||
flatMap((request: RestRequest) => this.requestService.getByHref(request.href))
|
|
||||||
);
|
|
||||||
|
|
||||||
const rmrObs: Observable<RegistryMetadatafieldsResponse> = requestEntryObs.pipe(
|
|
||||||
getResponseFromEntry(),
|
|
||||||
map((response: RegistryMetadatafieldsSuccessResponse) => response.metadatafieldsResponse)
|
|
||||||
);
|
|
||||||
|
|
||||||
const metadatafieldsObs: Observable<MetadataField[]> = rmrObs.pipe(
|
|
||||||
map((rmr: RegistryMetadatafieldsResponse) => rmr.metadatafields)
|
|
||||||
);
|
|
||||||
|
|
||||||
const pageInfoObs: Observable<PageInfo> = requestEntryObs.pipe(
|
|
||||||
getResponseFromEntry(),
|
|
||||||
|
|
||||||
map((response: RegistryMetadatafieldsSuccessResponse) => response.pageInfo)
|
|
||||||
);
|
|
||||||
|
|
||||||
const payloadObs = observableCombineLatest(metadatafieldsObs, pageInfoObs).pipe(
|
|
||||||
map(([metadatafields, pageInfo]) => {
|
|
||||||
return new PaginatedList(pageInfo, metadatafields);
|
|
||||||
})
|
|
||||||
);
|
|
||||||
|
|
||||||
return this.rdb.toRemoteDataObservable(requestEntryObs, payloadObs);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Retrieve all existing metadata fields as a paginated list
|
* Retrieve all existing metadata fields as a paginated list
|
||||||
* @param pagination Pagination options to determine which page of metadata fields should be requested
|
* @param options Options to determine which page of metadata fields should be requested
|
||||||
* When no pagination is provided, all metadata fields are requested in one large page
|
* When no options are provided, all metadata fields are requested in one large page
|
||||||
|
* @param linksToFollow List of {@link FollowLinkConfig} that indicate which {@link HALLink}s should be automatically resolved
|
||||||
* @returns an observable that emits a remote data object with a page of metadata fields
|
* @returns an observable that emits a remote data object with a page of metadata fields
|
||||||
*/
|
*/
|
||||||
public getAllMetadataFields(pagination?: PaginationComponentOptions): Observable<RemoteData<PaginatedList<MetadataField>>> {
|
public getAllMetadataFields(options?: FindListOptions, ...linksToFollow: Array<FollowLinkConfig<MetadataField>>): Observable<RemoteData<PaginatedList<MetadataField>>> {
|
||||||
if (hasNoValue(pagination)) {
|
if (hasNoValue(options)) {
|
||||||
pagination = {currentPage: 1, pageSize: 10000} as any;
|
options = {currentPage: 1, elementsPerPage: 10000} as any;
|
||||||
}
|
}
|
||||||
const requestObs = this.getMetadataFieldsRequestObs(pagination);
|
return this.metadataFieldService.findAll(options, ...linksToFollow);
|
||||||
|
|
||||||
const requestEntryObs = requestObs.pipe(
|
|
||||||
flatMap((request: RestRequest) => this.requestService.getByHref(request.href))
|
|
||||||
);
|
|
||||||
|
|
||||||
const rmrObs: Observable<RegistryMetadatafieldsResponse> = requestEntryObs.pipe(
|
|
||||||
getResponseFromEntry(),
|
|
||||||
map((response: RegistryMetadatafieldsSuccessResponse) => response.metadatafieldsResponse)
|
|
||||||
);
|
|
||||||
|
|
||||||
const metadatafieldsObs: Observable<MetadataField[]> = rmrObs.pipe(
|
|
||||||
map((rmr: RegistryMetadatafieldsResponse) => rmr.metadatafields),
|
|
||||||
/* Make sure to explicitly cast this into a MetadataField object, on first page loads this object comes from the object cache created by the server and its prototype is unknown */
|
|
||||||
map((metadataFields: MetadataField[]) => metadataFields.map((metadataField: MetadataField) => Object.assign(new MetadataField(), metadataField)))
|
|
||||||
);
|
|
||||||
|
|
||||||
const pageInfoObs: Observable<PageInfo> = requestEntryObs.pipe(
|
|
||||||
getResponseFromEntry(),
|
|
||||||
|
|
||||||
map((response: RegistryMetadatafieldsSuccessResponse) => response.pageInfo)
|
|
||||||
);
|
|
||||||
|
|
||||||
const payloadObs = observableCombineLatest(metadatafieldsObs, pageInfoObs).pipe(
|
|
||||||
map(([metadatafields, pageInfo]) => {
|
|
||||||
return new PaginatedList(pageInfo, metadatafields);
|
|
||||||
})
|
|
||||||
);
|
|
||||||
|
|
||||||
return this.rdb.toRemoteDataObservable(requestEntryObs, payloadObs);
|
|
||||||
}
|
|
||||||
|
|
||||||
public getMetadataSchemasRequestObs(pagination: PaginationComponentOptions): Observable<RestRequest> {
|
|
||||||
return this.halService.getEndpoint(this.metadataSchemasPath).pipe(
|
|
||||||
map((url: string) => {
|
|
||||||
const args: string[] = [];
|
|
||||||
args.push(`size=${pagination.pageSize}`);
|
|
||||||
args.push(`page=${pagination.currentPage - 1}`);
|
|
||||||
if (isNotEmpty(args)) {
|
|
||||||
url = new URLCombiner(url, `?${args.join('&')}`).toString();
|
|
||||||
}
|
|
||||||
const request = new GetRequest(this.requestService.generateRequestId(), url);
|
|
||||||
return Object.assign(request, {
|
|
||||||
getResponseParser(): GenericConstructor<ResponseParsingService> {
|
|
||||||
return RegistryMetadataschemasResponseParsingService;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}),
|
|
||||||
tap((request: RestRequest) => this.requestService.configure(request)),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
private getMetadataFieldsBySchemaRequestObs(pagination: PaginationComponentOptions, schema: MetadataSchema): Observable<RestRequest> {
|
|
||||||
return this.halService.getEndpoint(this.metadataFieldsPath + '/search/bySchema').pipe(
|
|
||||||
// return this.halService.getEndpoint(this.metadataFieldsPath).pipe(
|
|
||||||
map((url: string) => {
|
|
||||||
const args: string[] = [];
|
|
||||||
args.push(`schema=${schema.prefix}`);
|
|
||||||
args.push(`size=${pagination.pageSize}`);
|
|
||||||
args.push(`page=${pagination.currentPage - 1}`);
|
|
||||||
if (isNotEmpty(args)) {
|
|
||||||
url = new URLCombiner(url, `?${args.join('&')}`).toString();
|
|
||||||
}
|
|
||||||
const request = new GetRequest(this.requestService.generateRequestId(), url);
|
|
||||||
return Object.assign(request, {
|
|
||||||
getResponseParser(): GenericConstructor<ResponseParsingService> {
|
|
||||||
return RegistryMetadatafieldsResponseParsingService;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}),
|
|
||||||
tap((request: RestRequest) => this.requestService.configure(request)),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
private getMetadataFieldsRequestObs(pagination: PaginationComponentOptions): Observable<RestRequest> {
|
|
||||||
return this.halService.getEndpoint(this.metadataFieldsPath).pipe(
|
|
||||||
map((url: string) => {
|
|
||||||
const args: string[] = [];
|
|
||||||
args.push(`size=${pagination.pageSize}`);
|
|
||||||
args.push(`page=${pagination.currentPage - 1}`);
|
|
||||||
if (isNotEmpty(args)) {
|
|
||||||
url = new URLCombiner(url, `?${args.join('&')}`).toString();
|
|
||||||
}
|
|
||||||
const request = new GetRequest(this.requestService.generateRequestId(), url);
|
|
||||||
return Object.assign(request, {
|
|
||||||
getResponseParser(): GenericConstructor<ResponseParsingService> {
|
|
||||||
return RegistryMetadatafieldsResponseParsingService;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}),
|
|
||||||
tap((request: RestRequest) => this.requestService.configure(request)),
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public editMetadataSchema(schema: MetadataSchema) {
|
public editMetadataSchema(schema: MetadataSchema) {
|
||||||
@@ -386,59 +209,17 @@ export class RegistryService {
|
|||||||
* Create or Update a MetadataSchema
|
* Create or Update a MetadataSchema
|
||||||
* If the MetadataSchema contains an id, it is assumed the schema already exists and is updated instead
|
* If the MetadataSchema contains an id, it is assumed the schema already exists and is updated instead
|
||||||
* Since creating or updating is nearly identical, the only real difference is the request (and slight difference in endpoint):
|
* Since creating or updating is nearly identical, the only real difference is the request (and slight difference in endpoint):
|
||||||
* - On creation, a CreateMetadataSchemaRequest is used
|
* - On creation, a CreateRequest is used
|
||||||
* - On update, a UpdateMetadataSchemaRequest is used
|
* - On update, a PutRequest is used
|
||||||
* @param schema The MetadataSchema to create or update
|
* @param schema The MetadataSchema to create or update
|
||||||
*/
|
*/
|
||||||
public createOrUpdateMetadataSchema(schema: MetadataSchema): Observable<MetadataSchema> {
|
public createOrUpdateMetadataSchema(schema: MetadataSchema): Observable<MetadataSchema> {
|
||||||
const isUpdate = hasValue(schema.id);
|
const isUpdate = hasValue(schema.id);
|
||||||
const requestId = this.requestService.generateRequestId();
|
return this.metadataSchemaService.createOrUpdateMetadataSchema(schema).pipe(
|
||||||
const endpoint$ = this.halService.getEndpoint(this.metadataSchemasPath).pipe(
|
getFirstSucceededRemoteDataPayload(),
|
||||||
isNotEmptyOperator(),
|
hasValueOperator(),
|
||||||
map((endpoint: string) => (isUpdate ? `${endpoint}/${schema.id}` : endpoint)),
|
tap(() => {
|
||||||
distinctUntilChanged()
|
|
||||||
);
|
|
||||||
|
|
||||||
const serializedSchema = new DSpaceSerializer(getClassForType(MetadataSchema.type)).serialize(schema);
|
|
||||||
|
|
||||||
const request$ = endpoint$.pipe(
|
|
||||||
take(1),
|
|
||||||
map((endpoint: string) => {
|
|
||||||
if (isUpdate) {
|
|
||||||
const options: HttpOptions = Object.create({});
|
|
||||||
let headers = new HttpHeaders();
|
|
||||||
headers = headers.append('Content-Type', 'application/json');
|
|
||||||
options.headers = headers;
|
|
||||||
return new UpdateMetadataSchemaRequest(requestId, endpoint, JSON.stringify(serializedSchema), options);
|
|
||||||
} else {
|
|
||||||
return new CreateMetadataSchemaRequest(requestId, endpoint, JSON.stringify(serializedSchema));
|
|
||||||
}
|
|
||||||
})
|
|
||||||
);
|
|
||||||
|
|
||||||
// Execute the post/put request
|
|
||||||
request$.pipe(
|
|
||||||
configureRequest(this.requestService)
|
|
||||||
).subscribe();
|
|
||||||
|
|
||||||
// Return created/updated schema
|
|
||||||
return this.requestService.getByUUID(requestId).pipe(
|
|
||||||
getResponseFromEntry(),
|
|
||||||
map((response: RestResponse) => {
|
|
||||||
if (!response.isSuccessful) {
|
|
||||||
if (hasValue((response as any).errorMessage)) {
|
|
||||||
this.notificationsService.error('Server Error:', (response as any).errorMessage, new NotificationOptions(-1));
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
this.showNotifications(true, isUpdate, false, {prefix: schema.prefix});
|
this.showNotifications(true, isUpdate, false, {prefix: schema.prefix});
|
||||||
return response;
|
|
||||||
}
|
|
||||||
}),
|
|
||||||
isNotEmptyOperator(),
|
|
||||||
map((response: MetadataschemaSuccessResponse) => {
|
|
||||||
if (isNotEmpty(response.metadataschema)) {
|
|
||||||
return response.metadataschema;
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -448,74 +229,32 @@ export class RegistryService {
|
|||||||
* @param id The id of the metadata schema to delete
|
* @param id The id of the metadata schema to delete
|
||||||
*/
|
*/
|
||||||
public deleteMetadataSchema(id: number): Observable<RestResponse> {
|
public deleteMetadataSchema(id: number): Observable<RestResponse> {
|
||||||
return this.delete(this.metadataSchemasPath, id);
|
return this.metadataSchemaService.deleteAndReturnResponse(`${id}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Method that clears a cached metadata schema request and returns its REST url
|
* Method that clears a cached metadata schema request and returns its REST url
|
||||||
*/
|
*/
|
||||||
public clearMetadataSchemaRequests(): Observable<string> {
|
public clearMetadataSchemaRequests(): Observable<string> {
|
||||||
return this.halService.getEndpoint(this.metadataSchemasPath).pipe(
|
return this.metadataSchemaService.clearRequests();
|
||||||
tap((href: string) => this.requestService.removeByHrefSubstring(href))
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create or Update a MetadataField
|
* Create or Update a MetadataField
|
||||||
* If the MetadataField contains an id, it is assumed the field already exists and is updated instead
|
* If the MetadataField contains an id, it is assumed the field already exists and is updated instead
|
||||||
* Since creating or updating is nearly identical, the only real difference is the request (and slight difference in endpoint):
|
* Since creating or updating is nearly identical, the only real difference is the request (and slight difference in endpoint):
|
||||||
* - On creation, a CreateMetadataFieldRequest is used
|
* - On creation, a CreateRequest is used
|
||||||
* - On update, a UpdateMetadataFieldRequest is used
|
* - On update, a PutRequest is used
|
||||||
* @param field The MetadataField to create or update
|
* @param field The MetadataField to create or update
|
||||||
*/
|
*/
|
||||||
public createOrUpdateMetadataField(field: MetadataField): Observable<MetadataField> {
|
public createOrUpdateMetadataField(field: MetadataField): Observable<MetadataField> {
|
||||||
const isUpdate = hasValue(field.id);
|
const isUpdate = hasValue(field.id);
|
||||||
const requestId = this.requestService.generateRequestId();
|
return this.metadataFieldService.createOrUpdateMetadataField(field).pipe(
|
||||||
const endpoint$ = this.halService.getEndpoint(this.metadataFieldsPath).pipe(
|
getFirstSucceededRemoteDataPayload(),
|
||||||
isNotEmptyOperator(),
|
hasValueOperator(),
|
||||||
map((endpoint: string) => (isUpdate ? `${endpoint}/${field.id}` : `${endpoint}?schemaId=${field.schema.id}`)),
|
tap(() => {
|
||||||
distinctUntilChanged()
|
|
||||||
);
|
|
||||||
|
|
||||||
const request$ = endpoint$.pipe(
|
|
||||||
take(1),
|
|
||||||
map((endpoint: string) => {
|
|
||||||
if (isUpdate) {
|
|
||||||
const options: HttpOptions = Object.create({});
|
|
||||||
let headers = new HttpHeaders();
|
|
||||||
headers = headers.append('Content-Type', 'application/json');
|
|
||||||
options.headers = headers;
|
|
||||||
return new UpdateMetadataFieldRequest(requestId, endpoint, JSON.stringify(field), options);
|
|
||||||
} else {
|
|
||||||
return new CreateMetadataFieldRequest(requestId, endpoint, JSON.stringify(field));
|
|
||||||
}
|
|
||||||
})
|
|
||||||
);
|
|
||||||
|
|
||||||
// Execute the post/put request
|
|
||||||
request$.pipe(
|
|
||||||
configureRequest(this.requestService)
|
|
||||||
).subscribe();
|
|
||||||
|
|
||||||
// Return created/updated field
|
|
||||||
return this.requestService.getByUUID(requestId).pipe(
|
|
||||||
getResponseFromEntry(),
|
|
||||||
map((response: RestResponse) => {
|
|
||||||
if (!response.isSuccessful) {
|
|
||||||
if (hasValue((response as any).errorMessage)) {
|
|
||||||
this.notificationsService.error('Server Error:', (response as any).errorMessage, new NotificationOptions(-1));
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
const fieldString = `${field.schema.prefix}.${field.element}${field.qualifier ? `.${field.qualifier}` : ''}`;
|
const fieldString = `${field.schema.prefix}.${field.element}${field.qualifier ? `.${field.qualifier}` : ''}`;
|
||||||
this.showNotifications(true, isUpdate, true, {field: fieldString});
|
this.showNotifications(true, isUpdate, true, {field: fieldString});
|
||||||
return response;
|
|
||||||
}
|
|
||||||
}),
|
|
||||||
isNotEmptyOperator(),
|
|
||||||
map((response: MetadatafieldSuccessResponse) => {
|
|
||||||
if (isNotEmpty(response.metadatafield)) {
|
|
||||||
return response.metadatafield;
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -525,38 +264,13 @@ export class RegistryService {
|
|||||||
* @param id The id of the metadata field to delete
|
* @param id The id of the metadata field to delete
|
||||||
*/
|
*/
|
||||||
public deleteMetadataField(id: number): Observable<RestResponse> {
|
public deleteMetadataField(id: number): Observable<RestResponse> {
|
||||||
return this.delete(this.metadataFieldsPath, id);
|
return this.metadataFieldService.deleteAndReturnResponse(`${id}`);
|
||||||
}
|
}
|
||||||
/**
|
/**
|
||||||
* Method that clears a cached metadata field request and returns its REST url
|
* Method that clears a cached metadata field request and returns its REST url
|
||||||
*/
|
*/
|
||||||
public clearMetadataFieldRequests(): Observable<string> {
|
public clearMetadataFieldRequests(): Observable<string> {
|
||||||
return this.halService.getEndpoint(this.metadataFieldsPath).pipe(
|
return this.metadataFieldService.clearRequests();
|
||||||
tap((href: string) => this.requestService.removeByHrefSubstring(href))
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
private delete(path: string, id: number): Observable<RestResponse> {
|
|
||||||
const requestId = this.requestService.generateRequestId();
|
|
||||||
const endpoint$ = this.halService.getEndpoint(path).pipe(
|
|
||||||
isNotEmptyOperator(),
|
|
||||||
map((endpoint: string) => `${endpoint}/${id}`),
|
|
||||||
distinctUntilChanged()
|
|
||||||
);
|
|
||||||
|
|
||||||
const request$ = endpoint$.pipe(
|
|
||||||
take(1),
|
|
||||||
map((endpoint: string) => new DeleteRequest(requestId, endpoint))
|
|
||||||
);
|
|
||||||
|
|
||||||
// Execute the delete request
|
|
||||||
request$.pipe(
|
|
||||||
configureRequest(this.requestService)
|
|
||||||
).subscribe();
|
|
||||||
|
|
||||||
return this.requestService.getByUUID(requestId).pipe(
|
|
||||||
getResponseFromEntry()
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private showNotifications(success: boolean, edited: boolean, isField: boolean, options: any) {
|
private showNotifications(success: boolean, edited: boolean, isField: boolean, options: any) {
|
||||||
|
@@ -0,0 +1,43 @@
|
|||||||
|
<div class="form-group w-100 pr-2 pl-2">
|
||||||
|
<input *ngIf="searchField"
|
||||||
|
type="search"
|
||||||
|
class="form-control w-100"
|
||||||
|
(click)="$event.stopPropagation();"
|
||||||
|
placeholder="{{ 'submission.sections.general.search-collection' | translate }}"
|
||||||
|
[formControl]="searchField"
|
||||||
|
#searchFieldEl>
|
||||||
|
</div>
|
||||||
|
<div class="dropdown-divider"></div>
|
||||||
|
<div
|
||||||
|
class="scrollable-menu"
|
||||||
|
aria-labelledby="dropdownMenuButton"
|
||||||
|
(scroll)="onScroll($event)">
|
||||||
|
<div
|
||||||
|
infiniteScroll
|
||||||
|
[infiniteScrollDistance]="2"
|
||||||
|
[infiniteScrollThrottle]="300"
|
||||||
|
[infiniteScrollUpDistance]="1.5"
|
||||||
|
[infiniteScrollContainer]="'.scrollable-menu'"
|
||||||
|
[fromRoot]="true"
|
||||||
|
(scrolled)="onScrollDown()">
|
||||||
|
<button class="dropdown-item disabled" *ngIf="searchListCollection?.length == 0 && !(isLoadingList | async)">
|
||||||
|
{{'submission.sections.general.no-collection' | translate}}
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
*ngFor="let listItem of searchListCollection"
|
||||||
|
class="dropdown-item collection-item"
|
||||||
|
title="{{ listItem.collection.name }}"
|
||||||
|
(click)="onSelect(listItem)">
|
||||||
|
<ul class="list-unstyled mb-0">
|
||||||
|
<li class="list-item text-truncate text-secondary" *ngFor="let item of listItem.communities">
|
||||||
|
{{ item.name}} <i class="fa fa-level-down" aria-hidden="true"></i>
|
||||||
|
</li>
|
||||||
|
<li class="list-item text-truncate text-primary font-weight-bold">{{ listItem.collection.name}}</li>
|
||||||
|
</ul>
|
||||||
|
</button>
|
||||||
|
<button class="dropdown-item disabled" *ngIf="(isLoadingList | async)" >
|
||||||
|
<ds-loading message="{{'loading.default' | translate}}">
|
||||||
|
</ds-loading>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
@@ -0,0 +1,15 @@
|
|||||||
|
.scrollable-menu {
|
||||||
|
height: auto;
|
||||||
|
max-height: $dropdown-menu-max-height;
|
||||||
|
overflow-x: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.collection-item {
|
||||||
|
border-bottom: $dropdown-border-width solid $dropdown-border-color;
|
||||||
|
}
|
||||||
|
|
||||||
|
#collectionControlsDropdownMenu {
|
||||||
|
outline: 0;
|
||||||
|
left: 0 !important;
|
||||||
|
box-shadow: $btn-focus-box-shadow;
|
||||||
|
}
|
@@ -0,0 +1,241 @@
|
|||||||
|
import { async, ComponentFixture, TestBed, fakeAsync, tick } from '@angular/core/testing';
|
||||||
|
|
||||||
|
import { CollectionDropdownComponent } from './collection-dropdown.component';
|
||||||
|
import { FollowLinkConfig } from '../utils/follow-link-config.model';
|
||||||
|
import { Observable, of } from 'rxjs';
|
||||||
|
import { RemoteData } from 'src/app/core/data/remote-data';
|
||||||
|
import { PaginatedList } from 'src/app/core/data/paginated-list';
|
||||||
|
import { cold, getTestScheduler, hot } from 'jasmine-marbles';
|
||||||
|
import { createSuccessfulRemoteDataObject } from '../remote-data.utils';
|
||||||
|
import { PageInfo } from 'src/app/core/shared/page-info.model';
|
||||||
|
import { Collection } from '../../core/shared/collection.model';
|
||||||
|
import { NO_ERRORS_SCHEMA, ChangeDetectorRef, ElementRef } from '@angular/core';
|
||||||
|
import { CollectionDataService } from 'src/app/core/data/collection-data.service';
|
||||||
|
import { FindListOptions } from 'src/app/core/data/request.models';
|
||||||
|
import { TranslateModule, TranslateLoader } from '@ngx-translate/core';
|
||||||
|
import { TranslateLoaderMock } from '../mocks/translate-loader.mock';
|
||||||
|
import { TestScheduler } from 'rxjs/testing';
|
||||||
|
import { By } from '@angular/platform-browser';
|
||||||
|
import { Community } from 'src/app/core/shared/community.model';
|
||||||
|
|
||||||
|
const community: Community = Object.assign(new Community(), {
|
||||||
|
id: 'ce64f48e-2c9b-411a-ac36-ee429c0e6a88',
|
||||||
|
uuid: 'ce64f48e-2c9b-411a-ac36-ee429c0e6a88',
|
||||||
|
name: 'Community 1'
|
||||||
|
});
|
||||||
|
|
||||||
|
const collections: Collection[] = [
|
||||||
|
Object.assign(new Collection(), {
|
||||||
|
id: 'ce64f48e-2c9b-411a-ac36-ee429c0e6a88',
|
||||||
|
name: 'Collection 1',
|
||||||
|
metadata: [
|
||||||
|
{
|
||||||
|
key: 'dc.title',
|
||||||
|
language: 'en_US',
|
||||||
|
value: 'Community 1-Collection 1'
|
||||||
|
}],
|
||||||
|
parentCommunity: of(
|
||||||
|
new RemoteData(false, false, true, undefined, community, 200)
|
||||||
|
)
|
||||||
|
}),
|
||||||
|
Object.assign(new Collection(), {
|
||||||
|
id: '59ee713b-ee53-4220-8c3f-9860dc84fe33',
|
||||||
|
name: 'Collection 2',
|
||||||
|
metadata: [
|
||||||
|
{
|
||||||
|
key: 'dc.title',
|
||||||
|
language: 'en_US',
|
||||||
|
value: 'Community 1-Collection 2'
|
||||||
|
}],
|
||||||
|
parentCommunity: of(
|
||||||
|
new RemoteData(false, false, true, undefined, community, 200)
|
||||||
|
)
|
||||||
|
}),
|
||||||
|
Object.assign(new Collection(), {
|
||||||
|
id: 'e9dbf393-7127-415f-8919-55be34a6e9ed',
|
||||||
|
name: 'Collection 3',
|
||||||
|
metadata: [
|
||||||
|
{
|
||||||
|
key: 'dc.title',
|
||||||
|
language: 'en_US',
|
||||||
|
value: 'Community 1-Collection 3'
|
||||||
|
}],
|
||||||
|
parentCommunity: of(
|
||||||
|
new RemoteData(false, false, true, undefined, community, 200)
|
||||||
|
)
|
||||||
|
}),
|
||||||
|
Object.assign(new Collection(), {
|
||||||
|
id: '59da2ff0-9bf4-45bf-88be-e35abd33f304',
|
||||||
|
name: 'Collection 4',
|
||||||
|
metadata: [
|
||||||
|
{
|
||||||
|
key: 'dc.title',
|
||||||
|
language: 'en_US',
|
||||||
|
value: 'Community 1-Collection 4'
|
||||||
|
}],
|
||||||
|
parentCommunity: of(
|
||||||
|
new RemoteData(false, false, true, undefined, community, 200)
|
||||||
|
)
|
||||||
|
}),
|
||||||
|
Object.assign(new Collection(), {
|
||||||
|
id: 'a5159760-f362-4659-9e81-e3253ad91ede',
|
||||||
|
name: 'Collection 5',
|
||||||
|
metadata: [
|
||||||
|
{
|
||||||
|
key: 'dc.title',
|
||||||
|
language: 'en_US',
|
||||||
|
value: 'Community 1-Collection 5'
|
||||||
|
}],
|
||||||
|
parentCommunity: of(
|
||||||
|
new RemoteData(false, false, true, undefined, community, 200)
|
||||||
|
)
|
||||||
|
})
|
||||||
|
];
|
||||||
|
|
||||||
|
const listElementMock = {
|
||||||
|
communities: [
|
||||||
|
{
|
||||||
|
id: 'ce64f48e-2c9b-411a-ac36-ee429c0e6a88',
|
||||||
|
name: 'Community 1'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
collection: {
|
||||||
|
id: 'e9dbf393-7127-415f-8919-55be34a6e9ed',
|
||||||
|
uuid: 'e9dbf393-7127-415f-8919-55be34a6e9ed',
|
||||||
|
name: 'Collection 3'
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// tslint:disable-next-line: max-classes-per-file
|
||||||
|
class CollectionDataServiceMock {
|
||||||
|
getAuthorizedCollection(query: string, options: FindListOptions = {}, ...linksToFollow: Array<FollowLinkConfig<Collection>>): Observable<RemoteData<PaginatedList<Collection>>> {
|
||||||
|
return of(
|
||||||
|
createSuccessfulRemoteDataObject(
|
||||||
|
new PaginatedList(new PageInfo(), collections)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
describe('CollectionDropdownComponent', () => {
|
||||||
|
let component: CollectionDropdownComponent;
|
||||||
|
let fixture: ComponentFixture<CollectionDropdownComponent>;
|
||||||
|
let scheduler: TestScheduler;
|
||||||
|
const searchedCollection = 'TEXT';
|
||||||
|
|
||||||
|
beforeEach(async(() => {
|
||||||
|
TestBed.configureTestingModule({
|
||||||
|
imports: [
|
||||||
|
TranslateModule.forRoot({
|
||||||
|
loader: {
|
||||||
|
provide: TranslateLoader,
|
||||||
|
useClass: TranslateLoaderMock
|
||||||
|
}
|
||||||
|
})
|
||||||
|
],
|
||||||
|
declarations: [ CollectionDropdownComponent ],
|
||||||
|
providers: [
|
||||||
|
{provide: CollectionDataService, useClass: CollectionDataServiceMock},
|
||||||
|
{provide: ChangeDetectorRef, useValue: {}},
|
||||||
|
{provide: ElementRef, userValue: {}}
|
||||||
|
],
|
||||||
|
schemas: [NO_ERRORS_SCHEMA]
|
||||||
|
})
|
||||||
|
.compileComponents();
|
||||||
|
}));
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
scheduler = getTestScheduler();
|
||||||
|
fixture = TestBed.createComponent(CollectionDropdownComponent);
|
||||||
|
component = fixture.componentInstance;
|
||||||
|
fixture.detectChanges();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should populate collections list with five items', () => {
|
||||||
|
const elements = fixture.debugElement.queryAll(By.css('.collection-item'));
|
||||||
|
expect(elements.length).toEqual(5);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should trigger onSelect method when select a new collection from list', fakeAsync(() => {
|
||||||
|
spyOn(component, 'onSelect');
|
||||||
|
const collectionItem = fixture.debugElement.query(By.css('.collection-item:nth-child(2)'));
|
||||||
|
collectionItem.triggerEventHandler('click', null);
|
||||||
|
fixture.detectChanges();
|
||||||
|
tick();
|
||||||
|
fixture.whenStable().then(() => {
|
||||||
|
expect(component.onSelect).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
}));
|
||||||
|
|
||||||
|
it('should init component with collection list', fakeAsync(() => {
|
||||||
|
spyOn(component.subs, 'push').and.callThrough();
|
||||||
|
spyOn(component, 'resetPagination').and.callThrough();
|
||||||
|
spyOn(component, 'populateCollectionList').and.callThrough();
|
||||||
|
component.ngOnInit();
|
||||||
|
tick();
|
||||||
|
fixture.detectChanges();
|
||||||
|
|
||||||
|
fixture.whenStable().then(() => {
|
||||||
|
expect(component.subs.push).toHaveBeenCalled();
|
||||||
|
expect(component.resetPagination).toHaveBeenCalled();
|
||||||
|
expect(component.populateCollectionList).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
}));
|
||||||
|
|
||||||
|
it('should emit collectionChange event when selecting a new collection', () => {
|
||||||
|
spyOn(component.selectionChange, 'emit').and.callThrough();
|
||||||
|
component.ngOnInit();
|
||||||
|
component.onSelect(listElementMock as any);
|
||||||
|
fixture.detectChanges();
|
||||||
|
|
||||||
|
expect(component.selectionChange.emit).toHaveBeenCalledWith(listElementMock as any);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should reset collections list after reset of searchField', fakeAsync(() => {
|
||||||
|
spyOn(component.subs, 'push').and.callThrough();
|
||||||
|
spyOn(component, 'reset').and.callThrough();
|
||||||
|
spyOn(component.searchField, 'setValue').and.callThrough();
|
||||||
|
spyOn(component, 'resetPagination').and.callThrough();
|
||||||
|
spyOn(component, 'populateCollectionList').and.callThrough();
|
||||||
|
component.reset();
|
||||||
|
const input = fixture.debugElement.query(By.css('input.form-control'));
|
||||||
|
const el = input.nativeElement;
|
||||||
|
el.value = searchedCollection;
|
||||||
|
el.dispatchEvent(new Event('input'));
|
||||||
|
fixture.detectChanges();
|
||||||
|
tick(500);
|
||||||
|
|
||||||
|
fixture.whenStable().then(() => {
|
||||||
|
expect(component.reset).toHaveBeenCalled();
|
||||||
|
expect(component.searchField.setValue).toHaveBeenCalledWith('');
|
||||||
|
expect(component.resetPagination).toHaveBeenCalled();
|
||||||
|
expect(component.currentQuery).toEqual('');
|
||||||
|
expect(component.populateCollectionList).toHaveBeenCalledWith(component.currentQuery, component.currentPage);
|
||||||
|
expect(component.searchListCollection).toEqual(collections as any);
|
||||||
|
expect(component.subs.push).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
}));
|
||||||
|
|
||||||
|
it('should reset searchField when dropdown menu has been closed', () => {
|
||||||
|
spyOn(component.searchField, 'setValue').and.callThrough();
|
||||||
|
component.reset();
|
||||||
|
|
||||||
|
expect(component.searchField.setValue).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should change loader status', () => {
|
||||||
|
spyOn(component.isLoadingList, 'next').and.callThrough();
|
||||||
|
component.hideShowLoader(true);
|
||||||
|
|
||||||
|
expect(component.isLoadingList.next).toHaveBeenCalledWith(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('reset pagination fields', () => {
|
||||||
|
component.resetPagination();
|
||||||
|
|
||||||
|
expect(component.currentPage).toEqual(1);
|
||||||
|
expect(component.currentQuery).toEqual('');
|
||||||
|
expect(component.hasNextPage).toEqual(true);
|
||||||
|
expect(component.searchListCollection).toEqual([]);
|
||||||
|
});
|
||||||
|
});
|
@@ -0,0 +1,236 @@
|
|||||||
|
import { Component, OnInit, HostListener, ChangeDetectorRef, OnDestroy, Output, EventEmitter, ElementRef } from '@angular/core';
|
||||||
|
import { FormControl } from '@angular/forms';
|
||||||
|
import { Observable, Subscription, BehaviorSubject } from 'rxjs';
|
||||||
|
import { hasValue } from '../empty.util';
|
||||||
|
import { map, mergeMap, startWith, debounceTime, distinctUntilChanged, switchMap, reduce } from 'rxjs/operators';
|
||||||
|
import { RemoteData } from 'src/app/core/data/remote-data';
|
||||||
|
import { FindListOptions } from 'src/app/core/data/request.models';
|
||||||
|
import { PaginatedList } from 'src/app/core/data/paginated-list';
|
||||||
|
import { Community } from 'src/app/core/shared/community.model';
|
||||||
|
import { CollectionDataService } from 'src/app/core/data/collection-data.service';
|
||||||
|
import { Collection } from '../../core/shared/collection.model';
|
||||||
|
import { followLink } from '../utils/follow-link-config.model';
|
||||||
|
import { getFirstSucceededRemoteDataPayload, getSucceededRemoteWithNotEmptyData } from '../../core/shared/operators';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An interface to represent a collection entry
|
||||||
|
*/
|
||||||
|
interface CollectionListEntryItem {
|
||||||
|
id: string;
|
||||||
|
uuid: string;
|
||||||
|
name: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An interface to represent an entry in the collection list
|
||||||
|
*/
|
||||||
|
interface CollectionListEntry {
|
||||||
|
communities: CollectionListEntryItem[],
|
||||||
|
collection: CollectionListEntryItem
|
||||||
|
}
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
selector: 'ds-collection-dropdown',
|
||||||
|
templateUrl: './collection-dropdown.component.html',
|
||||||
|
styleUrls: ['./collection-dropdown.component.scss']
|
||||||
|
})
|
||||||
|
export class CollectionDropdownComponent implements OnInit, OnDestroy {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The search form control
|
||||||
|
* @type {FormControl}
|
||||||
|
*/
|
||||||
|
public searchField: FormControl = new FormControl();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The collection list obtained from a search
|
||||||
|
* @type {Observable<CollectionListEntry[]>}
|
||||||
|
*/
|
||||||
|
public searchListCollection$: Observable<CollectionListEntry[]>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A boolean representing if dropdown list is scrollable to the bottom
|
||||||
|
* @type {boolean}
|
||||||
|
*/
|
||||||
|
private scrollableBottom = false;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A boolean representing if dropdown list is scrollable to the top
|
||||||
|
* @type {boolean}
|
||||||
|
*/
|
||||||
|
private scrollableTop = false;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Array to track all subscriptions and unsubscribe them onDestroy
|
||||||
|
* @type {Array}
|
||||||
|
*/
|
||||||
|
public subs: Subscription[] = [];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The list of collection to render
|
||||||
|
*/
|
||||||
|
searchListCollection: CollectionListEntry[] = [];
|
||||||
|
|
||||||
|
@Output() selectionChange = new EventEmitter<CollectionListEntry>();
|
||||||
|
/**
|
||||||
|
* A boolean representing if the loader is visible or not
|
||||||
|
*/
|
||||||
|
isLoadingList: BehaviorSubject<boolean> = new BehaviorSubject(false);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A numeric representig current page
|
||||||
|
*/
|
||||||
|
currentPage: number;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A boolean representing if exist another page to render
|
||||||
|
*/
|
||||||
|
hasNextPage: boolean;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Current seach query used to filter collection list
|
||||||
|
*/
|
||||||
|
currentQuery: string;
|
||||||
|
|
||||||
|
constructor(
|
||||||
|
private changeDetectorRef: ChangeDetectorRef,
|
||||||
|
private collectionDataService: CollectionDataService,
|
||||||
|
private el: ElementRef
|
||||||
|
) { }
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Method called on mousewheel event, it prevent the page scroll
|
||||||
|
* when arriving at the top/bottom of dropdown menu
|
||||||
|
*
|
||||||
|
* @param event
|
||||||
|
* mousewheel event
|
||||||
|
*/
|
||||||
|
@HostListener('mousewheel', ['$event']) onMousewheel(event) {
|
||||||
|
if (event.wheelDelta > 0 && this.scrollableTop) {
|
||||||
|
event.preventDefault();
|
||||||
|
}
|
||||||
|
if (event.wheelDelta < 0 && this.scrollableBottom) {
|
||||||
|
event.preventDefault();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Initialize collection list
|
||||||
|
*/
|
||||||
|
ngOnInit() {
|
||||||
|
this.subs.push(this.searchField.valueChanges.pipe(
|
||||||
|
debounceTime(500),
|
||||||
|
distinctUntilChanged(),
|
||||||
|
startWith('')
|
||||||
|
).subscribe(
|
||||||
|
(next) => {
|
||||||
|
if (hasValue(next) && next !== this.currentQuery) {
|
||||||
|
this.resetPagination();
|
||||||
|
this.currentQuery = next;
|
||||||
|
this.populateCollectionList(this.currentQuery, this.currentPage);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
));
|
||||||
|
// Workaround for prevent the scroll of main page when this component is placed in a dialog
|
||||||
|
setTimeout(() => this.el.nativeElement.querySelector('input').focus(), 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if dropdown scrollbar is at the top or bottom of the dropdown list
|
||||||
|
*
|
||||||
|
* @param event
|
||||||
|
*/
|
||||||
|
onScroll(event) {
|
||||||
|
this.scrollableBottom = (event.target.scrollTop + event.target.clientHeight === event.target.scrollHeight);
|
||||||
|
this.scrollableTop = (event.target.scrollTop === 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Method used from infitity scroll for retrive more data on scroll down
|
||||||
|
*/
|
||||||
|
onScrollDown() {
|
||||||
|
if ( this.hasNextPage ) {
|
||||||
|
this.populateCollectionList(this.currentQuery, ++this.currentPage);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Emit a [selectionChange] event when a new collection is selected from list
|
||||||
|
*
|
||||||
|
* @param event
|
||||||
|
* the selected [CollectionListEntry]
|
||||||
|
*/
|
||||||
|
onSelect(event: CollectionListEntry) {
|
||||||
|
this.selectionChange.emit(event);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Method called for populate the collection list
|
||||||
|
* @param query text for filter the collection list
|
||||||
|
* @param page page number
|
||||||
|
*/
|
||||||
|
populateCollectionList(query: string, page: number) {
|
||||||
|
this.isLoadingList.next(true);
|
||||||
|
// Set the pagination info
|
||||||
|
const findOptions: FindListOptions = {
|
||||||
|
elementsPerPage: 10,
|
||||||
|
currentPage: page
|
||||||
|
};
|
||||||
|
this.searchListCollection$ = this.collectionDataService
|
||||||
|
.getAuthorizedCollection(query, findOptions, followLink('parentCommunity'))
|
||||||
|
.pipe(
|
||||||
|
getSucceededRemoteWithNotEmptyData(),
|
||||||
|
switchMap((collections: RemoteData<PaginatedList<Collection>>) => {
|
||||||
|
if ( (this.searchListCollection.length + findOptions.elementsPerPage) >= collections.payload.totalElements ) {
|
||||||
|
this.hasNextPage = false;
|
||||||
|
}
|
||||||
|
return collections.payload.page;
|
||||||
|
}),
|
||||||
|
mergeMap((collection: Collection) => collection.parentCommunity.pipe(
|
||||||
|
getFirstSucceededRemoteDataPayload(),
|
||||||
|
map((community: Community) => ({
|
||||||
|
communities: [{ id: community.id, name: community.name }],
|
||||||
|
collection: { id: collection.id, uuid: collection.id, name: collection.name }
|
||||||
|
})
|
||||||
|
))),
|
||||||
|
reduce((acc: any, value: any) => [...acc, ...value], []),
|
||||||
|
startWith([])
|
||||||
|
);
|
||||||
|
this.subs.push(this.searchListCollection$.subscribe(
|
||||||
|
(next) => { this.searchListCollection.push(...next); }, undefined,
|
||||||
|
() => { this.hideShowLoader(false); this.changeDetectorRef.detectChanges(); }
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Unsubscribe from all subscriptions
|
||||||
|
*/
|
||||||
|
ngOnDestroy(): void {
|
||||||
|
this.subs.filter((sub) => hasValue(sub)).forEach((sub) => sub.unsubscribe());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reset search form control
|
||||||
|
*/
|
||||||
|
reset() {
|
||||||
|
this.searchField.setValue('');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reset pagination values
|
||||||
|
*/
|
||||||
|
resetPagination() {
|
||||||
|
this.currentPage = 1;
|
||||||
|
this.currentQuery = '';
|
||||||
|
this.hasNextPage = true;
|
||||||
|
this.searchListCollection = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Hide/Show the collection list loader
|
||||||
|
* @param hideShow true for show, false otherwise
|
||||||
|
*/
|
||||||
|
hideShowLoader(hideShow: boolean) {
|
||||||
|
this.isLoadingList.next(hideShow);
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,11 @@
|
|||||||
|
<div>
|
||||||
|
<div class="modal-header">{{'dso-selector.'+ action + '.' + objectType.toString().toLowerCase() + '.head' | translate}}
|
||||||
|
<button type="button" class="close" (click)="close()" aria-label="Close">
|
||||||
|
<span aria-hidden="true">×</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<div class="modal-body">
|
||||||
|
<ds-collection-dropdown (selectionChange)="selectObject($event.collection)">
|
||||||
|
</ds-collection-dropdown>
|
||||||
|
</div>
|
||||||
|
</div>
|
@@ -13,7 +13,8 @@ import { DSOSelectorModalWrapperComponent, SelectorActionType } from '../dso-sel
|
|||||||
@Component({
|
@Component({
|
||||||
selector: 'ds-create-item-parent-selector',
|
selector: 'ds-create-item-parent-selector',
|
||||||
// styleUrls: ['./create-item-parent-selector.component.scss'],
|
// styleUrls: ['./create-item-parent-selector.component.scss'],
|
||||||
templateUrl: '../dso-selector-modal-wrapper.component.html',
|
// templateUrl: '../dso-selector-modal-wrapper.component.html',
|
||||||
|
templateUrl: './create-item-parent-selector.component.html'
|
||||||
})
|
})
|
||||||
export class CreateItemParentSelectorComponent extends DSOSelectorModalWrapperComponent implements OnInit {
|
export class CreateItemParentSelectorComponent extends DSOSelectorModalWrapperComponent implements OnInit {
|
||||||
objectType = DSpaceObjectType.ITEM;
|
objectType = DSpaceObjectType.ITEM;
|
||||||
|
@@ -11,7 +11,7 @@ export function getMockRequestService(requestEntry$: Observable<RequestEntry> =
|
|||||||
getByUUID: requestEntry$,
|
getByUUID: requestEntry$,
|
||||||
uriEncodeBody: jasmine.createSpy('uriEncodeBody'),
|
uriEncodeBody: jasmine.createSpy('uriEncodeBody'),
|
||||||
isCachedOrPending: false,
|
isCachedOrPending: false,
|
||||||
removeByHrefSubstring: jasmine.createSpy('removeByHrefSubstring'),
|
removeByHrefSubstring: jasmine.createSpy('removeByHrefSubstring').and.returnValue(observableOf(true)),
|
||||||
hasByHrefObservable: observableOf(false)
|
hasByHrefObservable: observableOf(false)
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@@ -75,7 +75,7 @@ export class NotificationsService {
|
|||||||
this.translate.get(hrefTranslateLabel)
|
this.translate.get(hrefTranslateLabel)
|
||||||
.pipe(first())
|
.pipe(first())
|
||||||
.subscribe((hrefMsg) => {
|
.subscribe((hrefMsg) => {
|
||||||
const anchor = `<a class="btn btn-link p-0 m-0" href="${href}" >
|
const anchor = `<a class="align-baseline btn btn-link p-0 m-0" href="${href}" >
|
||||||
<strong>${hrefMsg}</strong>
|
<strong>${hrefMsg}</strong>
|
||||||
</a>`;
|
</a>`;
|
||||||
const interpolateParams = Object.create({});
|
const interpolateParams = Object.create({});
|
||||||
|
@@ -12,14 +12,16 @@ import { take } from 'rxjs/operators';
|
|||||||
import { PaginationComponent } from '../pagination/pagination.component';
|
import { PaginationComponent } from '../pagination/pagination.component';
|
||||||
import { createSuccessfulRemoteDataObject } from '../remote-data.utils';
|
import { createSuccessfulRemoteDataObject } from '../remote-data.utils';
|
||||||
import { createPaginatedList } from '../testing/utils.test';
|
import { createPaginatedList } from '../testing/utils.test';
|
||||||
|
import { ObjectValuesPipe } from '../utils/object-values-pipe';
|
||||||
|
|
||||||
class MockAbstractPaginatedDragAndDropListComponent extends AbstractPaginatedDragAndDropListComponent<DSpaceObject> {
|
class MockAbstractPaginatedDragAndDropListComponent extends AbstractPaginatedDragAndDropListComponent<DSpaceObject> {
|
||||||
|
|
||||||
constructor(protected objectUpdatesService: ObjectUpdatesService,
|
constructor(protected objectUpdatesService: ObjectUpdatesService,
|
||||||
protected elRef: ElementRef,
|
protected elRef: ElementRef,
|
||||||
|
protected objectValuesPipe: ObjectValuesPipe,
|
||||||
protected mockUrl: string,
|
protected mockUrl: string,
|
||||||
protected mockObjectsRD$: Observable<RemoteData<PaginatedList<DSpaceObject>>>) {
|
protected mockObjectsRD$: Observable<RemoteData<PaginatedList<DSpaceObject>>>) {
|
||||||
super(objectUpdatesService, elRef);
|
super(objectUpdatesService, elRef, objectValuesPipe);
|
||||||
}
|
}
|
||||||
|
|
||||||
initializeObjectsRD(): void {
|
initializeObjectsRD(): void {
|
||||||
@@ -35,6 +37,7 @@ describe('AbstractPaginatedDragAndDropListComponent', () => {
|
|||||||
let component: MockAbstractPaginatedDragAndDropListComponent;
|
let component: MockAbstractPaginatedDragAndDropListComponent;
|
||||||
let objectUpdatesService: ObjectUpdatesService;
|
let objectUpdatesService: ObjectUpdatesService;
|
||||||
let elRef: ElementRef;
|
let elRef: ElementRef;
|
||||||
|
let objectValuesPipe: ObjectValuesPipe;
|
||||||
|
|
||||||
const url = 'mock-abstract-paginated-drag-and-drop-list-component';
|
const url = 'mock-abstract-paginated-drag-and-drop-list-component';
|
||||||
|
|
||||||
@@ -52,32 +55,26 @@ describe('AbstractPaginatedDragAndDropListComponent', () => {
|
|||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
objectUpdatesService = jasmine.createSpyObj('objectUpdatesService', {
|
objectUpdatesService = jasmine.createSpyObj('objectUpdatesService', {
|
||||||
initializeWithCustomOrder: {},
|
initialize: {},
|
||||||
addPageToCustomOrder: {},
|
getFieldUpdatesExclusive: observableOf(updates)
|
||||||
getFieldUpdatesByCustomOrder: observableOf(updates),
|
|
||||||
saveMoveFieldUpdate: {}
|
|
||||||
});
|
});
|
||||||
elRef = {
|
elRef = {
|
||||||
nativeElement: jasmine.createSpyObj('nativeElement', {
|
nativeElement: jasmine.createSpyObj('nativeElement', {
|
||||||
querySelector: {}
|
querySelector: {}
|
||||||
})
|
})
|
||||||
};
|
};
|
||||||
|
objectValuesPipe = new ObjectValuesPipe();
|
||||||
paginationComponent = jasmine.createSpyObj('paginationComponent', {
|
paginationComponent = jasmine.createSpyObj('paginationComponent', {
|
||||||
doPageChange: {}
|
doPageChange: {}
|
||||||
});
|
});
|
||||||
objectsRD$ = new BehaviorSubject(objectsRD);
|
objectsRD$ = new BehaviorSubject(objectsRD);
|
||||||
component = new MockAbstractPaginatedDragAndDropListComponent(objectUpdatesService, elRef, url, objectsRD$);
|
component = new MockAbstractPaginatedDragAndDropListComponent(objectUpdatesService, elRef, objectValuesPipe, url, objectsRD$);
|
||||||
component.paginationComponent = paginationComponent;
|
component.paginationComponent = paginationComponent;
|
||||||
component.ngOnInit();
|
component.ngOnInit();
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should call initializeWithCustomOrder to initialize the first page and add it to initializedPages', (done) => {
|
it('should call initialize to initialize the objects in the store', () => {
|
||||||
expect(component.initializedPages.indexOf(0)).toBeLessThan(0);
|
expect(objectUpdatesService.initialize).toHaveBeenCalled();
|
||||||
component.updates$.pipe(take(1)).subscribe(() => {
|
|
||||||
expect(objectUpdatesService.initializeWithCustomOrder).toHaveBeenCalled();
|
|
||||||
expect(component.initializedPages.indexOf(0)).toBeGreaterThanOrEqual(0);
|
|
||||||
done();
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should initialize the updates correctly', (done) => {
|
it('should initialize the updates correctly', (done) => {
|
||||||
@@ -87,43 +84,6 @@ describe('AbstractPaginatedDragAndDropListComponent', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('when a new page is loaded', () => {
|
|
||||||
const page = 5;
|
|
||||||
|
|
||||||
beforeEach((done) => {
|
|
||||||
component.updates$.pipe(take(1)).subscribe(() => {
|
|
||||||
component.currentPage$.next(page);
|
|
||||||
objectsRD$.next(objectsRD);
|
|
||||||
done();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should call addPageToCustomOrder to initialize the new page and add it to initializedPages', (done) => {
|
|
||||||
component.updates$.pipe(take(1)).subscribe(() => {
|
|
||||||
expect(objectUpdatesService.addPageToCustomOrder).toHaveBeenCalled();
|
|
||||||
expect(component.initializedPages.indexOf(page - 1)).toBeGreaterThanOrEqual(0);
|
|
||||||
done();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('twice', () => {
|
|
||||||
beforeEach((done) => {
|
|
||||||
component.updates$.pipe(take(1)).subscribe(() => {
|
|
||||||
component.currentPage$.next(page);
|
|
||||||
objectsRD$.next(objectsRD);
|
|
||||||
done();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('shouldn\'t call addPageToCustomOrder again, as the page has already been initialized', (done) => {
|
|
||||||
component.updates$.pipe(take(1)).subscribe(() => {
|
|
||||||
expect(objectUpdatesService.addPageToCustomOrder).toHaveBeenCalledTimes(1);
|
|
||||||
done();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('switchPage', () => {
|
describe('switchPage', () => {
|
||||||
const page = 3;
|
const page = 3;
|
||||||
|
|
||||||
@@ -149,30 +109,31 @@ describe('AbstractPaginatedDragAndDropListComponent', () => {
|
|||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
elRef.nativeElement.querySelector.and.returnValue(hoverElement);
|
elRef.nativeElement.querySelector.and.returnValue(hoverElement);
|
||||||
component.initializedPages.push(hoverPage - 1);
|
spyOn(component.dropObject, 'emit');
|
||||||
component.drop(event);
|
component.drop(event);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should detect the page and set currentPage$ to its value', () => {
|
it('should send out a dropObject event with the expected processed paginated indexes', () => {
|
||||||
expect(component.currentPage$.value).toEqual(hoverPage);
|
expect(component.dropObject.emit).toHaveBeenCalledWith(Object.assign({
|
||||||
});
|
fromIndex: ((component.currentPage$.value - 1) * component.pageSize) + event.previousIndex,
|
||||||
|
toIndex: ((hoverPage - 1) * component.pageSize),
|
||||||
it('should detect the page and update the pagination component with its value', () => {
|
finish: jasmine.anything()
|
||||||
expect(paginationComponent.doPageChange).toHaveBeenCalledWith(hoverPage);
|
}));
|
||||||
});
|
|
||||||
|
|
||||||
it('should send out a saveMoveFieldUpdate with the correct values', () => {
|
|
||||||
expect(objectUpdatesService.saveMoveFieldUpdate).toHaveBeenCalledWith(url, event.previousIndex, 0, 0, hoverPage - 1, object1);
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('when the user is not hovering over a new page', () => {
|
describe('when the user is not hovering over a new page', () => {
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
|
spyOn(component.dropObject, 'emit');
|
||||||
component.drop(event);
|
component.drop(event);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should send out a saveMoveFieldUpdate with the correct values', () => {
|
it('should send out a dropObject event with the expected properties', () => {
|
||||||
expect(objectUpdatesService.saveMoveFieldUpdate).toHaveBeenCalledWith(url, event.previousIndex, event.currentIndex, 0, 0);
|
expect(component.dropObject.emit).toHaveBeenCalledWith(Object.assign({
|
||||||
|
fromIndex: event.previousIndex,
|
||||||
|
toIndex: event.currentIndex,
|
||||||
|
finish: jasmine.anything()
|
||||||
|
}));
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@@ -1,23 +1,33 @@
|
|||||||
import { FieldUpdates } from '../../core/data/object-updates/object-updates.reducer';
|
import { FieldUpdate, FieldUpdates } from '../../core/data/object-updates/object-updates.reducer';
|
||||||
import { Observable } from 'rxjs/internal/Observable';
|
import { Observable } from 'rxjs/internal/Observable';
|
||||||
import { RemoteData } from '../../core/data/remote-data';
|
import { RemoteData } from '../../core/data/remote-data';
|
||||||
import { PaginatedList } from '../../core/data/paginated-list';
|
import { PaginatedList } from '../../core/data/paginated-list';
|
||||||
import { PaginationComponentOptions } from '../pagination/pagination-component-options.model';
|
import { PaginationComponentOptions } from '../pagination/pagination-component-options.model';
|
||||||
import { BehaviorSubject } from 'rxjs/internal/BehaviorSubject';
|
import { BehaviorSubject } from 'rxjs/internal/BehaviorSubject';
|
||||||
import { ObjectUpdatesService } from '../../core/data/object-updates/object-updates.service';
|
import { ObjectUpdatesService } from '../../core/data/object-updates/object-updates.service';
|
||||||
import { switchMap, take, tap } from 'rxjs/operators';
|
import { distinctUntilChanged, map, switchMap, take } from 'rxjs/operators';
|
||||||
import { hasValue, isEmpty, isNotEmpty } from '../empty.util';
|
import { hasValue, isNotEmpty } from '../empty.util';
|
||||||
import { paginatedListToArray } from '../../core/shared/operators';
|
import { paginatedListToArray } from '../../core/shared/operators';
|
||||||
import { DSpaceObject } from '../../core/shared/dspace-object.model';
|
import { DSpaceObject } from '../../core/shared/dspace-object.model';
|
||||||
import { CdkDragDrop } from '@angular/cdk/drag-drop';
|
import { CdkDragDrop, moveItemInArray } from '@angular/cdk/drag-drop';
|
||||||
import { ElementRef, ViewChild } from '@angular/core';
|
import { ElementRef, EventEmitter, OnDestroy, Output, ViewChild } from '@angular/core';
|
||||||
import { PaginationComponent } from '../pagination/pagination.component';
|
import { PaginationComponent } from '../pagination/pagination.component';
|
||||||
|
import { ObjectValuesPipe } from '../utils/object-values-pipe';
|
||||||
|
import { compareArraysUsing } from '../../+item-page/simple/item-types/shared/item-relationships-utils';
|
||||||
|
import { Subscription } from 'rxjs/internal/Subscription';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Operator used for comparing {@link FieldUpdate}s by their field's UUID
|
||||||
|
*/
|
||||||
|
export const compareArraysUsingFieldUuids = () =>
|
||||||
|
compareArraysUsing((fieldUpdate: FieldUpdate) => (hasValue(fieldUpdate) && hasValue(fieldUpdate.field)) ? fieldUpdate.field.uuid : undefined);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* An abstract component containing general methods and logic to be able to drag and drop objects within a paginated
|
* An abstract component containing general methods and logic to be able to drag and drop objects within a paginated
|
||||||
* list. This implementation supports being able to drag and drop objects between pages.
|
* list. This implementation supports being able to drag and drop objects between pages.
|
||||||
* Dragging an object on top of a page number will automatically detect the page it's being dropped on, send an update
|
* Dragging an object on top of a page number will automatically detect the page it's being dropped on and send a
|
||||||
* to the store and add the object on top of that page.
|
* dropObject event to the parent component containing detailed information about the indexes the object was dropped from
|
||||||
|
* and to.
|
||||||
*
|
*
|
||||||
* To extend this component, it is important to make sure to:
|
* To extend this component, it is important to make sure to:
|
||||||
* - Initialize objectsRD$ within the initializeObjectsRD() method
|
* - Initialize objectsRD$ within the initializeObjectsRD() method
|
||||||
@@ -28,12 +38,19 @@ import { PaginationComponent } from '../pagination/pagination.component';
|
|||||||
*
|
*
|
||||||
* An example component extending from this abstract component: PaginatedDragAndDropBitstreamListComponent
|
* An example component extending from this abstract component: PaginatedDragAndDropBitstreamListComponent
|
||||||
*/
|
*/
|
||||||
export abstract class AbstractPaginatedDragAndDropListComponent<T extends DSpaceObject> {
|
export abstract class AbstractPaginatedDragAndDropListComponent<T extends DSpaceObject> implements OnDestroy {
|
||||||
/**
|
/**
|
||||||
* A view on the child pagination component
|
* A view on the child pagination component
|
||||||
*/
|
*/
|
||||||
@ViewChild(PaginationComponent, {static: false}) paginationComponent: PaginationComponent;
|
@ViewChild(PaginationComponent, {static: false}) paginationComponent: PaginationComponent;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Send an event when the user drops an object on the pagination
|
||||||
|
* The event contains details about the index the object came from and is dropped to (across the entirety of the list,
|
||||||
|
* not just within a single page)
|
||||||
|
*/
|
||||||
|
@Output() dropObject: EventEmitter<any> = new EventEmitter<any>();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The URL to use for accessing the object updates from this list
|
* The URL to use for accessing the object updates from this list
|
||||||
*/
|
*/
|
||||||
@@ -49,6 +66,12 @@ export abstract class AbstractPaginatedDragAndDropListComponent<T extends DSpace
|
|||||||
*/
|
*/
|
||||||
updates$: Observable<FieldUpdates>;
|
updates$: Observable<FieldUpdates>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A list of object UUIDs
|
||||||
|
* This is the order the objects will be displayed in
|
||||||
|
*/
|
||||||
|
customOrder: string[];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The amount of objects to display per page
|
* The amount of objects to display per page
|
||||||
*/
|
*/
|
||||||
@@ -70,23 +93,21 @@ export abstract class AbstractPaginatedDragAndDropListComponent<T extends DSpace
|
|||||||
currentPage$ = new BehaviorSubject<number>(1);
|
currentPage$ = new BehaviorSubject<number>(1);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A list of pages that have been initialized in the field-update store
|
* Whether or not we should display a loading animation
|
||||||
|
* This is used to display a loading page when the user drops a bitstream onto a new page. The loading animation
|
||||||
|
* should stop once the bitstream has moved to the new page and the new page's response has loaded and contains the
|
||||||
|
* dropped object on top (see this.stopLoadingWhenFirstIs below)
|
||||||
*/
|
*/
|
||||||
initializedPages: number[] = [];
|
loading$: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(false);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* An object storing information about an update that should be fired whenever fireToUpdate is called
|
* List of subscriptions
|
||||||
*/
|
*/
|
||||||
toUpdate: {
|
subs: Subscription[] = [];
|
||||||
fromIndex: number,
|
|
||||||
toIndex: number,
|
|
||||||
fromPage: number,
|
|
||||||
toPage: number,
|
|
||||||
field?: T
|
|
||||||
};
|
|
||||||
|
|
||||||
protected constructor(protected objectUpdatesService: ObjectUpdatesService,
|
protected constructor(protected objectUpdatesService: ObjectUpdatesService,
|
||||||
protected elRef: ElementRef) {
|
protected elRef: ElementRef,
|
||||||
|
protected objectValuesPipe: ObjectValuesPipe) {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -110,28 +131,29 @@ export abstract class AbstractPaginatedDragAndDropListComponent<T extends DSpace
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Initialize the field-updates in the store
|
* Initialize the field-updates in the store
|
||||||
* This method ensures (new) pages displayed are automatically added to the field-update store when the objectsRD updates
|
|
||||||
*/
|
*/
|
||||||
initializeUpdates(): void {
|
initializeUpdates(): void {
|
||||||
|
this.objectsRD$.pipe(
|
||||||
|
paginatedListToArray(),
|
||||||
|
take(1)
|
||||||
|
).subscribe((objects: T[]) => {
|
||||||
|
this.objectUpdatesService.initialize(this.url, objects, new Date());
|
||||||
|
});
|
||||||
this.updates$ = this.objectsRD$.pipe(
|
this.updates$ = this.objectsRD$.pipe(
|
||||||
paginatedListToArray(),
|
paginatedListToArray(),
|
||||||
tap((objects: T[]) => {
|
switchMap((objects: T[]) => this.objectUpdatesService.getFieldUpdatesExclusive(this.url, objects))
|
||||||
// Pages in the field-update store are indexed starting at 0 (because they're stored in an array of pages)
|
);
|
||||||
const updatesPage = this.currentPage$.value - 1;
|
this.subs.push(
|
||||||
if (isEmpty(this.initializedPages)) {
|
this.updates$.pipe(
|
||||||
// No updates have been initialized yet for this list, initialize the first page
|
map((fieldUpdates) => this.objectValuesPipe.transform(fieldUpdates)),
|
||||||
this.objectUpdatesService.initializeWithCustomOrder(this.url, objects, new Date(), this.pageSize, updatesPage);
|
distinctUntilChanged(compareArraysUsingFieldUuids())
|
||||||
this.initializedPages.push(updatesPage);
|
).subscribe((updateValues) => {
|
||||||
} else if (this.initializedPages.indexOf(updatesPage) < 0) {
|
this.customOrder = updateValues.map((fieldUpdate) => fieldUpdate.field.uuid);
|
||||||
// Updates were initialized for this list, but not the page we're on. Add the current page to the field-update store for this list
|
// We received new values, stop displaying a loading indicator if it's present
|
||||||
this.objectUpdatesService.addPageToCustomOrder(this.url, objects, updatesPage);
|
this.loading$.next(false);
|
||||||
this.initializedPages.push(updatesPage);
|
|
||||||
}
|
|
||||||
|
|
||||||
// The new page is loaded into the store, check if there are any updates waiting and fire those as well
|
|
||||||
this.fireToUpdate();
|
|
||||||
}),
|
}),
|
||||||
switchMap((objects: T[]) => this.objectUpdatesService.getFieldUpdatesByCustomOrder(this.url, objects, this.currentPage$.value - 1))
|
// Disable the pagination when objects are loading
|
||||||
|
this.loading$.subscribe((loading) => this.options.disabled = loading)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -144,52 +166,60 @@ export abstract class AbstractPaginatedDragAndDropListComponent<T extends DSpace
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* An object was moved, send updates to the store.
|
* An object was moved, send updates to the dropObject EventEmitter
|
||||||
* When the object is dropped on a page within the pagination of this component, the object moves to the top of that
|
* When the object is dropped on a page within the pagination of this component, the object moves to the top of that
|
||||||
* page and the pagination automatically loads and switches the view to that page.
|
* page and the pagination automatically loads and switches the view to that page (this is done by calling the event's
|
||||||
|
* finish() method after sending patch requests to the REST API)
|
||||||
* @param event
|
* @param event
|
||||||
*/
|
*/
|
||||||
drop(event: CdkDragDrop<any>) {
|
drop(event: CdkDragDrop<any>) {
|
||||||
|
const dragIndex = event.previousIndex;
|
||||||
|
let dropIndex = event.currentIndex;
|
||||||
|
const dragPage = this.currentPage$.value - 1;
|
||||||
|
let dropPage = this.currentPage$.value - 1;
|
||||||
|
|
||||||
// Check if the user is hovering over any of the pagination's pages at the time of dropping the object
|
// Check if the user is hovering over any of the pagination's pages at the time of dropping the object
|
||||||
const droppedOnElement = this.elRef.nativeElement.querySelector('.page-item:hover');
|
const droppedOnElement = this.elRef.nativeElement.querySelector('.page-item:hover');
|
||||||
if (hasValue(droppedOnElement) && hasValue(droppedOnElement.textContent)) {
|
if (hasValue(droppedOnElement) && hasValue(droppedOnElement.textContent)) {
|
||||||
// The user is hovering over a page, fetch the page's number from the element
|
// The user is hovering over a page, fetch the page's number from the element
|
||||||
const page = Number(droppedOnElement.textContent);
|
const droppedPage = Number(droppedOnElement.textContent);
|
||||||
if (hasValue(page) && !Number.isNaN(page)) {
|
if (hasValue(droppedPage) && !Number.isNaN(droppedPage)) {
|
||||||
const id = event.item.element.nativeElement.id;
|
dropPage = droppedPage - 1;
|
||||||
this.updates$.pipe(take(1)).subscribe((updates: FieldUpdates) => {
|
dropIndex = 0;
|
||||||
const field = hasValue(updates[id]) ? updates[id].field : undefined;
|
|
||||||
this.toUpdate = Object.assign({
|
|
||||||
fromIndex: event.previousIndex,
|
|
||||||
toIndex: 0,
|
|
||||||
fromPage: this.currentPage$.value - 1,
|
|
||||||
toPage: page - 1,
|
|
||||||
field
|
|
||||||
});
|
|
||||||
// Switch to the dropped-on page and force a page update for the pagination component
|
|
||||||
this.currentPage$.next(page);
|
|
||||||
this.paginationComponent.doPageChange(page);
|
|
||||||
if (this.initializedPages.indexOf(page - 1) >= 0) {
|
|
||||||
// The page the object is being dropped to has already been loaded before, directly fire an update to the store.
|
|
||||||
// For pages that haven't been loaded before, the updates$ observable will call fireToUpdate after the new page
|
|
||||||
// has loaded
|
|
||||||
this.fireToUpdate();
|
|
||||||
}
|
}
|
||||||
});
|
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
this.objectUpdatesService.saveMoveFieldUpdate(this.url, event.previousIndex, event.currentIndex, this.currentPage$.value - 1, this.currentPage$.value - 1);
|
const isNewPage = dragPage !== dropPage;
|
||||||
|
// Move the object in the custom order array if the drop happened within the same page
|
||||||
|
// This allows us to instantly display a change in the order, instead of waiting for the REST API's response first
|
||||||
|
if (!isNewPage && dragIndex !== dropIndex) {
|
||||||
|
moveItemInArray(this.customOrder, dragIndex, dropIndex);
|
||||||
|
}
|
||||||
|
|
||||||
|
const redirectPage = dropPage + 1;
|
||||||
|
const fromIndex = (dragPage * this.pageSize) + dragIndex;
|
||||||
|
const toIndex = (dropPage * this.pageSize) + dropIndex;
|
||||||
|
// Send out a drop event (and navigate to the new page) when the "from" and "to" indexes are different from each other
|
||||||
|
if (fromIndex !== toIndex) {
|
||||||
|
if (isNewPage) {
|
||||||
|
this.loading$.next(true);
|
||||||
|
}
|
||||||
|
this.dropObject.emit(Object.assign({
|
||||||
|
fromIndex,
|
||||||
|
toIndex,
|
||||||
|
finish: () => {
|
||||||
|
if (isNewPage) {
|
||||||
|
this.paginationComponent.doPageChange(redirectPage);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Method checking if there's an update ready to be fired. Send out a MoveFieldUpdate to the store if there's an
|
* unsub all subscriptions
|
||||||
* update present and clear the update afterwards.
|
|
||||||
*/
|
*/
|
||||||
fireToUpdate() {
|
ngOnDestroy(): void {
|
||||||
if (hasValue(this.toUpdate)) {
|
this.subs.filter((sub) => hasValue(sub)).forEach((sub) => sub.unsubscribe());
|
||||||
this.objectUpdatesService.saveMoveFieldUpdate(this.url, this.toUpdate.fromIndex, this.toUpdate.toIndex, this.toUpdate.fromPage, this.toUpdate.toPage, this.toUpdate.field);
|
|
||||||
this.toUpdate = undefined;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
14
src/app/shared/pagination/pagination.utils.ts
Normal file
14
src/app/shared/pagination/pagination.utils.ts
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
import { PaginationComponentOptions } from './pagination-component-options.model';
|
||||||
|
import { FindListOptions } from '../../core/data/request.models';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transform a PaginationComponentOptions object into a FindListOptions object
|
||||||
|
* @param pagination The PaginationComponentOptions to transform
|
||||||
|
* @param original An original FindListOptions object to start from
|
||||||
|
*/
|
||||||
|
export function toFindListOptions(pagination: PaginationComponentOptions, original?: FindListOptions): FindListOptions {
|
||||||
|
return Object.assign(new FindListOptions(), original, {
|
||||||
|
currentPage: pagination.currentPage,
|
||||||
|
elementsPerPage: pagination.pageSize
|
||||||
|
});
|
||||||
|
}
|
@@ -202,6 +202,7 @@ import { ResourcePolicyTargetResolver } from './resource-policies/resolvers/reso
|
|||||||
import { ResourcePolicyResolver } from './resource-policies/resolvers/resource-policy.resolver';
|
import { ResourcePolicyResolver } from './resource-policies/resolvers/resource-policy.resolver';
|
||||||
import { EpersonSearchBoxComponent } from './resource-policies/form/eperson-group-list/eperson-search-box/eperson-search-box.component';
|
import { EpersonSearchBoxComponent } from './resource-policies/form/eperson-group-list/eperson-search-box/eperson-search-box.component';
|
||||||
import { GroupSearchBoxComponent } from './resource-policies/form/eperson-group-list/group-search-box/group-search-box.component';
|
import { GroupSearchBoxComponent } from './resource-policies/form/eperson-group-list/group-search-box/group-search-box.component';
|
||||||
|
import { CollectionDropdownComponent } from './collection-dropdown/collection-dropdown.component';
|
||||||
|
|
||||||
const MODULES = [
|
const MODULES = [
|
||||||
// Do NOT include UniversalModule, HttpModule, or JsonpModule here
|
// Do NOT include UniversalModule, HttpModule, or JsonpModule here
|
||||||
@@ -386,7 +387,8 @@ const COMPONENTS = [
|
|||||||
ResourcePolicyFormComponent,
|
ResourcePolicyFormComponent,
|
||||||
EpersonGroupListComponent,
|
EpersonGroupListComponent,
|
||||||
EpersonSearchBoxComponent,
|
EpersonSearchBoxComponent,
|
||||||
GroupSearchBoxComponent
|
GroupSearchBoxComponent,
|
||||||
|
CollectionDropdownComponent
|
||||||
];
|
];
|
||||||
|
|
||||||
const ENTRY_COMPONENTS = [
|
const ENTRY_COMPONENTS = [
|
||||||
@@ -504,8 +506,7 @@ const DIRECTIVES = [
|
|||||||
...COMPONENTS,
|
...COMPONENTS,
|
||||||
...DIRECTIVES,
|
...DIRECTIVES,
|
||||||
...ENTRY_COMPONENTS,
|
...ENTRY_COMPONENTS,
|
||||||
...SHARED_ITEM_PAGE_COMPONENTS,
|
...SHARED_ITEM_PAGE_COMPONENTS
|
||||||
|
|
||||||
],
|
],
|
||||||
providers: [
|
providers: [
|
||||||
...PROVIDERS
|
...PROVIDERS
|
||||||
|
@@ -1,5 +1,20 @@
|
|||||||
<div>
|
<div>
|
||||||
<div ngbDropdown #collectionControls="ngbDropdown" class="btn-group input-group" (openChange)="toggled($event)">
|
<div
|
||||||
|
*ngIf="!(available$ | async)"
|
||||||
|
class="input-group mb-3">
|
||||||
|
<div class="input-group-prepend">
|
||||||
|
<span class="input-group-text">{{ 'submission.sections.general.collection' | translate }}</span>
|
||||||
|
</div>
|
||||||
|
<div class="input-group-append">
|
||||||
|
<span class="input-group-text">{{ selectedCollectionName$ | async }}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
ngbDropdown
|
||||||
|
#collectionControls="ngbDropdown"
|
||||||
|
*ngIf="(available$ | async)"
|
||||||
|
class="btn-group input-group"
|
||||||
|
(openChange)="toggled($event)">
|
||||||
<div class="input-group-prepend">
|
<div class="input-group-prepend">
|
||||||
<span id="collectionControlsMenuLabel" class="input-group-text">
|
<span id="collectionControlsMenuLabel" class="input-group-text">
|
||||||
{{ 'submission.sections.general.collection' | translate }}
|
{{ 'submission.sections.general.collection' | translate }}
|
||||||
@@ -10,7 +25,7 @@
|
|||||||
class="btn btn-outline-primary"
|
class="btn btn-outline-primary"
|
||||||
(blur)="onClose()"
|
(blur)="onClose()"
|
||||||
(click)="onClose()"
|
(click)="onClose()"
|
||||||
[disabled]="(disabled$ | async) || (processingChange$ | async)"
|
[disabled]="(processingChange$ | async)"
|
||||||
ngbDropdownToggle>
|
ngbDropdownToggle>
|
||||||
<span *ngIf="(processingChange$ | async)"><i class='fas fa-circle-notch fa-spin'></i></span>
|
<span *ngIf="(processingChange$ | async)"><i class='fas fa-circle-notch fa-spin'></i></span>
|
||||||
<span *ngIf="!(processingChange$ | async)">{{ selectedCollectionName$ | async }}</span>
|
<span *ngIf="!(processingChange$ | async)">{{ selectedCollectionName$ | async }}</span>
|
||||||
@@ -20,31 +35,9 @@
|
|||||||
class="dropdown-menu"
|
class="dropdown-menu"
|
||||||
id="collectionControlsDropdownMenu"
|
id="collectionControlsDropdownMenu"
|
||||||
aria-labelledby="collectionControlsMenuButton">
|
aria-labelledby="collectionControlsMenuButton">
|
||||||
<div class="form-group w-100 pr-2 pl-2">
|
<ds-collection-dropdown
|
||||||
<input *ngIf="searchField"
|
(selectionChange)="onSelect($event)">
|
||||||
type="search"
|
</ds-collection-dropdown>
|
||||||
class="form-control w-100"
|
|
||||||
(click)="$event.stopPropagation();"
|
|
||||||
placeholder="{{ 'submission.sections.general.search-collection' | translate }}"
|
|
||||||
[formControl]="searchField">
|
|
||||||
</div>
|
|
||||||
<div class="dropdown-divider"></div>
|
|
||||||
<div class="scrollable-menu" aria-labelledby="dropdownMenuButton" (scroll)="onScroll($event)">
|
|
||||||
<button class="dropdown-item disabled" *ngIf="(searchListCollection$ | async)?.length == 0">
|
|
||||||
{{'submission.sections.general.no-collection' | translate}}
|
|
||||||
</button>
|
|
||||||
<button *ngFor="let listItem of (searchListCollection$ | async)"
|
|
||||||
class="dropdown-item collection-item"
|
|
||||||
title="{{ listItem.collection.name }}"
|
|
||||||
(click)="onSelect(listItem)">
|
|
||||||
<ul class="list-unstyled mb-0">
|
|
||||||
<li class="list-item text-truncate text-secondary" *ngFor="let item of listItem.communities">
|
|
||||||
{{ item.name}} <i class="fa fa-level-down" aria-hidden="true"></i>
|
|
||||||
</li>
|
|
||||||
<li class="list-item text-truncate text-primary font-weight-bold">{{ listItem.collection.name}}</li>
|
|
||||||
</ul>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@@ -1,14 +1,11 @@
|
|||||||
import { ChangeDetectorRef, Component, CUSTOM_ELEMENTS_SCHEMA, DebugElement, SimpleChange } from '@angular/core';
|
import { ChangeDetectorRef, Component, CUSTOM_ELEMENTS_SCHEMA, DebugElement } from '@angular/core';
|
||||||
import { async, ComponentFixture, fakeAsync, inject, TestBed, tick } from '@angular/core/testing';
|
import { async, ComponentFixture, fakeAsync, inject, TestBed, tick } from '@angular/core/testing';
|
||||||
import { By } from '@angular/platform-browser';
|
import { By } from '@angular/platform-browser';
|
||||||
import { FormsModule, ReactiveFormsModule } from '@angular/forms';
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms';
|
||||||
|
|
||||||
import { of as observableOf } from 'rxjs';
|
|
||||||
import { filter } from 'rxjs/operators';
|
|
||||||
import { TranslateModule } from '@ngx-translate/core';
|
import { TranslateModule } from '@ngx-translate/core';
|
||||||
import { NgbModule } from '@ng-bootstrap/ng-bootstrap';
|
import { NgbModule } from '@ng-bootstrap/ng-bootstrap';
|
||||||
import { Store } from '@ngrx/store';
|
import { Store } from '@ngrx/store';
|
||||||
import { cold } from 'jasmine-marbles';
|
|
||||||
|
|
||||||
import { SubmissionServiceStub } from '../../../shared/testing/submission-service.stub';
|
import { SubmissionServiceStub } from '../../../shared/testing/submission-service.stub';
|
||||||
import { mockSubmissionId, mockSubmissionRestResponse } from '../../../shared/mocks/submission.mock';
|
import { mockSubmissionId, mockSubmissionRestResponse } from '../../../shared/mocks/submission.mock';
|
||||||
@@ -19,123 +16,28 @@ import { SubmissionJsonPatchOperationsService } from '../../../core/submission/s
|
|||||||
import { SubmissionJsonPatchOperationsServiceStub } from '../../../shared/testing/submission-json-patch-operations-service.stub';
|
import { SubmissionJsonPatchOperationsServiceStub } from '../../../shared/testing/submission-json-patch-operations-service.stub';
|
||||||
import { JsonPatchOperationsBuilder } from '../../../core/json-patch/builder/json-patch-operations-builder';
|
import { JsonPatchOperationsBuilder } from '../../../core/json-patch/builder/json-patch-operations-builder';
|
||||||
import { JsonPatchOperationPathCombiner } from '../../../core/json-patch/builder/json-patch-operation-path-combiner';
|
import { JsonPatchOperationPathCombiner } from '../../../core/json-patch/builder/json-patch-operation-path-combiner';
|
||||||
import { RemoteData } from '../../../core/data/remote-data';
|
|
||||||
import { Community } from '../../../core/shared/community.model';
|
|
||||||
import { PaginatedList } from '../../../core/data/paginated-list';
|
|
||||||
import { PageInfo } from '../../../core/shared/page-info.model';
|
|
||||||
import { Collection } from '../../../core/shared/collection.model';
|
|
||||||
import { createTestComponent } from '../../../shared/testing/utils.test';
|
import { createTestComponent } from '../../../shared/testing/utils.test';
|
||||||
import { CollectionDataService } from '../../../core/data/collection-data.service';
|
import { CollectionDataService } from '../../../core/data/collection-data.service';
|
||||||
|
import { hot, cold } from 'jasmine-marbles';
|
||||||
|
import { of } from 'rxjs';
|
||||||
|
import { SectionsService } from '../../sections/sections.service';
|
||||||
|
import { componentFactoryName } from '@angular/compiler';
|
||||||
|
import { Collection } from 'src/app/core/shared/collection.model';
|
||||||
|
|
||||||
const subcommunities = [Object.assign(new Community(), {
|
describe('SubmissionFormCollectionComponent Component', () => {
|
||||||
name: 'SubCommunity 1',
|
|
||||||
id: '123456789-1',
|
|
||||||
metadata: [
|
|
||||||
{
|
|
||||||
key: 'dc.title',
|
|
||||||
language: 'en_US',
|
|
||||||
value: 'SubCommunity 1'
|
|
||||||
}]
|
|
||||||
}),
|
|
||||||
Object.assign(new Community(), {
|
|
||||||
name: 'SubCommunity 1',
|
|
||||||
id: '123456789s-1',
|
|
||||||
metadata: [
|
|
||||||
{
|
|
||||||
key: 'dc.title',
|
|
||||||
language: 'en_US',
|
|
||||||
value: 'SubCommunity 1'
|
|
||||||
}]
|
|
||||||
})
|
|
||||||
];
|
|
||||||
|
|
||||||
const mockCommunity1Collection1 = Object.assign(new Collection(), {
|
let comp: SubmissionFormCollectionComponent;
|
||||||
name: 'Community 1-Collection 1',
|
let compAsAny: any;
|
||||||
id: '1234567890-1',
|
let fixture: ComponentFixture<SubmissionFormCollectionComponent>;
|
||||||
metadata: [
|
let submissionServiceStub: SubmissionServiceStub;
|
||||||
{
|
let jsonPatchOpServiceStub: SubmissionJsonPatchOperationsServiceStub;
|
||||||
key: 'dc.title',
|
|
||||||
language: 'en_US',
|
|
||||||
value: 'Community 1-Collection 1'
|
|
||||||
}]
|
|
||||||
});
|
|
||||||
|
|
||||||
const mockCommunity1Collection2 = Object.assign(new Collection(), {
|
const submissionId = mockSubmissionId;
|
||||||
name: 'Community 1-Collection 2',
|
const collectionId = '1234567890-1';
|
||||||
id: '1234567890-2',
|
const definition = 'traditional';
|
||||||
metadata: [
|
const submissionRestResponse = mockSubmissionRestResponse;
|
||||||
{
|
|
||||||
key: 'dc.title',
|
|
||||||
language: 'en_US',
|
|
||||||
value: 'Community 1-Collection 2'
|
|
||||||
}]
|
|
||||||
});
|
|
||||||
|
|
||||||
const mockCommunity2Collection1 = Object.assign(new Collection(), {
|
const mockCollectionList = [
|
||||||
name: 'Community 2-Collection 1',
|
|
||||||
id: '1234567890-3',
|
|
||||||
metadata: [
|
|
||||||
{
|
|
||||||
key: 'dc.title',
|
|
||||||
language: 'en_US',
|
|
||||||
value: 'Community 2-Collection 1'
|
|
||||||
}]
|
|
||||||
});
|
|
||||||
|
|
||||||
const mockCommunity2Collection2 = Object.assign(new Collection(), {
|
|
||||||
name: 'Community 2-Collection 2',
|
|
||||||
id: '1234567890-4',
|
|
||||||
metadata: [
|
|
||||||
{
|
|
||||||
key: 'dc.title',
|
|
||||||
language: 'en_US',
|
|
||||||
value: 'Community 2-Collection 2'
|
|
||||||
}]
|
|
||||||
});
|
|
||||||
|
|
||||||
const mockCommunity = Object.assign(new Community(), {
|
|
||||||
name: 'Community 1',
|
|
||||||
id: '123456789-1',
|
|
||||||
metadata: [
|
|
||||||
{
|
|
||||||
key: 'dc.title',
|
|
||||||
language: 'en_US',
|
|
||||||
value: 'Community 1'
|
|
||||||
}],
|
|
||||||
collections: observableOf(new RemoteData(true, true, true,
|
|
||||||
undefined, new PaginatedList(new PageInfo(), [mockCommunity1Collection1, mockCommunity1Collection2]))),
|
|
||||||
subcommunities: observableOf(new RemoteData(true, true, true,
|
|
||||||
undefined, new PaginatedList(new PageInfo(), subcommunities))),
|
|
||||||
});
|
|
||||||
|
|
||||||
const mockCommunity2 = Object.assign(new Community(), {
|
|
||||||
name: 'Community 2',
|
|
||||||
id: '123456789-2',
|
|
||||||
metadata: [
|
|
||||||
{
|
|
||||||
key: 'dc.title',
|
|
||||||
language: 'en_US',
|
|
||||||
value: 'Community 2'
|
|
||||||
}],
|
|
||||||
collections: observableOf(new RemoteData(true, true, true,
|
|
||||||
undefined, new PaginatedList(new PageInfo(), [mockCommunity2Collection1, mockCommunity2Collection2]))),
|
|
||||||
subcommunities: observableOf(new RemoteData(true, true, true,
|
|
||||||
undefined, new PaginatedList(new PageInfo(), []))),
|
|
||||||
});
|
|
||||||
|
|
||||||
const mockCommunity1Collection1Rd = observableOf(new RemoteData(true, true, true,
|
|
||||||
undefined, mockCommunity1Collection1));
|
|
||||||
|
|
||||||
const mockCommunityList = observableOf(new RemoteData(true, true, true,
|
|
||||||
undefined, new PaginatedList(new PageInfo(), [mockCommunity, mockCommunity2])));
|
|
||||||
|
|
||||||
const mockCommunityCollectionList = observableOf(new RemoteData(true, true, true,
|
|
||||||
undefined, new PaginatedList(new PageInfo(), [mockCommunity1Collection1, mockCommunity1Collection2])));
|
|
||||||
|
|
||||||
const mockCommunity2CollectionList = observableOf(new RemoteData(true, true, true,
|
|
||||||
undefined, new PaginatedList(new PageInfo(), [mockCommunity2Collection1, mockCommunity2Collection2])));
|
|
||||||
|
|
||||||
const mockCollectionList = [
|
|
||||||
{
|
{
|
||||||
communities: [
|
communities: [
|
||||||
{
|
{
|
||||||
@@ -184,21 +86,7 @@ const mockCollectionList = [
|
|||||||
name: 'Community 2-Collection 2'
|
name: 'Community 2-Collection 2'
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
];
|
];
|
||||||
|
|
||||||
describe('SubmissionFormCollectionComponent Component', () => {
|
|
||||||
|
|
||||||
let comp: SubmissionFormCollectionComponent;
|
|
||||||
let compAsAny: any;
|
|
||||||
let fixture: ComponentFixture<SubmissionFormCollectionComponent>;
|
|
||||||
let submissionServiceStub: SubmissionServiceStub;
|
|
||||||
let jsonPatchOpServiceStub: SubmissionJsonPatchOperationsServiceStub;
|
|
||||||
|
|
||||||
const submissionId = mockSubmissionId;
|
|
||||||
const collectionId = '1234567890-1';
|
|
||||||
const definition = 'traditional';
|
|
||||||
const submissionRestResponse = mockSubmissionRestResponse;
|
|
||||||
const searchedCollection = 'Community 2-Collection 2';
|
|
||||||
|
|
||||||
const communityDataService: any = jasmine.createSpyObj('communityDataService', {
|
const communityDataService: any = jasmine.createSpyObj('communityDataService', {
|
||||||
findAll: jasmine.createSpy('findAll')
|
findAll: jasmine.createSpy('findAll')
|
||||||
@@ -217,6 +105,10 @@ describe('SubmissionFormCollectionComponent Component', () => {
|
|||||||
replace: jasmine.createSpy('replace')
|
replace: jasmine.createSpy('replace')
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const sectionsService: any = jasmine.createSpyObj('sectionsService', {
|
||||||
|
isSectionAvailable: of(true)
|
||||||
|
});
|
||||||
|
|
||||||
beforeEach(async(() => {
|
beforeEach(async(() => {
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
imports: [
|
imports: [
|
||||||
@@ -236,6 +128,7 @@ describe('SubmissionFormCollectionComponent Component', () => {
|
|||||||
{ provide: CommunityDataService, useValue: communityDataService },
|
{ provide: CommunityDataService, useValue: communityDataService },
|
||||||
{ provide: JsonPatchOperationsBuilder, useValue: jsonPatchOpBuilder },
|
{ provide: JsonPatchOperationsBuilder, useValue: jsonPatchOpBuilder },
|
||||||
{ provide: Store, useValue: store },
|
{ provide: Store, useValue: store },
|
||||||
|
{ provide: SectionsService, useValue: sectionsService },
|
||||||
ChangeDetectorRef,
|
ChangeDetectorRef,
|
||||||
SubmissionFormCollectionComponent
|
SubmissionFormCollectionComponent
|
||||||
],
|
],
|
||||||
@@ -299,72 +192,11 @@ describe('SubmissionFormCollectionComponent Component', () => {
|
|||||||
expect(compAsAny.pathCombiner).toEqual(expected);
|
expect(compAsAny.pathCombiner).toEqual(expected);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should init collection list properly', () => {
|
|
||||||
communityDataService.findAll.and.returnValue(mockCommunityList);
|
|
||||||
collectionDataService.findById.and.returnValue(mockCommunity1Collection1Rd);
|
|
||||||
collectionDataService.getAuthorizedCollectionByCommunity.and.returnValues(mockCommunityCollectionList, mockCommunity2CollectionList);
|
|
||||||
|
|
||||||
comp.ngOnChanges({
|
|
||||||
currentCollectionId: new SimpleChange(null, collectionId, true)
|
|
||||||
});
|
|
||||||
|
|
||||||
expect(comp.searchListCollection$).toBeObservable(cold('(ab)', {
|
|
||||||
a: [],
|
|
||||||
b: mockCollectionList
|
|
||||||
}));
|
|
||||||
|
|
||||||
expect(comp.selectedCollectionName$).toBeObservable(cold('(a|)', {
|
|
||||||
a: 'Community 1-Collection 1'
|
|
||||||
}));
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should show only the searched collection', () => {
|
|
||||||
comp.searchListCollection$ = observableOf(mockCollectionList);
|
|
||||||
fixture.detectChanges();
|
|
||||||
|
|
||||||
comp.searchField.setValue(searchedCollection);
|
|
||||||
fixture.detectChanges();
|
|
||||||
|
|
||||||
comp.searchListCollection$.pipe(
|
|
||||||
filter(() => !comp.disabled$.getValue())
|
|
||||||
).subscribe((list) => {
|
|
||||||
expect(list).toEqual([mockCollectionList[3]]);
|
|
||||||
});
|
|
||||||
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should emit collectionChange event when selecting a new collection', () => {
|
|
||||||
spyOn(comp.searchField, 'reset').and.callThrough();
|
|
||||||
spyOn(comp.collectionChange, 'emit').and.callThrough();
|
|
||||||
jsonPatchOpServiceStub.jsonPatchByResourceID.and.returnValue(observableOf(submissionRestResponse));
|
|
||||||
comp.ngOnInit();
|
|
||||||
comp.onSelect(mockCollectionList[1]);
|
|
||||||
fixture.detectChanges();
|
|
||||||
|
|
||||||
expect(comp.searchField.reset).toHaveBeenCalled();
|
|
||||||
expect(comp.collectionChange.emit).toHaveBeenCalledWith(submissionRestResponse[0] as any);
|
|
||||||
expect(submissionServiceStub.changeSubmissionCollection).toHaveBeenCalled();
|
|
||||||
expect(comp.selectedCollectionId).toBe(mockCollectionList[1].collection.id);
|
|
||||||
expect(comp.selectedCollectionName$).toBeObservable(cold('(a|)', {
|
|
||||||
a: mockCollectionList[1].collection.name
|
|
||||||
}));
|
|
||||||
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should reset searchField when dropdown menu has been closed', () => {
|
|
||||||
spyOn(comp.searchField, 'reset').and.callThrough();
|
|
||||||
comp.toggled(false);
|
|
||||||
|
|
||||||
expect(comp.searchField.reset).toHaveBeenCalled();
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('', () => {
|
describe('', () => {
|
||||||
let dropdowBtn: DebugElement;
|
let dropdowBtn: DebugElement;
|
||||||
let dropdownMenu: DebugElement;
|
let dropdownMenu: DebugElement;
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
|
|
||||||
comp.searchListCollection$ = observableOf(mockCollectionList);
|
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
dropdowBtn = fixture.debugElement.query(By.css('#collectionControlsMenuButton'));
|
dropdowBtn = fixture.debugElement.query(By.css('#collectionControlsMenuButton'));
|
||||||
dropdownMenu = fixture.debugElement.query(By.css('#collectionControlsDropdownMenu'));
|
dropdownMenu = fixture.debugElement.query(By.css('#collectionControlsDropdownMenu'));
|
||||||
@@ -387,49 +219,46 @@ describe('SubmissionFormCollectionComponent Component', () => {
|
|||||||
fixture.whenStable().then(() => {
|
fixture.whenStable().then(() => {
|
||||||
expect(comp.onClose).toHaveBeenCalled();
|
expect(comp.onClose).toHaveBeenCalled();
|
||||||
expect(dropdownMenu.nativeElement.classList).toContain('show');
|
expect(dropdownMenu.nativeElement.classList).toContain('show');
|
||||||
expect(dropdownMenu.queryAll(By.css('.collection-item')).length).toBe(4);
|
|
||||||
});
|
});
|
||||||
}));
|
}));
|
||||||
|
|
||||||
it('should trigger onSelect method when select a new collection from dropdown menu', fakeAsync(() => {
|
it('the dropdown menu should be enable', () => {
|
||||||
|
const dropDown = fixture.debugElement.query(By.css('#collectionControlsDropdownMenu'));
|
||||||
spyOn(comp, 'onSelect');
|
expect(dropDown).toBeTruthy();
|
||||||
dropdowBtn.triggerEventHandler('click', null);
|
|
||||||
tick();
|
|
||||||
fixture.detectChanges();
|
|
||||||
|
|
||||||
const secondLink: DebugElement = dropdownMenu.query(By.css('.collection-item:nth-child(2)'));
|
|
||||||
secondLink.triggerEventHandler('click', null);
|
|
||||||
tick();
|
|
||||||
fixture.detectChanges();
|
|
||||||
|
|
||||||
fixture.whenStable().then(() => {
|
|
||||||
|
|
||||||
expect(comp.onSelect).toHaveBeenCalled();
|
|
||||||
});
|
});
|
||||||
}));
|
|
||||||
|
|
||||||
it('should update searchField on input type', fakeAsync(() => {
|
it('the dropdown menu should be disabled', () => {
|
||||||
|
comp.available$ = of(false);
|
||||||
dropdowBtn.triggerEventHandler('click', null);
|
|
||||||
tick();
|
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
|
const dropDown = fixture.debugElement.query(By.css('#collectionControlsDropdownMenu'));
|
||||||
fixture.whenStable().then(() => {
|
expect(dropDown).toBeFalsy();
|
||||||
const input = fixture.debugElement.query(By.css('input.form-control'));
|
|
||||||
const el = input.nativeElement;
|
|
||||||
|
|
||||||
expect(el.value).toBe('');
|
|
||||||
|
|
||||||
el.value = searchedCollection;
|
|
||||||
el.dispatchEvent(new Event('input'));
|
|
||||||
|
|
||||||
fixture.detectChanges();
|
|
||||||
|
|
||||||
expect(fixture.componentInstance.searchField.value).toEqual(searchedCollection);
|
|
||||||
});
|
});
|
||||||
}));
|
|
||||||
|
|
||||||
|
it('should be simulated when the drop-down menu is closed', () => {
|
||||||
|
spyOn(comp, 'onClose');
|
||||||
|
comp.onClose();
|
||||||
|
expect(comp.onClose).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should be simulated when the drop-down menu is toggled', () => {
|
||||||
|
spyOn(comp, 'toggled');
|
||||||
|
comp.toggled(false);
|
||||||
|
expect(comp.toggled).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should ', () => {
|
||||||
|
spyOn(comp.collectionChange, 'emit').and.callThrough();
|
||||||
|
jsonPatchOpServiceStub.jsonPatchByResourceID.and.returnValue(of(submissionRestResponse));
|
||||||
|
comp.ngOnInit();
|
||||||
|
comp.onSelect(mockCollectionList[1]);
|
||||||
|
fixture.detectChanges();
|
||||||
|
|
||||||
|
expect(submissionServiceStub.changeSubmissionCollection).toHaveBeenCalled();
|
||||||
|
expect(comp.selectedCollectionId).toBe(mockCollectionList[1].collection.id);
|
||||||
|
expect(comp.selectedCollectionName$).toBeObservable(cold('(a|)', {
|
||||||
|
a: mockCollectionList[1].collection.name
|
||||||
|
}));
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
});
|
});
|
||||||
|
@@ -2,57 +2,31 @@ import {
|
|||||||
ChangeDetectorRef,
|
ChangeDetectorRef,
|
||||||
Component,
|
Component,
|
||||||
EventEmitter,
|
EventEmitter,
|
||||||
HostListener,
|
|
||||||
Input,
|
Input,
|
||||||
OnChanges,
|
OnChanges,
|
||||||
OnInit,
|
OnInit,
|
||||||
Output,
|
Output,
|
||||||
SimpleChanges
|
SimpleChanges,
|
||||||
|
ViewChild
|
||||||
} from '@angular/core';
|
} from '@angular/core';
|
||||||
import { FormControl } from '@angular/forms';
|
|
||||||
|
|
||||||
import { BehaviorSubject, combineLatest, Observable, of as observableOf, Subscription } from 'rxjs';
|
import { BehaviorSubject, Observable, of as observableOf, Subscription } from 'rxjs';
|
||||||
import {
|
import {
|
||||||
debounceTime,
|
|
||||||
distinctUntilChanged,
|
|
||||||
filter,
|
|
||||||
find,
|
find,
|
||||||
flatMap,
|
map
|
||||||
map,
|
|
||||||
mergeMap,
|
|
||||||
reduce,
|
|
||||||
startWith
|
|
||||||
} from 'rxjs/operators';
|
} from 'rxjs/operators';
|
||||||
|
|
||||||
import { Collection } from '../../../core/shared/collection.model';
|
import { Collection } from '../../../core/shared/collection.model';
|
||||||
import { CommunityDataService } from '../../../core/data/community-data.service';
|
import { hasValue, isNotEmpty } from '../../../shared/empty.util';
|
||||||
import { Community } from '../../../core/shared/community.model';
|
|
||||||
import { hasValue, isEmpty, isNotEmpty } from '../../../shared/empty.util';
|
|
||||||
import { RemoteData } from '../../../core/data/remote-data';
|
import { RemoteData } from '../../../core/data/remote-data';
|
||||||
import { JsonPatchOperationPathCombiner } from '../../../core/json-patch/builder/json-patch-operation-path-combiner';
|
import { JsonPatchOperationPathCombiner } from '../../../core/json-patch/builder/json-patch-operation-path-combiner';
|
||||||
import { JsonPatchOperationsBuilder } from '../../../core/json-patch/builder/json-patch-operations-builder';
|
import { JsonPatchOperationsBuilder } from '../../../core/json-patch/builder/json-patch-operations-builder';
|
||||||
import { PaginatedList } from '../../../core/data/paginated-list';
|
|
||||||
import { SubmissionService } from '../../submission.service';
|
import { SubmissionService } from '../../submission.service';
|
||||||
import { SubmissionObject } from '../../../core/submission/models/submission-object.model';
|
import { SubmissionObject } from '../../../core/submission/models/submission-object.model';
|
||||||
import { SubmissionJsonPatchOperationsService } from '../../../core/submission/submission-json-patch-operations.service';
|
import { SubmissionJsonPatchOperationsService } from '../../../core/submission/submission-json-patch-operations.service';
|
||||||
import { CollectionDataService } from '../../../core/data/collection-data.service';
|
import { CollectionDataService } from '../../../core/data/collection-data.service';
|
||||||
import { FindListOptions } from '../../../core/data/request.models';
|
import { CollectionDropdownComponent } from 'src/app/shared/collection-dropdown/collection-dropdown.component';
|
||||||
|
import { SectionsService } from '../../sections/sections.service';
|
||||||
/**
|
|
||||||
* An interface to represent a collection entry
|
|
||||||
*/
|
|
||||||
interface CollectionListEntryItem {
|
|
||||||
id: string;
|
|
||||||
name: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* An interface to represent an entry in the collection list
|
|
||||||
*/
|
|
||||||
interface CollectionListEntry {
|
|
||||||
communities: CollectionListEntryItem[],
|
|
||||||
collection: CollectionListEntryItem
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This component allows to show the current collection the submission belonging to and to change it.
|
* This component allows to show the current collection the submission belonging to and to change it.
|
||||||
@@ -88,30 +62,12 @@ export class SubmissionFormCollectionComponent implements OnChanges, OnInit {
|
|||||||
*/
|
*/
|
||||||
@Output() collectionChange: EventEmitter<SubmissionObject> = new EventEmitter<SubmissionObject>();
|
@Output() collectionChange: EventEmitter<SubmissionObject> = new EventEmitter<SubmissionObject>();
|
||||||
|
|
||||||
/**
|
|
||||||
* A boolean representing if this dropdown button is disabled
|
|
||||||
* @type {BehaviorSubject<boolean>}
|
|
||||||
*/
|
|
||||||
public disabled$ = new BehaviorSubject<boolean>(true);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A boolean representing if a collection change operation is processing
|
* A boolean representing if a collection change operation is processing
|
||||||
* @type {BehaviorSubject<boolean>}
|
* @type {BehaviorSubject<boolean>}
|
||||||
*/
|
*/
|
||||||
public processingChange$ = new BehaviorSubject<boolean>(false);
|
public processingChange$ = new BehaviorSubject<boolean>(false);
|
||||||
|
|
||||||
/**
|
|
||||||
* The search form control
|
|
||||||
* @type {FormControl}
|
|
||||||
*/
|
|
||||||
public searchField: FormControl = new FormControl();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The collection list obtained from a search
|
|
||||||
* @type {Observable<CollectionListEntry[]>}
|
|
||||||
*/
|
|
||||||
public searchListCollection$: Observable<CollectionListEntry[]>;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The selected collection id
|
* The selected collection id
|
||||||
* @type {string}
|
* @type {string}
|
||||||
@@ -130,24 +86,23 @@ export class SubmissionFormCollectionComponent implements OnChanges, OnInit {
|
|||||||
*/
|
*/
|
||||||
protected pathCombiner: JsonPatchOperationPathCombiner;
|
protected pathCombiner: JsonPatchOperationPathCombiner;
|
||||||
|
|
||||||
/**
|
|
||||||
* A boolean representing if dropdown list is scrollable to the bottom
|
|
||||||
* @type {boolean}
|
|
||||||
*/
|
|
||||||
private scrollableBottom = false;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A boolean representing if dropdown list is scrollable to the top
|
|
||||||
* @type {boolean}
|
|
||||||
*/
|
|
||||||
private scrollableTop = false;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Array to track all subscriptions and unsubscribe them onDestroy
|
* Array to track all subscriptions and unsubscribe them onDestroy
|
||||||
* @type {Array}
|
* @type {Array}
|
||||||
*/
|
*/
|
||||||
private subs: Subscription[] = [];
|
private subs: Subscription[] = [];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The html child that contains the collections list
|
||||||
|
*/
|
||||||
|
@ViewChild(CollectionDropdownComponent, {static: false}) collectionDropdown: CollectionDropdownComponent;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A boolean representing if the collection section is available
|
||||||
|
* @type {BehaviorSubject<boolean>}
|
||||||
|
*/
|
||||||
|
available$: Observable<boolean>;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Initialize instance variables
|
* Initialize instance variables
|
||||||
*
|
*
|
||||||
@@ -159,37 +114,11 @@ export class SubmissionFormCollectionComponent implements OnChanges, OnInit {
|
|||||||
* @param {SubmissionService} submissionService
|
* @param {SubmissionService} submissionService
|
||||||
*/
|
*/
|
||||||
constructor(protected cdr: ChangeDetectorRef,
|
constructor(protected cdr: ChangeDetectorRef,
|
||||||
private communityDataService: CommunityDataService,
|
|
||||||
private collectionDataService: CollectionDataService,
|
private collectionDataService: CollectionDataService,
|
||||||
private operationsBuilder: JsonPatchOperationsBuilder,
|
private operationsBuilder: JsonPatchOperationsBuilder,
|
||||||
private operationsService: SubmissionJsonPatchOperationsService,
|
private operationsService: SubmissionJsonPatchOperationsService,
|
||||||
private submissionService: SubmissionService) {
|
private submissionService: SubmissionService,
|
||||||
}
|
private sectionsService: SectionsService) {
|
||||||
|
|
||||||
/**
|
|
||||||
* Method called on mousewheel event, it prevent the page scroll
|
|
||||||
* when arriving at the top/bottom of dropdown menu
|
|
||||||
*
|
|
||||||
* @param event
|
|
||||||
* mousewheel event
|
|
||||||
*/
|
|
||||||
@HostListener('mousewheel', ['$event']) onMousewheel(event) {
|
|
||||||
if (event.wheelDelta > 0 && this.scrollableTop) {
|
|
||||||
event.preventDefault();
|
|
||||||
}
|
|
||||||
if (event.wheelDelta < 0 && this.scrollableBottom) {
|
|
||||||
event.preventDefault();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Check if dropdown scrollbar is at the top or bottom of the dropdown list
|
|
||||||
*
|
|
||||||
* @param event
|
|
||||||
*/
|
|
||||||
onScroll(event) {
|
|
||||||
this.scrollableBottom = (event.target.scrollTop + event.target.clientHeight === event.target.scrollHeight);
|
|
||||||
this.scrollableTop = (event.target.scrollTop === 0);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -204,51 +133,6 @@ export class SubmissionFormCollectionComponent implements OnChanges, OnInit {
|
|||||||
find((collectionRD: RemoteData<Collection>) => isNotEmpty(collectionRD.payload)),
|
find((collectionRD: RemoteData<Collection>) => isNotEmpty(collectionRD.payload)),
|
||||||
map((collectionRD: RemoteData<Collection>) => collectionRD.payload.name)
|
map((collectionRD: RemoteData<Collection>) => collectionRD.payload.name)
|
||||||
);
|
);
|
||||||
|
|
||||||
const findOptions: FindListOptions = {
|
|
||||||
elementsPerPage: 1000
|
|
||||||
};
|
|
||||||
|
|
||||||
// Retrieve collection list only when is the first change
|
|
||||||
if (changes.currentCollectionId.isFirstChange()) {
|
|
||||||
// @TODO replace with search/top browse endpoint
|
|
||||||
// @TODO implement community/subcommunity hierarchy
|
|
||||||
const communities$ = this.communityDataService.findAll(findOptions).pipe(
|
|
||||||
find((communities: RemoteData<PaginatedList<Community>>) => isNotEmpty(communities.payload)),
|
|
||||||
mergeMap((communities: RemoteData<PaginatedList<Community>>) => communities.payload.page));
|
|
||||||
|
|
||||||
const listCollection$ = communities$.pipe(
|
|
||||||
flatMap((communityData: Community) => {
|
|
||||||
return this.collectionDataService.getAuthorizedCollectionByCommunity(communityData.uuid, findOptions).pipe(
|
|
||||||
find((collections: RemoteData<PaginatedList<Collection>>) => !collections.isResponsePending && collections.hasSucceeded),
|
|
||||||
mergeMap((collections: RemoteData<PaginatedList<Collection>>) => collections.payload.page),
|
|
||||||
filter((collectionData: Collection) => isNotEmpty(collectionData)),
|
|
||||||
map((collectionData: Collection) => ({
|
|
||||||
communities: [{ id: communityData.id, name: communityData.name }],
|
|
||||||
collection: { id: collectionData.id, name: collectionData.name }
|
|
||||||
}))
|
|
||||||
);
|
|
||||||
}),
|
|
||||||
reduce((acc: any, value: any) => [...acc, ...value], []),
|
|
||||||
startWith([])
|
|
||||||
);
|
|
||||||
|
|
||||||
const searchTerm$ = this.searchField.valueChanges.pipe(
|
|
||||||
debounceTime(200),
|
|
||||||
distinctUntilChanged(),
|
|
||||||
startWith('')
|
|
||||||
);
|
|
||||||
|
|
||||||
this.searchListCollection$ = combineLatest(searchTerm$, listCollection$).pipe(
|
|
||||||
map(([searchTerm, listCollection]) => {
|
|
||||||
this.disabled$.next(isEmpty(listCollection));
|
|
||||||
if (isEmpty(searchTerm)) {
|
|
||||||
return listCollection;
|
|
||||||
} else {
|
|
||||||
return listCollection.filter((v) => v.collection.name.toLowerCase().indexOf(searchTerm.toLowerCase()) > -1).slice(0, 5);
|
|
||||||
}
|
|
||||||
}));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -257,6 +141,7 @@ export class SubmissionFormCollectionComponent implements OnChanges, OnInit {
|
|||||||
*/
|
*/
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.pathCombiner = new JsonPatchOperationPathCombiner('sections', 'collection');
|
this.pathCombiner = new JsonPatchOperationPathCombiner('sections', 'collection');
|
||||||
|
this.available$ = this.sectionsService.isSectionAvailable(this.submissionId, 'collection');
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -273,7 +158,6 @@ export class SubmissionFormCollectionComponent implements OnChanges, OnInit {
|
|||||||
* the selected [CollectionListEntryItem]
|
* the selected [CollectionListEntryItem]
|
||||||
*/
|
*/
|
||||||
onSelect(event) {
|
onSelect(event) {
|
||||||
this.searchField.reset();
|
|
||||||
this.processingChange$.next(true);
|
this.processingChange$.next(true);
|
||||||
this.operationsBuilder.replace(this.pathCombiner.getPath(), event.collection.id, true);
|
this.operationsBuilder.replace(this.pathCombiner.getPath(), event.collection.id, true);
|
||||||
this.subs.push(this.operationsService.jsonPatchByResourceID(
|
this.subs.push(this.operationsService.jsonPatchByResourceID(
|
||||||
@@ -296,7 +180,7 @@ export class SubmissionFormCollectionComponent implements OnChanges, OnInit {
|
|||||||
* Reset search form control on dropdown menu close
|
* Reset search form control on dropdown menu close
|
||||||
*/
|
*/
|
||||||
onClose() {
|
onClose() {
|
||||||
this.searchField.reset();
|
this.collectionDropdown.reset();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -307,7 +191,7 @@ export class SubmissionFormCollectionComponent implements OnChanges, OnInit {
|
|||||||
*/
|
*/
|
||||||
toggled(isOpen: boolean) {
|
toggled(isOpen: boolean) {
|
||||||
if (!isOpen) {
|
if (!isOpen) {
|
||||||
this.searchField.reset();
|
this.collectionDropdown.reset();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -8,6 +8,10 @@
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
"admin.registries.bitstream-formats.breadcrumbs": "Format registry",
|
||||||
|
|
||||||
|
"admin.registries.bitstream-formats.create.breadcrumbs": "Bitstream format",
|
||||||
|
|
||||||
"admin.registries.bitstream-formats.create.failure.content": "An error occurred while creating the new bitstream format.",
|
"admin.registries.bitstream-formats.create.failure.content": "An error occurred while creating the new bitstream format.",
|
||||||
|
|
||||||
"admin.registries.bitstream-formats.create.failure.head": "Failure",
|
"admin.registries.bitstream-formats.create.failure.head": "Failure",
|
||||||
@@ -30,6 +34,8 @@
|
|||||||
|
|
||||||
"admin.registries.bitstream-formats.description": "This list of bitstream formats provides information about known formats and their support level.",
|
"admin.registries.bitstream-formats.description": "This list of bitstream formats provides information about known formats and their support level.",
|
||||||
|
|
||||||
|
"admin.registries.bitstream-formats.edit.breadcrumbs": "Bitstream format",
|
||||||
|
|
||||||
"admin.registries.bitstream-formats.edit.description.hint": "",
|
"admin.registries.bitstream-formats.edit.description.hint": "",
|
||||||
|
|
||||||
"admin.registries.bitstream-formats.edit.description.label": "Description",
|
"admin.registries.bitstream-formats.edit.description.label": "Description",
|
||||||
@@ -94,6 +100,8 @@
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
"admin.registries.metadata.breadcrumbs": "Metadata registry",
|
||||||
|
|
||||||
"admin.registries.metadata.description": "The metadata registry maintains a list of all metadata fields available in the repository. These fields may be divided amongst multiple schemas. However, DSpace requires the qualified Dublin Core schema.",
|
"admin.registries.metadata.description": "The metadata registry maintains a list of all metadata fields available in the repository. These fields may be divided amongst multiple schemas. However, DSpace requires the qualified Dublin Core schema.",
|
||||||
|
|
||||||
"admin.registries.metadata.form.create": "Create metadata schema",
|
"admin.registries.metadata.form.create": "Create metadata schema",
|
||||||
@@ -120,6 +128,8 @@
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
"admin.registries.schema.breadcrumbs": "Metadata schema",
|
||||||
|
|
||||||
"admin.registries.schema.description": "This is the metadata schema for \"{{namespace}}\".",
|
"admin.registries.schema.description": "This is the metadata schema for \"{{namespace}}\".",
|
||||||
|
|
||||||
"admin.registries.schema.fields.head": "Schema metadata fields",
|
"admin.registries.schema.fields.head": "Schema metadata fields",
|
||||||
@@ -1741,7 +1751,7 @@
|
|||||||
|
|
||||||
"mydspace.description": "",
|
"mydspace.description": "",
|
||||||
|
|
||||||
"mydspace.general.text-here": "HERE",
|
"mydspace.general.text-here": "here",
|
||||||
|
|
||||||
"mydspace.messages.controller-help": "Select this option to send a message to item's submitter.",
|
"mydspace.messages.controller-help": "Select this option to send a message to item's submitter.",
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user