mirror of
https://github.com/DSpace/dspace-angular.git
synced 2025-10-10 19:43:04 +00:00

# Conflicts: # src/app/+admin/admin-access-control/epeople-registry/epeople-registry.component.spec.ts # src/app/+admin/admin-access-control/epeople-registry/epeople-registry.component.ts # src/app/+admin/admin-access-control/epeople-registry/eperson-form/eperson-form.component.spec.ts # src/app/+admin/admin-access-control/group-registry/group-form/group-form.component.spec.ts # src/app/+admin/admin-access-control/group-registry/group-form/members-list/members-list.component.spec.ts # src/app/+admin/admin-access-control/group-registry/group-form/subgroup-list/subgroups-list.component.spec.ts # src/app/+admin/admin-access-control/group-registry/groups-registry.component.ts # src/app/+admin/admin-import-metadata-page/metadata-import-page.component.ts # src/app/+admin/admin-registries/metadata-registry/metadata-registry.component.ts # src/app/+admin/admin-registries/metadata-schema/metadata-schema.component.ts # src/app/+admin/admin-search-page/admin-search-results/admin-search-result-grid-element/item-search-result/item-admin-search-result-grid-element.component.spec.ts # src/app/+admin/admin-search-page/admin-search-results/admin-search-result-list-element/item-search-result/item-admin-search-result-list-element.component.spec.ts # src/app/+bitstream-page/bitstream-page.resolver.ts # src/app/+bitstream-page/edit-bitstream-page/edit-bitstream-page.component.spec.ts # src/app/+bitstream-page/edit-bitstream-page/edit-bitstream-page.component.ts # src/app/+collection-page/collection-item-mapper/collection-item-mapper.component.ts # src/app/+collection-page/collection-page.component.ts # src/app/+collection-page/edit-collection-page/collection-roles/collection-roles.component.ts # src/app/+collection-page/edit-item-template-page/item-template-page.resolver.spec.ts # src/app/+collection-page/edit-item-template-page/item-template-page.resolver.ts # src/app/+item-page/edit-item-page/item-authorizations/item-authorizations.component.ts # src/app/+item-page/edit-item-page/item-bitstreams/item-bitstreams.component.spec.ts # src/app/+item-page/edit-item-page/item-bitstreams/item-bitstreams.component.ts # src/app/+item-page/edit-item-page/item-collection-mapper/item-collection-mapper.component.ts # src/app/+item-page/edit-item-page/item-private/item-private.component.ts # src/app/+item-page/edit-item-page/item-public/item-public.component.ts # src/app/+item-page/edit-item-page/item-reinstate/item-reinstate.component.ts # src/app/+item-page/edit-item-page/item-relationships/edit-relationship-list/edit-relationship-list.component.ts # src/app/+item-page/edit-item-page/item-relationships/item-relationships.component.ts # src/app/+item-page/edit-item-page/item-withdraw/item-withdraw.component.ts # src/app/+item-page/simple/item-types/shared/item-relationships-utils.ts # src/app/+my-dspace-page/collection-selector/collection-selector.component.spec.ts # src/app/+search-page/search-tracker.component.ts # src/app/app-routing.module.ts # src/app/community-list-page/community-list-service.ts # src/app/community-list-page/community-list.reducer.spec.ts # src/app/core/auth/auth-response-parsing.service.spec.ts # src/app/core/auth/auth.interceptor.spec.ts # src/app/core/browse/browse.service.spec.ts # src/app/core/browse/browse.service.ts # src/app/core/cache/builders/link.service.spec.ts # src/app/core/cache/builders/remote-data-build.service.ts # src/app/core/cache/object-cache.service.spec.ts # src/app/core/cache/response.models.ts # src/app/core/data/base-response-parsing.service.ts # src/app/core/data/bitstream-data.service.ts # src/app/core/data/bitstream-format-data.service.ts # src/app/core/data/bundle-data.service.spec.ts # src/app/core/data/bundle-data.service.ts # src/app/core/data/change-analyzer.ts # src/app/core/data/collection-data.service.spec.ts # src/app/core/data/collection-data.service.ts # src/app/core/data/comcol-data.service.spec.ts # src/app/core/data/comcol-data.service.ts # src/app/core/data/data.service.ts # src/app/core/data/entity-type-data.service.ts # src/app/core/data/entity-type.service.ts # src/app/core/data/feature-authorization/authorization-data.service.ts # src/app/core/data/feature-authorization/feature-data.service.ts # src/app/core/data/item-template-data.service.ts # src/app/core/data/license-data.service.ts # src/app/core/data/metadata-field-data.service.ts # src/app/core/data/processes/process-data.service.ts # src/app/core/data/processes/script-data.service.ts # src/app/core/data/relationship.service.spec.ts # src/app/core/data/relationship.service.ts # src/app/core/data/remote-data.ts # src/app/core/data/request.actions.ts # src/app/core/data/request.effects.ts # src/app/core/data/request.reducer.spec.ts # src/app/core/data/request.reducer.ts # src/app/core/data/request.service.spec.ts # src/app/core/data/request.service.ts # src/app/core/dspace-rest/dspace-rest.service.spec.ts # src/app/core/eperson/eperson-data.service.spec.ts # src/app/core/eperson/eperson-data.service.ts # src/app/core/eperson/group-data.service.spec.ts # src/app/core/eperson/group-data.service.ts # src/app/core/forward-client-ip/forward-client-ip.interceptor.spec.ts # src/app/core/index/index.reducer.ts # src/app/core/json-patch/json-patch-operations.service.spec.ts # src/app/core/json-patch/json-patch-operations.service.ts # src/app/core/locale/locale.interceptor.spec.ts # src/app/core/registry/registry.service.ts # src/app/core/resource-policy/resource-policy.service.ts # src/app/core/shared/operators.ts # src/app/core/shared/search/search.service.ts # src/app/core/submission/submission-object-data.service.ts # src/app/core/submission/submission-rest.service.ts # src/app/core/submission/vocabularies/vocabulary-entries-response-parsing.service.spec.ts # src/app/core/submission/vocabularies/vocabulary.service.spec.ts # src/app/core/submission/vocabularies/vocabulary.service.ts # src/app/core/submission/workflowitem-data.service.ts # src/app/core/tasks/tasks.service.ts # src/app/curation-form/curation-form.component.ts # src/app/process-page/process-page.resolver.ts # src/app/shared/collection-dropdown/collection-dropdown.component.spec.ts # src/app/shared/collection-dropdown/collection-dropdown.component.ts # src/app/shared/comcol-forms/create-comcol-page/create-comcol-page.component.spec.ts # src/app/shared/comcol-forms/create-comcol-page/create-comcol-page.component.ts # src/app/shared/comcol-forms/edit-comcol-page/comcol-metadata/comcol-metadata.component.spec.ts # src/app/shared/comcol-forms/edit-comcol-page/comcol-role/comcol-role.component.spec.ts # src/app/shared/comcol-forms/edit-comcol-page/comcol-role/comcol-role.component.ts # src/app/shared/dso-selector/dso-selector/dso-selector.component.ts # src/app/shared/dso-selector/modal-wrappers/export-metadata-selector/export-metadata-selector.component.ts # src/app/shared/form/builder/ds-dynamic-form-ui/relation-lookup-modal/relationship.effects.ts # src/app/shared/form/builder/form-builder.service.ts # src/app/shared/item/item-versions/item-versions.component.ts # src/app/shared/mocks/remote-data-build.service.mock.ts # src/app/shared/mydspace-actions/workspaceitem/workspaceitem-actions.component.ts # src/app/shared/object-detail/my-dspace-result-detail-element/item-detail-preview/item-detail-preview.component.spec.ts # src/app/shared/object-select/collection-select/collection-select.component.spec.ts # src/app/shared/object-select/item-select/item-select.component.spec.ts # src/app/shared/pagination-drag-and-drop/abstract-paginated-drag-and-drop-list.component.spec.ts # src/app/shared/pagination-drag-and-drop/abstract-paginated-drag-and-drop-list.component.ts # src/app/shared/resource-policies/form/eperson-group-list/eperson-group-list.component.ts # src/app/shared/search/search-query-response.model.ts # src/app/statistics/statistics.module.ts # src/app/submission/import-external/import-external-searchbar/submission-import-external-searchbar.component.spec.ts # src/app/submission/sections/form/section-form.component.ts # src/app/submission/sections/upload/section-upload.component.spec.ts # src/app/submission/sections/upload/section-upload.component.ts # src/app/submission/submission.service.spec.ts # src/app/submission/submission.service.ts
464 lines
20 KiB
TypeScript
464 lines
20 KiB
TypeScript
import { Injectable } from '@angular/core';
|
|
import { Actions, Effect, ofType } from '@ngrx/effects';
|
|
import { Store } from '@ngrx/store';
|
|
import { TranslateService } from '@ngx-translate/core';
|
|
import { isEqual, union } from 'lodash';
|
|
|
|
import { from as observableFrom, Observable, of as observableOf } from 'rxjs';
|
|
import {
|
|
catchError,
|
|
filter,
|
|
map,
|
|
mergeMap,
|
|
switchMap,
|
|
take,
|
|
tap,
|
|
withLatestFrom
|
|
} from 'rxjs/operators';
|
|
import { SubmissionObject } from '../../core/submission/models/submission-object.model';
|
|
import { WorkflowItem } from '../../core/submission/models/workflowitem.model';
|
|
import { WorkspaceitemSectionUploadObject } from '../../core/submission/models/workspaceitem-section-upload.model';
|
|
import { WorkspaceitemSectionsObject } from '../../core/submission/models/workspaceitem-sections.model';
|
|
import { WorkspaceItem } from '../../core/submission/models/workspaceitem.model';
|
|
import { SubmissionJsonPatchOperationsService } from '../../core/submission/submission-json-patch-operations.service';
|
|
import { isEmpty, isNotEmpty, isNotUndefined } from '../../shared/empty.util';
|
|
import { NotificationsService } from '../../shared/notifications/notifications.service';
|
|
import { SectionsType } from '../sections/sections-type';
|
|
import { SectionsService } from '../sections/sections.service';
|
|
import { SubmissionState } from '../submission.reducers';
|
|
import { SubmissionService } from '../submission.service';
|
|
import parseSectionErrors from '../utils/parseSectionErrors';
|
|
import {
|
|
CompleteInitSubmissionFormAction,
|
|
DepositSubmissionAction,
|
|
DepositSubmissionErrorAction,
|
|
DepositSubmissionSuccessAction,
|
|
DiscardSubmissionErrorAction,
|
|
DiscardSubmissionSuccessAction,
|
|
InitSectionAction,
|
|
InitSubmissionFormAction,
|
|
ResetSubmissionFormAction,
|
|
SaveAndDepositSubmissionAction,
|
|
SaveForLaterSubmissionFormAction,
|
|
SaveForLaterSubmissionFormSuccessAction,
|
|
SaveSubmissionFormAction,
|
|
SaveSubmissionFormErrorAction,
|
|
SaveSubmissionFormSuccessAction,
|
|
SaveSubmissionSectionFormAction,
|
|
SaveSubmissionSectionFormErrorAction,
|
|
SaveSubmissionSectionFormSuccessAction,
|
|
SubmissionObjectAction,
|
|
SubmissionObjectActionTypes,
|
|
UpdateSectionDataAction,
|
|
UpdateSectionDataSuccessAction
|
|
} from './submission-objects.actions';
|
|
import {SubmissionObjectEntry, SubmissionSectionError, SubmissionSectionObject} from './submission-objects.reducer';
|
|
import { Item } from '../../core/shared/item.model';
|
|
import { RemoteData } from '../../core/data/remote-data';
|
|
import { getFirstSucceededRemoteDataPayload } from '../../core/shared/operators';
|
|
import { SubmissionObjectDataService } from '../../core/submission/submission-object-data.service';
|
|
import { followLink } from '../../shared/utils/follow-link-config.model';
|
|
import parseSectionErrorPaths, {SectionErrorPath} from '../utils/parseSectionErrorPaths';
|
|
import { FormState } from '../../shared/form/form.reducer';
|
|
|
|
@Injectable()
|
|
export class SubmissionObjectEffects {
|
|
|
|
/**
|
|
* Dispatch a [InitSectionAction] for every submission sections and dispatch a [CompleteInitSubmissionFormAction]
|
|
*/
|
|
@Effect() loadForm$ = this.actions$.pipe(
|
|
ofType(SubmissionObjectActionTypes.INIT_SUBMISSION_FORM),
|
|
map((action: InitSubmissionFormAction) => {
|
|
const definition = action.payload.submissionDefinition;
|
|
const mappedActions = [];
|
|
definition.sections.page.forEach((sectionDefinition: any) => {
|
|
const selfLink = sectionDefinition._links.self.href || sectionDefinition._links.self;
|
|
const sectionId = selfLink.substr(selfLink.lastIndexOf('/') + 1);
|
|
const config = sectionDefinition._links.config ? (sectionDefinition._links.config.href || sectionDefinition._links.config) : '';
|
|
const enabled = (sectionDefinition.mandatory) || (isNotEmpty(action.payload.sections) && action.payload.sections.hasOwnProperty(sectionId));
|
|
let sectionData;
|
|
if (sectionDefinition.sectionType !== SectionsType.SubmissionForm) {
|
|
sectionData = (isNotUndefined(action.payload.sections) && isNotUndefined(action.payload.sections[sectionId])) ? action.payload.sections[sectionId] : Object.create(null);
|
|
} else {
|
|
sectionData = action.payload.item.metadata;
|
|
}
|
|
const sectionErrors = null;
|
|
mappedActions.push(
|
|
new InitSectionAction(
|
|
action.payload.submissionId,
|
|
sectionId,
|
|
sectionDefinition.header,
|
|
config,
|
|
sectionDefinition.mandatory,
|
|
sectionDefinition.sectionType,
|
|
sectionDefinition.visibility,
|
|
enabled,
|
|
sectionData,
|
|
sectionErrors
|
|
)
|
|
);
|
|
});
|
|
return { action: action, definition: definition, mappedActions: mappedActions };
|
|
}),
|
|
mergeMap((result) => {
|
|
return observableFrom(
|
|
result.mappedActions.concat(
|
|
new CompleteInitSubmissionFormAction(result.action.payload.submissionId)
|
|
));
|
|
}));
|
|
|
|
/**
|
|
* Dispatch a [InitSubmissionFormAction]
|
|
*/
|
|
@Effect() resetForm$ = this.actions$.pipe(
|
|
ofType(SubmissionObjectActionTypes.RESET_SUBMISSION_FORM),
|
|
map((action: ResetSubmissionFormAction) =>
|
|
new InitSubmissionFormAction(
|
|
action.payload.collectionId,
|
|
action.payload.submissionId,
|
|
action.payload.selfUrl,
|
|
action.payload.submissionDefinition,
|
|
action.payload.sections,
|
|
action.payload.item,
|
|
null
|
|
)));
|
|
|
|
/**
|
|
* Dispatch a [SaveSubmissionFormSuccessAction] or a [SaveSubmissionFormErrorAction] on error
|
|
*/
|
|
@Effect() saveSubmission$ = this.actions$.pipe(
|
|
ofType(SubmissionObjectActionTypes.SAVE_SUBMISSION_FORM),
|
|
switchMap((action: SaveSubmissionFormAction) => {
|
|
return this.operationsService.jsonPatchByResourceType(
|
|
this.submissionService.getSubmissionObjectLinkName(),
|
|
action.payload.submissionId,
|
|
'sections').pipe(
|
|
map((response: SubmissionObject[]) => new SaveSubmissionFormSuccessAction(action.payload.submissionId, response, action.payload.isManual)),
|
|
catchError(() => observableOf(new SaveSubmissionFormErrorAction(action.payload.submissionId))));
|
|
}));
|
|
|
|
/**
|
|
* Dispatch a [SaveForLaterSubmissionFormSuccessAction] or a [SaveSubmissionFormErrorAction] on error
|
|
*/
|
|
@Effect() saveForLaterSubmission$ = this.actions$.pipe(
|
|
ofType(SubmissionObjectActionTypes.SAVE_FOR_LATER_SUBMISSION_FORM),
|
|
switchMap((action: SaveForLaterSubmissionFormAction) => {
|
|
return this.operationsService.jsonPatchByResourceType(
|
|
this.submissionService.getSubmissionObjectLinkName(),
|
|
action.payload.submissionId,
|
|
'sections').pipe(
|
|
map((response: SubmissionObject[]) => new SaveForLaterSubmissionFormSuccessAction(action.payload.submissionId, response)),
|
|
catchError(() => observableOf(new SaveSubmissionFormErrorAction(action.payload.submissionId))));
|
|
}));
|
|
|
|
/**
|
|
* Call parseSaveResponse and dispatch actions
|
|
*/
|
|
@Effect() saveSubmissionSuccess$ = this.actions$.pipe(
|
|
ofType(SubmissionObjectActionTypes.SAVE_SUBMISSION_FORM_SUCCESS),
|
|
withLatestFrom(this.store$),
|
|
map(([action, currentState]: [SaveSubmissionFormSuccessAction, any]) => {
|
|
return this.parseSaveResponse((currentState.submission as SubmissionState).objects[action.payload.submissionId],
|
|
action.payload.submissionObject, action.payload.submissionId, currentState.forms, action.payload.notify);
|
|
}),
|
|
mergeMap((actions) => observableFrom(actions)));
|
|
|
|
/**
|
|
* Call parseSaveResponse and dispatch actions.
|
|
* Notification system is forced to be disabled.
|
|
*/
|
|
@Effect() saveSubmissionSectionSuccess$ = this.actions$.pipe(
|
|
ofType(SubmissionObjectActionTypes.SAVE_SUBMISSION_SECTION_FORM_SUCCESS),
|
|
withLatestFrom(this.store$),
|
|
map(([action, currentState]: [SaveSubmissionSectionFormSuccessAction, any]) => {
|
|
return this.parseSaveResponse((currentState.submission as SubmissionState).objects[action.payload.submissionId],
|
|
action.payload.submissionObject, action.payload.submissionId, currentState.forms, false);
|
|
}),
|
|
mergeMap((actions) => observableFrom(actions)));
|
|
|
|
/**
|
|
* Dispatch a [SaveSubmissionSectionFormSuccessAction] or a [SaveSubmissionSectionFormErrorAction] on error
|
|
*/
|
|
@Effect() saveSection$ = this.actions$.pipe(
|
|
ofType(SubmissionObjectActionTypes.SAVE_SUBMISSION_SECTION_FORM),
|
|
switchMap((action: SaveSubmissionSectionFormAction) => {
|
|
return this.operationsService.jsonPatchByResourceID(
|
|
this.submissionService.getSubmissionObjectLinkName(),
|
|
action.payload.submissionId,
|
|
'sections',
|
|
action.payload.sectionId).pipe(
|
|
map((response: SubmissionObject[]) => new SaveSubmissionSectionFormSuccessAction(action.payload.submissionId, response)),
|
|
catchError(() => observableOf(new SaveSubmissionSectionFormErrorAction(action.payload.submissionId))));
|
|
}));
|
|
|
|
/**
|
|
* Show a notification on error
|
|
*/
|
|
@Effect({ dispatch: false }) saveError$ = this.actions$.pipe(
|
|
ofType(SubmissionObjectActionTypes.SAVE_SUBMISSION_FORM_ERROR, SubmissionObjectActionTypes.SAVE_SUBMISSION_SECTION_FORM_ERROR),
|
|
withLatestFrom(this.store$),
|
|
tap(() => this.notificationsService.error(null, this.translate.get('submission.sections.general.save_error_notice'))));
|
|
|
|
/**
|
|
* Call parseSaveResponse and dispatch actions or dispatch [SaveSubmissionFormErrorAction] on error
|
|
*/
|
|
@Effect() saveAndDeposit$ = this.actions$.pipe(
|
|
ofType(SubmissionObjectActionTypes.SAVE_AND_DEPOSIT_SUBMISSION),
|
|
withLatestFrom(this.store$),
|
|
switchMap(([action, currentState]: [SaveAndDepositSubmissionAction, any]) => {
|
|
return this.operationsService.jsonPatchByResourceType(
|
|
this.submissionService.getSubmissionObjectLinkName(),
|
|
action.payload.submissionId,
|
|
'sections').pipe(
|
|
map((response: SubmissionObject[]) => {
|
|
if (this.canDeposit(response)) {
|
|
return new DepositSubmissionAction(action.payload.submissionId);
|
|
} else {
|
|
this.notificationsService.warning(null, this.translate.get('submission.sections.general.sections_not_valid'));
|
|
return this.parseSaveResponse((currentState.submission as SubmissionState).objects[action.payload.submissionId],
|
|
response, action.payload.submissionId, currentState.forms);
|
|
}
|
|
}),
|
|
catchError(() => observableOf(new SaveSubmissionFormErrorAction(action.payload.submissionId))));
|
|
}));
|
|
|
|
/**
|
|
* Dispatch a [DepositSubmissionSuccessAction] or a [DepositSubmissionErrorAction] on error
|
|
*/
|
|
@Effect() depositSubmission$ = this.actions$.pipe(
|
|
ofType(SubmissionObjectActionTypes.DEPOSIT_SUBMISSION),
|
|
withLatestFrom(this.store$),
|
|
switchMap(([action, state]: [DepositSubmissionAction, any]) => {
|
|
return this.submissionService.depositSubmission(state.submission.objects[action.payload.submissionId].selfUrl).pipe(
|
|
map(() => new DepositSubmissionSuccessAction(action.payload.submissionId)),
|
|
catchError((error) => {
|
|
console.log('submission error', error);
|
|
return observableOf(new DepositSubmissionErrorAction(action.payload.submissionId));
|
|
}));
|
|
}));
|
|
|
|
/**
|
|
* Show a notification on success and redirect to MyDSpace page
|
|
*/
|
|
@Effect({ dispatch: false }) saveForLaterSubmissionSuccess$ = this.actions$.pipe(
|
|
ofType(SubmissionObjectActionTypes.SAVE_FOR_LATER_SUBMISSION_FORM_SUCCESS),
|
|
tap(() => this.notificationsService.success(null, this.translate.get('submission.sections.general.save_success_notice'))),
|
|
tap(() => this.submissionService.redirectToMyDSpace()));
|
|
|
|
/**
|
|
* Show a notification on success and redirect to MyDSpace page
|
|
*/
|
|
@Effect({ dispatch: false }) depositSubmissionSuccess$ = this.actions$.pipe(
|
|
ofType(SubmissionObjectActionTypes.DEPOSIT_SUBMISSION_SUCCESS),
|
|
tap(() => this.notificationsService.success(null, this.translate.get('submission.sections.general.deposit_success_notice'))),
|
|
tap(() => this.submissionService.redirectToMyDSpace()));
|
|
|
|
/**
|
|
* Show a notification on error
|
|
*/
|
|
@Effect({ dispatch: false }) depositSubmissionError$ = this.actions$.pipe(
|
|
ofType(SubmissionObjectActionTypes.DEPOSIT_SUBMISSION_ERROR),
|
|
tap(() => this.notificationsService.error(null, this.translate.get('submission.sections.general.deposit_error_notice'))));
|
|
|
|
/**
|
|
* Dispatch a [DiscardSubmissionSuccessAction] or a [DiscardSubmissionErrorAction] on error
|
|
*/
|
|
@Effect() discardSubmission$ = this.actions$.pipe(
|
|
ofType(SubmissionObjectActionTypes.DISCARD_SUBMISSION),
|
|
switchMap((action: DepositSubmissionAction) => {
|
|
return this.submissionService.discardSubmission(action.payload.submissionId).pipe(
|
|
map(() => new DiscardSubmissionSuccessAction(action.payload.submissionId)),
|
|
catchError(() => observableOf(new DiscardSubmissionErrorAction(action.payload.submissionId))));
|
|
}));
|
|
|
|
/**
|
|
* Adds all metadata an item to the SubmissionForm sections of the submission
|
|
*/
|
|
@Effect() addAllMetadataToSectionData = this.actions$.pipe(
|
|
ofType(SubmissionObjectActionTypes.UPDATE_SECTION_DATA),
|
|
switchMap((action: UpdateSectionDataAction) => {
|
|
return this.sectionService.getSectionState(action.payload.submissionId, action.payload.sectionId, SectionsType.Upload)
|
|
.pipe(map((section: SubmissionSectionObject) => [action, section]), take(1));
|
|
}),
|
|
filter(([action, section]: [UpdateSectionDataAction, SubmissionSectionObject]) => section.sectionType === SectionsType.SubmissionForm),
|
|
switchMap(([action, section]: [UpdateSectionDataAction, SubmissionSectionObject]) => {
|
|
if (section.sectionType === SectionsType.SubmissionForm) {
|
|
const submissionObject$ = this.submissionObjectService
|
|
.findById(action.payload.submissionId, false, followLink('item')).pipe(
|
|
getFirstSucceededRemoteDataPayload()
|
|
);
|
|
|
|
const item$ = submissionObject$.pipe(
|
|
switchMap((submissionObject: SubmissionObject) => (submissionObject.item as Observable<RemoteData<Item>>).pipe(
|
|
getFirstSucceededRemoteDataPayload(),
|
|
)));
|
|
|
|
return item$.pipe(
|
|
map((item: Item) => item.metadata),
|
|
filter((metadata) => !isEqual(action.payload.data, metadata)),
|
|
map((metadata: any) => new UpdateSectionDataAction(action.payload.submissionId, action.payload.sectionId, metadata, action.payload.errors, action.payload.metadata))
|
|
);
|
|
} else {
|
|
return observableOf(new UpdateSectionDataSuccessAction());
|
|
}
|
|
}),
|
|
);
|
|
|
|
/**
|
|
* Show a notification on success and redirect to MyDSpace page
|
|
*/
|
|
@Effect({ dispatch: false })
|
|
discardSubmissionSuccess$ = this.actions$.pipe(
|
|
ofType(SubmissionObjectActionTypes.DISCARD_SUBMISSION_SUCCESS),
|
|
tap(() => this.notificationsService.success(null, this.translate.get('submission.sections.general.discard_success_notice'))),
|
|
tap(() => this.submissionService.redirectToMyDSpace()));
|
|
|
|
/**
|
|
* Show a notification on error
|
|
*/
|
|
@Effect({ dispatch: false }) discardSubmissionError$ = this.actions$.pipe(
|
|
ofType(SubmissionObjectActionTypes.DISCARD_SUBMISSION_ERROR),
|
|
tap(() => this.notificationsService.error(null, this.translate.get('submission.sections.general.discard_error_notice'))));
|
|
|
|
constructor(private actions$: Actions,
|
|
private notificationsService: NotificationsService,
|
|
private operationsService: SubmissionJsonPatchOperationsService,
|
|
private sectionService: SectionsService,
|
|
private store$: Store<any>,
|
|
private submissionService: SubmissionService,
|
|
private submissionObjectService: SubmissionObjectDataService,
|
|
private translate: TranslateService) {
|
|
}
|
|
|
|
/**
|
|
* Check if the submission object retrieved from REST haven't section errors
|
|
*
|
|
* @param response
|
|
* The submission object retrieved from REST
|
|
*/
|
|
protected canDeposit(response: SubmissionObject[]) {
|
|
let canDeposit = true;
|
|
|
|
if (isNotEmpty(response)) {
|
|
response.forEach((item: WorkspaceItem | WorkflowItem) => {
|
|
const { errors } = item;
|
|
|
|
if (errors && !isEmpty(errors)) {
|
|
canDeposit = false;
|
|
}
|
|
});
|
|
}
|
|
return canDeposit;
|
|
}
|
|
|
|
/**
|
|
* Parse the submission object retrieved from REST and return actions to dispatch
|
|
*
|
|
* @param currentState
|
|
* The current SubmissionObjectEntry
|
|
* @param response
|
|
* The submission object retrieved from REST
|
|
* @param submissionId
|
|
* The submission id
|
|
* @param notify
|
|
* A boolean that indicate if show notification or not
|
|
* @return SubmissionObjectAction[]
|
|
* List of SubmissionObjectAction to dispatch
|
|
*/
|
|
protected parseSaveResponse(
|
|
currentState: SubmissionObjectEntry,
|
|
response: SubmissionObject[],
|
|
submissionId: string,
|
|
forms,
|
|
notify: boolean = true): SubmissionObjectAction[] {
|
|
|
|
const mappedActions = [];
|
|
|
|
if (isNotEmpty(response)) {
|
|
if (notify) {
|
|
this.notificationsService.success(null, this.translate.get('submission.sections.general.save_success_notice'));
|
|
}
|
|
|
|
response.forEach((item: WorkspaceItem | WorkflowItem) => {
|
|
|
|
let errorsList = Object.create({});
|
|
const { errors } = item;
|
|
|
|
if (errors && !isEmpty(errors)) {
|
|
// to avoid dispatching an action for every error, create an array of errors per section
|
|
errorsList = parseSectionErrors(errors);
|
|
if (notify) {
|
|
this.notificationsService.warning(null, this.translate.get('submission.sections.general.sections_not_valid'));
|
|
}
|
|
}
|
|
|
|
const sections: WorkspaceitemSectionsObject = (item.sections && isNotEmpty(item.sections)) ? item.sections : {};
|
|
const sectionsKeys: string[] = union(Object.keys(sections), Object.keys(errorsList));
|
|
|
|
for (const sectionId of sectionsKeys) {
|
|
const sectionErrors = errorsList[sectionId] || [];
|
|
const sectionData = sections[sectionId] || {};
|
|
|
|
// When Upload section is disabled, add to submission only if there are files
|
|
if (currentState.sections[sectionId].sectionType === SectionsType.Upload
|
|
&& isEmpty((sectionData as WorkspaceitemSectionUploadObject).files)
|
|
&& !currentState.sections[sectionId].enabled) {
|
|
continue;
|
|
}
|
|
|
|
if (notify && !currentState.sections[sectionId].enabled) {
|
|
this.submissionService.notifyNewSection(submissionId, sectionId, currentState.sections[sectionId].sectionType);
|
|
}
|
|
|
|
const sectionForm = getForm(forms, currentState, sectionId);
|
|
const filteredErrors = filterErrors(sectionForm, sectionErrors, currentState.sections[sectionId].sectionType, notify);
|
|
mappedActions.push(new UpdateSectionDataAction(submissionId, sectionId, sectionData, filteredErrors));
|
|
}
|
|
});
|
|
}
|
|
return mappedActions;
|
|
}
|
|
}
|
|
|
|
function getForm(forms, currentState, sectionId) {
|
|
if (!forms) {
|
|
return null;
|
|
}
|
|
const formId = currentState.sections[sectionId].formId;
|
|
return forms[formId];
|
|
}
|
|
|
|
/**
|
|
* Filter sectionErrors accordingly to this rules:
|
|
* 1. if notifications are enabled return all errors
|
|
* 2. if sectionType is different from 'submission-form' return all errors
|
|
* 3. otherwise return errors only for those fields marked as touched inside the section form
|
|
* @param sectionForm
|
|
* The form related to the section
|
|
* @param sectionErrors
|
|
* The section errors array
|
|
* @param sectionType
|
|
* The section type
|
|
* @param notify
|
|
* Whether notifications are enabled
|
|
*/
|
|
function filterErrors(sectionForm: FormState, sectionErrors: SubmissionSectionError[], sectionType: string, notify: boolean): SubmissionSectionError[] {
|
|
if (notify || sectionType !== SectionsType.SubmissionForm) {
|
|
return sectionErrors;
|
|
}
|
|
if (!sectionForm || !sectionForm.touched) {
|
|
return [];
|
|
}
|
|
const filteredErrors = [];
|
|
sectionErrors.forEach((error: SubmissionSectionError) => {
|
|
const errorPaths: SectionErrorPath[] = parseSectionErrorPaths(error.path);
|
|
errorPaths.forEach((path: SectionErrorPath) => {
|
|
if (path.fieldId && sectionForm.touched[path.fieldId]) {
|
|
filteredErrors.push(error);
|
|
}
|
|
});
|
|
});
|
|
return filteredErrors;
|
|
}
|