您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Ultimate Pixiv Script: Direct Links, Auto-Paging, Preview, IQDB/Danbooru, Filter/Sort using Bookmark,views,rating,total score. | Safe Search | plus more. Works best with "Pixiv++ Manga Viewer" and "Generic Image Viewer". 自動ページング|ポケベル|ロード次ページ|フィルター|並べ替え|注文|ダイレクトリンク
当前为
// ==UserScript== // @name [TS] Pixiv++ // @namespace TimidScript // @version 3.2.78a // @description Ultimate Pixiv Script: Direct Links, Auto-Paging, Preview, IQDB/Danbooru, Filter/Sort using Bookmark,views,rating,total score. | Safe Search | plus more. Works best with "Pixiv++ Manga Viewer" and "Generic Image Viewer". 自動ページング|ポケベル|ロード次ページ|フィルター|並べ替え|注文|ダイレクトリンク // @author TimidScript // @homepageURL https://openuserjs.org/users/TimidScript // @copyright © 2014 TimidScript, All Rights Reserved. // @license Creative Commons BY-NC-SA + Read the License inside the script // @include http://www.pixiv.net/* // @exclude http://www.pixiv.net/member_illust.php?mode=manga&illust_id* // @exclude http://www.pixiv.net/member_illust.php?mode=big&illust_id* // @require https://openuserjs.org/src/libs/TimidScript/TSL_-_Generic.js // @require https://openuserjs.org/src/libs/TimidScript/TSL_-_GM_Update.js // @homeURL https://openuserjs.org/scripts/TimidScript/[TS]_Pixiv++ // @grant GM_info // @grant GM_getMetadata // @grant GM_registerMenuCommand // @grant GM_getValue // @grant GM_setValue // @grant GM_listValues // @grant GM_deleteValue // @grant GM_xmlhttpRequest // @icon  // ==/UserScript== /* License + Copyright Notice ******************************************************************************************** Copyright © TimidScript, All Rights Reserved. [Creative Commons BY-NC-SA](http://en.wikipedia.org/wiki/Creative_Commons_license) Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: 1) This copyright must be included 2) Due credits and link to original author's homepage (included in copyright). 3) Notify the original author of redistribution TimidScript's Homepages: [GitHub](https://github.com/TimidScript) [OpenUserJS](https://openuserjs.org/users/TimidScript) [GreasyFork](https://greasyforks.org/users/1455-timidscript) */ /* Information ******************************************************************************************** Version History ------------------------------------ 3.2.78 (2015-08-23) - Filter fixed. Neat code provided by HoldMyPizza from OUJS 3.2.77 (2015-07-26) - Bug Fix: Corrected latest fix 3.2.76 (2015-07-25) - Bug Fix: Capture new homepage url 3.2.75 (2015-06-27) - Bug Fix: Default preview time and height were switched - Bug Fix: Removal of extra title from thumbnail - Support for other 3rd party of login by adding full support of xhr to be default rather than GM_xhr. Default request method is now is xhr - Removed iframe support 3.2.74 (2015-06-06) - Bug fixes to support own illustrations both public and hidden 3.2.73 (2015-05-30) - Outgoing links are direct now - Bug Fix: getIllustID extracts id from unique urls now http://www.pixiv.net/member_illust.php?illust_id=1111111&mode=medium http://www.pixiv.net/member_illust.php?mode=medium&illust_id=1111111 - Bug Fix: Bookmark count was not being attained in "Works" page (3) 3.2.72 (2015-05-25) - Bug fix: Removes the token information from image link (illustURL) 3.2.71 (2015-05-16) - As of 11/05/2015 the Phone API (SPAPI) is dead. Using public API and HTML instead. - Single paged mangas will be marked as illustrations rather than manga as in most cases that is the case - Changed much of the metadata and information used - Created a separate library for the illustration data getter - Manga hotbox preview images dimensions unfixed - Removed a lot of redundant code. including linker methods, Cache and rated paging - GM_setValue("LinkerMethod", 2); //[2] - GM_setValue("EnableCache", true); //[3] - GM_setValue("APIAgeRating", true); //[4] - RequestBoomarkCount setting added - Huge amount of optimisation and removal of redundant code - IQDB now uses size 240 (240x480) - Need to remove AgeRating searches. Should only be a filter. 3.1.70 (2015-05-08) - Page count added in information box 3.1.69 (2015-04-14) - Hidden comix ads from the illustration page - Changed the illustration page interface. Illustration information is always shown and thumbnails for pages are bellow the illustration. . . . *************************************************************************************************/ //if (window.self !== window.top) return; var IsIllustrationPage = (document.URL.indexOf("http://www.pixiv.net/member_illust.php?") != -1 && document.URL.match(/mode=medium/i) != null); var Illustrations = {}; var PAGETYPE = (function () { if (IsIllustrationPage) return 0; else if (document.URL.match(/http:\/\/www\.pixiv\.net\/((cate_r18|mypage|member)\.php|$)/i)) return 1; else if (document.URL.match(/http:\/\/www\.pixiv\.net\/member_illust\.php\?id/i)) return 3; //Artist Work Page else if (document.URL.match(/http:\/\/www\.pixiv\.net\/member_illust\.php/i)) return 4; //Personal Work Page else if (document.URL.match(/http:\/\/www\.pixiv\.net\/bookmark(_add)?\.php/i)) return 5; //Personal Bookmarks, Added new bookmarks else if (document.URL.match(/http:\/\/www\.pixiv\.net\/bookmark\.php\?id=/i)) return 6; //Artist Bookmark else if (document.URL.match(/http:\/\/www\.pixiv\.net\/bookmark_new_illust(_r18)?\.php/i)) return 7; //Works from favourite artists else if (document.URL.match(/http:\/\/www\.pixiv\.net\/new_illust(_r18)?\.php/i)) return 8; //New illustrations else if (document.URL.match(/http:\/\/www\.pixiv\.net\/search\.php\?/i)) return 9; //Search return -1; })(); console.info("Pixiv++ (" + PAGETYPE + ")"); if (!(typeof GM_getValue === "function" && GM_getValue("", "?") === "?")) { GM_setValue = function (key, val) { localStorage.setItem(key, val); }; GM_getValue = function (key, def) { return (localStorage.getItem(key) ? localStorage.getItem(key) : def); } } var containerClasses = { UnPaged: [ "works_display", // Illustration Page (http://www.pixiv.net/member_illust.php?mode) "content", //Personal Home Page (http://www.pixiv.net/mypage.php) "top_display_works linkStyleWorks", //Personal Home Page Adult Content "worksListOthers", //Artist's Profile Page (http://www.pixiv.net/member.php?) "search_a2_result linkStyleWorks", //Response Page ], Paged: [ "_image-items", "display_works linkStyleWorks", //Artist's & Personal Work and Bookmark Page (http://www.pixiv.net/member_illust.php)(http://www.pixiv.net/bookmark.php) "_image-items autopagerize_page_element", // Illustrations Search; New Illustrations; Your Favourite Artist's New Illustrations; (http://www.pixiv.net/search.php?)(http://www.pixiv.net/new_illust.php)(http://www.pixiv.net/new_illust_r18.php)(http://www.pixiv.net/bookmark_new_illust.php) ] }; /* =================================================================================================================================== Handles all functions to do with getting and adding all Illustration links and metadata. This includes: Image Links, IQDB, Bookmark Count, Views, Rating and score. ===================================================================================================================================*/ var IllustrationLinker = { //Contains all thumbnail links that need to be parsed processList: new Array(), //Max of simultaneous link calls. The higher the count the more stress on server and ISP simultaneousCallsMAX: 6, simultaneousCalls: 0, intervalID: null, msgHandle: null, thumbcounter: 0, //Used by the pager when running in "SafeMode" as it temporarily removes session cookie shortPause: false, //If false the thumbnail interval parser stops running enabled: false, //First time it gets turned on is when the SideBar is loaded TIMESTART: 0, TIMEEND: 0, requestBookmarkCount: GM_getValue("RequestBookmarkCount", 0), getIllustID: function (url) { var id = url.replace(/.+www\.pixiv\.net\/member_illust\.php(\?.+&|\?)illust_id=(\d+)(&.+)?/, "$2"); id = id.replace(/.+i\d+\.pixiv\.net\/(img|c\/).+\/(\d+)(_.+)?\.\w+(\?.+)?/, "$2"); return id; }, getIllust: function (id) { var properties = "userID userName userProfileImageURL illustType illustID illustTitle illust128URL illust150URL illust240URL illust480URL illust600URL illustURL pageCount description time tags tools ratings totalRatings viewCount bookmarkCount responseCount R18"; var illust = Illustrations["i" + id]; if (!illust) { illust = {}; Illustrations["i" + id] = illust; } return makeStruct(properties, illust); }, /* ------------------------------------------------------------------------------------------- Parses through all the image links and add full image link -------------------------------------------------------------------------------------------*/ getContainerLinks: function (containers, pageNumber) { if (!containers) return; var id, thumbnail; if (IsIllustrationPage) ////Illustration Page that excludes Novels { IllustrationLinker.getDataHTML(IllustrationLinker.getIllustID(document.URL)) return; } for (var i = 0, i; container = containers[i], i < containers.length; i++) { links = container.getElementsByTagName("a"); for (var j = 0; link = links[j], j < links.length; j++) { if (link.href.indexOf("http://www.pixiv.net/member_illust.php?mode=") != -1 && link.getElementsByTagName('IMG').length == 1) { id = IllustrationLinker.getIllustID(link.href); thumbnail = link.parentNode; thumbnail.id = "i" + id; thumbnail.setAttribute("illustration-id", id); thumbnail.setAttribute("name", "pppThumb"); if (pageNumber > 0) { TSL.addClass(thumbnail, "pppThumb" + (pageNumber % 3)); thumbnail.setAttribute("page", pageNumber); thumbnail.setAttribute("position", ++IllustrationLinker.thumbcounter); } if (IllustrationLinker.getIllust(id).illustID) IllustrationLinker.createLinksBox(id); else IllustrationLinker.processList.push(id); } } } PaginatorHQ.displayResultInfo(); if (this.enabled) this.runIntevalThumbnailParser(); }, /* ------------------------------------------------------------------------------------------- Get links again. Called when using ARS -------------------------------------------------------------------------------------------*/ resetSettings: function () { clearInterval(IllustrationLinker.intervalID); IllustrationLinker.intervalID = null; IllustrationLinker.processList = new Array(); if (IllustrationLinker.msgHandle) IllustrationLinker.msgHandle.textContent = ""; }, /* ------------------------------------------------------------------------------------------- Contains the interval that requests image information -------------------------------------------------------------------------------------------*/ runIntevalThumbnailParser: function () { IllustrationLinker.TIMESTART = new Date(); if (!IllustrationLinker.msgHandle) IllustrationLinker.msgHandle = DisplayMessage("Getting metadata for [" + IllustrationLinker.processList.length + "] illustrations..."); if (IllustrationLinker.intervalID == null && IllustrationLinker.enabled) IllustrationLinker.intervalID = setInterval(processNextID, 100); function processNextID() { //shortPause used when removing session cookie for safe page search if (IllustrationLinker.shortPause || IllustrationLinker.simultaneousCalls >= IllustrationLinker.simultaneousCallsMAX) return; if (IllustrationLinker.processList.length == 0 && IllustrationLinker.simultaneousCalls == 0) { IllustrationLinker.TIMEEND = new Date(); //console.log(IllustrationLinker.TIMEEND - IllustrationLinker.TIMESTART); clearInterval(IllustrationLinker.intervalID); IllustrationLinker.intervalID = null; RemoveMessage(IllustrationLinker.msgHandle); IllustrationLinker.msgHandle = null; PreviewHQ.adjustAllHotboxPositions(); PaginatorHQ.displayResultInfo(); DisplayMessage("Illustration metadata acquired", 1000); } else if (IllustrationLinker.processList.length % 5 == 0) { IllustrationLinker.msgHandle.textContent = "Getting metadata for [" + (IllustrationLinker.simultaneousCalls + IllustrationLinker.processList.length) + "] illustrations..."; PaginatorHQ.displayResultInfo(); PreviewHQ.adjustAllHotboxPositions(); } //Last check if (IllustrationLinker.enabled && IllustrationLinker.processList.length > 0) { IllustrationLinker.simultaneousCalls++; IllustrationLinker.getDataHTML(IllustrationLinker.processList.shift()); } } }, /* ------------------------------------------------------------------------------------------- If "value" is set to true the linker is temporarily paused. This is used only for short term pausing. switchOff is used for longer pausing method. -------------------------------------------------------------------------------------------*/ pause: function (value) { IllustrationLinker.shortPause = value; if (value && msgHandle) msgHandle.textContent = ""; }, /* ------------------------------------------------------------------------------------------- Thumbnail parser interval is cleared and the link generation halts until it switchOn is called. -------------------------------------------------------------------------------------------*/ switchOff: function () { PreviewHQ.adjustAllHotboxPositions(); IllustrationLinker.enabled = false; clearInterval(IllustrationLinker.intervalID); IllustrationLinker.intervalID = null; if (IllustrationLinker.msgHandle) RemoveMessage(IllustrationLinker.msgHandle); IllustrationLinker.msgHandle = null; }, switchOn: function () { PreviewHQ.adjustAllHotboxPositions(); IllustrationLinker.enabled = true; IllustrationLinker.runIntevalThumbnailParser(); }, /* ------------------------------------------------------------------------------------------- Extracts data from illustration page -------------------------------------------------------------------------------------------*/ setMetadata: function (id, doc) { if (IsIllustrationPage) console.warn("Illustration page debugging: ", id); var el, m, context metadata = IllustrationLinker.getIllust(id), script = doc.evaluate("//div[@id='wrapper']//script[contains(text(),'pixiv.context.illustId')]", doc, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue; unsafeWindow.eval("pixiv" + id + "= {}; pixiv" + id + ".context= {};"); script.innerHTML = script.innerHTML.replace(/pixiv\.context/g, "pixiv" + id + ".context"); unsafeWindow.eval(script.innerHTML); context = unsafeWindow["pixiv" + id].context; metadata.userID = context.userId; //el.parentElement.href.match(/member\.php\?id=(\d+)/i)[1]; metadata.userName = context.userName; //el.nextElementSibling.textContent; el = doc.querySelector(".user-image"); if (el) { metadata.userProfileImageURL = el.src; if (el.src.match("/profile/")) metadata.userLoginName = el.src.match(/\/profile\/([^\/])+\//i)[1]; } metadata.illustID = id; metadata.illustTitle = context.illustTitle; //doc.querySelector('meta[property="og:title"]'); //Translation scripts mess with the title metadata.R18 = 0; if (doc.querySelector(".r-18")) metadata.R18 = 1; if (doc.querySelector(".r-18g")) metadata.R18 = 2; metadata.pageCount = 1; if (doc.querySelector(".tools")) metadata.tools = doc.querySelector(".tools").textContent; metadata.tags = [].map.call(doc.querySelectorAll('li.tag a.text'), function (v, i) { return v.childNodes[0].textContent; }).join(' '); metadata.illust128URL = doc.querySelector('meta[property="og:image"]'); metadata.illust128URL = (metadata.illust128URL) ? metadata.illust128URL.getAttribute("content") : doc.querySelector('img[class="original-image"]').getAttribute("data-src"); metadata.description = doc.querySelector('meta[property="og:description"]'); metadata.description = (metadata.description) ? metadata.description.getAttribute("content") : doc.querySelector("._unit .caption").innerHTML; metadata.bookmarkCount = "?"; metadata.viewCount = parseInt(doc.querySelector(".view-count").textContent); metadata.ratings = parseInt(doc.querySelector(".rated-count").textContent); metadata.totalRatings = parseInt(doc.querySelector(".score-count").textContent); el = doc.querySelector(".response-in-work-more"); metadata.responseCount = (el) ? parseInt(el.textContent.match(/\d+/)[0]) : 0; if (context.ugokuIllustFullscreenData) { metadata.illustType = 3; metadata.illustURL = context.ugokuIllustFullscreenData.src; metadata.illust128URL = metadata.illust128URL.replace(/\/img-inf\/(.+)_s/, "/c/128x128/img-master/$1_master1200"); metadata.illust128URL = metadata.illust128URL.replace(".png", ".jpg"); metadata.illust150URL = metadata.illust128URL.replace("/128x128/", "/150x150/"); metadata.illust240URL = metadata.illust128URL.replace("/128x128/", "/240x480/"); metadata.illust480URL = metadata.illust128URL.replace("/128x128/", "/480x960/"); metadata.illust600URL = metadata.illust128URL.replace("/128x128/", "/600x600/"); metadata.illust1200URL = metadata.illust128URL.replace("/128x128/", "/1200x1200/"); } else { el = doc.querySelector(".works_display ._layout-thumbnail img"); metadata.illustURL = metadata.illust600URL = el.src; el = doc.querySelector(".original-image"); if (el) //Single Image { metadata.illustType = 1; metadata.illustURL = el.getAttribute("data-src").replace(/\?.+/, ""); } else //Manga { metadata.illustType = 2; el = doc.querySelectorAll(".work-info .meta li")[1]; m = el.textContent.match(/(\d+)P$/i); if (m) { metadata.pageCount = parseInt(m[1]); } else //Single paged manga { metadata.pageCount = 1; metadata.illustType = 1; } metadata.illustURL = metadata.illustURL.replace(/(\d+)_m\.\w+/, "$1_big_p0" + metadata.illust128URL.match(/(.\w+)$/)[1]); metadata.illustURL = metadata.illustURL.replace(/c\/600x600\/img-master(.+)\/.+/, "img-original$1/" + metadata.illustID + "_p0" + metadata.illust128URL.match(/(\.\w+)$/)[1]); if (metadata.illustID < 11319936) metadata.illustURL = metadata.illustURL.replace("_big", ""); } if (metadata.illust128URL.indexOf("/img-master/") > 0) { metadata.illust150URL = metadata.illust128URL.replace("/128x128/", "/150x150/"); metadata.illust128URL = metadata.illust150URL.replace("/150x150/", "/128x128/"); metadata.illust240URL = metadata.illust128URL.replace("/128x128/", "/240x480/"); metadata.illust480URL = metadata.illust128URL.replace("/128x128/", "/480x960/"); metadata.illust480URL = metadata.illust128URL.replace("/128x128/", "/600x600/"); metadata.illust1200URL = metadata.illust128URL.replace("/128x128/", "/1200x1200/"); } else { metadata.illust240URL = metadata.illust600URL.replace(/([^\/]+)\/(\d+)_m.+/, "$1/mobile/$2_240mw.jpg"); metadata.illust480URL = metadata.illust600URL; metadata.illust1200URL = metadata.illust600URL; if (metadata.illustType == 2) metadata.illust240URL = metadata.illust240URL.replace("_240mw", "_240mw_p0"); if (metadata.pageCount > 1 && metadata.illustID >= 11319936) metadata.illust1200URL = metadata.illustURL.replace("_big", "") metadata.illust150URL = metadata.illust240URL; } } metadata.bookmarkCount = IllustrationLinker.getBookmarkCount(id); if (IsIllustrationPage) console.log("HTML", metadata); //if (IsIllustrationPage) //{ // var arr = ["illust128URL", "illust150URL", "illust240URL", "illust480URL", "illust600URL", "illust1200URL"]; // var wd = document.querySelector(".works_display"); // wd.innerHTML = ""; // for (var i = 0; i < arr.length; i++) // { // var img = document.createElement("img"); // img.src = metadata[arr[i]]; // img.title = arr[i] + ": " + metadata[arr[i]]; // wd.appendChild(img); // wd.appendChild(document.createElement("br")); // } //} return metadata; }, /* ------------------------------------------------------------------------------------------- Gets bookmark count from thumbnail if one exists -------------------------------------------------------------------------------------------*/ getBookmarkCount: function (id) { var thumbnail = document.getElementById("i" + id); if (thumbnail) { var bm = thumbnail.querySelector(".bookmark-count"); if (bm) return parseInt(bm.textContent); if (PAGETYPE == 1 || PAGETYPE == 3 || PAGETYPE == 7 || PAGETYPE == 8) return "?"; //Pages that do not contain bookmark information else return 0; } return "?"; }, /* ------------------------------------------------------------------------------------------- Creates main links that are used by Downloaders. These links can be hidden. It also creates calls the creation of the hotbox. -------------------------------------------------------------------------------------------*/ createLinksBox: function (id) { var thumbnail = document.getElementById("i" + id), metadata = IllustrationLinker.getIllust(id); var linksBox = document.createElement("div"); linksBox.className = "pppLinksBox"; var directLinks = document.createElement("span"); if (IsIllustrationPage) { TSL.addStyle("LINXBOX", ".pppLinksBox {margin: 0 50px; text-align: center;}"); var el = document.querySelector(".works_display"); el.parentNode.insertBefore(linksBox, el.nextSibling); } else { PaginatorHQ.tidyThumbnail(id); thumbnail.insertBefore(linksBox, thumbnail.querySelector("a").nextSibling); } linksBox.appendChild(directLinks); directLinks.className = "pppDirectLinks"; if (PAGETYPE > 1) { var evaluator = new XPathEvaluator(); //document.evaluate var sortButton = evaluator.evaluate("//a[@name='Sort']", SideBar.iDoc.body, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue; if (sortButton.firstElementChild.style.borderColor == "rgb(0, 255, 0)") sortButton.firstElementChild.style.borderColor = "#F00"; } if (metadata.pageCount <= 1) //Single Illustration { var directLink = document.createElement("a"); directLink.textContent = "[" + ((metadata.illustType == 3) ? "?" : "❀") + "]"; //S❂❀ directLink.href = metadata.illustURL; if (!Settings.display.illustLink) directLinks.style.display = "none"; directLinks.appendChild(directLink); } else //Manga Illustration { directLinks.setAttribute("style", "font-size:xx-small"); var mangaLinks = '[ '; for (var i = 0, href; i < metadata.pageCount; i++) { href = metadata.illustURL.replace(/_p\d+/, "_p" + i); if (IsIllustrationPage) mangaLinks += '<a title ="Page #' + i + '" href="' + href + '">' + i + '</a> | '; else mangaLinks += '<a title ="Page #' + i + '" href="' + href + '">' + (((i + 1) % 5 == 0) ? "◆" : "◇") + '</a>' } mangaLinks = mangaLinks.replace(/ | $/, "") + "]"; if (IsIllustrationPage) { directLinks.innerHTML = mangaLinks; directLinks.setAttribute("style", "font-size:small"); } else directLinks.innerHTML = mangaLinks; if (!Settings.display.mangaLinks) directLinks.style.display = "none"; } if (IsIllustrationPage) { TSL.addStyle("RemoveAds", ".ads_anchor {display:none;}"); TSL.addStyle("AddBorder", ".work-tags, .works_display {margin-bottom: 0px;}"); //TSL.addStyle("VLB", "#VisibleLinkBox {background-color: red; height: 40px; margin: 5px 20px 10px 20px;}"); TSL.addStyle("VLB", "#VisibleLinkBox {margin: 1px 20px 10px 20px; background-color: gray;}"); var linkbox = document.createElement("section"); linkbox.id = "VisibleLinkBox"; linkbox.className = "work-tags"; var metascore = [metadata.bookmarkCount, metadata.viewCount, metadata.ratings, metadata.totalRatings, metadata.responseCount]; var metaname = ["★", "Views", "Rating", "Total", "Response"]; linkbox.innerHTML = "<a class='bookmark-count' style='background-color:#FFFD00;' href='http://" + Settings.IQDBType + ".IQDB.org/?url=" + ((metadata.illustType == 3) ? metadata.illust480URL : metadata.illust150URL) + "&NULL'>IQDB<a>" + '<a href="/response.php?type=illust&id=' + metadata.illustID + '" class="image-response-count ui-tooltip" data-tooltip="Received ' + metascore[4] + ' image responses"><i class="_icon sprites-image-response-badge"></i>' + metascore[4] + '</a>' + '<a href="/bookmark_detail.php?illust_id=' + metadata.illustID + '" class="bookmark-count ui-tooltip" data-tooltip="Received ' + metascore[0] + ' bookmarks"><i class="_icon sprites-bookmark-badge"></i>' + metascore[0] + '</a>'; if (metadata.pageCount > 1) linkbox.innerHTML = "<a class='bookmark-count' style='background-color:#81FF00;'>P" + metadata.pageCount + "</a>" + linkbox.innerHTML; for (var i = 1; i < 4; i++) linkbox.innerHTML += "<span class='bookmark-count' style='background-color:#FDFDFB;'>" + metaname[i] + " " + metascore[i] + "</span>"; var el = document.querySelector(".work-tags"); el.parentNode.insertBefore(linkbox, el.nextElementSibling); if (metadata.illustType == 2 && metadata.pageCount > 1) //Manga { var previewThumbs = document.createElement("section"); previewThumbs.setAttribute("style", "text-align: center;"); linksBox.parentElement.insertBefore(previewThumbs, linksBox.nextElementSibling); var ShowThumbs = document.createElement("button"); ShowThumbs.textContent = "Show Thumbnails"; ShowThumbs.setAttribute("style", "display: inline-block; width: 300px; height: margin: 10px 0 20px 0; padding: 5px;"); previewThumbs.appendChild(ShowThumbs); ShowThumbs.onclick = function () { if (this.shown) { ShowThumbs.textContent = "Show Thumbnails"; document.getElementById("MangaThumbnails").style.display = "none"; this.shown = false; } else { ShowThumbs.textContent = "Hide Thumbnails"; this.shown = true; if (document.getElementById("MangaThumbnails")) document.getElementById("MangaThumbnails").style.display = "block"; else { TSL.addStyle("MangaT", "#MangaThumbnails img {max-width: 150px; min-width: 150px; margin: 2px; box-shadow: 5px 5px 3px #888888;}"); var thumbs = document.createElement("section"); thumbs.id = "MangaThumbnails"; thumbs.setAttribute("style", "margin: 10px 30px;"); linksBox.parentElement.insertBefore(thumbs, previewThumbs.nextElementSibling); for (var i = 0; i < metadata.pageCount; i++) { el = document.createElement("a"); el.href = metadata.illustURL.replace(/\/(\d+|\d+_big)_p\d+/i, "/$1_p" + i); el.innerHTML = "<img src='" + metadata.illust240URL.replace(/_p\d+/i, "_p" + i) + "'/>"; //el.innerHTML = "<img src='" + metadata.illust150URL.replace(/_p\d+/i, "_p" + i) + "'/>"; thumbs.appendChild(el); } } } } } } else { var IQDBLink = document.createElement("a"); IQDBLink.textContent = "IQDB"; IQDBLink.className = "IQDBLink"; linksBox.appendChild(IQDBLink); PaginatorHQ.updateIQDBLink(linksBox); PreviewHQ.createHotBox(id); } if (PAGETYPE > 1) PaginatorHQ.filterThumbnail(thumbnail); }, /* ------------------------------------------------------------------------------------------- Gets illustration information using Illustration page -------------------------------------------------------------------------------------------*/ getDataHTML: function (id) { var extensionKnown = false, illust = IllustrationLinker.getIllust(id), START = new Date(); if (IsIllustrationPage) ////Illustration Page that excludes Novels { IllustrationLinker.setMetadata(id, document); finalise(); return; } if (Settings.requestMethod & 4) { GM_xmlhttpRequest({ url: "http://www.pixiv.net/member_illust.php?mode=medium&illust_id=" + id, method: "GET", timeout: 15000, //headers: { "User-agent": navigator.userAgent, "Accept": "text/html", Referer: "http://www.pixiv.net" }, headers: { "User-agent": navigator.userAgent, "Accept": "text/html" }, onload: function (response) { if (response.status == 200) //Response 200 implies that link exist and then most likely a Manga (or an Error XD) { var END = new Date(); //console.info(END - START); var doc = new DOMParser().parseFromString(response.responseText, "text/html"); IllustrationLinker.setMetadata(id, doc); } finalise(); } }); } else { var oReq = new XMLHttpRequest(); oReq.open("GET", "http://www.pixiv.net/member_illust.php?mode=medium&illust_id=" + id, true); oReq.responseType = "text"; oReq.timeout = 15000; oReq.onload = function (e) { if (oReq.status == 200) //Response 200 implies that link exist and then most likely a Manga (or an Error XD) { var END = new Date(); //console.info(END - START); var doc = new DOMParser().parseFromString(oReq.response, "text/html"); IllustrationLinker.setMetadata(id, doc); } finalise(); }; oReq.send(); } function finalise() { if (illust.bookmarkCount == "?" && (IllustrationLinker.requestBookmarkCount == 2 || (IllustrationLinker.requestBookmarkCount == 1 && !IsIllustrationPage))) { getBookmarkCount(); } else if (!extensionKnown && illust.pageCount > 1) { extensionKnown = true; getMangaExtension(); } else { IllustrationLinker.createLinksBox(id); if (IllustrationLinker.simultaneousCalls > 0) IllustrationLinker.simultaneousCalls--; } } function getBookmarkCount() { if ((IsIllustrationPage && (Settings.requestMethod & 8)) || (!IsIllustrationPage && (Settings.requestMethod & 4))) { GM_xmlhttpRequest({ url: "http://www.pixiv.net/bookmark_detail.php?illust_id=" + id, method: "GET", timeout: 15000, //headers: { "User-agent": navigator.userAgent, "Accept": "text/html", Referer: "http://www.pixiv.net" }, headers: { "User-agent": navigator.userAgent, "Accept": "text/html" }, onload: function (response) { if (response.status == 200) //Response 200 implies that link exist and then most likely a Manga (or an Error XD) { var doc = new DOMParser().parseFromString(response.responseText, "text/html"); var bm = doc.querySelector(".bookmark-count"); if (bm) illust.bookmarkCount = parseInt(bm.textContent); else illust.bookmarkCount = 0; } finalise(); } }); } else { var oReq = new XMLHttpRequest(); oReq.open("GET", "http://www.pixiv.net/bookmark_detail.php?illust_id=" + id, true); oReq.responseType = "text"; oReq.timeout = 15000; oReq.onload = function (e) { if (oReq.status == 200) //Response 200 implies that link exist and then most likely a Manga (or an Error XD) { var doc = new DOMParser().parseFromString(oReq.response, "text/html"); var bm = doc.querySelector(".bookmark-count"); if (bm) illust.bookmarkCount = parseInt(bm.textContent); else illust.bookmarkCount = 0; } finalise(); }; oReq.send(); } } /* There is no way of finding out the extension of manga files in the new naming method using the illustration page. We need to make another http request. -------------------------------------------------------------------------------------------*/ function getMangaExtension() { if (illust.illust128URL.indexOf("/img-master/") > 0) //New naming format { if ((IsIllustrationPage && (Settings.requestMethod & 8)) || (!IsIllustrationPage && (Settings.requestMethod & 4))) { GM_xmlhttpRequest({ url: "http://www.pixiv.net/member_illust.php?mode=manga_big&illust_id=" + id + "&page=0", method: "GET", timeout: 15000, //headers: { "User-agent": navigator.userAgent, "Accept": "text/html", Referer: "http://www.pixiv.net" }, headers: { "User-agent": navigator.userAgent, "Accept": "text/html" }, onload: function (response) { if (response.status == 200) { var doc = new DOMParser().parseFromString(response.responseText, "text/html"); try { illust.illustURL = doc.getElementsByTagName("img")[0].src; } catch (err) { } } finalise(); } }); } else { var oReq = new XMLHttpRequest(); oReq.open("GET", "http://www.pixiv.net/member_illust.php?mode=manga_big&illust_id=" + id + "&page=0", true); oReq.responseType = "text"; oReq.timeout = 15000; oReq.onload = function (e) { if (oReq.status == 200) { var doc = new DOMParser().parseFromString(oReq.response, "text/html"); try { illust.illustURL = doc.getElementsByTagName("img")[0].src; } catch (err) { } } finalise(); }; oReq.send(); } } else finalise(); //Old naming format } } }; /* =================================================================================================================================== Handles getting the next page ===================================================================================================================================*/ var Pager = { nextPageURL: null, intervalID: null, //Time to wait out before attempting to get the next page timeOutLength: 2000, pause: false, //Event that fires off passing the loaded document onPageLoad: null, //If true it scraps any currently loading pages ageRatingChanged: false, requestingPage: false, scrollOffset: null, /* ------------------------------------------------------------------------------ Initialises the pager. - pageLoadEventHandler must be set as it gets called when a new page is loaded. It takes (document,nextURL) parameters. - scrollOffset function that return the bottom position of the main container. Used to calculate when to load next page. ------------------------------------------------------------------------------*/ initalise: function (pageLoadEventHandler, funcScrollOffset) { this.scrollOffset = funcScrollOffset; this.onPageLoad = pageLoadEventHandler; this.getNextPageURL(document.body); if (this.intervalID) clearInterval(this.intervalID); if (this.nextPageURL) this.intervalID = setInterval(this.checkScrollPosition, 500); return (this.nextPageURL != null); }, checkScrollPosition: function () { if (Pager.ageRatingChanged || !Settings.fetch.nextPage) return; if ((Pager.scrollOffset() - window.scrollY - window.innerHeight) < Settings.pagingOffset) { Pager.getNextPage(); } }, /* ------------------------------------------------------------------------------ Gets the next page url from an element or full document. Returns null if there isn't a next page other the url. ------------------------------------------------------------------------------*/ getNextPageURL: function (xml) { this.nextPageURL = null; var evaluator = new XPathEvaluator(); //document.evaluate var btnNext = evaluator.evaluate(".//a[@rel='next' and @class='_button']", xml, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue; if (btnNext) this.nextPageURL = btnNext.href; //else btnNext = evaluator.evaluate(".//a[@rel='next' and @class='button']", xml, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue; //if (btnNext) this.nextPageURL = btnNext.href; return this.nextPageURL; }, NextPageReceived: function (doc, pageNumber) { DisplayMessage("Page [" + pageNumber + "] received", 1500); if (Pager.onPageLoad) Pager.onPageLoad(doc, Pager.nextPageURL); Pager.getNextPageURL(doc.body); if (Pager.nextPageURL) { setTimeout( function () { Pager.intervalID = setInterval(Pager.checkScrollPosition, 500); } , Pager.timeOutLength); } }, /* ------------------------------------------------------------------------------ Gets the next page asynchronously. ------------------------------------------------------------------------------*/ getNextPage: function () { Pager.requestingPage = true; clearInterval(Pager.intervalID); var pageNumber = PaginatorHQ.getPageNumber(this.nextPageURL); var msg = DisplayMessage("Requesting page [" + pageNumber + "] ..."); if (Settings.requestMethod & 2) { GM_xmlhttpRequest({ url: this.nextPageURL, method: "GET", headers: { "User-agent": navigator.userAgent, "Accept": "document" }, //"Accept": "text/html" timeout: 20000, onload: function (response) { Pager.requestingPage = false; //Change age rate, so no need to process return result as new content will be loaded if (Pager.ageRatingChanged) return; if (response.status == 200) { RemoveMessage(msg); var doc = document.implementation.createHTMLDocument("P++"); doc.documentElement.innerHTML = response.responseText; //Not supported by Opera //var doc = new DOMParser().parseFromString(response.responseText, "text/html") Pager.NextPageReceived(doc, pageNumber); } else Pager.pageErrorTimeout("ERROR: Page [" + pageNumber + "] failed request", response); }, ontimeout: function (response) { RemoveMessage(msg); Pager.pageErrorTimeout("ERROR: Timeout on Page [" + pageNumber + "].", response) }, onerror: function (response) { RemoveMessage(msg); Pager.pageErrorTimeout("ERROR: Page [" + pageNumber + "] failed request", response) } }); } else { var oReq = new XMLHttpRequest(); oReq.open("GET", this.nextPageURL, true); oReq.responseType = "document"; oReq.timeout = 20000; oReq.onload = function (e) { { Pager.requestingPage = false; //Change age rate, so no need to process return result as new content will be loaded if (Pager.ageRatingChanged) return; if (oReq.status == 200) { RemoveMessage(msg); Pager.NextPageReceived(oReq.response, pageNumber); } else Pager.pageErrorTimeout("ERROR: Page [" + pageNumber + "] failed request", oReq); } }; oReq.ontimeout = function (response) { RemoveMessage(msg); Pager.pageErrorTimeout("ERROR: Timeout on Page [" + pageNumber + "].", response) }; oReq.onerror = function (response) { RemoveMessage(msg); Pager.pageErrorTimeout("ERROR: Page [" + pageNumber + "] failed request", response) }; oReq.send(); } }, pageErrorTimeout: function (msg, response) { console.error("Failed to get next page: (" + response.status + ") : " + response.statusText); DisplayMessage(msg, 5000); setTimeout( function () { Pager.intervalID = setInterval(Pager.checkScrollPosition, 500); } , Pager.timeOutLength); Pager.requestingPage = false; } } /* =================================================================================================================================== PaginatorHQ handles all functions to do with formatting the layout of the results ===================================================================================================================================*/ var PaginatorHQ = { pageTable: null, status: 0, styled: false, /* ------------------------------------------------------------------------------------------- Prepares current page for pagination -------------------------------------------------------------------------------------------*/ intialise: function () { PaginatorHQ.setStyles(); PaginatorHQ.removeUnwantedElements(document); var paged = (Pager.getNextPageURL(document) != null); var pageNumber = (paged) ? PaginatorHQ.getPageNumber(document.URL) : 0; console.log("Paged Content: ", paged); if (!paged) //Either has no pages or it is the last page { var containers = PaginatorHQ.getContainers(document); if (PAGETYPE > 1 && containers) containers[0].setAttribute("name", "pageContainer"); //Done for age rating issues. IllustrationLinker.getContainerLinks(containers, pageNumber); } else { var paginator = document.getElementsByClassName("column-order-menu"); while (paginator.length > 1) paginator[1].parentNode.removeChild(paginator[1]); paginator = paginator[0]; paginator.className += " paginator"; var pageContainer = PaginatorHQ.getContainers(document)[0]; if (pageContainer.className == "display_works linkStyleWorks") //display_works linkStyleWorks breaks sets UL style { //You need to add this otherwise "DIV._unit action-unit" element does not expand and will end up with transparent background. var divider = document.createElement("div"); divider.className = "clear"; pageContainer.appendChild(divider); pageContainer = pageContainer.firstElementChild; } else { pageContainer.parentElement.insertBefore(paginator, pageContainer); //pageContainer.insertBefore(paginator, pageContainer.firstElementChild); } pageContainer.style.marginBottom = "0"; pageContainer.style.marginTop = "0"; pageContainer.setAttribute("name", "pageContainer"); pageContainer.setAttribute("page", pageNumber); PaginatorHQ.pageTable = pageContainer.parentElement; IllustrationLinker.getContainerLinks([pageContainer], pageNumber); pageContainer.className += " pppPage" + (pageNumber % 3); Pager.initalise(PaginatorHQ.addNewPage, PaginatorHQ.getMainTableBottom); } PaginatorHQ.updateVisibilityOfAllElements(document); }, /* ------------------------------------------------------------------------------------------- Gets bottom of page table, used to calculate when the next page is going to be loaded -------------------------------------------------------------------------------------------*/ getMainTableBottom: function () { var pos = GetAbsolutePosition(PaginatorHQ.pageTable); return pos.top + PaginatorHQ.pageTable.offsetHeight; }, addStyle: function (id, textContent) { var style = document.createElement("style"); style.type = "text/css"; style.innerHTML = textContent; if (id) style.id = id; document.head.appendChild(style); }, setStyles: function () { if (PaginatorHQ.styled) return; PaginatorHQ.styled = true; PaginatorHQ.addStyle(null, '.IQDBLink{font-family:"Times New Roman";background-color: #000000;border-radius: 3px;padding: 2px 3px 1px 3px;color: #66CCFF !important;font-size: x-small;text-decoration: none;}'); PaginatorHQ.addStyle(null, "#pppMsgBox{position: fixed;z-index: 500;bottom: 30px;left: 10px;padding: 0 20px 0 10px;color: #FF0;background-color: #0000D5;padding-left: 10px;border-style: solid;border-color: #FFF;border: solid;}.pppHotBoxTi, .pppHotBoxTm{position: absolute;top: 0px;left: 0px;z-index: 20;border: 2px solid;}.pppHotBoxTi{background-color: rgba(255,255,0,0.6);border-color: red;width: 16px;height: 16px;}.pppHotBoxTm{background-color: rgba(0,255,255,0.2);border-color: red;width: 30px;height: 30px;}.pppHotBoxI{background-color: rgba(255,255,0,0.6);border: 2px solid rgba(255,0,0,0.6);width: 200px;height: 12px;display: inline-block;}.pppHotBoxTi:visited, .pppHotBoxTm:visited, .pppHotBoxI:visited{border-color: purple;/*background-color: rgba(128,0,128,0.4);*/}.pppPreviewWindow{position: absolute;background-color: #000;}"); PaginatorHQ.addStyle("pppPaged", ".pppPage1{background-color: yellow;}.pppPage2{background-color: #63FF00;}.pppPage0{background-color: #FF00FC;}.pppThumb1{background-color: #FCFCA8 !important;}.pppThumb2{background-color: #D9FFD9 !important;}.pppThumb0{background-color: #EAD3EF !important;}"); PaginatorHQ.addStyle("pppPaginator", ".paginator{background-color: white; margin:0; border: ridge;}"); PaginatorHQ.addStyle("pppVisitedIllusPage", ".image-item > a.work:visited > img { border: 22px solid red; !important}"); //Used to fix broken artist and bookmarks pages PaginatorHQ.addStyle(null, ".pppPage1 > li, .pppPage2 > li, .pppPage0 > li { clear: none; float: none !important; display:inline-block;}"); PaginatorHQ.addStyle(null, ".paginator li {clear:none;float:none;display:inline:block; width:auto;}"); PaginatorHQ.addStyle("linkColours", ".pppThumb0 a:link, .pppThumb1 a:link, .pppThumb2 a:link{color:#0507FF;} .pppThumb0 a:visited, .pppThumb1 a:visited, .pppThumb2 a:visited{color:#BCBBB9;}") }, getPageNumber: function (url) { var pageNumber = url.replace(/.+(&|\?)p=(\d+)$/gi, "$2"); if (isNaN(pageNumber)) pageNumber = 1; return parseInt(pageNumber); }, /* ------------------------------------------------------------------------------------------- Adds new content. -------------------------------------------------------------------------------------------*/ addNewPage: function (doc, url) { PaginatorHQ.removeUnwantedElements(doc); var pageNumber = PaginatorHQ.getPageNumber(url); //PageContainer var pageContainer = PaginatorHQ.getContainers(doc)[0]; pageContainer.setAttribute("name", "pageContainer"); pageContainer.setAttribute("page", pageNumber); pageContainer.style.marginBottom = "0"; pageContainer.style.marginTop = "0"; //Navigation Bar var paginator = doc.getElementsByClassName("column-order-menu")[0]; paginator.className += " paginator pppPagedChild"; for (var i = paginator.children.length - 1; child = paginator.children[i], i >= 0; i--) if (child.className != "pager-container") paginator.removeChild(child); paginator.getElementsByTagName("ul")[0].className = "clear " + paginator.getElementsByTagName("ul")[0].className; pageContainer.className += " pppPagedChild pppPage" + (pageNumber % 3); //TODO: Readjust painator bars //pageContainer.insertBefore(paginator, pageContainer.firstElementChild); PaginatorHQ.pageTable.appendChild(paginator); PaginatorHQ.pageTable.appendChild(pageContainer); //IllustrationLinker.getContainerLinks should always after page is added. IllustrationLinker.getContainerLinks([pageContainer], pageNumber); PaginatorHQ.updateVisibilityOfAllElements(pageContainer); if (Settings.filterSwitchFlag > 0) { pageContainer.style.display = "none"; paginator.style.display = "none"; PaginatorHQ.filterContainer(pageContainer); } }, /* ----------------------------------------------------------------------------------------- Returns containers that contain thumbnails. If doc is left out it uses current document. This does not return dynamic containers, i.e. containers like recommended illustrations whose content are auto-updated by Pixiv. -----------------------------------------------------------------------------------------*/ getContainers: function (doc) { //console.info("Getting Containers"); if (!doc) doc = document; for (var i = 0; i < containerClasses.Paged.length; i++) { var nodes = doc.getElementsByClassName(containerClasses.Paged[i]); if (nodes.length > 0) return nodes; } for (var i = 0; i < containerClasses.UnPaged.length; i++) { var nodes = doc.getElementsByClassName(containerClasses.UnPaged[i]); if (nodes.length > 0) return nodes; } //console.warn("No thumbnail containers"); return null; }, filterThumbnail: function (thumbnail) { if (Settings.filterSwitchFlag == 0) return; var metadata = IllustrationLinker.getIllust(thumbnail.getAttribute("illustration-id")); var filterOut = false; if (metadata.pageCount == null) return; //Filter using age rating filterOut = ((Settings.filterSwitchFlag & 16) == 16 && metadata.R18 == 0) || ((Settings.filterSwitchFlag & 32) == 32 && metadata.R18 == 1) || ((Settings.filterSwitchFlag & 64) == 64 && metadata.R18 == 2); if (!filterOut && (Settings.filterSwitchFlag & 2) == 2) { var flag = Settings.filter.flag; var values = Settings.filters[Settings.filter.set]; try { if ((flag & 2) == 2 && metadata.pageCount == 0) filterOut = true; //Filtering out Illustrations else if ((flag & 4) == 4 && metadata.pageCount > 0) filterOut = true; //Filtering out Manga else if ((flag & 8) == 8 && metadata.bookmarkCount < values[0]) filterOut = true; //Bookmarks else if ((flag & 16) == 16 && metadata.viewCount < values[1]) filterOut = true; //Views else if ((flag & 32) == 32 && metadata.ratings < values[2]) filterOut = true; //Ratings else if ((flag & 64) == 64 && metadata.totalRatings < values[3]) filterOut = true; //Total } catch (e) { }; } if (!filterOut) { var found; var tags = metadata.tags; while (tags.indexOf(" ") >= 0) tags = tags.replace(" ", " "); tags = tags.split(" "); if ((Settings.filterSwitchFlag & 4) == 4) { var tagsF = Settings.filter.tagsInclude; while (tagsF.indexOf(" ") >= 0) tagsF = tagsF.replace(" ", " "); tagsF = tagsF.split(" "); tagsF = Settings.filter.tagsInclude.split(" "); found = true; for (var i = 0; i < tagsF.length; i++) { found = false for (var j = 0; j < tags.length; j++) { if (tagsF[i].trim().toLowerCase() == tags[j].trim().toLowerCase()) { found = true; break; } } if (!found) break; } filterOut = !found; } if (!filterOut && (Settings.filterSwitchFlag & 8) == 8) { var tagsF = Settings.filter.tagsExclude; while (tagsF.indexOf(" ") >= 0) tagsF = tagsF.replace(" ", " "); tagsF = tagsF.split(" "); found = false; for (var i = 0; i < tagsF.length; i++) { for (var j = 0; j < tags.length; j++) { if (tagsF[i].trim().toLowerCase() == tags[j].trim().toLowerCase()) { found = true; break; } } if (found) break; } filterOut = found; } } //Illustration metadata has yet to be retrieved if (filterOut) { PaginatorHQ.filterThumbnailLinks(thumbnail, true); thumbnail.style.display = "none"; } else { PaginatorHQ.filterThumbnailLinks(thumbnail, false); thumbnail.style.display = null; } }, filterThumbnailLinks: function (thumbnail, remove) { var links = thumbnail.querySelectorAll(".pppHotBoxTi, .pppDirectLinks > a"); //var links = thumbnail.querySelectorAll("a"); if (!links) return; for (var i = 0; i < links.length; i++) { if (remove) { if (!links[i].getAttribute("hrefOriginal")) links[i].setAttribute("hrefOriginal", links[i].href); links[i].removeAttribute("href"); } else if (!links[i].href && links[i].getAttribute("hrefOriginal")) { links[i].href = links[i].getAttribute("hrefOriginal"); } } }, filterContainer: function (container) { if (Settings.filterSwitchFlag == 0) return; var mainContainer = document.getElementsByName("pageContainer")[0]; var evaluator = new XPathEvaluator(); var nodesSnapshot = evaluator.evaluate(".//li[@name='pppThumb']", container, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null); for (var j = 0 ; thumb = nodesSnapshot.snapshotItem(j), j < nodesSnapshot.snapshotLength; j++) { PaginatorHQ.filterThumbnail(thumb); if (thumb.parentElement != mainContainer) mainContainer.appendChild(thumb); } return nodesSnapshot.snapshotLength; }, /* ------------------------------------------------------------------------------------------- Goes through all thumbnail containers and filters the result accordingly -------------------------------------------------------------------------------------------*/ filterResult: function () { if (PaginatorHQ.status == 0) { PaginatorHQ.status = 1; setTimeout(PaginatorHQ.filterResult, 0); return true; } if (PaginatorHQ.status == 2) { return false; } PaginatorHQ.status = 2; var msg = DisplayMessage("Filtering content (" + Settings.filterSwitchFlag + ")"); var filtering = (Settings.filterSwitchFlag > 0); var containers = document.getElementsByName("pageContainer"); var paginators = document.getElementsByClassName(" paginator"); var mainContainer = containers[0]; var mainContainerPage = mainContainer.getAttribute("page"); //Parse through page containers and paginators and set visibility according to filter for (var i = 1; container = containers[i], i < containers.length; i++) { containers[i].style.display = (filtering) ? "none" : null; paginators[i].style.display = (filtering) ? "none" : null; } if (filtering) //Displaying all items { mainContainer.style.backgroundColor = "transparent"; for (var i = 0; i < containers.length; i++) { msg.textContent = ("Filtering Page " + (i + 2) + "/" + containers.length); PaginatorHQ.filterContainer(containers[i]); } } else //Filtering out { mainContainer.style.backgroundColor = null; var evaluator = new XPathEvaluator(); var nodesSnapshot = evaluator.evaluate(".//li[@name='pppThumb']", mainContainer, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null); for (var i = 0 ; thumb = nodesSnapshot.snapshotItem(i), i < nodesSnapshot.snapshotLength; i++) { //msg.textContent = ("Filtering thumbnail " + (i + 1) + "/" + nodesSnapshot.snapshotLength); thumbPage = thumb.getAttribute("page"); if (thumbPage > mainContainerPage) containers[thumbPage - mainContainerPage].appendChild(thumb); PaginatorHQ.filterThumbnailLinks(thumb, false); thumb.style.display = null; } } if (filtering) { DisplayMessage("Filtering completed", 2000); // (totalCount - PaginatorHQ.filteredCount) + " (" + totalCount + ")"; } else { DisplayMessage("Filter disabled", 2000); } RemoveMessage(msg); PaginatorHQ.status = 0; PaginatorHQ.displayResultInfo(); PreviewHQ.adjustAllHotboxPositions(); }, getResultCount: function (doc) { if (!doc) doc = document; var evaluator = new XPathEvaluator(); //document.evaluate //var result = evaluator.evaluate("//div[@class='column-label' or '_unit manage-unit']/span[@class='count-badge']", doc, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue; var result = evaluator.evaluate("//div[@class='column-label' or @class='_unit manage-unit']/span[@class='count-badge']", doc, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue; if (result) return parseInt(result.textContent); else return -1; }, /* ------------------------------------------------------------------------------------------- Displays the amount of thumbnails listed -------------------------------------------------------------------------------------------*/ displayResultInfo: function () { if (SideBar.iDoc == null) { setTimeout(function () { PaginatorHQ.displayResultInfo(); }, 250); //Sidebar has yet to load return; } var thumbs = document.getElementsByName("pppThumb"); var it = SideBar.iDoc.getElementById("InfoTable"); var resultCount = PaginatorHQ.getResultCount(); //Number of items returned by search var linked = (thumbs.length - IllustrationLinker.processList.length - IllustrationLinker.simultaneousCalls); if (resultCount < 0) //We do not know how many pages can be displayed. Might not be paged { it.rows[0].style.display = "none"; it.rows[1].style.display = null; it.rows[1].cells[1].textContent = linked + " / " + thumbs.length; //Number of images if (PAGETYPE > 2) //It is paged. Go figure. { var startPage = PaginatorHQ.getPageNumber(document.URL); var pagesLoaded = Math.ceil(thumbs.length / 20); it.rows[0].style.display = null; it.rows[0].cells[1].textContent = pagesLoaded + " (" + startPage + "-" + (startPage + pagesLoaded - 1) + ")"; } } else if (thumbs.length == 0 || resultCount == 0) { it.rows[0].cells[1].textContent = 0; it.rows[1].cells[1].textContent = 0; } else { var visibleCount = thumbs.length; if (Settings.filterSwitchFlag > 0) { visibleCount = 0; for (var i = 0; i < thumbs.length; i++) { if (!thumbs[i].style.display) visibleCount++; } } var pagesLoaded = Math.ceil(thumbs.length / 20); var pageS = PaginatorHQ.getPageNumber(document.URL); var pageE = pageS + pagesLoaded - 1; var pageCount = Math.ceil(resultCount / 20); it.rows[0].cells[1].textContent = pagesLoaded + " (" + pageS + "-" + pageE + " | " + pageCount + ")"; var leftCount = resultCount - ((pageS - 1) * 20) if (Settings.filterSwitchFlag > 0) it.rows[1].cells[1].innerHTML = visibleCount + " / " + linked + " | " + thumbs.length + "<br /> (" + leftCount + " | " + resultCount + ")"; else it.rows[1].cells[1].textContent = linked + " / " + thumbs.length + " (" + leftCount + " | " + resultCount + ")"; } //Adjust frame size to fit info text if (SideBar.iDoc.body.clientHeight < SideBar.iDoc.body.scrollHeight) SideBar.adjustFrameSize(); }, /* ------------------------------------------------------------------------------------------- Sorts the thumbnails according to filter type -------------------------------------------------------------------------------------------*/ sortByScore: function (sort) { if (PaginatorHQ.status == 0) { PaginatorHQ.status = 1; setTimeout(PaginatorHQ.sortByScore, 0, sort); return true; } if (PaginatorHQ.status == 2) { return false; } PaginatorHQ.status = 2; var msg = DisplayMessage("Sorting content"); var containers = document.getElementsByName("pageContainer"); var sortType = Settings.sortType; //A very bad way of doing sort but it seems efficient. Problem is having items spread over containers var evaluator = new XPathEvaluator(); var nodesSnapshot = evaluator.evaluate(".//li[@name='pppThumb']", PaginatorHQ.pageTable, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null); var arr = new Array(); for (var i = 0; thumb = nodesSnapshot.snapshotItem(i), i < nodesSnapshot.snapshotLength; i++) { var value = (sort) ? parseInt(getMetaScore(thumb)[sortType]) : parseInt(thumb.getAttribute("position")); arr.push({ "index": i, "value": value }); } //Stores thumbnail metascore and position in an array and then sorts the array and then proceeds to //to apply sorted result to the thumbnails for (var i = 0 ; i < arr.length; i++) { for (var j = i + 1 ; j < arr.length; j++) { if ((sort && arr[i].value < arr[j].value) || (!sort && arr[i].value > arr[j].value)) { var temp = arr[i].value; arr[i].value = arr[j].value; arr[j].value = temp; temp = arr[i].index; arr[i].index = arr[j].index; arr[j].index = temp; } } } //Apply sorted array to thumbnails for (var i = 0 ; i < arr.length; i++) { if (Settings.filterSwitchFlag > 0) container = containers[0]; //Items are all in one container as filter is enabled else container = containers[Math.floor(i / 20)]; container.appendChild(nodesSnapshot.snapshotItem(arr[i].index)); } var evaluator = new XPathEvaluator(); //document.evaluate var sortButton = evaluator.evaluate("//a[@name='Sort']", SideBar.iDoc.body, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue; if (sort) { DisplayMessage("Sorted by " + FilterTypeToString(sortType), 3000); sortButton.firstElementChild.style.borderColor = "#0F0"; } else { DisplayMessage("Removed sorting", 3000); sortButton.firstElementChild.style.borderColor = "#000"; } RemoveMessage(msg); PaginatorHQ.status = 0; PreviewHQ.adjustAllHotboxPositions(); function getMetaScore(thumbnail) { var illust = IllustrationLinker.getIllust(thumbnail.getAttribute("illustration-id")) return [illust.bookmarkCount, illust.viewCount, illust.ratings, illust.totalRatings]; } }, updateIQDBLink: function (linksBox) { var IQDBLink = linksBox.getElementsByClassName("IQDBLink")[0]; if (Settings.display.IQDBLink) { var directLinks = linksBox.getElementsByClassName("pppDirectLinks")[0].getElementsByTagName("a"); IQDBLink.href = "http://" + Settings.IQDBType + ".IQDB.org/?url=" + linksBox.parentElement.getElementsByTagName('IMG')[0].src + "&fullimage=" + directLinks[0].href + "&"; IQDBLink.style.display = null; } else { IQDBLink.removeAttribute("href"); IQDBLink.style.display = "none"; } }, updateAllIQDBLinks: function () { var linksBoxes = document.getElementsByClassName("pppLinksBox"); for (var i = 0; i < linksBoxes.length; i++) PaginatorHQ.updateIQDBLink(linksBoxes[i]); }, /* Set display information for thumbnails ----------------------------------------------------------------------------------------*/ updateVisibilityOfElement: function (thumbnail) { var el = thumbnail.getElementsByClassName("title2")[0]; if (el) el.style.display = (Settings.display.illustTitle) ? null : "none"; else if (PAGETYPE == 1 || PAGETYPE == 3 || PAGETYPE == 4 || PAGETYPE == 5) //Bookmark page { var el = thumbnail.getElementsByTagName("H1")[0]; if (el) el.style.display = (Settings.display.illustTitle) ? null : "none"; } el = thumbnail.getElementsByClassName("user")[0]; if (el) el.style.display = (Settings.display.artistName) ? "block" : "none"; el = thumbnail.getElementsByClassName("count-list")[0]; if (el) el.style.display = (Settings.display.countList) ? null : "none"; var links = thumbnail.getElementsByClassName("pppDirectLinks")[0]; if (links) { if (links.getElementsByTagName("a").length == 1) links.style.display = (Settings.display.illustLink) ? null : "none"; else links.style.display = (Settings.display.mangaLinks) ? null : "none"; } }, /* ---------------------------------------------------------------------------------------- Clean's up the thumbnail and adds any missing information. It gets called by the IllustrationLinker.createLinkBox function when the metadata has been recieved. ----------------------------------------------------------------------------------------*/ tidyThumbnail: function (id) { var thumbnail = document.getElementById("i" + id), link = thumbnail.getElementsByTagName("a")[0], metadata = IllustrationLinker.getIllust(id); //We clean up here. Removing anything we do not desire or wanting to recreate //var el = thumbnail.getElementsByClassName("count-list"); //if (el.length == 1) thumbnail.removeChild(el[0]); //var el = thumbnail.getElementsByClassName("f10"); //if (el.length == 1) thumbnail.removeChild(el[0]); //el = thumbnail.getElementsByClassName("bookmark_artist"); //if (el.length == 1) thumbnail.removeChild(el[0]); TSL.removeNode(thumbnail.querySelector(".count-list")); TSL.removeNode(thumbnail.querySelector(".f10")); TSL.removeNode(thumbnail.querySelector(".bookmark_artist")); TSL.removeNode(thumbnail.querySelector("h1.title")); //Cleans image link by first removing any unwanted text. This is the case in Bookmarks. var brs = thumbnail.getElementsByTagName("br"); while (brs.length > 0) brs[0].parentElement.removeChild(brs[0]); if (link.getElementsByClassName("title2").length == 0) { link.innerHTML = link.innerHTML.replace(/(<img[^>]*>).*/gi, "$1"); link.style.display = "block"; var h1 = document.createElement("h1"); h1.className = "title2"; h1.textContent = metadata.illustTitle; link.appendChild(h1); } if (thumbnail.getElementsByClassName("user").length == 0) { var el = thumbnail.getElementsByClassName("f10"); if (el.length == 1) thumbnail.removeChild(el[0]); el = thumbnail.getElementsByClassName("bookmark_artist"); if (el.length == 1) thumbnail.removeChild(el[0]); var a = document.createElement("a"); a.className = "user"; a.textContent = metadata.userName; a.href = "http://www.pixiv.net/member.php?id=" + metadata.userID; thumbnail.insertBefore(a, link.nextElementSibling); } var ul = document.createElement("ul"); ul.className = "count-list"; ul.innerHTML = '<a href="/bookmark_detail.php?illust_id=' + metadata.illustID + '" class="bookmark-count ui-tooltip" data-tooltip="Received ' + metadata.bookmarkCount + ' bookmarks"><i class="_icon sprites-bookmark-badge"></i>' + metadata.bookmarkCount + '</a>' + '<a href="/response.php?type=illust&id=' + metadata.illustID + '" class="image-response-count ui-tooltip" data-tooltip="Received ' + metadata.responseCount + ' image responses"><i class="_icon sprites-image-response-badge"></i>' + metadata.responseCount + '</a>'; thumbnail.appendChild(ul); //<ul class="count-list"><li><a href="/bookmark_detail.php?illust_id=33473196" class="bookmark-count ui-tooltip" data-tooltip="Received 31 bookmarks"><i class="_icon sprites-bookmark-badge"></i>31</a></li><li><a href="/response.php?type=illust&id=33473196" class="image-response-count ui-tooltip" data-tooltip="Received 1 image responses"><i class="_icon sprites-image-response-badge"></i>1</a></li></ul> PaginatorHQ.updateVisibilityOfElement(thumbnail); }, /* ---------------------------------------------------------------------------------------- Sets the visibility of the elements according to sidebar settings. ----------------------------------------------------------------------------------------*/ updateVisibilityOfAllElements: function (container) { if (!container) container = document; var evaluator = new XPathEvaluator(); var nodesSnapshot = evaluator.evaluate(".//li[@name='pppThumb']", container, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null); for (var i = 0 ; thumb = nodesSnapshot.snapshotItem(i), i < nodesSnapshot.snapshotLength; i++) { PaginatorHQ.updateVisibilityOfElement(nodesSnapshot.snapshotItem(i)); } PreviewHQ.adjustAllHotboxPositions(document); }, removeUnwantedElements: function (doc) { var classes = ["popular-introduction", "user-ad-container"]; var ids = ["header-banner"]; for (var i = 0; i < classes.length; i++) { var els = doc.getElementsByClassName(classes[i]); while (els.length > 0) els[0].parentNode.removeChild(els[0]); } for (var i = 0; i < ids.length; i++) { var el = doc.getElementById(ids[i]); if (el) el.parentNode.removeChild(el); } } }; /* =================================================================================================================================== Handles the preview dialog window ===================================================================================================================================*/ var PreviewHQ = { interval: null, intervalImageLoadCheck: null, delay: new Object(), /* ------------------------------------------------------------------------------------------- Creates hotbox and it's event handlers -------------------------------------------------------------------------------------------*/ createHotBox: function (id) { var metadata = IllustrationLinker.getIllust(id); var hotbox = document.createElement("a"); hotbox.name = "HotBox"; document.getElementById("i" + id).querySelector("a").appendChild(hotbox); if (metadata.illustType != 2) { hotbox.className = "pppHotBoxTi"; hotbox.href = metadata.illustURL; } else { hotbox.className += "pppHotBoxTm"; hotbox.href = "http://www.pixiv.net/member_illust.php?mode=manga&illust_id=" + metadata.illustID; //DirectLink to Manga Page Viewer } PreviewHQ.adjustHotboxPosition(hotbox); hotbox.onmouseenter = function () { if (PreviewHQ.delay.iID != metadata.illustID) { PreviewHQ.delay.tID = setTimeout(function () { //PreviewHQ.delay = new object(); PreviewHQ.onMouseOverHBShowPreview(hotbox); }, 250); } if (PreviewHQ.interval) { clearTimeout(PreviewHQ.interval); PreviewHQ.interval = null; } } hotbox.onmouseout = PreviewHQ.onMouseOut; }, /* ------------------------------------------------------------------------------------------- Adjust hotbox position. Called often because images sometimes not loaded, or when elements are removed the hotbox position goes out of whack. -------------------------------------------------------------------------------------------*/ adjustHotboxPosition: function (hotbox) { //If image is not loaded you might need to readjust position. //var img = hotbox.parentElement.getElementsByTagName("img")[0]; var img = hotbox.parentElement.getElementsByClassName("_layout-thumbnail")[0]; //hotbox.style.left = (img.offsetLeft + 2) + "px"; //hotbox.style.top = (img.offsetTop + 2) + "px"; hotbox.style.left = img.offsetLeft + "px"; hotbox.style.top = img.offsetTop + "px"; }, /* ------------------------------------------------------------------------------------------- Parses through all hotboxes and adjusts them ------------------------------------------------------------------------------------------*/ adjustAllHotboxPositions: function (container) { clearInterval(PreviewHQ.cid); var hotboxes = document.getElementsByName("HotBox"); var i = 0; PreviewHQ.cid = setInterval(AP, 0); function AP() { if (i > hotboxes.length) clearInterval(PreviewHQ.cid); PreviewHQ.adjustHotboxPosition(hotboxes[i]); i++ } }, /* ------------------------------------------------------------------------------------------- Loads preview window when mouse hovers over hotbox ------------------------------------------------------------------------------------------*/ onMouseOverHBShowPreview: function (hotbox) { //var hotbox = e.target; var pThumb = hotbox.parentElement.parentElement; var imageID = pThumb.getAttribute("illustration-id"); var metadata = IllustrationLinker.getIllust(imageID); var previewWindow = document.getElementById("previewWindow"); if (previewWindow && previewWindow.getAttribute("illustration-id") == imageID) return; else if (previewWindow) document.body.removeChild(previewWindow); if (PreviewHQ.intervalImageLoadCheck) clearInterval(PreviewHQ.intervalImageLoadCheck); PreviewHQ.intervalImageLoadCheck = null; previewWindow = document.createElement("div"); previewWindow.id = "previewWindow"; document.body.appendChild(previewWindow); previewWindow.setAttribute("illustration-id", imageID); previewWindow.onmouseout = PreviewHQ.onMouseOut; previewWindow.onmouseover = function () { if (PreviewHQ.interval) { clearTimeout(PreviewHQ.interval); PreviewHQ.interval = null; } }; var metascore = [metadata.bookmarkCount, metadata.viewCount, metadata.ratings, metadata.totalRatings, metadata.responseCount]; var metaname = ["★", "Views", "Rating", "Total", "Response"]; var data = document.createElement("div"); data.className = "pwMetadata"; var info = document.createElement("div"); var a = document.createElement("a"); a.href = "http://www.pixiv.net/member.php?id=" + metadata.userID //if (metadata.userProfileImageURL.indexOf("http://") == 0) a.innerHTML = "<img src='" + metadata.userProfileImageURL + "' / >"; a.innerHTML = '<img alt="" src="" />'; info.appendChild(a); a = document.createElement("a"); a.style.color = "#00FD00"; a.href = "http://www.pixiv.net/member_illust.php?id=" + metadata.userID; a.textContent = metadata.userName; info.appendChild(a); a = document.createElement("a"); a.style.marginRight = "10px"; a.style.color = "#00FD00"; a.textContent = metadata.illustTitle; if (metadata.pageCount > 1) a.href = "http://www.pixiv.net/member_illust.php?mode=manga&illust_id=" + metadata.illustID; else a.href = "http://www.pixiv.net/member_illust.php?mode=medium&illust_id=" + metadata.illustID + "#SKIP"; info.innerHTML = a.outerHTML + "(" + info.innerHTML + ")"; data.appendChild(info); var IQDBPrefix = "http://" + Settings.IQDBType + ".IQDB.org/?url="; //Different informational links //http://www.pixiv.net/bookmark_detail.php?illust_id=34470652 //http://www.pixiv.net/response.php?type=illust&id=34470652 //http://www.pixiv.net/member_illust.php?id=68447 var links; links = pThumb.getElementsByClassName("pppDirectLinks")[0].getElementsByTagName("a"); if (metadata.pageCount > 1) data.innerHTML = "<a class='bookmark-count' style='background-color:#81FF00;'>P" + metadata.pageCount + "</a>"; data.innerHTML += "<a class='bookmark-count' style='background-color:#FFFD00;' href='http://" + Settings.IQDBType + ".IQDB.org/?url=" + ((metadata.illustType == 3) ? metadata.illust480URL : metadata.illust150URL) + "&NULL'>IQDB<a>"; data.setAttribute("style", "padding: 2px 5px 2px 5px; text-decoration: none; color: #FFF; font-weight:bold;"); data.innerHTML += '<a href="/response.php?type=illust&id=' + imageID + '" class="image-response-count ui-tooltip" data-tooltip="Received ' + metascore[4] + ' image responses"><i class="_icon sprites-image-response-badge"></i>' + metascore[4] + '</a>'; data.innerHTML += '<a href="/bookmark_detail.php?illust_id=' + imageID + '" class="bookmark-count ui-tooltip" data-tooltip="Received ' + metascore[0] + ' bookmarks"><i class="_icon sprites-bookmark-badge"></i>' + metascore[0] + '</a>'; for (var i = 1; i < 4; i++) data.innerHTML += "<span class='bookmark-count' style='background-color:#FDFDFB;'>" + metaname[i] + " " + metascore[i] + "</span>"; previewWindow.setAttribute("style", "position: absolute; z-index: 1000; background-color:#000;border: 5px ridge #565757; max-width: 95%;"); if (metadata.illustType == 2 && metadata.pageCount > 1) //Manga { TSL.addStyle("MangaPreview", "#ThumbContainer{top: 0; left: 0; overflow: auto; white-space: nowrap; max-width: 100%; width: 100%;}" + "#ThumbContainer > a > img {border: 2px ridge #FFF; max-width: 128px;}"); var thumbContainer = document.createElement("div"); thumbContainer.id = "ThumbContainer"; thumbContainer.setAttribute("style", ""); for (var i = 0; i < links.length; i++) { var link = document.createElement("a"); var thumbnail = document.createElement("img"); thumbnail.setAttribute("style", "border: 2px ridge #FFF;"); thumbnail.setAttribute("style", "border: 2px ridge #FFF; max-width: 128px;"); thumbnail.src = metadata.illust150URL.replace(/_p\d+/, "_p" + i); thumbnail.alt = "Page #" + i; link.href = links[i].href; link.appendChild(thumbnail); thumbContainer.appendChild(link); thumbnail.onload = (function () { setTimeout(function () { PreviewHQ.adjustPreviewWindow(previewWindow, hotbox); }, 1000); })(); } previewWindow.appendChild(thumbContainer); } else { var prethumb = document.createElement("div"); var link = document.createElement("a"); link.href = links[0].href; var preImg = document.createElement("img"); preImg.setAttribute("style", "margin: auto auto; height: " + Settings.preview.height + "px;text-align:center;"); var imgsrc = (Settings.preview.height > 450) ? metadata.illust600URL : metadata.illust480URL; var topOffSet = 0; if (Settings.display.autoPreview) { preImg.src = imgsrc; prethumb.setAttribute("style", "height: " + Settings.preview.height + "px;text-align:center;"); preImg.alt = "Loading image"; } else { preImg.alt = "Click to load image"; prethumb.setAttribute("style", "text-align:center;"); topOffSet = 100; preImg.onclick = function () { preImg.onclick = null; preImg.onmouseover = null; preImg.alt = "Loading image"; preImg.src = imgsrc; prethumb.setAttribute("style", "height: " + Settings.preview.height + "px;text-align:center;"); PreviewHQ.adjustPreviewWindow(previewWindow, hotbox, topOffSet); return false; }; } PreviewHQ.intervalImageLoadCheck = setInterval(function () { if (preImg.naturalWidth > 0) { clearInterval(PreviewHQ.intervalImageLoadCheck); PreviewHQ.intervalImageLoadCheck = null; PreviewHQ.adjustPreviewWindow(previewWindow, hotbox, topOffSet); //console.log(preImg.width, 'x', preImg.height); } }, 0); //if (preImg.offsetWidth == 0) preImg.onload = function () { preImg.onload = null; }; //PreviewHQ.adjustPreviewWindow(previewWindow, hotbox, 65); link.appendChild(preImg); prethumb.appendChild(link) previewWindow.appendChild(prethumb); } previewWindow.appendChild(data); PreviewHQ.adjustPreviewWindow(previewWindow, hotbox); }, onMouseOut: function (e) { //Clear preview timeout so it does not display if hover is too short clearTimeout(PreviewHQ.delay.tID); PreviewHQ.delay = new Object(); PreviewHQ.interval = setTimeout(function () { if (PreviewHQ.intervalImageLoadCheck) clearInterval(PreviewHQ.intervalImageLoadCheck); PreviewHQ.intervalImageLoadCheck = null; var previewWindow = document.getElementById("previewWindow"); //if (previewWindow.getElementsByTagName("img")[0]) previewWindow.getElementsByTagName("img")[0].onload = null; if (previewWindow) document.body.removeChild(previewWindow); PreviewHQ.interval = null; }, Settings.preview.timeLength); }, adjustPreviewWindow: function (previewWindow, hotbox, topOffset) { if (previewWindow.offsetWidth == 0) return; //PreviewWindow unloaded. Setting onload to null in onMouseOut does not work. if (!topOffset) topOffset = 0; var offset = GetAbsolutePosition(hotbox); //TODO: adjustPreviewWindow: Change it to take into account top scroll-space, so it should not always choose bottom preview by default. //console.log(hotbox.getClientRects()[0].top, offset.top, previewWindow.offsetHeight, offset.top - previewWindow.offsetHeight) previewWindow.style.top = (offset.top - previewWindow.offsetHeight + 2) + "px"; if ((previewWindow.getClientRects()[0].top) < 0) { previewWindow.style.top = (offset.top + hotbox.offsetHeight - topOffset + 4) + "px"; var data = previewWindow.getElementsByClassName("pwMetadata")[0]; if (data != previewWindow.firstElementChild) previewWindow.insertBefore(data, previewWindow.firstElementChild); } previewWindow.style.left = 0 + "px"; //You do this to stop the wrapping that occurs with the metadata on the right most thumbnails previewWindow.style.left = (offset.left - (previewWindow.offsetWidth / 2) + Math.round(hotbox.offsetWidth / 2)) + "px"; var right = previewWindow.getClientRects()[0].right; if (right > document.body.clientWidth) previewWindow.style.left = (document.body.clientWidth - previewWindow.offsetWidth - 10) + "px"; var left = previewWindow.getClientRects()[0].left; if (left < 0) previewWindow.style.left = "0px"; } } /* =================================================================================================================================== Handles all events and functions to do with the sidebar ===================================================================================================================================*/ var SideBar = { iframe: null, sidebar: null, iDoc: null, interval: null, scrollWidth: (function getScrollBarWidth() { var inner = document.createElement('p'); inner.style.width = "100%"; inner.style.height = "200px"; var outer = document.createElement('div'); outer.style.position = "absolute"; outer.style.top = "0px"; outer.style.left = "0px"; outer.style.visibility = "hidden"; outer.style.width = "200px"; outer.style.height = "150px"; outer.style.overflow = "hidden"; outer.appendChild(inner); document.body.appendChild(outer); var w1 = inner.offsetWidth; outer.style.overflow = 'scroll'; var w2 = inner.offsetWidth; if (w1 == w2) w2 = outer.clientWidth; document.body.removeChild(outer); return (w1 - w2); })(), addStyle: function (id, styleScript) { var style = SideBar.iDoc.createElement("style"); style.type = "text/css"; style.innerHTML = styleScript; if (id) style.id = id; SideBar.iDoc.head.appendChild(style); }, initalise: function () { if (PAGETYPE < 1) return; var img = document.createElement("img"); img.setAttribute("style", "position:fixed; z-index: 100; top: 15px; left: 10px;"); img.src = "" document.body.appendChild(img); img.onmouseover = function () { img.style.cursor = "pointer"; }; img.onmousedown = function () { SideBar.iframe.style.visibility = null; SideBar.adjustFrameSize(); document.getElementById("wrapper").style.margin = "10px 20px 0px auto"; Settings.display.sidebar = true; Settings.saveSettings(); }; var iframe = document.createElement("iframe"); iframe.onload = function () { var iDoc = iframe.contentDocument || iframe.contentWindow.document; iDoc.head.innerHTML = '<meta charset="utf-8" />'; iDoc.body.innerHTML = '<section id="PSideBar" style="min-width: 130px;"><div style="text-align: right; margin: 2px 5px 0 0; padding: 0;"><span style="float: left; display: inline-block; margin-left: 5px;"><a name="nextPage" href="#" title="Paging" class="buttonR buttonOn"><span></span></a><a name="metadata" href="#" title="Illustartion Linker" class="buttonG buttonOn"><span></span></a></span><a href="#" id="SidebarClose"><img src="" /></a></div><div id="hiddenCtrl" style="font-size: x-small; text-align: center; margin: 2px; display: none;"><input id="pagingOffset" type="text" style="width: 40px; margin: 0px 3px;" title="Page Offset" /><select id="pageFetchMethod" title="Page Fetch Method"><option title="GM_xmlHttpRequest">1G</option><option title="xmlHttpRequest">2X</option><option title="iframe">3F</option></select></div><hr style="margin-top: 2px;" /><div><table id="InfoTable"><tbody><tr><td>Pages</td><td></td></tr><tr><td>Images</td><td></td></tr></tbody></table></div><hr /><section><div style="text-align: center;"><a name="displayOptions" href="#"><div class="bgiY blockButtonOn" style="border-color: yellow;">Display Options</div></a></div><div id="DisplayOptions" class="aY" style="border: 2px solid yellow; padding-bottom: 5px;"><table class="switchTable"><tbody><tr class="switchCellOn"><td><a name="artistName" href="#"><div></div></a></td><td><a name="artistName" href="#">Artist Name</a></td></tr><tr class="switchCellOn"><td><a name="illustTitle" href="#"><div></div></a></td><td><a name="illustTitle" href="#">Illustration Title</a></td></tr><tr class="switchCellOff"><td><a name="illustLink" href="#"><div></div></a></td><td><a name="illustLink" href="#">Illustration Link</a></td></tr><tr class="switchCellOff"><td><a name="mangaLinks" href="#"><div></div></a></td><td><a name="mangaLinks" href="#">Manga Links</a></td></tr><tr class="switchCellOn"><td><a name="countList" href="#"><div></div></a></td><td><a name="countList" href="#">Count-List</a></td></tr></tbody></table><hr /><table class="switchTable"><tbody><tr class="switchCellOff"><td><a name="IQDBLink" href="#"><div></div></a></td><td><a name="IQDBLink" href="#">IQDB Link</a></td></tr></tbody></table><div class="selectHolder"><select id="IQDBOptions"><option>danbooru</option></select></div><hr /><table class="switchTable"><tbody><tr class="switchCellOn"><td><a name="autoPreview" href="#"><div></div></a></td><td><a name="autoPreview" href="#">Preview Image</a></td></tr></tbody></table><div class="selectHolder"><select id="previewHeight"><option>250px</option></select><select id="previewLength"><option>10ms</option></select></div></div></section><section><div style="text-align: center;"><a name="filterOptions" href="#"><div class="bgiC blockButtonOn" style="border-color: cyan;">Filter Options</div></a></div><div id="FilterOptions" class="aC" style="border: 2px solid cyan;"><ul><li><input type="checkbox" name="ageRating" />Safe</li><li><input type="checkbox" name="ageRating" />R18</li><li><input type="checkbox" name="ageRating" />R18G</li></ul><table class="filterTable"><tbody><tr><td><input id="tagsInclude" type="text" /></td><td><a name="tagsInclude" href="#"><div class="bgiG"></div></a></td></tr><tr><td><input id="tagsExclude" type="text" /></td><td><a name="tagsExclude" href="#"><div class="bgiG"></div></a></td></tr></tbody></table><hr /><div style="text-align: center;"><a name="filterSwitch" href="#"><div class="miniButtonOff" style="width: 80%;">Filter Options</div></a></div><table><tbody><tr><td style="width: 16px;"><input type="checkbox" name="filterType" value="2" checked="checked" /></td><td colspan="3">Illustrations</td></tr><tr><td><input type="checkbox" name="filterType" value="4" checked="checked" /></td><td colspan="3">Mangas</td></tr><tr><td><input type="checkbox" name="filterType" value="8" checked="checked" /></td><td>Bookmarks</td><td><input class="filterValue" type="text" /></td><td style="width: 10px;"><input type="radio" name="sortType" value="0" /></td></tr><tr><td><input type="checkbox" name="filterType" value="16" checked="checked" /></td><td>Views</td><td><input class="filterValue" type="text" /></td><td><input type="radio" name="sortType" value="1" /></td></tr><tr><td><input type="checkbox" name="filterType" value="32" checked="checked" /></td><td>Ratings</td><td><input class="filterValue" type="text" /></td><td><input type="radio" name="sortType" value="2" /></td></tr><tr><td><input type="checkbox" name="filterType" value="32" checked="checked" /></td><td>Total</td><td><input class="filterValue" type="text" /></td><td><input type="radio" name="sortType" value="3" /></td></tr></tbody></table><div style="text-align: center; margin-bottom: 3px;"><a name="Sort" href="#"><div class="blockButtonOn bgiC" style="width: 50px;">Sort</div></a><a name="Unsort" href="#"><div class="blockButtonOn bgiC" style="width: 50px;">Unsort</div></a></div><ul class="filterSet"><li><span style="background-color: yellow;"><input type="radio" name="filterSet" value="0" /></span></li><li><span style="background-color: pink;"><input type="radio" name="filterSet" value="1" /></span></li><li><span style="background-color: red;"><input type="radio" name="filterSet" value="2" /></span></li><li><span style="background-color: darkgreen;"><input type="radio" name="filterSet" value="3" /></span></li><li><span style="background-color: blue;"><input type="radio" name="filterSet" value="4" /></span></li></ul></div></section></section>'; var sidebar = iDoc.body.firstElementChild; iDoc.body.appendChild(sidebar); SideBar.sidebar = sidebar; SideBar.iDoc = iDoc; SideBar.iframe = iframe; SideBar.addStyle(null, ".bgiY{background-color: yellow;background-image: linear-gradient(bottom, rgb(255,255,250) 0%, rgb(255,255,0) 50%, rgb(255,255,250) 100%);background-image: -o-linear-gradient(bottom, rgb(255,255,250) 0%, rgb(255,255,0) 50%, rgb(255,255,250) 100%);background-image: -moz-linear-gradient(bottom, rgb(255,255,250) 0%, rgb(255,255,0) 50%, rgb(255,255,250) 100%);background-image: -webkit-linear-gradient(bottom, rgb(255,255,250) 0%, rgb(255,255,0) 50%, rgb(255,255,250) 100%);background-image: -ms-linear-gradient(bottom, rgb(255,255,250) 0%, rgb(255,255,0) 50%, rgb(255,255,250) 100%);background-image: -webkit-gradient( linear, left bottom, left top, color-stop(0, rgb(255,255,250)), color-stop(0.5, rgb(255,255,0)), color-stop(1, rgb(255,255,250)) );}.bgiC{background-color: cyan;background-image: linear-gradient(bottom, rgb(200,255,255) 0%, rgb(0,255,255) 50%, rgb(200,255,255) 100%);background-image: -o-linear-gradient(bottom, rgb(200,255,255) 0%, rgb(0,255,255) 50%, rgb(200,255,255) 100%);background-image: -moz-linear-gradient(bottom, rgb(200,255,255) 0%, rgb(0,255,255) 50%, rgb(200,255,255) 100%);background-image: -webkit-linear-gradient(bottom, rgb(200,255,255) 0%, rgb(0,255,255) 50%, rgb(200,255,255) 100%);background-image: -ms-linear-gradient(bottom, rgb(200,255,255) 0%, rgb(0,255,255) 50%, rgb(200,255,255) 100%);background-image: -webkit-gradient( linear, left bottom, left top, color-stop(0, rgb(200,255,255)), color-stop(0.5, rgb(0,255,255)), color-stop(1, rgb(200,255,255)) );}.bgiL, .switchCellOn > td:first-child, .miniButtonOn{background-color: rgb(0,255,0);background-image: linear-gradient(bottom,rgb(200,255,200) 0%, rgb(0,255,0) 50%, rgb(200,255,200) 100%);background-image: -o-linear-gradient(bottom, rgb(200,255,200) 0%, rgb(0,255,0) 50%, rgb(200,255,200) 100%);background-image: -moz-linear-gradient(bottom, rgb(200,255,200) 0%, rgb(0,255,0) 50%, rgb(200,255,200) 100%);background-image: -webkit-linear-gradient(bottom, rgb(200,255,200) 0%, rgb(0,255,0) 50%, rgb(200,255,200) 100%);background-image: -ms-linear-gradient(bottom, rgb(200,255,200) 0%, rgb(0,255,0) 50%, rgb(200,255,200) 100%);background-image: -webkit-gradient( linear, left bottom, left top, color-stop(0, rgb(200,255,200)), color-stop(0.5, rgb(0,255,0) ), color-stop(1, rgb(200,255,200)) );}.bgiG, .switchCellOff > td:first-child, .miniButtonOff{background-color: rgb(156,156,156);background-image: linear-gradient(bottom, rgb(238,238,238) 0%, rgb(156,156,156) 50%, rgb(238,238,238) 100%);background-image: -o-linear-gradient(bottom, rgb(238,238,238) 0%, rgb(156,156,156) 50%, rgb(238,238,238) 100%);background-image: -moz-linear-gradient(bottom, rgb(238,238,238) 0%, rgb(156,156,156) 50%, rgb(238,238,238) 100%);background-image: -webkit-linear-gradient(bottom, rgb(238,238,238) 0%, rgb(156,156,156) 50%, rgb(238,238,238) 100%);background-image: -ms-linear-gradient(bottom, rgb(238,238,238) 0%, rgb(156,156,156) 50%, rgb(238,238,238) 100%);background-image: -webkit-gradient( linear, left bottom, left top, color-stop(0, rgb(238,238,238)), color-stop(0.5, rgb(156,156,156)), color-stop(1, rgb(238,238,238)) );}.bgiB, #PSideBar, .blockButtonOff{background-color: black;background-image: linear-gradient(bottom, rgb(80,80,80) 0%, rgb(0,0,0) 50%, rgb(80,80,80) 100%);background-image: -o-linear-gradient(bottom, rgb(80,80,80) 0%, rgb(0,0,0) 50%, rgb(80,80,80) 100%);background-image: -moz-linear-gradient(bottom, rgb(80,80,80) 0%, rgb(0,0,0) 50%, rgb(80,80,80) 100%);background-image: -webkit-linear-gradient(bottom, rgb(80,80,80) 0%, rgb(0,0,0) 50%, rgb(80,80,80) 100%);background-image: -ms-linear-gradient(bottom, rgb(80,80,80) 0%, rgb(0,0,0) 50%, rgb(80,80,80) 100%);background-image: -webkit-gradient( linear, left bottom, left top, color-stop(0, rgb(80,80,80)), color-stop(0.5, rgb(0,0,0)), color-stop(1, rgb(80,80,80)) );}.aY a:link, .aY a:visited{color: yellow;}.aC a:link, .aC a:visited{color: cyan;}a, td, input, select, option, ul, span{text-decoration: none;font-size: small;}a, td, ul{color: white;}input, select, option, span{text-align: right;font-size: smaller;}hr{margin: 5px;}"); SideBar.addStyle(null, "#PSideBar{top: 0px;left: 0px;border: 5px ridge black;display: inline-block;color: white;}#PSideBar > div, #PSideBar > section > div{margin: 2px 5px 2px 5px;}.blockButtonOff, .blockButtonOn, .miniButtonOn, .miniButtonOff{width: 90%;display: inline-block;text-align: center;border-radius: 20px;border: 2px solid;}.blockButtonOn{color: black;}.blockButtonOff{color: rgb(156,156,156);}.miniButtonOff, .miniButtonOn{color: black;}.switchTable, .filterTable{padding: 2px 5px 0px 5px;text-decoration: none;}.switchTable * tr > td:first-child > a > div{width: 100%;height: 100%;}.switchTable * tr > td:first-child{width: 12px;height: 12px;border-radius: 10px;display: inline-block;}.switchCellOff > td:last-child > a:link, .switchCellOff a:visited{color: white;}#InfoTable tr > td:first-child{padding-right: 5px;border-right: 1px solid gray;}#InfoTable tr > td:last-child{padding-left: 5px;}.selectHolder{padding: 0 8px 0 8px;}.selectHolder > select{font-size: small;width: 100%;}ul{margin: 0;padding: 0;list-style-type: none;text-align: center;}ul > li{display: inline;border-right: 1px solid gray;padding-right: 5px;}ul > li:last-child{border-right: none;}ul > li > span{display: inline-block;}.filterSet{margin-bottom: 5px;}.filterSet > li{border: none;padding-right: 1px;}.filterSet input{margin: 4px;}.filterValue{color: black;width: 38px;}.buttonG > span{background-image: url();}.buttonR > span{background-image: url();}.buttonOff > span, .buttonOn > span{height: 20px;width: 20px;display: inline-block;}.buttonOff > span{background-image: url();}.filterTable div{width: 15px;height: 15px;border-radius: 2px;border: 2px solid black;}.filterTable input{text-align: left;}"); // Settings to hide according to the page type. if (Settings.displayHidden & 2) if (PAGETYPE > 1 && Settings.displayHidden & 2) { var hc = iDoc.getElementById("hiddenCtrl"); hc.style.display = null; hc.style.textAlign = "left"; var offset = iDoc.getElementById("pagingOffset"); offset.value = Settings.pagingOffset; offset.onfocus = function (e) { e.target.select(); } offset.oninput = function () { if (offset.value.length > 5 || isNaN(offset.value[offset.value.length - 1])) offset.value = offset.value.substring(0, offset.value.length - 1); if (isNaN(offset.value) || offset.value < 100) Settings.pagingOffset = 100; else Settings.pagingOffset = offset.value; Settings.saveSettings(); }; hc.removeChild(iDoc.getElementById("pageFetchMethod")); hc.title = "If checked implies usage of GM_xmlhttpRequest rather than XMLHttpRequest.\nP > Next page fetching.\nA > All pages apart from illustration page. \nI > Illustration page" hc.innerHTML += '<select id="gbm" title="Get Bookmark on..."><option>None</option><option>All-I</option><option>All+I</option></select>' hc.innerHTML += 'P<input type="checkbox" />A<input type="checkbox" />I<input type="checkbox" />'; var els = iDoc.getElementById("gbm"); els.selectedIndex = GM_getValue("RequestBookmarkCount", 0); els.onchange = function () { GM_setValue("RequestBookmarkCount", this.selectedIndex); }; var els = hc.querySelectorAll("input[type=checkbox]"); for (var i = 0; i < els.length; i++) { els[i].checked = (Settings.requestMethod & Math.pow(2, i + 1)); els[i].onclick = function () { Settings.requestMethod = 0; var chk = iDoc.querySelectorAll("#hiddenCtrl input[type=checkbox]"); for (var j = 0; j < chk.length; j++) { Settings.requestMethod += (chk[j].checked) ? Math.pow(2, j + 1) : 0; } console.log("New Request Method: " + Settings.requestMethod); GM_setValue("RequestMethod", Settings.requestMethod) }; } } else { iDoc.getElementById("pagingOffset").style.display = "none"; iDoc.getElementById("pageFetchMethod").style.display = "none"; } if (PAGETYPE < 2) { iDoc.getElementById("FilterOptions").parentElement.style.display = "none"; iDoc.getElementsByName("nextPage")[0].style.display = "none"; } else { iDoc.getElementById("FilterOptions").style.display = (Settings.display.filterOptions) ? null : "none"; //How to handle AgeRating functions if (PAGETYPE != 8) { var ars = iDoc.getElementsByName("ageRating"); for (var i = 0; i < ars.length; i++) { ars[i].onclick = SideBar.onAgeRatingFilter; } } else iDoc.getElementsByName("ageRating")[0].parentElement.parentElement.style.display = "none"; } iDoc.getElementById("DisplayOptions").style.display = (Settings.display.displayOptions) ? null : "none"; var switchLinks = sidebar.getElementsByTagName("a"); for (var i = 0; s = switchLinks[i], i < switchLinks.length; i++) { if (s.id == "SidebarClose") //No toggling here { s.onclick = function () { SideBar.iframe.style.visibility = "hidden"; document.getElementById("wrapper").style.margin = null; Settings.display.sidebar = false; Settings.saveSettings(); } } else if (s.name) { switch (s.name) { case "filterSwitch": SideBar.switchSet(s, false); break; case "Sort": case "Unsort": break; case "tagsInclude": case "tagsExclude": //Filter search text var txt = iDoc.getElementById(s.name); txt.value = Settings.filter[s.name]; txt.oninput = SideBar.onInputTagFilter; break; case "nextPage": case "metadata": SideBar.switchSet(s, Settings.fetch[s.name]); break; default: SideBar.switchSet(s, Settings.display[s.name]); break; } s.onclick = SideBar.onSwitchPressed; } } //-------------------------------------------// //Checking filters var filters = iDoc.getElementsByName("filterType"); for (var i = 0; i < filters.length; i++) { var bitValue = Math.pow(2, i + 1); filters[i].checked = (bitValue & Settings.filter.flag) ? true : false; filters[i].onclick = SideBar.onToggledFilter; } //-------------------------------------------// //Loading filter values var filterSets = iDoc.getElementsByName("filterSet"); for (var i = 0; i < filterSets.length; i++) { filterSets[i].onclick = SideBar.onRadioClick; } iDoc.getElementsByName("filterSet")[Settings.filter.set].checked = true; SideBar.loadFilterSet(); var els = sidebar.getElementsByClassName("filterValue"); for (var i = 0; i < els.length; i++) { els[i].oninput = SideBar.onInputFilterValue; els[i].onfocus = function (e) { e.target.select(); }; } //-------------------------------------------// //Sort type var sortTypes = iDoc.getElementsByName("sortType"); for (var i = 0; i < sortTypes.length; i++) sortTypes[i].onclick = SideBar.onRadioClick; SideBar.iDoc.getElementsByName("sortType")[Settings.sortType].checked = true; //-------------------------------------------// //Load IQDB options and set them var IQDBTypes = ["All", "anime-pictures", "danbooru", "e-shuushuu", "haruhidoujins", "gelbooru", "konachan", "mangadrawing", "sankaku", "theanimegallery", "yandere", "zerochan"]; var sel = iDoc.getElementById("IQDBOptions"); sel.innerHTML = ""; for (i = 0; i < IQDBTypes.length; i++) { var opt = iDoc.createElement("option"); opt.textContent = IQDBTypes[i]; sel.add(opt); if (IQDBTypes[i] == Settings.IQDBType) opt.selected = true; } if (Settings.IQDBType == "www") sel.selectedIndex = 0; sel.onchange = function (e) { var sel = e.target; if (sel.selectedIndex == 0) Settings.IQDBType = "www"; else Settings.IQDBType = sel.options[sel.selectedIndex].textContent; if (Settings.display.IQDBLink) PaginatorHQ.updateAllIQDBLinks(); Settings.saveSettings(); }; //-------------------------------------------// //Preview height settings sel = iDoc.getElementById("previewHeight"); sel.innerHTML = ""; for (i = 0; i <= 10; i++) { var size = i * 50 + 200; var opt = iDoc.createElement("option"); opt.textContent = size + "px"; opt.value = size; sel.add(opt); if (size == Settings.preview.height) opt.selected = true; } sel.onchange = function (e) { var sel = e.target; Settings.preview.height = sel.options[sel.selectedIndex].value; Settings.saveSettings(); }; //-------------------------------------------// //Preview length settings sel = iDoc.getElementById("previewLength"); sel.innerHTML = ""; for (i = 1; i < 9; i++) { var period = (i < 7) ? (i * 500) : 3000 + ((i - 6) * 1000); var opt = iDoc.createElement("option"); opt.textContent = period + "ms"; opt.value = period; sel.add(opt); if (period == Settings.preview.timeLength) opt.selected = true; } sel.onchange = function (e) { var sel = e.target; Settings.preview.timeLength = sel.options[sel.selectedIndex].value; Settings.saveSettings(); }; iDoc.body.setAttribute("style", "width: auto; height: auto;"); iframe.setAttribute("style", "z-index: 1000; position:fixed; top: 0px; left:0px;"); SideBar.adjustFrameSize(); SideBar.iframe.style.visibility = (Settings.display.sidebar) ? null : "hidden"; document.getElementById("wrapper").style.margin = (Settings.display.sidebar) ? "10px 20px 0px auto" : null; window.onresize = SideBar.adjustFrameSize; if (Settings.fetch.metadata) IllustrationLinker.switchOn(); else IllustrationLinker.switchOff(); } document.body.appendChild(iframe); }, switchMain: function (s) { if (s.className) return s; if (s.firstElementChild && s.firstElementChild.className) return s.firstElementChild; var depth = 0; while (!s.className) { s = s.parentElement; if (s.className) return s; if (++depth == 2) return s; } }, switchCheck: function (s) { return (SideBar.switchMain(s).className.indexOf("On") >= 0); }, switchSet: function (s, enable) { var m = SideBar.switchMain(s); m.className = (enable) ? m.className.replace("Off", "On") : m.className.replace("On", "Off"); }, switchToggle: function (s) { var m = SideBar.switchMain(s); var enabled = !SideBar.switchCheck(m); SideBar.switchSet(m, enabled); return enabled; }, adjustFrameSize: function (e) { if (SideBar.sidebar.offsetHeight > window.innerHeight) { SideBar.iframe.style.height = (window.innerHeight) + "px"; SideBar.iframe.style.width = (SideBar.sidebar.offsetWidth + 20 + SideBar.scrollWidth) + "px"; } else { SideBar.iframe.style.height = (SideBar.sidebar.offsetHeight + 30) + "px"; SideBar.iframe.style.width = (SideBar.sidebar.offsetWidth + 20) + "px"; } }, /* ------------------------------------------------------ Handles all link elements that behave like a button ------------------------------------------------------*/ onSwitchPressed: function (e) { e.stopPropagation(); //e.preventDefault(); //Stops propagating to parent var s = e.target; while (s.tagName != "A") s = s.parentElement; var enabled; if (s.name != "Sort" && s.name != "Unsort") enabled = SideBar.switchToggle(s); if (Settings.display[s.name] != undefined) Settings.display[s.name] = enabled; if (Settings.fetch[s.name] != undefined) Settings.fetch[s.name] = enabled; switch (s.name) { case "Sort": case "Unsort": case "tagsInclude": case "tagsExclude": case "filterSwitch": if (PaginatorHQ.status != 0) { alert("Sort/Filter is already in progress."); } else if (s.name == "filterSwitch") { if (enabled) Settings.filterSwitchFlag = (Settings.filterSwitchFlag | 2); else Settings.filterSwitchFlag -= (Settings.filterSwitchFlag & 2); PaginatorHQ.filterResult(); } else if (s.name == "tagsInclude" || s.name == "tagsExclude") { var btn = e.target; if (btn.className == "bgiL" && btn.style.borderColor == "rgb(0, 0, 0)") { btn.className = "bgiG"; enabled = false; } else { btn.style.borderColor = "rgb(0, 0, 0)"; btn.className = "bgiL"; enabled = true; } if (s.name == "tagsInclude") { if (enabled) Settings.filterSwitchFlag = (Settings.filterSwitchFlag | 4); else Settings.filterSwitchFlag -= (Settings.filterSwitchFlag & 4); } if (s.name == "tagsExclude") { if (enabled) Settings.filterSwitchFlag = (Settings.filterSwitchFlag | 8); else Settings.filterSwitchFlag -= (Settings.filterSwitchFlag & 8); } PaginatorHQ.filterResult(); } else { PaginatorHQ.sortingMethod = (PaginatorHQ.sortingMethod != 0) ? 0 : 1; PaginatorHQ.sortByScore((s.name == "Sort")); } break; case "displayOptions": SideBar.iDoc.getElementById("DisplayOptions").style.display = (enabled) ? null : "none"; break; case "filterOptions": SideBar.iDoc.getElementById("FilterOptions").style.display = (enabled) ? null : "none"; break; case "IQDBLink": PaginatorHQ.updateAllIQDBLinks(); break; case "metadata": if (enabled) IllustrationLinker.switchOn(); else IllustrationLinker.switchOff(); break; case "Paging": break; default: PaginatorHQ.updateVisibilityOfAllElements(); break; } Settings.saveSettings(); SideBar.adjustFrameSize(); return false; }, onToggledFilter: function (e) { Settings.filter.flag = 0; var filters = SideBar.iDoc.getElementsByName("filterType"); for (var i = 0; i < filters.length; i++) { var bitValue = Math.pow(2, i + 1); Settings.filter.flag += (filters[i].checked) ? bitValue : 0; } Settings.saveSettings(); PaginatorHQ.filterResult(); }, onAgeRatingFilter: function (e) { var ars = SideBar.iDoc.getElementsByName("ageRating"); for (var i = 0; i < ars.length; i++) { var val = Math.pow(2, i + 4); Settings.filterSwitchFlag -= (Settings.filterSwitchFlag & val); //Remove the bit switch if (ars[i].checked) Settings.filterSwitchFlag += val; } PaginatorHQ.filterResult(); }, onRadioClick: function (e) { switch (e.target.name) { case "filterSet": Settings.filter.set = e.target.value; SideBar.loadFilterSet(); Settings.saveSettings(); if ((Settings.filterSwitchFlag & 2) == 2) PaginatorHQ.filterResult(); break; case "sortType": Settings.sortType = e.target.value; Settings.saveSettings(); break; } }, onInputFilterValue: function (e) { e.target.value = parseInt(e.target.value); if (e.target.value.length > 6) e.target.value = e.target.value.substring(0, 6); var values = SideBar.sidebar.getElementsByClassName("filterValue"); for (var i = 0; i < values.length; i++) Settings.filters[Settings.filter.set][i] = parseInt(values[i].value); Settings.saveSettings(); if ((Settings.filterSwitchFlag & 2) == 2 && Settings.filter.flag == e.target.nextElementSibling.value) PaginatorHQ.filterResult(); }, onInputTagFilter: function (e) { var name = e.target.id; var btn = SideBar.iDoc.getElementsByName(name)[0].firstElementChild; if (btn.className == "bgiL" && btn.style.borderColor == "rgb(0, 0, 0)") { btn.style.borderColor = "rgb(255, 0, 0)"; } Settings.filter[name] = e.target.value; Settings.saveSettings(); }, loadFilterSet: function () { var values = Settings.filters[Settings.filter.set]; var filters = SideBar.sidebar.getElementsByClassName("filterValue"); for (var i = 0; i < filters.length; i++) filters[i].value = values[i]; } } /* =================================================================================================================================== ===================================================================================================================================*/ var Settings = { filterSwitchFlag: 0, sortType: 0, versionCheck: function (current) { var saved = GM_getValue("Version", "0.0").toString(); var v1 = saved.split("."); var v2 = current.split("."); if (saved != current) alert("(Some) Pixiv++ settings will be reset due to major changes in the update."); if (v1[0] != v2[0]) { var names = GM_listValues(); for (var i = 0; name = names[i], i < names.length; i++) { var skipNames = ["Filters", "DisplayHidden"]; var found = false; for (var j = 0; j < skipNames.length; j++) found = found || (name.indexOf(skipNames[j]) == 0); if (!found) GM_deleteValue(name); } } if (v1[1] != v2[1]) { var names = GM_listValues(); for (var i = 0; name = names[i], i < names.length; i++) { var deleteNames = ["Filters"]; if ("Filters" == name) GM_deleteValue(name); } } GM_setValue("Version", current); }, loadSettings: function () { Settings.versionCheck("126.125"); //GM_setValue("DisplayHidden", 2); Settings.displayHidden = GM_getValue("DisplayHidden", 2); Settings.requestMethod = GM_getValue("RequestMethod", 0); var vals = GM_getValue(Settings.valueName("Generic"), "4094|6|350,1500|500|0|0|0").split("|"); // ----------- Sidebar Display Settings Options (PAGETYPE > 0) Settings.display = makeStruct("artistName illustTitle countList illustLink mangaLinks IQDBLink autoPreview sidebar displayOptions filterOptions"); Settings.setObjectPropertiesByFlag(Settings.display, vals[0]); // ----------- Sidebar Fetch Settings (PAGETYPE > 0) Settings.fetch = makeStruct("nextPage metadata"); Settings.setObjectPropertiesByFlag(Settings.fetch, vals[1]); // ----------- Preview Settings Settings.preview = makeStruct("timeLength height"); Settings.setObjectPropertiesValues(Settings.preview, vals[2].split(",")); // ----------- Other Values Settings.pagingOffset = parseInt(vals[3]); Settings.sortType = parseInt(vals[4]); Settings.filter = new Object(); Settings.filter.flag = parseInt(vals[5]); Settings.filter.set = parseInt(vals[6]); // ----------- Fixed Settings vals = GM_getValue("General", "danbooru|R-18|R-18").split("|"); Settings.IQDBType = vals[0]; Settings.filter.tagsInclude = vals[1]; Settings.filter.tagsExclude = vals[2]; // ----------- Filters vals = GM_getValue("Filters", "5, 10, 5, 10|10, 50, 10, 20|30, 400, 30, 100|50, 1000, 50, 200|80, 2000, 60, 400").split("|"); Settings.filters = new Array(); for (var i = 0; i < 5; i++) Settings.filters[i] = vals[i].split(","); if (Settings.pagingOffset > 3000) Settings.pagingOffset = 3000; }, setObjectPropertiesByFlag: function (obj, flag) { var i = 0; for (var key in obj) { if (obj.hasOwnProperty(key)) { i++; obj[key] = (flag & Math.pow(2, i)) > 0; } } return obj; }, getObjectPropertiesFlag: function (obj) { var i = 0; var flag = 0; for (var key in obj) { if (obj.hasOwnProperty(key)) { i++; if (obj[key]) flag += Math.pow(2, i); } } return flag; }, setObjectPropertiesValues: function (obj, arr) { var i = 0; for (var key in obj) { if (obj.hasOwnProperty(key)) { obj[key] = arr[i]; i++; } } return obj; }, getObjectPropertiesValues: function (obj) { var arr = new Array(); for (var key in obj) if (obj.hasOwnProperty(key)) arr.push(obj[key]); return arr; }, saveSettings: function () { GM_setValue(Settings.valueName("Generic"), Settings.getObjectPropertiesFlag(Settings.display) + "|" + Settings.getObjectPropertiesFlag(Settings.fetch) + "|" + Settings.getObjectPropertiesValues(Settings.preview) + "|" + Settings.pagingOffset + "|" + Settings.sortType + "|" + Settings.filter.flag + "|" + Settings.filter.set); GM_setValue("General", Settings.IQDBType + "|" + Settings.filter.tagsInclude + "|" + Settings.filter.tagsExclude); GM_setValue("Filters", (function () { var val = ""; for (var i = 0; i < 5; i++) val += Settings.filters[i].toString() + ((i < 4) ? "|" : ""); return val; })()); }, valueName: function (name, saveset) { if (!saveset) saveset = PAGETYPE; return "[" + saveset + "] " + name; } } function GetAbsolutePosition(el) { var x = 0; var y = 0; while (el && !isNaN(el.offsetLeft) && !isNaN(el.offsetTop)) { x += el.offsetLeft; y += el.offsetTop; el = el.offsetParent; } return { top: y, left: x }; } function FilterTypeToString(type) { var names = ["Bookmarks", "Views", "Ratings", "Total"]; return names[type]; } function RemoveMessage(msg) { if (msg) { var msgBox = msg.parentElement; msgBox.removeChild(msg); if (msgBox.children.length == 0) msgBox.style.visibility = "hidden"; } } //function AdjustDisplayMessage() //{ // var msgBox = document.getElementById("pppMsgBox"); // if (!msgBox) return; // if (SideBar.iframe) msgBox.style.marginLeft = (SideBar.pinned || (SideBar.iframe.style.visibility != "hidden")) ? SideBar.iframe.offsetWidth + "px" : "0px"; //} function DisplayMessage(msgTxt, timeout) { //text-align: center; display:inline-block; width: 100px; background-color: #D3D3D3; border: 1; var msgBox = document.getElementById("pppMsgBox"); if (!msgBox) { msgBox = document.createElement("span"); msgBox.id = "pppMsgBox"; document.body.appendChild(msgBox); } //AdjustDisplayMessage(); msgBox.style.visibility = null; var msg = document.createElement("div"); msg.textContent = msgTxt; msgBox.appendChild(msg); if (!isNaN(timeout)) setTimeout(RemoveMessage, timeout, msg); return msg } function makeStruct(keys, obj) { if (!obj) obj = {}; var names = keys.split(" ").sort(); for (var i = 0; i < names.length; i++) { obj[names[i]] = obj[names[i]]; } return obj; } /* ================================================================================================= [VYCS] VARIABLE YOU CAN SET =================================================================================================*/ //GM_setValue("RequestMethod", 0); // [1] //GM_setValue("RequestBookmarkCount", 2); //[4] /* [1] Default: 0 Default gets all meta-data using XmlHttpRequest otherwise uses the flags below: 2: Get next pages using GM_XmlHttpRequest 4: Get illustration data using GM_XmlHttpRequest on every page apart from illustration page 8: Get data using GM_XmlHttpRequest on illustration page If you want to use GM_XmlHttpRequest on everything then the value set the value to 14 (2+4+8) [2] Does an extra call to get bookmark if it is missing. Default value is 0. 0: Do not make any extra http request to get bookmark 1: Get bookmark count on every page except illustration page 2: Get bookmark for every page */ /* ================================================================================================= Do not touch =================================================================================================*/ //Removes pop dialog that appears when there isn't a cookie if (window.self === window.top) (function () { PaginatorHQ.addStyle("Adjust-Pixiv", "li.before a img, li.after a img, img._thumbnail {color: white; border: 3px solid; padding: 1px; background-color: white;}" + "li.before a:visited img, li.after a:visited img, a:visited img._thumbnail{color: cyan !important;}" ); console.info("Pixiv Main"); var counter = 0; var id = setInterval(function () { var pop = document.getElementById("register-introduction-modal"); if (pop) pop.getElementsByClassName("close")[0].click(); pop = document.getElementsByClassName("signup-notice"); for (var i = 0; i < pop.length; i++) pop[i].style.display = "none"; counter++; if (counter = 10) clearInterval(id); }, 100); Settings.loadSettings(); if (PAGETYPE >= 0) { console.info("Pixiv++ Initalising"); var els = document.querySelectorAll("a[href^='/jump']"); for (var i = 0; i < els.length; i++) { els[i].href = decodeURIComponent(els[i].href.replace(/.+\/jump\.php\?/, "")); console.log("Direct URL: " + els[i].href); } SideBar.initalise(); PaginatorHQ.intialise(); } }()); /* <script> pixiv.context.illustId = '44305721'; pixiv.context.illustTitle = 'フランスパンこいしちゃんgif'; pixiv.context.userId = '42949'; pixiv.context.userName = 'ゆぬき うた'; pixiv.context.hasQuestionnaire = false; pixiv.context.embedId = '44305721_0286523bc768bf54bbc69e3163d75256'; pixiv.context.explicit = false; pixiv.context.illustSize = [514, 487]; pixiv.context.ugokuIllustData = {"src":"http:\/\/i2.pixiv.net\/img-zip-ugoira\/img\/2014\/06\/25\/21\/24\/51\/44305721_ugoira600x600.zip","mime_type":"image\/jpeg","frames":[{"file":"000000.jpg","delay":100},{"file":"000001.jpg","delay":100},{"file":"000002.jpg","delay":100},{"file":"000003.jpg","delay":100},{"file":"000004.jpg","delay":100},{"file":"000005.jpg","delay":100},{"file":"000006.jpg","delay":100},{"file":"000007.jpg","delay":100},{"file":"000008.jpg","delay":100},{"file":"000009.jpg","delay":100},{"file":"000010.jpg","delay":100},{"file":"000011.jpg","delay":100}]}; pixiv.context.ugokuIllustFullscreenData = {"src":"http:\/\/i2.pixiv.net\/img-zip-ugoira\/img\/2014\/06\/25\/21\/24\/51\/44305721_ugoira1920x1080.zip","mime_type":"image\/jpeg","frames":[{"file":"000000.jpg","delay":100},{"file":"000001.jpg","delay":100},{"file":"000002.jpg","delay":100},{"file":"000003.jpg","delay":100},{"file":"000004.jpg","delay":100},{"file":"000005.jpg","delay":100},{"file":"000006.jpg","delay":100},{"file":"000007.jpg","delay":100},{"file":"000008.jpg","delay":100},{"file":"000009.jpg","delay":100},{"file":"000010.jpg","delay":100},{"file":"000011.jpg","delay":100}]}; </script> http://pixiv.me/<userID> */