// ==UserScript== // @name 抖音续火花自动发送助手-集成一言API和TXTAPI // @namespace http://tampermonkey.net/ // @version 2.6 // @description 每天自动发送续火消息,支持自定义时间,集成一言API和TXTAPI,消息内容更丰富 // @author 飔梦 / 阚泥 // @match https://creator.douyin.com/creator-micro/data/following/chat // @icon  // @grant GM_setValue // @grant GM_getValue // @grant GM_registerMenuCommand // @grant GM_notification // @grant GM_listValues // @grant GM_deleteValue // @grant GM_xmlhttpRequest // @connect hitokoto.cn // @connect self // ==/UserScript== (function() { 'use strict'; // 默认配置 const DEFAULT_CONFIG = { baseMessage: "续火", sendTime: "00:01:00", checkInterval: 1000, maxWaitTime: 30000, maxRetryCount: 3, hitokotoTimeout: 60000, txtApiTimeout: 60000, useHitokoto: true, useTxtApi: true, // 默认开启TXTAPI txtApiMode: "manual", // 默认手动模式 txtApiManualRandom: true, // 手动模式是否随机选择 customMessage: "—————每日续火—————\n\n[TXTAPI]\n\n—————每日一言—————\n\n[API]\n", hitokotoFormat: "{hitokoto}\n—— {from}{from_who}", fromFormat: "{from}", fromWhoFormat: "「{from_who}」", txtApiUrl: "https://v1.hitokoto.cn/?encode=text", txtApiManualText: "文本1\n文本2\n文本3" }; // 状态变量 let isProcessing = false; let retryCount = 0; let countdownInterval = null; let isScriptCat = false; let userConfig = {}; let nextSendTime = null; // 检测是否是ScriptCat function detectScriptCat() { return typeof ScriptCat !== 'undefined' || (typeof GM_info !== 'undefined' && GM_info.scriptHandler === 'ScriptCat'); } // 初始化配置 function initConfig() { // 从存储中获取用户配置 const savedConfig = GM_getValue('userConfig'); userConfig = savedConfig ? {...DEFAULT_CONFIG, ...savedConfig} : {...DEFAULT_CONFIG}; // 确保所有配置项都存在 for (const key in DEFAULT_CONFIG) { if (userConfig[key] === undefined) { userConfig[key] = DEFAULT_CONFIG[key]; } } // 初始化手动文本发送记录 if (!GM_getValue('txtApiManualSentIndexes')) { GM_setValue('txtApiManualSentIndexes', []); } GM_setValue('userConfig', userConfig); return userConfig; } // 保存配置 function saveConfig() { GM_setValue('userConfig', userConfig); } // 创建UI控制面板 function createControlPanel() { // 移除可能已存在的面板 const existingPanel = document.getElementById('dy-fire-helper'); if (existingPanel) { existingPanel.remove(); } const panel = document.createElement('div'); panel.id = 'dy-fire-helper'; panel.style.cssText = ` position: fixed; top: 20px; right: 20px; width: 380px; background: rgba(255, 255, 255, 0.98); border-radius: 12px; box-shadow: 0 4px 20px rgba(0, 0, 0, 0.15); z-index: 9999; font-family: 'PingFang SC', 'Microsoft YaHei', sans-serif; padding: 15px; color: #333; transition: all 0.3s ease; max-height: 500px; overflow: hidden; border: 1px solid #eee; `; panel.innerHTML = `

抖音续火助手 ${isScriptCat ? '(ScriptCat)' : ''}

今日状态: 已发送
下次发送: 2023-11-05 00:01:00
倒计时: 23:45:12
重试次数: 0/${userConfig.maxRetryCount}
一言状态: 未获取
TXTAPI状态: 未获取
操作日志
系统已就绪,等待执行...
`; document.body.appendChild(panel); // 添加关闭按钮事件 document.getElementById('dy-fire-helper-close').addEventListener('click', function() { panel.style.display = 'none'; }); // 添加按钮事件 document.getElementById('dy-fire-send').addEventListener('click', sendMessage); document.getElementById('dy-fire-settings').addEventListener('click', showSettingsPanel); document.getElementById('dy-fire-clear').addEventListener('click', clearData); document.getElementById('dy-fire-reset').addEventListener('click', resetAllConfig); } // 显示设置面板 function showSettingsPanel() { // 移除可能已存在的设置面板 const existingSettings = document.getElementById('dy-fire-settings-panel'); if (existingSettings) { existingSettings.remove(); return; } const settingsPanel = document.createElement('div'); settingsPanel.id = 'dy-fire-settings-panel'; settingsPanel.style.cssText = ` position: fixed; top: 50%; left: 50%; transform: translate(-50%, -50%); max-width: 90vw; width: 500px; background: white; border-radius: 12px; box-shadow: 0 5px 25px rgba(0, 0, 0, 0.2); z-index: 10000; padding: 20px; font-family: 'PingFang SC', 'Microsoft YaHei', sans-serif; max-height: 90vh; overflow-y: auto; box-sizing: border-box; `; settingsPanel.innerHTML = `

