当前位置:   article > 正文

uniapp小程序开发 | 从零实现一款影视类app (横向滚动和下拉刷新的实现)

uniapp小程序开发 | 从零实现一款影视类app (横向滚动和下拉刷新的实现)

uniapp小程序开发实战系列,完整介绍从零实现一款影视类小程序。包含小程序前端和后台接口的全部完整实现。系列连载中,喜欢的可以点击收藏。

这里介绍下我的电影小程序的完整实现过程。这个系列将会详细讲解每个步骤,包括接口设计、数据结构优化、用户体验提升等方面,帮助开发者从零开始构建一个完整的电影小程序。希望这个系列对你有所帮助,别忘了点击收藏,以便随时查阅和跟进更新内容。接下来后续的文章,还将深入探讨如何实现电影详情页以及评论功能。

该篇着重介绍下实现过程中的横向滚动和下拉刷新的实现。

电影展示界面,是个明显的横向滚动。在没实现之前,一直以为可能需要借助listView或者uni-grid组件来实现,最后发现都不行。最终仅使用内置组件的scroll-view配合flex布局就完美达到效果啦。这里总结分享下,给需要的小伙伴。

效果截图: 

在uniapp小程序开发中,实现横向滚动和下拉刷新功能是一项常见的需求,尤其是在构建电影展示界面这样的场景。

首先,我们来看看横向滚动的实现。

横向滚动(Horizontal Scrolling)

使用<scroll-view>组件,设置scroll-x属性为true,这将使内容在水平方向上滚动。

通过width和height属性控制scroll-view的尺寸,确保内容能正确显示。

为了确保内容宽度大于容器宽度,我们需要通过CSS设置display: flex和flex-wrap: nowrap,这样内容会水平排列并持续滚动。

实现代码

index.vue:

  1. <template>
  2. <view class="content">
  3. <view class="title">
  4. <view class="title-item">
  5. 影院热映
  6. </view>
  7. <view class="title-more" @click="goToMore(1)">
  8. 查看更多 >
  9. </view>
  10. </view>
  11. <scroll-view
  12. :scroll-x="true"
  13. :show-scrollbar="false"
  14. class="scroll"
  15. >
  16. <view class="movie-box">
  17. <view v-for="(item, index) in hotList"
  18. :key="index"
  19. @click="goToDetail(item.id)" class="movie-item">
  20. <image :src="item.imageUrl" mode="heightFix" />
  21. <view class="movie-item-title">{{ ellipsis(item.title) }}</view>
  22. <view class="movie-rate">
  23. <uni-rate :readonly="true" :value="item.rate/2" size=12 active-color="#ffaa00" color="#DADADA">
  24. </uni-rate>
  25. <text class="movie-rate-t">{{item.rate}}</text>
  26. </view>
  27. </view>
  28. </view>
  29. </scroll-view>
  30. </view>
  31. </template>

样式代码

  1. <style scoped>
  2. .content {
  3. display: flex;
  4. flex-direction: column;
  5. align-items: center;
  6. justify-content: center;
  7. }
  8. .uni-margin-wrap {
  9. width: 690rpx;
  10. width: 100%;
  11. }
  12. .scroll{
  13. height:320rpx;
  14. width: 100%;
  15. white-space: nowrap;
  16. margin-top:15rpx;
  17. }
  18. .movie-box{
  19. display: flex;
  20. flex-direction: row;
  21. padding-left: 10rpx;
  22. padding-right: 10rpx;
  23. }
  24. .movie-item {
  25. display: flex;
  26. flex-direction: column;
  27. align-items: center;
  28. justify-content: center;
  29. width: 200rpx;
  30. height: 330rpx;
  31. margin-right: 22rpx;
  32. }
  33. .movie-item-img {
  34. border-radius: 5rpx;
  35. }
  36. .movie-item-title {
  37. color: #606266;
  38. font-size: 10rpx;
  39. margin-top: 10rpx;
  40. }
  41. .movie-rate {
  42. height: 40rpx;
  43. line-height: 40rpx;
  44. display: flex;
  45. justify-content: space-between;
  46. align-items: center;
  47. margin-bottom: 15rpx;
  48. }
  49. .movie-rate-t {
  50. margin-right: 4rpx;
  51. font-size: 8rpx;
  52. }
  53. </style>

接口实现

以下为测试横向滚动使用的接口,获取正在热映的电影。

  1. // 获取当前正在热映电影
  2. export const getNowHot = async (start,count,city) => {
  3. try {
  4. console.log('getNowHot request');
  5. const response = await uni.$http.post('/movie/in_theaters',{
  6. apikey: uni.$apiKey,city:city,start:start,count:count});
  7. console.log(response);
  8. if (response.statusCode !== 200) {
  9. uni.showToast({
  10. title: '数据请求失败! ',
  11. duration: 1500,
  12. icon: 'none',
  13. });
  14. return [];
  15. }
  16. return response.data;
  17. } catch (error) {
  18. console.error('Network request failed:', error);
  19. uni.showToast({
  20. title: '网络请求失败! ',
  21. duration: 1500,
  22. icon: 'none',
  23. });
  24. return [];
  25. }
  26. };

