更新记录

1.4.0(2022-07-27)

本次主要更新: 1.修改andorid设置音轨方法不生效,修改方法名 2.增加视频轨相关功能 3.andorid增加硬件加速参数

1.3.0(2022-07-20)

本次主要更新: 1.增加 插件内 全屏方法

1.2.0(2022-07-18)

本次主要更新: 1.新增自定义播放器示列 2.增加窗口缩放、视频纵横比、音轨等等功能

查看更多

平台兼容性

Android Android CPU类型 iOS
适用版本区间:4.4 - 14.0 armeabi-v7a:未测试,arm64-v8a:支持,x86:未测试 适用版本区间:9 - 17

原生插件通用使用流程:

  1. 购买插件,选择该插件绑定的项目。
  2. 在HBuilderX里找到项目,在manifest的app原生插件配置中勾选模块,如需要填写参数则参考插件作者的文档添加。
  3. 根据插件作者的提供的文档开发代码,在代码中引用插件,调用插件功能。
  4. 打包自定义基座,选择插件,得到自定义基座,然后运行时选择自定义基座,进行log输出测试。
  5. 开发完毕后正式云打包

付费原生插件目前不支持离线打包。
Android 离线打包原生插件另见文档 https://nativesupport.dcloud.net.cn/NativePlugin/offline_package/android
iOS 离线打包原生插件另见文档 https://nativesupport.dcloud.net.cn/NativePlugin/offline_package/ios

注意事项:使用HBuilderX2.7.14以下版本,如果同一插件且同一appid下购买并绑定了多个包名,提交云打包界面提示包名绑定不一致时,需要在HBuilderX项目中manifest.json->“App原生插件配置”->”云端插件“列表中删除该插件重新选择


KJ-VLC

VLC多媒体播放器、支持rtsp、rtmp、mms、ftp、udp/rtp等等大多数格式、截图、录制、速率、快进、倒退、音量、窗口缩放、视频纵横比、音轨。。

注意事项

andorid 需要配置:manifest.json -> App常用其它设置 -> 支持CPU类型 勾选 arm64-v8a
要不某些手机会闪退

海康rtsp流规则 https://blog.csdn.net/NBA_1/article/details/118086332

rtsp://用户名:密码@172.16.2.79:554/Streaming/Channels/101

使用

<template>
    <view class="content">
        <KJ-VLC class="vlc" ref="vlc" :style="{width:'100%',height:palyerHeight}" @onStateChanged="onStateChanged"
            @onTimeChanged="onTimeChanged" @onStartedRecording="onStartedRecording" @onStoppedAtPath="onStoppedAtPath">
            <cover-view class="video-view" :style="{width:'100%',height:palyerHeight}" @click="tap"
                @touchstart="touchStartHandle" @touchend="touchEndHandle" @touchcancel="touchCancelHandle"
                @touchmove="touchMoveHandle">
                <text class="tip" v-if="isShowTip"
                    :style="{position:'absolute',left:0,right:0,top:tipTop}">{{tip}}</text>
                <view class=" top-title" :style="{width:'100%'}" v-if="isShowTitle">
                    <view></view>
                    <text :style="{color:'#fff'}">标题</text>
                    <view></view>
                </view>
                <view class="bottom-title" :style="{width:'100%'}" v-if="isShowTitle">
                    <button type="primary" @click="playOrPause">{{playOrPauseText}}</button>
                    <text class="text">{{leftTime}}</text>
                    <view class="fullControls-center">
                        <chunLei-slider :max="duration" :value="current" :style="{width:`${fullControlsWidth-40}px`}"
                            :screenLeft="40" :width="fullControlsWidth-40" @change="changeCurrent" :ratio="1">
                        </chunLei-slider>
                    </view>
                    <text class="text">{{rightTime}}</text>
                    <button type="primary" @click="full">全屏</button>
                </view>
            </cover-view>
        </KJ-VLC>

        <view class="btns">
            <button type="primary" @click="init">初始化</button>
            <button type="primary" @click="play">播放</button>
            <button type="primary" @click="isPlaying">是否在播放</button>
            <button type="primary" @click="pause">暂停</button>
            <button type="primary" @click="stop">停止</button>
            <button type="primary" @click="getLength">获取视频时长</button>
            <button type="primary" @click="getRate">获取速率</button>
            <button type="primary" @click="setRate">设置速率</button>
            <button type="primary" @click="getTime">获取进度</button>
            <button type="primary" @click="setTime">设置进度</button>
            <button type="primary" @click="saveVideoSnapshotAt">保存截图</button>
            <button type="primary" @click="startRecordingAtPath">开始录屏</button>
            <button type="primary" @click="stopRecording">停止录屏</button>
            <button type="primary" @click="getVolume">获取音量</button>
            <button type="primary" @click="setVolume">设置音量</button>
            <button type="primary" @click="setScale">设置视频缩放</button>
            <button type="primary" @click="getScale">获取视频缩放</button>
            <button type="primary" @click="setAspectRatio">设置视频纵横比</button>
            <button type="primary" @click="getAspectRatio">获取视频纵横比</button>
            <button type="primary" @click="getAudioTracks">获取所有音轨</button>
            <button type="primary" @click="getAudioTracksCount">获取音轨个数</button>
            <button type="primary" @click="getAudioTrack">获取当前音轨</button>
            <button type="primary" @click="setAudioTrack">设置音轨(原唱)</button>
            <button type="primary" @click="setAudioTrack2">设置音轨(伴唱)</button>
            <button type="primary" @click="getVideoTracks">获取所有视频轨</button>
            <button type="primary" @click="getVideoTracksCount">获取视频轨个数</button>
            <button type="primary" @click="getVideoTrack">获取当前视频轨</button>
            <button type="primary" @click="setVideoTrack">设置视频轨</button>
            <button type="primary" @click="setVideoTrack2">设置视频轨2</button>
        </view>
    </view>
