[TS] Generic Image Viewer

A more Powerful Image Viewer with info panel support for Pixiv, deviantArt, imgur, Seiga Nico and nijie.info. NEW: Image rotation and flip added.

当前为 2016-05-27 提交的版本,查看 最新版本

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