uni-app直播实例|仿抖音小视频|uniapp仿陌陌直播

优直播uni-liveShow是基于vue+uni-app+vuex+nvue+swiper等技术开发仿制抖音|火山小视频/陌陌直播实战项目,支持编译到三端(H5、小程序、App端) 且兼容效果一致。

使用技术

  • 编码器+技术:HBuilderX + vue/NVue/uniapp/vuex
  • iconfont图标:阿里字体图标库
  • 自定义导航栏 + 底部Tabbar
  • 弹窗组件:uniPop(uni-app封装自定义弹出窗)
  • 测试环境:H5端 + 小程序 + App端

◆ uniapp透明导航栏

顶部导航栏采用的是自定义模式,可设置透明背景(如:个人主页/朋友圈动态) 具体可参看这篇文章:uni-app自定义导航栏按钮|uniapp仿微信顶部导航条

 /**
  * @desc 	uni-app主入口页面
  * @about 	Q:282310962  wx:xy190310
  */

import Vue from 'vue'
import App from './App'

// ***引入css
import './static/fonts/iconfont.css'
import './assets/css/reset.css'
import './assets/css/layout.css'

// ***引入状态管理
import store from './store'
Vue.prototype.$store = store

// ***引入公共组件
import headerBar from './components/header/header.vue'
import tabBar from './components/tabbar/tabbar.vue'
import popupWindow from './components/popupWindow.vue'
Vue.component('header-bar', headerBar)
Vue.component('tab-bar', tabBar)
Vue.component('popup-window', popupWindow)

// ***引入自定义弹窗组件uniPop
import uniPop from './components/uniPop/uniPop.vue'
Vue.component('uni-pop', uniPop)

Vue.config.productionTip = false
App.mpType = 'app'

const app = new Vue({
    ...App
})
app.$mount()

◆ uniapp实现抖音小视频界面

项目中小视频界面功能效果类似抖音/火山小视频,使用swiper组件实现上下滑动切换视频播放。

<swiper :indicator-dots="false" :duration="200" :vertical="true" :current="videoIndex" @change="handleSlider" style="height: 100%;">
    <block v-for="(item,index) in vlist" :key="index">
        <swiper-item>
            <view class="uni_vdplayer">
                <video :id="'myVideo' + index" :ref="'myVideo' + index" class="player-video" :src="item.src" 
                :controls="false" :loop="true" :show-center-play-btn="false" objectFit="fill">
                </video>
                <!-- 中间播放按钮 -->
                <view class="vd-cover flexbox" @click="handleClicked(index)"><text v-if="!isPlay" class="iconfont icon-bofang"></text></view>
                <!-- 底部信息 -->
                <view class="vd-footToolbar flexbox flex_alignb">
                    <view class="vd-info flex1">
                        <view class="item at">
                            <view class="kw" v-for="(kwItem,kwIndex) in item.keyword" :key="kwIndex"><text class="bold fs_18 mr_5">#</text> {{kwItem}}</view>
                        </view>
                        <view class="item subtext">{{item.subtitle}}</view>
                        <view class="item uinfo flexbox flex_alignc">
                            <image class="avator" :src="item.avator" mode="aspectFill" /><text class="name">{{item.author}}</text> <text class="btn-attention bg_linear1" :class="item.attention ? 'on' : ''" @tap="handleAttention(index)">{{item.attention ? '已关注' : '关注'}}</text>
                        </view>
                        <view class="item reply" @tap="handleVideoComment"><text class="iconfont icon-pinglun mr_5"></text> 写评论...</view>
                    </view>
                    <view class="vd-sidebar">
                        <view v-if="item.cart" class="ls cart flexbox bg_linear3" @tap="handleVideoCart(index)"><text class="iconfont icon-cart"></text></view>
                        <view class="ls" @tap="handleIsLike(index)"><text class="iconfont icon-like" :class="item.islike ? 'like' : ''"></text><text class="num">{{ item.likeNum+(item.islike ? 1: 0) }}</text></view>
                        <view class="ls" @tap="handleVideoComment"><text class="iconfont icon-liuyan"></text><text class="num">{{item.replyNum}}</text></view>
                        <view class="ls"><text class="iconfont icon-share"></text><text class="num">{{item.shareNum}}</text></view>
                    </view>
                </view>
            </view>
        </swiper-item>
    </block>
</swiper>

视频video不能覆盖的问题,使用nvue页面就可以解决view覆盖在video之上,更多关于nvue页面开发,可以参看:uniapp开发nvue页面

