小程序頭部導航欄添加圖片背景


原理其實就是通過將原來的頭禁用,然后PAGE自然而然的頂上去以后,定義一個頭的組件,將他設置成fixed布局固定在原來頭的部分,然后給page加上Margin-top,所以還原原來的感覺。

1.app配置

​ 首先禁用所有頭導航,在app.json的window里加一行這個,你會發現所有頭都消失了。然后禁止滑動頁面,滑動問題用scroll-view解決

"window": {
    "navigationStyle": "custom"
}, 
"disableScroll": true

 

​ 然后在app.js里獲取導航頭的高度的全局數據

// app.js

App({
  globalData: {
    statusBarHeight: wx.getSystemInfoSync()['statusBarHeight']
  },
  // 判斷是否由分享進入小程序
    if (e.scene == 1007 || e.scene == 1008) {
      this.globalData.share = true
    } else {
      this.globalData.share = false
    }
    //獲取設備頂部窗口的高度(不同設備窗口高度不一樣,根據這個來設置自定義導航欄的高度)
    //這個最初我是在組件中獲取,但是出現了一個問題,當第一次進入小程序時導航欄會把
    //頁面內容蓋住一部分,當打開調試重新進入時就沒有問題,這個問題弄得我是莫名其妙
    //雖然最后解決了,但是花費了不少時間
    wx.getSystemInfo({
      success: res => {
        this.globalData.height = res.statusBarHeight
      }
    })
  },
  globalData: {
    userInfo: null,
    share: false, // 分享默認為false
    height: 0 // 頂部高度
  }
})

 

​ 在app.wxss給page加一個高度百分之百。

/* app.wxss */
page {
  height: 100%;
}

 

​ app配置到這里應該完事了。

2.組件配置

組件結構:

放源碼吧

// navbar.wxml

<view class='nav-wrap' style='height: {{height*2 + 20}}px;'>
  <!-- 導航欄背景圖片 -->
  <image class="backgroundimg" src="{{navbarData.address}}" bindload="imgLoaded" style="width:{{imageWidth}}px;height:{{imageHeight}}px" />
  <!-- // 導航欄 中間的標題 -->
  <view class='nav-title' wx:if='{{!navbarData.white}}' style='line-height: {{height*2 + 44}}px;'>
    {{navbarData.title}}
  </view>
  <view class='nav-title' wx:else='{{!navbarData.white}}' style='line-height: {{height*2 + 44}}px; color:#ffffff'>
    {{navbarData.title}}
  </view>
  <view style='display: flex; justify-content: space-around;flex-direction: column'>
    <!-- // 導航欄  左上角的返回按鈕 -->
    <!-- //  其中wx:if='{{navbarData.showCapsule}}' 是控制左上角按鈕的顯示隱藏,首頁不顯示 -->
    <view class='nav-capsule' style='height: {{height*2 + 44}}px;' wx:if='{{navbarData.showCapsule}}'>
      <!-- //左上角的返回按鈕,wx:if='{{!share}}'空制返回按鈕顯示 -->
      <!-- //從分享進入小程序時 返回上一級按鈕不應該存在 -->
      <!-- navbarData.white是控制按鈕顏色的,因為背景有深淺色,返回按鈕自己找圖片 -->
      <view bindtap='_navback' wx:if='{{!share&&navbarData.white}}'>
        <image src='../../images/返 回 (1).svg' mode='aspectFit' class='back-pre'></image>
      </view>
      <view bindtap='_navback' wx:else='{{!share}}'>
        <image src='../../images/返 回.svg' mode='aspectFit' class='back-pre'></image>
      </view>
    </view>
  </view>
</view>
<!-- 導航欄下面的背景圖片 -->
<image class="backgroundimg" src="{{navbarData.address}}" bindload="imgLoaded" style="width:{{imageWidth}}px;height:{{imageHeight}}px" />

 

CSS:

/* navbar.wxss */

/* 頂部要固定定位   標題要居中   自定義按鈕和標題要和右邊微信原生的膠囊上下對齊 */
.nav-wrap {
  /* display: none; */
  position: fixed;
  width: 100%;
  top: 0;
  background: #fff;
  color: #000;
  z-index: 9999999;
  background: #000;
  overflow: hidden;
}
/* 背景圖 */
.backgroundimg {
  position: absolute;
  z-index: -1;
}
/* 標題要居中 */
.nav-title {
  position: absolute;
  text-align: center;
  max-width: 400rpx;
  overflow: hidden;
  text-overflow: ellipsis;
  white-space: nowrap;
  top: 0;
  left: 0;
  right: 0;
  bottom: 0;
  margin: auto;
  font-size: 36rpx;
  color: #2c2b2b;
  font-weight: 450;
}

.nav-capsule {
  display: flex;
  align-items: center;
  margin-left: 30rpx;
  width: 140rpx;
  justify-content: space-between;
  height: 100%;
}

