// ==UserScript==
// @name HuyaTool 虎牙直播插件
// @namespace https://github.com/WithoutHair/Huya-Tool
// @version 1.0
// @description 查看主播数据/B站直播同屏观看
// @author Mywait
// @match *://www.huya.com/*
// @icon 
// @grant GM_xmlhttpRequest
// @grant GM_log
// @grant unsafeWindow
// @connect doseeing.com
// @connect bilibili.com
// @connect huya.com
// @require https://cdn.bootcdn.net/ajax/libs/flv.js/1.6.2/flv.min.js
// @downloadURL https://update.greasyfork.icu/scripts/446620/HuyaTool%20%E8%99%8E%E7%89%99%E7%9B%B4%E6%92%AD%E6%8F%92%E4%BB%B6.user.js
// @updateURL https://update.greasyfork.icu/scripts/446620/HuyaTool%20%E8%99%8E%E7%89%99%E7%9B%B4%E6%92%AD%E6%8F%92%E4%BB%B6.meta.js
// ==/UserScript==
let isHostRoom = () => {
try {
let room_id = document.getElementsByClassName('host-rid')[0].children[1].textContent
return room_id
} catch (error) {
return false
}
}
let getHostStat = () => {
let room_id = isHostRoom()
GM_xmlhttpRequest({
method: 'GET',
url: `https://www.doseeing.com/huya/data/api/rank?rids=${room_id}&dt=0&rank_type=chat_pv`,
responseType: 'json',
onload: function(res) {
res = res.response.result.result[0]
let parentNode = document.getElementsByClassName('ht-item')[0],
statDom = document.createElement('div')
statDom.className = 'ht-stat'
parentNode.appendChild(statDom)
statDom.innerHTML = `
总收入
${res['gift.paid.price'] / 100}元
付费人数 ${res['gift.paid.uv']}
总礼物
${res['gift.all.price'] / 100}元
送礼人数 ${res['gift.all.uv']}
总弹幕
${res['chat.pv']}条
弹幕人数 ${res['chat.uv']}
开播时长
${res['online.minutes']}分
活跃人数 ${res['active.uv']}
`
}
})
}
let getRoomId = (id) => {
return new Promise((resolve, reject) => {
GM_xmlhttpRequest({
method: 'GET',
url: `https://api.live.bilibili.com/room/v1/Room/room_init?id=${id}`,
responseType: 'json',
onload: function(response) {
resolve(response.response)
}
})
})
}
let getStreaming = (room_id, qn = 150) => {
/* 当platform为web流才为flv格式 h5为m3u8
0: {qn: 10000, desc: '原画'}
1: {qn: 400, desc: '蓝光'}
2: {qn: 250, desc: '超清'}
3: {qn: 150, desc: '高清'}
*/
return new Promise((resolve, reject) => {
GM_xmlhttpRequest({
method: "GET",
url: `https://api.live.bilibili.com/room/v1/Room/playUrl?cid=${room_id}&qn=${qn}&platform=web`,
responseType: "json",
onload: function(response) {
resolve(response.response)
}
})
})
}
let InitInput = () => {
let inputUrlCon = document.createElement('div')
inputUrlCon.className = 'ht-input-con'
inputUrlCon.innerHTML = `
同屏观看(暂仅支持B站)
`
document.body.appendChild(inputUrlCon)
let hide = () => {
inputUrlCon.className = inputUrlCon.className.replace('display', '')
}
document.getElementsByClassName('ht-button-ok')[0].onclick = async () => {
hide()
let urlId = inputUrlCon.children[2].value.split('/').pop()
let key = `b-${urlId}`
let {data} = await getRoomId(urlId)
let stream = await getStreaming(data.room_id)
createVideo(stream.data, key)
}
document.getElementsByClassName('ht-button-cancel')[0].onclick = () => {
hide()
}
}
let InitMenu = () => {
let menuDom = document.createElement('div')
menuDom.className = 'ht-fixed'
menuDom.onclick = (e) => {
if (e.target.parentNode.parentNode.className.includes('active')) {
menuDom.className = 'ht-fixed'
} else {
menuDom.className = 'ht-fixed active'
}
}
document.body.appendChild(menuDom)
menuDom.innerHTML = `
`
document.getElementsByClassName('ht-item')[1].onclick = () => {
document.getElementsByClassName('ht-input-con')[0].className += ' display'
}
}
let createVideo = (data, key) => {
if (flvjs.isSupported()) {
let video = document.createElement('video'),
con = document.createElement('div'),
title = document.createElement('div'),
quality = document.createElement('select'),
path = document.createElement('select'),
resizeDot = document.createElement('div'),
insertButton = document.createElement('div'),
restoreButton = document.createElement('div'),
flvPlayer = flvjs.createPlayer({
type: 'flv',
url: data.durl[0].url
})
flvPlayer.attachMediaElement(video)
flvPlayer.load()
let now = document.getElementById('hy-video') || document.getElementById('player-recommend')
video.controls = true
con.className = `ht-video-con ${key}`
con.draggable = true
setDragEvent(con)
title.className = 'ht-video-title'
insertButton.className = 'ht-button'
insertButton.innerHTML = '嵌入直播间'
restoreButton.className = 'ht-button'
restoreButton.innerHTML = '恢复直播间'
restoreButton.style.display = 'none'
resizeDot.className = 'ht-video-resize'
resizeDot.draggable = true
let qualitys = data.quality_description,
paths = data.durl
qualitys.forEach(q => {
quality.innerHTML += `