mirror of
https://github.com/gosticks/react-bootstrap-table2.git
synced 2026-06-29 13:40:07 +00:00
Compare commits
33 Commits
react-boot
...
react-boot
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
096799c403 | ||
|
|
6dee718081 | ||
|
|
936a82954c | ||
|
|
ba24990994 | ||
|
|
e7ccd47817 | ||
|
|
a0af964d76 | ||
|
|
865be93ef7 | ||
|
|
65a596a0e9 | ||
|
|
024bba15fa | ||
|
|
f9217930e7 | ||
|
|
fc34ea12e6 | ||
|
|
b0f411e934 | ||
|
|
28a1077bad | ||
|
|
ca32eee28e | ||
|
|
7030b54cbd | ||
|
|
4d7378e3f1 | ||
|
|
88234fead0 | ||
|
|
dea780519f | ||
|
|
577973a147 | ||
|
|
c4f14e2b69 | ||
|
|
38bb2290dc | ||
|
|
feedcb9f4b | ||
|
|
8bfbc14bd9 | ||
|
|
ee4eb8f2c6 | ||
|
|
8fa6389c81 | ||
|
|
9a354444d0 | ||
|
|
2533a63430 | ||
|
|
81e0080aa6 | ||
|
|
094a0682f1 | ||
|
|
3f2c6201d9 | ||
|
|
280c423298 | ||
|
|
fc813e80b6 | ||
|
|
4bb2ae2ba0 |
21
LICENSE
Normal file
21
LICENSE
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
MIT License
|
||||||
|
|
||||||
|
Copyright (c) 2018 react-bootstrap-table2
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
@@ -145,7 +145,7 @@ Custom the events on row:
|
|||||||
|
|
||||||
```js
|
```js
|
||||||
const rowEvents = {
|
const rowEvents = {
|
||||||
onClick: (e) => {
|
onClick: (e, row, rowIndex) => {
|
||||||
....
|
....
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -12,6 +12,7 @@ Available properties in a column object:
|
|||||||
* [formatExtraData](#formatExtraData)
|
* [formatExtraData](#formatExtraData)
|
||||||
* [sort](#sort)
|
* [sort](#sort)
|
||||||
* [sortFunc](#sortFunc)
|
* [sortFunc](#sortFunc)
|
||||||
|
* [onSort](#onSort)
|
||||||
* [classes](#classes)
|
* [classes](#classes)
|
||||||
* [style](#style)
|
* [style](#style)
|
||||||
* [title](#title)
|
* [title](#title)
|
||||||
@@ -122,6 +123,19 @@ Enable the column sort via a `true` value given.
|
|||||||
```
|
```
|
||||||
> The possible value of `order` argument is **`asc`** and **`desc`**.
|
> The possible value of `order` argument is **`asc`** and **`desc`**.
|
||||||
|
|
||||||
|
## <a name='sortFunc'>column.onSort - [Function]</a>
|
||||||
|
`column.onSort` is an event listener for sort change event:
|
||||||
|
|
||||||
|
```js
|
||||||
|
{
|
||||||
|
// omit...
|
||||||
|
sort: true,
|
||||||
|
onSort: (field, order) => {
|
||||||
|
// ....
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
## <a name='classes'>column.classes - [String | Function]</a>
|
## <a name='classes'>column.classes - [String | Function]</a>
|
||||||
It's availabe to have custom class on table column:
|
It's availabe to have custom class on table column:
|
||||||
|
|
||||||
@@ -542,6 +556,7 @@ Or take a callback function
|
|||||||
Configure `column.filter` will able to setup a column level filter on the header column. Currently, `react-bootstrap-table2` support following filters:
|
Configure `column.filter` will able to setup a column level filter on the header column. Currently, `react-bootstrap-table2` support following filters:
|
||||||
|
|
||||||
* Text(`textFilter`)
|
* Text(`textFilter`)
|
||||||
|
* Select(`selectFilter`)
|
||||||
|
|
||||||
We have a quick example to show you how to use `column.filter`:
|
We have a quick example to show you how to use `column.filter`:
|
||||||
|
|
||||||
|
|||||||
@@ -60,6 +60,7 @@ Please see [Work with table sort](https://react-bootstrap-table.github.io/react-
|
|||||||
- [x] Default Sort
|
- [x] Default Sort
|
||||||
- [x] Remote mode
|
- [x] Remote mode
|
||||||
- [x] Custom the sorting header
|
- [x] Custom the sorting header
|
||||||
|
- [x] Sort event listener
|
||||||
- [ ] Custom the sort caret
|
- [ ] Custom the sort caret
|
||||||
- [ ] Sort management
|
- [ ] Sort management
|
||||||
- [ ] Multi sort
|
- [ ] Multi sort
|
||||||
@@ -83,8 +84,9 @@ Please see [available filter configuration](https://react-bootstrap-table.github
|
|||||||
- [x] Remote Filter
|
- [x] Remote Filter
|
||||||
- [ ] Custom Filter Component
|
- [ ] Custom Filter Component
|
||||||
- [ ] Regex Filter
|
- [ ] Regex Filter
|
||||||
- [ ] Select Filter
|
- [x] Select Filter
|
||||||
- [ ] Number Filter
|
- [x] Custom Select Filter
|
||||||
|
- [X] Number Filter
|
||||||
- [ ] Date Filter
|
- [ ] Date Filter
|
||||||
- [ ] Array Filter
|
- [ ] Array Filter
|
||||||
- [ ] Programmatically Filter
|
- [ ] Programmatically Filter
|
||||||
|
|||||||
@@ -24,6 +24,7 @@ const JS_SKIPS = `+(${TEST}|${LIB}|${DIST}|${NODE_MODULES})`;
|
|||||||
|
|
||||||
const STYLE_PKGS = [
|
const STYLE_PKGS = [
|
||||||
'react-bootstrap-table2',
|
'react-bootstrap-table2',
|
||||||
|
'react-bootstrap-table2-filter',
|
||||||
'react-bootstrap-table2-paginator'
|
'react-bootstrap-table2-paginator'
|
||||||
].reduce((pkg, curr) => `${curr}|${pkg}`, '');
|
].reduce((pkg, curr) => `${curr}|${pkg}`, '');
|
||||||
|
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ const filterSourcePath = path.join(__dirname, '../../react-bootstrap-table2-filt
|
|||||||
const editorSourcePath = path.join(__dirname, '../../react-bootstrap-table2-editor/index.js');
|
const editorSourcePath = path.join(__dirname, '../../react-bootstrap-table2-editor/index.js');
|
||||||
const sourceStylePath = path.join(__dirname, '../../react-bootstrap-table2/style');
|
const sourceStylePath = path.join(__dirname, '../../react-bootstrap-table2/style');
|
||||||
const paginationStylePath = path.join(__dirname, '../../react-bootstrap-table2-paginator/style');
|
const paginationStylePath = path.join(__dirname, '../../react-bootstrap-table2-paginator/style');
|
||||||
|
const filterStylePath = path.join(__dirname, '../../react-bootstrap-table2-filter/style');
|
||||||
const storyPath = path.join(__dirname, '../stories');
|
const storyPath = path.join(__dirname, '../stories');
|
||||||
const examplesPath = path.join(__dirname, '../examples');
|
const examplesPath = path.join(__dirname, '../examples');
|
||||||
const srcPath = path.join(__dirname, '../src');
|
const srcPath = path.join(__dirname, '../src');
|
||||||
@@ -40,7 +41,7 @@ const loaders = [{
|
|||||||
}, {
|
}, {
|
||||||
test: /\.scss$/,
|
test: /\.scss$/,
|
||||||
use: ['style-loader', 'css-loader', 'sass-loader'],
|
use: ['style-loader', 'css-loader', 'sass-loader'],
|
||||||
include: [storyPath, sourceStylePath, paginationStylePath],
|
include: [storyPath, sourceStylePath, paginationStylePath, filterStylePath],
|
||||||
}, {
|
}, {
|
||||||
test: /\.(jpg|png|woff|woff2|eot|ttf|svg)$/,
|
test: /\.(jpg|png|woff|woff2|eot|ttf|svg)$/,
|
||||||
loader: 'url-loader?limit=100000',
|
loader: 'url-loader?limit=100000',
|
||||||
|
|||||||
74
packages/react-bootstrap-table2-example/examples/column-filter/custom-number-filter.js
vendored
Normal file
74
packages/react-bootstrap-table2-example/examples/column-filter/custom-number-filter.js
vendored
Normal file
@@ -0,0 +1,74 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import BootstrapTable from 'react-bootstrap-table-next';
|
||||||
|
import filterFactory, { numberFilter, Comparator } from 'react-bootstrap-table2-filter';
|
||||||
|
import Code from 'components/common/code-block';
|
||||||
|
import { productsGenerator } from 'utils/common';
|
||||||
|
|
||||||
|
const products = productsGenerator(8);
|
||||||
|
|
||||||
|
const columns = [{
|
||||||
|
dataField: 'id',
|
||||||
|
text: 'Product ID'
|
||||||
|
}, {
|
||||||
|
dataField: 'name',
|
||||||
|
text: 'Product Name'
|
||||||
|
}, {
|
||||||
|
dataField: 'price',
|
||||||
|
text: 'Product Price',
|
||||||
|
filter: numberFilter({
|
||||||
|
options: [2100, 2103, 2105],
|
||||||
|
delay: 600,
|
||||||
|
placeholder: 'custom placeholder',
|
||||||
|
withoutEmptyComparatorOption: true,
|
||||||
|
comparators: [Comparator.EQ, Comparator.GT, Comparator.LT],
|
||||||
|
style: { display: 'inline-grid' },
|
||||||
|
className: 'custom-numberfilter-class',
|
||||||
|
comparatorStyle: { backgroundColor: 'antiquewhite' },
|
||||||
|
comparatorClassName: 'custom-comparator-class',
|
||||||
|
numberStyle: { backgroundColor: 'cadetblue', margin: '0px' },
|
||||||
|
numberClassName: 'custom-number-class'
|
||||||
|
})
|
||||||
|
}];
|
||||||
|
|
||||||
|
const sourceCode = `\
|
||||||
|
import BootstrapTable from 'react-bootstrap-table-next';
|
||||||
|
import filterFactory, { numberFilter, Comparator } from 'react-bootstrap-table2-filter';
|
||||||
|
|
||||||
|
const columns = [{
|
||||||
|
dataField: 'id',
|
||||||
|
text: 'Product ID'
|
||||||
|
}, {
|
||||||
|
dataField: 'name',
|
||||||
|
text: 'Product Name'
|
||||||
|
}, {
|
||||||
|
dataField: 'price',
|
||||||
|
text: 'Product Price',
|
||||||
|
filter: numberFilter({
|
||||||
|
options: [2100, 2103, 2105],
|
||||||
|
delay: 600,
|
||||||
|
placeholder: 'custom placeholder',
|
||||||
|
withoutEmptyComparatorOption: true,
|
||||||
|
comparators: [Comparator.EQ, Comparator.GT, Comparator.LT],
|
||||||
|
style: { display: 'inline-grid' },
|
||||||
|
className: 'custom-numberfilter-class',
|
||||||
|
comparatorStyle: { backgroundColor: 'antiquewhite' },
|
||||||
|
comparatorClassName: 'custom-comparator-class',
|
||||||
|
numberStyle: { backgroundColor: 'cadetblue', margin: '0px' },
|
||||||
|
numberClassName: 'custom-number-class'
|
||||||
|
})
|
||||||
|
}];
|
||||||
|
|
||||||
|
<BootstrapTable keyField='id' data={ products } columns={ columns } filter={ filterFactory() } />
|
||||||
|
`;
|
||||||
|
|
||||||
|
export default () => (
|
||||||
|
<div>
|
||||||
|
<BootstrapTable
|
||||||
|
keyField="id"
|
||||||
|
data={ products }
|
||||||
|
columns={ columns }
|
||||||
|
filter={ filterFactory() }
|
||||||
|
/>
|
||||||
|
<Code>{ sourceCode }</Code>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
80
packages/react-bootstrap-table2-example/examples/column-filter/custom-select-filter.js
vendored
Normal file
80
packages/react-bootstrap-table2-example/examples/column-filter/custom-select-filter.js
vendored
Normal file
@@ -0,0 +1,80 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import BootstrapTable from 'react-bootstrap-table-next';
|
||||||
|
import filterFactory, { selectFilter } from 'react-bootstrap-table2-filter';
|
||||||
|
import Code from 'components/common/code-block';
|
||||||
|
import { productsQualityGenerator } from 'utils/common';
|
||||||
|
|
||||||
|
const products = productsQualityGenerator(6);
|
||||||
|
|
||||||
|
const selectOptions = {
|
||||||
|
0: 'good',
|
||||||
|
1: 'Bad',
|
||||||
|
2: 'unknown'
|
||||||
|
};
|
||||||
|
|
||||||
|
const columns = [{
|
||||||
|
dataField: 'id',
|
||||||
|
text: 'Product ID'
|
||||||
|
}, {
|
||||||
|
dataField: 'name',
|
||||||
|
text: 'Product Name'
|
||||||
|
}, {
|
||||||
|
dataField: 'quality',
|
||||||
|
text: 'Product Quailty',
|
||||||
|
formatter: cell => selectOptions[cell],
|
||||||
|
filter: selectFilter({
|
||||||
|
options: selectOptions,
|
||||||
|
withoutEmptyOption: true,
|
||||||
|
style: {
|
||||||
|
backgroundColor: 'pink'
|
||||||
|
},
|
||||||
|
className: 'test-classname',
|
||||||
|
datamycustomattr: 'datamycustomattr'
|
||||||
|
})
|
||||||
|
}];
|
||||||
|
|
||||||
|
const sourceCode = `\
|
||||||
|
import BootstrapTable from 'react-bootstrap-table-next';
|
||||||
|
import filterFactory, { selectFilter } from 'react-bootstrap-table2-filter';
|
||||||
|
|
||||||
|
const selectOptions = {
|
||||||
|
0: 'good',
|
||||||
|
1: 'Bad',
|
||||||
|
2: 'unknown'
|
||||||
|
};
|
||||||
|
|
||||||
|
const columns = [{
|
||||||
|
dataField: 'id',
|
||||||
|
text: 'Product ID'
|
||||||
|
}, {
|
||||||
|
dataField: 'name',
|
||||||
|
text: 'Product Name'
|
||||||
|
}, {
|
||||||
|
dataField: 'quality',
|
||||||
|
text: 'Product Quailty',
|
||||||
|
formatter: cell => selectOptions[cell],
|
||||||
|
filter: selectFilter({
|
||||||
|
options: selectOptions,
|
||||||
|
withoutEmptyOption: true,
|
||||||
|
style: {
|
||||||
|
backgroundColor: 'pink'
|
||||||
|
},
|
||||||
|
className: 'test-classname',
|
||||||
|
datamycustomattr: 'datamycustomattr'
|
||||||
|
})
|
||||||
|
}];
|
||||||
|
|
||||||
|
<BootstrapTable keyField='id' data={ products } columns={ columns } filter={ filterFactory() } />
|
||||||
|
`;
|
||||||
|
|
||||||
|
export default () => (
|
||||||
|
<div>
|
||||||
|
<BootstrapTable
|
||||||
|
keyField="id"
|
||||||
|
data={ products }
|
||||||
|
columns={ columns }
|
||||||
|
filter={ filterFactory() }
|
||||||
|
/>
|
||||||
|
<Code>{ sourceCode }</Code>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
54
packages/react-bootstrap-table2-example/examples/column-filter/number-filter-default-value.js
vendored
Normal file
54
packages/react-bootstrap-table2-example/examples/column-filter/number-filter-default-value.js
vendored
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import BootstrapTable from 'react-bootstrap-table-next';
|
||||||
|
import filterFactory, { numberFilter, Comparator } from 'react-bootstrap-table2-filter';
|
||||||
|
import Code from 'components/common/code-block';
|
||||||
|
import { productsGenerator } from 'utils/common';
|
||||||
|
|
||||||
|
const products = productsGenerator(8);
|
||||||
|
|
||||||
|
const columns = [{
|
||||||
|
dataField: 'id',
|
||||||
|
text: 'Product ID'
|
||||||
|
}, {
|
||||||
|
dataField: 'name',
|
||||||
|
text: 'Product Name'
|
||||||
|
}, {
|
||||||
|
dataField: 'price',
|
||||||
|
text: 'Product Price',
|
||||||
|
filter: numberFilter({
|
||||||
|
defaultValue: { number: 2103, comparator: Comparator.GT }
|
||||||
|
})
|
||||||
|
}];
|
||||||
|
|
||||||
|
const sourceCode = `\
|
||||||
|
import BootstrapTable from 'react-bootstrap-table-next';
|
||||||
|
import filterFactory, { numberFilter, Comparator } from 'react-bootstrap-table2-filter';
|
||||||
|
|
||||||
|
const columns = [{
|
||||||
|
dataField: 'id',
|
||||||
|
text: 'Product ID'
|
||||||
|
}, {
|
||||||
|
dataField: 'name',
|
||||||
|
text: 'Product Name'
|
||||||
|
}, {
|
||||||
|
dataField: 'price',
|
||||||
|
text: 'Product Price',
|
||||||
|
filter: numberFilter({
|
||||||
|
defaultValue: { number: 2103, comparator: Comparator.GT }
|
||||||
|
})
|
||||||
|
}];
|
||||||
|
|
||||||
|
<BootstrapTable keyField='id' data={ products } columns={ columns } filter={ filterFactory() } />
|
||||||
|
`;
|
||||||
|
|
||||||
|
export default () => (
|
||||||
|
<div>
|
||||||
|
<BootstrapTable
|
||||||
|
keyField="id"
|
||||||
|
data={ products }
|
||||||
|
columns={ columns }
|
||||||
|
filter={ filterFactory() }
|
||||||
|
/>
|
||||||
|
<Code>{ sourceCode }</Code>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
50
packages/react-bootstrap-table2-example/examples/column-filter/number-filter.js
vendored
Normal file
50
packages/react-bootstrap-table2-example/examples/column-filter/number-filter.js
vendored
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import BootstrapTable from 'react-bootstrap-table-next';
|
||||||
|
import filterFactory, { numberFilter } from 'react-bootstrap-table2-filter';
|
||||||
|
import Code from 'components/common/code-block';
|
||||||
|
import { productsGenerator } from 'utils/common';
|
||||||
|
|
||||||
|
const products = productsGenerator(8);
|
||||||
|
|
||||||
|
const columns = [{
|
||||||
|
dataField: 'id',
|
||||||
|
text: 'Product ID'
|
||||||
|
}, {
|
||||||
|
dataField: 'name',
|
||||||
|
text: 'Product Name'
|
||||||
|
}, {
|
||||||
|
dataField: 'price',
|
||||||
|
text: 'Product Price',
|
||||||
|
filter: numberFilter()
|
||||||
|
}];
|
||||||
|
|
||||||
|
const sourceCode = `\
|
||||||
|
import BootstrapTable from 'react-bootstrap-table-next';
|
||||||
|
import filterFactory, { numberFilter } from 'react-bootstrap-table2-filter';
|
||||||
|
|
||||||
|
const columns = [{
|
||||||
|
dataField: 'id',
|
||||||
|
text: 'Product ID'
|
||||||
|
}, {
|
||||||
|
dataField: 'name',
|
||||||
|
text: 'Product Name'
|
||||||
|
}, {
|
||||||
|
dataField: 'price',
|
||||||
|
text: 'Product Price',
|
||||||
|
filter: numberFilter()
|
||||||
|
}];
|
||||||
|
|
||||||
|
<BootstrapTable keyField='id' data={ products } columns={ columns } filter={ filterFactory() } />
|
||||||
|
`;
|
||||||
|
|
||||||
|
export default () => (
|
||||||
|
<div>
|
||||||
|
<BootstrapTable
|
||||||
|
keyField="id"
|
||||||
|
data={ products }
|
||||||
|
columns={ columns }
|
||||||
|
filter={ filterFactory() }
|
||||||
|
/>
|
||||||
|
<Code>{ sourceCode }</Code>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
70
packages/react-bootstrap-table2-example/examples/column-filter/select-filter-default-value.js
vendored
Normal file
70
packages/react-bootstrap-table2-example/examples/column-filter/select-filter-default-value.js
vendored
Normal file
@@ -0,0 +1,70 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import BootstrapTable from 'react-bootstrap-table-next';
|
||||||
|
import filterFactory, { selectFilter } from 'react-bootstrap-table2-filter';
|
||||||
|
import Code from 'components/common/code-block';
|
||||||
|
import { productsQualityGenerator } from 'utils/common';
|
||||||
|
|
||||||
|
const products = productsQualityGenerator(6);
|
||||||
|
|
||||||
|
const selectOptions = {
|
||||||
|
0: 'good',
|
||||||
|
1: 'Bad',
|
||||||
|
2: 'unknown'
|
||||||
|
};
|
||||||
|
|
||||||
|
const columns = [{
|
||||||
|
dataField: 'id',
|
||||||
|
text: 'Product ID'
|
||||||
|
}, {
|
||||||
|
dataField: 'name',
|
||||||
|
text: 'Product Name'
|
||||||
|
}, {
|
||||||
|
dataField: 'quality',
|
||||||
|
text: 'Product Quailty',
|
||||||
|
formatter: cell => selectOptions[cell],
|
||||||
|
filter: selectFilter({
|
||||||
|
options: selectOptions,
|
||||||
|
defaultValue: 2
|
||||||
|
})
|
||||||
|
}];
|
||||||
|
|
||||||
|
const sourceCode = `\
|
||||||
|
import BootstrapTable from 'react-bootstrap-table-next';
|
||||||
|
import filterFactory, { selectFilter } from 'react-bootstrap-table2-filter';
|
||||||
|
|
||||||
|
const selectOptions = {
|
||||||
|
0: 'good',
|
||||||
|
1: 'Bad',
|
||||||
|
2: 'unknown'
|
||||||
|
};
|
||||||
|
|
||||||
|
const columns = [{
|
||||||
|
dataField: 'id',
|
||||||
|
text: 'Product ID'
|
||||||
|
}, {
|
||||||
|
dataField: 'name',
|
||||||
|
text: 'Product Name'
|
||||||
|
}, {
|
||||||
|
dataField: 'quality',
|
||||||
|
text: 'Product Quailty',
|
||||||
|
formatter: cell => selectOptions[cell],
|
||||||
|
filter: selectFilter({
|
||||||
|
options: selectOptions,
|
||||||
|
defaultValue: 2
|
||||||
|
})
|
||||||
|
}];
|
||||||
|
|
||||||
|
<BootstrapTable keyField='id' data={ products } columns={ columns } filter={ filterFactory() } />
|
||||||
|
`;
|
||||||
|
|
||||||
|
export default () => (
|
||||||
|
<div>
|
||||||
|
<BootstrapTable
|
||||||
|
keyField="id"
|
||||||
|
data={ products }
|
||||||
|
columns={ columns }
|
||||||
|
filter={ filterFactory() }
|
||||||
|
/>
|
||||||
|
<Code>{ sourceCode }</Code>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
69
packages/react-bootstrap-table2-example/examples/column-filter/select-filter-like-comparator.js
vendored
Normal file
69
packages/react-bootstrap-table2-example/examples/column-filter/select-filter-like-comparator.js
vendored
Normal file
@@ -0,0 +1,69 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import BootstrapTable from 'react-bootstrap-table-next';
|
||||||
|
import filterFactory, { selectFilter, Comparator } from 'react-bootstrap-table2-filter';
|
||||||
|
import Code from 'components/common/code-block';
|
||||||
|
import { productsGenerator } from 'utils/common';
|
||||||
|
|
||||||
|
const products = productsGenerator(6);
|
||||||
|
|
||||||
|
const selectOptions = {
|
||||||
|
'03': '03',
|
||||||
|
'04': '04',
|
||||||
|
'01': '01'
|
||||||
|
};
|
||||||
|
|
||||||
|
const columns = [{
|
||||||
|
dataField: 'id',
|
||||||
|
text: 'Product ID'
|
||||||
|
}, {
|
||||||
|
dataField: 'name',
|
||||||
|
text: 'Product Name'
|
||||||
|
}, {
|
||||||
|
dataField: 'price',
|
||||||
|
text: 'Product Price',
|
||||||
|
filter: selectFilter({
|
||||||
|
options: selectOptions,
|
||||||
|
comparator: Comparator.LIKE // default is Comparator.EQ
|
||||||
|
})
|
||||||
|
}];
|
||||||
|
|
||||||
|
const sourceCode = `\
|
||||||
|
import BootstrapTable from 'react-bootstrap-table-next';
|
||||||
|
import filterFactory, { selectFilter } from 'react-bootstrap-table2-filter';
|
||||||
|
|
||||||
|
const selectOptions = {
|
||||||
|
'03': '03',
|
||||||
|
'04': '04',
|
||||||
|
'01': '01'
|
||||||
|
};
|
||||||
|
|
||||||
|
const columns = [{
|
||||||
|
dataField: 'id',
|
||||||
|
text: 'Product ID'
|
||||||
|
}, {
|
||||||
|
dataField: 'name',
|
||||||
|
text: 'Product Name'
|
||||||
|
}, {
|
||||||
|
dataField: 'price',
|
||||||
|
text: 'Product Price',
|
||||||
|
filter: selectFilter({
|
||||||
|
options: selectOptions,
|
||||||
|
comparator: Comparator.LIKE // default is Comparator.EQ
|
||||||
|
})
|
||||||
|
}];
|
||||||
|
|
||||||
|
<BootstrapTable keyField='id' data={ products } columns={ columns } filter={ filterFactory() } />
|
||||||
|
`;
|
||||||
|
|
||||||
|
export default () => (
|
||||||
|
<div>
|
||||||
|
<h3>Select Filter with LIKE Comparator</h3>
|
||||||
|
<BootstrapTable
|
||||||
|
keyField="id"
|
||||||
|
data={ products }
|
||||||
|
columns={ columns }
|
||||||
|
filter={ filterFactory() }
|
||||||
|
/>
|
||||||
|
<Code>{ sourceCode }</Code>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
68
packages/react-bootstrap-table2-example/examples/column-filter/select-filter.js
vendored
Normal file
68
packages/react-bootstrap-table2-example/examples/column-filter/select-filter.js
vendored
Normal file
@@ -0,0 +1,68 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import BootstrapTable from 'react-bootstrap-table-next';
|
||||||
|
import filterFactory, { selectFilter } from 'react-bootstrap-table2-filter';
|
||||||
|
import Code from 'components/common/code-block';
|
||||||
|
import { productsQualityGenerator } from 'utils/common';
|
||||||
|
|
||||||
|
const products = productsQualityGenerator(6);
|
||||||
|
|
||||||
|
const selectOptions = {
|
||||||
|
0: 'good',
|
||||||
|
1: 'Bad',
|
||||||
|
2: 'unknown'
|
||||||
|
};
|
||||||
|
|
||||||
|
const columns = [{
|
||||||
|
dataField: 'id',
|
||||||
|
text: 'Product ID'
|
||||||
|
}, {
|
||||||
|
dataField: 'name',
|
||||||
|
text: 'Product Name'
|
||||||
|
}, {
|
||||||
|
dataField: 'quality',
|
||||||
|
text: 'Product Quailty',
|
||||||
|
formatter: cell => selectOptions[cell],
|
||||||
|
filter: selectFilter({
|
||||||
|
options: selectOptions
|
||||||
|
})
|
||||||
|
}];
|
||||||
|
|
||||||
|
const sourceCode = `\
|
||||||
|
import BootstrapTable from 'react-bootstrap-table-next';
|
||||||
|
import filterFactory, { selectFilter } from 'react-bootstrap-table2-filter';
|
||||||
|
|
||||||
|
const selectOptions = {
|
||||||
|
0: 'good',
|
||||||
|
1: 'Bad',
|
||||||
|
2: 'unknown'
|
||||||
|
};
|
||||||
|
|
||||||
|
const columns = [{
|
||||||
|
dataField: 'id',
|
||||||
|
text: 'Product ID'
|
||||||
|
}, {
|
||||||
|
dataField: 'name',
|
||||||
|
text: 'Product Name'
|
||||||
|
}, {
|
||||||
|
dataField: 'quality',
|
||||||
|
text: 'Product Quailty',
|
||||||
|
formatter: cell => selectOptions[cell],
|
||||||
|
filter: selectFilter({
|
||||||
|
options: selectOptions
|
||||||
|
})
|
||||||
|
}];
|
||||||
|
|
||||||
|
<BootstrapTable keyField='id' data={ products } columns={ columns } filter={ filterFactory() } />
|
||||||
|
`;
|
||||||
|
|
||||||
|
export default () => (
|
||||||
|
<div>
|
||||||
|
<BootstrapTable
|
||||||
|
keyField="id"
|
||||||
|
data={ products }
|
||||||
|
columns={ columns }
|
||||||
|
filter={ filterFactory() }
|
||||||
|
/>
|
||||||
|
<Code>{ sourceCode }</Code>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
51
packages/react-bootstrap-table2-example/examples/column-filter/text-filter-caseSensitive.js
vendored
Normal file
51
packages/react-bootstrap-table2-example/examples/column-filter/text-filter-caseSensitive.js
vendored
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import BootstrapTable from 'react-bootstrap-table-next';
|
||||||
|
import filterFactory, { textFilter } from 'react-bootstrap-table2-filter';
|
||||||
|
import Code from 'components/common/code-block';
|
||||||
|
import { productsGenerator } from 'utils/common';
|
||||||
|
|
||||||
|
const products = productsGenerator(8);
|
||||||
|
|
||||||
|
const columns = [{
|
||||||
|
dataField: 'id',
|
||||||
|
text: 'Product ID'
|
||||||
|
}, {
|
||||||
|
dataField: 'name',
|
||||||
|
text: 'Product Name',
|
||||||
|
filter: textFilter({ caseSensitive: true })
|
||||||
|
}, {
|
||||||
|
dataField: 'price',
|
||||||
|
text: 'Product Price'
|
||||||
|
}];
|
||||||
|
|
||||||
|
const sourceCode = `\
|
||||||
|
import BootstrapTable from 'react-bootstrap-table-next';
|
||||||
|
import filterFactory, { textFilter } from 'react-bootstrap-table2-filter';
|
||||||
|
|
||||||
|
const columns = [{
|
||||||
|
dataField: 'id',
|
||||||
|
text: 'Product ID'
|
||||||
|
}, {
|
||||||
|
dataField: 'name',
|
||||||
|
text: 'Product Name',
|
||||||
|
filter: textFilter({ caseSensitive: true })
|
||||||
|
}, {
|
||||||
|
dataField: 'price',
|
||||||
|
text: 'Product Price'
|
||||||
|
}];
|
||||||
|
|
||||||
|
<BootstrapTable keyField='id' data={ products } columns={ columns } filter={ filterFactory() } />
|
||||||
|
`;
|
||||||
|
|
||||||
|
export default () => (
|
||||||
|
<div>
|
||||||
|
<h3>Product Name is case sensitive</h3>
|
||||||
|
<BootstrapTable
|
||||||
|
keyField="id"
|
||||||
|
data={ products }
|
||||||
|
columns={ columns }
|
||||||
|
filter={ filterFactory() }
|
||||||
|
/>
|
||||||
|
<Code>{ sourceCode }</Code>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
@@ -20,8 +20,8 @@ const columns = [{
|
|||||||
}];
|
}];
|
||||||
|
|
||||||
const rowEvents = {
|
const rowEvents = {
|
||||||
onClick: (e) => {
|
onClick: (e, row, rowIndex) => {
|
||||||
alert('click on row');
|
alert(`clicked on row with index: ${rowIndex}`);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -40,8 +40,8 @@ const columns = [{
|
|||||||
}];
|
}];
|
||||||
|
|
||||||
const rowEvents = {
|
const rowEvents = {
|
||||||
onClick: (e) => {
|
onClick: (e, row, rowIndex) => {
|
||||||
alert('click on row');
|
alert(\`clicked on row with index: \${rowIndex}\`);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
58
packages/react-bootstrap-table2-example/examples/sort/sort-events.js
vendored
Normal file
58
packages/react-bootstrap-table2-example/examples/sort/sort-events.js
vendored
Normal file
@@ -0,0 +1,58 @@
|
|||||||
|
/* eslint no-console: 0 */
|
||||||
|
import React from 'react';
|
||||||
|
|
||||||
|
import BootstrapTable from 'react-bootstrap-table-next';
|
||||||
|
import Code from 'components/common/code-block';
|
||||||
|
import { productsGenerator } from 'utils/common';
|
||||||
|
|
||||||
|
const products = productsGenerator();
|
||||||
|
|
||||||
|
const columns = [{
|
||||||
|
dataField: 'id',
|
||||||
|
text: 'Product ID',
|
||||||
|
sort: true
|
||||||
|
}, {
|
||||||
|
dataField: 'name',
|
||||||
|
text: 'Product Name',
|
||||||
|
sort: true,
|
||||||
|
onSort: (field, order) => {
|
||||||
|
console.log(`Sort Field: ${field}, Sort Order: ${order}`);
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
dataField: 'price',
|
||||||
|
text: 'Product Price'
|
||||||
|
}];
|
||||||
|
|
||||||
|
const defaultSorted = [{
|
||||||
|
dataField: 'name',
|
||||||
|
order: 'desc'
|
||||||
|
}];
|
||||||
|
|
||||||
|
const sourceCode = `\
|
||||||
|
import BootstrapTable from 'react-bootstrap-table-next';
|
||||||
|
|
||||||
|
const columns = [{
|
||||||
|
dataField: 'id',
|
||||||
|
text: 'Product ID',
|
||||||
|
sort: true
|
||||||
|
}, {
|
||||||
|
dataField: 'name',
|
||||||
|
text: 'Product Name',
|
||||||
|
sort: true,
|
||||||
|
onSort: (field, order) => {
|
||||||
|
console.log(....);
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
dataField: 'price',
|
||||||
|
text: 'Product Price'
|
||||||
|
}];
|
||||||
|
|
||||||
|
<BootstrapTable keyField='id' data={ products } columns={ columns } />
|
||||||
|
`;
|
||||||
|
|
||||||
|
export default () => (
|
||||||
|
<div>
|
||||||
|
<BootstrapTable keyField="id" data={ products } columns={ columns } defaultSorted={ defaultSorted } />
|
||||||
|
<Code>{ sourceCode }</Code>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "react-bootstrap-table2-example",
|
"name": "react-bootstrap-table2-example",
|
||||||
"version": "0.1.1",
|
"version": "0.1.2",
|
||||||
"description": "",
|
"description": "",
|
||||||
"main": "index.js",
|
"main": "index.js",
|
||||||
"private": true,
|
"private": true,
|
||||||
|
|||||||
@@ -20,6 +20,13 @@ export const productsGenerator = (quantity = 5, callback) => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const productsQualityGenerator = (quantity = 5) =>
|
||||||
|
Array.from({ length: quantity }, (value, index) => ({
|
||||||
|
id: index,
|
||||||
|
name: `Item name ${index}`,
|
||||||
|
quality: index % 3
|
||||||
|
}));
|
||||||
|
|
||||||
export const jobsGenerator = (quantity = 5) =>
|
export const jobsGenerator = (quantity = 5) =>
|
||||||
Array.from({ length: quantity }, (value, index) => ({
|
Array.from({ length: quantity }, (value, index) => ({
|
||||||
id: index,
|
id: index,
|
||||||
|
|||||||
@@ -37,8 +37,16 @@ import HeaderColumnAttrsTable from 'examples/header-columns/column-attrs-table';
|
|||||||
import TextFilter from 'examples/column-filter/text-filter';
|
import TextFilter from 'examples/column-filter/text-filter';
|
||||||
import TextFilterWithDefaultValue from 'examples/column-filter/text-filter-default-value';
|
import TextFilterWithDefaultValue from 'examples/column-filter/text-filter-default-value';
|
||||||
import TextFilterComparator from 'examples/column-filter/text-filter-eq-comparator';
|
import TextFilterComparator from 'examples/column-filter/text-filter-eq-comparator';
|
||||||
|
import TextFilterCaseSensitive from 'examples/column-filter/text-filter-caseSensitive';
|
||||||
import CustomTextFilter from 'examples/column-filter/custom-text-filter';
|
import CustomTextFilter from 'examples/column-filter/custom-text-filter';
|
||||||
import CustomFilterValue from 'examples/column-filter/custom-filter-value';
|
import CustomFilterValue from 'examples/column-filter/custom-filter-value';
|
||||||
|
import SelectFilter from 'examples/column-filter/select-filter';
|
||||||
|
import SelectFilterWithDefaultValue from 'examples/column-filter/select-filter-default-value';
|
||||||
|
import SelectFilterComparator from 'examples/column-filter/select-filter-like-comparator';
|
||||||
|
import CustomSelectFilter from 'examples/column-filter/custom-select-filter';
|
||||||
|
import NumberFilter from 'examples/column-filter/number-filter';
|
||||||
|
import NumberFilterWithDefaultValue from 'examples/column-filter/number-filter-default-value';
|
||||||
|
import CustomNumberFilter from 'examples/column-filter/custom-number-filter';
|
||||||
|
|
||||||
// work on rows
|
// work on rows
|
||||||
import RowStyleTable from 'examples/rows/row-style';
|
import RowStyleTable from 'examples/rows/row-style';
|
||||||
@@ -48,6 +56,7 @@ import RowEventTable from 'examples/rows/row-event';
|
|||||||
// table sort
|
// table sort
|
||||||
import EnableSortTable from 'examples/sort/enable-sort-table';
|
import EnableSortTable from 'examples/sort/enable-sort-table';
|
||||||
import DefaultSortTable from 'examples/sort/default-sort-table';
|
import DefaultSortTable from 'examples/sort/default-sort-table';
|
||||||
|
import SortEvents from 'examples/sort/sort-events';
|
||||||
import CustomSortTable from 'examples/sort/custom-sort-table';
|
import CustomSortTable from 'examples/sort/custom-sort-table';
|
||||||
import HeaderSortingClassesTable from 'examples/sort/header-sorting-classes';
|
import HeaderSortingClassesTable from 'examples/sort/header-sorting-classes';
|
||||||
import HeaderSortingStyleTable from 'examples/sort/header-sorting-style';
|
import HeaderSortingStyleTable from 'examples/sort/header-sorting-style';
|
||||||
@@ -98,6 +107,7 @@ import 'stories/stylesheet/tomorrow.min.css';
|
|||||||
import 'stories/stylesheet/storybook.scss';
|
import 'stories/stylesheet/storybook.scss';
|
||||||
import '../../react-bootstrap-table2/style/react-bootstrap-table2.scss';
|
import '../../react-bootstrap-table2/style/react-bootstrap-table2.scss';
|
||||||
import '../../react-bootstrap-table2-paginator/style/react-bootstrap-table2-paginator.scss';
|
import '../../react-bootstrap-table2-paginator/style/react-bootstrap-table2-paginator.scss';
|
||||||
|
import '../../react-bootstrap-table2-filter/style/react-bootstrap-table2-filter.scss';
|
||||||
|
|
||||||
// import { action } from '@storybook/addon-actions';
|
// import { action } from '@storybook/addon-actions';
|
||||||
|
|
||||||
@@ -138,8 +148,16 @@ storiesOf('Column Filter', module)
|
|||||||
.add('Text Filter', () => <TextFilter />)
|
.add('Text Filter', () => <TextFilter />)
|
||||||
.add('Text Filter with Default Value', () => <TextFilterWithDefaultValue />)
|
.add('Text Filter with Default Value', () => <TextFilterWithDefaultValue />)
|
||||||
.add('Text Filter with Comparator', () => <TextFilterComparator />)
|
.add('Text Filter with Comparator', () => <TextFilterComparator />)
|
||||||
.add('Custom Text Filter', () => <CustomTextFilter />)
|
.add('Text Filter with Case Sensitive', () => <TextFilterCaseSensitive />)
|
||||||
// add another filter type example right here.
|
// add another filter type example right here.
|
||||||
|
.add('Select Filter', () => <SelectFilter />)
|
||||||
|
.add('Select Filter with Default Value', () => <SelectFilterWithDefaultValue />)
|
||||||
|
.add('Select Filter with Comparator', () => <SelectFilterComparator />)
|
||||||
|
.add('Number Filter', () => <NumberFilter />)
|
||||||
|
.add('Number Filter with Default Value', () => <NumberFilterWithDefaultValue />)
|
||||||
|
.add('Custom Text Filter', () => <CustomTextFilter />)
|
||||||
|
.add('Custom Select Filter', () => <CustomSelectFilter />)
|
||||||
|
.add('Custom Number Filter', () => <CustomNumberFilter />)
|
||||||
.add('Custom Filter Value', () => <CustomFilterValue />);
|
.add('Custom Filter Value', () => <CustomFilterValue />);
|
||||||
|
|
||||||
storiesOf('Work on Rows', module)
|
storiesOf('Work on Rows', module)
|
||||||
@@ -150,6 +168,7 @@ storiesOf('Work on Rows', module)
|
|||||||
storiesOf('Sort Table', module)
|
storiesOf('Sort Table', module)
|
||||||
.add('Enable Sort', () => <EnableSortTable />)
|
.add('Enable Sort', () => <EnableSortTable />)
|
||||||
.add('Default Sort Table', () => <DefaultSortTable />)
|
.add('Default Sort Table', () => <DefaultSortTable />)
|
||||||
|
.add('Sort Events', () => <SortEvents />)
|
||||||
.add('Custom Sort Fuction', () => <CustomSortTable />)
|
.add('Custom Sort Fuction', () => <CustomSortTable />)
|
||||||
.add('Custom Classes on Sorting Header Column', () => <HeaderSortingClassesTable />)
|
.add('Custom Classes on Sorting Header Column', () => <HeaderSortingClassesTable />)
|
||||||
.add('Custom Style on Sorting Header Column', () => <HeaderSortingStyleTable />);
|
.add('Custom Style on Sorting Header Column', () => <HeaderSortingStyleTable />);
|
||||||
|
|||||||
@@ -17,8 +17,20 @@ $ npm install react-bootstrap-table2-filter --save
|
|||||||
You can get all types of filters via import and these filters are a factory function to create a individual filter instance. Currently, we support following filters:
|
You can get all types of filters via import and these filters are a factory function to create a individual filter instance. Currently, we support following filters:
|
||||||
|
|
||||||
* TextFilter
|
* TextFilter
|
||||||
|
* SelectFilter
|
||||||
|
* NumberFilter
|
||||||
* **Coming soon!**
|
* **Coming soon!**
|
||||||
|
|
||||||
|
## Add CSS
|
||||||
|
|
||||||
|
```js
|
||||||
|
// es5
|
||||||
|
require('react-bootstrap-table2-filter/dist/react-bootstrap-table2-filter.min.css');
|
||||||
|
|
||||||
|
// es6
|
||||||
|
import 'react-bootstrap-table2-filter/dist/react-bootstrap-table2-filter.min.css';
|
||||||
|
```
|
||||||
|
|
||||||
## Text Filter
|
## Text Filter
|
||||||
Following is a quick demo for enable the column filter on **Product Price** column!!
|
Following is a quick demo for enable the column filter on **Product Price** column!!
|
||||||
|
|
||||||
@@ -47,9 +59,94 @@ const priceFilter = textFilter({
|
|||||||
className: 'my-custom-text-filter', // custom classname on input
|
className: 'my-custom-text-filter', // custom classname on input
|
||||||
defaultValue: 'test', // default filtering value
|
defaultValue: 'test', // default filtering value
|
||||||
comparator: Comparator.EQ, // default is Comparator.LIKE
|
comparator: Comparator.EQ, // default is Comparator.LIKE
|
||||||
|
caseSensitive: true, // default is false, and true will only work when comparator is LIKE
|
||||||
style: { ... }, // your custom styles on input
|
style: { ... }, // your custom styles on input
|
||||||
delay: 1000 // how long will trigger filtering after user typing, default is 500 ms
|
delay: 1000 // how long will trigger filtering after user typing, default is 500 ms
|
||||||
});
|
});
|
||||||
|
|
||||||
// omit...
|
// omit...
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## Select Filter
|
||||||
|
A quick example:
|
||||||
|
|
||||||
|
```js
|
||||||
|
import filterFactory, { selectFilter } from 'react-bootstrap-table2-filter';
|
||||||
|
|
||||||
|
// omit...
|
||||||
|
const selectOptions = {
|
||||||
|
0: 'good',
|
||||||
|
1: 'Bad',
|
||||||
|
2: 'unknown'
|
||||||
|
};
|
||||||
|
|
||||||
|
const columns = [
|
||||||
|
..., {
|
||||||
|
dataField: 'quality',
|
||||||
|
text: 'Product Quailty',
|
||||||
|
formatter: cell => selectOptions[cell],
|
||||||
|
filter: selectFilter({
|
||||||
|
options: selectOptions
|
||||||
|
})
|
||||||
|
}];
|
||||||
|
|
||||||
|
<BootstrapTable keyField='id' data={ products } columns={ columns } filter={ filterFactory() } />
|
||||||
|
```
|
||||||
|
|
||||||
|
Following is an example for custom select filter:
|
||||||
|
|
||||||
|
```js
|
||||||
|
import filterFactory, { selectFilter, Comparator } from 'react-bootstrap-table2-filter';
|
||||||
|
// omit...
|
||||||
|
|
||||||
|
const qualityFilter = selectFilter({
|
||||||
|
options: selectOptions,
|
||||||
|
placeholder: 'My Custom PlaceHolder', // custom the input placeholder
|
||||||
|
className: 'my-custom-text-filter', // custom classname on input
|
||||||
|
defaultValue: '2', // default filtering value
|
||||||
|
comparator: Comparator.LIKE, // default is Comparator.EQ
|
||||||
|
style: { ... }, // your custom styles on input
|
||||||
|
withoutEmptyOption: true // hide the default select option
|
||||||
|
});
|
||||||
|
|
||||||
|
// omit...
|
||||||
|
```
|
||||||
|
|
||||||
|
## Number Filter
|
||||||
|
|
||||||
|
```js
|
||||||
|
import filterFactory, { numberFilter } from 'react-bootstrap-table2-filter';
|
||||||
|
|
||||||
|
const columns = [..., {
|
||||||
|
dataField: 'price',
|
||||||
|
text: 'Product Price',
|
||||||
|
filter: numberFilter()
|
||||||
|
}];
|
||||||
|
|
||||||
|
<BootstrapTable keyField='id' data={ products } columns={ columns } filter={ filterFactory() } />
|
||||||
|
```
|
||||||
|
|
||||||
|
Numner filter is same as other filter, you can custom the number filter via `numberFilter` factory function:
|
||||||
|
|
||||||
|
```js
|
||||||
|
import filterFactory, { selectFilter, Comparator } from 'react-bootstrap-table2-filter';
|
||||||
|
// omit...
|
||||||
|
|
||||||
|
const numberFilter = numberFilter({
|
||||||
|
options: [2100, 2103, 2105], // if options defined, will render number select instead of number input
|
||||||
|
delay: 600, // how long will trigger filtering after user typing, default is 500 ms
|
||||||
|
placeholder: 'custom placeholder', // placeholder for number input
|
||||||
|
withoutEmptyComparatorOption: true, // dont render empty option for comparator
|
||||||
|
withoutEmptyNumberOption: true, // dont render empty option for numner select if it is defined
|
||||||
|
comparators: [Comparator.EQ, Comparator.GT, Comparator.LT], // Custom the comparators
|
||||||
|
style: { display: 'inline-grid' }, // custom the style on number filter
|
||||||
|
className: 'custom-numberfilter-class', // custom the class on number filter
|
||||||
|
comparatorStyle: { backgroundColor: 'antiquewhite' }, // custom the style on comparator select
|
||||||
|
comparatorClassName: 'custom-comparator-class', // custom the class on comparator select
|
||||||
|
numberStyle: { backgroundColor: 'cadetblue', margin: '0px' }, // custom the style on number input/select
|
||||||
|
numberClassName: 'custom-number-class', // custom the class on ber input/select
|
||||||
|
defaultValue: { number: 2103, comparator: Comparator.GT } // default value
|
||||||
|
})
|
||||||
|
|
||||||
|
// omit...
|
||||||
|
```
|
||||||
12
packages/react-bootstrap-table2-filter/index.js
vendored
12
packages/react-bootstrap-table2-filter/index.js
vendored
@@ -1,4 +1,6 @@
|
|||||||
import TextFilter from './src/components/text';
|
import TextFilter from './src/components/text';
|
||||||
|
import SelectFilter from './src/components/select';
|
||||||
|
import NumberFilter from './src/components/number';
|
||||||
import wrapperFactory from './src/wrapper';
|
import wrapperFactory from './src/wrapper';
|
||||||
import * as Comparison from './src/comparison';
|
import * as Comparison from './src/comparison';
|
||||||
|
|
||||||
@@ -13,3 +15,13 @@ export const textFilter = (props = {}) => ({
|
|||||||
Filter: TextFilter,
|
Filter: TextFilter,
|
||||||
props
|
props
|
||||||
});
|
});
|
||||||
|
|
||||||
|
export const selectFilter = (props = {}) => ({
|
||||||
|
Filter: SelectFilter,
|
||||||
|
props
|
||||||
|
});
|
||||||
|
|
||||||
|
export const numberFilter = (props = {}) => ({
|
||||||
|
Filter: NumberFilter,
|
||||||
|
props
|
||||||
|
});
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "react-bootstrap-table2-filter",
|
"name": "react-bootstrap-table2-filter",
|
||||||
"version": "0.1.1",
|
"version": "0.1.3",
|
||||||
"description": "it's a column filter addon for react-bootstrap-table2",
|
"description": "it's a column filter addon for react-bootstrap-table2",
|
||||||
"main": "./lib/index.js",
|
"main": "./lib/index.js",
|
||||||
"repository": {
|
"repository": {
|
||||||
|
|||||||
@@ -1,2 +1,7 @@
|
|||||||
export const LIKE = 'LIKE';
|
export const LIKE = 'LIKE';
|
||||||
export const EQ = '=';
|
export const EQ = '=';
|
||||||
|
export const NE = '!=';
|
||||||
|
export const GT = '>';
|
||||||
|
export const GE = '>=';
|
||||||
|
export const LT = '<';
|
||||||
|
export const LE = '<=';
|
||||||
|
|||||||
249
packages/react-bootstrap-table2-filter/src/components/number.js
vendored
Normal file
249
packages/react-bootstrap-table2-filter/src/components/number.js
vendored
Normal file
@@ -0,0 +1,249 @@
|
|||||||
|
/* eslint no-return-assign: 0 */
|
||||||
|
|
||||||
|
import React, { Component } from 'react';
|
||||||
|
import PropTypes from 'prop-types';
|
||||||
|
import * as Comparator from '../comparison';
|
||||||
|
import { FILTER_TYPE, FILTER_DELAY } from '../const';
|
||||||
|
|
||||||
|
const legalComparators = [
|
||||||
|
Comparator.EQ,
|
||||||
|
Comparator.NE,
|
||||||
|
Comparator.GT,
|
||||||
|
Comparator.GE,
|
||||||
|
Comparator.LT,
|
||||||
|
Comparator.LE
|
||||||
|
];
|
||||||
|
|
||||||
|
class NumberFilter extends Component {
|
||||||
|
constructor(props) {
|
||||||
|
super(props);
|
||||||
|
this.comparators = props.comparators || legalComparators;
|
||||||
|
this.timeout = null;
|
||||||
|
let isSelected = props.defaultValue !== undefined && props.defaultValue.number !== undefined;
|
||||||
|
if (props.options && isSelected) {
|
||||||
|
isSelected = props.options.indexOf(props.defaultValue.number) > -1;
|
||||||
|
}
|
||||||
|
this.state = { isSelected };
|
||||||
|
this.onChangeNumber = this.onChangeNumber.bind(this);
|
||||||
|
this.onChangeNumberSet = this.onChangeNumberSet.bind(this);
|
||||||
|
this.onChangeComparator = this.onChangeComparator.bind(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
componentDidMount() {
|
||||||
|
const { column, onFilter } = this.props;
|
||||||
|
const comparator = this.numberFilterComparator.value;
|
||||||
|
const number = this.numberFilter.value;
|
||||||
|
if (comparator && number) {
|
||||||
|
onFilter(column, { number, comparator }, FILTER_TYPE.NUMBER);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
componentWillUnmount() {
|
||||||
|
clearTimeout(this.timeout);
|
||||||
|
}
|
||||||
|
|
||||||
|
onChangeNumber(e) {
|
||||||
|
const { delay, column, onFilter } = this.props;
|
||||||
|
const comparator = this.numberFilterComparator.value;
|
||||||
|
if (comparator === '') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (this.timeout) {
|
||||||
|
clearTimeout(this.timeout);
|
||||||
|
}
|
||||||
|
const filterValue = e.target.value;
|
||||||
|
this.timeout = setTimeout(() => {
|
||||||
|
onFilter(column, { number: filterValue, comparator }, FILTER_TYPE.NUMBER);
|
||||||
|
}, delay);
|
||||||
|
}
|
||||||
|
|
||||||
|
onChangeNumberSet(e) {
|
||||||
|
const { column, onFilter } = this.props;
|
||||||
|
const comparator = this.numberFilterComparator.value;
|
||||||
|
const { value } = e.target;
|
||||||
|
this.setState(() => ({ isSelected: (value !== '') }));
|
||||||
|
// if (comparator === '') {
|
||||||
|
// return;
|
||||||
|
// }
|
||||||
|
onFilter(column, { number: value, comparator }, FILTER_TYPE.NUMBER);
|
||||||
|
}
|
||||||
|
|
||||||
|
onChangeComparator(e) {
|
||||||
|
const { column, onFilter } = this.props;
|
||||||
|
const value = this.numberFilter.value;
|
||||||
|
const comparator = e.target.value;
|
||||||
|
// if (value === '') {
|
||||||
|
// return;
|
||||||
|
// }
|
||||||
|
onFilter(column, { number: value, comparator }, FILTER_TYPE.NUMBER);
|
||||||
|
}
|
||||||
|
|
||||||
|
getComparatorOptions() {
|
||||||
|
const optionTags = [];
|
||||||
|
const { withoutEmptyComparatorOption } = this.props;
|
||||||
|
if (!withoutEmptyComparatorOption) {
|
||||||
|
optionTags.push(<option key="-1" />);
|
||||||
|
}
|
||||||
|
for (let i = 0; i < this.comparators.length; i += 1) {
|
||||||
|
optionTags.push(
|
||||||
|
<option key={ i } value={ this.comparators[i] }>
|
||||||
|
{ this.comparators[i] }
|
||||||
|
</option>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return optionTags;
|
||||||
|
}
|
||||||
|
|
||||||
|
getNumberOptions() {
|
||||||
|
const optionTags = [];
|
||||||
|
const { options, column, withoutEmptyNumberOption } = this.props;
|
||||||
|
if (!withoutEmptyNumberOption) {
|
||||||
|
optionTags.push(
|
||||||
|
<option key="-1" value="">
|
||||||
|
{ this.props.placeholder || `Select ${column.text}...` }
|
||||||
|
</option>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
for (let i = 0; i < options.length; i += 1) {
|
||||||
|
optionTags.push(<option key={ i } value={ options[i] }>{ options[i] }</option>);
|
||||||
|
}
|
||||||
|
return optionTags;
|
||||||
|
}
|
||||||
|
|
||||||
|
applyFilter(filterObj) {
|
||||||
|
const { column, onFilter } = this.props;
|
||||||
|
const { number, comparator } = filterObj;
|
||||||
|
this.setState(() => ({ isSelected: (number !== '') }));
|
||||||
|
this.numberFilterComparator.value = comparator;
|
||||||
|
this.numberFilter.value = number;
|
||||||
|
onFilter(column, { number, comparator }, FILTER_TYPE.NUMBER);
|
||||||
|
}
|
||||||
|
|
||||||
|
cleanFiltered() {
|
||||||
|
const { column, onFilter, defaultValue } = this.props;
|
||||||
|
const value = defaultValue ? defaultValue.number : '';
|
||||||
|
const comparator = defaultValue ? defaultValue.comparator : '';
|
||||||
|
this.setState(() => ({ isSelected: (value !== '') }));
|
||||||
|
this.numberFilterComparator.value = comparator;
|
||||||
|
this.numberFilter.value = value;
|
||||||
|
onFilter(column, { number: value, comparator }, FILTER_TYPE.NUMBER);
|
||||||
|
}
|
||||||
|
|
||||||
|
render() {
|
||||||
|
const { isSelected } = this.state;
|
||||||
|
const {
|
||||||
|
defaultValue,
|
||||||
|
column,
|
||||||
|
options,
|
||||||
|
style,
|
||||||
|
className,
|
||||||
|
numberStyle,
|
||||||
|
numberClassName,
|
||||||
|
comparatorStyle,
|
||||||
|
comparatorClassName,
|
||||||
|
placeholder
|
||||||
|
} = this.props;
|
||||||
|
const selectClass = `
|
||||||
|
select-filter
|
||||||
|
number-filter-input
|
||||||
|
form-control
|
||||||
|
${numberClassName}
|
||||||
|
${!isSelected ? 'placeholder-selected' : ''}
|
||||||
|
`;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={ `filter number-filter ${className}` } style={ style }>
|
||||||
|
<select
|
||||||
|
ref={ n => this.numberFilterComparator = n }
|
||||||
|
style={ comparatorStyle }
|
||||||
|
className={ `number-filter-comparator form-control ${comparatorClassName}` }
|
||||||
|
onChange={ this.onChangeComparator }
|
||||||
|
defaultValue={ defaultValue ? defaultValue.comparator : '' }
|
||||||
|
>
|
||||||
|
{ this.getComparatorOptions() }
|
||||||
|
</select>
|
||||||
|
{
|
||||||
|
options ?
|
||||||
|
<select
|
||||||
|
ref={ n => this.numberFilter = n }
|
||||||
|
style={ numberStyle }
|
||||||
|
className={ selectClass }
|
||||||
|
onChange={ this.onChangeNumberSet }
|
||||||
|
defaultValue={ defaultValue ? defaultValue.number : '' }
|
||||||
|
>
|
||||||
|
{ this.getNumberOptions() }
|
||||||
|
</select> :
|
||||||
|
<input
|
||||||
|
ref={ n => this.numberFilter = n }
|
||||||
|
type="number"
|
||||||
|
style={ numberStyle }
|
||||||
|
className={ `number-filter-input form-control ${numberClassName}` }
|
||||||
|
placeholder={ placeholder || `Enter ${column.text}...` }
|
||||||
|
onChange={ this.onChangeNumber }
|
||||||
|
defaultValue={ defaultValue ? defaultValue.number : '' }
|
||||||
|
/>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
NumberFilter.propTypes = {
|
||||||
|
onFilter: PropTypes.func.isRequired,
|
||||||
|
column: PropTypes.object.isRequired,
|
||||||
|
options: PropTypes.arrayOf(PropTypes.number),
|
||||||
|
defaultValue: PropTypes.shape({
|
||||||
|
number: PropTypes.oneOfType([PropTypes.string, PropTypes.number]),
|
||||||
|
comparator: PropTypes.oneOf([...legalComparators, ''])
|
||||||
|
}),
|
||||||
|
delay: PropTypes.number,
|
||||||
|
/* eslint consistent-return: 0 */
|
||||||
|
comparators: (props, propName) => {
|
||||||
|
if (!props[propName]) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
for (let i = 0; i < props[propName].length; i += 1) {
|
||||||
|
let comparatorIsValid = false;
|
||||||
|
for (let j = 0; j < legalComparators.length; j += 1) {
|
||||||
|
if (legalComparators[j] === props[propName][i] || props[propName][i] === '') {
|
||||||
|
comparatorIsValid = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!comparatorIsValid) {
|
||||||
|
return new Error(`Number comparator provided is not supported.
|
||||||
|
Use only ${legalComparators}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
placeholder: PropTypes.string,
|
||||||
|
withoutEmptyComparatorOption: PropTypes.bool,
|
||||||
|
withoutEmptyNumberOption: PropTypes.bool,
|
||||||
|
style: PropTypes.object,
|
||||||
|
className: PropTypes.string,
|
||||||
|
comparatorStyle: PropTypes.object,
|
||||||
|
comparatorClassName: PropTypes.string,
|
||||||
|
numberStyle: PropTypes.object,
|
||||||
|
numberClassName: PropTypes.string
|
||||||
|
};
|
||||||
|
|
||||||
|
NumberFilter.defaultProps = {
|
||||||
|
delay: FILTER_DELAY,
|
||||||
|
options: undefined,
|
||||||
|
defaultValue: {
|
||||||
|
number: undefined,
|
||||||
|
comparator: ''
|
||||||
|
},
|
||||||
|
withoutEmptyComparatorOption: false,
|
||||||
|
withoutEmptyNumberOption: false,
|
||||||
|
comparators: legalComparators,
|
||||||
|
placeholder: undefined,
|
||||||
|
style: undefined,
|
||||||
|
className: '',
|
||||||
|
comparatorStyle: undefined,
|
||||||
|
comparatorClassName: '',
|
||||||
|
numberStyle: undefined,
|
||||||
|
numberClassName: ''
|
||||||
|
};
|
||||||
|
|
||||||
|
export default NumberFilter;
|
||||||
135
packages/react-bootstrap-table2-filter/src/components/select.js
vendored
Normal file
135
packages/react-bootstrap-table2-filter/src/components/select.js
vendored
Normal file
@@ -0,0 +1,135 @@
|
|||||||
|
/* eslint react/require-default-props: 0 */
|
||||||
|
/* eslint no-return-assign: 0 */
|
||||||
|
/* eslint react/no-unused-prop-types: 0 */
|
||||||
|
import React, { Component } from 'react';
|
||||||
|
import PropTypes from 'prop-types';
|
||||||
|
import { LIKE, EQ } from '../comparison';
|
||||||
|
import { FILTER_TYPE } from '../const';
|
||||||
|
|
||||||
|
function optionsEquals(currOpts, prevOpts) {
|
||||||
|
const keys = Object.keys(currOpts);
|
||||||
|
for (let i = 0; i < keys.length; i += 1) {
|
||||||
|
if (currOpts[keys[i]] !== prevOpts[keys[i]]) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return Object.keys(currOpts).length === Object.keys(prevOpts).length;
|
||||||
|
}
|
||||||
|
|
||||||
|
class SelectFilter extends Component {
|
||||||
|
constructor(props) {
|
||||||
|
super(props);
|
||||||
|
this.filter = this.filter.bind(this);
|
||||||
|
const isSelected = props.options[props.defaultValue] !== undefined;
|
||||||
|
this.state = { isSelected };
|
||||||
|
}
|
||||||
|
|
||||||
|
componentDidMount() {
|
||||||
|
const value = this.selectInput.value;
|
||||||
|
if (value && value !== '') {
|
||||||
|
this.props.onFilter(this.props.column, value, FILTER_TYPE.SELECT);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
componentDidUpdate(prevProps) {
|
||||||
|
let needFilter = false;
|
||||||
|
if (this.props.defaultValue !== prevProps.defaultValue) {
|
||||||
|
needFilter = true;
|
||||||
|
} else if (!optionsEquals(this.props.options, prevProps.options)) {
|
||||||
|
needFilter = true;
|
||||||
|
}
|
||||||
|
if (needFilter) {
|
||||||
|
const value = this.selectInput.value;
|
||||||
|
if (value) {
|
||||||
|
this.props.onFilter(this.props.column, value, FILTER_TYPE.SELECT);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
getOptions() {
|
||||||
|
const optionTags = [];
|
||||||
|
const { options, placeholder, column, withoutEmptyOption } = this.props;
|
||||||
|
if (!withoutEmptyOption) {
|
||||||
|
optionTags.push((
|
||||||
|
<option key="-1" value="">{ placeholder || `Select ${column.text}...` }</option>
|
||||||
|
));
|
||||||
|
}
|
||||||
|
Object.keys(options).forEach(key =>
|
||||||
|
optionTags.push(<option key={ key } value={ key }>{ options[key] }</option>)
|
||||||
|
);
|
||||||
|
return optionTags;
|
||||||
|
}
|
||||||
|
|
||||||
|
cleanFiltered() {
|
||||||
|
const value = (this.props.defaultValue !== undefined) ? this.props.defaultValue : '';
|
||||||
|
this.setState(() => ({ isSelected: value !== '' }));
|
||||||
|
this.selectInput.value = value;
|
||||||
|
this.props.onFilter(this.props.column, value, FILTER_TYPE.SELECT);
|
||||||
|
}
|
||||||
|
|
||||||
|
applyFilter(value) {
|
||||||
|
this.selectInput.value = value;
|
||||||
|
this.setState(() => ({ isSelected: value !== '' }));
|
||||||
|
this.props.onFilter(this.props.column, value, FILTER_TYPE.SELECT);
|
||||||
|
}
|
||||||
|
|
||||||
|
filter(e) {
|
||||||
|
const { value } = e.target;
|
||||||
|
this.setState(() => ({ isSelected: value !== '' }));
|
||||||
|
this.props.onFilter(this.props.column, value, FILTER_TYPE.SELECT);
|
||||||
|
}
|
||||||
|
|
||||||
|
render() {
|
||||||
|
const {
|
||||||
|
style,
|
||||||
|
className,
|
||||||
|
defaultValue,
|
||||||
|
onFilter,
|
||||||
|
column,
|
||||||
|
options,
|
||||||
|
comparator,
|
||||||
|
withoutEmptyOption,
|
||||||
|
caseSensitive,
|
||||||
|
...rest
|
||||||
|
} = this.props;
|
||||||
|
|
||||||
|
const selectClass =
|
||||||
|
`filter select-filter form-control ${className} ${this.state.isSelected ? '' : 'placeholder-selected'}`;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<select
|
||||||
|
{ ...rest }
|
||||||
|
ref={ n => this.selectInput = n }
|
||||||
|
style={ style }
|
||||||
|
className={ selectClass }
|
||||||
|
onChange={ this.filter }
|
||||||
|
defaultValue={ defaultValue !== undefined ? defaultValue : '' }
|
||||||
|
>
|
||||||
|
{ this.getOptions() }
|
||||||
|
</select>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
SelectFilter.propTypes = {
|
||||||
|
onFilter: PropTypes.func.isRequired,
|
||||||
|
column: PropTypes.object.isRequired,
|
||||||
|
options: PropTypes.object.isRequired,
|
||||||
|
comparator: PropTypes.oneOf([LIKE, EQ]),
|
||||||
|
placeholder: PropTypes.string,
|
||||||
|
style: PropTypes.object,
|
||||||
|
className: PropTypes.string,
|
||||||
|
withoutEmptyOption: PropTypes.bool,
|
||||||
|
defaultValue: PropTypes.any,
|
||||||
|
caseSensitive: PropTypes.bool
|
||||||
|
};
|
||||||
|
|
||||||
|
SelectFilter.defaultProps = {
|
||||||
|
defaultValue: '',
|
||||||
|
className: '',
|
||||||
|
withoutEmptyOption: false,
|
||||||
|
comparator: EQ,
|
||||||
|
caseSensitive: true
|
||||||
|
};
|
||||||
|
|
||||||
|
export default SelectFilter;
|
||||||
@@ -69,7 +69,16 @@ class TextFilter extends Component {
|
|||||||
}
|
}
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
const { placeholder, column: { text }, style, className, onFilter, ...rest } = this.props;
|
const {
|
||||||
|
placeholder,
|
||||||
|
column: { text },
|
||||||
|
style,
|
||||||
|
className,
|
||||||
|
onFilter,
|
||||||
|
caseSensitive,
|
||||||
|
defaultValue,
|
||||||
|
...rest
|
||||||
|
} = this.props;
|
||||||
// stopPropagation for onClick event is try to prevent sort was triggered.
|
// stopPropagation for onClick event is try to prevent sort was triggered.
|
||||||
return (
|
return (
|
||||||
<input
|
<input
|
||||||
@@ -95,12 +104,14 @@ TextFilter.propTypes = {
|
|||||||
delay: PropTypes.number,
|
delay: PropTypes.number,
|
||||||
placeholder: PropTypes.string,
|
placeholder: PropTypes.string,
|
||||||
style: PropTypes.object,
|
style: PropTypes.object,
|
||||||
className: PropTypes.string
|
className: PropTypes.string,
|
||||||
|
caseSensitive: PropTypes.bool
|
||||||
};
|
};
|
||||||
|
|
||||||
TextFilter.defaultProps = {
|
TextFilter.defaultProps = {
|
||||||
delay: FILTER_DELAY,
|
delay: FILTER_DELAY,
|
||||||
defaultValue: ''
|
defaultValue: '',
|
||||||
|
caseSensitive: false
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
export const FILTER_TYPE = {
|
export const FILTER_TYPE = {
|
||||||
TEXT: 'TEXT'
|
TEXT: 'TEXT',
|
||||||
|
SELECT: 'SELECT',
|
||||||
|
NUMBER: 'NUMBER'
|
||||||
};
|
};
|
||||||
|
|
||||||
export const FILTER_DELAY = 500;
|
export const FILTER_DELAY = 500;
|
||||||
|
|||||||
@@ -1,10 +1,12 @@
|
|||||||
|
/* eslint eqeqeq: 0 */
|
||||||
|
/* eslint no-console: 0 */
|
||||||
import { FILTER_TYPE } from './const';
|
import { FILTER_TYPE } from './const';
|
||||||
import { LIKE, EQ } from './comparison';
|
import { LIKE, EQ, NE, GT, GE, LT, LE } from './comparison';
|
||||||
|
|
||||||
export const filterByText = _ => (
|
export const filterByText = _ => (
|
||||||
data,
|
data,
|
||||||
dataField,
|
dataField,
|
||||||
{ filterVal, comparator = LIKE },
|
{ filterVal = '', comparator = LIKE, caseSensitive },
|
||||||
customFilterValue
|
customFilterValue
|
||||||
) =>
|
) =>
|
||||||
data.filter((row) => {
|
data.filter((row) => {
|
||||||
@@ -16,15 +18,81 @@ export const filterByText = _ => (
|
|||||||
if (comparator === EQ) {
|
if (comparator === EQ) {
|
||||||
return cellStr === filterVal;
|
return cellStr === filterVal;
|
||||||
}
|
}
|
||||||
return cellStr.indexOf(filterVal) > -1;
|
if (caseSensitive) {
|
||||||
|
return cellStr.includes(filterVal);
|
||||||
|
}
|
||||||
|
return cellStr.toLocaleUpperCase().includes(filterVal.toLocaleUpperCase());
|
||||||
|
});
|
||||||
|
|
||||||
|
export const filterByNumber = _ => (
|
||||||
|
data,
|
||||||
|
dataField,
|
||||||
|
{ filterVal: { comparator, number } },
|
||||||
|
customFilterValue
|
||||||
|
) =>
|
||||||
|
data.filter((row) => {
|
||||||
|
if (number === '' || !comparator) return true;
|
||||||
|
let valid = true;
|
||||||
|
let cell = _.get(row, dataField);
|
||||||
|
if (customFilterValue) {
|
||||||
|
cell = customFilterValue(cell, row);
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (comparator) {
|
||||||
|
case EQ: {
|
||||||
|
if (cell != number) {
|
||||||
|
valid = false;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case GT: {
|
||||||
|
if (cell <= number) {
|
||||||
|
valid = false;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case GE: {
|
||||||
|
if (cell < number) {
|
||||||
|
valid = false;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case LT: {
|
||||||
|
if (cell >= number) {
|
||||||
|
valid = false;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case LE: {
|
||||||
|
if (cell > number) {
|
||||||
|
valid = false;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case NE: {
|
||||||
|
if (cell == number) {
|
||||||
|
valid = false;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default: {
|
||||||
|
console.error('Number comparator provided is not supported');
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return valid;
|
||||||
});
|
});
|
||||||
|
|
||||||
export const filterFactory = _ => (filterType) => {
|
export const filterFactory = _ => (filterType) => {
|
||||||
let filterFn;
|
let filterFn;
|
||||||
switch (filterType) {
|
switch (filterType) {
|
||||||
case FILTER_TYPE.TEXT:
|
case FILTER_TYPE.TEXT:
|
||||||
|
case FILTER_TYPE.SELECT:
|
||||||
filterFn = filterByText(_);
|
filterFn = filterByText(_);
|
||||||
break;
|
break;
|
||||||
|
case FILTER_TYPE.NUMBER:
|
||||||
|
filterFn = filterByNumber(_);
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
filterFn = filterByText(_);
|
filterFn = filterByText(_);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,7 +3,8 @@
|
|||||||
import React, { Component } from 'react';
|
import React, { Component } from 'react';
|
||||||
import PropTypes from 'prop-types';
|
import PropTypes from 'prop-types';
|
||||||
import { filters } from './filter';
|
import { filters } from './filter';
|
||||||
import { LIKE } from './comparison';
|
import { LIKE, EQ } from './comparison';
|
||||||
|
import { FILTER_TYPE } from './const';
|
||||||
|
|
||||||
export default (Base, {
|
export default (Base, {
|
||||||
_,
|
_,
|
||||||
@@ -47,8 +48,12 @@ export default (Base, {
|
|||||||
if (!_.isDefined(filterVal) || filterVal === '') {
|
if (!_.isDefined(filterVal) || filterVal === '') {
|
||||||
delete currFilters[dataField];
|
delete currFilters[dataField];
|
||||||
} else {
|
} else {
|
||||||
const { comparator = LIKE } = filter.props;
|
// select default comparator is EQ, others are LIKE
|
||||||
currFilters[dataField] = { filterVal, filterType, comparator };
|
const {
|
||||||
|
comparator = (filterType === FILTER_TYPE.SELECT ? EQ : LIKE),
|
||||||
|
caseSensitive = false
|
||||||
|
} = filter.props;
|
||||||
|
currFilters[dataField] = { filterVal, filterType, comparator, caseSensitive };
|
||||||
}
|
}
|
||||||
store.filters = currFilters;
|
store.filters = currFilters;
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,31 @@
|
|||||||
|
.react-bootstrap-table > table > thead > tr > th .filter {
|
||||||
|
font-weight: normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
.react-bootstrap-table > table > thead > tr > th .select-filter option[value=''],
|
||||||
|
.react-bootstrap-table > table > thead > tr > th .select-filter.placeholder-selected,
|
||||||
|
.react-bootstrap-table > table > thead > tr > th .filter::-webkit-input-placeholder,
|
||||||
|
.react-bootstrap-table > table > thead > tr > th .number-filter-input::-webkit-input-placeholder {
|
||||||
|
color: lightgrey;
|
||||||
|
font-style: italic;
|
||||||
|
}
|
||||||
|
|
||||||
|
.react-bootstrap-table > table > thead > tr > th .select-filter.placeholder-selected option:not([value='']) {
|
||||||
|
color: initial;
|
||||||
|
font-style: initial;
|
||||||
|
}
|
||||||
|
|
||||||
|
.react-bootstrap-table > table > thead > tr > th .number-filter {
|
||||||
|
display: flex;
|
||||||
|
}
|
||||||
|
|
||||||
|
.react-bootstrap-table > table > thead > tr > th .number-filter-input {
|
||||||
|
margin-left: 5px;
|
||||||
|
float: left;
|
||||||
|
width: calc(100% - 67px - 5px);
|
||||||
|
}
|
||||||
|
|
||||||
|
.react-bootstrap-table > table > thead > tr > th .number-filter-comparator {
|
||||||
|
width: 67px;
|
||||||
|
float: left;
|
||||||
|
}
|
||||||
@@ -0,0 +1,310 @@
|
|||||||
|
import 'jsdom-global/register';
|
||||||
|
import React from 'react';
|
||||||
|
import sinon from 'sinon';
|
||||||
|
import { mount } from 'enzyme';
|
||||||
|
import NumberFilter from '../../src/components/number';
|
||||||
|
import { FILTER_TYPE } from '../../src/const';
|
||||||
|
import * as Comparator from '../../src/comparison';
|
||||||
|
|
||||||
|
|
||||||
|
describe('Number Filter', () => {
|
||||||
|
let wrapper;
|
||||||
|
const onFilter = sinon.stub();
|
||||||
|
const column = {
|
||||||
|
dataField: 'price',
|
||||||
|
text: 'Product Price'
|
||||||
|
};
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
onFilter.reset();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('initialization', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
wrapper = mount(
|
||||||
|
<NumberFilter onFilter={ onFilter } column={ column } />
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should have correct state', () => {
|
||||||
|
expect(wrapper.state().isSelected).toBeFalsy();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should rendering component successfully', () => {
|
||||||
|
expect(wrapper).toHaveLength(1);
|
||||||
|
expect(wrapper.find('select')).toHaveLength(1);
|
||||||
|
expect(wrapper.find('input[type="number"]')).toHaveLength(1);
|
||||||
|
expect(wrapper.find('.number-filter')).toHaveLength(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should rendering comparator options correctly', () => {
|
||||||
|
const select = wrapper.find('select');
|
||||||
|
expect(select.find('option')).toHaveLength(wrapper.prop('comparators').length + 1);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('when withoutEmptyComparatorOption prop is true', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
wrapper = mount(
|
||||||
|
<NumberFilter onFilter={ onFilter } column={ column } withoutEmptyComparatorOption />
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should rendering comparator options correctly', () => {
|
||||||
|
const select = wrapper.find('select');
|
||||||
|
expect(select.find('option')).toHaveLength(wrapper.prop('comparators').length);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('when defaultValue.number props is defined', () => {
|
||||||
|
const number = 203;
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
wrapper = mount(
|
||||||
|
<NumberFilter onFilter={ onFilter } column={ column } defaultValue={ { number } } />
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should rendering input successfully', () => {
|
||||||
|
expect(wrapper).toHaveLength(1);
|
||||||
|
const input = wrapper.find('input[type="number"]');
|
||||||
|
expect(input).toHaveLength(1);
|
||||||
|
expect(input.props().defaultValue).toEqual(number);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('when defaultValue.comparator props is defined', () => {
|
||||||
|
const comparator = Comparator.EQ;
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
wrapper = mount(
|
||||||
|
<NumberFilter onFilter={ onFilter } column={ column } defaultValue={ { comparator } } />
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should rendering comparator select successfully', () => {
|
||||||
|
expect(wrapper).toHaveLength(1);
|
||||||
|
const select = wrapper.find('.number-filter-comparator');
|
||||||
|
expect(select).toHaveLength(1);
|
||||||
|
expect(select.props().defaultValue).toEqual(comparator);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('when defaultValue.number and defaultValue.comparator props is defined', () => {
|
||||||
|
const number = 203;
|
||||||
|
const comparator = Comparator.EQ;
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
wrapper = mount(
|
||||||
|
<NumberFilter
|
||||||
|
onFilter={ onFilter }
|
||||||
|
column={ column }
|
||||||
|
defaultValue={ { number, comparator } }
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should have correct state', () => {
|
||||||
|
expect(wrapper.state().isSelected).toBeTruthy();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should calling onFilter on componentDidMount', () => {
|
||||||
|
expect(onFilter.calledOnce).toBeTruthy();
|
||||||
|
expect(onFilter.calledWith(
|
||||||
|
column, { number: `${number}`, comparator }, FILTER_TYPE.NUMBER)).toBeTruthy();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('when options props is defined', () => {
|
||||||
|
const options = [2100, 2103, 2105];
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
wrapper = mount(
|
||||||
|
<NumberFilter
|
||||||
|
onFilter={ onFilter }
|
||||||
|
column={ column }
|
||||||
|
options={ options }
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should rendering number options instead of number input', () => {
|
||||||
|
expect(wrapper).toHaveLength(1);
|
||||||
|
const select = wrapper.find('.select-filter.placeholder-selected');
|
||||||
|
expect(select).toHaveLength(1);
|
||||||
|
expect(select.find('option')).toHaveLength(options.length + 1);
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('when withoutEmptyNumberOption props is defined', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
wrapper = mount(
|
||||||
|
<NumberFilter
|
||||||
|
onFilter={ onFilter }
|
||||||
|
column={ column }
|
||||||
|
options={ options }
|
||||||
|
withoutEmptyNumberOption
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should rendering number options instead of number input', () => {
|
||||||
|
const select = wrapper.find('.select-filter.placeholder-selected');
|
||||||
|
expect(select).toHaveLength(1);
|
||||||
|
expect(select.find('option')).toHaveLength(options.length);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('when defaultValue.number props is defined', () => {
|
||||||
|
const number = 203;
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
wrapper = mount(
|
||||||
|
<NumberFilter
|
||||||
|
onFilter={ onFilter }
|
||||||
|
column={ column }
|
||||||
|
defaultValue={ { number } }
|
||||||
|
options={ options }
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should rendering number options successfully', () => {
|
||||||
|
const select = wrapper.find('.select-filter.placeholder-selected');
|
||||||
|
expect(select).toHaveLength(1);
|
||||||
|
expect(select.props().defaultValue).toEqual(number);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('when defaultValue.number and defaultValue.comparator props is defined', () => {
|
||||||
|
const number = options[1];
|
||||||
|
const comparator = Comparator.EQ;
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
wrapper = mount(
|
||||||
|
<NumberFilter
|
||||||
|
onFilter={ onFilter }
|
||||||
|
column={ column }
|
||||||
|
defaultValue={ { number, comparator } }
|
||||||
|
options={ options }
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should rendering number options successfully', () => {
|
||||||
|
let select = wrapper.find('.placeholder-selected');
|
||||||
|
expect(select).toHaveLength(0);
|
||||||
|
|
||||||
|
select = wrapper.find('.select-filter');
|
||||||
|
expect(select).toHaveLength(1);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('when style props is defined', () => {
|
||||||
|
const style = { backgroundColor: 'red' };
|
||||||
|
beforeEach(() => {
|
||||||
|
wrapper = mount(
|
||||||
|
<NumberFilter
|
||||||
|
onFilter={ onFilter }
|
||||||
|
column={ column }
|
||||||
|
style={ style }
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should rendering component successfully', () => {
|
||||||
|
expect(wrapper).toHaveLength(1);
|
||||||
|
expect(wrapper.find('.number-filter').prop('style')).toEqual(style);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('when numberStyle props is defined', () => {
|
||||||
|
const numberStyle = { backgroundColor: 'red' };
|
||||||
|
beforeEach(() => {
|
||||||
|
wrapper = mount(
|
||||||
|
<NumberFilter
|
||||||
|
onFilter={ onFilter }
|
||||||
|
column={ column }
|
||||||
|
numberStyle={ numberStyle }
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should rendering component successfully', () => {
|
||||||
|
expect(wrapper).toHaveLength(1);
|
||||||
|
expect(wrapper.find('.number-filter-input').prop('style')).toEqual(numberStyle);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('when comparatorStyle props is defined', () => {
|
||||||
|
const comparatorStyle = { backgroundColor: 'red' };
|
||||||
|
beforeEach(() => {
|
||||||
|
wrapper = mount(
|
||||||
|
<NumberFilter
|
||||||
|
onFilter={ onFilter }
|
||||||
|
column={ column }
|
||||||
|
comparatorStyle={ comparatorStyle }
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should rendering component successfully', () => {
|
||||||
|
expect(wrapper).toHaveLength(1);
|
||||||
|
expect(wrapper.find('select').prop('style')).toEqual(comparatorStyle);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('when className props is defined', () => {
|
||||||
|
const className = 'test';
|
||||||
|
beforeEach(() => {
|
||||||
|
wrapper = mount(
|
||||||
|
<NumberFilter
|
||||||
|
onFilter={ onFilter }
|
||||||
|
column={ column }
|
||||||
|
className={ className }
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should rendering component successfully', () => {
|
||||||
|
expect(wrapper).toHaveLength(1);
|
||||||
|
expect(wrapper.hasClass(className)).toBeTruthy();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('when numberClassName props is defined', () => {
|
||||||
|
const className = 'test';
|
||||||
|
beforeEach(() => {
|
||||||
|
wrapper = mount(
|
||||||
|
<NumberFilter
|
||||||
|
onFilter={ onFilter }
|
||||||
|
column={ column }
|
||||||
|
numberClassName={ className }
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should rendering component successfully', () => {
|
||||||
|
expect(wrapper).toHaveLength(1);
|
||||||
|
expect(wrapper.find('.number-filter-input').prop('className').indexOf(className) > -1).toBeTruthy();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('when comparatorClassName props is defined', () => {
|
||||||
|
const className = 'test';
|
||||||
|
beforeEach(() => {
|
||||||
|
wrapper = mount(
|
||||||
|
<NumberFilter
|
||||||
|
onFilter={ onFilter }
|
||||||
|
column={ column }
|
||||||
|
comparatorClassName={ className }
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should rendering component successfully', () => {
|
||||||
|
expect(wrapper).toHaveLength(1);
|
||||||
|
expect(wrapper.find('select').prop('className').indexOf(className) > -1).toBeTruthy();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -0,0 +1,296 @@
|
|||||||
|
import 'jsdom-global/register';
|
||||||
|
import React from 'react';
|
||||||
|
import sinon from 'sinon';
|
||||||
|
import { mount } from 'enzyme';
|
||||||
|
import SelectFilter from '../../src/components/select';
|
||||||
|
import { FILTER_TYPE } from '../../src/const';
|
||||||
|
|
||||||
|
|
||||||
|
describe('Select Filter', () => {
|
||||||
|
let wrapper;
|
||||||
|
let instance;
|
||||||
|
const onFilter = sinon.stub();
|
||||||
|
const column = {
|
||||||
|
dataField: 'quality',
|
||||||
|
text: 'Product Quality'
|
||||||
|
};
|
||||||
|
const options = {
|
||||||
|
0: 'Bad',
|
||||||
|
1: 'Good',
|
||||||
|
2: 'Unknow'
|
||||||
|
};
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
onFilter.reset();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('initialization', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
wrapper = mount(
|
||||||
|
<SelectFilter onFilter={ onFilter } column={ column } options={ options } />
|
||||||
|
);
|
||||||
|
instance = wrapper.instance();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should have correct state', () => {
|
||||||
|
expect(instance.state.isSelected).toBeFalsy();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should rendering component successfully', () => {
|
||||||
|
expect(wrapper).toHaveLength(1);
|
||||||
|
expect(wrapper.find('select')).toHaveLength(1);
|
||||||
|
expect(wrapper.find('.select-filter')).toHaveLength(1);
|
||||||
|
expect(wrapper.find('.placeholder-selected')).toHaveLength(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should rendering select options correctly', () => {
|
||||||
|
const select = wrapper.find('select');
|
||||||
|
expect(select.find('option')).toHaveLength(Object.keys(options).length + 1);
|
||||||
|
expect(select.childAt(0).text()).toEqual(`Select ${column.text}...`);
|
||||||
|
|
||||||
|
Object.keys(options).forEach((key, i) => {
|
||||||
|
expect(select.childAt(i + 1).prop('value')).toEqual(key);
|
||||||
|
expect(select.childAt(i + 1).text()).toEqual(options[key]);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('when defaultValue is defined', () => {
|
||||||
|
let defaultValue;
|
||||||
|
|
||||||
|
describe('and it is valid', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
defaultValue = '0';
|
||||||
|
wrapper = mount(
|
||||||
|
<SelectFilter
|
||||||
|
onFilter={ onFilter }
|
||||||
|
column={ column }
|
||||||
|
options={ options }
|
||||||
|
defaultValue={ defaultValue }
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
instance = wrapper.instance();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should have correct state', () => {
|
||||||
|
expect(instance.state.isSelected).toBeTruthy();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should rendering component successfully', () => {
|
||||||
|
expect(wrapper).toHaveLength(1);
|
||||||
|
expect(wrapper.find('.placeholder-selected')).toHaveLength(0);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should calling onFilter on componentDidMount', () => {
|
||||||
|
expect(onFilter.calledOnce).toBeTruthy();
|
||||||
|
expect(onFilter.calledWith(column, defaultValue, FILTER_TYPE.SELECT)).toBeTruthy();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('when placeholder is defined', () => {
|
||||||
|
const placeholder = 'test';
|
||||||
|
beforeEach(() => {
|
||||||
|
wrapper = mount(
|
||||||
|
<SelectFilter
|
||||||
|
onFilter={ onFilter }
|
||||||
|
column={ column }
|
||||||
|
options={ options }
|
||||||
|
placeholder={ placeholder }
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
instance = wrapper.instance();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should rendering component successfully', () => {
|
||||||
|
expect(wrapper).toHaveLength(1);
|
||||||
|
const select = wrapper.find('select');
|
||||||
|
expect(select.childAt(0).text()).toEqual(placeholder);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('when style is defined', () => {
|
||||||
|
const style = { backgroundColor: 'red' };
|
||||||
|
beforeEach(() => {
|
||||||
|
wrapper = mount(
|
||||||
|
<SelectFilter
|
||||||
|
onFilter={ onFilter }
|
||||||
|
column={ column }
|
||||||
|
options={ options }
|
||||||
|
style={ style }
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should rendering component successfully', () => {
|
||||||
|
expect(wrapper).toHaveLength(1);
|
||||||
|
expect(wrapper.find('select').prop('style')).toEqual(style);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('when withoutEmptyOption is defined', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
wrapper = mount(
|
||||||
|
<SelectFilter
|
||||||
|
onFilter={ onFilter }
|
||||||
|
column={ column }
|
||||||
|
options={ options }
|
||||||
|
withoutEmptyOption
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should rendering select without default empty option', () => {
|
||||||
|
const select = wrapper.find('select');
|
||||||
|
expect(select.find('option')).toHaveLength(Object.keys(options).length);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('componentDidUpdate', () => {
|
||||||
|
let prevProps;
|
||||||
|
|
||||||
|
describe('when props.defaultValue is diff from prevProps.defaultValue', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
wrapper = mount(
|
||||||
|
<SelectFilter
|
||||||
|
onFilter={ onFilter }
|
||||||
|
column={ column }
|
||||||
|
options={ options }
|
||||||
|
defaultValue="0"
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
prevProps = {
|
||||||
|
column,
|
||||||
|
options,
|
||||||
|
defaultValue: '1'
|
||||||
|
};
|
||||||
|
instance = wrapper.instance();
|
||||||
|
instance.componentDidUpdate(prevProps);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should update', () => {
|
||||||
|
expect(onFilter.callCount).toBe(2);
|
||||||
|
expect(onFilter.calledWith(
|
||||||
|
column, instance.props.defaultValue, FILTER_TYPE.SELECT)).toBeTruthy();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('when props.options is diff from prevProps.options', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
wrapper = mount(
|
||||||
|
<SelectFilter
|
||||||
|
onFilter={ onFilter }
|
||||||
|
column={ column }
|
||||||
|
options={ {
|
||||||
|
...options,
|
||||||
|
3: 'Best'
|
||||||
|
} }
|
||||||
|
defaultValue="1"
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
prevProps = {
|
||||||
|
column,
|
||||||
|
options
|
||||||
|
};
|
||||||
|
instance = wrapper.instance();
|
||||||
|
instance.componentDidUpdate(prevProps);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should update', () => {
|
||||||
|
expect(onFilter.callCount).toBe(2);
|
||||||
|
expect(onFilter.calledWith(
|
||||||
|
column, instance.props.defaultValue, FILTER_TYPE.SELECT)).toBeTruthy();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('cleanFiltered', () => {
|
||||||
|
describe('when props.defaultValue is defined', () => {
|
||||||
|
const defaultValue = '0';
|
||||||
|
beforeEach(() => {
|
||||||
|
wrapper = mount(
|
||||||
|
<SelectFilter
|
||||||
|
onFilter={ onFilter }
|
||||||
|
column={ column }
|
||||||
|
options={ options }
|
||||||
|
defaultValue={ defaultValue }
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
instance = wrapper.instance();
|
||||||
|
instance.cleanFiltered();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should setting state correctly', () => {
|
||||||
|
expect(instance.state.isSelected).toBeTruthy();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should calling onFilter correctly', () => {
|
||||||
|
expect(onFilter.callCount).toBe(2);
|
||||||
|
expect(onFilter.calledWith(column, defaultValue, FILTER_TYPE.SELECT)).toBeTruthy();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('when props.defaultValue is not defined', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
wrapper = mount(
|
||||||
|
<SelectFilter
|
||||||
|
onFilter={ onFilter }
|
||||||
|
column={ column }
|
||||||
|
options={ options }
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
instance = wrapper.instance();
|
||||||
|
instance.cleanFiltered();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should setting state correctly', () => {
|
||||||
|
expect(instance.state.isSelected).toBeFalsy();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should calling onFilter correctly', () => {
|
||||||
|
expect(onFilter.callCount).toBe(1);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('applyFilter', () => {
|
||||||
|
const value = '2';
|
||||||
|
beforeEach(() => {
|
||||||
|
wrapper = mount(
|
||||||
|
<SelectFilter onFilter={ onFilter } column={ column } options={ options } />
|
||||||
|
);
|
||||||
|
instance = wrapper.instance();
|
||||||
|
instance.applyFilter(value);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should setting state correctly', () => {
|
||||||
|
expect(instance.state.isSelected).toBeTruthy();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should calling onFilter correctly', () => {
|
||||||
|
expect(onFilter.callCount).toBe(1);
|
||||||
|
expect(onFilter.calledWith(column, value, FILTER_TYPE.SELECT)).toBeTruthy();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('filter', () => {
|
||||||
|
const event = { target: { value: 'tester' } };
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
wrapper = mount(
|
||||||
|
<SelectFilter onFilter={ onFilter } column={ column } options={ options } />
|
||||||
|
);
|
||||||
|
instance = wrapper.instance();
|
||||||
|
instance.filter(event);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should setting state correctly', () => {
|
||||||
|
expect(instance.state.isSelected).toBeTruthy();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should calling onFilter correctly', () => {
|
||||||
|
expect(onFilter.callCount).toBe(1);
|
||||||
|
expect(onFilter.calledWith(column, event.target.value, FILTER_TYPE.SELECT)).toBeTruthy();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -4,7 +4,7 @@ import Store from 'react-bootstrap-table-next/src/store';
|
|||||||
|
|
||||||
import { filters } from '../src/filter';
|
import { filters } from '../src/filter';
|
||||||
import { FILTER_TYPE } from '../src/const';
|
import { FILTER_TYPE } from '../src/const';
|
||||||
import { LIKE, EQ } from '../src/comparison';
|
import { LIKE, EQ, GT, GE, LT, LE, NE } from '../src/comparison';
|
||||||
|
|
||||||
const data = [];
|
const data = [];
|
||||||
for (let i = 0; i < 20; i += 1) {
|
for (let i = 0; i < 20; i += 1) {
|
||||||
@@ -37,7 +37,7 @@ describe('filter', () => {
|
|||||||
}];
|
}];
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('text filter', () => {
|
describe('filterByText', () => {
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
filterFn = filters(store, columns, _);
|
filterFn = filters(store, columns, _);
|
||||||
});
|
});
|
||||||
@@ -55,6 +55,20 @@ describe('filter', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('when caseSensitive is true', () => {
|
||||||
|
it('should returning correct result', () => {
|
||||||
|
currFilters.name = {
|
||||||
|
filterVal: 'NAME',
|
||||||
|
caseSensitive: true,
|
||||||
|
filterType: FILTER_TYPE.TEXT
|
||||||
|
};
|
||||||
|
|
||||||
|
const result = filterFn(currFilters);
|
||||||
|
expect(result).toBeDefined();
|
||||||
|
expect(result).toHaveLength(0);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
describe(`when default comparator is ${EQ}`, () => {
|
describe(`when default comparator is ${EQ}`, () => {
|
||||||
it('should returning correct result', () => {
|
it('should returning correct result', () => {
|
||||||
currFilters.name = {
|
currFilters.name = {
|
||||||
@@ -91,4 +105,114 @@ describe('filter', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('filterByNumber', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
filterFn = filters(store, columns, _);
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('when currFilters.filterVal.comparator is empty', () => {
|
||||||
|
it('should returning correct result', () => {
|
||||||
|
currFilters.price = {
|
||||||
|
filterVal: { comparator: '', number: '203' },
|
||||||
|
filterType: FILTER_TYPE.NUMBER
|
||||||
|
};
|
||||||
|
|
||||||
|
let result = filterFn(currFilters);
|
||||||
|
expect(result).toHaveLength(data.length);
|
||||||
|
|
||||||
|
currFilters.price.filterVal.comparator = undefined;
|
||||||
|
result = filterFn(currFilters);
|
||||||
|
expect(result).toHaveLength(data.length);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('when currFilters.filterVal.number is empty', () => {
|
||||||
|
it('should returning correct result', () => {
|
||||||
|
currFilters.price = {
|
||||||
|
filterVal: { comparator: EQ, number: '' },
|
||||||
|
filterType: FILTER_TYPE.NUMBER
|
||||||
|
};
|
||||||
|
|
||||||
|
const result = filterFn(currFilters);
|
||||||
|
expect(result).toHaveLength(data.length);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe(`when currFilters.filterVal.comparator is ${EQ}`, () => {
|
||||||
|
it('should returning correct result', () => {
|
||||||
|
currFilters.price = {
|
||||||
|
filterVal: { comparator: EQ, number: '203' },
|
||||||
|
filterType: FILTER_TYPE.NUMBER
|
||||||
|
};
|
||||||
|
|
||||||
|
let result = filterFn(currFilters);
|
||||||
|
expect(result).toHaveLength(1);
|
||||||
|
|
||||||
|
currFilters.price.filterVal.number = '0';
|
||||||
|
result = filterFn(currFilters);
|
||||||
|
expect(result).toHaveLength(0);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe(`when currFilters.filterVal.comparator is ${GT}`, () => {
|
||||||
|
it('should returning correct result', () => {
|
||||||
|
currFilters.price = {
|
||||||
|
filterVal: { comparator: GT, number: '203' },
|
||||||
|
filterType: FILTER_TYPE.NUMBER
|
||||||
|
};
|
||||||
|
|
||||||
|
const result = filterFn(currFilters);
|
||||||
|
expect(result).toHaveLength(16);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe(`when currFilters.filterVal.comparator is ${GE}`, () => {
|
||||||
|
it('should returning correct result', () => {
|
||||||
|
currFilters.price = {
|
||||||
|
filterVal: { comparator: GE, number: '203' },
|
||||||
|
filterType: FILTER_TYPE.NUMBER
|
||||||
|
};
|
||||||
|
|
||||||
|
const result = filterFn(currFilters);
|
||||||
|
expect(result).toHaveLength(17);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe(`when currFilters.filterVal.comparator is ${LT}`, () => {
|
||||||
|
it('should returning correct result', () => {
|
||||||
|
currFilters.price = {
|
||||||
|
filterVal: { comparator: LT, number: '203' },
|
||||||
|
filterType: FILTER_TYPE.NUMBER
|
||||||
|
};
|
||||||
|
|
||||||
|
const result = filterFn(currFilters);
|
||||||
|
expect(result).toHaveLength(3);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe(`when currFilters.filterVal.comparator is ${LE}`, () => {
|
||||||
|
it('should returning correct result', () => {
|
||||||
|
currFilters.price = {
|
||||||
|
filterVal: { comparator: LE, number: '203' },
|
||||||
|
filterType: FILTER_TYPE.NUMBER
|
||||||
|
};
|
||||||
|
|
||||||
|
const result = filterFn(currFilters);
|
||||||
|
expect(result).toHaveLength(4);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe(`when currFilters.filterVal.comparator is ${NE}`, () => {
|
||||||
|
it('should returning correct result', () => {
|
||||||
|
currFilters.price = {
|
||||||
|
filterVal: { comparator: NE, number: '203' },
|
||||||
|
filterType: FILTER_TYPE.NUMBER
|
||||||
|
};
|
||||||
|
|
||||||
|
const result = filterFn(currFilters);
|
||||||
|
expect(result).toHaveLength(19);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
# react-bootstrap-table2-pagination
|
# react-bootstrap-table2-paginator
|
||||||
|
|
||||||
`react-bootstrap-table2` separate the pagination code base to [`react-bootstrap-table2-pagination`](https://github.com/react-bootstrap-table/react-bootstrap-table2/tree/develop/packages/react-bootstrap-table2-paginator), so there's a little bit different when you use pagination. In the following, we are going to show you how to enable and configure the a pagination table
|
`react-bootstrap-table2` separate the pagination code base to [`react-bootstrap-table2-paginator`](https://github.com/react-bootstrap-table/react-bootstrap-table2/tree/develop/packages/react-bootstrap-table2-paginator), so there's a little bit different when you use pagination. In the following, we are going to show you how to enable and configure the a pagination table
|
||||||
|
|
||||||
**[Live Demo For Pagination](https://react-bootstrap-table.github.io/react-bootstrap-table2/storybook/index.html?selectedKind=Pagination)**
|
**[Live Demo For Pagination](https://react-bootstrap-table.github.io/react-bootstrap-table2/storybook/index.html?selectedKind=Pagination)**
|
||||||
|
|
||||||
@@ -11,7 +11,7 @@
|
|||||||
## Install
|
## Install
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
$ npm install react-bootstrap-table2-pagination --save
|
$ npm install react-bootstrap-table2-paginator --save
|
||||||
```
|
```
|
||||||
|
|
||||||
## Add CSS
|
## Add CSS
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "react-bootstrap-table-next",
|
"name": "react-bootstrap-table-next",
|
||||||
"version": "0.1.1",
|
"version": "0.1.3",
|
||||||
"description": "Next generation of react-bootstrap-table",
|
"description": "Next generation of react-bootstrap-table",
|
||||||
"main": "./lib/index.js",
|
"main": "./lib/index.js",
|
||||||
"repository": {
|
"repository": {
|
||||||
|
|||||||
@@ -131,6 +131,7 @@ HeaderCell.propTypes = {
|
|||||||
attrs: PropTypes.oneOfType([PropTypes.object, PropTypes.func]),
|
attrs: PropTypes.oneOfType([PropTypes.object, PropTypes.func]),
|
||||||
sort: PropTypes.bool,
|
sort: PropTypes.bool,
|
||||||
sortFunc: PropTypes.func,
|
sortFunc: PropTypes.func,
|
||||||
|
onSort: PropTypes.func,
|
||||||
editable: PropTypes.oneOfType([PropTypes.bool, PropTypes.func]),
|
editable: PropTypes.oneOfType([PropTypes.bool, PropTypes.func]),
|
||||||
editCellStyle: PropTypes.oneOfType([PropTypes.object, PropTypes.func]),
|
editCellStyle: PropTypes.oneOfType([PropTypes.object, PropTypes.func]),
|
||||||
editCellClasses: PropTypes.oneOfType([PropTypes.string, PropTypes.func]),
|
editCellClasses: PropTypes.oneOfType([PropTypes.string, PropTypes.func]),
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
export default ExtendBase =>
|
export default ExtendBase =>
|
||||||
class ColumnResolver extends ExtendBase {
|
class ColumnResolver extends ExtendBase {
|
||||||
visibleColumnSize() {
|
visibleColumnSize() {
|
||||||
return this.props.columns.length;
|
return this.props.columns.filter(c => !c.hidden).length;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -13,12 +13,13 @@ export default class SelectionCell extends Component {
|
|||||||
selected: PropTypes.bool,
|
selected: PropTypes.bool,
|
||||||
onRowSelect: PropTypes.func,
|
onRowSelect: PropTypes.func,
|
||||||
disabled: PropTypes.bool,
|
disabled: PropTypes.bool,
|
||||||
rowIndex: PropTypes.number
|
rowIndex: PropTypes.number,
|
||||||
|
clickToSelect: PropTypes.bool
|
||||||
}
|
}
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
super();
|
super();
|
||||||
this.handleRowClick = this.handleRowClick.bind(this);
|
this.handleClick = this.handleClick.bind(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
shouldComponentUpdate(nextProps) {
|
shouldComponentUpdate(nextProps) {
|
||||||
@@ -27,17 +28,19 @@ export default class SelectionCell extends Component {
|
|||||||
return nextProps.selected !== selected;
|
return nextProps.selected !== selected;
|
||||||
}
|
}
|
||||||
|
|
||||||
handleRowClick() {
|
handleClick() {
|
||||||
const {
|
const {
|
||||||
mode: inputType,
|
mode: inputType,
|
||||||
rowKey,
|
rowKey,
|
||||||
selected,
|
selected,
|
||||||
onRowSelect,
|
onRowSelect,
|
||||||
disabled,
|
disabled,
|
||||||
rowIndex
|
rowIndex,
|
||||||
|
clickToSelect
|
||||||
} = this.props;
|
} = this.props;
|
||||||
|
|
||||||
if (disabled) return;
|
if (disabled) return;
|
||||||
|
if (clickToSelect) return;
|
||||||
|
|
||||||
const checked = inputType === Const.ROW_SELECT_SINGLE
|
const checked = inputType === Const.ROW_SELECT_SINGLE
|
||||||
? true
|
? true
|
||||||
@@ -54,7 +57,7 @@ export default class SelectionCell extends Component {
|
|||||||
} = this.props;
|
} = this.props;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<td onClick={ this.handleRowClick }>
|
<td onClick={ this.handleClick }>
|
||||||
<input
|
<input
|
||||||
type={ inputType }
|
type={ inputType }
|
||||||
checked={ selected }
|
checked={ selected }
|
||||||
|
|||||||
20
packages/react-bootstrap-table2/src/row.js
vendored
20
packages/react-bootstrap-table2/src/row.js
vendored
@@ -1,4 +1,5 @@
|
|||||||
/* eslint react/prop-types: 0 */
|
/* eslint react/prop-types: 0 */
|
||||||
|
/* eslint react/no-array-index-key: 0 */
|
||||||
import React, { Component } from 'react';
|
import React, { Component } from 'react';
|
||||||
import PropTypes from 'prop-types';
|
import PropTypes from 'prop-types';
|
||||||
|
|
||||||
@@ -12,6 +13,7 @@ class Row extends Component {
|
|||||||
super(props);
|
super(props);
|
||||||
this.clickNum = 0;
|
this.clickNum = 0;
|
||||||
this.handleRowClick = this.handleRowClick.bind(this);
|
this.handleRowClick = this.handleRowClick.bind(this);
|
||||||
|
this.handleSimpleRowClick = this.handleSimpleRowClick.bind(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
handleRowClick(e) {
|
handleRowClick(e) {
|
||||||
@@ -35,7 +37,7 @@ class Row extends Component {
|
|||||||
|
|
||||||
const clickFn = () => {
|
const clickFn = () => {
|
||||||
if (attrs.onClick) {
|
if (attrs.onClick) {
|
||||||
attrs.onClick(e);
|
attrs.onClick(e, row, rowIndex);
|
||||||
}
|
}
|
||||||
if (selectable) {
|
if (selectable) {
|
||||||
const key = _.get(row, keyField);
|
const key = _.get(row, keyField);
|
||||||
@@ -56,6 +58,16 @@ class Row extends Component {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
handleSimpleRowClick(e) {
|
||||||
|
const {
|
||||||
|
row,
|
||||||
|
rowIndex,
|
||||||
|
attrs
|
||||||
|
} = this.props;
|
||||||
|
|
||||||
|
attrs.onClick(e, row, rowIndex);
|
||||||
|
}
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
const {
|
const {
|
||||||
row,
|
row,
|
||||||
@@ -89,6 +101,8 @@ class Row extends Component {
|
|||||||
const trAttrs = { ...attrs };
|
const trAttrs = { ...attrs };
|
||||||
if (clickToSelect) {
|
if (clickToSelect) {
|
||||||
trAttrs.onClick = this.handleRowClick;
|
trAttrs.onClick = this.handleRowClick;
|
||||||
|
} else if (attrs.onClick) {
|
||||||
|
trAttrs.onClick = this.handleSimpleRowClick;
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@@ -126,7 +140,7 @@ class Row extends Component {
|
|||||||
}
|
}
|
||||||
return (
|
return (
|
||||||
<EditingCell
|
<EditingCell
|
||||||
key={ content }
|
key={ `${content}-${index}` }
|
||||||
row={ row }
|
row={ row }
|
||||||
column={ column }
|
column={ column }
|
||||||
className={ editCellclasses }
|
className={ editCellclasses }
|
||||||
@@ -137,7 +151,7 @@ class Row extends Component {
|
|||||||
}
|
}
|
||||||
return (
|
return (
|
||||||
<Cell
|
<Cell
|
||||||
key={ content }
|
key={ `${content}-${index}` }
|
||||||
row={ row }
|
row={ row }
|
||||||
rowIndex={ rowIndex }
|
rowIndex={ rowIndex }
|
||||||
columnIndex={ index }
|
columnIndex={ index }
|
||||||
|
|||||||
@@ -25,6 +25,10 @@ export default Base =>
|
|||||||
if (column.length > 0) {
|
if (column.length > 0) {
|
||||||
store.setSort(column[0], order);
|
store.setSort(column[0], order);
|
||||||
|
|
||||||
|
if (column[0].onSort) {
|
||||||
|
column[0].onSort(store.sortField, store.sortOrder);
|
||||||
|
}
|
||||||
|
|
||||||
if (this.isRemoteSort() || this.isRemotePagination()) {
|
if (this.isRemoteSort() || this.isRemotePagination()) {
|
||||||
this.handleSortChange();
|
this.handleSortChange();
|
||||||
} else {
|
} else {
|
||||||
@@ -48,6 +52,10 @@ export default Base =>
|
|||||||
const { store } = this.props;
|
const { store } = this.props;
|
||||||
store.setSort(column);
|
store.setSort(column);
|
||||||
|
|
||||||
|
if (column.onSort) {
|
||||||
|
column.onSort(store.sortField, store.sortOrder);
|
||||||
|
}
|
||||||
|
|
||||||
if (this.isRemoteSort() || this.isRemotePagination()) {
|
if (this.isRemoteSort() || this.isRemotePagination()) {
|
||||||
this.handleSortChange();
|
this.handleSortChange();
|
||||||
} else {
|
} else {
|
||||||
|
|||||||
@@ -34,12 +34,12 @@ describe('<SelectionCell />', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('handleRowClick', () => {
|
describe('handleClick', () => {
|
||||||
describe('when <input /> was been clicked', () => {
|
describe('when <input /> was been clicked', () => {
|
||||||
const rowKey = 1;
|
const rowKey = 1;
|
||||||
const selected = true;
|
const selected = true;
|
||||||
let mockOnRowSelect;
|
let mockOnRowSelect;
|
||||||
const spy = sinon.spy(SelectionCell.prototype, 'handleRowClick');
|
const spy = sinon.spy(SelectionCell.prototype, 'handleClick');
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
mockOnRowSelect = sinon.stub();
|
mockOnRowSelect = sinon.stub();
|
||||||
|
|||||||
@@ -10,16 +10,7 @@ import wrapperFactory from '../../src/sort/wrapper';
|
|||||||
|
|
||||||
describe('SortWrapper', () => {
|
describe('SortWrapper', () => {
|
||||||
let wrapper;
|
let wrapper;
|
||||||
|
let columns;
|
||||||
const columns = [{
|
|
||||||
dataField: 'id',
|
|
||||||
text: 'ID',
|
|
||||||
sort: true
|
|
||||||
}, {
|
|
||||||
dataField: 'name',
|
|
||||||
text: 'Name',
|
|
||||||
sort: true
|
|
||||||
}];
|
|
||||||
|
|
||||||
const data = [{
|
const data = [{
|
||||||
id: 1,
|
id: 1,
|
||||||
@@ -37,6 +28,15 @@ describe('SortWrapper', () => {
|
|||||||
const SortWrapper = wrapperFactory(BootstrapTable);
|
const SortWrapper = wrapperFactory(BootstrapTable);
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
|
columns = [{
|
||||||
|
dataField: 'id',
|
||||||
|
text: 'ID',
|
||||||
|
sort: true
|
||||||
|
}, {
|
||||||
|
dataField: 'name',
|
||||||
|
text: 'Name',
|
||||||
|
sort: true
|
||||||
|
}];
|
||||||
wrapper = shallow(
|
wrapper = shallow(
|
||||||
<SortWrapper
|
<SortWrapper
|
||||||
keyField={ keyField }
|
keyField={ keyField }
|
||||||
@@ -58,9 +58,10 @@ describe('SortWrapper', () => {
|
|||||||
|
|
||||||
describe('call handleSort function', () => {
|
describe('call handleSort function', () => {
|
||||||
let sortBySpy;
|
let sortBySpy;
|
||||||
const sortColumn = columns[0];
|
let sortColumn;
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
|
sortColumn = columns[0];
|
||||||
store = new Store(keyField);
|
store = new Store(keyField);
|
||||||
store.data = data;
|
store.data = data;
|
||||||
sortBySpy = sinon.spy(store, 'sortBy');
|
sortBySpy = sinon.spy(store, 'sortBy');
|
||||||
@@ -130,6 +131,32 @@ describe('SortWrapper', () => {
|
|||||||
expect(onTableChangeCB.calledOnce).toBeTruthy();
|
expect(onTableChangeCB.calledOnce).toBeTruthy();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('when column.onSort prop is defined', () => {
|
||||||
|
const onSortCB = jest.fn();
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
columns[0].onSort = onSortCB;
|
||||||
|
wrapper = shallow(
|
||||||
|
<SortWrapper
|
||||||
|
keyField={ keyField }
|
||||||
|
data={ data }
|
||||||
|
columns={ columns }
|
||||||
|
store={ store }
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
wrapper.instance().handleSort(sortColumn);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should calling column.onSort function correctly', () => {
|
||||||
|
expect(onSortCB).toHaveBeenCalledTimes(1);
|
||||||
|
expect(onSortCB).toHaveBeenCalledWith(columns[0].dataField, Const.SORT_DESC);
|
||||||
|
|
||||||
|
wrapper.instance().handleSort(sortColumn);
|
||||||
|
expect(onSortCB).toHaveBeenCalledTimes(2);
|
||||||
|
expect(onSortCB).toHaveBeenCalledWith(columns[0].dataField, Const.SORT_ASC);
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('when defaultSorted prop is defined', () => {
|
describe('when defaultSorted prop is defined', () => {
|
||||||
@@ -161,6 +188,28 @@ describe('SortWrapper', () => {
|
|||||||
it('should update store.sortOrder correctly', () => {
|
it('should update store.sortOrder correctly', () => {
|
||||||
expect(store.sortOrder).toEqual(defaultSorted[0].order);
|
expect(store.sortOrder).toEqual(defaultSorted[0].order);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('when column.onSort prop is defined', () => {
|
||||||
|
const onSortCB = jest.fn();
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
columns[1].onSort = onSortCB;
|
||||||
|
wrapper = shallow(
|
||||||
|
<SortWrapper
|
||||||
|
keyField={ keyField }
|
||||||
|
data={ data }
|
||||||
|
columns={ columns }
|
||||||
|
store={ store }
|
||||||
|
defaultSorted={ defaultSorted }
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should calling column.onSort function correctly', () => {
|
||||||
|
expect(onSortCB).toHaveBeenCalledTimes(1);
|
||||||
|
expect(onSortCB).toHaveBeenCalledWith(defaultSorted[0].dataField, defaultSorted[0].order);
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('componentWillReceiveProps', () => {
|
describe('componentWillReceiveProps', () => {
|
||||||
|
|||||||
Reference in New Issue
Block a user