项目中的聊天页面,功能效果这里就不详细介绍了,可参看这篇:uni-app聊天室|vue+uniapp仿微信聊天实例

◆ uniapp仿陌陌直播页面

解决video不能覆盖问题,直播页面采用的是nvue编写

<template>
	<div class="nlv__container">
		<view class="nlv_main">
			<swiper class="nlv-swiper" :indicator-dots="false" :vertical="false" :current="videoIndex" @change="handleSlider">
				<swiper-item v-for="(item, index) in vlist" :key="index">
					<!-- //直播区 -->
					<view class="nlv-playerbox">
						<video :id="'myVideo' + index" :ref="'myVideo' + index" class="player-video" :src="item.src" :autoplay="index == videoIndex"
						:controls="false" :loop="true" :show-center-play-btn="false" objectFit="fill" :style="{height: winHeight, width: winWidth}">
						</video>
						
						<!-- //顶部 -->
						<view class="nlv_topbar" :style="{ height: headerBarH, 'padding-top': statusBarH }">
							...
						</view>
						
						<!-- //排名信息 -->
						<view class="nlv-rankbox" :style="{top: headerBarH}">
							<view class="nlv-rkls">
								<text class="rkitem">总排名{{item.allRank}}</text>
								<text v-if="item.hourRank" class="rkitem">小时榜第{{item.hourRank}}名</text>
							</view>
							<text class="nlv-uid">U直播:{{item.uid}}</text>
						</view>
						
						<!-- 底部信息栏 -->
						<view class="nlv-footToolbar">
							<!-- 送礼物提示 -->
							...
							
							<!-- 滚动msg信息 -->
							<scroll-view class="nlv-rollMsgPanel" scroll-y show-scrollbar="false">
								<block v-for="(msgitem, msgidx) in item.rollmsg" :key="msgidx">
									<view class="nlv-msglist"><view class="msg_bg"><text class="msg_name">{{msgitem.uname}}</text> <text class="msg_text">{{msgitem.content}}</text></view></view>
								</block>
							</scroll-view>
							
							<view class="nlv-infobox">
								<view class="nlv_reply" @tap="handleRollMsg(index)"><text class="nlv_reply_text">说点什么...</text></view>
								<view class="nlv_btntool">
									<view class="btn-toolitem"><text class="iconfont i-btntool">&#xe61f;</text></view>
									<view v-if="item.cart" class="btn-toolitem" @tap="handleLiveCart(index)"><text class="iconfont i-btntool" style="color: #ff4e0e;font-size: 20px;">&#xe61e;</text></view>
									<view class="btn-toolitem btn-toolitem-cart" @tap="handleLiveGift"><text class="iconfont i-btntool">&#xe600;</text></view>
									<view class="btn-toolitem"><text class="iconfont i-btntool">&#xe682;</text></view>
									<view class="btn-toolitem"><text class="iconfont i-btntool">&#xe656;</text></view>
								</view>
							</view>
						</view>
					</view>
				</swiper-item>
			</swiper>
		</view>
		
		<!-- 商品广告、滚动消息、礼物 -->
		...
	</div>
</template>

<script>
	// 引入视频数据
	const liveJson = require('../mock-live.js')
	
	// 引入商品广告、滚动消息json、礼物
	import liveCart from './cp-live/cart.vue'
	import rollMsg from './cp-live/rollmsg'
	import liveGift from './cp-live/gift'
	
	
	let timer = null
	export default {
		data() {
			return {
				statusBarH: '',
				headerBarH: '',
				winHeight: '',
				winWidth: '',
				
				videoIndex: 0,
				vlist: liveJson,
				
				clickNum: 0,	//记录点击次数
			}
		},
		components: {
			liveCart, rollMsg, liveGift
		},
		beforeCreate() {
			// 引入iconfont字体
			// #ifdef APP-PLUS
			const domModule = weex.requireModule('dom')
			domModule.addRule('fontFace', {
				fontFamily: "nvueIcon",
				'src': "url('../../../static/fonts/iconfont.ttf')"
			});
			// #endif
		},
		created() {
			// 获取设备宽、高、状态栏高度
			let _sH = uni.getSystemInfoSync().statusBarHeight
			let _hH = _sH + 50
			let _wH = uni.getSystemInfoSync().windowHeight
			let _wW = uni.getSystemInfoSync().windowWidth
			this.statusBarH = `${_sH}px`
			this.headerBarH = `${_hH}px`
			this.winHeight = `${_wH}px`
			this.winWidth = `${_wW}px`
		},
		onLoad(option) {
			// 获取列表页索引
			this.videoIndex = parseInt(option.index)
		},
		onReady() {
			this.init()
		},
		methods: {
			init() {
				this.videoContextList = []
				for(var i = 0; i < this.vlist.length; i++) {
					// this.videoContextList.push(uni.createVideoContext('myVideo' + i, this))
					// #ifdef APP-PLUS-NVUE
					this.videoContextList.push(this.$refs['myVideo' + i][0])
					// #endif
					// #ifndef APP-PLUS
					this.videoContextList.push(uni.createVideoContext('myVideo' + i, this))
					// #endif
				}
			},
			
			
			// >>> 【视频播放处理模块】-------------------------------------
			// 滑动切换
			handleSlider(e) {
				let curIndex = e.detail.current
				if(this.videoIndex >= 0){
					this.videoContextList[this.videoIndex].pause()
					this.videoContextList[this.videoIndex].seek(0)
				}
				if(curIndex === this.videoIndex + 1) {
					console.log('向左滑动')
					this.videoContextList[this.videoIndex + 1].play()
				}else if(curIndex === this.videoIndex - 1) {
					console.log('向右滑动')
					this.videoContextList[this.videoIndex - 1].play()
				}
				this.videoIndex = curIndex
			},
			// 播放
			play(index) {
				this.videoContextList[index].play()
			},
			// 暂停
			pause(index) {
				this.videoContextList[index].pause()
			},
			
			...
		}
	}
