微信小程式開發完工+問題彙總

dmego發表於2019-03-04

出發吧一起

前言

經過將近一個多月的開發,我們團隊開發的微信小程式 "出發吧一起" 終於開發完成,現在的線上版本為 2.2.4-beta 版

本文件主要介紹該小程式在開發中所用到的技術,已經在開發中遇到問題的採取的解決方法

小程式簡介

“讓興趣不再孤單,讓愛好不再流浪” 是微信小程式《出發吧一起》的主題,這款小程式旨在解決當代大學生在校園生活中的孤獨感,讓大家找到志同道合的朋友,在跑步、健身、競賽等活動中找到夥伴。利用小程式即開即用,用完就走的特點與交友相結合,它將會是一款高效快捷、無負擔的線下交友利器

本小程式由 bmob 後端雲提供資料處理與儲存支援

小程式碼

歡迎掃描體驗

開發中技術問題彙總

1.使用e.target.dataset的出現問題

在小程式開發過程中,我們經常會用到標籤中屬性的屬性值,我們通常會在 <view> 中 設定 data-*="{{XXX}}" 然後在 JS 裡通過 e.target.dateset.* 來獲取XXX值,但是我經常遇到獲取的是undefined,使用 console.log(e) 檢視輸出資訊會發現,在 e物件中包含兩個物件分別是currentTargettarget,而往往有些時候資料在currentTarget中,

此時可以將程式碼替換成這樣來獲取值

  • WXML
<view bindtap="bintap" data-id="1"></view>
複製程式碼
  • JS
bintap:function(e){
    var id = e.currentTarget.dataset.id;
}
複製程式碼

網上還有一直說法是 data-** 命名的問題,去掉駝峰式命名,純小寫也能解決

2.小程式 textarea 文字框如何顯示實時字數

  • WXML
<view>
    <view>
        <textarea name="content" bindinput="bindTextAreaChange" maxlength="{{noteMaxLen}}" />
        <view class="chnumber">{{noteNowLen}}/{{noteMaxLen}}</view>
    </view>
</view>
複製程式碼
  • JS
data:{
    noteMaxLen: 200,//備註最多字數
    noteNowLen: 0,//備註當前字數
}

  //字數改變觸發事件
  bindTextAreaChange: function (e) {
    var that = this
    var value = e.detail.value,
      len = parseInt(value.length);
    if (len > that.data.noteMaxLen)
      return;
    that.setData({
      content: value, noteNowLen: len
    })
  },
複製程式碼

3.利用 JS 實現模糊查詢

由於我們使用的是 Bmob 後端雲提供的資料處理與儲存支援,根據 Bmob 提供的開發文件,免費版的應用無法進行模糊查詢,看到這裡,再看看已經快完工的活動檢索介面,感受無法言說。正當準備放棄的時候,突然想到一個方法,那就是先把所有的後臺所有資料都存到集合裡,然後根據輸入的檢索值一個個匹配,想到之後馬上就開始著手幹了,先查了一下javaScript 文件,String 物件有一個方法是 indexOf() ,可返回某個指定的字串值在字串中首次出現的位置,這樣就成了,遍歷 所以資料,檢索每一條資料的每個字元,如果出現了則將它加入到檢索結果的集合中.

  • JS
//js 實現模糊匹配查詢
  findEach: function (e) {
    var that = this
    var strFind = that.data.wxSearchData.value; //這裡使用的 wxSearch 搜尋UI外掛,
    if (strFind == null || strFind == "") {
      wx.showToast({
        title: '輸入為空',
        icon: 'loading',
      })
    }
    if (strFind != "") {
      var nPos;
      var resultPost = [];
      for (var i in smoodList) {
        var sTxt = smoodList[i].title || ''; //活動的標題
        nPos = sTxt.indexOf(strFind); 
        if (nPos >= 0) {//如果輸入的關鍵字在該活動標題中出現過,則匹配該活動
          resultPost.push(smoodList[i]); //將該活動加入到搜尋到的活動列表中
        }
      }
      that.setData({
        moodList: resultPost
      })
    }
  },

複製程式碼

更加詳細的程式碼請前往Github檢視

4.使用 JS 將字串格式的時間轉換成幾秒前,幾分鐘前...

由於小程式中涉及評論,加入活動,收藏等一系列包括事件時間的功能,而資料庫中存的時間格式為 2017-11-30 23:36:10 現在想要在介面上不顯示具體時間,而是顯示與當前時間的差,即幾秒前,幾分鐘前等等