</template>

<script>
    import chunLeiSlider from '@/components/chunLei-slider/chunLei-slider.nvue'

    export default {
        components: {
            chunLeiSlider
        },
        data() {
            return {
                duration: 60 * 60 * 1000, //精确总时长
                fullControlsWidth: '200', //全屏宽
                current: 0, //video进度
                leftTime: "00:00:00",
                rightTime: "00:00:00",
                statusBarHeight: "20px",
                getStatusBarHeight: "20px",
                headerHeight: "44px",
                palyerHeight: "500rpx",
                tipTop: "250rpx",
                tip: "加载中...",
                isShowTip: true,
                isShowTitle: true,
                hideTitleTime: 6000,
                isFull: false,
                playOrPauseText: "播放",
                videoNowSatus: false, //当前播放的视频是否已经播放完成
                touchHandleType: false, //滑动记录 progress // light // voice
                handleDynamicX: 0, // 动态记录滑动的记录X轴
                handleDynamicY: 0, // 动态记录Y轴滑动距离
                videoNowLight: 0, // 当前屏幕的亮度
                videoNowVoice: 0, // 当前的音量
                onceClickTimer: null,
                changing: false
            }
        },
        onLoad() {
            this.getStatusBarHeight = plus.navigator.getStatusbarHeight();
            this.statusBarHeight = this.getStatusBarHeight;

            this.videoNowVoice = plus.device.getVolume(); // 获取当前屏幕的音量
            this.onceClickTimer = setTimeout(() => {
                clearTimeout(this.onceClickTimer);
                this.onceClickTimer = null;
                this.isShowTitle = false;
            }, this.hideTitleTime)
        },
        onReady() {
            this.init();
            this.play();
            //this.$refs.vlc.requestFullScreen()
        },
        (res) {
            console.log(JSON.stringify(res))
            if (res.deviceOrientation == "portrait") {
                this.isFull = false;
                plus.navigator.showSystemNavigation();
                // this.statusBarHeight = this.getStatusBarHeight;
                // this.headerHeight = "45px";
                this.palyerHeight = '500rpx';
                this.tipTop = '250rpx';
                this.fullControlsWidth = uni.getSystemInfoSync().screenWidth - 200;
                this.$refs.vlc.exitFullScreen()
                //this.init();
                //this.play();

            } else if (res.deviceOrientation == "landscape") {
                this.isFull = true;
                plus.navigator.hideSystemNavigation();
                // this.statusBarHeight = "0";
                // this.headerHeight = "0";
                this.palyerHeight = '750rpx'
                this.tipTop = '375rpx';
                this.fullControlsWidth = uni.getSystemInfoSync().screenWidth - 200;
                this.$refs.vlc.requestFullScreen()
                //this.init();
                //this.play();

            }
            //this.full();
        },
        methods: {
            onStateChanged(res) {
                console.log("onStateChanged:" + JSON.stringify(res.detail))
                /**
                 * statr 状态  Opening(打开) Buffering(推流) Error(错误) Stopped(停止) 
                 * Playing(播放) Paused(暂停) Ended(结束) ESAdded()
                 * */
                var state = res.detail.state;
                this.$refs.vlc.isPlaying((res) => {
                    var isPlaying = res.result;
                    if (isPlaying == true) {
                        this.playOrPauseText = "暂停";
                    } else {
                        this.playOrPauseText = "播放";
                    }
                })
                if (state == "Playing") {
                    this.isShowTip = false;
                } else if (state == "Ended") {
                    this.$refs.vlc.stop();
                    this.init();
                    this.$refs.vlc.play();
                } else if (res.detail.state == "Error") {
                    this.init();
                    this.tip = "加载错误";
                    this.isShowTip = true;
                }
            },
            onTimeChanged(res) {
                // console.log("onTimeChanged:" + JSON.stringify(res.detail))
                if (this.changing == false) {
                    this.duration = res.detail.length;
                    this.current = res.detail.time;
                    this.leftTime = this.formatDuring(this.current)
                    this.rightTime = this.formatDuring(this.duration)
                }

            },
            onStartedRecording(res) {
                console.log("onStartedRecording:" + JSON.stringify(res.detail))
            },
            onStoppedAtPath(res) {
                console.log("onStoppedAtPath:" + JSON.stringify(res.detail))
                var path = res.detail.path;
                path = path.replace("//", "/");
                console.log("path:" + path)
                uni.saveVideoToPhotosAlbum({
                    filePath: path
                })
            },
            init() {
                this.tip = "等待播放";
                this.isShowTip = true;
                this.$refs.vlc.init({
                    "urlType": "network", //local(本地路径-传绝对路径plus.io.convertLocalFileSystemURL) network(网络地址)
                    "url": "http://baobab.kaiyanapp.com/api/v1/playUrl?vid=164016&resourceType=video&editionType=high&source=aliyun&playUrlType=url_oss",
                    "options": ["--rtsp-tcp",
                        //"--aout=opensles", //加这个,方法setVolume会失效 
                        "--audio-time-stretch",
                        "-vvv"
                    ],
                    "mediaOptions": [":network-caching=3000", ":clock-jitter=0", ":clock-synchro=0",
                        ":fullscreen"
                    ],
                    "HWDecoder": { //andorid有效,是否开启硬件加速
                        "enabled": false,
                        "force": false
                    }
                })
                this.playOrPauseText = "播放";
                this.duration = 60 * 60 * 1000;
                this.current = 0;
                this.leftTime = this.formatDuring(this.current)
                this.rightTime = this.formatDuring(this.duration)
            },
            isPlaying() {
                this.$refs.vlc.isPlaying((res) => {
                    console.log("isPlaying:" + JSON.stringify(res))
                })
            },
            playOrPause() {
                this.$refs.vlc.isPlaying((res) => {
                    var isPlaying = res.result;
                    if (isPlaying == true) {
                        this.$refs.vlc.pause()
                    } else {
                        this.$refs.vlc.play()
                    }
                })
            },
            play() {
                this.$refs.vlc.play()
            },
            pause() {
                this.$refs.vlc.pause()
            },
            stop() {
                this.$refs.vlc.stop()
            },
            getLength() {
                this.$refs.vlc.getLength((res) => {
                    console.log("getLength:" + JSON.stringify(res))
                })
            },
            getRate() {
                this.$refs.vlc.getRate((res) => {
                    console.log("getRate:" + JSON.stringify(res))
                })
            },
            setRate() {
                this.$refs.vlc.setRate({
                    "rate": "8.0" //0.25 - 8.0
                })
            },
            getTime() {
                this.$refs.vlc.getTime((res) => {
                    console.log("getTime:" + JSON.stringify(res))
                })
            },
            setTime() {
                this.$refs.vlc.setTime({
                    "time": "" + 1000 * 60 //单位毫秒
                })
            },
            saveVideoSnapshotAt() {
                var path = plus.io.convertLocalFileSystemURL("_doc/" + new Date()
                    .getTime() +
                    ".png")
                this.$refs.vlc.saveVideoSnapshotAt({
                    "path": path
                })
                uni.saveImageToPhotosAlbum({
                    filePath: path
                })
            },
            startRecordingAtPath() {
                this.$refs.vlc.startRecordingAtPath({
                    "path": plus.io.convertLocalFileSystemURL("_doc")
                })
            },
            stopRecording() {
                this.$refs.vlc.stopRecording()
            },
            getVolume() {
                this.$refs.vlc.getVolume((res) => {
                    console.log("getVolume:" + JSON.stringify(res))
                })
            },
            setVolume() {
                this.$refs.vlc.setVolume({
                    "volume": "1" //1-100 播放器的声音
                })
            },
            setScale() {
                this.$refs.vlc.setScale({
                    "scale": "0"
                })
            },
            getScale() {
                this.$refs.vlc.getScale((res) => {
                    console.log("getScale:" + JSON.stringify(res))
                })
            },
            setAspectRatio() {
                this.$refs.vlc.setAspectRatio({
                    "aspectRatio": "1000:600"
                })
            },
            getAspectRatio() {
                this.$refs.vlc.getAspectRatio((res) => {
                    console.log("getAspectRatio:" + JSON.stringify(res))
                })
            },
            getAudioTracks() {
                this.$refs.vlc.getAudioTracks((res) => {
                    console.log("getAudioTracks:" + JSON.stringify(res))
                })
            },
            getAudioTracksCount() {
                this.$refs.vlc.getAudioTracksCount((res) => {
                    console.log("getAudioTracksCount:" + JSON.stringify(res))
                })
            },
            getAudioTrack() {
                this.$refs.vlc.getAudioTrack((res) => {
                    console.log("getAudioTrack:" + JSON.stringify(res))
                })
            },
            setAudioTrack() {
                this.$refs.vlc.setAudioTrack({
                    "id": "1"
                })
            },
            setAudioTrack2() {
                this.$refs.vlc.setAudioTrack({
                    "id": "2"
                })
            },
            getVideoTracks() {
                this.$refs.vlc.getVideoTracks((res) => {
                    console.log("getVideoTracks:" + JSON.stringify(res))
                })
            },
            getVideoTracksCount() {
                this.$refs.vlc.getVideoTracksCount((res) => {
                    console.log("getVideoTracksCount:" + JSON.stringify(res))
                })
            },
            getVideoTrack() {
                this.$refs.vlc.getVideoTrack((res) => {
                    console.log("getVideoTrack:" + JSON.stringify(res))
                })
            },
            setVideoTrack() {
                this.$refs.vlc.setVideoTrack({
                    "id": "-1"
                })
            },
            setVideoTrack2() {
                this.$refs.vlc.setVideoTrack({
                    "id": "0"
                })
            },
            formatDuring(mss) {
                //console.log(mss)
                var hours = parseInt((mss % (1000 * 60 * 60 * 24)) / (1000 * 60 * 60));
                if (hours < 10) {
                    hours = "0" + hours
                }
                var minutes = parseInt((mss % (1000 * 60 * 60)) / (1000 * 60));
                if (minutes < 10) {
                    minutes = "0" + minutes
                }
                var seconds = parseInt((mss % (1000 * 60)) / 1000);
                if (seconds < 10) {
                    seconds = "0" + seconds
                }
                return hours + ":" + minutes + ":" + seconds;
            },
            //拖动滑块
            changeCurrent(e) {
                this.current = e.detail.value
                console.log(JSON.stringify(e));
                this.changing = true;
                this.$refs.vlc.setTime({
                    "time": "" + parseInt(this.current)
                })
            },
            changing(e) {
                this.changing = false;
            },
            full() {
                if (!this.isFull) {
                    plus.screen.lockOrientation("landscape-primary");
                    //this.$refs.vlc.requestFullScreen()
                } else {
                    plus.screen.lockOrientation("portrait-primary");
                    //this.$refs.vlc.exitFullScreen()
                }
                this.isFull = !this.isFull;
            },
            tap() {
                console.log("tap")
                if (this.onceClickTimer != null) {
                    clearTimeout(this.onceClickTimer);
                    this.onceClickTimer = null;
                }
                this.isShowTitle = !this.isShowTitle;
                this.$forceUpdate()
                if (this.isShowTitle) {
                    this.onceClickTimer = setTimeout(() => {
                        clearTimeout(this.onceClickTimer);
                        this.onceClickTimer = null;
                        this.isShowTitle = false;
                    }, this.hideTitleTime)
                }
            },
            //监听当前操作
            touchStartHandle(e) {
                console.log(e);
                this.touchStartTimeStamp = new Date().getTime();
                this.handleDynamicY = e.changedTouches[0].screenY;
                this.handleDynamicX = e.changedTouches[0].screenX;
            },
            //监听抬起操作
            touchEndHandle(e) {
                //this.tap();
                if (this.touchHandleType) {
                    this.touchStartTimeStamp = 0;
                    this.handleDynamicY = 0;
                    this.handleDynamicX = 0;
                    this.touchHandleType = false;
                    if (this.progressBtnStatus) {
                        this.progressBtnStatus = false;
                        this.progressPercent = this.progressBtnPercent;
                    }
                }

            },
            // 监听是否触发滑动事件
            touchMoveHandle(e) {
                // 判断左右上下滑动是否超过10px
                let X = e.changedTouches[0].screenX;
                let Y = e.changedTouches[0].screenY;
                if (this.touchHandleType) {
                    this.handleFun(X, Y);
                    return;
                }
                let chaX = Math.abs(X - this.handleDynamicX);
                let chaY = Math.abs(Y - this.handleDynamicY);
                if (chaX > 10 || chaY > 10) {
                    if (chaX > 10) {
                        this.touchHandleType = 'progress';
                        this.handleDynamicX = X;
                    } else {
                        let left = uni.getSystemInfoSync().screenWidth / 2;
                        if (this.handleDynamicX > left) {
                            this.touchHandleType = 'voice'
                        } else {
                            this.touchHandleType = 'light'
                        }
                        this.handleDynamicY = Y;
                    }
                }
            },
            // 操作当前的方法
            handleFun(X, Y) {
                let chaX = X - this.handleDynamicX;
                let chaY = Y - this.handleDynamicY;
                switch (this.touchHandleType) {
                    case 'progress':
                        console.log("progress")
                        this.progressBtnStatus = true;
                        if (chaX > 0) {
                            if (this.current >= this.duration) {
                                this.current = this.duration;
                            } else {
                                this.current += 1000;
                            }
                        } else {
                            if (this.current <= 0) {
                                this.current = 0;
                            } else {
                                this.current -= 1000;
                            }
                        }
                        this.$refs.vlc.setTime({
                            "time": "" + parseInt(this.current)
                        })
                        break;
                    case 'light':
                        console.log("light")
                        if (chaY < 0) {
                            if (this.videoNowLight + 0.02 >= 1) {
                                this.videoNowLight = 1;
                            } else {
                                this.videoNowLight += 0.02
                            }
                        } else {
                            if (this.videoNowLight - 0.02 <= 0) {
                                this.videoNowLight = 0;
                            } else {
                                this.videoNowLight -= 0.02
                            }
                        }
                        plus.screen.setBrightness(this.videoNowLight)
                        break;
                    case 'voice':
                        console.log("voice")

                        if (chaY < 0) {
                            if (this.videoNowVoice + 0.02 >= 1) {
                                this.videoNowVoice = 1;
                            } else {
                                this.videoNowVoice += 0.02
                            }
                        } else {
                            if (this.videoNowVoice - 0.02 <= 0) {
                                this.videoNowVoice = 0;
                            } else {
                                this.videoNowVoice -= 0.02
                            }
                        }
                        plus.device.setVolume(this.videoNowVoice);

                        break;
                    default:
                        this.touchHandleType = false;
                        break;
                }
                this.handleDynamicX = X;
                this.handleDynamicY = Y;
            }
        }
    }
