// ==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
// ==/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 = "<h3>候选清单</h3>";
// 添加导出候选清单按钮
const exportButton = createExportButton();
panel.appendChild(exportButton);
const clearButton = document.createElement("button");
clearButton.innerText = "清空";
clearButton.style.marginBottom = "10px";
clearButton.onclick = clearCandidates;
panel.appendChild(clearButton);
// 渲染所有候选项
candidates.forEach((app, index) => {
const appDiv = document.createElement("div");
appDiv.style.display = "flex";
appDiv.style.justifyContent = "space-between";
appDiv.style.marginBottom = "5px";
const appInfo = document.createElement("span");
appInfo.innerText = `${app.name} (${app.category})`;
const deleteButton = document.createElement("button");
deleteButton.innerText = "删除";
deleteButton.style.backgroundColor = "#ff6666";
deleteButton.style.color = "white";
deleteButton.style.border = "none";
deleteButton.style.borderRadius = "3px";
deleteButton.style.cursor = "pointer";
deleteButton.onclick = () => deleteCandidate(index);
appDiv.appendChild(appInfo);
appDiv.appendChild(deleteButton);
panel.appendChild(appDiv);
});
}
// **删除候选清单中的某个应用**
function deleteCandidate(index) {
candidates.splice(index, 1);
localStorage.setItem("candidates", JSON.stringify(candidates));
renderCandidatesPanel(); // 更新面板
}
// **清空候选清单**
function clearCandidates() {
candidates.length = 0;
localStorage.setItem("candidates", JSON.stringify(candidates));
renderCandidatesPanel(); // 更新面板
}
// **获取应用类别和下载量**
function getCategoryAndDownloads() {
// 先获取正确的父容器
const parentContainer = document.querySelector("div.app-info-card.dd-flex");
if (!parentContainer) {
console.log("未找到正确的父容器!");
return "未知类别";
}
// 获取该容器下所有 app-info-card-item
const items = parentContainer.querySelectorAll("div.app-info-card-item");
// 获取items中的第3个元素
const blockCategory = items[2].querySelector("div.app-desc-value");
const categoryElement = blockCategory.querySelector("a.dd-desc-color");
// 获取items中的倒数第2个元素
const blockDownloads =
items[items.length - 2].querySelector("div.app-value");
const downloadsElement = blockDownloads.querySelector("a.app-value");
// 返回类别和下载量
return {
appCategory: categoryElement
? categoryElement.innerText.trim()
: "未知类别",
appDownloads: downloadsElement
? downloadsElement.innerText.trim()
: "未知下载量",
};
}
// **获取应用发布时间**
function getPubTime() {
const parentContainer = document.querySelector("div.app-base-info-wrap");
if (!parentContainer) {
return "-";
}
const item = parentContainer.querySelectorAll(
"div.content-title.dd-flex.dd-flex-end"
)[1];
return item ? item.innerText.trim() : "-";
}
// **模拟点击"排行榜全部"按钮**
function clickBtn() {
const btn = document
.querySelector("ul.filter-list.filter-group.dd-overflow-visible")
.querySelectorAll("a.toggle-item")[0];
if (btn) {
btn.click();
}
}
// **将存储的jsonData进行处理**
// "rank_type": 2, 游戏榜
// "genre_id": 33, 游戏总榜
// "brand_id": 1, 免费
// "brand_id": 2, 畅销
// "brand_id": 3, 付费
// "brand_id": 5, 人气蹿升
function processJsonData(rankData) {
let rank1 = "-";
let rank2 = "-";
let rank3 = "-";
rankData.forEach((item) => {
if (
item.rank_type === 2 &&
item.genre_id === 33 &&
(item.brand_id === 1 || item.brand_id === 3)
) {
// 免费/付费榜排名
rank1 = item.stats.at(-1).at(-1) ? item.stats.at(-1).at(-1) : "-";
}
if (item.rank_type === 2 && item.genre_id === 33 && item.brand_id === 2) {
// 畅销榜排名
rank2 = item.stats.at(-1).at(-1) ? item.stats.at(-1).at(-1) : "-";
}
if (item.rank_type === 2 && item.genre_id === 33 && item.brand_id === 5) {
// 人气蹿升榜排名
rank3 = item.stats.at(-1).at(-1) ? item.stats.at(-1).at(-1) : "-";
}
});
return { rank1, rank2, rank3 };
}
// **通用 CSV 导出函数**
function exportToCSV(data, filename) {
// 去重:基于应用名称进行去重
const uniqueData = Array.from(new Set(data.map((app) => app.name))).map(
(name) => {
return data.find((app) => app.name === name);
}
);
let csvContent =
"名称,类别,免费/付费榜,畅销榜,人气蹿升榜,下载量,发布时间,链接\n";
uniqueData.forEach((app) => {
csvContent += `"${app.name}","${app.category}","${app.rank1}","${app.rank2}","${app.rank3}","${app.downloads}","${app.pubtime}","${app.link}"\n`;
});
// 生成带日期的文件名
const fullFilename = `${filename}_${getCurrentDate()}.csv`;
// 创建 CSV 下载链接
const blob = new Blob([csvContent], { type: "text/csv;charset=utf-8;" });
const link = document.createElement("a");
link.href = URL.createObjectURL(blob);
link.setAttribute("download", fullFilename);
document.body.appendChild(link);
link.click();
document.body.removeChild(link);
}
// **导出候选清单**
function exportCandidateList() {
const candidateApps = [...candidates];
if (candidateApps.length === 0) {
alert("候选清单为空!");
return;
}
exportToCSV(candidateApps, "详情页_候选清单");
// **清空候选清单**
clearCandidates();
}
// **主逻辑**
requestAnimationFrame(startListener); // 开始监听页面可见性变化
createCandidatesPanel(); // 创建候选清单面板
renderCandidatesPanel(); // 渲染候选清单
})();