diff --git a/docs/components/Layout/index.jsx b/docs/components/Layout/index.jsx index 3bd7cf8bf..6d6767629 100644 --- a/docs/components/Layout/index.jsx +++ b/docs/components/Layout/index.jsx @@ -16,6 +16,7 @@ import AlertExample from '../../examples/AlertExample'; import CheckboxExample from '../../examples/CheckboxExample'; import CheckboxGroupExample from '../../examples/CheckboxGroupExample'; import RadioExample from '../../examples/RadioExample'; +import RadioGroupExample from '../../examples/RadioGroupExample'; import SelectExample from '../../examples/SelectExample'; import DatePickerExample from '../../examples/DatePickerExample'; import BorderedWellExample from '../../examples/BorderedWellExample'; @@ -178,6 +179,7 @@ class PageLayout extends React.Component { + diff --git a/docs/examples/RadioExample.jsx b/docs/examples/RadioExample.jsx index 4e4f0672f..1ecd58483 100644 --- a/docs/examples/RadioExample.jsx +++ b/docs/examples/RadioExample.jsx @@ -1,52 +1,78 @@ import React from 'react'; import Example from '../components/Example'; -import { Radio, RadioGroup } from '../../src'; +import Radio from 'adslot-ui/Radio'; class RadioExample extends React.PureComponent { + onChange(event) { + _.noop(); + } + render() { return ( - - - - + ); } } const exampleProps = { componentName: 'Radio', - designNotes: ( -

- Radio buttons used for making a single selection from multiple options. Only - one selection can ever be made from the radio button group at a time. -

- ), - notes: ( -

- See React iCheck Documentation -

- ), - exampleCodeSnippet: ` - - - - `, + notes: '', + exampleCodeSnippet: ``, propTypeSectionArray: [ { propTypes: [ + { + propType: 'id', + type: 'string', + }, + { + propType: 'className', + type: 'string', + note: 'This class will be applied to the input element', + }, + { + propType: 'name', + type: 'string', + }, { propType: 'label', - type: 'node', - note: 'Usually fine to rely on a string but can pass HTML e.g. for a url.', + type: 'string', }, { - propType: 'value', + propType: 'dts', type: 'string', + note: 'render `data-test-selector` onto the component. It can be useful for testing.', + }, + { + propType: 'disabled', + type: 'bool', + defaultValue: false, + }, + { + propType: 'checked', + type: 'bool', + defaultValue: false, }, { propType: 'onChange', type: 'func', }, + { + propType: 'value', + type: 'string', + }, ], }, ], diff --git a/docs/examples/RadioGroupExample.jsx b/docs/examples/RadioGroupExample.jsx new file mode 100644 index 000000000..c1fce4fb3 --- /dev/null +++ b/docs/examples/RadioGroupExample.jsx @@ -0,0 +1,89 @@ +import React from 'react'; +import Example from '../components/Example'; +import RadioGroup from 'adslot-ui/RadioGroup'; +import Radio from 'adslot-ui/Radio'; + +class RadioGroupExample extends React.PureComponent { + onChangeGroup(event) { + _.noop(); + } + + onChangeIndividual(event) { + _.noop(); + } + + render() { + return ( + + + + + + ); + } +} + +const exampleProps = { + componentName: 'RadioGroup', + designNotes: ( +

+ Radio buttons used for making a single selection from multiple options. Only + one selection can ever be made from the radio button group at a time. +

+ ), + notes: '', + exampleCodeSnippet: ` + + + +`, + propTypeSectionArray: [ + { + propTypes: [ + { + propType: 'name', + type: 'string', + note: ( + + Required. All Radio buttons within this group will have the same name + + ), + }, + { + propType: 'value', + type: 'string', + note: 'value of the selected radio button', + }, + { + propType: 'onChange', + type: 'func', + note: 'Triggers when selection changes.', + }, + { + propType: 'children', + type: 'node', + note: `Should be an array of components`, + }, + { + propType: 'className', + type: 'string', + }, + { + propType: 'dts', + type: 'string', + note: 'render `data-test-selector` onto the component. It can be useful for testing.', + }, + { + propType: 'id', + type: 'string', + }, + ], + }, + ], +}; + +export default () => ( + + + +); diff --git a/src/components/adslot-ui/Radio/index.jsx b/src/components/adslot-ui/Radio/index.jsx new file mode 100644 index 000000000..f812230cb --- /dev/null +++ b/src/components/adslot-ui/Radio/index.jsx @@ -0,0 +1,65 @@ +import _ from 'lodash'; +import React from 'react'; +import PropTypes from 'prop-types'; +import { expandDts } from 'lib/utils'; + +class RadioButton extends React.Component { + static getDerivedStateFromProps(newProps, prevState) { + return newProps.checked === prevState.checked ? null : { checked: newProps.checked }; + } + + constructor(props) { + super(props); + this.state = { checked: props.checked }; + + this.onChangeDefault = this.onChangeDefault.bind(this); + } + + onChangeDefault(event) { + this.setState({ checked: Boolean(event.target.checked) }); + if (this.props.onChange) { + this.props.onChange(event); + } + } + + render() { + const { name, className, label, dts, disabled, id, value } = this.props; + + const radioInputProps = { + type: 'radio', + name, + checked: this.state.checked, + disabled, + onChange: this.onChangeDefault, + value, + id, + className, + }; + + return ( +
+ + {label ? {label} : null} +
+ ); + } +} + +RadioButton.propTypes = { + id: PropTypes.string, + className: PropTypes.string, + name: PropTypes.string, + label: PropTypes.node, + dts: PropTypes.string, + disabled: PropTypes.bool, + checked: PropTypes.bool, + onChange: PropTypes.func, + value: PropTypes.string, +}; + +RadioButton.defaultProps = { + disabled: false, + checked: false, +}; + +export default RadioButton; diff --git a/src/components/adslot-ui/Radio/index.spec.jsx b/src/components/adslot-ui/Radio/index.spec.jsx new file mode 100644 index 000000000..45550e94f --- /dev/null +++ b/src/components/adslot-ui/Radio/index.spec.jsx @@ -0,0 +1,81 @@ +import _ from 'lodash'; +import React from 'react'; +import { shallow } from 'enzyme'; +import sinon from 'sinon'; +import Radio from '.'; + +describe('', () => { + let props; + + beforeEach(() => { + props = { + name: 'radio-name', + value: 'radio-value', + label: 'Radio 1', + dts: 'radio-dts', + id: 'radio-id', + className: 'radio-class', + disabled: false, + checked: false, + onChange: sinon.spy(), + }; + }); + + it('should render with props', () => { + const component = shallow(); + expect(component.find('input[type="radio"]')).to.have.length(1); + expect(component.text()).to.equal('Radio 1'); + expect(component.find('[name="radio-name"]')).to.have.length(1); + expect(component.find('[value="radio-value"]')).to.have.length(1); + expect(component.find('[data-test-selector="radio-dts"]')).to.have.length(1); + }); + + it('should not render label if props.label is undefined', () => { + delete props.label; + const component = shallow(); + expect(component.text()).to.equal(''); + }); + + it('should trigger state change and `props.onChange` when change event is triggered', () => { + const component = shallow(); + const event = { target: { checked: true } }; + + expect(component.state('checked')).to.equal(false); + + component.find('input').simulate('change', event); + expect(component.state('checked')).to.equal(true); + expect(props.onChange.calledOnce).to.equal(true); + }); + + it('should still trigger state change when `props.onChange` is not present', () => { + delete props.onChange; + const component = shallow(); + const event = { target: { checked: true } }; + + expect(component.state('checked')).to.equal(false); + + component.find('input').simulate('change', event); + expect(component.state('checked')).to.equal(true); + }); + + it('should override state value when `prop.value` changes', () => { + const component = shallow(); + expect(component.state('checked')).to.equal(false); + + props.checked = true; + component.setProps(props); + expect(component.state('checked')).to.equal(true); + }); + + it('should NOT override state value when other props change', () => { + const component = shallow(); + expect(component.state('checked')).to.equal(false); + + _.assign(props, { + name: 'some-other-name', + label: 'New Label', + }); + component.setProps(props); + expect(component.state('checked')).to.equal(false); + }); +}); diff --git a/src/components/adslot-ui/RadioGroup/index.jsx b/src/components/adslot-ui/RadioGroup/index.jsx new file mode 100644 index 000000000..afd778d91 --- /dev/null +++ b/src/components/adslot-ui/RadioGroup/index.jsx @@ -0,0 +1,71 @@ +import _ from 'lodash'; +import React from 'react'; +import PropTypes from 'prop-types'; +import { expandDts } from 'lib/utils'; + +class RadioGroup extends React.Component { + static getDerivedStateFromProps(nextProps, prevState) { + return nextProps.value === prevState.value ? null : { value: nextProps.value }; + } + + constructor(props) { + super(props); + + this.state = { + value: props.value, + }; + + this.onChangeDefault = this.onChangeDefault.bind(this); + this.renderChildren = this.renderChildren.bind(this); + } + + onChangeDefault(event) { + this.setState({ value: event.target.value }); + if (this.props.onChange) { + this.props.onChange(event); + } + } + + renderChildren() { + return React.Children.map(this.props.children, child => { + const childProps = _.assign({}, child.props, { + name: this.props.name, + checked: this.state.value === child.props.value, + onChange: (...args) => { + if (child.props.onChange) child.props.onChange(...args); + this.onChangeDefault(...args); + }, + }); + + return React.cloneElement(child, childProps); + }); + } + + render() { + const { dts, className, id } = this.props; + const props = _.pickBy({ + id, + className: _(['radio-group-component', className]) + .compact() + .join(' '), + }); + + return ( +
+ {this.renderChildren()} +
+ ); + } +} + +RadioGroup.propTypes = { + name: PropTypes.string.isRequired, + value: PropTypes.string, + onChange: PropTypes.func, + children: PropTypes.node, + className: PropTypes.string, + dts: PropTypes.string, + id: PropTypes.string, +}; + +export default RadioGroup; diff --git a/src/components/adslot-ui/RadioGroup/index.spec.jsx b/src/components/adslot-ui/RadioGroup/index.spec.jsx new file mode 100644 index 000000000..3d8e062a7 --- /dev/null +++ b/src/components/adslot-ui/RadioGroup/index.spec.jsx @@ -0,0 +1,119 @@ +import React from 'react'; +import { shallow } from 'enzyme'; +import sinon from 'sinon'; +import Radio from '../Radio'; +import RadioGroup from '.'; + +describe('', () => { + let props; + + beforeEach(() => { + props = { + id: 'radio-group-id', + className: 'radio-group-custom-class', + name: 'hobbies', + value: 'badminton', + onChange: sinon.spy(), + dts: 'radio-group-dts', + }; + }); + + it('should render with props', () => { + const component = shallow( + + + + + + ); + + expect(component.find(Radio)).to.have.length(3); + expect(component.find('#radio-group-id')).to.have.length(1); + expect(component.find('.radio-group-custom-class')).to.have.length(1); + expect(component.find('[data-test-selector="radio-group-dts"]')).to.have.length(1); + }); + + it('should update state and trigger props.onChange when selection changes', () => { + const component = shallow( + + + + + + ); + + expect(component.state('value')).to.equal('badminton'); + component + .find(Radio) + .at(0) + .simulate('change', { target: { value: 'swimming' } }); + expect(component.state('value')).to.equal('swimming'); + expect(props.onChange.calledOnce).to.equal(true); + }); + + it('should still update state when props.onChange is not available', () => { + delete props.onChange; + const component = shallow( + + + + + + ); + + expect(component.state('value')).to.equal('badminton'); + component + .find(Radio) + .at(1) + .simulate('change', { target: { value: 'soccer' } }); + expect(component.state('value')).to.equal('soccer'); + }); + + it('should call Radio props.onChange when selecting that radio button', () => { + const onChangeSwimming = sinon.spy(); + const onChangeSoccer = sinon.spy(); + + const component = shallow( + + + + + + ); + + component + .find(Radio) + .at(0) + .simulate('change', { target: { value: 'swimming' } }); + expect(onChangeSwimming.calledOnce).to.equal(true); + expect(onChangeSoccer.called).to.equal(false); + expect(props.onChange.calledOnce).to.equal(true); + }); + + describe('getDerivedStateFromProps()', () => { + let component; + + beforeEach(() => { + component = shallow( + + + + + + ); + }); + + it('should update selected value when `prop.value` changes', () => { + props.value = 'soccer'; + component.setProps(props); + expect(component.state('value')).to.equal('soccer'); + }); + + it('should NOT update selected value when other props attributes change', () => { + props.id = 'new-id'; + props.name = 'some-other-name'; + component.setProps(props); + expect(component.state('value')).to.equal('badminton'); + }); + }); +}); diff --git a/src/components/adslot-ui/index.js b/src/components/adslot-ui/index.js index de528cda9..eb8841557 100644 --- a/src/components/adslot-ui/index.js +++ b/src/components/adslot-ui/index.js @@ -11,6 +11,8 @@ import ListPicker from 'adslot-ui/ListPicker'; import ListPickerPure from 'adslot-ui/ListPickerPure'; import PagedGrid from 'adslot-ui/PagedGrid'; import Panel from 'adslot-ui/Panel'; +import Radio from 'adslot-ui/Radio'; +import RadioGroup from 'adslot-ui/RadioGroup'; import Search from 'adslot-ui/Search'; import SearchBar from 'adslot-ui/SearchBar'; import SplitPane from 'adslot-ui/SplitPane'; @@ -42,6 +44,8 @@ export { Nav, PagedGrid, Panel, + Radio, + RadioGroup, Search, SearchBar, SplitPane, diff --git a/src/index.js b/src/index.js index f032877a3..45075ffd0 100644 --- a/src/index.js +++ b/src/index.js @@ -1,6 +1,4 @@ // Export the consumable components. -import Radio from 'react-icheck/lib/Radio'; -import RadioGroup from 'react-icheck/lib/RadioGroup'; import Select from 'react-select'; import DatePicker from 'react-datepicker'; @@ -63,6 +61,8 @@ import { Nav, PagedGrid, Panel, + Radio, + RadioGroup, Search, SearchBar, SplitPane,