-
-
Notifications
You must be signed in to change notification settings - Fork 38
/
missav-explorer.user.js
2330 lines (1998 loc) · 187 KB
/
missav-explorer.user.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
// ==UserScript==
// @name MissAv批量备份收藏视频
// @name:zh-CN MissAv批量备份收藏视频
// @namespace https://github.com/ChinaGodMan/UserScripts
// @version 1.2.3.73
// @description 从当前missav页面获取图片文件和视频信息,并合并结果后提供下载生成的网页文件
// @description:zh-CN 从当前missav页面获取图片文件和视频信息,并合并结果后提供下载生成的网页文件
// @license MIT
// @author 人民的勤务员 <[email protected]> & ChatGPT
// @match https://missav.com/*
// @grant GM_setValue
// @grant GM_getValue
// @grant GM_xmlhttpRequest
// @grant GM_download
// @grant GM_openInTab
// @icon 
// @iconbak https://pic.616pic.com/ys_bnew_img/00/35/79/Gv93yQh7v6.jpg
// @require https://cdnjs.cloudflare.com/ajax/libs/jszip/3.7.1/jszip.min.js
// @require https://update.greasyfork.org/scripts/498124/1396763/video.js
// @require https://update.greasyfork.org/scripts/498149/1395619/%E4%BF%A1%E6%81%AF%E6%9F%A5%E7%9C%8B%E5%99%A8.js
// @supportURL https://github.com/ChinaGodMan/UserScripts/issues
// @homepageURL https://github.com/ChinaGodMan/UserScripts
// ==/UserScript==
(function () {
'use strict'
var controlButton = createButton('备份', '10px', '10px')
var buttonA = createButton('备份当前', '70px', '10px')
var buttonB = createButton('备份片单', '130px', '10px')
var buttonC = createButton('设置选项', '190px', '10px')
var webdavbutton = createButton('WebDav', '250px', '10px')
// 设置按钮的背景颜色和样式
controlButton.style.backgroundColor = 'blue' // 控制按钮改为蓝色背景
buttonA.style.backgroundColor = 'green'
buttonB.style.backgroundColor = 'blue'
buttonC.style.backgroundColor = 'red'
webdavbutton.style.backgroundColor = 'blue'
// 隐藏初始的三个按钮
buttonA.style.display = 'none'
buttonB.style.display = 'none'
buttonC.style.display = 'none'
webdavbutton.style.display = 'none'
// 添加按钮到页面
document.body.appendChild(controlButton)
document.body.appendChild(buttonA)
document.body.appendChild(buttonB)
document.body.appendChild(buttonC)
document.body.appendChild(webdavbutton)
// 控制按钮的点击事件
controlButton.addEventListener('click', function () {
if (buttonA.style.display === 'none') {
// 显示三个按钮
buttonA.style.display = 'block'
buttonB.style.display = 'block'
buttonC.style.display = 'block'
webdavbutton.style.display = 'block'
controlButton.innerHTML = '隐藏'
} else {
// 隐藏三个按钮
buttonA.style.display = 'none'
buttonB.style.display = 'none'
buttonC.style.display = 'none'
webdavbutton.style.display = 'none'
controlButton.innerHTML = '备份'
}
})
webdavbutton.addEventListener('click', function () {
// 点击按钮时执行的操作
WebDAVManager.listFilesAndFolders(webdavfold)
})
// 按钮A的点击事件
buttonA.addEventListener('click', function () {
resetGlobalVariables()
singleFileDownload = true
window.showLogContainer()
var currentDate = new Date()
var currentTime = currentDate.getFullYear() + '-' + (currentDate.getMonth() + 1) + '-' + currentDate.getDate() + '_' + currentDate.getHours() + '-' + currentDate.getMinutes() + '-' + currentDate.getSeconds()
if (useDefaultTitle) {
name = document.querySelector('meta[name="twitter:title"]').content
} else {
const twitterTitleContent = document.querySelector('meta[name="twitter:title"]').content
name = prompt('请输入自定义名称:', twitterTitleContent)
if (name === null) {
name = twitterTitleContent
}
}
inurl = window.location.href
const defaultPages = getTotalPagesd()
const totalPages = setTotalPage(defaultPages)
allpages = totalPages
//const delay = settime();
if (totalPages) {
start(totalPages)
}
})
// 按钮B的点击事件
buttonB.addEventListener('click', function () {
// 点击按钮时执行的操作
resetGlobalVariables()
fetchJsonData()
})
// 按钮C的点击事件
buttonC.addEventListener('click', function () {
createSettingsUI()
// 这里可以添加按钮C点击后的具体操作,比如打开链接或执行其他动作
})
// 创建按钮的辅助函数
function createButton(text, top, left) {
var button = document.createElement('button')
button.innerHTML = text
button.style.position = 'fixed'
button.style.bottom = top
button.style.right = left
button.style.zIndex = '1000'
button.style.padding = '10px'
button.style.border = 'none'
button.style.cursor = 'pointer'
button.style.color = '#fff'
button.style.fontSize = '14px'
button.style.fontWeight = 'bold'
button.style.textAlign = 'center'
button.style.width = '100px' // 调整按钮宽度
return button
}
// 全局变量
var allResults = [] // 存储所有的结果数据
var zip = new JSZip() // 创建一个压缩文件实例
var allzip = new JSZip() // 另一个可能的压缩文件实例
var imgFolder = zip.folder('img') // 在 zip 中创建一个名为 "img" 的文件夹,用于存储图片文件
var allimgFolder = allzip.folder('img') // 在 allzip 中创建一个名为 "img" 的文件夹,可能用于另一个压缩文件的图片存储
var ALLfiledown = false // 标识是否所有文件已下载完毕的布尔变量
var videos = [] // 存储视频文件或相关信息的数组
var finalData = [] // 存储最终处理数据的数组
var inurl = '' // 当前下载地址的变量
var pendingRequests = 0 // 当前待处理的请求数量
var delayTime // 延迟时间,以毫秒为单位,用于控制异步操作的时间间隔
var currentPage = 1 // 当前处理的页数,可能用于分页处理或其他进度跟踪
var currentUrlIndex = 0 // 当前处理的 urls 数组中的索引位置
var name = '' // 当前下载的名称
var urls = [] // 存储需要处理的网址数组
var a = -1 // 循环中的计数或索引,初始值为 -1
var allZipContents = [] // 存储所有压缩文件内容的数组
var singleFileDownload = false // 标识是否为单个文件下载模式的布尔变量
var names = [] // 存储下载名称列表的数组
var allpages = 0 // 存储总页数或其他页面处理相关信息的变量
var modalContainer = null // 存储模态窗口容器的全局变量,用于显示下载进度或其他信息
var shouldReplace = false // 控制是否在下载大图时进行替换操作的布尔变量
var temporaryData = []
var saveJson = false
var useDefaultTitle = true
var pageCount = true
var saveVideoInfo = false
var saveImage = false
var downloadLog = {}
var errorLogs = {}
var downloadLogFileA = false // 这里设置为 true 时载日志
var webdavfold = 'missavsave'
//var webdavfold="1111";
var savetowebdav = false
var webdavUrl = ''
var webdavUsername = ''
var webdavPassword = ''
var deleteSelected = false
ini()//读取配置
function resetGlobalVariables() {
zip = new JSZip() // 重置为一个新的 JSZip 实例,用于创建新的压缩文件
allzip = new JSZip() // 可能是另一个新的 JSZip 实例,用于其他用途的压缩文件
if (saveImage) {
imgFolder = zip.folder('img') // 在 zip 中创建一个名为 "img" 的文件夹,用于存储图片文件
allimgFolder = allzip.folder('img') // 在 allzip 中创建一个名为 "img" 的文件夹,可能用于另一个压缩文件的图片存储
}
downloadLog = {}
errorLogs = {}
ALLfiledown = false // 重置为 false,表示所有文件未下载完毕
videos = [] // 清空存储视频文件或相关信息的数组
finalData = [] // 清空存储最终处理数据的数组
inurl = '' // 重置当前下载地址为空字符串
pendingRequests = 0 // 重置待处理的请求数量为 0
currentPage = 1 // 重置当前处理的页数为 1
currentUrlIndex = 0 // 重置当前处理的 urls 数组索引为 0
name = '' // 重置当前下载的名称为空字符串
urls = [] // 清空存储需要处理的网址数组
a = -1 // 重置循环中的计数或索引为 -1
allZipContents = [] // 清空存储所有压缩文件内容的数组
singleFileDownload = false // 重置为 false,表示不是单个文件下载模式
names = [] // 清空存储下载名称列表的数组
allpages = 0 // 重置存储总页数或其他页面处理相关信息的变量为 0
temporaryData = []
}
async function processUrls() {
//delayTime = 20;
let completedTasks = 0 // 计数已完成的任务数量
for (const url of urls) {
a = a + 1 // 每次循环递增 a
inurl = url
console.log('正在处理网址:', url, names[a])
window.addToLog('处理:' + url + names[a], 'info')
name = names[a]
try {
const totalPages = await getTotalPages(url) // 等待 getTotalPages 返回结果
console.log('Total pages for', url, ':', totalPages) // 显示总页数
window.addToLog(name + ' 总页数:' + url + totalPages, 'info')
allpages = totalPages
start(totalPages) // 启动处理流程
// 等待当前页面的请求完成
while (pendingRequests > 0) {
await new Promise(resolve => setTimeout(resolve, 100)) // 每隔 100 毫秒检查一次是否所有请求都已完成
}
completedTasks++ // 标记当前任务已完成
} catch (error) {
console.error('Error processing URL:', url, error) // 处理错误信息
allpages = 1
start(1) // 启动处理流程
while (pendingRequests > 0) {
await new Promise(resolve => setTimeout(resolve, 100)) // 每隔 100 毫秒检查一次是否所有请求都已完成
}
completedTasks++ // 标记当前任务已完成
}
}
// 如果所有任务都已完成且 urls 数组不为空,则调用下载函数
if (completedTasks === urls.length && urls.length !== 0) {
downloadAllZips()
}
}
function getAllCookies() {
return document.cookie
}
// 获取指定 JSON 数据的函数
function fetchJsonData() {
const cookies = getAllCookies()
//alert(cookies);
console.log('Current page cookies:', cookies)
// 构建 API URL
const apiUrl = 'https://missav.com/api/playlists/dfe-057'
// 发送带有 cookies 的请求
GM_xmlhttpRequest({
method: 'GET',
url: apiUrl,
headers: {
'Cookie': cookies
},
onload: function (response) {
if (response.status === 200) {
try {
const jsonResponse = JSON.parse(response.responseText)
if (jsonResponse && Array.isArray(jsonResponse.data)) {
createReportUI(jsonResponse.data, 500) // 假设每页显示 10 个项目
// 调用 processUrls 函数处理 URLs
} else {
console.error('JSON 格式无效')
showModal('JSON 格式无效', 2000)
}
} catch (error) {
console.error('Error parsing JSON:', error)
showModal('解析错误' + error, 2000)
}
} else {
console.error('Request failed with status:', response.status)
showModal('解析错误', 2000)
}
},
onerror: function (error) {
showModal('解析错误' + error, 2000)
}
})
}
function processUrl(url) {
// 检查是否包含 `page=` 参数
var pageIndex = url.indexOf('page=')
if (pageIndex !== -1) {
// 找到 `page=` 参数并删除它及其后的所有内容
var baseUrl = url.substring(0, pageIndex + 5) // +5 to include `page=`
return baseUrl
} else {
// 检查是否已有其他参数
if (url.includes('?')) {
// 有其他参数,添加 `&page=`
return url + '&page='
} else {
// 没有其他参数,添加 `?page=`
return url + '?page='
}
}
}
function settime() {
// 让用户输入延时时间
delayTime = prompt('请输入每页请求的延时时间(毫秒):', '1000')
// 检查用户是否取消输入
if (delayTime === null) {
alert('输入取消')
return
}
delayTime = parseInt(delayTime)
// 检查输入的延时时间是否有效
if (isNaN(delayTime) || delayTime <= 0) {
alert('请输入有效的延时时间(正整数)!')
return
}
// 返回有效的延时时间
return delayTime
}
function getTotalPagesd() { // 获取总页数
var totalPagesElement = document.querySelector('#price-currency')
var totalPagesText = totalPagesElement ? totalPagesElement.innerText : ''
var totalPages = parseInt(totalPagesText.replace('/', '').trim(), 10)
// 如果获取总页数失败,则返回 1
if (isNaN(totalPages) || totalPages <= 0) {
totalPages = 1
}
return totalPages
}
function getTotalPages(url) {
return new Promise((resolve, reject) => {
// 如果没有提供 URL,则使用当前页面的 URL
if (!url) {
url = window.location.href
}
// 发起 GM_xmlhttpRequest 请求获取页面内容
GM_xmlhttpRequest({
method: 'GET',
url: url,
headers: { 'Cookie': document.cookie },
onload: function (response) {
// 处理响应
if (response.status === 200) {
const parser = new DOMParser()
const doc = parser.parseFromString(response.responseText, 'text/html')
const totalPagesElement = doc.querySelector('#price-currency') // 替换为实际选择器
if (totalPagesElement) {
const totalPagesText = totalPagesElement.innerText
const totalPages = parseInt(totalPagesText.replace('/', '').trim(), 10)
resolve(totalPages) // 成功时返回总页数
} else {
window.addToLog('页面中没有找到总页数,默认为1页', 'warning')
reject('Total pages element not found') // 页面中没有找到总页数元素
}
} else {
window.addToLog('请求失败', 'warning')
reject(`Request failed with status ${response.status}`) // 请求失败
}
},
onerror: function () {
window.addToLog('请求出错', 'warning')
reject('Request failed') // 请求出错
}
})
})
}
// 设置总页数
function setTotalPage(defaultPages) {
if (!pageCount) {
return defaultPages
}
const inputPages = parseInt(prompt(`当前 ${name} 总页数为 ${defaultPages}。请输入你想要抓取的页数(不输入抓取全部):`, defaultPages), 10)
if (isNaN(inputPages) || inputPages <= 0) {
return defaultPages
}
return inputPages
}
// 开始处理页面抓取
function start(totalPages, callback) {
const pages = Array.from({ length: totalPages }, (_, i) => i + 1)
pendingRequests = pages.length
fetchPage(pages.shift(), pages, callback)
}
// 点击按钮时执行操作
// 异步获取页面内容
function fetchPage(pageNum, pages, callback) {
const pageUrl = `${processUrl(inurl)}${pageNum}`
console.log(`正在获取第 ${pageNum} 页的内容...`)
//showModal(`正在获取${name} 第 ${pageNum} / ${allpages}页 `);
if (a !== -1) {
showModal(`${a + 1}/${names.length} 正在获取 ${name} 第 ${pageNum} / ${allpages} 页`)
} else {
showModal(`正在获取 ${name} 第 ${pageNum} / ${allpages} 页`)
}
GM_xmlhttpRequest({
method: 'GET',
url: pageUrl,
headers: { 'Cookie': document.cookie },
onload: function (response) {
if (response.status === 200) {
processPageContent(response.responseText, pageNum, pages, callback)
} else {
pendingRequests--
checkIfComplete(callback)
if (pages.length > 0) {
setTimeout(() => fetchPage(pages.shift(), pages, callback), delayTime)
}
}
}
})
}
//获取视频信息
function extractInformation(htmlContent) {
let data = {} // 创建一个对象来存储提取的数据
let xhr = new XMLHttpRequest()
xhr.open('GET', htmlContent, false) // 同步方式打开请求
xhr.send()
// 创建一个虚拟的 <div> 元素来加载 HTML 内容
let tempDiv = document.createElement('div')
tempDiv.innerHTML = xhr.responseText
// 获取所有包含信息的父元素列表
let parentElements = tempDiv.querySelectorAll('div.space-y-2 > div')
if (parentElements.length > 0) {
let allInfo = {} // 初始化一个空对象来存储所有信息
// 遍历每个包含信息的 <div> 元素
parentElements.forEach(div => {
let span = div.querySelector('span') // 获取第一个 <span> 元素
if (span) {
let category = span.textContent.trim() // 获取主分类名称
if (!allInfo[category]) {
allInfo[category] = [] // 初始化一个空数组来存储该分类下的所有信息
}
// 查找所有的 <a> 元素和 <time> 元素
div.querySelectorAll('a, time').forEach(element => {
let info = {}
// 判断元素是否是 <a> 元素
if (element.tagName === 'A') {
info['name'] = element.textContent.trim() // 获取名称
info['link'] = element.href.trim() // 获取链接
} else if (element.tagName === 'TIME') {
info = element.textContent.trim() // 如果是 <time> 元素,则直接保存其文本内容
}
// 添加信息对象到相应的主分类数组中
allInfo[category].push(info)
})
// 如果没有找到 <a> 元素,则尝试获取 <span> 标签内的文本内容
if (div.querySelectorAll('a').length === 0) {
let spanText = div.querySelector('span.font-medium')
if (spanText) {
let info = spanText.textContent.trim()
allInfo[category].push(info)
}
}
}
})
// 提取 class="mb-1 text-secondary break-all line-clamp-2" 的内容
let descriptionElement = tempDiv.querySelector('.mb-1.text-secondary.break-all.line-clamp-2')
let descriptionContent = descriptionElement ? descriptionElement.textContent.trim() : ''
allInfo['简介'] = descriptionContent
// 将所有信息存储到 data 对象中
data['videosinfo'] = allInfo
// 查找包含 x-cloak 和 x-show="currentTab === 'magnets'" 的第二个元素
let secondElement = tempDiv.querySelector('div[x-cloak][x-show="currentTab === \'magnets\'"]')
if (secondElement) {
let linksAndInfo = []
// 遍历第二个元素内的 <a> 元素
secondElement.querySelectorAll('a[rel="nofollow"]').forEach(a => {
let linkInfo = {
name: a.textContent.trim(),
link: a.href.trim()
}
// 查找相邻的 <td> 元素,获取大小和日期信息
let sizeTd = a.closest('td').nextElementSibling
if (sizeTd && sizeTd.classList.contains('font-mono')) {
linkInfo['size'] = sizeTd.textContent.trim() // 获取大小信息
}
let dateTd = sizeTd ? sizeTd.nextElementSibling : null
if (dateTd && dateTd.classList.contains('hidden')) {
linkInfo['date'] = dateTd.textContent.trim() // 获取日期信息
}
let nextSibling = a.nextElementSibling
// 循环处理所有紧邻的<span>元素
while (nextSibling && nextSibling.tagName === 'SPAN') {
let spanText = nextSibling.textContent.trim()
linkInfo['name'] += ' ' + spanText // 将<span>元素的文本内容追加到name中
nextSibling = nextSibling.nextElementSibling // 继续查找下一个兄弟元素
}
linksAndInfo.push(linkInfo)
})
// 将第二个元素的链接和信息添加到 data 中
data['secondElementLinksInfo'] = linksAndInfo
} else {
console.error('未找到包含 x-cloak 和 x-show="currentTab === \'magnets\'" 的第二个元素。')
}
return data // 返回结构化的数据
} else {
console.error('未找到匹配的父元素 div.space-y-2')
return null // 如果未找到匹配的父元素,返回 null
}
} ///大
// 使用XMLHttpRequest获取页面内容
function fetchPageforinfo(url) {
let xhr = new XMLHttpRequest()
xhr.onreadystatechange = function () {
if (xhr.readyState === XMLHttpRequest.DONE) {
if (xhr.status === 200) {
extractInformation(xhr.responseText) // 将获取的页面内容传递给提取信息的函数
} else {
console.error('请求失败:' + xhr.status)
}
}
}
xhr.open('GET', url, true)
xhr.send()
}
// 处理获取到的页面内容
function processPageContent(htmlContent, pageNum, pages, callback) {
const parser = new DOMParser()
const doc = parser.parseFromString(htmlContent, 'text/html')
const divElements = doc.querySelectorAll('div.relative.aspect-w-16.aspect-h-9.rounded.overflow-hidden.shadow-lg')
const logEntry = {
url: `${processUrl(inurl)}${pageNum}`,
elementsFetched: divElements.length
}
// 如果当前名称的日志组不存在,则创建一个新数组
if (!downloadLog[name]) {
downloadLog[name] = []
}
// 将日志条目添加到日志数组中
downloadLog[name].push(logEntry)
if (divElements.length === 0) {
const logEntry = {
url: `${processUrl(inurl)}${pageNum}`,
elementsFetched: 0, // 这里可以根据实际需求设置其他信息
errorMessage: `获取第 ${pageNum} 页失败。`
}
if (!errorLogs[name]) {
errorLogs[name] = []
}
errorLogs[name].push(logEntry)
console.log(`获取第 ${pageNum} 页失败。`)
window.addToLog(`${name}${processUrl(inurl)}${pageNum}+获取失败 数量:` + divElements.length, 'error')
}
divElements.forEach(div => {
var imgUrl = div.querySelector('img').getAttribute('data-src')
if (shouldReplace) {
imgUrl = imgUrl.replace('cover-t.jpg', 'cover-n.jpg')
}
const video = {
fileName: div.querySelector('a').getAttribute('alt'),
imgUrl: imgUrl,
videoUrl: div.querySelector('video').getAttribute('data-src'),
markContent: Array.from(div.querySelectorAll('span')).map(mark => mark.textContent).join(' '),
altText: div.querySelector('img').getAttribute('alt'),
jumpUrl: div.querySelector('a').getAttribute('href')
}
if (saveVideoInfo) {
video.info = extractInformation(video.jumpUrl)
//showBanner(`正在获取 ${video.fileName} 信息`);
window.addToLog(`正在获取 ${video.fileName} 信息`, 'info')
console.log()
}
if (video.imgUrl && video.altText) {
videos.push(video)
if (saveImage) {
window.addToLog('保存' + video.imgUrl, 'info')
pendingRequests++
GM_xmlhttpRequest({
method: 'GET',
url: video.imgUrl,
responseType: 'blob',
onload: function (response) {
if (response.status === 200) {
if (saveImage) {
if (singleFileDownload) {
console.log('这是单个文件下载')
imgFolder.file(`${video.fileName}.jpg`, response.response, { binary: true })
} else {
console.log('这是批量文件下载')
allimgFolder.file(`${video.fileName}.jpg`, response.response, { binary: true })
}
}
pendingRequests--
checkIfComplete(callback)
} else {
pendingRequests--
checkIfComplete(callback)
}
}
})
}
} else {
pendingRequests--
checkIfComplete(callback)
}
})
showModal(`获取第 ${pageNum} 页的内容完成,等待 ${delayTime} 毫秒加载第 ${pageNum + 1} 页。`)
pendingRequests--
checkIfComplete(callback)
if (pages.length > 0) {
setTimeout(() => fetchPage(pages.shift(), pages, callback), delayTime)
} else {
}
}
closeModal()
function downloadLogFile() {
if (!downloadLogFileA) {
console.log('日志下载已被跳过')
return
}
if (Object.keys(errorLogs).length === 0) {
// 如果错误日志为空,直接下载正常日志文件
const logBlob = new Blob([JSON.stringify(downloadLog, null, 4)], { type: 'application/json' })
const logUrl = URL.createObjectURL(logBlob)
const logLink = document.createElement('a')
logLink.href = logUrl
logLink.download = 'download_log.json'
logLink.click()
URL.revokeObjectURL(logUrl)
} else {
// 创建一个JSZip实例
const zip = new JSZip()
// 添加正常日志文件到压缩包
const logBlob = new Blob([JSON.stringify(downloadLog, null, 4)], { type: 'application/json' })
zip.file('download_log.json', logBlob)
// 添加错误日志文件到压缩包
const errorLogBlob = new Blob([JSON.stringify(errorLogs, null, 4)], { type: 'application/json' })
zip.file('error_log.json', errorLogBlob)
// 生成压缩包并触发下载
zip.generateAsync({ type: 'blob' }).then(function (content) {
const zipUrl = URL.createObjectURL(content)
const link = document.createElement('a')
link.href = zipUrl
link.download = 'logs.zip'
link.click()
URL.revokeObjectURL(zipUrl)
})
}
}
function sanitizeFileName(name) {
return name.replace(/[\\/:*?"<>|]/g, '_')
}
function checkIfComplete(callback) {
if (pendingRequests === 0) {
const additionalInfo = {
timestamp: new Date().toISOString(),
inurl: inurl
}
if (singleFileDownload) {
showModal('获取完毕,正在生成单个文件...')
finalData = {
info: additionalInfo,
video: videos
}
if (saveJson) {
zip.file('data.json', JSON.stringify(finalData, null, 4))
}
if (savetowebdav) {
WebDAVManager.uploadFile(webdavfold, `${sanitizeFileName(name)}.json`, JSON.stringify(finalData, null, 4))
}
const jsonIndexContent = generateJsonIndexContent(finalData)
const numFiles = Object.keys(zip.files).length // 获取压缩包中文件的数量
if (numFiles === 0) {
const htmlContent = jsonIndexContent // 替换为实际的HTML内容
const htmlBlob = new Blob([htmlContent], { type: 'text/html' })
const htmlUrl = URL.createObjectURL(htmlBlob)
const a = document.createElement('a')
a.href = htmlUrl
a.download = `${sanitizeFileName(name)}.html`
a.click()
closeModal()
downloadLogFile()
if (callback) callback()
} else {
zip.file(`${sanitizeFileName(name)}.html`, jsonIndexContent)
// 生成并下载单个文件
zip.generateAsync({ type: 'blob' }, function updateCallback(metadata) {
const progress = metadata.percent.toFixed(2)
showModal(`压缩进度: ${progress}%`)
}).then(content => {
const zipUrl = URL.createObjectURL(content)
const a = document.createElement('a')
a.href = zipUrl
a.download = `${name}.zip`
a.click()
URL.revokeObjectURL(zipUrl)
closeModal()
downloadLogFile()
if (callback) callback()
})
}
} else {
finalData = {
info: additionalInfo,
video: videos
}
if (saveJson) {
allzip.file(`${sanitizeFileName(name)}.json`, JSON.stringify(finalData, null, 4))
}
if (savetowebdav) {
WebDAVManager.uploadFile(webdavfold, `${sanitizeFileName(name)}.json`, JSON.stringify(finalData, null, 4))
}
const jsonIndexContent = generateJsonIndexContent(finalData);
allzip.file(`${sanitizeFileName(name)}.html`, jsonIndexContent)
finalData = []
videos = []
if (callback) callback()
}
}
}
function downloadAllZips() {
if (singleFileDownload === false) {
showModal('获取完毕,正在生成压缩文件...')
const numFiles = Object.keys(allzip.files).length // 获取压缩包中文件的数量
if (numFiles === 1) {
// 如果压缩包中只有一个文件,直接处理该文件
const fileName = Object.keys(allzip.files)[0] // 获取唯一的文件名
const file = allzip.files[fileName]
// 根据文件类型获取文件内容
file.async('blob').then(content => {
// 创建一个Blob对象,并下载
const blob = new Blob([content])
const url = URL.createObjectURL(blob)
const a = document.createElement('a')
a.href = url
a.download = fileName
document.body.appendChild(a) // 添加到文档中以确保点击有效
a.click()
document.body.removeChild(a) // 下载完成后移除元素
URL.revokeObjectURL(url)
closeModal()
}).catch(error => {
console.error('Error fetching file content:', error)
closeModal()
})
downloadLogFile()
return // 结束函数执行,不生成压缩包
}
allzip.generateAsync({ type: 'blob' }, function updateCallback(metadata) {
const progress = metadata.percent.toFixed(2)
showModal(`压缩进度: ${progress}%`)
}).then(content => {
const zipUrl = URL.createObjectURL(content)
const a = document.createElement('a')
a.href = zipUrl
a.download = `批量备份${urls.length}个片单.zip`
a.click()
URL.revokeObjectURL(zipUrl)
closeModal()
downloadLogFile()
if (callback) callback()
})
// 如果 singleFileDownload 等于假,则执行这里的代码
}
}
function showBanner(text) {
// 查找现有的横幅元素
var existingBanner = document.querySelector('.banner')
if (existingBanner) {
// 如果横幅已经存在,直接更新文本内容
existingBanner.textContent = text
} else {
// 如果横幅不存在,创建一个新的横幅
var banner = document.createElement('div')
banner.className = 'banner' // 添加一个类名以便识别
banner.style.position = 'fixed'
banner.style.bottom = '20px' // 距离底部的距离
banner.style.left = '20px' // 距离左侧的距离
banner.style.width = 'auto' // 根据文本自动调整宽度
banner.style.backgroundColor = 'rgba(255, 255, 255, 0.9)'
banner.style.color = '#000' // 黑色文本
banner.style.textAlign = 'center'
banner.style.padding = '20px'
banner.style.borderRadius = '8px'
banner.style.boxShadow = '0 4px 8px rgba(0, 0, 0, 0.1)'
banner.style.zIndex = '9999'
banner.textContent = text // 将传入的文本设置为横幅内容
document.body.appendChild(banner) // 将横幅添加到文档的末尾
// 3秒后移除横幅提示
setTimeout(function () {
banner.remove()
}, 3000)
}
}
// 创建或更新模态窗口
function showModal(message, autoCloseDelay = 0) {
// 如果模态窗口不存在,则创建新的模态窗口
if (!modalContainer) {
modalContainer = document.createElement('div')
modalContainer.className = 'modal-container'
modalContainer.style.cssText = `
position: fixed;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
background-color: rgba(255, 255, 255, 0.9);
border-radius: 8px;
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1);
z-index: 9999;
padding: 20px;
`
document.body.appendChild(modalContainer)
}
// 更新模态窗口的内容
modalContainer.textContent = message
// 自动关闭模态窗口
if (autoCloseDelay > 0) {
setTimeout(closeModal, autoCloseDelay)
}
}
// 关闭模态窗口
function closeModal() {
// 如果模态窗口存在,则从 DOM 中移除
if (modalContainer) {
document.body.removeChild(modalContainer)
modalContainer = null // 将变量重置为 null,以便下次创建新的模态窗口
}
}
// 创建JSONindex
function createReportUI(data, itemsPerPage) {
temporaryData = data
// 创建全屏遮罩层
const overlay = document.createElement('div')
overlay.className = 'overlay'
overlay.style.cssText = `
position: fixed;
top: 0;
left: 0;
width: 100%;
height: 100%;
background-color: rgba(0, 0, 0, 1); /* 全黑不透明背景 */
z-index: 9999; /* 确保遮罩层位于所有内容之上 */
`
// document.body.appendChild(overlay);
const modalContainer = document.createElement('div')
modalContainer.className = 'modal-container'
modalContainer.style.cssText = `
position: fixed;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
background-color: rgba(255, 255, 255, 1);
border-radius: 8px;
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.2);
z-index: 10000; /* 确保弹出框位于遮罩层之上 */
padding: 20px;
width: 80%;
max-width: 800px;
`
const title = document.createElement('h2')
title.textContent = `当前共有片单数量: ${temporaryData.length}`
title.style.textAlign = 'center'
modalContainer.appendChild(title)
const closeButton = document.createElement('button')
closeButton.textContent = '×'
closeButton.style.position = 'absolute'
closeButton.style.top = '10px'
closeButton.style.right = '10px'
closeButton.style.backgroundColor = 'transparent'
closeButton.style.border = 'none'
closeButton.style.fontSize = '24px'
closeButton.style.cursor = 'pointer'
modalContainer.appendChild(closeButton)
closeButton.addEventListener('click', () => {
// document.body.removeChild(overlay); // 移除遮罩层
document.body.removeChild(modalContainer) // 移除模态框
})