Jessibuca 插件播放直播流视频

jessibuca官网:http://jessibuca.monibuca.com/player.html
git地址:https://gitee.com/huangz2350_admin/jessibuca#https://gitee.com/link?target=http%3A%2F%2Fjessibuca.monibuca.com%2F
项目需要的文件
在这里插入图片描述
在这里插入图片描述

1.播放组件

<template >
    <div id="jessibuca" style="width: auto; height: auto; position: relative" @mouseover="showbtn">
        <div :id="'container' + index" ref="container" :style="'position: relative;width: 100%; height:' +
            height +
            '; background:url(' +  background +
            ') no-repeat;background-size: 100% 100%;background-color: #000;'
            " @dblclick="fullscreenSwich">
            <div :id="'buttonsBox' + index" class="buttons-box" style="position: absolute; left: 0; bottom: 0">
                <div class="buttons-box-left">
                    <i v-if="!playing" class="iconfont icon-play jessibuca-btn" @click="playBtnClick" />
                    <i v-if="playing" class="iconfont icon-pause jessibuca-btn" @click="pause" />
                    <i class="iconfont icon-stop jessibuca-btn" @click="destroy" />
                    <i v-if="isNotMute" class="iconfont icon-audio-high jessibuca-btn" @click="jessibuca.mute()" />
                    <i v-if="!isNotMute" class="iconfont icon-audio-mute jessibuca-btn" @click="jessibuca.cancelMute()" />
                </div>
                <div class="buttons-box-right">
                    <span class="jessibuca-btn">{{ kBps }} kb/s</span>
                    <!--          <i class="iconfont icon-file-record1 jessibuca-btn"></i>-->
                    <!--          <i class="iconfont icon-xiangqing2 jessibuca-btn" ></i>-->
                    <i class="iconfont icon-camera1196054easyiconnet jessibuca-btn"
                        @click="jessibuca.screenshot('截图', 'png', 0.5)" />
                    <i class="iconfont icon-shuaxin11 jessibuca-btn" @click="playBtnClick" />
                    <i v-if="!fullscreen" class="iconfont icon-weibiaoti10 jessibuca-btn" @click="fullscreenSwich" />
                    <i v-if="fullscreen" class="iconfont icon-weibiaoti11 jessibuca-btn" @click="fullscreenSwich" />
                </div>
            </div>
        </div>
        <div v-if="!playing && !autoplay" class="btn-image" :style="'height:' + height">
            <div class="btn-image-center" @click="playBtnClick" :style="'width:' +
                btnimageW +
                ';height: ' +
                btnimageH +
                ';background:url(' +
                btnimage +
                ') no-repeat;background-size: 100% 100%;'
                "></div>
        </div>
    </div>
</template>
  
