Greasy Fork is available in English.

[TS] Pixiv Manga Viewer

A more Powerful Pixiv Manga Viewer with no lazy loading (loads all images) and better interface. Works with Pixiv++ & Generic Image Viewer ロードし、すべての画像|いいえ遅延読み込み|マンガビューア|ダイレクトリンク|ベターインタフェース

// ==UserScript==
// @name            [TS] Pixiv Manga Viewer
// @namespace       TimidScript
// @version         2.2.31
// @description     A more Powerful Pixiv Manga Viewer with no lazy loading (loads all images) and better interface. Works with Pixiv++ & Generic Image Viewer ロードし、すべての画像|いいえ遅延読み込み|マンガビューア|ダイレクトリンク|ベターインタフェース
// @author          TimidScript
// @homepageURL     https://github.com/TimidScript
// @copyright       © 2013+ TimidScript, Some Rights Reserved.
// @license         https://github.com/TimidScript/UserScripts/blob/master/license.txt
// @match           *://www.pixiv.net/member_illust.php?mode=manga&illust_id=*
// @require         https://greasyfork.org/scripts/19967/code/TSL - GM_update.js
// @require         https://greasyfork.org/scripts/19968/code/TSLibrary - Generic.js
// @homeURL         https://greasyfork.org/en/scripts/4684
// @grant           GM_xmlhttpRequest
// @grant           GM_listValues
// @grant           GM_info
// @grant           GM_getMetadata
// @grant           GM_getValue
// @grant           GM_setValue
// @grant           GM_deleteValue
// @grant           GM_registerMenuCommand
// @icon            
// ==/UserScript==


/* License + Copyright Notice
********************************************************************************************
License can be found at: https://github.com/TimidScript/UserScripts/blob/master/license.txt
Below is a copy of the license the may not be up-to-date.

Copyright © TimidScript, Some Rights Reserved.

Redistribution and use in source and binary forms, with or without modification, are permitted provided that the
following conditions are met:

1) GPL-3 License is met that does not conflict with the rest of the license (http://www.gnu.org/licenses/gpl-3.0.en.html)
2) This notice must be included
3) Due credits and link to original author's homepage (included in this notice).
4) Notify the original author of redistribution
5) Clear clarification of the License and Notice to the end user
6) Do not upload on OpenUserJS.org or any other site that infringes on this license

TimidScript's Homepages:  GitHub:      https://github.com/TimidScript
                          GreasyFork:  https://greasyfork.org/users/1455
*/

/* Information
********************************************************************************************
    Version History
------------------------------------

Hotkeys:
  A = ← = K => Previous Page
  Z = → = J => Next Page
  Q = Num 6 => Thumbnail Gallery

  1 = Num 1 => Auto-Height
  2 = Num 2 => Auto-Width
  3 = Num 3 => Auto-Stretch
  4 = Num 4 => Enlarge/Shrink to Client Area
  5 = Num 5 => Alternate through colour schemes
  ` = Num 0 => Reset Size
----------------------------------------------
    Version History
----------------------------------------------
2.2.31 (2017-04-18)
 - Quickfix to support https protocol
2.2.30 (2016-05-27)
 - Altered license
2.2.29 (2016-05-25)
 - Moving to GreasyFork and preparing to remove OUJS files
2.2.28 (2016-04-10)
 - updateURL added
2.2.28 (2016-04-03)
 - Changed license to GPL-3
2.2.26 (2015-10-05)
 - Replaced base64 bmp icon with png version
2.2.25 (2015-08-23)
 - Removed redundant code
2.2.24 (2015-06-27)
 - Bug Fix: userId not userID when getting artist's id
2.2.23 (2015-06-27)
 - Changed to using XMLHttpRequest to supprt Pixiv 3rd party login
2.2.22 (2015-05-15)
 - As of 11/05/2015 the Phone API (SPAPI) is dead. Extract the information from the document instead.
 - Avoided the usage of Public API to bypass multiple login and timeout
 - Added bgc selector array
 - Removed alternating background colours. All saved settings are reset.
 - Changed the content styling
 - Changed the hotkeys to match Generic Image Viewer
2.1.21 (2014-10-02)
 - Bug fixes due to changes in pixiv url syntax
2.1.20 (2014-09-08)
 - Bug Fix: Due to changes to TSL-Generic.
2.1.19 (2014-08-29)
 - Added GM_update
2.1.18  (2014-08-19)
 - Cleaned up header for OUJS
 - Removed old history
2.1.17 (2014-03-28)
 - Bug Fix: Resize issue.
2.1.16 (2014-03-25)
 - Behaviour changed to mimic Generic Image Viewer. See "Information" section above for hotkeys.
 - Multiple background colour options
 - Settings are removed with update
 - Optimised and cleaned up the code
 - Need to manually place the image to make sure correct CSS is applied to image background
 in correct order.
2.0.15 (2013-10-14)
 - document.replaceChild used
2.0.14 (2013-10-03)
 - Previous and next page with mouse click
 - Slight changes in interface
 - Increase top, left active area
**********************************************************************************************/

