// ==UserScript==
// @name Porn Blocker | 色情内容过滤器
// @name:en Porn Blocker
// @name:zh-CN 色情内容过滤器
// @name:zh-TW 色情內容過濾器
// @name:zh-HK 色情內容過濾器
// @name:ja アダルトコンテンツブロッカー
// @name:ko 성인 컨텐츠 차단기
// @name:ru Блокировщик порнографии
// @namespace https://noctiro.moe
// @version 2.1.2
// @description A powerful content blocker that helps protect you from inappropriate websites. Features: Auto-detection of adult content, Multi-language support, Smart scoring system, Safe browsing protection.
// @description:en A powerful content blocker that helps protect you from inappropriate websites. Features: Auto-detection of adult content, Multi-language support, Smart scoring system, Safe browsing protection.
// @description:zh-CN 强大的网页过滤工具,帮助你远离不良网站。功能特点:智能检测色情内容,多语言支持,评分系统,安全浏览保护,支持自定义过滤规则。为了更好的网络环境,从我做起。
// @description:zh-TW 強大的網頁過濾工具,幫助你遠離不良網站。功能特點:智能檢測色情內容,多語言支持,評分系統,安全瀏覽保護,支持自定義過濾規則。為了更好的網絡環境,從我做起。
// @description:zh-HK 強大的網頁過濾工具,幫助你遠離不良網站。功能特點:智能檢測色情內容,多語言支持,評分系統,安全瀏覽保護,支持自定義過濾規則。為了更好的網絡環境,從我做起。
// @description:ja アダルトコンテンツを自動的にブロックする強力なツールです。機能:アダルトコンテンツの自動検出、多言語対応、スコアリングシステム、カスタマイズ可能なフィルタリング。より良いインターネット環境のために。
// @description:ko 성인 컨텐츠를 자동으로 차단하는 강력한 도구입니다. 기능: 성인 컨텐츠 자동 감지, 다국어 지원, 점수 시스템, 안전 브라우징 보호, 맞춤형 필터링 규칙。
// @description:ru Мощный инструмент для блокировки неприемлемого контента. Функции: автоматическое определение, многоязычная поддержка, система оценки, настраиваемые правила фильтрации。
// @license Apache-2.0
// @icon 
// @match *://*/*
// @run-at document-start
// @run-at document-end
// @run-at document-idle
// @grant none
// @grant GM_setValue
// @grant GM_getValue
// ==/UserScript==
(function () {
'use strict';
// 多语言支持
const i18n = {
'en': {
title: '🚫 Access Denied',
message: 'This page contains content that may harm your well-being.',
redirect: 'You will be redirected in <span class="countdown">4</span> seconds…',
footer: 'Cherish your mind · Stay away from harmful sites'
},
'zh-CN': {
title: '🚫 访问受限',
message: '该页面包含有害信息,可能危害您的身心健康。',
redirect: '将在 <span class="countdown">4</span> 秒后自动跳转……',
footer: '珍爱健康 · 远离有害信息'
},
'zh-TW': {
title: '🚫 存取受限',
message: '此頁面含有有害資訊,可能危害您的身心健康。',
redirect: '將於 <span class="countdown">4</span> 秒後自動跳轉……',
footer: '珍愛健康 · 遠離有害資訊'
},
'zh-HK': {
title: '🚫 存取受限',
message: '此網頁含有有害資訊,或會損害您的身心健康。',
redirect: '<span class="countdown">4</span> 秒後將自動引導離開……',
footer: '珍重健康 · 遠離有害內容'
},
'ja': {
title: '🚫 アクセス制限',
message: 'このページには心身に悪影響を及ぼす可能性のある情報が含まれています。',
redirect: '<span class="countdown">4</span> 秒後に自動的にページが移動します……',
footer: '心と体を大切に · 有害サイトに近づかない'
},
'ko': {
title: '🚫 접근 제한',
message: '이 페이지에는 신체와 정신에 해를 끼칠 수 있는 정보가 포함되어 있습니다.',
redirect: '<span class="countdown">4</span>초 후 자동으로 이동됩니다……',
footer: '건강을 소중히 · 유해 사이트는 멀리'
},
'ru': {
title: '🚫 Доступ ограничен',
message: 'Эта страница содержит материалы, которые могут нанести вред вашему здоровью.',
redirect: 'Перенаправление произойдёт через <span class="countdown">4</span> секунды……',
footer: 'Берегите здоровье · Держитесь подальше от вредных сайтов'
}
};
// 获取用户语言
const getUserLanguage = () => {
const lang = navigator.language || navigator.userLanguage;
// 检查完整语言代码
if (i18n[lang]) return lang;
// 处理中文的特殊情况
if (lang.startsWith('zh')) {
const region = lang.toLowerCase();
if (region.includes('tw') || region.includes('hant')) return 'zh-TW';
if (region.includes('hk')) return 'zh-HK';
return 'zh-CN';
}
// 检查简单语言代码
const shortLang = lang.split('-')[0];
if (i18n[shortLang]) return shortLang;
return 'en';
};
// 浏览器检测函数
const getBrowserType = () => {
const ua = navigator.userAgent.toLowerCase();
// 1. User-Agent Client Hints (modern Chromium-based browsers)
if (navigator.userAgentData && Array.isArray(navigator.userAgentData.brands)) {
const brands = navigator.userAgentData.brands.map(b => b.brand.toLowerCase());
if (brands.includes('microsoft edge')) return 'edge';
if (brands.includes('google chrome')) return 'chrome';
if (brands.includes('brave')) return 'brave';
if (brands.includes('vivaldi')) return 'vivaldi';
if (brands.includes('opera') || brands.includes('opr')) return 'opera';
if (brands.includes('arc')) return 'arc';
// If none of the above, it's some other Chromium variant
if (brands.includes('chromium')) return 'chromium';
}
// 2. Arc-specific CSS variable detection (Arc adds --arc-palette-background)
if (window.getComputedStyle(document.documentElement)
.getPropertyValue('--arc-palette-background')) {
return 'arc';
}
// 3. Traditional UA substring checks for non-Chromium or unhinted cases
if (ua.includes('ucbrowser')) return 'uc';
if (ua.includes('qqbrowser')) return 'qq';
if (ua.includes('2345explorer')) return '2345';
if (ua.includes('360') || ua.includes('qihu')) return '360';
if (ua.includes('maxthon')) return 'maxthon';
if (ua.includes('via')) return 'via';
if (ua.includes('waterfox')) return 'waterfox';
if (ua.includes('palemoon')) return 'palemoon';
if (ua.includes('torbrowser') || (ua.includes('firefox') && ua.includes('tor'))) return 'tor';
if (ua.includes('focus')) return 'firefox-focus';
if (ua.includes('firefox')) return 'firefox';
if (ua.includes('edg/')) return 'edge'; // Edge Chromium
if (ua.includes('opr/') || ua.includes('opera')) return 'opera';
if (ua.includes('brave')) return 'brave';
if (ua.includes('vivaldi')) return 'vivaldi';
if (ua.includes('yabrowser')) return 'yandex';
if (ua.includes('chrome')) return 'chrome';
if (ua.includes('safari') && !ua.includes('chrome')) return 'safari';
return 'other';
};
// 获取浏览器主页URL
const getHomePageUrl = () => {
switch (getBrowserType()) {
case 'firefox': return 'about:home';
case 'tor': return 'about:home'; // Tor uses Firefox's UI
case 'waterfox': return 'about:home'; // Waterfox mirrors Firefox
case 'palemoon': return 'about:home'; // Pale Moon custom but similar
case 'chrome': return 'chrome://newtab';
case 'edge': return 'edge://newtab';
case 'safari': return 'topsites://';
case 'opera': return 'opera://startpage';
case 'brave': return 'brave://newtab';
case 'vivaldi': return 'vivaldi://newtab';
case 'yandex': return 'yandex://newtab';
case 'arc': return 'arc://start'; // Arc’s default start page
case 'via': return 'via://home';
// Fallbacks for lesser-known or legacy browsers
case 'uc': return 'ucenterhome://';
case 'qq': return 'qbrowser://home';
case '360': return 'se://newtab';
case 'maxthon': return 'mx://newtab';
case '2345': return '2345explorer://newtab';
default: return 'about:blank';
}
};
// ----------------- 配置项 -----------------
const config = {
// ================== 域名专用黑名单词汇 ==================
domainKeywords: {
// 常见成人网站域名关键词(权重4)
'pornhub': 4, 'xvideo': 4, 'redtube': 4,
'xnxx': 4, 'xhamster': 4, '4tube': 4,
'youporn': 4, 'spankbang': 4,
'myfreecams': 4, 'missav': 4,
'rule34': 4, 'youjizz': 4,
'onlyfans': 4, 'paidaa': 4,
'haijiao': 4,
// 核心违规词(权重3-4)
'porn': 3, 'nsfw': 3, 'hentai': 3,
'incest': 4, 'rape': 4, 'childporn': 4,
// 身体部位关键词(权重2)
'pussy': 2, 'cock': 2, 'dick': 2,
'boobs': 2, 'tits': 2, 'ass': 2,
'beaver': 1,
// 特定群体(权重2-3)
'cuckold': 3, 'virgin': 2, 'luoli': 2,
'gay': 2,
// 具体违规行为(权重2-3)
'blowjob': 3, 'creampie': 2,
'bdsm': 2, 'masturbat': 2, 'handjob': 3,
'footjob': 3, 'rimjob': 3,
// 其他相关词汇(权重1-2)
'camgirl': 2,
'nude': 3, 'naked': 3, 'upskirt': 2,
// 特定地区成人站点域名特征(权重4)
'jav': 4,
// 域名变体检测(权重3)
'p0rn': 3, 'pr0n': 3, 'pron': 3,
's3x': 3, 'sexx': 3,
},
// ================== 内容检测关键词 ==================
contentKeywords: {
// 核心违规词(权重3-4)- 严格边界检测
'\\b(?:po*r*n|pr[o0]n)\\b': 3, // porn及其变体
'nsfw': 3,
'\\bhentai\\b': 3,
'\\binces*t\\b': 4,
'\\br[a@]pe\\b': 4,
'(?:child|kid|teen)(?:po*r*n)': 4,
'海角社区': 4,
// 身体部位关键词(权重2)- 优化边界和上下文检测
'puss(?:y|ies)\\b': 2,
'\\bco*ck(?:s)?(?!tail|roach|pit|er)\\b': 2, // 排除cocktail等
'\\bdick(?:s)?(?!ens|tionary|tate)\\b': 2, // 排除dickens等
'\\bb[o0]{2,}bs?\\b': 2,
'\\btits?\\b': 2,
'(?<!cl|gl|gr|br|m|b|h)ass(?:es)?(?!ign|et|ist|ume|ess|ert|embl|oci|ault|essment|emble|ume|uming|ured)\\b': 2, // 优化ass检测
'\\bbeaver(?!s\\s+dam)\\b': 1, // 排除海狸相关
// 特定群体(权重2-3)- 上下文敏感
'\\bteen(?!age\\s+mutant)\\b': 3, // 排除 Teenage Mutant
'\\bsis(?!ter|temp)\\b': 2, // 排除 sister, system
'\\bmilfs?\\b': 2,
'\\bcuck[o0]ld\\b': 3,
'\\bvirgins?(?!ia|\\s+islands?)\\b': 2, // 排除地名
'lu[o0]li': 2,
'\\bg[a@]y(?!lord|le|le\\s+storm)\\b': 2, // 排除人名
// 具体违规行为(权重2-3)- 严格检测
'\\banal(?!ys[it]|og)\\b': 3, // 排除analysis等
'\\bbl[o0]w\\s*j[o0]b\\b': 3,
'cream\\s*pie(?!\\s+recipe)\\b': 2, // 排除食物相关
'\\bbdsm\\b': 2,
'masturba?t(?:ion|e|ing)\\b': 2,
'\\bhand\\s*j[o0]b\\b': 3,
'\\bf[o0]{2}t\\s*j[o0]b\\b': 3,
'\\brim\\s*j[o0]b\\b': 3,
// 新增违规行为(权重2-3)
'\\bstr[i1]p(?:p(?:er|ing)|tease)\\b': 3,
'\\bh[o0]{2}ker(?:s)?\\b': 3,
'pr[o0]st[i1]tut(?:e|ion)\\b': 3,
'b[o0]{2}ty(?!\\s+call)\\b': 2, // 排除 booty call
'sp[a@]nk(?:ing)?\\b': 2,
'deepthroat': 3,
'bukk[a@]ke': 3,
'org(?:y|ies)\\b': 3,
'gangbang': 3,
'thr[e3]{2}s[o0]me': 2,
'c[u|v]msh[o0]t': 3,
'f[e3]tish': 2,
// 其他相关词汇(权重1-2)- 上下文敏感
'\\bcamgirls?\\b': 2,
'\\bwebcam(?!era)\\b': 2, // 排除webcamera
'\\ble[a@]ked(?!\\s+(?:pipe|gas|oil))\\b': 2, // 排除工程相关
'\\bf[a@]p(?:p(?:ing)?)?\\b': 2,
'\\ber[o0]tic(?!a\\s+books?)\\b': 1, // 排除文学相关
'\\besc[o0]rt(?!\\s+mission)\\b': 3, // 排除游戏相关
'\\bnude(?!\\s+color)\\b': 3, // 排除色彩相关
'n[a@]ked(?!\\s+juice)\\b': 3, // 排除品牌
'\\bupskirt\\b': 2,
'\\b[o0]nlyfans\\b': 3,
// 多语言支持 (按原有配置)
'情色': 3, '成人': 3, '做爱': 4,
'セックス': 3, 'エロ': 3, '淫': 4,
'секс': 3, 'порн': 3, '性爱': 3,
'無修正': 3, 'ポルノ': 3, 'порно': 3,
'色情': 3, '骚': 1, '啪啪': 2,
'自慰': 3, '口交': 3, '肛交': 3,
'吞精': 3, '诱惑': 1, '全裸': 3,
'内射': 3, '乳交': 3, '射精': 3,
'反差': 0.5, '调教': 1.5, '性交': 3,
'性奴': 3, '高潮': 0.3, '白虎': 0.8,
'少女': 0.1, '女友': 0.1, '狂操': 3,
'捆绑': 0.1, '约炮': 3, '鸡吧': 3,
'鸡巴': 3, '阴茎': 1, '阴道': 1,
'女优': 3, '裸体': 3, '男优': 3,
'乱伦': 3, '偷情': 2, '母狗': 3,
'内射': 4, '喷水': 0.8, '潮吹': 3,
'轮奸': 2, '少妇': 2, '熟女': 2,
// 新增中文词汇(更细致的分级)
'色情': 3, '情色': 3, '黄色': 2,
'淫(?:秽|荡|乱|贱|液|穴|水)': 4,
'肉(?:棒|根|穴|缝|臀|奶|体|欲)': 3,
'(?:巨|大|小|翘|白|圆|肥)(?:乳|臀|胸)': 2,
'(?:舔|添|吸|吮|插|干|操|草|日|艹)(?:穴|逼|屄|阴|蜜|菊|屌|鸡|肉)': 4,
'(?:销|骚|浪|淫)(?:魂|女|货|逼|贱|荡)': 3,
// 新增日语词汇
'オナニー': 3, // 自慰
'手コキ': 3, // 手淫
'パイズリ': 3, // 乳交
'中出し': 4, // 中出
'素人': 2, // 素人
'アヘ顔': 3, // 阿黑颜
'痴女': 3, // 痴女
'処女': 2, // 处女
// 新增韩语词汇
'섹스': 3, // 性
'야동': 3, // 成人视频
'자위': 2, // 自慰
'음란': 3, // 淫乱
'성인': 2, // 成人
'누드': 2, // 裸体
// 新兴词汇、变体、谐音、emoji(权重2-4)
// 英文新兴变体
'lewd': 2, 'fap': 2, 'simp': 2, 'thicc': 2, 'bussy': 2, 'sloot': 2, 'nut': 2, 'noods': 2, 'lewdies': 2,
'camwhore': 3, 'onlyfams': 3, 'fansly': 3, 'sugardaddy': 2, 'sugarbaby': 2, 'egirl': 2, 'eboy': 2,
// 谐音与变体
'pron': 3, 'prawn': 2, 'p0rn': 3, 'p*rn': 3, 's3x': 3, 'shex': 2, 'seggs': 2, 's3ggs': 2, 'sx': 2,
'lo1i': 3, 'l0li': 3, 'loli': 3, 'shota': 3, 'sh0ta': 3, 'sh0t4': 3, '萝莉': 3, '正太': 3,
// emoji
'🍆': 0.5, '👅': 0.5, '👙': 0.5, '👠': 0.5, '👄': 0.5, '🔞': 2,
// 新兴中文网络词
'涩涩': 2, '涩图': 2, '涩气': 2, '涩女': 2, '涩男': 2, '涩会': 2, '涩图群': 2, '涩图包': 2, '涩图控': 2,
'色批': 2, '色图': 2, '色气': 2, '色女': 2, '色男': 2, '色会': 2, '色图群': 2, '色图包': 2, '色图控': 2,
'约p': 3, '约啪': 3, '约炮': 3, '约x': 3, '约会炮': 3, '约会啪': 3, '约会p': 3, '约会x': 3,
// 日语新兴词
'エッチ': 2, 'えっち': 2, 'えちえち': 2, 'えち': 2, 'エロい': 2, 'エロ画像': 2, 'エロ動画': 2,
// 韩语新兴词
'야짤': 2, '야사': 2, '야한': 2, '야동': 3, '야설': 2,
},
// ================== 白名单减分规则 ==================
whitelist: {
// 强豁免词(权重-30)
'edu': -30, 'health': -30, 'medical': -30, 'science': -30,
'gov': -30, 'org': -30, 'official': -30,
// 常用场景豁免(权重-15)
'academy': -15, 'clinic': -15, 'therapy': -15,
'university': -4, 'research': -15, 'news': -15,
'dictionary': -15, 'library': -15, 'museum': -15,
// 动物/自然相关(权重-1)
'animal': -4, 'zoo': -1, 'cat': -1, 'dog': -1,
'pet': -6, 'bird': -1, 'vet': -1,
// 科技类(权重-5)
'tech': -5, 'cloud': -5, 'software': -5, 'cyber': -3,
// 支持正则和后缀
'/\\.edu$/': -30, '/\\.gov$/': -30, '/\\.org$/': -30, '/\\.ac\\./': -20,
// 在线聊天/论坛豁免
'telegram': -20, 'discord': -20, 'slack': -20, 'line': -20, 'whatsapp': -20, 'skype': -20, 'teams': -20, 'twitter': -20, 'facebook': -20,
'forum': -10, 'bbs': -10, 'reddit': -10, 'tieba': -10, '知乎': -10, '豆瓣': -10, 'quora': -10, 'stack': -10, 'stackoverflow': -10,
},
// ================== 阈值配置 ==================
thresholds: {
// 总分触发阈值(建议3~4)
block: 3,
// URL路径加分阈值
path: 2,
// 进行白名单减分的最低阈值
whitelist: 2
},
// ================== 域名正则表达式规则 ==================
domainPatterns: [
/^mogu\d+\.[a-z]{2,3}$/i,
/^mg\d+\.app$/i,
],
// ================== 内容检测规则 ==================
// 需要内容检测的域名规则
contentCheckDomains: [
/\d{3}[a-z]{2,3}/i,
/[a-z]{2,3}\d{2,3}/i,
/[a-z]{1,3}\d{1,3}[a-z]{1,3}/i,
// 海角社区
/^[a-z0-9]{0,5}(\.[a-z0-5]{0,5})?h[a-z0-5]{0,5}j[a-z0-5]{0,5}(\.[a-z]{0,5})?/i,
],
// 内容检测相关配置
contentCheck: {
// 成人内容分数
adultContentThreshold: 25,
suspiciousTagNames: [
// 主要内容区域
'article', 'main', 'section', 'content',
// 文本块
'p', 'h1', 'h2', 'h3', 'h4', 'h5', 'h6',
// 列表和表格
'li', 'td', 'th', 'figcaption',
// 链接和按钮文本
'a', 'button',
// 通用容器
'div.content', 'div.text', 'div.description',
'span.text', 'span.content'
],
// 文本节点最小长度
textNodeMinLength: 5,
// 防抖等待时间(毫秒)
debounceWait: 1000,
// 观察者最大运行时间(毫秒)
observerTimeout: 30000,
// 添加局部内容检测配置
localizedCheck: {
// 单个元素的内容阈值,超过此值才会影响整体评分
elementThreshold: 8,
// 需要触发的违规元素数量
minViolationCount: 3,
// 违规内容占总内容的比例阈值
violationRatio: 0.3,
// 排除检测的元素
excludeSelectors: [
'.comment', '.reply', '.user-content',
'[id*="comment"]', '[class*="comment"]',
'[id*="reply"]', '[class*="reply"]',
'.social-feed', '.user-post'
],
// 高风险元素选择器(权重更高)
highRiskSelectors: [
'article', 'main', '.main-content',
'.article-content', '.post-content'
]
}
},
// ================== 搜索引擎白名单 ==================
searchEngines: [
'google.com', 'bing.com', 'baidu.com', 'yahoo.com', 'duckduckgo.com',
'yandex.com', 'so.com', 'sogou.com', 'sm.cn', 'search.brave.com',
'ecosia.org', 'qwant.com', 'searx', 'startpage.com', 'you.com',
'naver.com', 'daum.net', 'ask.com', 'aol.com', 'dogpile.com',
'gibiru.com', 'mojeek.com', 'metager.org', 'swisscows.com',
'search.com', 'search.yahoo.com', 'search.aol.com', 'search.naver.com',
'search.daum.net', 'search.sogou.com', 'search.sm.cn', 'search.yandex.com',
'search.ecosia.org', 'search.qwant.com', 'search.searx', 'search.startpage.com',
'search.you.com', 'search.brave.com', 'search.metager.org', 'search.swisscows.com'
],
// ================== 误判正则白名单 ==================
falsePositiveRegexList: [
/cocktail/i, /class/i, /classic/i, /associate/i, /assignment/i, /passage/i, /passion/i, /pass/i, /mass/i, /massive/i, /dickens/i, /dickinson/i, /analysis/i, /analogy/i, /webcamera/i, /booty call/i, /virginia/i, /virgin islands/i, /teenage mutant/i, /system/i, /sister/i, /mission/i, /juice/i, /color/i, /pipe/i, /gas/i, /oil/i, /roach/i, /pit/i, /er/i, /tate/i, /ens/i, /dictionary/i, /museum/i, /library/i, /academy/i, /clinic/i, /therapy/i, /research/i, /news/i, /animal/i, /zoo/i, /cat/i, /dog/i, /pet/i, /bird/i, /vet/i, /tech/i, /cloud/i, /software/i, /cyber/i, /gov/i, /org/i, /official/i, /edu/i, /health/i, /medical/i, /science/i
],
};
// 预编译 domainKeywords、whitelist、falsePositiveRegexList
const compiledDomainRegexes = Object.keys(config.domainKeywords).map(k => new RegExp(`\\b${k}\\b`, 'gi'));
const compiledWhitelistRegexes = Object.keys(config.whitelist).map(k => {
if (k.startsWith('/') && k.endsWith('/')) {
return new RegExp(k.slice(1, -1), 'i');
} else {
return new RegExp(`\\b${k}\\b`, 'i');
}
});
// ----------------- 工具函数 -----------------
function isWhitelisted(text) {
let i = 0;
for (const [w, wv] of Object.entries(config.whitelist)) {
const reg = compiledWhitelistRegexes[i++];
if (reg.test(text)) return wv;
}
return 0;
}
const compiledContentRegexes = Object.entries(config.contentKeywords).map(([k, v]) => ({ regex: new RegExp(k, 'i'), weight: v, raw: k }));
// 只检测可见文本
function getAllVisibleText(element) {
if (!element) return "";
const textSet = new Set();
try {
const walker = document.createTreeWalker(
element,
NodeFilter.SHOW_TEXT,
{
acceptNode: (node) => {
const parent = node.parentElement;
if (!parent ||
/^(SCRIPT|STYLE|NOSCRIPT|IFRAME|META|LINK)$/i.test(parent.tagName) ||
parent.hidden ||
getComputedStyle(parent).display === 'none' ||
getComputedStyle(parent).visibility === 'hidden' ||
getComputedStyle(parent).opacity === '0') {
return NodeFilter.FILTER_REJECT;
}
const text = node.textContent.trim();
if (!text || text.length < config.contentCheck.textNodeMinLength) {
return NodeFilter.FILTER_REJECT;
}
return NodeFilter.FILTER_ACCEPT;
}
}
);
let node;
while (node = walker.nextNode()) {
textSet.add(node.textContent.trim());
}
} catch (e) { }
return Array.from(textSet).join(' ');
}
// 评分函数
function calculateScore(text, isDomain = false) {
if (!text) return 0;
const white = isWhitelisted(text);
if (white) return white;
for (const fp of config.falsePositiveRegexList) {
if (fp.test(text)) return 0;
}
let score = 0;
if (isDomain) {
let i = 0;
for (const [k, v] of Object.entries(config.domainKeywords)) {
const reg = compiledDomainRegexes[i++];
const matches = text.match(reg);
if (matches) score += v * matches.length;
}
} else {
for (const { regex, weight, raw } of compiledContentRegexes) {
const matches = text.match(regex);
if (matches) {
let contextSafe = false;
for (const w of Object.keys(config.whitelist)) {
if (text.match(new RegExp(`.{0,10}${w}.{0,10}${raw}|${raw}.{0,10}${w}.{0,10}`, 'i'))) {
contextSafe = true;
break;
}
}
if (!contextSafe) score += weight * matches.length;
}
}
}
return score;
}
// 主内容检测
function detectAdultContent() {
let totalScore = 0;
let violationCount = 0;
const mainSelectors = ['main', 'article', '.main-content', '.article-content', '.post-content', '#main', '#content'];
let mainElements = [];
for (const sel of mainSelectors) {
mainElements = mainElements.concat(Array.from(document.querySelectorAll(sel)));
}
if (mainElements.length === 0) mainElements = [document.body];
let mainRisk = false;
for (const el of mainElements) {
const text = getAllVisibleText(el).slice(0, 2000);
const score = calculateScore(text);
if (score >= config.contentCheck.localizedCheck.elementThreshold) mainRisk = true;
totalScore += score;
}
if (!mainRisk) {
const globalText = getAllVisibleText(document.body).slice(0, 2000);
const globalScore = calculateScore(globalText);
if (globalScore >= config.contentCheck.localizedCheck.elementThreshold) violationCount++;
totalScore += globalScore;
}
// 图片alt/title检测
const images = document.querySelectorAll('img[alt], img[title]');
for (const img of images) {
const imgText = `${img.alt} ${img.title}`.trim();
if (imgText) {
const score = calculateScore(imgText);
if (score >= 3) violationCount++;
totalScore += score * 0.3;
}
}
// 元数据检测
const metaTags = document.querySelectorAll('meta[name="description"], meta[name="keywords"]');
for (const meta of metaTags) {
const content = meta.content;
if (content) {
const score = calculateScore(content);
if (score >= 3) violationCount++;
totalScore += score * 0.2;
}
}
if (violationCount > 0 || mainRisk) return true;
return totalScore >= config.contentCheck.adultContentThreshold;
}
// 动态内容检测增强
function enhancedDynamicContentCheck() {
let checkCount = 0;
const maxChecks = 10;
const interval = 2000;
const hostname = window.location.hostname;
function tryCheck() {
if (detectAdultContent()) {
blacklistManager.addToBlacklist(hostname, 'dynamic-content');
handleBlockedContent();
return;
}
checkCount++;
if (checkCount < maxChecks) {
setTimeout(tryCheck, interval);
}
}
setTimeout(tryCheck, 1000);
}
// GM_value 封装
async function gmGet(key, def) {
if (typeof GM_getValue === 'function') {
const v = await GM_getValue(key);
return v === undefined ? def : v;
}
return def;
}
async function gmSet(key, value) {
if (typeof GM_setValue === 'function') {
await GM_setValue(key, value);
}
}
// 黑名单管理器(GM_setValue/GM_getValue实现)
const blacklistManager = {
BLACKLIST_KEY: 'pornblocker-blacklist',
BLACKLIST_VERSION_KEY: 'pornblocker-blacklist-version',
CURRENT_VERSION: '3.0', // 升级数据库版本,弃用旧数据
// 只在版本号不一致时清空旧数据
async checkAndUpgradeVersion() {
const storedVersion = await gmGet(this.BLACKLIST_VERSION_KEY, null);
if (storedVersion !== this.CURRENT_VERSION) {
await gmSet(this.BLACKLIST_VERSION_KEY, this.CURRENT_VERSION);
await gmSet(this.BLACKLIST_KEY, []);
}
},
// 获取黑名单
async getBlacklist() {
// 确保版本检查已完成
await this.checkAndUpgradeVersion();
let data = await gmGet(this.BLACKLIST_KEY, []);
// 自动清理过期和升级结构
const now = Date.now();
let changed = false;
const valid = (Array.isArray(data) ? data : []).filter(item => {
if (typeof item === 'string') return true; // 兼容老数据
if (item && item.host && item.expire && item.expire > now) return true;
changed = true;
return false;
}).map(item => {
if (typeof item === 'string') {
changed = true;
return createBlacklistEntry(item, 'legacy', '自动升级');
}
// 结构升级:补全缺失字段
if (!item.version) item.version = this.CURRENT_VERSION;
if (!item.added) item.added = now;
if (!item.reason) item.reason = '';
if (!item.note) item.note = '';
return item;
});
if (changed) {
this.saveBlacklist(valid);
}
return valid;
},
async saveBlacklist(list) {
await gmSet(this.BLACKLIST_KEY, list);
},
async addToBlacklist(hostname, reason = '', note = '') {
if (!hostname) return false;
// 搜索引擎白名单,禁止加入黑名单
if (config.searchEngines.some(domain => hostname === domain || hostname.endsWith('.' + domain))) {
return false;
}
let list = await this.getBlacklist();
if (list.some(item => (typeof item === 'string' ? item : item.host) === hostname)) return true;
list.push(createBlacklistEntry(hostname, reason, note));
await this.saveBlacklist(list);
return true;
},
async isBlacklisted(hostname) {
let list = await this.getBlacklist();
return list.some(item => (typeof item === 'string' ? item : item.host) === hostname);
},
async removeFromBlacklist(hostname) {
let list = await this.getBlacklist();
list = list.filter(item => (typeof item === 'string' ? item : item.host) !== hostname);
await this.saveBlacklist(list);
return true;
},
// 新增批量清理过期条目方法
async cleanExpired() {
let list = await this.getBlacklist();
const now = Date.now();
const valid = list.filter(item => (typeof item === 'string') || (item && item.expire && item.expire > now));
await this.saveBlacklist(valid);
return valid.length;
}
};
// 数据库结构优化:黑名单支持更多元数据,未来可扩展
// 支持来源、拦截原因、添加时间、过期时间、用户备注等
// 统一黑名单条目结构
function createBlacklistEntry(host, reason = '', note = '') {
return {
host,
reason,
note,
added: Date.now(),
expire: getExpireTimestamp(),
version: blacklistManager.CURRENT_VERSION
};
}
// 立即执行版本检查
(async function initBlacklist() {
await blacklistManager.checkAndUpgradeVersion();
})();
function getExpireTimestamp() {
const BLACKLIST_EXPIRE_DAYS = 30;
return Date.now() + BLACKLIST_EXPIRE_DAYS * 24 * 60 * 60 * 1000;
}
const regexCache = {
domainRegex: new RegExp(Object.keys(config.domainKeywords).join('|'), 'gi'),
whitelistRegex: new RegExp(Object.keys(config.whitelist).join('|'), 'gi'),
xxxRegex: /\.xxx$/i
};
function checkDomainPatterns(hostname) {
return config.domainPatterns.some(pattern => pattern.test(hostname));
}
function shouldCheckContent(hostname) {
return config.contentCheckDomains.some(pattern => pattern.test(hostname));
}
// ----------------- 主检测逻辑 -----------------
async function checkUrl() {
const url = new URL(window.location.href);
const hostname = url.hostname;
if (await blacklistManager.isBlacklisted(hostname)) {
return { shouldBlock: true, url, reason: 'blacklist' };
}
if (checkDomainPatterns(url.hostname)) {
await blacklistManager.addToBlacklist(hostname, 'domain-pattern');
return { shouldBlock: true, url, reason: 'domain-pattern' };
}
for (const w of Object.keys(config.whitelist)) {
if (hostname.match(new RegExp(`\\b${w}\\b`, 'i')) || (document.title || '').match(new RegExp(`\\b${w}\\b`, 'i'))) {
return { shouldBlock: false, url };
}
}
if (shouldCheckContent(url.hostname)) {
if (document.body) {
if (detectAdultContent()) {
await blacklistManager.addToBlacklist(hostname, 'content');
return { shouldBlock: true, url, reason: 'content' };
}
enhancedDynamicContentCheck();
} else {
document.addEventListener('DOMContentLoaded', () => {
if (detectAdultContent()) {
blacklistManager.addToBlacklist(hostname, 'content');
handleBlockedContent();
}
enhancedDynamicContentCheck();
});
}
}
let score = 0;
const pornMatches = url.hostname.match(regexCache.domainRegex) || [];
pornMatches.forEach(match => {
const keyword = match.toLowerCase();
const domainScore = config.domainKeywords[keyword] || 0;
if (domainScore !== 0) score += domainScore;
});
const path = url.pathname + url.search;
score += calculateScore(path) * 0.4;
score += calculateScore(document.title || "");
if (score >= config.thresholds.whitelist) {
const hostMatches = url.hostname.match(regexCache.whitelistRegex) || [];
const titleMatches = (document.title || "").match(regexCache.whitelistRegex) || [];
let whitelistScore = 0;
const whitelistMatchCount = (matches) => {
matches.forEach(match => {
const term = match.toLowerCase();
const reduction = config.whitelist[term] || 0;
whitelistScore += reduction;
});
};
whitelistMatchCount(hostMatches);
whitelistMatchCount(titleMatches);
score += whitelistScore;
}
return { shouldBlock: score >= config.thresholds.block, url };
}
// 检测结果处理函数
const handleBlockedContent = () => {
const lang = getUserLanguage();
const text = i18n[lang];
document.title = text.title;
window.stop();
document.documentElement.innerHTML = `
<body>
<div class="container">
<div class="card">
<div class="icon-wrapper">
<svg width="32" height="32" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<path d="M12 9v2m0 4h.01m-6.938 4h13.856c1.54 0 2.502-1.667 1.732-3L13.732 4c-.77-1.333-2.694-1.333-3.464 0L3.34 16c-.77 1.333.192 3 1.732 3z"/>
</svg>
</div>
<h1>${text.title}</h1>
<p>${text.message}<br>${text.redirect}</p>
<div class="footer">${text.footer}</div>
</div>
</div>
<style>
:root {
--bg-light: #f0f2f5;
--card-light: #ffffff;
--text-light: #2d3436;
--text-secondary-light: #636e72;
--text-muted-light: #b2bec3;
--accent-light: #ff4757;
--bg-dark: #1a1a1a;
--card-dark: #2d2d2d;
--text-dark: #ffffff;
--text-secondary-dark: #a0a0a0;
--text-muted-dark: #808080;
--accent-dark: #ff6b6b;
}
@media (prefers-color-scheme: dark) {
body {
background: var(--bg-dark) !important;
}
.card {
background: var(--card-dark) !important;
box-shadow: 0 4px 12px rgba(0,0,0,0.3) !important;
}
h1 { color: var(--text-dark) !important; }
p { color: var(--text-secondary-dark) !important; }
.footer { color: var(--text-muted-dark) !important; }
.icon-wrapper {
background: var(--accent-dark) !important; }
.countdown {
color: var(--accent-dark);
}
}
body {
background: var(--bg-light);
margin: 0;
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, sans-serif;
min-height: 100vh;
display: flex;
align-items: center;
justify-content: center;
}
.container {
max-width: 500px;
width: 100%;
}
.card {
background: var(--card-light);
border-radius: 16px;
box-shadow: 0 4px 12px rgba(0,0,0,0.1);
padding: 32px;
text-align: center;
animation: slideIn 0.5s ease-out;
}
.icon-wrapper {
width: 64px;
height: 64px;
background: var(--accent-light);
border-radius: 50%;
display: flex;
align-items: center;
justify-content: center;
margin: 0 auto 24px;
animation: pulse 2s infinite;
}
.icon-wrapper svg {
stroke: white;
}
h1 {
color: var(--text-light);
margin: 0 0 16px;
font-size: 24px;
font-weight: 600;
}
p {
color: var(--text-secondary-light);
margin: 0 0 24px;
line-height: 1.6;
font-size: 16px;
}
.footer {
color: var(--text-muted-light);
font-size: 14px;
animation: fadeIn 1s ease-out;
}
.countdown {
font-weight: bold;
color: var(--accent-light);
}
@keyframes slideIn {
from { transform: translateY(20px); opacity: 0; }
to { transform: translateY(0); opacity: 1; }
}
@keyframes pulse {
0% { transform: scale(1); }
50% { transform: scale(1.05); }
100% { transform: scale(1); }
}
@keyframes fadeIn {
from { opacity: 0; }
to { opacity: 1; }
}
</style>
</body>
`;
let timeLeft = 4;
const countdownEl = document.querySelector('.countdown');
const countdownInterval = setInterval(() => {
timeLeft--;
if (countdownEl) countdownEl.textContent = timeLeft;
if (timeLeft <= 0) {
clearInterval(countdownInterval);
try {
const homeUrl = getHomePageUrl();
if (window.history.length > 1) {
const iframe = document.createElement('iframe');
iframe.style.display = 'none';
document.body.appendChild(iframe);
iframe.onload = () => {
try {
const prevUrl = iframe.contentWindow.location.href;
const prevScore = calculateScore(new URL(prevUrl).hostname, true);
if (prevScore >= config.thresholds.block) {
window.location.href = homeUrl;
} else {
window.history.back();
}
} catch (e) {
window.location.href = homeUrl;
}
document.body.removeChild(iframe);
};
iframe.src = 'about:blank';
} else {
window.location.href = homeUrl;
}
} catch (e) {
window.location.href = getHomePageUrl();
}
}
}, 1000);
};
// setupDynamicContentCheck 函数之前添加新函数
const setupTitleObserver = () => {
let titleObserver = null;
try {
// 监听 title 标签变化
const titleElement = document.querySelector('title');
if (titleElement) {
titleObserver = new MutationObserver(async (mutations) => {
for (const mutation of mutations) {
const newTitle = mutation.target.textContent;
console.log(`[Title Change] New title: "${newTitle}"`);
// 计算新标题的分数
const titleScore = calculateScore(newTitle || "");
if (titleScore >= config.thresholds.block) {
console.log(`[Title Score] ${titleScore} exceeds threshold`);
const hostname = window.location.hostname;
await blacklistManager.addToBlacklist(hostname);
titleObserver.disconnect();
handleBlockedContent();
return;
}
}
});
titleObserver.observe(titleElement, {
subtree: true,
characterData: true,
childList: true
});
}
// 监听 title 标签的添加
const headObserver = new MutationObserver((mutations) => {
for (const mutation of mutations) {
for (const node of mutation.addedNodes) {
if (node.nodeName === 'TITLE') {
setupTitleObserver();
headObserver.disconnect();
return;
}
}
}
});
headObserver.observe(document.head, {
childList: true,
subtree: true
});
// 设置超时清理
setTimeout(() => {
titleObserver?.disconnect();
headObserver?.disconnect();
}, config.contentCheck.observerTimeout);
} catch (e) {
console.error('Error in setupTitleObserver:', e);
}
return titleObserver;
};
// ----------------- 主检测逻辑 -----------------
(async function () {
const { shouldBlock, url: currentUrl } = await checkUrl();
if (shouldBlock || regexCache.xxxRegex.test(currentUrl.hostname)) {
handleBlockedContent();
} else {
// 添加标题监听
setupTitleObserver();
}
})();
// 可选:定期自动清理过期黑名单(每天一次)
(function autoCleanBlacklist() {
try {
const key = 'pornblocker-last-clean';
const now = Date.now();
let last = 0;
try { last = parseInt(localStorage.getItem(key) || '0', 10); } catch (e) { }
if (!last || now - last > 86400000) {
blacklistManager.cleanExpired();
localStorage.setItem(key, now.toString());
}
} catch (e) { }
})();
})();