Picviewer CE+

在线看图工具,支持图片翻转、旋转、缩放、弹出大图、批量保存

当前为 2022-03-13 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name Picviewer CE+
  3. // @name:zh-CN Picviewer CE+
  4. // @name:zh-TW Picviewer CE+
  5. // @name:pt-BR Picviewer CE+
  6. // @name:ru Picviewer CE+
  7. // @author NLF && ywzhaiqi && hoothin
  8. // @description Powerful picture viewing tool online, which can popup/scale/rotate/batch save pictures automatically
  9. // @description:zh-CN 在线看图工具,支持图片翻转、旋转、缩放、弹出大图、批量保存
  10. // @description:zh-TW 線上看圖工具,支援圖片翻轉、旋轉、縮放、彈出大圖、批量儲存
  11. // @description:pt-BR Poderosa ferramenta de visualização de imagens on-line, que pode pop-up/dimensionar/girar/salvar em lote imagens automaticamente
  12. // @description:ru Мощный онлайн-инструмент для просмотра изображений, который может автоматически отображать/масштабировать/вращать/пакетно сохранять изображения
  13. // @version 2022.3.13.1
  14. // @created 2011-6-15
  15. // @namespace https://github.com/hoothin/UserScripts
  16. // @homepage http://hoothin.com
  17. // @connect www.google.com
  18. // @connect www.google.com.hk
  19. // @connect www.google.co.jp
  20. // @connect ipv4.google.com
  21. // @connect image.baidu.com
  22. // @connect www.tineye.com
  23. // @grant GM_getValue
  24. // @grant GM_setValue
  25. // @grant GM_addStyle
  26. // @grant GM_openInTab
  27. // @grant GM_setClipboard
  28. // @grant GM_xmlhttpRequest
  29. // @grant GM_registerMenuCommand
  30. // @grant GM_notification
  31. // @grant GM_download
  32. // @grant GM.getValue
  33. // @grant GM.setValue
  34. // @grant GM.addStyle
  35. // @grant GM.openInTab
  36. // @grant GM.setClipboard
  37. // @grant GM.xmlHttpRequest
  38. // @grant GM.registerMenuCommand
  39. // @grant GM.notification
  40. // @grant unsafeWindow
  41. // @require https://cdnjs.cloudflare.com/ajax/libs/FileSaver.js/2.0.2/FileSaver.min.js
  42. // @require https://cdnjs.cloudflare.com/ajax/libs/jszip/3.7.1/jszip.js
  43. // @require https://greasyfork.org/scripts/6158-gm-config-cn/code/GM_config%20CN.js?version=23710
  44. // @require https://greasyfork.org/scripts/438080-pvcep-rules/code/pvcep_rules.js?version=1023785
  45. // @require https://greasyfork.org/scripts/440698-pvcep-lang/code/pvcep_lang.js?version=1027618
  46. // @contributionURL https://www.paypal.com/cgi-bin/webscr?cmd=_donations&business=rixixi@sina.com&item_name=Greasy+Fork+donation
  47. // @contributionAmount 1
  48. // @include http://*
  49. // @include https://*
  50. // @include ftp://*
  51. // @exclude http://www.toodledo.com/tasks/*
  52. // @exclude http*://maps.google.com*/*
  53. // @exclude *://www.google.*/_/chrome/newtab*
  54. // @exclude *://mega.*/*
  55. // @exclude *://*.mega.*/*
  56. // ==/UserScript==
  57.  
  58. ;(function(topObject,window,document,unsafeWindow){
  59. 'use strict';
  60.  
  61. var debug;
  62. var lang;
  63. function initLang(){
  64. let customLang=storage.getItem("customLang")||'auto';
  65. if(customLang=="auto")lang = navigator.appName=="Netscape"?navigator.language:navigator.userLanguage;
  66. else lang=customLang;
  67. setLang(lang);
  68. }
  69. function i18n(key,inserts){
  70. var result=i18nData[key],i;
  71. if(inserts){
  72. if(typeof inserts!="object")inserts=[inserts];
  73. for(i=0;i<inserts.length;i++){
  74. result=result.replace("#t#",inserts[i]);
  75. }
  76. }
  77. return result?result:key;
  78. }
  79. var defaultSearchData=`Google | https://www.google.com/searchbyimage?image_url=#t#
  80. Yandex | https://yandex.com/images/search?source=collections&rpt=imageview&url=#t#
  81. SauceNAO | https://saucenao.com/search.php?db=999&url=#t#
  82. IQDB | https://iqdb.org/?url=#t#
  83. 3D IQDB | https://3d.iqdb.org/?url=#t#
  84. Baidu | https://graph.baidu.com/details?isfromtusoupc=1&tn=pc&carousel=0&promotion_name=pc_image_shituindex&extUiData%5bisLogoShow%5d=1&image=#t#
  85. Bing | https://www.bing.com/images/search?view=detailv2&iss=sbi&form=SBIVSP&sbisrc=UrlPaste&q=imgurl:#t#
  86. TinEye | https://www.tineye.com/search?url=#t#
  87. Sogou | https://pic.sogou.com/ris?query=#t#
  88. 360 | http://st.so.com/stu?imgurl=#t#
  89. WhatAnime | https://trace.moe/?url=#t#
  90. Ascii2D | https://ascii2d.net/search/url/#t#
  91. Trace Moe | https://trace.moe/?url=#t#
  92. KarmaDecay | http://karmadecay.com/#t#
  93. ZXing QRCode | https://zxing.org/w/decode?full=true&u=#t#
  94. ImgOps | https://imgops.com/#b#`;
  95.  
  96. var _GM_openInTab,_GM_setClipboard,_GM_xmlhttpRequest,_GM_registerMenuCommand,_GM_notification;
  97. if(typeof GM_openInTab!='undefined'){
  98. _GM_openInTab=GM_openInTab;
  99. }else if(typeof GM!='undefined' && typeof GM.openInTab!='undefined'){
  100. _GM_openInTab=GM.openInTab;
  101. }else{
  102. _GM_openInTab=(s,t)=>{window.open(s)};
  103. }
  104. if(typeof GM_setClipboard!='undefined'){
  105. _GM_setClipboard=GM_setClipboard;
  106. }else if(typeof GM!='undefined' && typeof GM.setClipboard!='undefined'){
  107. _GM_setClipboard=GM.setClipboard;
  108. }else{
  109. _GM_setClipboard=(s)=>{};
  110. }
  111. if(typeof GM_xmlhttpRequest!='undefined'){
  112. _GM_xmlhttpRequest=GM_xmlhttpRequest;
  113. }else if(typeof GM!='undefined' && typeof GM.xmlHttpRequest!='undefined'){
  114. _GM_xmlhttpRequest=GM.xmlHttpRequest;
  115. }else{
  116. _GM_xmlhttpRequest=(f)=>{fetch(f.url).then(response=>response.text()).then(data=>{let res={response:data};f.onload(res)}).catch(f.onerror())};
  117. }
  118. if(typeof GM_registerMenuCommand!='undefined'){
  119. _GM_registerMenuCommand=GM_registerMenuCommand;
  120. }else if(typeof GM!='undefined' && typeof GM.registerMenuCommand!='undefined'){
  121. _GM_registerMenuCommand=GM.registerMenuCommand;
  122. }else{
  123. _GM_registerMenuCommand=(s,f)=>{};
  124. }
  125. if(typeof GM_notification!='undefined'){
  126. _GM_notification=GM_notification;
  127. }else if(typeof GM!='undefined' && typeof GM.notification!='undefined'){
  128. _GM_notification=GM.notification;
  129. }else{
  130. _GM_notification=(s)=>{alert(s)};
  131. }
  132. var _GM_download=(typeof GM_download=='undefined')?saveAs:GM_download;
  133. var prefs;
  134. var escapeHTMLPolicy;
  135. if (unsafeWindow.trustedTypes && unsafeWindow.trustedTypes.createPolicy) {
  136. escapeHTMLPolicy=unsafeWindow.trustedTypes.createPolicy('default', {
  137. createHTML: (string, sink) => string
  138. });
  139. }
  140.  
  141. function createHTML(html){
  142. return escapeHTMLPolicy?escapeHTMLPolicy.createHTML(html):html;
  143. }
  144. function init(topObject,window,document,arrayFn,envir,storage,unsafeWindow){
  145. // 默认设置,请到设置界面修改
  146. prefs={
  147. floatBar:{//浮动工具栏相关设置.
  148. butonOrder:['actual','current','gallery','magnifier'],//按钮排列顺序'actual'(实际的图片),'current'(当前显示的图片),'magnifier'(放大镜观察),'gallery'(图集),'search'(搜索原图)
  149. listenBg:true,//监听背景图
  150. showDelay:366,//浮动工具栏显示延时.单位(毫秒)
  151. hideDelay:566,//浮动工具栏隐藏延时.单位(毫秒)
  152. position:'top left',// 取值为: 'top left'(图片左上角) 或者 'top right'(图片右上角) 'bottom right'(图片右下角) 'bottom left'(图片左下角);
  153. offset:{//浮动工具栏偏移.单位(像素)
  154. x:-15,//x轴偏移(正值,向右偏移,负值向左)
  155. y:-15,//y轴偏移(正值,向下,负值向上)
  156. },
  157. forceShow:{//在没有被缩放的图片上,但是大小超过下面设定的尺寸时,强制显示浮动框.
  158. enabled:true,//启用强制显示.
  159. size:{//图片尺寸.单位(像素);
  160. w:45,
  161. h:45,
  162. },
  163. },
  164. minSizeLimit:{//就算是图片被缩放了(看到的图片被设定了width或者height限定了大小,这种情况下),如果图片显示大小小于设定值,那么也不显示浮动工具栏.
  165. w:15,
  166. h:15,
  167. },
  168. sizeLimitOr:false,
  169.  
  170. keys: {
  171. enable: true,
  172. actual: 'a', // 当出现悬浮条时按下 `a` 打开原图
  173. search: 's',
  174. current: 'c',
  175. magnifier: 'm',
  176. gallery: 'g',
  177. },
  178. globalkeys: {
  179. invertInitShow: false,
  180. ctrl: true,
  181. alt: false,
  182. shift: false,
  183. command: false,
  184. type: "hold"
  185. }
  186. },
  187.  
  188. magnifier:{//放大镜的设置.
  189. radius: 77,//默认半径.单位(像素).
  190. wheelZoom:{//滚轮缩放.
  191. enabled:true,
  192. pauseFirst:true,//需要暂停(单击暂停)后,才能缩放.(推荐,否则因为放大镜会跟着鼠标,如果放大镜过大,那么会影响滚动.)..
  193. range:[0.4,0.5,0.6,0.7,0.8,0.9,1,1.1,1.2,1.3,1.4,1.5,1.7,1.9,2,2.5,3.0,4.0],//缩放的范围
  194. },
  195. },
  196.  
  197. gallery:{//图库相关设定
  198. loadMore:false,
  199. loadAll:true,//加载更多时是否加载全部页面
  200. viewmoreEndless:true,
  201. fitToScreen:true,//图片适应屏幕(适应方式为contain,非cover).
  202. fitToScreenSmall:false,
  203. scrollEndToChange:true,
  204. exportType:'grid',
  205. sidebarPosition: 'bottom',//'top' 'right' 'bottom' 'left' 四个可能值
  206. sidebarSize: 120,//侧栏的高(如果是水平放置)或者宽(如果是垂直放置)
  207. sidebarToggle: true, // 是否显示隐藏按钮
  208. transition:true,//大图片区的动画。
  209. preload:true,//对附近的图片进行预读。
  210. max:5,//最多预读多少张(前后各多少张)
  211.  
  212. zoomresized: 25, // 图片尺寸最少相差比例,单位:%
  213. scaleSmallSize: 250, // 图库的新类别,缩放的图片,尺寸的高或宽都小于该值
  214. showSmallSize:true,//是否默认显示小尺寸图片
  215.  
  216. scrollEndAndLoad: false, // 滚动主窗口到最底部,然后自动重载库的图片。还有bug,有待进一步测试
  217. scrollEndAndLoad_num: 3, // 最后几张图片执行
  218.  
  219. autoZoom: true, // 如果有放大,则把图片及 sidebar 部分的缩放改回 100%,增大可视面积(仅在 chrome 下有效)
  220. descriptionLength: 32, // 注释的最大宽度
  221. editSite: "Lunapic",
  222. defaultSizeLimit:{
  223. w:200,
  224. h:200
  225. },
  226. searchData:defaultSearchData,
  227. downloadWithZip:false,
  228. autoOpenViewmore:false,
  229. viewmoreLayout:0
  230. },
  231.  
  232. imgWindow:{// 图片窗相关设置
  233. suitLongImg: true,
  234. fitToScreen: false,//适应屏幕,并且水平垂直居中(适应方式为contain,非cover).
  235. syncSelectedTool:true,//同步当前选择的工具,如果开了多个图片窗口,其中修改一个会反映到其他的上面。
  236. defaultTool:'hand',//"hand","rotate","zoom";打开窗口的时候默认选择的工具
  237. close:{//关闭的方式
  238. escKey:true,//按esc键
  239. dblClickImgWindow: true,//双击图片窗口
  240. clickOutside:'', // 点击图片外部关闭。值为''|'click'|'dblclick';无或点击或双击
  241. },
  242. overlayer:{// 覆盖层.
  243. shown:false,//显示
  244. color:'rgba(0,0,0,0.8)',//颜色和不透明度设置.
  245. },
  246. shiftRotateStep:15,// 旋转的时候,按住shift键时,旋转的步进.单位:度.
  247. zoom:{//滚轮缩放
  248. range:[0.1,0.2,0.3,0.4,0.5,0.6,0.7,0.8,0.9,1,1.1,1.2,1.3,1.4,1.5,1.7,1.9,2,2.5,3.0,4.0],//缩放比例.(不要出现负数,谢谢-_-!~)
  249. mouseWheelZoom:true,//是否允许使用滚轮缩放。
  250. },
  251. },
  252.  
  253. //等图片完全载入后,才开始执行弹出,放大等等操作,
  254. //按住ctrl键的时候,可以临时执行和这个设定相反的设定.
  255. waitImgLoad: false,
  256.  
  257. //框架里面的图片在顶层窗口展示出来,但是当frame与顶层窗口domain不一样的时候,可能导致图片被反盗链拦截,
  258. //按住shift键,可以临时执行和这个设定相反的设定
  259. framesPicOpenInTopWindow: true,
  260.  
  261. // lowLevel: true, // 如果有多个图片,优先选择低一级的
  262.  
  263. debug: false,
  264. customLang:'auto',
  265. customRules:`[
  266. //{
  267. //name: "Google picture",
  268. //url: /https?:\/\/www.google(\.\w{1,3}){1,3}\/search\?.*/,
  269. //getImage: function(a){
  270. //},
  271. //src: /avatar/i,
  272. //r: /\?.*$/i,
  273. //s: ''
  274. //}
  275. ]`,
  276. firstEngine:"Tineye"
  277. };
  278.  
  279. var tprules=[
  280. function(a) {
  281. var oldsrc = this.src,newsrc = this.src;
  282. if(this.getAttribute("_src") && !this.src){
  283. newsrc=this.getAttribute("_src");
  284. }else if(this.dataset && this.dataset.original){
  285. newsrc=this.dataset.original;
  286. }else if(this.dataset && this.dataset.src){
  287. newsrc=this.dataset.src;
  288. }else if(this._lazyrias && this._lazyrias.srcset){
  289. newsrc=this._lazyrias.srcset[this._lazyrias.srcset.length-1];
  290. }else if(this.dataset && this.dataset.origFile){
  291. newsrc=this.dataset.origFile;
  292. }else if(this.srcset){
  293. var srcs=this.srcset.split(","),largeSize=0;
  294. srcs.forEach(srci=>{
  295. let srcInfo=srci.trim().split(" "),curSize=parseInt(srcInfo[1]);
  296. if(srcInfo[1] && curSize>largeSize){
  297. largeSize=curSize;
  298. newsrc=srcInfo[0];
  299. }
  300. });
  301. }
  302. return oldsrc != newsrc ? newsrc : null;
  303. }
  304. ];
  305.  
  306. //图标
  307. prefs.icons={
  308. actual:'',
  309. current:'',
  310. magnifier:'',
  311. gallery:'',
  312. search:'',
  313.  
  314. retry:'',
  315. loading:'',
  316. loadingCancle:'',
  317.  
  318. hand:'',
  319. rotate:'',
  320. zoom:'',
  321. flipVertical:'',
  322. flipHorizontal:'',
  323. close:'',
  324. searchBtn:'',
  325. rotateIndicatorBG:'',
  326. rotateIndicatorPointer:'',
  327.  
  328. arrowTop:'',
  329. arrowBottom:'',
  330. arrowLeft:'',
  331. arrowRight:'',
  332.  
  333. fivePointedStar:'',
  334. lock:'',
  335. brokenImg:'',
  336. brokenImg_small:'',
  337. };
  338.  
  339. prefs.share={
  340. weibo:{
  341. disabled:false,
  342. name:'新浪微博',
  343. limitLang:'zh-CN',
  344. icon:'',
  345. api:function(args){
  346. var url='http://service.weibo.com/share/share.php?'+
  347. 'title='+args.title+
  348. '&url='+args.url+
  349. '&pic='+args.pic;
  350. return {
  351. url:url,
  352. wSize:{
  353. h:500,
  354. w:620,
  355. },
  356. };
  357. },
  358. },
  359. t:{
  360. name:'腾讯微博',
  361. limitLang:'zh-CN',
  362. icon:'',
  363. api:function(args){
  364. var url='http://v.t.qq.com/share/share.php?'+
  365. 'title='+args.title+
  366. '&url='+args.url+
  367. '&pic='+args.pic;
  368. return {
  369. url:url,
  370. wSize:{
  371. h:500,
  372. w:620,
  373. },
  374. };
  375. },
  376. },
  377. qZone:{
  378. name:'QQ空间',
  379. limitLang:'zh-CN',
  380. icon:'',
  381. api:function(args){
  382. var url='http://sns.qzone.qq.com/cgi-bin/qzshare/cgi_qzshare_onekey?'+
  383. 'title='+args.title+
  384. '&pics='+args.pic+
  385. '&url='+args.url;
  386. return {
  387. url:url,
  388. wSize:{
  389. h:650,
  390. w:620,
  391. },
  392. };
  393. },
  394. },
  395. fanfou:{
  396. name:'饭否',
  397. limitLang:'zh-CN',
  398. icon:'',
  399. api:function(args){
  400. var url='http://fanfou.com/sharer/image?'+
  401. 'u='+args.url+
  402. '&t='+args.title+
  403. '&img_src='+args.pic;
  404. return{
  405. url:url,
  406. wSize:{
  407. h:550,
  408. w:650,
  409. },
  410. };
  411. },
  412. },
  413. tieba:{
  414. name:'百度贴吧',
  415. limitLang:'zh-CN',
  416. icon:'',
  417. api:function(args){
  418. var url = 'http://tieba.baidu.com/f/commit/share/openShareApi?'+
  419. 'title='+args.title+
  420. '&url='+args.url+
  421. '&pic='+args.pic;
  422. return {
  423. url:url,
  424. wSize:{
  425. h:600,
  426. w:630,
  427. },
  428. };
  429. },
  430. },
  431. renren:{
  432. name:'人人网',
  433. limitLang:'zh-CN',
  434. icon:'',
  435. api:function(args){
  436. var url='http://widget.renren.com/dialog/share?'+
  437. 'link='+args.url+
  438. '&title='+args.title+
  439. '&pic='+args.pic;
  440. return {
  441. url:url,
  442. wSize:{
  443. h:600,
  444. w:650,
  445. },
  446. };
  447. },
  448. },
  449. douban:{
  450. name:'豆瓣',
  451. limitLang:'zh-CN',
  452. icon:'',
  453. api:function(args){
  454. var url='http://shuo.douban.com/%21service/share?'+
  455. 'href='+args.url+
  456. '&name='+args.title+
  457. '&image='+args.pic;
  458. return {
  459. url:url,
  460. wSize:{
  461. h:350,
  462. w:600,
  463. },
  464. };
  465. },
  466. },
  467. facebook:{
  468. name:'Facebook',
  469. icon:'',
  470. api:function(args){
  471. var url='https://www.facebook.com/sharer/sharer.php?'+
  472. 'u='+encodeURIComponent(args.url)+
  473. '&t='+args.title;
  474. return {
  475. url:url,
  476. wSize:{
  477. h:436,
  478. w:626,
  479. },
  480. };
  481. },
  482. },
  483. twitter:{
  484. name:'Twitter',
  485. icon:'',
  486. api:function(args){
  487. var url='https://twitter.com/intent/tweet?'+
  488. 'url='+args.url+
  489. '&text='+args.title;
  490. return {
  491. url:url,
  492. wSize:{
  493. h:350,
  494. w:600,
  495. },
  496. };
  497. },
  498. },
  499. pinterest:{
  500. name:'Pinterest',
  501. icon:'',
  502. api:function(args){
  503. var url='https://pinterest.com/pin/create/button/?'+
  504. 'url='+args.url+
  505. '&media='+encodeURIComponent(args.pic)+
  506. '&description='+encodeURIComponent(args.title);
  507. return {
  508. url:url,
  509. wSize:{
  510. h:350,
  511. w:600,
  512. },
  513. };
  514. },
  515. }
  516. };
  517.  
  518. if (typeof String.prototype.startsWith != 'function') {
  519. String.prototype.startsWith = function(str) {
  520. return this.slice(0, str.length) == str;
  521. };
  522. }
  523.  
  524. if (typeof String.prototype.visualLength != 'function') {
  525. var rulerEle = document.createElement("span");
  526. rulerEle.style.visibility = "hidden";
  527. rulerEle.style.whiteSpace = "nowrap";
  528. String.prototype.visualLength = function(size,family){
  529. rulerEle.style.fontSize = size || "inherit";
  530. rulerEle.style.fontFamily = family || "inherit";
  531. rulerEle.innerText = this;
  532. document.body.appendChild(rulerEle);
  533. let w = rulerEle.offsetWidth;
  534. document.body.removeChild(rulerEle);
  535. return w;
  536. }
  537. }
  538.  
  539. function getMStr(func) {
  540. var lines = func.toString();
  541. lines = lines.substring(lines.indexOf("/*") + 3, lines.lastIndexOf("*/"));
  542. return lines;
  543. }
  544.  
  545. function toRE(obj, flag) {
  546. if (!obj) {
  547. return obj;
  548. } else if (obj instanceof RegExp) {
  549. return obj;
  550. } else if (flag) {
  551. return new RegExp(obj, flag);
  552. } else if (obj instanceof Array) {
  553. return new RegExp(obj[0], obj[1]);
  554. } else if (typeof obj === 'string') {
  555. if (obj.indexOf('*') != -1 && obj.indexOf('.*') == -1) {
  556. obj = wildcardToRegExpStr(obj);
  557. }
  558. return new RegExp(obj);
  559. }
  560. }
  561.  
  562. function wildcardToRegExpStr(urlstr) {
  563. if (urlstr.source) return urlstr.source;
  564. var reg = urlstr.replace(/[()\[\]{}|+.,^$?\\]/g, "\\$&").replace(/\*+/g, function(str){
  565. return str === "*" ? ".*" : "[^/]*";
  566. });
  567. return "^" + reg + "$";
  568. }
  569.  
  570. function isXPath(xpath) {
  571. return xpath.startsWith('./') || xpath.startsWith('//') || xpath.startsWith('id(');
  572. }
  573.  
  574. function getElementMix(selector, contextNode, doc) {
  575. var ret;
  576. if (!selector || !contextNode) return ret;
  577. doc = doc || document;
  578.  
  579. var type = typeof selector;
  580. if (type == 'string') {
  581. if (isXPath(selector)) {
  582. ret = getElementByXpath(selector, contextNode, doc);
  583. } else {
  584. ret = contextNode.parentNode.querySelector(selector);
  585. }
  586. } else if (type == 'function') {
  587. ret = selector(contextNode, doc);
  588. }
  589. return ret;
  590. }
  591.  
  592. function launchFullScreen(element) {
  593. if (element.requestFullscreen) {
  594. element.requestFullscreen();
  595. } else if (element.msRequestFullscreen) {
  596. element.msRequestFullscreen();
  597. } else if (element.mozRequestFullScreen) {
  598. element.mozRequestFullScreen();
  599. } else if (element.webkitRequestFullscreen) {
  600. element.webkitRequestFullscreen(Element.ALLOW_KEYBOARD_INPUT);
  601. }
  602. }
  603.  
  604. function cancelFullScreen() {
  605. if (document.exitFullscreen) {
  606. document.exitFullscreen();
  607. } else if (document.msExitFullscreen) {
  608. document.msExitFullscreen();
  609. } else if (document.mozCancelFullScreen) {
  610. document.mozCancelFullScreen();
  611. } else if (document.webkitExitFullscreen) {
  612. document.webkitExitFullscreen();
  613. }
  614. }
  615.  
  616. // 检测缩放
  617. function detectZoom (){
  618. var ratio = 0,
  619. screen = window.screen,
  620. ua = navigator.userAgent.toLowerCase();
  621.  
  622. if (window.devicePixelRatio !== undefined) {
  623. ratio = window.devicePixelRatio;
  624. }
  625. else if (~ua.indexOf('msie')) {
  626. if (screen.deviceXDPI && screen.logicalXDPI) {
  627. ratio = screen.deviceXDPI / screen.logicalXDPI;
  628. }
  629. }
  630. else if (window.outerWidth !== undefined && window.innerWidth !== undefined) {
  631. ratio = window.outerWidth / window.innerWidth;
  632. }
  633.  
  634. if (ratio){
  635. ratio = Math.round(ratio * 100);
  636. }
  637.  
  638. return ratio;
  639. }
  640.  
  641. //获取位置
  642. function getContentClientRect(target){
  643. var rect=target.getBoundingClientRect();
  644. var compStyle=unsafeWindow.getComputedStyle(target);
  645. var pFloat=parseFloat;
  646. var top=rect.top + pFloat(compStyle.paddingTop) + pFloat(compStyle.borderTopWidth);
  647. var right=rect.right - pFloat(compStyle.paddingRight) - pFloat(compStyle.borderRightWidth);
  648. var bottom=rect.bottom - pFloat(compStyle.paddingBottom) - pFloat(compStyle.borderBottomWidth);
  649. var left=rect.left + pFloat(compStyle.paddingLeft) + pFloat(compStyle.borderLeftWidth);
  650. return {
  651. top:top,
  652. right:right,
  653. bottom:bottom,
  654. left:left,
  655. width:right-left,
  656. height:bottom-top,
  657. };
  658. };
  659.  
  660. //获取窗口大小.
  661. function getWindowSize(){
  662. /*
  663. //包含滚动条
  664. return {
  665. h:window.innerHeight,
  666. w:window.innerWidth,
  667. };
  668. */
  669.  
  670. //去除滚动条的窗口大小
  671. var de=document.documentElement;
  672. var body=document.body;
  673. var backCompat=document.compatMode=='BackCompat';
  674. return {
  675. h:backCompat? body.clientHeight : de.clientHeight,
  676. w:backCompat? body.clientWidth : de.clientWidth,
  677. };
  678.  
  679. };
  680.  
  681. //获取已滚动的距离
  682. function getScrolled(container){
  683. if(container){
  684. return {
  685. x:container.scrollLeft,
  686. y:container.scrollTop,
  687. };
  688. };
  689. return {
  690. x:'scrollX' in window ? window.scrollX : ('pageXOffset' in window ? window.pageXOffset : document.documentElement.scrollLeft || document.body.scrollLeft),
  691. y:'scrollY' in window ? window.scrollY : ('pageYOffset' in window ? window.pageYOffset : document.documentElement.scrollTop || document.body.scrollTop),
  692. };
  693. };
  694.  
  695. //xpath 获取单个元素
  696. function getElementByXpath(xpath,contextNode,doc){
  697. doc=doc || document;
  698. contextNode=contextNode || doc;
  699. return doc.evaluate(xpath,contextNode,null,9,null).singleNodeValue;
  700. };
  701.  
  702.  
  703. //事件支持检测.
  704. function eventSupported( eventName,elem ){
  705. elem = elem || document.createElement("div");
  706. eventName = "on" + eventName;
  707. var isSupported = (eventName in elem);
  708. if (!isSupported){
  709. if(!elem.setAttribute){//setAttribute是元素节点的方法
  710. elem=document.createElement("div");
  711. };
  712. var setAttr;
  713. if(!elem.hasAttribute(eventName)){
  714. setAttr=true;
  715. elem.setAttribute(eventName, "return;");
  716. };
  717. isSupported = typeof elem[eventName] == "function";
  718. if(setAttr)elem.removeAttribute(eventName);
  719. };
  720. return isSupported;
  721. };
  722.  
  723.  
  724. //检测属性支持.dom属性
  725. //返回带前缀的可以直接执行是属性
  726. function proSupported(proName,elem){
  727. //判断第一个字母是否大写,如果是的话,为构造函数,前缀也要大写
  728. var prefix=/^[A-Z]/.test(proName)? ['','WebKit-','O-','Moz-','MS-'] : ['','webkit-','o-','moz-','ms-'];
  729. var i=0;
  730. var p_i;
  731. var sProName;
  732. elem = elem || document.createElement("div");
  733. while(typeof (p_i=prefix[i++])!='undefined'){
  734. sProName=(p_i+proName).replace(/-([A-z])/g,function(a,b){
  735. return b.toUpperCase();
  736. });
  737. if(sProName in elem)return sProName;
  738. };
  739. };
  740.  
  741.  
  742. //css属性支持
  743. //带前缀的默认为大写(所有浏览器支持)
  744. //比如WebkitTransform,MozTransform,OTransfomr
  745. //chrome浏览器大小写前缀都行。
  746. //firefox,opera只能大写
  747. //ie 9+只能小写
  748. function cssProSupported(proName,elem,capitalize){
  749. if(capitalize!==false)capitalize=true;
  750. proName=proName.toLowerCase();
  751.  
  752. var prefix=['','-webkit-','-o-','-moz-','-ms-'];
  753. elem=elem || document.createElement('div');
  754. var style=elem.style;
  755. var camelPro;
  756.  
  757. // 会有个错误 invalid 'in' operand style
  758. try {
  759. for(var i=0,ii=prefix.length;i<ii;i++){
  760. var first=true;
  761. camelPro=(prefix[i]+proName).replace(/-([a-z])/g,function(a,b){
  762. b=b.toUpperCase();
  763. if(first){
  764. first=false;
  765. if(!capitalize){
  766. b=b.toLowerCase();
  767. };
  768. };
  769. return b;
  770. });
  771. if(camelPro in style){
  772. return camelPro;
  773. }
  774. }
  775. } catch(ex) {}
  776.  
  777. if(!capitalize)return;
  778. return cssProSupported(proName,elem,false);
  779.  
  780. };
  781.  
  782. //css属性值支持
  783. function cssValueSupported(proName,value,elem){
  784. var prefix=['','-webkit-','-o-','-moz-','-ms-'];
  785. elem=elem || document.createElement('div');
  786. var style=elem.style;
  787. if(!style)return false;
  788. var prefixedValue;
  789. for(var i=0,ii=prefix.length;i<ii;i++){
  790. prefixedValue=prefix[i] + value;
  791. style[proName]=prefixedValue;
  792. if(style[proName]==prefixedValue){
  793. return prefixedValue;
  794. };
  795. };
  796. };
  797.  
  798.  
  799. //elem.dataset的兼容实现
  800. //ie不支持;firefoxGM储存不能反映到元素属性上。
  801. function dataset(elem,pro,value){
  802.  
  803. function getDataPrefix(){
  804. return 'data-' + pro.replace(/[A-Z]/g,function(m){
  805. return '-' + m.toLowerCase();
  806. });
  807. };
  808.  
  809. if(typeof value=='undefined'){//取值
  810. if(elem.dataset){
  811. value = elem.dataset[pro];
  812. }else{//没有取到值,返回undefined,getAttribute默认是返回null,所以判断一下。
  813. var prefixedPro=getDataPrefix();
  814. if(elem.hasAttribute(prefixedPro)){
  815. value=elem.getAttribute(prefixedPro);
  816. };
  817. };
  818. return value;
  819. }else{
  820. elem.setAttribute(getDataPrefix(),value);
  821. };
  822. };
  823.  
  824.  
  825. //重新检查悬浮图片
  826. function imgReHover(img){
  827. //要检查的图片,是当前悬浮的。
  828. if(!floatBar.shown || floatBar.data.img != img)return;
  829.  
  830. var mHover=document.createEvent('MouseEvent');
  831. var cr=img.getBoundingClientRect();
  832. mHover.initMouseEvent('mouseover',true,true,window,0, cr.left + 10, cr.top + 10, cr.left + 10, cr.top + 10, false,false,false,false, 0,null);
  833. img.dispatchEvent(mHover);
  834. };
  835.  
  836. // 获取真正的unsafeWindow,chrome里面也能访问到真实环境的变量
  837. // 在 chrome 37 测试无效
  838. if(!envir.firefox && !envir.opera && !envir.ie && !storage.supportGM){
  839. ;(function(){
  840. document.addEventListener('picViewer-return-unsafeWindow',function(e){
  841. unsafeWindow = e.detail;
  842. // alert(unsafeWindow.$);
  843. },true);
  844.  
  845. //页面脚本
  846. var s=document.createElement('script');
  847. s.textContent='(' + (function(){
  848. var cusEvent=document.createEvent('CustomEvent');
  849. cusEvent.initCustomEvent('picViewer-return-unsafeWindow',false,false,window);
  850. document.dispatchEvent(cusEvent);
  851. }).toString() +')()';
  852. document.head.appendChild(s);
  853. })();
  854. }
  855.  
  856.  
  857. //抛出错误到错误控制台
  858. function throwErrorInfo(err){
  859. if(console && console.error){
  860. console.error(err.message + '\n\n' + (err.stacktrace? err.stacktrace : '') + '\n\n' , err);
  861. };
  862. };
  863.  
  864. //对象克隆
  865. function cloneObject(obj,deep){
  866. var obj_i;
  867. var ret=Array.isArray(obj)? [] : {};
  868. for(var i in obj){
  869. if(!obj.hasOwnProperty(i))continue;
  870. obj_i=obj[i];
  871. if(!deep || typeof obj_i!='object' || obj_i===null || obj_i.nodeType){
  872. ret[i]=obj_i;
  873. }else{
  874. ret[i]=cloneObject(obj_i,deep);
  875. };
  876. };
  877. return ret;
  878. };
  879.  
  880. //闪烁元素。
  881. function flashEle(ele,duration){
  882. if(dataset(ele,'pvFlashing'))return;
  883. if(ele.offsetHeight==0)return;
  884. dataset(ele,'pvFlashing','1');
  885.  
  886. var oOutline=ele.style.outline;
  887. var oOutlineOffset=ele.style.outlineOffset;
  888. var oOpacity=ele.style.opacity;
  889. var oTransform=ele.style[support.cssTransform];
  890.  
  891. var count=0;
  892. var startTime=Date.now();
  893. duration=duration? duration : 1200;
  894.  
  895. var flashInterval=setInterval(function(){
  896. var outline='none',
  897. outlineOffset=0,
  898. opacity=0.3,
  899. transform='';
  900.  
  901. if(count % 2 == 0){
  902. outline='5px dashed rgba(255,0,0,0.95)';
  903. opacity=0.95;
  904. outlineOffset='1px';
  905. transform='scale(1.1)';
  906. }else{
  907. if((Date.now() - startTime) > duration){
  908. clearInterval(flashInterval);
  909. outline=oOutline;
  910. opacity=oOpacity;
  911. outlineOffset=oOutlineOffset;
  912. transform=oTransform;
  913. ele.removeAttribute('data-pv-flashing');
  914. };
  915. };
  916.  
  917. ele.style.outline=outline;
  918. ele.style.outlineOffset=outlineOffset;
  919. ele.style.opacity=opacity;
  920. ele.style[support.cssTransform]=transform;
  921.  
  922. count++;
  923. },80);
  924. };
  925.  
  926. //支持情况.
  927. var support={
  928. cssTransform:cssProSupported('transform'),
  929. cssCursorValue:{
  930. zoomIn:cssValueSupported('cursor','zoom-in'),
  931. zoomOut:cssValueSupported('cursor','zoom-out'),
  932. grab:cssValueSupported('cursor','grab'),
  933. grabbing:cssValueSupported('cursor','grabbing'),
  934. },
  935. };
  936.  
  937.  
  938.  
  939. //动画算法
  940. /*
  941. t: current time(当前时间);
  942. b: beginning value(初始值);
  943. c: change in value(变化量);
  944. d: duration(持续时间)。
  945. */
  946.  
  947. var Tween = {
  948. Cubic: {
  949. easeInOut:function(t,b,c,d){
  950. return -c/2*(Math.cos(Math.PI*t/d)-1)+b
  951. }
  952. },
  953. };
  954.  
  955. //imgReady
  956. var imgReady=(function(){
  957. var iRInterval,
  958. iRReadyFn=[],
  959. isrcs=[]
  960. ;
  961.  
  962. var timeLimit=3 * 60 * 1000;//3分钟
  963.  
  964. function checkReady(){
  965. var now= Date.now();
  966. for(var i=0,ii=iRReadyFn.length,iRReadyFn_i;i<ii;i++){
  967. iRReadyFn_i=iRReadyFn[i];
  968. //now - iRReadyFn_i.startTime >= timeLimit ||
  969. if(iRReadyFn_i()){
  970. iRReadyFn.splice(i,1);
  971. isrcs.splice(i,1);
  972. i--;
  973. ii--;
  974. };
  975. };
  976. if(iRReadyFn.length==0){
  977. clearInterval(iRInterval);
  978. iRInterval=null;
  979. };
  980. };
  981.  
  982.  
  983.  
  984. var imgReady=function(img,opts){
  985.  
  986. if(/NodeList|HTMLCollection/.test(Object.prototype.toString.call(img)) || Array.isArray(img)){
  987. arrayFn.forEach.call(img,function(img,index,array){
  988. if(img instanceof HTMLImageElement){
  989. imgReady(img,opts);
  990. };
  991. });
  992. return;
  993. };
  994.  
  995. if(!(img instanceof HTMLImageElement)){
  996. var t_img=document.createElement('img');
  997. t_img.src=img;
  998. img=t_img;
  999. t_img=null;
  1000. };
  1001.  
  1002. var ready,load,error,loadEnd,abort,timeout,time;
  1003. ready=opts.ready;
  1004. load=opts.load;
  1005. error=opts.error;
  1006. loadEnd=opts.loadEnd;
  1007. abort=opts.abort;
  1008. timeout=opts.timeout;
  1009. time=typeof opts.time=='number'? opts.time : 0;
  1010.  
  1011. if(time){
  1012. setTimeout(function(){
  1013. if(!loadEndDone){
  1014. aborted=true;
  1015. removeListener();
  1016. img.src= prefs.icons.brokenImg_small;
  1017. if(timeout){
  1018. timeout.call(img,{
  1019. target:img,
  1020. type:'timeout',
  1021. });
  1022. };
  1023. loadEndDone=true;
  1024. if(loadEnd){
  1025. loadEnd.call(img,{
  1026. target:img,
  1027. type:'timeout',
  1028. });
  1029. };
  1030.  
  1031. };
  1032. },time);
  1033. };
  1034.  
  1035. var src=img.src;
  1036. var loadEndDone;
  1037.  
  1038. function go(type,e){
  1039. switch(type){
  1040. case 'load':{
  1041. removeListener();
  1042. go('ready');//如果直接触发load,那么先触发ready
  1043. if(load){
  1044. load.call(img,e);
  1045. };
  1046.  
  1047. if(!loadEndDone){
  1048. loadEndDone=true;
  1049. if(loadEnd){
  1050. loadEnd.call(img,e);
  1051. };
  1052. };
  1053. }break;
  1054. case 'ready':{
  1055. if(!ready || readyHandler.done)return;
  1056. readyHandler.done=true;
  1057. ready.call(img,{
  1058. target:img,
  1059. type:'ready',
  1060. });
  1061. }break;
  1062. case 'error':{
  1063. removeListener();
  1064. if(error){
  1065. error.call(img,e);
  1066. };
  1067. if(!loadEndDone){
  1068. loadEndDone=true;
  1069. if(loadEnd){
  1070. loadEnd.call(img,e);
  1071. };
  1072. };
  1073. }break;
  1074. };
  1075. };
  1076.  
  1077. var aborted;
  1078. var ret={
  1079. img:img,
  1080. abort:function(){
  1081. if(!loadEndDone){
  1082. aborted=true;
  1083. removeListener();
  1084. img.src= prefs.icons.brokenImg_small;
  1085. if(abort){
  1086. abort.call(img,{
  1087. target:img,
  1088. type:'abort',
  1089. });
  1090. };
  1091. loadEndDone=true;
  1092. if(loadEnd){
  1093. loadEnd.call(img,{
  1094. target:img,
  1095. type:'abort',
  1096. });
  1097. };
  1098. };
  1099. },
  1100. };
  1101.  
  1102. function readyHandler(){//尽快的检测图片大小.
  1103. if(loadEndDone || aborted)return true;
  1104. if(img.naturalWidth==0 || img.naturalHeight==0)return;
  1105. go('ready');
  1106. return true;
  1107. };
  1108.  
  1109.  
  1110. function loadHandler(e){
  1111. go('load',e);
  1112. };
  1113.  
  1114. function errorHandler(e){
  1115. go('error',e);
  1116. };
  1117.  
  1118. function removeListener(){
  1119. img.removeEventListener('load',loadHandler,true);
  1120. img.removeEventListener('error',errorHandler,true);
  1121. };
  1122.  
  1123. //ready必须在load之前触发。
  1124.  
  1125. if(img.complete){//图片已经加载完成.
  1126. if(typeof img.width=='number' && img.width && img.height){//图片
  1127. setTimeout(function(){
  1128. if(aborted)return;
  1129. go('load',{
  1130. type:'load',
  1131. target:img,
  1132. });
  1133. },0);
  1134. }else{//这不是图片.opera会识别错误.
  1135. setTimeout(function(){
  1136. if(aborted)return;
  1137. go('error',{
  1138. type:'error',
  1139. target:img,
  1140. });
  1141. },0);
  1142. };
  1143. return ret;
  1144. };
  1145.  
  1146.  
  1147. img.addEventListener('error',errorHandler,true);
  1148. img.addEventListener('load',loadHandler,true);
  1149.  
  1150.  
  1151. if(ready){
  1152. var index=isrcs.indexOf(src);
  1153. if(index==-1){
  1154. isrcs.push(src);
  1155. readyHandler.startTime= Date.now();
  1156. iRReadyFn.push(readyHandler);
  1157. }else{
  1158. iRReadyFn[index].startTime= Date.now();
  1159. };
  1160.  
  1161. if(!iRInterval){
  1162. iRInterval=setInterval(checkReady,66);
  1163. };
  1164. };
  1165.  
  1166. return ret;
  1167. };
  1168.  
  1169. return imgReady;
  1170. })();
  1171.  
  1172.  
  1173. var addWheelEvent=(function(){
  1174.  
  1175. function getSupportEventName(){
  1176. var ret='DOMMouseScroll';
  1177. if(eventSupported('wheel')){//w3c FF>=17 ie>=9
  1178. ret='wheel';
  1179. }else if(eventSupported('mousewheel')){//opera,chrome
  1180. ret='mousewheel';
  1181. };
  1182. return ret;
  1183. };
  1184.  
  1185. var eventName;
  1186.  
  1187. return function(ele,callback,useCapture){
  1188. if(!eventName){
  1189. eventName=getSupportEventName();
  1190. };
  1191.  
  1192. ele.addEventListener(eventName,function(e){
  1193. var type=e.type;
  1194. var ne;
  1195. if(type!='wheel'){
  1196. ne={};
  1197. for(var i in e){
  1198. ne[i]=e[i];
  1199. };
  1200.  
  1201. ne.type='wheel';
  1202. ne.deltaX=0;
  1203. ne.deltaY=0;
  1204. ne.deltaZ=0;
  1205. ne.deltaMode=1;//line
  1206. ne.preventDefault=e.preventDefault.bind(e);
  1207. ne.stopPropagation=e.stopPropagation.bind(e);
  1208.  
  1209. var x=0,y=0;
  1210. if(typeof e.axis=='number'){//DOMMouseScroll
  1211. if(e.axis==2){
  1212. y=e.detail;
  1213. }else{
  1214. x=e.detail;
  1215. };
  1216. }else{
  1217. //opera早起版本的mousewheel只支持y轴的滚动,e.wheelDeltaY undefined
  1218. if(typeof e.wheelDeltaY=='undefined' || e.wheelDeltaY!=0){
  1219. y=-e.wheelDelta/40;
  1220. }else{
  1221. x=-e.wheelDelta/40;
  1222. };
  1223. };
  1224. ne.deltaY =y;
  1225. ne.deltaX =x;
  1226.  
  1227. };
  1228.  
  1229. callback.call(this,ne? ne : e);
  1230. },useCapture || false);
  1231. };
  1232. })();
  1233.  
  1234.  
  1235. var addCusMouseEvent=(function(){
  1236.  
  1237. function getSupported(){
  1238. return {
  1239. mouseleave:eventSupported('mouseleave'),
  1240. mouseenter:eventSupported('mouseenter'),
  1241. };
  1242. };
  1243.  
  1244. var support;
  1245. var map={
  1246. mouseleave:'mouseout',
  1247. mouseenter:'mouseover',
  1248. };
  1249.  
  1250. return function(type, ele, fn){//事件类型,元素,监听函数
  1251. if(!support){
  1252. support=getSupported();
  1253. };
  1254.  
  1255. // chrome 30+ 虽然支持 mouseenter,但是存在问题
  1256. if(support[type] && !(type == 'mouseenter' && window.chrome)){
  1257. ele.addEventListener(type,fn,false);//mouseleave,enter不冒泡
  1258. }else{
  1259. ele.addEventListener(map[type],function(e){
  1260. var relatedTarget=e.relatedTarget;//mouseout,去往的元素;mouseover,来自的元素
  1261. if(!this.contains(relatedTarget)){
  1262. fn.call(this,e);
  1263. };
  1264. },true);
  1265. };
  1266. };
  1267.  
  1268. })();
  1269.  
  1270.  
  1271. //库
  1272. function GalleryC(){
  1273. this.init();
  1274. };
  1275.  
  1276. var gallery;
  1277. var galleryMode;
  1278.  
  1279. GalleryC.prototype={
  1280. init:function(){
  1281. this.addStyle();
  1282. var container=document.createElement('span');
  1283.  
  1284. this.gallery=container;
  1285. container.className='pv-gallery-container';
  1286. container.tabIndex=1;//为了获取焦点,来截获键盘事件
  1287. container.innerHTML=createHTML(
  1288. '<span class="pv-gallery-head">'+
  1289. '<span class="pv-gallery-head-float-left">'+
  1290. '<span title="'+i18n("picInfo")+'" class="pv-gallery-head-left-img-info">'+
  1291. '<span class="pv-gallery-head-left-img-info-resolution" title="'+i18n("resolution")+'">0 x 0</span>'+
  1292. '<span class="pv-gallery-head-left-img-info-count" title="'+i18n("picNum")+'">(1 / 1)</span>'+
  1293. '<span class="pv-gallery-head-left-img-info-scaling" title="'+i18n("scaleRatio")+'">(100%)</span>'+
  1294. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1295. '<span class="pv-gallery-head-left-img-info-description" title="'+i18n("picNote")+'"></span>'+
  1296. '<div class="pv-gallery-range-box"><input type="range" id="minsizeW" min="0" max="100" value="0" title="Width"> <span id="minsizeWSpan">0px</span> '+
  1297. '<input type="range" id="minsizeH" min="0" max="100" value="0" title="Height"> <span id="minsizeHSpan">0px</span></div>'+
  1298. '<span class="pv-gallery-head-left-lock-icon" title="'+i18n("lockSizeTip")+'"></span>'+
  1299. '</span>'+
  1300. '</span>'+
  1301.  
  1302. '<span title="'+i18n("exitCollectionTip")+'" class="pv-gallery-head-command pv-gallery-head-command-exit-collection">'+
  1303. '<span>'+i18n("exitCollection")+'</span>'+
  1304. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1305. '</span>'+
  1306.  
  1307. '<span title="'+i18n("loadAllTip")+'" class="pv-gallery-head-command pv-gallery-head-command-nextPage">'+
  1308. '<span>'+i18n("loadAll")+'</span>'+
  1309. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1310. '</span>'+
  1311.  
  1312. '<span title="'+i18n("urlFilterTip")+'" class="pv-gallery-head-command pv-gallery-head-command-urlFilter">'+
  1313. '<span>'+i18n("urlFilter")+'</span>'+
  1314. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1315. '</span>'+
  1316.  
  1317. '<span title="'+i18n("fiddleTip")+'" class="pv-gallery-head-command pv-gallery-head-command-operate">'+
  1318. '<span>'+i18n("fiddle")+'</span>'+
  1319. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1320. '</span>'+
  1321.  
  1322. '<span class="pv-gallery-head-command-container">'+
  1323. '<span class="pv-gallery-head-command pv-gallery-head-command-collect">'+
  1324. '<span class="pv-gallery-head-command-collect-icon"></span>'+
  1325. '<span class="pv-gallery-head-command-collect-text"></span>'+
  1326. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1327. '</span>'+
  1328. '<span class="pv-gallery-head-command-drop-list pv-gallery-head-command-drop-list-collect">'+
  1329. '<span title="'+i18n("collectDetailTip")+'" class="pv-gallery-head-command-drop-list-item pv-gallery-head-command-drop-list-item-collect-description">'+
  1330. '<span>'+i18n("collectDetail")+':</span>'+
  1331. '<textarea data-prefs="description" cols="25" rows="5"></textarea>'+
  1332. '</span>'+
  1333. '</span>'+
  1334. '</span>'+
  1335.  
  1336. '<span class="pv-gallery-head-command-container">'+
  1337. '<span title="'+i18n("playSlide")+'" class="pv-gallery-head-command pv-gallery-head-command-slide-show">'+
  1338. '<span class="pv-gallery-head-command_overlayer"></span>'+
  1339. '<span class="pv-gallery-head-command-slide-show-button">'+
  1340. '<span class="pv-gallery-head-command-slide-show-button-inner"></span>'+
  1341. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1342. '</span>'+
  1343. '<span class="pv-gallery-head-command-slide-show-countdown" title="'+i18n("countDown")+'"></span>'+
  1344. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1345. '</span>'+
  1346. '<span class="pv-gallery-head-command-drop-list pv-gallery-head-command-drop-list-slide-show">'+
  1347. '<span class="pv-gallery-head-command-drop-list-item" title="'+i18n("slideGapTip")+'">'+
  1348. '<input data-prefs="interval" step="1" min="1" type="number" value="5" />'+
  1349. '<span>'+i18n("slideGap")+'</span>'+
  1350. '</span>'+
  1351. '<span class="pv-gallery-head-command-drop-list-item" title="'+i18n("slideBackTip")+'">'+
  1352. '<input id="pv-gallery-head-command-drop-list-item-slide-show-backward" data-prefs="backward" type="checkbox" />'+
  1353. '<label for="pv-gallery-head-command-drop-list-item-slide-show-backward">'+i18n("slideBack")+'   </label>'+
  1354. '</span>'+
  1355. '<span class="pv-gallery-head-command-drop-list-item" title="'+i18n("slideWaitTip")+'">'+
  1356. '<input id="pv-gallery-head-command-drop-list-item-slide-show-wait" data-prefs="wait" type="checkbox" checked="checked" />'+
  1357. '<label for="pv-gallery-head-command-drop-list-item-slide-show-wait">'+i18n("slideWait")+'</label>'+
  1358. '</span>'+
  1359. '<span class="pv-gallery-head-command-drop-list-item" title="'+i18n("slideSkipErrorTip")+'">'+
  1360. '<input id="pv-gallery-head-command-drop-list-item-slide-show-skipErrorImg" data-prefs="skipErrorImg" type="checkbox" checked="checked" />'+
  1361. '<label for="pv-gallery-head-command-drop-list-item-slide-show-skipErrorImg">'+i18n("slideSkipError")+'</label>'+
  1362. '</span>'+
  1363. '</span>'+
  1364. '</span>'+
  1365.  
  1366. '<span class="pv-gallery-head-command-container">'+
  1367. '<span title="'+i18n("typeTip")+'" class="pv-gallery-head-command pv-gallery-head-command-category">'+
  1368. '<span>'+i18n("type")+'</span>'+
  1369. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1370. '</span>'+
  1371. '<span class="pv-gallery-head-command-drop-list pv-gallery-head-command-drop-list-category">'+
  1372. '</span>'+
  1373. '</span>'+
  1374.  
  1375. '<span class="pv-gallery-head-command-container">'+
  1376. '<span title="'+i18n("commandTip")+'" class="pv-gallery-head-command pv-gallery-head-command-others">'+
  1377. '<span>'+i18n("command")+'</span>'+
  1378. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1379. '</span>'+
  1380. '<span class="pv-gallery-head-command-drop-list pv-gallery-head-command-drop-list-others">'+
  1381. '<span class="pv-gallery-head-command-drop-list-item" data-command="psImage" title="'+i18n("onlineEditTip",prefs.gallery.editSite)+'">'+i18n("onlineEdit")+'</span>'+
  1382. '<span class="pv-gallery-head-command-drop-list-item" data-command="exportImages" title="'+i18n("exportImagesTip")+'">'+i18n("exportImages")+'</span>'+
  1383. '<span class="pv-gallery-head-command-drop-list-item" data-command="copyImages" title="'+i18n("copyImagesUrlTip")+'">'+i18n("copyImagesUrl")+'</span>'+
  1384. '<span class="pv-gallery-head-command-drop-list-item" data-command="downloadImage" title="'+i18n("downloadImageTip")+'">'+i18n("downloadImage")+'</span>'+
  1385. '<span class="pv-gallery-head-command-drop-list-item" data-command="openInNewWindow" title="'+i18n("openInNewWindowTip")+'">'+i18n("openInNewWindow")+'</span>'+
  1386. '<span class="pv-gallery-head-command-drop-list-item" data-command="enterCollection" title="'+i18n("viewCollectionTip")+'">'+i18n("viewCollection")+'</span>'+
  1387. '<span class="pv-gallery-head-command-drop-list-item" data-command="scrollIntoView" title="'+i18n("findInPageTip")+'">'+i18n("findInPage")+'</span>'+
  1388. '<span class="pv-gallery-head-command-drop-list-item" title="'+i18n("autoRefreshTip")+'">'+
  1389. '<label data-command="scrollToEndAndReload">'+i18n("autoRefresh")+'</label>'+
  1390. '<input type="checkbox" data-command="scrollToEndAndReload"/>'+
  1391. '</span>'+
  1392. '<span id="pv-gallery-fullscreenbtn" class="pv-gallery-head-command-drop-list-item" data-command="fullScreen">'+i18n("enterFullsc")+'</span>'+
  1393. '<span class="pv-gallery-head-command-drop-list-item" data-command="openPrefs">'+i18n("openConfig")+'</span>'+
  1394. '</span>'+
  1395. '</span>'+
  1396.  
  1397. '<span class="pv-gallery-head-command-container">'+
  1398. '<span title="'+i18n("headSearchTip")+'" class="pv-gallery-head-command pv-gallery-head-command-search">'+
  1399. '<span>'+i18n("headSearchTip")+'</span>'+
  1400. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1401. '</span>'+
  1402. '<span class="pv-gallery-head-command-drop-list pv-gallery-head-command-drop-list-search">'+
  1403. '<span class="pv-gallery-head-command-drop-list-item" id="headSearchAll" data-command="headSearchAll">'+i18n("headSearchAll")+'</span>'+
  1404. '</span>'+
  1405. '</span>'+
  1406.  
  1407. '<span class="pv-gallery-head-command-container">'+
  1408. '<span title="'+i18n("share")+'" class="pv-gallery-head-command pv-gallery-head-command-share">'+
  1409. '<span>'+i18n("share")+'</span>'+
  1410. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1411. '</span>'+
  1412. '<span class="pv-gallery-head-command-drop-list pv-gallery-head-command-drop-list-share">'+
  1413. '</span>'+
  1414. '</span>'+
  1415.  
  1416. '<span title="'+i18n("closeGallery")+'" class="pv-gallery-head-command pv-gallery-head-command-close">'+
  1417. '</span>'+
  1418.  
  1419. '</span>'+
  1420.  
  1421. '<span class="pv-gallery-body">'+
  1422.  
  1423. '<span class="pv-gallery-img-container">'+
  1424.  
  1425. '<span class="pv-gallery-img-content">'+
  1426. '<span class="pv-gallery-img-parent">'+
  1427. '<img title="'+i18n("refreshWhenError")+'" class="pv-gallery-img_broken" src="'+prefs.icons.brokenImg+'" />'+
  1428. '</span>'+
  1429. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1430. '</span>'+
  1431.  
  1432. '<span class="pv-gallery-img-controler pv-gallery-img-controler-pre"></span>'+
  1433. '<span class="pv-gallery-img-controler pv-gallery-img-controler-next"></span>'+
  1434.  
  1435. '<span class="pv-gallery-scrollbar-h pv-gallery-img-scrollbar-h">'+
  1436. '<span class="pv-gallery-scrollbar-h-track pv-gallery-img-scrollbar-h-track">'+
  1437. '<span class="pv-gallery-scrollbar-h-handle pv-gallery-img-scrollbar-h-handle"></span>'+
  1438. '</span>'+
  1439. '</span>'+
  1440.  
  1441. '<span class="pv-gallery-scrollbar-v pv-gallery-img-scrollbar-v">'+
  1442. '<span class="pv-gallery-scrollbar-v-track pv-gallery-img-scrollbar-v-track">'+
  1443. '<span class="pv-gallery-scrollbar-v-handle pv-gallery-img-scrollbar-v-handle"></span>'+
  1444. '</span>'+
  1445. '</span>'+
  1446.  
  1447. '<span class="pv-gallery-sidebar-toggle" title="'+i18n("switchSlide")+'">'+
  1448. '<span class="pv-gallery-sidebar-toggle-content">▼</span>'+
  1449. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1450. '</span>'+
  1451.  
  1452. '<span class="pv-gallery-sidebar-viewmore" title="'+i18n("viewmore")+'">'+
  1453. '<span class="pv-gallery-sidebar-viewmore-content">✚</span>'+
  1454. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1455. '</span>'+
  1456.  
  1457. '</span>'+
  1458.  
  1459. '<span class="pv-gallery-sidebar-container" unselectable="on">'+
  1460. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1461. '<span class="pv-gallery-sidebar-content" >'+
  1462.  
  1463. '<span class="pv-gallery-sidebar-controler pv-gallery-sidebar-controler-pre"></span>'+
  1464. '<span class="pv-gallery-sidebar-controler pv-gallery-sidebar-controler-next"></span>'+
  1465.  
  1466. '<span class="pv-gallery-sidebar-thumbnails-container">'+
  1467. '</span>'+
  1468.  
  1469. '<span class="pv-gallery-scrollbar-h pv-gallery-thumb-scrollbar-h">'+
  1470. '<span class="pv-gallery-scrollbar-h-track pv-gallery-thumb-scrollbar-h-track">'+
  1471. '<span class="pv-gallery-scrollbar-h-handle pv-gallery-thumb-scrollbar-h-handle"></span>'+
  1472. '</span>'+
  1473. '</span>'+
  1474. '<span class="pv-gallery-scrollbar-v pv-gallery-thumb-scrollbar-v">'+
  1475. '<span class="pv-gallery-scrollbar-v-track pv-gallery-thumb-scrollbar-v-track">'+
  1476. '<span class="pv-gallery-scrollbar-v-handle pv-gallery-thumb-scrollbar-v-handle"></span>'+
  1477. '</span>'+
  1478. '</span>'+
  1479.  
  1480. '</span>'+
  1481. '</span>'+
  1482. '<span class="pv-gallery-maximize-scroll"><span class="pv-gallery-maximize-container"></span></span>'+
  1483. '</span>');
  1484. document.body.appendChild(container);
  1485.  
  1486. var self=this;
  1487.  
  1488. var hideBodyStyle=document.createElement('style');
  1489. this.hideBodyStyle=hideBodyStyle;
  1490. hideBodyStyle.textContent=`body>*:not([class^="pv-"]) img,body>img{display:none}`;
  1491.  
  1492. container.querySelector("#minsizeW").oninput=function(){self.changeMinView();};
  1493. container.querySelector("#minsizeH").oninput=function(){self.changeMinView();};
  1494. container.querySelector("#minsizeWSpan").onclick=function(){
  1495. var minsizeW=window.prompt("Width:",this.value);
  1496. if(!minsizeW)return;
  1497. container.querySelector("#minsizeW").value=minsizeW;
  1498. self.changeMinView();
  1499. };
  1500. container.querySelector("#minsizeHSpan").onclick=function(){
  1501. var minsizeH=window.prompt("Height:",this.value);
  1502. if(!minsizeH)return;
  1503. container.querySelector("#minsizeH").value=minsizeH;
  1504. self.changeMinView();
  1505. };
  1506. container.querySelector(".pv-gallery-head-left-lock-icon").onclick=function(){
  1507. if(self.lockMaxSize){
  1508. self.lockMaxSize=null;
  1509. self.changeMinView();
  1510. this.style.filter="";
  1511. this.title=i18n("lockSizeTip");
  1512. }else{
  1513. var maxsizeW=window.prompt("Width:");
  1514. if(!maxsizeW)return;
  1515. var maxsizeH=window.prompt("Height:");
  1516. if(!maxsizeH)return;
  1517. self.lockMaxSize={w:maxsizeW,h:maxsizeH};
  1518. self.changeMinView();
  1519. this.style.filter="brightness(5)";
  1520. this.title=maxsizeW+" x "+maxsizeH;
  1521. }
  1522. };
  1523.  
  1524. var maximizeTrigger=document.createElement('span');
  1525. this.maximizeTrigger=maximizeTrigger;
  1526. maximizeTrigger.innerHTML=createHTML('-'+i18n("returnToGallery")+'-<span class="pv-gallery-maximize-trigger-close" title="'+i18n("closeGallery")+'"></span>');
  1527. maximizeTrigger.className='pv-gallery-maximize-trigger';
  1528.  
  1529. document.body.appendChild(maximizeTrigger);
  1530.  
  1531.  
  1532. var validPos=['top','right','bottom','left'];
  1533. var sBarPosition=prefs.gallery.sidebarPosition;
  1534. if(validPos.indexOf(sBarPosition)==-1){
  1535. sBarPosition='bottom';
  1536. };
  1537.  
  1538. this.sBarPosition=sBarPosition;
  1539. this.selectedClassName='pv-gallery-sidebar-thumb_selected-' + sBarPosition;
  1540.  
  1541.  
  1542. var sBarDirection='v';//垂直放置
  1543. var isHorizontal=false;
  1544. if(sBarPosition=='top' || sBarPosition=='bottom'){
  1545. sBarDirection='h';//水平放置
  1546. isHorizontal=true;
  1547. };
  1548. this.sBarDirection=sBarDirection;
  1549. this.isHorizontal=isHorizontal;
  1550.  
  1551. var classPrefix='pv-gallery-';
  1552. var validClass=[
  1553. 'head',
  1554.  
  1555. 'head-left-img-info',
  1556. 'head-left-img-info-description',
  1557. 'head-left-img-info-resolution',
  1558. 'head-left-img-info-count',
  1559. 'head-left-img-info-scaling',
  1560.  
  1561. 'head-command-close',
  1562. 'head-command-nextPage',
  1563. 'head-command-operate',
  1564. 'head-command-urlFilter',
  1565. 'head-command-slide-show',
  1566. 'head-command-slide-show-button-inner',
  1567. 'head-command-slide-show-countdown',
  1568. 'head-command-collect',
  1569. 'head-command-exit-collection',
  1570.  
  1571. 'head-command-drop-list-category',
  1572. 'head-command-drop-list-others',
  1573. 'head-command-drop-list-share',
  1574. 'head-command-drop-list-slide-show',
  1575. 'head-command-drop-list-collect',
  1576. 'head-command-drop-list-search',
  1577.  
  1578. 'body',
  1579.  
  1580. 'img-container',
  1581.  
  1582. 'img-scrollbar-h',
  1583. 'img-scrollbar-h-handle',
  1584. 'img-scrollbar-h-track',
  1585.  
  1586. 'img-scrollbar-v',
  1587. 'img-scrollbar-v-handle',
  1588. 'img-scrollbar-v-track',
  1589.  
  1590. 'thumb-scrollbar-h',
  1591. 'thumb-scrollbar-h-handle',
  1592. 'thumb-scrollbar-h-track',
  1593.  
  1594. 'thumb-scrollbar-v',
  1595. 'thumb-scrollbar-v-handle',
  1596. 'thumb-scrollbar-v-track',
  1597.  
  1598. 'img-content',
  1599. 'img-parent',
  1600. 'img_broken',
  1601.  
  1602. 'img-controler-pre',
  1603. 'img-controler-next',
  1604.  
  1605. 'sidebar-toggle',
  1606. 'sidebar-toggle-content',
  1607. 'sidebar-viewmore',
  1608. 'sidebar-viewmore-content',
  1609. 'maximize-container',
  1610.  
  1611. 'sidebar-container',
  1612. 'sidebar-content',
  1613.  
  1614. 'sidebar-controler-pre',
  1615. 'sidebar-controler-next',
  1616.  
  1617. 'sidebar-thumbnails-container',
  1618. ];
  1619.  
  1620. var eleMaps={};
  1621. this.eleMaps=eleMaps;
  1622.  
  1623. validClass.forEach(function(c){
  1624. eleMaps[c]=container.querySelector('.'+ classPrefix + c);
  1625. });
  1626.  
  1627. var posClass=[//需要添加'top bottom left right'class的元素
  1628. 'img-container',
  1629. 'sidebar-toggle',
  1630. 'sidebar-viewmore',
  1631. 'sidebar-container',
  1632. 'sidebar-thumbnails-container',
  1633. ];
  1634. posClass.forEach(function(c){
  1635. eleMaps[c].classList.add(classPrefix + c + '-' +sBarPosition);
  1636. });
  1637.  
  1638. var hvClass=[//需要添加'v h'class的元素
  1639. 'sidebar-toggle',
  1640. 'sidebar-toggle-content',
  1641. 'sidebar-viewmore',
  1642. 'sidebar-viewmore-content',
  1643. 'sidebar-container',
  1644. 'sidebar-content',
  1645. 'sidebar-controler-pre',
  1646. 'sidebar-controler-next',
  1647. 'sidebar-thumbnails-container',
  1648. ];
  1649. hvClass.forEach(function(c){
  1650. eleMaps[c].classList.add(classPrefix + c + '-' + sBarDirection);
  1651. });
  1652.  
  1653.  
  1654.  
  1655. //图片区域水平方向的滚动条
  1656. var imgScrollbarH=new this.Scrollbar({
  1657. bar:eleMaps['img-scrollbar-h'],
  1658. handle:eleMaps['img-scrollbar-h-handle'],
  1659. track:eleMaps['img-scrollbar-h-track'],
  1660. },eleMaps['img-content'],true);
  1661. this.imgScrollbarH=imgScrollbarH;
  1662.  
  1663. //图片区域垂直方向的滚动条
  1664. var imgScrollbarV=new this.Scrollbar({
  1665. bar:eleMaps['img-scrollbar-v'],
  1666. handle:eleMaps['img-scrollbar-v-handle'],
  1667. track:eleMaps['img-scrollbar-v-track'],
  1668. },eleMaps['img-content'],false);
  1669. this.imgScrollbarV=imgScrollbarV;
  1670.  
  1671. //缩略图区域的滚动条
  1672. var thumbScrollbar;
  1673. if(isHorizontal){
  1674. thumbScrollbar=new this.Scrollbar({
  1675. bar:eleMaps['thumb-scrollbar-h'],
  1676. handle:eleMaps['thumb-scrollbar-h-handle'],
  1677. track:eleMaps['thumb-scrollbar-h-track'],
  1678. },eleMaps['sidebar-thumbnails-container'],true);
  1679. }else{
  1680. thumbScrollbar=new this.Scrollbar({
  1681. bar:eleMaps['thumb-scrollbar-v'],
  1682. handle:eleMaps['thumb-scrollbar-v-handle'],
  1683. track:eleMaps['thumb-scrollbar-v-track'],
  1684. },eleMaps['sidebar-thumbnails-container'],false);
  1685. };
  1686. this.thumbScrollbar=thumbScrollbar;
  1687.  
  1688. self=this;
  1689.  
  1690. var imgStatistics={//图片的总类,统计,初始化值
  1691. rule:{
  1692. shown:true,
  1693. count:0,
  1694. description:i18n("advancedRulesTip"),
  1695. name:i18n("advancedRules"),
  1696. },
  1697. tpRule:{
  1698. shown:true,
  1699. count:0,
  1700. description:i18n("tpRulesTip"),
  1701. name:i18n("tpRules"),
  1702. },
  1703. scale:{
  1704. shown:true,
  1705. count:0,
  1706. description:i18n("scaleRulesTip"),
  1707. name:i18n("scaleRules"),
  1708. },
  1709. force:{
  1710. shown:true,
  1711. count:0,
  1712. description:i18n("noScaleRulesTip"),
  1713. name:i18n("noScaleRules"),
  1714. },
  1715.  
  1716. // new
  1717. // scaleZoomResized: {
  1718. // shown: false,
  1719. // count: 0,
  1720. // description: '缩放的图片,图片尺寸最少相差比例 ' + prefs.gallery.zoomresized + '%',
  1721. // name: '小缩放'
  1722. // },
  1723. scaleSmall: {
  1724. shown: prefs.gallery.showSmallSize,
  1725. count: 0,
  1726. description: i18n("smallRulesTip",prefs.gallery.scaleSmallSize),
  1727. name: i18n("smallRules")
  1728. },
  1729. };
  1730. this.imgStatistics=imgStatistics;
  1731.  
  1732. //生成分类下拉列表
  1733. var typeMark='';
  1734. var imgStatistics_i;
  1735. for(var i in imgStatistics){
  1736. if(!imgStatistics.hasOwnProperty(i))continue;
  1737. imgStatistics_i=imgStatistics[i];
  1738. typeMark+=
  1739. '<span class="pv-gallery-head-command-drop-list-item" title="'+imgStatistics_i.description+'">'+
  1740. '<input type="checkbox" data-type="'+i+'" id="pv-gallery-head-command-drop-list-item-category-'+i+'" />'+
  1741. '<label for="pv-gallery-head-command-drop-list-item-category-'+i+'">'+imgStatistics_i.name+'</label>'+
  1742. '</span>';
  1743. };
  1744. eleMaps['head-command-drop-list-category'].innerHTML=createHTML(typeMark);
  1745.  
  1746.  
  1747. //收藏相关
  1748. var collection={
  1749. getMatched:function(){
  1750. return (this.all || this.get())._find(function(value,index){
  1751. if(value.src==self.src){
  1752. return true;
  1753. };
  1754. });
  1755. },
  1756. check:function(){
  1757. //从缓存数据中检查。
  1758. var matched=this.getMatched();
  1759. this.favorite=matched? matched[0] : null;
  1760.  
  1761. this.tAreaValue();
  1762. this.highLight();
  1763. },
  1764. tAreaValue:function(){
  1765. this.textArea.value=this.favorite? this.favorite.description : self.eleMaps['head-left-img-info-description'].textContent;
  1766. },
  1767. highLight:function(){
  1768. eleMaps['head-command-collect'].classList[this.favorite? 'add' : 'remove']('pv-gallery-head-command-collect-favorite');
  1769. },
  1770. add:function(){
  1771. this.favorite={
  1772. src:self.src,
  1773. thumbSrc:dataset(self.relatedThumb,'thumbSrc'),
  1774. naturalSize:self.imgNaturalSize,
  1775. description:this.textArea.value,
  1776. };
  1777.  
  1778. //为了防止多个页面同时的储存,添加前,先载入最新的数据。
  1779. this.get();
  1780. //检查是否已经在里面了
  1781. var matched=this.getMatched();
  1782.  
  1783. if(matched){//如果已经存在,删除旧的。
  1784. this.all.splice(matched[1],1);
  1785. };
  1786. this.all.unshift(this.favorite);//添加到最前面。
  1787. this.highLight();
  1788. this.save();
  1789. },
  1790. remove:function(){
  1791. //获得最新数据
  1792. this.get();
  1793. //检查是否已经在里面了
  1794. var matched=this.getMatched();
  1795. if(matched){
  1796. this.all.splice(matched[1],1);
  1797. this.save();
  1798. };
  1799. this.favorite=null;
  1800. this.highLight();
  1801. },
  1802. save:function(){
  1803. storage.setItem('pv_collection',encodeURIComponent(JSON.stringify(this.all)));
  1804. },
  1805. get:function(){
  1806. var ret=storage.getItem('pv_collection') || '[]';
  1807. try{
  1808. ret=JSON.parse(decodeURIComponent(ret));
  1809. }catch(e){
  1810. ret=[];
  1811. };
  1812. this.all=ret;
  1813. return ret;
  1814. },
  1815. enter:function(){
  1816.  
  1817. if(this.all.length==0){
  1818. _GM_notification(i18n("noCollectionYet"));
  1819. return;
  1820. };
  1821.  
  1822. this.mMode=true;
  1823. var button=this.dropListButton;
  1824. button.textContent=i18n("exitCollection");
  1825. dataset(button,'command','exitCollection');
  1826. this.headButton.style.display='inline-block';
  1827. eleMaps['sidebar-thumbnails-container'].classList.add('pv-gallery-sidebar-thumbnails_hide-span');
  1828.  
  1829. //生成dom
  1830. var container=document.createElement('span');
  1831.  
  1832. this.container=container;
  1833.  
  1834. var data_i;
  1835. var spanMark='';
  1836. var i=0;
  1837. while(data_i=this.all[i++]){
  1838. spanMark +=
  1839. '<span class="pv-gallery-sidebar-thumb-container" '+
  1840. 'data-description="' + data_i.description + '"' +
  1841. ' data-natural-size="' + JSON.stringify(data_i.naturalSize).replace(/"/g,'&quot;') +
  1842. '" data-src="' + data_i.src +
  1843. '" data-thumb-src="' + data_i.thumbSrc +
  1844. '">'+
  1845. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1846. '<span class="pv-gallery-sidebar-thumb-loading" title="'+i18n("loading")+'......"></span>'+
  1847. '</span>';
  1848. };
  1849. container.innerHTML=createHTML(spanMark);
  1850. eleMaps['sidebar-thumbnails-container'].appendChild(container);
  1851.  
  1852.  
  1853. this.selected=self.selected;//备份
  1854.  
  1855. self.select(container.children[0]);
  1856. self.thumbScrollbar.reset();
  1857. self.loadThumb();
  1858. },
  1859. exit:function(){
  1860. if(!this.mMode)return;
  1861.  
  1862. this.mMode=false;
  1863. var button=this.dropListButton;
  1864. button.textContent=i18n("viewCollection");
  1865. dataset(button,'command','enterCollection');
  1866. this.headButton.style.display='none';
  1867. eleMaps['sidebar-thumbnails-container'].removeChild(this.container);
  1868. eleMaps['sidebar-thumbnails-container'].classList.remove('pv-gallery-sidebar-thumbnails_hide-span');
  1869.  
  1870. self.select(this.selected);
  1871. self.thumbScrollbar.reset();
  1872. self.loadThumb();
  1873. },
  1874. textArea:eleMaps['head-command-drop-list-collect'].querySelector('textarea'),
  1875. dropListButton:eleMaps['head-command-drop-list-others'].querySelector('[data-command$="Collection"]'),
  1876. headButton:eleMaps['head-command-exit-collection'],
  1877. };
  1878.  
  1879. this.collection=collection;
  1880. this.collection.textArea.addEventListener('focus',function(e){
  1881. eleMaps['head-command-drop-list-collect'].classList.add("focus");
  1882. });
  1883. this.collection.textArea.addEventListener('blur',function(e){
  1884. eleMaps['head-command-drop-list-collect'].classList.remove("focus");
  1885. });
  1886.  
  1887. eleMaps['head-command-drop-list-collect'].addEventListener('input',function(e){
  1888. var target=e.target;
  1889. if(!collection.favorite)return;
  1890. collection.favorite[dataset(target,'prefs')]=target.value;
  1891. clearTimeout(collection.saveTimer);
  1892. collection.saveTimer=setTimeout(function(){
  1893. collection.save();
  1894. },500);
  1895. },true);
  1896.  
  1897.  
  1898. var slideShow={
  1899. opts:{
  1900. interval:5000,
  1901. wait:true,
  1902. backward:false,
  1903. skipErrorImg:true,
  1904. run:false,
  1905. },
  1906. //timing:
  1907. //select(选中下一个图片后(缩略图栏选中了),还没开始读取大图(一般选中后,延时200ms开始读取大图)),
  1908. //loadEnd(当前显示图片已经读取完成后),
  1909. //click(点击按钮),
  1910. //change(改变设置)
  1911. run:function(timing){
  1912. if(!this.opts.run)return;
  1913.  
  1914. if(timing!='loadEnd'){
  1915. this.stop();
  1916. };
  1917.  
  1918. if(timing=='click' || timing=='select'){
  1919. if(!this.getEle()){//没有要切换到的图片了,停止
  1920. this.exit();
  1921. return;
  1922. };
  1923. };
  1924.  
  1925. if(this.opts.skipErrorImg){
  1926. if(self.imgError && !self.isLoading){//确保是当前图片和选中缩略图一致的时候
  1927. self.select(this.getEle());
  1928. return;
  1929. };
  1930. };
  1931.  
  1932.  
  1933. if(this.opts.wait){
  1934. if(timing!='select' && (timing=='loadEnd' || (!self.isLoading && (self.img.complete || self.imgError)))){
  1935. this.go();
  1936. };
  1937. }else{
  1938. if(timing!='loadEnd'){
  1939. this.go();
  1940. };
  1941. };
  1942.  
  1943. },
  1944. getEle:function(){
  1945. return self.getThumSpan(this.opts.backward)
  1946. },
  1947. go:function(){
  1948. this.stop();//停止上次的。
  1949. var interval=this.opts.interval;
  1950. var _self=this;
  1951. this.timer=setTimeout(function(){
  1952. _self.setCountdown(0);
  1953. clearInterval(_self.countdownTimer);
  1954. self.select(_self.getEle());
  1955. },interval);
  1956.  
  1957. var startTime=Date.now();
  1958. this.countdownTimer=setInterval(function(){
  1959. _self.setCountdown(interval - (Date.now()-startTime));
  1960. },100);
  1961. },
  1962. stop:function(){
  1963. this.setCountdown(this.opts.interval);
  1964. clearTimeout(this.timer);
  1965. clearInterval(this.countdownTimer);
  1966. },
  1967. exit:function(){
  1968. this.opts.run=true;
  1969. this.switchStatus();
  1970. this.stop();
  1971. },
  1972. setCountdown:function(value){
  1973. eleMaps['head-command-slide-show-countdown'].textContent=(value/1000).toFixed(2);
  1974. },
  1975. switchStatus:function(){
  1976. this.opts.run=!this.opts.run;
  1977. eleMaps['head-command-slide-show-button-inner'].classList[this.opts.run? 'add' : 'remove']('pv-gallery-head-command-slide-show-button-inner_stop');
  1978. },
  1979. check:function(){
  1980. this.opts.run? this.run('click') : this.stop();
  1981. },
  1982. };
  1983.  
  1984. slideShow.setCountdown(slideShow.opts.interval);;
  1985. this.slideShow=slideShow;
  1986.  
  1987. //幻灯片播放下拉列表change事件的处理
  1988. eleMaps['head-command-drop-list-slide-show'].addEventListener('change',function(e){
  1989. var target=e.target;
  1990. var value;
  1991. var prefs=dataset(target,'prefs');
  1992. if(target.type=='checkbox'){
  1993. value=target.checked;
  1994. }else{
  1995. value=parseFloat(target.value);
  1996. if(isNaN(value)){//无效
  1997. value=slideShow.opts[prefs] / 1000;
  1998. };
  1999. value=value>0 ? value : 1;
  2000. target.value=value;
  2001. value *= 1000;
  2002. };
  2003. slideShow.opts[prefs]=value;
  2004. slideShow.run('change');
  2005. },true);
  2006.  
  2007.  
  2008. //分类下拉列表的点击发生change事件的处理
  2009. eleMaps['head-command-drop-list-category'].addEventListener('change',function(e){
  2010. var target=e.target;
  2011. var type=dataset(target,'type');
  2012. self.iStatisCopy[type].shown=target.checked;
  2013. self.switchThumbVisible();//切换图片类别显隐;
  2014. },true);
  2015.  
  2016.  
  2017. var srcSplit,downloading=false;
  2018. //命令下拉列表的点击处理
  2019. eleMaps['head-command-drop-list-others'].addEventListener('click',function(e){
  2020. if(e.button!=0)return;//左键
  2021. var target=e.target;
  2022. var command=dataset(target,'command');
  2023. if(!command)return;
  2024. switch(command){
  2025. case 'openInNewWindow':{
  2026. window.open(self.src,'_blank');
  2027. }break;
  2028. case 'psImage':{
  2029. //window.open((prefs.gallery.editSite=='Pixlr'?'https://pixlr.com/editor/?image=':'https://www.toolpic.com/apieditor.html?image=')+self.src,'_blank');
  2030. window.open('https://www.lunapic.com/editor/index.php?action=url&url='+self.src,'_blank');
  2031. }break;
  2032. case 'scrollIntoView':{
  2033. if(collection.mMode){
  2034. _GM_notification(i18n("inCollection"));
  2035. return;
  2036. };
  2037. var relatedThumb=self.relatedThumb;
  2038. var index=arrayFn.indexOf.call(self.imgSpans,relatedThumb);
  2039. var targetImg=self.data[index].img;
  2040.  
  2041. if(targetImg){
  2042. if(!document.documentElement.contains(targetImg) || unsafeWindow.getComputedStyle(targetImg).display=='none'){//图片不存在文档中,或者隐藏了。
  2043. _GM_notification(i18n("cantFind"));
  2044. return;
  2045. };
  2046. self.minimize();
  2047. setTimeout(function(){
  2048. self.navigateToImg(targetImg);
  2049. flashEle(targetImg);
  2050. },0);
  2051.  
  2052. }else{//frame发送过来的时候删除了不能传送的图片
  2053.  
  2054. document.addEventListener('pv-navigateToImg',function(e){
  2055. if(!e.detail){
  2056. _GM_notification(i18n("cantFind"));
  2057. return;
  2058. };
  2059. self.minimize();
  2060. setTimeout(function(){//将frame滚动到中间位置
  2061. if(self.iframe){
  2062. self.navigateToImg(self.iframe);
  2063. };
  2064. },0);
  2065. },true);
  2066. window.postMessage({//问问frame。。
  2067. messageID:messageID,
  2068. command:'navigateToImg',
  2069. index:index,
  2070. to:self.from,
  2071. },'*');
  2072. };
  2073.  
  2074. }break;
  2075. case 'exportImages':
  2076. self.exportImages();
  2077. break;
  2078. case 'downloadImage':
  2079. if(downloading)break;
  2080. downloading=true;
  2081. var nodes = document.querySelectorAll('.pv-gallery-sidebar-thumb-container[data-src]');
  2082. var saveParams = [],saveIndex=0;
  2083. [].forEach.call(nodes, function(node){
  2084. if(unsafeWindow.getComputedStyle(node).display!="none"){
  2085. saveIndex++;
  2086. srcSplit=node.dataset.src.replace(/[\?#].*/,"").split("/");
  2087. srcSplit=srcSplit[srcSplit.length-1];
  2088. if(srcSplit.length>30 && (srcSplit.indexOf(".")==-1 || /[&\?=,]/i.test(srcSplit))){
  2089. srcSplit="";
  2090. }
  2091. var picName=document.title + "-" + (saveIndex<10?"00"+saveIndex:(saveIndex<100?"0"+saveIndex:saveIndex)) + (node.title?"-" + node.title:"") + "-" + srcSplit,hostArr=location.host.split(".");
  2092. var host=hostArr[hostArr.length-2];
  2093. saveParams.push([node.dataset.src, picName]);
  2094. //saveAs(node.dataset.src, location.host+"-"+srcSplit[srcSplit.length-1]);
  2095. }
  2096. });
  2097. self.batchDownload(saveParams, ()=>{
  2098. downloading=false;
  2099. });
  2100. break;
  2101. case 'copyImages':
  2102. self.copyImages(true);
  2103. break;
  2104. case 'scrollToEndAndReload':
  2105. var checkbox = target.parentNode.querySelector("input");
  2106. if(target.nodeName=="LABEL"){
  2107. checkbox.checked = !checkbox.checked;
  2108. }
  2109.  
  2110. prefs.gallery.scrollEndAndLoad = checkbox.checked;
  2111. break;
  2112. case 'fullScreen':
  2113. if (target.classList.contains('fullscreenbtn')) {
  2114. if (cancelFullScreen()) return;
  2115. target.textContent = i18n("enterFullsc");
  2116. target.classList.remove('fullscreenbtn');
  2117. return;
  2118. }
  2119.  
  2120. if (launchFullScreen(document.documentElement)) return;
  2121. target.classList.toggle('fullscreenbtn');
  2122. target.textContent = i18n("exitFullsc");
  2123. target.classList.add('fullscreenbtn');
  2124. break;
  2125. case 'openPrefs':
  2126. openPrefs();
  2127. break;
  2128. case 'enterCollection':{
  2129. //进入管理模式
  2130. collection.enter();
  2131. }break;
  2132. case 'exitCollection':{
  2133. //退出管理模式
  2134. collection.exit();
  2135. }break;
  2136. };
  2137. },true);
  2138.  
  2139. // 监视全屏的变化
  2140. function fullScreenChanged() {
  2141. if (!document.fullscreenElement && // alternative standard method
  2142. !document.mozFullScreenElement &&
  2143. !document.webkitFullscreenElement &&
  2144. !document.msFullscreenElement) {
  2145.  
  2146. var btn = document.getElementById("pv-gallery-fullscreenbtn");
  2147. if (btn) {
  2148. btn.textContent = i18n("enterFullsc");
  2149. btn.removeClass('fullscreenbtn');
  2150. }
  2151. }
  2152. }
  2153. document.addEventListener('webkitfullscreenchange', fullScreenChanged, false);
  2154. document.addEventListener('mozfullscreenchange', fullScreenChanged, false);
  2155. document.addEventListener('fullscreenchange', fullScreenChanged, false);
  2156.  
  2157. //生成分享的下拉列表
  2158. var shareMark='';
  2159. var shareItem;
  2160. for(let i in prefs.share){
  2161. if(!prefs.share.hasOwnProperty(i))continue;
  2162. shareItem=prefs.share[i];
  2163. if(shareItem.limitLang && shareItem.limitLang.indexOf(lang)==-1)continue;
  2164. if(shareItem.disabled)continue;
  2165. shareMark+=(
  2166. '<span class="pv-gallery-head-command-drop-list-item" data-site="'+i+'" style="\
  2167. background-image:url(\''+ shareItem.icon +'\');\
  2168. background-position:4px center;\
  2169. background-repeat:no-repeat;\
  2170. padding-left:24px;">'+shareItem.name+'</span>');
  2171. };
  2172.  
  2173. eleMaps['head-command-drop-list-share'].innerHTML=createHTML(shareMark);
  2174.  
  2175. //分享下拉列表的点击处理
  2176. eleMaps['head-command-drop-list-share'].addEventListener('click',function(e){
  2177. if(e.button!=0)return;//左键
  2178. var target=e.target;
  2179. var site=dataset(target,'site');
  2180. if(!site)return;
  2181. var site_info=prefs.share[site];
  2182. var param=site_info.api.call(self.img,{
  2183. title:encodeURIComponent(document.title),
  2184. pic:encodeURIComponent(self.src),
  2185. url:encodeURIComponent(location.href),
  2186. });
  2187. if(!param)return;
  2188. window.open(param.url,'_blank','height='+param.wSize.h+',width='+param.wSize.w+',left=30,top=30,location=no,status=no,toolbar=no,menubar=no,scrollbars=yes');
  2189. },true);
  2190.  
  2191. /*eleMaps['head'].addEventListener('click',function(e){
  2192. if(e.target.className.indexOf('pv-gallery-head-command')!=-1){
  2193. self.closeViewMore();
  2194. }
  2195. });*/
  2196.  
  2197. if(!prefs.gallery.searchData || defaultSearchData.indexOf(prefs.gallery.searchData)!=-1)prefs.gallery.searchData=defaultSearchData;
  2198. var searchRules=prefs.gallery.searchData.split("\n"),searchUploadUrl,searchItems=[];
  2199. var searchAll=eleMaps['head-command-drop-list-search'].querySelector("#headSearchAll");
  2200. searchRules.forEach(rule=>{
  2201. if(!searchUploadUrl){
  2202. var uploadMatch=rule.match(/\s*{(.*)}\s*/);//todo: upload 2 search, need a Long-Term Servicing server for base64 to url like ainoob.com/api/uploadImage/
  2203. if(uploadMatch){
  2204. searchUploadUrl=uploadMatch[1];
  2205. return;
  2206. }
  2207. }
  2208. let ruleArr=rule.trim().split("|");
  2209. if(ruleArr.length==2){
  2210. var item=document.createElement('span');
  2211. item.className="pv-gallery-head-command-drop-list-item";
  2212. item.innerHTML=createHTML(ruleArr[0]);
  2213. item.addEventListener('click',function(e){
  2214. let url=encodeURIComponent(self.src);
  2215. let urlb=self.src.replace(/https?:\/\//i,"");
  2216. window.open(ruleArr[1].replace("#b#", urlb).replace("#t#", url), "_blank", "width=1024, height=768, toolbar=1");
  2217. });
  2218. searchItems.push(item);
  2219. eleMaps['head-command-drop-list-search'].insertBefore(item, searchAll);
  2220. }
  2221. });
  2222. searchAll.addEventListener('click',function(e){
  2223. searchItems.forEach(item=>{item.click()});
  2224. });
  2225.  
  2226.  
  2227.  
  2228. var loadThumbsTimer;
  2229. eleMaps['sidebar-thumbnails-container'].addEventListener('scroll',function(e){//发生scroll事件时加载缩略图
  2230. clearTimeout(loadThumbsTimer);//加个延时,在连续触发的时候缓一缓。
  2231. loadThumbsTimer=setTimeout(function(){
  2232. self.loadThumb();
  2233. },200);
  2234. },false);
  2235.  
  2236. this.canScroll=true;
  2237. var scrollToChange=function(next){
  2238. if(self.canScroll){
  2239. if(prefs.gallery.transition){
  2240. self.canScroll=false;
  2241. setTimeout(function(){
  2242. self.canScroll=true;
  2243. },500);
  2244. }
  2245. next ? self.selectNext() : self.selectPrevious();
  2246. }
  2247. }
  2248. addWheelEvent(eleMaps['body'],function(e){//wheel事件
  2249. if(e.deltaZ!=0)return;//z轴
  2250. if(eleMaps['sidebar-toggle'].style.visibility == 'hidden')return;
  2251. var target=e.target;
  2252. //e.preventDefault();
  2253. if(eleMaps['sidebar-container'].contains(target)){//缩略图区滚动滚轮翻图片
  2254. let distance=self.thumbSpanOuterSize;
  2255.  
  2256. if(e.deltaY<0 || e.deltaX<0){//向上滚
  2257. distance=-distance;
  2258. };
  2259. thumbScrollbar.scrollBy(distance)
  2260. }else{//图片区域滚动
  2261. let distance=100;
  2262. if(e.deltaY!=0){//y轴
  2263. if(self.img && self.img.classList.contains('pv-gallery-img_zoom-out')){//图片可以缩小时,滚动图片,否则切换图片。
  2264. if(e.deltaY < 0){
  2265. distance=-distance;
  2266. };
  2267. if(eleMaps['img-scrollbar-h'].contains(target)){//如果在横向滚动条上。
  2268. imgScrollbarH.scrollBy(distance);
  2269. }else{
  2270. if(imgScrollbarV.scrollBy(distance) && prefs.gallery.scrollEndToChange){
  2271. scrollToChange(e.deltaY > 0);
  2272. }
  2273. };
  2274. }else{
  2275. scrollToChange(e.deltaY > 0);
  2276. };
  2277. }else{//x轴
  2278. if(e.deltaX < 0){
  2279. distance=-distance;
  2280. };
  2281. imgScrollbarH.scrollBy(distance);
  2282. }
  2283. }
  2284. },true);
  2285.  
  2286.  
  2287. //focus,blur;
  2288. addCusMouseEvent('mouseenter',container,function(){
  2289. this.focus();
  2290. });
  2291. addCusMouseEvent('mouseleave',container,function(){
  2292. this.blur();
  2293. });
  2294.  
  2295. var lastX,lastY;
  2296. const minLength=10000,tg=0.5;
  2297. function tracer(e) {
  2298. let curX=e.changedTouches[0].clientX;
  2299. let curY=e.changedTouches[0].clientY;
  2300. let distanceX=curX-lastX,distanceY=curY-lastY;
  2301. let distance=distanceX*distanceX+distanceY*distanceY;
  2302. if (distance>minLength) {
  2303. lastX=curX;
  2304. lastY=curY;
  2305. let direction="";
  2306. let slope=Math.abs(distanceY/distanceX);
  2307. if(slope>tg){
  2308. if(distanceY>0) {
  2309. direction="↓";
  2310. }else{
  2311. direction="↑";
  2312. }
  2313. }else if(slope<=1/tg) {
  2314. if(distanceX>0) {
  2315. direction="→";
  2316. }else{
  2317. direction="←";
  2318. }
  2319. }
  2320. switch(direction){
  2321. case "←":
  2322. self.selectNext();
  2323. break;
  2324. case "→":
  2325. self.selectPrevious();
  2326. break;
  2327. case "↑":
  2328. if(self.eleMaps['sidebar-toggle'].style.visibility != 'hidden'){
  2329. self.maximizeSidebar();
  2330. }
  2331. break;
  2332. default:
  2333. break;
  2334. }
  2335. }
  2336. }
  2337. self.eleMaps['img-content'].addEventListener('touchstart',function(e){
  2338. lastX=e.changedTouches[0].clientX;
  2339. lastY=e.changedTouches[0].clientY;
  2340. self.eleMaps['img-content'].addEventListener('touchmove',tracer);
  2341. });
  2342. self.eleMaps['img-content'].addEventListener('touchend',function(e){
  2343. self.eleMaps['img-content'].removeEventListener('touchmove',tracer);
  2344. });
  2345.  
  2346. //上下左右切换图片,空格键模拟滚动一页
  2347.  
  2348. var validKeyCode=[38,39,40,37,32,9]//上右下左,32空格,tab禁止焦点切换。
  2349. var keyDown;
  2350.  
  2351. container.addEventListener('keydown',function(e){
  2352. var keyCode=e.keyCode;
  2353. var index=validKeyCode.indexOf(keyCode);
  2354. if(index==-1)return;
  2355.  
  2356. var target=e.target;
  2357.  
  2358. if(!container.contains(target))return;//触发焦点不再gallery里面。
  2359. e.preventDefault();
  2360.  
  2361. if(keyCode==9)return;//tab键
  2362. if(keyCode==32){//32空格,模拟滚动一页
  2363. imgScrollbarV.scrollByPages(1);
  2364. return;
  2365. };
  2366.  
  2367. if(keyDown)return;//已按下。
  2368. keyDown=true;
  2369.  
  2370. var stop;
  2371. switch(index){
  2372. case 0:;
  2373. case 3:{
  2374. self.selectPrevious();
  2375. stop=self.simpleSlideShow(true);
  2376. }break;
  2377. case 1:;
  2378. case 2:{
  2379. self.selectNext();
  2380. stop=self.simpleSlideShow();
  2381. }break;
  2382. };
  2383.  
  2384. function keyUpHandler(e){
  2385. if(e.keyCode!=validKeyCode[index])return;
  2386. container.removeEventListener('keyup',keyUpHandler,false);
  2387. keyDown=false;
  2388. stop();
  2389. };
  2390. container.addEventListener('keyup',keyUpHandler,false);
  2391.  
  2392. },true);
  2393.  
  2394.  
  2395. var imgDraged;
  2396. eleMaps['img-parent'].addEventListener('mousedown',function(e){//如果图片尺寸大于屏幕的时候按住图片进行拖移
  2397. var target=e.target;
  2398. if(e.button!=0 || target.nodeName!='IMG')return;
  2399. var bigger=target.classList.contains('pv-gallery-img_zoom-out');//如果是大于屏幕
  2400.  
  2401. var oClient={
  2402. x:e.clientX,
  2403. y:e.clientY,
  2404. };
  2405.  
  2406. var oScroll={
  2407. left:self.imgScrollbarH.getScrolled(),
  2408. top:self.imgScrollbarV.getScrolled(),
  2409. };
  2410.  
  2411. var moveFiredCount=0;
  2412. var moveHandler=function(e){
  2413. moveFiredCount++;
  2414. if(moveFiredCount<2){//给个缓冲。。
  2415. return;
  2416. };
  2417. imgDraged=true;
  2418. if(bigger){
  2419. target.style.cursor= support.cssCursorValue.grabbing || 'pointer';
  2420. self.imgScrollbarV.scroll(oScroll.top-(e.clientY-oClient.y));
  2421. self.imgScrollbarH.scroll(oScroll.left-(e.clientX-oClient.x));
  2422. };
  2423. };
  2424.  
  2425. var upHandler=function(){
  2426. target.style.cursor='';
  2427.  
  2428. //拖曳之后阻止随后可能产生click事件产生的大小切换。
  2429. //确保在随后的click事件发生后执行
  2430. setTimeout(function(){
  2431. imgDraged=false;
  2432. },0);
  2433.  
  2434. document.removeEventListener('mousemove',moveHandler,true);
  2435. document.removeEventListener('mouseup',upHandler,true);
  2436. };
  2437.  
  2438. document.addEventListener('mousemove',moveHandler,true);
  2439. document.addEventListener('mouseup',upHandler,true);
  2440. },true);
  2441.  
  2442. eleMaps['img-parent'].addEventListener('click',function(e){//点击图片本身就行图片缩放处理
  2443. var target=e.target;
  2444. if(e.button!=0 || target.nodeName!='IMG')return;
  2445.  
  2446. if(imgDraged){//在拖动后触发的click事件,取消掉。免得一拖动完就立即进行的缩放。。。
  2447. imgDraged=false;
  2448. return;
  2449. };
  2450.  
  2451. if(target.classList.contains('pv-gallery-img_zoom-in')){//放大
  2452. self.fitContains=false;
  2453. var zoomX = typeof e.offsetX=='undefined' ? e.layerX : e.offsetX;
  2454. var zoomY = typeof e.offsetY=='undefined' ? e.layerY : e.offsetY;
  2455. var scaleX=zoomX/target.offsetWidth;
  2456. var scaleY=zoomY/target.offsetHeight;
  2457. self.fitToScreen({
  2458. x:scaleX,
  2459. y:scaleY,
  2460. });
  2461. }else if(target.classList.contains('pv-gallery-img_zoom-out')){
  2462. self.fitContains=true;
  2463. self.fitToScreen();
  2464. };
  2465. },true);
  2466.  
  2467.  
  2468. container.addEventListener('mousedown',function(e){//鼠标按在导航上,切换图片
  2469. if(e.button!=0)return;//左键
  2470. var target=e.target;
  2471. if(target.nodeName=='IMG')e.preventDefault();
  2472.  
  2473. var matched=true;
  2474. var stop;
  2475. switch(target){
  2476. case eleMaps['img-controler-pre']:;
  2477. case eleMaps['sidebar-controler-pre']:{//上一个
  2478. self.selectPrevious();
  2479. stop=self.simpleSlideShow(true);
  2480. }break;
  2481. case eleMaps['img-controler-next']:;
  2482. case eleMaps['sidebar-controler-next']:{//下一个
  2483. self.selectNext();
  2484. stop=self.simpleSlideShow();
  2485. }break;
  2486. default:{
  2487. matched=false;
  2488. }break;
  2489. };
  2490.  
  2491. function mouseUpHandler(e){
  2492. document.removeEventListener('mouseup',mouseUpHandler,true);
  2493. stop();
  2494. };
  2495.  
  2496. if(matched){
  2497. e.preventDefault();
  2498. document.addEventListener('mouseup',mouseUpHandler,true);
  2499. };
  2500. },false);
  2501.  
  2502. eleMaps['sidebar-thumbnails-container'].addEventListener('click',function(e){//点击缩略图切换
  2503. if(e.button!=0)return;//左键
  2504. var target=e.target;
  2505. var targetP;
  2506. if(!dataset(target,'src') && (targetP=target.parentNode) && !dataset(targetP,'src'))return;
  2507.  
  2508. self.select(targetP? targetP : target);
  2509. },false);
  2510.  
  2511. //点击读取错误的图片占位符重新读取
  2512. eleMaps['img_broken'].addEventListener('click',function(e){
  2513. if(self.isLoading){
  2514. self.select(self.errorSpan);
  2515. }else{
  2516. self.getImg(self.errorSpan);
  2517. };
  2518. },false);
  2519. if(prefs.gallery.viewmoreLayout==1){
  2520. eleMaps['maximize-container'].classList.add("pv-gallery-flex-maximize");
  2521. }
  2522.  
  2523. if(prefs.gallery.viewmoreEndless){
  2524. addWheelEvent(eleMaps['maximize-container'],function(e){
  2525. if(!self.haveMorePage)return;
  2526. let scrollCon=self.eleMaps['maximize-container'].parentNode;
  2527. let scrollPercent=scrollCon.scrollTop / (scrollCon.scrollHeight - scrollCon.clientHeight);
  2528. if(scrollPercent>0.8){
  2529. var textSpan=self.eleMaps['head-command-nextPage'].querySelector("span");
  2530. if(textSpan.innerHTML==i18n("loading")){
  2531. return;
  2532. }
  2533. textSpan.innerHTML=createHTML(i18n("loading"));
  2534. self.completePages=[];
  2535. self.pageAllReady=false;
  2536. self.pageAction(true, true);
  2537. }
  2538. })
  2539. }
  2540. self.urlFilter="";
  2541.  
  2542. eleMaps['head'].addEventListener('click',function(e){//顶栏上面的命令
  2543. if(e.button!=0)return;
  2544. var target=e.target;
  2545. if(eleMaps['head-command-close']==target){
  2546. self.close();
  2547. }else if(eleMaps['head-command-operate'].contains(target)){
  2548. imgReady(self.src,{
  2549. ready:function(){
  2550. new ImgWindowC(this);
  2551. },
  2552. });
  2553. }else if(eleMaps['head-command-nextPage'].contains(target)){
  2554. var textSpan=eleMaps['head-command-nextPage'].querySelector("span");
  2555. if(textSpan.innerHTML==i18n("loading")){
  2556. textSpan.innerHTML=createHTML(i18n("loadAll"));
  2557. return;
  2558. }
  2559. textSpan.innerHTML=createHTML(i18n("loading"));
  2560. self.completePages=[];
  2561. self.pageAllReady=false;
  2562. self.pageAction(true);
  2563. }else if(eleMaps['head-command-urlFilter'].contains(target)){
  2564. let urlFilterSpan=eleMaps['head-command-urlFilter'];
  2565. let textSpan = urlFilterSpan.querySelector("span");
  2566. if(self.urlFilter){
  2567. self.urlFilter="";
  2568. textSpan.style.color="";
  2569. urlFilterSpan.title=i18n("urlFilterTip");
  2570. self.changeMinView();
  2571. }else{
  2572. let regStr=prompt(i18n("urlFilterTip"));
  2573. if(regStr){
  2574. self.urlFilter=regStr;
  2575. textSpan.style.color="#e9cccc";
  2576. urlFilterSpan.title=regStr;
  2577. self.changeMinView();
  2578. }
  2579. }
  2580. }else if(eleMaps['head-command-collect'].contains(target)){
  2581. if(collection.favorite){
  2582. collection.remove();
  2583. }else{
  2584. collection.add();
  2585. };
  2586. }else if(eleMaps['head-command-exit-collection'].contains(target)){
  2587. collection.exit();
  2588. }else if(eleMaps['head-command-slide-show'].contains(target)){
  2589. slideShow.switchStatus();
  2590. slideShow.check();
  2591. };
  2592.  
  2593. },false);
  2594.  
  2595.  
  2596. //点击还原。
  2597. maximizeTrigger.addEventListener('click',function(e){
  2598. var target=e.target;
  2599. this.style.display='none';
  2600. if(target==this){
  2601. self.show();
  2602. self.resizeHandler();
  2603. }else{
  2604. self.minimized=false;
  2605. };
  2606. },true);
  2607.  
  2608.  
  2609. this._resizeHandler=this.resizeHandler.bind(this);
  2610. this._keyDownListener=this.keyDownListener.bind(this);
  2611. this._keyUpListener=this.keyUpListener.bind(this);
  2612.  
  2613. //插入动态生成的css数据。
  2614. this.globalSSheet.insertRule('.pv-gallery-sidebar-thumb-container{'+
  2615. ((isHorizontal ? 'width' : 'height') + ':' + (isHorizontal ? unsafeWindow.getComputedStyle(eleMaps['sidebar-thumbnails-container']).height : unsafeWindow.getComputedStyle(eleMaps['sidebar-thumbnails-container']).width)) +
  2616. '}',this.globalSSheet.cssRules.length);
  2617.  
  2618. this.forceRepaintTimes=0;
  2619.  
  2620. container.style.display='none';
  2621. this.shown=false;
  2622.  
  2623. // 我添加的部分
  2624. this.initToggleBar();
  2625. this.initZoom();
  2626. },
  2627. batchDownload:function(saveParams, callback){
  2628. var self=this;
  2629. if(prefs.gallery.downloadWithZip){
  2630. _GM_notification(i18n("galleryDownloadWithZipAlert"));
  2631. var zip = new JSZip(),downloaded=0;
  2632. var fileName = document.title + ".zip";
  2633. for(let i=0; i<saveParams.length; i++){
  2634. self.dataURLToCanvas(saveParams[i][0], canvas=>{
  2635. if(!canvas){
  2636. downloaded++;
  2637. if(downloaded == saveParams.length){
  2638. zip.generateAsync({type:"blob"}).then(function(content){
  2639. saveAs(content, fileName);
  2640. callback();
  2641. })
  2642. }
  2643. return;
  2644. }
  2645. canvas.toBlob(blob=>{
  2646. zip.file(saveParams[i][1].replace(/\.[^\.]+$/,"")+'.jpg',blob);
  2647. downloaded++;
  2648. if(downloaded == saveParams.length){
  2649. zip.generateAsync({type:"blob"}).then(function(content){
  2650. saveAs(content, fileName);
  2651. callback();
  2652. })
  2653. }
  2654. }, "image/jpg");
  2655. });
  2656. };
  2657. return;
  2658. }
  2659.  
  2660. let download5Times=function(){
  2661. for(let i=0;i<5;i++){
  2662. let saveParam=saveParams.shift();
  2663. if(saveParam){
  2664. _GM_download(saveParam[0], saveParam[1]);
  2665. }else{
  2666. callback();
  2667. break;
  2668. }
  2669. }
  2670. if(saveParams.length>0){
  2671. setTimeout(()=>{
  2672. download5Times();
  2673. },1000);
  2674. }
  2675. };
  2676. download5Times();
  2677. },
  2678. changeMinView:function(){
  2679. var urlReg=new RegExp(this.urlFilter);
  2680. var sizeInputH=this.gallery.querySelector("#minsizeH");
  2681. var sizeInputW=this.gallery.querySelector("#minsizeW");
  2682. var sizeInputHSpan=this.gallery.querySelector("#minsizeHSpan");
  2683. var sizeInputWSpan=this.gallery.querySelector("#minsizeWSpan");
  2684. sizeInputH.title=sizeInputH.value+"px";
  2685. sizeInputHSpan.innerHTML=createHTML(Math.floor(sizeInputH.value)+"px");
  2686. sizeInputW.title=sizeInputW.value+"px";
  2687. sizeInputWSpan.innerHTML=createHTML(Math.floor(sizeInputW.value)+"px");
  2688.  
  2689. var self=this;
  2690. var viewmoreShow = this.eleMaps['sidebar-toggle'].style.visibility == 'hidden';
  2691. if(viewmoreShow){
  2692. var maxSizeH=0,minSizeH=0,maxSizeW=0,minSizeW=0;
  2693. [].forEach.call(document.querySelectorAll(".maximizeChild>img"),function(item){
  2694. var spanMark;
  2695. try{
  2696. spanMark=document.querySelector('span.pv-gallery-sidebar-thumb-container[data-src="'+item.src+'"]');
  2697. }catch(e){
  2698. spanMark=document.querySelector("span.pv-gallery-sidebar-thumb-container[data-src='"+item.src+"']");
  2699. }
  2700. if(spanMark && !spanMark.dataset.naturalSize && item.naturalWidth && item.naturalHeight){
  2701. spanMark.dataset.naturalSize=JSON.stringify({w:item.naturalWidth,h:item.naturalHeight});
  2702. }
  2703. if(item.naturalWidth<sizeInputW.value || item.naturalHeight<sizeInputH.value || !urlReg.test(item.src) || (self.lockMaxSize && (item.naturalWidth>self.lockMaxSize.w || item.naturalHeight>self.lockMaxSize.h))){
  2704. item.parentNode.style.display="none";
  2705. if(spanMark)spanMark.style.display="none";
  2706. }else{
  2707. item.parentNode.style.display="";
  2708. if(spanMark)spanMark.style.display="";
  2709. }
  2710. if(item.naturalHeight>maxSizeH)
  2711. maxSizeH=item.naturalHeight;
  2712. if(item.naturalHeight<minSizeH || minSizeH==0)
  2713. minSizeH=item.naturalHeight;
  2714. if(item.naturalWidth>maxSizeW)
  2715. maxSizeW=item.naturalWidth;
  2716. if(item.naturalWidth<minSizeW || minSizeW==0)
  2717. minSizeW=item.naturalWidth;
  2718. });
  2719. sizeInputH.max=maxSizeH;
  2720. sizeInputH.min=minSizeH;
  2721. sizeInputH.title=sizeInputH.value+"px";
  2722. sizeInputHSpan.innerHTML=createHTML(Math.floor(sizeInputH.value)+"px");
  2723.  
  2724. sizeInputW.max=maxSizeW;
  2725. sizeInputW.min=minSizeW;
  2726. sizeInputW.title=sizeInputW.value+"px";
  2727. sizeInputWSpan.innerHTML=createHTML(Math.floor(sizeInputW.value)+"px");
  2728. }else{
  2729. this.data.forEach(function(item) {
  2730. if(!item)return;
  2731. var spanMark=document.querySelector("span.pv-gallery-sidebar-thumb-container[data-thumb-src='"+item.imgSrc+"']");
  2732. if(spanMark){
  2733. var naturalSize=spanMark.dataset.naturalSize,itemW=item.sizeW,itemH=item.sizeH;
  2734. if(naturalSize){
  2735. naturalSize=JSON.parse(naturalSize);
  2736. itemW=naturalSize.w;
  2737. itemH=naturalSize.h;
  2738. if(itemW>sizeInputW.max)sizeInputW.max=itemW;
  2739. if(itemH>sizeInputH.max)sizeInputH.max=itemH;
  2740. }else if(!item.noActual){
  2741. //itemW=99999;
  2742. //itemH=99999;
  2743. }
  2744. if(itemW<sizeInputW.value || itemH<sizeInputH.value || !urlReg.test(item.src) || (self.lockMaxSize && (itemW>self.lockMaxSize.w || itemH>self.lockMaxSize.h))){
  2745. spanMark.style.display="none";
  2746. }else{
  2747. spanMark.style.display="";
  2748. }
  2749. }
  2750. });
  2751. this.switchThumbVisible();
  2752. }
  2753. },
  2754. changeSizeInputReset:function(){
  2755. var maxSizeH=0,minSizeH=0,maxSizeW=0,minSizeW=0;
  2756. var sizeInputH=this.gallery.querySelector("#minsizeH");
  2757. var sizeInputW=this.gallery.querySelector("#minsizeW");
  2758. let self=this;
  2759. this.data.forEach(function(item) {
  2760. if(!item)return;
  2761. var itemW=item.sizeW,itemH=item.sizeH;
  2762. var spanMark=self._spanMarkPool[item.imgSrc];
  2763. if(spanMark){
  2764. var naturalSize=spanMark.dataset.naturalSize;
  2765. if(naturalSize){
  2766. naturalSize=JSON.parse(naturalSize);
  2767. itemW=naturalSize.w;
  2768. itemH=naturalSize.h;
  2769. }
  2770. }
  2771. if(itemH>maxSizeH)
  2772. maxSizeH=itemH;
  2773. if(itemH<minSizeH || minSizeH==0)
  2774. minSizeH=itemH;
  2775. if(itemW>maxSizeW)
  2776. maxSizeW=itemW;
  2777. if(itemW<minSizeW || minSizeW==0)
  2778. minSizeW=itemW;
  2779. });
  2780. sizeInputH.max=maxSizeH;
  2781. sizeInputH.min=minSizeH;
  2782. sizeInputH.value=prefs.gallery.defaultSizeLimit.h;
  2783. sizeInputH.title=sizeInputH.value+"px";
  2784. var sizeInputHSpan=this.gallery.querySelector("#minsizeHSpan");
  2785. sizeInputHSpan.innerHTML=createHTML(Math.floor(sizeInputH.value)+"px");
  2786.  
  2787. sizeInputW.max=maxSizeW;
  2788. sizeInputW.min=minSizeW;
  2789. sizeInputW.value=prefs.gallery.defaultSizeLimit.w;
  2790. sizeInputW.title=sizeInputW.value+"px";
  2791. var sizeInputWSpan=this.gallery.querySelector("#minsizeWSpan");
  2792. sizeInputWSpan.innerHTML=createHTML(Math.floor(sizeInputW.value)+"px");
  2793. },
  2794. initToggleBar: function() { // 是否显示切换 sidebar 按钮
  2795. /**
  2796. * TODO:仿造下面的链接重新改造过?
  2797. * http://image.baidu.com/detail/newindex?col=%E8%B5%84%E8%AE%AF&tag=%E4%BD%93%E8%82%B2&pn=0&pid=5123662821688142478&aid=&user_id=10086&setid=-1&sort=0&newsPn=4&star=&fr=hotword&from=1
  2798. */
  2799. if (prefs.gallery.sidebarToggle) {
  2800. var toggleBar = this.eleMaps['sidebar-toggle'];
  2801. toggleBar.style.display = 'block';
  2802. toggleBar.style.height = '16px';
  2803. toggleBar.addEventListener('click', this.showHideBottom.bind(this), false);
  2804.  
  2805. var viewmoreBar = this.eleMaps['sidebar-viewmore'];
  2806. viewmoreBar.style.display = 'block';
  2807. viewmoreBar.addEventListener('click', this.maximizeSidebar.bind(this), false);
  2808.  
  2809. // 顶部圆角
  2810. switch (prefs.gallery.sidebarPosition) {
  2811. case 'bottom':
  2812. toggleBar.style.borderRadius = '8px 8px 0 0'; // 左上、右上、右下、左下
  2813. break;
  2814. case 'top':
  2815. toggleBar.style.borderRadius = '0 0 8px 8px';
  2816. break;
  2817. case 'left':
  2818. toggleBar.style.height = '60px';
  2819. toggleBar.style.borderRadius = '0 8px 8px 0';
  2820. break;
  2821. case 'right':
  2822. toggleBar.style.height = '60px';
  2823. toggleBar.style.borderRadius = '8px 0 0 8px';
  2824. break;
  2825. }
  2826. }
  2827. },
  2828. closeViewMore: function() {
  2829. var toggleBar = this.eleMaps['sidebar-toggle'],
  2830. imgCon = this.eleMaps['img-container'],
  2831. viewmoreBar = this.eleMaps['sidebar-viewmore-content'],
  2832. imgPre = this.eleMaps['img-controler-pre'],
  2833. imgNext = this.eleMaps['img-controler-next'],
  2834. alreadyShow = toggleBar.style.visibility == 'hidden';
  2835. if(!alreadyShow) return;
  2836. var sidebarContainer = this.eleMaps['sidebar-container'];
  2837. var maximizeContainer = this.eleMaps['maximize-container'];
  2838. var sidebarPosition = prefs.gallery.sidebarPosition,
  2839. capitalize = function(string) { // 将字符串中每个单词首字母大写
  2840. var words = string.split(" ");
  2841. for (var i = 0; i < words.length; i++) {
  2842. words[i] = words[i].charAt(0).toUpperCase() + words[i].slice(1);
  2843. }
  2844. return words.join(" ");
  2845. };
  2846. maximizeContainer.style.minHeight = 0;
  2847. maximizeContainer.parentNode.style.visibility = "hidden";
  2848. if(this.hideBodyStyle.parentNode)
  2849. this.hideBodyStyle.parentNode.removeChild(this.hideBodyStyle);
  2850. imgPre.style.visibility = imgNext.style.visibility = toggleBar.style.visibility = sidebarContainer.style.visibility = 'visible';
  2851. imgCon.style['border' + capitalize(sidebarPosition)] = prefs.gallery.sidebarSize + 'px solid transparent';
  2852. toggleBar.style[sidebarPosition] = '-5px';
  2853. while (maximizeContainer.firstChild) {
  2854. maximizeContainer.removeChild(maximizeContainer.firstChild);
  2855. }
  2856. viewmoreBar.innerHTML = createHTML('✚');
  2857. viewmoreBar.parentNode.classList.remove("showmore");
  2858. //viewmoreBar.parentNode.style.backgroundColor = "#000000";
  2859.  
  2860. toggleBar.innerHTML = createHTML('▼');
  2861. this.changeSizeInputReset();
  2862. },
  2863. selectViewmore: function(imgSpan, src) {
  2864. if(this.curImgSpan)this.curImgSpan.classList.remove("selected");
  2865. imgSpan.classList.add("selected");
  2866. this.curImgSpan=imgSpan;
  2867. var node = document.querySelector('.pv-gallery-sidebar-thumb-container[data-src="'+src+'"]');
  2868. if(node)this.select(node);
  2869. },
  2870. addViewmoreItem: function(nodes) {
  2871. var alreadyShow = this.eleMaps['sidebar-toggle'].style.visibility == 'hidden';
  2872. if(!alreadyShow)return;
  2873. var self=this;
  2874. var maximizeContainer = this.eleMaps['maximize-container'];
  2875. [].forEach.call(nodes, function(node){
  2876. var nodeStyle=unsafeWindow.getComputedStyle(node);
  2877. let curNode=node;
  2878. let imgSpan = document.createElement('span');
  2879. if(nodeStyle.display=="none")imgSpan.style.display="none";
  2880. imgSpan.className = "maximizeChild";
  2881. imgSpan.innerHTML = createHTML('<img src="'+curNode.dataset.src+'">');
  2882. imgSpan.addEventListener("click", function(e){
  2883. imgReady(curNode.dataset.src,{
  2884. ready:function(){
  2885. let imgwin=new ImgWindowC(this);
  2886. self.selectViewmore(imgSpan, curNode.dataset.src);
  2887. if(prefs.imgWindow.overlayer.shown){
  2888. imgwin.blur(true);
  2889. self.curImgWin=imgwin;
  2890. self.curImgSpan=imgSpan;
  2891. if(!self.scrollInit){
  2892. self.scrollInit=true;
  2893. let wheelHandler=function(e){
  2894. if(self.canScroll && self.curImgWin && !self.curImgWin.removed && !self.curImgWin.focused){
  2895. self.canScroll=false;
  2896. let targetImgSpan=self.curImgSpan;
  2897. while(targetImgSpan){
  2898. targetImgSpan=e.deltaY<0?targetImgSpan.previousElementSibling:targetImgSpan.nextElementSibling;
  2899. if(targetImgSpan && targetImgSpan.style.display!="none" && targetImgSpan.clientWidth>1)break;
  2900. }
  2901. if(targetImgSpan){
  2902. self.curImgWin.remove();
  2903. let curImgEle=document.createElement("img");
  2904. curImgEle.src=targetImgSpan.querySelector("img").src;
  2905. let imgwin=new ImgWindowC(curImgEle);
  2906. imgwin.blur(true);
  2907. self.curImgWin=imgwin;
  2908. self.selectViewmore(targetImgSpan, curImgEle.src);
  2909. targetImgSpan.scrollIntoView({block: "nearest", inline: "nearest"});
  2910. self.canScroll=true;
  2911. /*imgReady(targetImgSpan.querySelector("img").src,{
  2912. ready:function(){
  2913. self.curImgWin.remove(true);
  2914. let imgwin=new ImgWindowC(this);
  2915. imgwin.blur(true);
  2916. self.curImgWin.imgWindow.style.opacity=0;
  2917. self.curImgWin=imgwin;
  2918. self.curImgSpan=targetImgSpan;
  2919. self.canScroll=true;
  2920. }
  2921. });*/
  2922. }else{
  2923. self.canScroll=true;
  2924. }
  2925. }
  2926. };
  2927. addWheelEvent(document.body,wheelHandler,true);
  2928. }
  2929. }
  2930. }
  2931. });
  2932. });
  2933. imgSpan.title=curNode.title;
  2934. let img=imgSpan.querySelector("img");
  2935. var addDlSpan=(img, imgSpan, curNode, clickCb)=>{
  2936. var dlSpan=document.createElement('p');
  2937. dlSpan.className="pv-bottom-banner";
  2938. dlSpan.innerHTML=createHTML('<svg class="icon" style="width: 20px;height: 20px;vertical-align: middle;fill: currentColor;overflow: hidden;" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="1100"><path d="M768 768q0-14.857143-10.857143-25.714286t-25.714286-10.857143-25.714285 10.857143-10.857143 25.714286 10.857143 25.714286 25.714285 10.857143 25.714286-10.857143 10.857143-25.714286z m146.285714 0q0-14.857143-10.857143-25.714286t-25.714285-10.857143-25.714286 10.857143-10.857143 25.714286 10.857143 25.714286 25.714286 10.857143 25.714285-10.857143 10.857143-25.714286z m73.142857-128v182.857143q0 22.857143-16 38.857143t-38.857142 16H91.428571q-22.857143 0-38.857142-16t-16-38.857143v-182.857143q0-22.857143 16-38.857143t38.857142-16h265.714286l77.142857 77.714286q33.142857 32 77.714286 32t77.714286-32l77.714285-77.714286h265.142858q22.857143 0 38.857142 16t16 38.857143z m-185.714285-325.142857q9.714286 23.428571-8 40l-256 256q-10.285714 10.857143-25.714286 10.857143t-25.714286-10.857143L230.285714 354.857143q-17.714286-16.571429-8-40 9.714286-22.285714 33.714286-22.285714h146.285714V36.571429q0-14.857143 10.857143-25.714286t25.714286-10.857143h146.285714q14.857143 0 25.714286 10.857143t10.857143 25.714286v256h146.285714q24 0 33.714286 22.285714z" p-id="1101"></path></svg> '+i18n("download"));
  2939. dlSpan.src=curNode.dataset.src;
  2940. dlSpan.title=curNode.title||document.title;
  2941. dlSpan.onclick=clickCb;
  2942. var topP=document.createElement('p');
  2943. topP.className="pv-top-banner";
  2944. topP.innerHTML=createHTML(img.naturalWidth+' x '+img.naturalHeight);
  2945. topP.title=img.src;
  2946. var checkBox=document.createElement('input');
  2947. checkBox.type="checkbox";
  2948. checkBox.onclick=function(e){
  2949. let checkBoxs=maximizeContainer.querySelectorAll(".maximizeChild>input:checked");
  2950. if(!self.batchDl || self.batchDl.parentNode!=maximizeContainer){
  2951. self.batchDl=document.createElement('p');
  2952. let batchDlBtn=document.createElement('input');
  2953. let cancelBtn=document.createElement('input');
  2954. let invertBtn=document.createElement('input');
  2955. batchDlBtn.value=i18n("download");
  2956. cancelBtn.value=i18n("closeBtn");
  2957. invertBtn.value=i18n("invertBtn");
  2958. batchDlBtn.type="button";
  2959. cancelBtn.type="button";
  2960. invertBtn.type="button";
  2961. batchDlBtn.onclick=function(e){
  2962. checkBoxs=maximizeContainer.querySelectorAll(".maximizeChild>input:checked");
  2963. if(checkBoxs.length<1)return;
  2964.  
  2965. var saveParams = [],saveIndex=0;
  2966. [].forEach.call(checkBoxs, function(node){
  2967. let conItem=node.parentNode;
  2968. if(conItem.style.display=="none")return;
  2969. saveIndex++;
  2970. let imgSrc=conItem.querySelector("img").src;
  2971. let title=node.nextElementSibling.title;
  2972. let srcSplit=imgSrc.replace(/[\?#].*/,"").split("/");
  2973. srcSplit=srcSplit[srcSplit.length-1];
  2974. if(srcSplit.length>30 && (srcSplit.indexOf(".")==-1 || /[&\?=,]/i.test(srcSplit))){
  2975. srcSplit="";
  2976. }
  2977. var picName=document.title + "-" + (saveIndex<10?"00"+saveIndex:(saveIndex<100?"0"+saveIndex:saveIndex)) + (title==document.title?"":"-" + title) + "-" + srcSplit;
  2978. saveParams.push([imgSrc, picName]);
  2979. });
  2980. self.batchDownload(saveParams, ()=>{
  2981. });
  2982. };
  2983. cancelBtn.onclick=function(e){
  2984. checkBoxs=maximizeContainer.querySelectorAll(".maximizeChild>input:checked");
  2985. if(checkBoxs.length<1)return;
  2986. [].forEach.call(checkBoxs, i=>{
  2987. i.checked=false;
  2988. });
  2989. maximizeContainer.removeChild(self.batchDl);
  2990. maximizeContainer.classList.remove("checked");
  2991. };
  2992. invertBtn.onclick=function(e){
  2993. checkBoxs=maximizeContainer.querySelectorAll(".maximizeChild>input");
  2994. if(checkBoxs.length<1)return;
  2995. [].forEach.call(checkBoxs, i=>{
  2996. let conItem=i.parentNode;
  2997. if(conItem.style.display=="none")i.checked=false;
  2998. else i.checked=!i.checked;
  2999. });
  3000. checkBoxs=maximizeContainer.querySelectorAll(".maximizeChild>input:checked");
  3001. if(checkBoxs.length==0){
  3002. maximizeContainer.removeChild(self.batchDl);
  3003. maximizeContainer.classList.remove("checked");
  3004. }
  3005. };
  3006. self.batchDl.appendChild(batchDlBtn);
  3007. self.batchDl.appendChild(cancelBtn);
  3008. self.batchDl.appendChild(invertBtn);
  3009. maximizeContainer.appendChild(self.batchDl);
  3010. }
  3011. if(checkBoxs.length>0){
  3012. maximizeContainer.appendChild(self.batchDl);
  3013. maximizeContainer.classList.add("checked");
  3014. }else{
  3015. maximizeContainer.removeChild(self.batchDl);
  3016. maximizeContainer.classList.remove("checked");
  3017. }
  3018. e.stopPropagation();
  3019. };
  3020. imgSpan.appendChild(topP);
  3021. imgSpan.appendChild(checkBox);
  3022. imgSpan.appendChild(dlSpan);
  3023. };
  3024. fetch(curNode.dataset.src).then(response=>{
  3025. return response.blob();
  3026. }).then(blob=>{
  3027. imgReady(img,{
  3028. ready:function(){
  3029. if(img.width>=88 && img.height>=88){
  3030. addDlSpan(img, imgSpan, curNode, e=>{
  3031. e.stopPropagation();
  3032. if(blob.type=="image/webp"){
  3033. self.blobToCanvas(blob, canvas=>{
  3034. canvas.toBlob(blob=>{
  3035. saveAs(blob,e.target.title);
  3036. }, "image/png");
  3037. });
  3038. }else{
  3039. _GM_download(e.target.src,e.target.title);
  3040. }
  3041. return true;
  3042. });
  3043. }
  3044. }
  3045. });
  3046. }).catch(e=>{
  3047. imgReady(img,{
  3048. ready:function(){
  3049. if(img.width>=88 && img.height>=88){
  3050. addDlSpan(img, imgSpan, curNode, e=>{
  3051. e.stopPropagation();
  3052. _GM_download(e.target.src,e.target.title);
  3053. return true;
  3054. });
  3055. }
  3056. }
  3057. });
  3058. });
  3059. maximizeContainer.appendChild(imgSpan);
  3060. });
  3061. },
  3062. maximizeSidebar: function() {
  3063. var toggleBar = this.eleMaps['sidebar-toggle'],
  3064. imgCon = this.eleMaps['img-container'],
  3065. viewmoreBar = this.eleMaps['sidebar-viewmore-content'],
  3066. imgPre = this.eleMaps['img-controler-pre'],
  3067. imgNext = this.eleMaps['img-controler-next'],
  3068. alreadyShow = toggleBar.style.visibility == 'hidden';
  3069. var sidebarContainer = this.eleMaps['sidebar-container'];
  3070. var maximizeContainer = this.eleMaps['maximize-container'];
  3071. var sidebarPosition = prefs.gallery.sidebarPosition,
  3072. capitalize = function(string) { // 将字符串中每个单词首字母大写
  3073. var words = string.split(" ");
  3074. for (var i = 0; i < words.length; i++) {
  3075. words[i] = words[i].charAt(0).toUpperCase() + words[i].slice(1);
  3076. }
  3077. return words.join(" ");
  3078. };
  3079. if(alreadyShow){
  3080. this.closeViewMore();
  3081. }else{
  3082. maximizeContainer.style.minHeight = "100%";
  3083. maximizeContainer.parentNode.style.visibility = "visible";
  3084. document.head.appendChild(this.hideBodyStyle);
  3085. imgPre.style.visibility = imgNext.style.visibility = toggleBar.style.visibility = sidebarContainer.style.visibility = 'hidden';
  3086. imgCon.style['border' + capitalize(sidebarPosition)] = '0';
  3087. toggleBar.style[sidebarPosition] = '0';
  3088. maximizeContainer.innerHTML = createHTML("");
  3089. viewmoreBar.innerHTML = createHTML('✖');
  3090. viewmoreBar.parentNode.classList.add("showmore");//.backgroundColor = "#2a2a2a";
  3091.  
  3092. var nodes = document.querySelectorAll('.pv-gallery-sidebar-thumb-container[data-src]');
  3093. this.addViewmoreItem(nodes);
  3094. }
  3095. },
  3096. dataURLToCanvas:function (dataurl, cb){
  3097. if(!dataurl)return cb(null);
  3098. var canvas = document.createElement('CANVAS');
  3099. var ctx = canvas.getContext('2d');
  3100. var img = new Image();
  3101. img.setAttribute("crossOrigin","anonymous");
  3102. img.onload = function(){
  3103. canvas.width = img.width;
  3104. canvas.height = img.height;
  3105. ctx.drawImage(img, 0, 0);
  3106. cb(canvas);
  3107. };
  3108. img.onerror = function(){
  3109. cb(null);
  3110. };
  3111. img.src = dataurl;
  3112. },
  3113. blobToDataURL:function(blob, cb){
  3114. var a = new FileReader();
  3115. a.readAsDataURL(blob);
  3116. a.onload = function (e){
  3117. cb(e.target.result);
  3118. };
  3119. a.onerror = function (e){
  3120. cb(null);
  3121. }
  3122. },
  3123. blobToCanvas: function (blob, cb){
  3124. var self=this;
  3125. this.blobToDataURL(blob, function (dataurl){
  3126. self.dataURLToCanvas(dataurl, cb);
  3127. });
  3128. },
  3129. showHideBottom: function() { // 显示隐藏 sidebar-container
  3130. var sidebarContainer = this.eleMaps['sidebar-container'],
  3131. isHidden = sidebarContainer.style.visibility == 'hidden';
  3132.  
  3133. sidebarContainer.style.visibility = isHidden ? 'visible' : 'hidden';
  3134.  
  3135. var sidebarPosition = prefs.gallery.sidebarPosition,
  3136. capitalize = function(string) { // 将字符串中每个单词首字母大写
  3137. var words = string.split(" ");
  3138. for (var i = 0; i < words.length; i++) {
  3139. words[i] = words[i].charAt(0).toUpperCase() + words[i].slice(1);
  3140. }
  3141. return words.join(" ");
  3142. };
  3143.  
  3144. // 修正下图片底部的高度
  3145. this.eleMaps['img-container'].style['border' + capitalize(sidebarPosition)] = isHidden ?
  3146. prefs.gallery.sidebarSize + 'px solid transparent' :
  3147. '0';
  3148. // 修正底部距离
  3149. this.eleMaps['sidebar-toggle'].style[sidebarPosition] = isHidden ? '-5px' : '0';
  3150. this.eleMaps['sidebar-toggle'].innerHTML = createHTML(isHidden ? '▼' : '▲');
  3151. this.eleMaps['sidebar-viewmore'].style.visibility = isHidden ? 'visible' : 'hidden';
  3152. },
  3153. initZoom: function() { // 如果有放大,则把图片及 sidebar 部分缩放比率改为 1
  3154. if (prefs.gallery.autoZoom && document.body.style.zoom != undefined) {
  3155. var oZoom = detectZoom();
  3156. if (oZoom > 100) {
  3157. this.eleMaps['body'].style.zoom = 100 / oZoom;
  3158. }
  3159. }
  3160. },
  3161.  
  3162. getThumSpan:function(previous,relatedTarget){
  3163. var ret;
  3164. var rt = relatedTarget || this.selected;
  3165. if(!rt)return;
  3166. while((rt=previous ? rt.previousElementSibling : rt.nextElementSibling)){
  3167. if(rt.clientWidth!=0){
  3168. ret=rt;
  3169. break;
  3170. };
  3171. };
  3172. return ret;
  3173. },
  3174. previous:false,
  3175. selectPrevious:function(){
  3176. this.previous=true;
  3177. this.select(this.getThumSpan(true));
  3178. },
  3179. selectNext:function(){
  3180. this.select(this.getThumSpan());
  3181. },
  3182. select:function(ele,noTransition){
  3183. if(!ele || this.selected==ele)return;
  3184. if(this.selected){
  3185. this.selected.classList.remove(this.selectedClassName);
  3186. this.selected.classList.remove('pv-gallery-sidebar-thumb_selected');
  3187. }
  3188. ele.classList.add(this.selectedClassName);
  3189. ele.classList.add('pv-gallery-sidebar-thumb_selected');
  3190.  
  3191. this.selected=ele;
  3192. this.arrowVisib();
  3193.  
  3194. var self=this;
  3195. clearTimeout(this.loadImgTimer);
  3196. if(prefs.gallery.transition){
  3197. this.loadImgTimer=setTimeout(function(){//快速跳转的时候不要尝试读取图片。
  3198. self.loadImg(ele);
  3199. },200);
  3200. }else{
  3201. self.loadImg(ele);
  3202. }
  3203.  
  3204. this.selectedIntoView(noTransition);
  3205. this.forceRepaint();
  3206. this.slideShow.run('select');
  3207. },
  3208. loadThumb:function(){//读取可视范围里面的缩略图
  3209. var self=this;
  3210. var pro=self.isHorizontal ? ['scrollLeft','clientWidth','offsetLeft','offsetWidth'] : ['scrollTop','clientHeight','offsetTop','offsetHeight'];
  3211. var thumbC=self.eleMaps['sidebar-thumbnails-container'];
  3212.  
  3213. var scrolled=thumbC[pro[0]];
  3214.  
  3215. var loadStopDis=scrolled + thumbC[pro[1]];
  3216.  
  3217. var imgSpans=self.selected.parentNode.children;
  3218. var span_i;
  3219. var spanOffset;
  3220. var thumb;
  3221.  
  3222. var i=0
  3223. while(span_i=imgSpans[i++]){
  3224. if(span_i.clientWidth==0)continue;//隐藏的
  3225.  
  3226. spanOffset=span_i[pro[2]];
  3227. if(spanOffset + span_i[pro[3]] <= scrolled)continue;//在滚动条上面了
  3228. if(spanOffset >= loadStopDis)break;//在滚动条下面了
  3229.  
  3230. if(dataset(span_i,'thumbLoaded'))continue;//已经加载了缩略图
  3231.  
  3232. thumb=document.createElement('img');
  3233. thumb.src=dataset(span_i,'thumbSrc') || dataset(span_i,'src') || prefs.icons.brokenImg_small;
  3234. thumb.className='pv-gallery-sidebar-thumb';
  3235.  
  3236. dataset(span_i,'thumbLoaded','true');
  3237. span_i.appendChild(thumb);
  3238.  
  3239. imgReady(thumb,{
  3240. error:function(e){
  3241. this.src=prefs.icons.brokenImg_small;
  3242. },
  3243. });
  3244. };
  3245. },
  3246. selectedIntoView:function(noTransition){
  3247. var thumBC=this.eleMaps['sidebar-thumbnails-container'];
  3248. var pro=this.isHorizontal ? ['offsetLeft','clientWidth','offsetWidth'] : ['offsetTop','clientHeight','offsetHeight'] ;
  3249. //需要滚动的距离。
  3250. var needScrollDis= this.selected[pro[0]];
  3251. //尽可能的居中显示
  3252. var thumBCClient=thumBC[pro[1]];
  3253. var scrollCenter=Math.max((thumBCClient - this.selected[pro[2]])/2,0);
  3254.  
  3255. this.thumbScrollbar.scroll(needScrollDis - scrollCenter,false,!noTransition);
  3256. },
  3257. getImg:function(ele){
  3258. var self = this;
  3259.  
  3260. var src = dataset(ele,'src');
  3261.  
  3262. this.lastLoading=src;//记住最后读取的图片
  3263. this.isLoading=true;//表示选择的图片正在读取
  3264.  
  3265. // 特殊的 xhr 方式获取
  3266. var xhr = dataset(ele, 'xhr');
  3267. if (xhr) {
  3268. var xhrError = function() {
  3269. dataset(ele, 'xhr', '');
  3270. dataset(ele, 'src', dataset(ele, 'thumb-src'));
  3271. self.getImg(ele);
  3272. };
  3273. xhrLoad.load({
  3274. url: src,
  3275. xhr: JSON.parse(decodeURIComponent(xhr)),
  3276. cb: function(imgSrc, imgSrcs, caption) {
  3277. if (imgSrc) {
  3278. dataset(ele, 'src', imgSrc);
  3279. dataset(ele, 'xhr', '');
  3280. self.getImg(ele);
  3281. } else {
  3282. xhrError();
  3283. }
  3284. },
  3285. onerror: xhrError
  3286. });
  3287. return;
  3288. }
  3289.  
  3290. var allLoading=this.allLoading;
  3291. if(allLoading.indexOf(src)!=-1){//在读取队列中。
  3292. return;
  3293. };
  3294. allLoading.push(src);
  3295.  
  3296. //上一个读取中的图片,不是当前显示的。那么直接终止
  3297. var preImgR=this.imgReady;
  3298. if(preImgR && this.img){
  3299. if(preImgR.img.src!=this.src){
  3300. preImgR.abort();
  3301. preImgR.removeLI();
  3302. };
  3303. };
  3304.  
  3305.  
  3306. //显示读取指示器。
  3307. var loadingIndicator=ele.querySelector('.pv-gallery-sidebar-thumb-loading');
  3308. loadingIndicator.style.display='block';
  3309.  
  3310.  
  3311. this.imgReady=imgReady(src,{
  3312. ready:function(){
  3313. //从读取队列中删除自己
  3314. var index=allLoading.indexOf(src);
  3315. if(index!=-1){
  3316. allLoading.splice(index,1);
  3317. };
  3318.  
  3319. if(src!=self.lastLoading)return;
  3320.  
  3321. loadingIndicator.style.display='';
  3322. if(preImgR)preImgR.abort();
  3323. self.loadImg(this,ele);
  3324. },
  3325. loadEnd:function(e){//在loadend后开始预读。
  3326. //从读取队列中删除自己
  3327. var index=allLoading.indexOf(src);
  3328. if(index!=-1){
  3329. allLoading.splice(index,1);
  3330. };
  3331.  
  3332. if(src!=self.lastLoading)return;
  3333.  
  3334. if(e.type=='error'){
  3335. loadingIndicator.style.display='';
  3336. self.errorSpan=ele;
  3337. if(preImgR)preImgR.abort();
  3338. self.loadImg(this,ele,true);
  3339. };
  3340.  
  3341. self.slideShow.run('loadEnd');
  3342.  
  3343. if(prefs.gallery.preload){
  3344. if(self.preloading){//结束上次的预读。
  3345. self.preloading.abort();
  3346. };
  3347. self.preloading=new self.Preload(ele,self);
  3348. self.preloading.preload();
  3349. };
  3350. },
  3351. });
  3352.  
  3353. this.imgReady.removeLI=function(){
  3354. loadingIndicator.style.display='';
  3355. };
  3356.  
  3357. },
  3358. loadImg:function(img,relatedThumb,error){
  3359. if(img.nodeName!='IMG'){//先读取。
  3360. this.getImg(img);
  3361. return;
  3362. };
  3363.  
  3364. if(this.img){
  3365. this.img.parentNode.removeChild(this.img);
  3366. };
  3367.  
  3368. var imgNaturalSize={
  3369. h:img.naturalHeight,
  3370. w:img.naturalWidth,
  3371. };
  3372. this.imgNaturalSize=imgNaturalSize;
  3373.  
  3374. this.eleMaps['head-left-img-info-resolution'].textContent=imgNaturalSize.w + ' x ' + imgNaturalSize.h;
  3375. var thumbnails=this.eleMaps['sidebar-thumbnails-container'].childNodes,i=0;
  3376. while(thumbnails[i]!=relatedThumb && i<thumbnails.length)i++;
  3377. if(i<thumbnails.length)this.eleMaps['head-left-img-info-count'].textContent="("+(i+1)+" / "+thumbnails.length+")";
  3378. // 加上图片的注释
  3379. var description = decodeURIComponent(dataset(relatedThumb, 'description')),
  3380. defaultLength = prefs.gallery.descriptionLength;
  3381. this.eleMaps['head-left-img-info-description'].title = description;
  3382. this.eleMaps['head-left-img-info-description'].textContent= description.length > defaultLength ?
  3383. description.slice(0, defaultLength) + '...' :
  3384. description;
  3385.  
  3386. this.img=img;
  3387. this.img.title=description;
  3388. this.src=img.src;
  3389. this.isLoading=false;
  3390.  
  3391. this.relatedThumb=relatedThumb;
  3392. img.className='pv-gallery-img';
  3393.  
  3394. if(error){
  3395. if(relatedThumb.querySelector("img").src==this.img.src){
  3396. this.imgError=true;
  3397. this.img.style.display='none';
  3398. this.eleMaps['img_broken'].style.display='inline-block';
  3399. }else{
  3400. var srcs=dataset(relatedThumb, 'srcs');
  3401. if(srcs && srcs.length>0)srcs=srcs.split(",");
  3402. var self=this;
  3403. this.img.onload=function(){
  3404. var imgNaturalSize={
  3405. h:this.naturalHeight,
  3406. w:this.naturalWidth,
  3407. };
  3408.  
  3409. self.imgNaturalSize=imgNaturalSize;
  3410. self.fitToScreen();
  3411. }
  3412. if(srcs && srcs.length>0){
  3413. var src=srcs.shift();
  3414. dataset(relatedThumb, 'srcs',srcs.join(","));
  3415. if(src){
  3416. dataset(relatedThumb, 'src',src);
  3417. this.img.onerror=function(e){
  3418. this.src=relatedThumb.querySelector("img").src;
  3419. dataset(relatedThumb, 'src',this.src);
  3420. }
  3421. this.img.src=src;
  3422. }
  3423. }
  3424. else this.img.src=relatedThumb.querySelector("img").src;
  3425. }
  3426. }else{
  3427. this.imgError=false;
  3428. this.eleMaps['img_broken'].style.display='';
  3429. if(!dataset(relatedThumb,'naturalSize')){
  3430. dataset(relatedThumb,'naturalSize',JSON.stringify(imgNaturalSize));
  3431. };
  3432. };
  3433.  
  3434. function styled(){
  3435. img.style.opacity=1;
  3436. img.style[support.cssTransform]='scale(1)';
  3437. };
  3438.  
  3439.  
  3440. if(prefs.gallery.transition){
  3441. setTimeout(styled,0);
  3442. }else{
  3443. styled();
  3444. };
  3445.  
  3446. this.eleMaps['img-parent'].appendChild(img);
  3447.  
  3448. this.fitContains=prefs.gallery.fitToScreen;//适应屏幕
  3449.  
  3450. this.fitToScreen({
  3451. x:0,
  3452. y:0,
  3453. });
  3454.  
  3455. if(this.previous){
  3456. this.previous=false;
  3457. this.imgScrollbarV.scrollBy(999999);
  3458. }
  3459. this.collection.check();//检查是否在收藏里面。
  3460.  
  3461. },
  3462. fitToScreen:function(scale){
  3463.  
  3464. var container=this.eleMaps['img-content'];
  3465. var containerSize={
  3466. h:container.clientHeight,
  3467. w:container.clientWidth,
  3468. };
  3469.  
  3470. var img=this.img;
  3471.  
  3472. if(!img || !img.classList)return;
  3473. img.classList.remove('pv-gallery-img_zoom-in');
  3474. img.classList.remove('pv-gallery-img_zoom-out');
  3475.  
  3476. var imgSty=img.style;
  3477. imgSty.width='';
  3478. imgSty.height='';
  3479.  
  3480. var contentSSize={
  3481. h:container.scrollHeight,
  3482. w:container.scrollWidth,
  3483. };
  3484. var larger=contentSSize.h>containerSize.h || contentSSize.w>containerSize.w;
  3485.  
  3486. var scaled='100%';
  3487.  
  3488. if(this.fitContains && !(scale && scale.x==0 && scale.y==0 && this.imgNaturalSize.h/this.imgNaturalSize.w > 2.5)){//适应屏幕
  3489. this.imgScrollbarV.hide();
  3490. this.imgScrollbarH.hide();
  3491. if(larger){
  3492. img.classList.add('pv-gallery-img_zoom-in');
  3493. if(contentSSize.h/contentSSize.w >=containerSize.h/containerSize.w){
  3494. let height=this.imgNaturalSize.h-(contentSSize.h - containerSize.h);
  3495. imgSty.height=height + 'px';
  3496. scaled=height/this.imgNaturalSize.h;
  3497. }else{
  3498. let width=this.imgNaturalSize.w-(contentSSize.w - containerSize.w);
  3499. imgSty.width=width + 'px';
  3500. scaled=width/this.imgNaturalSize.w;
  3501. };
  3502. scaled=(scaled*100).toFixed(2) + '%';
  3503. }else if(prefs.gallery.fitToScreenSmall){
  3504. if(this.imgNaturalSize.h/this.imgNaturalSize.w >=containerSize.h/containerSize.w){
  3505. let height=contentSSize.h-50;
  3506. height=height<0?contentSSize.h:height;
  3507. imgSty.height=height + 'px';
  3508. scaled=height/this.imgNaturalSize.h;
  3509. }else{
  3510. let width=contentSSize.w-50;
  3511. width=width<0?contentSSize.w:width;
  3512. imgSty.width=width + 'px';
  3513. scaled=width/this.imgNaturalSize.w;
  3514. };
  3515. scaled=(scaled*100).toFixed(2) + '%';
  3516. }
  3517. }else{//不做尺寸调整
  3518. this.imgScrollbarV.reset();
  3519. this.imgScrollbarH.reset();
  3520.  
  3521. if(larger){
  3522. img.classList.add('pv-gallery-img_zoom-out');
  3523. if(scale){//通过鼠标点击进行的切换。
  3524. this.imgScrollbarH.scroll(container.scrollWidth * scale.x - containerSize.w/2);
  3525. this.imgScrollbarV.scroll(container.scrollHeight * scale.y - containerSize.h/2);
  3526. };
  3527. };
  3528. };
  3529.  
  3530.  
  3531. var imgScaledInfo=this.eleMaps['head-left-img-info-scaling'];
  3532. imgScaledInfo.textContent='('+scaled+')';
  3533. if(scaled!='100%'){
  3534. imgScaledInfo.style.color='#E9CCCC';
  3535. }else{
  3536. imgScaledInfo.style.color='';
  3537. };
  3538.  
  3539. },
  3540.  
  3541. _dataCache: {},
  3542. _spanMarkPool: {},
  3543. _appendThumbSpans: function(data, index) { // 添加缩略图栏的 spans
  3544. var iStatisCopy = this.iStatisCopy;
  3545.  
  3546. if (typeof index == 'undefined' && this.selected) {
  3547. index = Array.prototype.slice.call(this.imgSpans).indexOf(this.selected);
  3548. }
  3549.  
  3550. var sizeInputH=this.gallery.querySelector("#minsizeH");
  3551. var sizeInputW=this.gallery.querySelector("#minsizeW");
  3552. var thumbnails=this.eleMaps['sidebar-thumbnails-container'];
  3553. var selectData=this.data[index];
  3554. if(selectData){
  3555. let spanMark=this._spanMarkPool[selectData.imgSrc];
  3556. if(spanMark && spanMark.dataset.naturalSize){
  3557. let naturalSize=JSON.parse(spanMark.dataset.naturalSize);
  3558. selectData.sizeW=naturalSize.w;
  3559. selectData.sizeH=naturalSize.h;
  3560. }else{
  3561. if(selectData.sizeW<sizeInputW.value){
  3562. var sizeInputWSpan=this.gallery.querySelector("#minsizeWSpan");
  3563. sizeInputW.value=selectData.sizeW;
  3564. sizeInputW.title=sizeInputW.value+"px";
  3565. sizeInputWSpan.innerHTML=createHTML(Math.floor(sizeInputW.value)+"px");
  3566. }
  3567. if(selectData.sizeH<sizeInputH.value){
  3568. var sizeInputHSpan=this.gallery.querySelector("#minsizeHSpan");
  3569. sizeInputH.value=selectData.sizeH;
  3570. sizeInputH.title=sizeInputH.value+"px";
  3571. sizeInputHSpan.innerHTML=createHTML(Math.floor(sizeInputH.value)+"px");
  3572. }
  3573. }
  3574. }
  3575. // 如果是新的,则添加,否则重置并添加。
  3576. if (!data){
  3577. thumbnails.innerHTML = createHTML("");
  3578. this._dataCache = {};
  3579. this.eleMaps['maximize-container'].innerHTML = createHTML("");
  3580. }
  3581. var self=this;
  3582. (data || this.data).forEach(function(item) {
  3583. if(!item)return;
  3584. iStatisCopy[item.type].count++;
  3585. var spanMark=self._spanMarkPool[item.imgSrc];
  3586. if(!spanMark){
  3587. spanMark = document.createElement("span");
  3588. try{
  3589. spanMark.className="pv-gallery-sidebar-thumb-container";
  3590. spanMark.dataset.type=item.type;
  3591. spanMark.dataset.src=item.src;
  3592. spanMark.dataset.srcs=item.srcs||"";
  3593. if(item.xhr)spanMark.dataset.xhr=encodeURIComponent(JSON.stringify(item.xhr));
  3594. spanMark.dataset.description=encodeURIComponent(item.description || '');
  3595. spanMark.dataset.thumbSrc=item.imgSrc;
  3596. spanMark.title=(item.img?(item.img.title||item.img.alt||""):"");
  3597. spanMark.innerHTML=createHTML('<span class="pv-gallery-vertical-align-helper"></span>' +
  3598. '<span class="pv-gallery-sidebar-thumb-loading" title="'+i18n("loading")+'......"></span>');
  3599. }catch(e){};
  3600. self._spanMarkPool[item.imgSrc] = spanMark;
  3601. }
  3602. if(spanMark.dataset.naturalSize){
  3603. let naturalSize=JSON.parse(spanMark.dataset.naturalSize);
  3604. item.sizeW=naturalSize.w;
  3605. item.sizeH=naturalSize.h;
  3606. }
  3607. if(item.sizeW<sizeInputW.value || item.sizeH<sizeInputH.value){
  3608. spanMark.style.display="none";
  3609. }else{
  3610. spanMark.style.display="";
  3611. }
  3612. thumbnails.appendChild(spanMark);
  3613. self.addViewmoreItem([spanMark]);
  3614. });
  3615.  
  3616. (data || this.data).forEach(function(d) {
  3617. if(!d)return;
  3618. self._dataCache[d.imgSrc] = true;
  3619. });
  3620.  
  3621. //写入类别数据。
  3622. var gallery = this.gallery;
  3623. var input, label, iStatisCopy_i;
  3624.  
  3625. for (var i in iStatisCopy) {
  3626. if (!iStatisCopy.hasOwnProperty(i)) continue;
  3627. iStatisCopy_i = iStatisCopy[i];
  3628. input = gallery.querySelector('#pv-gallery-head-command-drop-list-item-category-' + i);
  3629. input.checked = iStatisCopy_i.shown;
  3630. if (iStatisCopy_i.count == 0) {
  3631. input.disabled = true;
  3632. input.parentNode.classList.add('pv-gallery-head-command-drop-list-item_disabled');
  3633. } else {
  3634. input.disabled = false;
  3635. input.parentNode.classList.remove('pv-gallery-head-command-drop-list-item_disabled');
  3636. };
  3637.  
  3638. label = gallery.querySelector('label[for="pv-gallery-head-command-drop-list-item-category-' + i + '"]');
  3639. label.textContent = label.textContent.replace(/(.*)/i, '') + '(' + iStatisCopy_i.count + ')';
  3640. };
  3641.  
  3642. this.imgSpans = thumbnails.children;
  3643.  
  3644. this.thumbScrollbar.reset();
  3645.  
  3646. if(!this.imgSpans[index] || (this.imgSpans[index].style.display=="none" && !selectData)){
  3647. for(var j in this.imgSpans){
  3648. if (!this.imgSpans.hasOwnProperty(j)) continue;
  3649. var curSpan=this.imgSpans[j];
  3650. if(curSpan.style.display!="none"){
  3651. this.select(curSpan, true);
  3652. return;
  3653. }
  3654. }
  3655. }
  3656. this.select(this.imgSpans[index], true);
  3657. },
  3658. load:function(data, from, reload){
  3659. if(this.shown || this.minimized){//只允许打开一个,请先关掉当前已经打开的库
  3660.  
  3661. if(from){//frame发送过来的数据。
  3662. window.postMessage({
  3663. messageID:messageID,
  3664. command:'sendFail',
  3665. to:from,
  3666. },'*');
  3667. };
  3668.  
  3669. if(this.minimized){
  3670. _GM_notification(i18n('closeFirst'));
  3671. flashEle(this.maximizeTrigger);
  3672. };
  3673. return;
  3674. };
  3675.  
  3676. var self=this;
  3677. if(from){//来自frame,获取这个frame所在的iframe标签。定位到图片的时候要用到。
  3678. window.postMessage({
  3679. messageID:messageID,
  3680. command:'getIframeObject',
  3681. windowId:from,
  3682. },'*');
  3683. document.addEventListener('pv-getIframeObject',function(e){
  3684. self.iframe=e.detail;
  3685. },true);
  3686. };
  3687.  
  3688. var unique=this.unique(data);
  3689. data=unique.data;
  3690. var index=unique.index;
  3691.  
  3692. if (reload && this.data.length >= data.length) {
  3693. // alert('没有新增的图片');
  3694. return;
  3695. }
  3696.  
  3697. this.clear();//还原对象的一些修改,以便复用。
  3698. this.data=data;
  3699. this.show(reload);
  3700.  
  3701. this.from=from;//如果来自frame,那么这个from应该保存了那个frame的窗口id,便于以后通信。
  3702.  
  3703. this._appendThumbSpans(null, index);
  3704.  
  3705. this.runOnce();
  3706.  
  3707. this.switchThumbVisible();
  3708.  
  3709. var pageObj = this.getPage(),textSpan = this.eleMaps['head-command-nextPage'].querySelector("span");
  3710. this.haveMorePage = !!pageObj.pre || !!pageObj.next;
  3711. textSpan.style.color=this.haveMorePage?"#e9cccc":"";
  3712. },
  3713. haveMorePage:false,
  3714. clear:function(){
  3715. this._dataCache = {};
  3716.  
  3717. this.allLoading=[];//读取中的图片数组
  3718. this.iStatisCopy=cloneObject(this.imgStatistics,true);//图片统计副本
  3719. if(this.selected){
  3720. this.selected.classList.remove(this.selectedClassName);
  3721. this.selected.classList.remove('pv-gallery-sidebar-thumb_selected');
  3722. }
  3723. this.selected=null;
  3724. if(this.img){
  3725. this.img.style.display='none';
  3726. this.img=null;
  3727. };
  3728. //读取错误的图片占位符
  3729. this.eleMaps['img_broken'].style.display='';
  3730. //清空dom
  3731. this.eleMaps['sidebar-thumbnails-container'].innerHTML=createHTML('');
  3732. this.eleMaps['head-left-img-info-resolution'].textContent='0 x 0';
  3733. this.eleMaps['head-left-img-info-count'].textContent='(1 / 1)';
  3734. this.eleMaps['head-left-img-info-scaling'].textContent='(100%)';
  3735. //隐藏滚动条
  3736. this.imgScrollbarV.hide();
  3737. this.imgScrollbarH.hide();
  3738. this.thumbScrollbar.hide();
  3739. //重置style;
  3740. this.thumbVisibleStyle.textContent='';
  3741. },
  3742.  
  3743. unique:function(data){
  3744. var imgSrc;
  3745. if(data.target){
  3746. imgSrc=data.target.src;
  3747. }
  3748.  
  3749. var data_i,
  3750. data_i_src,
  3751. dataSrcs=[];
  3752.  
  3753. var index;
  3754.  
  3755. for(var i=0,ii=data.length;i<ii;i++){
  3756. data_i=data[i];
  3757. data_i_src=data_i.src;
  3758. if(dataSrcs.indexOf(data_i_src)!=-1){//已经存在
  3759. data.splice(i,1);//移除
  3760. i--;
  3761. ii--;
  3762. continue;
  3763. };
  3764. dataSrcs.push(data_i_src);
  3765.  
  3766. if(imgSrc==data_i_src){
  3767. index=i;
  3768. };
  3769. };
  3770.  
  3771. if(typeof index =='undefined'){
  3772. index=0;
  3773. data.unshift(data.target);
  3774. };
  3775.  
  3776. delete data.target;
  3777.  
  3778. return {
  3779. data:data,
  3780. index:index,
  3781. };
  3782. },
  3783. keyDownListener:function(e){
  3784. switch(e.keyCode){
  3785. case 27:
  3786. if(prefs.imgWindow.close.escKey){
  3787. this.close();
  3788. }
  3789. break;
  3790. }
  3791. },
  3792. keyUpListener:function(e){
  3793. switch(e.keyCode){
  3794. case 82:
  3795. var img=this.img;
  3796. var cssTransform=img.style[support.cssTransform];
  3797. var iTransform=cssTransform.replace(/rotate\([^)]*\)/i,'');
  3798. var rotatedRadians=cssTransform.indexOf("rotate")!=-1?cssTransform.replace(/.*rotate\(([-\d\.]+).*/i,'$1'):0;
  3799. var PI=Math.PI;
  3800. var origin=parseFloat(rotatedRadians) +(e.shiftKey?-90:90) * PI/180;
  3801. if(origin>=2*PI || origin<=-2*PI ||(-0.1<origin && origin<0.1)){
  3802. origin=0;
  3803. }
  3804. img.style[support.cssTransform] = ' rotate('+ origin +'rad) ' + iTransform;
  3805. break;
  3806. }
  3807. },
  3808. show:function(reload){
  3809. this.shown=true;
  3810. galleryMode=true;
  3811.  
  3812. if (!reload) {
  3813. var des=document.documentElement.style;
  3814. this.deOverflow={
  3815. x:des.overflowX,
  3816. y:des.overflowY,
  3817. };
  3818. des.overflow='hidden';
  3819. this.gallery.style.display='';
  3820. this.gallery.focus();
  3821. window.addEventListener('resize',this._resizeHandler,true);
  3822. }
  3823. document.addEventListener('keydown',this._keyDownListener,true);
  3824. document.addEventListener('keyup',this._keyUpListener,true);
  3825.  
  3826. if(prefs.gallery.loadMore){
  3827. this.eleMaps['head-command-nextPage'].click();
  3828. }
  3829.  
  3830. this.changeSizeInputReset();
  3831.  
  3832. var self=this;
  3833. if(prefs.gallery.autoOpenViewmore){
  3834. setTimeout(function(){
  3835. self.maximizeSidebar();
  3836. },1);
  3837. }
  3838. },
  3839. close:function(reload){
  3840. if(this.hideBodyStyle.parentNode)
  3841. this.hideBodyStyle.parentNode.removeChild(this.hideBodyStyle);
  3842. document.removeEventListener('keydown',this._keyDownListener,true);
  3843. document.removeEventListener('keyup',this._keyUpListener,true);
  3844. this.shown=false;
  3845. this.minimized=false;
  3846.  
  3847. if (!reload) {
  3848. galleryMode=false;
  3849. this.gallery.blur();
  3850. this.gallery.style.display='none';
  3851. var des=document.documentElement.style;
  3852. des.overflowX=this.deOverflow.x;
  3853. des.overflowY=this.deOverflow.y;
  3854. this.slideShow.exit();
  3855. this.collection.exit();
  3856. window.removeEventListener('resize',this._resizeHandler,true);
  3857.  
  3858. // 退出全屏
  3859. var btn = document.getElementById('pv-gallery-fullscreenbtn');
  3860. if (btn.classList.contains('fullscreenbtn')) {
  3861. cancelFullScreen();
  3862. btn.textContent = i18n("enterFullsc");
  3863. btn.classList.remove('fullscreenbtn');
  3864. }
  3865. }
  3866. },
  3867. curPage:document,
  3868. getPage:function(){
  3869. let pageNum=0,preStr="",afterStr="";
  3870. let pageMatch1=this.href.match(/(.*[a-zA-Z0-9\/][\-_](?:p|page)?)(\d+)(\.html?$|$)/i);
  3871. let pageMatch2=this.href.match(/(.*[\?&]p(?:age)?=)(\d+)($|[#&].*)/i);
  3872. if(pageMatch1){
  3873. preStr=pageMatch1[1];
  3874. pageNum=pageMatch1[2];
  3875. afterStr=pageMatch1[3];
  3876. }else if(pageMatch2){
  3877. preStr=pageMatch2[1];
  3878. pageNum=pageMatch2[2];
  3879. afterStr=pageMatch2[3];
  3880. }
  3881. var curPage=this.curPage;
  3882. let pre=curPage.querySelector("a.prev")||
  3883. curPage.querySelector("a#prev")||
  3884. curPage.querySelector("a#leftFix")||
  3885. curPage.querySelector("a.prev_page")||
  3886. curPage.querySelector(".prev>a");
  3887. let next=curPage.querySelector("a.next")||
  3888. curPage.querySelector("a#next")||
  3889. curPage.querySelector("a#rightFix")||
  3890. curPage.querySelector("a.next_page")||
  3891. curPage.querySelector(".next>a");
  3892. if(!pre && !next){
  3893. let pageDiv=curPage.querySelector("div.wp-pagenavi");
  3894. if(pageDiv){
  3895. let cur=pageDiv.querySelector("span.current");
  3896. pre=cur.previousSibling;
  3897. next=cur.nextSibling;
  3898. }else{
  3899. let cur=curPage.querySelector("div.article-paging>span");
  3900. if(cur){
  3901. pre=cur.previousElementSibling;
  3902. next=cur.nextElementSibling;
  3903. }
  3904. }
  3905. }
  3906. if(pre && (!pre.href || /javascript:/.test(pre.href.trim())))pre=null;
  3907. if(next && (!next.href || /javascript:/.test(next.href.trim())))next=null;
  3908. if(!pre || !next){
  3909. let aTags=curPage.querySelectorAll("a");
  3910. if(!pre){
  3911. let pref,pres,pret;
  3912. for(let i=0;i<aTags.length;i++){
  3913. let aTag=aTags[i];
  3914. if(!aTag.href || /javascript:/.test(aTag.href.trim()))continue;
  3915. if(pref && pres && pret)break;
  3916. if(!pref){
  3917. if(/(\s|^)上[一1]?[页頁张張]|^previous( page)?\s*$|前のページ/i.test(aTag.innerHTML)){
  3918. pref=aTag;
  3919. }
  3920. }
  3921. if(!pres){
  3922. if(aTag.innerHTML=="&lt;"){
  3923. pres=aTag;
  3924. }
  3925. }
  3926. if(!pret){
  3927. if(aTag.innerHTML=="«"){
  3928. pret=aTag;
  3929. }else if(pageNum==1){
  3930. if(aTag.href.indexOf(this.href.replace(/.*\/([^\/]+)$/,"$1").replace(/[_-]\d+/,""))!=-1){
  3931. pret=aTag;
  3932. }
  3933. }else if(aTag.href.replace(preStr,"").replace(afterStr,"")==pageNum-1){
  3934. pret=aTag;
  3935. }
  3936. }
  3937. }
  3938. pre=pref||pres||pret;
  3939. }
  3940. if(!next){
  3941. let nextf,nexts,nextt;
  3942. for(let i=0;i<aTags.length;i++){
  3943. let aTag=aTags[i];
  3944. if(!aTag.href || /^\s*javascript:/.test(aTag.href.trim()))continue;
  3945. if(nextf && nexts && nextt)break;
  3946. if(!nextf){
  3947. if(/(\s|^)下[一1]?[页頁张張]|^next( page)?\s*$|次のページ/i.test(aTag.innerHTML)){
  3948. nextf=aTag;
  3949. }
  3950. }
  3951. if(!nexts){
  3952. if(aTag.innerHTML=="&gt;"){
  3953. nexts=aTag;
  3954. }
  3955. }
  3956. if(!nextt){
  3957. if(aTag.innerHTML=="»"){
  3958. nextt=aTag;
  3959. }else if(aTag.href.replace(preStr,"").replace(afterStr,"")==parseInt(pageNum)+1){
  3960. nextt=aTag;
  3961. }else if(aTag.href.indexOf(this.href)!=-1 && /^[\/\?&]?[_-]?(p|page)?=?[12](\?|&|$)/i.test(aTag.href.replace(this.href,""))){
  3962. nextt=aTag;
  3963. }
  3964. }
  3965. }
  3966. next=nextf||nexts||nextt;
  3967. }
  3968. }
  3969. if(!pre)pre=curPage.querySelector('[rel="prev"],[rel="previous"]');
  3970. if(!next)next=curPage.querySelector('[rel="next"]');
  3971. return {pre:pre,next:next};
  3972. },
  3973. canonicalUri:function(src, base_path){
  3974. if(src.charAt(0)=="#")return location.href+src;
  3975. var root_page = /^[^?#]*\//.exec(location.href)[0],
  3976. root_domain = /^\w+\:\/\/\/?[^\/]+/.exec(root_page)[0],
  3977. absolute_regex = /^\w+\:\/\//;
  3978. src=src.replace(/\.\//,"");
  3979. if (/^\/\/\/?/.test(src)){
  3980. src = location.protocol + src;
  3981. }
  3982. else if (!absolute_regex.test(src) && src.charAt(0) != "/"){
  3983. src = (base_path || "") + src;
  3984. }
  3985. return (absolute_regex.test(src) ? src : ((src.charAt(0) == "/" ? root_domain : root_page) + src));
  3986. },
  3987. completePages:[location.href],
  3988. href:location.href,
  3989. pageAllReady:false,
  3990. loadingImgNum:0,
  3991. pageImgReady:function(){
  3992. var textSpan=this.eleMaps['head-command-nextPage'].querySelector("span");
  3993. if(this.pageAllReady && this.loadingImgNum<=0){
  3994. textSpan.innerHTML=createHTML("<font color='red'>"+i18n("loadedAll")+"</font>");
  3995. setTimeout(function(){textSpan.innerHTML=createHTML(i18n("loadAll"));},1500);
  3996. }
  3997. },
  3998. pageAction:function(next, single){
  3999. var pageObj=this.getPage(),self=this,textSpan=this.eleMaps['head-command-nextPage'].querySelector("span");
  4000. if(textSpan.innerHTML!=i18n("loading")){
  4001. return;
  4002. }
  4003. var loadOver=function(){
  4004. if(!next || !prefs.gallery.loadAll || single){
  4005. self.pageAllReady=true;
  4006. self.pageImgReady();
  4007. }else{
  4008. self.curPage=document;
  4009. self.href=location.href;
  4010. self.pageAction(false);
  4011. }
  4012. };
  4013. if((next && !pageObj.next) || (!next && !pageObj.pre)){
  4014. loadOver();
  4015. return;
  4016. }
  4017. var targetUrl=next?pageObj.next:pageObj.pre;
  4018. if(targetUrl.tagName!="A"){
  4019. var childA=targetUrl.querySelector("a");
  4020. if(childA){
  4021. targetUrl=childA;
  4022. }else{
  4023. while(targetUrl=targetUrl.parentElement){
  4024. if(targetUrl.nodeName=='A'){
  4025. break;
  4026. }
  4027. }
  4028. if(!targetUrl)return;
  4029. }
  4030. }
  4031. var href=targetUrl.getAttribute("href");
  4032. if(self.completePages.indexOf(href)!=-1){
  4033. loadOver();
  4034. return;
  4035. }else{
  4036. self.completePages.push(href);
  4037. }
  4038. self.href=self.canonicalUri(href);
  4039. _GM_xmlhttpRequest({
  4040. method: 'GET',
  4041. url: self.href,
  4042. headers:{"Referer": + window.location.href},
  4043. overrideMimeType:"text/html;charset="+document.charset,
  4044. onload: function(d) {
  4045. let html=document.implementation.createHTMLDocument('');
  4046. html.documentElement.innerHTML = d.responseText;
  4047. self.curPage=html;
  4048. let imgs=html.querySelectorAll('img');
  4049. var container = document.querySelector('.pv-gallery-container'),
  4050. preloadContainer = document.querySelector('.pv-gallery-preloaded-img-container');
  4051. imgs = Array.prototype.slice.call(imgs).filter(function(img){
  4052. return !(container.contains(img) || (preloadContainer&&preloadContainer.contains(img)));
  4053. });
  4054. imgs.forEach(function(img) {
  4055. pretreatment(img);
  4056. if(!img.src)return;
  4057. var isrc=img.src.trim();
  4058. if(!isrc)return;
  4059. isrc=self.canonicalUri(isrc);
  4060. if (self._dataCache[isrc]) return;
  4061. self._dataCache[isrc] = true;
  4062. var nimg = new Image();
  4063. self.loadingImgNum++;
  4064. nimg.onload=function(){
  4065. self.loadingImgNum--;
  4066. self.pageImgReady();
  4067. var result = findPic(this);
  4068. if (result) {
  4069. self.data.push(result);
  4070. self._appendThumbSpans([result]);
  4071. self.loadThumb();
  4072. }
  4073. };
  4074. nimg.onerror=function(){
  4075. self.loadingImgNum--;
  4076. self.pageImgReady();
  4077. };
  4078. nimg.src = isrc;
  4079. });
  4080. if(prefs.gallery.loadAll && !single)self.pageAction(next);
  4081. else loadOver();
  4082. },
  4083. onerror: function(e) {
  4084. if(prefs.gallery.loadAll && !single)self.pageAction(next);
  4085. else loadOver();
  4086. }
  4087. });
  4088. },
  4089. runOnce:function(){//运行一次来获取某些数据。
  4090. var thumbSpanCS=unsafeWindow.getComputedStyle(this.selected);
  4091. this.thumbSpanOuterSize=this.isHorizontal?
  4092. this.selected.offsetWidth + parseFloat(thumbSpanCS.marginLeft) + parseFloat(thumbSpanCS.marginRight) :
  4093. this.selected.offsetHeight + parseFloat(thumbSpanCS.marginTop) + parseFloat(thumbSpanCS.marginBottom);
  4094.  
  4095.  
  4096.  
  4097. this.runOnce=function(){
  4098. };
  4099. },
  4100.  
  4101. minimize:function(){
  4102. this.close();
  4103. this.maximizeTrigger.style.display='block';
  4104. this.minimized=true;
  4105. },
  4106. navigateToImg:function(targetImg){
  4107. targetImg.scrollIntoView();//先调用原方法,可以让overflow hidden的滚动出来。
  4108.  
  4109. //让图片近可能的居中
  4110. var imgBCRect=getContentClientRect(targetImg);
  4111. var wSize=getWindowSize();
  4112.  
  4113. window.scrollBy(imgBCRect.left - (wSize.w - imgBCRect.width)/2,
  4114. imgBCRect.top - (wSize.h - imgBCRect.height)/2);
  4115.  
  4116. },
  4117. switchThumbVisible:function(){
  4118. var style=this.thumbVisibleStyle;
  4119. var count=0;
  4120. var styleText=[];
  4121. var iStatisCopy=this.iStatisCopy;
  4122. var iStatisCopy_i;
  4123.  
  4124. for(var i in iStatisCopy){
  4125. if(!iStatisCopy.hasOwnProperty(i))continue;
  4126. iStatisCopy_i=iStatisCopy[i];
  4127. if(iStatisCopy_i.shown){
  4128. count+=iStatisCopy_i.count;
  4129. }else{
  4130. styleText.push('.pv-gallery-sidebar-thumb-container[data-type="'+i+'"]');
  4131. };
  4132. };
  4133.  
  4134. //写入style;
  4135. style.textContent=styleText.join(',') + '{display:none !important;}';
  4136.  
  4137. //初始化缩略图区的滚动条
  4138. this.thumbScrollbar.reset();
  4139. this.arrowVisib();
  4140.  
  4141. //载入缩略图
  4142. this.loadThumb();
  4143. },
  4144. forceRepaint:function(){//解决opera的fixed元素,当滚动条不再最高处的时候,不重绘fixed元素的问题。
  4145. clearTimeout(this.forceRepaintTimer);
  4146. var self=this;
  4147. this.forceRepaintTimer=setTimeout(function(){
  4148. if(envir.opera){
  4149. self.forceRepaintTimes % 2 ==0 ? window.scrollBy(0,1) : window.scrollBy(0,-1);
  4150. self.forceRepaintTimes++;
  4151. };
  4152. },333);
  4153. },
  4154. resizeHandler:function(){//窗口变化时,调整一些东西。
  4155. this.thumbScrollbar.reset();
  4156. //this.selectedIntoView();
  4157. this.fitToScreen();
  4158. this.loadThumb();
  4159. },
  4160. _isLastSpan: function(span) {// 用于判断是否自动重载,是否是最后几个图片
  4161. if (this.selected.clientWidth == 0) return false;
  4162. if (!span) return true;
  4163.  
  4164. var index = Array.prototype.slice.call(this.imgSpans).indexOf(span);
  4165. if (index != -1) {
  4166. var total = this.imgSpans.length;
  4167. if (total - index < prefs.gallery.scrollEndAndLoad_num) {
  4168. return true;
  4169. }
  4170. }
  4171. },
  4172. arrowVisib:function(){//当当前选择元素的前面或者后面没有元素的时候隐藏控制箭头
  4173.  
  4174. var icps=this.eleMaps['img-controler-pre'].style;
  4175. var icns=this.eleMaps['img-controler-next'].style;
  4176. var scps=this.eleMaps['sidebar-controler-pre'].style;
  4177. var scns=this.eleMaps['sidebar-controler-next'].style;
  4178.  
  4179. //下一张的箭头
  4180. var nextSpan = this.getThumSpan();
  4181. if (nextSpan) {
  4182. icns.display='';
  4183. scns.display='';
  4184. }else{
  4185. icns.display='none';
  4186. scns.display='none';
  4187. };
  4188.  
  4189. // 最后几张图片,滚到底部添加新的图片
  4190. if (prefs.gallery.scrollEndAndLoad && this._isLastSpan(nextSpan)) {
  4191. this.scrollToEndAndReload();
  4192. }
  4193.  
  4194. //上一张的箭头
  4195. if(this.getThumSpan(true)){
  4196. icps.display='';
  4197. scps.display='';
  4198. }else{
  4199. icps.display='none';
  4200. scps.display='none';
  4201. };
  4202. },
  4203. simpleSlideShow:function(backward,interval){
  4204. clearInterval(this.slideShowInterval);//幻灯播放,只允许存在一个,否则得乱套
  4205.  
  4206. var self=this;
  4207. var slideShowInterval=setInterval(function(){
  4208. var before=self.selected;
  4209. backward ? self.selectPrevious() : self.selectNext();
  4210. if(before == self.selected){//没有下一个元素了。。
  4211. stop();
  4212. };
  4213. },(interval? interval : 800));
  4214.  
  4215. this.slideShowInterval=slideShowInterval;
  4216.  
  4217. function stop(){
  4218. clearInterval(slideShowInterval);
  4219. };
  4220.  
  4221. return stop;
  4222. },
  4223.  
  4224. reload: function() {// 重新加载所有图片到库里面
  4225. // 函数在 LoadingAnimC 中
  4226. var data = this.getAllValidImgs();
  4227. // 设置当前选中的图片
  4228. data.target = {
  4229. src: this.selected.dataset.src
  4230. };
  4231.  
  4232. this.close(true);
  4233.  
  4234. this.load(data, null, true);
  4235. },
  4236. reloadNew: function() {// 加载新的图片到库里面
  4237. var newer = true;
  4238. var data = this.getAllValidImgs(newer);
  4239. if (data.length) {
  4240. this._appendThumbSpans(data);
  4241. }
  4242. },
  4243. getAllValidImgs:function(newer){
  4244. var validImgs = [];
  4245. var imgs = document.getElementsByTagName('img'),
  4246. container = document.querySelector('.pv-gallery-container'),
  4247. preloadContainer = document.querySelector('.pv-gallery-preloaded-img-container');
  4248.  
  4249. imgs = Array.prototype.slice.call(imgs);
  4250. arrayFn.forEach.call(document.querySelectorAll("iframe"),function(iframe){
  4251. if(iframe.src && iframe.src.replace(/\/[^\/]*$/,"").indexOf(location.hostname)!=-1){
  4252. try{
  4253. arrayFn.forEach.call(iframe.contentWindow.document.getElementsByTagName('img'),function(img){
  4254. imgs.push(img);
  4255. });
  4256. }catch(e){
  4257. debug(e.toString());
  4258. }
  4259. }
  4260. });
  4261. var bgReg=/.*url\(\s*["']?(.+?)["']?\s*\)/i;
  4262. var bgImgs=Array.from(document.querySelectorAll('*'))
  4263. .reduce((total, node) => {
  4264. if(node.nodeName != "IMG" && !node.src && (!node.className || !node.className.indexOf || node.className.indexOf("pv-")==-1)){
  4265. let prop = unsafeWindow.getComputedStyle(node).backgroundImage;
  4266. let match = bgReg.exec(prop)
  4267. if (match) {
  4268. node.src=match[1];
  4269. total.push(node);
  4270. }
  4271. }
  4272. return total;
  4273. }, []);
  4274. if(bgImgs)imgs=imgs.concat(bgImgs);
  4275. // 排除库里面的图片
  4276. imgs = imgs.filter(function(img){
  4277. return !(container.contains(img) || (preloadContainer&&preloadContainer.contains(img)));
  4278. });
  4279.  
  4280. // 已经在图库里面的
  4281. var self = this;
  4282. imgs.forEach(function(img) {
  4283. pretreatment(img);
  4284. if(!img.src) return;
  4285. if (newer && self._dataCache[img.src]) return;
  4286.  
  4287. var result = findPic(img);
  4288. if (result) {
  4289. validImgs.push(result);
  4290. self.data.push(result);
  4291. }
  4292.  
  4293. self._dataCache[img.src] = true;
  4294. });
  4295.  
  4296. return validImgs;
  4297. },
  4298. scrollToEndAndReload: function() {// 滚动主窗口到最底部,然后自动重载库的图片
  4299.  
  4300. window.scrollTo(0, 9999999);
  4301.  
  4302. var self = this;
  4303. clearTimeout(self.reloadTimeout);
  4304. self.reloadTimeout = setTimeout(function(){
  4305. // self.reload();
  4306. self.reloadNew();
  4307. }, 1000);
  4308. },
  4309. exportImages: function () {// 导出所有图片到新窗口
  4310. var nodes = document.querySelectorAll('.pv-gallery-sidebar-thumb-container[data-src]'),i;
  4311. //var arr = Array.prototype.map.call(nodes, function(node){
  4312. // if(unsafeWindow.getComputedStyle(node).display=="none")return "";
  4313. // else return '<div><img src=' + node.dataset.src + '></div>'
  4314. //});
  4315.  
  4316. var arr=[];
  4317. for (i = 0; i < nodes.length; ++i) {
  4318. if(unsafeWindow.getComputedStyle(nodes[i]).display=="none")arr.push("");
  4319. else arr.push('<div><img src=' + nodes[i].dataset.src + '></div>');
  4320. }
  4321.  
  4322. var title = document.title;
  4323.  
  4324. var html = '\
  4325. <head>\
  4326. <title>' + title + ' '+i18n("exportImages")+'</title>\
  4327. <style>\
  4328. .toTop{width:28px;height:28px;border-radius:14px;position: fixed;right:2px;bottom: 2px;cursor: pointer;background-color:#000;opacity:.3;padding:0em!important;border:0px!important;z-index:1}\
  4329. .toTop:hover{opacity:1}\
  4330. .toTop>span{height:28px;line-height:28px;display:block;color:#FFF;text-align:center;font-size:20px;}\
  4331. .grid{-moz-column-count:4;-webkit-column-count:4;column-count:4;-moz-column-gap: 1em;-webkit-column-gap: 1em;column-gap: 1em;}\
  4332. .grid>div{padding: 1em;margin: 0 0 1em 0;-moz-page-break-inside: avoid;-webkit-column-break-inside: avoid;break-inside: avoid;}\
  4333. .grid>div>img{width: 100%;margin-bottom:10px;}\
  4334. .list>div {text-align:center;}\
  4335. .list>div>img { max-width: 100%; }\
  4336. .gridBig{margin: 0px;}\
  4337. .gridBig>div { float: left;margin: 0px 0px 1px 1px;}\
  4338. .gridBig>div>img { max-width: 100%; }\
  4339. .select{opacity: 0.8;border: 5px solid red!important;}\
  4340. body>div{border: 5px solid black;margin: 1px;}\
  4341. body>div:hover{border: 5px solid #dbdbdb;}\
  4342. body>div{position:relative;}\
  4343. </style>\
  4344. </head>\
  4345. <span class="toTop">\
  4346. <span>↑</span>\
  4347. </span>\
  4348. <body class="'+prefs.gallery.exportType+'">\
  4349. <p style="width:100vw;display:flex;flex-direction:column;">\
  4350. <img id="bigImg" style="pointer-events:none;position:fixed;z-index:999;width:100vw;top:0px;align-self:center;"></p>\
  4351. <p>【'+i18n("picTitle")+'】:' + title + '</p>\
  4352. <p>【'+i18n("picNum")+'】:' + nodes.length + ' <select onchange="document.body.className=this.options[this.options.selectedIndex].value"><option value="grid" '+(prefs.gallery.exportType=="grid"?"selected='selected'":"")+'>'+i18n("grid")+'</option><option value="gridBig" '+(prefs.gallery.exportType=="gridBig"?"selected='selected'":"")+'>'+i18n("gridBig")+'</option><option value="list" '+(prefs.gallery.exportType=="list"?"selected='selected'":"")+'>'+i18n("list")+'</option> </select> \
  4353. <input type="button" value="'+i18n("exportImagesUrl")+'" onclick="var imgStr=\'\',selList=document.querySelectorAll(\'.select>img\');if(selList.length==0)[].forEach.call(document.querySelectorAll(\'img\'),function(i){imgStr+=i.src+\' \\n\'});else{[].forEach.call(selList,function(i){imgStr+=i.src+\' \\n\'});}window.prompt(\''+i18n("exportImagesUrlPop")+'\',imgStr);">\
  4354. ('+i18n("picTips")+')</p><p>'+i18n("savePageTips")+"</p>";
  4355.  
  4356. html += arr.join('\n') +
  4357. '<script type="text/javascript">\
  4358. document.querySelector(".toTop").addEventListener("click", function(){\
  4359. document.body.scrollIntoView();\
  4360. });\
  4361. var bigImg=document.querySelector("#bigImg"),body=document.body;\
  4362. [].forEach.call(document.querySelectorAll("div>img"),function(i){\
  4363. i.onerror=function(e){i.style.display="none"};\
  4364. i.onmouseover=i.onmousemove=function(e){bigImg.style.top=(this.width/this.height>body.clientWidth/body.clientHeight?10+(body.clientHeight*0.95-body.clientWidth*this.height/this.width)*e.offsetY/this.height:10-(body.clientWidth*this.height/this.width-body.clientHeight*0.95)*e.offsetY/this.height);bigImg.src=e.ctrlKey?this.src:"";bigImg.style.display=e.ctrlKey?"":"none";};\
  4365. });\
  4366. [].forEach.call(document.querySelectorAll("body>div"),function(i){\
  4367. i.onclick=function(e){if(e.ctrlKey&&i.firstChild.src){window.open(i.firstChild.src,"_blank")}else{this.classList.toggle("select")}}\
  4368. });\
  4369. </script></body>';
  4370. _GM_openInTab('data:text/html;charset=utf-8,' + encodeURIComponent(html),{active:true});
  4371. },
  4372. copyImages: function(isAlert) {
  4373. var nodes = document.querySelectorAll('.pv-gallery-sidebar-thumb-container[data-src]');
  4374. var urls = [];
  4375. [].forEach.call(nodes, function(node){
  4376. if(unsafeWindow.getComputedStyle(node).display!="none"){
  4377. urls.push(node.dataset.src);
  4378. }
  4379. });
  4380.  
  4381. _GM_setClipboard(urls.join("\n"));
  4382.  
  4383. if (isAlert) {
  4384. _GM_notification(i18n("copySuccess",urls.length));
  4385. }
  4386. },
  4387.  
  4388. Preload:function(ele,oriThis){
  4389. this.ele=ele;
  4390. this.oriThis=oriThis;//主this
  4391. this.init();
  4392. },
  4393. Scrollbar:function(scrollbar,container,isHorizontal){
  4394. this.scrollbar=scrollbar;
  4395. this.container=container;
  4396. this.isHorizontal=isHorizontal
  4397. this.init();
  4398. },
  4399.  
  4400. addStyle:function(){
  4401. var style=document.createElement('style');
  4402. style.type='text/css';
  4403. style.textContent='\
  4404. /*最外层容器*/\
  4405. .pv-gallery-container {\
  4406. position: fixed;\
  4407. top: 0;\
  4408. left: 0;\
  4409. width: 100%;\
  4410. height: 100%;\
  4411. min-width:none;\
  4412. min-height:none;\
  4413. padding: 0;\
  4414. margin: 0;\
  4415. border: none;\
  4416. z-index:2147483646;\
  4417. background-color: transparent;\
  4418. }\
  4419. /*全局border-box*/\
  4420. .pv-gallery-container span{\
  4421. -moz-box-sizing: border-box;\
  4422. box-sizing: border-box;\
  4423. line-height: 1.6;\
  4424. }\
  4425. .pv-gallery-container * {\
  4426. font-size: 14px;\
  4427. }\
  4428. /*点击还原的工具条*/\
  4429. .pv-gallery-maximize-trigger{\
  4430. position:fixed;\
  4431. bottom:15px;\
  4432. left:15px;\
  4433. display:none;\
  4434. background:#000;\
  4435. opacity:0.6;\
  4436. padding-left:10px;\
  4437. font-size:16px;\
  4438. line-height:0;\
  4439. color:white;\
  4440. cursor:pointer;\
  4441. box-shadow:3px 3px 0 0 #333;\
  4442. z-index:899999998;\
  4443. }\
  4444. .pv-gallery-maximize-trigger:hover{\
  4445. opacity:0.9;\
  4446. }\
  4447. .pv-gallery-maximize-trigger-close{\
  4448. display:inline-block;\
  4449. padding-left:10px;\
  4450. vertical-align:middle;\
  4451. height:30px;\
  4452. padding:10px 0;\
  4453. width:24px;\
  4454. background:url("'+prefs.icons.loadingCancle+'") center no-repeat;\
  4455. }\
  4456. .pv-gallery-maximize-trigger-close:hover{\
  4457. background-color:#333;\
  4458. }\
  4459. @media only screen and (max-width: 800px) {\
  4460. .pv-gallery-range-box>input {\
  4461. display: none;\
  4462. }\
  4463. .pv-gallery-maximize-container{\
  4464. column-count: 2;\
  4465. -moz-column-count: 2;\
  4466. -webkit-column-count: 2;\
  4467. padding-top: 300px;\
  4468. }\
  4469. .pv-gallery-sidebar-viewmore-bottom.showmore{\
  4470. transform: scale(3.5);\
  4471. bottom: 50px;\
  4472. }\
  4473. .pv-gallery-maximize-container span>p{\
  4474. opacity: 0.6;\
  4475. }\
  4476. }\
  4477. @media only screen and (min-width: 800px) {\
  4478. .pv-gallery-maximize-container{\
  4479. column-count: 5;\
  4480. -moz-column-count: 5;\
  4481. -webkit-column-count: 5;\
  4482. padding-top: 30px;\
  4483. }\
  4484. .pv-gallery-maximize-container span>p{\
  4485. opacity: 0;\
  4486. }\
  4487. }\
  4488. /*顶栏*/\
  4489. .pv-gallery-head {\
  4490. position: absolute;\
  4491. top: 0;\
  4492. left: 0;\
  4493. width: 100%;\
  4494. flex-wrap: wrap;\
  4495. min-height: 30px;\
  4496. height: auto;\
  4497. z-index:1;\
  4498. background-color:rgb(0,0,0);\
  4499. border:none;\
  4500. border-bottom:1px solid #333333;\
  4501. text-align:right;\
  4502. line-height:0;\
  4503. font-size: 14px;\
  4504. color:#757575;\
  4505. padding-right:42px;\
  4506. display: table;\
  4507. }\
  4508. .pv-gallery-head > span{\
  4509. vertical-align:middle;\
  4510. }\
  4511. /*顶栏左边*/\
  4512. .pv-gallery-head-float-left{\
  4513. float:left;\
  4514. height:100%;\
  4515. text-align:left;\
  4516. padding-left:5px;\
  4517. display: table;\
  4518. }\
  4519. .pv-gallery-head-float-left > span{\
  4520. display:inline-block;\
  4521. height:100%;\
  4522. vertical-align:middle;\
  4523. }\
  4524. .pv-gallery-head-float-left > span > *{\
  4525. vertical-align:middle;\
  4526. }\
  4527. .pv-gallery-head-left-img-info{\
  4528. cursor:help;\
  4529. }\
  4530. .pv-gallery-head-left-img-info-description {\
  4531. margin-left: 10px;\
  4532. margin-right: 10px;\
  4533. }\
  4534. .pv-gallery-range-box{\
  4535. display: inline-flex;\
  4536. justify-content: center;\
  4537. align-items: center;\
  4538. }\
  4539. .pv-gallery-range-box>span{\
  4540. padding: 0 5px 0 5px;\
  4541. }\
  4542. .pv-gallery-range-box>input{\
  4543. background: white;\
  4544. }\
  4545. /*顶栏里面的按钮样式-开始*/\
  4546. .pv-gallery-head-command{\
  4547. display:inline-block;\
  4548. cursor:pointer;\
  4549. height:100%;\
  4550. padding:0 8px;\
  4551. text-align:center;\
  4552. position:relative;\
  4553. z-index:1;\
  4554. vertical-align:middle;\
  4555. -o-user-select: none;\
  4556. -ms-user-select: none;\
  4557. -webkit-user-select: none;\
  4558. -moz-user-select: -moz-none;\
  4559. user-select: none;\
  4560. }\
  4561. /*辅助点击事件的生成,countdown*/\
  4562. .pv-gallery-head-command_overlayer{\
  4563. top:0;\
  4564. left:0;\
  4565. right:0;\
  4566. bottom:0;\
  4567. position:absolute;\
  4568. opacity:0;\
  4569. }\
  4570. .pv-gallery-head-command > *{\
  4571. vertical-align:middle;\
  4572. }\
  4573. .pv-gallery-head-command-close{\
  4574. position:absolute;\
  4575. top:0;\
  4576. right:0;\
  4577. width:40px;\
  4578. border-left: 1px solid #333333;\
  4579. background:transparent no-repeat center;\
  4580. background-image:url("'+prefs.icons.loadingCancle+'");\
  4581. }\
  4582. .pv-gallery-head-command-slide-show-countdown{\
  4583. font-size:0.8em;\
  4584. }\
  4585. .pv-gallery-head-command-slide-show-button{\
  4586. border-radius:36px;\
  4587. display:inline-block;\
  4588. width:18px;\
  4589. height:18px;\
  4590. border:2px solid #757575;\
  4591. margin-right:3px;\
  4592. line-height:0;\
  4593. }\
  4594. .pv-gallery-head-command-slide-show-button-inner{\
  4595. display:inline-block;\
  4596. border:none;\
  4597. border-top:4px solid transparent;\
  4598. border-bottom:4px solid transparent;\
  4599. border-left:8px solid #757575;\
  4600. vertical-align:middle;\
  4601. }\
  4602. .pv-gallery-head-command-slide-show-button-inner_stop{\
  4603. border-color:#757575;\
  4604. }\
  4605. .pv-gallery-head-command-collect-icon{\
  4606. display:inline-block;\
  4607. height:20px;\
  4608. width:20px;\
  4609. background:transparent url("' + prefs.icons.fivePointedStar + '") 0 0 no-repeat;\
  4610. }\
  4611. .pv-gallery-head-left-lock-icon{\
  4612. display:inline-block;\
  4613. height:20px;\
  4614. width:20px;\
  4615. cursor:pointer;\
  4616. background:transparent url("' + prefs.icons.lock + '") 0 0 no-repeat;\
  4617. }\
  4618. .pv-gallery-head-command-collect-icon ~ .pv-gallery-head-command-collect-text::after{\
  4619. content:"'+i18n("collect")+'";\
  4620. }\
  4621. .pv-gallery-head-command-collect-favorite > .pv-gallery-head-command-collect-icon{\
  4622. background-position:-40px 0 !important;\
  4623. }\
  4624. .pv-gallery-head-command-collect-favorite > .pv-gallery-head-command-collect-text::after{\
  4625. content:"'+i18n("collected")+'";\
  4626. }\
  4627. .pv-gallery-head-command-exit-collection{\
  4628. color:#939300 !important;\
  4629. display:none;\
  4630. }\
  4631. .pv-gallery-head-command:hover{\
  4632. background-color:#272727;\
  4633. color:#ccc;\
  4634. }\
  4635. /*droplist*/\
  4636. .pv-gallery-head-command-drop-list{\
  4637. position:absolute;\
  4638. right:0;\
  4639. display:none;\
  4640. box-shadow:0 0 3px #808080;\
  4641. background-color:#272727;\
  4642. line-height: 1.6;\
  4643. text-align:left;\
  4644. padding:10px;\
  4645. color:#ccc;\
  4646. margin-top:-1px;\
  4647. z-index:9;\
  4648. }\
  4649. .pv-gallery-head-command-drop-list-item{\
  4650. display:block;\
  4651. padding:2px 5px;\
  4652. cursor:pointer;\
  4653. white-space:nowrap;\
  4654. }\
  4655. .pv-gallery-head-command-drop-list-item-collect-description{\
  4656. cursor:default;\
  4657. }\
  4658. .pv-gallery-head-command-drop-list-item-collect-description > textarea{\
  4659. resize:both;\
  4660. width:auto;\
  4661. height:auto;\
  4662. }\
  4663. .pv-gallery-head-command-drop-list-item_disabled{\
  4664. color:#757575;\
  4665. }\
  4666. .pv-gallery-head-command-drop-list-item input + *{\
  4667. padding-left:3px;\
  4668. }\
  4669. .pv-gallery-head-command-drop-list-item input[type=number]{\
  4670. text-align:left;\
  4671. max-width:50px;\
  4672. height:20px;\
  4673. }\
  4674. .pv-gallery-head-command-drop-list-item input[type=checkbox]{\
  4675. width:20px\
  4676. }\
  4677. .pv-gallery-head-command-drop-list-item > * {\
  4678. vertical-align:middle;\
  4679. }\
  4680. .pv-gallery-head-command-drop-list-item label {\
  4681. font-weight: normal;\
  4682. display:inline\
  4683. }\
  4684. .pv-gallery-head-command-drop-list-item:hover{\
  4685. background-color:#404040;\
  4686. }\
  4687. /*container*/\
  4688. .pv-gallery-head-command-container{\
  4689. display:inline-block;\
  4690. height:100%;\
  4691. position:relative;\
  4692. }\
  4693. /* after伪类生成标识下拉菜单的三角图标*/\
  4694. .pv-gallery-head-command-container > .pv-gallery-head-command::after{\
  4695. content:"";\
  4696. display:inline-block;\
  4697. vertical-align:middle;\
  4698. border:none;\
  4699. border-top:7px solid #757575;\
  4700. border-left:5px solid transparent;\
  4701. border-right:5px solid transparent;\
  4702. margin-left:5px;\
  4703. -moz-transition:all 0.3s ease-in-out 0s;\
  4704. -webkit-transition:all 0.3s ease-in-out 0s;\
  4705. transition:all 0.3s ease-in-out 0s;\
  4706. }\
  4707. .pv-gallery-head-command-container:hover{\
  4708. box-shadow:0 0 3px #808080;\
  4709. }\
  4710. .pv-gallery-head-command-container:hover > .pv-gallery-head-command{\
  4711. background-color:#272727;\
  4712. color:#ccc;\
  4713. }\
  4714. .pv-gallery-head-command-container:hover > .pv-gallery-head-command::after{\
  4715. -webkit-transform:rotate(180deg);\
  4716. -moz-transform:rotate(180deg);\
  4717. transform:rotate(180deg);\
  4718. border-top:7px solid #ccc;\
  4719. }\
  4720. .pv-gallery-head-command-container:hover .pv-gallery-head-command-collect-icon{\
  4721. background-position:-20px 0;\
  4722. }\
  4723. .pv-gallery-head-command-container:hover .pv-gallery-head-command-slide-show-button{\
  4724. border-color:#ccc;\
  4725. }\
  4726. .pv-gallery-head-command-container:hover .pv-gallery-head-command-slide-show-button-inner{\
  4727. border-left-color:#ccc;\
  4728. }\
  4729. .pv-gallery-head-command-container:hover .pv-gallery-head-command-slide-show-button-inner_stop{\
  4730. border-color:#ccc;\
  4731. }\
  4732. .pv-gallery-head-command-container:hover > .pv-gallery-head-command-drop-list,.pv-gallery-head-command-container > .pv-gallery-head-command-drop-list.focus{\
  4733. display:block;\
  4734. }\
  4735. /*顶栏里面的按钮样式-结束*/\
  4736. .pv-gallery-body {\
  4737. display: block;\
  4738. height: 100%;\
  4739. width: 100%;\
  4740. margin: 0;\
  4741. padding: 0;\
  4742. border: none;\
  4743. border-top: 30px solid transparent;\
  4744. position: relative;\
  4745. background-clip: padding-box;\
  4746. z-index:0;\
  4747. }\
  4748. .pv-gallery-img-container {\
  4749. display: block;\
  4750. padding: 0;\
  4751. margin: 0;\
  4752. border: none;\
  4753. height: 100%;\
  4754. width: 100%;\
  4755. background-clip: padding-box;\
  4756. background-color: rgba(20,20,20,0.96);\
  4757. position:relative;\
  4758. }\
  4759. .pv-gallery-img-container-top {\
  4760. border-top: '+ prefs.gallery.sidebarSize +'px solid transparent;\
  4761. }\
  4762. .pv-gallery-img-container-right {\
  4763. border-right: '+ prefs.gallery.sidebarSize +'px solid transparent;\
  4764. }\
  4765. .pv-gallery-img-container-bottom {\
  4766. border-bottom: '+ prefs.gallery.sidebarSize +'px solid transparent;\
  4767. }\
  4768. .pv-gallery-img-container-left {\
  4769. border-left: '+ prefs.gallery.sidebarSize +'px solid transparent;\
  4770. }\
  4771. /*大图区域的切换控制按钮*/\
  4772. .pv-gallery-img-controler{\
  4773. position:absolute;\
  4774. top:50%;\
  4775. height:60px;\
  4776. width:50px;\
  4777. margin-top:-30px;\
  4778. cursor:pointer;\
  4779. opacity:0.3;\
  4780. z-index:1;\
  4781. }\
  4782. .pv-gallery-img-controler-pre{\
  4783. background:rgba(70,70,70,0.5) url("'+prefs.icons.arrowLeft+'") no-repeat center;\
  4784. left:10px;\
  4785. }\
  4786. .pv-gallery-img-controler-next{\
  4787. background:rgba(70,70,70,0.5) url("'+prefs.icons.arrowRight+'") no-repeat center;\
  4788. right:10px;\
  4789. }\
  4790. .pv-gallery-img-controler:hover{\
  4791. background-color:rgba(140,140,140,0.5);\
  4792. opacity:0.9;\
  4793. z-index:2;\
  4794. }\
  4795. /*滚动条样式--开始*/\
  4796. .pv-gallery-scrollbar-h,\
  4797. .pv-gallery-scrollbar-v{\
  4798. display:none;\
  4799. z-index:1;\
  4800. position:absolute;\
  4801. margin:0;\
  4802. padding:0;\
  4803. border:none;\
  4804. }\
  4805. .pv-gallery-scrollbar-h{\
  4806. bottom:10px;\
  4807. left:0;\
  4808. right:0;\
  4809. height:10px;\
  4810. margin:0 2px;\
  4811. }\
  4812. .pv-gallery-scrollbar-v{\
  4813. top:0;\
  4814. bottom:0;\
  4815. right:10px;\
  4816. width:10px;\
  4817. margin:2px 0;\
  4818. }\
  4819. .pv-gallery-scrollbar-h:hover{\
  4820. height:25px;\
  4821. bottom:0;\
  4822. }\
  4823. .pv-gallery-scrollbar-v:hover{\
  4824. width:25px;\
  4825. }\
  4826. .pv-gallery-scrollbar-h:hover,\
  4827. .pv-gallery-scrollbar-v:hover{\
  4828. background-color:rgba(100,100,100,0.9);\
  4829. z-index:2;\
  4830. }\
  4831. .pv-gallery-scrollbar-h-track,\
  4832. .pv-gallery-scrollbar-v-track{\
  4833. position:absolute;\
  4834. top:0;\
  4835. left:0;\
  4836. right:0;\
  4837. bottom:0;\
  4838. background-color:rgba(100,100,100,0.3);\
  4839. border:2px solid transparent;\
  4840. }\
  4841. .pv-gallery-scrollbar-h-handle,\
  4842. .pv-gallery-scrollbar-v-handle{\
  4843. position:absolute;\
  4844. background-color:rgba(0,0,0,0.5);\
  4845. }\
  4846. .pv-gallery-scrollbar-h-handle{\
  4847. height:100%;\
  4848. }\
  4849. .pv-gallery-scrollbar-v-handle{\
  4850. width:100%;\
  4851. }\
  4852. .pv-gallery-scrollbar-h-handle:hover,\
  4853. .pv-gallery-scrollbar-v-handle:hover{\
  4854. background-color:rgba(80,33,33,1);\
  4855. border: 2px solid #585757;\
  4856. }\
  4857. .pv-gallery-scrollbar-h-handle:active,\
  4858. .pv-gallery-scrollbar-v-handle:active{\
  4859. background-color:rgba(57,26,26,1);\
  4860. border: 2px solid #878484;\
  4861. }\
  4862. /*滚动条样式--结束*/\
  4863. .pv-gallery-img-content{\
  4864. display:block;\
  4865. width:100%;\
  4866. height:100%;\
  4867. overflow:hidden;\
  4868. text-align:center;\
  4869. padding:0;\
  4870. border:none;\
  4871. margin:0;\
  4872. line-height:0;\
  4873. font-size:0;\
  4874. white-space:nowrap;\
  4875. }\
  4876. .pv-gallery-img-parent{\
  4877. display:inline-block;\
  4878. vertical-align:middle;\
  4879. line-height:0;\
  4880. }\
  4881. .pv-gallery-img_broken{\
  4882. display:none;\
  4883. cursor:pointer;\
  4884. }\
  4885. .pv-gallery-img{\
  4886. position:relative;\/*辅助e.layerX,layerY*/\
  4887. display:inline-block;\
  4888. vertical-align:middle;\
  4889. width:auto;\
  4890. height:auto;\
  4891. padding:0;\
  4892. border:5px solid #313131;\
  4893. margin:10px;\
  4894. opacity:0.6;\
  4895. -webkit-transform:scale(1.2);\
  4896. -moz-transform:scale(1.2);\
  4897. transform:scale(1.2);\
  4898. '+
  4899. (prefs.gallery.transition ? ('\
  4900. -webkit-transition: opacity 0.15s ease-in-out,\
  4901. -webkit-transform 0.1s ease-in-out;\
  4902. -moz-transition: opacity 0.15s ease-in-out,\
  4903. -moz-transform 0.1s ease-in-out;\
  4904. transition: opacity 0.15s ease-in-out,\
  4905. transform 0.1s ease-in-out;\
  4906. ') : '') + '\
  4907. }\
  4908. .pv-gallery-img_zoom-out{\
  4909. cursor:'+support.cssCursorValue.zoomOut+';\
  4910. }\
  4911. .pv-gallery-img_zoom-in{\
  4912. cursor:'+support.cssCursorValue.zoomIn+';\
  4913. }\
  4914. .pv-gallery-sidebar-toggle{\
  4915. position:absolute;\
  4916. line-height:0;\
  4917. text-align:center;\
  4918. background-color:rgb(0,0,0);\
  4919. color:#757575;\
  4920. white-space:nowrap;\
  4921. cursor:pointer;\
  4922. z-index:1;\
  4923. display:none;\
  4924. }\
  4925. .pv-gallery-sidebar-viewmore{\
  4926. position:absolute;\
  4927. line-height:0;\
  4928. text-align:center;\
  4929. background-color:#000000;\
  4930. color:#757575;\
  4931. white-space:nowrap;\
  4932. cursor:pointer;\
  4933. z-index:1;\
  4934. display:none;\
  4935. height: 30px;\
  4936. width:30px;\
  4937. border-radius: 15px;\
  4938. line-height: 2 !important;\
  4939. font-family: auto;\
  4940. opacity: 0.5;\
  4941. }\
  4942. .pv-gallery-sidebar-viewmore:hover{\
  4943. opacity: 1;\
  4944. }\
  4945. .pv-gallery-maximize-container>p{\
  4946. position: fixed;\
  4947. width: 100%;\
  4948. text-align: center;\
  4949. pointer-events: none;\
  4950. margin-top: 25px;\
  4951. }\
  4952. .pv-gallery-maximize-container>p>input{\
  4953. pointer-events: all;\
  4954. color: white;\
  4955. background-color: black;\
  4956. border: 0;\
  4957. opacity: 0.8;\
  4958. padding: 5px 10px;\
  4959. cursor: pointer;\
  4960. margin: 1px;\
  4961. font-size: 20px;\
  4962. }\
  4963. .pv-gallery-maximize-container{\
  4964. width: 100%;\
  4965. display: block;\
  4966. background: black;\
  4967. }\
  4968. .pv-gallery-maximize-container.pv-gallery-flex-maximize{\
  4969. column-count: unset;\
  4970. -moz-column-count: unset;\
  4971. -webkit-column-count: unset;\
  4972. display: flex;\
  4973. flex-flow: wrap;\
  4974. }\
  4975. .pv-gallery-maximize-container.pv-gallery-flex-maximize span{\
  4976. width: 18.5%;\
  4977. height: 30vh;\
  4978. }\
  4979. .pv-gallery-maximize-container.pv-gallery-flex-maximize img{\
  4980. position: relative;\
  4981. width: unset;\
  4982. max-height: 100%;\
  4983. max-width: 100%;\
  4984. top: 50%;\
  4985. transform: translateY(-50%) scale3d(1, 1, 1);\
  4986. }\
  4987. .pv-gallery-maximize-container.pv-gallery-flex-maximize img:hover {\
  4988. transform: translateY(-50%) scale3d(1.1, 1.1, 1.1);\
  4989. }\
  4990. .pv-gallery-maximize-container span{\
  4991. -moz-page-break-inside: avoid;\
  4992. -webkit-column-break-inside: avoid;\
  4993. break-inside: avoid;\
  4994. float: left;\
  4995. margin-bottom: 15px;\
  4996. margin-right: 15px;\
  4997. overflow: hidden;\
  4998. position: relative;\
  4999. }\
  5000. .pv-gallery-maximize-container>.maximizeChild{\
  5001. display: inline-block;\
  5002. vertical-align: middle;\
  5003. text-align: center;\
  5004. }\
  5005. .pv-gallery-maximize-container>.maximizeChild.selected{\
  5006. border: 5px solid #ff0000;\
  5007. }\
  5008. .pv-gallery-maximize-container img{\
  5009. width:100%;\
  5010. transition: transform .3s ease 0s;\
  5011. transform: scale3d(1, 1, 1);\
  5012. cursor: zoom-in;\
  5013. }\
  5014. .pv-gallery-maximize-container img:hover {\
  5015. transform: scale3d(1.1, 1.1, 1.1);\
  5016. opacity: .9;\
  5017. }\
  5018. .pv-gallery-maximize-container span>p{\
  5019. position: absolute;\
  5020. width: 100%;\
  5021. max-height: 40%;\
  5022. font-size: 18px;\
  5023. text-align: center;\
  5024. background: #000;\
  5025. color: #fff;\
  5026. left: 0;\
  5027. user-select: none;\
  5028. word-break: break-all;\
  5029. display: inline;\
  5030. margin: 0 auto;\
  5031. }\
  5032. .pv-gallery-maximize-container span>p.pv-bottom-banner{\
  5033. bottom: 0;\
  5034. height: 35px;\
  5035. cursor: pointer;\
  5036. line-height: 40px;\
  5037. }\
  5038. .pv-gallery-maximize-container span>p.pv-top-banner{\
  5039. top: 0;\
  5040. height: 25px;\
  5041. }\
  5042. .pv-gallery-maximize-container span:hover>p{\
  5043. opacity: 0.6;\
  5044. }\
  5045. .pv-gallery-maximize-container span>p.pv-bottom-banner:hover{\
  5046. color:red;\
  5047. font-weight:bold;\
  5048. }\
  5049. .pv-gallery-maximize-container span>input{\
  5050. position: absolute;\
  5051. top: 2px;\
  5052. width: 20px;\
  5053. height: 20px;\
  5054. opacity: 0;\
  5055. left: 0;\
  5056. display: inline;\
  5057. }\
  5058. .pv-gallery-maximize-container.checked span>input{\
  5059. opacity: 1;\
  5060. }\
  5061. .pv-gallery-maximize-container.checked span>.pv-top-banner{\
  5062. opacity: 0.6;\
  5063. }\
  5064. .pv-gallery-maximize-container span:hover>input{\
  5065. opacity: 1;\
  5066. }\
  5067. .pv-gallery-maximize-scroll{\
  5068. overflow-y: scroll;\
  5069. height: 100%;\
  5070. width: 100%;\
  5071. position: absolute;\
  5072. visibility: hidden;\
  5073. top: 0;\
  5074. left: 0;\
  5075. }\
  5076. .pv-gallery-sidebar-toggle:hover,.pv-gallery-sidebar-viewmore:hover{\
  5077. color:#ccc;\
  5078. }\
  5079. .pv-gallery-sidebar-toggle-h{\
  5080. width:80px;\
  5081. margin-left:-40px;\
  5082. left:50%;\
  5083. }\
  5084. .pv-gallery-sidebar-viewmore-h{\
  5085. margin-left:-15px;\
  5086. left:50%;\
  5087. }\
  5088. .pv-gallery-sidebar-toggle-v{\
  5089. height:80px;\
  5090. margin-top:-40px;\
  5091. top:50%;\
  5092. }\
  5093. .pv-gallery-sidebar-viewmore-v{\
  5094. height:30px;\
  5095. top:6%;\
  5096. }\
  5097. .pv-gallery-sidebar-toggle-top{\
  5098. top:-5px;\
  5099. }\
  5100. .pv-gallery-sidebar-viewmore-top{\
  5101. top:15px;\
  5102. }\
  5103. .pv-gallery-sidebar-toggle-right,.pv-gallery-sidebar-viewmore-right{\
  5104. right:-5px;\
  5105. }\
  5106. .pv-gallery-sidebar-toggle-bottom{\
  5107. bottom:-5px;\
  5108. }\
  5109. .pv-gallery-sidebar-viewmore-bottom{\
  5110. display: block;\
  5111. background-color: #000000;\
  5112. bottom:12px;\
  5113. }\
  5114. .pv-gallery-sidebar-viewmore-bottom.showmore{\
  5115. background-color: rgb(42, 42, 42);\
  5116. }\
  5117. .pv-gallery-sidebar-toggle-left,.pv-gallery-sidebar-viewmore-left{\
  5118. left:-5px;\
  5119. }\
  5120. .pv-gallery-sidebar-toggle-content{\
  5121. display:inline-block;\
  5122. vertical-align:middle;\
  5123. white-space:normal;\
  5124. word-wrap:break-word;\
  5125. overflow-wrap:break-word;\
  5126. line-height:1.1;\
  5127. font-size:12px;\
  5128. text-align:center;\
  5129. margin-bottom:8px;\
  5130. }\
  5131. .pv-gallery-sidebar-viewmore-content{\
  5132. display:inline-block;\
  5133. vertical-align:middle;\
  5134. white-space:normal;\
  5135. word-wrap:break-word;\
  5136. overflow-wrap:break-word;\
  5137. line-height:1.1;\
  5138. font-size:16px;\
  5139. text-align:center;\
  5140. }\
  5141. .pv-gallery-sidebar-toggle-content-v,.pv-gallery-sidebar-viewmore-content-v{\
  5142. width:1.1em;\
  5143. }\
  5144. /*侧边栏开始*/\
  5145. .pv-gallery-sidebar-container {\
  5146. position: absolute;\
  5147. background-color:rgb(0,0,0);\
  5148. padding:5px;\
  5149. border:none;\
  5150. margin:none;\
  5151. text-align:center;\
  5152. line-height:0;\
  5153. white-space:nowrap;\
  5154. -o-user-select: none;\
  5155. -webkit-user-select: none;\
  5156. -moz-user-select: -moz-none;\
  5157. user-select: none;\
  5158. }\
  5159. .pv-gallery-sidebar-container-h {\
  5160. height: '+ prefs.gallery.sidebarSize +'px;\
  5161. width: 100%;\
  5162. }\
  5163. .pv-gallery-sidebar-container-v {\
  5164. width: '+ prefs.gallery.sidebarSize +'px;\
  5165. height: 100%;\
  5166. }\
  5167. .pv-gallery-sidebar-container-top {\
  5168. top: 0;\
  5169. left: 0;\
  5170. border-bottom:1px solid #333333;\
  5171. }\
  5172. .pv-gallery-sidebar-container-right {\
  5173. top: 0;\
  5174. right: 0;\
  5175. border-left:1px solid #333333;\
  5176. }\
  5177. .pv-gallery-sidebar-container-bottom {\
  5178. bottom: 0;\
  5179. left: 0;\
  5180. border-top:1px solid #333333;\
  5181. }\
  5182. .pv-gallery-sidebar-container-left {\
  5183. top: 0;\
  5184. left: 0;\
  5185. border-right:1px solid #333333;\
  5186. }\
  5187. .pv-gallery-sidebar-content {\
  5188. display: inline-block;\
  5189. margin: 0;\
  5190. padding: 0;\
  5191. border: none;\
  5192. background-clip: padding-box;\
  5193. vertical-align:middle;\
  5194. position:relative;\
  5195. text-align:left;\
  5196. }\
  5197. .pv-gallery-sidebar-content-h {\
  5198. height: 100%;\
  5199. width: 90%;\
  5200. border-left: 40px solid transparent;\
  5201. border-right: 40px solid transparent;\
  5202. }\
  5203. .pv-gallery-sidebar-content-v {\
  5204. height: 90%;\
  5205. width: 100%;\
  5206. border-top: 40px solid transparent;\
  5207. border-bottom: 40px solid transparent;\
  5208. }\
  5209. .pv-gallery-sidebar-controler{\
  5210. cursor:pointer;\
  5211. position:absolute;\
  5212. background:rgba(255,255,255,0.1) no-repeat center;\
  5213. }\
  5214. .pv-gallery-sidebar-controler:hover{\
  5215. background-color:rgba(255,255,255,0.3);\
  5216. }\
  5217. .pv-gallery-sidebar-controler-pre-h,\
  5218. .pv-gallery-sidebar-controler-next-h{\
  5219. top:0;\
  5220. width:36px;\
  5221. height:100%;\
  5222. }\
  5223. .pv-gallery-sidebar-controler-pre-v,\
  5224. .pv-gallery-sidebar-controler-next-v{\
  5225. left:0;\
  5226. width:100%;\
  5227. height:36px;\
  5228. }\
  5229. .pv-gallery-sidebar-controler-pre-h {\
  5230. left: -40px;\
  5231. background-image: url("'+prefs.icons.arrowLeft+'");\
  5232. }\
  5233. .pv-gallery-sidebar-controler-next-h {\
  5234. right: -40px;\
  5235. background-image: url("'+prefs.icons.arrowRight+'");\
  5236. }\
  5237. .pv-gallery-sidebar-controler-pre-v {\
  5238. top: -40px;\
  5239. background-image: url("'+prefs.icons.arrowTop+'");\
  5240. }\
  5241. .pv-gallery-sidebar-controler-next-v {\
  5242. bottom: -40px;\
  5243. background-image: url("'+prefs.icons.arrowBottom+'");\
  5244. }\
  5245. .pv-gallery-sidebar-thumbnails-container {\
  5246. display: block;\
  5247. overflow: hidden;\
  5248. height: 100%;\
  5249. width: 100%;\
  5250. margin:0;\
  5251. border:none;\
  5252. padding:0;\
  5253. line-height:0;\
  5254. position:relative;\
  5255. }\
  5256. .pv-gallery-sidebar-thumbnails-container span{\
  5257. vertical-align:middle;\
  5258. }\
  5259. .pv-gallery-sidebar-thumbnails-container-h{\
  5260. border-left:1px solid #464646;\
  5261. border-right:1px solid #464646;\
  5262. white-space:nowrap;\
  5263. }\
  5264. .pv-gallery-sidebar-thumbnails-container-v{\
  5265. border-top:1px solid #464646;\
  5266. border-bottom:1px solid #464646;\
  5267. white-space:normal;\
  5268. }\
  5269. .pv-gallery-sidebar-thumbnails-container-top {\
  5270. padding-bottom:5px;\
  5271. }\
  5272. .pv-gallery-sidebar-thumbnails-container-right {\
  5273. padding-left:5px;\
  5274. }\
  5275. .pv-gallery-sidebar-thumbnails-container-bottom {\
  5276. padding-top:5px;\
  5277. }\
  5278. .pv-gallery-sidebar-thumbnails-container-left {\
  5279. padding-right:5px;\
  5280. }\
  5281. .pv-gallery-sidebar-thumb-container {\
  5282. display:inline-block;\
  5283. text-align: center;\
  5284. border:2px solid rgb(52,52,52);\
  5285. cursor:pointer;\
  5286. position:relative;\
  5287. padding:2px;\
  5288. font-size:0;\
  5289. line-height:0;\
  5290. white-space:nowrap;\
  5291. vertical-align: middle;\
  5292. top:0;\
  5293. left:0;\
  5294. -webkit-transition:all 0.2s ease-in-out;\
  5295. transition:all 0.2s ease-in-out;\
  5296. }\
  5297. .pv-gallery-sidebar-thumbnails-container-h .pv-gallery-sidebar-thumb-container {\
  5298. margin:0 2px;\
  5299. height:100%;\
  5300. }\
  5301. .pv-gallery-sidebar-thumbnails-container-v .pv-gallery-sidebar-thumb-container {\
  5302. margin:2px 0;\
  5303. width:100%;\
  5304. }\
  5305. .pv-gallery-sidebar-thumbnails_hide-span > .pv-gallery-sidebar-thumb-container {\
  5306. display:none;\
  5307. }\
  5308. .pv-gallery-sidebar-thumb-container:hover {\
  5309. border:2px solid rgb(57,149,211);\
  5310. }\
  5311. .pv-gallery-sidebar-thumb_selected {\
  5312. border:2px solid rgb(229,59,62);\
  5313. }\
  5314. .pv-gallery-sidebar-thumb_selected-top {\
  5315. top:5px;\
  5316. }\
  5317. .pv-gallery-sidebar-thumb_selected-right {\
  5318. left:-5px;\
  5319. }\
  5320. .pv-gallery-sidebar-thumb_selected-bottom {\
  5321. top:-5px;\
  5322. }\
  5323. .pv-gallery-sidebar-thumb_selected-left {\
  5324. left:5px;\
  5325. }\
  5326. .pv-gallery-sidebar-thumb-loading{\
  5327. position:absolute;\
  5328. top:0;\
  5329. left:0;\
  5330. text-align:center;\
  5331. width:100%;\
  5332. height:100%;\
  5333. display:none;\
  5334. opacity:0.6;\
  5335. background:black url("'+ prefs.icons.loading + '") no-repeat center ;\
  5336. }\
  5337. .pv-gallery-sidebar-thumb-loading:hover{\
  5338. opacity:0.8;\
  5339. }\
  5340. .pv-gallery-sidebar-thumb {\
  5341. display: inline-block;\
  5342. vertical-align: middle;\
  5343. max-width: 100% !important;\
  5344. max-height: 100% !important;\
  5345. height: auto !important;\
  5346. width: auto !important;\
  5347. }\
  5348. .pv-gallery-vertical-align-helper{\
  5349. display:inline-block;\
  5350. vertical-align:middle;\
  5351. width:0;\
  5352. height:100%;\
  5353. margin:0;\
  5354. border:0;\
  5355. padding:0;\
  5356. visibility:hidden;\
  5357. white-space:nowrap;\
  5358. background-color:red;\
  5359. }\
  5360. ';
  5361. var head=document.head;
  5362. head.appendChild(style);
  5363. this.globalSSheet=style.sheet;
  5364.  
  5365. var style2=document.createElement('style');
  5366. this.thumbVisibleStyle=style2;
  5367. style2.type='text/css';
  5368. head.appendChild(style2);
  5369.  
  5370. // 让 description 的文字内容溢出用点点点(...)省略号表示
  5371. // .pv-gallery-head-left-img-info-description {
  5372. // overflow: hidden;
  5373. // text-overflow: ellipsis;
  5374. // white-space: nowrap;
  5375. // width: 27em;
  5376. // }
  5377. },
  5378.  
  5379. };
  5380.  
  5381.  
  5382. GalleryC.prototype.Preload.prototype={//预读对象
  5383. init:function(){
  5384. if(!this.container){//预读的图片都仍里面
  5385. var div=document.createElement('div');
  5386. div.className='pv-gallery-preloaded-img-container';
  5387. div.style.display='none';
  5388. document.body.appendChild(div);
  5389. GalleryC.prototype.Preload.prototype.container=div;
  5390. };
  5391. this.max=prefs.gallery.max;
  5392. this.nextNumber=0;
  5393. this.nextEle=this.ele;
  5394. this.preNumber=0;
  5395. this.preEle=this.ele;
  5396. this.direction='pre';
  5397. },
  5398. preload:function(){
  5399. var ele=this.getPreloadEle();
  5400. if(!ele){
  5401. return;
  5402. };
  5403.  
  5404. var self=this;
  5405. this.imgReady=imgReady(dataset(ele,'src'),{
  5406. loadEnd:function(){
  5407. if(self.aborted){
  5408. return;
  5409. };
  5410. dataset(ele,'preloaded','true')
  5411. self.container.appendChild(this);
  5412. self.preload();
  5413. },
  5414. time:60 * 1000,//限时一分钟,否则强制结束并开始预读下一张。
  5415. });
  5416. },
  5417. getPreloadEle:function(){
  5418. if((this.max<=this.nextNumber && this.max<=this.preNumber) || (!this.nextEle && !this.preEle)){
  5419. return;
  5420. };
  5421. var ele=this.direction=='pre'? this.getNext() : this.getPrevious();
  5422. if(ele && !dataset(ele,'preloaded')){
  5423. return ele;
  5424. }else{
  5425. return this.getPreloadEle();
  5426. };
  5427. },
  5428. getNext:function(){
  5429. this.nextNumber++;
  5430. this.direction='next';
  5431. if(!this.nextEle)return;
  5432. return (this.nextEle = this.oriThis.getThumSpan(false,this.nextEle));
  5433. },
  5434. getPrevious:function(){
  5435. this.preNumber++;
  5436. this.direction='pre';
  5437. if(!this.preEle)return;
  5438. return (this.preEle = this.oriThis.getThumSpan(true,this.preEle));
  5439. },
  5440. abort:function(){
  5441. this.aborted=true;
  5442. if(this.imgReady){
  5443. this.imgReady.abort();
  5444. };
  5445. },
  5446. };
  5447.  
  5448.  
  5449. GalleryC.prototype.Scrollbar.prototype={//滚动条对象
  5450. init:function(){
  5451. var bar=this.scrollbar.bar;
  5452. this.shown=bar.offsetWidth!=0;
  5453. var self=this;
  5454. bar.addEventListener('mousedown',function(e){//点击滚动条区域,该干点什么!
  5455. e.preventDefault();
  5456. var target=e.target;
  5457. var handle=self.scrollbar.handle;
  5458. var track=self.scrollbar.track;
  5459. switch(target){
  5460. case handle:{//手柄;功能,拖动手柄来滚动窗口
  5461. let pro=self.isHorizontal ? ['left','clientX'] : ['top','clientY'];
  5462. var oHOffset=parseFloat(handle.style[pro[0]]);
  5463. var oClient=e[pro[1]];
  5464.  
  5465. var moveHandler=function(e){
  5466. self.scroll(oHOffset + e[pro[1]] - oClient,true);
  5467. };
  5468. let upHandler=function(){
  5469. document.removeEventListener('mousemove',moveHandler,true);
  5470. document.removeEventListener('mouseup',upHandler,true);
  5471. };
  5472. document.addEventListener('mousemove',moveHandler,true);
  5473. document.addEventListener('mouseup',upHandler,true);
  5474. }break;
  5475. case track:{//轨道;功能,按住不放来连续滚动一个页面的距离
  5476. let pro=self.isHorizontal ? ['left','offsetX','layerX','clientWidth','offsetWidth'] : ['top' , 'offsetY' ,'layerY','clientHeight','offsetHeight'];
  5477. var clickOffset=typeof e[pro[1]]=='undefined' ? e[pro[2]] : e[pro[1]];
  5478. var handleOffset=parseFloat(handle.style[pro[0]]);
  5479. var handleSize=handle[pro[4]];
  5480. var under= clickOffset > handleOffset ;//点击在滚动手柄的下方
  5481. var containerSize=self.container[pro[3]];
  5482.  
  5483. var scroll=function(){
  5484. self.scrollBy(under? (containerSize - 10) : (-containerSize + 10));//滚动一个页面距离少一点
  5485. };
  5486. scroll();
  5487.  
  5488. var checkStop=function(){//当手柄到达点击位置时停止
  5489. var handleOffset=parseFloat(handle.style[pro[0]]);
  5490. if(clickOffset >= handleOffset && clickOffset <= (handleOffset + handleSize)){
  5491. clearTimeout(scrollTimeout);
  5492. clearInterval(scrollInterval);
  5493. };
  5494. };
  5495.  
  5496.  
  5497. var scrollInterval;
  5498. var scrollTimeout=setTimeout(function(){
  5499. scroll();
  5500. scrollInterval=setInterval(function(){
  5501. scroll();
  5502. checkStop();
  5503. },120);
  5504. checkStop();
  5505. },300);
  5506.  
  5507.  
  5508. checkStop();
  5509.  
  5510. let upHandler=function(){
  5511. clearTimeout(scrollTimeout);
  5512. clearInterval(scrollInterval);
  5513. document.removeEventListener('mouseup',upHandler,true);
  5514. };
  5515. document.addEventListener('mouseup',upHandler,true);
  5516. }break;
  5517. };
  5518.  
  5519. },true);
  5520. },
  5521. reset:function(){//判断滚动条该显示还是隐藏
  5522.  
  5523. var pro=this.isHorizontal ? ['scrollWidth','clientWidth','width'] : ['scrollHeight','clientHeight','height'];
  5524.  
  5525. //如果内容大于容器的content区域
  5526.  
  5527. var scrollSize=this.container[pro[0]];
  5528. var clientSize=this.container[pro[1]];
  5529. var scrollMax=scrollSize - clientSize;
  5530. this.scrollMax=scrollMax;
  5531. if(scrollMax>0){
  5532. this.show();
  5533. var trackSize=this.scrollbar.track[pro[1]];
  5534. this.trackSize=trackSize;
  5535. var handleSize=Math.floor((clientSize/scrollSize) * trackSize);
  5536. handleSize=Math.max(20,handleSize);//限制手柄的最小大小;
  5537. this.handleSize=handleSize;
  5538. this.one=(trackSize-handleSize) / scrollMax;//一个像素对应的滚动条长度
  5539. this.scrollbar.handle.style[pro[2]]= handleSize + 'px';
  5540. this.scroll(this.getScrolled());
  5541. }else{
  5542. this.hide();
  5543. };
  5544. },
  5545. show:function(){
  5546. if(this.shown)return;
  5547. this.shown=true;
  5548. this.scrollbar.bar.style.display='block';
  5549. },
  5550. hide:function(){
  5551. if(!this.shown)return;
  5552. this.shown=false;
  5553. this.scrollbar.bar.style.display='none';
  5554. },
  5555. scrollBy:function(distance,handleDistance){
  5556. return this.scroll(this.getScrolled() + (handleDistance? distance / this.one : distance));
  5557. },
  5558. scrollByPages:function(num){
  5559. this.scroll(this.getScrolled() + (this.container[(this.isHorizontal ? 'clientWidth' : 'clientHeight')] - 10) * num);
  5560. },
  5561. scroll:function(distance,handleDistance,transition){
  5562. if(!this.shown)return;
  5563.  
  5564. //滚动实际滚动条
  5565. var _distance=distance;
  5566. _distance=handleDistance? distance / this.one : distance;
  5567. _distance=Math.max(0,_distance);
  5568. _distance=Math.min(_distance,this.scrollMax);
  5569.  
  5570.  
  5571. var pro=this.isHorizontal? ['left','scrollLeft'] : ['top','scrollTop'];
  5572.  
  5573.  
  5574. //滚动虚拟滚动条
  5575. //根据比例转换为滚动条上应该滚动的距离。
  5576. distance=handleDistance? distance : this.one * distance;
  5577. //处理非法值
  5578. distance=Math.max(0,distance);//如果值小于0那么取0
  5579. distance=Math.min(distance,this.trackSize - this.handleSize);//大于极限值,取极限值
  5580.  
  5581. var shs=this.scrollbar.handle.style;
  5582. var container=this.container;
  5583. if(transition){
  5584. clearInterval(this.transitionInterval);
  5585.  
  5586. var start=0;
  5587. var duration=10;
  5588.  
  5589. var cStart=this.getScrolled();
  5590. var cChange=_distance-cStart;
  5591. var sStart=parseFloat(shs[pro[0]]);
  5592. var sChange=distance-sStart;
  5593.  
  5594. var transitionInterval=setInterval(function(){
  5595. var cEnd=Tween.Cubic.easeInOut(start,cStart,cChange,duration);
  5596. var sEnd=Tween.Cubic.easeInOut(start,sStart,sChange,duration);
  5597.  
  5598. container[pro[1]]=cEnd;
  5599. shs[pro[0]]=sEnd + 'px';
  5600.  
  5601. start++;
  5602. if(start>=duration){
  5603. clearInterval(transitionInterval);
  5604. };
  5605. },35);
  5606.  
  5607. this.transitionInterval=transitionInterval;
  5608.  
  5609. return;
  5610. };
  5611.  
  5612. var noScroll=shs[pro[0]].replace(/(\.\d*)?\s*px/,"")==Math.floor(distance);
  5613. shs[pro[0]]=distance + 'px';
  5614. container[pro[1]]=_distance;
  5615. return noScroll;
  5616. },
  5617. getScrolled:function(){
  5618. return this.container[(this.isHorizontal ? 'scrollLeft' : 'scrollTop')];
  5619. },
  5620. };
  5621.  
  5622.  
  5623. //放大镜
  5624. function MagnifierC(img,data){
  5625. this.img=img;
  5626. this.data=data;
  5627. this.init();
  5628. };
  5629.  
  5630. MagnifierC.all=[];
  5631. MagnifierC.styleZIndex=900000000;//全局z-index;
  5632.  
  5633. MagnifierC.prototype={
  5634. init:function(){
  5635. MagnifierC.zoomRange=prefs.magnifier.wheelZoom.range.slice(0).sort((a, b)=>{return a - b});
  5636. MagnifierC.zoomRangeR=MagnifierC.zoomRange.slice(0).reverse();//降序
  5637. this.addStyle();
  5638. MagnifierC.all.push(this);
  5639. var container=document.createElement('span');
  5640.  
  5641. container.className='pv-magnifier-container';
  5642. document.body.appendChild(container);
  5643.  
  5644. this.magnifier=container;
  5645.  
  5646. var imgNaturalSize={
  5647. h:this.img.naturalHeight,
  5648. w:this.img.naturalWidth,
  5649. };
  5650.  
  5651. this.imgNaturalSize=imgNaturalSize;
  5652.  
  5653. var cs=container.style;
  5654. cs.zIndex=MagnifierC.styleZIndex++;
  5655.  
  5656.  
  5657.  
  5658. var maxDia=Math.ceil(Math.sqrt(Math.pow(1/2*imgNaturalSize.w,2) + Math.pow(1/2*imgNaturalSize.h,2)) * 2);
  5659. this.maxDia=maxDia;
  5660.  
  5661. var radius=prefs.magnifier.radius;
  5662. radius=Math.min(maxDia/2,radius);
  5663. this.radius=radius;
  5664. var diameter=radius * 2;
  5665. this.diameter=diameter;
  5666.  
  5667. cs.width=diameter + 'px';
  5668. cs.height=diameter + 'px';
  5669. cs.borderRadius=radius+1 + 'px';
  5670. cs.backgroundImage='url("'+ this.img.src +'")';
  5671. cs.marginLeft= -radius +'px';
  5672. cs.marginTop= -radius +'px';
  5673.  
  5674. var imgPos=getContentClientRect(this.data.img);
  5675. var wScrolled=getScrolled();
  5676. var imgRange={//图片所在范围
  5677. x:[imgPos.left + wScrolled.x , imgPos.right + wScrolled.x],
  5678. y:[imgPos.top + wScrolled.y, imgPos.bottom + wScrolled.y],
  5679. };
  5680. var imgW=imgRange.x[1] - imgRange.x[0];
  5681. var imgH=imgRange.y[1] - imgRange.y[0];
  5682. //如果图片太小的话,进行范围扩大。
  5683. var minSize=60;
  5684. if(imgW < minSize){
  5685. imgRange.x[1] +=(minSize - imgW)/2;
  5686. imgRange.x[0] -=(minSize - imgW)/2;
  5687. imgW=minSize;
  5688. };
  5689. if(imgH < minSize){
  5690. imgRange.y[1] +=(minSize - imgH)/2;
  5691. imgRange.y[0] -=(minSize - imgH)/2;
  5692. imgH=minSize;
  5693. };
  5694. this.imgSize={
  5695. w:imgW,
  5696. h:imgH,
  5697. };
  5698. this.imgRange=imgRange;
  5699.  
  5700. this.setMouseRange();
  5701.  
  5702.  
  5703. this.move({
  5704. pageX:imgRange.x[0],
  5705. pageY:imgRange.y[0],
  5706. });
  5707.  
  5708. this._focus=this.focus.bind(this);
  5709. this._blur=this.blur.bind(this);
  5710. this._move=this.move.bind(this);
  5711. this._remove=this.remove.bind(this);
  5712. this._pause=this.pause.bind(this);
  5713. this._zoom=this.zoom.bind(this);
  5714. this._clickOut=this.clickOut.bind(this);
  5715.  
  5716. if(prefs.magnifier.wheelZoom.enabled){
  5717. this.zoomLevel=1;
  5718. this.defaultDia=diameter;
  5719. addWheelEvent(container,this._zoom,false);
  5720. };
  5721.  
  5722. container.addEventListener('mouseover',this._focus,false);
  5723. container.addEventListener('mouseout',this._blur,false);
  5724. container.addEventListener('dblclick',this._remove,false);
  5725. container.addEventListener('click',this._pause,false);
  5726.  
  5727.  
  5728. document.addEventListener('mousemove',this._move,true);
  5729. document.addEventListener('mouseup',this._clickOut,true);
  5730. },
  5731. addStyle:function(){
  5732. if(MagnifierC.style)return;
  5733. var style=document.createElement('style');
  5734. style.type='text/css';
  5735. MagnifierC.style=style;
  5736. style.textContent='\
  5737. .pv-magnifier-container{\
  5738. position:absolute;\
  5739. padding:0;\
  5740. margin:0;\
  5741. background-origin:border-box;\
  5742. -moz-box-sizing:border-box;\
  5743. box-sizing:border-box;\
  5744. border:3px solid #CCCCCC;\
  5745. background:rgba(40, 40, 40, 0.9) no-repeat;\
  5746. }\
  5747. .pv-magnifier-container_focus{\
  5748. box-shadow: 0px 0px 6px rgba(0, 0, 0, 0.7);\
  5749. }\
  5750. .pv-magnifier-container_pause{\
  5751. border-color:red;\
  5752. }\
  5753. ';
  5754. document.head.appendChild(style);
  5755. },
  5756. clickOut:function(){
  5757. if(!this.magnifier.classList.contains("pv-magnifier-container_focus")){
  5758. this.remove();
  5759. }
  5760. },
  5761. focus:function(){
  5762. this.magnifier.classList.add('pv-magnifier-container_focus');
  5763. this.magnifier.style.zIndex=MagnifierC.styleZIndex++;
  5764. },
  5765. blur:function(){
  5766. this.magnifier.classList.remove('pv-magnifier-container_focus');
  5767. },
  5768. move:function(e){
  5769. var mouseCoor={
  5770. x:e.pageX,
  5771. y:e.pageY,
  5772. };
  5773. var mouseRange=this.mouseRange;
  5774. var imgRange=this.imgRange;
  5775.  
  5776. if( !(mouseCoor.x >= mouseRange.x[0] && mouseCoor.x <= mouseRange.x[1] && mouseCoor.y >= mouseRange.y[0] && mouseCoor.y <= mouseRange.y[1]))return;//如果不再鼠标范围
  5777. if(mouseCoor.x > imgRange.x[1]){
  5778. mouseCoor.x = imgRange.x[1];
  5779. }else if(mouseCoor.x < imgRange.x[0]){
  5780. mouseCoor.x = imgRange.x[0];
  5781. };
  5782. if(mouseCoor.y > imgRange.y[1]){
  5783. mouseCoor.y = imgRange.y[1];
  5784. }else if(mouseCoor.y < imgRange.y[0]){
  5785. mouseCoor.y = imgRange.y[0];
  5786. };
  5787.  
  5788. var ms=this.magnifier.style;
  5789. ms.top= mouseCoor.y + 'px';
  5790. ms.left= mouseCoor.x + 'px';
  5791.  
  5792. var radius=this.radius;
  5793. var imgSize=this.imgSize;
  5794. var imgNaturalSize=this.imgNaturalSize;
  5795. var px=-((mouseCoor.x-imgRange.x[0])/imgSize.w * imgNaturalSize.w) + radius +'px';
  5796. var py=-((mouseCoor.y-imgRange.y[0])/imgSize.h * imgNaturalSize.h) + radius +'px';
  5797. ms.backgroundPosition=px + ' ' + py;
  5798. },
  5799. getNextZoomLevel:function(){
  5800. var level;
  5801. var self=this;
  5802. if(this.zoomOut){//缩小
  5803. MagnifierC.zoomRangeR._find(function(value){
  5804. if(value < self.zoomLevel){
  5805. level=value;
  5806. return true;
  5807. }
  5808. })
  5809. }else{
  5810. MagnifierC.zoomRange._find(function(value){
  5811. if(value > self.zoomLevel){
  5812. level=value;
  5813. return true;
  5814. };
  5815. });
  5816. }
  5817. return level;
  5818. },
  5819. zoom:function(e){
  5820. if(e.deltaY===0)return;//非Y轴的滚动
  5821. if(prefs.magnifier.wheelZoom.pauseFirst && !this.paused)return;
  5822. e.preventDefault();
  5823. if(e.deltaY < 0){//向上滚,放大;
  5824. if(this.diameter >= this.maxDia)return;
  5825. this.zoomOut=false;
  5826. }else{
  5827. this.zoomOut=true;
  5828. };
  5829. var level=this.getNextZoomLevel();
  5830. if(!level)return;
  5831.  
  5832. this.zoomLevel=level;
  5833. var diameter=this.defaultDia * level;
  5834. if(diameter > this.maxDia){
  5835. diameter = this.maxDia;
  5836. };
  5837.  
  5838. var radius=diameter/2
  5839. this.diameter=diameter;
  5840. var bRadius=this.radius;
  5841. this.radius=radius;
  5842. this.setMouseRange();
  5843. var ms=this.magnifier.style;
  5844. ms.width=diameter+'px';
  5845. ms.height=diameter+'px';
  5846. ms.borderRadius=radius+1 + 'px';
  5847. ms.marginLeft=-radius+'px';
  5848. ms.marginTop=-radius+'px';
  5849. var bBP=ms.backgroundPosition.split(' ');
  5850. ms.backgroundPosition=parseFloat(bBP[0]) + (radius - bRadius) + 'px' + ' ' + (parseFloat(bBP[1]) + ( radius - bRadius) + 'px');
  5851.  
  5852. },
  5853. pause:function(){
  5854. if(this.paused){
  5855. this.magnifier.classList.remove('pv-magnifier-container_pause');
  5856. document.addEventListener('mousemove',this._move,true);
  5857. }else{
  5858. this.magnifier.classList.add('pv-magnifier-container_pause');
  5859. document.removeEventListener('mousemove',this._move,true);
  5860. };
  5861. this.paused=!this.paused;
  5862. },
  5863. setMouseRange:function(){
  5864. var imgRange=this.imgRange;
  5865. var radius=this.radius;
  5866. this.mouseRange={//鼠标活动范围
  5867. x:[imgRange.x[0]-radius , imgRange.x[1] + radius],
  5868. y:[imgRange.y[0]-radius , imgRange.y[1] + radius],
  5869. };
  5870. },
  5871. remove:function(){
  5872. this.magnifier.parentNode.removeChild(this.magnifier);
  5873. document.removeEventListener('mousemove',this._move,true);
  5874. MagnifierC.all.splice(MagnifierC.all.indexOf(this),1);
  5875. },
  5876. };
  5877.  
  5878. //图片窗口
  5879. function ImgWindowC(img, data){
  5880. this.loaded=false;
  5881. this.img=img;
  5882. this.src=data?data.src:img.src;
  5883. this.data = data;
  5884.  
  5885. this.init();
  5886. if(data){
  5887. this.img.src = location.protocol == "https"?data.src.replace(/^http:/,"https:"):data.src;
  5888. }
  5889. };
  5890.  
  5891. ImgWindowC.all=[];//所有的窗口对象
  5892. ImgWindowC.styleZIndex=2147483647;//全局z-index;
  5893. ImgWindowC.overlayer=null;
  5894.  
  5895.  
  5896. ImgWindowC.prototype={
  5897. init:function(){
  5898. ImgWindowC.zoomRange=prefs.imgWindow.zoom.range.slice(0).sort((a, b)=>{return a - b});
  5899. ImgWindowC.zoomRangeR=ImgWindowC.zoomRange.slice(0).reverse();//降序
  5900. var self=this;
  5901. if(uniqueImgWin && !uniqueImgWin.removed){
  5902. uniqueImgWin.remove();
  5903. }
  5904. //图片是否已经被打开
  5905. if(ImgWindowC.all._find(function(iwin){
  5906. if(iwin.src==self.src){
  5907. iwin.firstOpen();
  5908. return true;
  5909. };
  5910. }))return;
  5911.  
  5912. this.addStyle();
  5913.  
  5914. var img=this.img;
  5915. img.className='pv-pic-window-pic pv-pic-ignored';
  5916. img.style.cssText='\
  5917. top:0px;\
  5918. left:0px;\
  5919. ';
  5920.  
  5921. var imgNaturalSize={
  5922. h:img.naturalHeight,
  5923. w:img.naturalWidth,
  5924. };
  5925. this.imgNaturalSize=imgNaturalSize;
  5926.  
  5927. var container=document.createElement('span');
  5928. container.style.cssText='\
  5929. cursor:pointer;\
  5930. top:0px;\
  5931. left:0px;\
  5932. opacity:0\
  5933. ';
  5934. container.className='pv-pic-window-container';
  5935. container.innerHTML=createHTML(
  5936. '<span class="pv-pic-window-rotate-indicator">'+
  5937. '<span class="pv-pic-window-rotate-indicator-pointer"></span>'+
  5938. '</span>'+
  5939. '<span class="pv-pic-window-rotate-overlayer"></span>'+
  5940. '<span class="pv-pic-window-toolbar" unselectable="on">'+
  5941. '<span class="pv-pic-window-tb-hand pv-pic-window-tb-tool" title="'+i18n("hand")+'"></span>'+
  5942. '<span class="pv-pic-window-tb-tool-badge-container pv-pic-window-tb-tool-extend-menu-container">'+
  5943. '<span class="pv-pic-window-tb-rotate pv-pic-window-tb-tool" title="'+i18n("rotate")+'"></span>'+
  5944. '<span class="pv-pic-window-tb-tool-badge">0</span>'+
  5945. '<span class="pv-pic-window-tb-tool-extend-menu pv-pic-window-tb-tool-extend-menu-rotate">'+
  5946. '<span class="pv-pic-window-tb-tool-extend-menu-item" title="+90"><svg class="icon" style="width: 1em;height: 1em;vertical-align: middle;fill: currentColor;overflow: hidden;" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="4332"><path d="M435.2 362.666667l179.2-179.2L435.2 0 375.466667 59.733333l76.8 76.8H341.333333c-119.466667 0-213.333333 93.866667-213.333333 213.333334v170.666666h85.333333v-170.666666c0-72.533333 55.466667-128 128-128h115.2L375.466667 302.933333l59.733333 59.733334zM853.333333 384H426.666667c-25.6 0-42.666667 17.066667-42.666667 42.666667v426.666666c0 25.6 17.066667 42.666667 42.666667 42.666667h426.666666c25.6 0 42.666667-17.066667 42.666667-42.666667V426.666667c0-25.6-17.066667-42.666667-42.666667-42.666667z m-42.666666 426.666667h-341.333334v-341.333334h341.333334v341.333334z" p-id="4333"></path></svg></span>'+
  5947. '<span class="pv-pic-window-tb-tool-extend-menu-item" title="-90"><svg class="icon" style="width: 1em;height: 1em;vertical-align: middle;fill: currentColor;overflow: hidden;" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="4595"><path d="M682.666667 136.533333h-115.2l76.8-76.8L588.8 0 405.333333 179.2l179.2 179.2L644.266667 298.666667l-76.8-76.8H682.666667c72.533333 0 128 55.466667 128 128v170.666666h85.333333v-170.666666c0-115.2-93.866667-213.333333-213.333333-213.333334zM597.333333 384H170.666667c-25.6 0-42.666667 17.066667-42.666667 42.666667v426.666666c0 25.6 17.066667 42.666667 42.666667 42.666667h426.666666c25.6 0 42.666667-17.066667 42.666667-42.666667V426.666667c0-25.6-17.066667-42.666667-42.666667-42.666667z m-42.666666 426.666667H213.333333v-341.333334h341.333334v341.333334z" p-id="4596"></path></svg></span>'+
  5948. '<span class="pv-pic-window-tb-tool-extend-menu-item" title="0"><svg class="icon" style="width: 1em;height: 1em;vertical-align: middle;fill: currentColor;overflow: hidden;" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="3716"><path d="M867.89 574.16a30.73 30.73 0 0 0-37.52 21.92c-38 144.83-169.31 246-319.25 246a330.71 330.71 0 0 1-306.27-206.82h60.29l-92.78-123.5-91.82 123.5h58.88q1.23 3.72 2.53 7.4A391.65 391.65 0 0 0 511.12 903.5c177.86 0 333.58-120 378.69-291.82a30.73 30.73 0 0 0-21.92-37.52zM153.88 452.57a30.69 30.69 0 0 0 37.35-22.2A329.68 329.68 0 0 1 511.12 182c136.8 0 256.58 82 306.4 207h-60.66l92.78 123.5L941.46 389h-58.58a391.63 391.63 0 0 0-751.2 26.24 30.73 30.73 0 0 0 22.2 37.33z" p-id="3717"></path></svg></span>'+
  5949. '</span>'+
  5950. '</span>'+
  5951. '<span class="pv-pic-window-tb-tool-badge-container pv-pic-window-tb-tool-extend-menu-container">'+
  5952. '<span class="pv-pic-window-tb-zoom pv-pic-window-tb-tool" title="'+i18n("scale")+'"></span>'+
  5953. '<span class="pv-pic-window-tb-tool-badge">0</span>'+
  5954. '<span class="pv-pic-window-tb-tool-extend-menu pv-pic-window-tb-tool-extend-menu-zoom">'+
  5955. '<span id="pv-pic-zoom-in" class="pv-pic-window-tb-tool-extend-menu-item" title="+0.1"><svg class="icon" style="width: 1em;height: 1em;vertical-align: middle;fill: currentColor;overflow: hidden;" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="3134"><path d="M754.2 151.5h-89.5v42.7h89.5c59.5 0 108 48.4 108 108v67.6h42.7v-67.6c0-83.1-67.6-150.7-150.7-150.7zM862.2 737.3c0 59.5-48.4 108-108 108h-89.5V888h89.5c83.1 0 150.7-67.6 150.7-150.7v-67.6h-42.7v67.6zM166.3 737.8v-67.6h-42.7v67.6c0 83.1 67.6 150.7 150.7 150.7h89.5v-42.7h-89.5c-59.5 0-108-48.4-108-108zM416.3 261.8h-42.8v126H247.6v42.7h125.9V556h42.8V430.5h125.4v-42.7H416.3zM773.6 789.4l30.2-30.2-190.1-190.6c32.7-44.8 52-99.9 52-159.5 0-149.7-121.6-271.3-271.3-271.3-149.3 0-271.3 121.6-271.3 271.3 0 149.3 121.5 271.3 271.3 271.3 74.5 0 142.3-30.3 191.4-79.3l187.8 188.3zM394.4 637.7c-126 0-228.6-102.5-228.6-228.6s102.5-228.6 228.6-228.6S623 283.1 623 409.2 520.5 637.7 394.4 637.7z" p-id="3135"></path></svg></span>'+
  5956. '<span id="pv-pic-zoom-out" class="pv-pic-window-tb-tool-extend-menu-item" title="-0.1"><svg class="icon" style="width: 1em;height: 1em;vertical-align: middle;fill: currentColor;overflow: hidden;" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="2674"><path d="M754.2 151.5h-89.5v42.7h89.5c59.5 0 108 48.4 108 108v67.6h42.7v-67.6c0-83.1-67.6-150.7-150.7-150.7zM862.2 737.3c0 59.5-48.4 108-108 108h-89.5V888h89.5c83.1 0 150.7-67.6 150.7-150.7v-67.6h-42.7v67.6zM166.3 737.8v-67.6h-42.7v67.6c0 83.1 67.6 150.7 150.7 150.7h89.5v-42.7h-89.5c-59.5 0-108-48.4-108-108zM247.6 387.8h294.2v42.7H247.6zM773.6 789.4l30.2-30.2-190.1-190.6c32.7-44.8 52-99.9 52-159.5 0-149.7-121.6-271.3-271.3-271.3-149.3 0-271.3 121.6-271.3 271.3 0 149.3 121.6 271.3 271.3 271.3 74.5 0 142.3-30.3 191.4-79.3l187.8 188.3zM394.4 637.7c-126 0-228.6-102.5-228.6-228.6s102.5-228.6 228.6-228.6S623 283.1 623 409.2 520.5 637.7 394.4 637.7z" p-id="2675"></path></svg></span>'+
  5957. '<span class="pv-pic-window-tb-tool-extend-menu-item" title="1"><svg class="icon" style="width: 1em;height: 1em;vertical-align: middle;fill: currentColor;overflow: hidden;" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="3716"><path d="M867.89 574.16a30.73 30.73 0 0 0-37.52 21.92c-38 144.83-169.31 246-319.25 246a330.71 330.71 0 0 1-306.27-206.82h60.29l-92.78-123.5-91.82 123.5h58.88q1.23 3.72 2.53 7.4A391.65 391.65 0 0 0 511.12 903.5c177.86 0 333.58-120 378.69-291.82a30.73 30.73 0 0 0-21.92-37.52zM153.88 452.57a30.69 30.69 0 0 0 37.35-22.2A329.68 329.68 0 0 1 511.12 182c136.8 0 256.58 82 306.4 207h-60.66l92.78 123.5L941.46 389h-58.58a391.63 391.63 0 0 0-751.2 26.24 30.73 30.73 0 0 0 22.2 37.33z" p-id="3717"></path></svg></span>'+
  5958. '</span>'+
  5959. '</span>'+
  5960. '<span class="pv-pic-window-tb-flip-horizontal pv-pic-window-tb-command" title="'+i18n("horizontalFlip")+'"></span>'+
  5961. '<span class="pv-pic-window-tb-flip-vertical pv-pic-window-tb-command" title="'+i18n("verticalFlip")+'"></span>'+
  5962. '</span>'+
  5963. '<span class="pv-pic-window-close"></span>' +
  5964. //'<span class="pv-pic-window-search" title="'+i18n("similarImage")+'"></span>' +
  5965. '<span class="pv-pic-window-range"></span>' +
  5966. '<span class="pv-pic-window-description"></span>'+
  5967. '<span class="pv-pic-search-state"></span>');
  5968.  
  5969. container.insertBefore(img,container.firstChild);
  5970.  
  5971. this.imgWindow=container;
  5972.  
  5973. var toolMap={
  5974. 'hand':container.querySelector('.pv-pic-window-tb-hand'),
  5975. 'rotate':container.querySelector('.pv-pic-window-tb-rotate'),
  5976. 'zoom':container.querySelector('.pv-pic-window-tb-zoom'),
  5977. 'fh':container.querySelector('.pv-pic-window-tb-flip-horizontal'),
  5978. 'fv':container.querySelector('.pv-pic-window-tb-flip-vertical'),
  5979. };
  5980. this.toolMap=toolMap;
  5981.  
  5982.  
  5983. //关闭
  5984. var closeButton=container.querySelector('.pv-pic-window-close');
  5985. closeButton.style.cssText='top: -24px;right: 0px;';
  5986. this.closeButton=closeButton;
  5987. closeButton.addEventListener('click',function(e){
  5988. self.remove();
  5989. },false);
  5990.  
  5991. //var searchButton=container.querySelector('.pv-pic-window-search');
  5992. //searchButton.style.cssText='top: -24px;right: 50px;';
  5993. //this.searchButton=searchButton;
  5994. var srcs, from;
  5995. img.onerror=function(e){
  5996. //setSearchState(i18n("loadNextSimilar"),img.parentNode);
  5997. console.info(img.src+" "+i18n("loadError"));
  5998. if(self.removed || !self.data)return;
  5999. var src;
  6000. if(self.data.srcs)
  6001. src=self.data.srcs.shift();
  6002. if(src)img.src=src;
  6003. else{
  6004. if(img.src!=self.data.imgSrc)
  6005. img.src=self.data.imgSrc;
  6006. return;
  6007. if(from<searchSort.length){
  6008. from++;
  6009. searchImgByImg(self.img.src, self.img.parentNode, function(srcs, index){
  6010. from=index;
  6011. self.data.srcs=srcs;
  6012. self.img.src=srcs.shift();
  6013. },null,null,from);
  6014. }else{
  6015. setSearchState(i18n("findNoPic"),img.parentNode);
  6016. setTimeout(function(){
  6017. setSearchState("",img.parentNode);
  6018. },2000);
  6019. }
  6020. }
  6021. };
  6022. img.onload=function(e){
  6023. if(self.removed)return;
  6024. self.loaded=true;
  6025. if(img.naturalHeight ==1 && img.naturalWidth ==1){
  6026. self.remove();
  6027. return;
  6028. }
  6029. //self.imgWindow.classList.remove("pv-pic-window-transition-all");
  6030. self.imgWindow.style.display="";
  6031. setSearchState("",img.parentNode);
  6032. self.imgNaturalSize={
  6033. h:img.naturalHeight,
  6034. w:img.naturalWidth,
  6035. };
  6036. if(self.imgWindow.style.overflow!="scroll"){
  6037. self.zoomLevel=0;
  6038. self.zoom(1);
  6039. }
  6040. if(prefs.imgWindow.fitToScreen)
  6041. self.fitToScreen();
  6042. self.center(true,true);
  6043. self.imgWindow.style.opacity=1;
  6044. self.keepScreenInside();
  6045.  
  6046. var wSize=getWindowSize();
  6047. wSize.h -= 16;
  6048. wSize.w -= 16;
  6049. self.isLongImg=self.imgNaturalSize.h >= wSize.h && self.imgNaturalSize.h/self.imgNaturalSize.w > 3.5;
  6050. }
  6051. /*searchButton.addEventListener('click',function(e){
  6052. sortSearch();
  6053. searchImgByImg(self.img.src, self.img.parentNode, function(srcs, index){
  6054. from=index;
  6055. self.srcs=srcs;
  6056. self.img.src=srcs.shift();
  6057. });
  6058. },false);*/
  6059.  
  6060. /**
  6061. * 说明
  6062. * 1、对原来的适应屏幕等功能会有影响,暂时禁用。
  6063. * 2、分为 absolute 和默认的2种情况
  6064. */
  6065. if (this.data) {
  6066. var descriptionSpan = container.querySelector('.pv-pic-window-description');
  6067. // descriptionSpan.style.cssText = '\
  6068. // bottom: -40px;\
  6069. // left: 10px;\
  6070. // ';
  6071. descriptionSpan.textContent = this.data.description || '';
  6072. // descriptionSpan.style.display = this.data.description ? 'block' : 'none';
  6073. descriptionSpan.style.display = 'none';
  6074. this.descriptionSpan = descriptionSpan;
  6075. }
  6076.  
  6077. var toolbar=container.querySelector('.pv-pic-window-toolbar');
  6078. toolbar.style.cssText='\
  6079. top: 0px;\
  6080. left: -45px;\
  6081. ';
  6082. this.toolbar=toolbar;
  6083.  
  6084. this.selectedToolClass='pv-pic-window-tb-tool-selected';
  6085.  
  6086. this.viewRange=container.querySelector('.pv-pic-window-range');
  6087.  
  6088. this.rotateIndicator=container.querySelector('.pv-pic-window-rotate-indicator');
  6089. this.rotateIPointer=container.querySelector('.pv-pic-window-rotate-indicator-pointer');
  6090. this.rotateOverlayer=container.querySelector('.pv-pic-window-rotate-overlayer');
  6091.  
  6092.  
  6093. this.hKeyUp=true;
  6094. this.rKeyUp=true;
  6095. this.zKeyUp=true;
  6096.  
  6097. this.spaceKeyUp=true;
  6098. this.ctrlKeyUp=true;
  6099. this.altKeyUp=true;
  6100. this.shiftKeyUp=true;
  6101. this.moving=false;
  6102.  
  6103. //缩放工具的扩展菜单
  6104. container.querySelector('.pv-pic-window-tb-tool-extend-menu-zoom').addEventListener('click',function(e){
  6105. var target=e.target;
  6106. var text=target.title;
  6107. var value;
  6108. switch(text){
  6109. case '1':{
  6110. value=1;
  6111. }break;
  6112. case '+0.1':{
  6113. value=self.zoomLevel + 0.1;
  6114. }break;
  6115. case '-0.1':{
  6116. value=self.zoomLevel - 0.1;
  6117. }break;
  6118. };
  6119. if(typeof value!='undefined'){
  6120. self.zoom(value,{x:0,y:0});
  6121. };
  6122. },true);
  6123.  
  6124. //旋转工具的扩展菜单
  6125. container.querySelector('.pv-pic-window-tb-tool-extend-menu-rotate').addEventListener('click',function(e){
  6126. var target=e.target;
  6127. var text=target.title;
  6128. var value;
  6129. function convert(deg){
  6130. return deg * Math.PI/180;
  6131. };
  6132.  
  6133. switch(text){
  6134. case '0':{
  6135. value=0;
  6136. }break;
  6137. case '+90':{
  6138. value=self.rotatedRadians + convert(90);
  6139. }break;
  6140. case '-90':{
  6141. value=self.rotatedRadians - convert(90);
  6142. }break;
  6143. };
  6144.  
  6145. var PI=Math.PI;
  6146. if(typeof value!='undefined'){
  6147. if(value>=2*PI){
  6148. value-=2*PI;
  6149. }else if(value<0){
  6150. value+=2*PI;
  6151. };
  6152. self.rotate(value,true);
  6153. };
  6154. },true);
  6155.  
  6156. toolbar.addEventListener('mousedown',function(e){//鼠标按下选择工具
  6157. self.toolbarEventHandler(e);
  6158. },false);
  6159.  
  6160.  
  6161. toolbar.addEventListener('dblclick',function(e){//鼠标双击工具
  6162. self.toolbarEventHandler(e);
  6163. },false);
  6164.  
  6165.  
  6166. //阻止浏览器对图片的默认控制行为
  6167. img.addEventListener('mousedown',function(e){
  6168. e.preventDefault();
  6169. },false);
  6170.  
  6171.  
  6172. container.addEventListener('mousedown',function(e){//当按下的时,执行平移,缩放,旋转操作
  6173. self.imgWindowEventHandler(e);
  6174. },false);
  6175.  
  6176. container.addEventListener('touchstart',function(e){//当按下的时,执行平移,缩放,旋转操作
  6177. self.imgWindowEventHandler(e);
  6178. },false);
  6179.  
  6180. container.addEventListener('click',function(e){//阻止opera ctrl+点击保存图片
  6181. self.imgWindowEventHandler(e);
  6182. },false);
  6183.  
  6184. if(prefs.imgWindow.zoom.mouseWheelZoom){//是否使用鼠标缩放
  6185. addWheelEvent(container,function(e){//滚轮缩放
  6186. self.imgWindowEventHandler(e);
  6187. },false);
  6188. };
  6189.  
  6190.  
  6191.  
  6192. //是否点击图片外部关闭
  6193. if(prefs.imgWindow.overlayer.shown && prefs.imgWindow.close.clickOutside){
  6194. var clickOutside=function(e){
  6195. var target=e.target;
  6196. if(!container.contains(target)){
  6197. self.remove();
  6198. };
  6199. };
  6200. this.clickOutside=clickOutside;
  6201. document.addEventListener(prefs.imgWindow.close.clickOutside,clickOutside,true);
  6202. };
  6203.  
  6204. //是否双击图片本身关闭
  6205. if(prefs.imgWindow.close.dblClickImgWindow){
  6206. var dblClickImgWindow=function(e){
  6207. var target=e.target;
  6208. if(target==container || target==img || target==self.rotateOverlayer){
  6209. self.remove();
  6210. };
  6211. };
  6212. container.addEventListener('dblclick',dblClickImgWindow,true);
  6213. };
  6214.  
  6215.  
  6216. ImgWindowC.all.push(this);
  6217.  
  6218. this._blur=this.blur.bind(this);
  6219. this._focusedKeydown=this.focusedKeydown.bind(this);
  6220. this._focusedKeyup=this.focusedKeyup.bind(this);
  6221.  
  6222. if(prefs.imgWindow.overlayer.shown){//是否显示覆盖层
  6223. var overlayer=ImgWindowC.overlayer;
  6224. if(!overlayer){
  6225. overlayer=document.createElement('span');
  6226. ImgWindowC.overlayer=overlayer;
  6227. overlayer.className='pv-pic-window-overlayer';
  6228. overlayer.style.backgroundColor=prefs.imgWindow.overlayer.color;
  6229. };
  6230. document.body.appendChild(overlayer);
  6231. overlayer.style.display='block';
  6232. };
  6233. document.body.appendChild(container);
  6234.  
  6235. this.rotatedRadians=0;//已经旋转的角度
  6236. this.zoomLevel=1;//缩放级别
  6237. this.setToolBadge('zoom',1);
  6238.  
  6239. //选中默认工具
  6240. this.selectTool(prefs.imgWindow.defaultTool);
  6241.  
  6242. this.firstOpen();
  6243. this.imgWindow.style.opacity=1;
  6244. },
  6245. changeData:function(result){
  6246. if(this.src != result.src){
  6247. //this.imgWindow.classList.add("pv-pic-window-transition-all");
  6248. this.loaded = false;
  6249. this.data = result;
  6250. this.src = result.src;
  6251. this.img.src = location.protocol == "https"?result.src.replace(/^https?:/,""):result.src;
  6252. }
  6253. },
  6254. addStyle:function(){
  6255. if(ImgWindowC.style)return;
  6256. var style=document.createElement('style');
  6257. ImgWindowC.style=style;
  6258. style.textContent='\
  6259. .pv-pic-window-container {\
  6260. position: absolute;\
  6261. background-color: rgba(40,40,40,0.8);\
  6262. padding: 8px;\
  6263. border: 5px solid rgb(255 255 255 / 60%);\
  6264. border-radius: 1px;\
  6265. line-height: 0;\
  6266. text-align: left;\
  6267. box-sizing: content-box;\
  6268. -webkit-transition: opacity 0.1s ease-out;\
  6269. transition: opacity 0.1s ease-out;\
  6270. overscroll-behavior: none;\
  6271. }\
  6272. .pv-pic-window-transition-all{\
  6273. -webkit-transition: all 0.3s ease-out;\
  6274. transition: all 0.3s ease-out;\
  6275. }\
  6276. .pv-pic-window-container_focus {\
  6277. box-shadow: 0 0 10px 5px rgba(0,0,0,0.6);\
  6278. box-sizing: content-box;\
  6279. }\
  6280. .pv-pic-window-close,\
  6281. .pv-pic-window-search,\
  6282. .pv-pic-window-toolbar,\
  6283. .pv-pic-window-tb-tool-extend-menu{\
  6284. -webkit-transition: opacity 0.2s ease-in-out;\
  6285. transition: opacity 0.2s ease-in-out;\
  6286. }\
  6287. .pv-pic-window-toolbar {\
  6288. position: absolute;\
  6289. background-color: #535353;\
  6290. padding: 0;\
  6291. opacity: 0.9;\
  6292. display: none;\
  6293. cursor: default;\
  6294. -o-user-select: none;\
  6295. -webkit-user-select: none;\
  6296. -moz-user-select: -moz-none;\
  6297. user-select: none;\
  6298. }\
  6299. .pv-pic-window-toolbar:hover {\
  6300. opacity: 1;\
  6301. }\
  6302. .pv-pic-window-toolbar_focus {\
  6303. display: block;\
  6304. }\
  6305. .pv-pic-window-close {\
  6306. cursor: pointer;\
  6307. position: absolute;\
  6308. right: 0px;\
  6309. top: -24px;\
  6310. background: url("'+prefs.icons.close+'") no-repeat center bottom;\
  6311. height: 17px;\
  6312. width: 46px;\
  6313. opacity: 0.9;\
  6314. border:none;\
  6315. padding:0;\
  6316. padding-top:2px;\
  6317. background-color:#1771FF;\
  6318. display: none;\
  6319. }\
  6320. .pv-pic-window-close:hover {\
  6321. background-color:red;\
  6322. opacity: 1;\
  6323. }\
  6324. .pv-pic-window-close_focus {\
  6325. display: block;\
  6326. }\
  6327. .pv-pic-window-search {\
  6328. cursor: pointer;\
  6329. position: absolute;\
  6330. right: 50px;\
  6331. top: -24px;\
  6332. background: url("'+prefs.icons.searchBtn+'") no-repeat center bottom;\
  6333. height: 17px;\
  6334. width: 46px;\
  6335. opacity: 0.9;\
  6336. border:none;\
  6337. padding:0;\
  6338. padding-top:2px;\
  6339. background-color:#1771FF;\
  6340. display: none;\
  6341. }\
  6342. .pv-pic-window-search:hover {\
  6343. background-color:red;\
  6344. opacity: 1;\
  6345. }\
  6346. .pv-pic-window-search_focus {\
  6347. display: block;\
  6348. }\
  6349. .pv-pic-window-description {\
  6350. margin-top: 20px;\
  6351. min-height: 20px;\
  6352. }\
  6353. .pv-pic-search-state {\
  6354. top: 10px;\
  6355. left: 10px;\
  6356. display: block;\
  6357. position: absolute;\
  6358. z-index: 1;\
  6359. color: #ffff00;\
  6360. width: 500px;\
  6361. font-size: large;\
  6362. text-shadow: 1px 0 0 #000,-1px 0 0 #000,0 1px 0 #000,0 -1px 0 #000;\
  6363. -webkit-text-shadow:#000 1px 0 0,#000 0 1px 0,#000 -1px 0 0,#000 0 -1px 0;\
  6364. -moz-text-shadow:#000 1px 0 0,#000 0 1px 0,#000 -1px 0 0,#000 0 -1px 0;\
  6365. }\
  6366. .pv-pic-window-pic {\
  6367. position: relative;\
  6368. display:inline-block;\/*opera把图片设置display:block会出现渲染问题,会有残影,还会引发其他各种问题,吓尿*/\
  6369. max-width:none;\
  6370. min-width:none;\
  6371. max-height:none;\
  6372. min-height:none;\
  6373. padding:0;\
  6374. margin:0;\
  6375. border:none;\
  6376. vertical-align:middle;\
  6377. }\
  6378. .pv-pic-window-pic_focus {\
  6379. box-shadow: 0 0 6px black;\
  6380. }\
  6381. .pv-pic-window-tb-tool,\
  6382. .pv-pic-window-tb-command{\
  6383. box-sizing:content-box;\
  6384. -moz-box-sizing:content-box;\
  6385. -webkit-box-sizing:content-box;\
  6386. height: 24px;\
  6387. width: 24px;\
  6388. padding: 12px 8px 6px 6px;\
  6389. margin:0;\
  6390. display: block;\
  6391. background: transparent no-repeat center;\
  6392. cursor: pointer;\
  6393. position: relative;\
  6394. border: none;\
  6395. border-left: 2px solid transparent;\
  6396. border-bottom: 1px solid #868686;\
  6397. background-origin: content-box;\
  6398. }\
  6399. .pv-pic-window-toolbar > span:last-child {\
  6400. border-bottom: none;\
  6401. }\
  6402. .pv-pic-window-tb-tool:hover,\
  6403. .pv-pic-window-tb-command:hover{\
  6404. border-left: 2px solid red;\
  6405. }\
  6406. .pv-pic-window-tb-tool-selected{\
  6407. box-shadow: inset 0 21px 0 rgba(255,255,255,0.3) ,inset 0 -21px 0 rgba(0,0,0,0.3);\
  6408. border-left:2px solid #1771FF;\
  6409. }\
  6410. .pv-pic-window-tb-hand {\
  6411. background-image: url("'+prefs.icons.hand+'");\
  6412. }\
  6413. .pv-pic-window-tb-rotate {\
  6414. background-image: url("'+prefs.icons.rotate+'");\
  6415. }\
  6416. .pv-pic-window-tb-zoom {\
  6417. background-image: url("'+prefs.icons.zoom+'");\
  6418. }\
  6419. .pv-pic-window-tb-flip-horizontal {\
  6420. background-image: url("'+prefs.icons.flipHorizontal+'");\
  6421. }\
  6422. .pv-pic-window-tb-flip-vertical {\
  6423. background-image: url("'+prefs.icons.flipVertical+'");\
  6424. }\
  6425. .pv-pic-window-tb-tool-badge-container {\
  6426. display: block;\
  6427. position: relative;\
  6428. }\
  6429. .pv-pic-window-tb-tool-badge {\
  6430. position: absolute;\
  6431. top: -3px;\
  6432. right: 1px;\
  6433. font-size: 10px;\
  6434. line-height: 1.5;\
  6435. padding: 0 3px;\
  6436. background-color: #F93;\
  6437. border-radius: 50px;\
  6438. opacity: 0.5;\
  6439. color: black;\
  6440. }\
  6441. .pv-pic-window-tb-tool-extend-menu{\
  6442. position:absolute;\
  6443. top:0;\
  6444. margin-left:-1px;\
  6445. background-color:#535353;\
  6446. display:none;\
  6447. left:40px;\
  6448. color:#C3C3C3;\
  6449. font-size:12px;\
  6450. text-shadow:0px -1px 0px black;\
  6451. opacity:0.7;\
  6452. }\
  6453. .pv-pic-window-tb-tool-extend-menu:hover{\
  6454. opacity:0.9;\
  6455. }\
  6456. .pv-pic-window-tb-tool-extend-menu-item{\
  6457. display:block;\
  6458. line-height:1.5;\
  6459. text-align:center;\
  6460. padding:10px;\
  6461. cursor:pointer;\
  6462. border: none;\
  6463. border-right: 2px solid transparent;\
  6464. border-bottom: 1px solid #868686;\
  6465. font-size: 20px;\
  6466. }\
  6467. .pv-pic-window-tb-tool-extend-menu-item>svg{\
  6468. pointer-events: none;\
  6469. }\
  6470. .pv-pic-window-tb-tool-extend-menu-item:last-child{\
  6471. border-bottom: none;\
  6472. }\
  6473. .pv-pic-window-tb-tool-extend-menu-item:hover{\
  6474. border-right:2px solid red;\
  6475. }\
  6476. .pv-pic-window-tb-tool-extend-menu-item:active{\
  6477. padding:11px 9px 9px 11px;\
  6478. }\
  6479. .pv-pic-window-tb-tool-extend-menu-container:hover .pv-pic-window-tb-tool{\
  6480. border-left:2px solid red;\
  6481. }\
  6482. .pv-pic-window-tb-tool-extend-menu-container:hover .pv-pic-window-tb-tool-extend-menu{\
  6483. display:block;\
  6484. }\
  6485. .pv-pic-window-tb-tool-extend-menu-container::after{\
  6486. content:"";\
  6487. position:absolute;\
  6488. right:1px;\
  6489. bottom:2px;\
  6490. width:0;\
  6491. height:0;\
  6492. padding:0;\
  6493. margin:0;\
  6494. border:3px solid #C3C3C3;\
  6495. border-top-color:transparent;\
  6496. border-left-color:transparent;\
  6497. opacity:0.5;\
  6498. }\
  6499. .pv-pic-window-overlayer{\
  6500. height:100%;\
  6501. width:100%;\
  6502. position:fixed;\
  6503. top:0;\
  6504. left:0;\
  6505. z-index: 2147483646;\
  6506. }\
  6507. .pv-pic-window-rotate-indicator{\
  6508. display:none;\
  6509. position:fixed;\
  6510. width:250px;\
  6511. height:250px;\
  6512. padding:10px;\
  6513. margin-top:-135px;\
  6514. margin-left:-135px;\
  6515. background:transparent url("'+ prefs.icons.rotateIndicatorBG +'") no-repeat center;\
  6516. }\
  6517. .pv-pic-window-rotate-indicator-pointer{\
  6518. display:block;\
  6519. margin-left:auto;\
  6520. margin-right:auto;\
  6521. background:transparent url("'+ prefs.icons.rotateIndicatorPointer +'") no-repeat center;\
  6522. width:60px;\
  6523. height:240px;\
  6524. position:relative;\
  6525. top:5px;\
  6526. transform:rotate(0.1deg);\
  6527. }\
  6528. .pv-pic-window-rotate-overlayer{/*当切换到旋转工具的时候显示这个覆盖层,然后旋转指示器显示在这个覆盖层的下面*/\
  6529. position:absolute;\
  6530. top:0;\
  6531. bottom:0;\
  6532. left:0;\
  6533. right:0;\
  6534. display:none;\
  6535. background-color:transparent;\
  6536. }\
  6537. .pv-pic-window-range{\
  6538. position:absolute;\
  6539. border:none;\
  6540. width:100px;\
  6541. height:100px;\
  6542. box-shadow: 0px 0px 5px rgba(0, 0, 0, 0.8);\
  6543. display:none;\
  6544. padding:0;\
  6545. background-color:rgba(255, 0, 0, 0.150);\
  6546. }\
  6547. .pv-pic-window-container::-webkit-scrollbar { width: 0 !important }\
  6548. .pv-pic-window-container { -ms-overflow-style: none;overflow: -moz-scrollbars-none; }\
  6549. ';
  6550. document.head.appendChild(style);
  6551. },
  6552.  
  6553. firstOpen:function(){
  6554. ImgWindowC.selectedTool='hand';
  6555. this.focus();
  6556. var imgWindow=this.imgWindow;
  6557. var scrolled=getScrolled();
  6558. imgWindow.style.left=-5 + scrolled.x + 'px';
  6559. imgWindow.style.top=-5 + scrolled.y + 'px';
  6560.  
  6561. //window的尺寸
  6562. var wSize=getWindowSize();
  6563. //空隙
  6564. wSize.h -= 16;
  6565. wSize.w -= 16;
  6566.  
  6567. var imgWindowCS=unsafeWindow.getComputedStyle(imgWindow);
  6568. var rectSize={
  6569. h:parseFloat(imgWindowCS.height),
  6570. w:parseFloat(imgWindowCS.width),
  6571. };
  6572. this.isLongImg=rectSize.h > wSize.h && rectSize.h/rectSize.w > 3.5;
  6573. if(prefs.imgWindow.suitLongImg && this.isLongImg){
  6574. this.center(rectSize.w <= wSize.w,false);
  6575. this.imgWindow.style.overflow="scroll";
  6576. this.imgWindow.style.height="100%";
  6577. this.imgWindow.style.maxWidth="100%";
  6578. this.closeButton.style.display="none";
  6579. }else if(prefs.imgWindow.fitToScreen){
  6580. this.fitToScreen();
  6581. this.center(true,true);
  6582. }else{
  6583. this.center(rectSize.w <= wSize.w , rectSize.h <= wSize.h);
  6584. };
  6585.  
  6586. this.keepScreenInside();
  6587. },
  6588. keepScreenInside:function(){//保持按钮在屏幕里面.
  6589. var imgWindow=this.imgWindow;
  6590. var imgWindowFullSize={
  6591. h:imgWindow.offsetHeight,
  6592. w:imgWindow.offsetWidth,
  6593. };
  6594.  
  6595. var windowSize=getWindowSize();
  6596.  
  6597. function keepSI(obj,offsetDirection,defaultValue, out){
  6598. var objRect=obj.getBoundingClientRect();
  6599. var objStyle=obj.style;
  6600.  
  6601. while(offsetDirection.length){
  6602. var oD=offsetDirection[0];
  6603. var oDV=defaultValue[0];
  6604. offsetDirection.shift();
  6605. defaultValue.shift();
  6606. var oValue=parseFloat(objStyle[oD]);
  6607. var newValue;
  6608. switch(oD){
  6609. case 'top':{
  6610. newValue=oValue - objRect.top;
  6611. if(objRect.top<0){
  6612. newValue=Math.min(newValue,imgWindowFullSize.h);
  6613. }else{
  6614. newValue=Math.max(newValue,oDV);
  6615. };
  6616. }break;
  6617. case 'right':{
  6618. newValue=oValue + (objRect.right - windowSize.w);
  6619. if(objRect.right > windowSize.w){//屏幕外
  6620. newValue=Math.min(newValue,imgWindowFullSize.w);
  6621. }else{
  6622. newValue=Math.max(newValue,oDV);
  6623. };
  6624. }break;
  6625. case 'bottom':{
  6626. newValue=oValue + (objRect.bottom - windowSize.h);
  6627. if(objRect.bottom > windowSize.h){//屏幕外
  6628. newValue=Math.min(newValue,imgWindowFullSize.h);
  6629. }else{
  6630. newValue=Math.max(newValue,oDV);
  6631. };
  6632. }break;
  6633. case 'left':{
  6634. newValue=oValue - objRect.left;
  6635. if(objRect.left<0){
  6636. newValue=Math.min(newValue,imgWindowFullSize.w);
  6637. }else{
  6638. newValue=Math.max(newValue,oDV);
  6639. }
  6640. }break;
  6641. };
  6642. objStyle[oD]=newValue + 'px';
  6643.  
  6644. };
  6645. };
  6646.  
  6647. keepSI(this.closeButton,['top','right'],[-24,0]);
  6648. //keepSI(this.searchButton,['top','right'],[-24,50]);
  6649. keepSI(this.toolbar,['top','left'],[0,-45]);
  6650.  
  6651. // 保持注释在图片里面
  6652. // keepSI(this.descriptionSpan,['bottom', 'left'],[-40, 10]);
  6653. },
  6654. fitToScreen:function(){
  6655. var imgWindow=this.imgWindow;
  6656. if(!prefs.imgWindow.fitToScreen || imgWindow.style.overflow=="scroll")return;
  6657. var wSize=getWindowSize();
  6658. //空隙
  6659. wSize.h -= 16;
  6660. wSize.w -= 16;
  6661.  
  6662. var imgWindowCS=unsafeWindow.getComputedStyle(imgWindow);
  6663. var rectSize={
  6664. h:parseFloat(imgWindowCS.height),
  6665. w:parseFloat(imgWindowCS.width),
  6666. };
  6667.  
  6668. var size;
  6669. if(rectSize.w - wSize.w>0 || rectSize.h - wSize.h>0){//超出屏幕,那么缩小。
  6670. if(rectSize.w/rectSize.h > wSize.w/wSize.h){
  6671. size={
  6672. w:wSize.w,
  6673. h:wSize.w / (rectSize.w/rectSize.h),
  6674. };
  6675. }else{
  6676. size={
  6677. h:wSize.h,
  6678. w:wSize.h * (rectSize.w/rectSize.h),
  6679. }
  6680. };
  6681.  
  6682. this.zoom(this.getRotatedImgCliSize(size).w/this.imgNaturalSize.w);
  6683. };
  6684. },
  6685. center:function(horizontal,vertical){
  6686. if(!horizontal && !vertical)return;
  6687. var wSize=getWindowSize();
  6688. var imgWindow=this.imgWindow;
  6689. var scrolled=getScrolled();
  6690. if(horizontal)imgWindow.style.left= (wSize.w - imgWindow.offsetWidth)/2 + scrolled.x +'px';
  6691. if(vertical)imgWindow.style.top= (wSize.h - imgWindow.offsetHeight)/2 + scrolled.y +'px';
  6692. },
  6693.  
  6694.  
  6695. move:function(e){
  6696. this.working=true;
  6697. var cursor=this.cursor;
  6698. this.changeCursor('handing');
  6699.  
  6700. var mouseCoor={
  6701. x:e.pageX || e.touches[0].pageX,
  6702. y:e.pageY || e.touches[0].pageY,
  6703. };
  6704. var imgWindow=this.imgWindow;
  6705. var imgWStyle=imgWindow.style;
  6706. var oriOffset={
  6707. left:parseFloat(imgWStyle.left),
  6708. top:parseFloat(imgWStyle.top),
  6709. };
  6710. var self=this;
  6711. var moveHandler=function(e){
  6712. imgWStyle.left=oriOffset.left+ (e.pageX || e.touches[0].pageX)-mouseCoor.x +'px';
  6713. imgWStyle.top=oriOffset.top + (e.pageY || e.touches[0].pageY)-mouseCoor.y +'px';
  6714. self.keepScreenInside();
  6715. self.moving=true;
  6716. };
  6717. var mouseupHandler=function(){
  6718. e.preventDefault();
  6719. self.changeCursor(cursor);
  6720. self.working=false;
  6721. if(self.tempHand && self.spaceKeyUp){//如果是临时切换到抓手工具,平移完成后返回上个工具
  6722. self.tempHand=false;
  6723. self.changeCursor(self.selectedTool);
  6724. };
  6725. document.removeEventListener('mousemove',moveHandler,true);
  6726. document.removeEventListener('mouseup',mouseupHandler,true);
  6727. document.removeEventListener('touchmove',moveHandler,true);
  6728. document.removeEventListener('touchend',mouseupHandler,true);
  6729. };
  6730. document.addEventListener('mousemove',moveHandler,true);
  6731. document.addEventListener('mouseup',mouseupHandler,true);
  6732. document.addEventListener('touchmove',moveHandler,true);
  6733. document.addEventListener('touchend',mouseupHandler,true);
  6734. },
  6735. rotate:function(origin,topLeft){
  6736.  
  6737. var img=this.img;
  6738. var imgWindow=this.imgWindow;
  6739.  
  6740. var iTransform=img.style[support.cssTransform].replace(/rotate\([^)]*\)/i,'');
  6741.  
  6742. var imgWindowCS=unsafeWindow.getComputedStyle(imgWindow);
  6743. var imgRectSize={
  6744. h:parseFloat(imgWindowCS.height),
  6745. w:parseFloat(imgWindowCS.width),
  6746. };
  6747.  
  6748. var rectOffset={
  6749. top:parseFloat(imgWindow.style.top),
  6750. left:parseFloat(imgWindow.style.left),
  6751. };
  6752.  
  6753. var imgSize={
  6754. h:img.clientHeight,
  6755. w:img.clientWidth,
  6756. };
  6757.  
  6758. var imgOffset={
  6759. top:parseFloat(img.style.top),
  6760. left:parseFloat(img.style.left),
  6761. };
  6762.  
  6763. var self=this;
  6764. var PI=Math.PI;
  6765.  
  6766. var rotate=function (radians){
  6767. if(self.rotatedRadians==radians)return;
  6768. img.style[support.cssTransform] = ' rotate('+ radians +'rad) ' + iTransform;//旋转图片
  6769. self.rotateIPointer.style[support.cssTransform]='rotate('+ radians +'rad)';//旋转指示器
  6770.  
  6771. self.rotatedRadians=radians;
  6772. self.setToolBadge('rotate',radians/(PI/180));
  6773.  
  6774. var afterimgRectSize=self.getRotatedImgRectSize( radians, imgSize );
  6775. imgWindow.style.width=afterimgRectSize.w +'px';
  6776. imgWindow.style.height=afterimgRectSize.h + 'px';
  6777.  
  6778. if(!topLeft){
  6779. self.setImgWindowOffset(rectOffset,imgRectSize,afterimgRectSize);
  6780. };
  6781.  
  6782. self.setImgOffset(imgOffset,imgRectSize,afterimgRectSize);
  6783. self.keepScreenInside();
  6784. };
  6785.  
  6786.  
  6787. if(typeof origin=='number'){
  6788. rotate(origin);
  6789. return;
  6790. };
  6791.  
  6792.  
  6793. this.working=true;
  6794.  
  6795. var lastRotatedRadians=this.rotatedRadians;
  6796. this.shiftKeyUp=true;
  6797. var shiftRotateStep=prefs.imgWindow.shiftRotateStep / (180/Math.PI);//转成弧度
  6798.  
  6799. var moveHandler=function(e){
  6800. self.rotateIndicator.style.display='block';
  6801. var radians=lastRotatedRadians + Math.atan2( e.clientY - origin.y, e.clientX - origin.x );
  6802. if(radians>=2*PI){
  6803. radians-=2*PI;
  6804. }else if(radians<0){
  6805. radians+=2*PI;
  6806. };
  6807.  
  6808. if(!self.shiftKeyUp){//如果按下了shift键,那么步进缩放
  6809. radians -= radians % shiftRotateStep;
  6810. radians += shiftRotateStep;
  6811. };
  6812. rotate(radians);
  6813. };
  6814.  
  6815. var mouseupHandler=function(){
  6816. self.working=false;
  6817. self.rotateIndicator.style.display='none';
  6818. document.removeEventListener('mousemove',moveHandler,true);
  6819. document.removeEventListener('mouseup',mouseupHandler,true);
  6820. };
  6821.  
  6822. document.addEventListener('mousemove',moveHandler,true);
  6823. document.addEventListener('mouseup',mouseupHandler,true);
  6824. },
  6825. convertToValidRadians:function(radians){
  6826. //转成0-90的等价角度。
  6827. var PI=Math.PI;
  6828. if(radians > PI){
  6829. radians = 2*PI - radians;
  6830. };
  6831. if(radians > 1/2*PI){
  6832. radians = PI - radians;
  6833. };
  6834. return radians;
  6835. },
  6836. getRotatedImgRectSize:function( radians, imgSize ){//通过旋转后的角度和图片的大小,求虚拟矩形的大小
  6837. imgSize= imgSize ? imgSize :{
  6838. h:this.img.clientHeight,
  6839. w:this.img.clentWidth,
  6840. };
  6841.  
  6842. if(typeof radians==='undefined'){
  6843. radians = this.rotatedRadians;
  6844. };
  6845.  
  6846. radians=this.convertToValidRadians(radians);
  6847.  
  6848. return {
  6849. h:this.notExponential(imgSize.h* Math.cos(radians) + imgSize.w * Math.sin(radians)),
  6850. w:this.notExponential(imgSize.h* Math.sin(radians) + imgSize.w * Math.cos(radians)),
  6851. };
  6852. },
  6853. getRotatedImgCliSize:function(rectSize,radians){//通过虚拟矩形的大小和图片的旋转角度,求图片的大小
  6854.  
  6855. if(typeof radians==='undefined'){
  6856. radians = this.rotatedRadians;
  6857. };
  6858.  
  6859. radians=this.convertToValidRadians(radians);
  6860.  
  6861. if(radians==0){
  6862. //radians=Math.PI/180 * 1/100;
  6863. return rectSize;
  6864. };
  6865.  
  6866. var h=(rectSize.h-rectSize.w * Math.tan(radians))/(Math.cos(radians)-Math.sin(radians)*Math.tan(radians));
  6867. var w=(rectSize.h - h*Math.cos(radians))/Math.sin(radians);
  6868. return {
  6869. h:h,
  6870. w:w,
  6871. };
  6872.  
  6873. },
  6874. setImgOffset:function(oriOffset,bImgSize,aImgSize){
  6875. var imgStyle=this.img.style;
  6876.  
  6877. //避免出现指数形式的数字和单位相加,导致变成无效值
  6878. var top=this.notExponential(oriOffset.top + (aImgSize.h-bImgSize.h)*1/2) + 'px';
  6879. var left=this.notExponential(oriOffset.left + (aImgSize.w-bImgSize.w)*1/2) + 'px';
  6880. imgStyle.top= top;
  6881. imgStyle.left= left;
  6882. },
  6883. setImgWindowOffset:function(oriOffset,bImgWindowSize,aImgWidnowSize,ratio){
  6884. ratio= ratio? ratio : {x:1/2,y:1/2};
  6885. var imgWindowStyle=this.imgWindow.style;
  6886. var top=oriOffset.top - (aImgWidnowSize.h-bImgWindowSize.h)*ratio.y + 'px';
  6887. var left=oriOffset.left - (aImgWidnowSize.w-bImgWindowSize.w)*ratio.x + 'px';
  6888. imgWindowStyle.top= top;
  6889. imgWindowStyle.left= left;
  6890. },
  6891. zoom:function(e,ratio){//e可能是undefined,可能是事件对象,可能是直接的缩放级别数字
  6892. var imgWindow=this.imgWindow;
  6893. var imgWindowCS=unsafeWindow.getComputedStyle(imgWindow);
  6894. var imgRectSize={
  6895. h:parseFloat(imgWindowCS.height),
  6896. w:parseFloat(imgWindowCS.width),
  6897. };
  6898.  
  6899. var rectOffset={
  6900. top:parseFloat(imgWindow.style.top),
  6901. left:parseFloat(imgWindow.style.left),
  6902. };
  6903.  
  6904. var img=this.img;
  6905. var self=this;
  6906.  
  6907. var zoom=function(level){//缩放到指定级别
  6908. if(typeof level=='undefined' || level<0 || level==self.zoomLevel)return;
  6909.  
  6910. var afterImgSize={
  6911. h:self.imgNaturalSize.h * level,
  6912. w:self.imgNaturalSize.w * level,
  6913. };
  6914. img.width=afterImgSize.w;
  6915. img.height=afterImgSize.h;
  6916.  
  6917. var afterimgRectSize=self.getRotatedImgRectSize( self.rotatedRadians, afterImgSize );
  6918. imgWindow.style.width=afterimgRectSize.w +'px';
  6919. imgWindow.style.height=afterimgRectSize.h + 'px';
  6920. self.setImgWindowOffset(rectOffset,imgRectSize,afterimgRectSize,ratio);
  6921. self.setImgOffset({top:0,left:0},afterImgSize,afterimgRectSize);//如果旋转了,调整偏移
  6922. self.zoomLevel=level;
  6923. self.setToolBadge('zoom',level);
  6924. self.keepScreenInside();
  6925. };
  6926.  
  6927. if(typeof e!='object'){
  6928. ratio=ratio? ratio : {
  6929. x:1/2,
  6930. y:1/2,
  6931. };
  6932. zoom(e);
  6933. return;
  6934. };
  6935.  
  6936. this.working=true;
  6937.  
  6938. ratio=this.getZoomRatio({
  6939. x:e.clientX,
  6940. y:e.clientY,
  6941. });
  6942.  
  6943.  
  6944. var moved;
  6945. var lastPageX=e.pageX;
  6946. var currentLevel=this.zoomLevel;
  6947. var moveFired=0;
  6948. var moveHandler=function(e){
  6949. moveFired++
  6950. if(moveFired < 2){//有时候点击的时候不小心会触发一发move
  6951. return;
  6952. };
  6953. moved=true;
  6954. var pageX=e.pageX;
  6955. var level;
  6956. if(pageX > lastPageX){//向右移,zoomin扩大
  6957. self.changeCursor('zoom',false);
  6958. level=0.05;
  6959. }else{//向左移,zoomout缩小
  6960. self.changeCursor('zoom',true);
  6961. level=-0.05;
  6962. };
  6963. lastPageX=pageX;
  6964. currentLevel += level;
  6965. zoom(currentLevel);
  6966. };
  6967.  
  6968. var mouseupHandler=function(e){
  6969. self.working=false;
  6970. document.removeEventListener('mousemove',moveHandler,true);
  6971. document.removeEventListener('mouseup',mouseupHandler,true);
  6972.  
  6973. var level=self.getNextZoomLevel();
  6974.  
  6975. if(self.zoomOut && self.altKeyUp){
  6976. self.zoomOut=false;
  6977. };
  6978.  
  6979. if(!moved){//如果没有平移缩放。
  6980. zoom(level);
  6981. };
  6982.  
  6983. self.changeCursor('zoom',self.zoomOut);
  6984.  
  6985. if(self.tempZoom && self.ctrlKeyUp && self.altKeyUp){
  6986. self.tempZoom=false;
  6987. self.changeCursor(self.selectedTool);
  6988. };
  6989.  
  6990. };
  6991.  
  6992. document.addEventListener('mousemove',moveHandler,true);
  6993. document.addEventListener('mouseup',mouseupHandler,true);
  6994. },
  6995. getNextZoomLevel:function(){
  6996. var level;
  6997. var self=this;
  6998. if(this.zoomOut){//缩小
  6999. ImgWindowC.zoomRangeR._find(function(value){
  7000. if(value < self.zoomLevel){
  7001. level=value;
  7002. return true;
  7003. }
  7004. })
  7005. }else{
  7006. ImgWindowC.zoomRange._find(function(value){
  7007. if(value > self.zoomLevel){
  7008. level=value;
  7009. return true;
  7010. };
  7011. });
  7012. }
  7013. return level;
  7014. },
  7015. getZoomRatio:function(mouseCoor){
  7016. var ibcRect=this.img.getBoundingClientRect();
  7017. var ratio={
  7018. x:(mouseCoor.x-ibcRect.left)/ibcRect.width,
  7019. y:(mouseCoor.y-ibcRect.top)/ibcRect.height,
  7020. };
  7021. if(ratio.x<0){
  7022. ratio.x=0
  7023. }else if(ratio.x>1){
  7024. ratio.x=1
  7025. };
  7026. if(ratio.y<0){
  7027. ratio.y=0
  7028. }else if(ratio.y>1){
  7029. ratio.y=1
  7030. };
  7031. return ratio;
  7032. },
  7033. aerialView:function(e){
  7034. this.working=true;
  7035. //记住现在的缩放比例
  7036. var cLevel=this.zoomLevel;
  7037.  
  7038. var wSize=getWindowSize();
  7039. wSize.h -= 16;
  7040. wSize.w -= 16;
  7041.  
  7042. var imgWindow=this.imgWindow;
  7043. var imgWindowCS=unsafeWindow.getComputedStyle(imgWindow);
  7044. var rectSize={
  7045. h:parseFloat(imgWindowCS.height),
  7046. w:parseFloat(imgWindowCS.width),
  7047. };
  7048. var rectRatio=rectSize.h/rectSize.w;
  7049. var windowRatio=wSize.h/wSize.w;
  7050.  
  7051. var size;
  7052. var rangeSize={};
  7053. if(rectRatio > windowRatio){
  7054. size={
  7055. h:wSize.h,
  7056. w:wSize.h / rectRatio,
  7057. };
  7058. rangeSize.h=Math.min(wSize.h * (size.h / rectSize.h), size.h);
  7059. rangeSize.w=Math.min(rangeSize.h / windowRatio , size.w);
  7060. }else{
  7061. size={
  7062. w:wSize.w,
  7063. h:wSize.w * rectRatio,
  7064. };
  7065. rangeSize.w=Math.min(wSize.w * (size.w / rectSize.w), size.w);
  7066. rangeSize.h=Math.min(rangeSize.w * windowRatio , size.h);
  7067. };
  7068.  
  7069.  
  7070. this.zoom(this.getRotatedImgCliSize(size).w/this.imgNaturalSize.w);
  7071.  
  7072. this.center(true,true);
  7073.  
  7074. this.keepScreenInside();
  7075.  
  7076. var viewRange=this.viewRange;
  7077. var vRS=viewRange.style;
  7078. vRS.display='block';
  7079. vRS.height=rangeSize.h + 'px';
  7080. vRS.width=rangeSize.w + 'px';
  7081. vRS.top=0 + 'px';
  7082. vRS.left=0 + 'px';
  7083.  
  7084.  
  7085.  
  7086. var viewRangeRect=viewRange.getBoundingClientRect();
  7087. var scrolled=getScrolled();
  7088. var viewRangeCenterCoor={
  7089. x:viewRangeRect.left + scrolled.x + 1/2 * rangeSize.w,
  7090. y:viewRangeRect.top + scrolled.y + 1/2 * rangeSize.h,
  7091. };
  7092.  
  7093. var self=this;
  7094.  
  7095. var moveRange={
  7096. x:[8,8+size.w-rangeSize.w],
  7097. y:[8,8+size.h-rangeSize.h]
  7098. };
  7099.  
  7100.  
  7101. function setViewRangePosition(pageXY){
  7102. var top=pageXY.y - viewRangeCenterCoor.y;
  7103. var left=pageXY.x - viewRangeCenterCoor.x;
  7104. if(top<=moveRange.y[0]){
  7105. top=moveRange.y[0];
  7106. }else if(top>=moveRange.y[1]){
  7107. top=moveRange.y[1];
  7108. };
  7109. vRS.top= top + 'px';
  7110. if(left<=moveRange.x[0]){
  7111. left=moveRange.x[0];
  7112. }else if(left>=moveRange.x[1]){
  7113. left=moveRange.x[1];
  7114. };
  7115. vRS.left= left + 'px';
  7116. };
  7117.  
  7118. setViewRangePosition({
  7119. x:e.pageX,
  7120. y:e.pageY,
  7121. });
  7122.  
  7123. var moveHandler=function(e){
  7124. setViewRangePosition({
  7125. x:e.pageX,
  7126. y:e.pageY,
  7127. });
  7128. };
  7129.  
  7130. var mouseupHandler=function(){
  7131. self.working=false;
  7132. viewRange.style.display='none';
  7133. self.zoom(cLevel);
  7134. var scrolled=getScrolled();
  7135. imgWindow.style.top= -13 - rectSize.h * ((parseFloat(vRS.top) - moveRange.y[0])/size.h) + scrolled.y +'px';
  7136. imgWindow.style.left= -13 - rectSize.w * ((parseFloat(vRS.left) - moveRange.x[0])/size.w) + scrolled.x +'px';
  7137.  
  7138. //说明图片的高度没有屏幕高,居中
  7139. //说明图片的宽度没有屏幕宽,居中
  7140. self.center(rangeSize.w == size.w , rangeSize.h == size.h);
  7141.  
  7142. self.keepScreenInside();
  7143.  
  7144. document.removeEventListener('mousemove',moveHandler,true);
  7145. document.removeEventListener('mouseup',mouseupHandler,true);
  7146. };
  7147. document.addEventListener('mousemove',moveHandler,true);
  7148. document.addEventListener('mouseup',mouseupHandler,true);
  7149. },
  7150. setToolBadge:function(tool,content){
  7151. var scale=0;
  7152. switch(tool){
  7153. case 'zoom':{
  7154. scale=2;
  7155. }break;
  7156. case 'rotate':{
  7157. scale=1;
  7158. }break;
  7159. default:break;
  7160. }
  7161. content=typeof content=='string'? content : content.toFixed(scale);
  7162. this.toolMap[tool].nextElementSibling.textContent=content;
  7163. },
  7164. notExponential:function(num){//不要转为指数形势
  7165. if(num>0){
  7166. if(num >= 999999999999999934463){
  7167. return 999999999999999934463;
  7168. }else if(num <= 0.000001){
  7169. return 0.000001;
  7170. };
  7171. }else if(num < 0){
  7172. if(num >= -0.000001){
  7173. return -0.000001;
  7174. }else if(num <= -999999999999999934463){
  7175. return -999999999999999934463
  7176. };
  7177. };
  7178.  
  7179. return num;
  7180. },
  7181.  
  7182. blur:function(e){
  7183. if(!this.focused)return;
  7184. var imgWindow =this.imgWindow;
  7185. //点击imgWinodw的外部的时候失去焦点
  7186. if(e!==true && imgWindow.contains(e.target))return;
  7187. imgWindow.classList.remove('pv-pic-window-container_focus');
  7188. this.toolbar.classList.remove('pv-pic-window-toolbar_focus');
  7189. this.closeButton.classList.remove('pv-pic-window-close_focus');
  7190. //this.searchButton.classList.remove('pv-pic-window-search_focus');
  7191. this.img.classList.remove('pv-pic-window-pic_focus');
  7192. document.removeEventListener('mousedown',this._blur,true);
  7193. document.removeEventListener('keydown',this._focusedKeydown,true);
  7194. document.removeEventListener('keyup',this._focusedKeyup,true);
  7195. this.changeCursor('default');
  7196. ImgWindowC.selectedTool=this.selectedTool;
  7197. this.focused=false;
  7198. },
  7199. focus:function(){
  7200. if(this.focused)return;
  7201. this.imgWindow.classList.add('pv-pic-window-container_focus');
  7202. this.toolbar.classList.add('pv-pic-window-toolbar_focus');
  7203. this.closeButton.classList.add('pv-pic-window-close_focus');
  7204. //this.searchButton.classList.add('pv-pic-window-search_focus');
  7205. this.img.classList.add('pv-pic-window-pic_focus');
  7206. this.imgWindow.style.zIndex= ImgWindowC.styleZIndex;
  7207. this.zIndex=ImgWindowC.styleZIndex;
  7208. ImgWindowC.styleZIndex ++;
  7209. document.addEventListener('keydown',this._focusedKeydown,true);
  7210. document.addEventListener('keyup',this._focusedKeyup,true);
  7211. document.addEventListener('mousedown',this._blur,true);
  7212.  
  7213. //还原鼠标样式。
  7214. this.changeCursor(this.selectedTool);
  7215.  
  7216. if(prefs.imgWindow.syncSelectedTool && ImgWindowC.selectedTool){
  7217. this.selectTool(ImgWindowC.selectedTool);
  7218. };
  7219.  
  7220. this.focused=true;
  7221. },
  7222. focusedKeyup:function(e){
  7223. var keyCode=e.keyCode;
  7224. var valid=[32,18,16,72,17,72,82,90,67];
  7225. if(valid.indexOf(keyCode)==-1)return;
  7226.  
  7227. e.preventDefault();
  7228.  
  7229. switch(keyCode){
  7230. case 32:{//空格键,临时切换到移动
  7231. this.spaceKeyUp=true;
  7232. if(!this.tempHand)return;//如果之前没有临时切换到抓手工具(当已经在工作的时候,按下空格不会临时切换到抓手工具)
  7233. if(!this.working){//松开按键的时候,没有在继续平移了。
  7234. this.tempHand=false;
  7235. this.changeCursor(this.selectedTool);
  7236. };
  7237. }break;
  7238. case 18:{//alt键盘切换缩小放大。
  7239. this.altKeyUp=true;
  7240. if(!this.zoomOut)return;
  7241. if(!this.working){
  7242. this.zoomOut=false;
  7243. this.changeCursor('zoom');
  7244. if(this.tempZoom && this.ctrlKeyUp){
  7245. this.tempZoom=false;
  7246. this.changeCursor(this.selectedTool);
  7247. };
  7248. };
  7249. }break;
  7250. case 16:{//shift键,旋转的时候按住shift键,步进缩放。
  7251. this.shiftKeyUp=true;
  7252. }break;
  7253. case 17:{//ctrl键
  7254. clearTimeout(this.ctrlkeyDownTimer);
  7255. if(!this.justCKeyUp){//如果刚才没有松开c,规避划词软件的ctrl+c松开
  7256. this.ctrlKeyUp=true;
  7257. if(!this.tempZoom)return;//如果没有切换到了缩放
  7258. if(!this.working && this.altKeyUp){
  7259. this.tempZoom=false;
  7260. this.changeCursor(this.selectedTool);
  7261. };
  7262. };
  7263. }break;
  7264. case 67:{//c键
  7265. this.justCKeyUp=true;
  7266. var self=this;
  7267. clearTimeout(this.justCKeyUpTimer);
  7268. this.justCKeyUpTimer=setTimeout(function(){
  7269. self.justCKeyUp=false;
  7270. },100)
  7271. }break;
  7272. case 72:{//h键
  7273. this.hKeyUp=true;
  7274. }break;
  7275. case 82:{//r键
  7276. this.rKeyUp=true;
  7277. }break;
  7278. case 90:{//z键
  7279. this.zKeyUp=true;
  7280. }break;
  7281. default:break;
  7282. };
  7283.  
  7284. if([72,82,90].indexOf(keyCode)!=-1){
  7285. if(!this.working && this.restoreBeforeTool){
  7286. this.restoreBeforeTool=false;
  7287. this.selectTool(this.beforeTool);
  7288. };
  7289. };
  7290. },
  7291. focusedKeydown:function(e){
  7292. var keyCode=e.keyCode;
  7293. var valid=[32,82,72,90,18,16,17,27,67];//有效的按键
  7294. if(valid.indexOf(keyCode)==-1) return;
  7295.  
  7296. e.preventDefault();
  7297.  
  7298. if(this.working){//working的时候也可以接受按下shift键,以便旋转的时候可以任何时候按下
  7299. if(keyCode==16){//shift键
  7300. this.shiftKeyUp=false;
  7301. };
  7302. return;
  7303. };
  7304.  
  7305. switch(keyCode){
  7306. case 82:{//r键,切换到旋转工具
  7307. if(this.rKeyUp){
  7308. this.rKeyUp=false;
  7309. this.beforeTool=this.selectedTool;
  7310. this.selectTool('rotate');
  7311. };
  7312. }break;
  7313. case 72:{//h键,切换到抓手工具
  7314. if(this.hKeyUp){
  7315. this.hKeyUp=false;
  7316. this.beforeTool=this.selectedTool;
  7317. this.selectTool('hand');
  7318. };
  7319. }break;
  7320. case 90:{//z键,切换到缩放工具
  7321. if(this.zKeyUp){
  7322. this.zKeyUp=false;
  7323. this.beforeTool=this.selectedTool;
  7324. this.selectTool('zoom');
  7325. };
  7326. }break;
  7327. case 32:{//空格键阻止,临时切换到抓手功能
  7328. if(this.spaceKeyUp){
  7329. this.spaceKeyUp=false;
  7330. if(this.selectedTool!='hand'){
  7331. this.tempHand=true;
  7332. this.changeCursor('hand');
  7333. };
  7334. };
  7335. }break;
  7336. case 18:{//alt键,在当前选择是缩放工具的时候,按下的时候切换到缩小功能
  7337. if(this.altKeyUp){
  7338. if((this.selectedTool!='zoom' && !this.tempZoom) || this.zoomOut)return;
  7339. this.zoomOut=true;
  7340. this.altKeyUp=false;
  7341. this.changeCursor('zoom',true);
  7342. };
  7343. }break;
  7344. case 17:{//ctrl键临时切换到缩放工具
  7345. if(this.ctrlKeyUp){
  7346. var self=this;
  7347. this.ctrlkeyDownTimer=setTimeout(function(){//规避词典软件的ctrl+c,一瞬间切换到缩放的问题
  7348. self.ctrlKeyUp=false;
  7349. if(self.selectedTool!='zoom'){
  7350. self.tempZoom=true;
  7351. self.changeCursor('zoom');
  7352. };
  7353. },100);
  7354. };
  7355. }break;
  7356. case 67:{//c键
  7357. clearTimeout(this.ctrlkeyDownTimer);
  7358. }break;
  7359. case 27:{//ese关闭窗口
  7360. if(prefs.imgWindow.close.escKey){
  7361. this.remove();
  7362. };
  7363. }break;
  7364. default:break;
  7365. };
  7366. },
  7367.  
  7368. toolbarEventHandler:function(e){
  7369. e.stopPropagation();
  7370. var target=e.target;
  7371. var toolMap=this.toolMap;
  7372. for(var i in toolMap){
  7373. if(toolMap.hasOwnProperty(i) && toolMap[i]==target){
  7374. switch(e.type){
  7375. case 'mousedown':{
  7376. this.selectTool(i);
  7377. }break;
  7378. case 'dblclick':{
  7379. this.dblclickCommand(i);
  7380. }break;
  7381. default:break;
  7382. };
  7383. break;
  7384. };
  7385. };
  7386. },
  7387. imgWindowEventHandler:function(e){
  7388. e.stopPropagation();
  7389. var selectedTool=this.selectedTool;
  7390. if(selectedTool == "hand" && prefs.imgWindow.suitLongImg && this.isLongImg){
  7391. var inScroll=this.imgWindow.style.overflow=="scroll";
  7392. if(e.type == "wheel" && inScroll)
  7393. return;
  7394. if(e.type == "click" && !this.moving){
  7395. var wSize=getWindowSize();
  7396. var imgWindow=this.imgWindow;
  7397. var scrolled=getScrolled();
  7398. var origTop=parseFloat(imgWindow.style.top);
  7399. if(inScroll){
  7400. imgWindow.style.top = parseFloat(imgWindow.style.top) - getScrolled(imgWindow).y +'px';
  7401. }
  7402. this.imgWindow.style.height=inScroll?"":"100%";
  7403. this.imgWindow.style.maxWidth=inScroll?"":"100%";
  7404. this.imgWindow.style.overflow=inScroll?"":"scroll";
  7405. this.closeButton.style.display=inScroll?"":"none";
  7406. //this.center(true , true);
  7407. if(!inScroll){
  7408. imgWindow.style.top= (wSize.h - imgWindow.offsetHeight)/2 + scrolled.y +'px';
  7409. var scrollTop=parseFloat(imgWindow.style.top)-origTop;
  7410. if(this.imgWindow.scrollTop)this.imgWindow.scrollTop = scrollTop;
  7411. else if(this.imgWindow.pageYOffset)this.imgWindow.pageYOffset = scrollTop;
  7412. else if(this.imgWindow.scrollY)this.imgWindow.scrollY = scrollTop;
  7413. }
  7414. this.keepScreenInside();
  7415. }
  7416. }
  7417. switch(e.type){
  7418. case 'click':{//阻止opera的图片保存
  7419. this.moving=false;
  7420. if(e.ctrlKey && e.target.nodeName=='IMG'){
  7421. e.preventDefault();
  7422. }
  7423. }break;
  7424. case 'mousedown':
  7425. case 'touchstart':{
  7426. if(!this.focused){//如果没有focus,先focus
  7427. this.focus();
  7428. this.keepScreenInside();
  7429. };
  7430.  
  7431. var target=e.target;
  7432. if(e.button==2){//由于rotate时候的覆盖层问题,修复右键的图片菜单弹出
  7433. if(target!=this.rotateOverlayer)return;
  7434. var self=this;
  7435. this.rotateOverlayer.style.display='none';
  7436. var upHandler=function(){
  7437. document.removeEventListener('mouseup',upHandler,true);
  7438. setTimeout(function(){
  7439. self.rotateOverlayer.style.display='block';
  7440. },10);
  7441. };
  7442. document.addEventListener('mouseup',upHandler,true);
  7443. return;
  7444. };
  7445.  
  7446. if((e.button!=0 && e.type!="touchstart") || (target!=this.imgWindow && target!=this.img && target!=this.rotateOverlayer))return;
  7447. e.preventDefault();
  7448. if(this.tempHand){
  7449. this.move(e);
  7450. }else if(this.tempZoom){
  7451. this.zoom(e);
  7452. }else if(selectedTool=='hand'){
  7453. this.restoreBeforeTool=!this.hKeyUp;
  7454. if(this.hKeyUp){
  7455. this.move(e);
  7456. }else{//鸟瞰视图
  7457. this.aerialView(e);
  7458. };
  7459. }else if(selectedTool=='rotate'){
  7460. var origin={//旋转原点
  7461. x:e.clientX - 30,//稍微偏左一点。
  7462. y:e.clientY ,
  7463. };
  7464.  
  7465. var rIS=this.rotateIndicator.style;
  7466. //rIS.display='block';
  7467. rIS.top=origin.y + 'px';
  7468. rIS.left=origin.x + 'px';
  7469.  
  7470. this.restoreBeforeTool=!this.rKeyUp;
  7471. this.rotate(origin);
  7472. }else if(selectedTool=='zoom'){
  7473. this.restoreBeforeTool=!this.zKeyUp;
  7474. this.zoom(e);
  7475. };
  7476. }break;
  7477. case 'wheel':{
  7478. if(!this.focused)return;//如果没有focus
  7479. if(e.deltaY===0)return;//非Y轴的滚动
  7480. e.preventDefault();
  7481. if(this.working)return;
  7482. var oriZoomOut=this.zoomOut;
  7483. this.zoomOut = !!(e.deltaY > 0);
  7484.  
  7485. var ratio=this.getZoomRatio({
  7486. x:e.clientX,
  7487. y:e.clientY,
  7488. });
  7489.  
  7490. var level=this.getNextZoomLevel();
  7491.  
  7492. this.zoom(level,ratio);
  7493. this.zoomOut=oriZoomOut;
  7494. }break;
  7495. default:break;
  7496. };
  7497. },
  7498.  
  7499. dblclickCommand:function(tool){
  7500. var done;
  7501. switch(tool){
  7502. case 'hand':{//双击居中,并且适应屏幕
  7503. this.zoom(1);
  7504. this.fitToScreen();
  7505. this.center(true,true);
  7506. this.keepScreenInside();
  7507. }break;
  7508. case 'rotate':{//双击还原旋转
  7509. if(this.rotatedRadians==0)return;
  7510. done=true;
  7511. this.rotate(0,true);
  7512. }break;
  7513. case 'zoom':{//双击还原缩放
  7514. if(this.zoomLevel==1)return;
  7515. done=true;
  7516. this.zoom(1,{x:0,y:0});
  7517. }break;
  7518. default:break;
  7519. };
  7520.  
  7521. if((tool=='rotate' || tool=='zoom') && done){
  7522. var scrolled=getScrolled();
  7523. var imgWindow=this.imgWindow;
  7524. var imgWinodowRect=imgWindow.getBoundingClientRect();
  7525. var imgWindowStyle=imgWindow.style;
  7526. if(imgWinodowRect.left<40){
  7527. imgWindowStyle.left=40 + scrolled.x + 'px';
  7528. };
  7529. if(imgWinodowRect.top<-5){
  7530. imgWindowStyle.top=-5 + scrolled.y +'px';
  7531. };
  7532. this.keepScreenInside();
  7533. };
  7534.  
  7535. },
  7536. doFlipCommand:function(command){
  7537. var map={
  7538. fv:[/scaleY\([^)]*\)/i,' scaleY(-1) '],
  7539. fh:[/scaleX\([^)]*\)/i,' scaleX(-1) '],
  7540. };
  7541.  
  7542. var iTransform=this.img.style[support.cssTransform];
  7543.  
  7544. var toolClassList=this.toolMap[command].classList;
  7545.  
  7546. if(map[command][0].test(iTransform)){
  7547. iTransform=iTransform.replace(map[command][0],'');
  7548. toolClassList.remove(this.selectedToolClass);
  7549. }else{
  7550. iTransform += map[command][1];
  7551. toolClassList.add(this.selectedToolClass);
  7552. };
  7553. this.img.style[support.cssTransform]=iTransform;
  7554.  
  7555. },
  7556. selectTool:function(tool){
  7557. var command=['fv','fh'];
  7558. if(command.indexOf(tool)==-1){//工具选择
  7559. if(this.selectedTool==tool){
  7560. if(tool=="rotate"){
  7561. var PI=Math.PI;
  7562. var value=this.rotatedRadians + 90 * PI/180;
  7563. if(value>=2*PI){
  7564. value-=2*PI;
  7565. }
  7566. this.rotate(value,true);
  7567. }
  7568. return;
  7569. }
  7570. var selectedTool=this.selectedTool;
  7571. this.selectedTool=tool;
  7572. if(this.tempHand || this.tempZoom){//临时工具中。不变鼠标
  7573. return;
  7574. };
  7575.  
  7576. this.rotateOverlayer.style.display=(tool=='rotate'? 'block' : 'none');//这个覆盖层是为了捕捉双击或者单击事件。
  7577.  
  7578. if(selectedTool){
  7579. this.toolMap[selectedTool].classList.remove(this.selectedToolClass);
  7580. };
  7581. this.toolMap[tool].classList.add(this.selectedToolClass);
  7582. this.changeCursor(tool);
  7583. }else{//命令
  7584. this.doFlipCommand(tool);
  7585. };
  7586. },
  7587. changeCursor:function(tool,zoomOut){
  7588. if(tool=='zoom'){
  7589. tool+=zoomOut? '-out' : '-in';
  7590. };
  7591. if(this.cursor==tool)return;
  7592. this.cursor=tool;
  7593.  
  7594. var cursor;
  7595.  
  7596. switch(tool){
  7597. case 'hand':{
  7598. cursor=support.cssCursorValue.grab || 'pointer';
  7599. }break;
  7600. case 'handing':{
  7601. cursor=support.cssCursorValue.grabbing || 'pointer';
  7602. }break;
  7603. case 'zoom-in':{
  7604. cursor=support.cssCursorValue.zoomIn;
  7605. }break;
  7606. case 'zoom-out':{
  7607. cursor=support.cssCursorValue.zoomOut;
  7608. }break;
  7609. case 'rotate':{
  7610. cursor='progress';
  7611. }break;
  7612. case 'default':{
  7613. cursor='';
  7614. }break;
  7615. };
  7616.  
  7617. if(typeof cursor!='undefined'){
  7618. this.imgWindow.style.cursor=cursor;
  7619. };
  7620.  
  7621. },
  7622.  
  7623. remove:function(opacity){
  7624. if(this.removed)return;
  7625. this.removed=true;
  7626. //this.imgWindow.classList.remove("pv-pic-window-transition-all");
  7627. this.blur(true);
  7628. if(!opacity)this.imgWindow.style.opacity=0;
  7629. let self = this;
  7630. setTimeout(function(){
  7631. self.img.src= prefs.icons.brokenImg_small;//如果在加载中取消,图片也取消读取。
  7632. self.imgWindow.parentNode.removeChild(self.imgWindow);
  7633. },300);
  7634.  
  7635. var index=ImgWindowC.all.indexOf(this);
  7636. ImgWindowC.all.splice(index,1);
  7637.  
  7638. //focus next
  7639. if(ImgWindowC.all.length==0){
  7640. if(ImgWindowC.overlayer){
  7641. ImgWindowC.overlayer.style.display='none';
  7642. };
  7643. }else{
  7644. var topmost=0;
  7645. ImgWindowC.all.forEach(function(iwin){
  7646. if(iwin.zIndex > topmost){
  7647. topmost=iwin;
  7648. };
  7649. });
  7650. if(topmost){
  7651. topmost.focus();
  7652. };
  7653. };
  7654.  
  7655. },
  7656.  
  7657. };
  7658.  
  7659. // 载入动画
  7660. function LoadingAnimC(data,buttonType,waitImgLoad,openInTopWindow){
  7661. this.args=arrayFn.slice.call(arguments,0);
  7662. this.data=data;//data
  7663. this.buttonType=buttonType;//点击的按钮类型
  7664. this.openInTopWindow=openInTopWindow;//是否在顶层窗口打开,如果在frame里面的话
  7665. this.waitImgLoad=waitImgLoad;//是否等待完全读取后打开
  7666. this.init();
  7667. };
  7668.  
  7669. LoadingAnimC.all=[];
  7670.  
  7671. LoadingAnimC.prototype={
  7672. init:function(){
  7673. LoadingAnimC.all.push(this);
  7674. this.addStyle();
  7675. var container=document.createElement('span');
  7676.  
  7677. container.className='pv-loading-container';
  7678. this.loadingAnim=container;
  7679.  
  7680. container.title=i18n("loading")+':' + this.data.src;
  7681. let retrySpan=document.createElement('span');
  7682. retrySpan.className='pv-loading-button pv-loading-retry';
  7683. retrySpan.title='Retry';
  7684. container.appendChild(retrySpan);
  7685. let cancelSpan=document.createElement('span');
  7686. cancelSpan.className='pv-loading-button pv-loading-cancle';
  7687. cancelSpan.title='Cancel';
  7688. container.appendChild(cancelSpan);
  7689. /*container.innerHTML=
  7690. '<span class="pv-loading-button pv-loading-retry" title="重试"></span>'+
  7691. '<span class="pv-loading-button pv-loading-cancle" title="取消"></span>';*/
  7692.  
  7693. if (this.buttonType == 'popup'){
  7694. container.style.pointerEvents="none";
  7695. }
  7696. document.body.appendChild(container);
  7697.  
  7698. var self = this;
  7699. container.addEventListener('click',function(e){
  7700. var tcl=e.target.classList;
  7701. if(tcl.contains('pv-loading-cancle')){
  7702. self.imgReady.abort();
  7703. self.remove();
  7704. }else if(tcl.contains('pv-loading-retry')){
  7705. self.remove();
  7706. new LoadingAnimC(self.args[0],self.args[1],self.args[2],self.args[3]);
  7707. };
  7708. },true);
  7709.  
  7710. this.setPosition();
  7711.  
  7712. if (this.buttonType == 'current') {
  7713. this.loadImg(this.data.imgSrc);
  7714. } else {
  7715. if (!this.data.xhr) {
  7716. if(this.buttonType == 'search'){
  7717. sortSearch();
  7718. let from=0;
  7719. let searchFun=function(){
  7720. searchImgByImg(self.data.imgSrc, null, function(srcs, index){
  7721. let src=srcs.shift();
  7722. if(index==3){
  7723. self.loadImg(src, srcs);
  7724. }else{
  7725. from=index+1;
  7726. self.loadImg(src, srcs, searchFun);
  7727. }
  7728. },function(e) {
  7729. self.error("网络错误");
  7730. },function(e) {
  7731. self.error("没有找到原图");
  7732. }, from);
  7733. };
  7734. searchFun();
  7735. }else{
  7736. this.loadImg(this.data.src||this.data.imgSrc, this.data.srcs);
  7737. }
  7738. } else {
  7739. xhrLoad.load({
  7740. url: this.data.src,
  7741. xhr: this.data.xhr,
  7742. cb: function(imgSrc, imgSrcs, caption) {
  7743. if (imgSrc) {
  7744. self.loadImg(imgSrc, imgSrcs);
  7745. } else {
  7746. self.error();
  7747. }
  7748. },
  7749. onerror: function() {
  7750. self.error();
  7751. }
  7752. });
  7753. }
  7754. }
  7755. },
  7756. addStyle:function(){
  7757. if(LoadingAnimC.styleAdded)return;
  7758. LoadingAnimC.styleAdded=true;
  7759. var style=document.createElement('style');
  7760. style.type='text/css';
  7761. style.textContent='\
  7762. .pv-loading-container {\
  7763. position: absolute;\
  7764. z-index:999999997;\
  7765. background: black url("'+prefs.icons.loading+'") center no-repeat;\
  7766. background-origin: content-box;\
  7767. border: none;\
  7768. padding: 1px 30px 1px 2px;\
  7769. margin: 0;\
  7770. opacity: 0.5;\
  7771. height: 24px;\
  7772. min-width: 24px;\
  7773. box-shadow: 2px 2px 0px #666;\
  7774. -webkit-transition: opacity 0.15s ease-in-out;\
  7775. transition: opacity 0.15s ease-in-out;\
  7776. }\
  7777. .pv-loading-container:hover {\
  7778. opacity: 0.9;\
  7779. }\
  7780. .pv-loading-button {\
  7781. cursor: pointer;\
  7782. height: 24px;\
  7783. width: 24px;\
  7784. position: absolute;\
  7785. right: 0;\
  7786. top: 0;\
  7787. opacity: 0.4;\
  7788. background:transparent center no-repeat;\
  7789. -webkit-transition: opacity 0.15s ease-in-out;\
  7790. transition: opacity 0.15s ease-in-out;\
  7791. }\
  7792. .pv-loading-button:hover {\
  7793. opacity: 1;\
  7794. }\
  7795. .pv-loading-cancle{\
  7796. background-image: url("'+prefs.icons.loadingCancle+'");\
  7797. }\
  7798. .pv-loading-retry{\
  7799. display:none;\
  7800. background-image: url("'+prefs.icons.retry+'");\
  7801. }\
  7802. .pv-loading-container_error{\
  7803. background-image:none;\
  7804. }\
  7805. .pv-loading-container_error::after{\
  7806. content:"'+i18n("loadError")+'";\
  7807. line-height: 24px;\
  7808. color: red;\
  7809. font-size: 14px;\
  7810. display:inline;\
  7811. }\
  7812. .pv-loading-container_error .pv-loading-cancle{\
  7813. display:none;\
  7814. }\
  7815. .pv-loading-container_error .pv-loading-retry{\
  7816. display:block;\
  7817. }\
  7818. ';
  7819. document.head.appendChild(style);
  7820. },
  7821. remove:function(){
  7822. if(!this.removed){
  7823. this.removed=true;
  7824. this.loadingAnim.parentNode.removeChild(this.loadingAnim);
  7825. LoadingAnimC.all.splice(LoadingAnimC.all.indexOf(this),1);
  7826. };
  7827. },
  7828. error:function(msg,img,e){
  7829. if(msg)debug(msg);
  7830. this.loadingAnim.style.pointerEvents="";
  7831. this.loadingAnim.classList.add('pv-loading-container_error');
  7832. debug('Picviewer CE+ 载入大图错误:%o', this.data);
  7833.  
  7834. var self=this;
  7835. setTimeout(function(){
  7836. self.remove();
  7837. },3000);
  7838. },
  7839. setPosition:function(){
  7840. var position=getContentClientRect(this.data.img);
  7841. var cs=this.loadingAnim.style;
  7842. var scrolled=getScrolled();
  7843. cs.top=position.top + scrolled.y +1 + 'px';
  7844. cs.left=position.left + scrolled.x +1 + 'px';
  7845. cs.removeProperty('display');
  7846. },
  7847.  
  7848. // 根据 imgSrc 载入图片,imgSrcs 为备用图片地址,imgSrc 加载失败备用
  7849. loadImg: function(imgSrc, imgSrcs, nextFun) {
  7850. var self = this;
  7851.  
  7852. var img = document.createElement('img');
  7853. img.src = imgSrc;
  7854.  
  7855. var opts = {
  7856. error: function(e) {
  7857. if (Array.isArray(imgSrcs)) {
  7858. var src = imgSrcs.shift();
  7859. if (src) {
  7860. self.loadImg(src, imgSrcs, nextFun);
  7861. return;
  7862. }
  7863. }
  7864.  
  7865. if(nextFun) nextFun();
  7866. else self.error('', this, e);
  7867. },
  7868. };
  7869.  
  7870. opts[self.waitImgLoad ? 'load' : 'ready'] = function(e) {
  7871. self.load(this, e);
  7872. };
  7873.  
  7874. self.imgReady = imgReady(img, opts);
  7875. },
  7876.  
  7877. load:function(img,e){
  7878. this.remove();
  7879. this.img=img;
  7880. var buttonType=this.buttonType;
  7881.  
  7882. if(buttonType=='gallery'){
  7883. if(!gallery){
  7884. gallery=new GalleryC();
  7885. gallery.data=[];
  7886. }
  7887. var allData=gallery.getAllValidImgs();
  7888. allData.target=this.data;
  7889. this.data=allData;
  7890. };
  7891.  
  7892. var self=this;
  7893. function openInTop(){
  7894. var data=self.data;
  7895.  
  7896. //删除不能发送的项。
  7897. var delCantClone=function(obj){
  7898. if(!obj)return;
  7899. delete obj.img;
  7900. delete obj.imgPA;
  7901. };
  7902.  
  7903. if(Array.isArray(data)){
  7904. frameSentSuccessData=frameSentData;
  7905. frameSentData=cloneObject(data,true);
  7906. delCantClone(data.target);
  7907. data.forEach(function(obj){
  7908. delCantClone(obj);
  7909. });
  7910. }else{
  7911. delCantClone(data);
  7912. };
  7913.  
  7914. window.postMessage({
  7915. messageID:messageID,
  7916. src:img.src,
  7917. data:data,
  7918. command:'open',
  7919. buttonType:buttonType,
  7920. to:'top',
  7921. },'*');
  7922. };
  7923.  
  7924. if(this.openInTopWindow && isFrame && topWindowValid!==false && buttonType!='magnifier'){
  7925. if(topWindowValid){
  7926. openInTop();
  7927. }else{//先发消息问问顶层窗口是不是非frameset窗口
  7928. window.postMessage({
  7929. messageID:messageID,
  7930. command:'topWindowValid',
  7931. to:'top',
  7932. },'*');
  7933.  
  7934. document.addEventListener('pv-topWindowValid',function(e){
  7935. topWindowValid=e.detail;
  7936. if(topWindowValid){//如果顶层窗口有效
  7937. openInTop();
  7938. }else{
  7939. self.open();
  7940. };
  7941. },true);
  7942. };
  7943.  
  7944. }else{
  7945. this.open();
  7946. };
  7947.  
  7948.  
  7949. },
  7950. open:function(){
  7951. switch(this.buttonType){
  7952. case 'popup':
  7953. if(!uniqueImgWin || uniqueImgWin.removed){
  7954. uniqueImgWin = new ImgWindowC(this.img, this.data);
  7955. //uniqueImgWin.imgWindow.classList.add("pv-pic-window-transition-all");
  7956. }
  7957. if(uniqueImgWin.src != this.data.src && (!this.data.srcs || !this.data.srcs.includes(uniqueImgWin.src))){
  7958. uniqueImgWin.changeData(this.data);
  7959. }
  7960. uniqueImgWin.blur({target:this.data.img});
  7961. if(!uniqueImgWin.loaded){
  7962. if(prefs.waitImgLoad){
  7963. uniqueImgWin.imgWindow.style.display = "none";
  7964. uniqueImgWin.imgWindow.style.opacity = 0;
  7965. }else{
  7966. uniqueImgWin.center(true,true);
  7967. if(centerInterval)clearInterval(centerInterval);
  7968. centerInterval=setInterval(function(){
  7969. if(!uniqueImgWin || uniqueImgWin.removed || uniqueImgWin.loaded)
  7970. clearInterval(centerInterval);
  7971. else{
  7972. uniqueImgWin.center(true,true);
  7973. }
  7974. },300);
  7975. }
  7976. }
  7977. uniqueImgWin.imgWindow.style.pointerEvents = "none";
  7978. break;
  7979. case 'gallery':
  7980. if(!gallery){
  7981. gallery=new GalleryC();
  7982. };
  7983. gallery.load(this.data,this.from);
  7984. break;
  7985. case 'magnifier':
  7986. new MagnifierC(this.img,this.data);
  7987. break;
  7988. case 'actual':;
  7989. case 'search':;
  7990. case 'current':;
  7991. case 'original'://original 是为了兼容以前的规则
  7992. if(this.data.src!=this.img.src)this.data.src=this.img.src;
  7993. new ImgWindowC(this.img, this.data);
  7994. break;
  7995. };
  7996. },
  7997. };
  7998.  
  7999. //工具栏
  8000. function FloatBarC(){
  8001. this.init();
  8002. };
  8003.  
  8004. FloatBarC.prototype={
  8005. init:function(){
  8006. this.addStyle();
  8007. var container=document.createElement('span');
  8008. container.id='pv-float-bar-container';
  8009. document.body.appendChild(container);
  8010. for(let i=0;i<4;i++){
  8011. let spanChild=document.createElement('span');
  8012. spanChild.className='pv-float-bar-button';
  8013. container.appendChild(spanChild);
  8014. }
  8015. /*container.innerHTML=
  8016. '<span class="pv-float-bar-button"></span>'+
  8017. '<span class="pv-float-bar-button"></span>'+
  8018. '<span class="pv-float-bar-button"></span>'+
  8019. //'<span class="pv-float-bar-button"></span>'+
  8020. '<span class="pv-float-bar-button"></span>';*/
  8021.  
  8022. var buttons={
  8023. };
  8024. this.buttons=buttons;
  8025. this.children=container.children;
  8026.  
  8027. arrayFn.forEach.call(this.children,function(child,index){
  8028. var titleMap={
  8029. actual:i18n("actualBtn"),
  8030. search:i18n("searchBtn"),
  8031. gallery:i18n("galleryBtn"),
  8032. current:i18n("currentBtn"),
  8033. magnifier:i18n("magnifierBtn"),
  8034. };
  8035. var buttonName=prefs.floatBar.butonOrder[index];
  8036. if(!buttonName){
  8037. child.style.display="none";
  8038. return;
  8039. }
  8040. buttons[buttonName]=child;
  8041. child.title=titleMap[buttonName];
  8042. child.classList.add('pv-float-bar-button-' + buttonName);
  8043. });
  8044.  
  8045.  
  8046. this.floatBar=container;
  8047.  
  8048.  
  8049. var self=this;
  8050. container.addEventListener('click',function(e){
  8051. var buttonType;
  8052. var target=e.target;
  8053. for(var type in buttons){
  8054. if(!buttons.hasOwnProperty(type))return;
  8055. if(target==buttons[type]){
  8056. buttonType=type;
  8057. break;
  8058. };
  8059. };
  8060. if(!buttonType)return;
  8061.  
  8062. self.hide();
  8063. self.open(e,buttonType);
  8064.  
  8065. },true);
  8066.  
  8067.  
  8068. addCusMouseEvent('mouseleave',container,function(e){
  8069. clearTimeout(self.hideTimer);
  8070. self.hideTimer=setTimeout(function(){
  8071. self.hide();
  8072. },prefs.floatBar.hideDelay);
  8073. });
  8074.  
  8075. addCusMouseEvent('mouseenter',container,function(e){
  8076. clearTimeout(self.hideTimer);
  8077. clearTimeout(self.showTimer);
  8078. clearTimeout(self.globarOutTimer);
  8079. });
  8080.  
  8081. this._scrollHandler=this.scrollHandler.bind(this);
  8082. },
  8083. addStyle:function(){
  8084. var style=document.createElement('style');
  8085. style.type='text/css';
  8086. style.textContent='\
  8087. #pv-float-bar-container {\
  8088. position: absolute;\
  8089. z-index:9999999998;\
  8090. padding: 5px;\
  8091. margin: 0;\
  8092. border: none;\
  8093. opacity: 0.6;\
  8094. line-height: 0;\
  8095. -webkit-transition: opacity 0.2s ease-in-out;\
  8096. transition: opacity 0.2s ease-in-out;\
  8097. display:none;\
  8098. }\
  8099. #pv-float-bar-container:hover {\
  8100. opacity: 1;\
  8101. }\
  8102. #pv-float-bar-container .pv-float-bar-button {\
  8103. vertical-align:middle;\
  8104. cursor: pointer;\
  8105. width: 18px;\
  8106. height: 18px;\
  8107. padding: 0;\
  8108. margin:0;\
  8109. border: none;\
  8110. display: inline-block;\
  8111. position: relative;\
  8112. box-shadow: 1px 0 3px 0px rgba(0,0,0,0.9);\
  8113. background: transparent center no-repeat;\
  8114. background-size:100% 100%;\
  8115. background-origin: content-box;\
  8116. -webkit-transition: margin-right 0.15s ease-in-out , width 0.15s ease-in-out , height 0.15s ease-in-out ;\
  8117. transition: margin-right 0.15s ease-in-out , width 0.15s ease-in-out , height 0.15s ease-in-out ;\
  8118. }\
  8119. #pv-float-bar-container .pv-float-bar-button:not(:last-child){\
  8120. margin-right: -14px;\
  8121. }\
  8122. #pv-float-bar-container .pv-float-bar-button:first-child {\
  8123. z-index: 4;\
  8124. }\
  8125. #pv-float-bar-container .pv-float-bar-button:nth-child(2) {\
  8126. z-index: 3;\
  8127. }\
  8128. #pv-float-bar-container .pv-float-bar-button:nth-child(3) {\
  8129. z-index: 2;\
  8130. }\
  8131. #pv-float-bar-container .pv-float-bar-button:last-child {\
  8132. z-index: 1;\
  8133. }\
  8134. #pv-float-bar-container:hover > .pv-float-bar-button {\
  8135. width: 24px;\
  8136. height: 24px;\
  8137. }\
  8138. #pv-float-bar-container:hover > .pv-float-bar-button:not(:last-child) {\
  8139. margin-right: 4px;\
  8140. }\
  8141. #pv-float-bar-container .pv-float-bar-button-actual {\
  8142. background-image:url("'+ prefs.icons.actual +'");\
  8143. }\
  8144. #pv-float-bar-container .pv-float-bar-button-search {\
  8145. background-image:url("'+ prefs.icons.search +'");\
  8146. }\
  8147. #pv-float-bar-container .pv-float-bar-button-gallery {\
  8148. background-image:url("'+ prefs.icons.gallery +'");\
  8149. }\
  8150. #pv-float-bar-container .pv-float-bar-button-current {\
  8151. background-image:url("'+ prefs.icons.current +'");\
  8152. }\
  8153. #pv-float-bar-container .pv-float-bar-button-magnifier {\
  8154. background-image:url("'+ prefs.icons.magnifier +'");\
  8155. }\
  8156. ';
  8157. document.head.appendChild(style);
  8158. },
  8159. start:function(data){
  8160.  
  8161. //读取中的图片,不显示浮动栏,调整读取图标的位置.
  8162. if(LoadingAnimC.all._find(function(item,index,array){
  8163. if(data.img==item.data.img){
  8164. return true;
  8165. };
  8166. }))return;
  8167.  
  8168.  
  8169. //被放大镜盯上的图片,不要显示浮动栏.
  8170. if(MagnifierC.all._find(function(item,index,array){
  8171. if(data.img==item.data.img){
  8172. return true;
  8173. };
  8174. }))return;
  8175.  
  8176. var self=this;
  8177. clearTimeout(this.hideTimer);
  8178.  
  8179. var imgOutHandler=function(e){
  8180. document.removeEventListener('mouseout',imgOutHandler,true);
  8181. clearTimeout(self.showTimer);
  8182. clearTimeout(self.hideTimer);
  8183. self.hideTimer=setTimeout(function(){
  8184. self.hide();
  8185. },prefs.floatBar.hideDelay);
  8186. };
  8187.  
  8188. clearTimeout(this.globarOutTimer);
  8189. this.globarOutTimer=setTimeout(function(){//稍微延时。错开由于css hover样式发生的out;
  8190. document.addEventListener('mouseout',imgOutHandler,true);
  8191. },150);
  8192.  
  8193. clearTimeout(this.showTimer);
  8194. this.showTimer=setTimeout(function(){
  8195. self.data=data;
  8196. self.show();
  8197. },prefs.floatBar.showDelay);
  8198. },
  8199. setButton:function(){
  8200. if(this.data.noActual){
  8201. this.buttons['actual'].style.display='none';
  8202. }else{
  8203. this.buttons['actual'].style.removeProperty('display');
  8204. }
  8205. if(this.data.type != "force" && this.data.img.nodeName == 'IMG'){
  8206. this.buttons['magnifier'].style.removeProperty('display');
  8207. }else{
  8208. this.buttons['magnifier'].style.display='none';
  8209. }
  8210. if (this.data.img.nodeName != 'IMG') {
  8211. //this.buttons['gallery'].style.display = 'none';
  8212. //this.buttons['current'].style.display = 'none';
  8213. } else {
  8214. //this.buttons['gallery'].style.removeProperty('display');
  8215. //this.buttons['current'].style.removeProperty('display');
  8216. }
  8217. },
  8218. setPosition:function(){
  8219. //如果图片被删除了,或者隐藏了。
  8220. if(this.data.img.offsetWidth==0){
  8221. return true;
  8222. };
  8223. var targetPosi=getContentClientRect(this.data.img);
  8224. var windowSize=getWindowSize();
  8225. var img=this.data.img;
  8226.  
  8227. var floatBarPosi=prefs.floatBar.position.toLowerCase().split(/\s+/);
  8228.  
  8229. var offsetX=prefs.floatBar.offset.x;
  8230. var offsetY=prefs.floatBar.offset.y;
  8231.  
  8232.  
  8233. var scrolled=getScrolled();
  8234.  
  8235. var fbs=this.floatBar.style;
  8236. var setPosition={
  8237. top:function(){
  8238. fbs.opacity="";
  8239. var top=targetPosi.top + scrolled.y;
  8240. if(targetPosi.top + offsetY < 0){//满足图标被遮住的条件.
  8241. top=scrolled.y;
  8242. offsetY=0;
  8243. }
  8244. top=top + offsetY;
  8245. if(targetPosi.height<=50)top-=10;
  8246. fbs.top=top + 'px';
  8247. },
  8248. right:function(){
  8249. fbs.opacity="";
  8250. var right=windowSize.w - targetPosi.right;
  8251. if(right < offsetX){
  8252. right= -scrolled.x;
  8253. offsetX=0;
  8254. }else{
  8255. right -=scrolled.x;
  8256. }
  8257. right=right - offsetX;
  8258. if(targetPosi.width<=50)right+=10;
  8259. fbs.right=right + 'px';
  8260. },
  8261. bottom:function(){
  8262. fbs.opacity="";
  8263. var bottom=windowSize.h - targetPosi.bottom;
  8264. if(bottom <= offsetY){
  8265. bottom=-scrolled.y;
  8266. offsetY=0;
  8267. }else{
  8268. bottom -= scrolled.y;
  8269. }
  8270. bottom=bottom - offsetY;
  8271. if(targetPosi.height<=50)bottom+=10;
  8272. fbs.bottom=bottom + 'px';
  8273. },
  8274. left:function(){
  8275. fbs.opacity="";
  8276. var left=targetPosi.left + scrolled.x;
  8277. if(targetPosi.left + offsetX < 0){
  8278. left=scrolled.x;
  8279. offsetX=0;
  8280. }
  8281. left=left + offsetX;
  8282. if(targetPosi.width<=50)left-=10;
  8283. fbs.left=left + 'px';
  8284. },
  8285. center:function(){
  8286. fbs.opacity="";
  8287. var left=targetPosi.left + scrolled.x + offsetX;
  8288. fbs.left=left + img.width/2 + 'px';
  8289. },
  8290. hide:function(){
  8291. fbs.opacity=0;
  8292. fbs.display="none";
  8293. },
  8294. };
  8295.  
  8296. setPosition[floatBarPosi[0]]();
  8297. if(floatBarPosi.length>1)
  8298. setPosition[floatBarPosi[1]]();
  8299. },
  8300. show:function(){
  8301. if(this.setPosition())return;
  8302. this.shown=true;
  8303. this.setButton();
  8304. if(this.floatBar.style.opacity!=="0")this.floatBar.style.display='block';
  8305. clearTimeout(this.hideTimer);
  8306. window.removeEventListener('scroll',this._scrollHandler,true);
  8307. window.addEventListener('scroll',this._scrollHandler,true);
  8308. },
  8309. hide:function(){
  8310. clearTimeout(this.showTimer);
  8311. this.shown=false;
  8312. this.floatBar.style.display='none';
  8313. window.removeEventListener('scroll',this._scrollHandler,true);
  8314. },
  8315. scrollHandler:function(){//更新坐标
  8316. clearTimeout(this.scrollUpdateTimer);
  8317. var self=this;
  8318. this.scrollUpdateTimer=setTimeout(function(){
  8319. self.setPosition();
  8320. },100);
  8321. },
  8322. open:function(e,buttonType){
  8323. var waitImgLoad = e && e.ctrlKey ? !prefs.waitImgLoad : prefs.waitImgLoad; //按住ctrl取反向值
  8324. var openInTopWindow = e && e.shiftKey ? !prefs.framesPicOpenInTopWindow : prefs.framesPicOpenInTopWindow; //按住shift取反向值
  8325. if (!waitImgLoad && buttonType == 'magnifier' && !envir.chrome) { //非chrome的background-image需要全部载入后才能显示出来
  8326. waitImgLoad = true;
  8327. };
  8328. new LoadingAnimC(this.data, buttonType, waitImgLoad, openInTopWindow);
  8329. },
  8330. update:function(img,src){
  8331. if(this.data.img==img && this.data.imgSrc!=src){
  8332. this.data.src=src;
  8333. this.data.noActual=false;
  8334. this.data.type="rule";
  8335. if(this.shown){
  8336. this.setButton();
  8337. }
  8338. }
  8339. }
  8340. };
  8341.  
  8342. /**
  8343. * 提取自 Mouseover Popup Image Viewer 脚本,用于 xhr 方式的获取
  8344. */
  8345. var xhrLoad = function() {
  8346. var _ = {};
  8347.  
  8348. var caches = {};
  8349. var handleError;
  8350.  
  8351. /**
  8352. * @param q 图片的选择器或函数
  8353. * @param c 图片说明的选择器或函数
  8354. */
  8355. function parsePage(url, q, c, post, cb) {
  8356. downloadPage(url, post, function(html) {
  8357. var iurl, iurls = [], cap, doc = createDoc(html);
  8358.  
  8359. if(typeof q == 'function') {
  8360. iurl = q(html, doc);
  8361. } else {
  8362. var inodes = findNodes(q, doc);
  8363. inodes.forEach(function(node) {
  8364. iurls.push(findFile(node, url));
  8365. });
  8366. iurl = iurls.shift();
  8367. }
  8368.  
  8369. if(typeof c == 'function') {
  8370. cap = c(html, doc);
  8371. } else {
  8372. var cnodes = findNodes(c, doc);
  8373. cap = cnodes.length ? findCaption(cnode[0]) : false;
  8374. }
  8375.  
  8376. // 缓存
  8377. if (iurl) {
  8378. caches[url] = {
  8379. iurl: iurl,
  8380. iurls: iurls,
  8381. cap: cap
  8382. };
  8383. }
  8384.  
  8385. cb(iurl, iurls, cap);
  8386. });
  8387. }
  8388.  
  8389. function downloadPage(url, post, cb) {
  8390. var opts = {
  8391. method: 'GET',
  8392. url: url,
  8393. onload: function(req) {
  8394. try {
  8395. if(req.status > 399) throw 'Server error: ' + req.status;
  8396. cb(req.responseText, req.finalUrl || url);
  8397. } catch(ex) {
  8398. handleError(ex);
  8399. }
  8400. },
  8401. onerror: handleError
  8402. };
  8403. if(post) {
  8404. opts.method = 'POST';
  8405. opts.data = post;
  8406. opts.headers = {'Content-Type':'application/x-www-form-urlencoded','Referer':url};
  8407. }
  8408.  
  8409. _GM_xmlhttpRequest(opts);
  8410. }
  8411.  
  8412. function createDoc(text) {
  8413. var doc = document.implementation.createHTMLDocument('PicViewerCE');
  8414. doc.documentElement.innerHTML = text;
  8415. return doc;
  8416. }
  8417.  
  8418. function findNodes(q, doc) {
  8419. var nodes = [],
  8420. node;
  8421. if (!Array.isArray(q)) q = [q];
  8422. for (var i = 0, len = q.length; i < len; i++) {
  8423. node = qs(q[i], doc);
  8424. if (node) {
  8425. nodes.push(node);
  8426. }
  8427. }
  8428. return nodes;
  8429. }
  8430.  
  8431. function findFile(n, url) {
  8432. var path = n.src || n.href;
  8433. return path ? path.trim() : false;
  8434. }
  8435.  
  8436. function findCaption(n) {
  8437. return n.getAttribute('content') || n.getAttribute('title') || n.textContent;
  8438. }
  8439.  
  8440. function qs(s, n) {
  8441. return n.querySelector(s);
  8442. }
  8443.  
  8444. _.load = function(opt) {
  8445. var info = caches[opt.url];
  8446. if (info) {
  8447. opt.cb(info.iurl, info.iruls, info.cap);
  8448. return;
  8449. }
  8450.  
  8451. handleError = opt.onerror || function() {};
  8452.  
  8453. parsePage(opt.url, opt.xhr.q, opt.xhr.c, opt.post, opt.cb);
  8454. };
  8455.  
  8456. return _;
  8457. }();
  8458.  
  8459.  
  8460. // ------------------- run -------------------------
  8461.  
  8462. var matchedRule,
  8463. URL=location.href,
  8464. floatBar;
  8465.  
  8466. function pretreatment(img){
  8467. if(img.nodeName != "IMG" || img.src)return;
  8468. if(img._lazyrias && img._lazyrias.srcset){
  8469. img.src=img._lazyrias.srcset[0];
  8470. }else if(img.srcset){
  8471. var srcs=img.srcset.split(","),minSize=0,newSrc;
  8472. srcs.forEach(srci=>{
  8473. let srcInfo=srci.trim().split(" "),curSize=parseInt(srcInfo[1]);
  8474. if(srcInfo[1] && (curSize<minSize || minSize==0)){
  8475. minSize=curSize;
  8476. newsrc=srcInfo[0];
  8477. }
  8478. });
  8479. if(newSrc)img.src=newSrc;
  8480. }else if(img.currentSrc){
  8481. img.src=img.currentSrc;
  8482. }
  8483. }
  8484.  
  8485. function findPic(img){
  8486. var imgPN=img;
  8487. var imgPA,imgPE=[];
  8488. while(imgPN=imgPN.parentElement){
  8489. if(imgPN.nodeName=='A'){
  8490. imgPA=imgPN;
  8491. break;
  8492. }
  8493. }
  8494. imgPN=img;
  8495. while(imgPN=imgPN.parentElement){
  8496. if(imgPN.nodeName=='BODY'){
  8497. break;
  8498. }else{
  8499. imgPE.push(imgPN);
  8500. }
  8501. }
  8502.  
  8503. var iPASrc=imgPA? imgPA.href : '';
  8504. //base64字符串过长导致正则匹配卡死浏览器
  8505. var base64Img=/^data:/i.test(img.src);
  8506. var src, // 大图地址
  8507. srcs, // 备用的大图地址
  8508. type, // 类别
  8509. noActual = false, //没有原图
  8510. imgSrc = img.currentSrc||img.dataset.lazySrc||img.src, // img 节点的 src
  8511. xhr,
  8512. description; // 图片的注释
  8513. var imgCStyle=unsafeWindow.getComputedStyle(img);
  8514. var imgCS={
  8515. h: parseFloat(imgCStyle.height)||img.height||img.offsetHeight,
  8516. w: parseFloat(imgCStyle.width)||img.width||img.offsetWidth,
  8517. };
  8518. var imgAS={//实际尺寸。
  8519. h:img.naturalHeight||imgCS.h,
  8520. w:img.naturalWidth||imgCS.w,
  8521. };
  8522. if(!src && matchedRule.rules.length>0){// 通过高级规则获取.
  8523. // 排除
  8524. try{
  8525. var newSrc=matchedRule.getImage(img,imgPA,imgPE);
  8526. if(newSrc && imgSrc!=newSrc) src=newSrc;
  8527. }catch(err){
  8528. throwErrorInfo(err);
  8529. }
  8530.  
  8531. if(src) {
  8532. if (Array.isArray(src)) {
  8533. srcs = src;
  8534. src = srcs.shift();
  8535. }
  8536.  
  8537. type = 'rule';
  8538. xhr = matchedRule.xhr;
  8539.  
  8540. if (matchedRule.lazyAttr) { // 由于采用了延迟加载技术,所以图片可能为 loading.gif
  8541. imgSrc = img.getAttribute(matchedRule.lazyAttr) || img.src;
  8542. }
  8543.  
  8544. if (matchedRule.description) {
  8545. var node = getElementMix(matchedRule.description, img);
  8546. if (node) {
  8547. description = node.getAttribute('title') || node.textContent;
  8548. }
  8549. }
  8550. }
  8551. }
  8552.  
  8553. if(!src && !base64Img){//遍历通配规则
  8554. tprules._find(function(rule,index,array){
  8555. try{
  8556. src=rule.call(img,imgPA);
  8557. if(src){
  8558. return true;
  8559. };
  8560. }catch(err){
  8561. throwErrorInfo(err);
  8562. };
  8563. });
  8564. if(src)type='tpRule';
  8565. }
  8566.  
  8567. if(!src && iPASrc){//链接可能是一张图片...
  8568. if(iPASrc!=img.src && /\.(jpg|jpeg|png|gif|bmp)(\?[^\?]*)?$/i.test(iPASrc)){
  8569. src=iPASrc;
  8570. }
  8571. if(src)type='scale';
  8572. }
  8573.  
  8574. if(!src || src==imgSrc){//本图片是否被缩放.
  8575. noActual=true;
  8576. if(!(imgAS.w==imgCS.w && imgAS.h==imgCS.h)){//如果不是两者完全相等,那么被缩放了.
  8577. src=imgSrc;
  8578. type='scale';
  8579. if (imgAS.h < prefs.gallery.scaleSmallSize && imgAS.w < prefs.gallery.scaleSmallSize) {
  8580. type='scaleSmall';
  8581. }
  8582. }else{
  8583. src=imgSrc;
  8584. type='force';
  8585. }
  8586. }
  8587.  
  8588. if(!src)return;
  8589.  
  8590. if(/^blob/i.test(imgSrc)){
  8591. imgSrc=drawTobase64(img);
  8592. src=imgSrc;
  8593. }
  8594.  
  8595. var ret = {
  8596. src: src, // 得到的src
  8597. srcs: srcs, // 多个 src,失败了会尝试下一个
  8598. type: type, // 通过哪种方式得到的
  8599. imgSrc: imgSrc, // 处理的图片的src
  8600. iPASrc: iPASrc, // 图片的第一个父a元素的链接地址
  8601. sizeH:imgAS.h,
  8602. sizeW:imgAS.w,
  8603. imgCS:imgCS,
  8604. imgAS:imgAS,
  8605.  
  8606. noActual:noActual,
  8607. xhr: xhr,
  8608. description: description || '',
  8609.  
  8610. img: img, // 处理的图片
  8611. imgPA: imgPA, // 图片的第一个父a元素
  8612. };
  8613. return ret;
  8614. }
  8615.  
  8616. function getMatchedRule() {
  8617. return new MatchedRuleC();
  8618. /*var rule = siteInfo._find(function(site, index, array) {
  8619. if (site.enabled != false && site.url && toRE(site.url).test(URL)) {
  8620. return true;
  8621. }
  8622. });
  8623.  
  8624. rule = rule ? rule[0] : false;
  8625.  
  8626. return rule;*/
  8627. }
  8628.  
  8629. function MatchedRuleC(){
  8630. this.init();
  8631. }
  8632.  
  8633. MatchedRuleC.prototype={
  8634. init:function(){
  8635. try{
  8636. var customRules=unsafeWindow.eval(prefs.customRules);
  8637. if(Array.isArray(customRules)){
  8638. customRules.forEach(rule=>{
  8639. let hasRule = false;
  8640. for(let s in siteInfo){
  8641. if(siteInfo[s].name == rule.name){
  8642. hasRule = true;
  8643. for(let si in rule){
  8644. siteInfo[s][si]=rule[si];
  8645. }
  8646. break;
  8647. }
  8648. }
  8649. if(!hasRule)siteInfo.unshift(rule);
  8650. })
  8651. //siteInfo=customRules.concat(siteInfo);
  8652. }
  8653. }catch(e){}
  8654.  
  8655. var self=this,r=0;
  8656. self.rules=[];
  8657. function searchByTime(){
  8658. setTimeout(()=>{
  8659. let end=r+30;
  8660. end=end>siteInfo.length?siteInfo.length:end;
  8661. for(;r<end;r++){
  8662. let site=siteInfo[r];
  8663. if (site.enabled != false && (!site.url || toRE(site.url).test(URL))) {
  8664. if(site.url){
  8665. if(site.css){
  8666. var style = document.createElement('style');
  8667. style.type = 'text/css';
  8668. style.id = 'gm-picviewer-site-style';
  8669. style.textContent = site.css;
  8670. document.head.appendChild(style);
  8671. }
  8672. if(site.xhr){
  8673. self.xhr=site.xhr;
  8674. }
  8675. if(site.lazyAttr){
  8676. self.lazyAttr=site.lazyAttr;
  8677. }
  8678. if(site.description){
  8679. self.description=site.description;
  8680. }
  8681. if(site.clickToOpen){
  8682. self.clickToOpen=site.clickToOpen;
  8683. }
  8684. if(site.ext){
  8685. self.ext=site.ext;
  8686. }
  8687. }
  8688. self.rules.push(site);
  8689. }
  8690. }
  8691. if(end<siteInfo.length){
  8692. searchByTime();
  8693. }
  8694. },10);
  8695. }
  8696. searchByTime();
  8697. },
  8698. replace:function(str, r, s){
  8699. var results=[],rt;
  8700. if(Array.isArray(s)){
  8701. s.forEach(_s=>{
  8702. rt=str.replace(r, _s);
  8703. if(rt && rt!=str)results.push(rt);
  8704. });
  8705. }else{
  8706. rt=str.replace(r, s);
  8707. if(rt && rt!=str)return str.replace(r, s);
  8708. }
  8709. return results;
  8710. },
  8711. getImage:function(img, a, p){
  8712. var newSrc,rule;
  8713. var base64Img=/^data:/i.test(img.src);
  8714. for(var i in this.rules){
  8715. rule=this.rules[i];
  8716. if((!rule.url || !rule.getImage) && base64Img)continue;
  8717. if(rule.src && !rule.src.test(img.src))continue;
  8718. if(rule.exclude && rule.exclude.test(img.src))continue;
  8719. if(rule.getImage){
  8720. newSrc = rule.getImage.call(img, a, p);
  8721. }else{
  8722. if(rule.r){
  8723. if(Array.isArray(rule.r)){//r最多一层
  8724. for(var j in rule.r){
  8725. var _r=rule.r[j];
  8726. if(_r && _r.test && _r.test(img.src)){
  8727. if(Array.isArray(rule.s)){//s对上r最多两层
  8728. var _s=rule.s[j];
  8729. newSrc=this.replace(img.src, _r, _s);
  8730. }else{
  8731. newSrc=this.replace(img.src, _r, rule.s);
  8732. }
  8733. break;
  8734. }
  8735. }
  8736. }else{
  8737. newSrc=this.replace(img.src, rule.r, rule.s);
  8738. }
  8739. }
  8740. }
  8741. if(newSrc && newSrc.length>0 && newSrc!=img.src){
  8742. debug(rule);
  8743. break;
  8744. }
  8745. }
  8746. if(newSrc && newSrc.length==0)newSrc=null;
  8747. return newSrc;
  8748. }
  8749. };
  8750.  
  8751. var isFrame=window!=window.parent;
  8752. var topWindowValid;
  8753. var frameSentData;
  8754. var frameSentSuccessData;
  8755. function handleMessage(e){
  8756. var data=e.data;
  8757. if( !data || !data.messageID || data.messageID != messageID )return;
  8758. var source=e.source,command,cusEvent;
  8759. if(typeof source=='undefined' || source!==window){
  8760. if(!isFrame){
  8761. command=data.command;
  8762. switch(command){
  8763. case 'open':{
  8764. var img=document.createElement('img');
  8765. img.src=data.src;
  8766.  
  8767. imgReady(img,{
  8768. ready:function(){
  8769. LoadingAnimC.prototype.open.call({
  8770. img:img,
  8771. data:data.data,
  8772. buttonType:data.buttonType,
  8773. from:data.from,
  8774. });
  8775. },
  8776. });
  8777. }break;
  8778. case 'navigateToImg':{
  8779. cusEvent=document.createEvent('CustomEvent');
  8780. cusEvent.initCustomEvent('pv-navigateToImg',false,false,data.exist);
  8781. document.dispatchEvent(cusEvent);
  8782. }break;
  8783. case 'topWindowValid':{
  8784. if(data.from)
  8785. window.postMessage({
  8786. messageID:messageID,
  8787. command:'topWindowValid_frame',
  8788. valid:document.body.nodeName!='FRAMESET',
  8789. to:data.from,
  8790. },'*');
  8791. }break;
  8792. };
  8793.  
  8794. }else{
  8795. command=data.command;
  8796. switch(command){
  8797. case 'navigateToImg':{
  8798.  
  8799. if(!frameSentData.unique){
  8800. var unique=GalleryC.prototype.unique(frameSentData);
  8801. frameSentData=unique.data;
  8802. frameSentData.unique=true;
  8803. };
  8804. var targetImg=frameSentData[data.index].img;
  8805. var exist=(document.documentElement.contains(targetImg) && unsafeWindow.getComputedStyle(targetImg).display!='none');
  8806.  
  8807. if(exist){
  8808. if(gallery && gallery.shown){
  8809. gallery.minimize();
  8810. };
  8811. setTimeout(function(){
  8812. GalleryC.prototype.navigateToImg(targetImg);
  8813. flashEle(targetImg);
  8814. },0);
  8815. };
  8816. window.postMessage({
  8817. messageID:messageID,
  8818. command:'navigateToImg',
  8819. exist:exist,
  8820. to:data.from,
  8821. },'*');
  8822. }break;
  8823. case 'sendFail':{
  8824. frameSentData=frameSentSuccessData;
  8825. }break;
  8826. case 'topWindowValid_frame':{
  8827. cusEvent=document.createEvent('CustomEvent');
  8828. cusEvent.initCustomEvent('pv-topWindowValid',false,false,data.valid);
  8829. document.dispatchEvent(cusEvent);
  8830. }break;
  8831. };
  8832. };
  8833.  
  8834. };
  8835. }
  8836.  
  8837. //页面脚本用来转发消息
  8838. //原因chrome的contentscript无法访问非自己外的别的窗口。都会返回undefined,自然也无法向其他的窗口发送信息,这里用pagescript做个中间代理
  8839. //通讯逻辑..A页面的contentscript发送到A页面的pagescript,pagescript转交给B页面的contentscript
  8840. var messageID='pv-0.5106795670312598';
  8841.  
  8842. function isunsafe(){
  8843. try {
  8844. return eval("false");
  8845. } catch (e) {
  8846. return true;
  8847. }
  8848. }
  8849. function addPageScript() {
  8850.  
  8851. if(isunsafe())return;
  8852. var pageScript=document.createElement('script');
  8853. pageScript.id = 'picviewer-page-script';
  8854.  
  8855. var pageScriptText=function(messageID){
  8856. var frameID=Math.random();
  8857. var frames={
  8858. top:window.top,
  8859. };
  8860.  
  8861. window.addEventListener('message',function(e){
  8862. var data=e.data;
  8863. if( !data || !data.messageID || data.messageID != messageID )return;//通信ID认证
  8864. var source=e.source;
  8865. if(source===window){//来自contentscript,发送出去,或者干嘛。
  8866. if(data.to){
  8867. data.from=frameID;
  8868. frames[data.to].postMessage(data,'*');
  8869. }else{
  8870. switch(data.command){
  8871. case 'getIframeObject':{
  8872. var frameWindow=frames[data.windowId];
  8873. var iframes=document.getElementsByTagName('iframe');
  8874. var iframe;
  8875. var targetIframe;
  8876. for(var i=iframes.length-1 ; i>=0 ; i--){
  8877. iframe=iframes[i];
  8878. if(iframe.contentWindow===frameWindow){
  8879. targetIframe=iframe;
  8880. break;
  8881. };
  8882. };
  8883. var cusEvent=document.createEvent('CustomEvent');
  8884. cusEvent.initCustomEvent('pv-getIframeObject',false,false,targetIframe);
  8885. document.dispatchEvent(cusEvent);
  8886. }break;
  8887. };
  8888. };
  8889.  
  8890. }else{//来自别的窗口的,contentscript可以直接接收,这里保存下来自的窗口的引用
  8891. frames[data.from]=source;
  8892. };
  8893. },true)
  8894. };
  8895.  
  8896. pageScript.textContent='(' + pageScriptText.toString() + ')('+ JSON.stringify(messageID) +')';
  8897. if(document.head)document.head.appendChild(pageScript);
  8898. }
  8899.  
  8900. function clickToOpen(data){
  8901. var preventDefault = matchedRule.clickToOpen.preventDefault;
  8902.  
  8903. function mouseout(){
  8904. document.removeEventListener('mouseout',mouseout,true);
  8905. document.removeEventListener('click',click,true);
  8906. if(data.imgPA && preventDefault){
  8907. data.imgPA.removeEventListener('click',clickA,false);
  8908. };
  8909. };
  8910.  
  8911. function click(e){
  8912. if(e.button!=0)return;
  8913. FloatBarC.prototype.open.call({
  8914. data:data,
  8915. },e,matchedRule.clickToOpen.type);
  8916. if(preventDefault){
  8917. e.preventDefault();
  8918. e.stopPropagation();
  8919. return false;
  8920. }
  8921. };
  8922.  
  8923. function clickA(e){//阻止a的默认行为
  8924. e.preventDefault();
  8925. };
  8926.  
  8927. document.addEventListener('click',click,true);
  8928.  
  8929. if(data.imgPA && preventDefault){
  8930. data.imgPA.addEventListener('click',clickA,false);
  8931. };
  8932.  
  8933. setTimeout(function(){//稍微延时。错开由于css hover样式发生的out;
  8934. document.addEventListener('mouseout',mouseout,true);
  8935. },100);
  8936.  
  8937. return function(){
  8938. mouseout();
  8939. };
  8940. }
  8941.  
  8942. var canclePreCTO,uniqueImgWin,centerInterval,removeUniqueWinTimer,globalFuncEnabled=false;
  8943. function checkGlobalKeydown(e){
  8944. return(!((!e.ctrlKey && prefs.floatBar.globalkeys.ctrl)||
  8945. (!e.altKey && prefs.floatBar.globalkeys.alt)||
  8946. (!e.shiftKey && prefs.floatBar.globalkeys.shift)||
  8947. (!e.metaKey && prefs.floatBar.globalkeys.command)||
  8948. (!prefs.floatBar.globalkeys.ctrl && !prefs.floatBar.globalkeys.alt && !prefs.floatBar.globalkeys.shift && !prefs.floatBar.globalkeys.command)));
  8949. }
  8950.  
  8951. function checkPreview(e){
  8952. let keyActive=(prefs.floatBar.globalkeys.type == "hold" && checkGlobalKeydown(e)) ||
  8953. (prefs.floatBar.globalkeys.type == "press" && globalFuncEnabled);
  8954. return prefs.floatBar.globalkeys.invertInitShow?!keyActive:keyActive;
  8955. }
  8956.  
  8957. //监听 mouseover
  8958. function globalMouseoverHandler(e){
  8959.  
  8960. if(galleryMode)return;//库模式全屏中......
  8961.  
  8962. var target = e.target;
  8963.  
  8964. if (!target || target.id=="pv-float-bar-container" ||
  8965. (target.className && target.className.indexOf &&
  8966. (/^pv\-/.test(target.className) ||
  8967. target.classList.contains("ks-imagezoom-lens")))) {
  8968. return;
  8969. }
  8970. if (target.nodeName=="PICTURE"){
  8971. target=target.querySelector("img");
  8972. }
  8973. if(e.type=="mousemove"){
  8974. if(target.nodeName != 'IMG' || !checkPreview(e) || (uniqueImgWin && !uniqueImgWin.removed)){
  8975. return;
  8976. }
  8977. }
  8978.  
  8979. // 扩展模式,检查前面一个是否为 img
  8980. if (target.nodeName != 'IMG' && matchedRule.rules.length>0 && matchedRule.ext) {
  8981. var _type = typeof matchedRule.ext;
  8982. if (_type == 'string') {
  8983. switch (matchedRule.ext) {
  8984. case 'previous':
  8985. target = target.previousElementSibling || target;
  8986. break;
  8987. case 'next':
  8988. target = target.nextElementSibling || target;
  8989. break;
  8990. case 'previous-2':
  8991. target = (target.previousElementSibling &&
  8992. target.previousElementSibling.previousElementSibling) || target;
  8993. break;
  8994. }
  8995. } else if (_type == 'function') {
  8996. try {
  8997. target = matchedRule.ext(target);
  8998. } catch(ex) {
  8999. throwErrorInfo(ex);
  9000. }
  9001.  
  9002. if (!target) return;
  9003. }
  9004. }
  9005. var result, hasBg = node => {
  9006. if(node.nodeName == "HTML" || node.nodeName == "#document"){
  9007. return false;
  9008. }
  9009. let nodeStyle = unsafeWindow.getComputedStyle(node);
  9010. return node && nodeStyle.backgroundImage && /^\s*url\(\s*['"]?\s*[^a\s]/i.test(nodeStyle.backgroundImage) && parseFloat(nodeStyle.width) > prefs.floatBar.minSizeLimit.w && parseFloat(nodeStyle.height) > prefs.floatBar.minSizeLimit.h;
  9011. };
  9012. if (target.nodeName != 'IMG'){
  9013. if(target.nodeName == "AREA")target=target.parentNode;
  9014. var targetBg;
  9015. var bgReg=/^\s*url\(\s*["']?(.+?)["']?\s*\)/i;
  9016. if(prefs.floatBar.listenBg && hasBg(target)){
  9017. targetBg = unsafeWindow.getComputedStyle(target).backgroundImage.replace(bgReg,"$1");
  9018. let src=targetBg,nsrc=src,noActual=true,type="scale";
  9019. let img={src:src};
  9020. result = {
  9021. src: nsrc,
  9022. type: type,
  9023. imgSrc: src,
  9024. noActual:noActual,
  9025. img: target
  9026. };
  9027. }else if(target.previousElementSibling && target.previousElementSibling.tagName=="IMG"){
  9028. if(unsafeWindow.getComputedStyle(target).position=="absolute" || target.nodeName == "MAP"){
  9029. target=target.previousElementSibling;
  9030. }
  9031. }else if(target.childNodes.length<=2 && target.querySelectorAll("img").length==1){
  9032. target=target.querySelector("img");
  9033. }else if(target.parentNode){
  9034. if(target.parentNode.nodeName=='IMG'){
  9035. target=target.parentNode;
  9036. }else if(prefs.floatBar.listenBg && hasBg(target.parentNode)){
  9037. target=target.parentNode;
  9038. targetBg=unsafeWindow.getComputedStyle(target).backgroundImage.replace(bgReg,"$1");
  9039. let src=targetBg,nsrc=src,noActual=true,type="scale";
  9040. let img={src:src};
  9041. result = {
  9042. src: nsrc,
  9043. type: type,
  9044. imgSrc: src,
  9045. noActual:noActual,
  9046. img: target
  9047. };
  9048. }/*else if(unsafeWindow.getComputedStyle(target).position=="absolute" || target.nodeName == "MAP"){
  9049. var imgChildren=[],availableImgs = [];
  9050. [].forEach.call(target.parentNode.querySelectorAll('img'),function(img){
  9051. var imgStyle=unsafeWindow.getComputedStyle(img);
  9052. if(imgStyle.display != "none"){
  9053. imgChildren.push(img);
  9054. if(imgStyle.width > 200 || imgStyle.position != "absolute"){
  9055. availableImgs.push(img);
  9056. }
  9057. }
  9058. });
  9059. if(imgChildren.length == 1){
  9060. target=imgChildren[0];
  9061. }else if(availableImgs.length == 1){
  9062. target=availableImgs[0];
  9063. }else if(imgChildren.length == 0 && unsafeWindow.getComputedStyle(target.parentNode).position=="absolute"){
  9064. imgChildren=[];availableImgs = [];
  9065. [].forEach.call(target.parentNode.parentNode.querySelectorAll('img'),function(img){
  9066. var imgStyle=unsafeWindow.getComputedStyle(img);
  9067. if(imgStyle.display != "none"){
  9068. imgChildren.push(img);
  9069. if(imgStyle.width > 200 || imgStyle.position != "absolute"){
  9070. availableImgs.push(img);
  9071. }
  9072. }
  9073. });
  9074. if(imgChildren.length == 1){
  9075. target=imgChildren[0];
  9076. }else if(availableImgs.length == 1){
  9077. target=availableImgs[0];
  9078. }
  9079. }
  9080. }*/
  9081. }
  9082. if(result && !/^data:/i.test(result.src)){
  9083. if(matchedRule.rules.length>0 && target.nodeName != 'IMG'){
  9084. let src=result.src,img={src:src},type,imgSrc=src;
  9085. try{
  9086. var newSrc=matchedRule.getImage(img);
  9087. if(newSrc && imgSrc!=newSrc) {
  9088. src=newSrc;
  9089. if (Array.isArray(src)) {
  9090. srcs = src;
  9091. src = srcs.shift();
  9092. }
  9093. type = 'rule';
  9094.  
  9095. if (matchedRule.description) {
  9096. var node = getElementMix(matchedRule.description, img);
  9097. if (node) {
  9098. description = node.getAttribute('title') || node.textContent;
  9099. }
  9100. }
  9101. result.src=src;
  9102. result.type=type;
  9103. result.noActual=false;
  9104. result.xhr=matchedRule.xhr;
  9105. result.description=description || '';
  9106. }
  9107. }catch(err){}
  9108. if(result.type!="rule"){
  9109. tprules._find(function(rule,index,array){
  9110. try{
  9111. src=rule.call(img);
  9112. if(src){
  9113. return true;
  9114. };
  9115. }catch(err){
  9116. }
  9117. });
  9118. if(src && src != imgSrc){
  9119. result.src=src;
  9120. result.type="tpRule";
  9121. result.noActual=false;
  9122. }
  9123. }
  9124. }
  9125. }
  9126. }
  9127. var checkUniqueImgWin=function(){
  9128. //metaKey altKey shiftKey ctrlKey
  9129. if(checkPreview(e)){
  9130. if(removeUniqueWinTimer)clearTimeout(removeUniqueWinTimer);
  9131. if(uniqueImgWin && !uniqueImgWin.removed) uniqueImgWin.remove();
  9132. new LoadingAnimC(result, 'popup', prefs.waitImgLoad, prefs.framesPicOpenInTopWindow);
  9133. return true;
  9134. }else {
  9135. if(uniqueImgWin && !uniqueImgWin.removed)
  9136. uniqueImgWin.imgWindow.style.pointerEvents = "auto";
  9137. return false;
  9138. }
  9139. };
  9140.  
  9141. if (!result && target.nodeName != 'IMG') {
  9142. if(target.nodeName == 'A' && /\.(jpg|png|jpeg|gif|webp)\b/.test(target.href)){
  9143. result = {
  9144. src: target.href,
  9145. type: "",
  9146. imgSrc: target.href,
  9147. noActual:true,
  9148. img: target
  9149. };
  9150. checkUniqueImgWin();
  9151. }else if(target.parentNode.nodeName == 'A' && /\.(jpg|png|jpeg|gif|webp)\b/.test(target.parentNode.href)){
  9152. result = {
  9153. src: target.parentNode.href,
  9154. type: "",
  9155. imgSrc: target.parentNode.href,
  9156. noActual:true,
  9157. img: target.parentNode
  9158. };
  9159. checkUniqueImgWin();
  9160. }
  9161. return;
  9162. }
  9163.  
  9164. if (!result) {
  9165. pretreatment(target)
  9166. result = findPic(target);
  9167. if(!result)return;
  9168. if(!(result.imgAS.w==result.imgCS.w && result.imgAS.h==result.imgCS.h)){//如果不是两者完全相等,那么被缩放了.
  9169. if(prefs.floatBar.sizeLimitOr){
  9170. if(result.imgCS.h <= prefs.floatBar.minSizeLimit.h && result.imgCS.w <= prefs.floatBar.minSizeLimit.w){//最小限定判断.
  9171. return;
  9172. }
  9173. }else{
  9174. if(result.imgCS.h <= prefs.floatBar.minSizeLimit.h || result.imgCS.w <= prefs.floatBar.minSizeLimit.w){//最小限定判断.
  9175. return;
  9176. }
  9177. }
  9178. }else{
  9179. if(prefs.floatBar.sizeLimitOr){
  9180. if(result.imgCS.w <= prefs.floatBar.forceShow.size.w && result.imgCS.h <= prefs.floatBar.forceShow.size.h){
  9181. return;
  9182. }
  9183. }else{
  9184. if(result.imgCS.w <= prefs.floatBar.forceShow.size.w || result.imgCS.h <= prefs.floatBar.forceShow.size.h){
  9185. return;
  9186. }
  9187. }
  9188. }
  9189. }
  9190.  
  9191. if(result){
  9192. debug(result);
  9193. if(!result.noActual){
  9194. if(!result.srcs){
  9195. result.srcs=[result.imgSrc];
  9196. }else{
  9197. if(result.imgSrc && result.srcs.join(" ").indexOf(result.imgSrc)==-1){
  9198. result.srcs.push(result.imgSrc);
  9199. }
  9200. }
  9201. }
  9202. if(!floatBar){
  9203. floatBar=new FloatBarC();
  9204. }
  9205. if(result.type=='rule' && matchedRule.clickToOpen && matchedRule.clickToOpen.enabled){
  9206. if(canclePreCTO){//取消上次的,防止一次点击打开多张图片
  9207. canclePreCTO();
  9208. }
  9209. canclePreCTO=clickToOpen(result);
  9210. }
  9211.  
  9212. if(!checkUniqueImgWin() && !e.altKey)
  9213. floatBar.start(result);//出现悬浮工具栏
  9214. };
  9215. }
  9216.  
  9217. function isKeyDownEffectiveTarget(target) {
  9218. var localName = target.localName;
  9219.  
  9220. // 确保光标不是定位在文字输入框或选择框
  9221. if (localName == 'textarea' || localName == 'input' || localName == 'select')
  9222. return false;
  9223.  
  9224. // 视频播放器
  9225. if (localName == 'object' || localName == 'embed')
  9226. return false;
  9227.  
  9228. // 百度贴吧回复输入的问题
  9229. if (target.getAttribute('contenteditable') == 'true')
  9230. return false;
  9231.  
  9232. return true;
  9233. }
  9234.  
  9235. function openGallery(){
  9236. if(!gallery){
  9237. gallery=new GalleryC();
  9238. gallery.data=[];
  9239. }
  9240. var allData=gallery.getAllValidImgs();
  9241. if(allData.length<1)return true;
  9242. gallery.data=allData;
  9243. gallery.load(gallery.data);
  9244. return gallery;
  9245. }
  9246.  
  9247. function keydown(event) {
  9248. var key = String.fromCharCode(event.keyCode).toLowerCase();
  9249. if(checkGlobalKeydown(event)){
  9250. if(key==prefs.floatBar.keys['gallery']){
  9251. openGallery();
  9252. event.stopPropagation();
  9253. event.preventDefault();
  9254. globalFuncEnabled = !globalFuncEnabled;
  9255. }else if((!gallery || (!gallery.shown && !gallery.minimized)) && prefs.floatBar.globalkeys.type == "press"){
  9256. globalFuncEnabled = !globalFuncEnabled;
  9257. }
  9258. return true;
  9259. }else{
  9260. if (event.ctrlKey || event.shiftKey || event.altKey || event.metaKey)
  9261. return false;
  9262.  
  9263. if (floatBar && floatBar.shown && isKeyDownEffectiveTarget(event.target)) {
  9264. Object.keys(prefs.floatBar.keys).some(function(action) {
  9265. if (action == 'enable') return;
  9266. if (key == prefs.floatBar.keys[action]) {
  9267. floatBar.open(null, action);
  9268. event.stopPropagation();
  9269. event.preventDefault();
  9270. return true;
  9271. }
  9272. })
  9273. }
  9274. }
  9275. }
  9276.  
  9277. window.addEventListener('message', handleMessage, true);
  9278.  
  9279. addPageScript();
  9280.  
  9281. document.addEventListener('mouseover', globalMouseoverHandler, true);
  9282. document.addEventListener('mousemove', globalMouseoverHandler, true);
  9283.  
  9284. document.addEventListener('mouseout',e=>{
  9285. if(uniqueImgWin && !uniqueImgWin.removed){
  9286. if(checkPreview(e)){
  9287. if(removeUniqueWinTimer)clearTimeout(removeUniqueWinTimer);
  9288. removeUniqueWinTimer = setTimeout(()=>{uniqueImgWin.remove()},100);
  9289. }else{
  9290. //if(e.target.tagName!="IMG")return;
  9291. uniqueImgWin.imgWindow.style.pointerEvents = "auto";
  9292. uniqueImgWin.focus();
  9293. }
  9294. }
  9295. }, true);
  9296.  
  9297. initLang();
  9298. var customLangOption={
  9299. 'auto': i18n("defaultLang")
  9300. };
  9301. for(let key in langList){
  9302. customLangOption[key]=langList[key];
  9303. }
  9304. GM_config.init({
  9305. id: 'pv-prefs',
  9306. title: GM_config.create('a', {
  9307. href: 'https://greasyfork.org/scripts/24204-picviewer-ce',
  9308. target: '_blank',
  9309. textContent: 'Picviewer CE+ '+i18n("config"),
  9310. title: i18n("openHomePage")
  9311. }),
  9312. isTabs: true,
  9313. skin: 'tab',
  9314. frameStyle: {
  9315. minWidth: "480px",
  9316. width: (((i18n("floatBar") + i18n("magnifier") + i18n("gallery") + i18n("imgWindow") + i18n("others")).visualLength("14px","arial,tahoma,myriad pro,sans-serif") + 250) || 480) + 'px',
  9317. zIndex:'2147483648',
  9318. },
  9319. css: [
  9320. "#pv-prefs input[type='text'] { width: 50px; } ",
  9321. "#pv-prefs input[type='number'] { width: 50px; } ",
  9322. "#pv-prefs .inline .config_var { margin-left: 6px; }",
  9323. "#pv-prefs label.size { width: 205px; }",
  9324. "#pv-prefs span.sep-x { margin-left: 0px !important; }",
  9325. "#pv-prefs label.sep-x { margin-right: 5px; }",
  9326. "#pv-prefs label.floatBar-key { margin-left: 20px; width: 100px; }",
  9327. "#pv-prefs input.color { width: 120px; }",
  9328. "#pv-prefs input.order { width: 250px; }",
  9329. "#pv-prefs .config_header>a { border-bottom: solid 2px; }",
  9330. "#pv-prefs .config_header>a:hover { color: #9f9f9f; }",,
  9331. ].join('\n'),
  9332. fields: {
  9333. // 浮动工具栏
  9334. 'floatBar.position': {
  9335. label: i18n("position"),
  9336. title: i18n("positionTips"),
  9337. type: 'select',
  9338. options: {
  9339. 'top left': i18n("topLeft"),
  9340. 'top right': i18n("topRight"),
  9341. 'bottom right': i18n("bottomRight"),
  9342. 'bottom left': i18n("bottomLeft"),
  9343. 'top center': i18n("topCenter"),
  9344. 'bottom center': i18n("bottomCenter"),
  9345. 'hide': i18n("hide")
  9346. },
  9347. "default": prefs.floatBar.position,
  9348. section: [i18n("floatBar")],
  9349. },
  9350. 'floatBar.showDelay': {
  9351. label: i18n("showDelay"),
  9352. type: 'int',
  9353. "default": prefs.floatBar.showDelay,
  9354. after: ' '+i18n("ms"),
  9355. },
  9356. 'floatBar.hideDelay': {
  9357. label: i18n("hideDelay"),
  9358. type: 'int',
  9359. className: 'hideDelay',
  9360. "default": prefs.floatBar.hideDelay,
  9361. after: ' '+i18n("ms")
  9362. },
  9363. 'floatBar.forceShow.size.w': {
  9364. label: i18n("forceShow"),
  9365. type: 'int',
  9366. className: 'size',
  9367. "default": prefs.floatBar.forceShow.size.w,
  9368. title: i18n("forceShowTip"),
  9369. line: 'start',
  9370. },
  9371. 'floatBar.forceShow.size.h': {
  9372. label: ' x ',
  9373. type: 'int',
  9374. className: 'sep-x',
  9375. after: ' '+i18n("px"),
  9376. "default": prefs.floatBar.forceShow.size.h,
  9377. line: 'end',
  9378. },
  9379. 'floatBar.minSizeLimit.w': {
  9380. label: i18n("minSizeLimit"),
  9381. type: 'int',
  9382. className: 'size',
  9383. "default": prefs.floatBar.minSizeLimit.w,
  9384. title: i18n("minSizeLimitTip"),
  9385. line: 'start',
  9386. },
  9387. 'floatBar.minSizeLimit.h': {
  9388. label: ' x ',
  9389. type: 'int',
  9390. className: 'sep-x',
  9391. after: ' '+i18n("px"),
  9392. "default": prefs.floatBar.minSizeLimit.h,
  9393. line: 'end',
  9394. },
  9395. 'floatBar.sizeLimitOr': {
  9396. label: i18n("sizeLimitOr"),
  9397. type: "checkbox",
  9398. "default": false
  9399. },
  9400. 'floatBar.butonOrder': {
  9401. label: i18n("butonOrder"),
  9402. type: 'text',
  9403. className: 'order',
  9404. "default": prefs.floatBar.butonOrder.join(', '),
  9405. },
  9406. 'floatBar.listenBg': {
  9407. label: i18n("listenBg"),
  9408. type: 'checkbox',
  9409. "default": prefs.floatBar.listenBg,
  9410. title: i18n("listenBgTip")
  9411. },
  9412. 'floatBar.globalkeys.ctrl': {
  9413. label: i18n("globalkeys"),
  9414. type: 'checkbox',
  9415. after: "CTRL +",
  9416. "default": true,
  9417. line: 'start'
  9418. },
  9419. 'floatBar.globalkeys.alt': {
  9420. after: "ALT +",
  9421. type: 'checkbox',
  9422. className: 'sep-x',
  9423. "default": false,
  9424. },
  9425. 'floatBar.globalkeys.shift': {
  9426. after: "SHIFT +",
  9427. type: 'checkbox',
  9428. className: 'sep-x',
  9429. "default": false,
  9430. },
  9431. 'floatBar.globalkeys.command': {
  9432. after: "COMMAND",
  9433. type: 'checkbox',
  9434. className: 'sep-x',
  9435. "default": false,
  9436. line: 'end',
  9437. },
  9438. 'floatBar.globalkeys.type': {
  9439. label: i18n("globalkeysType"),
  9440. type: 'select',
  9441. options: {
  9442. 'press': i18n("globalkeysPress"),
  9443. 'hold': i18n("globalkeysHold")
  9444. },
  9445. "default": prefs.floatBar.globalkeys.type
  9446. },
  9447. 'floatBar.globalkeys.invertInitShow': {
  9448. label: i18n("initShow"),
  9449. type: 'checkbox',
  9450. "default": prefs.floatBar.globalkeys.invertInitShow
  9451. },
  9452. // 按键
  9453. 'floatBar.keys.enable': {
  9454. label: i18n("keysEnable"),
  9455. type: 'checkbox',
  9456. "default": prefs.floatBar.keys.enable
  9457. },
  9458. 'floatBar.keys.actual': {
  9459. label: i18n("keysActual"),
  9460. type: 'text',
  9461. className: 'floatBar-key',
  9462. "default": prefs.floatBar.keys.actual,
  9463. title: i18n("keysActualTip")
  9464. },
  9465. /*'floatBar.keys.search': {
  9466. label: i18n("keysSearch"),
  9467. type: 'text',
  9468. className: 'floatBar-key',
  9469. "default": prefs.floatBar.keys.search,
  9470. title: i18n("keysSearchTip")
  9471. },*/
  9472. 'floatBar.keys.current': {
  9473. label: i18n("keysCurrent"),
  9474. type: 'text',
  9475. className: 'floatBar-key',
  9476. "default": prefs.floatBar.keys.current,
  9477. title: i18n("keysCurrentTip")
  9478. },
  9479. 'floatBar.keys.magnifier': {
  9480. label: i18n("keysMagnifier"),
  9481. type: 'text',
  9482. className: 'floatBar-key',
  9483. "default": prefs.floatBar.keys.magnifier,
  9484. title: i18n("keysMagnifierTip")
  9485. },
  9486. 'floatBar.keys.gallery': {
  9487. label: i18n("keysGallery"),
  9488. type: 'text',
  9489. className: 'floatBar-key',
  9490. "default": prefs.floatBar.keys.gallery,
  9491. title: i18n("keysGalleryTip")
  9492. },
  9493.  
  9494. // 放大镜
  9495. 'magnifier.radius': {
  9496. label: i18n("magnifierRadius"),
  9497. type: 'int',
  9498. "default": prefs.magnifier.radius,
  9499. section: [i18n("magnifier")],
  9500. after: ' '+i18n("px")
  9501. },
  9502. 'magnifier.wheelZoom.enabled': {
  9503. label: i18n("magnifierWheelZoomEnabled"),
  9504. type: 'checkbox',
  9505. "default": prefs.magnifier.wheelZoom.enabled,
  9506. },
  9507. 'magnifier.wheelZoom.range': {
  9508. label: i18n("magnifierWheelZoomRange"),
  9509. type: 'textarea',
  9510. "default": prefs.magnifier.wheelZoom.range.join(', '),
  9511. },
  9512.  
  9513. // 图库
  9514. 'gallery.defaultSizeLimit.w': {
  9515. label: i18n("defaultSizeLimit"),
  9516. type: 'int',
  9517. className: 'size',
  9518. section: [i18n("gallery")],
  9519. "default": prefs.gallery.defaultSizeLimit.w,
  9520. line: 'start',
  9521. },
  9522. 'gallery.defaultSizeLimit.h': {
  9523. label: ' x ',
  9524. type: 'int',
  9525. className: 'sep-x',
  9526. after: ' '+i18n("px"),
  9527. "default": prefs.gallery.defaultSizeLimit.h,
  9528. line: 'end',
  9529. },
  9530. 'gallery.fitToScreen': {
  9531. label: i18n("galleryFitToScreen"),
  9532. type: 'checkbox',
  9533. "default": prefs.gallery.fitToScreen,
  9534. title: i18n("galleryFitToScreenTip"),
  9535. line: 'start',
  9536. },
  9537. 'gallery.fitToScreenSmall': {
  9538. label: i18n("galleryFitToScreenSmall"),
  9539. type: 'checkbox',
  9540. "default": prefs.gallery.fitToScreenSmall,
  9541. line: 'end',
  9542. },
  9543. 'gallery.scrollEndToChange': {
  9544. label: i18n("galleryScrollEndToChange"),
  9545. type: 'checkbox',
  9546. "default": prefs.gallery.scrollEndToChange,
  9547. title: i18n("galleryScrollEndToChangeTip")
  9548. },
  9549. 'gallery.exportType': {
  9550. label: i18n("galleryExportType"),
  9551. type: 'select',
  9552. options: {
  9553. 'grid': i18n("grid"),
  9554. 'gridBig': i18n("gridBig"),
  9555. 'list': i18n("list")
  9556. },
  9557. "default": prefs.gallery.exportType,
  9558. },
  9559. 'gallery.loadMore': {
  9560. label: i18n("galleryAutoLoad"),
  9561. type: 'checkbox',
  9562. "default": prefs.gallery.loadMore
  9563. },
  9564. 'gallery.loadAll': {
  9565. label: i18n("galleryLoadAll"),
  9566. type: 'checkbox',
  9567. "default": prefs.gallery.loadAll,
  9568. title: i18n("galleryLoadAllTip")
  9569. },
  9570. 'gallery.viewmoreEndless': {
  9571. label: i18n("viewmoreEndless"),
  9572. type: 'checkbox',
  9573. "default": prefs.gallery.viewmoreEndless
  9574. },
  9575. 'gallery.downloadWithZip': {
  9576. label: i18n("galleryDownloadWithZip"),
  9577. type: 'checkbox',
  9578. "default": prefs.gallery.downloadWithZip
  9579. },
  9580. 'gallery.scaleSmallSize': {
  9581. label: i18n("galleryScaleSmallSize1"),
  9582. type: 'int',
  9583. "default": prefs.gallery.scaleSmallSize,
  9584. after: i18n("galleryScaleSmallSize2")
  9585. },
  9586. 'gallery.showSmallSize':{
  9587. label: i18n("galleryShowSmallSize"),
  9588. type: 'checkbox',
  9589. "default": prefs.gallery.showSmallSize
  9590. },
  9591. 'gallery.transition': {
  9592. label: i18n("galleryTransition"),
  9593. type: 'checkbox',
  9594. "default": prefs.gallery.transition
  9595. },
  9596. 'gallery.sidebarPosition': {
  9597. label: i18n("gallerySidebarPosition"),
  9598. type: 'select',
  9599. options: {
  9600. 'bottom': i18n("bottom"),
  9601. 'right': i18n("right"),
  9602. 'left': i18n("left"),
  9603. 'top': i18n("top")
  9604. },
  9605. "default": prefs.gallery.sidebarPosition,
  9606. line: 'start',
  9607. },
  9608. 'gallery.sidebarSize': {
  9609. label: i18n("gallerySidebarSize"),
  9610. type: 'int',
  9611. "default": prefs.gallery.sidebarSize,
  9612. title: i18n("gallerySidebarSizeTip"),
  9613. after: ' '+i18n("px"),
  9614. line: 'end',
  9615. },
  9616. 'gallery.max': {
  9617. label: i18n("galleryMax1"),
  9618. type: 'number',
  9619. "default": prefs.gallery.max,
  9620. after: i18n("galleryMax2")
  9621. },
  9622. 'gallery.autoZoom': {
  9623. label: i18n("galleryAutoZoom"),
  9624. type: 'checkbox',
  9625. "default": prefs.gallery.autoZoom,
  9626. title: i18n("galleryAutoZoomTip")
  9627. },
  9628. 'gallery.descriptionLength': {
  9629. label: i18n("galleryDescriptionLength1"),
  9630. type: 'int',
  9631. "default": prefs.gallery.descriptionLength,
  9632. after: i18n("galleryDescriptionLength2")
  9633. },
  9634. 'gallery.viewmoreLayout': {
  9635. label: i18n("galleryViewmoreLayout"),
  9636. type: 'select',
  9637. options: {
  9638. '0': "default",
  9639. '1': "flex-box"
  9640. },
  9641. "default": prefs.gallery.viewmoreLayout
  9642. },
  9643. 'gallery.autoOpenViewmore': {
  9644. label: i18n("autoOpenViewmore"),
  9645. type: 'checkbox',
  9646. "default": prefs.gallery.autoOpenViewmore
  9647. },
  9648. 'gallery.autoOpenSites': {
  9649. label: i18n("galleryAutoOpenSites"),
  9650. type: 'textarea',
  9651. "default": prefs.gallery.autoOpenSites
  9652. },
  9653. 'gallery.searchData': {
  9654. label: i18n("gallerySearchData"),
  9655. type: 'textarea',
  9656. "default": prefs.gallery.searchData
  9657. },
  9658. /*'gallery.editSite': {
  9659. label: i18n("galleryEditSite"),
  9660. type: 'select',
  9661. options: {
  9662. 'Pixlr': 'Pixlr',
  9663. 'Toolpic': 'Toolpic'
  9664. },
  9665. "default": prefs.gallery.editSite,
  9666. },*/
  9667.  
  9668. // 图片窗口
  9669. 'imgWindow.fitToScreen': {
  9670. label: i18n("imgWindowFitToScreen"),
  9671. type: 'checkbox',
  9672. "default": prefs.imgWindow.fitToScreen,
  9673. section: [i18n("imgWindow")],
  9674. title: i18n("imgWindowFitToScreenTip"),
  9675. },
  9676. 'imgWindow.suitLongImg': {
  9677. label: i18n("suitLongImg"),
  9678. type: 'checkbox',
  9679. "default": prefs.imgWindow.suitLongImg
  9680. },
  9681. 'imgWindow.close.defaultTool': {
  9682. label: i18n("imgWindowDefaultTool"),
  9683. type: 'select',
  9684. options: {
  9685. 'hand': i18n("hand"),
  9686. 'rotate': i18n("rotate"),
  9687. 'zoom': i18n("zoom"),
  9688. },
  9689. "default": prefs.imgWindow.close.defaultTool,
  9690. },
  9691. 'imgWindow.close.escKey': {
  9692. label: i18n("imgWindowEscKey"),
  9693. type: 'checkbox',
  9694. "default": prefs.imgWindow.close.escKey,
  9695. line: 'start',
  9696. },
  9697. 'imgWindow.close.dblClickImgWindow': {
  9698. label: i18n("imgWindowDblClickImgWindow"),
  9699. type: 'checkbox',
  9700. "default": prefs.imgWindow.close.dblClickImgWindow,
  9701. },
  9702. 'imgWindow.close.clickOutside': {
  9703. label: i18n("imgWindowClickOutside"),
  9704. type: 'select',
  9705. options: {
  9706. '': i18n("none"),
  9707. 'click': i18n("click"),
  9708. 'dblclick': i18n("dblclick"),
  9709. },
  9710. "default": prefs.imgWindow.close.clickOutside,
  9711. title: i18n("imgWindowClickOutsideTip"),
  9712. line: 'end',
  9713. },
  9714. 'imgWindow.overlayer.shown': {
  9715. label: i18n("imgWindowOverlayerShown"),
  9716. type: 'checkbox',
  9717. "default": prefs.imgWindow.overlayer.shown,
  9718. line: 'start',
  9719. },
  9720. 'imgWindow.overlayer.color': {
  9721. label: i18n("imgWindowOverlayerColor"),
  9722. type: 'text',
  9723. className: 'color',
  9724. "default": prefs.imgWindow.overlayer.color,
  9725. line: 'end'
  9726. },
  9727. 'imgWindow.shiftRotateStep': {
  9728. label: i18n("imgWindowShiftRotateStep1"),
  9729. type: 'int',
  9730. "default": prefs.imgWindow.shiftRotateStep,
  9731. after: i18n("imgWindowShiftRotateStep2")
  9732. },
  9733. 'imgWindow.zoom.mouseWheelZoom': {
  9734. label: i18n("imgWindowMouseWheelZoom"),
  9735. type: 'checkbox',
  9736. "default": prefs.imgWindow.zoom.mouseWheelZoom,
  9737. },
  9738. 'imgWindow.zoom.range': {
  9739. label: i18n("imgWindowZoomRange"),
  9740. type: 'textarea',
  9741. "default": prefs.imgWindow.zoom.range.join(', '),
  9742. title: i18n("imgWindowZoomRangeTip"),
  9743. attr: {
  9744. "spellcheck": "false"
  9745. }
  9746. },
  9747.  
  9748. // 其它
  9749. 'waitImgLoad': {
  9750. label: i18n("waitImgLoad"),
  9751. type: 'checkbox',
  9752. "default": prefs.waitImgLoad,
  9753. section: [i18n("others")],
  9754. title: i18n("waitImgLoadTip")
  9755. },
  9756. 'customLang': {
  9757. label: i18n("customLang"),
  9758. type: 'select',
  9759. options: customLangOption,
  9760. "default": prefs.customLang,
  9761. line: 'end',
  9762. },
  9763. 'debug': {
  9764. label: i18n("debug"),
  9765. type: 'checkbox',
  9766. "default": prefs.debug
  9767. },
  9768. 'customRules': {
  9769. label: i18n("customRules"),
  9770. type: 'textarea',
  9771. "default": prefs.customRules
  9772. }
  9773. /*'firstEngine': {
  9774. label: i18n("firstEngine"),
  9775. type: 'select',
  9776. options: {
  9777. "Tineye":"Tineye",
  9778. "Google":"Google",
  9779. "Baidu":"Baidu"
  9780. },
  9781. "default": prefs.firstEngine,
  9782. },*/
  9783. },
  9784. events: {
  9785. open: function(doc, win, frame) {
  9786. let saveBtn=doc.querySelector("#"+this.id+"_saveBtn");
  9787. let closeBtn=doc.querySelector("#"+this.id+"_closeBtn");
  9788. let resetLink=doc.querySelector("#"+this.id+"_resetLink");
  9789. saveBtn.textContent=i18n("saveBtn");
  9790. saveBtn.title=i18n("saveBtnTips");
  9791. closeBtn.textContent=i18n("closeBtn");
  9792. closeBtn.title=i18n("closeBtnTips");
  9793. resetLink.textContent=i18n("resetLink");
  9794. resetLink.title=i18n("resetLinkTips");
  9795. let searchData=doc.getElementById(this.id+"_field_gallery.searchData");
  9796. if(searchData && searchData.value==""){
  9797. searchData.value=defaultSearchData;
  9798. }
  9799. },
  9800. save: function() {
  9801. loadPrefs();
  9802. storage.setItem("customLang", prefs.customLang);
  9803. }
  9804. }
  9805. });
  9806.  
  9807.  
  9808. var hideIcon=false;
  9809. var hideIconStyle=document.createElement('style');
  9810. hideIconStyle.textContent=`#pv-float-bar-container{display:none!important}`;
  9811. _GM_registerMenuCommand(i18n("openConfig"), openPrefs);
  9812. _GM_registerMenuCommand(i18n("openGallery"), openGallery);
  9813. _GM_registerMenuCommand(i18n("toggleIcon"), ()=>{
  9814. hideIcon=!hideIcon;
  9815. if(hideIcon){
  9816. document.head.appendChild(hideIconStyle);
  9817. }else{
  9818. document.head.removeChild(hideIconStyle);
  9819. }
  9820. });
  9821.  
  9822. loadPrefs();
  9823.  
  9824. matchedRule = getMatchedRule();
  9825.  
  9826. if(prefs.gallery.autoOpenSites){
  9827. var sitesArr=prefs.gallery.autoOpenSites.split("\n");
  9828. for(var s=0;s<sitesArr.length;s++){
  9829. var siteReg=sitesArr[s].trim();
  9830. var autoViewMore=siteReg[0]=="@";
  9831. if(autoViewMore)siteReg=siteReg.substr(1);
  9832. if(new RegExp(siteReg).test(location.href)){
  9833. setTimeout(function(){
  9834. var gallery = openGallery();
  9835. if(autoViewMore)gallery.maximizeSidebar();
  9836. },2000);
  9837. break;
  9838. }
  9839. }
  9840. }
  9841.  
  9842. // 注册按键
  9843. if (prefs.floatBar.keys.enable) {
  9844. document.addEventListener('keydown', keydown, false);
  9845. }
  9846.  
  9847. function openPrefs() {
  9848. GM_config.open();
  9849. setTimeout(()=>{
  9850. if(GM_config.frame && GM_config.frame.style && GM_config.frame.style.display=="none"){
  9851. GM_config.frame.src="";
  9852. }
  9853. },500);
  9854. }
  9855.  
  9856. function loadPrefs() {
  9857. // 根据 GM_config 的 key 载入设置到 prefs
  9858. Object.keys(GM_config.fields).forEach(function(keyStr) {
  9859. var keys = keyStr.split('.');
  9860. var lastKey = keys.pop();
  9861.  
  9862. var lastPref = keys.reduce(function(previousValue, curKey) {
  9863. return previousValue[curKey];
  9864. }, prefs) || prefs;
  9865.  
  9866. var value = GM_config.get(keyStr);
  9867. if (typeof value != 'undefined') {
  9868. // 特殊的
  9869. if (keyStr == 'magnifier.wheelZoom.range' || keyStr == 'imgWindow.zoom.range') {
  9870. lastPref[lastKey] = value.split(/[,,]\s*/).map(function(s) { return parseFloat(s)});
  9871. } else if(keyStr == 'floatBar.butonOrder') {
  9872. lastPref[lastKey] = value.trim().split(/\s*[,,]\s*/);
  9873. } else {
  9874. lastPref[lastKey] = value;
  9875. }
  9876. }
  9877. });
  9878.  
  9879. debug = prefs.debug ? console.debug.bind(console) : function() {};
  9880. }
  9881.  
  9882. };
  9883.  
  9884. function drawTobase64(img){
  9885. let canvas = document.createElement('CANVAS');
  9886. canvas.width = img.width;
  9887. canvas.height = img.height;
  9888. let ctx = canvas.getContext('2d');
  9889. ctx.drawImage(img, 0, 0);
  9890. return canvas.toDataURL("image/png");
  9891. }
  9892.  
  9893. function init2(){
  9894. init(topObject,window,document,arrayFn,envir,storage,unsafeWindow);
  9895. };
  9896.  
  9897.  
  9898. //大致检测运行环境
  9899. var envir={
  9900. ie:typeof document.documentMode == 'number',
  9901. firefox:typeof XPCNativeWrapper == 'function',
  9902. opera:!!window.opera,
  9903. chrome:!!window.chrome,
  9904. };
  9905.  
  9906. //ie的话,不支持 < ie9的版本
  9907. if(envir.ie && document.documentMode < 9){
  9908. return;
  9909. };
  9910.  
  9911.  
  9912. var arrayFn=(function(){
  9913. //Array的某些方法对所有的类数组都有效,比如HTMLCollection,NodeList,DOMStringList.....
  9914.  
  9915. //添加一个当函数返回true时,返回[array[index],index],并且跳出循环的方法
  9916. //类似做到 for 循环,在满足条件的时候直接break跳出的效果。
  9917. if(typeof Array.prototype['_find']!='function'){
  9918. Object.defineProperty(Array.prototype,'_find',{
  9919. value:function(callback , thisArg){
  9920. if (this == null){
  9921. throw new TypeError( "this is null or not defined" );
  9922. };
  9923.  
  9924. if(typeof callback != 'function') {
  9925. throw new TypeError( callback + " is not a function" );
  9926. };
  9927.  
  9928. var i = 0,
  9929. l = this.length,
  9930. value,
  9931. hasOwnProperty=Object.prototype.hasOwnProperty;
  9932.  
  9933.  
  9934. while(i<l){
  9935. if(hasOwnProperty.call(this,i)){
  9936. value = this[i];
  9937. if(callback.call( thisArg, value, i, this )===true){
  9938. return [value,i,this];
  9939. };
  9940. };
  9941. i++;
  9942. };
  9943. },
  9944. writable:true,
  9945. enumerable:false,//与原生方法一样不可枚举,维护网页和谐。。。
  9946. configurable:true,
  9947. });
  9948. };
  9949.  
  9950. var arrayProto=Array.prototype;
  9951. return {
  9952. _find:arrayProto._find,
  9953. slice:arrayProto.slice,
  9954. forEach:arrayProto.forEach,
  9955. some:arrayProto.some,
  9956. every:arrayProto.every,
  9957. map:arrayProto.map,
  9958. filter:arrayProto.filter,
  9959. indexOf:arrayProto.indexOf,
  9960. lastIndexOf:arrayProto.lastIndexOf,
  9961. };
  9962.  
  9963. })();
  9964.  
  9965. var storage={
  9966. supportGM: typeof GM_getValue=='function' && typeof GM_getValue('a','b')!='undefined',//chrome的gm函数式空函数
  9967. mxAppStorage:(function(){//傲游扩展储存接口
  9968. try{
  9969. return window.external.mxGetRuntime().storage;
  9970. }catch(e){
  9971. };
  9972. })(),
  9973. operaUJSStorage:(function(){//opera userjs全局存储接口
  9974. try{
  9975. return window.opera.scriptStorage;
  9976. }catch(e){
  9977. };
  9978. })(),
  9979. setItem:function(key,value){
  9980. if(this.operaUJSStorage){
  9981. this.operaUJSStorage.setItem(key,value);
  9982. }else if(this.mxAppStorage){
  9983. this.mxAppStorage.setConfig(key,value);
  9984. }else if(this.supportGM){
  9985. GM_setValue(key,value);
  9986. }else if(window.localStorage){
  9987. window.localStorage.setItem(key,value);
  9988. };
  9989. },
  9990. getItem:function(key){
  9991. var value;
  9992. if(this.operaUJSStorage){
  9993. value=this.operaUJSStorage.getItem(key);
  9994. }else if(this.mxAppStorage){
  9995. value=this.mxAppStorage.getConfig(key);
  9996. }else if(this.supportGM){
  9997. value=GM_getValue(key);
  9998. }else if(window.localStorage){
  9999. value=window.localStorage.getItem(key);
  10000. };
  10001. return value;
  10002. },
  10003. };
  10004.  
  10005. function getUrl(url, callback, onError){
  10006. _GM_xmlhttpRequest({
  10007. method: 'GET',
  10008. url: url,
  10009. onload: callback,
  10010. onerror: onError
  10011. });
  10012. }
  10013.  
  10014. function setSearchState(words,imgCon){
  10015. if(words)console.info(words);
  10016. var searchState = (imgCon?imgCon:document).querySelector('.pv-pic-search-state');
  10017. if(searchState)searchState.innerHTML=createHTML(words);
  10018. }
  10019.  
  10020. var searchSort=["Tineye","Google","Baidu"];
  10021. function sortSearch(){
  10022. for(var i=0;i<searchSort.length;i++){
  10023. if(searchSort[i]==prefs.firstEngine){
  10024. searchSort.splice(i,1);
  10025. break;
  10026. }
  10027. }
  10028. searchSort.unshift(prefs.firstEngine);
  10029. }
  10030.  
  10031. function searchImgByImg(imgSrc, imgCon, callBack, onError, noneResult, searchFrom){
  10032. let srcs=[];
  10033. var searchBaidu=function(){
  10034. setSearchState(i18n("beginSearchImg","百度"),imgCon);
  10035. getUrl("http://image.baidu.com/n/same?queryImageUrl="+encodeURIComponent(imgSrc)+"&isguessword=1&rn=30&fr=pc&pn=0&sort=size", function(d){
  10036. let baiduJson;
  10037. try{
  10038. baiduJson=JSON.parse(d.responseText);
  10039. }catch(e){
  10040. setSearchState(i18n("findNoPic"),imgCon);
  10041. setTimeout(function(){
  10042. setSearchState("",imgCon);
  10043. },2000);
  10044. searchNext();
  10045. return;
  10046. }
  10047. if(baiduJson.data[0]){
  10048. srcs=[];
  10049. for(let imgData of baiduJson.data){
  10050. if(srcs.length>2)break;
  10051. srcs.push(imgData.objURL);
  10052. }
  10053. setSearchState(i18n("findOverBeginLoad",["百度",srcs.length]),imgCon);
  10054. callBackFun(srcs);
  10055. }else{
  10056. searchNext();
  10057. return;
  10058. }
  10059. }, onError);
  10060. };
  10061. var searchGoogle=function(){
  10062. setSearchState(i18n("beginSearchImg","Google"),imgCon);
  10063. getUrl("https://www.google.com/searchbyimage?safe=off&image_url="+encodeURIComponent(imgSrc), function(d){
  10064. let googleHtml=document.implementation.createHTMLDocument('');
  10065. googleHtml.documentElement.innerHTML = d.responseText;
  10066. let sizeUrl=googleHtml.querySelector("div.card-section>div>div>span.gl>a");
  10067. if(sizeUrl){
  10068. getUrl("https://www.google.com"+sizeUrl.getAttribute("href"), function(d){
  10069. googleHtml.documentElement.innerHTML = d.responseText;
  10070. let imgs=googleHtml.querySelectorAll("div.rg_meta");
  10071. if(imgs.length==0){searchNext();return;}
  10072. srcs=[];
  10073. for(var i=0;i<imgs.length;i++){
  10074. if(srcs.length>2)break;
  10075. let jsonData=JSON.parse(imgs[i].innerHTML);
  10076. srcs.push(jsonData.ou);
  10077. }
  10078. setSearchState(i18n("findOverBeginLoad",["Google",srcs.length]),imgCon);
  10079. callBackFun(srcs);
  10080. }, onError);
  10081. }else{
  10082. searchNext();
  10083. }
  10084. }, onError);
  10085. };
  10086. var searchTineye=function(){
  10087. setSearchState(i18n("beginSearchImg","Tineye"),imgCon);
  10088. getUrl("https://www.tineye.com/search?url="+encodeURIComponent(imgSrc)+"&sort=size", function(d){
  10089. let tineyeHtml=document.implementation.createHTMLDocument('');
  10090. tineyeHtml.documentElement.innerHTML = d.responseText;
  10091. let searchImg=tineyeHtml.querySelectorAll(".match-details>div.match:first-of-type>p.image-link:first-of-type>a");
  10092. if(searchImg.length>0){
  10093. srcs=[];
  10094. for(var i=0;i<searchImg.length;i++){
  10095. if(srcs.length>2)break;
  10096. srcs.push(searchImg[i].href);
  10097. }
  10098. setSearchState(i18n("findOverBeginLoad",["Tineye",srcs.length]),imgCon);
  10099. callBackFun(srcs);
  10100. }else{
  10101. searchNext();
  10102. }
  10103. }, onError);
  10104. };
  10105. var searchNext=function(){
  10106. searchFrom++;
  10107. if(searchFrom<=searchSort.length)switchSearch();
  10108. else{
  10109. if(noneResult)noneResult();
  10110. setSearchState(i18n("findNoPic"),imgCon);
  10111. setTimeout(function(){
  10112. setSearchState("",imgCon);
  10113. },2000);
  10114. }
  10115. };
  10116. var callBackFun=function(srcs){
  10117. callBack(srcs, searchFrom);
  10118. };
  10119. if(!searchFrom)searchFrom=1;
  10120. var switchSearch=function(){
  10121. switch(searchSort[searchFrom-1]){
  10122. case "Baidu":
  10123. searchBaidu();
  10124. break;
  10125. case "Google":
  10126. searchGoogle();
  10127. break;
  10128. case "Tineye":
  10129. searchTineye();
  10130. break;
  10131. default:
  10132. searchTineye();
  10133. break;
  10134. }
  10135. };
  10136. switchSearch();
  10137. }
  10138.  
  10139. init2();
  10140.  
  10141. })(this,window,document,(typeof unsafeWindow=='undefined'? window : unsafeWindow));