//#region Global Variables
/*
==============================================================================================
 VYCC: Variables You Can Change
==============================================================================================*/
//GM_setValue("SyncCalls", "5"); //Number of images it loads at a given time
//GM_setValue("ColourSets", '[["DDF0F5","E2E2E1"],["F1F18A","C8F3C8"],["000071","000"],["EACCE6","A15F5F"]]');  //Background colour + Image frame colour


/*
==============================================================================================
 Variables you should not touch
==============================================================================================*/
var MangaBigPagesID = 11319936;
var MaxSyncCalls = GM_getValue("SyncCalls", "5");
var ResizeMode = GM_getValue("ResizeMode", 0); //Bit operator for fitting and expanding images. (1 = FV, 2 = FH, 4 = Expand)
var IQDBType = GM_getValue("IQDBType", 2);
var IQDBTypes = ["All", "anime-pictures", "danbooru", "e-shuushuu", "haruhidoujins", "gelbooru", "konachan", "mangadrawing", "sankaku", "theanimegallery", "yandere", "zerochan"];
var ViewingPage = { previous: -1, current: 0, next: 1 };
var Illust = new Object();


var ColourSets = JSON.parse(GM_getValue("ColourSets", '[["EEF0F5","BFCBE9"],["DDF0F5","E2E2E1"],["F1F18A","C8F3C8"],["000071","000"],["EACCE6","A15F5F"],["FFF","C8C6C6"],["000","C8C6C6"]]'));
var ColoursActive = GM_getValue("ColoursActive", 0);
//#endregion

var ScrollBarThickness = TSL.getScrollBarThickness();
//#endregion

function GetCurrentViewedPage()
{
    var pages = document.getElementsByClassName("mPage");
    var thumbnails;
    if (document.getElementById("Thumbnails"))
        thumbnails = document.getElementById("Thumbnails").getElementsByTagName("a");

    ViewingPage = { previous: -1, current: -1, next: -1 };
    for (i = 0; page = pages[i], i < pages.length; i++)
    {
        if (thumbnails && thumbnails.length > 0) thumbnails[i].style.backgroundColor = null;
        var top = page.getClientRects()[0].top;

        if ((ViewingPage.previous == -1 || ViewingPage.next == -1))
        {
            if (top >= -1 && top <= 50)
            {
                ViewingPage.previous = i;
                ViewingPage.current = i + 1;
                if (ViewingPage.current < Illust.pageCount) ViewingPage.next = i + 2;
            }
            else if (top > 50)
            {
                ViewingPage.previous = i;
                ViewingPage.next = i + 1;
            }
        }
    }
    if (ViewingPage.previous == -1 && ViewingPage.current == -1 && ViewingPage.next == -1)
    {
        ViewingPage.previous = Illust.pageCount - 1;
        ViewingPage.current = Illust.pageCount;
    }
    //console.log("Page Count: " + Illust.pageCount, ViewingPage);

    if (thumbnails && thumbnails.length > 0)
    {
        if (ViewingPage.previous > 0) thumbnails[ViewingPage.previous - 1].style.backgroundColor = "#FFFFBD";
        if (ViewingPage.current > 0) thumbnails[ViewingPage.current - 1].style.backgroundColor = "#FF0";
        if (ViewingPage.next > 0) thumbnails[ViewingPage.next - 1].style.backgroundColor = "#FFFFBD";
    }
}


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 GotoToPageID(id)
{
    var page = document.getElementById(id);
    if (page)
    {
        DisplayMessage("Page #" + (parseInt(id.match(/\d+/)[0])));
        var offset = GetAbsolutePosition(page);
        //console.log(page.id + " Top: " + offset.top);
        scrollTo(null, offset.top);
    }
    else console.error("Passed erroneous element ID: " + id);
}

function GotoToPageNumber(number)
{
    GotoToPageID("Page" + number);
}

function GotoToPage(e)
{
    var link = e.target;
    while (link.tagName != "A") link = link.parentElement;
    GotoToPageID(link.name);
    return false;
}


