From d550931820ed228a5d95f4ffe2d4e5f4f90d951f Mon Sep 17 00:00:00 2001 From: Lorenzo Cavazzi <43481553+lorenzo-cavazzi@users.noreply.github.com> Date: Wed, 15 Dec 2021 09:08:16 +0100 Subject: [PATCH] feat(client): support template variables metadata (#1579) * Style and show input elements based on the variable type. * Automatically set default values. fix #1320 --- client/src/project/new/ProjectNew.present.js | 158 +++++++++++++++++-- client/src/project/new/ProjectNew.state.js | 29 +++- 2 files changed, 170 insertions(+), 17 deletions(-) diff --git a/client/src/project/new/ProjectNew.present.js b/client/src/project/new/ProjectNew.present.js index bdd7a845fd..537fbda38f 100644 --- a/client/src/project/new/ProjectNew.present.js +++ b/client/src/project/new/ProjectNew.present.js @@ -34,7 +34,7 @@ import { } from "reactstrap"; import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; import { - faExclamationTriangle, faInfoCircle, faLink, faQuestionCircle, faSyncAlt + faExclamationTriangle, faInfoCircle, faLink, faQuestionCircle, faSyncAlt, faUndo } from "@fortawesome/free-solid-svg-icons"; import { @@ -974,6 +974,32 @@ function TemplateGalleryRow(props) { ); } +/** + * Create a "restore default" button. + * + * @param {function} restore - function to invoke + * @param {string} tip - message to display in the tooltip + * @param {boolean} disabled - whether it's disabled or not + */ +function RestoreButton(props) { + const { restore, name, disabled } = props; + + const id = `restore_${name}`; + const tip = disabled ? + "Default value already selected" : + "Restore default value"; + + return ( +
+ + {tip} +
+ ); +} + class Variables extends Component { render() { const { input, handlers } = this.props; @@ -987,19 +1013,111 @@ class Variables extends Component { const template = templates.all.filter(t => t.id === input.template)[0]; if (!template || !template.variables || !Object.keys(template.variables).length) return null; - const variables = Object.keys(template.variables).map(variable => ( - - - handlers.setVariable(variable, e.target.value)} /> - {capitalize(template.variables[variable])} - - )); + const variables = Object.keys(template.variables).map(variable => { + const data = template.variables[variable]; + + // fallback to avoid breaking old variable structure + if (typeof data !== "object") { + return ( + + + handlers.setVariable(variable, e.target.value)} /> + {capitalize(template.variables[variable])} + + ); + } + + // expected `data` properties: default_value, description, enum, type. + // changing enum to enumValues to avoid using js reserved word + return ( + + ); + }); return variables; } } +function Variable(props) { + const { default_value, description, enumValues, handlers, input, name, type } = props; + const id = `parameter-${name}`; + + const descriptionOutput = description ? + ({capitalize(description)}) : + null; + + const defaultOutput = default_value != null ? + `Default: ${default_value}` : + null; + + const restoreButton = default_value != null ? + ( + handlers.setVariable(name, default_value)} + /> + ) : + null; + + let inputElement = null; + if (type === "boolean") { + inputElement = ( + + handlers.setVariable(name, e.target.checked)} + className="form-check-input rounded-pill" /> + + {restoreButton} + + ); + // inputElement = null; + } + else if (type === "enum") { + const enumObjects = enumValues.map(enumObject => { + const enumId = `enum-${id}-${enumObject.toString()}`; + return ( + + ); + }); + inputElement = ( + + {restoreButton} + handlers.setVariable(name, e.target.value)}> + {enumObjects} + + {descriptionOutput} + + ); + } + else { + const inputType = type === "number" ? + "number" : + "text"; + inputElement = ( + + {restoreButton} + handlers.setVariable(name, e.target.value)} + placeholder={defaultOutput} /> + {descriptionOutput} + + ); + } + + return inputElement; +} + class Create extends Component { constructor(props) { super(props); @@ -1109,6 +1227,25 @@ class Create extends Component { "based on " + (templates.all.find(t => t.id === input.template).name) : ""; + const errorFields = meta.validation.errors ? + Object.keys(meta.validation.errors) + .filter(field => !input[`${field}Pristine`]) // don't consider pristine fields + .map(field => capitalize(field)) : + []; + const plural = errorFields.length > 1 ? + "s" : + ""; + const errorMessage = errorFields.length ? + ( + + + To create a new project, please first fix problems with the following field{plural}:{" "} + {errorFields.join(", ")} + + + ) : + null; + return ( {alert} @@ -1124,6 +1261,7 @@ class Create extends Component { meta={meta} createUrl={this.props.handlers.createEncodedUrl} /> + {errorMessage} ); } @@ -1204,7 +1342,7 @@ function ShareLinkModal(props) { if (include.variables) { let variablesObject = {}; for (let variable of Object.keys(input.variables)) { - if (input.variables[variable]) + if (input.variables[variable] != null) variablesObject[variable] = input.variables[variable]; } dataObject.variables = variablesObject; diff --git a/client/src/project/new/ProjectNew.state.js b/client/src/project/new/ProjectNew.state.js index b5d171c23a..9c4432e1ca 100644 --- a/client/src/project/new/ProjectNew.state.js +++ b/client/src/project/new/ProjectNew.state.js @@ -91,17 +91,32 @@ class NewProjectCoordinator { Object.keys(template.variables) : []; - // preserve already set values + // preserve already set values or set default values when available const oldValues = currentInput.template ? currentInput.variables : {}; const oldVariables = Object.keys(oldValues); const values = variables.reduce((values, variable) => { - const text = oldVariables.includes(variable) ? - oldValues[variable] : - ""; - return { ...values, [variable]: text }; + let value = ""; + + const variableData = template.variables[variable]; + if (typeof variableData === "object") { + // set first value for enum, and "false" for boolean + if (variableData["type"] === "enum" && variableData["enum"] && variableData["enum"].length) + value = variableData["enum"][0]; + + // set default, if any + if (typeof variableData === "object" && variableData["default_value"] != null) + value = variableData["default_value"]; + } + + // set older value, if any + if (oldVariables.includes(variable)) + value = oldValues[variable]; + + return { ...values, [variable]: value }; }, {}); + return values; } @@ -535,10 +550,10 @@ class NewProjectCoordinator { newProjectData.ref = userTemplates.ref; } - // add variables + // add variables after converting to string (renku core accept string only) let parameters = []; for (let variable of Object.keys(input.variables)) - parameters.push({ key: variable, value: input.variables[variable] }); + parameters.push({ key: variable, value: input.variables[variable].toString() }); newProjectData.parameters = parameters; // reset all previous creation progresses and invoke the project creation API