</script>

至此,基于uniapp+vue直播项目就介绍这么多了,希望能够喜欢!!

◆ 最后附上react、RN项目实战案例

RN仿微信聊天室:https://blog.csdn.net/yanxinyun1990/article/details/100573360

react+redux网页版聊天实例:https://blog.csdn.net/yanxinyun1990/article/details/94143575

 

要实现仿抖音视频滑动效果,你可以使用uni-app框架中的swiper组件结合相关的动画效果来实现。下面是一种可能的实现方式: 1. 在你的uni-app项目中,使用swiper组件创建一个滑动容器,设置为横向滑动。 ```html <swiper class="swiper" :autoplay="false" :indicator-dots="false" :circular="true"> <swiper-item v-for="(item, index) in videoList" :key="index"> <video :src="item.url" autoplay muted loop></video> </swiper-item> </swiper> ``` 2. 使用css样式来设置容器的布局和样式。 ```css .swiper { width: 100%; height: 100%; overflow: hidden; } .swiper-item { width: 100%; height: 100%; } video { width: 100%; height: 100%; object-fit: cover; } ``` 3. 使用JavaScript或者Vue的生命周期钩子函数来监听滑动事件,并根据滑动的距离和方向来实现动画效果。 ```javascript export default { data() { return { videoList: [...], // 视频列表数据 startX: 0, // 触摸起始点的X坐标 startY: 0, // 触摸起始点的Y坐标 moveX: 0, // 触摸移动中的X坐标 moveY: 0, // 触摸移动中的Y坐标 direction: '', // 滑动方向 currentIndex: 0 // 当前显示的视频索引 }; }, methods: { onTouchStart(e) { this.startX = e.changedTouches[0].pageX; this.startY = e.changedTouches[0].pageY; }, onTouchMove(e) { this.moveX = e.changedTouches[0].pageX; this.moveY = e.changedTouches[0].pageY; const offsetX = this.moveX - this.startX; const offsetY = this.moveY - this.startY; if (Math.abs(offsetY) < Math.abs(offsetX)) { // 水平滑动 if (offsetX > 0) { this.direction = 'right'; } else { this.direction = 'left'; } } else { // 垂直滑动 if (offsetY > 0) { this.direction = 'down'; } else { this.direction = 'up'; } } }, onTouchEnd() { if (this.direction === 'left') { // 向左滑动,切换到下一个视频 this.currentIndex++; if (this.currentIndex >= this.videoList.length) { this.currentIndex = 0; } } else if (this.direction === 'right') { // 向右滑动,切换到上一个视频 this.currentIndex--; if (this.currentIndex < 0) { this.currentIndex = this.videoList.length - 1; } } // 根据currentIndex更新swiper组件的activeIndex属性,实现视图切换 this.$refs.swiper.swiperRef.setActiveItem(this.currentIndex); } } }; ``` 上述代码中,通过监听触摸事件,根据滑动方向切换到对应的视频,并将视图更新到当前的视频。 这是一种简单的实现方式,你可以根据自己的需求进行扩展和优化。希望对你有帮助!
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

xiaoyan_2018

你的鼓励将是我持续作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值