// ==UserScript== // @name Pagetual // @name:zh-CN 东方永页机 // @name:zh-TW 東方永頁機 // @name:ja 東方永頁機 // @name:ko 東方永頁機 // @name:ru Pagetual // @name:de Pagetual // @name:es Pagetual // @name:fr Pagetual // @name:it Pagetual // @namespace hoothin // @version 1.9.37.29 // @description Perpetual pages - powerful auto-pager script. Auto fetching next paginated web pages and inserting into current page for infinite scroll. Support thousands of web sites without any rule. // @description:zh-CN 终极自动翻页 - 加载并拼接下一分页内容至当前页尾,智能适配任意网页 // @description:zh-TW 終極自動翻頁 - 加載並拼接下一分頁內容至當前頁尾,智能適配任意網頁 // @description:ja Webページを自動で読み込み継ぎ足し表示を行うブラウザ拡張です、次のページ付けされた Web ページの自動読み込みと現在のページへの挿入 ルールなしで何千もの Web サイトをサポートします。 // @description:ko 페이지가 매겨진 다음 웹 페이지를 자동으로 로드하고 현재 페이지에 삽입합니다. 규칙 없이 수천 개의 웹 사이트를 지원합니다. // @description:ru Автоматическая подгрузка следующих страниц и вставка их содержимого в текущую страницу. Поддерживает тысячи сайтов даже с настройками по умолчанию. // @description:de Automatisches Laden der nächsten paginierten Webseiten und Einfügen in die aktuelle Seite. Unterstützen Sie Tausende von Websites ohne Regeln. // @description:es Carga automática de las siguientes páginas web paginadas e inserción en la página actual. Admite miles de sitios web sin ninguna regla. // @description:fr Chargement automatique des pages Web paginées suivantes et insertion dans la page en cours. Prend en charge des milliers de sites Web sans aucune règle. // @description:it Caricamento automatico delle pagine Web impaginate successive e inserimento nella pagina corrente. Supporta migliaia di siti web senza alcuna regola. // @author hoothin // @license MPL-2.0 // @match *://*/* // @icon  // @grant GM_xmlhttpRequest // @grant GM_registerMenuCommand // @grant GM_notification // @grant GM_getValue // @grant GM_setValue // @grant GM_addStyle // @grant GM_openInTab // @grant GM_deleteValue // @grant GM_info // @grant GM_setClipboard // @grant GM.xmlHttpRequest // @grant GM.registerMenuCommand // @grant GM.notification // @grant GM.getValue // @grant GM.setValue // @grant GM.addStyle // @grant GM.openInTab // @grant GM.deleteValue // @grant GM.info // @grant GM.setClipboard // @supportURL https://github.com/hoothin/UserScripts/issues // @connect wedata.net // @connect githubusercontent.com // @connect ghproxy.com // @connect hoothin.github.io // @run-at document-idle // @connect * // @contributionURL https://ko-fi.com/hoothin // @contributionAmount 1 // @downloadURL none // ==/UserScript== (function() { 'use strict'; const pauseVideo = () => { setTimeout(() => { [].forEach.call(document.querySelectorAll("video"), video => { video.removeAttribute && video.removeAttribute("autoplay"); video.pause && video.pause(); }); }, 1000); }; if (window.name === 'pagetual-iframe' || (window.frameElement && window.frameElement.name === 'pagetual-iframe')) { [].forEach.call(document.querySelectorAll("iframe"), iframe => { iframe.name = 'pagetual-iframe'; }); var domloaded = function() { window.parent.postMessage('pagetual-iframe:DOMLoaded', '*'); }; if (window.opera) { document.addEventListener('DOMContentLoaded', domloaded, false); pauseVideo(); } else { domloaded(); if (document.readystate === 'complete') { pauseVideo(); } else { window.addEventListener('load', () => { pauseVideo(); }, false); } } if (getComputedStyle(document.documentElement).display === 'none') { document.documentElement.style.display = 'block'; } if (document.body && getComputedStyle(document.body).display === 'none') { document.body.style.display = 'block'; } return; } if (window.top !== window.self) { try { if (window.self.innerWidth < 300 || window.self.innerHeight < 300) { return; } } catch(e) { return; } } const noRuleTest = false; const lang = navigator.appName === "Netscape" ? navigator.language : navigator.userLanguage; const langData = [ { name: "English", match: ["en"], lang: { enableDebug: "Enable debug output", updateNotification: "Notification after rules updated", disable: "Disable", disableSite: "Toggle disabled state", disableSiteTips: "Disabled on this site", enableSiteTips: "Enabled on this site", enable: "Enable", toTop: "To Top", toBottom: "To Bottom", current: "Current Page", forceIframe: "Force to join next page", cancelForceIframe: "Cancel Force join", configure: "Configure", firstUpdate: "Click here to initialize the rules", update: "Update online rules", click2update: "Click to update rules from url now", loadNow: "Load next automatically", loadConfirm: "How much pages do you want to load? (0 means infinite)", noNext: "No next link found, please create a new rule", passSec: "Updated #t# seconds ago", passMin: "Updated #t# minutes ago", passHour: "Updated #t# hours ago", passDay: "Updated #t# days ago", cantDel: "Can't delete buildin rules", confirmDel: "Are you sure you want to delete this rule?", updateSucc: "Update succeeded", beginUpdate: "Begin update, wait a minute please", customUrls: "Import Pagetual or AutoPagerize rule url, One url per line", customRules: "Input custom rules. ✍️Contribute rules", save: "Save", loadingText: "Shojo Now Loading...", opacity: "Opacity", opacityPlaceholder: "0: hide spacer", hideBar: "Hide the paging spacer", hideBarButNoStop: "Hide but not stop", dbClick2Stop: "Double-click on the blank space to pause", sortTitle: "Sorting takes effect after the next rule update", autoRun: "Auto enable (blacklist mode)", autoLoadNum: "Amount for preload pages", turnRate: "Turn the next page when it's less than 【X】 times page height from the footer", inputPageNum: "Enter page number to jump", enableHistory: "Write browsing history after page turning", enableHistoryAfterInsert: "Write browsing history immediately after splicing, otherwise write after browsing", contentVisibility: "Automatically switch contentVisibility to improve rendering performance", initRun: "Turn pages immediately after opening", preload: "Preload next page for speeding up", click2ImportRule: "Click to import base rules link, then wait until the update is complete: ", forceAllBody: "Join full body of page?", openInNewTab: "Open urls of additions in new tab", importSucc: "Import completed", import: "Import", editCurrent: "Edit rule for current", editBlacklist: "Edit the blacklist urls, line by line, Support ? * for wildcard", upBtnImg: "Icon of back to top", downBtnImg: "Icon of go to footer", sideControllerIcon: "Icon of sidebar", loadingTextTitle: "Loading text", dbClick2StopCtrl: "Ctrl key", dbClick2StopAlt: "Alt key", dbClick2StopShift: "Shift key", dbClick2StopMeta: "Meta key", dbClick2StopKey: "Shortcut key", pageElementCss: "Custom style for main page elements", customCss: "Custom complete css", firstAlert: "You have not imported the base rule, please select the appropriate rule to import", picker: "Pagetual page element picker", closePicker: "Close Pagetual picker", pickerPlaceholder: "Element selector, Leave empty if you have no idea", pickerCheck: "Check selector and copy", switchSelector: "Click to switch element", gotoEdit: "Go to edit rule with current selector", manualMode: "Disable splicing, manually turn pages with the right arrow keys (or dispatch event 'pagetual.next')", clickMode: "Disable splicing, automatically click the next page when scrolling to the end of the page", pageBarMenu: "Click the center of the page bar to open the picker menu", nextSwitch: "Switch next link", arrowToScroll: "Press left arrow key to scroll prev and right arrow key to scroll next", sideController: "Display the paging control bar in the sidebar", sideControllerScroll: "Scroll toggle", sideControllerAlways: "Always show", hideLoadingIcon: "Hide loading animation", hideBarArrow: "Hide arrow for page bar", duplicate: "Duplicate Pagetual have been installed, check your script manager!", forceStateIframe: "Embed full page as iframe", forceStateDynamic: "Load dynamic content via iframe", forceStateDisable: "Disable page turning on this site", autoScrollRate: "Scroll speed (1~1000)", disableAutoScroll: "Stop Auto Scroll", enableAutoScroll: "Enable Auto Scroll", toggleAutoScroll: "Toggle Auto Scroll", page: "Page ", prevPage: "Prev page", nextPage: "Next page", errorRulesMustBeArray: "Rules must be a Array!", errorJson: "JSON error, check again!", editSuccess: "Edit successfully", errorWrongUrl: "Wrong url, check again!", errorAlreadyExists: "Already exists!", settingsSaved: "The settings are saved, refresh to view", iframe: "Iframe", dynamic: "Dynamic", reloadPage: "Edit completed, reload page now?", copied: "Copied", noValidContent: "No valid content detected, Captcha action may be required, click to view", outOfDate: "The script is outdated, update to the latest version in time!", hideBarTips: "Hide the pagination bar, toggle immersive experience", setConfigPage: "Set current page as the default configuration page", wedata2github: "Change the wedata address to the mirror address in the github repository", addOtherProp: "Add rule property", addNextSelector: "Add selector content as nextLink", addPageSelector: "Add selector content as pageElement", propName: "Enter rule property name", propValue: "Enter rule property value", customFirst: "Ignore cache for local custom rules", rulesExample: "Rules Example", lastPage: "Reached the last page", lastPageTips: "Show tips when reaching the last page" } }, { name: "简体中文", match: ["zh-CN", "zh-SG"], lang: { enableDebug: "调试模式,输出信息至控制台", updateNotification: "规则更新后提示", disable: "暂时禁用", disableSite: "站点禁用开关", disableSiteTips: "已在此站禁用", enableSiteTips: "已在此站启用", enable: "启用自动翻页", toTop: "回到顶部", toBottom: "前往页尾", current: "当前页", forceIframe: "强制拼接", cancelForceIframe: "取消强制拼接", configure: "打开配置页面", firstUpdate: "点击此处初始化规则", update: "更新在线规则", click2update: "点击立即更新规则", loadNow: "立即翻页", loadConfirm: "要翻几页?(0为不间断)", noNext: "没有找到下一页,请新建规则", passSec: "更新于 #t# 秒前", passMin: "更新于 #t# 分钟前", passHour: "更新于 #t# 小时前", passDay: "更新于 #t# 天前", cantDel: "无法删除内置规则", confirmDel: "是否确认要删除此规则?", updateSucc: "更新成功", beginUpdate: "正在更新,请耐心等待,不要关闭页面", customUrls: "导入 Pagetual 或 AutoPagerize 规则 url,一行一条", customRules: "输入【东方永页机】格式的自定义规则 ✍️贡献规则", save: "保存设置", loadingText: "少女祈祷中...", opacity: "不透明值", opacityPlaceholder: "0: 隐藏分隔条", hideBar: "隐藏分页隔条", hideBarButNoStop: "隐藏但不停止", dbClick2Stop: "空白处双击暂停翻页", sortTitle: "排序在下次更新规则后生效", autoRun: "自动启用,否则为白名单模式", autoLoadNum: "自动加载指定页数", turnRate: "距离页尾【X】倍页面高度时就开始翻页", inputPageNum: "输入页码跳转", enableHistory: "翻页后写入历史记录", enableHistoryAfterInsert: "拼接后立即写入历史记录,否则浏览完毕后再行写入", contentVisibility: "自动切换 contentVisibility,提升渲染性能", initRun: "打开页面后立即尝试翻页,否则滚动至页尾再翻页", preload: "翻页前预读下一页,加速浏览", click2ImportRule: "点击下方任意一条添加规则库,并静待更新成功,请预先打开链接确认能正常访问再行导入:", forceAllBody: "是否拼接整个页面?", openInNewTab: "使拼接页面的内容在新页面打开", importSucc: "导入成功", import: "导入", editCurrent: "编辑此站规则", editBlacklist: "编辑黑名单网址,一行一条,支持? *通配符与正则", upBtnImg: "回到页首图标", downBtnImg: "前往页尾图标", sideControllerIcon: "侧边栏图标", loadingTextTitle: "加载中文字", dbClick2StopCtrl: "Ctrl", dbClick2StopAlt: "Alt", dbClick2StopShift: "Shift", dbClick2StopMeta: "Meta", dbClick2StopKey: "快捷键", pageElementCss: "页面主体框架的样式", customCss: "自定义 css", firstAlert: "你还未导入规则库,请选择合适的规则库导入哦", picker: "东方永页机主体元素抓取器", closePicker: "关闭东方永页机抓取器", pickerPlaceholder: "元素选择器,没想法建议留空", pickerCheck: "检查你编辑的选择器并复制", switchSelector: "点击切换元素", gotoEdit: "使用当前的选择器前往编辑规则", manualMode: "禁用拼接,手动用右方向键翻页,可使用左方向键返回", clickMode: "禁用拼接,滚动至页尾时自动点击下一页", pageBarMenu: "点击分隔条中间弹出菜单", nextSwitch: "切换其他页码", arrowToScroll: "左方向键滚动至上一页,右方向键滚动至下一页", sideController: "在侧边显示翻页控制栏", sideControllerScroll: "滚动开关", sideControllerAlways: "始终显示", hideLoadingIcon: "隐藏加载动画", hideBarArrow: "隐藏分隔条定位箭头", duplicate: "检测到永页机重复安装,请删除其他脚本管理器中的永页机!", forceStateIframe: "以 iframe 嵌入整页", forceStateDynamic: "通过 iframe 加载动态内容后取出", forceStateDisable: "在此站禁用", autoScrollRate: "滚动速度(1~1000)", disableAutoScroll: "停止自动滚动", enableAutoScroll: "开启自动滚动", toggleAutoScroll: "自动滚动开关", page: "Page ", prevPage: "上一页", nextPage: "下一页", errorRulesMustBeArray: "规则必须为数组形式!", errorJson: "JSON 格式有错,请重新检查!", editSuccess: "编辑成功", errorWrongUrl: "URL 错误, 请重新检查!", errorAlreadyExists: "已经存在!", settingsSaved: "设置已保存,刷新后生效", iframe: "强制拼接", dynamic: "动态加载", reloadPage: "编辑完成,是否立即刷新页面?", copied: "已复制", noValidContent: "没有检测到有效内容,可能需要人机校验,点击查看", outOfDate: "脚本已过时,请及时更新到最新版本!", hideBarTips: "隐藏分页隔条,沉浸式体验", setConfigPage: "将当前页面设为默认配置页", wedata2github: "将 wedata 地址更改为 github 仓库内的镜像地址", addOtherProp: "添加规则属性", addNextSelector: "添加选择器内容为 nextLink", addPageSelector: "添加选择器内容为 pageElement", propName: "输入规则属性名", propValue: "输入规则属性值", customFirst: "为本地自定义规则忽略缓存", rulesExample: "自定义规则详解", lastPage: "已到达最后一页", lastPageTips: "到达最后一页时进行提示" } }, { name: "正體中文", match: ["zh-TW", "zh-HK"], lang: { enableDebug: "調試模式,輸出信息至控制台", updateNotification: "規則更新后提示", disable: "暫時禁用", disableSite: "站點禁用開關", disableSiteTips: "已在此站禁用", enableSiteTips: "已在此站啟用", enable: "啟用自動翻頁", toTop: "回到頂部", toBottom: "前往頁尾", current: "當前頁", forceIframe: "強制拼接", cancelForceIframe: "取消强制拼接", configure: "打開配置頁面", firstUpdate: "點擊此處初始化規則", update: "更新在綫規則", click2update: "點擊立即更新規則", loadNow: "立即翻頁", loadConfirm: "要翻几頁?(0為不間斷)", noNext: "沒有找到下一頁,請新建規則", passSec: "更新于 #t# 秒前", passMin: "更新于 #t# 分鐘前", passHour: "更新于 #t# 小時前", passDay: "更新于 #t# 天前", cantDel: "無法刪除内置規則", confirmDel: "是否確認要刪除此規則?", updateSucc: "更新成功", beginUpdate: "正在更新,請稍候", customUrls: "導入 Pagetual 或 AutoPagerize 規則 url,一行一條", customRules: "輸入【東方永頁機】格式的自定義規則 ✍️貢獻規則", save: "存儲設置", loadingText: "少女祈禱中...", opacity: "不透明值", opacityPlaceholder: "0: 隱藏分隔條", hideBar: "隱藏分頁隔條", hideBarButNoStop: "隱藏但不停止", dbClick2Stop: "空白處雙擊暫停翻頁", sortTitle: "排序在下次更新規則後生效", autoRun: "自動啓用,否則為白名單模式", autoLoadNum: "自動加載指定頁數", turnRate: "距離頁尾【X】倍頁面高度時就開始翻頁", inputPageNum: "輸入頁碼跳轉", enableHistory: "翻頁后寫入歷史記錄", enableHistoryAfterInsert: "拼接後立即寫入歷史記錄,否則瀏覽完畢後再行寫入", contentVisibility: "自動切換 contentVisibility,提升渲染性能", initRun: "打開頁面后立即嘗試翻頁,否則滾動至頁尾再翻頁", preload: "翻頁前預讀下一頁,加速瀏覽", click2ImportRule: "點擊下方添加特殊規則庫,并靜待更新成功:", forceAllBody: "是否拼接整個頁面?", openInNewTab: "使拼接頁面的内容在新頁面打開", importSucc: "導入成功", import: "導入", editCurrent: "編輯此站規則", editBlacklist: "編輯黑名單網址,一行一條,支持? *通配符與正則", upBtnImg: "回到頁首圖標", downBtnImg: "前往頁尾圖標", sideControllerIcon: "側邊欄圖標", loadingTextTitle: "加載中文字", dbClick2StopCtrl: "Ctrl", dbClick2StopAlt: "Alt", dbClick2StopShift: "Shift", dbClick2StopMeta: "Meta", dbClick2StopKey: "快捷鍵", pageElementCss: "頁面主體框架的樣式", customCss: "自定義 css", firstAlert: "你還未導入規則庫,請選擇合適的規則庫導入哦", picker: "東方永頁機主體元素抓取器", closePicker: "關閉東方永頁機抓取器", pickerPlaceholder: "元素選擇器,沒想法建議留空", pickerCheck: "檢查你編輯的選擇器並複製", switchSelector: "點擊切換元素", gotoEdit: "使用當前的選擇器前往編輯規則", manualMode: "禁用拼接,手動用右方向鍵翻頁,左方向鍵返回", clickMode: "禁用拼接,滾動至頁尾時自動點擊下一頁", pageBarMenu: "點擊分隔條中間彈出菜單", nextSwitch: "切換其他頁碼", arrowToScroll: "左方向鍵滾動至上一頁,右方向鍵滾動至下一頁", sideController: "在側邊顯示翻頁控制欄", sideControllerScroll: "滾燈開關", sideControllerAlways: "始終顯示", hideLoadingIcon: "隱藏加載動畫", hideBarArrow: "隱藏分隔條定位箭頭", duplicate: "檢測到永頁機重複安裝,請刪除其他腳本管理器中的永頁機!", forceStateIframe: "以 iframe 嵌入整頁", forceStateDynamic: "通過 iframe 加載動態內容後取出", forceStateDisable: "在此站禁用", autoScrollRate: "滾動速度(1~1000)", disableAutoScroll: "停止自動滾動", enableAutoScroll: "開啟自動滾動", toggleAutoScroll: "自動滾動開關", page: "Page ", prevPage: "上一頁", nextPage: "下一頁", errorRulesMustBeArray: "規則必須為陣列形式!", errorJson: "JSON 格式有錯,請重新檢查!", editSuccess: "編輯成功", errorWrongUrl: "URL 錯誤, 請重新檢查!", errorAlreadyExists: "已經存在!", settingsSaved: "設置已保存,刷新後生效", iframe: "強制拼接", dynamic: "動態加載", reloadPage: "編輯完成,是否立即刷新頁面?", copied: "已復制", noValidContent: "沒有檢測到有效内容,可能需要人機校驗,點擊查看", outOfDate: "脚本已過時,請及時更新到最新版本!", hideBarTips: "隱藏分頁隔條,沉浸式體驗", setConfigPage: "將當前頁面設為默認配置頁", wedata2github: "將 wedata 地址更改為 github 倉庫內的鏡像地址", addOtherProp: "添加規則屬性", addNextSelector: "添加選擇器內容為 nextLink", addPageSelector: "添加選擇器內容為 pageElement", propName: "輸入規則屬性名", propValue: "輸入規則屬性值", customFirst: "為本地自定義規則忽略緩存", rulesExample: "自定義規則詳解", lastPage: "已到達最後一頁", lastPageTips: "到達最後一頁時進行提示" } }, { name: "日本語", match: ["ja"], lang: { enableDebug: "デバッグモード", updateNotification: "ルール更新後の通知", disable: "一時的に無効", disableSite: "無効状態の切り替え", disableSiteTips: "このサイトで既に無効になっています", enableSiteTips: "このサイトで既に有効になっています", enable: "ページめくりを有効にする", toTop: "トップに戻る", toBottom: "ページの下部に移動", current: "現在のページ", forceIframe: "強制ステッチ", cancelForceIframe: "強制ステッチをキャンセル", configure: "設定ページを開く", firstUpdate: "ここをクリックしてルールを初期化します", update: "更新ルール", click2update: "今すぐルールを更新してください", loadNow: "今すぐページをめくる", loadConfirm: "数ページめくりたいですか?(0は途切れない)", noNext: "次のページが見つかりません、新しいルールを作成してください", passSec: "#t#秒前に更新", passMin: "#t#分前に更新", passHour: "#t#時間前に更新", passDay: "#t#日前に更新", cantDel: "組み込みルールを削除できません", confirmDel: "このルールを削除してもよろしいですか?", updateSucc: "更新に成功しました", beginUpdate: "更新中、お待ちください", customUrls: "インポートルールのURL、1行に1つ", customRules: "【東方永頁機】の形式でカスタムルールを入力してください ✍️寄稿ルール", save: "設定を保存", loadingText: "少女祈祷中...", opacity: "不透明値", opacityPlaceholder: "0: 隠す", hideBar: "ページ区切り文字を非表示にします", hideBarButNoStop: "非表示にするが停止しない", dbClick2Stop: "空白部分をダブルクリックしてページめくりを一時停止します", sortTitle: "並べ替えは、次のルールの更新後に有効になります", autoRun: "自動的に有効", autoLoadNum: "指定したページ数を自動的に読み込みます", turnRate: "ページの端からページの高さの【X】倍になったらページをめくる", inputPageNum: "ジャンプするページ番号を入力", enableHistory: "ページめくり後の履歴を書く", enableHistoryAfterInsert: "スプライシングの直後に履歴レコードを書き込みます。それ以外の場合は、閲覧後に書き込みます", contentVisibility: "contentVisibility を自動的に切り替えてレンダリング パフォーマンスを向上させる", initRun: "Webページを開いた直後にページをめくる", preload: "事前に次のページを読む", click2ImportRule: "以下をクリックして、ルールベースを追加します:", forceAllBody: "フルページ埋め込み?", openInNewTab: "スプライスされたページのコンテンツを新しいページで開きます", importSucc: "インポート完了", import: "インポート", editCurrent: "現在のルールの編集", editBlacklist: "ブラックリストのURLを編集し、1行ずつ、サポート? *ワイルドカード", upBtnImg: "トップアイコンに戻る", downBtnImg: "フッターアイコンに移動", sideControllerIcon: "サイドバー アイコン", loadingTextTitle: "テキストをロード", dbClick2StopCtrl: "Ctrlキー", dbClick2StopAlt: "Altキー", dbClick2StopShift: "Shiftキー", dbClick2StopMeta: "Metaキー", dbClick2StopKey: "Shortcutキー", pageElementCss: "ページ本文フレームの STYLE", customCss: "カスタム css", firstAlert: "ルールベースをインポートしていないため、インポートする適切なルールベースを選択してください", picker: "Pagetualページ要素ピッカー", closePicker: "Pagetualピッカーを閉じる", pickerPlaceholder: "わからない場合は空のままにしてください", pickerCheck: "セレクターをチェックしてコピー", switchSelector: "クリックして要素を切り替えます", gotoEdit: "現在のセレクターでルールを編集する", manualMode: "スプライシングを無効にします。手動で右の矢印キーを使用してページをめくります", clickMode: "スティッチングを無効にします。ページの最後までスクロールすると、次のページが自動的にクリックされます", pageBarMenu: "ページバーの中央をクリックしてメニューをポップアップ表示", nextSwitch: "次のページに切り替え", arrowToScroll: "左矢印キーで前へ、右矢印キーで次へ", sideController: "サイドバーにページング コントロール バーを表示する", sideControllerScroll: "スクロール スイッチ", sideControllerAlways: "常に表示", hideLoadingIcon: "読み込み中のアニメーションを隠す", hideBarArrow: "分割線の位置矢印を隠す", duplicate: "Pagetual の重複インストールが検出されました。他のスクリプト マネージャで永続的なページ マシンを削除してください!", forceStateIframe: "iframe にページ全体を埋め込む", forceStateDynamic: "iframe 経由で動的コンテンツを読み込む", forceStateDisable: "このステーションでのページめくりを無効にする", autoScrollRate: "スクロール速度 (1~1000)", disableAutoScroll: "自動スクロールを停止します", EnableAutoScroll: "自動スクロールを有効にする", toggleAutoScroll: "自動スクロールの切り替え", page: "Page ", prevPage: "Prev page", nextPage: "Next page", errorRulesMustBeArray: "Rules must be a Array!", errorJson: "JSON error, check again!", editSuccess: "Edit successfully", errorWrongUrl: "Wrong url, check again!", errorAlreadyExists: "Already exists!", settingsSaved: "The settings are saved, refresh to view", iframe: "Iframe", dynamic: "Dynamic", reloadPage: "Edit completed, reload page now?", copied: "Copied", noValidContent: "有効なコンテンツが検出されませんでした。クリックして表示", outOfDate: "スクリプトが古くなっています。最新バージョンに更新してください。", hideBarTips: "ページネーション バーを非表示にします。没入型エクスペリエンス", setConfigPage: "現在のページをデフォルト設定ページとして設定", wedata2github: "wedata アドレスを github ウェアハウスのミラー アドレスに変更", addOtherProp: "ルールプロパティを追加", addNextSelector: "セレクターのコンテンツを nextLink として追加", addPageSelector: "セレクタ コンテンツを pageElement として追加", propName: "ルールのプロパティ名を入力してください", propValue: "ルールのプロパティ値を入力してください", customFirst: "ローカルカスタムルールのキャッシュを無視する", rulesExample: "カスタムルールの詳しい説明", lastPage: "最後のページに到達しました", lastPageTips: "最後のページに到達した時にヒントを表示する" } }, { name: "Русский", match: ["ru", "ru-RU"], lang: { enableDebug: "Включить отладку", updateNotification: "Уведомление после обновления правила", disable: "Выключить", disableSite: "Включить/выключить на сайте", disableSiteTips: "Выключено для этого сайта", enableSiteTips: "Включено для этого сайта", enable: "Включить", toTop: "Наверх", toBottom: "Вниз", current: "Текущая страница", forceIframe: "Подгрузить ещё страницу", cancelForceIframe: "Отменить подгрузку", configure: "Настройки", firstUpdate: "Нажмите здесь, чтобы инициализировать правила", update: "Обновить правила", click2update: "Нажмите, чтобы обновить правила", loadNow: "Загрузить следующую страницу", loadConfirm: "Сколько страниц вы хотите загрузить? (0 означает бесконечность)", noNext: "Ссылка на следующую страницу не найдена. Пожалуйста, создайте новое правило", passSec: "Обновлено #t# секунд назад", passMin: "Обновлено #t# минут назад", passHour: "Обновлено #t# часов назад", passDay: "Обновлено #t# дней назад", cantDel: "Нельзя удалить правила по умолчанию", confirmDel: "Вы уверены, что хотите удалить эту ссылку?", updateSucc: "Правила обновлены", beginUpdate: "Обновление. Пожалуйста, немного подождите", customUrls: "Ссылки с правилами для импорта. Одна ссылка на строку", customRules: "Введите пользовательские правила в формате Pagetual. ✍️Улучшить встроенные правила", save: "Сохранить настройки", loadingText: "Следующая страница подгружается…", opacity: "Прозрачность", opacityPlaceholder: "0 - скрыть", hideBar: "Скрыть разделитель страниц", hideBarButNoStop: "Скрыть, но не останавливать", dbClick2Stop: "Двойной клик на странице для выключения", sortTitle: "Правило сортировки применится после следующего обновления правил", autoRun: "Автозапуск (режим черного списка)", autoLoadNum: "Количество страниц для предзагрузки", turnRate: "Подгрузить страницу, когда она будет в 【X】 раз больше высоты страницы от конца страницы", inputPageNum: "Введите номер страницы для перехода", enableHistory: "Записать историю после переключения страниц", enableHistoryAfterInsert: "Записать запись истории сразу после вставки, иначе записать после просмотра", contentVisibility: "Автоматически переключать contentVisibility для повышения производительности рендеринга", initRun: "Подгружать страницы сразу после открытия", preload: "Предзагрузка следующей страницы", click2ImportRule: "Нажмите, чтобы импортировать базовые правила: ", forceAllBody: "Присоединить страницу целиком?", openInNewTab: "Открыть дополнительные ссылки в новой вкладке", importSucc: "Импорт завершен", import: "Импорт", editCurrent: "Изменить правило для текущего сайта", editBlacklist: "Изменить черный список. Одна ссылка на строку. Поддерживаются метасимволы: ? и *", upBtnImg: "Иконка перехода к началу", downBtnImg: "Иконка перехода к концу", sideControllerIcon: "Значок боковой панели", loadingTextTitle: "Текст во время загрузки", dbClick2StopCtrl: "Ctrl", dbClick2StopAlt: "Alt", dbClick2StopShift: "Shift", dbClick2StopMeta: "Meta", dbClick2StopKey: "Клавиша", pageElementCss: "Пользовательский стиль для основных элементов страницы", customCss: "Полный пользовательский CSS", firstAlert: "Вы не импортировали базовое правило. Пожалуйста, выберите соответствующее правило для импорта", picker: "Pagetual: выбор элемента страницы", closePicker: "Закрыть окно Pagetual", pickerPlaceholder: "Если не знаете, что тут писать — оставьте поле пустым", pickerCheck: "Проверить и скопировать селектор", switchSelector: "Нажмите для выбора элемента", gotoEdit: "Перейти к редактированию правила с текущим селектором", manualMode: "Отключить автоматическую загрузку страниц. Загружать вручную с помощью стрелки вправо (или вызова события 'pagetual.next')", clickMode: "Отключить \"сшивание\" страниц. При прокрутке до конца автоматически переходить на следующую страницу", pageBarMenu: "Открывать меню кликом на середину панели страниц", nextSwitch: "Переключить ссылку на следующую страницу", arrowToScroll: "Листать страницы клавишами со стрелками влево и вправо", sideController: "Показать справа панель перемещения по вкладке", sideControllerScroll: "Переключатель прокрутки", sideControllerAlways: "Всегда показывать", hideLoadingIcon: "Скрыть анимацию загрузки", hideBarArrow: "Скрыть кнопки перемещения на разделителе", duplicate: "Похоже, Pagetual установлен несколько раз. Пожалуйста, удалите Pagetual из других менеджеров скриптов!", forceStateIframe: "Вставить полную страницу как iframe", forceStateDynamic: "Загружать динамический контент через iframe", forceStateDisable: "Отключить перелистывание страниц на этой станции", autoScrollRate: "Скорость прокрутки (1~1000)", disableAutoScroll: "Остановить автоматическую прокрутку", enableAutoScroll: "Включить автопрокрутку", toggleAutoScroll: "Переключить автопрокрутку", page: "Страница ", prevPage: "Предыдущая страница", nextPage: "Следующая страница", errorRulesMustBeArray: "Правила должны быть массивом!", errorJson: "Ошибка разбора JSON. Пожалуйста, исправьте его", editSuccess: "Редактирование успешно", errorWrongUrl: "Ссылка некорректна. Пожалуйста, исправьте её", errorAlreadyExists: "Уже существует!", settingsSaved: "Настройки сохранены. Обновите страницы", iframe: "iframe", dynamic: "Динамически", reloadPage: "Редактирование завершено. Обновить страницу?", copied: "Скопировано", noValidContent: "Действительный контент не обнаружен, нажмите для просмотра", outOfDate: "Скрипт устарел, своевременно обновляйте до последней версии!", hideBarTips: "Скрыть панель разбиения на страницы, иммерсивный опыт", setConfigPage: "Установить текущую страницу в качестве страницы конфигурации по умолчанию", wedata2github: "Изменить адрес wedata на зеркальный адрес на складе github", addOtherProp: "Добавить свойство правила", addNextSelector: "Добавить содержимое селектора как nextLink", addPageSelector: "Добавить содержимое селектора как pageElement", propName: "Введите имя свойства правила", propValue: "Введите значение свойства правила", customFirst: "Игнорировать кеш для локальных пользовательских правил", rulesExample: "Подробное объяснение настраиваемых правил", lastPage: "Достигнута последняя страница", lastPageTips: "Показывать подсказки при достижении последней страницы" } } ]; var langList = {}; langData.forEach(lang => { langList[lang.match[0]] = lang.name; }); var i18nData = langData[0].lang; function setLang(la) { for (let i = 0; i < langData.length; i++) { let lang = langData[i]; if (lang && lang.match.indexOf(la) !== -1) { i18nData = lang.lang; break; } } } setLang(lang); var enableDebug = true; var _GM_xmlhttpRequest, _GM_registerMenuCommand, _GM_notification, _GM_addStyle, _GM_openInTab, _GM_info, _GM_setClipboard; function i18n(name, param) { return i18nData[name] ? i18nData[name].replace("#t#", param) : name; } function debug(str, title) { if (enableDebug) { console.log( `%c【Pagetual v.${_GM_info.script.version}】 ${title ? title : 'debug'}:`, 'color: yellow;font-size: large;font-weight: bold;background-color: darkblue;', str ); } } if (typeof GM_xmlhttpRequest !== 'undefined') { _GM_xmlhttpRequest = GM_xmlhttpRequest; } else if (typeof GM !== 'undefined' && typeof GM.xmlHttpRequest !== 'undefined') { _GM_xmlhttpRequest = GM.xmlHttpRequest; } else { _GM_xmlhttpRequest = (f) => {fetch(f.url, {method: f.method || 'GET', body: f.data, headers: f.headers}).then(response => response.text()).then(data => {f.onload && f.onload({response: data})}).catch(f.onerror && f.onerror())}; } if (typeof GM_registerMenuCommand !== 'undefined') { _GM_registerMenuCommand = GM_registerMenuCommand; } else if (typeof GM !== 'undefined' && typeof GM.registerMenuCommand !== 'undefined') { _GM_registerMenuCommand = GM.registerMenuCommand; } else { _GM_registerMenuCommand = (s, f) => {debug(s); debug(f);}; } if (typeof GM_info !== 'undefined') { _GM_info = GM_info; } else if (typeof GM !== 'undefined' && typeof GM.info !== 'undefined') { _GM_info = GM.info; } else { _GM_info = {script: {}}; } if (typeof GM_notification !== 'undefined') { _GM_notification = GM_notification; } else if (typeof GM !== 'undefined' && typeof GM.notification !== 'undefined') { _GM_notification = GM.notification; } else { _GM_notification = (s) => {showTips(String(s.text || s));}; } if (typeof GM_openInTab !== 'undefined') { _GM_openInTab = GM_openInTab; } else if (typeof GM !== 'undefined' && typeof GM.openInTab !== 'undefined') { _GM_openInTab = GM.openInTab; } else { _GM_openInTab = (s,t) => {window.open(s); debug(t);}; } if (typeof GM_addStyle !== 'undefined') { _GM_addStyle = GM_addStyle; } else if (typeof GM !== 'undefined' && typeof GM.addStyle !== 'undefined') { _GM_addStyle = GM.addStyle; } else { _GM_addStyle = cssStr => { let styleEle = document.createElement("style"); styleEle.innerHTML = cssStr; document.head.appendChild(styleEle); return styleEle; }; } if (typeof GM_setClipboard !== 'undefined') { _GM_setClipboard = GM_setClipboard; } else if (typeof GM !== 'undefined' && typeof GM.setClipboard !== 'undefined') { _GM_setClipboard = GM.setClipboard; } else { _GM_setClipboard = (s, i) => {debug(s); debug(i);}; } const _unsafeWindow = (typeof unsafeWindow === 'undefined') ? window : unsafeWindow;//兼容 ios userscripts 的寫法 const storage = { supportGM: typeof GM_getValue === 'function' && typeof GM_getValue('a', 'b') !== 'undefined', supportGMPromise: typeof GM !== 'undefined' && typeof GM.getValue === 'function' && typeof GM.getValue('a', 'b') !== 'undefined' && typeof GM.getValue('a', 'b').then === 'function', supportCrossSave: function() { return this.supportGM || this.supportGMPromise; }, mxAppStorage: (function() { try { return window.external.mxGetRuntime().storage; } catch(e) { } })(), operaUJSStorage: (function() { try { return window.opera.scriptStorage; } catch(e) { } })(), setItem: function(key, value) { if (this.supportGMPromise) { GM.setValue(key, value); if (value === "" && typeof GM !== 'undefined' && typeof GM.deleteValue !== 'undefined') { GM.deleteValue(key); } } else if (this.supportGM) { GM_setValue(key, value); if (value === "" && typeof GM_deleteValue !== 'undefined') { GM_deleteValue(key); } } else if (this.operaUJSStorage) { this.operaUJSStorage.setItem(key, value); } else if (this.mxAppStorage) { this.mxAppStorage.setConfig(key, value); } else if (window.localStorage) { window.localStorage.setItem(key, value); } }, getItem: function(key, cb) { var value; if (this.supportGMPromise) { value = GM.getValue(key).then(v => {cb(v);}); return; } else if (this.supportGM) { value = GM_getValue(key); } else if (this.operaUJSStorage) { value = this.operaUJSStorage.getItem(key); } else if (this.mxAppStorage) { value = this.mxAppStorage.getConfig(key); } else if (window.localStorage) { value = window.localStorage.getItem(key); } cb(value); } }; async function getData(key) { return new Promise((resolve) => { storage.getItem(key, value => { resolve(value); }); }); } async function getListData(list, key) { return new Promise((resolve) => { storage.getItem(list, listData => { let value; if (listData) { for(var i = 0; i < listData.length; i++) { let data = listData[i]; if (data.k === key) { value = data.v; break; } } } resolve(value); }); }); } function setListData(list, key, value, length) { storage.getItem(list, listData => { if (!listData) { listData = []; } listData = listData.filter(data => data && data.k !== key); if (value !== "") { listData.unshift({k: key, v: value}); if (listData.length > (length || 100)) { listData.pop(); } } storage.setItem(list, listData); }); } const isMobile = ('ontouchstart' in document.documentElement); const configPage = [`https://pagetual.hoothin.com/${lang === 'zh-CN' ? 'cn/' : ''}rule.html`, "https://github.com/hoothin/UserScripts/tree/master/Pagetual", "https://hoothin.github.io/UserScripts/Pagetual/"]; const firstRunPage = "https://pagetual.hoothin.com/firstRun"; const guidePage = /^https?:\/\/.*pagetual.*rule\.html/i; const ruleImportUrlReg = /greasyfork\.org\/.*scripts\/438684(\-[^\/]*)?(\/discussions|\/?$|\/feedback)|github\.com\/hoothin\/UserScripts\/(tree\/master\/Pagetual|issues)/i; const allOfBody = "body>*"; const mainSel = "article,.article,[role=main],main,.main,#main"; const nextTextReg1 = new RegExp("\u005e\u7ffb\u003f\u005b\u4e0b\u540e\u5f8c\u6b21\u005d\u005b\u4e00\u30fc\u0031\u005d\u003f\u005b\u9875\u9801\u5f20\u5f35\u005d\u007c\u005e\u006e\u0065\u0078\u0074\u005b\u005c\u0073\u005f\u002d\u005d\u003f\u0070\u0061\u0067\u0065\u005c\u0073\u002a\u005b\u203a\u003e\u2192\u00bb\u005d\u003f\u0024\u007c\u6b21\u306e\u30da\u30fc\u30b8\u007c\u005e\u6b21\u3078\u003f\u0024\u007c\u0412\u043f\u0435\u0440\u0435\u0434", "i"); const nextTextReg2 = new RegExp("\u005e\u0028\u005b\u4e0b\u540e\u5f8c\u6b21\u005d\u005b\u4e00\u30fc\u0031\u005d\u003f\u005b\u7ae0\u8bdd\u8a71\u8282\u7bc0\u5e45\u005d\u007c\u006e\u0065\u0078\u0074\u002e\u003f\u0063\u0068\u0061\u0070\u0074\u0065\u0072\u007c\u00bb\u007c\u003e\u003e\u0029\u0028\u005b\u003a\uff1a\u005c\u002d\u005f\u2014\u005c\u0073\u005c\u002e\u3002\u003e\u0023\u00b7\u005c\u005b\u3010\u3001\uff08\u005c\u0028\u002f\u002c\uff0c\uff1b\u003b\u2192\u005d\u007c\u0024\u0029", "i"); const lazyImgAttr = ["data-lazy-src", "data-lazy", "data-isrc", "data-url", "data-orig-file", "zoomfile", "file", "original", "load-src", "imgsrc", "real_src", "src2", "origin-src", "data-lazyload", "data-lazyload-src", "data-lazy-load-src", "data-ks-lazyload", "data-ks-lazyload-custom", "data-src", "data-defer-src", "data-actualsrc", "data-cover", "data-original", "data-thumb", "data-imageurl", "data-placeholder", "lazysrc"]; var rulesData = {uninited: true, firstRun: true, sideController: !isMobile}, ruleUrls, updateDate, clickedSth = false, loadNowNum = 5, autoScrollRate = 50; var isPause = false, manualPause = false, isHideBar = false, isLoading = false, curPage = 1, forceState = 0, autoScroll = 0, autoScrollInterval, bottomGap = 1000, autoLoadNum = -1, nextIndex = 0, stopScroll = false, clickMode = false, openInNewTab = 0, charset = "UTF-8", charsetValid = true, urlWillChange = false; var tryTimes = 0, showedLastPageTips = false, rate = 1, author = ''; function getBody(doc) { return doc.body || doc.querySelector('body') || doc; } function getElementByXpath(xpath, doc, contextNode) { doc = doc || document; contextNode = contextNode || doc; try { let result = doc.evaluate(xpath, contextNode, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null); return result.singleNodeValue && result.singleNodeValue.nodeType === 1 && result.singleNodeValue; } catch (err) { debug(`Invalid xpath: ${xpath}`); } return null; } function getAllElementsByXpath(xpath, contextNode, doc) { doc = doc || document; contextNode = contextNode || doc; var result = []; try { var query = doc.evaluate(xpath, contextNode, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null); for (var i = 0; i < query.snapshotLength; i++) { var node = query.snapshotItem(i); if (node.nodeType === 1) { result.push(node); } } } catch (err) { debug(`Invalid xpath: ${xpath}`); } return result; } function isXPath(xpath) { if (!xpath) { return false; } return /^\(*(descendant::|\.\/|\/|id\()/.test(xpath); } function getAllElements(sel, doc, contextNode) { try { if (!isXPath(sel)) { return doc.querySelectorAll(sel); } } catch(e) { debug(e, 'Error selector'); } return getAllElementsByXpath(sel, contextNode, doc); } function getElement(sel, doc, contextNode, bySort) { try { if (!isXPath(sel)) { if (!bySort) { return doc.querySelector(sel); } else { let selArr = sel.split(","); try { for (let i = 0; i < selArr.length; i++) { let ele = doc.querySelector(selArr[i].trim()); if (ele) { return ele; } } } catch(e) { return doc.querySelector(sel); } return null; } } } catch(e) { debug(e, 'Error selector'); } return getElementByXpath(sel, doc, contextNode); } function compareNodeName(node, names) { if (!node || !node.nodeName || !node.nodeName.toLowerCase) { return false; } let nodeName = node.nodeName.toLowerCase(); for (let i = 0; i < names.length; i++) { if (names[i] === nodeName) { return true; } } return false; } function geneSelector(ele, addID, exact) { let selector = ele.nodeName.toLowerCase(); //Google id class都是隨機。百度更過分,style script順序都是隨機的 if (selector !== "html" && selector !== "body") { let hasId = false; if (addID && ele.id && /^[a-z_][\w\-_]*$/i.test(ele.id)) { if (ele.ownerDocument && ele.ownerDocument.querySelectorAll("#" + ele.id).length === 1) { hasId = true; selector = '#' + ele.id; } } if (!hasId) { let className = ""; if (ele.className) { let classList = ele.classList; for (let i = 0; i < classList.length; i++) { let c = classList[i]; if (/^[\w\-_]+$/.test(c) && !/\d{4,}/.test(c)) { className += '.' + c; } } selector += className; } let parent = ele.parentElement; if (parent) { if (exact) { let i, nth = 0, all = 0; for (i = 0; i < parent.children.length; i++) { if (parent.children[i].nodeName === ele.nodeName) { all++; if (parent.children[i] === ele) { nth = all; } if (nth > 0 && all > 1) { break; } } } selector += (all === 1 ? "" : `:nth-of-type(${nth})`); } else if (!className && !hasId && parent.children.length > 1 && !compareNodeName(parent, ["html"])) { let prevE = ele.previousElementSibling; if (prevE && prevE.className) { let classList = prevE.classList; for (let i = 0; i < classList.length; i++) { let c = classList[i]; if (/^[\w\-_]+$/.test(c) && !/\d{4,}/.test(c)) { className += '.' + c; } } if (className) { selector = prevE.nodeName.toLowerCase() + className + "+" + selector; } } if (!className) { let i, nth = 0, all = 0; for (i = 0; i < parent.children.length; i++) { if (parent.children[i].nodeName === ele.nodeName) { all++; if (parent.children[i] === ele) { nth = all; } if (nth > 0 && all > 1) { break; } } } selector += (all === 1 ? "" : `:nth-of-type(${nth})`); } } selector = geneSelector(parent, addID, exact) + ' > ' + selector; } } } return selector; } function createXPathFromElement(elm) { let allNodes = document.getElementsByTagName('*'), segs; for (segs = []; elm && elm.nodeType === 1; elm = elm.parentNode) { if (compareNodeName(elm, ["body", "html"])) { segs.unshift(elm.localName.toLowerCase()); continue; } if (elm.hasAttribute && elm.hasAttribute('id')) { var uniqueIdCount = 0; for (var n = 0; n < allNodes.length; n++) { if (allNodes[n].hasAttribute('id') && allNodes[n].id === elm.id) { uniqueIdCount++; } if (uniqueIdCount > 1) { break; } } if ( uniqueIdCount === 1) { segs.unshift('id("' + elm.getAttribute('id') + '")'); return segs.join('/'); } else { segs.unshift(elm.localName.toLowerCase() + '[@id="' + elm.getAttribute('id') + '"]'); } } else if (elm.hasAttribute && elm.hasAttribute('class')) { segs.unshift(elm.localName.toLowerCase() + '[@class="' + elm.getAttribute('class') + '"]'); } else { let i, sib; for (i = 1, sib = elm.previousSibling; sib; sib = sib.previousSibling) { if (sib.localName === elm.localName) { i++; } } segs.unshift(elm.localName.toLowerCase() + '[' + i + ']'); } } return segs.length ? '/' + segs.join('/') : null; } const escapeHTMLPolicy = (_unsafeWindow.trustedTypes && _unsafeWindow.trustedTypes.createPolicy) ? _unsafeWindow.trustedTypes.createPolicy('pagetual_default', { createHTML: (string, sink) => string }) : null; function createHTML(html) { return escapeHTMLPolicy ? escapeHTMLPolicy.createHTML(html) : html; } const AsyncFunction = Object.getPrototypeOf(async function(){}).constructor; async function sleep(time) { await new Promise((resolve) => { setTimeout(() => { resolve(); }, time); }); } class RuleParser { constructor() { this.hpRules = []; this.smartRules = []; this.customRules = []; this.rules = []; this.pageDoc = document; this.nextLinkHref = null; this.nextTitle = ""; this.oldUrl = ""; this.curUrl = location.href; this.curSiteRule = {}; } async initSavedRules(callback) { var self = this; let smartRules = await getData("smartRules"); if (smartRules) self.smartRules = smartRules; let hpRules = await getData("hpRules"); if (hpRules) self.hpRules = hpRules; let customRules = await getData("customRules"); if (customRules) self.customRules = customRules; if (_unsafeWindow.pagetualRules && _unsafeWindow.pagetualRules.length) { _unsafeWindow.pagetualRules.forEach(rule => { rule.isScript = true; }); self.customRules = _unsafeWindow.pagetualRules.concat(self.customRules); } let rules = await getData("rules"); if (rules) self.rules = rules; callback(); } saveCurSiteRule() { /*if(!this.curSiteRule || !this.curSiteRule.url || this.curSiteRule.smart || this.curSiteRule.url.length<13)return; this.hpRules=this.hpRules.filter(item=>{return item&&item.url!=this.curSiteRule.url}); this.hpRules.unshift(this.curSiteRule); if(this.hpRules.length>30){ this.hpRules.pop(); } storage.setItem("hpRules", this.hpRules);*/ } requestJSON(url, callback) { _GM_xmlhttpRequest({ url: url, method: 'GET', timeout: 20000, headers: { 'accept': 'application/json,text/html' }, onload: function(res) { let json = null; try { json = JSON.parse(res.response || res.responseText); } catch(e) { debug(e, 'Error json'); } callback(json); }, onerror: function(e) { callback(null, e); }, ontimeout: function(e) { callback(null, e); } }); } formatRule(item, from) { if (item.data && item.data.url) { let result = { name: item.name, from: from, action: item.data.forceIframe === "true" ? 1 : undefined, url: item.data.url, pageElement: item.data.pageElement, nextLink: item.data.nextLink, insert: item.data.insertBefore||undefined, updatedAt: item.updated_at }; let _css = (item.data.Stylus || '') + (item.data.CSS || ''); if (_css) result.css = _css; if (item.data.bookmarklet) result.pageAction = item.data.bookmarklet; return result; } else { item.from = from; return item; } return null; } addRuleByUrl(url, from, callback) { if (url.indexOf("?") === -1) { url = url + "?" + Date.now(); } else { url = url + "&" + Date.now(); } this.requestJSON(url, (json,err) => { if (!json) { debug(err, "Failed to update " + url + " rules!"); } this.addRules(json, from); callback(json, err); }); } addRules(rules, from) { if (rules && rules.length > 0) { let first = -1; this.rules = this.rules.filter((item, i) => { if (item.from === from) { if (first === -1) first = i; return false; } else return true; }); if (first === -1) first = 0; rules.forEach(item => { let rule = this.formatRule(item, from); if (rule) { this.rules.splice(first, 0, rule) } }); } } ruleMatchPre(r) { if (r.include) { let include; if (Array && Array.isArray && Array.isArray(r.include)) { include = r.include.every((sel, i) => { let ele = getElement(sel, document); return !!ele; }); } else include = getElement(r.include, document); if (!include) return false; } if (r.exclude) { let exclude; if (Array && Array.isArray && Array.isArray(r.exclude)) { exclude = !r.exclude.every((sel, i) => { let ele = getElement(sel, document); return !ele; }); } else exclude = getElement(r.exclude, document); if (exclude) return false; } return true; } ruleMatchReady(r) { let findIndex = 0; if (r.nextLink && r.nextLink !== 0) { let nextLinkSel = r.nextLink, nextLink; if (Array && Array.isArray && Array.isArray(nextLinkSel)) { nextLink = !nextLinkSel.every((sel, i) => { let ele = getElement(sel, document); if (ele) findIndex = i; return !ele; }); } else nextLink = getElement(nextLinkSel, document); if (!nextLink) return false; } if (r.pageElement) { let pageElementSel = r.pageElement, pageElement; if (Array && Array.isArray && Array.isArray(pageElementSel)) { pageElementSel = pageElementSel[findIndex]; } pageElement = getElement(pageElementSel, document); if (!pageElement) return false; } if (r.insert) { let insertSel = r.insert, insert; if (Array && Array.isArray && Array.isArray(insertSel)) { insertSel = insertSel[findIndex]; } insert = getElement(insertSel, document); if (!insert) return false; } //if (findIndex !== 0) nextIndex = findIndex; return true; } ruleMatch(r) { return this.ruleMatchPre(r) && this.ruleMatchReady(r); } scrollToShow(sel, doc) { let exclude = getElement(sel, doc); if (exclude) { var actualTop = exclude.offsetTop; var current = exclude.offsetParent; while (current !== null) { actualTop += current.offsetTop; current = current.offsetParent; } getBody(doc).scrollTop = 0; doc.documentElement.scrollTop = 0; let maxHeight = Math.max(getBody(doc).scrollHeight, doc.documentElement.scrollHeight); getBody(doc).scrollTop = actualTop - 10; doc.documentElement.scrollTop = actualTop - 10; setTimeout(() => { while (actualTop < maxHeight) { actualTop += 200; getBody(doc).scrollTop = actualTop; doc.documentElement.scrollTop = actualTop; } }, 0); return false; } return true; } waitElement(doc, selArr) { if (!selArr) selArr = this.curSiteRule.waitElement; if (!Array.isArray(selArr)) { selArr = [selArr]; } let includeSel = selArr[0].trim(), excludeSel; if (selArr.length === 2) { excludeSel = selArr[1].trim().replace(/^!/, ''); } else if (includeSel.indexOf('!') === 0) { excludeSel = includeSel.replace(/^!/, ''); includeSel = ''; } if (includeSel) { let include = getElement(includeSel, doc); if (!include) { return false; } } if (doc === document) return true; if (excludeSel) { if (!this.scrollToShow(excludeSel, doc)) { if (!loadingDiv.offsetParent && this.insert.parentNode) { this.insertElement(loadingDiv); } return false; } } return true; } runPageBar(pageBar) { if (this.curSiteRule.pageBar && this.curSiteRule.pageBar !== 0) { try { ((typeof this.curSiteRule.pageBar === 'function') ? this.curSiteRule.pageBar : Function("pageBar",'"use strict";' + this.curSiteRule.pageBar))(pageBar); } catch(e) { debug(e); } } } runWait(cb) { let checkEval = null, waitTime = 0; if (this.curSiteRule.waitElement) { checkEval = doc => { return this.waitElement(doc); }; } else if(this.curSiteRule.wait) { if (isNaN(this.curSiteRule.wait)) { try { checkEval = (typeof this.curSiteRule.wait === 'function') ? this.curSiteRule.wait : Function("doc", '"use strict";' + this.curSiteRule.wait); } catch(e) { debug(e); } } else { waitTime = this.curSiteRule.wait; } } cb(checkEval, waitTime); } findNoNext() { if (!this.curSiteRule || !this.curSiteRule.smart || this.curSiteRule.nextLink === 0 || this.possibleRule) return; let self = this; self.curSiteRule.nextLink = 0; self.smartRules = self.smartRules.filter(item => {return item && item.url !== self.curSiteRule.url;}); self.smartRules.unshift(self.curSiteRule); storage.setItem("smartRules", self.smartRules); } async getRule(callback) { var href = location.href.slice(0, 500); if(noRuleTest) { this.curSiteRule = {}; this.curSiteRule.url = href; this.curSiteRule.smart = true; callback(); return; } if (_unsafeWindow.pagetualRule) { this.curSiteRule = _unsafeWindow.pagetualRule; if (!this.curSiteRule.url) this.curSiteRule.url = "."; this.curSiteRule.isScript = true; } if (this.curSiteRule && this.curSiteRule.url && !this.curSiteRule.smart) { let urlReg = new RegExp(this.curSiteRule.url, "i"); if (urlReg.test(href) && this.ruleMatch(this.curSiteRule)) { return callback(); } } if (this.possibleRule) { let urlReg = new RegExp(this.possibleRule.url, "i"); if (urlReg.test(href) && this.ruleMatch(this.possibleRule)) { this.curSiteRule = this.possibleRule; debug(this.curSiteRule, 'Match'); return callback(); } } this.curSiteRule = {}; var self = this; function setRule(r) { if (self.preSiteRule) { href = location.href.slice(0, 500); let urlReg = new RegExp(self.preSiteRule.url, "i"); if (urlReg.test(href) && self.ruleMatch(self.preSiteRule)) { self.curSiteRule = self.preSiteRule; return callback(); } } if (r.from === 2) { delete r.autoLoadNum; delete r.history; delete r.sideController; if (r.pageBar === 0) delete r.pageBar; } if (!r.smart) { self.insert = null; self.curSiteRule = r; self.preSiteRule = r; if (r.enable !== 0) debug(r, 'Match'); } else if (!self.curSiteRule || !self.curSiteRule.smart) self.curSiteRule = r; callback(); } function checkRule(r) { if (r.from === 1 && r.url.length <= 13) return false; let urlReg = new RegExp(r.url, "i"); if (urlReg.test(href)) { if (!self.ruleMatchPre(r)) return false; if (r.url.length > 15 && r.from !== 1) { self.possibleRule = r; } if (r.waitElement) { let waitTime = 500; let checkReady = () => { setTimeout(() => { if (!self.waitElement(document, r.waitElement) || !self.ruleMatchReady(r)) { checkReady(); } else { setRule(r); } }, parseInt(waitTime)); }; checkReady(); debug(r, 'Wait for'); return true; } else if (r.wait) { let waitTime = 500, checkEval, maxCheckTimes = 50; if (isNaN(r.wait)) { try { checkEval = (typeof r.wait === 'function') ? r.wait : Function("doc",'"use strict";' + r.wait); } catch(e) { debug(e, 'Error when checkeval'); } } else { waitTime = r.wait; } let checkReady = () => { if (maxCheckTimes-- <= 0) { debug("Wait for rule ready but failed"); setRule(r); return; } setTimeout(() => { if (!self.ruleMatchReady(r) || (checkEval && !checkEval(document))) { checkReady(); } else { setRule(r); } }, parseInt(waitTime)); }; checkReady(); debug(r, 'Wait for'); return true; } if (r.pinUrl) { setRule(r); return true; } if (!self.ruleMatchReady(r)) { return false; } setRule(r); return true; } return false; } function checkHpRules() { for (let i in self.hpRules) { let rule = self.hpRules[i]; if (!rule || !rule.url || rule.smart) continue; if (checkRule(rule)) return true; } return false; } function checkCustomRules() { for (let i in self.customRules) { let rule = self.customRules[i]; if (!rule || !rule.url) continue; if (checkRule(rule)) return true; } return false; } if (rulesData.customFirst) { if (checkCustomRules()) return; await sleep(1); if (checkHpRules()) return; } else { if (checkHpRules()) return; await sleep(1); if (checkCustomRules()) return; } await sleep(1); for (let i in this.smartRules) { let rule = this.smartRules[i]; if (!rule || !rule.url || !rule.smart) continue; if (href == rule.url) { setRule(rule); return; } } let r = 0; async function searchByTime() { while (document.hidden) { await sleep(1000); } setTimeout(() => { let end = r + 20; end = end > self.rules.length ? self.rules.length : end; for (; r < end; r++) { let rule = self.rules[r]; if (checkRule(rule)) return; } if (end >= self.rules.length) { setRule({url: href, smart: true}); return; } else { searchByTime(); } }, 1); } searchByTime(); } addToHpRules(instead) { try { if (this.curSiteRule.isScript) return; if (!this.hpRules) this.hpRules = []; let url = this.curSiteRule && this.curSiteRule.url, self = this; let href = location.href.slice(0, 500); let matchedRules = this.hpRules.filter(rule => rule != self.curSiteRule && new RegExp(rule.url, "i").test(href) && self.ruleMatch(rule)); if (url) matchedRules.unshift(this.curSiteRule); matchedRules.sort((a, b) => { if ((a.include || a.exclude) && (!b.include && !b.exclude)) { return -1; } else if ((b.include || b.exclude) && (!a.include && !a.exclude)) { return 1; } else { if ((a.nextLink || a.pageElement) && (!b.nextLink && !b.pageElement)) { return -1; } else if ((b.nextLink || b.pageElement) && (!a.nextLink && !a.pageElement)) { return 1; } else { if (a.url.length > b.url.length) { return -1; } else if (a.url.length < b.url.length) { return 1; } else { return 0; } } } }); this.hpRules = this.hpRules.filter(item => { return item && !matchedRules.find(rule => item.url == rule.url && JSON.stringify(item) == JSON.stringify(rule)); }); if (instead) { if (url) { this.hpRules.unshift(this.curSiteRule); matchedRules = []; } } else { this.hpRules = matchedRules.concat(this.hpRules); } if (matchedRules && matchedRules.length) this.curSiteRule = matchedRules[0]; if (this.hpRules.length > 30) { this.hpRules.pop(); } if (!rulesData.sort) rulesData.sort = [1]; this.hpRules.sort((a, b) => { let aSort = -1, bSort = -1; for (let s = 0; s < rulesData.sort.length; s++) { if (a.from == rulesData.sort[s]) aSort = s; if (b.from == rulesData.sort[s]) bSort = s; if (aSort > -1 && bSort > -1) break; } return aSort - bSort; }); storage.setItem("hpRules", this.hpRules); } catch (e) { debug(e); } } replaceElement(doc) { if (!doc || doc === document) return; let replaceElementSel = this.curSiteRule.replaceElement; if (replaceElementSel) { if (!Array.isArray(replaceElementSel)) { replaceElementSel = [replaceElementSel]; } replaceElementSel.forEach(sel => { let pageEles = getAllElements(sel, document); let replaceEles = getAllElements(sel, doc); for (let i = 0; i < pageEles.length; i++) { let replaceEle = replaceEles[i]; let pageEle = pageEles[i]; if (replaceEle) { pageEle.parentNode.replaceChild(replaceEle.cloneNode(true), pageEle); } else break; } }); } } refreshByClick() { let refreshByClickSel = this.curSiteRule.refreshByClick; if (refreshByClickSel) { let self = this; if (!this.refreshByClickHandler) { this.refreshByClickHandler = e => { if (!self.refreshing) { self.refreshing = true; setTimeout(() => { self.refreshing = false; let checkEles = getAllElements(refreshByClickSel, document); for (let i = 0; i < checkEles.length; i++) { let curEle = checkEles[i]; if (curEle === e.target || curEle.contains(e.target)) { urlChanged = true; isPause = true; if (!ruleParser.nextLinkHref) isLoading = false; break; } } }, 300); } } } document.removeEventListener("click", this.refreshByClickHandler); document.addEventListener("click", this.refreshByClickHandler); } } getValidSize(ele, win) { if (!win) return {h: 0, w: 0}; let eleStyle = win.getComputedStyle(ele); if (!ele.offsetParent && (eleStyle.position !== "fixed" || eleStyle.opacity === 0)) { return {h: 0, w: 0}; } let h = ele.scrollHeight; if (eleStyle.overflow === "hidden") { h = ele.offsetHeight; } if (h === 0 && ele.parentNode && ele.parentNode.children.length === 1) { h = ele.parentNode.scrollHeight; } while (h === 0 && ele.children && ele.children.length === 1) { ele = ele.children[0]; h = ele.scrollHeight; } if (h === 0 && ele.children && ele.children.length) { let maxChildSize = {h: 0}, self = this; [].forEach.call(ele.children, el => { let childSize = self.getValidSize(el, win); if (childSize.h > maxChildSize.h) maxChildSize = childSize; }); if (maxChildSize.h !== 0) return maxChildSize; } const maxNum = 2147483647; let moreChild = ele.children[0], minOffsetTop = maxNum; while (moreChild) { if ((moreChild.offsetParent === ele || moreChild.offsetParent === ele.offsetParent)) { let curOffsetTop = moreChild.offsetParent === ele.offsetParent ? moreChild.offsetTop - ele.offsetTop : moreChild.offsetTop; if (curOffsetTop < minOffsetTop) { minOffsetTop = curOffsetTop; } } moreChild = moreChild.nextElementSibling; } if (h && minOffsetTop !== maxNum && minOffsetTop > 0) { h -= minOffsetTop; } return {h: h, w: parseInt(ele.offsetWidth || ele.scrollWidth)}; } checkTargetChildren(ele, curWin, articleNum, curHeight) { let pf = false; if (ele.parentNode) { let paStyle = curWin.getComputedStyle(ele.parentNode); let paDisplay = paStyle.display; let paOverflow = paStyle.overflow; pf = (paDisplay.indexOf('flex') !== -1 && paStyle.flexDirection === "row" && paStyle.flexWrap !== "wrap") || compareNodeName(ele.parentNode, ["ul"]) || paDisplay.indexOf('grid') !== -1 || paOverflow === "hidden"; } let curStyle = curWin.getComputedStyle(ele); if (ele.children.length > 1) { if (articleNum > 1) { return ">article"; } else { let hasText = false; for (let i in ele.childNodes) { let child = ele.childNodes[i]; if (child.nodeType === 3 && child.nodeValue.trim() !== '') { return ""; } } if (!hasText) { let gridArea = curStyle.gridArea; if (gridArea && gridArea !== "auto" && gridArea !== "auto / auto / auto / auto") { return ">*"; } else { let middleChild = ele.children[parseInt(ele.children.length / 2)]; if ((curStyle.display === 'flex' && curStyle.flexDirection === "row" && curStyle.flexWrap !== "wrap") || (curStyle.float === "none" && rulesData.opacity !== 0 && !pf)) { return ""; } else if ((middleChild.style && middleChild.style.position === "absolute" && middleChild.style.left && middleChild.style.top) || compareNodeName(ele, ["ul"]) || curHeight === 0) { return ""; } else { return ">*"; } } } } } else if (pf || curStyle.position === "absolute") { return ">*"; } return ""; } getPageElement(doc, curWin, dontFind) { if (doc === document && this.docPageElement && document.documentElement.contains(this.docPageElement[0])) { return this.docPageElement; } let pageElement = null; let self = this; let body = getBody(doc); if (this.curSiteRule.pageElement) { let pageElementSel = this.curSiteRule.pageElement; if (Array && Array.isArray && Array.isArray(pageElementSel)) { pageElementSel = pageElementSel[nextIndex < pageElementSel.length ? nextIndex : 0]; } pageElement = getAllElements(pageElementSel, doc); if (this.curSiteRule.smart && (!pageElement || pageElement.length === 0)) { const childSelMatch = />\s*\*$/; const targetChild = childSelMatch.test(pageElementSel); if (targetChild) pageElementSel = pageElementSel.replace(childSelMatch, ""); let pageElementSelSplit = pageElementSel.split(">"); while(pageElementSelSplit && pageElementSelSplit.length > 5) { pageElementSelSplit.shift(); let tempSel = pageElementSelSplit.join(">"); pageElement = getAllElements(tempSel, doc); if (pageElement && pageElement.length === 1) { if (targetChild) { pageElement = pageElement.children; } this.curSiteRule.pageElement = tempSel + (targetChild ? ">*" : ""); break; } } if (!pageElement || pageElement.length === 0) { pageElementSel = pageElementSel.replace(/:nth-of-type\(\d+\)/g, ""); pageElement = getAllElements(pageElementSel, doc); if (pageElement && pageElement.length === 1) { if (targetChild) { pageElement = pageElement.children; } this.curSiteRule.pageElement = pageElementSel + (targetChild ? ">*" : ""); } } } if (this.curSiteRule.smart && pageElement && pageElement.length && curWin && curWin !== _unsafeWindow) { let parent = pageElement[0].parentNode; let loading = parent.querySelector('[class*=loading]'); if (loading && loading.offsetParent && loading.offsetHeight > parent.offsetHeight>>2) { pageElement = null; } } } if (pageElement && pageElement.length === 1 && pageElement[0].style.display === 'none') { pageElement = [body]; } if (this.curSiteRule.smart && pageElement && pageElement.length > 0 && compareNodeName(pageElement[0], ["tr"])) { let mainTr = this.insert.parentNode.querySelectorAll('tr'), mainTdNum = 0, newTdNum = 0; mainTr = mainTr[mainTr.length - 1]; [].forEach.call(mainTr.children, el => { if (compareNodeName(el, ["td", "th"])) { mainTdNum += el.colSpan || 1; } }); [].forEach.call(pageElement[0].children, el => { if (compareNodeName(el, ["td", "th"])) { newTdNum += el.colSpan || 1; } }); if (mainTdNum !== newTdNum) { this.curSiteRule.pageElement = this.curSiteRule.pageElement.replace(/> *table.*/, ">table"); this.getInsert(true); return this.getPageElement(doc, curWin, dontFind); } } if ((this.curSiteRule.smart || !this.curSiteRule.pageElement) && (!pageElement || pageElement.length == 0) && curWin && !dontFind) { if (!body) return null; let bodyHeight = parseInt(body.offsetHeight || body.scrollHeight); let curHeight = bodyHeight, curWidth = 0; let windowHeight = window.innerHeight || document.documentElement.clientHeight; let windowWidth = window.innerWidth || document.documentElement.clientWidth; let needCheckNext = (doc == document && this.initNext), nextLeftPos = 0; if (needCheckNext && this.initNext.getBoundingClientRect) { nextLeftPos = this.initNext.getBoundingClientRect().left; } function checkElement(ele) { if (compareNodeName(ele, ["pre", "code"])) { self.curSiteRule.pageElement = geneSelector(ele.parentNode); debug(self.curSiteRule.pageElement, 'Page element'); return [ele.parentNode]; } if (ele.children && ele.children.length === 1) { let hasText = false; for (let i in ele.childNodes) { let child = ele.childNodes[i]; if (child.nodeType === 3 && child.nodeValue.trim() !== '') { hasText = true; break; } } if (!hasText) { ele = ele.children[0]; let validSize = self.getValidSize(ele, curWin); curHeight = validSize.h; curWidth = validSize.w; } } if (compareNodeName(ele, ["picture"]) || !ele.innerText || ele.innerText.trim() === '') { self.curSiteRule.pageElement = geneSelector(ele.parentNode) + ">" + ele.nodeName.toLowerCase(); debug(self.curSiteRule.pageElement, 'Page element'); let eles = []; for (let i = 0; i < ele.parentNode.children.length; i++) { let curNode = ele.parentNode.children[i]; if (curNode.nodeName === ele.nodeName && curNode.id === ele.id && curNode.className === ele.className) { eles.push(curNode); } } return eles; } if (compareNodeName(ele, ["form"]) && ele.parentNode !== getBody(document)) { self.curSiteRule.pageElement = geneSelector(ele) + ">*"; debug(self.curSiteRule.pageElement, 'Page element'); return ele.children; } if (ele.children.length === 0 && !self.curSiteRule.pageElement) { if (compareNodeName(ele.parentNode, ["p"])) ele = ele.parentNode; self.curSiteRule.pageElement = geneSelector(ele.parentNode) + ">" + ele.nodeName.toLowerCase(); debug(self.curSiteRule.pageElement, 'Page element'); return getAllElements(self.curSiteRule.pageElement, doc); } let i, minHeight = curHeight * 0.52, curMaxEle = null, curMaxArea = 0, minWidth = Math.min(curWidth * 0.38, 500); let isHori, preOffsetTop = -1; let articleNum = 0; for (i = 0; i < ele.children.length; i++) { let curNode = ele.children[i]; if (ele !== body && /^H\d$/i.test(curNode.nodeName) && curNode.offsetParent) { curMaxEle = null; break; } if (compareNodeName(curNode, ["canvas", "nav"])) continue; let curStyle = curWin.getComputedStyle(curNode); if (!curNode.offsetParent && (curStyle.position !== "fixed" || curStyle.opacity === 0)) { continue; } if (!compareNodeName(curNode, ["img"]) && curNode.querySelector('img') === null && /^\s*$/.test(curNode.innerText)) continue; if (needCheckNext && !curNode.contains(self.initNext) && getElementTop(curNode) > windowHeight) { continue; } if (compareNodeName(curNode, ["article"])) articleNum++; let validSize = self.getValidSize(curNode, curWin); let h = validSize.h; let w = validSize.w; if (isNaN(h) || isNaN(w)) continue; isHori = Math.abs(preOffsetTop - curNode.offsetTop) <= 20 ? true : (preOffsetTop === -1 && curNode.nextElementSibling && curNode.nextElementSibling.offsetTop === curNode.offsetTop); if (isHori && h <= 50) continue; /*if (isHori && nextLeftPos && curMaxEle && curWidth > 500 && curHeight > 500) { let curRect = curNode.getBoundingClientRect(); if (curRect.left > windowWidth>>2 && curRect.left <= nextLeftPos && curRect.right > nextLeftPos) { continue; } }*/ let a = h * w, moreChild = curNode.children[0]; while (moreChild) { let validSize = self.getValidSize(moreChild, curWin); let ch = validSize.h; let cw = validSize.w; if (h < ch) { h = ch; } if (moreChild.innerText !== "" && ch && cw) { a += ch * cw; } moreChild = moreChild.nextElementSibling; } let isMax = false; if (isHori) { if (curMaxEle) { if (w > curWidth && (windowWidth>>1) > curWidth) { isMax = true; } else if (w + 300 > curWidth && a > curMaxArea) { isMax = true; } } } else { if (curMaxEle && curMaxEle.offsetParent === curNode.offsetParent && curMaxEle.offsetTop === curNode.offsetTop) { if (curMaxArea * 2 > a) continue; } isMax = curMaxArea < a; } if (curMaxEle === null || isMax) { if (isHori) { if (w < minWidth) { continue; } } if (h < minHeight) { if (!needCheckNext || h < (windowHeight>>1) || !ele.contains(self.initNext)) { continue; } } curHeight = h; curMaxArea = a; curWidth = w; curMaxEle = curNode; preOffsetTop = curNode.offsetTop; } } if (curMaxEle && curHeight / bodyHeight <= 0.18) { let article = doc.querySelectorAll(mainSel); if (article && article.length === 1) { article = article[0]; let childrenEnd = self.checkTargetChildren(article, curWin, articleNum, curHeight); self.curSiteRule.pageElement = article.nodeName.toLowerCase() + (article.id ? "#" + article.id : "") + (article.className ? "." + article.className.replace(/ /g, ".") : "") + childrenEnd; debug(self.curSiteRule.pageElement, 'Page element'); return childrenEnd ? article.children : [article]; } curMaxEle = null; } if (curMaxEle) { let sameClassNum = 0, hasDifferent = false; if (curMaxEle.className) { for(i = 0; i < ele.children.length; i++) { let curNode = ele.children[i]; if (curMaxEle !== curNode && curNode.style.display !== 'none' && curMaxEle.className === curNode.className && curMaxEle.nodeName === curNode.nodeName){ sameClassNum++; } else if (curMaxEle.className !== curNode.className) { hasDifferent = true; } } } if (sameClassNum < 2 || (sameClassNum < 5 && hasDifferent)) { return checkElement(curMaxEle); } } if (ele.nodeName === "APP-ROOT") { isPause = true; debug(ele, "Angular root"); return null; } while (compareNodeName(ele, ["p", "br", "td"])) { ele = ele.parentNode; } if (compareNodeName(ele, ["tbody"])) { self.curSiteRule.pageElement = geneSelector(ele) + ">*"; if (ele.children.length > 0 && ele.children[0].querySelector("th")) { self.curSiteRule.pageElement += ":not(:first-child)"; } debug(self.curSiteRule.pageElement, 'Page element'); return ele.children; } let imgs = ele.querySelectorAll('img'); if (imgs.length === 1) { let img = imgs[0]; if (img.offsetWidth > ele.offsetWidth / 3 * 2 && img.offsetHeight > ele.offsetHeight / 2) { ele = img; } } self.curSiteRule.pageElement = geneSelector(ele); let childrenEnd = self.checkTargetChildren(ele, curWin, articleNum, curHeight); if (childrenEnd) { self.curSiteRule.pageElement += childrenEnd; ele = ele.children; } else { ele = [ele]; } debug(self.curSiteRule.pageElement, 'Page element'); return ele; } pageElement = checkElement(body); if (pageElement && pageElement.length > 0 && self.initNext) { let lastBottom = getElementBottom(pageElement[pageElement.length - 1]); if (lastBottom && getElementTop(self.initNext) - lastBottom > 1000) { debug("Stop as too long between next & page element"); isPause = true; pageElement = []; sideController.remove(); } else { if (pageElement.length === 1 && compareNodeName(pageElement[0], ["img"])) { self.curSiteRule.pageBar = 0; } } } //if(pageElement)this.saveCurSiteRule(); } if (doc !== document) { this.setPageElementCss(pageElement); this.lazyImgAction(pageElement, doc); this.filterEles(doc, pageElement); } else if (!this.docPageElement) { this.setPageElementCss(pageElement, true); this.docPageElement = pageElement; this.filterEles(doc, pageElement); if (this.nextLinkHref) { this.openInNewTab(pageElement); } } /* if (this.curSiteRule.smart && pageElement && pageElement.length > 0) { let targetEle = pageElement.length > 1 ? pageElement[0].parentNode : pageElement[0]; let video = targetEle.querySelector("video,iframe[id*=play],[id*=play]>iframe,iframe[src*=player],iframe[src*=m3u8]"); if (video && doc == document) { if (video.offsetParent && video.name != 'pagetual-iframe') { let scrollWidth = video.scrollWidth || video.offsetWidth; let scrollHeight = video.scrollHeight || video.offsetHeight; if (/IFRAME/i.test(video.nodeName)) { } else if (scrollWidth > 100 && scrollHeight > 100) { let winWidth = window.innerWidth || document.documentElement.clientWidth; let winHeight = window.innerHeight || document.documentElement.clientHeight; if (scrollWidth > winWidth>>1 && scrollHeight > winHeight>>1) { debug("Disable when large media found"); } else { video = null; } } else { video = null; } } else { video = null; } } if (video) { isPause = true; this.clearAddedElements(); return null; } } */ return pageElement; } showAddedElements() { if (!this.addedElementsIsHide) return; if (this.addedElePool && this.addedElePool.length) { this.addedElePool.forEach(ele => { ele.classList && ele.classList.remove("pagetual-hide"); }); } this.addedElementsIsHide = false; } hideAddedElements() { if (this.addedElePool && this.addedElePool.length) { this.addedElePool.forEach(ele => { ele.classList && ele.classList.add("pagetual-hide"); }); } this.addedElementsIsHide = true; } toggleAddedElements() { if (this.addedElementsIsHide) { this.showAddedElements(); } else { this.hideAddedElements(); } } changeVisibility() { let contentVisibility = this.curSiteRule.contentVisibility || rulesData.contentVisibility; if (!contentVisibility) return; if (!this.changingVisibility) { clearTimeout(this.changeVisibilityTimer); this.changeVisibilityTimer = setTimeout(() => { this.changingVisibility = true; this.changeVisibility(); }, 300); return; } this.changingVisibility = false; if (!this.visibilityItems || !this.visibilityItems.length || this.visibleIndex < 0) return; let tempIndex = this.visibleIndex, findVisible = false, lastVisible = 0; let viewPortHeight = window.innerHeight || document.documentElement.clientHeight || getBody(document).clientHeight; let checkItem = this.visibilityItems[tempIndex]; while(checkItem) { if (checkItem.offsetParent) { if (!checkItem.style.containIntrinsicSize) return; let clientRect = checkItem.getBoundingClientRect(); let top = clientRect && clientRect.top; let bottom = clientRect && clientRect.bottom; if (bottom > 0 && top < viewPortHeight) { if (!findVisible) { findVisible = true; lastVisible = tempIndex; } checkItem.style.contentVisibility = "visible"; } else { if (top < viewPortHeight && checkItem.style.contentVisibility === "auto") { break; } else checkItem.style.contentVisibility = "auto"; } } if (tempIndex === 0) break; tempIndex--; checkItem = this.visibilityItems[tempIndex]; } tempIndex = this.visibleIndex + 1; if (findVisible) { this.visibleIndex = lastVisible; } checkItem = this.visibilityItems[tempIndex]; while(checkItem) { if (checkItem.offsetParent) { if (!checkItem.style.containIntrinsicSize) return; let clientRect = checkItem.getBoundingClientRect(); let top = clientRect && clientRect.top; let bottom = clientRect && clientRect.bottom; if (bottom > 0 && top < viewPortHeight) { findVisible = true; lastVisible = tempIndex; checkItem.style.contentVisibility = "visible"; } else { if (findVisible && checkItem.style.contentVisibility === "auto") { break; } else checkItem.style.contentVisibility = "auto"; } } if (tempIndex === this.visibilityItems.length - 1) break; tempIndex++; checkItem = this.visibilityItems[tempIndex]; } if (findVisible) { this.visibleIndex = lastVisible; } } setPageElementCss(pageElement, init) { let self = this; if (pageElement && pageElement.length > 0) { let pageElementCss = this.curSiteRule.pageElementCss || rulesData.pageElementCss; if (!pageElementCss && init && !this.nextLinkHref) return; let contentVisibility = this.curSiteRule.contentVisibility || rulesData.contentVisibility; if (!contentVisibility && !pageElementCss) return; [].forEach.call(pageElement, (ele, i) => { if (!compareNodeName(ele, ["link", "meta", "style", "script"])) { if (pageElementCss) { if (pageElementCss !== '0' && !ele.dataset.pagetualPageElement) { ele.style.cssText = (ele.style.cssText || '') + pageElementCss; ele.dataset.pagetualPageElement = 1; } } if (contentVisibility) { ele.style.containIntrinsicSize = `auto ${ele.offsetHeight || self.preVisibleHeight || 100}px`; if (ele.style.containIntrinsicSize) { if (ele.offsetHeight) self.preVisibleHeight = ele.offsetHeight; if (init) { ele.style.contentVisibility = "visible"; self.visibilityItems.push(ele); self.visibleIndex++; } else { ele.style.contentVisibility = emuIframe ? "visible" : "auto"; } } } } }); } } clearAddedElements() { if (this.addedElePool && this.addedElePool.length) { this.addedElePool.forEach(ele => { if (ele.parentNode) ele.parentNode.removeChild(ele); }); } this.addedElePool = []; sideController.remove(); } linkHasHref(link) { return link.href && link.href.replace && !this.hrefIsJs(link.href); } hrefIsJs(href) { return /^(javascript|#|$)/.test(href.trim().replace("#p{", "").replace(location.href, "")); } async querySelectorList(source, list) { for (let i = 0; i < list.length; i++) { await sleep(1); let sel = list[i]; let result = source.querySelectorAll(sel); if (result.length > 0) return result[result.length - 1]; } return null; } verifyElement(ele) { let verifyHandler = e => { if (e.nodeType == 9) return true; if (e.style.display === "none" || e.getAttribute("aria-disabled") === "true") { return false; } if (e.className) { if (/slick|slide|gallery|disabled$/i.test(e.className)) { return false; } else if (e.classList) { if (e.classList.contains('disabled') || e.classList.contains('active')) { return false; } } } let ariaLabel = e.getAttribute("aria-label"); if (ariaLabel && /slick|slide|gallery/i.test(ariaLabel)) return false; return true; }; if (!ele) return false; let i = 0; while (ele && i++ < 6) { if (!verifyHandler(ele)) return false; ele = ele.parentNode; } return true; } async getPage(doc, exist) { if (document.documentElement.className.indexOf('discourse') !== -1) return {}; let body = getBody(doc); let canSave = false;//發現頁碼選擇器在其他頁對不上,還是別保存了 let url = this.curUrl.slice(0, 250).replace("index.php?", "?"); let _url = url.replace(/\.s?html?$/i, "").toLowerCase(); let pageNum = 1, preStr = "", afterStr = ""; let pageTwoReg = /^[\/\?&]?[_-]?(p|page)?=?\/?2(\/[^\/]*$|\?|&|$)/i; let pageMatch1 = url.match(/(.*[\?&]p(?:age)?=)(\d+)($|[#&].*)/i); let doubtTextReg = /^\s*(»|>>)\s*$/; if (pageMatch1) { preStr = pageMatch1[1]; pageNum = parseInt(pageMatch1[2]); afterStr = pageMatch1[3]; } else { let pageMatch2 = url.match(/(.*[a-z\/\-_](?:p|page)?\/?)(\d+)(\.s?html?$|\/?$)/i); if (pageMatch2) { preStr = pageMatch2[1]; pageNum = parseInt(pageMatch2[2]); afterStr = pageMatch2[3]; if (/^\/?$/.test(afterStr) && !/(p(age)?|_|\-|\/)$/.test(preStr)) { preStr = ""; afterStr = ""; } } } if (pageNum > 999) { pageNum = 1; preStr = ""; afterStr = ""; } let curPage = doc, i, cur, jsNext; let next1, next2, next3, next4, nextJs1, nextJs2; let selectorList = [ ".page-next>a", "a.next_page", "#next_page", ".curPage+a", ".nextPage", ".pagination-next>a", "a[data-pagination=next]", ".pageButtonsCurrent+a", "a[class*=nextpage]", "li.page-current+li>a", "[class^=pag] a[rel=next]", "[class^=Pag] [aria-label=next]", "[aria-label='Next page']", "[aria-label='next page']", ".pagination-nav__item--next>a", "a.pageright", ".page-numbers.current+a", "a.page-numbers.next", "body [class*=pagination] li.active+li>a", "body [class^=pag] .current+a", ".page_current+a", "input[value='next']", "input[value='Next page']", "input[value='下一页']", "input[value='下一頁']", "a#pb_next", "a#rightFix", "a#btnPreGn", "a.page-next", "a.pages-next", "a.page.right", ".paging>.active+.item", "a#next", ".next>a", ".next>button", "a[alt=next]", ".pg_area>em+a", "button.next:not([disabled])", ".btn_next:not([disabled])", ".btn-next:not([disabled])", "a#linkNext", "a[class*=page__next]", "[class*=pager]>a.next", "[class*=pagination-next]>a" ]; let next = await this.querySelectorList(body, selectorList); if (!next) { await sleep(1); let nexts = body.querySelectorAll("a.next"); const prevReg = /^\s*([上前首尾]|previous)/i; for (i = 0; i < nexts.length; i++) { let n = nexts[i]; if (this.verifyElement(n) && this.linkHasHref(n) && !prevReg.test(n.innerText.trim())) { next = n; break; } } } if (next) { let innerText = next.innerText; let isJs = !this.linkHasHref(next); if (innerText && nextTextReg2.test(innerText.trim())) { if (isJs) { if (this.verifyElement(next)) { nextJs2 = next; } } else { next2 = next; } next = null; } else { if (isJs) { if (this.verifyElement(next)) { jsNext = next; } next = null; } } } if (!next) { await sleep(1); next = body.querySelector("a.curr+a") || body.querySelector("div.wp-pagenavi>span.current+a,div.page-nav>span.current+a,div.article-paging>span+a") || body.querySelector(".number>ul>li.active+li>a"); } if (!next) { await sleep(1); let pageDiv = body.querySelector(".pages>ul,.page_no>ul"); if (pageDiv) { cur = pageDiv.querySelector("li>b,li>strong"); if (cur) next = cur.parentNode.nextElementSibling; if (next) next = next.querySelector("a"); } } if (!next) { await sleep(1); next = body.querySelector(".pages>a[href='javascript:;']+a"); if (next && (next.href === "javascript:;" || next.getAttribute("href") === "#")) next = null; } if (!next) { await sleep(1); let pageDiv = body.querySelector(".pagination"); if (pageDiv) { cur = pageDiv.querySelector("[class*=current]"); if (cur) next = cur.parentNode.nextElementSibling; if (next) next = next.querySelector("a"); } } if (!next) { await sleep(1); let pageDivs = body.querySelectorAll("[class*=pagination],[class*=Pagination]"); if (pageDivs && pageDivs.length) { for (let i = pageDivs.length - 1; i >= 0; i--) { let p = pageDivs[i]; if (/(next\s*(»|>>|>|›|→|❯)?|>|▶|>|›|→|❯)/i.test(p.title || p.value || '')) { next = p.querySelector('a') || p; break; } else if (/^(next\s*(»|>>|>|›|→|❯)?|>|▶|>|›|→|❯)$/i.test((p.innerText || '').trim())) { next = p.querySelector('a') || p; break; } } if (next && !this.linkHasHref(next)) { if (!jsNext) jsNext = next; next = null; } } } if (!next) { let isApp = !!body.querySelector("main#app"); let aTags = body.querySelectorAll("a,button,[type='button']"); for (i = aTags.length - 1; i >= 0; i--) { if (next1) break; if (i % 100 === 0) { await sleep(1); } let aTag = aTags[i]; let title = aTag.title || ""; let value = aTag.value || ""; let innerText = aTag.innerText || ""; if (innerText === "" || (title !== "" && title.length < innerText.length)) { innerText = title; } if (innerText === "" || (value !== "" && value.length < innerText.length)) { innerText = value; } if (innerText !== "") { if (innerText === "§") continue; innerText = innerText.trim(); if (innerText.length > 80) continue; } if (!this.verifyElement(aTag)) continue; if (aTag.dataset && aTag.dataset.preview) continue; let availableHref = aTag.href && aTag.href.length < 250 && /^http/.test(aTag.href); if (availableHref && /next\-?(page)?$/i.test(aTag.href)) continue; if (compareNodeName(aTag.parentNode, ["blockquote"])) continue; if (aTag.previousElementSibling && /\b(play|volume)\b/.test(aTag.previousElementSibling.className)) continue; if (aTag.nextElementSibling && /\b(play|volume)\b/.test(aTag.nextElementSibling.className)) continue; let isJs = !this.linkHasHref(aTag); if (exist && isJs && !aTag.offsetParent) continue; if (innerText) { innerText = innerText.split(/\n/)[0].replace(/ /g, ''); if (isJs && /^(»|>>|>|›|→|❯)$/.test(innerText)) continue; if (innerText && innerText.length <= 25) { if (!next1) { if (nextTextReg1.test(innerText)) { if (isJs) { if (!nextJs1) nextJs1 = aTag; } else { next1 = aTag; } } } if (!next4) { if (!next2) { if (nextTextReg2.test(innerText) || /nextpage|pager\-older/i.test(aTag.className) || /^(»|>>)$/.test(innerText)) { if (isJs) { if (!nextJs2) nextJs2 = aTag; } else { next2 = aTag; } } } if (!isApp && !next3 && !isJs) { if (/^(next\s*(»|>>|>|›|→|❯)?|>|▶|>|›|→|❯)$/i.test(aTag.textContent) && aTag.parentNode.hasAttribute && !aTag.parentNode.hasAttribute("jsaction")) { next3 = aTag; } } } } } if (isJs) continue; if (!next4) { let prevEle = aTag.previousElementSibling; if (prevEle && compareNodeName(prevEle, ["b", "span", "strong"])) { if (/^\d+$/.test(aTag.innerText.trim()) && /^\d+$/.test(prevEle.innerText.trim()) && parseInt(aTag.innerText) === parseInt(prevEle.innerText) + 1) { next4 = aTag; } } } if (urlWillChange) continue; if (!next4 && availableHref) { if (aTag.href.indexOf(location.hostname) === -1) continue; let _aHref = aTag.href.replace("?&", "?").replace("index.php?", "?").toLowerCase(); if (preStr || afterStr) { let _aHrefTrim = _aHref; if (preStr) _aHrefTrim = _aHrefTrim.replace(preStr, ""); if (afterStr) _aHrefTrim = _aHrefTrim.replace(afterStr, ""); if (_aHrefTrim == pageNum + 1) { next4 = aTag; } } else if (this.curUrl !== aTag.href) { _aHref = _aHref.replace(/\.s?html?$/i, ""); if (_aHref.indexOf(_url) !== -1) { let pageTwoMatch = _aHref.replace(_url, "").match(pageTwoReg); if (pageTwoMatch) { afterStr = pageTwoMatch[2]; next4 = aTag; } } } if (next4 && !/page/.test(next4.href)) { let curHref = next4.getAttribute("href"); let curPageReg = new RegExp("(.*)" + (pageNum + 1) + afterStr.replace(/([\.\?])/g, '\\$1')); let otherPageHref = curHref.replace(curPageReg, `$1${pageNum}${afterStr}`); let otherPageEle = body.querySelector(`a[href='${otherPageHref}']`); if (!otherPageEle) { otherPageHref = curHref.replace(curPageReg, `$1${pageNum + 2}${afterStr}`); otherPageEle = body.querySelector(`a[href='${otherPageHref}']`); } if (otherPageEle) { let parent = otherPageEle.parentNode; if (parent && parent.parentNode) { parent = parent.parentNode; if (parent.parentNode) { parent = parent.parentNode; } } if (parent && parent.contains(next4) && !/^\d+$/.test(otherPageEle.innerText.trim())) { next4 = null; } } } } } if (next2 && doubtTextReg.test(next2.innerText)) { next2 = this.verifyNext(next2, doc); } if (nextJs2 && doubtTextReg.test(nextJs2.innerText)) { nextJs2 = this.verifyNext(nextJs2, doc); } if (next3) { next3 = this.verifyNext(next3, doc); } } if (!next) next = next1 || next4 || next3 || next2; if (!next && location.pathname !== "/") { next = jsNext || nextJs1 || nextJs2; if (next && next.parentNode.className && next.parentNode.className.indexOf && next.parentNode.className.indexOf('tab') !== -1) next = null; } if (next && next.classList && (next.classList.contains("results-more") || next.classList.contains("no"))) next = null; if (next && next.hasAttribute && next.hasAttribute("disabled")) next = null; if (next && next.parentNode.href && compareNodeName(next.parentNode, ["a"])) next = next.parentNode; return {next:next, canSave:canSave}; } verifyNext(next, doc) { if (!next) return null; if (next.previousElementSibling && compareNodeName(next.previousElementSibling, ["br"])) return null; let eles = []; if (next.innerText && next.innerText.indexOf("\n") === -1) { eles = getAllElements(`//${next.nodeName}[text()='${next.innerText}']`, doc); } if (eles.length >= 2 && eles[0].href !== eles[1].href) { next = null; } else if (doc === document) { let left = getElementLeft(next); if (left < 20 || (document.documentElement.scrollWidth > 500 && left < 250)) { next = null; } else { let top = getElementTop(next); if (top < 20 || (left < window.innerWidth / 3 && top < window.innerHeight / 3)) { next = null; } else { let bottom = top + next.offsetHeight || 0; let scrollH = Math.max(document.documentElement.scrollHeight, getBody(document).scrollHeight); if (scrollH - bottom < 10) next = null; } } } return next; } canonicalUri(src) { if (!src) { return ""; } if (src.charAt(0) === "#") return this.curUrl + src; if (src.charAt(0) === "?") return this.curUrl.replace(/^([^\?#]+).*/, "$1" + src); let origin = location.protocol + '//' + location.host; let url = this.basePath || origin; url = url.replace(/(\?|#).*/, ""); if (/https?:\/\/[^\/]+$/.test(url)) url = url + '/'; if (url.indexOf("http") !== 0) url = origin + url; var root_page = /^[^\?#]*\//.exec(url)[0], root_domain = /^\w+\:\/\/\/?[^\/]+/.exec(root_page)[0], absolute_regex = /^\w+\:\/\//; this.updateUrl = false; while (src.indexOf("../") === 0) { src = src.substr(3); root_page = root_page.replace(/\/[^\/]+\/$/, "/"); this.updateUrl = true; } src = src.replace(/\.\//, ""); if (/^\/\/\/?/.test(src)) { src = location.protocol + src; } return (absolute_regex.test(src) ? src : ((src.charAt(0) === "/" ? root_domain : root_page) + src)); } getLinkByPage(url, pageNum) { if (!url) return null; if (this.curSiteRule.pageNum) { let result = this.curSiteRule.pageNum; let strMatch = result.match(/\{.*?}/); if (!strMatch) return null; let urlReg = new RegExp("(" + result.replace(strMatch[0], ")\\d+(") + ")", "i"); let code = strMatch[0].replace(/^{/, "").replace(/}$/, "").replace(/\$p/g, pageNum); if (code === pageNum) { result = url.replace(urlReg, "$1" + code + "$2"); } else { try { code = Function('"use strict";return ' + code)(); if (code && code % 1 == 0) { result = url.replace(urlReg, "$1" + code + "$2"); } else return null; } catch(e) { debug(e); } } if (result != url) { return result; } } return url.replace(/([&\/\?](p=|page[=\/_-]?))\d+/i, "$1" + pageNum).replace(/([_-])\d+\./i, "$1" + pageNum + "."); } getPageNumFromUrl(url, defaultPage) { if (!url) return defaultPage; if (this.curSiteRule.pageNum) { let result = this.curSiteRule.pageNum; let strMatch = result.match(/\{.*?}/); if (!strMatch) return defaultPage; let urlReg = new RegExp(".*" + result.replace(strMatch[0], "(\\d+)") + ".*", "i"); let curShowNum = url.replace(urlReg, "$1"); if (curShowNum !== url) { let code = strMatch[0].replace(/^{/, "").replace(/}$/, ""); if (code === "$p") { return curShowNum; } else { try { let page1 = parseInt(Function('"use strict";return ' + code.replace("$p", "0"))()); let page2 = parseInt(Function('"use strict";return ' + code.replace("$p", "1"))()); let numGap = page2 - page1; let _page = (parseInt(curShowNum) - page1) / numGap; if (_page && _page % 1 === 0) return _page; else { this.curSiteRule.pageNum = null; return defaultPage; } } catch(e) { debug(e); } } } else { return defaultPage; } } let pageNum = url.replace(/.*[&\/\?](p=|page[=\/_-]?)(\d+).*/i, "$2"); return pageNum === url ? defaultPage : (pageNum.length > 4 ? defaultPage : pageNum); } async getNextLink(doc, exist) { let nextLink = null, page, href; let getNextLinkByForm = (form, submitBtn, n) => { let params = []; let formData = new FormData(form); if (submitBtn && submitBtn.getAttribute) { let btnValue, btnName; btnName = submitBtn.getAttribute("name"); btnValue = submitBtn.getAttribute("value"); if (btnName && btnValue) params = [btnName + "=" + encodeURIComponent(btnValue)]; } for (let [key, value] of formData) { if (n && /^(p|page)$/i.test(key)) { params.push(key + '=' + n); } else { params.push(key + '=' + encodeURIComponent(value)); } } params = params.join('&'); return form.action + (form.action.indexOf('?') === -1 ? '?' : '&') + params + (form.method === 'post' ? '#p{' + params + '}' : ''); }; if (this.curSiteRule.pageElementByJs) { this.nextLinkHref = "#"; return true; } else if (this.curSiteRule.nextLinkByJs) { try { let over = _url => {}; let targetUrl = await ((typeof this.curSiteRule.nextLinkByJs === 'function') ? this.curSiteRule.nextLinkByJs : new AsyncFunction("doc", '"use strict";' + this.curSiteRule.nextLinkByJs))(doc); if (targetUrl) nextLink = {href: targetUrl}; } catch(e) { debug(e); } } else if (this.curSiteRule.nextLinkByUrl) { let targetUrl = this.curUrl.replace(new RegExp(this.curSiteRule.nextLinkByUrl[0], "i"), this.curSiteRule.nextLinkByUrl[1]); if (targetUrl !== this.curUrl) { let includeSel = this.curSiteRule.nextLinkByUrl[2]; let excludeSel = this.curSiteRule.nextLinkByUrl[3]; if (includeSel) { includeSel = includeSel.trim(); if (!getElement(includeSel, doc)) { this.nextLinkHref=false; return null; } } if (excludeSel) { excludeSel = excludeSel.trim(); if (getElement(excludeSel, doc)) { this.nextLinkHref=false; return null; } } let reps = targetUrl.match(/{.*?}/g); if (reps) { reps.forEach(rep => { let code = rep.replace("{", "").replace("}", "").replace(/\(\)/g, "0"); let result = code.match(/^(\d*)\+1$/); if (result) { result = parseInt(result[1] || 1) + 1; } else { try { result = Function('"use strict";return ' + code)(); } catch(e) { debug(e); } } targetUrl = targetUrl.replace(rep, result); }); } } nextLink = {href: targetUrl}; } else if (typeof this.curSiteRule.nextLink !== 'undefined') { let nextLinkSel = this.curSiteRule.nextLink; if (nextLinkSel != 0) { if (Array && Array.isArray && Array.isArray(nextLinkSel)) { nextLink = getElement(nextLinkSel[nextIndex], doc, null, true); if (!nextLink && curPage === 1 && nextIndex !== 0) { nextIndex = 0; nextLink = getElement(nextLinkSel[nextIndex], doc, null, true); } } else nextLink = getElement(nextLinkSel, doc, null, true); } if (nextLink && (this.curSiteRule.action == 0 || this.curSiteRule.action == 1 || this.curSiteRule.action == 2)) { let form = doc.querySelector('#search-form'); if (!nextLink.href && nextLink.hasAttribute && nextLink.hasAttribute("onclick") && form) { if (/^\d+$/.test(nextLink.innerText)) { nextLink.href = getNextLinkByForm(form, nextLink, nextLink.innerText); } } else if (compareNodeName(nextLink, ["input"]) || nextLink.type === "submit") { form = nextLink.parentNode; while (form) { if (compareNodeName(form, ["form"])) break; else form = form.parentNode; } if (form) { nextLink.href = getNextLinkByForm(form, nextLink); } } if (nextLink.href && this.curSiteRule.action != 0) { nextLink.href = nextLink.href.replace(/#p{.*/, ""); } } } else { page = await this.getPage(doc, exist); nextLink = page.next; if (nextLink) { if (compareNodeName(nextLink, ["input"]) || nextLink.type === "submit") { if (!/next/i.test(nextLink.getAttribute("onclick"))) { let form = nextLink.parentNode; while (form) { if (compareNodeName(form, ["form"])) break; else form = form.parentNode; } if (form) { nextLink.href = getNextLinkByForm(form, nextLink); } } } let parent = nextLink; while (parent && !compareNodeName(parent, ["body"])) { if (parent.hasAttribute && parent.hasAttribute("disabled")) { this.nextLinkHref = false; return null; } if (parent.className && parent.classList) { if (parent.classList.contains("noClick") || parent.classList.contains("no-pages") || parent.classList.contains("disabled")) { this.nextLinkHref = false; return null; } } if (parent.style && parent.style.display === "none") { this.nextLinkHref = false; return null; } parent = parent.parentNode; } if (doc === document) { if (!this.linkHasHref(nextLink)) { if ((clickedSth && this.curSiteRule.smart) || !isVisible(nextLink, _unsafeWindow)) { this.nextLinkHref = false; return null; } let video = document.querySelector("video,iframe[id*=play],[id*=play]>iframe,iframe[src*=player],iframe[src*=m3u8]"); if (video) { if (video.offsetParent && video.name !== 'pagetual-iframe') { let scrollWidth = video.scrollWidth || video.offsetWidth; let scrollHeight = video.scrollHeight || video.offsetHeight; if (compareNodeName(video, ["iframe"])) { } else if (scrollWidth > 100 && scrollHeight > 100) { let winWidth = window.innerWidth || document.documentElement.clientWidth; let winHeight = window.innerHeight || document.documentElement.clientHeight; if (scrollWidth > winWidth>>1 && scrollHeight > winHeight>>1) { debug("Disable when large media found"); } else { video = null; } } else { video = null; } } else { video = null; } } if (video) { isPause = true; this.clearAddedElements(); this.nextLinkHref = false; return null; } } let nextLinkCs = _unsafeWindow.getComputedStyle(nextLink); if (nextLinkCs.cursor === "not-allowed") { this.nextLinkHref = false; return null; } this.initNext = nextLink; } let form = doc.querySelector('#search-form'); if (!nextLink.href && nextLink.hasAttribute("onclick") && form) { if (form && /^\d+$/.test(nextLink.innerText)) { href = getNextLinkByForm(form, nextLink, nextLink.innerText); } } } } if (nextLink) { if (!this.checkStopSign(nextLink, doc)) { if (curPage > 1 && rulesData.lastPageTips) showTips(i18n("lastPage"), "", 800); return null; } if (this.curSiteRule.action == 3) { if (doc == document) debug(nextLink, 'Next link'); this.nextLinkHref = '#'; } else { let needUrl = (this.curSiteRule.action == 0 || this.curSiteRule.action == 1 || this.curSiteRule.action == 2); if (!href) href = nextLink.href; if (href && nextLink.getAttribute) { let _href = nextLink.getAttribute("href"); if (_href) { if (_href.charAt(0) === "#" || _href === "?"){ href = "#"; } else { href = _href; } } else if (_href === "") { href = _href; } } if (needUrl && (href === "" || href === null)) { this.nextLinkHref = false; } else if (needUrl && /^(javascript:|#)/.test(href)) { this.nextLinkHref = false; } else { this.nextLinkHref = (href && !/^(javascript:|#)/.test(href)) ? this.canonicalUri(href) : "#"; let tempUrl = this.nextLinkHref; if (tempUrl !== "#" && (this.compareUrl(tempUrl, this.initUrl) || this.compareUrl(tempUrl, this.curUrl) || this.compareUrl(tempUrl, this.curUrl + "#") || this.compareUrl(tempUrl, this.oldUrl) || this.compareUrl(tempUrl, this.oldUrl + "#"))) { this.nextLinkHref = false; } else if (doc === document) { debug(nextLink, 'Next link'); } } } } else { this.nextLinkHref = false; } this.nextLinkEle = nextLink; this.preload(); return nextLink; } compareUrl(url1, url2) { if (url1 === url2) return true; if (!url1 || !url2) return false; let url1Arr = url1.split("?"); let url2Arr = url2.split("?"); if (url1Arr[0] != url2Arr[0]) return false; if (!url1Arr[1] || !url2Arr[1]) return false; url1Arr = url1Arr[1].split("&").sort().join("&"); url2Arr = url2Arr[1].split("&").sort().join("&"); return url1Arr === url2Arr; } filterEles(doc, eles) { let filter = this.curSiteRule.filter; if (!filter || !eles || eles.length === 0) return; if (eles.length === 1) { eles = eles[0].children; if (eles.length === 1) { eles = eles[0].children; } } if (typeof filter === "string") { if (/^\d+$/.test(filter)) { filter = {count: parseInt(filter)}; } else filter = {words: filter}; } [].forEach.call(eles, ele => { if (!ele.parentNode) return; let canKeep = (() => { let innerText = (ele.innerText && ele.innerText.trim()) || ""; if (filter.count) { if (innerText.length < filter.count) return false; } if (filter.words) { let wordsRegExp = new RegExp(filter.words, "i"); if (innerText && wordsRegExp.test(innerText)) return false; } if (filter.link) { let linkRegExp = new RegExp(filter.link, "i"); if (compareNodeName(ele, ["a"]) && linkRegExp.test(ele.href)) return false; let aChildren = ele.querySelectorAll("a"); for (let i = 0; i < aChildren.length; i++) { let child = aChildren[i]; if (linkRegExp.test(child.href)) return false; } } if (filter.selector) { if (getElement(filter.selector, doc, ele)) return false; } return true; })(); if (!canKeep) { ele.parentNode.removeChild(ele); } }); } checkStopSign(nextLink, doc) { if (this.curSiteRule.stopSign) { let typeArray = Array && Array.isArray && Array.isArray(this.curSiteRule.stopSign); let typeObject = !typeArray && (this.curSiteRule.stopSign.include || this.curSiteRule.stopSign.exclude || this.curSiteRule.stopSign.pageNum); if (typeArray || typeObject) { let includeSel, excludeSel, curSign, maxSign; if (typeArray) { includeSel = this.curSiteRule.stopSign[0]; excludeSel = this.curSiteRule.stopSign[1]; curSign = this.curSiteRule.stopSign[2]; maxSign = this.curSiteRule.stopSign[3]; if (Array && Array.isArray && Array.isArray(includeSel) && !curSign) { curSign = includeSel; includeSel = false; } if (excludeSel && Array && Array.isArray && Array.isArray(excludeSel) && !maxSign) { maxSign = excludeSel; excludeSel = false; } } else { includeSel = this.curSiteRule.stopSign.include; excludeSel = this.curSiteRule.stopSign.exclude; curSign = this.curSiteRule.stopSign.pageNum; } if (includeSel) { includeSel = includeSel.trim(); if (!getElement(includeSel, doc)) { isPause = true; this.nextLinkHref = false; return false; } } if (excludeSel) { excludeSel = excludeSel.trim(); if (getElement(excludeSel, doc)) { isPause = true; this.nextLinkHref = false; return false; } } if (curSign) { if (!maxSign) maxSign = curSign.slice(2); let currentEle = getElement(curSign[0], doc); let maxEle = getElement(maxSign[0], doc); if (currentEle && maxEle) { let currentSignNum, maxSignNum; if (/\(.*\)/.test(curSign[1])) { currentSignNum = currentEle.innerText.match(new RegExp(curSign[1])); if (currentSignNum) currentSignNum = currentSignNum[1]; } else if (currentEle.getAttribute) { currentSignNum = currentEle.getAttribute(curSign[1]); } if (/\(.*\)/.test(maxSign[1])) { maxSignNum = maxEle.innerText.match(new RegExp(maxSign[1])); if (maxSignNum) maxSignNum = maxSignNum[1]; } else if (maxEle.getAttribute) { maxSignNum = maxEle.getAttribute(maxSign[1]); } if (currentSignNum && maxSignNum && currentSignNum == maxSignNum) { isPause = true; this.nextLinkHref = false; return false; } } } } else { try { let stopSign = ((typeof this.curSiteRule.stopSign === 'function') ? this.curSiteRule.stopSign : Function("doc", "nextLink", '"use strict";' + this.curSiteRule.stopSign))(doc, nextLink); if (stopSign) { isPause = true; this.nextLinkHref = false; return false; } } catch(e) { debug(e); } } } return true; } preloadImageHandler() { if (this.preloadingImage || !this.unCheckedImgs.length) return; this.preloadingImage = true; setTimeout(() => { this.preloadingImage = false; this.preloadImageHandler(); }, 10); let iSrc = this.unCheckedImgs.shift(); let img = document.createElement('img'); img.src = iSrc; this.preloadDiv.appendChild(img); } preload() { if (!rulesData.preload) return; if (this.curSiteRule.preload === 0) return; if (!this.nextLinkHref || this.nextLinkHref == "#") return; let self = this, url = this.nextLinkHref; let postParams = url.match(/#p{(.*)}$/); if (postParams) { postParams = postParams[1]; url = url.replace(/#p{.*/, ""); } _GM_xmlhttpRequest({ url: url, method: postParams ? 'POST' : 'GET', data: postParams, overrideMimeType: 'text/html;charset=' + charset, headers: { 'Referer': location.href, 'User-Agent': navigator.userAgent, "Content-Type": (postParams ? "application/x-www-form-urlencoded" : "text/html") + ";charset=" + charset, }, timeout: 10000, onload: function(res) { var doc = null; try { doc = document.implementation.createHTMLDocument(''); doc.documentElement.innerHTML = createHTML(res.response); var body = getBody(doc); if (!self.preloadDiv) { self.preloadDiv = document.createElement('div'); self.preloadDiv.id = "pagetual-preload"; self.preloadDiv.style.cssText = 'display:none!important;'; getBody(document).appendChild(self.preloadDiv); self.checkedImgs = {}; self.unCheckedImgs = []; } let code = self.curSiteRule.preloadImages; if (code) { let imgSrcArr = new Function("doc", '"use strict";' + code)(doc); if (imgSrcArr && imgSrcArr.length) { imgSrcArr.forEach(imgSrc => { if (imgSrc && !self.checkedImgs[imgSrc]) { self.checkedImgs[imgSrc] = true; self.unCheckedImgs.push(imgSrc); } }); } self.preloadImageHandler(); } else if (code !== 0 && code !== false) { if (body && body.firstChild) { self.lazyImgAction(body.children, doc); } [].forEach.call(doc.images, i => { let iSrc = i.src; if (iSrc && !self.checkedImgs[iSrc]) { self.checkedImgs[iSrc] = true; self.unCheckedImgs.push(iSrc); } }); self.preloadImageHandler(); } } catch(e) { debug(e); return; } } }); } getInsert(refresh) { if (refresh) { this.insert = null; } if (this.insert && this.insert.parentNode && document.documentElement.contains(this.insert)) { return this.insert; } if (this.curSiteRule.insert) { let insertSel = this.curSiteRule.insert; if (Array && Array.isArray && Array.isArray(insertSel)) { insertSel = insertSel[nextIndex < insertSel.length ? nextIndex : 0]; } this.insert = getElement(insertSel, document, null, true); } else { this.docPageElement = null; let pageElement = this.getPageElement(document, _unsafeWindow); if (this.curSiteRule.smart && this.nextLinkHref == "#" && this.curSiteRule.pageElement === 'body') { debug("Stop as jsNext & whole body"); isPause = true; return null; } if (pageElement && pageElement.length > 0) { let pEIndex = pageElement.length - 1; let pELast = pageElement[pEIndex]; while(pELast && compareNodeName(pELast, ["link", "meta", "style", "script"])) { pEIndex--; pELast = pageElement[pEIndex]; } this.insert = pELast.nextSibling ? pELast.nextSibling : pELast.parentNode.appendChild(document.createTextNode(' ')); } } return this.insert; } pageInit(doc, eles) { let code = this.curSiteRule.pageInit; if (code) { let initFunc = ((typeof code == 'function') ? code : Function("doc", "eles", '"use strict";' + code)); let checkInit = (resolve) => { try { if (initFunc(doc, eles) === false) { setTimeout(() => { checkInit(resolve); }, 100); } else { resolve(true); } } catch(e) { resolve(false); debug(e); } }; return new Promise((resolve) => { checkInit(function(e) { resolve(e) }); }) } } pageAction(doc, eles) { let code = this.curSiteRule.pageAction; if (code) { try { ((typeof code == 'function') ? code : Function("doc", "eles", '"use strict";' + code))(doc, eles); } catch(e) { debug(e); } } this.openInNewTab(eles); this.replaceElement(doc); } openInNewTab(eles) { if (openInNewTab) { [].forEach.call(eles, ele => { if (compareNodeName(ele, ["a"]) && ele.href && !/^(mailto:|javascript:)|#/.test(ele.href)) { ele.setAttribute('target', openInNewTab == 1 ? '_blank' : '_self'); } else { [].forEach.call(ele.querySelectorAll('a[href]:not([href^="mailto:"]):not([href^="javascript:"]):not([href^="#"])'), a => { if (openInNewTab == 1) { a.setAttribute('target', '_blank'); if (a.getAttribute('onclick') == 'atarget(this)') { a.removeAttribute('onclick'); } } else a.setAttribute('target', '_self'); }); } }); } } lazyImgAction(eles, doc) { if (!eles || eles.length == 0) return; let lazyImgSrc = this.curSiteRule.lazyImgSrc; if (lazyImgSrc === 0 || lazyImgSrc === '0') return; let imgLazyAttrs = []; let lazyAttrs = ["div[data-thumb]|data-src", "div.img|data-src", "div.lazy|data-src", "div.lazy|data-original", "a.lazy|data-bg", "a.lazyload|data-original"]; let removeProps = []; let setLazyImg = img => { let realSrc; imgLazyAttrs.forEach(attr => { realSrc = img.getAttribute(lazyImgSrc[0]); if (realSrc) { removeProps.forEach(prop => { img.removeAttribute(prop.trim()); }); img.src = realSrc; return; } }) if (!realSrc) { let lazyAttr = ""; if (img.getAttribute("_src") && !img.src) { lazyAttr = "_src"; realSrc = img.getAttribute(lazyAttr); } else { for (let i in lazyImgAttr) { lazyAttr = lazyImgAttr[i]; let attrValue = img.getAttribute(lazyAttr); if (attrValue) { realSrc = attrValue; break; } } } if (!realSrc && img._lazyrias && img._lazyrias.srcset) { realSrc = img._lazyrias.srcset[img._lazyrias.srcset.length - 1]; lazyAttr = "_lazyrias"; } if (!realSrc && img.srcset) { lazyAttr = "srcset"; var srcs = img.srcset.split(/[xw],/i), largeSize = 0; srcs.forEach(srci => { let srcInfo = srci.trim().split(" "), curSize = parseInt(srcInfo[1]); if (srcInfo[1] && curSize > largeSize) { largeSize = curSize; realSrc = srcInfo[0]; } }); } if (realSrc) { img.src = realSrc; img.removeAttribute("srcset"); img.removeAttribute(lazyAttr); if (img.classList && img.classList.contains && img.classList.contains("lazy")) { img.classList.remove("lazy"); } if (img.style.display == "none") { img.style.display = ""; } if (img.style.visibility == "hidden") { img.style.visibility = ""; } if (img.style.opacity == 0) { img.style.opacity = ""; } } } }; if (lazyImgSrc) { if (!Array.isArray(lazyImgSrc)) { lazyAttrs = lazyImgSrc.split(","); } else { lazyAttrs = lazyImgSrc[0].split(","); removeProps = lazyImgSrc[1].split(","); } } lazyAttrs.forEach(attr => { let attrArr = attr.split("|"); if (attrArr.length !== 2) { imgLazyAttrs.push(attr.trim()); } else { let selector = attrArr[0].trim(); let lazyAttr = attrArr[1].trim(); if (selector == "img") { imgLazyAttrs.push(lazyAttr); } else { selector += "[" + lazyAttr + "]"; [].forEach.call(doc.querySelectorAll(selector), ele => { ele.style.setProperty("background-image", "url(" + ele.getAttribute(lazyAttr) + ")", "important"); removeProps.forEach(prop => { ele.removeAttribute(prop.trim()); }); }); } } }); [].forEach.call(doc.querySelectorAll("img,picture>source"), img => { setLazyImg(img); }); } initPage(callback) { let self = this; if (self.initing) return; self.initing = true; setTimeout(() => { self.initing = false; }, 100); curPage = 1; urlChanged = false; tryTimes = 0; this.clearAddedElements(); this.insert = null; this.visibilityItems = []; this.visibleIndex = -1; this.pageDoc = document; this.nextLinkHref = null; this.curUrl = location.href; this.oldUrl = ""; this.initUrl = location.href; this.historyUrl = ""; this.possibleCheck = 0; let base = document.querySelector("base"); this.basePath = base ? base.href : location.href; this.getRule(async () => { if (self.curSiteRule.sideController === true || (self.curSiteRule.sideController !== false && rulesData.sideController)) { isPause = manualPause; } if (typeof(self.curSiteRule.rate) !== "undefined") { rate = self.curSiteRule.rate; } if (self.curSiteRule.enable == 0) { debug("Stop as rule disable"); isPause = true; _GM_registerMenuCommand(i18n("enable"), () => { showTips(i18n("enableSiteTips")); if(!self.customRules) { self.customRules = []; } for (let i in self.customRules) { if (self.customRules[i].url == self.curSiteRule.url) { self.customRules.splice(i, 1); break; } } self.curSiteRule.enable = 1; self.customRules.unshift(self.curSiteRule); storage.setItem("customRules", self.customRules); location.reload(); }); return; } if (rulesData.sideControllerAlways) { sideController.setup(); } //若是再亂匹配就不緩存wedata,或者只在找完本地規則之後再考慮wedata的緩存 if (self.curSiteRule.smart) { delete self.curSiteRule.pageElement; if (!self.possibleRule) { self.smartRules = self.smartRules.filter(item => {return item && item.url != self.curSiteRule.url}); self.smartRules.unshift(self.curSiteRule); if (self.smartRules.length > 100) { self.smartRules.pop(); } storage.setItem("smartRules", self.smartRules); } } else if (self.curSiteRule && self.curSiteRule.url.length > 13) { self.addToHpRules(); } let css = self.curSiteRule.css || rulesData.customCss; if (css) { let cssArr = css.split("inIframe:"); if (cssArr && cssArr.length) { _GM_addStyle(cssArr[0]); } } if (/sidesearch=(1|true)$/.test(self.curUrl)) { openInNewTab = 0; } else if (typeof self.curSiteRule.openInNewTab !== 'undefined') { openInNewTab = self.curSiteRule.openInNewTab ? 1 : 2; } let autoClick = self.curSiteRule.autoClick; if (autoClick) { let autoClickBtn; autoClickBtn = getElement(autoClick, document, null, true); if (autoClickBtn) { emuClick(autoClickBtn); } } let code = self.curSiteRule.init; if (code) { try { await ((typeof code == 'function') ? code : new AsyncFunction('doc', 'win', 'iframe', 'click', 'enter', 'input', 'sleep', '"use strict";' + code))(null, null, null, async sel => {await clickAction(sel, document)}, async sel => {await enterAction(sel, document)}, async (sel, v) =>{await inputAction(sel, v, document)}, async time => {await sleep(time)}); } catch(e) { debug(e); } } await self.getNextLink(document, true); if (self.curSiteRule.pageNum && self.nextLinkHref && self.nextLinkHref != "#") { let num1st = self.getPageNumFromUrl(location.href, 1); let num2nd = self.getPageNumFromUrl(self.nextLinkHref, 1); if (num2nd != num1st + 1) { self.curSiteRule.pageNum = null; } } if (self.curSiteRule.smart && self.nextLinkHref == false && self.possibleRule) { let urlReg = new RegExp(self.possibleRule.url, "i"); function checkPossible () { if (self.possibleCheck++ < 3) { setTimeout(() => { if (self.curSiteRule.smart) { var href = location.href.slice(0, 500); if (urlReg.test(href) && self.ruleMatch(self.possibleRule)) { self.initPage(() => {}); } else checkPossible(); } }, 3000); } } checkPossible(); } self.refreshByClick(); if (emuIframe && emuIframe.parentNode) { emuIframe.parentNode.removeChild(emuIframe); emuIframe = null; } let pageElementCss = self.curSiteRule.pageElementCss || rulesData.pageElementCss; if (pageElementCss && pageElementCss !== '0') { self.getPageElement(document, _unsafeWindow); } callback(); let initRun = typeof self.curSiteRule.initRun == 'undefined' ? rulesData.initRun : self.curSiteRule.initRun; if (self.nextLinkHref) { sideController.setup(); if (initRun && initRun != false) { setTimeout(() => { nextPage(); }, 300); } } }); } async hookUrlSetEle(ele, doc) { let self = this; return new Promise((resolve) => { let catchUrl = e => { ele.dataset.url = self.catchedUrl; ele.setAttribute('onclick', 'window.open(this.dataset.url)'); window.removeEventListener('pagetual_openUrl', catchUrl); resolve(); }; window.addEventListener('pagetual_openUrl', catchUrl); emuClick(ele, doc); }); } async hookUrl(doc) { let sel = this.curSiteRule.hookUrl; if (!sel) return; let self = this; if (!this.initHook) { this.initHook = true; Object.defineProperty(doc.defaultView, 'open', { get: function () { return (s) => { self.catchedUrl = s; var e = new CustomEvent('pagetual_openUrl'); window.dispatchEvent(e); } } }); } let eles = getAllElements(sel, doc); for (let i = 0; i < eles.length; i++) { await sleep(1); let ele = eles[i]; if (!ele.dataset.url) { await this.hookUrlSetEle(ele, doc); } } } beginLoading() { isLoading = true; if (targetY >= 0) { window.scrollTo({ top: targetY, behavior: 'instant'}); targetY = -1; } let lastScrollTop = getBody(document).scrollTop || document.documentElement.scrollTop; ruleParser.insertElement(loadingDiv); if (forceState == 2) { getBody(document).appendChild(loadingDiv); } else { let parent = loadingDiv.parentNode; if (compareNodeName(parent, ["tbody"])) { parent = parent.parentNode; } if (compareNodeName(parent, ["table"])) { parent.parentNode.appendChild(loadingDiv); } } getBody(document).scrollTop = lastScrollTop; document.documentElement.scrollTop = lastScrollTop; } insertElement(ele) { if (!this.insert || !this.insert.parentNode) { this.getInsert(); } if (this.insert && this.insert.parentNode) { let self = this; if (ele.nodeName == "#document-fragment") { [].forEach.call(ele.children, el => { self.addedElePool.push(el); }); } else { this.addedElePool.push(ele); } if (this.curSiteRule.insertPos == 2 || this.curSiteRule.insertPos == "in") { this.insert.appendChild(ele); } else { this.insert.parentNode.insertBefore(ele, this.insert); } } } noValidContent(url) { if (!this.curSiteRule.nextLinkByUrl) showTips(i18n("noValidContent"), url); } async insertPage(doc, eles, url, callback, tried) { this.oldUrl = this.curUrl; let oldTitle = document.title; try { let oldTitle = this.pageDoc.title; } catch (e) {} this.pageDoc = doc; this.curUrl = url; isLoading = true; let nextLink = await this.getNextLink(doc); this.nextTitle = ""; if (this.curSiteRule.pageBarText) { if (this.curSiteRule.pageBarText == 1 || this.curSiteRule.pageBarText == true) { this.nextTitle = doc.title; } else { try { this.nextTitle = ((typeof this.curSiteRule.pageBarText == 'function') ? this.curSiteRule.pageBarText : Function("doc",'"use strict";' + this.curSiteRule.pageBarText))(doc); } catch(e) { debug(e); } } } if (curPage == 1 && !tried && !nextLink && this.curSiteRule.smart && this.curSiteRule.pageElement && this.curSiteRule.action != 0) { this.curSiteRule.action = 1; this.curUrl = location.href; isLoading = false; return false; } if (targetY >= 0) { window.scrollTo({ top: targetY, behavior: 'instant'}); targetY = -1; } let lastScrollTop = getBody(document).scrollTop || document.documentElement.scrollTop; this.getInsert(); await this.pageInit(doc, eles); var self = this, newEles = []; if (!eles || eles.length == 0 || !self.insert || !self.insert.parentNode) { if (callback) callback(eles); loadPageOver(); } else { if (callback) callback(eles); loadPageOver(); const collection = document.createDocumentFragment(); [].forEach.call(eles, ele => { let newEle = ele.cloneNode(true); let oldCanvass = ele.querySelectorAll("canvas"); let newCanvass = newEle.querySelectorAll("canvas"); if (self.updateUrl) { [].forEach.call(newEle.querySelectorAll("img"), img => { if (img.getAttribute("src")) img.src = self.canonicalUri(img.getAttribute("src")); }); [].forEach.call(newEle.querySelectorAll("a"), a => { if (a.getAttribute("href")) a.href = self.canonicalUri(a.getAttribute("href")); }); } for (let i = 0; i < oldCanvass.length; i++) { let oldCanvas = oldCanvass[i]; let newCanvas = newCanvass[i]; newCanvas.getContext('2d').drawImage(oldCanvas, 0, 0); } if (!compareNodeName(newEle, ["style", "script"])) self.visibilityItems.push(newEle); collection.appendChild(newEle) newEles.push(newEle); }); self.insertElement(collection); } getBody(document).scrollTop = lastScrollTop; document.documentElement.scrollTop = lastScrollTop; this.pageAction(doc, newEles); let enableHistory = this.curSiteRule.history; let enableHistoryAfterInsert = false; if (enableHistory == 1) { enableHistory = true; } else if (enableHistory == 2) { enableHistory = true; enableHistoryAfterInsert = true; } else if (enableHistory == 0) { enableHistory = false; } else { enableHistory = rulesData.enableHistory; enableHistoryAfterInsert = rulesData.enableHistoryAfterInsert; } if (enableHistory) { this.historyUrl = enableHistoryAfterInsert ? this.curUrl : this.oldUrl; if(this.historyUrl != location.href) { let isJs = this.hrefIsJs(this.historyUrl); if (!isJs) { let historyTitle = enableHistoryAfterInsert ? doc.title : oldTitle; _unsafeWindow.history.replaceState(undefined, historyTitle, this.historyUrl); document.title = historyTitle; } } } isLoading = false; return true; } } var ruleParser = new RuleParser(); class SideController { //static controller; constructor() { this.inited = false; } /*static getInstance() { if (!SideController.controller) { SideController.controller = new SideController(); } return SideController.controller; }*/ setup() { if (ruleParser.curSiteRule.sideController === false) return; if (!rulesData.sideController && !ruleParser.curSiteRule.sideController) return; this.addToStage(); } init() { if (this.inited) return; this.inited = true; let self = this; this.cssText = ` #pagetual-sideController { position: fixed; top: calc(50% - 83px); left: calc(99% - 40px); width: 40px; border-radius: 20px; box-shadow: rgb(0 0 0) 0px 0px 10px; text-align: center; background: #ffffffd0!important; user-select: none; z-index: 2147483646!important; padding: 0!important; opacity: 0.35; transition: opacity .5s ease, background .5s, box-shadow .5s; } #pagetual-sideController * { font-weight: bold; font-family: arial; font-style: normal; font-size: 20px!important; color: black!important; line-height: normal; float: none; text-align: center; } #pagetual-sideController.stop { -webkit-filter: invert(100%); filter: invert(100%); } .pagetual-sideController-btn { padding: 5px 0; cursor: pointer; transition: transform .15s ease-in-out, opacity .3s ease; } #pagetual-sideController .pagetual-sideController-btn:hover { transform: scale(1.5); color: red!important; } #pagetual-sideController #pagetual-sideController-move > svg { transition: transform .3s ease; } #pagetual-sideController #pagetual-sideController-move > svg:hover { transform: scale(1.2); } #pagetual-sideController.minSize #pagetual-sideController-move > svg { background: white; opacity: 0; } #pagetual-sideController #pagetual-sideController-move > img, #pagetual-sideController #pagetual-sideController-move > span { width: 35px; height: 35px; cursor: pointer; } #pagetual-sideController.minSize #pagetual-sideController-move > img, #pagetual-sideController.minSize #pagetual-sideController-move > span { border-radius: 50px; text-shadow: rgb(255 255 255) 0px 0px 10px; } #pagetual-sideController #pagetual-sideController-pagenum { font-size: 15px!important; line-height: 30px; text-align: center; position: absolute; right: calc(50% - 15px); top: calc(50% - 15px); border: 1px solid #00000099; display: inline-block; width: 30px; height: 30px; box-sizing: border-box; border-radius: 50%; background: white; opacity: 0; transition: opacity .5s ease; pointer-events: none; } #pagetual-sideController.minSize #pagetual-sideController-pagenum { opacity: 0.8; } #pagetual-sideController:hover { opacity: 1; } #pagetual-sideController>.extra { bottom: 170px; left: 0px; width: 40px; position: absolute; } #pagetual-sideController>.extra>svg { width: 30px; height: 30px; opacity: 0.1; cursor: pointer; margin: 0 0 5px 0; transition: opacity .3s ease; } #pagetual-sideController>.extra>svg:hover { opacity: 1; } #pagetual-sideController.minSize { box-shadow: rgb(0 0 0 / 0%) 0px 0px 0px; background: #00000000!important; } #pagetual-sideController.minSize .pagetual-sideController-btn { opacity: 0; } `; let frame = document.createElement("div"); frame.id = "pagetual-sideController"; frame.innerHTML = createHTML(`
${loadingText}
${rulesData.hideLoadingIcon ? "" : ``}`); } var upSvg = ``; var upSvgCSS = `text-align: center;display: initial;position: relative;cursor: pointer;margin: 0 8px;width: 30px;height: 30px;vertical-align: baseline;fill: currentColor;overflow: hidden;`; var downSvg = ``; var downSvgCSS = `text-align: center;display: initial;position: relative;cursor: pointer;margin: 0 8px;width: 30px;height: 30px;vertical-align: baseline;fill: currentColor;overflow: hidden;transform: rotate(180deg);`; const initStyle = `text-indent: initial;display: contents;right: unset;left: unset;top: unset;bottom: unset;inset: unset;clear: both;cy: initial;d: initial;dominant-baseline: initial;empty-cells: initial;fill: initial;fill-opacity: initial;fill-rule: initial;filter: initial;flex: initial;flex-flow: initial;float: initial;flood-color: initial;flood-opacity: initial;grid: initial;grid-area: initial;height: initial;hyphens: initial;image-orientation: initial;image-rendering: initial;inline-size: initial;inset-block: initial;inset-inline: initial;isolation: initial;letter-spacing: initial;lighting-color: initial;line-break: initial;list-style: initial;margin-block: initial;margin: 0px 5px;margin-inline: initial;marker: initial;mask: initial;mask-type: initial;max-block-size: initial;max-height: initial;max-inline-size: initial;max-width: initial;min-block-size: initial;min-height: initial;min-inline-size: initial;min-width: initial;mix-blend-mode: initial;object-fit: initial;object-position: initial;offset: initial;opacity: initial;order: initial;orphans: initial;outline: initial;outline-offset: initial;overflow-anchor: initial;overflow-clip-margin: initial;overflow-wrap: initial;overflow: initial;overscroll-behavior-block: initial;overscroll-behavior-inline: initial;overscroll-behavior: initial;padding-block: initial;padding: initial;padding-inline: initial;page: initial;page-orientation: initial;paint-order: initial;perspective: initial;perspective-origin: initial;pointer-events: initial;position: relative;quotes: initial;r: initial;resize: initial;ruby-position: initial;rx: initial;ry: initial;scroll-behavior: initial;scroll-margin-block: initial;scroll-margin: initial;scroll-margin-inline: initial;scroll-padding-block: initial;scroll-padding: initial;scroll-padding-inline: initial;scroll-snap-align: initial;scroll-snap-stop: initial;scroll-snap-type: initial;scrollbar-gutter: initial;shape-image-threshold: initial;shape-margin: initial;shape-outside: initial;shape-rendering: initial;size: initial;speak: initial;stop-color: initial;stop-opacity: initial;stroke: initial;stroke-dasharray: initial;stroke-dashoffset: initial;stroke-linecap: initial;stroke-linejoin: initial;stroke-miterlimit: initial;stroke-opacity: initial;stroke-width: initial;tab-size: initial;table-layout: initial;text-align: initial;text-align-last: initial;text-anchor: initial;text-combine-upright: initial;text-decoration: initial;text-decoration-skip-ink: initial;text-indent: initial;text-overflow: initial;text-shadow: initial;text-size-adjust: initial;text-transform: initial;text-underline-offset: initial;text-underline-position: initial;touch-action: initial;transform: initial;transform-box: initial;transform-origin: initial;transform-style: initial;transition: initial;user-select: initial;vector-effect: initial;vertical-align: initial;visibility: initial;border-spacing: initial;-webkit-border-image: initial;-webkit-box-align: initial;-webkit-box-decoration-break: initial;-webkit-box-direction: initial;-webkit-box-flex: initial;-webkit-box-ordinal-group: initial;-webkit-box-orient: initial;-webkit-box-pack: initial;-webkit-box-reflect: initial;-webkit-highlight: initial;-webkit-hyphenate-character: initial;-webkit-line-break: initial;-webkit-line-clamp: initial;-webkit-mask-box-image: initial;-webkit-mask: initial;-webkit-mask-composite: initial;-webkit-perspective-origin-x: initial;-webkit-perspective-origin-y: initial;-webkit-print-color-adjust: initial;-webkit-rtl-ordering: initial;-webkit-ruby-position: initial;-webkit-tap-highlight-color: initial;-webkit-text-combine: initial;-webkit-text-decorations-in-effect: initial;-webkit-text-emphasis: initial;-webkit-text-emphasis-position: initial;-webkit-text-fill-color: initial;-webkit-text-security: initial;-webkit-text-stroke: initial;-webkit-transform-origin-x: initial;-webkit-transform-origin-y: initial;-webkit-transform-origin-z: initial;-webkit-user-drag: initial;-webkit-user-modify: initial;white-space: initial;widows: initial;width: initial;will-change: initial;word-break: initial;word-spacing: initial;x: initial;y: initial;`; const pageTextStyle = `opacity: 1!important;text-indent: initial;padding: unset;border: none;background: unset!important;line-height: 30px;text-decoration: none;user-select: none;visibility: visible;position: initial;width: auto;max-width: 80%; white-space: nowrap; text-overflow: ellipsis;overflow: hidden;height: auto;float: none;clear: both;margin: 0px;text-align: center;display: inline-block;font-weight: bold;font-style: normal;font-size: 16px;letter-spacing: initial;vertical-align: top;color: rgb(85, 85, 95)!important;`; var sideControllerIcon = ''; var tipsWords = document.createElement("div"); tipsWords.className = "pagetual_tipsWords"; function changeStop(stop) { isPause = stop; [].forEach.call(getBody(document).querySelectorAll(".pagetual_pageBar,#pagetual-sideController"), bar => { if (isPause) { bar.classList.add("stop"); } else { bar.classList.remove("stop"); } }); if (!isPause) ruleParser.showAddedElements(); manualPause = isPause; if (sideController.inited) setListData("pauseState", location.host, isPause ? true : ""); } function changeHideBar(hide) { isHideBar = hide; [].forEach.call(getBody(document).querySelectorAll(".pagetual_pageBar"), bar => { if (isHideBar) { bar.classList.add("hide"); } else { bar.classList.remove("hide"); } }); } function isInViewPort(element) { if (!getBody(document).contains(element)) return false; const viewWidth = window.innerWidth || document.documentElement.clientWidth; const viewHeight = window.innerHeight || document.documentElement.clientHeight; const { top, right, bottom, left, } = element.getBoundingClientRect(); return ( top >= 0 && left >= 0 && right <= viewWidth + 1 && top <= viewHeight * rate && isVisible(element, _unsafeWindow) ); } function getPageBar() { let preBar = null, nextBar = null; let pageBars = [].slice.call(document.querySelectorAll(".pagetual_pageBar")); for (let i = 0; i < pageBars.length; i++) { let pageBar = pageBars[i]; if (!pageBar || !getBody(document).contains(pageBar)) continue; let { top, right, bottom, left, } = pageBar.getBoundingClientRect(); if (top > 500) { nextBar = pageBar; preBar = (i - 1 >= 0 ? pageBars[i - 1] : null); if (pageBar && getBody(document).contains(pageBar)) { let { top, right, bottom, left, } = pageBar.getBoundingClientRect(); if (top < -500) { preBar = pageBar; } else preBar = (i - 2 >= 0 ? pageBars[i - 2] : null); } break; } } if (!nextBar) preBar = pageBars[pageBars.length - 2]; return {preBar: preBar, nextBar: nextBar}; } var urlChanged = false; var urlChanging = false; var urlchangeHandler = e => { if (ruleParser && ruleParser.curSiteRule && ruleParser.curSiteRule.listenUrlChange == false) return; isPause = true; setTimeout(() => { lastActiveUrl = location.href; if (urlChanging) return; urlChanging = true; let href = location.href.slice(0, 60); if (href == configPage[1]) { setTimeout(() => { initConfig(href); urlChanging = false; }, 1000); } else { setTimeout(() => { if (guidePage.test(href)) { setTimeout(() => { initConfig(href); urlChanging = false; }, 1000); } else { urlChanged = true; if (!ruleParser.nextLinkHref) { isLoading = false; } urlChanging = false; //if (!pageReady && !ruleImportUrlReg.test(href)) location.reload(); } }, 500); } }, 1); }; window.addEventListener('pagetual_pushState', urlchangeHandler); /*var _wr = function(type) { var orig = history[type]; return function() { var rv = orig.apply(this, arguments); var e = new Event('pagetual_' + type); e.arguments = arguments; window.dispatchEvent(e); return rv; }; }; history.pushState = _wr('pushState');*/ function listenUrl() { var prevPathname = window.location.pathname; var prevSearch = window.location.search; var checkUrlTime = 100; var checkUrlTimer; var checkClickedEle = null; var checkFunc = () => { if (forceState == 1) return; if (checkClickedEle) { if (!clickedSth && checkClickedEle && checkClickedEle.nodeName) { if (compareNodeName(checkClickedEle, ["a", "button"])) { clickedSth = true; } else { let targetStyle = _unsafeWindow.getComputedStyle(checkClickedEle); if (targetStyle.cursor == "pointer") clickedSth = true; } } checkClickedEle = null; } if (checkUrlTime < 5000) { checkUrlTime += checkUrlTime>>1; } clearTimeout(checkUrlTimer); checkUrlTimer = setTimeout(checkFunc, checkUrlTime); if (document.hidden) return; if ((prevPathname !== window.location.pathname || prevSearch !== window.location.search) && window.location.href != ruleParser.historyUrl) { checkUrlTime = 2000; urlWillChange = true; var e = new Event('pagetual_pushState'); e.arguments = arguments; window.dispatchEvent(e); clickedSth = false; } prevPathname = window.location.pathname; prevSearch = window.location.search; }; checkUrlTimer = setTimeout(checkFunc, checkUrlTime); document.addEventListener("click", e => { if (!checkClickedEle) { checkClickedEle = e.target; checkUrlTime = 300; clearTimeout(checkUrlTimer); checkUrlTimer = setTimeout(checkFunc, checkUrlTime); } }); } let scrollContainer; function distToBottom () { let scrolly = window.scrollY; let windowHeight = window.innerHeight || document.documentElement.clientHeight; if (!scrollContainer || !document.documentElement.contains(scrollContainer)) { if (curPage > 1 || ruleParser.nextLinkHref) { let pageEle = ruleParser.getPageElement(document); if (pageEle && pageEle.length) { let parent = pageEle[0].parentNode, pageScrollY = parent.scrollTop; while (parent && pageScrollY == 0) { parent = parent.parentNode; pageScrollY = parent.scrollTop; } if (pageScrollY) { scrollContainer = parent; return scrollContainer.scrollHeight - pageScrollY - windowHeight; } } } } if (scrollContainer) { return scrollContainer.scrollHeight - scrollContainer.scrollTop - windowHeight; } let scrollH = Math.max(document.documentElement.scrollHeight, getBody(document).scrollHeight); return scrollH - scrolly - windowHeight; } let scrollHandler, clickToResetHandler, dblclickHandler, keydownHandler, hashchangeHandler, manualModeKeyHandler, pagetualNextHandler, keyupHandler; function initListener () { document.removeEventListener('scroll', scrollHandler, true); document.removeEventListener('wheel', scrollHandler, true); document.removeEventListener('dblclick', dblclickHandler); document.removeEventListener('keydown', keydownHandler); window.removeEventListener('hashchange', hashchangeHandler, false); document.removeEventListener('keydown', manualModeKeyHandler); document.removeEventListener('pagetual.next', pagetualNextHandler, false); document.removeEventListener('keyup', keyupHandler); let loadmoreBtn, loadingMore = true, lastScroll = 0, checkLoadMoreTimes = 0; if (ruleParser.curSiteRule.smart) { loadingMore = false; } else if (ruleParser.curSiteRule.loadMore) { loadingMore = false; } clickMode = typeof ruleParser.curSiteRule.clickMode == 'undefined' ? rulesData.clickMode : ruleParser.curSiteRule.clickMode; let clickingNext = false; let clickNext = async () => { if (clickingNext) return; clickingNext = true; setTimeout(() => { clickingNext = false; }, 1500); let nextLink = ruleParser.nextLinkHref; if (!nextLink) return; let isJs = ruleParser.hrefIsJs(nextLink); if (isJs) { let nextBtn = ruleParser.nextLinkEle; if (!nextBtn || !nextBtn.offsetParent) nextBtn = await ruleParser.getNextLink(document, true); if (nextBtn) emuClick(nextBtn); } else { window.location.href = nextLink; } }; let checkScrollReach = () => { let dist = distToBottom(); if (clickMode) { if (dist < 10) { clickNext(); } } else if (dist < bottomGap) { nextPage(); } }; scrollHandler = e => { if (urlChanged && !isLoading) { ruleParser.initPage(() => {}); urlChanged = false; loadingMore = false; return; } if (isPause) return; if (!loadingMore) { loadmoreBtn = getLoadMore(document, loadmoreBtn); if (loadmoreBtn) { if (isInViewPort(loadmoreBtn)) { emuClick(loadmoreBtn); loadingMore = true; setTimeout(() => {loadingMore = false}, 200); } } else { loadingMore = true; if (!ruleParser.curSiteRule.smart || checkLoadMoreTimes++ < 3) { setTimeout(() => {loadingMore = false}, 200); } } } if (!isLoading && !stopScroll) { checkScrollReach(); } ruleParser.changeVisibility(); let curScroll = getBody(document).scrollTop || document.documentElement.scrollTop; if (ruleParser.curSiteRule.lockScroll) { if (isLoading && Math.abs(lastScroll - curScroll) > 350) { getBody(document).scrollTop = lastScroll; document.documentElement.scrollTop = lastScroll; } else { lastScroll = curScroll; } } if (targetY >= 0) { if (Math.abs(targetY - curScroll) < 100) { targetY = -1; } } }; dblclickHandler = e => { if (forceState == 1 || compareNodeName(e.target, ["input", "textarea", "select", "a", "button", "svg", "use", "img", "path"])) return; if (!rulesData.dbClick2StopKey) { if ((rulesData.dbClick2StopCtrl && !e.ctrlKey) || (rulesData.dbClick2StopAlt && !e.altKey) || (rulesData.dbClick2StopShift && !e.shiftKey) || (rulesData.dbClick2StopMeta && !e.metaKey)) { return; } } if (!compareNodeName(e.target, ["body"]) && !e.target.classList.contains('pagetual_pageBar')) { try { let selection = window.getSelection(); let selStr = selection.toString().trim(); if (!selStr) { selection = selection.getRangeAt(0); selStr = selection && selection.cloneContents().children[0]; if (selStr && !compareNodeName(selStr, ["img"])) selStr = false; } if (selStr) { return; } } catch (e) {} } if (rulesData.dbClick2Stop && (ruleParser.nextLinkHref || loadmoreBtn)) { setTimeout(() => { if (rulesData.hideBarButNoStop || rulesData.hideBar) { changeHideBar(!isHideBar); } if (!rulesData.hideBarButNoStop) { changeStop(!isPause); showTips(i18n(isPause ? "disable" : "enable")); } if (!isPause) { checkScrollReach(); } }, 10); } }; document.addEventListener('dblclick', dblclickHandler); clickToResetHandler = e => { if (!ruleParser.nextLinkHref) isLoading = false; }; document.addEventListener('click', clickToResetHandler); if (rulesData.dbClick2StopKey) { keydownHandler = e => { if ((rulesData.dbClick2StopCtrl && !e.ctrlKey) || (rulesData.dbClick2StopAlt && !e.altKey) || (rulesData.dbClick2StopShift && !e.shiftKey) || (rulesData.dbClick2StopMeta && !e.metaKey)) { return; } if (document.activeElement && (compareNodeName(document.activeElement, ["input", "textarea"]) || document.activeElement.contentEditable == 'true')) { return; } var key = e.key.toLowerCase(); if (rulesData.dbClick2StopKey.toLowerCase() == key) { if (forceState == 1) { forceState = 0; showTips(i18n("enableSiteTips")); changeStop(false); } else { forceState = 1; showTips(i18n("disableSiteTips")); changeStop(true); sideController.remove(); } if (!ruleParser.curSiteRule.url) { setListData("forceState", location.host, forceState); setTimeout(() => { location.reload(); }, 500); } } }; document.addEventListener('keydown', keydownHandler); } if (ruleParser.curSiteRule.listenHashChange) { hashchangeHandler = () => { isPause = true; urlChanged = true; if (!ruleParser.nextLinkHref) isLoading = false; }; window.addEventListener('hashchange', hashchangeHandler, false); } let manualMode = typeof ruleParser.curSiteRule.manualMode == 'undefined' ? rulesData.manualMode : ruleParser.curSiteRule.manualMode; if (manualMode) { manualModeKeyHandler = e => { if (document.activeElement && (compareNodeName(document.activeElement, ["input", "textarea"]) || document.activeElement.contentEditable == 'true')) { return; } if (e.keyCode == 39) { clickNext(); } else if (e.keyCode == 37) { history.back(); } }; document.addEventListener('keydown', manualModeKeyHandler); pagetualNextHandler = () => { clickNext(); }; document.addEventListener('pagetual.next', pagetualNextHandler, false); return; } if (rulesData.arrowToScroll) { keyupHandler = e => { if (document.activeElement && (compareNodeName(document.activeElement, ["input", "textarea"]) || document.activeElement.contentEditable == 'true')) { return; } if (e.keyCode == 39) { let nextPageBar=getPageBar().nextBar; if (nextPageBar) { scrollToPageBar(nextPageBar); } else { let scrollTop = getBody(document).scrollTop || document.documentElement.scrollTop; window.scrollTo({ top: scrollTop + (window.innerHeight || document.documentElement.clientHeight), behavior: 'instant'}); } } else if (e.keyCode == 37) { let prePageBar = getPageBar().preBar; if (prePageBar) { scrollToPageBar(prePageBar); } else { let scrollTop = getBody(document).scrollTop || document.documentElement.scrollTop; window.scrollTo({ top: scrollTop - (window.innerHeight || document.documentElement.clientHeight), behavior: 'instant'}); } } }; document.addEventListener('keyup', keyupHandler); } if (!ruleParser.curSiteRule.wheel) { document.addEventListener('scroll', scrollHandler, true); } document.addEventListener('wheel', scrollHandler, true); } let hideTipsTimeout; function showTips(content, href, time, wordColor, backColor) { initView(); getBody(document).appendChild(tipsWords); tipsWords.style.color = wordColor || 0xFFFFFF; tipsWords.style.backgroundColor = backColor || 0x000; let _time = 1500; if (href) { _time = 3500; tipsWords.innerHTML = createHTML(`${content}`); tipsWords.style.pointerEvents = 'all'; } else { tipsWords.innerHTML = createHTML(content); } tipsWords.style.marginLeft = -tipsWords.offsetWidth / 2 + "px"; setTimeout(() => { tipsWords.style.marginLeft = -tipsWords.offsetWidth / 2 + "px"; }, 0); setTimeout(() => { tipsWords.style.opacity = 0.8; clearTimeout(hideTipsTimeout); hideTipsTimeout = setTimeout(() => { tipsWords.style.opacity = 0; tipsWords.style.pointerEvents = ''; }, time || _time); }, 1); } const loadmoreReg = /^\s*((点击)?加载更多|(點擊)?加載更多|load\s*more|もっと読み込む)[.…]*\s*$/i; const defaultLoadmoreSel = ".loadMore,.LoadMore,[class*='load-more'],button.show_more,.button-show-more,button[data-testid='more-results-button'],#btn_preview_remain,.view-more-btn"; function getLoadMore(doc, loadmoreBtn) { if (!loadmoreBtn || !getBody(doc).contains(loadmoreBtn) || /less/.test(loadmoreBtn.innerText)) loadmoreBtn = null; if (!ruleParser.curSiteRule.smart && !ruleParser.curSiteRule.loadMore) return null; if (loadmoreBtn) return loadmoreBtn; let btnSel = ruleParser.curSiteRule.loadMore || defaultLoadmoreSel; if (btnSel) { loadmoreBtn = getElement(btnSel, doc, null, true); } if (!loadmoreBtn) { let buttons = doc.querySelectorAll("input,button,a,div[onclick]"); for (let i = 0; i < buttons.length; i++) { let button = buttons[i]; if (!button.innerText || button.innerText.length > 20) continue; if (button && loadmoreReg.test(button.innerText)) { loadmoreBtn = button; break; } } } if (loadmoreBtn && !ruleParser.curSiteRule.loadMore && loadmoreBtn.dataset.ajax !== "true") { let href = loadmoreBtn.getAttribute("href"); if (href && href != "/" && !ruleParser.hrefIsJs(href)) { loadmoreBtn = null; } } if (loadmoreBtn && /less/.test(loadmoreBtn.innerText)) loadmoreBtn = null; if (loadmoreBtn) debug(loadmoreBtn, 'Load more button'); return loadmoreBtn; } var targetY = -1; function scrollToPageBar(bar){ let yOffset = -20; if (typeof ruleParser.curSiteRule.pageBarTop !== 'undefined') { yOffset = -ruleParser.curSiteRule.pageBarTop; } targetY = bar.getBoundingClientRect().top + window.pageYOffset + yOffset; window.scrollTo({ top: targetY, behavior: 'smooth'}); } const pageNumReg=/[&\/\?](p=|page[=\/_-]?)\d+|[_-]\d+\./; function createPageBar(url) { curPage++; sideController.setup(); let posEle = null; let scrollH = Math.max(document.documentElement.scrollHeight, getBody(document).scrollHeight); let insert = ruleParser.getInsert(); if (!insert || !insert.parentNode) return; if (forceState == 2) { posEle = getBody(document); posEle = posEle.children[posEle.children.length - 1]; } else { posEle = insert; } while (posEle && !posEle.offsetParent) { posEle = posEle.previousElementSibling || posEle.parentNode; } if (posEle) { let actualBottom = getElementBottom(posEle); bottomGap = scrollH - actualBottom + (window.innerHeight || document.documentElement.clientHeight) * rate; if (bottomGap < 100) bottomGap = 100; } else { bottomGap = 1000; } if (rulesData.opacity == 0 || ruleParser.curSiteRule.pageBar === 0) return null; url = url.replace(/#p{.*/, ""); let example = (ruleParser.curSiteRule.insertPos == 2 || ruleParser.curSiteRule.insertPos == "in") ? insert.children[0] : (insert.parentNode.children[0] || insert); while (example && (compareNodeName(example, ["script", "style"]) || example.className == "pagetual_pageBar")) { example = example.nextElementSibling; } if (!example || !example.parentNode) example = insert; if (example.nodeType != 1) { example = example.previousElementSibling || example.parentNode; if (!example || example.nodeType != 1) return; } let exampleStyle = _unsafeWindow.getComputedStyle(example); let inTable, inLi; if (forceState == 2) { inTable = inLi = false; } else { inTable = compareNodeName(example.parentNode, ["table", "tbody"]) || compareNodeName(example, ["tr", "tbody"]) || exampleStyle.display == "table-row" || (example.nextElementSibling && compareNodeName(example.nextElementSibling, ["tr", "tbody"])); inLi = compareNodeName(example, ["li"]) || (example.nextElementSibling && compareNodeName(example.nextElementSibling, ["li"])); } let pageBar = document.createElement(inTable ? "tr" : (inLi ? "li" : "div")); let upSpan = document.createElement("span"); let downSpan = document.createElement("span"); let pageText = document.createElement("a"); let pageNum; pageBar.className = isHideBar ? "pagetual_pageBar hide" : "pagetual_pageBar"; pageBar.id = "pagetual_pageBar" + curPage; pageBar.setAttribute("translate", "no"); if (isPause) { pageBar.classList.add("stop"); } pageBar.style.cssText = pageBarStyle; pageBar.title = i18n(isPause ? "enable" : "disable"); upSpan.innerHTML = createHTML(upSvg); upSpan.children[0].style.cssText = upSvgCSS; upSpan.title = i18n("toTop"); downSpan.innerHTML = createHTML(downSvg); downSpan.children[0].style.cssText = downSvgCSS; downSpan.title = i18n("toBottom"); upSpan.style.cssText = initStyle; downSpan.style.cssText = initStyle; pageText.href = url; pageText.style.cssText = pageTextStyle; pageText.title = i18n("current"); if (openInNewTab == 1) pageText.target = "_blank"; pageBar.appendChild(upSpan); pageBar.appendChild(pageText); let touched = false; let touchBodyHandler = e => { touched = false; getBody(document).removeEventListener('touchstart', touchBodyHandler, { passive: false, capture: false }); }; pageText.addEventListener("touchstart", e => { if (touched) return; touched = true; pageText.style.pointerEvents = 'none'; setTimeout(() => { pageText.style.pointerEvents = 'all'; }, 250); getBody(document).addEventListener("touchstart", touchBodyHandler, { passive: false, capture: false }); }, { passive: false, capture: false }); if (ruleParser.nextTitle) { pageText.innerHTML = createHTML(ruleParser.nextTitle + " "); pageText.title = ruleParser.nextTitle; } if (ruleParser.curSiteRule.pageNum || pageNumReg.test(url)) { pageText.innerHTML = createHTML(pageText.innerHTML + i18n("page")); pageNum = document.createElement("span"); let num = ruleParser.getPageNumFromUrl(url, curPage); pageNum.innerHTML = createHTML(num + " "); pageNum.className = "pagetual_pageNum"; pageNum.title = i18n("inputPageNum"); pageNum.style.cssText = pageTextStyle; pageNum.style.cursor = "pointer"; pageNum.style.color = ""; pageNum.style.marginLeft = "5px"; pageNum.addEventListener("click", e => { let pageInput = prompt(i18n("inputPageNum"), num || "1"); if (pageInput) { let localPageBar = document.querySelector("#pagetual_pageBar" + pageInput); if (localPageBar) { scrollToPageBar(localPageBar); } else { let pageLink = ruleParser.getLinkByPage(url, pageInput); if (pageLink) { _GM_openInTab(pageLink, {active:true}); } } } e.preventDefault(); e.stopPropagation(); }); pageBar.appendChild(pageNum); } else { pageText.innerHTML = createHTML(pageText.innerHTML + i18n("page") + curPage + " "); } let preBtn = document.createElement("span"); preBtn.innerHTML = createHTML("∧"); preBtn.title = i18n("prevPage"); preBtn.className = "prevScreen"; preBtn.style.cssText = "display: none;text-align: center;right: unset; float: left; width: 40px; background: rgba(240, 240, 240, 0.8); position: absolute; z-index: 9999999; box-shadow: rgb(0 0 0 / 50%) 0px -5px 5px; border-radius: 20px 20px 0 0; margin-top: -30px; "; let nextBtn = document.createElement("span"); nextBtn.innerHTML = createHTML("∨"); nextBtn.title = i18n("nextPage"); nextBtn.className = "nextScreen"; nextBtn.style.cssText = "display: none;text-align: center;right: unset; float: left; width: 40px; background: rgba(240, 240, 240, 0.8); position: absolute; z-index: 9999999; box-shadow: rgb(0 0 0 / 50%) 0px 5px 5px; border-radius: 0 0 20px 20px; margin-top: 30px; "; let localPage = curPage; preBtn.addEventListener("click", e => { e.stopPropagation(); e.preventDefault(); let prePageBar = document.querySelector("#pagetual_pageBar" + (localPage - 1)); if (prePageBar) { scrollToPageBar(prePageBar); } else { let scrollTop = getBody(document).scrollTop || document.documentElement.scrollTop; targetY = scrollTop - (window.innerHeight || document.documentElement.clientHeight); window.scrollTo({ top: targetY, behavior: 'smooth'}); } }); nextBtn.addEventListener("click", e => { e.stopPropagation(); e.preventDefault(); let nextPageBar = document.querySelector("#pagetual_pageBar" + (localPage + 1)); if (nextPageBar) { scrollToPageBar(nextPageBar); } else { scrollH = Math.max(document.documentElement.scrollHeight, getBody(document).scrollHeight); targetY = scrollH || 9999999; window.scrollTo({ top: targetY, behavior: 'smooth'}); } }); if (!rulesData.hideBarArrow) { pageText.insertBefore(preBtn, pageText.firstChild); pageText.insertBefore(nextBtn, pageText.firstChild); } if (curForceIframe) { let bgRing = document.createElement("span"); bgRing.className = "refreshRing"; bgRing.style.display = "none"; bgRing.innerHTML = createHTML(upSvg); pageText.title = "Refresh"; pageText.appendChild(bgRing); pageText.addEventListener("click", e => { e.stopPropagation(); if (e.ctrlKey || e.altKey || e.shiftKey || e.metaKey) return; e.preventDefault(); let nextEle = pageBar && pageBar.nextElementSibling; if (nextEle && nextEle.name == 'pagetual-iframe') { if (curForceIframe == nextEle) { nextEle.setAttribute("loaded", "refresh"); } nextEle.src = nextEle.src; } }); } else if (rulesData.pageBarMenu) { pageText.addEventListener("click", e => { e.stopPropagation(); if (e.ctrlKey || e.altKey || e.shiftKey || e.metaKey) return; e.preventDefault(); picker.start(); }); } pageBar.appendChild(downSpan); if (forceState == 2) { pageBar.style.width = "99%"; } else { let parentStyle = _unsafeWindow.getComputedStyle(example.parentNode); let parentWidth = example.parentNode.offsetWidth || parseInt(parentStyle.width); pageBar.style.width = parentWidth - parseInt(parentStyle.paddingLeft) - parseInt(parentStyle.paddingRight) - 10 + "px"; pageBar.style.margin = '10px 5px'; if (parentStyle.display == "grid" || parentStyle.display == "inline-grid") { pageBar.style.gridColumn = "1/-1"; } if (inTable) { example = compareNodeName(example, ["tr", "tbody"]) ? example : example.nextElementSibling || example; if (compareNodeName(example, ["tbody"])) example = example.querySelector("tr"); let nextTr = example; while (nextTr && nextTr.children.length == 0) nextTr = nextTr.nextElementSibling; if (nextTr) example = nextTr; let tdNum = 0; if (exampleStyle.display == "table-row") { [].forEach.call(example.children, el => { tdNum += el.colSpan || 1; }); } else { [].forEach.call(example.children, el => { if (compareNodeName(el, ["td", "th"])) { tdNum += el.colSpan || 1; } }); } pageBar.style.cssText = ""; pageBar.style.display = "table-row"; pageBar.style.backgroundColor = "unset"; pageBar.style.lineHeight = "20px"; pageBar.style.boxShadow = ""; let td = document.createElement("td"); td.colSpan = tdNum || 1; let inTd = document.createElement("div"); inTd.style.cssText = pageBarStyle; inTd.style.display = ""; inTd.className = pageBar.className; pageBar.className = ""; inTd.appendChild(upSpan); inTd.appendChild(pageText); if (pageNum) inTd.appendChild(pageNum); inTd.appendChild(downSpan); td.appendChild(inTd); pageBar.appendChild(td); } else if (inLi) { example = compareNodeName(example, ["li"]) ? example : example.nextElementSibling || example; pageBar.style.opacity = 1; pageBar.style.display = getComputedStyle(example).display; pageBar.style.backgroundColor = "unset"; pageBar.style.lineHeight = "20px"; pageBar.style.boxShadow = ""; pageBar.style.border = ""; pageBar.style.maxWidth = "unset"; pageBar.style.flex = "auto"; let inTd = document.createElement("div"); inTd.style.cssText = pageBarStyle; inTd.style.display = ""; inTd.style.margin = "0" inTd.style.padding = "0 0"; inTd.style.textAlign = "center"; inTd.style.minWidth = "150px"; inTd.style.width = 'calc(100% - 20px)'; inTd.className = pageBar.className; pageBar.className = ""; inTd.appendChild(upSpan); inTd.appendChild(pageText); if (pageNum) inTd.appendChild(pageNum); inTd.appendChild(downSpan); if (pageBar.style.display === 'table-row') { let td = document.createElement("td"); td.colSpan = example.children.length; td.style.width = '100%'; td.appendChild(inTd); pageBar.appendChild(td); } else { inTd.style.width = '100%'; pageBar.appendChild(inTd); } } } upSpan.addEventListener("click", e => { getBody(document).scrollTop = 0; document.documentElement.scrollTop = 0; e.preventDefault(); e.stopPropagation(); }); downSpan.addEventListener("click", e => { if (!e.altKey && !e.ctrlKey && !e.shiftKey && !e.metaKey) { changeStop(true); } pageBar.title = i18n(isPause ? "enable" : "disable"); scrollH=Math.max(document.documentElement.scrollHeight, getBody(document).scrollHeight); getBody(document).scrollTop = scrollH || 9999999; document.documentElement.scrollTop = scrollH || 9999999; e.preventDefault(); e.stopPropagation(); }); pageBar.addEventListener("click", e => { changeStop(!isPause); pageBar.title = i18n(isPause ? "enable" : "disable"); }); ruleParser.insertElement(pageBar); ruleParser.runPageBar(pageBar); return pageBar; } async function waitForElement(sel, doc) { if (!sel) return null; return new Promise((resolve) => { let checkInv = setInterval(() => { let result = getElement(sel, doc, null, true); if (result) { clearInterval(checkInv); resolve(result); } }, 100); }); } async function clickAction(sel, doc) { let btn = await waitForElement(sel, doc); emuClick(btn, doc); } async function enterAction(sel, doc) { let btn = await waitForElement(sel, doc); let eventParam = { key: "Enter", keyCode: 13, bubbles: true }; let event = new KeyboardEvent('keydown', eventParam); btn.dispatchEvent(event); event = new KeyboardEvent('keyup', eventParam); btn.dispatchEvent(event); event = new KeyboardEvent('keypress', eventParam); btn.dispatchEvent(event); } async function inputAction(sel, v, doc) { let input = await waitForElement(sel, doc); let event = new Event('focus', { bubbles: true }); input.dispatchEvent(event); let lastValue = input.value; if (input.nodeName.toUpperCase() == "INPUT") { var nativeInputValueSetter = Object.getOwnPropertyDescriptor(window.HTMLInputElement.prototype, "value").set; nativeInputValueSetter.call(input, v); } else if (input.nodeName.toUpperCase() == "TEXTAREA") { var nativeTextareaValueSetter = Object.getOwnPropertyDescriptor(window.HTMLTextAreaElement.prototype, "value").set; nativeTextareaValueSetter.call(input, v); } else { input.innerHTML = createHTML(v); } event = new Event('input', { bubbles: true }); let tracker = input._valueTracker; if (tracker) { tracker.setValue(lastValue); } input.dispatchEvent(event); event = new Event('change', { bubbles: true }); input.dispatchEvent(event); debug(input, `input ${sel}`); } function emuClick(btn, doc) { if (!doc) doc = document; let curScroll = getBody(doc).scrollTop || doc.documentElement.scrollTop; let orgHref = btn.getAttribute('href'); if (orgHref && orgHref.replace(location.origin + location.pathname, "").indexOf("#") === 0) { let hashAction = e => { e.preventDefault(); getBody(doc).scrollTop = curScroll; doc.documentElement.scrollTop = curScroll; btn.removeEventListener('click', hashAction, false); }; btn.addEventListener('click', hashAction, false); } if (!PointerEvent) return btn.click(); let eventParam = { isTrusted: true, altKey: false, azimuthAngle: 0, bubbles: true, button: 0, buttons: 0, clientX: 1, clientY: 1, cancelBubble: false, cancelable: true, composed: true, ctrlKey: false, defaultPrevented: false, detail: 1, eventPhase: 2, fromElement: null, height: 1, isPrimary: false, metaKey: false, pointerId: 1, pointerType: "mouse", pressure: 0, relatedTarget: null, returnValue: true, shiftKey: false, toElement: null, twist: 0, which: 1 }; let mouseEvent = new PointerEvent("mousedown", eventParam); btn.dispatchEvent(mouseEvent); mouseEvent = new PointerEvent("mouseup", eventParam); btn.dispatchEvent(mouseEvent); let dispatchTouchEvent = (ele, type) => { let touchEvent; try { touchEvent = document.createEvent('TouchEvent') touchEvent.initTouchEvent(type, true, true) } catch (err) { try { touchEvent = document.createEvent('UIEvent') touchEvent.initUIEvent(type, true, true) } catch (err) { touchEvent = document.createEvent('Event') touchEvent.initEvent(type, true, true) } } if (touchEvent) { try { touchEvent.targetTouches = [{ pageX: 1, pageY: 1, clientX: 1, clientY: 1, target: btn }]; touchEvent.touches = [{ pageX: 1, pageY: 1, clientX: 1, clientY: 1, target: btn }]; touchEvent.changedTouches = [{ pageX: 1, pageY: 1, clientX: 1, clientY: 1, target: btn }]; } catch (err) {} ele.dispatchEvent(touchEvent); } } dispatchTouchEvent(btn, "touchstart"); dispatchTouchEvent(btn, "touchend"); btn.click(); } function emuInput(input, v) { let result = false; if (!input) return true; let event = new Event('focus', { bubbles: true }); input.dispatchEvent(event); let lastValue = input.value; if (input.type == 'file') { let file = v; let blob = new Blob([file], { type: 'text/plain' }); file = new File([blob], 'noname.txt', { type: blob.type }); let dataTransfer = new DataTransfer(); dataTransfer.items.add(file); input.files = dataTransfer.files; v = "c:/fakepath/fakefile"; } else if (/INPUT/i.test(input.nodeName)) { var nativeInputValueSetter = Object.getOwnPropertyDescriptor(window.HTMLInputElement.prototype, "value").set; nativeInputValueSetter.call(input, v); } else if (/SELECT/i.test(input.nodeName)) { var nativeSelectValueSetter = Object.getOwnPropertyDescriptor(window.HTMLSelectElement.prototype, "value").set; nativeSelectValueSetter.call(input, v); } else if (input.nodeName.toUpperCase() == "TEXTAREA") { var nativeTextareaValueSetter = Object.getOwnPropertyDescriptor(window.HTMLTextAreaElement.prototype, "value").set; nativeTextareaValueSetter.call(input, v); } else { input.innerHTML = createHTML(v); } event = new Event('input', { bubbles: true }); let tracker = input._valueTracker; if (tracker) { tracker.setValue(lastValue); } input.dispatchEvent(event); event = new Event('change', { bubbles: true }); input.dispatchEvent(event); return result; } var failFromIframe = 0; var inCors = false; var checkRemoveIntv; function requestFromIframe(url, callback){ if (location.protocol === 'https:' && !/^https:/.test(url)) { ruleParser.noValidContent(url); } url = url.indexOf('=') == -1 ? url.replace(/#[^#]*/,"") : url; let iframe = document.createElement('iframe'); iframe.name = 'pagetual-iframe'; iframe.width = '100%'; iframe.height = '1000'; iframe.frameBorder = '0'; if (ruleParser.curSiteRule.sandbox != false) { iframe.sandbox = "allow-same-origin allow-scripts allow-popups allow-forms"; } iframe.style.cssText = 'margin:0!important;padding:0!important;visibility:hidden!important;flex:0;opacity:0!important;pointer-events:none!important;position:fixed;top:0px;left:0px;z-index:-2147483647;'; let waitTime = 100, checkEval; ruleParser.runWait((_checkEval, _waitTime) => { if (_checkEval) { checkEval = _checkEval; } if (_waitTime) { waitTime = _waitTime; } }); if (checkRemoveIntv) clearInterval(checkRemoveIntv); checkRemoveIntv = setInterval(() => { if (!iframe || !getBody(document).contains(iframe)) { clearInterval(checkRemoveIntv); loadPageOver(); } }, 500); let loadedHandler = async e => { if (e.data != 'pagetual-iframe:DOMLoaded' && e.type != 'load') return; clearInterval(checkRemoveIntv); window.removeEventListener('message', loadedHandler, false); iframe.removeEventListener('load', loadedHandler, false); let pageEleTryTimes = 0; async function checkIframe() { if (urlChanged || isPause) { return callback(false, false); } try { let doc = iframe.contentDocument || iframe.contentWindow.document; let base = doc.querySelector("base"); ruleParser.basePath = base ? base.href : url; let eles = ruleParser.getPageElement(doc, iframe.contentWindow, pageEleTryTimes < 25); if (checkEval && !checkEval(doc)) { setTimeout(() => { checkIframe(); }, waitTime); return; } else if (eles && eles.length > 0) { await ruleParser.hookUrl(doc); callback(doc, eles); } else if (pageEleTryTimes++ < 100) { getBody(doc).scrollTop = 9999999; doc.documentElement.scrollTop = 9999999; setTimeout(() => { checkIframe(); }, waitTime); return; } else { if (failFromIframe++ > 2) { failFromIframe = 0; debug("Stop as failFromIframe"); changeStop(true); callback(false, false); } else { ruleParser.noValidContent(url); callback(false, false); } } } catch(e) { inCors = true; if (forceState === 3) { debug("Stop as cors"); isPause = true; } if (!ruleParser.curSiteRule.pageElement) { ruleParser.curSiteRule.pageElement = allOfBody; ruleParser.getInsert(true); } ruleParser.curSiteRule.action = 0; ruleParser.nextLinkHref = url; callback(false, false); nextPage(); } if (iframe && iframe.parentNode) iframe.parentNode.removeChild(iframe); } setTimeout(() => { checkIframe(); }, waitTime); }; let code = ruleParser.curSiteRule.iframeInit; if (code) { let checkReady = setInterval(() => { let doc; try { doc = iframe.contentDocument || (iframe.contentWindow && iframe.contentWindow.document); } catch(e) { clearInterval(checkReady); return; } if (doc) { clearInterval(checkReady); try { Function('win', 'iframe', '"use strict";' + code)(iframe.contentWindow, iframe); } catch(e) { debug(e); } } }, 50); } window.addEventListener('message', loadedHandler, false); iframe.addEventListener('load', loadedHandler, false); iframe.src = url; try { getBody(document).appendChild(iframe); } catch (e) { return callback(false, false); } } var emuIframe, lastActiveUrl, orgContent, meetCors = false; function emuPage(callback) { let orgPage = null, preContent = null, iframeDoc, checkTimes = 0, loadmoreBtn, pageEle, nextLink, loadmoreEnd = false, waitTimes = 80, changed = false; function returnFalse(log) { if (curPage > 1) { if (rulesData.lastPageTips) showTips(i18n("lastPage"), "", 800); } else { sideController.remove(); } debug(log); isPause = true; callback(false, false); if (emuIframe && emuIframe.parentNode) { emuIframe.parentNode.removeChild(emuIframe); emuIframe = null; } } function cloneStatus() { if (!iframeDoc) return; let inputs = document.querySelectorAll("input:not([type=button],[type=image],[type=reset],[type=submit])"); let selectOptions = document.querySelectorAll("select>option"); [...inputs].forEach(input => { let sel = geneSelector(input, true, true); let mirrorEle = iframeDoc.querySelector(sel); if (!mirrorEle) return; emuInput(mirrorEle, input.value); }); [...selectOptions].forEach(option => { let sel = geneSelector(option, true, true); let mirrorEle = iframeDoc.querySelector(sel); if (!mirrorEle) return; let selected = option.selected; mirrorEle.selected = !!selected; mirrorEle.parentNode.dispatchEvent(new Event('change')); }); } async function checkPage() { if (isPause) return loadPageOver(); try { iframeDoc = emuIframe.contentDocument || emuIframe.contentWindow.document; } catch(e) { returnFalse("Stop as cors"); return; } let waitTime = 200, checkEval; ruleParser.runWait((_checkEval, _waitTime) => { if (_checkEval) { checkEval = _checkEval; } if (_waitTime) { waitTime = _waitTime; } }); if (!orgPage) { if (!loadmoreEnd) { loadmoreBtn = getLoadMore(iframeDoc); if (loadmoreBtn && isVisible(loadmoreBtn, emuIframe.contentWindow)) { emuClick(loadmoreBtn, iframeDoc); let intv = setInterval(() => { loadmoreBtn = getLoadMore(iframeDoc); if (!loadmoreBtn || !getBody(document).contains(loadmoreBtn) || !isVisible(loadmoreBtn, emuIframe.contentWindow)) { clearInterval(intv); loadmoreEnd = true; setTimeout(() => { checkPage(); }, 500); } else if (isInViewPort(loadmoreBtn)) { emuClick(loadmoreBtn, iframeDoc); } }, 200); return; } else { loadmoreEnd = true; } } if (checkEval && !checkEval(iframeDoc)) { waitTimes = 50; setTimeout(() => { checkPage(); }, waitTime); return; } else { if (!nextLink || !nextLink.offsetParent) nextLink = await ruleParser.getNextLink(iframeDoc, true); if (nextLink) pageEle = ruleParser.getPageElement(iframeDoc, emuIframe.contentWindow, true); if (!pageEle || pageEle.length == 0 || !nextLink) { getBody(iframeDoc).scrollTop = 9999999; iframeDoc.documentElement.scrollTop = 9999999; if (waitTimes-- > 0) { setTimeout(() => { checkPage(); }, waitTime); return; } } } if (!pageEle || pageEle.length == 0) { returnFalse("Stop as no page when emu"); return; } pageEle = [].filter.call(pageEle, ele => {return ele && !compareNodeName(ele, ["style", "script", "meta"])}); if (compareNodeName(pageEle[0], ["ul"]) || pageEle.length == 1) pageEle = pageEle[0]; else if (pageEle[0].parentNode == pageEle[1].parentNode) { pageEle = pageEle[0].parentNode; } else { pageEle = pageEle[0]; } if (ruleParser.curSiteRule.smart && orgContent != pageEle.innerHTML) { orgContent = pageEle.innerHTML; if (waitTimes-- > 0) { setTimeout(() => { checkPage(); }, 500); return; } } orgPage = pageEle; if (nextLink) { if (compareNodeName(orgPage, ["img"])) { if (!ruleParser.curSiteRule.lazyImgSrc) ruleParser.curSiteRule.lazyImgSrc = "0"; if (orgPage.src) { orgContent = orgPage.src; } else { setTimeout(() => { checkPage(); }, 500); return; } } else { orgContent = orgPage.innerHTML; } preContent = orgContent; if (!isVisible(nextLink, emuIframe.contentWindow)) { returnFalse("Stop as next hide when emu"); } else { emuClick(nextLink, iframeDoc); setTimeout(() => { checkPage(); }, 500); } } else { returnFalse("Stop as no next when emu"); } return; } if (!ruleParser.checkStopSign(nextLink, iframeDoc)) { return returnFalse("Stop as stopSign"); } if (checkTimes++ > 200) { returnFalse("Stop as timeout when emu"); return; } let eles = ruleParser.getPageElement(iframeDoc, emuIframe.contentWindow, true), checkItem; if (eles && eles.length > 0) { eles = [].filter.call(eles, ele => {return ele && !compareNodeName(ele, ["style", "script", "meta"])}); if (compareNodeName(eles[0], ["ul"]) || eles.length == 1) checkItem = eles[0]; else if (eles[0].parentNode == eles[1].parentNode) { checkItem = eles[0].parentNode; } else { checkItem = eles[0]; } } if (!checkItem || (checkEval && !checkEval(iframeDoc))) { if (checkEval) checkTimes = 0; setTimeout(() => { checkPage(); }, waitTime); } else { let checkInner; if (compareNodeName(checkItem, ["img"])) { if (checkItem.src) { checkInner = checkItem.src; } else { setTimeout(() => { checkPage(); }, waitTime); return; } } else { checkInner = checkItem.innerHTML; } if (orgPage != checkItem || checkInner != preContent) { changed = true; orgPage = checkItem; preContent = checkInner; setTimeout(() => { checkPage(); }, 500); } else if (changed) { checkTimes = 0; if (orgContent == preContent && (ruleParser.curSiteRule.smart || ruleParser.curSiteRule.stopSame)) { returnFalse("Stop as same content"); } else { orgContent = preContent; await ruleParser.hookUrl(iframeDoc); callback(iframeDoc, eles); } } else { if (checkTimes % 10 === 5) { if (!nextLink || !nextLink.offsetParent) { nextLink = await ruleParser.getNextLink(iframeDoc, true); } if (nextLink) { emuClick(nextLink, iframeDoc); } } setTimeout(() => { checkPage(); }, waitTime); } } } if (!emuIframe) { let loaded = false; emuIframe = document.createElement('iframe'); emuIframe.name = 'pagetual-iframe'; let notSetSandbox = typeof ruleParser.curSiteRule.sandbox == 'undefined'; if (notSetSandbox || ruleParser.curSiteRule.sandbox == true) { emuIframe.sandbox = "allow-same-origin allow-scripts allow-popups allow-forms"; } else if (ruleParser.curSiteRule.sandbox) { emuIframe.sandbox = ruleParser.curSiteRule.sandbox; } emuIframe.width = '100%'; emuIframe.height = '100%'; emuIframe.frameBorder = '0'; emuIframe.style.cssText = 'margin:0!important;padding:0!important;flex:0;opacity:0!important;pointer-events:none!important;position:fixed;top:0px;left:0px;z-index:-2147483647;'; emuIframe.addEventListener("load", e => { try { iframeDoc = emuIframe.contentDocument || emuIframe.contentWindow.document; } catch(e) { if (e.message && e.message.indexOf("cross-origin") != -1 && notSetSandbox && emuIframe.hasAttribute("sandbox")) { emuIframe.removeAttribute("sandbox"); meetCors = true; callback(false, false); if (ruleParser.curSiteRule.smart) { ruleParser.findNoNext(); } } else { returnFalse("Stop as cors"); } return; } setTimeout(async () => { if (meetCors && ruleParser.curSiteRule.smart) { ruleParser.smartRules = ruleParser.smartRules.filter(item => {return item && item.url !== ruleParser.curSiteRule.url;}); storage.setItem("smartRules", ruleParser.smartRules); } meetCors = false; let code = ruleParser.curSiteRule.init; if (code) { try { await new AsyncFunction('doc','win','iframe','click', 'enter', 'input', 'sleep', '"use strict";' + code)(iframeDoc, emuIframe.contentWindow, emuIframe, async sel => {await clickAction(sel, iframeDoc)}, async sel => {await enterAction(sel, iframeDoc)}, async (sel, v) =>{await inputAction(sel, v, iframeDoc)}, async time => {await sleep(time)}); } catch(e) { debug(e); } } else { let refreshByClickSel = ruleParser.curSiteRule.refreshByClick; if (iframeDoc && refreshByClickSel) { let clickBtn = await waitForElement(refreshByClickSel, iframeDoc); await sleep(500); cloneStatus(); emuClick(clickBtn, iframeDoc); await sleep(500); } } if (loaded) return; loaded = true; checkPage(); }, 500); }); let code = ruleParser.curSiteRule.iframeInit; if (code) { let checkReady = setInterval(() => { try { iframeDoc = emuIframe.contentDocument || (emuIframe.contentWindow && emuIframe.contentWindow.document); } catch(e) { clearInterval(checkReady); return; } if (iframeDoc) { clearInterval(checkReady); try { Function('win', 'iframe', '"use strict";' + code)(emuIframe.contentWindow, emuIframe); } catch(e) { debug(e); } } }, 50); } if (!lastActiveUrl) lastActiveUrl = location.href; emuIframe.src = lastActiveUrl; getBody(document).appendChild(emuIframe); } else { if (emuIframe.src != lastActiveUrl || meetCors) { emuIframe.src = lastActiveUrl; return; } checkPage(); } } var scrollToResizeInited = false; var resizePool = []; var scrollingToResize = false; function isTouchViewPort(element) { const viewWidth = window.innerWidth || document.documentElement.clientWidth; const viewHeight = window.innerHeight || document.documentElement.clientHeight; const { top, right, bottom, left, } = element.getBoundingClientRect(); return ( top < viewHeight && left < viewWidth && right > 0 && bottom > 0 ); } function resizeIframe(iframe, frameDoc, pageEle) { if (targetY >= 0) { window.scrollTo({ top: targetY, behavior: 'instant'}); targetY = -1; } let curScroll = getBody(document).scrollTop || document.documentElement.scrollTop; if (ruleParser.curSiteRule.smart || forceState === 2) { let height = (getBody(frameDoc).scrollHeight || getBody(frameDoc).offsetHeight || 500); if (!iframe.style.height || height - parseInt(iframe.style.height) > window.innerHeight) { iframe.style.height = height + "px"; iframe.style.minHeight = iframe.style.height; } iframe.style.width = "100%"; frameDoc.documentElement.scrollTop = 0; frameDoc.documentElement.scrollLeft = 0; } else { if (pageEle) { if (document.body.scrollWidth) frameDoc.documentElement.style.width = document.body.scrollWidth + "px"; let fitWidth = ruleParser.curSiteRule.fitWidth !== false; let targetElement = pageEle[0]; if (!targetElement) return; if (pageEle.length > 1) { targetElement = targetElement.parentNode; } let scrollHeight = targetElement.scrollHeight || targetElement.offsetHeight || 500; iframe.style.height = scrollHeight + "px"; let scrollTop = 0, scrollLeft = 0; getBody(frameDoc).scrollTop = 0; getBody(frameDoc).scrollLeft = 0; while (targetElement && targetElement.offsetParent) { targetElement.offsetParent.scrollTop = targetElement.offsetTop; if (targetElement.offsetParent.scrollTop == 0) { scrollTop += targetElement.offsetTop; } if (fitWidth) { targetElement.offsetParent.scrollLeft = targetElement.offsetLeft; if (targetElement.offsetParent.scrollLeft == 0) { scrollLeft += targetElement.offsetLeft; } } targetElement = targetElement.offsetParent; } frameDoc.documentElement.scrollTop = scrollTop; frameDoc.documentElement.scrollLeft = scrollLeft; if (frameDoc.documentElement.scrollTop == 0 && frameDoc.documentElement.scrollLeft == 0) { getBody(frameDoc).scrollTop += scrollTop; getBody(frameDoc).scrollLeft += scrollLeft; } if (!fitWidth && iframe.style.marginLeft == '0px') { iframe.style.width = "100vw"; iframe.style.maxWidth = "100vw"; iframe.style.minWidth = "100vw"; var cWidth = document.body.clientWidth || document.documentElement.clientWidth; var iWidth = window.innerWidth; iframe.style.marginLeft = -iframe.getBoundingClientRect().left - (iWidth - cWidth) / 2 + "px"; } } } let newScroll = getBody(document).scrollTop || document.documentElement.scrollTop; if (newScroll != curScroll) { getBody(document).scrollTop = curScroll; document.documentElement.scrollTop = curScroll; } } function scrollToResize(e) { if (scrollingToResize) return; else { scrollingToResize = true; let resizeHandler = () => { let touched = 0; for (let i = 0; i < resizePool.length; i++) { let resizeArr = resizePool[i]; let iframe = resizeArr[1](); if (isTouchViewPort(iframe)) { touched++; let pageEle = resizeArr[0](); let frameDoc = resizeArr[2](); resizeIframe(iframe, frameDoc, pageEle); } else if (touched) { if (touched == 1) { let pageEle = resizeArr[0](); let frameDoc = resizeArr[2](); resizeIframe(iframe, frameDoc, pageEle); } break; } else if (!iframe.parentNode) { resizePool.splice(i, 1); break; } } }; setTimeout(() => { scrollingToResize = false; }, 300); resizeHandler(); } } var curForceIframe; function forceIframe(url, callback) { url = url.indexOf('=') == -1 ? url.replace(/#[^#]*/,"") : url; let curIframe = document.createElement('iframe'), iframeDoc, pageElement = null, inAction = true; let loadedHandler = () => { let getPageEle = () => { if (ruleParser.curSiteRule.smart) { return null; } else { if (!pageElement || pageElement.length === 0 || !pageElement[0].offsetParent) { pageElement = ruleParser.getPageElement(iframeDoc, curIframe.contentWindow); } return pageElement; } }; resizeIframe(curIframe, iframeDoc, getPageEle()); let loaded = curIframe.getAttribute("loaded"); if (loaded == "true") { return; } curIframe.setAttribute("loaded", "true"); let getIframe = () => { return curIframe; }; let getFrameDoc = () => { return iframeDoc; }; ruleParser.insertPage(iframeDoc, [], url, ele => { callback(curIframe, loaded == "refresh"); inAction = false; }, true); if (!loaded) { resizePool.push([getPageEle, getIframe, getFrameDoc]); } }; let checkIframeTimer = setInterval(() => { if (!curIframe.parentNode) { clearInterval(checkIframeTimer); return curIframe.getAttribute("loaded") == "true" || callback(false); } }, 500); let code = ruleParser.curSiteRule.iframeInit; if (code) { let checkReady = setInterval(() => { try { iframeDoc = curIframe.contentDocument || (curIframe.contentWindow && curIframe.contentWindow.document); } catch(e) { clearInterval(checkReady); return; } if (iframeDoc) { clearInterval(checkReady); try { Function('win', 'iframe', '"use strict";' + code)(curIframe.contentWindow, curIframe); } catch(e) { debug(e); } } }, 50); } curIframe.name = 'pagetual-iframe'; curIframe.sandbox = "allow-same-origin allow-scripts allow-popups allow-forms"; curIframe.frameBorder = '0'; curIframe.scrolling = "no"; curIframe.style.cssText = 'display: block; visibility: visible; float: none; clear: both; width: 100%; height: 0; background: initial; border: 0px; border-radius: 0px; margin: 0px; padding: 0px; z-index: 2147483645;content-visibility: auto;contain-intrinsic-size: auto 300px;'; curIframe.addEventListener("load", async e => { clearInterval(checkIframeTimer); if (isPause) return callback(false); try { iframeDoc = curIframe.contentDocument || curIframe.contentWindow.document; } catch(e) { debug("Stop as cors"); isPause = true; callback(false); return; } let css = ruleParser.curSiteRule.css || rulesData.customCss; if (css) { let cssArr = css.split("inIframe:"); if (cssArr && cssArr.length > 1) { let styleEle = iframeDoc.createElement("style"); styleEle.innerHTML = cssArr[1]; iframeDoc.head.appendChild(styleEle); } } loadedHandler(); iframeDoc.addEventListener('wheel', e => { document.dispatchEvent(new Event('wheel')); }, true); }); let checkTimes = 0, findPageEle = false; let forceRefresh = e => { if (inAction || !iframeDoc) return; inAction = true; let foundNext = () => { document.removeEventListener("scroll", forceRefresh); } setTimeout(async () => { inAction = false; if (!ruleParser.nextLinkHref && !isPause) { checkTimes++; await ruleParser.getNextLink(iframeDoc, true); if (ruleParser.nextLinkHref) { foundNext(); if (isLoading) isLoading = false; } else if (checkTimes >= 10) { foundNext(); } else if (checkTimes >= 3 && !findPageEle) { if (!pageElement) pageElement = ruleParser.getPageElement(iframeDoc, curIframe.contentWindow); if (!pageElement) { inAction = true; curIframe.contentWindow.location.reload(); } else { findPageEle = true; } } } else { foundNext(); } }, 100); }; document.addEventListener("scroll", forceRefresh); curIframe.src = url; curForceIframe = curIframe; let insert = ruleParser.getInsert(); let body = getBody(document); let curScroll = body.scrollTop || document.documentElement.scrollTop; if (forceState == 2) { document.documentElement.appendChild(loadingDiv); body.appendChild(curIframe); let bodyStyle = getComputedStyle(body); if (bodyStyle.display == "flex" || bodyStyle.display == "inline-flex") { body.style.flexDirection = "column"; } } else { ruleParser.insertElement(curIframe); } body.scrollTop = curScroll; document.documentElement.scrollTop = curScroll; if (!scrollToResizeInited) { scrollToResizeInited = true; document.addEventListener("scroll", scrollToResize); } return curIframe; } function loadPageOver() { isLoading = false; stopScroll = true; setTimeout(() => {stopScroll = false;}, 300); if (loadingDiv.parentNode) { loadingDiv.parentNode.removeChild(loadingDiv); } if (rate !== 1 && !clickMode) { setTimeout(() => { if (distToBottom() < bottomGap) { nextPage(); } }, 1); } } function checkAutoLoadNum() { if (autoLoadNum >= 0) { if (autoLoadNum !== 0 && --autoLoadNum === 0) { autoLoadNum = -1; } else { setTimeout(() => nextPage(), 1); } } } async function nextPage() { if (typeof ruleParser.curSiteRule.manualMode === 'undefined' ? rulesData.manualMode : ruleParser.curSiteRule.manualMode) return; if (clickMode) return; if (isPause || isLoading || forceState === 1) return; if (ruleParser.curSiteRule.delay) { try { let checkDelay = ((typeof ruleParser.curSiteRule.delay === 'function') ? ruleParser.curSiteRule.delay : Function('"use strict";' + ruleParser.curSiteRule.delay))(); if (!checkDelay) return; } catch(e) { debug(e); } } let nextLink = ruleParser.nextLinkHref; if (!nextLink) { isLoading = true; if (curPage === 1) { await ruleParser.getNextLink(document, true); nextLink = ruleParser.nextLinkHref; } if (!nextLink) { if (curPage === 1) { if (ruleParser.curSiteRule.pinUrl) { setTimeout(() => {isLoading = false;}, 500); } else if (tryTimes++ < 3) { setTimeout(() => {isLoading = false;}, 1000); } else { ruleParser.findNoNext(); } } else if (rulesData.lastPageTips && !showedLastPageTips) { showTips(i18n("lastPage"), "", 800); showedLastPageTips = true; } return; } isLoading = false; } showedLastPageTips = false; let pvGallery = document.querySelector("span.pv-gallery-container"); if (pvGallery && pvGallery.style.display !== "none") return; let insert = ruleParser.getInsert(); if (insert) { if (curPage === 1) initView(); /*if (curPage == 1) { window.postMessage({ command: 'pagetual.insert' }, '*'); }*/ let isJs = ruleParser.curSiteRule.action == 3 || ruleParser.hrefIsJs(nextLink); if (!isJs) { emuIframe = null; lastActiveUrl = nextLink; if (location.protocol === "https:" && /^http:/.test(nextLink)) { nextLink = nextLink.replace(/^http/, "https"); } } isLoading = true; if (curPage !== 1 || !isJs || !ruleParser.curSiteRule.smart) { ruleParser.beginLoading(loadingDiv); } let sleep = ruleParser.curSiteRule.sleep || 0; setTimeout(() => { if (ruleParser.curSiteRule.pageElementByJs) { var over = eles => { if (urlChanged || isPause) { loadPageOver(); return; } if (eles) { ruleParser.insertPage(document, eles, nextLink, () => { createPageBar(nextLink); checkAutoLoadNum(); }, true); } else { debug("Stop as no page when get by js"); isPause = true; loadPageOver(); } }; try { ((typeof ruleParser.curSiteRule.pageElementByJs === 'function') ? ruleParser.curSiteRule.pageElementByJs : Function("over", "pageNum",'"use strict";' + ruleParser.curSiteRule.pageElementByJs))(over, curPage); } catch(e) { debug(e); } } else if ((forceState === 2 || ruleParser.curSiteRule.action == 2) && !isJs) { forceIframe(nextLink, (iframe, refresh) => { if (urlChanged || isPause) { loadPageOver(); return; } if (!refresh) { let pageBar = createPageBar(nextLink); if (pageBar && iframe && iframe.parentNode) iframe.parentNode.insertBefore(pageBar, iframe); } loadPageOver(); checkAutoLoadNum(); }); } else if ((forceState === 3 || ruleParser.curSiteRule.action == 1) && !isJs) { requestFromIframe(nextLink, (doc, eles) => { if (urlChanged || isPause) { loadPageOver(); return; } if (eles) { ruleParser.insertPage(doc, eles, nextLink, () => { createPageBar(nextLink); checkAutoLoadNum(); }, true); } else loadPageOver(); }); } else { if (!isJs) { requestDoc(nextLink, (eles) => { if (urlChanged || isPause) { loadPageOver(); return; } if (eles) { createPageBar(nextLink); checkAutoLoadNum(); } else loadPageOver(); }); } else { emuPage((doc, eles) => { if (urlChanged || isPause) { loadPageOver(); return; } if (eles) { ruleParser.insertPage(doc, eles, "", () => { createPageBar(nextLink); checkAutoLoadNum(); }, true); } else loadPageOver(); }); } } }, sleep); } } function init() { if (document.readyState === 'loading' || document.readyState === 'uninitialized') { let domReady = e => { initRules(() => { initPage(); }); document.removeEventListener("DOMContentLoaded", domReady, false); }; document.addEventListener("DOMContentLoaded", domReady, false); } else { initRules(() => { initPage(); }); } } function visibilitychangeHandler() { document.removeEventListener('visibilitychange', visibilitychangeHandler); init(); } setTimeout(() => { if (document.hidden) { document.addEventListener('visibilitychange', visibilitychangeHandler); } else { init(); } }, 300); })();