admin管理员组文章数量:1345891
I am having trouble figuring out how I can update the ponent with the new data from the database after the edit function has been run and the update to the database was successful.
I have tried doing this.forceUpdate()
but that doesn't work. I heard about using this.setState()
but I don't know how I would update it using that with the database as my classroom state is a JSON array.
I have been thinking and google-ing long and hard for days without any luck, so any advice or help with this problem would be greatly appreciated.
P.S. I am using MySQL for my Database. I don't know if that matters.
App.js
import React, { Component } from 'react';
import './App.css';
class App extends Component {
constructor(props) {
super(props);
this.state = {
toggle: false,
toggleIdxArray: [],
classroom: [],
childflagged: '',
flagreason: '',
id: ''
};
this.eventHandler = this.eventHandler.bind(this);
this.logChange = this.logChange.bind(this);
this.handleEdit = this.handleEdit.bind(this);
}
logChange(e) {
this.setState({
[e.target.name]: e.target.value
});
}
handleEdit(event) {
event.preventDefault()
var data = {
childflagged: this.state.childflagged,
flagreason: this.state.flagreason,
id: this.state.id
}
fetch("/classroom/edit", {
method: 'POST',
headers: {
'Content-Type': 'application/json'
},
body: JSON.stringify(data)
}).then(function(response) {
if (response.status >= 400) {
throw new Error("Bad response from server");
}
return response.json();
}).then(function(data) {
if (data.affectedRows === 1) {
console.log('Edited Flag!'); //UPDATE COMPONENT FUNCTION OF SOME-SORT HERE
}
}).catch(function(err) {
console.log(err)
});
}
eventHandler(event, idx) {
let updatedArr = this.state.toggleIdxArray.slice();
let checkIdx = updatedArr.indexOf(idx);
if (checkIdx === -1) updatedArr.push(idx);
else updatedArr.splice(checkIdx, 1);
this.setState((prevState) => ({
toggleIdxArray: updatedArr
})
);
}
ponentDidMount() {
fetch("/classroom")
.then(res => res.json())
.then(classroom => this.setState({ classroom }))
}
render() {
return (
<div className="App">
<div className="wrapper">
<h1>Classroom</h1>
{this.state.classroom.map((classroom, idx) =>
<div className="child" key={classroom.id}>
<p className="detail">Child's Name: <span className={classroom.childflagged}>{classroom.childsname}</span> <i title={classroom.flagreason} className={'fa fa-flag flag' + classroom.childflagged}></i><i title={classroom.starreason} className={'fa fa-star star' + classroom.starstudent}></i></p>
<p className="detail">Parent's Phone Number: {classroom.parentsnumber}</p>
<div className="actionMenu">
<button className="flags" id={classroom.id} onClick={e => this.eventHandler(e, idx)}>Edit Flags</button>
<button className="results">View Results</button>
<button className="profile">View Profile</button>
</div>
<div className="actionForm">
<div>
<form id={"flagform" + classroom.id} className={this.state.toggleIdxArray.includes(idx) ? '' : 'hide'} onSubmit={this.handleEdit} method="post" autoComplete="no">
<input type="hidden" name="id" value={classroom.id} />
<input type="text" onChange={this.logChange} value={this.state.childflagged} name="childflagged" placeholder="Flag Type" />
<input type="text" onChange={this.logChange} value={this.state.flagreason} name="flagreason" placeholder="Reason For Flag" />
<input type="submit" onClick={() => this.state.id = classroom.id} name="submit" value="Save Changes" />
</form>
</div>
</div>
</div>
)}
</div>
</div>
);
}
}
export default App;
I am having trouble figuring out how I can update the ponent with the new data from the database after the edit function has been run and the update to the database was successful.
I have tried doing this.forceUpdate()
but that doesn't work. I heard about using this.setState()
but I don't know how I would update it using that with the database as my classroom state is a JSON array.
I have been thinking and google-ing long and hard for days without any luck, so any advice or help with this problem would be greatly appreciated.
P.S. I am using MySQL for my Database. I don't know if that matters.
App.js
import React, { Component } from 'react';
import './App.css';
class App extends Component {
constructor(props) {
super(props);
this.state = {
toggle: false,
toggleIdxArray: [],
classroom: [],
childflagged: '',
flagreason: '',
id: ''
};
this.eventHandler = this.eventHandler.bind(this);
this.logChange = this.logChange.bind(this);
this.handleEdit = this.handleEdit.bind(this);
}
logChange(e) {
this.setState({
[e.target.name]: e.target.value
});
}
handleEdit(event) {
event.preventDefault()
var data = {
childflagged: this.state.childflagged,
flagreason: this.state.flagreason,
id: this.state.id
}
fetch("/classroom/edit", {
method: 'POST',
headers: {
'Content-Type': 'application/json'
},
body: JSON.stringify(data)
}).then(function(response) {
if (response.status >= 400) {
throw new Error("Bad response from server");
}
return response.json();
}).then(function(data) {
if (data.affectedRows === 1) {
console.log('Edited Flag!'); //UPDATE COMPONENT FUNCTION OF SOME-SORT HERE
}
}).catch(function(err) {
console.log(err)
});
}
eventHandler(event, idx) {
let updatedArr = this.state.toggleIdxArray.slice();
let checkIdx = updatedArr.indexOf(idx);
if (checkIdx === -1) updatedArr.push(idx);
else updatedArr.splice(checkIdx, 1);
this.setState((prevState) => ({
toggleIdxArray: updatedArr
})
);
}
ponentDidMount() {
fetch("/classroom")
.then(res => res.json())
.then(classroom => this.setState({ classroom }))
}
render() {
return (
<div className="App">
<div className="wrapper">
<h1>Classroom</h1>
{this.state.classroom.map((classroom, idx) =>
<div className="child" key={classroom.id}>
<p className="detail">Child's Name: <span className={classroom.childflagged}>{classroom.childsname}</span> <i title={classroom.flagreason} className={'fa fa-flag flag' + classroom.childflagged}></i><i title={classroom.starreason} className={'fa fa-star star' + classroom.starstudent}></i></p>
<p className="detail">Parent's Phone Number: {classroom.parentsnumber}</p>
<div className="actionMenu">
<button className="flags" id={classroom.id} onClick={e => this.eventHandler(e, idx)}>Edit Flags</button>
<button className="results">View Results</button>
<button className="profile">View Profile</button>
</div>
<div className="actionForm">
<div>
<form id={"flagform" + classroom.id} className={this.state.toggleIdxArray.includes(idx) ? '' : 'hide'} onSubmit={this.handleEdit} method="post" autoComplete="no">
<input type="hidden" name="id" value={classroom.id} />
<input type="text" onChange={this.logChange} value={this.state.childflagged} name="childflagged" placeholder="Flag Type" />
<input type="text" onChange={this.logChange} value={this.state.flagreason} name="flagreason" placeholder="Reason For Flag" />
<input type="submit" onClick={() => this.state.id = classroom.id} name="submit" value="Save Changes" />
</form>
</div>
</div>
</div>
)}
</div>
</div>
);
}
}
export default App;
Share
Improve this question
edited Dec 1, 2018 at 14:09
Roy Scheffers
3,90811 gold badges33 silver badges36 bronze badges
asked Mar 14, 2018 at 22:20
JulianJulian
6573 gold badges9 silver badges33 bronze badges
3 Answers
Reset to default 4If you get back the data in a useful format after updating it on the database, just use setState with that data:
handleEdit(event) {
event.preventDefault()
var data = {
childflagged: this.state.childflagged,
flagreason: this.state.flagreason,
id: this.state.id
}
fetch("/classroom/edit", {
method: 'POST',
headers: {
'Content-Type': 'application/json'
},
body: JSON.stringify(data)
}).then(function(response) {
if (response.status >= 400) {
//Aside: because of the way promises work, this is the correct way to throw the error. Doing this you can catch it in the "catch" callback
Promise.reject(new Error("Bad response from server"));
}
return response.json();
}).then((data) => {
if (data.affectedRows === 1) {
console.log('Edited Flag!'); //UPDATE COMPONENT FUNCTION OF SOME-SORT HERE
//use data to set state
this.setState({ classroom: data });
}
}).catch(function(err) {
console.log(err)
});
}
I fixed this by using:
}.then((data) => {
console.log('Update Complete');
this.setState({ classroom: data })
})
You can trigger the ponent re-render by calling setState
inside your handleEdit
function, after the request finished and data returned:
handleEdit(event) {
....
fetch("/classroom/edit", {
method: 'POST',
headers: {
'Content-Type': 'application/json'
},
body: JSON.stringify(data)
}).then(function(response) {
if (response.status >= 400) {
throw new Error("Bad response from server");
}
return response.json();
}).then(function(data) {
if (data.affectedRows === 1) {
/** Calling setState here
* For example, if the data return is a classroom object,
* you can update the classroom property of the state to
* reflect it
*/
this.setState({classroom:[...this.state.classroom, data.classroom]});
}
})
....
}
Basically, a react ponent will re-render if:
- Its props change.
- Its
setState
function is called. - Its
forceUpdate
function is called.
More details about this topic can be found here: https://reactjs/docs/state-and-lifecycle.html
版权声明:本文标题:javascript - ReactJS check database for changes and update component without refreshing browser - Stack Overflow 内容由网友自发贡献,该文观点仅代表作者本人, 转载请联系作者并注明出处:http://www.betaflare.com/web/1743818743a2544403.html, 本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容,一经查实,本站将立刻删除。
发表评论