// ==UserScript==
// @name 抖音续火花自动发送助手-集成一言API和TXTAPI
// @namespace http://tampermonkey.net/
// @version 2.0
// @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,
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",
enableTargetUser: false,
targetUsername: "",
userSearchTimeout: 10000,
maxHistoryLogs: 200,
searchDebounceDelay: 500,
searchThrottleDelay: 1000,
clickMethod: "direct",
pageLoadWaitTime: 5000,
chatInputCheckInterval: 1000
};
// 状态变量
let isProcessing = false;
let retryCount = 0;
let countdownInterval = null;
let isScriptCat = false;
let userConfig = {};
let nextSendTime = null;
let currentState = "idle";
let chatObserver = null;
let searchTimeout = null;
let lastSearchTime = 0;
let searchDebounceTimer = null;
let chatInputCheckTimer = 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', []);
}
if (!GM_getValue('historyLogs')) {
GM_setValue('historyLogs', []);
}
GM_setValue('userConfig', userConfig);
return userConfig;
}
// 保存配置
function saveConfig() {
GM_setValue('userConfig', userConfig);
}
// 添加历史日志
function addHistoryLog(message, type = 'info') {
const logs = GM_getValue('historyLogs', []);
const logEntry = {
timestamp: new Date().toISOString(),
message: message,
type: type
};
logs.unshift(logEntry);
if (logs.length > userConfig.maxHistoryLogs) {
logs.splice(userConfig.maxHistoryLogs);
}
GM_setValue('historyLogs', logs);
addLog(message, type);
}
// 获取历史日志
function getHistoryLogs() {
return GM_getValue('historyLogs', []);
}
// 清空历史日志
function clearHistoryLogs() {
GM_setValue('historyLogs', []);
addHistoryLog('历史日志已清空', 'info');
}
// 导出历史日志
function exportHistoryLogs() {
const logs = getHistoryLogs();
const logText = logs.map(log =>
`${new Date(log.timestamp).toLocaleString()} [${log.type.toUpperCase()}] ${log.message}`
).join('\n');
const blob = new Blob([logText], { type: 'text/plain' });
const url = URL.createObjectURL(blob);
const a = document.createElement('a');
a.href = url;
a.download = `抖音续火助手日志_${new Date().toISOString().split('T')[0]}.txt`;
document.body.appendChild(a);
a.click();
document.body.removeChild(a);
URL.revokeObjectURL(url);
addHistoryLog('日志已导出', 'success');
}
// 创建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: 420px;
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: 600px;
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-history').addEventListener('click', showHistoryPanel);
document.getElementById('dy-fire-clear').addEventListener('click', clearData);
document.getElementById('dy-fire-reset').addEventListener('click', resetAllConfig);
}
// 更新用户状态显示
function updateUserStatus(status, isSuccess = null) {
const statusEl = document.getElementById('dy-fire-user-status');
statusEl.textContent = status;
if (isSuccess === true) {
statusEl.style.color = '#28a745';
} else if (isSuccess === false) {
statusEl.style.color = '#dc3545';
} else {
statusEl.style.color = '#6c757d';
}
}
// 显示历史日志面板
function showHistoryPanel() {
const existingPanel = document.getElementById('dy-fire-history-panel');
if (existingPanel) {
existingPanel.remove();
return;
}
const historyPanel = document.createElement('div');
historyPanel.id = 'dy-fire-history-panel';
historyPanel.style.cssText = `
position: fixed;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
max-width: 90vw;
width: 700px;
height: 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;
display: flex;
flex-direction: column;
box-sizing: border-box;
`;
const logs = getHistoryLogs();
const logItems = logs.map(log => `
${new Date(log.timestamp).toLocaleString()}
${log.type.toUpperCase()}
${log.message}
`).join('');
historyPanel.innerHTML = `
历史日志 (${logs.length}/${userConfig.maxHistoryLogs})
${logs.length > 0 ? logItems : '
暂无日志记录
'}
`;
document.body.appendChild(historyPanel);
document.getElementById('dy-fire-history-close').addEventListener('click', function() {
historyPanel.remove();
});
document.getElementById('dy-fire-history-export').addEventListener('click', exportHistoryLogs);
document.getElementById('dy-fire-history-clear').addEventListener('click', clearHistoryLogs);
}
// 显示设置面板
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;
max-height: 80vh;
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;
overflow-y: auto;
box-sizing: border-box;
`;
settingsPanel.innerHTML = `
设置
可用变量: {hitokoto} {from} {from_who}
示例: {hitokoto} —— {from}{from_who}
使用 [API] 作为一言内容的占位符
使用 [TXTAPI] 作为TXTAPI内容的占位符
支持换行符,关闭API时占位符标记将保留
`;
document.body.appendChild(settingsPanel);
document.getElementById('dy-fire-settings-enable-target').addEventListener('change', function() {
document.getElementById('target-user-container').style.display = this.checked ? 'block' : 'none';
});
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';
});
});
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', saveSettings);
}
// 保存设置
function saveSettings() {
const timeValue = document.getElementById('dy-fire-settings-time').value;
const enableTargetUser = document.getElementById('dy-fire-settings-enable-target').checked;
const targetUsername = document.getElementById('dy-fire-settings-target-user').value;
const clickMethod = document.querySelector('input[name="click-method"]:checked').value;
const pageLoadWaitTime = parseInt(document.getElementById('dy-fire-settings-page-wait').value, 10);
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 userSearchTimeout = parseInt(document.getElementById('dy-fire-settings-user-timeout').value, 10);
const maxHistoryLogs = parseInt(document.getElementById('dy-fire-settings-max-logs').value, 10);
const debounceDelay = parseInt(document.getElementById('dy-fire-settings-debounce-delay').value, 10);
const throttleDelay = parseInt(document.getElementById('dy-fire-settings-throttle-delay').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;
if (!/^([0-1]?[0-9]|2[0-3]):[0-5][0-9]:[0-5][0-9]$/.test(timeValue)) {
addHistoryLog('时间格式错误,请使用HH:mm:ss格式', 'error');
return;
}
if (isNaN(maxRetryCount) || maxRetryCount < 1 || maxRetryCount > 10) {
addHistoryLog('重试次数必须是1-10之间的数字', 'error');
return;
}
if (isNaN(userSearchTimeout) || userSearchTimeout < 1000 || userSearchTimeout > 30000) {
addHistoryLog('用户查找超时必须介于1000-30000毫秒之间', 'error');
return;
}
if (isNaN(maxHistoryLogs) || maxHistoryLogs < 50 || maxHistoryLogs > 1000) {
addHistoryLog('最大历史日志数量必须介于50-1000之间', 'error');
return;
}
if (isNaN(debounceDelay) || debounceDelay < 100 || debounceDelay > 2000) {
addHistoryLog('防抖延迟必须介于100-2000毫秒之间', 'error');
return;
}
if (isNaN(throttleDelay) || throttleDelay < 500 || throttleDelay > 3000) {
addHistoryLog('节流延迟必须介于500-3000毫秒之间', 'error');
return;
}
if (isNaN(pageLoadWaitTime) || pageLoadWaitTime < 1000 || pageLoadWaitTime > 15000) {
addHistoryLog('页面加载等待时间必须介于1000-15000毫秒之间', 'error');
return;
}
if (useTxtApi && txtApiMode === 'api' && !txtApiUrl) {
addHistoryLog('请填写TXTAPI链接', 'error');
return;
}
if (useTxtApi && txtApiMode === 'manual' && !txtApiManualText.trim()) {
addHistoryLog('请填写手动文本内容', 'error');
return;
}
if (enableTargetUser && !targetUsername.trim()) {
addHistoryLog('启用目标用户查找时,必须填写目标用户名', 'error');
return;
}
userConfig.sendTime = timeValue;
userConfig.enableTargetUser = enableTargetUser;
userConfig.targetUsername = targetUsername.trim();
userConfig.clickMethod = clickMethod;
userConfig.pageLoadWaitTime = pageLoadWaitTime;
userConfig.useHitokoto = useHitokoto;
userConfig.useTxtApi = useTxtApi;
userConfig.txtApiMode = txtApiMode;
userConfig.txtApiManualRandom = txtApiRandom;
userConfig.txtApiUrl = txtApiUrl;
userConfig.txtApiManualText = txtApiManualText;
userConfig.maxRetryCount = maxRetryCount;
userConfig.userSearchTimeout = userSearchTimeout;
userConfig.maxHistoryLogs = maxHistoryLogs;
userConfig.searchDebounceDelay = debounceDelay;
userConfig.searchThrottleDelay = throttleDelay;
userConfig.hitokotoFormat = hitokotoFormat;
userConfig.fromFormat = fromFormat;
userConfig.fromWhoFormat = fromWhoFormat;
userConfig.customMessage = customMessage;
saveConfig();
updateUserStatus(userConfig.enableTargetUser ? `已启用: ${userConfig.targetUsername}` : '未启用');
updateRetryCount();
document.getElementById('dy-fire-settings-panel').remove();
addHistoryLog('设置已保存', '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';
}
// 初始化聊天列表观察器
function initChatObserver() {
if (chatObserver) {
chatObserver.disconnect();
chatObserver = null;
}
if (!userConfig.enableTargetUser || currentState !== 'searching') {
return;
}
chatObserver = new MutationObserver(function(mutations) {
clearTimeout(searchDebounceTimer);
searchDebounceTimer = setTimeout(() => {
const now = Date.now();
if (now - lastSearchTime < userConfig.searchThrottleDelay) {
return;
}
lastSearchTime = now;
findAndClickTargetUser();
}, userConfig.searchDebounceDelay);
});
const chatContainer = findChatContainer();
if (chatContainer) {
chatObserver.observe(chatContainer, {
childList: true,
subtree: true,
attributes: false,
characterData: false
});
addHistoryLog('聊天列表观察器已启动', 'info');
} else {
addHistoryLog('未找到聊天列表容器,将使用备用查找策略', 'warn');
chatObserver.observe(document.body, {
childList: true,
subtree: false,
attributes: false,
characterData: false
});
}
}
// 查找聊天容器
function findChatContainer() {
const possibleSelectors = [
'.chat-list-container',
'.semi-list',
'[role="list"]',
'.conversation-list',
'.message-list'
];
for (const selector of possibleSelectors) {
const container = document.querySelector(selector);
if (container) {
return container;
}
}
const sampleUser = document.querySelector('.item-header-name-vL_79m');
if (sampleUser) {
let parent = sampleUser;
for (let i = 0; i < 10; i++) {
parent = parent.parentElement;
if (parent && parent.children.length > 5) {
return parent;
}
if (!parent) break;
}
}
return null;
}
// 停止聊天观察器
function stopChatObserver() {
if (chatObserver) {
chatObserver.disconnect();
chatObserver = null;
}
clearTimeout(searchDebounceTimer);
addHistoryLog('聊天列表观察器已停止', 'info');
}
// 安全地创建鼠标事件
function createSafeMouseEvent(type, options = {}) {
try {
const safeOptions = {
bubbles: true,
cancelable: true,
view: window,
...options
};
return new MouseEvent(type, safeOptions);
} catch (error) {
try {
const safeOptions = {
bubbles: true,
cancelable: true,
...options
};
delete safeOptions.view;
return new MouseEvent(type, safeOptions);
} catch (error2) {
addHistoryLog(`创建鼠标事件失败: ${error2.message}`, 'error');
return null;
}
}
}
// 查找并点击目标用户
function findAndClickTargetUser() {
if (!userConfig.enableTargetUser || !userConfig.targetUsername) {
updateUserStatus('配置错误', false);
return false;
}
if (currentState !== 'searching') {
return false;
}
addHistoryLog(`查找目标用户: ${userConfig.targetUsername}`, 'info');
updateUserStatus('寻找中', null);
const userElements = document.querySelectorAll('.item-header-name-vL_79m');
let targetElement = null;
for (let element of userElements) {
if (element.textContent.trim() === userConfig.targetUsername) {
targetElement = element;
break;
}
}
if (targetElement) {
addHistoryLog(`找到目标用户: ${userConfig.targetUsername}`, 'success');
updateUserStatus('已找到', true);
stopChatObserver();
let clickSuccess = false;
if (userConfig.clickMethod === 'direct') {
try {
targetElement.click();
addHistoryLog('使用直接点击方法成功', 'success');
clickSuccess = true;
} catch (error) {
addHistoryLog(`直接点击失败: ${error.message}`, 'error');
}
} else {
try {
const clickEvent = createSafeMouseEvent('click');
if (clickEvent) {
targetElement.dispatchEvent(clickEvent);
addHistoryLog('使用事件触发方法成功', 'success');
clickSuccess = true;
} else {
targetElement.click();
addHistoryLog('事件创建失败,使用直接点击成功', 'success');
clickSuccess = true;
}
} catch (error) {
addHistoryLog(`事件触发失败: ${error.message}`, 'error');
}
}
if (clickSuccess) {
currentState = 'found';
// 点击用户后,等待页面加载完成
waitForPageLoad().then(() => {
addHistoryLog('页面加载完成,开始查找聊天输入框', 'info');
tryFindChatInput();
}).catch(error => {
addHistoryLog(`等待页面加载超时: ${error.message}`, 'error');
// 即使超时也尝试继续
tryFindChatInput();
});
return true;
} else {
try {
let clickableParent = targetElement;
for (let i = 0; i < 5; i++) {
clickableParent = clickableParent.parentElement;
if (!clickableParent) break;
const style = window.getComputedStyle(clickableParent);
if (style.cursor === 'pointer' || clickableParent.onclick) {
clickableParent.click();
addHistoryLog('通过父元素点击成功', 'success');
currentState = 'found';
// 点击用户后,等待页面加载完成
waitForPageLoad().then(() => {
addHistoryLog('页面加载完成,开始查找聊天输入框', 'info');
tryFindChatInput();
}).catch(error => {
addHistoryLog(`等待页面加载超时: ${error.message}`, 'error');
tryFindChatInput();
});
return true;
}
}
} catch (error) {
addHistoryLog(`父元素点击也失败: ${error.message}`, 'error');
}
updateUserStatus('点击失败', false);
return false;
}
} else {
addHistoryLog(`未找到目标用户: ${userConfig.targetUsername}`, 'warn');
updateUserStatus('寻找中', null);
return false;
}
}
// 等待页面加载完成
function waitForPageLoad() {
return new Promise((resolve, reject) => {
const timeout = setTimeout(() => {
reject(new Error(`页面加载等待超时 (${userConfig.pageLoadWaitTime}ms)`));
}, userConfig.pageLoadWaitTime);
// 检查页面是否已经加载完成
if (document.readyState === 'complete') {
clearTimeout(timeout);
resolve();
return;
}
// 监听页面加载完成事件
window.addEventListener('load', function onLoad() {
clearTimeout(timeout);
window.removeEventListener('load', onLoad);
resolve();
});
// 同时检查DOM是否已经稳定(没有频繁的DOM变化)
let checkCount = 0;
const maxChecks = userConfig.pageLoadWaitTime / 100;
const checkInterval = setInterval(() => {
checkCount++;
// 检查是否有聊天输入框出现
const chatInput = document.querySelector('.chat-input-dccKiL');
if (chatInput) {
clearTimeout(timeout);
clearInterval(checkInterval);
resolve();
return;
}
// 检查是否超时
if (checkCount >= maxChecks) {
clearTimeout(timeout);
clearInterval(checkInterval);
reject(new Error('页面DOM变化检查超时'));
}
}, 100);
});
}
// 发送消息函数
async function sendMessage() {
if (isProcessing) {
addHistoryLog('已有任务正在进行中', 'error');
return;
}
const lastSentDate = GM_getValue('lastSentDate', '');
const today = new Date().toDateString();
if (lastSentDate === today) {
addHistoryLog('今天已经发送过消息', 'info');
return;
}
isProcessing = true;
retryCount = 0;
currentState = 'idle';
updateRetryCount();
addHistoryLog('开始发送流程...', 'info');
executeSendProcess();
}
// 执行发送流程 - 修复版本
async function executeSendProcess() {
retryCount++;
updateRetryCount();
if (retryCount > userConfig.maxRetryCount) {
addHistoryLog(`已达到最大重试次数 (${userConfig.maxRetryCount})`, 'error');
isProcessing = false;
currentState = 'idle';
stopChatObserver();
return;
}
addHistoryLog(`尝试发送 (${retryCount}/${userConfig.maxRetryCount})`, 'info');
if (userConfig.enableTargetUser && userConfig.targetUsername) {
currentState = 'searching';
const searchTimeoutId = setTimeout(() => {
if (currentState === 'searching') {
addHistoryLog('用户查找超时', 'error');
updateUserStatus('查找超时', false);
stopChatObserver();
setTimeout(executeSendProcess, 2000);
}
}, userConfig.userSearchTimeout);
initChatObserver();
const found = findAndClickTargetUser();
if (!found) {
// 用户查找失败,观察器会继续工作
// 超时后会通过上面的setTimeout处理
}
} else {
setTimeout(tryFindChatInput, 1000);
}
}
// 尝试查找聊天输入框并发送消息
async function tryFindChatInput() {
// 先清除之前的检查计时器
if (chatInputCheckTimer) {
clearTimeout(chatInputCheckTimer);
}
const input = document.querySelector('.chat-input-dccKiL');
if (input) {
addHistoryLog('找到聊天输入框', 'info');
let messageToSend;
try {
messageToSend = await getMessageContent();
addHistoryLog('消息内容准备完成', 'success');
} catch (error) {
addHistoryLog(`消息获取失败: ${error.message}`, 'error');
messageToSend = `${userConfig.baseMessage} | 消息获取失败~`;
}
currentState = 'sending';
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) {
addHistoryLog('正在发送消息...', 'info');
sendBtn.click();
setTimeout(() => {
addHistoryLog('消息发送成功!', 'success');
const today = new Date().toDateString();
GM_setValue('lastSentDate', today);
updateStatus(true);
isProcessing = false;
currentState = 'idle';
stopChatObserver();
if (typeof GM_notification !== 'undefined') {
try {
GM_notification({
title: '抖音续火助手',
text: '续火消息发送成功!',
timeout: 3000
});
} catch (e) {
GM_notification('续火消息发送成功!', '抖音续火助手');
}
}
}, 1000);
} else {
addHistoryLog('发送按钮不可用', 'error');
setTimeout(executeSendProcess, 2000);
}
}, 500);
} else {
addHistoryLog('未找到输入框,继续查找中...', 'info');
// 持续检查聊天输入框,直到找到或超时
chatInputCheckTimer = setTimeout(() => {
tryFindChatInput();
}, userConfig.chatInputCheckInterval);
}
}
// 获取消息内容
async function getMessageContent() {
let customMessage = userConfig.customMessage || userConfig.baseMessage;
let hitokotoContent = '';
if (userConfig.useHitokoto) {
try {
addHistoryLog('正在获取一言内容...', 'info');
hitokotoContent = await getHitokoto();
addHistoryLog('一言内容获取成功', 'success');
} catch (error) {
addHistoryLog(`一言获取失败: ${error.message}`, 'error');
hitokotoContent = '一言获取失败~';
}
}
let txtApiContent = '';
if (userConfig.useTxtApi) {
try {
addHistoryLog('正在获取TXTAPI内容...', 'info');
txtApiContent = await getTxtApiContent();
addHistoryLog('TXTAPI内容获取成功', 'success');
} catch (error) {
addHistoryLog(`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 || '');
let fromFormatted = '';
if (data.from) {
fromFormatted = userConfig.fromFormat.replace(/{from}/g, data.from);
}
result = result.replace(/{from}/g, fromFormatted);
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') {
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('手动文本解析失败'));
}
}
});
}
// 解析时间字符串为日期对象
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) {
addHistoryLog(`检测到今日未发送且已过${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);
addHistoryLog('已清空发送记录,准备发送新消息', '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', []);
addHistoryLog('发送记录已清空', 'info');
updateStatus(false);
retryCount = 0;
updateRetryCount();
updateHitokotoStatus('未获取');
updateTxtApiStatus('未获取');
updateUserStatus(userConfig.enableTargetUser ? `已启用: ${userConfig.targetUsername}` : '未启用');
stopChatObserver();
if (chatInputCheckTimer) {
clearTimeout(chatInputCheckTimer);
}
}
// 重置所有配置
function resetAllConfig() {
if (typeof GM_listValues !== 'undefined' && typeof GM_deleteValue !== 'undefined') {
try {
const values = GM_listValues();
values.forEach(key => {
GM_deleteValue(key);
});
} catch (e) {
GM_setValue('lastSentDate', '');
GM_setValue('userConfig', '');
GM_setValue('txtApiManualSentIndexes', []);
GM_setValue('historyLogs', []);
}
} else {
GM_setValue('lastSentDate', '');
GM_setValue('userConfig', '');
GM_setValue('txtApiManualSentIndexes', []);
GM_setValue('historyLogs', []);
}
initConfig();
addHistoryLog('所有配置已重置', 'info');
updateStatus(false);
retryCount = 0;
updateRetryCount();
updateHitokotoStatus('未获取');
updateTxtApiStatus('未获取');
updateUserStatus('未启用');
stopChatObserver();
if (chatInputCheckTimer) {
clearTimeout(chatInputCheckTimer);
}
if (typeof GM_notification !== 'undefined') {
try {
GM_notification({
title: '抖音续火助手',
text: '所有配置已重置!',
timeout: 3000
});
} catch (e) {
GM_notification('所有配置已重置!', '抖音续火助手');
}
}
}
// 初始化函数
function init() {
isScriptCat = detectScriptCat();
initConfig();
createControlPanel();
const lastSentDate = GM_getValue('lastSentDate', '');
const today = new Date().toDateString();
const isSentToday = lastSentDate === today;
updateStatus(isSentToday);
updateUserStatus(userConfig.enableTargetUser ? `已启用: ${userConfig.targetUsername}` : '未启用');
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('历史日志', showHistoryPanel);
GM_registerMenuCommand('清空发送记录', clearData);
GM_registerMenuCommand('重置所有配置', resetAllConfig);
} catch (e) {
addHistoryLog('菜单命令注册失败,使用面板控制', 'error');
}
}
addHistoryLog('抖音续火助手已启动', '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) {
addHistoryLog('定时任务触发发送', 'info');
sendMessage();
}
}
}, 1000);
}
if (document.readyState === 'loading') {
document.addEventListener('DOMContentLoaded', init);
} else {
init();
}
})();