<script>
/* eslint-disable no-underscore-dangle */
export default {
    name: 'Jessibuca',
    props: {
        index: {
            //index
            type: Number,
            default: 0,
        },
        videoUrl: {
            //播放地址
            type: String,
            default: '',
        },
        error: {
            //报错信息
            type: Function,
            default: null,
        },
        hasAudio: {
            //静音
            type: Boolean,
            default: false,
        },
        height: {
            //播放器高度
            type: String,
            default: '500px',
        },
        isFullResize: {
            //播放面即是否填充满容器
            type: Boolean,
            default: true,
        },
        autoplay: {
            //是否自动播放
            type: Boolean,
            default: false,
        },
        background: {
            //封面图
            type: String,
            default: '',
        },
        btnimage: {
            //播放器按钮图片
            type: String,
            default: '',
        },
        btnimageH: {
            //播放器按钮图片高度
            type: String,
            default: '60px',
        },
        btnimageW: {
            //播放器按钮图片宽度
            type: String,
            default: '60px',
        },
    },
    data() {
        return {
            jessibuca: null,
            playing: false,
            isNotMute: false,
            quieting: false,
            fullscreen: false,
            loaded: false, // mute
            speed: 0,
            performance: '', // 工作情况
            kBps: 0,
            btnDom: null,
            videoInfo: null,
            volume: 1,
            rotate: 0,
            vod: true, // 点播
            forceNoOffscreen: false,
        }
    },
    watch: {
        videoUrl: {
            handler(newData, oldData) {
                if (!this.autoplay && !oldData) return
                this.play(newData)
            },
        },
    },
    mounted() {
        window.onerror = (msg) => {
            // console.error(msg)
        }
        const paramUrl = decodeURIComponent(this.$route.params.url)
        this.$nextTick(() => {
            if (typeof this.videoUrl === 'undefined') {
                this.videoUrl = paramUrl
            }
            this.btnDom = document.getElementById('buttonsBox' + this.index)
            console.log(`初始化时的地址为: ${this.videoUrl}`)
            this.autoplay ? this.play(this.videoUrl) : ''
        })
    },
    destroyed() {
        if (this.jessibuca) {
            this.jessibuca.destroy()
        }
        this.playing = false
        // this.loaded = false
        this.performance = ''
    },
    methods: {
        create() {
            const options = {}
            console.log(this.$refs.container)
            console.log(`hasAudio  ${this.hasAudio}`)

            this.jessibuca = new window.Jessibuca(
                Object.assign(
                    {
                        container: this.$refs.container,
                        videoBuffer: 0.5, // 最大缓冲时长,单位秒
                        isResize: true,
                        isFlv: true,
                        decoder: '/js/jessibuca/index.js',
                        // text: "WVP-PRO",
                        // background: "bg.jpg",
                        loadingText: '加载中',
                        hasAudio:
                            typeof this.hasAudio === 'undefined' ? true : this.hasAudio,
                        debug: false,
                        supportDblclickFullscreen: false, // 是否支持屏幕的双击事件,触发全屏,取消全屏事件。
                        operateBtns: {
                            fullscreen: false,
                            screenshot: false,
                            play: false,
                            audio: false,
                        },
                        record: 'record',
                        isFullResize: this.isFullResize || false,
                        vod: this.vod,
                        forceNoOffscreen: this.forceNoOffscreen,
                        isNotMute: this.isNotMute,
                    },
                    options
                )
            )

            // eslint-disable-next-line no-underscore-dangle
            const _this = this
            this.jessibuca.on('load', () => {
                console.log('on load init')
            })

            this.jessibuca.on('log', (msg) => {
                console.log('on log', msg)
            })
            this.jessibuca.on('record', (msg) => {
                console.log('on record:', msg)
            })
            this.jessibuca.on('pause', () => {
                _this.playing = false
            })
            this.jessibuca.on('play', () => {
                _this.playing = true
            })
            this.jessibuca.on('fullscreen', (msg) => {
                console.log('on fullscreen', msg)
                _this.fullscreen = msg
            })

            this.jessibuca.on('mute', (msg) => {
                console.log('on mute', msg)
                _this.isNotMute = !msg
            })
            this.jessibuca.on('audioInfo', (msg) => {
                // console.log("audioInfo", msg);
            })

            this.jessibuca.on('videoInfo', function (msg) {
                this.videoInfo = msg
                // console.log("videoInfo", msg);
            })

            this.jessibuca.on('bps', (bps) => {
                // console.log('bps', bps);
            })
            // eslint-disable-next-line no-unused-vars
            let _ts = 0
            this.jessibuca.on('timeUpdate', (ts) => {
                // console.log('timeUpdate,old,new,timestamp', _ts, ts, ts - _ts);
                _ts = ts
            })

            this.jessibuca.on('videoInfo', (info) => {
                console.log('videoInfo', info)
            })

            this.jessibuca.on('error', (error) => {
                console.log('error', error)
            })

            this.jessibuca.on('timeout', () => {
                console.log('timeout')
            })

            this.jessibuca.on('start', () => {
                console.log('start')
            })

            this.jessibuca.on('performance', (performance) => {
                let show = '卡顿'
                if (performance === 2) {
                    show = '非常流畅'
                } else if (performance === 1) {
                    show = '流畅'
                }
                _this.performance = show
            })
            this.jessibuca.on('buffer', (buffer) => {
                // console.log('buffer', buffer);
            })

            this.jessibuca.on('stats', (stats) => {
                // console.log('stats', stats);
            })

            this.jessibuca.on('kBps', (kBps) => {
                _this.kBps = Math.round(kBps)
            })

            // 显示时间戳 PTS
            this.jessibuca.on('videoFrame', () => { })

            //
            this.jessibuca.on('metadata', () => { })
        },
        playBtnClick(event) {
            this.play(this.videoUrl)
        },
        play(url) {
            if (!url || url.length == 0) {
                this.$message.warning('当前摄像头出错,请稍后重试')
                this.pause()
                return
            }
            if (this.jessibuca) {
                this.destroy()
            }
            this.create()
            this.jessibuca.on('play', () => {
                this.playing = true
                // this.loaded = true
                // this.quieting = this.jessibuca.quieting
            })
            if (this.jessibuca.hasLoaded()) {
                this.jessibuca.play(url)
            } else {
                this.jessibuca.on('load', () => {
                    console.log('load 播放')
                    this.jessibuca.play(url)
                })
            }
        },
        pause() {
            if (this.jessibuca) {
                this.jessibuca.pause()
            }
            this.playing = false
            this.err = ''
            this.performance = ''
        },
        destroy() {
            if (this.jessibuca) {
                this.jessibuca.destroy()
            }
            if (document.getElementById('buttonsBox' + this.index) == null) {
                document
                    .getElementById('container' + this.index)
                    .appendChild(this.btnDom)
            }
            this.jessibuca = null
            this.playing = false
            this.err = ''
            this.performance = ''
        },
        showbtn() {
            if (document.getElementById('buttonsBox' + this.index) == null) {
                document
                    .getElementById('container' + this.index)
                    .appendChild(this.btnDom)
            }
        },
        eventcallbacK(type, message) {
            // console.log("player 事件回调")
            // console.log(type)
            // console.log(message)
        },
        fullscreenSwich() {
            const isFull = this.isFullscreen()
            this.jessibuca.setFullscreen(!isFull)
            this.fullscreen = !isFull
        },
        isFullscreen() {
            return (
                document.fullscreenElement ||
                document.msFullscreenElement ||
                document.mozFullScreenElement ||
                document.webkitFullscreenElement ||
                false
            )
        },
    },
}
</script>
  