.back-pre {
  width: 32rpx;
  height: 36rpx;
  margin-top: 4rpx;
  padding: 10rpx;
}
.nav-capsule {
  width: 36rpx;
  height: 40rpx;
  margin-top: 3rpx;
}

 

在JSON里聲明我是個組件

{
  "component": true,
  "usingComponents": {}
}

 

最后是js。

const app = getApp()
Component({
  properties: {
    navbarData: {
      //navbarData   由父頁面傳遞的數據,變量名字自命名
      type: Object,
      value: {},
      observer: function(newVal, oldVal) {}
    }
  },
  data: {
    height: '',
    //默認值  默認顯示左上角
    navbarData: {
      showCapsule: 1
    },
    imageWidth: wx.getSystemInfoSync().windowWidth, // 背景圖片的高度
    imageHeight: '' // 背景圖片的長度,通過計算獲取
  },
  attached: function() {
    // 獲取是否是通過分享進入的小程序
    this.setData({
      share: app.globalData.share
    })
    // 定義導航欄的高度   方便對齊
    this.setData({
      height: app.globalData.height
    })
  },
  methods: {
    // 返回上一頁面
    _navback() {
      wx.navigateBack()
    },
    // 計算圖片高度
    imgLoaded(e) {
      this.setData({
        imageHeight:
          e.detail.height *
          (wx.getSystemInfoSync().windowWidth / e.detail.width)
      })
    }
    //返回到首頁
    // _backhome() {
    //   wx.switchTab({
    //     url: '/pages/index/index'
    //   })
    // }
  }
})

 

​ 大概就是這么多,怎么在頁面上用呢

3.具體頁面配置

​ 頁面的HTML,我是內容里面放頁面的東西。

<nav-bar navbar-data='{{nvabarData}}'></nav-bar>
<scroll-view scroll-y style="height: 100%;">
  <view class="scroll-view-item" style='padding-top: {{height}}px;'>我是內容</view>
</scroll-view>

 

​ 頁面的JSON,navigationBarTextStyle是用來配置膠囊顏色的,因為膠囊是微信給的,不能自定義,只能改顏色,所以委屈一下從這里改一下吧

{
  "usingComponents": {
    "nav-bar": "../../components/navbar/navbar"
  },
  "navigationBarTextStyle": "white"
}

 

​ 頁面JS,圖片自己填上地址就好了。注意getApp()不要省。

const app = getApp()
Page({
  data: {
    // 導航頭組件所需的參數
    nvabarData: {
      showCapsule: 1, //是否顯示左上角圖標   1表示顯示    0表示不顯示
      title: '標題', //導航欄 中間的標題
      white: true, // 是就顯示白的,不是就顯示黑的。
      address: '../../images/蒙版組 1@2x.png' // 加個背景 不加就是沒有
    },
    // 導航頭的高度
    height: app.globalData.height * 2 + 20
  }
})

 

4.存在的問題

上拉刷新

​ 我沒有試過哈,不過原生的微信上拉刷新這么用是准定不行了,如果喜歡IOS橡皮筋模式的同學或者想要刷新的同學可以在具體頁面里刪掉scroll-view組件換成view(記得保留那個padding-top!),然后把app.json的禁用滑動刪除掉。具體的我也沒有深入,大家自行解決吧。

5.踩的坑

​ 單純分享下,不看也可以,首先就是設置頁面的背景的時候,我考慮過直接在css上設置background image,但是有一個問題是,小程序的background image 只支持在線的地址或者是base64。我不知道為什么要這么做。真的很迷。但是線上的不穩定,base64太長了,代碼不好看也不好整理,所以考慮了一下還是用Image組件吧。

​ 然后第二個坑又來了,image組件自帶寬高,而且用Mode里的任何值都不能完成需求。如果我設置成width:100%占滿父元素的話,他的長還是默認的340px,所以還是鼓搗了鼓搗,先設置寬度不是100%了,而是通過wx.**getSystemInfoSync**().windowWidth;來獲取的屏幕寬度。然后再通過image組件的事件獲取原圖的長寬,探后計算屏幕寬和原圖寬的率,然后再將這個率乘上原圖長度,就可以獲取到一個占滿父元素又對着比例的圖了,然后給父元素套上overflow:hidden就好了。

​ 第三個坑,就是怎么做背景的拼接,想了想也不是個坑,直接在組件的最外層再加一個一模一樣的image標簽就行了,這樣就做到了標簽上顯示半個背景圖,然后在他的下層又能顯示一個完整的背景圖,因為上面被蓋住了,所以地下的下半部分和導航欄的上半部分背景正好拼接起來,所以問題也就這么解決了。

轉自:https://www.cnblogs.com/wangzirui98/p/11249317.html


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM