您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Finds and cancels shows at specific times within a given date range.
// ==UserScript== // @name Popmundo Mass Show Canceller // @namespace http://tampermonkey.net/ // @version 2.3 // @description Finds and cancels shows at specific times within a given date range. // @author Gemini // @match https://*.popmundo.com/World/Popmundo.aspx/Artist/UpcomingPerformances/* // @match https://*.popmundo.com/World/Popmundo.aspx/Artist/PerformanceDetails/* // @grant none // @icon  // ==/UserScript== (function() { 'use strict'; const CONFIG = { storageKey: 'pm_massCancel_status', startDateKey: 'pm_massCancel_startDate', endDateKey: 'pm_massCancel_endDate', // MODIFICATION: Key updated to reflect multiple times. targetTimesKey: 'pm_massCancel_targetTimes' }; // ================================================================================= // ## Logic for the "Upcoming Performances" Page // ================================================================================= function handleUpcomingPage() { const setupUI = () => { const tableWrapper = document.getElementById('tableupcoming_wrapper'); if (!tableWrapper || document.getElementById('massCancelPanel')) return; const controlPanel = document.createElement('div'); // MODIFICATION: The UI now includes a multi-select box for times. controlPanel.innerHTML = ` <div id="massCancelPanel" style="padding: 15px; margin-bottom: 10px; border: 2px solid #c00; background-color: #fdd; text-align: center;"> <h3 style="margin: 0 0 10px 0;">Mass Show Canceller</h3> <div style="display: flex; justify-content: center; align-items: flex-start; gap: 20px;"> <span> <label for="cancelStartDate">From Date:</label><br> <input type="date" id="cancelStartDate" style="padding: 5px;"> </span> <span> <label for="cancelEndDate">To Date:</label><br> <input type="date" id="cancelEndDate" style="padding: 5px;"> </span> <span> <label for="cancelTimes">At Times (Ctrl+Click):</label><br> <select id="cancelTimes" multiple size="5" style="padding: 5px; height: 100px; width: 100px;"> <option value="14:00" selected>14:00</option> <option value="16:00">16:00</option> <option value="18:00">18:00</option> <option value="20:00">20:00</option> <option value="22:00">22:00</option> </select> </span> </div> <div style="margin-top: 15px;"> <button id="startCancelBtn" style="padding: 8px 15px; background-color: #4CAF50; color: white; border: none; cursor: pointer; font-size: 14px;">Start Mass Cancel</button> <button id="stopCancelBtn" style="padding: 8px 15px; background-color: #f44336; color: white; border: none; cursor: pointer; margin-left: 10px; font-size: 14px;">Stop Process</button> </div> <p id="cancelStatus" style="margin-top: 10px; font-weight: bold; min-height: 1.2em;"></p> </div> `; tableWrapper.prepend(controlPanel); addEventListeners(); // MODIFICATION: Logic to restore the multi-select state on page load. if (sessionStorage.getItem(CONFIG.storageKey) === 'RUNNING') { document.getElementById('cancelStartDate').value = sessionStorage.getItem(CONFIG.startDateKey); document.getElementById('cancelEndDate').value = sessionStorage.getItem(CONFIG.endDateKey); const savedTimes = JSON.parse(sessionStorage.getItem(CONFIG.targetTimesKey) || '[]'); if (savedTimes.length > 0) { document.querySelectorAll('#cancelTimes option').forEach(opt => { opt.selected = savedTimes.includes(opt.value); }); } findAndCancelNextShow(); } }; const findAndCancelNextShow = () => { const statusElement = document.getElementById('cancelStatus'); const startDateStr = sessionStorage.getItem(CONFIG.startDateKey); const endDateStr = sessionStorage.getItem(CONFIG.endDateKey); // MODIFICATION: Retrieve the stored times array. const targetTimesStr = sessionStorage.getItem(CONFIG.targetTimesKey); if (!startDateStr || !endDateStr || !targetTimesStr) return; // MODIFICATION: Parse the JSON string into an array and format for display. const targetTimes = JSON.parse(targetTimesStr); const timesForDisplay = `<b>${targetTimes.join(', ')}</b>`; statusElement.innerHTML = `PROCESS RUNNING... Searching for shows at ${timesForDisplay} between ${startDateStr} and ${endDateStr}.`; statusElement.style.color = '#c00'; const startDate = new Date(startDateStr + 'T00:00:00'); const endDate = new Date(endDateStr + 'T00:00:00'); const showRows = document.querySelectorAll('#tableupcoming tbody tr'); let foundShowToCancel = false; for (const row of showRows) { const dateCell = row.querySelector('td:first-child'); const detailsCell = row.querySelector('td:last-child'); if (!dateCell || !detailsCell) continue; const sortKeySpan = dateCell.querySelector('span.sortkey'); let fullDateText = dateCell.textContent; if (sortKeySpan) { fullDateText = fullDateText.replace(sortKeySpan.textContent, '').trim(); } const parts = fullDateText.split(','); if (parts.length !== 2) continue; const showDateStr = parts[0].trim(); const showTimeStr = parts[1].trim(); const [day, month, year] = showDateStr.split('/'); const showDate = new Date(parseInt(year), parseInt(month) - 1, parseInt(day)); // MODIFICATION: Check if the show's time is in our array of target times. if ((showDate >= startDate && showDate <= endDate) && (targetTimes.includes(showTimeStr))) { const detailsLink = detailsCell.querySelector('a'); if (detailsLink) { statusElement.textContent = `Found matching show on ${showDateStr} at ${showTimeStr}. Navigating to cancel...`; foundShowToCancel = true; window.location.href = detailsLink.href; return; // Exit after finding one to process } } } if (!foundShowToCancel) { statusElement.textContent = `All shows at ${timesForDisplay} in the specified date range have been cancelled.`; statusElement.style.color = 'green'; alert('Mass cancel process finished! ✅'); sessionStorage.removeItem(CONFIG.storageKey); sessionStorage.removeItem(CONFIG.startDateKey); sessionStorage.removeItem(CONFIG.endDateKey); sessionStorage.removeItem(CONFIG.targetTimesKey); // Use updated key } }; const addEventListeners = () => { document.getElementById('startCancelBtn').addEventListener('click', () => { const startDate = document.getElementById('cancelStartDate').value; const endDate = document.getElementById('cancelEndDate').value; // MODIFICATION: Get all selected time values. const selectedOptions = document.querySelectorAll('#cancelTimes option:checked'); const targetTimes = Array.from(selectedOptions).map(el => el.value); if (!startDate || !endDate || targetTimes.length === 0) { alert('Please select a Start Date, End Date, and at least one Time.'); return; } if (new Date(startDate) > new Date(endDate)) { alert('The start date cannot be after the end date.'); return; } sessionStorage.setItem(CONFIG.storageKey, 'RUNNING'); sessionStorage.setItem(CONFIG.startDateKey, startDate); sessionStorage.setItem(CONFIG.endDateKey, endDate); // MODIFICATION: Stringify the array for storage. sessionStorage.setItem(CONFIG.targetTimesKey, JSON.stringify(targetTimes)); findAndCancelNextShow(); }); document.getElementById('stopCancelBtn').addEventListener('click', () => { sessionStorage.removeItem(CONFIG.storageKey); sessionStorage.removeItem(CONFIG.startDateKey); sessionStorage.removeItem(CONFIG.endDateKey); sessionStorage.removeItem(CONFIG.targetTimesKey); // Use updated key alert('Process stopped.'); location.reload(); }); }; const interval = setInterval(setupUI, 250); } // ================================================================================= // ## Logic for the "Performance Details" Page (No changes needed here) // ================================================================================= function handleDetailsPage() { if (sessionStorage.getItem(CONFIG.storageKey) === 'RUNNING') { const initialCancelButton = document.querySelector('#ctl00_cphLeftColumn_ctl01_btnCancelPerformance'); if (initialCancelButton) { initialCancelButton.click(); setTimeout(() => { const finalConfirmButton = document.querySelector('body > div:nth-child(4) > div.ui-dialog-buttonpane.ui-widget-content.ui-helper-clearfix > div > button:nth-child(1)'); if (finalConfirmButton && (finalConfirmButton.textContent.includes('Yes') || finalConfirmButton.textContent.includes('Ok'))) { finalConfirmButton.click(); } else { console.error("Could not find the FINAL confirmation button. Stopping process."); alert("Error: Could not find the final confirmation button. The process has been stopped."); sessionStorage.removeItem(CONFIG.storageKey); } }, 500); } else { console.log("No 'Cancel Show' button found, maybe it's already cancelled. Returning to list."); setTimeout(() => { history.back(); }, 250); } } } // --- Script Router --- const path = window.location.pathname; if (path.includes('/Artist/UpcomingPerformances/')) { handleUpcomingPage(); } else if (path.includes('/Artist/PerformanceDetails/')) { handleDetailsPage(); } })();