Picviewer CE+

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

当前为 2022-01-07 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name Picviewer CE+
  3. // @name:zh-CN Picviewer CE+
  4. // @name:zh-TW Picviewer CE+
  5. // @author NLF && ywzhaiqi && hoothin
  6. // @description Powerful picture viewing tool online, which can popup/scale/rotate/batch save pictures automatically
  7. // @description:zh-CN 在线看图工具,支持图片翻转、旋转、缩放、弹出大图、批量保存
  8. // @description:zh-TW 線上看圖工具,支援圖片翻轉、旋轉、縮放、彈出大圖、批量儲存
  9. // @version 2022.1.7.1
  10. // @created 2011-6-15
  11. // @namespace https://github.com/hoothin/UserScripts
  12. // @homepage http://hoothin.com
  13. // @connect www.google.com
  14. // @connect www.google.com.hk
  15. // @connect www.google.co.jp
  16. // @connect ipv4.google.com
  17. // @connect image.baidu.com
  18. // @connect www.tineye.com
  19. // @grant GM_getValue
  20. // @grant GM_setValue
  21. // @grant GM_addStyle
  22. // @grant GM_openInTab
  23. // @grant GM_setClipboard
  24. // @grant GM_xmlhttpRequest
  25. // @grant GM_registerMenuCommand
  26. // @grant GM_notification
  27. // @grant GM_download
  28. // @grant GM.getValue
  29. // @grant GM.setValue
  30. // @grant GM.addStyle
  31. // @grant GM.openInTab
  32. // @grant GM.setClipboard
  33. // @grant GM.xmlhttpRequest
  34. // @grant GM.registerMenuCommand
  35. // @grant GM.notification
  36. // @grant unsafeWindow
  37. // @require https://greasyfork.org/scripts/6158-gm-config-cn/code/GM_config%20CN.js?version=23710
  38. // @contributionURL https://www.paypal.com/cgi-bin/webscr?cmd=_donations&business=rixixi@sina.com&item_name=Greasy+Fork+donation
  39. // @contributionAmount 1
  40. // @require https://cdnjs.cloudflare.com/ajax/libs/FileSaver.js/2.0.2/FileSaver.min.js
  41. // @require https://cdnjs.cloudflare.com/ajax/libs/jszip/3.7.1/jszip.js
  42. // @require https://greasyfork.org/scripts/438080-pvcep-rules/code/pvcep_rules.js?version=1005390
  43. // @include http://*
  44. // @include https://*
  45. // @include ftp://*
  46. // @exclude http://www.toodledo.com/tasks/*
  47. // @exclude http*://maps.google.com*/*
  48. // @exclude *://www.google.*/_/chrome/newtab*
  49. // @exclude *://mega.*/*
  50. // @exclude *://*.mega.*/*
  51. // ==/UserScript==
  52.  
  53. ;(function(topObject,window,document,unsafeWindow){
  54. 'use strict';
  55.  
  56. var lang = navigator.appName=="Netscape"?navigator.language:navigator.userLanguage,debug;
  57. var i18nData={};
  58. switch (lang){
  59. case "zh-CN":
  60. case "zh-SG":
  61. i18nData={
  62. saveBtn:"确定",
  63. download:"下载",
  64. saveBtnTips:"部分选项需要刷新页面才能生效",
  65. closeBtn:"取消",
  66. closeBtnTips:"取消本次设置,所有选项还原",
  67. resetLink:"恢复默认设置",
  68. resetLinkTips:"恢复所有设置的内容为默认值",
  69. share:"分享",
  70. suitLongImg:"长图在滚动窗口显示",
  71. globalkeys:"预览功能键组合: ",
  72. globalkeysPress:"敲击开关预览",
  73. globalkeysHold:"按住开启预览",
  74. globalkeysType:"预览触发方式:",
  75. loadAll:"加载更多",
  76. loadedAll:"加载完毕",
  77. loading:"正在加载",
  78. loadAllTip:"加载下一页的图片",
  79. fiddle:"折腾",
  80. fiddleTip:"弹出图片进行复杂操作",
  81. collect:"收藏",
  82. collected:"已收藏",
  83. exitCollection:"退出收藏",
  84. exitCollectionTip:"点击退出收藏模式",
  85. noCollectionYet:"你还木有收藏任何图片",
  86. collectDetail:"描述",
  87. collectDetailTip:"给收藏的图片添加一些描述吧",
  88. playSlide:"播放幻灯片",
  89. slideGap:"间隔(s)",
  90. slideGapTip:"间隔时间,单位(秒)",
  91. slideBack:"后退",
  92. slideBackTip:"从后往前播放",
  93. slideWait:"等待图片读取",
  94. slideWaitTip:"从每张图片完全读取完成后才开始倒计时",
  95. slideSkipError:"跳过错误图片",
  96. slideSkipErrorTip:"快速跳过读取错误的图片",
  97. type:"类别",
  98. typeTip:"选择图片类别",
  99. advancedRules:"高级规则",
  100. advancedRulesTip:"由高级规则匹配出来的",
  101. tpRules:"通配规则",
  102. tpRulesTip:"由通配规则匹配出来的",
  103. scaleRules:"缩放过的",
  104. scaleRulesTip:"js自动查找,相对页面显示的图片有缩放过的",
  105. noScaleRules:"无缩放过",
  106. noScaleRulesTip:"js自动查找,无缩放过的,但是满足一定的大小",
  107. smallRules:"小尺寸的",
  108. smallRulesTip:"小尺寸图片,实际尺寸的高和宽都小于#t#像素",
  109. command:"命令",
  110. commandTip:"命令菜单",
  111. onlineEdit:"在线编辑",
  112. onlineEditTip:"使用#t#在线编辑该图片",
  113. openInNewWindow:"新窗口打开",
  114. openInNewWindowTip:"新窗口打开图片",
  115. findInPage:"定位到图片",
  116. findInPageTip:"滚动到当前图片所在的位置",
  117. viewCollection:"查看收藏",
  118. viewCollectionTip:"查看所有收藏的图片",
  119. inCollection:"收藏模式中,无法使用",
  120. cantFind:"图片不在文档中,或者被隐藏了,无法定位!",
  121. exportImages:"导出显示大图",
  122. exportImagesTip:"导出所有显示中的图片到新窗口",
  123. downloadImage:"下载当前所有",
  124. downloadImageTip:"下载当前库中所有显示图片,注意无法下载跨域图片",
  125. copyImagesUrl:"复制显示大图",
  126. copyImagesUrlTip:"复制所有显示中的大图地址",
  127. copySuccess:"已成功复制 #t# 张大图地址",
  128. autoRefresh:"自动重载",
  129. autoRefreshTip:"最后几张图片时,滚动主窗口到最底部,然后自动加载新的图片",
  130. enterFullsc:"进入全屏",
  131. exitFullsc:"退出全屏",
  132. config:"设置",
  133. openConfig:"打开设置",
  134. closeGallery:"关闭库",
  135. returnToGallery:"回到库",
  136. picInfo:"点击修改",
  137. picNote:"图片注释",
  138. resolution:"分辨率",
  139. scaleRatio:"缩放比",
  140. similarImage:"以图搜图",
  141. scale:"缩放",
  142. horizontalFlip:"水平翻转",
  143. verticalFlip:"垂直翻转",
  144. actualBtn:'查看原始(A)',
  145. searchBtn:'查找原图(S)',
  146. galleryBtn:'查看库(G)',
  147. currentBtn:'查看当前(C)',
  148. magnifierBtn:'放大镜(M)',
  149. picTitle:"图片标题",
  150. picNum:"图片数量",
  151. picTips:"按住Ctrl放大",
  152. savePageTips:"使用“网页另存为”即可保存所有图片",
  153. exportImagesUrl:"导出图片链接",
  154. exportImagesUrlPop:"Ctrl+C复制图片链接",
  155. beginSearchImg:"#t#识图开始……",
  156. findNoPic:"未找到原图",
  157. findOverBeginLoad:"#t#识图结束,共找到#t#张匹配图片,开始加载第一张",
  158. loadNextSimilar:"原图加载失败,尝试加载下一结果……",
  159. loadError:"加载失败",
  160. openHomePage:"点击此处打开主页",
  161. position:"显示位置",
  162. positionTips:"按住ALT隐藏",
  163. topLeft: '图片左上角',
  164. topRight: '图片右上角',
  165. bottomRight: '图片右下角',
  166. bottomLeft: '图片左下角',
  167. topCenter: '图片正上方',
  168. bottomCenter: '图片正下方',
  169. floatBar:"浮动工具栏",
  170. showDelay:"显示延时",
  171. ms:"毫秒",
  172. hide:"隐藏",
  173. hideDelay:"隐藏延时",
  174. forceShow:"非缩放图片,超过该尺寸,显示浮框",
  175. forceShowTip:"非缩放的图片大小超过下面设定的尺寸时显示浮动工具栏",
  176. sizeLimitOr:"以上长宽条件只需满足其一",
  177. px:"像素",
  178. minSizeLimit:"缩放图片,超过该尺寸,显示浮框",
  179. minSizeLimitTip:"图片被缩放(图片原始大小与实际大小不一致)后,显示长宽大于设定值时显示浮动工具栏",
  180. defaultSizeLimit:"图库默认筛选尺寸",
  181. listenBg:"监听背景图",
  182. listenBgTip:"在有背景图的元素上显示悬浮框",
  183. butonOrder:"工具栏图标排序",
  184. keysEnable:"启用以下快捷键",
  185. keysActual:"打开大图",
  186. keysActualTip:"当出现悬浮条时按下此按键打开大图",
  187. keysSearch:"查找原图",
  188. keysSearchTip:"当出现悬浮条时按下此按键查找原图",
  189. headSearchTip:"搜图",
  190. headSearchAll:"搜索以上全部",
  191. keysCurrent:"打开当前图片",
  192. keysCurrentTip:"当出现悬浮条时按下此按键打开当前显示的图片",
  193. keysMagnifier:"打开放大镜观察",
  194. keysMagnifierTip:"当出现悬浮条时按下此按键打开放大镜观察",
  195. keysGallery:"打开图库(加功能键为全局)",
  196. keysGalleryTip:"当出现悬浮条时按下此按键打开图库",
  197. openGallery:"打开图库",
  198. magnifier:"放大镜",
  199. magnifierRadius:"默认半径",
  200. magnifierWheelZoomEnabled:"启用滚轮缩放",
  201. magnifierWheelZoomRange:"滚轮缩放的倍率",
  202. gallery:"图库",
  203. galleryFitToScreen:"对图片进行缩放以适应屏幕",
  204. galleryFitToScreenSmall:"小图也缩放以适应屏幕",
  205. galleryFitToScreenTip:"适应方式为contain,非cover",
  206. galleryScrollEndToChange:"大图滚动到底后切换图片",
  207. galleryScrollEndToChangeTip:"取消上一选项后才有效",
  208. galleryExportType:"图片导出默认排序",
  209. grid:'平铺排序',
  210. gridBig:'原图平铺',
  211. list:'列表排序',
  212. galleryAutoLoad:"自动加载更多图片",
  213. galleryLoadAll:"加载更多图片时自动处理全部页",
  214. galleryLoadAllTip:"若页数过多可能影响体验",
  215. galleryDownloadWithZip:"下载所有时打包成zip",
  216. galleryDownloadWithZipAlert:"正在打包,请耐心等候数十秒",
  217. galleryScaleSmallSize1:"实际尺寸的高和宽都小于 ",
  218. galleryScaleSmallSize2:" 像素则归入小尺寸图片",
  219. galleryShowSmallSize:"默认显示小尺寸图片",
  220. galleryTransition:"显示图库切换图片的特效",
  221. gallerySidebarPosition:"缩略图栏位置",
  222. bottom:'底部',
  223. right:'右侧',
  224. left:'左侧',
  225. top:'顶部',
  226. gallerySidebarSize:"高度",
  227. gallerySidebarSizeTip:"缩略图栏的高(如果是水平放置)或者宽(如果是垂直放置)",
  228. galleryMax1:"最多预读 ",
  229. galleryMax2:" 张图片(前后各多少张)",
  230. galleryAutoZoom:"缩放改回 100%(chrome)",
  231. galleryAutoZoomTip:"如果有放大,则把图片及 sidebar 部分的缩放改回 100%,增大可视面积(仅在 chrome 下有效)",
  232. galleryDescriptionLength1:"注释的最大宽度",
  233. galleryDescriptionLength2:" 个字符",
  234. galleryAutoOpenSites:"自动打开图库的网址正则,一行一条,若开头加@则自动展开图库",
  235. gallerySearchData:"搜图站点设置,清空还原",
  236. galleryEditSite:"在线编辑站点",
  237. imgWindow:"图片窗口",
  238. imgWindowFitToScreen:"适应屏幕,并且水平垂直居中",
  239. imgWindowFitToScreenTip:"适应方式为contain,非cover",
  240. imgWindowDefaultTool:"打开窗口时默认选择的工具",
  241. hand:'抓手',
  242. rotate:'旋转',
  243. zoom:'放大镜',
  244. imgWindowEscKey:"Esc键关闭",
  245. imgWindowDblClickImgWindow:"双击图片窗口关闭",
  246. imgWindowClickOutside:"点击图片外部(覆盖层)关闭",
  247. imgWindowClickOutsideTip:"仅当覆盖层显示时生效",
  248. none:'无',
  249. click:'单击',
  250. dblclick:'双击',
  251. imgWindowOverlayerShown:"覆盖层",
  252. imgWindowOverlayerColor:"颜色和不透明度",
  253. imgWindowShiftRotateStep1:"旋转时,按住shift键,旋转的步进",
  254. imgWindowShiftRotateStep2:" 度",
  255. imgWindowMouseWheelZoom:"滚轮缩放",
  256. imgWindowZoomRange:"滚轮缩放比例",
  257. imgWindowZoomRangeTip:"缩放比例(必须为正数)",
  258. others:"其它",
  259. waitImgLoad:"等图片完全载入后,才开始执行弹出放大等操作",
  260. waitImgLoadTip:"按住ctrl键的时候,可以临时执行和这个设定相反的设定",
  261. debug:"调试模式",
  262. firstEngine:"首选搜图引擎",
  263. refreshWhenError:"读取错误,点击重载",
  264. switchSlide:"开关侧边栏",
  265. viewmore:"展开更多",
  266. countDown:"倒计时"
  267. };
  268. break;
  269. case "zh-TW":
  270. case "zh-HK":
  271. i18nData={
  272. saveBtn:"確定",
  273. download:"下載",
  274. saveBtnTips:"部分選項需要刷新頁面才能生效",
  275. closeBtn:"取消",
  276. closeBtnTips:"取消本次設置,所有選項還原",
  277. resetLink:"恢復默認設置",
  278. resetLinkTips:"恢復所有設置的內容為默認值",
  279. share:"分享",
  280. suitLongImg:"長圖在滾動窗口顯示",
  281. globalkeys:"預覽功能鍵組合: ",
  282. globalkeysPress:"敲擊開關預覽",
  283. globalkeysHold:"按住開啟預覽",
  284. globalkeysType:"預覽觸發方式:",
  285. loadAll:"載入更多",
  286. loadedAll:"載入完畢",
  287. loading:"正在載入",
  288. loadAllTip:"載入下一頁的圖片",
  289. fiddle:"折騰",
  290. fiddleTip:"彈出圖片進行複雜操作",
  291. collect:"收藏",
  292. collected:"已收藏",
  293. exitCollection:"退出收藏",
  294. exitCollectionTip:"點擊退出收藏模式",
  295. noCollectionYet:"你還木有收藏任何圖片",
  296. collectDetail:"描述",
  297. collectDetailTip:"給收藏的圖片添加一些描述吧",
  298. playSlide:"播放幻燈片",
  299. slideGap:"間隔(s)",
  300. slideGapTip:"間隔時間,單位(秒)",
  301. slideBack:"後退",
  302. slideBackTip:"從後往前播放",
  303. slideWait:"等待圖片讀取",
  304. slideWaitTip:"從每張圖片完全讀取完成後才開始倒計時",
  305. slideSkipError:"跳過錯誤圖片",
  306. slideSkipErrorTip:"快速跳過讀取錯誤的圖片",
  307. type:"類別",
  308. typeTip:"選擇圖片類別",
  309. advancedRules:"高級規則",
  310. advancedRulesTip:"由高級規則匹配出來的",
  311. tpRules:"通配規則",
  312. tpRulesTip:"由通配規則匹配出來的",
  313. scaleRules:"縮放過的",
  314. scaleRulesTip:"js自動查找,相對頁面顯示的圖片有縮放過的",
  315. noScaleRules:"無縮放過",
  316. noScaleRulesTip:"js自動查找,無縮放過的,但是滿足一定的大小",
  317. smallRules:"小尺寸的",
  318. smallRulesTip:"小尺寸圖片,實際尺寸的高和寬都小於#t#像素",
  319. command:"命令",
  320. commandTip:"命令菜單",
  321. onlineEdit:"在線編輯",
  322. onlineEditTip:"使用#t#在線編輯該圖片",
  323. openInNewWindow:"新窗口打開",
  324. openInNewWindowTip:"新窗口打開圖片",
  325. findInPage:"定位到圖片",
  326. findInPageTip:"滾動到當前圖片所在的位置",
  327. viewCollection:"查看收藏",
  328. viewCollectionTip:"查看所有收藏的圖片",
  329. inCollection:"收藏模式中,無法使用",
  330. cantFind:"圖片不在文檔中,或者被隱藏了,無法定位!",
  331. exportImages:"導出顯示大圖",
  332. exportImagesTip:"導出所有顯示中的圖片到新窗口",
  333. downloadImage:"下載當前所有",
  334. downloadImageTip:"下載當前庫中所有顯示圖片,注意無法下載跨域圖片",
  335. copyImagesUrl:"複製顯示大圖",
  336. copyImagesUrlTip:"複製所有顯示中的大圖地址",
  337. copySuccess:"已成功複製 #t# 張大圖地址",
  338. autoRefresh:"自動重載",
  339. autoRefreshTip:"最後幾張圖片時,滾動主窗口到最底部,然後自動載入新的圖片",
  340. enterFullsc:"進入全屏",
  341. exitFullsc:"退出全屏",
  342. config:"設置",
  343. openConfig:"打開設置",
  344. closeGallery:"關閉庫",
  345. returnToGallery:"回到庫",
  346. picInfo:"點擊修改",
  347. picNote:"圖片注釋",
  348. resolution:"解析度",
  349. scaleRatio:"縮放比",
  350. similarImage:"以圖搜圖",
  351. scale:"縮放",
  352. horizontalFlip:"水平翻轉",
  353. verticalFlip:"垂直翻轉",
  354. actualBtn:'查看原始(A)',
  355. searchBtn:'查找原圖(S)',
  356. galleryBtn:'查看庫(G)',
  357. currentBtn:'查看當前(C)',
  358. magnifierBtn:'放大鏡(M)',
  359. picTitle:"圖片標題",
  360. picNum:"圖片數量",
  361. picTips:"按住Ctrl放大",
  362. savePageTips:"使用「網頁另存為」即可保存所有圖片",
  363. exportImagesUrl:"導出圖片鏈接",
  364. exportImagesUrlPop:"Ctrl+C複製圖片鏈接",
  365. beginSearchImg:"#t#識圖開始……",
  366. findNoPic:"未找到原圖",
  367. findOverBeginLoad:"#t#識圖結束,共找到#t#張匹配圖片,開始載入第一張",
  368. loadNextSimilar:"原圖載入失敗,嘗試載入下一結果……",
  369. loadError:"載入失敗",
  370. openHomePage:"點擊此處打開主頁",
  371. position:"顯示位置",
  372. positionTips:"按住ALT隱藏",
  373. topLeft: '圖片左上角',
  374. topRight: '圖片右上角',
  375. bottomRight: '圖片右下角',
  376. bottomLeft: '圖片左下角',
  377. topCenter: '圖片正上方',
  378. bottomCenter: '圖片正下方',
  379. floatBar:"浮動工具欄",
  380. showDelay:"顯示延時",
  381. ms:"毫秒",
  382. hide:"隱藏",
  383. hideDelay:"隱藏延時",
  384. forceShow:"非縮放圖片,超過該尺寸,顯示浮框",
  385. forceShowTip:"非縮放的圖片大小超過下面設定的尺寸時顯示浮動工具欄",
  386. sizeLimitOr:"以上長寬條件只需滿足其一",
  387. px:"像素",
  388. minSizeLimit:"縮放圖片,超過該尺寸,顯示浮框",
  389. minSizeLimitTip:"圖片被縮放(圖片原始大小與實際大小不一致)後,顯示長寬大於設定值時顯示浮動工具欄",
  390. defaultSizeLimit:"圖庫默認篩選尺寸",
  391. listenBg:"監聽背景圖",
  392. listenBgTip:"在有背景圖的元素上顯示懸浮框",
  393. butonOrder:"工具欄圖標排序",
  394. keysEnable:"啟用以下快捷鍵",
  395. keysActual:"打開大圖",
  396. keysActualTip:"當出現懸浮條時按下此按鍵打開大圖",
  397. keysSearch:"查找原圖",
  398. keysSearchTip:"當出現懸浮條時按下此按鍵查找原圖",
  399. headSearchTip:"搜圖",
  400. headSearchAll:"搜索以上全部",
  401. keysCurrent:"打開當前圖片",
  402. keysCurrentTip:"當出現懸浮條時按下此按鍵打開當前顯示的圖片",
  403. keysMagnifier:"打開放大鏡觀察",
  404. keysMagnifierTip:"當出現懸浮條時按下此按鍵打開放大鏡觀察",
  405. keysGallery:"打開圖庫(加功能鍵為全局)",
  406. keysGalleryTip:"當出現懸浮條時按下此按鍵打開圖庫",
  407. openGallery:"打開圖庫",
  408. magnifier:"放大鏡",
  409. magnifierRadius:"默認半徑",
  410. magnifierWheelZoomEnabled:"啟用滾輪縮放",
  411. magnifierWheelZoomRange:"滾輪縮放的倍率",
  412. gallery:"圖庫",
  413. galleryFitToScreen:"對圖片進行縮放以適應屏幕",
  414. galleryFitToScreenSmall:"小圖也縮放以適應屏幕",
  415. galleryFitToScreenTip:"適應方式為contain,非cover",
  416. galleryScrollEndToChange:"大圖滾動到底後切換圖片",
  417. galleryScrollEndToChangeTip:"取消上一選項後才有效",
  418. galleryExportType:"圖片導出默認排序",
  419. grid:'平鋪排序',
  420. gridBig:'原圖平鋪',
  421. list:'列表排序',
  422. galleryAutoLoad:"自動載入更多圖片",
  423. galleryLoadAll:"載入更多圖片時自動處理全部頁",
  424. galleryLoadAllTip:"若頁數過多可能影響體驗",
  425. galleryDownloadWithZip:"下載所有時打包成zip",
  426. galleryDownloadWithZipAlert:"正在打包,請耐心等候數十秒",
  427. galleryScaleSmallSize1:"實際尺寸的高和寬都小於 ",
  428. galleryScaleSmallSize2:" 像素則歸入小尺寸圖片",
  429. galleryShowSmallSize:"默認顯示小尺寸圖片",
  430. galleryTransition:"顯示圖庫切換圖片的特效",
  431. gallerySidebarPosition:"縮略圖欄位置",
  432. bottom:'底部',
  433. right:'右側',
  434. left:'左側',
  435. top:'頂部',
  436. gallerySidebarSize:"高度",
  437. gallerySidebarSizeTip:"縮略圖欄的高(如果是水平放置)或者寬(如果是垂直放置)",
  438. galleryMax1:"最多預讀 ",
  439. galleryMax2:" 張圖片(前後各多少張)",
  440. galleryAutoZoom:"縮放改回 100%(chrome)",
  441. galleryAutoZoomTip:"如果有放大,則把圖片及 sidebar 部分的縮放改回 100%,增大可視面積(僅在 chrome 下有效)",
  442. galleryDescriptionLength1:"注釋的最大寬度",
  443. galleryDescriptionLength2:" 個字元",
  444. galleryAutoOpenSites:"自動打開圖庫的網址正則,一行一條,若前綴@則自動展開圖庫",
  445. gallerySearchData:"搜圖站點設置,清空還原",
  446. galleryEditSite:"在線編輯站點",
  447. imgWindow:"圖片窗口",
  448. imgWindowFitToScreen:"適應屏幕,並且水平垂直居中",
  449. imgWindowFitToScreenTip:"適應方式為contain,非cover",
  450. imgWindowDefaultTool:"打開窗口時默認選擇的工具",
  451. hand:'抓手',
  452. rotate:'旋轉',
  453. zoom:'放大鏡',
  454. imgWindowEscKey:"Esc鍵關閉",
  455. imgWindowDblClickImgWindow:"雙擊圖片窗口關閉",
  456. imgWindowClickOutside:"點擊圖片外部關閉(覆蓋層)",
  457. imgWindowClickOutsideTip:"僅當覆蓋層顯示時生效",
  458. none:'無',
  459. click:'單擊',
  460. dblclick:'雙擊',
  461. imgWindowOverlayerShown:"覆蓋層",
  462. imgWindowOverlayerColor:"顏色和不透明度",
  463. imgWindowShiftRotateStep1:"旋轉時,按住shift鍵,旋轉的步進",
  464. imgWindowShiftRotateStep2:" 度",
  465. imgWindowMouseWheelZoom:"滾輪縮放",
  466. imgWindowZoomRange:"滾輪縮放比例",
  467. imgWindowZoomRangeTip:"縮放比例(必須為正數)",
  468. others:"其它",
  469. waitImgLoad:"等圖片完全載入後,才開始執行彈出放大等操作",
  470. waitImgLoadTip:"按住ctrl鍵的時候,可以臨時執行和這個設定相反的設定",
  471. debug:"調試模式",
  472. firstEngine:"首選搜圖引擎",
  473. refreshWhenError:"讀取錯誤,點擊重載",
  474. switchSlide:"開關側邊欄",
  475. viewmore:"展開更多",
  476. countDown:"倒計時"
  477. };
  478. break;
  479. default:
  480. i18nData={
  481. saveBtn: "OK",
  482. download:"Download",
  483. saveBtnTips: "some options need to refresh the page to take effect",
  484. closeBtn: "Cancel",
  485. closeBtnTips: "cancel this setting and restore all options",
  486. resetLink: "Restore default settings",
  487. resetLinkTips: "restore all settings to default values",
  488. share:"Share",
  489. suitLongImg:"Suit long pics in scroll window",
  490. globalkeys:"Global keys for preview: ",
  491. globalkeysPress:"Press to toggle preview",
  492. globalkeysHold:"Hold to enable preview",
  493. globalkeysType:"Method to enable preview",
  494. loadAll:"Load more pages",
  495. loadedAll:"Load completed",
  496. loading:"Loading ...",
  497. loadAllTip:"Load the picture on the next page",
  498. fiddle:"Operate",
  499. fiddleTip:"Pop-up pictures for complex operations",
  500. collect:"Collection",
  501. collected:"Has been collected",
  502. exitCollection:"Exit collection",
  503. exitCollectionTip:"Click to exit the collection mode",
  504. noCollectionYet:"You have no picture in the collection",
  505. collectDetail:"Description",
  506. collectDetailTip:"Add some descriptions to your favorite pictures",
  507. playSlide:"Play slideshow",
  508. slideGap:"Interval (s)",
  509. slideGapTip:"Interval, unit (seconds)",
  510. slideBack:"Back",
  511. slideBackTip:"Play from back to front",
  512. slideWait:"Wait for image reading",
  513. slideWaitTip:"The countdown starts after each image is completely read.",
  514. slideSkipError:"Skip error pictures",
  515. slideSkipErrorTip:"Quickly skip to read the error pictures",
  516. type:"Category",
  517. typeTip:"Select image category",
  518. advancedRules:"Advanced Rules",
  519. advancedRulesTip:"Matched by advanced rules",
  520. tpRules:"Wildcard rules",
  521. tpRulesTip:"Matched by wildcard rules",
  522. scaleRules:"Zoomed",
  523. scaleRulesTip:"The image which is finded automatically but scaled",
  524. noScaleRules:"No scaling",
  525. noScaleRulesTip:"The image which is finded automatically but without scale",
  526. smallRules:"Small size",
  527. smallRulesTip:"Small size image, the actual size of the height and width are less than #t# pixels",
  528. command:"Command",
  529. commandTip:"Command Menu",
  530. onlineEdit:"Online editing",
  531. onlineEditTip:"Edit this image online using #t#",
  532. openInNewWindow:"Open in new window",
  533. openInNewWindowTip:"Open image in new window",
  534. findInPage:"Find In Page",
  535. findInPageTip:"Scroll to the current image in page",
  536. viewCollection:"View Collection",
  537. viewCollectionTip:"View all collected images",
  538. inCollection:"Unable to use in Collection mode",
  539. cantFind:"The image is not in the document, or it is hidden and cannot be located!",
  540. exportImages:"Export all Images",
  541. exportImagesTip:"Export all shown big size images to new window",
  542. downloadImage:"Download all shown",
  543. downloadImageTip:"Download the current shown pictures, support no cross-origin",
  544. copyImagesUrl:"Copy all images Urls",
  545. copyImagesUrlTip:"Copy all shown big size image Urls",
  546. copySuccess:"Copied #t# Urls successfully",
  547. autoRefresh:"Auto load page",
  548. autoRefreshTip:"When the last few images are viewed, scroll the window to the bottom, so that some webpage will load the new images",
  549. enterFullsc:"Enter full screen",
  550. exitFullsc:"Exit full screen",
  551. config:"Settings",
  552. openConfig:"Open Settings",
  553. closeGallery:"Close Gallery",
  554. returnToGallery:"Back to the Gallery",
  555. picInfo:"Click to change",
  556. picNote:"Img annotation",
  557. resolution:"Img Resolution",
  558. picNum:"Number of pictures",
  559. picTips:"View pictures with CTRL key",
  560. savePageTips:"Save this page to download all pics",
  561. scaleRatio:"Scaling ratio",
  562. similarImage:"Searching by image",
  563. scale:"Zoom",
  564. horizontalFlip:"Horizontal flip",
  565. verticalFlip:"Vertical flip",
  566. actualBtn:"View original (A)",
  567. searchBtn:"Find the original image (S)",
  568. galleryBtn:"View gallery (G)",
  569. currentBtn:"View current (C)",
  570. magnifierBtn:"Magnifier / ZooM (M)",
  571. picTitle:"Picture Title",
  572. exportImagesUrl:"Export image Url",
  573. exportImagesUrlPop:"Ctrl+C to copy image Url",
  574. beginSearchImg:"#t# begin Search Img ...",
  575. findNoPic:"The original image was not found",
  576. findOverBeginLoad:"#t# end of the map, find #t# matching pictures, start loading the first one",
  577. loadNextSimilar:"The original image failed to load, try to load the next result...",
  578. loadError:"Load failed",
  579. openHomePage:"Open Home page",
  580. position:"Display position",
  581. positionTips:"Hold ALT to hide",
  582. topLeft:"The top left corner of the picture",
  583. topRight:"The top right corner of the picture",
  584. bottomRight:"The bottom right corner of the picture",
  585. bottomLeft:"The bottom left corner of the picture",
  586. topCenter:"Beside the picture",
  587. bottomCenter:"Below the picture",
  588. floatBar:"Toolbar",
  589. showDelay:"Show delay",
  590. ms:"ms",
  591. hide:"Hide",
  592. hideDelay:"Hide Delay",
  593. forceShow:"Show toolbar over Non-zoomed image beyond that size, ",
  594. forceShowTip:"Show floating toolbar when non-scaled image size exceeds the size set below",
  595. sizeLimitOr:"Effected by height OR width only",
  596. px:"px",
  597. minSizeLimit:"Show toolbar over Zoomed image beyond that size",
  598. minSizeLimitTip:"After the image is scaled (the original size of the image does not match the actual size), the floating toolbar is displayed when the shown image length is greater than the set value.",
  599. defaultSizeLimit:"Default size limit for gallery",
  600. listenBg:"Listening background image",
  601. listenBgTip:"Show toolbar on the element with the background image",
  602. butonOrder:"Sort of toolbar icons",
  603. keysEnable:"Enable shortcuts",
  604. keysActual:"Open the big picture",
  605. keysActualTip:"Press this button to open a large image when floating bar appears",
  606. keysSearch:"Find the original image",
  607. keysSearchTip:"Press this button to find the original image when floating bar appears",
  608. headSearchTip:"Search by image",
  609. headSearchAll:"Search All",
  610. keysCurrent:"Open the current picture",
  611. keysCurrentTip:"Press this button to open the current image when floating bar appears",
  612. keysMagnifier:"Open the magnifier to observe",
  613. keysMagnifierTip:"Press this button to open the magnifier when floating bar appears",
  614. keysGallery:"Open Gallery(Global with funcKeys)",
  615. keysGalleryTip:"Press this button to open the Gallery when floating bar appears",
  616. openGallery:"Open Gallery",
  617. magnifier:"Zoom",
  618. magnifierRadius:"Default radius",
  619. magnifierWheelZoomEnabled:"Enable wheel zoom",
  620. magnifierWheelZoomRange:"Zoom ratio for magnifier",
  621. gallery:"Gallery",
  622. galleryFitToScreen:"Scale the image to fit the screen",
  623. galleryFitToScreenSmall:"Scale the small image also",
  624. galleryFitToScreenTip:"Adapt to be contain, not cover",
  625. galleryScrollEndToChange:"Switch the image after the long picture scrolls to the end",
  626. galleryScrollEndToChangeTip:"Valid after canceling the previous option",
  627. galleryExportType:"Default sort when images exported",
  628. grid:"Tile sorting",
  629. gridBig:"original sorting",
  630. list:"List sorting",
  631. galleryAutoLoad:"Automatically load more images on next page",
  632. galleryLoadAll:"Automatically process all pages when loading more images",
  633. galleryLoadAllTip:"Too many pages may affect the experience",
  634. galleryDownloadWithZip:"Compress to ZIP when download all",
  635. galleryDownloadWithZipAlert:"Compressing, wait for a while please",
  636. galleryScaleSmallSize1:"The actual size is less than the height and width",
  637. galleryScaleSmallSize2:"Pixels are grouped into small size images",
  638. galleryShowSmallSize:"Show small size pictures by default",
  639. galleryTransition:"Show effects when gallery switch",
  640. gallerySidebarPosition:"Thumbnail bar position",
  641. bottom:"Bottom",
  642. right:"Right",
  643. left:"Left",
  644. top:"Top",
  645. gallerySidebarSize:"Height",
  646. gallerySidebarSizeTip:"The height of the thumbnail bar (if it is horizontal) or the width (if it is vertical)",
  647. galleryMax1:"Maximum prefetch",
  648. galleryMax2:"Pictures (before and after)",
  649. galleryAutoZoom:"Zoom changes back to 100% (chrome)",
  650. galleryAutoZoomTip:"If you zoom in, change the zoom of the image and sidebar sections back to 100% and increase the viewable area (only valid under chrome)",
  651. galleryDescriptionLength1:"Maximum of annotation is",
  652. galleryDescriptionLength2:"Characters",
  653. galleryAutoOpenSites:"Regulars of urls for auto open gallery, one per line, start with @ for ViewMore",
  654. gallerySearchData:"Site rules for search, empty it to restore",
  655. galleryEditSite:"Online editing site",
  656. imgWindow:"ImgWindow",
  657. imgWindowFitToScreen:"Adapt to the screen",
  658. imgWindowFitToScreenTip:"Adapt to be contain, not cover",
  659. imgWindowDefaultTool:"The tool selected by default when opening the window",
  660. hand:"Hand",
  661. rotate:"Rotate",
  662. zoom:"Magnifier",
  663. imgWindowEscKey:"Esc key to close",
  664. imgWindowDblClickImgWindow:"Double click to close",
  665. imgWindowClickOutside:"Click overlayer to close",
  666. imgWindowClickOutsideTip:"Only enable when Overlayer is shown",
  667. none:"None",
  668. click:"Click",
  669. dblclick:"Double click",
  670. imgWindowOverlayerShown:"Overlay",
  671. imgWindowOverlayerColor:"Color and Opacity",
  672. imgWindowShiftRotateStep1:"Rotate when hold down the Shift key on every ",
  673. imgWindowShiftRotateStep2:"Degree",
  674. imgWindowMouseWheelZoom:"MouseWheel Zoom",
  675. imgWindowZoomRange:"Zoom Range",
  676. imgWindowZoomRangeTip:"Zoom ratio (must be positive)",
  677. others:"Other",
  678. waitImgLoad:"Start to perform operations such as zooming when image is loaded",
  679. waitImgLoadTip:"When holding down the Ctrl key, you can temporarily execute opposite to this setting",
  680. debug:"Debug mode",
  681. firstEngine:"Preferred (first) search engine",
  682. refreshWhenError:"Read error, click to overload",
  683. switchSlide:"Switch sidebar",
  684. viewmore:"View more",
  685. countDown:"CountDown"
  686. };
  687. break;
  688. }
  689. function i18n(key,inserts){
  690. var result=i18nData[key],i;
  691. if(inserts){
  692. if(typeof inserts!="object")inserts=[inserts];
  693. for(i=0;i<inserts.length;i++){
  694. result=result.replace("#t#",inserts[i]);
  695. }
  696. }
  697. return result?result:key;
  698. }
  699. var defaultSearchData=`Google | https://www.google.com/searchbyimage?image_url=#t#
  700. Yandex | https://yandex.com/images/search?source=collections&rpt=imageview&url=#t#
  701. SauceNAO | https://saucenao.com/search.php?db=999&url=#t#
  702. IQDB | https://iqdb.org/?url=#t#
  703. 3D IQDB | https://3d.iqdb.org/?url=#t#
  704. Baidu | https://graph.baidu.com/details?isfromtusoupc=1&tn=pc&carousel=0&promotion_name=pc_image_shituindex&extUiData%5bisLogoShow%5d=1&image=#t#
  705. Bing | https://www.bing.com/images/search?view=detailv2&iss=sbi&form=SBIVSP&sbisrc=UrlPaste&q=imgurl:#t#
  706. TinEye | https://www.tineye.com/search?url=#t#
  707. Sogou | https://pic.sogou.com/ris?query=#t#
  708. 360 | http://st.so.com/stu?imgurl=#t#
  709. WhatAnime | https://trace.moe/?url=#t#
  710. Ascii2D | https://ascii2d.net/search/url/#t#
  711. Trace Moe | https://trace.moe/?url=#t#`;
  712.  
  713. if(typeof GM_addStyle=='undefined')var GM_addStyle=GM&&GM.addStyle?GM.addStyle:(s)=>{};
  714. if(typeof GM_openInTab=='undefined')var GM_openInTab=GM&&GM.openInTab?GM.openInTab:(s)=>{};
  715. if(typeof GM_setClipboard=='undefined')var GM_setClipboard=GM&&GM.setClipboard?GM.setClipboard:(s)=>{};
  716. if(typeof GM_xmlhttpRequest=='undefined')var GM_xmlhttpRequest=GM&&GM.xmlhttpRequest?GM.xmlhttpRequest:(f)=>{};
  717. if(typeof GM_registerMenuCommand=='undefined')var GM_registerMenuCommand=GM&&GM.registerMenuCommand?GM.registerMenuCommand:(s,f)=>{};
  718. if(typeof GM_notification=='undefined')var GM_notification=GM&&GM.notification?GM.notification:(s)=>{};
  719. if(typeof GM_download=='undefined')var GM_download=(u,n)=>{saveAs(u,n)};
  720. var prefs;
  721. function init(topObject,window,document,arrayFn,envir,storage,unsafeWindow){
  722. // 默认设置,请到设置界面修改
  723. prefs={
  724. floatBar:{//浮动工具栏相关设置.
  725. butonOrder:['actual','current','gallery','magnifier'],//按钮排列顺序'actual'(实际的图片),'current'(当前显示的图片),'magnifier'(放大镜观察),'gallery'(图集),'search'(搜索原图)
  726. listenBg:true,//监听背景图
  727. showDelay:366,//浮动工具栏显示延时.单位(毫秒)
  728. hideDelay:566,//浮动工具栏隐藏延时.单位(毫秒)
  729. position:'top left',// 取值为: 'top left'(图片左上角) 或者 'top right'(图片右上角) 'bottom right'(图片右下角) 'bottom left'(图片左下角);
  730. offset:{//浮动工具栏偏移.单位(像素)
  731. x:-15,//x轴偏移(正值,向右偏移,负值向左)
  732. y:-15,//y轴偏移(正值,向下,负值向上)
  733. },
  734. forceShow:{//在没有被缩放的图片上,但是大小超过下面设定的尺寸时,强制显示浮动框.
  735. enabled:true,//启用强制显示.
  736. size:{//图片尺寸.单位(像素);
  737. w:45,
  738. h:45,
  739. },
  740. },
  741. minSizeLimit:{//就算是图片被缩放了(看到的图片被设定了width或者height限定了大小,这种情况下),如果图片显示大小小于设定值,那么也不显示浮动工具栏.
  742. w:15,
  743. h:15,
  744. },
  745. sizeLimitOr:false,
  746.  
  747. // 按键,感觉用不太到,默认禁用
  748. keys: {
  749. enable: false,
  750. actual: 'a', // 当出现悬浮条时按下 `a` 打开原图
  751. search: 's',
  752. current: 'c',
  753. magnifier: 'm',
  754. gallery: 'g',
  755. },
  756. globalkeys: {
  757. ctrl: true,
  758. alt: false,
  759. shift: false,
  760. command: false,
  761. type: "hold"
  762. }
  763. },
  764.  
  765. magnifier:{//放大镜的设置.
  766. radius: 77,//默认半径.单位(像素).
  767. wheelZoom:{//滚轮缩放.
  768. enabled:true,
  769. pauseFirst:true,//需要暂停(单击暂停)后,才能缩放.(推荐,否则因为放大镜会跟着鼠标,如果放大镜过大,那么会影响滚动.)..
  770. 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],//缩放的范围
  771. },
  772. },
  773.  
  774. gallery:{//图库相关设定
  775. loadMore:false,
  776. loadAll:true,//加载更多时是否加载全部页面
  777. fitToScreen:true,//图片适应屏幕(适应方式为contain,非cover).
  778. fitToScreenSmall:false,
  779. scrollEndToChange:true,
  780. exportType:'grid',
  781. sidebarPosition: 'bottom',//'top' 'right' 'bottom' 'left' 四个可能值
  782. sidebarSize: 120,//侧栏的高(如果是水平放置)或者宽(如果是垂直放置)
  783. sidebarToggle: true, // 是否显示隐藏按钮
  784. transition:true,//大图片区的动画。
  785. preload:true,//对附近的图片进行预读。
  786. max:5,//最多预读多少张(前后各多少张)
  787.  
  788. zoomresized: 25, // 图片尺寸最少相差比例,单位:%
  789. scaleSmallSize: 250, // 图库的新类别,缩放的图片,尺寸的高或宽都小于该值
  790. showSmallSize:true,//是否默认显示小尺寸图片
  791.  
  792. scrollEndAndLoad: false, // 滚动主窗口到最底部,然后自动重载库的图片。还有bug,有待进一步测试
  793. scrollEndAndLoad_num: 3, // 最后几张图片执行
  794.  
  795. autoZoom: true, // 如果有放大,则把图片及 sidebar 部分的缩放改回 100%,增大可视面积(仅在 chrome 下有效)
  796. descriptionLength: 32, // 注释的最大宽度
  797. editSite: "Lunapic",
  798. defaultSizeLimit:{
  799. w:200,
  800. h:200
  801. },
  802. searchData:defaultSearchData,
  803. downloadWithZip:false
  804. },
  805.  
  806. imgWindow:{// 图片窗相关设置
  807. suitLongImg: true,
  808. fitToScreen: false,//适应屏幕,并且水平垂直居中(适应方式为contain,非cover).
  809. syncSelectedTool:true,//同步当前选择的工具,如果开了多个图片窗口,其中修改一个会反映到其他的上面。
  810. defaultTool:'hand',//"hand","rotate","zoom";打开窗口的时候默认选择的工具
  811. close:{//关闭的方式
  812. escKey:true,//按esc键
  813. dblClickImgWindow: true,//双击图片窗口
  814. clickOutside:'', // 点击图片外部关闭。值为''|'click'|'dblclick';无或点击或双击
  815. },
  816. overlayer:{// 覆盖层.
  817. shown:false,//显示
  818. color:'rgba(0,0,0,0.8)',//颜色和不透明度设置.
  819. },
  820. shiftRotateStep:15,// 旋转的时候,按住shift键时,旋转的步进.单位:度.
  821. zoom:{//滚轮缩放
  822. 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],//缩放比例.(不要出现负数,谢谢-_-!~)
  823. mouseWheelZoom:true,//是否允许使用滚轮缩放。
  824. },
  825. },
  826.  
  827. //等图片完全载入后,才开始执行弹出,放大等等操作,
  828. //按住ctrl键的时候,可以临时执行和这个设定相反的设定.
  829. waitImgLoad: false,
  830.  
  831. //框架里面的图片在顶层窗口展示出来,但是当frame与顶层窗口domain不一样的时候,可能导致图片被反盗链拦截,
  832. //按住shift键,可以临时执行和这个设定相反的设定
  833. framesPicOpenInTopWindow: true,
  834.  
  835. // lowLevel: true, // 如果有多个图片,优先选择低一级的
  836.  
  837. debug: false,
  838. firstEngine:"Tineye"
  839. };
  840.  
  841. var tprules=[
  842. function(a) {
  843. var oldsrc = this.src,newsrc = this.src;
  844. if(this.dataset && this.dataset.original){
  845. newsrc=this.dataset.original;
  846. }else if(this._lazyrias && this._lazyrias.srcset){
  847. newsrc=this._lazyrias.srcset[this._lazyrias.srcset.length-1];
  848. }else if(this.dataset && this.dataset.origFile){
  849. newsrc=this.dataset.origFile;
  850. }else if(this.srcset){
  851. var srcs=this.srcset.split(","),largeSize=0;
  852. srcs.forEach(srci=>{
  853. let srcInfo=srci.trim().split(" "),curSize=parseInt(srcInfo[1]);
  854. if(srcInfo[1] && curSize>largeSize){
  855. largeSize=curSize;
  856. newsrc=srcInfo[0];
  857. }
  858. });
  859. }
  860. return oldsrc != newsrc ? newsrc : null;
  861. }
  862. ];
  863.  
  864. //图标
  865. prefs.icons={
  866. actual:'',
  867. current:'',
  868. magnifier:'',
  869. gallery:'',
  870. search:'',
  871.  
  872. retry:'',
  873. loading:'',
  874. loadingCancle:'',
  875.  
  876. hand:'',
  877. rotate:'',
  878. zoom:'',
  879. flipVertical:'',
  880. flipHorizontal:'',
  881. close:'',
  882. searchBtn:'',
  883. rotateIndicatorBG:'',
  884. rotateIndicatorPointer:'',
  885.  
  886. arrowTop:'',
  887. arrowBottom:'',
  888. arrowLeft:'',
  889. arrowRight:'',
  890.  
  891. fivePointedStar:'',
  892.  
  893. brokenImg:'',
  894. brokenImg_small:'',
  895. };
  896.  
  897. prefs.share={
  898. weibo:{
  899. disabled:false,
  900. name:'新浪微博',
  901. limitLang:'zh-CN',
  902. icon:'',
  903. api:function(args){
  904. var url='http://service.weibo.com/share/share.php?'+
  905. 'title='+args.title+
  906. '&url='+args.url+
  907. '&pic='+args.pic;
  908. return {
  909. url:url,
  910. wSize:{
  911. h:500,
  912. w:620,
  913. },
  914. };
  915. },
  916. },
  917. t:{
  918. name:'腾讯微博',
  919. limitLang:'zh-CN',
  920. icon:'',
  921. api:function(args){
  922. var url='http://v.t.qq.com/share/share.php?'+
  923. 'title='+args.title+
  924. '&url='+args.url+
  925. '&pic='+args.pic;
  926. return {
  927. url:url,
  928. wSize:{
  929. h:500,
  930. w:620,
  931. },
  932. };
  933. },
  934. },
  935. qZone:{
  936. name:'QQ空间',
  937. limitLang:'zh-CN',
  938. icon:'',
  939. api:function(args){
  940. var url='http://sns.qzone.qq.com/cgi-bin/qzshare/cgi_qzshare_onekey?'+
  941. 'title='+args.title+
  942. '&pics='+args.pic+
  943. '&url='+args.url;
  944. return {
  945. url:url,
  946. wSize:{
  947. h:650,
  948. w:620,
  949. },
  950. };
  951. },
  952. },
  953. fanfou:{
  954. name:'饭否',
  955. limitLang:'zh-CN',
  956. icon:'',
  957. api:function(args){
  958. var url='http://fanfou.com/sharer/image?'+
  959. 'u='+args.url+
  960. '&t='+args.title+
  961. '&img_src='+args.pic;
  962. return{
  963. url:url,
  964. wSize:{
  965. h:550,
  966. w:650,
  967. },
  968. };
  969. },
  970. },
  971. tieba:{
  972. name:'百度贴吧',
  973. limitLang:'zh-CN',
  974. icon:'',
  975. api:function(args){
  976. var url = 'http://tieba.baidu.com/f/commit/share/openShareApi?'+
  977. 'title='+args.title+
  978. '&url='+args.url+
  979. '&pic='+args.pic;
  980. return {
  981. url:url,
  982. wSize:{
  983. h:600,
  984. w:630,
  985. },
  986. };
  987. },
  988. },
  989. renren:{
  990. name:'人人网',
  991. limitLang:'zh-CN',
  992. icon:'',
  993. api:function(args){
  994. var url='http://widget.renren.com/dialog/share?'+
  995. 'link='+args.url+
  996. '&title='+args.title+
  997. '&pic='+args.pic;
  998. return {
  999. url:url,
  1000. wSize:{
  1001. h:600,
  1002. w:650,
  1003. },
  1004. };
  1005. },
  1006. },
  1007. douban:{
  1008. name:'豆瓣',
  1009. limitLang:'zh-CN',
  1010. icon:'',
  1011. api:function(args){
  1012. var url='http://shuo.douban.com/%21service/share?'+
  1013. 'href='+args.url+
  1014. '&name='+args.title+
  1015. '&image='+args.pic;
  1016. return {
  1017. url:url,
  1018. wSize:{
  1019. h:350,
  1020. w:600,
  1021. },
  1022. };
  1023. },
  1024. },
  1025. facebook:{
  1026. name:'Facebook',
  1027. icon:'',
  1028. api:function(args){
  1029. var url='https://www.facebook.com/sharer/sharer.php?'+
  1030. 'u='+encodeURIComponent(args.url)+
  1031. '&t='+args.title;
  1032. return {
  1033. url:url,
  1034. wSize:{
  1035. h:436,
  1036. w:626,
  1037. },
  1038. };
  1039. },
  1040. },
  1041. twitter:{
  1042. name:'Twitter',
  1043. icon:'',
  1044. api:function(args){
  1045. var url='https://twitter.com/intent/tweet?'+
  1046. 'url='+args.url+
  1047. '&text='+args.title;
  1048. return {
  1049. url:url,
  1050. wSize:{
  1051. h:350,
  1052. w:600,
  1053. },
  1054. };
  1055. },
  1056. },
  1057. pinterest:{
  1058. name:'Pinterest',
  1059. icon:'',
  1060. api:function(args){
  1061. var url='https://pinterest.com/pin/create/button/?'+
  1062. 'url='+args.url+
  1063. '&media='+encodeURIComponent(args.pic)+
  1064. '&description='+encodeURIComponent(args.title);
  1065. return {
  1066. url:url,
  1067. wSize:{
  1068. h:350,
  1069. w:600,
  1070. },
  1071. };
  1072. },
  1073. }
  1074. };
  1075.  
  1076. if (typeof String.prototype.startsWith != 'function') {
  1077. String.prototype.startsWith = function(str) {
  1078. return this.slice(0, str.length) == str;
  1079. };
  1080. }
  1081.  
  1082. function getMStr(func) {
  1083. var lines = func.toString();
  1084. lines = lines.substring(lines.indexOf("/*") + 3, lines.lastIndexOf("*/"));
  1085. return lines;
  1086. }
  1087.  
  1088. function toRE(obj, flag) {
  1089. if (!obj) {
  1090. return obj;
  1091. } else if (obj instanceof RegExp) {
  1092. return obj;
  1093. } else if (flag) {
  1094. return new RegExp(obj, flag);
  1095. } else if (obj instanceof Array) {
  1096. return new RegExp(obj[0], obj[1]);
  1097. } else if (typeof obj === 'string') {
  1098. if (obj.indexOf('*') != -1 && obj.indexOf('.*') == -1) {
  1099. obj = wildcardToRegExpStr(obj);
  1100. }
  1101. return new RegExp(obj);
  1102. }
  1103. }
  1104.  
  1105. function wildcardToRegExpStr(urlstr) {
  1106. if (urlstr.source) return urlstr.source;
  1107. var reg = urlstr.replace(/[()\[\]{}|+.,^$?\\]/g, "\\$&").replace(/\*+/g, function(str){
  1108. return str === "*" ? ".*" : "[^/]*";
  1109. });
  1110. return "^" + reg + "$";
  1111. }
  1112.  
  1113. function isXPath(xpath) {
  1114. return xpath.startsWith('./') || xpath.startsWith('//') || xpath.startsWith('id(');
  1115. }
  1116.  
  1117. function getElementMix(selector, contextNode, doc) {
  1118. var ret;
  1119. if (!selector || !contextNode) return ret;
  1120. doc = doc || document;
  1121.  
  1122. var type = typeof selector;
  1123. if (type == 'string') {
  1124. if (isXPath(selector)) {
  1125. ret = getElementByXpath(selector, contextNode, doc);
  1126. } else {
  1127. ret = contextNode.parentNode.querySelector(selector);
  1128. }
  1129. } else if (type == 'function') {
  1130. ret = selector(contextNode, doc);
  1131. }
  1132. return ret;
  1133. }
  1134.  
  1135. function launchFullScreen(element) {
  1136. if (element.requestFullscreen) {
  1137. element.requestFullscreen();
  1138. } else if (element.msRequestFullscreen) {
  1139. element.msRequestFullscreen();
  1140. } else if (element.mozRequestFullScreen) {
  1141. element.mozRequestFullScreen();
  1142. } else if (element.webkitRequestFullscreen) {
  1143. element.webkitRequestFullscreen(Element.ALLOW_KEYBOARD_INPUT);
  1144. }
  1145. }
  1146.  
  1147. function cancelFullScreen() {
  1148. if (document.exitFullscreen) {
  1149. document.exitFullscreen();
  1150. } else if (document.msExitFullscreen) {
  1151. document.msExitFullscreen();
  1152. } else if (document.mozCancelFullScreen) {
  1153. document.mozCancelFullScreen();
  1154. } else if (document.webkitExitFullscreen) {
  1155. document.webkitExitFullscreen();
  1156. }
  1157. }
  1158.  
  1159. // 检测缩放
  1160. function detectZoom (){
  1161. var ratio = 0,
  1162. screen = window.screen,
  1163. ua = navigator.userAgent.toLowerCase();
  1164.  
  1165. if (window.devicePixelRatio !== undefined) {
  1166. ratio = window.devicePixelRatio;
  1167. }
  1168. else if (~ua.indexOf('msie')) {
  1169. if (screen.deviceXDPI && screen.logicalXDPI) {
  1170. ratio = screen.deviceXDPI / screen.logicalXDPI;
  1171. }
  1172. }
  1173. else if (window.outerWidth !== undefined && window.innerWidth !== undefined) {
  1174. ratio = window.outerWidth / window.innerWidth;
  1175. }
  1176.  
  1177. if (ratio){
  1178. ratio = Math.round(ratio * 100);
  1179. }
  1180.  
  1181. return ratio;
  1182. }
  1183.  
  1184. //获取位置
  1185. function getContentClientRect(target){
  1186. var rect=target.getBoundingClientRect();
  1187. var compStyle=unsafeWindow.getComputedStyle(target);
  1188. var pFloat=parseFloat;
  1189. var top=rect.top + pFloat(compStyle.paddingTop) + pFloat(compStyle.borderTopWidth);
  1190. var right=rect.right - pFloat(compStyle.paddingRight) - pFloat(compStyle.borderRightWidth);
  1191. var bottom=rect.bottom - pFloat(compStyle.paddingBottom) - pFloat(compStyle.borderBottomWidth);
  1192. var left=rect.left + pFloat(compStyle.paddingLeft) + pFloat(compStyle.borderLeftWidth);
  1193. return {
  1194. top:top,
  1195. right:right,
  1196. bottom:bottom,
  1197. left:left,
  1198. width:right-left,
  1199. height:bottom-top,
  1200. };
  1201. };
  1202.  
  1203. //获取窗口大小.
  1204. function getWindowSize(){
  1205. /*
  1206. //包含滚动条
  1207. return {
  1208. h:window.innerHeight,
  1209. w:window.innerWidth,
  1210. };
  1211. */
  1212.  
  1213. //去除滚动条的窗口大小
  1214. var de=document.documentElement;
  1215. var body=document.body;
  1216. var backCompat=document.compatMode=='BackCompat';
  1217. return {
  1218. h:backCompat? body.clientHeight : de.clientHeight,
  1219. w:backCompat? body.clientWidth : de.clientWidth,
  1220. };
  1221.  
  1222. };
  1223.  
  1224. //获取已滚动的距离
  1225. function getScrolled(container){
  1226. if(container){
  1227. return {
  1228. x:container.scrollLeft,
  1229. y:container.scrollTop,
  1230. };
  1231. };
  1232. return {
  1233. x:'scrollX' in window ? window.scrollX : ('pageXOffset' in window ? window.pageXOffset : document.documentElement.scrollLeft || document.body.scrollLeft),
  1234. y:'scrollY' in window ? window.scrollY : ('pageYOffset' in window ? window.pageYOffset : document.documentElement.scrollTop || document.body.scrollTop),
  1235. };
  1236. };
  1237.  
  1238. //xpath 获取单个元素
  1239. function getElementByXpath(xpath,contextNode,doc){
  1240. doc=doc || document;
  1241. contextNode=contextNode || doc;
  1242. return doc.evaluate(xpath,contextNode,null,9,null).singleNodeValue;
  1243. };
  1244.  
  1245.  
  1246. //事件支持检测.
  1247. function eventSupported( eventName,elem ){
  1248. elem = elem || document.createElement("div");
  1249. eventName = "on" + eventName;
  1250. var isSupported = (eventName in elem);
  1251. if (!isSupported){
  1252. if(!elem.setAttribute){//setAttribute是元素节点的方法
  1253. elem=document.createElement("div");
  1254. };
  1255. var setAttr;
  1256. if(!elem.hasAttribute(eventName)){
  1257. setAttr=true;
  1258. elem.setAttribute(eventName, "return;");
  1259. };
  1260. isSupported = typeof elem[eventName] == "function";
  1261. if(setAttr)elem.removeAttribute(eventName);
  1262. };
  1263. return isSupported;
  1264. };
  1265.  
  1266.  
  1267. //检测属性支持.dom属性
  1268. //返回带前缀的可以直接执行是属性
  1269. function proSupported(proName,elem){
  1270. //判断第一个字母是否大写,如果是的话,为构造函数,前缀也要大写
  1271. var prefix=/^[A-Z]/.test(proName)? ['','WebKit-','O-','Moz-','MS-'] : ['','webkit-','o-','moz-','ms-'];
  1272. var i=0;
  1273. var p_i;
  1274. var sProName;
  1275. elem = elem || document.createElement("div");
  1276. while(typeof (p_i=prefix[i++])!='undefined'){
  1277. sProName=(p_i+proName).replace(/-([A-z])/g,function(a,b){
  1278. return b.toUpperCase();
  1279. });
  1280. if(sProName in elem)return sProName;
  1281. };
  1282. };
  1283.  
  1284.  
  1285. //css属性支持
  1286. //带前缀的默认为大写(所有浏览器支持)
  1287. //比如WebkitTransform,MozTransform,OTransfomr
  1288. //chrome浏览器大小写前缀都行。
  1289. //firefox,opera只能大写
  1290. //ie 9+只能小写
  1291. function cssProSupported(proName,elem,capitalize){
  1292. if(capitalize!==false)capitalize=true;
  1293. proName=proName.toLowerCase();
  1294.  
  1295. var prefix=['','-webkit-','-o-','-moz-','-ms-'];
  1296. elem=elem || document.createElement('div');
  1297. var style=elem.style;
  1298. var camelPro;
  1299.  
  1300. // 会有个错误 invalid 'in' operand style
  1301. try {
  1302. for(var i=0,ii=prefix.length;i<ii;i++){
  1303. var first=true;
  1304. camelPro=(prefix[i]+proName).replace(/-([a-z])/g,function(a,b){
  1305. b=b.toUpperCase();
  1306. if(first){
  1307. first=false;
  1308. if(!capitalize){
  1309. b=b.toLowerCase();
  1310. };
  1311. };
  1312. return b;
  1313. });
  1314. if(camelPro in style){
  1315. return camelPro;
  1316. }
  1317. }
  1318. } catch(ex) {}
  1319.  
  1320. if(!capitalize)return;
  1321. return cssProSupported(proName,elem,false);
  1322.  
  1323. };
  1324.  
  1325. //css属性值支持
  1326. function cssValueSupported(proName,value,elem){
  1327. var prefix=['','-webkit-','-o-','-moz-','-ms-'];
  1328. elem=elem || document.createElement('div');
  1329. var style=elem.style;
  1330. var prefixedValue;
  1331. for(var i=0,ii=prefix.length;i<ii;i++){
  1332. prefixedValue=prefix[i] + value;
  1333. style[proName]=prefixedValue;
  1334. if(style[proName]==prefixedValue){
  1335. return prefixedValue;
  1336. };
  1337. };
  1338. };
  1339.  
  1340.  
  1341. //elem.dataset的兼容实现
  1342. //ie不支持;firefoxGM储存不能反映到元素属性上。
  1343. function dataset(elem,pro,value){
  1344.  
  1345. function getDataPrefix(){
  1346. return 'data-' + pro.replace(/[A-Z]/g,function(m){
  1347. return '-' + m.toLowerCase();
  1348. });
  1349. };
  1350.  
  1351. if(typeof value=='undefined'){//取值
  1352. if(elem.dataset){
  1353. value = elem.dataset[pro];
  1354. }else{//没有取到值,返回undefined,getAttribute默认是返回null,所以判断一下。
  1355. var prefixedPro=getDataPrefix();
  1356. if(elem.hasAttribute(prefixedPro)){
  1357. value=elem.getAttribute(prefixedPro);
  1358. };
  1359. };
  1360. return value;
  1361. }else{
  1362. elem.setAttribute(getDataPrefix(),value);
  1363. };
  1364. };
  1365.  
  1366.  
  1367. //重新检查悬浮图片
  1368. function imgReHover(img){
  1369. //要检查的图片,是当前悬浮的。
  1370. if(!floatBar.shown || floatBar.data.img != img)return;
  1371.  
  1372. var mHover=document.createEvent('MouseEvent');
  1373. var cr=img.getBoundingClientRect();
  1374. 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);
  1375. img.dispatchEvent(mHover);
  1376. };
  1377.  
  1378. // 获取真正的unsafeWindow,chrome里面也能访问到真实环境的变量
  1379. // 在 chrome 37 测试无效
  1380. if(!envir.firefox && !envir.opera && !envir.ie && !storage.supportGM){
  1381. ;(function(){
  1382. document.addEventListener('picViewer-return-unsafeWindow',function(e){
  1383. unsafeWindow = e.detail;
  1384. // alert(unsafeWindow.$);
  1385. },true);
  1386.  
  1387. //页面脚本
  1388. var s=document.createElement('script');
  1389. s.textContent='(' + (function(){
  1390. var cusEvent=document.createEvent('CustomEvent');
  1391. cusEvent.initCustomEvent('picViewer-return-unsafeWindow',false,false,window);
  1392. document.dispatchEvent(cusEvent);
  1393. }).toString() +')()';
  1394. document.head.appendChild(s);
  1395. })();
  1396. };
  1397.  
  1398.  
  1399. //抛出错误到错误控制台
  1400. function throwErrorInfo(err){
  1401. if(console && console.error){
  1402. console.error(err.message + '\n\n' + (err.stacktrace? err.stacktrace : '') + '\n\n' , err);
  1403. };
  1404. };
  1405.  
  1406. //对象克隆
  1407. function cloneObject(obj,deep){
  1408. var obj_i;
  1409. var ret=Array.isArray(obj)? [] : {};
  1410. for(var i in obj){
  1411. if(!obj.hasOwnProperty(i))continue;
  1412. obj_i=obj[i];
  1413. if(!deep || typeof obj_i!='object' || obj_i===null || obj_i.nodeType){
  1414. ret[i]=obj_i;
  1415. }else{
  1416. ret[i]=cloneObject(obj_i,deep);
  1417. };
  1418. };
  1419. return ret;
  1420. };
  1421.  
  1422. //闪烁元素。
  1423. function flashEle(ele,duration){
  1424. if(dataset(ele,'pvFlashing'))return;
  1425. if(ele.offsetHeight==0)return;
  1426. dataset(ele,'pvFlashing','1');
  1427.  
  1428. var oOutline=ele.style.outline;
  1429. var oOutlineOffset=ele.style.outlineOffset;
  1430. var oOpacity=ele.style.opacity;
  1431. var oTransform=ele.style[support.cssTransform];
  1432.  
  1433. var count=0;
  1434. var startTime=Date.now();
  1435. duration=duration? duration : 1200;
  1436.  
  1437. var flashInterval=setInterval(function(){
  1438. var outline='none',
  1439. outlineOffset=0,
  1440. opacity=0.3,
  1441. transform='';
  1442.  
  1443. if(count % 2 == 0){
  1444. outline='5px dashed rgba(255,0,0,0.95)';
  1445. opacity=0.95;
  1446. outlineOffset='1px';
  1447. transform='scale(1.1)';
  1448. }else{
  1449. if((Date.now() - startTime) > duration){
  1450. clearInterval(flashInterval);
  1451. outline=oOutline;
  1452. opacity=oOpacity;
  1453. outlineOffset=oOutlineOffset;
  1454. transform=oTransform;
  1455. ele.removeAttribute('data-pv-flashing');
  1456. };
  1457. };
  1458.  
  1459. ele.style.outline=outline;
  1460. ele.style.outlineOffset=outlineOffset;
  1461. ele.style.opacity=opacity;
  1462. ele.style[support.cssTransform]=transform;
  1463.  
  1464. count++;
  1465. },80);
  1466. };
  1467.  
  1468. //支持情况.
  1469. var support={
  1470. cssTransform:cssProSupported('transform'),
  1471. cssCursorValue:{
  1472. zoomIn:cssValueSupported('cursor','zoom-in'),
  1473. zoomOut:cssValueSupported('cursor','zoom-out'),
  1474. grab:cssValueSupported('cursor','grab'),
  1475. grabbing:cssValueSupported('cursor','grabbing'),
  1476. },
  1477. };
  1478.  
  1479.  
  1480.  
  1481. //动画算法
  1482. /*
  1483. t: current time(当前时间);
  1484. b: beginning value(初始值);
  1485. c: change in value(变化量);
  1486. d: duration(持续时间)。
  1487. */
  1488.  
  1489. var Tween = {
  1490. Cubic: {
  1491. easeInOut:function(t,b,c,d){
  1492. return -c/2*(Math.cos(Math.PI*t/d)-1)+b
  1493. }
  1494. },
  1495. };
  1496.  
  1497. //imgReady
  1498. var imgReady=(function(){
  1499. var iRInterval,
  1500. iRReadyFn=[],
  1501. isrcs=[]
  1502. ;
  1503.  
  1504. var timeLimit=3 * 60 * 1000;//3分钟
  1505.  
  1506. function checkReady(){
  1507. var now= Date.now();
  1508. for(var i=0,ii=iRReadyFn.length,iRReadyFn_i;i<ii;i++){
  1509. iRReadyFn_i=iRReadyFn[i];
  1510. //now - iRReadyFn_i.startTime >= timeLimit ||
  1511. if(iRReadyFn_i()){
  1512. iRReadyFn.splice(i,1);
  1513. isrcs.splice(i,1);
  1514. i--;
  1515. ii--;
  1516. };
  1517. };
  1518. if(iRReadyFn.length==0){
  1519. clearInterval(iRInterval);
  1520. iRInterval=null;
  1521. };
  1522. };
  1523.  
  1524.  
  1525.  
  1526. var imgReady=function(img,opts){
  1527.  
  1528. if(/NodeList|HTMLCollection/.test(Object.prototype.toString.call(img)) || Array.isArray(img)){
  1529. arrayFn.forEach.call(img,function(img,index,array){
  1530. if(img instanceof HTMLImageElement){
  1531. imgReady(img,opts);
  1532. };
  1533. });
  1534. return;
  1535. };
  1536.  
  1537. if(!(img instanceof HTMLImageElement)){
  1538. var t_img=document.createElement('img');
  1539. t_img.src=img;
  1540. img=t_img;
  1541. t_img=null;
  1542. };
  1543.  
  1544. var ready,load,error,loadEnd,abort,timeout,time;
  1545. ready=opts.ready;
  1546. load=opts.load;
  1547. error=opts.error;
  1548. loadEnd=opts.loadEnd;
  1549. abort=opts.abort;
  1550. timeout=opts.timeout;
  1551. time=typeof opts.time=='number'? opts.time : 0;
  1552.  
  1553. if(time){
  1554. setTimeout(function(){
  1555. if(!loadEndDone){
  1556. aborted=true;
  1557. removeListener();
  1558. img.src= prefs.icons.brokenImg_small;
  1559. if(timeout){
  1560. timeout.call(img,{
  1561. target:img,
  1562. type:'timeout',
  1563. });
  1564. };
  1565. loadEndDone=true;
  1566. if(loadEnd){
  1567. loadEnd.call(img,{
  1568. target:img,
  1569. type:'timeout',
  1570. });
  1571. };
  1572.  
  1573. };
  1574. },time);
  1575. };
  1576.  
  1577. var src=img.src;
  1578. var loadEndDone;
  1579.  
  1580. function go(type,e){
  1581. switch(type){
  1582. case 'load':{
  1583. removeListener();
  1584. go('ready');//如果直接触发load,那么先触发ready
  1585. if(load){
  1586. load.call(img,e);
  1587. };
  1588.  
  1589. if(!loadEndDone){
  1590. loadEndDone=true;
  1591. if(loadEnd){
  1592. loadEnd.call(img,e);
  1593. };
  1594. };
  1595. }break;
  1596. case 'ready':{
  1597. if(!ready || readyHandler.done)return;
  1598. readyHandler.done=true;
  1599. ready.call(img,{
  1600. target:img,
  1601. type:'ready',
  1602. });
  1603. }break;
  1604. case 'error':{
  1605. removeListener();
  1606. if(error){
  1607. error.call(img,e);
  1608. };
  1609. if(!loadEndDone){
  1610. loadEndDone=true;
  1611. if(loadEnd){
  1612. loadEnd.call(img,e);
  1613. };
  1614. };
  1615. }break;
  1616. };
  1617. };
  1618.  
  1619. var aborted;
  1620. var ret={
  1621. img:img,
  1622. abort:function(){
  1623. if(!loadEndDone){
  1624. aborted=true;
  1625. removeListener();
  1626. img.src= prefs.icons.brokenImg_small;
  1627. if(abort){
  1628. abort.call(img,{
  1629. target:img,
  1630. type:'abort',
  1631. });
  1632. };
  1633. loadEndDone=true;
  1634. if(loadEnd){
  1635. loadEnd.call(img,{
  1636. target:img,
  1637. type:'abort',
  1638. });
  1639. };
  1640. };
  1641. },
  1642. };
  1643.  
  1644. function readyHandler(){//尽快的检测图片大小.
  1645. if(loadEndDone || aborted)return true;
  1646. if(img.naturalWidth==0 || img.naturalHeight==0)return;
  1647. go('ready');
  1648. return true;
  1649. };
  1650.  
  1651.  
  1652. function loadHandler(e){
  1653. go('load',e);
  1654. };
  1655.  
  1656. function errorHandler(e){
  1657. go('error',e);
  1658. };
  1659.  
  1660. function removeListener(){
  1661. img.removeEventListener('load',loadHandler,true);
  1662. img.removeEventListener('error',errorHandler,true);
  1663. };
  1664.  
  1665. //ready必须在load之前触发。
  1666.  
  1667. if(img.complete){//图片已经加载完成.
  1668. if(typeof img.width=='number' && img.width && img.height){//图片
  1669. setTimeout(function(){
  1670. if(aborted)return;
  1671. go('load',{
  1672. type:'load',
  1673. target:img,
  1674. });
  1675. },0);
  1676. }else{//这不是图片.opera会识别错误.
  1677. setTimeout(function(){
  1678. if(aborted)return;
  1679. go('error',{
  1680. type:'error',
  1681. target:img,
  1682. });
  1683. },0);
  1684. };
  1685. return ret;
  1686. };
  1687.  
  1688.  
  1689. img.addEventListener('error',errorHandler,true);
  1690. img.addEventListener('load',loadHandler,true);
  1691.  
  1692.  
  1693. if(ready){
  1694. var index=isrcs.indexOf(src);
  1695. if(index==-1){
  1696. isrcs.push(src);
  1697. readyHandler.startTime= Date.now();
  1698. iRReadyFn.push(readyHandler);
  1699. }else{
  1700. iRReadyFn[index].startTime= Date.now();
  1701. };
  1702.  
  1703. if(!iRInterval){
  1704. iRInterval=setInterval(checkReady,66);
  1705. };
  1706. };
  1707.  
  1708. return ret;
  1709. };
  1710.  
  1711. return imgReady;
  1712. })();
  1713.  
  1714.  
  1715. var addWheelEvent=(function(){
  1716.  
  1717. function getSupportEventName(){
  1718. var ret='DOMMouseScroll';
  1719. if(eventSupported('wheel')){//w3c FF>=17 ie>=9
  1720. ret='wheel';
  1721. }else if(eventSupported('mousewheel')){//opera,chrome
  1722. ret='mousewheel';
  1723. };
  1724. return ret;
  1725. };
  1726.  
  1727. var eventName;
  1728.  
  1729. return function(ele,callback,useCapture){
  1730. if(!eventName){
  1731. eventName=getSupportEventName();
  1732. };
  1733.  
  1734. ele.addEventListener(eventName,function(e){
  1735. var type=e.type;
  1736. var ne;
  1737. if(type!='wheel'){
  1738. ne={};
  1739. for(var i in e){
  1740. ne[i]=e[i];
  1741. };
  1742.  
  1743. ne.type='wheel';
  1744. ne.deltaX=0;
  1745. ne.deltaY=0;
  1746. ne.deltaZ=0;
  1747. ne.deltaMode=1;//line
  1748. ne.preventDefault=e.preventDefault.bind(e);
  1749. ne.stopPropagation=e.stopPropagation.bind(e);
  1750.  
  1751. var x=0,y=0;
  1752. if(typeof e.axis=='number'){//DOMMouseScroll
  1753. if(e.axis==2){
  1754. y=e.detail;
  1755. }else{
  1756. x=e.detail;
  1757. };
  1758. }else{
  1759. //opera早起版本的mousewheel只支持y轴的滚动,e.wheelDeltaY undefined
  1760. if(typeof e.wheelDeltaY=='undefined' || e.wheelDeltaY!=0){
  1761. y=-e.wheelDelta/40;
  1762. }else{
  1763. x=-e.wheelDelta/40;
  1764. };
  1765. };
  1766. ne.deltaY =y;
  1767. ne.deltaX =x;
  1768.  
  1769. };
  1770.  
  1771. callback.call(this,ne? ne : e);
  1772. },useCapture || false);
  1773. };
  1774. })();
  1775.  
  1776.  
  1777. var addCusMouseEvent=(function(){
  1778.  
  1779. function getSupported(){
  1780. return {
  1781. mouseleave:eventSupported('mouseleave'),
  1782. mouseenter:eventSupported('mouseenter'),
  1783. };
  1784. };
  1785.  
  1786. var support;
  1787. var map={
  1788. mouseleave:'mouseout',
  1789. mouseenter:'mouseover',
  1790. };
  1791.  
  1792. return function(type, ele, fn){//事件类型,元素,监听函数
  1793. if(!support){
  1794. support=getSupported();
  1795. };
  1796.  
  1797. // chrome 30+ 虽然支持 mouseenter,但是存在问题
  1798. if(support[type] && !(type == 'mouseenter' && window.chrome)){
  1799. ele.addEventListener(type,fn,false);//mouseleave,enter不冒泡
  1800. }else{
  1801. ele.addEventListener(map[type],function(e){
  1802. var relatedTarget=e.relatedTarget;//mouseout,去往的元素;mouseover,来自的元素
  1803. if(!this.contains(relatedTarget)){
  1804. fn.call(this,e);
  1805. };
  1806. },true);
  1807. };
  1808. };
  1809.  
  1810. })();
  1811.  
  1812.  
  1813. //库
  1814. function GalleryC(){
  1815. this.init();
  1816. };
  1817.  
  1818. var gallery;
  1819. var galleryMode;
  1820.  
  1821. GalleryC.prototype={
  1822. init:function(){
  1823. this.addStyle();
  1824. var container=document.createElement('span');
  1825.  
  1826. this.gallery=container;
  1827. container.className='pv-gallery-container';
  1828. container.tabIndex=1;//为了获取焦点,来截获键盘事件
  1829. container.innerHTML=
  1830. '<span class="pv-gallery-head">'+
  1831. '<span class="pv-gallery-head-float-left">'+
  1832. '<span title="'+i18n("picInfo")+'" class="pv-gallery-head-left-img-info">'+
  1833. '<span class="pv-gallery-head-left-img-info-resolution" title="'+i18n("resolution")+'">0 x 0</span>'+
  1834. '<span class="pv-gallery-head-left-img-info-count" title="'+i18n("picNum")+'">(1 / 1)</span>'+
  1835. '<span class="pv-gallery-head-left-img-info-scaling" title="'+i18n("scaleRatio")+'">(100%)</span>'+
  1836. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1837. '<span class="pv-gallery-head-left-img-info-description" title="'+i18n("picNote")+'"></span>'+
  1838. '<div class="pv-gallery-range-box"><input type="range" id="minsizeW" min="0" max="100" value="0" title="Width"> <span id="minsizeWSpan">0px</span> '+
  1839. '<input type="range" id="minsizeH" min="0" max="100" value="0" title="Height"> <span id="minsizeHSpan">0px</span></div>'+
  1840. '</span>'+
  1841. '</span>'+
  1842.  
  1843. '<span title="'+i18n("exitCollectionTip")+'" class="pv-gallery-head-command pv-gallery-head-command-exit-collection">'+
  1844. '<span>'+i18n("exitCollection")+'</span>'+
  1845. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1846. '</span>'+
  1847.  
  1848. '<span title="'+i18n("loadAllTip")+'" class="pv-gallery-head-command pv-gallery-head-command-nextPage">'+
  1849. '<span>'+i18n("loadAll")+'</span>'+
  1850. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1851. '</span>'+
  1852.  
  1853. '<span title="'+i18n("fiddleTip")+'" class="pv-gallery-head-command pv-gallery-head-command-operate">'+
  1854. '<span>'+i18n("fiddle")+'</span>'+
  1855. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1856. '</span>'+
  1857.  
  1858. '<span class="pv-gallery-head-command-container">'+
  1859. '<span class="pv-gallery-head-command pv-gallery-head-command-collect">'+
  1860. '<span class="pv-gallery-head-command-collect-icon"></span>'+
  1861. '<span class="pv-gallery-head-command-collect-text"></span>'+
  1862. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1863. '</span>'+
  1864. '<span class="pv-gallery-head-command-drop-list pv-gallery-head-command-drop-list-collect">'+
  1865. '<span title="'+i18n("collectDetailTip")+'" class="pv-gallery-head-command-drop-list-item pv-gallery-head-command-drop-list-item-collect-description">'+
  1866. '<span>'+i18n("collectDetail")+':</span>'+
  1867. '<textarea data-prefs="description" cols="25" rows="5"></textarea>'+
  1868. '</span>'+
  1869. '</span>'+
  1870. '</span>'+
  1871.  
  1872. '<span class="pv-gallery-head-command-container">'+
  1873. '<span title="'+i18n("playSlide")+'" class="pv-gallery-head-command pv-gallery-head-command-slide-show">'+
  1874. '<span class="pv-gallery-head-command_overlayer"></span>'+
  1875. '<span class="pv-gallery-head-command-slide-show-button">'+
  1876. '<span class="pv-gallery-head-command-slide-show-button-inner"></span>'+
  1877. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1878. '</span>'+
  1879. '<span class="pv-gallery-head-command-slide-show-countdown" title="'+i18n("countDown")+'"></span>'+
  1880. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1881. '</span>'+
  1882. '<span class="pv-gallery-head-command-drop-list pv-gallery-head-command-drop-list-slide-show">'+
  1883. '<span class="pv-gallery-head-command-drop-list-item" title="'+i18n("slideGapTip")+'">'+
  1884. '<input data-prefs="interval" step="1" min="1" type="number" value="5" />'+
  1885. '<span>'+i18n("slideGap")+'</span>'+
  1886. '</span>'+
  1887. '<span class="pv-gallery-head-command-drop-list-item" title="'+i18n("slideBackTip")+'">'+
  1888. '<input id="pv-gallery-head-command-drop-list-item-slide-show-backward" data-prefs="backward" type="checkbox" />'+
  1889. '<label for="pv-gallery-head-command-drop-list-item-slide-show-backward">'+i18n("slideBack")+'   </label>'+
  1890. '</span>'+
  1891. '<span class="pv-gallery-head-command-drop-list-item" title="'+i18n("slideWaitTip")+'">'+
  1892. '<input id="pv-gallery-head-command-drop-list-item-slide-show-wait" data-prefs="wait" type="checkbox" checked="checked" />'+
  1893. '<label for="pv-gallery-head-command-drop-list-item-slide-show-wait">'+i18n("slideWait")+'</label>'+
  1894. '</span>'+
  1895. '<span class="pv-gallery-head-command-drop-list-item" title="'+i18n("slideSkipErrorTip")+'">'+
  1896. '<input id="pv-gallery-head-command-drop-list-item-slide-show-skipErrorImg" data-prefs="skipErrorImg" type="checkbox" checked="checked" />'+
  1897. '<label for="pv-gallery-head-command-drop-list-item-slide-show-skipErrorImg">'+i18n("slideSkipError")+'</label>'+
  1898. '</span>'+
  1899. '</span>'+
  1900. '</span>'+
  1901.  
  1902. '<span class="pv-gallery-head-command-container">'+
  1903. '<span title="'+i18n("typeTip")+'" class="pv-gallery-head-command pv-gallery-head-command-category">'+
  1904. '<span>'+i18n("type")+'</span>'+
  1905. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1906. '</span>'+
  1907. '<span class="pv-gallery-head-command-drop-list pv-gallery-head-command-drop-list-category">'+
  1908. '</span>'+
  1909. '</span>'+
  1910.  
  1911. '<span class="pv-gallery-head-command-container">'+
  1912. '<span title="'+i18n("commandTip")+'" class="pv-gallery-head-command pv-gallery-head-command-others">'+
  1913. '<span>'+i18n("command")+'</span>'+
  1914. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1915. '</span>'+
  1916. '<span class="pv-gallery-head-command-drop-list pv-gallery-head-command-drop-list-others">'+
  1917. '<span class="pv-gallery-head-command-drop-list-item" data-command="psImage" title="'+i18n("onlineEditTip",prefs.gallery.editSite)+'">'+i18n("onlineEdit")+'</span>'+
  1918. '<span class="pv-gallery-head-command-drop-list-item" data-command="exportImages" title="'+i18n("exportImagesTip")+'">'+i18n("exportImages")+'</span>'+
  1919. '<span class="pv-gallery-head-command-drop-list-item" data-command="copyImages" title="'+i18n("copyImagesUrlTip")+'">'+i18n("copyImagesUrl")+'</span>'+
  1920. '<span class="pv-gallery-head-command-drop-list-item" data-command="downloadImage" title="'+i18n("downloadImageTip")+'">'+i18n("downloadImage")+'</span>'+
  1921. '<span class="pv-gallery-head-command-drop-list-item" data-command="openInNewWindow" title="'+i18n("openInNewWindowTip")+'">'+i18n("openInNewWindow")+'</span>'+
  1922. '<span class="pv-gallery-head-command-drop-list-item" data-command="enterCollection" title="'+i18n("viewCollectionTip")+'">'+i18n("viewCollection")+'</span>'+
  1923. '<span class="pv-gallery-head-command-drop-list-item" data-command="scrollIntoView" title="'+i18n("findInPageTip")+'">'+i18n("findInPage")+'</span>'+
  1924. '<span class="pv-gallery-head-command-drop-list-item" title="'+i18n("autoRefreshTip")+'">'+
  1925. '<label data-command="scrollToEndAndReload">'+i18n("autoRefresh")+'</label>'+
  1926. '<input type="checkbox" data-command="scrollToEndAndReload"/>'+
  1927. '</span>'+
  1928. '<span id="pv-gallery-fullscreenbtn" class="pv-gallery-head-command-drop-list-item" data-command="fullScreen">'+i18n("enterFullsc")+'</span>'+
  1929. '<span class="pv-gallery-head-command-drop-list-item" data-command="openPrefs">'+i18n("openConfig")+'</span>'+
  1930. '</span>'+
  1931. '</span>'+
  1932.  
  1933. '<span class="pv-gallery-head-command-container">'+
  1934. '<span title="'+i18n("headSearchTip")+'" class="pv-gallery-head-command pv-gallery-head-command-search">'+
  1935. '<span>'+i18n("headSearchTip")+'</span>'+
  1936. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1937. '</span>'+
  1938. '<span class="pv-gallery-head-command-drop-list pv-gallery-head-command-drop-list-search">'+
  1939. '<span class="pv-gallery-head-command-drop-list-item" id="headSearchAll" data-command="headSearchAll">'+i18n("headSearchAll")+'</span>'+
  1940. '</span>'+
  1941. '</span>'+
  1942.  
  1943. '<span class="pv-gallery-head-command-container">'+
  1944. '<span title="'+i18n("share")+'" class="pv-gallery-head-command pv-gallery-head-command-share">'+
  1945. '<span>'+i18n("share")+'</span>'+
  1946. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1947. '</span>'+
  1948. '<span class="pv-gallery-head-command-drop-list pv-gallery-head-command-drop-list-share">'+
  1949. '</span>'+
  1950. '</span>'+
  1951.  
  1952. '<span title="'+i18n("closeGallery")+'" class="pv-gallery-head-command pv-gallery-head-command-close">'+
  1953. '</span>'+
  1954.  
  1955. '</span>'+
  1956.  
  1957. '<span class="pv-gallery-body">'+
  1958.  
  1959. '<span class="pv-gallery-img-container">'+
  1960.  
  1961. '<span class="pv-gallery-img-content">'+
  1962. '<span class="pv-gallery-img-parent">'+
  1963. '<img title="'+i18n("refreshWhenError")+'" class="pv-gallery-img_broken" src="'+prefs.icons.brokenImg+'" />'+
  1964. '</span>'+
  1965. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1966. '</span>'+
  1967.  
  1968. '<span class="pv-gallery-img-controler pv-gallery-img-controler-pre"></span>'+
  1969. '<span class="pv-gallery-img-controler pv-gallery-img-controler-next"></span>'+
  1970.  
  1971. '<span class="pv-gallery-scrollbar-h pv-gallery-img-scrollbar-h">'+
  1972. '<span class="pv-gallery-scrollbar-h-track pv-gallery-img-scrollbar-h-track">'+
  1973. '<span class="pv-gallery-scrollbar-h-handle pv-gallery-img-scrollbar-h-handle"></span>'+
  1974. '</span>'+
  1975. '</span>'+
  1976.  
  1977. '<span class="pv-gallery-scrollbar-v pv-gallery-img-scrollbar-v">'+
  1978. '<span class="pv-gallery-scrollbar-v-track pv-gallery-img-scrollbar-v-track">'+
  1979. '<span class="pv-gallery-scrollbar-v-handle pv-gallery-img-scrollbar-v-handle"></span>'+
  1980. '</span>'+
  1981. '</span>'+
  1982.  
  1983. '<span class="pv-gallery-sidebar-toggle" title="'+i18n("switchSlide")+'">'+
  1984. '<span class="pv-gallery-sidebar-toggle-content">▼</span>'+
  1985. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1986. '</span>'+
  1987.  
  1988. '<span class="pv-gallery-sidebar-viewmore" title="'+i18n("viewmore")+'">'+
  1989. '<span class="pv-gallery-sidebar-viewmore-content">✚</span>'+
  1990. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1991. '</span>'+
  1992.  
  1993. '</span>'+
  1994.  
  1995. '<span class="pv-gallery-sidebar-container" unselectable="on">'+
  1996. '<span class="pv-gallery-vertical-align-helper"></span>'+
  1997. '<span class="pv-gallery-sidebar-content" >'+
  1998.  
  1999. '<span class="pv-gallery-sidebar-controler pv-gallery-sidebar-controler-pre"></span>'+
  2000. '<span class="pv-gallery-sidebar-controler pv-gallery-sidebar-controler-next"></span>'+
  2001.  
  2002. '<span class="pv-gallery-sidebar-thumbnails-container">'+
  2003. '</span>'+
  2004.  
  2005. '<span class="pv-gallery-scrollbar-h pv-gallery-thumb-scrollbar-h">'+
  2006. '<span class="pv-gallery-scrollbar-h-track pv-gallery-thumb-scrollbar-h-track">'+
  2007. '<span class="pv-gallery-scrollbar-h-handle pv-gallery-thumb-scrollbar-h-handle"></span>'+
  2008. '</span>'+
  2009. '</span>'+
  2010. '<span class="pv-gallery-scrollbar-v pv-gallery-thumb-scrollbar-v">'+
  2011. '<span class="pv-gallery-scrollbar-v-track pv-gallery-thumb-scrollbar-v-track">'+
  2012. '<span class="pv-gallery-scrollbar-v-handle pv-gallery-thumb-scrollbar-v-handle"></span>'+
  2013. '</span>'+
  2014. '</span>'+
  2015.  
  2016. '</span>'+
  2017. '</span>'+
  2018. '<span class="pv-gallery-maximize-scroll"><span class="pv-gallery-maximize-container"></span></span>'+
  2019. '</span>';
  2020. document.body.appendChild(container);
  2021.  
  2022. var self=this;
  2023.  
  2024. var hideBodyStyle=document.createElement('style');
  2025. this.hideBodyStyle=hideBodyStyle;
  2026. hideBodyStyle.textContent=`body>*:not([class^="pv-"]) img,body>img{display:none}`;
  2027.  
  2028. container.querySelector("#minsizeW").oninput=function(){self.changeMinView();};
  2029. container.querySelector("#minsizeH").oninput=function(){self.changeMinView();};
  2030. container.querySelector("#minsizeWSpan").onclick=function(){
  2031. var minsizeW=window.prompt("Width:",this.value);
  2032. if(!minsizeW)return;
  2033. container.querySelector("#minsizeW").value=minsizeW;
  2034. self.changeMinView();
  2035. };
  2036. container.querySelector("#minsizeHSpan").onclick=function(){
  2037. var minsizeH=window.prompt("Height:",this.value);
  2038. if(!minsizeH)return;
  2039. container.querySelector("#minsizeH").value=minsizeH;
  2040. self.changeMinView();
  2041. };
  2042.  
  2043. var maximizeTrigger=document.createElement('span');
  2044. this.maximizeTrigger=maximizeTrigger;
  2045. maximizeTrigger.innerHTML='-'+i18n("returnToGallery")+'-<span class="pv-gallery-maximize-trigger-close" title="'+i18n("closeGallery")+'"></span>';
  2046. maximizeTrigger.className='pv-gallery-maximize-trigger';
  2047.  
  2048. document.body.appendChild(maximizeTrigger);
  2049.  
  2050.  
  2051. var validPos=['top','right','bottom','left'];
  2052. var sBarPosition=prefs.gallery.sidebarPosition;
  2053. if(validPos.indexOf(sBarPosition)==-1){
  2054. sBarPosition='bottom';
  2055. };
  2056.  
  2057. this.sBarPosition=sBarPosition;
  2058. this.selectedClassName='pv-gallery-sidebar-thumb_selected-' + sBarPosition;
  2059.  
  2060.  
  2061. var sBarDirection='v';//垂直放置
  2062. var isHorizontal=false;
  2063. if(sBarPosition=='top' || sBarPosition=='bottom'){
  2064. sBarDirection='h';//水平放置
  2065. isHorizontal=true;
  2066. };
  2067. this.sBarDirection=sBarDirection;
  2068. this.isHorizontal=isHorizontal;
  2069.  
  2070. var classPrefix='pv-gallery-';
  2071. var validClass=[
  2072. 'head',
  2073.  
  2074. 'head-left-img-info',
  2075. 'head-left-img-info-description',
  2076. 'head-left-img-info-resolution',
  2077. 'head-left-img-info-count',
  2078. 'head-left-img-info-scaling',
  2079.  
  2080. 'head-command-close',
  2081. 'head-command-nextPage',
  2082. 'head-command-operate',
  2083. 'head-command-slide-show',
  2084. 'head-command-slide-show-button-inner',
  2085. 'head-command-slide-show-countdown',
  2086. 'head-command-collect',
  2087. 'head-command-exit-collection',
  2088.  
  2089. 'head-command-drop-list-category',
  2090. 'head-command-drop-list-others',
  2091. 'head-command-drop-list-share',
  2092. 'head-command-drop-list-slide-show',
  2093. 'head-command-drop-list-collect',
  2094. 'head-command-drop-list-search',
  2095.  
  2096. 'body',
  2097.  
  2098. 'img-container',
  2099.  
  2100. 'img-scrollbar-h',
  2101. 'img-scrollbar-h-handle',
  2102. 'img-scrollbar-h-track',
  2103.  
  2104. 'img-scrollbar-v',
  2105. 'img-scrollbar-v-handle',
  2106. 'img-scrollbar-v-track',
  2107.  
  2108. 'thumb-scrollbar-h',
  2109. 'thumb-scrollbar-h-handle',
  2110. 'thumb-scrollbar-h-track',
  2111.  
  2112. 'thumb-scrollbar-v',
  2113. 'thumb-scrollbar-v-handle',
  2114. 'thumb-scrollbar-v-track',
  2115.  
  2116. 'img-content',
  2117. 'img-parent',
  2118. 'img_broken',
  2119.  
  2120. 'img-controler-pre',
  2121. 'img-controler-next',
  2122.  
  2123. 'sidebar-toggle',
  2124. 'sidebar-toggle-content',
  2125. 'sidebar-viewmore',
  2126. 'sidebar-viewmore-content',
  2127. 'maximize-container',
  2128.  
  2129. 'sidebar-container',
  2130. 'sidebar-content',
  2131.  
  2132. 'sidebar-controler-pre',
  2133. 'sidebar-controler-next',
  2134.  
  2135. 'sidebar-thumbnails-container',
  2136. ];
  2137.  
  2138. var eleMaps={};
  2139. this.eleMaps=eleMaps;
  2140.  
  2141. validClass.forEach(function(c){
  2142. eleMaps[c]=container.querySelector('.'+ classPrefix + c);
  2143. });
  2144.  
  2145. var posClass=[//需要添加'top bottom left right'class的元素
  2146. 'img-container',
  2147. 'sidebar-toggle',
  2148. 'sidebar-viewmore',
  2149. 'sidebar-container',
  2150. 'sidebar-thumbnails-container',
  2151. ];
  2152. posClass.forEach(function(c){
  2153. eleMaps[c].classList.add(classPrefix + c + '-' +sBarPosition);
  2154. });
  2155.  
  2156. var hvClass=[//需要添加'v h'class的元素
  2157. 'sidebar-toggle',
  2158. 'sidebar-toggle-content',
  2159. 'sidebar-viewmore',
  2160. 'sidebar-viewmore-content',
  2161. 'sidebar-container',
  2162. 'sidebar-content',
  2163. 'sidebar-controler-pre',
  2164. 'sidebar-controler-next',
  2165. 'sidebar-thumbnails-container',
  2166. ];
  2167. hvClass.forEach(function(c){
  2168. eleMaps[c].classList.add(classPrefix + c + '-' + sBarDirection);
  2169. });
  2170.  
  2171.  
  2172.  
  2173. //图片区域水平方向的滚动条
  2174. var imgScrollbarH=new this.Scrollbar({
  2175. bar:eleMaps['img-scrollbar-h'],
  2176. handle:eleMaps['img-scrollbar-h-handle'],
  2177. track:eleMaps['img-scrollbar-h-track'],
  2178. },eleMaps['img-content'],true);
  2179. this.imgScrollbarH=imgScrollbarH;
  2180.  
  2181. //图片区域垂直方向的滚动条
  2182. var imgScrollbarV=new this.Scrollbar({
  2183. bar:eleMaps['img-scrollbar-v'],
  2184. handle:eleMaps['img-scrollbar-v-handle'],
  2185. track:eleMaps['img-scrollbar-v-track'],
  2186. },eleMaps['img-content'],false);
  2187. this.imgScrollbarV=imgScrollbarV;
  2188.  
  2189. //缩略图区域的滚动条
  2190. var thumbScrollbar;
  2191. if(isHorizontal){
  2192. thumbScrollbar=new this.Scrollbar({
  2193. bar:eleMaps['thumb-scrollbar-h'],
  2194. handle:eleMaps['thumb-scrollbar-h-handle'],
  2195. track:eleMaps['thumb-scrollbar-h-track'],
  2196. },eleMaps['sidebar-thumbnails-container'],true);
  2197. }else{
  2198. thumbScrollbar=new this.Scrollbar({
  2199. bar:eleMaps['thumb-scrollbar-v'],
  2200. handle:eleMaps['thumb-scrollbar-v-handle'],
  2201. track:eleMaps['thumb-scrollbar-v-track'],
  2202. },eleMaps['sidebar-thumbnails-container'],false);
  2203. };
  2204. this.thumbScrollbar=thumbScrollbar;
  2205.  
  2206. var self=this;
  2207.  
  2208. var imgStatistics={//图片的总类,统计,初始化值
  2209. rule:{
  2210. shown:true,
  2211. count:0,
  2212. description:i18n("advancedRulesTip"),
  2213. name:i18n("advancedRules"),
  2214. },
  2215. tpRule:{
  2216. shown:true,
  2217. count:0,
  2218. description:i18n("tpRulesTip"),
  2219. name:i18n("tpRules"),
  2220. },
  2221. scale:{
  2222. shown:true,
  2223. count:0,
  2224. description:i18n("scaleRulesTip"),
  2225. name:i18n("scaleRules"),
  2226. },
  2227. force:{
  2228. shown:true,
  2229. count:0,
  2230. description:i18n("noScaleRulesTip"),
  2231. name:i18n("noScaleRules"),
  2232. },
  2233.  
  2234. // new
  2235. // scaleZoomResized: {
  2236. // shown: false,
  2237. // count: 0,
  2238. // description: '缩放的图片,图片尺寸最少相差比例 ' + prefs.gallery.zoomresized + '%',
  2239. // name: '小缩放'
  2240. // },
  2241. scaleSmall: {
  2242. shown: prefs.gallery.showSmallSize,
  2243. count: 0,
  2244. description: i18n("smallRulesTip",prefs.gallery.scaleSmallSize),
  2245. name: i18n("smallRules")
  2246. },
  2247. };
  2248. this.imgStatistics=imgStatistics;
  2249.  
  2250. //生成分类下拉列表
  2251. var typeMark='';
  2252. var imgStatistics_i;
  2253. for(var i in imgStatistics){
  2254. if(!imgStatistics.hasOwnProperty(i))continue;
  2255. imgStatistics_i=imgStatistics[i];
  2256. typeMark+=
  2257. '<span class="pv-gallery-head-command-drop-list-item" title="'+imgStatistics_i.description+'">'+
  2258. '<input type="checkbox" data-type="'+i+'" id="pv-gallery-head-command-drop-list-item-category-'+i+'" />'+
  2259. '<label for="pv-gallery-head-command-drop-list-item-category-'+i+'">'+imgStatistics_i.name+'</label>'+
  2260. '</span>';
  2261. };
  2262. eleMaps['head-command-drop-list-category'].innerHTML=typeMark;
  2263.  
  2264.  
  2265. //收藏相关
  2266. var collection={
  2267. getMatched:function(){
  2268. return (this.all || this.get())._find(function(value,index){
  2269. if(value.src==self.src){
  2270. return true;
  2271. };
  2272. });
  2273. },
  2274. check:function(){
  2275. //从缓存数据中检查。
  2276. var matched=this.getMatched();
  2277. this.favorite=matched? matched[0] : null;
  2278.  
  2279. this.tAreaValue();
  2280. this.highLight();
  2281. },
  2282. tAreaValue:function(){
  2283. this.textArea.value=this.favorite? this.favorite.description : self.eleMaps['head-left-img-info-description'].textContent;
  2284. },
  2285. highLight:function(){
  2286. eleMaps['head-command-collect'].classList[this.favorite? 'add' : 'remove']('pv-gallery-head-command-collect-favorite');
  2287. },
  2288. add:function(){
  2289. this.favorite={
  2290. src:self.src,
  2291. thumbSrc:dataset(self.relatedThumb,'thumbSrc'),
  2292. naturalSize:self.imgNaturalSize,
  2293. description:this.textArea.value,
  2294. };
  2295.  
  2296. //为了防止多个页面同时的储存,添加前,先载入最新的数据。
  2297. this.get();
  2298. //检查是否已经在里面了
  2299. var matched=this.getMatched();
  2300.  
  2301. if(matched){//如果已经存在,删除旧的。
  2302. this.all.splice(matched[1],1);
  2303. };
  2304. this.all.unshift(this.favorite);//添加到最前面。
  2305. this.highLight();
  2306. this.save();
  2307. },
  2308. remove:function(){
  2309. //获得最新数据
  2310. this.get();
  2311. //检查是否已经在里面了
  2312. var matched=this.getMatched();
  2313. if(matched){
  2314. this.all.splice(matched[1],1);
  2315. this.save();
  2316. };
  2317. this.favorite=null;
  2318. this.highLight();
  2319. },
  2320. save:function(){
  2321. storage.setItem('pv_collection',encodeURIComponent(JSON.stringify(this.all)));
  2322. },
  2323. get:function(){
  2324. var ret=storage.getItem('pv_collection') || '[]';
  2325. try{
  2326. ret=JSON.parse(decodeURIComponent(ret));
  2327. }catch(e){
  2328. ret=[];
  2329. };
  2330. this.all=ret;
  2331. return ret;
  2332. },
  2333. enter:function(){
  2334.  
  2335. if(this.all.length==0){
  2336. GM_notification(i18n("noCollectionYet"));
  2337. return;
  2338. };
  2339.  
  2340. this.mMode=true;
  2341. var button=this.dropListButton;
  2342. button.textContent=i18n("exitCollection");
  2343. dataset(button,'command','exitCollection');
  2344. this.headButton.style.display='inline-block';
  2345. eleMaps['sidebar-thumbnails-container'].classList.add('pv-gallery-sidebar-thumbnails_hide-span');
  2346.  
  2347. //生成dom
  2348. var container=document.createElement('span');
  2349.  
  2350. this.container=container;
  2351.  
  2352. var data_i;
  2353. var spanMark='';
  2354. var i=0;
  2355. while(data_i=this.all[i++]){
  2356. spanMark +=
  2357. '<span class="pv-gallery-sidebar-thumb-container" '+
  2358. ' data-natural-size="' + JSON.stringify(data_i.naturalSize).replace(/"/g,'&quot;') +
  2359. '" data-src="' + data_i.src +
  2360. '" data-thumb-src="' + data_i.thumbSrc +
  2361. '">'+
  2362. '<span class="pv-gallery-vertical-align-helper"></span>'+
  2363. '<span class="pv-gallery-sidebar-thumb-loading" title="'+i18n("loading")+'......"></span>'+
  2364. '</span>';
  2365. };
  2366. container.innerHTML=spanMark;
  2367. eleMaps['sidebar-thumbnails-container'].appendChild(container);
  2368.  
  2369.  
  2370. this.selected=self.selected;//备份
  2371.  
  2372. self.select(container.children[0]);
  2373. self.thumbScrollbar.reset();
  2374. self.loadThumb();
  2375. },
  2376. exit:function(){
  2377. if(!this.mMode)return;
  2378.  
  2379. this.mMode=false;
  2380. var button=this.dropListButton;
  2381. button.textContent=i18n("viewCollection");
  2382. dataset(button,'command','enterCollection');
  2383. this.headButton.style.display='none';
  2384. eleMaps['sidebar-thumbnails-container'].removeChild(this.container);
  2385. eleMaps['sidebar-thumbnails-container'].classList.remove('pv-gallery-sidebar-thumbnails_hide-span');
  2386.  
  2387. self.select(this.selected);
  2388. self.thumbScrollbar.reset();
  2389. self.loadThumb();
  2390. },
  2391. textArea:eleMaps['head-command-drop-list-collect'].querySelector('textarea'),
  2392. dropListButton:eleMaps['head-command-drop-list-others'].querySelector('[data-command$="Collection"]'),
  2393. headButton:eleMaps['head-command-exit-collection'],
  2394. };
  2395.  
  2396. this.collection=collection;
  2397.  
  2398. eleMaps['head-command-drop-list-collect'].addEventListener('input',function(e){
  2399. var target=e.target;
  2400. if(!collection.favorite)return;
  2401. collection.favorite[dataset(target,'prefs')]=target.value;
  2402. clearTimeout(collection.saveTimer);
  2403. collection.saveTimer=setTimeout(function(){
  2404. collection.save();
  2405. },500);
  2406. },true);
  2407.  
  2408.  
  2409. var slideShow={
  2410. opts:{
  2411. interval:5000,
  2412. wait:true,
  2413. backward:false,
  2414. skipErrorImg:true,
  2415. run:false,
  2416. },
  2417. //timing:
  2418. //select(选中下一个图片后(缩略图栏选中了),还没开始读取大图(一般选中后,延时200ms开始读取大图)),
  2419. //loadEnd(当前显示图片已经读取完成后),
  2420. //click(点击按钮),
  2421. //change(改变设置)
  2422. run:function(timing){
  2423. if(!this.opts.run)return;
  2424.  
  2425. if(timing!='loadEnd'){
  2426. this.stop();
  2427. };
  2428.  
  2429. if(timing=='click' || timing=='select'){
  2430. if(!this.getEle()){//没有要切换到的图片了,停止
  2431. this.exit();
  2432. return;
  2433. };
  2434. };
  2435.  
  2436. if(this.opts.skipErrorImg){
  2437. if(self.imgError && !self.isLoading){//确保是当前图片和选中缩略图一致的时候
  2438. self.select(this.getEle());
  2439. return;
  2440. };
  2441. };
  2442.  
  2443.  
  2444. if(this.opts.wait){
  2445. if(timing!='select' && (timing=='loadEnd' || (!self.isLoading && (self.img.complete || self.imgError)))){
  2446. this.go();
  2447. };
  2448. }else{
  2449. if(timing!='loadEnd'){
  2450. this.go();
  2451. };
  2452. };
  2453.  
  2454. },
  2455. getEle:function(){
  2456. return self.getThumSpan(this.opts.backward)
  2457. },
  2458. go:function(){
  2459. this.stop();//停止上次的。
  2460. var interval=this.opts.interval;
  2461. var _self=this;
  2462. this.timer=setTimeout(function(){
  2463. _self.setCountdown(0);
  2464. clearInterval(_self.countdownTimer);
  2465. self.select(_self.getEle());
  2466. },interval);
  2467.  
  2468. var startTime=Date.now();
  2469. this.countdownTimer=setInterval(function(){
  2470. _self.setCountdown(interval - (Date.now()-startTime));
  2471. },100);
  2472. },
  2473. stop:function(){
  2474. this.setCountdown(this.opts.interval);
  2475. clearTimeout(this.timer);
  2476. clearInterval(this.countdownTimer);
  2477. },
  2478. exit:function(){
  2479. this.opts.run=true;
  2480. this.switchStatus();
  2481. this.stop();
  2482. },
  2483. setCountdown:function(value){
  2484. eleMaps['head-command-slide-show-countdown'].textContent=(value/1000).toFixed(2);
  2485. },
  2486. switchStatus:function(){
  2487. this.opts.run=!this.opts.run;
  2488. eleMaps['head-command-slide-show-button-inner'].classList[this.opts.run? 'add' : 'remove']('pv-gallery-head-command-slide-show-button-inner_stop');
  2489. },
  2490. check:function(){
  2491. this.opts.run? this.run('click') : this.stop();
  2492. },
  2493. };
  2494.  
  2495. slideShow.setCountdown(slideShow.opts.interval);;
  2496. this.slideShow=slideShow;
  2497.  
  2498. //幻灯片播放下拉列表change事件的处理
  2499. eleMaps['head-command-drop-list-slide-show'].addEventListener('change',function(e){
  2500. var target=e.target;
  2501. var value;
  2502. var prefs=dataset(target,'prefs');
  2503. if(target.type=='checkbox'){
  2504. value=target.checked;
  2505. }else{
  2506. value=parseFloat(target.value);
  2507. if(isNaN(value)){//无效
  2508. value=slideShow.opts[prefs] / 1000;
  2509. };
  2510. value=value>0 ? value : 1;
  2511. target.value=value;
  2512. value *= 1000;
  2513. };
  2514. slideShow.opts[prefs]=value;
  2515. slideShow.run('change');
  2516. },true);
  2517.  
  2518.  
  2519. //分类下拉列表的点击发生change事件的处理
  2520. eleMaps['head-command-drop-list-category'].addEventListener('change',function(e){
  2521. var target=e.target;
  2522. var type=dataset(target,'type');
  2523. self.iStatisCopy[type].shown=target.checked;
  2524. self.switchThumbVisible();//切换图片类别显隐;
  2525. },true);
  2526.  
  2527.  
  2528. var srcSplit,downloading=false;
  2529. //命令下拉列表的点击处理
  2530. eleMaps['head-command-drop-list-others'].addEventListener('click',function(e){
  2531. if(e.button!=0)return;//左键
  2532. var target=e.target;
  2533. var command=dataset(target,'command');
  2534. if(!command)return;
  2535. switch(command){
  2536. case 'openInNewWindow':{
  2537. window.open(self.src,'_blank');
  2538. }break;
  2539. case 'psImage':{
  2540. //window.open((prefs.gallery.editSite=='Pixlr'?'https://pixlr.com/editor/?image=':'https://www.toolpic.com/apieditor.html?image=')+self.src,'_blank');
  2541. window.open('https://www.lunapic.com/editor/index.php?action=url&url='+self.src,'_blank');
  2542. }break;
  2543. case 'scrollIntoView':{
  2544. if(collection.mMode){
  2545. GM_notification(i18n("inCollection"));
  2546. return;
  2547. };
  2548. var relatedThumb=self.relatedThumb;
  2549. var index=arrayFn.indexOf.call(self.imgSpans,relatedThumb);
  2550. var targetImg=self.data[index].img;
  2551.  
  2552. if(targetImg){
  2553. if(!document.documentElement.contains(targetImg) || unsafeWindow.getComputedStyle(targetImg).display=='none'){//图片不存在文档中,或者隐藏了。
  2554. GM_notification(i18n("cantFind"));
  2555. return;
  2556. };
  2557. self.minimize();
  2558. setTimeout(function(){
  2559. self.navigateToImg(targetImg);
  2560. flashEle(targetImg);
  2561. },0);
  2562.  
  2563. }else{//frame发送过来的时候删除了不能传送的图片
  2564.  
  2565. document.addEventListener('pv-navigateToImg',function(e){
  2566. if(!e.detail){
  2567. GM_notification(i18n("cantFind"));
  2568. return;
  2569. };
  2570. self.minimize();
  2571. setTimeout(function(){//将frame滚动到中间位置
  2572. if(self.iframe){
  2573. self.navigateToImg(self.iframe);
  2574. };
  2575. },0);
  2576. },true);
  2577. window.postMessage({//问问frame。。
  2578. messageID:messageID,
  2579. command:'navigateToImg',
  2580. index:index,
  2581. to:self.from,
  2582. },'*');
  2583. };
  2584.  
  2585. }break;
  2586. case 'exportImages':
  2587. self.exportImages();
  2588. break;
  2589. case 'downloadImage':
  2590. if(downloading)break;
  2591. downloading=true;
  2592. var nodes = document.querySelectorAll('.pv-gallery-sidebar-thumb-container[data-src]');
  2593. var saveParams = [],saveIndex=0;
  2594. [].forEach.call(nodes, function(node){
  2595. if(unsafeWindow.getComputedStyle(node).display!="none"){
  2596. saveIndex++;
  2597. srcSplit=node.dataset.src.replace(/[\?#].*/,"").split("/");
  2598. srcSplit=srcSplit[srcSplit.length-1];
  2599. if(srcSplit.length>30 && (srcSplit.indexOf(".")==-1 || /[&\?=,]/i.test(srcSplit))){
  2600. srcSplit="";
  2601. }
  2602. var picName=document.title + "-" + (saveIndex<10?"00"+saveIndex:(saveIndex<100?"0"+saveIndex:saveIndex)) + (node.title?"-" + node.title:"") + "-" + srcSplit,hostArr=location.host.split(".");
  2603. var host=hostArr[hostArr.length-2];
  2604. saveParams.push([node.dataset.src, picName]);
  2605. //saveAs(node.dataset.src, location.host+"-"+srcSplit[srcSplit.length-1]);
  2606. }
  2607. });
  2608. if(prefs.gallery.downloadWithZip){
  2609. GM_notification(i18n("galleryDownloadWithZipAlert"));
  2610. var zip = new JSZip(),downloaded=0;
  2611. var fileName = document.title + ".zip";
  2612. for(let i=0; i<saveParams.length; i++){
  2613. self.dataURLToCanvas(saveParams[i][0], canvas=>{
  2614. if(!canvas){
  2615. downloaded++;
  2616. if(downloaded == saveParams.length){
  2617. zip.generateAsync({type:"blob"}).then(function(content){
  2618. saveAs(content, fileName);
  2619. downloading=false;
  2620. })
  2621. }
  2622. return;
  2623. }
  2624. canvas.toBlob(blob=>{
  2625. zip.file(saveParams[i][1].replace(/\.[^\.]+$/,"")+'.jpg',blob);
  2626. downloaded++;
  2627. if(downloaded == saveParams.length){
  2628. zip.generateAsync({type:"blob"}).then(function(content){
  2629. saveAs(content, fileName);
  2630. downloading=false;
  2631. })
  2632. }
  2633. }, "image/jpg");
  2634. });
  2635. };
  2636. break;
  2637. }
  2638.  
  2639. let download5Times=function(){
  2640. for(let i=0;i<5;i++){
  2641. let saveParam=saveParams.shift();
  2642. if(saveParam){
  2643. GM_download(saveParam[0], saveParam[1]);
  2644. }else{
  2645. downloading=false;
  2646. break;
  2647. }
  2648. }
  2649. if(saveParams.length>0){
  2650. setTimeout(()=>{
  2651. download5Times();
  2652. },1000);
  2653. }
  2654. };
  2655. download5Times();
  2656. break;
  2657. case 'copyImages':
  2658. self.copyImages(true);
  2659. break;
  2660. case 'scrollToEndAndReload':
  2661. var checkbox = target.parentNode.querySelector("input");
  2662. if(target.nodeName=="LABEL"){
  2663. checkbox.checked = !checkbox.checked;
  2664. }
  2665.  
  2666. prefs.gallery.scrollEndAndLoad = checkbox.checked;
  2667. break;
  2668. case 'fullScreen':
  2669. if (target.classList.contains('fullscreenbtn')) {
  2670. if (cancelFullScreen()) return;
  2671. target.textContent = i18n("enterFullsc");
  2672. target.classList.remove('fullscreenbtn');
  2673. return;
  2674. }
  2675.  
  2676. if (launchFullScreen(document.documentElement)) return;
  2677. target.classList.toggle('fullscreenbtn');
  2678. target.textContent = i18n("exitFullsc");
  2679. target.classList.add('fullscreenbtn');
  2680. break;
  2681. case 'openPrefs':
  2682. openPrefs();
  2683. break;
  2684. case 'enterCollection':{
  2685. //进入管理模式
  2686. collection.enter();
  2687. }break;
  2688. case 'exitCollection':{
  2689. //退出管理模式
  2690. collection.exit();
  2691. }break;
  2692. };
  2693. },true);
  2694.  
  2695. // 监视全屏的变化
  2696. function fullScreenChanged() {
  2697. if (!document.fullscreenElement && // alternative standard method
  2698. !document.mozFullScreenElement &&
  2699. !document.webkitFullscreenElement &&
  2700. !document.msFullscreenElement) {
  2701.  
  2702. var btn = document.getElementById("pv-gallery-fullscreenbtn");
  2703. if (btn) {
  2704. btn.textContent = i18n("enterFullsc");
  2705. btn.removeClass('fullscreenbtn');
  2706. }
  2707. }
  2708. }
  2709. document.addEventListener('webkitfullscreenchange', fullScreenChanged, false);
  2710. document.addEventListener('mozfullscreenchange', fullScreenChanged, false);
  2711. document.addEventListener('fullscreenchange', fullScreenChanged, false);
  2712.  
  2713. //生成分享的下拉列表
  2714. var shareMark='';
  2715. var shareItem;
  2716. for(var i in prefs.share){
  2717. if(!prefs.share.hasOwnProperty(i))continue;
  2718. shareItem=prefs.share[i];
  2719. if(shareItem.limitLang && shareItem.limitLang.indexOf(lang)==-1)continue;
  2720. if(shareItem.disabled)continue;
  2721. shareMark+=(
  2722. '<span class="pv-gallery-head-command-drop-list-item" data-site="'+i+'" style="\
  2723. background-image:url(\''+ shareItem.icon +'\');\
  2724. background-position:4px center;\
  2725. background-repeat:no-repeat;\
  2726. padding-left:24px;">'+shareItem.name+'</span>');
  2727. };
  2728.  
  2729. eleMaps['head-command-drop-list-share'].innerHTML=shareMark;
  2730.  
  2731. //分享下拉列表的点击处理
  2732. eleMaps['head-command-drop-list-share'].addEventListener('click',function(e){
  2733. if(e.button!=0)return;//左键
  2734. var target=e.target;
  2735. var site=dataset(target,'site');
  2736. if(!site)return;
  2737. var site_info=prefs.share[site];
  2738. var param=site_info.api.call(self.img,{
  2739. title:encodeURIComponent(document.title),
  2740. pic:encodeURIComponent(self.src),
  2741. url:encodeURIComponent(location.href),
  2742. });
  2743. if(!param)return;
  2744. 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');
  2745. },true);
  2746.  
  2747. eleMaps['head'].addEventListener('click',function(e){
  2748. if(e.target.className.indexOf('pv-gallery-head-command')!=-1)
  2749. self.closeViewMore();
  2750. });
  2751.  
  2752. if(!prefs.gallery.searchData)prefs.gallery.searchData=defaultSearchData;
  2753. var searchRules=prefs.gallery.searchData.split("\n"),searchUploadUrl,searchItems=[];
  2754. var searchAll=eleMaps['head-command-drop-list-search'].querySelector("#headSearchAll");
  2755. searchRules.forEach(rule=>{
  2756. if(!searchUploadUrl){
  2757. var uploadMatch=rule.match(/\s*{(.*)}\s*/);//todo: upload 2 search, need a Long-Term Servicing server for base64 to url like ainoob.com/api/uploadImage/
  2758. if(uploadMatch){
  2759. searchUploadUrl=uploadMatch[1];
  2760. return;
  2761. }
  2762. }
  2763. let ruleArr=rule.trim().split("|");
  2764. if(ruleArr.length==2){
  2765. var item=document.createElement('span');
  2766. item.className="pv-gallery-head-command-drop-list-item";
  2767. item.innerHTML=ruleArr[0];
  2768. item.addEventListener('click',function(e){
  2769. window.open(ruleArr[1].replace("#t#", encodeURIComponent(self.src)), "_blank", "width=1024, height=768, toolbar=1");
  2770. });
  2771. searchItems.push(item);
  2772. eleMaps['head-command-drop-list-search'].insertBefore(item, searchAll);
  2773. }
  2774. });
  2775. searchAll.addEventListener('click',function(e){
  2776. searchItems.forEach(item=>{item.click()});
  2777. });
  2778.  
  2779.  
  2780.  
  2781. var loadThumbsTimer;
  2782. eleMaps['sidebar-thumbnails-container'].addEventListener('scroll',function(e){//发生scroll事件时加载缩略图
  2783. clearTimeout(loadThumbsTimer);//加个延时,在连续触发的时候缓一缓。
  2784. loadThumbsTimer=setTimeout(function(){
  2785. self.loadThumb();
  2786. },200);
  2787. },false);
  2788.  
  2789. var canScroll=true;
  2790. var scrollToChange=function(next){
  2791. if(canScroll){
  2792. if(prefs.gallery.transition){
  2793. canScroll=false;
  2794. setTimeout(function(){
  2795. canScroll=true;
  2796. },500);
  2797. }
  2798. next ? self.selectNext() : self.selectPrevious();
  2799. }
  2800. }
  2801. addWheelEvent(eleMaps['body'],function(e){//wheel事件
  2802. if(e.deltaZ!=0)return;//z轴
  2803. if(eleMaps['sidebar-toggle'].style.visibility == 'hidden')return;
  2804. var target=e.target;
  2805. //e.preventDefault();
  2806. if(eleMaps['sidebar-container'].contains(target)){//缩略图区滚动滚轮翻图片
  2807. var distance=self.thumbSpanOuterSize;
  2808.  
  2809. if(e.deltaY<0 || e.deltaX<0){//向上滚
  2810. distance=-distance;
  2811. };
  2812. thumbScrollbar.scrollBy(distance)
  2813. }else{//图片区域滚动
  2814. var distance=100;
  2815. if(e.deltaY!=0){//y轴
  2816. if(self.img && self.img.classList.contains('pv-gallery-img_zoom-out')){//图片可以缩小时,滚动图片,否则切换图片。
  2817. if(e.deltaY < 0){
  2818. distance=-distance;
  2819. };
  2820. if(eleMaps['img-scrollbar-h'].contains(target)){//如果在横向滚动条上。
  2821. imgScrollbarH.scrollBy(distance);
  2822. }else{
  2823. if(imgScrollbarV.scrollBy(distance) && prefs.gallery.scrollEndToChange){
  2824. scrollToChange(e.deltaY > 0);
  2825. }
  2826. };
  2827. }else{
  2828. scrollToChange(e.deltaY > 0);
  2829. };
  2830. }else{//x轴
  2831. if(e.deltaX < 0){
  2832. distance=-distance;
  2833. };
  2834. imgScrollbarH.scrollBy(distance);
  2835. };
  2836. };
  2837. },true);
  2838.  
  2839.  
  2840. //focus,blur;
  2841. addCusMouseEvent('mouseenter',container,function(){
  2842. this.focus();
  2843. });
  2844. addCusMouseEvent('mouseleave',container,function(){
  2845. this.blur();
  2846. });
  2847.  
  2848. var lastX,lastY;
  2849. const minLength=10000,tg=0.5;
  2850. function tracer(e) {
  2851. let curX=e.changedTouches[0].clientX;
  2852. let curY=e.changedTouches[0].clientY;
  2853. let distanceX=curX-lastX,distanceY=curY-lastY;
  2854. let distance=distanceX*distanceX+distanceY*distanceY;
  2855. if (distance>minLength) {
  2856. lastX=curX;
  2857. lastY=curY;
  2858. let direction="";
  2859. let slope=Math.abs(distanceY/distanceX);
  2860. if(slope>tg){
  2861. if(distanceY>0) {
  2862. direction="↓";
  2863. }else{
  2864. direction="↑";
  2865. }
  2866. }else if(slope<=1/tg) {
  2867. if(distanceX>0) {
  2868. direction="→";
  2869. }else{
  2870. direction="←";
  2871. }
  2872. }
  2873. switch(direction){
  2874. case "←":
  2875. self.selectNext();
  2876. break;
  2877. case "→":
  2878. self.selectPrevious();
  2879. break;
  2880. case "↑":
  2881. if(self.eleMaps['sidebar-toggle'].style.visibility != 'hidden'){
  2882. self.maximizeSidebar();
  2883. }
  2884. break;
  2885. default:
  2886. break;
  2887. }
  2888. }
  2889. }
  2890. self.eleMaps['img-content'].addEventListener('touchstart',function(e){
  2891. lastX=e.changedTouches[0].clientX;
  2892. lastY=e.changedTouches[0].clientY;
  2893. self.eleMaps['img-content'].addEventListener('touchmove',tracer);
  2894. });
  2895. self.eleMaps['img-content'].addEventListener('touchend',function(e){
  2896. self.eleMaps['img-content'].removeEventListener('touchmove',tracer);
  2897. });
  2898.  
  2899. //上下左右切换图片,空格键模拟滚动一页
  2900.  
  2901. var validKeyCode=[38,39,40,37,32,9]//上右下左,32空格,tab禁止焦点切换。
  2902. var keyDown;
  2903.  
  2904. container.addEventListener('keydown',function(e){
  2905. var keyCode=e.keyCode;
  2906. var index=validKeyCode.indexOf(keyCode);
  2907. if(index==-1)return;
  2908.  
  2909. var target=e.target;
  2910.  
  2911. if(!container.contains(target))return;//触发焦点不再gallery里面。
  2912. e.preventDefault();
  2913.  
  2914. if(keyCode==9)return;//tab键
  2915. if(keyCode==32){//32空格,模拟滚动一页
  2916. imgScrollbarV.scrollByPages(1);
  2917. return;
  2918. };
  2919.  
  2920. if(keyDown)return;//已按下。
  2921. keyDown=true;
  2922.  
  2923. var stop;
  2924. switch(index){
  2925. case 0:;
  2926. case 3:{
  2927. self.selectPrevious();
  2928. stop=self.simpleSlideShow(true);
  2929. }break;
  2930. case 1:;
  2931. case 2:{
  2932. self.selectNext();
  2933. stop=self.simpleSlideShow();
  2934. }break;
  2935. };
  2936.  
  2937. function keyUpHandler(e){
  2938. if(e.keyCode!=validKeyCode[index])return;
  2939. container.removeEventListener('keyup',keyUpHandler,false);
  2940. keyDown=false;
  2941. stop();
  2942. };
  2943. container.addEventListener('keyup',keyUpHandler,false);
  2944.  
  2945. },true);
  2946.  
  2947.  
  2948. var imgDraged;
  2949. eleMaps['img-parent'].addEventListener('mousedown',function(e){//如果图片尺寸大于屏幕的时候按住图片进行拖移
  2950. var target=e.target;
  2951. if(e.button!=0 || target.nodeName!='IMG')return;
  2952. var bigger=target.classList.contains('pv-gallery-img_zoom-out');//如果是大于屏幕
  2953.  
  2954. var oClient={
  2955. x:e.clientX,
  2956. y:e.clientY,
  2957. };
  2958.  
  2959. var oScroll={
  2960. left:self.imgScrollbarH.getScrolled(),
  2961. top:self.imgScrollbarV.getScrolled(),
  2962. };
  2963.  
  2964. var moveFiredCount=0;
  2965. var moveHandler=function(e){
  2966. moveFiredCount++;
  2967. if(moveFiredCount<2){//给个缓冲。。
  2968. return;
  2969. };
  2970. imgDraged=true;
  2971. if(bigger){
  2972. target.style.cursor= support.cssCursorValue.grabbing || 'pointer';
  2973. self.imgScrollbarV.scroll(oScroll.top-(e.clientY-oClient.y));
  2974. self.imgScrollbarH.scroll(oScroll.left-(e.clientX-oClient.x));
  2975. };
  2976. };
  2977.  
  2978. var upHandler=function(){
  2979. target.style.cursor='';
  2980.  
  2981. //拖曳之后阻止随后可能产生click事件产生的大小切换。
  2982. //确保在随后的click事件发生后执行
  2983. setTimeout(function(){
  2984. imgDraged=false;
  2985. },0);
  2986.  
  2987. document.removeEventListener('mousemove',moveHandler,true);
  2988. document.removeEventListener('mouseup',upHandler,true);
  2989. };
  2990.  
  2991. document.addEventListener('mousemove',moveHandler,true);
  2992. document.addEventListener('mouseup',upHandler,true);
  2993. },true);
  2994.  
  2995. eleMaps['img-parent'].addEventListener('click',function(e){//点击图片本身就行图片缩放处理
  2996. var target=e.target;
  2997. if(e.button!=0 || target.nodeName!='IMG')return;
  2998.  
  2999. if(imgDraged){//在拖动后触发的click事件,取消掉。免得一拖动完就立即进行的缩放。。。
  3000. imgDraged=false;
  3001. return;
  3002. };
  3003.  
  3004. if(target.classList.contains('pv-gallery-img_zoom-in')){//放大
  3005. self.fitContains=false;
  3006. var zoomX = typeof e.offsetX=='undefined' ? e.layerX : e.offsetX;
  3007. var zoomY = typeof e.offsetY=='undefined' ? e.layerY : e.offsetY;
  3008. var scaleX=zoomX/target.offsetWidth;
  3009. var scaleY=zoomY/target.offsetHeight;
  3010. self.fitToScreen({
  3011. x:scaleX,
  3012. y:scaleY,
  3013. });
  3014. }else if(target.classList.contains('pv-gallery-img_zoom-out')){
  3015. self.fitContains=true;
  3016. self.fitToScreen();
  3017. };
  3018. },true);
  3019.  
  3020.  
  3021. container.addEventListener('mousedown',function(e){//鼠标按在导航上,切换图片
  3022. if(e.button!=0)return;//左键
  3023. var target=e.target;
  3024. if(target.nodeName=='IMG')e.preventDefault();
  3025.  
  3026. var matched=true;
  3027. var stop;
  3028. switch(target){
  3029. case eleMaps['img-controler-pre']:;
  3030. case eleMaps['sidebar-controler-pre']:{//上一个
  3031. self.selectPrevious();
  3032. stop=self.simpleSlideShow(true);
  3033. }break;
  3034. case eleMaps['img-controler-next']:;
  3035. case eleMaps['sidebar-controler-next']:{//下一个
  3036. self.selectNext();
  3037. stop=self.simpleSlideShow();
  3038. }break;
  3039. default:{
  3040. matched=false;
  3041. }break;
  3042. };
  3043.  
  3044. function mouseUpHandler(e){
  3045. document.removeEventListener('mouseup',mouseUpHandler,true);
  3046. stop();
  3047. };
  3048.  
  3049. if(matched){
  3050. e.preventDefault();
  3051. document.addEventListener('mouseup',mouseUpHandler,true);
  3052. };
  3053. },false);
  3054.  
  3055. eleMaps['sidebar-thumbnails-container'].addEventListener('click',function(e){//点击缩略图切换
  3056. if(e.button!=0)return;//左键
  3057. var target=e.target;
  3058. var targetP;
  3059. if(!dataset(target,'src') && (targetP=target.parentNode) && !dataset(targetP,'src'))return;
  3060.  
  3061. self.select(targetP? targetP : target);
  3062. },false);
  3063.  
  3064. //点击读取错误的图片占位符重新读取
  3065. eleMaps['img_broken'].addEventListener('click',function(e){
  3066. if(self.isLoading){
  3067. self.select(self.errorSpan);
  3068. }else{
  3069. self.getImg(self.errorSpan);
  3070. };
  3071. },false);
  3072.  
  3073.  
  3074. eleMaps['head'].addEventListener('click',function(e){//顶栏上面的命令
  3075. if(e.button!=0)return;
  3076. var target=e.target;
  3077. if(eleMaps['head-command-close']==target){
  3078. self.close();
  3079. }else if(eleMaps['head-command-operate'].contains(target)){
  3080. imgReady(self.src,{
  3081. ready:function(){
  3082. new ImgWindowC(this);
  3083. },
  3084. });
  3085. }else if(eleMaps['head-command-nextPage'].contains(target)){
  3086. var textSpan=eleMaps['head-command-nextPage'].querySelector("span");
  3087. if(textSpan.innerHTML==i18n("loading")){
  3088. textSpan.innerHTML=i18n("loadAll");
  3089. return;
  3090. }
  3091. textSpan.innerHTML=i18n("loading");
  3092. self.completePages=[];
  3093. self.pageAllReady=false;
  3094. self.nextPage();
  3095. }else if(eleMaps['head-command-collect'].contains(target)){
  3096. if(collection.favorite){
  3097. collection.remove();
  3098. }else{
  3099. collection.add();
  3100. };
  3101. }else if(eleMaps['head-command-exit-collection'].contains(target)){
  3102. collection.exit();
  3103. }else if(eleMaps['head-command-slide-show'].contains(target)){
  3104. slideShow.switchStatus();
  3105. slideShow.check();
  3106. };
  3107.  
  3108. },false);
  3109.  
  3110.  
  3111. //点击还原。
  3112. maximizeTrigger.addEventListener('click',function(e){
  3113. var target=e.target;
  3114. this.style.display='none';
  3115. if(target==this){
  3116. self.show();
  3117. self.resizeHandler();
  3118. }else{
  3119. self.minimized=false;
  3120. };
  3121. },true);
  3122.  
  3123.  
  3124. this._resizeHandler=this.resizeHandler.bind(this);
  3125. this._keyDownListener=this.keyDownListener.bind(this);
  3126. this._keyUpListener=this.keyUpListener.bind(this);
  3127.  
  3128. //插入动态生成的css数据。
  3129. this.globalSSheet.insertRule('.pv-gallery-sidebar-thumb-container{'+
  3130. ((isHorizontal ? 'width' : 'height') + ':' + (isHorizontal ? unsafeWindow.getComputedStyle(eleMaps['sidebar-thumbnails-container']).height : unsafeWindow.getComputedStyle(eleMaps['sidebar-thumbnails-container']).width)) +
  3131. '}',this.globalSSheet.cssRules.length);
  3132.  
  3133. this.forceRepaintTimes=0;
  3134.  
  3135. container.style.display='none';
  3136. this.shown=false;
  3137.  
  3138. // 我添加的部分
  3139. this.initToggleBar();
  3140. this.initZoom();
  3141. },
  3142.  
  3143. changeMinView:function(){
  3144. var sizeInputH=this.gallery.querySelector("#minsizeH");
  3145. var sizeInputW=this.gallery.querySelector("#minsizeW");
  3146. var sizeInputHSpan=this.gallery.querySelector("#minsizeHSpan");
  3147. var sizeInputWSpan=this.gallery.querySelector("#minsizeWSpan");
  3148. sizeInputH.title=sizeInputH.value+"px";
  3149. sizeInputHSpan.innerHTML=Math.floor(sizeInputH.value)+"px";
  3150. sizeInputW.title=sizeInputW.value+"px";
  3151. sizeInputWSpan.innerHTML=Math.floor(sizeInputW.value)+"px";
  3152.  
  3153. var viewmoreShow = this.eleMaps['sidebar-toggle'].style.visibility == 'hidden';
  3154. if(viewmoreShow){
  3155. var maxSizeH=0,minSizeH=0,maxSizeW=0,minSizeW=0;
  3156. [].forEach.call(document.querySelectorAll(".maximizeChild>img"),function(item){
  3157. var spanMark=document.querySelector("span.pv-gallery-sidebar-thumb-container[data-src='"+item.src+"']");
  3158. if(spanMark && !spanMark.dataset.naturalSize && item.naturalWidth && item.naturalHeight){
  3159. spanMark.dataset.naturalSize=JSON.stringify({w:item.naturalWidth,h:item.naturalHeight});
  3160. }
  3161. if(item.naturalWidth<sizeInputW.value || item.naturalHeight<sizeInputH.value){
  3162. item.parentNode.style.display="none";
  3163. if(spanMark)spanMark.style.display="none";
  3164. }else{
  3165. item.parentNode.style.display="";
  3166. if(spanMark)spanMark.style.display="";
  3167. }
  3168. if(item.naturalHeight>maxSizeH)
  3169. maxSizeH=item.naturalHeight;
  3170. if(item.naturalHeight<minSizeH || minSizeH==0)
  3171. minSizeH=item.naturalHeight;
  3172. if(item.naturalWidth>maxSizeW)
  3173. maxSizeW=item.naturalWidth;
  3174. if(item.naturalWidth<minSizeW || minSizeW==0)
  3175. minSizeW=item.naturalWidth;
  3176. });
  3177. sizeInputH.max=maxSizeH;
  3178. sizeInputH.min=minSizeH;
  3179. sizeInputH.title=sizeInputH.value+"px";
  3180. sizeInputHSpan.innerHTML=Math.floor(sizeInputH.value)+"px";
  3181.  
  3182. sizeInputW.max=maxSizeW;
  3183. sizeInputW.min=minSizeW;
  3184. sizeInputW.title=sizeInputW.value+"px";
  3185. sizeInputWSpan.innerHTML=Math.floor(sizeInputW.value)+"px";
  3186. }else{
  3187. this.data.forEach(function(item) {
  3188. if(!item)return;
  3189. var spanMark=document.querySelector("span.pv-gallery-sidebar-thumb-container[data-thumb-src='"+item.imgSrc+"']");
  3190. if(spanMark){
  3191. var naturalSize=spanMark.dataset.naturalSize,itemW=item.sizeW,itemH=item.sizeH;
  3192. if(naturalSize){
  3193. naturalSize=JSON.parse(naturalSize);
  3194. itemW=naturalSize.w;
  3195. itemH=naturalSize.h;
  3196. if(itemW>sizeInputW.max)sizeInputW.max=itemW;
  3197. if(itemH>sizeInputH.max)sizeInputH.max=itemH;
  3198. }else if(!item.noActual){
  3199. itemW=99999;
  3200. itemH=99999;
  3201. }
  3202. if(itemW<sizeInputW.value || itemH<sizeInputH.value){
  3203. spanMark.style.display="none";
  3204. }else{
  3205. spanMark.style.display="";
  3206. }
  3207. }
  3208. });
  3209. this.switchThumbVisible();
  3210. }
  3211. },
  3212. changeSizeInputReset:function(){
  3213. var maxSizeH=0,minSizeH=0,maxSizeW=0,minSizeW=0;
  3214. var sizeInputH=this.gallery.querySelector("#minsizeH");
  3215. var sizeInputW=this.gallery.querySelector("#minsizeW");
  3216. this.data.forEach(function(item) {
  3217. if(!item)return;
  3218. var itemW=item.sizeW,itemH=item.sizeH;
  3219. var spanMark=document.querySelector("span.pv-gallery-sidebar-thumb-container[data-src='"+item.src+"']");
  3220. if(spanMark){
  3221. var naturalSize=spanMark.dataset.naturalSize;
  3222. if(naturalSize){
  3223. naturalSize=JSON.parse(naturalSize);
  3224. itemW=naturalSize.w;
  3225. itemH=naturalSize.h;
  3226. }
  3227. }
  3228. if(itemH>maxSizeH)
  3229. maxSizeH=itemH;
  3230. if(itemH<minSizeH || minSizeH==0)
  3231. minSizeH=itemH;
  3232. if(itemW>maxSizeW)
  3233. maxSizeW=itemW;
  3234. if(itemW<minSizeW || minSizeW==0)
  3235. minSizeW=itemW;
  3236. });
  3237. sizeInputH.max=maxSizeH;
  3238. sizeInputH.min=minSizeH;
  3239. sizeInputH.value=prefs.gallery.defaultSizeLimit.h;
  3240. sizeInputH.title=sizeInputH.value+"px";
  3241. var sizeInputHSpan=this.gallery.querySelector("#minsizeHSpan");
  3242. sizeInputHSpan.innerHTML=Math.floor(sizeInputH.value)+"px";
  3243.  
  3244. sizeInputW.max=maxSizeW;
  3245. sizeInputW.min=minSizeW;
  3246. sizeInputW.value=prefs.gallery.defaultSizeLimit.w;
  3247. sizeInputW.title=sizeInputW.value+"px";
  3248. var sizeInputWSpan=this.gallery.querySelector("#minsizeWSpan");
  3249. sizeInputWSpan.innerHTML=Math.floor(sizeInputW.value)+"px";
  3250. },
  3251. initToggleBar: function() { // 是否显示切换 sidebar 按钮
  3252. /**
  3253. * TODO:仿造下面的链接重新改造过?
  3254. * http://image.baidu.com/detail/newindex?col=%E8%B5%84%E8%AE%AF&tag=%E4%BD%93%E8%82%B2&pn=0&pid=5123662821688142478&aid=&user_id=10086&setid=-1&sort=0&newsPn=4&star=&fr=hotword&from=1
  3255. */
  3256. if (prefs.gallery.sidebarToggle) {
  3257. var toggleBar = this.eleMaps['sidebar-toggle'];
  3258. toggleBar.style.display = 'block';
  3259. toggleBar.style.height = '16px';
  3260. toggleBar.addEventListener('click', this.showHideBottom.bind(this), false);
  3261.  
  3262. var viewmoreBar = this.eleMaps['sidebar-viewmore'];
  3263. viewmoreBar.style.display = 'block';
  3264. viewmoreBar.addEventListener('click', this.maximizeSidebar.bind(this), false);
  3265.  
  3266. // 顶部圆角
  3267. switch (prefs.gallery.sidebarPosition) {
  3268. case 'bottom':
  3269. toggleBar.style.borderRadius = '8px 8px 0 0'; // 左上、右上、右下、左下
  3270. break;
  3271. case 'top':
  3272. toggleBar.style.borderRadius = '0 0 8px 8px';
  3273. break;
  3274. case 'left':
  3275. toggleBar.style.height = '60px';
  3276. toggleBar.style.borderRadius = '0 8px 8px 0';
  3277. break;
  3278. case 'right':
  3279. toggleBar.style.height = '60px';
  3280. toggleBar.style.borderRadius = '8px 0 0 8px';
  3281. break;
  3282. }
  3283. }
  3284. },
  3285. closeViewMore: function() {
  3286. var toggleBar = this.eleMaps['sidebar-toggle'],
  3287. imgCon = this.eleMaps['img-container'],
  3288. viewmoreBar = this.eleMaps['sidebar-viewmore-content'],
  3289. imgPre = this.eleMaps['img-controler-pre'],
  3290. imgNext = this.eleMaps['img-controler-next'],
  3291. alreadyShow = toggleBar.style.visibility == 'hidden';
  3292. if(!alreadyShow) return;
  3293. var sidebarContainer = this.eleMaps['sidebar-container'];
  3294. var maximizeContainer = this.eleMaps['maximize-container'];
  3295. var sidebarPosition = prefs.gallery.sidebarPosition,
  3296. capitalize = function(string) { // 将字符串中每个单词首字母大写
  3297. var words = string.split(" ");
  3298. for (var i = 0; i < words.length; i++) {
  3299. words[i] = words[i].charAt(0).toUpperCase() + words[i].slice(1);
  3300. }
  3301. return words.join(" ");
  3302. };
  3303. maximizeContainer.style.minHeight = 0;
  3304. maximizeContainer.parentNode.style.visibility = "hidden";
  3305. if(this.hideBodyStyle.parentNode)
  3306. this.hideBodyStyle.parentNode.removeChild(this.hideBodyStyle);
  3307. imgPre.style.visibility = imgNext.style.visibility = toggleBar.style.visibility = sidebarContainer.style.visibility = 'visible';
  3308. imgCon.style['border' + capitalize(sidebarPosition)] = prefs.gallery.sidebarSize + 'px solid transparent';
  3309. toggleBar.style[sidebarPosition] = '-5px';
  3310. while (maximizeContainer.firstChild) {
  3311. maximizeContainer.removeChild(maximizeContainer.firstChild);
  3312. }
  3313. viewmoreBar.innerHTML = '✚';
  3314. viewmoreBar.parentNode.classList.remove("showmore");
  3315. //viewmoreBar.parentNode.style.backgroundColor = "#000000";
  3316.  
  3317. toggleBar.innerHTML = '▼';
  3318. this.changeSizeInputReset();
  3319. },
  3320. maximizeSidebar: function() {
  3321. var toggleBar = this.eleMaps['sidebar-toggle'],
  3322. imgCon = this.eleMaps['img-container'],
  3323. viewmoreBar = this.eleMaps['sidebar-viewmore-content'],
  3324. imgPre = this.eleMaps['img-controler-pre'],
  3325. imgNext = this.eleMaps['img-controler-next'],
  3326. alreadyShow = toggleBar.style.visibility == 'hidden';
  3327. var sidebarContainer = this.eleMaps['sidebar-container'];
  3328. var maximizeContainer = this.eleMaps['maximize-container'];
  3329. var sidebarPosition = prefs.gallery.sidebarPosition,
  3330. capitalize = function(string) { // 将字符串中每个单词首字母大写
  3331. var words = string.split(" ");
  3332. for (var i = 0; i < words.length; i++) {
  3333. words[i] = words[i].charAt(0).toUpperCase() + words[i].slice(1);
  3334. }
  3335. return words.join(" ");
  3336. };
  3337. if(alreadyShow){
  3338. this.closeViewMore();
  3339. }else{
  3340. maximizeContainer.style.minHeight = "100%";
  3341. maximizeContainer.parentNode.style.visibility = "visible";
  3342. document.head.appendChild(this.hideBodyStyle);
  3343. imgPre.style.visibility = imgNext.style.visibility = toggleBar.style.visibility = sidebarContainer.style.visibility = 'hidden';
  3344. imgCon.style['border' + capitalize(sidebarPosition)] = '0';
  3345. toggleBar.style[sidebarPosition] = '0';
  3346. maximizeContainer.innerHTML = "";
  3347. viewmoreBar.innerHTML = '✖';
  3348. viewmoreBar.parentNode.classList.add("showmore");//.backgroundColor = "#2a2a2a";
  3349.  
  3350. var nodes = document.querySelectorAll('.pv-gallery-sidebar-thumb-container[data-src]');
  3351. var self=this;
  3352. [].forEach.call(nodes, function(node){
  3353. var nodeStyle=unsafeWindow.getComputedStyle(node);
  3354. let curNode=node;
  3355. let imgSpan = document.createElement('span');
  3356. imgSpan.style.display=nodeStyle.display;
  3357. imgSpan.className = "maximizeChild";
  3358. imgSpan.innerHTML = '<img src="'+curNode.dataset.src+'">';
  3359. imgSpan.addEventListener("click", function(e){
  3360. imgReady(curNode.dataset.src,{
  3361. ready:function(){
  3362. new ImgWindowC(this);
  3363. }
  3364. });
  3365. });
  3366. let img=imgSpan.querySelector("img");
  3367. var addDlSpan=(img, imgSpan, curNode, clickCb)=>{
  3368. var dlSpan = document.createElement('p');
  3369. dlSpan.innerHTML='<svg class="icon" style="width: 20px;height: 20px;vertical-align: middle;fill: currentColor;overflow: hidden;" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="1100"><path d="M768 768q0-14.857143-10.857143-25.714286t-25.714286-10.857143-25.714285 10.857143-10.857143 25.714286 10.857143 25.714286 25.714285 10.857143 25.714286-10.857143 10.857143-25.714286z m146.285714 0q0-14.857143-10.857143-25.714286t-25.714285-10.857143-25.714286 10.857143-10.857143 25.714286 10.857143 25.714286 25.714286 10.857143 25.714285-10.857143 10.857143-25.714286z m73.142857-128v182.857143q0 22.857143-16 38.857143t-38.857142 16H91.428571q-22.857143 0-38.857142-16t-16-38.857143v-182.857143q0-22.857143 16-38.857143t38.857142-16h265.714286l77.142857 77.714286q33.142857 32 77.714286 32t77.714286-32l77.714285-77.714286h265.142858q22.857143 0 38.857142 16t16 38.857143z m-185.714285-325.142857q9.714286 23.428571-8 40l-256 256q-10.285714 10.857143-25.714286 10.857143t-25.714286-10.857143L230.285714 354.857143q-17.714286-16.571429-8-40 9.714286-22.285714 33.714286-22.285714h146.285714V36.571429q0-14.857143 10.857143-25.714286t25.714286-10.857143h146.285714q14.857143 0 25.714286 10.857143t10.857143 25.714286v256h146.285714q24 0 33.714286 22.285714z" p-id="1101"></path></svg> '+i18n("download");
  3370. dlSpan.src=curNode.dataset.src;
  3371. dlSpan.title=curNode.title||document.title;
  3372. dlSpan.onclick=clickCb;
  3373. imgSpan.appendChild(dlSpan);
  3374. };
  3375. fetch(curNode.dataset.src).then(response=>{
  3376. return response.blob();
  3377. }).then(blob=>{
  3378. imgReady(img,{
  3379. ready:function(){
  3380. if(img.width>=88 && img.height>=88){
  3381. addDlSpan(img, imgSpan, curNode, e=>{
  3382. e.stopPropagation();
  3383. if(blob.type=="image/webp"){
  3384. self.blobToCanvas(blob, canvas=>{
  3385. canvas.toBlob(blob=>{
  3386. saveAs(blob,e.target.title);
  3387. }, "image/png");
  3388. });
  3389. }else{
  3390. GM_download(e.target.src,e.target.title);
  3391. }
  3392. return true;
  3393. });
  3394. }
  3395. }
  3396. });
  3397. }).catch(e=>{
  3398. imgReady(img,{
  3399. ready:function(){
  3400. if(img.width>=88 && img.height>=88){
  3401. addDlSpan(img, imgSpan, curNode, e=>{
  3402. e.stopPropagation();
  3403. GM_download(e.target.src,e.target.title);
  3404. return true;
  3405. });
  3406. }
  3407. }
  3408. });
  3409. });
  3410. maximizeContainer.appendChild(imgSpan);
  3411. });
  3412. }
  3413. },
  3414. dataURLToCanvas:function (dataurl, cb){
  3415. if(!dataurl)return cb(null);
  3416. var canvas = document.createElement('CANVAS');
  3417. var ctx = canvas.getContext('2d');
  3418. var img = new Image();
  3419. img.setAttribute("crossOrigin","anonymous");
  3420. img.onload = function(){
  3421. canvas.width = img.width;
  3422. canvas.height = img.height;
  3423. ctx.drawImage(img, 0, 0);
  3424. cb(canvas);
  3425. };
  3426. img.onerror = function(){
  3427. cb(null);
  3428. };
  3429. img.src = dataurl;
  3430. },
  3431. blobToDataURL:function(blob, cb){
  3432. var a = new FileReader();
  3433. a.readAsDataURL(blob);
  3434. a.onload = function (e){
  3435. cb(e.target.result);
  3436. };
  3437. a.onerror = function (e){
  3438. cb(null);
  3439. }
  3440. },
  3441. blobToCanvas: function (blob, cb){
  3442. var self=this;
  3443. this.blobToDataURL(blob, function (dataurl){
  3444. self.dataURLToCanvas(dataurl, cb);
  3445. });
  3446. },
  3447. showHideBottom: function() { // 显示隐藏 sidebar-container
  3448. var sidebarContainer = this.eleMaps['sidebar-container'],
  3449. isHidden = sidebarContainer.style.visibility == 'hidden';
  3450.  
  3451. sidebarContainer.style.visibility = isHidden ? 'visible' : 'hidden';
  3452.  
  3453. var sidebarPosition = prefs.gallery.sidebarPosition,
  3454. capitalize = function(string) { // 将字符串中每个单词首字母大写
  3455. var words = string.split(" ");
  3456. for (var i = 0; i < words.length; i++) {
  3457. words[i] = words[i].charAt(0).toUpperCase() + words[i].slice(1);
  3458. }
  3459. return words.join(" ");
  3460. };
  3461.  
  3462. // 修正下图片底部的高度
  3463. this.eleMaps['img-container'].style['border' + capitalize(sidebarPosition)] = isHidden ?
  3464. prefs.gallery.sidebarSize + 'px solid transparent' :
  3465. '0';
  3466. // 修正底部距离
  3467. this.eleMaps['sidebar-toggle'].style[sidebarPosition] = isHidden ? '-5px' : '0';
  3468. this.eleMaps['sidebar-toggle'].innerHTML = isHidden ? '▼' : '▲';
  3469. this.eleMaps['sidebar-viewmore'].style.visibility = isHidden ? 'visible' : 'hidden';
  3470. },
  3471. initZoom: function() { // 如果有放大,则把图片及 sidebar 部分缩放比率改为 1
  3472. if (prefs.gallery.autoZoom && document.body.style.zoom != undefined) {
  3473. var oZoom = detectZoom();
  3474. if (oZoom > 100) {
  3475. this.eleMaps['body'].style.zoom = 100 / oZoom;
  3476. }
  3477. }
  3478. },
  3479.  
  3480. getThumSpan:function(previous,relatedTarget){
  3481. var ret;
  3482. var rt = relatedTarget || this.selected;
  3483. if(!rt)return;
  3484. while((rt=previous ? rt.previousElementSibling : rt.nextElementSibling)){
  3485. if(rt.clientWidth!=0){
  3486. ret=rt;
  3487. break;
  3488. };
  3489. };
  3490. return ret;
  3491. },
  3492. previous:false,
  3493. selectPrevious:function(){
  3494. this.previous=true;
  3495. this.select(this.getThumSpan(true));
  3496. },
  3497. selectNext:function(){
  3498. this.select(this.getThumSpan());
  3499. },
  3500. select:function(ele,noTransition){
  3501. if(!ele || this.selected==ele)return;
  3502. if(this.selected){
  3503. this.selected.classList.remove(this.selectedClassName);
  3504. this.selected.classList.remove('pv-gallery-sidebar-thumb_selected');
  3505. };
  3506. ele.classList.add(this.selectedClassName);
  3507. ele.classList.add('pv-gallery-sidebar-thumb_selected');
  3508.  
  3509. this.selected=ele;
  3510. this.arrowVisib();
  3511.  
  3512. var self=this;
  3513. clearTimeout(this.loadImgTimer);
  3514. if(prefs.gallery.transition){
  3515. this.loadImgTimer=setTimeout(function(){//快速跳转的时候不要尝试读取图片。
  3516. self.loadImg(ele);
  3517. },200);
  3518. }else{
  3519. self.loadImg(ele);
  3520. }
  3521.  
  3522. this.selectedIntoView(noTransition);
  3523. this.forceRepaint();
  3524. this.slideShow.run('select');
  3525. },
  3526. loadThumb:function(){//读取可视范围里面的缩略图
  3527. var self=this;
  3528. var pro=self.isHorizontal ? ['scrollLeft','clientWidth','offsetLeft','offsetWidth'] : ['scrollTop','clientHeight','offsetTop','offsetHeight'];
  3529. var thumbC=self.eleMaps['sidebar-thumbnails-container'];
  3530.  
  3531. var scrolled=thumbC[pro[0]];
  3532.  
  3533. var loadStopDis=scrolled + thumbC[pro[1]];
  3534.  
  3535. var imgSpans=self.selected.parentNode.children;
  3536. var span_i;
  3537. var spanOffset;
  3538. var thumb;
  3539.  
  3540. var i=0
  3541. while(span_i=imgSpans[i++]){
  3542. if(span_i.clientWidth==0)continue;//隐藏的
  3543.  
  3544. spanOffset=span_i[pro[2]];
  3545. if(spanOffset + span_i[pro[3]] <= scrolled)continue;//在滚动条上面了
  3546. if(spanOffset >= loadStopDis)break;//在滚动条下面了
  3547.  
  3548. if(dataset(span_i,'thumbLoaded'))continue;//已经加载了缩略图
  3549.  
  3550. thumb=document.createElement('img');
  3551. thumb.src=dataset(span_i,'thumbSrc') || dataset(span_i,'src') || prefs.icons.brokenImg_small;
  3552. thumb.className='pv-gallery-sidebar-thumb';
  3553.  
  3554. dataset(span_i,'thumbLoaded','true');
  3555. span_i.appendChild(thumb);
  3556.  
  3557. imgReady(thumb,{
  3558. error:function(e){
  3559. this.src=prefs.icons.brokenImg_small;
  3560. },
  3561. });
  3562. };
  3563. },
  3564. selectedIntoView:function(noTransition){
  3565. var thumBC=this.eleMaps['sidebar-thumbnails-container'];
  3566. var pro=this.isHorizontal ? ['offsetLeft','clientWidth','offsetWidth'] : ['offsetTop','clientHeight','offsetHeight'] ;
  3567. //需要滚动的距离。
  3568. var needScrollDis= this.selected[pro[0]];
  3569. //尽可能的居中显示
  3570. var thumBCClient=thumBC[pro[1]];
  3571. var scrollCenter=Math.max((thumBCClient - this.selected[pro[2]])/2,0);
  3572.  
  3573. this.thumbScrollbar.scroll(needScrollDis - scrollCenter,false,!noTransition);
  3574. },
  3575. getImg:function(ele){
  3576. var self = this;
  3577.  
  3578. var src = dataset(ele,'src');
  3579.  
  3580. this.lastLoading=src;//记住最后读取的图片
  3581. this.isLoading=true;//表示选择的图片正在读取
  3582.  
  3583. // 特殊的 xhr 方式获取
  3584. var xhr = dataset(ele, 'xhr');
  3585. if (xhr) {
  3586. var xhrError = function() {
  3587. dataset(ele, 'xhr', '');
  3588. dataset(ele, 'src', dataset(ele, 'thumb-src'));
  3589. self.getImg(ele);
  3590. };
  3591. xhrLoad.load({
  3592. url: src,
  3593. xhr: JSON.parse(decodeURIComponent(xhr)),
  3594. cb: function(imgSrc, imgSrcs, caption) {
  3595. if (imgSrc) {
  3596. dataset(ele, 'src', imgSrc);
  3597. dataset(ele, 'xhr', '');
  3598. self.getImg(ele);
  3599. } else {
  3600. xhrError();
  3601. }
  3602. },
  3603. onerror: xhrError
  3604. });
  3605. return;
  3606. }
  3607.  
  3608. var allLoading=this.allLoading;
  3609. if(allLoading.indexOf(src)!=-1){//在读取队列中。
  3610. return;
  3611. };
  3612. allLoading.push(src);
  3613.  
  3614. //上一个读取中的图片,不是当前显示的。那么直接终止
  3615. var preImgR=this.imgReady;
  3616. if(preImgR && this.img){
  3617. if(preImgR.img.src!=this.src){
  3618. preImgR.abort();
  3619. preImgR.removeLI();
  3620. };
  3621. };
  3622.  
  3623.  
  3624. //显示读取指示器。
  3625. var loadingIndicator=ele.querySelector('.pv-gallery-sidebar-thumb-loading');
  3626. loadingIndicator.style.display='block';
  3627.  
  3628.  
  3629. this.imgReady=imgReady(src,{
  3630. ready:function(){
  3631. //从读取队列中删除自己
  3632. var index=allLoading.indexOf(src);
  3633. if(index!=-1){
  3634. allLoading.splice(index,1);
  3635. };
  3636.  
  3637. if(src!=self.lastLoading)return;
  3638.  
  3639. loadingIndicator.style.display='';
  3640. if(preImgR)preImgR.abort();
  3641. self.loadImg(this,ele);
  3642. },
  3643. loadEnd:function(e){//在loadend后开始预读。
  3644. //从读取队列中删除自己
  3645. var index=allLoading.indexOf(src);
  3646. if(index!=-1){
  3647. allLoading.splice(index,1);
  3648. };
  3649.  
  3650. if(src!=self.lastLoading)return;
  3651.  
  3652. if(e.type=='error'){
  3653. loadingIndicator.style.display='';
  3654. self.errorSpan=ele;
  3655. if(preImgR)preImgR.abort();
  3656. self.loadImg(this,ele,true);
  3657. };
  3658.  
  3659. self.slideShow.run('loadEnd');
  3660.  
  3661. if(prefs.gallery.preload){
  3662. if(self.preloading){//结束上次的预读。
  3663. self.preloading.abort();
  3664. };
  3665. self.preloading=new self.Preload(ele,self);
  3666. self.preloading.preload();
  3667. };
  3668. },
  3669. });
  3670.  
  3671. this.imgReady.removeLI=function(){
  3672. loadingIndicator.style.display='';
  3673. };
  3674.  
  3675. },
  3676. loadImg:function(img,relatedThumb,error){
  3677. if(img.nodeName!='IMG'){//先读取。
  3678. this.getImg(img);
  3679. return;
  3680. };
  3681.  
  3682. if(this.img){
  3683. this.img.parentNode.removeChild(this.img);
  3684. };
  3685.  
  3686. var imgNaturalSize={
  3687. h:img.naturalHeight,
  3688. w:img.naturalWidth,
  3689. };
  3690. this.imgNaturalSize=imgNaturalSize;
  3691.  
  3692. this.eleMaps['head-left-img-info-resolution'].textContent=imgNaturalSize.w + ' x ' + imgNaturalSize.h;
  3693. var thumbnails=this.eleMaps['sidebar-thumbnails-container'].childNodes,i=0;
  3694. while(thumbnails[i]!=relatedThumb && i<thumbnails.length)i++;
  3695. if(i<thumbnails.length)this.eleMaps['head-left-img-info-count'].textContent="("+(i+1)+" / "+thumbnails.length+")";
  3696. // 加上图片的注释
  3697. var description = decodeURIComponent(dataset(relatedThumb, 'description')),
  3698. defaultLength = prefs.gallery.descriptionLength;
  3699. this.eleMaps['head-left-img-info-description'].title = description;
  3700. this.eleMaps['head-left-img-info-description'].textContent= description.length > defaultLength ?
  3701. description.slice(0, defaultLength) + '...' :
  3702. description;
  3703.  
  3704. this.img=img;
  3705. this.src=img.src;
  3706. this.isLoading=false;
  3707.  
  3708. this.relatedThumb=relatedThumb;
  3709. img.className='pv-gallery-img';
  3710.  
  3711. if(error){
  3712. if(relatedThumb.querySelector("img").src==this.img.src){
  3713. this.imgError=true;
  3714. this.img.style.display='none';
  3715. this.eleMaps['img_broken'].style.display='inline-block';
  3716. }else{
  3717. var srcs=dataset(relatedThumb, 'srcs').split(",");
  3718. var self=this;
  3719. this.img.onload=function(){
  3720. var imgNaturalSize={
  3721. h:this.naturalHeight,
  3722. w:this.naturalWidth,
  3723. };
  3724.  
  3725. self.imgNaturalSize=imgNaturalSize;
  3726. self.fitToScreen();
  3727. }
  3728. if(srcs){
  3729. var src=srcs.shift();
  3730. dataset(relatedThumb, 'srcs',srcs.join(","));
  3731. if(src){
  3732. dataset(relatedThumb, 'src',src);
  3733. this.img.onerror=function(e){
  3734. this.src=relatedThumb.querySelector("img").src;
  3735. dataset(relatedThumb, 'src',this.src);
  3736. }
  3737. this.img.src=src;
  3738. }
  3739. }
  3740. else this.img.src=relatedThumb.querySelector("img").src;
  3741. }
  3742. }else{
  3743. this.imgError=false;
  3744. this.eleMaps['img_broken'].style.display='';
  3745. if(!dataset(relatedThumb,'naturalSize')){
  3746. dataset(relatedThumb,'naturalSize',JSON.stringify(imgNaturalSize));
  3747. };
  3748. };
  3749.  
  3750. function styled(){
  3751. img.style.opacity=1;
  3752. img.style[support.cssTransform]='scale(1)';
  3753. };
  3754.  
  3755.  
  3756. if(prefs.gallery.transition){
  3757. setTimeout(styled,0);
  3758. }else{
  3759. styled();
  3760. };
  3761.  
  3762. this.eleMaps['img-parent'].appendChild(img);
  3763.  
  3764. this.fitContains=prefs.gallery.fitToScreen;//适应屏幕
  3765.  
  3766. this.fitToScreen({
  3767. x:0,
  3768. y:0,
  3769. });
  3770.  
  3771. if(this.previous){
  3772. this.previous=false;
  3773. this.imgScrollbarV.scrollBy(999999);
  3774. }
  3775. this.collection.check();//检查是否在收藏里面。
  3776.  
  3777. },
  3778. fitToScreen:function(scale){
  3779.  
  3780. var container=this.eleMaps['img-content'];
  3781. var containerSize={
  3782. h:container.clientHeight,
  3783. w:container.clientWidth,
  3784. };
  3785.  
  3786. var img=this.img;
  3787.  
  3788. if(!img.classList)return;
  3789. img.classList.remove('pv-gallery-img_zoom-in');
  3790. img.classList.remove('pv-gallery-img_zoom-out');
  3791.  
  3792. var imgSty=img.style;
  3793. imgSty.width='';
  3794. imgSty.height='';
  3795.  
  3796. var contentSSize={
  3797. h:container.scrollHeight,
  3798. w:container.scrollWidth,
  3799. };
  3800. var larger=contentSSize.h>containerSize.h || contentSSize.w>containerSize.w;
  3801.  
  3802. var scaled='100%';
  3803.  
  3804. if(this.fitContains && !(scale && scale.x==0 && scale.y==0 && this.imgNaturalSize.h/this.imgNaturalSize.w > 2.5)){//适应屏幕
  3805. this.imgScrollbarV.hide();
  3806. this.imgScrollbarH.hide();
  3807. if(larger){
  3808. img.classList.add('pv-gallery-img_zoom-in');
  3809. if(contentSSize.h/contentSSize.w >=containerSize.h/containerSize.w){
  3810. var height=this.imgNaturalSize.h-(contentSSize.h - containerSize.h);
  3811. imgSty.height=height + 'px';
  3812. scaled=height/this.imgNaturalSize.h;
  3813. }else{
  3814. var width=this.imgNaturalSize.w-(contentSSize.w - containerSize.w);
  3815. imgSty.width=width + 'px';
  3816. scaled=width/this.imgNaturalSize.w;
  3817. };
  3818. scaled=(scaled*100).toFixed(2) + '%';
  3819. }else if(prefs.gallery.fitToScreenSmall){
  3820. if(this.imgNaturalSize.h/this.imgNaturalSize.w >=containerSize.h/containerSize.w){
  3821. var height=contentSSize.h-50;
  3822. height=height<0?contentSSize.h:height;
  3823. imgSty.height=height + 'px';
  3824. scaled=height/this.imgNaturalSize.h;
  3825. }else{
  3826. var width=contentSSize.w-50;
  3827. width=width<0?contentSSize.w:width;
  3828. imgSty.width=width + 'px';
  3829. scaled=width/this.imgNaturalSize.w;
  3830. };
  3831. scaled=(scaled*100).toFixed(2) + '%';
  3832. }
  3833. }else{//不做尺寸调整
  3834. this.imgScrollbarV.reset();
  3835. this.imgScrollbarH.reset();
  3836.  
  3837. if(larger){
  3838. img.classList.add('pv-gallery-img_zoom-out');
  3839. if(scale){//通过鼠标点击进行的切换。
  3840. this.imgScrollbarH.scroll(container.scrollWidth * scale.x - containerSize.w/2);
  3841. this.imgScrollbarV.scroll(container.scrollHeight * scale.y - containerSize.h/2);
  3842. };
  3843. };
  3844. };
  3845.  
  3846.  
  3847. var imgScaledInfo=this.eleMaps['head-left-img-info-scaling'];
  3848. imgScaledInfo.textContent='('+scaled+')';
  3849. if(scaled!='100%'){
  3850. imgScaledInfo.style.color='#E9CCCC';
  3851. }else{
  3852. imgScaledInfo.style.color='';
  3853. };
  3854.  
  3855. },
  3856.  
  3857. _dataCache: {},
  3858. _spanMarkPool: {},
  3859. _appendThumbSpans: function(data, index) { // 添加缩略图栏的 spans
  3860. var iStatisCopy = this.iStatisCopy;
  3861.  
  3862. if (typeof index == 'undefined' && this.selected) {
  3863. index = Array.prototype.slice.call(this.imgSpans).indexOf(this.selected);
  3864. }
  3865.  
  3866. var sizeInputH=this.gallery.querySelector("#minsizeH");
  3867. var sizeInputW=this.gallery.querySelector("#minsizeW");
  3868. var thumbnails = this.eleMaps['sidebar-thumbnails-container'];
  3869. // 如果是新的,则添加,否则重置并添加。
  3870. if (!data){
  3871. thumbnails.innerHTML = "";
  3872. this._dataCache = {};
  3873. }
  3874. var self=this;
  3875. (data || this.data).forEach(function(item) {
  3876. if(!item)return;
  3877. iStatisCopy[item.type].count++;
  3878. var spanMark=self._spanMarkPool[item.imgSrc];
  3879. if(!spanMark){
  3880. spanMark = document.createElement("span");
  3881. try{
  3882. spanMark.className="pv-gallery-sidebar-thumb-container";
  3883. spanMark.dataset.type=item.type;
  3884. spanMark.dataset.src=item.src;
  3885. spanMark.dataset.srcs=item.srcs||"";
  3886. if(item.xhr)spanMark.dataset.xhr=encodeURIComponent(JSON.stringify(item.xhr));
  3887. spanMark.dataset.description=encodeURIComponent(item.description || '');
  3888. spanMark.dataset.thumbSrc=item.imgSrc;
  3889. spanMark.title=(item.img?(item.img.title||item.img.alt||""):"");
  3890. spanMark.innerHTML='<span class="pv-gallery-vertical-align-helper"></span>' +
  3891. '<span class="pv-gallery-sidebar-thumb-loading" title="'+i18n("loading")+'......"></span>';
  3892. }catch(e){};
  3893. self._spanMarkPool[item.imgSrc] = spanMark;
  3894. }
  3895. if(item.noActual && (item.sizeW<sizeInputW.value || item.sizeH<sizeInputH.value)){
  3896. spanMark.style.display="none";
  3897. }else{
  3898. spanMark.style.display="";
  3899. }
  3900. thumbnails.appendChild(spanMark);
  3901. });
  3902.  
  3903. var self = this;
  3904. (data || this.data).forEach(function(d) {
  3905. if(!d)return;
  3906. self._dataCache[d.imgSrc] = true;
  3907. });
  3908.  
  3909. //写入类别数据。
  3910. var gallery = this.gallery;
  3911. var input, label, iStatisCopy_i;
  3912.  
  3913. for (var i in iStatisCopy) {
  3914. if (!iStatisCopy.hasOwnProperty(i)) continue;
  3915. iStatisCopy_i = iStatisCopy[i];
  3916. input = gallery.querySelector('#pv-gallery-head-command-drop-list-item-category-' + i);
  3917. input.checked = iStatisCopy_i.shown;
  3918. if (iStatisCopy_i.count == 0) {
  3919. input.disabled = true;
  3920. input.parentNode.classList.add('pv-gallery-head-command-drop-list-item_disabled');
  3921. } else {
  3922. input.disabled = false;
  3923. input.parentNode.classList.remove('pv-gallery-head-command-drop-list-item_disabled');
  3924. };
  3925.  
  3926. label = gallery.querySelector('label[for="pv-gallery-head-command-drop-list-item-category-' + i + '"]');
  3927. label.textContent = label.textContent.replace(/(.*)/i, '') + '(' + iStatisCopy_i.count + ')';
  3928. };
  3929.  
  3930. this.imgSpans = thumbnails.children;
  3931.  
  3932. this.thumbScrollbar.reset();
  3933.  
  3934. if(this.imgSpans[index].style.display=="none"){
  3935. for(var j in this.imgSpans){
  3936. if (!this.imgSpans.hasOwnProperty(j)) continue;
  3937. var curSpan=this.imgSpans[j];
  3938. if(curSpan.style.display!="none"){
  3939. this.select(curSpan, true);
  3940. return;
  3941. }
  3942. }
  3943. }
  3944. this.select(this.imgSpans[index], true);
  3945. },
  3946. load:function(data, from, reload){
  3947. if(this.shown || this.minimized){//只允许打开一个,请先关掉当前已经打开的库
  3948.  
  3949. if(from){//frame发送过来的数据。
  3950. window.postMessage({
  3951. messageID:messageID,
  3952. command:'sendFail',
  3953. to:from,
  3954. },'*');
  3955. };
  3956.  
  3957. if(this.minimized){
  3958. GM_notification('请先关掉当前已经打开的库');
  3959. flashEle(this.maximizeTrigger);
  3960. };
  3961. return;
  3962. };
  3963.  
  3964. var self=this;
  3965. if(from){//来自frame,获取这个frame所在的iframe标签。定位到图片的时候要用到。
  3966. window.postMessage({
  3967. messageID:messageID,
  3968. command:'getIframeObject',
  3969. windowId:from,
  3970. },'*');
  3971. document.addEventListener('pv-getIframeObject',function(e){
  3972. self.iframe=e.detail;
  3973. },true);
  3974. };
  3975.  
  3976. var unique=this.unique(data);
  3977. data=unique.data;
  3978. var index=unique.index;
  3979.  
  3980. if (reload && this.data.length >= data.length) {
  3981. // alert('没有新增的图片');
  3982. return;
  3983. }
  3984.  
  3985. this.clear();//还原对象的一些修改,以便复用。
  3986. this.data=data;
  3987. this.show(reload);
  3988.  
  3989. this.from=from;//如果来自frame,那么这个from应该保存了那个frame的窗口id,便于以后通信。
  3990.  
  3991. this._appendThumbSpans(null, index);
  3992.  
  3993. this.runOnce();
  3994.  
  3995. this.switchThumbVisible();
  3996.  
  3997. var pageObj = this.getPage(),textSpan = this.eleMaps['head-command-nextPage'].querySelector("span");
  3998. var haveMorePage = pageObj.pre || pageObj.next;
  3999. textSpan.style.color=haveMorePage?"#e9cccc":"#757575";
  4000. },
  4001. clear:function(){
  4002. this._dataCache = {};
  4003.  
  4004. this.allLoading=[];//读取中的图片数组
  4005. this.iStatisCopy=cloneObject(this.imgStatistics,true);//图片统计副本
  4006. this.selected==null;
  4007. if(this.img){
  4008. this.img.style.display='none';
  4009. this.img=null;
  4010. };
  4011. //读取错误的图片占位符
  4012. this.eleMaps['img_broken'].style.display='';
  4013. //清空dom
  4014. this.eleMaps['sidebar-thumbnails-container'].innerHTML='';
  4015. this.eleMaps['head-left-img-info-resolution'].textContent='0 x 0';
  4016. this.eleMaps['head-left-img-info-count'].textContent='(1 / 1)';
  4017. this.eleMaps['head-left-img-info-scaling'].textContent='(100%)';
  4018. //隐藏滚动条
  4019. this.imgScrollbarV.hide();
  4020. this.imgScrollbarH.hide();
  4021. this.thumbScrollbar.hide();
  4022. //重置style;
  4023. this.thumbVisibleStyle.textContent='';
  4024. },
  4025.  
  4026. unique:function(data){
  4027. var imgSrc;
  4028. if(data.target)
  4029. imgSrc=data.target.src;
  4030.  
  4031. var data_i,
  4032. data_i_src,
  4033. dataSrcs=[];
  4034.  
  4035. var index;
  4036.  
  4037. for(var i=0,ii=data.length;i<ii;i++){
  4038. data_i=data[i];
  4039. data_i_src=data_i.src;
  4040. if(dataSrcs.indexOf(data_i_src)!=-1){//已经存在
  4041. data.splice(i,1);//移除
  4042. i--;
  4043. ii--;
  4044. continue;
  4045. };
  4046. dataSrcs.push(data_i_src);
  4047.  
  4048. if(imgSrc==data_i_src){
  4049. index=i;
  4050. };
  4051. };
  4052.  
  4053. if(typeof index =='undefined'){
  4054. index=0;
  4055. data.unshift(data.target);
  4056. };
  4057.  
  4058. delete data.target;
  4059.  
  4060. return {
  4061. data:data,
  4062. index:index,
  4063. };
  4064. },
  4065. keyDownListener:function(e){
  4066. switch(e.keyCode){
  4067. case 27:
  4068. if(prefs.imgWindow.close.escKey){
  4069. this.close();
  4070. }
  4071. break;
  4072. }
  4073. },
  4074. keyUpListener:function(e){
  4075. switch(e.keyCode){
  4076. case 82:
  4077. var img=this.img;
  4078. var cssTransform=img.style[support.cssTransform];
  4079. var iTransform=cssTransform.replace(/rotate\([^)]*\)/i,'');
  4080. var rotatedRadians=cssTransform.indexOf("rotate")!=-1?cssTransform.replace(/.*rotate\(([-\d\.]+).*/i,'$1'):0;
  4081. var PI=Math.PI;
  4082. var origin=parseFloat(rotatedRadians) +(e.shiftKey?-90:90) * PI/180;
  4083. if(origin>=2*PI || origin<=-2*PI ||(-0.1<origin && origin<0.1)){
  4084. origin=0;
  4085. }
  4086. img.style[support.cssTransform] = ' rotate('+ origin +'rad) ' + iTransform;
  4087. break;
  4088. }
  4089. },
  4090. show:function(reload){
  4091. this.shown=true;
  4092. galleryMode=true;
  4093.  
  4094. if (!reload) {
  4095. var des=document.documentElement.style;
  4096. this.deOverflow={
  4097. x:des.overflowX,
  4098. y:des.overflowY,
  4099. };
  4100. des.overflow='hidden';
  4101. this.gallery.style.display='';
  4102. this.gallery.focus();
  4103. window.addEventListener('resize',this._resizeHandler,true);
  4104. }
  4105. document.addEventListener('keydown',this._keyDownListener,true);
  4106. document.addEventListener('keyup',this._keyUpListener,true);
  4107.  
  4108. if(prefs.gallery.loadMore){
  4109. this.eleMaps['head-command-nextPage'].click();
  4110. }
  4111.  
  4112. this.changeSizeInputReset();
  4113. },
  4114. close:function(reload){
  4115. if(this.hideBodyStyle.parentNode)
  4116. this.hideBodyStyle.parentNode.removeChild(this.hideBodyStyle);
  4117. document.removeEventListener('keydown',this._keyDownListener,true);
  4118. document.removeEventListener('keyup',this._keyUpListener,true);
  4119. this.shown=false;
  4120. this.minimized=false;
  4121.  
  4122. if (!reload) {
  4123. galleryMode=false;
  4124. this.gallery.blur();
  4125. this.gallery.style.display='none';
  4126. var des=document.documentElement.style;
  4127. des.overflowX=this.deOverflow.x;
  4128. des.overflowY=this.deOverflow.y;
  4129. this.slideShow.exit();
  4130. this.collection.exit();
  4131. window.removeEventListener('resize',this._resizeHandler,true);
  4132.  
  4133. // 退出全屏
  4134. var btn = document.getElementById('pv-gallery-fullscreenbtn');
  4135. if (btn.classList.contains('fullscreenbtn')) {
  4136. cancelFullScreen();
  4137. btn.textContent = i18n("enterFullsc");
  4138. btn.classList.remove('fullscreenbtn');
  4139. }
  4140. }
  4141. },
  4142. curPage:document,
  4143. getPage:function(){
  4144. var pageNum=0;
  4145. if(/[a-zA-Z0-9][_\-]\d+(\.html)?$/.test(this.href)){
  4146. pageNum=this.href.replace(/.*[a-zA-Z0-9][\-_](\d+)(\.html)?$/,"$1");
  4147. }
  4148. var curPage=this.curPage;
  4149. let pre=curPage.querySelector("a.prev");
  4150. let next=curPage.querySelector("a.next");
  4151. if(!pre)pre=curPage.querySelector("a#prev");
  4152. if(!next)next=curPage.querySelector("a#next");
  4153. if(!pre)pre=curPage.querySelector("a#leftFix");
  4154. if(!next)next=curPage.querySelector("a#rightFix");
  4155. if(!pre || !next){
  4156. let aTags=curPage.querySelectorAll("a");
  4157. if(!pre){
  4158. let pref,pres,pret;
  4159. for(var i=0;i<aTags.length;i++){
  4160. let aTag=aTags[i];
  4161. if(pref && pres && pret)break;
  4162. if(!pref){
  4163. if(/上[一1]?[页頁张張]|previous page|前のページ/i.test(aTag.innerHTML)){
  4164. pref=aTag;
  4165. }
  4166. }
  4167. if(!pres){
  4168. if(aTag.innerHTML=="&lt;"){
  4169. pres=aTag;
  4170. }
  4171. }
  4172. if(!pret){
  4173. if(aTag.innerHTML=="«"){
  4174. pret=aTag;
  4175. }else if(pageNum==1){
  4176. if(aTag.getAttribute("href") && aTag.getAttribute("href").indexOf(this.href.replace(/.*\/([^\/]+)$/,"$1").replace(/[_-]\d+/,""))!=-1){
  4177. pret=aTag;
  4178. }
  4179. }else if(aTag.getAttribute("href") && aTag.getAttribute("href").replace(/.*[a-zA-Z0-9][\-_](\d+)(\.html)?$/,"$1")==pageNum-1){
  4180. pret=aTag;
  4181. }
  4182. }
  4183. }
  4184. pre=pref||pres||pret;
  4185. }
  4186. if(!next){
  4187. let nextf,nexts,nextt;
  4188. for(var i=0;i<aTags.length;i++){
  4189. let aTag=aTags[i];
  4190. if(nextf && nexts && nextt)break;
  4191. if(!nextf){
  4192. if(/下[一1]?[页頁张張]|next page|次のページ/i.test(aTag.innerHTML)){
  4193. nextf=aTag;
  4194. }
  4195. }
  4196. if(!nexts){
  4197. if(aTag.innerHTML=="&gt;"){
  4198. nexts=aTag;
  4199. }
  4200. }
  4201. if(!nextt){
  4202. if(aTag.innerHTML=="»"){
  4203. nextt=aTag;
  4204. }else if(aTag.hasAttribute("href") && aTag.getAttribute("href").replace(/.*[a-zA-Z0-9][\-_](\d+)(\.html)?$/,"$1")==parseInt(pageNum)+1){
  4205. nextt=aTag;
  4206. }
  4207. }
  4208. }
  4209. next=nextf||nexts||nextt;
  4210. }
  4211. }
  4212. if(!pre)pre=curPage.querySelector(".prev>a");
  4213. if(!next)next=curPage.querySelector(".next>a");
  4214. if(!pre && !next){
  4215. let pageDiv=curPage.querySelector("div.wp-pagenavi");
  4216. if(pageDiv){
  4217. var cur=pageDiv.querySelector("span.current");
  4218. pre=cur.previousSibling;
  4219. next=cur.nextSibling;
  4220. }else{
  4221. var cur=curPage.querySelector("div.article-paging>span");
  4222. if(cur){
  4223. pre=cur.previousElementSibling;
  4224. next=cur.nextElementSibling;
  4225. }
  4226. }
  4227. }
  4228. return {pre:pre,next:next};
  4229. },
  4230. canonicalUri:function(src, base_path){
  4231. if(src.charAt(0)=="#")return location.href+src;
  4232. var root_page = /^[^?#]*\//.exec(location.href)[0],
  4233. root_domain = /^\w+\:\/\/\/?[^\/]+/.exec(root_page)[0],
  4234. absolute_regex = /^\w+\:\/\//;
  4235. src=src.replace(/\.\//,"");
  4236. if (/^\/\/\/?/.test(src)){
  4237. src = location.protocol + src;
  4238. }
  4239. else if (!absolute_regex.test(src) && src.charAt(0) != "/"){
  4240. src = (base_path || "") + src;
  4241. }
  4242. return (absolute_regex.test(src) ? src : ((src.charAt(0) == "/" ? root_domain : root_page) + src));
  4243. },
  4244. completePages:[location.href],
  4245. href:location.href,
  4246. pageAllReady:false,
  4247. londingImgNum:0,
  4248. pageImgReady:function(){
  4249. var textSpan=this.eleMaps['head-command-nextPage'].querySelector("span");
  4250. if(this.pageAllReady && this.londingImgNum<=0){
  4251. textSpan.innerHTML="<font color='red'>"+i18n("loadedAll")+"</font>";
  4252. setTimeout(function(){textSpan.innerHTML=i18n("loadAll");},1500);
  4253. }
  4254. },
  4255. prePage:function(){
  4256. var pageObj=this.getPage(),self=this,textSpan=this.eleMaps['head-command-nextPage'].querySelector("span");
  4257. if(textSpan.innerHTML!=i18n("loading")){
  4258. return;
  4259. }
  4260. var loadOver=function(){
  4261. self.pageAllReady=true;
  4262. self.pageImgReady();
  4263. };
  4264. if(!pageObj.pre){
  4265. loadOver();
  4266. return;
  4267. }
  4268. var preUrl=pageObj.pre;
  4269. if(preUrl.tagName!="A"){
  4270. var childA=preUrl.querySelector("a");
  4271. if(childA){
  4272. preUrl=childA;
  4273. }else{
  4274. while(preUrl=preUrl.parentElement){
  4275. if(preUrl.nodeName=='A'){
  4276. break;
  4277. }
  4278. }
  4279. if(!preUrl)return;
  4280. }
  4281. }
  4282. var href=preUrl.getAttribute("href");
  4283. if(self.completePages.indexOf(href)!=-1){
  4284. loadOver();
  4285. return;
  4286. }else{
  4287. self.completePages.push(href);
  4288. }
  4289. self.href=self.canonicalUri(href);
  4290. GM_xmlhttpRequest({
  4291. method: 'GET',
  4292. headers:{"Referer": + window.location.href},
  4293. url: self.href,
  4294. overrideMimeType:"text/html;charset="+document.charset,
  4295. onload: function(d) {
  4296. let html=document.implementation.createHTMLDocument('');
  4297. html.documentElement.innerHTML = d.responseText;
  4298. self.curPage=html;
  4299. let imgs=html.querySelectorAll('img');
  4300. var container = document.querySelector('.pv-gallery-container'),
  4301. preloadContainer = document.querySelector('.pv-gallery-preloaded-img-container');
  4302. imgs = Array.prototype.slice.call(imgs).filter(function(img){
  4303. return !(container.contains(img) || (preloadContainer&&preloadContainer.contains(img)));
  4304. });
  4305. imgs.forEach(function(img) {
  4306. pretreatment(img);
  4307. if(!img.src)return;
  4308. var isrc=img.src.trim();
  4309. if(!isrc)return;
  4310. isrc=self.canonicalUri(isrc);
  4311. if (self._dataCache[isrc]) return;
  4312. self._dataCache[isrc] = true;
  4313. self.londingImgNum++;
  4314. var nimg = new Image();
  4315. nimg.src = isrc;
  4316. nimg.onload=function(){
  4317. self.londingImgNum--;
  4318. self.pageImgReady();
  4319. var result = findPic(this);
  4320. if (result) {
  4321. self.data.push(result);
  4322. self._appendThumbSpans([result]);
  4323. self.loadThumb();
  4324. }
  4325. };
  4326. nimg.onerror=function(){
  4327. self.londingImgNum--;
  4328. self.pageImgReady();
  4329. };
  4330. });
  4331. if(prefs.gallery.loadAll)self.prePage();
  4332. else loadOver();
  4333. },
  4334. onerror: function(e) {
  4335. if(prefs.gallery.loadAll)self.prePage();
  4336. else loadOver();
  4337. }
  4338. });
  4339. },
  4340. nextPage:function(){
  4341. var pageObj=this.getPage(),self=this,textSpan=this.eleMaps['head-command-nextPage'].querySelector("span");
  4342. if(textSpan.innerHTML!=i18n("loading")){
  4343. return;
  4344. }
  4345. var loadOver=function(){
  4346. if(prefs.gallery.loadAll){
  4347. self.curPage=document;
  4348. self.href=location.href;
  4349. self.prePage();
  4350. }else{
  4351. self.pageAllReady=true;
  4352. self.pageImgReady();
  4353. }
  4354. };
  4355. if(!pageObj.next){
  4356. loadOver();
  4357. return;
  4358. }
  4359. var nextUrl=pageObj.next;
  4360. if(nextUrl.tagName!="A"){
  4361. var childA=nextUrl.querySelector("a");
  4362. if(childA){
  4363. nextUrl=childA;
  4364. }else{
  4365. while(nextUrl=nextUrl.parentElement){
  4366. if(nextUrl.nodeName=='A'){
  4367. break;
  4368. }
  4369. }
  4370. if(!nextUrl)return;
  4371. }
  4372. }
  4373. var href=nextUrl.getAttribute("href");
  4374. if(self.completePages.indexOf(href)!=-1){
  4375. loadOver();
  4376. return;
  4377. }else{
  4378. self.completePages.push(href);
  4379. }
  4380. self.href=self.canonicalUri(href);
  4381. GM_xmlhttpRequest({
  4382. method: 'GET',
  4383. url: self.href,
  4384. headers:{"Referer": + window.location.href},
  4385. overrideMimeType:"text/html;charset="+document.charset,
  4386. onload: function(d) {
  4387. let html=document.implementation.createHTMLDocument('');
  4388. html.documentElement.innerHTML = d.responseText;
  4389. self.curPage=html;
  4390. let imgs=html.querySelectorAll('img');
  4391. var container = document.querySelector('.pv-gallery-container'),
  4392. preloadContainer = document.querySelector('.pv-gallery-preloaded-img-container');
  4393. imgs = Array.prototype.slice.call(imgs).filter(function(img){
  4394. return !(container.contains(img) || (preloadContainer&&preloadContainer.contains(img)));
  4395. });
  4396. imgs.forEach(function(img) {
  4397. pretreatment(img);
  4398. if(!img.src)return;
  4399. var isrc=img.src.trim();
  4400. if(!isrc)return;
  4401. isrc=self.canonicalUri(isrc);
  4402. if (self._dataCache[isrc]) return;
  4403. self._dataCache[isrc] = true;
  4404. var nimg = new Image();
  4405. self.londingImgNum++;
  4406. nimg.src = isrc;
  4407. nimg.onload=function(){
  4408. self.londingImgNum--;
  4409. self.pageImgReady();
  4410. var result = findPic(this);
  4411. if (result) {
  4412. self.data.push(result);
  4413. self._appendThumbSpans([result]);
  4414. self.loadThumb();
  4415. }
  4416. };
  4417. nimg.onerror=function(){
  4418. self.londingImgNum--;
  4419. self.pageImgReady();
  4420. };
  4421. });
  4422. if(prefs.gallery.loadAll)self.nextPage();
  4423. else loadOver();
  4424. },
  4425. onerror: function(e) {
  4426. if(prefs.gallery.loadAll)self.nextPage();
  4427. else loadOver();
  4428. }
  4429. });
  4430. },
  4431. runOnce:function(){//运行一次来获取某些数据。
  4432. var thumbSpanCS=unsafeWindow.getComputedStyle(this.selected);
  4433. this.thumbSpanOuterSize=this.isHorizontal?
  4434. this.selected.offsetWidth + parseFloat(thumbSpanCS.marginLeft) + parseFloat(thumbSpanCS.marginRight) :
  4435. this.selected.offsetHeight + parseFloat(thumbSpanCS.marginTop) + parseFloat(thumbSpanCS.marginBottom);
  4436.  
  4437.  
  4438.  
  4439. this.runOnce=function(){
  4440. };
  4441. },
  4442.  
  4443. minimize:function(){
  4444. this.close();
  4445. this.maximizeTrigger.style.display='block';
  4446. this.minimized=true;
  4447. },
  4448. navigateToImg:function(targetImg){
  4449. targetImg.scrollIntoView();//先调用原方法,可以让overflow hidden的滚动出来。
  4450.  
  4451. //让图片近可能的居中
  4452. var imgBCRect=getContentClientRect(targetImg);
  4453. var wSize=getWindowSize();
  4454.  
  4455. window.scrollBy(imgBCRect.left - (wSize.w - imgBCRect.width)/2,
  4456. imgBCRect.top - (wSize.h - imgBCRect.height)/2);
  4457.  
  4458. },
  4459. switchThumbVisible:function(){
  4460. var style=this.thumbVisibleStyle;
  4461. var count=0;
  4462. var styleText=[];
  4463. var iStatisCopy=this.iStatisCopy;
  4464. var iStatisCopy_i;
  4465.  
  4466. for(var i in iStatisCopy){
  4467. if(!iStatisCopy.hasOwnProperty(i))continue;
  4468. iStatisCopy_i=iStatisCopy[i];
  4469. if(iStatisCopy_i.shown){
  4470. count+=iStatisCopy_i.count;
  4471. }else{
  4472. styleText.push('.pv-gallery-sidebar-thumb-container[data-type="'+i+'"]');
  4473. };
  4474. };
  4475.  
  4476. //写入style;
  4477. style.textContent=styleText.join(',') + '{\
  4478. display:none !important;\
  4479. }';
  4480.  
  4481. //初始化缩略图区的滚动条
  4482. this.thumbScrollbar.reset();
  4483. this.arrowVisib();
  4484.  
  4485. //载入缩略图
  4486. this.loadThumb();
  4487. },
  4488. forceRepaint:function(){//解决opera的fixed元素,当滚动条不再最高处的时候,不重绘fixed元素的问题。
  4489. clearTimeout(this.forceRepaintTimer);
  4490. var self=this;
  4491. this.forceRepaintTimer=setTimeout(function(){
  4492. if(envir.opera){
  4493. self.forceRepaintTimes % 2 ==0 ? window.scrollBy(0,1) : window.scrollBy(0,-1);
  4494. self.forceRepaintTimes++;
  4495. };
  4496. },333);
  4497. },
  4498. resizeHandler:function(){//窗口变化时,调整一些东西。
  4499. this.thumbScrollbar.reset();
  4500. //this.selectedIntoView();
  4501. this.fitToScreen();
  4502. this.loadThumb();
  4503. },
  4504. _isLastSpan: function(span) { // 用于判断是否自动重载,是否是最后几个图片
  4505. if (this.selected.clientWidth == 0) return false;
  4506. if (!span) return true;
  4507.  
  4508. var index = Array.prototype.slice.call(this.imgSpans).indexOf(span);
  4509. if (index != -1) {
  4510. var total = this.imgSpans.length;
  4511. if (total - index < prefs.gallery.scrollEndAndLoad_num) {
  4512. return true;
  4513. }
  4514. }
  4515. },
  4516. arrowVisib:function(){//当当前选择元素的前面或者后面没有元素的时候隐藏控制箭头
  4517.  
  4518. var icps=this.eleMaps['img-controler-pre'].style;
  4519. var icns=this.eleMaps['img-controler-next'].style;
  4520. var scps=this.eleMaps['sidebar-controler-pre'].style;
  4521. var scns=this.eleMaps['sidebar-controler-next'].style;
  4522.  
  4523. //下一张的箭头
  4524. var nextSpan = this.getThumSpan();
  4525. if (nextSpan) {
  4526. icns.display='';
  4527. scns.display='';
  4528. }else{
  4529. icns.display='none';
  4530. scns.display='none';
  4531. };
  4532.  
  4533. // 最后几张图片,滚到底部添加新的图片
  4534. if (prefs.gallery.scrollEndAndLoad && this._isLastSpan(nextSpan)) {
  4535. this.scrollToEndAndReload();
  4536. }
  4537.  
  4538. //上一张的箭头
  4539. if(this.getThumSpan(true)){
  4540. icps.display='';
  4541. scps.display='';
  4542. }else{
  4543. icps.display='none';
  4544. scps.display='none';
  4545. };
  4546. },
  4547. simpleSlideShow:function(backward,interval){
  4548. clearInterval(this.slideShowInterval);//幻灯播放,只允许存在一个,否则得乱套
  4549.  
  4550. var self=this;
  4551. var slideShowInterval=setInterval(function(){
  4552. var before=self.selected;
  4553. backward ? self.selectPrevious() : self.selectNext();
  4554. if(before == self.selected){//没有下一个元素了。。
  4555. stop();
  4556. };
  4557. },(interval? interval : 800));
  4558.  
  4559. this.slideShowInterval=slideShowInterval;
  4560.  
  4561. function stop(){
  4562. clearInterval(slideShowInterval);
  4563. };
  4564.  
  4565. return stop;
  4566. },
  4567.  
  4568. reload: function() { // 重新加载所有图片到库里面
  4569. // 函数在 LoadingAnimC 中
  4570. var data = this.getAllValidImgs();
  4571. // 设置当前选中的图片
  4572. data.target = {
  4573. src: this.selected.dataset.src
  4574. };
  4575.  
  4576. this.close(true);
  4577.  
  4578. this.load(data, null, true);
  4579. },
  4580. reloadNew: function() { // 加载新的图片到库里面
  4581. var newer = true;
  4582. var data = this.getAllValidImgs(newer);
  4583. if (data.length) {
  4584. this._appendThumbSpans(data);
  4585. }
  4586. },
  4587. getAllValidImgs:function(newer){
  4588. var validImgs = [];
  4589. var imgs = document.getElementsByTagName('img'),
  4590. container = document.querySelector('.pv-gallery-container'),
  4591. preloadContainer = document.querySelector('.pv-gallery-preloaded-img-container');
  4592.  
  4593. imgs = Array.prototype.slice.call(imgs);
  4594. arrayFn.forEach.call(document.querySelectorAll("iframe"),function(iframe){
  4595. if(iframe.src && iframe.src.replace(/\/[^\/]*$/,"").indexOf(location.hostname)!=-1)
  4596. try{
  4597. arrayFn.forEach.call(iframe.contentWindow.document.getElementsByTagName('img'),function(img){
  4598. imgs.push(img);
  4599. });
  4600. }catch(e){
  4601. debug(e.toString());
  4602. }
  4603. });
  4604. var bgReg=/.*url\(\s*["']?(.+?)["']?\s*\)/i;
  4605. var bgImgs=Array.from(document.querySelectorAll('*'))
  4606. .reduce((total, node) => {
  4607. if(node.nodeName != "IMG" && !node.src && (!node.className || !node.className.indexOf || node.className.indexOf("pv-")==-1)){
  4608. let prop = unsafeWindow.getComputedStyle(node).backgroundImage;
  4609. let match = bgReg.exec(prop)
  4610. if (match) {
  4611. node.src=match[1];
  4612. total.push(node);
  4613. }
  4614. }
  4615. return total;
  4616. }, []);
  4617. if(bgImgs)imgs=imgs.concat(bgImgs);
  4618. // 排除库里面的图片
  4619. imgs = imgs.filter(function(img){
  4620. return !(container.contains(img) || (preloadContainer&&preloadContainer.contains(img)));
  4621. });
  4622.  
  4623. // 已经在图库里面的
  4624. var self = this;
  4625. imgs.forEach(function(img) {
  4626. pretreatment(img);
  4627. if(!img.src) return;
  4628. if (newer && self._dataCache[img.src]) return;
  4629.  
  4630. var result = findPic(img);
  4631. if (result) {
  4632. validImgs.push(result);
  4633. self.data.push(result);
  4634. }
  4635.  
  4636. self._dataCache[img.src] = true;
  4637. });
  4638.  
  4639. return validImgs;
  4640. },
  4641. scrollToEndAndReload: function() { // 滚动主窗口到最底部,然后自动重载库的图片
  4642.  
  4643. window.scrollTo(0, 9999999);
  4644.  
  4645. var self = this;
  4646. clearTimeout(self.reloadTimeout);
  4647. self.reloadTimeout = setTimeout(function(){
  4648. // self.reload();
  4649. self.reloadNew();
  4650. }, 1000);
  4651. },
  4652. exportImages: function () { // 导出所有图片到新窗口
  4653. var nodes = document.querySelectorAll('.pv-gallery-sidebar-thumb-container[data-src]'),i;
  4654. //var arr = Array.prototype.map.call(nodes, function(node){
  4655. // if(unsafeWindow.getComputedStyle(node).display=="none")return "";
  4656. // else return '<div><img src=' + node.dataset.src + '></div>'
  4657. //});
  4658.  
  4659. var arr=[];
  4660. for (i = 0; i < nodes.length; ++i) {
  4661. if(unsafeWindow.getComputedStyle(nodes[i]).display=="none")arr.push("");
  4662. else arr.push('<div><img src=' + nodes[i].dataset.src + '></div>');
  4663. }
  4664.  
  4665. var title = document.title;
  4666.  
  4667. var html = '\
  4668. <head>\
  4669. <title>' + title + ' '+i18n("exportImages")+'</title>\
  4670. <style>\
  4671. .toTop{width:28px;height:28px;border-radius:14px;position: fixed;right:2px;bottom: 2px;cursor: pointer;background-color:#000;opacity:.3;padding:0em!important;border:0px!important;z-index:1}\
  4672. .toTop:hover{opacity:1}\
  4673. .toTop>span{height:28px;line-height:28px;display:block;color:#FFF;text-align:center;font-size:20px;}\
  4674. .grid{-moz-column-count:4;-webkit-column-count:4;column-count:4;-moz-column-gap: 1em;-webkit-column-gap: 1em;column-gap: 1em;}\
  4675. .grid>div{padding: 1em;margin: 0 0 1em 0;-moz-page-break-inside: avoid;-webkit-column-break-inside: avoid;break-inside: avoid;}\
  4676. .grid>div>img{width: 100%;margin-bottom:10px;}\
  4677. .list>div {text-align:center;}\
  4678. .list>div>img { max-width: 100%; }\
  4679. .gridBig{margin: 0px;}\
  4680. .gridBig>div { float: left;margin: 0px 0px 1px 1px;}\
  4681. .gridBig>div>img { max-width: 100%; }\
  4682. .select{opacity: 0.8;border: 5px solid red!important;}\
  4683. body>div{border: 5px solid black;margin: 1px;}\
  4684. body>div:hover{border: 5px solid #dbdbdb;}\
  4685. body>div{position:relative;}\
  4686. </style>\
  4687. </head>\
  4688. <span class="toTop">\
  4689. <span>↑</span>\
  4690. </span>\
  4691. <body class="'+prefs.gallery.exportType+'">\
  4692. <p style="width:100vw;display:flex;flex-direction:column;">\
  4693. <img id="bigImg" style="pointer-events:none;position:fixed;z-index:999;width:100vw;top:0px;align-self:center;"></p>\
  4694. <p>【'+i18n("picTitle")+'】:' + title + '</p>\
  4695. <p>【'+i18n("picNum")+'】:' + nodes.length + ' <select onchange="document.body.className=this.options[this.options.selectedIndex].value"><option value="grid" '+(prefs.gallery.exportType=="grid"?"selected='selected'":"")+'>'+i18n("grid")+'</option><option value="gridBig" '+(prefs.gallery.exportType=="gridBig"?"selected='selected'":"")+'>'+i18n("gridBig")+'</option><option value="list" '+(prefs.gallery.exportType=="list"?"selected='selected'":"")+'>'+i18n("list")+'</option> </select> \
  4696. <input type="button" value="'+i18n("exportImagesUrl")+'" onclick="var imgStr=\'\',selList=document.querySelectorAll(\'.select>img\');if(selList.length==0)[].forEach.call(document.querySelectorAll(\'img\'),function(i){imgStr+=i.src+\' \\n\'});else{[].forEach.call(selList,function(i){imgStr+=i.src+\' \\n\'});}window.prompt(\''+i18n("exportImagesUrlPop")+'\',imgStr);">\
  4697. ('+i18n("picTips")+')</p><p>'+i18n("savePageTips")+"</p>";
  4698.  
  4699. html += arr.join('\n') +
  4700. '<script type="text/javascript">\
  4701. document.querySelector(".toTop").addEventListener("click", function(){\
  4702. document.body.scrollIntoView();\
  4703. });\
  4704. var bigImg=document.querySelector("#bigImg"),body=document.body;\
  4705. [].forEach.call(document.querySelectorAll("div>img"),function(i){\
  4706. i.onerror=function(e){i.style.display="none"};\
  4707. i.onmouseover=i.onmousemove=function(e){bigImg.style.top=(this.width/this.height>body.clientWidth/body.clientHeight?10+(body.clientHeight*0.95-body.clientWidth*this.height/this.width)*e.offsetY/this.height:10-(body.clientWidth*this.height/this.width-body.clientHeight*0.95)*e.offsetY/this.height);bigImg.src=e.ctrlKey?this.src:"";bigImg.style.display=e.ctrlKey?"":"none";};\
  4708. });\
  4709. [].forEach.call(document.querySelectorAll("body>div"),function(i){\
  4710. i.onclick=function(e){if(e.ctrlKey&&i.firstChild.src){window.open(i.firstChild.src,"_blank")}else{this.classList.toggle("select")}}\
  4711. });\
  4712. </script></body>';
  4713. GM_openInTab('data:text/html;charset=utf-8,' + encodeURIComponent(html));
  4714. },
  4715. copyImages: function(isAlert) {
  4716. var nodes = document.querySelectorAll('.pv-gallery-sidebar-thumb-container[data-src]');
  4717. var urls = [];
  4718. [].forEach.call(nodes, function(node){
  4719. if(unsafeWindow.getComputedStyle(node).display!="none")
  4720. urls.push(node.dataset.src);
  4721. });
  4722.  
  4723. GM_setClipboard(urls.join("\n"));
  4724.  
  4725. if (isAlert) {
  4726. GM_notification(i18n("copySuccess",urls.length));
  4727. }
  4728. },
  4729.  
  4730. Preload:function(ele,oriThis){
  4731. this.ele=ele;
  4732. this.oriThis=oriThis;//主this
  4733. this.init();
  4734. },
  4735. Scrollbar:function(scrollbar,container,isHorizontal){
  4736. this.scrollbar=scrollbar;
  4737. this.container=container;
  4738. this.isHorizontal=isHorizontal
  4739. this.init();
  4740. },
  4741.  
  4742. addStyle:function(){
  4743. var style=document.createElement('style');
  4744. style.type='text/css';
  4745. style.textContent='\
  4746. /*最外层容器*/\
  4747. .pv-gallery-container {\
  4748. position: fixed;\
  4749. top: 0;\
  4750. left: 0;\
  4751. width: 100%;\
  4752. height: 100%;\
  4753. min-width:none;\
  4754. min-height:none;\
  4755. padding: 0;\
  4756. margin: 0;\
  4757. border: none;\
  4758. z-index:2147483647;\
  4759. background-color: transparent;\
  4760. }\
  4761. /*全局border-box*/\
  4762. .pv-gallery-container span{\
  4763. -moz-box-sizing: border-box;\
  4764. box-sizing: border-box;\
  4765. line-height: 1.6;\
  4766. }\
  4767. .pv-gallery-container * {\
  4768. font-size: 14px;\
  4769. }\
  4770. /*点击还原的工具条*/\
  4771. .pv-gallery-maximize-trigger{\
  4772. position:fixed;\
  4773. bottom:15px;\
  4774. left:15px;\
  4775. display:none;\
  4776. background:#000;\
  4777. opacity:0.6;\
  4778. padding-left:10px;\
  4779. font-size:16px;\
  4780. line-height:0;\
  4781. color:white;\
  4782. cursor:pointer;\
  4783. box-shadow:3px 3px 0 0 #333;\
  4784. z-index:899999998;\
  4785. }\
  4786. .pv-gallery-maximize-trigger:hover{\
  4787. opacity:0.9;\
  4788. }\
  4789. .pv-gallery-maximize-trigger-close{\
  4790. display:inline-block;\
  4791. padding-left:10px;\
  4792. vertical-align:middle;\
  4793. height:30px;\
  4794. padding:10px 0;\
  4795. width:24px;\
  4796. background:url("'+prefs.icons.loadingCancle+'") center no-repeat;\
  4797. }\
  4798. .pv-gallery-maximize-trigger-close:hover{\
  4799. background-color:#333;\
  4800. }\
  4801. @media screen and (max-width: 800px) {\
  4802. .pv-gallery-range-box>input {\
  4803. display: none;\
  4804. }\
  4805. .pv-gallery-maximize-container{\
  4806. column-count: 2!important;\
  4807. -moz-column-count: 2!important;\
  4808. -webkit-column-count: 2!important;\
  4809. padding-top: 300px!important;\
  4810. }\
  4811. .pv-gallery-sidebar-viewmore-bottom.showmore{\
  4812. transform: scale(3.5);\
  4813. bottom: 50px;\
  4814. }\
  4815. .pv-gallery-maximize-container span>p{\
  4816. opacity: 0.6!important;\
  4817. }\
  4818. }\
  4819. /*顶栏*/\
  4820. .pv-gallery-head {\
  4821. position: absolute;\
  4822. top: 0;\
  4823. left: 0;\
  4824. width: 100%;\
  4825. flex-wrap: wrap;\
  4826. min-height: 30px;\
  4827. height: auto;\
  4828. z-index:1;\
  4829. background-color:rgb(0,0,0);\
  4830. border:none;\
  4831. border-bottom:1px solid #333333;\
  4832. text-align:right;\
  4833. line-height:0;\
  4834. font-size: 14px;\
  4835. color:#757575;\
  4836. padding-right:42px;\
  4837. display: table;\
  4838. }\
  4839. .pv-gallery-head > span{\
  4840. vertical-align:middle;\
  4841. }\
  4842. /*顶栏左边*/\
  4843. .pv-gallery-head-float-left{\
  4844. float:left;\
  4845. height:100%;\
  4846. text-align:left;\
  4847. padding-left:5px;\
  4848. display: table;\
  4849. }\
  4850. .pv-gallery-head-float-left > span{\
  4851. display:inline-block;\
  4852. height:100%;\
  4853. vertical-align:middle;\
  4854. }\
  4855. .pv-gallery-head-float-left > span > *{\
  4856. vertical-align:middle;\
  4857. }\
  4858. .pv-gallery-head-left-img-info{\
  4859. cursor:help;\
  4860. }\
  4861. .pv-gallery-head-left-img-info-description {\
  4862. margin-left: 10px;\
  4863. }\
  4864. .pv-gallery-range-box{\
  4865. display: inline-flex;\
  4866. justify-content: center;\
  4867. align-items: center;\
  4868. }\
  4869. .pv-gallery-range-box>span{\
  4870. padding: 0 5px 0 5px;\
  4871. }\
  4872. .pv-gallery-range-box>input{\
  4873. background: white;\
  4874. }\
  4875. /*顶栏里面的按钮样式-开始*/\
  4876. .pv-gallery-head-command{\
  4877. display:inline-block;\
  4878. cursor:pointer;\
  4879. height:100%;\
  4880. padding:0 8px;\
  4881. text-align:center;\
  4882. position:relative;\
  4883. z-index:1;\
  4884. vertical-align:middle;\
  4885. -o-user-select: none;\
  4886. -ms-user-select: none;\
  4887. -webkit-user-select: none;\
  4888. -moz-user-select: -moz-none;\
  4889. user-select: none;\
  4890. }\
  4891. /*辅助点击事件的生成,countdown*/\
  4892. .pv-gallery-head-command_overlayer{\
  4893. top:0;\
  4894. left:0;\
  4895. right:0;\
  4896. bottom:0;\
  4897. position:absolute;\
  4898. opacity:0;\
  4899. }\
  4900. .pv-gallery-head-command > *{\
  4901. vertical-align:middle;\
  4902. }\
  4903. .pv-gallery-head-command-close{\
  4904. position:absolute;\
  4905. top:0;\
  4906. right:0;\
  4907. width:40px;\
  4908. border-left: 1px solid #333333;\
  4909. background:transparent no-repeat center;\
  4910. background-image:url("'+prefs.icons.loadingCancle+'");\
  4911. }\
  4912. .pv-gallery-head-command-slide-show-countdown{\
  4913. font-size:0.8em;\
  4914. }\
  4915. .pv-gallery-head-command-slide-show-button{\
  4916. border-radius:36px;\
  4917. display:inline-block;\
  4918. width:18px;\
  4919. height:18px;\
  4920. border:2px solid #757575;\
  4921. margin-right:3px;\
  4922. line-height:0;\
  4923. }\
  4924. .pv-gallery-head-command-slide-show-button-inner{\
  4925. display:inline-block;\
  4926. border:none;\
  4927. border-top:4px solid transparent;\
  4928. border-bottom:4px solid transparent;\
  4929. border-left:8px solid #757575;\
  4930. vertical-align:middle;\
  4931. }\
  4932. .pv-gallery-head-command-slide-show-button-inner_stop{\
  4933. border-color:#757575;\
  4934. }\
  4935. .pv-gallery-head-command-collect-icon{\
  4936. display:inline-block;\
  4937. height:20px;\
  4938. width:20px;\
  4939. background:transparent url("' + prefs.icons.fivePointedStar + '") 0 0 no-repeat;\
  4940. }\
  4941. .pv-gallery-head-command-collect-icon ~ .pv-gallery-head-command-collect-text::after{\
  4942. content:"'+i18n("collect")+'";\
  4943. }\
  4944. .pv-gallery-head-command-collect-favorite > .pv-gallery-head-command-collect-icon{\
  4945. background-position:-40px 0 !important;\
  4946. }\
  4947. .pv-gallery-head-command-collect-favorite > .pv-gallery-head-command-collect-text::after{\
  4948. content:"'+i18n("collected")+'";\
  4949. }\
  4950. .pv-gallery-head-command-exit-collection{\
  4951. color:#939300 !important;\
  4952. display:none;\
  4953. }\
  4954. .pv-gallery-head-command:hover{\
  4955. background-color:#272727;\
  4956. color:#ccc;\
  4957. }\
  4958. /*droplist*/\
  4959. .pv-gallery-head-command-drop-list{\
  4960. position:absolute;\
  4961. right:0;\
  4962. display:none;\
  4963. box-shadow:0 0 3px #808080;\
  4964. background-color:#272727;\
  4965. line-height: 1.6;\
  4966. text-align:left;\
  4967. padding:10px;\
  4968. color:#ccc;\
  4969. margin-top:-1px;\
  4970. z-index:9;\
  4971. }\
  4972. .pv-gallery-head-command-drop-list-item{\
  4973. display:block;\
  4974. padding:2px 5px;\
  4975. cursor:pointer;\
  4976. white-space:nowrap;\
  4977. }\
  4978. .pv-gallery-head-command-drop-list-item-collect-description{\
  4979. cursor:default;\
  4980. }\
  4981. .pv-gallery-head-command-drop-list-item-collect-description > textarea{\
  4982. resize:both;\
  4983. width:auto;\
  4984. height:auto;\
  4985. }\
  4986. .pv-gallery-head-command-drop-list-item_disabled{\
  4987. color:#757575;\
  4988. }\
  4989. .pv-gallery-head-command-drop-list-item input + *{\
  4990. padding-left:3px;\
  4991. }\
  4992. .pv-gallery-head-command-drop-list-item input[type=number]{\
  4993. text-align:left;\
  4994. max-width:50px;\
  4995. height:20px;\
  4996. }\
  4997. .pv-gallery-head-command-drop-list-item input[type=checkbox]{\
  4998. width:20px\
  4999. }\
  5000. .pv-gallery-head-command-drop-list-item > * {\
  5001. vertical-align:middle;\
  5002. }\
  5003. .pv-gallery-head-command-drop-list-item label {\
  5004. font-weight: normal;\
  5005. display:inline\
  5006. }\
  5007. .pv-gallery-head-command-drop-list-item:hover{\
  5008. background-color:#404040;\
  5009. }\
  5010. /*container*/\
  5011. .pv-gallery-head-command-container{\
  5012. display:inline-block;\
  5013. height:100%;\
  5014. position:relative;\
  5015. }\
  5016. /* after伪类生成标识下拉菜单的三角图标*/\
  5017. .pv-gallery-head-command-container > .pv-gallery-head-command::after{\
  5018. content:"";\
  5019. display:inline-block;\
  5020. vertical-align:middle;\
  5021. border:none;\
  5022. border-top:7px solid #757575;\
  5023. border-left:5px solid transparent;\
  5024. border-right:5px solid transparent;\
  5025. margin-left:5px;\
  5026. -moz-transition:all 0.3s ease-in-out 0s;\
  5027. -webkit-transition:all 0.3s ease-in-out 0s;\
  5028. transition:all 0.3s ease-in-out 0s;\
  5029. }\
  5030. .pv-gallery-head-command-container:hover{\
  5031. box-shadow:0 0 3px #808080;\
  5032. }\
  5033. .pv-gallery-head-command-container:hover > .pv-gallery-head-command{\
  5034. background-color:#272727;\
  5035. color:#ccc;\
  5036. }\
  5037. .pv-gallery-head-command-container:hover > .pv-gallery-head-command::after{\
  5038. -webkit-transform:rotate(180deg);\
  5039. -moz-transform:rotate(180deg);\
  5040. transform:rotate(180deg);\
  5041. border-top:7px solid #ccc;\
  5042. }\
  5043. .pv-gallery-head-command-container:hover .pv-gallery-head-command-collect-icon{\
  5044. background-position:-20px 0;\
  5045. }\
  5046. .pv-gallery-head-command-container:hover .pv-gallery-head-command-slide-show-button{\
  5047. border-color:#ccc;\
  5048. }\
  5049. .pv-gallery-head-command-container:hover .pv-gallery-head-command-slide-show-button-inner{\
  5050. border-left-color:#ccc;\
  5051. }\
  5052. .pv-gallery-head-command-container:hover .pv-gallery-head-command-slide-show-button-inner_stop{\
  5053. border-color:#ccc;\
  5054. }\
  5055. .pv-gallery-head-command-container:hover > .pv-gallery-head-command-drop-list{\
  5056. display:block;\
  5057. }\
  5058. /*顶栏里面的按钮样式-结束*/\
  5059. .pv-gallery-body {\
  5060. display: block;\
  5061. height: 100%;\
  5062. width: 100%;\
  5063. margin: 0;\
  5064. padding: 0;\
  5065. border: none;\
  5066. border-top: 30px solid transparent;\
  5067. position: relative;\
  5068. background-clip: padding-box;\
  5069. z-index:0;\
  5070. }\
  5071. .pv-gallery-img-container {\
  5072. display: block;\
  5073. padding: 0;\
  5074. margin: 0;\
  5075. border: none;\
  5076. height: 100%;\
  5077. width: 100%;\
  5078. background-clip: padding-box;\
  5079. background-color: rgba(20,20,20,0.96);\
  5080. position:relative;\
  5081. }\
  5082. .pv-gallery-img-container-top {\
  5083. border-top: '+ prefs.gallery.sidebarSize +'px solid transparent;\
  5084. }\
  5085. .pv-gallery-img-container-right {\
  5086. border-right: '+ prefs.gallery.sidebarSize +'px solid transparent;\
  5087. }\
  5088. .pv-gallery-img-container-bottom {\
  5089. border-bottom: '+ prefs.gallery.sidebarSize +'px solid transparent;\
  5090. }\
  5091. .pv-gallery-img-container-left {\
  5092. border-left: '+ prefs.gallery.sidebarSize +'px solid transparent;\
  5093. }\
  5094. /*大图区域的切换控制按钮*/\
  5095. .pv-gallery-img-controler{\
  5096. position:absolute;\
  5097. top:50%;\
  5098. height:60px;\
  5099. width:50px;\
  5100. margin-top:-30px;\
  5101. cursor:pointer;\
  5102. opacity:0.3;\
  5103. z-index:1;\
  5104. }\
  5105. .pv-gallery-img-controler-pre{\
  5106. background:rgba(70,70,70,0.5) url("'+prefs.icons.arrowLeft+'") no-repeat center;\
  5107. left:10px;\
  5108. }\
  5109. .pv-gallery-img-controler-next{\
  5110. background:rgba(70,70,70,0.5) url("'+prefs.icons.arrowRight+'") no-repeat center;\
  5111. right:10px;\
  5112. }\
  5113. .pv-gallery-img-controler:hover{\
  5114. background-color:rgba(140,140,140,0.5);\
  5115. opacity:0.9;\
  5116. z-index:2;\
  5117. }\
  5118. /*滚动条样式--开始*/\
  5119. .pv-gallery-scrollbar-h,\
  5120. .pv-gallery-scrollbar-v{\
  5121. display:none;\
  5122. z-index:1;\
  5123. position:absolute;\
  5124. margin:0;\
  5125. padding:0;\
  5126. border:none;\
  5127. }\
  5128. .pv-gallery-scrollbar-h{\
  5129. bottom:10px;\
  5130. left:0;\
  5131. right:0;\
  5132. height:10px;\
  5133. margin:0 2px;\
  5134. }\
  5135. .pv-gallery-scrollbar-v{\
  5136. top:0;\
  5137. bottom:0;\
  5138. right:10px;\
  5139. width:10px;\
  5140. margin:2px 0;\
  5141. }\
  5142. .pv-gallery-scrollbar-h:hover{\
  5143. height:25px;\
  5144. bottom:0;\
  5145. }\
  5146. .pv-gallery-scrollbar-v:hover{\
  5147. width:25px;\
  5148. }\
  5149. .pv-gallery-scrollbar-h:hover,\
  5150. .pv-gallery-scrollbar-v:hover{\
  5151. background-color:rgba(100,100,100,0.9);\
  5152. z-index:2;\
  5153. }\
  5154. .pv-gallery-scrollbar-h-track,\
  5155. .pv-gallery-scrollbar-v-track{\
  5156. position:absolute;\
  5157. top:0;\
  5158. left:0;\
  5159. right:0;\
  5160. bottom:0;\
  5161. background-color:rgba(100,100,100,0.3);\
  5162. border:2px solid transparent;\
  5163. }\
  5164. .pv-gallery-scrollbar-h-handle,\
  5165. .pv-gallery-scrollbar-v-handle{\
  5166. position:absolute;\
  5167. background-color:rgba(0,0,0,0.5);\
  5168. }\
  5169. .pv-gallery-scrollbar-h-handle{\
  5170. height:100%;\
  5171. }\
  5172. .pv-gallery-scrollbar-v-handle{\
  5173. width:100%;\
  5174. }\
  5175. .pv-gallery-scrollbar-h-handle:hover,\
  5176. .pv-gallery-scrollbar-v-handle:hover{\
  5177. background-color:rgba(80,33,33,1);\
  5178. border: 2px solid #585757;\
  5179. }\
  5180. .pv-gallery-scrollbar-h-handle:active,\
  5181. .pv-gallery-scrollbar-v-handle:active{\
  5182. background-color:rgba(57,26,26,1);\
  5183. border: 2px solid #878484;\
  5184. }\
  5185. /*滚动条样式--结束*/\
  5186. .pv-gallery-img-content{\
  5187. display:block;\
  5188. width:100%;\
  5189. height:100%;\
  5190. overflow:hidden;\
  5191. text-align:center;\
  5192. padding:0;\
  5193. border:none;\
  5194. margin:0;\
  5195. line-height:0;\
  5196. font-size:0;\
  5197. white-space:nowrap;\
  5198. }\
  5199. .pv-gallery-img-parent{\
  5200. display:inline-block;\
  5201. vertical-align:middle;\
  5202. line-height:0;\
  5203. }\
  5204. .pv-gallery-img_broken{\
  5205. display:none;\
  5206. cursor:pointer;\
  5207. }\
  5208. .pv-gallery-img{\
  5209. position:relative;\/*辅助e.layerX,layerY*/\
  5210. display:inline-block;\
  5211. vertical-align:middle;\
  5212. width:auto;\
  5213. height:auto;\
  5214. padding:0;\
  5215. border:5px solid #313131;\
  5216. margin:10px;\
  5217. opacity:0.6;\
  5218. -webkit-transform:scale(1.2);\
  5219. -moz-transform:scale(1.2);\
  5220. transform:scale(1.2);\
  5221. '+
  5222. (prefs.gallery.transition ? ('\
  5223. -webkit-transition: opacity 0.15s ease-in-out,\
  5224. -webkit-transform 0.1s ease-in-out;\
  5225. -moz-transition: opacity 0.15s ease-in-out,\
  5226. -moz-transform 0.1s ease-in-out;\
  5227. transition: opacity 0.15s ease-in-out,\
  5228. transform 0.1s ease-in-out;\
  5229. ') : '') + '\
  5230. }\
  5231. .pv-gallery-img_zoom-out{\
  5232. cursor:'+support.cssCursorValue.zoomOut+';\
  5233. }\
  5234. .pv-gallery-img_zoom-in{\
  5235. cursor:'+support.cssCursorValue.zoomIn+';\
  5236. }\
  5237. .pv-gallery-sidebar-toggle{\
  5238. position:absolute;\
  5239. line-height:0;\
  5240. text-align:center;\
  5241. background-color:rgb(0,0,0);\
  5242. color:#757575;\
  5243. white-space:nowrap;\
  5244. cursor:pointer;\
  5245. z-index:1;\
  5246. display:none;\
  5247. }\
  5248. .pv-gallery-sidebar-viewmore{\
  5249. position:absolute;\
  5250. line-height:0;\
  5251. text-align:center;\
  5252. background-color:#000000;\
  5253. color:#757575;\
  5254. white-space:nowrap;\
  5255. cursor:pointer;\
  5256. z-index:1;\
  5257. display:none;\
  5258. height: 30px;\
  5259. width:30px;\
  5260. border-radius: 15px;\
  5261. line-height: 2 !important;\
  5262. font-family: auto;\
  5263. }\
  5264. .pv-gallery-maximize-container{\
  5265. column-count: 5;\
  5266. -moz-column-count: 5;\
  5267. -webkit-column-count: 5;\
  5268. width: 100%;\
  5269. display: block;\
  5270. background: black;\
  5271. padding-top: 30px;\
  5272. }\
  5273. .pv-gallery-maximize-container span{\
  5274. -moz-page-break-inside: avoid;\
  5275. -webkit-column-break-inside: avoid;\
  5276. break-inside: avoid;\
  5277. float: left;\
  5278. margin-bottom: 15px;\
  5279. margin-right: 15px;\
  5280. overflow: hidden;\
  5281. position: relative;\
  5282. }\
  5283. .pv-gallery-maximize-container img{\
  5284. width:100%;\
  5285. transition: transform .3s ease 0s;\
  5286. transform: scale3d(1, 1, 1);\
  5287. cursor: zoom-in;\
  5288. }\
  5289. .pv-gallery-maximize-container img:hover {\
  5290. transform: scale3d(1.1, 1.1, 1.1);\
  5291. opacity: .9;\
  5292. }\
  5293. .pv-gallery-maximize-container span>p{\
  5294. position: absolute;\
  5295. bottom: 0;\
  5296. width: 100%;\
  5297. height: 35px;\
  5298. max-height: 40%;\
  5299. font-size: 18px;\
  5300. line-height: 40px;\
  5301. text-align: center;\
  5302. background: #000;\
  5303. color: #fff;\
  5304. opacity: 0;\
  5305. left: 0;\
  5306. user-select: none;\
  5307. cursor: pointer;\
  5308. word-break: break-all;\
  5309. display: inline;\
  5310. }\
  5311. .pv-gallery-maximize-container span:hover>p{\
  5312. opacity: 0.6;\
  5313. }\
  5314. .pv-gallery-maximize-container span>p:hover{\
  5315. color:red;\
  5316. font-weight:bold;\
  5317. }\
  5318. .pv-gallery-maximize-scroll{\
  5319. overflow-y: scroll;\
  5320. height: 100%;\
  5321. width: 100%;\
  5322. position: absolute;\
  5323. visibility: hidden;\
  5324. top: 0;\
  5325. left: 0;\
  5326. }\
  5327. .pv-gallery-sidebar-toggle:hover,.pv-gallery-sidebar-viewmore:hover{\
  5328. color:#ccc;\
  5329. }\
  5330. .pv-gallery-sidebar-toggle-h{\
  5331. width:80px;\
  5332. margin-left:-40px;\
  5333. left:50%;\
  5334. }\
  5335. .pv-gallery-sidebar-viewmore-h{\
  5336. margin-left:-15px;\
  5337. left:50%;\
  5338. }\
  5339. .pv-gallery-sidebar-toggle-v{\
  5340. height:80px;\
  5341. margin-top:-40px;\
  5342. top:50%;\
  5343. }\
  5344. .pv-gallery-sidebar-viewmore-v{\
  5345. height:30px;\
  5346. top:6%;\
  5347. }\
  5348. .pv-gallery-sidebar-toggle-top{\
  5349. top:-5px;\
  5350. }\
  5351. .pv-gallery-sidebar-viewmore-top{\
  5352. top:15px;\
  5353. }\
  5354. .pv-gallery-sidebar-toggle-right,.pv-gallery-sidebar-viewmore-right{\
  5355. right:-5px;\
  5356. }\
  5357. .pv-gallery-sidebar-toggle-bottom{\
  5358. bottom:-5px;\
  5359. }\
  5360. .pv-gallery-sidebar-viewmore-bottom{\
  5361. display: block;\
  5362. background-color: #000000;\
  5363. bottom:12px;\
  5364. }\
  5365. .pv-gallery-sidebar-viewmore-bottom.showmore{\
  5366. background-color: rgb(42, 42, 42);\
  5367. }\
  5368. .pv-gallery-sidebar-toggle-left,.pv-gallery-sidebar-viewmore-left{\
  5369. left:-5px;\
  5370. }\
  5371. .pv-gallery-sidebar-toggle-content{\
  5372. display:inline-block;\
  5373. vertical-align:middle;\
  5374. white-space:normal;\
  5375. word-wrap:break-word;\
  5376. overflow-wrap:break-word;\
  5377. line-height:1.1;\
  5378. font-size:12px;\
  5379. text-align:center;\
  5380. margin-bottom:8px;\
  5381. }\
  5382. .pv-gallery-sidebar-viewmore-content{\
  5383. display:inline-block;\
  5384. vertical-align:middle;\
  5385. white-space:normal;\
  5386. word-wrap:break-word;\
  5387. overflow-wrap:break-word;\
  5388. line-height:1.1;\
  5389. font-size:16px;\
  5390. text-align:center;\
  5391. }\
  5392. .pv-gallery-sidebar-toggle-content-v,.pv-gallery-sidebar-viewmore-content-v{\
  5393. width:1.1em;\
  5394. }\
  5395. /*侧边栏开始*/\
  5396. .pv-gallery-sidebar-container {\
  5397. position: absolute;\
  5398. background-color:rgb(0,0,0);\
  5399. padding:5px;\
  5400. border:none;\
  5401. margin:none;\
  5402. text-align:center;\
  5403. line-height:0;\
  5404. white-space:nowrap;\
  5405. -o-user-select: none;\
  5406. -webkit-user-select: none;\
  5407. -moz-user-select: -moz-none;\
  5408. user-select: none;\
  5409. }\
  5410. .pv-gallery-sidebar-container-h {\
  5411. height: '+ prefs.gallery.sidebarSize +'px;\
  5412. width: 100%;\
  5413. }\
  5414. .pv-gallery-sidebar-container-v {\
  5415. width: '+ prefs.gallery.sidebarSize +'px;\
  5416. height: 100%;\
  5417. }\
  5418. .pv-gallery-sidebar-container-top {\
  5419. top: 0;\
  5420. left: 0;\
  5421. border-bottom:1px solid #333333;\
  5422. }\
  5423. .pv-gallery-sidebar-container-right {\
  5424. top: 0;\
  5425. right: 0;\
  5426. border-left:1px solid #333333;\
  5427. }\
  5428. .pv-gallery-sidebar-container-bottom {\
  5429. bottom: 0;\
  5430. left: 0;\
  5431. border-top:1px solid #333333;\
  5432. }\
  5433. .pv-gallery-sidebar-container-left {\
  5434. top: 0;\
  5435. left: 0;\
  5436. border-right:1px solid #333333;\
  5437. }\
  5438. .pv-gallery-sidebar-content {\
  5439. display: inline-block;\
  5440. margin: 0;\
  5441. padding: 0;\
  5442. border: none;\
  5443. background-clip: padding-box;\
  5444. vertical-align:middle;\
  5445. position:relative;\
  5446. text-align:left;\
  5447. }\
  5448. .pv-gallery-sidebar-content-h {\
  5449. height: 100%;\
  5450. width: 90%;\
  5451. border-left: 40px solid transparent;\
  5452. border-right: 40px solid transparent;\
  5453. }\
  5454. .pv-gallery-sidebar-content-v {\
  5455. height: 90%;\
  5456. width: 100%;\
  5457. border-top: 40px solid transparent;\
  5458. border-bottom: 40px solid transparent;\
  5459. }\
  5460. .pv-gallery-sidebar-controler{\
  5461. cursor:pointer;\
  5462. position:absolute;\
  5463. background:rgba(255,255,255,0.1) no-repeat center;\
  5464. }\
  5465. .pv-gallery-sidebar-controler:hover{\
  5466. background-color:rgba(255,255,255,0.3);\
  5467. }\
  5468. .pv-gallery-sidebar-controler-pre-h,\
  5469. .pv-gallery-sidebar-controler-next-h{\
  5470. top:0;\
  5471. width:36px;\
  5472. height:100%;\
  5473. }\
  5474. .pv-gallery-sidebar-controler-pre-v,\
  5475. .pv-gallery-sidebar-controler-next-v{\
  5476. left:0;\
  5477. width:100%;\
  5478. height:36px;\
  5479. }\
  5480. .pv-gallery-sidebar-controler-pre-h {\
  5481. left: -40px;\
  5482. background-image: url("'+prefs.icons.arrowLeft+'");\
  5483. }\
  5484. .pv-gallery-sidebar-controler-next-h {\
  5485. right: -40px;\
  5486. background-image: url("'+prefs.icons.arrowRight+'");\
  5487. }\
  5488. .pv-gallery-sidebar-controler-pre-v {\
  5489. top: -40px;\
  5490. background-image: url("'+prefs.icons.arrowTop+'");\
  5491. }\
  5492. .pv-gallery-sidebar-controler-next-v {\
  5493. bottom: -40px;\
  5494. background-image: url("'+prefs.icons.arrowBottom+'");\
  5495. }\
  5496. .pv-gallery-sidebar-thumbnails-container {\
  5497. display: block;\
  5498. overflow: hidden;\
  5499. height: 100%;\
  5500. width: 100%;\
  5501. margin:0;\
  5502. border:none;\
  5503. padding:0;\
  5504. line-height:0;\
  5505. position:relative;\
  5506. }\
  5507. .pv-gallery-sidebar-thumbnails-container span{\
  5508. vertical-align:middle;\
  5509. }\
  5510. .pv-gallery-sidebar-thumbnails-container-h{\
  5511. border-left:1px solid #464646;\
  5512. border-right:1px solid #464646;\
  5513. white-space:nowrap;\
  5514. }\
  5515. .pv-gallery-sidebar-thumbnails-container-v{\
  5516. border-top:1px solid #464646;\
  5517. border-bottom:1px solid #464646;\
  5518. white-space:normal;\
  5519. }\
  5520. .pv-gallery-sidebar-thumbnails-container-top {\
  5521. padding-bottom:5px;\
  5522. }\
  5523. .pv-gallery-sidebar-thumbnails-container-right {\
  5524. padding-left:5px;\
  5525. }\
  5526. .pv-gallery-sidebar-thumbnails-container-bottom {\
  5527. padding-top:5px;\
  5528. }\
  5529. .pv-gallery-sidebar-thumbnails-container-left {\
  5530. padding-right:5px;\
  5531. }\
  5532. .pv-gallery-sidebar-thumb-container {\
  5533. display:inline-block;\
  5534. text-align: center;\
  5535. border:2px solid rgb(52,52,52);\
  5536. cursor:pointer;\
  5537. position:relative;\
  5538. padding:2px;\
  5539. font-size:0;\
  5540. line-height:0;\
  5541. white-space:nowrap;\
  5542. vertical-align: middle;\
  5543. top:0;\
  5544. left:0;\
  5545. -webkit-transition:all 0.2s ease-in-out;\
  5546. transition:all 0.2s ease-in-out;\
  5547. }\
  5548. .pv-gallery-sidebar-thumbnails-container-h .pv-gallery-sidebar-thumb-container {\
  5549. margin:0 2px;\
  5550. height:100%;\
  5551. }\
  5552. .pv-gallery-sidebar-thumbnails-container-v .pv-gallery-sidebar-thumb-container {\
  5553. margin:2px 0;\
  5554. width:100%;\
  5555. }\
  5556. .pv-gallery-sidebar-thumbnails_hide-span > .pv-gallery-sidebar-thumb-container {\
  5557. display:none;\
  5558. }\
  5559. .pv-gallery-sidebar-thumb-container:hover {\
  5560. border:2px solid rgb(57,149,211);\
  5561. }\
  5562. .pv-gallery-sidebar-thumb_selected {\
  5563. border:2px solid rgb(229,59,62);\
  5564. }\
  5565. .pv-gallery-sidebar-thumb_selected-top {\
  5566. top:5px;\
  5567. }\
  5568. .pv-gallery-sidebar-thumb_selected-right {\
  5569. left:-5px;\
  5570. }\
  5571. .pv-gallery-sidebar-thumb_selected-bottom {\
  5572. top:-5px;\
  5573. }\
  5574. .pv-gallery-sidebar-thumb_selected-left {\
  5575. left:5px;\
  5576. }\
  5577. .pv-gallery-sidebar-thumb-loading{\
  5578. position:absolute;\
  5579. top:0;\
  5580. left:0;\
  5581. text-align:center;\
  5582. width:100%;\
  5583. height:100%;\
  5584. display:none;\
  5585. opacity:0.6;\
  5586. background:black url("'+ prefs.icons.loading + '") no-repeat center ;\
  5587. }\
  5588. .pv-gallery-sidebar-thumb-loading:hover{\
  5589. opacity:0.8;\
  5590. }\
  5591. .pv-gallery-sidebar-thumb {\
  5592. display: inline-block;\
  5593. vertical-align: middle;\
  5594. max-width: 100% !important;\
  5595. max-height: 100% !important;\
  5596. height: auto !important;\
  5597. width: auto !important;\
  5598. }\
  5599. .pv-gallery-vertical-align-helper{\
  5600. display:inline-block;\
  5601. vertical-align:middle;\
  5602. width:0;\
  5603. height:100%;\
  5604. margin:0;\
  5605. border:0;\
  5606. padding:0;\
  5607. visibility:hidden;\
  5608. white-space:nowrap;\
  5609. background-color:red;\
  5610. }\
  5611. ';
  5612. var head=document.head;
  5613. head.appendChild(style);
  5614. this.globalSSheet=style.sheet;
  5615.  
  5616. var style2=document.createElement('style');
  5617. this.thumbVisibleStyle=style2;
  5618. style2.type='text/css';
  5619. head.appendChild(style2);
  5620.  
  5621. // 让 description 的文字内容溢出用点点点(...)省略号表示
  5622. // .pv-gallery-head-left-img-info-description {
  5623. // overflow: hidden;
  5624. // text-overflow: ellipsis;
  5625. // white-space: nowrap;
  5626. // width: 27em;
  5627. // }
  5628. },
  5629.  
  5630. };
  5631.  
  5632.  
  5633. GalleryC.prototype.Preload.prototype={//预读对象
  5634. init:function(){
  5635. if(!this.container){//预读的图片都仍里面
  5636. var div=document.createElement('div');
  5637. div.className='pv-gallery-preloaded-img-container';
  5638. div.style.display='none';
  5639. document.body.appendChild(div);
  5640. GalleryC.prototype.Preload.prototype.container=div;
  5641. };
  5642. this.max=prefs.gallery.max;
  5643. this.nextNumber=0;
  5644. this.nextEle=this.ele;
  5645. this.preNumber=0;
  5646. this.preEle=this.ele;
  5647. this.direction='pre';
  5648. },
  5649. preload:function(){
  5650. var ele=this.getPreloadEle();
  5651. if(!ele){
  5652. return;
  5653. };
  5654.  
  5655. var self=this;
  5656. this.imgReady=imgReady(dataset(ele,'src'),{
  5657. loadEnd:function(){
  5658. if(self.aborted){
  5659. return;
  5660. };
  5661. dataset(ele,'preloaded','true')
  5662. self.container.appendChild(this);
  5663. self.preload();
  5664. },
  5665. time:60 * 1000,//限时一分钟,否则强制结束并开始预读下一张。
  5666. });
  5667. },
  5668. getPreloadEle:function(){
  5669. if((this.max<=this.nextNumber && this.max<=this.preNumber) || (!this.nextEle && !this.preEle)){
  5670. return;
  5671. };
  5672. var ele=this.direction=='pre'? this.getNext() : this.getPrevious();
  5673. if(ele && !dataset(ele,'preloaded')){
  5674. return ele;
  5675. }else{
  5676. return this.getPreloadEle();
  5677. };
  5678. },
  5679. getNext:function(){
  5680. this.nextNumber++;
  5681. this.direction='next';
  5682. if(!this.nextEle)return;
  5683. return (this.nextEle = this.oriThis.getThumSpan(false,this.nextEle));
  5684. },
  5685. getPrevious:function(){
  5686. this.preNumber++;
  5687. this.direction='pre';
  5688. if(!this.preEle)return;
  5689. return (this.preEle = this.oriThis.getThumSpan(true,this.preEle));
  5690. },
  5691. abort:function(){
  5692. this.aborted=true;
  5693. if(this.imgReady){
  5694. this.imgReady.abort();
  5695. };
  5696. },
  5697. };
  5698.  
  5699.  
  5700. GalleryC.prototype.Scrollbar.prototype={//滚动条对象
  5701. init:function(){
  5702. var bar=this.scrollbar.bar;
  5703. this.shown=bar.offsetWidth!=0;
  5704. var self=this;
  5705. bar.addEventListener('mousedown',function(e){//点击滚动条区域,该干点什么!
  5706. e.preventDefault();
  5707. var target=e.target;
  5708. var handle=self.scrollbar.handle;
  5709. var track=self.scrollbar.track;
  5710. switch(target){
  5711. case handle:{//手柄;功能,拖动手柄来滚动窗口
  5712. var pro=self.isHorizontal ? ['left','clientX'] : ['top','clientY'];
  5713. var oHOffset=parseFloat(handle.style[pro[0]]);
  5714. var oClient=e[pro[1]];
  5715.  
  5716. var moveHandler=function(e){
  5717. self.scroll(oHOffset + e[pro[1]] - oClient,true);
  5718. };
  5719. var upHandler=function(){
  5720. document.removeEventListener('mousemove',moveHandler,true);
  5721. document.removeEventListener('mouseup',upHandler,true);
  5722. };
  5723. document.addEventListener('mousemove',moveHandler,true);
  5724. document.addEventListener('mouseup',upHandler,true);
  5725. }break;
  5726. case track:{//轨道;功能,按住不放来连续滚动一个页面的距离
  5727. var pro=self.isHorizontal ? ['left','offsetX','layerX','clientWidth','offsetWidth'] : ['top' , 'offsetY' ,'layerY','clientHeight','offsetHeight'];
  5728. var clickOffset=typeof e[pro[1]]=='undefined' ? e[pro[2]] : e[pro[1]];
  5729. var handleOffset=parseFloat(handle.style[pro[0]]);
  5730. var handleSize=handle[pro[4]];
  5731. var under= clickOffset > handleOffset ;//点击在滚动手柄的下方
  5732. var containerSize=self.container[pro[3]];
  5733.  
  5734. var scroll=function(){
  5735. self.scrollBy(under? (containerSize - 10) : (-containerSize + 10));//滚动一个页面距离少一点
  5736. };
  5737. scroll();
  5738.  
  5739. var checkStop=function(){//当手柄到达点击位置时停止
  5740. var handleOffset=parseFloat(handle.style[pro[0]]);
  5741. if(clickOffset >= handleOffset && clickOffset <= (handleOffset + handleSize)){
  5742. clearTimeout(scrollTimeout);
  5743. clearInterval(scrollInterval);
  5744. };
  5745. };
  5746.  
  5747.  
  5748. var scrollInterval;
  5749. var scrollTimeout=setTimeout(function(){
  5750. scroll();
  5751. scrollInterval=setInterval(function(){
  5752. scroll();
  5753. checkStop();
  5754. },120);
  5755. checkStop();
  5756. },300);
  5757.  
  5758.  
  5759. checkStop();
  5760.  
  5761. var upHandler=function(){
  5762. clearTimeout(scrollTimeout);
  5763. clearInterval(scrollInterval);
  5764. document.removeEventListener('mouseup',upHandler,true);
  5765. };
  5766. document.addEventListener('mouseup',upHandler,true);
  5767. }break;
  5768. };
  5769.  
  5770. },true);
  5771. },
  5772. reset:function(){//判断滚动条该显示还是隐藏
  5773.  
  5774. var pro=this.isHorizontal ? ['scrollWidth','clientWidth','width'] : ['scrollHeight','clientHeight','height'];
  5775.  
  5776. //如果内容大于容器的content区域
  5777.  
  5778. var scrollSize=this.container[pro[0]];
  5779. var clientSize=this.container[pro[1]];
  5780. var scrollMax=scrollSize - clientSize;
  5781. this.scrollMax=scrollMax;
  5782. if(scrollMax>0){
  5783. this.show();
  5784. var trackSize=this.scrollbar.track[pro[1]];
  5785. this.trackSize=trackSize;
  5786. var handleSize=Math.floor((clientSize/scrollSize) * trackSize);
  5787. handleSize=Math.max(20,handleSize);//限制手柄的最小大小;
  5788. this.handleSize=handleSize;
  5789. this.one=(trackSize-handleSize) / scrollMax;//一个像素对应的滚动条长度
  5790. this.scrollbar.handle.style[pro[2]]= handleSize + 'px';
  5791. this.scroll(this.getScrolled());
  5792. }else{
  5793. this.hide();
  5794. };
  5795. },
  5796. show:function(){
  5797. if(this.shown)return;
  5798. this.shown=true;
  5799. this.scrollbar.bar.style.display='block';
  5800. },
  5801. hide:function(){
  5802. if(!this.shown)return;
  5803. this.shown=false;
  5804. this.scrollbar.bar.style.display='none';
  5805. },
  5806. scrollBy:function(distance,handleDistance){
  5807. return this.scroll(this.getScrolled() + (handleDistance? distance / this.one : distance));
  5808. },
  5809. scrollByPages:function(num){
  5810. this.scroll(this.getScrolled() + (this.container[(this.isHorizontal ? 'clientWidth' : 'clientHeight')] - 10) * num);
  5811. },
  5812. scroll:function(distance,handleDistance,transition){
  5813. if(!this.shown)return;
  5814.  
  5815. //滚动实际滚动条
  5816. var _distance=distance;
  5817. _distance=handleDistance? distance / this.one : distance;
  5818. _distance=Math.max(0,_distance);
  5819. _distance=Math.min(_distance,this.scrollMax);
  5820.  
  5821.  
  5822. var pro=this.isHorizontal? ['left','scrollLeft'] : ['top','scrollTop'];
  5823.  
  5824.  
  5825. //滚动虚拟滚动条
  5826. //根据比例转换为滚动条上应该滚动的距离。
  5827. distance=handleDistance? distance : this.one * distance;
  5828. //处理非法值
  5829. distance=Math.max(0,distance);//如果值小于0那么取0
  5830. distance=Math.min(distance,this.trackSize - this.handleSize);//大于极限值,取极限值
  5831.  
  5832. var shs=this.scrollbar.handle.style;
  5833. var container=this.container;
  5834. if(transition){
  5835. clearInterval(this.transitionInterval);
  5836.  
  5837. var start=0;
  5838. var duration=10;
  5839.  
  5840. var cStart=this.getScrolled();
  5841. var cChange=_distance-cStart;
  5842. var sStart=parseFloat(shs[pro[0]]);
  5843. var sChange=distance-sStart;
  5844.  
  5845. var transitionInterval=setInterval(function(){
  5846. var cEnd=Tween.Cubic.easeInOut(start,cStart,cChange,duration);
  5847. var sEnd=Tween.Cubic.easeInOut(start,sStart,sChange,duration);
  5848.  
  5849. container[pro[1]]=cEnd;
  5850. shs[pro[0]]=sEnd + 'px';
  5851.  
  5852. start++;
  5853. if(start>=duration){
  5854. clearInterval(transitionInterval);
  5855. };
  5856. },35);
  5857.  
  5858. this.transitionInterval=transitionInterval;
  5859.  
  5860. return;
  5861. };
  5862.  
  5863. var noScroll=shs[pro[0]].replace(/(\.\d*)?\s*px/,"")==Math.floor(distance);
  5864. shs[pro[0]]=distance + 'px';
  5865. container[pro[1]]=_distance;
  5866. return noScroll;
  5867. },
  5868. getScrolled:function(){
  5869. return this.container[(this.isHorizontal ? 'scrollLeft' : 'scrollTop')];
  5870. },
  5871. };
  5872.  
  5873.  
  5874. //放大镜
  5875. function MagnifierC(img,data){
  5876. this.img=img;
  5877. this.data=data;
  5878. this.init();
  5879. };
  5880.  
  5881. MagnifierC.all=[];
  5882. MagnifierC.styleZIndex=900000000;//全局z-index;
  5883.  
  5884. MagnifierC.prototype={
  5885. init:function(){
  5886. MagnifierC.zoomRange=prefs.magnifier.wheelZoom.range.slice(0).sort((a, b)=>{return a - b});
  5887. MagnifierC.zoomRangeR=MagnifierC.zoomRange.slice(0).reverse();//降序
  5888. this.addStyle();
  5889. MagnifierC.all.push(this);
  5890. var container=document.createElement('span');
  5891.  
  5892. container.className='pv-magnifier-container';
  5893. document.body.appendChild(container);
  5894.  
  5895. this.magnifier=container;
  5896.  
  5897. var imgNaturalSize={
  5898. h:this.img.naturalHeight,
  5899. w:this.img.naturalWidth,
  5900. };
  5901.  
  5902. this.imgNaturalSize=imgNaturalSize;
  5903.  
  5904. var cs=container.style;
  5905. cs.zIndex=MagnifierC.styleZIndex++;
  5906.  
  5907.  
  5908.  
  5909. var maxDia=Math.ceil(Math.sqrt(Math.pow(1/2*imgNaturalSize.w,2) + Math.pow(1/2*imgNaturalSize.h,2)) * 2);
  5910. this.maxDia=maxDia;
  5911.  
  5912. var radius=prefs.magnifier.radius;
  5913. radius=Math.min(maxDia/2,radius);
  5914. this.radius=radius;
  5915. var diameter=radius * 2;
  5916. this.diameter=diameter;
  5917.  
  5918. cs.width=diameter + 'px';
  5919. cs.height=diameter + 'px';
  5920. cs.borderRadius=radius+1 + 'px';
  5921. cs.backgroundImage='url("'+ this.img.src +'")';
  5922. cs.marginLeft= -radius +'px';
  5923. cs.marginTop= -radius +'px';
  5924.  
  5925. var imgPos=getContentClientRect(this.data.img);
  5926. var wScrolled=getScrolled();
  5927. var imgRange={//图片所在范围
  5928. x:[imgPos.left + wScrolled.x , imgPos.right + wScrolled.x],
  5929. y:[imgPos.top + wScrolled.y, imgPos.bottom + wScrolled.y],
  5930. };
  5931. var imgW=imgRange.x[1] - imgRange.x[0];
  5932. var imgH=imgRange.y[1] - imgRange.y[0];
  5933. //如果图片太小的话,进行范围扩大。
  5934. var minSize=60;
  5935. if(imgW < minSize){
  5936. imgRange.x[1] +=(minSize - imgW)/2;
  5937. imgRange.x[0] -=(minSize - imgW)/2;
  5938. imgW=minSize;
  5939. };
  5940. if(imgH < minSize){
  5941. imgRange.y[1] +=(minSize - imgH)/2;
  5942. imgRange.y[0] -=(minSize - imgH)/2;
  5943. imgH=minSize;
  5944. };
  5945. this.imgSize={
  5946. w:imgW,
  5947. h:imgH,
  5948. };
  5949. this.imgRange=imgRange;
  5950.  
  5951. this.setMouseRange();
  5952.  
  5953.  
  5954. this.move({
  5955. pageX:imgRange.x[0],
  5956. pageY:imgRange.y[0],
  5957. });
  5958.  
  5959. this._focus=this.focus.bind(this);
  5960. this._blur=this.blur.bind(this);
  5961. this._move=this.move.bind(this);
  5962. this._remove=this.remove.bind(this);
  5963. this._pause=this.pause.bind(this);
  5964. this._zoom=this.zoom.bind(this);
  5965. this._clickOut=this.clickOut.bind(this);
  5966.  
  5967. if(prefs.magnifier.wheelZoom.enabled){
  5968. this.zoomLevel=1;
  5969. this.defaultDia=diameter;
  5970. addWheelEvent(container,this._zoom,false);
  5971. };
  5972.  
  5973. container.addEventListener('mouseover',this._focus,false);
  5974. container.addEventListener('mouseout',this._blur,false);
  5975. container.addEventListener('dblclick',this._remove,false);
  5976. container.addEventListener('click',this._pause,false);
  5977.  
  5978.  
  5979. document.addEventListener('mousemove',this._move,true);
  5980. document.addEventListener('mouseup',this._clickOut,true);
  5981. },
  5982. addStyle:function(){
  5983. if(MagnifierC.style)return;
  5984. var style=document.createElement('style');
  5985. style.type='text/css';
  5986. MagnifierC.style=style;
  5987. style.textContent='\
  5988. .pv-magnifier-container{\
  5989. position:absolute;\
  5990. padding:0;\
  5991. margin:0;\
  5992. background-origin:border-box;\
  5993. -moz-box-sizing:border-box;\
  5994. box-sizing:border-box;\
  5995. border:3px solid #CCCCCC;\
  5996. background:rgba(40, 40, 40, 0.9) no-repeat;\
  5997. }\
  5998. .pv-magnifier-container_focus{\
  5999. box-shadow: 0px 0px 6px rgba(0, 0, 0, 0.7);\
  6000. }\
  6001. .pv-magnifier-container_pause{\
  6002. border-color:red;\
  6003. }\
  6004. ';
  6005. document.head.appendChild(style);
  6006. },
  6007. clickOut:function(){
  6008. if(!this.magnifier.classList.contains("pv-magnifier-container_focus")){
  6009. this.remove();
  6010. }
  6011. },
  6012. focus:function(){
  6013. this.magnifier.classList.add('pv-magnifier-container_focus');
  6014. this.magnifier.style.zIndex=MagnifierC.styleZIndex++;
  6015. },
  6016. blur:function(){
  6017. this.magnifier.classList.remove('pv-magnifier-container_focus');
  6018. },
  6019. move:function(e){
  6020. var mouseCoor={
  6021. x:e.pageX,
  6022. y:e.pageY,
  6023. };
  6024. var mouseRange=this.mouseRange;
  6025. var imgRange=this.imgRange;
  6026.  
  6027. if( !(mouseCoor.x >= mouseRange.x[0] && mouseCoor.x <= mouseRange.x[1] && mouseCoor.y >= mouseRange.y[0] && mouseCoor.y <= mouseRange.y[1]))return;//如果不再鼠标范围
  6028. if(mouseCoor.x > imgRange.x[1]){
  6029. mouseCoor.x = imgRange.x[1];
  6030. }else if(mouseCoor.x < imgRange.x[0]){
  6031. mouseCoor.x = imgRange.x[0];
  6032. };
  6033. if(mouseCoor.y > imgRange.y[1]){
  6034. mouseCoor.y = imgRange.y[1];
  6035. }else if(mouseCoor.y < imgRange.y[0]){
  6036. mouseCoor.y = imgRange.y[0];
  6037. };
  6038.  
  6039. var ms=this.magnifier.style;
  6040. ms.top= mouseCoor.y + 'px';
  6041. ms.left= mouseCoor.x + 'px';
  6042.  
  6043. var radius=this.radius;
  6044. var imgSize=this.imgSize;
  6045. var imgNaturalSize=this.imgNaturalSize;
  6046. var px=-((mouseCoor.x-imgRange.x[0])/imgSize.w * imgNaturalSize.w) + radius +'px';
  6047. var py=-((mouseCoor.y-imgRange.y[0])/imgSize.h * imgNaturalSize.h) + radius +'px';
  6048. ms.backgroundPosition=px + ' ' + py;
  6049. },
  6050. getNextZoomLevel:function(){
  6051. var level;
  6052. var self=this;
  6053. if(this.zoomOut){//缩小
  6054. MagnifierC.zoomRangeR._find(function(value){
  6055. if(value < self.zoomLevel){
  6056. level=value;
  6057. return true;
  6058. }
  6059. })
  6060. }else{
  6061. MagnifierC.zoomRange._find(function(value){
  6062. if(value > self.zoomLevel){
  6063. level=value;
  6064. return true;
  6065. };
  6066. });
  6067. }
  6068. return level;
  6069. },
  6070. zoom:function(e){
  6071. if(e.deltaY===0)return;//非Y轴的滚动
  6072. if(prefs.magnifier.wheelZoom.pauseFirst && !this.paused)return;
  6073. e.preventDefault();
  6074. if(e.deltaY < 0){//向上滚,放大;
  6075. if(this.diameter >= this.maxDia)return;
  6076. this.zoomOut=false;
  6077. }else{
  6078. this.zoomOut=true;
  6079. };
  6080. var level=this.getNextZoomLevel();
  6081. if(!level)return;
  6082.  
  6083. this.zoomLevel=level;
  6084. var diameter=this.defaultDia * level;
  6085. if(diameter > this.maxDia){
  6086. diameter = this.maxDia;
  6087. };
  6088.  
  6089. var radius=diameter/2
  6090. this.diameter=diameter;
  6091. var bRadius=this.radius;
  6092. this.radius=radius;
  6093. this.setMouseRange();
  6094. var ms=this.magnifier.style;
  6095. ms.width=diameter+'px';
  6096. ms.height=diameter+'px';
  6097. ms.borderRadius=radius+1 + 'px';
  6098. ms.marginLeft=-radius+'px';
  6099. ms.marginTop=-radius+'px';
  6100. var bBP=ms.backgroundPosition.split(' ');
  6101. ms.backgroundPosition=parseFloat(bBP[0]) + (radius - bRadius) + 'px' + ' ' + (parseFloat(bBP[1]) + ( radius - bRadius) + 'px');
  6102.  
  6103. },
  6104. pause:function(){
  6105. if(this.paused){
  6106. this.magnifier.classList.remove('pv-magnifier-container_pause');
  6107. document.addEventListener('mousemove',this._move,true);
  6108. }else{
  6109. this.magnifier.classList.add('pv-magnifier-container_pause');
  6110. document.removeEventListener('mousemove',this._move,true);
  6111. };
  6112. this.paused=!this.paused;
  6113. },
  6114. setMouseRange:function(){
  6115. var imgRange=this.imgRange;
  6116. var radius=this.radius;
  6117. this.mouseRange={//鼠标活动范围
  6118. x:[imgRange.x[0]-radius , imgRange.x[1] + radius],
  6119. y:[imgRange.y[0]-radius , imgRange.y[1] + radius],
  6120. };
  6121. },
  6122. remove:function(){
  6123. this.magnifier.parentNode.removeChild(this.magnifier);
  6124. document.removeEventListener('mousemove',this._move,true);
  6125. MagnifierC.all.splice(MagnifierC.all.indexOf(this),1);
  6126. },
  6127. };
  6128.  
  6129. //图片窗口
  6130. function ImgWindowC(img, data){
  6131. this.loaded=false;
  6132. this.img=img;
  6133. this.src=data?data.src:img.src;
  6134. this.data = data;
  6135.  
  6136. this.init();
  6137. if(data){
  6138. this.img.src = location.protocol == "https"?data.src.replace(/^http:/,"https:"):data.src;
  6139. }
  6140. };
  6141.  
  6142. ImgWindowC.all=[];//所有的窗口对象
  6143. ImgWindowC.styleZIndex=2147483647;//全局z-index;
  6144. ImgWindowC.overlayer=null;
  6145.  
  6146.  
  6147. ImgWindowC.prototype={
  6148. init:function(){
  6149. ImgWindowC.zoomRange=prefs.imgWindow.zoom.range.slice(0).sort((a, b)=>{return a - b});
  6150. ImgWindowC.zoomRangeR=ImgWindowC.zoomRange.slice(0).reverse();//降序
  6151. var self=this;
  6152. if(uniqueImgWin && !uniqueImgWin.removed){
  6153. uniqueImgWin.remove();
  6154. }
  6155. //图片是否已经被打开
  6156. if(ImgWindowC.all._find(function(iwin){
  6157. if(iwin.src==self.src){
  6158. iwin.firstOpen();
  6159. return true;
  6160. };
  6161. }))return;
  6162.  
  6163. this.addStyle();
  6164.  
  6165. var img=this.img;
  6166. img.className='pv-pic-window-pic pv-pic-ignored';
  6167. img.style.cssText='\
  6168. top:0px;\
  6169. left:0px;\
  6170. ';
  6171.  
  6172. var imgNaturalSize={
  6173. h:img.naturalHeight,
  6174. w:img.naturalWidth,
  6175. };
  6176. this.imgNaturalSize=imgNaturalSize;
  6177.  
  6178. var container=document.createElement('span');
  6179. container.style.cssText='\
  6180. cursor:pointer;\
  6181. top:0px;\
  6182. left:0px;\
  6183. opacity:0\
  6184. ';
  6185. container.className='pv-pic-window-container';
  6186. container.innerHTML=
  6187. '<span class="pv-pic-window-rotate-indicator">'+
  6188. '<span class="pv-pic-window-rotate-indicator-pointer"></span>'+
  6189. '</span>'+
  6190. '<span class="pv-pic-window-rotate-overlayer"></span>'+
  6191. '<span class="pv-pic-window-toolbar" unselectable="on">'+
  6192. '<span class="pv-pic-window-tb-hand pv-pic-window-tb-tool" title="'+i18n("hand")+'"></span>'+
  6193. '<span class="pv-pic-window-tb-tool-badge-container pv-pic-window-tb-tool-extend-menu-container">'+
  6194. '<span class="pv-pic-window-tb-rotate pv-pic-window-tb-tool" title="'+i18n("rotate")+'"></span>'+
  6195. '<span class="pv-pic-window-tb-tool-badge">0</span>'+
  6196. '<span class="pv-pic-window-tb-tool-extend-menu pv-pic-window-tb-tool-extend-menu-rotate">'+
  6197. '<span class="pv-pic-window-tb-tool-extend-menu-item" title="+90"><svg class="icon" style="width: 1em;height: 1em;vertical-align: middle;fill: currentColor;overflow: hidden;" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="4332"><path d="M435.2 362.666667l179.2-179.2L435.2 0 375.466667 59.733333l76.8 76.8H341.333333c-119.466667 0-213.333333 93.866667-213.333333 213.333334v170.666666h85.333333v-170.666666c0-72.533333 55.466667-128 128-128h115.2L375.466667 302.933333l59.733333 59.733334zM853.333333 384H426.666667c-25.6 0-42.666667 17.066667-42.666667 42.666667v426.666666c0 25.6 17.066667 42.666667 42.666667 42.666667h426.666666c25.6 0 42.666667-17.066667 42.666667-42.666667V426.666667c0-25.6-17.066667-42.666667-42.666667-42.666667z m-42.666666 426.666667h-341.333334v-341.333334h341.333334v341.333334z" p-id="4333"></path></svg></span>'+
  6198. '<span class="pv-pic-window-tb-tool-extend-menu-item" title="-90"><svg class="icon" style="width: 1em;height: 1em;vertical-align: middle;fill: currentColor;overflow: hidden;" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="4595"><path d="M682.666667 136.533333h-115.2l76.8-76.8L588.8 0 405.333333 179.2l179.2 179.2L644.266667 298.666667l-76.8-76.8H682.666667c72.533333 0 128 55.466667 128 128v170.666666h85.333333v-170.666666c0-115.2-93.866667-213.333333-213.333333-213.333334zM597.333333 384H170.666667c-25.6 0-42.666667 17.066667-42.666667 42.666667v426.666666c0 25.6 17.066667 42.666667 42.666667 42.666667h426.666666c25.6 0 42.666667-17.066667 42.666667-42.666667V426.666667c0-25.6-17.066667-42.666667-42.666667-42.666667z m-42.666666 426.666667H213.333333v-341.333334h341.333334v341.333334z" p-id="4596"></path></svg></span>'+
  6199. '<span class="pv-pic-window-tb-tool-extend-menu-item" title="0"><svg class="icon" style="width: 1em;height: 1em;vertical-align: middle;fill: currentColor;overflow: hidden;" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="3716"><path d="M867.89 574.16a30.73 30.73 0 0 0-37.52 21.92c-38 144.83-169.31 246-319.25 246a330.71 330.71 0 0 1-306.27-206.82h60.29l-92.78-123.5-91.82 123.5h58.88q1.23 3.72 2.53 7.4A391.65 391.65 0 0 0 511.12 903.5c177.86 0 333.58-120 378.69-291.82a30.73 30.73 0 0 0-21.92-37.52zM153.88 452.57a30.69 30.69 0 0 0 37.35-22.2A329.68 329.68 0 0 1 511.12 182c136.8 0 256.58 82 306.4 207h-60.66l92.78 123.5L941.46 389h-58.58a391.63 391.63 0 0 0-751.2 26.24 30.73 30.73 0 0 0 22.2 37.33z" p-id="3717"></path></svg></span>'+
  6200. '</span>'+
  6201. '</span>'+
  6202. '<span class="pv-pic-window-tb-tool-badge-container pv-pic-window-tb-tool-extend-menu-container">'+
  6203. '<span class="pv-pic-window-tb-zoom pv-pic-window-tb-tool" title="'+i18n("scale")+'"></span>'+
  6204. '<span class="pv-pic-window-tb-tool-badge">0</span>'+
  6205. '<span class="pv-pic-window-tb-tool-extend-menu pv-pic-window-tb-tool-extend-menu-zoom">'+
  6206. '<span id="pv-pic-zoom-in" class="pv-pic-window-tb-tool-extend-menu-item" title="+0.1"><svg class="icon" style="width: 1em;height: 1em;vertical-align: middle;fill: currentColor;overflow: hidden;" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="3134"><path d="M754.2 151.5h-89.5v42.7h89.5c59.5 0 108 48.4 108 108v67.6h42.7v-67.6c0-83.1-67.6-150.7-150.7-150.7zM862.2 737.3c0 59.5-48.4 108-108 108h-89.5V888h89.5c83.1 0 150.7-67.6 150.7-150.7v-67.6h-42.7v67.6zM166.3 737.8v-67.6h-42.7v67.6c0 83.1 67.6 150.7 150.7 150.7h89.5v-42.7h-89.5c-59.5 0-108-48.4-108-108zM416.3 261.8h-42.8v126H247.6v42.7h125.9V556h42.8V430.5h125.4v-42.7H416.3zM773.6 789.4l30.2-30.2-190.1-190.6c32.7-44.8 52-99.9 52-159.5 0-149.7-121.6-271.3-271.3-271.3-149.3 0-271.3 121.6-271.3 271.3 0 149.3 121.5 271.3 271.3 271.3 74.5 0 142.3-30.3 191.4-79.3l187.8 188.3zM394.4 637.7c-126 0-228.6-102.5-228.6-228.6s102.5-228.6 228.6-228.6S623 283.1 623 409.2 520.5 637.7 394.4 637.7z" p-id="3135"></path></svg></span>'+
  6207. '<span id="pv-pic-zoom-out" class="pv-pic-window-tb-tool-extend-menu-item" title="-0.1"><svg class="icon" style="width: 1em;height: 1em;vertical-align: middle;fill: currentColor;overflow: hidden;" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="2674"><path d="M754.2 151.5h-89.5v42.7h89.5c59.5 0 108 48.4 108 108v67.6h42.7v-67.6c0-83.1-67.6-150.7-150.7-150.7zM862.2 737.3c0 59.5-48.4 108-108 108h-89.5V888h89.5c83.1 0 150.7-67.6 150.7-150.7v-67.6h-42.7v67.6zM166.3 737.8v-67.6h-42.7v67.6c0 83.1 67.6 150.7 150.7 150.7h89.5v-42.7h-89.5c-59.5 0-108-48.4-108-108zM247.6 387.8h294.2v42.7H247.6zM773.6 789.4l30.2-30.2-190.1-190.6c32.7-44.8 52-99.9 52-159.5 0-149.7-121.6-271.3-271.3-271.3-149.3 0-271.3 121.6-271.3 271.3 0 149.3 121.6 271.3 271.3 271.3 74.5 0 142.3-30.3 191.4-79.3l187.8 188.3zM394.4 637.7c-126 0-228.6-102.5-228.6-228.6s102.5-228.6 228.6-228.6S623 283.1 623 409.2 520.5 637.7 394.4 637.7z" p-id="2675"></path></svg></span>'+
  6208. '<span class="pv-pic-window-tb-tool-extend-menu-item" title="1"><svg class="icon" style="width: 1em;height: 1em;vertical-align: middle;fill: currentColor;overflow: hidden;" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="3716"><path d="M867.89 574.16a30.73 30.73 0 0 0-37.52 21.92c-38 144.83-169.31 246-319.25 246a330.71 330.71 0 0 1-306.27-206.82h60.29l-92.78-123.5-91.82 123.5h58.88q1.23 3.72 2.53 7.4A391.65 391.65 0 0 0 511.12 903.5c177.86 0 333.58-120 378.69-291.82a30.73 30.73 0 0 0-21.92-37.52zM153.88 452.57a30.69 30.69 0 0 0 37.35-22.2A329.68 329.68 0 0 1 511.12 182c136.8 0 256.58 82 306.4 207h-60.66l92.78 123.5L941.46 389h-58.58a391.63 391.63 0 0 0-751.2 26.24 30.73 30.73 0 0 0 22.2 37.33z" p-id="3717"></path></svg></span>'+
  6209. '</span>'+
  6210. '</span>'+
  6211. '<span class="pv-pic-window-tb-flip-horizontal pv-pic-window-tb-command" title="'+i18n("horizontalFlip")+'"></span>'+
  6212. '<span class="pv-pic-window-tb-flip-vertical pv-pic-window-tb-command" title="'+i18n("verticalFlip")+'"></span>'+
  6213. '</span>'+
  6214. '<span class="pv-pic-window-close"></span>' +
  6215. //'<span class="pv-pic-window-search" title="'+i18n("similarImage")+'"></span>' +
  6216. '<span class="pv-pic-window-range"></span>' +
  6217. '<span class="pv-pic-window-description"></span>'+
  6218. '<span class="pv-pic-search-state"></span>';
  6219.  
  6220. container.insertBefore(img,container.firstChild);
  6221.  
  6222. this.imgWindow=container;
  6223.  
  6224. var toolMap={
  6225. 'hand':container.querySelector('.pv-pic-window-tb-hand'),
  6226. 'rotate':container.querySelector('.pv-pic-window-tb-rotate'),
  6227. 'zoom':container.querySelector('.pv-pic-window-tb-zoom'),
  6228. 'fh':container.querySelector('.pv-pic-window-tb-flip-horizontal'),
  6229. 'fv':container.querySelector('.pv-pic-window-tb-flip-vertical'),
  6230. };
  6231. this.toolMap=toolMap;
  6232.  
  6233.  
  6234. //关闭
  6235. var closeButton=container.querySelector('.pv-pic-window-close');
  6236. closeButton.style.cssText='top: -24px;right: 0px;';
  6237. this.closeButton=closeButton;
  6238. closeButton.addEventListener('click',function(e){
  6239. self.remove();
  6240. },false);
  6241.  
  6242. //var searchButton=container.querySelector('.pv-pic-window-search');
  6243. //searchButton.style.cssText='top: -24px;right: 50px;';
  6244. //this.searchButton=searchButton;
  6245. var srcs, from;
  6246. img.onerror=function(e){
  6247. //setSearchState(i18n("loadNextSimilar"),img.parentNode);
  6248. console.info(img.src+" "+i18n("loadError"));
  6249. if(self.removed || !self.data)return;
  6250. var src;
  6251. if(self.data.srcs)
  6252. src=self.data.srcs.shift();
  6253. if(src)img.src=src;
  6254. else{
  6255. if(img.src!=self.data.imgSrc)
  6256. img.src=self.data.imgSrc;
  6257. return;
  6258. if(from<searchSort.length){
  6259. from++;
  6260. searchImgByImg(self.img.src, self.img.parentNode, function(srcs, index){
  6261. from=index;
  6262. self.data.srcs=srcs;
  6263. self.img.src=srcs.shift();
  6264. },null,null,from);
  6265. }else{
  6266. setSearchState(i18n("findNoPic"),img.parentNode);
  6267. setTimeout(function(){
  6268. setSearchState("",img.parentNode);
  6269. },2000);
  6270. }
  6271. }
  6272. };
  6273. img.onload=function(e){
  6274. self.loaded=true;
  6275. if(img.naturalHeight ==1 && img.naturalWidth ==1){
  6276. self.remove();
  6277. return;
  6278. }
  6279. self.imgWindow.classList.remove("pv-pic-window-transition-all");
  6280. self.imgWindow.style.display="";
  6281. setSearchState("",img.parentNode);
  6282. self.imgNaturalSize={
  6283. h:img.naturalHeight,
  6284. w:img.naturalWidth,
  6285. };
  6286. if(self.imgWindow.style.overflow!="scroll"){
  6287. self.zoomLevel=0;
  6288. self.zoom(1);
  6289. }
  6290. if(prefs.imgWindow.fitToScreen)
  6291. self.fitToScreen();
  6292. self.center(true,true);
  6293. self.imgWindow.style.opacity=1;
  6294. self.keepScreenInside();
  6295.  
  6296. var wSize=getWindowSize();
  6297. wSize.h -= 16;
  6298. wSize.w -= 16;
  6299. self.isLongImg=self.imgNaturalSize.h >= wSize.h && self.imgNaturalSize.h/self.imgNaturalSize.w > 3.5;
  6300. }
  6301. /*searchButton.addEventListener('click',function(e){
  6302. sortSearch();
  6303. searchImgByImg(self.img.src, self.img.parentNode, function(srcs, index){
  6304. from=index;
  6305. self.srcs=srcs;
  6306. self.img.src=srcs.shift();
  6307. });
  6308. },false);*/
  6309.  
  6310. /**
  6311. * 说明
  6312. * 1、对原来的适应屏幕等功能会有影响,暂时禁用。
  6313. * 2、分为 absolute 和默认的2种情况
  6314. */
  6315. if (this.data) {
  6316. var descriptionSpan = container.querySelector('.pv-pic-window-description');
  6317. // descriptionSpan.style.cssText = '\
  6318. // bottom: -40px;\
  6319. // left: 10px;\
  6320. // ';
  6321. descriptionSpan.textContent = this.data.description || '';
  6322. // descriptionSpan.style.display = this.data.description ? 'block' : 'none';
  6323. descriptionSpan.style.display = 'none';
  6324. this.descriptionSpan = descriptionSpan;
  6325. }
  6326.  
  6327. var toolbar=container.querySelector('.pv-pic-window-toolbar');
  6328. toolbar.style.cssText='\
  6329. top: 0px;\
  6330. left: -45px;\
  6331. ';
  6332. this.toolbar=toolbar;
  6333.  
  6334. this.selectedToolClass='pv-pic-window-tb-tool-selected';
  6335.  
  6336. this.viewRange=container.querySelector('.pv-pic-window-range');
  6337.  
  6338. this.rotateIndicator=container.querySelector('.pv-pic-window-rotate-indicator');
  6339. this.rotateIPointer=container.querySelector('.pv-pic-window-rotate-indicator-pointer');
  6340. this.rotateOverlayer=container.querySelector('.pv-pic-window-rotate-overlayer');
  6341.  
  6342.  
  6343. this.hKeyUp=true;
  6344. this.rKeyUp=true;
  6345. this.zKeyUp=true;
  6346.  
  6347. this.spaceKeyUp=true;
  6348. this.ctrlKeyUp=true;
  6349. this.altKeyUp=true;
  6350. this.shiftKeyUp=true;
  6351. this.moving=false;
  6352.  
  6353. //缩放工具的扩展菜单
  6354. container.querySelector('.pv-pic-window-tb-tool-extend-menu-zoom').addEventListener('click',function(e){
  6355. var target=e.target;
  6356. var text=target.title;
  6357. var value;
  6358. switch(text){
  6359. case '1':{
  6360. value=1;
  6361. }break;
  6362. case '+0.1':{
  6363. value=self.zoomLevel + 0.1;
  6364. }break;
  6365. case '-0.1':{
  6366. value=self.zoomLevel - 0.1;
  6367. }break;
  6368. };
  6369. if(typeof value!='undefined'){
  6370. self.zoom(value,{x:0,y:0});
  6371. };
  6372. },true);
  6373.  
  6374. //旋转工具的扩展菜单
  6375. container.querySelector('.pv-pic-window-tb-tool-extend-menu-rotate').addEventListener('click',function(e){
  6376. var target=e.target;
  6377. var text=target.title;
  6378. var value;
  6379. function convert(deg){
  6380. return deg * Math.PI/180;
  6381. };
  6382.  
  6383. switch(text){
  6384. case '0':{
  6385. value=0;
  6386. }break;
  6387. case '+90':{
  6388. value=self.rotatedRadians + convert(90);
  6389. }break;
  6390. case '-90':{
  6391. value=self.rotatedRadians - convert(90);
  6392. }break;
  6393. };
  6394.  
  6395. var PI=Math.PI;
  6396. if(typeof value!='undefined'){
  6397. if(value>=2*PI){
  6398. value-=2*PI;
  6399. }else if(value<0){
  6400. value+=2*PI;
  6401. };
  6402. self.rotate(value,true);
  6403. };
  6404. },true);
  6405.  
  6406. toolbar.addEventListener('mousedown',function(e){//鼠标按下选择工具
  6407. self.toolbarEventHandler(e);
  6408. },false);
  6409.  
  6410.  
  6411. toolbar.addEventListener('dblclick',function(e){//鼠标双击工具
  6412. self.toolbarEventHandler(e);
  6413. },false);
  6414.  
  6415.  
  6416. //阻止浏览器对图片的默认控制行为
  6417. img.addEventListener('mousedown',function(e){
  6418. e.preventDefault();
  6419. },false);
  6420.  
  6421.  
  6422. container.addEventListener('mousedown',function(e){//当按下的时,执行平移,缩放,旋转操作
  6423. self.imgWindowEventHandler(e);
  6424. },false);
  6425.  
  6426. container.addEventListener('touchstart',function(e){//当按下的时,执行平移,缩放,旋转操作
  6427. self.imgWindowEventHandler(e);
  6428. },false);
  6429.  
  6430. container.addEventListener('click',function(e){//阻止opera ctrl+点击保存图片
  6431. self.imgWindowEventHandler(e);
  6432. },false);
  6433.  
  6434. if(prefs.imgWindow.zoom.mouseWheelZoom){//是否使用鼠标缩放
  6435. addWheelEvent(container,function(e){//滚轮缩放
  6436. self.imgWindowEventHandler(e);
  6437. },false);
  6438. };
  6439.  
  6440.  
  6441. if(prefs.imgWindow.overlayer.shown){//是否显示覆盖层
  6442. var overlayer=ImgWindowC.overlayer;
  6443. if(!overlayer){
  6444. var overlayer=document.createElement('span');
  6445. ImgWindowC.overlayer=overlayer;
  6446. overlayer.className='pv-pic-window-overlayer';
  6447. document.body.appendChild(overlayer);
  6448. overlayer.style.backgroundColor=prefs.imgWindow.overlayer.color;
  6449. };
  6450. overlayer.style.display='block';
  6451. };
  6452.  
  6453. //是否点击图片外部关闭
  6454. if(prefs.imgWindow.overlayer.shown && prefs.imgWindow.close.clickOutside){
  6455. var clickOutside=function(e){
  6456. var target=e.target;
  6457. if(!container.contains(target)){
  6458. self.remove();
  6459. };
  6460. };
  6461. this.clickOutside=clickOutside;
  6462. document.addEventListener(prefs.imgWindow.close.clickOutside,clickOutside,true);
  6463. };
  6464.  
  6465. //是否双击图片本身关闭
  6466. if(prefs.imgWindow.close.dblClickImgWindow){
  6467. var dblClickImgWindow=function(e){
  6468. var target=e.target;
  6469. if(target==container || target==img || target==self.rotateOverlayer){
  6470. self.remove();
  6471. };
  6472. };
  6473. container.addEventListener('dblclick',dblClickImgWindow,true);
  6474. };
  6475.  
  6476.  
  6477. document.body.appendChild(container);
  6478. ImgWindowC.all.push(this);
  6479.  
  6480. this._blur=this.blur.bind(this);
  6481. this._focusedKeydown=this.focusedKeydown.bind(this);
  6482. this._focusedKeyup=this.focusedKeyup.bind(this);
  6483.  
  6484. this.rotatedRadians=0;//已经旋转的角度
  6485. this.zoomLevel=1;//缩放级别
  6486. this.setToolBadge('zoom',1);
  6487.  
  6488. //选中默认工具
  6489. this.selectTool(prefs.imgWindow.defaultTool);
  6490.  
  6491. this.firstOpen();
  6492. this.imgWindow.style.opacity=1;
  6493. },
  6494. changeData:function(result){
  6495. if(this.src != result.src){
  6496. this.imgWindow.classList.add("pv-pic-window-transition-all");
  6497. this.loaded = false;
  6498. this.data = result;
  6499. this.src = result.src;
  6500. this.img.src = location.protocol == "https"?result.src.replace(/^https?:/,""):result.src;
  6501. }
  6502. },
  6503. addStyle:function(){
  6504. if(ImgWindowC.style)return;
  6505. var style=document.createElement('style');
  6506. ImgWindowC.style=style;
  6507. style.textContent='\
  6508. .pv-pic-window-container {\
  6509. position: absolute;\
  6510. background-color: rgba(40,40,40,0.9);\
  6511. padding: 8px;\
  6512. border: 5px solid #ccc;\
  6513. border-radius: 1px;\
  6514. line-height: 0;\
  6515. text-align: left;\
  6516. box-sizing: content-box;\
  6517. -webkit-transition: opacity 0.3s ease-out;\
  6518. transition: opacity 0.3s ease-out;\
  6519. overscroll-behavior: none;\
  6520. }\
  6521. .pv-pic-window-transition-all{\
  6522. -webkit-transition: all 0.5s ease-out;\
  6523. transition: all 0.5s ease-out;\
  6524. }\
  6525. .pv-pic-window-container_focus {\
  6526. box-shadow: 0 0 10px rgba(0,0,0,0.6);\
  6527. box-sizing: content-box;\
  6528. }\
  6529. .pv-pic-window-close,\
  6530. .pv-pic-window-search,\
  6531. .pv-pic-window-toolbar,\
  6532. .pv-pic-window-tb-tool-extend-menu{\
  6533. -webkit-transition: opacity 0.2s ease-in-out;\
  6534. transition: opacity 0.2s ease-in-out;\
  6535. }\
  6536. .pv-pic-window-toolbar {\
  6537. position: absolute;\
  6538. background-color: #535353;\
  6539. padding: 0;\
  6540. opacity: 0.9;\
  6541. display: none;\
  6542. cursor: default;\
  6543. -o-user-select: none;\
  6544. -webkit-user-select: none;\
  6545. -moz-user-select: -moz-none;\
  6546. user-select: none;\
  6547. }\
  6548. .pv-pic-window-toolbar:hover {\
  6549. opacity: 1;\
  6550. }\
  6551. .pv-pic-window-toolbar_focus {\
  6552. display: block;\
  6553. }\
  6554. .pv-pic-window-close {\
  6555. cursor: pointer;\
  6556. position: absolute;\
  6557. right: 0px;\
  6558. top: -24px;\
  6559. background: url("'+prefs.icons.close+'") no-repeat center bottom;\
  6560. height: 17px;\
  6561. width: 46px;\
  6562. opacity: 0.9;\
  6563. border:none;\
  6564. padding:0;\
  6565. padding-top:2px;\
  6566. background-color:#1771FF;\
  6567. display: none;\
  6568. }\
  6569. .pv-pic-window-close:hover {\
  6570. background-color:red;\
  6571. opacity: 1;\
  6572. }\
  6573. .pv-pic-window-close_focus {\
  6574. display: block;\
  6575. }\
  6576. .pv-pic-window-search {\
  6577. cursor: pointer;\
  6578. position: absolute;\
  6579. right: 50px;\
  6580. top: -24px;\
  6581. background: url("'+prefs.icons.searchBtn+'") no-repeat center bottom;\
  6582. height: 17px;\
  6583. width: 46px;\
  6584. opacity: 0.9;\
  6585. border:none;\
  6586. padding:0;\
  6587. padding-top:2px;\
  6588. background-color:#1771FF;\
  6589. display: none;\
  6590. }\
  6591. .pv-pic-window-search:hover {\
  6592. background-color:red;\
  6593. opacity: 1;\
  6594. }\
  6595. .pv-pic-window-search_focus {\
  6596. display: block;\
  6597. }\
  6598. .pv-pic-window-description {\
  6599. margin-top: 20px;\
  6600. min-height: 20px;\
  6601. }\
  6602. .pv-pic-search-state {\
  6603. top: 10px;\
  6604. left: 10px;\
  6605. display: block;\
  6606. position: absolute;\
  6607. z-index: 1;\
  6608. color: #ffff00;\
  6609. width: 500px;\
  6610. font-size: large;\
  6611. text-shadow: 1px 0 0 #000,-1px 0 0 #000,0 1px 0 #000,0 -1px 0 #000;\
  6612. -webkit-text-shadow:#000 1px 0 0,#000 0 1px 0,#000 -1px 0 0,#000 0 -1px 0;\
  6613. -moz-text-shadow:#000 1px 0 0,#000 0 1px 0,#000 -1px 0 0,#000 0 -1px 0;\
  6614. }\
  6615. .pv-pic-window-pic {\
  6616. position: relative;\
  6617. display:inline-block;\/*opera把图片设置display:block会出现渲染问题,会有残影,还会引发其他各种问题,吓尿*/\
  6618. max-width:none;\
  6619. min-width:none;\
  6620. max-height:none;\
  6621. min-height:none;\
  6622. padding:0;\
  6623. margin:0;\
  6624. border:none;\
  6625. vertical-align:middle;\
  6626. }\
  6627. .pv-pic-window-pic_focus {\
  6628. box-shadow: 0 0 6px black;\
  6629. }\
  6630. .pv-pic-window-tb-tool,\
  6631. .pv-pic-window-tb-command{\
  6632. box-sizing:content-box;\
  6633. -moz-box-sizing:content-box;\
  6634. -webkit-box-sizing:content-box;\
  6635. height: 24px;\
  6636. width: 24px;\
  6637. padding: 12px 8px 6px 6px;\
  6638. margin:0;\
  6639. display: block;\
  6640. background: transparent no-repeat center;\
  6641. cursor: pointer;\
  6642. position: relative;\
  6643. border: none;\
  6644. border-left: 2px solid transparent;\
  6645. border-bottom: 1px solid #868686;\
  6646. background-origin: content-box;\
  6647. }\
  6648. .pv-pic-window-toolbar > span:last-child {\
  6649. border-bottom: none;\
  6650. }\
  6651. .pv-pic-window-tb-tool:hover,\
  6652. .pv-pic-window-tb-command:hover{\
  6653. border-left: 2px solid red;\
  6654. }\
  6655. .pv-pic-window-tb-tool-selected{\
  6656. box-shadow: inset 0 21px 0 rgba(255,255,255,0.3) ,inset 0 -21px 0 rgba(0,0,0,0.3);\
  6657. border-left:2px solid #1771FF;\
  6658. }\
  6659. .pv-pic-window-tb-hand {\
  6660. background-image: url("'+prefs.icons.hand+'");\
  6661. }\
  6662. .pv-pic-window-tb-rotate {\
  6663. background-image: url("'+prefs.icons.rotate+'");\
  6664. }\
  6665. .pv-pic-window-tb-zoom {\
  6666. background-image: url("'+prefs.icons.zoom+'");\
  6667. }\
  6668. .pv-pic-window-tb-flip-horizontal {\
  6669. background-image: url("'+prefs.icons.flipHorizontal+'");\
  6670. }\
  6671. .pv-pic-window-tb-flip-vertical {\
  6672. background-image: url("'+prefs.icons.flipVertical+'");\
  6673. }\
  6674. .pv-pic-window-tb-tool-badge-container {\
  6675. display: block;\
  6676. position: relative;\
  6677. }\
  6678. .pv-pic-window-tb-tool-badge {\
  6679. position: absolute;\
  6680. top: -3px;\
  6681. right: 1px;\
  6682. font-size: 10px;\
  6683. line-height: 1.5;\
  6684. padding: 0 3px;\
  6685. background-color: #F93;\
  6686. border-radius: 50px;\
  6687. opacity: 0.5;\
  6688. color: black;\
  6689. }\
  6690. .pv-pic-window-tb-tool-extend-menu{\
  6691. position:absolute;\
  6692. top:0;\
  6693. margin-left:-1px;\
  6694. background-color:#535353;\
  6695. display:none;\
  6696. left:40px;\
  6697. color:#C3C3C3;\
  6698. font-size:12px;\
  6699. text-shadow:0px -1px 0px black;\
  6700. opacity:0.7;\
  6701. }\
  6702. .pv-pic-window-tb-tool-extend-menu:hover{\
  6703. opacity:0.9;\
  6704. }\
  6705. .pv-pic-window-tb-tool-extend-menu-item{\
  6706. display:block;\
  6707. line-height:1.5;\
  6708. text-align:center;\
  6709. padding:10px;\
  6710. cursor:pointer;\
  6711. border: none;\
  6712. border-right: 2px solid transparent;\
  6713. border-bottom: 1px solid #868686;\
  6714. font-size: 20px;\
  6715. }\
  6716. .pv-pic-window-tb-tool-extend-menu-item>svg{\
  6717. pointer-events: none;\
  6718. }\
  6719. .pv-pic-window-tb-tool-extend-menu-item:last-child{\
  6720. border-bottom: none;\
  6721. }\
  6722. .pv-pic-window-tb-tool-extend-menu-item:hover{\
  6723. border-right:2px solid red;\
  6724. }\
  6725. .pv-pic-window-tb-tool-extend-menu-item:active{\
  6726. padding:11px 9px 9px 11px;\
  6727. }\
  6728. .pv-pic-window-tb-tool-extend-menu-container:hover .pv-pic-window-tb-tool{\
  6729. border-left:2px solid red;\
  6730. }\
  6731. .pv-pic-window-tb-tool-extend-menu-container:hover .pv-pic-window-tb-tool-extend-menu{\
  6732. display:block;\
  6733. }\
  6734. .pv-pic-window-tb-tool-extend-menu-container::after{\
  6735. content:"";\
  6736. position:absolute;\
  6737. right:1px;\
  6738. bottom:2px;\
  6739. width:0;\
  6740. height:0;\
  6741. padding:0;\
  6742. margin:0;\
  6743. border:3px solid #C3C3C3;\
  6744. border-top-color:transparent;\
  6745. border-left-color:transparent;\
  6746. opacity:0.5;\
  6747. }\
  6748. .pv-pic-window-overlayer{\
  6749. height:100%;\
  6750. width:100%;\
  6751. position:fixed;\
  6752. z-index:999999999;\
  6753. top:0;\
  6754. left:0;\
  6755. }\
  6756. .pv-pic-window-rotate-indicator{\
  6757. display:none;\
  6758. position:fixed;\
  6759. width:250px;\
  6760. height:250px;\
  6761. padding:10px;\
  6762. margin-top:-135px;\
  6763. margin-left:-135px;\
  6764. background:transparent url("'+ prefs.icons.rotateIndicatorBG +'") no-repeat center;\
  6765. }\
  6766. .pv-pic-window-rotate-indicator-pointer{\
  6767. display:block;\
  6768. margin-left:auto;\
  6769. margin-right:auto;\
  6770. background:transparent url("'+ prefs.icons.rotateIndicatorPointer +'") no-repeat center;\
  6771. width:60px;\
  6772. height:240px;\
  6773. position:relative;\
  6774. top:5px;\
  6775. transform:rotate(0.1deg);\
  6776. }\
  6777. .pv-pic-window-rotate-overlayer{/*当切换到旋转工具的时候显示这个覆盖层,然后旋转指示器显示在这个覆盖层的下面*/\
  6778. position:absolute;\
  6779. top:0;\
  6780. bottom:0;\
  6781. left:0;\
  6782. right:0;\
  6783. display:none;\
  6784. background-color:transparent;\
  6785. }\
  6786. .pv-pic-window-range{\
  6787. position:absolute;\
  6788. border:none;\
  6789. width:100px;\
  6790. height:100px;\
  6791. box-shadow: 0px 0px 5px rgba(0, 0, 0, 0.8);\
  6792. display:none;\
  6793. padding:0;\
  6794. background-color:rgba(255, 0, 0, 0.150);\
  6795. }\
  6796. .pv-pic-window-container::-webkit-scrollbar { width: 0 !important }\
  6797. .pv-pic-window-container { -ms-overflow-style: none;overflow: -moz-scrollbars-none; }\
  6798. ';
  6799. document.head.appendChild(style);
  6800. },
  6801.  
  6802. firstOpen:function(){
  6803. ImgWindowC.selectedTool='hand';
  6804. this.focus();
  6805. var imgWindow=this.imgWindow;
  6806. var scrolled=getScrolled();
  6807. imgWindow.style.left=-5 + scrolled.x + 'px';
  6808. imgWindow.style.top=-5 + scrolled.y + 'px';
  6809.  
  6810. //window的尺寸
  6811. var wSize=getWindowSize();
  6812. //空隙
  6813. wSize.h -= 16;
  6814. wSize.w -= 16;
  6815.  
  6816. var imgWindowCS=unsafeWindow.getComputedStyle(imgWindow);
  6817. var rectSize={
  6818. h:parseFloat(imgWindowCS.height),
  6819. w:parseFloat(imgWindowCS.width),
  6820. };
  6821. this.isLongImg=rectSize.h > wSize.h && rectSize.h/rectSize.w > 3.5;
  6822. if(prefs.imgWindow.suitLongImg && this.isLongImg){
  6823. this.center(rectSize.w <= wSize.w,false);
  6824. this.imgWindow.style.overflow="scroll";
  6825. this.imgWindow.style.height="100%";
  6826. this.imgWindow.style.maxWidth="100%";
  6827. this.closeButton.style.display="none";
  6828. }else if(prefs.imgWindow.fitToScreen){
  6829. this.fitToScreen();
  6830. this.center(true,true);
  6831. }else{
  6832. this.center(rectSize.w <= wSize.w , rectSize.h <= wSize.h);
  6833. };
  6834.  
  6835. this.keepScreenInside();
  6836. },
  6837. keepScreenInside:function(){//保持按钮在屏幕里面.
  6838. var imgWindow=this.imgWindow;
  6839. var imgWindowFullSize={
  6840. h:imgWindow.offsetHeight,
  6841. w:imgWindow.offsetWidth,
  6842. };
  6843.  
  6844. var windowSize=getWindowSize();
  6845.  
  6846. function keepSI(obj,offsetDirection,defaultValue, out){
  6847. var objRect=obj.getBoundingClientRect();
  6848. var objStyle=obj.style;
  6849.  
  6850. while(offsetDirection.length){
  6851. var oD=offsetDirection[0];
  6852. var oDV=defaultValue[0];
  6853. offsetDirection.shift();
  6854. defaultValue.shift();
  6855. var oValue=parseFloat(objStyle[oD]);
  6856. var newValue;
  6857. switch(oD){
  6858. case 'top':{
  6859. newValue=oValue - objRect.top;
  6860. if(objRect.top<0){
  6861. newValue=Math.min(newValue,imgWindowFullSize.h);
  6862. }else{
  6863. newValue=Math.max(newValue,oDV);
  6864. };
  6865. }break;
  6866. case 'right':{
  6867. newValue=oValue + (objRect.right - windowSize.w);
  6868. if(objRect.right > windowSize.w){//屏幕外
  6869. newValue=Math.min(newValue,imgWindowFullSize.w);
  6870. }else{
  6871. newValue=Math.max(newValue,oDV);
  6872. };
  6873. }break;
  6874. case 'bottom':{
  6875. newValue=oValue + (objRect.bottom - windowSize.h);
  6876. if(objRect.bottom > windowSize.h){//屏幕外
  6877. newValue=Math.min(newValue,imgWindowFullSize.h);
  6878. }else{
  6879. newValue=Math.max(newValue,oDV);
  6880. };
  6881. }break;
  6882. case 'left':{
  6883. newValue=oValue - objRect.left;
  6884. if(objRect.left<0){
  6885. newValue=Math.min(newValue,imgWindowFullSize.w);
  6886. }else{
  6887. newValue=Math.max(newValue,oDV);
  6888. }
  6889. }break;
  6890. };
  6891. objStyle[oD]=newValue + 'px';
  6892.  
  6893. };
  6894. };
  6895.  
  6896. keepSI(this.closeButton,['top','right'],[-24,0]);
  6897. //keepSI(this.searchButton,['top','right'],[-24,50]);
  6898. keepSI(this.toolbar,['top','left'],[0,-45]);
  6899.  
  6900. // 保持注释在图片里面
  6901. // keepSI(this.descriptionSpan,['bottom', 'left'],[-40, 10]);
  6902. },
  6903. fitToScreen:function(){
  6904. var imgWindow=this.imgWindow;
  6905. if(!prefs.imgWindow.fitToScreen || imgWindow.style.overflow=="scroll")return;
  6906. var wSize=getWindowSize();
  6907. //空隙
  6908. wSize.h -= 16;
  6909. wSize.w -= 16;
  6910.  
  6911. var imgWindowCS=unsafeWindow.getComputedStyle(imgWindow);
  6912. var rectSize={
  6913. h:parseFloat(imgWindowCS.height),
  6914. w:parseFloat(imgWindowCS.width),
  6915. };
  6916.  
  6917. var size;
  6918. if(rectSize.w - wSize.w>0 || rectSize.h - wSize.h>0){//超出屏幕,那么缩小。
  6919. if(rectSize.w/rectSize.h > wSize.w/wSize.h){
  6920. size={
  6921. w:wSize.w,
  6922. h:wSize.w / (rectSize.w/rectSize.h),
  6923. };
  6924. }else{
  6925. size={
  6926. h:wSize.h,
  6927. w:wSize.h * (rectSize.w/rectSize.h),
  6928. }
  6929. };
  6930.  
  6931. this.zoom(this.getRotatedImgCliSize(size).w/this.imgNaturalSize.w);
  6932. };
  6933. },
  6934. center:function(horizontal,vertical){
  6935. if(!horizontal && !vertical)return;
  6936. var wSize=getWindowSize();
  6937. var imgWindow=this.imgWindow;
  6938. var scrolled=getScrolled();
  6939. if(horizontal)imgWindow.style.left= (wSize.w - imgWindow.offsetWidth)/2 + scrolled.x +'px';
  6940. if(vertical)imgWindow.style.top= (wSize.h - imgWindow.offsetHeight)/2 + scrolled.y +'px';
  6941. },
  6942.  
  6943.  
  6944. move:function(e){
  6945. this.working=true;
  6946. var cursor=this.cursor;
  6947. this.changeCursor('handing');
  6948.  
  6949. var mouseCoor={
  6950. x:e.pageX || e.touches[0].pageX,
  6951. y:e.pageY || e.touches[0].pageY,
  6952. };
  6953. var imgWindow=this.imgWindow;
  6954. var imgWStyle=imgWindow.style;
  6955. var oriOffset={
  6956. left:parseFloat(imgWStyle.left),
  6957. top:parseFloat(imgWStyle.top),
  6958. };
  6959. var self=this;
  6960. var moveHandler=function(e){
  6961. imgWStyle.left=oriOffset.left+ (e.pageX || e.touches[0].pageX)-mouseCoor.x +'px';
  6962. imgWStyle.top=oriOffset.top + (e.pageY || e.touches[0].pageY)-mouseCoor.y +'px';
  6963. self.keepScreenInside();
  6964. self.moving=true;
  6965. };
  6966. var mouseupHandler=function(){
  6967. e.preventDefault();
  6968. self.changeCursor(cursor);
  6969. self.working=false;
  6970. if(self.tempHand && self.spaceKeyUp){//如果是临时切换到抓手工具,平移完成后返回上个工具
  6971. self.tempHand=false;
  6972. self.changeCursor(self.selectedTool);
  6973. };
  6974. document.removeEventListener('mousemove',moveHandler,true);
  6975. document.removeEventListener('mouseup',mouseupHandler,true);
  6976. document.removeEventListener('touchmove',moveHandler,true);
  6977. document.removeEventListener('touchend',mouseupHandler,true);
  6978. };
  6979. document.addEventListener('mousemove',moveHandler,true);
  6980. document.addEventListener('mouseup',mouseupHandler,true);
  6981. document.addEventListener('touchmove',moveHandler,true);
  6982. document.addEventListener('touchend',mouseupHandler,true);
  6983. },
  6984. rotate:function(origin,topLeft){
  6985.  
  6986. var img=this.img;
  6987. var imgWindow=this.imgWindow;
  6988.  
  6989. var iTransform=img.style[support.cssTransform].replace(/rotate\([^)]*\)/i,'');
  6990.  
  6991. var imgWindowCS=unsafeWindow.getComputedStyle(imgWindow);
  6992. var imgRectSize={
  6993. h:parseFloat(imgWindowCS.height),
  6994. w:parseFloat(imgWindowCS.width),
  6995. };
  6996.  
  6997. var rectOffset={
  6998. top:parseFloat(imgWindow.style.top),
  6999. left:parseFloat(imgWindow.style.left),
  7000. };
  7001.  
  7002. var imgSize={
  7003. h:img.clientHeight,
  7004. w:img.clientWidth,
  7005. };
  7006.  
  7007. var imgOffset={
  7008. top:parseFloat(img.style.top),
  7009. left:parseFloat(img.style.left),
  7010. };
  7011.  
  7012. var self=this;
  7013. var PI=Math.PI;
  7014.  
  7015. var rotate=function (radians){
  7016. if(self.rotatedRadians==radians)return;
  7017. img.style[support.cssTransform] = ' rotate('+ radians +'rad) ' + iTransform;//旋转图片
  7018. self.rotateIPointer.style[support.cssTransform]='rotate('+ radians +'rad)';//旋转指示器
  7019.  
  7020. self.rotatedRadians=radians;
  7021. self.setToolBadge('rotate',radians/(PI/180));
  7022.  
  7023. var afterimgRectSize=self.getRotatedImgRectSize( radians, imgSize );
  7024. imgWindow.style.width=afterimgRectSize.w +'px';
  7025. imgWindow.style.height=afterimgRectSize.h + 'px';
  7026.  
  7027. if(!topLeft){
  7028. self.setImgWindowOffset(rectOffset,imgRectSize,afterimgRectSize);
  7029. };
  7030.  
  7031. self.setImgOffset(imgOffset,imgRectSize,afterimgRectSize);
  7032. self.keepScreenInside();
  7033. };
  7034.  
  7035.  
  7036. if(typeof origin=='number'){
  7037. rotate(origin);
  7038. return;
  7039. };
  7040.  
  7041.  
  7042. this.working=true;
  7043.  
  7044. var lastRotatedRadians=this.rotatedRadians;
  7045. this.shiftKeyUp=true;
  7046. var shiftRotateStep=prefs.imgWindow.shiftRotateStep / (180/Math.PI);//转成弧度
  7047.  
  7048. var moveHandler=function(e){
  7049. self.rotateIndicator.style.display='block';
  7050. var radians=lastRotatedRadians + Math.atan2( e.clientY - origin.y, e.clientX - origin.x );
  7051. if(radians>=2*PI){
  7052. radians-=2*PI;
  7053. }else if(radians<0){
  7054. radians+=2*PI;
  7055. };
  7056.  
  7057. if(!self.shiftKeyUp){//如果按下了shift键,那么步进缩放
  7058. radians -= radians % shiftRotateStep;
  7059. radians += shiftRotateStep;
  7060. };
  7061. rotate(radians);
  7062. };
  7063.  
  7064. var mouseupHandler=function(){
  7065. self.working=false;
  7066. self.rotateIndicator.style.display='none';
  7067. document.removeEventListener('mousemove',moveHandler,true);
  7068. document.removeEventListener('mouseup',mouseupHandler,true);
  7069. };
  7070.  
  7071. document.addEventListener('mousemove',moveHandler,true);
  7072. document.addEventListener('mouseup',mouseupHandler,true);
  7073. },
  7074. convertToValidRadians:function(radians){
  7075. //转成0-90的等价角度。
  7076. var PI=Math.PI;
  7077. if(radians > PI){
  7078. radians = 2*PI - radians;
  7079. };
  7080. if(radians > 1/2*PI){
  7081. radians = PI - radians;
  7082. };
  7083. return radians;
  7084. },
  7085. getRotatedImgRectSize:function( radians, imgSize ){//通过旋转后的角度和图片的大小,求虚拟矩形的大小
  7086. imgSize= imgSize ? imgSize :{
  7087. h:this.img.clientHeight,
  7088. w:this.img.clentWidth,
  7089. };
  7090.  
  7091. if(typeof radians==='undefined'){
  7092. radians = this.rotatedRadians;
  7093. };
  7094.  
  7095. radians=this.convertToValidRadians(radians);
  7096.  
  7097. return {
  7098. h:this.notExponential(imgSize.h* Math.cos(radians) + imgSize.w * Math.sin(radians)),
  7099. w:this.notExponential(imgSize.h* Math.sin(radians) + imgSize.w * Math.cos(radians)),
  7100. };
  7101. },
  7102. getRotatedImgCliSize:function(rectSize,radians){//通过虚拟矩形的大小和图片的旋转角度,求图片的大小
  7103.  
  7104. if(typeof radians==='undefined'){
  7105. radians = this.rotatedRadians;
  7106. };
  7107.  
  7108. radians=this.convertToValidRadians(radians);
  7109.  
  7110. if(radians==0){
  7111. //radians=Math.PI/180 * 1/100;
  7112. return rectSize;
  7113. };
  7114.  
  7115. var h=(rectSize.h-rectSize.w * Math.tan(radians))/(Math.cos(radians)-Math.sin(radians)*Math.tan(radians));
  7116. var w=(rectSize.h - h*Math.cos(radians))/Math.sin(radians);
  7117. return {
  7118. h:h,
  7119. w:w,
  7120. };
  7121.  
  7122. },
  7123. setImgOffset:function(oriOffset,bImgSize,aImgSize){
  7124. var imgStyle=this.img.style;
  7125.  
  7126. //避免出现指数形式的数字和单位相加,导致变成无效值
  7127. var top=this.notExponential(oriOffset.top + (aImgSize.h-bImgSize.h)*1/2) + 'px';
  7128. var left=this.notExponential(oriOffset.left + (aImgSize.w-bImgSize.w)*1/2) + 'px';
  7129. imgStyle.top= top;
  7130. imgStyle.left= left;
  7131. },
  7132. setImgWindowOffset:function(oriOffset,bImgWindowSize,aImgWidnowSize,ratio){
  7133. ratio= ratio? ratio : {x:1/2,y:1/2};
  7134. var imgWindowStyle=this.imgWindow.style;
  7135. var top=oriOffset.top - (aImgWidnowSize.h-bImgWindowSize.h)*ratio.y + 'px';
  7136. var left=oriOffset.left - (aImgWidnowSize.w-bImgWindowSize.w)*ratio.x + 'px';
  7137. imgWindowStyle.top= top;
  7138. imgWindowStyle.left= left;
  7139. },
  7140. zoom:function(e,ratio){//e可能是undefined,可能是事件对象,可能是直接的缩放级别数字
  7141. var imgWindow=this.imgWindow;
  7142. var imgWindowCS=unsafeWindow.getComputedStyle(imgWindow);
  7143. var imgRectSize={
  7144. h:parseFloat(imgWindowCS.height),
  7145. w:parseFloat(imgWindowCS.width),
  7146. };
  7147.  
  7148. var rectOffset={
  7149. top:parseFloat(imgWindow.style.top),
  7150. left:parseFloat(imgWindow.style.left),
  7151. };
  7152.  
  7153. var img=this.img;
  7154. var self=this;
  7155.  
  7156. var zoom=function(level){//缩放到指定级别
  7157. if(typeof level=='undefined' || level<0 || level==self.zoomLevel)return;
  7158.  
  7159. var afterImgSize={
  7160. h:self.imgNaturalSize.h * level,
  7161. w:self.imgNaturalSize.w * level,
  7162. };
  7163. img.width=afterImgSize.w;
  7164. img.height=afterImgSize.h;
  7165.  
  7166. var afterimgRectSize=self.getRotatedImgRectSize( self.rotatedRadians, afterImgSize );
  7167. imgWindow.style.width=afterimgRectSize.w +'px';
  7168. imgWindow.style.height=afterimgRectSize.h + 'px';
  7169. self.setImgWindowOffset(rectOffset,imgRectSize,afterimgRectSize,ratio);
  7170. self.setImgOffset({top:0,left:0},afterImgSize,afterimgRectSize);//如果旋转了,调整偏移
  7171. self.zoomLevel=level;
  7172. self.setToolBadge('zoom',level);
  7173. self.keepScreenInside();
  7174. };
  7175.  
  7176. if(typeof e!='object'){
  7177. ratio=ratio? ratio : {
  7178. x:1/2,
  7179. y:1/2,
  7180. };
  7181. zoom(e);
  7182. return;
  7183. };
  7184.  
  7185. this.working=true;
  7186.  
  7187. ratio=this.getZoomRatio({
  7188. x:e.clientX,
  7189. y:e.clientY,
  7190. });
  7191.  
  7192.  
  7193. var moved;
  7194. var lastPageX=e.pageX;
  7195. var currentLevel=this.zoomLevel;
  7196. var moveFired=0;
  7197. var moveHandler=function(e){
  7198. moveFired++
  7199. if(moveFired < 2){//有时候点击的时候不小心会触发一发move
  7200. return;
  7201. };
  7202. moved=true;
  7203. var pageX=e.pageX;
  7204. var level;
  7205. if(pageX > lastPageX){//向右移,zoomin扩大
  7206. self.changeCursor('zoom',false);
  7207. level=0.05;
  7208. }else{//向左移,zoomout缩小
  7209. self.changeCursor('zoom',true);
  7210. level=-0.05;
  7211. };
  7212. lastPageX=pageX;
  7213. currentLevel += level;
  7214. zoom(currentLevel);
  7215. };
  7216.  
  7217. var mouseupHandler=function(e){
  7218. self.working=false;
  7219. document.removeEventListener('mousemove',moveHandler,true);
  7220. document.removeEventListener('mouseup',mouseupHandler,true);
  7221.  
  7222. var level=self.getNextZoomLevel();
  7223.  
  7224. if(self.zoomOut && self.altKeyUp){
  7225. self.zoomOut=false;
  7226. };
  7227.  
  7228. if(!moved){//如果没有平移缩放。
  7229. zoom(level);
  7230. };
  7231.  
  7232. self.changeCursor('zoom',self.zoomOut);
  7233.  
  7234. if(self.tempZoom && self.ctrlKeyUp && self.altKeyUp){
  7235. self.tempZoom=false;
  7236. self.changeCursor(self.selectedTool);
  7237. };
  7238.  
  7239. };
  7240.  
  7241. document.addEventListener('mousemove',moveHandler,true);
  7242. document.addEventListener('mouseup',mouseupHandler,true);
  7243. },
  7244. getNextZoomLevel:function(){
  7245. var level;
  7246. var self=this;
  7247. if(this.zoomOut){//缩小
  7248. ImgWindowC.zoomRangeR._find(function(value){
  7249. if(value < self.zoomLevel){
  7250. level=value;
  7251. return true;
  7252. }
  7253. })
  7254. }else{
  7255. ImgWindowC.zoomRange._find(function(value){
  7256. if(value > self.zoomLevel){
  7257. level=value;
  7258. return true;
  7259. };
  7260. });
  7261. }
  7262. return level;
  7263. },
  7264. getZoomRatio:function(mouseCoor){
  7265. var ibcRect=this.img.getBoundingClientRect();
  7266. var ratio={
  7267. x:(mouseCoor.x-ibcRect.left)/ibcRect.width,
  7268. y:(mouseCoor.y-ibcRect.top)/ibcRect.height,
  7269. };
  7270. if(ratio.x<0){
  7271. ratio.x=0
  7272. }else if(ratio.x>1){
  7273. ratio.x=1
  7274. };
  7275. if(ratio.y<0){
  7276. ratio.y=0
  7277. }else if(ratio.y>1){
  7278. ratio.y=1
  7279. };
  7280. return ratio;
  7281. },
  7282. aerialView:function(e){
  7283. this.working=true;
  7284. //记住现在的缩放比例
  7285. var cLevel=this.zoomLevel;
  7286.  
  7287. var wSize=getWindowSize();
  7288. wSize.h -= 16;
  7289. wSize.w -= 16;
  7290.  
  7291. var imgWindow=this.imgWindow;
  7292. var imgWindowCS=unsafeWindow.getComputedStyle(imgWindow);
  7293. var rectSize={
  7294. h:parseFloat(imgWindowCS.height),
  7295. w:parseFloat(imgWindowCS.width),
  7296. };
  7297. var rectRatio=rectSize.h/rectSize.w;
  7298. var windowRatio=wSize.h/wSize.w;
  7299.  
  7300. var size;
  7301. var rangeSize={};
  7302. if(rectRatio > windowRatio){
  7303. size={
  7304. h:wSize.h,
  7305. w:wSize.h / rectRatio,
  7306. };
  7307. rangeSize.h=Math.min(wSize.h * (size.h / rectSize.h), size.h);
  7308. rangeSize.w=Math.min(rangeSize.h / windowRatio , size.w);
  7309. }else{
  7310. size={
  7311. w:wSize.w,
  7312. h:wSize.w * rectRatio,
  7313. };
  7314. rangeSize.w=Math.min(wSize.w * (size.w / rectSize.w), size.w);
  7315. rangeSize.h=Math.min(rangeSize.w * windowRatio , size.h);
  7316. };
  7317.  
  7318.  
  7319. this.zoom(this.getRotatedImgCliSize(size).w/this.imgNaturalSize.w);
  7320.  
  7321. this.center(true,true);
  7322.  
  7323. this.keepScreenInside();
  7324.  
  7325. var viewRange=this.viewRange;
  7326. var vRS=viewRange.style;
  7327. vRS.display='block';
  7328. vRS.height=rangeSize.h + 'px';
  7329. vRS.width=rangeSize.w + 'px';
  7330. vRS.top=0 + 'px';
  7331. vRS.left=0 + 'px';
  7332.  
  7333.  
  7334.  
  7335. var viewRangeRect=viewRange.getBoundingClientRect();
  7336. var scrolled=getScrolled();
  7337. var viewRangeCenterCoor={
  7338. x:viewRangeRect.left + scrolled.x + 1/2 * rangeSize.w,
  7339. y:viewRangeRect.top + scrolled.y + 1/2 * rangeSize.h,
  7340. };
  7341.  
  7342. var self=this;
  7343.  
  7344. var moveRange={
  7345. x:[8,8+size.w-rangeSize.w],
  7346. y:[8,8+size.h-rangeSize.h]
  7347. };
  7348.  
  7349.  
  7350. function setViewRangePosition(pageXY){
  7351. var top=pageXY.y - viewRangeCenterCoor.y;
  7352. var left=pageXY.x - viewRangeCenterCoor.x;
  7353. if(top<=moveRange.y[0]){
  7354. top=moveRange.y[0];
  7355. }else if(top>=moveRange.y[1]){
  7356. top=moveRange.y[1];
  7357. };
  7358. vRS.top= top + 'px';
  7359. if(left<=moveRange.x[0]){
  7360. left=moveRange.x[0];
  7361. }else if(left>=moveRange.x[1]){
  7362. left=moveRange.x[1];
  7363. };
  7364. vRS.left= left + 'px';
  7365. };
  7366.  
  7367. setViewRangePosition({
  7368. x:e.pageX,
  7369. y:e.pageY,
  7370. });
  7371.  
  7372. var moveHandler=function(e){
  7373. setViewRangePosition({
  7374. x:e.pageX,
  7375. y:e.pageY,
  7376. });
  7377. };
  7378.  
  7379. var mouseupHandler=function(){
  7380. self.working=false;
  7381. viewRange.style.display='none';
  7382. self.zoom(cLevel);
  7383. var scrolled=getScrolled();
  7384. imgWindow.style.top= -13 - rectSize.h * ((parseFloat(vRS.top) - moveRange.y[0])/size.h) + scrolled.y +'px';
  7385. imgWindow.style.left= -13 - rectSize.w * ((parseFloat(vRS.left) - moveRange.x[0])/size.w) + scrolled.x +'px';
  7386.  
  7387. //说明图片的高度没有屏幕高,居中
  7388. //说明图片的宽度没有屏幕宽,居中
  7389. self.center(rangeSize.w == size.w , rangeSize.h == size.h);
  7390.  
  7391. self.keepScreenInside();
  7392.  
  7393. document.removeEventListener('mousemove',moveHandler,true);
  7394. document.removeEventListener('mouseup',mouseupHandler,true);
  7395. };
  7396. document.addEventListener('mousemove',moveHandler,true);
  7397. document.addEventListener('mouseup',mouseupHandler,true);
  7398. },
  7399. setToolBadge:function(tool,content){
  7400. var scale=0;
  7401. switch(tool){
  7402. case 'zoom':{
  7403. scale=2;
  7404. }break;
  7405. case 'rotate':{
  7406. scale=1;
  7407. }break;
  7408. default:break;
  7409. }
  7410. content=typeof content=='string'? content : content.toFixed(scale);
  7411. this.toolMap[tool].nextElementSibling.textContent=content;
  7412. },
  7413. notExponential:function(num){//不要转为指数形势
  7414. if(num>0){
  7415. if(num >= 999999999999999934463){
  7416. return 999999999999999934463;
  7417. }else if(num <= 0.000001){
  7418. return 0.000001;
  7419. };
  7420. }else if(num < 0){
  7421. if(num >= -0.000001){
  7422. return -0.000001;
  7423. }else if(num <= -999999999999999934463){
  7424. return -999999999999999934463
  7425. };
  7426. };
  7427.  
  7428. return num;
  7429. },
  7430.  
  7431. blur:function(e){
  7432. if(!this.focused)return;
  7433. var imgWindow =this.imgWindow;
  7434. //点击imgWinodw的外部的时候失去焦点
  7435. if(e!==true && imgWindow.contains(e.target))return;
  7436. imgWindow.classList.remove('pv-pic-window-container_focus');
  7437. this.toolbar.classList.remove('pv-pic-window-toolbar_focus');
  7438. this.closeButton.classList.remove('pv-pic-window-close_focus');
  7439. //this.searchButton.classList.remove('pv-pic-window-search_focus');
  7440. this.img.classList.remove('pv-pic-window-pic_focus');
  7441. document.removeEventListener('mousedown',this._blur,true);
  7442. document.removeEventListener('keydown',this._focusedKeydown,true);
  7443. document.removeEventListener('keyup',this._focusedKeyup,true);
  7444. this.changeCursor('default');
  7445. ImgWindowC.selectedTool=this.selectedTool;
  7446. this.focused=false;
  7447. },
  7448. focus:function(){
  7449. if(this.focused)return;
  7450. this.imgWindow.classList.add('pv-pic-window-container_focus');
  7451. this.toolbar.classList.add('pv-pic-window-toolbar_focus');
  7452. this.closeButton.classList.add('pv-pic-window-close_focus');
  7453. //this.searchButton.classList.add('pv-pic-window-search_focus');
  7454. this.img.classList.add('pv-pic-window-pic_focus');
  7455. this.imgWindow.style.zIndex= ImgWindowC.styleZIndex;
  7456. this.zIndex=ImgWindowC.styleZIndex;
  7457. ImgWindowC.styleZIndex ++;
  7458. document.addEventListener('keydown',this._focusedKeydown,true);
  7459. document.addEventListener('keyup',this._focusedKeyup,true);
  7460. document.addEventListener('mousedown',this._blur,true);
  7461.  
  7462. //还原鼠标样式。
  7463. this.changeCursor(this.selectedTool);
  7464.  
  7465. if(prefs.imgWindow.syncSelectedTool && ImgWindowC.selectedTool){
  7466. this.selectTool(ImgWindowC.selectedTool);
  7467. };
  7468.  
  7469. this.focused=true;
  7470. },
  7471. focusedKeyup:function(e){
  7472. var keyCode=e.keyCode;
  7473. var valid=[32,18,16,72,17,72,82,90,67];
  7474. if(valid.indexOf(keyCode)==-1)return;
  7475.  
  7476. e.preventDefault();
  7477.  
  7478. switch(keyCode){
  7479. case 32:{//空格键,临时切换到移动
  7480. this.spaceKeyUp=true;
  7481. if(!this.tempHand)return;//如果之前没有临时切换到抓手工具(当已经在工作的时候,按下空格不会临时切换到抓手工具)
  7482. if(!this.working){//松开按键的时候,没有在继续平移了。
  7483. this.tempHand=false;
  7484. this.changeCursor(this.selectedTool);
  7485. };
  7486. }break;
  7487. case 18:{//alt键盘切换缩小放大。
  7488. this.altKeyUp=true;
  7489. if(!this.zoomOut)return;
  7490. if(!this.working){
  7491. this.zoomOut=false;
  7492. this.changeCursor('zoom');
  7493. if(this.tempZoom && this.ctrlKeyUp){
  7494. this.tempZoom=false;
  7495. this.changeCursor(this.selectedTool);
  7496. };
  7497. };
  7498. }break;
  7499. case 16:{//shift键,旋转的时候按住shift键,步进缩放。
  7500. this.shiftKeyUp=true;
  7501. }break;
  7502. case 17:{//ctrl键
  7503. clearTimeout(this.ctrlkeyDownTimer);
  7504. if(!this.justCKeyUp){//如果刚才没有松开c,规避划词软件的ctrl+c松开
  7505. this.ctrlKeyUp=true;
  7506. if(!this.tempZoom)return;//如果没有切换到了缩放
  7507. if(!this.working && this.altKeyUp){
  7508. this.tempZoom=false;
  7509. this.changeCursor(this.selectedTool);
  7510. };
  7511. };
  7512. }break;
  7513. case 67:{//c键
  7514. this.justCKeyUp=true;
  7515. var self=this;
  7516. clearTimeout(this.justCKeyUpTimer);
  7517. this.justCKeyUpTimer=setTimeout(function(){
  7518. self.justCKeyUp=false;
  7519. },100)
  7520. }break;
  7521. case 72:{//h键
  7522. this.hKeyUp=true;
  7523. }break;
  7524. case 82:{//r键
  7525. this.rKeyUp=true;
  7526. }break;
  7527. case 90:{//z键
  7528. this.zKeyUp=true;
  7529. }break;
  7530. default:break;
  7531. };
  7532.  
  7533. if([72,82,90].indexOf(keyCode)!=-1){
  7534. if(!this.working && this.restoreBeforeTool){
  7535. this.restoreBeforeTool=false;
  7536. this.selectTool(this.beforeTool);
  7537. };
  7538. };
  7539. },
  7540. focusedKeydown:function(e){
  7541. var keyCode=e.keyCode;
  7542. var valid=[32,82,72,90,18,16,17,27,67];//有效的按键
  7543. if(valid.indexOf(keyCode)==-1) return;
  7544.  
  7545. e.preventDefault();
  7546.  
  7547. if(this.working){//working的时候也可以接受按下shift键,以便旋转的时候可以任何时候按下
  7548. if(keyCode==16){//shift键
  7549. this.shiftKeyUp=false;
  7550. };
  7551. return;
  7552. };
  7553.  
  7554. switch(keyCode){
  7555. case 82:{//r键,切换到旋转工具
  7556. if(this.rKeyUp){
  7557. this.rKeyUp=false;
  7558. this.beforeTool=this.selectedTool;
  7559. this.selectTool('rotate');
  7560. };
  7561. }break;
  7562. case 72:{//h键,切换到抓手工具
  7563. if(this.hKeyUp){
  7564. this.hKeyUp=false;
  7565. this.beforeTool=this.selectedTool;
  7566. this.selectTool('hand');
  7567. };
  7568. }break;
  7569. case 90:{//z键,切换到缩放工具
  7570. if(this.zKeyUp){
  7571. this.zKeyUp=false;
  7572. this.beforeTool=this.selectedTool;
  7573. this.selectTool('zoom');
  7574. };
  7575. }break;
  7576. case 32:{//空格键阻止,临时切换到抓手功能
  7577. if(this.spaceKeyUp){
  7578. this.spaceKeyUp=false;
  7579. if(this.selectedTool!='hand'){
  7580. this.tempHand=true;
  7581. this.changeCursor('hand');
  7582. };
  7583. };
  7584. }break;
  7585. case 18:{//alt键,在当前选择是缩放工具的时候,按下的时候切换到缩小功能
  7586. if(this.altKeyUp){
  7587. if((this.selectedTool!='zoom' && !this.tempZoom) || this.zoomOut)return;
  7588. this.zoomOut=true;
  7589. this.altKeyUp=false;
  7590. this.changeCursor('zoom',true);
  7591. };
  7592. }break;
  7593. case 17:{//ctrl键临时切换到缩放工具
  7594. if(this.ctrlKeyUp){
  7595. var self=this;
  7596. this.ctrlkeyDownTimer=setTimeout(function(){//规避词典软件的ctrl+c,一瞬间切换到缩放的问题
  7597. self.ctrlKeyUp=false;
  7598. if(self.selectedTool!='zoom'){
  7599. self.tempZoom=true;
  7600. self.changeCursor('zoom');
  7601. };
  7602. },100);
  7603. };
  7604. }break;
  7605. case 67:{//c键
  7606. clearTimeout(this.ctrlkeyDownTimer);
  7607. }break;
  7608. case 27:{//ese关闭窗口
  7609. if(prefs.imgWindow.close.escKey){
  7610. this.remove();
  7611. };
  7612. }break;
  7613. default:break;
  7614. };
  7615. },
  7616.  
  7617. toolbarEventHandler:function(e){
  7618. e.stopPropagation();
  7619. var target=e.target;
  7620. var toolMap=this.toolMap;
  7621. for(var i in toolMap){
  7622. if(toolMap.hasOwnProperty(i) && toolMap[i]==target){
  7623. switch(e.type){
  7624. case 'mousedown':{
  7625. this.selectTool(i);
  7626. }break;
  7627. case 'dblclick':{
  7628. this.dblclickCommand(i);
  7629. }break;
  7630. default:break;
  7631. };
  7632. break;
  7633. };
  7634. };
  7635. },
  7636. imgWindowEventHandler:function(e){
  7637. e.stopPropagation();
  7638. var selectedTool=this.selectedTool;
  7639. if(selectedTool == "hand" && prefs.imgWindow.suitLongImg && this.isLongImg){
  7640. var inScroll=this.imgWindow.style.overflow=="scroll";
  7641. if(e.type == "wheel" && inScroll)
  7642. return;
  7643. if(e.type == "click" && !this.moving){
  7644. var wSize=getWindowSize();
  7645. var imgWindow=this.imgWindow;
  7646. var scrolled=getScrolled();
  7647. var origTop=parseFloat(imgWindow.style.top);
  7648. if(inScroll){
  7649. imgWindow.style.top = parseFloat(imgWindow.style.top) - getScrolled(imgWindow).y +'px';
  7650. }
  7651. this.imgWindow.style.height=inScroll?"":"100%";
  7652. this.imgWindow.style.maxWidth=inScroll?"":"100%";
  7653. this.imgWindow.style.overflow=inScroll?"":"scroll";
  7654. this.closeButton.style.display=inScroll?"":"none";
  7655. //this.center(true , true);
  7656. if(!inScroll){
  7657. imgWindow.style.top= (wSize.h - imgWindow.offsetHeight)/2 + scrolled.y +'px';
  7658. var scrollTop=parseFloat(imgWindow.style.top)-origTop;
  7659. if(this.imgWindow.scrollTop)this.imgWindow.scrollTop = scrollTop;
  7660. else if(this.imgWindow.pageYOffset)this.imgWindow.pageYOffset = scrollTop;
  7661. else if(this.imgWindow.scrollY)this.imgWindow.scrollY = scrollTop;
  7662. }
  7663. this.keepScreenInside();
  7664. }
  7665. }
  7666. switch(e.type){
  7667. case 'click':{//阻止opera的图片保存
  7668. this.moving=false;
  7669. if(e.ctrlKey && e.target.nodeName=='IMG'){
  7670. e.preventDefault();
  7671. }
  7672. }break;
  7673. case 'mousedown':
  7674. case 'touchstart':{
  7675. if(!this.focused){//如果没有focus,先focus
  7676. this.focus();
  7677. this.keepScreenInside();
  7678. };
  7679.  
  7680. var target=e.target;
  7681. if(e.button==2){//由于rotate时候的覆盖层问题,修复右键的图片菜单弹出
  7682. if(target!=this.rotateOverlayer)return;
  7683. var self=this;
  7684. this.rotateOverlayer.style.display='none';
  7685. var upHandler=function(){
  7686. document.removeEventListener('mouseup',upHandler,true);
  7687. setTimeout(function(){
  7688. self.rotateOverlayer.style.display='block';
  7689. },10);
  7690. };
  7691. document.addEventListener('mouseup',upHandler,true);
  7692. return;
  7693. };
  7694.  
  7695. if((e.button!=0 && e.type!="touchstart") || (target!=this.imgWindow && target!=this.img && target!=this.rotateOverlayer))return;
  7696. e.preventDefault();
  7697. if(this.tempHand){
  7698. this.move(e);
  7699. }else if(this.tempZoom){
  7700. this.zoom(e);
  7701. }else if(selectedTool=='hand'){
  7702. this.restoreBeforeTool=!this.hKeyUp;
  7703. if(this.hKeyUp){
  7704. this.move(e);
  7705. }else{//鸟瞰视图
  7706. this.aerialView(e);
  7707. };
  7708. }else if(selectedTool=='rotate'){
  7709. var origin={//旋转原点
  7710. x:e.clientX - 30,//稍微偏左一点。
  7711. y:e.clientY ,
  7712. };
  7713.  
  7714. var rIS=this.rotateIndicator.style;
  7715. //rIS.display='block';
  7716. rIS.top=origin.y + 'px';
  7717. rIS.left=origin.x + 'px';
  7718.  
  7719. this.restoreBeforeTool=!this.rKeyUp;
  7720. this.rotate(origin);
  7721. }else if(selectedTool=='zoom'){
  7722. this.restoreBeforeTool=!this.zKeyUp;
  7723. this.zoom(e);
  7724. };
  7725. }break;
  7726. case 'wheel':{
  7727. if(!this.focused)return;//如果没有focus
  7728. if(e.deltaY===0)return;//非Y轴的滚动
  7729. e.preventDefault();
  7730. if(this.working)return;
  7731. var oriZoomOut=this.zoomOut;
  7732. this.zoomOut = !!(e.deltaY > 0);
  7733.  
  7734. var ratio=this.getZoomRatio({
  7735. x:e.clientX,
  7736. y:e.clientY,
  7737. });
  7738.  
  7739. var level=this.getNextZoomLevel();
  7740.  
  7741. this.zoom(level,ratio);
  7742. this.zoomOut=oriZoomOut;
  7743. }break;
  7744. default:break;
  7745. };
  7746. },
  7747.  
  7748. dblclickCommand:function(tool){
  7749. var done;
  7750. switch(tool){
  7751. case 'hand':{//双击居中,并且适应屏幕
  7752. this.zoom(1);
  7753. this.fitToScreen();
  7754. this.center(true,true);
  7755. this.keepScreenInside();
  7756. }break;
  7757. case 'rotate':{//双击还原旋转
  7758. if(this.rotatedRadians==0)return;
  7759. done=true;
  7760. this.rotate(0,true);
  7761. }break;
  7762. case 'zoom':{//双击还原缩放
  7763. if(this.zoomLevel==1)return;
  7764. done=true;
  7765. this.zoom(1,{x:0,y:0});
  7766. }break;
  7767. default:break;
  7768. };
  7769.  
  7770. if((tool=='rotate' || tool=='zoom') && done){
  7771. var scrolled=getScrolled();
  7772. var imgWindow=this.imgWindow;
  7773. var imgWinodowRect=imgWindow.getBoundingClientRect();
  7774. var imgWindowStyle=imgWindow.style;
  7775. if(imgWinodowRect.left<40){
  7776. imgWindowStyle.left=40 + scrolled.x + 'px';
  7777. };
  7778. if(imgWinodowRect.top<-5){
  7779. imgWindowStyle.top=-5 + scrolled.y +'px';
  7780. };
  7781. this.keepScreenInside();
  7782. };
  7783.  
  7784. },
  7785. doFlipCommand:function(command){
  7786. var map={
  7787. fv:[/scaleY\([^)]*\)/i,' scaleY(-1) '],
  7788. fh:[/scaleX\([^)]*\)/i,' scaleX(-1) '],
  7789. };
  7790.  
  7791. var iTransform=this.img.style[support.cssTransform];
  7792.  
  7793. var toolClassList=this.toolMap[command].classList;
  7794.  
  7795. if(map[command][0].test(iTransform)){
  7796. iTransform=iTransform.replace(map[command][0],'');
  7797. toolClassList.remove(this.selectedToolClass);
  7798. }else{
  7799. iTransform += map[command][1];
  7800. toolClassList.add(this.selectedToolClass);
  7801. };
  7802. this.img.style[support.cssTransform]=iTransform;
  7803.  
  7804. },
  7805. selectTool:function(tool){
  7806. var command=['fv','fh'];
  7807. if(command.indexOf(tool)==-1){//工具选择
  7808. if(this.selectedTool==tool){
  7809. if(tool=="rotate"){
  7810. var PI=Math.PI;
  7811. var value=this.rotatedRadians + 90 * PI/180;
  7812. if(value>=2*PI){
  7813. value-=2*PI;
  7814. }
  7815. this.rotate(value,true);
  7816. }
  7817. return;
  7818. }
  7819. var selectedTool=this.selectedTool;
  7820. this.selectedTool=tool;
  7821. if(this.tempHand || this.tempZoom){//临时工具中。不变鼠标
  7822. return;
  7823. };
  7824.  
  7825. this.rotateOverlayer.style.display=(tool=='rotate'? 'block' : 'none');//这个覆盖层是为了捕捉双击或者单击事件。
  7826.  
  7827. if(selectedTool){
  7828. this.toolMap[selectedTool].classList.remove(this.selectedToolClass);
  7829. };
  7830. this.toolMap[tool].classList.add(this.selectedToolClass);
  7831. this.changeCursor(tool);
  7832. }else{//命令
  7833. this.doFlipCommand(tool);
  7834. };
  7835. },
  7836. changeCursor:function(tool,zoomOut){
  7837. if(tool=='zoom'){
  7838. tool+=zoomOut? '-out' : '-in';
  7839. };
  7840. if(this.cursor==tool)return;
  7841. this.cursor=tool;
  7842.  
  7843. var cursor;
  7844.  
  7845. switch(tool){
  7846. case 'hand':{
  7847. cursor=support.cssCursorValue.grab || 'pointer';
  7848. }break;
  7849. case 'handing':{
  7850. cursor=support.cssCursorValue.grabbing || 'pointer';
  7851. }break;
  7852. case 'zoom-in':{
  7853. cursor=support.cssCursorValue.zoomIn;
  7854. }break;
  7855. case 'zoom-out':{
  7856. cursor=support.cssCursorValue.zoomOut;
  7857. }break;
  7858. case 'rotate':{
  7859. cursor='progress';
  7860. }break;
  7861. case 'default':{
  7862. cursor='';
  7863. }break;
  7864. };
  7865.  
  7866. if(typeof cursor!='undefined'){
  7867. this.imgWindow.style.cursor=cursor;
  7868. };
  7869.  
  7870. },
  7871.  
  7872. remove:function(){
  7873. if(this.removed)return;
  7874. this.removed=true;
  7875. this.imgWindow.classList.remove("pv-pic-window-transition-all");
  7876. this.blur(true);
  7877. this.imgWindow.style.opacity=0;
  7878. let self = this;
  7879. setTimeout(function(){
  7880. self.img.src= prefs.icons.brokenImg_small;//如果在加载中取消,图片也取消读取。
  7881. self.imgWindow.parentNode.removeChild(self.imgWindow);
  7882. },300);
  7883.  
  7884. var index=ImgWindowC.all.indexOf(this);
  7885. ImgWindowC.all.splice(index,1);
  7886.  
  7887. //focus next
  7888. if(ImgWindowC.all.length==0){
  7889. if(ImgWindowC.overlayer){
  7890. ImgWindowC.overlayer.style.display='none';
  7891. };
  7892. }else{
  7893. var topmost=0;
  7894. ImgWindowC.all.forEach(function(iwin){
  7895. if(iwin.zIndex > topmost){
  7896. topmost=iwin;
  7897. };
  7898. });
  7899. if(topmost){
  7900. topmost.focus();
  7901. };
  7902. };
  7903.  
  7904. },
  7905.  
  7906. };
  7907.  
  7908. // 载入动画
  7909. function LoadingAnimC(data,buttonType,waitImgLoad,openInTopWindow){
  7910. this.args=arrayFn.slice.call(arguments,0);
  7911. this.data=data;//data
  7912. this.buttonType=buttonType;//点击的按钮类型
  7913. this.openInTopWindow=openInTopWindow;//是否在顶层窗口打开,如果在frame里面的话
  7914. this.waitImgLoad=waitImgLoad;//是否等待完全读取后打开
  7915. this.init();
  7916. };
  7917.  
  7918. LoadingAnimC.all=[];
  7919.  
  7920. LoadingAnimC.prototype={
  7921. init:function(){
  7922. LoadingAnimC.all.push(this);
  7923. this.addStyle();
  7924. var container=document.createElement('span');
  7925.  
  7926. container.className='pv-loading-container';
  7927. this.loadingAnim=container;
  7928.  
  7929. container.title=i18n("loading")+':' + this.data.src;
  7930. container.innerHTML=
  7931. '<span class="pv-loading-button pv-loading-retry" title="重试"></span>'+
  7932. '<span class="pv-loading-button pv-loading-cancle" title="取消"></span>';
  7933.  
  7934. document.body.appendChild(container);
  7935.  
  7936. var self = this;
  7937. container.addEventListener('click',function(e){
  7938. var tcl=e.target.classList;
  7939. if(tcl.contains('pv-loading-cancle')){
  7940. self.imgReady.abort();
  7941. self.remove();
  7942. }else if(tcl.contains('pv-loading-retry')){
  7943. self.remove();
  7944. new LoadingAnimC(self.args[0],self.args[1],self.args[2],self.args[3]);
  7945. };
  7946. },true);
  7947.  
  7948. this.setPosition();
  7949.  
  7950. if (this.buttonType == 'current') {
  7951. this.loadImg(this.data.imgSrc);
  7952. } else {
  7953. if (!this.data.xhr) {
  7954. if(this.buttonType == 'search'){
  7955. sortSearch();
  7956. let from=0;
  7957. let searchFun=function(){
  7958. searchImgByImg(self.data.imgSrc, null, function(srcs, index){
  7959. let src=srcs.shift();
  7960. if(index==3){
  7961. self.loadImg(src, srcs);
  7962. }else{
  7963. from=index+1;
  7964. self.loadImg(src, srcs, searchFun);
  7965. }
  7966. },function(e) {
  7967. self.error("网络错误");
  7968. },function(e) {
  7969. self.error("没有找到原图");
  7970. }, from);
  7971. };
  7972. searchFun();
  7973. }else{
  7974. this.loadImg(this.data.src||this.data.imgSrc, this.data.srcs);
  7975. }
  7976. } else {
  7977. xhrLoad.load({
  7978. url: this.data.src,
  7979. xhr: this.data.xhr,
  7980. cb: function(imgSrc, imgSrcs, caption) {
  7981. if (imgSrc) {
  7982. self.loadImg(imgSrc, imgSrcs);
  7983. } else {
  7984. self.error();
  7985. }
  7986. },
  7987. onerror: function() {
  7988. self.error();
  7989. }
  7990. });
  7991. }
  7992. }
  7993. },
  7994. addStyle:function(){
  7995. if(LoadingAnimC.styleAdded)return;
  7996. LoadingAnimC.styleAdded=true;
  7997. var style=document.createElement('style');
  7998. style.type='text/css';
  7999. style.textContent='\
  8000. .pv-loading-container {\
  8001. position: absolute;\
  8002. z-index:999999997;\
  8003. background: black url("'+prefs.icons.loading+'") center no-repeat;\
  8004. background-origin: content-box;\
  8005. border: none;\
  8006. padding: 1px 30px 1px 2px;\
  8007. margin: 0;\
  8008. opacity: 0.7;\
  8009. height: 24px;\
  8010. min-width: 24px;\
  8011. box-shadow: 2px 2px 0px #666;\
  8012. -webkit-transition: opacity 0.15s ease-in-out;\
  8013. transition: opacity 0.15s ease-in-out;\
  8014. }\
  8015. .pv-loading-container:hover {\
  8016. opacity: 0.9;\
  8017. }\
  8018. .pv-loading-button {\
  8019. cursor: pointer;\
  8020. height: 24px;\
  8021. width: 24px;\
  8022. position: absolute;\
  8023. right: 0;\
  8024. top: 0;\
  8025. opacity: 0.4;\
  8026. background:transparent center no-repeat;\
  8027. -webkit-transition: opacity 0.15s ease-in-out;\
  8028. transition: opacity 0.15s ease-in-out;\
  8029. }\
  8030. .pv-loading-button:hover {\
  8031. opacity: 1;\
  8032. }\
  8033. .pv-loading-cancle{\
  8034. background-image: url("'+prefs.icons.loadingCancle+'");\
  8035. }\
  8036. .pv-loading-retry{\
  8037. display:none;\
  8038. background-image: url("'+prefs.icons.retry+'");\
  8039. }\
  8040. .pv-loading-container_error{\
  8041. background-image:none;\
  8042. }\
  8043. .pv-loading-container_error::after{\
  8044. content:"'+i18n("loadError")+'";\
  8045. line-height: 24px;\
  8046. color: red;\
  8047. font-size: 14px;\
  8048. display:inline;\
  8049. }\
  8050. .pv-loading-container_error .pv-loading-cancle{\
  8051. display:none;\
  8052. }\
  8053. .pv-loading-container_error .pv-loading-retry{\
  8054. display:block;\
  8055. }\
  8056. ';
  8057. document.head.appendChild(style);
  8058. },
  8059. remove:function(){
  8060. if(!this.removed){
  8061. this.removed=true;
  8062. this.loadingAnim.parentNode.removeChild(this.loadingAnim);
  8063. LoadingAnimC.all.splice(LoadingAnimC.all.indexOf(this),1);
  8064. };
  8065. },
  8066. error:function(msg,img,e){
  8067. if(msg)debug(msg);
  8068. this.loadingAnim.classList.add('pv-loading-container_error');
  8069. debug('Picviewer CE+ 载入大图错误:%o', this.data);
  8070.  
  8071. var self=this;
  8072. setTimeout(function(){
  8073. self.remove();
  8074. },3000);
  8075. },
  8076. setPosition:function(){
  8077. var position=getContentClientRect(this.data.img);
  8078. var cs=this.loadingAnim.style;
  8079. var scrolled=getScrolled();
  8080. cs.top=position.top + scrolled.y +1 + 'px';
  8081. cs.left=position.left + scrolled.x +1 + 'px';
  8082. cs.removeProperty('display');
  8083. },
  8084.  
  8085. // 根据 imgSrc 载入图片,imgSrcs 为备用图片地址,imgSrc 加载失败备用
  8086. loadImg: function(imgSrc, imgSrcs, nextFun) {
  8087. var self = this;
  8088.  
  8089. var img = document.createElement('img');
  8090. img.src = imgSrc;
  8091.  
  8092. var opts = {
  8093. error: function(e) {
  8094. if (Array.isArray(imgSrcs)) {
  8095. var src = imgSrcs.shift();
  8096. if (src) {
  8097. self.loadImg(src, imgSrcs, nextFun);
  8098. return;
  8099. }
  8100. }
  8101.  
  8102. if(nextFun) nextFun();
  8103. else self.error('', this, e);
  8104. },
  8105. };
  8106.  
  8107. opts[self.waitImgLoad ? 'load' : 'ready'] = function(e) {
  8108. self.load(this, e);
  8109. };
  8110.  
  8111. self.imgReady = imgReady(img, opts);
  8112. },
  8113.  
  8114. load:function(img,e){
  8115. this.remove();
  8116. this.img=img;
  8117. var buttonType=this.buttonType;
  8118.  
  8119. if(buttonType=='gallery'){
  8120. if(!gallery){
  8121. gallery=new GalleryC();
  8122. gallery.data=[];
  8123. }
  8124. var allData=gallery.getAllValidImgs();
  8125. allData.target=this.data;
  8126. this.data=allData;
  8127. };
  8128.  
  8129. var self=this;
  8130. function openInTop(){
  8131. var data=self.data;
  8132.  
  8133. //删除不能发送的项。
  8134. var delCantClone=function(obj){
  8135. if(!obj)return;
  8136. delete obj.img;
  8137. delete obj.imgPA;
  8138. };
  8139.  
  8140. if(Array.isArray(data)){
  8141. frameSentSuccessData=frameSentData;
  8142. frameSentData=cloneObject(data,true);
  8143. delCantClone(data.target);
  8144. data.forEach(function(obj){
  8145. delCantClone(obj);
  8146. });
  8147. }else{
  8148. delCantClone(data);
  8149. };
  8150.  
  8151. window.postMessage({
  8152. messageID:messageID,
  8153. src:img.src,
  8154. data:data,
  8155. command:'open',
  8156. buttonType:buttonType,
  8157. to:'top',
  8158. },'*');
  8159. };
  8160.  
  8161. if(this.openInTopWindow && isFrame && topWindowValid!==false && buttonType!='magnifier'){
  8162. if(topWindowValid){
  8163. openInTop();
  8164. }else{//先发消息问问顶层窗口是不是非frameset窗口
  8165. window.postMessage({
  8166. messageID:messageID,
  8167. command:'topWindowValid',
  8168. to:'top',
  8169. },'*');
  8170.  
  8171. document.addEventListener('pv-topWindowValid',function(e){
  8172. topWindowValid=e.detail;
  8173. if(topWindowValid){//如果顶层窗口有效
  8174. openInTop();
  8175. }else{
  8176. self.open();
  8177. };
  8178. },true);
  8179. };
  8180.  
  8181. }else{
  8182. this.open();
  8183. };
  8184.  
  8185.  
  8186. },
  8187. open:function(){
  8188. switch(this.buttonType){
  8189. case 'gallery':{
  8190. if(!gallery){
  8191. gallery=new GalleryC();
  8192. };
  8193. gallery.load(this.data,this.from);
  8194. }break;
  8195. case 'magnifier':{
  8196. new MagnifierC(this.img,this.data);
  8197. }break;
  8198. case 'actual':;
  8199. case 'search':;
  8200. case 'current':;
  8201. case 'original':{//original 是为了兼容以前的规则
  8202. if(this.data.src!=this.img.src)this.data.src=this.img.src;
  8203. new ImgWindowC(this.img, this.data);
  8204. }break;
  8205. };
  8206. },
  8207. };
  8208.  
  8209. //工具栏
  8210. function FloatBarC(){
  8211. this.init();
  8212. };
  8213.  
  8214. FloatBarC.prototype={
  8215. init:function(){
  8216. this.addStyle();
  8217. var container=document.createElement('span');
  8218. container.id='pv-float-bar-container';
  8219. container.innerHTML=
  8220. '<span class="pv-float-bar-button"></span>'+
  8221. '<span class="pv-float-bar-button"></span>'+
  8222. '<span class="pv-float-bar-button"></span>'+
  8223. //'<span class="pv-float-bar-button"></span>'+
  8224. '<span class="pv-float-bar-button"></span>';
  8225. document.body.appendChild(container);
  8226.  
  8227. var buttons={
  8228. };
  8229. this.buttons=buttons;
  8230. this.children=container.children;
  8231.  
  8232. arrayFn.forEach.call(this.children,function(child,index){
  8233. var titleMap={
  8234. actual:i18n("actualBtn"),
  8235. search:i18n("searchBtn"),
  8236. gallery:i18n("galleryBtn"),
  8237. current:i18n("currentBtn"),
  8238. magnifier:i18n("magnifierBtn"),
  8239. };
  8240. var buttonName=prefs.floatBar.butonOrder[index];
  8241. buttons[buttonName]=child;
  8242. child.title=titleMap[buttonName];
  8243. child.classList.add('pv-float-bar-button-' + buttonName);
  8244. });
  8245.  
  8246.  
  8247. this.floatBar=container;
  8248.  
  8249.  
  8250. var self=this;
  8251. container.addEventListener('click',function(e){
  8252. var buttonType;
  8253. var target=e.target;
  8254. for(var type in buttons){
  8255. if(!buttons.hasOwnProperty(type))return;
  8256. if(target==buttons[type]){
  8257. buttonType=type;
  8258. break;
  8259. };
  8260. };
  8261. if(!buttonType)return;
  8262.  
  8263. self.hide();
  8264. self.open(e,buttonType);
  8265.  
  8266. },true);
  8267.  
  8268.  
  8269. addCusMouseEvent('mouseleave',container,function(e){
  8270. clearTimeout(self.hideTimer);
  8271. self.hideTimer=setTimeout(function(){
  8272. self.hide();
  8273. },prefs.floatBar.hideDelay);
  8274. });
  8275.  
  8276. addCusMouseEvent('mouseenter',container,function(e){
  8277. clearTimeout(self.hideTimer);
  8278. clearTimeout(self.showTimer);
  8279. clearTimeout(self.globarOutTimer);
  8280. });
  8281.  
  8282. this._scrollHandler=this.scrollHandler.bind(this);
  8283. },
  8284. addStyle:function(){
  8285. var style=document.createElement('style');
  8286. style.type='text/css';
  8287. style.textContent='\
  8288. #pv-float-bar-container {\
  8289. position: absolute;\
  8290. z-index:9999999998;\
  8291. padding: 5px;\
  8292. margin: 0;\
  8293. border: none;\
  8294. opacity: 0.6;\
  8295. line-height: 0;\
  8296. -webkit-transition: opacity 0.2s ease-in-out;\
  8297. transition: opacity 0.2s ease-in-out;\
  8298. display:none;\
  8299. }\
  8300. #pv-float-bar-container:hover {\
  8301. opacity: 1;\
  8302. }\
  8303. #pv-float-bar-container .pv-float-bar-button {\
  8304. vertical-align:middle;\
  8305. cursor: pointer;\
  8306. width: 18px;\
  8307. height: 18px;\
  8308. padding: 0;\
  8309. margin:0;\
  8310. border: none;\
  8311. display: inline-block;\
  8312. position: relative;\
  8313. box-shadow: 1px 0 3px 0px rgba(0,0,0,0.9);\
  8314. background: transparent center no-repeat;\
  8315. background-size:100% 100%;\
  8316. background-origin: content-box;\
  8317. -webkit-transition: margin-right 0.15s ease-in-out , width 0.15s ease-in-out , height 0.15s ease-in-out ;\
  8318. transition: margin-right 0.15s ease-in-out , width 0.15s ease-in-out , height 0.15s ease-in-out ;\
  8319. }\
  8320. #pv-float-bar-container .pv-float-bar-button:not(:last-child){\
  8321. margin-right: -14px;\
  8322. }\
  8323. #pv-float-bar-container .pv-float-bar-button:first-child {\
  8324. z-index: 4;\
  8325. }\
  8326. #pv-float-bar-container .pv-float-bar-button:nth-child(2) {\
  8327. z-index: 3;\
  8328. }\
  8329. #pv-float-bar-container .pv-float-bar-button:nth-child(3) {\
  8330. z-index: 2;\
  8331. }\
  8332. #pv-float-bar-container .pv-float-bar-button:last-child {\
  8333. z-index: 1;\
  8334. }\
  8335. #pv-float-bar-container:hover > .pv-float-bar-button {\
  8336. width: 24px;\
  8337. height: 24px;\
  8338. }\
  8339. #pv-float-bar-container:hover > .pv-float-bar-button:not(:last-child) {\
  8340. margin-right: 4px;\
  8341. }\
  8342. #pv-float-bar-container .pv-float-bar-button-actual {\
  8343. background-image:url("'+ prefs.icons.actual +'");\
  8344. }\
  8345. #pv-float-bar-container .pv-float-bar-button-search {\
  8346. background-image:url("'+ prefs.icons.search +'");\
  8347. }\
  8348. #pv-float-bar-container .pv-float-bar-button-gallery {\
  8349. background-image:url("'+ prefs.icons.gallery +'");\
  8350. }\
  8351. #pv-float-bar-container .pv-float-bar-button-current {\
  8352. background-image:url("'+ prefs.icons.current +'");\
  8353. }\
  8354. #pv-float-bar-container .pv-float-bar-button-magnifier {\
  8355. background-image:url("'+ prefs.icons.magnifier +'");\
  8356. }\
  8357. ';
  8358. document.head.appendChild(style);
  8359. },
  8360. start:function(data){
  8361.  
  8362. //读取中的图片,不显示浮动栏,调整读取图标的位置.
  8363. if(LoadingAnimC.all._find(function(item,index,array){
  8364. if(data.img==item.data.img){
  8365. return true;
  8366. };
  8367. }))return;
  8368.  
  8369.  
  8370. //被放大镜盯上的图片,不要显示浮动栏.
  8371. if(MagnifierC.all._find(function(item,index,array){
  8372. if(data.img==item.data.img){
  8373. return true;
  8374. };
  8375. }))return;
  8376.  
  8377. var self=this;
  8378. clearTimeout(this.hideTimer);
  8379.  
  8380. var imgOutHandler=function(e){
  8381. document.removeEventListener('mouseout',imgOutHandler,true);
  8382. clearTimeout(self.showTimer);
  8383. clearTimeout(self.hideTimer);
  8384. self.hideTimer=setTimeout(function(){
  8385. self.hide();
  8386. },prefs.floatBar.hideDelay);
  8387. };
  8388.  
  8389. clearTimeout(this.globarOutTimer);
  8390. this.globarOutTimer=setTimeout(function(){//稍微延时。错开由于css hover样式发生的out;
  8391. document.addEventListener('mouseout',imgOutHandler,true);
  8392. },150);
  8393.  
  8394. clearTimeout(this.showTimer);
  8395. this.showTimer=setTimeout(function(){
  8396. self.data=data;
  8397. self.show();
  8398. },prefs.floatBar.showDelay);
  8399. },
  8400. setButton:function(){
  8401. if(this.data.noActual){
  8402. this.buttons['actual'].style.display='none';
  8403. }else{
  8404. this.buttons['actual'].style.removeProperty('display');
  8405. }
  8406. if(this.data.type != "force" && this.data.img.nodeName == 'IMG'){
  8407. this.buttons['magnifier'].style.removeProperty('display');
  8408. }else{
  8409. this.buttons['magnifier'].style.display='none';
  8410. }
  8411. if (this.data.img.nodeName != 'IMG') {
  8412. //this.buttons['gallery'].style.display = 'none';
  8413. //this.buttons['current'].style.display = 'none';
  8414. } else {
  8415. //this.buttons['gallery'].style.removeProperty('display');
  8416. //this.buttons['current'].style.removeProperty('display');
  8417. }
  8418. },
  8419. setPosition:function(){
  8420. //如果图片被删除了,或者隐藏了。
  8421. if(this.data.img.offsetWidth==0){
  8422. return true;
  8423. };
  8424. var targetPosi=getContentClientRect(this.data.img);
  8425. var windowSize=getWindowSize();
  8426. var img=this.data.img;
  8427.  
  8428. var floatBarPosi=prefs.floatBar.position.toLowerCase().split(/\s+/);
  8429.  
  8430. var offsetX=prefs.floatBar.offset.x;
  8431. var offsetY=prefs.floatBar.offset.y;
  8432.  
  8433.  
  8434. var scrolled=getScrolled();
  8435.  
  8436. var fbs=this.floatBar.style;
  8437. var setPosition={
  8438. top:function(){
  8439. fbs.opacity="";
  8440. var top=targetPosi.top + scrolled.y;
  8441. if(targetPosi.top + offsetY < 0){//满足图标被遮住的条件.
  8442. top=scrolled.y;
  8443. offsetY=0;
  8444. }
  8445. top=top + offsetY;
  8446. if(targetPosi.height<=50)top-=10;
  8447. fbs.top=top + 'px';
  8448. },
  8449. right:function(){
  8450. fbs.opacity="";
  8451. var right=windowSize.w - targetPosi.right;
  8452. if(right < offsetX){
  8453. right= -scrolled.x;
  8454. offsetX=0;
  8455. }else{
  8456. right -=scrolled.x;
  8457. }
  8458. right=right - offsetX;
  8459. if(targetPosi.width<=50)right+=10;
  8460. fbs.right=right + 'px';
  8461. },
  8462. bottom:function(){
  8463. fbs.opacity="";
  8464. var bottom=windowSize.h - targetPosi.bottom;
  8465. if(bottom <= offsetY){
  8466. bottom=-scrolled.y;
  8467. offsetY=0;
  8468. }else{
  8469. bottom -= scrolled.y;
  8470. }
  8471. bottom=bottom - offsetY;
  8472. if(targetPosi.height<=50)bottom+=10;
  8473. fbs.bottom=bottom + 'px';
  8474. },
  8475. left:function(){
  8476. fbs.opacity="";
  8477. var left=targetPosi.left + scrolled.x;
  8478. if(targetPosi.left + offsetX < 0){
  8479. left=scrolled.x;
  8480. offsetX=0;
  8481. }
  8482. left=left + offsetX;
  8483. if(targetPosi.width<=50)left-=10;
  8484. fbs.left=left + 'px';
  8485. },
  8486. center:function(){
  8487. fbs.opacity="";
  8488. var left=targetPosi.left + scrolled.x + offsetX;
  8489. fbs.left=left + img.width/2 + 'px';
  8490. },
  8491. hide:function(){
  8492. fbs.opacity=0;
  8493. },
  8494. };
  8495.  
  8496. setPosition[floatBarPosi[0]]();
  8497. if(floatBarPosi.length>1)
  8498. setPosition[floatBarPosi[1]]();
  8499. },
  8500. show:function(){
  8501. if(this.setPosition())return;
  8502. this.shown=true;
  8503. this.setButton();
  8504. this.floatBar.style.display='block';
  8505. clearTimeout(this.hideTimer);
  8506. window.removeEventListener('scroll',this._scrollHandler,true);
  8507. window.addEventListener('scroll',this._scrollHandler,true);
  8508. },
  8509. hide:function(){
  8510. clearTimeout(this.showTimer);
  8511. this.shown=false;
  8512. this.floatBar.style.display='none';
  8513. window.removeEventListener('scroll',this._scrollHandler,true);
  8514. },
  8515. scrollHandler:function(){//更新坐标
  8516. clearTimeout(this.scrollUpdateTimer);
  8517. var self=this;
  8518. this.scrollUpdateTimer=setTimeout(function(){
  8519. self.setPosition();
  8520. },100);
  8521. },
  8522. open:function(e,buttonType){
  8523. var waitImgLoad = e && e.ctrlKey ? !prefs.waitImgLoad : prefs.waitImgLoad; //按住ctrl取反向值
  8524. var openInTopWindow = e && e.shiftKey ? !prefs.framesPicOpenInTopWindow : prefs.framesPicOpenInTopWindow; //按住shift取反向值
  8525. if (!waitImgLoad && buttonType == 'magnifier' && !envir.chrome) { //非chrome的background-image需要全部载入后才能显示出来
  8526. waitImgLoad = true;
  8527. };
  8528. new LoadingAnimC(this.data, buttonType, waitImgLoad, openInTopWindow);
  8529. },
  8530. update:function(img,src){
  8531. if(this.data.img==img && this.data.imgSrc!=src){
  8532. this.data.src=src;
  8533. this.data.noActual=false;
  8534. this.data.type="rule";
  8535. if(this.shown){
  8536. this.setButton();
  8537. }
  8538. }
  8539. }
  8540. };
  8541.  
  8542. /**
  8543. * 提取自 Mouseover Popup Image Viewer 脚本,用于 xhr 方式的获取
  8544. */
  8545. var xhrLoad = function() {
  8546. var _ = {};
  8547.  
  8548. var caches = {};
  8549. var handleError;
  8550.  
  8551. /**
  8552. * @param q 图片的选择器或函数
  8553. * @param c 图片说明的选择器或函数
  8554. */
  8555. function parsePage(url, q, c, post, cb) {
  8556. downloadPage(url, post, function(html) {
  8557. var iurl, iurls = [], cap, doc = createDoc(html);
  8558.  
  8559. if(typeof q == 'function') {
  8560. iurl = q(html, doc);
  8561. } else {
  8562. var inodes = findNodes(q, doc);
  8563. inodes.forEach(function(node) {
  8564. iurls.push(findFile(node, url));
  8565. });
  8566. iurl = iurls.shift();
  8567. }
  8568.  
  8569. if(typeof c == 'function') {
  8570. cap = c(html, doc);
  8571. } else {
  8572. var cnodes = findNodes(c, doc);
  8573. cap = cnodes.length ? findCaption(cnode[0]) : false;
  8574. }
  8575.  
  8576. // 缓存
  8577. if (iurl) {
  8578. caches[url] = {
  8579. iurl: iurl,
  8580. iurls: iurls,
  8581. cap: cap
  8582. };
  8583. }
  8584.  
  8585. cb(iurl, iurls, cap);
  8586. });
  8587. }
  8588.  
  8589. function downloadPage(url, post, cb) {
  8590. var opts = {
  8591. method: 'GET',
  8592. url: url,
  8593. onload: function(req) {
  8594. try {
  8595. if(req.status > 399) throw 'Server error: ' + req.status;
  8596. cb(req.responseText, req.finalUrl || url);
  8597. } catch(ex) {
  8598. handleError(ex);
  8599. }
  8600. },
  8601. onerror: handleError
  8602. };
  8603. if(post) {
  8604. opts.method = 'POST';
  8605. opts.data = post;
  8606. opts.headers = {'Content-Type':'application/x-www-form-urlencoded','Referer':url};
  8607. }
  8608.  
  8609. GM_xmlhttpRequest(opts);
  8610. }
  8611.  
  8612. function createDoc(text) {
  8613. var doc = document.implementation.createHTMLDocument('picViewerCE');
  8614. doc.documentElement.innerHTML = text;
  8615. return doc;
  8616. }
  8617.  
  8618. function findNodes(q, doc) {
  8619. var nodes = [],
  8620. node;
  8621. if (!Array.isArray(q)) q = [q];
  8622. for (var i = 0, len = q.length; i < len; i++) {
  8623. node = qs(q[i], doc);
  8624. if (node) {
  8625. nodes.push(node);
  8626. }
  8627. }
  8628. return nodes;
  8629. }
  8630.  
  8631. function findFile(n, url) {
  8632. var path = n.src || n.href;
  8633. return path ? path.trim() : false;
  8634. }
  8635.  
  8636. function findCaption(n) {
  8637. return n.getAttribute('content') || n.getAttribute('title') || n.textContent;
  8638. }
  8639.  
  8640. function qs(s, n) {
  8641. return n.querySelector(s);
  8642. }
  8643.  
  8644. _.load = function(opt) {
  8645. var info = caches[opt.url];
  8646. if (info) {
  8647. opt.cb(info.iurl, info.iruls, info.cap);
  8648. return;
  8649. }
  8650.  
  8651. handleError = opt.onerror || function() {};
  8652.  
  8653. parsePage(opt.url, opt.xhr.q, opt.xhr.c, opt.post, opt.cb);
  8654. };
  8655.  
  8656. return _;
  8657. }();
  8658.  
  8659.  
  8660. // ------------------- run -------------------------
  8661.  
  8662. var matchedRule,
  8663. URL=location.href,
  8664. floatBar;
  8665.  
  8666. function pretreatment(img){
  8667. if(img.nodeName != "IMG" || img.src)return;
  8668. if(img._lazyrias && img._lazyrias.srcset){
  8669. img.src=img._lazyrias.srcset[0];
  8670. }else if(img.srcset){
  8671. var srcs=img.srcset.split(","),minSize=0,newSrc;
  8672. srcs.forEach(srci=>{
  8673. let srcInfo=srci.trim().split(" "),curSize=parseInt(srcInfo[1]);
  8674. if(srcInfo[1] && (curSize<minSize || minSize==0)){
  8675. minSize=curSize;
  8676. newsrc=srcInfo[0];
  8677. }
  8678. });
  8679. if(newSrc)img.src=newSrc;
  8680. }else if(img.currentSrc){
  8681. img.src=img.currentSrc;
  8682. }
  8683. }
  8684.  
  8685. function findPic(img){
  8686. var imgPN=img;
  8687. var imgPA,imgPE=[];
  8688. while(imgPN=imgPN.parentElement){
  8689. if(imgPN.nodeName=='A'){
  8690. imgPA=imgPN;
  8691. break;
  8692. }
  8693. }
  8694. imgPN=img;
  8695. while(imgPN=imgPN.parentElement){
  8696. if(imgPN.nodeName=='BODY'){
  8697. break;
  8698. }else{
  8699. imgPE.push(imgPN);
  8700. }
  8701. }
  8702.  
  8703. var iPASrc=imgPA? imgPA.href : '';
  8704. //base64字符串过长导致正则匹配卡死浏览器
  8705. var base64Img=/^data:[^;]+;base64,/i.test(img.src);
  8706. var src, // 大图地址
  8707. srcs, // 备用的大图地址
  8708. type, // 类别
  8709. noActual = false, //没有原图
  8710. imgSrc = img.src||img.currentSrc, // img 节点的 src
  8711. xhr,
  8712. description; // 图片的注释
  8713. var imgCStyle=unsafeWindow.getComputedStyle(img);
  8714. var imgCS={
  8715. h: parseFloat(imgCStyle.height)||img.height,
  8716. w: parseFloat(imgCStyle.width)||img.width,
  8717. };
  8718. var imgAS={//实际尺寸。
  8719. h:img.naturalHeight||imgCS.h,
  8720. w:img.naturalWidth||imgCS.w,
  8721. };
  8722. if(!src && matchedRule && !base64Img){// 通过高级规则获取.
  8723. // 排除
  8724. try{
  8725. var newSrc=matchedRule.getImage(img,imgPA,imgPE);
  8726. if(newSrc && imgSrc!=newSrc) src=newSrc;
  8727. }catch(err){
  8728. throwErrorInfo(err);
  8729. }
  8730.  
  8731. if(src) {
  8732. if (Array.isArray(src)) {
  8733. srcs = src;
  8734. src = srcs.shift();
  8735. }
  8736.  
  8737. type = 'rule';
  8738. xhr = matchedRule.xhr;
  8739.  
  8740. if (matchedRule.lazyAttr) { // 由于采用了延迟加载技术,所以图片可能为 loading.gif
  8741. imgSrc = img.getAttribute(matchedRule.lazyAttr) || img.src;
  8742. }
  8743.  
  8744. if (matchedRule.description) {
  8745. var node = getElementMix(matchedRule.description, img);
  8746. if (node) {
  8747. description = node.getAttribute('title') || node.textContent;
  8748. }
  8749. }
  8750. }
  8751. }
  8752.  
  8753. if(!src && !base64Img){//遍历通配规则
  8754. tprules._find(function(rule,index,array){
  8755. try{
  8756. src=rule.call(img,imgPA);
  8757. if(src){
  8758. return true;
  8759. };
  8760. }catch(err){
  8761. throwErrorInfo(err);
  8762. };
  8763. });
  8764. if(src)type='tpRule';
  8765. }
  8766.  
  8767. if(!src && imgPA){//链接可能是一张图片...
  8768. if(/\.(?:jpg|jpeg|png|gif|bmp)(\?[^\?]*)?$/i.test(iPASrc) && iPASrc!=img.src){
  8769. src=iPASrc;
  8770. }
  8771. if(src)type='scale';
  8772. }
  8773.  
  8774. if(!src || src==imgSrc){//本图片是否被缩放.
  8775. noActual=true;
  8776. if(!(imgAS.w==imgCS.w && imgAS.h==imgCS.h)){//如果不是两者完全相等,那么被缩放了.
  8777. src=imgSrc;
  8778. type='scale';
  8779. }else{
  8780. src=imgSrc;
  8781. type='force';
  8782. }
  8783. }
  8784.  
  8785. if(!src)return;
  8786. if (imgAS.h < prefs.gallery.scaleSmallSize && imgAS.w < prefs.gallery.scaleSmallSize) {
  8787. type = 'scaleSmall';
  8788. }
  8789. if(img.dataset.lazySrc && (!imgSrc || /^\s*data:image/.test(imgSrc))){
  8790. imgSrc=img.dataset.lazySrc;
  8791. }
  8792.  
  8793. var ret = {
  8794. src: src, // 得到的src
  8795. srcs: srcs, // 多个 src,失败了会尝试下一个
  8796. type: type, // 通过哪种方式得到的
  8797. imgSrc: imgSrc, // 处理的图片的src
  8798. iPASrc: iPASrc, // 图片的第一个父a元素的链接地址
  8799. sizeH:imgAS.h,
  8800. sizeW:imgAS.w,
  8801. imgCS:imgCS,
  8802. imgAS:imgAS,
  8803.  
  8804. noActual:noActual,
  8805. xhr: xhr,
  8806. description: description || '',
  8807.  
  8808. img: img, // 处理的图片
  8809. imgPA: imgPA, // 图片的第一个父a元素
  8810. };
  8811. return ret;
  8812. }
  8813.  
  8814. function getMatchedRule() {
  8815. var rules=new MatchedRuleC();
  8816. return rules.rules.length>0?rules:false;
  8817. /*var rule = siteInfo._find(function(site, index, array) {
  8818. if (site.enabled != false && site.url && toRE(site.url).test(URL)) {
  8819. return true;
  8820. }
  8821. });
  8822.  
  8823. rule = rule ? rule[0] : false;
  8824.  
  8825. return rule;*/
  8826. }
  8827.  
  8828. function MatchedRuleC(){
  8829. this.init();
  8830. }
  8831.  
  8832. MatchedRuleC.prototype={
  8833. init:function(){
  8834. var self=this;
  8835. self.rules=[];
  8836. siteInfo.forEach(site=>{
  8837. if (site.enabled != false && (!site.url || toRE(site.url).test(URL))) {
  8838. if(site.url){
  8839. if(site.css){
  8840. var style = document.createElement('style');
  8841. style.type = 'text/css';
  8842. style.id = 'gm-picviewer-site-style';
  8843. style.textContent = site.css;
  8844. document.head.appendChild(style);
  8845. }
  8846. if(site.xhr){
  8847. self.xhr=site.xhr;
  8848. }
  8849. if(site.lazyAttr){
  8850. self.lazyAttr=site.lazyAttr;
  8851. }
  8852. if(site.description){
  8853. self.description=site.description;
  8854. }
  8855. if(site.clikToOpen){
  8856. self.clikToOpen=site.clikToOpen;
  8857. }
  8858. if(site.ext){
  8859. self.ext=site.ext;
  8860. }
  8861. }
  8862. self.rules.push(site);
  8863. }
  8864. });
  8865. },
  8866. replace:function(str, r, s){
  8867. var results=[],rt;
  8868. if(Array.isArray(s)){
  8869. s.forEach(_s=>{
  8870. rt=str.replace(r, _s);
  8871. if(rt && rt!=str)results.push(rt);
  8872. });
  8873. }else{
  8874. rt=str.replace(r, s);
  8875. if(rt && rt!=str)return str.replace(r, s);
  8876. }
  8877. return results;
  8878. },
  8879. getImage:function(img, a, p){
  8880. var newSrc,rule;
  8881. for(var i in this.rules){
  8882. rule=this.rules[i];
  8883. if(rule.src && !rule.src.test(img.src))continue;
  8884. if(rule.exclude && rule.exclude.test(img.src))continue;
  8885. if(rule.getImage){
  8886. debug(rule);
  8887. newSrc = rule.getImage.call(img, a, p);
  8888. }else{
  8889. if(rule.r){
  8890. if(Array.isArray(rule.r)){//r最多一层
  8891. for(var j in rule.r){
  8892. var _r=rule.r[j];
  8893. if(_r.test(img.src)){
  8894. if(Array.isArray(rule.s)){//s对上r最多两层
  8895. var _s=rule.s[j];
  8896. newSrc=this.replace(img.src, _r, _s);
  8897. }else{
  8898. newSrc=this.replace(img.src, _r, rule.s);
  8899. }
  8900. break;
  8901. }
  8902. }
  8903. }else{
  8904. if(rule.r.test(img.src)){
  8905. newSrc=this.replace(img.src, rule.r, rule.s);
  8906. }
  8907. }
  8908. }
  8909. }
  8910. if(newSrc && newSrc.length>0 && newSrc!=img.src){
  8911. debug(rule);
  8912. break;
  8913. }
  8914. }
  8915. return newSrc;
  8916. }
  8917. };
  8918.  
  8919. var isFrame=window!=window.parent;
  8920. var topWindowValid;
  8921. var frameSentData;
  8922. var frameSentSuccessData;
  8923. function handleMessage(e){
  8924. var data=e.data;
  8925. if( !data || !data.messageID || data.messageID != messageID )return;
  8926. var source=e.source;
  8927. if(typeof source=='undefined' || source!==window){
  8928. if(!isFrame){
  8929. var command=data.command;
  8930. switch(command){
  8931. case 'open':{
  8932. var img=document.createElement('img');
  8933. img.src=data.src;
  8934.  
  8935. imgReady(img,{
  8936. ready:function(){
  8937. LoadingAnimC.prototype.open.call({
  8938. img:img,
  8939. data:data.data,
  8940. buttonType:data.buttonType,
  8941. from:data.from,
  8942. });
  8943. },
  8944. });
  8945. }break;
  8946. case 'navigateToImg':{
  8947. var cusEvent=document.createEvent('CustomEvent');
  8948. cusEvent.initCustomEvent('pv-navigateToImg',false,false,data.exist);
  8949. document.dispatchEvent(cusEvent);
  8950. }break;
  8951. case 'topWindowValid':{
  8952. if(data.from)
  8953. window.postMessage({
  8954. messageID:messageID,
  8955. command:'topWindowValid_frame',
  8956. valid:document.body.nodeName!='FRAMESET',
  8957. to:data.from,
  8958. },'*');
  8959. }break;
  8960. };
  8961.  
  8962. }else{
  8963. var command=data.command;
  8964. switch(command){
  8965. case 'navigateToImg':{
  8966.  
  8967. if(!frameSentData.unique){
  8968. var unique=GalleryC.prototype.unique(frameSentData);
  8969. frameSentData=unique.data;
  8970. frameSentData.unique=true;
  8971. };
  8972. var targetImg=frameSentData[data.index].img;
  8973. var exist=(document.documentElement.contains(targetImg) && unsafeWindow.getComputedStyle(targetImg).display!='none');
  8974.  
  8975. if(exist){
  8976. if(gallery && gallery.shown){
  8977. gallery.minimize();
  8978. };
  8979. setTimeout(function(){
  8980. GalleryC.prototype.navigateToImg(targetImg);
  8981. flashEle(targetImg);
  8982. },0);
  8983. };
  8984. window.postMessage({
  8985. messageID:messageID,
  8986. command:'navigateToImg',
  8987. exist:exist,
  8988. to:data.from,
  8989. },'*');
  8990. }break;
  8991. case 'sendFail':{
  8992. frameSentData=frameSentSuccessData;
  8993. }break;
  8994. case 'topWindowValid_frame':{
  8995. var cusEvent=document.createEvent('CustomEvent');
  8996. cusEvent.initCustomEvent('pv-topWindowValid',false,false,data.valid);
  8997. document.dispatchEvent(cusEvent);
  8998. }break;
  8999. };
  9000. };
  9001.  
  9002. };
  9003. }
  9004.  
  9005. //页面脚本用来转发消息
  9006. //原因chrome的contentscript无法访问非自己外的别的窗口。都会返回undefined,自然也无法向其他的窗口发送信息,这里用pagescript做个中间代理
  9007. //通讯逻辑..A页面的contentscript发送到A页面的pagescript,pagescript转交给B页面的contentscript
  9008. var messageID='pv-0.5106795670312598';
  9009.  
  9010. function isunsafe(){
  9011. try {
  9012. return eval("false");
  9013. } catch (e) {
  9014. return true;
  9015. }
  9016. }
  9017. function addPageScript() {
  9018.  
  9019. if(isunsafe())return;
  9020. var pageScript=document.createElement('script');
  9021. pageScript.id = 'picviewer-page-script';
  9022.  
  9023. var pageScriptText=function(messageID){
  9024. var frameID=Math.random();
  9025. var frames={
  9026. top:window.top,
  9027. };
  9028.  
  9029. window.addEventListener('message',function(e){
  9030. var data=e.data;
  9031. if( !data || !data.messageID || data.messageID != messageID )return;//通信ID认证
  9032. var source=e.source;
  9033. if(source===window){//来自contentscript,发送出去,或者干嘛。
  9034. if(data.to){
  9035. data.from=frameID;
  9036. frames[data.to].postMessage(data,'*');
  9037. }else{
  9038. switch(data.command){
  9039. case 'getIframeObject':{
  9040. var frameWindow=frames[data.windowId];
  9041. var iframes=document.getElementsByTagName('iframe');
  9042. var iframe;
  9043. var targetIframe;
  9044. for(var i=iframes.length-1 ; i>=0 ; i--){
  9045. iframe=iframes[i];
  9046. if(iframe.contentWindow===frameWindow){
  9047. targetIframe=iframe;
  9048. break;
  9049. };
  9050. };
  9051. var cusEvent=document.createEvent('CustomEvent');
  9052. cusEvent.initCustomEvent('pv-getIframeObject',false,false,targetIframe);
  9053. document.dispatchEvent(cusEvent);
  9054. }break;
  9055. };
  9056. };
  9057.  
  9058. }else{//来自别的窗口的,contentscript可以直接接收,这里保存下来自的窗口的引用
  9059. frames[data.from]=source;
  9060. };
  9061. },true)
  9062. };
  9063.  
  9064. pageScript.textContent='(' + pageScriptText.toString() + ')('+ JSON.stringify(messageID) +')';
  9065. document.head.appendChild(pageScript);
  9066. }
  9067.  
  9068. function clikToOpen(data){
  9069. var preventDefault = matchedRule.clikToOpen.preventDefault;
  9070.  
  9071. function mouseout(){
  9072. document.removeEventListener('mouseout',mouseout,true);
  9073. document.removeEventListener('click',click,true);
  9074. if(data.imgPA && preventDefault){
  9075. data.imgPA.removeEventListener('click',clickA,false);
  9076. };
  9077. };
  9078.  
  9079. function click(e){
  9080. if(e.button!=0)return;
  9081. FloatBarC.prototype.open.call({
  9082. data:data,
  9083. },e,matchedRule.clikToOpen.type);
  9084. };
  9085.  
  9086. function clickA(e){//阻止a的默认行为
  9087. e.preventDefault();
  9088. };
  9089.  
  9090. document.addEventListener('click',click,true);
  9091.  
  9092. if(data.imgPA && preventDefault){
  9093. data.imgPA.addEventListener('click',clickA,false);
  9094. };
  9095.  
  9096. setTimeout(function(){//稍微延时。错开由于css hover样式发生的out;
  9097. document.addEventListener('mouseout',mouseout,true);
  9098. },100);
  9099.  
  9100. return function(){
  9101. mouseout();
  9102. };
  9103. }
  9104.  
  9105. var canclePreCTO,uniqueImgWin,centerInterval,removeUniqueWinTimer,globalFuncEnabled=false;
  9106. function checkGlobalKeydown(e){
  9107. return(!((!e.ctrlKey && prefs.floatBar.globalkeys.ctrl)||
  9108. (!e.altKey && prefs.floatBar.globalkeys.alt)||
  9109. (!e.shiftKey && prefs.floatBar.globalkeys.shift)||
  9110. (!e.metaKey && prefs.floatBar.globalkeys.command)||
  9111. (!prefs.floatBar.globalkeys.ctrl && !prefs.floatBar.globalkeys.alt && !prefs.floatBar.globalkeys.shift && !prefs.floatBar.globalkeys.command)));
  9112. }
  9113.  
  9114. function checkPreview(e){
  9115. return (prefs.floatBar.globalkeys.type == "hold" && checkGlobalKeydown(e)) ||
  9116. (prefs.floatBar.globalkeys.type == "press" && globalFuncEnabled);
  9117. }
  9118.  
  9119. //监听 mouseover
  9120. function globalMouseoverHandler(e){
  9121.  
  9122. if(galleryMode)return;//库模式全屏中......
  9123.  
  9124. var target = e.target;
  9125.  
  9126. if (!target || target.id=="pv-float-bar-container" ||
  9127. (target.className && target.className.indexOf &&
  9128. (target.className.indexOf('pv-') != -1 ||
  9129. target.classList.contains("ks-imagezoom-lens")))) {
  9130. return;
  9131. }
  9132.  
  9133. // 扩展模式,检查前面一个是否为 img
  9134. if (target.nodeName != 'IMG' && matchedRule && matchedRule.ext) {
  9135. var _type = typeof matchedRule.ext;
  9136. if (_type == 'string') {
  9137. switch (matchedRule.ext) {
  9138. case 'previous':
  9139. target = target.previousElementSibling || target;
  9140. break;
  9141. case 'next':
  9142. target = target.nextElementSibling || target;
  9143. break;
  9144. case 'previous-2':
  9145. target = (target.previousElementSibling &&
  9146. target.previousElementSibling.previousElementSibling) || target;
  9147. break;
  9148. }
  9149. } else if (_type == 'function') {
  9150. try {
  9151. target = matchedRule.ext(target);
  9152. } catch(ex) {
  9153. throwErrorInfo(ex);
  9154. }
  9155.  
  9156. if (!target) return;
  9157. }
  9158. }
  9159. var result, hasBg = node => {
  9160. if(node.nodeName == "HTML" || node.nodeName == "#document")
  9161. return false;
  9162. let nodeStyle = unsafeWindow.getComputedStyle(node);
  9163. return node && nodeStyle.backgroundImage && /url\(/i.test(nodeStyle.backgroundImage) && nodeStyle.backgroundImage.indexOf("about:blank") == -1 && parseFloat(nodeStyle.width) > prefs.floatBar.minSizeLimit.w && parseFloat(nodeStyle.height) > prefs.floatBar.minSizeLimit.h;
  9164. };
  9165. if (target.nodeName != 'IMG'){
  9166. if(target.nodeName == "AREA")target=target.parentNode;
  9167. var targetBg;
  9168. var bgReg=/.*url\(\s*["']?(.+?)["']?\s*\)/i;
  9169. if(prefs.floatBar.listenBg && hasBg(target)){
  9170. targetBg = unsafeWindow.getComputedStyle(target).backgroundImage.replace(bgReg,"$1");
  9171. var src=targetBg,nsrc=src,noActual=true,type="scale";
  9172. var img={src:src};
  9173. result = {
  9174. src: nsrc,
  9175. type: type,
  9176. imgSrc: src,
  9177. noActual:noActual,
  9178. img: target
  9179. };
  9180. }else if(target.childNodes.length<=2 && target.querySelectorAll("img").length==1){
  9181. target=target.querySelector("img");
  9182. }else if(target.parentNode){
  9183. if(target.parentNode.nodeName=='IMG'){
  9184. target=target.parentNode;
  9185. }else if(prefs.floatBar.listenBg && hasBg(target.parentNode)){
  9186. target=target.parentNode;
  9187. targetBg=unsafeWindow.getComputedStyle(target).backgroundImage.replace(bgReg,"$1");
  9188. var src=targetBg,nsrc=src,noActual=true,type="scale";
  9189. var img={src:src};
  9190. result = {
  9191. src: nsrc,
  9192. type: type,
  9193. imgSrc: src,
  9194. noActual:noActual,
  9195. img: target
  9196. };
  9197. }else if(unsafeWindow.getComputedStyle(target).position=="absolute" || target.nodeName == "MAP"){
  9198. var imgChildren=[],availableImgs = [];
  9199. [].forEach.call(target.parentNode.querySelectorAll('img'),function(img){
  9200. var imgStyle=unsafeWindow.getComputedStyle(img);
  9201. if(imgStyle.display != "none"){
  9202. imgChildren.push(img);
  9203. if(imgStyle.width > 200 || imgStyle.position != "absolute"){
  9204. availableImgs.push(img);
  9205. }
  9206. }
  9207. });
  9208. if(imgChildren.length == 1){
  9209. target=imgChildren[0];
  9210. }else if(availableImgs.length == 1){
  9211. target=availableImgs[0];
  9212. }else if(imgChildren.length == 0 && unsafeWindow.getComputedStyle(target.parentNode).position=="absolute"){
  9213. imgChildren=[];availableImgs = [];
  9214. [].forEach.call(target.parentNode.parentNode.querySelectorAll('img'),function(img){
  9215. var imgStyle=unsafeWindow.getComputedStyle(img);
  9216. if(imgStyle.display != "none"){
  9217. imgChildren.push(img);
  9218. if(imgStyle.width > 200 || imgStyle.position != "absolute"){
  9219. availableImgs.push(img);
  9220. }
  9221. }
  9222. });
  9223. if(imgChildren.length == 1){
  9224. target=imgChildren[0];
  9225. }else if(availableImgs.length == 1){
  9226. target=availableImgs[0];
  9227. }
  9228. }
  9229. }
  9230. }
  9231. if(result && !/^data:[^;]+;base64,/i.test(result.src)){
  9232. if(matchedRule && target.nodeName != 'IMG'){
  9233. let src=result.src,img={src:src},type,imgSrc=src;
  9234. try{
  9235. var newSrc=matchedRule.getImage(img);
  9236. if(newSrc && imgSrc!=newSrc) {
  9237. src=newSrc;
  9238. if (Array.isArray(src)) {
  9239. srcs = src;
  9240. src = srcs.shift();
  9241. }
  9242. type = 'rule';
  9243.  
  9244. if (matchedRule.description) {
  9245. var node = getElementMix(matchedRule.description, img);
  9246. if (node) {
  9247. description = node.getAttribute('title') || node.textContent;
  9248. }
  9249. }
  9250. result.src=src;
  9251. result.type=type;
  9252. result.noActual=false;
  9253. result.xhr=matchedRule.xhr;
  9254. result.description=description || '';
  9255. }
  9256. }catch(err){}
  9257. if(result.type!="rule"){
  9258. tprules._find(function(rule,index,array){
  9259. try{
  9260. src=rule.call(img);
  9261. if(src){
  9262. return true;
  9263. };
  9264. }catch(err){
  9265. }
  9266. });
  9267. if(src && src != imgSrc){
  9268. result.src=src;
  9269. result.type="tpRule";
  9270. result.noActual=false;
  9271. }
  9272. }
  9273. }
  9274. }
  9275. }
  9276. var checkUniqueImgWin=function(){
  9277. //metaKey altKey shiftKey ctrlKey
  9278. if(checkPreview(e)){
  9279. if(removeUniqueWinTimer)clearTimeout(removeUniqueWinTimer);
  9280. if(!uniqueImgWin || uniqueImgWin.removed){
  9281. var img = document.createElement('img');
  9282. uniqueImgWin = new ImgWindowC(img, result);
  9283. uniqueImgWin.imgWindow.classList.add("pv-pic-window-transition-all");
  9284. }
  9285. if(uniqueImgWin.src != result.src && (!result.srcs || !result.srcs.includes(uniqueImgWin.src))){
  9286. uniqueImgWin.changeData(result);
  9287. }
  9288. uniqueImgWin.blur(e);
  9289. if(!uniqueImgWin.loaded){
  9290. if(prefs.waitImgLoad){
  9291. uniqueImgWin.imgWindow.style.display = "none";
  9292. uniqueImgWin.imgWindow.style.opacity = 0;
  9293. }else{
  9294. uniqueImgWin.center(true,true);
  9295. if(centerInterval)clearInterval(centerInterval);
  9296. centerInterval=setInterval(function(){
  9297. if(!uniqueImgWin || uniqueImgWin.removed || uniqueImgWin.loaded)
  9298. clearInterval(centerInterval);
  9299. else{
  9300. uniqueImgWin.center(true,true);
  9301. }
  9302. },300);
  9303. }
  9304. }
  9305. uniqueImgWin.imgWindow.style.pointerEvents = "none";
  9306. return true;
  9307. }else {
  9308. if(uniqueImgWin && !uniqueImgWin.removed)
  9309. uniqueImgWin.imgWindow.style.pointerEvents = "auto";
  9310. return false;
  9311. }
  9312. };
  9313.  
  9314. if (!result && target.nodeName != 'IMG') {
  9315. if(target.nodeName == 'A' && /\.(jpg|png|jpeg|gif|webp)\b/.test(target.href)){
  9316. result = {
  9317. src: target.href,
  9318. type: "",
  9319. imgSrc: target.href,
  9320. noActual:true,
  9321. img: target
  9322. };
  9323. checkUniqueImgWin();
  9324. }else if(target.parentNode.nodeName == 'A' && /\.(jpg|png|jpeg|gif|webp)\b/.test(target.parentNode.href)){
  9325. result = {
  9326. src: target.parentNode.href,
  9327. type: "",
  9328. imgSrc: target.parentNode.href,
  9329. noActual:true,
  9330. img: target.parentNode
  9331. };
  9332. checkUniqueImgWin();
  9333. }
  9334. return;
  9335. }
  9336.  
  9337. if (!result) {
  9338. pretreatment(target)
  9339. result = findPic(target);
  9340. if(!(result.imgAS.w==result.imgCS.w && result.imgAS.h==result.imgCS.h)){//如果不是两者完全相等,那么被缩放了.
  9341. if(prefs.floatBar.sizeLimitOr){
  9342. if(result.imgCS.h <= prefs.floatBar.minSizeLimit.h && result.imgCS.w <= prefs.floatBar.minSizeLimit.w){//最小限定判断.
  9343. return;
  9344. }
  9345. }else{
  9346. if(result.imgCS.h <= prefs.floatBar.minSizeLimit.h || result.imgCS.w <= prefs.floatBar.minSizeLimit.w){//最小限定判断.
  9347. return;
  9348. }
  9349. }
  9350. }else{
  9351. if(prefs.floatBar.sizeLimitOr){
  9352. if(result.imgCS.w <= prefs.floatBar.forceShow.size.w && result.imgCS.h <= prefs.floatBar.forceShow.size.h){
  9353. return;
  9354. }
  9355. }else{
  9356. if(result.imgCS.w <= prefs.floatBar.forceShow.size.w || result.imgCS.h <= prefs.floatBar.forceShow.size.h){
  9357. return;
  9358. }
  9359. }
  9360. }
  9361. }
  9362.  
  9363. if(result){
  9364. debug(result);
  9365. if(!result.noActual){
  9366. if(!result.srcs){
  9367. result.srcs=[result.imgSrc];
  9368. }else{
  9369. if(result.imgSrc && result.srcs.join(" ").indexOf(result.imgSrc)==-1){
  9370. result.srcs.push(result.imgSrc);
  9371. }
  9372. }
  9373. }
  9374. if(!floatBar){
  9375. floatBar=new FloatBarC();
  9376. }
  9377. if(result.type=='rule' && matchedRule.clikToOpen && matchedRule.clikToOpen.enabled){
  9378. if(canclePreCTO){//取消上次的,防止一次点击打开多张图片
  9379. canclePreCTO();
  9380. }
  9381. canclePreCTO=clikToOpen(result);
  9382. }
  9383.  
  9384. if(!checkUniqueImgWin() && !e.altKey)
  9385. floatBar.start(result);//出现悬浮工具栏
  9386. };
  9387. }
  9388.  
  9389. function isKeyDownEffectiveTarget(target) {
  9390. var localName = target.localName;
  9391.  
  9392. // 确保光标不是定位在文字输入框或选择框
  9393. if (localName == 'textarea' || localName == 'input' || localName == 'select')
  9394. return false;
  9395.  
  9396. // 视频播放器
  9397. if (localName == 'object' || localName == 'embed')
  9398. return false;
  9399.  
  9400. // 百度贴吧回复输入的问题
  9401. if (target.getAttribute('contenteditable') == 'true')
  9402. return false;
  9403.  
  9404. return true;
  9405. }
  9406.  
  9407. function openGallery(){
  9408. if(!gallery){
  9409. gallery=new GalleryC();
  9410. gallery.data=[];
  9411. }
  9412. var allData=gallery.getAllValidImgs();
  9413. if(allData.length<1)return true;
  9414. gallery.data=allData;
  9415. gallery.load(gallery.data);
  9416. return gallery;
  9417. }
  9418.  
  9419. function keydown(event) {
  9420. var key = String.fromCharCode(event.keyCode).toLowerCase();
  9421. if(checkGlobalKeydown(event)){
  9422. if(key==prefs.floatBar.keys['gallery']){
  9423. openGallery();
  9424. event.stopPropagation();
  9425. event.preventDefault();
  9426. globalFuncEnabled = !globalFuncEnabled;
  9427. }else if(prefs.floatBar.globalkeys.type == "press"){
  9428. globalFuncEnabled = !globalFuncEnabled;
  9429. }
  9430. return true;
  9431. }else{
  9432. if (event.ctrlKey || event.shiftKey || event.altKey || event.metaKey)
  9433. return false;
  9434.  
  9435. if (floatBar && floatBar.shown && isKeyDownEffectiveTarget(event.target)) {
  9436. Object.keys(prefs.floatBar.keys).some(function(action) {
  9437. if (action == 'enable') return;
  9438. if (key == prefs.floatBar.keys[action]) {
  9439. floatBar.open(null, action);
  9440. event.stopPropagation();
  9441. event.preventDefault();
  9442. return true;
  9443. }
  9444. })
  9445. }
  9446. }
  9447. }
  9448.  
  9449. matchedRule = getMatchedRule();
  9450.  
  9451. window.addEventListener('message', handleMessage, true);
  9452.  
  9453. addPageScript();
  9454.  
  9455. document.addEventListener('mouseover', globalMouseoverHandler, true);
  9456.  
  9457. document.addEventListener('mouseout',e=>{
  9458. if(uniqueImgWin && !uniqueImgWin.removed){
  9459. if(checkPreview(e)){
  9460. if(removeUniqueWinTimer)clearTimeout(removeUniqueWinTimer);
  9461. removeUniqueWinTimer = setTimeout(()=>{uniqueImgWin.remove()},100);
  9462. }else{
  9463. //if(e.target.tagName!="IMG")return;
  9464. uniqueImgWin.imgWindow.style.pointerEvents = "auto";
  9465. uniqueImgWin.focus();
  9466. }
  9467. }
  9468. }, true);
  9469.  
  9470. GM_config.init({
  9471. id: 'pv-prefs',
  9472. title: GM_config.create('a', {
  9473. href: 'https://greasyfork.org/scripts/24204-picviewer-ce',
  9474. target: '_blank',
  9475. textContent: 'Picviewer CE+ '+i18n("config"),
  9476. title: i18n("openHomePage")
  9477. }),
  9478. isTabs: true,
  9479. skin: 'tab',
  9480. frameStyle: {
  9481. width: '480px',
  9482. zIndex:'2147483648',
  9483. },
  9484. css: [
  9485. "#pv-prefs input[type='text'] { width: 50px; } ",
  9486. "#pv-prefs input[type='number'] { width: 50px; } ",
  9487. "#pv-prefs .inline .config_var { margin-left: 6px; }",
  9488. "#pv-prefs label.size { width: 205px; }",
  9489. "#pv-prefs span.sep-x { margin-left: 0px !important; }",
  9490. "#pv-prefs label.sep-x { margin-right: 5px; }",
  9491. "#pv-prefs label.floatBar-key { margin-left: 20px; width: 100px; }",
  9492. "#pv-prefs input.color { width: 120px; }",
  9493. "#pv-prefs input.order { width: 250px; }",
  9494. "#pv-prefs .config_header>a { border-bottom: solid 2px; }",
  9495. "#pv-prefs .config_header>a:hover { color: #9f9f9f; }",,
  9496. ].join('\n'),
  9497. fields: {
  9498. // 浮动工具栏
  9499. 'floatBar.position': {
  9500. label: i18n("position"),
  9501. title: i18n("positionTips"),
  9502. type: 'select',
  9503. options: {
  9504. 'top left': i18n("topLeft"),
  9505. 'top right': i18n("topRight"),
  9506. 'bottom right': i18n("bottomRight"),
  9507. 'bottom left': i18n("bottomLeft"),
  9508. 'top center': i18n("topCenter"),
  9509. 'bottom center': i18n("bottomCenter"),
  9510. 'hide': i18n("hide")
  9511. },
  9512. "default": prefs.floatBar.position,
  9513. section: [i18n("floatBar")],
  9514. },
  9515. 'floatBar.showDelay': {
  9516. label: i18n("showDelay"),
  9517. type: 'int',
  9518. "default": prefs.floatBar.showDelay,
  9519. after: ' '+i18n("ms"),
  9520. },
  9521. 'floatBar.hideDelay': {
  9522. label: i18n("hideDelay"),
  9523. type: 'int',
  9524. className: 'hideDelay',
  9525. "default": prefs.floatBar.hideDelay,
  9526. after: ' '+i18n("ms")
  9527. },
  9528. 'floatBar.forceShow.size.w': {
  9529. label: i18n("forceShow"),
  9530. type: 'int',
  9531. className: 'size',
  9532. "default": prefs.floatBar.forceShow.size.w,
  9533. title: i18n("forceShowTip"),
  9534. line: 'start',
  9535. },
  9536. 'floatBar.forceShow.size.h': {
  9537. label: ' x ',
  9538. type: 'int',
  9539. className: 'sep-x',
  9540. after: ' '+i18n("px"),
  9541. "default": prefs.floatBar.forceShow.size.h,
  9542. line: 'end',
  9543. },
  9544. 'floatBar.minSizeLimit.w': {
  9545. label: i18n("minSizeLimit"),
  9546. type: 'int',
  9547. className: 'size',
  9548. "default": prefs.floatBar.minSizeLimit.w,
  9549. title: i18n("minSizeLimitTip"),
  9550. line: 'start',
  9551. },
  9552. 'floatBar.minSizeLimit.h': {
  9553. label: ' x ',
  9554. type: 'int',
  9555. className: 'sep-x',
  9556. after: ' '+i18n("px"),
  9557. "default": prefs.floatBar.minSizeLimit.h,
  9558. line: 'end',
  9559. },
  9560. 'floatBar.sizeLimitOr': {
  9561. label: i18n("sizeLimitOr"),
  9562. type: "checkbox",
  9563. "default": false
  9564. },
  9565. 'floatBar.butonOrder': {
  9566. label: i18n("butonOrder"),
  9567. type: 'text',
  9568. className: 'order',
  9569. "default": prefs.floatBar.butonOrder.join(', '),
  9570. },
  9571. 'floatBar.listenBg': {
  9572. label: i18n("listenBg"),
  9573. type: 'checkbox',
  9574. "default": prefs.floatBar.listenBg,
  9575. title: i18n("listenBgTip")
  9576. },
  9577. 'floatBar.globalkeys.ctrl': {
  9578. label: i18n("globalkeys"),
  9579. type: 'checkbox',
  9580. after: "CTRL"+" +",
  9581. "default": true,
  9582. line: 'start'
  9583. },
  9584. 'floatBar.globalkeys.alt': {
  9585. after: "ALT"+" +",
  9586. type: 'checkbox',
  9587. className: 'sep-x',
  9588. "default": false,
  9589. },
  9590. 'floatBar.globalkeys.shift': {
  9591. after: "SHIFT"+" +",
  9592. type: 'checkbox',
  9593. className: 'sep-x',
  9594. "default": false,
  9595. },
  9596. 'floatBar.globalkeys.command': {
  9597. after: "COMMAND",
  9598. type: 'checkbox',
  9599. className: 'sep-x',
  9600. "default": false,
  9601. line: 'end',
  9602. },
  9603. 'floatBar.globalkeys.type': {
  9604. label: i18n("globalkeysType"),
  9605. type: 'select',
  9606. options: {
  9607. 'press': i18n("globalkeysPress"),
  9608. 'hold': i18n("globalkeysHold")
  9609. },
  9610. "default": prefs.floatBar.globalkeys.type
  9611. },
  9612. // 按键
  9613. 'floatBar.keys.enable': {
  9614. label: i18n("keysEnable"),
  9615. type: 'checkbox',
  9616. "default": prefs.floatBar.keys.enable
  9617. },
  9618. 'floatBar.keys.actual': {
  9619. label: i18n("keysActual"),
  9620. type: 'text',
  9621. className: 'floatBar-key',
  9622. "default": prefs.floatBar.keys.actual,
  9623. title: i18n("keysActualTip")
  9624. },
  9625. /*'floatBar.keys.search': {
  9626. label: i18n("keysSearch"),
  9627. type: 'text',
  9628. className: 'floatBar-key',
  9629. "default": prefs.floatBar.keys.search,
  9630. title: i18n("keysSearchTip")
  9631. },*/
  9632. 'floatBar.keys.current': {
  9633. label: i18n("keysCurrent"),
  9634. type: 'text',
  9635. className: 'floatBar-key',
  9636. "default": prefs.floatBar.keys.current,
  9637. title: i18n("keysCurrentTip")
  9638. },
  9639. 'floatBar.keys.magnifier': {
  9640. label: i18n("keysMagnifier"),
  9641. type: 'text',
  9642. className: 'floatBar-key',
  9643. "default": prefs.floatBar.keys.magnifier,
  9644. title: i18n("keysMagnifierTip")
  9645. },
  9646. 'floatBar.keys.gallery': {
  9647. label: i18n("keysGallery"),
  9648. type: 'text',
  9649. className: 'floatBar-key',
  9650. "default": prefs.floatBar.keys.gallery,
  9651. title: i18n("keysGalleryTip")
  9652. },
  9653.  
  9654. // 放大镜
  9655. 'magnifier.radius': {
  9656. label: i18n("magnifierRadius"),
  9657. type: 'int',
  9658. "default": prefs.magnifier.radius,
  9659. section: [i18n("magnifier")],
  9660. after: ' '+i18n("px")
  9661. },
  9662. 'magnifier.wheelZoom.enabled': {
  9663. label: i18n("magnifierWheelZoomEnabled"),
  9664. type: 'checkbox',
  9665. "default": prefs.magnifier.wheelZoom.enabled,
  9666. },
  9667. 'magnifier.wheelZoom.range': {
  9668. label: i18n("magnifierWheelZoomRange"),
  9669. type: 'textarea',
  9670. "default": prefs.magnifier.wheelZoom.range.join(', '),
  9671. },
  9672.  
  9673. // 图库
  9674. 'gallery.defaultSizeLimit.w': {
  9675. label: i18n("defaultSizeLimit"),
  9676. type: 'int',
  9677. className: 'size',
  9678. section: [i18n("gallery")],
  9679. "default": prefs.gallery.defaultSizeLimit.w,
  9680. line: 'start',
  9681. },
  9682. 'gallery.defaultSizeLimit.h': {
  9683. label: ' x ',
  9684. type: 'int',
  9685. className: 'sep-x',
  9686. after: ' '+i18n("px"),
  9687. "default": prefs.gallery.defaultSizeLimit.h,
  9688. line: 'end',
  9689. },
  9690. 'gallery.fitToScreen': {
  9691. label: i18n("galleryFitToScreen"),
  9692. type: 'checkbox',
  9693. "default": prefs.gallery.fitToScreen,
  9694. title: i18n("galleryFitToScreenTip"),
  9695. line: 'start',
  9696. },
  9697. 'gallery.fitToScreenSmall': {
  9698. label: i18n("galleryFitToScreenSmall"),
  9699. type: 'checkbox',
  9700. "default": prefs.gallery.fitToScreenSmall,
  9701. line: 'end',
  9702. },
  9703. 'gallery.scrollEndToChange': {
  9704. label: i18n("galleryScrollEndToChange"),
  9705. type: 'checkbox',
  9706. "default": prefs.gallery.scrollEndToChange,
  9707. title: i18n("galleryScrollEndToChangeTip")
  9708. },
  9709. 'gallery.exportType': {
  9710. label: i18n("galleryExportType"),
  9711. type: 'select',
  9712. options: {
  9713. 'grid': i18n("grid"),
  9714. 'gridBig': i18n("gridBig"),
  9715. 'list': i18n("list")
  9716. },
  9717. "default": prefs.gallery.exportType,
  9718. },
  9719. 'gallery.loadMore': {
  9720. label: i18n("galleryAutoLoad"),
  9721. type: 'checkbox',
  9722. "default": prefs.gallery.loadMore
  9723. },
  9724. 'gallery.loadAll': {
  9725. label: i18n("galleryLoadAll"),
  9726. type: 'checkbox',
  9727. "default": prefs.gallery.loadAll,
  9728. title: i18n("galleryLoadAllTip")
  9729. },
  9730. 'gallery.downloadWithZip': {
  9731. label: i18n("galleryDownloadWithZip"),
  9732. type: 'checkbox',
  9733. "default": prefs.gallery.downloadWithZip
  9734. },
  9735. 'gallery.scaleSmallSize': {
  9736. label: i18n("galleryScaleSmallSize1"),
  9737. type: 'int',
  9738. "default": prefs.gallery.scaleSmallSize,
  9739. after: i18n("galleryScaleSmallSize2")
  9740. },
  9741. 'gallery.showSmallSize':{
  9742. label: i18n("galleryShowSmallSize"),
  9743. type: 'checkbox',
  9744. "default": prefs.gallery.showSmallSize
  9745. },
  9746. 'gallery.transition': {
  9747. label: i18n("galleryTransition"),
  9748. type: 'checkbox',
  9749. "default": prefs.gallery.transition
  9750. },
  9751. 'gallery.sidebarPosition': {
  9752. label: i18n("gallerySidebarPosition"),
  9753. type: 'select',
  9754. options: {
  9755. 'bottom': i18n("bottom"),
  9756. 'right': i18n("right"),
  9757. 'left': i18n("left"),
  9758. 'top': i18n("top")
  9759. },
  9760. "default": prefs.gallery.sidebarPosition,
  9761. line: 'start',
  9762. },
  9763. 'gallery.sidebarSize': {
  9764. label: i18n("gallerySidebarSize"),
  9765. type: 'int',
  9766. "default": prefs.gallery.sidebarSize,
  9767. title: i18n("gallerySidebarSizeTip"),
  9768. after: ' '+i18n("px"),
  9769. line: 'end',
  9770. },
  9771. 'gallery.max': {
  9772. label: i18n("galleryMax1"),
  9773. type: 'number',
  9774. "default": prefs.gallery.max,
  9775. after: i18n("galleryMax2")
  9776. },
  9777. 'gallery.autoZoom': {
  9778. label: i18n("galleryAutoZoom"),
  9779. type: 'checkbox',
  9780. "default": prefs.gallery.autoZoom,
  9781. title: i18n("galleryAutoZoomTip")
  9782. },
  9783. 'gallery.descriptionLength': {
  9784. label: i18n("galleryDescriptionLength1"),
  9785. type: 'int',
  9786. "default": prefs.gallery.descriptionLength,
  9787. after: i18n("galleryDescriptionLength2")
  9788. },
  9789. 'gallery.autoOpenSites': {
  9790. label: i18n("galleryAutoOpenSites"),
  9791. type: 'textarea',
  9792. "default": prefs.gallery.autoOpenSites
  9793. },
  9794. 'gallery.searchData': {
  9795. label: i18n("gallerySearchData"),
  9796. type: 'textarea',
  9797. "default": prefs.gallery.searchData
  9798. },
  9799. /*'gallery.editSite': {
  9800. label: i18n("galleryEditSite"),
  9801. type: 'select',
  9802. options: {
  9803. 'Pixlr': 'Pixlr',
  9804. 'Toolpic': 'Toolpic'
  9805. },
  9806. "default": prefs.gallery.editSite,
  9807. },*/
  9808.  
  9809. // 图片窗口
  9810. 'imgWindow.fitToScreen': {
  9811. label: i18n("imgWindowFitToScreen"),
  9812. type: 'checkbox',
  9813. "default": prefs.imgWindow.fitToScreen,
  9814. section: [i18n("imgWindow")],
  9815. title: i18n("imgWindowFitToScreenTip"),
  9816. },
  9817. 'imgWindow.suitLongImg': {
  9818. label: i18n("suitLongImg"),
  9819. type: 'checkbox',
  9820. "default": prefs.imgWindow.suitLongImg
  9821. },
  9822. 'imgWindow.close.defaultTool': {
  9823. label: i18n("imgWindowDefaultTool"),
  9824. type: 'select',
  9825. options: {
  9826. 'hand': i18n("hand"),
  9827. 'rotate': i18n("rotate"),
  9828. 'zoom': i18n("zoom"),
  9829. },
  9830. "default": prefs.imgWindow.close.defaultTool,
  9831. },
  9832. 'imgWindow.close.escKey': {
  9833. label: i18n("imgWindowEscKey"),
  9834. type: 'checkbox',
  9835. "default": prefs.imgWindow.close.escKey,
  9836. line: 'start',
  9837. },
  9838. 'imgWindow.close.dblClickImgWindow': {
  9839. label: i18n("imgWindowDblClickImgWindow"),
  9840. type: 'checkbox',
  9841. "default": prefs.imgWindow.close.dblClickImgWindow,
  9842. },
  9843. 'imgWindow.close.clickOutside': {
  9844. label: i18n("imgWindowClickOutside"),
  9845. type: 'select',
  9846. options: {
  9847. '': i18n("none"),
  9848. 'click': i18n("click"),
  9849. 'dblclick': i18n("dblclick"),
  9850. },
  9851. "default": prefs.imgWindow.close.clickOutside,
  9852. title: i18n("imgWindowClickOutsideTip"),
  9853. line: 'end',
  9854. },
  9855. 'imgWindow.overlayer.shown': {
  9856. label: i18n("imgWindowOverlayerShown"),
  9857. type: 'checkbox',
  9858. "default": prefs.imgWindow.overlayer.shown,
  9859. line: 'start',
  9860. },
  9861. 'imgWindow.overlayer.color': {
  9862. label: i18n("imgWindowOverlayerColor"),
  9863. type: 'text',
  9864. className: 'color',
  9865. "default": prefs.imgWindow.overlayer.color,
  9866. line: 'end'
  9867. },
  9868. 'imgWindow.shiftRotateStep': {
  9869. label: i18n("imgWindowShiftRotateStep1"),
  9870. type: 'int',
  9871. "default": prefs.imgWindow.shiftRotateStep,
  9872. after: i18n("imgWindowShiftRotateStep2")
  9873. },
  9874. 'imgWindow.zoom.mouseWheelZoom': {
  9875. label: i18n("imgWindowMouseWheelZoom"),
  9876. type: 'checkbox',
  9877. "default": prefs.imgWindow.zoom.mouseWheelZoom,
  9878. },
  9879. 'imgWindow.zoom.range': {
  9880. label: i18n("imgWindowZoomRange"),
  9881. type: 'textarea',
  9882. "default": prefs.imgWindow.zoom.range.join(', '),
  9883. title: i18n("imgWindowZoomRangeTip"),
  9884. attr: {
  9885. "spellcheck": "false"
  9886. }
  9887. },
  9888.  
  9889. // 其它
  9890. 'waitImgLoad': {
  9891. label: i18n("waitImgLoad"),
  9892. type: 'checkbox',
  9893. "default": prefs.waitImgLoad,
  9894. section: [i18n("others")],
  9895. title: i18n("waitImgLoadTip")
  9896. },
  9897. 'debug': {
  9898. label: i18n("debug"),
  9899. type: 'checkbox',
  9900. "default": prefs.debug
  9901. },
  9902. /*'firstEngine': {
  9903. label: i18n("firstEngine"),
  9904. type: 'select',
  9905. options: {
  9906. "Tineye":"Tineye",
  9907. "Google":"Google",
  9908. "Baidu":"Baidu"
  9909. },
  9910. "default": prefs.firstEngine,
  9911. },*/
  9912. },
  9913. events: {
  9914. open: function(doc, win, frame) {
  9915. let saveBtn=doc.querySelector("#"+this.id+"_saveBtn");
  9916. let closeBtn=doc.querySelector("#"+this.id+"_closeBtn");
  9917. let resetLink=doc.querySelector("#"+this.id+"_resetLink");
  9918. saveBtn.textContent=i18n("saveBtn");
  9919. saveBtn.title=i18n("saveBtnTips");
  9920. closeBtn.textContent=i18n("closeBtn");
  9921. closeBtn.title=i18n("closeBtnTips");
  9922. resetLink.textContent=i18n("resetLink");
  9923. resetLink.title=i18n("resetLinkTips");
  9924. let searchData=doc.getElementById(this.id+"_field_gallery.searchData");
  9925. if(searchData && searchData.value==""){
  9926. searchData.value=defaultSearchData;
  9927. }
  9928. },
  9929. save: function() {
  9930. loadPrefs();
  9931. }
  9932. }
  9933. });
  9934.  
  9935.  
  9936. GM_registerMenuCommand('Picviewer CE+ '+i18n("config"), openPrefs);
  9937. GM_registerMenuCommand(i18n("openGallery"), openGallery);
  9938.  
  9939. loadPrefs();
  9940.  
  9941. if(prefs.gallery.autoOpenSites){
  9942. var sitesArr=prefs.gallery.autoOpenSites.split("\n");
  9943. for(var s=0;s<sitesArr.length;s++){
  9944. var siteReg=sitesArr[s].trim();
  9945. var autoViewMore=siteReg[0]=="@";
  9946. if(autoViewMore)siteReg=siteReg.substr(1);
  9947. if(new RegExp(siteReg).test(location.href)){
  9948. setTimeout(function(){
  9949. var gallery = openGallery();
  9950. if(autoViewMore)gallery.maximizeSidebar();
  9951. },2000);
  9952. break;
  9953. }
  9954. }
  9955. }
  9956.  
  9957. // 注册按键
  9958. if (prefs.floatBar.keys.enable) {
  9959. document.addEventListener('keydown', keydown, false);
  9960. }
  9961.  
  9962. function openPrefs() {
  9963. GM_config.open();
  9964. }
  9965.  
  9966. function loadPrefs() {
  9967. // 根据 GM_config 的 key 载入设置到 prefs
  9968. Object.keys(GM_config.fields).forEach(function(keyStr) {
  9969. var keys = keyStr.split('.');
  9970. var lastKey = keys.pop();
  9971.  
  9972. var lastPref = keys.reduce(function(previousValue, curKey) {
  9973. return previousValue[curKey];
  9974. }, prefs) || prefs;
  9975.  
  9976. var value = GM_config.get(keyStr);
  9977. if (typeof value != 'undefined') {
  9978. // 特殊的
  9979. if (keyStr == 'magnifier.wheelZoom.range' || keyStr == 'imgWindow.zoom.range') {
  9980. lastPref[lastKey] = value.split(/[,,]\s*/).map(function(s) { return parseFloat(s)});
  9981. } else if(keyStr == 'floatBar.butonOrder') {
  9982. lastPref[lastKey] = value.replace(/^\s*|\s*$/g,"").split(/\s*[,,]\s*/);
  9983. } else {
  9984. lastPref[lastKey] = value;
  9985. }
  9986. }
  9987. });
  9988.  
  9989. debug = prefs.debug ? console.debug.bind(console) : function() {};
  9990. }
  9991.  
  9992. };
  9993.  
  9994. function init2(){
  9995. init(topObject,window,document,arrayFn,envir,storage,unsafeWindow);
  9996. };
  9997.  
  9998.  
  9999. //大致检测运行环境
  10000. var envir={
  10001. ie:typeof document.documentMode == 'number',
  10002. firefox:typeof XPCNativeWrapper == 'function',
  10003. opera:!!window.opera,
  10004. chrome:!!window.chrome,
  10005. };
  10006.  
  10007. //ie的话,不支持 < ie9的版本
  10008. if(envir.ie && document.documentMode < 9){
  10009. return;
  10010. };
  10011.  
  10012.  
  10013. var arrayFn=(function(){
  10014. //Array的某些方法对所有的类数组都有效,比如HTMLCollection,NodeList,DOMStringList.....
  10015.  
  10016. //添加一个当函数返回true时,返回[array[index],index],并且跳出循环的方法
  10017. //类似做到 for 循环,在满足条件的时候直接break跳出的效果。
  10018. if(typeof Array.prototype['_find']!='function'){
  10019. Object.defineProperty(Array.prototype,'_find',{
  10020. value:function(callback , thisArg){
  10021. if (this == null){
  10022. throw new TypeError( "this is null or not defined" );
  10023. };
  10024.  
  10025. if(typeof callback != 'function') {
  10026. throw new TypeError( callback + " is not a function" );
  10027. };
  10028.  
  10029. var i = 0,
  10030. l = this.length,
  10031. value,
  10032. hasOwnProperty=Object.prototype.hasOwnProperty;
  10033.  
  10034.  
  10035. while(i<l){
  10036. if(hasOwnProperty.call(this,i)){
  10037. value = this[i];
  10038. if(callback.call( thisArg, value, i, this )===true){
  10039. return [value,i,this];
  10040. };
  10041. };
  10042. i++;
  10043. };
  10044. },
  10045. writable:true,
  10046. enumerable:false,//与原生方法一样不可枚举,维护网页和谐。。。
  10047. configurable:true,
  10048. });
  10049. };
  10050.  
  10051. var arrayProto=Array.prototype;
  10052. return {
  10053. _find:arrayProto._find,
  10054. slice:arrayProto.slice,
  10055. forEach:arrayProto.forEach,
  10056. some:arrayProto.some,
  10057. every:arrayProto.every,
  10058. map:arrayProto.map,
  10059. filter:arrayProto.filter,
  10060. indexOf:arrayProto.indexOf,
  10061. lastIndexOf:arrayProto.lastIndexOf,
  10062. };
  10063.  
  10064. })();
  10065.  
  10066.  
  10067. if(typeof GM_getValue=='undefined' && GM && GM.getValue){
  10068. var GM_getValue=GM.getValue;
  10069. var GM_setValue=GM.setValue;
  10070. }
  10071. var storage={
  10072. supportGM: typeof GM_getValue=='function' && typeof GM_getValue('a','b')!='undefined',//chrome的gm函数式空函数
  10073. mxAppStorage:(function(){//傲游扩展储存接口
  10074. try{
  10075. return window.external.mxGetRuntime().storage;
  10076. }catch(e){
  10077. };
  10078. })(),
  10079. operaUJSStorage:(function(){//opera userjs全局存储接口
  10080. try{
  10081. return window.opera.scriptStorage;
  10082. }catch(e){
  10083. };
  10084. })(),
  10085. setItem:function(key,value){
  10086. if(this.operaUJSStorage){
  10087. this.operaUJSStorage.setItem(key,value);
  10088. }else if(this.mxAppStorage){
  10089. this.mxAppStorage.setConfig(key,value);
  10090. }else if(this.supportGM){
  10091. GM_setValue(key,value);
  10092. }else if(window.localStorage){
  10093. window.localStorage.setItem(key,value);
  10094. };
  10095. },
  10096. getItem:function(key){
  10097. var value;
  10098. if(this.operaUJSStorage){
  10099. value=this.operaUJSStorage.getItem(key);
  10100. }else if(this.mxAppStorage){
  10101. value=this.mxAppStorage.getConfig(key);
  10102. }else if(this.supportGM){
  10103. value=GM_getValue(key);
  10104. }else if(window.localStorage){
  10105. value=window.localStorage.getItem(key);
  10106. };
  10107. return value;
  10108. },
  10109. };
  10110.  
  10111. function getUrl(url, callback, onError){
  10112. GM_xmlhttpRequest({
  10113. method: 'GET',
  10114. url: url,
  10115. onload: callback,
  10116. onerror: onError
  10117. });
  10118. }
  10119.  
  10120. function setSearchState(words,imgCon){
  10121. if(words)console.info(words);
  10122. var searchState = (imgCon?imgCon:document).querySelector('.pv-pic-search-state');
  10123. if(searchState)searchState.innerHTML=words;
  10124. }
  10125.  
  10126. var searchSort=["Tineye","Google","Baidu"];
  10127. function sortSearch(){
  10128. for(var i=0;i<searchSort.length;i++){
  10129. if(searchSort[i]==prefs.firstEngine){
  10130. searchSort.splice(i,1);
  10131. break;
  10132. }
  10133. }
  10134. searchSort.unshift(prefs.firstEngine);
  10135. }
  10136.  
  10137. function searchImgByImg(imgSrc, imgCon, callBack, onError, noneResult, searchFrom){
  10138. let srcs=[];
  10139. var searchBaidu=function(){
  10140. setSearchState(i18n("beginSearchImg","百度"),imgCon);
  10141. getUrl("http://image.baidu.com/n/same?queryImageUrl="+encodeURIComponent(imgSrc)+"&isguessword=1&rn=30&fr=pc&pn=0&sort=size", function(d){
  10142. let baiduJson;
  10143. try{
  10144. baiduJson=JSON.parse(d.responseText);
  10145. }catch(e){
  10146. setSearchState(i18n("findNoPic"),imgCon);
  10147. setTimeout(function(){
  10148. setSearchState("",imgCon);
  10149. },2000);
  10150. searchNext();
  10151. return;
  10152. }
  10153. if(baiduJson.data[0]){
  10154. srcs=[];
  10155. for(let imgData of baiduJson.data){
  10156. if(srcs.length>2)break;
  10157. srcs.push(imgData.objURL);
  10158. }
  10159. setSearchState(i18n("findOverBeginLoad",["百度",srcs.length]),imgCon);
  10160. callBackFun(srcs);
  10161. }else{
  10162. searchNext();
  10163. return;
  10164. }
  10165. }, onError);
  10166. };
  10167. var searchGoogle=function(){
  10168. setSearchState(i18n("beginSearchImg","Google"),imgCon);
  10169. getUrl("https://www.google.com/searchbyimage?safe=off&image_url="+encodeURIComponent(imgSrc), function(d){
  10170. let googleHtml=document.implementation.createHTMLDocument('');
  10171. googleHtml.documentElement.innerHTML = d.responseText;
  10172. let sizeUrl=googleHtml.querySelector("div.card-section>div>div>span.gl>a");
  10173. if(sizeUrl){
  10174. getUrl("https://www.google.com"+sizeUrl.getAttribute("href"), function(d){
  10175. googleHtml.documentElement.innerHTML = d.responseText;
  10176. let imgs=googleHtml.querySelectorAll("div.rg_meta");
  10177. if(imgs.length==0){searchNext();return;}
  10178. srcs=[];
  10179. for(var i=0;i<imgs.length;i++){
  10180. if(srcs.length>2)break;
  10181. let jsonData=JSON.parse(imgs[i].innerHTML);
  10182. srcs.push(jsonData.ou);
  10183. }
  10184. setSearchState(i18n("findOverBeginLoad",["Google",srcs.length]),imgCon);
  10185. callBackFun(srcs);
  10186. }, onError);
  10187. }else{
  10188. searchNext();
  10189. }
  10190. }, onError);
  10191. };
  10192. var searchTineye=function(){
  10193. setSearchState(i18n("beginSearchImg","Tineye"),imgCon);
  10194. getUrl("https://www.tineye.com/search?url="+encodeURIComponent(imgSrc)+"&sort=size", function(d){
  10195. let tineyeHtml=document.implementation.createHTMLDocument('');
  10196. tineyeHtml.documentElement.innerHTML = d.responseText;
  10197. let searchImg=tineyeHtml.querySelectorAll(".match-details>div.match:first-of-type>p.image-link:first-of-type>a");
  10198. if(searchImg.length>0){
  10199. srcs=[];
  10200. for(var i=0;i<searchImg.length;i++){
  10201. if(srcs.length>2)break;
  10202. srcs.push(searchImg[i].href);
  10203. }
  10204. setSearchState(i18n("findOverBeginLoad",["Tineye",srcs.length]),imgCon);
  10205. callBackFun(srcs);
  10206. }else{
  10207. searchNext();
  10208. }
  10209. }, onError);
  10210. };
  10211. var searchNext=function(){
  10212. searchFrom++;
  10213. if(searchFrom<=searchSort.length)switchSearch();
  10214. else{
  10215. if(noneResult)noneResult();
  10216. setSearchState(i18n("findNoPic"),imgCon);
  10217. setTimeout(function(){
  10218. setSearchState("",imgCon);
  10219. },2000);
  10220. }
  10221. };
  10222. var callBackFun=function(srcs){
  10223. callBack(srcs, searchFrom);
  10224. };
  10225. if(!searchFrom)searchFrom=1;
  10226. var switchSearch=function(){
  10227. switch(searchSort[searchFrom-1]){
  10228. case "Baidu":
  10229. searchBaidu();
  10230. break;
  10231. case "Google":
  10232. searchGoogle();
  10233. break;
  10234. case "Tineye":
  10235. searchTineye();
  10236. break;
  10237. default:
  10238. searchTineye();
  10239. break;
  10240. }
  10241. };
  10242. switchSearch();
  10243. }
  10244.  
  10245. init2();
  10246.  
  10247. })(this,window,document,(typeof unsafeWindow=='undefined'? window : unsafeWindow));