微信小程序本质上来说就是一个 HTML 5(移动网页) 应用,用view、scoll-view代替了div标签等,换汤不换药。在微信中运行时,微信小程序获得更多的系统权限。首先是数据缓存能力,这可以让用户在打开一个小程序的时候将程序的主要框架缓存到微信上,下一次就可以快速打开了。微信创始人张小龙曾说过,微信应用号希望实现的目标是“用完即走,无需安装和卸载”,也就是说以后当你要使用一个应用时,只需要在微信里搜索就可以直接使用了,如摩拜、美团等小型使用低频的app使用该套技术可大量节省开发成本。最近又新增了开放个人开发、公众号关联推送的加强,可谓使用场景不容小觑。
由于本人吃货一枚,家门口有一家KFC,之前KFC的app经常有一些福利卷,既然用惯了这个便捷实惠的app,于是就做它了。 言归正传,先来分析一下一步一步该做啥,做一个小demo成就感还是满满的。
附近位置选择-联动菜单导航-模拟数据-抽屉式购物车-获取用户微信信息-页面传值-数据生成订单
https://xurenjie.cn:3000/img/KFC/KFC_gif.gif
抱歉!!GIF太卡了太卡了,简易clone下来本地跑起来效果最佳 直接点餐会自动为你找到最近的餐厅,不过离最近的kfc太远的不太行
- 开发调试工具: 微信小程序平台教程,安装好最基本的啦!稍微过一遍简易文档。
- 开发‘字典’文档: 微信小程序开发教程手册文档功能使用方法都在上面挺全的。通过 阅读文档了解页面的搭建、页面路由、导航跳转、数据绑定、条件渲染、列表渲染、scroll-view、request、radio、modal、toast、map、位置、数据缓存、动画、验证信息。
- easy-mock: easy-mock模拟后端数据,后面会简单介绍配置。
- 腾讯地图API: 腾讯地图小程序有地点搜索、关键词输入提示、逆地址解析(坐标位置描述)地址解析(地址转坐标)距离计算、获取城市列表、获取城市区县的一步配置教程。
- WeUI WeUI github微信专用的结构样式组件库,加速开发速度,BEM规范代码,增强可读性
├── app.js
├── app.json
├── app.wxss
├── pages
│ ├── .DS_Store
│ ├── KFC
│ │ ├── KFC.js
│ │ ├── KFC.wxml
│ │ └── KFC.wxss
│ ├── card
│ │ ├── card.js
│ │ ├── card.wxml
│ │ └── card.wxss
│ ├── halladdress
│ │ ├── .DS_Store
│ │ ├── halladdress.js
│ │ ├── halladdress.wxml
│ │ └── halladdress.wxss
│ ├── index
│ │ ├── index.js
│ │ ├── index.wxml
│ │ └── index.wxss
│ ├── logs
│ │ ├── logs.js
│ │ ├── logs.json
│ │ ├── logs.wxml
│ │ └── logs.wxss
│ ├── menu
│ │ ├── index.html
│ │ ├── menu.js
│ │ ├── menu.wxml
│ │ └── menu.wxss
│ ├── order
│ │ ├── order.js
│ │ ├── order.wxml
│ │ └── order.wxss
│ └── takeout
│ ├── message
│ │ ├── message.js
│ │ ├── message.wxml
│ │ └── message.wxss
│ ├── qqmap-wx-jssdk.min.js
│ ├── takeout.js
│ ├── takeout.wxml
│ └── takeout.wxss
├── style
│ ├── .DS_Store
│ └── weui.wxss
└── utils
└── util.js
"pages": [
"pages/index/index", // 首页
"pages/KFC/KFC", // K金商城页
"pages/menu/menu", // 菜单页
"pages/card/card", // 卡包页
"pages/order/order", // 订单页
"pages/takeout/takeout", // 外卖地图页
"pages/takeout/message/message", // 填写外卖信息页
"pages/halladdress/halladdress", // 附近餐厅页
"pages/logs/logs" // 日志页
]
我们可以通过官网的文档或W3C教程上初始化了一个小程序目录,小程序的每个页面都放在pages目录下 每次添加一个新页面,都需要先在app.json.page下注册。
mock.js大红大紫,让前端独立于后端,用它来模拟KFC数据 不太清楚使用的同学可以参考:
因为菜单中每个左侧的分类对应一组数据,在右侧也需要渲染类名,因此简单模拟结构
[{ "title": "这里放左边列表的组名",
"foodsIndex": [{
"name": "这里放每个食物的名字",
"price": "11.0",
"url": "https://imgm.4008823823.com.cn/kfcmwos/img//S/269_116012.jpg"
},
{},{},{}]
用toast优化耗时加载
wx.showToast({
title: '地图加载中',
icon: 'loading',
duration: 0,
mask: true
})
画图完成后用回调将Toast去除
this.mapCtx = wx.createMapContext('myMap', function () {
wx.hideToast();
})
WXML:
<map id="myMap" longitude="{{longitude}}" latitude="{{latitude}}"
style="width: 100%; height: 100%" markers="{{markers}}" covers="{{covers}}" scale="18">
</map>
以搜附近地点渲染至页面列表为例
- 引入核心类 腾讯地图小程序版下载js并获取key
let QQMapWX = require('qqmap-wx-jssdk.min.js');
let demo = new QQMapWX({
key: '5Q2BZ-O3W24-V6DUN-DZ4Z7-H427K-WCB7R' // 必填
});
- 调用APIreverseGeocoder
demo.reverseGeocoder({
location: {
latitude: _latitude,
longitude: _longitude
},
get_poi: 1,
success: function (res) {
// console.log(res);
},
fail: function (res) {
console.log(res);
},
complete: function (res) {
console.log(res);
that.setData({
pois: res.result.pois
})
}
});
通过setData() 我们的数据就传到data上去中了便用此渲染页面上去,that保持对原page对象的引用哟
<view class="address-item" wx:for="{{pois}}" wx:for-item="poi"
data-name="{{poi.address}}" catchtap="ToDetailPage">
<image src="../../image/position.png" data-name="{{poi.address}}"
catchtap="ToDetailPage"></image>
<text catchtap="ToDetailPage" data-name="{{poi.address}}">{{poi.address}}</text>
</view>
以搜周围的KFC为例
demo.search({
keyword: '肯德基',
location: {
latitude: _latitude,
longitude: _longitude
},
success: function (res) {
// console.log(res);
},
fail: function (res) {
// console.log(res);
},
complete: function (res) {
console.log(res.data[0].location.lat,res.data[0].location.lng)
console.log(res.data[0])
// .address._distance
that.setData({
poi: res.data[0].address,
distance: res.data[0]._distance,
latitude: res.data[0].location.lat,
longitude: res.data[0].location.lng,
markers: [{
latitude: res.data[0].location.lat,
longitude: res.data[0].location.lng,
name: 'KFC',
desc: 'KFC在您身边'
}]
})
}
});
换汤不换药核心还是通过setData改变data从而让页面显示当前kfc,没有用输入框搜索,用设置的自动搜索
- 侧nav与内容区的联动
- 商品的增减改变总价格并弹出购物车和抽屉详细面板
- 抽屉的动画
- 在抽屉面板中改变商品数量,没有则隐藏组件
gif炸了的话直接看下面部分的图吧,忽略底下的购物车
左边点击菜单的不同种类,右边转到相应的的内容 这里的实现,用到了scroll-view的API 给每个右边的内容对象渲染时附上id
<view class="food-list" wx:for="{{foodArray}}" wx:for-item="item" id="foodtype{{index}}">
再给每个nav的点击事件dataset解析一下
let goPage = e.currentTarget.id
this.setData({
scroll_into_view: "foodtype" + goPage
})
我在这里的做法是给每个商品都赋了一个dataset,以便点击不同的商品让不同的对象进入购物车数组,通过e.target.dataset拿到
// 是否有同种商品判断
if (this.data.shoppingList.length > 0) {
// 商品名是否相同判断,不重复添加同名商品
let isHave = this.data.shoppingList.findIndex(item => item.name == e.target.dataset.name)
if (isHave != -1) {
that.data.shoppingList[isHave].num++
} else {
// 购物车数组加进新的一样食品
that.data.shoppingList.push({
price: e.target.dataset.price,
name: e.target.dataset.name,
num: itemNum
})
// 动画效果的长度添加
move_length++
}
// 没有商品时直接添加
} else {
this.data.shoppingList.push({
price: e.target.dataset.price,
name: e.target.dataset.name,
num: itemNum
})
move_length++
}
可参照API 我在这里是做了一个增加商品时,抽屉往上滚动,删除为空时抽屉向下滚动
data: {
totalCount: 0, // 购物车的总数量
movelength: 0, // 上移或下拉动画的单位距离
cartIsHidden: true, // 购物车是否隐藏
cartIndexIsHidden: true, // 购物车详情菜单是否隐藏
animationData: {} // 动画动作对象
}
滚动动画初始设置
let animation = wx.createAnimation({
duration: 400,
timingFunction: "linear",
delay: 0
});
动画产生的效果就以bottom的变化而产生
let mlength = move_length * 55;
if (move_length > 1) {
mlength = 55 + (move_length - 1) * 65;
}
this.animation = animation
animation.bottom(mlength).step()
加入动画序列,并设置好movelength
this.setData({
animationData: animation.export()
})
this.setData({
shoppingList: shopping_list,
totalPrice: total_price,
totalCount: total_count,
// 购物车当有商品时弹出
cartIsHidden: false,
movelength: move_length
})
}
- 设置全局的数据缓存看这API
- url 附带字串传值
- 引入事件订阅和发布onfire.js(最近准备撸撸)
比如这个页面,它的所有数据都来之于之前的选择 我在自己项目里目前用的是本地存储的方式,比如地址的设置获取
在选择页设置本地存储
let OrderAddress = {
address: [],
isHall: false
}
//遍历去重
let item = OrderAddress.address.find(item=>item==event.target.dataset.name)
if(!item){
OrderAddress.address.push(event.target.dataset.name)
}
wx.setStorage({
key: "OrderAddress",
data: OrderAddress,
});
wx.navigateTo({
url:'/pages/takeout/message/message'
})
在订单页拿到地址,对api不熟悉多console.log几下,没有什么解决不了的~
wx.getStorage({
key: 'OrderAddress',
success: function (res) {
console.log(res.data);
that.setData({
address: res.data.address[0],
elementToggle: res.data.isHall
})
}
})
-
因为还没有上rpx:真机样式有差距;同样设置几行文本,高度是不一致的。
-
腾讯地图的大坑z-index,本来用地图的api想做一个搜索自动提示,但绘制的地图是微信内置的,z-index再高都根本无法覆盖在地图上面,解决办法是另外跳入一个页面处理
-
需要申请合法域名,请求里合法域名有个数限制。
-
页面内跳转不能超过5级。