您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Advanced bulk message deletion tool for Snapchat Web. Features theme options, persistent settings, automatic message detection, efficient deletion, auto-scroll, configurable deletion order, and a sleek, modern, minimizable interface. Operates by simulating user interactions.
// ==UserScript== // @name Snapchat Chat Bulk Delete // @namespace https://github.com/mathe00/snapchat-chat-bulk-delete-ts // @version 3.1.5 // @description Advanced bulk message deletion tool for Snapchat Web. Features theme options, persistent settings, automatic message detection, efficient deletion, auto-scroll, configurable deletion order, and a sleek, modern, minimizable interface. Operates by simulating user interactions. // @author mathe00 (Ported by T3 Chat) // @match https://web.snapchat.com/* // @match https://www.snapchat.com/web/* // @icon  // @grant GM_addStyle // @grant GM_setValue // @grant GM_getValue // @run-at document-idle // @license MIT // @compatible firefox Violentmonkey Tampermonkey // @compatible chrome Violentmonkey Tampermonkey // @compatible edge Violentmonkey Tampermonkey // @supportURL https://github.com/mathe00/snapchat-chat-bulk-delete-ts/issues // @homepageURL https://github.com/mathe00/snapchat-chat-bulk-delete-ts // ==/UserScript== (async function() { 'use strict'; const INJECTED_LOGO_BASE64 = ""; // src/config.ts var CONFIG = { SCRIPT_NAME: "Snapchat Chat Bulk Delete", VERSION: "3.1.5", // Synchronize with package.json and build.mjs // ICON_BASE64_DATA_URL is removed, will be injected by build script STORAGE_KEYS: { REVERSE_ORDER: "scbd_reverse_order_v3_1_5", AUTO_SCROLL: "scbd_auto_scroll_v3_1_5", UI_POSITION: "scbd_ui_position_v3_1_5", THEME_PREFERENCE: "scbd_theme_preference_v3_1_5", PANEL_MINIMIZED: "scbd_panel_minimized_v3_1_5" }, AUTO_SCROLL_DELAY: 5e3, MESSAGE_HOVER_DELAY: 300, POST_DELETE_DELAY: 50, MAX_SCROLL_ATTEMPTS: 5, SCROLL_INTERVAL: 200, LOAD_WAIT_AFTER_SCROLL: 1e3, UI_UPDATE_INTERVAL: 1e3, SELECTORS: { MESSAGE_CONTAINER: ".T1yt2", // Example, verify if still valid MESSAGE_HEADER_SELECTOR: "header", USER_MESSAGE_IDENTIFIER_ELEMENT: "header .nonIntl", // Example MESSAGE_HOVER_TARGET_1: ".KB4Aq", // Example MESSAGE_HOVER_TARGET_2: '[role="button"]', // Example CONTEXT_MENU: ".gFryU", // Example DELETE_BUTTON_CANDIDATE: "button.NcaQH", // Example DELETE_BUTTON_SVG_PATH_D: "9.75 4.624", // Example CHAT_SCROLL_CONTAINER_PARENT_SELECTOR: ".T1yt2", // Example SNAPCHAT_ROOT_THEME_DARK: ':root[theme="dark"]', SNAPCHAT_ROOT_THEME_LIGHT: ':root[theme="light"]' }, THEME_OPTIONS: ["auto", "light", "dark"], DEFAULT_THEME: "auto" }; // src/state.ts var STATE = { observerActive: false, reverseOrder: true, autoScrollEnabled: false, lastDeleteTime: 0, isScrolling: false, isDragging: false, uiPosition: { x: 20, y: 20 }, processedButtons: /* @__PURE__ */ new WeakSet(), processedMessages: /* @__PURE__ */ new WeakSet(), domObserver: null, snapchatThemeObserver: null, autoScrollIntervalId: null, displayUpdateIntervalId: null, ui: { container: null, mainControlsSection: null, toggleButton: null, counterDisplay: null, settingsSection: null, orderToggleButton: null, autoScrollToggleButton: null, themeToggleButton: null, infoSection: null, infoLogoImg: null, dragHandle: null, minimizeButton: null, isMinimized: false, isInfoVisible: false, isSettingsVisible: false, versionDisplay: null, currentThemeDisplay: null }, themePreference: CONFIG.DEFAULT_THEME, currentAppliedTheme: "light" }; function handleError(error, context = "General") { console.error( `[${CONFIG.SCRIPT_NAME} Error] ${context}: ${error.message}`, error.stack || "" ); } // src/domUtils.ts function simulateHover(element, enter) { if (!element) return; const eventTypes = enter ? ["mouseenter", "mouseover"] : ["mouseleave", "mouseout"]; eventTypes.forEach((type) => { const event = new MouseEvent(type, { bubbles: true, cancelable: true, view: window }); element.dispatchEvent(event); let parent = element.parentElement; while (parent) { parent.dispatchEvent( new MouseEvent(type, { bubbles: true, cancelable: true, view: window }) ); parent = parent.parentElement; } }); } function isDeleteButton(button) { try { if (!button.closest(CONFIG.SELECTORS.CONTEXT_MENU)) return false; const menu = button.closest(CONFIG.SELECTORS.CONTEXT_MENU); if (!menu) return false; const buttonsInMenu = menu.querySelectorAll( CONFIG.SELECTORS.DELETE_BUTTON_CANDIDATE ); if (buttonsInMenu[buttonsInMenu.length - 1] !== button) return false; const svg = button.querySelector("svg"); if (!svg) return false; const path = svg.querySelector("path"); if (!path) return false; const pathData = path.getAttribute("d"); return pathData ? pathData.includes(CONFIG.SELECTORS.DELETE_BUTTON_SVG_PATH_D) : false; } catch (error) { handleError(error, "isDeleteButton check"); return false; } } function getSnapchatPageTheme() { if (document.documentElement.matches(CONFIG.SELECTORS.SNAPCHAT_ROOT_THEME_DARK)) { return "dark"; } if (document.documentElement.matches(CONFIG.SELECTORS.SNAPCHAT_ROOT_THEME_LIGHT)) { return "light"; } return "light"; } // src/gmApi.ts function setLogoFromBase64(imgElement, base64DataUrl, scriptName, handleErrorFunction) { if (!(imgElement instanceof HTMLImageElement)) { handleErrorFunction( new Error("Invalid element passed to setLogoFromBase64."), "Logo Init" ); return; } imgElement.onerror = function() { handleErrorFunction( new Error( `Image failed to load from base64 data URL. Check if the data URL is correct.` ), `Logo img.onerror (base64)` ); this.alt = "Error loading logo from base64"; }; try { imgElement.src = base64DataUrl; } catch (error) { handleErrorFunction(error, `Logo Set Function (Outer Catch)`); imgElement.alt = "Exception loading logo from base64"; } } // src/themeManager.ts function applyPanelTheme(theme) { if (!STATE.ui.container) return; STATE.ui.container.classList.remove("theme-light", "theme-dark"); STATE.ui.container.classList.add(`theme-${theme}`); STATE.currentAppliedTheme = theme; if (updateUICore.currentThemeDisplay) updateUICore.currentThemeDisplay(); } function cycleThemePreference() { const currentIndex = CONFIG.THEME_OPTIONS.indexOf(STATE.themePreference); const nextIndex = (currentIndex + 1) % CONFIG.THEME_OPTIONS.length; STATE.themePreference = CONFIG.THEME_OPTIONS[nextIndex]; if (typeof GM_setValue === "function") { try { GM_setValue(CONFIG.STORAGE_KEYS.THEME_PREFERENCE, STATE.themePreference); } catch (e) { handleError(e, "Error saving theme preference"); } } else { handleError(new Error("GM_setValue is not a function in cycleThemePreference"), "GM_setValue Check"); } if (updateUICore.settings?.themeToggleButton) updateUICore.settings.themeToggleButton(); if (STATE.themePreference === "auto") { applyPanelTheme(getSnapchatPageTheme()); } else { applyPanelTheme(STATE.themePreference); } } function setupSnapchatThemeObserver() { if (STATE.snapchatThemeObserver) STATE.snapchatThemeObserver.disconnect(); STATE.snapchatThemeObserver = new MutationObserver(() => { if (STATE.themePreference === "auto") { applyPanelTheme(getSnapchatPageTheme()); } }); STATE.snapchatThemeObserver.observe(document.documentElement, { attributes: true, attributeFilter: ["theme", "class"] }); } // src/ui.ts var deleteCounterInstance; function setUIDeleteCounterInstance(instance) { deleteCounterInstance = instance; } var updateUICore = { toggleButtonState: () => { if (!STATE.ui.toggleButton) return; const b = STATE.ui.toggleButton; const isActive = STATE.observerActive; b.textContent = isActive ? "PAUSE" : "START"; b.classList.toggle("active", isActive); b.classList.toggle("inactive", !isActive); const pulseStyleId = "scbd-pulse-style"; let pulseElement = document.getElementById(pulseStyleId); if (isActive && !pulseElement) { const s = document.createElement("style"); s.id = pulseStyleId; s.textContent = `@keyframes scbdPulse{0%{box-shadow:0 0 0 0 rgba(255,69,0,.7)}70%{box-shadow:0 0 0 10px rgba(255,69,0,0)}100%{box-shadow:0 0 0 0 rgba(255,69,0,0)}} .scbd-button.active{animation:scbdPulse 2s infinite}`; document.head.appendChild(s); } else if (!isActive && pulseElement) { pulseElement.remove(); } }, uiPosition: () => { if (STATE.ui.container && !STATE.ui.isMinimized) { STATE.ui.container.style.transform = `translate(${STATE.uiPosition.x}px, ${STATE.uiPosition.y}px)`; } }, panelVisibility: () => { if (!STATE.ui.container || !STATE.ui.minimizeButton) return; const isMinimized = STATE.ui.isMinimized; STATE.ui.container.classList.toggle("minimized", isMinimized); STATE.ui.minimizeButton.innerHTML = isMinimized ? "❐" : "✕"; STATE.ui.minimizeButton.title = isMinimized ? "Restore Panel" : "Minimize Panel"; if (isMinimized) { if (STATE.ui.infoSection) STATE.ui.infoSection.style.display = "none"; if (STATE.ui.settingsSection) STATE.ui.settingsSection.style.display = "none"; STATE.ui.isInfoVisible = false; STATE.ui.isSettingsVisible = false; if (STATE.ui.container) STATE.ui.container.style.transform = ""; } else { if (STATE.ui.infoSection) STATE.ui.infoSection.style.display = STATE.ui.isInfoVisible ? "block" : "none"; if (STATE.ui.settingsSection) STATE.ui.settingsSection.style.display = STATE.ui.isSettingsVisible ? "block" : "none"; updateUICore.uiPosition(); } if (typeof GM_setValue === "function") { try { GM_setValue(CONFIG.STORAGE_KEYS.PANEL_MINIMIZED, isMinimized); } catch (e) { handleError(e, "Error calling GM_setValue in panelVisibility"); } } else { handleError( new Error("GM_setValue is not a function in panelVisibility"), "GM_setValue Check" ); } }, toggleSection: (sectionName) => { const sectionElement = STATE.ui[`${sectionName}Section`]; const visibilityFlag = `is${sectionName.charAt(0).toUpperCase() + sectionName.slice(1)}Visible`; if (!sectionElement || STATE.ui.isMinimized) return; STATE.ui[visibilityFlag] = !STATE.ui[visibilityFlag]; sectionElement.style.display = STATE.ui[visibilityFlag] ? "block" : "none"; if (STATE.ui[visibilityFlag]) { const otherSectionName = sectionName === "info" ? "settings" : "info"; const otherSectionElement = STATE.ui[`${otherSectionName}Section`]; const otherVisibilityFlag = `is${otherSectionName.charAt(0).toUpperCase() + otherSectionName.slice(1)}Visible`; if (otherSectionElement && STATE.ui[otherVisibilityFlag]) { STATE.ui[otherVisibilityFlag] = false; otherSectionElement.style.display = "none"; } } }, settings: { orderToggleButton: () => { if (!STATE.ui.orderToggleButton) return; STATE.ui.orderToggleButton.textContent = STATE.reverseOrder ? "Newest → Oldest" : "Oldest → Newest"; STATE.ui.orderToggleButton.disabled = STATE.observerActive; }, autoScrollToggleButton: () => { if (!STATE.ui.autoScrollToggleButton) return; STATE.ui.autoScrollToggleButton.classList.toggle( "enabled", STATE.autoScrollEnabled ); STATE.ui.autoScrollToggleButton.setAttribute( "aria-checked", String(STATE.autoScrollEnabled) ); STATE.ui.autoScrollToggleButton.disabled = STATE.observerActive; }, themeToggleButton: () => { if (!STATE.ui.themeToggleButton) return; const themeName = STATE.themePreference.charAt(0).toUpperCase() + STATE.themePreference.slice(1); STATE.ui.themeToggleButton.title = `Theme: ${themeName}`; if (updateUICore.currentThemeDisplay) updateUICore.currentThemeDisplay(); } }, versionDisplay: () => { if (STATE.ui.versionDisplay) STATE.ui.versionDisplay.textContent = `v${CONFIG.VERSION}`; }, currentThemeDisplay: () => { if (STATE.ui.currentThemeDisplay) { const themeText = STATE.themePreference === "auto" ? `Auto (${STATE.currentAppliedTheme})` : STATE.themePreference.charAt(0).toUpperCase() + STATE.themePreference.slice(1); STATE.ui.currentThemeDisplay.textContent = `Theme: ${themeText}`; } }, updateInfoSectionContent: () => { if (STATE.ui.infoSection) { STATE.ui.infoSection.innerHTML = `<h4>Information</h4> <p><strong>${CONFIG.SCRIPT_NAME} v${CONFIG.VERSION}</strong></p> <p>Operates by simulating UI clicks. No network requests are modified. User's own messages are identified based on their unique appearance in the chat.</p> <p>Settings (theme, order, auto-scroll) are saved.</p> <div class="scbd-info-logo-container"> <img id="scbd-info-logo-img" alt="Script Logo" class="scbd-info-logo"/> </div>`; STATE.ui.infoLogoImg = STATE.ui.infoSection.querySelector( "#scbd-info-logo-img" ); if (STATE.ui.infoLogoImg) { setLogoFromBase64( STATE.ui.infoLogoImg, typeof INJECTED_LOGO_BASE64 !== "undefined" ? INJECTED_LOGO_BASE64 : "", CONFIG.SCRIPT_NAME, handleError ); } else { handleError( new Error( "Could not find #scbd-info-logo-img after setting innerHTML." ), "Info Logo Setup" ); } } } }; function createPanelUI() { STATE.ui.container = document.createElement("div"); STATE.ui.container.id = "scbd-panel-container"; STATE.ui.dragHandle = document.createElement("div"); STATE.ui.dragHandle.id = "scbd-drag-handle"; STATE.ui.container.appendChild(STATE.ui.dragHandle); const header = document.createElement("div"); header.className = "scbd-header"; const logoImg = document.createElement("img"); logoImg.alt = "Logo"; logoImg.className = "scbd-logo"; setLogoFromBase64( logoImg, typeof INJECTED_LOGO_BASE64 !== "undefined" ? INJECTED_LOGO_BASE64 : "", CONFIG.SCRIPT_NAME, handleError ); const title = document.createElement("span"); title.className = "scbd-title"; title.textContent = CONFIG.SCRIPT_NAME; header.appendChild(logoImg); header.appendChild(title); const headerControls = document.createElement("div"); headerControls.className = "scbd-header-controls"; STATE.ui.themeToggleButton = document.createElement("button"); STATE.ui.themeToggleButton.className = "scbd-icon-button"; STATE.ui.themeToggleButton.innerHTML = "🎨"; STATE.ui.themeToggleButton.title = "Cycle Theme"; STATE.ui.themeToggleButton.addEventListener("click", (e) => { e.stopPropagation(); cycleThemePreference(); }); headerControls.appendChild(STATE.ui.themeToggleButton); const settingsButton = document.createElement("button"); settingsButton.className = "scbd-icon-button"; settingsButton.innerHTML = "⚙️"; settingsButton.title = "Settings"; settingsButton.addEventListener("click", (e) => { e.stopPropagation(); updateUICore.toggleSection("settings"); }); headerControls.appendChild(settingsButton); const infoButton = document.createElement("button"); infoButton.className = "scbd-icon-button"; infoButton.innerHTML = "ℹ️"; infoButton.title = "Information"; infoButton.addEventListener("click", (e) => { e.stopPropagation(); updateUICore.toggleSection("info"); }); headerControls.appendChild(infoButton); STATE.ui.minimizeButton = document.createElement("button"); STATE.ui.minimizeButton.className = "scbd-icon-button scbd-minimize-btn"; STATE.ui.minimizeButton.innerHTML = "✕"; STATE.ui.minimizeButton.addEventListener("click", (e) => { e.stopPropagation(); STATE.ui.isMinimized = !STATE.ui.isMinimized; updateUICore.panelVisibility(); }); headerControls.appendChild(STATE.ui.minimizeButton); header.appendChild(headerControls); STATE.ui.container.appendChild(header); const contentArea = document.createElement("div"); contentArea.className = "scbd-content-area"; STATE.ui.mainControlsSection = document.createElement("div"); STATE.ui.mainControlsSection.id = "scbd-main-controls"; STATE.ui.toggleButton = document.createElement("button"); STATE.ui.toggleButton.id = "scbd-toggle-button"; STATE.ui.toggleButton.className = "scbd-button"; STATE.ui.mainControlsSection.appendChild(STATE.ui.toggleButton); STATE.ui.counterDisplay = document.createElement("div"); STATE.ui.counterDisplay.id = "scbd-counter-display"; STATE.ui.mainControlsSection.appendChild(STATE.ui.counterDisplay); contentArea.appendChild(STATE.ui.mainControlsSection); STATE.ui.settingsSection = document.createElement("div"); STATE.ui.settingsSection.id = "scbd-settings-section"; STATE.ui.settingsSection.style.display = "none"; const settingsTitle = document.createElement("h4"); settingsTitle.textContent = "Settings"; STATE.ui.settingsSection.appendChild(settingsTitle); const orderDiv = document.createElement("div"); orderDiv.className = "scbd-setting-row"; orderDiv.innerHTML = `<span>Order:</span> `; STATE.ui.orderToggleButton = document.createElement("button"); STATE.ui.orderToggleButton.className = "scbd-setting-button"; STATE.ui.orderToggleButton.addEventListener("click", () => { if (STATE.observerActive) return; STATE.reverseOrder = !STATE.reverseOrder; if (typeof GM_setValue === "function") { try { GM_setValue(CONFIG.STORAGE_KEYS.REVERSE_ORDER, STATE.reverseOrder); } catch (e) { handleError(e, "Error saving order preference"); } } else { handleError( new Error("GM_setValue is not a function for order toggle"), "GM_setValue Check" ); } if (updateUICore.settings?.orderToggleButton) updateUICore.settings.orderToggleButton(); if (deleteCounterInstance) deleteCounterInstance.updateDisplay(); }); orderDiv.appendChild(STATE.ui.orderToggleButton); STATE.ui.settingsSection.appendChild(orderDiv); const autoScrollDiv = document.createElement("div"); autoScrollDiv.className = "scbd-setting-row"; const autoScrollLabel = document.createElement("span"); autoScrollLabel.textContent = "Auto-Scroll:"; autoScrollDiv.appendChild(autoScrollLabel); STATE.ui.autoScrollToggleButton = document.createElement("button"); STATE.ui.autoScrollToggleButton.className = "scbd-toggle-switch"; STATE.ui.autoScrollToggleButton.setAttribute("role", "switch"); STATE.ui.autoScrollToggleButton.setAttribute( "aria-checked", String(STATE.autoScrollEnabled) ); STATE.ui.autoScrollToggleButton.addEventListener("click", () => { if (STATE.observerActive) return; STATE.autoScrollEnabled = !STATE.autoScrollEnabled; if (typeof GM_setValue === "function") { try { GM_setValue(CONFIG.STORAGE_KEYS.AUTO_SCROLL, STATE.autoScrollEnabled); } catch (e) { handleError(e, "Error saving auto-scroll preference"); } } else { handleError( new Error("GM_setValue is not a function for auto-scroll"), "GM_setValue Check" ); } if (updateUICore.settings?.autoScrollToggleButton) updateUICore.settings.autoScrollToggleButton(); if (deleteCounterInstance) deleteCounterInstance.updateDisplay(); }); autoScrollDiv.appendChild(STATE.ui.autoScrollToggleButton); STATE.ui.settingsSection.appendChild(autoScrollDiv); contentArea.appendChild(STATE.ui.settingsSection); STATE.ui.infoSection = document.createElement("div"); STATE.ui.infoSection.id = "scbd-info-section"; STATE.ui.infoSection.style.display = "none"; if (updateUICore.updateInfoSectionContent) updateUICore.updateInfoSectionContent(); contentArea.appendChild(STATE.ui.infoSection); STATE.ui.container.appendChild(contentArea); const footer = document.createElement("div"); footer.className = "scbd-footer"; STATE.ui.currentThemeDisplay = document.createElement("span"); STATE.ui.currentThemeDisplay.id = "scbd-current-theme"; footer.appendChild(STATE.ui.currentThemeDisplay); STATE.ui.versionDisplay = document.createElement("span"); STATE.ui.versionDisplay.id = "scbd-version"; footer.appendChild(STATE.ui.versionDisplay); STATE.ui.container.appendChild(footer); document.body.appendChild(STATE.ui.container); STATE.ui.container.addEventListener("click", (e) => { if (STATE.ui.isMinimized && e.target === STATE.ui.container) { STATE.ui.isMinimized = false; updateUICore.panelVisibility(); } }); setupDragAndDrop(); updateUICore.toggleButtonState(); if (updateUICore.settings) { updateUICore.settings.orderToggleButton(); updateUICore.settings.autoScrollToggleButton(); updateUICore.settings.themeToggleButton(); } if (updateUICore.versionDisplay) updateUICore.versionDisplay(); if (updateUICore.currentThemeDisplay) updateUICore.currentThemeDisplay(); if (deleteCounterInstance) deleteCounterInstance.updateDisplay(); if (STATE.themePreference === "auto") { applyPanelTheme(getSnapchatPageTheme()); } else { applyPanelTheme(STATE.themePreference); } updateUICore.panelVisibility(); } function setupDragAndDrop() { let initialX, initialY; function dragStart(e) { if (e.target === STATE.ui.dragHandle && STATE.ui.container && !STATE.ui.isMinimized) { initialX = e.clientX - STATE.uiPosition.x; initialY = e.clientY - STATE.uiPosition.y; STATE.isDragging = true; document.body.classList.add("scbd-dragging"); if (STATE.ui.container) STATE.ui.container.classList.add("scbd-dragging-panel"); document.addEventListener("mousemove", dragMove); document.addEventListener("mouseup", dragEnd); } } function dragMove(e) { if (STATE.isDragging && STATE.ui.container) { e.preventDefault(); const newX = e.clientX - initialX; const newY = e.clientY - initialY; const rect = STATE.ui.container.getBoundingClientRect(); STATE.uiPosition.x = Math.min( window.innerWidth - rect.width - 5, Math.max(5, newX) ); STATE.uiPosition.y = Math.min( window.innerHeight - rect.height - 5, Math.max(5, newY) ); updateUICore.uiPosition(); } } function dragEnd() { if (!STATE.isDragging) return; STATE.isDragging = false; document.body.classList.remove("scbd-dragging"); if (STATE.ui.container) { STATE.ui.container.classList.remove("scbd-dragging-panel"); } document.removeEventListener("mousemove", dragMove); document.removeEventListener("mouseup", dragEnd); if (typeof GM_setValue === "function") { try { GM_setValue(CONFIG.STORAGE_KEYS.UI_POSITION, STATE.uiPosition); } catch (e) { handleError(e, "Error calling GM_setValue in dragEnd"); } } else { handleError( new Error("GM_setValue is not a function in dragEnd"), "GM_setValue Check" ); } } if (STATE.ui.container) { STATE.ui.container.addEventListener("mousedown", dragStart); } } function injectStyles() { if (typeof GM_addStyle === "function") { GM_addStyle(` body.scbd-dragging, body.scbd-dragging * { user-select: none !important; cursor: grabbing !important; } #scbd-panel-container.scbd-dragging-panel, #scbd-panel-container.scbd-dragging-panel * { cursor: grabbing !important; } #scbd-panel-container { position: fixed; top: 0; left: 0; z-index: 100000; border-radius: 10px; box-shadow: 0 8px 25px rgba(0,0,0,0.2); display: flex; flex-direction: column; gap: 8px; font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif; min-width: 260px; max-width: 320px; padding: 10px; transition: transform 0.25s ease-out, opacity 0.25s ease-out, width 0.25s ease-out, height 0.25s ease-out, bottom 0.25s ease-out, right 0.25s ease-out; overflow: hidden; } #scbd-panel-container.theme-light { background-color: rgba(255,255,255,0.85); color: #333; border: 1px solid rgba(0,0,0,0.1); } #scbd-panel-container.theme-dark { background-color: rgba(40,40,40,0.9); color: #eee; border: 1px solid rgba(255,255,255,0.15); } #scbd-panel-container.minimized { width: 44px !important; height: 44px !important; padding: 0 !important; border-radius: 50%; opacity: 0.7; bottom: 20px; right: 20px; top: auto !important; left: auto !important; transform: none !important; cursor: pointer; display: flex; align-items: center; justify-content: center; } #scbd-panel-container.minimized:hover { opacity: 1; } #scbd-panel-container.minimized .scbd-header-controls .scbd-minimize-btn { font-size: 20px; } #scbd-panel-container.minimized > *:not(.scbd-header), #scbd-panel-container.minimized .scbd-header > *:not(.scbd-header-controls), #scbd-panel-container.minimized .scbd-header-controls > *:not(.scbd-minimize-btn) { display: none !important; } #scbd-drag-handle { position: absolute; top: 0; left: 0; right: 0; height: 30px; cursor: move; z-index: 1; } .scbd-header { display: flex; align-items: center; gap: 8px; padding-bottom: 5px; border-bottom: 1px solid; position: relative; z-index: 2; } .theme-light .scbd-header { border-bottom-color: rgba(0,0,0,0.1); } .theme-dark .scbd-header { border-bottom-color: rgba(255,255,255,0.1); } .scbd-logo { width: 18px; height: 18px; object-fit: contain; display: inline-block; vertical-align: middle; } .scbd-title { font-weight: bold; font-size: 13px; flex-grow: 1; white-space: nowrap; overflow: hidden; text-overflow: ellipsis; } .scbd-header-controls { display: flex; align-items: center; gap: 4px; margin-left: auto; } .scbd-icon-button { background: transparent; border: none; cursor: pointer; font-size: 16px; padding: 4px; border-radius: 4px; transition: background-color 0.2s; line-height: 1; } .theme-light .scbd-icon-button { color: #555; } .theme-dark .scbd-icon-button { color: #bbb; } .theme-light .scbd-icon-button:hover { background-color: rgba(0,0,0,0.08); } .theme-dark .scbd-icon-button:hover { background-color: rgba(255,255,255,0.1); } .scbd-minimize-btn { font-weight: bold; } .scbd-content-area { display: flex; flex-direction: column; gap: 8px; } #scbd-main-controls { display: flex; flex-direction: column; gap: 8px; } #scbd-toggle-button { padding: 10px 15px; border: none; border-radius: 6px; cursor: pointer; font-weight: bold; font-size: 14px; transition: background-color 0.2s, color 0.2s, box-shadow 0.2s; } #scbd-toggle-button.inactive.theme-light { background-color: #FFFC00; color: black; box-shadow: 0 2px 4px rgba(0,0,0,0.1); } #scbd-toggle-button.inactive.theme-dark { background-color: #FFFC00; color: black; box-shadow: 0 2px 4px rgba(0,0,0,0.3); } #scbd-toggle-button.active { background-color: #FF4500; color: white; box-shadow: 0 2px 4px rgba(255,69,0,0.3); } #scbd-counter-display { font-size: 11px; text-align: center; opacity: 0.8; line-height: 1.4; } .theme-light #scbd-counter-display { color: #444; } .theme-dark #scbd-counter-display { color: #ccc; } #scbd-settings-section, #scbd-info-section { padding: 10px; border-radius: 6px; font-size: 12px; line-height: 1.5; animation: scbd-fade-in 0.3s ease-out; } .theme-light #scbd-settings-section, .theme-light #scbd-info-section { background-color: rgba(0,0,0,0.03); border: 1px solid rgba(0,0,0,0.07); } .theme-dark #scbd-settings-section, .theme-dark #scbd-info-section { background-color: rgba(255,255,255,0.05); border: 1px solid rgba(255,255,255,0.1); } #scbd-settings-section h4, #scbd-info-section h4 { margin-top: 0; margin-bottom: 8px; font-size: 13px; opacity: 0.9; } .scbd-setting-row { display: flex; justify-content: space-between; align-items: center; margin-bottom: 8px; gap: 8px; } .scbd-setting-row > span { flex-shrink: 0; font-size: 12px; opacity: 0.9;} .scbd-setting-button { font-size: 11px; padding: 5px 10px; border-radius: 5px; border: 1px solid; cursor: pointer; transition: background-color 0.2s, border-color 0.2s; } .theme-light .scbd-setting-button { background-color: #f0f0f0; border-color: #ddd; color: #333; } .theme-dark .scbd-setting-button { background-color: #383838; border-color: #555; color: #ddd; } .theme-light .scbd-setting-button:hover { background-color: #e0e0e0; border-color: #ccc; } .theme-dark .scbd-setting-button:hover { background-color: #484848; border-color: #666; } .scbd-setting-button:disabled { opacity: 0.5; cursor: not-allowed; } .scbd-toggle-switch { position: relative; display: inline-block; width: 40px; height: 20px; background-color: #ccc; border-radius: 20px; cursor: pointer; transition: background-color 0.3s ease; } .scbd-toggle-switch::before { content: ""; position: absolute; width: 16px; height: 16px; border-radius: 50%; background-color: white; top: 2px; left: 2px; transition: transform 0.3s ease; } .scbd-toggle-switch.enabled { background-color: #4CAF50; } .theme-dark .scbd-toggle-switch.enabled { background-color: #66BB6A; } .theme-dark .scbd-toggle-switch { background-color: #555; } .scbd-toggle-switch.enabled::before { transform: translateX(20px); } .scbd-toggle-switch:disabled { opacity: 0.5; cursor: not-allowed; } .scbd-toggle-switch.scrolling { animation: scbd-pulse-scroll 1s infinite alternate; } @keyframes scbd-pulse-scroll { from { box-shadow: 0 0 0 0px rgba(76, 175, 80, 0.7); } to { box-shadow: 0 0 0 5px rgba(76, 175, 80, 0); } } .scbd-info-logo-container { text-align: center; margin-top: 15px; } .scbd-info-logo { max-width: 64px; max-height: 64px; object-fit: contain; display: block; margin: 0 auto; opacity: 0.8; } .scbd-footer { display: flex; justify-content: space-between; align-items: center; padding-top: 5px; border-top: 1px solid; font-size: 10px; opacity: 0.6; } #scbd-current-theme { text-align: left; } #scbd-version { text-align: right; } .theme-light .scbd-footer { border-top-color: rgba(0,0,0,0.1); } .theme-dark .scbd-footer { border-top-color: rgba(255,255,255,0.1); } @keyframes scbd-fade-in { from { opacity: 0; transform: translateY(-5px); } to { opacity: 1; transform: translateY(0); } } `); } else { handleError( new Error("GM_addStyle is not available for injecting styles."), "injectStyles" ); } } // src/deleteCounter.ts var DeleteCountManager = class { counters; currentChatId; constructor() { this.counters = { total: 0, chats: {} }; this.currentChatId = this.extractChatId(); this.setupUrlMonitoring(); this.updateDisplay(); } extractChatId() { const match = window.location.pathname.match(/\/([^\/]+)$/); return match ? match[1] : "unknown_chat"; } setupUrlMonitoring() { setInterval(() => { const newChatId = this.extractChatId(); if (newChatId !== this.currentChatId) { this.currentChatId = newChatId; this.updateDisplay(); } }, CONFIG.UI_UPDATE_INTERVAL); } async increment() { if (!this.counters.chats[this.currentChatId]) { this.counters.chats[this.currentChatId] = 0; } this.counters.chats[this.currentChatId]++; this.counters.total++; this.updateDisplay(); } updateDisplay() { if (STATE.ui.counterDisplay) { const currentChatCount = this.counters.chats[this.currentChatId] || 0; const timeUntilScroll = STATE.autoScrollEnabled && STATE.observerActive ? Math.max( 0, Math.ceil( (CONFIG.AUTO_SCROLL_DELAY - (Date.now() - STATE.lastDeleteTime)) / 1e3 ) ) : "-"; STATE.ui.counterDisplay.innerHTML = `<div>Chat: ${currentChatCount}</div><div>Total: ${this.counters.total}</div><div>Next scroll: ${timeUntilScroll}s</div>`; } } resetCurrentChatCounter() { this.counters.chats[this.currentChatId] = 0; this.updateDisplay(); } resetAllCounters() { this.counters = { total: 0, chats: {} }; this.updateDisplay(); } }; // src/main.ts console.log(`[${CONFIG.SCRIPT_NAME}] GM_getValue type: ${typeof GM_getValue}`); console.log(`[${CONFIG.SCRIPT_NAME}] GM_setValue type: ${typeof GM_setValue}`); console.log(`[${CONFIG.SCRIPT_NAME}] GM_addStyle type: ${typeof GM_addStyle}`); console.log(`[${CONFIG.SCRIPT_NAME}] GM_xmlhttpRequest type: ${typeof GM_xmlhttpRequest}`); var deleteCounter; async function loadPreferences() { try { if (typeof GM_getValue !== "function") { handleError(new Error("GM_getValue is not available for loading preferences."), "loadPreferences"); STATE.reverseOrder = true; STATE.autoScrollEnabled = false; STATE.uiPosition = { x: 20, y: 20 }; STATE.themePreference = CONFIG.DEFAULT_THEME; STATE.ui.isMinimized = false; console.log(`[${CONFIG.SCRIPT_NAME}] GM_getValue not found, defaults set.`); return; } STATE.reverseOrder = await GM_getValue(CONFIG.STORAGE_KEYS.REVERSE_ORDER, true); STATE.autoScrollEnabled = await GM_getValue(CONFIG.STORAGE_KEYS.AUTO_SCROLL, false); const storedPosition = await GM_getValue(CONFIG.STORAGE_KEYS.UI_POSITION); STATE.uiPosition = storedPosition || { x: 20, y: 20 }; STATE.themePreference = await GM_getValue(CONFIG.STORAGE_KEYS.THEME_PREFERENCE, CONFIG.DEFAULT_THEME); STATE.ui.isMinimized = await GM_getValue(CONFIG.STORAGE_KEYS.PANEL_MINIMIZED, false); console.log(`[${CONFIG.SCRIPT_NAME}] Preferences loaded via GM_getValue.`); } catch (error) { handleError(error, "Loading preferences"); STATE.reverseOrder = true; STATE.autoScrollEnabled = false; STATE.uiPosition = { x: 20, y: 20 }; STATE.themePreference = CONFIG.DEFAULT_THEME; STATE.ui.isMinimized = false; } } async function handlePotentialDeleteButton(button) { if (STATE.processedButtons.has(button)) return; try { if (isDeleteButton(button)) { STATE.processedButtons.add(button); button.click(); STATE.lastDeleteTime = Date.now(); await deleteCounter.increment(); await new Promise((resolve) => setTimeout(resolve, CONFIG.POST_DELETE_DELAY)); } } catch (error) { handleError(error, "Processing delete button"); } } async function hoverNextMessage() { if (!STATE.observerActive || STATE.isScrolling) return; try { let messages = Array.from( document.querySelectorAll(CONFIG.SELECTORS.MESSAGE_CONTAINER) ).filter((msgContainer) => { const header = msgContainer.querySelector(CONFIG.SELECTORS.MESSAGE_HEADER_SELECTOR); if (!header) return false; const potentialSelfIdentifier = header.querySelector( CONFIG.SELECTORS.USER_MESSAGE_IDENTIFIER_ELEMENT ); const isUserMessage = !!potentialSelfIdentifier; return isUserMessage && !STATE.processedMessages.has(msgContainer); }); if (STATE.reverseOrder) { messages.reverse(); } for (const message of messages) { if (!STATE.observerActive) break; STATE.processedMessages.add(message); const hoverTargets = [ message.querySelector(CONFIG.SELECTORS.MESSAGE_HOVER_TARGET_1), message.querySelector(CONFIG.SELECTORS.MESSAGE_HOVER_TARGET_2), message ].filter(Boolean); for (const target of hoverTargets) { if (!STATE.observerActive || !target) break; simulateHover(target, true); await new Promise((resolve) => setTimeout(resolve, CONFIG.MESSAGE_HOVER_DELAY)); const menuVisible = !!document.querySelector(CONFIG.SELECTORS.CONTEXT_MENU); if (menuVisible) { await new Promise((resolve) => setTimeout(resolve, 50)); break; } else { simulateHover(target, false); await new Promise((resolve) => setTimeout(resolve, 10)); } } break; } } catch (error) { handleError(error, "Automatic message hover"); } if (STATE.observerActive) { setTimeout(hoverNextMessage, CONFIG.MESSAGE_HOVER_DELAY * 2); } } async function autoScrollChat() { if (!STATE.autoScrollEnabled || !STATE.observerActive || STATE.isScrolling) return; const now = Date.now(); if (STATE.lastDeleteTime === 0 || now - STATE.lastDeleteTime > CONFIG.AUTO_SCROLL_DELAY) { STATE.isScrolling = true; if (STATE.ui.autoScrollToggleButton) { STATE.ui.autoScrollToggleButton.classList.add("scrolling"); } const scrollContainerParent = document.querySelector( CONFIG.SELECTORS.CHAT_SCROLL_CONTAINER_PARENT_SELECTOR ); let chatContainer = null; if (scrollContainerParent) { let currentElement = scrollContainerParent; while (currentElement && !chatContainer) { const style = window.getComputedStyle(currentElement); if (style.overflowY === "scroll" || style.overflowY === "auto") { chatContainer = currentElement; break; } currentElement = currentElement.parentElement; } } if (chatContainer) { for (let i = 0; i < CONFIG.MAX_SCROLL_ATTEMPTS; i++) { chatContainer.scrollTop = 0; await new Promise((resolve) => setTimeout(resolve, CONFIG.SCROLL_INTERVAL)); } await new Promise((resolve) => setTimeout(resolve, CONFIG.LOAD_WAIT_AFTER_SCROLL)); } else { console.warn(`[${CONFIG.SCRIPT_NAME}] Chat scroll container not found.`); } if (STATE.ui.autoScrollToggleButton) { STATE.ui.autoScrollToggleButton.classList.remove("scrolling"); if (updateUICore.settings?.autoScrollToggleButton) updateUICore.settings.autoScrollToggleButton(); } STATE.isScrolling = false; STATE.lastDeleteTime = Date.now(); deleteCounter.updateDisplay(); } } function setupDomObserver() { if (STATE.domObserver) STATE.domObserver.disconnect(); STATE.domObserver = new MutationObserver((mutations) => { if (!STATE.observerActive) return; for (const mutation of mutations) { if (mutation.type === "childList") { mutation.addedNodes.forEach((node) => { if (node.nodeType === Node.ELEMENT_NODE) { const elementNode = node; if (elementNode.matches && elementNode.matches(CONFIG.SELECTORS.DELETE_BUTTON_CANDIDATE)) { handlePotentialDeleteButton(elementNode).catch((err) => handleError(err, "DOM Observer - direct match")); } elementNode.querySelectorAll(CONFIG.SELECTORS.DELETE_BUTTON_CANDIDATE).forEach((btn) => handlePotentialDeleteButton(btn).catch((err) => handleError(err, "DOM Observer - querySelectorAll"))); } }); } } }); STATE.domObserver.observe(document.body, { childList: true, subtree: true }); } function toggleDeletionProcess() { STATE.observerActive = !STATE.observerActive; if (STATE.observerActive) { STATE.lastDeleteTime = Date.now(); setupDomObserver(); hoverNextMessage().catch((err) => handleError(err, "Initial hoverNextMessage call")); if (STATE.autoScrollEnabled) { if (STATE.autoScrollIntervalId) clearInterval(STATE.autoScrollIntervalId); STATE.autoScrollIntervalId = window.setInterval(autoScrollChat, CONFIG.UI_UPDATE_INTERVAL); } if (STATE.displayUpdateIntervalId) clearInterval(STATE.displayUpdateIntervalId); STATE.displayUpdateIntervalId = window.setInterval( () => deleteCounter.updateDisplay(), CONFIG.UI_UPDATE_INTERVAL ); } else { if (STATE.domObserver) { STATE.domObserver.disconnect(); STATE.domObserver = null; } if (STATE.autoScrollIntervalId) { clearInterval(STATE.autoScrollIntervalId); STATE.autoScrollIntervalId = null; } if (STATE.displayUpdateIntervalId) { clearInterval(STATE.displayUpdateIntervalId); STATE.displayUpdateIntervalId = null; } STATE.processedMessages = /* @__PURE__ */ new WeakSet(); STATE.processedButtons = /* @__PURE__ */ new WeakSet(); } updateUICore.toggleButtonState(); if (updateUICore.settings) { updateUICore.settings.orderToggleButton(); updateUICore.settings.autoScrollToggleButton(); } deleteCounter.updateDisplay(); } async function initialize() { console.log(`[${CONFIG.SCRIPT_NAME}] Initializing v${CONFIG.VERSION}...`); try { deleteCounter = new DeleteCountManager(); setUIDeleteCounterInstance(deleteCounter); await loadPreferences(); if (typeof GM_addStyle !== "function") { handleError(new Error("GM_addStyle is not available for injecting styles."), "initialize"); } else { injectStyles(); } createPanelUI(); if (STATE.ui.toggleButton) { STATE.ui.toggleButton.addEventListener("click", toggleDeletionProcess); } else { handleError(new Error("Toggle button not found after UI creation."), "Initialization"); } setupSnapchatThemeObserver(); console.log(`[${CONFIG.SCRIPT_NAME}] Initialized successfully.`); } catch (error) { handleError(error, "Initialization"); } } if (document.readyState === "interactive" || document.readyState === "complete") { initialize().catch((err) => console.error("Initialization failed:", err)); } else { document.addEventListener("DOMContentLoaded", () => { initialize().catch((err) => console.error("Initialization failed (DOMContentLoaded):", err)); }); } })();