// ==UserScript== // @name Pagetual // @name:zh-CN 东方永页机 // @name:zh-TW 東方永頁機 // @name:ja 東方永頁機 // @name:ru Pagetual // @name:de Pagetual // @name:es Pagetual // @name:fr Pagetual // @name:it Pagetual // @name:ko 東方永頁機 // @namespace hoothin // @version 1.9.35.17 // @description Perpetual pages - Most powerful auto-pager script. Auto loading next paginated web pages and inserting into current page. Support thousands of web sites without any rule. // @description:zh-CN 终极自动翻页 - 加载并拼接下一分页内容至当前页尾,智能适配任意网页 // @description:zh-TW 終極自動翻頁 - 加載並拼接下一分頁內容至當前頁尾,智能適配任意網頁 // @description:ja Webページを自動で読み込み継ぎ足し表示を行うブラウザ拡張です、次のページ付けされた Web ページの自動読み込みと現在のページへの挿入 ルールなしで何千もの Web サイトをサポートします。 // @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. // @description:ko 페이지가 매겨진 다음 웹 페이지를 자동으로 로드하고 현재 페이지에 삽입합니다. 규칙 없이 수천 개의 웹 사이트를 지원합니다. // @author hoothin // @license MPL License // @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 * // @downloadURL none // ==/UserScript== (function() { 'use strict'; if (window.name === 'pagetual-iframe' || (window.frameElement && window.frameElement.name === 'pagetual-iframe')) { var domloaded = function (){ window.scroll(window.scrollX, 999999); window.parent.postMessage('pagetual-iframe:DOMLoaded', '*'); }; if (window.opera) { document.addEventListener('DOMContentLoaded', domloaded, false); } else { domloaded(); } 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 < 100 || window.self.innerHeight < 100) { return; } } catch(e) { return; } } const noRuleTest = false; const lang = navigator.appName == "Netscape" ? navigator.language : navigator.userLanguage; let config = {}; switch (lang) { case "zh-CN": case "zh-SG": config = { enableDebug: "调试模式,输出信息至控制台", 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: "拼接后立即写入历史记录,否则浏览完毕后再行写入", 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: "在侧边显示翻页控制栏", hideLoadingIcon: "隐藏加载动画", hideBarArrow: "隐藏分隔条定位箭头", duplicate: "检测到永页机重复安装,请删除其他脚本管理器中的永页机!", forceStateIframe: "以 iframe 嵌入整页", forceStateDynamic: "通过 iframe 加载动态内容后取出", forceStateDisable: "在此站禁用", page: "Page ", prevPage: "上一页", nextPage: "下一页", errorRulesMustBeArray: "规则必须为数组形式!", errorJson: "JSON 格式有错,请重新检查!", editSuccess: "编辑成功", errorWrongUrl: "URL 错误, 请重新检查!", errorAlreadyExists: "已经存在!", settingsSaved: "设置已保存,刷新后生效", iframe: "强制拼接", dynamic: "动态加载", reloadPage: "编辑完成,是否立即刷新页面?", copied: "已复制", noValidContent: "没有检测到有效内容,点击查看", outOfDate: "脚本已过时,请及时更新到最新版本!" }; break; case "zh-TW": case "zh-HK": config = { enableDebug: "調試模式,輸出信息至控制台", 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: "拼接後立即寫入歷史記錄,否則瀏覽完畢後再行寫入", 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: "在側邊顯示翻頁控制欄", hideLoadingIcon: "隱藏加載動畫", hideBarArrow: "隱藏分隔條定位箭頭", duplicate: "檢測到永頁機重複安裝,請刪除其他腳本管理器中的永頁機!", forceStateIframe: "以 iframe 嵌入整頁", forceStateDynamic: "通過 iframe 加載動態內容後取出", forceStateDisable: "在此站禁用", page: "Page ", prevPage: "上一頁", nextPage: "下一頁", errorRulesMustBeArray: "規則必須為陣列形式!", errorJson: "JSON 格式有錯,請重新檢查!", editSuccess: "編輯成功", errorWrongUrl: "URL 錯誤, 請重新檢查!", errorAlreadyExists: "已經存在!", settingsSaved: "設置已保存,刷新後生效", iframe: "強制拼接", dynamic: "動態加載", reloadPage: "編輯完成,是否立即刷新頁面?", copied: "已復制", noValidContent: "沒有檢測到有效内容,點擊查看", outOfDate: "脚本已過時,請及時更新到最新版本!" }; break; case "ja": config = { enableDebug: "デバッグモード", 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: "スプライシングの直後に履歴レコードを書き込みます。それ以外の場合は、閲覧後に書き込みます", 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: "サイドバーにページング コントロール バーを表示する", hideLoadingIcon: "読み込み中のアニメーションを隠す", hideBarArrow: "分割線の位置矢印を隠す", duplicate: "Pagetual の重複インストールが検出されました。他のスクリプト マネージャで永続的なページ マシンを削除してください!", forceStateIframe: "iframe にページ全体を埋め込む", forceStateDynamic: "iframe 経由で動的コンテンツを読み込む", forceStateDisable: "このステーションでのページめくりを無効にする", 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: "スクリプトが古くなっています。最新バージョンに更新してください。" }; break; case "ru": case "ru-RU": config = { enableDebug: "Включить отладку", 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: "Записать запись истории сразу после вставки, иначе записать после просмотра", 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: "Показать справа панель перемещения по вкладке", hideLoadingIcon: "Скрыть анимацию загрузки", hideBarArrow: "Скрыть кнопки перемещения на разделителе", duplicate: "Похоже, Pagetual установлен несколько раз. Пожалуйста, удалите Pagetual из других менеджеров скриптов!", forceStateIframe: "Вставить полную страницу как iframe", forceStateDynamic: "Загружать динамический контент через iframe", forceStateDisable: "Отключить перелистывание страниц на этой станции", page: "Страница ", prevPage: "Предыдущая страница", nextPage: "Следующая страница", errorRulesMustBeArray: "Правила должны быть массивом!", errorJson: "Ошибка разбора JSON. Пожалуйста, исправьте его", editSuccess: "Редактирование успешно", errorWrongUrl: "Ссылка некорректна. Пожалуйста, исправьте её", errorAlreadyExists: "Уже существует!", settingsSaved: "Настройки сохранены. Обновите страницы", iframe: "iframe", dynamic: "Динамически", reloadPage: "Редактирование завершено. Обновить страницу?", copied: "Скопировано", noValidContent: "Действительный контент не обнаружен, нажмите для просмотра", outOfDate: "Скрипт устарел, своевременно обновляйте до последней версии!" }; break; default: config = { enableDebug: "Enable debug output", 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 page manually", 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 with [Pagetual] format. 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 run (black list 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 history after page turning", enableHistoryAfterInsert: "Write history immediately after splicing, otherwise write after browsing", 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: "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 middle of the page bar to open the 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", 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", 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, click to view", outOfDate: "The script is outdated, update to the latest version in time!" }; break; } var enableDebug = true; function i18n(name, param) { return config[name] ? config[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 ); } }; var _GM_xmlhttpRequest, _GM_registerMenuCommand, _GM_notification, _GM_addStyle, _GM_openInTab, _GM_info, _GM_setClipboard; 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).then(response => response.text()).then(data => {let res = {response:data}; f.onload(res)}).catch(e => f.onerror(e))}; } 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) => {}; } 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(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)}; } 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) => {}; } var _unsafeWindow = (typeof unsafeWindow == 'undefined') ? window : unsafeWindow;//兼容 ios userscripts 的寫法 var 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); } }; var rulesData = {}, ruleUrls, updateDate; const configPage = ["https://github.com/hoothin/UserScripts/tree/master/Pagetual", "https://hoothin.github.io/UserScripts/Pagetual/"]; 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\u0028\u006e\u0065\u0078\u0074\u005b\u0020\u005f\u002d\u005d\u003f\u0070\u0061\u0067\u0065\u007c\u006f\u006c\u0064\u0065\u0072\u0029\u005c\u0073\u002a\u005b\u203a\u003e\u2192\u00bb\u005d\u003f\u0024\u007c\u6b21\u306e\u30da\u30fc\u30b8\u007c\u005e\u6b21\u3078\u003f\u0024\u007cВперед", "i"); const nextTextReg2 = new RegExp("\u005e\u005b\u4e0b\u540e\u5f8c\u6b21\u005d\u005b\u4e00\u30fc\u0031\u005d\u003f\u005b\u7ae0\u8bdd\u8a71\u8282\u7bc0\u4e2a\u500b\u5e45\u005d", "i"); const lazyImgAttr = ["data-lazy-src", "data-lazy", "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"]; _GM_registerMenuCommand(i18n("configure"), () => { _GM_openInTab(configPage[0], {active: true}); }); _GM_registerMenuCommand(i18n("editCurrent"), () => { Picker.getInstance().start(); }); function getBody(doc) { return doc.body || doc.querySelector('body') || doc; } function getElementByXpath(xpath, contextNode, doc) { doc = doc || document; contextNode = contextNode || doc; try { var 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) { try { if (!isXPath(sel)) { return doc.querySelectorAll(sel); } } catch(e) { debug(e, 'Error selector'); } return getAllElementsByXpath(sel, doc, doc); } function getElement(sel, doc) { try { if (!isXPath(sel)) { return doc.querySelector(sel); } } catch(e) { debug(e, 'Error selector'); } return getElementByXpath(sel, doc, doc); } function geneSelector(ele, addID) { 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)) { hasId = true; selector = '#' + ele.id; } else { let className = ""; if (ele.className) { let classList = ele.classList, i = 0; for (let i = 0; i < classList.length; i++) { let c = classList[i]; if (/^[\w\-_]+$/.test(c) && !/\d{3,}/.test(c)) { className += '.' + c; } } selector += className; } let parent = ele.parentElement; if (parent) { selector = geneSelector(parent, addID) + ' > ' + selector; if (!className && !hasId && parent.children.length > 1) { let i, j = 0; for (i = 0; i < parent.children.length; i++) { if (parent.children[i].nodeName == ele.nodeName) { j++; if (parent.children[i] == ele) { break; } } } selector += (/^HTML$/i.test(parent.nodeName) ? "" : `:nth-of-type(${j})`); } } } } return selector; } function createXPathFromElement(elm) { let allNodes = document.getElementsByTagName('*'), segs; for (segs = []; elm && elm.nodeType == 1; elm = elm.parentNode) { if (/^(BODY|HTML)$/i.test(elm.nodeName)) { 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; } var escapeHTMLPolicy; if (_unsafeWindow.trustedTypes && _unsafeWindow.trustedTypes.createPolicy) { escapeHTMLPolicy = _unsafeWindow.trustedTypes.createPolicy('pagetual_default', { createHTML: (string, sink) => string }); } 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 = {}; } initSavedRules(callback) { var self = this; storage.getItem("smartRules", smartRules => { if (smartRules) self.smartRules = smartRules; storage.getItem("hpRules", hpRules => { if (hpRules) self.hpRules = hpRules; storage.getItem("customRules", customRules => { if (customRules) self.customRules = customRules; storage.getItem("rules", rules => { if (rules) self.rules = rules; callback(); }); }); }); }); } saveCurSiteRule() { /*if(!this.curSiteRule || !this.curSiteRule.url || this.curSiteRule.singleUrl || 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, 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, "Update " + url + " rules fail!"); } 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; 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; } getRule(callback) { if(noRuleTest) { this.curSiteRule = {}; this.curSiteRule.url = location.origin + location.pathname; this.curSiteRule.singleUrl = true; callback(); return; } if (this.curSiteRule && this.curSiteRule.url && !this.curSiteRule.singleUrl) { if (this.ruleMatch(this.curSiteRule)) { return callback(); } } this.curSiteRule = {}; var self = this; var href = location.href.slice(0, 500); function setRule(r) { if (self.preSiteRule && self.ruleMatch(self.preSiteRule)) { self.curSiteRule = self.preSiteRule; return callback(); } if (!r.singleUrl) { self.curSiteRule = r; self.preSiteRule = r; if (r.enable !== 0) debug(r, 'Match rule'); } else if (!self.curSiteRule || !self.curSiteRule.singleUrl) 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.waitElement) { let waitTime = 500; let checkReady = () => { setTimeout(() => { if (!self.waitElement(document, r.waitElement)) { checkReady(); } else { setRule(r); } }, parseInt(waitTime)); }; checkReady(); return true; } else if (r.wait) { let waitTime = 500, checkEval, maxCheckTimes = 50; if (isNaN(r.wait)) { try { checkEval = (typeof _unsafeWindow.pagetualWait == 'undefined') ? Function("doc",'"use strict";' + r.wait) : _unsafeWindow.pagetualWait; } 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(); return true; } if (r.pinUrl) { setRule(r); return true; } if (!self.ruleMatchReady(r)) { return false; } setRule(r); return true; } return false; } for (let i in this.hpRules) { let rule = this.hpRules[i]; if (!rule || !rule.url) continue; if (rule.singleUrl) { continue; } if (checkRule(rule)) return; } for (let i in this.customRules) { let rule = this.customRules[i]; if (!rule || !rule.url) continue; if (checkRule(rule)) return; } for (let i in this.smartRules) { let rule = this.smartRules[i]; if (!rule || !rule.url) continue; if (rule.singleUrl) { let singleUrl = location.origin + location.pathname; if (singleUrl == rule.url) { setRule(rule); callback = () => { if (self.curSiteRule && !self.curSiteRule.singleUrl) { self.smartRules = self.smartRules.filter(item => {return item && item.url != singleUrl}); storage.setItem("smartRules", self.smartRules); if (self.curSiteRule.url.length > 13) { self.hpRules = self.hpRules.filter(item => {return item && item.url != self.curSiteRule.url}); self.hpRules.unshift(self.curSiteRule); if (self.hpRules.length > 30) { self.hpRules.pop(); } storage.setItem("hpRules", self.hpRules); } } }; break; } continue; } } let r = 0; function searchByTime() { setTimeout(() => { let end = r + 50; end = end > self.rules.length ? self.rules.length : end; for (; r < end; r++) { let rule = self.rules[r]; if (rule.from == 2) { delete rule.autoLoadNum; delete rule.history; delete rule.sideController; if (rule.pageBar == 0) delete rule.pageBar; } if (checkRule(rule)) return; } if (end >= self.rules.length) { setRule({url: location.origin + location.pathname, singleUrl: true}); return; } else { searchByTime(); } }, 1); } searchByTime(); } 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; document.addEventListener("click", e => { if (!self.refreshing) { self.refreshing = true; setTimeout(() => { self.refreshing = false; let checkEles = getAllElements(refreshByClickSel, document); for (let i = 0; i < checkEles.length; i++) { if (checkEles[i] === e.target) { urlChanged = true; isPause = true; if (!ruleParser.nextLinkHref) isLoading = false; break; } } }, 300); } }); } } getValidHeight(ele) { if (!ele.offsetParent) return 0; let h = ele.scrollHeight; let moreChild = ele.children[0], minOffsetTop = h; while (moreChild) { if (moreChild.offsetParent && moreChild.offsetTop < minOffsetTop) { minOffsetTop = moreChild.offsetTop; } moreChild = moreChild.nextElementSibling; } if (minOffsetTop != h && minOffsetTop > 0) { h -= minOffsetTop; } return h; } getPageElement(doc, curWin, dontFind) { if (doc == document && this.docPageElement) { let parent = this.docPageElement && this.docPageElement[0]; while (parent && !/^BODY$/i.test(parent.nodeName)) { parent = parent.parentNode; } if (parent && /^BODY$/i.test(parent.nodeName)) { 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 (pageElement && pageElement.length === 1 && pageElement[0].style.display === 'none') { pageElement = [body]; } if (this.curSiteRule.singleUrl && pageElement && pageElement.length > 0 && /^TR$/i.test(pageElement[0].nodeName)) { let mainTr = this.insert.parentNode.querySelector('tr'), mainTdNum = 0, newTdNum = 0; [].forEach.call(mainTr.children, el => { if (/^(TD|TH)$/i.test(el.nodeName)) { mainTdNum += el.colSpan || 1; } }); [].forEach.call(pageElement[0].children, el => { if (/^(TD|TH)$/i.test(el.nodeName)) { newTdNum += el.colSpan || 1; } }); if (mainTdNum != newTdNum) { this.curSiteRule.pageElement = allOfBody; pageElement = [body]; this.getInsert(true); } } if ((this.curSiteRule.singleUrl || !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 needCheckNext = (doc == document && this.initNext); function checkElement(ele) { if (/^(PRE|CODE)$/i.test(ele.nodeName)) { 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]; curHeight = self.getValidHeight(ele); curWidth = parseInt(ele.offsetWidth || ele.scrollWidth); } } if (/^PICTURE$/i.test(ele.nodeName)) { self.curSiteRule.pageElement = geneSelector(ele.parentNode) + ">" + ele.nodeName.toLowerCase(); debug(self.curSiteRule.pageElement, 'Page element'); return [ele]; } if (curHeight / bodyHeight <= 0.25) { let article = doc.querySelectorAll(mainSel); if (article && article.length > 0) { if (article.length == 1) { article = article[0]; self.curSiteRule.pageElement = article.nodeName.toLowerCase() + (article.id ? "#" + article.id : "") + (article.className ? "." + article.className : "") + ">*"; debug(self.curSiteRule.pageElement, 'Page element'); return article.children; } else { self.curSiteRule.pageElement = mainSel; debug(self.curSiteRule.pageElement, 'Page element'); return article; } } self.curSiteRule.pageElement = allOfBody; debug(self.curSiteRule.pageElement, 'Page element'); return [body]; } if (/^FORM$/i.test(ele.nodeName) && 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 (/^P$/i.test(ele.parentNode.nodeName)) 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 = curWidth * 0.38; let isHori, preOffsetTop = -1; let articleNum = 0; for (i = 0; i < ele.children.length; i++) { let curNode = ele.children[i]; if (/^(CANVAS|NAV)$/i.test(curNode.nodeName)) continue; if (!curNode.offsetParent) continue; if (!/^IMG$/i.test(curNode.nodeName) && curNode.querySelector('img') == null && /^\s*$/.test(curNode.innerText)) continue; if (needCheckNext && !curNode.contains(self.initNext) && getElementTop(curNode) > windowHeight) { continue; } if (/^ARTICLE$/i.test(curNode.nodeName)) articleNum++; let h = self.getValidHeight(curNode); let w = curNode.scrollWidth; if (isNaN(h) || isNaN(w)) continue; isHori = preOffsetTop == curNode.offsetTop ? true : (preOffsetTop == -1 ? (curNode.nextElementSibling && curNode.nextElementSibling.offsetTop == curNode.offsetTop) : false); if (isHori && h <= 50) continue; let a = h * w, moreChild = curNode.children[0]; while (moreChild) { let ch = self.getValidHeight(moreChild); let cw = moreChild.scrollWidth; if (h < ch) { h = ch; } if (moreChild.innerText != "" && ch && cw) { a += ch * cw; } moreChild = moreChild.nextElementSibling; } let isMax = false; if (isHori) { if (curMaxEle) { if (curWidth < w) { isMax = true; } else if (curWidth < w + 300 && curMaxArea < a) { 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 (curMaxEle && w < minWidth) { continue; } } else { if (h < minHeight) { if (!needCheckNext || h < windowHeight || !ele.contains(self.initNext)) { continue; } } } curHeight = h; curMaxArea = a; curWidth = w; curMaxEle = curNode; preOffsetTop == curNode.offsetTop; } } if (curMaxEle) { let sameClassNum = 0; 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++; } } } if (sameClassNum < 5) { return checkElement(curMaxEle); } } if (ele.parentNode.children.length == 1 && curWin.getComputedStyle(ele.parentNode).float == 'none') ele = ele.parentNode; else if (/^(P|BR|TD)$/i.test(ele.nodeName)) ele = ele.parentNode; else if (/^TBODY$/i.test(ele.nodeName)) { 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 pf = false; if (ele.parentNode) { let paDisplay = curWin.getComputedStyle(ele.parentNode).display; pf = paDisplay.indexOf('flex') !== -1 || paDisplay.indexOf('grid') !== -1; } if (ele.children.length > 1) { if (articleNum > 1) { self.curSiteRule.pageElement += ">article"; ele = ele.children; } else { let hasText = false; for (let i in ele.childNodes) { let child = ele.childNodes[i]; if (child.nodeType == 3 && child.nodeValue.trim() !== '') { hasText = true; ele = [ele]; break; } } if (!hasText) { let gridArea = curWin.getComputedStyle(ele).gridArea; if (gridArea && gridArea != "auto / auto / auto / auto") { self.curSiteRule.pageElement += ">*"; ele = ele.children; } else { let middleChild = ele.children[parseInt(ele.children.length / 2)]; if (curWin.getComputedStyle(ele).display === 'flex' || (rulesData.opacity != 0 && !pf)) { ele = [ele]; } else if ((middleChild.style && middleChild.style.position === "absolute" && middleChild.style.left && middleChild.style.top) || /^UL$/i.test(ele.nodeName) || curHeight == 0) { ele = [ele]; } else { self.curSiteRule.pageElement += ">*"; ele = ele.children; } } } } } else if (pf) { self.curSiteRule.pageElement += ">*"; 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 = []; } else { if (pageElement.length == 1 && /^IMG$/i.test(pageElement[0].nodeName)) { self.curSiteRule.pageBar = 0; } } } //if(pageElement)this.saveCurSiteRule(); } if (doc !== document) { this.setPageElementCss(pageElement); this.lazyImgAction(pageElement); } else if (!this.docPageElement) { this.setPageElementCss(pageElement, true); this.docPageElement = pageElement; if (this.nextLinkHref) { this.openInNewTab(pageElement); } } return pageElement; } changeVisibility() { let pageElementCss = this.curSiteRule.pageElementCss || this.curSiteRule.pageElementStyle || rulesData.pageElementCss; if (pageElementCss) 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 || this.curSiteRule.pageElementStyle || rulesData.pageElementCss; if (!pageElementCss && init && !this.nextLinkHref) return; [].forEach.call(pageElement, (ele, i) => { if (!/LINK|META|STYLE|SCRIPT/i.test(ele.nodeName)) { if (pageElementCss) { if (pageElementCss !== '0' && !ele.dataset.pagetualPageElement) { ele.style.cssText = (ele.style.cssText || '') + pageElementCss; ele.dataset.pagetualPageElement = 1; } } else { 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"; } } } } }); } } async getPage(doc) { if (document.documentElement.className.indexOf('discourse') != -1) return {}; let body = getBody(doc); let video = body.querySelector("video") || body.querySelector("iframe[id*=play],[id*=play]>iframe,iframe[src*=player],iframe[src*=m3u8]") || body.querySelector("canvas"); if (video && video.offsetParent && video.name != 'pagetual-iframe') { let scrollWidth = video.scrollWidth || video.offsetWidth; let scrollHeight = video.scrollHeight || video.offsetHeight; if (scrollWidth > 500 && scrollHeight > 500) { isPause = true; debug("Disable when large media found"); return {}; } 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) { isPause = true; debug("Disable when large media found under mobile"); return {}; } } } 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, nextJs3; let next = body.querySelector(".page-next>a") || body.querySelector("a.next_page") || body.querySelector("#next_page") || body.querySelector(".nextPage") || body.querySelector(".pagination-next>a") || body.querySelector("a[data-pagination=next]") || body.querySelector("ul.pagination>li.active+li>a") || body.querySelector(".pagination a.current+a") || body.querySelector(".pagination a[rel=next]") || body.querySelector(".pagination-nav__item--next>a") || body.querySelector("a.pageright") || body.querySelector(".page-numbers.current+a"); if (!next) { await sleep(1); next = body.querySelector("input[value='next']") || body.querySelector("input[value='Next page']") || body.querySelector("input[value='下一页']") || body.querySelector("input[value='下一頁']") || body.querySelector("a#pb_next") || body.querySelector("a#rightFix") || body.querySelector("a#btnPreGn") || body.querySelector("a.page-next") || body.querySelector("a.pages-next") || body.querySelector("a.page.right"); } if (!next) { await sleep(1); next = body.querySelector("a#next") || body.querySelector(".next>a") || body.querySelector(".next>button") || body.querySelector("a[alt=next]") || body.querySelector(".pg_area>em+a") || body.querySelector("button.next:not([disabled])") || body.querySelector(".btn_next:not([disabled])") || body.querySelector(".btn-next:not([disabled])") || body.querySelector("[title=next]") || body.querySelector("a#linkNext") || body.querySelector("a[class*=page__next]"); } if (!next) { await sleep(1); let nexts = body.querySelectorAll("a.next"); for (i = 0; i < nexts.length; i++) { if (nexts[i].style.display !== "none" && nexts[i].parentNode.style.display !== "none" && !/^\s*([上前首尾]|previous)/i.test(nexts[i].innerText.trim())) { next = nexts[i]; break; } } } if (next && (!next.href || /^(javascript|#)/.test(next.href.replace(location.href, "")))) { jsNext = next; next = null; } if (!next) { next = body.querySelectorAll("[aria-label='Next page']"); if (next && next.length == 1) { next = next[0]; if (!next.href || /^(javascript|#)/.test(next.href.replace(location.href, ""))) { if (!jsNext) jsNext = next; next = null; } } else { next = null; } } if (jsNext) { if (jsNext.className) { if (/slick|slide|gallery/i.test(jsNext.className)) jsNext = null; else if (jsNext.classList && jsNext.classList.contains('disabled')) jsNext = null; } if (jsNext) { let ariaLabel = jsNext.getAttribute("aria-label"); if (ariaLabel && /slick|slide|gallery/i.test(ariaLabel)) jsNext = null; } } if (next && nextTextReg2.test(next.innerText.trim())) { next2 = next; next = null; } if (!next) { 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) { let pageDiv = body.querySelector("div.pages>ul"); if (pageDiv) { cur = pageDiv.querySelector("li>b"); if (cur) next = cur.parentNode.nextElementSibling; if (next) next = next.querySelector("a"); } } if (!next) { 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) { let aTags = body.querySelectorAll("a,button,[type='button']"); for (i = aTags.length - 1; i >= 0; i--) { if (next1) break; if (i % 500 == 0) { await sleep(1); } let aTag = aTags[i]; if (aTag.style.display == "none") continue; if (aTag.innerText) { if (aTag.innerText.trim().length > 80) continue; if (aTag.innerText == "§") continue; } let availableHref = aTag.href && aTag.href.length < 250 && /^http/.test(aTag.href); if (availableHref && /next$/i.test(aTag.href)) continue; if (aTag.className) { if (/slick|slide|gallery/i.test(aTag.className)) continue; if (aTag.classList && aTag.classList.contains('disabled')) continue; } let ariaLabel = aTag.getAttribute("aria-label"); if (ariaLabel && /slick|slide|gallery/i.test(ariaLabel)) continue; if (aTag.parentNode) { if (aTag.parentNode.className && /slick|slide|gallery/i.test(aTag.parentNode.className)) continue; if (aTag.parentNode.classList && aTag.parentNode.classList.contains('disabled')) continue; if (/^BLOCKQUOTE$/i.test(aTag.parentNode.nodeName)) continue; } let isJs = !aTag.href || /^(javascript|#)/.test(aTag.href.replace(location.href, "")); let innerText = (aTag.innerText || aTag.value || aTag.title || ''); if (innerText && innerText.length < 250) { innerText = innerText.trim().replace(/\n.*/, '').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 (!next3) { if (/^(next\s*(»|>>|>|›|→|❯)?|>|▶|>|›|→|❯)$/i.test(innerText)) { if (isJs) { if (!nextJs3) nextJs3 = aTag; } else { next3 = aTag; } } } } } } if (isJs) continue; if (!next4) { let prevEle = aTag.previousElementSibling; if (prevEle && (/^(B|SPAN|STRONG)$/i.test(prevEle.nodeName))) { if (/^\d+$/.test(aTag.innerText.trim()) && /^\d+$/.test(prevEle.innerText.trim()) && parseInt(aTag.innerText) == parseInt(prevEle.innerText) + 1) { next4 = aTag; } } } 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) { 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 && !/^\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 (nextJs3) { nextJs3 = this.verifyNext(nextJs3, doc); } } if (!next) next = next1 || next4 || next3 || next2; if (!next) { next = jsNext || nextJs1 || nextJs3 || nextJs2; if (next && next.parentNode.className.indexOf('tab') != -1) next = null; } if (next && next.classList && next.classList.contains("results-more")) next=null; if (next && next.hasAttribute && next.hasAttribute("disabled")) next=null; return {next:next, canSave:canSave}; } verifyNext(next, doc) { if (!next) return null; if (next.previousElementSibling && /^BR$/i.test(next.previousElementSibling.nodeName)) return null; let 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) { 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; 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)(); result = url.replace(urlReg, "$1" + code + "$2"); } 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) { if (!url) return curPage; 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 curShowNum = url.replace(urlReg, "$1"); if (curShowNum != url) { let code = strMatch[0].replace(/^{/, "").replace(/}$/, ""); if (code == "$p") { return curShowNum; } else { try { let page1 = Function('"use strict";return ' + code.replace("$p", "0"))(); let page2 = Function('"use strict";return ' + code.replace("$p", "1"))(); let numGap = page2 - page1; return (curShowNum - page1) / numGap; } catch(e) { debug(e); } } } else { return curPage; } } let pageNum = url.replace(/.*[&\/\?](p=|page[=\/_-]?)(\d+).*/i, "$2"); return pageNum == url ? curPage : (pageNum.length > 4 ? curPage : pageNum); } async getNextLink(doc) { 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 + "=" + 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 _unsafeWindow.pagetualNextLinkByJs == 'undefined') ? new AsyncFunction("doc", '"use strict";' + this.curSiteRule.nextLinkByJs) : _unsafeWindow.pagetualNextLinkByJs)(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 (this.curSiteRule.nextLink) { let nextLinkSel = this.curSiteRule.nextLink; if (nextLinkSel != 0) { if (Array && Array.isArray && Array.isArray(nextLinkSel)) { nextLink = getElement(nextLinkSel[nextIndex], doc); if (!nextLink && curPage == 1 && nextIndex != 0) { nextIndex = 0; nextLink = getElement(nextLinkSel[nextIndex], doc); } } else nextLink = getElement(nextLinkSel, doc); } 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 (/^INPUT$/i.test(nextLink.nodeName) || nextLink.type == "submit") { form = nextLink.parentNode; while (form) { if (/^FORM$/i.test(form.nodeName)) 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); nextLink = page.next; if (nextLink) { if (/^INPUT$/i.test(nextLink.nodeName) || nextLink.type == "submit") { let form = nextLink.parentNode; while (form) { if (/^FORM$/i.test(form.nodeName)) break; else form = form.parentNode; } if (form) { nextLink.href = getNextLinkByForm(form, nextLink); } } let parent = nextLink; while (parent && !/^BODY$/i.test(parent.nodeName)) { 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 ((!nextLink.href || /^(javascript|#)/.test(nextLink.href.replace(location.href, ""))) && !isVisible(nextLink, _unsafeWindow)) { this.nextLinkHref = false; return null; } else { 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)) 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; } } } 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 compareUrl = this.nextLinkHref.replace(/#p{.*/, ""); if (compareUrl != "#" && (compareUrl == this.initUrl || compareUrl == this.curUrl || compareUrl == this.curUrl + "#" || compareUrl == this.oldUrl || compareUrl == this.oldUrl + "#")) { this.nextLinkHref = false; } else if (doc == document) debug(nextLink, 'Next link'); } } } else { this.nextLinkHref = false; } this.preload(); return nextLink; } checkStopSign(nextLink, doc) { if (this.curSiteRule.stopSign) { if (Array && Array.isArray && Array.isArray(this.curSiteRule.stopSign)) { let includeSel = this.curSiteRule.stopSign[0]; let excludeSel = this.curSiteRule.stopSign[1]; let curSign = this.curSiteRule.stopSign[2]; let maxSign = this.curSiteRule.stopSign[3]; if (Array && Array.isArray && Array.isArray(includeSel) && !curSign) { curSign = includeSel; includeSel = false; } if (Array && Array.isArray && Array.isArray(excludeSel) && !maxSign) { maxSign = excludeSel; excludeSel = false; } 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 && maxSign) { 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 _unsafeWindow.stopSign == 'undefined') ? Function("doc", "nextLink", '"use strict";' + this.curSiteRule.stopSign) : _unsafeWindow.stopSign)(doc, nextLink); if (stopSign) { isPause = true; this.nextLinkHref = false; return false; } } catch(e) { debug(e); } } } return true; } preload() { if (!rulesData.preload) 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=' + (document.characterSet || document.charset || document.inputEncoding), headers: { 'Referer': location.href, 'User-Agent': navigator.userAgent, "Content-Type": (postParams ? "application/x-www-form-urlencoded" : "text/html") + ";charset=" + (document.characterSet || document.charset || document.inputEncoding), }, timeout: 10000, onload: function(res) { var doc = null; try { doc = document.implementation.createHTMLDocument(''); doc.documentElement.innerHTML = res.response; var body = getBody(doc); if (body && body.firstChild) { self.lazyImgAction(body.children); } 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 = {}; } [].forEach.call(doc.images, i => { let iSrc = i.src; if (iSrc && !self.checkedImgs[iSrc]) { self.checkedImgs[iSrc] = true; let img = document.createElement('img'); img.src = iSrc; self.preloadDiv.appendChild(img); } }); 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; let img = document.createElement('img'); img.src = imgSrc; self.preloadDiv.appendChild(img); } }); } } } catch(e) { debug(e); return; } } }); } getInsert(refresh) { if (refresh) { this.insert = null; } if (this.insert && 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); } else { let pageElement = this.getPageElement(document, _unsafeWindow); if (this.curSiteRule.singleUrl && this.nextLinkHref == "#" && this.curSiteRule.pageElement === 'body') { debug("Stop as jsNext & whole body"); isPause = true; return null; } if (pageElement && pageElement.length > 0) { var pELast = pageElement[pageElement.length - 1]; 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 _unsafeWindow.pagetualPageInit=='undefined') ? Function("doc", "eles", '"use strict";' + code) : _unsafeWindow.pagetualPageInit); 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 _unsafeWindow.pagetualPageAction == 'undefined') ? Function("doc", "eles", '"use strict";' + code) : _unsafeWindow.pagetualPageAction)(doc, eles); } catch(e) { debug(e); } } this.openInNewTab(eles); this.replaceElement(doc); } openInNewTab(eles) { if (openInNewTab) { [].forEach.call(eles, ele => { if (/^A$/i.test(ele.nodeName) && 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) { if (!eles || eles.length == 0) return; let lazyImgSrc = this.curSiteRule.lazyImgSrc; if (lazyImgSrc === 0 || lazyImgSrc === '0') return; let setLazyImg = img => { let realSrc; if (lazyImgSrc) { if (!Array.isArray(lazyImgSrc)) { lazyImgSrc = [lazyImgSrc]; } realSrc = img.getAttribute(lazyImgSrc[0]); if (lazyImgSrc.length == 2) { let removeProps = lazyImgSrc[1].split(","); removeProps.forEach(prop => { img.removeAttribute(prop.trim()); }); } } if (!realSrc) { if (img.getAttribute("_src") && !img.src) { realSrc = img.getAttribute("_src"); } else { for (let i in lazyImgAttr) { let attrName = lazyImgAttr[i]; let attrValue = img.getAttribute(attrName); if (attrValue) { realSrc = attrValue; break; } } } if (!realSrc && img._lazyrias && img._lazyrias.srcset) { realSrc = img._lazyrias.srcset[img._lazyrias.srcset.length - 1]; } if (realSrc) { img.removeAttribute("srcset"); } else if (img.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; if (img.style.display == "none") { img.style.display = ""; } if (img.style.visibility == "hidden") { img.style.visibility = ""; } if (img.style.opacity == 0) { img.style.opacity = ""; } } }; [].forEach.call(eles, ele => { if (/^IMG$/i.test(ele.nodeName)) { setLazyImg(ele); } else { [].forEach.call(ele.querySelectorAll("img,picture>source"), img => { setLazyImg(img); }); [].forEach.call(ele.querySelectorAll("div[data-src][data-thumb]"), div => { div.style.setProperty("background-image", "url(" + div.dataset.src + ")", "important"); }); } if (/^A$/i.test(ele.nodeName) && ele.classList.contains("lazyload")) { if (ele.dataset.original) { ele.style.backgroundImage = 'url("' + ele.dataset.original + '")'; } } else { [].forEach.call(ele.querySelectorAll("a.lazyload"), a => { if (a.dataset.original) { a.style.backgroundImage = 'url("' + a.dataset.original + '")'; } }); } }); } initPage(callback) { let self = this; if (self.initing) return; self.initing = true; setTimeout(() => { self.initing = false; }, 100); curPage = 1; urlChanged = false; SideController.getInstance().remove(); if (this.addedElePool && this.addedElePool.length) { this.addedElePool.forEach(ele => { if (ele.parentNode) ele.parentNode.removeChild(ele); }); } this.insert = null; this.addedElePool = []; this.visibilityItems = []; this.visibleIndex = -1; this.pageDoc = document; this.nextLinkHref = null; this.curUrl = location.href; this.oldUrl = ""; this.initUrl = location.href; this.historyUrl = ""; let base = document.querySelector("base"); this.basePath = base ? base.href : location.href; this.getRule(async () => { isPause = false; if (self.curSiteRule.enable == 0) { debug("Stop as rule disable"); isPause = true; return; } //若是再亂匹配就不緩存wedata,或者只在找完本地規則之後再考慮wedata的緩存 if (self.curSiteRule.singleUrl) { delete self.curSiteRule.pageElement; 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.hpRules = self.hpRules.filter(item => {return item && item.url != self.curSiteRule.url}); self.hpRules.unshift(self.curSiteRule); if (self.hpRules.length > 30) { self.hpRules.pop(); } storage.setItem("hpRules", self.hpRules); } let css = self.curSiteRule.css || rulesData.customCss; if (css && !/^inIframe:/.test(css)) { _GM_addStyle(css); } 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); if (autoClickBtn) { emuClick(autoClickBtn); } } let code = self.curSiteRule.init; if (code) { try { ((typeof _unsafeWindow.pagetualInit == 'undefined') ? Function('doc','win','iframe','"use strict";' + code) : _unsafeWindow.pagetualInit)(null, null, null); } catch(e) { debug(e); } } await self.getNextLink(document); self.refreshByClick(); let pageElementCss = self.curSiteRule.pageElementCss || self.curSiteRule.pageElementStyle || rulesData.pageElementCss; if (pageElementCss && pageElementCss !== '0') { self.getPageElement(document, _unsafeWindow); } callback(); let initRun = typeof self.curSiteRule.initRun == 'undefined' ? rulesData.initRun : self.curSiteRule.initRun; if (initRun && initRun != false && self.nextLinkHref) nextPage(); }); } insertElement(ele) { if (!this.insert || !this.insert.parentNode) { this.getInsert(); } if (this.insert) { this.addedElePool.push(ele); if (this.curSiteRule.insertPos == 2) { this.insert.appendChild(ele); } else { this.insert.parentNode.insertBefore(ele, this.insert); } } } async insertPage(doc, eles, url, callback, tried) { this.oldUrl = this.curUrl; let oldTitle = this.pageDoc.title; this.pageDoc = doc; this.curUrl = url; 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 _unsafeWindow.pagetualPageBarText == 'undefined') ? Function("doc",'"use strict";' + this.curSiteRule.pageBarText) : _unsafeWindow.pagetualPageBarText)(doc); } catch(e) { debug(e); } } } if (curPage == 1 && !tried && !nextLink && this.curSiteRule.singleUrl && this.curSiteRule.pageElement && this.curSiteRule.action != 0) { this.curSiteRule.action = 1; this.curUrl = location.href; return false; } let curScroll = getBody(document).scrollTop || document.documentElement.scrollTop; if (callback) callback(eles); this.getInsert(); var self = this, newEles = []; if (!eles || eles.length == 0 || !self.insert || !self.insert.parentNode) { } else { isLoading = true; await this.pageInit(doc, eles); [].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 (!/^(STYLE|SCRIPT)$/.test(newEle.nodeName)) self.visibilityItems.push(newEle); self.insertElement(newEle); newEles.push(newEle); }); getBody(document).scrollTop = curScroll; document.documentElement.scrollTop = curScroll; } 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 = /^(javascript|#)/.test(this.historyUrl.replace(location.href, "")); 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; } static setup() { if (ruleParser.curSiteRule.sideController === false) return; if (!rulesData.sideController && !ruleParser.curSiteRule.sideController) return; SideController.getInstance().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; color: black!important; user-select: none; z-index: 2147483646!important; padding: 0!important; opacity: 0.5; transition: opacity .5s ease, background .5s, box-shadow .5s; } #pagetual-sideController:hover { opacity: 1; } #pagetual-sideController * { font-weight: bold; font-family: arial; font-style: normal; font-size: 20px!important; line-height: normal; float: none; } #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-btn:hover { transform: scale(1.5); color: red; } #pagetual-sideController.minSize { box-shadow: rgb(0 0 0 / 0%) 0px 0px 0px; background: #00000000!important; } #pagetual-sideController.minSize .pagetual-sideController-btn { opacity: 0; } #pagetual-sideController.minSize #pagetual-sideController-move > svg { background: white; } #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; } `; this.styleEle = _GM_addStyle(this.cssText); 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: middle;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: middle;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"; var isPause = false, isHideBar = false, isLoading = false, curPage = 1, forceState = 0, bottomGap = 1000, autoLoadNum = -1, nextIndex = 0, stopScroll = false, clickMode = false; var openInNewTab = 0; function changeStop(stop) { isPause = stop; [].forEach.call(document.querySelectorAll(".pagetual_pageBar,#pagetual-sideController"), bar => { if (isPause) { bar.classList.add("stop"); } else { bar.classList.remove("stop"); } }); } function changeHideBar(hide) { isHideBar = hide; [].forEach.call(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; if (_unsafeWindow.getComputedStyle(element).display == "none") 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 * (ruleParser.curSiteRule.rate || rulesData.rate || 1) ); } 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 urlchangeHandler = e => { if (ruleParser && ruleParser.curSiteRule && ruleParser.curSiteRule.listenUrlChange == false) return; isPause = true; setTimeout(() => { lastActiveUrl = location.href; if (location.href == configPage[0]) { location.reload(); } else { setTimeout(() => { if (guidePage.test(location.href.slice(0, 250))) { if (typeof JSONEditor !== 'undefined') { createEdit(); } else { window.onload = e => { createEdit(); } } } else { urlChanged = true; if (!ruleParser.nextLinkHref) { isLoading = false; } } }, 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 prevState = window.location.pathname + window.location.search; var checkUrlTime = 100; var checkUrlTimer; var checkFunc = () => { if (forceState == 1) return; if (checkUrlTime < 5000) { checkUrlTime += checkUrlTime>>1; } clearTimeout(checkUrlTimer); checkUrlTimer = setTimeout(checkFunc, checkUrlTime); if (document.hidden) return; let url = window.location.pathname + window.location.search; if (prevState !== url && window.location.href != ruleParser.historyUrl) { checkUrlTime = 2000; prevState = url; var e = new Event('pagetual_pushState'); e.arguments = arguments; window.dispatchEvent(e); } }; checkUrlTimer = setTimeout(checkFunc, checkUrlTime); document.addEventListener("click", e => { checkUrlTime = 100; clearTimeout(checkUrlTimer); checkUrlTimer = setTimeout(checkFunc, checkUrlTime); }); } function distToBottom () { let scrolly = window.scrollY; let windowHeight = window.innerHeight || document.documentElement.clientHeight; let scrollH = Math.max(document.documentElement.scrollHeight, getBody(document).scrollHeight); return scrollH - scrolly - windowHeight; } let checkLoadMore, scrollHandler, clickToResetHandler, dblclickHandler, keydownHandler, hashchangeHandler, manualModeKeyHandler, pagetualNextHandler, keyupHandler; function initListener () { clearInterval(checkLoadMore); 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.singleUrl) { if (ruleParser.curSiteRule.loadMore) { loadingMore = false; } } else { checkLoadMore = setInterval(() => { loadmoreBtn = getLoadMore(document); if (loadmoreBtn && isVisible(loadmoreBtn, _unsafeWindow)) { loadingMore = false; clearInterval(checkLoadMore); } else if (checkLoadMoreTimes++ > 10) { clearInterval(checkLoadMore); } }, 1000); } clickMode = typeof ruleParser.curSiteRule.clickMode == 'undefined' ? rulesData.clickMode : ruleParser.curSiteRule.clickMode; let clickNext = async () => { let nextLink = ruleParser.nextLinkHref; if (!nextLink) return; let isJs = /^(javascript|#)/.test(nextLink.replace(location.href, "")); if (isJs) { let nextBtn = await ruleParser.getNextLink(document); 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; } 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; setTimeout(() => {loadingMore = false}, 200); } } if (!isLoading && !stopScroll) { checkScrollReach(); } ruleParser.changeVisibility(); if (ruleParser.curSiteRule.lockScroll) { let curScroll = getBody(document).scrollTop || document.documentElement.scrollTop; if (isLoading && Math.abs(lastScroll - curScroll) > 350) { getBody(document).scrollTop = lastScroll; document.documentElement.scrollTop = lastScroll; } else { lastScroll = curScroll; } } }; dblclickHandler = e => { if (forceState == 1 || /^(INPUT|TEXTAREA|SELECT)$/i.test(e.target.nodeName)) return; if (!rulesData.dbClick2StopKey) { if ((rulesData.dbClick2StopCtrl && !e.ctrlKey) || (rulesData.dbClick2StopAlt && !e.altKey) || (rulesData.dbClick2StopShift && !e.shiftKey) || (rulesData.dbClick2StopMeta && !e.metaKey)) { return; } } if (!/^BODY$/i.test(e.target.nodeName) && e.target.className !== 'pagetual_pageBar') { let selStr = document.getSelection().toString(); if (selStr && selStr.trim()) { return; } } 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 && (/^(INPUT|TEXTAREA)$/i.test(document.activeElement.nodeName) || document.activeElement.contentEditable == 'true')) { return; } var key = e.key.toLowerCase(); if (rulesData.dbClick2StopKey.toLowerCase() == key) { forceState = (forceState == 1 ? 0 : 1); showTips(i18n(forceState == 1 ? "disableSiteTips" : "enableSiteTips")); if (!ruleParser.curSiteRule.url) { storage.setItem("forceState_" + location.host, forceState); location.reload(); } } }; 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 && (/^(INPUT|TEXTAREA)$/i.test(document.activeElement.nodeName) || 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 && (/^(INPUT|TEXTAREA)$/i.test(document.activeElement.nodeName) || 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: 'smooth'}); } } 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: 'smooth'}); } } }; document.addEventListener('keyup', keyupHandler); } if (!ruleParser.curSiteRule.wheel) { document.addEventListener('scroll', scrollHandler, true); } document.addEventListener('wheel', scrollHandler, true); } function showTips(content, wordColor, backColor, href) { 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; setTimeout(() => { tipsWords.style.opacity = 0; tipsWords.style.pointerEvents = ''; }, time); }, 1); } function getLoadMore(doc, loadmoreBtn) { if (!loadmoreBtn || !getBody(doc).contains(loadmoreBtn) || /less/.test(loadmoreBtn.innerText)) loadmoreBtn = null; if (!ruleParser.curSiteRule.singleUrl && !ruleParser.curSiteRule.loadMore) return null; if (loadmoreBtn) return loadmoreBtn; let btnSel = ruleParser.curSiteRule.loadMore || ".loadMore,.LoadMore,.load-more,.button-show-more,button[data-testid='more-results-button'],#btn_preview_remain"; if (btnSel) { loadmoreBtn = getElement(btnSel, doc); } if (!loadmoreBtn) { let buttons = doc.querySelectorAll("input,button,a,div[onclick]"), loadmoreReg = /^\s*(加载更多|加載更多|load\s*more|もっと読み込む)[.…]*\s*$/i; for (let i = 0; i < buttons.length; i++) { let button = buttons[i]; if (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 != "/" && !/^(javascript|#)/.test(href.replace(location.href, ""))) { loadmoreBtn = null; } } if (loadmoreBtn && /less/.test(loadmoreBtn.innerText)) loadmoreBtn = null; return loadmoreBtn; } function scrollToPageBar(bar){ let yOffset = -20; if (typeof ruleParser.curSiteRule.pageBarTop !== 'undefined') { yOffset = -ruleParser.curSiteRule.pageBarTop; } const y = bar.getBoundingClientRect().top + window.pageYOffset + yOffset; window.scrollTo({ top: y, behavior: 'smooth'}); } const pageNumReg=/[&\/\?](p=|page[=\/_-]?)\d+|[_-]\d+\./; function createPageBar(url) { curPage++; let posEle = null; let scrollH = Math.max(document.documentElement.scrollHeight, getBody(document).scrollHeight); let insert = ruleParser.getInsert(); if (!insert || !insert.parentNode) return; 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) * (ruleParser.curSiteRule.rate || rulesData.rate || 1); 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 ? insert.children[0] : (insert.parentNode.children[0] || insert); while (example && (/^(SCRIPT|STYLE)$/i.test(example.nodeName) || 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 = /^(TABLE|TBODY)$/i.test(example.parentNode.nodeName) || /^(TR|TBODY)$/i.test(example.nodeName) || exampleStyle.display == "table-row" || (example.nextElementSibling && /^(TR|TBODY)$/i.test(example.nextElementSibling.nodeName)); inLi = /^LI$/i.test(example.nodeName) || (example.nextElementSibling && /^LI$/i.test(example.nextElementSibling.nodeName)); } 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 = upSvg; upSpan.children[0].style.cssText = upSvgCSS; upSpan.title = i18n("toTop"); downSpan.innerHTML = 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); if (rulesData.pageBarMenu) { pageText.addEventListener("click", e => { e.stopPropagation(); if (e.ctrlKey || e.altKey || e.shiftKey || e.metaKey) return; e.preventDefault(); Picker.getInstance().start(); }); } 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 = ruleParser.nextTitle + " "; pageText.title = ruleParser.nextTitle; } if (ruleParser.curSiteRule.pageNum || pageNumReg.test(url)) { pageText.innerHTML += i18n("page"); pageNum = document.createElement("span"); pageNum.innerHTML = ruleParser.getPageNumFromUrl(url); 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"), "1"); if (pageInput) { let pageLink = ruleParser.getLinkByPage(url, pageInput); if (pageLink) { _GM_openInTab(pageLink, {active:true}); } } e.preventDefault(); e.stopPropagation(); }); pageBar.appendChild(pageNum); } else { pageText.innerHTML += i18n("page") + curPage; } let preBtn = document.createElement("span"); preBtn.innerHTML = "∧"; 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 = "∨"; 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; window.scrollTo({ top: scrollTop - (window.innerHeight || document.documentElement.clientHeight), 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); window.scrollTo({ top: scrollH || 9999999, behavior: 'smooth'}); } }); if (!rulesData.hideBarArrow) { pageText.insertBefore(preBtn, pageText.firstChild); pageText.insertBefore(nextBtn, pageText.firstChild); } 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 = (/^(TR|TBODY)$/i.test(example.nodeName)) ? example : example.nextElementSibling || example; if (/^TBODY$/i.test(example.nodeName)) 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 (/^(TD|TH)$/i.test(el.nodeName)) { tdNum += el.colSpan || 1; } }); } pageBar.style.display = "table-row"; pageBar.style.backgroundColor = "unset"; pageBar.style.lineHeight = "20px"; pageBar.style.boxShadow = ""; //pageBar.style.height="35px"; let td = document.createElement("td"); td.colSpan = tdNum || 1; let inTd = document.createElement("div"); inTd.style.backgroundColor = "rgb(240 240 240 / 80%)"; inTd.style.borderRadius = "20px"; inTd.style.padding = "0 0"; inTd.style.margin = "0"; inTd.style.lineHeight = "20px"; inTd.style.textAlign = "center"; inTd.style.boxShadow = "rgb(0 0 0 / 67%) 0px 0px 10px 0px"; inTd.appendChild(upSpan); inTd.appendChild(pageText); if (pageNum) inTd.appendChild(pageNum); inTd.appendChild(downSpan); td.appendChild(inTd); pageBar.appendChild(td); } else if(inLi) { example = /^LI$/i.test(example.nodeName) ? example : example.nextElementSibling || example; pageBar.style.display = getComputedStyle(example).display; pageBar.style.backgroundColor = "unset"; pageBar.style.lineHeight = "20px"; pageBar.style.boxShadow = ""; pageBar.style.maxWidth = "unset"; pageBar.style.flex = "auto"; //pageBar.style.height="35px"; let td = document.createElement("td"); td.colSpan = example.children.length; td.style.width = '100%'; let inTd = document.createElement("div"); inTd.style.backgroundColor = "rgb(240 240 240 / 80%)"; inTd.style.borderRadius = "20px"; inTd.style.margin = "0" inTd.style.padding = "0 0"; inTd.style.textAlign = "center"; inTd.style.minWidth = "150px"; inTd.appendChild(upSpan); inTd.appendChild(pageText); inTd.style.width = 'calc(100% - 20px)'; inTd.style.boxShadow = "rgb(0 0 0 / 67%) 0px 0px 10px 0px"; if (pageNum) inTd.appendChild(pageNum); inTd.appendChild(downSpan); if (pageBar.style.display === 'table-row') { 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 => { 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); if (ruleParser.curSiteRule.pageBar && ruleParser.curSiteRule.pageBar !== 0) { try { ((typeof _unsafeWindow.pagetualPageBar == 'undefined') ? Function("pageBar",'"use strict";' + ruleParser.curSiteRule.pageBar) : _unsafeWindow.pagetualPageBar)(pageBar); } catch(e) { debug(e); } } return pageBar; } function emuClick(btn) { let orgHref = btn.getAttribute('href'); if (orgHref && orgHref != "#") { btn.setAttribute('href', orgHref.replace(/#$/,"")); } 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(); if (orgHref && orgHref != "#") { setTimeout(() => btn.setAttribute('href', orgHref), 0); } } var failFromIframe = 0; var inCors = false; var checkRemoveIntv; function requestFromIframe(url, callback){ if (location.protocol === 'https:' && !/^https:/.test(url)) { showTips(i18n("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; if (ruleParser.curSiteRule.waitElement) { checkEval = doc => { return ruleParser.waitElement(doc); }; } else if (ruleParser.curSiteRule.wait) { if (isNaN(ruleParser.curSiteRule.wait)) { try { checkEval = (typeof _unsafeWindow.pagetualWait == 'undefined') ? Function("doc",'"use strict";' + ruleParser.curSiteRule.wait) : _unsafeWindow.pagetualWait; } catch(e) { debug(e); } } else { waitTime = ruleParser.curSiteRule.wait; } } if (checkRemoveIntv) clearInterval(checkRemoveIntv); checkRemoveIntv = setInterval(() => { if (!iframe || !getBody(document).contains(iframe)) { clearInterval(checkRemoveIntv); loadPageOver(); } }, 500); let loadedHandler = e => { if (e.data != 'pagetual-iframe:DOMLoaded' && e.type != 'load') return; clearInterval(checkRemoveIntv); window.removeEventListener('message', loadedHandler, false); iframe.removeEventListener('load', loadedHandler, false); let tryTimes = 0; 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); if (checkEval && !checkEval(doc)) { setTimeout(() => { checkIframe(); }, waitTime); return; } else if (eles && eles.length > 0) { callback(doc, eles); } else if (tryTimes++ < 100) { setTimeout(() => { checkIframe(); }, waitTime); return; } else { if (failFromIframe++ > 2) { failFromIframe = 0; debug("Stop as failFromIframe"); isPause = true; callback(false, false); } else { showTips(i18n("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); }; 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, times = 0, loadmoreBtn, pageEle, nextLink, loadmoreEnd = false, waitTimes = 80, changed = false; function returnFalse(log) { debug(log); isPause = true; callback(false, false); if (emuIframe && emuIframe.parentNode) { emuIframe.parentNode.removeChild(emuIframe); emuIframe = null; } } 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; if (ruleParser.curSiteRule.waitElement) { checkEval = doc => { return ruleParser.waitElement(doc); }; } else if(ruleParser.curSiteRule.wait) { if (isNaN(ruleParser.curSiteRule.wait)) { try { checkEval = (typeof _unsafeWindow.pagetualWait == 'undefined') ? Function("doc", '"use strict";' + ruleParser.curSiteRule.wait) : _unsafeWindow.pagetualWait; } catch(e) { debug(e); } } else { waitTime = ruleParser.curSiteRule.wait; } } if (!orgPage) { if (!loadmoreEnd) { loadmoreBtn = getLoadMore(iframeDoc); if (loadmoreBtn && isVisible(loadmoreBtn, iframeDoc.defaultView)) { emuClick(loadmoreBtn); let intv = setInterval(() => { loadmoreBtn = getLoadMore(iframeDoc); if (!loadmoreBtn || !getBody(document).contains(loadmoreBtn) || !isVisible(loadmoreBtn, iframeDoc.defaultView)) { clearInterval(intv); loadmoreEnd = true; setTimeout(() => { checkPage(); }, 500); } else if (isInViewPort(loadmoreBtn)) { emuClick(loadmoreBtn); } }, 200); return; } else { loadmoreEnd = true; } } if (checkEval && !checkEval(iframeDoc)) { waitTimes = 50; setTimeout(() => { checkPage(); }, waitTime); return; } else { nextLink = await ruleParser.getNextLink(iframeDoc); pageEle = ruleParser.getPageElement(iframeDoc, iframeDoc.defaultView, true); if (!pageEle || pageEle.length==0 || !nextLink) { if (waitTimes-- > 0) { setTimeout(() => { checkPage(); }, waitTime); return; } } } if (!pageEle || pageEle.length == 0) { returnFalse("Stop as no page when emu"); return; } if (/^UL$/i.test(pageEle[0].nodeName)) pageEle = pageEle[0].children; pageEle = pageEle[parseInt(pageEle.length / 2)]; while(pageEle && !pageEle.offsetParent) { if (pageEle.nextElementSibling) pageEle = pageEle.nextElementSibling; else break; } if (ruleParser.curSiteRule.singleUrl && orgContent != pageEle.innerHTML) { orgContent = pageEle.innerHTML; if (waitTimes-- > 0) { setTimeout(() => { checkPage(); }, waitTime); return; } } orgPage = pageEle; if (nextLink) { if (/^IMG$/i.test(orgPage.nodeName)) { 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, iframeDoc.defaultView)) { returnFalse("Stop as next hide when emu"); } else { emuClick(nextLink); setTimeout(() => { checkPage(); }, 500); } } else { returnFalse("Stop as no next when emu"); } return; } if (!ruleParser.checkStopSign(nextLink, iframeDoc)) return returnFalse("Stop as stopSign");; if (times++ > 200) { returnFalse("Stop as timeout when emu"); return; } let eles = ruleParser.getPageElement(iframeDoc, iframeDoc.defaultView, true), checkItem; if (eles && eles.length > 0) { checkItem = eles; if (/^UL$/i.test(eles[0].nodeName)) checkItem = eles[0].children; checkItem = checkItem[parseInt(checkItem.length / 2)]; while(checkItem && !checkItem.offsetParent) { if (checkItem.nextElementSibling) checkItem = checkItem.nextElementSibling; else break; } } if (!checkItem || (checkEval && !checkEval(iframeDoc))) { if (checkEval) times = 0; setTimeout(() => { checkPage(); }, waitTime); } else { let checkInner; if (/^IMG$/i.test(checkItem.nodeName)) { 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(); }, waitTime); } else if (changed) { times = 0; if (orgContent == preContent && (ruleParser.curSiteRule.singleUrl || ruleParser.curSiteRule.stopSame)) { returnFalse("Stop as same content"); } else { orgContent = preContent; callback(iframeDoc, eles); } } else { if (times % 10 === 1) { if (!nextLink || !nextLink.offsetParent) { nextLink = await ruleParser.getNextLink(iframeDoc); } if (nextLink) { emuClick(nextLink); } } 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 => { setTimeout(() => { 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); } else { returnFalse("Stop as cors"); } return; } meetCors = false; let code = ruleParser.curSiteRule.init; if (code) { try { Function('doc','win','iframe','"use strict";' + code)(iframeDoc, iframeDoc.defaultView, emuIframe); } catch(e) { debug(e); } } if (loaded) return; loaded = true; checkPage(); }, 500); }); 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 (ruleParser.curSiteRule.singleUrl || forceState === 2) { iframe.style.height = (getBody(frameDoc).scrollHeight || getBody(frameDoc).offsetHeight || 500) + "px"; iframe.style.minHeight = iframe.style.height; iframe.style.width = "100%"; frameDoc.documentElement.scrollTop = 0; frameDoc.documentElement.scrollLeft = 0; } else { if (pageEle) { 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.marginLeft = -iframe.getBoundingClientRect().left + "px"; } } } } function scrollToResize(e) { if (scrollingToResize) return; else { scrollingToResize = true; let resizeHandler = () => { let touched = 0; for (let i in resizePool) { 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(); } } function forceIframe(url, callback) { url = url.indexOf('=') == -1 ? url.replace(/#[^#]*/,"") : url; let curIframe = document.createElement('iframe'), iframeDoc, pageElement = null, isloaded = false, inAction = true; let loadedHandler = () => { inAction = false; let getPageEle = () => { if (ruleParser.curSiteRule.singleUrl) { return null; } else { if (!pageElement || pageElement.length === 0 || !pageElement[0].offsetParent) { pageElement = ruleParser.getPageElement(iframeDoc, iframeDoc.defaultView); } return pageElement; } }; resizeIframe(curIframe, iframeDoc, getPageEle()); if (isloaded) return; isloaded = true; let getIframe = () => { return curIframe; }; let getFrameDoc = () => { return iframeDoc; }; ruleParser.insertPage(iframeDoc, [], url, ele => { callback(curIframe); }, true); resizePool.push([getPageEle, getIframe, getFrameDoc]); }; let checkIframeTimer = setInterval(() => { if (!curIframe.parentNode) { clearInterval(checkIframeTimer); return isloaded || callback(false); } }, 500); 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: 2147483647;content-visibility: auto;contain-intrinsic-size: auto 300px;'; curIframe.addEventListener("load", e => { clearInterval(checkIframeTimer); 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) { css = css.replace(/^inIframe:/, ""); let styleEle=iframeDoc.createElement("style"); styleEle.innerHTML=css; iframeDoc.head.appendChild(styleEle); } loadedHandler(); let code = ruleParser.curSiteRule.init; if (code) { try { Function('doc','win','iframe','"use strict";' + code)(iframeDoc,iframeDoc.defaultView,curIframe); } catch(e) { debug(e); } } }); 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) { checkTimes++; await ruleParser.getNextLink(iframeDoc); if (ruleParser.nextLinkHref) { foundNext(); if (isLoading) isLoading = false; } else if (checkTimes >= 10) { foundNext(); } else if (checkTimes >= 3 && !findPageEle) { if (!pageElement) pageElement = ruleParser.getPageElement(iframeDoc, iframeDoc.defaultView); if (!pageElement) { inAction = true; curIframe.contentWindow.location.reload(); } else { findPageEle = true; } } } else { foundNext(); } }, 100); }; document.addEventListener("scroll", forceRefresh); curIframe.src = url; let insert = ruleParser.getInsert(); if (ruleParser.curSiteRule.singleUrl || forceState == 2) { getBody(document).appendChild(loadingDiv); getBody(document).appendChild(curIframe); } else { ruleParser.insertElement(curIframe); } if (!scrollToResizeInited) { scrollToResizeInited = true; document.addEventListener("scroll", scrollToResize); } return curIframe; } function loadPageOver(){ isLoading = false; stopScroll = true; let dist = distToBottom(); let rate = (ruleParser.curSiteRule.rate || rulesData.rate || 1); if (rate != 1 && !clickMode) { setTimeout(() => { if (dist < bottomGap) { nextPage(); } }, 1); } setTimeout(() => {stopScroll = false}, 300); if(loadingDiv.parentNode){ loadingDiv.parentNode.removeChild(loadingDiv); } } function checkAutoLoadNum() { if (autoLoadNum >= 0) { if (autoLoadNum != 0 && --autoLoadNum == 0) { autoLoadNum = -1; } else { setTimeout(() => nextPage(), 1); } } } var tryTimes = 0; 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 _unsafeWindow.pagetualDelay=='undefined') ? Function('"use strict";' + ruleParser.curSiteRule.delay) : _unsafeWindow.pagetualDelay)(); if (!checkDelay) return; } catch(e) { debug(e); } } let nextLink = ruleParser.nextLinkHref; if (!nextLink) { isLoading = true; if (curPage == 1) { await ruleParser.getNextLink(document); nextLink = ruleParser.nextLinkHref; } if (!nextLink) { if (curPage == 1 && (ruleParser.curSiteRule.pinUrl || tryTimes++ <= 10)) { setTimeout(() => {isLoading = false}, 500); } return; } isLoading = 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(); SideController.setup(); /*if (curPage == 1) { window.postMessage({ insert: geneSelector(ruleParser.curSiteRule.insertPos == 2 ? insert : insert.parentNode, true), command: 'pagetual.insert' }, '*'); }*/ let isJs = ruleParser.curSiteRule.action == 3 || /^(javascript|#)/.test(nextLink.replace(location.href, "")); if (!isJs) { emuIframe = null; lastActiveUrl = nextLink; if (location.protocol == "https:" && /^http:/.test(nextLink)) { nextLink = nextLink.replace(/^http/, "https"); } } isLoading = true; ruleParser.insertElement(loadingDiv); let parent = loadingDiv.parentNode; if (/^TBODY$/i.test(parent.nodeName)) { parent = parent.parentNode; } if (/^TABLE$/i.test(parent.nodeName)) { parent.parentNode.appendChild(loadingDiv); } loadingDiv.style.cssText = loadingCSS; if (curPage == 1 && isJs && ruleParser.curSiteRule.singleUrl) { loadingDiv.style.display = "none"; } let sleep = ruleParser.curSiteRule.sleep || 0; setTimeout(() => { if (ruleParser.curSiteRule.pageElementByJs) { var over = eles => { loadPageOver(); if (urlChanged || isPause) return; if (eles) { ruleParser.insertPage(document, eles, nextLink, () => { createPageBar(nextLink); checkAutoLoadNum(); }, true); } else { debug("Stop as no page when get by js"); isPause = true; } }; try { ((typeof _unsafeWindow.pagetualPageElementByJs == 'undefined') ? Function("over", "pageNum",'"use strict";' + ruleParser.curSiteRule.pageElementByJs) : _unsafeWindow.pagetualPageElementByJs)(over, curPage); } catch(e) { debug(e); } } else if ((forceState == 2 || ruleParser.curSiteRule.action == 2) && !isJs) { forceIframe(nextLink, (iframe) => { loadPageOver(); if (urlChanged || isPause) return; let pageBar = createPageBar(nextLink); if (pageBar && iframe && iframe.parentNode) iframe.parentNode.insertBefore(pageBar, iframe); checkAutoLoadNum(); }); } else if ((forceState == 3 || ruleParser.curSiteRule.action == 1) && !isJs) { requestFromIframe(nextLink, (doc, eles) => { loadPageOver(); if (urlChanged || isPause) return; if (eles) { ruleParser.insertPage(doc, eles, nextLink, () => { createPageBar(nextLink); checkAutoLoadNum(); }, true); } }); } else { if (!isJs) { requestDoc(nextLink, (eles) => { loadPageOver(); if (urlChanged || isPause) return; if (eles) { createPageBar(nextLink); checkAutoLoadNum(); } }); } else { emuPage((doc, eles) => { loadPageOver(); if (urlChanged || isPause) return; if (eles) { ruleParser.insertPage(doc, eles, "", () => { createPageBar(nextLink); checkAutoLoadNum(); }, true); } }); } } },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); })();