實現起來並不複雜,主要思路是先把字串的時間轉換成時間戳,然後與當前的時間戳進行比較,這樣就能轉換成幾秒前、幾分鐘前、幾小時前、幾天前等形式了

  • JS
//字串轉換為時間戳
function getDateTimeStamp(dateStr) {
  return Date.parse(dateStr.replace(/-/gi, "/"));
}
//格式化時間
function getDateDiff(dateStr) {
  var publishTime = getDateTimeStamp(dateStr) / 1000,
    d_seconds,
    d_minutes,
    d_hours,
    d_days,
    timeNow = parseInt(new Date().getTime() / 1000),
    d,

    date = new Date(publishTime * 1000),
    Y = date.getFullYear(),
    M = date.getMonth() + 1,
    D = date.getDate(),
    H = date.getHours(),
    m = date.getMinutes(),
    s = date.getSeconds();
  //小於10的在前面補0
  if (M < 10) {
    M = '0' + M;
  }
  if (D < 10) {
    D = '0' + D;
  }
  if (H < 10) {
    H = '0' + H;
  }
  if (m < 10) {
    m = '0' + m;
  }
  if (s < 10) {
    s = '0' + s;
  }

  d = timeNow - publishTime;
  d_days = parseInt(d / 86400);
  d_hours = parseInt(d / 3600);
  d_minutes = parseInt(d / 60);
  d_seconds = parseInt(d);

  if (d_days > 0 && d_days < 3) {
    return d_days + '天前';
  } else if (d_days <= 0 && d_hours > 0) {
    return d_hours + '小時前';
  } else if (d_hours <= 0 && d_minutes > 0) {
    return d_minutes + '分鐘前';
  } else if (d_seconds < 60) {
    if (d_seconds <= 0) {
      return '剛剛';
    } else {
      return d_seconds + '秒前';
    }
  } else if (d_days >= 3 && d_days < 30) {
    return M + '-' + D + ' ' + H + ':' + m;
  } else if (d_days >= 30) {
    return Y + '-' + M + '-' + D + ' ' + H + ':' + m;
  }
}
複製程式碼

5.微信小程式提交表單清空表單資料

在釋出活動之後,由於表單中的資料沒有清空,給使用者的體驗必定不好,然而小程式的資料互動並不像html + jS 那樣,使用 dataSet({}) 來給賦值,檢視層就能通過非同步的方式活動到值,於是想到,在提交表單後,給這些input都賦值為空,那樣就實現了清空表單的效果,當然,表單中並不只包含input,但是都可以通過這種方式實現清空效果

  • WXML
<form bindsubmit="submitForm">
    <text class="key">活動名稱</text>
    <input name="title"  maxlength="100" value="{{title}}" />
    <button  formType="submit">確定</button>
</form>
複製程式碼
  • JS
