xxxxxxxxxx
1
<!DOCTYPE html>
2
<html lang="en">
3
4
<head>
5
<meta charset="utf-8">
6
<meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no">
7
<meta name="theme-color" content="#000000">
8
<link rel="manifest" href="%PUBLIC_URL%/manifest.json">
9
<link rel="shortcut icon" href="%PUBLIC_URL%/favicon.ico">
10
<title>React App</title>
11
</head>
12
13
<body>
14
<link rel="stylesheet" href="https://pro.fontawesome.com/releases/v5.2.0/css/all.css" />
15
<link rel="stylesheet" href="https://static.fontawesome.com/css/fontawesome-app.css" />
16
<noscript>You need to enable JavaScript to run this app.</noscript>
17
<div id="root"></div>
18
</body>
19
20
</html>
21
1
1
xxxxxxxxxx
1
import React from 'react';
2
import {MDBDataTable} from "mdbreact";
3
4
export default class App extends React.Component {
5
6
constructor(props) {
7
super(props);
8
this.handleUserChange = this.handleUserChange.bind(this);
9
10
this.state = {
11
userList: [],
12
selectedUsers: []
13
};
14
}
15
16
componentDidMount() {
17
const userList = ['Tiger Nixon', 'Charde Marshall', 'Jena Gaines', 'Quinn Flynn', 'Rhona Davidson', 'Sonya Frost', 'Colleen Hurst', 'Herrod Chandler', 'Garrett Winters', 'Brielle Williamson', 'Ashton Cox', 'Cedric Kelly', 'Airi Satou'];
18
this.setState(() => ({userList}));
19
}
20
21
handleUserChange(e) {
22
/* If the nodeValue ('ADD' or 'REMOVE') we add or remove the selected user to the selectedUsers array and update state */
23
const {value, attributes: {action: {nodeValue}}} = e.currentTarget;
24
25
this.setState((prevState) => {
26
let selectedUsers = [];
27
if (nodeValue === 'ADD') {
28
selectedUsers = [...prevState.selectedUsers, value];
29
} else {
30
selectedUsers = prevState.selectedUsers.filter((user) => (value !== user));
31
}
32
return ({selectedUsers});
33
});
34
}
35
36
buildRowData(userList) {
37
/*
38
Build up the data needed for the table.
39
In order to have a component inside the datatable, this pattern with the 'searchValue' needs to be used.
40
ref: https://mdbootstrap.com/support/react/mdbdatatable-search-not-returning-results-if-rendered-by-another-component/
41
*/
42
const {selectedUsers} = this.state,
43
MyButton = ({searchValue, btnClass, meta, label, handler}) => {
44
return <button className={`btn btn-block btn-sm ${btnClass}`} value={'actions'} type='button' {...meta} onClick={(e) => (handler(e))}>{label}</button>;
45
};
46
let data = {
47
rows: [],
48
columns: [
49
{label: 'Name', field: 'Name'},
50
{label: 'Actions', field: 'Actions'}
51
]
52
};
53
54
userList.forEach((user) => {
55
/* Populate both columns, and depending on the current state, the My_Button component will display either 'Remove' or 'Add' */
56
const added = (selectedUsers.includes(user)),
57
label = (added) ? 'Remove' : 'Add',
58
btnClass = (added) ? 'btn-danger' : 'btn-default',
59
meta = {action: (added) ? 'REMOVE' : 'ADD', value: user},
60
rowObj = {
61
'Name': user,
62
'Actions': <MyButton btnClass={btnClass} searchValue={'actions'} meta={meta} label={label} handler={this.handleUserChange} />
63
};
64
65
data['rows'].push(rowObj);
66
});
67
68
return data;
69
};
70
71
render() {
72
const {userList, selectedUsers} = this.state;
73
return (
74
<div className="container" style={{marginTop: '3rem'}}>
75
<div className="row">
76
<h4>
77
<a href='https://mdbootstrap.com/support/react/mdbdatatable-search-filter-is-not-honored-after-state-change/'>https://mdbootstrap.com/support/react/mdbdatatable-search-filter-is-not-honored-after-state-change/</a>
78
</h4>
79
<div className="row">
80
<div className="col-lg-6"><h5>Expected behavior:</h5>
81
<p>The datatable is listens to the search term, and the rows are filtered. State updates, which causes the datatable to re-render, but the search term in the filter should be honored.</p>
82
</div>
83
<div className="col-lg-6">
84
<h5>Actual behavior:</h5>
85
<p>After a state change, the datatable re-renders, and the search term is ignored until the next onChange event.</p>
86
</div>
87
</div>
88
<h5>Screen recording of it happening on a live application: <a href='https://www.youtube.com/watch?v=AS5STv01Es4'>https://www.youtube.com/watch?v=AS5STv01Es4</a>
89
</h5>
90
</div>
91
<br />
92
<div className='row'>
93
<div className="col-lg-6">
94
<MDBDataTable
95
data={this.buildRowData(userList)}
96
entries={3}
97
entriesOptions={[3, 5, 10, 15]} />
98
</div>
99
<div className="col-lg-5 col-lg-offset-1">
100
<h3>Selected Users:</h3>
101
<ul>
102
{
103
selectedUsers.map((user, idx) => {
104
return (
105
<li key={idx}>{user}</li>
106
)
107
})
108
}
109
</ul>
110
</div>
111
</div>
112
</div>
113
);
114
}
115
116
}
Console errors: 0