// ==UserScript== // @name 点点数据详细页完整版 // @version 2025-02-12 // @author DethanZ // @description 在榜单详细页面加入候选, 并提供清单查看、删除和清空功能, 导出所有候选应用的 CSV 文件 // @icon  // @match https://app.diandian.com/rank/* // @match https://app.diandian.com/app/* // @license GPL-3.0 License // @run-at document-start // @namespace http://tampermonkey.net/ // @supportURL https://github.com/dethanzhang/UserScript // @homepageURL https://github.com/dethanzhang/UserScript // @downloadURL https://update.greasyfork.icu/scripts/526647/%E7%82%B9%E7%82%B9%E6%95%B0%E6%8D%AE%E8%AF%A6%E7%BB%86%E9%A1%B5%E5%AE%8C%E6%95%B4%E7%89%88.user.js // @updateURL https://update.greasyfork.icu/scripts/526647/%E7%82%B9%E7%82%B9%E6%95%B0%E6%8D%AE%E8%AF%A6%E7%BB%86%E9%A1%B5%E5%AE%8C%E6%95%B4%E7%89%88.meta.js // ==/UserScript== (function () { "use strict"; // 存储候选应用清单 const candidates = JSON.parse(localStorage.getItem("candidates")) || []; // 备份原始的 open 和 send 方法 const _open = XMLHttpRequest.prototype.open; const _send = XMLHttpRequest.prototype.send; // 重写 open 方法 XMLHttpRequest.prototype.open = function ( method, url, async, user, password ) { this._url = url; // 记录请求的 URL return _open.apply(this, arguments); }; // 使用闭包来确保每个页面有独立的 captured 和 rankjson 变量 (function () { let captured = false; let rankjson = []; const pattern = /\/trend\?.*&brand_id=0/; // 判断是否为指定的请求链接 // 重写 send 方法 XMLHttpRequest.prototype.send = function (body) { // 如果是符合条件的url则捕获 if (!captured && pattern.test(this._url)) { let _onload = this.onload; // 备份原来的 onload 事件(如果有) this.onload = function (event) { if (_onload) _onload.call(this, event); // 保持原来的逻辑 rankjson.push(JSON.parse(this.responseText)); captured = true; // 处理完后设置标志为 true,停止进一步捕获 }; } return _send.apply(this, arguments); }; // **添加应用到候选清单&存储所有数据的逻辑** async function addToCandidates() { if (!captured) { clickBtn(); } // 等待 captured 变为 true while (!captured) { await new Promise((resolve) => setTimeout(resolve, 500)); // 等待 } const { rank1, rank2, rank3 } = processJsonData(rankjson[0].data.list); // 处理jsonData const appName = document.querySelector("div.ellip.font-600") ? document.querySelector("div.ellip.font-600").innerText.trim() : "未知"; const { appCategory, appDownloads } = getCategoryAndDownloads(); // 应用类别和下载量 const appLink = window.location.href.replace( "/googleplay-rank?", "/googleplay?" ); // 将当前应用的信息加入候选清单 const appData = { name: appName, category: appCategory, rank1: rank1, rank2: rank2, rank3: rank3, downloads: formatNumberToChinese(appDownloads), pubtime: getPubTime(), link: appLink, }; if (!candidates.some((app) => app.link === appData.link)) { candidates.push(appData); // 确保去重 localStorage.setItem("candidates", JSON.stringify(candidates)); // alert('已加入候选清单!'); renderCandidatesPanel(); // 更新候选清单面板 } else { alert("此应用已在候选清单中!"); } } // 详情页显示“加入候选” if (window.location.href.includes("/app/")) { createButton("加入候选", addToCandidates); } })(); // **获取当前日期(格式:YYYYMMDD)** function getCurrentDate() { const now = new Date(); const year = now.getFullYear(); const month = String(now.getMonth() + 1).padStart(2, "0"); // 月份是从 0 开始的 const day = String(now.getDate()).padStart(2, "0"); return `${year}${month}${day}`; } // **格式化数字为中文格式** function formatNumberToChinese(numStr) { numStr = numStr.replace(/,/g, ""); const hasPlus = numStr.includes("+"); const num = parseFloat(numStr.replace("+", "")); let result; if (num >= 100000000) { result = Math.round(num / 100000000) + "亿"; // 四舍五入到整数 } else if (num >= 10000) { result = Math.round(num / 10000) + "万"; // 四舍五入到整数 } else { result = num.toString(); } if (hasPlus) { result += "+"; } return result; } // **监听页面可见性变化** function visibilityChangeHandler() { if (document.visibilityState === "hidden") { stopListener(); // 切到后台时停止监听 requestAnimationFrame(startListener); // 下次回到前台时重新监听 } } // **开始监听** function startListener() { // 重新读取 localStorage 中的 candidates const updatedCandidates = JSON.parse(localStorage.getItem("candidates")) || []; candidates.length = 0; // 清空当前数组 candidates.push(...updatedCandidates); // 更新为最新的候选清单 renderCandidatesPanel(); // 更新候选清单面板 document.addEventListener("visibilitychange", visibilityChangeHandler); } // **停止监听** function stopListener() { document.removeEventListener("visibilitychange", visibilityChangeHandler); } // **创建上方按钮: 加入候选** function createButton(text, onClickHandler) { const button = document.createElement("button"); button.innerText = text; button.style.position = "fixed"; button.style.top = "60px"; button.style.right = "20px"; button.style.padding = "10px"; button.style.background = "#007bff"; button.style.color = "white"; button.style.border = "none"; button.style.borderRadius = "5px"; button.style.cursor = "pointer"; button.style.zIndex = "1000"; button.onclick = onClickHandler; document.body.appendChild(button); } // **创建按钮: 导出候选清单** function createExportButton() { const exportButton = document.createElement("button"); exportButton.textContent = "导出候选清单CSV"; exportButton.style.position = "absolute"; exportButton.style.top = "10px"; exportButton.style.right = "10px"; exportButton.style.padding = "5px"; exportButton.style.background = "#28a745"; exportButton.style.color = "white"; exportButton.style.border = "none"; exportButton.style.borderRadius = "5px"; exportButton.style.cursor = "pointer"; exportButton.style.zIndex = "1000"; exportButton.onclick = exportCandidateList; return exportButton; } // **创建候选清单面板** function createCandidatesPanel() { const panel = document.createElement("div"); panel.id = "candidatesPanel"; panel.style.position = "fixed"; panel.style.bottom = "20px"; panel.style.right = "20px"; panel.style.width = "400px"; panel.style.height = "200px"; panel.style.overflowY = "auto"; panel.style.backgroundColor = "white"; panel.style.border = "1px solid #ccc"; panel.style.padding = "10px"; panel.style.zIndex = "1000"; panel.style.display = "none"; // 默认隐藏 const title = document.createElement("h3"); title.innerText = "候选清单"; panel.appendChild(title); // 添加导出候选清单按钮 const exportButton = createExportButton(); panel.appendChild(exportButton); // 添加清空候选清单按钮 const clearButton = document.createElement("button"); clearButton.innerText = "清空"; clearButton.style.marginBottom = "10px"; clearButton.onclick = clearCandidates; panel.appendChild(clearButton); document.body.appendChild(panel); } // **渲染候选清单** function renderCandidatesPanel() { const panel = document.getElementById("candidatesPanel"); panel.style.display = "block"; // 显示候选面板 // 清空现有清单 panel.innerHTML = "