// ==UserScript==
// @name Yet Another Weibo Filter
// @namespace https://github.com/tiansh
// @description 新浪微博根据关键词、作者、话题、来源等过滤微博;修改版面。 新浪微博根據關鍵字、作者、話題、來源等篩選微博;修改版面。 filter Sina Weibo by keywords, original, topic, source, etc.; modify layout
// @include http://weibo.com/*
// @include http://www.weibo.com/*
// @version 0.1.27 alpha
// @author 田生
// @copyright 2013+, 田生
// @license The MIT License (MIT); http://opensource.org/licenses/MIT
// @grant GM_xmlhttpRequest
// @grant GM_setValue
// @grant GM_getValue
// @grant GM_addStyle
// @grant GM_registerMenuCommand
// @grant GM_info
// @grant unsafeWindow
// @run-at document-start
// @downloadURL none
// ==/UserScript==
// 图片
var images = {
'filter': '',
};
// 多行字符串
var funcStr = function (f) {
var s = f.toString().split(/\r\n|\r|\n/g).slice(1, -1).join('\n');
return s;
};
// 快速创建一段文档元素
var cewih = function (tag, inner) {
var d = document.createElement(tag);
d.innerHTML = inner;
return d;
};
// 文本常量
// 请以简体中文为原文,参考这些资料翻译
// http://zh.wikipedia.org/wiki/Template:CGroup/IT
// http://www.microsoft.com/Language/zh-cn/Search.aspx
var text = {
// 基本按钮
'okButtonTitle': { 'zh-cn': '确定', 'zh-hk': '確定', 'zh-tw': '確定', 'en': 'Confirm' },
'cancelButtonTitle': { 'zh-cn': '取消', 'zh-hk': '取消', 'zh-tw': '取消', 'en': 'Cancel' },
'closeButtonTitle': { 'zh-cn': '关闭', 'zh-hk': '關閉', 'zh-tw': '關閉', 'en': 'Close' },
'configStringsAdd': { 'zh-cn': '添加', 'zh-hk': '新增', 'zh-tw': '新增', 'en': 'Add' },
'configUsersAdd': { 'zh-cn': '添加', 'zh-hk': '新增', 'zh-tw': '新增', 'en': 'Add' },
'foldedWeiboText': { 'zh-cn': '"来自 @" attr(yawf-author) " 的一条微博被折叠,请点击查看"', 'zh-hk': '"來自 @" attr(yawf-author) " 的一條微博被折疊,請點擊查看"', 'zh-tw': '"來自 @" attr(yawf-author) " 的一條微博被折疊,請點擊查看"', 'en': '"A Weibo from @" attr(yawf-author) " was folded, click to view."' },
// 设置框
'filter': { 'zh-cn': '过滤器', 'zh-hk': '篩選器', 'zh-tw': '篩選器', 'en': 'Filter' },
'configDialogTitle': { 'zh-cn': '过滤器设置', 'zh-hk': '篩選器設定', 'zh-tw': '篩選器設定', 'en': 'Filter Settings' },
'configRefreshNotice': { 'zh-cn': '部分设置修改需要保存后重新载入页面才能生效', 'zh-hk': '部分設定變更需要儲存後重新載入網頁才能生效', 'zh-tw': '部分設定變更需要儲存後重新載入網頁才能生效', 'en': 'Some settings need refresh webpage after save to apply' },
'whitelistFilterDesc': { 'zh-cn': '总是显示{{{typed}}}', 'zh-hk': '總是顯示{{{typed}}}', 'zh-tw': '總是顯示{{{typed}}}', 'en': 'Always show {{{typed}}}' },
'blacklistFilterDesc': { 'zh-cn': '隐藏{{{typed}}}', 'zh-hk': '隱藏{{{typed}}}', 'zh-tw': '隱藏{{{typed}}}', 'en': 'Hide {{{typed}}}' },
'foldlistFilterDesc': { 'zh-cn': '折叠{{{typed}}}', 'zh-hk': '折疊{{{typed}}}', 'zh-tw': '折疊{{{typed}}}', 'en': 'Fold {{{typed}}}' },
'whitelistActionDesc': { 'zh-cn': '显示', 'zh-hk': '顯示', 'zh-tw': '顯示', 'en': 'Show' },
'blacklistActionDesc': { 'zh-cn': '隐藏', 'zh-hk': '隱藏', 'zh-tw': '隱藏', 'en': 'Hide' },
'foldlistActionDesc': { 'zh-cn': '折叠', 'zh-hk': '折疊', 'zh-tw': '折疊', 'en': 'Fold' },
// 关键词
'keywordFilterGroupTitle': { 'zh-cn': '内容', 'zh-hk': '內容', 'zh-tw': '內容', 'en': 'Content' },
'keywordFilterDesc': { 'zh-cn': '关键词', 'zh-hk': '關鍵字', 'zh-tw': '關鍵字', 'en': 'Keyword' },
'keywordFilterDetails': { 'zh-cn': '包含以下关键词的微博', 'zh-hk': '包含以下關鍵字的微博', 'zh-tw': '包含以下關鍵字的微博', 'en': 'Weibo with these keywords' },
'keywordFilterFast': { 'zh-cn': '包含关键词“{{text}}”的微博', 'zh-hk': '包含關鍵字“{{text}}”的微博', 'zh-tw': '包含關鍵字“{{text}}”的微博', 'en': 'Weibo contains keyword "{{text}}"' },
// 正则表达式
'regexpFilterGroupTitle': { 'zh-cn': '正则', 'zh-hk': '正則', 'zh-tw': '正規', 'en': 'Regexp' },
'regexpFilterDesc': { 'zh-cn': '正则式', 'zh-hk': '正則式', 'zh-tw': '正規式', 'en': 'Regexp' },
'regexpFilterDetails': { 'zh-cn': '匹配以下正则表达式的微博', 'zh-hk': '匹配以下正則表達式的微博', 'zh-tw': '匹配以下正規表示式的微博', 'en': 'Weibo matches these regular expressions' },
'regexpFilterRemark': { 'zh-cn': '书写时不需要对“/”字符转义', 'zh-hk': '書寫時不需要對“/”字符轉義', 'zh-tw': '書寫時不需要對“/”字符轉義', 'en': 'Do not escape "/" in your regexp.' },
'regexpBadFormedTitle': { 'zh-cn': '非法的正则表达式', 'zh-hk': '不合法的正則表達式', 'zh-tw': '不合法的正規表示式', 'en': 'Illegal Regexp' },
'regexpBadFormed': { 'zh-cn': '您输入的/{{regexp}}/不能被正确地解析为正则表达式,请检查您的输入。如需关键词屏蔽请到内容标签页设置。', 'zh-hk': '您输入的/{{regexp}}/不能被正确地解析为正则表达式,请检查您的输入。如需关键词屏蔽请到内容标签页面设置。', 'zh-tw': '您輸入的/{{regexp}}/不能被正確地解析為正規表示式,請檢查您的輸入。如需關鍵詞屏蔽請到內容標籤頁面設置。', 'en': 'Can not parse /{{regexp}}/ as regexp. Please check your input. You may hide Weibo by keywords in Content tab page.' },
// 帐号
'accountFilterGroupTitle': { 'zh-cn': '帐号', 'zh-hk': '帳號', 'zh-tw': '帳號', 'en': 'Account' },
'accountFilterDesc': { 'zh-cn': '帐号', 'zh-hk': '帳號', 'zh-tw': '帳號', 'en': 'Account' },
'accountFilterDetails': { 'zh-cn': '来自以下帐号的微博', 'zh-hk': '來自以下帳號的微博', 'zh-tw': '來自以下帳號的微博', 'en': 'Weibo from these accounts' },
'accountFilterFast': { 'zh-cn': '作者是“@{{name}}”的微博', 'zh-hk': '作者是“@{{name}}”的微博', 'zh-tw': '作者是“@{{name}}”的微博', 'en': 'Weibo from "@{{name}}"', },
'accountFilterRemark': {
'zh-cn': '推荐您到隐私设置 - 屏蔽账号 隐藏您关注了但不想在首页看到的账号。',
'zh-hk': '推薦您到隱私設置 - 屏蔽帐号 隱藏您關注了但不想在首頁看到的帳號。',
'zh-tw': '推薦您到隱私設置 - 屏蔽帐号 隱藏您關注了但不想在首頁看到的帳號。',
'en': 'You can hide Weibo from accounts you followed in the page Privacy - Block account .'
},
'accountNotExistErrorTitle': { 'zh-cn': '帐号不存在', 'zh-hk': '帳號不存在', 'zh-tw': '帳號不存在', 'en': 'Account does not exist' },
'accountNotExistError': { 'zh-cn': '不存在名为{{name}}的账号', 'zh-hk': '不存在名為{{name}}的賬號', 'zh-tw': '不存在名為{{name}}的賬號', 'en': 'Account named {{name}} does not exist' },
// 原创
'originalFilterGroupTitle': { 'zh-cn': '原创', 'zh-hk': '原創', 'zh-tw': '原創', 'en': 'Original' },
'originalFilterDesc': { 'zh-cn': '帐号', 'zh-hk': '帳號', 'zh-tw': '帳號', 'en': 'Account' },
'originalFilterDetails': { 'zh-cn': '转发自以下账号的微博', 'zh-hk': '隱藏轉發自以下帳號的微博', 'zh-tw': '隱藏轉發自以下帳號的微博', 'en': 'Hide Weibo forwarded from these accounts' },
'originalFilterFast': { 'zh-cn': '转发自“@{{name}}”的微博', 'zh-hk': '轉發自“@{{name}}”的微博', 'zh-tw': '轉發自“@{{name}}”的微博', 'en': 'Weibo forwarded from "@{{name}}"' },
// 提到
'mentionFilterGroupTitle': { 'zh-cn': '提到', 'zh-hk': '提到', 'zh-tw': '提到', 'en': 'Mention' },
'mentionFilterDesc': { 'zh-cn': '帐号', 'zh-hk': '帳號', 'zh-tw': '帳號', 'en': 'Account' },
'mentionFilterDetails': { 'zh-cn': '提到以下账号的微博', 'zh-hk': '提到以下帳號的微博', 'zh-tw': '提到以下帳號的微博', 'en': 'Weibo mentioned these accounts' },
'mentionFilterFast': { 'zh-cn': '提到了“@{{name}}”的微博', 'zh-hk': '提到了“@{{name}}”的微博', 'zh-tw': '提到了“@{{name}}”的微博', 'en': 'Weibo mentioned "@{{name}}"' },
// 话题
'topicFilterGroupTitle': { 'zh-cn': '话题', 'zh-hk': '話題', 'zh-tw': '話題', 'en': 'Topic' },
'topicFilterDesc': { 'zh-cn': '话题', 'zh-hk': '話題', 'zh-tw': '話題', 'en': 'Topic' },
'topicFilterDetails': { 'zh-cn': '包含以下话题的微博', 'zh-hk': '包含以下話題的微博', 'zh-tw': '包含以下話題的微博', 'en': 'Weibo with these topics' },
'topicFilterFast': { 'zh-cn': '包含“#{{topic}}#”话题的微博', 'zh-hk': '包含“#{{topic}}#”話題的微博', 'zh-tw': '包含“#{{topic}}#”話題的微博', 'en': 'Weibo contains topic "#{{topic}}#"' },
// 来源
'sourceFilterGroupTitle': { 'zh-cn': '来源', 'zh-hk': '來源', 'zh-tw': '來源', 'en': 'Source' },
'sourceFilterDesc': { 'zh-cn': '来自', 'zh-hk': '來自', 'zh-tw': '來自', 'en': 'Via' },
'sourceFilterDetails': { 'zh-cn': '以下来源的微博', 'zh-hk': '以下來源的微博', 'zh-tw': '以下來源的微博', 'en': 'Weibo from these sources' },
'sourceFilterFast': { 'zh-cn': '来自“{{source}}”的微博', 'zh-hk': '來自“{{source}}”的微博', 'zh-tw': '來自“{{source}}”的微博', 'en': 'Weibo via "{{source}}"' },
'sourceFilterWarningTitle': { 'zh-cn': '默认来源', 'zh-hk': '預設來源', 'zh-tw': '預設來源', 'en': 'Default Source' },
'sourceFilterWarning': { 'zh-cn': '不能添加默认来源', 'zh-hk': '不能新增預設來源', 'zh-tw': '不能新增預設來源', 'en': 'You cannot add default source' },
// 超链接
'hyperlinkFilterGroupTitle': { 'zh-cn': '链接', 'zh-hk': '連結', 'zh-tw': '連結', 'en': 'Link' },
'hyperlinkFilterDesc': { 'zh-cn': '超链接', 'zh-hk': '超連結', 'zh-tw': '超連結', 'en': 'Hyperlink' },
'hyperlinkFilterDetails': { 'zh-cn': '包含指向以下网站的超链接的微博', 'zh-hk': '包含指向以下站點的超連結的微博', 'zh-tw': '包含指向以下站點的超連結的微博', 'en': 'Weibo with hyperlink to these website' },
'hyperlinkFilterFast': { 'zh-cn': '包含指向包含“{{host}}”地址链接的微博', 'zh-hk': '包含指向包含“{{host}}”位址連結的微博', 'zh-tw': '包含指向包含“{{host}}”位址連結的微博', 'en': 'Weibo contains hyperlink to "{{host}}"' },
// 更多
'otherFilterGroupTitle': { 'zh-cn': '更多', 'zh-hk': '其他', 'zh-tw': '其他', 'en': 'More' },
// 显示
'otherWhitelistTitle': { 'zh-cn': '显示以下内容(不计入白名单)', 'zh-hk': '顯示以下內容(不計入白名單)', 'zh-tw': '顯示以下內容(不計入白名單)', 'en': 'Show following content (not regard as whitelist)' },
'showMyWeiboDesc': { 'zh-cn': '自己的微博', 'zh-hk': '自己的微博', 'zh-tw': '自己的微博', 'en': 'Weibo by myself' },
'showMyOriginalDesc': { 'zh-cn': '自己微博的转发', 'zh-hk': '自己微博的轉發', 'zh-tw': '自己微博的轉發', 'en': 'Forward of my Weibo' },
'showMentionMeDesc': { 'zh-cn': '提到自己的微博', 'zh-hk': '提到自己的微博', 'zh-tw': '提到自己的微博', 'en': 'Weibo mentioned me' },
// 隐藏
'otherBlacklistTitle': { 'zh-cn': '隐藏以下内容', 'zh-hk': '以下內容', 'zh-tw': '隱藏以下內容', 'en': 'Hide following content' },
'adfeedFilterDesc': { 'zh-cn': '推广微博', 'zh-hk': '推廣微博', 'zh-tw': '推廣微博', 'en': 'Ad Weibo' },
'recommandFeedDesc': { 'zh-cn': '推荐微博', 'zh-hk': '建議微博', 'zh-tw': '建議微博', 'en': 'Recommand Weibo' },
'fakeWeiboFilterDesc': { 'zh-cn': '混入微博列表的其它内容', 'zh-hk': '混入微博列表的其它內容', 'zh-tw': '混入微博列表的其它內容', 'en': 'Other contents in Weibo list' },
'deletedForwardFilterDesc': { 'zh-cn': '已删除微博的转发', 'zh-hk': '已刪除微博的轉發', 'zh-tw': '已刪除微博的轉發', 'en': 'Forward of deleted Weibo' },
'voteWeiboFilterDesc': { 'zh-cn': '投票微博', 'zh-hk': '投票微博', 'zh-tw': '投票微博', 'en': 'Vote weibo' },
// 刷屏与版聊
'otherSpammingTitle': { 'zh-cn': '刷屏与版聊', 'zh-hk': '洗版與版聊', 'zh-tw': '洗版與版聊', 'en': 'Spamming & Chating' },
'sameAccountFilterDesc': { 'zh-cn': '相同作者的微博:', 'zh-hk': '相同作者的微博:', 'zh-tw': '相同作者的微博:', 'en': 'Weibo from same account: ' },
'sameForwardFilterDesc': { 'zh-cn': '相同微博的转发:', 'zh-hk': '相同微博的轉發:', 'zh-tw': '相同微博的轉發:', 'en': 'Forward from same Weibo: ' },
'sameNumberOptionDesc': { 'zh-cn': '超过{{number}}条时', 'zh-hk': '超過{{number}}條時', 'zh-tw': '超過{{number}}條時', 'en': 'display no more than {{number}} ' },
'sameActionOptionDesc': { 'zh-cn': '{{{select}}}', 'zh-hk': '{{{select}}}', 'zh-tw': '{{{select}}}', 'en': 'or {{{select}}} it' },
// 分组浏览
'otherGroupTitle': { 'zh-cn': '分组浏览', 'zh-hk': '分組流覽', 'zh-tw': '分組流覽', 'en': 'Browse by Group' },
'accountByGroup': { 'zh-cn': '分组浏览时禁用按账号隐藏', 'zh-hk': '分組流覽時禁用按帳號隱藏', 'zh-tw': '分組流覽時禁用按帳號隱藏', 'en': 'Disable hide by account filter when browsing by group' },
'sameAccountByGroup': { 'zh-cn': '浏览分组时禁用相同作者数量限制', 'zh-hk': '流覽分組時禁用相同作者數量限制', 'zh-tw': '流覽分組時禁用相同作者數量限制', 'en': 'Disable hide too many Weibo from same account filter when browsing by group' },
// 模块
'layoutFilterGroupTitle': { 'zh-cn': '模块', 'zh-hk': '模組', 'zh-tw': '模組', 'en': 'Module' },
'layoutFilterGroupDesc': { 'zh-cn': '隐藏以下模块', 'zh-hk': '隱藏以下模組', 'zh-tw': '隱藏以下模組', 'en': 'Hide following modules' },
// 标识图标
'layoutHideIcon': { 'zh-cn': '标识/图标', 'zh-hk': '標誌/圖示', 'zh-tw': '標誌/圖示', 'en': 'Logo / Icon' },
'layoutHideIconLevel': { 'zh-cn': '等级', 'zh-hk': 'Level', 'zh-tw': '等級', 'en': 'Level' },
'layoutHideIconMember': { 'zh-cn': '微博会员', 'zh-hk': '微博會員', 'zh-tw': '微博會員', 'en': 'Weibo VIP / Member' },
'layoutHideIconApprove': { 'zh-cn': '个人认证', 'zh-hk': '個人認證', 'zh-tw': '個人認證', 'en': 'Personal Authentication / 個人認證' },
'layoutHideIconApproveCo': { 'zh-cn': '机构认证', 'zh-hk': '企業認證', 'zh-tw': '企業認證', 'en': 'Weibo Verification / 企業認證' },
'layoutHideIconClub': { 'zh-cn': '微博达人', 'zh-hk': '微博達人', 'zh-tw': '微博達人', 'en': 'Pioneer' },
'layoutHideIconVGirl': { 'zh-cn': '微博女郎', 'zh-hk': '微博女郎', 'zh-tw': '微博女郎', 'en': 'Weibo girl' },
'layoutHideIconTaobao': { 'zh-cn': '淘宝商户', 'zh-hk': '淘寶商戶', 'zh-tw': '淘寶商戶', 'en': 'Taobao Merchant' },
// 导航栏
'layoutHideNav': { 'zh-cn': '导航栏', 'zh-hk': '導覽列', 'zh-tw': '導覽列', 'en': 'Navigation Bar' },
'layoutHideNavMain': { 'zh-cn': '首页', 'zh-hk': '首頁', 'zh-tw': '首頁', 'en': 'Home' },
'layoutHideNavHot': { 'zh-cn': '热门', 'zh-hk': '熱門', 'zh-tw': '熱門', 'en': 'Hot' },
'layoutHideNavApp': { 'zh-cn': '应用', 'zh-hk': '應用', 'zh-tw': '應用', 'en': 'Apps' },
'layoutHideNavGame': { 'zh-cn': '游戏', 'zh-hk': '遊戲', 'zh-tw': '遊戲', 'en': 'Game' },
'layoutHideNavMember': { 'zh-cn': '会员菜单', 'zh-hk': '會員功能表', 'zh-tw': '會員功能表', 'en': 'VIP Menu' },
// 左栏
'layoutHideLeft': { 'zh-cn': '左栏', 'zh-hk': '左欄', 'zh-tw': '左欄', 'en': 'Left Column' },
'layoutHideLeftToMe': { 'zh-cn': '发给我的', 'zh-hk': '發給我的', 'zh-tw': '發給我的', 'en': 'Send to me' },
'layoutHideLeftFriends': { 'zh-cn': '好友圈', 'zh-hk': '好友圈', 'zh-tw': '好友圈', 'en': 'Friends' },
'layoutHideLeftApp': { 'zh-cn': '应用', 'zh-hk': '应用', 'zh-tw': '应用'/* as is */, 'en': 'Apps' },
// 中栏
'layoutHideMiddle': { 'zh-cn': '中栏', 'zh-hk': '中欄', 'zh-tw': '中欄', 'en': 'Middle Column' },
'layoutHideMiddleRecommendedTopic': { 'zh-cn': '热门微博(发布框上方)', 'zh-hk': '热门微博(發布框上方)', 'zh-tw': '热门微博(發布框上方)'/* as is */, 'en': '热门微博 (Hot Weibo), top of publisher' },
'layoutHideMiddleMemberTip': { 'zh-cn': '开通会员提示(底部)', 'zh-hk': '開通會員提示(底部)', 'zh-tw': '開通會員提示(底部)', 'en': 'Tip of Join Weibo VIP, bottom' },
// 右栏
'layoutHideRight': { 'zh-cn': '右栏', 'zh-hk': '右欄', 'zh-tw': '右欄', 'en': 'Right Column' },
'layoutHideRightWhole': { 'zh-cn': '整个右栏', 'zh-hk': '整個右欄', 'zh-tw': '整個右欄', 'en': 'Whole Right Column' },
'layoutHideRightTemplate': { 'zh-cn': '设置模板', 'zh-hk': '背景設定', 'zh-tw': '背景設定', 'en': 'Template Settings' },
'layoutHideRightInfo': { 'zh-cn': '头像', 'zh-hk': '頭像', 'zh-tw': '頭像', 'en': 'Avatar' },
'layoutHideRightTrial': { 'zh-cn': '会员资格体验', 'zh-hk': '会员资格体验', 'zh-tw': '会员资格体验'/* as is */, 'en': '会员资格体验 (Trial of VIP)' },
'layoutHideRightAtten': { 'zh-cn': '关注/粉丝/微博数', 'zh-hk': '關注/粉絲/微博數', 'zh-tw': '關注/粉絲/微博數', 'en': 'Numbers of Following/Followers/Weibo' },
'layoutHideRightInterest': { 'zh-cn': '可能感兴趣的人', 'zh-hk': '可能感興趣的人', 'zh-tw': '可能感興趣的人', 'en': 'You may know' },
'layoutHideRightHotTopic': { 'zh-cn': '热门话题', 'zh-hk': '熱門話題', 'zh-tw': '熱門話題', 'en': 'Hot Topic' },
'layoutHideRightMember': { 'zh-cn': '会员专区', 'zh-hk': '會員專區', 'zh-tw': '會員專區', 'en': 'Weibo VIP' },
'layoutHideRightWeibo': { 'zh-cn': '热门微博', 'zh-hk': '熱門微博', 'zh-tw': '熱門微博', 'en': 'Hot Weibo' },
'layoutHideRightLocation': { 'zh-cn': '地点推荐', 'zh-hk': '地點推薦', 'zh-tw': '地點推薦', 'en': 'Location' },
'layoutHideRightMusic': { 'zh-cn': ' 热门歌曲', 'zh-hk': '熱門歌曲', 'zh-tw': '熱門歌曲', 'en': 'Hot Music' },
'layoutHideRightMovie': { 'zh-cn': '最新电影', 'zh-hk': '最新電影', 'zh-tw': '最新電影', 'en': 'Hot Movie' },
'layoutHideRightBook': { 'zh-cn': '人气图书', 'zh-hk': '人氣圖書', 'zh-tw': '人氣圖書', 'en': 'Hot Book' },
'layoutHideRightNotice': { 'zh-cn': '公告栏', 'zh-hk': '公告欄', 'zh-tw': '公告欄', 'en': 'Bulletin Board' },
// 微博内
'layoutHideWeibo': { 'zh-cn': '微博内', 'zh-hk': '微博內', 'zh-tw': '微博內', 'en': 'In Weibo' },
'layoutHideWeiboRecomFeed': { 'zh-cn': '精彩微博推荐', 'zh-hk': '精彩微博推薦', 'zh-tw': '精彩微博推薦', 'en': '精彩微博推荐 (Weibo you may interested in)' },
'layoutHideWeiboTopicCard': { 'zh-cn': '话题卡片', 'zh-hk': '話題卡片', 'zh-tw': '話題卡片', 'en': 'Topic Cards' },
'layoutHideWeiboLocationCard': { 'zh-cn': '位置卡片', 'zh-hk': '位置卡片', 'zh-tw': '位置卡片', 'en': 'Location Cards' },
'layoutHideWeiboFeedTip': { 'zh-cn': '评论框提示横幅', 'zh-hk': '評論框提示橫幅', 'zh-tw': '評論框提示橫幅', 'en': 'Tips for Comment' },
// 个人主页
'layoutHidePerson': { 'zh-cn': '个人主页', 'zh-hk': '個人主頁', 'zh-tw': '個人主頁', 'en': 'Ones home page' },
'layoutHidePersonCover': { 'zh-cn': '封面图', 'zh-hk': '封面圖', 'zh-tw': '封面圖', 'en': 'Cover Picture' },
'layoutHidePersonTemplate': { 'zh-cn': '模板设置', 'zh-hk': '模板設置', 'zh-tw': '模板設置', 'en': 'Template Settings' },
'layoutHidePersonBadgeIcon': { 'zh-cn': '勋章', 'zh-hk': '勳章', 'zh-tw': '勳章', 'en': 'Badges' },
'layoutHidePersonStats': { 'zh-cn': '关注/粉丝/微博数', 'zh-hk': '關注/粉絲/微博數', 'zh-tw': '關注/粉絲/微博數', 'en': 'Numbers of Following/Followers/Weibo' },
'layoutHidePersonMyData': { 'zh-cn': '我的微博人气', 'zh-hk': '我的微博人气', 'zh-tw': '我的微博人气', 'en': '我的微博人气 (My Micro World)' },
'layoutHidePersonSuggestUser': { 'zh-cn': '可能感兴趣的人', 'zh-hk': '可能感兴趣的人', 'zh-tw': '可能感兴趣的人'/* as is */, 'en': 'Suggested' },
'layoutHidePersonGroup': { 'zh-cn': '推荐的人', 'zh-hk': '推荐的人', 'zh-tw': '推荐的人'/* as is */, 'en': '推荐的人 (Suggested Group)' },
'layoutHidePersonRelation': { 'zh-cn': '微关系', 'zh-hk': '微关系', 'zh-tw': '微关系'/* as is */, 'en': 'Relationship' },
'layoutHidePersonAlbum': { 'zh-cn': '图片', 'zh-hk': '相冊', 'zh-tw': '相冊', 'en': 'Album' },
'layoutHidePersonHotTopic': { 'zh-cn': '话题', 'zh-hk': '話題', 'zh-tw': '話題', 'en': 'Topic' },
'layoutHidePersonHotWeibo': { 'zh-cn': '热门微博', 'zh-hk': '熱門微博', 'zh-tw': '熱門微博', 'en': 'Hot Weibo' },
// 杂项
'layoutHideOther': { 'zh-cn': '杂项', 'zh-hk': '雜項', 'zh-tw': '雜項', 'en': 'Others' },
'layoutHideOtherAds': { 'zh-cn': '广告', 'zh-hk': '廣告', 'zh-tw': '廣告', 'en': 'Advertisement' },
'layoutHideOtherFeedRecom': { 'zh-cn': '相关微博推荐', 'zh-hk': '相关推荐', 'zh-tw': '相关推荐', 'en': '相关推荐 (Related Weibo Suggestion)' },
'layoutHideOtherFooter': { 'zh-cn': '页面底部', 'zh-hk': '頁面底部', 'zh-tw': '頁面底部', 'en': 'Footer' },
'layoutHideOtherWbIm': { 'zh-cn': '微博桌面推荐(右下)', 'zh-hk': '微博桌面推薦(右下)', 'zh-tw': '微博桌面推薦(右下)', 'en': '微博桌面2014 (Desktop Weibo), bottom right' },
'layoutHideOtherTip': { 'zh-cn': '功能提示框', 'zh-hk': '功能提示框', 'zh-tw': '功能提示框', 'en': 'Function Tips' },
// 工具
'toolFilterGroupTitle': { 'zh-cn': '工具', 'zh-hk': '工具', 'zh-tw': '工具', 'en': 'Tool' },
'useFastCreator': { 'zh-cn': '使用拖放快速创建过滤器', 'zh-hk': '使用拖放快速創建篩選器', 'zh-tw': '使用拖放快速創建篩選器', 'en': 'Use drag and drop to create filters' },
'clearDefTopicDesc': { 'zh-cn': '清除发布框中的默认话题', 'zh-hk': '清除發布框中的預設話題', 'zh-tw': '清除發布框中的預設話題', 'en': 'Remove default topic in Publisher' },
'userstyleTitle': { 'zh-cn': '自定义CSS', 'zh-hk': '自訂CSS', 'zh-tw': '自訂CSS', 'en': 'Customize CSS' },
'userstyleEditDesc': { 'zh-cn': '编辑微博自定义CSS', 'zh-hk': '編輯微博自訂CSS', 'zh-tw': '編輯微博自訂CSS', 'en': 'Edit Weibo Customize CSS' },
'userstyleEditDetails': { 'zh-cn': 'YAWF CSS: ', 'zh-hk': 'YAWF CSS: ', 'zh-tw': 'YAWF CSS: ', 'en': 'YAWF CSS: ' },
'noFloatNav': { 'zh-cn': '禁用导航栏顶端固定', 'zh-hk': '禁用導覽列頂端固定', 'zh-tw': '禁用導覽列頂端固定', 'en': 'Make navigation scroll with page' },
'showAllGroupDesc': { 'zh-cn': '展开左栏分组', 'zh-hk': '展開左欄分組', 'zh-tw': '展開左欄分組', 'en': 'Unfold groups in left column' },
'showAllMsgNavDesc': { 'zh-cn': '展开左栏消息', 'zh-hk': '展開左欄消息', 'zh-tw': '展開左欄消息', 'en': 'Unfold news in left column' },
'unwrapTextDesc': { 'zh-cn': '微博作者和正文同行', 'zh-hk': '微博作者和正文同行', 'zh-tw': '微博作者和正文同行', 'en': 'No line break after author' },
'personalRedirectWeibo': { 'zh-cn': '访问账号主页显示微博页面', 'zh-hk': '訪問帳號主頁顯示微博頁面', 'zh-tw': '訪問帳號主頁顯示微博頁面', 'en': 'Show Weibo page instead of personal mainpage by default' },
'viewOriginalDesc': { 'zh-cn': '添加“查看原图”链接', 'zh-hk': '添加“查看原圖”連結', 'zh-tw': '添加“查看原圖”連結', 'en': 'add "Original Picture" link' },
'viewOriginalText': { 'zh-cn': '查看原图', 'zh-hk': '查看原圖', 'zh-tw': '查看原圖', 'en': 'Original Picture' },
'blockHiddenWeiboDesc': { 'zh-cn': '告知服务器被隐藏的微博以避免再次加载', 'zh-hk': '告知伺服器被隱藏的微博以避免再次載入', 'zh-tw': '告知伺服器被隱藏的微博以避免再次載入', 'en': 'Send blocked Weibo to server to avoid load it again' },
'whitelistHighlightDesc': { 'zh-cn': '以该背景色显示白名单的微博:', 'zh-hk': '以該背景色顯示白名單的微博:', 'zh-tw': '以該背景色顯示白名單的微博:', 'en': 'Show Weibo in whitelist with background color: ' },
'transparencyInput': { 'zh-cn': '透明度{{number}}%', 'zh-hk': '透明度{{number}}%', 'zh-tw': '透明度{{number}}%', 'en': ' transparency {{number}} %' },
'mainBackgroundColorOverride': { 'zh-cn': '首页背景颜色', 'zh-hk': '首頁背景色彩', 'zh-tw': '首頁背景色彩', 'en': 'Background color for home page' },
'profileBackgroundColorOverride': { 'zh-cn': '个人主页背景颜色', 'zh-hk': '個人主頁背景色彩', 'zh-tw': '個人主頁背景色彩', 'en': 'Background color for personal home page' },
// 脚本
'scriptFilterGroupTitle': { 'zh-cn': '脚本', 'zh-hk': '腳本', 'zh-tw': '腳本', 'en': 'Script' },
// 导入导出
'configImportAndExport': { 'zh-cn': '设置', 'zh-hk': '設定', 'zh-tw': '設定', 'en': 'Setting' },
'configImportButton': { 'zh-cn': '导入', 'zh-hk': '匯入', 'zh-tw': '匯入', 'en': 'Import' },
'configImportWarningTitle': { 'zh-cn': '设置导入', 'zh-hk': '設定匯入', 'zh-tw': '設定匯入', 'en': 'Setting Import' },
'configImportWarning': { 'zh-cn': '导入的设置会覆盖您当前已有的设置,确实要导入设置吗?', 'zh-hk': '匯入的設定會覆蓋您當前已有的設定,您確定要匯入設定嗎?', 'zh-tw': '匯入的設定會覆蓋您當前已有的設定,您確定要匯入設定嗎?', 'en': 'The imported settings may replace your current settings. Are you sure you want to import this file?' },
'configImportSuccessTitle': { 'zh-cn': '设置导入完成', 'zh-hk': '設定匯入完成', 'zh-tw': '設定匯入完成', 'en': 'Import settings' },
'configImportSuccess': { 'zh-cn': '已经成功地导入了设置', 'zh-hk': '已经成功地匯入了設定', 'zh-tw': '已经成功地匯入了設定', 'en': 'Successfully imported settings' },
'configImportFailTitle': { 'zh-cn': '设置导入失败', 'zh-hk': '設定匯入失败', 'zh-tw': '設定匯入失败', 'en': 'Import settings' },
'configImportFail': { 'zh-cn': '导入设置文件时出现错误,可能是使用了错误的文件,文件被损坏或文件的版本不支持', 'zh-hk': '匯入設定檔案時出現錯誤,可能是使用了錯誤的檔案,檔案被損壞或設定檔案的版本不支持', 'zh-tw': '匯入設定檔案時出現錯誤,可能是使用了錯誤的檔案,檔案被損壞或設定檔案的版本不支持', 'en': 'Error occurred during importing process. Wrong file may be used, the file may be broken, or the version of setting file is not supported.' },
'configExportButton': { 'zh-cn': '导出', 'zh-hk': '匯出', 'zh-tw': '匯出', 'en': 'Export' },
'configResetButton': { 'zh-cn': '重置', 'zh-hk': '重設', 'zh-tw': '重設', 'en': 'Reset' },
'configResetWarningTitle': { 'zh-cn': '设置重置', 'zh-hk': '設定重設', 'zh-tw': '設定重設', 'en': 'Setting Reset' },
'configResetWarning': { 'zh-cn': '这将会清空您当前的所有配置,确实要重置设置吗?', 'zh-hk': '這將會清空您當前的所有設定,您確定要重置設定嗎?', 'zh-tw': '這將會清空您當前的所有設定,您確定要重置設定嗎?', 'en': 'You are deleting all your settings. Are you sure you want to reset your settings?' },
// 调试
'scriptDebugTitle': { 'zh-cn': '调试', 'zh-hk': '偵錯', 'zh-tw': '偵錯', 'en': 'Debug' },
'scriptDebug': { 'zh-cn': '在控制台打印调试信息', 'zh-hk': '將偵錯訊息列印到主控台', 'zh-tw': '將偵錯訊息列印到主控台', 'en': 'Print debug info to console' },
// 关于
'scriptAboutTitle': { 'zh-cn': '关于', 'zh-hk': '關於', 'zh-tw': '關於', 'en': 'About' },
'scriptAbout': {
'zh-cn': '
Yet Another Weibo Filter (YAWF) {{version}}
YAWF 使用 MIT 协议授权。您可以访问脚本主页 获取详细信息。 如果您在使用过程中遇到任何脚本的错误,或对脚本有任何建议,您可以到反馈页面 提供报告,或直接私信作者 。
本脚本参考并使用了眼不见心不烦 脚本的部分代码。
',
'zh-hk': 'Yet Another Weibo Filter (YAWF) {{version}}
YAWF 使用 MIT 協定授權。您可以訪問腳本主頁 獲取詳細資訊。 如果您在使用過程中遇到任何腳本的錯誤,或對腳本有任何建議,您可以到回饋頁面 提供報告,或直接私信作者 。
本腳本參考並使用了眼不見心不煩 腳本的部分原始碼。
',
'zh-tw': 'Yet Another Weibo Filter (YAWF) {{version}}
YAWF 使用 MIT 協定授權。您可以訪問腳本主頁 獲取詳細資訊。 如果您在使用過程中遇到任何腳本的錯誤,或對腳本有任何建議,您可以到回饋頁面 提供報告,或直接私信作者 。
本腳本參考並使用了眼不見心不煩 腳本的部分原始碼。
',
'en': 'Yet Another Weibo Filter (YAWF) {{version}}
YAWF is under the MIT License. You may want to visit project homepage for more information. If you find any bugs or have feature request, please report them in the feedback page , or send message to author .
Some codes of this script come from 眼不见心不烦 (Weibo Content Filter) script.
',
},
// 拖拽
'dropAreaTitle': { 'zh-cn': '拖放至此 快速创建过滤器', 'zh-hk': '拖放至此 快速創建篩選器', 'zh-tw': '拖放至此 快速創建篩篩器', 'en': 'Drop Here to Create Filter' },
'dropAreaText': { 'zh-cn': '将文字或链接拖放至此,快速创建过滤器。', 'zh-hk': '將文字或連結拖放至此,快速創建篩選器。', 'zh-tw': '將文字或連結拖放至此,快速創建篩選器。', 'en': 'Drop text or link here to create filter.' },
'fastCreateChoseTitle': { 'zh-cn': '创建过滤器', 'zh-hk': '創建篩篩器', 'zh-tw': '創建篩篩器', 'en': 'Create Filter' },
'fastFilterChoseText': { 'zh-cn': '请选择要创建的过滤器:', 'zh-hk': '請選擇要創建的篩選器:', 'zh-tw': '請選擇要創建的篩選器:', 'en': 'Chose the filter(s) you want:' },
};
// 页面常量
var html = {
'cover': '
',
'dialog': '',
'alert': '',
'confirm': '',
'icon': '',
'configHeaderTop': '',
'configLayerTop': '',
'configLayerItem': '
',
'configLayerBottom': '
',
'configFooter': '',
'configSubtitle': '{{{text}}}
',
'configText': '{{{text}}}
',
'configText': '',
'configBoolean': '{{{text}}}
',
'configString': '{{{text}}}
',
'configStrings': '',
'configStringsItem': '{{[item]}} ',
'configUsers': '',
'configUsersItem': ' ',
'configImportExport': '',
'viewOriginalLink': ' {{viewOriginalText}}| ',
'colorInput': ' ',
'numberInput': ' ',
'transparencyInput': '
',
'select': '{{options}} ',
'option': '{{{text}}} ',
'dropArea': '{{dropAreaTitle}}
{{dropAreaText}}
',
'fastFilterHeader': '',
};
var url = {
'newcard': 'http://weibo.com/aj/user/newcard?type=1&{{query}}&_t=1&callback={{callback}}',
'view_ori': 'http://photo.weibo.com/{{uid}}/wbphotos/large/mid/{{mid}}/pid/{{pid}}',
'block_wb': 'http://weibo.com/aj/user/block?_wv=5&__rnd={{rnd}}',
};
// 微博过滤规则
var rules = (function () {
var list = [];
var add = function (priority, rule) {
list.push({ 'priority': priority, 'rule': rule });
list.sort(function (x, y) { return y.priority - x.priority; });
};
var parse = function (feed) {
var result = null;
list.some(function (item) {
try { result = item.rule(feed) || result; }
catch (e) { debug('error while parsing rule %o: %o', item.rule, e); }
if (result) debug('%o(%o) -> %s', item.rule, feed, result);
return result;
});
return result;
};
return {
'add': add,
'parse': parse,
};
}());
// 根据用户界面上的语言做不同调整
var i18n = (function () {
var defaultLang = 'zh-cn';
var lang = null;
var pending = [];
var chose = function (langObj) {
langObj.local = langObj[lang] || langObj[defaultLang];
};
var langs = function (langObj) {
pending.push(langObj);
};
var setLang = function (l) {
lang = l;
pending.map(chose);
pending = [];
i18n = chose;
i18n.lang = l;
};
langs.setLang = setLang;
return langs;
}());
// 将字符串用dd的形式转义
var escapeXml = function (s) {
return s.replace(/./g, function (c) { return '' + c.charCodeAt(0); });
};
// 以参数填充字符串
var fillStr = function (base) {
var argdatas = Array.apply(Array, arguments).slice(1);
var datas = argdatas.concat([text]);
return base.replace(/{{([\[{]?([a-zA-Z0-9_-]*)[\]}]?)}}/g, function (o, i, p) {
var ret = null;
datas.some(function (data) {
if (p in data && typeof data[p] === 'string') ret = data[p];
return ret !== null;
});
if (ret) {
if (i[0] === '{') return fillStr.bind(this, ret).apply(null, argdatas);
if (i[0] === '[') return escapeXml(ret);
}
return ret || o;
});
};
// 设置项
var config = function (uid) {
var config = {}, keys = [], onputs = [], storageKey = 'user' + uid + 'config';
var tonputs = function (key, value, oldValue) {
onputs.map(function (f) { f(key, value, oldValue); });
};
// 读取到内存
var read = function () {
try { config = JSON.parse(GM_getValue(storageKey, '{}')); }
catch (e) { config = {}; }
};
// 从内存写出
var write = function () {
GM_setValue(storageKey, JSON.stringify(config));
};
// 写入到内存
var put = function (key, value) {
if (keys.indexOf(key) === -1) return;
tonputs(key, value, config[key]);
config[key] = value;
write();
return value;
};
// 从内存读取
var get = function (key, value, type) {
read();
if (!(key in config)) return value;
var val = config[key];
if (typeof val === 'undefined') return value;
if (type && (val === null || val.constructor !== type)) return value;
return val;
};
// 当内存配置被修改时调用
var onput = function (f) {
onputs.push(f);
};
// 从字串导入
var import_ = function (s) {
try {
clear();
s = JSON.parse(s).conf;
Object.keys(s).forEach(function (key) {
put(key, s[key]);
});
write();
return true;
} catch (e) { }
return false;
};
// 导出成为字串
var export_ = function () {
var info = GM_info || {}, script = info.script || {};
return JSON.stringify({
'ua': navigator.userAgent,
'yawf': script.name,
'ver': script.version,
'gm': (info.scriptHandler || '') + info.version,
'conf': config,
}, null, 2);
};
// 清空设置
var clear = function () {
config = {};
tonputs();
write();
};
// 注册键
var reg = function (key) { keys.push(key); };
// 初始化
return {
'uid': uid,
'put': put, 'get': get, 'onput': onput,
'read': read, 'write': write,
'import': import_, 'export': export_,
'clear': clear,
'reg': reg,
};
};
var debug = !!GM_getValue('debug', false) ?
console.log.bind(console) : function () { };
// 显示右上角过滤器图标
var showIcon = function () {
var p = document.querySelector('.WB_global_nav .gn_person');
if (!p) return setTimeout(showIcon, 100);
var d = cewih('div', html.icon).firstChild;
p.appendChild(d);
var f = document.querySelector('.gn_filter');
f.addEventListener('click', function (e) {
filters.showDialog();
e.preventDefault();
});
};
// 对话框
var Form = function (dom, display, details) {
var ok = dom.querySelector('[node-type="OK"]');
var cancel = dom.querySelector('[node-type="cancel"]');
var close = dom.querySelector('[node-type="close"]');
var title = dom.querySelector('.title');
var mouse = null, pos;
var setPos = function (pos) {
var left = pos[0], top = pos[1];
left = Math.min(Math.max(0, left), document.body.clientWidth - dom.clientWidth - 2);
top = Math.min(Math.max(pageYOffset, top), pageYOffset + window.innerHeight - dom.clientHeight - 2);
dom.style.left = left + 'px';
dom.style.top = top + 'px';
return [left, top];
};
var dragMove = function (e) {
var mouse_new = [e.clientX, e.clientY];
pos[0] += mouse_new[0] - mouse[0];
pos[1] += mouse_new[1] - mouse[1];
setPos(pos);
mouse = mouse_new;
};
var dragMoveDone = function () {
document.removeEventListener('mousemove', dragMove);
document.removeEventListener('mouseup', dragMoveDone);
dom.classList.remove('yawf-drag');
if (dom.releaseCapture) { dom.releaseCapture(); }
pos = setPos(pos);
mouse = null;
};
if (title) {
title.style.cursor = 'move';
title.addEventListener('mousedown', function (e) {
mouse = [e.clientX, e.clientY];
document.addEventListener('mousemove', dragMove);
document.addEventListener('mouseup', dragMoveDone);
dom.classList.add('yawf-drag');
if (dom.setCapture) { dom.setCapture(); }
});
}
if (details.onOk) if (ok) ok.addEventListener('click', details.onOk);
if (details.onCancel) {
if (cancel) cancel.addEventListener('click', details.onCancel);
if (close) close.addEventListener('click', details.onCancel);
}
var cover = cewih('div', html.cover).firstChild;
var keys = function (e) {
if (e.keyCode === 13) if (ok) ok.click();
if (e.keyCode === 27) if (close) close.click();
};
var hide = function () {
document.body.removeChild(dom);
document.body.removeChild(cover);
document.removeEventListener('keypress', keys);
};
var show = function (top, left) {
document.body.appendChild(cover);
document.body.appendChild(dom);
if (top == null) top = (window.innerHeight - dom.clientHeight) / 2;
if (left == null) left = (window.innerWidth - dom.clientWidth) / 2;
pos = [left, top + pageYOffset];
setPos(pos);
document.addEventListener('keypress', keys);
document.activeElement.blur();
};
if (display) show();
if (ok) ok.addEventListener('click', hide);
if (cancel) cancel.addEventListener('click', hide);
if (close) close.addEventListener('click', hide);
return {
'hide': hide,
'show': show,
};
};
// 显示一个对话框
var Dialog = function (id, title, fillFun) {
var dom = cewih('div', fillStr(html.dialog, { 'id': id, 'title': fillStr(title) })).firstChild;
var form = Form(dom, false, {});
fillFun(dom.querySelector('[node-type="inner"]'));
return form;
};
// 显示一个提示框
var Alert = function (id, details) {
var dom = cewih('div', fillStr(html.alert, { 'id': id, 'title': details.title, 'text': details.text, 'icon': details.icon || 'warn' })).firstChild;
var form = Form(dom, true, details);
return form;
};
var Confirm = function (id, details) {
var dom = cewih('div', fillStr(html.confirm, { 'id': id, 'title': details.title, 'text': details.text, 'icon': details.icon || 'question' })).firstChild;
var form = Form(dom, true, details);
return form;
};
// 延迟调用函数
var call = function (f) {
setTimeout.bind(this, f, 0).apply(null, Array.apply(Array, arguments).slice(1));
};
// 套上try-catch
var withTry = function (f, fc) {
return function () {
try { f.apply(this, arguments); }
catch (e) {
debug('Exception while run %o: %o (%o)', f, e, e.stack);
if (fc) fc(e);
}
};
};
// 管理样式
var css = (function () {
var styleText = '';
var fun = function (css) {
return function () {
if (this.conf) styleText += css + '\n';
};
};
fun.init = function () { GM_addStyle(styleText); };
fun.add = function (css) { styleText += css + '\n'; };
return fun;
}());
// 产生一个假的回调函数
var dateStr = (function () {
var last = 0;
return function () {
return '' + (last = Math.max(last + 1, Number(new Date())));
};
}());
// 维护账号信息,用于显示
var account = (function () {
var idCache = {}, nameCache = {}, working = {};
var request = function (queryStr, onsucc, onerror) {
if (working[queryStr]) return working[queryStr].push([onsucc, onerror]);
working[queryStr] = [[onsucc, onerror]];
var done = function (success, data) {
working[queryStr].forEach(function (w) { w[success ? 0 : 1](data); });
delete working[queryStr];
};
GM_xmlhttpRequest({
'method': 'GET',
'url': fillStr(url.newcard, { 'query': queryStr, 'callback': 'STK_' + dateStr() }),
'onload': withTry(function (resp) {
var respJson = JSON.parse(resp.responseText.replace(/^try{[^{]*\(/, '').replace(/\)}catch\(e\){};$/, ''));
var namecard = cewih('div', respJson.data);
var avatar = namecard.querySelector('.name dt img').getAttribute('src');
var name = namecard.querySelector('.name dd a[uid]').getAttribute('title');
var uid = namecard.querySelector('.name dd a[uid]').getAttribute('uid');
var data = { 'avatar': avatar, 'id': uid, 'name': name };
nameCache[name] = idCache[uid] = data;
done(true, data);
}, done.bind(this, false)),
'onerror': function () { done(false); },
});
return queryStr;
};
var byId = function (id, onsucc, onerror) {
if (idCache[id]) onsucc(idCache[id]);
else request('id=' + id, onsucc, onerror);
};
var byName = function (name, onsucc, onerror) {
if (nameCache[name]) onsucc(nameCache[name]);
else request('name=' + encodeURIComponent(name), onsucc, onerror);
};
return { 'id': byId, 'name': byName };
}());
// 过滤器管理器
var filters = (function () {
var list = [], preinit = true;
var add = function (details) {
list.push(details);
if (!preinit) details.init();
return details;
};
var init = function () {
list.forEach(function (x) { x.init(); });
preinit = false;
};
var dialog = null;
var lastTab = 0;
var dialogTabs = function (list, inner, page) {
var alist = Array.apply(Array, inner.querySelectorAll('.yawf-config-header a'));
var llist = Array.apply(Array, inner.querySelectorAll('.yawf-config-body .yawf-config-layer'));
var choseLList = function (i) {
llist.forEach(function (l) { l.style.display = 'none'; });
alist.forEach(function (a) { a.classList.remove('current'); a.classList.remove('S_bg5'); a.classList.add('S_bg1'); });
llist[i].innerHTML = ''; list[i].show(llist[i]); llist[i].style.display = 'block';
alist[i].classList.add('current'); alist[i].classList.remove('S_bg1'); alist[i].classList.add('S_bg5');
lastTab = i;
};
list.map(function (filter, i) {
var a = alist[i];
a.addEventListener('mousedown', function () { choseLList(i); });
a.addEventListener('keydown', function () { choseLList(i); });
});
choseLList(page);
};
var showDialog = function (page, count) {
var showDialogInner = function (inner) {
inner.innerHTML = [html.configHeaderTop,
list.map(function (filter, index) {
return fillStr(html.configHeaderItem, {
'name': text[filter.name + 'Title'],
'aclass': index === 0 ? 'S_bg5 current' : 'S_bg1',
'liclass': index === list.length - 1 ? 'pftb_itm_lst' : ' ',
});
}).join(''),
html.configHeaderBottom,
html.configLayerTop,
list.map(function (filter, index) {
return fillStr(html.configLayerItem, {
'name': filter.name + 'Layer',
});
}).join(''),
html.configLayerBottom,
html.configFooter,
].join('');
call(function () {
dialog.show(0);
});
if (list.indexOf(page) === -1) dialogTabs(list, inner, lastTab);
else dialogTabs(list, inner, list.indexOf(page));
};
if (!(dialog = Dialog('yawf-config', '{{configDialogTitle}}', showDialogInner))) {
if (!count || count < 100) setTimeout(showDialog, 100, page, (count || 0) + 1);
}
};
return {
'add': add,
'init': init,
'showDialog': showDialog,
};
}());
// 快速创建播放器的对话框
var fastFilterDialog = function (chose) {
var dialogInner = function (inner) {
inner.innerHTML = [html.fastFilterHeader,
chose.map(function (c) {
return fillStr(html.fastFilterItem, { 'inner': c.filter.desc(c.val) });
}).join(''),
html.fastFilterFooter].join('');
var checkboxList = Array.apply(Array, inner.querySelectorAll('input[type="checkbox"]'));
var selectList = Array.apply(Array, inner.querySelectorAll('select'));
selectList.forEach(function (select) { select.value = 'blacklist'; });
var allChecked = function () {
var active = [];
checkboxList.forEach(function (checkbox, i) {
if (checkbox.checked)
active.push({ 'chose': chose[i], 'action': selectList[i].value });
});
return active;
};
if (checkboxList.length === 1) checkboxList[0].checked = true;
else checkboxList.forEach(function (checkbox) { checkbox.checked = false; });
var updateOkButton = function () {
if (allChecked().length) ok.classList.remove('W_btn_a_disable');
else ok.classList.add('W_btn_a_disable');
};
var ok = inner.querySelector('[action-type="ok"]');
ok.addEventListener('click', function () {
var active = allChecked();
if (!active.length) return null;
active.forEach(function (act) {
act.chose.filter.add(act.chose.val, act.action);
});
dialog.hide();
});
checkboxList.forEach(function (checkbox) {
checkbox.addEventListener('change', updateOkButton);
});
updateOkButton();
var cancel = inner.querySelector('[action-type="cancel"]');
cancel.addEventListener('click', function () { dialog.hide(); })
};
var dialog = Dialog('yawf-drop-select', '{{fastCreateChoseTitle}}', dialogInner);
return dialog;
};
// 将文本、链接等拖拽到框内,快速创建过滤器
var dropdown = (function () {
var dropArea, content;
var dropdownFilters = [];
var got = function (element) {
if (!element) return;
// 忽略最外面套的em/strong
if (element.tagName && element.firstChild === element.lastChild &&
['strong', 'em'].indexOf(element.tagName.toLowerCase()) !== -1)
element = element.firstChild;
debug('got %o', element);
var chose = [];
dropdownFilters.forEach(function (filter, i) { chose[i] = null; });
dropdownFilters.forEach(function (filter, i) {
filter.valid(element, function (val) {
if (!val) chose[i] = [];
else if (val.constructor !== Array) chose[i] = [{ 'filter': filter, 'val': val }];
else chose[i] = val.map(function (val) { return { 'filter': filter, 'val': val }; });
if (chose.filter(function (x) { return !x; }).length) return;
var chosen = chose.reduce(function (x, y) { return x.concat(y); });
console.log('chose: %o', chosen);
if (chosen.length) {
var dialog = fastFilterDialog(chosen);
call(function () { dialog.show(); });
}
});
});
};
var init = function () {
// 拖放到的框
dropArea = cewih('div', html.dropArea).firstChild;
var active = false, target = null;
content = dropArea.querySelector('#yawf-drop-area-content');
document.body.appendChild(dropArea);
var hideDropArea = function () {
if (target && target.hover) target.hover(); target = null;
dropArea.style.display = 'none';
};
// 只接受从微博拽过来的东西
var checkTarget = function () {
var checker = target;
for (var checker = target; checker && checker !== document; checker = checker.parentNode) {
if (!checker.classList || !checker.classList.contains || !checker.tagName) continue;
if (checker.classList.contains('WB_feed')) return true;
if (checker.classList.contains('W_miniblog')) return false;
if (checker.tagName.toLowerCase() === 'body') return true;
}
return false;
};
// 开始拽
document.addEventListener('dragstart', function (e) {
var cover = document.querySelector('body>div[node-type="outer"]');
if (cover && cover.clientHeight) return;
target = e.explicitOriginalTarget; if (target && !checkTarget()) return;
active = true;
dropArea.style.display = 'block';
}, false);
// 拽完了
document.addEventListener('dragend', function (e) {
console.log('dragend! %o', e);
if (!active) return; active = false;
e.preventDefault(); e.stopPropagation();
if (content.firstChild === content.lastChild) got(content.firstChild);
else got(content);
content.innerHTML = '';
hideDropArea();
}, false);
// 拽出去了
document.addEventListener('mouseout', function (e) {
if (!active) return; active = false;
hideDropArea();
});
};
var add = function (details) {
dropdownFilters.push(details);
};
return {
'init': init,
'add': add,
}
}());
var bindInputValue = function (input, obj, attr, standlize) {
var onchange = function () {
var val = input.value, valid;
if (standlize) valid = standlize(val); else valid = val;
if (String(valid) !== val) input.value = valid;
obj['putconf' + attr](valid);
};
input.value = obj['getconf' + attr]();
input.addEventListener('change', onchange);
return onchange;
};
// 告诉服务器屏蔽被隐藏的微博
var blockWeibo = (function () {
var buffer = [], busy = false;
var delay = function () { return 3000 + Math.round(20 * Math.random()) * 100; };
var block = function (mid, callback) {
var done = function () { setTimeout(callback, delay()); };
debug('blocking weibo %s', mid);
GM_xmlhttpRequest({
'method': 'POST',
'url': fillStr(url.block_wb, { 'rnd': dateStr() }),
'headers': {
'Accept': 'text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8',
'Cache-Control': 'no-cache',
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8',
'Cookie': document.cookie,
'Referer': location.href,
'X-Requested-With': 'XMLHttpRequest',
},
'data': 'filter_type=0&mid=' + mid + '&justhide=0&location=home&_t=0',
'onload': function (resp) { debug('block %s response: %s', mid, resp.responseText); done(); },
'onerror': function () { debug('block %s network error', mid); done(); },
});
};
var active = function () {
if (!(busy = !!buffer.length)) return;
var mid = buffer.shift();
block(mid, active);
};
return function (mid) {
buffer.push(mid);
if (!busy) active();
return mid;
};
}());
// 检查是否有新的节点
var newNode = (function () {
var callbacks = [], actived = false;
var callAll = function (mutation) {
callbacks.forEach(function (c) { c(mutation); });
};
var observe = function () {
callAll();
(new MutationObserver(function (mutations) {
mutations.forEach(function (mutation) { callAll(mutation); });
})).observe(document.body, { 'childList': true, 'subtree': true });
};
var add = function (callback) {
callbacks.push(withTry(callback));
return callback;
};
var remove = function (callback) {
var found = false;
callbacks = callbacks.filter(function (x) {
if (x === callback) return found = true; return false;
});
};
var active = function () {
if (actived) return;
actived = true;
observe();
};
return {
'add': add,
'remove': remove,
'active': active,
};
}());
// 添加点击后展开折叠消息的事件
var fixFoldWeibo = (function () {
var fixOne = withTry(function (feed) {
var display = feed.getAttribute('yawf-display').replace(/fold$/g, 'unfold');
var showFeed = function () {
feed.setAttribute('yawf-display', display);
feed.removeEventListener('click', showFeed);
};
var author = feed.querySelector('.WB_name[usercard]').getAttribute('title');
feed.setAttribute('yawf-author', author);
feed.addEventListener('click', showFeed);
});
var fix = function (feed) {
var feeds = [feed].concat(Array.apply(Array, feed.querySelectorAll('.WB_feed_type')));
feeds.forEach(function (feed) {
if (feed.getAttribute('yawf-display').lastIndexOf('-fold') === -1) return;
if (feed.getAttribute('yawf-author')) return;
fixOne(feed);
});
};
fix.init = function () {
css.add(fillStr('[node-type="feed_list"] .WB_feed_type[yawf-display$="-fold"]::before { content: {{foldedWeiboText}}; }'));
};
return fix;
}());
var swapParentSonWeibo = function (parent, son) {
var x = cewih('div', '');
['.WB_face', '.WB_info', '.WB_text', '.WB_func'].map(function (q) {
(function (a, b) {
b.parentNode.replaceChild(x, b);
a.parentNode.replaceChild(b, a);
x.parentNode.replaceChild(a, x);
}(parent.querySelector(q), son.querySelector(q)));
});
var mid = parent.getAttribute('mid');
parent.setAttribute('mid', son.getAttribute('mid'));
son.setAttribute('mid', mid);
// 各种细节的修补(原网站做的太乱了……)
var pf = parent.querySelector('.WB_face'), sf = son.querySelector('.WB_face');
var pfa = pf.querySelector('a'), pfi = pf.querySelector('img');
var sfa = sf.querySelector('a'), sfi = sf.querySelector('img');
if (pfa.href.indexOf('?') === -1) pfa.href += '?from=feed&loc=avatar';
if (sfa.href.indexOf('?') !== -1) sfa.href = sfa.href.slice(0, sfa.href.indexOf('?'));
if (!pfa.title) pfa.title = pfi.title;
if (sfa.title) sfa.removeAttribute('title');
sfi.width = sfi.height = '30'; pfi.width = pfi.height = '50';
return son;
};
// 如果有一个微博的子微博都隐藏了,那么就隐藏这个微博的子微博框
var fixSonWeiboDisplay = function (feed) {
if (!feed.querySelector('.WB_feed_together')) return fixFoldWeibo(feed);
var sonList = Array.apply(Array, feed.querySelectorAll('.WB_feed_together .WB_sonFeed .WB_feed_type[yawf-display]'));
// 交换两个子微博
var swapSon = function (p, q) {
var x = sonList[p], y = sonList[q];
var fakeNode = document.createElement('div');
x.parentNode.insertBefore(fakeNode, x);
y.parentNode.insertBefore(x, y);
fakeNode.parentNode.insertBefore(y, fakeNode);
fakeNode.parentNode.removeChild(fakeNode);
sonList[p] = y; sonList[q] = x;
};
// 把子微博重新排序一下,按照从想看到不想看的顺序排列,排序算法和一趟快排差不多
var p = 0, q;
['show', 'unset', 'unfold', 'fold', 'hidden'].forEach(function (display) {
q = sonList.length - 1;
while (p < q) {
if (sonList[p].getAttribute('yawf-display').lastIndexOf(display) !== -1) p++;
else if (sonList[q].getAttribute('yawf-display').lastIndexOf(display) === -1) q--;
else swapSon(p, q);
}
});
// 看看还有多少显示出来的子微博,更新一下子微博的计数
var sonCount = feed.querySelectorAll('.WB_feed_together .WB_sonFeed .WB_feed_type:not([yawf-display$="-hidden"])').length;
if (sonCount === 0) feed.querySelector('.WB_feed_together').setAttribute('yawf-display', 'display-hidden');
else feed.querySelector('[node-type="followNum"]').textContent = sonCount;
// 如果下面更多的按钮已经没用了,就藏起来吧
var foldSonCount = feed.querySelectorAll('[node-type="feed_list_wrapForward"] .WB_feed_type:not([yawf-display$="-hidden"])').length;
if (foldSonCount === 0 && feed.querySelector('[node-type="feed_list_wrapForward"]')) {
feed.querySelector('.WB_feed_together').setAttribute('yawf-sonfold', 'display');
}
// 把原始微博和子微博拆开
var another = feed.cloneNode(true);
feed.parentNode.insertBefore(another, feed.nextSibling);
feed.setAttribute('yawf-withson', 'son');
another.setAttribute('yawf-display', 'display-son');
fixFoldWeibo(feed);
fixFoldWeibo(another);
};
// 真正微博过滤的核心模块
var weiboFilter = function () {
var feeds = Array.apply(Array,
document.querySelectorAll('[node-type="feed_list"] .WB_feed_type:not([yawf-display])'));
feeds.forEach(function (feed) {
// 同源合并的微博
var sonFeeds = Array.apply(Array, feed.querySelectorAll('[node-type="feed_list"] .WB_feed_type:not([yawf-display])'));
var action = null, parentAction = null;
var needSwap = function (action) {
if (!sonFeeds.length) return false;
if (['hidden', 'fold'].indexOf(action) === -1) return false;
return true;
};
var setAction = function (feed, action) {
feed.setAttribute('yawf-display', 'display-' + action);
};
while (true) {
action = rules.parse(feed) || 'unset';
// 如果父微博被屏蔽或折叠,那么就把下面一条没被屏蔽的拉上来换个位置
if (!needSwap(action)) break;
setAction(swapParentSonWeibo(feed, sonFeeds.pop()), action);
}
parentAction = action;
// 最后处理所有下面的子微博
var fixSonWeibo = function (son) {
var action = rules.parse(son) || 'unset';
setAction(son, action);
// 如果一列里面有白名单的,那么把白名单的特别换到外面去
if (parentAction !== 'show' && action === 'show') {
setAction(swapParentSonWeibo(feed, son), parentAction);
parentAction = action;
}
};
while (sonFeeds.length) fixSonWeibo(sonFeeds.shift());
setAction(feed, parentAction);
fixSonWeiboDisplay(feed);
});
return feeds;
};
var addWeiboFilterListener = (function () {
var listeners = [];
// 逐条进行过滤
newNode.add(function () {
var feeds = weiboFilter();
feeds.forEach(function (feed) {
listeners.forEach(function (f) { f(feed); });
});
});
return function (callback) {
listeners.push(withTry(callback));
};
}());
var typedConfig = (function () {
var inits = [];
// 字符串
var baseConfig = function (type) {
return function (key) {
return function (item) {
var skey = item.key + (key ? '.' + key : key);
if (!item['getconf' + key]) item['getconf' + key] = function () {
return item['conf' + key] = config.get(skey, item['default' + key] || type(), type);
};
if (!item['putconf' + key]) item['putconf' + key] = function (conf) {
return config.put(skey, item['conf' + key] = conf);
};
inits.push(function () {
config.reg(skey);
item['putconf' + key](item['getconf' + key]());
});
};
};
};
// 集合类型的add/del操作
var itemsConfig = function (key) {
return function (item) {
baseConfig(Array)(key)(item);
if (!item['delconf' + key]) item['delconf' + key] = function (str) {
var val = item['getconf' + key]();
val = val.filter(function (x) { return x !== str; });
return item['putconf' + key](val)
};
if (!item['addconf' + key]) item['addconf' + key] = function (str) {
item['delconf' + key](str);
var val = item['getconf' + key](); val.push(str);
return item['putconf' + key](val)
};
};
};
var init = function () {
while (inits.length) inits.shift()();
};
init.string = baseConfig(String);
init.strings = itemsConfig;
init.boolean = baseConfig(Boolean);
init.users = itemsConfig;
init.number = baseConfig(Number);
return init;
}());
// 根据不同类型生成带有事件的文档节点
var typedHtml = (function () {
// 副标题
var subtitle = function (item) {
return cewih('div', fillStr(html.configSubtitle, item)).firstChild;
};
// 文本
var text = function (item) {
return cewih('div', fillStr(html.configText, item)).firstChild;
};
// 不缩进的文本
var remark = function (item) {
return cewih('div', fillStr(html.configRemark, item)).firstChild;
};
// 真假值的设置项
var boolean = function (item) {
var dom = cewih('div', fillStr(html.configBoolean, item)).firstChild;
var input = dom.querySelector('input');
input.checked = item.getconf();
input.addEventListener('change', function () {
item.putconf(input.checked);
});
return dom;
};
// 字符串的设置项
var string = function (item) {
var dom = cewih('div', fillStr(html.configString, item)).firstChild;
var textarea = dom.querySelector('textarea');
textarea.value = item.getconf();
textarea.addEventListener('change', function () { item.putconf(textarea.value); });
textarea.addEventListener('keypress', function () { item.putconf(textarea.value); });
return dom;
};
// 字符串数组设置项模板
var items = function (base, genli, item) {
var dom = cewih('div', fillStr(base, item)).firstChild;
var form = dom.querySelector('form'), input = dom.querySelector('input'), ul = dom.querySelector('ul');
var shown = {};
// 将某个已经有的字符串显示到末尾
var moveToEnd = function (x) {
var p = x.parentNode; p.appendChild(p.removeChild(x));
};
// 显示一个新的字符串
var showStrings = function (userinput, str, onsucc, ondone) {
genli(item, userinput, str, function (str, li) {
if (ondone) ondone();
if (!str || !li) return;
if (shown[str]) return moveToEnd(shown[str]);
var del = li.querySelector('a.icon_close');
del.addEventListener('click', function () {
delete shown[str];
if (item.del) item.del(str);
li.parentNode.removeChild(li);
item.delconf(str);
});
ul.appendChild(shown[str] = li);
if (onsucc) onsucc(str);
});
};
item.getconf().forEach(function (str) { showStrings(false, str); });
form.addEventListener('submit', function (e) {
e.preventDefault();
var str = input.value;
input.disabled = true;
showStrings(true, str, function (str) {
item.addconf(str);
}, function () {
input.value = '';
input.disabled = false;
});
});
return dom;
};
// 字符串设置项
var strings = items.bind(null, html.configStrings, function (item, userinput, str, callback) {
if (userinput && item.add && !(str = item.add(str))) callback();
else callback(str, cewih('ul', fillStr(html.configStringsItem, { 'item': item.display ? item.display(str) : str })).firstChild);
});
// 用户列表的设置项
var users = items.bind(null, html.configUsers, function (item, userinput, str, callback) {
var showUserNotExistError = function () {
Alert('yawf-user-not-exist', {
'title': fillStr('{{accountNotExistErrorTitle}}'),
'text': fillStr('{{{accountNotExistError}}}', { 'name': escapeXml(str) }),
'icon': 'error'
});
callback();
};
if (userinput) {
if (!(str = str.trim().replace(/^@/, ''))) return callback();
account.name(str, function (info) {
if (!info) showUserNotExistError();
else if (item.add && !item.add(info)) callback();
else callback(info.id, cewih('ul', fillStr(html.configUsersItem, info)).firstChild);
}, showUserNotExistError);
} else {
var emptyInfo = { 'id': str, 'name': ' ', 'avatar': ' ' };
var li = cewih('ul', fillStr(html.configUsersItem, emptyInfo)).firstChild;
callback(str, li);
account.id(str, function (info) {
var u = li.querySelector('[uid]');
u.setAttribute('uid', info.id); u.setAttribute('title', info.name); u.textContent = info.name;
var p = li.querySelector('.pic img');
p.src = info.avatar;
});
}
});
return {
'subtitle': subtitle,
'text': text,
'remark': remark,
'string': string,
'strings': strings,
'boolean': boolean,
'users': users,
};
}());
// 过滤器组
var filterGroup = function (groupName) {
var items = [];
// 向过滤器组里面添加一个项目
var add = function (item) {
items.push(item);
try {
if (item.type && typedConfig[item.type])
typedConfig[item.type]('')(item);
if (item.keys) Object.keys(item.keys).forEach(function (key) {
if (typedConfig[item.keys[key]]) typedConfig[item.keys[key]](key)(item);
});
} catch (e) { }
return item;
};
// 网页被初始化时初始化所有过滤器
var init = function () {
items.forEach(withTry(function (item) {
if (item.init) item.init();
if (item.rule) rules.add(item.priority || 0, item.rule.bind(item));
}));
};
// 需要显示选项时生成界面
var show = function (inner) {
items.forEach(withTry(function (item) {
var dom = null;
if (item.show) dom = item.show(dom);
else if (item.type && typedHtml[item.type])
dom = typedHtml[item.type](item);
if (dom && item.shown) item.shown(dom);
if (dom) inner.appendChild(dom);
}));
};
// 注册到过滤器分组
var group = {
'name': groupName,
'show': show,
'init': init,
'add': add
};
return filters.add(group);
};
// 当前是查看分组的页面吗?
var onGroupPage = function () {
return location.pathname.slice(-9) === '/mygroups';
};
// 白名单、黑名单和折叠名单一式三份
var allInOneFilters = function (details) {
var filters = [
{ 'type': 'whitelist', 'priority': 1e5, 'action': 'show' },
{ 'type': 'blacklist', 'priority': 0, 'action': 'hidden' },
{ 'type': 'foldlist', 'priority': -1e5, 'action': 'fold' },
];
var rules = {};
var typedFilterGroup = filterGroup(details.name + 'FilterGroup');
filters.forEach(function (filter) {
typedFilterGroup.add({
'type': 'subtitle',
'text': '{{{' + filter.type + 'FilterDesc}}}',
'typed': '{{' + details.name + 'FilterDetails}}',
'shown': function (dom) {
dom.classList.add('yawf-' + filter.type + 'FilterTitle');
},
});
var rule = {
'type': details.type || 'strings',
'key': 'weibo.filters.' + details.name + '.' + filter.type,
'priority': filter.priority,
'text': '{{' + details.name + 'FilterDesc}}',
};
if (details.add) rule.add = details.add.bind(rule);
if (details.display) rule.display = details.display.bind(rule);
rule.rule = details.rule.bind(rule, filter.action);
if (details[filter.type]) {
Object.keys(details[filter.type]).forEach(function (override) {
rule[override] = details[filter.type][override](rule[override]);
});
}
rules[filter.type] = typedFilterGroup.add(rule);
});
if (details.fast) {
dropdown.add({
'valid': details.fast.valid,
'desc': details.fast.desc,
'add': function (val, action) {
rules[action].addconf(details.fast.add(val));
},
});
}
return typedFilterGroup;
};
// 方便的选择器
var weiboContentSelector = function (feed, f) {
var content = feed.querySelector('[node-type="feed_list_content"]');
var reason = feed.querySelector('[node-type="feed_list_reason"] em');
var items = [];
if (content) items = items.concat(Array.apply(Array, f(content)));
if (reason) items = items.concat(Array.apply(Array, f(reason)));
return items;
};
// 关键字过滤
var keywordFilterGroup = allInOneFilters({
'name': 'keyword',
'add': function (s) { return s.trim(); },
'rule': function keywordMatch(action, feed) {
var keywords = this.conf;
var texts = weiboContentSelector(feed, function (m) {
return Array.apply(Array, m.childNodes)
.filter(function (node) { return node.nodeType === Node.TEXT_NODE; })
.map(function (node) { return node.textContent; });
}).join(' ').toUpperCase();
var match = keywords.some(function (keyword) {
if (!keyword) return false;
return texts.indexOf(keyword.toUpperCase()) !== -1;
});
if (match) return action; else return null;
},
'fast': {
'valid': function (element, callback) {
var valid = false;
if (element.nodeType === Node.TEXT_NODE) valid = true;
else if ((element.tagName || '').toLowerCase() === 'em' &&
element.firstChild === element.lastChild &&
element.firstChild.nodeType === Node.TEXT_NODE) valid = true;
if (valid) callback({ 'text': element.textContent }); else callback();
},
'desc': function (val) {
return fillStr('{{{keywordFilterFast}}}', { 'text': escapeXml(val.text) });
},
'add': function (val) { return val.text; },
}
});
// 按照正则式过滤
var regexpFilterGroup = allInOneFilters({
'name': 'regexp',
'add': function (s) {
s = s.trim();
if (s[0] === '/' && s[s.length - 1] === '/') s = s.slice(1, -1);
try { RegExp(s).exec(''); } catch (e) {
Alert('yawf-regexp-bad-formed', {
'title': fillStr('{{regexpBadFormedTitle}}'),
'text': fillStr('{{{regexpBadFormed}}}', { 'regexp': escapeXml(s) }),
'icon': 'error'
});
s = null;
}
return s;
},
'display': function (s) { return '/' + s + '/'; },
'rule': function regexpMatch(action, feed) {
if (!this.regexen) this.regexen = this.conf.map(function (s) {
try { return RegExp(s); }
catch (e) { debug('erorr while compile regexp %s : %o', s, e); }
}).filter(function (x) { return x; });
var regexen = this.regexen;
var texts = weiboContentSelector(feed, function (m) {
return Array.apply(Array, m.childNodes)
.filter(function (node) { return node.nodeType === Node.TEXT_NODE; })
.map(function (node) { return node.textContent; });
}).join(' ');
var match = regexen.some(function (regexp) {
return !!(regexp.exec(texts));
});
if (match) return action; else return null;
},
});
regexpFilterGroup.add({
'type': 'text',
'text': '{{regexpFilterRemark}}',
});
// 从一条微博中找到他的作者
var getFeedAuthorId = function (feed) {
var author = feed.querySelector('.WB_name[usercard]');
if (!author) return null;
return author.getAttribute('usercard').split('=')[1];
};
// 检查一个元素是不是用户相关的
var validUserElement = function (element, callback) {
var user = null;
if (element.nodeType === Node.TEXT_NODE) return callback();
var a = null;
// 作者或来源
try {
if (element.classList.contains('WB_name')) a = element;
else a = element.querySelector('.WB_name');
if (a && (a.getAttribute('usercard') || '').indexOf('id=') === 0) {
return callback({
'name': a.getAttribute('title'),
'id': a.getAttribute('usercard').slice(3),
});
}
} catch (e) {}
// 提及
try {
if (element.getAttribute('usercard')) a = element;
else a = element.querySelector('[usercard]');
if (a && (a.getAttribute('usercard') || '').indexOf('name=') === 0) {
return account.name(a.getAttribute('usercard').slice(5), callback, callback);
}
} catch (e) {}
// 其他
try {
if (element.getAttribute('title') && element.getAttribute('uid')) a = element;
else if (element.getAttribute('title') && element.getAttribute('usercard').indexOf('id=') === 0) a = element;
else a = element.querySelector('[title][uid], [title][usercard^="id="]');
if (a) {
return callback({
'name': a.getAttribute('title'),
'id': a.getAttribute('uid') || a.getAttribute('usercard').slice(3),
});
}
} catch (e) { }
callback();
};
// 作者用户过滤
var accountFilterGroup = allInOneFilters({
'name': 'account',
'type': 'users',
'rule': function accountMatch(action, feed) {
var accounts = this.conf, id = getFeedAuthorId(feed);
if (!id) return null;
var match = accounts.some(function (x) { return x === id; });
if (match) return action; else return null;
},
'blacklist': {
'rule': function accountMatchBlacklistOverride(_super) {
return function accountMatchBlacklist(feed) {
if (!accountByGroup.conf || !onGroupPage()) return _super(feed);
return null;
};
},
},
'fast': {
'valid': validUserElement,
'desc': function (val) {
return fillStr('{{{accountFilterFast}}}', { 'name': escapeXml(val.name) });
},
'add': function (val) { return val.id; },
},
});
accountFilterGroup.add({
'type': 'text',
'text': '{{accountFilterRemark}}',
});
// 从一条微博中找到他的作者
var getFeedOriginalId = function (feed) {
var originalAuthor = feed.querySelector('.WB_media_expand .WB_info .WB_name');
if (!originalAuthor) return null;
return originalAuthor.getAttribute('usercard').split('=')[1];
};
// 原创用户过滤
var originalFilterGroup = allInOneFilters({
'name': 'original',
'type': 'users',
'rule': function originalMatch(action, feed) {
var originals = this.conf, id = getFeedOriginalId(feed);
if (!id) return null;
var match = originals.some(function (x) { return x === id; });
if (match) return action; else return null;
},
'fast': {
'valid': validUserElement,
'desc': function (val) {
return fillStr('{{{originalFilterFast}}}', { 'name': escapeXml(val.name) });
},
'add': function (val) { return val.id; },
},
});
// 找到在一条微博里面被提到的人的昵称
var getFeedMentionList = function (feed) {
return weiboContentSelector(feed, function (m) {
return Array.apply(Array, m.querySelectorAll('a[usercard^="name="][href$="loc=at"]'));
}).map(function (link) {
return link.getAttribute('usercard').slice('name='.length);
});
};
// 提到某人的微博
var mentionFilterGroup = allInOneFilters({
'name': 'mention',
'type': 'strings',
'add': function (s) { return s.trim().replace(/^@/, ''); },
'display': function (s) { return '@' + s; },
'rule': function mentionMatch(action, feed) {
var mentions = this.conf, users = getFeedMentionList(feed);
var match = users.some(function (name) {
return mentions.indexOf(name) !== -1;
});
if (match) return action; else return null;
},
'fast': {
'valid': validUserElement,
'desc': function (val) {
return fillStr('{{{mentionFilterFast}}}', { 'name': escapeXml(val.name) });
},
'add': function (val) { return val.name; },
},
});
var getFeedTopicList = function (feed) {
return weiboContentSelector(feed, function (m) {
return Array.apply(Array, m.querySelectorAll('.a_topic'));
}).map(function (topic) { return topic.textContent; });
};
// 话题过滤
var topicFilterGroup = allInOneFilters({
'name': 'topic',
'add': function (s) { return s.trim().replace(/#/g, ''); },
'display': function (s) { return '#' + s + '#'; },
'rule': function topicMatch(action, feed) {
var topics = this.conf;
var text = getFeedTopicList(feed).join('##');
var match = topics.some(function (topic) { return text.indexOf(topic) !== -1; });
if (match) return action; else return null;
},
'fast': {
'valid': function (element, callback) {
var a = null;
if (element.nodeType === Node.TEXT_NODE) return callback();
try {
if (element.tagName.toLowerCase() === 'a' && element.classList.contains('a_topic')) a = element;
else a = element.querySelector('a.a_topic');
if (a) return callback({ 'topic': a.textContent.trim().replace(/#/g, '') });
} catch (e) { }
callback();
},
'desc': function (val) {
return fillStr('{{{topicFilterFast}}}', { 'topic': escapeXml(val.topic) });
},
'add': function (val) { return val.topic; },
}
});
// 获取一条微博的所有来源(包括转发)
var getFeedSourceList = function (feed) {
return ['[node-type="feed_list_funcLink"] [action-type="app_source"]',
'.WB_media_expand [action-type="app_source"]'].map(function (qs) {
var st = feed.querySelector(qs); if (!st) return null;
return st.getAttribute('title') || st.textContent || '未通过审核应用';
}).filter(function (x) { return x; });
};
// 来源过滤
var sourceFilterGroup = allInOneFilters({
'name': 'source',
'add': function (s) {
s = s.trim();
if (s === '微博 weibo.com') {
Alert('yawf-source-filter-warning', {
'title': fillStr('{{sourceFilterWarningTitle}}'),
'text': fillStr('{{sourceFilterWarning}}'),
'icon': 'error'
});
s = null;
}
return s;
},
'rule': function sourceMatch(action, feed) {
var sources = this.conf, _sources = getFeedSourceList(feed);
var match = _sources.some(function (s) { return sources.indexOf(s) !== -1; })
if (match) return action; else return null;
},
'fast': {
'valid': function (element, callback) {
var a = null;
if (element.nodeType === Node.TEXT_NODE) return callback();
try {
if (element.getAttribute('action-type') === 'app_source') a = element;
else a = element.querySelector('[action-type="app_source"]');
if (a) {
var source = a.getAttribute('title') || a.textContent || '未通过审核应用';
if (source && source !== '微博 weibo.com') return callback({ 'source': source });
}
} catch (e) { console.log('%o: %o', e, e.stack); }
callback();
},
'desc': function (val) {
return fillStr('{{{sourceFilterFast}}}', { 'source': escapeXml(val.source) });
},
'add': function (val) { return val.source; },
}
});
var getFeedHyperlinkList = function (feed) {
return Array.apply(Array, feed.querySelectorAll('a[title][href^="http://t.cn/"]')).map(function (a) {
return a.getAttribute('title');
});
};
// 超链接过滤
var hyperlinkFilterGroup = allInOneFilters({
'name': 'hyperlink',
'add': function (s) { return s.trim(); },
'rule': function hyperlinkMatch(action, feed) {
var links = this.conf, _links = getFeedHyperlinkList(feed);
var match = _links.some(function (l) {
return links.some(function (link) {
return l.indexOf(link) !== -1;
});
});
if (match) return action; else return null;
},
'fast': {
'valid': function (element, callback) {
if (element.nodeType === Node.TEXT_NODE) return callback();
try {
if (element.tagName.toLowerCase() === 'a' &&
element.getAttribute('title') &&
element.getAttribute('href').indexOf('http://t.cn/') === 0) a = element;
else a = element.querySelector('a[title][href^="http://t.cn/"]');
if (a) {
var x = document.createElement('a'); x.href = a.getAttribute('title');
if (x.host !== 'weibo.com') return callback({ 'host': x.host });
}
} catch (e) { }
callback();
},
'desc': function (val) {
return fillStr('{{{hyperlinkFilterFast}}}', { 'host': escapeXml(val.host) });
},
'add': function (val) { return val.host; },
},
});
var otherFilterGroup = filterGroup('otherFilterGroup');
otherFilterGroup.add({
'type': 'subtitle',
'text': '{{otherWhitelistTitle}}'
});
// 总是显示自己的微博
otherFilterGroup.add({
'type': 'boolean',
'key': 'weibo.other.my_weibo',
'text': '{{showMyWeiboDesc}}',
'priority': 1e5 - 1e3, // 略低于白名单,但高于其他
'rule': function showMyWeiboRule(feed) {
if (!this.conf) return;
if (getFeedAuthorId(feed) === config.uid) return 'showme'; else return null;
},
});
// 总是显示自己原创的微博
otherFilterGroup.add({
'type': 'boolean',
'key': 'weibo.other.my_original',
'text': '{{showMyOriginalDesc}}',
'priority': 1e5 - 1e3, // 略低于白名单,但高于其他
'rule': function showMyOriginalRule(feed) {
if (!this.conf) return;
if (getFeedOriginalId(feed) === config.uid) return 'showme'; else return null;
},
});
// 总是显示自己原创的微博
otherFilterGroup.add({
'type': 'boolean',
'key': 'weibo.other.mention_me',
'text': '{{showMentionMeDesc}}',
'priority': 1e5 - 1e3, // 略低于白名单,但高于其他
'rule': function showMentionMeRule(feed) {
if (!this.conf) return;
if (getFeedMentionList(feed).indexOf(config.uid) !== -1) return 'showme'; else return null;
},
});
otherFilterGroup.add({
'type': 'subtitle',
'text': '{{otherBlacklistTitle}}'
});
// 推广微博
otherFilterGroup.add({
'type': 'boolean',
'key': 'weibo.other.ad_feed',
'text': '{{adfeedFilterDesc}}',
'rule': function adFeedFilterRule(feed) {
if (!this.conf) return null;
return feed.getAttribute('feedtype') === 'ad' ? 'hidden' : null;
},
});
// 关注推荐微博
otherFilterGroup.add({
'type': 'boolean',
'key': 'weibo.other.fake_weibo',
'text': '{{fakeWeiboFilterDesc}}',
'rule': function fakeWeiboFilterRule(feed) {
if (!this.conf) return null;
if (!feed.getAttribute('mid')) return 'hidden';
return null;
},
});
// 已删除或没有权限查看的微博的转发
otherFilterGroup.add({
'type': 'boolean',
'key': 'weibo.other.deleted_forward',
'text': '{{deletedForwardFilterDesc}}',
'rule': function deletedForwardFilterRule(feed) {
if (!this.conf) return null;
if (feed.getAttribute('isforward') === '1' &&
!feed.querySelector('.WB_media_expand .WB_info .WB_name')) return 'hidden';
return null;
},
});
// 投票微博
otherFilterGroup.add({
'type': 'boolean',
'key': 'weibo.other.vote_weibo',
'text': '{{voteWeiboFilterDesc}}',
'rule': function voteWeiboFilterRule(feed) {
if (!this.conf) return null;
if (feed.querySelector('.WB_from a[href^="http://vote.weibo.com/"]'))
return 'hidden';
return null;
},
});
otherFilterGroup.add({
'type': 'subtitle',
'text': '{{otherSpammingTitle}}',
});
// 添加数量和折叠/隐藏的
var lotShown = function (dom) {
var nl = cewih('label', fillStr(text.sameNumberOptionDesc, { 'number': html.numberInput }));
var sl = cewih('label', fillStr(text.sameActionOptionDesc, {
'select': html.select,
'options': fillStr(html.option, { 'value': 'fold', 'text': '{{foldlistActionDesc}}' }) +
fillStr(html.option, { 'value': 'hidden', 'text': '{{blacklistActionDesc}}' }),
}));
var n = nl.querySelector('input'), s = sl.querySelector('select');
n.setAttribute('min', '1');
bindInputValue(n, this, 'number', function (val) {
if (isNaN(val)) val = 1; else val = parseInt(val);
return Math.max(val, 1);
});
bindInputValue(s, this, 'action');
dom.appendChild(nl); dom.appendChild(sl);
};
// 相同账号的过多微博
otherFilterGroup.add({
'type': 'boolean',
'priority': -1e6, // 低优先级
'key': 'weibo.other.same_account',
'keys': { 'number': 'number', 'action': 'string' },
'defaultnumber': 5,
'defaultaction': 'fold',
'text': '{{sameAccountFilterDesc}}',
'shown': lotShown,
'rule': function sameAccountRule(feed) {
if (!this.conf) return null;
if (sameAccountByGroup.conf && onGroupPage()) return null;
var author = feed.querySelector('.WB_name[usercard]');
if (!author) return null;
var id = author.getAttribute('usercard').split('=')[1];
var number = document.querySelectorAll('[node-type="feed_list"] ' +
'.WB_feed_type[yawf-display]:not([yawf-display$="-fold"]):not([yawf-display$="-unfold"]):not([yawf-display$="-hidden"])' +
'>.WB_feed_datail>.WB_detail>.WB_info>a.WB_name[usercard="id=' + id + '"]').length;
if (number >= this.confnumber) return 'account-' + this.confaction; else return null;
},
});
// 相同微博的过多转发
otherFilterGroup.add({
'type': 'boolean',
'priority': -1e6, // 低优先级
'key': 'weibo.other.same_forward',
'keys': { 'number': 'number', 'action': 'string' },
'defaultnumber': 3,
'defaultaction': 'fold',
'text': '{{sameForwardFilterDesc}}',
'shown': lotShown,
'rule': function sameForwardRule(feed) {
if (!this.conf) return null;
var omid = feed.getAttribute('omid');
if (!omid) return null;
var number = document.querySelectorAll('[node-type="feed_list"] ' +
'.WB_feed_type[omid="' + omid + '"][yawf-display]:not([yawf-display$="-fold"]):not([yawf-display$="-unfold"]):not([yawf-display$="-hidden"])').length;
if (number >= this.confnumber) return 'forward-' + this.confaction; else return null;
},
});
// 分组浏览
otherFilterGroup.add({
'type': 'subtitle',
'text': '{{otherGroupTitle}}',
});
// 分组浏览不做按帐号隐藏
var accountByGroup = otherFilterGroup.add({
'type': 'boolean',
'key': 'weibo.other.group_account',
'text': '{{accountByGroup}}',
});
// 分组浏览不做刷屏检查
var sameAccountByGroup = otherFilterGroup.add({
'type': 'boolean',
'key': 'weibo.other.group_same_account',
'text': '{{sameAccountByGroup}}',
});
var layoutFilterGroup = filterGroup('layoutFilterGroup');
layoutFilterGroup.add({
'type': 'subtitle',
'text': '{{layoutFilterGroupDesc}}',
});
// 大部分选择器参考了 眼不见心不烦 脚本
var layouts = (function () {
var current = null;
var subtitle = function (name) {
layoutFilterGroup.add({
'type': 'subtitle',
'text': '{{layoutHide' + name + '}}',
});
current = name;
};
var item = function (name, cssText, defaultValue) {
layoutFilterGroup.add({
'type': 'boolean',
'key': 'weibo.layoutHide' + current + name,
'default': defaultValue || false,
'text': '{{layoutHide' + current + name + '}}',
'init': css(cssText),
});
};
subtitle('Icon');
item('Level', '.icon_bed[node-type="level"] { display: none !important; }');
item('Member', '.W_ico16[class*="ico_member"], .ico_member_dis { display: none !important; }');
item('Approve', '.approve { display: none !important; }');
item('ApproveCo', '.approve_co { display: none !important; }');
item('Club', '.ico_club { display: none !important; }');
item('VGirl', '.ico_vlady { display: none !important; }');
item('Taobao', '.ico_taobao { display: none !important; }');
subtitle('Nav');
item('Main', '.gn_nav>div:nth-child(1) { display: none !important; }');
item('Hot', '.gn_nav>div:nth-child(2) { display: none !important; }');
item('App', '.gn_nav>div:nth-child(3) { display: none !important; }');
item('Game', '.gn_nav>div:nth-child(4) { display: none !important; }');
item('Member', '.gn_setting[node-type="member"] { display: none !important; }');
subtitle('Left');
item('ToMe', '#pl_leftnav_common a[href^="/direct/tome"] { display: none !important; }');
item('Friends', '#pl_leftnav_group > div[node-type="groupList"] > .level_1_Box, #pl_leftnav_common .level_1_Box > form.left_nav_line { display: none !important; }');
item('App', '#pl_leftnav_app { display: none !important; }');
subtitle('Middle');
item('MemberTip', '[node-type="feed_list_shieldKeyword"] { display: none !important; }');
item('RecommendedTopic', '#pl_content_publisherTop div[node-type="recommendTopic"] { display: none !important; }');
subtitle('Right');
item('Whole', '.B_profile .W_main_c, .B_profile .WB_feed .repeat .input textarea { width: 100% } .B_profile .WB_feed .WB_screen { margin-left: 928px } .B_profile .W_main_2r { display: none !important; }');
item('Template', '.templete_enter { display: none !important; }');
item('Info', '.W_person_info { display: none !important; }');
item('Atten', '#pl_rightmod_myinfo .user_atten { display: none !important; }');
item('Trial', '#trustPagelet_checkin_lotteryv5 { display: none !important; }');
item('Interest', '[yawf-id="rightmod_recom_interest"] { display: none !important; }');
item('HotTopic', '[yawf-id="rightmod_zt_hottopic"] { display: none !important; }');
item('Member', '#trustPagelet_recom_memberv5 { display: none !important; }');
item('Weibo', '[yawf-id="rightmod_recom_weibo"] { display: none !important; }');
item('Location', '[yawf-id="rightmod_recom_location"] { display: none !important; }');
item('Music', '[yawf-id="rightmod_recom_music"] { display: none !important; }');
item('Movie', '[yawf-id="rightmod_recom_movie"] { display: none !important; }');
item('Book', '[yawf-id="rightmod_recom_book"] { display: none !important; }');
item('Notice', '#pl_rightmod_noticeboard { display: none !important; }');
subtitle('Weibo');
item('RecomFeed', '[node-type="feed_list_recommend"] { display: none !important; }');
item('TopicCard', '.WB_feed_spec[exp-data*="value=1022-topic"] { display: none !important; }');
item('LocationCard', '.WB_feed_spec[exp-data*="value=1022-place"] { display: none !important; }');
item('FeedTip', '[node-type="feed_privateset_tip"] { display: none !important; }');
subtitle('Person');
item('Cover', funcStr(function () { /* !CSS
.S_profile_pic { display: none; }
.profile_top { margin-top: 20px; }
.profile_top .pf_head { top: 5px; margin-top: 0 !important; }
.profile_top .pf_head_pic { height: 120px; width: 120px; float: right; }
.profile_top .pf_head_pic img { height: 120px; }
.profile_top .pf_head .user_atten { width: 60px; float: left; height: 120px; }
.profile_top .pf_head .user_atten li, .profile_top .pf_head .user_atten .follower { width: 54px; padding: 0 3px 3px; height: 37px; border-right: none; }
.profile_top .pf_head .user_atten li strong { margin: 3px 0 0; }
*/ }));
item('BadgeIcon', '.pf_badge_icon { display: none !important; }');
item('Stats', '.profile_top .user_atten { display: none !important; } .profile_top .pf_head { margin-top: 51px; } ');
item('MyData', '.W_main_c [id^="Pl_Official_MyMicroworld__"] { display: none !important; }');
item('Group', '.W_main_2r [id^="Pl_Core_RightGroupsBtn__"] { display: none !important; }');
item('SuggestUser', '.W_main_2r [id^="Pl_Core_RightUserList__"] { display: none !important; }');
item('Relation', '.W_main_2r [id^="Pl_Core_RightUserGrid__"] { display: none !important; }');
item('Album', '.W_main_2r [id^="Pl_Core_RightPicMulti__"] { display: none !important; }'); // FIXME
item('HotTopic', '.W_main_2r [id^="Pl_Core_RightTextSingle__"] { display: none !important; }');
item('HotWeibo', '.W_main_2r [id^="Pl_Core_RightPicText__"] { display: none !important; }');
subtitle('Other');
item('Ads', '#plc_main [id^="pl_rightmod_ads"], #Box_right [id^="ads_"], #trustPagelet_zt_hottopicv5 [class*="hot_topicad"], div[ad-data], .WB_feed .popular_buss, [id^="sinaadToolkitBox"] { display: none !important; } #wrapAD, .news_logo { visibility: hidden !important; }');
item('FeedRecom', '.W_main_2r [id^="Pl_Third_Inline__"] { display: none !important; }');
item('Footer', '.global_footer { display: none !important; }');
item('WbIm', '.WBIM_news { display: none !important; }');
item('Tip', '.layer_tips { display: none !important; }');
var tagRightbarMods = function () {
var mods = document.querySelectorAll('#trustPagelet_indexright_recom .WB_right_module:not([yawf-id])');
if (!mods) return;
var identifiers = {
'.right_content.hot_topic': 'rightmod_zt_hottopic',
'.right_content.person_list': 'rightmod_recom_interest',
'[change-data*="key=index_weibo"]': 'rightmod_recom_weibo',
'[change-data*="key=index_LBS"]': 'rightmod_recom_location',
'[change-data*="key=index_song"]': 'rightmod_recom_music',
'[change-data*="key=index_mov"]': 'rightmod_recom_movie',
'[change-data*="key=index_book"]': 'rightmod_recom_book'
};
Array.apply(Array, mods).forEach(function (mod) {
Object.keys(identifiers).forEach(function (qs) {
if (mod.querySelector(qs)) mod.setAttribute('yawf-id', identifiers[qs]);
});
});
};
newNode.add(tagRightbarMods);
tagRightbarMods();
css.add('.W_miniblog { visibility: visible !important; }');
}());
// 改造设置
var toolFilterGroup = filterGroup('toolFilterGroup');
// 快速创建过滤器
toolFilterGroup.add({
'type': 'boolean',
'key': 'weibo.tool.use_fast_creator',
'default': true,
'text': '{{useFastCreator}}',
'init': function () {
if (!this.conf) return;
dropdown.init();
},
});
// 清除发布框中的默认话题 (wcf)
toolFilterGroup.add({
'type': 'boolean',
'key': 'weibo.tool.clear_def_topic',
'text': '{{clearDefTopicDesc}}',
'init': function () {
if (!this.conf) return;
var clearDefTopic = function () {
var inputBox = document.querySelector('#pl_content_publisherTop .send_weibo .input textarea');
if (inputBox && inputBox.hasAttribute('hottopic')) {
inputBox.removeAttribute('hottopic'); inputBox.removeAttribute('hottopicid');
inputBox.value = 'DUMMY'; inputBox.focus();
inputBox.value = ''; inputBox.blur();
}
};
newNode.add(clearDefTopic);
},
});
// 展开左栏分组
toolFilterGroup.add({
'type': 'boolean',
'key': 'weibo.tool.noFloatNav',
'text': '{{noFloatNav}}',
'init': css('.WB_global_nav { position: absolute !important; }'),
});
// 展开左栏分组
toolFilterGroup.add({
'type': 'boolean',
'key': 'weibo.tool.showAllGroup',
'text': '{{showAllGroupDesc}}',
'init': css('#pl_leftnav_group div[node-type="moreList"] { display: block !important } #pl_leftnav_group > div[node-type="groupList"] > .level_2_Box > .levmore { display: none }'),
});
// 展开左栏消息
toolFilterGroup.add({
'type': 'boolean',
'key': 'weibo.tool.showAllMsgNav',
'text': '{{showAllMsgNavDesc}}',
'init': css('#pl_leftnav_common > .level_1_Box > .lev2_new { display: block !important }'),
});
// 微博作者与正文同行
toolFilterGroup.add({
'type': 'boolean',
'key': 'weibo.tool.unwrapText',
'text': '{{unwrapTextDesc}}',
'init': css('.WB_info, .WB_text { display: inline } .WB_info+.WB_text::before { content: ": " } .WB_func { margin-top: 5px } .B_index .WB_feed .W_ico16 { vertical-align: -3px !important }'),
});
// 个人主页自动打开微博列表
toolFilterGroup.add({
'type': 'boolean',
'key': 'weibo.tool.redirectWeibo',
'text': '{{personalRedirectWeibo}}',
'init': function () {
if (!this.conf) return;
var locat = unsafeWindow.$CONFIG.location;
if (locat.slice(-5) !== '_home') return;
if (!document.body.classList.contains('B_profile')) return;
var from = (location.search.match(/from=([^&]*)/) || {})[1];
if (locat.indexOf(from) === 0) return;
var redirect = function () {
var link = document.querySelector('.PRF_tab_noicon li.pftb_itm a[href*="/weibo?"]'); if (!link) return;
if (!link) return false;
newNode.remove(redirect);
location.replace(link.href);
};
newNode.add(redirect);
redirect();
},
});
// 查看大图旁添加查看原图链接
toolFilterGroup.add({
'type': 'boolean',
'default': true,
'key': 'weibo.tool.viewOriginal',
'text': '{{viewOriginalDesc}}',
'init': function () {
if (!this.conf) return;
var addOriLink = function () {
var a = document.querySelector('a.show_big[action-data]:not([yawf-viewori])');
if (!a) return; a.setAttribute('yawf-viewori', 'yawf-viewori');
var arg = a.getAttribute('action-data').match(/pid=(\w+)&mid=(\d+)&uid=(\d+)/);
if (!arg) return;
var vol = cewih('div', fillStr(html.viewOriginalLink)), l = vol.firstChild;
var img = fillStr(url.view_ori, { 'uid': arg[3], 'mid': arg[2], 'pid': arg[1] });
while (vol.firstChild) a.parentNode.insertBefore(vol.firstChild, a);
if (0) GM_xmlhttpRequest({
'method': 'GET',
'url': img,
'onload': function (resp) {
var h = (new DOMParser()).parseFromString(resp.responseText, 'text/html');
l.href = h.querySelector('#pic').src;
},
}); l.href = img;
};
newNode.add(addOriLink);
},
});
// 屏蔽隐藏微博
toolFilterGroup.add({
'type': 'boolean',
'key': 'weibo.tool.block_hidden',
'text': '{{blockHiddenWeiboDesc}}',
'init': function () {
if (!this.conf) return;
addWeiboFilterListener(function (feed) {
[feed].concat(Array.apply(Array, feed.querySelectorAll('.WB_feed_type'))).forEach(function (feed) {
var display = feed.getAttribute('yawf-display');
if (display !== 'display-hidden') return;
if (!feed.getAttribute('mid')) return;
blockWeibo(feed.getAttribute('mid'));
feed.setAttribute('yawf-block', 'block');
});
});
}
});
// 显示一个透明度设置框
var genTransparencyInput = function (binder) {
var l = cewih('label', fillStr(text.transparencyInput, { 'number': html.transparencyInput }));
var n = l.querySelector('input[type="number"]'), r = l.querySelector('input[type="range"]');
var onchange = binder(n, function (val) {
if (isNaN(parseInt(val))) return 0;
return Math.min(Math.max(parseInt(val), 0), 100);
});
r.value = n.value;
var updateN = function () { if (r.value !== n.value) n.value = r.value; onchange(); };
var updateR = function () { if (r.value !== n.value) r.value = n.value; onchange(); };
r.addEventListener('change', updateN); r.addEventListener('mousemove', updateN);
n.addEventListener('change', updateR); n.addEventListener('keypress', updateR);
return l;
};
// 显示一个颜色&透明度设置框
var genColorWithTransparencyInput = function (binder1, binder2) {
var color = cewih('label', html.colorInput);
var c = color.querySelector('input[type="color"]');
binder1(c);
var transparency = genTransparencyInput(binder2);
return [color, transparency];
};
// 将颜色和透明度转换为一个表示颜色的字符串
var colorStr = function (color, transparency) {
return 'rgba(' + color.slice(1)
.split(/(..)/).filter(function (x) { return x; })
.map(function (x) { return parseInt(x, 16); }).join(',') +
',' + (100 - transparency) / 100 + ')';
};
// 一个带有颜色/透明度的选框项
var coloredConfigItem = function (details) {
var conf = {
'type': 'boolean',
'key': details.key,
'keys': { 'color': 'string', 'transparency': 'number', 'blur': 'boolean' },
'defaultcolor': details.defaultcolor,
'defaulttransparency': details.defaulttransparency,
'text': details.text,
'shown': function (dom) {
var elements = genColorWithTransparencyInput(function (c, s) {
bindInputValue(c, this, 'color', s);
}.bind(this), function (t, s) {
return bindInputValue(t, this, 'transparency', s);
}.bind(this));
elements.forEach(dom.appendChild.bind(dom));
},
'init': function () {
if (!this.conf) return;
details.init(colorStr(this.confcolor, this.conftransparency));
},
};
return conf;
};
// 高亮显示白名单微博
toolFilterGroup.add(coloredConfigItem({
'key': 'weibo.tool.whitelist_highlight',
'defaultcolor': '#fef3da',
'text': '{{whitelistHighlightDesc}}',
'init': function (color) {
css.add(fillStr(funcStr(function () { /*
[node-type="feed_list"] .WB_feed_type[yawf-display$="-show"] { background-color: {{color}} !important; box-shadow: -20px 0 0 {{color}}, 20px 0 0 {{color}}; }
[node-type="feed_list"] .WB_feed_together .WB_feed_type[yawf-display$="-show"] { background-color: {{color}} !important; box-shadow: -10px 0 0 {{color}}, 10px 0 0 {{color}}; }
*/ }), { 'color': color }));
},
}));
// 首页背景
toolFilterGroup.add(coloredConfigItem({
'key': 'weibo.tool.my_background_color',
'defaultcolor': '#ffffff',
'defaulttransparency': 30,
'text': '{{mainBackgroundColorOverride}}',
'init': function (color) {
css.add(fillStr(funcStr(function () { /*
body:not(.S_profile) .W_main { background-image: none !important; background-color: {{color}} !important }
body:not(.S_profile) .S_bg4, body:not(.S_profile) .W_main_a, body:not(.S_profile) .W_main_bg { background: transparent !important; }
*/ }), { 'color': color }));
},
}));
// 个人主页背景
toolFilterGroup.add(coloredConfigItem({
'key': 'weibo.tool.other_background_color',
'defaultcolor': '#ffffff',
'defaulttransparency': 30,
'text': '{{profileBackgroundColorOverride}}',
'init': function (color) {
css.add(fillStr(funcStr(function () { /*
.S_profile .W_profile_bg, .S_profile .S_bg5 { background-color: {{color}} !important; }
.S_profile .S_bg4:not(.W_profile_bg) { background: none transparent !important }
*/ }), { 'color': color }));
},
}));
// 自定义样式
toolFilterGroup.add({
'type': 'string',
'text': '{{userstyleTitle}}',
'key': 'weibo.tool.userstyle',
'init': function () {
var conf = this.conf; GM_addStyle(conf);
var set = this.putconf.bind(this);
var putconf = function (css) {
conf = css;
setTimeout(function () { set(css); config.write(); location.reload(); }, 0);
};
GM_registerMenuCommand(fillStr('{{userstyleEditDesc}}'), function () {
var newcss = prompt(fillStr('{{userstyleEditDetails}}'), conf);
if (newcss !== null) putconf(newcss);
}, "S");
},
});
// 脚本设置
var scriptFilterGroup = filterGroup('scriptFilterGroup');
// 导入导出
scriptFilterGroup.add({
'type': 'subtitle',
'text': '{{configImportAndExport}}',
});
scriptFilterGroup.add({
'show': function () {
var dom = cewih('div', html.configImportExport).firstChild;
var bii = dom.querySelector('input[type="file"]');
var be = dom.querySelector('[node-type="export"]');
var br = dom.querySelector('[node-type="reset"]');
// 导出按钮
var updateExportButton = function () {
be.href = 'data:application/octet-stream;base64,' +
btoa(unescape(encodeURIComponent(config.export())));
be.setAttribute('download', 'yawf-config.yawf');
};
// 导入按钮
var doImport = function (file) {
var reader = new FileReader();
var success = function () {
Alert('yawf-config-import-success', {
'title': fillStr('{{configImportSuccessTitle}}'),
'text': fillStr('{{configImportSuccess}}'),
'icon': 'succ'
});
};
var error = function () {
Alert('yawf-config-import-fail', {
'title': fillStr('{{configImportFailTitle}}'),
'text': fillStr('{{configImportFail}}'),
'icon': 'error'
});
};
if (file.size > (1 << 24)) error();
else reader.addEventListener('load', function () {
if (config.import(reader.result)) {
updateExportButton();
success();
} else error();
});
reader.readAsText(file);
bii.value = '';
};
bii.addEventListener('change', function () {
var file = bii.files[0];
Confirm('yawf-config-import-warning', {
'title': fillStr('{{configImportWarningTitle}}'),
'text': fillStr('{{configImportWarning}}'),
'onOk': function () { doImport(file); },
});
});
updateExportButton();
// 重置按钮
var doReset = function () {
config.clear();
updateExportButton();
};
br.addEventListener('click', function () {
Confirm('yawf-config-reset-warning', {
'title': fillStr('{{configResetWarningTitle}}'),
'text': fillStr('{{configResetWarning}}'),
'onOk': doReset,
});
});
return dom;
},
});
// 调试
scriptFilterGroup.add({
'type': 'subtitle',
'text': '{{scriptDebugTitle}}',
});
scriptFilterGroup.add({
'show': function () {
var dom = cewih('div', fillStr(html.configBoolean, { 'key': 'debug', 'text': '{{scriptDebug}}' })).firstChild;
var input = dom.querySelector('input');
input.checked = !!GM_getValue('debug', false);
input.addEventListener('change', function () {
GM_setValue('debug', !GM_getValue('debug', false));
});
return dom;
}
});
// 关于
scriptFilterGroup.add({
'type': 'subtitle',
'text': '{{scriptAboutTitle}}',
});
scriptFilterGroup.add({
'type': 'text', 'text': '',
'shown': function (dom) {
dom.innerHTML = fillStr(text.scriptAbout, {
'version': ((GM_info || {}).script || {}).version || '?'
});
},
});
scriptFilterGroup.add({
'init': function () {
var isEn = i18n.lang === 'en';
css.add(fillStr(funcStr(function () { /*!CSS
.profile_tab .pftb_lk { padding-left: {{headerWidth}}; padding-right: {{headerWidth}}; }
.profile_tab .current.pftb_lk { padding-left: {{headerWidth2}}; padding-right: {{headerWidth2}}; }
.layoutFilterGroupLayer .yawf-configBoolean { width: {{layoutOptionWidth}}; }
*/ }), {
'headerWidth': isEn ? '12px' : '15px',
'headerWidth2': isEn ? '9px' : '12px',
'layoutOptionWidth': isEn ? '320px' : '160px',
}));
},
});
// 检查是否要在本页上运行
var validPage = function () {
if (self !== top) return false;
if (!unsafeWindow.$CONFIG.uid) return false;
if (!unsafeWindow.$CONFIG.lang) return false;
return true;
};
// 完成加载时
var dcl = function () {
if (!validPage()) return;
// 初始化用户语言
i18n.setLang(unsafeWindow.$CONFIG.lang);
// 加载用户配置
config = config(unsafeWindow.$CONFIG.uid);
typedConfig();
// 初始化文本和网页数据(基于用户选择的语言)
Object.keys(text).map(function (key) { i18n(text[key]); text[key] = text[key].local; });
Object.keys(html).map(function (key) { html[key] = fillStr(html[key]); });
// 初始化所有过滤器
filters.init();
// 初始化折叠微博
fixFoldWeibo.init();
// 注册样式
css.init();
// 初始化界面
showIcon();
// 开始过滤
newNode.active();
};
if (document.body) call(dcl);
else document.addEventListener('DOMContentLoaded', dcl);
GM_addStyle(fillStr((funcStr(function () { /*!CSS
// 在顶部添加按钮
.gn_setting[node-type="member"]:last-child { margin-right: 44px; }
.WB_global_nav .gn_setting .gn_tab.gn_filter .ico { background-image: url("{{filter-img}}"); !important; background-position: 0 0 !important; }
.WB_global_nav .gn_search { width: 210px !important; left: 440px !important; position: absolute !important; }
.WB_global_nav .gn_search .gn_input { width: 168px !important; }
// 设置框相关样式
.yawf-Layer.yawf-drag { opacity: 0.67; -moz-user-select: none; user-select: none; }
#yawf-config [node-type="inner"] { padding: 20px; }
.yawf-config-body { margin: -20px; max-height: 360px; overflow-y: auto; padding: 20px; width: 760px; }
#yawf-config .profile_tab { font-size: 12px; margin: -20px -20px 20px; width: 800px; }
.yawf-config-layer { padding-bottom: 20px; }
.yawf-groupSubtitle, .yawf-groupRemark { margin: 5px 10px; padding: 10px 0 0 0; }
.yawf-groupSubtitle { font-weight: bold; }
.yawf-configItem, .yawf-groupText { margin: 5px 20px; padding: 5px 0; }
.yawf-whitelistFilterTitle::before, .yawf-blacklistFilterTitle::before, .yawf-foldlistFilterTitle::before { content: " "; display: inline-block; width: 0.8em; height: 0.8em; border-radius: 1em; margin-right: 0.5em; border: 1px solid white; vertical-align: middle; }
.yawf-whitelistFilterTitle::before { background: #37c837; box-shadow: 0 0 2px #37c837; }
.yawf-blacklistFilterTitle::before { background: #c83737; box-shadow: 0 0 2px #c83737; }
.yawf-foldlistFilterTitle::before { background: #c8c837; box-shadow: 0 0 2px #c8c837; }
.yawf-configString span { line-height: 16px; padding: 2px 10px; margin-bottom: -20px; display: block; width: calc(100% - 20px); position: relative; }
.yawf-configString textarea.W_input { width: calc(100% - 20px); padding-top: 20px; min-height: 80px; resize: vertical; }
.yawf-configStringsInput, .yawf-configUsersInput { margin: 5px; }
.yawf-configStringsItems, .yawf-configUsersItems { padding: 5px 10px; }
.yawf-configStringsItem, .yawf-configUsersItem { display: inline-block; margin: 2px; }
.yawf-configStringsItem a.icon_close, .yawf-configUsersItem a.icon_close { margin-left: 3px; vertical-align: -2px; }
.yawf-configUsersItem .shield_object_card { display: inline-block; }
.yawf-configUsersItem .shield_object_card .card_bg { border: 1px solid #e6e6e6; border-radius: 2px; box-shadow: 0 1px 1px rgba(0, 0, 0, 0.02); padding: 1px 5px 1px 1px; }
.yawf-configUsersItem .shield_object_card .card_pic { float: left; width: 50px; }
.yawf-configUsersItem .shield_object_card .card_pic .pic_box { display: block; }
.yawf-configUsersItem .shield_object_card .card_content { margin-left: 60px; }
.yawf-configUsersItem .shield_object_card .object_info { height: 22px; margin-top: 2px; }
.yawf-configUsersItem .shield_object_card .object_name { line-height: 22px; overflow: hidden; }
.yawf-configUsersItem .shield_object_card .other_info { line-height: 24px; }
.yawf-configImportExport [node-type] { margin-right: 20px; }
.yawf-configAdd { appearance: none; }
#yawf-config .btn { border-top: 1px solid #ccc; margin: 15px 0 0; padding: 10px 0 0; }
#yawf-config .btn .W_btn_b_disable:hover { border-color: #d9d9d9; }
#yawf-config .btn .W_btn_b_disable:hover span { border-color: #ffffff; }
.layoutFilterGroupLayer .yawf-configBoolean { display: inline-block; margin-right: 0; }
// 隐藏微博
[yawf-display$="-hidden"] { display: none !important; }
[node-type="feed_list"] .WB_feed_type:not([yawf-display]), [node-type="feed_list"] .WB_feed_type .WB_feed_type:not([yawf-display]) { visibility: hidden !important; }
// 折叠微博
[node-type="feed_list"] .WB_feed_type[yawf-display$="-fold"]::before { display: block; width: 100%; height: 24px; line-height: 24px; padding: 0 2em 20px; border-bottom: #e6e6e6 1px solid; }
[node-type="feed_list"] .WB_feed_type[yawf-display$="-fold"] { height: 45px; overflow: hidden; cursor: pointer; }
[node-type="feed_list"] .WB_feed_type[yawf-display$="-fold"] .WB_screen { margin-top: -40px !important; }
[node-type="feed_list"] .WB_feed_type[yawf-display$="-fold"] .WB_feed_datail { display: none !important; }
// 子微博
[node-type="feed_list"] .WB_feed_type[yawf-display$="-hidden"]+.WB_feed_type[yawf-display$="-son"],
[node-type="feed_list"] .WB_feed_type[yawf-display$="-fold"]+.WB_feed_type[yawf-display$="-son"] { display: none !important; }
[node-type="feed_list"] .WB_feed_type:not([yawf-display$="-son"]) .WB_feed_together { display: none !important; }
[node-type="feed_list"] .WB_feed_type[yawf-withson="son"] .WB_feed_datail,
[node-type="feed_list"] .WB_feed_type[yawf-withson="son"][yawf-display$="-fold"]::before { border: 0 none !important; }
[node-type="feed_list"] .WB_feed_type[yawf-display$="-son"] { padding-top: 0 !important; }
[node-type="feed_list"] .WB_feed_type[yawf-display$="-son"]>.WB_screen,
[node-type="feed_list"] .WB_feed_type[yawf-display$="-son"]>.WB_feed_datail>.WB_face,
[node-type="feed_list"] .WB_feed_type[yawf-display$="-son"]>.WB_feed_datail>.WB_detail>*:not(.WB_feed_together) { display: none !important; }
[node-type="feed_list"] .WB_feed_type[yawf-display$="-son"]>.WB_feed_datail>.WB_detail { margin-top: -20px; margin-bottom: -10px; }
// 其他
.WB_feed_together .wft_users { display: none; }
.WB_feed_together[yawf-sonfold="display"] [node-type="feed_list_wrapForward"] { display: block !important; }
.WB_feed_together[yawf-sonfold="display"] [action-type="feed_list_seeAll"],
.WB_feed_together[yawf-sonfold="display"] [action-type="feed_list_foldForward"] { display: none !important; }
.W_miniblog { visibility: hidden; }
input[type="number"]:not(:focus) ~ .yawf-transparency-range:not(:hover) > input[type="range"]:not(:focus) { display: none; }
// 拖拽
#yawf-drop-area { background: rgba(251, 251, 216, 0.8); display: none; height: 230px; left: calc(50% + 260px); position: fixed; top: 40px; width: 230px; z-index: 9999; }
.yawf-drop-area-desc { height: 170px; width: 170px; margin: 16px 16px -206px 16px; padding: 10px; -moz-user-select: none; user-select: none; border: 4px dashed #ddd; border-radius: 20px; }
.yawf-drop-area-title { font-size: 150%; font-weight: bold; }
.yawf-drop-area-text { padding: 10px; }
#yawf-drop-area-content { height: 230px; width: 230px; position: relative; z-index: 10002; opacity: 0; }
#yawf-fast-filter-chose { padding: 20px 40px; }
#yawf-fast-filter-text { font-weight: bold; }
*/ }) + '\n').replace(/\/\/.*\n/g, '\n'), {
'filter-img': images.filter,
}));