submitForm:function(e){
     var title = e.detail.value.title;
     ......
     success: function (res) {
         //將title值設定空
        that.setData({
            title: ''
         }
     }
}
複製程式碼

6.微訊號,QQ號,手機號 正則校驗

由於申請加入活動需要填寫真實姓名,聯絡方式等資訊,為了防止使用者隨意填寫資訊,必須要對這些資訊進行校驗

  • JS
    var wxReg = new RegExp("^[a-zA-Z]([-_a-zA-Z0-9]{5,19})+$"); //微訊號正則校驗
    var qqReg = new RegExp("[1-9][0-9]{4,}"); //QQ號正則校驗
    var phReg = /^1[34578]\d{9}$/; //手機號正則校驗
    var nameReg = new RegExp("^[\u4e00-\u9fa5]{2,4}$"); //2-4位中文姓名正則校驗
複製程式碼

7.使用 Bmob SDK 實現報名成功傳送模板訊息,生成小程式二維碼等

在開發過程中,由於想要實現,當使用者報名成功後如何通知使用者,查閱了小程式的開發文件發現有一個傳送模板訊息的API,再查詢 Bmob 的開發文件,發現實現了這個功能,這個真的太有用了.模板訊息只能再真機上才能傳送成功,經過配置,重要成功,但是有在使用中出現一個問題 ,就是在小程式釋出後 模板訊息中如果帶有 page 引數將不會傳送,但是在開發版中能傳送成功, 這個問題已經反饋了,估計等Bmob小程式SDK更新後會解決這個問題.

具體程式碼我就不寫了,bmob開發文件直達

截圖 & GIF

啟動畫面

訊息通知

地圖模式.gif

Bmob 資料庫表結構設計

使用者表:(_User,自帶表)

|--objectId //Id
|--userPic(String) //使用者頭像
|--username(String) //使用者名稱
|--password(String) //密碼
|--nickname(String) //暱稱
|--sex(Number) //性別
|--userData(Object) //微信登入使用者資料
|--eventJoin(Array) //參加的活動Id 陣列Array
|--eventFavo(Array) //收藏的活動Id 陣列Array
|--feednum(Number) //反饋次數
複製程式碼

活動資訊表:(Events)

|--objectId //活動Id
|--publisher(Pointer-->_User) //發起人
|--title(String) //活動主題
|--content(String) //活動內容
|--actpic(File) //活動宣傳照片
|--acttype(String) //活動類別
{
    1:運動,2:遊戲,3:交友,
    4:旅行,5:讀書,6:競賽,
    7:電影,8: 音樂,9: 其他
}
|--isShow(Number) //是否公開顯示在首頁
|--endtime(String) //組隊截止時間
|--address(String) //活動地點
|--latitude(Number)  //地址緯度
|--longitude(Number) //地址經度
|--peoplenum(String)//人數限制
|--likenum(Number)  //點贊數
|--liker(Array) //點贊人Id集合
|--commentnum(Number) //評論數
|--joinnumber(Number) // 現在參加的人數
|--joinArray(Array) // 現在參加的人集合
複製程式碼

活動資訊擴充套件表:(EventMore)

|--objectId //活動資訊擴充套件表Id
|--event(Pointer-->Events) //活動
|--Status(Number) //活動狀態,(1:準備中,2:進行中,3:已結束)
|--Statusname(String) //活動狀態名稱
|--qrcode(File) //活動群聊二維碼
複製程式碼

評論表:(Comments)

|--objectId //評論Id
|--publisher(Pointer-->_User) //評論釋出者
|--olderUsername(String) //上一條評論人暱稱
|--olderComment(Pointer-->Comments) //上一條評論
|--event(Pointer-->Events) //評論的活動
|--content(String)  //評論內容
複製程式碼

點贊表:(Likes)

|--objectId //點讚的Id
|--liker(Pointer-->_User) //點贊人
|--event(Pointer-->Events) //點讚的活動
複製程式碼

收藏表:(Favos)

|--objectId //收藏的Id
|--favor(Pointer-->_User)  //收藏者
|--event(Pointer-->Events) //收藏的活動
複製程式碼

訊息通知表:(Plyre)

|--objectId //訊息通知的Id
|--fid(String)  //活動釋出者Id(被贊或者被取消讚的人的ID,或者被回覆,被評論的人的ID)(被通知的人)
|--uid(Pointer-->_User)   //訊息通知人
|--wid (String) //被贊,或者取消贊,被評論,或者被回覆,加入,取消加入的活動id
|--avatar (String) //訊息通知人的頭像
|--username (String) //訊息通知人的姓名
|--is_read(Number) //這條訊息是否已讀(1代表已讀,0代表未讀)
|--bigtype(Number) //訊息通知大類(1代表訊息,2代表通知)
|--behavior(Number) //(訊息提醒型別)
{
	1:贊
	2:取消贊
	3:被評論
	4:被回覆
	5:加入活動
	6:取消加入活動
	7:修改了加入資訊
}
複製程式碼

活動聯絡表:(Contact)

|--objectId //聯絡表Id
|--publisher(Pointer-->_User)  //活動釋出者
|--currentUser (Pointer-->_User) //當前使用者
|--event(Pointer-->Events)  //想要加入的活動
|--realname (String) //真實姓名
|--contactWay(String) //聯絡方式(微訊號,手機號,QQ號)
|--contactValue(String) //聯絡方式的號碼
複製程式碼

反饋表:(Feedback)

|--objectId //反饋Id
|--feedUser(Pointer-->_User) //反饋人Id
|--title(String) //反饋標題
|--content(String) //反饋內容
|--feedpic(File) //反饋圖片
|--feedinfo(String) //反饋使用者的裝置資訊 	
複製程式碼

感謝以下開源專案、網站社群

相關文章