From 2749cfdf8fc893ce0f068a7e41045f72a3f39cbc Mon Sep 17 00:00:00 2001 From: Alex Abenoja Date: Fri, 15 May 2015 00:14:49 -0600 Subject: [PATCH 1/4] [added] CustomPropTypes.singlePropFrom Throw an error if multiple properties in the given list have a value --- src/utils/CustomPropTypes.js | 30 +++++++++++++++++++++++++++++- test/CustomPropTypesSpec.js | 20 ++++++++++++++++++++ 2 files changed, 49 insertions(+), 1 deletion(-) diff --git a/src/utils/CustomPropTypes.js b/src/utils/CustomPropTypes.js index c56a0a4e8a..5a6ae4e0fc 100644 --- a/src/utils/CustomPropTypes.js +++ b/src/utils/CustomPropTypes.js @@ -22,7 +22,17 @@ let CustomPropTypes = { * @param componentName * @returns {Error|undefined} */ - keyOf: createKeyOfChecker + keyOf: createKeyOfChecker, + /** + * Checks if only one of the listed properties is in use. An error is given + * if multiple have a value + * + * @param props + * @param propName + * @param componentName + * @returns {Error|undefined} + */ + singlePropFrom: createSinglePropFromChecker }; /** @@ -80,4 +90,22 @@ function createKeyOfChecker(obj) { return createChainableTypeChecker(validate); } +function createSinglePropFromChecker(arrOfProps) { + function validate(props, propName, componentName) { + const usedPropCount = arrOfProps + .map(listedProp => props[listedProp]) + .reduce((acc, curr) => acc + (curr !== undefined ? 1 : 0), 0); + + if (usedPropCount > 1) { + const [first, ...others] = arrOfProps; + const message = `${others.join(', ')} and ${first}`; + return new Error( + `Invalid prop '${propName}', only one of the following ` + + `may be provided: ${message}` + ); + } + } + return validate; +} + export default CustomPropTypes; diff --git a/test/CustomPropTypesSpec.js b/test/CustomPropTypesSpec.js index 9a0b94deb2..d28137f585 100644 --- a/test/CustomPropTypesSpec.js +++ b/test/CustomPropTypesSpec.js @@ -47,4 +47,24 @@ describe('CustomPropTypes', function () { assert.isUndefined(validate('bar')); }); }); + + describe('singlePropFrom', function () { + function validate(testProps) { + const propList = ['children', 'value']; + + return CustomPropTypes.singlePropFrom(propList)(testProps, 'value', 'Component'); + } + + it('Should return undefined if only one listed prop in used', function () { + const testProps = {value: 5}; + + assert.isUndefined(validate(testProps)); + }); + + it('Should return error if multiple of the listed properties have values', function () { + const testProps = {value: 5, children: 5}; + + validate(testProps).should.be.instanceOf(Error); + }); + }); }); From fd0972e08709219e2067272a000a501621bf6aca Mon Sep 17 00:00:00 2001 From: Alex Abenoja Date: Thu, 14 May 2015 14:04:50 -0600 Subject: [PATCH 2/4] DRYing ButtonInput --- src/ButtonInput.js | 16 ++++++---------- src/Input.js | 4 +--- src/utils/childrenValueInputValidation.js | 14 ++++++++++++++ test/ButtonInputSpec.js | 8 ++++---- 4 files changed, 25 insertions(+), 17 deletions(-) create mode 100644 src/utils/childrenValueInputValidation.js diff --git a/src/ButtonInput.js b/src/ButtonInput.js index 4df4c6a9f1..8667df0178 100644 --- a/src/ButtonInput.js +++ b/src/ButtonInput.js @@ -2,13 +2,7 @@ import React from 'react'; import Button from './Button'; import FormGroup from './FormGroup'; import InputBase from './InputBase'; - -function valueValidation({children, value}, propName, componentName) { - if (children && value) { - return new Error('Both value and children cannot be passed to ButtonInput'); - } - return React.PropTypes.oneOfType([React.PropTypes.string, React.PropTypes.number]).call(null, {children, value}, propName, componentName); -} +import childrenValueValidation from './utils/childrenValueInputValidation'; class ButtonInput extends InputBase { renderFormGroup(children) { @@ -23,18 +17,20 @@ class ButtonInput extends InputBase { } } +ButtonInput.types = ['button', 'reset', 'submit']; + ButtonInput.defaultProps = { type: 'button' }; ButtonInput.propTypes = { - type: React.PropTypes.oneOf(['button', 'reset', 'submit']), + type: React.PropTypes.oneOf(ButtonInput.types), bsStyle(props) { //defer to Button propTypes of bsStyle return null; }, - children: valueValidation, - value: valueValidation + children: childrenValueValidation, + value: childrenValueValidation }; export default ButtonInput; diff --git a/src/Input.js b/src/Input.js index 61456c70c3..6cd7c47776 100644 --- a/src/Input.js +++ b/src/Input.js @@ -3,11 +3,9 @@ import InputBase from './InputBase'; import ButtonInput from './ButtonInput'; import deprecationWarning from './utils/deprecationWarning'; -const buttonTypes = ['button', 'reset', 'submit']; - class Input extends InputBase { render() { - if (buttonTypes.indexOf(this.props.type) > -1) { + if (ButtonInput.types.indexOf(this.props.type) > -1) { deprecationWarning(`Input type=${this.props.type}`, 'ButtonInput'); return ; } diff --git a/src/utils/childrenValueInputValidation.js b/src/utils/childrenValueInputValidation.js new file mode 100644 index 0000000000..075a4c42b0 --- /dev/null +++ b/src/utils/childrenValueInputValidation.js @@ -0,0 +1,14 @@ +import React from 'react'; +import { singlePropFrom } from './CustomPropTypes'; + +const propList = ['children', 'value']; +const typeList = [React.PropTypes.number, React.PropTypes.string]; + +export default function valueValidation(props, propName, componentName) { + let error = singlePropFrom(propList)(props, propName, componentName); + if (!error) { + const oneOfType = React.PropTypes.oneOfType(typeList); + error = oneOfType(props, propName, componentName); + } + return error; +} diff --git a/test/ButtonInputSpec.js b/test/ButtonInputSpec.js index 5118181938..7f5e66ac66 100644 --- a/test/ButtonInputSpec.js +++ b/test/ButtonInputSpec.js @@ -55,11 +55,11 @@ describe('ButtonInput', () =>{ }); it('throws a warning if given both children and a value property', function () { - ReactTestUtils.renderIntoDocument( - button - ); + const testData = { value: 5, children: 'button' }; + const result = ButtonInput.propTypes.value(testData, 'value', 'ButtonInput'); - shouldWarn('Both value and children'); + result.should.be.instanceOf(Error); + result.message.should.have.string('value and children'); }); it('does not throw an error for strings and numbers', function () { From 0c61f4634b404171abdea856c17b0e88a9e63a3a Mon Sep 17 00:00:00 2001 From: Alex Abenoja Date: Thu, 14 May 2015 11:03:49 -0600 Subject: [PATCH 3/4] [changed] Moving type=static out of Input Introducing the Static component. Usage of type=static is now deprecated. Please use Static instead. --- src/FormControls/Static.js | 26 ++++++++++++++++++++++++++ src/FormControls/index.js | 5 +++++ src/Input.js | 4 ++++ src/index.js | 2 ++ test/FormControlsSpec.js | 37 +++++++++++++++++++++++++++++++++++++ test/InputSpec.js | 7 +++---- 6 files changed, 77 insertions(+), 4 deletions(-) create mode 100644 src/FormControls/Static.js create mode 100644 src/FormControls/index.js create mode 100644 test/FormControlsSpec.js diff --git a/src/FormControls/Static.js b/src/FormControls/Static.js new file mode 100644 index 0000000000..3f0f4a40a3 --- /dev/null +++ b/src/FormControls/Static.js @@ -0,0 +1,26 @@ +import React from 'react'; +import classNames from 'classnames'; +import InputBase from '../InputBase'; +import childrenValueValidation from '../utils/childrenValueInputValidation'; + +class Static extends InputBase { + getValue() { + const {children, value} = this.props; + return children ? children : value; + } + + renderInput() { + return ( +

+ {this.getValue()} +

+ ); + } +} + +Static.propTypes = { + value: childrenValueValidation, + children: childrenValueValidation +}; + +export default Static; diff --git a/src/FormControls/index.js b/src/FormControls/index.js new file mode 100644 index 0000000000..5a7c16286e --- /dev/null +++ b/src/FormControls/index.js @@ -0,0 +1,5 @@ +import Static from './Static'; + +export default { + Static +}; diff --git a/src/Input.js b/src/Input.js index 6cd7c47776..fa516a4ee4 100644 --- a/src/Input.js +++ b/src/Input.js @@ -1,6 +1,7 @@ import React from 'react'; import InputBase from './InputBase'; import ButtonInput from './ButtonInput'; +import FormControls from './FormControls'; import deprecationWarning from './utils/deprecationWarning'; class Input extends InputBase { @@ -8,6 +9,9 @@ class Input extends InputBase { if (ButtonInput.types.indexOf(this.props.type) > -1) { deprecationWarning(`Input type=${this.props.type}`, 'ButtonInput'); return ; + } else if (this.props.type === 'static') { + deprecationWarning('Input type=static', 'StaticText'); + return ; } return super.render(); diff --git a/src/index.js b/src/index.js index 118bacd7d9..2844c7ae9a 100644 --- a/src/index.js +++ b/src/index.js @@ -19,6 +19,7 @@ import DropdownButton from './DropdownButton'; import DropdownMenu from './DropdownMenu'; import DropdownStateMixin from './DropdownStateMixin'; import FadeMixin from './FadeMixin'; +import FormControls from './FormControls'; import Glyphicon from './Glyphicon'; import Grid from './Grid'; import Input from './Input'; @@ -75,6 +76,7 @@ export default { DropdownMenu, DropdownStateMixin, FadeMixin, + FormControls, Glyphicon, Grid, Input, diff --git a/test/FormControlsSpec.js b/test/FormControlsSpec.js new file mode 100644 index 0000000000..cc56ecd618 --- /dev/null +++ b/test/FormControlsSpec.js @@ -0,0 +1,37 @@ +import React from 'react'; +import ReactTestUtils from 'react/lib/ReactTestUtils'; +import FormControls from '../src/FormControls'; + +describe('Form Controls', function () { + describe('Static', function () { + it('renders a p element wrapped around the given value', function () { + const instance = ReactTestUtils.renderIntoDocument( + + ); + + 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( + + ); + + instance.getValue().should.equal('v'); + + instance = ReactTestUtils.renderIntoDocument( + 5 + ); + + 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 = FormControls.Static.propTypes.children(testData, 'children', 'Static'); + + result.should.be.instanceOf(Error); + }); + }); +}); diff --git a/test/InputSpec.js b/test/InputSpec.js index 065ff1824c..3a73cba5cf 100644 --- a/test/InputSpec.js +++ b/test/InputSpec.js @@ -65,13 +65,12 @@ describe('Input', function () { shouldWarn('deprecated'); }); - it('renders a p element when type=static', function () { - let instance = ReactTestUtils.renderIntoDocument( + it('throws a warning when type=static', function () { + ReactTestUtils.renderIntoDocument( ); - assert.ok(ReactTestUtils.findRenderedDOMComponentWithTag(instance, 'p')); - assert.equal(instance.getValue(), 'v'); + shouldWarn('deprecated'); }); it('renders an input element of given type when type is anything else', function () { From 93c95b6ca4668e288807558677b05155bf456a76 Mon Sep 17 00:00:00 2001 From: Alex Abenoja Date: Fri, 15 May 2015 10:45:13 -0600 Subject: [PATCH 4/4] Updating docs with new Static component --- docs/examples/.eslintrc | 1 + docs/examples/InputTypes.js | 1 - docs/examples/StaticText.js | 9 +++++++++ docs/src/ComponentsPage.js | 4 +++- docs/src/ReactPlayground.js | 2 ++ docs/src/Samples.js | 1 + 6 files changed, 16 insertions(+), 2 deletions(-) create mode 100644 docs/examples/StaticText.js diff --git a/docs/examples/.eslintrc b/docs/examples/.eslintrc index b6dc693611..358c61473e 100644 --- a/docs/examples/.eslintrc +++ b/docs/examples/.eslintrc @@ -20,6 +20,7 @@ "CarouselItem", "Col", "DropdownButton", + "FormControls", "Glyphicon", "Grid", "Input", diff --git a/docs/examples/InputTypes.js b/docs/examples/InputTypes.js index 036f6d6199..63a418e8a7 100644 --- a/docs/examples/InputTypes.js +++ b/docs/examples/InputTypes.js @@ -15,7 +15,6 @@ const inputTypeInstance = ( - diff --git a/docs/examples/StaticText.js b/docs/examples/StaticText.js new file mode 100644 index 0000000000..1b08ab93d9 --- /dev/null +++ b/docs/examples/StaticText.js @@ -0,0 +1,9 @@ +const staticTextExample = ( +
+ + + Bob + +); + +React.render(staticTextExample, mountNode); diff --git a/docs/src/ComponentsPage.js b/docs/src/ComponentsPage.js index e268bc27f5..863e71b78d 100644 --- a/docs/src/ComponentsPage.js +++ b/docs/src/ComponentsPage.js @@ -577,8 +577,10 @@ const ComponentsPage = React.createClass({ The helper method getInputDOMNode() returns the internal input element. If you don't want the form-group class applied apply the prop named standalone.

Types

-

Supports select, textarea, static as well as standard HTML input types. getValue() returns an array for multiple select.

+

Supports select, textarea, as well as standard HTML input types. getValue() returns an array for multiple select.

+

Static text can be added to your form controls through the use of the FormControls.Static component.

+

Button Input Types

Form buttons are encapsulated by ButtonInput. Pass in type="reset" or type="submit" to suit your needs. Styling is the same as Button.

diff --git a/docs/src/ReactPlayground.js b/docs/src/ReactPlayground.js index 6561e87287..097fd47fb5 100644 --- a/docs/src/ReactPlayground.js +++ b/docs/src/ReactPlayground.js @@ -13,6 +13,7 @@ import * as modCarousel from '../../src/Carousel'; import * as modCarouselItem from '../../src/CarouselItem'; import * as modCol from '../../src/Col'; import * as modDropdownButton from '../../src/DropdownButton'; +import * as modFormControls from '../../src/FormControls'; import * as modGlyphicon from '../../src/Glyphicon'; import * as modGrid from '../../src/Grid'; import * as modInput from '../../src/Input'; @@ -64,6 +65,7 @@ const Carousel = modCarousel.default; const CarouselItem = modCarouselItem.default; const Col = modCol.default; const DropdownButton = modDropdownButton.default; +const FormControls = modFormControls.default; const Glyphicon = modGlyphicon.default; const Grid = modGrid.default; const Input = modInput.default; diff --git a/docs/src/Samples.js b/docs/src/Samples.js index 3fc3a163a5..b15e400a85 100644 --- a/docs/src/Samples.js +++ b/docs/src/Samples.js @@ -89,6 +89,7 @@ export default { TableResponsive: require('fs').readFileSync(__dirname + '/../examples/TableResponsive.js', 'utf8'), Input: require('fs').readFileSync(__dirname + '/../examples/Input.js', 'utf8'), InputTypes: require('fs').readFileSync(__dirname + '/../examples/InputTypes.js', 'utf8'), + StaticText: require('fs').readFileSync(__dirname + '/../examples/StaticText.js', 'utf8'), ButtonInput: require('fs').readFileSync(__dirname + '/../examples/ButtonInput.js', 'utf8'), InputAddons: require('fs').readFileSync(__dirname + '/../examples/InputAddons.js', 'utf8'), InputSizes: require('fs').readFileSync(__dirname + '/../examples/InputSizes.js', 'utf8'),