// ==UserScript== // @name ChatGPT Voice On LiveKit Meet // @namespace github.com/hmjz100 // @version 1.0.1 // @description 跳转 ChatGPT 镜像站的语音功能到 LiveKit Meet 而不是镜像站的 LiveKit Meet。始皇的镜像站甚至支持选择高级语音模式和选择模型 // @license MIT // @author hmjz100 // @match https://new.oaifree.com/* // @match https://chat.rawchat.top/* // @match https://chat.sharedchat.cn/* // @match https://gpt.github.cn.com/* // @match https://free.xyhelper.cn/* // @icon  // @require https://unpkg.com/jquery@3.6.3/dist/jquery.min.js // @grant GM_setValue // @grant GM_getValue // @grant GM_openInTab // @grant unsafeWindow // @downloadURL none // ==/UserScript== (function () { 'use strict'; // 隐藏原来的按钮 waitForKeyElements('div:not(#ChatGPTVoice-On-LiveKitMeet-Button, #immersive-translate-popup) svg.icon[width="25"][height="25"], div#voiceButton svg, #of-custom-floating-ball svg', function (element) { element.parent().hide(); }); waitForKeyElements("body main div.mb-7.text-center, div.btn-voice, body main div.flex-shrink-0 svg", function (element) { if (element.hasClass("voice")) return; let checkOpacity = function () { if (element.hasClass("mb-7")) { if (element.find("h1.result-streaming").css('opacity') == '0') { let clone = element.clone(true); clone.addClass("voice") element.replaceWith(clone); clone.css({ 'cursor': 'pointer', 'user-select': 'none', '-webkit-user-select': 'none', '-ms-user-select': 'none', '-moz-user-select': 'none' }); clone.on('click touchend', handleVoiceClick); } else { setTimeout(checkOpacity, 100); } } else { element.addClass("voice") element.css({ 'cursor': 'pointer', 'user-select': 'none', '-webkit-user-select': 'none', '-ms-user-select': 'none', '-moz-user-select': 'none' }); element.on('click touchend', handleVoiceClick); } }; // 初次调用检查函数 checkOpacity(); }); let html = $(`
语音
`) let button = html.find('#ChatGPTVoice-On-LiveKitMeet-Button'); let isDragging = false; let offsetY = 0; let dragStartTime; // 从 GM 获取按钮位置 if (GM_getValue('buttonTop')) { button.css('top', GM_getValue('buttonTop') + 'px'); } // 点击事件处理 button.on('click touchend', handleVoiceClick); // 鼠标按下事件 button.on('mousedown touchstart', function (e) { e.preventDefault(); dragStartTime = Date.now(); // 记录拖动开始时间 offsetY = e.clientY - button.offset().top; }); // 鼠标移动事件 $(document).on('mousemove touchmove', function (e) { if (offsetY !== undefined) { let newTop = e.clientY - offsetY; const buttonHeight = button.outerHeight(); const windowHeight = $(window).height(); // 限制按钮位置 if (newTop < 0) newTop = 0; if (newTop + buttonHeight > windowHeight) newTop = windowHeight - buttonHeight; // 判断是否拖动 if (isDragging || (Date.now() - dragStartTime > 100)) { // 如果已经拖动或拖动时间超过100ms isDragging = true; button.addClass('is-dragging'); button.css('top', newTop + 'px'); GM_setValue('buttonTop', newTop); } } }); // 鼠标抬起事件 $(document).on('mouseup touchend', function () { if (isDragging) { setTimeout(function () { isDragging = false; button.removeClass('is-dragging'); }, 100) } offsetY = undefined; // 重置 offsetY }); setInterval(function () { if (!$('#ChatGPTVoice-On-LiveKitMeet-Button').length || !$('#ChatGPTVoice-On-LiveKitMeet-Style').length) { $('#ChatGPTVoice-On-LiveKitMeet').remove() $('body').append(html); } }, 500) // 绑定点击事件到新创建的按钮 async function handleVoiceClick(event) { if (!event?.currentTarget || isDragging) return; let element = $(event.currentTarget); if (element.attr('data-clicked') === 'true') return; element.attr('data-clicked', 'true'); // 异步获取语音链接 await goVoice(element).catch(function (error) { alert('获取语音对话(会议)链接错误: \n' + error.message); console.error(error); element.removeAttr('data-clicked'); }); }; async function goVoice(element) { // 定义不同服务器的配置 let servers = { "new.oaifree.com": { apiPath: "/api/voice/link", apiType: "POST", url: "wss://webrtc.oaifree.com", model: new URL(location.href).searchParams.get('model'), mode: [['标准语音', '高级语音'], ['std', 'adv']], getToken: data => new URL(data.url).searchParams.get('token'), getHash: data => new URL(data.url).hash }, "chat.rawchat.top": { apiPath: "/backend-api/voice_token", apiType: "GET", url: data => data.url, getToken: data => data.token, getHash: data => data.e2ee_key }, "chat.sharedchat.cn": { apiPath: "/backend-api/voice_token", apiType: "GET", url: data => data.url, getToken: data => data.token, getHash: data => data.e2ee_key }, "gpt.github.cn.com": { apiPath: "/backend-api/voice_token", apiType: "GET", url: data => data.url, getToken: data => data.token, getHash: data => data.e2ee_key }, "free.xyhelper.cn": { apiPath: "/backend-api/voice_token", apiType: "GET", url: data => data.url, getToken: data => data.token, getHash: data => data.e2ee_key } }; // 获取当前服务器的域名 let host = location.hostname; // 获取服务器配置 let config = servers[host]; if (!config) { throw new Error(`未知服务器: ${host}`); } let extra = { method: config.apiType, headers: { 'Content-Type': 'application/json' } } if (config.model !== undefined && config.mode !== undefined && config.apiType === 'POST') { let model = config.model; let mode = config.mode; let modeChoice; if (mode && mode.length) { let modeOptions = mode[0] .map((name, index) => `(${index + 1}) ${name}`) .join(" "); let userChoice = prompt(`请选择语音模式: (不输入则使用${mode[0][0]})\n${modeOptions}`); let choiceIndex = parseInt(userChoice) - 1; if (choiceIndex >= 0 && choiceIndex < mode[1].length) { modeChoice = mode[1][choiceIndex]; } else if (userChoice === null) { return element.removeAttr('data-clicked'); } else { modeChoice = mode[1][0]; } } if (!model) { let userInput = prompt("请输入模型名称: (不输入则使用默认模型)"); if (userInput === null) { return element.removeAttr('data-clicked'); } model = userInput; } extra.body = JSON.stringify({ model, mode: modeChoice }); } // 发送请求到语音API let response = await unsafeWindow.fetch(config.apiPath, extra); // 解析返回的JSON数据 let data = await response.json(); console.log('服务数据: \n', data); // 检查返回的模式,如果是高级模式,修改颜色 if (data.mode === "advanced") { element.css('color', '#f00'); } // 检查是否有url或者token,否则抛出错误 if (!data.url) { throw new Error(data.detail || 'No Data provided by server'); } // 获取url、token、hash let url = typeof config.url === 'function' ? config.url(data) : config.url; let token = config.getToken ? config.getToken(data) : null; let hash = config.getHash ? config.getHash(data) : null; // 打印日志方便调试 console.log('会议数据: \n', { token, hash, url }); // 检查是否有url或者token,否则抛出错误 if (!url || !token || !hash) throw new Error(data.detail || '语音服务未返回数据'); // 构建 meetUrl let meetUrl = new URL('https://meet.livekit.io/custom'); if (url) meetUrl.searchParams.set('liveKitUrl', url); if (token) meetUrl.searchParams.set('token', token); if (hash) meetUrl.hash = hash; // 打开新页面 GM_openInTab(meetUrl.href, { active: true, insert: true, setParent: true }) element.removeAttr('data-clicked'); } function waitForKeyElements(selectorTxt, actionFunction, bWaitOnce, iframeSelector) { function findInShadowRoots(root, selector) { let elements = $(root).find(selector).toArray(); $(root).find('*').each(function () { let shadowRoot = this.shadowRoot; if (shadowRoot) { elements = elements.concat(findInShadowRoots(shadowRoot, selector)); } }); return elements; } var targetElements; if (iframeSelector) { targetElements = $(iframeSelector).contents(); } else { targetElements = $(document); } let allElements = findInShadowRoots(targetElements, selectorTxt); if (allElements.length > 0) { allElements.forEach(function (element) { var jThis = $(element); var uniqueIdentifier = 'alreadyFound'; var alreadyFound = jThis.data(uniqueIdentifier) || false; if (!alreadyFound) { var cancelFound = actionFunction(jThis); if (cancelFound) { return false; } else { jThis.data(uniqueIdentifier, true); } } }); } var controlObj = waitForKeyElements.controlObj || {}; var controlKey = selectorTxt.replace(/[^\w]/g, "_"); var timeControl = controlObj[controlKey]; if (allElements.length > 0 && bWaitOnce && timeControl) { clearInterval(timeControl); delete controlObj[controlKey]; } else { if (!timeControl) { timeControl = setInterval(function () { waitForKeyElements(selectorTxt, actionFunction, bWaitOnce, iframeSelector); }, 1000); controlObj[controlKey] = timeControl; } } waitForKeyElements.controlObj = controlObj; } })();