ps-tcplayer
Version:
Tencent Cloud Player component with Vue2/Vue3 compatibility
1,309 lines • 90.1 kB
JavaScript
import { h, isVue2 } from "vue-demi";
import { version } from "vue-demi";
function loadResource(sources, cb) {
let arr = [...sources];
if (!arr.length) {
cb && cb();
console.log("resouce loaded");
return false;
}
let item = arr.shift();
if (!item) {
loadResource(arr, cb);
return false;
}
const isJsFile = /\.js(\?.*)?$/.test(item);
const isCssFile = /\.css(\?.*)?$/.test(item);
if (isJsFile) {
getJS(item, () => {
loadResource(arr, cb);
});
} else if (isCssFile) {
getCss(item, () => {
loadResource(arr, cb);
});
} else {
console.warn("no suport resource:", item);
loadResource(arr, cb);
}
}
function getJS(url, callback) {
if (document.querySelector(`script[src="${url}"]`)) {
console.log(url, ":has youle");
callback && callback();
return false;
}
var script = document.createElement("script"), fn = callback || function() {
};
script.type = "text/javascript";
if (script.readyState) {
script.onreadystatechange = function() {
if (script.readyState == "loaded" || script.readyState == "complete") {
script.onreadystatechange = null;
fn();
} else {
script.onreadystatechange = null;
fn();
}
console.log(script.readyState);
};
} else {
script.onload = function() {
fn();
};
script.onerror = function(err) {
fn();
console.log(err);
};
}
script.src = url;
document.querySelector("body").appendChild(script);
}
function getCss(url, callback) {
if (document.querySelector(`link[href="${url}"]`)) {
console.log(url, ":has youle");
callback && callback();
return false;
}
var css = document.createElement("link"), fn = callback || function() {
};
css.rel = "stylesheet";
if (css.readyState) {
css.onreadystatechange = function() {
if (css.readyState == "loaded" || css.readyState == "complete") {
css.onreadystatechange = null;
fn();
} else {
css.onreadystatechange = null;
fn();
}
console.log(css.readyState);
};
} else {
css.onload = function() {
fn();
};
css.onerror = function(err) {
fn();
console.log(err);
};
}
css.href = url;
document.querySelector("head").appendChild(css);
}
function parseDom(html2) {
let ele = document.createElement("div");
ele.innerHTML = html2.trim();
return ele.childNodes[0];
}
function afterinsert(newElement, targetElement) {
const parent = targetElement.parentNode;
if (parent.lastChild === targetElement) {
parent.appendChild(newElement);
} else {
parent.insertBefore(newElement, targetElement.nextSibling);
}
}
const STORAGE_KEYS = {
VOLUME: "cto_ali_player_vol",
RATE: "cto_video_Rate"
};
const localStore = {
get(key, defaultVal) {
try {
const val = localStorage.getItem(key);
return val !== null ? JSON.parse(val) : defaultVal;
} catch (e) {
console.error("\u83B7\u53D6\u672C\u5730\u5B58\u50A8\u5931\u8D25:", e);
return defaultVal;
}
},
set(key, val) {
try {
localStorage.setItem(key, JSON.stringify(val));
} catch (e) {
console.error("\u8BBE\u7F6E\u672C\u5730\u5B58\u50A8\u5931\u8D25:", e);
}
}
};
const playerNextHtml = `<div class="player-olympic-player-next"><div class="player-olympic-player-next-tip">\u4E0B\u4E00\u8282</div></div>`;
const index$9 = "";
const nextIcon = "";
class PlayerNextComponent {
constructor(player, clickHandle) {
this.player = player;
this.clickHandle = clickHandle;
this.$html = parseDom(playerNextHtml);
}
createEl(el) {
if (this.clickHandle) {
this.$html.onclick = () => {
console.log("this.clickHandle", this.clickHandle);
if (typeof this.clickHandle == "function") {
this.clickHandle();
} else if (typeof this.clickHandle == "string") {
window.location.href = this.clickHandle;
}
};
this.$html.style.backgroundImage = `url(${nextIcon})`;
const beforeNode = el.querySelector(".vjs-play-control");
afterinsert(this.$html, beforeNode);
}
}
dispose() {
this.$html.parentNode.removeChild(this.$html);
}
}
const playPrevHtml = `<div class="player-olympic-player-prev"><div class="player-olympic-player-prev-tip">\u4E0A\u4E00\u8282</div></div>`;
const index$8 = "";
const prevIcon = "";
class PlayPrevComponent {
constructor(player, clickHandle) {
this.player = player;
this.clickHandle = clickHandle;
this.$html = parseDom(playPrevHtml);
}
createEl(el) {
if (this.clickHandle) {
this.$html.onclick = () => {
if (typeof this.clickHandle == "function") {
this.clickHandle();
} else if (typeof this.clickHandle == "string") {
window.location.href = this.clickHandle;
}
};
this.$html.style.backgroundImage = `url(${prevIcon})`;
const beforeNode = el.querySelector(".vjs-play-control");
afterinsert(this.$html, beforeNode);
}
}
dispose() {
this.$html.parentNode.removeChild(this.$html);
}
}
const memoryPlayHtml = `<div class="memory-play-wrap"></div>`;
const index$7 = "";
class MemoryPlayComponent {
constructor(player, autoPlay = false, getTime, saveTimeFunction, htime) {
this.html = parseDom(memoryPlayHtml);
this.player = player;
this.autoPlay = autoPlay;
this.getTime = getTime || this._getTime;
this.saveTimeFunction = saveTimeFunction || this._saveTime;
this.hasMemoryDisplay = false;
this.htime = htime || 0;
this.memoryKey = "tcplayer_memory_";
}
createEl(el) {
el.appendChild(this.html);
}
ready() {
try {
const player = this.player;
let that = this;
let playerOptions = player.options_ || {};
let memoryVideo = "";
if (playerOptions.fileID) {
memoryVideo = playerOptions.fileID;
} else if (playerOptions.src) {
memoryVideo = playerOptions.src.replace(/\?.*$/, "");
}
this.memoryKey = "tcplayer_memory_" + memoryVideo;
let memoryTime = 0;
if (this.htime > 0) {
memoryTime = this.htime;
} else {
memoryTime = this.getTime(this.memoryKey) ? parseInt(this.getTime(this.memoryKey)) : 0;
}
this.hasMemoryDisplay = false;
var memoryVideoTime_cto, memoryVideoTime, lasttime, seektime;
memoryVideoTime = this.getVideoTime(memoryTime);
memoryVideoTime_cto = this.getVideoTime(this.htime);
if (!this.hasMemoryDisplay && memoryTime > 0) {
this.hasMemoryDisplay = true;
let duration = 0;
try {
duration = player.duration() || 0;
} catch (e) {
console.error("\u83B7\u53D6\u89C6\u9891\u65F6\u957F\u5931\u8D25:", e);
}
if (memoryTime <= parseInt(duration) - 2) {
var memoryDomString = "";
seektime = 0;
lasttime = memoryVideoTime;
if (this.autoPlay) {
memoryDomString = `<div class="memory-play"><span>\u60A8\u4E0A\u6B21\u5B66\u4E60\u5230${lasttime}\u5DF2\u81EA\u52A8\u4E3A\u60A8\u7EED\u64AD</span><span class="play-jump">\u4ECE\u5934\u64AD\u653E</span><i class="iconfont icon-close"></i></div>`;
player.currentTime(memoryTime);
} else {
memoryDomString = `<div class="memory-play"><span>\u60A8\u4E0A\u6B21\u5B66\u4E60\u5230${lasttime}</span><span class="play-jump">\u7EE7\u7EED\u64AD\u653E</span><i class="iconfont icon-close"></i></div>`;
}
if (player.paused()) {
if (this.autoPlay) {
}
}
this.html.innerHTML = memoryDomString;
let timeoutMemory = setTimeout(() => {
this.html.innerHTML = "";
}, 5e3);
var closebtn = this.html.querySelector(".icon-close");
if (!!closebtn) {
closebtn.onclick = () => {
this.html.innerHTML = "";
clearTimeout(timeoutMemory);
};
}
var jumpbtn = this.html.querySelector(".play-jump");
if (!!jumpbtn) {
jumpbtn.onclick = () => {
if (this.autoPlay) {
player.currentTime(0);
} else {
player.currentTime(memoryTime);
}
if (player.paused() && this.autoPlay) {
try {
player.play().catch((err) => {
console.warn("\u70B9\u51FB\u540E\u64AD\u653E\u88AB\u963B\u6B62\uFF0C\u8FD9\u79CD\u60C5\u51B5\u5F88\u5C11\u89C1:", err);
});
} catch (err) {
console.warn("\u64AD\u653E\u5C1D\u8BD5\u5931\u8D25:", err);
}
}
this.html.innerHTML = "";
clearTimeout(timeoutMemory);
};
}
}
}
setTimeout(() => {
if (!this.hasMemoryDisplay && memoryTime > 0) {
this.processMemoryPlayPrompt(memoryTime);
}
}, 500);
this.setupTimeSaving();
} catch (err) {
console.error(err);
}
}
error() {
this.setMemory();
}
dispose() {
this.setMemory();
if (this.visibilityHandler) {
document.removeEventListener("visibilitychange", this.visibilityHandler);
}
if (this.unloadHandler) {
window.removeEventListener("beforeunload", this.unloadHandler);
}
if (this.progressInterval) {
clearInterval(this.progressInterval);
}
}
processMemoryPlayPrompt(memoryTime) {
const player = this.player;
this.hasMemoryDisplay = true;
let duration = 0;
try {
duration = player.duration() || 0;
} catch (e) {
console.error("\u83B7\u53D6\u89C6\u9891\u65F6\u957F\u5931\u8D25:", e);
}
if (memoryTime <= parseInt(duration) - 2) {
const formattedTime = this.getVideoTime(memoryTime);
let memoryDomString = "";
if (this.autoPlay) {
memoryDomString = `<div class="memory-play"><span>\u60A8\u4E0A\u6B21\u5B66\u4E60\u5230${formattedTime}\u5DF2\u81EA\u52A8\u4E3A\u60A8\u7EED\u64AD</span><span class="play-jump">\u4ECE\u5934\u64AD\u653E</span><i class="iconfont icon-close"></i></div>`;
player.currentTime(memoryTime);
} else {
memoryDomString = `<div class="memory-play"><span>\u60A8\u4E0A\u6B21\u5B66\u4E60\u5230${formattedTime}</span><span class="play-jump">\u7EE7\u7EED\u64AD\u653E</span><i class="iconfont icon-close"></i></div>`;
}
this.html.innerHTML = memoryDomString;
let timeoutMemory = setTimeout(() => {
this.html.innerHTML = "";
}, 5e3);
const closeBtn = this.html.querySelector(".icon-close");
if (closeBtn) {
closeBtn.onclick = () => {
this.html.innerHTML = "";
clearTimeout(timeoutMemory);
};
}
const jumpBtn = this.html.querySelector(".play-jump");
if (jumpBtn) {
jumpBtn.onclick = () => {
if (this.autoPlay) {
player.currentTime(0);
} else {
player.currentTime(memoryTime);
}
if (player.paused()) {
try {
player.play().catch((err) => {
console.warn("\u70B9\u51FB\u540E\u64AD\u653E\u88AB\u963B\u6B62:", err);
});
} catch (err) {
console.warn("\u64AD\u653E\u5C1D\u8BD5\u5931\u8D25:", err);
}
}
this.html.innerHTML = "";
clearTimeout(timeoutMemory);
};
}
}
}
setupTimeSaving() {
const player = this.player;
let that = this;
document.removeEventListener("visibilitychange", this.visibilityHandler);
this.visibilityHandler = function() {
if (document.visibilityState === "hidden" && player.currentTime() > 0) {
that.saveTimeFunction(that.memoryKey, player.currentTime());
}
};
document.addEventListener("visibilitychange", this.visibilityHandler);
window.removeEventListener("beforeunload", this.unloadHandler);
this.unloadHandler = function() {
if (player.currentTime() > 0) {
that.saveTimeFunction(that.memoryKey, player.currentTime());
}
};
window.addEventListener("beforeunload", this.unloadHandler);
this.progressInterval = setInterval(() => {
if (!player.paused() && player.currentTime() > 0) {
that.saveTimeFunction(that.memoryKey, player.currentTime());
}
}, 3e4);
}
setMemory() {
const player = this.player;
try {
if (!player || typeof player.currentTime !== "function") {
return;
}
const currentTime = player.currentTime();
if (currentTime > 0) {
this.saveTimeFunction(this.memoryKey, currentTime);
}
} catch (err) {
console.error("\u4FDD\u5B58\u8BB0\u5FC6\u65F6\u95F4\u5931\u8D25:", err);
}
}
getVideoTime(duration) {
let secondTotal = Math.round(duration);
let hour = Math.floor(secondTotal / 3600);
let minute = Math.floor((secondTotal - hour * 3600) / 60);
let second = secondTotal - hour * 3600 - minute * 60;
if (minute < 10) {
minute = "0" + minute;
}
if (second < 10) {
second = "0" + second;
}
return hour === 0 ? minute + ":" + second : hour + ":" + minute + ":" + second;
}
_getTime(memoryVideo) {
return localStorage.getItem(memoryVideo);
}
_saveTime(memoryVideo, currentTime) {
localStorage.setItem(memoryVideo, currentTime);
}
}
const index$6 = "";
const rateHtml = `
<div class="rate-components">
<div class="current-rate">1.0x</div>
<ul class="rate-list">
<li data-rate="2.0">2.0x</li>
<li data-rate="1.75">1.75x</li>
<li data-rate="1.5">1.5x</li>
<li data-rate="1.25">1.25x</li>
<li data-rate="1.0" class="current">1.0x</li>
<li data-rate="0.5">0.5x</li>
</ul>
</div>
`;
const index$5 = "";
class RateComponent {
constructor(player) {
this.html = parseDom(rateHtml);
this.player = player;
this.hasCreated = false;
this.rate = this.getRate();
}
createEl(el) {
try {
if (!el) {
console.warn("RateComponent: DOM\u5143\u7D20\u4E0D\u5B58\u5728");
return;
}
let eleControlbar = el.querySelector(".vjs-control-bar");
if (!eleControlbar) {
console.warn("RateComponent: \u65E0\u6CD5\u627E\u5230\u63A7\u5236\u680F\u5143\u7D20\uFF0C\u5C1D\u8BD5\u6DFB\u52A0\u5230\u6839\u5143\u7D20");
el.appendChild(this.html);
return;
}
eleControlbar.appendChild(this.html);
} catch (err) {
console.error("RateComponent\u521B\u5EFADOM\u5143\u7D20\u5931\u8D25:", err);
}
}
created() {
if (this.hasCreated == false) {
if (this.rate) {
this.player.playbackRate(parseFloat(this.rate));
let li_target_current = this.html.querySelector(`li[class="current"]`);
if (li_target_current) {
li_target_current.className = "";
}
let li_target = this.html.querySelector(`li[data-rate="${this.rate}"]`);
if (li_target) {
li_target.className = "current";
}
} else {
this.rate = "1.0";
}
let currentRateEle = this.html.querySelector(".current-rate");
currentRateEle.innerText = this.rate + "x";
}
this.hasCreated = true;
}
ready() {
let currentRateEle = this.html.querySelector(".current-rate");
let rateListEle = this.html.querySelector(".rate-list");
let timeId = null;
let settingRate = document.querySelector(".vcp-rate");
if (settingRate) {
settingRate.style.display = "none";
}
currentRateEle.onclick = () => {
rateListEle.style.display = "block";
};
currentRateEle.onmouseleave = () => {
timeId = setTimeout(() => {
rateListEle.style.display = "none";
}, 100);
};
rateListEle.onmouseenter = () => {
clearTimeout(timeId);
};
rateListEle.onmouseleave = () => {
rateListEle.style.display = "none";
};
rateListEle.onclick = ({ target }) => {
let rate = target.dataset.rate;
if (rate) {
this.player.playbackRate(parseFloat(rate));
if (target.className !== "current") {
let currentEle = rateListEle.querySelector(".current");
if (currentEle) {
currentEle.className = "";
}
target.className = "current";
}
rateListEle.style.display = "none";
currentRateEle.innerText = rate + "x";
this.setRate(rate);
}
};
}
setRate(Rate) {
localStorage.setItem("cto_video_Rate", Rate);
}
getRate() {
return localStorage.getItem("cto_video_Rate");
}
}
const index$4 = "";
class BulletScreenComponent {
constructor(player, text, style2, bulletPosition = "random") {
this.text = text;
this.style = style2 || { fontSize: "14px", color: "#fff" };
this.html = parseDom('<div class="bullet-screen paused"></div>');
this.bulletPosition = bulletPosition;
this.mintime = 90;
this.maxtime = 180;
this.animationtime = 10;
this.time = 0;
this.stoptime = 0;
this.start = new Date();
this.player = player;
}
createEl(el) {
this.html.innerText = this.text;
el.appendChild(this.html);
}
ready(e) {
const player = this.player;
if (player.autoplay() === false) {
this.html.style.animationPlayState = "paused";
}
Object.keys(this.style).forEach((key) => this.html.style[key] = this.style[key]);
var bulletHeight = this.html.offsetHeight;
var playerHeight = parseInt(player._el.offsetHeight);
var maxHeight = playerHeight - bulletHeight;
if (this.bulletPosition === "bottom") {
this.html.style.bottom = 0;
} else {
let top = this.bulletPosition === "top" ? 0 : this.randomTop(maxHeight);
this.html.style.top = top;
}
if (this.bulletPosition === "random") {
this.html.addEventListener("animationiteration", () => {
this.html.style.top = this.randomTop(maxHeight);
});
}
this.html.style.webkitAnimationDuration = this.animationtime + "s";
this.start = new Date();
}
playing(e) {
this.html.style.animationPlayState = "running";
this.start = new Date();
}
timeupdate(timeStamp) {
const player = this.player;
var randomnum = Math.random();
if (this.maxtime - this.mintime > 0) {
randomnum = this.mintime + (this.maxtime - this.mintime) * randomnum;
} else {
randomnum = 0;
}
let el = player.el();
let componentEl = el.querySelector(".bullet-screen");
this.time = new Date() - this.start + this.stoptime;
if (!componentEl) {
if (this.time - this.animationtime * 1e3 - randomnum * 1e3 > 0) {
el.appendChild(this.html);
this.start = new Date();
this.stoptime = 0;
}
} else if (this.time - this.animationtime * 1e3 > 0) {
componentEl.remove();
this.stoptime = 0;
} else {
if (componentEl.className !== "bullet-screen") {
componentEl.className = "bullet-screen";
}
let cssStyles = getComputedStyle(componentEl);
let display = cssStyles.getPropertyValue("display");
let opacity = cssStyles.getPropertyValue("opacity");
let visibility = cssStyles.getPropertyValue("visibility");
let currentwords = this.text;
let modifywords = componentEl.innerText;
if (display === "none") {
componentEl.style.setProperty("display", "block");
}
if (opacity !== "1") {
componentEl.style.setProperty("opacity", "1");
}
if (visibility === "hidden") {
componentEl.style.setProperty("visibility", "visible");
}
if (currentwords != modifywords) {
componentEl.innerText = currentwords;
}
}
}
pause(e) {
this.html.style.animationPlayState = "paused";
}
randomTop(max) {
return Math.floor(Math.random() * max) + "px";
}
}
const html = {
CountdownHtml: `<div class="countdown-wrap"><div class="countdown"></div></div>`,
RateHtml: `<div class="player-rate-wrap"><div class="player-rate-tips">\u8FD9\u8282\u8BFE\u7A0B\u60A8\u662F\u5426\u6EE1\u610F\uFF1F</div><div class="player-rate-box"><div class="player-star-wrap"><i class="player-icon player-star" data-id="1"></i><i class="player-icon player-star" data-id="2"></i><i class="player-icon player-star" data-id="3"></i><i class="player-icon player-star" data-id="4"></i><i class="player-icon player-star" data-id="5"></i></div><div class="player-rate-txt">\u8BC4\u5206</div></div></div>`
};
const index$3 = "";
class CountdownComponent {
constructor(player, open = true, time, type, starOpen) {
this.html = parseDom(html.CountdownHtml);
this.player = player;
this.open = open;
this.time = time ? time : 15;
this.type = type;
this.isClicked = false;
this.starOpen = starOpen;
this.player.on("timeupdate", () => this.timeupdate());
}
createEl(el) {
if (this.open) {
el.appendChild(this.html);
}
}
timeupdate() {
const player = this.player;
if (this.type == "star") {
var lasttime = player.duration() - player.currentTime();
if (lasttime <= this.time) {
this.starOpen();
}
} else {
if (this.open) {
var txtbox = this.html.querySelector(".countdown");
var lasttime = player.duration() - player.currentTime();
if (lasttime <= this.time) {
lasttime = lasttime - lasttime % 1 + 1;
var countdownString = `<div class="countdown-txt">
<span class="time">${lasttime}s</span>\u540E\u4E3A\u60A8\u64AD\u653E\u4E0B\u4E00\u8282
</div>`;
txtbox.innerHTML = countdownString;
} else {
txtbox.innerHTML = "";
}
}
}
}
getVideoTime(duration) {
let secondTotal = Math.round(duration);
let hour = Math.floor(secondTotal / 3600);
let minute = Math.floor((secondTotal - hour * 3600) / 60);
let second = secondTotal - hour * 3600 - minute * 60;
if (minute < 10) {
minute = "0" + minute;
}
if (second < 10) {
second = "0" + second;
}
return hour === 0 ? minute + ":" + second : hour + ":" + minute + ":" + second;
}
}
const snapshotHtml = `
<div class="prism-snapshot-btn"></div>
`;
const index$2 = "";
class SnapshotComponent {
constructor(player) {
this.player = player;
this.html = parseDom(snapshotHtml);
}
createEl(el) {
const controlBar = el.querySelector(".vjs-control-bar");
if (controlBar) {
const tcpVideoQualitySwitcher = controlBar.querySelector(".vjs-fullscreen-control");
controlBar.insertBefore(this.html, tcpVideoQualitySwitcher.previousSibling);
} else {
console.warn("\u627E\u4E0D\u5230\u64AD\u653E\u5668\u63A7\u5236\u680F\uFF0C\u5C1D\u8BD5\u6DFB\u52A0\u5230\u6839\u5143\u7D20");
el.appendChild(this.html);
}
}
ready() {
this.setupVideoForSnapshot();
this.html.addEventListener("click", () => {
this.captureSnapshot();
});
}
setupVideoForSnapshot() {
const video = this.player.tech_.el_;
if (video && !video.crossOrigin) {
video.crossOrigin = "anonymous";
video.setAttribute("crossOrigin", "anonymous");
}
}
captureSnapshot() {
const player = this.player;
const video = player.tech_.el_;
if (!video) {
console.error("\u89C6\u9891\u5143\u7D20\u4E0D\u5B58\u5728");
this.showErrorMessage("\u89C6\u9891\u5143\u7D20\u4E0D\u5B58\u5728");
return;
}
const canvas = document.createElement("canvas");
canvas.width = video.videoWidth || video.clientWidth;
canvas.height = video.videoHeight || video.clientHeight;
const ctx = canvas.getContext("2d");
try {
ctx.drawImage(video, 0, 0, canvas.width, canvas.height);
const imageData = canvas.toDataURL("image/png");
const link = document.createElement("a");
link.href = imageData;
const timestamp = new Date().toISOString().replace(/[:.]/g, "-");
const title = player.el_.getAttribute("title") || "video";
const filename = `${title}_${timestamp}.png`;
link.download = filename;
document.body.appendChild(link);
link.click();
document.body.removeChild(link);
console.log("\u622A\u56FE\u6210\u529F:", filename);
} catch (error) {
console.error("\u622A\u56FE\u5931\u8D25:", error);
if (error.name === "SecurityError") {
this.tryAlternativeSnapshot(video);
} else {
this.showErrorMessage("\u622A\u56FE\u5931\u8D25\uFF1A" + error.message);
}
}
}
tryAlternativeSnapshot(video) {
try {
const videoSrc = video.src || video.currentSrc;
if (videoSrc) {
console.log("\u5C1D\u8BD5\u91CD\u65B0\u52A0\u8F7D\u89C6\u9891\u5E76\u8BBE\u7F6E\u8DE8\u57DF\u5C5E\u6027...");
const tempVideo = document.createElement("video");
tempVideo.crossOrigin = "anonymous";
tempVideo.muted = true;
tempVideo.style.display = "none";
tempVideo.addEventListener("loadeddata", () => {
tempVideo.currentTime = video.currentTime;
});
tempVideo.addEventListener("seeked", () => {
this.captureFromTempVideo(tempVideo);
document.body.removeChild(tempVideo);
});
document.body.appendChild(tempVideo);
tempVideo.src = videoSrc;
tempVideo.load();
}
} catch (altError) {
console.error("\u66FF\u4EE3\u65B9\u6848\u4E5F\u5931\u8D25\u4E86:", altError);
this.showErrorMessage("\u6240\u6709\u622A\u56FE\u65B9\u6848\u90FD\u5931\u8D25\u4E86\uFF0C\u8BF7\u68C0\u67E5\u89C6\u9891\u6E90\u7684 CORS \u914D\u7F6E");
}
}
captureFromTempVideo(video) {
const canvas = document.createElement("canvas");
canvas.width = video.videoWidth;
canvas.height = video.videoHeight;
const ctx = canvas.getContext("2d");
ctx.drawImage(video, 0, 0, canvas.width, canvas.height);
try {
const imageData = canvas.toDataURL("image/png");
const link = document.createElement("a");
link.href = imageData;
const timestamp = new Date().toISOString().replace(/[:.]/g, "-");
const title = this.player.el_.getAttribute("title") || "video";
const filename = `${title}_${timestamp}.png`;
link.download = filename;
document.body.appendChild(link);
link.click();
document.body.removeChild(link);
} catch (error) {
console.error("\u66FF\u4EE3\u65B9\u6848\u622A\u56FE\u5931\u8D25:", error);
this.showErrorMessage("\u66FF\u4EE3\u65B9\u6848\u622A\u56FE\u5931\u8D25");
}
}
showErrorMessage(message) {
console.warn("[SnapshotComponent]", message);
}
}
const webFullscreenHtml = `
<div class="prism-web-fullscreen-btn" title="\u7F51\u9875\u5168\u5C4F"></div>
`;
const index$1 = "";
class WebFullscreenComponent {
constructor(player) {
this.player = player;
this.html = parseDom(webFullscreenHtml);
this.isWebFullscreen = false;
this.handleToggle = this.handleToggle.bind(this);
this.handleKeydown = this.handleKeydown.bind(this);
}
createEl(el) {
const controlBar = el.querySelector(".vjs-control-bar");
if (controlBar) {
const fullscreenControl = controlBar.querySelector(".vjs-fullscreen-control");
if (fullscreenControl) {
controlBar.insertBefore(this.html, fullscreenControl);
} else {
controlBar.appendChild(this.html);
}
} else {
console.warn("\u627E\u4E0D\u5230\u64AD\u653E\u5668\u63A7\u5236\u680F\uFF0C\u5C1D\u8BD5\u6DFB\u52A0\u5230\u6839\u5143\u7D20");
el.appendChild(this.html);
}
}
ready() {
this.html.addEventListener("click", this.handleToggle);
document.addEventListener("keydown", this.handleKeydown);
}
handleToggle() {
if (this.isWebFullscreen) {
this.exitWebFullscreen();
} else {
this.enterWebFullscreen();
}
}
enterWebFullscreen() {
const playerEl = this.player.el_;
if (!playerEl) {
console.error("\u64AD\u653E\u5668\u5143\u7D20\u4E0D\u5B58\u5728");
return;
}
this.originalStyles = {
position: playerEl.style.position,
top: playerEl.style.top,
left: playerEl.style.left,
width: playerEl.style.width,
height: playerEl.style.height,
zIndex: playerEl.style.zIndex,
backgroundColor: playerEl.style.backgroundColor
};
playerEl.style.position = "fixed";
playerEl.style.top = "0";
playerEl.style.left = "0";
playerEl.style.width = "100vw";
playerEl.style.height = "100vh";
playerEl.style.zIndex = "9999";
playerEl.style.backgroundColor = "#000";
playerEl.classList.add("prism-web-fullscreen");
this.html.classList.add("web-fullscreen");
document.body.style.overflow = "hidden";
this.isWebFullscreen = true;
this.player.trigger("webfullscreenchange", { isWebFullscreen: true });
console.log("\u8FDB\u5165\u7F51\u9875\u5168\u5C4F\u6A21\u5F0F");
}
exitWebFullscreen() {
const playerEl = this.player.el_;
if (!playerEl) {
console.error("\u64AD\u653E\u5668\u5143\u7D20\u4E0D\u5B58\u5728");
return;
}
if (this.originalStyles) {
Object.keys(this.originalStyles).forEach((key) => {
playerEl.style[key] = this.originalStyles[key] || "";
});
}
playerEl.classList.remove("prism-web-fullscreen");
this.html.classList.remove("web-fullscreen");
document.body.style.overflow = "";
this.isWebFullscreen = false;
this.player.trigger("webfullscreenchange", { isWebFullscreen: false });
console.log("\u9000\u51FA\u7F51\u9875\u5168\u5C4F\u6A21\u5F0F");
}
handleKeydown(event) {
if (event.key === "Escape" && this.isWebFullscreen) {
this.exitWebFullscreen();
event.preventDefault();
}
if ((event.ctrlKey || event.metaKey) && event.key === "Enter") {
this.handleToggle();
event.preventDefault();
}
}
destroy() {
if (this.isWebFullscreen) {
this.exitWebFullscreen();
}
if (this.html) {
this.html.removeEventListener("click", this.handleToggle);
}
document.removeEventListener("keydown", this.handleKeydown);
this.player = null;
this.html = null;
}
}
const wideScreenHtml = `
<div class="prism-wide-screen-btn" title="\u5BBD\u5C4F\u663E\u793A"></div>
`;
const index = "";
class WideScreenComponent {
constructor(player, onWideScreenChange) {
this.player = player;
this.html = parseDom(wideScreenHtml);
this.isWideScreen = false;
this.onWideScreenChange = onWideScreenChange;
this.handleToggle = this.handleToggle.bind(this);
this.handleKeydown = this.handleKeydown.bind(this);
}
createEl(el) {
const controlBar = el.querySelector(".vjs-control-bar");
if (controlBar) {
const fullscreenControl = controlBar.querySelector(".vjs-fullscreen-control");
if (fullscreenControl) {
controlBar.insertBefore(this.html, fullscreenControl);
} else {
controlBar.appendChild(this.html);
}
} else {
console.warn("\u627E\u4E0D\u5230\u64AD\u653E\u5668\u63A7\u5236\u680F\uFF0C\u5C1D\u8BD5\u6DFB\u52A0\u5230\u6839\u5143\u7D20");
el.appendChild(this.html);
}
}
ready() {
this.html.addEventListener("click", this.handleToggle);
document.addEventListener("keydown", this.handleKeydown);
}
handleToggle() {
if (this.isWideScreen) {
this.exitWideScreen();
} else {
this.enterWideScreen();
}
}
enterWideScreen() {
if (this.isWideScreen) {
return;
}
this.isWideScreen = true;
this.html.classList.add("wide-screen");
this.player.trigger("widescreenchange", { isWideScreen: true });
if (this.onWideScreenChange && typeof this.onWideScreenChange === "function") {
this.onWideScreenChange(true, this.player);
}
}
exitWideScreen() {
if (!this.isWideScreen) {
return;
}
this.isWideScreen = false;
this.html.classList.remove("wide-screen");
this.player.trigger("widescreenchange", { isWideScreen: false });
if (this.onWideScreenChange && typeof this.onWideScreenChange === "function") {
this.onWideScreenChange(false, this.player);
}
}
handleKeydown(event) {
if ((event.ctrlKey || event.metaKey) && event.key === "w") {
this.handleToggle();
event.preventDefault();
}
}
getWideScreenState() {
return this.isWideScreen;
}
setWideScreenState(isWideScreen) {
if (isWideScreen && !this.isWideScreen) {
this.enterWideScreen();
} else if (!isWideScreen && this.isWideScreen) {
this.exitWideScreen();
}
}
destroy() {
if (this.isWideScreen) {
this.exitWideScreen();
}
if (this.html) {
this.html.removeEventListener("click", this.handleToggle);
}
document.removeEventListener("keydown", this.handleKeydown);
this.player = null;
this.html = null;
this.onWideScreenChange = null;
}
}
function Md5(e) {
var n = 8;
function t(e2) {
return s(r(i(e2), e2.length * n));
}
function r(e2, t2) {
e2[t2 >> 5] |= 128 << t2 % 32, e2[14 + (t2 + 64 >>> 9 << 4)] = t2;
for (var r2 = 1732584193, i2 = -271733879, a = -1732584194, n2 = 271733878, s2 = 0; s2 < e2.length; s2 += 16) {
var o2 = r2, l = i2, u = a, d = n2;
i2 = p(i2 = p(i2 = p(i2 = p(i2 = f(i2 = f(i2 = f(i2 = f(i2 = h2(i2 = h2(i2 = h2(i2 = h2(i2 = c(i2 = c(i2 = c(i2 = c(i2, a = c(a, n2 = c(n2, r2 = c(r2, i2, a, n2, e2[s2 + 0], 7, -680876936), i2, a, e2[s2 + 1], 12, -389564586), r2, i2, e2[s2 + 2], 17, 606105819), n2, r2, e2[s2 + 3], 22, -1044525330), a = c(a, n2 = c(n2, r2 = c(r2, i2, a, n2, e2[s2 + 4], 7, -176418897), i2, a, e2[s2 + 5], 12, 1200080426), r2, i2, e2[s2 + 6], 17, -1473231341), n2, r2, e2[s2 + 7], 22, -45705983), a = c(a, n2 = c(n2, r2 = c(r2, i2, a, n2, e2[s2 + 8], 7, 1770035416), i2, a, e2[s2 + 9], 12, -1958414417), r2, i2, e2[s2 + 10], 17, -42063), n2, r2, e2[s2 + 11], 22, -1990404162), a = c(a, n2 = c(n2, r2 = c(r2, i2, a, n2, e2[s2 + 12], 7, 1804603682), i2, a, e2[s2 + 13], 12, -40341101), r2, i2, e2[s2 + 14], 17, -1502002290), n2, r2, e2[s2 + 15], 22, 1236535329), a = h2(a, n2 = h2(n2, r2 = h2(r2, i2, a, n2, e2[s2 + 1], 5, -165796510), i2, a, e2[s2 + 6], 9, -1069501632), r2, i2, e2[s2 + 11], 14, 643717713), n2, r2, e2[s2 + 0], 20, -373897302), a = h2(a, n2 = h2(n2, r2 = h2(r2, i2, a, n2, e2[s2 + 5], 5, -701558691), i2, a, e2[s2 + 10], 9, 38016083), r2, i2, e2[s2 + 15], 14, -660478335), n2, r2, e2[s2 + 4], 20, -405537848), a = h2(a, n2 = h2(n2, r2 = h2(r2, i2, a, n2, e2[s2 + 9], 5, 568446438), i2, a, e2[s2 + 14], 9, -1019803690), r2, i2, e2[s2 + 3], 14, -187363961), n2, r2, e2[s2 + 8], 20, 1163531501), a = h2(a, n2 = h2(n2, r2 = h2(r2, i2, a, n2, e2[s2 + 13], 5, -1444681467), i2, a, e2[s2 + 2], 9, -51403784), r2, i2, e2[s2 + 7], 14, 1735328473), n2, r2, e2[s2 + 12], 20, -1926607734), a = f(a, n2 = f(n2, r2 = f(r2, i2, a, n2, e2[s2 + 5], 4, -378558), i2, a, e2[s2 + 8], 11, -2022574463), r2, i2, e2[s2 + 11], 16, 1839030562), n2, r2, e2[s2 + 14], 23, -35309556), a = f(a, n2 = f(n2, r2 = f(r2, i2, a, n2, e2[s2 + 1], 4, -1530992060), i2, a, e2[s2 + 4], 11, 1272893353), r2, i2, e2[s2 + 7], 16, -155497632), n2, r2, e2[s2 + 10], 23, -1094730640), a = f(a, n2 = f(n2, r2 = f(r2, i2, a, n2, e2[s2 + 13], 4, 681279174), i2, a, e2[s2 + 0], 11, -358537222), r2, i2, e2[s2 + 3], 16, -722521979), n2, r2, e2[s2 + 6], 23, 76029189), a = f(a, n2 = f(n2, r2 = f(r2, i2, a, n2, e2[s2 + 9], 4, -640364487), i2, a, e2[s2 + 12], 11, -421815835), r2, i2, e2[s2 + 15], 16, 530742520), n2, r2, e2[s2 + 2], 23, -995338651), a = p(a, n2 = p(n2, r2 = p(r2, i2, a, n2, e2[s2 + 0], 6, -198630844), i2, a, e2[s2 + 7], 10, 1126891415), r2, i2, e2[s2 + 14], 15, -1416354905), n2, r2, e2[s2 + 5], 21, -57434055), a = p(a, n2 = p(n2, r2 = p(r2, i2, a, n2, e2[s2 + 12], 6, 1700485571), i2, a, e2[s2 + 3], 10, -1894986606), r2, i2, e2[s2 + 10], 15, -1051523), n2, r2, e2[s2 + 1], 21, -2054922799), a = p(a, n2 = p(n2, r2 = p(r2, i2, a, n2, e2[s2 + 8], 6, 1873313359), i2, a, e2[s2 + 15], 10, -30611744), r2, i2, e2[s2 + 6], 15, -1560198380), n2, r2, e2[s2 + 13], 21, 1309151649), a = p(a, n2 = p(n2, r2 = p(r2, i2, a, n2, e2[s2 + 4], 6, -145523070), i2, a, e2[s2 + 11], 10, -1120210379), r2, i2, e2[s2 + 2], 15, 718787259), n2, r2, e2[s2 + 9], 21, -343485551), r2 = g(r2, o2), i2 = g(i2, l), a = g(a, u), n2 = g(n2, d);
}
return Array(r2, i2, a, n2);
}
function o(e2, t2, r2, i2, a, n2) {
return g(function(e3, t3) {
return e3 << t3 | e3 >>> 32 - t3;
}(g(g(t2, e2), g(i2, n2)), a), r2);
}
function c(e2, t2, r2, i2, a, n2, s2) {
return o(t2 & r2 | ~t2 & i2, e2, t2, a, n2, s2);
}
function h2(e2, t2, r2, i2, a, n2, s2) {
return o(t2 & i2 | r2 & ~i2, e2, t2, a, n2, s2);
}
function f(e2, t2, r2, i2, a, n2, s2) {
return o(t2 ^ r2 ^ i2, e2, t2, a, n2, s2);
}
function p(e2, t2, r2, i2, a, n2, s2) {
return o(r2 ^ (t2 | ~i2), e2, t2, a, n2, s2);
}
function g(e2, t2) {
var r2 = (65535 & e2) + (65535 & t2);
return (e2 >> 16) + (t2 >> 16) + (r2 >> 16) << 16 | 65535 & r2;
}
function i(e2) {
for (var t2 = Array(), r2 = (1 << n) - 1, i2 = 0; i2 < e2.length * n; i2 += n)
t2[i2 >> 5] |= (e2.charCodeAt(i2 / n) & r2) << i2 % 32;
return t2;
}
function s(e2) {
for (var t2 = "0123456789abcdef", r2 = "", i2 = 0; i2 < 4 * e2.length; i2++)
r2 += t2.charAt(e2[i2 >> 2] >> i2 % 4 * 8 + 4 & 15) + t2.charAt(e2[i2 >> 2] >> i2 % 4 * 8 & 15);
return r2;
}
return t(e);
}
const ApiSign = `HRkJaX81cAbXdFjWe4Qg6HXxtCR7bJRr`;
const ErrorTypeEnum = {
ERROR: "error",
CONNECT_BUFFER: "connectBuffer",
SEEK_VIDEO_BUFFER: "seekVideoBuffer",
VIDEO_BUFFER: "videoBuffer"
};
const PlayTypeEnum = {
M3U8: 1,
VIDEO: 2
};
const LogStatus = {
NORMAL: 0,
BLOCK_TIME_OVERFLOW: -1,
DOWNLINK_TOO_LOW: -2
};
const VIDEO_BLOCK_LIMIT = 1e3;
class ErrorReportComponent {
constructor(player, params, options) {
this.player = player;
this.params = params;
this.currentTime = 0;
this.ali_dt = "";
this.definition = "";
this.errorCode = 0;
this.block_time = 0;
this.blockStartTime = 0;
this.speed = 1;
this.errorType = "";
this.is_first = true;
this.net_speed = 0;
this.bitrate = 0;
this.videoResources = [];
this.hasOffline = false;
this.netTestTimer = null;
this.netTestTimeout = 6e4;
}
created() {
console.log("created----==--==-=--=--=--=-");
window.playerEl = this.player;
const player = this.player;
this.videoResources = player._urls;
this.net_speed = 0;
this.is_first = true;
this.systemEventListen();
this.logger("created videoResources", this.videoResources);
}
logger(...args) {
const isProduction = window.location.host.startsWith("edu.51cto") || false;
if (isProduction) {
return;
}
console.log(...args);
}
getQuality() {
let Quality = "\u81EA\u52A8";
try {
const targetElement = document.querySelector(
".tcp-video-quality-switcher .vjs-selected .vjs-menu-item-text"
);
if (targetElement) {
const textContent = targetElement.textContent.trim();
Quality = textContent;
} else {
console.warn("\u672A\u627E\u5230\u76EE\u6807\u5143\u7D20\uFF0C\u8BF7\u68C0\u67E5\u9009\u62E9\u5668\u6216\u9875\u9762\u72B6\u6001");
Quality = "\u81EA\u52A8";
}
} catch (error) {
Quality = "\u672A\u77E5";
}
this.fp = Quality;
}
getBitrate() {
let bitrate = "";
this.bitrate = bitrate;
}
async playing() {
this.logger("playing:", Date.now());
this.logger("playing-blockStartTime:", this.blockStartTime);
this.is_first = false;
this.ali_dt = this.player.getDuration();
if (this.blockStartTime > 0) {
const blockDuration = Date.now() - this.blockStartTime;
this.logger("\u7F13\u51B2\u4E86", `${this.block_time / 1e3}s`);
if (blockDuration > VIDEO_BLOCK_LIMIT) {
this.block_time = blockDuration;
const data = await this.generateRequestParams();
await this.errorRequest(data);
}
}
}
ended() {
this.errorCode = 0;
this.resetBlockState();
}
resetBlockState() {
this.blockStartTime = 0;
this.block_time = 0;
this.errorType = "";
}
pause() {
this.resetBlockState();
}
waiting() {
const player = this.player;
this.logger("\u5F00\u59CB\u7F13\u51B2:", Date.now());
this.logger("\u5F00\u59CB\u7F13\u51B2--seeking:", player._seeking);
this.currentTime = player.getCurrentTime();
this.blockStartTime = Date.now();
this.speed = localStorage.getItem("cto_video_Rate") ? localStorage.getItem("cto_video_Rate") : this.speed;
if (this.is_first) {
this.errorType = ErrorTypeEnum.CONNECT_BUFFER;
} else if (player._seeking) {
this.errorType = ErrorTypeEnum.SEEK_VIDEO_BUFFER;
} else {
this.errorType = ErrorTypeEnum.VIDEO_BUFFER;
}
}
timeupdate() {
}
async error(player, error) {
this.logger("error", error);
this.errorCode = error.error_code;
this.resetBlockState();
this.errorType = ErrorTypeEnum.ERROR;
const data = await this.generateRequestParams();
await this.errorRequest(data);
}
destroyed() {
this.errorCode = 0;
this.resetBlockState();
}
getQuality() {
let Quality = "\u81EA\u52A8";
try {
const targetElement = document.querySelector(
".tcp-video-quality-switcher .vjs-selected .vjs-menu-item-text"
);
if (targetElement) {
const textContent = targetElement.textContent.trim();
Quality = textContent;
} else {
console.warn("\u672A\u627E\u5230\u76EE\u6807\u5143\u7D20\uFF0C\u8BF7\u68C0\u67E5\u9009\u62E9\u5668\u6216\u9875\u9762\u72B6\u6001");
Quality = "\u81EA\u52A8";
}
} catch (error) {
Quality = "\u672A\u77E5";
}
this.definition = Quality;
}
getDownLink() {
if (window.navigator && window.navigator.connection) {
return window.navigator.connection.downlink || "";
}
}
getAccumulation() {
if (this.player) {
const ReportingComponent2 = this.player.getComponent("ReportingComponent");
this.logger("ReportingComponent", ReportingComponent2);
if (ReportingComponent2) {
return ReportingComponent2.accumulation;
}
}
}
systemEventListen() {
window.addEventListener("online", this.onNetworkOnline.bind(this));
window.addEventListener("offline", this.onNetworkOffLine.bind(this));
window.addEventListener("visibleChange", this.onVisibleChange.bind(this));
}
onVisibleChange() {
this.resetBlockState();
}
onNetworkOnline() {
this.logger("online");
this.resetBlockState();
}
onNetworkOffLine() {
this.logger("offline");
this.hasOffline = true;
this.resetBlockState();
}
startSeek() {
this.resetBlockState();
}
async generateRequestParams() {
const type = this.params.video_id.length === 32 ? PlayTypeEnum.M3U8 : PlayTypeEnum.VIDEO;
const pt = this.getAccumulation();
this.getBitrate();
this.getQuality();
var baseData = {
video_id: this.params.video_id,
user_id: this.params.user_id,
uuid: Md5("" + this.params.timestr + this.params.user_id + this.params.random).toString(),
type,
htime: this.currentTime,
dt: this.ali_dt,
pt,
ref: location.href,
sgin: Md5("" + this.params.timestr + ApiSign).toString(),
platform: 5,
ts_url: this.definition,
code: this.errorCode,
speed: this.speed,
eType: this.errorType,
bitrate: this.bitrate,
net_speed: 0,
downlink: this.getDownLink(),
has_offline: this.hasOffline,
log_status: LogStatus.NORMAL
};
baseData.log_status = this.calcLogStatus(baseData);
var data = {};
if ([ErrorTypeEnum.ERROR].includes(this.errorType)) {
data = Object.assign(baseData, {
time: new Date().getTime()
});
}
if ([ErrorTypeEnum.CONNECT_BUFFER, ErrorTypeEnum.SEEK_VIDEO_BUFFER, ErrorTypeEnum.VIDEO_BUFFER].includes(this.errorType)) {
data = Object.assign(baseData, {
time: this.params.timestr,
block_time: this.block_time
});
}
return data;
}
isParamInvalid(data) {
if (![ErrorTypeEnum.ERROR].includes(this.errorType) && data.block_time <= VIDEO_BLOCK_LIMIT) {
return true;
}
if (data.block_time > data.ali_dt) {
return true;
}
if (data.block_time > 10 * 60 * 1e3) {
return true;
}
return false;
}
async errorRequest(data) {
this.logger("\u9519\u8BEF\u4E0A\u62A5", data);
const that = this;
if (this.isParamInvalid(data)) {
return true;
}
return new Promise((resolve, reject) => {
$.ajax({
type: "post",
url: `${this.params.error_url}`,
data,
dataType: "json",
async: false,
success: function(result) {
},
complete: function(res) {
that.onReportComplete();
resolve();
}
});
});
}
onReportComplete() {
this.logger("onReportComplete");
this.resetBlockState();
this.errorCode = 0;
this.hasOffline = false;
}
calcLogStatus(baseData) {
const blockSecond = baseData.block_time ? baseData.block_time / 1e3 : 0;
if (baseData.eType === ErrorTypeEnum.CONNECT_BUFFER) {
if (blockSecond > 5 * 60) {
return LogStatus.BLOCK_TIME_OVERFLOW;
}
} else {
if (blockSecond > this.ali_dt) {
return LogStatus.BLOCK_TIME_OVERFLOW;
}
if (blockSecond > 5 * 60) {
return LogStatus.BLOCK_TIME_OVERFLOW;
}
}
if (baseData.downlink > 0 && baseData.bitrate > 0