Skip to content

Commit fe8838e

Browse files
committedFeb 24, 2025·
stubs for dataset creation
1 parent 3c673b6 commit fe8838e

File tree

7 files changed

+251
-7
lines changed

7 files changed

+251
-7
lines changed
 

‎frontend/src/actions/project.js

+37
Original file line numberDiff line numberDiff line change
@@ -41,3 +41,40 @@ export function fetchProject(id) {
4141
});
4242
};
4343
}
44+
45+
export const CREATE_PROJECT = "CREATE_PROJECT";
46+
47+
export function projectCreated(formData) {
48+
return (dispatch) => {
49+
// If licenseFormData is not present, directly save the dataset
50+
return V2.ProjectsService.saveProjectApiV2ProjectsPost(
51+
formData
52+
)
53+
.then((project) => {
54+
dispatch({
55+
type: CREATE_PROJECT,
56+
project: project,
57+
receivedAt: Date.now(),
58+
});
59+
})
60+
.catch((reason) => {
61+
dispatch(
62+
handleErrors(
63+
reason,
64+
projectCreated(formData)
65+
)
66+
);
67+
});
68+
};
69+
}
70+
71+
export const RESET_CREATE_PROJECT = "RESET_CREATE_PROJECT";
72+
73+
export function resetProjectCreated() {
74+
return (dispatch) => {
75+
dispatch({
76+
type: RESET_CREATE_PROJECT,
77+
receivedAt: Date.now(),
78+
});
79+
};
80+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,109 @@
1+
import React, {useEffect, useState} from "react";
2+
3+
import {Box, Step, StepContent, StepLabel, Stepper, Typography,} from "@mui/material";
4+
import {useDispatch, useSelector} from "react-redux";
5+
import {RootState} from "../../types/data";
6+
import {projectCreated} from "../../actions/project";
7+
import {useNavigate} from "react-router-dom";
8+
import Layout from "../Layout";
9+
import {ErrorModal} from "../errors/ErrorModal";
10+
import {CreateProjectModal} from "./CreateProjectModal"
11+
12+
export const CreateProject = (): JSX.Element => {
13+
const dispatch = useDispatch();
14+
const createProject = (
15+
formData: FormData,
16+
) => dispatch(projectCreated(formData));
17+
18+
const newProject = useSelector(
19+
(state: RootState) => state.project.newProject
20+
);
21+
22+
const [errorOpen, setErrorOpen] = useState(false);
23+
24+
const [projectRequestForm, setProjectRequestForm] = useState({});
25+
const [allowSubmit, setAllowSubmit] = React.useState<boolean>(false);
26+
27+
const history = useNavigate();
28+
29+
const checkIfFieldsAreRequired = () => {
30+
const required = false;
31+
32+
return required;
33+
};
34+
35+
// step 1 - pick datasets
36+
const onProjectSave = (formData: any) => {
37+
setProjectRequestForm(formData);
38+
39+
// If no metadata fields are marked as required, allow user to skip directly to submit
40+
if (checkIfFieldsAreRequired()) {
41+
setAllowSubmit(false);
42+
} else {
43+
setAllowSubmit(true);
44+
}
45+
46+
handleNext();
47+
};
48+
// step 2 - add users
49+
50+
// step
51+
const [activeStep, setActiveStep] = useState(0);
52+
const handleNext = () => {
53+
setActiveStep((prevActiveStep) => prevActiveStep + 1);
54+
};
55+
const handleBack = () => {
56+
setActiveStep((prevActiveStep) => prevActiveStep - 1);
57+
};
58+
59+
const handleFinish = () => {
60+
createProject(projectRequestForm);
61+
};
62+
63+
useEffect(() => {
64+
if (newProject && newProject.id) {
65+
history(`/projects/${newProject.id}`);
66+
}
67+
}, [newProject]);
68+
69+
return (
70+
<Layout>
71+
<Box className="outer-container">
72+
{/*Error Message dialogue*/}
73+
<ErrorModal errorOpen={errorOpen} setErrorOpen={setErrorOpen}/>
74+
<Box className="inner-container">
75+
<Box>
76+
<Stepper activeStep={activeStep} orientation="vertical">
77+
{/*step 1 Dataset*/}
78+
<Step key="create-dataset">
79+
<StepLabel>Basic Information</StepLabel>
80+
<StepContent>
81+
<Typography>
82+
A project is a collection of datasets and a community of contributors.
83+
</Typography>
84+
<Box>
85+
<CreateProjectModal onSave={onProjectSave}/>
86+
</Box>
87+
</StepContent>
88+
</Step>
89+
90+
<Step key="select-datasets">
91+
<StepLabel>Select Datasets</StepLabel>
92+
<StepContent>
93+
<Box />
94+
</StepContent>
95+
</Step>
96+
97+
<Step key="invite-users">
98+
<StepLabel>Invite Users</StepLabel>
99+
<StepContent>
100+
<Box />
101+
</StepContent>
102+
</Step>
103+
</Stepper>
104+
</Box>
105+
</Box>
106+
</Box>
107+
</Layout>
108+
);
109+
};
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
1+
import React from "react";
2+
3+
import {Box, Button} from "@mui/material";
4+
5+
import Form from "@rjsf/mui";
6+
import projectSchema from "../../schema/projectSchema.json";
7+
import {FormProps} from "@rjsf/core";
8+
import {ClowderRjsfTextWidget} from "../styledComponents/ClowderRjsfTextWidget";
9+
import {ClowderRjsfSelectWidget} from "../styledComponents/ClowderRjsfSelectWidget";
10+
import {ClowderRjsfTextAreaWidget} from "../styledComponents/ClowderRjsfTextAreaWidget";
11+
import validator from "@rjsf/validator-ajv8";
12+
13+
type CreateProjectModalProps = {
14+
onSave: any;
15+
};
16+
17+
const widgets = {
18+
TextWidget: ClowderRjsfTextWidget,
19+
TextAreaWidget: ClowderRjsfTextAreaWidget,
20+
SelectWidget: ClowderRjsfSelectWidget,
21+
};
22+
23+
export const CreateProjectModal: React.FC<CreateProjectModalProps> = (
24+
props: CreateProjectModalProps
25+
) => {
26+
const {onSave} = props;
27+
28+
return (
29+
<Form
30+
widgets={widgets}
31+
schema={projectSchema["schema"] as FormProps<any>["schema"]}
32+
uiSchema={projectSchema["uiSchema"] as FormProps<any>["uiSchema"]}
33+
validator={validator}
34+
onSubmit={({formData}) => {
35+
onSave(formData);
36+
}}
37+
>
38+
<Box className="inputGroup">
39+
<Button variant="contained" type="submit">
40+
Next
41+
</Button>
42+
</Box>
43+
</Form>
44+
);
45+
};

‎frontend/src/routes.tsx

+9
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,7 @@ import {MetadataDefinitionEntry} from "./components/metadata/MetadataDefinitionE
3535
import {Feeds} from "./components/listeners/Feeds";
3636
import {AllListeners} from "./components/listeners/AllListeners";
3737
import {FeedEntry} from "./components/listeners/FeedEntry";
38+
import {CreateProject} from "./components/projects/CreateProject";
3839

3940
// https://dev.to/iamandrewluca/private-route-in-react-router-v6-lg5
4041
const PrivateRoute = (props): JSX.Element => {
@@ -176,6 +177,14 @@ export const AppRoutes = (): JSX.Element => {
176177
</PrivateRoute>
177178
}
178179
/>
180+
<Route
181+
path="/create-project/"
182+
element={
183+
<PrivateRoute>
184+
<CreateProject/>
185+
</PrivateRoute>
186+
}
187+
/>
179188
<Route
180189
path="/create-dataset/"
181190
element={
+33
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
{
2+
"schema": {
3+
"type": "object",
4+
"required": [
5+
"name"
6+
],
7+
"properties": {
8+
"name": {
9+
"type": "string",
10+
"title": "Name"
11+
},
12+
"description": {
13+
"type": "string",
14+
"title": "Description"
15+
}
16+
}
17+
},
18+
"uiSchema": {
19+
"name": {
20+
"ui:autofocus": true,
21+
"ui:emptyValue": "",
22+
"ui:placeholder": "ui:emptyValue causes this field to always be valid despite being required",
23+
"ui:description": "Name of the dataset."
24+
},
25+
"description": {
26+
"ui:autofocus": true,
27+
"ui:emptyValue": "",
28+
"ui:placeholder": "ui:emptyValue causes this field to always be valid despite being required",
29+
"ui:description": "Description of a dataset.",
30+
"ui:widget": "TextAreaWidget"
31+
}
32+
}
33+
}

‎frontend/src/types/action.ts

+10-7
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import { ExtractedMetadata, FilePreview, Folder, MetadataJsonld } from "./data";
1+
import {ExtractedMetadata, FilePreview, Folder, MetadataJsonld} from "./data";
22
import {
33
AuthorizationBase,
44
DatasetFreezeOut,
@@ -7,8 +7,8 @@ import {
77
DatasetRoles,
88
EventListenerJobOut,
99
EventListenerJobUpdateOut,
10-
FeedOut,
1110
EventListenerOut,
11+
FeedOut,
1212
FileOut,
1313
FileOut as FileSummary,
1414
FileVersion,
@@ -18,17 +18,14 @@ import {
1818
MetadataDefinitionOut as MetadataDefinition,
1919
MetadataOut as Metadata,
2020
Paged,
21+
ProjectOut as Project,
2122
RoleType,
2223
UserAPIKeyOut,
2324
UserOut,
2425
VisualizationConfigOut,
2526
VisualizationDataOut,
2627
} from "../openapi/v2";
27-
import {
28-
LIST_USERS,
29-
PREFIX_SEARCH_USERS,
30-
RECEIVE_USER_PROFILE,
31-
} from "../actions/user";
28+
import {LIST_USERS, PREFIX_SEARCH_USERS, RECEIVE_USER_PROFILE,} from "../actions/user";
3229

3330
interface RECEIVE_FILES_IN_DATASET {
3431
type: "RECEIVE_FILES_IN_DATASET";
@@ -227,6 +224,11 @@ interface RECEIVE_USER_PROFILE {
227224
profile: UserOut;
228225
}
229226

227+
interface CREATE_PROJECT {
228+
type: "CREATE_PROJECT";
229+
project: Project;
230+
}
231+
230232
interface CREATE_DATASET {
231233
type: "CREATE_DATASET";
232234
dataset: Dataset;
@@ -768,6 +770,7 @@ export type DataAction =
768770
| DELETE_API_KEY
769771
| GENERATE_API_KEY
770772
| RESET_API_KEY
773+
| CREATE_PROJECT
771774
| CREATE_DATASET
772775
| RESET_CREATE_DATASET
773776
| CREATE_FILE

‎frontend/src/types/data.ts

+8
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@ import {
1313
MetadataDefinitionOut,
1414
MetadataOut as Metadata,
1515
Paged,
16+
ProjectOut,
1617
RoleType,
1718
UserAPIKeyOut,
1819
UserOut,
@@ -120,6 +121,12 @@ export interface Thumbnail {
120121
thumbnail: string;
121122
}
122123

124+
export interface ProjectState {
125+
newProject: ProjectOut;
126+
datasets: DatasetOut[];
127+
members: DatasetRoles;
128+
}
129+
123130
export interface DatasetState {
124131
foldersAndFiles: Paged;
125132
files: Paged;
@@ -285,6 +292,7 @@ export interface RootState {
285292
file: FileState;
286293
publicFile: PublicFileState;
287294
dataset: DatasetState;
295+
project: ProjectState;
288296
publicDataset: PublicDatasetState;
289297
listener: ListenerState;
290298
group: GroupState;

0 commit comments

Comments
 (0)
Please sign in to comment.