- SAP UI5 SimpleForm 控件的 adjustLab
- SAP UI5 按钮的类型和背景色设置例子
- SAP UI5 Page 控件的构造函数参数讲解
- SAP UI5 xml 视图里定义的控件,运行时如何创建其实例的
- SAP UI5 CSS 类 sapUiSmallMarginEn
- 关于 SAP UI5 XML 视图里控件事件处理函数名称中的 .
- SAP UI5 XML 视图在实例化时就已经确定了 Bindin
- SAP UI5 进阶 - XML 视图里定义的 UI 控件,运行
- ODataListBinding.filter 方法里 Filt
- SAP UI5 里的 Busy Indicator 控件使用概述
入口函数在 XMLTemplateProcessor 里:
data:image/s3,"s3://crabby-images/f3445/f344559fd82ea8ea3f075cb4514883ae22b0f9c8" alt=""
解析 xml 视图的源代码之后,调用 createRegularControls 进行实例创建:
data:image/s3,"s3://crabby-images/8c099/8c09973fffb616ef5a558384e02c7e1cb656087d" alt=""
这是我的 xml 视图源代码:
<mvc:View
controllerName="sap.ui5.walkthrough.controller.InvoiceList"
xmlns="sap.m"
xmlns:mvc="sap.ui.core.mvc">
<List
headerText="{i18n>invoiceListTitle}"
class="sapUiResponsiveMargin"
width="auto"
items="{invoice>/Invoices}">
<items>
<ObjectListItem
title="{invoice>Quantity} x {invoice>ProductName}"
number="{
parts: [{path: 'invoice>ExtendedPrice'}, {path: 'view>/currency'}],
type: 'sap.ui.model.type.Currency',
formatOptions: {
showMeasure: false
}
}"
numberUnit="{view>/currency}"/>
</items>
</List>
</mvc:View>
number 字段的绑定路径被解析了出来:
data:image/s3,"s3://crabby-images/d7443/d7443b634ee94ec6a6ecd9ce1cf52af3308353b2" alt=""
在 BindingParser 的代码里,上述字符串类型的绑定路径,被解析成了 json 对象:
data:image/s3,"s3://crabby-images/1078c/1078cedbf6aa504ab2897ce385ca1b30139e940f" alt=""
这里调用 sap.ui.model.type.Currency 的构造函数。
data:image/s3,"s3://crabby-images/59827/5982772c3fff07e1b98fca6b3758fa0d65d9c6d3" alt=""
我们再回过头来看 xml 视图的加载和解析过程。
在 XMLView.js 的 this._xContent 字段里,我们能找到 xml 视图的字符串格式的源代码。
data:image/s3,"s3://crabby-images/e94fd/e94fd98fe7ad1422b35baafafc5ffcf9bfdfe7cd" alt=""
在 XMLView.js 里根据字符串 _xContent 进行搜索,即可查到这个字段赋值的位置:
data:image/s3,"s3://crabby-images/ba1ab/ba1ab6484b259f6d62722fc72fef48cd3d9758df" alt=""
在代码第 607 行触发 xml 视图文件的加载:
data:image/s3,"s3://crabby-images/41d78/41d78ea9e3c94774c068ab72111920d323c0a118" alt=""
从方法名也能看出,xml 视图文件采用异步的方式进行加载:
loadResourceAsync(sResourceName).then(runPreprocessorsAsync).then(processView);
这里采用了 promise 异步编程模型:
function loadResourceAsync(sResourceName) {
return LoaderExtensions.loadResource(sResourceName, {async: true}).then(function(oData) {
return oData.documentElement; // result is the document node
});
}
LoaderExtensions.loadResource 执行异步加载 xml 视图文件的任务,加载成功的结果,通过输入参数 oData 传递到匿名回调函数内。
进入 loadResource 内部:
data:image/s3,"s3://crabby-images/b5b44/b5b4482d49639c2c581f1930dd4ddd915604ed01" alt=""
转发给 sap.ui.loader._.getModuleContent(sResourceName, mOptions.url);
data:image/s3,"s3://crabby-images/de37d/de37d9c522da65ac1eb394944ea6188406d5a606" alt=""
从缓存里读取。由于是第一次加载,缓存没有命中:
data:image/s3,"s3://crabby-images/eb1d5/eb1d5692efb1c4d376b5034f6c8bb30d985557c6" alt=""
最终还是用的 jQuery.ajax api 去加载的数据:
data:image/s3,"s3://crabby-images/b5050/b505055b93b4cedb2225932f7fc4e54f84ade719" alt=""
加载成功后,调用 335 行的 success 回调函数:
data:image/s3,"s3://crabby-images/234ad/234adeb3f182d44d7ad8e292e792af444f36337d" alt=""
ajax 请求的 dataType 字段值为 xml:
data:image/s3,"s3://crabby-images/137f9/137f993349112860e7547c263df1eb374dd3e7db" alt=""
加载成功的 xml document 对象:
data:image/s3,"s3://crabby-images/08f80/08f80b7e83af7d00604eeca2f012d4cb83426601" alt=""
调用 resolve 方法,将 xml document 传给 promise api 的 then 回调函数。
data:image/s3,"s3://crabby-images/b1427/b14275081d40934740702f7b7325bbec0feac339" alt=""
此处就开始递归地解析 xml document 的节点了:
data:image/s3,"s3://crabby-images/f29a1/f29a166f7a853f9c37a9580aa6506fa342b24c08" alt=""
function parseChildren(xmlNode, bRoot, bIgnoreToplevelTextNodes, pRequireContext) {
var children = xmlNode.childNodes;
for (var i = 0; i < children.length; i++) {
parseNode(children[i], bRoot, bIgnoreToplevelTextNodes, pRequireContext);
}
}
parseChildren 调用 parseNode.
更多Jerry的原创文章,尽在:"汪子熙":
data:image/s3,"s3://crabby-images/9d5ff/9d5fffea0e5ec258def5c77c56d04b5c06480366" alt=""
网友评论