你們好,我是@IT·平頭哥聯盟
的首席填坑官——蘇南(South·Su),今天要給你們分享的是最近公司作的一個小程序項目,過程當中的一些好的總結和遇到的坑,但願能給其餘攻城獅帶來些許便利,更但願能像標題所說,作完老闆給你加薪~
html
今天是中秋節的第一天,假日的清晨莫名的醒的特別早,不知道爲何,也許是由於,昨晚公司上線的項目回來的路上,發現了個小bug,內心有些忐忑吧~~
前端
以上純爲扯淡,如今開始一本正經的裝逼,請繫好安全帶,中間過程有可能會開車,請注意安全!!!!!node
最近這個項目跟團隊小夥伴溝通在衆多框架中最後選擇了wepy,沒有直接用原生的,小程序原生就……,你們都懂的,用wepy
框架,給本身帶來了便利,也帶來了很多坑,但縱是如此,我仍是懷着:「縱你虐我千百遍,我仍待你如初戀」的心態去認真把項目作好。
git
wx.showToast({
title: '成功',
icon: 'success',
duration: 2000
})
wx.showModal({
title: '提示',
content: '這是一個模態彈窗',
success: function(res) {
if (res.confirm) {
console.log('用戶點擊肯定')
} else if (res.cancel) {
console.log('用戶點擊取消')
}
}
})
複製代碼
wx.showModal的content的文字是不會居中的(如今不肯定有沒有擴展,能夠設置),依稀記得有一次由於問題差點跟產品經理吵起來,讓文字居中,我說最少要兩小時,當時產品就炸了,什麼鬼???讓文字居中一下要兩小時??兩小時??兩小時??呵呵~走了,後來就下決定本身封裝了一個屬於本身的toast組件,如下爲部分核心代碼
github
<template lang="wxml">
<view class="ui-toast {{ className }}" hidden="{{ !visible }}">
<view class="ui-toast_bd">
<icon wx:if="{{ options.icon}}" type="{{ options.icon }}" size="40" color="{{ options.color }}" class="ui-toast_icon" />
<view class="ui-toast_text">{{ options.text }}</view>
</view>
</view>
</template>
<script>
import wepy from 'wepy';
const __timer__ =1900;
//方法以 : __XX__ 命名,因併入其餘組件中後,可能引發方法名重複
class Toast extends wepy.component {
/**
* 默認數據
*/
data={
list:[
{
type: `success`,
icon: `success`,
className: `ui-toast-success`,
},
{
type: `cancel`,
icon: `cancel`,
className: `ui-toast-cancel`,
},
{
type: `forbidden`,
icon: `warn`,
className: `ui-toast-forbidden`,
},
{
type: `text`,
icon: ``,
className: `ui-toast-text`,
},
],
timer:null,
scope: `$ui.toast`,
animateCss:'animateCss',
className:'',
visible:!1,
options:{
type: ``,
timer: __timer__,
color: `#fff`,
text: `已完成`,
}
}
/**
* 默認參數
*/
__setDefaults__() {
return {
type: `success`,
timer: __timer__,
color: `#fff`,
text: `已完成`,
success() {},
}
}
/**
* 設置元素顯示
*/
__setVisible__(className = `ui-animate-fade-in`) {
this.className = `${this.animateCss} ${className}`;
this.visible = !0;
this.$apply();
}
/**
* 設置元素隱藏
*/
__setHidden__(className = `ui-animate-fade-out`, timer = 300) {
this.className = `${this.animateCss} ${className}`;
this.$apply();
setTimeout(() => {
this.visible = !1;
this.$apply();
}, timer)
}
/**
* 顯示toast組件 @IT·平頭哥聯盟,首席填坑官∙蘇南的專欄,梅斌的專欄
* @param {Object} opts 配置項
* @param {String} opts.type 提示類型
* @param {Number} opts.timer 提示延遲時間
* @param {String} opts.color 圖標顏色
* @param {String} opts.text 提示文本
* @param {Function} opts.success 關閉後的回調函數
*/
__show__(opts = {}) {
let options = Object.assign({}, this.__setDefaults__(), opts)
const TOAST_TYPES = this.list;
TOAST_TYPES.forEach((value, key) => {
if (value.type === opts.type) {
options.icon = value.icon;
options.className = value.className
}
})
this.options = options;
if(!this.options.text){
return ;
};
clearTimeout(this.timer);
this.__setVisible__();
this.$apply();
this.timer = setTimeout(() => {
this.__setHidden__()
options.success&&options.success();
}, options.timer);
}
__info__(args=[]){
let [ message, callback, duration ] = args;
this.__show__({
type: 'text',
timer: (duration||__timer__),
color: '#fff',
text: message,
success: () => {callback&&callback()}
});
}
__success__(args=[]){
let [ message, callback, duration ] = args;
this.__show__({
type: 'success',
timer: (duration||__timer__),
color: '#fff',
text: message,
success: () => {callback&&callback()}
});
}
__warning__(args){
let [ message, callback, duration ] = args;
this.__show__({
type: 'forbidden',
timer: (duration||__timer__),
color: '#fff',
text: message,
success: () => {callback&&callback()}
});
}
__error__(args){
let [ message, callback, duration ] = args;
this.__show__({
type: 'cancel',
timer: (duration||__timer__),
color: '#fff',
text: message,
success: () => {callback&&callback()}
});
}
__showLoading__(options){
wx.showLoading({
title: (options&&options.title||"加載中"),
});
}
__hideLoading__(){
wx.hideLoading();
}
onLoad(){
this.$apply()
}
}
export default Toast;
</script>複製代碼
調用示例:
<template>
<view class="demo-page">
<Toast />
<Modals />
</view>
</template>
<script>
import wepy from 'wepy' //@IT·平頭哥聯盟
import Toast from '../components/ui/Toast'
import Modals from '../components/ui/Modals'
import {fetchJson} from '../utils/fetch';
export default class Index extends wepy.page {
config = {
navigationBarBackgroundColor: "#0ECE8D",
navigationBarTextStyle:"white",
navigationBarTitleText: ''
}
components = {
Toast: Toast,
Modals: Modals
}
methods = {
tapToast(){
this.$invoke("Toast","__success__",[`您已提交成功,感謝您的支持與配合`]);
}
}
}
</script>複製代碼
cookie
、localStorage
、sessionStorage
等這些,特性就不一一說明了,小程序裏你們都知道,數據存儲只能調用 wx.setStorage、wx.setStorageSync,至關於h5的localStorage
,而 localStorage
是不會過時的,這個你們都知道,並且在不少的面試中,面試官都會問到這個問題,怎麼讓localStorage像cookie
同樣,只存兩小時、兩天、甚至只存兩分鐘呢?今天帶你解惑,讓你在職場面試中又減小一個難題,這也是咱們項目中一直在用的方式,小程序中也一樣實用:class storage {
constructor(props) {
this.props = props || {}
this.source = wx||this.props.source;
}
get(key) {
const data = this.source,
timeout = (data.getStorageSync(`${key}__expires__`)||0)
// 過時失效
if (Date.now() >= timeout) {
this.remove(key)
return;
}
const value = data.getStorageSync(key)
return value
}
// 設置緩存
// timeout:過時時間(分鐘)
set(key, value, timeout) {
let data = this.source
let _timeout = timeout||120;
data.setStorageSync(key,(value));
data.setStorageSync(`${key}__expires__`,(Date.now() + 1000*60*_timeout));
return value;
}
remove(key) {
let data = this.source
data.removeStorageSync(key)
data.removeStorageSync(`${key}__expires__`)
return undefined;
}
}
module.exports = new storage();
複製代碼
其實很簡單,你們看了以後就都 「哦,原來還能夠這樣」 懂了,只是一時沒想到而已,就是個小技巧,每次在存儲的時候同時也存入一個時效時間戳,而在獲取數據前,先與當前時間比較,若是小於當前時間則過時了,直接返回空的數據。面試
nodejs
以前,前端好像一直都在爲處理不一樣環境下調用對應的API而煩惱,作的更多的就是用域名來進行判斷,固然也有些高級一點的作法,後端在頁面渲染的時候,存一個變量到cookie
裏或者在頁面輸出一個全局的api變量(創建在沒有先後端分離的基礎上),到了小程序一樣也是如此,每次都要手動改環境,那麼一個項目可能有不一樣的業務,要調用不一樣域名api,又有不一樣的環境區分,怎麼維護會比較好呢??env/dev.js
//本地環境
module.exports = {
wabApi:{
host:"https://dev-ali.southsu.com/XX/api/**",
},
questionApi:{
host:"https://dev-ali.bin.com/question/api/**/question",
},
mockApi:{
host:"https://easy.com/mock/594635**c/miniPrograms"
},
inWelApi: {
host: "https://dev.**.com/Wab/api/escene/v2"
}
};
複製代碼
import dev from './env/dev'; //本地或開發
import uat from './env/pre'; //體驗環境
import prd from './env/prd'; //線上
var ENV = "prd"; //'dev | uat | prd';
let _base_ = {
dev,
uat,
prd
}[ENV];
var config = {
ENV,
baseAPI:{..._base_, env : ENV },
appID:"wx*****b625e", //公司帳號(指數)appid
isAuthorization:true,
'logId': 'gVDSMH****HAas4qSSOTb-gzGzoHsz',
'logKey': 'pxFOg****Jn3JyjOVr',
questionnaireNo:'z**Insu' // 問卷調查編號
};
export const __DEBUG__ = (ENV!="prd");
export default config;
複製代碼
請求調用api處理的示例
import wepy from 'wepy'
import _login_ from './login';
import config,{__DEBUG__} from './config';
import 'wepy-async-function';
export const fetchJson = (options)=>{
/*
* 請求前的公共數據處理,首席填坑官∙蘇南的專欄,梅斌的專欄
* @ param {String} url 請求的地址
* @ param {String} Type 請求類型
* @ param {String} sessionId 用戶userToken
* @ param {Boolean} openLoad 開啓加載提示,默認開啓,true-開,false-關
* @ param {function} StaticToast 靜態提示方法 ,詳細說明請參考 components/ui/Toast
* @ param {Object} header 重置請求頭
* @ param {Boolean} isMandatory 是否強制用戶受權,獲取用戶信息
*/
StaticToast = getCurrentPages()[getCurrentPages().length - 1];
let { url,openLoad=true, type, data={},header={}, ...others } = options||{};
let sessionId = (Storage.get(__login__.server+'_userToken')||"");
/*Start */
var regExp = /\/(.*?)\//,
hostkey = url.match(regExp)[1];
let baseUrl = config.baseAPI[hostkey].host;
url = url.replace(regExp, '/');
/*End */
__DEBUG__&&console.log('#--baseUrl:', baseUrl);
__DEBUG__&&console.log('#--請求地址:', `${baseUrl}${url}`);
__DEBUG__&&console.log('----------分割線---------------');
openLoad&&StaticToast.__showLoading__();
return new Promise((resolve, reject) => {
return wepy.request({
url:`${baseUrl}${url}`,
method:(type || 'POST'),
data,
header:{
"t9oken":sessionId,
'content-type': 'application/json',
// 'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8',
...header
},
success:(res)=>{
StaticToast.__hideLoading__();
return resolve(resHandler(res,options));
},
error:(err,status)=>{
StaticToast.__hideLoading__();
return reject(errorHandler(err,options,err.statusCode));
}
});
})
複製代碼
業務調用示例json
fetchJson({
type:"post",
// url:"/mockApi/service/XXX", 最後請求獲得的地址是 https://easy.com/mock/594635**c/miniPrograms/service/XXX (域名不一樣環境不同,在config裏的 ENV baseAPI控制)
data:{
name:"蘇南"
},
success:res=>{
console.log(`你們好,我是@IT·平頭哥聯盟-蘇南`,res)
}
})
複製代碼
wepy
框架中每一個組件內的生命週期回調 onload,只要是引入了組件,無論你視圖有沒有渲染,他都會執行,致使某些業務邏輯用不上它的時候也執行了產生異常(固然爲個鍋< 小程序 >確定說我不背~^~ ),詳細看連接:github.com/Tencent/wep… , github.com/Tencent/wep… ,不知道後面有沒有人解決。a
標籤,a
標籤啊,屬性沒有,那還要它何用啊,你都不要我跳轉,我還要用你嗎?b、i、span、em……哪一個我不能用?不知道設計這個組件的人是否是腦被驢踢了(願老天保佑,我在這罵他,可千萬別被看到了,哈哈~),又是業務需求後臺配置的內容有連接,沒辦法,來吧,搞吧,往死裏搞吧,一切的推脫都是你技術low的藉口(你看,你看,別人的怎麼能夠跳轉啊,別人怎麼作到的?給我一刀,我能把產品砍成渣),因此有了後面的填坑:<template>
<view class="test-page">
<button @tap="cutting">點擊解析html</button>
<view wx:if="{{result.length>0}}" class="parse-list">
<view class="parse-view" wx:for="{{result}}" wx:key="unique" wx:for-index="index" wx:for-item="items">
<block wx:if="{{items.children&&items.children.length}}">
<block wx:for="{{items.children}}" wx:for-item="child" >
<text wx:if="{{child.type == 'link'}}" class="parse-link" @tap="goToTap({{child.link}})">{{child.value}}</text>
<text class="parse-text" wx:else>{{child.value}}</text>
</block>
</block>
<text class="parse-text" wx:else>{{items.value}}</text>
</view>
</view>
<Toast />
<Modals />
</view>
</template>
<script>
//首席填坑官∙蘇南的專欄,梅斌的專欄
import wepy from 'wepy'
import { connect } from 'wepy-redux'
import Toast from '../components/ui/Toast'
import Modals from '../components/ui/Modals'
import {fetchJson} from '../utils/fetch';
import Storage from "../utils/storage";
function wxHtmlParse(htmlStr=''){
if(!htmlStr){
return []
};
const httpExp =/(http:\/\/|https:\/\/)((\w|=|\?|\.|\/|\&|-)+)/g;//提取網址正則
const aExp=/<a.[^>]*?>([(^<a|\s\S)]*?)<\/a>/ig; //a標籤分割正則
let cuttingArr = htmlStr.split(/[\n]/);
let result = [];
//有a標籤的html處理
let itemParse = (itemHtml='')=>{
let itemCutting = itemHtml.split(aExp)||[];
let itemResult = [];
for(var i = 0;i<itemCutting.length;i++){
let _html = itemCutting[i];
if(_html!==''){
let itemData = {
value:_html,
type:'text',
class:"parse-text"
};
let matchTag = itemHtml.match(aExp)||[]; //再次匹配有 a 標籤的
if(matchTag.length){
let itemIndex = matchTag.findIndex((k,v)=>(k.indexOf(_html)!==-1));
if(itemIndex>=0){
let link = matchTag[itemIndex].match(httpExp)[0];
itemData.type = 'link';
itemData.link = link;
itemData.class = "parse-link";
};
};
itemResult.push(itemData)
}
};
return itemResult;
};
cuttingArr.map((k,v)=>{
let itemData = {type : "view",class:"parse-view"};
let isATag = k.match(aExp);
if(isATag){
itemData.children = itemParse(k);
}else{
itemData.value = k;
};
result.push(itemData);
return k;
}) ;
return result;
};
export default class Index extends wepy.page {
config = {
navigationBarBackgroundColor: "#0ECE8D",
navigationBarTextStyle:"white",
navigationBarTitleText: '小程序解析數據中的a標籤'
}
components = {
Toast: Toast,
Modals: Modals
}
data = {
html:'你們好,我是蘇南(South·Su),\n職業:@IT·平頭哥聯盟-首席填坑官,\n身高:176cm,\n性別:男,\n性取向:女,\n公司:目前就任於由騰訊、阿里、平安三巨頭合資的一家互聯網金融公司深圳分公司某事業部、,\n簡介:寶劍鋒從磨礪出 梅花香自苦寒來,認真作本身,樂於分享,但願能盡綿薄之力 助其餘同窗少走一些彎路!,gitHub:https://github.com/meibin08/,\n興趣:跑步、羽毛球、登山、音樂、看書、分享本身的微薄知識幫助他人……,\n其餘:想了解更多嗎?能夠加入<a href="https://honeybadger8.github.io/blog/#/">386485473交流羣</a>,也能夠給我電話<a href="https://github.com/meibin08/">134XX852xx5</a> ,開玩笑啦',
result:[]
}
methods = {
cutting(e){
this.result = wxHtmlParse(this.html);
console.log(`result`,this.result);
this.$apply();
},
}
}
</script>複製代碼
PS:完整示例源碼 來啦~,以爲不錯記得 Star、Star
、Watch
哦,感謝!redux
今天的分享就到這裏,寫了蠻久,最近纔在開始嘗試寫博客,新手上路中,若是文章中有不對之處,煩請各位大神斧正。若是你以爲這篇文章對你有所幫助,請記得點贊哦~,想了解我更多?歡迎關注下方公衆號哦小程序
做者:蘇南 -首席填坑官
交流羣:912594095,公衆號: honeyBadger8
本文原創,著做權歸做者全部。商業轉載請聯繫 @IT·平頭哥聯盟得到受權,非商業轉載請註明原連接及出處。
本次徵文活動的連接:juejin.im/post/5b923a…後端