设置

可用变量: {hitokoto} {from} {from_who}
示例: {hitokoto} —— {from}{from_who}
当from不为空时显示此格式,为空时不显示
当from_who不为空时显示此格式,为空时不显示
使用 [API] 作为一言内容的占位符
使用 [TXTAPI] 作为TXTAPI内容的占位符
支持换行符,关闭API时占位符标记将保留
`; document.body.appendChild(settingsPanel); // 添加TXTAPI模式切换事件 const modeRadios = document.querySelectorAll('input[name="txt-api-mode"]'); modeRadios.forEach(radio => { radio.addEventListener('change', function() { const mode = this.value; document.getElementById('txt-api-url-container').style.display = mode === 'api' ? 'block' : 'none'; document.getElementById('txt-api-manual-container').style.display = mode === 'manual' ? 'block' : 'none'; }); }); // 添加TXTAPI开关切换事件 document.getElementById('dy-fire-settings-use-txtapi').addEventListener('change', function() { const useTxtApi = this.checked; document.getElementById('txt-api-mode-container').style.display = useTxtApi ? 'block' : 'none'; // 根据当前选择的模式显示相应的表单 const currentMode = document.querySelector('input[name="txt-api-mode"]:checked').value; document.getElementById('txt-api-url-container').style.display = (useTxtApi && currentMode === 'api') ? 'block' : 'none'; document.getElementById('txt-api-manual-container').style.display = (useTxtApi && currentMode === 'manual') ? 'block' : 'none'; }); // 添加关闭按钮事件 document.getElementById('dy-fire-settings-close').addEventListener('click', function() { settingsPanel.remove(); }); // 添加保存按钮事件 document.getElementById('dy-fire-settings-save').addEventListener('click', function() { const timeValue = document.getElementById('dy-fire-settings-time').value; const useHitokoto = document.getElementById('dy-fire-settings-use-hitokoto').checked; const useTxtApi = document.getElementById('dy-fire-settings-use-txtapi').checked; const txtApiMode = document.querySelector('input[name="txt-api-mode"]:checked').value; const txtApiRandom = document.getElementById('dy-fire-settings-txtapi-random').checked; const txtApiUrl = document.getElementById('dy-fire-settings-txtapi-url').value; const txtApiManualText = document.getElementById('dy-fire-settings-txtapi-manual').value; const maxRetryCount = parseInt(document.getElementById('dy-fire-settings-retry-count').value, 10); const hitokotoFormat = document.getElementById('dy-fire-settings-hitokoto-format').value; const fromFormat = document.getElementById('dy-fire-settings-from-format').value; const fromWhoFormat = document.getElementById('dy-fire-settings-from-who-format').value; const customMessage = document.getElementById('dy-fire-settings-custom-message').value; // 验证时间格式 (HH:mm:ss) if (!/^([0-1]?[0-9]|2[0-3]):[0-5][0-9]:[0-5][0-9]$/.test(timeValue)) { addLog('时间格式错误,请使用HH:mm:ss格式', 'error'); return; } // 验证重试次数 if (isNaN(maxRetryCount) || maxRetryCount < 1 || maxRetryCount > 10) { addLog('重试次数必须是1-10之间的数字', 'error'); return; } // 验证TXTAPI链接(仅在API模式下) if (useTxtApi && txtApiMode === 'api' && !txtApiUrl) { addLog('请填写TXTAPI链接', 'error'); return; } // 验证手动文本(仅在手动模式下) if (useTxtApi && txtApiMode === 'manual' && !txtApiManualText.trim()) { addLog('请填写手动文本内容', 'error'); return; } // 更新配置 userConfig.sendTime = timeValue; userConfig.useHitokoto = useHitokoto; userConfig.useTxtApi = useTxtApi; userConfig.txtApiMode = txtApiMode; userConfig.txtApiManualRandom = txtApiRandom; userConfig.txtApiUrl = txtApiUrl; userConfig.txtApiManualText = txtApiManualText; userConfig.maxRetryCount = maxRetryCount; userConfig.hitokotoFormat = hitokotoFormat; userConfig.fromFormat = fromFormat; userConfig.fromWhoFormat = fromWhoFormat; userConfig.customMessage = customMessage; // 保存配置 saveConfig(); // 更新状态 updateStatus(GM_getValue('lastSentDate', '') === new Date().toDateString()); updateRetryCount(); // 关闭设置面板 settingsPanel.remove(); addLog('设置已保存', 'success'); }); } // 添加日志 function addLog(message, type = 'info') { const now = new Date(); const timeString = now.toLocaleTimeString(); const logEntry = document.createElement('div'); logEntry.style.color = type === 'success' ? '#28a745' : type === 'error' ? '#dc3545' : '#17a2b8'; logEntry.textContent = `${timeString} - ${message}`; const logContainer = document.getElementById('dy-fire-log'); logContainer.prepend(logEntry); // 限制日志数量 if (logContainer.children.length > 8) { logContainer.removeChild(logContainer.lastChild); } // 自动滚动到顶部 logContainer.scrollTop = 0; } // 更新重试计数显示 function updateRetryCount() { document.getElementById('dy-fire-retry').textContent = `${retryCount}/${userConfig.maxRetryCount}`; } // 更新一言状态显示 function updateHitokotoStatus(status, isSuccess = true) { const statusEl = document.getElementById('dy-fire-hitokoto'); statusEl.textContent = status; statusEl.style.color = isSuccess ? '#28a745' : '#dc3545'; } // 更新TXTAPI状态显示 function updateTxtApiStatus(status, isSuccess = true) { const statusEl = document.getElementById('dy-fire-txtapi'); statusEl.textContent = status; statusEl.style.color = isSuccess ? '#28a745' : '#dc3545'; } // 获取消息内容 async function getMessageContent() { let customMessage = userConfig.customMessage || userConfig.baseMessage; // 获取一言内容 let hitokotoContent = ''; if (userConfig.useHitokoto) { try { addLog('正在获取一言内容...', 'info'); hitokotoContent = await getHitokoto(); addLog('一言内容获取成功', 'success'); } catch (error) { addLog(`一言获取失败: ${error.message}`, 'error'); hitokotoContent = '一言获取失败~'; } } // 获取TXTAPI内容 let txtApiContent = ''; if (userConfig.useTxtApi) { try { addLog('正在获取TXTAPI内容...', 'info'); txtApiContent = await getTxtApiContent(); addLog('TXTAPI内容获取成功', 'success'); } catch (error) { addLog(`TXTAPI获取失败: ${error.message}`, 'error'); txtApiContent = 'TXTAPI获取失败~'; } } // 替换自定义消息中的占位符 if (customMessage.includes('[API]')) { customMessage = customMessage.replace('[API]', hitokotoContent); } else if (userConfig.useHitokoto) { customMessage += ` | ${hitokotoContent}`; } if (customMessage.includes('[TXTAPI]')) { customMessage = customMessage.replace('[TXTAPI]', txtApiContent); } else if (userConfig.useTxtApi) { customMessage += ` | ${txtApiContent}`; } return customMessage; } // 获取一言内容 function getHitokoto() { return new Promise((resolve, reject) => { const timeout = setTimeout(() => { reject(new Error('一言API请求超时')); }, userConfig.hitokotoTimeout); GM_xmlhttpRequest({ method: 'GET', url: 'https://v1.hitokoto.cn/', responseType: 'json', onload: function(response) { clearTimeout(timeout); if (response.status === 200) { try { const data = response.response; let message = formatHitokoto(userConfig.hitokotoFormat, data); updateHitokotoStatus('获取成功'); resolve(message); } catch (e) { updateHitokotoStatus('解析失败', false); reject(new Error('一言API响应解析失败')); } } else { updateHitokotoStatus('请求失败', false); reject(new Error(`一言API请求失败: ${response.status}`)); } }, onerror: function(error) { clearTimeout(timeout); updateHitokotoStatus('网络错误', false); reject(new Error('一言API网络错误')); }, ontimeout: function() { clearTimeout(timeout); updateHitokotoStatus('请求超时', false); reject(new Error('一言API请求超时')); } }); }); } // 格式化一言内容 function formatHitokoto(format, data) { // 首先替换基本变量 let result = format .replace(/{hitokoto}/g, data.hitokoto || ''); // 格式化from let fromFormatted = ''; if (data.from) { fromFormatted = userConfig.fromFormat.replace(/{from}/g, data.from); } result = result.replace(/{from}/g, fromFormatted); // 格式化from_who let fromWhoFormatted = ''; if (data.from_who) { fromWhoFormatted = userConfig.fromWhoFormat.replace(/{from_who}/g, data.from_who); } result = result.replace(/{from_who}/g, fromWhoFormatted); return result; } // 获取TXTAPI内容 function getTxtApiContent() { return new Promise((resolve, reject) => { if (userConfig.txtApiMode === 'api') { // API模式 const timeout = setTimeout(() => { reject(new Error('TXTAPI请求超时')); }, userConfig.txtApiTimeout); GM_xmlhttpRequest({ method: 'GET', url: userConfig.txtApiUrl, onload: function(response) { clearTimeout(timeout); if (response.status === 200) { try { updateTxtApiStatus('获取成功'); resolve(response.responseText.trim()); } catch (e) { updateTxtApiStatus('解析失败', false); reject(new Error('TXTAPI响应解析失败')); } } else { updateTxtApiStatus('请求失败', false); reject(new Error(`TXTAPI请求失败: ${response.status}`)); } }, onerror: function(error) { clearTimeout(timeout); updateTxtApiStatus('网络错误', false); reject(new Error('TXTAPI网络错误')); }, ontimeout: function() { clearTimeout(timeout); updateTxtApiStatus('请求超时', false); reject(new Error('TXTAPI请求超时')); } }); } else { // 手动模式 try { const lines = userConfig.txtApiManualText.split('\n').filter(line => line.trim()); if (lines.length === 0) { updateTxtApiStatus('无内容', false); reject(new Error('手动文本内容为空')); return; } // 获取已发送的索引 let sentIndexes = GM_getValue('txtApiManualSentIndexes', []); if (userConfig.txtApiManualRandom) { // 随机模式:找出未发送的索引 let availableIndexes = []; for (let i = 0; i < lines.length; i++) { if (!sentIndexes.includes(i)) { availableIndexes.push(i); } } // 如果所有行都已发送,重置已发送记录 if (availableIndexes.length === 0) { sentIndexes = []; availableIndexes = Array.from({length: lines.length}, (_, i) => i); GM_setValue('txtApiManualSentIndexes', []); } // 随机选择一个未发送的索引 const randomIndex = Math.floor(Math.random() * availableIndexes.length); const selectedIndex = availableIndexes[randomIndex]; const selectedText = lines[selectedIndex].trim(); // 记录已发送的索引 sentIndexes.push(selectedIndex); GM_setValue('txtApiManualSentIndexes', sentIndexes); updateTxtApiStatus('获取成功'); resolve(selectedText); } else { // 顺序模式:按顺序选择下一行 let nextIndex = 0; if (sentIndexes.length > 0) { nextIndex = (sentIndexes[sentIndexes.length - 1] + 1) % lines.length; } const selectedText = lines[nextIndex].trim(); // 记录已发送的索引 sentIndexes.push(nextIndex); GM_setValue('txtApiManualSentIndexes', sentIndexes); updateTxtApiStatus('获取成功'); resolve(selectedText); } } catch (e) { updateTxtApiStatus('解析失败', false); reject(new Error('手动文本解析失败')); } } }); } // 发送消息函数 async function sendMessage() { if (isProcessing) { addLog('已有任务正在进行中', 'error'); return; } // 检查是否今天已发送 const lastSentDate = GM_getValue('lastSentDate', ''); const today = new Date().toDateString(); if (lastSentDate === today) { addLog('今天已经发送过消息', 'info'); return; } isProcessing = true; retryCount = 0; updateRetryCount(); addLog('开始发送流程...', 'info'); // 执行发送流程 executeSendProcess(); } // 执行发送流程 async function executeSendProcess() { retryCount++; updateRetryCount(); if (retryCount > userConfig.maxRetryCount) { addLog(`已达到最大重试次数 (${userConfig.maxRetryCount})`, 'error'); isProcessing = false; return; } addLog(`尝试发送 (${retryCount}/${userConfig.maxRetryCount})`, 'info'); // 直接查找聊天输入框 setTimeout(tryFindChatInput, 1000); } // 尝试查找聊天输入框并发送消息 async function tryFindChatInput() { const input = document.querySelector('.chat-input-dccKiL'); if (input) { addLog('找到聊天输入框', 'info'); // 获取消息内容 let messageToSend; try { messageToSend = await getMessageContent(); addLog('消息内容准备完成', 'success'); } catch (error) { addLog(`消息获取失败: ${error.message}`, 'error'); messageToSend = `${userConfig.baseMessage} | 消息获取失败~`; } // 清空输入框 input.textContent = ''; // 输入消息 input.focus(); // 处理换行符 const lines = messageToSend.split('\n'); for (let i = 0; i < lines.length; i++) { document.execCommand('insertText', false, lines[i]); if (i < lines.length - 1) { // 插入换行符 document.execCommand('insertLineBreak'); } } input.dispatchEvent(new Event('input', { bubbles: true })); // 检查发送按钮状态 setTimeout(() => { const sendBtn = document.querySelector('.chat-btn'); if (sendBtn && !sendBtn.disabled) { addLog('正在发送消息...', 'info'); sendBtn.click(); // 确认发送成功 setTimeout(() => { addLog('消息发送成功!', 'success'); // 立即记录发送状态 const today = new Date().toDateString(); GM_setValue('lastSentDate', today); // 更新状态显示 updateStatus(true); isProcessing = false; // 显示通知 if (typeof GM_notification !== 'undefined') { try { GM_notification({ title: '抖音续火助手', text: '续火消息发送成功!', timeout: 3000 }); } catch (e) { GM_notification('续火消息发送成功!', '抖音续火助手'); } } }, 1000); } else { addLog('发送按钮不可用', 'error'); setTimeout(executeSendProcess, 2000); } }, 500); } else { addLog('未找到输入框,重试中...', 'error'); setTimeout(executeSendProcess, 2000); } } // 解析时间字符串为日期对象 function parseTimeString(timeStr) { const [hours, minutes, seconds] = timeStr.split(':').map(Number); const now = new Date(); const targetTime = new Date(now); targetTime.setHours(hours, minutes, seconds || 0, 0); // 如果目标时间已经过去,设置为明天 if (targetTime <= now) { targetTime.setDate(targetTime.getDate() + 1); } return targetTime; } // 更新状态 function updateStatus(isSent) { const statusEl = document.getElementById('dy-fire-status'); if (isSent) { statusEl.textContent = '已发送'; statusEl.style.color = '#28a745'; } else { statusEl.textContent = '未发送'; statusEl.style.color = '#dc3545'; // 如果是"未发送"状态,检查是否需要自动发送 autoSendIfNeeded(); } // 更新下次发送时间 const now = new Date(); if (isSent) { // 如果已发送,下次发送时间是明天指定时间 nextSendTime = parseTimeString(userConfig.sendTime); // 确保是明天而不是后天 const tomorrow = new Date(now); tomorrow.setDate(tomorrow.getDate() + 1); if (nextSendTime.getDate() !== tomorrow.getDate()) { nextSendTime.setDate(tomorrow.getDate()); } } else { // 如果未发送,检查当前时间是否已过指定时间 nextSendTime = parseTimeString(userConfig.sendTime); // 如果目标时间已经过去,设置为明天 if (nextSendTime <= now) { nextSendTime.setDate(nextSendTime.getDate() + 1); } } document.getElementById('dy-fire-next').textContent = nextSendTime.toLocaleString(); // 开始倒计时 startCountdown(nextSendTime); } // 检查是否需要自动发送 function autoSendIfNeeded() { const now = new Date(); const today = new Date().toDateString(); const lastSentDate = GM_getValue('lastSentDate', ''); // 解析发送时间 const [targetHour, targetMinute, targetSecond] = userConfig.sendTime.split(':').map(Number); // 如果今天未发送且当前时间已过指定时间,则自动发送 if (lastSentDate !== today) { const targetTimeToday = new Date(); targetTimeToday.setHours(targetHour, targetMinute, targetSecond || 0, 0); if (now >= targetTimeToday && !isProcessing) { addLog(`检测到今日未发送且已过${userConfig.sendTime},自动发送`, 'info'); sendMessage(); } } } // 开始倒计时 function startCountdown(targetTime) { // 清除之前的倒计时 if (countdownInterval) { clearInterval(countdownInterval); } function update() { const now = new Date(); const diff = targetTime - now; if (diff <= 0) { document.getElementById('dy-fire-countdown').textContent = '00:00:00'; // 检查是否今天已发送 const lastSentDate = GM_getValue('lastSentDate', ''); const today = new Date().toDateString(); if (lastSentDate === today) { // 如果已发送,设置明天指定时间为目标时间 nextSendTime = parseTimeString(userConfig.sendTime); // 确保是明天而不是后天 const tomorrow = new Date(now); tomorrow.setDate(tomorrow.getDate() + 1); if (nextSendTime.getDate() !== tomorrow.getDate()) { nextSendTime.setDate(tomorrow.getDate()); } startCountdown(nextSendTime); } else { // 如果未发送,清空数据并发送消息 if (!isProcessing) { // 清空数据,避免重复发送 GM_setValue('lastSentDate', ''); updateStatus(false); addLog('已清空发送记录,准备发送新消息', 'info'); sendMessage(); } } return; } const hours = Math.floor(diff / (1000 * 60 * 60)); const minutes = Math.floor((diff % (1000 * 60 * 60)) / (1000 * 60)); const seconds = Math.floor((diff % (1000 * 60)) / 1000); document.getElementById('dy-fire-countdown').textContent = `${hours.toString().padStart(2, '0')}:${minutes.toString().padStart(2, '0')}:${seconds.toString().padStart(2, '0')}`; } update(); countdownInterval = setInterval(update, 1000); } // 清空数据 function clearData() { GM_setValue('lastSentDate', ''); GM_setValue('txtApiManualSentIndexes', []); addLog('发送记录已清空', 'info'); updateStatus(false); retryCount = 0; updateRetryCount(); updateHitokotoStatus('未获取'); updateTxtApiStatus('未获取'); } // 重置所有配置 function resetAllConfig() { // 获取所有存储的值并删除 if (typeof GM_listValues !== 'undefined' && typeof GM_deleteValue !== 'undefined') { try { const values = GM_listValues(); values.forEach(key => { GM_deleteValue(key); }); } catch (e) { // ScriptCat可能不支持GM_listValues GM_setValue('lastSentDate', ''); GM_setValue('userConfig', ''); GM_setValue('txtApiManualSentIndexes', []); } } else { // 如果GM_listValues不可用,只删除已知的key GM_setValue('lastSentDate', ''); GM_setValue('userConfig', ''); GM_setValue('txtApiManualSentIndexes', []); } // 重新初始化配置 initConfig(); addLog('所有配置已重置', 'info'); updateStatus(false); retryCount = 0; updateRetryCount(); updateHitokotoStatus('未获取'); updateTxtApiStatus('未获取'); // 显示通知 if (typeof GM_notification !== 'undefined') { try { GM_notification({ title: '抖音续火助手', text: '所有配置已重置!', timeout: 3000 }); } catch (e) { GM_notification('所有配置已重置!', '抖音续火助手'); } } } // 初始化函数 function init() { // 检测是否是ScriptCat isScriptCat = detectScriptCat(); // 初始化配置 initConfig(); // 创建控制面板 createControlPanel(); // 检查今天是否已发送 const lastSentDate = GM_getValue('lastSentDate', ''); const today = new Date().toDateString(); const isSentToday = lastSentDate === today; // 更新状态 updateStatus(isSentToday); // 注册菜单命令 if (typeof GM_registerMenuCommand !== 'undefined') { try { GM_registerMenuCommand('抖音续火助手-显示面板', function() { document.getElementById('dy-fire-helper').style.display = 'block'; }); GM_registerMenuCommand('立即发送续火消息', sendMessage); GM_registerMenuCommand('设置', showSettingsPanel); GM_registerMenuCommand('清空发送记录', clearData); GM_registerMenuCommand('重置所有配置', resetAllConfig); } catch (e) { addLog('菜单命令注册失败,使用面板控制', 'error'); } } addLog('抖音续火助手已启动', 'info'); // 设置定时任务,每秒检查一次是否到发送时间 setInterval(() => { const now = new Date(); const [targetHour, targetMinute, targetSecond] = userConfig.sendTime.split(':').map(Number); if (now.getHours() === targetHour && now.getMinutes() === targetMinute && now.getSeconds() === (targetSecond || 0)) { const today = new Date().toDateString(); const lastSentDate = GM_getValue('lastSentDate', ''); if (lastSentDate !== today) { addLog('定时任务触发发送', 'info'); sendMessage(); } } }, 1000); } // 等待页面加载完成后初始化 if (document.readyState === 'loading') { document.addEventListener('DOMContentLoaded', init); } else { init(); } })();