Sleazy Fork is available in English.

MissAv批量备份收藏视频

从当前missav页面获取图片文件和视频信息,并合并结果后提供下载生成的网页文件

  1. // ==UserScript==
  2. // @name MissAv批量备份收藏视频
  3. // @name:zh-CN MissAv批量备份收藏视频
  4. // @namespace https://github.com/ChinaGodMan/UserScripts
  5. // @version 1.2.3.73
  6. // @description 从当前missav页面获取图片文件和视频信息,并合并结果后提供下载生成的网页文件
  7. // @description:zh-CN 从当前missav页面获取图片文件和视频信息,并合并结果后提供下载生成的网页文件
  8. // @license MIT
  9. // @author 人民的勤务员 <china.qinwuyuan@gmail.com> & ChatGPT
  10. // @match https://missav.com/*
  11. // @grant GM_setValue
  12. // @grant GM_getValue
  13. // @grant GM_xmlhttpRequest
  14. // @grant GM_download
  15. // @grant GM_openInTab
  16. // @icon 
  17. // @iconbak https://pic.616pic.com/ys_bnew_img/00/35/79/Gv93yQh7v6.jpg
  18. // @require https://cdnjs.cloudflare.com/ajax/libs/jszip/3.7.1/jszip.min.js
  19. // @require https://update.greasyfork.org/scripts/498124/1396763/video.js
  20. // @require https://update.greasyfork.org/scripts/498149/1395619/%E4%BF%A1%E6%81%AF%E6%9F%A5%E7%9C%8B%E5%99%A8.js
  21. // @supportURL https://github.com/ChinaGodMan/UserScripts/issues
  22. // @homepageURL https://github.com/ChinaGodMan/UserScripts
  23. // ==/UserScript==
  24.  
  25. (function () {
  26. 'use strict'
  27.  
  28. var controlButton = createButton('备份', '10px', '10px')
  29. var buttonA = createButton('备份当前', '70px', '10px')
  30. var buttonB = createButton('备份片单', '130px', '10px')
  31. var buttonC = createButton('设置选项', '190px', '10px')
  32. var webdavbutton = createButton('WebDav', '250px', '10px')
  33. // 设置按钮的背景颜色和样式
  34. controlButton.style.backgroundColor = 'blue' // 控制按钮改为蓝色背景
  35. buttonA.style.backgroundColor = 'green'
  36. buttonB.style.backgroundColor = 'blue'
  37. buttonC.style.backgroundColor = 'red'
  38. webdavbutton.style.backgroundColor = 'blue'
  39. // 隐藏初始的三个按钮
  40. buttonA.style.display = 'none'
  41. buttonB.style.display = 'none'
  42. buttonC.style.display = 'none'
  43. webdavbutton.style.display = 'none'
  44. // 添加按钮到页面
  45. document.body.appendChild(controlButton)
  46. document.body.appendChild(buttonA)
  47. document.body.appendChild(buttonB)
  48. document.body.appendChild(buttonC)
  49. document.body.appendChild(webdavbutton)
  50. // 控制按钮的点击事件
  51. controlButton.addEventListener('click', function () {
  52. if (buttonA.style.display === 'none') {
  53. // 显示三个按钮
  54. buttonA.style.display = 'block'
  55. buttonB.style.display = 'block'
  56. buttonC.style.display = 'block'
  57. webdavbutton.style.display = 'block'
  58. controlButton.innerHTML = '隐藏'
  59. } else {
  60. // 隐藏三个按钮
  61. buttonA.style.display = 'none'
  62. buttonB.style.display = 'none'
  63. buttonC.style.display = 'none'
  64. webdavbutton.style.display = 'none'
  65. controlButton.innerHTML = '备份'
  66. }
  67. })
  68. webdavbutton.addEventListener('click', function () {
  69. // 点击按钮时执行的操作
  70. WebDAVManager.listFilesAndFolders(webdavfold)
  71.  
  72.  
  73. })
  74.  
  75. // 按钮A的点击事件
  76. buttonA.addEventListener('click', function () {
  77. resetGlobalVariables()
  78. singleFileDownload = true
  79. window.showLogContainer()
  80. var currentDate = new Date()
  81. var currentTime = currentDate.getFullYear() + '-' + (currentDate.getMonth() + 1) + '-' + currentDate.getDate() + '_' + currentDate.getHours() + '-' + currentDate.getMinutes() + '-' + currentDate.getSeconds()
  82. if (useDefaultTitle) {
  83. name = document.querySelector('meta[name="twitter:title"]').content
  84. } else {
  85. const twitterTitleContent = document.querySelector('meta[name="twitter:title"]').content
  86. name = prompt('请输入自定义名称:', twitterTitleContent)
  87. if (name === null) {
  88. name = twitterTitleContent
  89. }
  90. }
  91. inurl = window.location.href
  92. const defaultPages = getTotalPagesd()
  93.  
  94. const totalPages = setTotalPage(defaultPages)
  95. allpages = totalPages
  96. //const delay = settime();
  97. if (totalPages) {
  98. start(totalPages)
  99. }
  100.  
  101. })
  102.  
  103. // 按钮B的点击事件
  104. buttonB.addEventListener('click', function () {
  105. // 点击按钮时执行的操作
  106. resetGlobalVariables()
  107. fetchJsonData()
  108.  
  109.  
  110. })
  111.  
  112. // 按钮C的点击事件
  113. buttonC.addEventListener('click', function () {
  114. createSettingsUI()
  115.  
  116. // 这里可以添加按钮C点击后的具体操作,比如打开链接或执行其他动作
  117. })
  118.  
  119. // 创建按钮的辅助函数
  120. function createButton(text, top, left) {
  121. var button = document.createElement('button')
  122. button.innerHTML = text
  123. button.style.position = 'fixed'
  124. button.style.bottom = top
  125. button.style.right = left
  126. button.style.zIndex = '1000'
  127. button.style.padding = '10px'
  128. button.style.border = 'none'
  129. button.style.cursor = 'pointer'
  130. button.style.color = '#fff'
  131. button.style.fontSize = '14px'
  132. button.style.fontWeight = 'bold'
  133. button.style.textAlign = 'center'
  134. button.style.width = '100px' // 调整按钮宽度
  135. return button
  136. }
  137. // 全局变量
  138. var allResults = [] // 存储所有的结果数据
  139. var zip = new JSZip() // 创建一个压缩文件实例
  140. var allzip = new JSZip() // 另一个可能的压缩文件实例
  141. var imgFolder = zip.folder('img') // 在 zip 中创建一个名为 "img" 的文件夹,用于存储图片文件
  142. var allimgFolder = allzip.folder('img') // 在 allzip 中创建一个名为 "img" 的文件夹,可能用于另一个压缩文件的图片存储
  143. var ALLfiledown = false // 标识是否所有文件已下载完毕的布尔变量
  144. var videos = [] // 存储视频文件或相关信息的数组
  145. var finalData = [] // 存储最终处理数据的数组
  146. var inurl = '' // 当前下载地址的变量
  147. var pendingRequests = 0 // 当前待处理的请求数量
  148. var delayTime // 延迟时间,以毫秒为单位,用于控制异步操作的时间间隔
  149. var currentPage = 1 // 当前处理的页数,可能用于分页处理或其他进度跟踪
  150. var currentUrlIndex = 0 // 当前处理的 urls 数组中的索引位置
  151. var name = '' // 当前下载的名称
  152. var urls = [] // 存储需要处理的网址数组
  153. var a = -1 // 循环中的计数或索引,初始值为 -1
  154. var allZipContents = [] // 存储所有压缩文件内容的数组
  155. var singleFileDownload = false // 标识是否为单个文件下载模式的布尔变量
  156. var names = [] // 存储下载名称列表的数组
  157. var allpages = 0 // 存储总页数或其他页面处理相关信息的变量
  158. var modalContainer = null // 存储模态窗口容器的全局变量,用于显示下载进度或其他信息
  159. var shouldReplace = false // 控制是否在下载大图时进行替换操作的布尔变量
  160. var temporaryData = []
  161. var saveJson = false
  162. var useDefaultTitle = true
  163. var pageCount = true
  164. var saveVideoInfo = false
  165. var saveImage = false
  166. var downloadLog = {}
  167. var errorLogs = {}
  168. var downloadLogFileA = false // 这里设置为 true 时载日志
  169. var webdavfold = 'missavsave'
  170. //var webdavfold="1111";
  171. var savetowebdav = false
  172. var webdavUrl = ''
  173. var webdavUsername = ''
  174. var webdavPassword = ''
  175. var deleteSelected = false
  176. ini()//读取配置
  177. function resetGlobalVariables() {
  178. zip = new JSZip() // 重置为一个新的 JSZip 实例,用于创建新的压缩文件
  179. allzip = new JSZip() // 可能是另一个新的 JSZip 实例,用于其他用途的压缩文件
  180. if (saveImage) {
  181. imgFolder = zip.folder('img') // 在 zip 中创建一个名为 "img" 的文件夹,用于存储图片文件
  182. allimgFolder = allzip.folder('img') // 在 allzip 中创建一个名为 "img" 的文件夹,可能用于另一个压缩文件的图片存储
  183. }
  184. downloadLog = {}
  185. errorLogs = {}
  186. ALLfiledown = false // 重置为 false,表示所有文件未下载完毕
  187. videos = [] // 清空存储视频文件或相关信息的数组
  188. finalData = [] // 清空存储最终处理数据的数组
  189. inurl = '' // 重置当前下载地址为空字符串
  190. pendingRequests = 0 // 重置待处理的请求数量为 0
  191.  
  192. currentPage = 1 // 重置当前处理的页数为 1
  193. currentUrlIndex = 0 // 重置当前处理的 urls 数组索引为 0
  194. name = '' // 重置当前下载的名称为空字符串
  195. urls = [] // 清空存储需要处理的网址数组
  196. a = -1 // 重置循环中的计数或索引为 -1
  197. allZipContents = [] // 清空存储所有压缩文件内容的数组
  198. singleFileDownload = false // 重置为 false,表示不是单个文件下载模式
  199. names = [] // 清空存储下载名称列表的数组
  200. allpages = 0 // 重置存储总页数或其他页面处理相关信息的变量为 0
  201. temporaryData = []
  202. }
  203.  
  204.  
  205. async function processUrls() {
  206. //delayTime = 20;
  207. let completedTasks = 0 // 计数已完成的任务数量
  208.  
  209. for (const url of urls) {
  210. a = a + 1 // 每次循环递增 a
  211. inurl = url
  212. console.log('正在处理网址:', url, names[a])
  213. window.addToLog('处理:' + url + names[a], 'info')
  214. name = names[a]
  215.  
  216. try {
  217. const totalPages = await getTotalPages(url) // 等待 getTotalPages 返回结果
  218. console.log('Total pages for', url, ':', totalPages) // 显示总页数
  219. window.addToLog(name + ' 总页数:' + url + totalPages, 'info')
  220. allpages = totalPages
  221. start(totalPages) // 启动处理流程
  222.  
  223. // 等待当前页面的请求完成
  224. while (pendingRequests > 0) {
  225. await new Promise(resolve => setTimeout(resolve, 100)) // 每隔 100 毫秒检查一次是否所有请求都已完成
  226. }
  227.  
  228. completedTasks++ // 标记当前任务已完成
  229. } catch (error) {
  230. console.error('Error processing URL:', url, error) // 处理错误信息
  231. allpages = 1
  232. start(1) // 启动处理流程
  233. while (pendingRequests > 0) {
  234. await new Promise(resolve => setTimeout(resolve, 100)) // 每隔 100 毫秒检查一次是否所有请求都已完成
  235. }
  236. completedTasks++ // 标记当前任务已完成
  237. }
  238. }
  239.  
  240. // 如果所有任务都已完成且 urls 数组不为空,则调用下载函数
  241. if (completedTasks === urls.length && urls.length !== 0) {
  242. downloadAllZips()
  243. }
  244. }
  245.  
  246.  
  247.  
  248.  
  249. function getAllCookies() {
  250. return document.cookie
  251. }
  252.  
  253. // 获取指定 JSON 数据的函数
  254.  
  255.  
  256.  
  257.  
  258. function fetchJsonData() {
  259. const cookies = getAllCookies()
  260. //alert(cookies);
  261. console.log('Current page cookies:', cookies)
  262.  
  263. // 构建 API URL
  264. const apiUrl = 'https://missav.com/api/playlists/dfe-057'
  265.  
  266. // 发送带有 cookies 的请求
  267. GM_xmlhttpRequest({
  268. method: 'GET',
  269. url: apiUrl,
  270. headers: {
  271. 'Cookie': cookies
  272. },
  273. onload: function (response) {
  274. if (response.status === 200) {
  275. try {
  276. const jsonResponse = JSON.parse(response.responseText)
  277. if (jsonResponse && Array.isArray(jsonResponse.data)) {
  278. createReportUI(jsonResponse.data, 500) // 假设每页显示 10 个项目
  279.  
  280. // 调用 processUrls 函数处理 URLs
  281. } else {
  282. console.error('JSON 格式无效')
  283. showModal('JSON 格式无效', 2000)
  284. }
  285. } catch (error) {
  286. console.error('Error parsing JSON:', error)
  287. showModal('解析错误' + error, 2000)
  288. }
  289. } else {
  290. console.error('Request failed with status:', response.status)
  291. showModal('解析错误', 2000)
  292. }
  293. },
  294. onerror: function (error) {
  295. showModal('解析错误' + error, 2000)
  296. }
  297. })
  298. }
  299.  
  300. function processUrl(url) {
  301. // 检查是否包含 `page=` 参数
  302. var pageIndex = url.indexOf('page=')
  303.  
  304. if (pageIndex !== -1) {
  305. // 找到 `page=` 参数并删除它及其后的所有内容
  306. var baseUrl = url.substring(0, pageIndex + 5) // +5 to include `page=`
  307. return baseUrl
  308. } else {
  309. // 检查是否已有其他参数
  310. if (url.includes('?')) {
  311. // 有其他参数,添加 `&page=`
  312. return url + '&page='
  313. } else {
  314. // 没有其他参数,添加 `?page=`
  315. return url + '?page='
  316. }
  317. }
  318. }
  319. function settime() {
  320. // 让用户输入延时时间
  321. delayTime = prompt('请输入每页请求的延时时间(毫秒):', '1000')
  322.  
  323. // 检查用户是否取消输入
  324. if (delayTime === null) {
  325. alert('输入取消')
  326. return
  327. }
  328.  
  329. delayTime = parseInt(delayTime)
  330.  
  331. // 检查输入的延时时间是否有效
  332. if (isNaN(delayTime) || delayTime <= 0) {
  333. alert('请输入有效的延时时间(正整数)!')
  334. return
  335. }
  336.  
  337. // 返回有效的延时时间
  338. return delayTime
  339. }
  340.  
  341.  
  342. function getTotalPagesd() { // 获取总页数
  343. var totalPagesElement = document.querySelector('#price-currency')
  344. var totalPagesText = totalPagesElement ? totalPagesElement.innerText : ''
  345. var totalPages = parseInt(totalPagesText.replace('/', '').trim(), 10)
  346.  
  347. // 如果获取总页数失败,则返回 1
  348. if (isNaN(totalPages) || totalPages <= 0) {
  349. totalPages = 1
  350. }
  351.  
  352. return totalPages
  353. }
  354. function getTotalPages(url) {
  355. return new Promise((resolve, reject) => {
  356. // 如果没有提供 URL,则使用当前页面的 URL
  357. if (!url) {
  358. url = window.location.href
  359. }
  360.  
  361. // 发起 GM_xmlhttpRequest 请求获取页面内容
  362. GM_xmlhttpRequest({
  363. method: 'GET',
  364. url: url,
  365. headers: { 'Cookie': document.cookie },
  366. onload: function (response) {
  367. // 处理响应
  368. if (response.status === 200) {
  369. const parser = new DOMParser()
  370. const doc = parser.parseFromString(response.responseText, 'text/html')
  371. const totalPagesElement = doc.querySelector('#price-currency') // 替换为实际选择器
  372. if (totalPagesElement) {
  373. const totalPagesText = totalPagesElement.innerText
  374. const totalPages = parseInt(totalPagesText.replace('/', '').trim(), 10)
  375. resolve(totalPages) // 成功时返回总页数
  376. } else {
  377. window.addToLog('页面中没有找到总页数,默认为1页', 'warning')
  378. reject('Total pages element not found') // 页面中没有找到总页数元素
  379. }
  380. } else {
  381. window.addToLog('请求失败', 'warning')
  382. reject(`Request failed with status ${response.status}`) // 请求失败
  383. }
  384. },
  385. onerror: function () {
  386. window.addToLog('请求出错', 'warning')
  387. reject('Request failed') // 请求出错
  388.  
  389. }
  390. })
  391. })
  392. }
  393.  
  394.  
  395. // 设置总页数
  396. function setTotalPage(defaultPages) {
  397. if (!pageCount) {
  398. return defaultPages
  399. }
  400.  
  401. const inputPages = parseInt(prompt(`当前 ${name} 总页数为 ${defaultPages}。请输入你想要抓取的页数(不输入抓取全部):`, defaultPages), 10)
  402. if (isNaN(inputPages) || inputPages <= 0) {
  403. return defaultPages
  404. }
  405. return inputPages
  406. }
  407. // 开始处理页面抓取
  408. function start(totalPages, callback) {
  409. const pages = Array.from({ length: totalPages }, (_, i) => i + 1)
  410. pendingRequests = pages.length
  411. fetchPage(pages.shift(), pages, callback)
  412. }
  413.  
  414. // 点击按钮时执行操作
  415.  
  416.  
  417.  
  418. // 异步获取页面内容
  419. function fetchPage(pageNum, pages, callback) {
  420. const pageUrl = `${processUrl(inurl)}${pageNum}`
  421. console.log(`正在获取第 ${pageNum} 页的内容...`)
  422. //showModal(`正在获取${name} 第 ${pageNum} / ${allpages}页 `);
  423. if (a !== -1) {
  424. showModal(`${a + 1}/${names.length} 正在获取 ${name} ${pageNum} / ${allpages} 页`)
  425. } else {
  426. showModal(`正在获取 ${name} ${pageNum} / ${allpages} 页`)
  427. }
  428. GM_xmlhttpRequest({
  429. method: 'GET',
  430. url: pageUrl,
  431. headers: { 'Cookie': document.cookie },
  432. onload: function (response) {
  433. if (response.status === 200) {
  434. processPageContent(response.responseText, pageNum, pages, callback)
  435. } else {
  436. pendingRequests--
  437. checkIfComplete(callback)
  438. if (pages.length > 0) {
  439. setTimeout(() => fetchPage(pages.shift(), pages, callback), delayTime)
  440. }
  441. }
  442. }
  443. })
  444. }
  445.  
  446.  
  447. //获取视频信息
  448. function extractInformation(htmlContent) {
  449. let data = {} // 创建一个对象来存储提取的数据
  450. let xhr = new XMLHttpRequest()
  451. xhr.open('GET', htmlContent, false) // 同步方式打开请求
  452. xhr.send()
  453.  
  454. // 创建一个虚拟的 <div> 元素来加载 HTML 内容
  455. let tempDiv = document.createElement('div')
  456. tempDiv.innerHTML = xhr.responseText
  457.  
  458. // 获取所有包含信息的父元素列表
  459. let parentElements = tempDiv.querySelectorAll('div.space-y-2 > div')
  460.  
  461. if (parentElements.length > 0) {
  462. let allInfo = {} // 初始化一个空对象来存储所有信息
  463.  
  464. // 遍历每个包含信息的 <div> 元素
  465. parentElements.forEach(div => {
  466. let span = div.querySelector('span') // 获取第一个 <span> 元素
  467.  
  468. if (span) {
  469. let category = span.textContent.trim() // 获取主分类名称
  470.  
  471. if (!allInfo[category]) {
  472. allInfo[category] = [] // 初始化一个空数组来存储该分类下的所有信息
  473. }
  474.  
  475. // 查找所有的 <a> 元素和 <time> 元素
  476. div.querySelectorAll('a, time').forEach(element => {
  477. let info = {}
  478.  
  479. // 判断元素是否是 <a> 元素
  480. if (element.tagName === 'A') {
  481. info['name'] = element.textContent.trim() // 获取名称
  482. info['link'] = element.href.trim() // 获取链接
  483. } else if (element.tagName === 'TIME') {
  484. info = element.textContent.trim() // 如果是 <time> 元素,则直接保存其文本内容
  485. }
  486.  
  487. // 添加信息对象到相应的主分类数组中
  488. allInfo[category].push(info)
  489. })
  490.  
  491. // 如果没有找到 <a> 元素,则尝试获取 <span> 标签内的文本内容
  492. if (div.querySelectorAll('a').length === 0) {
  493. let spanText = div.querySelector('span.font-medium')
  494. if (spanText) {
  495. let info = spanText.textContent.trim()
  496. allInfo[category].push(info)
  497. }
  498. }
  499. }
  500. })
  501.  
  502. // 提取 class="mb-1 text-secondary break-all line-clamp-2" 的内容
  503. let descriptionElement = tempDiv.querySelector('.mb-1.text-secondary.break-all.line-clamp-2')
  504. let descriptionContent = descriptionElement ? descriptionElement.textContent.trim() : ''
  505. allInfo['简介'] = descriptionContent
  506.  
  507. // 将所有信息存储到 data 对象中
  508. data['videosinfo'] = allInfo
  509.  
  510. // 查找包含 x-cloak 和 x-show="currentTab === 'magnets'" 的第二个元素
  511. let secondElement = tempDiv.querySelector('div[x-cloak][x-show="currentTab === \'magnets\'"]')
  512.  
  513. if (secondElement) {
  514. let linksAndInfo = []
  515.  
  516. // 遍历第二个元素内的 <a> 元素
  517. secondElement.querySelectorAll('a[rel="nofollow"]').forEach(a => {
  518. let linkInfo = {
  519. name: a.textContent.trim(),
  520. link: a.href.trim()
  521. }
  522.  
  523. // 查找相邻的 <td> 元素,获取大小和日期信息
  524. let sizeTd = a.closest('td').nextElementSibling
  525. if (sizeTd && sizeTd.classList.contains('font-mono')) {
  526. linkInfo['size'] = sizeTd.textContent.trim() // 获取大小信息
  527. }
  528.  
  529. let dateTd = sizeTd ? sizeTd.nextElementSibling : null
  530. if (dateTd && dateTd.classList.contains('hidden')) {
  531. linkInfo['date'] = dateTd.textContent.trim() // 获取日期信息
  532. }
  533. let nextSibling = a.nextElementSibling
  534.  
  535. // 循环处理所有紧邻的<span>元素
  536. while (nextSibling && nextSibling.tagName === 'SPAN') {
  537. let spanText = nextSibling.textContent.trim()
  538. linkInfo['name'] += ' ' + spanText // 将<span>元素的文本内容追加到name中
  539.  
  540. nextSibling = nextSibling.nextElementSibling // 继续查找下一个兄弟元素
  541. }
  542. linksAndInfo.push(linkInfo)
  543. })
  544.  
  545. // 将第二个元素的链接和信息添加到 data 中
  546. data['secondElementLinksInfo'] = linksAndInfo
  547. } else {
  548. console.error('未找到包含 x-cloak 和 x-show="currentTab === \'magnets\'" 的第二个元素。')
  549. }
  550.  
  551. return data // 返回结构化的数据
  552.  
  553. } else {
  554. console.error('未找到匹配的父元素 div.space-y-2')
  555. return null // 如果未找到匹配的父元素,返回 null
  556. }
  557. } ///大
  558.  
  559.  
  560.  
  561.  
  562. // 使用XMLHttpRequest获取页面内容
  563. function fetchPageforinfo(url) {
  564. let xhr = new XMLHttpRequest()
  565.  
  566. xhr.onreadystatechange = function () {
  567. if (xhr.readyState === XMLHttpRequest.DONE) {
  568. if (xhr.status === 200) {
  569. extractInformation(xhr.responseText) // 将获取的页面内容传递给提取信息的函数
  570. } else {
  571. console.error('请求失败:' + xhr.status)
  572. }
  573. }
  574. }
  575.  
  576. xhr.open('GET', url, true)
  577. xhr.send()
  578. }
  579.  
  580.  
  581. // 处理获取到的页面内容
  582. function processPageContent(htmlContent, pageNum, pages, callback) {
  583. const parser = new DOMParser()
  584. const doc = parser.parseFromString(htmlContent, 'text/html')
  585. const divElements = doc.querySelectorAll('div.relative.aspect-w-16.aspect-h-9.rounded.overflow-hidden.shadow-lg')
  586. const logEntry = {
  587. url: `${processUrl(inurl)}${pageNum}`,
  588. elementsFetched: divElements.length
  589. }
  590.  
  591. // 如果当前名称的日志组不存在,则创建一个新数组
  592. if (!downloadLog[name]) {
  593. downloadLog[name] = []
  594. }
  595.  
  596. // 将日志条目添加到日志数组中
  597. downloadLog[name].push(logEntry)
  598. if (divElements.length === 0) {
  599. const logEntry = {
  600. url: `${processUrl(inurl)}${pageNum}`,
  601. elementsFetched: 0, // 这里可以根据实际需求设置其他信息
  602. errorMessage: `获取第 ${pageNum} 页失败。`
  603. }
  604. if (!errorLogs[name]) {
  605. errorLogs[name] = []
  606. }
  607. errorLogs[name].push(logEntry)
  608. console.log(`获取第 ${pageNum} 页失败。`)
  609. window.addToLog(`${name}${processUrl(inurl)}${pageNum}+获取失败 数量:` + divElements.length, 'error')
  610. }
  611.  
  612.  
  613.  
  614.  
  615. divElements.forEach(div => {
  616. var imgUrl = div.querySelector('img').getAttribute('data-src')
  617.  
  618.  
  619. if (shouldReplace) {
  620. imgUrl = imgUrl.replace('cover-t.jpg', 'cover-n.jpg')
  621. }
  622. const video = {
  623. fileName: div.querySelector('a').getAttribute('alt'),
  624. imgUrl: imgUrl,
  625. videoUrl: div.querySelector('video').getAttribute('data-src'),
  626. markContent: Array.from(div.querySelectorAll('span')).map(mark => mark.textContent).join(' '),
  627. altText: div.querySelector('img').getAttribute('alt'),
  628. jumpUrl: div.querySelector('a').getAttribute('href')
  629.  
  630. }
  631.  
  632. if (saveVideoInfo) {
  633. video.info = extractInformation(video.jumpUrl)
  634. //showBanner(`正在获取 ${video.fileName} 信息`);
  635. window.addToLog(`正在获取 ${video.fileName} 信息`, 'info')
  636. console.log()
  637. }
  638.  
  639.  
  640.  
  641. if (video.imgUrl && video.altText) {
  642. videos.push(video)
  643. if (saveImage) {
  644. window.addToLog('保存' + video.imgUrl, 'info')
  645. pendingRequests++
  646. GM_xmlhttpRequest({
  647. method: 'GET',
  648. url: video.imgUrl,
  649. responseType: 'blob',
  650. onload: function (response) {
  651. if (response.status === 200) {
  652. if (saveImage) {
  653. if (singleFileDownload) {
  654. console.log('这是单个文件下载')
  655. imgFolder.file(`${video.fileName}.jpg`, response.response, { binary: true })
  656. } else {
  657. console.log('这是批量文件下载')
  658. allimgFolder.file(`${video.fileName}.jpg`, response.response, { binary: true })
  659. }
  660. }
  661.  
  662. pendingRequests--
  663. checkIfComplete(callback)
  664. } else {
  665. pendingRequests--
  666. checkIfComplete(callback)
  667. }
  668. }
  669. })
  670. }
  671. } else {
  672. pendingRequests--
  673. checkIfComplete(callback)
  674. }
  675. })
  676.  
  677. showModal(`获取第 ${pageNum} 页的内容完成,等待 ${delayTime} 毫秒加载第 ${pageNum + 1} 页。`)
  678. pendingRequests--
  679. checkIfComplete(callback)
  680. if (pages.length > 0) {
  681. setTimeout(() => fetchPage(pages.shift(), pages, callback), delayTime)
  682. } else {
  683.  
  684. }
  685. }
  686.  
  687. closeModal()
  688.  
  689. function downloadLogFile() {
  690.  
  691.  
  692. if (!downloadLogFileA) {
  693. console.log('日志下载已被跳过')
  694. return
  695. }
  696. if (Object.keys(errorLogs).length === 0) {
  697. // 如果错误日志为空,直接下载正常日志文件
  698. const logBlob = new Blob([JSON.stringify(downloadLog, null, 4)], { type: 'application/json' })
  699. const logUrl = URL.createObjectURL(logBlob)
  700. const logLink = document.createElement('a')
  701. logLink.href = logUrl
  702. logLink.download = 'download_log.json'
  703. logLink.click()
  704. URL.revokeObjectURL(logUrl)
  705. } else {
  706. // 创建一个JSZip实例
  707. const zip = new JSZip()
  708.  
  709. // 添加正常日志文件到压缩包
  710. const logBlob = new Blob([JSON.stringify(downloadLog, null, 4)], { type: 'application/json' })
  711. zip.file('download_log.json', logBlob)
  712.  
  713. // 添加错误日志文件到压缩包
  714. const errorLogBlob = new Blob([JSON.stringify(errorLogs, null, 4)], { type: 'application/json' })
  715. zip.file('error_log.json', errorLogBlob)
  716.  
  717. // 生成压缩包并触发下载
  718. zip.generateAsync({ type: 'blob' }).then(function (content) {
  719. const zipUrl = URL.createObjectURL(content)
  720. const link = document.createElement('a')
  721. link.href = zipUrl
  722. link.download = 'logs.zip'
  723. link.click()
  724. URL.revokeObjectURL(zipUrl)
  725. })
  726. }
  727. }
  728.  
  729.  
  730.  
  731. function sanitizeFileName(name) {
  732. return name.replace(/[\\/:*?"<>|]/g, '_')
  733. }
  734.  
  735. function checkIfComplete(callback) {
  736. if (pendingRequests === 0) {
  737.  
  738. const additionalInfo = {
  739. timestamp: new Date().toISOString(),
  740. inurl: inurl
  741.  
  742. }
  743. if (singleFileDownload) {
  744. showModal('获取完毕,正在生成单个文件...')
  745.  
  746. finalData = {
  747. info: additionalInfo,
  748. video: videos
  749. }
  750. if (saveJson) {
  751. zip.file('data.json', JSON.stringify(finalData, null, 4))
  752. }
  753. if (savetowebdav) {
  754.  
  755. WebDAVManager.uploadFile(webdavfold, `${sanitizeFileName(name)}.json`, JSON.stringify(finalData, null, 4))
  756. }
  757.  
  758.  
  759.  
  760. const jsonIndexContent = generateJsonIndexContent(finalData)
  761. const numFiles = Object.keys(zip.files).length // 获取压缩包中文件的数量
  762.  
  763. if (numFiles === 0) {
  764. const htmlContent = jsonIndexContent // 替换为实际的HTML内容
  765. const htmlBlob = new Blob([htmlContent], { type: 'text/html' })
  766. const htmlUrl = URL.createObjectURL(htmlBlob)
  767.  
  768. const a = document.createElement('a')
  769. a.href = htmlUrl
  770. a.download = `${sanitizeFileName(name)}.html`
  771. a.click()
  772. closeModal()
  773. downloadLogFile()
  774.  
  775. if (callback) callback()
  776. } else {
  777. zip.file(`${sanitizeFileName(name)}.html`, jsonIndexContent)
  778.  
  779. // 生成并下载单个文件
  780. zip.generateAsync({ type: 'blob' }, function updateCallback(metadata) {
  781. const progress = metadata.percent.toFixed(2)
  782. showModal(`压缩进度: ${progress}%`)
  783. }).then(content => {
  784. const zipUrl = URL.createObjectURL(content)
  785. const a = document.createElement('a')
  786. a.href = zipUrl
  787. a.download = `${name}.zip`
  788. a.click()
  789. URL.revokeObjectURL(zipUrl)
  790. closeModal()
  791. downloadLogFile()
  792. if (callback) callback()
  793. })
  794. }
  795.  
  796.  
  797. } else {
  798. finalData = {
  799. info: additionalInfo,
  800. video: videos
  801. }
  802. if (saveJson) {
  803. allzip.file(`${sanitizeFileName(name)}.json`, JSON.stringify(finalData, null, 4))
  804. }
  805. if (savetowebdav) {
  806. WebDAVManager.uploadFile(webdavfold, `${sanitizeFileName(name)}.json`, JSON.stringify(finalData, null, 4))
  807. }
  808.  
  809. const jsonIndexContent = generateJsonIndexContent(finalData);
  810. allzip.file(`${sanitizeFileName(name)}.html`, jsonIndexContent)
  811. finalData = []
  812.  
  813. videos = []
  814. if (callback) callback()
  815. }
  816.  
  817. }
  818. }
  819. function downloadAllZips() {
  820. if (singleFileDownload === false) {
  821. showModal('获取完毕,正在生成压缩文件...')
  822.  
  823. const numFiles = Object.keys(allzip.files).length // 获取压缩包中文件的数量
  824.  
  825. if (numFiles === 1) {
  826. // 如果压缩包中只有一个文件,直接处理该文件
  827. const fileName = Object.keys(allzip.files)[0] // 获取唯一的文件名
  828. const file = allzip.files[fileName]
  829.  
  830. // 根据文件类型获取文件内容
  831. file.async('blob').then(content => {
  832. // 创建一个Blob对象,并下载
  833. const blob = new Blob([content])
  834. const url = URL.createObjectURL(blob)
  835. const a = document.createElement('a')
  836. a.href = url
  837. a.download = fileName
  838. document.body.appendChild(a) // 添加到文档中以确保点击有效
  839. a.click()
  840. document.body.removeChild(a) // 下载完成后移除元素
  841. URL.revokeObjectURL(url)
  842. closeModal()
  843. }).catch(error => {
  844. console.error('Error fetching file content:', error)
  845. closeModal()
  846. })
  847. downloadLogFile()
  848. return // 结束函数执行,不生成压缩包
  849. }
  850.  
  851.  
  852.  
  853. allzip.generateAsync({ type: 'blob' }, function updateCallback(metadata) {
  854. const progress = metadata.percent.toFixed(2)
  855. showModal(`压缩进度: ${progress}%`)
  856. }).then(content => {
  857. const zipUrl = URL.createObjectURL(content)
  858. const a = document.createElement('a')
  859. a.href = zipUrl
  860. a.download = `批量备份${urls.length}个片单.zip`
  861. a.click()
  862. URL.revokeObjectURL(zipUrl)
  863. closeModal()
  864. downloadLogFile()
  865. if (callback) callback()
  866. })
  867. // 如果 singleFileDownload 等于假,则执行这里的代码
  868. }
  869.  
  870.  
  871. }
  872.  
  873. function showBanner(text) {
  874. // 查找现有的横幅元素
  875. var existingBanner = document.querySelector('.banner')
  876.  
  877. if (existingBanner) {
  878. // 如果横幅已经存在,直接更新文本内容
  879. existingBanner.textContent = text
  880. } else {
  881. // 如果横幅不存在,创建一个新的横幅
  882. var banner = document.createElement('div')
  883. banner.className = 'banner' // 添加一个类名以便识别
  884. banner.style.position = 'fixed'
  885. banner.style.bottom = '20px' // 距离底部的距离
  886. banner.style.left = '20px' // 距离左侧的距离
  887. banner.style.width = 'auto' // 根据文本自动调整宽度
  888. banner.style.backgroundColor = 'rgba(255, 255, 255, 0.9)'
  889. banner.style.color = '#000' // 黑色文本
  890. banner.style.textAlign = 'center'
  891. banner.style.padding = '20px'
  892. banner.style.borderRadius = '8px'
  893. banner.style.boxShadow = '0 4px 8px rgba(0, 0, 0, 0.1)'
  894. banner.style.zIndex = '9999'
  895. banner.textContent = text // 将传入的文本设置为横幅内容
  896.  
  897. document.body.appendChild(banner) // 将横幅添加到文档的末尾
  898.  
  899. // 3秒后移除横幅提示
  900. setTimeout(function () {
  901. banner.remove()
  902. }, 3000)
  903. }
  904. }
  905.  
  906.  
  907.  
  908. // 创建或更新模态窗口
  909. function showModal(message, autoCloseDelay = 0) {
  910. // 如果模态窗口不存在,则创建新的模态窗口
  911. if (!modalContainer) {
  912. modalContainer = document.createElement('div')
  913. modalContainer.className = 'modal-container'
  914. modalContainer.style.cssText = `
  915. position: fixed;
  916. top: 50%;
  917. left: 50%;
  918. transform: translate(-50%, -50%);
  919. background-color: rgba(255, 255, 255, 0.9);
  920. border-radius: 8px;
  921. box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1);
  922. z-index: 9999;
  923. padding: 20px;
  924. `
  925. document.body.appendChild(modalContainer)
  926. }
  927.  
  928. // 更新模态窗口的内容
  929. modalContainer.textContent = message
  930.  
  931. // 自动关闭模态窗口
  932. if (autoCloseDelay > 0) {
  933. setTimeout(closeModal, autoCloseDelay)
  934. }
  935. }
  936.  
  937. // 关闭模态窗口
  938. function closeModal() {
  939. // 如果模态窗口存在,则从 DOM 中移除
  940. if (modalContainer) {
  941. document.body.removeChild(modalContainer)
  942. modalContainer = null // 将变量重置为 null,以便下次创建新的模态窗口
  943. }
  944. }
  945.  
  946. // 创建JSONindex
  947.  
  948. function createReportUI(data, itemsPerPage) {
  949. temporaryData = data
  950. // 创建全屏遮罩层
  951. const overlay = document.createElement('div')
  952. overlay.className = 'overlay'
  953. overlay.style.cssText = `
  954. position: fixed;
  955. top: 0;
  956. left: 0;
  957. width: 100%;
  958. height: 100%;
  959. background-color: rgba(0, 0, 0, 1); /* 全黑不透明背景 */
  960. z-index: 9999; /* 确保遮罩层位于所有内容之上 */
  961. `
  962. // document.body.appendChild(overlay);
  963.  
  964. const modalContainer = document.createElement('div')
  965. modalContainer.className = 'modal-container'
  966. modalContainer.style.cssText = `
  967. position: fixed;
  968. top: 50%;
  969. left: 50%;
  970. transform: translate(-50%, -50%);
  971. background-color: rgba(255, 255, 255, 1);
  972. border-radius: 8px;
  973. box-shadow: 0 4px 8px rgba(0, 0, 0, 0.2);
  974. z-index: 10000; /* 确保弹出框位于遮罩层之上 */
  975. padding: 20px;
  976. width: 80%;
  977. max-width: 800px;
  978. `
  979. const title = document.createElement('h2')
  980. title.textContent = `当前共有片单数量: ${temporaryData.length}`
  981.  
  982. title.style.textAlign = 'center'
  983. modalContainer.appendChild(title)
  984.  
  985. const closeButton = document.createElement('button')
  986. closeButton.textContent = '×'
  987. closeButton.style.position = 'absolute'
  988. closeButton.style.top = '10px'
  989. closeButton.style.right = '10px'
  990. closeButton.style.backgroundColor = 'transparent'
  991. closeButton.style.border = 'none'
  992. closeButton.style.fontSize = '24px'
  993. closeButton.style.cursor = 'pointer'
  994. modalContainer.appendChild(closeButton)
  995.  
  996. closeButton.addEventListener('click', () => {
  997.  
  998. // document.body.removeChild(overlay); // 移除遮罩层
  999. document.body.removeChild(modalContainer) // 移除模态框
  1000. })
  1001.  
  1002. const tableContainer = document.createElement('div')
  1003. tableContainer.style.cssText = `
  1004. max-height: 60vh;
  1005. overflow-y: auto;
  1006. `
  1007. modalContainer.appendChild(tableContainer)
  1008.  
  1009. const table = document.createElement('table')
  1010. table.style.width = '100%'
  1011. table.style.borderCollapse = 'collapse'
  1012. table.style.fontSize = '16px'
  1013. tableContainer.appendChild(table)
  1014.  
  1015. const thead = document.createElement('thead')
  1016. table.appendChild(thead)
  1017.  
  1018. const headerRow = document.createElement('tr')
  1019. thead.appendChild(headerRow)
  1020.  
  1021. const checkboxHeader = document.createElement('th')
  1022. checkboxHeader.textContent = '选择'
  1023. checkboxHeader.style.textAlign = 'center'
  1024. checkboxHeader.style.padding = '10px'
  1025. headerRow.appendChild(checkboxHeader)
  1026.  
  1027. const nameHeader = document.createElement('th')
  1028. nameHeader.textContent = '片单'
  1029. nameHeader.style.padding = '10px'
  1030. nameHeader.style.width = '40%'
  1031. headerRow.appendChild(nameHeader)
  1032.  
  1033. const urlHeader = document.createElement('th')
  1034. urlHeader.textContent = '地址'
  1035. urlHeader.style.padding = '10px'
  1036. urlHeader.style.width = '40%'
  1037. headerRow.appendChild(urlHeader)
  1038.  
  1039. const tbody = document.createElement('tbody')
  1040. table.appendChild(tbody)
  1041.  
  1042. let currentPage = 1
  1043. const totalItems = data.length
  1044. const totalPages = Math.ceil(totalItems / itemsPerPage)
  1045.  
  1046. function generateTableData(page) {
  1047. tbody.innerHTML = ''
  1048.  
  1049. const startIndex = (page - 1) * itemsPerPage
  1050. const endIndex = startIndex + itemsPerPage
  1051.  
  1052. for (let i = startIndex; i < endIndex && i < data.length; i++) {
  1053. const row = document.createElement('tr')
  1054. tbody.appendChild(row)
  1055. // 序号列
  1056. const indexCell = document.createElement('td')
  1057. indexCell.textContent = i + 1 // 显示序号,从1开始
  1058. indexCell.style.textAlign = 'center'
  1059. indexCell.style.padding = '5px'
  1060. row.appendChild(indexCell)
  1061. const checkboxCell = document.createElement('td')
  1062. checkboxCell.style.textAlign = 'center'
  1063. checkboxCell.style.padding = '5px'
  1064. const checkbox = document.createElement('input')
  1065. checkbox.type = 'checkbox'
  1066. checkbox.id = `checkbox_${i}`
  1067. checkbox.value = i
  1068. checkboxCell.appendChild(checkbox)
  1069. row.appendChild(checkboxCell)
  1070.  
  1071. const nameCell = document.createElement('td')
  1072. nameCell.textContent = data[i].name
  1073. nameCell.style.padding = '10px'
  1074. nameCell.style.borderBottom = '1px solid #ddd'
  1075. row.appendChild(nameCell)
  1076.  
  1077. const urlCell = document.createElement('td')
  1078. const fullUrl = 'https://missav.com/playlists/' + data[i].key
  1079. const link = document.createElement('a')
  1080. link.textContent = fullUrl
  1081. link.href = fullUrl
  1082. link.target = '_blank' // 在新标签页中打开链接
  1083. urlCell.appendChild(link)
  1084. urlCell.style.padding = '10px'
  1085. urlCell.style.borderBottom = '1px solid #ddd'
  1086. row.appendChild(urlCell)
  1087. }
  1088. }
  1089.  
  1090. generateTableData(currentPage)
  1091.  
  1092. const paginationContainer = document.createElement('div')
  1093. paginationContainer.style.marginTop = '20px'
  1094. paginationContainer.style.textAlign = 'center'
  1095. modalContainer.appendChild(paginationContainer)
  1096.  
  1097. const prevButton = document.createElement('button')
  1098. prevButton.textContent = '上一页'
  1099. prevButton.style.marginRight = '10px'
  1100. prevButton.disabled = true
  1101.  
  1102. const pageIndicator = document.createElement('span')
  1103. pageIndicator.style.marginRight = '10px'
  1104. updatePageIndicator()
  1105. paginationContainer.appendChild(pageIndicator)
  1106.  
  1107. const nextButton = document.createElement('button')
  1108. nextButton.textContent = '下一页'
  1109. nextButton.style.marginLeft = '10px'
  1110. if (totalPages <= 1) {
  1111. nextButton.disabled = true
  1112. }
  1113.  
  1114. prevButton.addEventListener('click', () => {
  1115. currentPage--
  1116. generateTableData(currentPage)
  1117. updatePaginationButtons()
  1118. updatePageIndicator()
  1119. })
  1120.  
  1121. nextButton.addEventListener('click', () => {
  1122. currentPage++
  1123. generateTableData(currentPage)
  1124. updatePaginationButtons()
  1125. updatePageIndicator()
  1126. })
  1127.  
  1128. function updatePaginationButtons() {
  1129. prevButton.disabled = currentPage === 1
  1130. nextButton.disabled = currentPage === totalPages
  1131. }
  1132.  
  1133. function updatePageIndicator() {
  1134. pageIndicator.textContent = `第 ${currentPage}/${totalPages} 页`
  1135. }
  1136. const selectAllButton = document.createElement('button')
  1137. selectAllButton.textContent = '全部选择'
  1138. selectAllButton.style.marginRight = '10px'
  1139. selectAllButton.style.marginTop = '20px'
  1140. selectAllButton.style.padding = '10px 20px'
  1141. selectAllButton.style.fontSize = '16px'
  1142. selectAllButton.style.backgroundColor = '#007bff'
  1143. selectAllButton.style.color = '#fff'
  1144. selectAllButton.style.border = 'none'
  1145. selectAllButton.style.borderRadius = '5px'
  1146. selectAllButton.style.cursor = 'pointer'
  1147. selectAllButton.style.float = 'left'
  1148. modalContainer.appendChild(selectAllButton)
  1149.  
  1150. let selectAll = true
  1151.  
  1152. selectAllButton.addEventListener('click', () => {
  1153. const checkboxes = document.querySelectorAll('input[type="checkbox"]')
  1154. checkboxes.forEach(checkbox => {
  1155. checkbox.checked = selectAll
  1156. })
  1157.  
  1158. if (selectAll) {
  1159. selectAllButton.textContent = '取消选择'
  1160. } else {
  1161. selectAllButton.textContent = '全部选择'
  1162. }
  1163.  
  1164. selectAll = !selectAll
  1165. })
  1166. //
  1167.  
  1168. const confirmButton = document.createElement('button')
  1169. confirmButton.textContent = '确认选择'
  1170. confirmButton.style.marginTop = '20px'
  1171. confirmButton.style.padding = '10px 20px'
  1172. confirmButton.style.fontSize = '16px'
  1173. confirmButton.style.backgroundColor = '#007bff'
  1174. confirmButton.style.color = '#fff'
  1175. confirmButton.style.border = 'none'
  1176. confirmButton.style.borderRadius = '5px'
  1177. confirmButton.style.cursor = 'pointer'
  1178. confirmButton.style.float = 'right'
  1179. modalContainer.appendChild(confirmButton)
  1180.  
  1181. document.body.appendChild(modalContainer)
  1182.  
  1183. confirmButton.addEventListener('click', () => {
  1184. const checkboxes = document.querySelectorAll('input[type="checkbox"]')
  1185. let anyCheckboxChecked = false
  1186. checkboxes.forEach(checkbox => {
  1187. if (checkbox.checked) {
  1188.  
  1189. const index = parseInt(checkbox.value, 10)
  1190. if (index >= 0 && index < temporaryData.length) {
  1191. // 检查 temporaryData[index] 是否为 undefined 或 null
  1192. if (temporaryData[index]) {
  1193. // 将选中的名称和URL推送到全局变量
  1194. names.push(temporaryData[index].name)
  1195. urls.push('https://missav.com/playlists/' + temporaryData[index].key)
  1196. anyCheckboxChecked = true
  1197. } else {
  1198. console.error(`temporaryData[${index}] is undefined or null.`)
  1199. }
  1200. } else {
  1201. console.error(`Index ${index} is out of bounds for temporaryData.`)
  1202. }
  1203. }
  1204.  
  1205. })
  1206.  
  1207. document.body.removeChild(modalContainer)
  1208. // document.body.removeChild(overlay);
  1209. if (anyCheckboxChecked) {
  1210.  
  1211. processUrls()
  1212. window.showLogContainer()
  1213. }
  1214.  
  1215. })
  1216. }
  1217. function ini() {
  1218. delayTime = GM_getValue('delayTime', 1000) // 从GM存储中读取延迟时间
  1219. shouldReplace = GM_getValue('shouldReplace', false) // 从GM存储中读取状态
  1220. saveJson = GM_getValue('saveJson', false) // 从GM存储中读取状态
  1221. useDefaultTitle = GM_getValue('useDefaultTitle', true)
  1222. pageCount = GM_getValue('pageCount', true)
  1223. saveVideoInfo = GM_getValue('saveVideoInfo', false)
  1224. saveImage = GM_getValue('saveImage', true)
  1225. downloadLogFileA = GM_getValue('downloadLogFileA', false)
  1226. savetowebdav = GM_getValue('savetowebdav', false)
  1227. webdavUrl = GM_getValue('webdavUrl', '')
  1228. webdavUsername = GM_getValue('webdavUsername', '')
  1229. webdavPassword = GM_getValue('webdavPassword', '')
  1230.  
  1231. }
  1232. // 创建设置界面
  1233. function createControl(tagName, attributes = {}, styles = {}, parent = document.body) {
  1234. const element = document.createElement(tagName)
  1235.  
  1236. // 设置属性
  1237. for (const key in attributes) {
  1238. element[key] = attributes[key]
  1239. }
  1240.  
  1241. // 设置样式
  1242. Object.assign(element.style, styles)
  1243.  
  1244. // 添加到父元素
  1245. if (parent) {
  1246. parent.appendChild(element)
  1247. }
  1248.  
  1249. return element
  1250. }
  1251.  
  1252. function createSettingsUI() {
  1253. const modalContainer = createControl('div', {
  1254. className: 'settings-modal'
  1255. }, {
  1256. position: 'fixed',
  1257. top: '50%',
  1258. left: '50%',
  1259. transform: 'translate(-50%, -50%)',
  1260. backgroundColor: 'rgba(255, 255, 255, 1)',
  1261. borderRadius: '8px',
  1262. boxShadow: '0 4px 8px rgba(0, 0, 0, 0.2)',
  1263. zIndex: '9999',
  1264. padding: '20px',
  1265. width: '400px',
  1266. maxWidth: '80%'
  1267. })
  1268.  
  1269. const title = createControl('h2', {
  1270. textContent: '设置'
  1271. }, {
  1272. textAlign: 'center'
  1273. })
  1274. modalContainer.appendChild(title)
  1275.  
  1276. // 创建控件并添加到模态框
  1277. const controls = [
  1278. {
  1279. type: 'checkbox',
  1280. id: 'saveImageCheckbox',
  1281. label: '下载图片',
  1282. checked: GM_getValue('saveImage', true),
  1283. onchange: function () { GM_setValue('saveImage', this.checked) }
  1284. },
  1285. {
  1286. type: 'checkbox',
  1287. id: 'saveJsonCheckbox',
  1288. label: '下载JSON',
  1289. checked: GM_getValue('saveJson', false),
  1290. onchange: function () { GM_setValue('saveJson', this.checked) }
  1291. },
  1292. {
  1293. type: 'checkbox',
  1294. id: 'hdImageCheckbox',
  1295. label: '下载高清大图',
  1296. checked: GM_getValue('shouldReplace', false),
  1297. onchange: function () { GM_setValue('shouldReplace', this.checked) }
  1298. },
  1299. {
  1300. type: 'checkbox',
  1301. id: 'defaultTitleCheckbox',
  1302. label: '使用网页标题名保存',
  1303. checked: GM_getValue('useDefaultTitle', true),
  1304. onchange: function () { GM_setValue('useDefaultTitle', this.checked) }
  1305. },
  1306. {
  1307. type: 'checkbox',
  1308. id: 'saveVideoInfoCheckbox',
  1309. label: '下载视频信息',
  1310. checked: GM_getValue('saveVideoInfo', false),
  1311. onchange: function () { GM_setValue('saveVideoInfo', this.checked) }
  1312. },
  1313. {
  1314. type: 'checkbox',
  1315. id: 'pageCountCheckbox',
  1316. label: '自定义抓取页数',
  1317. checked: GM_getValue('pageCount', true),
  1318. onchange: function () { GM_setValue('pageCount', this.checked) }
  1319. },
  1320. {
  1321. type: 'checkbox',
  1322. id: 'downloadLogFileA',
  1323. label: '保存下载日志',
  1324. checked: GM_getValue('downloadLogFileA', false),
  1325. onchange: function () { GM_setValue('downloadLogFileA', this.checked) }
  1326. },
  1327. {
  1328. type: 'checkbox',
  1329. id: 'savetowebdav',
  1330. label: '上传JSON到WebDav',
  1331. checked: GM_getValue('savetowebdav', false),
  1332. onchange: function () { GM_setValue('savetowebdav', this.checked) }
  1333. },
  1334. {
  1335. type: 'number',
  1336. id: 'delayInput',
  1337. label: '设置延迟(毫秒)',
  1338. value: GM_getValue('delayTime', 1000),
  1339. placeholder: '设置延迟(毫秒)',
  1340. onchange: function () { GM_setValue('delayTime', this.value) },
  1341. style: {
  1342. width: '1px' // 设置输入框宽度为100%
  1343. }
  1344. },
  1345. {
  1346. type: 'text',
  1347. id: 'webdavUrlInput',
  1348. label: 'WebDAV 网址',
  1349. value: GM_getValue('webdavUrl', ''),
  1350. placeholder: '输入WebDAV网址',
  1351. onchange: function () { GM_setValue('webdavUrl', this.value) }
  1352. },
  1353. {
  1354. type: 'text',
  1355. id: 'webdavUsernameInput',
  1356. label: 'WebDAV 账号',
  1357. value: GM_getValue('webdavUsername', ''),
  1358. placeholder: '输入WebDAV账号',
  1359. onchange: function () { GM_setValue('webdavUsername', this.value) }
  1360. },
  1361. {
  1362. type: 'text',
  1363. id: 'webdavPasswordInput',
  1364. label: 'WebDAV 密码',
  1365. value: GM_getValue('webdavPassword', ''),
  1366. placeholder: '输入WebDAV密码',
  1367. onchange: function () { GM_setValue('webdavPassword', this.value) }
  1368. }
  1369. ]
  1370.  
  1371. controls.forEach(control => {
  1372. const input = createControl('input', {
  1373. type: control.type,
  1374. id: control.id,
  1375. checked: control.checked,
  1376. value: control.value,
  1377. placeholder: control.placeholder,
  1378. onchange: control.onchange
  1379. }, {
  1380. marginRight: '10px'
  1381. })
  1382.  
  1383. const label = createControl('label', {
  1384. textContent: control.label,
  1385. htmlFor: control.id
  1386. }, {
  1387. fontSize: '14px',
  1388. marginLeft: '5px'
  1389. })
  1390.  
  1391. modalContainer.appendChild(input)
  1392. modalContainer.appendChild(label)
  1393. modalContainer.appendChild(createControl('br'))
  1394. })
  1395.  
  1396. // 关闭按钮
  1397. const closeButton = createControl('button', {
  1398. textContent: '关闭',
  1399. onclick: () => {
  1400. ini()
  1401. WebDAVManager.updateConfig(webdavUrl, webdavUsername, webdavPassword)
  1402. document.body.removeChild(modalContainer)
  1403. }
  1404. }, {
  1405. marginTop: '10px',
  1406. padding: '10px 20px',
  1407. fontSize: '16px',
  1408. backgroundColor: '#007bff',
  1409. color: '#fff',
  1410. border: 'none',
  1411. borderRadius: '5px',
  1412. cursor: 'pointer',
  1413. float: 'right'
  1414. })
  1415.  
  1416. modalContainer.appendChild(closeButton)
  1417.  
  1418. // 将模态框添加到页面
  1419. document.body.appendChild(modalContainer)
  1420.  
  1421. // 添加移动端样式
  1422. const mediaQuery = window.matchMedia('(max-width: 600px)')
  1423. if (mediaQuery.matches) {
  1424. modalContainer.style.width = '90%'
  1425. modalContainer.style.maxWidth = '90%'
  1426. modalContainer.style.padding = '10px'
  1427. }
  1428. }
  1429.  
  1430. // 在页面加载时调用设置界面创建函数
  1431.  
  1432.  
  1433. // 调用示例
  1434.  
  1435.  
  1436. const WebDAVManager = (function () {
  1437. // WebDAV 配置
  1438. let url = webdavUrl
  1439. let username = webdavUsername
  1440. let password = webdavPassword
  1441.  
  1442.  
  1443. // 通用 GM_xmlhttpRequest 封装函数
  1444. function GM_xhr({ path = '/', method, success, fail, headers = {}, data, ...config }) {
  1445.  
  1446. return new Promise(resolve => {
  1447. GM_xmlhttpRequest({
  1448. url: url + path,
  1449. method,
  1450. ...config,
  1451. headers: {
  1452. 'Connection': 'Keep-Alive', // 保持连接
  1453. 'User-Agent': 'Mozilla/5.0', // 用户代理
  1454. 'Authorization': 'Basic ' + btoa(username + ':' + password), // 基本认证
  1455. ...headers
  1456. },
  1457. data,
  1458. onload: xhr => {
  1459. if (xhr.status >= 200 && xhr.status < 300) {
  1460. if (success) success(xhr)
  1461. } else {
  1462. if (fail) fail(xhr)
  1463. }
  1464. resolve(xhr)
  1465. },
  1466. onerror: xhr => {
  1467. if (fail) fail(xhr)
  1468. resolve(xhr)
  1469. }
  1470. })
  1471. })
  1472. }
  1473. //登录
  1474. async function login() {
  1475. let retryCount = 2 // 设置重试次数为2次
  1476.  
  1477. while (retryCount > 0) {
  1478. // 构建登录请求
  1479. const LOGIN = {
  1480. method: 'PROPFIND', // 使用 PROPFIND 方法检查根目录
  1481. path: retryCount === 2 ? '/' : '', // 根据重试次数设置 path
  1482. headers: {
  1483. 'Depth': '1',
  1484. 'Authorization': 'Basic ' + btoa(username + ':' + password),
  1485. 'Connection': 'Keep-Alive', // 保持连接
  1486. 'User-Agent': 'Mozilla/5.0' // 用户代理
  1487. }
  1488. }
  1489.  
  1490. // 发起登录请求
  1491. const loginResponse = await GM_xhr(LOGIN)
  1492.  
  1493. // 判断登录结果
  1494. if (loginResponse.status === 207) {
  1495. console.log('登录成功')
  1496. // 登录成功后,可以执行其他操作
  1497. showModal('Webdav登录成功!', 1000)
  1498. return true // 返回登录成功标志
  1499. } else {
  1500. console.error('登录失败:', loginResponse.status)
  1501. if (retryCount === 1) {
  1502. showModal('Webdav登录失败!' + loginResponse.status, 1000)
  1503. return false // 返回登录失败标志
  1504. } else {
  1505. retryCount-- // 减少重试次数
  1506. }
  1507. }
  1508. }
  1509.  
  1510. // 如果重试次数用尽仍未登录成功,执行其他操作(可根据实际情况添加代码)
  1511. console.error('重试次数用尽,登录失败')
  1512. showModal('Webdav登录失败!重试次数用尽', 1000)
  1513. return false
  1514. }
  1515.  
  1516. //刷新
  1517. function updateConfig(newUrl, newUsername, newPassword) {
  1518. if (newUrl && newUsername && newPassword) {
  1519. url = newUrl
  1520. username = newUsername
  1521. password = newPassword
  1522. // 在这里调用登录函数
  1523. login()
  1524. } else {
  1525. console.error('WebDAV 配置信息不完整')
  1526. }
  1527. }
  1528. // 获取 WebDAV 中指定路径下的所有文件和文件夹
  1529. async function listFilesAndFolders(folderName) {
  1530. const path = folderName.endsWith('/') ? folderName : folderName + '/'
  1531. const PROPFIND = {
  1532. method: 'PROPFIND',
  1533. headers: {
  1534. 'Depth': '1'
  1535. },
  1536. success: (xhr) => {
  1537. const parser = new DOMParser()
  1538. const xmlDoc = parser.parseFromString(xhr.responseText, 'text/xml')
  1539. const responses = xmlDoc.getElementsByTagNameNS('DAV:', 'response')
  1540.  
  1541. let fileCount = 0
  1542. //let fileList = `<div class="custom-hr"></div>`;
  1543. let fileList = ''
  1544.  
  1545. for (let i = 0; i < responses.length; i++) {
  1546. const href = decodeURIComponent(responses[i].getElementsByTagNameNS('DAV:', 'href')[0].textContent.trim())
  1547. const displayName = href.substring(href.lastIndexOf('/') + 1)
  1548. const isCollection = responses[i].getElementsByTagNameNS('DAV:', 'collection').length > 0
  1549.  
  1550. if (!isCollection) {
  1551. fileList += `
  1552. <li class="file-item">
  1553. <input type="checkbox" name="fileCheckbox" class="file-checkbox">
  1554. <span>${fileCount + 1}.</span> <!-- 文件序号 -->
  1555. <a href="#" data-display-name="${displayName}" onclick="WebDAVManager.downloadAndDisplayFile('${folderName}', '${displayName}')" class="file-link">${displayName}</a> <!-- 文件名链接 -->
  1556. <div class="file-actions">
  1557. <a href="#" onclick="WebDAVManager.renameFile('${folderName}', '${displayName}')" class="file-action">更名</a>
  1558. <a href="#" onclick="WebDAVManager.deleteFile('${folderName}', '${displayName}')" class="file-action">删除</a>
  1559. <a href="#" onclick="WebDAVManager.downloadFile('${folderName}', '${displayName}')" class="file-action">下载</a>
  1560. </div>
  1561. <div style="clear: both;"></div> <!-- 清除浮动 -->
  1562. </li>
  1563. `
  1564. fileCount++
  1565. }
  1566. }
  1567. /*fileList += `
  1568. <div class="custom-hr"></div>
  1569. <p>文件数量: ${fileCount}</p>
  1570. </ul>
  1571. `;*/
  1572. fileList += '</ul>'
  1573.  
  1574. showDialog(`
  1575. <span style="font-size: 24px;">文件夹 ${folderName}</span><br>
  1576. <span style="color: green;"><span id="fileCountLabel">文件数量:</span> <span class="file-count">${fileCount}</span></span>
  1577. <div class="custom-hr"></div>
  1578. `, fileList)
  1579.  
  1580. },
  1581. fail: (xhr) => {
  1582. alert('获取文件列表失败:' + xhr.status)
  1583. }
  1584. }
  1585.  
  1586. await GM_xhr({ ...PROPFIND, path })
  1587. }
  1588.  
  1589. // 下载文件函数
  1590.  
  1591. // 在 WebDAV 中创建新文件夹
  1592. async function createFolder(folderName) {
  1593. const MKCOL = {
  1594. method: 'MKCOL',
  1595. path: folderName.endsWith('/') ? folderName : folderName + '/',
  1596. success: () => {
  1597. alert('文件夹创建成功')
  1598. },
  1599. fail: (xhr) => {
  1600. if (xhr.status === 409) {
  1601. // alert('冲突:文件夹可能已经存在或路径不正确');
  1602. } else {
  1603. //alert('创建文件夹失败:' + xhr.status);
  1604. }
  1605. }
  1606. }
  1607.  
  1608. await GM_xhr({ ...MKCOL })
  1609. }
  1610.  
  1611. // 上传文件到 WebDAV
  1612.  
  1613. async function uploadFile(folderName, fileName, fileContent) {
  1614. // 检查文件是否已存在
  1615. const HEAD = {
  1616. method: 'HEAD',
  1617. path: folderName + '/' + fileName,
  1618. success: () => {
  1619. //alert('文件已存在,无需重复上传');
  1620. window.addToLog(fileName + '文件存在', 'info')
  1621. // 可以在这里执行文件已存在时的逻辑,比如显示提示信息或执行其他操作
  1622. },
  1623. fail: async (xhr) => {
  1624. if (xhr.status === 404) {
  1625. // 文件不存在,执行上传操作
  1626. const PUT = {
  1627. method: 'PUT',
  1628. path: folderName + '/' + fileName,
  1629. data: fileContent,
  1630. success: () => {
  1631.  
  1632. window.addToLog(fileName + '文件上传成功', 'info')
  1633. },
  1634. fail: (xhr) => {
  1635. window.addToLog(fileName + '文件上传失败', 'error')
  1636. }
  1637. }
  1638.  
  1639. await GM_xhr({ ...PUT })
  1640. } else {
  1641. window.addToLog(fileName + '文件检查上传失败', 'error')
  1642. }
  1643. }
  1644. }
  1645.  
  1646. await GM_xhr({ ...HEAD })
  1647. }
  1648.  
  1649. // 重命名文件
  1650. async function renameFile(folderName, oldFileName) {
  1651. // 弹出对话框让用户输入新文件名
  1652. const newFileName = prompt(`请输入 ${oldFileName} 的新文件名:`)
  1653. if (!newFileName) {
  1654. alert('未输入新文件名,操作已取消。')
  1655. return
  1656. }
  1657. const encodedOldFileName = encodeURIComponent(oldFileName) // 对旧文件名进行编码
  1658.  
  1659. const encodedNewFileName = encodeURIComponent(newFileName + '.json') // 对新文件名进行编码
  1660. const sourcePath = folderName + '/' + encodedOldFileName
  1661. const destinationPath = folderName + '/' + encodedNewFileName
  1662.  
  1663. const MOVE = {
  1664. method: 'MOVE',
  1665. path: sourcePath,
  1666. headers: {
  1667. 'Destination': url + destinationPath,
  1668. 'Overwrite': 'T' // 允许覆盖目标文件
  1669. },
  1670. success: () => {
  1671. alert(`文件 ${oldFileName} 已成功重命名为 ${newFileName}`)
  1672. listFilesAndFolders(folderName) // 刷新文件列表
  1673. },
  1674. fail: (xhr) => {
  1675. if (xhr.status === 409) {
  1676. alert('冲突:文件名可能已存在或路径不正确')
  1677. } else {
  1678. alert('重命名文件失败:' + xhr.status)
  1679. }
  1680. }
  1681. }
  1682.  
  1683. await GM_xhr({ ...MOVE })
  1684. }
  1685.  
  1686.  
  1687. // 删除文件
  1688. async function deleteFile(folderName, fileName) {
  1689. let confirmdelete = false
  1690. if (deleteSelected) {
  1691. confirmdelete = true
  1692. } else {
  1693. confirmdelete = confirm(`确定要删除文件 ${fileName} 吗?`)
  1694. }
  1695.  
  1696. if (confirmdelete) {
  1697. try {
  1698. await deleteFileFromServer(folderName, fileName)
  1699. const fileItem = document.querySelector(`a[onclick*="${fileName}"]`).closest('li')
  1700. const checkbox = fileItem.querySelector('input[type="checkbox"]')
  1701. if (checkbox && checkbox.checked) {
  1702. checkbox.checked = false // 清除选中状态
  1703. checkbox.disabled = true // 禁止操作
  1704. }
  1705. markItemDeleted(fileItem)
  1706. } catch (error) {
  1707. alert('删除文件失败:' + error.message)
  1708. }
  1709. }
  1710. }
  1711. function markItemDeleted(fileItem) {
  1712. // 将文件名链接标记为已删除
  1713. const fileNameLink = fileItem.querySelector('a')
  1714. fileNameLink.style.textDecoration = 'line-through'
  1715. fileNameLink.removeAttribute('onclick')
  1716. fileNameLink.style.pointerEvents = 'none' // 不可点击
  1717. fileNameLink.style.color = 'gray' // 设置为灰色
  1718.  
  1719. // 将后续按钮标记为已删除
  1720. const actionButtons = fileItem.querySelectorAll('div > a')
  1721. actionButtons.forEach(button => {
  1722. button.style.textDecoration = 'line-through'
  1723. button.removeAttribute('onclick')
  1724. button.style.pointerEvents = 'none' // 不可点击
  1725. button.style.color = 'gray' // 设置为灰色
  1726. })
  1727. }
  1728. // 实际删除文件的函数
  1729. async function deleteFileFromServer(folderName, fileName) {
  1730. const DELETE = {
  1731. method: 'DELETE',
  1732. path: folderName + '/' + fileName
  1733. }
  1734.  
  1735. return new Promise((resolve, reject) => {
  1736. GM_xhr({ ...DELETE, success: resolve, fail: reject })
  1737. })
  1738. }
  1739.  
  1740. // 检查文件夹是否存在
  1741. async function checkFolderExists(folderName) {
  1742. const HEAD = {
  1743. method: 'HEAD',
  1744. path: folderName.endsWith('/') ? folderName : folderName + '/',
  1745. success: (xhr) => {
  1746. if (xhr.status === 200) {
  1747. // alert('文件夹存在');
  1748. } else if (xhr.status === 404) {
  1749. alert('文件夹不存在')
  1750. } else {
  1751. alert('检查文件夹状态失败:' + xhr.status)
  1752. }
  1753. },
  1754. fail: (xhr) => {
  1755. alert('检查文件夹失败:' + xhr.status)
  1756. }
  1757. }
  1758.  
  1759. await GM_xhr({ ...HEAD })
  1760. }
  1761.  
  1762. // 下载并展示文件
  1763. async function downloadFile(folderName, fileName, zip = null) {
  1764. return new Promise((resolve, reject) => {
  1765. // 发起 GET 请求下载文件内容
  1766. const GET = {
  1767. method: 'GET',
  1768. path: folderName + '/' + fileName,
  1769. success: (xhr) => {
  1770. const fileContent = xhr.responseText
  1771. const jsonData = JSON.parse(fileContent)
  1772. const content = generateJsonIndexContent(jsonData)
  1773.  
  1774. if (zip) {
  1775. // 如果传入了压缩包实例,则将文件内容添加到压缩包中
  1776. const sanitizedFileName = sanitizeFileName(fileName.replace('.json', '')) + '.html'
  1777. zip.file(sanitizedFileName, content)
  1778. resolve()
  1779. } else {
  1780. // 否则直接下载文件
  1781. const blob = new Blob([content], { type: 'text/html' })
  1782. const htmlUrl = URL.createObjectURL(blob)
  1783. const a = document.createElement('a')
  1784. a.href = htmlUrl
  1785. a.download = sanitizeFileName(fileName.replace('.json', '')) + '.html'
  1786. a.click()
  1787. resolve()
  1788. }
  1789. },
  1790. fail: (xhr) => {
  1791. reject(new Error('下载文件失败:' + xhr.status))
  1792. }
  1793. }
  1794.  
  1795. GM_xhr({ ...GET })
  1796. })
  1797. }
  1798.  
  1799.  
  1800.  
  1801. async function downloadAndDisplayFile(folderName, fileName) {
  1802. const GET = {
  1803. method: 'GET',
  1804. path: folderName + '/' + fileName,
  1805. success: (xhr) => {
  1806. try {
  1807. const fileContent = xhr.responseText
  1808. const jsonData = JSON.parse(fileContent)
  1809. const content = generateJsonIndexContent(jsonData)
  1810.  
  1811. // 打开一个新的浏览器标签页显示内容
  1812. const dataUrl = 'data:text/html;charset=utf-8,' + encodeURIComponent(content)
  1813. GM_openInTab(dataUrl)
  1814. } catch (e) {
  1815. alert('解析 JSON 文件失败:' + e.message)
  1816. }
  1817. },
  1818. fail: (xhr) => {
  1819. alert('下载文件失败:' + xhr.status)
  1820. }
  1821. }
  1822.  
  1823. await GM_xhr({ ...GET })
  1824. }
  1825. //我是你爹啊
  1826. // 显示对话框
  1827. function showDialog(title, fileList, folderName) {
  1828. // 添加CSS样式
  1829. const style = document.createElement('style')
  1830. style.textContent = `
  1831. .overlay {
  1832. position: fixed;
  1833. top: 0;
  1834. left: 0;
  1835. width: 100%;
  1836. height: 100%;
  1837. background-color: rgba(0, 0, 0, 0.5);
  1838. z-index: 9998;
  1839. }
  1840.  
  1841. .dialog {
  1842. position: fixed;
  1843. top: 50%;
  1844. left: 50%;
  1845. transform: translate(-50%, -50%);
  1846. background: #fff;
  1847. padding: 20px;
  1848. border-radius: 10px;
  1849. width: 80%;
  1850. max-height: 80%;
  1851. overflow-y: auto;
  1852. box-shadow: 0 0 20px rgba(0, 0, 0, 0.2);
  1853. z-index: 9999;
  1854. }
  1855.  
  1856. .dialog-title {
  1857. margin-bottom: 20px;
  1858. }
  1859.  
  1860. .dialog-button {
  1861. padding: 10px 20px;
  1862. border: none;
  1863. border-radius: 5px;
  1864. cursor: pointer;
  1865. margin-bottom: 20px;
  1866. }
  1867.  
  1868. .select-button {
  1869. background-color: #007BFF;
  1870. color: #fff;
  1871. }
  1872.  
  1873. .delete-button {
  1874. background-color: #dc3545;
  1875. color: #fff;
  1876. margin-left: 10px;
  1877. }
  1878.  
  1879. .select-all-button {
  1880. background-color: #28a745;
  1881. color: #fff;
  1882. margin-left: 10px;
  1883. }
  1884.  
  1885. .download-button {
  1886. background-color: #007BFF;
  1887. color: #fff;
  1888. margin-left: 10px;
  1889. }
  1890.  
  1891. .search-button {
  1892. background-color: #6c757d;
  1893. color: #fff;
  1894. margin-left: 10px;
  1895. border: none;
  1896. border-radius: 5px;
  1897. cursor: pointer;
  1898. }
  1899. .close-x-button {
  1900. background-color: #6c757d;
  1901. color: red;
  1902. margin-left: 10px;
  1903. border: none;
  1904. border-radius: 5px;
  1905. cursor: pointer;
  1906. }
  1907.  
  1908. .search-button:hover {
  1909. background-color: #495057;
  1910. }
  1911.  
  1912. .close-button {
  1913. background-color: #007BFF;
  1914. color: #fff;
  1915. margin-top: 20px;
  1916. }
  1917.  
  1918. .file-list {
  1919. list-style-type: none;
  1920. padding: 0;
  1921. }
  1922.  
  1923. .file-action {
  1924. margin-right: 10px;
  1925. float: left;
  1926. color: blue;
  1927. cursor: pointer;
  1928. }
  1929.  
  1930. .file-actions-container {
  1931. float: right;
  1932. }
  1933.  
  1934. .clear-float {
  1935. clear: both;
  1936. }
  1937.  
  1938. .file-item {
  1939. border-bottom: 1px solid #ddd;
  1940. padding: 10px 0;
  1941. display: flex;
  1942. align-items: center;
  1943. }
  1944.  
  1945. .file-checkbox {
  1946. margin-right: 10px;
  1947. }
  1948.  
  1949. .file-link {
  1950. margin-right: 10px;
  1951. color: blue;
  1952. flex-grow: 1;
  1953. }
  1954.  
  1955. .file-actions {
  1956. display: flex;
  1957. gap: 10px;
  1958. }
  1959.  
  1960. .file-action {
  1961. color: #007BFF;
  1962. cursor: pointer;
  1963. }
  1964.  
  1965. .custom-hr {
  1966. height: 1px;
  1967. background-color: #007BFF;
  1968. margin: 20px 0;
  1969. }
  1970.  
  1971. .file-action:hover {
  1972. text-decoration: underline;
  1973. }
  1974.  
  1975. @media (max-width: 600px) {
  1976. .dialog-button {
  1977. padding: 5px 10px;
  1978. font-size: 12px;
  1979. margin-bottom: 10px;
  1980. }
  1981.  
  1982. .delete-button,
  1983. .download-button,
  1984. .select-all-button,
  1985. .close-x-button,
  1986. .search-button {
  1987. margin-left: 0;
  1988. }
  1989.  
  1990. .button-container {
  1991. right: 5px;
  1992. }
  1993. }
  1994.  
  1995. `
  1996. document.head.appendChild(style)
  1997.  
  1998. // 创建对话框
  1999. const dialog = document.createElement('div')
  2000. dialog.className = 'dialog'
  2001. // 创建遮罩
  2002.  
  2003.  
  2004. function close() {
  2005. document.body.removeChild(dialog)
  2006.  
  2007. }
  2008.  
  2009. // 标题
  2010. const titleElement = document.createElement('h1')
  2011. titleElement.innerHTML = title
  2012. titleElement.className = 'dialog-title'
  2013. dialog.appendChild(titleElement)
  2014.  
  2015.  
  2016. const buttonContainer = document.createElement('div')
  2017. const buttonConfigs = [
  2018. { text: '选择列表', className: 'dialog-button select-button', onclick: toggleSelection },
  2019. { text: '搜索列表', className: 'dialog-button search-button', onclick: toggleSearch },
  2020. { text: '关闭窗口', className: 'dialog-button close-x-button', onclick: close },
  2021. { text: '全部选中', className: 'dialog-button select-all-button', onclick: toggleSelectAll },
  2022. { text: '删除选中', className: 'dialog-button delete-button', onclick: deleteSelectedFiles },
  2023. { text: '下载选中', className: 'dialog-button download-button', onclick: downloadSelectedFiles }
  2024.  
  2025. // 可以添加更多按钮配置
  2026. ]
  2027. let count = 0 // 计数器,用于控制显示状态
  2028. buttonConfigs.forEach(config => {
  2029. const button = document.createElement('button')
  2030. button.textContent = config.text
  2031. button.className = config.className
  2032. button.onclick = config.onclick
  2033.  
  2034. if (count < 3) {
  2035. button.style.display = 'inline-block' // 前两个按钮初始可见
  2036. } else {
  2037. button.style.display = 'none' // 后面的按钮初始隐藏
  2038. }
  2039.  
  2040. buttonContainer.appendChild(button)
  2041. count++ // 每创建一个按钮,计数器加一
  2042. })
  2043. // 将按钮容器添加到对话框中
  2044. dialog.appendChild(buttonContainer)
  2045.  
  2046. // 文件列表
  2047. const fileListContainer = document.createElement('ul')
  2048. fileListContainer.className = 'file-list'
  2049. fileListContainer.innerHTML = fileList
  2050. dialog.appendChild(fileListContainer)
  2051.  
  2052. // 添加到body
  2053.  
  2054. document.body.appendChild(dialog)
  2055. let selectButtonInitialTop = 0
  2056. let selectAllButton = document.querySelector('.select-all-button')
  2057. let searchButton = document.querySelector('.search-button')
  2058. let downloadSelectedButton = document.querySelector('.download-button')
  2059. let deleteSelectedButton = document.querySelector('.delete-button')
  2060. let selectButton = document.querySelector('.select-button')
  2061. let closeXButton = document.querySelector('.close-x-button')
  2062. let scrollTimeout = 0 // 定义全局变量存储定时器
  2063. // 记录选择按钮的初始位置
  2064. if (selectButton) {
  2065. selectButtonInitialTop = selectButton.offsetTop
  2066. }
  2067.  
  2068. // 监听对话框的滚动事件
  2069. dialog.addEventListener('scroll', function () {
  2070. clearTimeout(scrollTimeout)
  2071. const buttons = [
  2072. { button: selectButton, offsetHeight: true },
  2073. { button: selectAllButton, offsetHeight: true },
  2074.  
  2075. { button: deleteSelectedButton, offsetHeight: true },
  2076. { button: downloadSelectedButton, offsetHeight: true },
  2077. { button: searchButton, offsetHeight: true },
  2078. { button: closeXButton, offsetHeight: true }
  2079.  
  2080.  
  2081. // 添加更多按钮对象,如果有的话
  2082. ]
  2083. scrollTimeout = setTimeout(() => {
  2084. if (buttons.every(buttonObj => buttonObj.button)) {
  2085. const dialogRect = dialog.getBoundingClientRect()
  2086. const fileListRect = fileListContainer.getBoundingClientRect()
  2087. const newButtonTopBase = Math.max(fileListRect.top, dialog.scrollTop)
  2088. let newButtonTop = newButtonTopBase
  2089.  
  2090. for (const { button, offsetHeight } of buttons) {
  2091. if (button) {
  2092. button.style.position = dialog.scrollTop === 0 ? 'static' : 'absolute'
  2093. button.style.top = `${newButtonTop}px`
  2094. button.style.right = '10px'
  2095. if (offsetHeight) {
  2096. newButtonTop += button.offsetHeight
  2097. }
  2098. }
  2099. }
  2100. }
  2101. }, 300) // 设置 300 毫秒的超时
  2102. })
  2103.  
  2104.  
  2105. // 获取所有复选框
  2106. const checkboxes = fileListContainer.querySelectorAll('input[type="checkbox"]')
  2107.  
  2108. // 初始隐藏复选框
  2109. checkboxes.forEach(checkbox => {
  2110. checkbox.style.display = 'none'
  2111. })
  2112. //选中全部
  2113. function toggleSelectAll() {
  2114. const checkboxes = document.querySelectorAll('input[name="fileCheckbox"]')
  2115. const allChecked = Array.from(checkboxes).every(checkbox => checkbox.checked)
  2116.  
  2117. checkboxes.forEach(checkbox => {
  2118. checkbox.checked = !allChecked
  2119. })
  2120.  
  2121. // 切换按钮文本和背景颜色
  2122. if (allChecked) {
  2123. selectAllButton.textContent = '全部选中'
  2124. selectAllButton.style.backgroundColor = '' // 恢复默认背景颜色
  2125. } else {
  2126. selectAllButton.textContent = '全部取消选中'
  2127. selectAllButton.style.backgroundColor = 'red' // 改为红色背景
  2128. }
  2129. }
  2130. //搜索
  2131. //---------------
  2132. function toggleSearch() {
  2133. if (searchButton.textContent === '搜索列表') {
  2134. // 如果当前是搜索状态,则进行搜索
  2135. const searchTerm = prompt('请输入搜索内容:') // 弹出输入框等待用户输入搜索内容
  2136. if (searchTerm !== null) { // 用户点击了确定按钮
  2137. filterFiles(searchTerm) // 调用过滤文件函数,传入搜索关键词
  2138. searchButton.textContent = '回到列表' // 将搜索按钮文本改为"Back"
  2139. }
  2140. } else {
  2141. // 如果当前是Back状态,则恢复初始文件列表
  2142. resetFileList()
  2143. }
  2144. }
  2145.  
  2146. function filterFiles(searchTerm) {
  2147. const files = Array.from(fileListContainer.children) // 获取文件列表的所有子元素(文件项)
  2148. let matchCount = 0 // 初始化匹配数量为 0
  2149.  
  2150. // 遍历文件列表,根据搜索关键词过滤显示符合条件的文件项
  2151. files.forEach(file => {
  2152. const fileName = file.querySelector('.file-link').textContent.toLowerCase() // 获取文件名并转换为小写
  2153. if (fileName.includes(searchTerm.toLowerCase())) {
  2154. file.style.display = '' // 匹配到的文件项显示
  2155. matchCount++ // 匹配数量加一
  2156. } else {
  2157. file.style.display = 'none' // 不匹配的文件项隐藏
  2158. }
  2159. })
  2160.  
  2161. // 更新文件数量标签内容
  2162. const fileCountLabel = document.getElementById('fileCountLabel')
  2163. if (fileCountLabel) {
  2164. fileCountLabel.textContent = '搜索数量: '
  2165. }
  2166.  
  2167. // 更新包裹文件数量的元素内容
  2168. const fileCountElement = document.querySelector('.file-count')
  2169. if (fileCountElement) {
  2170. fileCountElement.textContent = matchCount
  2171. }
  2172. }
  2173.  
  2174.  
  2175. // 恢复初始文件列表函数
  2176. function resetFileList() {
  2177. const files = Array.from(fileListContainer.children) // 获取文件列表的所有子元素(文件项)
  2178. files.forEach(file => {
  2179. file.style.display = '' // 显示所有文件项
  2180. })
  2181. searchButton.textContent = '搜索列表' // 恢复搜索按钮文本为"搜索"
  2182.  
  2183. // 更新文件数量标签内容为 "文件数量:"
  2184. const fileCountLabel = document.getElementById('fileCountLabel')
  2185. if (fileCountLabel) {
  2186. fileCountLabel.textContent = '文件数量:'
  2187. }
  2188.  
  2189. // 更新包裹文件数量的元素内容为当前文件数量
  2190. const fileCountElement = document.querySelector('.file-count')
  2191. if (fileCountElement) {
  2192. const currentFileCount = files.length // 获取当前文件数量
  2193. fileCountElement.textContent = currentFileCount
  2194. }
  2195. }
  2196. //搜索结束
  2197. //---------------
  2198.  
  2199. // 选择列表
  2200. function toggleSelection() {
  2201.  
  2202. checkboxes.forEach(checkbox => {
  2203. checkbox.style.display = checkbox.style.display === 'none' ? 'inline-block' : 'none'
  2204. checkbox.checked = false // 取消复选框的选中状态
  2205. })
  2206.  
  2207. // 切换显示“删除选中”和“下载选中”按钮
  2208. deleteSelectedButton.style.display = deleteSelectedButton.style.display === 'none' ? 'inline-block' : 'none'
  2209. downloadSelectedButton.style.display = downloadSelectedButton.style.display === 'none' ? 'inline-block' : 'none'
  2210. selectAllButton.style.display = selectAllButton.style.display === 'none' ? 'inline-block' : 'none'
  2211. selectButton.textContent = downloadSelectedButton.style.display === 'none' ? '选择列表' : '取消选择'
  2212. }
  2213.  
  2214. }
  2215. //选中下载
  2216. async function downloadSelectedFiles() {
  2217. // 获取所有选中的复选框
  2218. const checkboxes = document.querySelectorAll('input[name="fileCheckbox"]:checked')
  2219. if (checkboxes.length === 0) {
  2220. showModal('没有选中的文件', 2000)
  2221. return
  2222. }
  2223. // 创建一个 JSZip 实例
  2224. const zip = new JSZip()
  2225.  
  2226. try {
  2227. // 遍历所有选中的复选框
  2228. for (const checkbox of checkboxes) {
  2229. // 获取文件项信息
  2230. const fileItem = checkbox.closest('li')
  2231. const fileName = fileItem.querySelector('a[data-display-name]').dataset.displayName
  2232.  
  2233. // 调用 downloadFile 函数,将文件添加到压缩包中
  2234. await downloadFile(webdavfold, fileName, zip)
  2235. }
  2236.  
  2237. // 生成 ZIP 文件
  2238. const zipBlob = await zip.generateAsync({ type: 'blob' })
  2239.  
  2240. // 创建下载链接并触发下载
  2241. const zipUrl = URL.createObjectURL(zipBlob)
  2242. const a = document.createElement('a')
  2243. a.href = zipUrl
  2244. a.download = 'selected_files.zip'
  2245. a.click()
  2246.  
  2247. console.log('压缩包下载完成')
  2248. showModal('下载选中完成:', 2000)
  2249. } catch (error) {
  2250. console.error('下载选中文件失败:', error)
  2251. showModal('下载选中文件失败: ' + error.message, 2000)
  2252.  
  2253. }
  2254. }
  2255.  
  2256.  
  2257. //选中删除
  2258. function deleteSelectedFiles() {
  2259. // 获取所有选中的复选框
  2260. const selectedCheckboxes = document.querySelectorAll('input[name="fileCheckbox"]:checked')
  2261.  
  2262. if (selectedCheckboxes.length === 0) {
  2263. alert('没有选中文件')
  2264. return
  2265. }
  2266.  
  2267. // 弹出确认删除的提示框
  2268. const confirmDelete = confirm('确定要删除选中的文件吗?')
  2269.  
  2270. if (confirmDelete) {
  2271. deleteSelected = true
  2272. selectedCheckboxes.forEach(checkbox => {
  2273. const fileItem = checkbox.closest('li')
  2274. const deleteButton = fileItem.querySelector('a[onclick*="deleteFile"]')
  2275.  
  2276. // 调试日志
  2277. console.log('fileItem:', fileItem)
  2278. console.log('deleteButton:', deleteButton)
  2279.  
  2280. // 获取文件名
  2281. const fileName = fileItem.querySelector('a[data-display-name]').dataset.displayName
  2282.  
  2283. // 添加调试日志
  2284. console.log('Deleting file:', fileName)
  2285.  
  2286. // 触发删除按钮的点击事件
  2287. deleteButton.click()
  2288. //deleteFile(webdavfold,fileName);
  2289.  
  2290. // 将文件项标记为已删除
  2291. // markItemDeleted(fileItem);
  2292. })
  2293. deleteSelected = false
  2294. console.log('删除选中文件')
  2295. }
  2296. }
  2297.  
  2298.  
  2299. // 将 API 方法公开
  2300. return {
  2301. listFilesAndFolders,
  2302. updateConfig,
  2303. createFolder,
  2304. uploadFile,
  2305. renameFile,
  2306. deleteFile,
  2307. downloadFile,
  2308. downloadAndDisplayFile,
  2309. checkFolderExists
  2310. }
  2311. })()
  2312.  
  2313. // 将库暴露到全局作用域
  2314. unsafeWindow.WebDAVManager = WebDAVManager
  2315.  
  2316. // 示例用法:列出文件和文件夹,创建新文件夹,上传文件,删除文件
  2317. if (!url || !username || !password) {
  2318. console.error('WebDAV 配置更新失败:缺少 URL、用户名或密码')
  2319. return
  2320. }
  2321. WebDAVManager.updateConfig(webdavUrl, webdavUsername, webdavPassword)
  2322. WebDAVManager.createFolder(webdavfold)
  2323. WebDAVManager.checkFolderExists(webdavfold)
  2324.  
  2325. //WebDAVManager.uploadFile(webdavfold, 'nidie.txt', '这是 example.txt 的内容');
  2326. // WebDAVManager.deleteFile('Missav保存', 'example.txt');
  2327.  
  2328. })();
  2329.  
  2330.