[TS] Generic Image Viewer

A more Powerful Image Viewer with info panel support for Pixiv, deviantArt, imgur, Seiga Nico and nijie.info. Support numpad keys now.

  1. // ==UserScript==
  2. // @name [TS] Generic Image Viewer
  3. // @namespace TimidScript
  4. // @version 2.2.47
  5. // @description A more Powerful Image Viewer with info panel support for Pixiv, deviantArt, imgur, Seiga Nico and nijie.info. Support numpad keys now.
  6. // @author TimidScript
  7. // @homepageURL https://github.com/TimidScript
  8. // @copyright © 2013+ TimidScript, Some Rights Reserved.
  9. // @license https://github.com/TimidScript/UserScripts/blob/master/license.txt
  10. // @include *
  11. // @require https://greasyfork.org/scripts/19967/code/TSL - GM_update.js
  12. // @require https://greasyfork.org/scripts/19968/code/TSLibrary - Generic.js
  13. // @homeURL https://greasyfork.org/en/scripts/18448
  14. // @grant GM_getValue
  15. // @grant GM_setValue
  16. // @grant GM_listValues
  17. // @grant GM_deleteValue
  18. // @grant GM_xmlhttpRequest
  19. // @grant GM_info
  20. // @grant GM_getMetadata
  21. // @grant GM_registerMenuCommand
  22. // @run-at document-start
  23. // @icon 
  24. // ==/UserScript==
  25.  
  26. /* License + Copyright Notice
  27. ********************************************************************************************
  28. License can be found at: https://github.com/TimidScript/UserScripts/blob/master/license.txt
  29. Below is a copy of the license the may not be up-to-date.
  30.  
  31. Copyright © TimidScript, Some Rights Reserved.
  32.  
  33. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the
  34. following conditions are met:
  35.  
  36. 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)
  37. 2) This notice must be included
  38. 3) Due credits and link to original author's homepage (included in this notice).
  39. 4) Notify the original author of redistribution
  40. 5) Clear clarification of the License and Notice to the end user
  41. 6) Do not upload on OpenUserJS.org or any other site that infringes on this license
  42.  
  43. TimidScript's Homepages: GitHub: https://github.com/TimidScript
  44. GreasyFork: https://greasyfork.org/users/1455
  45. */
  46.  
  47. /* Information
  48. ********************************************************************************************
  49.  
  50. There are two sets of keys now. One using character keys the second using are numpad keys.
  51. Hotkeys:
  52. [A], [Num 1] => Auto-Height
  53. [S], [Num 2] => Auto-Width
  54. [Z], [Num 3] => Auto-Stretch
  55. [X], [Num .] => Enlarge/Shrink to Client Area
  56. [Q], [Num 0] => Reset Size
  57.  
  58. [D], [Num 4] => Rotate Left
  59. [W], [Num 5] => Reset rotation transformation
  60. [F], [Num 6] => Rotate Right
  61.  
  62. [X], [Num 7] => Mirror Horizontally
  63. [E], [Num 8] => Reset mirror transformation
  64. [C], [Num 9] => Mirror Vertically
  65.  
  66. [T], [Num *] => Change colour scheme (down)
  67. [R], [Num /] => Change colour scheme (up)
  68.  
  69. TODO: Replace the video controls
  70. ------------------------------------
  71. Version History
  72. ------------------------------------
  73. 2.2.47 (2017-04-11)
  74. - Bugfix for Chrome support
  75. 2.2.46 Beta (2016-11-22)
  76. - BugFix: imgur link fix
  77. - Better support for videos
  78. - Support for video sites (gfycat)
  79. - Support for twitter :orig url
  80. 2.2.45 (2016-10-13)
  81. - Bugfix: Video loop
  82. 2.2.44 (2016-09-21)
  83. - Changed the hotkeys
  84. 2.2.43 (2016-09-13)
  85. - Separated hotkey for reset rotation and mirror image.
  86. - Revised hotkeys and added numpad keys
  87. 2.2.42 (2016-09-03)
  88. - Bugfix: "z-index" added to panels
  89. - Bugfix: Video panel no longer shows on sites like steam
  90. - Removed no longer needed nicoseiga image checking
  91. - Removed the article container and just kept the image in the default body parent node
  92. - Added my own center algorithm that handles the rotation aspect.
  93. 2.2.41 (2016-07-26)
  94. - Moved SWF controls to the bottom
  95. 2.2.40 (2016-05-27)
  96. - Altered license
  97. 2.2.39 (2016-25-25)
  98. - Moving to GreasyFork and preparing the removal of files from OUJS
  99. 2.2.38 (2016-04-10)
  100. - updateURL added
  101. 2.2.37 (2016-04-03)
  102. - Changed license to GPL-3
  103. 2.2.36 (2016-03-28)
  104. - Updated URL for Google Reverse Image Search
  105. 2.2.35 (2016-02-26)
  106. - Support for TamperMonkey window object separation
  107. 2.2.34 (2015-12-27)
  108. - Bugfix: Original title of image kept
  109. - Bugfix: getScrollBarThickness is called after the document has finished loading (interactive or complete).
  110. - Bugfix: Resizing and centring of image corrected
  111. - Better support for Pixiv images
  112. - Bugfix: SWF Support
  113. - Added support for mp4, flv abnd webm
  114. - Added loop button for videos
  115. 2.2.33 (2015-10-05)
  116. - Replaced base64 bmp icon with png version
  117. 2.2.32 (2015-09-23)
  118. - Changed the hotkeys
  119. - Removed colour scheme preview
  120. - Links only appear when you hover link panel
  121. 2.2.31 (2015-07-31) Major release due to major changes in inner workings and addition of CSS
  122. styler.
  123. - Bugfix: "Unknown" is used for missing DeviantArt artist's name
  124. - Does not wait for the image to load. It attempts to alter page at interactive stage
  125. - Corrected a few typos
  126. - Improved all regular expressions for extracting information from url. Using match instead of replace
  127. - Improved the way it extracts user id from nijie.info
  128. - Support for DeviantArt https
  129. - Improved auto-hide visibility of meta panel
  130. - Improved extraction of metadata, such in deviantArt underscore (_) is representation of hyphen (-)
  131. - Added preview of all youtube thumbnails
  132. - Changed content to use CSS styles
  133. - CSS preview over hover
  134. - Changed hotkeys
  135. - Links href is only added when control bar is visible. (Remove tag suffix)
  136. 1.2.30 (2015-06-27)
  137. - URI (Base64) Script icon
  138. 1.2.29 (2015-06-20)
  139. - Removed the comment for URL alteration
  140. 1.2.28 (2015-06-19)
  141. - Removed cloneInto as it's no longer needed
  142. - In flash controls are always visible
  143. - Bugfix: where controls are missing for deviantart images/swf that do not have id
  144. - Remove postfix "?timidscript_*" from image source and url. The tag is used in a personal script that has been in beta phase since forever.
  145. - Source links now have the prefix "timidscript_source" to avoid being picked up by downloaders
  146. - Using styles to toggle visibility
  147. - Removed the message box.
  148. 1.2.27 (2015-05-14)
  149. - As of 11/05/2015 the Phone API (SPAPI) is dead. Using TSL-Pixiv library to get information from HTML.
  150. - Avoided the usage of Public API to bypass multiple login and timeout
  151. - Changed the hotkeys to match Pixiv Manga Viewer
  152. .
  153. .
  154. .
  155. 1.0.1 (2013-10-06)
  156. - Initial Release with Pixiv Support Only
  157. **********************************************************************************************/
  158. //if (window !== window.top) return;
  159.  
  160. function makeStruct(names)
  161. {
  162. var names = names.split(' ');
  163. var count = names.length;
  164. function constructor()
  165. {
  166. for (var i = 0; i < count; i++)
  167. {
  168. this[names[i]] = null;
  169. }
  170. }
  171. return constructor;
  172. }
  173.  
  174. /*
  175. ==============================================================================================
  176. VYCC: Variables You Can Change
  177. ==============================================================================================*/
  178. //GM_setValue("BGColors", '["#252525", "#EDE9E9", "#E7FDF7", "#F1E8D2"]') //Image Background Colours
  179.  
  180.  
  181. var ResizeMode = GM_getValue("ResizeMode", 0); //Bit operator for fitting and expanding images. (1 = FV, 2 = FH, 4 = Expand)
  182. var ScrollBarThickness;
  183. var DELAY = 1500;
  184. //var DELAY = 500000000;
  185.  
  186. function Styler()
  187. {
  188. document.stopCapture = true;
  189. var styler = document.createElement("div");
  190. styler.id = "GIVStyler"
  191.  
  192. var tb = document.createElement("textarea");
  193. tb.setAttribute("style", "resize:none;width:100%;");
  194. //tb.setAttribute("wrap", "off");
  195. styler.appendChild(tb);
  196.  
  197. tb.value = decodeURI(GM_getValue("CSS"));
  198. tb.onkeydown = function (e)
  199. {
  200. var k = e.which || e.keyCode;
  201. if (k != 9) return;
  202. e.preventDefault();
  203. var start = this.selectionStart,
  204. end = this.selectionEnd;
  205.  
  206. this.value = this.value.substring(0, start) + " " + this.value.substring(end);
  207. this.selectionEnd = this.selectionStart = start + 4;
  208. }
  209.  
  210. tb.onkeypress = function (e)
  211. {
  212. clearInterval(tb.timeout);
  213. tb.timeout = setTimeout(function () { TSL.addStyle("GIViewerCSS", tb.value); }, 1000);
  214. }
  215.  
  216. var con = document.createElement("section");
  217. con.setAttribute("style", "");
  218. styler.appendChild(con);
  219.  
  220. var el = document.createElement("button");
  221. el.textContent = "Apply";
  222. el.style.width = "80px";
  223. el.onclick = function (e)
  224. {
  225. TSL.addStyle("GIViewerCSS", tb.value);
  226. var css = encodeURI(tb.value);
  227. GM_setValue("CSS", css);
  228. console.log(css);
  229. }
  230. con.appendChild(el);
  231.  
  232.  
  233. el = document.createElement("button");
  234. el.textContent = "Exit";
  235. el.style.width = "80px";
  236. el.onclick = function (e)
  237. {
  238. TSL.addStyle("GIViewerCSS", decodeURI(GM_getValue("CSS")));
  239. TSL.removeNode(styler);
  240. document.stopCapture = false;
  241. }
  242. con.appendChild(el);
  243. document.body.appendChild(styler);
  244.  
  245. window.addEventListener("resize", function ()
  246. {
  247. tb.style.height = (styler.clientHeight - con.clientHeight) + "px";
  248. });
  249.  
  250. tb.style.height = (styler.clientHeight - con.clientHeight) + "px";
  251. }
  252.  
  253. function CreatePanelMeta(text, href)
  254. {
  255. var panel = document.createElement("span");
  256. panel.className = "metaPanel";
  257. var link = document.createElement("a");
  258.  
  259. if (text) link.textContent = text;
  260. if (href)
  261. {
  262. link.href = href;
  263. link.setAttribute("data-href", href);
  264. }
  265.  
  266. panel.appendChild(link);
  267. return panel;
  268. }
  269.  
  270. function CreatePanelImage(href, imgSrc, title)
  271. {
  272. var panel = CreatePanelMeta(null, href);
  273. var image = document.createElement("img");
  274. image.src = imgSrc;
  275. panel.firstElementChild.appendChild(image);
  276. if (title) panel.title = title;
  277. return panel;
  278. }
  279.  
  280. function CreatePanelControlIcon(text, classname, href)
  281. {
  282. var panel = CreatePanelMeta(null, href);
  283. //panel.firstElementChild.setAttribute("src", href);
  284. panel.firstElementChild.href = href;
  285. var div = document.createElement("div");
  286. div.className = classname;
  287. panel.firstElementChild.appendChild(div);
  288. panel.firstElementChild.title = text;
  289. return panel;
  290. }
  291.  
  292. var ControlHQ =
  293. {
  294. data: new (makeStruct("imgTitle imgURL userIcon userHome userName userGallery"))(),
  295. flash: false,
  296. video: false,
  297.  
  298. addStyles: function ()
  299. {
  300. ControlHQ.flash
  301.  
  302. document.body.className = "CSS" + GM_getValue("CSS-Selection", 1);
  303.  
  304. TSL.addStyle("CSS-Body", 'body {padding:0;margin:0;}');
  305. TSL.addStyle("CSS-MainImage", '#imageBox {vertical-align: middle; text-align:center; display:table-cell;} #theImage {top:0;bottom:0;margin:0}');
  306.  
  307. TSL.addStyle("CSS-PanelHover", "#LinkPanel:hover, #ControlPanel:hover, #YoutubeThumbs:hover {visibility: visible !important;}");
  308.  
  309. TSL.addStyle("CSS-LinkPaneM", '#LinkPanel a {color:red; font-size:14px; font-weight:700;font-family:font-family: "Times New Roman", Times, serif;}');
  310.  
  311. TSL.addStyle("CSS-LinkPanel", '#LinkPanel {position: fixed; z-index: 100; left: 10px; top: 10px; z-index:100; border: 2px ridge white; padding: 2px 5px; background-image:linear-gradient(to bottom,#818180 0%,#F3F3F5 50%, #818180 100% ); box-shadow: 0 0 2px 3px black;}'
  312. + '.metaPanel {text-align: center;vertical-align: middle;display: table-cell;margin: 0;padding: 0;}'
  313. + '#LinkPanel img {height:24px;width:24px;border: 1px solid white; padding: 1px; background-color:black; margin-right: 5px;} .metaPanel + #UserHome {padding-left: 10px;} .metaPanel + [title=IQDB] {padding-left: 20px;}'
  314. );
  315.  
  316.  
  317. TSL.addStyle("YoutubeStyler", " #YoutubeThumbs {position:fixed;right:10px;top:10px;max-width:120px;padding: 1px;background-color:white; border:1px ridge black;}");
  318.  
  319. TSL.addStyle("CSS-ControlPanel", "#ControlPanel {position:fixed; z-index: 100; top: 60px; left: 10px; display:inline-block; background-image:linear-gradient(to right,#818180 0%,#F3F3F5 50%, #818180 100% ); padding: 2px; border: 2px ridge white; box-shadow: 0 0 2px 3px black;}"
  320. + '#ControlPanel div {display:inline-block; margin:1px;} #ControlPanel {width: 80px; text-align:center;}'
  321. + ".resizeBTN, .transformBTN {height:32px;width: 32px; background-color: #F9FAFA; border: 2px ridge #05F505; border-radius: 5px; background-position: center center; background-repeat: no-repeat; cursor:pointer;}"
  322. + ".resizeBTN + .transformBTN, .transformBTN + .cssBTN {margin-top: 8px;}"
  323. + ".resizeBTN:hover, .cssBTN:hover, .transformBTN:hover {border-color:red;}"
  324. + ".cssBTN {width: 70px;height:40px;border: 2px solid #05F505; border-radius: 5px; cursor:pointer;background-color: white; padding:0!important; position:relative;}"
  325. + '.cssPRE {width:40px;height:20px;background-color:white; margin: 0 !important; position: absolute;left:15px;top:10px;font-family: "Courier New", Courier, monospace;font-size: 20px;font-weight: 500; color:gray}'
  326. );
  327.  
  328. //INFO: Use encodeURI rather than JSON as JSON seems causes an unknown error when I try and paste it in
  329. var css = GM_getValue("CSS", null)
  330. if (!css)
  331. {
  332. css = '/*%0ACSS#%20is%20a%20class%20that%20is%20applied%20to%20the%20body.%20The%20hash%20is%20replaced%20by%20the%20button%20number%20from%20the%20style%20selection%20(1-6).%0A*/%0A%0A/*%20CSS1%20*******************************************/%0A.CSS1,%20#cssBTN1%0A%7B%0A%20%20%20%20background-color:black;%0A%7D%20%0A%0A.CSS1%20#theImage%0A%7B%0A%20%20%20box-shadow:%200%200%203px%206px%20white;%0A%7D%0A%0A#cssPRE1%0A%7B%0A%20%20%20box-shadow:%200%200%201px%203px%20white%7D%0A%0A%0A/*%20CSS2%20*******************************************/%0A.CSS2,%20#cssBTN2%0A%7B%0A%20%20%20%20background-color:white;%0A%7D%20%0A%0A.CSS2%20#theImage%0A%7B%0A%20%20%20box-shadow:%200%200%203px%206px%20black;%0A%7D%0A%0A#cssPRE2%0A%7B%0A%20%20%20box-shadow:%200%200%201.5px%203px%20black;%0A%7D%0A%0A/*%20CSS3%20*******************************************/%0A.CSS3,%20#cssBTN3%0A%7B%0A%20%20%20%20background-color:#E0E0E0;%0A%7D%20%0A%0A.CSS3%20#theImage%0A%7B%0A%20%20%20box-shadow:%200%200%203px%206px%20black;%0A%7D%0A%0A#cssPRE3%0A%7B%0A%20%20%20box-shadow:%200%200%201px%203px%20black;%0A%7D%0A%0A/*%20CSS4%20*******************************************/%0A.CSS4,%20#cssBTN4%0A%7B%0A%20%20%20%20background-color:lightblue;%0A%7D%20%0A%0A.CSS4%20#theImage%0A%7B%0A%20%20%20box-shadow:%200%200%203px%206px%20blue;%0A%7D%0A%0A#cssPRE4%0A%7B%0A%20%20%20box-shadow:%200%200%201px%203px%20blue;%0A%7D%0A%0A%0A/*%20CSS5%20*******************************************/%0A.CSS5,%20#cssBTN5%0A%7B%0A%20%20%20%20background-color:orange;%0A%7D%20%0A%0A.CSS5%20#theImage%0A%7B%0A%20%20%20box-shadow:%200%200%203px%206px%20black;%0A%7D%0A%0A#cssPRE5%0A%7B%0A%20%20%20box-shadow:%200%200%201px%203px%20black;%0A%7D%0A%0A%0A/*%20CSS6%20*******************************************/%0A.CSS6,%20#cssBTN6%0A%7B%0A%20%20%20%20background-color:yellow;%0A%7D%20%0A%0A.CSS6%20#theImage%0A%7B%0A%20%20%20box-shadow:%200%200%203px%206px%20black;%0A%7D%0A%0A#cssPRE6%0A%7B%0A%20%20%20box-shadow:%200%200%201px%203px%20black;%0A%7D%0A%0A/*%20CSS7%20*******************************************/%0A.CSS7,%20#cssBTN7%0A%7B%0A%20%20%20%20background-color:black;%0A%7D%20%0A%0A/*%20CSS8%20*******************************************/%0A.CSS8,%20#cssBTN8%0A%7B%0A%20%20%20%20background-color:darkgray;%0A%7D%0A%0A/*%20CSS9%20*******************************************/%0A.CSS9,%20#cssBTN9%0A%7B%0A%20%20%20%20background-color:lightblue;%0A%7D%0A%0A/*%20PREVIEW%20BUTTONS%20**********************************/%0A.CSS1%20#cssBTN1,%20.CSS2%20#cssBTN2,.CSS3%20#cssBTN3,%20.CSS4%20#cssBTN4,%20.CSS5%20#cssBTN5,%20.CSS6%20#cssBTN6,%20.CSS7%20#cssBTN7,%20.CSS8%20#cssBTN8,%20.CSS9%20#cssBTN9%0A%7B%0A%20%20%20%20border-color:%20red!important;%0A%20%20%20%20box-shadow:%200%200%201px%204px%20yellow;%0A%7D%0A%0A.CSS1%20#cssPRE1,%20.CSS2%20#cssPRE2,.CSS3%20#cssPRE3,%20.CSS4%20#cssPRE4,%20.CSS5%20#cssPRE5,%20.CSS6%20#cssPRE6,%20.CSS7%20#cssPRE7,%20.CSS8%20#cssPRE8,%20.CSS9%20#cssPRE9%20%20%0A%7B%0A%20%20%20%20color:black;%0A%20%20%20%20font-weight:%20900;%0A%7D%0A%0A.cssPRE%0A%7B%0A%20%20%20%20%20background-image:%20linear-gradient%20(orange,yellow,white,yellow,orange);%0A%7D%0A%0A/*%20THIS%20WINDOW%20**********************************/%0A#GIVStyler%0A%7B%0A%20%20%20%20position:fixed;%20%0A%20%20%20%20width:40%25;%20%0A%20%20%20%20height:80%25;%0A%20%20%20%20right:10px;%0A%20%20%20%20top:10%25;%0A%20%20%20%20background-color:%20black;%0A%20%20%20%20border:2px%20ridge%20white;%0A%20%20%20%20box-shadow:%200%200%202px%204px%20black;%0A%7D%0A%0A#GIVStyler%20%3E%20textarea%0A%7B%0A%20%20%20%20background-color:#FFFFFA;%0A%7D%0A%0A#GIVStyler%20%3E%20section%0A%7B%0A%20%20%20%20padding:%203px%205px;%20text-align:right;%0A%7D';
  333. GM_setValue("CSS", css);
  334. }
  335. TSL.addStyle("GIViewerCSS", decodeURI(css));
  336.  
  337. if (ControlHQ.flash || ControlHQ.video)
  338. {
  339. TSL.addStyle("CSS-LinkPanelFV", "#LinkPanel {position:static; display:block; height:28px;}");
  340. }
  341. },
  342.  
  343. displayImage: function ()
  344. {
  345. for (var i = document.head.children.length - 1; i >= 0; i--)
  346. {
  347. var child = document.head.children[i];
  348. if (child.tagName != "TITLE") document.head.removeChild(child);
  349. }
  350.  
  351. for (var i = document.body.children.length - 1; i >= 0; i--)
  352. {
  353. var child = document.body.children[i];
  354. //document.body.removeChild(child);
  355. //if (child.id != "USOUpdaterMenu") document.body.removeChild(child);
  356. if (child.tagName != "IMG") document.body.removeChild(child);
  357. }
  358.  
  359. /* Add Image
  360. ********************/
  361. var img = document.querySelector("img");
  362. img.id = "theImage";
  363.  
  364. ControlHQ.data.imgSrc = img.src;
  365. img.onload = ControlHQ.readjustImageSize;
  366. setTimeout(ControlHQ.readjustImageSize, 500);
  367.  
  368. var a = document.createElement("a");
  369. a.href = img.src.replace(/(\?\d+$|\?[^\\\.\/]+$)/, "");
  370. a.href = a.href.replace(/\?\d+$/, "");
  371. a.title = "***IMAGE***";
  372. document.body.appendChild(a);
  373.  
  374. ControlHQ.createControlPanel();
  375. ControlHQ.readjustImageSize();
  376.  
  377. window.addEventListener("keydown", ControlHQ.keyDownCallback, true);
  378. window.onresize = ControlHQ.readjustImageSize;
  379.  
  380. /* Set favourite icon
  381. **********************/
  382. var link = document.createElement('link');
  383. link.type = 'image/x-icon';
  384. link.rel = 'shortcut icon';
  385. link.href = img.src;
  386. document.head.appendChild(link);
  387. },
  388.  
  389. createLinkPanel: function ()
  390. {
  391. console.log("createLinkPanel");
  392. //if (document.location.protocol.match("file:")) return;
  393. TSL.removeNode("LinkPanel");
  394.  
  395. var img = document.getElementById("theImage");
  396. if (img) ControlHQ.data.imgTitle += " (" + img.naturalWidth + "×" + img.naturalHeight + ")";
  397.  
  398. var panel,
  399. linkPanel = document.createElement("div");
  400.  
  401. linkPanel.id = "LinkPanel";
  402. document.body.insertBefore(linkPanel, document.body.firstElementChild);
  403.  
  404. var data = ControlHQ.data;
  405. if (data.imgURL)
  406. {
  407. panel = CreatePanelMeta(data.imgTitle, data.imgURL);
  408. panel.title = "Illustration Page";
  409. linkPanel.appendChild(panel);
  410. }
  411. if (data.userHome)
  412. {
  413. if (!data.userIcon) data.userIcon = "";
  414. panel = CreatePanelImage(data.userHome, data.userIcon, "User Homepage");
  415. panel.id = "UserHome";
  416.  
  417. var image = panel.getElementsByTagName("IMG")[0];
  418. linkPanel.appendChild(panel);
  419. }
  420.  
  421. if (data.userGallery)
  422. {
  423. panel = CreatePanelMeta(data.userName, data.userGallery);
  424. panel.title = "User Gallery";
  425. linkPanel.appendChild(panel);
  426. }
  427.  
  428. if (ControlHQ.flash)
  429. {
  430. document.body.appendChild(linkPanel);
  431. //TSL.addStyle("","#LinkPanel {position:fixed;top:0;left:0;width:100%;z-index:100;}");
  432.  
  433. var flash = document.getElementsByTagName("Embed")[0];
  434. panel = CreatePanelMeta("[SWF]", flash.src);
  435. panel.firstElementChild.style.marginLeft = "10px";
  436. linkPanel.appendChild(panel);
  437. }
  438. else if (ControlHQ.video)
  439. {
  440. var video = document.getElementsByTagName("video")[0];
  441. video.loop = true;
  442.  
  443. panel = CreatePanelMeta("[VIDEO]", video.baseURI);
  444. panel.firstElementChild.style.marginLeft = "10px";
  445. linkPanel.appendChild(panel);
  446.  
  447.  
  448. var loopOff = "",
  449. loopOn = "";
  450.  
  451. panel = CreatePanelImage("", loopOn, "Video Repeat");
  452. panel.firstElementChild.setAttribute("style", "cursor:pointer; margin-left:30px;");
  453. linkPanel.appendChild(panel);
  454.  
  455. linkPanel.querySelector("img").style.backgroundColor = "gray";
  456. panel.firstElementChild.onclick = function ()
  457. {
  458. if (video.loop == true)
  459. {
  460. video.loop = false;
  461. linkPanel.querySelector("img").src = loopOff;
  462. }
  463. else
  464. {
  465. video.loop = true;
  466. linkPanel.querySelector("img").src = loopOn;
  467. video.play();
  468. }
  469. }
  470. }
  471. else //Image
  472. {
  473. panel = CreatePanelImage("https://www.iqdb.org/?url=" + data.imgSrc + "&"
  474. , ""
  475. , "IQDB");
  476. linkPanel.appendChild(panel);
  477.  
  478. panel = CreatePanelImage("https://saucenao.com/search.php?db=999&url=" + data.imgSrc + "&"
  479. , ""
  480. , "SauceNAO");
  481. linkPanel.appendChild(panel);
  482.  
  483. panel = CreatePanelImage("https://imgops.com/" + data.imgSrc.replace("http://", "") + "&"
  484. , ""
  485. , "ImgOps");
  486. linkPanel.appendChild(panel);
  487.  
  488. panel = CreatePanelImage("https://encrypted.google.com/searchbyimage?image_url=" + data.imgSrc + "&safe"
  489. , ""
  490. , "Google");
  491. linkPanel.appendChild(panel);
  492.  
  493. panel = CreatePanelImage("https://www.tineye.com/parse?url=" + data.imgSrc + "&"
  494. , ""
  495. , "TinEye");
  496. linkPanel.appendChild(panel);
  497.  
  498. setTimeout(ControlHQ.setPanelVisibility, DELAY, document.getElementById("LinkPanel"), false);
  499.  
  500. HideLinks();
  501. linkPanel.onmouseenter = ShowLinks;
  502. linkPanel.onmouseleave = HideLinks;
  503. }
  504.  
  505. function HideLinks()
  506. {
  507. var links = linkPanel.querySelectorAll("[data-href]");
  508. for (var i = 0; i < links.length; i++) links[i].href = "";
  509. }
  510.  
  511. function ShowLinks()
  512. {
  513. var links = linkPanel.querySelectorAll("[data-href]");
  514. for (var i = 0; i < links.length; i++) links[i].href = links[i].getAttribute("data-href");
  515. }
  516. },
  517.  
  518.  
  519. createControlPanel: function ()
  520. {
  521. var btn, panel = document.createElement("div");
  522. panel.id = "ControlPanel";
  523.  
  524. var keys = ["a", "s", "z", "x", "d", "f", "c", "v"]
  525. for (var i = 0; i < 4; i++)
  526. {
  527. btn = document.createElement("div");
  528. btn.className = "resizeBTN";
  529. btn.onclick = ControlHQ.resizeButtonClick;
  530. btn.setAttribute("style", "position:relative");
  531. panel.appendChild(btn);
  532. }
  533.  
  534. var btns = panel.getElementsByClassName("resizeBTN");
  535. btns[0].style.backgroundImage = "URL('')";
  536. btns[0].title = "Auto-Fit Height (a)";
  537. btns[1].style.backgroundImage = "URL('')";
  538. btns[1].title = "Auto-Fit Width (z)";
  539. btns[2].style.backgroundImage = "URL('')";
  540. btns[2].title = "Stretch (z)";
  541. btns[3].style.backgroundImage = "URL('')";
  542. btns[3].title = "Fill Client Area while keeping ratio (x)";
  543.  
  544.  
  545. for (var i = 0; i < 4; i++)
  546. {
  547. btn = document.createElement("div");
  548. btn.className = "transformBTN";
  549. btn.value = 0;
  550. btn.onclick = ControlHQ.transformButtonClick;
  551. panel.appendChild(btn);
  552. }
  553.  
  554. TSL.addStyle("SpinMeRightRound", ".rotate90 {-webkit-transform: rotate(90deg); -moz-transform: rotate(90deg); -o-transform: rotate(90deg); -ms-transform: rotate(90deg); transform: rotate(90deg);}"
  555. + ".reflectMe {-webkit-transform: scaleX(-1) ; -moz-transform: scaleX(-1) ; -o-transform: scaleX(-1) ; -ms-transform: scaleX(-1) ; transform: scaleX(-1);}"
  556. );
  557.  
  558. btn = document.createElement("button");
  559. btn.textContent = "Edit";
  560. btn.setAttribute("style", "width:100%;margin-top: 5px;");
  561. btn.onclick = Styler;
  562. panel.appendChild(btn);
  563.  
  564. var btns = panel.getElementsByClassName("transformBTN");
  565. btns[0].style.backgroundImage = btns[1].style.backgroundImage = "URL('')";
  566. TSL.addClass(btns[0], "reflectMe");
  567.  
  568. btns[2].style.backgroundImage = "URL('')";
  569. TSL.addClass(btns[2], "rotate90");
  570. btns[3].style.backgroundImage = "URL('')";
  571.  
  572.  
  573. for (var i = 0, preview; i < 9; i++)
  574. {
  575. btn = document.createElement("div");
  576. btn.className = "cssBTN";
  577. btn.id = "cssBTN" + (i + 1);
  578. btn.title = "Sets CSS" + (i + 1) + "style";
  579. btn.onclick = changeCSS;
  580. //btn.onmouseenter = previewCSS;
  581. //btn.onmouseleave = savedCSS;
  582.  
  583. preview = document.createElement("div");
  584. preview.textContent = (i + 1);
  585. preview.className = "cssPRE";
  586. preview.id = "cssPRE" + (i + 1);
  587. btn.appendChild(preview);
  588. btn.value = i + 1;
  589. panel.appendChild(btn);
  590. }
  591.  
  592. function changeCSS(e)
  593. {
  594. clearTimeout(this.tid);
  595. GM_setValue("CSS-Selection", this.value);
  596. document.body.className = "CSS" + this.value;
  597. }
  598.  
  599. function previewCSS(e)
  600. {
  601. clearTimeout(this.tid);
  602. var value = this.value;
  603. if (GM_getValue("CSS-Selection") != value)
  604. document.tid = setTimeout(function ()
  605. {
  606. document.body.className = "CSS" + value;
  607. }, 500)
  608. }
  609.  
  610. function savedCSS(e)
  611. {
  612. clearTimeout(this.tid);
  613. document.body.className = "CSS" + GM_getValue("CSS-Selection");
  614. }
  615.  
  616.  
  617. document.body.appendChild(panel);
  618. },
  619.  
  620. showControlsPanel: function ()
  621. {
  622. var panel = document.getElementById("ControlPanel");
  623. clearTimeout(panel.timeoutID);
  624.  
  625. panel.style.visibility = null;
  626.  
  627. var btns = document.getElementsByClassName("resizeBTN");
  628. btns[0].style.backgroundColor = (ResizeMode & 2) ? "yellow" : null;
  629. btns[1].style.backgroundColor = (ResizeMode & 4) ? "yellow" : null;
  630. btns[2].style.backgroundColor = (ResizeMode & 8) ? "yellow" : null;
  631. btns[3].style.backgroundColor = (ResizeMode & 16) ? "red" : null;
  632.  
  633. panel.timeoutID = setTimeout(function () { document.getElementById("ControlPanel").style.visibility = "hidden"; }, DELAY);
  634. },
  635.  
  636. resizeButtonClick: function ()
  637. {
  638. var btns = document.getElementsByClassName("resizeBTN");
  639. for (var i = 0; i < btns.length; i++)
  640. {
  641. if (btns[i] == this)
  642. {
  643. ControlHQ.adjustSizeMode(i + 1);
  644. return;
  645. }
  646. }
  647. },
  648.  
  649. transformButtonClick: function (e)
  650. {
  651. var img = document.getElementById("theImage");
  652. var css = "";
  653.  
  654. var btns = document.getElementsByClassName("transformBTN");
  655.  
  656. if (btns[0] == this) btns[0].value--;
  657. else if (btns[1] == this) btns[0].value++;
  658. else this.value++;
  659.  
  660. if (btns[0].value == -1) btns[0].value = 3;
  661.  
  662. if (btns[0].value == 1) css = "rotate(90deg)";
  663. else if (btns[0].value == 2) css = "rotate(180deg)";
  664. else if (btns[0].value == 3) css = "rotate(270deg)";
  665. else btns[0].value = 0;
  666.  
  667. if (btns[2].value == 1) css += " scaleX(-1)";
  668. else btns[2].value = 0;
  669.  
  670. if (btns[3].value == 1) css += " scaleY(-1)";
  671. else btns[3].value = 0;
  672.  
  673. css = css.trim() + "; ";
  674. css = "transform: " + css + "-webkit-transform: " + css + "-moz-transform: " + css + "-o-transform: " + css + "-ms-transform: " + css;
  675.  
  676. TSL.addStyle("RobotsinDisguise", "#theImage {" + css + "}");
  677.  
  678. if (this.value == 0) this.style.backgroundColor = null;
  679. else this.style.backgroundColor = "yellow";
  680.  
  681. btns[0].style.backgroundColor = (btns[0].value) ? "yellow" : null;
  682. btns[1].style.backgroundColor = btns[0].style.backgroundColor;
  683.  
  684. ControlHQ.readjustImageSize();
  685. },
  686.  
  687. readjustImageSize: function ()
  688. {
  689. var img = document.getElementById("theImage");
  690. img.removeAttribute("class");
  691. img.removeAttribute("width");
  692. img.removeAttribute("height");
  693.  
  694. var reverse = (document.getElementsByClassName("transformBTN")[0].value % 2 == 1);
  695.  
  696. // 2=Height, 4=Width, 8=Stretch, 16=Fit
  697. if (!reverse)
  698. {
  699. img.style.maxHeight = (ResizeMode & 2) ? window.innerHeight + "px" : null;
  700. img.style.maxWidth = (ResizeMode & 4) ? window.innerWidth + "px" : null;
  701. }
  702. else
  703. {
  704. img.style.maxWidth = (ResizeMode & 2) ? window.innerHeight + "px" : null;
  705. img.style.maxHeight = (ResizeMode & 4) ? window.innerWidth + "px" : null;
  706. }
  707.  
  708. if (ResizeMode & 16) //Fit image
  709. {
  710. var imageRatio = (reverse) ? (img.naturalHeight / img.naturalWidth) : (img.naturalWidth / img.naturalHeight);
  711. var clientRatio = window.innerWidth / window.innerHeight;
  712. if (!reverse)
  713. {
  714. img.style.width = (imageRatio >= clientRatio) ? window.innerWidth + "px" : null;
  715. img.style.height = (imageRatio < clientRatio) ? window.innerHeight + "px" : null;
  716. }
  717. else
  718. {
  719. img.style.height = (imageRatio >= clientRatio) ? window.innerWidth + "px" : null;
  720. img.style.width = (imageRatio < clientRatio) ? window.innerHeight + "px" : null;
  721. }
  722. }
  723. else if (ResizeMode & 8) //Stretch
  724. {
  725. if (!reverse)
  726. {
  727. img.style.height = (ResizeMode & 2) ? window.innerHeight + "px" : null;
  728. img.style.width = (ResizeMode & 4) ? window.innerWidth + "px" : null;
  729. //Resize taking into account ScrollBars
  730. if (ResizeMode & 2 && document.body.scrollWidth > document.body.clientWidth) img.style.height = (window.innerHeight - ScrollBarThickness) + "px";
  731. if (ResizeMode & 4 && document.body.scrollHeight > document.body.clientHeight) img.style.width = (window.innerWidth - ScrollBarThickness) + "px";
  732. }
  733. else
  734. {
  735. img.style.width = (ResizeMode & 2) ? window.innerHeight + "px" : null;
  736. img.style.height = (ResizeMode & 4) ? window.innerWidth + "px" : null;
  737. //Resize taking into account ScrollBars
  738. if (ResizeMode & 2 && document.body.scrollWidth > document.body.clientWidth) img.style.width = (window.innerHeight - ScrollBarThickness) + "px";
  739. if (ResizeMode & 4 && document.body.scrollHeight > document.body.clientHeight) img.style.height = (window.innerWidth - ScrollBarThickness) + "px";
  740. }
  741. }
  742. else //Height & Width
  743. {
  744. img.style.height = null;
  745. img.style.width = null;
  746.  
  747. if (ResizeMode & 2 && document.body.scrollWidth > document.body.clientWidth) img.style.maxHeight = (window.innerHeight - ScrollBarThickness) + "px";
  748. if (ResizeMode & 4 && document.body.scrollHeight > document.body.clientHeight) img.style.maxWidth = (window.innerWidth - ScrollBarThickness) + "px";
  749. }
  750.  
  751.  
  752. //Image placement algorithm
  753. var w = img.clientWidth,
  754. h = img.clientHeight;
  755. console.log(w / h, h / w);
  756. img.style.position = "absolute";
  757. img.style.left = 0;
  758. img.style.top = 0;
  759. if (reverse)
  760. {
  761. var offsetW = ((h - w) / 2);
  762. var offsetH = ((w - h) / 2);
  763.  
  764. if (h < window.innerWidth) img.style.marginLeft = (((window.innerWidth - h) / 2) + offsetW) + "px";
  765. else img.style.marginLeft = (offsetW) + "px";
  766.  
  767. if (w < window.innerHeight) img.style.marginTop = (((window.innerHeight - w) / 2) + offsetH) + "px";
  768. else img.style.marginTop = (offsetH) + "px";
  769. }
  770. else
  771. {
  772. if (w < window.innerWidth) img.style.marginLeft = ((window.innerWidth - w) / 2) + "px";
  773. else img.style.marginLeft = "0px";
  774.  
  775. if (h < window.innerHeight) img.style.marginTop = ((window.innerHeight - h) / 2) + "px";
  776. else img.style.marginTop = "0px";
  777. }
  778.  
  779.  
  780.  
  781. ControlHQ.showControlsPanel();
  782. },
  783.  
  784. adjustSizeMode: function (mode)
  785. {
  786. var n = Math.pow(2, mode);
  787. var enabled = (ResizeMode & n);
  788. if (enabled) ResizeMode -= n; else ResizeMode += n;
  789.  
  790. var msg = "";
  791. switch (mode)
  792. {
  793. case 1:
  794. msg = "Auto-height";
  795. break;
  796. case 2:
  797. msg = "Auto-width";
  798. break;
  799. case 3:
  800. msg = "Stretch";
  801. break;
  802. case 4:
  803. msg = "Fill Area";
  804. break;
  805. }
  806.  
  807. ControlHQ.readjustImageSize();
  808. GM_setValue("ResizeMode", ResizeMode);
  809. },
  810.  
  811.  
  812. keyDownCallback: function (e)
  813. {
  814. var key = e.which || e.keyCode;
  815. if (document.stopCapture) return false;
  816.  
  817. console.log("KEY", key);
  818. //Load Styles
  819. if (key == 82 || key == 84 || key == 106 || key == 111) // [r] [t] [Num *] [Num /]
  820. {
  821. var n = document.body.className.match(/CSS(\d)/)[1];
  822. if (key == 82 || key == 111) n++; else n--;
  823. if (n > 9) n = 1; else if (n < 1) n = 9;
  824.  
  825. document.body.className = "CSS" + n;
  826. GM_setValue("CSS-Selection", n);
  827. ControlHQ.showControlsPanel();
  828. return false;
  829. }
  830.  
  831.  
  832. if (key == 81 || key == 96) //Reset Size [Q] [Num .]
  833. {
  834. e.stopImmediatePropagation();
  835. var btns = document.querySelectorAll('.resizeBTN[style*=background-color]');
  836. for (var i = 0; i < btns.length; i++) btns[i].click();
  837. return false;
  838. }
  839.  
  840. if (key == 87 || key == 101) //Reset Rotation [W] [Num 5]
  841. {
  842. e.stopImmediatePropagation();
  843. btns = document.querySelectorAll('.transformBTN');
  844. btns[0].value = -5; btns[0].click();
  845. return false;
  846. }
  847.  
  848. if (key == 69 || key == 104) //Reset Mirror [E] [Num 8]
  849. {
  850. e.stopImmediatePropagation();
  851. btns = document.querySelectorAll('.transformBTN');
  852. btns[2].value = -5; btns[2].click();
  853. btns[3].value = -5; btns[3].click();
  854. return false;
  855. }
  856.  
  857. var hk = [65, 83, 90, 88, 68, 70, 67, 86]; // a s z x d f c v
  858. var hkNum = [97, 98, 99, 110, 100, 102, 103, 105]; // Num Keys: 1 2 3 . 4 6 7 9
  859. var btns = document.querySelectorAll(".resizeBtn, .transformBtn");
  860. for (var i = 0; i < hk.length; i++)
  861. {
  862. if (hk[i] == key || hkNum[i] == key)
  863. {
  864. e.stopImmediatePropagation();
  865. btns[i].click();
  866. return false;
  867. }
  868. }
  869. },
  870.  
  871. monitorMouse: function (e)
  872. {
  873. isMouseOverPanel(e, "ControlPanel");
  874. isMouseOverPanel(e, "LinkPanel");
  875. isMouseOverPanel(e, "YoutubeThumbs");
  876.  
  877. function isMouseOverPanel(e, id)
  878. {
  879. var panel = document.getElementById(id);
  880. if (!panel) return;
  881. if (TSL.isMouseEventInClientArea(e, panel))
  882. {
  883. panel.clean = true;
  884. clearTimeout(panel.timeoutID);
  885. if (panel.style.visibility) ControlHQ.setPanelVisibility(panel, true);
  886. }
  887. else if (!panel.style.visibility && panel.clean)
  888. {
  889. panel.clean = false;
  890. panel.timeoutID = setTimeout(ControlHQ.setPanelVisibility, DELAY, panel, false);
  891. }
  892. }
  893. },
  894.  
  895. setPanelVisibility: function (panel, visible)
  896. {
  897. panel.style.visibility = (visible) ? null : "hidden";
  898. }
  899. };
  900.  
  901. (function ()
  902. {
  903. var start = setInterval(function ()
  904. {
  905. if (document.readyState != "loading")
  906. {
  907. clearInterval(start);
  908. ScrollBarThickness = TSL.getScrollBarThickness();
  909. LoadGIV();
  910. }
  911. }, 50);
  912.  
  913. function LoadGIV()
  914. {
  915. console.info("Generic Image Viewer");
  916.  
  917. //Removes old settings
  918. var Version = 1002;
  919. if (GM_getValue("Version", 0) != Version)
  920. {
  921. var names = GM_listValues();
  922. for (var i = 0; name = names[i], i < names.length; i++)
  923. {
  924. var skipNames = ["USO-Updater"];
  925. var found = false;
  926. for (var j = 0; j < skipNames.length; j++) found = found || (name.indexOf(skipNames[j]) == 0);
  927. if (!found) GM_deleteValue(name);
  928. GM_setValue("Version", Version)
  929. }
  930.  
  931. ResizeMode = 0;
  932. GM_setValue("ResizeMode", 0)
  933. }
  934.  
  935. //Cleans the URL links of the postfix "?timidscript_". This tag is used in a personal script that is still in beta phase (2015/06)
  936. if (document.URL.match(/[\?&]timidscript_[_a-z]+$/i))
  937. {
  938. console.log("GIViewer: Removed URL search tag");
  939. window.history.pushState(null, "", document.URL.replace(/[\?&]timidscript_[_a-z]+$/i, ""));
  940. var imgs = document.getElementsByTagName("img");
  941. for (var i = 0; i < imgs.length; i++) imgs[i].src = imgs[i].src.replace(/[\?&]timidscript_[_a-z]+$/i, "");
  942. }
  943.  
  944. var hostname = document.location.hostname,
  945. pathname = document.location.pathname;
  946. if (document.body.children.length == 1 && document.querySelector('embed[type="application/x-shockwave-flash"]'))
  947. {
  948. console.info("GIViewer: FLASH");
  949. ControlHQ.flash = true;
  950. }
  951. else if (document.body.children.length == 1 && document.body.children[0].tagName == "VIDEO")
  952. {
  953. console.info("GIViewer: VIDEO");
  954. ControlHQ.video = true;
  955. }
  956. else if (document.body.children[0].tagName == "IMG" && (
  957. document.URL.match(/.+\.(jpg|gif|jpeg|png|bmp)(\?\d+|:large|:orig|\?[^\\\.\/]+)?$/i)
  958. || document.URL.match(/\/[a-zA-Z0-9]+$/)
  959. || document.URL.match(/\/\/www\.pixiv\.net.+mode=manga_big.*&illust_id=(\d+)/i)
  960. || hostname.match(/^thumbs\..+\.com/i)
  961. ))
  962. {
  963. var imgs = document.getElementsByTagName("img");
  964. if (!document.URL.match("img.youtube") && (imgs.length == 0 || imgs.length > 1 || document.body.children.length > 2)) return;
  965. ControlHQ.displayImage();
  966. }
  967. else if (hostname == "gfycat.com" && document.location.pathname.match(/\/\w+$/))
  968. {
  969. console.info("GIViewer: gfycat");
  970.  
  971. var name = document.location.pathname.substr(1);
  972. url = "https://gfycat.com/ifr/" + name;
  973.  
  974. GM_xmlhttpRequest(
  975. {
  976. url: url,
  977. method: "GET",
  978. headers: { "User-agent": navigator.userAgent, "Accept": "text/html", Referer: "" },
  979. timeout: 10000,
  980. onload: function (response)
  981. {
  982. if (response.status == 200)
  983. {
  984. var doc = new DOMParser().parseFromString(response.responseText, "text/html");
  985. var links = doc.querySelectorAll(".iframe-video .mp4-source, .iframe-video .webm-source");
  986.  
  987. if (links.length > 0)
  988. {
  989. var el = document.getElementById("small-gif")
  990. if (el) el.href += "?hide";
  991.  
  992. el = document.getElementById("large-gif");
  993. if (el) el.href += "?hide";
  994. }
  995.  
  996. var sizes = { thumbs: 0, zippy: 1, fat: 2, giant: 3 }, largestSize = 0;
  997. for (var i = 0; i < links.length; i++) if (sizes[links[i].src.match(/\/\/(\w+)/)[1]] > largestSize) largestSize = sizes[links[i].src.match(/\/\/(\w+)/)[1]];
  998.  
  999. var holder = document.querySelector("#controls-container .left");
  1000.  
  1001. for (var i = 0; i < links.length; i++)
  1002. {
  1003. console.log(links[i].src);
  1004. if (sizes[links[i].src.match(/\/\/(\w+)/)[1]] != largestSize) continue;
  1005. var a = document.createElement("a");
  1006. a.href = links[i].src;
  1007. a.textContent = (links[i].className.match("webm")) ? "WebM" : "MP4";
  1008. a.setAttribute("style", "margin: 0 5px");
  1009. holder.appendChild(a);
  1010. }
  1011. }
  1012. else console.warn("ERROR: ", url);
  1013. }
  1014. });
  1015.  
  1016. return;
  1017. }
  1018. else return;
  1019.  
  1020. if (hostname.match(/^[a-z0-9]+\.pixiv\.net$/i)) //Pixiv Site
  1021. {
  1022. console.info("GIViewer: Pixiv");
  1023.  
  1024. var id = document.querySelector("img").src.match(/\/(\d+)(?:_p\d+|_big_p\d+|_.*master.+)?\.(?:jpg|gif|jpeg|png|bmp)/i);
  1025.  
  1026. if (!id) return;
  1027.  
  1028. id = id[1];
  1029. if (document.URL.match("/profile/"))
  1030. {
  1031. ControlHQ.data.userName = "Gallery [" + id + "]";
  1032. ControlHQ.data.userHome = "http://www.pixiv.net/member.php?id=" + id;
  1033. ControlHQ.data.userGallery = "http://www.pixiv.net/member_illust.php?id=" + id;
  1034. ControlHQ.data.userIcon = document.URL;
  1035. ControlHQ.createLinkPanel();
  1036. }
  1037. else
  1038. {
  1039. ControlHQ.data.imgTitle = "Illustration Page";
  1040. ControlHQ.data.imgURL = "http://www.pixiv.net/member_illust.php?mode=medium&illust_id=" + id;
  1041. ControlHQ.createLinkPanel();
  1042.  
  1043. GM_xmlhttpRequest({
  1044. url: "http://www.pixiv.net/member_illust.php?mode=medium&illust_id=" + id,
  1045. method: "GET",
  1046. timeout: 15000,
  1047. headers: { "User-agent": navigator.userAgent, "Accept": "text/html", Referer: "http://www.pixiv.net" },
  1048. onload: function (response)
  1049. {
  1050. if (response.status == 200)
  1051. {
  1052. var doc = new DOMParser().parseFromString(response.responseText, "text/html");
  1053. var script = doc.evaluate("//div[@id='wrapper']//script[contains(text(),'pixiv.context.illustId')]", doc, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;
  1054. if (script)
  1055. {
  1056. unsafeWindow.eval("pixiv = {}; pixiv.context= {};");
  1057. unsafeWindow.eval(script.innerHTML);
  1058. ControlHQ.data.imgTitle = unsafeWindow.pixiv.context.illustTitle;
  1059. ControlHQ.data.userIcon = doc.querySelector(".user-image").src.replace(".jpg", "_s.jpg");
  1060. ControlHQ.data.userHome = "http://www.pixiv.net/member.php?id=" + unsafeWindow.pixiv.context.userId;
  1061. ControlHQ.data.userName = unsafeWindow.pixiv.context.userName;
  1062. ControlHQ.data.userGallery = "http://www.pixiv.net/member_illust.php?id=" + unsafeWindow.pixiv.context.userId;
  1063. ControlHQ.createLinkPanel();
  1064. }
  1065. }
  1066. }
  1067. });
  1068. }
  1069. }
  1070. else if (hostname.match(/^\w+\.deviantart\.net$/i)) //deviantart
  1071. {
  1072. console.info("GIViewer: DeviantArt");
  1073.  
  1074. var id = pathname.match(/-(d\w{6})\.(?:jpg|bmp|png|gif|swf)/i),
  1075. user = pathname.match(/_by_(\w+)(?:-d\w{6})?\.(?:jpg|bmp|png|gif|swf)/i);
  1076.  
  1077. if (id)
  1078. {
  1079. ControlHQ.data.imgTitle = "Illustration Page";
  1080. ControlHQ.data.imgURL = "http://www.deviantart.com/gallery/#/" + id[1];
  1081. }
  1082.  
  1083. if (user)
  1084. {
  1085. user = user[1].replace("_", "-");
  1086. ControlHQ.data.userName = user;
  1087. ControlHQ.data.userHome = "http://" + user + ".deviantart.com";
  1088. ControlHQ.data.userGallery = "http://" + user + ".deviantart.com/gallery/?catpath=/";
  1089. }
  1090.  
  1091. ControlHQ.createLinkPanel();
  1092. }
  1093. else if (hostname.match(/^(i\.)?imgur.com$/i)) //imgur
  1094. {
  1095. console.info("GIViewer: imgur");
  1096. var id = pathname.match(/\w+/)[0];
  1097.  
  1098. ControlHQ.data.imgTitle = "Image Page";
  1099. ControlHQ.data.imgURL = "http://imgur.com/" + id;
  1100.  
  1101. ControlHQ.createLinkPanel();
  1102. }
  1103. else if (hostname.match(/^pic\d+\.nijie\.info$/i)) //nijie.info
  1104. {
  1105. console.info("GIViewer: nijie.info");
  1106.  
  1107. var user, name = pathname.match(/([\d+|_]+)\.(?:jpg|bmp|png|gif)/i);
  1108. if (!name) return;
  1109. name = name[1];
  1110. //Timestamp appears to be always 14 digits long
  1111. user = name.match(/\d+_(\d+)_(\d+)_\d{14}/); //<count>_<imgID>_<userID>_<time>
  1112. if (user)
  1113. {
  1114. ControlHQ.data.imgTitle = "Illustration Page";
  1115. ControlHQ.data.imgURL = "http://nijie.info/view.php?id=" + user[1];
  1116. user = user[2];
  1117. }
  1118. else
  1119. {
  1120. if (!user) user = name.match(/(\d+)_\d{14}_\d+/); //<userID>_<time>_<count>
  1121. if (!user) user = name.match(/(\d+)_\d{14}/); //<userID>_<time>
  1122. if (!user) user = name.match(/\d{14}(\d+)(?:_\d)?/); //<time><userid> <time><userid>_<num>
  1123. user = user[1];
  1124. }
  1125.  
  1126. if (!user) return;
  1127.  
  1128. ControlHQ.data.userName = "Gallery [" + user + "]";
  1129. ControlHQ.data.userGallery = "http://nijie.info/members_illust.php?id=" + user;
  1130. ControlHQ.data.userHome = "http://nijie.info/members.php?id=" + user;
  1131.  
  1132. ControlHQ.createLinkPanel();
  1133. }
  1134. else if (hostname.match(/^\w+\.nicoseiga\.jp/i)) //Nico Nico Seiga
  1135. {
  1136. console.info("GIViewer: Nico Nico Seiga");
  1137.  
  1138. var id = pathname.match(/\d+$/)[0];
  1139. ControlHQ.data.imgTitle = "Illustration Page";
  1140. ControlHQ.data.imgURL = "http://seiga.nicovideo.jp/seiga/im" + id;
  1141.  
  1142. ControlHQ.createLinkPanel();
  1143. }
  1144. else if (document.URL.match(/https?:\/\/img\.youtube\..+\/vi\//i)) //Nico Nico Seiga
  1145. {
  1146. console.info("GIViewer: Youtube");
  1147.  
  1148. var id = document.URL.match(/\/vi\/([^\/]+)/i)[1];
  1149. ControlHQ.data.imgTitle = "Video Page";
  1150. ControlHQ.data.imgURL = "https://www.youtube.com/watch?v=" + id;
  1151.  
  1152. ControlHQ.createLinkPanel();
  1153. if (pathname.match(/(maxresdefault|hqdefault|\d)\.jpg/i))
  1154. {
  1155. var thumbnails = document.createElement("section");
  1156. thumbnails.id = "YoutubeThumbs";
  1157. document.body.appendChild(thumbnails);
  1158.  
  1159. for (var i = 0, img; i < 3; i++)
  1160. {
  1161. img = document.createElement("img");
  1162.  
  1163. img.src = document.URL.replace(/(maxresdefault|hqdefault|\d)\.jpg/i, i + ".jpg");
  1164. img.style.width = "120px";
  1165. thumbnails.appendChild(img);
  1166. }
  1167. thumbnails.timeoutID = setTimeout(function () { document.getElementById("YoutubeThumbs").style.visibility = "hidden" }, DELAY);
  1168. }
  1169. }
  1170. else ControlHQ.createLinkPanel();
  1171.  
  1172. ControlHQ.addStyles();
  1173.  
  1174. if (!ControlHQ.flash && !ControlHQ.video) window.addEventListener("mousemove", ControlHQ.monitorMouse);
  1175. }
  1176. })();
  1177.  
  1178.  
  1179.  
  1180. /* Helpful information (Reverse image search)
  1181. ********************************************************************************************
  1182. http://saucenao.com/search.php?db=999&url=
  1183. http://www.tineye.com/parse?url=
  1184. http://google.com/searchbyimage?hl=en&site=search&image_url=
  1185. http://imgops.com/<url-without-http>
  1186. http://www.iqdb.org/?url=
  1187.  
  1188.  
  1189. 999 All Databases
  1190. 0 H-Magazines
  1191. 1 H-Anime
  1192. 2 H-Game CG
  1193. 3 DoujinshiDB
  1194. 4 pixiv Images
  1195. 5 Anime
  1196. 6 Nico Nico Seiga
  1197. 7 Danbooru
  1198. 8 drawr Images
  1199. 9 Nijie Images
  1200. 10 yande.re
  1201. TBA...
  1202.  
  1203.  
  1204.  
  1205. https://greasyfork.org/en/forum/discussion/comment/12162/#Comment_12162
  1206. image search on "ascii2d.net"?
  1207.  
  1208.  
  1209. GM_xmlhttpRequest({
  1210. method: 'POST',
  1211. url: 'http://ascii2d.net/imagesearch/search',
  1212. data: 'uri=https://i.imgur.com/RqikjW1.jpg',
  1213. headers: {'Content-Type': 'multipart/form-data'},
  1214. onload: function(r) {
  1215. //parse r.responseText
  1216. }
  1217. });
  1218.  
  1219.  
  1220. ********************************************************************************************/