接口mock

以下为对今日热映电影的接口的mock(支持分页请求),在后台接口不具备的情况下方便测试。

  1. // 界面A的Mock Data
  2. import './better-mock/mock.mp.js'
  3. export function mockTest(){
  4. console.log('mockTest')
  5. // 今日热映 电影接口模拟
  6. const data2 = Mock.mock(uni.$http.baseUrl+'/movie/in_theaters','post',{
  7. // 属性 list 的值是一个数组,其中含有3个元素
  8. 'list|10': [{
  9. 'id|+1': 1,
  10. 'imageUrl|+1': ['/static/hot/1.jpg',
  11. '/static/hot/2.jpg',
  12. '/static/hot/3.jpg',
  13. '/static/hot/4.jpg',
  14. '/static/hot/5.jpg',
  15. '/static/hot/6.jpg'],
  16. 'title|4-6': '标题',
  17. 'rate|1-9':1,
  18. 'description|2-3':'这是描述'
  19. }],
  20. 'total|20-50': 1,
  21. 'currentPage|1-5': 1,
  22. 'pageSize': 5
  23. });
  24. // 输出结果
  25. console.log(JSON.stringify(data2, null, 4));//使用四个空格缩进
  26. }

下拉刷新的实现

在点击查看更多时,会进入今日热映页面,展示全部的热映影片。这时候就用到了下拉刷新的实现,因为数据是分页请求的,一次不可能全部加载完。

下拉刷新,分页加载的实现也很简单。在uniapp中有onReachBottom页面级别的生命周期事件。 

在uniapp中,onReachBottom 是一个页面级别的生命周期事件,当用户滚动到页面底部时触发。它的主要用途是用来实现下拉加载更多(Load More)的功能,即当用户滚动到页面的底部附近,系统自动触发该事件,开发者可以在这个事件的回调函数中加载新的数据并更新列表。

可在pages.json里定义具体页面底部的触发距离onReachBottomDistance

比如设为50,那么滚动页面到距离底部50px时,就会触发onReachBottom事件。

注意和scroll-view滚动到底部的事件的区别,scroll-view 组件自身的滚动到底部事件与页面级别的 onReachBottom 事件有以下几点区别:

1.触发范围:

scroll-view滚动到底部事件:此事件仅限于<scroll-view>组件内部的滚动。当scroll-view的内容滚动到其容器底部时触发。

onReachBottom:这是页面级别的事件,适用于整个页面的滚动。当用户滚动到页面底部时触发,无论滚动发生在scroll-view组件内还是页面的其他可滚动区域。

2.应用场景:

scroll-view滚动到底部事件:适用于那些需要在局部滚动区域(如列表、轮播等)实现加载更多功能的场景。

onReachBottom:适用于整个页面滚动到尽头时加载更多内容,适用于页面主体内容滚动的场景。

3.实现下拉刷新:

实现下拉刷新功能通常不直接使用上述两个事件中的任何一个。下拉刷新通常通过scroll-view组件的refresher-enabled属性和相关事件来实现。

设置scroll-view的refresher-enabled属性为true来开启下拉刷新功能。

使用@refresherpulling事件监听用户下拉动作,可以在此事件中更新下拉刷新的UI状态或显示刷新进度。

使用@refresherrefresh事件处理实际的刷新逻辑,即调用接口获取新数据。

数据加载完成后,调用this.$refs.scrollview.finishPullRefresh()来结束刷新状态。

scrollview的下拉刷新示例代码片段如下:

  1. <scroll-view
  2. scroll-y
  3. refresher-enabled
  4. @refresherpulling="onRefresherPulling"
  5. @refresherrefresh="onRefresherRefresh"
  6. ref="scrollview">
  7. <!-- 页面内容 -->
  8. </scroll-view>
  1. methods: {
  2. onRefresherPulling(e) {
  3. // 可以在这里根据e.detail.pullDistance等信息更新下拉刷新的UI
  4. },
  5. async onRefresherRefresh() {
  6. // 加载新数据
  7. await this.loadData();
  8. // 结束刷新状态
  9. this.$refs.scrollview.finishPullRefresh();
  10. },
  11. loadData() {
  12. // 实际的数据加载逻辑
  13. }
  14. }

 参见文档:页面 | uni-app官网

由于这个展示在单独的一个页面,因此简单的使用页面声明周期事件onReachBottom。