function RemoveAllDocumentContent()
{
    //Futile attempts to disable javascripts :/
    var doc = document.implementation.createHTMLDocument(document.title);
    document.replaceChild(doc.documentElement, document.documentElement);
}

function KeyDownCallback(e)
{
    //var key = String.fromCharCode(e.keyCode).toLowerCase();
    //e.stopImmediatePropagation();

    switch (e.keyCode)
    {
        case 38: //↑
        case 40: //↓
            e.stopImmediatePropagation();
            break;
        case 37: //←: Load previous page
        case 75: //K: (Pixiv's default for previous image)
        case 65: //A
            e.stopImmediatePropagation();
            GetCurrentViewedPage();
            if (ViewingPage.previous > 0) GotoToPageNumber(ViewingPage.previous);
            else GotoToPageNumber(ViewingPage.current);
            return false;
        case 39://→
        case 74: //J (Pixiv's default for next image)
        case 90: //Z: Load next page
            e.stopImmediatePropagation();
            GetCurrentViewedPage();
            if (ViewingPage.next > 0 && ViewingPage.next <= Illust.pageCount) GotoToPageNumber(ViewingPage.next);
            return false; //window.pageYOffset == document.documentElement.scrollTop
        case 81: //Q: Toggle Thumbnail Gallery Display
        case 102: //Num 6
            e.stopImmediatePropagation();
            var tg = document.getElementById("ThumbGallery");
            if (tg) tg.parentElement.removeChild(tg);
            else DisplayThumbGallery();
            return false;
        case 53: //5 Change Background Colour
        case 101: //Num 4
            e.stopImmediatePropagation();
            ResizeHQ.setColours();
            return false;
        case 79: //O: Pixiv's Original size
            e.stopImmediatePropagation();
            return false;
    }

    var mode = e.keyCode - 48;
    if (e.keyCode == 100) mode = 4;
    else if (e.keyCode != 101 && mode > 10) mode = mode - 48;
    if (e.keyCode == 192 || e.keyCode == 96) mode = 0;

    if (mode >= 0 && mode < 5)
    {
        e.stopImmediatePropagation();
        if (mode == 0) ResizeMode = 0;
        ResizeHQ.adjustSizeMode(mode);
        return false;
    }
}


function CreatePanelControl(text, href)
{
    var panel = document.createElement("span");
    panel.className = "controlPanel";
    var link = document.createElement("a");

    if (text) link.textContent = text;
    if (href) link.href = href;

    panel.appendChild(link);
    return panel;
}

function CreatePanelControlImage(href, imgSrc)
{
    var panel = CreatePanelControl(null, href, imgSrc);
    var image = document.createElement("img");
    image.src = imgSrc;
    panel.firstElementChild.appendChild(image);
    return panel;
}


function DisplayThumbGallery()
{
    var thumbgallery = document.getElementById("ThumbGallery");

    if (!thumbgallery)
    {
        thumbgallery = document.createElement("section");
        thumbgallery.id = "ThumbGallery";
        document.body.appendChild(thumbgallery);

        var infopanel = document.createElement("div");
        infopanel.style.display = "inline-block";
        infopanel.style.margin = "10px 0 0 10px";
        thumbgallery.appendChild(infopanel);


        var panel = CreatePanelControl(Illust.title, "https://www.pixiv.net/member_illust.php?mode=medium&illust_id=" + Illust.id);
        infopanel.appendChild(panel);

        panel = CreatePanelControlImage("https://www.pixiv.net/member.php?id=" + Illust.userID, Illust.userIcon);
        infopanel.appendChild(panel);
        var image = panel.getElementsByTagName("IMG")[0];
        image.style.margin = "0 5px 0 10px";
        image.style.height = "20px";
        image.style.width = "20px";
        infopanel.appendChild(panel);


        panel = CreatePanelControl(Illust.username, "https://www.pixiv.net/member_illust.php?id=" + Illust.userID);
        infopanel.appendChild(panel);

        //var seperator = document.createElement("span");
        //seperator.setAttribute("style", "width: 50px;");
        //infopanel.appendChild(seperator);

        panel = document.createElement("span");
        panel.className = "controlPanel";
        var sel = document.createElement("select");
        sel.style.borderRadius = "0 5px 5px 0";
        sel.style.marginLeft = "50px";
        sel.title = "IQDB Search options";
        sel.onchange = function ()
        {
            IQDBType = sel.selectedIndex;
            GM_setValue("IQDBType", IQDBType);
        };

        for (var i = 0; i < IQDBTypes.length; i++)
        {
            var opt = document.createElement("option");
            opt.textContent = IQDBTypes[i];
            sel.add(opt);
        }
        sel.selectedIndex = IQDBType;
        panel.appendChild(sel);
        infopanel.appendChild(panel);


        var thumbs = document.createElement("div");
        thumbs.id = "Thumbnails";
        thumbgallery.appendChild(thumbs);
        for (var i = 0; i < Illust.pageCount; i++)
        {
            var thumbnail = document.createElement("a");
            var image = document.createElement("img");

            image.src = Illust.thumbnail;
            image.src = image.src.replace(/_p\d+/, "_p" + i);

            var div = document.createElement("div");
            var j = i + 1;
            div.textContent = (j);

            thumbnail.appendChild(image);
            thumbnail.appendChild(div);
            thumbnail.href = "#Page" + j;
            thumbnail.name = "Page" + j;
            thumbnail.onclick = GotoToPage;
            thumbs.appendChild(thumbnail);
        }

        thumbs.onclick = function () { thumbgallery.style.display = "none"; };
        thumbgallery.onclick = function () { thumbgallery.style.display = "none"; };
    }

    thumbgallery.style.display = null;
    GetCurrentViewedPage();

    return false;
}

