您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
如果你的脚本干涉了某些人的利益,收到恶意差评并不意外。恶意差评有两个特点,一是账号通常新注册,二是注册后不久就会给差评,且基本不会有后续活动。本脚本获取greasyfork用户注册时间,并显示在用户名旁边。如果用户名旁边显示的时间是未来的时间,那么这个用户很可能是恶意注册的账号。
当前为
// ==UserScript== // @name Display GreasyFork user registration time // @description If your script interferes with the interests of certain individuals, receiving malicious negative reviews is not surprising. Malicious negative reviews have two characteristics: First, the account is usually newly registered, and second, the review is given shortly after registration with little to no subsequent activity. This script retrieves the registration time of a GreasyFork user and displays it next to their username. If the time displayed next to the username is in the future, this user is likely to be a maliciously registered account. // @name:zh-CN 显示Greasyfork用户注册时间,识别恶意评论 // @description:zh-CN 如果你的脚本干涉了某些人的利益,收到恶意差评并不意外。恶意差评有两个特点,一是账号通常新注册,二是注册后不久就会给差评,且基本不会有后续活动。本脚本获取greasyfork用户注册时间,并显示在用户名旁边。如果用户名旁边显示的时间是未来的时间,那么这个用户很可能是恶意注册的账号。 // @grant GM_setValue // @grant GM_getValue // @grant GM_xmlhttpRequest // @match *://greasyfork.org/* // @match *://sleazyfork.org/* // @require https://code.jquery.com/jquery-3.7.1.min.js // @author yysk.org,人民的勤务员 <[email protected]> // @namespace https://github.com/ChinaGodMan/UserScripts // @supportURL https://github.com/ChinaGodMan/UserScripts/issues // @homepageURL https://github.com/ChinaGodMan/UserScripts // @license MIT // @icon https://raw.githubusercontent.com/ChinaGodMan/UserScriptsHistory/main/scriptsIcon/greasyfork-webhook-sync-enhanced.svg // @compatible chrome // @compatible firefox // @compatible edge // @compatible opera // @compatible safari // @compatible kiwi // @version 2025.03.10.0959 // @created 2025-03-10 09:59:11 // @modified 2025-03-10 09:59:11 // ==/UserScript== /** * File: greasyfork-user-registration-time.user.js * Project: UserScripts * File Created: 2025/03/10,Monday 09:59:36 * Author: 人民的勤务员@ChinaGodMan ([email protected]) * ----- * Last Modified: 2025/03/10,Monday 12:12:45 * Modified By: 人民的勤务员@ChinaGodMan ([email protected]) * ----- * License: MIT License * Copyright © 2024 - 2025 ChinaGodMan,Inc */ (function () { 'use strict' const interval = 3000 const absoluteTime = 'relative-time1'//随便修改个类名,显示精确时间.比如relative-time1 var lang = document.querySelector('html').lang.toLowerCase() || navigator.language.toLowerCase() if (!['en', 'zh-cn', 'zh-tw', 'ja', 'ru', 'kr'].includes(lang)) lang = 'en' const i18n = { en: { title: 'Registration time', prefix: 'R' }, 'zh-cn': { title: '注册时间', prefix: '注' }, 'zh-tw': { title: '註冊時間', prefix: '注' }, ja: { title: '登録時間', prefix: 'R' }, ru: { title: 'время регистрации', prefix: 'R' }, kr: { title: '등록 시간', prefix: 'R' } } const title = i18n[lang].title const prefix = i18n[lang].prefix function pad(s, d) { s = `000000${s}` return s.substring(s.length - d) } const formatUFn = (dt) => { return `${dt.getFullYear()}.${pad(dt.getMonth() + 1, 2)}.${pad(dt.getDate(), 2)} ${pad(dt.getHours(), 2)}:${pad(dt.getMinutes(), 2)}:${pad(dt.getSeconds(), 2)}` } const formatFrFn = (dt) => { return `${pad(dt.getDate(), 2)}.${pad(dt.getMonth() + 1, 2)}.${dt.getFullYear()} ${pad(dt.getHours(), 2)}:${pad(dt.getMinutes(), 2)}:${pad(dt.getSeconds(), 2)}` } function formatTime(utcTime, lang = 'en') { const dt = new Date(utcTime) let formatFn = (lang === 'fr' || lang.startsWith('fr-')) ? formatFrFn : formatUFn return formatFn(dt) } async function genregtime(uid) { if (GM_getValue(uid) !== undefined && GM_getValue(uid) !== null) { return GM_getValue(uid) } const user_api = `https://api.greasyfork.org/users/${uid}.json` var created_at try { const response = await fetch(user_api) const data = await response.json() created_at = data.created_at console.log('🔍 ~ created_at:', created_at) } catch (error) { console.error('请求失败:', error) } GM_setValue(uid, created_at) return created_at } function gensnippet(regtime, uid) { return `<span class="regtime" style="margin-left: 10px; font-weight: bold; color: red;" title="uid ${uid} ${title}">${prefix} <${absoluteTime} datetime="${regtime}+00:00" prefix="">${regtime}</${absoluteTime}></span>` } let isProcessing = false function run() { if (isProcessing) return isProcessing = true var maxuid = 0 const uids = document.documentElement.innerHTML.matchAll(/\/users\/(\d+)/g) for (const uid of uids) { if (parseInt(uid[1]) > maxuid) maxuid = parseInt(uid[1]) } if (window.location.href.match(/(greasyfork|sleazyfork).org\/\w+-\w+\/users\/\d+(-[^/]*)?$/) && jQuery('section#about-user > span.regtime').length === 0) { let uid = jQuery('section#about-user > a.report-link').attr('href').match(/item_id=(\d+)/)[1] genregtime(uid).then(regtime => { regtime = formatTime(regtime, lang) jQuery('section#about-user > h2').after(gensnippet(regtime, uid)) }) } jQuery('a.user-link, dd.script-list-author > span > a, dd.script-show-author > span > a, table.log-table > tbody > tr > td > a, i:contains(\'Deleted user \')').each(function (i, el) { var lastele = jQuery(el).parent().children().last() if (lastele.attr('class') === 'regtime' || lastele.attr('class') === 'small-btn') return var button = jQuery('<button>').text(i18n[lang].prefix).addClass('small-btn') var m = (el.tagName === 'A') ? jQuery(el).attr('href').match(/\/users\/(\d+)/) : jQuery(el).text().match(/Deleted user (\d+)/) if (!m) return button.on('click', function () { isProcessing = true genregtime(m[1]).then(regtime => { regtime = formatTime(regtime, lang) lastele.after(gensnippet(regtime, m[1])) }) button.remove() }) // 如果存在时间就直接插入,不存在显示按钮 if (GM_getValue(m[1]) !== undefined && GM_getValue(m[1]) !== null) { lastele.after(gensnippet(formatTime(GM_getValue(m[1]), lang), m[1])) } else { lastele.after(button) } }) isProcessing = false return run } setInterval(run(), interval) })()