完整代码:

  1. <template>
  2. <view class="wrapper">
  3. <!-- 电影列表 -->
  4. <scroll-view :scroll-y="true"
  5. :show-scrollbar="false"
  6. class="scroll">
  7. <view class="movie-box">
  8. <view v-for="(item, index) in hotList" :key="index" class="movie-item">
  9. <image class="movie-item-img" :src="item.cover" mode="heightFix"></image>
  10. <view class="movie-item-title">{{ ellipsis(item.title) }}</view>
  11. <view class="movie-rate">
  12. <uni-rate :readonly="true" :value="item.rate/2" size=12 active-color="#ffaa00" color="#DADADA">
  13. </uni-rate>
  14. <text class="movie-rate-t">{{item.rate}}</text>
  15. </view>
  16. </view>
  17. </view>
  18. <uni-load-more :status="listStatus" :contentText="contentText" v-if="loadStatu"></uni-load-more>
  19. </scroll-view>
  20. </view>
  21. </template>
  22. <script>
  23. import { getNowHot } from '@/api/home.js';
  24. export default {
  25. data() {
  26. return {
  27. loadStatu: false, // loading是否显示
  28. listStatus: 'loading', // loading状态
  29. contentText: {
  30. contentdown: "加载更多",
  31. contentrefresh: "正在加载...",
  32. contentnomore: "我是有底线的"
  33. },
  34. pageNum: 1, // 电影列表初始页数
  35. movieInfo: [], // 电影列表
  36. hotList: [
  37. {
  38. id: 1,
  39. cover: '/static/hot/1.jpg',
  40. title: '标题提提提提提11111',
  41. description: '描述1',
  42. rate:10
  43. },
  44. {
  45. id: 2,
  46. cover: '/static/hot/2.jpg',
  47. title: '标题2',
  48. description: '描述2',
  49. rate:2
  50. },
  51. {
  52. id: 3,
  53. cover: '/static/hot/3.jpg',
  54. title: '标题3',
  55. description: '描述3',
  56. rate:8
  57. },
  58. {
  59. id: 4,
  60. cover: '/static/hot/4.jpg',
  61. title: '标题4',
  62. description: '描述4',
  63. rate:7
  64. },
  65. ,
  66. {
  67. id: 5,
  68. cover: '/static/hot/5.jpg',
  69. title: '标题5',
  70. description: '描述5',
  71. rate:5
  72. }
  73. ]
  74. };
  75. },
  76. onReachBottom: function() { // 页面触底触发
  77. console.log('触底’')
  78. this.getmorenews()
  79. },
  80. methods:{
  81. // 名称超出显示省略号
  82. ellipsis(value) {
  83. if (!value) return '';
  84. if (value.length > 7) {
  85. return value.slice(0, 6) + '...'
  86. }
  87. return value
  88. },
  89. // 触底之后触发函数,
  90. getmorenews() {
  91. var that = this
  92. that.loadStatu = true
  93. if (that.movieInfo.length > 89) {
  94. that.listStatus = 'noMore'
  95. return
  96. }
  97. that.listStatus = 'loading'
  98. console.log('page:'+this.pageNum);
  99. getNowHot(this.pageNum,10,"郑州").then(result => {
  100. //this.swiperList = item;
  101. //that.loadStatu = false
  102. this.listStatus = "more";
  103. this.pageNum++;
  104. console.log("getNowHot,result:");
  105. console.log(result);
  106. that.hotList = that.hotList.concat(res.data.subjects);
  107. that.listStatus = 'loading'
  108. //this.hotList = result;
  109. });
  110. }
  111. }
  112. }
  113. </script>
  114. <style scoped lang="scss">
  115. .wrapper {
  116. position: absolute;
  117. top: 0;
  118. bottom: 0;
  119. width: 100%;
  120. background-color: #F4F4F4;
  121. }
  122. .scroll{
  123. height:100%;
  124. width: 100%;
  125. white-space: nowrap;
  126. margin-top:10rpx;
  127. }
  128. .movie-box{
  129. display: flex;
  130. flex-direction: row;
  131. flex-wrap: wrap;
  132. padding-left: 10rpx;
  133. padding-right: 10rpx;
  134. }
  135. .movie-item {
  136. display: flex;
  137. flex-direction: column;
  138. align-items: center;
  139. justify-content: center;
  140. width: 220rpx;
  141. height: 350rpx;
  142. margin-right: 22rpx;
  143. }
  144. .movie-item-img {
  145. border-radius: 5rpx;
  146. }
  147. .movie-item-title {
  148. color: #606266;
  149. font-size: 8rpx;
  150. margin-top: 10rpx;
  151. }
  152. .movie-rate {
  153. height: 40rpx;
  154. line-height: 40rpx;
  155. display: flex;
  156. flex-direction: row;
  157. justify-content: space-between;
  158. align-items: center;
  159. margin-bottom: 15rpx;
  160. }
  161. .movie-rate-t {
  162. margin-right: 4rpx;
  163. font-size: 8rpx;
  164. }
  165. </style>

最后,附上测试的工程代码源码。

资源下载地址:https://download.csdn.net/download/qq8864/89377440

其他资源

豆瓣 Api V2(测试版) | doubanapi

豆瓣网接口-CSDN博客

豆瓣电影API接口_电影数据api接口购买-CSDN博客

https://github.com/liulongbin1314/request-miniprogram

小程序项目(uniapp)_uniapp小程序项目-CSDN博客

使用微信小程序开发制作一个简单的电影资讯应用-CSDN博客

uni-app 使用escook/request-miniprogram插件发请求说明_request-miniprogram post-CSDN博客

小程序调用豆瓣公开接口解决办法(转) - 简书

https://feizhaojun.com/?p=3813

https://www.cnblogs.com/winchance/p/13230063.html

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/我家小花儿/article/detail/680330
推荐阅读
相关标签
  

闽ICP备14008679号