From f286261e398e9b2463c294cc66109b0f89f1f2c4 Mon Sep 17 00:00:00 2001 From: Sawyer Hollenshead Date: Thu, 5 Apr 2018 11:27:50 -0400 Subject: [PATCH] Fix: Ensure cloned masked field receives updated props (#264) When the field is a class property set in the constructor, it was only receiving the initial props. Any subsequent updates to the child TextField's props weren't being recognized. --- .../core/src/components/TextField/Mask.jsx | 44 +++++++++++++------ .../components/TextField/TextField.test.jsx | 10 +++++ 2 files changed, 40 insertions(+), 14 deletions(-) diff --git a/packages/core/src/components/TextField/Mask.jsx b/packages/core/src/components/TextField/Mask.jsx index 64f431233b..24dc400085 100644 --- a/packages/core/src/components/TextField/Mask.jsx +++ b/packages/core/src/components/TextField/Mask.jsx @@ -107,7 +107,7 @@ Style guide: components.masked-field.react export class Mask extends React.PureComponent { constructor(props) { super(props); - this.field = React.Children.only(this.props.children); + this.state = { value: this.maskedValue(this.initialValue()) }; @@ -115,11 +115,20 @@ export class Mask extends React.PureComponent { componentDidUpdate() { if (this.debouncedOnBlurEvent) { - this.field.props.onBlur(this.debouncedOnBlurEvent); + this.field().props.onBlur(this.debouncedOnBlurEvent); this.debouncedOnBlurEvent = null; } } + /** + * Get the child text field. Called as a method so that + * updates to the field cause the mask to re-render + * @returns {React.ReactElement} Child TextField + */ + field() { + return React.Children.only(this.props.children); + } + /** * Returns the value with additional masking characters * @param {String} value @@ -147,16 +156,16 @@ export class Mask extends React.PureComponent { * add/remove characters after the field has been blurred, * rather than when the user is typing in the field * @param {Object} evt + * @param {React.Element} field - Child TextField */ - handleBlur(evt) { + handleBlur(evt, field) { const value = this.maskedValue(evt.target.value); // We only debounce the onBlur when we know for sure that // this component will re-render (AKA when the value changes) // and when an onBlur callback is present const debounce = - value !== this.state.value && - typeof this.field.props.onBlur === 'function'; + value !== this.state.value && typeof field.props.onBlur === 'function'; if (debounce) { // We need to retain a reference to the event after the callback @@ -171,30 +180,37 @@ export class Mask extends React.PureComponent { value }); - if (!debounce && typeof this.field.props.onBlur === 'function') { + if (!debounce && typeof field.props.onBlur === 'function') { // If we didn't debounce the onBlur event, then we need to // call the onBlur callback from here - this.field.props.onBlur(evt); + field.props.onBlur(evt); } } - handleChange(evt) { + /** + * @param {Object} evt + * @param {React.Element} field - Child TextField + */ + handleChange(evt, field) { this.setState({ value: evt.target.value }); - if (typeof this.field.props.onChange === 'function') { - this.field.props.onChange(evt); + if (typeof field.props.onChange === 'function') { + field.props.onChange(evt); } } initialValue() { - return this.field.props.value || this.field.props.defaultValue; + const field = this.field(); + return field.props.value || field.props.defaultValue; } render() { - return React.cloneElement(this.field, { + const field = this.field(); + + return React.cloneElement(field, { defaultValue: undefined, - onBlur: evt => this.handleBlur(evt), - onChange: evt => this.handleChange(evt), + onBlur: evt => this.handleBlur(evt, field), + onChange: evt => this.handleChange(evt, field), value: this.state.value }); } diff --git a/packages/core/src/components/TextField/TextField.test.jsx b/packages/core/src/components/TextField/TextField.test.jsx index 4968b1daaf..44e1769651 100644 --- a/packages/core/src/components/TextField/TextField.test.jsx +++ b/packages/core/src/components/TextField/TextField.test.jsx @@ -286,5 +286,15 @@ describe('TextField', function() { expect(data.wrapper).toMatchSnapshot(); }); + + it('updates input classes when props are updated', () => { + const wrapper = render({ mask: 'currency' }, true).wrapper; + + expect(wrapper.find('input').hasClass('ds-c-field--error')).toBe(false); + + wrapper.setProps({ errorMessage: 'Oh no' }); + + expect(wrapper.find('input').hasClass('ds-c-field--error')).toBe(true); + }); }); });