第六篇 导航、页面跳转、stream、webview
在一款应用中,导航、页面切换、获取网络数据是很常见的功能。因此,本节做一个简单的介绍。
wxc-navpage组件
什么是navigation?如下图:
在iOS,使用的控制是UINavigationController。我们可以看到Navigation的内容比较多,例如中间的标题、左侧的搜索图片、右侧的图片等。同样,Weex也提供了该组件,那就是wxc-navpage。
<template>
<wxc-navpage height={% raw %}{{...}}{% endraw %} background-color="..." title="..." title-color="..." left-item-title="..." left-item-color="..." right-item-src="...">
<content> ...</content>
</wxc-navpage>
</template>
<script>
require('weex-components');
module.exports = {
created: function() {
this.$on('naviBar.rightItem.click',function(e){
//handle your click event here.
});
this.$on('naviBar.leftItem.click',function(e){
//handle your click event here.
});
}
}
</script>
我们,可以看到 wxc-navpage不仅支持基础的样式,同样提供了 title、title-color、 left-item-title、left-item-color、right-item-src等等属性,这里可以参考http://alibaba.github.io/weex/doc/components/wxc-navpage.html 。
wxc-navpage内嵌子组件,即<content> ...</content>
。但是,我们改造上面的片段代码是跑不起来的。因此我们需要安装weex-components:
$ npm install weex-components
weex-components中包含的内容比较多,例如也提供了wxc-tabbar组件。但是,需求是多样的。有时候,发现weex-components组件不能满足一些特定的需求。那么,这里提供2个思路:(1)给weex提issues,让weex支持新的需求 (2)如果实在等不及,可以将weex-components从node_modules中拷贝出来,然后修改里面的内容。
页面跳转
很可能一个应用需要多个页面,那么多个页面如何跳转呢?其实Weex的Demo就给出了很好的例子。例如,这里的代码如下:
var bundleUrl = this.$getConfig().bundleUrl;
bundleUrl = new String(bundleUrl);
console.log('hit', bundleUrl);
var nativeBase;
var isAndroidAssets = bundleUrl.indexOf('file://assets/') >= 0;
var isiOSAssets = bundleUrl.indexOf('file:///') >= 0 && bundleUrl.indexOf('WeexDemo.app') > 0;
if (isAndroidAssets) {
nativeBase = 'file://assets/';
}
else if (isiOSAssets) {
// file:///var/mobile/Containers/Bundle/Application/{id}/WeexDemo.app/
// file:///Users/{user}/Library/Developer/CoreSimulator/Devices/{id}/data/Containers/Bundle/Application/{id}/WeexDemo.app/
nativeBase = bundleUrl.substring(0, bundleUrl.lastIndexOf('/') + 1);
}
else {
var host = 'localhost:12580';
var matches = /\/\/([^\/]+?)\//.exec(this.$getConfig().bundleUrl);
if (matches && matches.length >= 2) {
host = matches[1];
}
nativeBase = 'http://' + host + '/' + this.dir + '/build/';
}
var h5Base = './index.html?page=./' + this.dir + '/build/';
// in Native
var base = nativeBase;
if (typeof window === 'object') {
// in Browser or WebView
base = h5Base;
}
this.baseURL = base;
上面的代码是我从Weex Demo中拔出来的。这一块也是经常被开发者问到的,就是跳转路径的问题。其实,仔细看上面的代码,我们只需要关注两个地方,一个是this.dir
,另一个是var h5Base = './index.html?page=./'
。
我们在created方法中获取到this.dir。this.dir 则是我们打包完成的js bundle的目录。h5Base则是web端的路径。它的基本形式是:
http://127.0.0.1:8080/index.html?page=./dist/index.js
其中index.html则是web端运行的载体,page=./dist/index.js则是传入到weex init中的参数。我们一般在集成web时候,会写上如下代码:
(function () {
function getUrlParam (key) {
var reg = new RegExp('[?|&]' + key + '=([^&]+)')
var match = location.search.match(reg)
return match && match[1]
}
var loader = getUrlParam('loader') || 'xhr'
var page = getUrlParam('page') || 'examples/build/index.js'
// jsonp callback name should be specified or be the default
// value 'weexJsonpCallback' if the 'jsonp' loader is used.
var JSONP_CALLBACK_NAME = 'weexJsonpCallback'
window.weex.init({
jsonpCallback: JSONP_CALLBACK_NAME,
appId: location.href,
loader: loader,
source: page,
rootId: 'weex',
downgrade: [] // 'list', 'scroller'
})
})();
getUrlParam就是获取url中的参数。如果页面有page参数,我们则调用page参数对应的js文件的代码。
到这里,基本上,可以看出跳转路径了。那么使用什么API跳转呢?那就是navigator。
var url = this.recommend[index].url;
var title = this.recommend[index].title;
var navigator = require('@weex-module/navigator');
var params = {
'url': this.baseURL + 'yywebview.js?weixin_title=' + title + '&weixin_url=' + url,
'animated' : 'true',
}
console.log('------',params.url);
navigator.push(params, function(e) {});
我们获取到this.baseUR,然后拼接路径。this.baseUR则是created方法中获取到不同端的路径前缀。这里,引入了navigator = require('@weex-module/navigator');
组件,然后使用了push方法,将路由对象添加进了路由栈。如果我们需要回退到上一个视图,该如何呢?其实navigator api已经提供了该功能,那就是pop。具体可以参考http://alibaba.github.io/weex/doc/modules/navigator.html 。
网络请求
Weex提供了网络请求模块,那就是stream。简单的例子如下:
stream.fetch({
method: 'GET',
url: "http://httpbin.org/get",
type:'json'
}, function(response) {
//process response
},function(response){
//progress response
console.log("current bytes received:"+response.length);
});
基本上stream.fetch方法的几个参数都很容易理解。如果需要POST、PUT、DELETE等。可以参考: https://github.com/alibaba/weex/blob/dev/examples/module/stream-demo.we 。
webview
当然,在Weex中也支持Webview类似组件,只是Weex命名其为web。具体使用如下:
<template>
<div class="wrapper">
<web class="web" src="{{src}}" style="height:{{height}};" onpagestart="pagestart" onpagefinish="pagefinish" onerror="error"></web>
</div>
</template>
<style>
.text{
font-size:26;
}
.web{
flex:1;
}
</style>
其中src就是我们要打开的网页地址。onpagestart、onpagefinish、onerror几个事件很重要。比如在onpagefinish时去掉loading,onerror时给用户友好提示。 具体参考:http://alibaba.github.io/weex/doc/components/web.html 。