/*
==============================================================================================

==============================================================================================*/
function ImageCursor(e, img)
{
    var pos = GetAbsolutePosition(img);

    if ((e.pageX - pos.left) < (img.width / 2))
    {
        if (img.getAttribute("page") == 1 && img.className != "cursorS") img.className = "cursorS";
        else if (img.getAttribute("page") > 1 && img.className != "cursorP") img.className = "cursorP";
    }
    else
    {
        if (img.getAttribute("page") == Illust.pageCount && img.className != "cursorS") img.className = "cursorS";
        else if (img.getAttribute("page") < Illust.pageCount && img.className != "cursorN") img.className = "cursorN";
    }
}

function DisplayMessage(msg)
{
    //text-align: center; display:inline-block; width: 100px; background-color: #D3D3D3; border: 1;
    var msgBox = document.getElementById("msgBox");
    if (!msgBox)
    {
        msgBox = document.createElement("span");
        msgBox.id = "msgBox";
        msgBox.setAttribute("style", "position: fixed; bottom: 30px; left: 10px; min-width: 200px; background-color: #D3D3D3; padding-left:10px; border-style: solid; border-color: #FF0000;");
        document.body.appendChild(msgBox);
    }
    msgBox.style.visibility = null;
    var div = document.createElement("div");
    div.textContent = msg;
    msgBox.appendChild(div);

    setTimeout(function (el) { if (el.parentElement.children.length == 1) el.parentElement.style.visibility = "hidden"; el.parentElement.removeChild(el); }, 2000, div);
}


/* Mouse Monitor Functions
====================================================================
 Monitors mouse cursor and set visibility of panels according to
 cursor position.
====================================================================*/
var MouseMonitor =
{
    stID: null,

    onMove: function (e)
    {
        MouseMonitor.isMouseOverPanel(e, "ButtonPanel");
        MouseMonitor.isMouseOverPanel(e, "LinkPanel");
        MouseMonitor.isMouseOverPanel(e, "galleryIcon");
    },

    isMouseOverPanel: function (e, id)
    {
        var panel = document.getElementById(id);
        if (!panel) return;
        if (TSL.isMouseEventInClientArea(e, panel))
        {
            if (panel.style.visibility)
            {
                clearTimeout(MouseMonitor.btnID);
                MouseMonitor.stID = null;
                panel.style.visibility = null;
            }
        }
        else if (!panel.style.visibility && !MouseMonitor.btnID)
        {
            MouseMonitor.stID = setTimeout(function () { panel.style.visibility = "hidden"; MouseMonitor.stID = null; }, 500);
        }
    }
};


