Add float array input
[arvados-workbench2.git] / src / views / run-process-panel / run-process-inputs-form.tsx
1 // Copyright (C) The Arvados Authors. All rights reserved.
2 //
3 // SPDX-License-Identifier: AGPL-3.0
4
5 import * as React from 'react';
6 import { reduxForm, InjectedFormProps } from 'redux-form';
7 import { CommandInputParameter, CWLType, IntCommandInputParameter, BooleanCommandInputParameter, FileCommandInputParameter, DirectoryCommandInputParameter, DirectoryArrayCommandInputParameter, FloatArrayCommandInputParameter } from '~/models/workflow';
8 import { IntInput } from '~/views/run-process-panel/inputs/int-input';
9 import { StringInput } from '~/views/run-process-panel/inputs/string-input';
10 import { StringCommandInputParameter, FloatCommandInputParameter, isPrimitiveOfType, File, Directory, WorkflowInputsData, EnumCommandInputParameter, isArrayOfType, StringArrayCommandInputParameter, FileArrayCommandInputParameter } from '../../models/workflow';
11 import { FloatInput } from '~/views/run-process-panel/inputs/float-input';
12 import { BooleanInput } from './inputs/boolean-input';
13 import { FileInput } from './inputs/file-input';
14 import { connect } from 'react-redux';
15 import { compose } from 'redux';
16 import { Grid, StyleRulesCallback, withStyles, WithStyles } from '@material-ui/core';
17 import { EnumInput } from './inputs/enum-input';
18 import { DirectoryInput } from './inputs/directory-input';
19 import { StringArrayInput } from './inputs/string-array-input';
20 import { createStructuredSelector, createSelector } from 'reselect';
21 import { FileArrayInput } from './inputs/file-array-input';
22 import { DirectoryArrayInput } from './inputs/directory-array-input';
23 import { FloatArrayInput } from './inputs/float-array-input';
24
25 export const RUN_PROCESS_INPUTS_FORM = 'runProcessInputsForm';
26
27 export interface RunProcessInputFormProps {
28     inputs: CommandInputParameter[];
29 }
30
31 const inputsSelector = (props: RunProcessInputFormProps) =>
32     props.inputs;
33
34 const initialValuesSelector = createSelector(
35     inputsSelector,
36     inputs => inputs.reduce(
37         (values, input) => ({ ...values, [input.id]: input.default }),
38         {}));
39
40 const propsSelector = createStructuredSelector({
41     initialValues: initialValuesSelector,
42 });
43
44 const mapStateToProps = (_: any, props: RunProcessInputFormProps) =>
45     propsSelector(props);
46
47 export const RunProcessInputsForm = compose(
48     connect(mapStateToProps),
49     reduxForm<WorkflowInputsData, RunProcessInputFormProps>({
50         form: RUN_PROCESS_INPUTS_FORM
51     }))(
52         (props: InjectedFormProps & RunProcessInputFormProps) =>
53             <form>
54                 <Grid container spacing={32}>
55                     {props.inputs.map(input =>
56                         <InputItem input={input} key={input.id} />)}
57                 </Grid>
58             </form>);
59
60 type CssRules = 'inputItem';
61
62 const styles: StyleRulesCallback<CssRules> = theme => ({
63     inputItem: {
64         marginBottom: theme.spacing.unit * 2,
65     }
66 });
67
68 const InputItem = withStyles(styles)(
69     (props: WithStyles<CssRules> & { input: CommandInputParameter }) =>
70         <Grid item xs={12} md={6} className={props.classes.inputItem}>
71             {getInputComponent(props.input)}
72         </Grid>);
73
74 const getInputComponent = (input: CommandInputParameter) => {
75     switch (true) {
76         case isPrimitiveOfType(input, CWLType.BOOLEAN):
77             return <BooleanInput input={input as BooleanCommandInputParameter} />;
78
79         case isPrimitiveOfType(input, CWLType.INT):
80         case isPrimitiveOfType(input, CWLType.LONG):
81             return <IntInput input={input as IntCommandInputParameter} />;
82
83         case isPrimitiveOfType(input, CWLType.FLOAT):
84         case isPrimitiveOfType(input, CWLType.DOUBLE):
85             return <FloatInput input={input as FloatCommandInputParameter} />;
86
87         case isPrimitiveOfType(input, CWLType.STRING):
88             return <StringInput input={input as StringCommandInputParameter} />;
89
90         case isPrimitiveOfType(input, CWLType.FILE):
91             return <FileInput input={input as FileCommandInputParameter} />;
92
93         case isPrimitiveOfType(input, CWLType.DIRECTORY):
94             return <DirectoryInput input={input as DirectoryCommandInputParameter} />;
95
96         case typeof input.type === 'object' &&
97             !(input.type instanceof Array) &&
98             input.type.type === 'enum':
99             return <EnumInput input={input as EnumCommandInputParameter} />;
100
101         case isArrayOfType(input, CWLType.STRING):
102             return <StringArrayInput input={input as StringArrayCommandInputParameter} />;
103         
104         case isArrayOfType(input, CWLType.FLOAT):
105             return <FloatArrayInput input={input as FloatArrayCommandInputParameter} />;
106
107         case isArrayOfType(input, CWLType.FILE):
108             return <FileArrayInput input={input as FileArrayCommandInputParameter} />;
109         
110         case isArrayOfType(input, CWLType.DIRECTORY):
111             return <DirectoryArrayInput input={input as DirectoryArrayCommandInputParameter} />;
112
113         default:
114             return null;
115     }
116 };