一:編寫tabBar模板
眾所周知,微信小程序的tabBar都是新開頁面的,而微信文檔上又表明了最多只能打開5層頁面。這樣就很容易導(dǎo)致出問題啦,假如我的tabBar有5個呢?下面是微信原話:
一個應(yīng)用同時只能打開5個頁面,當(dāng)已經(jīng)打開了5個頁面之后,wx.navigateTo不能正常打開新頁面。請避免多層級的交互方式,或者使用wx.redirectTo
因此這幾天想著根據(jù)微信tabBar數(shù)組來自定義模板供頁面調(diào)用。不過我在list里面每個對象都增加了一個selectedColor和active屬性,方便對每個tabBar當(dāng)前頁做樣式,如果不傳直接使用設(shè)置的selectedColor。因此這串?dāng)?shù)據(jù)只能設(shè)定在各個頁面下,不能設(shè)定在公用的app.js配置文件下,稍微有點代碼冗余,下次研究下怎么直接配置到app.js完善下。
只要新建一個tarBar.wxml模板頁,然后引用模板的頁面?zhèn)魅霐?shù)據(jù)即可,代碼如下:
-
<template name="tabBar">
-
<view class="flex-h flex-hsb tab-bar" style="color: {{tabBar.color}}; background: {{tarBar.backgroundColor}}; {{tabBar.position=='top'? 'top: 0' : 'bottom: 0'}}; {{tabBar.borderStyle? (tabBar.position=='top'? 'border-bottom: solid 1px '+tabBar.borderStyle + ';' : 'border-top: solid 1px '+tabBar.borderStyle + ';') : ''}}">
-
<block wx:for="{{tabBar.list}}" wx:key="pagePath">
-
<navigator url="{{item.pagePath}}" open-type="redirect" class="menu-item" style="{{item.active? 'color: '+(item.selectedColor? item.selectedColor : tabBar.selectedColor) : ''}}">
-
<image src="{{item.selectedIconPath}}" wx:if="{{item.active}}"></image>
-
<image src="{{item.iconPath}}" wx:if="{{!item.active}}"></image>
-
<text>{{item.text}}</text>
-
</navigator>
-
</block>
-
</view>
-
</template>
復(fù)制代碼
接下來進(jìn)行測試,首先是index.js的配置對象:
-
//配置tabBar
-
tabBar: {
-
"color": "#9E9E9E",
-
"selectedColor": "#f00",
-
"backgroundColor": "#fff",
-
"borderStyle": "#ccc",
-
"list": [
-
{
-
"pagePath": "/pages/index/index",
-
"text": "主頁",
-
"iconPath": "../../img/tabBar_home.png",
-
"selectedIconPath": "../../img/tabBar_home_cur.png",
-
//"selectedColor": "#4EDF80",
-
active: true
-
},
-
{
-
"pagePath": "/pages/village/city/city",
-
"text": "目的地",
-
"iconPath": "../../img/tabBar_village.png",
-
"selectedIconPath": "../../img/tabBar_village_cur.png",
-
"selectedColor": "#4EDF80",
-
active: false
-
},
-
{
-
"pagePath": "/pages/mine/mine",
-
"text": "我的",
-
"iconPath": "../../img/tabBar_mine.png",
-
"selectedIconPath": "../../img/tabBar_mine_cur.png",
-
"selectedColor": "#4EDF80",
-
active: false
-
}
-
],
-
"position": "bottom"
-
}
復(fù)制代碼
index.wxml引入模板:
-
<import src="../../template/tabBar.wxml" />
-
<template is="tabBar" data="{{tabBar: tabBar}}" />
復(fù)制代碼
接下來是mine.js文件引入配置對象:
-
//配置tabBar
-
tabBar: {
-
"color": "#9E9E9E",
-
"selectedColor": "#f00",
-
"backgroundColor": "#fff",
-
"borderStyle": "#ccc",
-
"list": [
-
{
-
"pagePath": "/pages/index/index",
-
"text": "主頁",
-
"iconPath": "../../img/tabBar_home.png",
-
"selectedIconPath": "../../img/tabBar_home_cur.png",
-
//"selectedColor": "#4EDF80",
-
active: false
-
},
-
{
-
"pagePath": "/pages/village/city/city",
-
"text": "目的地",
-
"iconPath": "../../../img/tabBar_village.png",
-
"selectedIconPath": "../../../img/tabBar_village_cur.png",
-
"selectedColor": "#4EDF80",
-
active: false
-
},
-
{
-
"pagePath": "/pages/mine/mine",
-
"text": "我的",
-
"iconPath": "../../img/tabBar_mine.png",
-
"selectedIconPath": "../../img/tabBar_mine_cur.png",
-
"selectedColor": "#4EDF80",
-
active: true
-
}
-
],
-
"position": "bottom"
-
}
復(fù)制代碼
mine.wxml引入模板:
-
<import src="../../template/tabBar.wxml" />
-
<template is="tabBar" data="{{tabBar: tabBar}}" />
復(fù)制代碼
最后演示如下:
方案二
我把配置數(shù)據(jù)統(tǒng)一放在app.js文件,通過點擊跳轉(zhuǎn)頁面后在把數(shù)據(jù)添加到當(dāng)前頁面實例上,具體做法如下:
1、app.js文件配置:
-
//app.js
-
var net = require('common/net');
-
var a_l, a_d = {}, a_cbSucc, a_cbSuccFail, a_cbFail, a_cbCom, a_h, a_m;
-
App({
-
onLaunch: function () {
-
var that = this;
-
},
-
//修改tabBar的active值
-
editTabBar: function () {
-
var _curPageArr = getCurrentPages();
-
var _curPage = _curPageArr[_curPageArr.length - 1];<span style="font-family: Arial, Helvetica;">//相當(dāng)于Page({})里面的this對象</span>
-
var _pagePath=_curPage.__route__;
-
if(_pagePath.indexOf('/') != 0){
-
_pagePath='/'+_pagePath;
-
}
-
var tabBar=this.globalData.tabBar;
-
for(var i=0; i<tabBar.list.length; i++){
-
tabBar.list.active=false;
-
if(tabBar.list.pagePath==_pagePath){
-
tabBar.list.active=true;//根據(jù)頁面地址設(shè)置當(dāng)前頁面狀態(tài)
-
}
-
}
-
_curPage.setData({
-
tabBar: tabBar
-
});
-
},
-
globalData: {
-
userInfo: null,
-
//配置tabBar
-
tabBar: {
-
"color": "#9E9E9E",
-
"selectedColor": "#f00",
-
"backgroundColor": "#fff",
-
"borderStyle": "#ccc",
-
"list": [
-
{
-
"pagePath": "/pages/index/index",
-
"text": "主頁",
-
"iconPath": "/pages/templateImg/tabBar_home.png",
-
"selectedIconPath": "/pages/templateImg/tabBar_home_cur.png",
-
"selectedColor": "#4EDF80",
-
active: false
-
},
-
{
-
"pagePath": "/pages/village/city/city",
-
"text": "目的地",
-
"iconPath": "/pages/templateImg/tabBar_village.png",
-
"selectedIconPath": "/pages/templateImg/tabBar_village_cur.png",
-
"selectedColor": "#4EDF80",
-
active: false
-
},
-
{
-
"pagePath": "/pages/mine/mine",
-
"text": "我的",
-
"iconPath": "/pages/templateImg/tabBar_mine.png",
-
"selectedIconPath": "/pages/templateImg/tabBar_mine_cur.png",
-
"selectedColor": "#4EDF80",
-
active: false
-
}
-
],
-
"position": "bottom"
-
}
-
}
-
})
復(fù)制代碼
2、index.js+mine.js+city.js頁面使用:
-
var App=getApp();
-
Page({
-
data:{
-
detail: {},
-
},
-
onLoad:function(options){
-
App.editTabBar();//添加tabBar數(shù)據(jù)
-
var that=this;
-
}
-
})
復(fù)制代碼
最終演示和上圖一致!
二:map組件markers屬性動態(tài)初始化
今天在寫小程序詳情頁時候遇到一個問題,map組件的markers屬性是通過異步請求到數(shù)據(jù)后設(shè)置的,結(jié)果就導(dǎo)致了微信底層渲染出錯。但是如果我先在data初始化markers變量,每次小程序都不能正常渲染都是初始化的北京的數(shù)據(jù)。然后寫了如下測試:
test.js:
-
Page({
-
data:{
-
map:{
-
lat: 0,
-
lng: 0,
-
markers: [],
-
hasMarkers: false//解決方案
-
}
-
},
-
onLoad: function(options){
-
var that=this;
-
wx.getLocation({
-
type: 'wgs84', // 默認(rèn)為 wgs84 返回 gps 坐標(biāo),gcj02 返回可用于 wx.openLocation 的坐標(biāo)
-
success: function (res) {
-
// success
-
wx.request({
-
url: 'https://xxx.com/detail.htm?vid=3&latlng=' + res.latitude + ',' + res.longitude,
-
data: {},
-
method: 'GET', // OPTIONS, GET, HEAD, POST, PUT, DELETE, TRACE, CONNECT
-
// header: {}, // 設(shè)置請求的 header
-
success: function (res) {
-
// success
-
that.setData({
-
'map.lat': res.data.data.lat,
-
'map.lng': res.data.data.lng,
-
'map.markers': [{
-
latitude: res.data.data.lat,
-
longitude: res.data.data.lng,
-
name: res.data.data.title,
-
desc: res.data.data.address
-
}],
-
'map.hasMarkers': true//解決方案
-
});
-
console.log(that.data.map.markers);
-
}
-
})
-
}
-
})
-
}
-
});
復(fù)制代碼
test.wxml:
-
<view style="height: 100rpx;">
-
{{map.lat}}--{{map.lng}}--{{map.markers[0].name}}--{{map.markers[0].desc}}
-
</view>
-
<map latitude="{{map.lat}}" longitude="{{map.lng}}" markers="{{map.markers}}"></map>
復(fù)制代碼
測試一:data里面沒有初始化map變量
測試二:初始化map變量,并在wxml把map數(shù)據(jù)都打印出來
為什么會這樣呢??各種測試,最后發(fā)現(xiàn)只要不是經(jīng)過遠(yuǎn)程請求在來設(shè)置map值就不會出問題。
后來經(jīng)大神網(wǎng)友提點,由于異步設(shè)置值的過程,組件已經(jīng)渲染。但是map變量沒有預(yù)定義是undefined狀態(tài),map初始化拿不到數(shù)據(jù)直接報錯了,而異步過來又變成動態(tài)更新導(dǎo)致了wxml需要重新渲染map組件的情況導(dǎo)致的。因為官方文檔有提到:
地圖組件的經(jīng)緯度必填, 如果不填經(jīng)緯度則默認(rèn)值是北京的經(jīng)緯度。 標(biāo)記點markers只能在初始化的時候設(shè)置,不支持動態(tài)更新。
只能初始化一次,因此導(dǎo)致每次都是顯示的初始化信息。
然后,網(wǎng)友給出的解決方案完美解決了這個問題:
wx:if會渲染一下組件,那我們按照這個思路給他加個if就行了
-
<map markers="{{markers}}" style="width: 375px; height: 200px;" wx:if="{{map}}"></map>
復(fù)制代碼
默認(rèn)map是false,就是加載時不渲染map組件,等ajax回來后把map設(shè)置為true,這樣就動態(tài)渲染成你要的地址了