/* ResizeHQ Functions
====================================================================
 Functions that deal with the image resize control panel
====================================================================*/
var ResizeHQ =
{
    addButtons: function ()
    {
        var panel = document.createElement("div");
        panel.id = "ButtonPanel";
        panel.setAttribute("style", "position:fixed; top: 50px; left: 5px; display:inline-block; z-index: 100; padding: 1px; background-color: #E1DFDF; border: 1px solid black; border-radius: 5px;");
        TSL.addStyle("", "#ButtonPanel > div {margin: 1px; width: 32px; background-color: #F9FAFA; border-radius: 5px; background-position: center center; background-repeat: no-repeat; cursor:pointer;} .resizeBTN:hover, #BGColorBtn:hover{border-color:red;}");

        TSL.addStyle("", ".resizeBTN {height:32px; border-color: #05F505 !important; border: 2px ridge; }");
        TSL.addStyle("", ".ColourBTN {height:24px;}");



        //TSL.addStyle("Buttons", ".resizeBTN");
        //#BGColorBtn{background-color: #F9FAFA; margin-top:10px; height:32px;width: 32px; border: 2px ridge #05F505; border-radius: 5px; cursor:pointer;}

        for (var i = 0; i < 4 + ColourSets.length ; i++)
        {
            var btn = document.createElement("div");
            if (i < 4) btn.className = "resizeBTN";
            else btn.className = "ColourBTN";
            btn.onclick = ResizeHQ.buttonClick;
            panel.appendChild(btn);
        }

        var btns = panel.querySelectorAll("#ButtonPanel > div");
        btns[0].style.backgroundImage = "URL('')";
        btns[0].title = "Auto-Fit Height (1, Num 1)";
        btns[1].style.backgroundImage = "URL('')";
        btns[1].title = "Auto-Fit Width (2, Num 2)";
        btns[2].style.backgroundImage = "URL('')";
        btns[2].title = "Stretch (3, Num 3)";
        btns[3].title = "Fill Client Area while keeping ratio (4, Num 4)";
        btns[3].style.marginBottom = "5px";


        for (var i = 4; i < btns.length ; i++)
        {
            btns[i].value = i - 4;
            console.log(ColourSets[btns[i].value]);
            btns[i].style.backgroundColor = "#" + ColourSets[btns[i].value][0];
            btns[i].style.border = "2px solid #" + ColourSets[btns[i].value][1];
            btns[i].onclick = ResizeHQ.setColours;
        }

        document.body.appendChild(panel);
        ResizeHQ.resizeButtonsShow();
    },

    buttonClick: function (e)
    {
        var btns = document.getElementsByClassName("resizeBTN");
        for (var i = 0; i < btns.length; i++)
        {
            if (btns[i] == this)
            {
                ResizeHQ.adjustSizeMode(i + 1);
                return;
            }
        }
    },

    adjustSizeMode: function (mode)
    {
        var n = Math.pow(2, mode);
        var enabled = (ResizeMode & n);
        if (enabled) ResizeMode -= n; else ResizeMode += n;

        var msg = "";
        switch (mode)
        {
            case 1:
                msg = "Auto-height";
                break;
            case 2:
                msg = "Auto-width";
                break;
            case 3:
                msg = "Stretch";
                break;
            case 4:
                msg = "Fill Area";
                break;
        }

        if (mode > 0) DisplayMessage(msg + ((!enabled) ? " ON" : " OFF"));
        else DisplayMessage("Resize Off");

        ResizeHQ.readjustImageSizes();
        GM_setValue("ResizeMode", ResizeMode);
    },

    readjustImageSizes: function ()
    {
        var images = document.getElementsByName("MangaImage");
        for (var i = 0; i < images.length; i++)
        {
            var img = images[i];
            var maxH = window.innerHeight - 10;
            var maxW = window.innerWidth - 30;

            //Resize taking into account ScrollBars
            if (document.body.scrollWidth > document.body.clientWidth) maxH - ScrollBarThickness;
            if (document.body.scrollHeight >= document.body.clientHeight) img.style.width = maxW - ScrollBarThickness;

            img.style.maxHeight = (ResizeMode & 2) ? maxH + "px" : null;
            img.style.maxWidth = (ResizeMode & 4) ? maxW + "px" : null;

            if (ResizeMode & 16) //Fill Area: Max size without stretching
            {
                var imageRatio = img.naturalWidth / img.naturalHeight;
                var clientRatio = window.innerWidth / window.innerHeight;

                img.style.width = (imageRatio >= clientRatio) ? maxW + "px" : null;
                img.style.height = (imageRatio < clientRatio) ? maxH + "px" : null;
            }
            else if (ResizeMode & 8) //Stretch image
            {
                img.style.height = (ResizeMode & 2) ? maxH + "px" : null;
                img.style.width = (ResizeMode & 4) ? maxW + "px" : null;
            }
            else
            {
                img.style.height = null;
                img.style.width = null;
            }

            ResizeHQ.resizeButtonsShow();
        }

        GetCurrentViewedPage();
    },

    resizeButtonsShow: function ()
    {
        clearTimeout(ResizeHQ.stID);
        document.getElementById("ButtonPanel").style.visibility = null;

        var btns = document.getElementsByClassName("resizeBTN");
        btns[0].style.backgroundColor = (ResizeMode & 2) ? "yellow" : null;
        btns[1].style.backgroundColor = (ResizeMode & 4) ? "yellow" : null;
        btns[2].style.backgroundColor = (ResizeMode & 8) ? "yellow" : null;
        btns[3].style.backgroundColor = (ResizeMode & 16) ? "red" : null;

        //ResizeHQ.stID = setTimeout(function () { document.getElementById("ButtonPanel").style.visibility = "hidden"; }, 1000);
    },

    setColours: function ()
    {
        if (!isNaN(this.value)) ColoursActive = this.value;
        else if (++ColoursActive >= ColourSets.length) ColoursActive = 0;

        GM_setValue("ColoursActive", ColoursActive);
        MainHQ.addStyles();
    },

    windowResized: function ()
    {
        GetCurrentViewedPage();
        if (ResizeMode != 4 && ResizeMode != 0) ResizeHQ.readjustImageSizes();

        //Reason for this is due to the crap pixiv script that scrolls to the top after resize
        setTimeout(function ()
        {
            if (ViewingPage.current >= 0) GotoToPageNumber(ViewingPage.current);
            else if (ViewingPage.previous >= 0) GotoToPageNumber(ViewingPage.previous);
            else GotoToPageNumber(ViewingPage.next);
        }, 300);
    }
};

