// ==UserScript==
// @name [TS] Pixiv++
// @namespace TimidScript
// @version 3.1.63
// @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". 自動ページング|ポケベル|ロード次ページ|フィルター|並べ替え|注文|ダイレクトリンク
// @icon https://i.imgur.com/ZNBlNzI.png
// @author TimidScript
// @homepageURL https://openuserjs.org/users/TimidScript
// @copyright © 2014 TimidScript, All Rights Reserved.
// @license Creative Commons BY-NC-SA + Please notify me if distributing
// @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_-_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
// ==/UserScript==
/* Information
********************************************************************************************
Copyright © 2014 TimidScript, All Rights Reserved.
Script's Homepage: Check homepages below
TimidScript's Homepage: https://openuserjs.org/users/TimidScript
https://greasyfork.org/users/1455-timidscript
https://monkeyguts.com/author.php?un=timidscript
http://userscripts.org/users/TimidScript
http://userscripts-mirror.org/users/100610/scripts
------------------------------------
Version History
------------------------------------
3.1.64 (2014-10-31)
- Removes links when thumbnail is hidden
3.1.63 (2014-10-04)
- Bug fixes to handle more changes in pixiv layout
- Using new embedded images instead of thumbnail images for new layout as it gets full images rather than cropped version
- Manga page now has direct links to large size only
- Removed some of the redundant code.
- Ugoria has preview image now
- Slight improvement in code by making adjustAllHotboxPositions asynchronous
- CSS added for visited illustration page
- Bug fix to adjustHotboxPosition due to layout changes
3.1.62 (2014-09-27)
- Fixes direct links that got broken due to latest changes in Pixiv++. Uses API all the time now.
It is now impossible to get the necessary meta data from just the illustration page, therefore
LinkerMethod(1) is no longer supported.
- Current obsolete code still remains and needs to be removed.
- Changes in Pixic broke detection of Ugoria illustration. Two methods now implemented, checking 480URL
or tags. Choose tag method.
3.1.61 (2014-08-29)
- Bug Fix: Cropped the d in GM_registerMenuCommand in header
3.1.60 (2014-08-29)
- Added GM_update
3.1.59 (2014-08-19)
- Cleaned up header for OUJS
- Removed old history
3.1.58 (2014-08-17)
- Full support for Ugoira links
- Removed header data @versioninfo as there is no personalised updater
3.1.57 (2014-07-09)
- Bug Fix for Pixiv layout changes
3.1.56 (2014-07-03)
- Bug Fix: Links on Illustration page not working
- Bug Fix in SetMethod2 document.evaluate variable fix
3.1.55 (2014-07-03)
- Partial support for Ugoira animated files. More information found:
http://www.pixiv.net/info.php?id=2476&lang=en
http://danbooru.donmai.us/forum_topics/10712
https://github.com/r888888888/danbooru/issues/2212
3.1.54 (2014-05-28)
- Fix IQDB links captured as images
3.1.53 (2014-05-19)
- Small changes
*************************************************************************************************/
//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 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\/(response\.php\?type=illust\&id)/i)) return 2; //Response
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)
]
};
IllustrationData =
{
collection: new Array(),
cached: false,
reset: function ()
{
collection = new Array();
},
createData: function (link)
{
var data = new METADATA();
data.illustID = this.getIllustartionID(link);
IllustrationData.collection.push(data);
return data;
},
getIllustrationData: function (id)
{
for (var i = 0; data = IllustrationData.collection[i], i < IllustrationData.collection.length; i++)
{
if (IllustrationData.collection[i].illustID == id) return IllustrationData.collection[i];
}
return null;
},
getMetaScore: function (id)
{
var data = IllustrationData.getIllustrationData(id);
return [data.bookmarkCount, data.viewCount, data.ratings, data.totalRatings];
},
getIllustrationLinkData: function (link)
{
return IllustrationData.getIllustrationData(IllustrationData.getIllustartionID(link));
},
getIllustartionID: function (link)
{
return link.href.replace(/.+illust_id=(\d+).*$/, "$1");
},
/*
------------------------------------------------------------------------------------------------
Does not automatically reset cache. It checks if it can be used, if not it is reset and
cached flag is set accordingly
------------------------------------------------------------------------------------------------*/
resetCache: function ()
{
//Remove items with incomplete data
for (var i = 0; data = IllustrationData.collection[i], i < IllustrationData.collection.length; i++)
{
if (data.illustURL == null)
{
IllustrationData.collection.splice(i, 1);
i--;
continue;
}
}
IllustrationData.cached = (Settings.EnableCache && IllustrationData.collection.length > 0);
}
}
/*
===================================================================================================================================
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 =
{
//0 shows SML images. 1 shows BIG images if available otherwise SML, 2 Shows both types
mangaImageLinksShown: 1,
//Manga in different languages used to extract manga page count.
mangaStrings: new Array("Manga", "漫画", "만화", "Манга", "มังงะ", "漫畫"),
//Contains all thumbnail links that need to be parsed
thumbnailLinks: 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
requestMethod: 2, // GM_getValue("LinkerMethod", 2), //1: Uses illustration page otherwise it uses APIMethod
TIMESTART: 0,
TIMEEND: 0,
/*
-------------------------------------------------------------------------------------------
Parses through all the image links and add full image link
-------------------------------------------------------------------------------------------*/
getContainerLinks: function (containers, pageNumber)
{
if (!containers) return;
for (i = 0; container = containers[i], i < containers.length; i++)
{
links = container.getElementsByTagName("a");
for (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)
{
var thumbnail = link.parentNode;
thumbnail.setAttribute("name", "pppThumb");
if (pageNumber > 0)
{
thumbnail.className += " pppThumb" + (pageNumber % 3);
thumbnail.setAttribute("page", pageNumber);
thumbnail.setAttribute("position", ++IllustrationLinker.thumbcounter);
}
var data;
if (IllustrationData.cached && (data = IllustrationData.getIllustrationLinkData(link)))
{
IllustrationLinker.createLinksBox(link);
thumbnail.setAttribute("illustration-id", data.illustID);
continue;
}
data = IllustrationData.createData(link);
this.thumbnailLinks.push(link);
thumbnail.setAttribute("illustration-id", data.illustID);
}
}
}
PaginatorHQ.displayResultInfo();
if (IsIllustrationPage) ////Illustration Page that excludes Novels
{
var link = IllustrationLinker.thumbnailLinks.shift();
//Use API to get missing information such as bookmark count
IllustrationLinker.setMetadataM1(link, document);
IllustrationLinker.getIllustrationAPIData(link);
//IllustrationLinker.createLinksBox(link);
}
else if (this.enabled) this.runIntevalThumbnailParser();
},
/*
-------------------------------------------------------------------------------------------
Get links again. Called when using ARS
-------------------------------------------------------------------------------------------*/
resetSettings: function ()
{
clearInterval(IllustrationLinker.intervalID);
IllustrationLinker.intervalID = null;
IllustrationLinker.thumbnailLinks = 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.thumbnailLinks.length + "] illustrations...");
if (IllustrationLinker.intervalID == null && IllustrationLinker.enabled)
IllustrationLinker.intervalID = setInterval(
function ()
{
//shortPause used when removing session cookie for safe page search
if (IllustrationLinker.shortPause || IllustrationLinker.simultaneousCalls >= IllustrationLinker.simultaneousCallsMAX) return;
if (IllustrationLinker.thumbnailLinks.length == 0 && IllustrationLinker.simultaneousCalls == 0)
{
IllustrationLinker.TIMEEND = new Date();
//console.warn("METHOD" + IllustrationLinker.requestMethod + ": " + (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.thumbnailLinks.length % 5 == 0)
{
IllustrationLinker.msgHandle.textContent = "Getting metadata for [" + (IllustrationLinker.simultaneousCalls + IllustrationLinker.thumbnailLinks.length) + "] illustrations...";
PaginatorHQ.displayResultInfo();
PreviewHQ.adjustAllHotboxPositions();
}
//Last check
if (IllustrationLinker.enabled && IllustrationLinker.thumbnailLinks.length > 0)
{
IllustrationLinker.simultaneousCalls++;
link = IllustrationLinker.thumbnailLinks.shift();
if (IllustrationLinker.requestMethod == 1) IllustrationLinker.getIllustrationPage(link);
else IllustrationLinker.getIllustrationAPIData(link);
}
}
, 100);
},
/*
-------------------------------------------------------------------------------------------
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();
},
/*
-------------------------------------------------------------------------------------------
Parses through illustration page to get the page count
-------------------------------------------------------------------------------------------*/
getPageCount: function (doc)
{
var metaData = doc.evaluate("//ul[@class='meta']", doc, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;
if (metaData)
{
lis = metaData.getElementsByTagName("li");
for (var i = 0; i < lis.length; i++)
{
var regex = new RegExp(/^.+ (\d+)P$/gi);
var pageCount = lis[i].textContent.replace(regex, "$1");
if (!isNaN(pageCount) && pageCount > 0) return pageCount;
}
}
return 0;
},
/*
-------------------------------------------------------------------------------------------
Method 1 for extracting metadata. It takes it from the illustration page. Method 2
uses Phone API.
-------------------------------------------------------------------------------------------*/
setMetadataM1: function (link, doc)
{
var thumbnail = link.parentNode;
var metadata = IllustrationData.getIllustrationLinkData(link);
//var metadata = new METADATA();
var scoreBar = doc.evaluate("//section[@class='score']/dl", doc, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;
if (scoreBar)
{
var m = scoreBar.textContent.match(/[^0-9]+(\d+)[^0-9]+(\d+)[^0-9]+(\d+)$/);
var response = doc.evaluate("//div[@class='worksImageresponse']/p[@class='worksAlso']", doc, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;
if (response)
{
response = parseInt(response.textContent.replace(/.*\((\d+)\).*/, "$1"));
}
else response = IllustrationLinker.getResponseCount(thumbnail);
if (isNaN(response)) response = 0;
metadata.responseCount = response;
}
if (IsIllustrationPage) console.log("HTML", metadata);
},
/*
-------------------------------------------------------------------------------------------
Method 2 for extracting metadata. Method 2 uses Phone API while method 1 uses Illustration
Page
-------------------------------------------------------------------------------------------*/
setMetadataM2: function (link, APIresponse)
{
var rawlist = APIresponse.split(",");
var datalist = new APIDataFull();
var dataNames = APIMetaNames.split(" ");
for (var i = 0, n = 0; rawdata = rawlist[i], i < rawlist.length; i++)
{
var j = 0;
while ((rawdata.split('"').length - 1) % 2 == 1) //Quote number should always be even. If odd then you need to combine
{
j++;
rawdata += "," + rawlist[i + j]; //We add the comma that we removed as it was part of the value
}
i += j; //We appended items
rawdata = rawdata.replace(/^"|"$/g, ""); //We remove any starting or ending quotes
datalist[dataNames[n]] = rawdata;
n++;
}
datalist.illust480URL = datalist.illust480URL.replace(/\?\d+$/, "");
//console.log(datalist);
var metadata = IllustrationData.getIllustrationLinkData(link);
for (var i = 0; i < dataNames.length; i++)
{
//Null if item exists otherwise it would be undefined.
if (metadata[dataNames[i]] === null)
{
metadata[dataNames[i]] = datalist[dataNames[i]];
if (metadata[dataNames[i]].length > 0 && !isNaN(metadata[dataNames[i]])) metadata[dataNames[i]] = parseInt(metadata[dataNames[i]]);
}
}
//metadata = new METADATA();
//If empty or pageCount equal zero then its not a manga
if (isNaN(metadata.pageCount)) metadata.pageCount = 0;
if (metadata.pageCount > 0) metadata.illustType = 2; //Manga
//else if (datalist.illust480URL.match(/img-master\/img/)) metadata.illustType = 3; //Ugoira (Animated)
else if (metadata.illust480URL.match(/_master1200.jpg$/)) metadata.illustType = 3; //Ugoira (Animated)
else metadata.illustType = 1; //Single illustration
/* Possible image base URLs
[-]http://i2.pixiv.net/img14/profile/whatsoy/mobile/7330980_80.jpg - Profile image
[-]http://i2.pixiv.net/img14/img/whatsoy/mobile/16791530_128x128_p0.jpg - Thumbnail use in Manga View
[-]http://i2.pixiv.net/img14/img/whatsoy/16791530_s.jpg - OLD Thumbnail URL.
[*]http://i2.pixiv.net/img14/img/whatsoy/16791530_m.jpg - Illustration Page Size (Medium)
[X]http://i2.pixiv.net/img14/img/whatsoy/16791530.jpg - Full Size Illustration
[-]http://i2.pixiv.net/img14/img/whatsoy/16791530_p0.jpg - [Manga Page] Full Size For Old Images otherwise Large
[-]http://i2.pixiv.net/img14/img/whatsoy/16791530_big_p0.jpg - [Manga Page] Full Size For New Images
[-]http://i1.pixiv.net/img101/img/nahenaheko/mobile/34649107_240mw.jpg [Embed Image?] Taken from Illustration page
[*]http://i2.pixiv.net/img14/img/whatsoy/mobile/16791530_480mw.jpg - [JPG][Mobile] Illustration Page Size (Medium) Very low quality & large size
[*]http://i2.pixiv.net/img14/img/whatsoy/mobile/16791530_128x128.jpg - [JPG][Mobile] Thumbnail
[*]http://i2.pixiv.net/img14/img/whatsoy/mobile/16791530_128x128_p0.jpg - [Mobile] Paged Thumbnail
[*] Are imageURLBase URLs that we are expecting. The rest can be generated
[X] The imageURLBase URL we want
NB: Image URL may have a suffix similar to this "?342342131" */
/* NEW URL 2014-10 ---------------------------------------------------
http://i2.pixiv.net/img02/profile/vantmic/mobile/2727986_80.jpg //Profile image
http://www.pixiv.net/member_illust.php?mode=medium&illust_id=46250475
http://i2.pixiv.net/c/128x128/img-master/img/2014/09/29/12/49/18/46250475_128x128.jpg //API 128
http://i2.pixiv.net/c/480x960/img-master/img/2014/09/29/12/49/18/46250475_480mw.jpg //API 480
http://i2.pixiv.net/c/600x600/img-master/img/2014/09/29/12/49/18/46250475_p0_master1200.jpg //Illustration page
http://i2.pixiv.net/img-original/img/2014/09/29/12/49/18/46250475_p0.jpg //Full Image
http://i2.pixiv.net/c/600x600/img-master/img/2014/09/29/12/49/18/46250475_p0_master1200.jpg //Embed
http://i2.pixiv.net/c/240x480/img-master/img/2014/09/29/12/49/18/46250475_p0_master1200.jpg //Embed
http://i2.pixiv.net/c/150x150/img-master/img/2014/09/29/12/49/18/46250475_p0_master1200.jpg //Embed
Manga Paged
http://www.pixiv.net/member_illust.php?mode=medium&illust_id=46283246
http://i1.pixiv.net/c/128x128/img-master/img/2014/10/01/06/19/21/46283246_128x128.jpg //API Thumbnail
http://i1.pixiv.net/c/480x960/img-master/img/2014/10/01/06/19/21/46283246_480mw.jpg //API 480
http://i1.pixiv.net/c/128x128/img-master/img/2014/10/01/06/19/21/46283246_p0_square1200.jpg //Thumbnails
http://i1.pixiv.net/c/600x600/img-master/img/2014/10/01/06/19/21/46283246_p0_master1200.jpg //Illustration Page
http://i1.pixiv.net/c/1200x1200/img-master/img/2014/10/01/06/19/21/46283246_p0_master1200.jpg //Not FULL image.
http://i1.pixiv.net/img-original/img/2014/10/01/06/19/21/46283246_p0.png //Full Image
http://i1.pixiv.net/c/600x600/img-master/img/2014/10/01/06/19/21/46283246_p0_master1200.jpg //Embed
http://i1.pixiv.net/c/240x480/img-master/img/2014/10/01/06/19/21/46283246_p0_master1200.jpg //Embed
http://i1.pixiv.net/c/150x150/img-master/img/2014/10/01/06/19/21/46283246_p0_master1200.jpg //Embed
http://i1.pixiv.net/c/480x960/img-master/img/2014/10/01/06/19/21/46283246_big_p2.png //Unofficial but works
Ugoira Animations
http://www.pixiv.net/member_illust.php?mode=medium&illust_id=46283334
http://www.pixiv.net/member_illust.php?mode=ugoira_view&illust_id=46283334
http://i1.pixiv.net/c/128x128/img-master/img/2014/10/01/06/38/39/46283334_square1200.jpg //API 128
http://i1.pixiv.net/c/480x960/img-master/img/2014/10/01/06/38/39/46283334_master1200.jpg //API 480
http://i1.pixiv.net/img-zip-ugoira/img/2014/10/01/06/38/39/46283334_ugoira1920x1080.zip
*/
if (metadata.illustType == 3)
{
metadata.illustURL = datalist.illust480URL.replace(/c\/\d+x\d+\/img-master(.+)\/.+/, "img-zip-ugoira$1/" + datalist.illustID + "_ugoira1920x1080.zip");
metadata.illust600URL = metadata.illust480URL.replace("480x960", "600x600");
}
else if (datalist.illust480URL.indexOf("/img-master/") > 0) //2014-10 Pixiv URLs
{
//Default Manga thumbnails
//metadata.illust128URL = metadata.illust128URL.replace("_128x128", "_p0_square1200");
//Embedded images get full image this way
//Using datalist.time does not always work as if the image is updated the time changes but the original path remains
//metadata.illust150URL = metadata.illust128URL.replace(/\.pixiv\.net.+/, ".pixiv.net/c/150x150/img-master/img/") + datalist.time.replace(/-|:|\s/g, "/") + "/" + datalist.illustID + "_p0_master1200.jpg";
metadata.illust150URL = metadata.illust128URL.replace(/128x128(.+)_128x128/, "150x150$1_p0_master1200");
metadata.illust240URL = metadata.illust150URL.replace("/150x150/", "/240x480/");
metadata.illust480URL = metadata.illust150URL.replace("/150x150/", "/480x960/");
metadata.illust600URL = metadata.illust150URL.replace("/150x150/", "/600x600/");
metadata.illustURL = metadata.illust600URL.replace(/c\/600x600\/img-master(.+)\/.+/, "img-original$1/" + datalist.illustID + "_p0." + datalist.illustExt);
}
else //Old format
{
metadata.illust128URL = datalist.illust480URL.replace(/(\d+)_480mw(\.\w+)/, "$1_128x128_p0$2");
metadata.illust240URL = datalist.illust480URL.replace(/(\d+)_480mw(\.\w+)/, "$1_240mw_p0$2");
metadata.illust150URL = metadata.illust240URL;
metadata.illust600URL = datalist.illust480URL.replace(/\/mobile(\/.+)480mw\..+/, "$1m." + datalist.illustExt);
metadata.illustURL = metadata.illust600URL.replace(/_m\.\w+$/, "") + ((metadata.illustType == 1) ? "." : "_big_p0.") + datalist.illustExt;
//Old manga format does not support _big
if (datalist.illustID < 11319936) metadata.illustURL = metadata.illustURL.replace("_big", "");
}
if (IsIllustrationPage) //Never gets called because we use Method1 there XD
{
var response = document.evaluate("//div[@class='worksImageresponse']/p[@class='worksAlso']", document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;
if (response)
{
response = parseInt(response.textContent.replace(/.*\((\d+)\).*/, "$1"));
}
} else metadata.responseCount = IllustrationLinker.getResponseCount(link.parentElement);
if (IsIllustrationPage) console.log("API\n", datalist, "\n\n", metadata);
},
/*
-------------------------------------------------------------------------------------------
Gets bookmark count from thumbnail if one exists
-------------------------------------------------------------------------------------------*/
getBookmarkCount: function (thumbnail)
{
var bm = thumbnail.getElementsByClassName("bookmark-count ui-tooltip")[0];
if (bm) return bm.textContent;
//If thumbnail contains count-list that means the count is zero
return (thumbnail.getElementsByClassName("count-list").length == 1) ? 0 : "?";
},
/*
-------------------------------------------------------------------------------------------
Gets response count from thumbnail if one exists
-------------------------------------------------------------------------------------------*/
getResponseCount: function (thumbnail)
{
var bm = thumbnail.getElementsByClassName("image-response-count ui-tooltip")[0];
if (bm) return parseInt(bm.textContent);
//If thumbnail contains count-list that means the count is zero
return (thumbnail.getElementsByClassName("count-list").length == 1) ? 0 : "?";
},
/*
-------------------------------------------------------------------------------------------
Creates main links that are used by Downloaders. These links can be hidden. It also
creates calls the creation of the hotbox. It uses the metadata stored in
IllustrationData
-------------------------------------------------------------------------------------------*/
createLinksBox: function (link)
{
var thumbnail = link.parentNode;
var metadata = IllustrationData.getIllustrationLinkData(link);
//var metadata = new METADATA();
PaginatorHQ.tidyThumbnail(link);
var linksBox = document.createElement("div");
linksBox.className = "pppLinksBox";
var directLinks = document.createElement("span");
thumbnail.insertBefore(linksBox, link.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 == 0) //Single Illustration
{
/*
http://superuser.com/questions/364632/how-to-override-a-javascript-value-with-greasemonkey
You'll want to use unsafeWindow, which, as the name infers, is not the most secure method
-- but if all you are doing is changing the value of a page based variable, there's no risk involved.
If you just use window.SESSION_TIMEOUT, you more than likely will not be accessing the
correct scope, and your SESSION_TIMEOUT variable would go unmodified, and a new one
created inside another context / scope. */
//console.warn(unsafeWindow.pixiv.context.ugokuIllustData);
//console.warn(unsafeWindow.pixiv.context.ugokuIllustFullscreenData);
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;
else directLinks.innerHTML = mangaLinks;
if (!Settings.display.mangaLinks) directLinks.style.display = "none";
}
if (!IsIllustrationPage)
{
var IQDBLink = document.createElement("a");
IQDBLink.textContent = "IQDB";
IQDBLink.className = "IQDBLink";
linksBox.appendChild(IQDBLink);
PaginatorHQ.updateIQDBLink(linksBox);
}
PreviewHQ.createHotBox(link);
if (PAGETYPE > 1) PaginatorHQ.filterThumbnail(thumbnail);
},
/*
-------------------------------------------------------------------------------------------
Gets illustration information using API
-------------------------------------------------------------------------------------------*/
getIllustrationAPIData: function (link) //Use API
{
//http://www.pixiv.net/member_illust.php?mode=medium&illust_id=34645665
//http://spapi.pixiv.net/iphone/illust.php?illust_id=34117192
//http://spapi.pixiv.net/iphone/illust.php?PHPSESSID=673982_98a848b3f187cdb7b45df373a1c7d7e2&illust_id=34117192
var id = link.href.replace(/.+id=(\d+).*$/, "$1");
var sessionID = Pager.GetSessionID();
var apiLink = "http://spapi.pixiv.net/iphone/illust.php?PHPSESSID=" + sessionID + "&illust_id=" + id;
var START = new Date();
GM_xmlhttpRequest({
url: apiLink,
method: "GET",
timeout: 15000,
headers: { "User-agent": navigator.userAgent, "Accept": "text/html", Referer: "http://www.pixiv.net" },
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);
IllustrationLinker.setMetadataM2(link, response.responseText);
//if (!IsIllustrationPage)
IllustrationLinker.createLinksBox(link);
}
else console.error("hmmmmm");
IllustrationLinker.simultaneousCalls--;
}
});
},
/*
-------------------------------------------------------------------------------------------
Gets illustration information using Illustration page
-------------------------------------------------------------------------------------------*/
getIllustrationPage: function (link)
{
var START = new Date();
GM_xmlhttpRequest({
url: link.href,
method: "GET",
timeout: 15000,
headers: { "User-agent": navigator.userAgent, "Accept": "text/html", Referer: "http://www.pixiv.net" },
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");
try
{
var src = doc.getElementsByClassName("works_display")[0].getElementsByTagName("a")[0].getElementsByTagName("img")[0].src;
}
catch (err)
{
DisplayMessage("Failed to get illustration: " + link.href, 5000);
console.error("Failed to get illustration source (" + link.href + ")");
}
if (src)
{
IllustrationLinker.setMetadataM1(link, doc);
IllustrationLinker.createLinksBox(link);
}
}
IllustrationLinker.simultaneousCalls--;
}
});
}
};
/*
===================================================================================================================================
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,
ageRating: 0,
//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);
},
/*
------------------------------------------------------------------------------
Rated version of "initialise". Initialise still needs to be called first.
- firstPageLoadEventHandler - gets called when the content of the first page
is acquired. Used to replace content of current document.
Takes (doc, array, ageRating). Array contains the result count for All, Safe, R18.
Used mainly to compare difference result count and make sure that
R18 + Safe = All.
- ageRating - 0:All|1:SafeMode|2:R-18
------------------------------------------------------------------------------*/
intialiseRated: function (firstPageLoadEventHandler, ageRating)
{
Pager.ageRatingChanged = true;
Pager.ageRating = ageRating;
//Waits for current page to finish loading
if (Pager.requestingPage)
{
setTimeout(Pager.intialiseRated, 250, firstPageLoadEventHandler, ageRating);
}
if (Pager.intervalID) clearInterval(Pager.intervalID); Pager.intervalID = null;
IllustrationData.resetCache();
Pager.getRatedPages(firstPageLoadEventHandler, document.URL.replace(/&r18=\d|(\?)r18=\d/g, "$1"), 0, new Array());
},
/*
-------------------------------------------------------------------------------------------
Gets called by initalisedRated. It does three different types of searches:
1) Clean url
2) URl with &r18=1
3) Clean url without cookie information for safe search
-------------------------------------------------------------------------------------------*/
getRatedPages: function (firstPageLoadEventHandler, url, stage, countList)
{
var sessionID = null;
var SafeMode = ((Pager.ageRating == 1 && stage == 2) || (Pager.ageRating == 2 && stage == 1));
if (SafeMode) var sessionID = Pager.removeSessionID();
document.getElementById
GM_xmlhttpRequest({
url: url,
method: "GET",
headers: { "User-agent": navigator.userAgent, "Accept": "text/html" },
timeout: 10000,
onload: function (response)
{
if (response.status == 200)
{
var doc = new DOMParser().parseFromString(response.responseText, "text/html");
var resultCount = PaginatorHQ.getResultCount(doc);
if (Pager.ageRating == 1 && stage == 2) countList.splice(1, 0, resultCount);
else countList.push(resultCount);
if (Pager.ageRating == 0 || stage == 2) //All Search
{
//console.log(countList);
Pager.getNextPageURL(doc.body);
firstPageLoadEventHandler(doc, countList, Pager.ageRating);
Pager.ageRatingChanged = false; //Should only be set when Pager.Initalise is called
}
else
{
stage++;
if ((Pager.ageRating == 1 && stage == 2) || (Pager.ageRating == 2 && stage == 1)) url = url.replace(/&r18=\d|(\?)r18=\d/g, "$1");
else url += "&r18=1";
Pager.getRatedPages(firstPageLoadEventHandler, url, stage, countList);
}
}
}
});
if (SafeMode) Pager.restoreSessionID(sessionID);
},
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;
},
/*
------------------------------------------------------------------------------
Gets session ID
------------------------------------------------------------------------------*/
GetSessionID: function ()
{
var m = document.cookie.match(/PHPSESSID=[^;]+/);
if (m[0]) return (m[0].split("=")[1]);
return "";
},
/*
------------------------------------------------------------------------------
Session Cookie altered for safe search
------------------------------------------------------------------------------*/
removeSessionID: function ()
{
IllustrationLinker.pause = true;
var sessionID = Pager.GetSessionID();
//console.info(document.cookie);
//Fake Session ID. If you leave it blank you cannot re-create cookie with old sessionID.
//console.warn("Pixiv session cookie being removed.");
if (sessionID) document.cookie = "PHPSESSID=000000000000000;path=/;domain=.pixiv.net;"
return sessionID;
},
/*
------------------------------------------------------------------------------
Session Cookie restored
------------------------------------------------------------------------------*/
restoreSessionID: function (sessionID)
{
if (sessionID) document.cookie = "PHPSESSID=" + sessionID + ";path=/;domain=.pixiv.net;"
//console.info("Pixiv session cookie restored.");
IllustrationLinker.pause = false;
},
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;
var safeMode = (Pager.ageRating == 1);
var sessionID = null;
if (safeMode) sessionID = Pager.removeSessionID();
clearInterval(Pager.intervalID);
var pageNumber = PaginatorHQ.getPageNumber(this.nextPageURL);
var msg = DisplayMessage("Requesting page [" + pageNumber + "] ...");
if (Settings.pageFetchingMethod == 1)
{
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);
dt = document.implementation.createDocumentType("html", "-//W3C//DTD HTML 4.01 Transitional//EN", "http://www.w3.org/TR/html4/loose.dtd"),
doc = document.implementation.createDocument("", "", dt),
documentElement = doc.createElement("html");
documentElement.innerHTML = response.responseText;
doc.appendChild(documentElement);
//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 if (Settings.pageFetchingMethod == 2)
{
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();
}
else
{
var iframe = document.getElementById("HiddenPagerFrame");
if (!iframe)
{
iframe = document.createElement("iframe");
iframe.id = "HiddenPagerFrame";
iframe.style.display = "none";
iframe.onload = function ()
{
Pager.requestingPage = false;
if (Pager.ageRatingChanged) return;
var doc = iframe.contentDocument || iframe.contentWindow.document;
Pager.NextPageReceived(doc, PaginatorHQ.getPageNumber(this.src));
}
document.body.appendChild(iframe);
}
iframe.src = this.nextPageURL;
setTimeout(function () { RemoveMessage(msg); }, 3000);
}
if (safeMode) Pager.restoreSessionID(sessionID);
},
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);
},
/*
-------------------------------------------------------------------------------------------
Callback from the Pager when new rated page result have returned.
It strips current result and insert new content.
-------------------------------------------------------------------------------------------*/
intialiseRated: function (doc, countList, ageRating)
{
/* Right now nothing is done with the countList and ageRating. But ideally
some sort of GUI notification would be nice. */
if (ageRating > 0 && countList[1] + countList[2] != countList[0])
console.warn("ALL|Safe||R18 ", countList);
else console.log("ALL|Safe||R18 ", countList);
DisplayMessage(countList.join("|"), 5000);
var evaluator = new XPathEvaluator(); //document.evaluate
var resultCount = evaluator.evaluate("//div[@class='column-label']/span[@class='count-badge']", document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;
/* We do this to avoid language issues. Remember we are getting safe
pages without valid session cookie (logged out) */
resultCount.textContent = resultCount.textContent.replace(/\d+/, countList[ageRating]);
var containers = document.getElementsByName("pageContainer");
var paginators = document.getElementsByClassName(" paginator");
while (paginators.length > 0) PaginatorHQ.pageTable.removeChild(paginators[0]);
while (containers.length > 0) PaginatorHQ.pageTable.removeChild(containers[0]);
var containerNEW = PaginatorHQ.getContainers(doc)[0];
var paginatorNEW = doc.getElementsByClassName("column-order-menu")[0];
if (containerNEW.getElementsByTagName("img").length > 0)
{
/*containerNEW: Need to mark it as pageContainer so to make sure it gets removed if
agerating is set again. Reason sometimes result only returns one
page which means it will not get marked.*/
containerNEW.setAttribute("name", "pageContainer");
if (paginatorNEW) PaginatorHQ.pageTable.appendChild(paginatorNEW);
PaginatorHQ.pageTable.appendChild(containerNEW);
}
IllustrationLinker.resetSettings();
PaginatorHQ.intialise();
},
/*
-------------------------------------------------------------------------------------------
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 data = new METADATA();
var data = IllustrationData.getIllustrationData(thumbnail.getAttribute("illustration-id"));
var filterOut = false;
if (data.pageCount == null) return;
//Filter using age rating
filterOut = ((Settings.filterSwitchFlag & 16) == 16 && data.R18 == 0) ||
((Settings.filterSwitchFlag & 32) == 32 && data.R18 == 1) ||
((Settings.filterSwitchFlag & 64) == 64 && data.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 && data.pageCount == 0) filterOut = true; //Filtering out Illustrations
else if ((flag & 4) == 4 && data.pageCount > 0) filterOut = true; //Filtering out Manga
else if ((flag & 8) == 8 && data.bookmarkCount < values[0]) filterOut = true; //Bookmarks
else if ((flag & 16) == 16 && data.viewCount < values[1]) filterOut = true; //Views
else if ((flag & 32) == 32 && data.ratings < values[2]) filterOut = true; //Ratings
else if ((flag & 64) == 64 && data.totalRatings < values[3]) filterOut = true; //Total
} catch (e) { };
}
if (!filterOut)
{
var found;
var tags = data.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.thumbnailLinks.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(IllustrationData.getMetaScore(thumb.getAttribute("illustration-id"))[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();
},
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("title")[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 (link)
{
if (IsIllustrationPage) return;
var thumbnail = link.parentElement;
var metadata = IllustrationData.getIllustrationLinkData(link);
//var metadata = new METADATA();
//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]);
//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("title").length == 0)
{
link.innerHTML = link.innerHTML.replace(/(<img[^>]*>).*/gi, "$1");
link.style.display = "block";
var h1 = document.createElement("h1");
//h1.className = "title";
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 (link)
{
var metadata = IllustrationData.getIllustrationLinkData(link);
//var metadata = new METADATA();
var hotbox = document.createElement("a");
link.appendChild(hotbox);
hotbox.name = "HotBox";
if (metadata.pageCount == 0)
{
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
}
if (IsIllustrationPage)
{
console.log("Illustration Page");
var br = document.createElement("br");
link.insertBefore(br, hotbox);
hotbox.className = "pppHotBoxI";
var img = link.getElementsByTagName("img")[0];
hotbox.style.width = (img.offsetWidth - 4) + "px";
PreviewHQ.intervalImageLoadCheck = setInterval(function ()
{
if (img.naturalWidth > 0)
{
clearInterval(PreviewHQ.intervalImageLoadCheck); PreviewHQ.intervalImageLoadCheck = null;
hotbox.style.width = (img.offsetWidth - 2) + "px";
}
}, 0);
}
else PreviewHQ.adjustHotboxPosition(hotbox);
hotbox.onmouseover = function ()
{
if (PreviewHQ.delay.iID != metadata.illustID)
{
PreviewHQ.delay.tID = setTimeout(function ()
{
//PreviewHQ.delay = new object();
PreviewHQ.onMouseOverHBShowPreview(hotbox);
}, 350);
}
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 = IllustrationData.getIllustrationData(imageID);
//metadata = new METADATA();
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.illustType == 3)
{
data.innerHTML += "<a class='bookmark-count' style='background-color:#FFFD00;' href='" + IQDBPrefix + pThumb.getElementsByTagName("img")[0].src + "&'>IQDB<a>";
}
else
{
data.innerHTML += "<a class='bookmark-count' style='background-color:#FFFD00;' href='" + IQDBPrefix + pThumb.getElementsByTagName("img")[0].src
+ "&fullimage=" + links[0].href + "&'>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: 90%;");
if (metadata.illustType == 2) //Manga
{
/* Sample URLS
http://i2.pixiv.net/img14/img/whatsoy/16791530.jpg
http://i2.pixiv.net/img14/img/whatsoy/16791530_p0.jpg
http://i2.pixiv.net/img14/img/whatsoy/16791530_big_p0.jpg
http://i2.pixiv.net/img14/img/itiba/mobile/3447413_128x128_p0.jpg (ALWAYS JPG)
http://i2.pixiv.net/img14/img/whatsoy/mobile/16791530_128x128_p0.jpg
*/
var thumbContainer = document.createElement("div");
thumbContainer.setAttribute("style", "top: 0; left: 0; overflow: auto; white-space: nowrap; max-width: 100%; width: 100%; min-height: 128px;");
for (var i = 0; i < links.length; i++)
{
var link = document.createElement("a");
var thumbnail = document.createElement("img");
if (metadata.illust150URL)
{
thumbnail.setAttribute("style", "border: 2px ridge #FFF;");
thumbnail.setAttribute("style", "border: 2px ridge #FFF; width: 128px; height: 128px;");
thumbnail.src = metadata.illust150URL.replace(/_p\d+/, "_p" + i);
}
else //Old format
{
thumbnail.setAttribute("style", "border: 2px ridge #FFF; width: 128px; height: 128px;");
thumbnail.src = metadata.illust128URL.replace(/_p\d+/, "_p" + i);
//thumbnail.src = metadata.illust240URL.replace(/_p\d+/, "_p" + i);
}
thumbnail.alt = "Page #" + i;
link.href = links[i].href;
link.appendChild(thumbnail);
thumbContainer.appendChild(link);
}
previewWindow.appendChild(thumbContainer);
}
else if (!IsIllustrationPage)
{
/*
http://i1.pixiv.net/img101/img/nahenaheko/34649107_m.jpg
http://i1.pixiv.net/img101/img/nahenaheko/mobile/34649107_240mw.jpg
http://i2.pixiv.net/img14/img/whatsoy/mobile/16791530_480mw.jpg Size can be large and the quality awaful
http://i1.pixiv.net/img101/img/nahenaheko/34649107_s.jpg
*/
var prethumb = document.createElement("div");
var link = document.createElement("a");
console.log(links);
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) + "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;"><select id="linkerMethod" title="Linker Method"><option title="Illustration Page">1H</option><option title="Phone API">2A</option></select><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="radio" name="ageRating" value="0" />All</li><li><input type="radio" name="ageRating" value="1" />Safe</li><li><input type="radio" name="ageRating" value="2" />R18</li></ul><ul><li><input type="checkbox" name="ageRating2" />Safe</li><li><input type="checkbox" name="ageRating2" />R18</li><li><input type="checkbox" name="ageRating2" />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) iDoc.getElementById("hiddenCtrl").style.display = null;
if (PAGETYPE > 0 && Settings.displayHidden & 2)
{
var sel = iDoc.getElementById("linkerMethod");
IllustrationLinker.requestMethod = GM_getValue("linkerMethod", 2);
if (IllustrationLinker.requestMethod == 1) sel.selectedIndex = 0;
else sel.selectedIndex = 1;
sel.onchange = function (e)
{
IllustrationLinker.requestMethod = this.selectedIndex + 1;
GM_setValue("linkerMethod", IllustrationLinker.requestMethod);
};
}
else iDoc.getElementById("linkerMethod").style.display = "none";
if (PAGETYPE > 1 && Settings.displayHidden & 2)
{
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();
};
var sel = iDoc.getElementById("pageFetchMethod");
sel.selectedIndex = Settings.pageFetchingMethod - 1;
sel.onchange = function (e)
{
Settings.pageFetchingMethod = this.selectedIndex + 1;
GM_setValue("NextPageMethod", Settings.pageFetchingMethod);
};
}
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 (APIAgeRating && PAGETYPE != 8)
{
iDoc.getElementsByName("ageRating")[0].parentElement.parentElement.style.display = "none";
var ars = iDoc.getElementsByName("ageRating2");
for (var i = 0; i < ars.length; i++)
{
ars[i].onclick = SideBar.onAgeRatingFilter;
}
}
else if (!APIAgeRating && PAGETYPE >= 9)
{
iDoc.getElementsByName("ageRating2")[0].parentElement.parentElement.style.display = "none";
if (document.URL.match(/(&|\?)r18=1/)) iDoc.getElementsByName("ageRating")[2].checked = true;
else iDoc.getElementsByName("ageRating")[0].checked = true;
var ratings = iDoc.getElementsByName("ageRating");
for (var i = 0; i < ratings.length; i++) ratings[i].onclick = SideBar.onRadioClick;
}
else
{
iDoc.getElementsByName("ageRating")[0].parentElement.parentElement.style.display = "none";
iDoc.getElementsByName("ageRating2")[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("ageRating2");
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;
}
console.log(Settings.filterSwitchFlag);
PaginatorHQ.filterResult();
},
onRadioClick: function (e)
{
switch (e.target.name)
{
case "ageRating":
//e.target.parentElement.style.visibility = "hidden";
var agerating = e.target.parentElement;
agerating.style.color = "black";
for (var i = 0; i < agerating.children.length; i++) agerating.children[i].disabled = "true";
Pager.intialiseRated(PaginatorHQ.intialiseRated, e.target.value);
setTimeout(
function (agerating)
{ agerating.style.color = "white"; for (var i = 0; i < agerating.children.length; i++) agerating.children[i].disabled = null; },
10000, agerating);
break;
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,
enableCache: GM_getValue("EnableCache", true),
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", "USO-Updater", "DisplayHidden", "EnableCache", "APIAgeRating", "NextPageMethod", "LinkerMethod"];
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("125.125");
//GM_setValue("DisplayHidden", 2);
Settings.displayHidden = GM_getValue("DisplayHidden", 0);
var vals = GM_getValue(Settings.valueName("Generic"), "4094|6|1500,350|500|0|0|0").split("|");
// ----------- Sidebar Display Settings Options (PAGETYPE > 0)
Settings.display = new (makeStruct("artistName illustTitle countList illustLink mangaLinks IQDBLink autoPreview sidebar displayOptions filterOptions"))();
Settings.setObjectPropertiesByFlag(Settings.display, vals[0]);
// ----------- Sidebar Fetch Settings (PAGETYPE > 0)
Settings.fetch = new (makeStruct("nextPage metadata"))();
Settings.setObjectPropertiesByFlag(Settings.fetch, vals[1]);
// ----------- Preview Settings
Settings.preview = new (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(names)
{
var names = names.split(' ');
var count = names.length;
function constructor()
{
for (var i = 0; i < count; i++)
{
this[names[i]] = null;
}
}
return constructor;
}
var APIMetaNames = "illustID userID illustExt illustTitle imgDirectoryNumber userName illust128URL unused1 unused2 illust480URL unused3 unused4 time tags software ratings totalRatings viewCount description pageCount unused5 unused6 bookmarkCount unknown2 userLoginName unused7 R18 unused8 unused9 userProfileImageURL endMarker";
var APIDataFull = makeStruct(APIMetaNames);
var METADATA = makeStruct("userID userName userProfileImageURL illustType illustID illustTitle illust128URL illust480URL illust600URL illustURL pageCount description time tags software ratings totalRatings viewCount bookmarkCount responseCount R18");
var APIAgeRating = GM_getValue("APIAgeRating", true);
/*
=================================================================================================
[VYCS] VARIABLE YOU CAN SET
=================================================================================================*/
//GM_setValue("NextPageMethod", 1); // [1]
//GM_setValue("LinkerMethod", 2); //[2]
//GM_setValue("EnableCache", true); //[3]
//GM_setValue("APIAgeRating", false); //[4]
/*
[1] Default: 3
Method used to load next page. Default now is Method 3 as it is most compatible
method.
1: Uses GM_XmlHttpRequest (Not supported by Google Chrome or Opera)
2: Uses XmlHttpRequest (Not supported by Opera)
3: Uses iform. Slowest method (Default method)
[2] Default: 2
By default we are using PhoneAPI (value 2). Set it to 1 if you want to gete metadata from
Illustration Page (slower).
1: Relies on actual thumbnail to extract bookmark count which isn't always present.
2: Relies on actual thumbnail to extract response count which isn't always present
From version 3.1.47 Illustration page uses API also to get bookmark count
[3] Default: true
Caching of links when using SideBar Age Rated Search.
[4] Default: true
New feature from 3.1.38:
Uses the metadata value for age search by default. Before it XMLHttpRequest to make a new search and the
repopulate the result. Now it filters out the current result using the metadata.
This makes it a lot faster and backward. If you choose to go back to the old method, it is recommended
to enable caching [3] to make it a lot faster.
**This Feature only works with LinkerMethod 2.
**EnableCache should be disabled with this feature.
*/
/*
=================================================================================================
Do not touch
=================================================================================================*/
//Removes pop dialog that appears when there isn't a cookie
if (window.self === window.top)
(function ()
{
PaginatorHQ.addStyle("Adjust-Pixiv", "img._thumbnail {color: white; border: 3px solid; padding: 1px; background-color: white;} 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);
////chrome, firefox, opera, safari
//if (navigator.userAgent.match(/firefox/i)) Settings.pageFetchingMethod = 1;
//else if (navigator.userAgent.match(/chrome/i)) Settings.pageFetchingMethod = 2;
//else if (navigator.userAgent.match(/opera/i)) Settings.pageFetchingMethod = 3;
//else Settings.pageFetchingMethod = GM_getValue("pageFetchingMethod", 3);
Settings.pageFetchingMethod = GM_getValue("NextPageMethod", 3);
Settings.loadSettings();
if (PAGETYPE >= 0)
{
console.info("Pixiv++ Initalising");
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://www.pixiv.net/bookmark_detail.php?illust_id=
*/