picViewer CE

NLF 的围观图修改版。

目前為 2014-09-28 提交的版本,檢視 最新版本

  1. // ==UserScript==
  2. // @name picViewer CE
  3. // @author NLF && ywzhaiqi
  4. // @description NLF 的围观图修改版。
  5. // @version 2014.9.28.1
  6. // version 4.2.6.1
  7. // @created 2011-6-15
  8. // @lastUpdated 2013-5-29
  9.  
  10. // @grant GM_getValue
  11. // @grant GM_setValue
  12. // @grant GM_openInTab
  13. // @run-at document-end
  14.  
  15. // @namespace http://userscripts.org/users/NLF
  16. // @homepage https://github.com/ywzhaiqi/userscript/tree/master/picviewerCE
  17. // homepage http://userscripts.org/scripts/show/105741
  18. // downloadURL https://userscripts.org/scripts/source/105741.user.js
  19. // updateURL https://userscripts.org/scripts/source/105741.meta.js
  20. // @include http*
  21. // @exclude http://www.toodledo.com/tasks/*
  22. // @exclude http*://maps.google.com*/*
  23. // ==/UserScript==
  24.  
  25. // 原脚本的 document-start 会在 chrome 下加载 2 次
  26.  
  27. ;(function(topObject,window,document,unsafeWindow){
  28. 'use strict';
  29.  
  30. function init(topObject,window,document,arrayFn,envir,storage,unsafeWindow){
  31. //一些设定。
  32. var prefs={
  33. floatBar:{//浮动工具栏相关设置.
  34. butonOrder:['actual','current','magnifier','gallery'],//按钮排列顺序'actual'(实际的图片),'current'(当前显示的图片),'magnifier'(放大镜观察),'gallery'(图集)
  35. showDelay:366,//浮动工具栏显示延时.单位(毫秒)
  36. hideDelay:566,//浮动工具栏隐藏延时.单位(毫秒)
  37. position:'top left',// 取值为: 'top left'(图片左上角) 或者 'top right'(图片右上角) 'bottom right'(图片右下角) 'bottom left'(图片左下角);
  38. offset:{//浮动工具栏偏移.单位(像素)
  39. x:-15,//x轴偏移(正值,向右偏移,负值向左)
  40. y:-15,//y轴偏移(正值,向下,负值向上)
  41. },
  42. forceShow:{//在没有被缩放的图片上,但是大小超过下面设定的尺寸时,强制显示浮动框.(以便进行旋转,放大,翻转等等操作)..
  43. enabled:true,//启用强制显示.
  44. size:{//图片尺寸.单位(像素);
  45. w:166,
  46. h:166,
  47. },
  48. },
  49. minSizeLimit:{//就算是图片被缩放了(看到的图片被设定了width或者height限定了大小,这种情况下),如果没有被缩放的原图片小于设定值,那么也不显示浮动工具栏.
  50. w:100,
  51. h:100,
  52. },
  53. },
  54.  
  55. magnifier:{//放大镜的设置.
  56. radius:77,//默认半径.单位(像素).
  57. wheelZoom:{//滚轮缩放.
  58. enabled:true,
  59. pauseFirst:true,//需要暂停(单击暂停)后,才能缩放.(推荐,否则因为放大镜会跟着鼠标,如果放大镜过大,那么会影响滚动.)..
  60. 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],//缩放的范围
  61. },
  62. },
  63.  
  64. gallery:{//图库相关设定
  65. fitToScreen:true,//图片适应屏幕(适应方式为contain,非cover).
  66. sidebarPosition:'bottom',//'top' 'right' 'bottom' 'left' 四个可能值
  67. sidebarSize: 120,//侧栏的高(如果是水平放置)或者宽(如果是垂直放置)
  68. transition:true,//大图片区的动画。
  69. preload:true,//对附近的图片进行预读。
  70. max:5,//最多预读多少张(前后各多少张)
  71.  
  72. autoScrollAndReload: false, // 最后一张图片时,滚动主窗口到最底部,然后自动重载库的图片。还有bug,有待进一步测试
  73. // 按键
  74. keysEnabled: true,
  75. keys: {
  76. actual: 'a', // 当出现悬浮条时按下 `a` 打开原图
  77. current: 'c',
  78. magnifier: 'm',
  79. gallery: 'g',
  80. }
  81. },
  82.  
  83. imgWindow:{//图片窗相关设置
  84. fitToScreen: false,//适应屏幕,并且水平垂直居中(适应方式为contain,非cover).
  85. syncSelectedTool:true,//同步当前选择的工具,如果开了多个图片窗口,其中修改一个会反映到其他的上面。
  86. defaultTool:'hand',//"hand","rotate","zoom";打开窗口的时候默认选择的工具
  87. close:{//关闭的方式
  88. escKey:true,//按esc键
  89. dblClickImgWindow:true,//双击图片窗口
  90. clickOutside:{//是否点击图片外部关闭
  91. enabled:false,
  92. trigger:'click',//'click'|'dblclick';点击或者双击
  93. },
  94. },
  95. overlayer:{//覆盖层.
  96. shown:false,//显示
  97. color:'rgba(0,0,0,0.8)',//颜色和不透明度设置.
  98. },
  99. shiftRotateStep:15,//旋转的时候,按住shift键时,旋转的步进.单位:度.
  100. zoom:{//滚轮缩放
  101. 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],//缩放比例.(不要出现负数,谢谢-_-!~)
  102. mouseWheelZoom:true,//是否允许使用滚轮缩放。
  103. },
  104. },
  105.  
  106. //等图片完全载入后,才开始执行弹出,放大等等操作,
  107. //按住ctrl键的时候,可以临时执行和这个设定相反的设定.
  108. waitImgLoad:true,
  109.  
  110. //框架里面的图片在顶层窗口展示出来,但是当frame与顶层窗口domain不一样的时候,可能导致图片被反盗链拦截,
  111. //按住shift键,可以临时执行和这个设定相反的设定
  112. framesPicOpenInTopWindow:true,
  113. };
  114.  
  115.  
  116. //各网站高级规则;
  117. var siteInfo=[
  118. {sitename: "google 图片搜索",
  119. //网址例子.(方便测试.查看.之类的)
  120. siteExample:"http://www.google.com.hk/search?q=firefox&tbm=isch",
  121. //是否启用
  122. enabled:true,
  123. //站点正则
  124. url:/https?:\/\/www.google(\.\w{1,3}){1,3}\/search\?.*&tbm=isch/,
  125. //鼠标左键点击直接打开..(这个只是当高级规则的getImage()返回图片的时候生效)
  126. // 无效?只有少数情况下有作用?
  127. clikToOpen:{
  128. enabled:true,
  129. preventDefault:true,//是否尝试阻止点击的默认行为(比如如果是你点的是一个链接,默认行为是打开这个链接,如果是true,js会尝试阻止链接的打开(如果想临时打开这个链接,请使用右键的打开命令))
  130. type:'actual',//默认的打开方式: 'actual'(弹出,原始图片) 'magnifier'(放大镜) 'current'(弹出,当前图片)
  131. },
  132. //获取图片实际地址的处理函数,
  133. //this 为当前鼠标悬浮图片的引用,
  134. //第一个参数和this相同,也是当前鼠标悬浮图片的引用,
  135. //第二个参数为包裹当前图片的第一个a元素(可能不存在).
  136. getImage:function(img,a){
  137. if(!a)return;
  138. if (a.href.match(/imgurl=(.*?\.\w{1,5})&/i)) {
  139. return decodeURIComponent(RegExp.$1);
  140. }
  141. },
  142.  
  143. // ====== 我新增的 ======
  144. // 自定义样式
  145. css: '',
  146. // 排除的图片正则
  147. // exclude: /weixin_code\.png$/i,
  148. },
  149. {sitename: "Bing 图片搜索",
  150. siteExample:"http://cn.bing.com/images/search?q=%E7%BE%8E%E5%A5%B3",
  151. enabled:true,
  152. url: /^https?:\/\/[^.]*\.bing\.com\/images\//i,
  153. getImage:function(img, a){
  154. if (!a) return;
  155. var oldsrc=this.src;
  156. var $ = /,imgurl:"([^"]+)/.exec(a.getAttribute('m'));
  157. var newsrc= $ ? $[1] : '';
  158. if(newsrc!=oldsrc)return newsrc;
  159. }
  160. },
  161. // 百度自身的全屏查看方式更加好,跟这个脚本的库查看类似。
  162. {sitename: "百度图片搜索",
  163. siteExample: "http://image.baidu.com/i?ie=utf-8&word=%E9%A3%8E%E6%99%AF&oq=%E9%A3%8E%E6%99",
  164. enabled: true,
  165. url: /^https?:\/\/image\.baidu\.com\/.*&word=/i,
  166. getImage: function(img, a) {
  167. if (!a) return;
  168. var reg = /&objurl=(http.*?\.(?:jpg|jpeg|png|gif|bmp))/i;
  169. if (a.href.match(reg)) {
  170. return decodeURIComponent(RegExp.$1);
  171. }
  172. }
  173. },
  174. {sitename: "百度图片 - channel/detail",
  175. siteExample: "http://image.baidu.com/channel?c=%E7%BE%8E%E5%A5%B3&t=%E5%85%A8%E9%83%A8&s=0",
  176. enabled: true,
  177. url: /^https?:\/\/image\.baidu\.com\/(?:channel|detail)/i,
  178. getImage: function(img, a) {
  179. var src = this.src,
  180. ret = src;
  181. var pic = new RegExp("(hiphotos|imgsrc)\\.baidu\\.com/(.+?)/.+?([0-9a-f]{40})");
  182. ret = src.replace(pic, '$1.baidu.com/$2/pic/item/$3');
  183.  
  184. if (ret != src) {
  185. return ret;
  186. }
  187. }
  188. },
  189. // 自带的更好
  190. // {sitename:"百度图片(详细页面)",
  191. // enabled:true,
  192. // url:/^http:\/\/image\.baidu\.com\/detail\//i,
  193. // getImage:function(){
  194. // var src=this.src;
  195. // var ret=src.replace(/\/w%3D230\/sign=[^\/]+\//i, '/pic/item/');
  196. // if(src==ret)return;//非缩略图
  197. // return ret;
  198. // },
  199. // },
  200. {sitename:"百度贴吧",
  201. enabled:true,
  202. url:/^https?:\/\/tieba\.baidu\.[^\/]+\//i,
  203. getImage:function(img){
  204. var src=img.src;
  205. var reg=/^(http:\/\/imgsrc\.baidu\.com\/forum\/)ab(pic\/item\/[\w.]+)/i ;
  206. var result=src.match(reg);
  207. //帖子列表页面
  208. if(result){//小图的时候
  209. return result[1]+result[2];
  210. }else{//小图点击之后的较大图,或者帖子内容页面的图片。
  211. var prefix = 'http://imgsrc.baidu.com/forum/pic/item/';
  212. var reg2 = /\/sign=\w+\/([\w.]+)$/;
  213. var sign = src.match(reg2);
  214. return sign ? prefix + sign[1] : null;
  215. };
  216. },
  217. },
  218. {sitename:"豆瓣",
  219. siteExample:"http://movie.douban.com/photos/photo/1000656155/",
  220. enabled:true,
  221. url:/^https?:\/\/[^.]*\.douban\.com/i,
  222. getImage:function(){
  223. var oldsrc = this.src,
  224. newsrc = oldsrc;
  225. var pic = /\/view\/photo\/(?:photo|albumcover|albumicon|thumb)\/public\//i;
  226. var movieCover = /\/view\/movie_poster_cover\/[si]pst\/public\//i;
  227. var bookCover = /\/view\/ark_article_cover\/cut\/public\//i;
  228. var spic = /(img\d.douban.com)\/spic\//i
  229.  
  230. if (pic.test(oldsrc)) {
  231. newsrc = oldsrc.replace(pic, '/view/photo/raw/public/');
  232. } else if (movieCover.test(oldsrc)) {
  233. newsrc = oldsrc.replace(movieCover, '/view/photo/raw/public/');
  234. } else if (bookCover.test(oldsrc)) {
  235. newsrc = oldsrc.replace(bookCover, '/view/ark_article_cover/retina/public/');
  236. } else if (spic.test(oldsrc)) {
  237. newsrc = oldsrc.replace(spic, '$1/mpic/');
  238. }
  239.  
  240. return newsrc == oldsrc ? null : newsrc;
  241. }
  242. },
  243. {sitename:"新浪微博",
  244. siteExample:"http://weibo.com/pub/?source=toptray",
  245. enabled:true,
  246. url:/^https?:\/\/(?:[^.]+\.)*weibo\.com/i,
  247. getImage:function(img){
  248. var oldsrc=this.src;
  249. var pic=/(\.sinaimg\.cn\/)(?:bmiddle|thumbnail)/i;//微博内容图片.
  250. var head=/(\.sinaimg\.cn\/\d+)\/50\//i;//头像.
  251. var photoList=/\.sinaimg\.cn\/thumb150\/\w+/i//相册
  252. var newsrc;
  253. if(pic.test(oldsrc)){
  254. newsrc=oldsrc.replace(pic,'$1large');
  255. return newsrc==oldsrc? '' : newsrc;
  256. }else if(head.test(oldsrc)){
  257. newsrc=oldsrc.replace(head,'$1/180/');
  258. return newsrc==oldsrc? '' : newsrc;
  259. }else if(photoList.test(oldsrc)){
  260. newsrc=oldsrc.replace('/thumb150/','/mw690/');
  261. return newsrc==oldsrc? '' : newsrc;
  262. };
  263. },
  264. },
  265. {sitename:"腾讯微博",
  266. siteExample:"http://t.qq.com/p/news",
  267. enabled:true,
  268. url:/^http:\/\/[^\/]*t\.qq\.com\//i,
  269. getImage:function(img){
  270. var pic=/(\.qpic\.cn\/mblogpic\/\w+)\/\d+/i;//图片
  271. var head=/(\.qlogo\.cn\/mbloghead\/\w+)\/\d+/i;//头像.
  272. var oldsrc=this.src;
  273. var newsrc;
  274. if(pic.test(oldsrc)){
  275. newsrc=oldsrc.replace(pic,'$1/2000');
  276. return newsrc==oldsrc? '' : newsrc;;
  277. }else if(head.test(oldsrc)){
  278. newsrc=oldsrc.replace(head,'$1/0');
  279. return newsrc==oldsrc? '' : newsrc;;
  280. };
  281. },
  282. },
  283. {sitename:"淘宝搜索",
  284. enabled:true,
  285. url:/^http:\/\/[^\.]+\.taobao\.com\//i,
  286. getImage:function(){
  287. var src = this.src;
  288. var ret = src.replace(new RegExp("((?:img\\d\\d\\.taobaocdn|g(?:[^.]*\\.?){1,2}?\\.alicdn)\\.com/)(?:img/|tps/http:\\//img\\d\\d+\\.taobaocdn\\.com/)?((?:imgextra|bao/uploaded)/i\\d+/[^!]+![^.]+\\.[^_]+)_.+", 'i'),
  289. '$1/$2');
  290. if (ret != src) return ret;
  291. },
  292. },
  293. {sitename: "deviantart",
  294. siteExample: "http://www.deviantart.com",
  295. enabled:true,
  296. url:/^https?:\/\/[^.]*\.deviantart\.com/i,
  297. getImage:function(){
  298. var oldsrc=this.src;
  299. var newsrc=oldsrc.replace(/(http:\/\/[^\/]+\/fs\d+\/)200H\/(.*)/i,'$1$2');
  300. return newsrc==oldsrc? '' : newsrc;
  301. },
  302. },
  303. {sitename: '花瓣网',
  304. enabled: true,
  305. url: /^https?:\/\/huaban\.com\//i,
  306. getImage: function() {
  307. var pic = /(.*img.hb.aicdn.com\/.*)_fw236$/i
  308. if (this.src.match(pic)) {
  309. return RegExp.$1 + '_fw658';
  310. }
  311. },
  312. css: '.pin a.img .cover { display: none; }',
  313. exclude: /weixin_code\.png$/i,
  314. },
  315. // 其它
  316. {sitename:"wiki百科",
  317. enabled:true,
  318. url:/^http:\/\/[^.]+.wikipedia.org\/wiki\/\w+/i,
  319. getImage:function(){
  320. var src=this.src;
  321. var ret=src.replace('/thumb/','/');
  322. if(src==ret)return;//非缩略图
  323. return (ret.match(/(https?:\/\/.*)\/\d+px-.*/) || [])[1];
  324. },
  325. },
  326. {sitename: "cnbeta",
  327. enabled: true,
  328. url: /^https?:\/\/www.cnbeta.com\//i,
  329. getImage: function() {
  330. var oldsrc = this.src,
  331. newsrc = oldsrc;
  332. // http://static.cnbetacdn.com/newsimg/2014/0922/19_1411376098.png_180x132.png
  333. if (oldsrc.match(/(static.cnbetacdn.com\/.+)_\d+x\d+\.\w{2,4}$/)) {
  334. newsrc = 'http://' + RegExp.$1;
  335. }
  336.  
  337. return newsrc == oldsrc ? null : newsrc;
  338. }
  339. },
  340. {sitename:"沪江碎碎",
  341. enabled:true,
  342. url:/^https?:\/\/([^.]+\.)*(?:yeshj\.com|hjenglish\.com|hujiang\.com)/i,
  343. getImage:function(img){
  344. var oldsrc=this.src;
  345. var reg=/^(https?:\/\/(?:[^.]+\.)*hjfile.cn\/.+)(_(?:s|m))(\.\w+)$/i;
  346. if(reg.test(oldsrc)){
  347. return oldsrc.replace(reg,'$1$3');
  348. };
  349. },
  350. },
  351. {sitename: '大众点评',
  352. siteExample: 'http://www.dianping.com/shop/17873296/photos',
  353. url: /^https?:\/\/www.dianping.com\/shop/i,
  354. getImage: function() {
  355. var oldsrc = this.src,
  356. newsrc;
  357. var pic = /(.+?dpfile\.com\/.+)\(240c180\)\/(thumb\..+)/;
  358. newsrc = oldsrc.replace(pic, '$1(700x700)/$2');
  359.  
  360. return newsrc == oldsrc ? null : newsrc;
  361. }
  362. },
  363.  
  364. // ------------------------- 视频 --------------------------------
  365. {sitename: "人人影视",
  366. enabled: true,
  367. url: /^http:\/\/www\.yyets\.com\//i,
  368. getImage: function() {
  369. var src = this.src;
  370. var ret = src.replace(new RegExp('(res\\.yyets\\.com/ftp/(?:attachment/)?\\d+/\\d+)/[ms]_(.*)', 'i'), '$1/$2');
  371. if (src == ret) return; //非缩略图
  372. return ret;
  373. },
  374. },
  375. {sitename: 'trakt.tv',
  376. url: /^http:\/\/trakt\.tv\//i,
  377. siteExample: 'http://trakt.tv/shows',
  378. getImage: function() {
  379. var oldsrc = this.src;
  380. if (oldsrc.match(/(.*\/images\/posters\/\d+)-(?:300|138)\.jpg\?(\d+)$/)) {
  381. return RegExp.$1 + '.jpg?' + RegExp.$2;
  382. }
  383. }
  384. },
  385.  
  386. // 游戏
  387. {sitename:"178.com",
  388. enabled:true,
  389. url:/^https?:\/\/(?:\w+\.)+178\.com\//i,
  390. clikToOpen:{
  391. enabled:true,
  392. preventDefault:true,
  393. type:'actual',
  394. },
  395. getImage:function(img,a){
  396. if(!a)return;
  397. var reg=/^https?:\/\/(?:\w+\.)+178\.com\/.+?(https?:\/\/img\d*.178.com\/[^.]+\.(?:jpg|jpeg|png|gif|bmp))/i;
  398. return (a.href.match(reg) || [])[1];
  399. },
  400. },
  401. {sitename: "天极网",
  402. url: /^http:\/\/game\.yesky\.com\//i,
  403. enabled: true,
  404. siteExample: "http://game.yesky.com/tupian/165/37968665.shtml",
  405. getImage: function() {
  406. var src = this.src;
  407. var ret = src.replace(/_\d+x\d+\.([a-z]+)$/i, '.$1');
  408. if (ret!=src) return ret;
  409. }
  410. },
  411. {sitename: "超级玩家",
  412. url: /^http:\/\/dota2\.sgamer\.com\/albums\//i,
  413. enabled: true,
  414. siteExample: "http://dota2.sgamer.com/albums/201407/8263_330866.html",
  415. getImage: function() {
  416. var src = this.src;
  417. var ret = src.replace(/\/s([^\.\/]+\.[a-z]+$)/i, '/$1');
  418. if (ret!=src) return ret;
  419. }
  420. },
  421. // 漫画站
  422. {sitename: "nhentai",
  423. url: /^http:\/\/nhentai\.net\/g\/\d+\//i,
  424. enabled: true,
  425. siteExample: "http://nhentai.net/g/113475/",
  426. getImage: function() {
  427. var src = this.src;
  428. var ret = src.replace(/\/(\d+)t(\.[a-z]+)$/i, '/$1$2');
  429. if (ret!=src) return ret;
  430. }
  431. },
  432. // 论坛
  433. {sitename:"极限主题社区",
  434. enabled:true,
  435. url:/^https?:\/\/bbs\.themex\.net\/.+/i,
  436. clikToOpen:{
  437. enabled:true,
  438. preventDefault:true,
  439. type:'actual',
  440. },
  441. getImage:function(){
  442. var reg=/^(https?:\/\/bbs\.themex\.net\/attachment\.php\?.+)&thumb=1(.+)/i;
  443. var src=this.src;
  444. var ret=src.replace(reg,'$1$2');
  445. return ret!=src? ret : '';
  446. },
  447. },
  448. // {sitename:"opera官方论坛",
  449. // siteExample:"http://bbs.operachina.com",
  450. // enabled:true,
  451. // url:/^http:\/\/bbs\.operachina\.com/i,
  452. // getImage:function(){
  453. // var src=this.src;
  454. // if(/file.php\?id=\d+$/i.test(src)){
  455. // return src+'&mode=view';
  456. // };
  457. // },
  458. // },
  459.  
  460. // ------------------------- 特殊的需要修正 --------------------------------
  461. {sitename: 'github 修正',
  462. url: /^https?:\/\/github\.com\//i,
  463. clikToOpen: {
  464. enabled: true,
  465. preventDefault: true,
  466. type: 'actual',
  467. },
  468. getImage: function() {
  469. return this.src;
  470. }
  471. },
  472.  
  473. // ------------------------- 需要 xhr 获取的 --------------------------------
  474. // 有些页面不行,需要 xhr 获取
  475. {sitename:"pixiv",
  476. enabled:true,
  477. url:/^http:\/\/www\.pixiv\.net/i,
  478. getImage:function(img){
  479. var oldsrc = this.src,
  480. newsrc = oldsrc;
  481. var reg = /(pixiv.net\/img\d+\/img\/.+\/\d+)_[ms]\.(\w{2,5})$/i;
  482. if (reg.test(oldsrc)) {
  483. newsrc = oldsrc.replace(reg, '$1.$2');
  484. }
  485. // 这里的链接需要 xhr 获取?
  486. // http://www.pixiv.net/member_illust.php?id=341433
  487.  
  488. return newsrc == oldsrc ? null : newsrc;
  489. },
  490. },
  491. // {sitename: "Google plus",
  492. // enabled: true,
  493. // url: /^https?:\/\/plus\.google\.com/i,
  494. // siteExample: "https://plus.google.com/u/0/+LynxEvil/posts/dpJTXa2Lgwf",
  495. // getImage: function (img) {
  496. // //lh5.googleusercontent.com/-wmoUHKSaEkg/Ue3kP5ZkfQI/AAAAAAAAXkY/BcJoyTtZbP8/s405-p/2.jpg
  497. // //lh6.googleusercontent.com/-wmoUHKSaEkg/Ue3kP5ZkfQI/AAAAAAAAXkY/BcJoyTtZbP8/w334-h413-no/2.jpg
  498.  
  499. // //lh3.googleusercontent.com/-sLCbDrGjLug/Ue3kLooLKZI/AAAAAAAARFY/KRG3uRMWiBI/s202-p/a.jpg
  500. // //lh6.googleusercontent.com/-sLCbDrGjLug/Ue3kLooLKZI/AAAAAAAAXkk/1PKW5XBQBzM/w325-h415-no/a.jpg
  501. // },
  502. // },
  503. // {sitename: '优酷电视剧',
  504. // siteExample: 'http://www.youku.com/v_olist/c_97.html',
  505. // url: /^https?:\/\/www.youku.com\/v_olist\//,
  506. // getImage: function() {
  507. // // {"r":"www\\.youku\\.com\\/show_page\\/id_.*\\.html","q":".baseinfo > .thumb > img"}
  508. // }
  509. // },
  510. ];
  511.  
  512. //通配型规则,无视站点.
  513. var tprules=[
  514. function(img,a){//解决新的dz论坛的原图获取方式.
  515. var reg=/(.+\/attachments?\/.+)\.thumb\.\w{2,5}$/i;
  516. var oldsrc=this.src;
  517. var newsrc=oldsrc.replace(reg,'$1');
  518. if(oldsrc!=newsrc)return newsrc;
  519. },
  520. ];
  521.  
  522.  
  523. //图标
  524. prefs.icons={
  525. actual:'',
  526. current:'',
  527. magnifier:'',
  528. gallery:'',
  529.  
  530.  
  531. retry:'',
  532. loading:'',
  533. loadingCancle:'',
  534.  
  535. hand:'',
  536. rotate:'',
  537. zoom:'',
  538. flipVertical:'',
  539. flipHorizontal:'',
  540. close:'',
  541. rotateIndicatorBG:'',
  542. rotateIndicatorPointer:'',
  543.  
  544. arrowTop:'',
  545. arrowBottom:'',
  546. arrowLeft:'',
  547. arrowRight:'',
  548.  
  549. fivePointedStar:'',
  550.  
  551. brokenImg:'',
  552. brokenImg_small:'',
  553. };
  554.  
  555. //分享api;有需求的照着添加
  556. //api项,请返回给一个{url:url,wSize:{w:,h:}},脚本会自动调用window.open打开,如果不返回任何的话,脚本将不做任何其他事情。
  557. //api的参数
  558. /*{
  559. title
  560. pic
  561. url
  562. } */
  563. prefs.share={
  564. weibo:{
  565. disabled:false,
  566. name:'新浪微博',
  567. icon:'',
  568. api:function(args){
  569. var url='http://service.weibo.com/share/share.php?'+
  570. 'title='+args.title+
  571. '&url='+args.url+
  572. '&pic='+args.pic;
  573. return {
  574. url:url,
  575. wSize:{
  576. h:500,
  577. w:620,
  578. },
  579. };
  580. },
  581. },
  582. t:{
  583. name:'腾讯微博',
  584. icon:'',
  585. api:function(args){
  586. var url='http://v.t.qq.com/share/share.php?'+
  587. 'title='+args.title+
  588. '&url='+args.url+
  589. '&pic='+args.pic;
  590. return {
  591. url:url,
  592. wSize:{
  593. h:500,
  594. w:620,
  595. },
  596. };
  597. },
  598. },
  599. qZone:{
  600. name:'QQ空间',
  601. icon:'',
  602. api:function(args){
  603. var url='http://sns.qzone.qq.com/cgi-bin/qzshare/cgi_qzshare_onekey?'+
  604. 'title='+args.title+
  605. '&pics='+args.pic+
  606. '&url='+args.url;
  607. return {
  608. url:url,
  609. wSize:{
  610. h:650,
  611. w:620,
  612. },
  613. };
  614. },
  615. },
  616. fanfou:{
  617. name:'饭否',
  618. icon:'',
  619. api:function(args){
  620. var url='http://fanfou.com/sharer/image?'+
  621. 'u='+args.url+
  622. '&t='+args.title+
  623. '&img_src='+args.pic;
  624. return{
  625. url:url,
  626. wSize:{
  627. h:550,
  628. w:650,
  629. },
  630. };
  631. },
  632. },
  633. tieba:{
  634. name:'百度贴吧',
  635. icon:'',
  636. api:function(args){
  637. var url = 'http://tieba.baidu.com/f/commit/share/openShareApi?'+
  638. 'title='+args.title+
  639. '&url='+args.url+
  640. '&pic='+args.pic;
  641. return {
  642. url:url,
  643. wSize:{
  644. h:600,
  645. w:630,
  646. },
  647. };
  648. },
  649. },
  650. renren:{
  651. name:'人人网',
  652. icon:'',
  653. api:function(args){
  654. var url='http://widget.renren.com/dialog/share?'+
  655. 'link='+args.url+
  656. '&title='+args.title+
  657. '&pic='+args.pic;
  658. return {
  659. url:url,
  660. wSize:{
  661. h:600,
  662. w:650,
  663. },
  664. };
  665. },
  666. },
  667. douban:{
  668. name:'豆瓣',
  669. icon:'',
  670. api:function(args){
  671. var url='http://shuo.douban.com/%21service/share?'+
  672. 'href='+args.url+
  673. '&name='+args.title+
  674. '&image='+args.pic;
  675. return {
  676. url:url,
  677. wSize:{
  678. h:350,
  679. w:600,
  680. },
  681. };
  682. },
  683. },
  684. };
  685.  
  686.  
  687. function get(url, callback) {
  688. var xhr = new XMLHttpRequest();
  689. xhr.open('GET', url);
  690. xhr.overrideMimeType('text/html; charset=' + document.characterSet);
  691. xhr.onload = function() {
  692. callback(xhr.responseText);
  693. };
  694. xhr.send(null);
  695. }
  696.  
  697. function launchFullScreen(element) {
  698. if (element.requestFullscreen) {
  699. element.requestFullscreen();
  700. } else if (element.msRequestFullscreen) {
  701. element.msRequestFullscreen();
  702. } else if (element.mozRequestFullScreen) {
  703. element.mozRequestFullScreen();
  704. } else if (element.webkitRequestFullscreen) {
  705. element.webkitRequestFullscreen(Element.ALLOW_KEYBOARD_INPUT);
  706. }
  707. }
  708.  
  709. function cancelFullScreen() {
  710. if (document.exitFullscreen) {
  711. document.exitFullscreen();
  712. } else if (document.msExitFullscreen) {
  713. document.msExitFullscreen();
  714. } else if (document.mozCancelFullScreen) {
  715. document.mozCancelFullScreen();
  716. } else if (document.webkitExitFullscreen) {
  717. document.webkitExitFullscreen();
  718. }
  719. }
  720.  
  721. //获取位置
  722. function getContentClientRect(target){
  723. var rect=target.getBoundingClientRect();
  724. var compStyle=getComputedStyle(target);
  725. var pFloat=parseFloat;
  726. var top=rect.top + pFloat(compStyle.paddingTop) + pFloat(compStyle.borderTopWidth);
  727. var right=rect.right - pFloat(compStyle.paddingRight) - pFloat(compStyle.borderRightWidth);
  728. var bottom=rect.bottom - pFloat(compStyle.paddingBottom) - pFloat(compStyle.borderBottomWidth);
  729. var left=rect.left + pFloat(compStyle.paddingLeft) + pFloat(compStyle.borderLeftWidth);
  730. return {
  731. top:top,
  732. right:right,
  733. bottom:bottom,
  734. left:left,
  735. width:right-left,
  736. height:bottom-top,
  737. };
  738. };
  739.  
  740. //获取窗口大小.
  741. function getWindowSize(){
  742. /*
  743. //包含滚动条
  744. return {
  745. h:window.innerHeight,
  746. w:window.innerWidth,
  747. };
  748. */
  749.  
  750. //去除滚动条的窗口大小
  751. var de=document.documentElement;
  752. var body=document.body;
  753. var backCompat=document.compatMode=='BackCompat';
  754. return {
  755. h:backCompat? body.clientHeight : de.clientHeight,
  756. w:backCompat? body.clientWidth : de.clientWidth,
  757. };
  758.  
  759. };
  760.  
  761. //获取已滚动的距离
  762. function getScrolled(container){
  763. if(container){
  764. return {
  765. x:container.scrollLeft,
  766. y:container.scrollTop,
  767. };
  768. };
  769. return {
  770. x:'scrollX' in window ? window.scrollX : ('pageXOffset' in window ? window.pageXOffset : document.documentElement.scrollLeft || document.body.scrollLeft),
  771. y:'scrollY' in window ? window.scrollY : ('pageYOffset' in window ? window.pageYOffset : document.documentElement.scrollTop || document.body.scrollTop),
  772. };
  773. };
  774.  
  775. //xpath 获取单个元素
  776. function getElementByXpath(xpath,contextNode,doc){
  777. doc=doc || document;
  778. contextNode=contextNode || doc;
  779. return doc.evaluate(xpath,contextNode,null,9,null).singleNodeValue;
  780. };
  781.  
  782.  
  783. //事件支持检测.
  784. function eventSupported( eventName,elem ){
  785. elem = elem || document.createElement("div");
  786. eventName = "on" + eventName;
  787. var isSupported = (eventName in elem);
  788. if (!isSupported){
  789. if(!elem.setAttribute){//setAttribute是元素节点的方法
  790. elem=document.createElement("div");
  791. };
  792. var setAttr;
  793. if(!elem.hasAttribute(eventName)){
  794. setAttr=true;
  795. elem.setAttribute(eventName, "return;");
  796. };
  797. isSupported = typeof elem[eventName] == "function";
  798. if(setAttr)elem.removeAttribute(eventName);
  799. };
  800. return isSupported;
  801. };
  802.  
  803.  
  804. //检测属性支持.dom属性
  805. //返回带前缀的可以直接执行是属性
  806. function proSupported(proName,elem){
  807. //判断第一个字母是否大写,如果是的话,为构造函数,前缀也要大写
  808. var prefix=/^[A-Z]/.test(proName)? ['','WebKit-','O-','Moz-','MS-'] : ['','webkit-','o-','moz-','ms-'];
  809. var i=0;
  810. var p_i;
  811. var sProName;
  812. elem = elem || document.createElement("div");
  813. while(typeof (p_i=prefix[i++])!='undefined'){
  814. sProName=(p_i+proName).replace(/-([A-z])/g,function(a,b){
  815. return b.toUpperCase();
  816. });
  817. //console.log(sProName);
  818. if(sProName in elem)return sProName;
  819. };
  820. };
  821.  
  822.  
  823. //css属性支持
  824. //带前缀的默认为大写(所有浏览器支持)
  825. //比如WebkitTransform,MozTransform,OTransfomr
  826. //chrome浏览器大小写前缀都行。
  827. //firefox,opera只能大写
  828. //ie 9+只能小写
  829. function cssProSupported(proName,elem,capitalize){
  830. if(capitalize!==false)capitalize=true;
  831. proName=proName.toLowerCase();
  832.  
  833. var prefix=['','-webkit-','-o-','-moz-','-ms-'];
  834. elem=elem || document.createElement('div');
  835. var style=elem.style;
  836. var camelPro;
  837.  
  838. // 会有个错误 invalid 'in' operand style
  839. try {
  840. for(var i=0,ii=prefix.length;i<ii;i++){
  841. var first=true;
  842. camelPro=(prefix[i]+proName).replace(/-([a-z])/g,function(a,b){
  843. b=b.toUpperCase();
  844. if(first){
  845. first=false;
  846. if(!capitalize){
  847. b=b.toLowerCase();
  848. };
  849. };
  850. return b;
  851. });
  852. //console.log(camelPro);
  853. if(camelPro in style){
  854. return camelPro;
  855. }
  856. }
  857. } catch(ex) {}
  858.  
  859. if(!capitalize)return;
  860. return cssProSupported(proName,elem,false);
  861.  
  862. };
  863.  
  864. //css属性值支持
  865. function cssValueSupported(proName,value,elem){
  866. var prefix=['','-webkit-','-o-','-moz-','-ms-'];
  867. elem=elem || document.createElement('div');
  868. var style=elem.style;
  869. var prefixedValue;
  870. for(var i=0,ii=prefix.length;i<ii;i++){
  871. prefixedValue=prefix[i] + value;
  872. style[proName]=prefixedValue;
  873. if(style[proName]==prefixedValue){
  874. return prefixedValue;
  875. };
  876. };
  877. };
  878.  
  879.  
  880. //elem.dataset的兼容实现
  881. //ie不支持;firefoxGM储存不能反映到元素属性上。
  882. function dataset(elem,pro,value){
  883.  
  884. function getDataPrefix(){
  885. return 'data-' + pro.replace(/[A-Z]/g,function(m){
  886. return '-' + m.toLowerCase();
  887. });
  888. };
  889.  
  890. if(typeof value=='undefined'){//取值
  891. if(elem.dataset){
  892. value = elem.dataset[pro];
  893. }else{//没有取到值,返回undefined,getAttribute默认是返回null,所以判断一下。
  894. var prefixedPro=getDataPrefix();
  895. if(elem.hasAttribute(prefixedPro)){
  896. value=elem.getAttribute(prefixedPro);
  897. };
  898. };
  899. return value;
  900. }else{
  901. elem.setAttribute(getDataPrefix(),value);
  902. };
  903. };
  904.  
  905.  
  906. //重新检查悬浮图片
  907. function imgReHover(img){
  908. //要检查的图片,是当前悬浮的。
  909. if(!floatBar.shown || floatBar.data.img != img)return;
  910. //console.log(img);
  911.  
  912. var mHover=document.createEvent('MouseEvent');
  913. var cr=img.getBoundingClientRect();
  914. 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);
  915. img.dispatchEvent(mHover);
  916. };
  917.  
  918. // 获取真正的unsafeWindow,chrome里面也能访问到真实环境的变量
  919. // 在 chrome 37 测试无效
  920. if(!envir.firefox && !envir.opera && !envir.ie){
  921. ;(function(){
  922. document.addEventListener('picViewer-return-unsafeWindow',function(e){
  923. unsafeWindow = e.detail;
  924. // alert(unsafeWindow.$);
  925. },true);
  926.  
  927. //页面脚本
  928. var s=document.createElement('script');
  929. s.textContent='(' + (function(){
  930. var cusEvent=document.createEvent('CustomEvent');
  931. cusEvent.initCustomEvent('picViewer-return-unsafeWindow',false,false,window);
  932. document.dispatchEvent(cusEvent);
  933. }).toString() +')()';
  934. document.head.appendChild(s);
  935. })();
  936. };
  937.  
  938.  
  939. //ie9 的HTMLElement.classList兼容,懒的麻烦,直接修改原型得了。=.=
  940. if(document.body && !document.body.classList){
  941.  
  942. ;(function (){
  943. 'use strict';
  944.  
  945. var ClassList=function(elem){
  946. var classes=elem.className.trim();
  947. classes=classes? classes.split(/\s+/) : [];
  948. this.push.apply(this,classes);
  949.  
  950. this._updateClassName=function(){
  951. elem.className=this.toString();
  952. };
  953. };
  954.  
  955.  
  956. var checkToken = function (token) {
  957. token += '';//转成字符串
  958.  
  959. var error=true;
  960.  
  961. var message;
  962. var type;
  963. var name;
  964. var code;
  965.  
  966. if (token == '') {//空字符串
  967. message='An invalid or illegal string was specified';
  968. name='SYNTAX_ERR';
  969. code=12;
  970. }else if (/\s/.test(token)) {//包含空格
  971. message='String contains an invalid character';
  972. name='INVALID_CHARACTER_ERR';
  973. code=5;
  974. }else{
  975. error=false;
  976. };
  977. if(error){
  978. error = new Error();
  979. error.message=message;
  980. error.type=type;
  981. error.name=name;
  982. error.code=code;
  983. throw error;
  984. };
  985. };
  986.  
  987. var ClassListProto = ClassList.prototype = [];//继承数组的方法
  988.  
  989. ClassListProto.add=function(token){
  990. checkToken(token);
  991. this.push(token);
  992. this._updateClassName();
  993. };
  994. ClassListProto.remove=function(token){
  995. checkToken(token);
  996. var index=this.indexOf(token);
  997. if(index != -1){//存在
  998. this.splice(index,1);
  999. this._updateClassName();
  1000. };
  1001. };
  1002. ClassListProto.contains=function(token){
  1003. checkToken(token);
  1004. return (this.indexOf(token) != -1)? true : false;
  1005. };
  1006. ClassListProto.item=function(index){
  1007. return this[index];
  1008. };
  1009. ClassListProto.toggle=function(token){
  1010. checkToken(token);
  1011. var index=this.indexOf(token);
  1012. if(index != -1){//存在
  1013. this.splice(index,1);
  1014. }else{
  1015. this.push(token);
  1016. };
  1017. this._updateClassName();
  1018. };
  1019. ClassListProto.toString=function(){
  1020. return this.join(" ");
  1021. };
  1022.  
  1023.  
  1024. Object.defineProperty(HTMLElement.prototype,'classList',{
  1025. get:function(){
  1026. return new ClassList(this);
  1027. },
  1028. enumerable:true,
  1029. configurable:true,
  1030. });
  1031. })();
  1032.  
  1033. };
  1034.  
  1035.  
  1036. //抛出错误到错误控制台
  1037. function throwErrorInfo(err){
  1038. if(console && console.error){
  1039. console.error(err.message + '\n\n' + (err.stacktrace? err.stacktrace : '') + '\n\n' , err);
  1040. };
  1041. };
  1042.  
  1043. //对象克隆
  1044. function cloneObject(obj,deep){
  1045. var obj_i;
  1046. var ret=Array.isArray(obj)? [] : {};
  1047. for(var i in obj){
  1048. if(!obj.hasOwnProperty(i))continue;
  1049. obj_i=obj[i];
  1050. if(!deep || typeof obj_i!='object' || obj_i===null || obj_i.nodeType){
  1051. ret[i]=obj_i;
  1052. }else{
  1053. ret[i]=cloneObject(obj_i,deep);
  1054. };
  1055. };
  1056. return ret;
  1057. };
  1058.  
  1059. //闪烁元素。
  1060. function flashEle(ele,duration){
  1061. if(dataset(ele,'pvFlashing'))return;
  1062. if(ele.offsetHeight==0)return;
  1063. dataset(ele,'pvFlashing','1');
  1064.  
  1065. var oOutline=ele.style.outline;
  1066. var oOutlineOffset=ele.style.outlineOffset;
  1067. var oOpacity=ele.style.opacity;
  1068. var oTransform=ele.style[support.cssTransform];
  1069.  
  1070. var count=0;
  1071. var startTime=Date.now();
  1072. duration=duration? duration : 1200;
  1073.  
  1074. var flashInterval=setInterval(function(){
  1075. var outline='none',
  1076. outlineOffset=0,
  1077. opacity=0.3,
  1078. transform='';
  1079.  
  1080. if(count % 2 == 0){
  1081. outline='5px dashed rgba(255,0,0,0.95)';
  1082. opacity=0.95;
  1083. outlineOffset='1px';
  1084. transform='scale(1.1)';
  1085. }else{
  1086. if((Date.now() - startTime) > duration){
  1087. clearInterval(flashInterval);
  1088. outline=oOutline;
  1089. opacity=oOpacity;
  1090. outlineOffset=oOutlineOffset;
  1091. transform=oTransform;
  1092. ele.removeAttribute('data-pv-flashing');
  1093. };
  1094. };
  1095.  
  1096. ele.style.outline=outline;
  1097. ele.style.outlineOffset=outlineOffset;
  1098. ele.style.opacity=opacity;
  1099. ele.style[support.cssTransform]=transform;
  1100.  
  1101. count++;
  1102. },80);
  1103. };
  1104.  
  1105. //支持情况.
  1106. var support={
  1107. cssTransform:cssProSupported('transform'),
  1108. cssCursorValue:{
  1109. zoomIn:cssValueSupported('cursor','zoom-in'),
  1110. zoomOut:cssValueSupported('cursor','zoom-out'),
  1111. grab:cssValueSupported('cursor','grab'),
  1112. grabbing:cssValueSupported('cursor','grabbing'),
  1113. },
  1114. };
  1115.  
  1116.  
  1117. //console.log('浏览器的一些对象支持情况:',support);
  1118.  
  1119. //动画算法
  1120. /*
  1121. t: current time(当前时间);
  1122. b: beginning value(初始值);
  1123. c: change in value(变化量);
  1124. d: duration(持续时间)。
  1125. */
  1126.  
  1127. var Tween = {
  1128. Cubic: {
  1129. easeInOut: function(t,b,c,d){
  1130. if ((t/=d/2) < 1) return c/2*t*t*t + b;
  1131. return c/2*((t-=2)*t*t + 2) + b;
  1132. },
  1133. },
  1134. };
  1135.  
  1136. //imgReady
  1137. var imgReady=(function(){
  1138. var iRInterval,
  1139. iRReadyFn=[],
  1140. isrcs=[]
  1141. ;
  1142.  
  1143. var timeLimit=3 * 60 * 1000;//3分钟
  1144.  
  1145. function checkReady(){
  1146. var now= Date.now();
  1147. for(var i=0,ii=iRReadyFn.length,iRReadyFn_i;i<ii;i++){
  1148. iRReadyFn_i=iRReadyFn[i];
  1149. //now - iRReadyFn_i.startTime >= timeLimit ||
  1150. if(iRReadyFn_i()){
  1151. iRReadyFn.splice(i,1);
  1152. isrcs.splice(i,1);
  1153. i--;
  1154. ii--;
  1155. };
  1156. };
  1157. //console.log('checkReady',iRReadyFn.length)
  1158. if(iRReadyFn.length==0){
  1159. clearInterval(iRInterval);
  1160. iRInterval=null;
  1161. };
  1162. };
  1163.  
  1164.  
  1165.  
  1166. var imgReady=function(img,opts){
  1167.  
  1168. if(/NodeList|HTMLCollection/.test(Object.prototype.toString.call(img)) || Array.isArray(img)){
  1169. arrayFn.forEach.call(img,function(img,index,array){
  1170. if(img instanceof HTMLImageElement){
  1171. imgReady(img,opts);
  1172. };
  1173. });
  1174. return;
  1175. };
  1176.  
  1177. if(!(img instanceof HTMLImageElement)){
  1178. var t_img=document.createElement('img');
  1179. t_img.src=img;
  1180. img=t_img;
  1181. t_img=null;
  1182. };
  1183.  
  1184. var ready,load,error,loadEnd,abort,timeout,time;
  1185. ready=opts.ready;
  1186. load=opts.load;
  1187. error=opts.error;
  1188. loadEnd=opts.loadEnd;
  1189. abort=opts.abort;
  1190. timeout=opts.timeout;
  1191. time=typeof opts.time=='number'? opts.time : 0;
  1192.  
  1193. if(time){
  1194. setTimeout(function(){
  1195. if(!loadEndDone){
  1196. aborted=true;
  1197. removeListener();
  1198. img.src= prefs.icons.brokenImg_small;
  1199. if(timeout){
  1200. timeout.call(img,{
  1201. target:img,
  1202. type:'timeout',
  1203. });
  1204. };
  1205. loadEndDone=true;
  1206. if(loadEnd){
  1207. loadEnd.call(img,{
  1208. target:img,
  1209. type:'timeout',
  1210. });
  1211. };
  1212.  
  1213. };
  1214. },time);
  1215. };
  1216.  
  1217. var src=img.src;
  1218. var loadEndDone;
  1219.  
  1220. function go(type,e){
  1221. switch(type){
  1222. case 'load':{
  1223. removeListener();
  1224. go('ready');//如果直接触发load,那么先触发ready
  1225. if(load){
  1226. load.call(img,e);
  1227. };
  1228.  
  1229. if(!loadEndDone){
  1230. loadEndDone=true;
  1231. if(loadEnd){
  1232. loadEnd.call(img,e);
  1233. };
  1234. };
  1235. }break;
  1236. case 'ready':{
  1237. if(!ready || readyHandler.done)return;
  1238. readyHandler.done=true;
  1239. ready.call(img,{
  1240. target:img,
  1241. type:'ready',
  1242. });
  1243. }break;
  1244. case 'error':{
  1245. removeListener();
  1246. if(error){
  1247. error.call(img,e);
  1248. };
  1249. if(!loadEndDone){
  1250. loadEndDone=true;
  1251. if(loadEnd){
  1252. loadEnd.call(img,e);
  1253. };
  1254. };
  1255. }break;
  1256. };
  1257. };
  1258.  
  1259. var aborted;
  1260. var ret={
  1261. img:img,
  1262. abort:function(){
  1263. if(!loadEndDone){
  1264. aborted=true;
  1265. removeListener();
  1266. img.src= prefs.icons.brokenImg_small;
  1267. if(abort){
  1268. abort.call(img,{
  1269. target:img,
  1270. type:'abort',
  1271. });
  1272. };
  1273. loadEndDone=true;
  1274. if(loadEnd){
  1275. loadEnd.call(img,{
  1276. target:img,
  1277. type:'abort',
  1278. });
  1279. };
  1280. };
  1281. },
  1282. };
  1283.  
  1284. function readyHandler(){//尽快的检测图片大小.
  1285. if(loadEndDone || aborted)return true;
  1286. if(img.naturalWidth==0 || img.naturalHeight==0)return;
  1287. go('ready');
  1288. return true;
  1289. };
  1290.  
  1291.  
  1292. function loadHandler(e){
  1293. go('load',e);
  1294. };
  1295.  
  1296. function errorHandler(e){
  1297. go('error',e);
  1298. };
  1299.  
  1300. function removeListener(){
  1301. img.removeEventListener('load',loadHandler,true);
  1302. img.removeEventListener('error',errorHandler,true);
  1303. };
  1304.  
  1305. //ready必须在load之前触发。
  1306.  
  1307. if(img.complete){//图片已经加载完成.
  1308. if(typeof img.width=='number' && img.width && img.height){//图片
  1309. setTimeout(function(){
  1310. if(aborted)return;
  1311. go('load',{
  1312. type:'load',
  1313. target:img,
  1314. });
  1315. },0);
  1316. }else{//这不是图片.opera会识别错误.
  1317. setTimeout(function(){
  1318. if(aborted)return;
  1319. go('error',{
  1320. type:'error',
  1321. target:img,
  1322. });
  1323. },0);
  1324. };
  1325. return ret;
  1326. };
  1327.  
  1328.  
  1329. img.addEventListener('load',loadHandler,true);
  1330. img.addEventListener('error',errorHandler,true);
  1331.  
  1332.  
  1333. if(ready){
  1334. var index=isrcs.indexOf(src);
  1335. if(index==-1){
  1336. isrcs.push(src);
  1337. readyHandler.startTime= Date.now();
  1338. iRReadyFn.push(readyHandler);
  1339. }else{
  1340. iRReadyFn[index].startTime= Date.now();
  1341. };
  1342.  
  1343. if(!iRInterval){
  1344. iRInterval=setInterval(checkReady,66);
  1345. };
  1346. };
  1347.  
  1348. return ret;
  1349. };
  1350.  
  1351. return imgReady;
  1352. })();
  1353.  
  1354.  
  1355. var addWheelEvent=(function(){
  1356.  
  1357. function getSupportEventName(){
  1358. var ret='DOMMouseScroll';
  1359. if(eventSupported('wheel')){//w3c FF>=17 ie>=9
  1360. ret='wheel';
  1361. }else if(eventSupported('mousewheel')){//opera,chrome
  1362. ret='mousewheel';
  1363. };
  1364. return ret;
  1365. };
  1366.  
  1367. var eventName;
  1368.  
  1369. return function(ele,callback,useCapture){
  1370. if(!eventName){
  1371. eventName=getSupportEventName();
  1372. };
  1373.  
  1374. ele.addEventListener(eventName,function(e){
  1375. var type=e.type;
  1376. var ne;
  1377. if(type!='wheel'){
  1378. ne={};
  1379. for(var i in e){
  1380. ne[i]=e[i];
  1381. };
  1382.  
  1383. ne.type='wheel';
  1384. ne.deltaX=0;
  1385. ne.deltaY=0;
  1386. ne.deltaZ=0;
  1387. ne.deltaMode=1;//line
  1388. ne.preventDefault=e.preventDefault.bind(e);
  1389. ne.stopPropagation=e.stopPropagation.bind(e);
  1390.  
  1391. var x=0,y=0;
  1392. if(typeof e.axis=='number'){//DOMMouseScroll
  1393. if(e.axis==2){
  1394. y=e.detail;
  1395. }else{
  1396. x=e.detail;
  1397. };
  1398. }else{
  1399. //opera早起版本的mousewheel只支持y轴的滚动,e.wheelDeltaY undefined
  1400. if(typeof e.wheelDeltaY=='undefined' || e.wheelDeltaY!=0){
  1401. y=-e.wheelDelta/40;
  1402. }else{
  1403. x=-e.wheelDelta/40;
  1404. };
  1405. };
  1406. ne.deltaY =y;
  1407. ne.deltaX =x;
  1408.  
  1409. };
  1410.  
  1411. callback.call(this,ne? ne : e);
  1412. },useCapture || false);
  1413. };
  1414. })();
  1415.  
  1416.  
  1417. var addCusMouseEvent=(function(){
  1418.  
  1419. function getSupported(){
  1420. return {
  1421. mouseleave:eventSupported('mouseleave'),
  1422. mouseenter:eventSupported('mouseenter'),
  1423. };
  1424. };
  1425.  
  1426. var support;
  1427. var map={
  1428. mouseleave:'mouseout',
  1429. mouseenter:'mouseover',
  1430. };
  1431.  
  1432. return function(type, ele, fn){//事件类型,元素,监听函数
  1433. if(!support){
  1434. support=getSupported();
  1435. };
  1436.  
  1437. // chrome 30+ 虽然支持 mouseenter,但是存在问题
  1438. if(support[type] && !(type == 'mouseenter' && window.chrome)){
  1439. ele.addEventListener(type,fn,false);//mouseleave,enter不冒泡
  1440. }else{
  1441. ele.addEventListener(map[type],function(e){
  1442. var relatedTarget=e.relatedTarget;//mouseout,去往的元素;mouseover,来自的元素
  1443. if(!this.contains(relatedTarget)){
  1444. fn.call(this,e);
  1445. };
  1446. },true);
  1447. };
  1448. };
  1449.  
  1450. })();
  1451.  
  1452.  
  1453. //库
  1454. function GalleryC(){
  1455. this.init();
  1456. };
  1457.  
  1458. var gallery;
  1459. var galleryMode;
  1460.  
  1461. GalleryC.prototype={
  1462. init:function(){
  1463. this.addStyle();
  1464. var container=document.createElement('span');
  1465.  
  1466. this.gallery=container;
  1467. container.className='pv-gallery-container';
  1468. container.tabIndex=1;//为了获取焦点,来截获键盘事件
  1469. container.innerHTML=
  1470. '<span class="pv-gallery-head">'+
  1471. '<span class="pv-gallery-head-float-left">'+
  1472. '<span title="图片信息" class="pv-gallery-head-left-img-info">'+
  1473. '<span class="pv-gallery-head-left-img-info-resolution" title="分辨率">0 x 0</span>'+
  1474. '<span class="pv-gallery-head-left-img-info-scaling" title="缩放比">(100%)</span>'+
  1475. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1476. '</span>'+
  1477. '</span>'+
  1478.  
  1479. '<span title="点击退出收藏模式" class="pv-gallery-head-command pv-gallery-head-command-exit-collection">'+
  1480. '<span>退出收藏</span>'+
  1481. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1482. '</span>'+
  1483.  
  1484. '<span title="弹出照片进行复杂操作" class="pv-gallery-head-command pv-gallery-head-command-operate">'+
  1485. '<span>折腾</span>'+
  1486. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1487. '</span>'+
  1488.  
  1489. '<span class="pv-gallery-head-command-container">'+
  1490. '<span class="pv-gallery-head-command pv-gallery-head-command-collect">'+
  1491. '<span class="pv-gallery-head-command-collect-icon"></span>'+
  1492. '<span class="pv-gallery-head-command-collect-text"></span>'+
  1493. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1494. '</span>'+
  1495. '<span class="pv-gallery-head-command-drop-list pv-gallery-head-command-drop-list-collect">'+
  1496. '<span title="给收藏的图片添加一些描述吧" class="pv-gallery-head-command-drop-list-item pv-gallery-head-command-drop-list-item-collect-description">'+
  1497. '<span>描述:</span>'+
  1498. '<textarea data-prefs="description" cols="25" rows="5"></textarea>'+
  1499. '</span>'+
  1500. '</span>'+
  1501. '</span>'+
  1502.  
  1503. '<span class="pv-gallery-head-command-container">'+
  1504. '<span title="播放幻灯片" class="pv-gallery-head-command pv-gallery-head-command-slide-show">'+
  1505. '<span class="pv-gallery-head-command_overlayer"></span>'+
  1506. '<span class="pv-gallery-head-command-slide-show-button">'+
  1507. '<span class="pv-gallery-head-command-slide-show-button-inner"></span>'+
  1508. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1509. '</span>'+
  1510. '<span class="pv-gallery-head-command-slide-show-countdown" title="倒计时"></span>'+
  1511. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1512. '</span>'+
  1513. '<span class="pv-gallery-head-command-drop-list pv-gallery-head-command-drop-list-slide-show">'+
  1514. '<span class="pv-gallery-head-command-drop-list-item" title="间隔时间,单位(秒)">'+
  1515. '<input data-prefs="interval" step="1" min="1" type="number" value="5" />'+
  1516. '<span>间隔(s)</span>'+
  1517. '</span>'+
  1518. '<span class="pv-gallery-head-command-drop-list-item" title="从后往前播放">'+
  1519. '<input id="pv-gallery-head-command-drop-list-item-slide-show-backward" data-prefs="backward" type="checkbox" />'+
  1520. '<label for="pv-gallery-head-command-drop-list-item-slide-show-backward">后退   </label>'+
  1521. '</span>'+
  1522. '<span class="pv-gallery-head-command-drop-list-item" title="从每张图片完全读取完成后才开始倒计时">'+
  1523. '<input id="pv-gallery-head-command-drop-list-item-slide-show-wait" data-prefs="wait" type="checkbox" checked="checked" />'+
  1524. '<label for="pv-gallery-head-command-drop-list-item-slide-show-wait">等待图片读取</label>'+
  1525. '</span>'+
  1526. '<span class="pv-gallery-head-command-drop-list-item" title="快速跳过读取错误的图片">'+
  1527. '<input id="pv-gallery-head-command-drop-list-item-slide-show-skipErrorImg" data-prefs="skipErrorImg" type="checkbox" checked="checked" />'+
  1528. '<label for="pv-gallery-head-command-drop-list-item-slide-show-skipErrorImg">跳过错误图片</label>'+
  1529. '</span>'+
  1530. '</span>'+
  1531. '</span>'+
  1532.  
  1533. '<span class="pv-gallery-head-command-container">'+
  1534. '<span title="选择图片类别" class="pv-gallery-head-command pv-gallery-head-command-category">'+
  1535. '<span>类别</span>'+
  1536. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1537. '</span>'+
  1538. '<span class="pv-gallery-head-command-drop-list pv-gallery-head-command-drop-list-category">'+
  1539. '</span>'+
  1540. '</span>'+
  1541.  
  1542. '<span class="pv-gallery-head-command-container">'+
  1543. '<span title="一些命令菜单" class="pv-gallery-head-command pv-gallery-head-command-others">'+
  1544. '<span>命令</span>'+
  1545. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1546. '</span>'+
  1547. '<span class="pv-gallery-head-command-drop-list pv-gallery-head-command-drop-list-others">'+
  1548. '<span class="pv-gallery-head-command-drop-list-item" data-command="openInNewWindow" title="新窗口打开图片">新窗口打开</span>'+
  1549. '<span class="pv-gallery-head-command-drop-list-item" data-command="scrollIntoView" title="滚动到当前图片所在的位置">定位到图片</span>'+
  1550. '<span class="pv-gallery-head-command-drop-list-item" data-command="enterCollection" title="查看所有收藏的图片">查看所有收藏</span>'+
  1551. '<span class="pv-gallery-head-command-drop-list-item" data-command="exportImages" title="导出所有图片的链接到新窗口">导出所有图片</span>'+
  1552. '<span class="pv-gallery-head-command-drop-list-item" data-command="reloadGalleryC" title="重新载入所有有效的图片">手动重载</span>'+
  1553. '<span class="pv-gallery-head-command-drop-list-item" title="最后一张图片时,滚动主窗口到最底部,然后自动重载库的图片(测试)">'+
  1554. '<input type="checkbox" data-command="scrollToEndAndReload"/>'+
  1555. '<label data-command="scrollToEndAndReload">自动重载</label>'+
  1556. '</span>'+
  1557. '<span class="pv-gallery-head-command-drop-list-item" data-command="showHideBottom" title="显示或隐藏缩略图栏">切换缩略图栏</span>'+
  1558. '<span id="pv-gallery-fullscreenbtn" class="pv-gallery-head-command-drop-list-item" data-command="fullScreen">进入全屏</span>'+
  1559. '</span>'+
  1560. '</span>'+
  1561.  
  1562. '<span class="pv-gallery-head-command-container">'+
  1563. '<span title="分享" class="pv-gallery-head-command pv-gallery-head-command-share">'+
  1564. '<span>分享</span>'+
  1565. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1566. '</span>'+
  1567. '<span class="pv-gallery-head-command-drop-list pv-gallery-head-command-drop-list-share">'+
  1568. '</span>'+
  1569. '</span>'+
  1570.  
  1571. '<span title="关闭库" class="pv-gallery-head-command pv-gallery-head-command-close">'+
  1572. '</span>'+
  1573.  
  1574. '</span>'+
  1575.  
  1576. '<span class="pv-gallery-body">'+
  1577.  
  1578. '<span class="pv-gallery-img-container">'+
  1579.  
  1580. '<span class="pv-gallery-img-content">'+
  1581. '<span class="pv-gallery-img-parent">'+
  1582. '<img title="读取错误,点击重载" class="pv-gallery-img_broken" src="'+prefs.icons.brokenImg+'" />'+
  1583. '</span>'+
  1584. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1585. '</span>'+
  1586.  
  1587. '<span class="pv-gallery-img-controler pv-gallery-img-controler-pre"></span>'+
  1588. '<span class="pv-gallery-img-controler pv-gallery-img-controler-next"></span>'+
  1589.  
  1590. '<span class="pv-gallery-scrollbar-h pv-gallery-img-scrollbar-h">'+
  1591. '<span class="pv-gallery-scrollbar-h-track pv-gallery-img-scrollbar-h-track">'+
  1592. '<span class="pv-gallery-scrollbar-h-handle pv-gallery-img-scrollbar-h-handle"></span>'+
  1593. '</span>'+
  1594. '</span>'+
  1595.  
  1596. '<span class="pv-gallery-scrollbar-v pv-gallery-img-scrollbar-v">'+
  1597. '<span class="pv-gallery-scrollbar-v-track pv-gallery-img-scrollbar-v-track">'+
  1598. '<span class="pv-gallery-scrollbar-v-handle pv-gallery-img-scrollbar-v-handle"></span>'+
  1599. '</span>'+
  1600. '</span>'+
  1601.  
  1602. '<span class="pv-gallery-sidebar-toggle" title="开关侧边栏">'+
  1603. '<span class="pv-gallery-sidebar-toggle-content">隐藏</span>'+
  1604. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1605. '</span>'+
  1606.  
  1607. '</span>'+
  1608.  
  1609. '<span class="pv-gallery-sidebar-container" unselectable="on">'+
  1610. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1611. '<span class="pv-gallery-sidebar-content" >'+
  1612.  
  1613. '<span class="pv-gallery-sidebar-controler pv-gallery-sidebar-controler-pre"></span>'+
  1614. '<span class="pv-gallery-sidebar-controler pv-gallery-sidebar-controler-next"></span>'+
  1615.  
  1616. '<span class="pv-gallery-sidebar-thumbnails-container">'+
  1617. '</span>'+
  1618.  
  1619. '<span class="pv-gallery-scrollbar-h pv-gallery-thumb-scrollbar-h">'+
  1620. '<span class="pv-gallery-scrollbar-h-track pv-gallery-thumb-scrollbar-h-track">'+
  1621. '<span class="pv-gallery-scrollbar-h-handle pv-gallery-thumb-scrollbar-h-handle"></span>'+
  1622. '</span>'+
  1623. '</span>'+
  1624. '<span class="pv-gallery-scrollbar-v pv-gallery-thumb-scrollbar-v">'+
  1625. '<span class="pv-gallery-scrollbar-v-track pv-gallery-thumb-scrollbar-v-track">'+
  1626. '<span class="pv-gallery-scrollbar-v-handle pv-gallery-thumb-scrollbar-v-handle"></span>'+
  1627. '</span>'+
  1628. '</span>'+
  1629.  
  1630. '</span>'+
  1631. '</span>'+
  1632.  
  1633. '</span>';
  1634. document.body.appendChild(container);
  1635.  
  1636. var maximizeTrigger=document.createElement('span');
  1637. this.maximizeTrigger=maximizeTrigger;
  1638. maximizeTrigger.innerHTML='-回到库-<span class="pv-gallery-maximize-trigger-close" title="关闭库"></span>';
  1639. maximizeTrigger.className='pv-gallery-maximize-trigger';
  1640.  
  1641. document.body.appendChild(maximizeTrigger);
  1642.  
  1643. var validPos=['top','right','bottom','left'];
  1644. var sBarPosition=prefs.gallery.sidebarPosition;
  1645. if(validPos.indexOf(sBarPosition)==-1){
  1646. sBarPosition='bottom';
  1647. };
  1648.  
  1649. this.sBarPosition=sBarPosition;
  1650. this.selectedClassName='pv-gallery-sidebar-thumb_selected-' + sBarPosition;
  1651.  
  1652.  
  1653. var sBarDirection='v';//垂直放置
  1654. var isHorizontal=false;
  1655. if(sBarPosition=='top' || sBarPosition=='bottom'){
  1656. sBarDirection='h';//水平放置
  1657. isHorizontal=true;
  1658. };
  1659. this.sBarDirection=sBarDirection;
  1660. this.isHorizontal=isHorizontal;
  1661.  
  1662. var classPrefix='pv-gallery-';
  1663. var validClass=[
  1664. 'head',
  1665.  
  1666. 'head-left-img-info',
  1667. 'head-left-img-info-resolution',
  1668. 'head-left-img-info-scaling',
  1669.  
  1670. 'head-command-close',
  1671. 'head-command-operate',
  1672. 'head-command-slide-show',
  1673. 'head-command-slide-show-button-inner',
  1674. 'head-command-slide-show-countdown',
  1675. 'head-command-collect',
  1676. 'head-command-exit-collection',
  1677.  
  1678. 'head-command-drop-list-category',
  1679. 'head-command-drop-list-others',
  1680. 'head-command-drop-list-share',
  1681. 'head-command-drop-list-slide-show',
  1682. 'head-command-drop-list-collect',
  1683.  
  1684. 'body',
  1685.  
  1686. 'img-container',
  1687.  
  1688. 'img-scrollbar-h',
  1689. 'img-scrollbar-h-handle',
  1690. 'img-scrollbar-h-track',
  1691.  
  1692. 'img-scrollbar-v',
  1693. 'img-scrollbar-v-handle',
  1694. 'img-scrollbar-v-track',
  1695.  
  1696. 'thumb-scrollbar-h',
  1697. 'thumb-scrollbar-h-handle',
  1698. 'thumb-scrollbar-h-track',
  1699.  
  1700. 'thumb-scrollbar-v',
  1701. 'thumb-scrollbar-v-handle',
  1702. 'thumb-scrollbar-v-track',
  1703.  
  1704. 'img-content',
  1705. 'img-parent',
  1706. 'img_broken',
  1707.  
  1708. 'img-controler-pre',
  1709. 'img-controler-next',
  1710.  
  1711. 'sidebar-toggle',
  1712. 'sidebar-toggle-content',
  1713.  
  1714. 'sidebar-container',
  1715. 'sidebar-content',
  1716.  
  1717. 'sidebar-controler-pre',
  1718. 'sidebar-controler-next',
  1719.  
  1720. 'sidebar-thumbnails-container',
  1721. ];
  1722.  
  1723. var eleMaps={};
  1724. this.eleMaps=eleMaps;
  1725.  
  1726. validClass.forEach(function(c){
  1727. eleMaps[c]=container.querySelector('.'+ classPrefix + c);
  1728. });
  1729.  
  1730. var posClass=[//需要添加'top bottom left right'class的元素
  1731. 'img-container',
  1732. 'sidebar-toggle',
  1733. 'sidebar-container',
  1734. 'sidebar-thumbnails-container',
  1735. ];
  1736. posClass.forEach(function(c){
  1737. eleMaps[c].classList.add(classPrefix + c + '-' +sBarPosition);
  1738. });
  1739.  
  1740. var hvClass=[//需要添加'v h'class的元素
  1741. 'sidebar-toggle',
  1742. 'sidebar-toggle-content',
  1743. 'sidebar-container',
  1744. 'sidebar-content',
  1745. 'sidebar-controler-pre',
  1746. 'sidebar-controler-next',
  1747. 'sidebar-thumbnails-container',
  1748. ];
  1749. hvClass.forEach(function(c){
  1750. eleMaps[c].classList.add(classPrefix + c + '-' + sBarDirection);
  1751. });
  1752.  
  1753.  
  1754.  
  1755. //图片区域水平方向的滚动条
  1756. var imgScrollbarH=new this.Scrollbar({
  1757. bar:eleMaps['img-scrollbar-h'],
  1758. handle:eleMaps['img-scrollbar-h-handle'],
  1759. track:eleMaps['img-scrollbar-h-track'],
  1760. },
  1761. eleMaps['img-content'],
  1762. true);
  1763. this.imgScrollbarH=imgScrollbarH;
  1764.  
  1765. //图片区域垂直方向的滚动条
  1766. var imgScrollbarV=new this.Scrollbar({
  1767. bar:eleMaps['img-scrollbar-v'],
  1768. handle:eleMaps['img-scrollbar-v-handle'],
  1769. track:eleMaps['img-scrollbar-v-track'],
  1770. },
  1771. eleMaps['img-content'],
  1772. false);
  1773. this.imgScrollbarV=imgScrollbarV;
  1774.  
  1775. //缩略图区域的滚动条
  1776. var thumbScrollbar;
  1777. if(isHorizontal){
  1778. thumbScrollbar=new this.Scrollbar({
  1779. bar:eleMaps['thumb-scrollbar-h'],
  1780. handle:eleMaps['thumb-scrollbar-h-handle'],
  1781. track:eleMaps['thumb-scrollbar-h-track'],
  1782. },
  1783. eleMaps['sidebar-thumbnails-container'],
  1784. true);
  1785. }else{
  1786. thumbScrollbar=new this.Scrollbar({
  1787. bar:eleMaps['thumb-scrollbar-v'],
  1788. handle:eleMaps['thumb-scrollbar-v-handle'],
  1789. track:eleMaps['thumb-scrollbar-v-track'],
  1790. },
  1791. eleMaps['sidebar-thumbnails-container'],
  1792. false);
  1793. };
  1794. this.thumbScrollbar=thumbScrollbar;
  1795.  
  1796. var self=this;
  1797.  
  1798.  
  1799. var imgStatistics={//图片的总类,统计,初始化值
  1800. rule:{
  1801. shown:true,
  1802. count:0,
  1803. description:'由高级规则匹配出来的',
  1804. name:'高级规则',
  1805. },
  1806. tpRule:{
  1807. shown:true,
  1808. count:0,
  1809. description:'由通配规则匹配出来的',
  1810. name:'通配规则',
  1811. },
  1812. scale:{
  1813. shown:true,
  1814. count:0,
  1815. description:'js自动查找,相对页面显示的图片有缩放过的',
  1816. name:'缩放过的',
  1817. },
  1818. force:{
  1819. shown:true,
  1820. count:0,
  1821. description:'js自动查找,无缩放过的,但是满足一定的大小',
  1822. name:'无缩放过',
  1823. },
  1824. };
  1825. this.imgStatistics=imgStatistics;
  1826.  
  1827. //生成分类下拉列表
  1828. var typeMark='';
  1829. var imgStatistics_i;
  1830. for(var i in imgStatistics){
  1831. if(!imgStatistics.hasOwnProperty(i))continue;
  1832. imgStatistics_i=imgStatistics[i];
  1833. typeMark+=
  1834. '<span class="pv-gallery-head-command-drop-list-item" title="'+imgStatistics_i.description+'">'+
  1835. '<input type="checkbox" data-type="'+i+'" id="pv-gallery-head-command-drop-list-item-category-'+i+'" />'+
  1836. '<label for="pv-gallery-head-command-drop-list-item-category-'+i+'">'+imgStatistics_i.name+'</label>'+
  1837. '</span>';
  1838. };
  1839. eleMaps['head-command-drop-list-category'].innerHTML=typeMark;
  1840.  
  1841.  
  1842. //收藏相关
  1843. var collection={
  1844. getMatched:function(){
  1845. return (this.all || this.get())._find(function(value,index){
  1846. if(value.src==self.src){
  1847. return true;
  1848. };
  1849. });
  1850. },
  1851. check:function(){
  1852. //从缓存数据中检查。
  1853. var matched=this.getMatched();
  1854. this.favorite=matched? matched[0] : null;
  1855.  
  1856. this.tAreaValue();
  1857. this.highLight();
  1858. },
  1859. tAreaValue:function(){
  1860. this.textArea.value=this.favorite? this.favorite.description : '';
  1861. },
  1862. highLight:function(){
  1863. eleMaps['head-command-collect'].classList[this.favorite? 'add' : 'remove']('pv-gallery-head-command-collect-favorite');
  1864. },
  1865. add:function(){
  1866. this.favorite={
  1867. src:self.src,
  1868. thumbSrc:dataset(self.relatedThumb,'thumbSrc'),
  1869. naturalSize:self.imgNaturalSize,
  1870. description:this.textArea.value,
  1871. };
  1872.  
  1873. //为了防止多个页面同时的储存,添加前,先载入最新的数据。
  1874. this.get();
  1875. //检查是否已经在里面了
  1876. var matched=this.getMatched();
  1877.  
  1878. if(matched){//如果已经存在,删除旧的。
  1879. this.all.splice(matched[1],1);
  1880. };
  1881. this.all.unshift(this.favorite);//添加到最前面。
  1882. this.highLight();
  1883. this.save();
  1884. },
  1885. remove:function(){
  1886. //获得最新数据
  1887. this.get();
  1888. //检查是否已经在里面了
  1889. var matched=this.getMatched();
  1890. if(matched){
  1891. this.all.splice(matched[1],1);
  1892. this.save();
  1893. };
  1894. this.favorite=null;
  1895. this.highLight();
  1896. },
  1897. save:function(){
  1898. storage.setItem('pv_collection',encodeURIComponent(JSON.stringify(this.all)));
  1899. },
  1900. get:function(){
  1901. var ret=storage.getItem('pv_collection') || '[]';
  1902. try{
  1903. ret=JSON.parse(decodeURIComponent(ret));
  1904. }catch(e){
  1905. ret=[];
  1906. };
  1907. this.all=ret;
  1908. return ret;
  1909. },
  1910. enter:function(){
  1911.  
  1912. if(this.all.length==0){
  1913. alert('你还木有收藏任何图片');
  1914. return;
  1915. };
  1916.  
  1917. this.mMode=true;
  1918. var button=this.dropListButton;
  1919. button.textContent='退出收藏查看';
  1920. dataset(button,'command','exitCollection');
  1921. this.headButton.style.display='inline-block';
  1922. eleMaps['sidebar-thumbnails-container'].classList.add('pv-gallery-sidebar-thumbnails_hide-span');
  1923.  
  1924. //生成dom
  1925. var container=document.createElement('span');
  1926.  
  1927. this.container=container;
  1928.  
  1929. var data_i;
  1930. var spanMark='';
  1931. var i=0;
  1932. while(data_i=this.all[i++]){
  1933. spanMark +=
  1934. '<span class="pv-gallery-sidebar-thumb-container" '+
  1935. ' data-natural-size="' + JSON.stringify(data_i.naturalSize).replace(/"/g,'&quot;') +
  1936. '" data-src="' + data_i.src +
  1937. '" data-thumb-src="' + data_i.thumbSrc +
  1938. '">'+
  1939. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1940. '<span class="pv-gallery-sidebar-thumb-loading" title="正在读取中......"></span>'+
  1941. '</span>';
  1942. };
  1943. container.innerHTML=spanMark;
  1944. eleMaps['sidebar-thumbnails-container'].appendChild(container);
  1945.  
  1946.  
  1947. this.selected=self.selected;//备份
  1948.  
  1949. self.select(container.children[0]);
  1950. self.thumbScrollbar.reset();
  1951. self.loadThumb();
  1952. },
  1953. exit:function(){
  1954. if(!this.mMode)return;
  1955.  
  1956. this.mMode=false;
  1957. var button=this.dropListButton;
  1958. button.textContent='查看所有收藏';
  1959. dataset(button,'command','enterCollection');
  1960. this.headButton.style.display='none';
  1961. eleMaps['sidebar-thumbnails-container'].removeChild(this.container);
  1962. eleMaps['sidebar-thumbnails-container'].classList.remove('pv-gallery-sidebar-thumbnails_hide-span');
  1963.  
  1964. self.select(this.selected);
  1965. self.thumbScrollbar.reset();
  1966. self.loadThumb();
  1967. },
  1968. textArea:eleMaps['head-command-drop-list-collect'].querySelector('textarea'),
  1969. dropListButton:eleMaps['head-command-drop-list-others'].querySelector('[data-command$="Collection"]'),
  1970. headButton:eleMaps['head-command-exit-collection'],
  1971. };
  1972.  
  1973. this.collection=collection;
  1974.  
  1975. eleMaps['head-command-drop-list-collect'].addEventListener('input',function(e){
  1976. var target=e.target;
  1977. if(!collection.favorite)return;
  1978. collection.favorite[dataset(target,'prefs')]=target.value;
  1979. clearTimeout(collection.saveTimer);
  1980. collection.saveTimer=setTimeout(function(){
  1981. collection.save();
  1982. },500);
  1983. },true);
  1984.  
  1985.  
  1986. var slideShow={
  1987. opts:{
  1988. interval:5000,
  1989. wait:true,
  1990. backward:false,
  1991. skipErrorImg:true,
  1992. run:false,
  1993. },
  1994. //timing:
  1995. //select(选中下一个图片后(缩略图栏选中了),还没开始读取大图(一般选中后,延时200ms开始读取大图)),
  1996. //loadEnd(当前显示图片已经读取完成后),
  1997. //click(点击按钮),
  1998. //change(改变设置)
  1999. run:function(timing){
  2000. if(!this.opts.run)return;
  2001.  
  2002. if(timing!='loadEnd'){
  2003. this.stop();
  2004. };
  2005.  
  2006. if(timing=='click' || timing=='select'){
  2007. if(!this.getEle()){//没有要切换到的图片了,停止
  2008. this.exit();
  2009. return;
  2010. };
  2011. };
  2012.  
  2013. if(this.opts.skipErrorImg){
  2014. if(self.imgError && !self.isLoading){//确保是当前图片和选中缩略图一致的时候
  2015. self.select(this.getEle());
  2016. return;
  2017. };
  2018. };
  2019.  
  2020.  
  2021. if(this.opts.wait){
  2022. if(timing!='select' && (timing=='loadEnd' || (!self.isLoading && (self.img.complete || self.imgError)))){
  2023. this.go();
  2024. };
  2025. }else{
  2026. if(timing!='loadEnd'){
  2027. this.go();
  2028. };
  2029. };
  2030.  
  2031. },
  2032. getEle:function(){
  2033. return self.getThumSpan(this.opts.backward)
  2034. },
  2035. go:function(){
  2036. this.stop();//停止上次的。
  2037. var interval=this.opts.interval;
  2038. var _self=this;
  2039. this.timer=setTimeout(function(){
  2040. _self.setCountdown(0);
  2041. clearInterval(_self.countdownTimer);
  2042. self.select(_self.getEle());
  2043. },interval);
  2044.  
  2045. var startTime=Date.now();
  2046. this.countdownTimer=setInterval(function(){
  2047. _self.setCountdown(interval - (Date.now()-startTime));
  2048. },100);
  2049. },
  2050. stop:function(){
  2051. this.setCountdown(this.opts.interval);
  2052. clearTimeout(this.timer);
  2053. clearInterval(this.countdownTimer);
  2054. },
  2055. exit:function(){
  2056. this.opts.run=true;
  2057. this.switchStatus();
  2058. this.stop();
  2059. },
  2060. setCountdown:function(value){
  2061. eleMaps['head-command-slide-show-countdown'].textContent=(value/1000).toFixed(2);
  2062. },
  2063. switchStatus:function(){
  2064. this.opts.run=!this.opts.run;
  2065. eleMaps['head-command-slide-show-button-inner'].classList[this.opts.run? 'add' : 'remove']('pv-gallery-head-command-slide-show-button-inner_stop');
  2066. },
  2067. check:function(){
  2068. this.opts.run? this.run('click') : this.stop();
  2069. },
  2070. };
  2071.  
  2072. slideShow.setCountdown(slideShow.opts.interval);;
  2073. this.slideShow=slideShow;
  2074.  
  2075. //幻灯片播放下拉列表change事件的处理
  2076. eleMaps['head-command-drop-list-slide-show'].addEventListener('change',function(e){
  2077. var target=e.target;
  2078. var value;
  2079. var prefs=dataset(target,'prefs');
  2080. if(target.type=='checkbox'){
  2081. value=target.checked;
  2082. }else{
  2083. value=parseFloat(target.value);
  2084. if(isNaN(value)){//无效
  2085. value=slideShow.opts[prefs] / 1000;
  2086. };
  2087. value=value>0 ? value : 1;
  2088. target.value=value;
  2089. value *= 1000;
  2090. };
  2091. slideShow.opts[prefs]=value;
  2092. slideShow.run('change');
  2093. //console.log(slideShow.opts);
  2094. },true);
  2095.  
  2096.  
  2097. //分类下拉列表的点击发生change事件的处理
  2098. eleMaps['head-command-drop-list-category'].addEventListener('change',function(e){
  2099. var target=e.target;
  2100. self.iStatisCopy[dataset(target,'type')].shown=target.checked;
  2101. self.switchThumbVisible();//切换图片类别显隐;
  2102. },true);
  2103.  
  2104.  
  2105. //命令下拉列表的点击处理
  2106. eleMaps['head-command-drop-list-others'].addEventListener('click',function(e){
  2107. if(e.button!=0)return;//左键
  2108. var target=e.target;
  2109. var command=dataset(target,'command');
  2110. if(!command)return;
  2111. switch(command){
  2112. case 'openInNewWindow':{
  2113. window.open(self.src,'_blank');
  2114. }break;
  2115. case 'scrollIntoView':{
  2116. if(collection.mMode){
  2117. alert('收藏模式中,无法使用');
  2118. return;
  2119. };
  2120. var relatedThumb=self.relatedThumb;
  2121. var index=arrayFn.indexOf.call(self.imgSpans,relatedThumb);
  2122. var targetImg=self.data[index].img;
  2123.  
  2124. if(targetImg){
  2125. if(!document.documentElement.contains(targetImg) || getComputedStyle(targetImg).display=='none'){//图片不存在文档中,或者隐藏了。
  2126. alert('图片不在文档中,或者被隐藏了,无法定位!');
  2127. return;
  2128. };
  2129. self.minimize();
  2130. setTimeout(function(){
  2131. self.navigateToImg(targetImg);
  2132. flashEle(targetImg);
  2133. },0);
  2134.  
  2135. }else{//frame发送过来的时候删除了不能传送的图片
  2136.  
  2137. document.addEventListener('pv-navigateToImg',function(e){
  2138. //console.log('pv-navigateToImg',e);
  2139. if(!e.detail){
  2140. alert('图片不在文档中,或者被隐藏了,无法定位!');
  2141. return;
  2142. };
  2143. self.minimize();
  2144. setTimeout(function(){//将frame滚动到中间位置
  2145. if(self.iframe){
  2146. self.navigateToImg(self.iframe);
  2147. };
  2148. },0);
  2149. },true);
  2150. window.postMessage({//问问frame。。
  2151. messageID:messageID,
  2152. command:'navigateToImg',
  2153. index:index,
  2154. to:self.from,
  2155. },'*');
  2156. };
  2157.  
  2158. }break;
  2159. case 'exportImages':
  2160. var exportImages = function () { // 导出所有图片到新窗口
  2161. var nodes = document.querySelectorAll('.pv-gallery-sidebar-thumb-container[data-src]');
  2162. var arr = [].map.call(nodes, function(node){
  2163. return '<div><img src=' + node.dataset.src + ' /></div>'
  2164. });
  2165.  
  2166. var title = document.title;
  2167.  
  2168. var html = '\
  2169. <head>\
  2170. <title>' + title + ' 导出大图</title>\
  2171. <style>\
  2172. div {\
  2173. float: left;\
  2174. max-height: 180px;\
  2175. max-width: 320px;\
  2176. margin: 2px;\
  2177. }\
  2178. img {\
  2179. max-height: 180px;\
  2180. max-width: 320px;\
  2181. }\
  2182. </style>\
  2183. </head>\
  2184. <body>\
  2185. <p>【图片标题】:' + title + '</p>\
  2186. <p>【图片数量】:' + nodes.length + '</p>\
  2187. ';
  2188.  
  2189. html += arr.join('\n') + '</body>'
  2190. GM_openInTab('data:text/html;charset=utf-8,' + encodeURIComponent(html));
  2191. };
  2192.  
  2193. exportImages();
  2194. break;
  2195. case 'showHideBottom':
  2196. // 显示隐藏底部图片罗列栏
  2197. var imgContainer = document.querySelector('.pv-gallery-img-container-bottom'),
  2198. sidebarContainer = document.querySelector('.pv-gallery-sidebar-container-bottom'),
  2199. isHidden = !(sidebarContainer.style.visibility == 'hidden');
  2200. sidebarContainer.style.visibility = isHidden ? 'hidden' : 'visible';
  2201. // 修正下图片底部的高度
  2202. imgContainer.style.borderBottom = isHidden ? '0px' : prefs.gallery.sidebarSize + 'px solid transparent';
  2203. break;
  2204. case 'reloadGalleryC':
  2205. self.reload();
  2206. break;
  2207. case 'scrollToEndAndReload':
  2208. var checkbox = target.parentNode.firstChild;
  2209. checkbox.checked = !checkbox.checked;
  2210.  
  2211. prefs.gallery.autoScrollAndReload = checkbox.checked;
  2212. break;
  2213. case 'fullScreen':
  2214. if (target.classList.contains('fullscreenbtn')) {
  2215. if (cancelFullScreen()) return;
  2216. target.textContent = '进入全屏';
  2217. target.classList.remove('fullscreenbtn');
  2218. return;
  2219. }
  2220.  
  2221. if (launchFullScreen(document.documentElement)) return;
  2222. target.classList.toggle('fullscreenbtn');
  2223. target.textContent = '退出全屏';
  2224. target.classList.add('fullscreenbtn');
  2225. break;
  2226. case 'enterCollection':{
  2227. //进入管理模式
  2228. collection.enter();
  2229. }break;
  2230. case 'exitCollection':{
  2231. //退出管理模式
  2232. collection.exit();
  2233. }break;
  2234. };
  2235. },true);
  2236.  
  2237. // 监视全屏的变化
  2238. function fullScreenChanged() {
  2239. if (!document.fullscreenElement && // alternative standard method
  2240. !document.mozFullScreenElement &&
  2241. !document.webkitFullscreenElement &&
  2242. !document.msFullscreenElement) {
  2243.  
  2244. var btn = document.getElementById("pv-gallery-fullscreenbtn");
  2245. btn.textContent = '进入全屏';
  2246. btn.removeClass('fullscreenbtn');
  2247. }
  2248. }
  2249. document.addEventListener('webkitfullscreenchange', fullScreenChanged, false);
  2250. document.addEventListener('mozfullscreenchange', fullScreenChanged, false);
  2251. document.addEventListener('fullscreenchange', fullScreenChanged, false);
  2252.  
  2253. //生成分享的下拉列表
  2254. var shareMark='';
  2255. var shareItem;
  2256. for(var i in prefs.share){
  2257. if(!prefs.share.hasOwnProperty(i))continue;
  2258. shareItem=prefs.share[i];
  2259. if(shareItem.disabled)continue;
  2260. shareMark+=(
  2261. '<span class="pv-gallery-head-command-drop-list-item" data-site="'+i+'" style="\
  2262. background-image:url(\''+ shareItem.icon +'\');\
  2263. background-position:4px center;\
  2264. background-repeat:no-repeat;\
  2265. padding-left:24px;">'+shareItem.name+'</span>');
  2266. };
  2267.  
  2268. eleMaps['head-command-drop-list-share'].innerHTML=shareMark;
  2269.  
  2270. //分享下拉列表的点击处理
  2271. eleMaps['head-command-drop-list-share'].addEventListener('click',function(e){
  2272. if(e.button!=0)return;//左键
  2273. var target=e.target;
  2274. var site=dataset(target,'site');
  2275. if(!site)return;
  2276. var site_info=prefs.share[site];
  2277. var param=site_info.api.call(self.img,{
  2278. title:encodeURIComponent(document.title),
  2279. pic:encodeURIComponent(self.src),
  2280. url:encodeURIComponent(location.href),
  2281. });
  2282. if(!param)return;
  2283. 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');
  2284. },true);
  2285.  
  2286.  
  2287.  
  2288. var loadThumbsTimer;
  2289. eleMaps['sidebar-thumbnails-container'].addEventListener('scroll',function(e){//发生scroll事件时加载缩略图
  2290. clearTimeout(loadThumbsTimer);//加个延时,在连续触发的时候缓一缓。
  2291. loadThumbsTimer=setTimeout(function(){
  2292. self.loadThumb();
  2293. },200);
  2294. },false);
  2295.  
  2296. addWheelEvent(eleMaps['body'],function(e){//wheel事件
  2297. if(e.deltaZ!=0)return;//z轴
  2298. var target=e.target;
  2299. e.preventDefault();
  2300. if(eleMaps['sidebar-container'].contains(target)){//缩略图区滚动滚轮翻图片
  2301. var distance=self.thumbSpanOuterSize;
  2302.  
  2303. if(e.deltaY<0 || e.deltaX<0){//向上滚
  2304. distance=-distance;
  2305. };
  2306. thumbScrollbar.scrollBy(distance)
  2307. }else{//图片区域滚动
  2308. var distance=100;
  2309. if(e.deltaY!=0){//y轴
  2310. if(self.img.classList.contains('pv-gallery-img_zoom-out')){//图片可以缩小时,滚动图片,否则切换图片。
  2311. if(e.deltaY < 0){
  2312. distance=-distance;
  2313. };
  2314. if(eleMaps['img-scrollbar-h'].contains(target)){//如果在横向滚动条上。
  2315. imgScrollbarH.scrollBy(distance);
  2316. }else{
  2317. imgScrollbarV.scrollBy(distance);
  2318. };
  2319. }else{
  2320. e.deltaY < 0 ? self.selectPrevious() : self.selectNext();
  2321. };
  2322. }else{//x轴
  2323. if(e.deltaX < 0){
  2324. distance=-distance;
  2325. };
  2326. imgScrollbarH.scrollBy(distance);
  2327. };
  2328. };
  2329. },true);
  2330.  
  2331.  
  2332. //focus,blur;
  2333. addCusMouseEvent('mouseenter',container,function(){
  2334. this.focus();
  2335. });
  2336. addCusMouseEvent('mouseleave',container,function(){
  2337. this.blur();
  2338. });
  2339.  
  2340. //上下左右切换图片,空格键模拟滚动一页
  2341.  
  2342. var validKeyCode=[38,39,40,37,32,9]//上右下左,32空格,tab禁止焦点切换。
  2343. var keyDown;
  2344.  
  2345. document.addEventListener('keydown',function(e){
  2346. var keyCode=e.keyCode;
  2347. var index=validKeyCode.indexOf(keyCode);
  2348. if(index==-1)return;
  2349.  
  2350. var target=e.target;
  2351.  
  2352. if(!container.contains(target))return;//触发焦点不再gallery里面。
  2353. e.preventDefault();
  2354.  
  2355. if(keyCode==9)return;//tab键
  2356. if(keyCode==32){//32空格,模拟滚动一页
  2357. imgScrollbarV.scrollByPages(1);
  2358. return;
  2359. };
  2360.  
  2361. if(keyDown)return;//已按下。
  2362. keyDown=true;
  2363.  
  2364. var stop;
  2365. switch(index){
  2366. case 0:;
  2367. case 3:{
  2368. self.selectPrevious();
  2369. stop=self.simpleSlideShow(true);
  2370. }break;
  2371. case 1:;
  2372. case 2:{
  2373. self.selectNext();
  2374. stop=self.simpleSlideShow();
  2375. }break;
  2376. };
  2377.  
  2378. function keyUpHandler(e){
  2379. if(e.keyCode!=validKeyCode[index])return;
  2380. document.removeEventListener('keyup',keyUpHandler,false);
  2381. keyDown=false;
  2382. stop();
  2383. };
  2384. document.addEventListener('keyup',keyUpHandler,false);
  2385.  
  2386. },true);
  2387.  
  2388.  
  2389. var imgDraged;
  2390. eleMaps['img-parent'].addEventListener('mousedown',function(e){//如果图片尺寸大于屏幕的时候按住图片进行拖移
  2391. var target=e.target;
  2392. if(e.button!=0 || target.nodeName!='IMG')return;
  2393. var bigger=target.classList.contains('pv-gallery-img_zoom-out');//如果是大于屏幕
  2394.  
  2395. var oClient={
  2396. x:e.clientX,
  2397. y:e.clientY,
  2398. };
  2399.  
  2400. var oScroll={
  2401. left:self.imgScrollbarH.getScrolled(),
  2402. top:self.imgScrollbarV.getScrolled(),
  2403. };
  2404.  
  2405. var moveFiredCount=0;
  2406. var moveHandler=function(e){
  2407. moveFiredCount++;
  2408. if(moveFiredCount<2){//给个缓冲。。
  2409. return;
  2410. };
  2411. imgDraged=true;
  2412. if(bigger){
  2413. target.style.cursor= support.cssCursorValue.grabbing || 'pointer';
  2414. self.imgScrollbarV.scroll(oScroll.top-(e.clientY-oClient.y));
  2415. self.imgScrollbarH.scroll(oScroll.left-(e.clientX-oClient.x));
  2416. };
  2417. };
  2418.  
  2419. var upHandler=function(){
  2420. target.style.cursor='';
  2421.  
  2422. //拖曳之后阻止随后可能产生click事件产生的大小切换。
  2423. //确保在随后的click事件发生后执行
  2424. setTimeout(function(){
  2425. imgDraged=false;
  2426. },0);
  2427.  
  2428. document.removeEventListener('mousemove',moveHandler,true);
  2429. document.removeEventListener('mouseup',upHandler,true);
  2430. };
  2431.  
  2432. document.addEventListener('mousemove',moveHandler,true);
  2433. document.addEventListener('mouseup',upHandler,true);
  2434. },true);
  2435.  
  2436. eleMaps['img-parent'].addEventListener('click',function(e){//点击图片本身就行图片缩放处理
  2437. var target=e.target;
  2438. if(e.button!=0 || target.nodeName!='IMG')return;
  2439.  
  2440. if(imgDraged){//在拖动后触发的click事件,取消掉。免得一拖动完就立即进行的缩放。。。
  2441. imgDraged=false;
  2442. return;
  2443. };
  2444.  
  2445. if(target.classList.contains('pv-gallery-img_zoom-in')){//放大
  2446. self.fitContains=false;
  2447. var zoomX = typeof e.offsetX=='undefined' ? e.layerX : e.offsetX;
  2448. var zoomY = typeof e.offsetY=='undefined' ? e.layerY : e.offsetY;
  2449. var scaleX=zoomX/target.offsetWidth;
  2450. var scaleY=zoomY/target.offsetHeight;
  2451. self.fitToScreen({
  2452. x:scaleX,
  2453. y:scaleY,
  2454. });
  2455. }else if(target.classList.contains('pv-gallery-img_zoom-out')){
  2456. self.fitContains=true;
  2457. self.fitToScreen();
  2458. };
  2459. },true);
  2460.  
  2461.  
  2462. container.addEventListener('mousedown',function(e){//鼠标按在导航上,切换图片
  2463. if(e.button!=0)return;//左键
  2464. var target=e.target;
  2465. if(target.nodeName=='IMG')e.preventDefault();
  2466.  
  2467. var matched=true;
  2468. var stop;
  2469. switch(target){
  2470. case eleMaps['img-controler-pre']:;
  2471. case eleMaps['sidebar-controler-pre']:{//上一个
  2472. self.selectPrevious();
  2473. stop=self.simpleSlideShow(true);
  2474. }break;
  2475. case eleMaps['img-controler-next']:;
  2476. case eleMaps['sidebar-controler-next']:{//下一个
  2477. self.selectNext();
  2478. stop=self.simpleSlideShow();
  2479. }break;
  2480. default:{
  2481. matched=false;
  2482. }break;
  2483. };
  2484.  
  2485. function mouseUpHandler(e){
  2486. document.removeEventListener('mouseup',mouseUpHandler,true);
  2487. stop();
  2488. };
  2489.  
  2490. if(matched){
  2491. e.preventDefault();
  2492. document.addEventListener('mouseup',mouseUpHandler,true);
  2493. };
  2494. },false);
  2495.  
  2496. eleMaps['sidebar-thumbnails-container'].addEventListener('click',function(e){//点击缩略图切换
  2497. if(e.button!=0)return;//左键
  2498. var target=e.target;
  2499. var targetP;
  2500. if(!dataset(target,'src') && (targetP=target.parentNode) && !dataset(targetP,'src'))return;
  2501.  
  2502. self.select(targetP? targetP : target);
  2503. },false);
  2504.  
  2505. //点击读取错误的图片占位符重新读取
  2506. eleMaps['img_broken'].addEventListener('click',function(e){
  2507. if(self.isLoading){
  2508. self.select(self.errorSpan);
  2509. }else{
  2510. self.getImg(self.errorSpan);
  2511. };
  2512. },false);
  2513.  
  2514.  
  2515. eleMaps['head'].addEventListener('click',function(e){//顶栏上面的命令
  2516. if(e.button!=0)return;
  2517. var target=e.target;
  2518. if(eleMaps['head-command-close']==target){
  2519. self.close();
  2520. }else if(eleMaps['head-command-operate'].contains(target)){
  2521. imgReady(self.src,{
  2522. ready:function(){
  2523. new ImgWindowC(this);
  2524. },
  2525. });
  2526. }else if(eleMaps['head-command-collect'].contains(target)){
  2527. if(collection.favorite){
  2528. collection.remove();
  2529. }else{
  2530. collection.add();
  2531. };
  2532. }else if(eleMaps['head-command-exit-collection'].contains(target)){
  2533. collection.exit();
  2534. }else if(eleMaps['head-command-slide-show'].contains(target)){
  2535. slideShow.switchStatus();
  2536. slideShow.check();
  2537. };
  2538.  
  2539. },false);
  2540.  
  2541.  
  2542. //点击还原。
  2543. maximizeTrigger.addEventListener('click',function(e){
  2544. var target=e.target;
  2545. this.style.display='none';
  2546. if(target==this){
  2547. self.show();
  2548. self.resizeHandler();
  2549. }else{
  2550. self.minimized=false;
  2551. };
  2552. },true);
  2553.  
  2554.  
  2555. this._resizeHandler=this.resizeHandler.bind(this);
  2556.  
  2557. //插入动态生成的css数据。
  2558. this.globalSSheet.insertRule('.pv-gallery-sidebar-thumb-container{'+
  2559. ((isHorizontal ? 'width' : 'height') + ':' + (isHorizontal ? getComputedStyle(eleMaps['sidebar-thumbnails-container']).height : getComputedStyle(eleMaps['sidebar-thumbnails-container']).width)) +
  2560. '}',this.globalSSheet.cssRules.length);
  2561.  
  2562. this.forceRepaintTimes=0;
  2563.  
  2564. container.style.display='none';
  2565. this.shown=false;
  2566. },
  2567.  
  2568. getThumSpan:function(previous,relatedTarget){
  2569. var ret;
  2570. var rt = relatedTarget || this.selected;
  2571. if(!rt)return;
  2572. while((rt=previous ? rt.previousElementSibling : rt.nextElementSibling)){
  2573. if(rt.clientWidth!=0){
  2574. ret=rt;
  2575. break;
  2576. };
  2577. };
  2578. return ret;
  2579. },
  2580. selectPrevious:function(){
  2581. this.select(this.getThumSpan(true));
  2582. },
  2583. selectNext:function(){
  2584. this.select(this.getThumSpan());
  2585. },
  2586. select:function(ele,noTransition){
  2587. if(!ele || this.selected==ele)return;
  2588. if(this.selected){
  2589. this.selected.classList.remove(this.selectedClassName);
  2590. this.selected.classList.remove('pv-gallery-sidebar-thumb_selected');
  2591. };
  2592. ele.classList.add(this.selectedClassName);
  2593. ele.classList.add('pv-gallery-sidebar-thumb_selected');
  2594.  
  2595. this.selected=ele;
  2596. this.arrowVisib();
  2597.  
  2598. var self=this;
  2599. clearTimeout(this.loadImgTimer);
  2600. this.loadImgTimer=setTimeout(function(){//快速跳转的时候不要尝试读取图片。
  2601. self.loadImg(ele);
  2602. },200);
  2603.  
  2604. this.selectedIntoView(noTransition);
  2605. this.forceRepaint();
  2606. this.slideShow.run('select');
  2607. },
  2608. loadThumb:function(){//读取可视范围里面的缩略图
  2609.  
  2610. var self=this;
  2611.  
  2612. var pro=this.isHorizontal ? ['scrollLeft','clientWidth','offsetLeft','offsetWidth'] : ['scrollTop','clientHeight','offsetTop','offsetHeight'];
  2613. var thumbC=this.eleMaps['sidebar-thumbnails-container'];
  2614.  
  2615. var scrolled=thumbC[pro[0]];
  2616.  
  2617. var loadStopDis=scrolled + thumbC[pro[1]];
  2618.  
  2619. var imgSpans=this.selected.parentNode.children;
  2620. var span_i;
  2621. var spanOffset;
  2622. var thumb;
  2623.  
  2624. var i=0
  2625. while(span_i=imgSpans[i++]){
  2626. if(span_i.clientWidth==0)continue;//隐藏的
  2627.  
  2628. spanOffset=span_i[pro[2]];
  2629. if(spanOffset + span_i[pro[3]] <= scrolled)continue;//在滚动条上面了
  2630. if(spanOffset >= loadStopDis)break;//在滚动条下面了
  2631.  
  2632. if(dataset(span_i,'thumbLoaded'))continue;//已经加载了缩略图
  2633.  
  2634. thumb=document.createElement('img');
  2635. thumb.src=dataset(span_i,'thumbSrc') || dataset(span_i,'src') || prefs.icons.brokenImg_small;
  2636. //thumb.src='http://www.notexistwebsite.com/';
  2637. thumb.className='pv-gallery-sidebar-thumb';
  2638.  
  2639. dataset(span_i,'thumbLoaded','true')
  2640. span_i.appendChild(thumb);
  2641.  
  2642. imgReady(thumb,{
  2643. error:function(e){
  2644. this.src=prefs.icons.brokenImg_small;
  2645. },
  2646. });
  2647. };
  2648.  
  2649. },
  2650. selectedIntoView:function(noTransition){
  2651. var thumBC=this.eleMaps['sidebar-thumbnails-container'];
  2652. var pro=this.isHorizontal ? ['offsetLeft','clientWidth','offsetWidth'] : ['offsetTop','clientHeight','offsetHeight'] ;
  2653. //需要滚动的距离。
  2654. var needScrollDis= this.selected[pro[0]];
  2655. //尽可能的居中显示
  2656. var thumBCClient=thumBC[pro[1]];
  2657. var scrollCenter=Math.max((thumBCClient - this.selected[pro[2]])/2,0);
  2658.  
  2659. this.thumbScrollbar.scroll(needScrollDis - scrollCenter,false,!noTransition);
  2660. },
  2661. getImg:function(ele){
  2662. var src=dataset(ele,'src');
  2663. this.lastLoading=src;//记住最后读取的图片
  2664. this.isLoading=true;//表示选择的图片正在读取
  2665.  
  2666. var allLoading=this.allLoading;
  2667.  
  2668. if(allLoading.indexOf(src)!=-1){//在读取队列中。
  2669. return;
  2670. };
  2671.  
  2672. allLoading.push(src);
  2673. //console.log('正在读取中的图片:',cloneObject(allLoading));
  2674.  
  2675. //上一个读取中的图片,不是当前显示的。那么直接终止
  2676. var preImgR=this.imgReady;
  2677. if(preImgR && this.img){
  2678. if(preImgR.img.src!=this.src){
  2679. preImgR.abort();
  2680. preImgR.removeLI();
  2681. };
  2682. };
  2683.  
  2684.  
  2685. //显示读取指示器。
  2686. var loadingIndicator=ele.querySelector('.pv-gallery-sidebar-thumb-loading');
  2687. loadingIndicator.style.display='block';
  2688.  
  2689. var self=this;
  2690.  
  2691.  
  2692. this.imgReady=imgReady(src,{
  2693. ready:function(){
  2694. //从读取队列中删除自己
  2695. var index=allLoading.indexOf(src);
  2696. if(index!=-1){
  2697. allLoading.splice(index,1);
  2698. };
  2699.  
  2700. if(src!=self.lastLoading)return;
  2701.  
  2702. loadingIndicator.style.display='';
  2703. if(preImgR)preImgR.abort();
  2704. self.loadImg(this,ele);
  2705. },
  2706. loadEnd:function(e){//在loadend后开始预读。
  2707. //从读取队列中删除自己
  2708. var index=allLoading.indexOf(src);
  2709. if(index!=-1){
  2710. allLoading.splice(index,1);
  2711. };
  2712.  
  2713. if(src!=self.lastLoading)return;
  2714.  
  2715. if(e.type=='error'){
  2716. loadingIndicator.style.display='';
  2717. self.errorSpan=ele;
  2718. if(preImgR)preImgR.abort();
  2719. self.loadImg(this,ele,true);
  2720. };
  2721.  
  2722. self.slideShow.run('loadEnd');
  2723.  
  2724. //console.log(this,'预读开始');
  2725. if(prefs.gallery.preload){
  2726. if(self.preloading){//结束上次的预读。
  2727. self.preloading.abort();
  2728. };
  2729. self.preloading=new self.Preload(ele,self);
  2730. self.preloading.preload();
  2731. };
  2732. },
  2733. });
  2734.  
  2735. this.imgReady.removeLI=function(){
  2736. loadingIndicator.style.display='';
  2737. };
  2738.  
  2739. },
  2740. loadImg:function(img,relatedThumb,error){
  2741. if(img.nodeName!='IMG'){//先读取。
  2742. this.getImg(img);
  2743. return;
  2744. };
  2745.  
  2746. if(this.img){
  2747. this.img.style.display='none';
  2748. };
  2749.  
  2750. var imgNaturalSize={
  2751. h:img.naturalHeight,
  2752. w:img.naturalWidth,
  2753. };
  2754. this.imgNaturalSize=imgNaturalSize;
  2755.  
  2756. this.eleMaps['head-left-img-info-resolution'].textContent= imgNaturalSize.w + ' x ' + imgNaturalSize.h;
  2757.  
  2758. this.img=img;
  2759. this.src=img.src;
  2760. this.isLoading=false;
  2761.  
  2762. this.relatedThumb=relatedThumb;
  2763. img.className='pv-gallery-img';
  2764.  
  2765. if(error){
  2766. this.imgError=true;
  2767. this.img.style.display='none';
  2768. this.eleMaps['img_broken'].style.display='inline-block';
  2769. }else{
  2770. this.imgError=false;
  2771. this.eleMaps['img_broken'].style.display='';
  2772. if(!dataset(relatedThumb,'naturalSize')){
  2773. dataset(relatedThumb,'naturalSize',JSON.stringify(imgNaturalSize));
  2774. };
  2775. };
  2776.  
  2777. function styled(){
  2778. img.style.opacity=1;
  2779. img.style[support.cssTransform]='scale(1)';
  2780. };
  2781.  
  2782.  
  2783. if(prefs.gallery.transition){
  2784. setTimeout(styled,0);
  2785. }else{
  2786. styled();
  2787. };
  2788.  
  2789. this.eleMaps['img-parent'].appendChild(img);
  2790.  
  2791. this.fitContains=prefs.gallery.fitToScreen;//适应屏幕
  2792.  
  2793. this.fitToScreen({
  2794. x:0,
  2795. y:0,
  2796. });
  2797.  
  2798. this.collection.check();//检查是否在收藏里面。
  2799.  
  2800. },
  2801. fitToScreen:function(scale){
  2802.  
  2803. var container=this.eleMaps['img-content'];
  2804. var containerSize={
  2805. h:container.clientHeight,
  2806. w:container.clientWidth,
  2807. };
  2808.  
  2809. var img=this.img;
  2810.  
  2811. img.classList.remove('pv-gallery-img_zoom-in');
  2812. img.classList.remove('pv-gallery-img_zoom-out');
  2813.  
  2814. var imgSty=img.style;
  2815. imgSty.width='';
  2816. imgSty.height='';
  2817.  
  2818. var contentSSize={
  2819. h:container.scrollHeight,
  2820. w:container.scrollWidth,
  2821. };
  2822. var larger=contentSSize.h>containerSize.h || contentSSize.w>containerSize.w;
  2823.  
  2824. var scaled='100%';
  2825.  
  2826. if(this.fitContains){//适应屏幕
  2827. this.imgScrollbarV.hide();
  2828. this.imgScrollbarH.hide();
  2829. if(larger){
  2830. img.classList.add('pv-gallery-img_zoom-in');
  2831. if(contentSSize.h/contentSSize.w >=containerSize.h/containerSize.w){
  2832. var height=this.imgNaturalSize.h-(contentSSize.h - containerSize.h);
  2833. imgSty.height=height + 'px';
  2834. scaled=height/this.imgNaturalSize.h;
  2835. }else{
  2836. var width=this.imgNaturalSize.w-(contentSSize.w - containerSize.w);
  2837. imgSty.width=width + 'px';
  2838. scaled=width/this.imgNaturalSize.w;
  2839. };
  2840. scaled=(scaled*100).toFixed(2) + '%';
  2841. };
  2842. }else{//不做尺寸调整
  2843. this.imgScrollbarV.reset();
  2844. this.imgScrollbarH.reset();
  2845.  
  2846. if(larger){
  2847. img.classList.add('pv-gallery-img_zoom-out');
  2848. if(scale){//通过鼠标点击进行的切换。
  2849. this.imgScrollbarH.scroll(container.scrollWidth * scale.x - containerSize.w/2);
  2850. this.imgScrollbarV.scroll(container.scrollHeight * scale.y - containerSize.h/2);
  2851. };
  2852. };
  2853. };
  2854.  
  2855.  
  2856. var imgScaledInfo=this.eleMaps['head-left-img-info-scaling'];
  2857. imgScaledInfo.textContent='('+scaled+')';
  2858. if(scaled!='100%'){
  2859. imgScaledInfo.style.color='#E9CCCC';
  2860. }else{
  2861. imgScaledInfo.style.color='';
  2862. };
  2863.  
  2864. },
  2865.  
  2866.  
  2867. load:function(data, from, reload){
  2868. if(this.shown || this.minimized){//只允许打开一个,请先关掉当前已经打开的库
  2869.  
  2870. if(from){//frame发送过来的数据。
  2871. window.postMessage({
  2872. messageID:messageID,
  2873. command:'sendFail',
  2874. to:from,
  2875. },'*');
  2876. };
  2877.  
  2878. if(this.minimized){
  2879. alert('请先关掉当前已经打开的库');
  2880. flashEle(this.maximizeTrigger);
  2881. };
  2882. return;
  2883. };
  2884.  
  2885. var self=this;
  2886. if(from){//来自frame,获取这个frame所在的iframe标签。定位到图片的时候要用到。
  2887. window.postMessage({
  2888. messageID:messageID,
  2889. command:'getIframeObject',
  2890. windowId:from,
  2891. },'*');
  2892. document.addEventListener('pv-getIframeObject',function(e){
  2893. self.iframe=e.detail;
  2894. },true);
  2895. };
  2896.  
  2897. var unique=this.unique(data);
  2898. data=unique.data;
  2899. var index=unique.index;
  2900.  
  2901. if (reload && this.data.length >= data.length) {
  2902. alert('没有新增的图片');
  2903. return;
  2904. }
  2905.  
  2906. this.clear();//还原对象的一些修改,以便复用。
  2907. this.show(reload);
  2908.  
  2909. //console.log(data);
  2910.  
  2911. this.data=data;
  2912. this.from=from;//如果来自frame,那么这个from应该保存了那个frame的窗口id,便于以后通信。
  2913.  
  2914. var spanMark='';
  2915. var data_i;
  2916. var iStatisCopy=this.iStatisCopy;
  2917. for(var i=0,ii=data.length;i<ii;i++){
  2918. data_i=data[i];
  2919. iStatisCopy[data_i.type].count++;
  2920. spanMark +=
  2921. '<span class="pv-gallery-sidebar-thumb-container'+
  2922. '" data-type="' + data_i.type +
  2923. '" data-src="' + data_i.src +
  2924. '" data-thumb-src="' + data_i.imgSrc +
  2925. '" title="' + data_i.img.title +
  2926. '">'+
  2927. '<span class="pv-gallery-vertical-align-helper"></span>'+
  2928. '<span class="pv-gallery-sidebar-thumb-loading" title="正在读取中......"></span>'+
  2929. '</span>';
  2930. };
  2931.  
  2932.  
  2933. var thumbnails=this.eleMaps['sidebar-thumbnails-container'];
  2934. thumbnails.innerHTML=spanMark;
  2935.  
  2936. //写入类别数据。
  2937. var gallery=this.gallery;
  2938. var input,label,iStatisCopy_i;
  2939.  
  2940. for(var i in iStatisCopy){
  2941. if(!iStatisCopy.hasOwnProperty(i))continue;
  2942. iStatisCopy_i=iStatisCopy[i];
  2943. input=gallery.querySelector('#pv-gallery-head-command-drop-list-item-category-' + i);
  2944. input.checked=iStatisCopy_i.shown;
  2945. if(iStatisCopy_i.count==0){
  2946. input.disabled=true;
  2947. input.parentNode.classList.add('pv-gallery-head-command-drop-list-item_disabled');
  2948. }else{
  2949. input.disabled=false;
  2950. input.parentNode.classList.remove('pv-gallery-head-command-drop-list-item_disabled');
  2951. };
  2952.  
  2953. label=gallery.querySelector('label[for="pv-gallery-head-command-drop-list-item-category-' + i + '"]');
  2954. label.textContent=label.textContent.replace(/(.*)/i,'') + '(' + iStatisCopy_i.count + ')';
  2955. };
  2956.  
  2957. this.imgSpans=thumbnails.children;
  2958.  
  2959. this.thumbScrollbar.reset();
  2960. this.select(this.imgSpans[index],true);
  2961.  
  2962. this.runOnce();
  2963.  
  2964. this.switchThumbVisible();
  2965.  
  2966. },
  2967. clear:function(){
  2968.  
  2969. this.allLoading=[];//读取中的图片数组
  2970. this.iStatisCopy=cloneObject(this.imgStatistics,true);//图片统计副本
  2971. this.selected==null;
  2972. if(this.img){
  2973. this.img.style.display='none';
  2974. this.img=null;
  2975. };
  2976. //读取错误的图片占位符
  2977. this.eleMaps['img_broken'].style.display='';
  2978. //清空dom
  2979. this.eleMaps['sidebar-thumbnails-container'].innerHTML='';
  2980. this.eleMaps['head-left-img-info-resolution'].textContent='0 x 0';
  2981. this.eleMaps['head-left-img-info-scaling'].textContent='(100%)';
  2982. //隐藏滚动条
  2983. this.imgScrollbarV.hide();
  2984. this.imgScrollbarH.hide();
  2985. this.thumbScrollbar.hide();
  2986. //重置style;
  2987. this.thumbVisibleStyle.textContent='';
  2988. },
  2989. // --------- 我添加的部分 start ----------------
  2990. reload: function() {
  2991. // 函数在 LoadingAnimC 中
  2992. var data = this.getAllValidImgs();
  2993. // 设置当前选中的图片
  2994. data.target = {
  2995. src: this.selected.dataset.src
  2996. };
  2997.  
  2998. this.close(true);
  2999.  
  3000. this.load(data, null, true);
  3001. },
  3002. getAllValidImgs:function(){
  3003. var imgs = document.getElementsByTagName('img'),
  3004. container = document.querySelector('.pv-gallery-container'),
  3005. preloadContainer = document.querySelector('.pv-gallery-preloaded-img-container'),
  3006. validImgs = [];
  3007.  
  3008. arrayFn.forEach.call(imgs, function(img, index, imgs) {
  3009. // 排除库里面的图片
  3010. if (container.contains(img) || preloadContainer.contains(img)) return;
  3011.  
  3012. var result = findPic(img);
  3013. if (result) {
  3014. validImgs.push(result);
  3015. };
  3016. });
  3017. return validImgs;
  3018. },
  3019. scrollToEndAndReload: function() {
  3020. if (!prefs.gallery.autoScrollAndReload) {
  3021. return;
  3022. }
  3023.  
  3024. window.scrollTo(0, 99999);
  3025.  
  3026. // 滚动主窗口到最底部,然后自动重载库的图片,还有bug,有待进一步测试
  3027. window.removeEventListener('scroll', this.scrolled, false);
  3028. var self = this;
  3029. this.scrolled = function() {
  3030. clearTimeout(self.reloadTimeout);
  3031. self.reloadTimeout = setTimeout(function(){
  3032. window.removeEventListener('scroll', self.scrolled, false);
  3033. self.reload();
  3034. }, 500);
  3035. };
  3036.  
  3037. window.addEventListener('scroll', this.scrolled, false);
  3038. },
  3039. // --------- 我添加的部分 end ----------------
  3040.  
  3041. unique:function(data){
  3042. var imgSrc=data.target.src;
  3043.  
  3044. var data_i,
  3045. data_i_src,
  3046. dataSrcs=[];
  3047.  
  3048. var index;
  3049.  
  3050. for(var i=0,ii=data.length;i<ii;i++){
  3051. data_i=data[i];
  3052. data_i_src=data_i.src;
  3053. if(dataSrcs.indexOf(data_i_src)!=-1){//已经存在
  3054. data.splice(i,1);//移除
  3055. i--;
  3056. ii--;
  3057. continue;
  3058. };
  3059. dataSrcs.push(data_i_src);
  3060.  
  3061. if(imgSrc==data_i_src){
  3062. index=i;
  3063. };
  3064. };
  3065.  
  3066. if(typeof index =='undefined'){
  3067. index=0;
  3068. data.unshift(data.target);
  3069. };
  3070.  
  3071. delete data.target;
  3072.  
  3073. return {
  3074. data:data,
  3075. index:index,
  3076. };
  3077. },
  3078. show:function(reload){
  3079. this.shown=true;
  3080. galleryMode=true;
  3081.  
  3082. if (!reload) {
  3083. var des=document.documentElement.style;
  3084. this.deOverflow={
  3085. x:des.overflowX,
  3086. y:des.overflowY,
  3087. };
  3088. des.overflow='hidden';
  3089. this.gallery.style.display='';
  3090. this.gallery.focus();
  3091. window.addEventListener('resize',this._resizeHandler,true);
  3092. }
  3093. },
  3094. close:function(reload){
  3095. this.shown=false;
  3096. this.minimized=false;
  3097.  
  3098. if (!reload) {
  3099. galleryMode=false;
  3100. this.gallery.blur();
  3101. this.gallery.style.display='none';
  3102. var des=document.documentElement.style;
  3103. des.overflowX=this.deOverflow.x;
  3104. des.overflowY=this.deOverflow.y;
  3105. this.slideShow.exit();
  3106. this.collection.exit();
  3107. window.removeEventListener('resize',this._resizeHandler,true);
  3108. }
  3109. },
  3110. runOnce:function(){//运行一次来获取某些数据。
  3111. var thumbSpanCS=getComputedStyle(this.selected);
  3112. this.thumbSpanOuterSize=this.isHorizontal?
  3113. this.selected.offsetWidth + parseFloat(thumbSpanCS.marginLeft) + parseFloat(thumbSpanCS.marginRight) :
  3114. this.selected.offsetHeight + parseFloat(thumbSpanCS.marginTop) + parseFloat(thumbSpanCS.marginBottom);
  3115.  
  3116.  
  3117. //console.log(this.thumbSpanOuterSize);
  3118.  
  3119. this.runOnce=function(){
  3120. };
  3121. },
  3122.  
  3123. minimize:function(){
  3124. this.close();
  3125. this.maximizeTrigger.style.display='block';
  3126. this.minimized=true;
  3127. },
  3128. navigateToImg:function(targetImg){
  3129. targetImg.scrollIntoView();//先调用原方法,可以让overflow hidden的滚动出来。
  3130.  
  3131. //让图片近可能的居中
  3132. var imgBCRect=getContentClientRect(targetImg);
  3133. var wSize=getWindowSize();
  3134.  
  3135. window.scrollBy(imgBCRect.left - (wSize.w - imgBCRect.width)/2,
  3136. imgBCRect.top - (wSize.h - imgBCRect.height)/2);
  3137.  
  3138. },
  3139. switchThumbVisible:function(){
  3140. var style=this.thumbVisibleStyle;
  3141. var count=0;
  3142. var styleText=[];
  3143. var iStatisCopy=this.iStatisCopy;
  3144. var iStatisCopy_i;
  3145.  
  3146. for(var i in iStatisCopy){
  3147. if(!iStatisCopy.hasOwnProperty(i))continue;
  3148. iStatisCopy_i=iStatisCopy[i];
  3149. if(iStatisCopy_i.shown){
  3150. count+=iStatisCopy_i.count;
  3151. }else{
  3152. styleText.push('.pv-gallery-sidebar-thumb-container[data-type="'+i+'"]');
  3153. };
  3154. };
  3155.  
  3156. //写入style;
  3157. style.textContent=styleText.join(',') + '{\
  3158. display:none !important;\
  3159. }';
  3160.  
  3161. //初始化缩略图区的滚动条
  3162. this.thumbScrollbar.reset();
  3163. this.arrowVisib();
  3164.  
  3165. //载入缩略图
  3166. this.loadThumb();
  3167. },
  3168. forceRepaint:function(){//解决opera的fixed元素,当滚动条不再最高处的时候,不重绘fixed元素的问题。
  3169. clearTimeout(this.forceRepaintTimer);
  3170. var self=this;
  3171. this.forceRepaintTimer=setTimeout(function(){
  3172. if(envir.opera){
  3173. self.forceRepaintTimes % 2 ==0 ? window.scrollBy(0,1) : window.scrollBy(0,-1);
  3174. self.forceRepaintTimes++;
  3175. };
  3176. },333);
  3177. },
  3178. resizeHandler:function(){//窗口变化时,调整一些东西。
  3179. this.thumbScrollbar.reset();
  3180. //this.selectedIntoView();
  3181. this.fitToScreen();
  3182. this.loadThumb();
  3183. },
  3184. arrowVisib:function(){//当当前选择元素的前面或者后面没有元素的时候隐藏控制箭头
  3185.  
  3186. var icps=this.eleMaps['img-controler-pre'].style;
  3187. var icns=this.eleMaps['img-controler-next'].style;
  3188. var scps=this.eleMaps['sidebar-controler-pre'].style;
  3189. var scns=this.eleMaps['sidebar-controler-next'].style;
  3190.  
  3191. //下一张的箭头
  3192. if(this.getThumSpan()){
  3193. icns.display='';
  3194. scns.display='';
  3195. }else{
  3196. icns.display='none';
  3197. scns.display='none';
  3198.  
  3199. this.scrollToEndAndReload();
  3200. };
  3201.  
  3202. //上一张的箭头
  3203. if(this.getThumSpan(true)){
  3204. icps.display='';
  3205. scps.display='';
  3206. }else{
  3207. icps.display='none';
  3208. scps.display='none';
  3209. };
  3210. },
  3211. simpleSlideShow:function(backward,interval){
  3212. clearInterval(this.slideShowInterval);//幻灯播放,只允许存在一个,否则得乱套
  3213.  
  3214. var self=this;
  3215. var slideShowInterval=setInterval(function(){
  3216. var before=self.selected;
  3217. backward ? self.selectPrevious() : self.selectNext();
  3218. if(before == self.selected){//没有下一个元素了。。
  3219. stop();
  3220. };
  3221. },(interval? interval : 800));
  3222.  
  3223. this.slideShowInterval=slideShowInterval;
  3224.  
  3225. function stop(){
  3226. clearInterval(slideShowInterval);
  3227. };
  3228.  
  3229. return stop;
  3230. },
  3231.  
  3232.  
  3233. Preload:function(ele,oriThis){
  3234. this.ele=ele;
  3235. this.oriThis=oriThis;//主this
  3236. this.init();
  3237. },
  3238. Scrollbar:function(scrollbar,container,isHorizontal){
  3239. this.scrollbar=scrollbar;
  3240. this.container=container;
  3241. this.isHorizontal=isHorizontal
  3242. this.init();
  3243. },
  3244.  
  3245. addStyle:function(){
  3246. var style=document.createElement('style');
  3247. style.type='text/css';
  3248. style.textContent='\
  3249. /*最外层容器*/\
  3250. .pv-gallery-container {\
  3251. position: fixed;\
  3252. top: 0;\
  3253. left: 0;\
  3254. width: 100%;\
  3255. height: 100%;\
  3256. min-width:none;\
  3257. min-height:none;\
  3258. padding: 0;\
  3259. margin: 0;\
  3260. border: none;\
  3261. z-index:899999999;\
  3262. background-color: transparent;\
  3263. }\
  3264. /*全局border-box*/\
  3265. .pv-gallery-container span{\
  3266. -moz-box-sizing: border-box;\
  3267. box-sizing: border-box;\
  3268. line-height: 1.6;\
  3269. }\
  3270. .pv-gallery-container * {\
  3271. font-size: 14px;\
  3272. }\
  3273. /*点击还原的工具条*/\
  3274. .pv-gallery-maximize-trigger{\
  3275. position:fixed;\
  3276. bottom:15px;\
  3277. left:15px;\
  3278. display:none;\
  3279. background:#000;\
  3280. opacity:0.6;\
  3281. padding-left:10px;\
  3282. font-size:16px;\
  3283. line-height:0;\
  3284. color:white;\
  3285. cursor:pointer;\
  3286. box-shadow:3px 3px 0 0 #333;\
  3287. z-index:899999998;\
  3288. }\
  3289. .pv-gallery-maximize-trigger:hover{\
  3290. opacity:0.9;\
  3291. }\
  3292. .pv-gallery-maximize-trigger-close{\
  3293. display:inline-block;\
  3294. padding-left:10px;\
  3295. vertical-align:middle;\
  3296. height:30px;\
  3297. padding:10px 0;\
  3298. width:24px;\
  3299. background:url("'+prefs.icons.loadingCancle+'") center no-repeat;\
  3300. }\
  3301. .pv-gallery-maximize-trigger-close:hover{\
  3302. background-color:#333;\
  3303. }\
  3304. /*顶栏*/\
  3305. .pv-gallery-head {\
  3306. position: absolute;\
  3307. top: 0;\
  3308. left: 0;\
  3309. width: 100%;\
  3310. height:30px;\
  3311. z-index:1;\
  3312. background-color:rgb(0,0,0);\
  3313. border:none;\
  3314. border-bottom:1px solid #333333;\
  3315. text-align:right;\
  3316. line-height:0;\
  3317. font-size: 14px;\
  3318. color:#757575;\
  3319. padding-right:42px;\
  3320. }\
  3321. .pv-gallery-head > span{\
  3322. vertical-align:middle;\
  3323. }\
  3324. /*顶栏左边*/\
  3325. .pv-gallery-head-float-left{\
  3326. float:left;\
  3327. height:100%;\
  3328. text-align:left;\
  3329. padding-left:5px;\
  3330. }\
  3331. .pv-gallery-head-float-left > span{\
  3332. display:inline-block;\
  3333. height:100%;\
  3334. vertical-align:middle;\
  3335. }\
  3336. .pv-gallery-head-float-left > span > *{\
  3337. vertical-align:middle;\
  3338. }\
  3339. .pv-gallery-head-left-img-info{\
  3340. cursor:help;\
  3341. }\
  3342. /*顶栏里面的按钮样式-开始*/\
  3343. .pv-gallery-head-command{\
  3344. display:inline-block;\
  3345. cursor:pointer;\
  3346. height:100%;\
  3347. padding:0 8px;\
  3348. text-align:center;\
  3349. position:relative;\
  3350. z-index:1;\
  3351. vertical-align:middle;\
  3352. -o-user-select: none;\
  3353. -ms-user-select: none;\
  3354. -webkit-user-select: none;\
  3355. -moz-user-select: -moz-none;\
  3356. user-select: none;\
  3357. }\
  3358. /*辅助点击事件的生成,countdown*/\
  3359. .pv-gallery-head-command_overlayer{\
  3360. top:0;\
  3361. left:0;\
  3362. right:0;\
  3363. bottom:0;\
  3364. position:absolute;\
  3365. opacity:0;\
  3366. }\
  3367. .pv-gallery-head-command > *{\
  3368. vertical-align:middle;\
  3369. }\
  3370. .pv-gallery-head-command-close{\
  3371. position:absolute;\
  3372. top:0;\
  3373. right:0;\
  3374. width:40px;\
  3375. border-left: 1px solid #333333;\
  3376. background:transparent no-repeat center;\
  3377. background-image:url("'+prefs.icons.loadingCancle+'");\
  3378. }\
  3379. .pv-gallery-head-command-slide-show-countdown{\
  3380. font-size:0.8em;\
  3381. }\
  3382. .pv-gallery-head-command-slide-show-button{\
  3383. border-radius:36px;\
  3384. display:inline-block;\
  3385. width:18px;\
  3386. height:18px;\
  3387. border:2px solid #757575;\
  3388. margin-right:3px;\
  3389. line-height:0;\
  3390. }\
  3391. .pv-gallery-head-command-slide-show-button-inner{\
  3392. display:inline-block;\
  3393. border:none;\
  3394. border-top:4px solid transparent;\
  3395. border-bottom:4px solid transparent;\
  3396. border-left:8px solid #757575;\
  3397. vertical-align:middle;\
  3398. }\
  3399. .pv-gallery-head-command-slide-show-button-inner_stop{\
  3400. border-color:#757575;\
  3401. }\
  3402. .pv-gallery-head-command-collect-icon{\
  3403. display:inline-block;\
  3404. height:20px;\
  3405. width:20px;\
  3406. background:transparent url("' + prefs.icons.fivePointedStar + '") 0 0 no-repeat;\
  3407. }\
  3408. .pv-gallery-head-command-collect-icon ~ .pv-gallery-head-command-collect-text::after{\
  3409. content:"收藏";\
  3410. }\
  3411. .pv-gallery-head-command-collect-favorite > .pv-gallery-head-command-collect-icon{\
  3412. background-position:-40px 0 !important;\
  3413. }\
  3414. .pv-gallery-head-command-collect-favorite > .pv-gallery-head-command-collect-text::after{\
  3415. content:"已收藏";\
  3416. }\
  3417. .pv-gallery-head-command-exit-collection{\
  3418. color:#939300 !important;\
  3419. display:none;\
  3420. }\
  3421. .pv-gallery-head-command:hover{\
  3422. background-color:#272727;\
  3423. color:#ccc;\
  3424. }\
  3425. /*droplist*/\
  3426. .pv-gallery-head-command-drop-list{\
  3427. position:absolute;\
  3428. right:0;\
  3429. display:none;\
  3430. box-shadow:0 0 3px #808080;\
  3431. background-color:#272727;\
  3432. line-height: 1.6;\
  3433. text-align:left;\
  3434. padding:10px;\
  3435. color:#ccc;\
  3436. margin-top:-1px;\
  3437. }\
  3438. .pv-gallery-head-command-drop-list-item{\
  3439. display:block;\
  3440. padding:2px 5px;\
  3441. cursor:pointer;\
  3442. white-space:nowrap;\
  3443. }\
  3444. .pv-gallery-head-command-drop-list-item-collect-description{\
  3445. cursor:default;\
  3446. }\
  3447. .pv-gallery-head-command-drop-list-item-collect-description > textarea{\
  3448. resize:both;\
  3449. width:auto;\
  3450. height:auto;\
  3451. }\
  3452. .pv-gallery-head-command-drop-list-item_disabled{\
  3453. color:#757575;\
  3454. }\
  3455. .pv-gallery-head-command-drop-list-item input + *{\
  3456. padding-left:3px;\
  3457. }\
  3458. .pv-gallery-head-command-drop-list-item input[type=number]{\
  3459. text-align:left;\
  3460. max-width:50px;\
  3461. height:20px;\
  3462. }\
  3463. .pv-gallery-head-command-drop-list-item > * {\
  3464. vertical-align:middle;\
  3465. }\
  3466. .pv-gallery-head-command-drop-list-item label {\
  3467. font-weight: normal;\
  3468. }\
  3469. .pv-gallery-head-command-drop-list-item:hover{\
  3470. background-color:#404040;\
  3471. }\
  3472. /*container*/\
  3473. .pv-gallery-head-command-container{\
  3474. display:inline-block;\
  3475. height:100%;\
  3476. position:relative;\
  3477. }\
  3478. /* after伪类生成标识下拉菜单的三角图标*/\
  3479. .pv-gallery-head-command-container > .pv-gallery-head-command::after{\
  3480. content:"";\
  3481. display:inline-block;\
  3482. vertical-align:middle;\
  3483. border:none;\
  3484. border-top:7px solid #757575;\
  3485. border-left:5px solid transparent;\
  3486. border-right:5px solid transparent;\
  3487. margin-left:5px;\
  3488. -moz-transition:all 0.3s ease-in-out 0s;\
  3489. -webkit-transition:all 0.3s ease-in-out 0s;\
  3490. transition:all 0.3s ease-in-out 0s;\
  3491. }\
  3492. .pv-gallery-head-command-container:hover{\
  3493. box-shadow:0 0 3px #808080;\
  3494. }\
  3495. .pv-gallery-head-command-container:hover > .pv-gallery-head-command{\
  3496. background-color:#272727;\
  3497. color:#ccc;\
  3498. }\
  3499. .pv-gallery-head-command-container:hover > .pv-gallery-head-command::after{\
  3500. -webkit-transform:rotate(180deg);\
  3501. -moz-transform:rotate(180deg);\
  3502. transform:rotate(180deg);\
  3503. border-top:7px solid #ccc;\
  3504. }\
  3505. .pv-gallery-head-command-container:hover .pv-gallery-head-command-collect-icon{\
  3506. background-position:-20px 0;\
  3507. }\
  3508. .pv-gallery-head-command-container:hover .pv-gallery-head-command-slide-show-button{\
  3509. border-color:#ccc;\
  3510. }\
  3511. .pv-gallery-head-command-container:hover .pv-gallery-head-command-slide-show-button-inner{\
  3512. border-left-color:#ccc;\
  3513. }\
  3514. .pv-gallery-head-command-container:hover .pv-gallery-head-command-slide-show-button-inner_stop{\
  3515. border-color:#ccc;\
  3516. }\
  3517. .pv-gallery-head-command-container:hover > .pv-gallery-head-command-drop-list{\
  3518. display:block;\
  3519. }\
  3520. /*顶栏里面的按钮样式-结束*/\
  3521. .pv-gallery-body {\
  3522. display: block;\
  3523. height: 100%;\
  3524. width: 100%;\
  3525. margin: 0;\
  3526. padding: 0;\
  3527. border: none;\
  3528. border-top: 30px solid transparent;\
  3529. position: relative;\
  3530. background-clip: padding-box;\
  3531. z-index:0;\
  3532. }\
  3533. .pv-gallery-img-container {\
  3534. display: block;\
  3535. padding: 0;\
  3536. margin: 0;\
  3537. border: none;\
  3538. height: 100%;\
  3539. width: 100%;\
  3540. background-clip: padding-box;\
  3541. background-color: rgba(20,20,20,0.96);\
  3542. position:relative;\
  3543. }\
  3544. .pv-gallery-img-container-top {\
  3545. border-top: '+ prefs.gallery.sidebarSize +'px solid transparent;\
  3546. }\
  3547. .pv-gallery-img-container-right {\
  3548. border-right: '+ prefs.gallery.sidebarSize +'px solid transparent;\
  3549. }\
  3550. .pv-gallery-img-container-bottom {\
  3551. border-bottom: '+ prefs.gallery.sidebarSize +'px solid transparent;\
  3552. }\
  3553. .pv-gallery-img-container-left {\
  3554. border-left: '+ prefs.gallery.sidebarSize +'px solid transparent;\
  3555. }\
  3556. /*大图区域的切换控制按钮*/\
  3557. .pv-gallery-img-controler{\
  3558. position:absolute;\
  3559. top:50%;\
  3560. height:60px;\
  3561. width:50px;\
  3562. margin-top:-30px;\
  3563. cursor:pointer;\
  3564. opacity:0.3;\
  3565. z-index:1;\
  3566. }\
  3567. .pv-gallery-img-controler-pre{\
  3568. background:rgba(70,70,70,0.5) url("'+prefs.icons.arrowLeft+'") no-repeat center;\
  3569. left:10px;\
  3570. }\
  3571. .pv-gallery-img-controler-next{\
  3572. background:rgba(70,70,70,0.5) url("'+prefs.icons.arrowRight+'") no-repeat center;\
  3573. right:10px;\
  3574. }\
  3575. .pv-gallery-img-controler:hover{\
  3576. background-color:rgba(140,140,140,0.5);\
  3577. opacity:0.9;\
  3578. z-index:2;\
  3579. }\
  3580. /*滚动条样式--开始*/\
  3581. .pv-gallery-scrollbar-h,\
  3582. .pv-gallery-scrollbar-v{\
  3583. display:none;\
  3584. z-index:1;\
  3585. opacity:0.3;\
  3586. position:absolute;\
  3587. margin:0;\
  3588. padding:0;\
  3589. border:none;\
  3590. }\
  3591. .pv-gallery-scrollbar-h{\
  3592. bottom:10px;\
  3593. left:0;\
  3594. right:0;\
  3595. height:10px;\
  3596. margin:0 2px;\
  3597. }\
  3598. .pv-gallery-scrollbar-v{\
  3599. top:0;\
  3600. bottom:0;\
  3601. right:10px;\
  3602. width:10px;\
  3603. margin:2px 0;\
  3604. }\
  3605. .pv-gallery-scrollbar-h:hover{\
  3606. height:15px;\
  3607. }\
  3608. .pv-gallery-scrollbar-v:hover{\
  3609. width:15px;\
  3610. }\
  3611. .pv-gallery-scrollbar-h:hover,\
  3612. .pv-gallery-scrollbar-v:hover{\
  3613. opacity:0.9;\
  3614. z-index:2;\
  3615. }\
  3616. .pv-gallery-scrollbar-h-track,\
  3617. .pv-gallery-scrollbar-v-track{\
  3618. position:absolute;\
  3619. top:0;\
  3620. left:0;\
  3621. right:0;\
  3622. bottom:0;\
  3623. background-color:rgba(100,100,100,1);\
  3624. border:2px solid transparent;\
  3625. }\
  3626. .pv-gallery-scrollbar-h-handle,\
  3627. .pv-gallery-scrollbar-v-handle{\
  3628. position:absolute;\
  3629. background-color:black;\
  3630. }\
  3631. .pv-gallery-scrollbar-h-handle{\
  3632. height:100%;\
  3633. }\
  3634. .pv-gallery-scrollbar-v-handle{\
  3635. width:100%;\
  3636. }\
  3637. .pv-gallery-scrollbar-h-handle:hover,\
  3638. .pv-gallery-scrollbar-v-handle:hover{\
  3639. background-color:#502121;\
  3640. }\
  3641. .pv-gallery-scrollbar-h-handle:active,\
  3642. .pv-gallery-scrollbar-v-handle:active{\
  3643. background-color:#391A1A;\
  3644. }\
  3645. /*滚动条样式--结束*/\
  3646. .pv-gallery-img-content{\
  3647. display:block;\
  3648. width:100%;\
  3649. height:100%;\
  3650. overflow:hidden;\
  3651. text-align:center;\
  3652. padding:0;\
  3653. border:none;\
  3654. margin:0;\
  3655. line-height:0;\
  3656. font-size:0;\
  3657. white-space:nowrap;\
  3658. }\
  3659. .pv-gallery-img-parent{\
  3660. display:inline-block;\
  3661. vertical-align:middle;\
  3662. line-height:0;\
  3663. }\
  3664. .pv-gallery-img_broken{\
  3665. display:none;\
  3666. cursor:pointer;\
  3667. }\
  3668. .pv-gallery-img{\
  3669. position:relative;\/*辅助e.layerX,layerY*/\
  3670. display:inline-block;\
  3671. vertical-align:middle;\
  3672. width:auto;\
  3673. height:auto;\
  3674. padding:0;\
  3675. border:5px solid #313131;\
  3676. margin:10px;\
  3677. opacity:0.6;\
  3678. -webkit-transform:scale(0.9);\
  3679. -moz-transform:scale(0.9);\
  3680. transform:scale(0.9);\
  3681. '+
  3682. (prefs.gallery.transition ? ('\
  3683. -webkit-transition: opacity 0.15s ease-in-out,\
  3684. -webkit-transform 0.1s ease-in-out;\
  3685. -moz-transition: opacity 0.15s ease-in-out,\
  3686. -moz-transform 0.1s ease-in-out;\
  3687. transition: opacity 0.15s ease-in-out,\
  3688. transform 0.1s ease-in-out;\
  3689. ') : '') + '\
  3690. }\
  3691. .pv-gallery-img_zoom-out{\
  3692. cursor:'+support.cssCursorValue.zoomOut+';\
  3693. }\
  3694. .pv-gallery-img_zoom-in{\
  3695. cursor:'+support.cssCursorValue.zoomIn+';\
  3696. }\
  3697. .pv-gallery-sidebar-toggle{\
  3698. position:absolute;\
  3699. line-height:0;\
  3700. text-align:center;\
  3701. background-color:rgb(0,0,0);\
  3702. color:#757575;\
  3703. border:1px solid #333;\
  3704. white-space:nowrap;\
  3705. cursor:pointer;\
  3706. z-index:1;\
  3707. display:none;\
  3708. }\
  3709. .pv-gallery-sidebar-toggle:hover{\
  3710. color:#ccc;\
  3711. }\
  3712. .pv-gallery-sidebar-toggle-h{\
  3713. width:80px;\
  3714. margin-left:-40px;\
  3715. left:50%;\
  3716. }\
  3717. .pv-gallery-sidebar-toggle-v{\
  3718. height:80px;\
  3719. margin-top:-40px;\
  3720. top:50%;\
  3721. }\
  3722. .pv-gallery-sidebar-toggle-top{\
  3723. top:-5px;\
  3724. }\
  3725. .pv-gallery-sidebar-toggle-right{\
  3726. right:-5px;\
  3727. }\
  3728. .pv-gallery-sidebar-toggle-bottom{\
  3729. bottom:-5px;\
  3730. }\
  3731. .pv-gallery-sidebar-toggle-left{\
  3732. left:-5px;\
  3733. }\
  3734. .pv-gallery-sidebar-toggle-content{\
  3735. display:inline-block;\
  3736. vertical-align:middle;\
  3737. white-space:normal;\
  3738. word-wrap:break-word;\
  3739. overflow-wrap:break-word;\
  3740. line-height:1.1;\
  3741. font-size:12px;\
  3742. text-align:center;\
  3743. margin:2px;\
  3744. }\
  3745. .pv-gallery-sidebar-toggle-content-v{\
  3746. width:1.1em;\
  3747. }\
  3748. /*侧边栏开始*/\
  3749. .pv-gallery-sidebar-container {\
  3750. position: absolute;\
  3751. background-color:rgb(0,0,0);\
  3752. padding:5px;\
  3753. border:none;\
  3754. margin:none;\
  3755. text-align:center;\
  3756. line-height:0;\
  3757. white-space:nowrap;\
  3758. -o-user-select: none;\
  3759. -webkit-user-select: none;\
  3760. -moz-user-select: -moz-none;\
  3761. user-select: none;\
  3762. }\
  3763. .pv-gallery-sidebar-container-h {\
  3764. height: '+ prefs.gallery.sidebarSize +'px;\
  3765. width: 100%;\
  3766. }\
  3767. .pv-gallery-sidebar-container-v {\
  3768. width: '+ prefs.gallery.sidebarSize +'px;\
  3769. height: 100%;\
  3770. }\
  3771. .pv-gallery-sidebar-container-top {\
  3772. top: 0;\
  3773. left: 0;\
  3774. border-bottom:1px solid #333333;\
  3775. }\
  3776. .pv-gallery-sidebar-container-right {\
  3777. top: 0;\
  3778. right: 0;\
  3779. border-left:1px solid #333333;\
  3780. }\
  3781. .pv-gallery-sidebar-container-bottom {\
  3782. bottom: 0;\
  3783. left: 0;\
  3784. border-top:1px solid #333333;\
  3785. }\
  3786. .pv-gallery-sidebar-container-left {\
  3787. top: 0;\
  3788. left: 0;\
  3789. border-right:1px solid #333333;\
  3790. }\
  3791. .pv-gallery-sidebar-content {\
  3792. display: inline-block;\
  3793. margin: 0;\
  3794. padding: 0;\
  3795. border: none;\
  3796. background-clip: padding-box;\
  3797. vertical-align:middle;\
  3798. position:relative;\
  3799. text-align:left;\
  3800. }\
  3801. .pv-gallery-sidebar-content-h {\
  3802. height: 100%;\
  3803. width: 90%;\
  3804. border-left: 40px solid transparent;\
  3805. border-right: 40px solid transparent;\
  3806. }\
  3807. .pv-gallery-sidebar-content-v {\
  3808. height: 90%;\
  3809. width: 100%;\
  3810. border-top: 40px solid transparent;\
  3811. border-bottom: 40px solid transparent;\
  3812. }\
  3813. .pv-gallery-sidebar-controler{\
  3814. cursor:pointer;\
  3815. position:absolute;\
  3816. background:rgba(255,255,255,0.1) no-repeat center;\
  3817. }\
  3818. .pv-gallery-sidebar-controler:hover{\
  3819. background-color:rgba(255,255,255,0.3);\
  3820. }\
  3821. .pv-gallery-sidebar-controler-pre-h,\
  3822. .pv-gallery-sidebar-controler-next-h{\
  3823. top:0;\
  3824. width:36px;\
  3825. height:100%;\
  3826. }\
  3827. .pv-gallery-sidebar-controler-pre-v,\
  3828. .pv-gallery-sidebar-controler-next-v{\
  3829. left:0;\
  3830. width:100%;\
  3831. height:36px;\
  3832. }\
  3833. .pv-gallery-sidebar-controler-pre-h {\
  3834. left: -40px;\
  3835. background-image: url("'+prefs.icons.arrowLeft+'");\
  3836. }\
  3837. .pv-gallery-sidebar-controler-next-h {\
  3838. right: -40px;\
  3839. background-image: url("'+prefs.icons.arrowRight+'");\
  3840. }\
  3841. .pv-gallery-sidebar-controler-pre-v {\
  3842. top: -40px;\
  3843. background-image: url("'+prefs.icons.arrowTop+'");\
  3844. }\
  3845. .pv-gallery-sidebar-controler-next-v {\
  3846. bottom: -40px;\
  3847. background-image: url("'+prefs.icons.arrowBottom+'");\
  3848. }\
  3849. .pv-gallery-sidebar-thumbnails-container {\
  3850. display: block;\
  3851. overflow: hidden;\
  3852. height: 100%;\
  3853. width: 100%;\
  3854. margin:0;\
  3855. border:none;\
  3856. padding:0;\
  3857. line-height:0;\
  3858. position:relative;\
  3859. }\
  3860. .pv-gallery-sidebar-thumbnails-container span{\
  3861. vertical-align:middle;\
  3862. }\
  3863. .pv-gallery-sidebar-thumbnails-container-h{\
  3864. border-left:1px solid #464646;\
  3865. border-right:1px solid #464646;\
  3866. white-space:nowrap;\
  3867. }\
  3868. .pv-gallery-sidebar-thumbnails-container-v{\
  3869. border-top:1px solid #464646;\
  3870. border-bottom:1px solid #464646;\
  3871. white-space:normal;\
  3872. }\
  3873. .pv-gallery-sidebar-thumbnails-container-top {\
  3874. padding-bottom:5px;\
  3875. }\
  3876. .pv-gallery-sidebar-thumbnails-container-right {\
  3877. padding-left:5px;\
  3878. }\
  3879. .pv-gallery-sidebar-thumbnails-container-bottom {\
  3880. padding-top:5px;\
  3881. }\
  3882. .pv-gallery-sidebar-thumbnails-container-left {\
  3883. padding-right:5px;\
  3884. }\
  3885. .pv-gallery-sidebar-thumb-container {\
  3886. display:inline-block;\
  3887. text-align: center;\
  3888. border:2px solid rgb(52,52,52);\
  3889. cursor:pointer;\
  3890. position:relative;\
  3891. padding:2px;\
  3892. font-size:0;\
  3893. line-height:0;\
  3894. white-space:nowrap;\
  3895. vertical-align: middle;\
  3896. top:0;\
  3897. left:0;\
  3898. -webkit-transition:all 0.2s ease-in-out;\
  3899. transition:all 0.2s ease-in-out;\
  3900. }\
  3901. .pv-gallery-sidebar-thumbnails-container-h .pv-gallery-sidebar-thumb-container {\
  3902. margin:0 2px;\
  3903. height:100%;\
  3904. }\
  3905. .pv-gallery-sidebar-thumbnails-container-v .pv-gallery-sidebar-thumb-container {\
  3906. margin:2px 0;\
  3907. width:100%;\
  3908. }\
  3909. .pv-gallery-sidebar-thumbnails_hide-span > .pv-gallery-sidebar-thumb-container {\
  3910. display:none;\
  3911. }\
  3912. .pv-gallery-sidebar-thumb-container:hover {\
  3913. border:2px solid rgb(57,149,211);\
  3914. }\
  3915. .pv-gallery-sidebar-thumb_selected {\
  3916. border:2px solid rgb(229,59,62);\
  3917. }\
  3918. .pv-gallery-sidebar-thumb_selected-top {\
  3919. top:5px;\
  3920. }\
  3921. .pv-gallery-sidebar-thumb_selected-right {\
  3922. left:-5px;\
  3923. }\
  3924. .pv-gallery-sidebar-thumb_selected-bottom {\
  3925. top:-5px;\
  3926. }\
  3927. .pv-gallery-sidebar-thumb_selected-left {\
  3928. left:5px;\
  3929. }\
  3930. .pv-gallery-sidebar-thumb-loading{\
  3931. position:absolute;\
  3932. top:0;\
  3933. left:0;\
  3934. text-align:center;\
  3935. width:100%;\
  3936. height:100%;\
  3937. display:none;\
  3938. opacity:0.6;\
  3939. background:black url("'+ prefs.icons.loading + '") no-repeat center ;\
  3940. }\
  3941. .pv-gallery-sidebar-thumb-loading:hover{\
  3942. opacity:0.8;\
  3943. }\
  3944. .pv-gallery-sidebar-thumb {\
  3945. display: inline-block;\
  3946. vertical-align: middle;\
  3947. max-width: 100% !important;\
  3948. max-height: 100% !important;\
  3949. height: auto !important;\
  3950. width: auto !important;\
  3951. }\
  3952. .pv-gallery-vertical-align-helper{\
  3953. display:inline-block;\
  3954. vertical-align:middle;\
  3955. width:0;\
  3956. height:100%;\
  3957. margin:0;\
  3958. border:0;\
  3959. padding:0;\
  3960. visibility:hidden;\
  3961. white-space:nowrap;\
  3962. background-color:red;\
  3963. }\
  3964. ';
  3965. var head=document.head;
  3966. head.appendChild(style);
  3967. this.globalSSheet=style.sheet;
  3968.  
  3969. var style2=document.createElement('style');
  3970. this.thumbVisibleStyle=style2;
  3971. style2.type='text/css';
  3972. head.appendChild(style2);
  3973. },
  3974.  
  3975. };
  3976.  
  3977.  
  3978. GalleryC.prototype.Preload.prototype={//预读对象
  3979. init:function(){
  3980. if(!this.container){//预读的图片都仍里面
  3981. var div=document.createElement('div');
  3982. div.className='pv-gallery-preloaded-img-container';
  3983. div.style.display='none';
  3984. document.body.appendChild(div);
  3985. GalleryC.prototype.Preload.prototype.container=div;
  3986. };
  3987. this.max=prefs.gallery.max;
  3988. this.nextNumber=0;
  3989. this.nextEle=this.ele;
  3990. this.preNumber=0;
  3991. this.preEle=this.ele;
  3992. this.direction='pre';
  3993. },
  3994. preload:function(){
  3995. var ele=this.getPreloadEle();
  3996. if(!ele){
  3997. //console.log('预读正常结束');
  3998. return;
  3999. };
  4000.  
  4001. //console.log('正在预读:',ele);
  4002. var self=this;
  4003. this.imgReady=imgReady(dataset(ele,'src'),{
  4004. loadEnd:function(){
  4005. if(self.aborted){
  4006. //console.log('强制终止了');
  4007. return;
  4008. };
  4009. dataset(ele,'preloaded','true')
  4010. self.container.appendChild(this);
  4011. self.preload();
  4012. },
  4013. time:60 * 1000,//限时一分钟,否则强制结束并开始预读下一张。
  4014. });
  4015. },
  4016. getPreloadEle:function(){
  4017. if((this.max<=this.nextNumber && this.max<=this.preNumber) || (!this.nextEle && !this.preEle)){
  4018. return;
  4019. };
  4020. var ele=this.direction=='pre'? this.getNext() : this.getPrevious();
  4021. if(ele && !dataset(ele,'preloaded')){
  4022. return ele;
  4023. }else{
  4024. return this.getPreloadEle();
  4025. };
  4026. },
  4027. getNext:function(){
  4028. this.nextNumber++;
  4029. this.direction='next';
  4030. if(!this.nextEle)return;
  4031. return (this.nextEle = this.oriThis.getThumSpan(false,this.nextEle));
  4032. },
  4033. getPrevious:function(){
  4034. this.preNumber++;
  4035. this.direction='pre';
  4036. if(!this.preEle)return;
  4037. return (this.preEle = this.oriThis.getThumSpan(true,this.preEle));
  4038. },
  4039. abort:function(){
  4040. this.aborted=true;
  4041. if(this.imgReady){
  4042. this.imgReady.abort();
  4043. };
  4044. },
  4045. };
  4046.  
  4047.  
  4048. GalleryC.prototype.Scrollbar.prototype={//滚动条对象
  4049. init:function(){
  4050. var bar=this.scrollbar.bar;
  4051. this.shown=bar.offsetWidth!=0;
  4052. var self=this;
  4053. bar.addEventListener('mousedown',function(e){//点击滚动条区域,该干点什么!
  4054. e.preventDefault();
  4055. var target=e.target;
  4056. var handle=self.scrollbar.handle;
  4057. var track=self.scrollbar.track;
  4058. switch(target){
  4059. case handle:{//手柄;功能,拖动手柄来滚动窗口
  4060. var pro=self.isHorizontal ? ['left','clientX'] : ['top','clientY'];
  4061. var oHOffset=parseFloat(handle.style[pro[0]]);
  4062. var oClient=e[pro[1]];
  4063.  
  4064. var moveHandler=function(e){
  4065. self.scroll(oHOffset + e[pro[1]] - oClient,true);
  4066. };
  4067. var upHandler=function(){
  4068. document.removeEventListener('mousemove',moveHandler,true);
  4069. document.removeEventListener('mouseup',upHandler,true);
  4070. };
  4071. document.addEventListener('mousemove',moveHandler,true);
  4072. document.addEventListener('mouseup',upHandler,true);
  4073. }break;
  4074. case track:{//轨道;功能,按住不放来连续滚动一个页面的距离
  4075. var pro=self.isHorizontal ? ['left','offsetX','layerX','clientWidth','offsetWidth'] : ['top' , 'offsetY' ,'layerY','clientHeight','offsetHeight'];
  4076. var clickOffset=typeof e[pro[1]]=='undefined' ? e[pro[2]] : e[pro[1]];
  4077. var handleOffset=parseFloat(handle.style[pro[0]]);
  4078. var handleSize=handle[pro[4]];
  4079. var under= clickOffset > handleOffset ;//点击在滚动手柄的下方
  4080. var containerSize=self.container[pro[3]];
  4081.  
  4082. var scroll=function(){
  4083. self.scrollBy(under? (containerSize - 10) : (-containerSize + 10));//滚动一个页面距离少一点
  4084. };
  4085. scroll();
  4086.  
  4087. var checkStop=function(){//当手柄到达点击位置时停止
  4088. var handleOffset=parseFloat(handle.style[pro[0]]);
  4089. if(clickOffset >= handleOffset && clickOffset <= (handleOffset + handleSize)){
  4090. clearTimeout(scrollTimeout);
  4091. clearInterval(scrollInterval);
  4092. };
  4093. };
  4094.  
  4095.  
  4096. var scrollInterval;
  4097. var scrollTimeout=setTimeout(function(){
  4098. scroll();
  4099. scrollInterval=setInterval(function(){
  4100. scroll();
  4101. checkStop();
  4102. },120);
  4103. checkStop();
  4104. },300);
  4105.  
  4106.  
  4107. checkStop();
  4108.  
  4109. var upHandler=function(){
  4110. clearTimeout(scrollTimeout);
  4111. clearInterval(scrollInterval);
  4112. document.removeEventListener('mouseup',upHandler,true);
  4113. };
  4114. document.addEventListener('mouseup',upHandler,true);
  4115. }break;
  4116. };
  4117.  
  4118. },true);
  4119. },
  4120. reset:function(){//判断滚动条该显示还是隐藏
  4121.  
  4122. var pro=this.isHorizontal ? ['scrollWidth','clientWidth','width'] : ['scrollHeight','clientHeight','height'];
  4123.  
  4124. //如果内容大于容器的content区域
  4125.  
  4126. var scrollSize=this.container[pro[0]];
  4127. var clientSize=this.container[pro[1]];
  4128. var scrollMax=scrollSize - clientSize;
  4129. this.scrollMax=scrollMax;
  4130. if(scrollMax>0){
  4131. this.show();
  4132. var trackSize=this.scrollbar.track[pro[1]];
  4133. this.trackSize=trackSize;
  4134. var handleSize=Math.floor((clientSize/scrollSize) * trackSize);
  4135. handleSize=Math.max(20,handleSize);//限制手柄的最小大小;
  4136. this.handleSize=handleSize;
  4137. this.one=(trackSize-handleSize) / scrollMax;//一个像素对应的滚动条长度
  4138. this.scrollbar.handle.style[pro[2]]= handleSize + 'px';
  4139. this.scroll(this.getScrolled());
  4140. }else{
  4141. this.hide();
  4142. };
  4143. },
  4144. show:function(){
  4145. if(this.shown)return;
  4146. this.shown=true;
  4147. this.scrollbar.bar.style.display='block';
  4148. },
  4149. hide:function(){
  4150. if(!this.shown)return;
  4151. this.shown=false;
  4152. this.scrollbar.bar.style.display='none';
  4153. },
  4154. scrollBy:function(distance,handleDistance){
  4155. this.scroll(this.getScrolled() + (handleDistance? distance / this.one : distance));
  4156. },
  4157. scrollByPages:function(num){
  4158. this.scroll(this.getScrolled() + (this.container[(this.isHorizontal ? 'clientWidth' : 'clientHeight')] - 10) * num);
  4159. },
  4160. scroll:function(distance,handleDistance,transition){
  4161. if(!this.shown)return;
  4162.  
  4163. //滚动实际滚动条
  4164. var _distance=distance;
  4165. _distance=handleDistance? distance / this.one : distance;
  4166. _distance=Math.max(0,_distance);
  4167. _distance=Math.min(_distance,this.scrollMax);
  4168.  
  4169.  
  4170. var pro=this.isHorizontal? ['left','scrollLeft'] : ['top','scrollTop'];
  4171.  
  4172.  
  4173. //滚动虚拟滚动条
  4174. //根据比例转换为滚动条上应该滚动的距离。
  4175. distance=handleDistance? distance : this.one * distance;
  4176. //处理非法值
  4177. distance=Math.max(0,distance);//如果值小于0那么取0
  4178. distance=Math.min(distance,this.trackSize - this.handleSize);//大于极限值,取极限值
  4179.  
  4180. var shs=this.scrollbar.handle.style;
  4181. var container=this.container;
  4182. if(transition){
  4183. clearInterval(this.transitionInterval);
  4184.  
  4185. var start=0;
  4186. var duration=10;
  4187.  
  4188. var cStart=this.getScrolled();
  4189. var cChange=_distance-cStart;
  4190. var sStart=parseFloat(shs[pro[0]]);
  4191. var sChange=distance-sStart;
  4192.  
  4193. var transitionInterval=setInterval(function(){
  4194. var cEnd=Tween.Cubic.easeInOut(start,cStart,cChange,duration);
  4195. var sEnd=Tween.Cubic.easeInOut(start,sStart,sChange,duration);
  4196.  
  4197. container[pro[1]]=cEnd;
  4198. shs[pro[0]]=sEnd + 'px';
  4199.  
  4200. start++;
  4201. if(start>=duration){
  4202. clearInterval(transitionInterval);
  4203. };
  4204. },35);
  4205.  
  4206. this.transitionInterval=transitionInterval;
  4207.  
  4208. return;
  4209. };
  4210.  
  4211. shs[pro[0]]=distance + 'px';
  4212. container[pro[1]]=_distance;
  4213. },
  4214. getScrolled:function(){
  4215. return this.container[(this.isHorizontal ? 'scrollLeft' : 'scrollTop')];
  4216. },
  4217. };
  4218.  
  4219.  
  4220. //放大镜
  4221. function MagnifierC(img,data){
  4222. this.img=img;
  4223. this.data=data;
  4224. this.init();
  4225. };
  4226.  
  4227. MagnifierC.all=[];
  4228. MagnifierC.styleZIndex=900000000;//全局z-index;
  4229. MagnifierC.zoomRange=prefs.magnifier.wheelZoom.range.slice(0).sort();//升序
  4230. MagnifierC.zoomRangeR=MagnifierC.zoomRange.slice(0).reverse();//降序
  4231.  
  4232. MagnifierC.prototype={
  4233. init:function(){
  4234. this.addStyle();
  4235. MagnifierC.all.push(this);
  4236. var container=document.createElement('span');
  4237.  
  4238. container.className='pv-magnifier-container';
  4239. document.body.appendChild(container);
  4240.  
  4241. this.magnifier=container;
  4242.  
  4243. var imgNaturalSize={
  4244. h:this.img.naturalHeight,
  4245. w:this.img.naturalWidth,
  4246. };
  4247.  
  4248. this.imgNaturalSize=imgNaturalSize;
  4249.  
  4250. var cs=container.style;
  4251. cs.zIndex=MagnifierC.styleZIndex++;
  4252.  
  4253.  
  4254.  
  4255. var maxDia=Math.ceil(Math.sqrt(Math.pow(1/2*imgNaturalSize.w,2) + Math.pow(1/2*imgNaturalSize.h,2)) * 2);
  4256. this.maxDia=maxDia;
  4257.  
  4258. var radius=prefs.magnifier.radius;
  4259. radius=Math.min(maxDia/2,radius);
  4260. this.radius=radius;
  4261. var diameter=radius * 2;
  4262. this.diameter=diameter;
  4263.  
  4264. cs.width=diameter + 'px';
  4265. cs.height=diameter + 'px';
  4266. cs.borderRadius=radius+1 + 'px';
  4267. cs.backgroundImage='url("'+ this.img.src +'")';
  4268. cs.marginLeft= -radius +'px';
  4269. cs.marginTop= -radius +'px';
  4270.  
  4271. var imgPos=getContentClientRect(this.data.img);
  4272. var wScrolled=getScrolled();
  4273. var imgRange={//图片所在范围
  4274. x:[imgPos.left + wScrolled.x , imgPos.right + wScrolled.x],
  4275. y:[imgPos.top + wScrolled.y, imgPos.bottom + wScrolled.y],
  4276. };
  4277. var imgW=imgRange.x[1] - imgRange.x[0];
  4278. var imgH=imgRange.y[1] - imgRange.y[0];
  4279. //如果图片太小的话,进行范围扩大。
  4280. var minSize=60;
  4281. if(imgW < minSize){
  4282. imgRange.x[1] +=(minSize - imgW)/2;
  4283. imgRange.x[0] -=(minSize - imgW)/2;
  4284. imgW=minSize;
  4285. };
  4286. if(imgH < minSize){
  4287. imgRange.y[1] +=(minSize - imgH)/2;
  4288. imgRange.y[0] -=(minSize - imgH)/2;
  4289. imgH=minSize;
  4290. };
  4291. this.imgSize={
  4292. w:imgW,
  4293. h:imgH,
  4294. };
  4295. this.imgRange=imgRange;
  4296. //console.log(this.imgRange,this.imgSize);
  4297.  
  4298. this.setMouseRange();
  4299.  
  4300.  
  4301. this.move({
  4302. pageX:imgRange.x[0],
  4303. pageY:imgRange.y[0],
  4304. });
  4305.  
  4306. this._focus=this.focus.bind(this);
  4307. this._blur=this.blur.bind(this);
  4308. this._move=this.move.bind(this);
  4309. this._remove=this.remove.bind(this);
  4310. this._pause=this.pause.bind(this);
  4311. this._zoom=this.zoom.bind(this);
  4312.  
  4313. if(prefs.magnifier.wheelZoom.enabled){
  4314. this.zoomLevel=1;
  4315. this.defaultDia=diameter;
  4316. addWheelEvent(container,this._zoom,false);
  4317. };
  4318.  
  4319. container.addEventListener('mouseover',this._focus,false);
  4320. container.addEventListener('mouseout',this._blur,false);
  4321. container.addEventListener('dblclick',this._remove,false);
  4322. container.addEventListener('click',this._pause,false);
  4323.  
  4324.  
  4325. document.addEventListener('mousemove',this._move,true);
  4326. },
  4327. addStyle:function(){
  4328. if(MagnifierC.style)return;
  4329. var style=document.createElement('style');
  4330. style.type='text/css';
  4331. MagnifierC.style=style;
  4332. style.textContent='\
  4333. .pv-magnifier-container{\
  4334. position:absolute;\
  4335. padding:0;\
  4336. margin:0;\
  4337. background-origin:border-box;\
  4338. -moz-box-sizing:border-box;\
  4339. box-sizing:border-box;\
  4340. border:3px solid #CCCCCC;\
  4341. background:rgba(40, 40, 40, 0.9) no-repeat;\
  4342. }\
  4343. .pv-magnifier-container_focus{\
  4344. box-shadow: 0px 0px 6px rgba(0, 0, 0, 0.7);\
  4345. }\
  4346. .pv-magnifier-container_pause{\
  4347. border-color:red;\
  4348. }\
  4349. ';
  4350. document.head.appendChild(style);
  4351. },
  4352. focus:function(){
  4353. this.magnifier.classList.add('pv-magnifier-container_focus');
  4354. this.magnifier.style.zIndex=MagnifierC.styleZIndex++;
  4355. },
  4356. blur:function(){
  4357. this.magnifier.classList.remove('pv-magnifier-container_focus');
  4358. },
  4359. move:function(e){
  4360. var mouseCoor={
  4361. x:e.pageX,
  4362. y:e.pageY,
  4363. };
  4364. var mouseRange=this.mouseRange;
  4365. var imgRange=this.imgRange;
  4366.  
  4367. if( !(mouseCoor.x >= mouseRange.x[0] && mouseCoor.x <= mouseRange.x[1] && mouseCoor.y >= mouseRange.y[0] && mouseCoor.y <= mouseRange.y[1]))return;//如果不再鼠标范围
  4368. if(mouseCoor.x > imgRange.x[1]){
  4369. mouseCoor.x = imgRange.x[1];
  4370. }else if(mouseCoor.x < imgRange.x[0]){
  4371. mouseCoor.x = imgRange.x[0];
  4372. };
  4373. if(mouseCoor.y > imgRange.y[1]){
  4374. mouseCoor.y = imgRange.y[1];
  4375. }else if(mouseCoor.y < imgRange.y[0]){
  4376. mouseCoor.y = imgRange.y[0];
  4377. };
  4378.  
  4379. var ms=this.magnifier.style;
  4380. ms.top= mouseCoor.y + 'px';
  4381. ms.left= mouseCoor.x + 'px';
  4382.  
  4383. var radius=this.radius;
  4384. var imgSize=this.imgSize;
  4385. var imgNaturalSize=this.imgNaturalSize;
  4386. var px=-((mouseCoor.x-imgRange.x[0])/imgSize.w * imgNaturalSize.w) + radius +'px';
  4387. var py=-((mouseCoor.y-imgRange.y[0])/imgSize.h * imgNaturalSize.h) + radius +'px';
  4388. //console.log(px,py);
  4389. ms.backgroundPosition=px + ' ' + py;
  4390. },
  4391. getNextZoomLevel:function(){
  4392. var level;
  4393. var self=this;
  4394. if(this.zoomOut){//缩小
  4395. MagnifierC.zoomRangeR._find(function(value){
  4396. if(value < self.zoomLevel){
  4397. level=value;
  4398. return true;
  4399. }
  4400. })
  4401. }else{
  4402. MagnifierC.zoomRange._find(function(value){
  4403. if(value > self.zoomLevel){
  4404. level=value;
  4405. return true;
  4406. };
  4407. });
  4408. }
  4409. return level;
  4410. },
  4411. zoom:function(e){
  4412. if(e.deltaY===0)return;//非Y轴的滚动
  4413. if(prefs.magnifier.wheelZoom.pauseFirst && !this.paused)return;
  4414. e.preventDefault();
  4415. if(e.deltaY < 0){//向上滚,放大;
  4416. if(this.diameter >= this.maxDia)return;
  4417. this.zoomOut=false;
  4418. }else{
  4419. this.zoomOut=true;
  4420. };
  4421. var level=this.getNextZoomLevel();
  4422. if(!level)return;
  4423.  
  4424. this.zoomLevel=level;
  4425. var diameter=this.defaultDia * level;
  4426. if(diameter > this.maxDia){
  4427. diameter = this.maxDia;
  4428. };
  4429.  
  4430. var radius=diameter/2
  4431. this.diameter=diameter;
  4432. var bRadius=this.radius;
  4433. this.radius=radius;
  4434. this.setMouseRange();
  4435. var ms=this.magnifier.style;
  4436. ms.width=diameter+'px';
  4437. ms.height=diameter+'px';
  4438. ms.borderRadius=radius+1 + 'px';
  4439. ms.marginLeft=-radius+'px';
  4440. ms.marginTop=-radius+'px';
  4441. var bBP=ms.backgroundPosition.split(' ');
  4442. ms.backgroundPosition=parseFloat(bBP[0]) + (radius - bRadius) + 'px' + ' ' + (parseFloat(bBP[1]) + ( radius - bRadius) + 'px');
  4443.  
  4444. },
  4445. pause:function(){
  4446. if(this.paused){
  4447. this.magnifier.classList.remove('pv-magnifier-container_pause');
  4448. document.addEventListener('mousemove',this._move,true);
  4449. }else{
  4450. this.magnifier.classList.add('pv-magnifier-container_pause');
  4451. document.removeEventListener('mousemove',this._move,true);
  4452. };
  4453. this.paused=!this.paused;
  4454. },
  4455. setMouseRange:function(){
  4456. var imgRange=this.imgRange;
  4457. var radius=this.radius;
  4458. this.mouseRange={//鼠标活动范围
  4459. x:[imgRange.x[0]-radius , imgRange.x[1] + radius],
  4460. y:[imgRange.y[0]-radius , imgRange.y[1] + radius],
  4461. };
  4462. },
  4463. remove:function(){
  4464. this.magnifier.parentNode.removeChild(this.magnifier);
  4465. document.removeEventListener('mousemove',this._move,true);
  4466. MagnifierC.all.splice(MagnifierC.all.indexOf(this),1);
  4467. },
  4468. };
  4469.  
  4470.  
  4471.  
  4472.  
  4473. //图片窗口
  4474. function ImgWindowC(img){
  4475. this.img=img;
  4476. this.src=img.src;
  4477. this.init();
  4478. };
  4479.  
  4480. ImgWindowC.all=[];//所有的窗口对象
  4481. ImgWindowC.styleZIndex=1000000000;//全局z-index;
  4482. ImgWindowC.zoomRange=prefs.imgWindow.zoom.range.slice(0).sort();//升序
  4483. ImgWindowC.zoomRangeR=ImgWindowC.zoomRange.slice(0).reverse();//降序
  4484. ImgWindowC.overlayer=null;
  4485.  
  4486.  
  4487. ImgWindowC.prototype={
  4488. init:function(){
  4489. var self=this;
  4490. //图片是否已经被打开
  4491. if(ImgWindowC.all._find(function(iwin){
  4492. if(iwin.src==self.src){
  4493. iwin.firstOpen();
  4494. return true;
  4495. };
  4496. }))return;
  4497.  
  4498. this.addStyle();
  4499.  
  4500. var img=this.img;
  4501. img.className='pv-pic-window-pic pv-pic-ignored';
  4502. img.style.cssText='\
  4503. top:0px;\
  4504. left:0px;\
  4505. ';
  4506.  
  4507. var imgNaturalSize={
  4508. h:img.naturalHeight,
  4509. w:img.naturalWidth,
  4510. };
  4511. this.imgNaturalSize=imgNaturalSize;
  4512.  
  4513. var container=document.createElement('span');
  4514. container.style.cssText='\
  4515. cursor:pointer;\
  4516. top:0px;\
  4517. left:0px;\
  4518. ';
  4519. container.className='pv-pic-window-container';
  4520. container.innerHTML=
  4521. '<span class="pv-pic-window-rotate-indicator">'+
  4522. '<span class="pv-pic-window-rotate-indicator-pointer"></span>'+
  4523. '</span>'+
  4524. '<span class="pv-pic-window-rotate-overlayer"></span>'+
  4525. '<span class="pv-pic-window-toolbar" unselectable="on">'+
  4526. '<span class="pv-pic-window-tb-hand pv-pic-window-tb-tool" title="抓手"></span>'+
  4527. '<span class="pv-pic-window-tb-tool-badge-container pv-pic-window-tb-tool-extend-menu-container">'+
  4528. '<span class="pv-pic-window-tb-rotate pv-pic-window-tb-tool" title="旋转"></span>'+
  4529. '<span class="pv-pic-window-tb-tool-badge">0</span>'+
  4530. '<span class="pv-pic-window-tb-tool-extend-menu pv-pic-window-tb-tool-extend-menu-rotate">'+
  4531. '<span class="pv-pic-window-tb-tool-extend-menu-item">0</span>'+
  4532. '<span class="pv-pic-window-tb-tool-extend-menu-item">+90</span>'+
  4533. '<span class="pv-pic-window-tb-tool-extend-menu-item">-90</span>'+
  4534. '</span>'+
  4535. '</span>'+
  4536. '<span class="pv-pic-window-tb-tool-badge-container pv-pic-window-tb-tool-extend-menu-container">'+
  4537. '<span class="pv-pic-window-tb-zoom pv-pic-window-tb-tool" title="缩放"></span>'+
  4538. '<span class="pv-pic-window-tb-tool-badge">0</span>'+
  4539. '<span class="pv-pic-window-tb-tool-extend-menu pv-pic-window-tb-tool-extend-menu-zoom">'+
  4540. '<span class="pv-pic-window-tb-tool-extend-menu-item">1</span>'+
  4541. '<span class="pv-pic-window-tb-tool-extend-menu-item">+0.1</span>'+
  4542. '<span class="pv-pic-window-tb-tool-extend-menu-item">-0.1</span>'+
  4543. '</span>'+
  4544. '</span>'+
  4545. '<span class="pv-pic-window-tb-flip-horizontal pv-pic-window-tb-command" title="水平翻转"></span>'+
  4546. '<span class="pv-pic-window-tb-flip-vertical pv-pic-window-tb-command" title="垂直翻转"></span>'+
  4547. '</span>'+
  4548. '<span class="pv-pic-window-close"></span>'+
  4549. '<span class="pv-pic-window-range"></span>';
  4550.  
  4551. container.insertBefore(img,container.firstChild);
  4552.  
  4553. this.imgWindow=container;
  4554.  
  4555. var toolMap={
  4556. 'hand':container.querySelector('.pv-pic-window-tb-hand'),
  4557. 'rotate':container.querySelector('.pv-pic-window-tb-rotate'),
  4558. 'zoom':container.querySelector('.pv-pic-window-tb-zoom'),
  4559. 'fh':container.querySelector('.pv-pic-window-tb-flip-horizontal'),
  4560. 'fv':container.querySelector('.pv-pic-window-tb-flip-vertical'),
  4561. };
  4562. this.toolMap=toolMap;
  4563.  
  4564.  
  4565. //关闭
  4566. var closeButton=container.querySelector('.pv-pic-window-close');
  4567. closeButton.style.cssText='\
  4568. top: -24px;\
  4569. right: 0px;\
  4570. ';
  4571. this.closeButton=closeButton;
  4572.  
  4573. closeButton.addEventListener('click',function(e){
  4574. self.remove();
  4575. },false);
  4576.  
  4577. var toolbar=container.querySelector('.pv-pic-window-toolbar');
  4578. toolbar.style.cssText='\
  4579. top: 0px;\
  4580. left: -45px;\
  4581. ';
  4582. this.toolbar=toolbar;
  4583.  
  4584. this.selectedToolClass='pv-pic-window-tb-tool-selected';
  4585.  
  4586. this.viewRange=container.querySelector('.pv-pic-window-range');
  4587.  
  4588. this.rotateIndicator=container.querySelector('.pv-pic-window-rotate-indicator');
  4589. this.rotateIPointer=container.querySelector('.pv-pic-window-rotate-indicator-pointer');
  4590. this.rotateOverlayer=container.querySelector('.pv-pic-window-rotate-overlayer');
  4591.  
  4592.  
  4593. this.hKeyUp=true;
  4594. this.rKeyUp=true;
  4595. this.zKeyUp=true;
  4596.  
  4597. this.spaceKeyUp=true;
  4598. this.ctrlKeyUp=true;
  4599. this.altKeyUp=true;
  4600. this.shiftKeyUp=true;
  4601.  
  4602. //缩放工具的扩展菜单
  4603. container.querySelector('.pv-pic-window-tb-tool-extend-menu-zoom').addEventListener('click',function(e){
  4604. var target=e.target;
  4605. var text=target.textContent;
  4606. var value;
  4607. switch(text){
  4608. case '1':{
  4609. value=1;
  4610. }break;
  4611. case '+0.1':{
  4612. value=self.zoomLevel + 0.1;
  4613. }break;
  4614. case '-0.1':{
  4615. value=self.zoomLevel - 0.1;
  4616. }break;
  4617. };
  4618. if(typeof value!='undefined'){
  4619. self.zoom(value,{x:0,y:0});
  4620. };
  4621. },true);
  4622.  
  4623. //旋转工具的扩展菜单
  4624. container.querySelector('.pv-pic-window-tb-tool-extend-menu-rotate').addEventListener('click',function(e){
  4625. var target=e.target;
  4626. var text=target.textContent;
  4627. var value;
  4628. function convert(deg){
  4629. return deg * Math.PI/180;
  4630. };
  4631.  
  4632. switch(text){
  4633. case '0':{
  4634. value=0;
  4635. }break;
  4636. case '+90':{
  4637. value=self.rotatedRadians + convert(90);
  4638. }break;
  4639. case '-90':{
  4640. value=self.rotatedRadians - convert(90);
  4641. }break;
  4642. };
  4643.  
  4644. var PI=Math.PI;
  4645. if(typeof value!='undefined'){
  4646. if(value>=2*PI){
  4647. value-=2*PI;
  4648. }else if(value<0){
  4649. value+=2*PI;
  4650. };
  4651. self.rotate(value,true);
  4652. };
  4653. },true);
  4654.  
  4655. toolbar.addEventListener('mousedown',function(e){//鼠标按下选择工具
  4656. self.toolbarEventHandler(e);
  4657. },false);
  4658.  
  4659.  
  4660. toolbar.addEventListener('dblclick',function(e){//鼠标双击工具
  4661. self.toolbarEventHandler(e);
  4662. },false);
  4663.  
  4664.  
  4665. //阻止浏览器对图片的默认控制行为
  4666. img.addEventListener('mousedown',function(e){
  4667. e.preventDefault();
  4668. },false);
  4669.  
  4670.  
  4671. container.addEventListener('mousedown',function(e){//当按下的时,执行平移,缩放,旋转操作
  4672. self.imgWindowEventHandler(e);
  4673. },false);
  4674.  
  4675. container.addEventListener('click',function(e){//阻止opera ctrl+点击保存图片
  4676. self.imgWindowEventHandler(e);
  4677. },false);
  4678.  
  4679. if(prefs.imgWindow.zoom.mouseWheelZoom){//是否使用鼠标缩放
  4680. addWheelEvent(container,function(e){//滚轮缩放
  4681. self.imgWindowEventHandler(e);
  4682. },false);
  4683. };
  4684.  
  4685.  
  4686. if(prefs.imgWindow.overlayer.shown){//是否显示覆盖层
  4687. var overlayer=ImgWindowC.overlayer;
  4688. if(!overlayer){
  4689. var overlayer=document.createElement('span');
  4690. ImgWindowC.overlayer=overlayer;
  4691. overlayer.className='pv-pic-window-overlayer';
  4692. document.body.appendChild(overlayer);
  4693. overlayer.style.backgroundColor=prefs.imgWindow.overlayer.color;
  4694. };
  4695. overlayer.style.display='block';
  4696. };
  4697.  
  4698. //是否点击图片外部关闭
  4699. if(prefs.imgWindow.close.clickOutside.enabled){
  4700. var clickOutside=function(e){
  4701. var target=e.target;
  4702. if(!container.contains(target)){
  4703. self.remove();
  4704. };
  4705. };
  4706. this.clickOutside=clickOutside;
  4707. document.addEventListener(prefs.imgWindow.close.clickOutside.trigger,clickOutside,true);
  4708. };
  4709.  
  4710. //是否双击图片本身关闭
  4711. if(prefs.imgWindow.close.dblClickImgWindow){
  4712. var dblClickImgWindow=function(e){
  4713. var target=e.target;
  4714. if(target==container || target==img || target==self.rotateOverlayer){
  4715. self.remove();
  4716. };
  4717. };
  4718. container.addEventListener('dblclick',dblClickImgWindow,true);
  4719. };
  4720.  
  4721.  
  4722. document.body.appendChild(container);
  4723. ImgWindowC.all.push(this);
  4724.  
  4725. this._blur=this.blur.bind(this);
  4726. this._focusedKeydown=this.focusedKeydown.bind(this);
  4727. this._focusedKeyup=this.focusedKeyup.bind(this);
  4728.  
  4729. this.rotatedRadians=0;//已经旋转的角度
  4730. this.zoomLevel=1;//缩放级别
  4731. this.setToolBadge('zoom',1);
  4732.  
  4733. //选中默认工具
  4734. this.selectTool(prefs.imgWindow.defaultTool);
  4735.  
  4736. this.firstOpen();
  4737. },
  4738.  
  4739.  
  4740. addStyle:function(){
  4741. if(ImgWindowC.style)return;
  4742. var style=document.createElement('style');
  4743. ImgWindowC.style=style;
  4744. style.textContent='\
  4745. .pv-pic-window-container {\
  4746. position: absolute;\
  4747. background-color: rgba(40,40,40,0.9);\
  4748. padding: 8px;\
  4749. border: 5px solid #ccc;\
  4750. line-height: 0;\
  4751. text-align: left;\
  4752. }\
  4753. .pv-pic-window-container_focus {\
  4754. box-shadow: 0 0 10px rgba(0,0,0,0.6);\
  4755. }\
  4756. .pv-pic-window-close,\
  4757. .pv-pic-window-toolbar,\
  4758. .pv-pic-window-tb-tool-extend-menu{\
  4759. -webkit-transition: opacity 0.2s ease-in-out;\
  4760. transition: opacity 0.2s ease-in-out;\
  4761. }\
  4762. .pv-pic-window-toolbar {\
  4763. position: absolute;\
  4764. background-color: #535353;\
  4765. padding: 0;\
  4766. opacity: 0.9;\
  4767. display: none;\
  4768. cursor: default;\
  4769. -o-user-select: none;\
  4770. -webkit-user-select: none;\
  4771. -moz-user-select: -moz-none;\
  4772. user-select: none;\
  4773. }\
  4774. .pv-pic-window-toolbar:hover {\
  4775. opacity: 1;\
  4776. }\
  4777. .pv-pic-window-toolbar_focus {\
  4778. display: block;\
  4779. }\
  4780. .pv-pic-window-close {\
  4781. cursor: pointer;\
  4782. position: absolute;\
  4783. right: 0px;\
  4784. top: -24px;\
  4785. background: url("'+prefs.icons.close+'") no-repeat center bottom;\
  4786. height: 17px;\
  4787. width: 46px;\
  4788. opacity: 0.9;\
  4789. border:none;\
  4790. padding:0;\
  4791. padding-top:2px;\
  4792. background-color:#1771FF;\
  4793. display: none;\
  4794. }\
  4795. .pv-pic-window-close:hover {\
  4796. background-color:red;\
  4797. opacity: 1;\
  4798. }\
  4799. .pv-pic-window-close_focus {\
  4800. display: block;\
  4801. }\
  4802. .pv-pic-window-pic {\
  4803. position: relative;\
  4804. display:inline-block;\/*opera把图片设置display:block会出现渲染问题,会有残影,还会引发其他各种问题,吓尿*/\
  4805. max-width:none;\
  4806. min-width:none;\
  4807. max-height:none;\
  4808. min-height:none;\
  4809. padding:0;\
  4810. margin:0;\
  4811. border:none;\
  4812. vertical-align:middle;\
  4813. }\
  4814. .pv-pic-window-pic_focus {\
  4815. box-shadow: 0 0 6px black;\
  4816. }\
  4817. .pv-pic-window-tb-tool,\
  4818. .pv-pic-window-tb-command{\
  4819. height: 24px;\
  4820. width: 24px;\
  4821. padding: 12px 8px 6px 6px;\
  4822. margin:0;\
  4823. display: block;\
  4824. background: transparent no-repeat center;\
  4825. cursor: pointer;\
  4826. position: relative;\
  4827. border: none;\
  4828. border-left: 2px solid transparent;\
  4829. border-bottom: 1px solid #868686;\
  4830. background-origin: content-box;\
  4831. }\
  4832. .pv-pic-window-toolbar > span:last-child {\
  4833. border-bottom: none;\
  4834. }\
  4835. .pv-pic-window-tb-tool:hover,\
  4836. .pv-pic-window-tb-command:hover{\
  4837. border-left: 2px solid red;\
  4838. }\
  4839. .pv-pic-window-tb-tool-selected{\
  4840. box-shadow: inset 0 21px 0 rgba(255,255,255,0.3) ,inset 0 -21px 0 rgba(0,0,0,0.3);\
  4841. border-left:2px solid #1771FF;\
  4842. }\
  4843. .pv-pic-window-tb-hand {\
  4844. background-image: url("'+prefs.icons.hand+'");\
  4845. }\
  4846. .pv-pic-window-tb-rotate {\
  4847. background-image: url("'+prefs.icons.rotate+'");\
  4848. }\
  4849. .pv-pic-window-tb-zoom {\
  4850. background-image: url("'+prefs.icons.zoom+'");\
  4851. }\
  4852. .pv-pic-window-tb-flip-horizontal {\
  4853. background-image: url("'+prefs.icons.flipHorizontal+'");\
  4854. }\
  4855. .pv-pic-window-tb-flip-vertical {\
  4856. background-image: url("'+prefs.icons.flipVertical+'");\
  4857. }\
  4858. .pv-pic-window-tb-tool-badge-container {\
  4859. display: block;\
  4860. position: relative;\
  4861. }\
  4862. .pv-pic-window-tb-tool-badge {\
  4863. position: absolute;\
  4864. top: -3px;\
  4865. right: 1px;\
  4866. font-size: 10px;\
  4867. line-height: 1.5;\
  4868. padding: 0 3px;\
  4869. background-color: #F93;\
  4870. border-radius: 50px;\
  4871. opacity: 0.5;\
  4872. color: black;\
  4873. }\
  4874. .pv-pic-window-tb-tool-extend-menu{\
  4875. position:absolute;\
  4876. top:0;\
  4877. margin-left:-1px;\
  4878. background-color:#535353;\
  4879. display:none;\
  4880. left:40px;\
  4881. color:#C3C3C3;\
  4882. font-size:12px;\
  4883. text-shadow:0px -1px 0px black;\
  4884. opacity:0.7;\
  4885. }\
  4886. .pv-pic-window-tb-tool-extend-menu:hover{\
  4887. opacity:0.9;\
  4888. }\
  4889. .pv-pic-window-tb-tool-extend-menu-item{\
  4890. display:block;\
  4891. line-height:1.5;\
  4892. text-align:center;\
  4893. padding:10px;\
  4894. cursor:pointer;\
  4895. border: none;\
  4896. border-right: 2px solid transparent;\
  4897. border-bottom: 1px solid #868686;\
  4898. }\
  4899. .pv-pic-window-tb-tool-extend-menu-item:last-child{\
  4900. border-bottom: none;\
  4901. }\
  4902. .pv-pic-window-tb-tool-extend-menu-item:hover{\
  4903. border-right:2px solid red;\
  4904. }\
  4905. .pv-pic-window-tb-tool-extend-menu-item:active{\
  4906. padding:11px 9px 9px 11px;\
  4907. }\
  4908. .pv-pic-window-tb-tool-extend-menu-container:hover .pv-pic-window-tb-tool{\
  4909. border-left:2px solid red;\
  4910. }\
  4911. .pv-pic-window-tb-tool-extend-menu-container:hover .pv-pic-window-tb-tool-extend-menu{\
  4912. display:block;\
  4913. }\
  4914. .pv-pic-window-tb-tool-extend-menu-container::after{\
  4915. content:"";\
  4916. position:absolute;\
  4917. right:1px;\
  4918. bottom:2px;\
  4919. width:0;\
  4920. height:0;\
  4921. padding:0;\
  4922. margin:0;\
  4923. border:3px solid #C3C3C3;\
  4924. border-top-color:transparent;\
  4925. border-left-color:transparent;\
  4926. opacity:0.5;\
  4927. }\
  4928. .pv-pic-window-overlayer{\
  4929. height:100%;\
  4930. width:100%;\
  4931. position:fixed;\
  4932. z-index:999999999;\
  4933. top:0;\
  4934. left:0;\
  4935. }\
  4936. .pv-pic-window-rotate-indicator{\
  4937. display:none;\
  4938. position:fixed;\
  4939. width:250px;\
  4940. height:250px;\
  4941. padding:10px;\
  4942. margin-top:-135px;\
  4943. margin-left:-135px;\
  4944. background:transparent url("'+ prefs.icons.rotateIndicatorBG +'") no-repeat center;\
  4945. }\
  4946. .pv-pic-window-rotate-indicator-pointer{\
  4947. display:block;\
  4948. margin-left:auto;\
  4949. margin-right:auto;\
  4950. background:transparent url("'+ prefs.icons.rotateIndicatorPointer +'") no-repeat center;\
  4951. width:60px;\
  4952. height:240px;\
  4953. position:relative;\
  4954. top:5px;\
  4955. transform:rotate(0.1deg);\
  4956. }\
  4957. .pv-pic-window-rotate-overlayer{/*当切换到旋转工具的时候显示这个覆盖层,然后旋转指示器显示在这个覆盖层的下面*/\
  4958. position:absolute;\
  4959. top:0;\
  4960. bottom:0;\
  4961. left:0;\
  4962. right:0;\
  4963. display:none;\
  4964. background-color:transparent;\
  4965. }\
  4966. .pv-pic-window-range{\
  4967. position:absolute;\
  4968. border:none;\
  4969. width:100px;\
  4970. height:100px;\
  4971. box-shadow: 0px 0px 5px rgba(0, 0, 0, 0.8);\
  4972. display:none;\
  4973. padding:0;\
  4974. background-color:rgba(255, 0, 0, 0.150);\
  4975. }\
  4976. ';
  4977. document.head.appendChild(style);
  4978. },
  4979.  
  4980. firstOpen:function(){
  4981. this.focus();
  4982. var imgWindow=this.imgWindow;
  4983. var scrolled=getScrolled();
  4984. imgWindow.style.left=-5 + scrolled.x + 'px';
  4985. imgWindow.style.top=-5 + scrolled.y + 'px';
  4986.  
  4987. if(prefs.imgWindow.fitToScreen){
  4988. this.fitToScreen();
  4989. this.center(true,true);
  4990. }else{
  4991. //window的尺寸
  4992. var wSize=getWindowSize();
  4993. //空隙
  4994. wSize.h -= 16;
  4995. wSize.w -= 16;
  4996.  
  4997. var imgWindowCS=getComputedStyle(imgWindow);
  4998.  
  4999. var rectSize={
  5000. h:parseFloat(imgWindowCS.height),
  5001. w:parseFloat(imgWindowCS.width),
  5002. };
  5003.  
  5004. this.center(rectSize.w <= wSize.w , rectSize.h <= wSize.h);
  5005. };
  5006.  
  5007. this.keepScreenInside();
  5008. },
  5009. keepScreenInside:function(){//保持按钮在屏幕里面.
  5010. var imgWindow=this.imgWindow;
  5011. var imgWindowFullSize={
  5012. h:imgWindow.offsetHeight,
  5013. w:imgWindow.offsetWidth,
  5014. };
  5015.  
  5016. var windowSize=getWindowSize();
  5017.  
  5018. function keepSI(obj,offsetDirection,defaultValue){
  5019. var objRect=obj.getBoundingClientRect();
  5020. var objStyle=obj.style;
  5021.  
  5022. while(offsetDirection.length){
  5023. var oD=offsetDirection[0];
  5024. var oDV=defaultValue[0];
  5025. offsetDirection.shift();
  5026. defaultValue.shift();
  5027. var oValue=parseFloat(objStyle[oD]);
  5028. var newValue;
  5029. switch(oD){
  5030. case 'top':{
  5031. newValue=oValue - objRect.top;
  5032. if(objRect.top<0){
  5033. newValue=Math.min(newValue,imgWindowFullSize.h);
  5034. }else{
  5035. newValue=Math.max(newValue,oDV);
  5036. };
  5037. }break;
  5038. case 'right':{
  5039. newValue=oValue + (objRect.right - windowSize.w);
  5040. if(objRect.right > windowSize.w){//屏幕外
  5041. newValue=Math.min(newValue,imgWindowFullSize.w);
  5042. }else{
  5043. newValue=Math.max(newValue,oDV);
  5044. };
  5045. }break;
  5046. case 'bottom':{
  5047. newValue=oValue + (objRect.bottom - windowSize.h);
  5048. if(objRect.bottom > windowSize.h){//屏幕外
  5049. newValue=Math.min(newValue,imgWindowFullSize.h);
  5050. }else{
  5051. newValue=Math.max(newValue,oDV);
  5052. };
  5053. }break;
  5054. case 'left':{
  5055. newValue=oValue - objRect.left;
  5056. if(objRect.left<0){
  5057. newValue=Math.min(newValue,imgWindowFullSize.w);
  5058. }else{
  5059. newValue=Math.max(newValue,oDV);
  5060. }
  5061. }break;
  5062. };
  5063. //console.log(newValue);
  5064. objStyle[oD]=newValue + 'px';
  5065.  
  5066. };
  5067. };
  5068.  
  5069. keepSI(this.closeButton,['top','right'],[-24,0]);
  5070. keepSI(this.toolbar,['top','left'],[0,-45]);
  5071. },
  5072. fitToScreen:function(){
  5073. var wSize=getWindowSize();
  5074. //空隙
  5075. wSize.h -= 16;
  5076. wSize.w -= 16;
  5077.  
  5078. var imgWindow=this.imgWindow;
  5079. var imgWindowCS=getComputedStyle(imgWindow);
  5080. var rectSize={
  5081. h:parseFloat(imgWindowCS.height),
  5082. w:parseFloat(imgWindowCS.width),
  5083. };
  5084.  
  5085.  
  5086. var size;
  5087. if(rectSize.w - wSize.w>0 || rectSize.h - wSize.h>0){//超出屏幕,那么缩小。
  5088. if(rectSize.w/rectSize.h > wSize.w/wSize.h){
  5089. size={
  5090. w:wSize.w,
  5091. h:wSize.w / (rectSize.w/rectSize.h),
  5092. };
  5093. }else{
  5094. size={
  5095. h:wSize.h,
  5096. w:wSize.h * (rectSize.w/rectSize.h),
  5097. }
  5098. };
  5099.  
  5100. this.zoom(this.getRotatedImgCliSize(size).w/this.imgNaturalSize.w);
  5101. };
  5102. },
  5103. center:function(horizontal,vertical){
  5104. if(!horizontal && !vertical)return;
  5105. var wSize=getWindowSize();
  5106. var imgWindow=this.imgWindow;
  5107. var scrolled=getScrolled();
  5108. if(horizontal)imgWindow.style.left= (wSize.w - imgWindow.offsetWidth)/2 + scrolled.x +'px';
  5109. if(vertical)imgWindow.style.top= (wSize.h - imgWindow.offsetHeight)/2 + scrolled.y +'px';
  5110. },
  5111.  
  5112.  
  5113. move:function(e){
  5114. this.working=true;
  5115. var cursor=this.cursor;
  5116. this.changeCursor('handing');
  5117.  
  5118. var mouseCoor={
  5119. x:e.pageX,
  5120. y:e.pageY,
  5121. };
  5122. var imgWindow=this.imgWindow;
  5123. var imgWStyle=imgWindow.style;
  5124. var oriOffset={
  5125. left:parseFloat(imgWStyle.left),
  5126. top:parseFloat(imgWStyle.top),
  5127. };
  5128. var self=this;
  5129. var moveHandler=function(e){
  5130. imgWStyle.left=oriOffset.left+ e.pageX-mouseCoor.x +'px';
  5131. imgWStyle.top=oriOffset.top + e.pageY-mouseCoor.y +'px';
  5132. self.keepScreenInside();
  5133. };
  5134. var mouseupHandler=function(){
  5135. e.preventDefault();
  5136. self.changeCursor(cursor);
  5137. self.working=false;
  5138. if(self.tempHand && self.spaceKeyUp){//如果是临时切换到抓手工具,平移完成后返回上个工具
  5139. self.tempHand=false;
  5140. self.changeCursor(self.selectedTool);
  5141. };
  5142. document.removeEventListener('mousemove',moveHandler,true);
  5143. document.removeEventListener('mouseup',mouseupHandler,true);
  5144. };
  5145. document.addEventListener('mousemove',moveHandler,true);
  5146. document.addEventListener('mouseup',mouseupHandler,true);
  5147. },
  5148. rotate:function(origin,topLeft){
  5149.  
  5150. var img=this.img;
  5151. var imgWindow=this.imgWindow;
  5152.  
  5153. var iTransform=img.style[support.cssTransform].replace(/rotate\([^)]*\)/i,'');
  5154.  
  5155. var imgWindowCS=getComputedStyle(imgWindow);
  5156. var imgRectSize={
  5157. h:parseFloat(imgWindowCS.height),
  5158. w:parseFloat(imgWindowCS.width),
  5159. };
  5160.  
  5161. var rectOffset={
  5162. top:parseFloat(imgWindow.style.top),
  5163. left:parseFloat(imgWindow.style.left),
  5164. };
  5165.  
  5166. var imgSize={
  5167. h:img.clientHeight,
  5168. w:img.clientWidth,
  5169. };
  5170.  
  5171. var imgOffset={
  5172. top:parseFloat(img.style.top),
  5173. left:parseFloat(img.style.left),
  5174. };
  5175.  
  5176. var self=this;
  5177. var PI=Math.PI;
  5178.  
  5179. var rotate=function (radians){
  5180. if(self.rotatedRadians==radians)return;
  5181. img.style[support.cssTransform] = ' rotate('+ radians +'rad) ' + iTransform;//旋转图片
  5182. self.rotateIPointer.style[support.cssTransform]='rotate('+ radians +'rad)';//旋转指示器
  5183.  
  5184. self.rotatedRadians=radians;
  5185. self.setToolBadge('rotate',radians/(PI/180));
  5186.  
  5187. var afterimgRectSize=self.getRotatedImgRectSize( radians, imgSize );
  5188. imgWindow.style.width=afterimgRectSize.w +'px';
  5189. imgWindow.style.height=afterimgRectSize.h + 'px';
  5190.  
  5191. if(!topLeft){
  5192. self.setImgWindowOffset(rectOffset,imgRectSize,afterimgRectSize);
  5193. };
  5194.  
  5195. self.setImgOffset(imgOffset,imgRectSize,afterimgRectSize);
  5196. self.keepScreenInside();
  5197. };
  5198.  
  5199.  
  5200. if(typeof origin=='number'){
  5201. rotate(origin);
  5202. return;
  5203. };
  5204.  
  5205.  
  5206. this.working=true;
  5207.  
  5208. var lastRotatedRadians=this.rotatedRadians;
  5209. this.shiftKeyUp=true;
  5210. var shiftRotateStep=prefs.imgWindow.shiftRotateStep / (180/Math.PI);//转成弧度
  5211.  
  5212. var moveHandler=function(e){
  5213. var radians=lastRotatedRadians + Math.atan2( e.clientY - origin.y, e.clientX - origin.x );
  5214. if(radians>=2*PI){
  5215. radians-=2*PI;
  5216. }else if(radians<0){
  5217. radians+=2*PI;
  5218. };
  5219.  
  5220. if(!self.shiftKeyUp){//如果按下了shift键,那么步进缩放
  5221. radians -= radians % shiftRotateStep;
  5222. radians += shiftRotateStep;
  5223. };
  5224. rotate(radians);
  5225. };
  5226.  
  5227. var mouseupHandler=function(){
  5228. self.working=false;
  5229. self.rotateIndicator.style.display='none';
  5230. document.removeEventListener('mousemove',moveHandler,true);
  5231. document.removeEventListener('mouseup',mouseupHandler,true);
  5232. };
  5233.  
  5234. document.addEventListener('mousemove',moveHandler,true);
  5235. document.addEventListener('mouseup',mouseupHandler,true);
  5236. },
  5237. convertToValidRadians:function(radians){
  5238. //转成0-90的等价角度。
  5239. var PI=Math.PI;
  5240. if(radians > PI){
  5241. radians = 2*PI - radians;
  5242. };
  5243. if(radians > 1/2*PI){
  5244. radians = PI - radians;
  5245. };
  5246. return radians;
  5247. },
  5248. getRotatedImgRectSize:function( radians, imgSize ){//通过旋转后的角度和图片的大小,求虚拟矩形的大小
  5249. imgSize= imgSize ? imgSize :{
  5250. h:this.img.clientHeight,
  5251. w:this.img.clentWidth,
  5252. };
  5253.  
  5254. if(typeof radians==='undefined'){
  5255. radians = this.rotatedRadians;
  5256. };
  5257.  
  5258. radians=this.convertToValidRadians(radians);
  5259.  
  5260. return {
  5261. h:this.notExponential(imgSize.h* Math.cos(radians) + imgSize.w * Math.sin(radians)),
  5262. w:this.notExponential(imgSize.h* Math.sin(radians) + imgSize.w * Math.cos(radians)),
  5263. };
  5264. },
  5265. getRotatedImgCliSize:function(rectSize,radians){//通过虚拟矩形的大小和图片的旋转角度,求图片的大小
  5266.  
  5267. if(typeof radians==='undefined'){
  5268. radians = this.rotatedRadians;
  5269. };
  5270.  
  5271. radians=this.convertToValidRadians(radians);
  5272.  
  5273. if(radians==0){
  5274. //radians=Math.PI/180 * 1/100;
  5275. return rectSize;
  5276. };
  5277.  
  5278. var h=(rectSize.h-rectSize.w * Math.tan(radians))/(Math.cos(radians)-Math.sin(radians)*Math.tan(radians));
  5279. var w=(rectSize.h - h*Math.cos(radians))/Math.sin(radians);
  5280. return {
  5281. h:h,
  5282. w:w,
  5283. };
  5284.  
  5285. },
  5286. setImgOffset:function(oriOffset,bImgSize,aImgSize){
  5287. var imgStyle=this.img.style;
  5288.  
  5289. //避免出现指数形式的数字和单位相加,导致变成无效值
  5290. var top=this.notExponential(oriOffset.top + (aImgSize.h-bImgSize.h)*1/2) + 'px';
  5291. var left=this.notExponential(oriOffset.left + (aImgSize.w-bImgSize.w)*1/2) + 'px';
  5292. imgStyle.top= top;
  5293. imgStyle.left= left;
  5294. },
  5295. setImgWindowOffset:function(oriOffset,bImgWindowSize,aImgWidnowSize,ratio){
  5296. ratio= ratio? ratio : {x:1/2,y:1/2};
  5297. var imgWindowStyle=this.imgWindow.style;
  5298. var top=oriOffset.top - (aImgWidnowSize.h-bImgWindowSize.h)*ratio.y + 'px';
  5299. var left=oriOffset.left - (aImgWidnowSize.w-bImgWindowSize.w)*ratio.x + 'px';
  5300. imgWindowStyle.top= top;
  5301. imgWindowStyle.left= left;
  5302. },
  5303. zoom:function(e,ratio){//e可能是undefined,可能是事件对象,可能是直接的缩放级别数字
  5304. var imgWindow=this.imgWindow;
  5305. var imgWindowCS=getComputedStyle(imgWindow);
  5306. var imgRectSize={
  5307. h:parseFloat(imgWindowCS.height),
  5308. w:parseFloat(imgWindowCS.width),
  5309. };
  5310.  
  5311. var rectOffset={
  5312. top:parseFloat(imgWindow.style.top),
  5313. left:parseFloat(imgWindow.style.left),
  5314. };
  5315.  
  5316. var img=this.img;
  5317. var self=this;
  5318.  
  5319. var zoom=function(level){//缩放到指定级别
  5320. if(typeof level=='undefined' || level<0 || level==self.zoomLevel)return;
  5321.  
  5322. var afterImgSize={
  5323. h:self.imgNaturalSize.h * level,
  5324. w:self.imgNaturalSize.w * level,
  5325. };
  5326. img.width=afterImgSize.w;
  5327. img.height=afterImgSize.h;
  5328.  
  5329. var afterimgRectSize=self.getRotatedImgRectSize( self.rotatedRadians, afterImgSize );
  5330. //console.log(afterimgRectSize);
  5331. imgWindow.style.width=afterimgRectSize.w +'px';
  5332. imgWindow.style.height=afterimgRectSize.h + 'px';
  5333. self.setImgWindowOffset(rectOffset,imgRectSize,afterimgRectSize,ratio);
  5334. self.setImgOffset({top:0,left:0},afterImgSize,afterimgRectSize);//如果旋转了,调整偏移
  5335. self.zoomLevel=level;
  5336. self.setToolBadge('zoom',level);
  5337. self.keepScreenInside();
  5338. };
  5339.  
  5340. if(typeof e!='object'){
  5341. ratio=ratio? ratio : {
  5342. x:1/2,
  5343. y:1/2,
  5344. };
  5345. zoom(e);
  5346. return;
  5347. };
  5348.  
  5349. this.working=true;
  5350.  
  5351. ratio=this.getZoomRatio({
  5352. x:e.clientX,
  5353. y:e.clientY,
  5354. });
  5355.  
  5356.  
  5357. var moved;
  5358. var lastPageX=e.pageX;
  5359. var currentLevel=this.zoomLevel;
  5360. var moveFired=0;
  5361. var moveHandler=function(e){
  5362. moveFired++
  5363. if(moveFired < 2){//有时候点击的时候不小心会触发一发move
  5364. return;
  5365. };
  5366. moved=true;
  5367. var pageX=e.pageX;
  5368. var level;
  5369. if(pageX > lastPageX){//向右移,zoomin扩大
  5370. self.changeCursor('zoom',false);
  5371. level=0.05;
  5372. }else{//向左移,zoomout缩小
  5373. self.changeCursor('zoom',true);
  5374. level=-0.05;
  5375. };
  5376. lastPageX=pageX;
  5377. currentLevel += level;
  5378. zoom(currentLevel);
  5379. };
  5380.  
  5381. var mouseupHandler=function(e){
  5382. self.working=false;
  5383. document.removeEventListener('mousemove',moveHandler,true);
  5384. document.removeEventListener('mouseup',mouseupHandler,true);
  5385.  
  5386. var level=self.getNextZoomLevel();
  5387.  
  5388. if(self.zoomOut && self.altKeyUp){
  5389. self.zoomOut=false;
  5390. };
  5391.  
  5392. if(!moved){//如果没有平移缩放。
  5393. zoom(level);
  5394. };
  5395.  
  5396. self.changeCursor('zoom',self.zoomOut);
  5397.  
  5398. if(self.tempZoom && self.ctrlKeyUp && self.altKeyUp){
  5399. self.tempZoom=false;
  5400. self.changeCursor(self.selectedTool);
  5401. };
  5402.  
  5403. };
  5404.  
  5405. document.addEventListener('mousemove',moveHandler,true);
  5406. document.addEventListener('mouseup',mouseupHandler,true);
  5407. },
  5408. getNextZoomLevel:function(){
  5409. var level;
  5410. var self=this;
  5411. if(this.zoomOut){//缩小
  5412. ImgWindowC.zoomRangeR._find(function(value){
  5413. if(value < self.zoomLevel){
  5414. level=value;
  5415. return true;
  5416. }
  5417. })
  5418. }else{
  5419. ImgWindowC.zoomRange._find(function(value){
  5420. if(value > self.zoomLevel){
  5421. level=value;
  5422. return true;
  5423. };
  5424. });
  5425. }
  5426. return level;
  5427. },
  5428. getZoomRatio:function(mouseCoor){
  5429. var ibcRect=this.img.getBoundingClientRect();
  5430. var ratio={
  5431. x:(mouseCoor.x-ibcRect.left)/ibcRect.width,
  5432. y:(mouseCoor.y-ibcRect.top)/ibcRect.height,
  5433. };
  5434. if(ratio.x<0){
  5435. ratio.x=0
  5436. }else if(ratio.x>1){
  5437. ratio.x=1
  5438. };
  5439. if(ratio.y<0){
  5440. ratio.y=0
  5441. }else if(ratio.y>1){
  5442. ratio.y=1
  5443. };
  5444. return ratio;
  5445. },
  5446. aerialView:function(e){
  5447. this.working=true;
  5448. //记住现在的缩放比例
  5449. var cLevel=this.zoomLevel;
  5450.  
  5451. var wSize=getWindowSize();
  5452. wSize.h -= 16;
  5453. wSize.w -= 16;
  5454.  
  5455. var imgWindow=this.imgWindow;
  5456. var imgWindowCS=getComputedStyle(imgWindow);
  5457. var rectSize={
  5458. h:parseFloat(imgWindowCS.height),
  5459. w:parseFloat(imgWindowCS.width),
  5460. };
  5461. var rectRatio=rectSize.h/rectSize.w;
  5462. var windowRatio=wSize.h/wSize.w;
  5463.  
  5464. var size;
  5465. var rangeSize={};
  5466. if(rectRatio > windowRatio){
  5467. size={
  5468. h:wSize.h,
  5469. w:wSize.h / rectRatio,
  5470. };
  5471. rangeSize.h=Math.min(wSize.h * (size.h / rectSize.h), size.h);
  5472. rangeSize.w=Math.min(rangeSize.h / windowRatio , size.w);
  5473. }else{
  5474. size={
  5475. w:wSize.w,
  5476. h:wSize.w * rectRatio,
  5477. };
  5478. rangeSize.w=Math.min(wSize.w * (size.w / rectSize.w), size.w);
  5479. rangeSize.h=Math.min(rangeSize.w * windowRatio , size.h);
  5480. };
  5481.  
  5482.  
  5483. this.zoom(this.getRotatedImgCliSize(size).w/this.imgNaturalSize.w);
  5484.  
  5485. this.center(true,true);
  5486.  
  5487. this.keepScreenInside();
  5488.  
  5489. var viewRange=this.viewRange;
  5490. var vRS=viewRange.style;
  5491. vRS.display='block';
  5492. vRS.height=rangeSize.h + 'px';
  5493. vRS.width=rangeSize.w + 'px';
  5494. vRS.top=0 + 'px';
  5495. vRS.left=0 + 'px';
  5496.  
  5497.  
  5498.  
  5499. var viewRangeRect=viewRange.getBoundingClientRect();
  5500. var scrolled=getScrolled();
  5501. var viewRangeCenterCoor={
  5502. x:viewRangeRect.left + scrolled.x + 1/2 * rangeSize.w,
  5503. y:viewRangeRect.top + scrolled.y + 1/2 * rangeSize.h,
  5504. };
  5505.  
  5506. var self=this;
  5507.  
  5508. var moveRange={
  5509. x:[8,8+size.w-rangeSize.w],
  5510. y:[8,8+size.h-rangeSize.h]
  5511. };
  5512.  
  5513.  
  5514. function setViewRangePosition(pageXY){
  5515. var top=pageXY.y - viewRangeCenterCoor.y;
  5516. var left=pageXY.x - viewRangeCenterCoor.x;
  5517. if(top<=moveRange.y[0]){
  5518. top=moveRange.y[0];
  5519. }else if(top>=moveRange.y[1]){
  5520. top=moveRange.y[1];
  5521. };
  5522. vRS.top= top + 'px';
  5523. if(left<=moveRange.x[0]){
  5524. left=moveRange.x[0];
  5525. }else if(left>=moveRange.x[1]){
  5526. left=moveRange.x[1];
  5527. };
  5528. vRS.left= left + 'px';
  5529. };
  5530.  
  5531. setViewRangePosition({
  5532. x:e.pageX,
  5533. y:e.pageY,
  5534. });
  5535.  
  5536. var moveHandler=function(e){
  5537. setViewRangePosition({
  5538. x:e.pageX,
  5539. y:e.pageY,
  5540. });
  5541. };
  5542.  
  5543. var mouseupHandler=function(){
  5544. self.working=false;
  5545. viewRange.style.display='none';
  5546. self.zoom(cLevel);
  5547. var scrolled=getScrolled();
  5548. imgWindow.style.top= -13 - rectSize.h * ((parseFloat(vRS.top) - moveRange.y[0])/size.h) + scrolled.y +'px';
  5549. imgWindow.style.left= -13 - rectSize.w * ((parseFloat(vRS.left) - moveRange.x[0])/size.w) + scrolled.x +'px';
  5550.  
  5551. //说明图片的高度没有屏幕高,居中
  5552. //说明图片的宽度没有屏幕宽,居中
  5553. self.center(rangeSize.w == size.w , rangeSize.h == size.h);
  5554.  
  5555. self.keepScreenInside();
  5556.  
  5557. document.removeEventListener('mousemove',moveHandler,true);
  5558. document.removeEventListener('mouseup',mouseupHandler,true);
  5559. };
  5560. document.addEventListener('mousemove',moveHandler,true);
  5561. document.addEventListener('mouseup',mouseupHandler,true);
  5562. },
  5563. setToolBadge:function(tool,content){
  5564. var scale=0;
  5565. switch(tool){
  5566. case 'zoom':{
  5567. scale=2;
  5568. }break;
  5569. case 'rotate':{
  5570. scale=1;
  5571. }break;
  5572. default:break;
  5573. }
  5574. content=typeof content=='string'? content : content.toFixed(scale);
  5575. this.toolMap[tool].nextElementSibling.textContent=content;
  5576. },
  5577. notExponential:function(num){//不要转为指数形势
  5578. if(num>0){
  5579. if(num >= 999999999999999934463){
  5580. return 999999999999999934463;
  5581. }else if(num <= 0.000001){
  5582. return 0.000001;
  5583. };
  5584. }else if(num < 0){
  5585. if(num >= -0.000001){
  5586. return -0.000001;
  5587. }else if(num <= -999999999999999934463){
  5588. return -999999999999999934463
  5589. };
  5590. };
  5591.  
  5592. return num;
  5593. },
  5594.  
  5595. blur:function(e){
  5596. if(!this.focused)return;
  5597. var imgWindow =this.imgWindow;
  5598. //点击imgWinodw的外部的时候失去焦点
  5599. if(e!==true && imgWindow.contains(e.target))return;
  5600. imgWindow.classList.remove('pv-pic-window-container_focus');
  5601. this.toolbar.classList.remove('pv-pic-window-toolbar_focus');
  5602. this.closeButton.classList.remove('pv-pic-window-close_focus');
  5603. this.img.classList.remove('pv-pic-window-pic_focus');
  5604. document.removeEventListener('mousedown',this._blur,true);
  5605. document.removeEventListener('keydown',this._focusedKeydown,true);
  5606. document.removeEventListener('keyup',this._focusedKeyup,true);
  5607. this.changeCursor('default');
  5608. ImgWindowC.selectedTool=this.selectedTool;
  5609. this.focused=false;
  5610. },
  5611. focus:function(){
  5612. if(this.focused)return;
  5613. this.imgWindow.classList.add('pv-pic-window-container_focus');
  5614. this.toolbar.classList.add('pv-pic-window-toolbar_focus');
  5615. this.closeButton.classList.add('pv-pic-window-close_focus');
  5616. this.img.classList.add('pv-pic-window-pic_focus');
  5617. this.imgWindow.style.zIndex= ImgWindowC.styleZIndex;
  5618. this.zIndex=ImgWindowC.styleZIndex;
  5619. ImgWindowC.styleZIndex ++;
  5620. document.addEventListener('keydown',this._focusedKeydown,true);
  5621. document.addEventListener('keyup',this._focusedKeyup,true);
  5622. document.addEventListener('mousedown',this._blur,true);
  5623.  
  5624. //还原鼠标样式。
  5625. this.changeCursor(this.selectedTool);
  5626.  
  5627. if(prefs.imgWindow.syncSelectedTool && ImgWindowC.selectedTool){
  5628. this.selectTool(ImgWindowC.selectedTool);
  5629. };
  5630.  
  5631. this.focused=true;
  5632. },
  5633. focusedKeyup:function(e){
  5634. var keyCode=e.keyCode;
  5635. var valid=[32,18,16,72,17,72,82,90,67];
  5636. if(valid.indexOf(keyCode)==-1)return;
  5637.  
  5638. e.preventDefault();
  5639.  
  5640. switch(keyCode){
  5641. case 32:{//空格键,临时切换到移动
  5642. this.spaceKeyUp=true;
  5643. if(!this.tempHand)return;//如果之前没有临时切换到抓手工具(当已经在工作的时候,按下空格不会临时切换到抓手工具)
  5644. if(!this.working){//松开按键的时候,没有在继续平移了。
  5645. this.tempHand=false;
  5646. this.changeCursor(this.selectedTool);
  5647. };
  5648. }break;
  5649. case 18:{//alt键盘切换缩小放大。
  5650. this.altKeyUp=true;
  5651. if(!this.zoomOut)return;
  5652. if(!this.working){
  5653. this.zoomOut=false;
  5654. this.changeCursor('zoom');
  5655. if(this.tempZoom && this.ctrlKeyUp){
  5656. this.tempZoom=false;
  5657. this.changeCursor(this.selectedTool);
  5658. };
  5659. };
  5660. }break;
  5661. case 16:{//shift键,旋转的时候按住shift键,步进缩放。
  5662. this.shiftKeyUp=true;
  5663. }break;
  5664. case 17:{//ctrl键
  5665. clearTimeout(this.ctrlkeyDownTimer);
  5666. if(!this.justCKeyUp){//如果刚才没有松开c,规避划词软件的ctrl+c松开
  5667. this.ctrlKeyUp=true;
  5668. if(!this.tempZoom)return;//如果没有切换到了缩放
  5669. if(!this.working && this.altKeyUp){
  5670. this.tempZoom=false;
  5671. this.changeCursor(this.selectedTool);
  5672. };
  5673. };
  5674. }break;
  5675. case 67:{//c键
  5676. this.justCKeyUp=true;
  5677. var self=this;
  5678. clearTimeout(this.justCKeyUpTimer);
  5679. this.justCKeyUpTimer=setTimeout(function(){
  5680. self.justCKeyUp=false;
  5681. },100)
  5682. }break;
  5683. case 72:{//h键
  5684. this.hKeyUp=true;
  5685. }break;
  5686. case 82:{//r键
  5687. this.rKeyUp=true;
  5688. }break;
  5689. case 90:{//z键
  5690. this.zKeyUp=true;
  5691. }break;
  5692. default:break;
  5693. };
  5694.  
  5695. if([72,82,90].indexOf(keyCode)!=-1){
  5696. if(!this.working && this.restoreBeforeTool){
  5697. this.restoreBeforeTool=false;
  5698. this.selectTool(this.beforeTool);
  5699. };
  5700. };
  5701. },
  5702. focusedKeydown:function(e){
  5703. var keyCode=e.keyCode;
  5704. var valid=[32,82,72,90,18,16,17,27,67];//有效的按键
  5705. if(valid.indexOf(keyCode)==-1) return;
  5706.  
  5707. e.preventDefault();
  5708.  
  5709. if(this.working){//working的时候也可以接受按下shift键,以便旋转的时候可以任何时候按下
  5710. if(keyCode==16){//shift键
  5711. this.shiftKeyUp=false;
  5712. };
  5713. return;
  5714. };
  5715.  
  5716. switch(keyCode){
  5717. case 82:{//r键,切换到旋转工具
  5718. if(this.rKeyUp){
  5719. this.rKeyUp=false;
  5720. this.beforeTool=this.selectedTool;
  5721. this.selectTool('rotate');
  5722. };
  5723. }break;
  5724. case 72:{//h键,切换到抓手工具
  5725. if(this.hKeyUp){
  5726. this.hKeyUp=false;
  5727. this.beforeTool=this.selectedTool;
  5728. this.selectTool('hand');
  5729. };
  5730. }break;
  5731. case 90:{//z键,切换到缩放工具
  5732. if(this.zKeyUp){
  5733. this.zKeyUp=false;
  5734. this.beforeTool=this.selectedTool;
  5735. this.selectTool('zoom');
  5736. };
  5737. }break;
  5738. case 32:{//空格键阻止,临时切换到抓手功能
  5739. if(this.spaceKeyUp){
  5740. this.spaceKeyUp=false;
  5741. if(this.selectedTool!='hand'){
  5742. this.tempHand=true;
  5743. this.changeCursor('hand');
  5744. };
  5745. };
  5746. }break;
  5747. case 18:{//alt键,在当前选择是缩放工具的时候,按下的时候切换到缩小功能
  5748. if(this.altKeyUp){
  5749. if((this.selectedTool!='zoom' && !this.tempZoom) || this.zoomOut)return;
  5750. this.zoomOut=true;
  5751. this.altKeyUp=false;
  5752. this.changeCursor('zoom',true);
  5753. };
  5754. }break;
  5755. case 17:{//ctrl键临时切换到缩放工具
  5756. if(this.ctrlKeyUp){
  5757. var self=this;
  5758. this.ctrlkeyDownTimer=setTimeout(function(){//规避词典软件的ctrl+c,一瞬间切换到缩放的问题
  5759. self.ctrlKeyUp=false;
  5760. if(self.selectedTool!='zoom'){
  5761. self.tempZoom=true;
  5762. self.changeCursor('zoom');
  5763. };
  5764. },100);
  5765. };
  5766. }break;
  5767. case 67:{//c键
  5768. clearTimeout(this.ctrlkeyDownTimer);
  5769. }break;
  5770. case 27:{//ese关闭窗口
  5771. if(prefs.imgWindow.close.escKey){
  5772. this.remove();
  5773. };
  5774. }break;
  5775. default:break;
  5776. };
  5777. },
  5778.  
  5779. toolbarEventHandler:function(e){
  5780. e.stopPropagation();
  5781. var target=e.target;
  5782. var toolMap=this.toolMap;
  5783. for(var i in toolMap){
  5784. if(toolMap.hasOwnProperty(i) && toolMap[i]==target){
  5785. switch(e.type){
  5786. case 'mousedown':{
  5787. this.selectTool(i);
  5788. }break;
  5789. case 'dblclick':{
  5790. this.dblclickCommand(i);
  5791. }break;
  5792. default:break;
  5793. };
  5794. break;
  5795. };
  5796. };
  5797. },
  5798. imgWindowEventHandler:function(e){
  5799. e.stopPropagation();
  5800. switch(e.type){
  5801. case 'click':{//阻止opera的图片保存
  5802. if(e.ctrlKey && e.target.nodeName=='IMG'){
  5803. e.preventDefault();
  5804. };
  5805. }break;
  5806. case 'mousedown':{
  5807. if(!this.focused){//如果没有focus,先focus
  5808. this.focus();
  5809. this.keepScreenInside();
  5810. };
  5811.  
  5812. var target=e.target;
  5813. if(e.button==2){//由于rotate时候的覆盖层问题,修复右键的图片菜单弹出
  5814. if(target!=this.rotateOverlayer)return;
  5815. var self=this;
  5816. this.rotateOverlayer.style.display='none';
  5817. var upHandler=function(){
  5818. document.removeEventListener('mouseup',upHandler,true);
  5819. setTimeout(function(){
  5820. self.rotateOverlayer.style.display='block';
  5821. },10);
  5822. };
  5823. document.addEventListener('mouseup',upHandler,true);
  5824. return;
  5825. };
  5826.  
  5827. if(e.button!=0 || (target!=this.imgWindow && target!=this.img && target!=this.rotateOverlayer))return;
  5828. e.preventDefault();
  5829. var selectedTool=this.selectedTool;
  5830. if(this.tempHand){
  5831. this.move(e);
  5832. }else if(this.tempZoom){
  5833. this.zoom(e);
  5834. }else if(selectedTool=='hand'){
  5835. this.restoreBeforeTool=!this.hKeyUp;
  5836. if(this.hKeyUp){
  5837. this.move(e);
  5838. }else{//鸟瞰视图
  5839. this.aerialView(e);
  5840. };
  5841. }else if(selectedTool=='rotate'){
  5842. var origin={//旋转原点
  5843. x:e.clientX - 30,//稍微偏左一点。
  5844. y:e.clientY ,
  5845. };
  5846.  
  5847. var rIS=this.rotateIndicator.style;
  5848. rIS.display='block';
  5849. rIS.top=origin.y + 'px';
  5850. rIS.left=origin.x + 'px';
  5851.  
  5852. this.restoreBeforeTool=!this.rKeyUp;
  5853. this.rotate(origin);
  5854. }else if(selectedTool=='zoom'){
  5855. this.restoreBeforeTool=!this.zKeyUp;
  5856. this.zoom(e);
  5857. };
  5858. }break;
  5859. case 'wheel':{
  5860. if(!this.focused)return;//如果没有focus
  5861. if(e.deltaY===0)return;//非Y轴的滚动
  5862. e.preventDefault();
  5863. if(this.working)return;
  5864. var oriZoomOut=this.zoomOut;
  5865. this.zoomOut = !!(e.deltaY > 0);
  5866.  
  5867. var ratio=this.getZoomRatio({
  5868. x:e.clientX,
  5869. y:e.clientY,
  5870. });
  5871.  
  5872. var level=this.getNextZoomLevel();
  5873.  
  5874. this.zoom(level,ratio);
  5875. this.zoomOut=oriZoomOut;
  5876. }break;
  5877. default:break;
  5878. };
  5879. },
  5880.  
  5881. dblclickCommand:function(tool){
  5882. var done;
  5883. switch(tool){
  5884. case 'hand':{//双击居中,并且适应屏幕
  5885. this.zoom(1);
  5886. this.fitToScreen();
  5887. this.center(true,true);
  5888. this.keepScreenInside();
  5889. }break;
  5890. case 'rotate':{//双击还原旋转
  5891. if(this.rotatedRadians==0)return;
  5892. done=true;
  5893. this.rotate(0,true);
  5894. }break;
  5895. case 'zoom':{//双击还原缩放
  5896. if(this.zoomLevel==1)return;
  5897. done=true;
  5898. this.zoom(1,{x:0,y:0});
  5899. }break;
  5900. default:break;
  5901. };
  5902.  
  5903. if((tool=='rotate' || tool=='zoom') && done){
  5904. var scrolled=getScrolled();
  5905. var imgWindow=this.imgWindow;
  5906. var imgWinodowRect=imgWindow.getBoundingClientRect();
  5907. var imgWindowStyle=imgWindow.style;
  5908. if(imgWinodowRect.left<40){
  5909. imgWindowStyle.left=40 + scrolled.x + 'px';
  5910. };
  5911. if(imgWinodowRect.top<-5){
  5912. imgWindowStyle.top=-5 + scrolled.y +'px';
  5913. };
  5914. this.keepScreenInside();
  5915. };
  5916.  
  5917. },
  5918. doFlipCommand:function(command){
  5919. var map={
  5920. fv:[/scaleY\([^)]*\)/i,' scaleY(-1) '],
  5921. fh:[/scaleX\([^)]*\)/i,' scaleX(-1) '],
  5922. };
  5923.  
  5924. var iTransform=this.img.style[support.cssTransform];
  5925.  
  5926. var toolClassList=this.toolMap[command].classList;
  5927.  
  5928. if(map[command][0].test(iTransform)){
  5929. iTransform=iTransform.replace(map[command][0],'');
  5930. toolClassList.remove(this.selectedToolClass);
  5931. }else{
  5932. iTransform += map[command][1];
  5933. toolClassList.add(this.selectedToolClass);
  5934. };
  5935. this.img.style[support.cssTransform]=iTransform;
  5936.  
  5937. },
  5938. selectTool:function(tool){
  5939. var command=['fv','fh'];
  5940. if(command.indexOf(tool)==-1){//工具选择
  5941. if(this.selectedTool==tool)return;
  5942. var selectedTool=this.selectedTool;
  5943. this.selectedTool=tool;
  5944. if(this.tempHand || this.tempZoom){//临时工具中。不变鼠标
  5945. return;
  5946. };
  5947.  
  5948. this.rotateOverlayer.style.display=(tool=='rotate'? 'block' : 'none');//这个覆盖层是为了捕捉双击或者单击事件。
  5949.  
  5950. if(selectedTool){
  5951. this.toolMap[selectedTool].classList.remove(this.selectedToolClass);
  5952. };
  5953. this.toolMap[tool].classList.add(this.selectedToolClass);
  5954. this.changeCursor(tool);
  5955. }else{//命令
  5956. this.doFlipCommand(tool);
  5957. };
  5958. },
  5959. changeCursor:function(tool,zoomOut){
  5960. if(tool=='zoom'){
  5961. tool+=zoomOut? '-out' : '-in';
  5962. };
  5963. if(this.cursor==tool)return;
  5964. this.cursor=tool;
  5965.  
  5966. var cursor;
  5967.  
  5968. switch(tool){
  5969. case 'hand':{
  5970. cursor=support.cssCursorValue.grab || 'pointer';
  5971. }break;
  5972. case 'handing':{
  5973. cursor=support.cssCursorValue.grabbing || 'pointer';
  5974. }break;
  5975. case 'zoom-in':{
  5976. cursor=support.cssCursorValue.zoomIn;
  5977. }break;
  5978. case 'zoom-out':{
  5979. cursor=support.cssCursorValue.zoomOut;
  5980. }break;
  5981. case 'rotate':{
  5982. cursor='progress';
  5983. }break;
  5984. case 'default':{
  5985. cursor='';
  5986. }break;
  5987. };
  5988.  
  5989. if(typeof cursor!='undefined'){
  5990. this.imgWindow.style.cursor=cursor;
  5991. };
  5992.  
  5993. },
  5994.  
  5995. remove:function(){
  5996. if(this.removed)return;
  5997. this.removed=true;
  5998. this.blur(true);
  5999. this.img.src= prefs.icons.brokenImg_small;//如果在加载中取消,图片也取消读取。
  6000.  
  6001. this.imgWindow.parentNode.removeChild(this.imgWindow);
  6002.  
  6003. //点击点击外部关闭的监听
  6004. if(prefs.imgWindow.close.clickOutside.enabled){
  6005. document.removeEventListener(prefs.imgWindow.close.clickOutside.trigger,this.clickOutside,true);
  6006. };
  6007.  
  6008. var index=ImgWindowC.all.indexOf(this);
  6009. ImgWindowC.all.splice(index,1);
  6010.  
  6011. //focus next
  6012. if(ImgWindowC.all.length==0){
  6013. if(ImgWindowC.overlayer){
  6014. ImgWindowC.overlayer.style.display='none';
  6015. };
  6016. }else{
  6017. var topmost=0;
  6018. ImgWindowC.all.forEach(function(iwin){
  6019. if(iwin.zIndex > topmost){
  6020. topmost=iwin;
  6021. };
  6022. });
  6023. if(topmost){
  6024. topmost.focus();
  6025. };
  6026. };
  6027.  
  6028. },
  6029.  
  6030. };
  6031.  
  6032.  
  6033. //载入动画
  6034. function LoadingAnimC(data,buttonType,waitImgLoad,openInTopWindow){
  6035. this.args=arrayFn.slice.call(arguments,0);
  6036. this.data=data;//data
  6037. this.buttonType=buttonType;//点击的按钮类型
  6038. this.openInTopWindow=openInTopWindow;//是否在顶层窗口打开,如果在frame里面的话
  6039. this.waitImgLoad=waitImgLoad;//是否等待完全读取后打开
  6040. this.init();
  6041. };
  6042.  
  6043. LoadingAnimC.all=[];
  6044.  
  6045. LoadingAnimC.prototype={
  6046. init:function(){
  6047. LoadingAnimC.all.push(this);
  6048. this.addStyle();
  6049. var container=document.createElement('span');
  6050.  
  6051. container.className='pv-loading-container';
  6052. this.loadingAnim=container;
  6053.  
  6054. container.title='正在加载:' + this.data.src;
  6055. container.innerHTML=
  6056. '<span class="pv-loading-button pv-loading-retry" title="重试"></span>'+
  6057. '<span class="pv-loading-button pv-loading-cancle" title="取消"></span>';
  6058.  
  6059. document.body.appendChild(container);
  6060.  
  6061. var self=this;
  6062. container.addEventListener('click',function(e){
  6063. var tcl=e.target.classList;
  6064. if(tcl.contains('pv-loading-cancle')){
  6065. self.imgReady.abort();
  6066. self.remove();
  6067. }else if(tcl.contains('pv-loading-retry')){
  6068. self.remove();
  6069. new LoadingAnimC(self.args[0],self.args[1],self.args[2],self.args[3]);
  6070. };
  6071. },true);
  6072.  
  6073.  
  6074. this.setPosition();
  6075.  
  6076. var img=document.createElement('img');
  6077. img.src= this.buttonType=='current'? this.data.imgSrc : this.data.src;
  6078.  
  6079. var opts={
  6080. error:function(e){
  6081. self.error(this,e);
  6082. },
  6083. };
  6084.  
  6085. opts[this.waitImgLoad? 'load' : 'ready' ]=function(e){
  6086. self.load(this,e);
  6087. };
  6088.  
  6089. this.imgReady=imgReady(img,opts);
  6090. },
  6091. addStyle:function(){
  6092. if(LoadingAnimC.styleAdded)return;
  6093. LoadingAnimC.styleAdded=true;
  6094. var style=document.createElement('style');
  6095. style.type='text/css';
  6096. style.textContent='\
  6097. .pv-loading-container {\
  6098. position: absolute;\
  6099. z-index:999999997;\
  6100. background: black url("'+prefs.icons.loading+'") center no-repeat;\
  6101. background-origin: content-box;\
  6102. border: none;\
  6103. padding: 1px 30px 1px 2px;\
  6104. margin: 0;\
  6105. opacity: 0.7;\
  6106. height: 24px;\
  6107. min-width: 24px;\
  6108. box-shadow: 2px 2px 0px #666;\
  6109. -webkit-transition: opacity 0.15s ease-in-out;\
  6110. transition: opacity 0.15s ease-in-out;\
  6111. }\
  6112. .pv-loading-container:hover {\
  6113. opacity: 0.9;\
  6114. }\
  6115. .pv-loading-button {\
  6116. cursor: pointer;\
  6117. height: 24px;\
  6118. width: 24px;\
  6119. position: absolute;\
  6120. right: 0;\
  6121. top: 0;\
  6122. opacity: 0.4;\
  6123. background:transparent center no-repeat;\
  6124. -webkit-transition: opacity 0.15s ease-in-out;\
  6125. transition: opacity 0.15s ease-in-out;\
  6126. }\
  6127. .pv-loading-button:hover {\
  6128. opacity: 1;\
  6129. }\
  6130. .pv-loading-cancle{\
  6131. background-image: url("'+prefs.icons.loadingCancle+'");\
  6132. }\
  6133. .pv-loading-retry{\
  6134. display:none;\
  6135. background-image: url("'+prefs.icons.retry+'");\
  6136. }\
  6137. .pv-loading-container_error{\
  6138. background-image:none;\
  6139. }\
  6140. .pv-loading-container_error::after{\
  6141. content:"加载失败";\
  6142. line-height: 24px;\
  6143. color: red;\
  6144. font-size: 14px;\
  6145. display:inline;\
  6146. }\
  6147. .pv-loading-container_error .pv-loading-cancle{\
  6148. display:none;\
  6149. }\
  6150. .pv-loading-container_error .pv-loading-retry{\
  6151. display:block;\
  6152. }\
  6153. ';
  6154. document.head.appendChild(style);
  6155. },
  6156. remove:function(){
  6157. if(!this.removed){
  6158. this.removed=true;
  6159. this.loadingAnim.parentNode.removeChild(this.loadingAnim);
  6160. LoadingAnimC.all.splice(LoadingAnimC.all.indexOf(this),1);
  6161. };
  6162. },
  6163. error:function(img,e){
  6164. this.loadingAnim.classList.add('pv-loading-container_error');
  6165. console.error('picviewer CE 载入大图错误:%o', this.data);
  6166.  
  6167. var self=this;
  6168. setTimeout(function(){
  6169. self.remove();
  6170. },3000);
  6171. },
  6172. setPosition:function(){
  6173. var position=getContentClientRect(this.data.img);
  6174. var cs=this.loadingAnim.style;
  6175. var scrolled=getScrolled();
  6176. cs.top=position.top + scrolled.y +1 + 'px';
  6177. cs.left=position.left + scrolled.x +1 + 'px';
  6178. cs.removeProperty('display');
  6179. },
  6180. load:function(img,e){
  6181. this.remove();
  6182. this.img=img;
  6183. var buttonType=this.buttonType;
  6184.  
  6185. if(buttonType=='gallery'){
  6186. var allData=this.getAllValidImgs();
  6187. allData.target=this.data;
  6188. this.data=allData;
  6189. };
  6190.  
  6191. var self=this;
  6192. function openInTop(){
  6193. var data=self.data;
  6194.  
  6195. //删除不能发送的项。
  6196. var delCantClone=function(obj){
  6197. delete obj.img;
  6198. delete obj.imgPA;
  6199. };
  6200.  
  6201. if(Array.isArray(data)){
  6202. frameSentSuccessData=frameSentData;
  6203. frameSentData=cloneObject(data,true);//备份一次
  6204. //console.log(frameSentData);
  6205.  
  6206. delCantClone(data.target);
  6207. data.forEach(function(obj){
  6208. delCantClone(obj);
  6209. });
  6210. }else{
  6211. delCantClone(data);
  6212. };
  6213.  
  6214. window.postMessage({
  6215. messageID:messageID,
  6216. src:img.src,
  6217. data:data,
  6218. command:'open',
  6219. buttonType:buttonType,
  6220. to:'top',
  6221. },'*');
  6222. };
  6223.  
  6224. if(this.openInTopWindow && isFrame && topWindowValid!==false && buttonType!='magnifier'){
  6225. if(topWindowValid){
  6226. openInTop();
  6227. }else{//先发消息问问顶层窗口是不是非frameset窗口
  6228. window.postMessage({
  6229. messageID:messageID,
  6230. command:'topWindowValid',
  6231. to:'top',
  6232. },'*');
  6233.  
  6234. document.addEventListener('pv-topWindowValid',function(e){
  6235. topWindowValid=e.detail;
  6236. if(topWindowValid){//如果顶层窗口有效
  6237. openInTop()
  6238. }else{
  6239. self.open();
  6240. };
  6241. },true);
  6242. };
  6243.  
  6244. }else{
  6245. this.open();
  6246. };
  6247.  
  6248.  
  6249. },
  6250. getAllValidImgs:function(){
  6251. var imgs=document.getElementsByTagName('img'),//html collection
  6252. validImgs=[]
  6253. ;
  6254. arrayFn.forEach.call(imgs,function(img,index,imgs){
  6255. var result=findPic(img);
  6256. if(result){
  6257. validImgs.push(result);
  6258. };
  6259. });
  6260. return validImgs;
  6261. },
  6262. open:function(){
  6263. switch(this.buttonType){
  6264. case 'gallery':{
  6265. if(!gallery){
  6266. gallery=new GalleryC();
  6267. };
  6268. gallery.load(this.data,this.from);
  6269. }break;
  6270. case 'magnifier':{
  6271. new MagnifierC(this.img,this.data);
  6272. }break;
  6273. case 'actual':;
  6274. case 'current':;
  6275. case 'original':{//original 是为了兼容以前的规则
  6276. new ImgWindowC(this.img);
  6277. }break;
  6278. };
  6279. },
  6280. };
  6281.  
  6282.  
  6283. //工具栏
  6284. function FloatBarC(){
  6285. this.init();
  6286. };
  6287.  
  6288.  
  6289. FloatBarC.prototype={
  6290. init:function(){
  6291. this.addStyle();
  6292. var container=document.createElement('span');
  6293. container.id='pv-float-bar-container';
  6294. container.innerHTML=
  6295. '<span class="pv-float-bar-button"></span>'+
  6296. '<span class="pv-float-bar-button"></span>'+
  6297. '<span class="pv-float-bar-button"></span>'+
  6298. '<span class="pv-float-bar-button"></span>';
  6299. document.body.appendChild(container);
  6300.  
  6301. var buttons={
  6302. };
  6303. this.buttons=buttons;
  6304. this.children=container.children;
  6305.  
  6306. arrayFn.forEach.call(this.children,function(child,index){
  6307. var titleMap={
  6308. actual:'查看原始(A)',
  6309. gallery:'查看库(G)',
  6310. current:'查看当前(C)',
  6311. magnifier:'放大镜(M)',
  6312. };
  6313. var buttonName=prefs.floatBar.butonOrder[index];
  6314. buttons[buttonName]=child;
  6315. child.title=titleMap[buttonName];
  6316. child.classList.add('pv-float-bar-button-' + buttonName);
  6317. });
  6318.  
  6319.  
  6320. this.floatBar=container;
  6321.  
  6322.  
  6323. var self=this;
  6324. container.addEventListener('click',function(e){
  6325. var buttonType;
  6326. var target=e.target;
  6327. for(var type in buttons){
  6328. if(!buttons.hasOwnProperty(type))return;
  6329. if(target==buttons[type]){
  6330. buttonType=type;
  6331. break;
  6332. };
  6333. };
  6334. if(!buttonType)return;
  6335.  
  6336. self.hide();
  6337. self.open(e,buttonType);
  6338.  
  6339. },true);
  6340.  
  6341.  
  6342. addCusMouseEvent('mouseleave',container,function(e){
  6343. clearTimeout(self.hideTimer);
  6344. self.hideTimer=setTimeout(function(){
  6345. self.hide();
  6346. },prefs.floatBar.hideDelay);
  6347. });
  6348.  
  6349. addCusMouseEvent('mouseenter',container,function(e){
  6350. clearTimeout(self.hideTimer);
  6351. });
  6352.  
  6353. this._scrollHandler=this.scrollHandler.bind(this);
  6354. },
  6355. addStyle:function(){
  6356. var style=document.createElement('style');
  6357. style.type='text/css';
  6358. style.textContent='\
  6359. #pv-float-bar-container {\
  6360. position: absolute;\
  6361. z-index:999999998;\
  6362. padding: 5px;\
  6363. margin: 0;\
  6364. border: none;\
  6365. opacity: 0.6;\
  6366. line-height: 0;\
  6367. -webkit-transition: opacity 0.2s ease-in-out;\
  6368. transition: opacity 0.2s ease-in-out;\
  6369. display:none;\
  6370. }\
  6371. #pv-float-bar-container:hover {\
  6372. opacity: 1;\
  6373. }\
  6374. .pv-float-bar-button {\
  6375. vertical-align:middle;\
  6376. cursor: pointer;\
  6377. width: 18px;\
  6378. height: 18px;\
  6379. padding: 0;\
  6380. margin:0;\
  6381. border: none;\
  6382. display: inline-block;\
  6383. position: relative;\
  6384. box-shadow: 1px 0 3px 0px rgba(0,0,0,0.9);\
  6385. background: transparent center no-repeat;\
  6386. background-size:100% 100%;\
  6387. background-origin: content-box;\
  6388. -webkit-transition: margin-right 0.15s ease-in-out , width 0.15s ease-in-out , height 0.15s ease-in-out ;\
  6389. transition: margin-right 0.15s ease-in-out , width 0.15s ease-in-out , height 0.15s ease-in-out ;\
  6390. }\
  6391. .pv-float-bar-button:not(:last-child){\
  6392. margin-right: -14px;\
  6393. }\
  6394. .pv-float-bar-button:first-child {\
  6395. z-index: 4;\
  6396. }\
  6397. .pv-float-bar-button:nth-child(2) {\
  6398. z-index: 3;\
  6399. }\
  6400. .pv-float-bar-button:nth-child(3) {\
  6401. z-index: 2;\
  6402. }\
  6403. .pv-float-bar-button:last-child {\
  6404. z-index: 1;\
  6405. }\
  6406. #pv-float-bar-container:hover > .pv-float-bar-button {\
  6407. width: 24px;\
  6408. height: 24px;\
  6409. }\
  6410. #pv-float-bar-container:hover > .pv-float-bar-button:not(:last-child) {\
  6411. margin-right: 4px;\
  6412. }\
  6413. .pv-float-bar-button-actual {\
  6414. background-image:url("'+ prefs.icons.actual +'");\
  6415. }\
  6416. .pv-float-bar-button-gallery {\
  6417. background-image:url("'+ prefs.icons.gallery +'");\
  6418. }\
  6419. .pv-float-bar-button-current {\
  6420. background-image:url("'+ prefs.icons.current +'");\
  6421. }\
  6422. .pv-float-bar-button-magnifier {\
  6423. background-image:url("'+ prefs.icons.magnifier +'");\
  6424. }\
  6425. ';
  6426. document.head.appendChild(style);
  6427. },
  6428. start:function(data){
  6429.  
  6430. //读取中的图片,不显示浮动栏,调整读取图标的位置.
  6431. if(LoadingAnimC.all._find(function(item,index,array){
  6432. if(data.img==item.data.img){
  6433. return true;
  6434. };
  6435. }))return;
  6436.  
  6437.  
  6438. //被放大镜盯上的图片,不要显示浮动栏.
  6439. if(MagnifierC.all._find(function(item,index,array){
  6440. if(data.img==item.data.img){
  6441. return true;
  6442. };
  6443. }))return;
  6444.  
  6445. this.data=data;
  6446. var self=this;
  6447. clearTimeout(this.hideTimer);
  6448.  
  6449. var imgOutHandler=function(e){
  6450. document.removeEventListener('mouseout',imgOutHandler,true);
  6451. clearTimeout(self.showTimer);
  6452. clearTimeout(self.hideTimer);
  6453. self.hideTimer=setTimeout(function(){
  6454. self.hide();
  6455. },prefs.floatBar.hideDelay);
  6456. };
  6457.  
  6458. clearTimeout(this.globarOutTimer);
  6459. this.globarOutTimer=setTimeout(function(){//稍微延时。错开由于css hover样式发生的out;
  6460. document.addEventListener('mouseout',imgOutHandler,true);
  6461. },150);
  6462.  
  6463. clearTimeout(this.showTimer);
  6464. this.showTimer=setTimeout(function(){
  6465. self.show();
  6466. },prefs.floatBar.showDelay);
  6467. },
  6468. setButton:function(){
  6469. if(this.data.type=='force'){
  6470. this.buttons['actual'].style.display='none';
  6471. this.buttons['magnifier'].style.display='none';
  6472. }else{
  6473. this.buttons['actual'].style.removeProperty('display');
  6474. this.buttons['magnifier'].style.removeProperty('display');
  6475. };
  6476. },
  6477. setPosition:function(){
  6478. //如果图片被删除了,或者隐藏了。
  6479. if(this.data.img.offsetWidth==0){
  6480. return true;
  6481. };
  6482. var targetPosi=getContentClientRect(this.data.img);
  6483. var windowSize=getWindowSize();
  6484.  
  6485. var floatBarPosi=prefs.floatBar.position.toLowerCase().split(/\s+/);
  6486.  
  6487. var offsetX=prefs.floatBar.offset.x;
  6488. var offsetY=prefs.floatBar.offset.y;
  6489.  
  6490.  
  6491. var scrolled=getScrolled();
  6492.  
  6493. var fbs=this.floatBar.style;
  6494. var setPosition={
  6495. top:function(){
  6496. var top=targetPosi.top + scrolled.y;
  6497. if(targetPosi.top + offsetY < 0){//满足图标被遮住的条件.
  6498. top=scrolled.y;
  6499. offsetY=0;
  6500. };
  6501. fbs.top=top + offsetY + 'px';
  6502. },
  6503. right:function(){
  6504. var right=windowSize.w - targetPosi.right;
  6505. if(right < offsetX){
  6506. right= -scrolled.x;
  6507. offsetX=0;
  6508. }else{
  6509. right -=scrolled.x;
  6510. };
  6511. fbs.right=right - offsetX + 'px';
  6512. },
  6513. bottom:function(){
  6514. var bottom=windowSize.h - targetPosi.bottom;
  6515. if(bottom <= offsetY){
  6516. bottom=-scrolled.y;
  6517. offsetY=0;
  6518. }else{
  6519. bottom -= scrolled.y;
  6520. };
  6521. fbs.bottom=bottom - offsetY + 'px';
  6522. },
  6523. left:function(){
  6524. var left=targetPosi.left + scrolled.x;
  6525. if(targetPosi.left + offsetX < 0){
  6526. left=scrolled.x;
  6527. offsetX=0;
  6528. };
  6529. fbs.left=left + offsetX + 'px';
  6530. },
  6531. };
  6532.  
  6533. setPosition[floatBarPosi[0]]();
  6534. setPosition[floatBarPosi[1]]();
  6535. },
  6536. show:function(){
  6537. if(this.setPosition())return;
  6538. this.shown=true;
  6539. this.setButton();
  6540. this.floatBar.style.display='block';
  6541. clearTimeout(this.hideTimer);
  6542. window.removeEventListener('scroll',this._scrollHandler,true);
  6543. window.addEventListener('scroll',this._scrollHandler,true);
  6544. },
  6545. hide:function(){
  6546. clearTimeout(this.showTimer);
  6547. this.shown=false;
  6548. this.floatBar.style.display='none';
  6549. window.removeEventListener('scroll',this._scrollHandler,true);
  6550. },
  6551. scrollHandler:function(){//更新坐标
  6552. clearTimeout(this.scrollUpdateTimer);
  6553. var self=this;
  6554. this.scrollUpdateTimer=setTimeout(function(){
  6555. self.setPosition();
  6556. },100);
  6557. },
  6558. open:function(e,buttonType){
  6559. var waitImgLoad = e && e.ctrlKey ? !prefs.waitImgLoad : prefs.waitImgLoad; //按住ctrl取反向值
  6560. var openInTopWindow = e && e.shiftKey ? !prefs.framesPicOpenInTopWindow : prefs.framesPicOpenInTopWindow; //按住shift取反向值
  6561.  
  6562. if (!waitImgLoad && buttonType == 'magnifier' && !envir.chrome) { //非chrome的background-image需要全部载入后才能显示出来
  6563. waitImgLoad = true;
  6564. };
  6565. new LoadingAnimC(this.data, buttonType, waitImgLoad, openInTopWindow);
  6566. },
  6567. };
  6568.  
  6569.  
  6570.  
  6571. var matchedRule,
  6572. URL=location.href,
  6573. floatBar;
  6574.  
  6575. function findPic(img){
  6576. //获取包裹img的第一个a元素。
  6577. var imgPN=img;
  6578. var imgPA;
  6579. while(imgPN=imgPN.parentElement){
  6580. if(imgPN.nodeName=='A'){
  6581. imgPA=imgPN;
  6582. break;
  6583. };
  6584. };
  6585.  
  6586. var iPASrc=imgPA? imgPA.href : '';
  6587. //base64字符串过长导致正则匹配卡死浏览器
  6588. var base64Img=/^data:[^;]+;base64,/i.test(img.src);
  6589.  
  6590.  
  6591. if(typeof matchedRule=='undefined'){//找到符合站点的高级规则,并缓存.
  6592. matchedRule=siteInfo._find(function(site,index,array){
  6593. // if(site.enabled && site.url && site.url.test(URL)){
  6594. if(site.url && site.url.test(URL)){
  6595. return true;
  6596. };
  6597. });
  6598. matchedRule=matchedRule? matchedRule[0] : false;
  6599. // console.log('匹配的规则:',matchedRule);
  6600. };
  6601.  
  6602. var src, type;
  6603.  
  6604. if(!src && matchedRule){//通过高级规则获取.
  6605. // 添加修正的样式
  6606. if (!matchedRule.cssAdded && matchedRule.css) {
  6607. var style = document.createElement('style');
  6608. style.type = 'text/css';
  6609. style.id = 'gm-picviewer-site-style';
  6610. style.textContent = matchedRule.css;
  6611. document.head.appendChild(style);
  6612.  
  6613. matchedRule.cssAdded = true;
  6614. }
  6615.  
  6616. // 排除
  6617. if (matchedRule.exclude && matchedRule.exclude.test(img.src)) {
  6618. return;
  6619. } else {
  6620. try{
  6621. src=matchedRule.getImage.call(img,img,imgPA);
  6622. }catch(err){
  6623. throwErrorInfo(err);
  6624. };
  6625.  
  6626. if(src)type='rule';
  6627. }
  6628. };
  6629.  
  6630. if(!src && !base64Img){//遍历通配规则
  6631. tprules._find(function(rule,index,array){
  6632. try{
  6633. src=rule.call(img,img,imgPA);
  6634. if(src){
  6635. //console.log('匹配的通配规则',rule);
  6636. return true;
  6637. };
  6638. }catch(err){
  6639. throwErrorInfo(err);
  6640. };
  6641. });
  6642. if(src)type='tpRule';
  6643. };
  6644.  
  6645. if(!src && imgPA){//链接可能是一张图片...
  6646. if(/\.(?:jpg|jpeg|png|gif|bmp)$/i.test(iPASrc)){
  6647. src=iPASrc;
  6648. };
  6649. if(src)type='scale';
  6650. };
  6651.  
  6652. if(!src){//本图片是否被缩放.
  6653. var imgAS={//实际尺寸。
  6654. h:img.naturalHeight,
  6655. w:img.naturalWidth,
  6656. };
  6657.  
  6658. var imgCStyle=getComputedStyle(img);
  6659. var imgCS={
  6660. h:parseFloat(imgCStyle.height),
  6661. w:parseFloat(imgCStyle.width),
  6662. };
  6663.  
  6664. if(!(imgAS.w==imgCS.w && imgAS.h==imgCS.h)){//如果不是两者完全相等,那么被缩放了.
  6665. if(imgAS.h > prefs.floatBar.minSizeLimit.h || imgAS.w > prefs.floatBar.minSizeLimit.w){//最小限定判断.
  6666. src=img.src;
  6667. type='scale';
  6668. };
  6669. }else{
  6670. if(prefs.floatBar.forceShow.enabled && (imgCS.w>=prefs.floatBar.forceShow.size.w && imgCS.h>=prefs.floatBar.forceShow.size.h)){
  6671. src=img.src;
  6672. type='force';
  6673. };
  6674. };
  6675. };
  6676.  
  6677.  
  6678. if(!src)return;
  6679.  
  6680. var ret={
  6681. src:src,//得到的src
  6682. type:type,//通过哪种方式得到的
  6683. imgSrc:img.src,//处理的图片的src
  6684. iPASrc:iPASrc,//图片的第一个父a元素的链接地址
  6685.  
  6686. img:img,//处理的图片
  6687. imgPA:imgPA,//图片的第一个父a元素
  6688. };
  6689.  
  6690. //console.log('图片查找结果:',ret);
  6691. return ret;
  6692. };
  6693.  
  6694.  
  6695. var isFrame=window!=window.parent;
  6696. var topWindowValid;//frameset的窗口这个标记为false
  6697. var frameSentData;
  6698. var frameSentSuccessData;
  6699. window.addEventListener('message',function(e){//contentscript里面的message监听,监听来自别的窗口的数据。
  6700. var data=e.data;
  6701. if( !data || !data.messageID || data.messageID != messageID )return;//通信ID认证
  6702. var source=e.source;
  6703. //chrome中所有window窗口的引用都是undefined
  6704. if(typeof source=='undefined' || source!==window){//来自别的窗口
  6705. if(!isFrame){//顶层窗口
  6706. //console.log('top-contentscript接收到:',e);
  6707.  
  6708. var command=data.command;
  6709. switch(command){
  6710. case 'open':{
  6711. var img=document.createElement('img');
  6712. img.src=data.src;
  6713.  
  6714. imgReady(img,{
  6715. ready:function(){
  6716. LoadingAnimC.prototype.open.call({
  6717. img:img,
  6718. data:data.data,
  6719. buttonType:data.buttonType,
  6720. from:data.from,//来自哪个窗口
  6721. });
  6722. },
  6723. });
  6724. }break;
  6725. case 'navigateToImg':{
  6726. var cusEvent=document.createEvent('CustomEvent');
  6727. cusEvent.initCustomEvent('pv-navigateToImg',false,false,data.exist);
  6728. document.dispatchEvent(cusEvent);
  6729. }break;
  6730. case 'topWindowValid':{
  6731. window.postMessage({
  6732. messageID:messageID,
  6733. command:'topWindowValid',
  6734. valid:document.body.nodeName!='FRAMESET',
  6735. to:data.from,
  6736. },'*');
  6737. }break;
  6738. };
  6739.  
  6740. }else{//frame窗口
  6741. //console.log('frame-contentscript接收到',e);
  6742. var command=data.command;
  6743. switch(command){
  6744. case 'navigateToImg':{
  6745.  
  6746. if(!frameSentData.unique){
  6747. var unique=GalleryC.prototype.unique(frameSentData);
  6748. frameSentData=unique.data;
  6749. frameSentData.unique=true;
  6750. };
  6751. var targetImg=frameSentData[data.index].img;
  6752. var exist=(document.documentElement.contains(targetImg) && getComputedStyle(targetImg).display!='none');
  6753.  
  6754. if(exist){
  6755. if(gallery && gallery.shown){//frame里面也打开了一个呢。
  6756. gallery.minimize();
  6757. };
  6758. setTimeout(function(){
  6759. GalleryC.prototype.navigateToImg(targetImg);
  6760. flashEle(targetImg);
  6761. },0);
  6762. };
  6763. window.postMessage({
  6764. messageID:messageID,
  6765. command:'navigateToImg',
  6766. exist:exist,
  6767. to:data.from,
  6768. },'*');
  6769. }break;
  6770. case 'sendFail':{
  6771. frameSentData=frameSentSuccessData;//frameSentData重置为发送成功的数据。
  6772. }break;
  6773. case 'topWindowValid':{
  6774. var cusEvent=document.createEvent('CustomEvent');
  6775. cusEvent.initCustomEvent('pv-topWindowValid',false,false,data.valid);
  6776. document.dispatchEvent(cusEvent);
  6777. }break;
  6778. };
  6779. };
  6780.  
  6781. };
  6782. },true);
  6783.  
  6784.  
  6785.  
  6786. //页面脚本用来转发消息
  6787. //原因chrome的contentscript无法访问非自己外的别的窗口。都会返回undefined,自然也无法向其他的窗口发送信息,这里用pagescript做个中间代理
  6788. //通讯逻辑..A页面的contentscript发送到A页面的pagescript,pagescript转交给B页面的contentscript
  6789.  
  6790. var messageID='pv-0.5106795670312598';
  6791.  
  6792. var pageScript=document.createElement('script');
  6793.  
  6794. var pageScriptText=function(messageID){
  6795. var frameID=Math.random();
  6796. var frames={
  6797. top:window.top,
  6798. };
  6799.  
  6800. window.addEventListener('message',function(e){
  6801. var data=e.data;
  6802. if( !data || !data.messageID || data.messageID != messageID )return;//通信ID认证
  6803. var source=e.source;
  6804. if(source===window){//来自contentscript,发送出去,或者干嘛。
  6805. if(data.to){
  6806. data.from=frameID;
  6807. frames[data.to].postMessage(data,'*');
  6808. }else{
  6809. switch(data.command){
  6810. case 'getIframeObject':{
  6811. var frameWindow=frames[data.windowId];
  6812. var iframes=document.getElementsByTagName('iframe');
  6813. var iframe;
  6814. var targetIframe;
  6815. for(var i=iframes.length-1 ; i>=0 ; i--){
  6816. iframe=iframes[i];
  6817. if(iframe.contentWindow===frameWindow){
  6818. targetIframe=iframe;
  6819. break;
  6820. };
  6821. };
  6822. var cusEvent=document.createEvent('CustomEvent');
  6823. cusEvent.initCustomEvent('pv-getIframeObject',false,false,targetIframe);
  6824. document.dispatchEvent(cusEvent);
  6825. }break;
  6826. };
  6827. };
  6828.  
  6829. }else{//来自别的窗口的,contentscript可以直接接收,这里保存下来自的窗口的引用
  6830. frames[data.from]=source;
  6831. };
  6832. },true)
  6833. };
  6834.  
  6835. pageScript.textContent='(' + pageScriptText.toString() + ')('+ JSON.stringify(messageID) +')';
  6836. document.head.appendChild(pageScript);
  6837.  
  6838.  
  6839. function clikToOpen(data){
  6840.  
  6841. var preventDefault = matchedRule.clikToOpen.preventDefault;
  6842.  
  6843. function mouseout(){
  6844. document.removeEventListener('mouseout',mouseout,true);
  6845. document.removeEventListener('click',click,true);
  6846. if(data.imgPA && preventDefault){
  6847. data.imgPA.removeEventListener('click',clickA,false);
  6848. };
  6849. };
  6850.  
  6851. function click(e){
  6852. if(e.button!=0)return;
  6853. FloatBarC.prototype.open.call({
  6854. data:data,
  6855. },
  6856. e,
  6857. matchedRule.clikToOpen.type);
  6858. };
  6859.  
  6860. function clickA(e){//阻止a的默认行为
  6861. e.preventDefault();
  6862. };
  6863.  
  6864. document.addEventListener('click',click,true);
  6865.  
  6866. if(data.imgPA && preventDefault){
  6867. data.imgPA.addEventListener('click',clickA,false);
  6868. };
  6869.  
  6870. setTimeout(function(){//稍微延时。错开由于css hover样式发生的out;
  6871. document.addEventListener('mouseout',mouseout,true);
  6872. },100);
  6873.  
  6874. return function(){
  6875. mouseout()
  6876. };
  6877. };
  6878.  
  6879. //监听 mouseover
  6880. var canclePreCTO;
  6881. function globalMouseoverHandler(e){
  6882.  
  6883. //console.log(e);
  6884. if(galleryMode)return;//库模式全屏中......
  6885.  
  6886. var target=e.target;
  6887.  
  6888. if(target.nodeName!='IMG' || target.classList.contains('pv-pic-ignored')){
  6889. return;
  6890. };
  6891. var result=findPic(target);
  6892.  
  6893. if(result){
  6894. if(!floatBar){
  6895. floatBar=new FloatBarC();
  6896. };
  6897. if(result.type=='rule' && matchedRule.clikToOpen && matchedRule.clikToOpen.enabled){
  6898. if(canclePreCTO){//取消上次的,防止一次点击打开多张图片
  6899. canclePreCTO();
  6900. };
  6901. canclePreCTO=clikToOpen(result);
  6902. };
  6903. floatBar.start(result);//出现悬浮工具栏
  6904. };
  6905. };
  6906.  
  6907. document.addEventListener('mouseover',globalMouseoverHandler,true);
  6908.  
  6909. // 注册按键
  6910. if (prefs.gallery.keys.keysEnabled) {
  6911. document.addEventListener('keydown', function(event) {
  6912. if (floatBar && floatBar.shown) {
  6913. var key = String.fromCharCode(event.keyCode).toLowerCase();
  6914.  
  6915. Object.keys(prefs.gallery.keys).some(function(action) {
  6916. if (key == prefs.gallery.keys[action]) {
  6917. floatBar.open(null, action);
  6918. event.stopPropagation();
  6919. event.preventDefault();
  6920. return true;
  6921. }
  6922. })
  6923. }
  6924. }, true);
  6925. }
  6926.  
  6927. };
  6928.  
  6929. function init2(){
  6930. init(topObject,window,document,arrayFn,envir,storage,unsafeWindow);
  6931. };
  6932.  
  6933.  
  6934. //大致检测运行环境
  6935. var envir={
  6936. ie:typeof document.documentMode == 'number',
  6937. firefox:typeof XPCNativeWrapper == 'function',
  6938. opera:!!window.opera,
  6939. chrome:!!window.chrome,
  6940. };
  6941.  
  6942. //ie的话,不支持 < ie9的版本
  6943. if(envir.ie && document.documentMode < 9){
  6944. return;
  6945. };
  6946.  
  6947.  
  6948. var arrayFn=(function(){
  6949. //Array的某些方法对所有的类数组都有效,比如HTMLCollection,NodeList,DOMStringList.....
  6950.  
  6951. //添加一个当函数返回true时,返回[array[index],index],并且跳出循环的方法
  6952. //类似做到 for 循环,在满足条件的时候直接break跳出的效果。
  6953. if(typeof Array.prototype['_find']!='function'){
  6954. Object.defineProperty(Array.prototype,'_find',{
  6955. value:function(callback , thisArg){
  6956. if (this == null){
  6957. throw new TypeError( "this is null or not defined" );
  6958. };
  6959.  
  6960. if(typeof callback != 'function') {
  6961. throw new TypeError( callback + " is not a function" );
  6962. };
  6963.  
  6964. var i = 0,
  6965. l = this.length,
  6966. value,
  6967. hasOwnProperty=Object.prototype.hasOwnProperty
  6968. ;
  6969.  
  6970.  
  6971. while(i<l){
  6972. if(hasOwnProperty.call(this,i)){
  6973. value = this[i];
  6974. if(callback.call( thisArg, value, i, this )===true){
  6975. return [value,i,this];
  6976. };
  6977. };
  6978. i++;
  6979. };
  6980. },
  6981. writable:true,
  6982. enumerable:false,//与原生方法一样不可枚举,维护网页和谐。。。
  6983. configurable:true,
  6984. });
  6985. };
  6986.  
  6987. var arrayProto=Array.prototype;
  6988. return {
  6989. _find:arrayProto._find,
  6990. slice:arrayProto.slice,
  6991. forEach:arrayProto.forEach,
  6992. some:arrayProto.some,
  6993. every:arrayProto.every,
  6994. map:arrayProto.map,
  6995. filter:arrayProto.filter,
  6996. indexOf:arrayProto.indexOf,
  6997. lastIndexOf:arrayProto.lastIndexOf,
  6998. };
  6999.  
  7000. })();
  7001.  
  7002.  
  7003. var storage={
  7004. supportGM: typeof GM_getValue=='function' && typeof GM_getValue('a','b')!='undefined',//chrome的gm函数式空函数
  7005. mxAppStorage:(function(){//傲游扩展储存接口
  7006. try{
  7007. return window.external.mxGetRuntime().storage;
  7008. }catch(e){
  7009. };
  7010. })(),
  7011. operaUJSStorage:(function(){//opera userjs全局存储接口
  7012. try{
  7013. return window.opera.scriptStorage;
  7014. }catch(e){
  7015. };
  7016. })(),
  7017. setItem:function(key,value){
  7018. if(this.operaUJSStorage){
  7019. this.operaUJSStorage.setItem(key,value);
  7020. }else if(this.mxAppStorage){
  7021. this.mxAppStorage.setConfig(key,value);
  7022. }else if(this.supportGM){
  7023. GM_setValue(key,value);
  7024. }else if(window.localStorage){
  7025. window.localStorage.setItem(key,value);
  7026. };
  7027. },
  7028. getItem:function(key){
  7029. var value;
  7030. if(this.operaUJSStorage){
  7031. value=this.operaUJSStorage.getItem(key);
  7032. }else if(this.mxAppStorage){
  7033. value=this.mxAppStorage.getConfig(key);
  7034. }else if(this.supportGM){
  7035. value=GM_getValue(key);
  7036. }else if(window.localStorage){
  7037. value=window.localStorage.getItem(key);
  7038. };
  7039. return value;
  7040. },
  7041. };
  7042.  
  7043. init2();
  7044.  
  7045. })(this,window,document,(typeof unsafeWindow=='undefined'? window : unsafeWindow));