OPEN(SAP) UI5 学习入门系列之二: 最佳实践练习(下)
上期我们完成了一个简单的主从页面,但是页面是静态的,不能交互,功能也很简单,只有一个销售订单的列表。 我们今天就一鼓作气把代码全都写完,由于本次的代码量较大,所以只对重点代码部分进行讲解。 具体每个文件和代码就不一一贴出来了,代码都放在github中,需要的自行下载吧。
1 页面导航
可以先把代码下载到本地并跑起来,这样可以对这个最佳实践的程序有一个直观的了解。
页面导航如下:
销售订单列表(Master) -> 销售订单明细(Detail) -> 行项目明细(LineItem),在每个明细页面都可以返回到上一层。
具体页面之间的导航是如何实现的呢?
我们从页面的入口 index.html
开始
var oView = sap.ui.view({
id : "app",
viewName : "ui5.tutorial.bp.view.App",
type : "JS",
});
//...
oView.placeAt('content');
这一段代码初始化了一个叫做App的JS view,那我们就来看 App.view.js
// create app
this.app = new sap.m.SplitApp(); // load the master page
var master = sap.ui.xmlview("Master", "ui5.tutorial.bp.view.Master");
master.getController().nav = this.getController();
this.app.addPage(master, true); // load the empty page
var empty = sap.ui.xmlview("Empty", "ui5.tutorial.bp.view.Empty");
this.app.addPage(empty, false);
App view中在 createContent
中创建了一个SplitApp,稍微说下SplitApp这个控件,当宿主是PC或者平台的时候这个控件默认包含两个页面容器,而当宿主是手机的时候又可以只包含一个页面,所以一般主从结构的页面可以用这个控件。
随后,分别创建了两个页面,一个是Master页面,另一个页面是空白页——作为没有选中任何Master页面中的数据时的默认页面,最后把两个页面都加入到了SplitApp中。
到目前为止都是在上一篇中已经做过的,到这里,页面已经可以展示了。但是我们今天要研究的是导航,所以接着往下。
SplitApp本身带有 to()
这个函数,可以在已经加入其容器的页面之间导航,但是我们现在的功能稍稍有点复杂,当点击Master页面的时候,要求导航到详细页面,
点击详细页面的行项目可以进入到行项目的详细页面,对应每个详细页面可以返回至上一级页面,同时可能还需要做一些其他逻辑上的处理,比如判断和绑定数据,因此我们需要在原有的 to()
上增强一些功能并封装给其他的控制器使用。
这些功能都集中定义在 App.controller.js
这个控制器中
to : function (pageId, context) { var app = this.getView().app; // load page on demand
var master = ("Master" === pageId);
if (app.getPage(pageId, master) === null) {
var page = sap.ui.view({
id : pageId,
viewName : "ui5.tutorial.bp.view." + pageId,
type : "XML"
});
page.getController().nav = this;
app.addPage(page, master);
jQuery.sap.log.info("app controller > loaded page: " + pageId);
} // show the page
app.to(pageId); // set data context on the page
if (context) {
var page = app.getPage(pageId);
page.setBindingContext(context);
}
}, /**
* Navigates back to a previous page
* @param {string} pageId The id of the next page
*/
back : function (pageId) {
this.getView().app.backToPage(pageId);
}
重新封装后的 to()
有两个参数,一个是页面id,另一个是上下文,当传入的页面不存在的时候,系统会首先初始化这个页面并加入到当前的SplitApp中,随后直接调用SplitApp的 to()
完成导航动作。 最后,如果上下文不为空的话,把这个上下文绑定到新的页面中,这对于一个新页面来说是非常有意义的。
back则直接复用SplitApp的 backToPage()
函数。
接下来我们再看其他的页面如何复用定义在这里的导航函数的。
我们回过头来再看 App.view.js
, 其中在初始化Master页面之后,有这么一条语句
master.getController().nav = this.getController();
这条语句把当前的控制器赋给Master页面的控制器的nav,这个nav只是用来存放App控制器的一个key,叫什么名字都行,这样在Master页面中就可以通过nav来调用App控制器的所有函数了。
同样的,再回到 App.controller.js
中,看到这条语句 page.getController().nav = this;
也是类似的作用。
在首页面刚刚初始化时,Detail页面是没有加载的,当点击Master页面中的某个SalesOrder的时候,Master页面中的 handleListItemPress
被调用:
handleListItemPress : function (evt) {
var context = evt.getSource().getBindingContext();
this.nav.to("Detail", context);
}
首先获得被点击的item上绑定的上下文信息(数据),然后调用App的 to()
方法并传入 Detail
告诉页面要加载Detail这个页面,同时把上下文数据传递过去, 接下来又回到了 to()
, Detail页面被初始化,绑定上下文数据,加载到SplitApp中,然后导航成为当前显示页面。
以上,就完成了这个App的导航,可以看到有些繁琐,并且需要显式的初始化子页面的时候获得并共享App的控制器,另外,页面之间的跳转不会修改URL,无法将某个中间页面存为bookmark,所以这种方式并不是UI5所推荐的, 但是作为初学者了解UI5的页面导航机制还是非常的直观,另外对于简单的应用来说,如果页面较少也未尝不可以考虑。
作为稍大型的web应用,UI5在早期的版本中推荐使用EventBus通过Event的传递来实现复杂的页面导航,从1.6开始引入了新的导航机制,就是Routing,可以将页面之间的导航关系定义在component中,在最新的1.30版本中,导航定义则可以直接写在App的说明文件 manifest.jso
中。
导航就介绍到这里,Component和Routing是一个比较复杂但是非常强大的工具,我们可以在后续接着探讨。
2 数据绑定
在我们的代码中,数据绑定也是做了简化处理,都直接写在 index.html
中了。
一共绑定了三个模型:
- 业务数据模型:
因为我们使用的是离线的json格式数据,所以可以直接把相对路径传递给sap.ui.model.json.JSONModel
来初始化这个模型,并绑定到App这个根视图上。var oModel = new sap.ui.model.json.JSONModel("model/mock.json");
oView.setModel(oModel);随后在这个视图及其子视图中,都可以直接通过类似
{SoId}
这种语法格式来使用这个模型的数据字段,需要注意的是,如果需要绑定的字段是这个模型的根节点,需要在前面加一个/
,譬如在Master视图中绑定到列表的aggregate字段items
,是这样的语法格式:items="{/SalesOrderCollection}" 。 - 多语言模型:
UI5中使用了i18n
机制来处理多语言问题。i18n是 internationalization的简称,在首位两个字母之间有18个字母……具体如何使用非常的简单,
首先创建一个资源文件messageBundle.properties
,这里我们在根目录创建了一个i18n
目录,在这里目录中集中存放相关的i18n文件。
在这个资源文件里我们定义如下:MasterTitle=Sales Orders
DetailTitle=Sales Order
StatusTextN=New
StatusTextP=In Process
ApproveButtonText=Approve
…左边的是KEY,右边的是对应的语言描述,如果我们需要定义一个中文的语言文件,那么只需要拷贝这个文件并重命名为
messageBundle_zh-CN.properties
,并将对应的描述改为中文如下:MasterTitle=销售订单列表
DetailTitle=销售订单
StatusTextN=新建
StatusTextP=处理中
ApproveButtonText=批准
…系统会根据用户设定的浏览器语言顺序依次查找对应的语言资源文件,如果都找不到的话,就会找默认的
messageBundle.properties
。定义好了资源文件,我们接下来就在
index.html
中通过sap.ui.model.resource.ResourceModel
来初始化这个资源模型,接着就可以把它绑定到视图或者控件中使用了。
怎么使用呢?在视图中通过类似{i18n>MasterTiel}
这种语法格式来绑定到对应的空间的文本项上,实际使用中需要用引号把这个串包含进去。这个串中前面的i18n>
指的是引用绑定到本视图的叫做i18n的模型,这里的i18n是绑定时起的名字,oView.setModel(i18nModel, "i18n");
可以是任意符合格式的字符串。 - 设备模型
设备模型通过查询jQuery的device来获悉宿主是否手机,并设定相应的不同显示选项,然后将结果存为Json格式并初始化为一个JSON模型,最后绑定到模型中。
3 工具方法
大多数情况下,我们可以直接把业务数据直接绑定到控件中显示,但是在一些情况下,我们可能需要对其中的一些格式做一些调整,或者根据一些字段做一些简单的逻辑处理,
这个时候,我们就需要用到大多数控件中的某些属性的 formatter
方法。
<ObjectStatus
text="{
path: 'LifecycleStatus',
formatter: 'ui5.tutorial.bp.util.Formatter.statusText'
}"
state="{
path: 'LifecycleStatus',
formatter: 'ui5.tutorial.bp.util.Formatter.statusState'
}" />
上面这个例子中,我们来看 text
属性,如果我们希望直接把业务数据绑定到text中,我们这样定义 text="{LifecycleStatus}",但是我们知道这个字段可能是后台定义的技术字段,我们需要把它转化的比较有业务意义。 所以这个时候,我们就需要用到 formatter
了,首先定义 path
告之需要绑定的字段,这里不需要用大括号,随后给 formatter
赋予一个处理方法,这个方法可以定义在任何地方,我们这里是在util下单独定义了一个 Fomatter.js
来集中处理这类需求。
来看 Formatter.js
,我们就看 statusText
这个方法:
statusText : function (value) {
var bundle = this.getModel("i18n").getResourceBundle();
return bundle.getText("StatusText" + value, "?");
},
path
中绑定的字段对应的值会作为参数传入,然后用这个值结合StatusText生成一个KEY,并在 i18n
中取出相应的描述。
4 总结
基本上这个最佳实践应用已经被剖析完成了,通过这样一个最佳实践 Best Practice
的练习,我们学习到了一般的UI5应用的整体结构以及大多数重要控件的使用方法。
OPEN(SAP) UI5 学习入门系列之二: 最佳实践练习(下)的更多相关文章
- OPEN(SAP) UI5 学习入门系列之二: 最佳实践练习(上)
这篇博文难产了很久,原来是打算一周更新一篇的,上周原计划写MVC,但是写了一半,发现带入了太多的细节,不太符合这个入门系列的主题. 当我们学习一个新的技能的时候,如果一开始就面对大量的细节,很容易陷入 ...
- OPEN(SAP) UI5 学习入门系列之四:更好的入门系列-官方Walkthrough
好久没有更新了,实在不知道应该写一些什么内容,因为作为入门系列,实际上应该更多的是操作而不是理论,而在UI5 SDK中的EXPLORER里面有着各种控件的用法,所以在这里也没有必要再来一遍,还是看官方 ...
- OPEN(SAP) UI5 学习入门系列之一:扫盲与热身(下)
1 UI5代码结构 上一次我们一起用了20秒的时间完成一个UI5版的Hello World.应用打开后有一个按钮,按钮的文字是Hello World,点击这个按钮之后,按钮会慢慢的消失掉(Fade o ...
- OPEN(SAP) UI5 学习入门系列之三:MVC (上) - 模型
这次我们来一起学习MVC,这个专题分为两个小节,本次主要是总览以及模型,下一次着重会介绍视图以及控制器,因为控制器其实没有太多可以讲的,所以和视图合并在一块. 1 Model View Control ...
- [转]OPEN(SAP) UI5 学习入门系列之一:扫盲与热身(上)
本文转自:http://www.cnblogs.com/qianmarv/p/4671394.html 1 扫盲 什么是SAP Fiori? 了解SAP UI5必须要从SAP Fiori开始,两者概念 ...
- OPEN(SAP) UI5 学习入门系列之一:扫盲与热身(上)
什么是SAP Fiori? 了解SAP UI5必须要从SAP Fiori开始,两者概念经常被混淆,而两者也确实有着非常紧密的关系. 用过SAP的同学们都对SAP的传统的界面(SAP GUI)表示“呵呵 ...
- OPEN(SAP) UI5 学习入门系列之三:MVC (下) - 视图与控制器
继续来学习UI5的MVC模型吧,这次我们来探讨视图与控制器. 1 视图 在MVC中,视图用来定义和渲染UI.在UI5中,视图的类型是可以自定义的,除了以下预定义的四种视图类型之外,你也可以定制自己的视 ...
- ABP学习入门系列(二)(abp的数据迁移)
本文将介绍在ABP框架中将实体类迁移至数据库表 1.下图是abp的体系结构. 我们要是实现创建实体类并迁移数据到数据库的功能主要就是在下图中domain(领域层)做相应的一些操作. 2,看一下解决方案 ...
- Cordova入门系列(二)分析第一个helloworld项目 转发 https://www.cnblogs.com/lishuxue/p/6015420.html
Cordova入门系列(二)分析第一个helloworld项目 版权声明:本文为博主原创文章,转载请注明出处 上一章我们介绍了如何创建一个cordova android项目,这章我们介绍一下创建的 ...
随机推荐
- javascript 实现类似百度联想输入,自动补全功能
js 实现类似百度联想输入,自动补全功能 方案一: search是搜索框id="search" //点击页面隐藏自动补全提示框 document.onclick = functi ...
- 洛谷 P1879 [USACO06NOV]玉米田Corn Fields
题目描述 Farmer John has purchased a lush new rectangular pasture composed of M by N (1 ≤ M ≤ 12; 1 ≤ N ...
- Liunx在开机后,自动启动openldap、radius、memcached等程序的shell脚本
以下是脚本命令: #!/bin/bash #说明:此文件需放在/etc/rc.d/init.d/目录下,然后编辑文件/etc/rc.d/rc.local,在里面添加bash /etc/init.d/A ...
- Notification通知代码简洁使用
1.自定义发送 Notification 的使用 1.1 通知(消息)的创建 ---------------详细介绍篇 // 不带消息内容 NSNotification *notification1 ...
- 洛谷P3628 [APIO2010]特别行动队(斜率优化)
传送门 先写出转移方程$$dp[i]=max\{dp[j]+a*(sum[i]-sum[j])^2+b*(sum[i]-sum[j])+c\}$$ 假设$j$比$k$更优,则有$$dp[j]+a*(s ...
- 洛谷P3901 数列找不同(莫队)
传送门 我不管我不管我就是要用莫队 直接用莫队裸上 //minamoto #include<iostream> #include<cstdio> #include<alg ...
- Linux之杀掉进程
pkill命令可以按照进程名杀死进程.pkill和killall应用方法差不多,也是直接杀死运行中的程序:如果您想杀掉单个进程,请用kill来杀掉. 语法 pkill(选项)(参数) 选项 -o:仅向 ...
- springboot整合xxl-mq学习笔记
首先xxl-mq是大神xuxueli开发的一个消息中间件框架: 与springboot整合过程: <?xml version="1.0" encoding="UTF ...
- springdataRedis连接redis集群
配置文件: <?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http:// ...
- sharepoint_study_目录学习笔记(长期更新)
1. _catalogs/masterpage:这个是SharePoint网站的母版页样式库页面,这里放了网站上所有的母版页(网站设置--Web设计器库--母版页和页面布局). 2. 15\TEMP ...