Skip to content

Commit

Permalink
Merge pull request #109 from mbrookes/formsytext-fix-error-on-update
Browse files Browse the repository at this point in the history
[FormsyText] Add updateImmediately prop & fix handleChange
  • Loading branch information
mbrookes authored Sep 14, 2016
2 parents 46ef37c + 61314ac commit ff14bc3
Show file tree
Hide file tree
Showing 3 changed files with 86 additions and 38 deletions.
59 changes: 30 additions & 29 deletions examples/webpack-example/src/app/Main.js
Original file line number Diff line number Diff line change
Expand Up @@ -83,28 +83,15 @@ const Main = React.createClass({
onValidSubmit={this.submitForm}
onInvalidSubmit={this.notifyFormError}
>
<FormsyText
name="name"
validations="isWords"
validationError={wordsError}
<FormsyDate
name="date"
required
hintText="What is your name?"
floatingLabelText="Name"
/>
<FormsyText
name="age"
validations="isNumeric"
validationError={numericError}
hintText="Are you a wrinkly?"
floatingLabelText="Age (optional)"
floatingLabelText="Date"
/>
<FormsyText
name="url"
validations="isUrl"
validationError={urlError}
<FormsyTime
name="time"
required
hintText="http://www.example.com"
floatingLabelText="URL"
floatingLabelText="Time"
/>
<FormsySelect
name="frequency"
Expand All @@ -116,16 +103,6 @@ const Main = React.createClass({
<MenuItem value={'nightly'} primaryText="Every Night" />
<MenuItem value={'weeknights'} primaryText="Weeknights" />
</FormsySelect>
<FormsyDate
name="date"
required
floatingLabelText="Date"
/>
<FormsyTime
name="time"
required
floatingLabelText="Time"
/>
<FormsyCheckbox
name="agree"
label="Do you agree to disagree?"
Expand Down Expand Up @@ -154,6 +131,30 @@ const Main = React.createClass({
disabled={true}
/>
</FormsyRadioGroup>
<FormsyText
name="name"
validations="isWords"
validationError={wordsError}
required
hintText="What is your name?"
floatingLabelText="Name"
/>
<FormsyText
name="age"
validations="isNumeric"
validationError={numericError}
hintText="Are you a wrinkly?"
floatingLabelText="Age (optional)"
/>
<FormsyText
name="url"
validations="isUrl"
validationError={urlError}
required
hintText="http://www.example.com"
floatingLabelText="URL"
updateImmediately
/>
<RaisedButton
style={submitStyle}
type="submit"
Expand Down
54 changes: 45 additions & 9 deletions src/FormsyText.jsx
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ import React from 'react';
import keycode from 'keycode';
import Formsy from 'formsy-react';
import TextField from 'material-ui/TextField';
import { setMuiComponentAndMaybeFocus } from './utils';
import { setMuiComponentAndMaybeFocus, debounce } from './utils';

const FormsyText = React.createClass({

Expand All @@ -11,11 +11,11 @@ const FormsyText = React.createClass({
name: React.PropTypes.string.isRequired,
onBlur: React.PropTypes.func,
onChange: React.PropTypes.func,
onFocus: React.PropTypes.func,
onKeyDown: React.PropTypes.func,
validationError: React.PropTypes.string,
validationErrors: React.PropTypes.object,
validations: React.PropTypes.oneOfType([React.PropTypes.string, React.PropTypes.object]),
updateImmediately: React.PropTypes.bool,
value: React.PropTypes.any,
},

Expand Down Expand Up @@ -55,16 +55,51 @@ const FormsyText = React.createClass({
return props.value || props.defaultValue || '';
},

// Controlled component
componentWillReceiveProps(nextProps) {
if (nextProps.value !== this.props.value)
this.setState({
value: nextProps.value
});
},

handleBlur: function handleBlur(event) {
this.setValue(event.currentTarget.value);
delete this.changeValue;
if (this.props.onBlur) this.props.onBlur(event);
},

handleChange: function handleChange(event) {
this.setState({
value: event.currentTarget.value,
});
if (this.props.onChange) this.props.onChange(event);
// Update the value (and so display any error) after a timeout.
if (this.props.updateImmediately) {
if (!this.changeValue) {
this.changeValue = debounce(this.setValue, 400);
}
this.changeValue(event.currentTarget.value);
} else {
// If there was an error (on loss of focus) update on each keypress to resolve same.
if (this.getErrorMessage() != null) {
this.setValue(event.currentTarget.value);
} else {
// Only update on valid values, so as to not generate an error until focus is lost.
if (this.isValidValue(event.target.value)) {
this.setValue(event.currentTarget.value);
// If it becomes invalid, and there isn't an error message, invalidate without error.
} else {
this.resetValue();
}
}
}

// Controlled component
if (this.props.onChange) {
this.props.onChange(event, event.currentTarget.value);
// Uncontrolled component
} else {
this.setState({
value: event.currentTarget.value
});
}
},

handleKeyDown: function handleKeyDown(event) {
Expand All @@ -82,20 +117,21 @@ const FormsyText = React.createClass({
validationErrors, // eslint-disable-line no-unused-vars
onFocus,
value, // eslint-disable-line no-unused-vars
...rest } = this.props;
...rest
} = this.props;

return (
<TextField
{...rest}
errorText={this.getErrorMessage()}
onBlur={this.handleBlur}
onChange={this.handleChange}
onFocus={onFocus}
onKeyDown={this.handleKeyDown}
ref={this.setMuiComponentAndMaybeFocus}
value={this.state.value}
/>
);
},
}
});

export default FormsyText;
11 changes: 11 additions & 0 deletions src/utils.js
Original file line number Diff line number Diff line change
Expand Up @@ -9,3 +9,14 @@ export function setMuiComponentAndMaybeFocus(c) {
delete this.focus;
}
}

export function debounce(fn, delay) {
let timeout;
return function() {
const args = arguments;
clearTimeout(timeout);
timeout = setTimeout(() => {
fn.apply(this, args);
}, delay);
};
}

0 comments on commit ff14bc3

Please sign in to comment.