-
Notifications
You must be signed in to change notification settings - Fork 2
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
bug: Calculate component error handling fix #3805
Changes from all commits
9421fa0
c8f808a
100dcc9
78f801c
32b74a5
0d0d020
5fe58b5
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -3,11 +3,8 @@ import { styled } from "@mui/material/styles"; | |
import Switch from "@mui/material/Switch"; | ||
import Typography from "@mui/material/Typography"; | ||
import { ComponentType as TYPES } from "@opensystemslab/planx-core/types"; | ||
import { | ||
EditorProps, | ||
ICONS, | ||
} from "@planx/components/ui"; | ||
import { useFormik } from "formik"; | ||
import { EditorProps, ICONS } from "@planx/components/ui"; | ||
import { FormikErrors, useFormik } from "formik"; | ||
import React from "react"; | ||
import InputGroup from "ui/editor/InputGroup"; | ||
import { ModalFooter } from "ui/editor/ModalFooter"; | ||
|
@@ -36,12 +33,51 @@ export default function Component(props: Props) { | |
props.handleSubmit({ type: TYPES.Calculate, data: newValues }); | ||
} | ||
}, | ||
validate: () => { | ||
// can parse formula | ||
getVariables(formik.values.formula); | ||
validate: (values) => { | ||
const errors: FormikErrors<Calculate> = {}; | ||
try { | ||
// can parse formula | ||
getVariables(formik.values.formula); | ||
|
||
// Validate formula | ||
const result = evaluate( | ||
values.formula, | ||
values.samples, | ||
values.defaults, | ||
); | ||
|
||
if (Number.isNaN(Number(result))) { | ||
errors.formula = "Enter a formula which outputs a number"; | ||
} | ||
} catch (error: any) { | ||
errors.formula = error.message; | ||
} | ||
return errors; | ||
}, | ||
|
||
validateOnChange: false, | ||
}); | ||
|
||
const sampleResult = React.useMemo(() => { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Tried to ensure this was an additive fix, I didn't want to take away too much which makes the component work because I couldn't fully grasp how it all functioned within the timeboxed time There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. So what I think is happening here is that the The pattern we want to follow, and other components employ, is to keep I think this means removing this custom validation function, and just using formik 👍 There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. It also feeds the "Try" field which gives you a preview of the formula output: <p>
<strong>{formik.values.output || "<output>"}</strong> would be set
to <strong>{sampleResult}</strong>.
</p> I could maybe move it to state and use that in the validation step as well, updating with a useEffect or something but, this felt like a change on top of the bug fix. Happy to do a bit more on this if you think it is worth it? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Aha, thanks. This is what I've missed 👍 So we do need a way to get a runtime result out which is I think I'd aim to see if there's a way we can avoid duplicating the validation logic - maybe a function with a signature like this could work in both instances? type Validate = (values: Calculate) => [result: number | undefined, errors: FormikErrors<Calculate>] This would look something like this - validate: (values) => {
const [_, errors]: validate(values);
return errors;
},
...
const sampleResult = React.useMemo(() => {
const [result, _]: validate(formik.values);
if (!result) return UNKNOWN
}, [formik.values, validate]); There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @DafyddLlyr got to the end of my timebox and didn't get it working as required. It's definitely possible though and I'll make a note for a potential follow-up PR to tidy this up, if that sounds good to you? I was mostly caught resolving types and harmonising return messaging. |
||
try { | ||
const result = evaluate( | ||
formik.values.formula, | ||
formik.values.samples, | ||
formik.values.defaults, | ||
); | ||
// Type guard as mathjs evaluates `m` to a "Unit" object for "meter" | ||
if (!Number.isNaN(Number(result))) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Is There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. From what I could work out, the result would return a |
||
return result; | ||
} else if (result === undefined) { | ||
return "a number returned from the formula above"; | ||
} else { | ||
return `'${result}' which is of the type: ${typeof result}`; | ||
} | ||
} catch (e) { | ||
return UNKNOWN; | ||
} | ||
}, [formik.values.formula, formik.values.defaults, formik.values.samples]); | ||
|
||
/** | ||
* When the formula is updated, remove any defaults which are no longer used | ||
*/ | ||
|
@@ -65,24 +101,6 @@ export default function Component(props: Props) { | |
} | ||
}, [formik.values.formula]); | ||
|
||
const sampleResult = React.useMemo(() => { | ||
try { | ||
const result = evaluate( | ||
formik.values.formula, | ||
formik.values.samples, | ||
formik.values.defaults, | ||
); | ||
// Type guard as mathjs evaluates `m` to a "Unit" object for "meter" | ||
if (typeof result === "number") { | ||
return result; | ||
} else { | ||
return UNKNOWN; | ||
} | ||
} catch (e) { | ||
return UNKNOWN; | ||
} | ||
}, [formik.values.formula, formik.values.defaults, formik.values.samples]); | ||
|
||
return ( | ||
<form onSubmit={formik.handleSubmit} id="modal"> | ||
<ModalSection> | ||
|
@@ -140,11 +158,7 @@ export default function Component(props: Props) { | |
name="formula" | ||
value={formik.values.formula} | ||
onChange={formik.handleChange} | ||
errorMessage={ | ||
sampleResult === UNKNOWN | ||
? "Invalid formula or missing default values." | ||
: "" | ||
} | ||
errorMessage={formik.errors.formula} | ||
/> | ||
</InputRow> | ||
</ModalSectionContent> | ||
|
@@ -190,7 +204,7 @@ export default function Component(props: Props) { | |
)} | ||
<p> | ||
<strong>{formik.values.output || "<output>"}</strong> would be set | ||
to <strong>{sampleResult}</strong>. | ||
to <strong>{sampleResult}</strong> | ||
</p> | ||
</ModalSectionContent> | ||
</ModalSection> | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
As mentioned in the description, I have used this to catch issues with the
NumberInput
component which existed but wasn't picked up because we were never blocking the update of the comp