gunakan kait atau HOC
Dengan menggunakan kait atau pola HOC (komponen pesanan lebih tinggi) , Anda dapat memiliki pembaruan otomatis saat toko Anda berubah. Ini adalah pendekatan yang sangat ringan tanpa kerangka kerja.
useStore Hooks cara untuk menangani pembaruan toko
interface ISimpleStore {
on: (ev: string, fn: () => void) => void;
off: (ev: string, fn: () => void) => void;
}
export default function useStore<T extends ISimpleStore>(store: T) {
const [storeState, setStoreState] = useState({store});
useEffect(() => {
const onChange = () => {
setStoreState({store});
}
store.on('change', onChange);
return () => {
store.off('change', onChange);
}
}, []);
return storeState.store;
}
withStores HOC menangani pembaruan toko
export default function (...stores: SimpleStore[]) {
return function (WrappedComponent: React.ComponentType<any>) {
return class WithStore extends PureComponent<{}, {lastUpdated: number}> {
constructor(props: React.ComponentProps<any>) {
super(props);
this.state = {
lastUpdated: Date.now(),
};
this.stores = stores;
}
private stores?: SimpleStore[];
private onChange = () => {
this.setState({lastUpdated: Date.now()});
};
componentDidMount = () => {
this.stores &&
this.stores.forEach((store) => {
// each store has a common change event to subscribe to
store.on('change', this.onChange);
});
};
componentWillUnmount = () => {
this.stores &&
this.stores.forEach((store) => {
store.off('change', this.onChange);
});
};
render() {
return (
<WrappedComponent
lastUpdated={this.state.lastUpdated}
{...this.props}
/>
);
}
};
};
}
Kelas SimpleStore
import AsyncStorage from '@react-native-community/async-storage';
import ee, {Emitter} from 'event-emitter';
interface SimpleStoreArgs {
key?: string;
defaultState?: {[key: string]: any};
}
export default class SimpleStore {
constructor({key, defaultState}: SimpleStoreArgs) {
if (key) {
this.key = key;
// hydrate here if you want w/ localState or AsyncStorage
}
if (defaultState) {
this._state = {...defaultState, loaded: false};
} else {
this._state = {loaded: true};
}
}
protected key: string = '';
protected _state: {[key: string]: any} = {};
protected eventEmitter: Emitter = ee({});
public setState(newState: {[key: string]: any}) {
this._state = {...this._state, ...newState};
this.eventEmitter.emit('change');
if (this.key) {
// store on client w/ localState or AsyncStorage
}
}
public get state() {
return this._state;
}
public on(ev: string, fn:() => void) {
this.eventEmitter.on(ev, fn);
}
public off(ev: string, fn:() => void) {
this.eventEmitter.off(ev, fn);
}
public get loaded(): boolean {
return !!this._state.loaded;
}
}
Cara Penggunaan
Dalam hal kait:
// use inside function like so
const someState = useStore(myStore);
someState.myProp = 'something';
Dalam kasus HOC:
// inside your code get/set your store and stuff just updates
const val = myStore.myProp;
myOtherStore.myProp = 'something';
// return your wrapped component like so
export default withStores(myStore)(MyComponent);
PASTIKAN
Untuk mengekspor toko Anda sebagai singleton untuk mendapatkan manfaat dari perubahan global seperti:
class MyStore extends SimpleStore {
public get someProp() {
return this._state.someProp || '';
}
public set someProp(value: string) {
this.setState({...this._state, someProp: value});
}
}
// this is a singleton
const myStore = new MyStore();
export {myStore};
Pendekatan ini sangat sederhana dan berfungsi untuk saya. Saya juga bekerja di tim-tim besar dan menggunakan Redux dan MobX dan menemukan itu bagus juga tetapi hanya banyak boilerplate. Saya pribadi suka pendekatan saya sendiri karena saya selalu membenci banyak kode untuk sesuatu yang sederhana ketika Anda membutuhkannya.
this.forceUpdate()
adalah solusi yang tepat sedangkan sisa semua jawaban dan beberapa komentar yang melawan menggunakanforceUpdate()
. Akankah itu baik-baik saja untuk mengatakan bahwa pertanyaannya belum mendapatkan solusi / jawaban yang tepat?