hbuilder中进行窗口管理有2种方式,一种是用了mui框架,可以用mui提供的管理方式,都封装好了。
还有一种是没有用mui,可以直接通过最底层的plus.webview来管理窗口的操作。
官方文档地址:http://www.dcloud.io/docs/api/zh_cn/webview.html
我们创建一个H5+的项目,在common.js中默认有对窗口管理的代码,可以直接使用。
在使用的过程中我遇到了下面的问题:
1,在APP下面有一个toolbar,目前每次点击一个按钮都是通过webview创建一个新的view,view中去加载网页(我们的功能实现都是通过h5的方式嵌入到app中)
common.js提供的方法源码如下:
/**
* 打开新窗口
* @param {URIString} id : 要打开页面url
* @param {boolean} wa : 是否显示等待框
* @param {boolean} ns : 是否不自动显示
* @param {JSON} ws : Webview窗口属性
*/
w.clicked=function(id,wa,ns,ws){
if(openw){//避免多次打开同一个页面
return null;
}
if(w.plus){
wa&&(waiting=plus.nativeUI.showWaiting());
ws=ws||{};
ws.scrollIndicator||(ws.scrollIndicator='none');
ws.scalable||(ws.scalable=false);
var pre='';//'http://192.168.1.178:8080/h5/';
openw=plus.webview.create(pre+id,id,ws);
ns||openw.addEventListener('loaded',function(){//页面加载完成后才显示
// setTimeout(function(){//延后显示可避免低端机上动画时白屏
openw.show(as);
closeWaiting();
// },200);
},false);
openw.addEventListener('close',function(){//页面关闭后可再次打开
openw=null;
},false);
return openw;
}else{
w.open(id);
}
return null;
};
我们直接调用clicked('http://cxytiandi.com',true)就可以打开猿天地的首页,APP中toolbar对应的几个模块的主页肯定会来回切换的,导致每次切换都需要重新渲染,我通过改造clicked方法来实现下面的需求:
- 如果页面对应的view没创建,就创建新的
- 如果页面对应的view已经存在,就直接打开已经存在的
直接上改造后的代码:
wa&&(waiting=plus.nativeUI.showWaiting());
ws=ws||{};
ws.scrollIndicator||(ws.scrollIndicator='none');
ws.scalable||(ws.scalable=false);
var pre='';//'http://192.168.1.178:8080/h5/';
if(ws.createNew==false){
openw = plus.webview.getWebviewById( id );
if (openw == null) {
openw=plus.webview.create(pre+id,id,ws);
} else {
openw.show(as);
closeWaiting();
return openw;
}
} else {
openw=plus.webview.create(pre+id,id,ws);
}
通过在ws对象中加一个createNew参数来决定是否要创建新的view,如果配置为false的话,就通过plus.webview.getWebviewById去查找之前有没有创建过,有就直接显示,不用创建了,使用代码如下:
clicked('http://cxytiandi.com',true, false,{createNew:false});
2,通过webview打开的页面,可以通过back()来返回到上个页面,源码如下:
// 处理返回事件
w.back=function(hide){
if(w.plus){
ws||(ws=plus.webview.currentWebview());
if(hide||ws.preate){
ws.hide('auto');
}else{
ws.close('auto');
}
}else if(history.length>1){
history.back();
}else{
w.close();
}
};
有些场景,比如在我的主页点击关注的用户,进入用户主页,在用户主页取消关注,然后回退到我的主页中,回退相当于重新打开之前开启过的view,这个时候我的页面中的关注数还是之前的,取消关注的并没有从总关注数中减去,今天我在回退的基础上做了扩展,
可以支持回退的时候指定是否要刷新上个页面,可以解决这个问题。
但是刷新页面其实用户体验很差的,最好的办法是异步加载,只改变需要改变的地方,为了支持这个我还加了一个回调的方法,可以支持方法回调,你可以自己写逻辑去实现数据刷新。
主要还是改造clicked方法,因为view是在clicked中创建的,在回退的时候我们可以通过监听view的close方法来实现这个操作
openw.addEventListener('close',function(){//页面关闭后可再次打开
//可以设置页面关闭时回调的方法,可以对数据进行重新加载操作
if(callback){
callback(callbackParams);
}
//页面关闭时需要刷新上个页面
if (ws.closeRefresh==true){
window.location.reload();
}
openw=null;
},false);
完整的clicked方法如下:
/**
* 打开新窗口
* @param {URIString} id : 要打开页面url
* @param {boolean} wa : 是否显示等待框
* @param {boolean} ns : 是否不自动显示
* @param {JSON} ws : Webview窗口属性
* @param {function} callback : 页面回退关闭时执行的回调方法
* @param {JSON} callbackParams : 回调方法的参数
*/
w.clicked=function(id,wa,ns,ws,callback,callbackParams){
/*if(openw){//避免多次打开同一个页面
return null;
}*/
if(w.plus){
wa&&(waiting=plus.nativeUI.showWaiting());
ws=ws||{};
ws.scrollIndicator||(ws.scrollIndicator='none');
ws.scalable||(ws.scalable=false);
var pre='';//'http://192.168.1.178:8080/h5/';
if(ws.createNew==false){
openw = plus.webview.getWebviewById( id );
if (openw == null) {
openw=plus.webview.create(pre+id,id,ws);
} else {
openw.show(as);
closeWaiting();
return openw;
}
} else {
openw=plus.webview.create(pre+id,id,ws);
}
ns||openw.addEventListener('loaded',function(){//页面加载完成后才显示
// setTimeout(function(){//延后显示可避免低端机上动画时白屏
openw.show(as);
closeWaiting();
// },200);
},false);
openw.addEventListener('close',function(){//页面关闭后可再次打开
//可以设置页面关闭时回调的方法,可以对数据进行重新加载操作
if(callback){
callback(callbackParams);
}
//页面关闭时需要刷新上个页面
if (ws.closeRefresh==true){
window.location.reload();
}
openw=null;
},false);
return openw;
}else{
w.open(id);
}
return null;
};
回退关闭时刷新使用代码如下:
clicked('http://cxytiandi.com',true, false,{closeRefresh:true});
回退关闭时执行回调代码使用代码如下:
clicked('http://cxytiandi.com',true, false,{},ref,{id:1001});
function ref(data) {
console.log(data.id);
}
3, 安卓手机上都有一个回退的按钮,相当于苹果中从左往右滑动,到了主页的时候如果点击了2次回退的按钮,需要提示是否退出app,这个可以通过监听来实现:
function plusReady(){
ws=plus.webview.currentWebview();
// Android处理返回键
plus.key.addEventListener('backbutton',function(){
var hurl = window.location.href;
if(hurl.indexOf("/house/index") > 0 || hurl.indexOf("/quotations/index") > 0
|| hurl.indexOf("/loanInfo/list") > 0 || hurl.indexOf("/selfInfo/index") > 0) {
plus.nativeUI.confirm("确认退出",function(e) {
if(e.index == 0) {
plus.runtime.quit();
}
});
} else {
back();
}
},false);
compatibleAdjust();
}