@lzwme/m3u8-dl 1.6.0-0 → 1.6.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/client/index.html CHANGED
@@ -4,7 +4,7 @@
4
4
  <meta charset="UTF-8">
5
5
  <meta http-equiv="X-UA-Compatible" content="IE=edge">
6
6
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
7
- <meta name="apple-mobile-web-app-capable" content="yes">
7
+ <meta name="mobile-web-app-capable" content="yes">
8
8
  <title>M3U8 下载器</title>
9
9
  <link rel="icon" type="image/png" href="/logo.png">
10
10
  <link rel="stylesheet" href="https://s4.zstatic.net/ajax/libs/font-awesome/6.7.2/css/all.min.css"
@@ -12,8 +12,8 @@
12
12
  crossorigin="anonymous" referrerpolicy="no-referrer" />
13
13
  <script src="https://s4.zstatic.net/ajax/libs/blueimp-md5/2.19.0/js/md5.min.js" crossorigin="anonymous"
14
14
  referrerpolicy="no-referrer"></script>
15
- <script type="module" crossorigin src="/assets/main-XL0wiaDU.js"></script>
16
- <link rel="stylesheet" crossorigin href="/assets/main-DYJAIw1q.css">
15
+ <script type="module" crossorigin src="/assets/main-DZTEqg-V.js"></script>
16
+ <link rel="stylesheet" crossorigin href="/assets/main-ChJ1yjNN.css">
17
17
  </head>
18
18
  <body>
19
19
  <div id="app"></div>
