From 87d5bd9515f0c233a6aff61cc957be128f8585e5 Mon Sep 17 00:00:00 2001 From: geekact Date: Tue, 6 Dec 2022 23:12:08 +0800 Subject: [PATCH] feat: lazy creating reducer (#24) --- src/model/defineModel.ts | 16 +++++++++------ src/model/enhanceAction.ts | 2 ++ src/model/lazyLoad.ts | 5 +++++ src/model/useDefined.ts | 5 ++++- src/model/useModel.ts | 5 ++++- src/persist/PersistItem.ts | 2 ++ src/store/modelStore.ts | 41 ++++++++++++++++++++++++++++++-------- test/lifecycle.test.ts | 5 ++--- test/middleware.test.ts | 3 +++ 9 files changed, 65 insertions(+), 19 deletions(-) create mode 100644 src/model/lazyLoad.ts diff --git a/src/model/defineModel.ts b/src/model/defineModel.ts index d11f1ab..67a0430 100644 --- a/src/model/defineModel.ts +++ b/src/model/defineModel.ts @@ -24,6 +24,7 @@ import type { import { isFunction } from '../utils/isType'; import { Unsubscribe } from 'redux'; import { original } from 'immer'; +import { lazyLoad } from './lazyLoad'; export const defineModel = < Name extends string, @@ -95,6 +96,7 @@ export const defineModel = < const getState = (obj: T): T & GetState => { return defineGetter(obj, 'state', () => { + lazyLoad(uniqueName); const state = modelStore.getState()[uniqueName]; return depsCollector.active ? new ObjectDeps(modelStore, uniqueName).start(state) @@ -203,7 +205,9 @@ export const defineModel = < } } - if (events) { + const onLazyLoaded = () => { + if (!events) return; + const { onInit, onChange, onDestroy } = events; const eventCtx: EventCtx = Object.assign( composeGetter({ name: uniqueName }, getState), @@ -211,9 +215,8 @@ export const defineModel = < enhancedMethods.internal, ); - modelStore.onInitialized().then(() => { - const subscriptions: Unsubscribe[] = []; - + const subscriptions: Unsubscribe[] = []; + { if (onChange) { let prevState = eventCtx.state; subscriptions.push( @@ -243,8 +246,8 @@ export const defineModel = < } onInit && onInit.call(eventCtx); - }); - } + } + }; ModelStore.appendReducer.call( modelStore, @@ -254,6 +257,7 @@ export const defineModel = < initialState: parseState(initialStateStr), allowRefresh: !skipRefresh, }), + onLazyLoaded, ); const model: InternalModel = diff --git a/src/model/enhanceAction.ts b/src/model/enhanceAction.ts index ca95f48..6c904ac 100644 --- a/src/model/enhanceAction.ts +++ b/src/model/enhanceAction.ts @@ -1,6 +1,7 @@ import type { PreModelAction } from '../actions/model'; import { modelStore } from '../store/modelStore'; import { toArgs } from '../utils/toArgs'; +import { lazyLoad } from './lazyLoad'; import type { ActionCtx } from './types'; export interface EnhancedAction { @@ -26,6 +27,7 @@ export const enhanceAction = ( }; const fn: EnhancedAction = function () { + lazyLoad(modelName); return modelStore.dispatch>({ type: actionType, model: modelName, diff --git a/src/model/lazyLoad.ts b/src/model/lazyLoad.ts new file mode 100644 index 0000000..5292d5f --- /dev/null +++ b/src/model/lazyLoad.ts @@ -0,0 +1,5 @@ +import { modelStore, ModelStore } from '../store/modelStore'; + +export function lazyLoad(modelName: string): void { + ModelStore.lazyLoad.call(modelStore, modelName); +} diff --git a/src/model/useDefined.ts b/src/model/useDefined.ts index 5f5ee7e..47c4b97 100644 --- a/src/model/useDefined.ts +++ b/src/model/useDefined.ts @@ -3,6 +3,7 @@ import { DestroyLodingAction, DESTROY_LOADING } from '../actions/loading'; import { loadingStore } from '../store/loadingStore'; import { ModelStore, modelStore } from '../store/modelStore'; import { cloneModel } from './cloneModel'; +import { lazyLoad } from './lazyLoad'; import { HookModel as HookModel, Model } from './types'; let nameCounter = 0; @@ -25,7 +26,9 @@ export const useDefined = < : useDevName(modelName, initialCount, new Error()); const hookModel = useMemo(() => { - return cloneModel(uniqueName, globalModel); + const model = cloneModel(uniqueName, globalModel); + lazyLoad(uniqueName); + return model; }, [uniqueName]); return hookModel as any; diff --git a/src/model/useModel.ts b/src/model/useModel.ts index 5bd8a6e..bd7bfb8 100644 --- a/src/model/useModel.ts +++ b/src/model/useModel.ts @@ -4,6 +4,7 @@ import type { HookModel, Model } from './types'; import { toArgs } from '../utils/toArgs'; import { useModelSelector } from '../redux/useSelector'; import { isFunction, isString } from '../utils/isType'; +import { lazyLoad } from './lazyLoad'; /** * hooks新旧数据的对比方式: @@ -212,7 +213,9 @@ export function useModel(): any { const reducerNames: string[] = []; for (i = 0; i < modelsLength; ++i) { - reducerNames.push(models[i]!.name); + const modelName = models[i]!.name; + lazyLoad(modelName); + reducerNames.push(modelName); } return useModelSelector((state: Record) => { diff --git a/src/persist/PersistItem.ts b/src/persist/PersistItem.ts index 3b158f7..5c0264b 100644 --- a/src/persist/PersistItem.ts +++ b/src/persist/PersistItem.ts @@ -1,4 +1,5 @@ import type { StorageEngine } from '../engines'; +import { lazyLoad } from '../model/lazyLoad'; import type { InternalModel, Model, ModelPersist } from '../model/types'; import { isObject, isString } from '../utils/isType'; import { parseState, stringifyState } from '../utils/serialize'; @@ -96,6 +97,7 @@ export class PersistItem { this.key = keyPrefix + key; models.forEach((model) => { + lazyLoad(model.name); const { decode = defaultDecodeFn, maxAge: customMaxAge = maxAge, diff --git a/src/store/modelStore.ts b/src/store/modelStore.ts index 75af068..9b1c200 100644 --- a/src/store/modelStore.ts +++ b/src/store/modelStore.ts @@ -28,7 +28,7 @@ interface CreateStoreOptions { persist?: PersistOptions[]; } -export class ModelStore extends StoreBasic> { +export class ModelStore> extends StoreBasic { public topic: Topic<{ init: []; ready: []; @@ -38,6 +38,10 @@ export class ModelStore extends StoreBasic> { protected _isReady: boolean = false; protected consumers: Record = {}; protected reducerKeys: string[] = []; + protected lazyKeys: Record< + string, + { consumer: Reducer; onComplete: () => void } + > = {}; /** * @protected */ @@ -125,12 +129,16 @@ export class ModelStore extends StoreBasic> { this.topic.publish('unmount'); } - onInitialized(): Promise { + onInitialized(maybeSync?: () => void): Promise { return new Promise((resolve) => { if (this._isReady) { + maybeSync?.(); resolve(); } else { - this.topic.subscribeOnce('ready', resolve); + this.topic.subscribeOnce('ready', () => { + maybeSync?.(); + resolve(); + }); } }); } @@ -189,11 +197,7 @@ export class ModelStore extends StoreBasic> { }; } - public static appendReducer( - this: ModelStore, - key: string, - consumer: Reducer, - ): void { + protected appendReducer(key: string, consumer: Reducer) { const store = this.origin; const consumers = this.consumers; const exists = store && consumers.hasOwnProperty(key); @@ -203,6 +207,27 @@ export class ModelStore extends StoreBasic> { store && !exists && store.replaceReducer(this.reducer); } + public static appendReducer( + this: ModelStore, + key: string, + consumer: Reducer, + onComplete: () => void, + ): void { + this.lazyKeys[key] = { + consumer, + onComplete, + }; + } + + public static lazyLoad(this: ModelStore, key: string) { + if (this.lazyKeys.hasOwnProperty(key)) { + const { consumer, onComplete } = this.lazyKeys[key]!; + delete this.lazyKeys[key]; + this.appendReducer(key, consumer); + this.onInitialized(onComplete); + } + } + public static removeReducer(this: ModelStore, key: string): void { const store = this.origin; const consumers = this.consumers; diff --git a/test/lifecycle.test.ts b/test/lifecycle.test.ts index d72b3b4..ed1a2b0 100644 --- a/test/lifecycle.test.ts +++ b/test/lifecycle.test.ts @@ -1,4 +1,5 @@ import { cloneModel, defineModel, engines, store } from '../src'; +import { lazyLoad } from '../src/model/lazyLoad'; import { PersistSchema } from '../src/persist/PersistItem'; import { ModelStore } from '../src/store/modelStore'; @@ -130,10 +131,8 @@ describe('onChange', () => { }, }, }); - model.plus(); - model.minus(); - expect(testMessage).toBe(''); + lazyLoad(model.name); await store.onInitialized(); expect(testMessage).toBe('onInit-prev-0-next-2-'); diff --git a/test/middleware.test.ts b/test/middleware.test.ts index c0528dc..f4a3329 100644 --- a/test/middleware.test.ts +++ b/test/middleware.test.ts @@ -1,5 +1,6 @@ import { defineModel, getLoading, store } from '../src'; import { DestroyLodingAction, DESTROY_LOADING } from '../src/actions/loading'; +import { lazyLoad } from '../src/model/lazyLoad'; import { loadingStore } from '../src/store/loadingStore'; import { basicModel } from './models/basicModel'; import { complexModel } from './models/complexModel'; @@ -14,6 +15,8 @@ afterEach(() => { test('dispatch the same state should be intercepted', () => { const fn = jest.fn(); + lazyLoad(basicModel.name); + lazyLoad(complexModel.name); const unsubscribe = store.subscribe(fn); expect(fn).toHaveBeenCalledTimes(0);