Skip to content

Commit bfb3e6c

Browse files
committed
[added] standalone prop to Input, which will not render the form-group class
In some cases you may not want the `form-group` class applied to form input wrapper divs. To achieve this simply apply the `standalone` prop. Fixes react-bootstrap#205
1 parent 7d7e465 commit bfb3e6c

File tree

5 files changed

+159
-30
lines changed

5 files changed

+159
-30
lines changed

docs/src/ComponentsPage.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -548,7 +548,7 @@ const ComponentsPage = React.createClass({
548548
<h1 id='input' className='page-header'>Input</h1>
549549
<p>Renders an input in bootstrap wrappers. Supports label, help, text input add-ons, validation and use as wrapper.
550550
Use <code>getValue()</code> or <code>getChecked()</code> to get the current state.
551-
The helper method <code>getInputDOMNode()</code> returns the internal input element.</p>
551+
The helper method <code>getInputDOMNode()</code> returns the internal input element. If you don't want the <code>form-group</code> class applied apply the prop named <code>standalone</code>.</p>
552552
<ReactPlayground codeText={Samples.Input} />
553553
<h2 id='input-types'>Types</h2>
554554
<p>Supports <code>select</code>, <code>textarea</code>, <code>static</code> as well as standard HTML input types. <code>getValue()</code> returns an array for multiple select.</p>

src/FormGroup.js

+33
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
import React from 'react';
2+
import classSet from 'classnames';
3+
4+
class FormGroup extends React.Component {
5+
render() {
6+
let classes = {
7+
'form-group': !this.props.standalone,
8+
'has-feedback': this.props.hasFeedback,
9+
'has-success': this.props.bsStyle === 'success',
10+
'has-warning': this.props.bsStyle === 'warning',
11+
'has-error': this.props.bsStyle === 'error'
12+
};
13+
14+
return (
15+
<div className={classSet(classes, this.props.groupClassName)}>
16+
{this.props.children}
17+
</div>
18+
);
19+
}
20+
}
21+
22+
FormGroup.defaultProps = {
23+
standalone: false
24+
};
25+
26+
FormGroup.propTypes = {
27+
standalone: React.PropTypes.bool,
28+
hasFeedback: React.PropTypes.bool,
29+
bsStyle: React.PropTypes.oneOf(['success', 'warning', 'error']),
30+
groupClassName: React.PropTypes.string
31+
};
32+
33+
export default FormGroup;

src/Input.js

+15-29
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
import React from 'react';
22
import classSet from 'classnames';
33
import Button from './Button';
4+
import FormGroup from './FormGroup';
45

56
const Input = React.createClass({
67

@@ -215,38 +216,21 @@ const Input = React.createClass({
215216
) : children;
216217
},
217218

218-
renderFormGroup(children) {
219-
let classes = {
220-
'form-group': true,
221-
'has-feedback': this.props.hasFeedback,
222-
'has-success': this.props.bsStyle === 'success',
223-
'has-warning': this.props.bsStyle === 'warning',
224-
'has-error': this.props.bsStyle === 'error'
225-
};
226-
classes[this.props.groupClassName] = this.props.groupClassName;
227-
228-
return (
229-
<div className={classSet(classes)}>
230-
{children}
231-
</div>
232-
);
233-
},
234-
235219
render() {
220+
let children;
221+
236222
if (this.isCheckboxOrRadio()) {
237-
return this.renderFormGroup(
238-
this.renderWrapper([
239-
this.renderCheckboxandRadioWrapper(
240-
this.renderLabel(
241-
this.renderInput()
242-
)
243-
),
244-
this.renderHelp()
245-
])
246-
);
223+
children = this.renderWrapper([
224+
this.renderCheckboxandRadioWrapper(
225+
this.renderLabel(
226+
this.renderInput()
227+
)
228+
),
229+
this.renderHelp()
230+
]);
247231
}
248232
else {
249-
return this.renderFormGroup([
233+
children = [
250234
this.renderLabel(),
251235
this.renderWrapper([
252236
this.renderInputGroup(
@@ -255,8 +239,10 @@ const Input = React.createClass({
255239
this.renderIcon(),
256240
this.renderHelp()
257241
])
258-
]);
242+
];
259243
}
244+
245+
return <FormGroup {...this.props}>{children}</FormGroup>;
260246
}
261247
});
262248

test/FormGroupSpec.js

+86
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,86 @@
1+
import React from 'react';
2+
import ReactTestUtils from 'react/lib/ReactTestUtils';
3+
import FormGroup from '../src/FormGroup';
4+
5+
describe('FormGroup', function() {
6+
beforeEach(function() {
7+
sinon.spy(console, 'warn');
8+
});
9+
10+
afterEach(function() {
11+
if (typeof console.warn.restore === 'function') {
12+
console.warn.called.should.be.false;
13+
console.warn.restore();
14+
}
15+
});
16+
17+
it('renders children', function() {
18+
let instance = ReactTestUtils.renderIntoDocument(
19+
<FormGroup>
20+
<span className='child1' />
21+
<span className='child2' />
22+
</FormGroup>
23+
);
24+
25+
assert.ok(ReactTestUtils.findRenderedDOMComponentWithClass(instance, 'child1'));
26+
assert.ok(ReactTestUtils.findRenderedDOMComponentWithClass(instance, 'child2'));
27+
});
28+
29+
it('renders with form-group class', function() {
30+
let instance = ReactTestUtils.renderIntoDocument(
31+
<FormGroup>
32+
<span />
33+
</FormGroup>
34+
);
35+
36+
assert.ok(ReactTestUtils.findRenderedDOMComponentWithClass(instance, 'form-group'));
37+
});
38+
39+
it('renders no form-group class when standalone', function() {
40+
let instance = ReactTestUtils.renderIntoDocument(
41+
<FormGroup standalone>
42+
<span />
43+
</FormGroup>
44+
);
45+
46+
assert.equal(ReactTestUtils.scryRenderedDOMComponentsWithClass(instance, 'form-group').length, 0);
47+
});
48+
49+
[{
50+
className: 'has-feedback',
51+
props: { hasFeedback: true }
52+
}, {
53+
className: 'has-success',
54+
props: { bsStyle: 'success' }
55+
}, {
56+
className: 'has-warning',
57+
props: { bsStyle: 'warning' }
58+
}, {
59+
className: 'has-error',
60+
props: { bsStyle: 'error' }
61+
}, {
62+
className: 'custom-group',
63+
props: { groupClassName: 'custom-group' }
64+
}
65+
].forEach(function(testCase) {
66+
it(`does not render ${testCase.className} class`, function() {
67+
let instance = ReactTestUtils.renderIntoDocument(
68+
<FormGroup>
69+
<span />
70+
</FormGroup>
71+
);
72+
73+
assert.equal(ReactTestUtils.scryRenderedDOMComponentsWithClass(instance, testCase.className).length, 0);
74+
});
75+
76+
it(`renders with ${testCase.className} class`, function() {
77+
let instance = ReactTestUtils.renderIntoDocument(
78+
<FormGroup {...testCase.props}>
79+
<span />
80+
</FormGroup>
81+
);
82+
83+
assert.ok(ReactTestUtils.findRenderedDOMComponentWithClass(instance, testCase.className));
84+
});
85+
});
86+
});

test/InputSpec.js

+24
Original file line numberDiff line numberDiff line change
@@ -167,6 +167,30 @@ describe('Input', function () {
167167
assert.ok(ReactTestUtils.findRenderedDOMComponentWithClass(instance, 'help-block'));
168168
});
169169

170+
it('renders form-group class', function() {
171+
let instance = ReactTestUtils.renderIntoDocument(
172+
<Input />
173+
);
174+
175+
assert.ok(ReactTestUtils.findRenderedDOMComponentWithClass(instance, 'form-group'));
176+
});
177+
178+
it('renders no form-group class', function() {
179+
let instance = ReactTestUtils.renderIntoDocument(
180+
<Input standalone />
181+
);
182+
183+
assert.equal(ReactTestUtils.scryRenderedDOMComponentsWithClass(instance, 'form-group').length, 0);
184+
});
185+
186+
it('renders custom-form-group class', function() {
187+
let instance = ReactTestUtils.renderIntoDocument(
188+
<Input groupClassName='custom-form-class' />
189+
);
190+
191+
assert.ok(ReactTestUtils.findRenderedDOMComponentWithClass(instance, 'custom-form-class'));
192+
});
193+
170194
it('renders feedback icon', function () {
171195
let instance = ReactTestUtils.renderIntoDocument(
172196
<Input hasFeedback={true} bsStyle="error" />

0 commit comments

Comments
 (0)