1 // Copyright (C) The Arvados Authors. All rights reserved.
3 // SPDX-License-Identifier: AGPL-3.0
5 import { Dispatch } from "redux";
6 import { reset, startSubmit, stopSubmit, initialize, FormErrors, formValueSelector, change, arrayPush } from 'redux-form';
7 import { RootState } from '~/store/store';
8 import { dialogActions } from "~/store/dialog/dialog-actions";
9 import { getCommonResourceServiceError, CommonResourceServiceError } from '~/services/common-service/common-resource-service';
10 import { ProjectResource } from '~/models/project';
11 import { ServiceRepository } from '~/services/services';
12 import { matchProjectRoute, matchRunProcessRoute } from '~/routes/routes';
13 import { ResourcePropertiesFormData } from '~/views-components/resource-properties-form/resource-properties-form';
15 export interface ProjectCreateFormDialogData {
22 export const PROJECT_CREATE_FORM_NAME = 'projectCreateFormName';
23 export const PROJECT_CREATE_PROPERTIES_FORM_NAME = 'projectCreatePropertiesFormName';
25 export const isProjectOrRunProcessRoute = ({ router }: RootState) => {
26 const pathname = router.location ? router.location.pathname : '';
27 const matchProject = matchProjectRoute(pathname);
28 const matchRunProcess = matchRunProcessRoute(pathname);
29 return Boolean(matchProject || matchRunProcess);
32 export const isItemNotInProject = (properties: any) => {
33 if (properties.breadcrumbs) {
34 return Boolean(properties.breadcrumbs[0].label !== 'Projects');
40 export const openProjectCreateDialog = (ownerUuid: string) =>
41 (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
42 const router = getState();
43 const properties = getState().properties;
44 if (isItemNotInProject(properties) || !isProjectOrRunProcessRoute(router)) {
45 const userUuid = getState().auth.user!.uuid;
46 dispatch(initialize(PROJECT_CREATE_FORM_NAME, { userUuid }));
48 dispatch(initialize(PROJECT_CREATE_FORM_NAME, { ownerUuid }));
50 dispatch(dialogActions.OPEN_DIALOG({ id: PROJECT_CREATE_FORM_NAME, data: {} }));
53 export const createProject = (project: Partial<ProjectResource>) =>
54 async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
55 dispatch(startSubmit(PROJECT_CREATE_FORM_NAME));
57 const newProject = await services.projectService.create(project);
58 dispatch(dialogActions.CLOSE_DIALOG({ id: PROJECT_CREATE_FORM_NAME }));
59 dispatch(reset(PROJECT_CREATE_FORM_NAME));
62 const error = getCommonResourceServiceError(e);
63 if (error === CommonResourceServiceError.UNIQUE_VIOLATION) {
64 dispatch(stopSubmit(PROJECT_CREATE_FORM_NAME, { name: 'Project with the same name already exists.' } as FormErrors));
70 export const addPropertyToCreateProjectForm = (data: ResourcePropertiesFormData) =>
71 (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
72 const selector = formValueSelector(PROJECT_CREATE_FORM_NAME);
73 const properties = selector(getState(), 'properties') || {};
74 properties[data.key] = data.value;
75 dispatch(change(PROJECT_CREATE_FORM_NAME, 'properties', {...properties } ));
78 export const removePropertyFromCreateProjectForm = (key: string) =>
79 (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
80 const selector = formValueSelector(PROJECT_CREATE_FORM_NAME);
81 const properties = selector(getState(), 'properties');
82 delete properties[key];
83 dispatch(change(PROJECT_CREATE_FORM_NAME, 'properties', { ...properties } ));