/* Main Functions
====================================================================
 Loads manga pages and alters the html interface
====================================================================*/
var MainHQ =
{
    imageCount: 0,
    syncLoads: 0,
    intervalID: 0,

    addStyles: function ()
    {
        TSL.addStyle("DockStyle", "body{margin: 0px;padding: 0px;}div.mPage{width: 100%;max-width: 100%;text-align: center;}#floatingPanel{position: absolute;text-align: center;width: 100%;margin: 0;padding: 0;}.controlPanel{text-align: center;vertical-align: middle;display: table-cell;margin: 0;padding: 0;}.directImageLink{background-color: #FFF;border-radius: 5px;margin: 0 5px;border: groove;}.IQDBLink, .pageCount{border-radius: 5px;text-decoration: none;border: groove;padding: 0 2px 0 2px;}.IQDBLink{background-color: #FF0;color: #00F;}.directImageLink:hover, .IQDBLink:hover, pageCount:hover{border-color: #0FF;}.pageCount{background-color: #E8F2F5;color: #808080;}.resizeBtn, .resizeBtnSelected{background-color: #E9E9EB;border-radius: 5px;border: groove;height: 24px;width: 24px;margin-left: 2px;}.resizeBtn:hover, .resizeBtnSelected:hover{background-color: #0FF;}.resizeBtnSelected{background-color: #FF0;}#ThumbGallery{position: fixed;background-color: #E6E5E5;height: 100%;max-height: 100%;width: 100%;z-index: 500;margin: 0px;left: 0px;top: 0px;}#Thumbnails{overflow: auto;position: absolute;top: 40px;bottom: 0px;}#Thumbnails > a{padding: 5px;margin: 5px;min-width: 128px;background-color: #F8FDFF;display: inline-block;text-align: center;}.cursorP{cursor: url('') 16 16, auto;}.cursorN{cursor: url('') 16 16, auto;}.cursorS{cursor: url('') 24 24, auto;}");
        TSL.addStyle("GalleryThumbs", "#Thumbnails img {max-width: 200px; min-width: 200px;}");
        TSL.addStyle("", ".mPage {padding: 2px 0}");
        TSL.addStyle("BGColour", "body {background-color:#" + ColourSets[ColoursActive][0] + "}");
        TSL.addStyle("IMGFrame", "div.mPage > img{border: 2px solid #" + ColourSets[ColoursActive][1] + ";}");
        //TSL.addStyle("IMGShadow", "div.mPage > img{margin: 0 5px; box-shadow: 10px 10px 5px #888888;}");
    },

    LoadAllMangaPages: function ()
    {
        RemoveAllDocumentContent();
        MainHQ.addStyles();


        for (var i = 0; i < Illust.pageCount; i++)
        {
            var link = document.createElement("a");

            link.href = Illust.full.replace(/_p\d+/, "_p" + i);
            link.title = i;
            document.body.appendChild(link);
        }

        MainHQ.intervalID = setInterval(MainHQ.loadNextImage, 100);

        //Creating Thumbnail Gallery Button
        var d = document.createElement("d");
        d.id = "galleryIcon";
        d.setAttribute("style", "visibility:hidden; position:fixed; right: 0px; top:0px; z-index:100; padding:15px;");
        var a = document.createElement("a");
        a.href = "#ShowGallery";

        var img = document.createElement("img");
        img.src = "";

        img.className = "directImageLink";
        a.appendChild(img);
        d.appendChild(a)
        document.body.appendChild(d);

        //#region LinkPanel
        var linkPanel = document.createElement("div");
        linkPanel.setAttribute("style", "visibility:hidden;  position: fixed; left: 10px; top: 10px; z-index:100; border: 1px ridge gray; padding: 5px;");
        linkPanel.id = "LinkPanel"

        var panel = CreatePanelControl(Illust.title, "https://www.pixiv.net/member_illust.php?mode=medium&illust_id=" + Illust.id);
        linkPanel.appendChild(panel);

        panel = CreatePanelControlImage("https://www.pixiv.net/member.php?id=" + Illust.userID, Illust.userIcon);
        linkPanel.appendChild(panel);
        var image = panel.getElementsByTagName("IMG")[0];
        image.style.margin = "0 5px 0 10px";
        image.style.height = "20px";
        image.style.width = "20px";
        linkPanel.appendChild(panel);

        panel = CreatePanelControl(Illust.username, "https://www.pixiv.net/member_illust.php?id=" + Illust.userID);
        linkPanel.appendChild(panel);

        document.body.appendChild(linkPanel);
        a.onclick = DisplayThumbGallery;
        document.body.onmousemove = MouseMonitor.onMove;
    },

    loadNextImage: function ()
    {
        if (MaxSyncCalls == MainHQ.syncLoads) return;
        MainHQ.syncLoads++;

        MainHQ.imageCount++;
        var mPage = document.createElement("div");
        mPage.className = "mPage";
        mPage.id = "Page" + MainHQ.imageCount;


        var img = document.createElement("img");

        img.src = Illust.full.replace(/_p\d+/, "_p" + (MainHQ.imageCount - 1));
        img.setAttribute("page", (MainHQ.imageCount));
        img.alt = "Page " + (MainHQ.imageCount) + " out of " + Illust.pageCount + "pages";
        img.title = "Page " + (MainHQ.imageCount) + " out of " + Illust.pageCount + "pages";
        img.name = "MangaImage";
        img.id = "IMG" + MainHQ.imageCount;

        MainHQ.imageEvents(img);
        mPage.appendChild(img);
        //document.body.appendChild(mPage);
        document.body.insertBefore(mPage, document.body.children[MainHQ.imageCount - 1]);

        if (MainHQ.imageCount == Illust.pageCount) clearInterval(MainHQ.intervalID);
        TSL.removeNode("fb-root"); //Sometimes an extra div is created by Pixiv script. Just removing it.
    },

    imageEvents: function (img)
    {
        img.onload = function (e)
        {
            MainHQ.syncLoads--;
            ResizeHQ.readjustImageSizes();
            this.onload = null;
        };

        setTimeout(ResizeHQ.readjustImageSizes, 500, img);

        img.onclick = function (e)
        {
            var page = parseInt(img.getAttribute("page"));
            if (this.className == "cursorP")
            {
                GotoToPageNumber(--page);
                ImageCursor(e, document.getElementById("IMG" + page));
            }
            else if (this.className == "cursorN")
            {
                GotoToPageNumber(++page);
                ImageCursor(e, document.getElementById("IMG" + page));
            }
        };

        img.onmousemove = function (e)
        {
            ImageCursor(e, this);
            var pos = GetAbsolutePosition(this);

            var floatingPanel = document.getElementById("floatingPanel");
            if ((e.pageY - pos.top) > 100 && ((pos.top + this.height) - e.pageY) > 100)
            {
                if (floatingPanel) floatingPanel.parentElement.removeChild(floatingPanel);
                return;
            }

            if (!floatingPanel)
            {
                floatingPanel = document.createElement("section");
                floatingPanel.id = "floatingPanel";
                floatingPanel.style.position = "absolute";
                document.body.appendChild(floatingPanel);


                var panels = document.createElement("div");
                panels.style.display = "inline-block";
                floatingPanel.appendChild(panels);


                //Pager
                var panelLink = CreatePanelControl(this.getAttribute("page") + "/" + Illust.pageCount, "#" + this.parentElement.parentElement.id);
                panelLink.firstElementChild.name = this.parentElement.parentElement.id;
                panelLink.firstElementChild.onclick = GotoToPageID;
                panelLink.firstElementChild.className = "pageCount";
                panels.appendChild(panelLink);

                //Direct Image Link
                var panelLink = CreatePanelControlImage(null, "");
                panelLink.getElementsByTagName("img")[0].className = "directImageLink";
                panels.appendChild(panelLink);

                //
                var panelLink = CreatePanelControl("IQDB", null);
                panelLink.firstElementChild.className = "IQDBLink";
                panels.appendChild(panelLink);
            }


            if (floatingPanel.getAttribute("page") != img.getAttribute("page"))
            {
                var pos = GetAbsolutePosition(img);
                if ((e.pageY - pos.top) <= 100) floatingPanel.style.top = pos.top + "px";
                else floatingPanel.style.top = (pos.top + img.clientHeight - 50) + "px";

                floatingPanel.style.left = window.pageXOffset + "px";

                var links = floatingPanel.getElementsByTagName("a");
                links[0].textContent = img.getAttribute("page") + "/" + Illust.pageCount;
                links[1].href = img.src;
                links[2].href = "https://" + ((IQDBType == 0) ? "www" : IQDBTypes[IQDBType]) + ".iqdb.org/?url=" + Illust.thumbnail.replace("_p0", "_p" + (img.getAttribute("page") - 1)) + "&"; //& at the end so as not to get picked up by download managers
                floatingPanel.setAttribute("page", img.getAttribute("page"));
            }
        };
    }
};



