【引自第九程序的博客】本文探讨一下小程序的 view 模块和 service 模块是如何构成的。
成都创新互联公司主要从事网站制作、网站建设、网页设计、企业做网站、公司建网站等业务。立足成都服务浦口,十多年网站建设经验,价格优惠、服务专业,欢迎来电咨询建站服务:13518219792
打开微信 web 开发者工具,然后输入 openVendor() 便会打开 WeappVendor这个目录,这里包含了 view 模块和 service 模块使用的几个核心文件:
view 页面详解
view 页面的 template 如下:
其中 会在 dev 模式开启后被替换为一个时间锚点,例如:
会被 WAWebview.js 内代码替换
到 之间暂时没有被使用到
会被 wcc 命令生成后的 js 代码替换
除了上面这些,页面上还会被插入页面和应用的 style 标签,如:
这里的 wxss 文件包含的是原始 wxss 文件转换后的 css
以及生成 DOM 的启动脚本:
WAWebview.js 文件中的各个模块(行号为 jsbeautify 之后代码行号,开发者工具版本:092300):
从页面 data 到 dom 的主要流程如下:
- var vtree
- var rootNode
- document.addEventListener("generateFuncReady", function(e) {
- var generateFunc = e.detail.generateFunc;
- wx.onAppDataChange(function(obj) {
- // 合并 data 到现有 data
- DataStore.setData(obj.data)
- // 生成 virtual dom 的 javascript plain object
- var props = generateFunc(DataStore.getData())
- // ***次渲染
- if (obj.options.firstRender) {
- vtree = createVirtualTree(props, true)
- rootNode = vtree.render()
- rootNode.replaceDocumentElement(document.body)
- wx.initReady()
- } else {
- var other_vtree = createVirtualTree(props, false)
- var patches = vtree.diff(other_vtree)
- patches.apply(rootNode)
- vtree = other_vtree
- document.dispatchEvent(new CustomEvent("pageReRender", {}));
- }
- })
- })
上面的 DataStore 对象提供合并和获取当前页面 data 对象的功能,其实现如下:
- var DataStore = (function() {
- var data = {}
- return {
- getData: function() {
- return data
- },
- setData: function(e) {
- for (var t in e) {
- for (var n = (0, parsePath)(t), o = data, a = void 0, s = void 0, c = 0; c < n.length; c++) Number(n[c]) === n[c] && Number(n[c]) % 1 === 0 ? Array.isArray(o) || (a[s] = [], o = a[s]) : "[object Object]" !== Object.prototype.toString.call(o) && (a[s] = {}, o = a[s]), s = n[c], a = o, o = o[n[c]];
- a && (a[s] = e[t])
- }
- }
- }
- })()
- // 解析 key 为 data 内对象的路径字符串
- function parsePath(e) {
- for (var t = e.length, n = [], i = "", r = 0, o = !1, a = !1, s = 0; s < t; s++) {
- var c = e[s];
- if ("\\" === c) s + 1 < t && ("." === e[s + 1] || "[" === e[s + 1] || "]" === e[s + 1]) ? (i += e[s + 1], s++) : i += "\\";
- else if ("." === c) i && (n.push(i), i = "");
- else if ("[" === c) {
- if (i && (n.push(i), i = ""), 0 === n.length) throw new Error("path can not start with []: " + e);
- a = !0, o = !1
- } else if ("]" === c) {
- if (!o) throw new Error("must have number in []: " + e);
- a = !1, n.push(r), r = 0
- } else if (a) {
- if (c < "0" || c > "9") throw new Error("only number 0-9 could inside []: " + e);
- o = !0, r = 10 * r + c.charCodeAt(0) - 48
- } else i += c
- }
- if (i && n.push(i), 0 === n.length) throw new Error("path can not be empty");
- return n
- }
可以看到,每次 data 变化之后,小程序就会开始整个页面的 diff patch 过程。
对于原生实现的组件, exparser 会在监视到数据变化后发送对应事件到 WeixinJSBridge。
service 页面详解
service 页面会被被拼接为以下的样子:
除了配置和开发者编写的页面、app.js,页面还在加载了 asdebug.js 和 WAService.js 两个文件。
asdebug.js 文件位于 nwjs 项目目录下,路径为app/dist/weapp/appservice/asdebug.js。 它包含了两个部分,一个是 WeixinJSBridge 针对 service 模块的实现,另一块是一些方便命令使用的接口, 例如:help() 会告诉你一些可用的函数:
该文件只会在开发者工具内被引入,如果小程序在微信内运行,应该会由微信底层提供 WeixinJSBridge。
WAService 负责 service 模块的一些核心逻辑,它包含以下部分 (行号为 jsbeautify 之后代码行号,开发者工具版本:092300):
现在的 WAService 还有有很多地方依赖 window 对象,所以很有可能它在微信中和开发者工具内一样,依然运行于 webview 标签之内。
网站题目:微信小程序架构分析(中)
文章起源:http://www.shufengxianlan.com/qtweb/news3/239703.html
网站建设、网络推广公司-创新互联,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等
声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 创新互联