@@ -0,0 +1,94 @@
1
+ // ==UserScript==
2
+ // @name [M3U8-DL]媒体链接抓取器
3
+ // @namespace https://github.com/lzwme/m3u8-dl
4
+ // @homepage https://m3u8-player.lzw.me/download.html
5
+ // @supportURL https://github.com/lzwme/m3u8-dl/issues
6
+ // @icon https://gh-proxy.org/raw.githubusercontent.com/lzwme/m3u8-dl/refs/heads/main/packages/frontend/public/logo.png
7
+ // @version 1.0.0
8
+ // @description 自动抓取网页中的多种媒体链接(m3u8、mp4、mkv、avi、mov、音频等),支持可配置的媒体类型,支持跳转到 m3u8-dl webui 下载
9
+ // @author lzw
10
+ // @updateURL https://gh-proxy.org/raw.githubusercontent.com/lzwme/m3u8-dl/refs/heads/main/client/m3u8-capture.user.js
11
+ // @downloadURL https://raw.githubusercontent.com/lzwme/m3u8-dl/refs/heads/main/client/m3u8-capture.user.js
12
+ // @match *://*/*
13
+ // @grant GM_addElement
14
+ // @grant GM_setValue
15
+ // @grant GM_getValue
16
+ // @grant GM_getResourceText
17
+ // @grant GM_xmlhttpRequest
18
+ // @grant unsafeWindow
19
+ // @resource SwalJS https://s4.zstatic.net/ajax/libs/sweetalert2/11.16.1/sweetalert2.min.js
20
+ // @resource SwalCSS https://s4.zstatic.net/ajax/libs/sweetalert2/11.16.1/sweetalert2.css
21
+ // @resource TailwindCSS https://s4.zstatic.net/ajax/libs/tailwindcss/2.2.19/tailwind.min.css
22
+ // @run-at document-idle
23
+ // ==/UserScript==
24
+
25
+ (function(){"use strict";const j="m3u8_capture_webui_url",K="m3u8_capture_exclude_urls",J="m3u8_capture_panel_pos",Q="m3u8_capture_panel_visible",Z="m3u8_capture_media_ext_list",ut=["m3u8","mp4","mkv","avi","mov","wmv","flv","webm","m4v","m3u","m4a","aac","flac","ape","mp3","wav","ogg","wma"];function V(){return GM_getValue(j,"http://localhost:6600").replace(/\/$/,"")}function tt(){return GM_getValue(K,"")}function ft(t){GM_setValue(K,t)}function U(){const t=GM_getValue(Z,[]);return t&&Array.isArray(t)&&t.length>0?t:[...ut]}function pt(t){if(Array.isArray(t)&&t.length>0){const e=t.map(n=>n.trim().toLowerCase()).filter(n=>n&&/^[a-z0-9]+$/i.test(n));return GM_setValue(Z,e),e}return null}function mt(){return GM_getValue(J,null)}function xt(t){GM_setValue(J,t)}function wt(){return GM_getValue(Q,!1)}function et(t){GM_setValue(Q,t)}function W(t,e=document.head,n="css"){const o=t.length<50?GM_getResourceText(t):t;return Promise.resolve(GM_addElement(e,n==="css"?"style":"script",{type:n==="css"?"text/css":"text/javascript",textContent:o}))}function gt(){const t=U();return new RegExp(`\\.(${t.join("|")})(\\?|$|#)`,"i")}function k(t){const e=window.location.href;if(e.startsWith(V()))return!0;const n=tt();if(!n||!n.trim())return!1;const o=n.split(`
26
+ `).map(i=>i.trim()).filter(i=>i);for(const i of o)try{if(e.includes(i)||i.startsWith("/")&&i.endsWith("/")&&new RegExp(i.slice(1,-1)).test(e))return!0}catch(a){console.warn("[M3U8 Capture] 排除规则格式错误:",i,a)}return!1}function nt(t){return new Promise((e,n)=>{navigator.clipboard?.writeText?navigator.clipboard.writeText(t).then(()=>e(!0)).catch(()=>{ot(t)?e(!0):n(new Error("复制失败"))}):ot(t)?e(!0):n(new Error("复制失败"))})}function ot(t){try{const e=document.createElement("textarea");e.value=t,e.style.position="fixed",e.style.opacity="0",e.style.left="-9999px",document.body.appendChild(e),e.select(),e.setSelectionRange(0,t.length);const n=document.execCommand("copy");return document.body.removeChild(e),n}catch(e){return console.error("[M3U8 Capture] fallbackCopy failed:",e),!1}}function ht(t){try{const e=new URL(t);return`${e.protocol}//${e.host}${e.pathname}`}catch{return t}}function O(t){return"touches"in t&&t.touches.length>0?{x:t.touches[0].clientX,y:t.touches[0].clientY}:"changedTouches"in t&&t.changedTouches.length>0?{x:t.changedTouches[0].clientX,y:t.changedTouches[0].clientY}:{x:t.clientX,y:t.clientY}}function y(t){if(!t||typeof t!="string"||!t.startsWith("http"))return!1;const e=t.toLowerCase();return!!(gt().test(e)||e.includes(".m3u8"))}function it(t){if(!t||typeof t!="string")return"media";const e=t.toLowerCase(),n=U();for(const o of n)if(new RegExp(`\\.${o}(\\?|$|#)`,"i").test(e))return o;return/m3u8/i.test(e)?"m3u8":"media"}function q(t=document){let e="";const n=["h1.title","h2.title","h1","h2"];for(const o of n){const i=t.querySelector(o);if(i?.textContent){e=i.textContent.trim();break}}if(e||(e=(t.title||"").split(/[-|_]/)[0].trim()),!e&&window.top!==window.self)try{e=q(window.top?.document)}catch{}return e}function yt(t){if(!t||typeof t!="string")return null;try{const e=new URL(t);for(const[n,o]of e.searchParams.entries()){if(!o)continue;const i=decodeURIComponent(o);if(y(i))return i}}catch{}return null}function bt(){const t=XMLHttpRequest.prototype.open;XMLHttpRequest.prototype.open=function(n,o,i,a,s){return y(o.toString())&&this.addEventListener("load",function(){this.status>=200&&this.status<300&&h(o.toString())}),t.call(this,n,o,i??!0,a,s)};const e=window.fetch;window.fetch=function(n,o){const i=typeof n=="string"?n:n&&typeof n=="object"&&"url"in n?n.url:"";return y(i)?e.call(this,n,o).then(a=>(a.ok&&h(i),a)):e.call(this,n,o)}}function vt(){if(typeof PerformanceObserver>"u")return;const t=new PerformanceObserver(e=>{for(const n of e.getEntries())n.name&&y(n.name)&&h(n.name)});try{t.observe({entryTypes:["resource"]})}catch{console.log("[M3U8 Capture] PerformanceObserver not supported")}}function A(){if(document.querySelectorAll("video").forEach(t=>{t.src&&y(t.src)&&h(t.src,t.getAttribute("title")||""),t.querySelectorAll("source").forEach(e=>{e.src&&y(e.src)&&h(e.src,t.getAttribute("title")||"")})}),document.querySelectorAll("a[href]").forEach(t=>{const e=t.getAttribute("href");if(e&&y(e))try{const n=new URL(e,window.location.href).href;h(n,t.textContent?.trim()||"")}catch{}}),document.body){const t=document.body.innerText||"",n=U().join("|"),o=new RegExp(`https?:\\/\\/[^\\s"'<>]+\\.(${n})(\\?[^\\s"'<>]*)?`,"gi");let i;for(;(i=o.exec(t))!==null;)h(i[0])}}async function Et(){let t=document.body;const e=o=>{t=o},n=()=>t;unsafeWindow.SetSwalTarget=e,window.SetSwalTarget=e,unsafeWindow.GetSwalTarget=n,window.GetSwalTarget=n;try{const o=GM_getResourceText("SwalJS").replace(/document\.body/g,"GetSwalTarget()"),i=()=>W(o,document.head||document.documentElement,"script").then(()=>{const a=window.Swal||window.Sweetalert2||unsafeWindow.Swal||unsafeWindow.Sweetalert2;a&&(window.Swal=a,unsafeWindow.Swal=a)}).catch(a=>{console.error("[M3U8 Capture] Failed to add SweetAlert2 script:",a)});document.head||document.documentElement?await i():document.readyState==="loading"?document.addEventListener("DOMContentLoaded",()=>i()):setTimeout(i,50)}catch(o){console.error("[M3U8 Capture] Failed to load SweetAlert2:",o)}}function St(t,e){W(GM_getResourceText("SwalCSS").replace(/:root *{/,`#${e.id} {`).replace(/body/g,""),t,"css"),setTimeout(()=>{const n=window.Swal;typeof window.SetSwalTarget=="function"&&n&&(window.SetSwalTarget(e),window.Swal=n.mixin({target:e}))},500)}const m=window.top&&window.top!==window.self;let S=null,b=null,l=null,r=null,p=null,$=wt(),B=!1;const P={x:0,y:0};let M=!1;const R={x:0,y:0};let z={x:0,y:0},C=!1,L=null;const g={x:0,y:0};let x;function Mt(t){x=t}function rt(){if(S)return b;S=document.createElement("div"),S.id="m3u8-capture-shadow-host",S.style.cssText="position: fixed; top: 0; left: 0; width: 100%; height: 100%; pointer-events: none; z-index: 9999;",document.body.appendChild(S),b=S.attachShadow({mode:"open"}),W("TailwindCSS",b,"css");const t=document.createElement("div");t.id="m3u8-capture-swal-container",b.appendChild(t);const e=document.createElement("style");return e.textContent=[":host { all: initial; font-family: system-ui, -apple-system, sans-serif; }","* { box-sizing: border-box; }",".hidden { display: none !important; }",`#${t.id},`,`#${t.id} * { pointer-events: auto !important; }`].join(`
27
+ `),b.appendChild(e),Et().then(()=>{St(b,t)}),b}function at(){if(r||m)return;const t=rt();if(!t)return;r=document.createElement("div"),r.id="m3u8-capture-toggle-btn",r.style.cssText="position: fixed; bottom: 40px; right: 20px; width: 50px; height: 50px; pointer-events: auto; z-index: 999998; will-change: transform;",r.className=`fixed bottom-10 right-5 w-[50px] h-[50px] bg-blue-500 rounded-full flex items-center justify-center cursor-move shadow-lg text-2xl transition-all duration-200 hover:scale-110 hover:shadow-xl select-none touch-none ${$?"hidden":"flex"}`;const e=document.createElement("span");e.textContent="🎬",r.appendChild(e),p=document.createElement("span"),p.id="m3u8-capture-toggle-badge",p.style.cssText="position: absolute; top: -4px; right: -4px; min-width: 18px; height: 18px; background: #ef4444; color: white; border-radius: 9px; font-size: 11px; font-weight: bold; display: flex; align-items: center; justify-content: center; padding: 0 4px; box-shadow: 0 2px 4px rgba(0,0,0,0.2); line-height: 1;",p.textContent="0",p.classList.add("hidden"),r.appendChild(p);const n=o=>{if(!r)return;M=!0,C=!1;const i=O(o);z={x:i.x,y:i.y};const a=r.getBoundingClientRect();R.x=i.x-a.left,R.y=i.y-a.top,g.x=a.left,g.y=a.top;const s=window.getComputedStyle(r);s.transform&&s.transform!=="none"&&(r.style.transform="none",r.style.left=`${a.left}px`,r.style.top=`${a.top}px`,r.style.right="auto",r.style.bottom="auto"),r.style.cursor="move",r.style.transition="none",o.preventDefault(),o.stopPropagation()};r.addEventListener("mousedown",n),r.addEventListener("touchstart",n,{passive:!1}),r.addEventListener("click",()=>{C||st()}),r.addEventListener("touchend",o=>{M&&!C&&(o.preventDefault(),o.stopPropagation(),st())},{passive:!1}),t.appendChild(r),F()}function F(){if(!p||m||!x)return;const t=x.size;t>0?(p.textContent=t>99?"99+":t.toString(),p.classList.remove("hidden")):p.classList.add("hidden")}function st(){m||!ct()||($=!0,et(!0),l&&(l.style.display="flex"),r&&r.classList.add("hidden"))}function H(){m||($=!1,et(!1),l&&(l.style.display="none"),r?r.classList.remove("hidden"):at(),F())}function lt(){const t=window.Swal;t&&t.fire({title:"确认清空",text:"确定要清空所有媒体链接吗?",icon:"warning",showCancelButton:!0,confirmButtonText:"确定",cancelButtonText:"取消",confirmButtonColor:"#3b82f6"}).then(e=>{e.isConfirmed&&x&&(x.clear(),N())})}function ct(){if(!document.body||m)return null;if(l)return l;const t=rt();if(!t)return null;const e=document.createElement("div");e.id="m3u8-capture-panel";const n=mt(),o=n&&window.innerWidth>768?{left:`${Math.max(0,Math.min(n.x,window.innerWidth-450))}px`,top:`${Math.max(0,Math.min(n.y,window.innerHeight-350))}px`,right:"auto"}:{right:"20px",top:"20px"};e.className="fixed w-[420px] max-w-[90vw] max-h-[85vh] bg-white border-2 border-blue-500 rounded-xl shadow-2xl font-sans flex flex-col",e.style.cssText=`
28
+ position: fixed;
29
+ width: 420px;
30
+ max-width: 90vw;
31
+ max-height: 85vh;
32
+ pointer-events: auto;
33
+ z-index: 1059;
34
+ ${o.left?`left: ${o.left};`:""}
35
+ ${o.top?`top: ${o.top};`:""}
36
+ ${`right: ${o.right};`}
37
+ display: ${$?"flex":"none"};
38
+ `,e.innerHTML=`
39
+ <div id="m3u8-capture-header" class="bg-gradient-to-br from-blue-500 to-blue-600 text-white px-4 py-3.5 rounded-t-lg flex justify-between items-center cursor-move select-none touch-none">
40
+ <div class="font-semibold text-[15px] flex items-center gap-2">
41
+ <span>🎬</span>
42
+ <span>媒体链接抓取器</span>
43
+ <span id="m3u8-capture-count" class="bg-white bg-opacity-25 px-2 py-0.5 rounded-xl text-xs font-medium">0</span>
44
+ </div>
45
+ <div class="flex gap-1.5">
46
+ <button id="m3u8-capture-settings" class="bg-white bg-opacity-20 border-none text-white px-2.5 py-1.5 rounded-md cursor-pointer text-xs transition-colors duration-200 hover:bg-opacity-30 active:bg-opacity-40 touch-manipulation" title="设置">⚙️</button>
47
+ <button id="m3u8-capture-toggle" class="bg-white bg-opacity-20 border-none text-white px-2.5 py-1.5 rounded-md cursor-pointer text-xs transition-colors duration-200 hover:bg-opacity-30 active:bg-opacity-40 touch-manipulation" title="隐藏">−</button>
48
+ <button id="m3u8-capture-clear" class="bg-white bg-opacity-20 border-none text-white px-2.5 py-1.5 rounded-md cursor-pointer text-xs transition-colors duration-200 hover:bg-opacity-30 active:bg-opacity-40 touch-manipulation" title="清空">🗑️</button>
49
+ </div>
50
+ </div>
51
+ <div id="m3u8-capture-content" class="p-3 overflow-y-auto flex-1 bg-gray-50">
52
+ <div id="m3u8-capture-list" class="flex flex-col gap-2.5"></div>
53
+ <div id="m3u8-capture-empty" class="text-center text-gray-400 py-10 px-5 hidden">
54
+ <div class="text-5xl mb-3">📹</div>
55
+ <div class="text-sm">暂无媒体链接</div>
56
+ <div class="text-xs text-gray-300 mt-2">浏览网页时会自动抓取</div>
57
+ </div>
58
+ </div>
59
+ `,t.appendChild(e),l=e;const i=e.querySelector("#m3u8-capture-header");if(!i)return l;const a=u=>{const f=u.target;if(f.tagName==="BUTTON"||f.closest("button"))return;B=!0;const T=O(u),_=e.getBoundingClientRect();P.x=T.x-_.left,P.y=T.y-_.top,e.style.cursor="move",u.preventDefault(),u.stopPropagation()};i.addEventListener("mousedown",a),i.addEventListener("touchstart",a,{passive:!1});const s=u=>{const f=O(u);if(B&&l){u.preventDefault();const T=f.x-P.x,_=f.y-P.y,X=window.innerWidth-e.offsetWidth,Y=window.innerHeight-e.offsetHeight,D=Math.max(0,Math.min(T,X)),G=Math.max(0,Math.min(_,Y));e.style.left=`${D}px`,e.style.top=`${G}px`,e.style.right="auto",xt({x:D,y:G})}if(M&&r){u.preventDefault();const T=f.x-R.x,_=f.y-R.y,X=window.innerWidth-r.offsetWidth,Y=window.innerHeight-r.offsetHeight,D=Math.max(0,Math.min(T,X)),G=Math.max(0,Math.min(_,Y));g.x=D,g.y=G,L||(L=requestAnimationFrame(()=>{r&&M&&(r.style.transform=`translate(${g.x}px, ${g.y}px)`,r.style.left="0",r.style.top="0",r.style.right="auto",r.style.bottom="auto"),L=null})),Math.sqrt((f.x-z.x)**2+(f.y-z.y)**2)>5&&(C=!0)}},d=u=>{B&&(B=!1,l&&(l.style.cursor="default"),u.preventDefault()),M&&(M=!1,L&&(cancelAnimationFrame(L),L=null),r&&(r.style.cursor="move",r.style.transition="",C&&(r.style.transform=`translate(${g.x}px, ${g.y}px)`,r.style.left="0",r.style.top="0",r.style.right="auto",r.style.bottom="auto")),C&&u.preventDefault())};document.addEventListener("mousemove",s),document.addEventListener("mouseup",d),document.addEventListener("touchmove",s,{passive:!1}),document.addEventListener("touchend",d,{passive:!1}),document.addEventListener("touchcancel",d,{passive:!1});const c=u=>f=>{f.preventDefault(),f.stopPropagation(),u()},v=e.querySelector("#m3u8-capture-toggle");v&&(v.addEventListener("click",c(()=>H())),v.addEventListener("touchend",c(()=>H()),{passive:!1}));const w=e.querySelector("#m3u8-capture-clear");w&&(w.addEventListener("click",c(()=>lt())),w.addEventListener("touchend",c(()=>lt()),{passive:!1}));const E=e.querySelector("#m3u8-capture-settings");return E&&(E.addEventListener("click",c(()=>dt())),E.addEventListener("touchend",c(()=>dt()),{passive:!1})),!$&&!r&&at(),l}function dt(){const t=window.Swal;if(!t)return;const e=tt(),n=U();t.fire({title:"设置",html:`
60
+ <div class="text-left">
61
+ <label class="block text-sm font-medium text-gray-700 mb-1">WebUI 地址</label>
62
+ <input id="swal-webui-url" type="text" value="${V()}"
63
+ class="w-full p-2.5 border border-gray-300 rounded-md mb-4 focus:outline-none focus:ring-2 focus:ring-blue-500"
64
+ placeholder="http://localhost:6600">
65
+ <label class="block text-sm font-medium text-gray-700 mb-1">媒体扩展名(每行一个,用逗号或换行分隔)</label>
66
+ <textarea id="swal-media-ext-list" rows="3"
67
+ class="w-full p-2.5 border border-gray-300 rounded-md mb-4 focus:outline-none focus:ring-2 focus:ring-blue-500"
68
+ placeholder="例如:m3u8, mp4, mkv, avi, mov, wmv, flv, webm, m4v, ts, m3u, m4a, aac, flac, ape, mp3, wav, ogg, wma">${n.join(", ")}</textarea>
69
+ <p class="text-xs text-gray-500 mb-4">支持的媒体文件扩展名,将用于识别和抓取媒体链接</p>
70
+ <label class="block text-sm font-medium text-gray-700 mb-1">排除网址规则(每行一个,支持正则表达式,以 / 开头和结尾)</label>
71
+ <textarea id="swal-exclude-urls" rows="6"
72
+ class="w-full p-2.5 border border-gray-300 rounded-md focus:outline-none focus:ring-2 focus:ring-blue-500"
73
+ placeholder="例如:&#10;localhost:6600&#10;/example.com/&#10;127.0.0.1">${e}</textarea>
74
+ <p class="text-xs text-gray-500 mt-1">匹配的网址将不展示面板且不抓取媒体链接</p>
75
+ </div>
76
+ `,showCancelButton:!0,confirmButtonText:"保存",cancelButtonText:"取消",confirmButtonColor:"#3b82f6",width:"600px",preConfirm:()=>{const o=document.getElementById("swal-webui-url"),i=document.getElementById("swal-exclude-urls"),a=document.getElementById("swal-media-ext-list"),s=o?o.value.trim():"",d=i?i.value.trim():"",c=a?a.value.trim():"",v=window.Swal;if(!s)return v?.showValidationMessage("WebUI 地址不能为空"),!1;const w=c.split(/[,\n\s]+/).map(E=>E.trim()).filter(E=>E);return w.length===0?(v?.showValidationMessage("媒体扩展名列表不能为空"),!1):{url:s,excludeUrls:d,mediaExtList:w}}}).then(o=>{if(o.isConfirmed&&o.value){const i=o.value;GM_setValue(j,i.url),ft(i.excludeUrls);const a=pt(i.mediaExtList),s=window.Swal;a&&s?s.fire({icon:"success",title:"设置已保存",html:`已保存 ${a.length} 个媒体扩展名类型`,timer:2e3,showConfirmButton:!1}):s&&s.fire({icon:"error",title:"保存失败",text:"媒体扩展名列表格式错误",timer:2e3,showConfirmButton:!1})}})}function Ct(t){const e=window.top&&window.top!==window;try{const o=window.open(t,"_blank");if(o&&!o.closed)return!0}catch(o){console.log("[M3U8 Capture] window.open failed:",o)}if(e&&window.top){try{const o=window.top.open(t,"_blank");if(o&&!o.closed)return!0}catch(o){console.log("[M3U8 Capture] window.top.open failed:",o)}try{return window.top.location.href=t,!0}catch(o){console.log("[M3U8 Capture] window.top.location.href failed:",o)}}const n=()=>{const o=window.Swal;o&&o.fire({icon:"info",title:"链接已复制",html:`由于 iframe 限制,链接已复制到剪贴板<br><br><code style="word-break: break-all; background: #f3f4f6; padding: 8px; border-radius: 4px; display: block; font-size: 12px;">${t}</code><br><br>请手动打开`,timer:4e3,showConfirmButton:!0,confirmButtonText:"确定"})};return nt(t).then(()=>n()).catch(()=>{const o=window.Swal;o&&o.fire({icon:"warning",title:"无法复制链接",html:`由于 iframe 限制,请手动复制并打开:<br><br><code style="word-break: break-all; background: #f3f4f6; padding: 8px; border-radius: 4px; display: block; font-size: 12px;">${t}</code>`,confirmButtonText:"确定"})}),!1}function N(){if(m||!x||!ct())return;F();const t=l?.querySelector("#m3u8-capture-list"),e=l?.querySelector("#m3u8-capture-empty"),n=l?.querySelector("#m3u8-capture-count");if(!t||!e||!n)return;if(n.textContent=x.size.toString(),x.size===0){t.classList.add("hidden"),e.classList.remove("hidden");return}t.classList.remove("hidden"),e.classList.add("hidden"),t.innerHTML="",Array.from(x.values()).sort((i,a)=>a.timestamp-i.timestamp).forEach(i=>{const a=document.createElement("div");a.className="border border-gray-200 rounded-lg p-3 bg-white transition-all duration-200 shadow-sm hover:bg-gray-50 hover:shadow-md";const s=i.title||"",d=i.type.toUpperCase();let c="bg-gray-500";d==="M3U8"||d==="M3U"?c="bg-blue-500":["MP4","MKV","AVI","MOV","WMV","FLV","WEBM","M4V","TS"].includes(d)?c="bg-green-500":["MP3","M4A","AAC","FLAC","APE","WAV","OGG","WMA"].includes(d)&&(c="bg-purple-500"),a.innerHTML=`
77
+ <div class="flex justify-between items-start gap-2 mb-2">
78
+ <div class="flex-1 min-w-0">
79
+ <div class="flex items-center gap-1.5 mb-1.5">
80
+ <span class="font-semibold text-[13px] text-gray-900 overflow-hidden text-ellipsis whitespace-nowrap" title="${s}">${s||"未命名媒体"}</span>
81
+ <span class="${c} text-white px-2 py-0.5 rounded-xl text-[10px] font-bold">${d}</span>
82
+ </div>
83
+ <div class="text-[11px] text-gray-500 overflow-hidden text-ellipsis whitespace-nowrap leading-snug max-w-[320px]" title="${i.url}">${i.url}</div>
84
+ </div>
85
+ </div>
86
+ <div class="flex gap-2">
87
+ <button class="m3u8-capture-download-btn flex-1 bg-blue-500 text-white border-none px-3.5 py-2 rounded-md cursor-pointer text-xs font-medium transition-all duration-200 hover:bg-blue-600 hover:-translate-y-0.5" data-url="${encodeURIComponent(i.url)}" data-title="${encodeURIComponent(s)}">
88
+ 跳转下载
89
+ </button>
90
+ <button class="m3u8-capture-copy-btn bg-gray-500 text-white border-none px-3.5 py-2 rounded-md cursor-pointer text-xs transition-all duration-200 hover:bg-gray-600" data-url="${i.url}">
91
+ 复制
92
+ </button>
93
+ </div>
94
+ `,t.appendChild(a)}),l?.querySelectorAll(".m3u8-capture-download-btn").forEach(i=>{i.addEventListener("click",a=>{a.stopPropagation();const s=decodeURIComponent(i.getAttribute("data-url")||""),d=decodeURIComponent(i.getAttribute("data-title")||""),c=`${V()}/page/download?from=capture&action=new&url=${encodeURIComponent(s+(d?`|${d}`:""))}`;Ct(c)})}),l?.querySelectorAll(".m3u8-capture-copy-btn").forEach(i=>{i.addEventListener("click",async a=>{a.stopPropagation();const s=i.getAttribute("data-url")||"",d=i.textContent,c=i.className;try{await nt(s),i.textContent="已复制",i.className="m3u8-capture-copy-btn bg-green-500 text-white border-none px-3.5 py-2 rounded-md cursor-pointer text-xs transition-all duration-200",setTimeout(()=>{i.textContent=d,i.className=c},2e3)}catch{const w=window.Swal;if(!w)return;w.fire({icon:"error",title:"复制失败",text:"请手动复制链接",html:`<code style="word-break: break-all; background: #f3f4f6; padding: 8px; border-radius: 4px; display: block; font-size: 12px;">${s}</code>`,confirmButtonText:"确定"})}})})}const I=new Map;Mt(I);function h(t,e=""){if(t=yt(t)||t,!t)return;if(m){try{const i={url:t,title:e||q(),type:it(t),pageUrl:window.location.href,timestamp:Date.now()};window.top?.postMessage({type:"m3u8-capture-link",data:i},"*")}catch(i){console.warn("[M3U8 Capture] Failed to send link to top window:",i)}return}const n=ht(t);I.get(n)?.title||(I.set(n,{url:t,title:e||q(),type:it(t),pageUrl:window.location.href,timestamp:Date.now()}),N())}function Lt(){if(k())return;m||window.addEventListener("message",n=>{if(n.data?.type==="m3u8-capture-link"&&n.data.data){const o=n.data.data;h(o.url,o.title)}}),bt(),vt();const t=()=>{if(!k()){if(m)return A();document.body&&(A(),I.size>0&&N())}};document.readyState==="loading"?document.addEventListener("DOMContentLoaded",t):setTimeout(t,100);let e=location.href;new MutationObserver(()=>{const n=location.href;if(n!==e){if(e=n,k()){H();return}setTimeout(()=>A(),1e3)}}).observe(document,{subtree:!0,childList:!0}),setInterval(()=>{document.body&&!k()&&A()},5e3)}Lt()})();
package/client/play.html CHANGED
@@ -1,11 +1,11 @@
1
1
  <!DOCTYPE html>
2
- <html lang="zh">
2
+ <html lang="zh-CN">
3
3
 
4
4
  <head>
5
5
  <title>M3U8 Player</title>
6
6
  <meta charset="utf-8">
7
7
  <meta http-equiv="X-UA-Compatible" content="IE=edge">
8
- <meta content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=0,viewport-fit=cover"
8
+ <meta content="width=device-width, initial-scale=1.0, viewport-fit=cover"
9
9
  name="viewport" />
10
10
  <link rel="shortcut icon" href="logo.png">
11
11
  <style>
package/package.json CHANGED
@@ -1,11 +1,14 @@
1
1
  {
2
2
  "name": "@lzwme/m3u8-dl",
3
- "version": "1.6.0-0",
4
- "description": "Batch download of m3u8 files and convert to mp4",
3
+ "version": "1.6.0",
4
+ "description": "A free, open-source, and powerful m3u8 video batch downloader with multi-threaded downloading, play-while-downloading, WebUI management, video parsing, and more.",
5
5
  "main": "cjs/index.js",
6
6
  "types": "cjs/index.d.ts",
7
7
  "license": "MIT",
8
- "repository": "https://github.com/lzwme/m3u8-dl.git",
8
+ "repository": {
9
+ "type": "git",
10
+ "url": "git+https://github.com/lzwme/m3u8-dl.git"
11
+ },
9
12
  "author": {
10
13
  "name": "renxia",
11
14
  "email": "lzwy0820@qq.com",
@@ -13,15 +16,20 @@
13
16
  },
14
17
  "scripts": {
15
18
  "prepare": "husky || true",
16
- "dev": "npm run watch",
17
- "watch": "npm run build -- -- -w",
19
+ "dev": "concurrently \"pnpm dev:sdk\" \"pnpm dev:server\" \"pnpm dev:frontend\"",
20
+ "dev:sdk": "tsc -p tsconfig.cjs.json --watch",
21
+ "dev:frontend": "pnpm -F @lzwme/m3u8-dl-frontend dev",
22
+ "dev:app": "pnpm -F @lzwme/m3u8-dl-app dev",
23
+ "dev:server": "nodemon bin/m3u8dl.js server --debug",
18
24
  "lint:flh": "flh --eslint --tscheck --prettier",
19
25
  "lint": "biome lint",
26
+ "lint:all": "biome lint && pnpm -F @lzwme/m3u8-capture lint && pnpm -F @lzwme/m3u8-dl-frontend lint",
20
27
  "format": "biome format --write",
21
28
  "fix": "biome check --fix",
22
- "build": "npm run clean && npm run build:cjs && npm run build:frontend",
29
+ "build": "npm run clean && npm run build:cjs && npm run build:frontend && npm run build:capture",
23
30
  "build:cjs": "tsc -p tsconfig.cjs.json",
24
31
  "build:frontend": "pnpm -F @lzwme/m3u8-dl-frontend build",
32
+ "build:capture": "pnpm -F @lzwme/m3u8-capture build",
25
33
  "download-cdn": "node scripts/download-cdn-resources.js",
26
34
  "doc": "typedoc src/ --exclude **/*.spec.ts --out docs --tsconfig tsconfig.module.json",
27
35
  "version": "standard-version",
@@ -48,24 +56,26 @@
48
56
  "registry": "https://registry.npmjs.com"
49
57
  },
50
58
  "devDependencies": {
51
- "@biomejs/biome": "^2.3.4",
59
+ "@biomejs/biome": "^2.3.5",
52
60
  "@eslint/js": "^9.39.1",
53
61
  "@lzwme/fed-lint-helper": "^2.6.6",
54
62
  "@types/express": "^5.0.5",
55
63
  "@types/m3u8-parser": "^7.2.5",
56
64
  "@types/node": "^24.10.0",
57
65
  "@types/ws": "^8.18.1",
58
- "@typescript-eslint/eslint-plugin": "^8.46.3",
59
- "@typescript-eslint/parser": "^8.46.3",
66
+ "@typescript-eslint/eslint-plugin": "^8.46.4",
67
+ "@typescript-eslint/parser": "^8.46.4",
68
+ "concurrently": "^9.2.1",
60
69
  "eslint": "^9.39.1",
61
70
  "eslint-config-prettier": "^10.1.8",
62
71
  "eslint-plugin-prettier": "^5.5.4",
63
72
  "express": "^5.1.0",
64
73
  "husky": "^9.1.7",
74
+ "nodemon": "^3.1.11",
65
75
  "prettier": "^3.6.2",
66
76
  "standard-version": "^9.5.0",
67
77
  "typescript": "^5.9.3",
68
- "typescript-eslint": "^8.46.3",
78
+ "typescript-eslint": "^8.46.4",
69
79
  "ws": "^8.18.3"
70
80
  },
71
81
  "dependencies": {
@@ -1 +0,0 @@
1
- .app-layout[data-v-a75620cd]{position:relative;min-height:100vh}.main-content[data-v-a75620cd]{transition:margin-left .3s ease,width .3s ease;margin-left:16rem;width:calc(100% - 16rem);padding:.25rem}.main-content.sidebar-collapsed[data-v-a75620cd]{margin-left:0;width:100%}@media(max-width:768px){.main-content[data-v-a75620cd]{margin-left:0!important;width:100%!important;padding:.25rem}}@media(min-width:769px)and (max-width:1024px){.main-content[data-v-a75620cd]{margin-left:12rem;width:calc(100% - 12rem)}.sidebar[data-v-a75620cd]{width:12rem}}.fixed.inset-0[data-v-7740f374]{position:fixed;inset:0}.fade-enter-active[data-v-7740f374],.fade-leave-active[data-v-7740f374]{transition:opacity .3s ease}.fade-enter-from[data-v-7740f374],.fade-leave-to[data-v-7740f374]{opacity:0}.slide-enter-active[data-v-7740f374],.slide-leave-active[data-v-7740f374]{transition:transform .3s ease,opacity .3s ease}@media(max-width:767px){.playlist-sidebar[data-v-7740f374]{position:fixed;top:0;right:0;bottom:0;z-index:50;border-left:none;width:85vw;max-width:320px;box-shadow:-4px 0 24px #00000080}.slide-enter-from[data-v-7740f374],.slide-leave-to[data-v-7740f374]{transform:translate(100%);opacity:0}}input[type=checkbox][data-v-51fd32c8]{min-width:20px}.web-browser-container[data-v-8ef86d80]{box-shadow:0 1px 3px #0000001a}@layer properties{@supports ((-webkit-hyphens:none) and (not (margin-trim:inline))) or ((-moz-orient:inline) and (not (color:rgb(from red r g b)))){*,:before,:after,::backdrop{--tw-rotate-x:initial;--tw-rotate-y:initial;--tw-rotate-z:initial;--tw-skew-x:initial;--tw-skew-y:initial;--tw-space-y-reverse:0;--tw-space-x-reverse:0;--tw-divide-y-reverse:0;--tw-border-style:solid;--tw-font-weight:initial;--tw-tracking:initial;--tw-shadow:0 0 #0000;--tw-shadow-color:initial;--tw-shadow-alpha:100%;--tw-inset-shadow:0 0 #0000;--tw-inset-shadow-color:initial;--tw-inset-shadow-alpha:100%;--tw-ring-color:initial;--tw-ring-shadow:0 0 #0000;--tw-inset-ring-color:initial;--tw-inset-ring-shadow:0 0 #0000;--tw-ring-inset:initial;--tw-ring-offset-width:0px;--tw-ring-offset-color:#fff;--tw-ring-offset-shadow:0 0 #0000;--tw-backdrop-blur:initial;--tw-backdrop-brightness:initial;--tw-backdrop-contrast:initial;--tw-backdrop-grayscale:initial;--tw-backdrop-hue-rotate:initial;--tw-backdrop-invert:initial;--tw-backdrop-opacity:initial;--tw-backdrop-saturate:initial;--tw-backdrop-sepia:initial;--tw-duration:initial;--tw-ease:initial}}}@layer theme{:root,:host{--font-sans:ui-sans-serif,system-ui,sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol","Noto Color Emoji";--font-mono:ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace;--color-red-50:oklch(97.1% .013 17.38);--color-red-100:oklch(93.6% .032 17.717);--color-red-200:oklch(88.5% .062 18.334);--color-red-300:oklch(80.8% .114 19.571);--color-red-500:oklch(63.7% .237 25.331);--color-red-600:oklch(57.7% .245 27.325);--color-red-700:oklch(50.5% .213 27.518);--color-red-800:oklch(44.4% .177 26.899);--color-yellow-50:oklch(98.7% .026 102.212);--color-yellow-100:oklch(97.3% .071 103.193);--color-yellow-300:oklch(90.5% .182 98.111);--color-yellow-500:oklch(79.5% .184 86.047);--color-yellow-600:oklch(68.1% .162 75.834);--color-yellow-800:oklch(47.6% .114 61.907);--color-green-50:oklch(98.2% .018 155.826);--color-green-100:oklch(96.2% .044 156.743);--color-green-300:oklch(87.1% .15 154.449);--color-green-500:oklch(72.3% .219 149.579);--color-green-600:oklch(62.7% .194 149.214);--color-green-700:oklch(52.7% .154 150.069);--color-green-800:oklch(44.8% .119 151.328);--color-blue-50:oklch(97% .014 254.604);--color-blue-100:oklch(93.2% .032 255.585);--color-blue-200:oklch(88.2% .059 254.128);--color-blue-300:oklch(80.9% .105 251.813);--color-blue-400:oklch(70.7% .165 254.624);--color-blue-500:oklch(62.3% .214 259.815);--color-blue-600:oklch(54.6% .245 262.881);--color-blue-700:oklch(48.8% .243 264.376);--color-blue-800:oklch(42.4% .199 265.638);--color-gray-50:oklch(98.5% .002 247.839);--color-gray-100:oklch(96.7% .003 264.542);--color-gray-200:oklch(92.8% .006 264.531);--color-gray-300:oklch(87.2% .01 258.338);--color-gray-400:oklch(70.7% .022 261.325);--color-gray-500:oklch(55.1% .027 264.364);--color-gray-600:oklch(44.6% .03 256.802);--color-gray-700:oklch(37.3% .034 259.733);--color-gray-800:oklch(27.8% .033 256.848);--color-gray-900:oklch(21% .034 264.665);--color-black:#000;--color-white:#fff;--spacing:.25rem;--container-md:28rem;--container-3xl:48rem;--container-4xl:56rem;--text-xs:.75rem;--text-xs--line-height:calc(1/.75);--text-sm:.875rem;--text-sm--line-height:calc(1.25/.875);--text-base:1rem;--text-base--line-height: 1.5 ;--text-lg:1.125rem;--text-lg--line-height:calc(1.75/1.125);--text-xl:1.25rem;--text-xl--line-height:calc(1.75/1.25);--text-4xl:2.25rem;--text-4xl--line-height:calc(2.5/2.25);--font-weight-medium:500;--font-weight-semibold:600;--font-weight-bold:700;--tracking-wider:.05em;--radius-lg:.5rem;--ease-out:cubic-bezier(0,0,.2,1);--animate-spin:spin 1s linear infinite;--blur-sm:8px;--default-transition-duration:.15s;--default-transition-timing-function:cubic-bezier(.4,0,.2,1);--default-font-family:var(--font-sans);--default-mono-font-family:var(--font-mono)}}@layer base{*,:after,:before,::backdrop{box-sizing:border-box;border:0 solid;margin:0;padding:0}::file-selector-button{box-sizing:border-box;border:0 solid;margin:0;padding:0}html,:host{-webkit-text-size-adjust:100%;-moz-tab-size:4;-o-tab-size:4;tab-size:4;line-height:1.5;font-family:var(--default-font-family,ui-sans-serif,system-ui,sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol","Noto Color Emoji");font-feature-settings:var(--default-font-feature-settings,normal);font-variation-settings:var(--default-font-variation-settings,normal);-webkit-tap-highlight-color:transparent}hr{height:0;color:inherit;border-top-width:1px}abbr:where([title]){-webkit-text-decoration:underline dotted;text-decoration:underline dotted}h1,h2,h3,h4,h5,h6{font-size:inherit;font-weight:inherit}a{color:inherit;-webkit-text-decoration:inherit;text-decoration:inherit}b,strong{font-weight:bolder}code,kbd,samp,pre{font-family:var(--default-mono-font-family,ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace);font-feature-settings:var(--default-mono-font-feature-settings,normal);font-variation-settings:var(--default-mono-font-variation-settings,normal);font-size:1em}small{font-size:80%}sub,sup{vertical-align:baseline;font-size:75%;line-height:0;position:relative}sub{bottom:-.25em}sup{top:-.5em}table{text-indent:0;border-color:inherit;border-collapse:collapse}:-moz-focusring{outline:auto}progress{vertical-align:baseline}summary{display:list-item}ol,ul,menu{list-style:none}img,svg,video,canvas,audio,iframe,embed,object{vertical-align:middle;display:block}img,video{max-width:100%;height:auto}button,input,select,optgroup,textarea{font:inherit;font-feature-settings:inherit;font-variation-settings:inherit;letter-spacing:inherit;color:inherit;opacity:1;background-color:#0000;border-radius:0}::file-selector-button{font:inherit;font-feature-settings:inherit;font-variation-settings:inherit;letter-spacing:inherit;color:inherit;opacity:1;background-color:#0000;border-radius:0}:where(select:is([multiple],[size])) optgroup{font-weight:bolder}:where(select:is([multiple],[size])) optgroup option{padding-inline-start:20px}::file-selector-button{margin-inline-end:4px}::-moz-placeholder{opacity:1}::placeholder{opacity:1}@supports (not (-webkit-appearance:-apple-pay-button)) or (contain-intrinsic-size:1px){::-moz-placeholder{color:currentColor}::placeholder{color:currentColor}@supports (color:color-mix(in lab,red,red)){::-moz-placeholder{color:color-mix(in oklab,currentcolor 50%,transparent)}::placeholder{color:color-mix(in oklab,currentcolor 50%,transparent)}}}textarea{resize:vertical}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-date-and-time-value{min-height:1lh;text-align:inherit}::-webkit-datetime-edit{display:inline-flex}::-webkit-datetime-edit-fields-wrapper{padding:0}::-webkit-datetime-edit{padding-block:0}::-webkit-datetime-edit-year-field{padding-block:0}::-webkit-datetime-edit-month-field{padding-block:0}::-webkit-datetime-edit-day-field{padding-block:0}::-webkit-datetime-edit-hour-field{padding-block:0}::-webkit-datetime-edit-minute-field{padding-block:0}::-webkit-datetime-edit-second-field{padding-block:0}::-webkit-datetime-edit-millisecond-field{padding-block:0}::-webkit-datetime-edit-meridiem-field{padding-block:0}::-webkit-calendar-picker-indicator{line-height:1}:-moz-ui-invalid{box-shadow:none}button,input:where([type=button],[type=reset],[type=submit]){-webkit-appearance:button;-moz-appearance:button;appearance:button}::file-selector-button{-webkit-appearance:button;-moz-appearance:button;appearance:button}::-webkit-inner-spin-button{height:auto}::-webkit-outer-spin-button{height:auto}[hidden]:where(:not([hidden=until-found])){display:none!important}}@layer components;@layer utilities{.visible{visibility:visible}.absolute{position:absolute}.fixed{position:fixed}.relative{position:relative}.inset-0{inset:calc(var(--spacing)*0)}.top-1{top:calc(var(--spacing)*1)}.top-3{top:calc(var(--spacing)*3)}.right-1{right:calc(var(--spacing)*1)}.right-2{right:calc(var(--spacing)*2)}.left-3{left:calc(var(--spacing)*3)}.z-40{z-index:40}.z-50{z-index:50}.z-100{z-index:100}.z-\[9999\]{z-index:9999}.container{width:100%}@media(min-width:40rem){.container{max-width:40rem}}@media(min-width:48rem){.container{max-width:48rem}}@media(min-width:64rem){.container{max-width:64rem}}@media(min-width:80rem){.container{max-width:80rem}}@media(min-width:96rem){.container{max-width:96rem}}.mt-1{margin-top:calc(var(--spacing)*1)}.mt-2{margin-top:calc(var(--spacing)*2)}.mt-4{margin-top:calc(var(--spacing)*4)}.mt-6{margin-top:calc(var(--spacing)*6)}.mr-1{margin-right:calc(var(--spacing)*1)}.mr-2{margin-right:calc(var(--spacing)*2)}.mr-3{margin-right:calc(var(--spacing)*3)}.mb-1{margin-bottom:calc(var(--spacing)*1)}.mb-2{margin-bottom:calc(var(--spacing)*2)}.mb-3{margin-bottom:calc(var(--spacing)*3)}.mb-4{margin-bottom:calc(var(--spacing)*4)}.mb-6{margin-bottom:calc(var(--spacing)*6)}.mb-8{margin-bottom:calc(var(--spacing)*8)}.ml-1{margin-left:calc(var(--spacing)*1)}.ml-2{margin-left:calc(var(--spacing)*2)}.ml-3{margin-left:calc(var(--spacing)*3)}.ml-4{margin-left:calc(var(--spacing)*4)}.block{display:block}.flex{display:flex}.grid{display:grid}.inline-block{display:inline-block}.inline-flex{display:inline-flex}.h-2{height:calc(var(--spacing)*2)}.h-4{height:calc(var(--spacing)*4)}.h-5{height:calc(var(--spacing)*5)}.h-8{height:calc(var(--spacing)*8)}.h-10{height:calc(var(--spacing)*10)}.h-12{height:calc(var(--spacing)*12)}.h-full{height:100%}.max-h-96{max-height:calc(var(--spacing)*96)}.max-h-\[90vh\]{max-height:90vh}.max-h-\[calc\(100vh-200px\)\]{max-height:calc(100vh - 200px)}.w-4{width:calc(var(--spacing)*4)}.w-5{width:calc(var(--spacing)*5)}.w-8{width:calc(var(--spacing)*8)}.w-10{width:calc(var(--spacing)*10)}.w-12{width:calc(var(--spacing)*12)}.w-80{width:calc(var(--spacing)*80)}.w-full{width:100%}.max-w-3xl{max-width:var(--container-3xl)}.max-w-4xl{max-width:var(--container-4xl)}.max-w-\[calc\(100vw-100px\)\]{max-width:calc(100vw - 100px)}.max-w-md{max-width:var(--container-md)}.min-w-0{min-width:calc(var(--spacing)*0)}.flex-1{flex:1}.flex-shrink-0{flex-shrink:0}.transform{transform:var(--tw-rotate-x,)var(--tw-rotate-y,)var(--tw-rotate-z,)var(--tw-skew-x,)var(--tw-skew-y,)}.animate-spin{animation:var(--animate-spin)}.cursor-pointer{cursor:pointer}.resize{resize:both}.list-inside{list-style-position:inside}.list-disc{list-style-type:disc}.grid-cols-1{grid-template-columns:repeat(1,minmax(0,1fr))}.flex-col{flex-direction:column}.flex-row{flex-direction:row}.flex-wrap{flex-wrap:wrap}.items-center{align-items:center}.items-start{align-items:flex-start}.justify-between{justify-content:space-between}.justify-center{justify-content:center}.justify-end{justify-content:flex-end}.gap-1{gap:calc(var(--spacing)*1)}.gap-2{gap:calc(var(--spacing)*2)}.gap-3{gap:calc(var(--spacing)*3)}.gap-4{gap:calc(var(--spacing)*4)}.gap-6{gap:calc(var(--spacing)*6)}:where(.space-y-2>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing)*2)*var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing)*2)*calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-3>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing)*3)*var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing)*3)*calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-6>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing)*6)*var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing)*6)*calc(1 - var(--tw-space-y-reverse)))}:where(.space-x-2>:not(:last-child)){--tw-space-x-reverse:0;margin-inline-start:calc(calc(var(--spacing)*2)*var(--tw-space-x-reverse));margin-inline-end:calc(calc(var(--spacing)*2)*calc(1 - var(--tw-space-x-reverse)))}:where(.space-x-4>:not(:last-child)){--tw-space-x-reverse:0;margin-inline-start:calc(calc(var(--spacing)*4)*var(--tw-space-x-reverse));margin-inline-end:calc(calc(var(--spacing)*4)*calc(1 - var(--tw-space-x-reverse)))}:where(.divide-y>:not(:last-child)){--tw-divide-y-reverse:0;border-bottom-style:var(--tw-border-style);border-top-style:var(--tw-border-style);border-top-width:calc(1px*var(--tw-divide-y-reverse));border-bottom-width:calc(1px*calc(1 - var(--tw-divide-y-reverse)))}:where(.divide-gray-200>:not(:last-child)){border-color:var(--color-gray-200)}.truncate{text-overflow:ellipsis;white-space:nowrap;overflow:hidden}.overflow-auto{overflow:auto}.overflow-hidden{overflow:hidden}.overflow-x-auto{overflow-x:auto}.overflow-y-auto{overflow-y:auto}.rounded{border-radius:.25rem}.rounded-full{border-radius:3.40282e38px}.rounded-lg{border-radius:var(--radius-lg)}.border{border-style:var(--tw-border-style);border-width:1px}.border-0{border-style:var(--tw-border-style);border-width:0}.border-4{border-style:var(--tw-border-style);border-width:4px}.border-t{border-top-style:var(--tw-border-style);border-top-width:1px}.border-b{border-bottom-style:var(--tw-border-style);border-bottom-width:1px}.border-l{border-left-style:var(--tw-border-style);border-left-width:1px}.border-blue-200{border-color:var(--color-blue-200)}.border-gray-100{border-color:var(--color-gray-100)}.border-gray-200{border-color:var(--color-gray-200)}.border-gray-300{border-color:var(--color-gray-300)}.border-gray-700{border-color:var(--color-gray-700)}.border-red-200{border-color:var(--color-red-200)}.border-t-blue-500{border-top-color:var(--color-blue-500)}.bg-black{background-color:var(--color-black)}.bg-black\/50{background-color:#00000080}@supports (color:color-mix(in lab,red,red)){.bg-black\/50{background-color:color-mix(in oklab,var(--color-black)50%,transparent)}}.bg-blue-50{background-color:var(--color-blue-50)}.bg-blue-100{background-color:var(--color-blue-100)}.bg-blue-200{background-color:var(--color-blue-200)}.bg-blue-500{background-color:var(--color-blue-500)}.bg-blue-500\/20{background-color:#3080ff33}@supports (color:color-mix(in lab,red,red)){.bg-blue-500\/20{background-color:color-mix(in oklab,var(--color-blue-500)20%,transparent)}}.bg-blue-600{background-color:var(--color-blue-600)}.bg-gray-50{background-color:var(--color-gray-50)}.bg-gray-100{background-color:var(--color-gray-100)}.bg-gray-500{background-color:var(--color-gray-500)}.bg-gray-500\/20{background-color:#6a728233}@supports (color:color-mix(in lab,red,red)){.bg-gray-500\/20{background-color:color-mix(in oklab,var(--color-gray-500)20%,transparent)}}.bg-gray-800{background-color:var(--color-gray-800)}.bg-gray-900{background-color:var(--color-gray-900)}.bg-green-100{background-color:var(--color-green-100)}.bg-green-500{background-color:var(--color-green-500)}.bg-green-500\/20{background-color:#00c75833}@supports (color:color-mix(in lab,red,red)){.bg-green-500\/20{background-color:color-mix(in oklab,var(--color-green-500)20%,transparent)}}.bg-green-600{background-color:var(--color-green-600)}.bg-red-50{background-color:var(--color-red-50)}.bg-red-100{background-color:var(--color-red-100)}.bg-red-500{background-color:var(--color-red-500)}.bg-red-500\/20{background-color:#fb2c3633}@supports (color:color-mix(in lab,red,red)){.bg-red-500\/20{background-color:color-mix(in oklab,var(--color-red-500)20%,transparent)}}.bg-white{background-color:var(--color-white)}.bg-yellow-100{background-color:var(--color-yellow-100)}.bg-yellow-500{background-color:var(--color-yellow-500)}.bg-yellow-500\/20{background-color:#edb20033}@supports (color:color-mix(in lab,red,red)){.bg-yellow-500\/20{background-color:color-mix(in oklab,var(--color-yellow-500)20%,transparent)}}.p-1{padding:calc(var(--spacing)*1)}.p-1\.5{padding:calc(var(--spacing)*1.5)}.p-2{padding:calc(var(--spacing)*2)}.p-3{padding:calc(var(--spacing)*3)}.p-4{padding:calc(var(--spacing)*4)}.p-6{padding:calc(var(--spacing)*6)}.p-8{padding:calc(var(--spacing)*8)}.px-1\.5{padding-inline:calc(var(--spacing)*1.5)}.px-2{padding-inline:calc(var(--spacing)*2)}.px-3{padding-inline:calc(var(--spacing)*3)}.px-4{padding-inline:calc(var(--spacing)*4)}.px-6{padding-inline:calc(var(--spacing)*6)}.py-0\.5{padding-block:calc(var(--spacing)*.5)}.py-1{padding-block:calc(var(--spacing)*1)}.py-1\.5{padding-block:calc(var(--spacing)*1.5)}.py-2{padding-block:calc(var(--spacing)*2)}.py-3{padding-block:calc(var(--spacing)*3)}.py-4{padding-block:calc(var(--spacing)*4)}.py-8{padding-block:calc(var(--spacing)*8)}.pt-1{padding-top:calc(var(--spacing)*1)}.pt-2{padding-top:calc(var(--spacing)*2)}.pt-4{padding-top:calc(var(--spacing)*4)}.pr-4{padding-right:calc(var(--spacing)*4)}.pb-2{padding-bottom:calc(var(--spacing)*2)}.pl-10{padding-left:calc(var(--spacing)*10)}.text-center{text-align:center}.text-left{text-align:left}.text-right{text-align:right}.text-4xl{font-size:var(--text-4xl);line-height:var(--tw-leading,var(--text-4xl--line-height))}.text-lg{font-size:var(--text-lg);line-height:var(--tw-leading,var(--text-lg--line-height))}.text-sm{font-size:var(--text-sm);line-height:var(--tw-leading,var(--text-sm--line-height))}.text-xl{font-size:var(--text-xl);line-height:var(--tw-leading,var(--text-xl--line-height))}.text-xs{font-size:var(--text-xs);line-height:var(--tw-leading,var(--text-xs--line-height))}.font-bold{--tw-font-weight:var(--font-weight-bold);font-weight:var(--font-weight-bold)}.font-medium{--tw-font-weight:var(--font-weight-medium);font-weight:var(--font-weight-medium)}.font-semibold{--tw-font-weight:var(--font-weight-semibold);font-weight:var(--font-weight-semibold)}.tracking-wider{--tw-tracking:var(--tracking-wider);letter-spacing:var(--tracking-wider)}.break-words{overflow-wrap:break-word}.break-all{word-break:break-all}.whitespace-nowrap{white-space:nowrap}.whitespace-pre-line{white-space:pre-line}.text-blue-300{color:var(--color-blue-300)}.text-blue-500{color:var(--color-blue-500)}.text-blue-600{color:var(--color-blue-600)}.text-blue-700{color:var(--color-blue-700)}.text-blue-800{color:var(--color-blue-800)}.text-gray-100{color:var(--color-gray-100)}.text-gray-200{color:var(--color-gray-200)}.text-gray-300{color:var(--color-gray-300)}.text-gray-400{color:var(--color-gray-400)}.text-gray-500{color:var(--color-gray-500)}.text-gray-600{color:var(--color-gray-600)}.text-gray-700{color:var(--color-gray-700)}.text-gray-800{color:var(--color-gray-800)}.text-gray-900{color:var(--color-gray-900)}.text-green-300{color:var(--color-green-300)}.text-green-500{color:var(--color-green-500)}.text-green-600{color:var(--color-green-600)}.text-green-700{color:var(--color-green-700)}.text-green-800{color:var(--color-green-800)}.text-red-300{color:var(--color-red-300)}.text-red-500{color:var(--color-red-500)}.text-red-600{color:var(--color-red-600)}.text-red-700{color:var(--color-red-700)}.text-white{color:var(--color-white)}.text-yellow-300{color:var(--color-yellow-300)}.text-yellow-500{color:var(--color-yellow-500)}.text-yellow-600{color:var(--color-yellow-600)}.text-yellow-800{color:var(--color-yellow-800)}.uppercase{text-transform:uppercase}.shadow{--tw-shadow:0 1px 3px 0 var(--tw-shadow-color,#0000001a),0 1px 2px -1px var(--tw-shadow-color,#0000001a);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-none{--tw-shadow:0 0 #0000;box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-xl{--tw-shadow:0 20px 25px -5px var(--tw-shadow-color,#0000001a),0 8px 10px -6px var(--tw-shadow-color,#0000001a);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.backdrop-blur-sm{--tw-backdrop-blur:blur(var(--blur-sm));-webkit-backdrop-filter:var(--tw-backdrop-blur,)var(--tw-backdrop-brightness,)var(--tw-backdrop-contrast,)var(--tw-backdrop-grayscale,)var(--tw-backdrop-hue-rotate,)var(--tw-backdrop-invert,)var(--tw-backdrop-opacity,)var(--tw-backdrop-saturate,)var(--tw-backdrop-sepia,);backdrop-filter:var(--tw-backdrop-blur,)var(--tw-backdrop-brightness,)var(--tw-backdrop-contrast,)var(--tw-backdrop-grayscale,)var(--tw-backdrop-hue-rotate,)var(--tw-backdrop-invert,)var(--tw-backdrop-opacity,)var(--tw-backdrop-saturate,)var(--tw-backdrop-sepia,)}.transition{transition-property:color,background-color,border-color,outline-color,text-decoration-color,fill,stroke,--tw-gradient-from,--tw-gradient-via,--tw-gradient-to,opacity,box-shadow,transform,translate,scale,rotate,filter,backdrop-filter,display,content-visibility,overlay,pointer-events;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-all{transition-property:all;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-colors{transition-property:color,background-color,border-color,outline-color,text-decoration-color,fill,stroke,--tw-gradient-from,--tw-gradient-via,--tw-gradient-to;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.duration-300{--tw-duration:.3s;transition-duration:.3s}.ease-out{--tw-ease:var(--ease-out);transition-timing-function:var(--ease-out)}@media(hover:hover){.hover\:bg-blue-600:hover{background-color:var(--color-blue-600)}.hover\:bg-blue-700:hover{background-color:var(--color-blue-700)}.hover\:bg-gray-50:hover{background-color:var(--color-gray-50)}.hover\:bg-gray-100:hover{background-color:var(--color-gray-100)}.hover\:bg-gray-200:hover{background-color:var(--color-gray-200)}.hover\:bg-gray-600:hover{background-color:var(--color-gray-600)}.hover\:bg-gray-700:hover{background-color:var(--color-gray-700)}.hover\:bg-gray-800:hover{background-color:var(--color-gray-800)}.hover\:bg-green-50:hover{background-color:var(--color-green-50)}.hover\:bg-green-600:hover{background-color:var(--color-green-600)}.hover\:bg-green-700:hover{background-color:var(--color-green-700)}.hover\:bg-red-50:hover{background-color:var(--color-red-50)}.hover\:bg-red-600:hover{background-color:var(--color-red-600)}.hover\:bg-yellow-50:hover{background-color:var(--color-yellow-50)}.hover\:bg-yellow-600:hover{background-color:var(--color-yellow-600)}.hover\:text-blue-600:hover{color:var(--color-blue-600)}.hover\:text-blue-800:hover{color:var(--color-blue-800)}.hover\:text-gray-700:hover{color:var(--color-gray-700)}.hover\:text-gray-800:hover{color:var(--color-gray-800)}.hover\:text-green-600:hover{color:var(--color-green-600)}.hover\:text-green-800:hover{color:var(--color-green-800)}.hover\:text-red-600:hover{color:var(--color-red-600)}.hover\:text-red-800:hover{color:var(--color-red-800)}.hover\:text-white:hover{color:var(--color-white)}.hover\:text-yellow-600:hover{color:var(--color-yellow-600)}.hover\:underline:hover{text-decoration-line:underline}}.focus\:border-blue-500:focus{border-color:var(--color-blue-500)}.focus\:border-transparent:focus{border-color:#0000}.focus\:ring-2:focus{--tw-ring-shadow:var(--tw-ring-inset,)0 0 0 calc(2px + var(--tw-ring-offset-width))var(--tw-ring-color,currentcolor);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.focus\:ring-blue-400:focus{--tw-ring-color:var(--color-blue-400)}.focus\:ring-blue-500:focus{--tw-ring-color:var(--color-blue-500)}.focus\:outline-none:focus{--tw-outline-style:none;outline-style:none}.disabled\:cursor-not-allowed:disabled{cursor:not-allowed}.disabled\:opacity-50:disabled{opacity:.5}@media(min-width:48rem){.md\:ml-2{margin-left:calc(var(--spacing)*2)}.md\:hidden{display:none}.md\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.md\:gap-2{gap:calc(var(--spacing)*2)}:where(.md\:space-y-3>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing)*3)*var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing)*3)*calc(1 - var(--tw-space-y-reverse)))}.md\:p-2{padding:calc(var(--spacing)*2)}.md\:p-4{padding:calc(var(--spacing)*4)}.md\:px-3{padding-inline:calc(var(--spacing)*3)}.md\:px-4{padding-inline:calc(var(--spacing)*4)}.md\:py-2{padding-block:calc(var(--spacing)*2)}.md\:text-base{font-size:var(--text-base);line-height:var(--tw-leading,var(--text-base--line-height))}.md\:text-sm{font-size:var(--text-sm);line-height:var(--tw-leading,var(--text-sm--line-height))}.md\:text-xl{font-size:var(--text-xl);line-height:var(--tw-leading,var(--text-xl--line-height))}}}:root{--color-primary:#3b82f6;--color-primary-hover:#2563eb;--color-success:#10b981;--color-warning:#f59e0b;--color-danger:#ef4444;--color-info:#3b82f6;--color-border:#e5e7eb;--spacing-xs:.25rem;--spacing-sm:.5rem;--spacing-md:1rem;--spacing-lg:1.5rem;--spacing-xl:2rem}*,:after,:before{border-color:var(--color-border);box-sizing:border-box;border-style:solid}::-moz-placeholder{font-size:14px}::placeholder{font-size:14px}button{cursor:pointer}#app{background-color:#f5f5f5;max-width:1400px;min-height:100vh;margin:auto;position:relative}.sidebar{z-index:1;background-color:#fff;width:16rem;transition:all .3s;position:absolute;top:0;bottom:0;left:0;box-shadow:2px 0 5px #0000001a}.download-item{transition:all .3s}.download-item:hover{background-color:#f8f9fa}.progress-bar{height:4px;transition:width .3s}.nav-item{transition:all .3s}.nav-item:hover{background-color:#f0f0f0;transform:translate(4px)}.nav-item.active{color:#1890ff;background-color:#e6f3ff}@media(max-width:768px){.sidebar{transform:translate(-100%)}.sidebar.show{transform:translate(0)}.menu-toggle{display:block!important}}@media(min-width:769px)and (max-width:1024px){.sidebar{width:12rem}}.menu-toggle{z-index:51;cursor:pointer;background:#fff;border-radius:.5rem;width:42px;height:42px;padding:.5rem;display:none;position:fixed;top:1rem;left:1rem;box-shadow:0 2px 4px #0000001a}.main-content{width:calc(100% - 16rem);margin-left:16rem;transition:margin-left .3s}.custom-toast{color:#333;z-index:99999;background:#fff;border-radius:4px;align-items:center;min-width:250px;max-width:400px;padding:15px 25px 15px 15px;transition:transform .3s;display:flex;position:fixed;top:20px;right:20px;transform:translate(150%);box-shadow:0 4px 12px #00000026}.custom-toast.show{transform:translate(0)}.custom-toast.hide{transform:translate(150%)}.custom-toast-success{border-left:4px solid #28a745}.custom-toast-error{border-left:4px solid #dc3545}.custom-toast-warning{border-left:4px solid #ffc107}.custom-toast-info{border-left:4px solid #17a2b8}.modal-overlay{background-color:#00000080;font-size:14px}.modal-overlay textarea,.modal-overlay input{padding-top:.25rem;padding-bottom:.25rem;font-size:16px}@property --tw-rotate-x{syntax:"*";inherits:false}@property --tw-rotate-y{syntax:"*";inherits:false}@property --tw-rotate-z{syntax:"*";inherits:false}@property --tw-skew-x{syntax:"*";inherits:false}@property --tw-skew-y{syntax:"*";inherits:false}@property --tw-space-y-reverse{syntax:"*";inherits:false;initial-value:0}@property --tw-space-x-reverse{syntax:"*";inherits:false;initial-value:0}@property --tw-divide-y-reverse{syntax:"*";inherits:false;initial-value:0}@property --tw-border-style{syntax:"*";inherits:false;initial-value:solid}@property --tw-font-weight{syntax:"*";inherits:false}@property --tw-tracking{syntax:"*";inherits:false}@property --tw-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-shadow-color{syntax:"*";inherits:false}@property --tw-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-inset-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-inset-shadow-color{syntax:"*";inherits:false}@property --tw-inset-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-ring-color{syntax:"*";inherits:false}@property --tw-ring-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-inset-ring-color{syntax:"*";inherits:false}@property --tw-inset-ring-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-ring-inset{syntax:"*";inherits:false}@property --tw-ring-offset-width{syntax:"<length>";inherits:false;initial-value:0}@property --tw-ring-offset-color{syntax:"*";inherits:false;initial-value:#fff}@property --tw-ring-offset-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-backdrop-blur{syntax:"*";inherits:false}@property --tw-backdrop-brightness{syntax:"*";inherits:false}@property --tw-backdrop-contrast{syntax:"*";inherits:false}@property --tw-backdrop-grayscale{syntax:"*";inherits:false}@property --tw-backdrop-hue-rotate{syntax:"*";inherits:false}@property --tw-backdrop-invert{syntax:"*";inherits:false}@property --tw-backdrop-opacity{syntax:"*";inherits:false}@property --tw-backdrop-saturate{syntax:"*";inherits:false}@property --tw-backdrop-sepia{syntax:"*";inherits:false}@property --tw-duration{syntax:"*";inherits:false}@property --tw-ease{syntax:"*";inherits:false}@keyframes spin{to{transform:rotate(360deg)}}