// ==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 += `