在您安装前,Sleazy Fork 希望您知道此脚本声明其包含了一些负面功能。这些功能也许会使脚本作者获利,而不能给您带来任何直接的金钱收益。
您只有在付费后才能使用脚本的全部功能。Sleazy Fork 未参与到支付的流程,因此无法验证您是否获得了有价值的东西,亦无法帮助您申请退款。
无限制解锁收费资源,包括钻石贴、金币贴,下载视频,复制视频链接,自动展开帖子
// ==UserScript== // @name 海角天涯-解锁海角社区资源 // @namespace tianya365.top // @homepage https://vip.tianya365.top // @version 2.1.0 // @description 无限制解锁收费资源,包括钻石贴、金币贴,下载视频,复制视频链接,自动展开帖子 // @icon  // @author tianya365.top // @exclude https://*.tianya365.top/* // @include https://*.*/* // @connect * // @run-at document-start // @grant none // @antifeature payment // @charset UTF-8 // ==/UserScript== (function () { 'use strict'; const CONFIG = { VERSION: '2.1.0', DEFAULT_SERVER: 'https://vip.tianya365.top', STORAGE_KEYS: { SERVER: 'hjty_server', USER_TOKEN: 'hjty_user_token', USER_NAME: 'hjty_user_name' }, SELECTORS: { TOOLBAR_CONTAINER: '.hjty-toolbar-container', UNLOCK_BTN: '.hjty-unlock-btn', SETTINGS_BTN: '.hjty-settings-btn', USER_BTN: '.hjty-user-btn', TOGGLE_BTN: '.hjty-toggle-btn' }, MODAL_IDS: { LOADING: 'hjty-loading-modal', LOGIN: 'hjty-login-modal', SERVER: 'hjty-server-modal', USER_INFO: 'hjty-user-info-modal', LOGOUT_CONFIRM: 'hjty-logout-confirm-modal', UPDATE_CONFIRM: 'hjty-update-confirm-modal' } }; let topic = null; let newVersion = ''; let m3u8_url = ''; let play_url = ''; let shortVideoUrl = ''; let freeToken = window.localStorage.getItem('hjty_free_token'); let video = null; let hasPic = false; let server = localStorage.getItem(CONFIG.STORAGE_KEYS.SERVER) || CONFIG.DEFAULT_SERVER; let toolbarCollapsed = localStorage.getItem('hjty_toolbar_collapsed') === 'true'; if (!localStorage.getItem(CONFIG.STORAGE_KEYS.SERVER)) { localStorage.setItem(CONFIG.STORAGE_KEYS.SERVER, server); } if (freeToken == null) { freeToken = '7a78a8bf4bd2432f93a301bec4a17cf7' window.localStorage.setItem('hjty_free_token', freeToken); } const Utils = { debounce(func, wait, immediate = false) { let timeout; return function executedFunction(...args) { const later = () => { timeout = null; if (!immediate) func.apply(this, args); }; const callNow = immediate && !timeout; clearTimeout(timeout); timeout = setTimeout(later, wait); if (callNow) func.apply(this, args); }; }, base64: { encode: function (str) { try { return btoa(encodeURIComponent(str).replace(/%([0-9A-F]{2})/g, function (match, p1) { return String.fromCharCode(parseInt(p1, 16)); })); } catch (error) { return ''; } }, decode: function (base64Str) { try { return decodeURIComponent(Array.prototype.map.call(atob(base64Str), function (c) { return '%' + ('00' + c.charCodeAt(0).toString(16)).slice(-2); }).join('')); } catch (error) { return ''; } } }, getUserToken() { return localStorage.getItem(CONFIG.STORAGE_KEYS.USER_TOKEN); }, getCookieValue(name) { const match = document.cookie.match(new RegExp(name + '=([^;]+)')); return match ? match[1] : null; }, checkLoginStatus() { return !!this.getUserToken(); }, getPid() { if (/\/post\/details.*pid=\d+/.test(location.href)) { const match = location.href.match(/\/post\/details.*pid=(\d+)/i); return match ? match[1] : null; } return null; }, deleteCookie(name) { document.cookie = `${name}=; expires=Thu, 01 Jan 1970 00:00:00 GMT; path=/`; }, decrypt(data) { return JSON.parse(this.base64.decode(this.base64.decode(this.base64.decode(data)))); }, encrypt(data) { return this.base64.encode(this.base64.encode(this.base64.encode(JSON.stringify(data)))) }, async copyText(text) { try { if (navigator.clipboard && window.isSecureContext) { await navigator.clipboard.writeText(text); return true; } else { const textArea = document.createElement('textarea'); textArea.value = text; textArea.style.position = 'fixed'; textArea.style.opacity = '0'; document.body.appendChild(textArea); textArea.focus(); textArea.select(); const success = document.execCommand('copy'); document.body.removeChild(textArea); return success; } } catch (error) { return false; } }, smoothScrollTo(element, offset = 0) { if (!element) return; if (this.isMobile()) { this.mobileScrollTo(element, offset); } else { this.tryMultipleScrollMethods(element, offset); } }, isMobile() { return /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent) || window.innerWidth <= 768; }, mobileScrollTo(element, offset = 0) { try { element.scrollIntoView({ behavior: 'smooth', block: 'start', inline: 'nearest' }); if (offset !== 0) { setTimeout(() => { const currentPos = this.getScrollTop(); const targetPos = currentPos + offset; window.scrollTo({ top: targetPos, behavior: 'smooth' }); }, 300); } } catch (error) { const targetPosition = element.getBoundingClientRect().top + this.getScrollTop() + offset; this.animateScrollTo(targetPosition); } }, tryMultipleScrollMethods(element, offset = 0) { const targetPosition = element.getBoundingClientRect().top + this.getScrollTop() + offset; if (this.tryNativeScroll(targetPosition)) { return; } if (this.tryScrollContainer(targetPosition)) { return; } this.animateScrollTo(targetPosition); }, getScrollTop() { return window.pageYOffset || document.documentElement.scrollTop || document.body.scrollTop || 0; }, tryNativeScroll(targetPosition) { try { window.scrollTo({ top: targetPosition, behavior: 'smooth' }); return new Promise(resolve => { const startPos = this.getScrollTop(); setTimeout(() => { const currentPos = this.getScrollTop(); resolve(Math.abs(currentPos - startPos) > 5); }, 100); }); } catch (error) { return false; } }, tryScrollContainer(targetPosition) { const containerSelectors = [ 'body', 'html', '.app', '.main', '.container', '#app', '[data-scroll]', '.scroll-container' ]; for (const selector of containerSelectors) { const container = document.querySelector(selector); if (container && this.tryScrollElement(container, targetPosition)) { return true; } } return false; }, tryScrollElement(element, targetPosition) { try { if (element.scrollTo) { element.scrollTo({ top: targetPosition, behavior: 'smooth' }); return true; } else if (element.scrollTop !== undefined) { this.animateElementScroll(element, targetPosition); return true; } } catch (error) { } return false; }, animateElementScroll(element, targetPosition, duration = 500) { const startPosition = element.scrollTop || 0; const distance = targetPosition - startPosition; let startTime = null; function animation(currentTime) { if (startTime === null) startTime = currentTime; const timeElapsed = currentTime - startTime; const progress = Math.min(timeElapsed / duration, 1); const ease = progress < 0.5 ? 2 * progress * progress : 1 - Math.pow(-2 * progress + 2, 2) / 2; element.scrollTop = startPosition + distance * ease; if (timeElapsed < duration) { requestAnimationFrame(animation); } } requestAnimationFrame(animation); }, animateScrollTo(targetPosition, duration = 500) { const startPosition = this.getScrollTop(); const distance = targetPosition - startPosition; let startTime = null; const animation = (currentTime) => { if (startTime === null) startTime = currentTime; const timeElapsed = currentTime - startTime; const progress = Math.min(timeElapsed / duration, 1); const ease = progress < 0.5 ? 2 * progress * progress : 1 - Math.pow(-2 * progress + 2, 2) / 2; const currentPosition = startPosition + distance * ease; try { window.scrollTo(0, currentPosition); } catch (e) { document.documentElement.scrollTop = currentPosition; document.body.scrollTop = currentPosition; } if (timeElapsed < duration) { requestAnimationFrame(animation); } }; requestAnimationFrame(animation); }, imageDecode(t) { const e = "ABCD*EFGHIJKLMNOPQRSTUVWX#YZabcdefghijklmnopqrstuvwxyz1234567890"; let o, i, a, r, s, c, u, l = "", d = 0; function n(e) { let t, n = "", o = 0, i = 0, a = 0; while (o < e.length) i = e.charCodeAt(o), i < 128 ? (n += String.fromCharCode(i), o++) : i > 191 && i < 224 ? (a = e.charCodeAt(o + 1), n += String.fromCharCode((31 & i) << 6 | 63 & a), o += 2) : (a = e.charCodeAt(o + 1), t = e.charCodeAt(o + 2), n += String.fromCharCode((15 & i) << 12 | (63 & a) << 6 | 63 & t), o += 3); return n } t = t.replace(/[^A-Za-z0-9*#]/g, ""); while (d < t.length) r = e.indexOf(t.charAt(d++)), s = e.indexOf(t.charAt(d++)), c = e.indexOf(t.charAt(d++)), u = e.indexOf(t.charAt(d++)), o = r << 2 | s >> 4, i = (15 & s) << 4 | c >> 2, a = (3 & c) << 6 | u, l += String.fromCharCode(o), 64 !== c && (l += String.fromCharCode(i)), 64 !== u && (l += String.fromCharCode(a)); let reg = new RegExp('', "g"); l = l.replace(reg, ''); return l = n(l), l } }; const UI = { showToast(message, type = 'info') { const existingToast = document.getElementById('hjty-toast'); if (existingToast) { existingToast.remove(); } const icons = { success: `<svg width="20" height="20" viewBox="0 0 24 24" fill="none"> <path d="M9 12L11 14L15 10" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/> <circle cx="12" cy="12" r="10" stroke="currentColor" stroke-width="2"/> </svg>`, error: `<svg width="20" height="20" viewBox="0 0 24 24" fill="none"> <circle cx="12" cy="12" r="10" stroke="currentColor" stroke-width="2"/> <line x1="15" y1="9" x2="9" y2="15" stroke="currentColor" stroke-width="2"/> <line x1="9" y1="9" x2="15" y2="15" stroke="currentColor" stroke-width="2"/> </svg>`, warning: `<svg width="20" height="20" viewBox="0 0 24 24" fill="none"> <path d="M10.29 3.86L1.82 18a2 2 0 0 0 1.71 3h16.94a2 2 0 0 0 1.71-3L13.71 3.86a2 2 0 0 0-3.42 0z" stroke="currentColor" stroke-width="2"/> <line x1="12" y1="9" x2="12" y2="13" stroke="currentColor" stroke-width="2"/> <line x1="12" y1="17" x2="12.01" y2="17" stroke="currentColor" stroke-width="2"/> </svg>`, info: `<svg width="20" height="20" viewBox="0 0 24 24" fill="none"> <circle cx="12" cy="12" r="10" stroke="currentColor" stroke-width="2"/> <path d="M12 16V12" stroke="currentColor" stroke-width="2" stroke-linecap="round"/> <path d="M12 8H12.01" stroke="currentColor" stroke-width="2" stroke-linecap="round"/> </svg>` }; const toast = document.createElement('div'); toast.id = 'hjty-toast'; toast.className = `hjty-toast hjty-toast-${type}`; toast.innerHTML = ` <div class="hjty-toast-icon">${icons[type] || icons.info}</div> <div class="hjty-toast-message">${message}</div> `; document.body.appendChild(toast); setTimeout(() => toast.classList.add('hjty-toast-show'), 10); setTimeout(() => { toast.classList.remove('hjty-toast-show'); setTimeout(() => toast.remove(), 300); }, 3000); }, showLoadingModal(message = '加载中...', showCloseBtn = true, onClose = null) { const existingModal = document.getElementById(CONFIG.MODAL_IDS.LOADING); if (existingModal) { existingModal.remove(); } const loadingModal = document.createElement('div'); loadingModal.id = CONFIG.MODAL_IDS.LOADING; loadingModal.className = 'hjty-modal-overlay hjty-loading-overlay'; if (onClose && typeof onClose === 'function') { loadingModal._onCloseCallback = onClose; } const closeButtonHtml = showCloseBtn ? '<button class="hjty-modal-close hjty-loading-close" title="关闭">×</button>' : ''; loadingModal.innerHTML = ` <div class="hjty-modal-content hjty-loading-content"> <div class="hjty-loading-header">${closeButtonHtml}</div> <div class="hjty-loading-body"> <div class="hjty-loading-spinner"> <div class="hjty-spinner-ring"></div> </div> <p class="hjty-loading-message">${message}</p> </div> </div> `; document.body.appendChild(loadingModal); if (showCloseBtn) { const closeBtn = loadingModal.querySelector('.hjty-loading-close'); closeBtn.addEventListener('click', () => this.hideLoadingModal()); } loadingModal.addEventListener('click', (e) => { if (e.target === loadingModal) { e.preventDefault(); e.stopPropagation(); } }); return loadingModal; }, hideLoadingModal(executeCallback = true) { const loadingModal = document.getElementById(CONFIG.MODAL_IDS.LOADING); if (loadingModal) { if (executeCallback && loadingModal._onCloseCallback) { try { loadingModal._onCloseCallback(); } catch (error) { } } loadingModal.remove(); } }, showActions(title = '操作选项', actions = [], onAction = null) { const existingDrawer = document.getElementById('hjty-action-drawer'); if (existingDrawer) { existingDrawer.remove(); } const drawer = document.createElement('div'); drawer.id = 'hjty-action-drawer'; drawer.className = 'hjty-action-drawer-overlay'; const actionsHtml = actions.map(action => ` <div class="hjty-action-item" data-key="${action.key || ''}" data-url="${action.url || ''}"> <div class="hjty-action-content"> <div class="hjty-action-name">${action.name || ''}</div> ${action.subname ? `<div class="hjty-action-subname">${action.subname}</div>` : ''} </div> <div class="hjty-action-arrow"> <svg width="16" height="16" viewBox="0 0 24 24" fill="none"> <path d="M9 18L15 12L9 6" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/> </svg> </div> </div> `).join(''); drawer.innerHTML = ` <div class="hjty-action-drawer"> <div class="hjty-action-header"> <h3 class="hjty-action-title">${title}</h3> </div> <div class="hjty-action-list"> ${actionsHtml} </div> <div class="hjty-action-footer"> <button class="hjty-action-close-btn">关闭</button> </div> </div> `; document.body.appendChild(drawer); const actionItems = drawer.querySelectorAll('.hjty-action-item'); const closeBtn = drawer.querySelector('.hjty-action-close-btn'); actionItems.forEach(item => { item.addEventListener('click', () => { if (onAction && typeof onAction === 'function') { onAction(item.dataset); } }); }); closeBtn.addEventListener('click', () => this.hideActions()); drawer.addEventListener('click', (e) => { if (e.target === drawer) this.hideActions(); }); setTimeout(() => drawer.classList.add('hjty-action-drawer-show'), 10); return drawer; }, hideActions() { const drawer = document.getElementById('hjty-action-drawer'); if (drawer) { drawer.classList.remove('hjty-action-drawer-show'); setTimeout(() => drawer.remove(), 300); } }, showPlayer(src = '') { const existingPlayer = document.getElementById('hjty-player-modal'); if (existingPlayer) { existingPlayer.remove(); } const playerModal = document.createElement('div'); playerModal.id = 'hjty-player-modal'; playerModal.className = 'hjty-player-modal-overlay'; playerModal.innerHTML = ` <div class="hjty-player-modal-content"> <div class="hjty-player-header"> <button class="hjty-player-close" title="关闭播放器">×</button> </div> <div class="hjty-player-body"> <div id="hjty-play-box"> <iframe style="display:block;border:0;width: 100%;height: 500px" allow="autoplay; encrypted-media" allowfullscreen src="${src}"></iframe> </div> </div> </div> `; document.body.appendChild(playerModal); const closeBtn = playerModal.querySelector('.hjty-player-close'); closeBtn.addEventListener('click', () => this.hidePlayer()); playerModal.addEventListener('click', (e) => { if (e.target === playerModal) this.hidePlayer(); }); const handleEscKey = (e) => { if (e.key === 'Escape') { this.hidePlayer(); document.removeEventListener('keydown', handleEscKey); } }; document.addEventListener('keydown', handleEscKey); setTimeout(() => playerModal.classList.add('hjty-player-modal-show'), 10); return playerModal; }, hidePlayer() { const playerModal = document.getElementById('hjty-player-modal'); if (playerModal) { playerModal.classList.remove('hjty-player-modal-show'); setTimeout(() => playerModal.remove(), 300); } } }; const App = { getTopic(pid) { return new Promise(async (resolve, reject) => { try { const response = await fetch(`/api/topic/${pid}`, { method: 'GET', headers: { 'Content-Type': 'application/json' } }); if (!response.ok) { throw new Error(`HTTP ${response.status}`); } const res = await response.json(); if (res.isEncrypted) { topic = Utils.decrypt(res.data) } else { topic = res.data; } resolve(topic); } catch (error) { reject('网络错误: 3'); } }); }, showTopic(data, pid) { if (!data['attachments']) { UI.showToast('没有可解锁的内容', 'warning'); return; } this.getAttachments(data).then(html => { if (html) { const sellBtn = document.querySelector('.sell-btn'); const hjsellContainer = document.getElementById('hjsell-container'); if (sellBtn) { const newContainer = document.createElement('div'); newContainer.id = 'hjsell-container'; newContainer.className = 'hjsell-container'; newContainer.innerHTML = '<div class="hssell-content">' + html + '</div>'; sellBtn.parentNode.replaceChild(newContainer, sellBtn); } else if (hjsellContainer) { const contentElement = hjsellContainer.querySelector('.hssell-content'); if (contentElement) { contentElement.innerHTML = html; } } const finalContainer = document.getElementById('hjsell-container'); if (!finalContainer) { UI.showToast('请等待页面加载完成!', 'error'); return; } Utils.smoothScrollTo(finalContainer, -100); if (hasPic) { this.showImages(); } } if (video !== null) { if (!data['sale'] || data['sale']['money_type'] === 0 || data['sale']['is_buy']) { this.getVideo(video, pid); } else { this.queryTopic(pid); } } else { UI.showToast('解锁成功', 'success'); UI.hideLoadingModal(false) } }) }, queryTopic(pid) { fetch(server + '/api/topic/' + pid + '?v=' + CONFIG.VERSION, { method: 'GET', headers: { 'token': Utils.getUserToken(), 'Content-Type': 'application/json' } }) .then(response => { if (!response.ok) { throw new Error(`HTTP ${response.status}`); } return response.json(); }) .then(data => { UI.hideLoadingModal(false) if (data.code) { if (data.msg === '有新版本') { newVersion = data.data; this.showUpdateConfirm(); return; } if (data.data['m3u8_url'] !== '') { m3u8_url = server + data.data['m3u8_url']; play_url = server + '/play?url=' + Utils.base64.encode(m3u8_url); this.showVideoActions(); } else { UI.showToast('没有可解锁的内容', 'error'); } } else { UI.showToast(data.msg, 'error'); if (data.data['confirmButtonText'] === '重新登录') { localStorage.removeItem(CONFIG.STORAGE_KEYS.USER_TOKEN); localStorage.removeItem(CONFIG.STORAGE_KEYS.USER_NAME); this.updateToolbarButtons(false); showLoginModal(); } else if (data.data['confirmButtonText'] === '打开官网') { setTimeout(() => { window.open(server + '/'); }, 1500) } } }) .catch(error => { }); }, getVideo(item, pid) { if (item.remoteUrl.includes('haijiao.live')) { m3u8_url = item.remoteUrl; play_url = server + '/play?url=' + Utils.base64.encode(item.remoteUrl); this.showVideoActions() UI.hideLoadingModal(false) } else { const uid = Utils.getCookieValue('uid'); const token = Utils.getCookieValue('token'); const xhr = new XMLHttpRequest(); xhr.open('POST', '/api/attachment', true); xhr.setRequestHeader('Content-Type', 'application/json;charset=UTF-8'); xhr.setRequestHeader('Mver', '211112203214'); xhr.setRequestHeader('X-User-Id', uid && token ? uid : '172561377002'); xhr.setRequestHeader('X-User-Token', uid && token ? token : freeToken); xhr.onreadystatechange = function () { if (xhr.readyState === 4) { if (xhr.status === 200) { try { const res = JSON.parse(xhr.responseText); let data = {}; if (res.isEncrypted) { data = Utils.decrypt(res.data) } else { data = res.data; } if (!data.remoteUrl.startsWith('http')) { m3u8_url = play_url = window.location.origin + data.remoteUrl; } else { m3u8_url = play_url = data.remoteUrl; } play_url = server + '/play?url=' + btoa(play_url); App.showVideoActions(); UI.hideLoadingModal(false); } catch (error) { const accXhr = new XMLHttpRequest(); accXhr.open('GET', server + '/acc', true); accXhr.onreadystatechange = function () { if (accXhr.readyState === 4) { if (accXhr.status === 200) { try { const accRes = JSON.parse(accXhr.responseText); if (accRes.code) { freeToken = accRes.data; localStorage.setItem('hjty_free_token', freeToken); UI.showToast('网络错误: 请重试', 'error'); } else { UI.showToast('网络错误: 2', 'error'); } } catch (error) { UI.showToast('网络错误: 2', 'error'); } } else { UI.showToast('网络错误: 2', 'error'); } } UI.hideLoadingModal(false); }; accXhr.onerror = function () { UI.showToast('网络错误: 2', 'error'); UI.hideLoadingModal(false); }; accXhr.send(); } } else { UI.showToast('网络错误: 2', 'error'); UI.hideLoadingModal(false); } } }; xhr.onerror = function () { UI.showToast('网络连接失败', 'error'); }; xhr.send(JSON.stringify({ id: item.id, is_ios: 0, resource_id: parseInt(pid), resource_type: 'topic', line: 'normal1' })); } }, showImages() { document.querySelectorAll('#hjsell-container img').forEach(async img => { try { const response = await fetch(img.dataset.url); if (!response.ok) { throw new Error(`HTTP ${response.status}`); } const data = await response.text(); img.src = Utils.imageDecode(data); } catch (error) { } }); }, showVideoActions() { const actions = [ { name: '播放视频(站内)', key: 'play', subname: '如无法在线播放,可尝试其他在线播放器,或下载后再观看' }, { name: '复制视频地址', key: 'copy', subname: '可复制视频地址到其他地方播放或下载' }, { name: '在线播放器1', key: 'open', subname: '可正常播放', url: 'https://www.m3u8player.online/m3u8?url=' + encodeURIComponent(m3u8_url) }, { name: '在线播放器2', key: 'open', subname: '不支持播放收费视频', url: 'https://m3u8player.org/player.html?url=' + m3u8_url }, { name: '在线播放器3', key: 'open', subname: '不支持播放收费视频', url: 'https://m3u8play.dev/?url=' + m3u8_url }, { name: '视频下载通道1', key: 'open', subname: '【推荐】支持自动下载和在线播放', url: 'https://getm3u8.com/?source=' + m3u8_url + (m3u8_url.includes('?') ? '&' : '?') + 'title=' + topic.title }, { name: '视频下载通道2', key: 'open', subname: '推荐在电脑端操作', url: 'https://blog.luckly-mjw.cn/tool-show/m3u8-downloader/index.html?source=' + m3u8_url + (m3u8_url.includes('?') ? '&' : '?') + 'title=' + topic.title }, { name: '视频下载通道3', key: 'open', subname: '需要手动点击下载,推荐在电脑端操作', url: 'https://tools.thatwind.com/tool/m3u8downloader#m3u8=' + encodeURIComponent(m3u8_url) }, { name: '海角天涯官网', key: 'open', subname: '点击访问', url: server + '/' } ]; UI.showActions('帖子解锁成功,下滑查看更多操作', actions, (item) => { this.handleVideoAction(item) }); }, handleVideoAction(item) { switch (item.key) { case 'play': UI.showPlayer(play_url); UI.showToast('视频加载中,请稍后...', 'info'); break; case 'copy': Utils.copyText(m3u8_url).then(success => { if (success) { UI.showToast('视频地址已复制', 'success'); } else { UI.showToast('复制失败', 'error'); } }); break; case 'open': if (item.url) { UI.showToast('正在打开链接...', 'info'); setTimeout(() => { window.open(item.url, '_blank'); }, 500); } break; default: console.log('未知操作:', key); } }, getAttachments(data) { return new Promise((resolve, reject) => { let html = ''; if (!data || !data.attachments) { resolve(html); return; } data.attachments.forEach(item => { switch (item.category) { case 'video': video = item; break; case 'audio': html += '<p style="text-align: center"><audio src="' + item.remoteUrl + '" controls="true" id="showaudio"></audio></p>'; break; case 'images': hasPic = true; if (!data.content.includes(item.remoteUrl)) { html += '<p><img data-url="' + item.remoteUrl + '" src="/images/common/project/loading.gif" alt="" data-id="' + item.id + '" lazy="loaded"></p>'; } break; } }); resolve(html); }); }, async checkUpdate() { const updateUrl = `${server}/api/update?v=${CONFIG.VERSION}`; const token = Utils.getUserToken(); try { const response = await fetch(updateUrl, { method: 'GET', headers: { 'token': token, 'Content-Type': 'application/json' } }); const data = await response.json(); if (data.msg === '有新版本') { newVersion = data.data; this.showUpdateConfirm(); } } catch (error) { UI.showToast('检查更新失败,请切换其他服务器', 'error'); } }, showUpdateConfirm() { const updateModal = document.createElement('div'); updateModal.id = CONFIG.MODAL_IDS.UPDATE_CONFIRM; updateModal.className = 'hjty-modal-overlay'; updateModal.innerHTML = ` <div class="hjty-modal-content hjty-update-content"> <div class="hjty-modal-header"> <h3>脚本升级</h3> <button class="hjty-modal-close" onclick="this.closest('.hjty-modal-overlay').remove()">×</button> </div> <div class="hjty-modal-body"> <div class="hjty-update-info"> <div class="hjty-update-icon"> <svg width="48" height="48" viewBox="0 0 24 24" fill="none"> <circle cx="12" cy="12" r="10" stroke="#3b82f6" stroke-width="2"/> <path d="M8 12L12 8L16 12" stroke="#3b82f6" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/> <path d="M12 8V16" stroke="#3b82f6" stroke-width="2" stroke-linecap="round"/> <path d="M16 16H8" stroke="#60a5fa" stroke-width="2" stroke-linecap="round"/> </svg> </div> <p class="hjty-update-message">检测到新版本,请更新后再使用!</p> <p class="hjty-update-details">当前版本: ${CONFIG.VERSION}</p> ${newVersion ? `<p class="hjty-update-details">最新版本: ${newVersion}</p>` : ''} </div> </div> <div class="hjty-modal-footer"> <button class="hjty-btn hjty-btn-primary hjty-confirm-update-btn">立即更新</button> </div> </div> `; document.body.appendChild(updateModal); const confirmBtn = updateModal.querySelector('.hjty-confirm-update-btn'); confirmBtn.addEventListener('click', () => { updateModal.remove(); window.open(`${server}/js/script.user.js?v=${Math.random()}`, '_blank'); }); updateModal.addEventListener('click', (e) => { if (e.target === updateModal) updateModal.remove(); }); }, logout() { localStorage.removeItem(CONFIG.STORAGE_KEYS.USER_TOKEN); localStorage.removeItem(CONFIG.STORAGE_KEYS.USER_NAME); this.updateToolbarButtons(false); UI.showToast('已退出登录', 'info'); }, updateToolbarButtons(isLoggedIn) { const toolbarContainer = document.querySelector(CONFIG.SELECTORS.TOOLBAR_CONTAINER); if (!toolbarContainer) return; let userBtn = toolbarContainer.querySelector(CONFIG.SELECTORS.USER_BTN); if (isLoggedIn) { if (!userBtn) { userBtn = document.createElement('button'); userBtn.className = 'hjty-toolbar-btn hjty-user-btn'; userBtn.title = '用户信息'; userBtn.innerHTML = ` <svg width="16" height="16" viewBox="0 0 24 24" fill="none"> <path d="M20 21V19C20 17.9391 19.5786 16.9217 18.8284 16.1716C18.0783 15.4214 17.0609 15 16 15H8C6.93913 15 5.92172 15.4214 5.17157 16.1716C4.42143 16.9217 4 17.9391 4 19V21" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/> <circle cx="12" cy="7" r="4" stroke="currentColor" stroke-width="2"/> </svg> `; userBtn.addEventListener('click', () => this.showUserInfoModal()); const unlockBtn = toolbarContainer.querySelector(CONFIG.SELECTORS.UNLOCK_BTN); unlockBtn.insertAdjacentElement('afterend', userBtn); } } else { if (userBtn) userBtn.remove(); } }, showUserInfoModal() { const userName = localStorage.getItem(CONFIG.STORAGE_KEYS.USER_NAME) || '未知用户'; const userModal = document.createElement('div'); userModal.id = CONFIG.MODAL_IDS.USER_INFO; userModal.className = 'hjty-modal-overlay'; userModal.innerHTML = ` <div class="hjty-modal-content hjty-user-info-content"> <div class="hjty-modal-header"> <h3>用户信息</h3> <button class="hjty-modal-close" onclick="this.closest('.hjty-modal-overlay').remove()">×</button> </div> <div class="hjty-modal-body"> <div class="hjty-user-info"> <div class="hjty-user-avatar"> <svg width="64" height="64" viewBox="0 0 24 24" fill="none"> <circle cx="12" cy="12" r="10" fill="#8b5cf6" opacity="0.1"/> <path d="M18 19V17C18 15.3431 16.6569 14 15 14H9C7.34315 14 6 15.3431 6 17V19" stroke="#8b5cf6" stroke-width="1.8" stroke-linecap="round" stroke-linejoin="round"/> <circle cx="12" cy="8" r="3" stroke="#8b5cf6" stroke-width="1.8"/> </svg> </div> <div class="hjty-user-details"> <p class="hjty-user-name">${userName}</p> <p class="hjty-user-status">已登录</p> </div> </div> </div> <div class="hjty-modal-footer"> <button class="hjty-btn hjty-btn-secondary hjty-visit-website-btn">访问官网</button> <button class="hjty-btn hjty-btn-danger hjty-logout-btn">退出登录</button> </div> </div> `; document.body.appendChild(userModal); const visitWebsiteBtn = userModal.querySelector('.hjty-visit-website-btn'); const logoutBtn = userModal.querySelector('.hjty-logout-btn'); visitWebsiteBtn.addEventListener('click', () => window.open(server, '_blank')); logoutBtn.addEventListener('click', () => { userModal.remove(); this.showLogoutConfirm(); }); userModal.addEventListener('click', (e) => { if (e.target === userModal) userModal.remove(); }); }, showLogoutConfirm() { const confirmModal = document.createElement('div'); confirmModal.id = CONFIG.MODAL_IDS.LOGOUT_CONFIRM; confirmModal.className = 'hjty-modal-overlay'; confirmModal.innerHTML = ` <div class="hjty-modal-content hjty-confirm-content"> <div class="hjty-modal-header"> <h3>确认退出</h3> <button class="hjty-modal-close" onclick="this.closest('.hjty-modal-overlay').remove()">×</button> </div> <div class="hjty-modal-body"> <p class="hjty-confirm-message">确定要退出登录吗?</p> </div> <div class="hjty-modal-footer"> <button class="hjty-btn hjty-btn-secondary hjty-cancel-btn">取消</button> <button class="hjty-btn hjty-btn-primary hjty-confirm-btn">确认退出</button> </div> </div> `; document.body.appendChild(confirmModal); const cancelBtn = confirmModal.querySelector('.hjty-cancel-btn'); const confirmBtn = confirmModal.querySelector('.hjty-confirm-btn'); cancelBtn.addEventListener('click', () => confirmModal.remove()); confirmBtn.addEventListener('click', () => { confirmModal.remove(); this.logout(); }); confirmModal.addEventListener('click', (e) => { if (e.target === confirmModal) confirmModal.remove(); }); }, formatTitle(data) { data.results.forEach(item => { const tips = []; tips.push('money_type' in item ? ['', '💰', '💎'][item.money_type] : ''); item.hasVideo && tips.push('🎥'); item.hasAudio && tips.push('🎵'); item.title = `${tips.join(' ')} ${item.title}`; }); return data; }, getShortVideoUrl(responseText) { try { let res = JSON.parse(responseText); if (res['isEncrypted']) { let data = Utils.decrypt(res.data); if (data['category'] && data['category'] === 'video' && data['remoteUrl']) { if (!data['remoteUrl'].startsWith('http')) { shortVideoUrl = window.location.origin + data['remoteUrl']; } else { shortVideoUrl = data['remoteUrl']; } } } } catch (e) { } }, fixShortVideo(responseText) { try { responseText = JSON.parse(responseText); let video = Utils.decrypt(responseText.data); video.type = 1; video.amount = 0; video.money_type = 0; video.vip = 0; video.message = ''; responseText.data = Utils.encrypt(video); return JSON.stringify(responseText); } catch (e) { return responseText; } } }; let buttonInitialized = false; function checkTitleAndInitButton() { if (buttonInitialized) return true; const currentTitle = document.title; if (currentTitle === "海角社区") { console.log('海角社区助手已启动,版本:', CONFIG.VERSION); buttonInitialized = true; start(); return true; } return false; } function setupTitleObserver() { const titleObserver = new MutationObserver((mutations) => { if (!buttonInitialized && document.title === "海角社区") { buttonInitialized = true; console.log('海角社区助手已启动,版本:', CONFIG.VERSION); start(); titleObserver.disconnect(); } }); const titleElement = document.querySelector('title'); if (titleElement) { titleObserver.observe(titleElement, { childList: true, characterData: true, subtree: true }); } let checkCount = 0; const maxChecks = 10; const checkInterval = setInterval(() => { if (buttonInitialized || checkCount >= maxChecks) { clearInterval(checkInterval); return; } if (checkTitleAndInitButton()) { buttonInitialized = true; clearInterval(checkInterval); } checkCount++; }, 1000); } function start() { initStyles(); initFloatingButton(); if (Utils.checkLoginStatus()) { App.checkUpdate(); } } function initStyles() { if (document.getElementById('hjty-global-styles')) return; const style = document.createElement('style'); style.id = 'hjty-global-styles'; style.textContent = ` /* 浮动工具栏样式 */ #hjty-floating-toolbar { position: fixed; top: 50%; right: 20px; transform: translateY(-50%); z-index: 10000; font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif; transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1); } #hjty-floating-toolbar.hjty-collapsed { right: -48px; /* 隐藏主体,只露出切换按钮 */ } .hjty-toolbar-container { display: flex; flex-direction: column; gap: 8px; padding: 8px; background: rgba(255, 255, 255, 0.95); backdrop-filter: blur(10px); border-radius: 12px; box-shadow: 0 8px 32px rgba(0, 0, 0, 0.1); border: 1px solid rgba(255, 255, 255, 0.2); position: relative; } /* 切换按钮样式 */ .hjty-toggle-btn { position: absolute; left: -20px; top: 50%; transform: translateY(-50%); width: 24px; height: 48px; border: none; border-radius: 12px 0 0 12px; background: rgba(255, 255, 255, 0.95); backdrop-filter: blur(10px); color: #374151; cursor: pointer; display: flex; align-items: center; justify-content: center; transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1); z-index: 1; } .hjty-toggle-btn:hover { background: rgba(255, 255, 255, 1); transform: translateY(-50%) scale(1.05); color: #1f2937; } .hjty-toggle-btn:active { transform: translateY(-50%) scale(0.95); } .hjty-toggle-btn svg { transition: transform 0.3s ease; } #hjty-floating-toolbar.hjty-collapsed .hjty-toggle-btn svg { transform: rotate(180deg); } .hjty-toolbar-btn { width: 32px; height: 32px; border: none; border-radius: 8px; background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); color: white; cursor: pointer; display: flex; align-items: center; justify-content: center; transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1); position: relative; overflow: hidden; } .hjty-toolbar-btn:hover { box-shadow: 0 8px 25px rgba(102, 126, 234, 0.4); } .hjty-toolbar-btn:active { transform: translateY(0); box-shadow: 0 4px 15px rgba(102, 126, 234, 0.3); } .hjty-unlock-btn { background: linear-gradient(135deg, #11998e 0%, #38ef7d 100%); } .hjty-unlock-btn:hover { box-shadow: 0 8px 25px rgba(17, 153, 142, 0.4); } .hjty-settings-btn { background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); } .hjty-settings-btn:hover { box-shadow: 0 8px 25px rgba(102, 126, 234, 0.4); } .hjty-user-btn { background: linear-gradient(135deg, #8b5cf6 0%, #7c3aed 100%); } .hjty-user-btn:hover { box-shadow: 0 8px 25px rgba(139, 92, 246, 0.4); } .hjty-toolbar-btn svg { transition: transform 0.3s ease; } .hjty-toolbar-btn:hover svg { transform: scale(1.1); } /* Modal 通用样式 */ .hjty-modal-overlay { position: fixed; top: 0; left: 0; width: 100%; height: 100%; background: rgba(0, 0, 0, 0.5); backdrop-filter: blur(5px); display: flex; align-items: center; justify-content: center; animation: hjty-fadeIn 0.3s ease; z-index: 20000; } .hjty-modal-content { background: white; border-radius: 16px; box-shadow: 0 20px 60px rgba(0, 0, 0, 0.2); width: 90%; max-width: 600px; max-height: 80vh; overflow: hidden; animation: hjty-slideUp 0.3s ease; } .hjty-modal-header { display: flex; justify-content: space-between; align-items: center; padding: 20px 24px; border-bottom: 1px solid #e5e7eb; } .hjty-modal-header h3 { margin: 0; font-size: 18px; font-weight: 600; color: #1f2937; } .hjty-modal-close { background: none; border: none; font-size: 24px; color: #6b7280; cursor: pointer; padding: 0; width: 32px; height: 32px; display: flex; align-items: center; justify-content: center; border-radius: 8px; transition: all 0.2s ease; } .hjty-modal-close:hover { background: #f3f4f6; color: #374151; } .hjty-modal-body { padding: 16px 24px; max-height: 350px; overflow-y: auto; } .hjty-modal-footer { padding: 16px 24px; border-top: 1px solid #e5e7eb; display: flex; justify-content: space-between; gap: 12px; } /* 登录Modal样式 */ .hjty-login-content { max-width: 400px; width: 90%; } .hjty-login-content .hjty-modal-body { padding: 24px; } .hjty-login-form { padding: 0; } .hjty-form-group { margin-bottom: 20px; } .hjty-form-group:last-of-type { margin-bottom: 0; } .hjty-form-group label { display: block; margin-bottom: 8px; font-size: 14px; font-weight: 500; color: #374151; } .hjty-form-input { display: block; width: 100%; padding: 12px 16px; border: 1px solid #d1d5db; border-radius: 8px; font-size: 14px; transition: all 0.2s ease; background: white; box-sizing: border-box; } .hjty-form-input:focus { outline: none; border-color: #3b82f6; box-shadow: 0 0 0 3px rgba(59, 130, 246, 0.1); } .hjty-form-input::placeholder { color: #9ca3af; } .hjty-form-actions { display: flex; gap: 12px; margin-top: 32px; } /* 按钮样式 */ .hjty-btn { flex: 1; padding: 12px 20px; border: none; border-radius: 8px; font-size: 14px; font-weight: 500; cursor: pointer; transition: all 0.2s ease; display: flex; align-items: center; justify-content: center; gap: 8px; } .hjty-btn-primary { background: linear-gradient(135deg, #3b82f6 0%, #1d4ed8 100%); color: white; } .hjty-btn-primary:hover { background: linear-gradient(135deg, #2563eb 0%, #1e40af 100%); box-shadow: 0 4px 12px rgba(59, 130, 246, 0.3); } .hjty-btn-primary:disabled { background: #9ca3af; cursor: not-allowed; transform: none; box-shadow: none; min-height: 44px; } .hjty-btn-secondary { background: white; color: #374151; border: 1px solid #d1d5db; } .hjty-btn-secondary:hover { background: #f9fafb; border-color: #9ca3af; } .hjty-btn-danger { background: linear-gradient(135deg, #ef4444 0%, #dc2626 100%); color: white; } .hjty-btn-danger:hover { background: linear-gradient(135deg, #dc2626 0%, #b91c1c 100%); box-shadow: 0 4px 12px rgba(239, 68, 68, 0.3); } /* 按钮内加载spinner样式 */ .hjty-btn .hjty-loading-spinner { display: inline-flex; align-items: center; justify-content: center; width: 16px; height: 16px; margin: 0; flex-shrink: 0; } .hjty-btn .hjty-spinner-ring { width: 14px; height: 14px; border: 2px solid rgba(255, 255, 255, 0.3); border-top: 2px solid white; border-radius: 50%; animation: hjty-spin 1s linear infinite; } /* 加载框样式 */ #hjty-loading-modal { z-index: 40000; } .hjty-loading-overlay { background: rgba(0, 0, 0, 0.6); backdrop-filter: blur(8px); } .hjty-loading-content { max-width: 320px; width: 90%; } .hjty-loading-header { position: relative; height: 20px; padding: 16px 16px 0 16px; } .hjty-loading-close { position: absolute; top: 16px; right: 16px; width: 24px; height: 24px; font-size: 20px; border-radius: 6px; } .hjty-loading-body { padding: 20px 24px 32px 24px; text-align: center; } .hjty-loading-spinner { margin: 0 auto 20px auto; width: 40px; height: 40px; display: flex; align-items: center; justify-content: center; } .hjty-spinner-ring { width: 32px; height: 32px; border: 3px solid #e5e7eb; border-top: 3px solid #3b82f6; border-radius: 50%; animation: hjty-spin 1s linear infinite; } .hjty-loading-message { margin: 0; font-size: 16px; color: #374151; font-weight: 500; line-height: 1.5; } /* 用户信息对话框样式 */ .hjty-user-info-content { max-width: 400px; width: 90%; } .hjty-user-info { display: flex; align-items: center; gap: 16px; padding: 20px 0; } .hjty-user-avatar { flex-shrink: 0; } .hjty-user-details { flex: 1; } .hjty-user-name { margin: 0 0 8px 0; font-size: 18px; font-weight: 600; color: #1f2937; } .hjty-user-status { margin: 0; font-size: 14px; color: #10b981; font-weight: 500; } /* 确认对话框样式 */ .hjty-confirm-content { max-width: 400px; width: 90%; } .hjty-confirm-message { margin: 0; font-size: 16px; color: #374151; text-align: center; line-height: 1.5; } /* 服务器切换Modal样式 */ .hjty-server-list { display: flex; flex-direction: column; gap: 8px; } .hjty-server-item { display: flex; align-items: center; padding: 12px 16px; border: 1px solid #e5e7eb; border-radius: 8px; transition: all 0.2s ease; background: #fafafa; cursor: pointer; } .hjty-server-item:hover { border-color: #d1d5db; box-shadow: 0 2px 8px rgba(0, 0, 0, 0.08); background: #f9fafb; } .hjty-server-item.hjty-online { border-color: #10b981; background: #f0fdf4; } .hjty-server-item.hjty-offline { border-color: #ef4444; background: #fef2f2; } .hjty-server-item.hjty-selected { border-color: #3b82f6; background: #eff6ff; } .hjty-server-item:not(.hjty-selected):hover { border-color: #3b82f6; background: #f1f5f9; } .hjty-server-info { flex: 1; min-width: 0; } .hjty-server-name { font-weight: 500; color: #1f2937; font-size: 14px; position: relative; } .hjty-server-item.hjty-selected .hjty-server-name { color: #1e40af; font-weight: 600; } .hjty-radio-indicator { width: 16px; height: 16px; border: 2px solid #d1d5db; border-radius: 50%; margin-right: 12px; flex-shrink: 0; position: relative; transition: all 0.2s ease; } .hjty-radio-indicator.hjty-radio-selected { border-color: #3b82f6; background: #3b82f6; } .hjty-radio-indicator.hjty-radio-selected::after { content: ''; position: absolute; top: 50%; left: 50%; transform: translate(-50%, -50%); width: 6px; height: 6px; background: white; border-radius: 50%; } .hjty-server-status { display: flex; align-items: center; gap: 6px; margin: 0 12px; flex-shrink: 0; } .hjty-status-indicator { width: 8px; height: 8px; border-radius: 50%; position: relative; flex-shrink: 0; } .hjty-status-indicator.hjty-checking { background: #f59e0b; } .hjty-status-indicator.hjty-online { background: #10b981; } .hjty-status-indicator.hjty-offline { background: #ef4444; } .hjty-status-text { font-size: 11px; color: #374151; min-width: 40px; font-weight: 600; } .hjty-latency { font-size: 11px; color: #10b981; font-weight: 600; min-width: 35px; } .hjty-refresh-btn { padding: 10px 20px; border: none; border-radius: 8px; background: linear-gradient(135deg, #3b82f6 0%, #1d4ed8 100%); color: white; font-size: 14px; font-weight: 500; cursor: pointer; transition: all 0.2s ease; } .hjty-refresh-btn:hover:not(:disabled) { background: linear-gradient(135deg, #2563eb 0%, #1e40af 100%); box-shadow: 0 4px 12px rgba(59, 130, 246, 0.3); } .hjty-refresh-btn:disabled { background: #9ca3af; cursor: not-allowed; transform: none; box-shadow: none; } .hjty-switch-btn { padding: 10px 20px; border: none; border-radius: 8px; background: linear-gradient(135deg, #3b82f6 0%, #1d4ed8 100%); color: white; font-size: 14px; font-weight: 500; cursor: pointer; transition: all 0.2s ease; } .hjty-switch-btn:hover:not(:disabled) { background: linear-gradient(135deg, #2563eb 0%, #1e40af 100%); box-shadow: 0 4px 12px rgba(59, 130, 246, 0.3); } .hjty-switch-btn:disabled { background: #9ca3af; cursor: not-allowed; transform: none; box-shadow: none; } /* 升级确认框样式 */ .hjty-update-content { max-width: 450px; width: 90%; } .hjty-update-info { text-align: center; padding: 20px 0; } .hjty-update-icon { margin-bottom: 16px; display: flex; justify-content: center; } .hjty-update-message { margin: 0 0 16px 0; font-size: 18px; font-weight: 600; color: #1f2937; } .hjty-update-details { margin: 8px 0; font-size: 14px; color: #6b7280; } .hjty-update-description { margin: 16px 0 0 0; font-size: 14px; color: #374151; background: #f8fafc; padding: 12px; border-radius: 8px; border-left: 4px solid #3b82f6; text-align: left; } /* Toast 提示样式 */ .hjty-toast { position: fixed; top: 20px; right: 20px; display: flex; align-items: center; gap: 12px; padding: 16px 20px; border-radius: 8px; box-shadow: 0 4px 20px rgba(0, 0, 0, 0.15); backdrop-filter: blur(10px); z-index: 100000; font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif; font-size: 14px; font-weight: 500; max-width: 400px; min-width: 300px; transform: translateX(100%); opacity: 0; transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1); } .hjty-toast-show { transform: translateX(0); opacity: 1; } .hjty-toast-success { background: rgba(16, 185, 129, 0.95); color: white; border: 1px solid rgba(16, 185, 129, 0.3); } .hjty-toast-error { background: rgba(239, 68, 68, 0.95); color: white; border: 1px solid rgba(239, 68, 68, 0.3); } .hjty-toast-warning { background: rgba(245, 158, 11, 0.95); color: white; border: 1px solid rgba(245, 158, 11, 0.3); } .hjty-toast-info { background: rgba(59, 130, 246, 0.95); color: white; border: 1px solid rgba(59, 130, 246, 0.3); } .hjty-toast-icon { flex-shrink: 0; display: flex; align-items: center; justify-content: center; } .hjty-toast-message { flex: 1; line-height: 1.4; } /* 操作抽屉样式 */ .hjty-action-drawer-overlay { position: fixed; top: 0; left: 0; width: 100%; height: 100%; background: rgba(0, 0, 0, 0.5); backdrop-filter: blur(5px); z-index: 30000; display: flex; align-items: flex-end; justify-content: center; opacity: 0; transition: opacity 0.3s ease; } .hjty-action-drawer-overlay.hjty-action-drawer-show { opacity: 1; } .hjty-action-drawer { background: #f8fafc; border-radius: 16px 16px 0 0; width: 100%; max-height: 80vh; display: flex; flex-direction: column; transform: translateY(100%); transition: transform 0.3s cubic-bezier(0.4, 0, 0.2, 1); box-shadow: 0 -8px 32px rgba(0, 0, 0, 0.2); } .hjty-action-drawer-show .hjty-action-drawer { transform: translateY(0); } .hjty-action-header { padding: 20px 24px 16px 24px; border-bottom: 1px solid #e5e7eb; text-align: center; flex-shrink: 0; background: white; border-radius: 16px 16px 0 0; } .hjty-action-title { margin: 0; font-size: 18px; font-weight: 600; color: #1f2937; } .hjty-action-list { flex: 1; overflow-y: auto; padding: 8px 0; background: white; } .hjty-action-item { display: flex; align-items: center; padding: 16px 24px; cursor: pointer; transition: background-color 0.2s ease; border-bottom: 1px solid #f3f4f6; } .hjty-action-item:hover { background: #f9fafb; } .hjty-action-item:active { background: #f3f4f6; } .hjty-action-item:last-child { border-bottom: none; } .hjty-action-content { flex: 1; min-width: 0; } .hjty-action-name { font-size: 16px; font-weight: 500; color: #1f2937; margin-bottom: 4px; line-height: 1.4; } .hjty-action-subname { font-size: 13px; color: #6b7280; line-height: 1.4; word-break: break-all; } .hjty-action-arrow { flex-shrink: 0; margin-left: 12px; color: #9ca3af; transition: transform 0.2s ease; display: flex; align-items: center; align-self: center; } .hjty-action-item:hover .hjty-action-arrow { transform: translateX(2px); color: #6b7280; } .hjty-action-footer { padding: 12px 16px; border-top: 1px solid #e5e7eb; flex-shrink: 0; background: #f8fafc; } .hjty-action-close-btn { display: block; width: 100%; padding: 10px 20px; border: none; border-radius: 6px; background: #f3f4f6; color: #374151; font-size: 14px; font-weight: 500; cursor: pointer; transition: all 0.2s ease; } .hjty-action-close-btn:hover { background: #e5e7eb; color: #1f2937; } .hjty-action-close-btn:active { background: #d1d5db; transform: translateY(1px); } /* 播放器弹窗样式 */ .hjty-player-modal-overlay { position: fixed; top: 0; left: 0; width: 100%; height: 100%; background: rgba(0, 0, 0, 0.45); backdrop-filter: blur(5px); z-index: 40000; display: flex; align-items: center; justify-content: center; opacity: 0; transition: opacity 0.3s ease; } .hjty-player-modal-overlay.hjty-player-modal-show { opacity: 1; } .hjty-player-modal-content { background: #000000; border-radius: 12px; width: 95%; max-width: 1200px; height: 90%; max-height: 800px; display: flex; flex-direction: column; transform: scale(0.9); transition: transform 0.3s cubic-bezier(0.4, 0, 0.2, 1); box-shadow: 0 20px 60px rgba(0, 0, 0, 0.5); overflow: hidden; } .hjty-player-modal-show .hjty-player-modal-content { transform: scale(1); } .hjty-player-header { display: flex; justify-content: flex-end; align-items: center; padding: 16px 20px; background: #000000; flex-shrink: 0; } .hjty-player-close { background: none; border: none; font-size: 28px; color: #ffffff; cursor: pointer; padding: 0; width: 40px; height: 40px; display: flex; align-items: center; justify-content: center; border-radius: 8px; transition: all 0.2s ease; font-weight: 300; } .hjty-player-close:hover { background: rgba(255, 255, 255, 0.1); color: #ff4757; transform: scale(1.1); } .hjty-player-close:active { transform: scale(0.95); } .hjty-player-body { flex: 1; background: #000000; display: flex; align-items: center; justify-content: center; overflow: hidden; } #hjty-play-box { width: 100%; height: 100%; display: flex; align-items: center; justify-content: center; } /* 动画 */ @keyframes hjty-fadeIn { from { opacity: 0; } to { opacity: 1; } } @keyframes hjty-slideUp { from { transform: translateY(20px); opacity: 0; } to { transform: translateY(0); opacity: 1; } } @keyframes hjty-spin { from { transform: rotate(0deg); } to { transform: rotate(360deg); } } /* 移动端适配 */ @media (max-width: 768px) { #hjty-floating-toolbar { right: 15px; top: auto; bottom: 180px; transform: none; } #hjty-floating-toolbar.hjty-collapsed { right: -54px; /* 移动端收起时的位置 */ } .hjty-toggle-btn { left: -20px; width: 20px; height: 40px; } .hjty-toolbar-btn { width: 36px; height: 36px; } .hjty-modal-content { width: 95%; margin: 20px; } .hjty-modal-body { padding: 12px 16px; } .hjty-server-item { padding: 10px 12px; } .hjty-server-name { font-size: 13px; } .hjty-server-status { margin: 0 8px; } .hjty-login-content { width: 95%; margin: 20px; } .hjty-form-actions { flex-direction: column; } .hjty-toast { top: 10px; right: 10px; left: 10px; max-width: none; min-width: auto; } /* 播放器移动端适配 */ .hjty-player-modal-content { width: 100%; height: 100%; max-width: none; max-height: none; border-radius: 0; } .hjty-player-header { padding: 12px 16px; } .hjty-player-close { width: 36px; height: 36px; font-size: 24px; } } /* 暗色模式适配 */ @media (prefers-color-scheme: dark) { .hjty-toolbar-container { background: rgba(30, 30, 30, 0.95); border: 1px solid rgba(255, 255, 255, 0.1); } .hjty-toggle-btn { background: rgba(30, 30, 30, 0.95); backdrop-filter: blur(10px); color: #f3f4f6; } .hjty-toggle-btn:hover { background: rgba(30, 30, 30, 1); color: #ffffff; } .hjty-modal-content { background: #1f2937; } .hjty-modal-header { border-bottom-color: #374151; } .hjty-modal-header h3 { color: #f9fafb; } .hjty-modal-close { color: #9ca3af; } .hjty-modal-close:hover { background: #374151; color: #f3f4f6; } .hjty-modal-footer { border-top-color: #374151; } .hjty-btn-secondary { background: #374151; color: #f3f4f6; border-color: #4b5563; } .hjty-btn-secondary:hover { background: #4b5563; border-color: #6b7280; } .hjty-loading-content { background: #1f2937; } .hjty-loading-close:hover { background: #374151; color: #f3f4f6; } .hjty-loading-message { color: #f3f4f6; } .hjty-user-name { color: #f9fafb; } .hjty-user-status { color: #34d399; } .hjty-confirm-message { color: #f3f4f6; } .hjty-update-message { color: #f9fafb; } .hjty-update-details { color: #9ca3af; } .hjty-update-description { color: #f3f4f6; background: #374151; border-left-color: #60a5fa; } .hjty-server-item { border-color: #374151; background: #1f2937; } .hjty-server-item:hover { border-color: #4b5563; background: #374151; } .hjty-server-item.hjty-online { background: #064e3b; border-color: #059669; } .hjty-server-item.hjty-offline { background: #7f1d1d; border-color: #dc2626; } .hjty-server-item.hjty-selected { background: #1e3a8a; border-color: #3b82f6; } .hjty-server-item.hjty-selected .hjty-server-name { color: #93c5fd; } .hjty-server-name { color: #f9fafb; } .hjty-radio-indicator { border-color: #4b5563; } .hjty-radio-indicator.hjty-radio-selected { border-color: #60a5fa; background: #60a5fa; } .hjty-status-text { color: #d1d5db; } .hjty-refresh-btn:disabled { background: #4b5563; } .hjty-switch-btn:disabled { background: #4b5563; } .hjty-form-group label { color: #f3f4f6; } .hjty-form-input { background: #374151; border-color: #4b5563; color: #f3f4f6; } .hjty-form-input:focus { border-color: #60a5fa; box-shadow: 0 0 0 3px rgba(96, 165, 250, 0.1); } .hjty-form-input::placeholder { color: #6b7280; } /* 操作抽屉暗色模式 */ .hjty-action-drawer { background: #111827; } .hjty-action-header { border-bottom-color: #374151; background: #1f2937; } .hjty-action-title { color: #f9fafb; } .hjty-action-item { border-bottom-color: #374151; } .hjty-action-item:hover { background: #374151; } .hjty-action-item:active { background: #4b5563; } .hjty-action-name { color: #f9fafb; } .hjty-action-subname { color: #9ca3af; } .hjty-action-arrow { color: #6b7280; } .hjty-action-item:hover .hjty-action-arrow { color: #9ca3af; } .hjty-action-list { background: #1f2937; } .hjty-action-footer { border-top-color: #374151; background: #111827; } .hjty-action-close-btn { background: #374151; color: #f3f4f6; } .hjty-action-close-btn:hover { background: #4b5563; color: #f9fafb; } .hjty-action-close-btn:active { background: #6b7280; } } #play-box {height: 100%;display: flex;align-items: center;background-color: #000;} .van-popup.van-popup--center.play-box {width: 100%;height: 100%;max-width: 100%;} #bbs_float_menu{z-index:1!important;} #body.el-popup-parent--hidden{overflow: auto;} .van-list .van-list__loading, .sell-btn:before, .containeradvertising, .ishide, #tidio-chat{display: none !important;} `; document.head.appendChild(style); } function initFloatingButton() { if (document.getElementById('hjty-floating-toolbar')) return; const floatingToolbar = document.createElement('div'); floatingToolbar.id = 'hjty-floating-toolbar'; floatingToolbar.innerHTML = ` <div class="hjty-toolbar-container"> <button class="hjty-toggle-btn" title="收起工具栏"> <svg width="12" height="12" viewBox="0 0 24 24" fill="none"> <path d="M9 18L15 12L9 6" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/> </svg> </button> <button class="hjty-toolbar-btn hjty-unlock-btn" title="解锁"> <svg width="16" height="16" viewBox="0 0 24 24" fill="none"> <path d="M6 10V8C6 5.79086 7.79086 4 10 4H14C16.2091 4 18 5.79086 18 8V10" stroke="currentColor" stroke-width="2" stroke-linecap="round"/> <rect x="4" y="10" width="16" height="10" rx="2" stroke="currentColor" stroke-width="2" fill="none"/> <circle cx="12" cy="15" r="1" fill="currentColor"/> </svg> </button> <button class="hjty-toolbar-btn hjty-settings-btn" title="切换服务器"> <svg width="16" height="16" viewBox="0 0 24 24" fill="none"> <rect x="2" y="3" width="20" height="4" rx="1" stroke="currentColor" stroke-width="2"/> <rect x="2" y="10" width="20" height="4" rx="1" stroke="currentColor" stroke-width="2"/> <rect x="2" y="17" width="20" height="4" rx="1" stroke="currentColor" stroke-width="2"/> <circle cx="6" cy="5" r="1" fill="currentColor"/> <circle cx="6" cy="12" r="1" fill="currentColor"/> <circle cx="6" cy="19" r="1" fill="currentColor"/> <path d="M16 8L18 10L16 12" stroke="currentColor" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round"/> <path d="M8 15L10 17L8 19" stroke="currentColor" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round"/> </svg> </button> </div> `; document.body.appendChild(floatingToolbar); const unlockBtn = floatingToolbar.querySelector(CONFIG.SELECTORS.UNLOCK_BTN); const settingsBtn = floatingToolbar.querySelector(CONFIG.SELECTORS.SETTINGS_BTN); const toggleBtn = floatingToolbar.querySelector(CONFIG.SELECTORS.TOGGLE_BTN); unlockBtn.addEventListener('click', handleUnlockClick); const debouncedSwitchServer = Utils.debounce(() => switchServer(), 300); settingsBtn.addEventListener('click', debouncedSwitchServer); const debouncedToggleToolbar = Utils.debounce(() => toggleToolbar(), 200); toggleBtn.addEventListener('click', debouncedToggleToolbar); if (toolbarCollapsed) { floatingToolbar.classList.add('hjty-collapsed'); } toggleBtn.title = toolbarCollapsed ? '展开工具栏' : '收起工具栏'; App.updateToolbarButtons(Utils.checkLoginStatus()); document.addEventListener('keydown', (e) => { if (e.ctrlKey && e.shiftKey && e.key === 'H') { e.preventDefault(); toggleToolbar(); } }); } function toggleToolbar() { const toolbar = document.getElementById('hjty-floating-toolbar'); if (!toolbar) return; toolbarCollapsed = !toolbarCollapsed; if (toolbarCollapsed) { toolbar.classList.add('hjty-collapsed'); } else { toolbar.classList.remove('hjty-collapsed'); } const toggleBtn = toolbar.querySelector(CONFIG.SELECTORS.TOGGLE_BTN); if (toggleBtn) { toggleBtn.title = toolbarCollapsed ? '展开工具栏' : '收起工具栏'; } localStorage.setItem('hjty_toolbar_collapsed', toolbarCollapsed.toString()); } function handleUnlockClick() { if (newVersion !== '') { App.showUpdateConfirm(); return; } if (!Utils.checkLoginStatus()) { showLoginModal(); return; } Utils.deleteCookie('NOTLOGIN'); const pid = Utils.getPid(); if (!pid) { if (/\/videoplay/.test(location.href)) { Utils.copyText(shortVideoUrl).then(success => { if (success) { UI.showToast('短视频地址复制成功', 'success'); } else { UI.showToast('短视频地址复制失败', 'error'); } }); } else { UI.showToast('请进入帖子再解锁!', 'error'); } return; } if (m3u8_url !== '') { App.showVideoActions(); return; } UI.showLoadingModal('解锁中...', true, function () { UI.showToast('解锁已取消', 'warning'); }); App.getTopic(pid).then(topic => { App.showTopic(topic, pid); }).catch((err) => { UI.showToast(err, 'warning'); }); } function showLoginModal() { if (document.getElementById(CONFIG.MODAL_IDS.LOGIN)) return; const loginModal = document.createElement('div'); loginModal.id = CONFIG.MODAL_IDS.LOGIN; loginModal.className = 'hjty-modal-overlay'; loginModal.innerHTML = ` <div class="hjty-modal-content hjty-login-content"> <div class="hjty-modal-header"> <h3>用户登录</h3> <button class="hjty-modal-close" onclick="this.closest('.hjty-modal-overlay').remove()">×</button> </div> <div class="hjty-modal-body"> <form class="hjty-login-form"> <div class="hjty-form-group"> <label for="hjty-username">账号</label> <input type="text" id="hjty-username" class="hjty-form-input" placeholder="请输入账号" required> </div> <div class="hjty-form-group"> <label for="hjty-password">密码</label> <input type="password" id="hjty-password" class="hjty-form-input" placeholder="请输入密码" required> </div> <div class="hjty-form-actions"> <button type="submit" class="hjty-btn hjty-btn-primary hjty-login-btn">登录</button> <button type="button" class="hjty-btn hjty-btn-secondary hjty-register-btn">注册</button> </div> </form> </div> </div> `; document.body.appendChild(loginModal); const loginForm = loginModal.querySelector('.hjty-login-form'); const loginBtn = loginModal.querySelector('.hjty-login-btn'); const registerBtn = loginModal.querySelector('.hjty-register-btn'); loginForm.addEventListener('submit', async (e) => { e.preventDefault(); const username = document.getElementById('hjty-username').value.trim(); const password = document.getElementById('hjty-password').value.trim(); if (!username || !password) { UI.showToast('请填写完整信息', 'warning'); return; } await handleLogin(username, password, loginBtn, loginModal); }); registerBtn.addEventListener('click', () => { window.open(`${server}/register`, '_blank'); }); loginModal.addEventListener('click', (e) => { if (e.target === loginModal) loginModal.remove(); }); } async function handleLogin(username, password, loginBtn, loginModal) { try { loginBtn.disabled = true; loginBtn.innerHTML = '<div class="hjty-loading-spinner"><div class="hjty-spinner-ring"></div></div>登录中...'; const response = await fetch(`${server}/api/login`, { method: 'POST', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify({ username, password }) }); const result = await response.json(); if (!result.code) { localStorage.setItem(CONFIG.STORAGE_KEYS.USER_TOKEN, result.token); localStorage.setItem(CONFIG.STORAGE_KEYS.USER_NAME, username); UI.showToast('登录成功', 'success'); loginModal.remove(); App.updateToolbarButtons(true); } else { UI.showToast(result.msg || '登录失败', 'error'); } } catch (error) { UI.showToast('网络错误,请重试', 'error'); } finally { loginBtn.disabled = false; loginBtn.innerHTML = '登录'; } } function switchServer() { const servers = [ { id: 'vip', name: '服务器A', url: 'https://vip.tianya365.top' }, { id: 'b', name: '服务器B', url: 'https://haijiao.tianya365.top' }, { id: 'c', name: '服务器C', url: 'https://new.tianya365.top' }, { id: 'd', name: '服务器D', url: 'https://hj.tianya365.top' } ]; function getServerIdByUrl(url) { const foundServer = servers.find(s => s.url === url); return foundServer ? foundServer.id : 'vip'; } const currentServerId = getServerIdByUrl(server); const modal = document.createElement('div'); modal.id = CONFIG.MODAL_IDS.SERVER; modal.className = 'hjty-modal-overlay'; modal.style.zIndex = '25000'; modal.innerHTML = ` <div class="hjty-modal-content"> <div class="hjty-modal-header"> <h3>服务器节点切换</h3> <button class="hjty-modal-close" onclick="this.closest('.hjty-modal-overlay').remove()">×</button> </div> <div class="hjty-modal-body"> <div class="hjty-server-list"> ${servers.map(srv => ` <div class="hjty-server-item ${srv.id === currentServerId ? 'hjty-selected' : ''}" data-server-id="${srv.id}"> <div class="hjty-radio-indicator ${srv.id === currentServerId ? 'hjty-radio-selected' : ''}"></div> <div class="hjty-server-info"> <div class="hjty-server-name">${srv.name}</div> </div> <div class="hjty-server-status"> <div class="hjty-status-indicator hjty-checking"></div> <span class="hjty-status-text">检测中</span> <span class="hjty-latency"></span> </div> </div> `).join('')} </div> </div> <div class="hjty-modal-footer"> <button class="hjty-refresh-btn">重新检测</button> <button class="hjty-switch-btn" disabled>立即切换</button> </div> </div> `; document.body.appendChild(modal); let isChecking = false; let selectedServerId = currentServerId; async function checkServer(server) { const startTime = Date.now(); try { const controller = new AbortController(); const timeoutId = setTimeout(() => controller.abort(), 60000); await fetch(`${server.url}/api/login`, { method: 'HEAD', mode: 'no-cors', signal: controller.signal }); clearTimeout(timeoutId); const latency = Date.now() - startTime; return { success: true, latency }; } catch (error) { return { online: false, latency: 0 }; } } function updateServerStatus(serverId, result) { const serverItem = modal.querySelector(`[data-server-id="${serverId}"]`); const statusIndicator = serverItem.querySelector('.hjty-status-indicator'); const statusText = serverItem.querySelector('.hjty-status-text'); const latencySpan = serverItem.querySelector('.hjty-latency'); statusIndicator.className = `hjty-status-indicator ${result.success ? 'hjty-online' : 'hjty-offline'}`; statusText.textContent = result.success ? '正常' : '异常'; if (result.success) { latencySpan.textContent = `${result.latency}ms`; serverItem.classList.add('hjty-online'); serverItem.classList.remove('hjty-offline'); } else { latencySpan.textContent = ''; serverItem.classList.add('hjty-offline'); serverItem.classList.remove('hjty-online'); } } function selectServer(serverId) { modal.querySelectorAll('.hjty-server-item').forEach(item => { item.classList.remove('hjty-selected'); item.querySelector('.hjty-radio-indicator').classList.remove('hjty-radio-selected'); }); const selectedItem = modal.querySelector(`[data-server-id="${serverId}"]`); selectedItem.classList.add('hjty-selected'); selectedItem.querySelector('.hjty-radio-indicator').classList.add('hjty-radio-selected'); selectedServerId = serverId; const switchBtn = modal.querySelector('.hjty-switch-btn'); switchBtn.disabled = false; } function switchToServer(serverId) { const selectedServer = servers.find(s => s.id === serverId); if (selectedServer) { server = selectedServer.url; localStorage.setItem(CONFIG.STORAGE_KEYS.SERVER, server); UI.showToast(`已切换到:${selectedServer.name}`, 'success'); modal.remove(); App.updateToolbarButtons(Utils.checkLoginStatus()); } } async function checkAllServers() { if (isChecking) return; isChecking = true; const refreshBtn = modal.querySelector('.hjty-refresh-btn'); refreshBtn.disabled = true; refreshBtn.textContent = '检测中...'; modal.querySelectorAll('.hjty-server-item').forEach(item => { const statusIndicator = item.querySelector('.hjty-status-indicator'); const statusText = item.querySelector('.hjty-status-text'); const latencySpan = item.querySelector('.hjty-latency'); statusIndicator.className = 'hjty-status-indicator hjty-checking'; statusText.textContent = '检测中'; latencySpan.textContent = ''; item.classList.remove('hjty-online', 'hjty-offline'); }); const checkPromises = servers.map(async (srv) => { const result = await checkServer(srv); updateServerStatus(srv.id, result); }); await Promise.all(checkPromises); refreshBtn.disabled = false; refreshBtn.textContent = '重新检测'; isChecking = false; } modal.querySelectorAll('.hjty-server-item').forEach(item => { item.addEventListener('click', () => { const serverId = item.dataset.serverId; selectServer(serverId); }); }); const refreshBtn = modal.querySelector('.hjty-refresh-btn'); const switchBtn = modal.querySelector('.hjty-switch-btn'); refreshBtn.addEventListener('click', checkAllServers); switchBtn.addEventListener('click', () => switchToServer(selectedServerId)); modal.addEventListener('click', (e) => { if (e.target === modal) modal.remove(); }); checkAllServers(); } if (document.readyState === 'loading') { document.addEventListener('DOMContentLoaded', () => { buttonInitialized = checkTitleAndInitButton(); if (!buttonInitialized) { setupTitleObserver(); } }); } else { buttonInitialized = checkTitleAndInitButton(); if (!buttonInitialized) { setupTitleObserver(); } } (function () { const originOpen = XMLHttpRequest.prototype.open; XMLHttpRequest.prototype.open = function (method, url) { if (/\/api\/topic\/(hot\/topics\?|node\/(topics|news)|idol_list)/.test(url)) { const xhr = this; const getter = Object.getOwnPropertyDescriptor( XMLHttpRequest.prototype, "response" ).get; Object.defineProperty(xhr, "responseText", { get: () => { let result = getter.call(xhr); try { let res = JSON.parse(result); if (res['isEncrypted']) { res.data = Utils.encrypt(App.formatTitle(Utils.decrypt(res.data))) } else { res.data = App.formatTitle(res.data) } return JSON.stringify(res); } catch (e) { return result; } }, }); } else if (/\/api\/topic\/\d+/.test(url) && /tianya365/.test(url) === false) { m3u8_url = ''; } else if (/\/api\/attachment/.test(url)) { const xhr = this; const getter = Object.getOwnPropertyDescriptor( XMLHttpRequest.prototype, "response" ).get; Object.defineProperty(xhr, "responseText", { get: () => { let result = getter.call(xhr); App.getShortVideoUrl(result); return result; } }); } else if (/\/api\/video\/checkVideoCanPlay/.test(url)) { const xhr = this; const getter = Object.getOwnPropertyDescriptor( XMLHttpRequest.prototype, "response" ).get; Object.defineProperty(xhr, "responseText", { get: () => { return App.fixShortVideo(getter.call(xhr)); }, }); } else if (/\/api\/topic\/like/.test(url)) { const xhr = this; const getter = Object.getOwnPropertyDescriptor( XMLHttpRequest.prototype, "response" ).get; Object.defineProperty(xhr, "responseText", { get: () => { let result = getter.call(xhr); let res = JSON.parse(result); if (res.errorCode == 1) { res.errorCode = -1; } return JSON.stringify(res); }, }); } originOpen.apply(this, arguments); }; })(); })();