一个轻量性能好的微信小程序的状态管理库(已有多个线上项目)
该库我已不再维护,但需要的话可以提pr或者各自fork单独修改
工作中在开发小程序的时候,发现组件间通讯或跨页通讯会把程序搞得混乱不堪,变得极难维护和扩展,setData 的性能不是很好,浪费很多的资源,所以封装了一个 wenaox 作为使用,后决定开源出来给大家使用 如果觉得有什么问题或者建议,欢迎提 issue 和 pr,觉得不错,可以给个 star,鼓励一下 2333
虽然可以直接引入,但是建议使用 npm 安装开发,将会很方便
npm i -S wenaox
or
yarn add wenaox
关于小程序如何构建 npm
新建一个 store.js
import { Store } from "wenaox";
//数据
const state = {
count: 0,
};
//方法
const methods = {
//修改state的方法(只允许通过syncs的方法进行修改)
syncs: {
addCount(state, payload) {
state.count = state.count + 1;
},
},
//包含副作用的方法
asyncs: {
asyncAddCount(payload, rootState) {
setTimeout(() => {
this.addCount(c);
});
},
},
};
//注册store
const store = new Store({ state, methods });
store 中的 state 和 methods 打印如下:
{
"state": { "count": 0 },
"methods": { "addCount": fn, "asyncAddCount": fn }
//略
}
在中大型小程序中的实践中,共享的状态和方法将会很多,如果全部都定义在一起会很混乱,所以提供一个多 contro 的机制,可以根据页面或者功能来进行划分
// 下面是多 contro 的注册写法
const store = new Store({ controA: { state, methods } });
将会 Store 对 store 的 state 和 methods 通过 contro 的变量名进行一个细化区分:
{
"state": { "controA": { "count": 0 } },
"methods": { "controA": { "addCount": fn, "asyncAddCount": fn } }
//略
}
//app.js
import { Provider } from "wenaox";
import store from "xxx路径/store";
const appConfig = {
//some config
};
App(Provider(store)(appConfig));
-在 page.js 中连接 state 和 methods
import { orm } from 'wenaox';
// 返回需要的state和methods
const mapState = state => ({
count: state.count,
});
const mapMethods = methods => ({
addCount: methods.addCount,
asyncAddCount: methods.asyncAddCount,
});
const pageConfig = {
//some config
};
// 使用orm连接
Page(orm(mapState, mapMethods)(pageConfig));
<view class="count">count</view>
<button bindtap="addCount">count + 1</button>
<button bindtap="asyncAddCount">async count + 1</button>
点击按钮就会发生变化!一个简单的计数器!
由于小程序中的属性没有分辨组件还是 page 页面所以我另外写了一个对 自定义组件 的方法就是 ormComp
所以在自定义组件中使用的话仅仅只需要 js 中的 orm 替换成 ormComp 就可以了
Component(ormComp(mapState, mapMethods)(compConfig));
使用 wenaox 你不用关心跨页数据同步,任何地方的修改,都会同步到使用到的地方[仅限于正在显示的页面/组建]
这是因为 wenaox 在页面栈中 hide 的页面不执行更新,而是等待 onshow 事件才重新进行更新,这是为了更好的性能!
在头部引入 regeneratorRuntime 即可使用 async/await
import { regeneratorRuntime } from 'wenaox'
const methods = {
// ...略
asyncs: {
async asyncAddCount(payload, rootState) {
await new Promise(resolve => {
setTimeout(resolve, 2000);
});
this.addCount(1);
},
},
};
而在使用 async/await 之后自动会生成一个 loading 变量
{
"loading": state.loading.asyncAddCount
}
可以在 mapState 中引入,再也不用手动写 loading 了!! 当然你不用的话,你不引入 对应的 loading 变量的话,wenaox 也不会再对 对应的 loading 进行更新,避免性能损失
wenaox 为了方便,提供了中间件的一个开发和使用,下面是一个 wenaox 的一个 log 的中间件
保证流动完所有的中间件才进行更新数据
const log = (store) => (next) => (fn, payload) => {
console.group("改变前:", store.state);
next(fn, payload);
console.log("改变后:", store.state);
console.groupEnd();
};
小程序是可以自定义 tabbar 的,通过 wenaox 可以随时更改底部的 tab 的数量以及跳转的方法
所有的具体在下面的例子中也有展示
计数器
MIT