Skip to content

Commit 6a541ff

Browse files
author
Igor Scekic
committed
[added] buttonComponentClass prop for Pagination
wrap pagination buttons in a custom element usage: <Pagination ... buttonComponentClass={ButtonWrapper}/> where ButtonWrapper is your custom element default behavior (no prop given) remains unchanged note: Bootstrap css targets .pagination > li > a
1 parent ac7ada5 commit 6a541ff

File tree

4 files changed

+93
-12
lines changed

4 files changed

+93
-12
lines changed

src/Pagination.js

+22-8
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,8 @@ import React from 'react';
22
import classNames from 'classnames';
33
import BootstrapMixin from './BootstrapMixin';
44
import PaginationButton from './PaginationButton';
5+
import CustomPropTypes from './utils/CustomPropTypes';
6+
import SafeAnchor from './SafeAnchor';
57

68
const Pagination = React.createClass({
79
mixins: [BootstrapMixin],
@@ -15,7 +17,11 @@ const Pagination = React.createClass({
1517
last: React.PropTypes.bool,
1618
prev: React.PropTypes.bool,
1719
next: React.PropTypes.bool,
18-
onSelect: React.PropTypes.func
20+
onSelect: React.PropTypes.func,
21+
/**
22+
* You can use a custom element for the buttons
23+
*/
24+
buttonComponentClass: CustomPropTypes.elementType
1925
},
2026

2127
getDefaultProps() {
@@ -28,6 +34,7 @@ const Pagination = React.createClass({
2834
prev: false,
2935
next: false,
3036
ellipsis: true,
37+
buttonComponentClass: SafeAnchor,
3138
bsClass: 'pagination'
3239
};
3340
},
@@ -40,7 +47,8 @@ const Pagination = React.createClass({
4047
activePage,
4148
items,
4249
onSelect,
43-
ellipsis
50+
ellipsis,
51+
buttonComponentClass
4452
} = this.props;
4553

4654
if(maxButtons){
@@ -68,7 +76,8 @@ const Pagination = React.createClass({
6876
key={pagenumber}
6977
eventKey={pagenumber}
7078
active={pagenumber === activePage}
71-
onSelect={onSelect}>
79+
onSelect={onSelect}
80+
buttonComponentClass={buttonComponentClass}>
7281
{pagenumber}
7382
</PaginationButton>
7483
);
@@ -78,7 +87,8 @@ const Pagination = React.createClass({
7887
pageButtons.push(
7988
<PaginationButton
8089
key='ellipsis'
81-
disabled>
90+
disabled
91+
buttonComponentClass={buttonComponentClass}>
8292
<span aria-label='More'>...</span>
8393
</PaginationButton>
8494
);
@@ -97,7 +107,8 @@ const Pagination = React.createClass({
97107
key='prev'
98108
eventKey={this.props.activePage - 1}
99109
disabled={this.props.activePage === 1}
100-
onSelect={this.props.onSelect}>
110+
onSelect={this.props.onSelect}
111+
buttonComponentClass={this.props.buttonComponentClass}>
101112
<span aria-label='Previous'>&lsaquo;</span>
102113
</PaginationButton>
103114
);
@@ -113,7 +124,8 @@ const Pagination = React.createClass({
113124
key='next'
114125
eventKey={this.props.activePage + 1}
115126
disabled={this.props.activePage >= this.props.items}
116-
onSelect={this.props.onSelect}>
127+
onSelect={this.props.onSelect}
128+
buttonComponentClass={this.props.buttonComponentClass}>
117129
<span aria-label='Next'>&rsaquo;</span>
118130
</PaginationButton>
119131
);
@@ -129,7 +141,8 @@ const Pagination = React.createClass({
129141
key='first'
130142
eventKey={1}
131143
disabled={this.props.activePage === 1 }
132-
onSelect={this.props.onSelect}>
144+
onSelect={this.props.onSelect}
145+
buttonComponentClass={this.props.buttonComponentClass}>
133146
<span aria-label='First'>&laquo;</span>
134147
</PaginationButton>
135148
);
@@ -145,7 +158,8 @@ const Pagination = React.createClass({
145158
key='last'
146159
eventKey={this.props.items}
147160
disabled={this.props.activePage >= this.props.items}
148-
onSelect={this.props.onSelect}>
161+
onSelect={this.props.onSelect}
162+
buttonComponentClass={this.props.buttonComponentClass}>
149163
<span aria-label='Last'>&raquo;</span>
150164
</PaginationButton>
151165
);

src/PaginationButton.js

+9-3
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@ import React from 'react';
22
import classNames from 'classnames';
33
import BootstrapMixin from './BootstrapMixin';
44
import createSelectedEvent from './utils/createSelectedEvent';
5-
import SafeAnchor from './SafeAnchor';
5+
import CustomPropTypes from './utils/CustomPropTypes';
66

77
const PaginationButton = React.createClass({
88
mixins: [BootstrapMixin],
@@ -15,7 +15,11 @@ const PaginationButton = React.createClass({
1515
]),
1616
onSelect: React.PropTypes.func,
1717
disabled: React.PropTypes.bool,
18-
active: React.PropTypes.bool
18+
active: React.PropTypes.bool,
19+
/**
20+
* You can use a custom element for this component
21+
*/
22+
buttonComponentClass: CustomPropTypes.elementType
1923
},
2024

2125
getDefaultProps() {
@@ -44,9 +48,11 @@ const PaginationButton = React.createClass({
4448
...anchorProps
4549
} = this.props;
4650

51+
let ButtonComponentClass = this.props.buttonComponentClass;
52+
4753
return (
4854
<li className={classNames(className, classes)}>
49-
<SafeAnchor
55+
<ButtonComponentClass
5056
{...anchorProps}
5157
onClick={this.handleClick} />
5258
</li>

test/OverlayTriggerSpec.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -49,7 +49,7 @@ describe('OverlayTrigger', function() {
4949
const overlayTrigger = React.findDOMNode(instance);
5050
ReactTestUtils.Simulate.click(overlayTrigger);
5151

52-
expect(document.getElementsByClassName('test-overlay').length).to.equal(1)
52+
expect(document.getElementsByClassName('test-overlay').length).to.equal(1);
5353
});
5454

5555
it('Should pass transition callbacks to Transition', function (done) {

test/PaginationSpec.js

+61
Original file line numberDiff line numberDiff line change
@@ -114,4 +114,65 @@ describe('Pagination', function () {
114114
assert.include(React.findDOMNode(pageButtons[0]).className, 'disabled');
115115
assert.include(React.findDOMNode(pageButtons[1]).className, 'disabled');
116116
});
117+
118+
it('Should wrap buttons in SafeAnchor when no buttonComponentClass prop is supplied', function () {
119+
let instance = ReactTestUtils.renderIntoDocument(
120+
<Pagination
121+
maxButtons={2}
122+
activePage={1}
123+
items={2} />
124+
);
125+
let pageButtons = ReactTestUtils.scryRenderedDOMComponentsWithTag(instance, 'li');
126+
127+
let tagName = 'A';
128+
129+
assert.equal(React.findDOMNode(pageButtons[0]).children[0].tagName, tagName);
130+
assert.equal(React.findDOMNode(pageButtons[1]).children[0].tagName, tagName);
131+
132+
assert.equal(React.findDOMNode(pageButtons[0]).children[0].getAttribute('href'), '');
133+
assert.equal(React.findDOMNode(pageButtons[1]).children[0].getAttribute('href'), '');
134+
});
135+
136+
it('Should wrap each button in a buttonComponentClass when it is present', function () {
137+
class DummyElement extends React.Component {
138+
render() {
139+
return <div {...this.props}/>;
140+
}
141+
}
142+
143+
let instance = ReactTestUtils.renderIntoDocument(
144+
<Pagination
145+
maxButtons={2}
146+
activePage={1}
147+
items={2}
148+
buttonComponentClass={DummyElement} />
149+
);
150+
let pageButtons = ReactTestUtils.scryRenderedDOMComponentsWithTag(instance, 'li');
151+
152+
let tagName = 'DIV';
153+
154+
assert.equal(React.findDOMNode(pageButtons[0]).children[0].tagName, tagName);
155+
assert.equal(React.findDOMNode(pageButtons[1]).children[0].tagName, tagName);
156+
});
157+
158+
it('Should call onSelect with custom buttonComponentClass', function (done) {
159+
class DummyElement extends React.Component {
160+
render() {
161+
return <div {...this.props}/>;
162+
}
163+
}
164+
165+
function onSelect(event, selectedEvent) {
166+
assert.equal(selectedEvent.eventKey, 3);
167+
done();
168+
}
169+
170+
let instance = ReactTestUtils.renderIntoDocument(
171+
<Pagination items={5} onSelect={onSelect} buttonComponentClass={DummyElement}/>
172+
);
173+
174+
ReactTestUtils.Simulate.click(
175+
ReactTestUtils.scryRenderedDOMComponentsWithTag(instance, 'div')[2]
176+
);
177+
});
117178
});

0 commit comments

Comments
 (0)