/* Main Function
====================================================================
Main function that extracts illustration information from the
document.
====================================================================*/
(function ()
{
    //Removes old settings
    var Version = 1002;
    if (GM_getValue("Version", 0) != Version)
    {
        var names = GM_listValues();
        for (var i = 0; name = names[i], i < names.length; i++)
        {
            var skipNames = ["SyncCalls", "IQDBType", "GMU-CoolingPeriod", "GMU-Timestamp"];
            var found = false;
            for (var j = 0; j < skipNames.length; j++) found = found || (name.indexOf(skipNames[j]) == 0);
            if (!found) GM_deleteValue(name);
        }
        ResizeMode = 0;
        GM_setValue("ResizeMode", 0);
        GM_setValue("Version", Version);
    }

    Illust.id = unsafeWindow.pixiv.context.illustId;
    Illust.title = document.querySelector(".breadcrumbs h1").textContent;
    Illust.userID = unsafeWindow.pixiv.context.userId;
    Illust.username = document.querySelector(".user").textContent;
    Illust.userIcon = document.querySelector(".user-icon").src.replace("_ss", "_s");
    //Illust.userLoginName = Illust.userIcon.match(/\/profile\/([^\/]+)\//)[1];
    Illust.pageCount = unsafeWindow.pixiv.context.images.length;
    Illust.full = unsafeWindow.pixiv.context.images[0];
    Illust.full = Illust.full.replace(/\/c\/\d+.+img-master\/(.+\d+_p0)(_master\d+)?/, "/img-original/$1");
    Illust.thumbnail = unsafeWindow.pixiv.context.thumbnailImages[0];
    Illust.thumbnail = Illust.thumbnail.replace("_128x128_p0", "_240mw_p0"); //Handles old naming format to display none-square larger thumbnails
    //Illust.thumbnail = Illust.thumbnail.replace(".png", ".jpg");
    Illust.thumbnail = Illust.thumbnail.replace("_p0_square1200", "_p0_master1200"); //Handles new naming format to display none-square larger thumbnails
    //Illust.thumbnail = Illust.thumbnail.replace("/128x128/", "/150x150/");
    Illust.thumbnail = Illust.thumbnail.replace("/128x128/", "/240x480/");

    if (Illust.thumbnail.indexOf("/img-master/") < 0) //Old format
    {
        if (Illust.id >= 11319936) Illust.full = Illust.full.replace("_p0", "_big_p0");
        changeInterface();
    }
    else //New format. We do not know the extension of the large size so we have to get information on it.
    {
        var oReq = new XMLHttpRequest();
        oReq.open("GET", "https://www.pixiv.net/member_illust.php?mode=manga_big&illust_id=" + Illust.id + "&page=0", true);
        oReq.responseType = "text";
        oReq.timeout = 15000;
        oReq.onload = function (e)
        {
            if (oReq.status == 200)
            {
                var doc = new DOMParser().parseFromString(oReq.response, "text/html");
                try
                {
                    Illust.full = doc.getElementsByTagName("img")[0].src;
                }
                catch (err) { }
            }
            changeInterface();
        };
        oReq.send();
    }

    function changeInterface()
    {
        console.log(Illust);
        MainHQ.LoadAllMangaPages();
        ResizeHQ.addButtons();
        window.onresize = ResizeHQ.windowResized;
        window.addEventListener("keydown", KeyDownCallback, true);
    }
})();