From bd4fc697cc1c50ce53c3f9b6ef6bdf27f226ea82 Mon Sep 17 00:00:00 2001 From: Kadee80 Date: Tue, 12 Mar 2019 15:00:02 -0400 Subject: [PATCH 1/5] feat(app): Enable pipette config modal and form Render hidden config fields behind new feat flag closes #3112 --- .../components/ConfigurePipette/ConfigForm.js | 31 +++++++++++++++++-- app/src/components/ConfigurePipette/index.js | 6 +++- .../AttachedPipettesCard.js | 8 ----- .../InstrumentSettings/InstrumentInfo.js | 3 +- app/src/config/index.js | 2 +- 5 files changed, 35 insertions(+), 15 deletions(-) diff --git a/app/src/components/ConfigurePipette/ConfigForm.js b/app/src/components/ConfigurePipette/ConfigForm.js index ca192c7660c..2b49095c0ef 100644 --- a/app/src/components/ConfigurePipette/ConfigForm.js +++ b/app/src/components/ConfigurePipette/ConfigForm.js @@ -5,9 +5,11 @@ import {Formik, Form} from 'formik' import startCase from 'lodash/startCase' import mapValues from 'lodash/mapValues' +import forOwn from 'lodash/forOwn' +import keys from 'lodash/keys' import pick from 'lodash/pick' +import omit from 'lodash/omit' import set from 'lodash/set' -import forOwn from 'lodash/forOwn' import isEmpty from 'lodash/isEmpty' import FormButtonBar from './FormButtonBar' @@ -33,6 +35,7 @@ type Props = { pipette: Pipette, pipetteConfig: PipetteConfigResponse, updateConfig: (id: string, PipetteConfigRequest) => mixed, + showHiddenFields: boolean, } const PLUNGER_KEYS = ['top', 'bottom', 'blowout', 'dropTip'] @@ -64,6 +67,14 @@ export default class ConfigForm extends React.Component { ]) } + getHiddenFields = () => { + return omit(this.props.pipetteConfig.fields, [ + ...PLUNGER_KEYS, + ...POWER_KEYS, + ...TIP_KEYS, + ]) + } + handleSubmit = (values: FormValues) => { const params = mapValues(values, v => { return v === '' ? null : {value: Number(v)} @@ -84,7 +95,9 @@ export default class ConfigForm extends React.Component { validate = (values: FormValues) => { const errors = {} - const fields = this.getVisibleFields() + const fields = this.props.showHiddenFields + ? this.props.pipetteConfig.fields + : this.getVisibleFields() const plungerFields = this.getFieldsByKey(PLUNGER_KEYS, fields) // validate all visible fields with min and max @@ -118,12 +131,18 @@ export default class ConfigForm extends React.Component { render () { const {parentUrl} = this.props const fields = this.getVisibleFields() - const initialValues = mapValues(fields, f => { + const hiddenFields = this.getHiddenFields() + const HIDDEN_KEYS = keys(hiddenFields) + const visibleFields = this.props.showHiddenFields + ? this.props.pipetteConfig.fields + : fields + const initialValues = mapValues(visibleFields, f => { return f.value !== f.default ? f.value.toString() : '' }) const plungerFields = this.getFieldsByKey(PLUNGER_KEYS, fields) const powerFields = this.getFieldsByKey(POWER_KEYS, fields) const tipFields = this.getFieldsByKey(TIP_KEYS, fields) + const devFields = this.getFieldsByKey(HIDDEN_KEYS, hiddenFields) return ( { groupLabel="Tip Pickup / Drop " formFields={tipFields} /> + {this.props.showHiddenFields && ( + + )} )} @@ -93,10 +95,12 @@ function makeMapStateToProps (): (state: State, ownProps: OP) => SP { pipette && configResponse && configResponse[pipette.id] const configSetConfigCall = pipette && getPipetteRequestById(state, ownProps.robot, pipette.id) + const devInternal = getConfig(state).devInternal return { pipette, pipetteConfig, configError: configSetConfigCall && configSetConfigCall.error, + __featureEnabled: !!devInternal && !!devInternal.allPipetteConfig, } } } diff --git a/app/src/components/InstrumentSettings/AttachedPipettesCard.js b/app/src/components/InstrumentSettings/AttachedPipettesCard.js index fe24b9d6ab9..7f5f82432d4 100644 --- a/app/src/components/InstrumentSettings/AttachedPipettesCard.js +++ b/app/src/components/InstrumentSettings/AttachedPipettesCard.js @@ -2,7 +2,6 @@ // attached pipettes container card import * as React from 'react' import {connect} from 'react-redux' -import {getIn} from '@thi.ng/paths' import { makeGetRobotPipettes, @@ -20,7 +19,6 @@ import {Card, IntervalWrapper} from '@opentrons/components' import type {State} from '../../types' import type {Robot} from '../../discovery' import type {Pipette} from '../../http-api-client' -import {getConfig} from '../../config' type OP = Robot @@ -29,7 +27,6 @@ type SP = { left: ?Pipette, right: ?Pipette, showSettings: boolean, - __featureEnabled: boolean, } type DP = { @@ -39,8 +36,6 @@ type DP = { type Props = OP & SP & DP -const __FEATURE_FLAG = 'devInternal.newPipetteConfig' - const TITLE = 'Pipettes' export default connect( @@ -59,7 +54,6 @@ function AttachedPipettesCard (props: Props) { {...props.left} onChangeClick={props.clearMove} showSettings={props.showSettings} - __enableConfig={props.__featureEnabled} /> @@ -88,7 +81,6 @@ function makeMapStateToProps (): (state: State, ownProps: OP) => SP { left, right, showSettings: !!configCall.response, - __featureEnabled: !!getIn(getConfig(state), __FEATURE_FLAG), } } } diff --git a/app/src/components/InstrumentSettings/InstrumentInfo.js b/app/src/components/InstrumentSettings/InstrumentInfo.js index 9e2a1e9f350..98794d1bd36 100644 --- a/app/src/components/InstrumentSettings/InstrumentInfo.js +++ b/app/src/components/InstrumentSettings/InstrumentInfo.js @@ -18,7 +18,6 @@ type Props = { name: string, onChangeClick: () => mixed, showSettings: boolean, - __enableConfig: boolean, } // TODO(mc, 2018-03-30): volume and channels should come from the API @@ -54,7 +53,7 @@ export default function PipetteInfo (props: Props) { {direction} - {props.__enableConfig && model && showSettings && ( + {model && showSettings && ( settings diff --git a/app/src/config/index.js b/app/src/config/index.js index 6594aaeb4f4..8b2a083bc90 100644 --- a/app/src/config/index.js +++ b/app/src/config/index.js @@ -64,7 +64,7 @@ export type Config = { // internal development flags devInternal?: { - newPipetteConfig?: boolean, + allPipetteConfig?: boolean, manualIp?: boolean, }, } From 0c6c5a98870960f482fcbafee571176eda31c3af Mon Sep 17 00:00:00 2001 From: Kadee80 Date: Wed, 13 Mar 2019 12:39:59 -0400 Subject: [PATCH 2/5] fixup: Cleanup unknown field logic and validation --- .../components/ConfigurePipette/ConfigForm.js | 34 ++++++++++--------- 1 file changed, 18 insertions(+), 16 deletions(-) diff --git a/app/src/components/ConfigurePipette/ConfigForm.js b/app/src/components/ConfigurePipette/ConfigForm.js index 2b49095c0ef..05eb3990292 100644 --- a/app/src/components/ConfigurePipette/ConfigForm.js +++ b/app/src/components/ConfigurePipette/ConfigForm.js @@ -60,6 +60,7 @@ export default class ConfigForm extends React.Component { } getVisibleFields = () => { + if (this.props.showHiddenFields) return this.props.pipetteConfig.fields return pick(this.props.pipetteConfig.fields, [ ...PLUNGER_KEYS, ...POWER_KEYS, @@ -67,12 +68,14 @@ export default class ConfigForm extends React.Component { ]) } - getHiddenFields = () => { - return omit(this.props.pipetteConfig.fields, [ - ...PLUNGER_KEYS, - ...POWER_KEYS, - ...TIP_KEYS, - ]) + getUnknownKeys = () => { + return keys( + omit(this.props.pipetteConfig.fields, [ + ...PLUNGER_KEYS, + ...POWER_KEYS, + ...TIP_KEYS, + ]) + ) } handleSubmit = (values: FormValues) => { @@ -102,14 +105,17 @@ export default class ConfigForm extends React.Component { // validate all visible fields with min and max forOwn(fields, (field, name) => { - const value = values[name] + const value = values[name].trim() const {min, max} = field - if (value !== '') { const parsed = Number(value) if (Number.isNaN(parsed)) { set(errors, name, `number required`) - } else if (min && max && (parsed < min || value > max)) { + } else if ( + typeof min === 'number' && + max && + (parsed < min || value > max) + ) { set(errors, name, `Min ${min} / Max ${max}`) } } @@ -131,18 +137,14 @@ export default class ConfigForm extends React.Component { render () { const {parentUrl} = this.props const fields = this.getVisibleFields() - const hiddenFields = this.getHiddenFields() - const HIDDEN_KEYS = keys(hiddenFields) - const visibleFields = this.props.showHiddenFields - ? this.props.pipetteConfig.fields - : fields - const initialValues = mapValues(visibleFields, f => { + const HIDDEN_KEYS = this.getUnknownKeys() + const initialValues = mapValues(fields, f => { return f.value !== f.default ? f.value.toString() : '' }) const plungerFields = this.getFieldsByKey(PLUNGER_KEYS, fields) const powerFields = this.getFieldsByKey(POWER_KEYS, fields) const tipFields = this.getFieldsByKey(TIP_KEYS, fields) - const devFields = this.getFieldsByKey(HIDDEN_KEYS, hiddenFields) + const devFields = this.getFieldsByKey(HIDDEN_KEYS, fields) return ( Date: Wed, 13 Mar 2019 12:50:25 -0400 Subject: [PATCH 3/5] fixup: Check max type --- app/src/components/ConfigurePipette/ConfigForm.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/src/components/ConfigurePipette/ConfigForm.js b/app/src/components/ConfigurePipette/ConfigForm.js index 05eb3990292..42becf6f1f1 100644 --- a/app/src/components/ConfigurePipette/ConfigForm.js +++ b/app/src/components/ConfigurePipette/ConfigForm.js @@ -113,7 +113,7 @@ export default class ConfigForm extends React.Component { set(errors, name, `number required`) } else if ( typeof min === 'number' && - max && + typeof max === 'number' && (parsed < min || value > max) ) { set(errors, name, `Min ${min} / Max ${max}`) From 2f5716de6e223c00c686ed2ff2cd9df9705e13e1 Mon Sep 17 00:00:00 2001 From: Kadee80 Date: Wed, 13 Mar 2019 12:58:42 -0400 Subject: [PATCH 4/5] fixup: Consistent naming for unknown field consts and methods --- app/src/components/ConfigurePipette/ConfigForm.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/app/src/components/ConfigurePipette/ConfigForm.js b/app/src/components/ConfigurePipette/ConfigForm.js index 42becf6f1f1..805d178fa21 100644 --- a/app/src/components/ConfigurePipette/ConfigForm.js +++ b/app/src/components/ConfigurePipette/ConfigForm.js @@ -137,14 +137,14 @@ export default class ConfigForm extends React.Component { render () { const {parentUrl} = this.props const fields = this.getVisibleFields() - const HIDDEN_KEYS = this.getUnknownKeys() + const UNKNOWN_KEYS = this.getUnknownKeys() const initialValues = mapValues(fields, f => { return f.value !== f.default ? f.value.toString() : '' }) const plungerFields = this.getFieldsByKey(PLUNGER_KEYS, fields) const powerFields = this.getFieldsByKey(POWER_KEYS, fields) const tipFields = this.getFieldsByKey(TIP_KEYS, fields) - const devFields = this.getFieldsByKey(HIDDEN_KEYS, fields) + const devFields = this.getFieldsByKey(UNKNOWN_KEYS, fields) return ( Date: Wed, 13 Mar 2019 15:16:59 -0400 Subject: [PATCH 5/5] fixup: Remove duplicate logic --- app/src/components/ConfigurePipette/ConfigForm.js | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/app/src/components/ConfigurePipette/ConfigForm.js b/app/src/components/ConfigurePipette/ConfigForm.js index 805d178fa21..0981905d8c1 100644 --- a/app/src/components/ConfigurePipette/ConfigForm.js +++ b/app/src/components/ConfigurePipette/ConfigForm.js @@ -98,9 +98,7 @@ export default class ConfigForm extends React.Component { validate = (values: FormValues) => { const errors = {} - const fields = this.props.showHiddenFields - ? this.props.pipetteConfig.fields - : this.getVisibleFields() + const fields = this.getVisibleFields() const plungerFields = this.getFieldsByKey(PLUNGER_KEYS, fields) // validate all visible fields with min and max