// ==UserScript==
// @name linux.do 等级查看
// @namespace http://tampermonkey.net/
// @version 0.0.5
// @description 一个linux.do论坛的小插件,可查询用户等级和升级到下一级的要求
// @author Reno, Hua, NullUser
// @icon https://www.google.com/s2/favicons?domain=linux.do
// @match https://linux.do/*
// @connect connect.linux.do
// @grant GM_xmlhttpRequest
// @license MIT
// ==/UserScript==
(function() {
'use strict';
const StyleManager = {
styles: `
@keyframes breathAnimation {
0%, 100% { transform: scale(1); box-shadow: 0 0 10px rgba(0,0,0,0.15); }
50% { transform: scale(1.1); box-shadow: 0 0 20px rgba(0,0,0,0.3); }
}
.breath-animation {
animation: breathAnimation 3s ease-in-out infinite;
}
.minimized {
border-radius: 50%;
cursor: pointer;
transition: transform 0.3s ease, box-shadow 0.3s ease;
width: 50px;
height: 50px;
display: flex;
justify-content: center;
align-items: center;
background: var(--minimized-bg);
box-shadow: 0 4px 12px rgba(0,0,0,0.1);
}
.minimized:hover {
transform: scale(1.1);
box-shadow: 0 0 15px rgba(0,0,0,0.3);
}
.linuxDoLevelPopup {
position: fixed;
width: 360px;
height: auto;
background: var(--popup-bg);
box-shadow: 0 8px 30px rgba(0,0,0,0.1);
padding: 15px;
z-index: 10000;
font-size: 14px;
border-radius: 15px;
cursor: move;
transition: transform 0.2s ease, box-shadow 0.2s ease, opacity 0.2s ease;
}
.linuxDoLevelPopup.hidden {
opacity: 0;
visibility: hidden;
}
.linuxDoLevelPopup:hover {
box-shadow: 0 12px 40px rgba(0,0,0,0.2);
}
.linuxDoLevelPopup input,
.linuxDoLevelPopup button {
width: 100%;
background: transparent;
margin-top: 8px;
padding: 10px;
border-radius: 6px;
border: 1px solid var(--input-border);
box-sizing: border-box;
font-size: 14px;
transition: border-color 0.3s ease, box-shadow 0.3s ease;
}
.linuxDoLevelPopup input:focus,
.linuxDoLevelPopup button:focus {
outline: none;
border-color: #007BFF;
box-shadow: 0 0 5px rgba(0,123,255,0.5);
}
.linuxDoLevelPopup button {
background-color: var(--button-bg);
color: var(--button-color);
border: none;
cursor: pointer;
transition: background-color 0.3s ease, transform 0.2s ease, box-shadow 0.3s ease;
}
.linuxDoLevelPopup button:hover {
background-color: var(--button-hover-bg);
transform: translateY(-2px);
box-shadow: 0 6px 15px rgba(0,0,0,0.1);
}
.minimizeButton {
position: absolute;
top: 5px;
right: 5px;
background: transparent;
border: none;
cursor: pointer;
width: 25px;
height: 25px;
font-size: 16px;
color: var(--minimize-btn-color);
transition: color 0.3s ease;
}
.minimizeButton:hover {
color: var(--minimize-btn-hover-color);
}
.summary-table {
width: 100%;
border-collapse: collapse;
animation: fadeIn 0.5s ease-in-out;
font-size: 14px;
}
.summary-table td {
padding: 4px;
text-align: left;
border-bottom: none;
white-space: nowrap;
}
.progress-bar {
position: relative;
height: 10px;
background-color: var(--progress-bg);
border-radius: 5px;
overflow: hidden;
width: 50%;
display: inline-block;
vertical-align: middle;
margin-right: 10px;
}
.progress-bar-fill {
height: 100%;
background-color: #28a745;
text-align: right;
line-height: 10px;
color: white;
transition: width 0.4s ease-in-out;
padding-right: 5px;
border-radius: 5px 0 0 5px;
}
.progress-bar-fill::after {
content: '';
position: absolute;
top: 0;
left: 0;
right: 0;
bottom: 0;
background-image: linear-gradient(90deg, transparent 10%, rgba(0,0,0,0.2) 10%, rgba(0,0,0,0.2) 15%, transparent 15%);
background-size: 30px 10px;
z-index: 1;
}
.progress-text {
display: inline-block;
vertical-align: middle;
font-size: 13px;
visibility: hidden;
position: absolute;
top: -25px; /* Adjust position */
left: 0;
background-color: #f39c12;
color: #fff;
border: 1px solid #e67e22;
padding: 2px 5px;
border-radius: 4px;
box-shadow: 0px 0px 5px rgba(0,0,0,0.1);
z-index: 1000;
}
.summary-row {
display: flex;
align-items: center;
justify-content: space-between;
margin-bottom: 5px;
position: relative;
}
.summary-row:hover .progress-text {
visibility: visible;
}
.progress-percentage {
position: absolute;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
font-size: 12px;
font-weight: bold;
}
@media (prefers-color-scheme: dark) {
:root {
--minimized-bg: #2c2c2c;
--popup-bg: #333;
--input-border: #555;
--button-bg: #444;
--button-color: #f0f0f0;
--button-hover-bg: #555;
--minimize-btn-color: #888;
--minimize-btn-hover-color: #fff;
--progress-bg: #3d3d3d;
}
.progress-percentage {
color: #fff;
}
}
@media (prefers-color-scheme: light) {
:root {
--minimized-bg: #f0f0f0;
--popup-bg: #fff;
--input-border: #ddd;
--button-bg: #e0e0e0;
--button-color: #333;
--button-hover-bg: #d5d5d5;
--minimize-btn-color: #888;
--minimize-btn-hover-color: #333;
--progress-bg: #f3f3f3;
}
.progress-percentage {
color: #000;
}
}
`,
injectStyles: function() {
const styleSheet = document.createElement('style');
styleSheet.type = 'text/css';
styleSheet.innerText = this.styles;
document.head.appendChild(styleSheet);
}
};
const DataManager = {
Config: {
BASE_URL: 'https://linux.do',
PATHS: {
ABOUT: '/about.json',
USER_SUMMARY: '/u/{username}/summary.json',
USER_DETAIL: '/u/{username}.json',
},
},
levelRequirements: {
0: { 'topics_entered': 5, 'posts_read_count': 30, 'time_read': 600 },
1: { 'days_visited': 15, 'likes_given': 1, 'likes_received': 1, 'post_count': 3, 'topics_entered': 20, 'posts_read_count': 100, 'time_read': 3600 },
2: { 'days_visited': 50, 'likes_given': 30, 'likes_received': 20, 'post_count': 10 },
},
levelDescriptions: {
0: "新用户 🌱",
1: "基本用户 ⭐ ",
2: "成员 ⭐⭐",
3: "活跃用户 ⭐⭐⭐",
4: "领导者 🏆"
},
fetch: async function(url, options = {}) {
try {
const response = await fetch(url, {
...options,
headers: { "Accept": "application/json", "User-Agent": "Mozilla/5.0" },
method: options.method || "GET",
});
if (!response.ok) throw new Error(`HTTP error! status: ${response.status}`);
return await response.json();
} catch (error) {
console.error(`Error fetching data from ${url}:`, error);
throw error;
}
},
fetchAboutData: function() {
const url = this.buildUrl(this.Config.PATHS.ABOUT);
return this.fetch(url);
},
fetchSummaryData: function(username) {
const url = this.buildUrl(this.Config.PATHS.USER_SUMMARY, { username });
return this.fetch(url);
},
fetchUserData: function(username) {
const url = this.buildUrl(this.Config.PATHS.USER_DETAIL, { username });
return this.fetch(url);
},
buildUrl: function(path, params = {}) {
let url = this.Config.BASE_URL + path;
Object.keys(params).forEach(key => {
url = url.replace(`{${key}}`, encodeURIComponent(params[key]));
});
return url;
},
};
const UIManager = {
initPopup: async function() {
this.popup = this.createElement('div', { id: 'linuxDoLevelPopup', class: 'linuxDoLevelPopup' });
this.content = this.createElement('div', { id: 'linuxDoLevelPopupContent' }, '欢迎使用 Linux do 等级增强插件');
this.searchBox = this.createElement('input', { placeholder: '请输入用户名...', type: 'text', class: 'searchBox' });
this.searchButton = this.createElement('button', { class: 'searchButton' }, '搜索');
this.minimizeButton = this.createElement('button', { }, '隐藏');
this.popup.style.bottom = '20px'; // 示例:距离顶部20px
this.popup.style.right = '20px'; // 示例:距离左侧20px
this.popup.style.width = '360px'; // 初始化宽度
this.popup.style.height = 'auto'; // 高度自适应内容
this.searchButton.classList.add('btn', 'btn-icon-text', 'btn-default')
this.minimizeButton.classList.add('btn', 'btn-icon-text', 'btn-default')
this.popup.append(this.content, this.searchBox, this.searchButton, this.minimizeButton);
document.body.appendChild(this.popup);
this.minimizeButton.addEventListener('click', () => this.togglePopupSize());
this.searchButton.addEventListener('click', () => EventHandler.handleSearch());
// 添加输入框的回车键事件监听器
this.searchBox.addEventListener('keypress', (event) => {
// 检查是否按下了回车键并且弹窗不处于最小化状态
if (event.key === 'Enter' && !this.popup.classList.contains('minimized')) {
EventHandler.handleSearch();
}
});
try {
const userName = await getUserName();
this.searchBox.value = userName;
} catch (e) {
console.log(e);
}
},
createElement: function(tag, attributes, text) {
const element = document.createElement(tag);
for (const attr in attributes) {
if (attr === 'class') {
element.classList.add(attributes[attr]);
} else {
element.setAttribute(attr, attributes[attr]);
}
}
if (text) element.textContent = text;
return element;
},
async updatePopupContent(userSummary, user, userDetail, status, username) {
if (!userSummary || !user || !userDetail) {
return;
}
let content = `<strong>信任等级🏅:</strong>${DataManager.levelDescriptions[user.trust_level]}<br>`;
if (userDetail.gamification_score) {
content += `<strong>你的点数🪙:</strong><span style="color: green;">${userDetail.gamification_score}</span><br>`;
}
content += `<strong>最近活跃🕒:</strong>${formatTimestamp(userDetail.last_seen_at)}<br>`;
if (user.trust_level === 2 && user.username === username) {
content += await fetchConnect();
} else if (user.trust_level > 2) {
if (userSummary.top_categories) {
content += analyzeAbility(userSummary.top_categories);
}
} else {
content += summaryRequired(DataManager.levelRequirements[user.trust_level] || {}, userSummary, UIManager.translateStat.bind(UIManager));
}
this.content.innerHTML = content;
},
togglePopupSize: function() {
if (this.popup.classList.contains('minimized')) {
this.popup.classList.remove('minimized');
this.popup.style.width = '360px';
this.popup.style.height = 'auto';
this.content.style.display = 'block';
this.searchBox.style.display = 'block';
this.searchButton.style.display = 'block';
this.minimizeButton.textContent = '隐藏';
this.minimizeButton.style.color = 'black';
this.popup.classList.remove('breath-animation');
} else {
this.popup.classList.add('minimized');
this.popup.style.width = '50px';
this.popup.style.height = '50px';
this.content.style.display = 'none';
this.searchBox.style.display = 'none';
this.searchButton.style.display = 'none';
this.minimizeButton.textContent = '显示';
this.popup.classList.add('breath-animation');
// 调用 updatePercentage 函数并更新按钮文本
updatePercentage().then(percentage => {
if (this.popup.classList.contains('minimized')) {
let color;
// 根据百分比设置颜色
if (percentage > 50) {
color = 'purple';
} else if (percentage > 30) {
color = 'red';
} else {
color = 'green';
}
// 更新按钮的文本和文本颜色
this.minimizeButton.textContent = `${percentage.toFixed(2)}%`;
this.minimizeButton.style.color = color; // 设置文本颜色
}
}).catch(error => {
console.error('Error calculating percentage:', error);
// 出错时保持原有文本
this.minimizeButton.textContent = '展开';
this.minimizeButton.style.color = 'black';
});
}
// 自动校正窗口位置
addDraggableFeature(this.popup);
const windowWidth = window.innerWidth;
const windowHeight = window.innerHeight;
const popupWidth = this.popup.offsetWidth;
const popupHeight = this.popup.offsetHeight;
const popupTop = parseInt(this.popup.style.top);
const popupLeft = parseInt(this.popup.style.left);
// 初始化新的位置
let newTop = popupTop;
let newLeft = popupLeft;
// 上下边界同时检查
newTop = Math.min(Math.max(70, popupTop), windowHeight - popupHeight);
// 左右边界同时检查
newLeft = Math.min(Math.max(5, popupLeft), windowWidth - popupWidth - 20);
this.popup.style.top = newTop + 'px';
this.popup.style.left = newLeft + 'px';
},
displayError: function(message) {
this.content.innerHTML = `<strong>错误:</strong>用户隐藏信息或不存在`;
},
translateStat: function(stat) {
const translations = {
'days_visited': '访问天数',
'likes_given': '给出的赞',
'likes_received': '收到的赞',
'post_count': '帖子数量',
'posts_read_count': '已读帖子',
'topics_entered': '已读主题',
'time_read': '阅读时间(秒)'
};
return translations[stat] || stat;
}
};
const EventHandler = {
handleSearch: async function() {
const username = UIManager.searchBox.value.trim();
if (!username) return;
try {
UIManager.searchButton.textContent = '搜索中,请稍等!';
UIManager.searchButton.disabled = true;
const [aboutData, summaryData, userData] = await Promise.all([
DataManager.fetchAboutData(),
DataManager.fetchSummaryData(username),
DataManager.fetchUserData(username)
]);
if (summaryData && userData && aboutData) {
await UIManager.updatePopupContent(summaryData.user_summary, summaryData.users ? summaryData.users[0] : { 'trust_level': 0 }, userData.user, aboutData.about.stats, username);
}
} catch (error) {
console.error(error);
UIManager.displayError('Failed to load data');
}
UIManager.searchButton.textContent = '搜索';
UIManager.searchButton.disabled = false;
},
// 更新拖动状态
handleDragEnd: function() {
UIManager.updateDragStatus(true);
}
};
// 2级以上添加技能分析
function analyzeAbility(topCategories) {
let resultStr = "<strong>技能分析🎯:</strong><br>";
const icons = {
"常规话题": "🌐",
"wiki": "📚",
"快问快答": "❓",
"人工智能": "🤖",
"周周热点": "🔥",
"精华神贴": "✨",
"高阶秘辛": "🔮",
"读书成诗": "📖",
"配置调优": "⚙️",
"网络安全": "🔒",
"软件分享": "💾",
"软件开发": "💻",
"嵌入式": "🔌",
"机器学习": "🧠",
"代码审查": "👀",
"new-api": "🆕",
"一机难求": "📱",
"速来拼车": "🚗",
"网络记忆": "💭",
"非我莫属": "🏆",
"赏金猎人": "💰",
"搞七捻三": "🎲",
"碎碎碎念": "🗨️",
"金融经济": "💹",
"新闻": "📰",
"旅行": "✈️",
"美食": "🍽️",
"健身": "🏋️",
"音乐": "🎵",
"游戏": "🎮",
"羊毛": "🐑",
"树洞": "🌳",
"病友": "🤒",
"职场": "💼",
"断舍离": "♻️",
"二次元": "🎎",
"运营反馈": "🔄",
"老干部疗养院": "🛌",
"活动": "🎉",
};
const totalScore = topCategories.reduce((sum, category) => sum + (category.topic_count * 2) + (category.post_count * 1), 0);
topCategories.sort((a, b) => a.name.length - b.name.length);
topCategories.forEach((category, index) => {
const score = (category.topic_count * 2) + (category.post_count * 1);
const percentage = ((score / totalScore) * 100).toFixed(1) + "%";
let numStars;
if (score >= 999) {
numStars = 7; // 满分7颗红星
} else {
numStars = Math.round((score / 999) * 7); // 其他按比例显示
}
const stars = "❤️".repeat(numStars) + "🤍".repeat(7 - numStars); // 显示红星和空星
let icon = icons[category.name] || "❓"; // 如果没有找到图标,显示默认图标
resultStr += `
<div style='display: flex; justify-content: space-between; align-items: center; margin-bottom: 8px; opacity: 0; animation: fadeIn 0.5s forwards; animation-delay: ${index * 0.1}s; font-size: 13px;'>
<div style='flex: 0 0 20px; text-align: center;'>${icon}</div>
<div style='flex: 2; text-align: left;'>${category.name}</div>
<div style='flex: 4; text-align: left;'>${stars}</div>
<div style='flex: 1; text-align: right;'>${percentage}</div>
</div>`;
});
resultStr += `
<style>
@keyframes fadeIn {
to { opacity: 1; }
}
</style>
`;
return resultStr;
}
// 2级添加Connect数据
async function fetchConnect() {
return new Promise((resolve, reject) => {
GM_xmlhttpRequest({
method: 'GET',
url: 'https://connect.linux.do',
onload: (response) => {
const bodyRegex = /<body[^>]*>([\s\S]+?)<\/body>/i;
const match = bodyRegex.exec(response.responseText);
if (match) {
const doc = new DOMParser().parseFromString(match[1], 'text/html');
let summary = '<strong>升级进度🌟:</strong><br><div class="summary-table">';
let violationExists = false;
let violationStats = []; // 违规项名称
const rows = doc.querySelectorAll('tr');
rows.forEach((row, index) => {
if (row) {
const cells = Array.from(row.querySelectorAll('td'), cell => cell.innerText.trim());
if (cells.length >= 3) {
const stat = cells[0];
const curMatches = cells[1].match(/(\d+)/);
const reqMatches = cells[2].match(/(\d+)/);
const curValue = curMatches ? parseInt(curMatches[0]) : 0;
const reqValue = reqMatches ? parseInt(reqMatches[0]) : 0;
// 检查是否存在违规
if ([7, 8, 13, 14].includes(index) && curValue > reqValue) {
violationExists = true;
violationStats.push(stat); // 添加违规项名称
}
// 选择性添加到摘要
if ([1, 2, 3,4, 5,6, 9, 10,11,12].includes(index)) {
const percentage = Math.min((curValue / reqValue) * 100, 100);
let color = curValue >= reqValue ? '#28a745' : '#dc3545';
summary += `
<div class="summary-row">
<div>${stat}</div>
<div class="progress-bar" title="${curValue}/${reqValue}">
<div class="progress-bar-fill" style="width: ${percentage}%; background-color: ${color};"></div>
<div class="progress-percentage">${Math.round(percentage)}%</div>
</div>
<div class="progress-text">${curValue}/${reqValue}</div>
</div>`;
}
}
}
});
if (violationExists) {
summary += `<div style="color: red;">用户存在违规行为:${violationStats.join(', ')}</div>`;
} else {
summary += '<div style="color: green;">用户不存在违规行为</div>';
}
summary += '</div>';
resolve(summary);
} else {
reject(new Error("No content extracted from response."));
}
},
onerror: (error) => {
reject(error);
}
});
});
}
async function getUserName() {
return new Promise((resolve, reject) => {
GM_xmlhttpRequest({
method: 'GET',
url: 'https://connect.linux.do',
onload: (response) => {
const bodyRegex = /<body[^>]*>([\s\S]+?)<\/body>/i;
const match = bodyRegex.exec(response.responseText);
if (match) {
const doc = new DOMParser().parseFromString(match[1], 'text/html');
if(doc){
const userNameDom = doc.querySelector('h1');
if(userNameDom){
const text = doc.querySelector('h1').textContent;
resolve(extractUserName(text));
}
}
} else {
reject(new Error("No content extracted from response."));
}
},
onerror: (error) => {
reject(error);
}
});
});
}
// 2级以下添加升级进度功能
function summaryRequired(required, current, translateStat) {
let summary = '<strong>升级进度🌟:</strong><br>';
summary += '<div class="summary-table">';
for (const stat in required) {
if (required.hasOwnProperty(stat) && current.hasOwnProperty(stat)) {
const reqValue = required[stat];
const curValue = current[stat] || 0; // 使用 || 0 确保未定义的情况下使用0
const percentage = Math.min((curValue / reqValue) * 100, 100); // 计算百分比
let color = curValue >= reqValue ? '#28a745' : '#dc3545'; // 使用绿色或红色
summary += `
<div class="summary-row">
<div>${translateStat(stat)}</div>
<div class="progress-bar" title="${curValue}/${reqValue}">
<div class="progress-bar-fill" style="width: ${percentage}%; background-color: ${color};"></div>
<div class="progress-percentage">${Math.round(percentage)}%</div>
</div>
<div class="progress-text">${curValue}/${reqValue}</div>
</div>`;
}
}
summary += '</div>';
return summary;
}
// 添加含水率
function updatePercentage() {
return new Promise((resolve, reject) => {
let badIds = [11, 16, 34, 17, 18, 19, 29, 36, 35, 22, 26, 25];
const badScore = [];
const goodScore = [];
const urls = [
'https://linux.do/latest.json?order=created',
'https://linux.do/new.json',
'https://linux.do/top.json?period=daily'
];
Promise.all(urls.map(url => fetch(url).then(resp => resp.json())))
.then(data => {
data.forEach(({ topic_list: { topics } }) => {
topics.forEach(topic => {
const score = topic.posts_count + topic.like_count + topic.reply_count;
(badIds.includes(topic.category_id) ? badScore : goodScore).push(score);
});
});
const badTotal = badScore.reduce((acc, curr) => acc + curr, 0);
const goodTotal = goodScore.reduce((acc, curr) => acc + curr, 0);
const percentage = (badTotal / (badTotal + goodTotal)) * 100;
resolve(percentage);
})
.catch(reject);
});
};
// 添加时间格式化
function formatTimestamp(lastSeenAt) {
// 解析时间戳并去除毫秒
let timestamp = new Date(lastSeenAt);
// 使用Intl.DateTimeFormat格式化时间为上海时区
let formatter = new Intl.DateTimeFormat('zh-CN', {
timeZone: 'Asia/Shanghai',
year: 'numeric',
month: 'numeric',
day: 'numeric',
hour: 'numeric',
minute: 'numeric',
second: 'numeric',
});
// 获取格式化后的字符串
let formattedTimestamp = formatter.format(timestamp);
return formattedTimestamp;
}
// 添加拖动功能
function addDraggableFeature(element) {
let pos1 = 0, pos2 = 0, pos3 = 0, pos4 = 0;
let isDragging = false;
const dragMouseDown = function(e) {
// 检查事件的目标是否是输入框,按钮或其他可以忽略拖动逻辑的元素
if (e.target.tagName.toUpperCase() === 'INPUT' || e.target.tagName.toUpperCase() === 'TEXTAREA' || e.target.tagName.toUpperCase() === 'BUTTON') {
return; // 如果是,则不执行拖动逻辑
}
e = e || window.event;
e.preventDefault();
pos3 = e.clientX;
pos4 = e.clientY;
document.onmouseup = closeDragElement;
document.onmousemove = elementDrag;
isDragging = true;
};
const elementDrag = function(e) {
if (!isDragging) return;
e = e || window.event;
e.preventDefault();
pos1 = pos3 - e.clientX;
pos2 = pos4 - e.clientY;
pos3 = e.clientX;
pos4 = e.clientY;
// 使用requestAnimationFrame优化拖动
requestAnimationFrame(() => {
element.style.top = Math.max(0, Math.min(window.innerHeight - element.offsetHeight, element.offsetTop - pos2)) + "px";
element.style.left = Math.max(0, Math.min(window.innerWidth - element.offsetWidth, element.offsetLeft - pos1)) + "px";
// 为了避免与拖动冲突,在此移除bottom和right样式
element.style.bottom = '';
element.style.right = '';
});
};
const closeDragElement = function() {
document.onmouseup = null;
document.onmousemove = null;
isDragging = false;
// 在拖动结束时更新拖动状态
EventHandler.handleDragEnd();
};
element.onmousedown = dragMouseDown;
}
// 提取用户名
function extractUserName(input) {
const regex = /\((.*?)\)/;
const match = input.match(regex);
return match ? match[1] : null;
}
const init = () => {
StyleManager.injectStyles();
UIManager.initPopup();
addDraggableFeature(document.getElementById('linuxDoLevelPopup')); // 确保已设置该ID
UIManager.togglePopupSize(); // 初始最小化
};
init();
})();