Làm quen với new context API React
This post hasn't been updated for 6 years
Context của React thì ngày xưa cũng đã có. Nhưng trên docs cũng có khuyến cáo là nếu được thì không nên sử dụng vì chúng tôi sẽ sớm viết lại context. Gần đây khi phiên bản 16.3 được ra mắt thì API context mới cũng được ra mắt. Cùng thử dùng new context nào.
Khi nào thì dùng context
Dùng context khi bạn muốn share data
đến tất cả các component con. Như là thông tin user đã đăng nhập, ngôn ngữ... Trên docs có khuyến cao "Nếu mục đích chỉ là tránh truyền props qua nhiều tầng thì không nên dùng context mà hãy dùng component composition thì sẽ đơn giản hơn"
Sử dụng context API
React.createContext
const {Provider, Consumer} = React.createContext(defaultValue);
Khi React render Consumer
sẽ đọc giá trị của Provider
gần nhất với nó. defaultValue
là giá trị mặc định của Consumer
Provider
<Provider value={/* some value */}>
Value được truyền vào Provider
sẽ được truyền đến tất cả các Consumer
con. Một Provider
có thể có nhiều Consumer
. Provider
có thể lồng nhau.
Consumer
<Consumer>
{value => /* render something based on the context value */}
</Consumer>
Consumer
là component sẽ được cập nhật khi value
truyền vào Provider
. "children" của Consumer
là một hàm, có tham số là value
được truyền vào Provider
và trả về một component. Xem thêm render props
Consumer
sẽ được render lại khi mà value
thay đổi. Chứ không phụ thuộc vào hàm shouldComponentUpdate
. Tức là cho dù shouldComponentUpdate
trả về false nhưng mà value
có cập nhật thì Consumer
vẫn render lại.
Demo
Cùng bắt đầu dùng context API với một ví dụ nhỏ. Bạn có một login modal, Modal này có thể được mở từ bất kỳ đâu, click vào nút login trên header, click vào button trên một component nào đó khi chưa đăng nhập chưa đăng nhập... Cái này sẽ được giải quyết đơn giản bằng context.
Ở component App.js thì sẽ có state isShowModal
và hàm showModal
để điều khiển trạng thái mở/đóng của modal.
import React from "react";
import ReactDOM from "react-dom";
import Modal from "./components/Modal";
import "./styles.css";
import { Provider } from "./contexts/modal";
import ChildComponent from "./components/ChildComponent";
class App extends React.Component {
state = {
isShowModal: false
};
showModal = (bool = true) => {
this.setState({ isShowModal: bool });
};
render() {
const { isShowModal } = this.state;
return (
<Provider
value={{
isShowModal: isShowModal,
showModal: this.showModal
}}
>
<div className="App">
<h1>Try new context API</h1>
<button onClick={this.showModal}>Open modal from parent</button>
<ChildComponent />
{isShowModal && <Modal />}
</div>
</Provider>
);
}
}
const rootElement = document.getElementById("root");
ReactDOM.render(<App />, rootElement);
Cho dễ quản lý code thì mình tạo riêng 1 file modal.js
để tạo context và export ra component khác dùng. hàm withModal
mình sử dụng Higher order component để có thể tái sử dụng Consumer
ở nhiều chỗ.
import React from "react";
const { Provider, Consumer } = React.createContext({
isShowModal: false,
showModal: () => {}
});
const withModal = Component => props => {
return (
<Consumer>
{({ showModal }) => <Component {...props} showModal={showModal} />}
</Consumer>
);
};
export { Provider, withModal };
Các bạn có thể xem toàn bộ code ở đây https://codesandbox.io/s/o91vrxlywy
All Rights Reserved