ios_rule_script/script/zheye/zheye.js

1210 lines
75 KiB
JavaScript
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

const scriptName = "哲也同学";
const blockedUsersKey = "zhihu_blocked_users";
const currentUserInfoKey = "zhihu_current_userinfo";
const keywordBlockKey = "zhihu_keyword_block";
const blackAnswersIdKey = "zhihu_black_answers";
const userCreditScore = "zhihu_credit_score";
// 默认屏蔽推荐列表的用户,通常不是真实用户,无法通过加入黑名单屏蔽
const defaultAnswerBlockedUsers = ["会员推荐", "盐选推荐"];
const keywordMaxCount = 1000; // 允许设置的关键词数量
const $ = MagicJS(scriptName, "INFO");
/**
* @description: 获取用户信息
* @return {*}
*/
function getUserInfo() {
let defaultUserInfo = { id: "default", is_vip: false };
try {
let userInfo = $.data.read(currentUserInfoKey);
if (typeof userInfo === "string") userInfo = JSON.parse(userInfo);
if (!!userInfo && userInfo.hasOwnProperty("id")) {
return userInfo;
} else {
return defaultUserInfo;
}
} catch (err) {
$.logger.error(`获取用户信息出现异常:${err}`);
return defaultUserInfo;
}
}
/**
* 优化软件配置
* @return {*}
*/
function modifyAppConfig() {
let response = null;
try {
if (!!$.response.body) {
let obj = JSON.parse($.response.body);
obj["config"]["homepage_feed_tab"]["tab_infos"] = obj["config"]["homepage_feed_tab"]["tab_infos"].filter(
(e) => {
// 将活动标签设置为已过期
if (e["tab_type"] === "activity_tab") {
e["end_time"] = (new Date() - 120000)
.toString()
.slice(0, 10);
return true;
} else {
return false;
}
}
);
obj["config"]["zvideo_max_number"] = 1;
// 似乎是控制内部弹窗
obj["config"]["is_show_followguide_alert"] = false;
// 似乎是某个地方的标签,待定
delete obj["config"]["hp_channel_tab"];
// 灰色模式
if (obj["config"]["zombie_conf"]) {
obj["config"]["zombie_conf"]["zombieEnable"] = false;
}
if (obj["config"]["gray_mode"]) {
obj["config"]["gray_mode"]["enable"] = false;
obj["config"]["gray_mode"]["start_time"] = '4092566400';
obj["config"]["gray_mode"]["end_time"] = '4092566400';
}
// 屏蔽8.X版本以上本地DNS解析以下修改不清楚哪些是有效的暂时全部保留
if (obj["config"].hasOwnProperty("zhcnh_thread_sync")) {
$.logger.debug(JSON.stringify(obj["config"]["zhcnh_thread_sync"]));
obj["config"]["zhcnh_thread_sync"]["LocalDNSSetHostWhiteList"] = [];
obj["config"]["zhcnh_thread_sync"]["isOpenLocalDNS"] = "0";
obj["config"]["zhcnh_thread_sync"]["ZHBackUpIP_Switch_Open"] = "0";
obj["config"]["zhcnh_thread_sync"]["dns_ip_detector_operation_lock"] =
"1";
obj["config"]["zhcnh_thread_sync"][
"ZHHTTPSessionManager_setupZHHTTPHeaderField"
] = "1";
}
response = { body: JSON.stringify(obj) };
}
} catch (err) {
$.logger.error(`优化软件配置出现异常:${err}`);
}
return response;
}
/**
* 修改云端下发的配置
* @return {*}
*/
function modifyMCloudConfig() {
let response = null;
try {
if (!!$.response.body) {
let obj = JSON.parse($.response.body);
if (obj.data && obj.data["configs"]) {
// 去除灰色主题
obj.data["configs"].forEach(element => {
if (element["configKey"] === "feed_gray_theme") {
$.notification.debug("修改灰色主题生效时间");
element["configValue"].start_time = "1669824000";
element["configValue"].end_time = "1669824001";
element.status = false;
}
});
}
const body = JSON.stringify(obj);
$.logger.debug(body);
response = { body: body };
}
} catch (err) {
$.logger.error(`优化软件配置出现异常:${err}`);
}
return response;
}
/**
* 屏蔽关键词解锁
* @return {*}
*/
function unlockBlockedKeywords() {
let response = null;
try {
const userInfo = getUserInfo();
// 获取屏蔽关键词列表
if ($.request.method === "GET") {
let keywords = $.data.read(keywordBlockKey, null, userInfo.id);
if (!keywords) {
keywords = [];
}
let headers = {
"Cache-Control":
"no-cache, no-store, must-revalidate, private, max-age=0",
Connection: "keep-alive",
"Content-Type": "application/json;charset=utf-8",
Pragma: "no-cache",
"Referrer-Policy": "no-referrer-when-downgrade",
Server: "CLOUD ELB 1.0.0",
Vary: "Accept-Encoding",
"X-Cache-Lookup": "Cache Miss",
"x-cdn-provider": "tencent",
};
let body = JSON.stringify({
success: true,
is_vip: true,
kw_min_length: 2,
kw_max_length: 100,
kw_max_count: keywordMaxCount,
data: keywords,
});
if ($.env.isQuanX) {
response = { body: body, headers: headers, status: "HTTP/1.1 200 OK" };
} else {
response = { response: { body: body, headers: headers, status: 200 } };
}
$.logger.debug(`获取本地脚本屏蔽关键词:\n${keywords.join("、")}`);
}
// 添加屏蔽关键词
else if ($.request.method === "POST") {
if (!!$.request.body) {
// 构造 response headers
let headers = {
"Cache-Control":
"no-cache, no-store, must-revalidate, private, max-age=0",
Connection: "keep-alive",
"Content-Type": "application/json;charset=utf-8",
Pragma: "no-cache",
"Referrer-Policy": "no-referrer-when-downgrade",
Server: "CLOUD ELB 1.0.0",
Vary: "Accept-Encoding",
"X-Cache-Lookup": "Cache Miss",
"x-cdn-provider": "tencent",
};
// 读取关键词
let keyword = decodeURIComponent($.request.body).match(
/keyword=(.*)/
)[1];
let keywords = $.data.read(keywordBlockKey, null, userInfo.id);
if (!keywords) {
keywords = [];
}
// 判断关键词是否存在
let keywordExists = false;
for (let i = 0; i < keywords.length; i++) {
if (keyword === keywords[i]) {
keywordExists = true;
break;
}
}
// 不存在添加,存在返回异常
if (keywordExists === false) {
keywords.push(keyword);
$.data.write(keywordBlockKey, keywords, userInfo.id);
let body = JSON.stringify({ success: true });
if ($.env.isQuanX) {
response = {
body: body,
headers: headers,
status: "HTTP/1.1 200 OK",
};
} else {
response = {
response: { body: body, headers: headers, status: 200 },
};
}
$.logger.debug(`添加本地脚本屏蔽关键词“${keyword}`);
} else {
let body = JSON.stringify({
error: {
message: "关键词已存在",
code: 100002,
},
});
if ($.env.isQuanX) {
response = {
body: body,
headers: headers,
status: "HTTP/1.1 400 Bad Request",
};
} else {
response = {
response: { body: body, headers: headers, status: 400 },
};
}
}
}
}
// 删除屏蔽关键词
else if ($.request.method === "DELETE") {
let keyword = decodeURIComponent($.request.url).match(/keyword=(.*)/)[1];
let keywords = $.data.read(keywordBlockKey, null, userInfo.id);
if (!keywords) {
keywords = [];
}
keywords = keywords.filter((e) => {
return e !== keyword;
});
$.data.write(keywordBlockKey, keywords, userInfo.id);
let headers = {
"Cache-Control":
"no-cache, no-store, must-revalidate, private, max-age=0",
Connection: "keep-alive",
"Content-Type": "application/json;charset=utf-8",
Pragma: "no-cache",
"Referrer-Policy": "no-referrer-when-downgrade",
Server: "CLOUD ELB 1.0.0",
Vary: "Accept-Encoding",
"X-Cache-Lookup": "Cache Miss",
"x-cdn-provider": "tencent",
};
let body = JSON.stringify({ success: true });
if ($.env.isQuanX) {
response = { body: body, headers: headers, status: "HTTP/1.1 200 OK" };
} else {
response = { response: { body: body, headers: headers, status: 200 } };
}
$.logger.debug(`删除本地脚本屏蔽关键词:“${keyword}`);
}
} catch (err) {
$.logger.debug(`关键词屏蔽操作出现异常:${err}`);
}
return response;
}
/**
* 处理登录用户信息
*
* @return {*}
*/
function processUserInfo() {
let response = null;
try {
let obj = JSON.parse($.response.body);
$.data.write(blackAnswersIdKey, []);
$.logger.debug(`用户登录用户信息,接口响应:${$.response.body}`);
if (
obj &&
obj["id"] &&
obj.hasOwnProperty("vip_info") &&
obj["vip_info"].hasOwnProperty("is_vip")
) {
const userInfo = {
id: obj["id"],
is_vip: obj["vip_info"]["is_vip"]
? obj["vip_info"]["is_vip"] !== undefined
: false,
};
$.logger.debug(
`当前用户id${obj["id"]}是否为VIP${obj["vip_info"]["is_vip"]}`
);
$.data.write(currentUserInfoKey, userInfo);
// 在APP显示VIP仅自己可见打开后才能使用屏蔽关键词解锁
if (
$.data.read("zhihu_settings_fake_vip") !== false &&
obj["vip_info"]["is_vip"] === false
) {
obj["vip_info"]["is_vip"] = true;
obj["vip_info"]["vip_type"] = 2;
obj["vip_info"]["vip_icon"] = {
"url": "https://picx.zhimg.com/v2-aa8a1823abfc46f14136f01d55224925.jpg?source=88ceefae",
"night_mode_url": "https://picx.zhimg.com/v2-aa8a1823abfc46f14136f01d55224925.jpg?source=88ceefae"
};
obj["vip_info"]["vip_icon_v2"] = {
"url": "https://picx.zhimg.com/v2-aa8a1823abfc46f14136f01d55224925.jpg?source=88ceefae",
"night_mode_url": "https://picx.zhimg.com/v2-aa8a1823abfc46f14136f01d55224925.jpg?source=88ceefae"
};
obj["vip_info"]["entrance"] = {
"icon": {
"url": "https://pic3.zhimg.com/v2-5b7012c8c22fd520f5677305e1e551bf.png",
"night_mode_url": "https://pic4.zhimg.com/v2-e51e3252d7a2cb016a70879defd5da0b.png"
},
"title": "盐选会员 为你严选好内容",
"expires_day": "2099-12-31",
"sub_title": null,
"button_text": "首月 9 元",
"jump_url": "zhihu://vip/purchase",
"button_jump_url": "zhihu://vip/purchase",
"sub_title_new": null,
"identity": "super_svip"
};
obj["vip_info"]["entrance_new"] = {
"left_button": {
"title": "精选会员内容",
"description": "为您严选好内容",
"jump_url": "zhihu://market/home"
},
"right_button": {
"title": "开通盐选会员",
"description": "畅享 10w+ 场优质内容等特权",
"jump_url": "zhihu://vip/purchase"
}
};
obj["vip_info"]["entrance_v2"] = {
"title": "我的超级盐选会员",
"sub_title": "畅享 5000W+ 优质内容",
"jump_url": "zhihu://market/home",
"button_text": "查看会员",
"sub_title_color": "#F8E2C4",
"sub_title_list": [
"畅享 5000W+ 优质内容"
],
"card_jump_url": "zhihu://market/home"
};
$.logger.debug("设置用户为本地盐选会员");
response = { body: JSON.stringify(obj) };
}
} else {
$.logger.warning(
`没有获取到本次登录用户信息,如未对功能造成影响,请忽略此日志。`
);
}
} catch (err) {
$.logger.error(`获取当前用户信息出现异常:${err}`);
}
return response;
}
/**
* @description: 黑名单管理
* @return {*}
*/
function manageBlackUser() {
const userInfo = getUserInfo();
let defaultBlockedUsers = {};
let customBlockedUsers = $.data.read(blockedUsersKey, "", userInfo.id);
customBlockedUsers =
typeof customBlockedUsers === "object" && !!customBlockedUsers
? customBlockedUsers
: {};
defaultAnswerBlockedUsers.forEach((element) => {
customBlockedUsers[element] = "00000000000000000000000000000000";
defaultBlockedUsers[element] = "00000000000000000000000000000000";
});
$.logger.debug(
`当前用户id${userInfo.id},脚本黑名单:${JSON.stringify(
customBlockedUsers
)}`
);
// 获取黑名单
if ($.request.method === "GET") {
try {
// 加载黑名单首页时,清空历史黑名单,仅保留脚本默认黑名单
if ($.request.url.indexOf("offset") < 0) {
customBlockedUsers = defaultBlockedUsers;
$.logger.debug("脚本黑名单已清空,请滑动至黑名单末尾保证重新获取完成。");
$.notification.post(
"开始同步黑名单数据,请滑动至黑名单末尾,直至弹出“同步成功”的通知。"
);
}
let obj = JSON.parse($.response.body);
if (!!obj["data"]) {
$.logger.debug(`本次滑动获取的黑名单信息:${JSON.stringify(obj["data"])}`);
obj["data"].forEach((element) => {
if (element["name"] !== "[已重置]") {
customBlockedUsers[element["name"]] = element["id"];
}
});
$.data.write(blockedUsersKey, customBlockedUsers, userInfo.id);
if (obj["paging"]["is_end"] === true) {
$.notification.post(
`同步黑名单数据成功!当前黑名单共${Object.keys(customBlockedUsers).length -
defaultAnswerBlockedUsers.length
}人。\n脚本内置黑名单${defaultAnswerBlockedUsers.length}人。`
);
$.logger.debug(`脚本黑名单内容:${JSON.stringify(customBlockedUsers)}`);
}
} else {
$.logger.warning(`获取黑名单失败,接口响应不合法:${$.response.body}`);
}
} catch (err) {
$.data.del(blockedUsersKey);
$.logger.error(`获取黑名单失败,异常信息:${err}`);
$.notification.post("获取黑名单失败,执行异常,已清空黑名单。");
}
}
// 写入黑名单
else if ($.request.method === "POST") {
try {
let obj = JSON.parse($.response.body);
if (obj.hasOwnProperty("name") && obj.hasOwnProperty("id")) {
$.logger.debug(
`当前需要加入黑名单的用户Id${obj["id"]},用户名:${obj["name"]}`
);
if (obj["id"]) {
customBlockedUsers[obj["name"]] = obj["id"];
$.data.write(blockedUsersKey, customBlockedUsers, userInfo.id);
$.logger.debug(
`${obj["name"]
}写入脚本黑名单成功,当前脚本黑名单数据:${JSON.stringify(
customBlockedUsers
)}`
);
$.notification.post(`已将用户“${obj["name"]}”写入脚本黑名单。`);
} else {
$.logger.error(`${obj["name"]}写入脚本黑名单失败没有获取到用户Id。`);
$.notification.post(`将用户“${obj["name"]}”写入脚本黑名单失败!`);
}
} else {
$.logger.warning(`写入黑名单失败,接口响应不合法:${$.response.body}`);
$.notification.post("写入脚本黑名单失败,接口返回不合法。");
}
} catch (err) {
$.logger.error(`写入黑名单失败,异常信息:${err}`);
$.notification.post("写入脚本黑名单失败,执行异常,请查阅日志。");
}
}
// 移出黑名单
else if ($.request.method === "DELETE") {
try {
let obj = JSON.parse($.response.body);
if (obj.success) {
let user_id = $.request.url.match(
/^https?:\/\/api\.zhihu\.com\/settings\/blocked_users\/([0-9a-zA-Z]*)/
)[1];
if (user_id) {
$.logger.debug(`当前需要移出黑名单的用户Id${user_id}`);
for (let username in customBlockedUsers) {
if (customBlockedUsers[username] === user_id) {
delete customBlockedUsers[username];
$.data.write(blockedUsersKey, customBlockedUsers, userInfo.id);
$.logger.debug(
`${username}移出脚本黑名单成功,当前脚本黑名单数据:${JSON.stringify(
customBlockedUsers
)}`
);
$.notification.post(`已将用户“${username}”移出脚本黑名单!`);
break;
}
}
} else {
$.logger.error("将用户移出脚本黑名单失败!\n建议从设置中刷新黑名单数据。");
$.notification.post(`将用户移出脚本黑名单失败没有获取到用户Id。\n建议从设置中刷新黑名单数据。`);
}
} else {
$.logger.warning(`移出黑名单失败,接口响应不合法:${$.response.body}`);
$.notification.post("移出脚本黑名单失败,接口返回不合法。");
}
} catch (err) {
$.logger.error(`移出黑名单失败,异常信息:${err}`);
$.notification.post("移出脚本黑名单失败,执行异常,请查阅日志。");
}
}
}
/**
* 黑名单增强 - 浏览黑名单用户信息时自动加入脚本黑名单
* @return {*}
*/
function autoInsertBlackList() {
let response = null;
try {
let obj = JSON.parse($.response.body);
// 删除MCN信息
delete obj["mcn_user_info"];
response = { body: JSON.stringify(obj) };
// 如已是黑名单用户,但不在脚本黑名单中,则自动加入
if (obj.name && obj.id && obj["is_blocking"] === true) {
const userInfo = getUserInfo();
let customBlockedUsers = $.data.read(blockedUsersKey, "", userInfo.id);
customBlockedUsers =
typeof customBlockedUsers === "object" && !!customBlockedUsers
? customBlockedUsers
: {};
if (!customBlockedUsers[obj.name]) {
$.logger.debug(
`当前需要加入黑名单的用户Id${obj["id"]},用户名:${obj["name"]}`
);
customBlockedUsers[obj["name"]] = obj["id"];
$.data.write(blockedUsersKey, customBlockedUsers, userInfo.id);
$.logger.debug(
`${obj["name"]
}写入脚本黑名单成功,当前脚本黑名单数据:${JSON.stringify(
customBlockedUsers
)}`
);
$.notification.post(`已自动将用户“${obj["name"]}”写入脚本黑名单。`);
}
}
} catch (err) {
$.logger.error(`去除MCN信息出现异常${err}`);
}
return response;
}
/**
* 关注列表去广告
*
* @return {*}
*/
function removeMoments() {
let response = null;
try {
let obj = JSON.parse($.response.body.replace(/(\w+"+\s?):\s?(\d{15,})/g, '$1:"$2"'));
const user_info = getUserInfo();
let customBlockedUsers = $.data.read(blockedUsersKey, "", user_info.id);
customBlockedUsers = !!customBlockedUsers ? customBlockedUsers : {};
let data;
const settings_remove_stream = $.data.read("zhihu_settings_moments_stream", false);
const settings_remove_recommend = $.data.read("zhihu_settings_moments_recommend", false);
const settings_remove_activity = $.data.read("zhihu_settings_moments_activity", false);
const settings_blocked_users = $.data.read("zhihu_settings_blocked_users", false);
data = obj.data.filter(
(item) => {
// 转发的想法是否含有黑名单用户
const isBlackUserPin = settings_blocked_users && item.target &&
item.target["origin_pin"] && item.target["origin_pin"].author &&
typeof customBlockedUsers[item.target["origin_pin"].author.name] != "undefined";
// 是否为流媒体
const isStream = settings_remove_stream && item["target_type"] === "zvideo";
// 是否为推荐关注用户
const isRecommend = settings_remove_recommend && item.type === "recommend_user_card_list";
// 是否为关注的问题有新动态
const isActivity = settings_remove_activity && item.type === "message_activity_card";
return !(isBlackUserPin || isStream || isRecommend || isActivity);
}
)
obj["data"] = data;
response = { body: JSON.stringify(obj) };
} catch (err) {
$.logger.error(`关注列表去广告出现异常:${err}`);
}
return response;
}
/**
* 推荐去广告与黑名单增强
*
* @return {*}
*/
function removeRecommend() {
let response = null;
try {
// 移除推荐列表中的想法
const settings_remove_pin = $.data.read("zhihu_settings_recommend_pin", false);
// 移除推荐列表的流媒体
const settings_recommend_stream = $.data.read("zhihu_settings_recommend_stream", false);
// 移除推荐列表的文章
const settings_remove_article = $.data.read("zhihu_settings_remove_article", false);
// 屏蔽黑名单用户
const settings_blocked_users = $.data.read("zhihu_settings_blocked_users", false);
// 屏蔽关键词内容
const settings_blocked_keywords = $.data.read("zhihu_settings_blocked_keywords", true);
// 获取用户信息
const user_info = getUserInfo();
let keywords = $.data.read(keywordBlockKey, "", user_info.id);
keywords = settings_blocked_keywords && !!keywords ? keywords : [];
let customBlockedUsers = $.data.read(blockedUsersKey, "", user_info.id);
customBlockedUsers = settings_blocked_users && !!customBlockedUsers ? customBlockedUsers : {};
const newData = (element) => {
const elementStr = JSON.stringify(element);
// 是否为广告
const isAd =
element["card_type"] === "slot_event_card" ||
element["card_type"] === "slot_video_event_card" ||
element.hasOwnProperty("ad") ||
// 训练营
(element["extra"] && element["extra"]["type"] === "Training");
// 是否为流媒体
const isStream =
isAd !== true &&
elementStr.search(
/"(type|style)+"\s?:\s?"(drama|zvideo|Video|BIG_IMAGE)+"/i
) >= 0;
const removeStream = isStream && settings_recommend_stream;
// 是否为想法
const isPin = isStream !== true && elementStr.search(/"(type|style)+"\s?:\s?"pin"/i) >= 0;
const removePin = isPin && settings_remove_pin;
// 是否为文章
const isArticle =
elementStr.search(/"(type|style)+"\s?:\s?"article"/i) >= 0;
const removeArticle = isArticle && settings_remove_article;
// 是否匹配脚本关键词过滤
let matchKeyword = false;
if (isStream !== true && settings_blocked_keywords) {
for (let i = 0; i < keywords.length; i++) {
if (elementStr.search(keywords[i]) >= 0) {
if ($.isDebug) {
let elementTitle =
element["common_card"]["feed_content"]["title"]["panel_text"];
let elementContent =
element["common_card"]["feed_content"]["content"]["panel_text"];
let actionUrl = "";
try {
actionUrl =
element["common_card"]["feed_content"]["title"]["action"]["intent_url"];
} catch { }
$.logger.debug(
`匹配关键字:\n${keywords[i]}\n标题:\n${elementTitle}\n内容:\n${elementContent}`
);
$.notification.debug(
scriptName,
`关键字:${keywords[i]}`,
`${elementTitle}\n${elementContent}`,
actionUrl
);
}
matchKeyword = true;
break;
}
}
}
// 是否为黑名单用户
let isBlockedUser;
try {
isBlockedUser =
matchKeyword !== true &&
settings_blocked_users &&
customBlockedUsers &&
element["common_card"]["feed_content"]["source_line"]["elements"][1][
"text"
]["panel_text"] in customBlockedUsers;
} catch {
isBlockedUser = false;
}
return !(
isAd ||
removePin ||
removeArticle ||
removeStream ||
matchKeyword ||
isBlockedUser
);
};
// 修复number类型精度丢失
let obj = JSON.parse(
$.response.body.replace(/(\w+"+\s?):\s?(\d{15,})/g, '$1:"$2"')
);
if (obj["data"].length > 0 && newData.length === 0) {
$.notification.post("所有推荐内容都已被过滤,建议调整脚本过滤配置。")
}
obj["data"] = obj["data"].filter(newData);
response = { body: JSON.stringify(obj) };
} catch (err) {
$.logger.error(`推荐列表去广告出现异常:${err}`);
}
return response;
}
/**
* 回答列表去广告与黑名单增强
*
* @return {*}
*/
function removeQuestions() {
let response = null;
try {
const userInfo = getUserInfo();
let customBlockedUsers = $.data.read(blockedUsersKey, "", userInfo.id);
customBlockedUsers = !!customBlockedUsers ? customBlockedUsers : {};
let obj = JSON.parse($.response.body);
const settingsBlockedUsers = $.data.read("zhihu_settings_blocked_users", false);
$.logger.debug(`当前黑名单列表: ${JSON.stringify(customBlockedUsers)}`);
// 黑名单用户的回答Id
let blackUserAnswersId = $.data.read(blackAnswersIdKey, []);
// 去除广告
delete obj["ad_info"];
// 去除回答列表中的黑名单用户
if (obj["data"]) {
let newData = [];
for (let element of obj.data) {
let blackUserName = "";
const answerId = element.target.id.toString();
try {
if ("target" in element) {
blackUserName = element["target"]["author"]["name"];
}
} catch (ex) {
$.logger.error(`获取回答列表用户名出现异常:${ex}`);
}
const isBlackUser = typeof customBlockedUsers[blackUserName] != "undefined";
const removeBlackUserAnswer = settingsBlockedUsers && isBlackUser;
// 显示仅作者自己可见的回答,允许复制
if ("target" in element) {
element["target"]["visible_only_to_author"] = false;
element["target"]["is_visible"] = true;
element["target"]["is_copyable"] = true;
}
if (!removeBlackUserAnswer) {
newData.push(element);
}
else if (removeBlackUserAnswer === true && blackUserAnswersId.includes(answerId) === false){
blackUserAnswersId.push(answerId);
$.notification.debug(`记录黑名单用户${blackUserName}的回答Id:${answerId}`);
}
}
obj.data = newData;
}
$.data.write(blackAnswersIdKey, blackUserAnswersId);
const body = JSON.stringify(obj);
$.logger.debug(`修改后的回答列表数据:${body}`);
response = { body: body };
} catch (err) {
$.logger.error(`回答列表去广告出现异常:${err}`);
}
return response;
}
/**
* 回答内容优化
*
* @return {*}
*/
function modifyAnswer() {
let response = null;
try {
let html = $.response.body;
let insertText = "";
// 付费内容提醒
if ((html.indexOf("查看完整内容") >= 0 || html.indexOf("查看全部章节") >= 0) && html.indexOf("paid") >= 0) {
insertText = '<a style="height: 42px;padding: 0 12px;border-radius: 6px;background-color: rgb(247 104 104 / 8%);display: block;text-decoration: none;" href="#"><div style="color: #f36;display: flex;-webkit-box-align: center;align-items: center;height: 100%;"><svg style="width: 1.2em; height: 1em;vertical-align: middle;fill: currentColor;overflow: hidden;" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="1957"><path d="M821.333333 138.666667c64.8 0 117.333333 52.533333 117.333334 117.333333v149.333333a32 32 0 0 1-32 32 74.666667 74.666667 0 0 0 0 149.333334 32 32 0 0 1 32 32v149.333333c0 64.8-52.533333 117.333333-117.333334 117.333333H202.666667c-64.8 0-117.333333-52.533333-117.333334-117.333333V618.666667a32 32 0 0 1 32-32 74.666667 74.666667 0 0 0 0-149.333334 32 32 0 0 1-32-32V256c0-64.8 52.533333-117.333333 117.333334-117.333333h618.666666zM428.576 329.994667a32 32 0 0 0-43.733333-2.581334l-1.514667 1.344a32 32 0 0 0-2.581333 43.733334L452.565333 458.666667H405.333333l-1.877333 0.053333A32 32 0 0 0 373.333333 490.666667l0.053334 1.877333A32 32 0 0 0 405.333333 522.666667h80v42.666666H405.333333l-1.877333 0.053334A32 32 0 0 0 373.333333 597.333333l0.053334 1.877334A32 32 0 0 0 405.333333 629.333333h80v42.666667l0.053334 1.877333A32 32 0 0 0 517.333333 704l1.877334-0.053333A32 32 0 0 0 549.333333 672v-42.666667H618.666667l1.877333-0.053333A32 32 0 0 0 650.666667 597.333333l-0.053334-1.877333A32 32 0 0 0 618.666667 565.333333h-69.333334v-42.666666H618.666667l1.877333-0.053334A32 32 0 0 0 650.666667 490.666667l-0.053334-1.877334A32 32 0 0 0 618.666667 458.666667h-47.253334l71.84-86.186667 1.248-1.589333a32 32 0 0 0-50.421333-39.381334L512 430.016l-82.08-98.506667z" p-id="1958"></path></svg><div style="flex: 1 1;white-space: nowrap;text-overflow: ellipsis;padding-left:4px"><span style="font-family: -apple-system,BlinkMacSystemFont,Helvetica Neue,PingFang SC,Microsoft YaHei,Source Han Sans SC,Noto Sans CJK SC,WenQuanYi Micro Hei,sans-serif;-webkit-tap-highlight-color: rgba(26,26,26,0);font-size: 14px;line-height: 20px;color: #f36;white-space: nowrap;font-weight: 600;">哲也同学 · 本文为付费内容</span></div><div><span style="font-family: -apple-system,BlinkMacSystemFont,Helvetica Neue,PingFang SC,Microsoft YaHei,Source Han Sans SC,Noto Sans CJK SC,WenQuanYi Micro Hei,sans-serif;-webkit-tap-highlight-color: rgba(26,26,26,0);font-size: 14px;color: #f36;line-height: normal;display: flex;-webkit-box-align: center;align-items: center;"><svg style="font-family: -apple-system,BlinkMacSystemFont,Helvetica Neue,PingFang SC,Microsoft YaHei,Source Han Sans SC,Noto Sans CJK SC,WenQuanYi Micro Hei,sans-serif;-webkit-tap-highlight-color: rgba(26,26,26,0);font-size: 14px;color: #f36;line-height: normal;fill: currentcolor;width: 24px;height: 24px;margin: -2px;" fill="currentColor" viewBox="0 0 24 24" width="24" height="24"><path d="M9.218 16.78a.737.737 0 0 0 1.052 0l4.512-4.249a.758.758 0 0 0 0-1.063L10.27 7.22a.737.737 0 0 0-1.052 0 .759.759 0 0 0-.001 1.063L13 12l-3.782 3.716a.758.758 0 0 0 0 1.063z" fill-rule="evenodd"></path></svg></span></div></div></a>';
}
// 营销推广提醒
else if (html.indexOf("ad-link-card") >= 0 || html.indexOf("xg.zhihu.com") >= 0 || html.indexOf("营销平台") >= 0) {
insertText = '<a style="height: 42px;padding: 0 12px;border-radius: 6px;background-color: rgb(8 188 212 / 8%);display: block;text-decoration: none;" href="#"><div style="color: #00bcd4;display: flex;-webkit-box-align: center;align-items: center;height: 100%;"><svg style="width: 1.2em; height: 1em;vertical-align: middle;fill: currentColor;overflow: hidden;" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="1957"><path d="M128 765.952q0 26.624 18.432 45.056t45.056 18.432l191.488 0 0 128-254.976 0q-26.624 0-49.664-10.24t-40.448-27.648-27.648-40.448-10.24-49.664l0-637.952q0-25.6 10.24-49.152t27.648-40.448 40.448-27.136 49.664-10.24l701.44 0q26.624 0 49.664 10.24t40.448 27.136 27.648 40.448 10.24 49.152l0 251.904-128 1.024 0-61.44q0-26.624-18.432-45.056t-45.056-18.432l-574.464 0q-26.624 0-45.056 18.432t-18.432 45.056l0 382.976zM990.208 705.536q21.504 18.432 22.016 34.304t-20.992 33.28q-21.504 18.432-51.2 41.472t-60.928 48.128-61.952 49.152-55.296 43.52q-26.624 20.48-43.52 15.36t-16.896-31.744q1.024-16.384 0-40.448t-1.024-41.472q0-19.456-10.752-24.064t-31.232-4.608q-21.504 0-39.936-0.512t-35.84-0.512-36.352-0.512-41.472-0.512q-9.216 0-19.968-2.048t-19.456-7.168-14.336-15.36-5.632-27.648l0-80.896q0-31.744 15.36-42.496t48.128-10.752q30.72 1.024 61.44 1.024t71.68 1.024q29.696 0 46.08-5.12t16.384-25.6q-1.024-14.336 0.512-35.328t1.536-37.376q0-26.624 14.336-33.28t36.864 10.752q22.528 18.432 52.736 43.008t61.952 50.688 62.976 51.2 54.784 44.544z" p-id="5859"></path></svg><div style="flex: 1 1;white-space: nowrap;text-overflow: ellipsis;padding-left:4px"><span style="font-family: -apple-system,BlinkMacSystemFont,Helvetica Neue,PingFang SC,Microsoft YaHei,Source Han Sans SC,Noto Sans CJK SC,WenQuanYi Micro Hei,sans-serif;-webkit-tap-highlight-color: rgba(26,26,26,0);font-size: 14px;line-height: 20px;color: #00bcd4;white-space: nowrap;font-weight: 600;">哲也同学 · 本文含有营销推广</span></div><div><span style="font-family: -apple-system,BlinkMacSystemFont,Helvetica Neue,PingFang SC,Microsoft YaHei,Source Han Sans SC,Noto Sans CJK SC,WenQuanYi Micro Hei,sans-serif;-webkit-tap-highlight-color: rgba(26,26,26,0);font-size: 14px;color: #f36;line-height: normal;display: flex;-webkit-box-align: center;align-items: center;"><svg style="font-family: -apple-system,BlinkMacSystemFont,Helvetica Neue,PingFang SC,Microsoft YaHei,Source Han Sans SC,Noto Sans CJK SC,WenQuanYi Micro Hei,sans-serif;-webkit-tap-highlight-color: rgba(26,26,26,0);font-size: 14px;color: #00BCD4;line-height: normal;fill: currentcolor;width: 24px;height: 24px;margin: -2px;" fill="currentColor" viewBox="0 0 24 24" width="24" height="24"><path d="M9.218 16.78a.737.737 0 0 0 1.052 0l4.512-4.249a.758.758 0 0 0 0-1.063L10.27 7.22a.737.737 0 0 0-1.052 0 .759.759 0 0 0-.001 1.063L13 12l-3.782 3.716a.758.758 0 0 0 0 1.063z" fill-rule="evenodd"></path></svg></span></div></div></a>';
}
// 购物推广提醒
else if (html.indexOf("mcn-link-card") >= 0) {
insertText = '<a style="height: 42px;padding: 0 12px;border-radius: 6px;background-color: rgb(8 188 212 / 8%);display: block;text-decoration: none;" href="#"><div style="color: #00bcd4;display: flex;-webkit-box-align: center;align-items: center;height: 100%;"><svg style="width: 1.2em; height: 1em;vertical-align: middle;fill: currentColor;overflow: hidden;" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="1957"><path d="M346.112 806.912q19.456 0 36.864 7.168t30.208 19.968 20.48 30.208 7.68 36.864-7.68 36.864-20.48 30.208-30.208 20.48-36.864 7.68q-20.48 0-37.888-7.68t-30.208-20.48-20.48-30.208-7.68-36.864 7.68-36.864 20.48-30.208 30.208-19.968 37.888-7.168zM772.096 808.96q19.456 0 37.376 7.168t30.72 19.968 20.48 30.208 7.68 36.864-7.68 36.864-20.48 30.208-30.72 20.48-37.376 7.68-36.864-7.68-30.208-20.48-20.48-30.208-7.68-36.864 7.68-36.864 20.48-30.208 30.208-19.968 36.864-7.168zM944.128 227.328q28.672 0 44.544 7.68t22.528 18.944 6.144 24.064-3.584 22.016-13.312 37.888-22.016 62.976-23.552 68.096-18.944 53.248q-13.312 40.96-33.28 56.832t-49.664 15.872l-35.84 0-65.536 0-86.016 0-96.256 0-253.952 0 14.336 92.16 517.12 0q49.152 0 49.152 41.984 0 20.48-9.728 35.328t-38.4 14.848l-49.152 0-94.208 0-118.784 0-119.808 0-99.328 0-55.296 0q-20.48 0-34.304-9.216t-23.04-24.064-14.848-32.256-8.704-32.768q-1.024-6.144-5.632-29.696t-11.264-58.88-14.848-78.848-16.384-87.552q-19.456-103.424-44.032-230.4l-76.8 0q-15.36 0-25.6-7.68t-16.896-18.432-9.216-23.04-2.56-22.528q0-20.48 13.824-33.792t37.376-13.312l21.504 0 21.504 0 25.6 0 34.816 0q20.48 0 32.768 6.144t19.456 15.36 10.24 19.456 5.12 17.408q2.048 8.192 4.096 23.04t4.096 30.208q3.072 18.432 6.144 38.912l700.416 0zM867.328 194.56l-374.784 0 135.168-135.168q23.552-23.552 51.712-24.064t51.712 23.04z"></path></svg><div style="flex: 1 1;white-space: nowrap;text-overflow: ellipsis;padding-left:4px"><span style="font-family: -apple-system,BlinkMacSystemFont,Helvetica Neue,PingFang SC,Microsoft YaHei,Source Han Sans SC,Noto Sans CJK SC,WenQuanYi Micro Hei,sans-serif;-webkit-tap-highlight-color: rgba(26,26,26,0);font-size: 14px;line-height: 20px;color: #00bcd4;white-space: nowrap;font-weight: 600;">哲也同学 · 本文含有购物推广</span></div><div><span style="font-family: -apple-system,BlinkMacSystemFont,Helvetica Neue,PingFang SC,Microsoft YaHei,Source Han Sans SC,Noto Sans CJK SC,WenQuanYi Micro Hei,sans-serif;-webkit-tap-highlight-color: rgba(26,26,26,0);font-size: 14px;color: #f36;line-height: normal;display: flex;-webkit-box-align: center;align-items: center;"><svg style="font-family: -apple-system,BlinkMacSystemFont,Helvetica Neue,PingFang SC,Microsoft YaHei,Source Han Sans SC,Noto Sans CJK SC,WenQuanYi Micro Hei,sans-serif;-webkit-tap-highlight-color: rgba(26,26,26,0);font-size: 14px;color: #00BCD4;line-height: normal;fill: currentcolor;width: 24px;height: 24px;margin: -2px;" fill="currentColor" viewBox="0 0 24 24" width="24" height="24"><path d="M9.218 16.78a.737.737 0 0 0 1.052 0l4.512-4.249a.758.758 0 0 0 0-1.063L10.27 7.22a.737.737 0 0 0-1.052 0 .759.759 0 0 0-.001 1.063L13 12l-3.782 3.716a.758.758 0 0 0 0 1.063z" fill-rule="evenodd"></path></svg></span></div></div></a>';
}
// 彩蛋
else if (Math.floor(Math.random() * 200) === 7) {
insertText = '<a style="height: 42px;padding: 0 12px;border-radius: 6px;background-color: rgb(74 162 56 / 8%);display: block;text-decoration: none;" href="#"><div style="color: #619201;display: flex;-webkit-box-align: center;align-items: center;height: 100%;"><svg class="icon" style="width: 1.2em; height: 1em;vertical-align: middle;fill: currentColor;overflow: hidden;" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="1465"><path d="M512 85.333333c71.477333 0 159.68 57.546667 229.258667 147.018667C817.845333 330.826667 864 455.978667 864 586.666667c0 211.808-148.501333 352-352 352S160 798.474667 160 586.666667c0-130.688 46.144-255.84 122.741333-354.314667C352.32 142.88 440.522667 85.333333 512 85.333333z m0 64c-48.213333 0-120.096 46.912-178.741333 122.314667C265.109333 359.253333 224 470.762667 224 586.666667c0 175.616 119.050667 288 288 288s288-112.384 288-288c0-115.904-41.109333-227.402667-109.258667-315.018667C632.096 196.234667 560.213333 149.333333 512 149.333333z m-74.666667 522.666667a53.333333 53.333333 0 1 1 0 106.666667 53.333333 53.333333 0 0 1 0-106.666667z m-96-128a42.666667 42.666667 0 1 1 0 85.333333 42.666667 42.666667 0 0 1 0-85.333333z" p-id="1466"></path></svg><div style="flex: 1 1;white-space: nowrap;text-overflow: ellipsis;padding-left:4px"><span style="font-family: -apple-system,BlinkMacSystemFont,Helvetica Neue,PingFang SC,Microsoft YaHei,Source Han Sans SC,Noto Sans CJK SC,WenQuanYi Micro Hei,sans-serif;-webkit-tap-highlight-color: rgba(26,26,26,0);font-size: 14px;line-height: 20px;color: #619201;white-space: nowrap;font-weight: 600;">哲也同学 · 本文为免费内容</span></div><div><span style="font-family: -apple-system,BlinkMacSystemFont,Helvetica Neue,PingFang SC,Microsoft YaHei,Source Han Sans SC,Noto Sans CJK SC,WenQuanYi Micro Hei,sans-serif;-webkit-tap-highlight-color: rgba(26,26,26,0);font-size: 14px;color: #619201;line-height: normal;display: flex;-webkit-box-align: center;align-items: center;"><svg style="font-family: -apple-system,BlinkMacSystemFont,Helvetica Neue,PingFang SC,Microsoft YaHei,Source Han Sans SC,Noto Sans CJK SC,WenQuanYi Micro Hei,sans-serif;-webkit-tap-highlight-color: rgba(26,26,26,0);font-size: 14px;color: #619201;line-height: normal;fill: currentcolor;width: 24px;height: 24px;margin: -2px;" fill="currentColor" viewBox="0 0 24 24" width="24" height="24"><path d="M9.218 16.78a.737.737 0 0 0 1.052 0l4.512-4.249a.758.758 0 0 0 0-1.063L10.27 7.22a.737.737 0 0 0-1.052 0 .759.759 0 0 0-.001 1.063L13 12l-3.782 3.716a.758.758 0 0 0 0 1.063z" fill-rule="evenodd"></path></svg></span></div></div></a>';
}
if (insertText !== "") {
const matchStr = html.match(/(richText[^<]*>)(.)/)[1];
const start = html.lastIndexOf(matchStr) + matchStr.length;
response = { body: html.slice(0, start) + insertText + html.slice(start) };
}
} catch (err) {
$.logger.error(`付费内容提醒出现异常:${err}`);
}
return response;
}
/**
* 评论去广告及黑名单增强
*
* @return {*}
*/
function removeComment() {
let response = null;
try {
if (!!$.response.body) {
let obj = JSON.parse($.response.body);
obj["ad_info"] = {};
// 屏蔽黑名单用户
if ($.data.read("zhihu_settings_blocked_users", false) === true) {
let user_info = getUserInfo();
let customBlockedUsers = $.data.read(blockedUsersKey, "", user_info.id);
customBlockedUsers = !!customBlockedUsers ? customBlockedUsers : {};
let newComments = [];
let blockCommentIdObj = {};
if (typeof obj.root != "undefined") {
// 屏蔽黑名单用户的评论
const rootUserName = obj.root.author.name;
const isBlackRootUser = typeof customBlockedUsers[rootUserName] != "undefined";
if (isBlackRootUser === true) {
obj.root.is_delete = true;
obj.root.can_reply = false;
obj.root.can_like = false;
obj.root.author.name = "黑名单用户";
obj.root.author.avatar_url = "https://picx.zhimg.com/v2-abed1a8c04700ba7d72b45195223e0ff_xll.jpg";
}
}
if (typeof obj.data != "undefined") {
obj.data.forEach((comment) => {
// 屏蔽黑名单用户的评论
// 评论人昵称
const commentUserName = comment.author.name;
// 回复哪个人的评论(仅适用于独立子评论页面请求)
let replyUserName = "";
if (comment["reply_to_author"] && comment["reply_to_author"].name) {
replyUserName = comment["reply_to_author"].name;
}
const isSubComment = replyUserName !== "";
const isBlackCommentUser = typeof customBlockedUsers[commentUserName] != "undefined";
const isBlackReplyUser = typeof customBlockedUsers[replyUserName] != "undefined";
if (isBlackCommentUser === true || isBlackReplyUser === true) {
if (isBlackCommentUser && !isSubComment && $.request.url.indexOf("root_comment") > 0) {
$.notification.debug(`屏蔽黑名单用户“${commentUserName}”的主评论。`);
} else if (!isBlackCommentUser && isSubComment && !isBlackReplyUser && $.request.url.indexOf("child_comment") > 0) {
$.notification.debug(`屏蔽黑名单用户“${commentUserName}”的子评论。`);
} else if (isBlackCommentUser && !isBlackReplyUser && $.request.url.indexOf("child_comment") > 0) {
$.notification.debug(`屏蔽黑名单用户“${commentUserName}”回复“${replyUserName}”的子评论。`);
} else if (isBlackCommentUser && isBlackReplyUser && $.request.url.indexOf("child_comment") > 0) {
$.notification.debug(`屏蔽黑名单用户“${commentUserName}”回复黑名单用户“${replyUserName}”的子评论。`);
}
blockCommentIdObj[comment.id] = commentUserName;
if (isBlackCommentUser) {
comment.is_delete = true;
comment.can_reply = false;
comment.can_like = false;
comment.author.exposed_medal = {};
comment.author.name = "[黑名单用户]";
comment.author.avatar_url = "https://picx.zhimg.com/v2-abed1a8c04700ba7d72b45195223e0ff_xll.jpg";
}
if (isBlackReplyUser) {
comment["reply_to_author"].name = "[黑名单用户]";
comment["reply_to_author"].exposed_medal = {};
comment["reply_to_author"].avatar_url = "https://picx.zhimg.com/v2-abed1a8c04700ba7d72b45195223e0ff_xll.jpg";
}
}
if (comment.child_comments) {
let newChildComments = [];
comment.child_comments.forEach((childComment) => {
// 屏蔽黑名单用户的评论
const childCommentUserName = childComment.author.name;
const childCommentReplyUserName = typeof childComment["reply_to_author"] != "undefined" ? childComment["reply_to_author"].name: "";
const isChildBlackCommentUser = typeof customBlockedUsers[childCommentUserName] != "undefined";
const isChildBlackReplyUser = typeof customBlockedUsers[childCommentReplyUserName] != "undefined";
if (isChildBlackCommentUser || isChildBlackReplyUser) {
if (isChildBlackCommentUser === true) {
$.notification.debug(`屏蔽黑名单用户“${childCommentUserName}”的子评论。`);
blockCommentIdObj[childComment.id] = childCommentUserName;
childComment.is_delete = true;
childComment.can_reply = false;
childComment.can_like = false;
childComment.author.name = "[黑名单用户]";
childComment.author.exposed_medal = {};
childComment.author.avatar_url = "https://picx.zhimg.com/v2-abed1a8c04700ba7d72b45195223e0ff_xll.jpg";
}
if (isChildBlackReplyUser === true) {
$.logger.debug(`修改前的子评论数据:\n${JSON.stringify(childComment)}`);
childComment["reply_to_author"].name = "[黑名单用户]";
childComment["reply_to_author"].exposed_medal = {};
childComment["reply_to_author"].avatar_url = "https://picx.zhimg.com/v2-abed1a8c04700ba7d72b45195223e0ff_xll.jpg";
$.notification.debug(`隐藏“${childCommentUserName}”回复黑名单用户“${childCommentReplyUserName}”的名称与头像。`);
$.logger.debug(`修改后的子评论数据:\n${JSON.stringify(childComment)}`);
}
}
newChildComments.push(childComment);
})
comment.child_comments = newChildComments;
}
newComments.push(comment);
});
}
obj.data = newComments;
}
const body = JSON.stringify(obj);
$.logger.debug(`过滤后的评论数据:\n${body}`);
response = { body: body };
}
} catch (err) {
$.logger.error(`去除评论广告出现异常:${err}`);
}
return response;
}
/**
* 移除文章页面的广告
* @return {*}
*/
function removeArticleAd() {
let response = null;
try {
if (!!$.response.body) {
let obj = JSON.parse($.response.body);
obj["ad_info"] = {};
const body = JSON.stringify(obj);
$.logger.debug(`过滤后的文章数据:\n${body}`);
response = { body: body };
}
} catch (err) {
$.logger.error(`去除文章广告出现异常:${err}`);
}
return response;
}
/**
* 屏蔽官方营销消息
*
* @return {*}
*/
function removeMarketingMsg() {
let response = null;
try {
let obj = JSON.parse($.response.body);
let newItems = [];
for (let item of obj["data"]) {
if (item["detail_title"] === "官方帐号消息") {
let unread_count = item["unread_count"];
if (unread_count > 0) {
item["content"]["text"] = "未读消息" + unread_count + "条";
} else {
item["content"]["text"] = "全部消息已读";
}
item["is_read"] = true;
item["unread_count"] = 0;
newItems.push(item);
} else if (item["detail_title"] !== "活动助手") {
newItems.push(item);
}
}
obj["data"] = newItems;
response = { body: JSON.stringify(obj) };
} catch (err) {
$.logger.error(`屏蔽官方营销消息出现异常:${err}`);
}
return response;
}
/**
* 热榜去广告
*
* @return {*}
*/
function removeHotListAds() {
let response = null;
try {
if (!!$.response.body) {
let obj = JSON.parse($.response.body);
if ("data" in obj) {
obj["data"] = obj["data"].filter((e) => {
return (
e["type"] === "hot_list_feed" || e["type"] === "hot_list_feed_video"
);
});
}
response = { body: JSON.stringify(obj) };
}
} catch (err) {
$.logger.error(`去除热榜广告出现异常:${err}`);
}
return response;
}
/**
* 去除预置关键字广告
*
* @return {*}
*/
function removeKeywordAds() {
let response = null;
try {
if (!!$.response.body) {
$.logger.debug(`预置关键字返回:${$.response.body}`);
let obj = JSON.parse($.response.body);
if (obj.hasOwnProperty("preset_words") && obj["preset_words"]["words"]) {
obj["preset_words"]["words"] = obj["preset_words"]["words"].filter((element) => {
return element["type"] !== "ad";
});
response = { body: JSON.stringify(obj) };
}
}
} catch (err) {
$.logger.error(`去除预置关键字广告出现异常:${err}`);
}
return response;
}
/**
* 移除回答翻页时出现的黑名单用户的回答
* 小概率会移除失败
* @return {*}
*/
function removeNextBlackUserAnswer(){
let response = null;
try {
if (!!$.response.body) {
let obj = JSON.parse($.response.body);
const blackUserAnswersId = $.data.read(blackAnswersIdKey, []);
if (blackUserAnswersId.length > 0){
let newData = [];
obj.data.forEach(element => {
const tag = blackUserAnswersId.includes(element.id.toString());
if (tag === false){
// 去除可能的广告
element.ad_info = {"data": ""};
newData.push(element);
}
else {
$.notification.debug(`屏蔽翻页过程中出现的黑名单用户回答Id:${element.id}`);
}
});
// 重新为答案排序
for(let i=0; i < newData.length; i++){
if (newData[i]["extra"] && newData[i]["extra"]["question_index"]){
newData[i]["extra"]["question_index"] = i+1;
}
if (newData[i]["strategy_info"]){
newData[i]["strategy_info"]["global_index"] = i+1;
newData[i]["strategy_info"]["strategy_index"] = i+1;
}
}
obj.data = newData;
}
response = { body: JSON.stringify(obj) };
}
} catch (err) {
$.logger.error(`屏蔽下翻黑名单用户的回答出现异常:${err}`);
}
return response;
}
/**
* 修改盐值
*
* @return {*}
*/
function changeUserCredit(){
$.notification.debug("开始修改用户盐值");
let response = null;
try {
if (!!$.response.body) {
// 自定义盐值
const score = parseInt($.data.read(userCreditScore, 780));
$.logger.debug(`准备修改用户盐值为${score}`);
let obj = JSON.parse($.response.body);
if (obj["credit_basis"].total_score < score){
obj["credit_basis"].total_score = score;
$.logger.debug(`已修改用户盐值为:${score}`);
}
response = { body: JSON.stringify(obj) };
}
} catch (err) {
$.logger.error(`修改用户盐值出现异常:${err}`);
}
return response;
}
(() => {
let response = null;
if ($.isResponse) {
switch (true) {
// 获取用户信息 - 隔离用户数据,开启本地盐选会员等
case /^https:\/\/api\.zhihu\.com\/people\/self$/.test($.request.url):
response = processUserInfo();
break;
// 优化软件配置 - 优化下发的配置文件来实现某些效果
case $.data.read("zhihu_settings_app_conf", false) === true &&
/^https?:\/\/appcloud2\.zhihu\.com\/v\d+\/config/.test($.request.url):
response = modifyAppConfig();
break;
case $.data.read("zhihu_settings_app_conf", false) === true &&
/^https?:\/\/m-cloud\.zhihu\.com\/api\/cloud\/config\/all\?/.test($.request.url):
response = modifyMCloudConfig();
break;
// 修改用户盐值 - 仅当自定义盐值大于真实盐值时生效
case /^https?:\/\/api\.zhihu\.com\/user-credit\/basis/.test($.request.url):
$.notification.debug("准备修改用户盐值");
response = changeUserCredit();
break;
// 推荐页 - 移除黑名单用户发布的文章、去除广告,及自定义一些屏蔽项目
case /^https:\/\/api\.zhihu\.com\/topstory\/recommend\?/.test($.request.url):
response = removeRecommend();
break;
// 问题的回答列表 - 移除黑名单用户的回答、去除广告
case /^https?:\/\/api\.zhihu\.com\/(v4\/)?questions\/\d+/.test($.request.url):
response = removeQuestions();
break;
// 消息页 - 折叠官方消息、屏蔽营销消息
case $.data.read("zhihu_settings_sys_msg", true) !== false &&
/^https?:\/\/api\.zhihu\.com\/notifications\/v3\/message/.test($.request.url):
response = removeMarketingMsg();
break;
// 评论页及子页面 - 去除黑名单用户发表的评论
case /^https?:\/\/api\.zhihu\.com\/comment_v5\/(answers|pins|comments?|articles)\/\d+\/(root|child)_comment/.test($.request.url):
response = removeComment();
break;
// 文章页 - 去除底部卡片广告
case /^https?:\/\/www\.zhihu\.com\/api\/v\d\/articles\/\d+\/recommendation\?/.test($.request.url):
response = removeArticleAd();
break;
// 回答页底部评论摘要 - 移除黑名单用户发表的评论
case /^https?:\/\/www\.zhihu\.com\/api\/v4\/comment_v5\/answers\/\d+\/abstract_comment\?/.test($.request.url):
response = removeComment();
break;
// 回答内容优化 - 付费、营销、推广内容文首提醒
case $.data.read("zhihu_settings_answer_tip", true) === true &&
/^https?:\/\/www\.zhihu\.com\/appview\/v2\/answer\/.*(entry=(?!(preload-topstory|preload-search|preload-subscription)))?/.test($.request.url):
response = modifyAnswer();
break;
// 回答页 - 屏蔽下翻出现的黑名单用户
case $.data.read("zhihu_settings_blocked_users", false) !== false &&
/^https?:\/\/api\.zhihu\.com\/next\?/.test($.request.url):
response = removeNextBlackUserAnswer();
break;
// 黑名单增强 - 浏览黑名单用户信息时自动加入脚本黑名单
case $.data.read("zhihu_settings_blocked_users", true) === true &&
/^https?:\/\/api\.zhihu\.com\/people\/((?!self).)*$/.test($.request.url):
response = autoInsertBlackList();
break;
// 关注页 - 去广告
case /^https?:\/\/api\.zhihu\.com\/moments_v3\?/.test($.request.url):
response = removeMoments();
break;
// 热榜页 - 去广告
case $.data.read("zhihu_settings_hot_list", true) === true &&
/^https?:\/\/api\.zhihu\.com\/topstory\/hot-lists(\?|\/)/.test($.request.url):
response = removeHotListAds();
break;
// 搜索页 - 去除预置广告
case $.data.read("zhihu_settings_preset_words", true) === true &&
/^https?:\/\/api\.zhihu\.com\/search\/preset_words\?/.test($.request.url):
response = removeKeywordAds();
break;
// 黑名单页 - 同步黑名单数据
case $.data.read("zhihu_settings_blocked_users", false) !== false &&
/^https?:\/\/api\.zhihu\.com\/settings\/blocked_users/.test($.request.url):
manageBlackUser();
break;
default:
$.logger.debug("没有匹配到任何请求,请确认配置正确。");
break;
}
} else if ($.isRequest) {
// 屏蔽关键词解锁
if (
$.data.read("zhihu_settings_blocked_keywords", false) !== false &&
/^https?:\/\/api\.zhihu\.com\/feed-root\/block/.test(
$.request.url
) === true
) {
response = unlockBlockedKeywords(response);
}
} else {
$.data.del(currentUserInfoKey);
$.data.del(blockedUsersKey);
$.data.del(keywordBlockKey);
$.notification.post("哲也同学数据清理完毕");
}
if (response) {
$.done(response);
} else {
$.done();
}
})();
// prettier-ignore
/**
*
* $$\ $$\ $$\ $$$$$\ $$$$$$\ $$$$$$\
* $$$\ $$$ | \__| \__$$ |$$ __$$\ $$ ___$$\
* $$$$\ $$$$ | $$$$$$\ $$$$$$\ $$\ $$$$$$$\ $$ |$$ / \__| \_/ $$ |
* $$\$$\$$ $$ | \____$$\ $$ __$$\ $$ |$$ _____| $$ |\$$$$$$\ $$$$$ /
* $$ \$$$ $$ | $$$$$$$ |$$ / $$ |$$ |$$ / $$\ $$ | \____$$\ \___$$\
* $$ |\$ /$$ |$$ __$$ |$$ | $$ |$$ |$$ | $$ | $$ |$$\ $$ | $$\ $$ |
* $$ | \_/ $$ |\$$$$$$$ |\$$$$$$$ |$$ |\$$$$$$$\\$$$$$$ |\$$$$$$ | \$$$$$$ |
* \__| \__| \_______| \____$$ |\__| \_______|\______/ \______/ \______/
* $$\ $$ |
* \$$$$$$ |
* \______/
*
*/
// prettier-ignore
function MagicJS(scriptName="MagicJS",logLevel="INFO"){const MagicEnvironment=()=>{const isLoon=typeof $loon!=="undefined";const isQuanX=typeof $task!=="undefined";const isNode=typeof module!=="undefined";const isSurge=typeof $httpClient!=="undefined"&&!isLoon;const isStorm=typeof $storm!=="undefined";const isStash=typeof $environment!=="undefined"&&typeof $environment["stash-build"]!=="undefined";const isSurgeLike=isSurge||isLoon||isStorm||isStash;const isScriptable=typeof importModule!=="undefined";return{isLoon:isLoon,isQuanX:isQuanX,isNode:isNode,isSurge:isSurge,isStorm:isStorm,isStash:isStash,isSurgeLike:isSurgeLike,isScriptable:isScriptable,get name(){if(isLoon){return"Loon"}else if(isQuanX){return"QuantumultX"}else if(isNode){return"NodeJS"}else if(isSurge){return"Surge"}else if(isScriptable){return"Scriptable"}else{return"unknown"}},get build(){if(isSurge){return $environment["surge-build"]}else if(isStash){return $environment["stash-build"]}else if(isStorm){return $storm.buildVersion}},get language(){if(isSurge||isStash){return $environment["language"]}},get version(){if(isSurge){return $environment["surge-version"]}else if(isStash){return $environment["stash-version"]}else if(isStorm){return $storm.appVersion}else if(isNode){return process.version}},get system(){if(isSurge){return $environment["system"]}else if(isNode){return process.platform}},get systemVersion(){if(isStorm){return $storm.systemVersion}},get deviceName(){if(isStorm){return $storm.deviceName}}}};const MagicLogger=(scriptName,logLevel="INFO")=>{let _level=logLevel;const logLevels={SNIFFER:6,DEBUG:5,INFO:4,NOTIFY:3,WARNING:2,ERROR:1,CRITICAL:0,NONE:-1};const logEmoji={SNIFFER:"",DEBUG:"",INFO:"",NOTIFY:"",WARNING:"❗ ",ERROR:"❌ ",CRITICAL:"❌ ",NONE:""};const _log=(msg,level="INFO")=>{if(!(logLevels[_level]<logLevels[level.toUpperCase()]))console.log(`[${level}] [${scriptName}]\n${logEmoji[level.toUpperCase()]}${msg}\n`)};const setLevel=logLevel=>{_level=logLevel};return{getLevel:()=>{return _level},setLevel:setLevel,sniffer:msg=>{_log(msg,"SNIFFER")},debug:msg=>{_log(msg,"DEBUG")},info:msg=>{_log(msg,"INFO")},notify:msg=>{_log(msg,"NOTIFY")},warning:msg=>{_log(msg,"WARNING")},error:msg=>{_log(msg,"ERROR")},retry:msg=>{_log(msg,"RETRY")}}};return new class{constructor(scriptName,logLevel){this._startTime=Date.now();this.version="3.0.0";this.scriptName=scriptName;this.env=MagicEnvironment();this.logger=MagicLogger(scriptName,logLevel);this.http=typeof MagicHttp==="function"?MagicHttp(this.env,this.logger):undefined;this.data=typeof MagicData==="function"?MagicData(this.env,this.logger):undefined;this.notification=typeof MagicNotification==="function"?MagicNotification(this.scriptName,this.env,this.logger,this.http):undefined;this.utils=typeof MagicUtils==="function"?MagicUtils(this.env,this.logger):undefined;this.qinglong=typeof MagicQingLong==="function"?MagicQingLong(this.env,this.data,this.logger):undefined;if(typeof this.data!=="undefined"){let magicLoglevel=this.data.read("magic_loglevel");const barkUrl=this.data.read("magic_bark_url");if(magicLoglevel){this.logger.setLevel(magicLoglevel.toUpperCase())}if(barkUrl){this.notification.setBark(barkUrl)}}}get isRequest(){return typeof $request!=="undefined"&&typeof $response==="undefined"}get isResponse(){return typeof $response!=="undefined"}get isDebug(){return this.logger.level==="DEBUG"}get request(){return typeof $request!=="undefined"?$request:undefined}get response(){if(typeof $response!=="undefined"){if($response.hasOwnProperty("status"))$response["statusCode"]=$response["status"];if($response.hasOwnProperty("statusCode"))$response["status"]=$response["statusCode"];return $response}else{return undefined}}done=(value={})=>{this._endTime=Date.now();let span=(this._endTime-this._startTime)/1e3;this.logger.info(`SCRIPT COMPLETED: ${span} S.`);if(typeof $done!=="undefined"){$done(value)}}}(scriptName,logLevel)}
// prettier-ignore
function MagicNotification(scriptName,env,logger,http){let _barkUrl=null;let _barkKey=null;const setBark=url=>{try{let _url=url.replace(/\/+$/g,"");_barkUrl=`${/^https?:\/\/([^/]*)/.exec(_url)[0]}/push`;_barkKey=/\/([^\/]+)\/?$/.exec(_url)[1]}catch(ex){logger.error(`Bark url error: ${ex}.`)}};function post(title=scriptName,subTitle="",body="",opts=""){const _adaptOpts=_opts=>{try{let newOpts={};if(typeof _opts==="string"){if(env.isLoon)newOpts={openUrl:_opts};else if(env.isQuanX)newOpts={"open-url":_opts};else if(env.isSurge)newOpts={url:_opts}}else if(typeof _opts==="object"){if(env.isLoon){newOpts["openUrl"]=!!_opts["open-url"]?_opts["open-url"]:"";newOpts["mediaUrl"]=!!_opts["media-url"]?_opts["media-url"]:""}else if(env.isQuanX){newOpts=!!_opts["open-url"]||!!_opts["media-url"]?_opts:{}}else if(env.isSurge){let openUrl=_opts["open-url"]||_opts["openUrl"];newOpts=openUrl?{url:openUrl}:{}}}return newOpts}catch(err){logger.error(`通知选项转换失败${err}`)}return _opts};opts=_adaptOpts(opts);if(arguments.length==1){title=scriptName;subTitle="",body=arguments[0]}logger.notify(`title:${title}\nsubTitle:${subTitle}\nbody:${body}\noptions:${typeof opts==="object"?JSON.stringify(opts):opts}`);if(env.isSurge){$notification.post(title,subTitle,body,opts)}else if(env.isLoon){if(!!opts)$notification.post(title,subTitle,body,opts);else $notification.post(title,subTitle,body)}else if(env.isQuanX){$notify(title,subTitle,body,opts)}if(_barkUrl&&_barkKey){bark(title,subTitle,body)}}function debug(title=scriptName,subTitle="",body="",opts=""){if(logger.getLevel()==="DEBUG"){if(arguments.length==1){title=scriptName;subTitle="",body=arguments[0]}this.post(title,subTitle,body,opts)}}function bark(title=scriptName,subTitle="",body="",opts=""){if(typeof http==="undefined"||typeof http.post==="undefined"){throw"Bark notification needs to import MagicHttp module."}let options={url:_barkUrl,headers:{"Content-Type":"application/json; charset=utf-8"},body:{title:title,body:subTitle?`${subTitle}\n${body}`:body,device_key:_barkKey}};http.post(options).catch(ex=>{logger.error(`Bark notify error: ${ex}`)})}return{post:post,debug:debug,bark:bark,setBark:setBark}}
// prettier-ignore
function MagicData(env,logger){let node={fs:undefined,data:{}};if(env.isNode){node.fs=require("fs");try{node.fs.accessSync("./magic.json",node.fs.constants.R_OK|node.fs.constants.W_OK)}catch(err){node.fs.writeFileSync("./magic.json","{}",{encoding:"utf8"})}node.data=require("./magic.json")}const defaultValueComparator=(oldVal,newVal)=>{if(typeof newVal==="object"){return false}else{return oldVal===newVal}};const _typeConvertor=val=>{if(val==="true"){return true}else if(val==="false"){return false}else if(typeof val==="undefined"){return null}else{return val}};const _valConvertor=(val,default_,session,read_no_session)=>{if(session){try{if(typeof val==="string")val=JSON.parse(val);if(val["magic_session"]===true){val=val[session]}else{val=null}}catch{val=null}}if(typeof val==="string"&&val!=="null"){try{val=JSON.parse(val)}catch{}}if(read_no_session===false&&!!val&&val["magic_session"]===true){val=null}if((val===null||typeof val==="undefined")&&default_!==null&&typeof default_!=="undefined"){val=default_}val=_typeConvertor(val);return val};const convertToObject=obj=>{if(typeof obj==="string"){let data={};try{data=JSON.parse(obj);const type=typeof data;if(type!=="object"||data instanceof Array||type==="bool"||data===null){data={}}}catch{}return data}else if(obj instanceof Array||obj===null||typeof obj==="undefined"||obj!==obj||typeof obj==="boolean"){return{}}else{return obj}};const readForNode=(key,default_=null,session="",read_no_session=false,externalData=null)=>{let data=externalData||node.data;if(!!data&&typeof data[key]!=="undefined"&&data[key]!==null){val=data[key]}else{val=!!session?{}:null}val=_valConvertor(val,default_,session,read_no_session);return val};const read=(key,default_=null,session="",read_no_session=false,externalData=null)=>{let val="";if(externalData||env.isNode){val=readForNode(key,default_,session,read_no_session,externalData)}else{if(env.isSurgeLike){val=$persistentStore.read(key)}else if(env.isQuanX){val=$prefs.valueForKey(key)}val=_valConvertor(val,default_,session,read_no_session)}logger.debug(`READ DATA [${key}]${!!session?`[${session}]`:""} <${typeof val}>\n${JSON.stringify(val)}`);return val};const writeForNode=(key,val,session="",externalData=null)=>{let data=externalData||node.data;data=convertToObject(data);if(!!session){let obj=convertToObject(data[key]);obj["magic_session"]=true;obj[session]=val;data[key]=obj}else{data[key]=val}if(externalData!==null){externalData=data}return data};const write=(key,val,session="",externalData=null)=>{if(typeof val==="undefined"||val!==val){return false}if(!env.isNode&&(typeof val==="boolean"||typeof val==="number")){val=String(val)}let data="";if(externalData||env.isNode){data=writeForNode(key,val,session,externalData)}else{if(!session){data=val}else{if(env.isSurgeLike){data=!!$persistentStore.read(key)?$persistentStore.read(key):data}else if(env.isQuanX){data=!!$prefs.valueForKey(key)?$prefs.valueForKey(key):data}data=convertToObject(data);data["magic_session"]=true;data[session]=val}}if(!!data&&typeof data==="object"){data=JSON.stringify(data)}logger.debug(`WRITE DATA [${key}]${session?`[${session}]`:""} <${typeof val}>\n${JSON.stringify(val)}`);if(!externalData){if(env.isSurgeLike){return $persistentStore.write(data,key)}else if(env.isQuanX){return $prefs.setValueForKey(data,key)}else if(env.isNode){try{node.fs.writeFileSync("./magic.json",data);return true}catch(err){logger.error(err);return false}}}return true};const update=(key,val,session,comparator=defaultValueComparator,externalData=null)=>{val=_typeConvertor(val);const oldValue=read(key,null,session,false,externalData);if(comparator(oldValue,val)===true){return false}else{const result=write(key,val,session,externalData);let newVal=read(key,null,session,false,externalData);if(comparator===defaultValueComparator&&typeof newVal==="object"){return result}return comparator(val,newVal)}};const delForNode=(key,session,externalData)=>{let data=externalData||node.data;data=convertToObject(data);if(!!session){obj=convertToObject(data[key]);delete obj[session];data[key]=obj}else{delete data[key]}if(!!externalData){externalData=data}return data};const del=(key,session="",externalData=null)=>{let data={};if(externalData||env.isNode){data=delForNode(key,session,externalData);if(!externalData){node.fs.writeFileSync("./magic.json",JSON.stringify(data))}else{externalData=data}}else{if(!session){if(env.isStorm){return $persistentStore.remove(key)}else if(env.isSurgeLike){return $persistentStore.write(null,key)}else if(env.isQuanX){return $prefs.removeValueForKey(key)}}else{if(env.isSurgeLike){data=$persistentStore.read(key)}else if(env.isQuanX){data=$prefs.valueForKey(key)}data=convertToObject(data);delete data[session];const json=JSON.stringify(data);write(key,json)}}logger.debug(`DELETE KEY [${key}]${!!session?`[${session}]`:""}`)};const allSessions=(key,externalData=null)=>{let _sessions=[];let data=read(key,null,null,true,externalData);data=convertToObject(data);if(data["magic_session"]!==true){_sessions=[]}else{_sessions=Object.keys(data).filter(key=>key!=="magic_session")}logger.debug(`READ ALL SESSIONS [${key}] <${typeof _sessions}>\n${JSON.stringify(_sessions)}`);return _sessions};return{read:read,write:write,del:del,update:update,allSessions:allSessions,defaultValueComparator:defaultValueComparator,convertToObject:convertToObject}}