model

补充信息

默认情况下,本节所有 API 的导出包名为:@modern-js/runtime/model

如果是在 Modern.js 以外单独集成 Reduck,导出包名为:@modern-js-reduck/react

提示

Reduck 原始类型较为复杂,以下涉及类型定义的地方,展示的是简化后的类型信息。原始类型见 model

model

用于创建管理应用状态的 Model。

function model(name: string): { define: function }

  • name:string,创建的 Model 的唯一标识。
示例
model('foo');

define

用于定义 Model 的详细结构,支持传入一个对象类型或函数类型的参数。

对象类型

function define(modelDesc: ModelDesc): Model;

  • modelDesc: ModelDesc,对 Model 结构的定义,包含 statecomputedactionseffects 等属性。
示例
const fooModel = model('foo').define({
  state: 'foo',
  computed: {
    cFoo: state => `c${state}`,
  },
  actions: {
    setState: (state, value) => {
      return value;
    },
  },
  effects: {
    loadState: async () => {
      // 从服务端获取 state
    },
  },
});

函数类型

function define((context: Context, utils: Utils) => ModelDesc): Model;

  • context: Context,Reduck 上下文对象,可以获取底层的 store 对象。store 除支持 Redux Store 的所有 API 以外,还挂载了用于消费 Model 的 use 的方法,和用于卸载 Model 的 unmount 方法。
  • utils: Utils,定义 Model 时,常用的工具函数:useonMountuse 作用同 store 对象上的 useonMount 是 Model 挂载后的钩子函数。
interface Utils {
  use: UseModel;
  onMount: OnMountHook;
}

interface Context {
  store: ReduxStore & {
    use: UseModel;
    unmount: (model: Model) => void;
  };
}

如通过 use,可以获取 Model 自身及其它 Model 的 stateactions

示例
const fooModel = model('foo').define(() => {
  return {
    state: 'foo',
    actions: {
      setState: (state, value) => {
        return value;
      },
    },
  };
});

const barModel = model('bar').define((_, { use }) => {
  return {
    state: 'bar',
    effects: {
      syncFoo() {
        const [state, actions] = use(fooModel);
        actions.setState(state);
      },
    },
  };
});

参数

ModelDesc.state

定义 Model 的状态。技术上,支持任意类型的 State,但是实践中建议使用可进行 JSON 序列化的类型。

interface ModelDesc {
  state: any;
}

ModelDesc.actions

定义 Model 的 Actions。Actions 的函数类型为:

interface ModelDesc {
  actions: {
    [actionKey: string]: (state: State, payload: any) => State | void;
  };
}

Reduck 内部集成了 immer,可以直接原始的 state,当 Action 没有显式返回值时,Reduck 内部会返回修改过的新的 State 对象。

ModelDesc.computed

定义 Model 的衍生状态。衍生状态的定义支持两种类型:

  1. 只依赖 Model 自身的状态
interface ModelDesc {
  computed: {
    [computedKey: string]: (state: State) => any;
  };
}
  1. 依赖其他 Model 的状态
interface ModelDesc {
  computed: {
    [computedKey: string]: [
      ...models: Model[],
      (state: State, ...args: ModelState[]) => any,
    ];
  };
}
示例
const fooModel = model('foo').define({
  state: 'foo',
});

const barModel = model('bar').define({
  state: 'bar',
  computed: {
    combineFoo: [fooModel, (state, fooState) => state + fooState],
  },
});

ModelDesc.effects

定义 Model 的 Effects。Effects 中定义的函数类型为:

interface ModelDesc {
  effects: {
    [effectKey: string]: (...args: any[]) => any;
  };
}
示例
const fooModel = model('foo').define((context, { use }) => ({
  state: 'foo',
  effects: {
    persist() {
      const [state] = use(fooModel);
      localStorage.setItem('state', state);
    },
  },
}));
更多参考