</script>
<style>
    .text {
        color: #fff;
        font-size: 10px;
    }

    .vlc {
        justify-content: center;
        align-items: center;
        background-color: black;
    }

    .video-view {
        position: absolute;
        top: 0;
        bottom: 0;
        left: 0;
        right: 0;
        z-index: 999;
    }

    .tip {
        position: absolute;
        left: 0;
        right: 0;
        top: 250rpx;
        text-align: center;
        color: #fff
    }

    .top-title,
    .bottom-title {
        position: absolute;
        left: 0;
        right: 0;
        height: 70px;
        background-color: rgba(0, 0, 0, 0.1);
        display: flex;
        justify-content: space-between;
        flex-direction: row;
        align-items: center;
        color: #fff;
        padding: 16px;
    }

    .top-title {
        top: 0;
        width: 750rpx;
    }

    .bottom-title {
        bottom: 0;
        width: 750rpx;
    }

    .fullControls-center {
        flex-direction: row;
        align-items: center;
        justify-content: center;
        height: 40px;
    }

    .btns {
        margin-top: 40px;
        display: flex;
        flex-wrap: wrap;
        flex-direction: row;
        align-items: center;
        justify-content: center;
    }
</style>

隐私、权限声明

1. 本插件需要申请的系统权限列表:

2. 本插件采集的数据、发送的服务器地址、以及数据用途说明:

插件不采集任何数据

3. 本插件是否包含广告,如包含需详细说明广告表达方式、展示频率:

使用中有什么不明白的地方,就向插件作者提问吧~ 我要提问