微信小程序--优购商城项目(4)
文章目录
- 前言
- 五、搜索
- 1、创建 search 分支
- 2、自定义搜索组件
- (1)自定义 my-search 组件
- (2) 通过自定义属性增强组件的通用性
- (3)为自定义组件封装 click 事件
- (4)实现首页搜索组件的吸顶效果
- 3、搜索建议
- (1)渲染搜索页面的基本结构
- (2)实现搜索框自动获取焦点
- (3)实现搜索框的防抖处理
- (4)根据关键词查询搜索建议列表
- (5)渲染搜索建议列表
- 4、搜索历史
- (1)渲染搜索历史记录的基本结构
- (2)实现搜索建议和搜索历史的按需展示
- (3)将搜索关键词存入 historyList
- (4)解决关键字前后顺序的问题
- (5)解决关键词重复的问题
- (6)将搜索历史记录持久化存储到本地
- (7)清空搜索历史记录
- (8)点击搜索历史跳转到商品列表页面
- 5、分支的合并与提交
前言
基于 uni-app 开发的微信小程序商城项目
- 备用接口地址:https://api-hmugo-web.itheima.net
- 搜索页面
五、搜索
1、创建 search 分支
运行如下的命令,基于 master 分支在本地创建 search 子分支,用来开发搜索相关的功能:
git checkout -b search
2、自定义搜索组件
(1)自定义 my-search 组件
在项目根目录的 components 目录上,鼠标右键,选择 新建组件,填写组件信息后,最后点击 创建 按钮:
在分类页面的 UI 结构中,直接以标签的形式使用 my-search 自定义组件:
<!-- 使用自定义的搜索组件 -->
<my-search></my-search>
- 定义 my-search 组件的 UI 结构如下:
<template><view class="my-search-container"><!-- 使用 view 组件模拟 input 输入框的样式 --><view class="my-search-box"><uni-icons type="search" size="17"></uni-icons><text class="placeholder">搜索</text></view></view>
</template>
注意:在当前组件中,我们使用 view 组件模拟 input 输入框的效果;并不会在页面上渲染真正的 input 输入框
- 美化自定义 search 组件的样式:
.my-search-container {background-color: #c00000;height: 50px;padding: 0 10px;display: flex;align-items: center;
}.my-search-box {height: 36px;background-color: #ffffff;border-radius: 15px;width: 100%;display: flex;align-items: center;justify-content: center;.placeholder {font-size: 15px;margin-left: 5px;}
}
- 由于自定义的 my-search 组件高度为 50px,因此,需要重新计算分类页面窗口的可用高度:
onLoad() {const sysInfo = uni.getSystemInfoSync()// 可用高度 = 屏幕高度 - navigationBar高度 - tabBar高度 - 自定义的search组件高度this.wh = sysInfo.windowHeight - 50
}
(2) 通过自定义属性增强组件的通用性
为了增强组件的通用性,我们允许使用者自定义搜索组件的 背景颜色 和 圆角尺寸。
- 通过 props 定义 bgcolor 和 radius 两个属性,并指定值类型和属性默认值:
props: {// 背景颜色bgcolor: {type: String,default: '#C00000'},// 圆角尺寸radius: {type: Number,// 单位是 pxdefault: 18}
}
- 通过属性绑定的形式,为 .my-search-container 盒子和 .my-search-box 盒子动态绑定 style 属性:
<view class="my-search-container" :style="{'background-color': bgcolor}"><view class="my-search-box" :style="{'border-radius': radius + 'px'}"><uni-icons type="search" size="17"></uni-icons><text class="placeholder">搜索</text></view>
</view>
- 移除对应 scss 样式中的 背景颜色 和 圆角尺寸:
.my-search-container {// 移除背景颜色,改由 props 属性控制// background-color: #C00000;height: 50px;padding: 0 10px;display: flex;align-items: center;
}.my-search-box {height: 36px;background-color: #ffffff;// 移除圆角尺寸,改由 props 属性控制// border-radius: 15px;width: 100%;display: flex;align-items: center;justify-content: center;.placeholder {font-size: 15px;margin-left: 5px;}
}
(3)为自定义组件封装 click 事件
- 在 my-search 自定义组件内部,给类名为 .my-search-box 的 view 绑定 click 事件处理函数:
<view class="my-search-box" :style="{'border-radius': radius + 'px'}" @click="searchBoxHandler"><uni-icons type="search" size="17"></uni-icons><text class="placeholder">搜索</text>
</view>
- 在 my-search 自定义组件的 methods 节点中,声明事件处理函数如下:
methods: {// 点击了模拟的 input 输入框searchBoxHandler() {// 触发外界通过 @click 绑定的 click 事件处理函数this.$emit('click')}
}
- 在分类页面中使用 my-search 自定义组件时,即可通过 @click 为其绑定点击事件处理函数:
<!-- 使用自定义的搜索组件 -->
<my-search @click="gotoSearch"></my-search>
同时在subpkg中新建search页面,在分类页面中,定义 gotoSearch 事件处理函数如下:
methods: {// 跳转到分包中的搜索页面gotoSearch() {uni.navigateTo({url: '/subpkg/search/search'})}
}
(4)实现首页搜索组件的吸顶效果
- 在 home 首页定义如下的 UI 结构:
<!-- 使用自定义的搜索组件 -->
<view class="search-box"><my-search @click="gotoSearch"></my-search>
</view>
- 在 home 首页定义如下的事件处理函数:
gotoSearch() {uni.navigateTo({url: '/subpkg/search/search'})
}
- 通过如下的样式实现吸顶的效果:
.search-box {// 设置定位效果为“吸顶”position: sticky;// 吸顶的“位置”top: 0;// 提高层级,防止被轮播图覆盖z-index: 999;
}
3、搜索建议
(1)渲染搜索页面的基本结构
- search页面定义如下的 UI 结构
<view class="search-box"><!-- 使用 uni-ui 提供的搜索组件 --><uni-search-bar @input="input" :radius="100" cancelButton="none"></uni-search-bar>
</view>
- 修改 components -> uni-search-bar -> uni-search-bar.vue 组件,将默认的白色搜索背景改为 #C00000 的红色背景:
.uni-searchbar {/* #ifndef APP-NVUE */display: flex;/* #endif */flex-direction: row;position: relative;padding: 16rpx;/* 将默认的 #FFFFFF 改为 #C00000 */background-color: #c00000;
}
- 实现搜索框的吸顶效果:
.search-box {position: sticky;top: 0;z-index: 999;
}
- 定义如下的 input 事件处理函数:
methods: {input(e) {// e是最新的搜索内容console.log(e)}
}
(2)实现搜索框自动获取焦点
- 修改 components -> uni-search-bar -> uni-search-bar.vue 组件,把 data 数据中的 show 和 showSync 的值,从默认的 false 改为 true 即可:
data() {return {show: true,showSync: true,searchVal: ""}
}
- 使用手机扫码预览,即可在真机上查看效果
(3)实现搜索框的防抖处理
- 在 data 中定义防抖的延时器 timerId 如下:
data() {return {// 延时器的 timerIdtimer: null,// 搜索关键词kw: ''}
}
- 修改 input 事件处理函数如下:
input(e) {// 清除 timer 对应的延时器clearTimeout(this.timer)// 重新启动一个延时器,并把 timerId 赋值给 this.timerthis.timer = setTimeout(() => {// 如果 500 毫秒内,没有触发新的输入事件,则为搜索关键词赋值this.kw = econsole.log(this.kw)}, 500)
}
(4)根据关键词查询搜索建议列表
- 在 data 中定义如下的数据节点,用来存放搜索建议的列表数据:
data() {return {// 搜索结果列表searchResults: []}
}
- 在防抖的 setTimeout 中,调用 getSearchList 方法获取搜索建议列表:
this.timer = setTimeout(() => {this.kw = e.value// 根据关键词,查询搜索建议列表this.getSearchList()
}, 500)
- 在 methods 中定义 getSearchList 方法如下:
// 根据搜索关键词,搜索商品建议列表
async getSearchList() {// 判断关键词是否为空if (this.kw === '') {this.searchResults = []return}// 发起请求,获取搜索建议列表const { data: res } = await uni.$http.get('/api/public/v1/goods/qsearch', { query: this.kw })if (res.meta.status !== 200) return uni.$showMsg()this.searchResults = res.message
}
(5)渲染搜索建议列表
- 定义如下的 UI 结构:
<!-- 搜索建议列表 -->
<view class="sugg-list"><view class="sugg-item" v-for="(item, i) in searchResults" :key="i" @click="gotoDetail(item.goods_id)"><view class="goods-name">{{item.goods_name}}</view><uni-icons type="arrowright" size="16"></uni-icons></view>
</view>
- 美化搜索建议列表:
.sugg-list {padding: 0 5px;.sugg-item {font-size: 12px;padding: 13px 0;border-bottom: 1px solid #efefef;display: flex;align-items: center;justify-content: space-between;.goods-name {// 文字不允许换行(单行文本)white-space: nowrap;// 溢出部分隐藏overflow: hidden;// 文本溢出后,使用 ... 代替text-overflow: ellipsis;margin-right: 3px;}}
}
- 点击搜索建议的 Item 项,跳转到商品详情页面:
gotoDetail(goods_id) {uni.navigateTo({// 指定详情页面的 URL 地址,并传递 goods_id 参数url: '/subpkg/goods_detail/goods_detail?goods_id=' + goods_id})
}
4、搜索历史
(1)渲染搜索历史记录的基本结构
- 在 data 中定义搜索历史的假数据:
data() {return {// 搜索关键词的历史记录historyList: ['a', 'app', 'apple']}
}
- 渲染搜索历史区域的 UI 结构:
<!-- 搜索历史 -->
<view class="history-box"><!-- 标题区域 --><view class="history-title"><text>搜索历史</text><uni-icons type="trash" size="17"></uni-icons></view><!-- 列表区域 --><view class="history-list"><uni-tag :text="item" v-for="(item, i) in historyList" :key="i"></uni-tag></view>
</view>
- 美化搜索历史区域的样式:
.history-box {padding: 0 5px;.history-title {display: flex;justify-content: space-between;align-items: center;height: 40px;font-size: 13px;border-bottom: 1px solid #efefef;}.history-list {display: flex;flex-wrap: wrap;.uni-tag {margin-top: 5px;margin-right: 5px;}}
}
(2)实现搜索建议和搜索历史的按需展示
当搜索结果列表的长度不为 0的时候(searchResults.length !== 0),需要展示搜索建议区域,隐藏搜索历史区域
当搜索结果列表的长度等于 0的时候(searchResults.length === 0),需要隐藏搜索建议区域,展示搜索历史区域
使用 v-if 和 v-else 控制这两个区域的显示和隐藏,示例代码如下
<!-- 搜索建议列表 -->
<view class="sugg-list" v-if="searchResults.length !== 0"><!-- 省略其它代码... -->
</view><!-- 搜索历史 -->
<view class="history-box" v-else><!-- 省略其它代码... -->
</view>
(3)将搜索关键词存入 historyList
- 直接将搜索关键词 push 到 historyList 数组中即可
methods: {// 根据搜索关键词,搜索商品建议列表async getSearchList() {// 省略其它不必要的代码...// 1. 查询到搜索建议之后,调用 saveSearchHistory() 方法保存搜索关键词this.saveSearchHistory()},// 2. 保存搜索关键词的方法saveSearchHistory() {// 2.1 直接把搜索关键词 push 到 historyList 数组中this.historyList.push(this.kw)}
}
- 上述实现思路存在的问题:
- 关键词前后顺序的问题(可以调用数组的 reverse() 方法对数组进行反转)
- 关键词重复的问题(可以使用 Set 对象进行去重操作
(4)解决关键字前后顺序的问题
data 中的 historyList 不做任何修改,依然使用 push 进行末尾追加
定义一个计算属性 historys,将 historyList 数组 reverse 反转之后,就是此计算属性的值:
computed: {historys() {// 注意:由于数组是引用类型,所以不要直接基于原数组调用 reverse 方法,以免修改原数组中元素的顺序// 而是应该新建一个内存无关的数组,再进行 reverse 反转return [...this.historyList].reverse()}
}
- 页面中渲染搜索关键词的时候,不再使用 data 中的 historyList,而是使用计算属性 historys:
<view class="history-list"><uni-tag :text="item" v-for="(item, i) in historys" :key="i"></uni-tag>
</view>
(5)解决关键词重复的问题
- 修改 saveSearchHistory 方法如下:
// 保存搜索关键词为历史记录
saveSearchHistory() {// this.historyList.push(this.kw)// 1. 将 Array 数组转化为 Set 对象const set = new Set(this.historyList)// 2. 调用 Set 对象的 delete 方法,移除对应的元素set.delete(this.kw)// 3. 调用 Set 对象的 add 方法,向 Set 中添加元素set.add(this.kw)// 4. 将 Set 对象转化为 Array 数组this.historyList = Array.from(set)
}
(6)将搜索历史记录持久化存储到本地
- 修改 saveSearchHistory 方法如下:
// 保存搜索关键词为历史记录
saveSearchHistory() {const set = new Set(this.historyList)set.delete(this.kw)set.add(this.kw)this.historyList = Array.from(set)// 调用 uni.setStorageSync(key, value) 将搜索历史记录持久化存储到本地uni.setStorageSync('kw', JSON.stringify(this.historyList))
}
- 在 onLoad 生命周期函数中,加载本地存储的搜索历史记录:
onLoad() {this.historyList = JSON.parse(uni.getStorageSync('kw') || '[]')
}
(7)清空搜索历史记录
- 为清空的图标按钮绑定 click 事件:
<uni-icons type="trash" size="17" @click="cleanHistory"></uni-icons>
- 在 methods 中定义 cleanHistory 处理函数:
// 清空搜索历史记录
cleanHistory() {// 清空 data 中保存的搜索历史this.historyList = []// 清空本地存储中记录的搜索历史uni.setStorageSync('kw', '[]')
}
(8)点击搜索历史跳转到商品列表页面
- 为搜索历史的 Item 项绑定 click 事件处理函数:
<uni-tag :text="item" v-for="(item, i) in historys" :key="i" @click="gotoGoodsList(item)"></uni-tag>
- 在 methods 中定义 gotoGoodsList 处理函数:
// 点击跳转到商品列表页面
gotoGoodsList(kw) {uni.navigateTo({url: '/subpkg/goods_list/goods_list?query=' + kw})
}
5、分支的合并与提交
- 将 search 分支进行本地提交:
git add .
git commit -m "完成了搜索功能的开发"
- 将本地的 search 分支推送到码云:
git push -u origin search
- 将本地 search 分支中的代码合并到 master 分支:
git checkout master
git merge search
git push
- 删除本地的 search 分支:
git branch -d search
微信小程序--优购商城项目(4)相关推荐
- 微信小程序优购商城项目
这篇文章主要就是把我写的优购商城进行简单的说明,希望可以对大家有所帮助 1.先配置页面 在app.json中设置所需要的页面路径 2.然后再封装我们需要的组件和wx.request wx.reques ...
- 微信小程序--优购商城项目(1)
文章目录 前言 一.配置uni-app开发环境 1.开发工具--HBuilderX 2.新建uni-app项目 3.把项目运行到微信开发者工具 4.使用Git管理 二.tarBar 1.创建 tabB ...
- 微信小程序--优购商城项目(6)
文章目录 前言 七.商品详情 1.创建 goodsdetail 分支并添加编译模式 2.获取商品详情数据 3.渲染商品详情页的 UI 结构 (1)渲染轮播图区域 (2)实现轮播图预览效果 (3) 渲染 ...
- 微信小程序--优购商城项目(8)
文章目录 前言 九.购物车页面 1.商品列表区域 (1)渲染购物车商品列表的标题区域 (2)渲染商品列表区域的基本结构 (3) 为 my-goods 组件封装 radio 勾选状态 (4)为 my-g ...
- 小程序优购商城项目讲解
1.新建⼩程序项⽬ 填入自己的appid 搭建⽬录结构 styles 存放公共样式 components 存放组件 lib 存放第三⽅库 utils ⾃⼰的帮助库 request ⾃⼰的接⼝帮助库 2 ...
- 微信小程序优购商城项目分析
商城项目分析 首页页面布局,最上面是搜索框,搜索框下边是个轮播图,轮播图下面这个分类这一栏,再往下就是介绍,轮播图.分类.和下边的图片都是通过接口实现的.点击分类.秒杀拍.超市购.母婴品可以进入分类页 ...
- 【愚公系列】2022年11月 微信小程序-优购电商项目-授权页面
文章目录 前言 1. 授权页面 一.授权页面 1.业务逻辑 2.涉及的接口数据 二.授权页面相关代码 1.页面代码 2.效果 前言 前言:由于微信官方修改了 getUserInfo 接口,所以现在无法 ...
- 微信小程序--优购页面制作
优购–项目 这个项目学习源自:黑马程序员微信小程序开发前端教程_零基础玩转微信小程序-哔哩哔哩 想要更深入的了解此项目,就去黑马程序员学习该项目.该项目个人觉得对于初学者来说是有很大的帮助的.做好该项 ...
- 【愚公系列】2022年11月 微信小程序-优购电商项目-商品详情页面
文章目录 前言 一.商品详情⻚⾯ 1.业务逻辑 2.涉及的接口数据 3. 关键技术 二.商品详情⻚⾯相关代码 1.页面代码 2.效果 前言 商品详情页是展示商品详细信息的一个页面,承载在网站的大部分流 ...
最新文章
- Load average in Linux的精确含义
- 百度地图开发的时候遇到的问题(二)
- 专有网络、云产品、路由器和交换机
- mysql 主从复制原理【转】
- 从云效1.0到2.0的升级,看技术如何驱动企业提效
- excel线性拟合的斜率_如何利用EXCEL求直线斜率?
- 苹果手机长截屏_iPhone终于自带长截屏了?苹果手机这些截图方式,你用过几种?...
- Linux面试题附答案
- ADNI静息态功能核磁共振成像数据预处理总流程
- centos7磁盘分区格式化
- uniapp app端自定义启动页与广告页
- 迅雷领航 WPF/E?---电影预览功能
- 如何免费破解安装正版Adobe Photoshop CC2017
- safari浏览器找不到服务器怎么办,IOS系统中Safari图标不见了怎么办 safari浏览器不见了找回方法图解...
- python假设检验--两个总体参数的检验(方差)
- Cisco Packet Trancer中小型校园网/企业网/园区网网络设计规划/无线网络
- linux是实时系统还是分时操作系统
- WEB开发技能树-JavaScript-DOM
- 前端husky中commitlint配置
- 贴片红外LED灯珠封装失效怎么应对
热门文章
- Erlang 下载安装
- 电脑故障排除方法(风扇转一下,马上就停,主板没有其它任何响应)
- 全球与中国矿物加工工程市场深度研究分析报告
- 云函数.批量操作数据库
- PLC通过智能网关 MQTT对接阿里云(带CA证书加密),实现物模型数据显示
- 鸿蒙系统和安卓有哪些区别?鸿蒙会取代安卓吗?
- 中国无尘室饮水机市场趋势报告、技术动态创新及市场预测
- android 高德地图定位圈,android ------ 实现高德定位并获取相应信息 ( 最新版高德SDK 和 Android SDK版本)...
- 计算机怎样格式u盘,win10系统电脑怎么操作把u盘格式化为fat32格式?
- Word文档中多个编号放同一行的方法(非技术)