The world’s smallest but powerful small program framework – more than 100 lines of code for global state management and cross-page communication
Making: github.com/dntzhang/we…
Small programs are known to make a paste of their own setData on the page or component, plus various father-son, grand-child, brother-in-law, sister-in-law, Cousins, and so on, and to make them very difficult to maintain and debug when combined with cross-page component communication. Although there are many technologies on the market stack to compile small program technology, but I feel no poke small program pain point. Small program from the components, development, debugging, release, gray scale, rollback, reporting, statistics, monitoring and recent cloud capabilities are very perfect, small program engineering is simply the model of the front end. As developer tools continue to be updated, you can imagine a future where component layouts don’t necessarily require writing code. So the biggest pain points are just state management and cross-page communication.
Inspired by the Omi framework and developed specifically for small programs, the WEStore global state management and cross-page communication framework keeps things under control, and thanks to the high performance JSON Diff library, the long list scrolling loading display becomes easy to navigate. In summary, it has the following features and advantages:
- The same neat Store API as Omi
- Super small code size (over 100 lines including JSON diff)
- Respect and obey the design of small programs (other translation libraries are equivalent to doing the opposite)
- This. update is better and smarter than native setData
- API
- Use guide
- Defining a global Store
- Create the page
- Data binding
- Update the page
- Create components
- Update the components
- Compare setData and update
- Synchronize data across pages
- debugging
- Best practices for very large small programs
- The principle of
- JSON Diff
- Update
- License
API
There are only three Westore apis, and the main path is simple:
- Create (Store, option) Creates a page
- Create (option) Creates a component
- This.update () updates the page or component
Use guide
Defining a global Store
export default {
data: {
motto: 'Hello World'.userInfo: {},
hasUserInfo: false.canIUse: wx.canIUse('button.open-type.getUserInfo'),
logs: []},logMotto: function () {
console.log(this.data.motto)
}
}
Copy the code
You don’t need to declare data properties on pages and components. If it is, it will be overwritten by object.assign on store.data. Then just change this.store.data.
Create the page
import store from '.. /.. /store'
import create from '.. /.. /utils/create'
const app = getApp()
create(store, {
onLoad: function () {
if (app.globalData.userInfo) {
this.store.data.userInfo = app.globalData.userInfo
this.store.data.hasUserInfo = true
this.update()
} else if (this.data.canIUse) {
app.userInfoReadyCallback = res= > {
this.store.data.userInfo = res.userInfo
this.store.data.hasUserInfo = true
this.update()
}
} else {
wx.getUserInfo({
success: res= > {
app.globalData.userInfo = res.userInfo
this.store.data.userInfo = res.userInfo
this.store.data.hasUserInfo = true
this.update()
}
})
}
}
})
Copy the code
The Page is created by passing just two arguments, the store is injected from the root node, and all child components can be accessed through this.store.
Data binding
<view class="container"> <view class="userinfo"> <button wx:if="{{! HasUserInfo && canIUse}}" open-type="getUserInfo" bindgetUserInfo ="getUserInfo"> </button> <block wx:else> <image bindtap="bindViewTap" class="userinfo-avatar" src="{{userInfo.avatarUrl}}" mode="cover"></image> <text class="userinfo-nickname">{{userInfo.nickName}}</text> </block> </view> <view class="usermotto"> <text class="user-motto">{{motto}}</text> </view> <hello></hello> </view>Copy the code
As before, we use store.data as the binding data source.
Update the page
this.store.data.any_prop_you_want_to_change = 'any_thing_you_want_change_to'
this.update()
Copy the code
Create components
import create from '.. /.. /utils/create'
create({
ready: function () {
//you can use this.store here
},
methods: {
//you can use this.store here}})Copy the code
Unlike creating a Page, you only need to pass a parameter to create a component, not a store, because it has already been injected from the root node.
Update the components
this.store.data.any_prop_you_want_to_change = 'any_thing_you_want_change_to'
this.update()
Copy the code
Compare setData and update
Take the log page of the official template example as an example:
this.setData({
logs: (wx.getStorageSync('logs') || []).map(log= > {
return util.formatTime(new Date(log))
})
})
Copy the code
After using WeStore:
this.store.data.logs = (wx.getStorageSync('logs') || []).map(log= > {
return util.formatTime(new Date(log))
})
this.update()
Copy the code
It looks like one statement becomes two, but the setData of this. Update call is after diff, so less data is passed.
Synchronize data across pages
With WeStore you don’t need to synchronize data across pages, you just need to focus on this.store.data and call update anywhere:
this.update()
Copy the code
debugging
console.log(getApp().globalData.store.data)
Copy the code
Best Practices for Very Large Applets (two scenarios)
Do not rule out the small program is made large may, contact the largest small program has 60+ pages, so how to manage? Here are two best practices.
- The first solution is to split the data of store into different modules, such as:
export default {
data: {
commonA: 'a'.commonB: 'b'.pageA: {
a: 1
xx: 'xxx'
},
pageB: {
b: 2.c: 3}},xxx: function () {
console.log(this.data)
}
}
Copy the code
- The second option, splitting the store data into different files and merging it into a single store, is exposed to the create method, such as:
a.js
export default {
data: {
a: 1
xx: 'xxx'
},
aMethod: function (num) {
this.data.a += num
}
}
Copy the code
b.js
export default {
data: {
b: 2.c: 3
},
bMethod: function () {}}Copy the code
store.js
import a from 'a.js'
import b from 'b.js'
export default {
data: {
commonNum: 1.commonB: 'b'.pageA: a.data
pageB: b.data
},
xxx: function () {
//you can call the methods of a or b and can pass args to them
console.log(a.aMethod(commonNum))
},
xx: function(){}}Copy the code
Of course, instead of splitting files or modules by page, you can split them by domain, which is very free, depending on the situation.
The principle of
-- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- - | enclosing the update | - > | json diff | - > |setData()-setData()... | - followed by black box (small program official implementation, but dom/apply little diff is certainly not) -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --Copy the code
Although it is the same store. Updata as Omi, there are essential differences. Here’s Omi:
-- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- - | enclosing the update | - > |setState | - > | JSX rerender | - > | vdom diff to apply the diff... | --------------- ------------------- ---------------- ------------------------------Copy the code
Both data-driven views, but different in nature for the following reasons:
- The small program store and DOM are not in the same environment. First, perform JSON diff in js environment, and then use the diff result to communicate through setData
- If omi is used on the Web, store and DOM are in the same environment. SetState directly drives the VDOM diff and then applies the diff result to the real DOM
JSON Diff
Let’s take a look at the capabilities of the JSON Diff library I developed specifically for WeStore:
diff({
a: 1.b: 2.c: "str".d: { e: [2, { a: 4 }, 5]},f: true.h: [1].g: { a: [1.2].j: 111}}, {a: [].b: "aa".c: 3.d: { e: [3, { a: 3}},f: false.h: [1.2].g: { a: [1.1.1].i: "delete" }, k: 'del'
})
Copy the code
Diff’s results are:
{ "a": 1."b": 2."c": "str"."d.e[0]": 2."d.e[1].a": 4."d.e[2]": 5."f": true."h": [1]."g.a": [1.2]."g.j": 111."g.i": null."k": null }
Copy the code
The Diff principle:
- Synchronizes all keys to the current store.data
- Recursively traverse all key values with PATH and result
export default function diff(current, pre) {
const result = {}
syncKeys(current, pre)
_diff(current, pre, ' ', result)
return result
}
Copy the code
Synchronizes the keys of the last round of state.data to detect deleted elements in array or deleted keys in OBj.
Small program setData
SetData is the most frequently used interface in small program development, and it is also the one most likely to cause performance problems. Before I cover some of the common misuses, let’s take a quick look at how setData works. The setData function is used to send data from the logical layer to the view layer (asynchronously) and change the corresponding value of this.data (synchronously).
The key can be given in the form of a data path to change an item in an array or a property of an object, such as array[2]. Message, A.B.C.D, and does not need to be defined in this.data. Such as:
this.setData({
'array[0].text':'changed data'
})
Copy the code
So the result of diff can be passed directly to setData, which is this.update.
How setData works
The view layer of the applet currently uses WebView as the rendering carrier, while the logic layer is run by a separate JavascriptCore. In terms of architecture, WebView and JavascriptCore are independent modules with no direct data sharing channels. Currently, the data transfer between the view layer and the logical layer is actually done through the evaluateJavascript provided on both sides. That is, the data transferred by the user needs to be converted into a string form and passed. At the same time, the converted data content is pieced into a JS script, and then passed to the independent environments on both sides by executing the JS script.
While the execution of evaluateJavascript is affected in many ways, data does not arrive at the view layer in real time.
Common setData operation errors:
- Go to setData frequently
- Every time setData is passed a lot of new data
- Background state page for setData
This is an official intercept. Using webStore’s this.update essentially diff and then execute a sequence of setData, so you can keep the number of data passed to a minimum each time. The first and third points are debatable violations since the transmission of data can be minimized.
Update
This distinguishes between an Update in a page and an Update in a component. The update in the page is instance collected in the onLoad event.
const onLoad = option.onLoad
option.onLoad = function () {
this.store = store
rewriteUpdate(this)
store.instances[this.route] = []
store.instances[this.route].push(this)
onLoad && onLoad.call(this)
}
Page(option)
Copy the code
The update in the component collects row instances in the Ready event:
const ready = store.ready
store.ready = function () {
this.page = getCurrentPages()[getCurrentPages().length - 1]
this.store = this.page.store;
this.setData.call(this.this.store.data)
rewriteUpdate(this)
this.store.instances[this.page.route].push(this)
ready && ready.call(this)
}
Component(store)
Copy the code
The implementation of rewriteUpdate is as follows:
function rewriteUpdate(ctx){
ctx.update = (a)= > {
const diffResult = diff(ctx.store.data, originData)
for(let key in ctx.store.instances){
ctx.store.instances[key].forEach(ins= > {
ins.setData.call(ins, diffResult)
})
}
for (let key in diffResult) {
updateOriginData(originData, key, diffResult[key])
}
}
}
Copy the code
Star & Fork
Making: github.com/dntzhang/we…
License
MIT @dntzhang