<style>
@font-face {
    font-family: 'iconfont';
    /* Project id 1291092 */
    src: url('iconfont.woff2') format('woff2');
}

.iconfont {
    font-family: 'iconfont' !important;
    font-size: 16px;
    font-style: normal;
    -webkit-font-smoothing: antialiased;
    -moz-osx-font-smoothing: grayscale;
}

.icon-play:before {
    content: '\e603';
}

.icon-pause:before {
    content: '\e6c6';
}

.icon-stop:before {
    content: '\e6a8';
}

.icon-audio-high:before {
    content: '\e793';
}

.icon-audio-mute:before {
    content: '\e792';
}

.icon-shuaxin11:before {
    content: '\e720';
}

.icon-weibiaoti10:before {
    content: '\e78f';
}

.icon-weibiaoti11:before {
    content: '\e790';
}

.icon-camera1196054easyiconnet:before {
    content: '\e791';
}

.buttons-box {
    width: 100%;
    height: 0px;
    transition: 0.4s;
    background-color: rgba(43, 51, 63, 0.7);
    display: -webkit-box;
    display: -ms-flexbox;
    display: flex;
    user-select: none;
    z-index: 100;
    display: flex;
    align-items: center;
    overflow: hidden;
}

#jessibuca:hover .buttons-box {
    height: 40px;
}

.buttons-box-left {
    display: flex;
    flex: 1;
    align-items: center;
}

.jessibuca-btn {
    color: rgb(255, 255, 255);
    line-height: 27px;
    margin: 0px 5px;
    padding: 0px 2px;
    cursor: pointer;
    text-align: center;
    font-size: 20px !important;
}

.buttons-box-right {
    height: 100%;
    position: absolute;
    right: 0;
    display: flex;
    flex: 1;
    align-items: center;
    font-size: 20px;
}

.btn-image {
    width: 100%;
    /* height: 100%; */
    position: absolute;
    left: 0;
    top: 0;
    z-index: 99 !important;
}

.btn-image-center {
    position: absolute;
    left: 0;
    right: 0;
    bottom: 0;
    top: 0;
    margin: auto;
}
</style>
  
  
.jessibuca-btn 设置播放控件的间距
decoder: '/js/jessibuca/index.js',地址是服务静态文件的相对地址
 src: url('iconfont.woff2') format('woff2'); 相对地址 iconfont.woff2需要和组件放在同一目录下

2.父组件引用

<template>
  <div class="app-container">
    <player video-url="视频地址" :has-audio="false" height="200px" :isFullResize="false" :autoplay="true"
      :index="getIndex" />
  </div>
</template>
<script>
import player from '@/components/jessibuca/index.vue' // h265播放器

export default {
  name: 'video',
  components: {
    player,
  },
  data() {
    return {
    }
  },
  computed: {
    // 获取时间戳
    getIndex() {
      let time = new Date().getTime()
      return time
    },
  },
}
</script>


3.index.html 需要引入静态js文件

<script src="./public/js/jessibuca/index.js"></script>
  • 9
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
### 回答1: Jessibuca M3U8是一种视频文件格式,它使用的是HLS(HTTP Live Streaming)协议。M3U8文件是一种播放列表文件,其中包含了视频文件的地址,而HLS则是一种基于HTTP协议的视频传输协议。这种协议可以将整个视频文件分割成多个小的TS(Transport Stream)文件,然后逐个下载和播放,这就可以保证在网络带宽较低的情况下也能够播放视频Jessibuca M3U8常常被用于在线视频直播、点播等场景,例如在视频网站、移动端应用等平台。通过对M3U8文件的解析,就可以获取到视频文件的真实地址,然后下载和播放相应的TS文件来实现视频的传输和播放。对于视频制作者和平台运营者来说,使用HLS和M3U8格式可以提供更好的使用体验和更高的观看质量。 ### 回答2: Jessibuca m3u8是一种视频文件格式,通常用于媒体播放。这种格式可以在各种设备和平台上播放,包括电视、电脑、手机和平板电脑等。相比其他视频格式,Jessibuca m3u8具有高清晰度、畅性和稳定性等优势。 Jessibuca m3u8本质上是一个播放列表文件,其中包含各种视频片段的URL地址。当用户打开视频文件时,播放器会自动下载这些片段并播放,从而实现无缝的视频播放。 对于视频制作人员和平台运营商来说,使用Jessibuca m3u8可以大幅度提升用户观看体验。通过将视频文件转换为这种格式,可以提高视频的质量和畅度,同时也可以减少用户的等待时间和缓冲时间。 总之,Jessibuca m3u8是一种非常有用的视频文件格式,它为用户提供了更好的观看体验,同时也为视频制作人员和平台运营商带来了许多好处。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

长影缚苍龙

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

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

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

打赏作者

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

抵扣说明:

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

余额充值