-
Notifications
You must be signed in to change notification settings - Fork 3.6k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
[changed] Deprecating Input type=radio in favor of new RadioButton an…
…d RadioGroup components
- Loading branch information
Showing
12 changed files
with
317 additions
and
39 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,33 @@ | ||
const SimpleRadioGroup = React.createClass({ | ||
getInitialState() { | ||
return {message: undefined}; | ||
}, | ||
|
||
handleChange(newValue) { | ||
this.setState({message: newValue}); | ||
}, | ||
|
||
render() { | ||
const message = this.state.message ? | ||
<FormControls.Static>{this.state.message}</FormControls.Static> : | ||
null; | ||
|
||
const handleChange = this.handleChange; | ||
|
||
return ( | ||
<form> | ||
{message} | ||
<FormControls.RadioGroup name="greeting"> | ||
<FormControls.RadioButton label="Hi!" value="Hi!" /> | ||
<FormControls.RadioButton label="Hello!" value="Hello!" /> | ||
</FormControls.RadioGroup> | ||
<FormControls.RadioGroup name="farewell" onChange={handleChange}> | ||
<FormControls.RadioButton label="Bye!" value="Bye!" /> | ||
<FormControls.RadioButton label="Goodbye!" value="Goodbye!" /> | ||
</FormControls.RadioGroup> | ||
</form> | ||
); | ||
} | ||
}); | ||
|
||
React.render(<SimpleRadioGroup />, mountNode); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,57 @@ | ||
import React from 'react'; | ||
import classNames from 'classnames'; | ||
|
||
export default class RadioButton extends React.Component { | ||
constructor(props) { | ||
super(props); | ||
} | ||
|
||
getInputDOMNode() { | ||
return React.findDOMNode(this.refs.input); | ||
} | ||
|
||
getValue() { | ||
return this.props.value; | ||
} | ||
|
||
getChecked() { | ||
return this.getInputDOMNode().checked; | ||
} | ||
|
||
renderLabel(input) { | ||
if (!this.props.label) { | ||
return input; | ||
} | ||
|
||
const {id, label} = this.props; | ||
|
||
const classes = { | ||
'radio': !this.props.inline, | ||
'radio-inline': this.props.inline | ||
}; | ||
|
||
return ( | ||
<label className={classNames(classes)} htmlFor={id}> | ||
{input} | ||
{label} | ||
</label> | ||
); | ||
} | ||
|
||
render() { | ||
const {inline, label, ...other} = this.props; | ||
return this.renderLabel( | ||
<input type="radio" {...other} ref="input" /> | ||
); | ||
} | ||
} | ||
|
||
RadioButton.propTypes = { | ||
disabled: React.PropTypes.bool, | ||
id: React.PropTypes.string, | ||
inline: React.PropTypes.bool, | ||
label: React.PropTypes.node, | ||
name: React.PropTypes.string, | ||
onChange: React.PropTypes.func, | ||
value: React.PropTypes.any | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,55 @@ | ||
import React from 'react'; | ||
|
||
export default class RadioGroup extends React.Component { | ||
constructor(props) { | ||
super(props); | ||
this.state = {value: undefined}; | ||
this.handleChange = this.handleChange.bind(this); | ||
} | ||
|
||
getValue() { | ||
return this.state.value; | ||
} | ||
|
||
renderLegend() { | ||
return this.props.legend ? | ||
<legend>{this.props.legend}</legend> : | ||
null; | ||
} | ||
|
||
handleChange(evt) { | ||
if (!evt.target) { | ||
return; | ||
} | ||
|
||
const {value} = evt.target; | ||
this.setState({value}); | ||
this.props.onChange(value); | ||
} | ||
|
||
renderChildren() { | ||
const {name, inline} = this.props; | ||
return React.Children.map(this.props.children, child => React.cloneElement(child, {name, inline, onChange: this.handleChange})); | ||
} | ||
|
||
render() { | ||
const children = this.renderChildren(); | ||
return ( | ||
<fieldset ref="radioGroup" className="radio"> | ||
{this.renderLegend()} | ||
{children} | ||
</fieldset> | ||
); | ||
} | ||
} | ||
|
||
RadioGroup.propTypes = { | ||
inline: React.PropTypes.bool, | ||
legend: React.PropTypes.node, | ||
name: React.PropTypes.string.isRequired, | ||
onChange: React.PropTypes.func | ||
}; | ||
|
||
RadioGroup.defaultProps = { | ||
onChange: () => {} | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,5 +1,9 @@ | ||
import RadioButton from './RadioButton'; | ||
import RadioGroup from './RadioGroup'; | ||
import Static from './Static'; | ||
|
||
export default { | ||
Static | ||
Static, | ||
RadioGroup, | ||
RadioButton | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -5,6 +5,7 @@ | |
"globals": { | ||
"assert": true, | ||
"expect": true, | ||
"should": true, | ||
"sinon": true | ||
}, | ||
"plugins": [ | ||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,49 @@ | ||
import React from 'react'; | ||
import ReactTestUtils from 'react/lib/ReactTestUtils'; | ||
import RadioButton from '../../src/FormControls/RadioButton'; | ||
|
||
describe('FormControls.RadioButton', function () { | ||
it('does not apply htmlFor to the label if no id is provided', function () { | ||
const instance = ReactTestUtils.renderIntoDocument( | ||
<RadioButton label="test" value="test" /> | ||
); | ||
|
||
const label = ReactTestUtils.findRenderedDOMComponentWithTag(instance, 'label'); | ||
should.not.exist(label.props.htmlFor); | ||
}); | ||
|
||
it('does not apply htmlFor to the label if no id is provided', function () { | ||
const testId = 'thing'; | ||
const instance = ReactTestUtils.renderIntoDocument( | ||
<RadioButton id={testId} label="test" value="test" /> | ||
); | ||
|
||
const label = ReactTestUtils.findRenderedDOMComponentWithTag(instance, 'label'); | ||
label.props.htmlFor.should.eql(testId); | ||
}); | ||
|
||
it('does not render a label if no label prop is provided', function () { | ||
const instance = ReactTestUtils.renderIntoDocument( | ||
<RadioButton value="test" /> | ||
); | ||
|
||
React.findDOMNode(instance).tagName.toLowerCase().should.eql('input'); | ||
}); | ||
|
||
it('renders a label if label prop is provided', function () { | ||
const instance = ReactTestUtils.renderIntoDocument( | ||
<RadioButton label="thing" value="test" /> | ||
); | ||
|
||
React.findDOMNode(instance).tagName.toLowerCase().should.eql('label'); | ||
}); | ||
|
||
it('should render the label content', function () { | ||
const testText = 'thing'; | ||
const instance = ReactTestUtils.renderIntoDocument( | ||
<RadioButton label={testText} value="test" /> | ||
); | ||
|
||
React.findDOMNode(instance).textContent.should.eql(testText); | ||
}); | ||
}); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,74 @@ | ||
import React from 'react'; | ||
import ReactTestUtils from 'react/lib/ReactTestUtils'; | ||
import RadioButton from '../../src/FormControls/RadioButton'; | ||
import RadioGroup from '../../src/FormControls/RadioGroup'; | ||
|
||
describe('RadioGroup', function () { | ||
it('applies an on change to children', function () { | ||
const instance = ReactTestUtils.renderIntoDocument( | ||
<RadioGroup name="test"> | ||
<RadioButton label="thing" value="1" /> | ||
<RadioButton label="pants" value="1" /> | ||
<RadioButton label="jeans" value="1" /> | ||
</RadioGroup> | ||
); | ||
|
||
const buttons = ReactTestUtils.scryRenderedDOMComponentsWithTag(instance, 'INPUT'); | ||
buttons.every(button => button.props.onChange === instance.handleChange).should.be.true; | ||
}); | ||
|
||
it('invokes a callback when a selection is changed', function() { | ||
const callback = sinon.spy(); | ||
|
||
const instance = ReactTestUtils.renderIntoDocument( | ||
<RadioGroup name="test" onChange={callback}> | ||
<RadioButton label="thing" value="thing" id="thing" className="thing" /> | ||
<RadioButton label="jeans" value="jeans" id="jeans" /> | ||
</RadioGroup> | ||
); | ||
|
||
const thing = ReactTestUtils.findRenderedDOMComponentWithClass(instance, 'thing'); | ||
ReactTestUtils.Simulate.change(thing, {target: {value: 5}}); | ||
callback.should.have.been.calledWith(5); | ||
}); | ||
|
||
it('applies the name property to all children', function () { | ||
const testName = 'test'; | ||
const instance = ReactTestUtils.renderIntoDocument( | ||
<RadioGroup name={testName}> | ||
<RadioButton label="thing" value="1" /> | ||
<RadioButton label="pants" value="1" /> | ||
<RadioButton label="jeans" value="1" /> | ||
</RadioGroup> | ||
); | ||
|
||
const buttons = ReactTestUtils.scryRenderedDOMComponentsWithTag(instance, 'INPUT'); | ||
buttons.every(button => button.props.name === testName).should.be.true; | ||
}); | ||
|
||
it('does not inline if the inline property is false', function () { | ||
const instance = ReactTestUtils.renderIntoDocument( | ||
<RadioGroup name="test" inline={false}> | ||
<RadioButton label="thing" value="1" /> | ||
<RadioButton label="pants" value="1" /> | ||
<RadioButton label="jeans" value="1" /> | ||
</RadioGroup> | ||
); | ||
|
||
const labels = ReactTestUtils.scryRenderedDOMComponentsWithTag(instance, 'LABEL'); | ||
labels.every(label => label.props.className === 'radio').should.be.true; | ||
}); | ||
|
||
it('applies the radio-inline class when the inline prop is used', function () { | ||
const instance = ReactTestUtils.renderIntoDocument( | ||
<RadioGroup name="test" inline> | ||
<RadioButton label="thing" value="1" /> | ||
<RadioButton label="pants" value="1" /> | ||
<RadioButton label="jeans" value="1" /> | ||
</RadioGroup> | ||
); | ||
|
||
const labels = ReactTestUtils.scryRenderedDOMComponentsWithTag(instance, 'LABEL'); | ||
labels.every(label => label.props.className === 'radio-inline').should.be.true; | ||
}); | ||
}); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,35 @@ | ||
import React from 'react'; | ||
import ReactTestUtils from 'react/lib/ReactTestUtils'; | ||
import Static from '../../src/FormControls/Static'; | ||
|
||
describe('FormControls.Static', function () { | ||
it('renders a p element wrapped around the given value', function () { | ||
const instance = ReactTestUtils.renderIntoDocument( | ||
<Static value='v' /> | ||
); | ||
|
||
const result = ReactTestUtils.findRenderedDOMComponentWithTag(instance, 'p'); | ||
result.props.children.should.equal('v'); | ||
}); | ||
|
||
it('getValue() pulls from either value or children', function () { | ||
let instance = ReactTestUtils.renderIntoDocument( | ||
<Static value='v' /> | ||
); | ||
|
||
instance.getValue().should.equal('v'); | ||
|
||
instance = ReactTestUtils.renderIntoDocument( | ||
<Static>5</Static> | ||
); | ||
|
||
instance.getValue().should.equal('5'); | ||
}); | ||
|
||
it('throws an error if both value and children are provided', function () { | ||
const testData = { value: 'blah', children: 'meh' }; | ||
const result = Static.propTypes.children(testData, 'children', 'Static'); | ||
|
||
result.should.be.instanceOf(Error); | ||
}); | ||
}); |
This file was deleted.
Oops, something went wrong.