SAP UI5 setProperty 的执行逻辑单步调式和分析
按照 Jerry 这篇文章介绍的代码,运行之后,给类型为 sap.ui.model.type.Currency
的字段设置一个非 number
类型的值之后,触发该数据类型自带的数据校验机制,显示 Enter a number
的错误消息。
SAP UI5 应用开发教程之四十六 - 使用 Message Manager 实现开箱即用的验证(Validation)信息抛出
本文介绍 SAP UI5 ManagedObject
的 setProperty
方法的执行原理。
首先明确,什么是 ManagedObject
?
托管属性表示 ManagedObject 的状态。 它们可以存储简单数据类型(如“字符串”或“整数”)的单个值。 它们具有名称(例如“size”)和获取当前值(getSize)或设置新值(setSize)的方法。 当通过调用 setter 修改属性时,ManagedObject 被标记为无效。 可以使用 #bindProperty 方法将托管属性绑定到 sap.ui.model.Model 中的属性。 对模型属性的更新将自动反映在托管属性中,并且 - 如果 TwoWay 数据绑定处于活动状态,则对托管属性的更改将反映在模型中。 可以通过调用#unbindProperty 删除现有绑定。
data:image/s3,"s3://crabby-images/cce96/cce967c34ccd12e06eb80d034a04ac8640c0f385" alt=""
当 input 控件失去 focus 之后,触发 onsapfocusleave
,这个方法调用 onChange
:
data:image/s3,"s3://crabby-images/1b898/1b89893d10dd37cd37e8abe6acec549c4ea2a99e" alt=""
传入的输入参数,event
类型为 sapfocusleave
:
data:image/s3,"s3://crabby-images/dc4cd/dc4cdd60e43c5fe230ef152c37bcd03c4aad12e8" alt=""
如果当前最新的值和之前的值 getLastValue()
不相等,则调用 this.setValue
:
data:image/s3,"s3://crabby-images/dbf5f/dbf5f876aeb8ab39f61c14edce582eed66c26489" alt=""
Input.setValue 调用 InputBase.setValue:
data:image/s3,"s3://crabby-images/03e6f/03e6fdd58e4b6a795955d6aa62cbf4dac445aa0b" alt=""
setValue 最终调用 setProperty,属性名称为 value
:
data:image/s3,"s3://crabby-images/6ca98/6ca9839b268e453d264d1f47c71623c47bd59987" alt=""
正式进入 setProperty 方法,首先从 mProperties
里得到修改之前的值:
data:image/s3,"s3://crabby-images/6a22a/6a22a67b2ce1f9842f4d218593b57eba601e4715" alt=""
然后进行 validateProperty
:
data:image/s3,"s3://crabby-images/8c8bb/8c8bb6d48e72207262c9532cce94532ef35453a6" alt=""
validateProperty 内部,首先从 this.getMetadata().getManagedProperty(sPropertyName)
读取该 value 属性的 元数据
:
data:image/s3,"s3://crabby-images/286ba/286baa3a23a6f8b5808ab37fb2154c99ab0f6d25" alt=""
关于 SAP UI5 控件的元数据设计,请参考我这篇文章:
深入学习SAP UI5框架代码系列之四:SAP UI5控件的元数据实现。
获得 string 类型对象:
data:image/s3,"s3://crabby-images/4b53c/4b53c2c8e82c2cb61e85e25b5c7d7e1c4d8e906a" alt=""
类型对象由一系列函数组成:
data:image/s3,"s3://crabby-images/36e5d/36e5dd2ceb4dce8ad68c2bccaea0abb8389d065a" alt=""
执行 normalize 操作:
data:image/s3,"s3://crabby-images/ae4cf/ae4cf669465f450dbaf46ca070bc71a08826386d" alt=""
将这个不合法的 value,设置到 this.mProperties[sPropertyName]
中。
data:image/s3,"s3://crabby-images/a3a33/a3a335bb6eccc3959598a882a330d11856d1096e" alt=""
进行双向绑定的逻辑处理:this.updateModelProperty
data:image/s3,"s3://crabby-images/c955d/c955ddd90c487e41df8d897ed22c206c873d6f26" alt=""
拿到绑定对象:
data:image/s3,"s3://crabby-images/cf8c4/cf8c437075cf7b47f19b3e9cbb66c7d52f79b01f" alt=""
进入双向绑定的处理分支:
data:image/s3,"s3://crabby-images/2f574/2f57437906a16c563f54303c9107a663e9529748" alt=""
if (oBinding && oBinding.getBindingMode() == BindingMode.TwoWay) {
oBindingInfo.skipPropertyUpdate++;
SyncPromise.resolve(oValue).then(function(oValue) {
return oBinding.setExternalValue(oValue);
}).then(function() {
oBindingInfo.skipPropertyUpdate--;
return oBinding.getExternalValue();
}).then(function(oExternalValue) {
if (oValue != oExternalValue) {
that.updateProperty(sName);
}
handleSuccess();
}).catch(function(oException) {
oBindingInfo.skipPropertyUpdate--;
handleException(oException);
}).unwrap();
data:image/s3,"s3://crabby-images/6b63e/6b63ef547958ff3332fe411e251779aa3db5836d" alt=""
SyncPromise.resolve(oValue).then(function(oValue) {
return oBinding.setExternalValue(oValue);
})
data:image/s3,"s3://crabby-images/41977/419777d01c64cb3a88b41941d009f725cace2574" alt=""
在 CompositeBinding.js
内部,核心逻辑是第 325 行的代码:
that.oType.parseValue(oValue, that.sInternalType, aCurrentValues);
data:image/s3,"s3://crabby-images/4c777/4c7774c1649aa57c400b928bad3358ef3873cc72" alt=""
调用 outputFormat
进行输入值的解析工作:
data:image/s3,"s3://crabby-images/feacc/feacc8c20e594e4f66abf218148ccf3641c25cbd" alt=""