// ==UserScript== // @name Extract Douyin Live Stream URLs // @name:zh-CN 抖音直播流提取 // @namespace Cassandre // @version 2.0 // @description Extract stream URLs from Douyin live streams // @description:zh-CN 提取抖音直播地址 // @author Cassandre Cora // @license MIT // @icon https://p3-pc-weboff.byteimg.com/tos-cn-i-9r5gewecjs/logo-horizontal-small.svg // @match https://live.douyin.com/* // @match https://www.douyin.com/* // @connect live.douyin.com // @run-at document-end // @grant GM_addStyle // @grant GM_setClipboard // @grant GM_xmlhttpRequest // @grant GM_getValue // @grant GM_setValue // @downloadURL https://update.greasyfork.icu/scripts/515942/Extract%20Douyin%20Live%20Stream%20URLs.user.js // @updateURL https://update.greasyfork.icu/scripts/515942/Extract%20Douyin%20Live%20Stream%20URLs.meta.js // ==/UserScript== (function () { 'use strict'; let dragBallTop = GM_getValue('dragBallTop'); dragBallTop = dragBallTop ?? '50%'; GM_setValue('dragBallTop', dragBallTop); const STYLES = ` .douyin-stream-url-side-button { position: fixed; z-index: 19998; right: 0; width: 40px; height: 40px; border: none; outline: none; cursor: pointer; color: white; text-align: center; background: linear-gradient(135deg, #FE2C55 0%, #FF4B75 100%); border-radius: 50%; transition: all 0.3s ease; box-shadow: 0 2px 8px rgba(0, 0, 0, 0.2); display: flex; align-items: center; justify-content: center; background-image: url(); background-size: 100% 100%; } .douyin-stream-url-side-button:hover { transform: scale(1.08); } .douyin-stream-url-close-button { position: absolute; top: -8px; right: -8px; width: 24px; height: 24px; border: 2px solid #FE2C55; border-radius: 50%; background: white; cursor: pointer; outline: none; font-size: 14px; display: flex; align-items: center; justify-content: center; transition: all 0.3s ease; box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1); } .douyin-stream-url-close-button:hover { transform: scale(1.1); background: rgb(254, 44, 85); color: white; font-weight: bold; } #douyin-stream-url-app { position: fixed; right: 20px; width: 320px; height: auto; opacity: 0; background-color: rgba(24, 24, 24, 0.95); color: #e0e0e0; padding: 15px; font-size: 13px; font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif; z-index: 9999; border-radius: 16px; transform: translateX(110%); transition: all 0.5s cubic-bezier(0.4, 0, 0.2, 1); box-shadow: 0 8px 24px rgba(0, 0, 0, 0.2); backdrop-filter: blur(10px); border: 2px solid rgba(255,255,255,.7) } .douyin-stream-url-list-container { background-color: rgba(31, 31, 31, 0.8); border-radius: 12px; box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15); margin-bottom: 12px; overflow: hidden; } .douyin-stream-url-list-container:last-child { margin-bottom: 0; } .douyin-stream-url-list-header { background-color: rgba(45, 45, 45, 0.8); padding: 10px 12px; font-weight: 600; font-size: 14px; color: #ffffff; border-bottom: 1px solid rgba(255, 255, 255, 0.1); } .douyin-hls-stream-url-list-content, .douyin-flv-stream-url-list-content { padding: 8px; overflow-x: auto; white-space: nowrap; background-color: rgba(38, 38, 38, 0.8); font-family: "SFMono-Regular", Consolas, "Liberation Mono", Menlo, Courier, monospace; font-size: 13px; line-height: 1.6; color: #d1d1d1; max-height: 150px; overflow-y: auto; border-bottom: 1px solid rgba(255, 255, 255, 0.1); } .douyin-stream-url-button-container { display: flex; gap: 4px; padding: 4px; } .douyin-hls-stream-url-copy-button, .douyin-flv-stream-url-copy-button { display: flex; align-items: center; justify-content: center; width: 50%; padding: 10px; background: rgb(254, 44, 85); color: white; border: none; border-radius: 8px; cursor: pointer; transition: all 0.3s ease; font-size: 14px; font-weight: 500; } .douyin-hls-stream-url-copy-button:hover, .douyin-flv-stream-url-copy-button:hover { transform: translateY(-1px); background: rgb(210, 27, 70); } .douyin-stream-url-download-all-button { display: flex; align-items: center; justify-content: center; width: 100%; padding: 12px; background: rgb(254, 44, 85); color: white; border: none; cursor: pointer; transition: all 0.3s ease; font-size: 14px; font-weight: 500; border-radius: 10px; margin-top: 8px; } .douyin-stream-url-download-all-button:hover { transform: translateY(-1px); background: rgb(210, 27, 70); } `; const QUALITY_LEVELS = ['FULL_HD1', 'HD1', 'SD2', 'SD1']; const QUALITY_MAP = { 'FULL_HD1': '原画', 'HD1': '超清', 'SD2': '高清', 'SD1': '标清' }; // current room ID let currentRid = null; // Inject styles GM_addStyle(STYLES); // Debounce function function debounce(func, wait) { let timeout; return function (...args) { clearTimeout(timeout); timeout = setTimeout(() => func.apply(this, args), wait); }; } // Initialize based on domain function init() { const domain = window.location.hostname; switch (domain) { case 'live.douyin.com': initLivePage(); break; case 'www.douyin.com': initUrlChangeListener(); break; } } // Initialize live page function initLivePage() { const streamData = getStreamDataFromPage(); if (streamData?.status) { createUI(streamData); } } // Initialize URL change listener function initUrlChangeListener() { onUrlChange(handleUrlChange); } // Handle URL changes async function handleUrlChange(urlData) { const isLivePage = ['root/live', 'follow/live'].some(path => urlData.url.includes(`www.douyin.com/${path}`)); if (!isLivePage) { currentRid = null; const selectors = ['douyin-stream-url-app', '.douyin-stream-url-side-button']; selectors.forEach(selector => { const element = selector.startsWith('.') ? document.querySelector(selector) : document.getElementById(selector); element?.remove(); }); return; } const rid = extractRoomId(urlData.url); if (!rid) { console.warn('Failed to extract room ID'); return; } if (rid === currentRid) { console.warn(`Room ID unchanged: ${rid}`); return; } currentRid = rid; try { console.log(`Getting stream data from API, room ID: ${rid}`); await getStreamDataFromApi(rid); } catch (err) { console.error('Failed to get stream data:', err); } } // Extract room ID from URL function extractRoomId(url) { const match = url.match(/\/(\d+)(?:\/|\?|$)/); return match ? match[1] : null; } // URL change listener function onUrlChange(callback) { window.addEventListener('popstate', () => callback({ type: 'popstate', url: window.location.href, timestamp: Date.now() })); const originalPushState = history.pushState; history.pushState = function (...args) { originalPushState.apply(this, args); callback({ type: 'pushState', url: window.location.href, timestamp: Date.now() }); }; const originalReplaceState = history.replaceState; history.replaceState = function (...args) { originalReplaceState.apply(this, args); callback({ type: 'replaceState', url: window.location.href, timestamp: Date.now() }); }; window.addEventListener('hashchange', () => callback({ type: 'hashchange', url: window.location.href, timestamp: Date.now() })); } // Extract JSON from page function extractJSON(pattern, page) { const pageHTML = page || document.documentElement.outerHTML; const match = pageHTML?.match(pattern); return match ? match[1].replace(/\\/g, '').replace(/u0026/g, '&') : null; } // Get stream data from page function getStreamDataFromPage(pageHTML) { try { const jsonStr = extractJSON(/(\{\\"state\\":.*?)]\\n"]\)/, pageHTML) || extractJSON(/(\{\\"common\\":.*?)]\\n"]\)<\/script>