// ==UserScript== // @name Pagetual // @name:zh-CN 东方永页机 // @name:zh-TW 東方永頁機 // @namespace hoothin // @version 0.3.2 // @description Simply auto load the next page // @description:zh-CN 自动翻页 // @description:zh-TW 自動翻頁 // @author hoothin // @include http* // @license MIT // @icon  // @grant GM_xmlhttpRequest // @grant GM_registerMenuCommand // @grant GM_notification // @grant GM_getValue // @grant GM_setValue // @grant GM_addStyle // @grant GM.xmlhttpRequest // @grant GM.registerMenuCommand // @grant GM.notification // @grant GM.getValue // @grant GM.setValue // @grant GM.addStyle // @connect wedata.net // @connect githubusercontent.com // @run-at document-idle // @downloadURL none // ==/UserScript== (function() { 'use strict'; if (window.name === 'pagetual-iframe') { var domloaded = function (){ window.scroll(window.scrollX, 99999); window.parent.postMessage('pagetual-iframe:DOMLoaded', '*'); }; if(window.opera){ document.addEventListener('DOMContentLoaded', domloaded, false); } else { domloaded(); } return; } if(window.top != window.self){ return; } const lang = navigator.appName=="Netscape"?navigator.language:navigator.userLanguage; var i18n=(name, param)=>{ let config={}; switch (lang){ case "zh-CN": case "zh-SG": config={ saveCurrent:"保存当前设置", disable:"暂时禁用", enable:"启用翻页", toTop:"回到顶部", toBottom:"前往页尾", current:"当前页", forceIframe:"强制拼接", configure:"打开配置页", update:"立即更新规则", passSec:"更新于 #t# 秒前", passMin:"更新于 #t# 分钟前", passHour:"更新于 #t# 小时前", passDay:"更新于 #t# 天前", cantDel:"无法删除内置规则", confirmDel:"是否确认要删除此规则?", updateSucc:"更新成功", beginUpdate:"正在更新,请稍候", customUrls:"导入规则url,一行一条,AutoPagerize 格式规则需要以\"0|\"开头", customRules:"输入【东方永页机】格式的自定义规则", save:"保存设置" }; break; case "zh-TW": case "zh-HK": config={ disable:"暫時禁用", enable:"啟用翻頁", toTop:"回到頂部", toBottom:"前往頁尾", current:"當前頁", forceIframe:"強制拼接", configure:"打開配置頁", update:"立即更新規則", passSec:"更新于 #t# 秒前", passMin:"更新于 #t# 分鐘前", passHour:"更新于 #t# 小時前", passDay:"更新于 #t# 天前", cantDel:"無法刪除内置規則", confirmDel:"是否確認要刪除此規則?", updateSucc:"更新成功", beginUpdate:"正在更新,請稍候", customUrls:"導入規則url,一行一條,AutoPagerize 格式規則需要以\"0|\"開頭", customRules:"輸入【東方永頁機】格式的自定義規則", save:"存儲設置" }; break; default: config={ disable:"Disable", enable:"Enable", toTop:"To Top", toBottom:"To Bottom", current:"Current Page", forceIframe:"Force to join", configure:"Configure", update:"Update rules from url now", 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 rule url, One url per line, rules on AutoPagerize format need to start with \"0|\"", customRules:"Input custom rules with [Pagetual] format", save:"Save" }; break; } return config[name]?config[name].replace("#t#",param):name; }; var enableDebug=true; var debug=str=>{ if(enableDebug){ console.debug(str); } }; var _GM_xmlhttpRequest,_GM_registerMenuCommand,_GM_notification,_GM_addStyle; if(typeof GM_xmlhttpRequest!='undefined'){ _GM_xmlhttpRequest=GM_xmlhttpRequest; }else if(typeof GM!='undefined' && typeof GM.xmlhttpRequest!='undefined'){ _GM_xmlhttpRequest=GM.xmlhttpRequest; } if(typeof GM_registerMenuCommand!='undefined'){ _GM_registerMenuCommand=GM_registerMenuCommand; }else if(typeof GM!='undefined' && typeof GM.registerMenuCommand!='undefined'){ _GM_registerMenuCommand=GM.registerMenuCommand; } if(typeof GM_notification!='undefined'){ _GM_notification=GM_notification; }else if(typeof GM!='undefined' && typeof GM.notification!='undefined'){ _GM_notification=GM.notification; } if(typeof GM_addStyle!='undefined'){ _GM_addStyle=GM_addStyle; }else if(typeof GM!='undefined' && typeof GM.addStyle!='undefined'){ _GM_addStyle=GM.addStyle; } if(typeof _GM_xmlhttpRequest=='undefined')_GM_xmlhttpRequest=(f)=>{}; if(typeof _GM_registerMenuCommand=='undefined')_GM_registerMenuCommand=(s,f)=>{}; if(typeof _GM_notification=='undefined')_GM_notification=(s)=>{}; var _unsafeWindow=(typeof unsafeWindow=='undefined')?window:unsafeWindow; 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', 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.operaUJSStorage){ this.operaUJSStorage.setItem(key,value); }else if(this.mxAppStorage){ this.mxAppStorage.setConfig(key,value); }else if(this.supportGM){ GM_setValue(key,value); }else if(this.supportGMPromise){ GM.setValue(key,value); }else if(window.localStorage){ window.localStorage.setItem(key,value); } }, getItem:function(key,cb){ var value; if(this.operaUJSStorage){ value=this.operaUJSStorage.getItem(key); }else if(this.mxAppStorage){ value=this.mxAppStorage.getConfig(key); }else if(this.supportGM){ value=GM_getValue(key); }else if(this.supportGMPromise){ value=GM.getValue(key).then(v=>{cb(v)}); return; }else if(window.localStorage){ value=window.localStorage.getItem(key); }; cb(value); } }; _GM_registerMenuCommand(i18n("forceIframe"), ()=>{ if(ruleParser.curSiteRule.action==2){ ruleParser.curSiteRule.action=0; }else{ ruleParser.curSiteRule.action=2; } ruleParser.saveCurSiteRule(); location.reload(); }); _GM_registerMenuCommand(i18n("configure"), ()=>{ location.href="https://github.com/hoothin/UserScripts/tree/master/Pagetual"; }); 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) { throw new Error(`Invalid xpath: ${xpath}`); } } 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) { throw new Error(`Invalid xpath: ${xpath}`); } return result; } class RuleParser { /* name url enable type action 0 div 1 iframe 2強行塞入 nextLink 下一頁的xpath或者selector pageElement //頁面主體的xpath或者selector lazyImgSrc //圖片延後加載的屬性直接賦值到src css //添加樣式 insert: '//div[@id="res"]', insertPos: 1, 1 之前 2 裏面最後 pageAction:(document, eles) //對每一個插入的頁面進行修剪 init:(document) // 對主頁面進行處理 */ constructor() { this.customRules=[ { from:2, name:"yande", type:1, action:0, url:"^https:\/\/yande\.re\/", pageElement:"ul#post-list-posts>li", nextLink:"a.next_page", css:".javascript-hide {display: inline-block !important;}" }, { from:2, name:"tieba", type:1, action:1, url:"^https:\/\/tieba\.baidu.com\/f\?kw=", pageElement:"ul#thread_list>li", nextLink:".next.pagination-item " } ]; this.rules=[]; this.pageDoc=document; this.curUrl=location.href; this.curSiteRule={}; } initSavedRules(callback){ var self=this; storage.getItem("customRules", customRules=>{ if(customRules)self.customRules=customRules; storage.getItem("rules", rules=>{ if(rules)this.rules=rules; callback(); }); }); } saveCurSiteRule(){ if(!this.curSiteRule || !this.curSiteRule.url)return; this.customRules=this.customRules.filter(item=>{return item.url!=this.curSiteRule.url}); this.customRules.push(this.curSiteRule); storage.setItem("customRules", this.customRules); } requestJSON(url, callback){ _GM_xmlhttpRequest({ url: url, onload: function(res) { let json=null; try{ json=JSON.parse(res.response); }catch(e){ debug(e); } callback(json); } }); } formatRule(item, type, from){ switch(type){ case 0: return { from:from, type:type, name:item.name, action:item.data.forceIframe=="true"?1:0, url:item.data.url, pageElement:item.data.pageElement, nextLink:item.data.nextLink, insert:item.data.insertBefore, insertPos:1, updatedAt:item.data.updated_at, css:item.data.Stylus+item.data.CSS, pageAction:item.data.bookmarklet }; break; case 1: default: item.from=from; item.type=type; return item; break; } return null; } addRuleByUrl(url, type, from, callback) { this.requestJSON(url, json=>{ this.addRules(json, type, from); storage.setItem("rules", this.rules); callback(); }); } addRules(rules, type, from) { if(rules && rules.length>0){ this.rules=this.rules.filter(item=>{return item.from!=from}); rules.forEach(item=>{ let rule=this.formatRule(item, type, from); if(rule){ this.rules.push(rule); } }); } } getRule() { if(this.curSiteRule && this.curSiteRule.url){ return this.curSiteRule; } var self=this; for(let i in this.customRules){ let rule=this.customRules[i]; if(rule.enable==0)continue; let urlReg=new RegExp(rule.url, "i"); if(urlReg.test(location.href)){ let pageElement,nextLink,insert; if(rule.pageElement)pageElement=document.querySelector(rule.pageElement); if(rule.nextLink)nextLink=document.querySelector(rule.nextLink); if(rule.insert)insert=document.querySelector(rule.insert); if((rule.pageElement && !pageElement) || (rule.nextLink && !nextLink) || (rule.insert && !insert)){ continue; } this.curSiteRule=rule; debug(rule); return rule; } } for(let i in this.rules){ let rule=this.rules[i]; if(rule.enable==0)continue; let urlReg=new RegExp(rule.url, "i"); if(urlReg.test(location.href)){ let pageElement,nextLink,insert; if(rule.pageElement)pageElement=rule.type==0?getElementByXpath(rule.pageElement):document.querySelector(rule.pageElement); if(rule.nextLink)nextLink=rule.type==0?getElementByXpath(rule.nextLink):document.querySelector(rule.nextLink); if(rule.insert)insert=rule.type==0?getElementByXpath(rule.insert):document.querySelector(rule.insert); if((rule.pageElement && !pageElement) || (rule.nextLink && !nextLink) || (rule.insert && !insert)){ continue; } this.curSiteRule=rule; debug(rule); return rule; } } this.curSiteRule={}; this.curSiteRule.url=location.origin.replace(/\./g,"\\."); return null; } geneSelector(ele){ let selector=ele.tagName; //Google id class都是隨機。百度更過分,style script順序都是隨機的 //if(ele.id) selector += '#' + ele.id; //if(ele.classList) selector += [].map.call(ele.classList,d=>'.'+d).join(''); let parent = ele.parentElement; if(parent){ let i,j=0; for(i=0;i ' + selector + (parent.tagName=="HTML"?"":`:nth-of-type(${j})`); } return selector; } getPageElement(doc) { let pageElement=null; let self=this; if(this.curSiteRule.pageElement){ pageElement=this.curSiteRule.type==0?getAllElementsByXpath(this.curSiteRule.pageElement,doc):doc.querySelectorAll(this.curSiteRule.pageElement); } if(!pageElement && !this.curSiteRule.pageElement){ let body=doc.body,bodyHeight=parseInt(_unsafeWindow.getComputedStyle(body).height); function checkElement(ele){ let curHeight=parseInt(_unsafeWindow.getComputedStyle(ele).height); if(curHeight/bodyHeight<=0.48)return null; if(ele.children.length==0){ self.curSiteRule.pageElement=self.geneSelector(ele.parentNode)+">"+ele.tagName; self.curSiteRule.type=1; debug(self.curSiteRule.pageElement); return [ele]; } let i,maxHeight=curHeight*0.35,curMaxEle=null,curMaxArea=0; for(i=0;imaxHeight){ return checkElement(curMaxEle); } self.curSiteRule.pageElement=self.geneSelector(ele)+">*"; self.curSiteRule.type=1; debug(self.curSiteRule.pageElement); return ele.children; } pageElement=checkElement(body); if(pageElement)this.saveCurSiteRule(); } return pageElement; } getPage(){ let canSave=true; let url=this.curUrl; let doc=this.pageDoc; let pageNum=0,preStr="",afterStr=""; let pageMatch1=url.match(/(.*[a-zA-Z0-9\/][\-_](?:p|page)?)(\d+)(\.html?$|$)/i); let pageMatch2=url.match(/(.*[\?&]p(?:age)?=)(\d+)($|#.*)/i); if(pageMatch1){ preStr=pageMatch1[1]; pageNum=pageMatch1[2]; afterStr=pageMatch1[3]; }else if(pageMatch2){ preStr=pageMatch2[1]; pageNum=pageMatch2[2]; afterStr=pageMatch2[3]; } var curPage=doc,i,cur; let next=curPage.querySelector("a.next"); if(!next)next=curPage.querySelector("a#next"); if(!next)next=curPage.querySelector("a#rightFix"); if(!next)next=curPage.querySelector("a.next_page"); if(next && (!next.href || /javascript:/.test(next.href)))next=null; if(!next){ let aTags=curPage.querySelectorAll("a"); let nextf,nexts,nextt,nextfo; for(i=0;ia"); if(!next){ canSave=false; let pageDiv=curPage.querySelector("div.wp-pagenavi"); if(pageDiv){ cur=pageDiv.querySelector("span.current"); next=cur.nextSibling; }else{ cur=curPage.querySelector("div.article-paging>span"); if(cur){ next=cur.nextElementSibling; } } } if(!next)next=curPage.querySelector('[rel="next"]'); return {next:next,canSave:canSave}; } getNextLink(doc) { let nextLink=null,page; let curDoc=doc||this.pageDoc; if(this.curSiteRule.nextLink)nextLink=this.curSiteRule.type==0?getElementByXpath(this.curSiteRule.nextLink,curDoc):curDoc.querySelector(this.curSiteRule.nextLink); if(!nextLink){ page=this.getPage(); nextLink=page.next; } if(nextLink){ if(!this.basePageElement){ this.basePageElement=this.getPageElement(document); } if(!this.curSiteRule.nextLink && page && page.canSave){ this.curSiteRule.nextLink=this.geneSelector(nextLink); this.curSiteRule.type=1; this.saveCurSiteRule(); } } return nextLink; } getInsert(refresh) { if(this.insert && !refresh && this.insert.parentNode)return this.insert; if(this.curSiteRule.insert){ this.insert=this.curSiteRule.type==0?getElementByXpath(this.curSiteRule.insert,document):document.querySelector(this.curSiteRule.insert); }else{ let pageElement=this.basePageElement; 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; } pageAction(document,eles){ let code=this.curSiteRule.pageAction; if(code){ _unsafeWindow.eval(code); } let css=this.curSiteRule.css; if(css){ _GM_addStyle(css); } let lazyImgSrc=this.curSiteRule.lazyImgSrc; if(lazyImgSrc){ [].forEach.call(eles, ele=>{ [].forEach.call(ele.querySelectorAll("img"), img=>{ if(img[lazyImgSrc]){ img.src=img[lazyImgSrc]; } }); }); } } initPage(){ this.getRule(); let code=this.curSiteRule.init; if(code){ _unsafeWindow.eval(code); } } insertPage(doc, eles, url){ this.pageDoc=doc; this.curUrl=url; this.pageAction(doc, eles); this.getInsert(); var self=this; if(!eles || eles.length==0 || !self.insert || !self.insert.parentNode){ }else{ [].forEach.call(eles, ele=>{ if(self.curSiteRule.insertPos==2){ self.insert.appendChild(ele.cloneNode(true)); }else{ self.insert.parentNode.insertBefore(ele.cloneNode(true), self.insert); } }); } } } var ruleParser = new RuleParser(); var rulesDate={},ruleUrls,updateDate; function initConfig(){ _GM_registerMenuCommand(i18n(isDisabled?"enable":"disable"), ()=>{ storage.setItem("disable_"+location.host, !isDisabled); location.reload(); }); var configCon,insertPos; if(location.href=="https://github.com/hoothin/UserScripts/tree/master/Pagetual"){ _GM_addStyle(` p>span:nth-child(1),p>span:nth-child(2),p>span:nth-child(3){ cursor: pointer; user-select: none; } p>span:nth-child(1):hover,p>span:nth-child(2):hover,p>span:nth-child(3):hover{ color:red; } .updateDate{ cursor: pointer; user-select: none; } .updateDate:hover{ color:red; } `); configCon=document.querySelector(".markdown-body"); insertPos=configCon.querySelector("hr"); }else if(location.href=="https://github.com/hoothin/UserScripts/tree/master/Pagetual"){ }else return; class Rulebar { init(ruleUrl){ this.ruleUrl=ruleUrl; this.item=document.createElement("p"); this.item.title=ruleUrl.type==0?"AutoPagerize Rules":"Pagetual Rules"; this.item.dataset.id=this.ruleUrl.id; let url=document.createElement("span"); url.innerHTML=ruleUrl.url; let up=document.createElement("span"); up.innerHTML="↑ "; let down=document.createElement("span"); down.innerHTML="↓ "; let del=document.createElement("span"); del.innerHTML="× "; up.onclick=e=>{ this.moveUp(); }; down.onclick=e=>{ this.moveDown(); }; del.onclick=e=>{ this.del(); }; this.item.appendChild(up); this.item.appendChild(down); this.item.appendChild(del); this.item.appendChild(url); configCon.insertBefore(this.item, insertPos); } saveSort(){ let sort=[]; [].forEach.call(this.item.parentNode.querySelectorAll("p[data-id]"), i=>{ sort.push(i.dataset.id); }); rulesDate.sort=sort; storage.setItem("importRuleUrl", rulesDate); } moveUp(){ let preE=this.item.previousElementSibling; if(preE.tagName=="P" && preE.children.length>1){ this.item.parentNode.insertBefore(this.item,preE); this.saveSort(); } } moveDown(){ let nextE=this.item.nextElementSibling; if(nextE.tagName=="P" && nextE.children.length>1){ this.item.parentNode.insertBefore(nextE,this.item); this.saveSort(); } } del(){ if(this.ruleUrl.id<2){ alert(i18n("cantDel")); }else if(window.confirm(i18n("confirmDel"))){ for(let u=0;u{return item.from!=this.ruleUrl.id}); storage.setItem("rules", ruleParser.rules); this.item.parentNode.removeChild(this.item); //location.reload(); } } } let updateP=document.createElement("p"),i=0; let now=new Date().getTime(),inUpdate=false; updateP.className="updateDate"; updateP.innerHTML=updateDate; updateP.title=i18n("update"); updateP.onclick=e=>{ if(inUpdate)return; inUpdate=true; ruleUrls.forEach(rule=>{ ruleParser.addRuleByUrl(rule.url, rule.type, rule.id, ()=>{ if(++i==ruleUrls.length){ storage.setItem("ruleLastUpdate", now); alert(i18n("updateSucc")); inUpdate=false; } }) }); alert(i18n("beginUpdate")); }; configCon.insertBefore(updateP, insertPos); if(ruleUrls){ ruleUrls.forEach(ruleUrl=>{ var rulebar=new Rulebar(); rulebar.init(ruleUrl); }); } let customUrlsTitle=document.createElement("h2"); customUrlsTitle.innerHTML=i18n("customUrls") configCon.insertBefore(customUrlsTitle, insertPos); let customUrlsInput=document.createElement("textarea"); customUrlsInput.style.width="100%"; customUrlsInput.placeholder="0|http://wedata.net/databases/AutoPagerize/items_all.json"; configCon.insertBefore(customUrlsInput, insertPos); let customRulesTitle=document.createElement("h2"); customRulesTitle.innerHTML=i18n("customRules") configCon.insertBefore(customRulesTitle, insertPos); let customRulesInput=document.createElement("textarea"); customRulesInput.style.width="100%"; customRulesInput.style.height="500px"; customRulesInput.placeholder=`[\n{\n "type":"1",\n "name":"yande",\n "action":"0",\n "url":"^https:\/\/yande\\.re\/",\n "pageElement":"ul#post-list-posts>li",\n "nextLink":"a.next_page",\n "css":".javascript-hide {display: inline-block !important;}"\n},\n{\n "type":"1",\n "name":"tieba",\n "action":"1",\n "url":"^https:\/\/tieba\\.baidu.com\/f\\?kw=",\n "pageElement":"ul#thread_list>li",\n "nextLink":".next.pagination-item "\n}\n]`; customRulesInput.value=JSON.stringify(ruleParser.customRules); configCon.insertBefore(customRulesInput, insertPos); let saveBtn=document.createElement("button"); saveBtn.innerHTML=i18n("save"); saveBtn.style.width="100%"; configCon.insertBefore(saveBtn, insertPos); saveBtn.onclick=e=>{ try{ let customRules=JSON.parse(customRulesInput.value); debug(customRules); storage.setItem("customRules", customRules); }catch(e){ debug(e); alert("JSON error!"); } let customUrls=customUrlsInput.value.trim(); if(customUrls){ customUrls=customUrls.split(/\n/); for(let c=0;c{ if(maxId{ storage.getItem("importRuleUrl", data=>{ if(data){ rulesDate=data; if(data.urls)ruleUrls=ruleUrls.concat(data.urls); if(data.sort){ let urls=[]; data.sort.forEach(id=>{ for(let s=0;s{ storage.getItem("ruleLastUpdate", date=>{ isDisabled=v==true; getTimeStr(date); initConfig(); if(isDisabled)return; let now=new Date().getTime(); if(!date || now-date>3*24*60*60*1000){ ruleUrls.forEach(rule=>{ ruleParser.addRuleByUrl(rule.url, rule.type, rule.id, ()=>{ if(++i==ruleUrls.length){ storage.setItem("ruleLastUpdate", now); callback(); } }) }); }else{ callback(); } }); }); }); }); } function requestDoc(url, callback){ _GM_xmlhttpRequest({ url: url, overrideMimeType:"text/html;charset="+document.charset, onload: function(res) { var doc=null; try { doc=document.implementation.createHTMLDocument(''); doc.documentElement.innerHTML=res.response; } catch (e) { debug('parse error'+e.toString()); } let pageElement=ruleParser.getPageElement(doc); //只有1的話怕不是圖片哦 if(pageElement && pageElement.length>1){ callback(pageElement); ruleParser.insertPage(doc, pageElement, url); }else{ requestFromIframe(url, (doc, eles)=>{ callback(eles); if(eles){ ruleParser.insertPage(doc, eles, url); } }); } } }); } function requestFromIframe(url, callback){ let orgPage,curPage; let iframe = document.createElement('iframe'); iframe.name = 'pagetual-iframe'; iframe.width = '100%'; iframe.height = '0'; iframe.frameBorder = '0'; iframe.style.cssText = 'margin:0!important;padding:0!important;visibility:hidden!important;'; iframe.addEventListener("load", e=>{ setTimeout(()=>{ //可能會延遲加載 let doc=iframe.contentWindow.document; let eles=ruleParser.getPageElement(doc); if(eles && eles.length>0){ callback(doc, eles); }else{ isPause=true; callback(false, false); } document.body.removeChild(iframe); },300); }); iframe.src=url; document.body.appendChild(iframe); } function initPage(){ ruleParser.initPage() nextPage(); initListener(); } function initView(){ _GM_addStyle(` .pagetual_pageBar.stop { -webkit-filter: invert(100%); filter: invert(100%); opacity: 1; } .pagetual_pageBar { opacity: 0.1; } .pagetual_pageBar:hover { opacity: 1; } .pagetual_pageBar>span { vertical-align: super; } .pagetual_pageBar>span>svg:hover { animation: touhouAni 1s infinite; } @keyframes touhouAni{ from {transform: rotate(0deg) scale3d(1.2, 1.2, 1.2);} to {transform: rotate(360deg);} } @-webkit-keyframes touhouAni{ from {transform: rotate(0deg) scale3d(1.2, 1.2, 1.2);} to {transform: rotate(360deg);} } `); } var loading=document.createElement("div"); loading.style.cssText="display:none;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 auto;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;origin-trial-test-property: 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: initial;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;z-index: 2147483647;"; loading.innerHTML=``; document.body.appendChild(loading); var upSvg=``; var downSvg=``; var pageBarStyle=`box-shadow: 0px 0px 10px 0px #000000aa;border-radius: 20px;background-color: rgb(240 240 240 / 80%);visibility: visible; position: initial; width: auto; height: 30px; float: none; clear: both; margin: 20px auto; text-align: center; display: block;`; var pageTextStyle=`line-height: 30px;text-decoration: none;user-select: none;visibility: visible;position: initial;width: auto;height: auto;float: none;clear: both;margin: 0px auto;text-align: center;display: inline;font-weight: bold;font-style: normal;font-size: 16px;letter-spacing: initial;vertical-align: super;color: rgb(85, 85, 95);`; var isPause=false,isLoading=false,curPage=1,isDisabled=false; function changeStop(stop){ isPause=stop; [].forEach.call(document.querySelectorAll(".pagetual_pageBar"), bar=>{ if(isPause){ bar.classList.add("stop"); }else{ bar.classList.remove("stop"); } }); } function initListener(){ document.addEventListener('scroll', e=>{ setTimeout(()=>{ if(!isPause && !isLoading){ let scrolly=window.scrollY; let windowHeight=window.innerHeight; let scrollH=Math.max(document.documentElement.scrollHeight, document.body.scrollHeight); if(scrollH-scrolly-windowHeight<500){ nextPage(); } } },100); }, false); document.addEventListener('dblclick', e=>{ changeStop(!isPause); }); } function createPageBar(url){ let insert=ruleParser.getInsert(); if(!insert || !insert.parentNode)return; curPage++; let inTable=insert.tagName=="TR" || insert.previousElementSibling.tagName=="TR"; let pageBar=document.createElement(inTable?"tr":"div"); let upSpan=document.createElement("span"); let downSpan=document.createElement("span"); let pageText=document.createElement("a"); pageBar.className="pagetual_pageBar"; pageBar.id="pagetual_pageBar"; if(isPause){ pageBar.classList.add("stop"); } pageBar.style.cssText=pageBarStyle; pageBar.title=i18n(isPause?"enable":"disable"); upSpan.innerHTML=upSvg; upSpan.title=i18n("toTop"); downSpan.innerHTML=downSvg; downSpan.title=i18n("toBottom"); pageText.href=url; pageText.style=pageTextStyle; pageText.innerHTML="Page "+curPage; pageText.title=i18n("currentPage"); pageBar.appendChild(upSpan); pageBar.appendChild(pageText); pageBar.appendChild(downSpan); if(inTable){ pageBar.style.display="table-row"; let td=document.createElement("td"); td.colSpan=99; td.appendChild(upSpan); td.appendChild(pageText); td.appendChild(downSpan); pageBar.appendChild(td); } upSpan.addEventListener("click", e=>{ changeStop(true); document.body.scrollTop=0; document.documentElement.scrollTop=0; e.preventDefault(); e.stopPropagation(); }); downSpan.addEventListener("click", e=>{ changeStop(true); document.body.scrollTop=9999999; document.documentElement.scrollTop=9999999; e.preventDefault(); e.stopPropagation(); }); pageBar.addEventListener("click", e=>{ changeStop(!isPause); pageBar.title=i18n(isPause?"enable":"disable"); }); pageText.addEventListener("click", e=>{ e.stopPropagation(); }); pageBar.style.width=parseInt(_unsafeWindow.getComputedStyle(insert.parentNode).width)*.9+"px"; insert.parentNode.insertBefore(pageBar, insert); return pageBar; } var emuIframe; function emuPage(callback){ let orgPage,curPage,iframeDoc; function checkPage(doc){ let eles=ruleParser.getPageElement(doc); if(orgPage == eles[0]){ setTimeout(()=>{ checkPage(doc); },500); }else{ if(eles && eles.length>0){ callback(doc, eles); }else{ isPause=true; callback(false, false); } } } if(!emuIframe){ emuIframe = document.createElement('iframe'); emuIframe.name = 'pagetual-iframe'; emuIframe.width = '100%'; emuIframe.height = '0'; emuIframe.frameBorder = '0'; emuIframe.style.cssText = 'margin:0!important;padding:0!important;visibility:hidden!important;'; emuIframe.addEventListener("load", e=>{ iframeDoc=emuIframe.contentDocument || emuIframe.contentWindow.document; setTimeout(()=>{ orgPage=ruleParser.getPageElement(iframeDoc)[0]; ruleParser.getNextLink(iframeDoc).click(); checkPage(iframeDoc); },300); }); emuIframe.src=location.href; document.body.appendChild(emuIframe); }else{ iframeDoc=emuIframe.contentDocument || emuIframe.contentWindow.document; orgPage=ruleParser.getPageElement(iframeDoc)[0]; ruleParser.getNextLink(iframeDoc).click(); checkPage(iframeDoc); } } function forceIframe(url, callback){ let curIframe = document.createElement('iframe'); curIframe.name = 'pagetual-iframe'; 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 0px 2rem; padding: 0px; z-index: 2147483647;'; curIframe.addEventListener("load", e=>{ let iframeDoc=curIframe.contentDocument || curIframe.contentWindow.document; let eles=ruleParser.getPageElement(iframeDoc); if(eles && eles.length>0){ ruleParser.insertPage(iframeDoc, [], url); callback(curIframe, eles); curIframe.style.height=iframeDoc.body.scrollHeight+"px"; curIframe.style.width=iframeDoc.body.scrollWidth+"px"; curIframe.scrollIntoView(); }else{ isPause=true; callback(false, false); curIframe.parentNode.removeChild(curIframe); } }); curIframe.src=url; let insert=ruleParser.getInsert(); document.body.appendChild(curIframe); return curIframe; } function nextPage(){ if(isPause || isLoading)return; let nextLink=ruleParser.getNextLink(); let insert=ruleParser.getInsert(); if(nextLink && insert){ isLoading=true; loading.style.display=""; if(ruleParser.curSiteRule.action==1 && nextLink.href){ requestFromIframe(nextLink.href, (doc, eles)=>{ isLoading=false; loading.style.display="none"; if(eles){ createPageBar(nextLink.href); ruleParser.insertPage(doc, eles, nextLink.href); } }); }else if(ruleParser.curSiteRule.action==2 && nextLink.href){ forceIframe(nextLink.href, (iframe, eles)=>{ isLoading=false; loading.style.display="none"; if(eles){ let pageBar=createPageBar(nextLink.href); iframe.parentNode.insertBefore(pageBar, iframe); } }); }else{ if(nextLink.href){ requestDoc(nextLink.href, (eles)=>{ isLoading=false; loading.style.display="none"; if(eles){ createPageBar(nextLink.href); } }); }else{ emuPage((doc, eles)=>{ isLoading=false; loading.style.display="none"; if(eles){ createPageBar(nextLink.href); ruleParser.insertPage(doc, eles, ""); } }); } } } } function init(){ initView(); initRules(()=>{ initPage(); }); } init(); })();