uniapp 小程序实现类似抖音的简易刷视频功能

uniapp 小程序实现类似抖音的简易刷视频功能

  1. 先上视频

20240725-163843

  1. 直接上代码

    1. 代码中使用的是 uniapp 组件 swiper slider
      slider
      swiper
    2. 代码中使用的  <tab-bar></tab-bar> 组件 是我自定义的组件  可以删除
    3. 代码中的图片地址以及视频地址请更换为自己的路径
    
<template><view><view class="uni-padding-wrap"><view class="page-section swiper"><view class="page-section-spacing"><swiper class="swiper" @change="changefun" @animationfinish="animationfinishfun" :current="index_":vertical="true"><swiper-item v-for="(item,index) in videoList"><view class="swiper-item" @click.stop="playPause(index)"><image src="https://xxxxxxxxxxxxxx.com/2021083111362225433.png" v-if="!isPlay" class="btn play"></image><image src="https://xxxxxxxxxxxxxx.com/2021083111373735978.png" v-if="isPlay&&firstclick"class="btn pause"></image><video :custom-cache="false" loop="true" class="video" :id="'id'+index" :duration="item.duration":enable-progress-gesture="true" :controls="false" @timeupdate="(e)=>videoUpdate(e,item,index)":src="item.vlogUrl" :show-center-play-btn="false"></video></view></swiper-item></swiper></view></view><tab-bar></tab-bar></view><view v-if="is_active"><view class="left"><view class="left_box"><!--   #ifdef MP-ALIPAY   --><view><!--   #endif  --><view class="slider"><slider v-model="activeObj.currentTime" @change="sliderChange" @changing="sliderChanging":activeColor="!isPlay || !updateState ?'#ffffff':'#ffffff4D'":block-size="!isPlay || !updateState?14:12" :block-color="!isPlay|| !updateState?'#ffffff':'#6e6f71'"backgroundColor="#4f5253"></slider></view></view></view></view></view>
</template><script>import tabBar from "../../../components/tabBar/tabBar";export default {components: {tabBar},data() {return {videoList: [],index_: null,index: '0',firstclick: false,is_active: true,isPlay: true,activeObj: null,sending: false,contentId: '',updateState: true,videoContext: null,}},onLoad(option) {// 此接口用来查第一个视频详细信息// this.$uniApi.dataRequestNoLoading('GET', 'base/app/v1/contentApp/getDefaultVlogId').then(res => {// 我在此处替换为假数据let res = {"code": 1,"data": {"id": "1806517903331479553","title": "一日游","subtitle": null,"coverImage": "https://xxxxxxxxxxxxxx.com/171954161525350063.jpg", // 请更换为自己的视频"readingSum": "239","rankingInclude": null,"includeContain": null,"vlogUrl": "https://xxxxxxxxxxxxxx.com/2024062810390955101.mp4", // 请更换为自己的图片"type": 9,"salesNum": null,"marketPrice": null,"coverImageHeight": 702,"coverImageWeight": 1080,"headImgUrl": "https://xxxxxxxxxxxxxx.com/171954159983539103.png", // 请更换为自己的图片"author": "一日游","isTopping": "0"},"msg": "成功","success": true}let id = res.data.id || ""this.contentId = idlet obj = Object.assign({}, {id: id})obj['istrue'] = falseobj['contentId'] = idobj['currentTime'] = 0//  用来计算滚动条obj['duration'] = 0// 视频链接obj['vlogUrl'] = res.data.vlogUrl// this.videoList.push(obj)// 查找视频列表   我的接口是根据当前的视频 id 去查后面的视频// 一次查的太多会导致卡顿this.oprateVideoList(this.contentId)// 初始化第一个视频 并播放this.$nextTick(function() {let videoContext = uni.createVideoContext(`id0`)videoContext.play()})// })},watch: {index_(val) {this.videoContext = uni.createVideoContext(`id${this.index_}`)this.contentId = this.videoList[val]['contentId']},activeObj(val, oldval) {if (!oldval) {this.$nextTick(function() {let videoContext = uni.createVideoContext(`id${this.index_}`)videoContext.play()})}}},methods: {// current 改变时会触发 change 事件,event.detail = {current: current, source: source}// 暂停当前的视频播放changefun(e) {this.is_active = falselet videoContext = uni.createVideoContext('id' + this.index_)videoContext.pause()},playPause(current) {// 点击视频时  如果播放就暂停 如果是暂停就开始播放let videoContext = uni.createVideoContext('id' + current)this.firstclick = trueif (this.isPlay) {videoContext.pause()this.isPlay = false} else {videoContext.play()this.isPlay = true}},// 	动画结束时会触发 animationfinish 事件,event.detail = {current: current, source: source}animationfinishfun(e) {let that = thisthis.isPlay = true// 获取下标let current = e.detail.currentthis.activeObj = this.videoList[current]this.index_ = currentthis.is_active = truethis.videoList[current]['istrue'] = true// 获取当前的 video 标签 并暂停播放let videoContext = uni.createVideoContext('id' + this.index_)videoContext.pause()// 初始化新的video 并播放videoContext = uni.createVideoContext('id' + current)videoContext.play()//  视频滑到最后一个 就再去请求新的 视频列表if (this.videoList.length > 0 && current + 1 == this.videoList.length && !this.sending) {this.oprateVideoList(this.videoList[current]['contentId'])}},videoUpdate(e, item, index) {// 此处用来计算进度条if (this.updateState) {// #ifdef MP-WEIXINif (e.target.id == 'id' + index) {this.videoList[this.index_]['currentTime'] = e.detail.currentTime / e.detail.duration * 100this.videoList[this.index_]['duration'] = e.detail.duration}// #endif// #ifdef MP-ALIPAYthis.videoList[this.index_]['currentTime'] = e.detail.currentTime / e.detail.videoDuration * 100this.videoList[this.index_]['duration'] = e.detail.videoDuration// #endif}},// 拖动过程中触发的事件,event.detail = {value: value}sliderChanging(e) {this.updateState = false},//拖动进度条触发事件sliderChange(e) {let duration = this.videoList[this.index_]['duration']if (duration) {this.videoContext.seek(e.detail.value / 100 * duration); //完成拖动后,计算对应时间并跳转到指定位置this.videoList[this.index_]['duration'] = e.detail.valuethis.updateState = true}},// 获取视频列表oprateVideoList(id) {let that = thisthis.sending = true// this.$uniApi.dataRequestNoLoading('GET', 'base/app/v1/contentVlog/getVlogListByContentId', {// 	contentId: id,// }).then(resp => {let resp = {"code": 1,"data": [{"contentId": "1606207777364082690","title": "vlog啊","coverImage": "https://xxxxxxxxxxxxxx.com/167178470537619117.png","vlogUrl": "https://xxxxxxxxxxxxxx.com/2022122316383441583.mp4"}, {"contentId": "1684011338528985090","title": "vlog","coverImage": "https://xxxxxxxxxxxxxx.com/169033454163171300.png","vlogUrl": "https://xxxxxxxxxxxxxx.com/2023072609222991294.mp4"}, {"contentId": "1678315077972848642","title": "阿达啊","coverImage": "https://xxxxxxxxxxxxxx.com/168897636035107597.png","vlogUrl": "https://xxxxxxxxxxxxxx.com/2023071016074361328.mp4"}],"msg": "成功","success": true}resp.data.map(res => {res['istrue'] = falseres['currentTime'] = 0res['duration'] = 0// return res})if (that.index_ === null) {that.index_ = 0}that.videoList = that.videoList.concat(resp.data)that.activeObj = that.videoList[that.index_]that.sending = falsethat.videoList[0]['istrue'] = true// })}},}
</script>
<style>page {background: #000;}
</style>
<style scoped lang="scss">.circle {background: rgba(34, 34, 34, 0.4);border-radius: 50%;z-index: 2;height: 70px;width: 70px;position: fixed;top: 0;bottom: 441rpx;left: 31rpx;margin: auto;.red {position: absolute;top: 0;right: 0;bottom: 0;left: 0;margin: auto;z-index: 3;height: 35px;width: 35px;}}.swiper {height: 100vh;.slider {height: 20rpx;position: absolute;bottom: 67rpx;left: 0;width: 750rpx;slider {margin: 0;}}.swiper-item {height: 100vh;position: relative;.btn {position: absolute;width: 120rpx;height: 120rpx;background: none;z-index: 2;top: 50%;margin-top: -60rpx;left: 50%;margin-left: -60rpx;&.pause {animation: benone .2s 2s linear forwards;}}}}.video {width: 100%;height: 100vh;position: relative;}.left_box {position: fixed;bottom: 55px;left: 0rpx;width: 100%;height: 600rpx;pointer-events: none;z-index: 8;background: linear-gradient(to bottom, rgba(0, 0, 0, 0), rgba(0, 0, 0, 1));.slider {pointer-events: initial;height: 20rpx;position: absolute;bottom: 67rpx;left: 0;width: 750rpx;slider {margin: 0;}}::v-deep .u-icon {position: fixed;right: 40rpx;bottom: 280rpx;pointer-events: initial;z-index: 10;}}@keyframes benone {from {opacity: 1;}to {opacity: 0;}}.uni-padding-wrap {background-color: #ffffff;}
</style>
  1. 已完成!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/50247.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

sed利用脚本处理文件

一、sed是什么 sed 命令是利用脚本来处理文本文件。它可以依照脚本的指令来处理、编辑文本文件。主要用来自动编 辑一个或多个文件、简化对文件的反复操作、编写转换程序等。 二、sed的原理 读入新的一行内容到缓存空间&#xff1b; 从指定的操作指令中取出第一条指令&…

【时时三省】(C语言基础)分支语句2

山不在高&#xff0c;有仙则名。水不在深&#xff0c;有龙则灵。 ——csdn时时三省 多分支语句 if&#xff08;表达式1&#xff09; 语句1; else if&#xff08;表达式2&#xff09; 语句2; else 语句3; 如果表达式1成立语句1会执行 如果不成立表达式2执行 如果表达式2成…

【Spring Framework】使用完全注解方式开发

Spring Framework 是一个非常灵活且强大的 Java 企业级开发框架&#xff0c;它允许开发人员以多种方式进行配置和开发。在现代 Java 开发中&#xff0c;使用完全注解的方式来进行配置和开发已经成为趋势&#xff0c;这种方式能够减少 XML 配置文件的使用&#xff0c;使代码更加…

高级及架构师高频面试题

一、微服务多节点批量应该怎么设计&#xff1f; 1、异步任务分类&#xff1a; 周期性定时任务调度任务批量任务 2、需要考虑并解决的问题&#xff1a; 2.1、避免同一任务同时被多个节点捞取。 1&#xff09;数据库的行级锁 2&#xff09;redis分布式锁 3&#xff09;quartz…

【运维笔记】数据库无法启动,数据库炸后备份恢复数据

事情起因 在做docker作业的时候&#xff0c;把卷映射到了宿主机原来的mysql数据库目录上&#xff0c;宿主机原来的mysql版本为8.0&#xff0c;docker容器版本为5.6&#xff0c;导致翻车。 具体操作 备份目录 将/var/lib/mysql备份到~/mysql_backup&#xff1a;cp /var/lib/…

Multiview LM-ICP 配准算法

Multiview LM-ICP 配准算法针对一些大型的物体&#xff08;比如建筑物&#xff09;或者需要精细化建模的物体&#xff08;比如某个文物&#xff09;&#xff0c;仅仅进行成对的配准难以还原物体的全貌和细节。所以&#xff0c;多个视角的配准十分关键。 多视角的配准存在以下两…

[STM32]FlyMcu同时烧写BootLoader和APP文件-HEX文件组成

目录 一、前言 二、HEX文件的格式 三、组合HEX文件 四、使用FlyMcu烧录 一、前言 如题&#xff0c;BootLoader每次烧写都是全部擦除&#xff0c;当我们烧写APP程序的时候&#xff0c;BootLoader程序将不复存在&#xff0c;很多开发者或许只有USB转TTL模块&#xff0c;没有其…

grep命令搜索部分命令

首先 然后可以输入&#xff5c;以及grep命令 比如 bjobs| grep "3075*"bjobs| grep "3075"这个结果是这样的&#xff0c;

mysql的存储过程:

mysql的存储过程&#xff1a; 存储过程的概念&#xff1a; 完成特点功能的sql语句的集合。把定义好的sql集合在一个特定的sql的函数当中 每次执行调用函数即可&#xff0c;还可以实现传参的调用 存储过程的语法&#xff1a; delimiter $$ #delimiter 开始和结束的语法&…

MYSQL 第四次作业

任务要求&#xff1a; 具体操作&#xff1a; 新建数据库&#xff1a; mysql> CREATE DATABASE mydb15_indexstu; Query OK, 1 row affected (0.01 sec) mysql> USE mydb15_indexstu; Database changed 新建表&#xff1a; mysql> CREATE TABLE student( ->…

遇到总条数count(*)返回不了数据

文章目录 前提1.准备数据1.1 建表语句1.2 插入数据 2.程序代码3.返回结果与分析4.验证 前提 获取h_user表中count(*)字段的值打印出来&#xff0c;打印出来是0&#xff0c;数据库中执行sql返回不是0。端点调试找到原因。下面先把数据库表数据及程序贴出来。 1.准备数据 1.1 …

CSS技巧专栏:一日一例 12 -纯CSS实现边框上下交错的按钮特效

CSS技巧专栏&#xff1a;一日一例 12 -纯CSS实现边框上下交错的按钮特效 大家好&#xff0c;今天我们来做一个上下边框交错闪动的按钮特效。 本例图片 案例分析 虽说这按钮给人的感觉就是上下两个边框交错变换了位置&#xff0c;但我们都知道border是没法移动的。那么这个按…

【无标KaiwuDB CTO 魏可伟:差异化创新,面向行业的多模架构题】

2024年7月16日&#xff0c;KaiwuDB CTO 魏可伟受邀于 2024 可信数据库发展大会主论坛发表演讲《多模一库 —— KaiwuDB 的现代数据库架构探索》&#xff0c;以下是演讲精华实录。 多模数据库 是顺应时代发展与融合趋势的产物 数据模型最早始于网状模型和层次模型&#xff0c;…

Spark实时(五):InputSource数据源案例演示

文章目录 InputSource数据源案例演示 一、​​​​​​​File Source 1、读取text文件 2、读取csv文件 3、读取json文件 二、Socket Source 三、Rate Source InputSource数据源案例演示 在Spark2.0版本之后&#xff0c;DataFrame和Dataset可以表示静态有边界的数据&am…

移动式气象站:便携科技的天气守望者

在科技日新月异的今天&#xff0c;我们身边的许多设备都在向着更加智能化、便携化的方向发展。而在气象观测领域&#xff0c;移动式气象站的出现&#xff0c;不仅改变了传统气象观测的固有模式&#xff0c;更以其灵活性和实时性&#xff0c;在气象监测、灾害预警等领域发挥着越…

MySQL练习05

题目 步骤 触发器 use mydb16_trigger; #使用数据库create table goods( gid char(8) primary key, name varchar(10), price decimal(8,2), num int);create table orders( oid int primary key auto_increment, gid char(10) not null, name varchar(10), price decima…

基于python的BP神经网络红酒品质分类预测模型

1 导入必要的库 import pandas as pd import numpy as np import matplotlib.pyplot as plt from sklearn.model_selection import train_test_split from sklearn.preprocessing import LabelEncoder from tensorflow.keras.models import Sequential from tenso…

NET8部署Kestrel服务HTTPS深入解读TLS协议之Certificate证书

Certificate证书 Certificate称为数字证书。数字证书是一种证明身份的电子凭证&#xff0c;它包含一个公钥和一些身份信息&#xff0c;用于验证数字签名和加密通信。数字证书在网络通信、电子签名、认证授权等场景中都有广泛应用。其特征如下&#xff1a; 由权威机构颁发&…

跟李沐学AI:池化层

目录 二维最大池化 填充、步幅和多个通道 平均池化层 池化层总结 二维最大池化 返回滑动窗口中的最大值。 图为池化窗口形状为 22 的最大池化层。着色部分是第一个输出元素&#xff0c;以及用于计算这个输出的输入元素: max(0,1,3,4)4。池化层与卷积层类似&#xff0c;不断…

Spark核心知识要点(八)Shuffle配置调优

1、Shuffle优化配置 -spark.shuffle.file.buffer 默认值&#xff1a;32k参数说明&#xff1a;该参数用于设置shuffle write task的BufferedOutputStream的buffer缓冲大小。将数据写到磁盘文件之前&#xff0c;会先写入buffer缓冲中&#xff0c;待缓冲写满之后&#xff0c;才会…