From 3d391415c6fe2276c930ec50fcbf23f9f58b3e2a Mon Sep 17 00:00:00 2001 From: Eeveid <448859157@qq.com> Date: Sat, 28 Mar 2026 18:58:52 +0800 Subject: [PATCH] =?UTF-8?q?=E4=BF=AE=E5=A4=8Dwss=E8=BF=9E=E6=8E=A5?= =?UTF-8?q?=E3=80=81=E4=B8=80=E4=BA=9B=E6=9C=AA=E7=9F=A5=E9=94=99=E8=AF=AF?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- backend/internal/ws/hub.go | 76 +-------- ...{index-CQ9sinZs.css => index-C-7tVt-S.css} | 2 +- frontend/dist/assets/index-DE3lDjdM.js | 17 -- frontend/dist/assets/index-DPzeYqvr.js | 17 ++ frontend/dist/index.html | 4 +- frontend/src/App.vue | 146 +++++++++++++++++- frontend/src/components/TransferPanel.vue | 5 + frontend/src/styles.css | 9 ++ 8 files changed, 179 insertions(+), 97 deletions(-) rename frontend/dist/assets/{index-CQ9sinZs.css => index-C-7tVt-S.css} (52%) delete mode 100644 frontend/dist/assets/index-DE3lDjdM.js create mode 100644 frontend/dist/assets/index-DPzeYqvr.js diff --git a/backend/internal/ws/hub.go b/backend/internal/ws/hub.go index cb2965a..09af4ed 100644 --- a/backend/internal/ws/hub.go +++ b/backend/internal/ws/hub.go @@ -5,9 +5,7 @@ import ( "encoding/json" "fmt" "log/slog" - "net" "net/http" - "net/url" "strings" "time" @@ -68,17 +66,10 @@ type realtimeBackplane interface { var upgrader = websocket.Upgrader{ CheckOrigin: func(r *http.Request) bool { - origin := strings.TrimSpace(r.Header.Get("Origin")) - if origin == "" { - return true - } - - parsed, err := url.Parse(origin) - if err != nil { - return false - } - - return originHostMatchesRequest(parsed.Host, r) + // Device sessions are authenticated before upgrade. Keeping origin + // permissive avoids false negatives behind reverse proxies or custom + // domains where Host/Forwarded headers are rewritten. + return true }, } @@ -307,62 +298,3 @@ func (c *Client) writePump() { } } } - -func originHostMatchesRequest(originHost string, r *http.Request) bool { - requestHosts := []string{ - strings.TrimSpace(r.Host), - strings.TrimSpace(r.Header.Get("X-Forwarded-Host")), - } - - originName, err := normalizeHost(originHost) - if err != nil { - return false - } - - for _, host := range requestHosts { - if host == "" { - continue - } - - requestName, err := normalizeHost(host) - if err != nil { - continue - } - - if requestName == originName { - return true - } - } - - return false -} - -func normalizeHost(host string) (string, error) { - host = strings.TrimSpace(host) - if host == "" { - return "", fmt.Errorf("empty host") - } - - if strings.Contains(host, "://") { - parsed, err := url.Parse(host) - if err != nil { - return "", err - } - host = parsed.Host - } - - name, _, err := net.SplitHostPort(host) - if err == nil { - host = name - } else if strings.HasPrefix(host, "[") && strings.HasSuffix(host, "]") { - host = strings.TrimPrefix(strings.TrimSuffix(host, "]"), "[") - } - - host = strings.ToLower(strings.TrimSpace(host)) - switch host { - case "127.0.0.1", "::1": - return "localhost", nil - default: - return host, nil - } -} diff --git a/frontend/dist/assets/index-CQ9sinZs.css b/frontend/dist/assets/index-C-7tVt-S.css similarity index 52% rename from frontend/dist/assets/index-CQ9sinZs.css rename to frontend/dist/assets/index-C-7tVt-S.css index 238e1a5..786ed78 100644 --- a/frontend/dist/assets/index-CQ9sinZs.css +++ b/frontend/dist/assets/index-C-7tVt-S.css @@ -1 +1 @@ -:root{--bg-color: #09090b;--card-bg: rgba(24, 24, 27, .65);--card-border: rgba(255, 255, 255, .08);--card-shadow: 0 20px 40px -10px rgba(0, 0, 0, .5), 0 0 2px rgba(255, 255, 255, .05) inset;--text-main: #f4f4f5;--text-secondary: #a1a1aa;--item-bg: rgba(255, 255, 255, .03);--item-bg-hover: rgba(255, 255, 255, .06);--item-border: rgba(255, 255, 255, .05);--icon-bg: rgba(255, 255, 255, .05);--input-bg: rgba(0, 0, 0, .3);--divider-color: rgba(255, 255, 255, .15);--divider-line: rgba(255, 255, 255, .08);--glow-1: rgba(0, 113, 227, .15);--glow-2: rgba(34, 211, 238, .12);--glow-3: rgba(168, 85, 247, .1);--accent-blue: #0a84ff;--accent-cyan: #30d158;--accent-cyan-light: rgba(10, 132, 255, .2);--success-green: #30d158;--danger-red: #ff453a;--border-radius-lg: 28px;--border-radius-sm: 18px;--transition-base: all .4s cubic-bezier(.2, .8, .2, 1);--font-stack: -apple-system, BlinkMacSystemFont, "SF Pro Text", "SF Pro Display", "Helvetica Neue", Roboto, sans-serif}body[data-theme=light]{--bg-color: #f2f4f8;--card-bg: rgba(255, 255, 255, .85);--card-border: rgba(255, 255, 255, 1);--card-shadow: 0 10px 30px -5px rgba(12, 43, 100, .08), 0 4px 10px -3px rgba(12, 43, 100, .04), 0 0 1px rgba(0, 0, 0, .05);--text-main: #1d1d1f;--text-secondary: #86868b;--item-bg: #ffffff;--item-bg-hover: #f8fafd;--item-border: rgba(0, 0, 0, .04);--icon-bg: #f5f5f7;--input-bg: #ffffff;--divider-color: #86868b;--divider-line: #e5e5ea;--glow-1: rgba(0, 113, 227, .05);--glow-2: rgba(34, 211, 238, .05);--glow-3: rgba(168, 85, 247, .03);--accent-blue: #0071e3;--accent-cyan: #0071e3;--accent-cyan-light: rgba(0, 113, 227, .1);--success-green: #34c759;--danger-red: #ff3b30}*{box-sizing:border-box;margin:0;padding:0;outline:none}html,body,#app{min-height:100%}body{font-family:var(--font-stack);background-color:var(--bg-color);color:var(--text-main);display:flex;justify-content:center;align-items:center;min-height:100vh;padding:20px 20px 100px;position:relative;overflow-x:hidden;overflow-y:auto;transition:background-color .5s ease,color .5s ease;-webkit-font-smoothing:antialiased}body:before{content:"";position:fixed;top:-25%;right:-25%;bottom:-25%;left:-25%;width:150%;height:150%;z-index:-1;background-image:radial-gradient(circle at 20% 20%,var(--glow-1) 0%,transparent 50%),radial-gradient(circle at 80% 20%,var(--glow-2) 0%,transparent 40%),radial-gradient(circle at 50% 80%,var(--glow-3) 0%,transparent 60%);animation:backgroundMove 25s infinite alternate ease-in-out}body,button,input{font:inherit}button{border:0;color:inherit}a{color:inherit}.app-icon{display:inline-flex;align-items:center;justify-content:center;flex:0 0 auto;line-height:1}.app-icon svg{width:100%;height:100%;overflow:visible}.container{width:100%;max-width:1000px;margin:auto;position:relative;z-index:1;transition:var(--transition-base)}.header{position:relative;display:flex;flex-direction:column;justify-content:center;align-items:center;min-height:80px;margin-top:30px;margin-bottom:40px;text-align:center}.header h1{font-size:42px;font-weight:800;letter-spacing:-1px;margin-bottom:8px}.header p{color:var(--text-secondary);font-size:16px;font-weight:400;letter-spacing:.5px}.theme-toggle{position:absolute;top:50%;right:0;width:44px;height:44px;margin-top:-22px;border-radius:50%;background:var(--card-bg);border:1px solid var(--card-border);color:var(--text-main);box-shadow:var(--card-shadow);cursor:pointer;display:flex;justify-content:center;align-items:center;transition:var(--transition-base);backdrop-filter:blur(20px);-webkit-backdrop-filter:blur(20px)}.theme-toggle:hover{transform:scale(1.08);box-shadow:0 8px 25px #0071e326}.main-grid{display:grid;grid-template-columns:1fr;gap:30px}.card{position:relative;display:flex;flex-direction:column;padding:40px 30px;background:var(--card-bg);border:1px solid var(--card-border);border-radius:var(--border-radius-lg);box-shadow:var(--card-shadow);transition:var(--transition-base);backdrop-filter:blur(40px) saturate(150%);-webkit-backdrop-filter:blur(40px) saturate(150%)}.card:before{content:"";position:absolute;inset:0 0 auto;height:1px;border-radius:var(--border-radius-lg) var(--border-radius-lg) 0 0;opacity:.5;background:linear-gradient(90deg,#fff0,#ffffff4d,#fff0)}.section-title{margin-bottom:25px;text-align:center;font-size:12px;font-weight:700;color:var(--text-secondary);text-transform:uppercase;letter-spacing:2px}.local-device-name{margin-top:-10px;margin-bottom:18px;text-align:center;font-size:13px;color:var(--text-secondary)}.local-device-name strong{color:var(--text-main);font-weight:600}.radar-container,.waiting-area{flex:1;min-height:200px;display:flex;flex-direction:column;justify-content:center;align-items:center;text-align:center}.radar{position:relative;width:80px;height:80px;margin-bottom:25px;display:flex;justify-content:center;align-items:center;border-radius:50%;background:var(--accent-cyan-light);border:1px solid rgba(0,113,227,.1);box-shadow:0 0 20px var(--accent-cyan-light)}.radar:before,.radar:after{content:"";position:absolute;top:0;right:0;bottom:0;left:0;border-radius:50%;border:2px solid var(--accent-blue);opacity:0;animation:ripplePro 3s infinite cubic-bezier(.1,.8,.3,1)}.radar:before{animation-delay:1.5s}.radar-icon{font-size:36px;color:var(--accent-blue)}.scan-status,.waiting-subtitle{margin-top:10px;font-size:14px;color:var(--text-secondary)}.device-list{width:100%;flex:1;display:flex;flex-direction:column;gap:12px}.device-item{position:relative;width:100%;display:flex;align-items:center;padding:16px 20px;background:var(--item-bg);border:1px solid var(--item-border);border-radius:var(--border-radius-sm);box-shadow:0 2px 8px #00000005;cursor:pointer;color:var(--text-main);text-align:left;transition:var(--transition-base)}.device-item:hover{background:var(--item-bg-hover);transform:translateY(-2px);border-color:#0071e34d;box-shadow:0 8px 20px -5px #0071e326,0 0 0 1px #0071e31a inset}.device-icon{width:48px;height:48px;margin-right:16px;display:flex;justify-content:center;align-items:center;border-radius:50%;background:var(--icon-bg);border:1px solid var(--item-border);transition:var(--transition-base)}.device-info{flex:1}.device-status-beacon{position:absolute;top:50%;right:22px;width:14px;height:14px;transform:translateY(-50%);pointer-events:none}.device-status-dot,.device-status-ring{position:absolute;top:0;right:0;bottom:0;left:0;border-radius:50%}.device-status-dot{top:3px;right:3px;bottom:3px;left:3px;background:var(--accent-blue);box-shadow:0 0 0 1px #ffffffbf,0 0 14px #0a84ff47}.device-status-ring{border:2px solid rgba(10,132,255,.22);opacity:0;transform:scale(.7);animation:deviceBeaconPing 2s infinite cubic-bezier(.16,.84,.44,1)}.device-status-ring-delay{animation-delay:1s}.device-icon .app-icon{position:relative;z-index:1;color:var(--text-main)}.device-item:hover .device-icon{background:var(--accent-cyan-light);border-color:transparent}.device-item:hover .device-icon .app-icon{color:var(--accent-blue)}.device-item:hover .device-status-dot{box-shadow:0 0 0 1px #ffffffd9,0 0 18px #0a84ff5c}.device-item:hover .device-status-ring{border-color:#0a84ff4d;animation-duration:1.4s}.device-info h4{margin-bottom:4px;font-size:16px;font-weight:600;color:var(--text-main)}.device-info p{font-size:13px;color:var(--text-secondary)}.room-action-area{flex:1;display:flex;flex-direction:column;justify-content:center}.btn-create{width:100%;margin-bottom:25px;padding:18px;display:flex;align-items:center;justify-content:center;gap:8px;background:var(--item-bg);color:var(--accent-blue);border:1px solid var(--item-border);border-radius:var(--border-radius-sm);box-shadow:0 2px 8px #00000005;font-size:16px;font-weight:600;cursor:pointer;transition:var(--transition-base)}.btn-create:hover{background:var(--item-bg-hover);border-color:#0071e34d;transform:translateY(-2px);box-shadow:0 8px 20px -5px #0071e326}.divider{position:relative;margin-bottom:25px;text-align:center;color:var(--divider-color);font-size:12px;font-weight:500;text-transform:uppercase;letter-spacing:1px}.divider:before,.divider:after{content:"";position:absolute;top:50%;width:35%;height:1px;background:var(--divider-line)}.divider:before{left:0}.divider:after{right:0}.room-input-group{display:flex;flex-direction:column;gap:12px}.pending-downloads{margin-top:22px;padding:16px 10px 16px 16px;border:1px solid var(--item-border);border-radius:var(--border-radius-sm);background:var(--item-bg);max-height:236px;overflow-y:auto}.pending-downloads::-webkit-scrollbar{width:6px}.pending-downloads::-webkit-scrollbar-thumb{background:var(--divider-color);border-radius:999px}.pending-downloads-head{display:flex;align-items:center;justify-content:space-between;margin-bottom:12px;font-size:12px;font-weight:700;letter-spacing:.08em;color:var(--text-secondary);position:sticky;top:0;z-index:1;background:var(--item-bg);padding-bottom:8px}.pending-download-item{display:flex;align-items:center;justify-content:space-between;gap:12px;width:100%;min-width:0;box-sizing:border-box;padding:12px 14px;border-radius:16px;background:var(--input-bg);border:1px solid var(--item-border);text-decoration:none;transition:var(--transition-base)}.pending-download-item+.pending-download-item{margin-top:10px}.pending-download-item:hover{background:var(--item-bg-hover);border-color:#0071e340;color:var(--accent-blue)}.pending-download-copy{flex:1;min-width:0;overflow:hidden}.pending-download-copy strong{display:-webkit-box;margin-bottom:4px;overflow:hidden;text-overflow:ellipsis;white-space:normal;overflow-wrap:anywhere;-webkit-box-orient:vertical;-webkit-line-clamp:2;line-height:1.35;max-height:2.7em;font-size:14px;font-weight:600;color:var(--text-main)}.pending-download-copy p{overflow:hidden;text-overflow:ellipsis;white-space:nowrap;font-size:12px;color:var(--text-secondary)}.pending-download-icon{flex:none}input.room-code,.text-input-group input,.text-input-group textarea{width:100%;min-width:0;padding:18px;border:1px solid var(--item-border);border-radius:var(--border-radius-sm);background:var(--input-bg);color:var(--text-main);box-shadow:inset 0 2px 4px #00000005;transition:var(--transition-base)}input.room-code{font-size:24px;font-weight:700;text-align:center;letter-spacing:8px;font-family:ui-monospace,SFMono-Regular,Menlo,monospace}input.room-code:focus,.text-input-group input:focus,.text-input-group textarea:focus{background:var(--item-bg);border-color:var(--accent-blue);box-shadow:0 0 0 3px var(--accent-cyan-light)}input.room-code::placeholder{letter-spacing:normal;font-size:16px;font-weight:400;color:var(--text-secondary);opacity:.6}.btn-primary{width:100%;padding:18px;background:var(--accent-blue);color:#fff;border-radius:var(--border-radius-sm);box-shadow:0 4px 12px #0071e34d;font-size:16px;font-weight:600;cursor:pointer;transition:var(--transition-base)}.btn-primary:hover,.text-input-group button:hover{background:#06c;transform:translateY(-2px);box-shadow:0 6px 16px #0071e366}.huge-code{margin:15px 0 25px;padding:15px 30px;font-size:56px;font-weight:800;color:var(--accent-blue);letter-spacing:10px;font-family:ui-monospace,monospace;border-radius:20px;border:1px solid rgba(0,113,227,.2);background:var(--accent-cyan-light)}.spinner{width:28px;height:28px;margin:0 auto 15px;border-radius:50%;border:3px solid var(--item-border);border-top-color:var(--accent-blue);animation:spin .8s linear infinite}.waiting-tip{font-size:14px;font-weight:500;color:var(--accent-blue)}.btn-cancel{margin-top:25px;background:none;color:var(--text-secondary);font-size:14px;cursor:pointer;transition:var(--transition-base)}.btn-cancel:hover{color:var(--text-main)}.transfer-panel,.admin-panel{width:100%;margin:0 auto;opacity:0;transform:scale(.95);animation:showPanelPro .5s forwards cubic-bezier(.2,.8,.2,1)}.transfer-panel{max-width:650px}.transfer-head{display:flex;align-items:center;justify-content:space-between;margin-bottom:30px;padding-bottom:20px;border-bottom:1px solid var(--divider-line)}.connected-to h2{margin-bottom:4px;font-size:20px;font-weight:700;color:var(--text-main)}.connected-to p{display:flex;align-items:center;gap:6px;font-size:13px;font-weight:500;color:var(--success-green)}.close-btn{width:36px;height:36px;display:flex;justify-content:center;align-items:center;border-radius:50%;background:var(--icon-bg);border:1px solid var(--item-border);color:var(--text-secondary);cursor:pointer;transition:var(--transition-base)}.close-btn:hover{background:var(--item-bg-hover);color:var(--text-main);transform:rotate(90deg)}.drop-zone{padding:40px 30px;margin-bottom:20px;text-align:center;background:var(--input-bg);border:2px dashed var(--divider-color);border-radius:var(--border-radius-sm);cursor:pointer;transition:var(--transition-base)}.drop-zone:hover,.drop-zone-active{border-color:var(--accent-blue);background-color:var(--accent-cyan-light)}.drop-zone-icon{margin-bottom:12px;font-size:42px;color:var(--text-secondary);transition:var(--transition-base)}.drop-zone:hover .drop-zone-icon,.drop-zone-active .drop-zone-icon{transform:translateY(-5px);color:var(--accent-blue)}.drop-zone-text{font-size:15px;font-weight:500;color:var(--text-main)}.text-input-group{display:flex;gap:12px;margin-bottom:20px}.text-input-group input{flex:1;padding:16px 20px;font-size:15px}.text-input-group button{width:52px;display:flex;align-items:center;justify-content:center;border-radius:var(--border-radius-sm);background:var(--accent-blue);color:#fff;box-shadow:0 4px 12px #0071e34d;cursor:pointer;transition:var(--transition-base)}.batch-actions{display:flex;justify-content:flex-end;margin-bottom:12px;height:0;overflow:hidden;opacity:0;pointer-events:none;transition:var(--transition-base)}.batch-actions.active{height:36px;overflow:visible;opacity:1;pointer-events:auto}.btn-small-primary{display:flex;align-items:center;gap:6px;padding:8px 16px;background:var(--item-bg);color:var(--text-main);border:1px solid var(--item-border);border-radius:12px;box-shadow:0 2px 6px #0000000a;font-size:13px;font-weight:600;cursor:pointer;transition:var(--transition-base)}.btn-small-primary:hover{background:var(--item-bg-hover);color:var(--accent-blue);border-color:#0071e34d;transform:translateY(-1px)}.batch-progress-container{max-height:280px;overflow-y:auto;padding-right:8px;padding-bottom:10px;text-align:left}.batch-progress-container::-webkit-scrollbar{width:6px}.batch-progress-container::-webkit-scrollbar-thumb{background:var(--divider-color);border-radius:4px}.batch-item{margin-bottom:12px;width:100%;min-width:0;box-sizing:border-box;padding:16px;background:var(--item-bg);border:1px solid var(--item-border);border-radius:var(--border-radius-sm);box-shadow:0 2px 8px #00000005;transition:all .3s ease}.batch-item:hover{border-color:#00000014;box-shadow:0 4px 12px #0000000a}body[data-theme=dark] .batch-item:hover{border-color:#ffffff1a}.file-info{display:flex;justify-content:space-between;align-items:center;gap:10px;min-width:0;margin-bottom:12px;font-size:14px}.file-info-left{flex:1;min-width:0;max-width:60%;display:flex;align-items:center;gap:10px;overflow:hidden}.file-icon-wrapper{flex:none;width:32px;height:32px;display:flex;justify-content:center;align-items:center;border-radius:8px;background:var(--icon-bg);color:var(--accent-blue)}.file-name{display:-webkit-box;flex:1;min-width:0;overflow:hidden;text-overflow:ellipsis;white-space:normal;overflow-wrap:anywhere;-webkit-box-orient:vertical;-webkit-line-clamp:2;line-height:1.35;max-height:2.7em;font-weight:500;color:var(--text-main)}.file-info-right{display:flex;flex:none;align-items:center;min-width:0;gap:8px}.file-status{margin-right:4px;white-space:nowrap;font-size:12px;font-weight:500;font-variant-numeric:tabular-nums}.action-btn{width:28px;height:28px;display:flex;justify-content:center;align-items:center;border-radius:8px;background:transparent;border:1px solid transparent;color:var(--text-secondary);cursor:pointer;transition:var(--transition-base)}.action-btn:hover{color:var(--text-main);background:var(--icon-bg);border-color:var(--item-border)}.action-btn.primary:hover{color:var(--accent-blue);background:var(--accent-cyan-light)}.action-btn.danger:hover{color:var(--danger-red);background:#ff3b301a}.progress-bar-container{height:6px;overflow:hidden;background:var(--icon-bg);border-radius:4px}.progress-bar-fill{height:100%;width:0;background:var(--accent-blue);border-radius:4px;transition:width .1s linear}.progress-bar-fill.success{background:var(--success-green)}.admin-header-card{margin-bottom:30px;padding:30px}.transfer-head-compact{margin-bottom:0;padding-bottom:0;border-bottom:0}.admin-title{font-size:24px;color:var(--text-main)}.connected-to .admin-subtitle{display:block;margin-top:4px;color:var(--text-secondary)}.admin-summary-grid{margin-bottom:30px}.admin-stats-card{height:100%}.admin-stats-panel{display:flex;flex-direction:column;height:100%}.admin-stats-row{flex:1;display:grid;grid-template-columns:repeat(2,minmax(0,1fr));grid-template-rows:repeat(2,minmax(0,1fr));gap:12px}.admin-stat-item{display:flex;flex-direction:column;justify-content:space-between;min-height:0;padding:16px 15px;text-align:left;border:1px solid var(--item-border);border-radius:20px;background:var(--item-bg);box-shadow:0 8px 24px -18px #0c2b6433}.admin-stat-item-fluid{grid-column:1 / -1;min-height:142px;padding:0;border:0;background:transparent;box-shadow:none}.admin-fluid-card{position:relative;isolation:isolate;display:flex;height:100%;min-height:0;overflow:hidden;padding:0;border:1px solid rgba(123,166,255,.5);border-radius:16px;background-color:#edf1f9;box-shadow:0 8px 24px #0000000a}.admin-fluid-fill{position:absolute;left:0;top:0;width:100%;height:100%;pointer-events:none}.admin-fluid-wave{position:absolute;left:50%;top:calc(100% - var(--fluid-level));width:2500px;height:2500px;transform:translate(-50%);transform-origin:50% 50%;transition:top .8s cubic-bezier(.4,0,.2,1)}.admin-fluid-wave-a{background-color:#afc8f599;border-radius:48.5%;animation:fluidSpin 20s linear infinite}.admin-fluid-wave-b{background-color:#91b2ebe0;border-radius:49%;animation:fluidSpin 25s linear infinite;animation-delay:-10s}.admin-fluid-content{position:relative;z-index:1;display:flex;align-items:center;gap:18px;width:100%;min-height:110px;padding:24px}.admin-fluid-icon{width:52px;height:52px;display:flex;align-items:center;justify-content:center;flex:none;border-radius:14px;background-color:#d7e1f4;color:#3b60c4;box-shadow:none}.admin-fluid-copy{min-width:0;flex:1}.admin-fluid-copy h3{margin:0 0 8px;font-size:22px;font-weight:600;line-height:1.15;color:#333}.admin-fluid-copy p{margin:0;font-size:13px;color:#778299}.admin-fluid-copy small{display:block;margin-top:10px;font-size:12px;color:var(--text-secondary)}.admin-stat-kicker{font-size:11px;font-weight:700;letter-spacing:.08em;color:var(--text-secondary)}.admin-stat-item h3{margin-top:10px;margin-bottom:6px;font-size:34px;font-weight:800;font-family:ui-monospace,monospace;line-height:1}.admin-stat-item p{font-size:13px;font-weight:500;color:var(--text-secondary)}.stat-suffix{font-size:16px}.admin-config-stack{display:flex;flex-direction:column;height:100%;gap:15px}.admin-config-card{height:100%}.admin-config-row{margin-bottom:0}.admin-config-row-field{flex-direction:column;align-items:stretch}.admin-config-row-last{margin-bottom:0}.admin-field-meta{display:flex;flex-direction:column;gap:6px}.admin-field-label{font-size:13px;font-weight:700;letter-spacing:.04em;color:var(--text-main)}.admin-field-hint{font-size:12px;line-height:1.5;color:var(--text-secondary)}.admin-field-control-row{display:flex;gap:12px}.admin-field-control-row input{flex:1}.admin-config-insights{display:flex;flex:1;flex-direction:column;gap:14px}.admin-config-highlight{padding:18px 20px;border:1px solid var(--item-border);border-radius:22px;background:linear-gradient(145deg,var(--item-bg-hover),var(--item-bg));box-shadow:0 16px 30px -24px #0c2b6459}.admin-config-badge{display:inline-flex;align-items:center;padding:6px 10px;border-radius:999px;background:var(--accent-cyan-light);color:var(--accent-blue);font-size:11px;font-weight:700;letter-spacing:.12em}.admin-config-highlight h3{margin-top:14px;margin-bottom:8px;font-size:30px;font-weight:800;font-family:ui-monospace,monospace;color:var(--text-main)}.admin-config-highlight p{font-size:13px;line-height:1.7;color:var(--text-secondary)}.admin-table-card{min-height:0}.admin-table-wrapper{max-height:420px;overflow-x:auto;overflow-y:auto;border-radius:18px}.admin-table-wrapper::-webkit-scrollbar{width:8px;height:8px}.admin-table-wrapper::-webkit-scrollbar-thumb{background:var(--divider-color);border-radius:999px}.admin-table{width:100%;margin-top:10px;border-collapse:collapse;font-size:14px;text-align:left}.admin-table th{position:sticky;top:0;z-index:1;padding:14px 12px;color:var(--text-secondary);border-bottom:1px solid var(--divider-line);background:var(--card-bg);font-weight:600;white-space:nowrap}.admin-table td{padding:14px 12px;color:var(--text-main);border-bottom:1px solid var(--item-border)}.admin-table tbody tr:last-child td{border-bottom:0}.admin-table tr:hover td{background-color:var(--item-bg-hover)}.hidden{display:none!important}.footer{position:absolute;right:0;bottom:24px;left:0;text-align:center;font-size:13px;color:var(--text-secondary);opacity:.6;line-height:1.8;letter-spacing:.5px;transition:var(--transition-base)}.footer:hover{opacity:1}.footer .divider-line{margin:0 8px;opacity:.5}.footer a{text-decoration:none}.footer a:hover{color:var(--text-main)}#admin-trigger{cursor:pointer;-webkit-user-select:none;user-select:none}@keyframes backgroundMove{0%{transform:scale(1) translate(0)}to{transform:scale(1.05) translate(2%,2%)}}@keyframes ripplePro{0%{transform:scale(.8);opacity:0}20%{opacity:.5}to{transform:scale(3);opacity:0}}@keyframes deviceBeaconPing{0%{opacity:0;transform:scale(.7)}22%{opacity:.9}to{opacity:0;transform:scale(2.8)}}@keyframes spin{to{transform:rotate(360deg)}}@keyframes showPanelPro{to{opacity:1;transform:scale(1)}}@keyframes fluidSpin{0%{transform:translate(-50%) rotate(0)}to{transform:translate(-50%) rotate(360deg)}}@media (min-width: 768px){.main-grid{grid-template-columns:1fr 1fr;gap:40px}}@media (min-width: 1024px){.container{max-width:1000px;padding:40px 0}.header h1{font-size:48px}.card{padding:40px}.btn-create,.btn-primary,input.room-code{padding:18px}.transfer-panel{max-width:700px}}@media (max-width: 767px){.admin-stats-row{grid-template-columns:1fr;grid-template-rows:none}.file-info,.file-info-right{align-items:flex-start}.file-info{flex-direction:column}.file-info-left,.file-info-right{max-width:100%;width:100%}.file-info-right{flex-wrap:wrap;row-gap:8px;justify-content:flex-end}.file-status{max-width:100%;overflow:hidden;text-overflow:ellipsis}}@media (max-width: 600px){body{padding-inline:16px}.theme-toggle{position:fixed;top:auto;right:20px;bottom:30px;width:50px;height:50px;margin-top:0;box-shadow:0 10px 30px #00000026}.card,.admin-header-card{padding:28px 22px}.header{margin-bottom:28px}.header h1{font-size:36px}.huge-code{width:100%;font-size:44px;letter-spacing:8px}.text-input-group{flex-direction:column}.text-input-group button{width:100%;min-height:52px}} +:root{--bg-color: #09090b;--card-bg: rgba(24, 24, 27, .65);--card-border: rgba(255, 255, 255, .08);--card-shadow: 0 20px 40px -10px rgba(0, 0, 0, .5), 0 0 2px rgba(255, 255, 255, .05) inset;--text-main: #f4f4f5;--text-secondary: #a1a1aa;--item-bg: rgba(255, 255, 255, .03);--item-bg-hover: rgba(255, 255, 255, .06);--item-border: rgba(255, 255, 255, .05);--icon-bg: rgba(255, 255, 255, .05);--input-bg: rgba(0, 0, 0, .3);--divider-color: rgba(255, 255, 255, .15);--divider-line: rgba(255, 255, 255, .08);--glow-1: rgba(0, 113, 227, .15);--glow-2: rgba(34, 211, 238, .12);--glow-3: rgba(168, 85, 247, .1);--accent-blue: #0a84ff;--accent-cyan: #30d158;--accent-cyan-light: rgba(10, 132, 255, .2);--success-green: #30d158;--danger-red: #ff453a;--border-radius-lg: 28px;--border-radius-sm: 18px;--transition-base: all .4s cubic-bezier(.2, .8, .2, 1);--font-stack: -apple-system, BlinkMacSystemFont, "SF Pro Text", "SF Pro Display", "Helvetica Neue", Roboto, sans-serif}body[data-theme=light]{--bg-color: #f2f4f8;--card-bg: rgba(255, 255, 255, .85);--card-border: rgba(255, 255, 255, 1);--card-shadow: 0 10px 30px -5px rgba(12, 43, 100, .08), 0 4px 10px -3px rgba(12, 43, 100, .04), 0 0 1px rgba(0, 0, 0, .05);--text-main: #1d1d1f;--text-secondary: #86868b;--item-bg: #ffffff;--item-bg-hover: #f8fafd;--item-border: rgba(0, 0, 0, .04);--icon-bg: #f5f5f7;--input-bg: #ffffff;--divider-color: #86868b;--divider-line: #e5e5ea;--glow-1: rgba(0, 113, 227, .05);--glow-2: rgba(34, 211, 238, .05);--glow-3: rgba(168, 85, 247, .03);--accent-blue: #0071e3;--accent-cyan: #0071e3;--accent-cyan-light: rgba(0, 113, 227, .1);--success-green: #34c759;--danger-red: #ff3b30}*{box-sizing:border-box;margin:0;padding:0;outline:none}html,body,#app{min-height:100%}body{font-family:var(--font-stack);background-color:var(--bg-color);color:var(--text-main);display:flex;justify-content:center;align-items:center;min-height:100vh;padding:20px 20px 100px;position:relative;overflow-x:hidden;overflow-y:auto;transition:background-color .5s ease,color .5s ease;-webkit-font-smoothing:antialiased}body:before{content:"";position:fixed;top:-25%;right:-25%;bottom:-25%;left:-25%;width:150%;height:150%;z-index:-1;background-image:radial-gradient(circle at 20% 20%,var(--glow-1) 0%,transparent 50%),radial-gradient(circle at 80% 20%,var(--glow-2) 0%,transparent 40%),radial-gradient(circle at 50% 80%,var(--glow-3) 0%,transparent 60%);animation:backgroundMove 25s infinite alternate ease-in-out}body,button,input{font:inherit}button{border:0;color:inherit}a{color:inherit}.app-icon{display:inline-flex;align-items:center;justify-content:center;flex:0 0 auto;line-height:1}.app-icon svg{width:100%;height:100%;overflow:visible}.container{width:100%;max-width:1000px;margin:auto;position:relative;z-index:1;transition:var(--transition-base)}.header{position:relative;display:flex;flex-direction:column;justify-content:center;align-items:center;min-height:80px;margin-top:30px;margin-bottom:40px;text-align:center}.header h1{font-size:42px;font-weight:800;letter-spacing:-1px;margin-bottom:8px}.header p{color:var(--text-secondary);font-size:16px;font-weight:400;letter-spacing:.5px}.theme-toggle{position:absolute;top:50%;right:0;width:44px;height:44px;margin-top:-22px;border-radius:50%;background:var(--card-bg);border:1px solid var(--card-border);color:var(--text-main);box-shadow:var(--card-shadow);cursor:pointer;display:flex;justify-content:center;align-items:center;transition:var(--transition-base);backdrop-filter:blur(20px);-webkit-backdrop-filter:blur(20px)}.theme-toggle:hover{transform:scale(1.08);box-shadow:0 8px 25px #0071e326}.main-grid{display:grid;grid-template-columns:1fr;gap:30px}.card{position:relative;display:flex;flex-direction:column;padding:40px 30px;background:var(--card-bg);border:1px solid var(--card-border);border-radius:var(--border-radius-lg);box-shadow:var(--card-shadow);transition:var(--transition-base);backdrop-filter:blur(40px) saturate(150%);-webkit-backdrop-filter:blur(40px) saturate(150%)}.card:before{content:"";position:absolute;inset:0 0 auto;height:1px;border-radius:var(--border-radius-lg) var(--border-radius-lg) 0 0;opacity:.5;background:linear-gradient(90deg,#fff0,#ffffff4d,#fff0)}.section-title{margin-bottom:25px;text-align:center;font-size:12px;font-weight:700;color:var(--text-secondary);text-transform:uppercase;letter-spacing:2px}.local-device-name{margin-top:-10px;margin-bottom:18px;text-align:center;font-size:13px;color:var(--text-secondary)}.local-device-name strong{color:var(--text-main);font-weight:600}.radar-container,.waiting-area{flex:1;min-height:200px;display:flex;flex-direction:column;justify-content:center;align-items:center;text-align:center}.radar{position:relative;width:80px;height:80px;margin-bottom:25px;display:flex;justify-content:center;align-items:center;border-radius:50%;background:var(--accent-cyan-light);border:1px solid rgba(0,113,227,.1);box-shadow:0 0 20px var(--accent-cyan-light)}.radar:before,.radar:after{content:"";position:absolute;top:0;right:0;bottom:0;left:0;border-radius:50%;border:2px solid var(--accent-blue);opacity:0;animation:ripplePro 3s infinite cubic-bezier(.1,.8,.3,1)}.radar:before{animation-delay:1.5s}.radar-icon{font-size:36px;color:var(--accent-blue)}.scan-status,.waiting-subtitle{margin-top:10px;font-size:14px;color:var(--text-secondary)}.device-list{width:100%;flex:1;display:flex;flex-direction:column;gap:12px}.device-item{position:relative;width:100%;display:flex;align-items:center;padding:16px 20px;background:var(--item-bg);border:1px solid var(--item-border);border-radius:var(--border-radius-sm);box-shadow:0 2px 8px #00000005;cursor:pointer;color:var(--text-main);text-align:left;transition:var(--transition-base)}.device-item:hover{background:var(--item-bg-hover);transform:translateY(-2px);border-color:#0071e34d;box-shadow:0 8px 20px -5px #0071e326,0 0 0 1px #0071e31a inset}.device-icon{width:48px;height:48px;margin-right:16px;display:flex;justify-content:center;align-items:center;border-radius:50%;background:var(--icon-bg);border:1px solid var(--item-border);transition:var(--transition-base)}.device-info{flex:1}.device-status-beacon{position:absolute;top:50%;right:22px;width:14px;height:14px;transform:translateY(-50%);pointer-events:none}.device-status-dot,.device-status-ring{position:absolute;top:0;right:0;bottom:0;left:0;border-radius:50%}.device-status-dot{top:3px;right:3px;bottom:3px;left:3px;background:var(--accent-blue);box-shadow:0 0 0 1px #ffffffbf,0 0 14px #0a84ff47}.device-status-ring{border:2px solid rgba(10,132,255,.22);opacity:0;transform:scale(.7);animation:deviceBeaconPing 2s infinite cubic-bezier(.16,.84,.44,1)}.device-status-ring-delay{animation-delay:1s}.device-icon .app-icon{position:relative;z-index:1;color:var(--text-main)}.device-item:hover .device-icon{background:var(--accent-cyan-light);border-color:transparent}.device-item:hover .device-icon .app-icon{color:var(--accent-blue)}.device-item:hover .device-status-dot{box-shadow:0 0 0 1px #ffffffd9,0 0 18px #0a84ff5c}.device-item:hover .device-status-ring{border-color:#0a84ff4d;animation-duration:1.4s}.device-info h4{margin-bottom:4px;font-size:16px;font-weight:600;color:var(--text-main)}.device-info p{font-size:13px;color:var(--text-secondary)}.room-action-area{flex:1;display:flex;flex-direction:column;justify-content:center}.btn-create{width:100%;margin-bottom:25px;padding:18px;display:flex;align-items:center;justify-content:center;gap:8px;background:var(--item-bg);color:var(--accent-blue);border:1px solid var(--item-border);border-radius:var(--border-radius-sm);box-shadow:0 2px 8px #00000005;font-size:16px;font-weight:600;cursor:pointer;transition:var(--transition-base)}.btn-create:hover{background:var(--item-bg-hover);border-color:#0071e34d;transform:translateY(-2px);box-shadow:0 8px 20px -5px #0071e326}.divider{position:relative;margin-bottom:25px;text-align:center;color:var(--divider-color);font-size:12px;font-weight:500;text-transform:uppercase;letter-spacing:1px}.divider:before,.divider:after{content:"";position:absolute;top:50%;width:35%;height:1px;background:var(--divider-line)}.divider:before{left:0}.divider:after{right:0}.room-input-group{display:flex;flex-direction:column;gap:12px}.pending-downloads{margin-top:22px;padding:16px 10px 16px 16px;border:1px solid var(--item-border);border-radius:var(--border-radius-sm);background:var(--item-bg);max-height:236px;overflow-y:auto}.pending-downloads::-webkit-scrollbar{width:6px}.pending-downloads::-webkit-scrollbar-thumb{background:var(--divider-color);border-radius:999px}.pending-downloads-head{display:flex;align-items:center;justify-content:space-between;margin-bottom:12px;font-size:12px;font-weight:700;letter-spacing:.08em;color:var(--text-secondary);position:sticky;top:0;z-index:1;background:var(--item-bg);padding-bottom:8px}.pending-download-item{display:flex;align-items:center;justify-content:space-between;gap:12px;width:100%;min-width:0;box-sizing:border-box;padding:12px 14px;border-radius:16px;background:var(--input-bg);border:1px solid var(--item-border);text-decoration:none;transition:var(--transition-base)}.pending-download-item+.pending-download-item{margin-top:10px}.pending-download-item:hover{background:var(--item-bg-hover);border-color:#0071e340;color:var(--accent-blue)}.pending-download-copy{flex:1;min-width:0;overflow:hidden}.pending-download-copy strong{display:-webkit-box;margin-bottom:4px;overflow:hidden;text-overflow:ellipsis;white-space:normal;overflow-wrap:anywhere;-webkit-box-orient:vertical;-webkit-line-clamp:2;line-height:1.35;max-height:2.7em;font-size:14px;font-weight:600;color:var(--text-main)}.pending-download-copy p{overflow:hidden;text-overflow:ellipsis;white-space:nowrap;font-size:12px;color:var(--text-secondary)}.pending-download-icon{flex:none}input.room-code,.text-input-group input,.text-input-group textarea{width:100%;min-width:0;padding:18px;border:1px solid var(--item-border);border-radius:var(--border-radius-sm);background:var(--input-bg);color:var(--text-main);box-shadow:inset 0 2px 4px #00000005;transition:var(--transition-base)}input.room-code{font-size:24px;font-weight:700;text-align:center;letter-spacing:8px;font-family:ui-monospace,SFMono-Regular,Menlo,monospace}input.room-code:focus,.text-input-group input:focus,.text-input-group textarea:focus{background:var(--item-bg);border-color:var(--accent-blue);box-shadow:0 0 0 3px var(--accent-cyan-light)}input.room-code::placeholder{letter-spacing:normal;font-size:16px;font-weight:400;color:var(--text-secondary);opacity:.6}.btn-primary{width:100%;padding:18px;background:var(--accent-blue);color:#fff;border-radius:var(--border-radius-sm);box-shadow:0 4px 12px #0071e34d;font-size:16px;font-weight:600;cursor:pointer;transition:var(--transition-base)}.btn-primary:hover,.text-input-group button:hover{background:#06c;transform:translateY(-2px);box-shadow:0 6px 16px #0071e366}.huge-code{margin:15px 0 25px;padding:15px 30px;font-size:56px;font-weight:800;color:var(--accent-blue);letter-spacing:10px;font-family:ui-monospace,monospace;border-radius:20px;border:1px solid rgba(0,113,227,.2);background:var(--accent-cyan-light)}.spinner{width:28px;height:28px;margin:0 auto 15px;border-radius:50%;border:3px solid var(--item-border);border-top-color:var(--accent-blue);animation:spin .8s linear infinite}.waiting-tip{font-size:14px;font-weight:500;color:var(--accent-blue)}.btn-cancel{margin-top:25px;background:none;color:var(--text-secondary);font-size:14px;cursor:pointer;transition:var(--transition-base)}.btn-cancel:hover{color:var(--text-main)}.transfer-panel,.admin-panel{width:100%;margin:0 auto;opacity:0;transform:scale(.95);animation:showPanelPro .5s forwards cubic-bezier(.2,.8,.2,1)}.transfer-panel{max-width:650px}.transfer-head{display:flex;align-items:center;justify-content:space-between;margin-bottom:30px;padding-bottom:20px;border-bottom:1px solid var(--divider-line)}.connected-to h2{margin-bottom:4px;font-size:20px;font-weight:700;color:var(--text-main)}.connected-to p{display:flex;align-items:center;gap:6px;font-size:13px;font-weight:500;color:var(--success-green)}.connection-hint{display:block;margin-top:8px;max-width:42ch;font-size:12px;line-height:1.5;color:var(--text-secondary)}.close-btn{width:36px;height:36px;display:flex;justify-content:center;align-items:center;border-radius:50%;background:var(--icon-bg);border:1px solid var(--item-border);color:var(--text-secondary);cursor:pointer;transition:var(--transition-base)}.close-btn:hover{background:var(--item-bg-hover);color:var(--text-main);transform:rotate(90deg)}.drop-zone{padding:40px 30px;margin-bottom:20px;text-align:center;background:var(--input-bg);border:2px dashed var(--divider-color);border-radius:var(--border-radius-sm);cursor:pointer;transition:var(--transition-base)}.drop-zone:hover,.drop-zone-active{border-color:var(--accent-blue);background-color:var(--accent-cyan-light)}.drop-zone-icon{margin-bottom:12px;font-size:42px;color:var(--text-secondary);transition:var(--transition-base)}.drop-zone:hover .drop-zone-icon,.drop-zone-active .drop-zone-icon{transform:translateY(-5px);color:var(--accent-blue)}.drop-zone-text{font-size:15px;font-weight:500;color:var(--text-main)}.text-input-group{display:flex;gap:12px;margin-bottom:20px}.text-input-group input{flex:1;padding:16px 20px;font-size:15px}.text-input-group button{width:52px;display:flex;align-items:center;justify-content:center;border-radius:var(--border-radius-sm);background:var(--accent-blue);color:#fff;box-shadow:0 4px 12px #0071e34d;cursor:pointer;transition:var(--transition-base)}.batch-actions{display:flex;justify-content:flex-end;margin-bottom:12px;height:0;overflow:hidden;opacity:0;pointer-events:none;transition:var(--transition-base)}.batch-actions.active{height:36px;overflow:visible;opacity:1;pointer-events:auto}.btn-small-primary{display:flex;align-items:center;gap:6px;padding:8px 16px;background:var(--item-bg);color:var(--text-main);border:1px solid var(--item-border);border-radius:12px;box-shadow:0 2px 6px #0000000a;font-size:13px;font-weight:600;cursor:pointer;transition:var(--transition-base)}.btn-small-primary:hover{background:var(--item-bg-hover);color:var(--accent-blue);border-color:#0071e34d;transform:translateY(-1px)}.batch-progress-container{max-height:280px;overflow-y:auto;padding-right:8px;padding-bottom:10px;text-align:left}.batch-progress-container::-webkit-scrollbar{width:6px}.batch-progress-container::-webkit-scrollbar-thumb{background:var(--divider-color);border-radius:4px}.batch-item{margin-bottom:12px;width:100%;min-width:0;box-sizing:border-box;padding:16px;background:var(--item-bg);border:1px solid var(--item-border);border-radius:var(--border-radius-sm);box-shadow:0 2px 8px #00000005;transition:all .3s ease}.batch-item:hover{border-color:#00000014;box-shadow:0 4px 12px #0000000a}body[data-theme=dark] .batch-item:hover{border-color:#ffffff1a}.file-info{display:flex;justify-content:space-between;align-items:center;gap:10px;min-width:0;margin-bottom:12px;font-size:14px}.file-info-left{flex:1;min-width:0;max-width:60%;display:flex;align-items:center;gap:10px;overflow:hidden}.file-icon-wrapper{flex:none;width:32px;height:32px;display:flex;justify-content:center;align-items:center;border-radius:8px;background:var(--icon-bg);color:var(--accent-blue)}.file-name{display:-webkit-box;flex:1;min-width:0;overflow:hidden;text-overflow:ellipsis;white-space:normal;overflow-wrap:anywhere;-webkit-box-orient:vertical;-webkit-line-clamp:2;line-height:1.35;max-height:2.7em;font-weight:500;color:var(--text-main)}.file-info-right{display:flex;flex:none;align-items:center;min-width:0;gap:8px}.file-status{margin-right:4px;white-space:nowrap;font-size:12px;font-weight:500;font-variant-numeric:tabular-nums}.action-btn{width:28px;height:28px;display:flex;justify-content:center;align-items:center;border-radius:8px;background:transparent;border:1px solid transparent;color:var(--text-secondary);cursor:pointer;transition:var(--transition-base)}.action-btn:hover{color:var(--text-main);background:var(--icon-bg);border-color:var(--item-border)}.action-btn.primary:hover{color:var(--accent-blue);background:var(--accent-cyan-light)}.action-btn.danger:hover{color:var(--danger-red);background:#ff3b301a}.progress-bar-container{height:6px;overflow:hidden;background:var(--icon-bg);border-radius:4px}.progress-bar-fill{height:100%;width:0;background:var(--accent-blue);border-radius:4px;transition:width .1s linear}.progress-bar-fill.success{background:var(--success-green)}.admin-header-card{margin-bottom:30px;padding:30px}.transfer-head-compact{margin-bottom:0;padding-bottom:0;border-bottom:0}.admin-title{font-size:24px;color:var(--text-main)}.connected-to .admin-subtitle{display:block;margin-top:4px;color:var(--text-secondary)}.admin-summary-grid{margin-bottom:30px}.admin-stats-card{height:100%}.admin-stats-panel{display:flex;flex-direction:column;height:100%}.admin-stats-row{flex:1;display:grid;grid-template-columns:repeat(2,minmax(0,1fr));grid-template-rows:repeat(2,minmax(0,1fr));gap:12px}.admin-stat-item{display:flex;flex-direction:column;justify-content:space-between;min-height:0;padding:16px 15px;text-align:left;border:1px solid var(--item-border);border-radius:20px;background:var(--item-bg);box-shadow:0 8px 24px -18px #0c2b6433}.admin-stat-item-fluid{grid-column:1 / -1;min-height:142px;padding:0;border:0;background:transparent;box-shadow:none}.admin-fluid-card{position:relative;isolation:isolate;display:flex;height:100%;min-height:0;overflow:hidden;padding:0;border:1px solid rgba(123,166,255,.5);border-radius:16px;background-color:#edf1f9;box-shadow:0 8px 24px #0000000a}.admin-fluid-fill{position:absolute;left:0;top:0;width:100%;height:100%;pointer-events:none}.admin-fluid-wave{position:absolute;left:50%;top:calc(100% - var(--fluid-level));width:2500px;height:2500px;transform:translate(-50%);transform-origin:50% 50%;transition:top .8s cubic-bezier(.4,0,.2,1)}.admin-fluid-wave-a{background-color:#afc8f599;border-radius:48.5%;animation:fluidSpin 20s linear infinite}.admin-fluid-wave-b{background-color:#91b2ebe0;border-radius:49%;animation:fluidSpin 25s linear infinite;animation-delay:-10s}.admin-fluid-content{position:relative;z-index:1;display:flex;align-items:center;gap:18px;width:100%;min-height:110px;padding:24px}.admin-fluid-icon{width:52px;height:52px;display:flex;align-items:center;justify-content:center;flex:none;border-radius:14px;background-color:#d7e1f4;color:#3b60c4;box-shadow:none}.admin-fluid-copy{min-width:0;flex:1}.admin-fluid-copy h3{margin:0 0 8px;font-size:22px;font-weight:600;line-height:1.15;color:#333}.admin-fluid-copy p{margin:0;font-size:13px;color:#778299}.admin-fluid-copy small{display:block;margin-top:10px;font-size:12px;color:var(--text-secondary)}.admin-stat-kicker{font-size:11px;font-weight:700;letter-spacing:.08em;color:var(--text-secondary)}.admin-stat-item h3{margin-top:10px;margin-bottom:6px;font-size:34px;font-weight:800;font-family:ui-monospace,monospace;line-height:1}.admin-stat-item p{font-size:13px;font-weight:500;color:var(--text-secondary)}.stat-suffix{font-size:16px}.admin-config-stack{display:flex;flex-direction:column;height:100%;gap:15px}.admin-config-card{height:100%}.admin-config-row{margin-bottom:0}.admin-config-row-field{flex-direction:column;align-items:stretch}.admin-config-row-last{margin-bottom:0}.admin-field-meta{display:flex;flex-direction:column;gap:6px}.admin-field-label{font-size:13px;font-weight:700;letter-spacing:.04em;color:var(--text-main)}.admin-field-hint{font-size:12px;line-height:1.5;color:var(--text-secondary)}.admin-field-control-row{display:flex;gap:12px}.admin-field-control-row input{flex:1}.admin-config-insights{display:flex;flex:1;flex-direction:column;gap:14px}.admin-config-highlight{padding:18px 20px;border:1px solid var(--item-border);border-radius:22px;background:linear-gradient(145deg,var(--item-bg-hover),var(--item-bg));box-shadow:0 16px 30px -24px #0c2b6459}.admin-config-badge{display:inline-flex;align-items:center;padding:6px 10px;border-radius:999px;background:var(--accent-cyan-light);color:var(--accent-blue);font-size:11px;font-weight:700;letter-spacing:.12em}.admin-config-highlight h3{margin-top:14px;margin-bottom:8px;font-size:30px;font-weight:800;font-family:ui-monospace,monospace;color:var(--text-main)}.admin-config-highlight p{font-size:13px;line-height:1.7;color:var(--text-secondary)}.admin-table-card{min-height:0}.admin-table-wrapper{max-height:420px;overflow-x:auto;overflow-y:auto;border-radius:18px}.admin-table-wrapper::-webkit-scrollbar{width:8px;height:8px}.admin-table-wrapper::-webkit-scrollbar-thumb{background:var(--divider-color);border-radius:999px}.admin-table{width:100%;margin-top:10px;border-collapse:collapse;font-size:14px;text-align:left}.admin-table th{position:sticky;top:0;z-index:1;padding:14px 12px;color:var(--text-secondary);border-bottom:1px solid var(--divider-line);background:var(--card-bg);font-weight:600;white-space:nowrap}.admin-table td{padding:14px 12px;color:var(--text-main);border-bottom:1px solid var(--item-border)}.admin-table tbody tr:last-child td{border-bottom:0}.admin-table tr:hover td{background-color:var(--item-bg-hover)}.hidden{display:none!important}.footer{position:absolute;right:0;bottom:24px;left:0;text-align:center;font-size:13px;color:var(--text-secondary);opacity:.6;line-height:1.8;letter-spacing:.5px;transition:var(--transition-base)}.footer:hover{opacity:1}.footer .divider-line{margin:0 8px;opacity:.5}.footer a{text-decoration:none}.footer a:hover{color:var(--text-main)}#admin-trigger{cursor:pointer;-webkit-user-select:none;user-select:none}@keyframes backgroundMove{0%{transform:scale(1) translate(0)}to{transform:scale(1.05) translate(2%,2%)}}@keyframes ripplePro{0%{transform:scale(.8);opacity:0}20%{opacity:.5}to{transform:scale(3);opacity:0}}@keyframes deviceBeaconPing{0%{opacity:0;transform:scale(.7)}22%{opacity:.9}to{opacity:0;transform:scale(2.8)}}@keyframes spin{to{transform:rotate(360deg)}}@keyframes showPanelPro{to{opacity:1;transform:scale(1)}}@keyframes fluidSpin{0%{transform:translate(-50%) rotate(0)}to{transform:translate(-50%) rotate(360deg)}}@media (min-width: 768px){.main-grid{grid-template-columns:1fr 1fr;gap:40px}}@media (min-width: 1024px){.container{max-width:1000px;padding:40px 0}.header h1{font-size:48px}.card{padding:40px}.btn-create,.btn-primary,input.room-code{padding:18px}.transfer-panel{max-width:700px}}@media (max-width: 767px){.admin-stats-row{grid-template-columns:1fr;grid-template-rows:none}.file-info,.file-info-right{align-items:flex-start}.file-info{flex-direction:column}.file-info-left,.file-info-right{max-width:100%;width:100%}.file-info-right{flex-wrap:wrap;row-gap:8px;justify-content:flex-end}.file-status{max-width:100%;overflow:hidden;text-overflow:ellipsis}}@media (max-width: 600px){body{padding-inline:16px}.theme-toggle{position:fixed;top:auto;right:20px;bottom:30px;width:50px;height:50px;margin-top:0;box-shadow:0 10px 30px #00000026}.card,.admin-header-card{padding:28px 22px}.header{margin-bottom:28px}.header h1{font-size:36px}.huge-code{width:100%;font-size:44px;letter-spacing:8px}.text-input-group{flex-direction:column}.text-input-group button{width:100%;min-height:52px}} diff --git a/frontend/dist/assets/index-DE3lDjdM.js b/frontend/dist/assets/index-DE3lDjdM.js deleted file mode 100644 index 04710aa..0000000 --- a/frontend/dist/assets/index-DE3lDjdM.js +++ /dev/null @@ -1,17 +0,0 @@ -(function(){const t=document.createElement("link").relList;if(t&&t.supports&&t.supports("modulepreload"))return;for(const i of document.querySelectorAll('link[rel="modulepreload"]'))s(i);new MutationObserver(i=>{for(const r of i)if(r.type==="childList")for(const l of r.addedNodes)l.tagName==="LINK"&&l.rel==="modulepreload"&&s(l)}).observe(document,{childList:!0,subtree:!0});function n(i){const r={};return i.integrity&&(r.integrity=i.integrity),i.referrerPolicy&&(r.referrerPolicy=i.referrerPolicy),i.crossOrigin==="use-credentials"?r.credentials="include":i.crossOrigin==="anonymous"?r.credentials="omit":r.credentials="same-origin",r}function s(i){if(i.ep)return;i.ep=!0;const r=n(i);fetch(i.href,r)}})();/** -* @vue/shared v3.5.30 -* (c) 2018-present Yuxi (Evan) You and Vue contributors -* @license MIT -**/function Ds(e){const t=Object.create(null);for(const n of e.split(","))t[n]=1;return n=>n in t}const ie={},Wt=[],tt=()=>{},rr=()=>!1,Gn=e=>e.charCodeAt(0)===111&&e.charCodeAt(1)===110&&(e.charCodeAt(2)>122||e.charCodeAt(2)<97),Fs=e=>e.startsWith("onUpdate:"),Te=Object.assign,Ns=(e,t)=>{const n=e.indexOf(t);n>-1&&e.splice(n,1)},Ko=Object.prototype.hasOwnProperty,Z=(e,t)=>Ko.call(e,t),B=Array.isArray,Vt=e=>wn(e)==="[object Map]",or=e=>wn(e)==="[object Set]",wi=e=>wn(e)==="[object Date]",z=e=>typeof e=="function",ae=e=>typeof e=="string",qe=e=>typeof e=="symbol",te=e=>e!==null&&typeof e=="object",lr=e=>(te(e)||z(e))&&z(e.then)&&z(e.catch),ar=Object.prototype.toString,wn=e=>ar.call(e),Wo=e=>wn(e).slice(8,-1),cr=e=>wn(e)==="[object Object]",Us=e=>ae(e)&&e!=="NaN"&&e[0]!=="-"&&""+parseInt(e,10)===e,cn=Ds(",key,ref,ref_for,ref_key,onVnodeBeforeMount,onVnodeMounted,onVnodeBeforeUpdate,onVnodeUpdated,onVnodeBeforeUnmount,onVnodeUnmounted"),Jn=e=>{const t=Object.create(null);return n=>t[n]||(t[n]=e(n))},Vo=/-\w/g,Ee=Jn(e=>e.replace(Vo,t=>t.slice(1).toUpperCase())),qo=/\B([A-Z])/g,It=Jn(e=>e.replace(qo,"-$1").toLowerCase()),Yn=Jn(e=>e.charAt(0).toUpperCase()+e.slice(1)),fs=Jn(e=>e?`on${Yn(e)}`:""),et=(e,t)=>!Object.is(e,t),Un=(e,...t)=>{for(let n=0;n{Object.defineProperty(e,t,{configurable:!0,enumerable:!1,writable:s,value:n})},Bs=e=>{const t=parseFloat(e);return isNaN(t)?e:t};let xi;const Xn=()=>xi||(xi=typeof globalThis<"u"?globalThis:typeof self<"u"?self:typeof window<"u"?window:typeof global<"u"?global:{});function Be(e){if(B(e)){const t={};for(let n=0;n{if(n){const s=n.split(Jo);s.length>1&&(t[s[0].trim()]=s[1].trim())}}),t}function xt(e){let t="";if(ae(e))t=e;else if(B(e))for(let n=0;n!!(e&&e.__v_isRef===!0),se=e=>ae(e)?e:e==null?"":B(e)||te(e)&&(e.toString===ar||!z(e.toString))?dr(e)?se(e.value):JSON.stringify(e,pr,2):String(e),pr=(e,t)=>dr(t)?pr(e,t.value):Vt(t)?{[`Map(${t.size})`]:[...t.entries()].reduce((n,[s,i],r)=>(n[ds(s,r)+" =>"]=i,n),{})}:or(t)?{[`Set(${t.size})`]:[...t.values()].map(n=>ds(n))}:qe(t)?ds(t):te(t)&&!B(t)&&!cr(t)?String(t):t,ds=(e,t="")=>{var n;return qe(e)?`Symbol(${(n=e.description)!=null?n:t})`:e};/** -* @vue/reactivity v3.5.30 -* (c) 2018-present Yuxi (Evan) You and Vue contributors -* @license MIT -**/let Oe;class tl{constructor(t=!1){this.detached=t,this._active=!0,this._on=0,this.effects=[],this.cleanups=[],this._isPaused=!1,this.__v_skip=!0,this.parent=Oe,!t&&Oe&&(this.index=(Oe.scopes||(Oe.scopes=[])).push(this)-1)}get active(){return this._active}pause(){if(this._active){this._isPaused=!0;let t,n;if(this.scopes)for(t=0,n=this.scopes.length;t0&&--this._on===0&&(Oe=this.prevScope,this.prevScope=void 0)}stop(t){if(this._active){this._active=!1;let n,s;for(n=0,s=this.effects.length;n0)return;if(fn){let t=fn;for(fn=void 0;t;){const n=t.next;t.next=void 0,t.flags&=-9,t=n}}let e;for(;un;){let t=un;for(un=void 0;t;){const n=t.next;if(t.next=void 0,t.flags&=-9,t.flags&1)try{t.trigger()}catch(s){e||(e=s)}t=n}}if(e)throw e}function vr(e){for(let t=e.deps;t;t=t.nextDep)t.version=-1,t.prevActiveLink=t.dep.activeLink,t.dep.activeLink=t}function yr(e){let t,n=e.depsTail,s=n;for(;s;){const i=s.prevDep;s.version===-1?(s===n&&(n=i),zs(s),sl(s)):t=s,s.dep.activeLink=s.prevActiveLink,s.prevActiveLink=void 0,s=i}e.deps=t,e.depsTail=n}function Ss(e){for(let t=e.deps;t;t=t.nextDep)if(t.dep.version!==t.version||t.dep.computed&&(_r(t.dep.computed)||t.dep.version!==t.version))return!0;return!!e._dirty}function _r(e){if(e.flags&4&&!(e.flags&16)||(e.flags&=-17,e.globalVersion===gn)||(e.globalVersion=gn,!e.isSSR&&e.flags&128&&(!e.deps&&!e._dirty||!Ss(e))))return;e.flags|=2;const t=e.dep,n=re,s=Ve;re=e,Ve=!0;try{vr(e);const i=e.fn(e._value);(t.version===0||et(i,e._value))&&(e.flags|=128,e._value=i,t.version++)}catch(i){throw t.version++,i}finally{re=n,Ve=s,yr(e),e.flags&=-3}}function zs(e,t=!1){const{dep:n,prevSub:s,nextSub:i}=e;if(s&&(s.nextSub=i,e.prevSub=void 0),i&&(i.prevSub=s,e.nextSub=void 0),n.subs===e&&(n.subs=s,!s&&n.computed)){n.computed.flags&=-5;for(let r=n.computed.deps;r;r=r.nextDep)zs(r,!0)}!t&&!--n.sc&&n.map&&n.map.delete(n.key)}function sl(e){const{prevDep:t,nextDep:n}=e;t&&(t.nextDep=n,e.prevDep=void 0),n&&(n.prevDep=t,e.nextDep=void 0)}let Ve=!0;const br=[];function ft(){br.push(Ve),Ve=!1}function dt(){const e=br.pop();Ve=e===void 0?!0:e}function Ii(e){const{cleanup:t}=e;if(e.cleanup=void 0,t){const n=re;re=void 0;try{t()}finally{re=n}}}let gn=0;class il{constructor(t,n){this.sub=t,this.dep=n,this.version=n.version,this.nextDep=this.prevDep=this.nextSub=this.prevSub=this.prevActiveLink=void 0}}class Ks{constructor(t){this.computed=t,this.version=0,this.activeLink=void 0,this.subs=void 0,this.map=void 0,this.key=void 0,this.sc=0,this.__v_skip=!0}track(t){if(!re||!Ve||re===this.computed)return;let n=this.activeLink;if(n===void 0||n.sub!==re)n=this.activeLink=new il(re,this),re.deps?(n.prevDep=re.depsTail,re.depsTail.nextDep=n,re.depsTail=n):re.deps=re.depsTail=n,wr(n);else if(n.version===-1&&(n.version=this.version,n.nextDep)){const s=n.nextDep;s.prevDep=n.prevDep,n.prevDep&&(n.prevDep.nextDep=s),n.prevDep=re.depsTail,n.nextDep=void 0,re.depsTail.nextDep=n,re.depsTail=n,re.deps===n&&(re.deps=s)}return n}trigger(t){this.version++,gn++,this.notify(t)}notify(t){js();try{for(let n=this.subs;n;n=n.prevSub)n.sub.notify()&&n.sub.dep.notify()}finally{Hs()}}}function wr(e){if(e.dep.sc++,e.sub.flags&4){const t=e.dep.computed;if(t&&!e.dep.subs){t.flags|=20;for(let s=t.deps;s;s=s.nextDep)wr(s)}const n=e.dep.subs;n!==e&&(e.prevSub=n,n&&(n.nextSub=e)),e.dep.subs=e}}const Cs=new WeakMap,Ot=Symbol(""),Ts=Symbol(""),vn=Symbol("");function Ie(e,t,n){if(Ve&&re){let s=Cs.get(e);s||Cs.set(e,s=new Map);let i=s.get(n);i||(s.set(n,i=new Ks),i.map=s,i.key=n),i.track()}}function ct(e,t,n,s,i,r){const l=Cs.get(e);if(!l){gn++;return}const a=f=>{f&&f.trigger()};if(js(),t==="clear")l.forEach(a);else{const f=B(e),m=f&&Us(n);if(f&&n==="length"){const p=Number(s);l.forEach((_,T)=>{(T==="length"||T===vn||!qe(T)&&T>=p)&&a(_)})}else switch((n!==void 0||l.has(void 0))&&a(l.get(n)),m&&a(l.get(vn)),t){case"add":f?m&&a(l.get("length")):(a(l.get(Ot)),Vt(e)&&a(l.get(Ts)));break;case"delete":f||(a(l.get(Ot)),Vt(e)&&a(l.get(Ts)));break;case"set":Vt(e)&&a(l.get(Ot));break}}Hs()}function Lt(e){const t=X(e);return t===e?t:(Ie(t,"iterate",vn),Le(e)?t:t.map(Ge))}function Zn(e){return Ie(e=X(e),"iterate",vn),e}function Ze(e,t){return pt(e)?Xt(Pt(e)?Ge(t):t):Ge(t)}const rl={__proto__:null,[Symbol.iterator](){return hs(this,Symbol.iterator,e=>Ze(this,e))},concat(...e){return Lt(this).concat(...e.map(t=>B(t)?Lt(t):t))},entries(){return hs(this,"entries",e=>(e[1]=Ze(this,e[1]),e))},every(e,t){return ot(this,"every",e,t,void 0,arguments)},filter(e,t){return ot(this,"filter",e,t,n=>n.map(s=>Ze(this,s)),arguments)},find(e,t){return ot(this,"find",e,t,n=>Ze(this,n),arguments)},findIndex(e,t){return ot(this,"findIndex",e,t,void 0,arguments)},findLast(e,t){return ot(this,"findLast",e,t,n=>Ze(this,n),arguments)},findLastIndex(e,t){return ot(this,"findLastIndex",e,t,void 0,arguments)},forEach(e,t){return ot(this,"forEach",e,t,void 0,arguments)},includes(...e){return ms(this,"includes",e)},indexOf(...e){return ms(this,"indexOf",e)},join(e){return Lt(this).join(e)},lastIndexOf(...e){return ms(this,"lastIndexOf",e)},map(e,t){return ot(this,"map",e,t,void 0,arguments)},pop(){return on(this,"pop")},push(...e){return on(this,"push",e)},reduce(e,...t){return Si(this,"reduce",e,t)},reduceRight(e,...t){return Si(this,"reduceRight",e,t)},shift(){return on(this,"shift")},some(e,t){return ot(this,"some",e,t,void 0,arguments)},splice(...e){return on(this,"splice",e)},toReversed(){return Lt(this).toReversed()},toSorted(e){return Lt(this).toSorted(e)},toSpliced(...e){return Lt(this).toSpliced(...e)},unshift(...e){return on(this,"unshift",e)},values(){return hs(this,"values",e=>Ze(this,e))}};function hs(e,t,n){const s=Zn(e),i=s[t]();return s!==e&&!Le(e)&&(i._next=i.next,i.next=()=>{const r=i._next();return r.done||(r.value=n(r.value)),r}),i}const ol=Array.prototype;function ot(e,t,n,s,i,r){const l=Zn(e),a=l!==e&&!Le(e),f=l[t];if(f!==ol[t]){const _=f.apply(e,r);return a?Ge(_):_}let m=n;l!==e&&(a?m=function(_,T){return n.call(this,Ze(e,_),T,e)}:n.length>2&&(m=function(_,T){return n.call(this,_,T,e)}));const p=f.call(l,m,s);return a&&i?i(p):p}function Si(e,t,n,s){const i=Zn(e),r=i!==e&&!Le(e);let l=n,a=!1;i!==e&&(r?(a=s.length===0,l=function(m,p,_){return a&&(a=!1,m=Ze(e,m)),n.call(this,m,Ze(e,p),_,e)}):n.length>3&&(l=function(m,p,_){return n.call(this,m,p,_,e)}));const f=i[t](l,...s);return a?Ze(e,f):f}function ms(e,t,n){const s=X(e);Ie(s,"iterate",vn);const i=s[t](...n);return(i===-1||i===!1)&&Gs(n[0])?(n[0]=X(n[0]),s[t](...n)):i}function on(e,t,n=[]){ft(),js();const s=X(e)[t].apply(e,n);return Hs(),dt(),s}const ll=Ds("__proto__,__v_isRef,__isVue"),xr=new Set(Object.getOwnPropertyNames(Symbol).filter(e=>e!=="arguments"&&e!=="caller").map(e=>Symbol[e]).filter(qe));function al(e){qe(e)||(e=String(e));const t=X(this);return Ie(t,"has",e),t.hasOwnProperty(e)}class Ir{constructor(t=!1,n=!1){this._isReadonly=t,this._isShallow=n}get(t,n,s){if(n==="__v_skip")return t.__v_skip;const i=this._isReadonly,r=this._isShallow;if(n==="__v_isReactive")return!i;if(n==="__v_isReadonly")return i;if(n==="__v_isShallow")return r;if(n==="__v_raw")return s===(i?r?yl:$r:r?Tr:Cr).get(t)||Object.getPrototypeOf(t)===Object.getPrototypeOf(s)?t:void 0;const l=B(t);if(!i){let f;if(l&&(f=rl[n]))return f;if(n==="hasOwnProperty")return al}const a=Reflect.get(t,n,Ce(t)?t:s);if((qe(n)?xr.has(n):ll(n))||(i||Ie(t,"get",n),r))return a;if(Ce(a)){const f=l&&Us(n)?a:a.value;return i&&te(f)?Ms(f):f}return te(a)?i?Ms(a):Vs(a):a}}class Sr extends Ir{constructor(t=!1){super(!1,t)}set(t,n,s,i){let r=t[n];const l=B(t)&&Us(n);if(!this._isShallow){const m=pt(r);if(!Le(s)&&!pt(s)&&(r=X(r),s=X(s)),!l&&Ce(r)&&!Ce(s))return m||(r.value=s),!0}const a=l?Number(n)e,Mn=e=>Reflect.getPrototypeOf(e);function pl(e,t,n){return function(...s){const i=this.__v_raw,r=X(i),l=Vt(r),a=e==="entries"||e===Symbol.iterator&&l,f=e==="keys"&&l,m=i[e](...s),p=n?$s:t?Xt:Ge;return!t&&Ie(r,"iterate",f?Ts:Ot),Te(Object.create(m),{next(){const{value:_,done:T}=m.next();return T?{value:_,done:T}:{value:a?[p(_[0]),p(_[1])]:p(_),done:T}}})}}function kn(e){return function(...t){return e==="delete"?!1:e==="clear"?void 0:this}}function hl(e,t){const n={get(i){const r=this.__v_raw,l=X(r),a=X(i);e||(et(i,a)&&Ie(l,"get",i),Ie(l,"get",a));const{has:f}=Mn(l),m=t?$s:e?Xt:Ge;if(f.call(l,i))return m(r.get(i));if(f.call(l,a))return m(r.get(a));r!==l&&r.get(i)},get size(){const i=this.__v_raw;return!e&&Ie(X(i),"iterate",Ot),i.size},has(i){const r=this.__v_raw,l=X(r),a=X(i);return e||(et(i,a)&&Ie(l,"has",i),Ie(l,"has",a)),i===a?r.has(i):r.has(i)||r.has(a)},forEach(i,r){const l=this,a=l.__v_raw,f=X(a),m=t?$s:e?Xt:Ge;return!e&&Ie(f,"iterate",Ot),a.forEach((p,_)=>i.call(r,m(p),m(_),l))}};return Te(n,e?{add:kn("add"),set:kn("set"),delete:kn("delete"),clear:kn("clear")}:{add(i){const r=X(this),l=Mn(r),a=X(i),f=!t&&!Le(i)&&!pt(i)?a:i;return l.has.call(r,f)||et(i,f)&&l.has.call(r,i)||et(a,f)&&l.has.call(r,a)||(r.add(f),ct(r,"add",f,f)),this},set(i,r){!t&&!Le(r)&&!pt(r)&&(r=X(r));const l=X(this),{has:a,get:f}=Mn(l);let m=a.call(l,i);m||(i=X(i),m=a.call(l,i));const p=f.call(l,i);return l.set(i,r),m?et(r,p)&&ct(l,"set",i,r):ct(l,"add",i,r),this},delete(i){const r=X(this),{has:l,get:a}=Mn(r);let f=l.call(r,i);f||(i=X(i),f=l.call(r,i)),a&&a.call(r,i);const m=r.delete(i);return f&&ct(r,"delete",i,void 0),m},clear(){const i=X(this),r=i.size!==0,l=i.clear();return r&&ct(i,"clear",void 0,void 0),l}}),["keys","values","entries",Symbol.iterator].forEach(i=>{n[i]=pl(i,e,t)}),n}function Ws(e,t){const n=hl(e,t);return(s,i,r)=>i==="__v_isReactive"?!e:i==="__v_isReadonly"?e:i==="__v_raw"?s:Reflect.get(Z(n,i)&&i in s?n:s,i,r)}const ml={get:Ws(!1,!1)},gl={get:Ws(!1,!0)},vl={get:Ws(!0,!1)};const Cr=new WeakMap,Tr=new WeakMap,$r=new WeakMap,yl=new WeakMap;function _l(e){switch(e){case"Object":case"Array":return 1;case"Map":case"Set":case"WeakMap":case"WeakSet":return 2;default:return 0}}function bl(e){return e.__v_skip||!Object.isExtensible(e)?0:_l(Wo(e))}function Vs(e){return pt(e)?e:qs(e,!1,ul,ml,Cr)}function wl(e){return qs(e,!1,dl,gl,Tr)}function Ms(e){return qs(e,!0,fl,vl,$r)}function qs(e,t,n,s,i){if(!te(e)||e.__v_raw&&!(t&&e.__v_isReactive))return e;const r=bl(e);if(r===0)return e;const l=i.get(e);if(l)return l;const a=new Proxy(e,r===2?s:n);return i.set(e,a),a}function Pt(e){return pt(e)?Pt(e.__v_raw):!!(e&&e.__v_isReactive)}function pt(e){return!!(e&&e.__v_isReadonly)}function Le(e){return!!(e&&e.__v_isShallow)}function Gs(e){return e?!!e.__v_raw:!1}function X(e){const t=e&&e.__v_raw;return t?X(t):e}function xl(e){return!Z(e,"__v_skip")&&Object.isExtensible(e)&&ur(e,"__v_skip",!0),e}const Ge=e=>te(e)?Vs(e):e,Xt=e=>te(e)?Ms(e):e;function Ce(e){return e?e.__v_isRef===!0:!1}function oe(e){return Il(e,!1)}function Il(e,t){return Ce(e)?e:new Sl(e,t)}class Sl{constructor(t,n){this.dep=new Ks,this.__v_isRef=!0,this.__v_isShallow=!1,this._rawValue=n?t:X(t),this._value=n?t:Ge(t),this.__v_isShallow=n}get value(){return this.dep.track(),this._value}set value(t){const n=this._rawValue,s=this.__v_isShallow||Le(t)||pt(t);t=s?t:X(t),et(t,n)&&(this._rawValue=t,this._value=s?t:Ge(t),this.dep.trigger())}}function Cl(e){return Ce(e)?e.value:e}const Tl={get:(e,t,n)=>t==="__v_raw"?e:Cl(Reflect.get(e,t,n)),set:(e,t,n,s)=>{const i=e[t];return Ce(i)&&!Ce(n)?(i.value=n,!0):Reflect.set(e,t,n,s)}};function Mr(e){return Pt(e)?e:new Proxy(e,Tl)}class $l{constructor(t,n,s){this.fn=t,this.setter=n,this._value=void 0,this.dep=new Ks(this),this.__v_isRef=!0,this.deps=void 0,this.depsTail=void 0,this.flags=16,this.globalVersion=gn-1,this.next=void 0,this.effect=this,this.__v_isReadonly=!n,this.isSSR=s}notify(){if(this.flags|=16,!(this.flags&8)&&re!==this)return gr(this,!0),!0}get value(){const t=this.dep.track();return _r(this),t&&(t.version=this.dep.version),this._value}set value(t){this.setter&&this.setter(t)}}function Ml(e,t,n=!1){let s,i;return z(e)?s=e:(s=e.get,i=e.set),new $l(s,i,n)}const Rn={},Hn=new WeakMap;let At;function kl(e,t=!1,n=At){if(n){let s=Hn.get(n);s||Hn.set(n,s=[]),s.push(e)}}function Rl(e,t,n=ie){const{immediate:s,deep:i,once:r,scheduler:l,augmentJob:a,call:f}=n,m=F=>i?F:Le(F)||i===!1||i===0?ut(F,1):ut(F);let p,_,T,S,U=!1,D=!1;if(Ce(e)?(_=()=>e.value,U=Le(e)):Pt(e)?(_=()=>m(e),U=!0):B(e)?(D=!0,U=e.some(F=>Pt(F)||Le(F)),_=()=>e.map(F=>{if(Ce(F))return F.value;if(Pt(F))return m(F);if(z(F))return f?f(F,2):F()})):z(e)?t?_=f?()=>f(e,2):e:_=()=>{if(T){ft();try{T()}finally{dt()}}const F=At;At=p;try{return f?f(e,3,[S]):e(S)}finally{At=F}}:_=tt,t&&i){const F=_,le=i===!0?1/0:i;_=()=>ut(F(),le)}const Q=nl(),K=()=>{p.stop(),Q&&Q.active&&Ns(Q.effects,p)};if(r&&t){const F=t;t=(...le)=>{F(...le),K()}}let R=D?new Array(e.length).fill(Rn):Rn;const ee=F=>{if(!(!(p.flags&1)||!p.dirty&&!F))if(t){const le=p.run();if(i||U||(D?le.some((he,$e)=>et(he,R[$e])):et(le,R))){T&&T();const he=At;At=p;try{const $e=[le,R===Rn?void 0:D&&R[0]===Rn?[]:R,S];R=le,f?f(t,3,$e):t(...$e)}finally{At=he}}}else p.run()};return a&&a(ee),p=new hr(_),p.scheduler=l?()=>l(ee,!1):ee,S=F=>kl(F,!1,p),T=p.onStop=()=>{const F=Hn.get(p);if(F){if(f)f(F,4);else for(const le of F)le();Hn.delete(p)}},t?s?ee(!0):R=p.run():l?l(ee.bind(null,!0),!0):p.run(),K.pause=p.pause.bind(p),K.resume=p.resume.bind(p),K.stop=K,K}function ut(e,t=1/0,n){if(t<=0||!te(e)||e.__v_skip||(n=n||new Map,(n.get(e)||0)>=t))return e;if(n.set(e,t),t--,Ce(e))ut(e.value,t,n);else if(B(e))for(let s=0;s{ut(s,t,n)});else if(cr(e)){for(const s in e)ut(e[s],t,n);for(const s of Object.getOwnPropertySymbols(e))Object.prototype.propertyIsEnumerable.call(e,s)&&ut(e[s],t,n)}return e}/** -* @vue/runtime-core v3.5.30 -* (c) 2018-present Yuxi (Evan) You and Vue contributors -* @license MIT -**/function xn(e,t,n,s){try{return s?e(...s):e()}catch(i){Qn(i,t,n)}}function nt(e,t,n,s){if(z(e)){const i=xn(e,t,n,s);return i&&lr(i)&&i.catch(r=>{Qn(r,t,n)}),i}if(B(e)){const i=[];for(let r=0;r>>1,i=Re[s],r=yn(i);r=yn(n)?Re.push(e):Re.splice(Al(t),0,e),e.flags|=1,Er()}}function Er(){zn||(zn=kr.then(Or))}function Ol(e){B(e)?qt.push(...e):wt&&e.id===-1?wt.splice(Ht+1,0,e):e.flags&1||(qt.push(e),e.flags|=1),Er()}function Ci(e,t,n=Xe+1){for(;nyn(n)-yn(s));if(qt.length=0,wt){wt.push(...t);return}for(wt=t,Ht=0;Hte.id==null?e.flags&2?-1:1/0:e.id;function Or(e){try{for(Xe=0;Xe{s._d&&Ni(-1);const r=Kn(t);let l;try{l=e(...i)}finally{Kn(r),s._d&&Ni(1)}return l};return s._n=!0,s._c=!0,s._d=!0,s}function Pl(e,t){if(ye===null)return e;const n=ss(ye),s=e.dirs||(e.dirs=[]);for(let i=0;i1)return n&&z(t)?t.call(s&&s.proxy):t}}const Fl=Symbol.for("v-scx"),Nl=()=>Bn(Fl);function Dt(e,t,n){return Dr(e,t,n)}function Dr(e,t,n=ie){const{immediate:s,deep:i,flush:r,once:l}=n,a=Te({},n),f=t&&s||!t&&r!=="post";let m;if(bn){if(r==="sync"){const S=Nl();m=S.__watcherHandles||(S.__watcherHandles=[])}else if(!f){const S=()=>{};return S.stop=tt,S.resume=tt,S.pause=tt,S}}const p=Se;a.call=(S,U,D)=>nt(S,p,U,D);let _=!1;r==="post"?a.scheduler=S=>{Ae(S,p&&p.suspense)}:r!=="sync"&&(_=!0,a.scheduler=(S,U)=>{U?S():Js(S)}),a.augmentJob=S=>{t&&(S.flags|=4),_&&(S.flags|=2,p&&(S.id=p.uid,S.i=p))};const T=Rl(e,t,a);return bn&&(m?m.push(T):f&&T()),T}function Ul(e,t,n){const s=this.proxy,i=ae(e)?e.includes(".")?Fr(s,e):()=>s[e]:e.bind(s,s);let r;z(t)?r=t:(r=t.handler,n=t);const l=In(this),a=Dr(i,r.bind(s),n);return l(),a}function Fr(e,t){const n=t.split(".");return()=>{let s=e;for(let i=0;ie.__isTeleport,jl=Symbol("_leaveCb");function Ys(e,t){e.shapeFlag&6&&e.component?(e.transition=t,Ys(e.component.subTree,t)):e.shapeFlag&128?(e.ssContent.transition=t.clone(e.ssContent),e.ssFallback.transition=t.clone(e.ssFallback)):e.transition=t}function Nr(e){e.ids=[e.ids[0]+e.ids[2]+++"-",0,0]}function Ti(e,t){let n;return!!((n=Object.getOwnPropertyDescriptor(e,t))&&!n.configurable)}const Wn=new WeakMap;function dn(e,t,n,s,i=!1){if(B(e)){e.forEach((D,Q)=>dn(D,t&&(B(t)?t[Q]:t),n,s,i));return}if(Jt(s)&&!i){s.shapeFlag&512&&s.type.__asyncResolved&&s.component.subTree.component&&dn(e,t,n,s.component.subTree);return}const r=s.shapeFlag&4?ss(s.component):s.el,l=i?null:r,{i:a,r:f}=e,m=t&&t.r,p=a.refs===ie?a.refs={}:a.refs,_=a.setupState,T=X(_),S=_===ie?rr:D=>Ti(p,D)?!1:Z(T,D),U=(D,Q)=>!(Q&&Ti(p,Q));if(m!=null&&m!==f){if($i(t),ae(m))p[m]=null,S(m)&&(_[m]=null);else if(Ce(m)){const D=t;U(m,D.k)&&(m.value=null),D.k&&(p[D.k]=null)}}if(z(f))xn(f,a,12,[l,p]);else{const D=ae(f),Q=Ce(f);if(D||Q){const K=()=>{if(e.f){const R=D?S(f)?_[f]:p[f]:U()||!e.k?f.value:p[e.k];if(i)B(R)&&Ns(R,r);else if(B(R))R.includes(r)||R.push(r);else if(D)p[f]=[r],S(f)&&(_[f]=p[f]);else{const ee=[r];U(f,e.k)&&(f.value=ee),e.k&&(p[e.k]=ee)}}else D?(p[f]=l,S(f)&&(_[f]=l)):Q&&(U(f,e.k)&&(f.value=l),e.k&&(p[e.k]=l))};if(l){const R=()=>{K(),Wn.delete(e)};R.id=-1,Wn.set(e,R),Ae(R,n)}else $i(e),K()}}}function $i(e){const t=Wn.get(e);t&&(t.flags|=8,Wn.delete(e))}Xn().requestIdleCallback;Xn().cancelIdleCallback;const Jt=e=>!!e.type.__asyncLoader,Ur=e=>e.type.__isKeepAlive;function Hl(e,t){Br(e,"a",t)}function zl(e,t){Br(e,"da",t)}function Br(e,t,n=Se){const s=e.__wdc||(e.__wdc=()=>{let i=n;for(;i;){if(i.isDeactivated)return;i=i.parent}return e()});if(es(t,s,n),n){let i=n.parent;for(;i&&i.parent;)Ur(i.parent.vnode)&&Kl(s,t,n,i),i=i.parent}}function Kl(e,t,n,s){const i=es(t,e,s,!0);jr(()=>{Ns(s[t],i)},n)}function es(e,t,n=Se,s=!1){if(n){const i=n[e]||(n[e]=[]),r=t.__weh||(t.__weh=(...l)=>{ft();const a=In(n),f=nt(t,n,e,l);return a(),dt(),f});return s?i.unshift(r):i.push(r),r}}const gt=e=>(t,n=Se)=>{(!bn||e==="sp")&&es(e,(...s)=>t(...s),n)},Wl=gt("bm"),Lr=gt("m"),Vl=gt("bu"),ql=gt("u"),Xs=gt("bum"),jr=gt("um"),Gl=gt("sp"),Jl=gt("rtg"),Yl=gt("rtc");function Xl(e,t=Se){es("ec",e,t)}const Zl="components",Hr=Symbol.for("v-ndc");function Ql(e){return ae(e)?ea(Zl,e,!1)||e:e||Hr}function ea(e,t,n=!0,s=!1){const i=ye||Se;if(i){const r=i.type;{const a=Ba(r,!1);if(a&&(a===t||a===Ee(t)||a===Yn(Ee(t))))return r}const l=Mi(i[e]||r[e],t)||Mi(i.appContext[e],t);return!l&&s?r:l}}function Mi(e,t){return e&&(e[t]||e[Ee(t)]||e[Yn(Ee(t))])}function Zt(e,t,n,s){let i;const r=n,l=B(e);if(l||ae(e)){const a=l&&Pt(e);let f=!1,m=!1;a&&(f=!Le(e),m=pt(e),e=Zn(e)),i=new Array(e.length);for(let p=0,_=e.length;p<_;p++)i[p]=t(f?m?Xt(Ge(e[p])):Ge(e[p]):e[p],p,void 0,r)}else if(typeof e=="number"){i=new Array(e);for(let a=0;at(a,f,void 0,r));else{const a=Object.keys(e);i=new Array(a.length);for(let f=0,m=a.length;f0;return L(),mt(fe,null,[q("slot",n,s)],m?-2:64)}let r=e[t];r&&r._c&&(r._d=!1),L();const l=r&&zr(r(n)),a=n.key||l&&l.key,f=mt(fe,{key:(a&&!qe(a)?a:`_${t}`)+(!l&&s?"_fb":"")},l||[],l&&e._===1?64:-2);return f.scopeId&&(f.slotScopeIds=[f.scopeId+"-s"]),r&&r._c&&(r._d=!0),f}function zr(e){return e.some(t=>ei(t)?!(t.type===ht||t.type===fe&&!zr(t.children)):!0)?e:null}const ks=e=>e?uo(e)?ss(e):ks(e.parent):null,pn=Te(Object.create(null),{$:e=>e,$el:e=>e.vnode.el,$data:e=>e.data,$props:e=>e.props,$attrs:e=>e.attrs,$slots:e=>e.slots,$refs:e=>e.refs,$parent:e=>ks(e.parent),$root:e=>ks(e.root),$host:e=>e.ce,$emit:e=>e.emit,$options:e=>Wr(e),$forceUpdate:e=>e.f||(e.f=()=>{Js(e.update)}),$nextTick:e=>e.n||(e.n=Rr.bind(e.proxy)),$watch:e=>Ul.bind(e)}),gs=(e,t)=>e!==ie&&!e.__isScriptSetup&&Z(e,t),na={get({_:e},t){if(t==="__v_skip")return!0;const{ctx:n,setupState:s,data:i,props:r,accessCache:l,type:a,appContext:f}=e;if(t[0]!=="$"){const T=l[t];if(T!==void 0)switch(T){case 1:return s[t];case 2:return i[t];case 4:return n[t];case 3:return r[t]}else{if(gs(s,t))return l[t]=1,s[t];if(i!==ie&&Z(i,t))return l[t]=2,i[t];if(Z(r,t))return l[t]=3,r[t];if(n!==ie&&Z(n,t))return l[t]=4,n[t];Rs&&(l[t]=0)}}const m=pn[t];let p,_;if(m)return t==="$attrs"&&Ie(e.attrs,"get",""),m(e);if((p=a.__cssModules)&&(p=p[t]))return p;if(n!==ie&&Z(n,t))return l[t]=4,n[t];if(_=f.config.globalProperties,Z(_,t))return _[t]},set({_:e},t,n){const{data:s,setupState:i,ctx:r}=e;return gs(i,t)?(i[t]=n,!0):s!==ie&&Z(s,t)?(s[t]=n,!0):Z(e.props,t)||t[0]==="$"&&t.slice(1)in e?!1:(r[t]=n,!0)},has({_:{data:e,setupState:t,accessCache:n,ctx:s,appContext:i,props:r,type:l}},a){let f;return!!(n[a]||e!==ie&&a[0]!=="$"&&Z(e,a)||gs(t,a)||Z(r,a)||Z(s,a)||Z(pn,a)||Z(i.config.globalProperties,a)||(f=l.__cssModules)&&f[a])},defineProperty(e,t,n){return n.get!=null?e._.accessCache[t]=0:Z(n,"value")&&this.set(e,t,n.value,null),Reflect.defineProperty(e,t,n)}};function ki(e){return B(e)?e.reduce((t,n)=>(t[n]=null,t),{}):e}let Rs=!0;function sa(e){const t=Wr(e),n=e.proxy,s=e.ctx;Rs=!1,t.beforeCreate&&Ri(t.beforeCreate,e,"bc");const{data:i,computed:r,methods:l,watch:a,provide:f,inject:m,created:p,beforeMount:_,mounted:T,beforeUpdate:S,updated:U,activated:D,deactivated:Q,beforeDestroy:K,beforeUnmount:R,destroyed:ee,unmounted:F,render:le,renderTracked:he,renderTriggered:$e,errorCaptured:Me,serverPrefetch:vt,expose:je,inheritAttrs:Je,components:He,directives:yt,filters:de}=t;if(m&&ia(m,s,null),l)for(const j in l){const G=l[j];z(G)&&(s[j]=G.bind(n))}if(i){const j=i.call(n,n);te(j)&&(e.data=Vs(j))}if(Rs=!0,r)for(const j in r){const G=r[j],_e=z(G)?G.bind(n,n):z(G.get)?G.get.bind(n,n):tt,ze=!z(G)&&z(G.set)?G.set.bind(n):tt,be=Ft({get:_e,set:ze});Object.defineProperty(s,j,{enumerable:!0,configurable:!0,get:()=>be.value,set:ce=>be.value=ce})}if(a)for(const j in a)Kr(a[j],s,n,j);if(f){const j=z(f)?f.call(n):f;Reflect.ownKeys(j).forEach(G=>{Dl(G,j[G])})}p&&Ri(p,e,"c");function H(j,G){B(G)?G.forEach(_e=>j(_e.bind(n))):G&&j(G.bind(n))}if(H(Wl,_),H(Lr,T),H(Vl,S),H(ql,U),H(Hl,D),H(zl,Q),H(Xl,Me),H(Yl,he),H(Jl,$e),H(Xs,R),H(jr,F),H(Gl,vt),B(je))if(je.length){const j=e.exposed||(e.exposed={});je.forEach(G=>{Object.defineProperty(j,G,{get:()=>n[G],set:_e=>n[G]=_e,enumerable:!0})})}else e.exposed||(e.exposed={});le&&e.render===tt&&(e.render=le),Je!=null&&(e.inheritAttrs=Je),He&&(e.components=He),yt&&(e.directives=yt),vt&&Nr(e)}function ia(e,t,n=tt){B(e)&&(e=Es(e));for(const s in e){const i=e[s];let r;te(i)?"default"in i?r=Bn(i.from||s,i.default,!0):r=Bn(i.from||s):r=Bn(i),Ce(r)?Object.defineProperty(t,s,{enumerable:!0,configurable:!0,get:()=>r.value,set:l=>r.value=l}):t[s]=r}}function Ri(e,t,n){nt(B(e)?e.map(s=>s.bind(t.proxy)):e.bind(t.proxy),t,n)}function Kr(e,t,n,s){let i=s.includes(".")?Fr(n,s):()=>n[s];if(ae(e)){const r=t[e];z(r)&&Dt(i,r)}else if(z(e))Dt(i,e.bind(n));else if(te(e))if(B(e))e.forEach(r=>Kr(r,t,n,s));else{const r=z(e.handler)?e.handler.bind(n):t[e.handler];z(r)&&Dt(i,r,e)}}function Wr(e){const t=e.type,{mixins:n,extends:s}=t,{mixins:i,optionsCache:r,config:{optionMergeStrategies:l}}=e.appContext,a=r.get(t);let f;return a?f=a:!i.length&&!n&&!s?f=t:(f={},i.length&&i.forEach(m=>Vn(f,m,l,!0)),Vn(f,t,l)),te(t)&&r.set(t,f),f}function Vn(e,t,n,s=!1){const{mixins:i,extends:r}=t;r&&Vn(e,r,n,!0),i&&i.forEach(l=>Vn(e,l,n,!0));for(const l in t)if(!(s&&l==="expose")){const a=ra[l]||n&&n[l];e[l]=a?a(e[l],t[l]):t[l]}return e}const ra={data:Ei,props:Ai,emits:Ai,methods:an,computed:an,beforeCreate:ke,created:ke,beforeMount:ke,mounted:ke,beforeUpdate:ke,updated:ke,beforeDestroy:ke,beforeUnmount:ke,destroyed:ke,unmounted:ke,activated:ke,deactivated:ke,errorCaptured:ke,serverPrefetch:ke,components:an,directives:an,watch:la,provide:Ei,inject:oa};function Ei(e,t){return t?e?function(){return Te(z(e)?e.call(this,this):e,z(t)?t.call(this,this):t)}:t:e}function oa(e,t){return an(Es(e),Es(t))}function Es(e){if(B(e)){const t={};for(let n=0;nt==="modelValue"||t==="model-value"?e.modelModifiers:e[`${t}Modifiers`]||e[`${Ee(t)}Modifiers`]||e[`${It(t)}Modifiers`];function fa(e,t,...n){if(e.isUnmounted)return;const s=e.vnode.props||ie;let i=n;const r=t.startsWith("update:"),l=r&&ua(s,t.slice(7));l&&(l.trim&&(i=n.map(p=>ae(p)?p.trim():p)),l.number&&(i=n.map(Bs)));let a,f=s[a=fs(t)]||s[a=fs(Ee(t))];!f&&r&&(f=s[a=fs(It(t))]),f&&nt(f,e,6,i);const m=s[a+"Once"];if(m){if(!e.emitted)e.emitted={};else if(e.emitted[a])return;e.emitted[a]=!0,nt(m,e,6,i)}}const da=new WeakMap;function qr(e,t,n=!1){const s=n?da:t.emitsCache,i=s.get(e);if(i!==void 0)return i;const r=e.emits;let l={},a=!1;if(!z(e)){const f=m=>{const p=qr(m,t,!0);p&&(a=!0,Te(l,p))};!n&&t.mixins.length&&t.mixins.forEach(f),e.extends&&f(e.extends),e.mixins&&e.mixins.forEach(f)}return!r&&!a?(te(e)&&s.set(e,null),null):(B(r)?r.forEach(f=>l[f]=null):Te(l,r),te(e)&&s.set(e,l),l)}function ts(e,t){return!e||!Gn(t)?!1:(t=t.slice(2).replace(/Once$/,""),Z(e,t[0].toLowerCase()+t.slice(1))||Z(e,It(t))||Z(e,t))}function Oi(e){const{type:t,vnode:n,proxy:s,withProxy:i,propsOptions:[r],slots:l,attrs:a,emit:f,render:m,renderCache:p,props:_,data:T,setupState:S,ctx:U,inheritAttrs:D}=e,Q=Kn(e);let K,R;try{if(n.shapeFlag&4){const F=i||s,le=F;K=Qe(m.call(le,F,p,_,S,T,U)),R=a}else{const F=t;K=Qe(F.length>1?F(_,{attrs:a,slots:l,emit:f}):F(_,null)),R=t.props?a:pa(a)}}catch(F){hn.length=0,Qn(F,e,1),K=q(ht)}let ee=K;if(R&&D!==!1){const F=Object.keys(R),{shapeFlag:le}=ee;F.length&&le&7&&(r&&F.some(Fs)&&(R=ha(R,r)),ee=Qt(ee,R,!1,!0))}return n.dirs&&(ee=Qt(ee,null,!1,!0),ee.dirs=ee.dirs?ee.dirs.concat(n.dirs):n.dirs),n.transition&&Ys(ee,n.transition),K=ee,Kn(Q),K}const pa=e=>{let t;for(const n in e)(n==="class"||n==="style"||Gn(n))&&((t||(t={}))[n]=e[n]);return t},ha=(e,t)=>{const n={};for(const s in e)(!Fs(s)||!(s.slice(9)in t))&&(n[s]=e[s]);return n};function ma(e,t,n){const{props:s,children:i,component:r}=e,{props:l,children:a,patchFlag:f}=t,m=r.emitsOptions;if(t.dirs||t.transition)return!0;if(n&&f>=0){if(f&1024)return!0;if(f&16)return s?Pi(s,l,m):!!l;if(f&8){const p=t.dynamicProps;for(let _=0;_Object.create(Jr),Xr=e=>Object.getPrototypeOf(e)===Jr;function va(e,t,n,s=!1){const i={},r=Yr();e.propsDefaults=Object.create(null),Zr(e,t,i,r);for(const l in e.propsOptions[0])l in i||(i[l]=void 0);n?e.props=s?i:wl(i):e.type.props?e.props=i:e.props=r,e.attrs=r}function ya(e,t,n,s){const{props:i,attrs:r,vnode:{patchFlag:l}}=e,a=X(i),[f]=e.propsOptions;let m=!1;if((s||l>0)&&!(l&16)){if(l&8){const p=e.vnode.dynamicProps;for(let _=0;_{f=!0;const[T,S]=Qr(_,t,!0);Te(l,T),S&&a.push(...S)};!n&&t.mixins.length&&t.mixins.forEach(p),e.extends&&p(e.extends),e.mixins&&e.mixins.forEach(p)}if(!r&&!f)return te(e)&&s.set(e,Wt),Wt;if(B(r))for(let p=0;pe==="_"||e==="_ctx"||e==="$stable",Qs=e=>B(e)?e.map(Qe):[Qe(e)],ba=(e,t,n)=>{if(t._n)return t;const s=Gt((...i)=>Qs(t(...i)),n);return s._c=!1,s},eo=(e,t,n)=>{const s=e._ctx;for(const i in e){if(Zs(i))continue;const r=e[i];if(z(r))t[i]=ba(i,r,s);else if(r!=null){const l=Qs(r);t[i]=()=>l}}},to=(e,t)=>{const n=Qs(t);e.slots.default=()=>n},no=(e,t,n)=>{for(const s in t)(n||!Zs(s))&&(e[s]=t[s])},wa=(e,t,n)=>{const s=e.slots=Yr();if(e.vnode.shapeFlag&32){const i=t._;i?(no(s,t,n),n&&ur(s,"_",i,!0)):eo(t,s)}else t&&to(e,t)},xa=(e,t,n)=>{const{vnode:s,slots:i}=e;let r=!0,l=ie;if(s.shapeFlag&32){const a=t._;a?n&&a===1?r=!1:no(i,t,n):(r=!t.$stable,eo(t,i)),l=t}else t&&(to(e,t),l={default:1});if(r)for(const a in i)!Zs(a)&&l[a]==null&&delete i[a]},Ae=$a;function Ia(e){return Sa(e)}function Sa(e,t){const n=Xn();n.__VUE__=!0;const{insert:s,remove:i,patchProp:r,createElement:l,createText:a,createComment:f,setText:m,setElementText:p,parentNode:_,nextSibling:T,setScopeId:S=tt,insertStaticContent:U}=e,D=(d,h,v,I=null,w=null,x=null,k=void 0,$=null,C=!!h.dynamicChildren)=>{if(d===h)return;d&&!ln(d,h)&&(I=Nt(d),ce(d,w,x,!0),d=null),h.patchFlag===-2&&(C=!1,h.dynamicChildren=null);const{type:b,ref:P,shapeFlag:E}=h;switch(b){case ns:Q(d,h,v,I);break;case ht:K(d,h,v,I);break;case ys:d==null&&R(h,v,I,k);break;case fe:He(d,h,v,I,w,x,k,$,C);break;default:E&1?le(d,h,v,I,w,x,k,$,C):E&6?yt(d,h,v,I,w,x,k,$,C):(E&64||E&128)&&b.process(d,h,v,I,w,x,k,$,C,it)}P!=null&&w?dn(P,d&&d.ref,x,h||d,!h):P==null&&d&&d.ref!=null&&dn(d.ref,null,x,d,!0)},Q=(d,h,v,I)=>{if(d==null)s(h.el=a(h.children),v,I);else{const w=h.el=d.el;h.children!==d.children&&m(w,h.children)}},K=(d,h,v,I)=>{d==null?s(h.el=f(h.children||""),v,I):h.el=d.el},R=(d,h,v,I)=>{[d.el,d.anchor]=U(d.children,h,v,I,d.el,d.anchor)},ee=({el:d,anchor:h},v,I)=>{let w;for(;d&&d!==h;)w=T(d),s(d,v,I),d=w;s(h,v,I)},F=({el:d,anchor:h})=>{let v;for(;d&&d!==h;)v=T(d),i(d),d=v;i(h)},le=(d,h,v,I,w,x,k,$,C)=>{if(h.type==="svg"?k="svg":h.type==="math"&&(k="mathml"),d==null)he(h,v,I,w,x,k,$,C);else{const b=d.el&&d.el._isVueCE?d.el:null;try{b&&b._beginPatch(),vt(d,h,w,x,k,$,C)}finally{b&&b._endPatch()}}},he=(d,h,v,I,w,x,k,$)=>{let C,b;const{props:P,shapeFlag:E,transition:A,dirs:N}=d;if(C=d.el=l(d.type,x,P&&P.is,P),E&8?p(C,d.children):E&16&&Me(d.children,C,null,I,w,vs(d,x),k,$),N&&Rt(d,null,I,"created"),$e(C,d,d.scopeId,k,I),P){for(const Y in P)Y!=="value"&&!cn(Y)&&r(C,Y,null,P[Y],x,I);"value"in P&&r(C,"value",null,P.value,x),(b=P.onVnodeBeforeMount)&&Ye(b,I,d)}N&&Rt(d,null,I,"beforeMount");const W=Ca(w,A);W&&A.beforeEnter(C),s(C,h,v),((b=P&&P.onVnodeMounted)||W||N)&&Ae(()=>{b&&Ye(b,I,d),W&&A.enter(C),N&&Rt(d,null,I,"mounted")},w)},$e=(d,h,v,I,w)=>{if(v&&S(d,v),I)for(let x=0;x{for(let b=C;b{const $=h.el=d.el;let{patchFlag:C,dynamicChildren:b,dirs:P}=h;C|=d.patchFlag&16;const E=d.props||ie,A=h.props||ie;let N;if(v&&Et(v,!1),(N=A.onVnodeBeforeUpdate)&&Ye(N,v,h,d),P&&Rt(h,d,v,"beforeUpdate"),v&&Et(v,!0),(E.innerHTML&&A.innerHTML==null||E.textContent&&A.textContent==null)&&p($,""),b?je(d.dynamicChildren,b,$,v,I,vs(h,w),x):k||G(d,h,$,null,v,I,vs(h,w),x,!1),C>0){if(C&16)Je($,E,A,v,w);else if(C&2&&E.class!==A.class&&r($,"class",null,A.class,w),C&4&&r($,"style",E.style,A.style,w),C&8){const W=h.dynamicProps;for(let Y=0;Y{N&&Ye(N,v,h,d),P&&Rt(h,d,v,"updated")},I)},je=(d,h,v,I,w,x,k)=>{for(let $=0;${if(h!==v){if(h!==ie)for(const x in h)!cn(x)&&!(x in v)&&r(d,x,h[x],null,w,I);for(const x in v){if(cn(x))continue;const k=v[x],$=h[x];k!==$&&x!=="value"&&r(d,x,$,k,w,I)}"value"in v&&r(d,"value",h.value,v.value,w)}},He=(d,h,v,I,w,x,k,$,C)=>{const b=h.el=d?d.el:a(""),P=h.anchor=d?d.anchor:a("");let{patchFlag:E,dynamicChildren:A,slotScopeIds:N}=h;N&&($=$?$.concat(N):N),d==null?(s(b,v,I),s(P,v,I),Me(h.children||[],v,P,w,x,k,$,C)):E>0&&E&64&&A&&d.dynamicChildren&&d.dynamicChildren.length===A.length?(je(d.dynamicChildren,A,v,w,x,k,$),(h.key!=null||w&&h===w.subTree)&&so(d,h,!0)):G(d,h,v,P,w,x,k,$,C)},yt=(d,h,v,I,w,x,k,$,C)=>{h.slotScopeIds=$,d==null?h.shapeFlag&512?w.ctx.activate(h,v,I,k,C):de(h,v,I,w,x,k,C):st(d,h,C)},de=(d,h,v,I,w,x,k)=>{const $=d.component=Oa(d,I,w);if(Ur(d)&&($.ctx.renderer=it),Da($,!1,k),$.asyncDep){if(w&&w.registerDep($,H,k),!d.el){const C=$.subTree=q(ht);K(null,C,h,v),d.placeholder=C.el}}else H($,d,h,v,w,x,k)},st=(d,h,v)=>{const I=h.component=d.component;if(ma(d,h,v))if(I.asyncDep&&!I.asyncResolved){j(I,h,v);return}else I.next=h,I.update();else h.el=d.el,I.vnode=h},H=(d,h,v,I,w,x,k)=>{const $=()=>{if(d.isMounted){let{next:E,bu:A,u:N,parent:W,vnode:Y}=d;{const Ne=io(d);if(Ne){E&&(E.el=Y.el,j(d,E,k)),Ne.asyncDep.then(()=>{Ae(()=>{d.isUnmounted||b()},w)});return}}let J=E,we;Et(d,!1),E?(E.el=Y.el,j(d,E,k)):E=Y,A&&Un(A),(we=E.props&&E.props.onVnodeBeforeUpdate)&&Ye(we,W,E,Y),Et(d,!0);const xe=Oi(d),Fe=d.subTree;d.subTree=xe,D(Fe,xe,_(Fe.el),Nt(Fe),d,w,x),E.el=xe.el,J===null&&ga(d,xe.el),N&&Ae(N,w),(we=E.props&&E.props.onVnodeUpdated)&&Ae(()=>Ye(we,W,E,Y),w)}else{let E;const{el:A,props:N}=h,{bm:W,m:Y,parent:J,root:we,type:xe}=d,Fe=Jt(h);Et(d,!1),W&&Un(W),!Fe&&(E=N&&N.onVnodeBeforeMount)&&Ye(E,J,h),Et(d,!0);{we.ce&&we.ce._hasShadowRoot()&&we.ce._injectChildStyle(xe,d.parent?d.parent.type:void 0);const Ne=d.subTree=Oi(d);D(null,Ne,v,I,d,w,x),h.el=Ne.el}if(Y&&Ae(Y,w),!Fe&&(E=N&&N.onVnodeMounted)){const Ne=h;Ae(()=>Ye(E,J,Ne),w)}(h.shapeFlag&256||J&&Jt(J.vnode)&&J.vnode.shapeFlag&256)&&d.a&&Ae(d.a,w),d.isMounted=!0,h=v=I=null}};d.scope.on();const C=d.effect=new hr($);d.scope.off();const b=d.update=C.run.bind(C),P=d.job=C.runIfDirty.bind(C);P.i=d,P.id=d.uid,C.scheduler=()=>Js(P),Et(d,!0),b()},j=(d,h,v)=>{h.component=d;const I=d.vnode.props;d.vnode=h,d.next=null,ya(d,h.props,I,v),xa(d,h.children,v),ft(),Ci(d),dt()},G=(d,h,v,I,w,x,k,$,C=!1)=>{const b=d&&d.children,P=d?d.shapeFlag:0,E=h.children,{patchFlag:A,shapeFlag:N}=h;if(A>0){if(A&128){ze(b,E,v,I,w,x,k,$,C);return}else if(A&256){_e(b,E,v,I,w,x,k,$,C);return}}N&8?(P&16&&Ct(b,w,x),E!==b&&p(v,E)):P&16?N&16?ze(b,E,v,I,w,x,k,$,C):Ct(b,w,x,!0):(P&8&&p(v,""),N&16&&Me(E,v,I,w,x,k,$,C))},_e=(d,h,v,I,w,x,k,$,C)=>{d=d||Wt,h=h||Wt;const b=d.length,P=h.length,E=Math.min(b,P);let A;for(A=0;AP?Ct(d,w,x,!0,!1,E):Me(h,v,I,w,x,k,$,C,E)},ze=(d,h,v,I,w,x,k,$,C)=>{let b=0;const P=h.length;let E=d.length-1,A=P-1;for(;b<=E&&b<=A;){const N=d[b],W=h[b]=C?at(h[b]):Qe(h[b]);if(ln(N,W))D(N,W,v,null,w,x,k,$,C);else break;b++}for(;b<=E&&b<=A;){const N=d[E],W=h[A]=C?at(h[A]):Qe(h[A]);if(ln(N,W))D(N,W,v,null,w,x,k,$,C);else break;E--,A--}if(b>E){if(b<=A){const N=A+1,W=NA)for(;b<=E;)ce(d[b],w,x,!0),b++;else{const N=b,W=b,Y=new Map;for(b=W;b<=A;b++){const me=h[b]=C?at(h[b]):Qe(h[b]);me.key!=null&&Y.set(me.key,b)}let J,we=0;const xe=A-W+1;let Fe=!1,Ne=0;const Tt=new Array(xe);for(b=0;b=xe){ce(me,w,x,!0);continue}let Pe;if(me.key!=null)Pe=Y.get(me.key);else for(J=W;J<=A;J++)if(Tt[J-W]===0&&ln(me,h[J])){Pe=J;break}Pe===void 0?ce(me,w,x,!0):(Tt[Pe-W]=b+1,Pe>=Ne?Ne=Pe:Fe=!0,D(me,h[Pe],v,null,w,x,k,$,C),we++)}const nn=Fe?Ta(Tt):Wt;for(J=nn.length-1,b=xe-1;b>=0;b--){const me=W+b,Pe=h[me],sn=h[me+1],rn=me+1{const{el:x,type:k,transition:$,children:C,shapeFlag:b}=d;if(b&6){be(d.component.subTree,h,v,I);return}if(b&128){d.suspense.move(h,v,I);return}if(b&64){k.move(d,h,v,it);return}if(k===fe){s(x,h,v);for(let E=0;E$.enter(x),w);else{const{leave:E,delayLeave:A,afterLeave:N}=$,W=()=>{d.ctx.isUnmounted?i(x):s(x,h,v)},Y=()=>{x._isLeaving&&x[jl](!0),E(x,()=>{W(),N&&N()})};A?A(x,W,Y):Y()}else s(x,h,v)},ce=(d,h,v,I=!1,w=!1)=>{const{type:x,props:k,ref:$,children:C,dynamicChildren:b,shapeFlag:P,patchFlag:E,dirs:A,cacheIndex:N}=d;if(E===-2&&(w=!1),$!=null&&(ft(),dn($,null,v,d,!0),dt()),N!=null&&(h.renderCache[N]=void 0),P&256){h.ctx.deactivate(d);return}const W=P&1&&A,Y=!Jt(d);let J;if(Y&&(J=k&&k.onVnodeBeforeUnmount)&&Ye(J,h,d),P&6)Sn(d.component,v,I);else{if(P&128){d.suspense.unmount(v,I);return}W&&Rt(d,null,h,"beforeUnmount"),P&64?d.type.remove(d,h,v,it,I):b&&!b.hasOnce&&(x!==fe||E>0&&E&64)?Ct(b,h,v,!1,!0):(x===fe&&E&384||!w&&P&16)&&Ct(C,h,v),I&&_t(d)}(Y&&(J=k&&k.onVnodeUnmounted)||W)&&Ae(()=>{J&&Ye(J,h,d),W&&Rt(d,null,h,"unmounted")},v)},_t=d=>{const{type:h,el:v,anchor:I,transition:w}=d;if(h===fe){St(v,I);return}if(h===ys){F(d);return}const x=()=>{i(v),w&&!w.persisted&&w.afterLeave&&w.afterLeave()};if(d.shapeFlag&1&&w&&!w.persisted){const{leave:k,delayLeave:$}=w,C=()=>k(v,x);$?$(d.el,x,C):C()}else x()},St=(d,h)=>{let v;for(;d!==h;)v=T(d),i(d),d=v;i(h)},Sn=(d,h,v)=>{const{bum:I,scope:w,job:x,subTree:k,um:$,m:C,a:b}=d;Fi(C),Fi(b),I&&Un(I),w.stop(),x&&(x.flags|=8,ce(k,d,h,v)),$&&Ae($,h),Ae(()=>{d.isUnmounted=!0},h)},Ct=(d,h,v,I=!1,w=!1,x=0)=>{for(let k=x;k{if(d.shapeFlag&6)return Nt(d.component.subTree);if(d.shapeFlag&128)return d.suspense.next();const h=T(d.anchor||d.el),v=h&&h[Bl];return v?T(v):h};let tn=!1;const Cn=(d,h,v)=>{let I;d==null?h._vnode&&(ce(h._vnode,null,null,!0),I=h._vnode.component):D(h._vnode||null,d,h,null,null,null,v),h._vnode=d,tn||(tn=!0,Ci(I),Ar(),tn=!1)},it={p:D,um:ce,m:be,r:_t,mt:de,mc:Me,pc:G,pbc:je,n:Nt,o:e};return{render:Cn,hydrate:void 0,createApp:ca(Cn)}}function vs({type:e,props:t},n){return n==="svg"&&e==="foreignObject"||n==="mathml"&&e==="annotation-xml"&&t&&t.encoding&&t.encoding.includes("html")?void 0:n}function Et({effect:e,job:t},n){n?(e.flags|=32,t.flags|=4):(e.flags&=-33,t.flags&=-5)}function Ca(e,t){return(!e||e&&!e.pendingBranch)&&t&&!t.persisted}function so(e,t,n=!1){const s=e.children,i=t.children;if(B(s)&&B(i))for(let r=0;r>1,e[n[a]]0&&(t[s]=n[r-1]),n[r]=s)}}for(r=n.length,l=n[r-1];r-- >0;)n[r]=l,l=t[l];return n}function io(e){const t=e.subTree.component;if(t)return t.asyncDep&&!t.asyncResolved?t:io(t)}function Fi(e){if(e)for(let t=0;te.__isSuspense;function $a(e,t){t&&t.pendingBranch?B(e)?t.effects.push(...e):t.effects.push(e):Ol(e)}const fe=Symbol.for("v-fgt"),ns=Symbol.for("v-txt"),ht=Symbol.for("v-cmt"),ys=Symbol.for("v-stc"),hn=[];let De=null;function L(e=!1){hn.push(De=e?null:[])}function Ma(){hn.pop(),De=hn[hn.length-1]||null}let _n=1;function Ni(e,t=!1){_n+=e,e<0&&De&&t&&(De.hasOnce=!0)}function lo(e){return e.dynamicChildren=_n>0?De||Wt:null,Ma(),_n>0&&De&&De.push(e),e}function V(e,t,n,s,i,r){return lo(y(e,t,n,s,i,r,!0))}function mt(e,t,n,s,i){return lo(q(e,t,n,s,i,!0))}function ei(e){return e?e.__v_isVNode===!0:!1}function ln(e,t){return e.type===t.type&&e.key===t.key}const ao=({key:e})=>e??null,Ln=({ref:e,ref_key:t,ref_for:n})=>(typeof e=="number"&&(e=""+e),e!=null?ae(e)||Ce(e)||z(e)?{i:ye,r:e,k:t,f:!!n}:e:null);function y(e,t=null,n=null,s=0,i=null,r=e===fe?0:1,l=!1,a=!1){const f={__v_isVNode:!0,__v_skip:!0,type:e,props:t,key:t&&ao(t),ref:t&&Ln(t),scopeId:Pr,slotScopeIds:null,children:n,component:null,suspense:null,ssContent:null,ssFallback:null,dirs:null,transition:null,el:null,anchor:null,target:null,targetStart:null,targetAnchor:null,staticCount:0,shapeFlag:r,patchFlag:s,dynamicProps:i,dynamicChildren:null,appContext:null,ctx:ye};return a?(ti(f,n),r&128&&e.normalize(f)):n&&(f.shapeFlag|=ae(n)?8:16),_n>0&&!l&&De&&(f.patchFlag>0||r&6)&&f.patchFlag!==32&&De.push(f),f}const q=ka;function ka(e,t=null,n=null,s=0,i=null,r=!1){if((!e||e===Hr)&&(e=ht),ei(e)){const a=Qt(e,t,!0);return n&&ti(a,n),_n>0&&!r&&De&&(a.shapeFlag&6?De[De.indexOf(e)]=a:De.push(a)),a.patchFlag=-2,a}if(La(e)&&(e=e.__vccOpts),t){t=Ra(t);let{class:a,style:f}=t;a&&!ae(a)&&(t.class=xt(a)),te(f)&&(Gs(f)&&!B(f)&&(f=Te({},f)),t.style=Be(f))}const l=ae(e)?1:oo(e)?128:Ll(e)?64:te(e)?4:z(e)?2:0;return y(e,t,n,s,i,l,r,!0)}function Ra(e){return e?Gs(e)||Xr(e)?Te({},e):e:null}function Qt(e,t,n=!1,s=!1){const{props:i,ref:r,patchFlag:l,children:a,transition:f}=e,m=t?co(i||{},t):i,p={__v_isVNode:!0,__v_skip:!0,type:e.type,props:m,key:m&&ao(m),ref:t&&t.ref?n&&r?B(r)?r.concat(Ln(t)):[r,Ln(t)]:Ln(t):r,scopeId:e.scopeId,slotScopeIds:e.slotScopeIds,children:a,target:e.target,targetStart:e.targetStart,targetAnchor:e.targetAnchor,staticCount:e.staticCount,shapeFlag:e.shapeFlag,patchFlag:t&&e.type!==fe?l===-1?16:l|16:l,dynamicProps:e.dynamicProps,dynamicChildren:e.dynamicChildren,appContext:e.appContext,dirs:e.dirs,transition:f,component:e.component,suspense:e.suspense,ssContent:e.ssContent&&Qt(e.ssContent),ssFallback:e.ssFallback&&Qt(e.ssFallback),placeholder:e.placeholder,el:e.el,anchor:e.anchor,ctx:e.ctx,ce:e.ce};return f&&s&&Ys(p,f.clone(p)),p}function en(e=" ",t=0){return q(ns,null,e,t)}function We(e="",t=!1){return t?(L(),mt(ht,null,e)):q(ht,null,e)}function Qe(e){return e==null||typeof e=="boolean"?q(ht):B(e)?q(fe,null,e.slice()):ei(e)?at(e):q(ns,null,String(e))}function at(e){return e.el===null&&e.patchFlag!==-1||e.memo?e:Qt(e)}function ti(e,t){let n=0;const{shapeFlag:s}=e;if(t==null)t=null;else if(B(t))n=16;else if(typeof t=="object")if(s&65){const i=t.default;i&&(i._c&&(i._d=!1),ti(e,i()),i._c&&(i._d=!0));return}else{n=32;const i=t._;!i&&!Xr(t)?t._ctx=ye:i===3&&ye&&(ye.slots._===1?t._=1:(t._=2,e.patchFlag|=1024))}else z(t)?(t={default:t,_ctx:ye},n=32):(t=String(t),s&64?(n=16,t=[en(t)]):n=8);e.children=t,e.shapeFlag|=n}function co(...e){const t={};for(let n=0;nSe||ye;let qn,Os;{const e=Xn(),t=(n,s)=>{let i;return(i=e[n])||(i=e[n]=[]),i.push(s),r=>{i.length>1?i.forEach(l=>l(r)):i[0](r)}};qn=t("__VUE_INSTANCE_SETTERS__",n=>Se=n),Os=t("__VUE_SSR_SETTERS__",n=>bn=n)}const In=e=>{const t=Se;return qn(e),e.scope.on(),()=>{e.scope.off(),qn(t)}},Ui=()=>{Se&&Se.scope.off(),qn(null)};function uo(e){return e.vnode.shapeFlag&4}let bn=!1;function Da(e,t=!1,n=!1){t&&Os(t);const{props:s,children:i}=e.vnode,r=uo(e);va(e,s,r,t),wa(e,i,n||t);const l=r?Fa(e,t):void 0;return t&&Os(!1),l}function Fa(e,t){const n=e.type;e.accessCache=Object.create(null),e.proxy=new Proxy(e.ctx,na);const{setup:s}=n;if(s){ft();const i=e.setupContext=s.length>1?Ua(e):null,r=In(e),l=xn(s,e,0,[e.props,i]),a=lr(l);if(dt(),r(),(a||e.sp)&&!Jt(e)&&Nr(e),a){if(l.then(Ui,Ui),t)return l.then(f=>{Bi(e,f)}).catch(f=>{Qn(f,e,0)});e.asyncDep=l}else Bi(e,l)}else fo(e)}function Bi(e,t,n){z(t)?e.type.__ssrInlineRender?e.ssrRender=t:e.render=t:te(t)&&(e.setupState=Mr(t)),fo(e)}function fo(e,t,n){const s=e.type;e.render||(e.render=s.render||tt);{const i=In(e);ft();try{sa(e)}finally{dt(),i()}}}const Na={get(e,t){return Ie(e,"get",""),e[t]}};function Ua(e){const t=n=>{e.exposed=n||{}};return{attrs:new Proxy(e.attrs,Na),slots:e.slots,emit:e.emit,expose:t}}function ss(e){return e.exposed?e.exposeProxy||(e.exposeProxy=new Proxy(Mr(xl(e.exposed)),{get(t,n){if(n in t)return t[n];if(n in pn)return pn[n](e)},has(t,n){return n in t||n in pn}})):e.proxy}function Ba(e,t=!0){return z(e)?e.displayName||e.name:e.name||t&&e.__name}function La(e){return z(e)&&"__vccOpts"in e}const Ft=(e,t)=>Ml(e,t,bn),ja="3.5.30";/** -* @vue/runtime-dom v3.5.30 -* (c) 2018-present Yuxi (Evan) You and Vue contributors -* @license MIT -**/let Ps;const Li=typeof window<"u"&&window.trustedTypes;if(Li)try{Ps=Li.createPolicy("vue",{createHTML:e=>e})}catch{}const po=Ps?e=>Ps.createHTML(e):e=>e,Ha="http://www.w3.org/2000/svg",za="http://www.w3.org/1998/Math/MathML",lt=typeof document<"u"?document:null,ji=lt&<.createElement("template"),Ka={insert:(e,t,n)=>{t.insertBefore(e,n||null)},remove:e=>{const t=e.parentNode;t&&t.removeChild(e)},createElement:(e,t,n,s)=>{const i=t==="svg"?lt.createElementNS(Ha,e):t==="mathml"?lt.createElementNS(za,e):n?lt.createElement(e,{is:n}):lt.createElement(e);return e==="select"&&s&&s.multiple!=null&&i.setAttribute("multiple",s.multiple),i},createText:e=>lt.createTextNode(e),createComment:e=>lt.createComment(e),setText:(e,t)=>{e.nodeValue=t},setElementText:(e,t)=>{e.textContent=t},parentNode:e=>e.parentNode,nextSibling:e=>e.nextSibling,querySelector:e=>lt.querySelector(e),setScopeId(e,t){e.setAttribute(t,"")},insertStaticContent(e,t,n,s,i,r){const l=n?n.previousSibling:t.lastChild;if(i&&(i===r||i.nextSibling))for(;t.insertBefore(i.cloneNode(!0),n),!(i===r||!(i=i.nextSibling)););else{ji.innerHTML=po(s==="svg"?`${e}`:s==="mathml"?`${e}`:e);const a=ji.content;if(s==="svg"||s==="mathml"){const f=a.firstChild;for(;f.firstChild;)a.appendChild(f.firstChild);a.removeChild(f)}t.insertBefore(a,n)}return[l?l.nextSibling:t.firstChild,n?n.previousSibling:t.lastChild]}},Wa=Symbol("_vtc");function Va(e,t,n){const s=e[Wa];s&&(t=(t?[t,...s]:[...s]).join(" ")),t==null?e.removeAttribute("class"):n?e.setAttribute("class",t):e.className=t}const Hi=Symbol("_vod"),qa=Symbol("_vsh"),Ga=Symbol(""),Ja=/(?:^|;)\s*display\s*:/;function Ya(e,t,n){const s=e.style,i=ae(n);let r=!1;if(n&&!i){if(t)if(ae(t))for(const l of t.split(";")){const a=l.slice(0,l.indexOf(":")).trim();n[a]==null&&jn(s,a,"")}else for(const l in t)n[l]==null&&jn(s,l,"");for(const l in n)l==="display"&&(r=!0),jn(s,l,n[l])}else if(i){if(t!==n){const l=s[Ga];l&&(n+=";"+l),s.cssText=n,r=Ja.test(n)}}else t&&e.removeAttribute("style");Hi in e&&(e[Hi]=r?s.display:"",e[qa]&&(s.display="none"))}const zi=/\s*!important$/;function jn(e,t,n){if(B(n))n.forEach(s=>jn(e,t,s));else if(n==null&&(n=""),t.startsWith("--"))e.setProperty(t,n);else{const s=Xa(e,t);zi.test(n)?e.setProperty(It(s),n.replace(zi,""),"important"):e[s]=n}}const Ki=["Webkit","Moz","ms"],_s={};function Xa(e,t){const n=_s[t];if(n)return n;let s=Ee(t);if(s!=="filter"&&s in e)return _s[t]=s;s=Yn(s);for(let i=0;ibs||(tc.then(()=>bs=0),bs=Date.now());function sc(e,t){const n=s=>{if(!s._vts)s._vts=Date.now();else if(s._vts<=n.attached)return;nt(ic(s,n.value),t,5,[s])};return n.value=e,n.attached=nc(),n}function ic(e,t){if(B(t)){const n=e.stopImmediatePropagation;return e.stopImmediatePropagation=()=>{n.call(e),e._stopped=!0},t.map(s=>i=>!i._stopped&&s&&s(i))}else return t}const Yi=e=>e.charCodeAt(0)===111&&e.charCodeAt(1)===110&&e.charCodeAt(2)>96&&e.charCodeAt(2)<123,rc=(e,t,n,s,i,r)=>{const l=i==="svg";t==="class"?Va(e,s,l):t==="style"?Ya(e,n,s):Gn(t)?Fs(t)||Qa(e,t,n,s,r):(t[0]==="."?(t=t.slice(1),!0):t[0]==="^"?(t=t.slice(1),!1):oc(e,t,s,l))?(qi(e,t,s),!e.tagName.includes("-")&&(t==="value"||t==="checked"||t==="selected")&&Vi(e,t,s,l,r,t!=="value")):e._isVueCE&&(lc(e,t)||e._def.__asyncLoader&&(/[A-Z]/.test(t)||!ae(s)))?qi(e,Ee(t),s,r,t):(t==="true-value"?e._trueValue=s:t==="false-value"&&(e._falseValue=s),Vi(e,t,s,l))};function oc(e,t,n,s){if(s)return!!(t==="innerHTML"||t==="textContent"||t in e&&Yi(t)&&z(n));if(t==="spellcheck"||t==="draggable"||t==="translate"||t==="autocorrect"||t==="sandbox"&&e.tagName==="IFRAME"||t==="form"||t==="list"&&e.tagName==="INPUT"||t==="type"&&e.tagName==="TEXTAREA")return!1;if(t==="width"||t==="height"){const i=e.tagName;if(i==="IMG"||i==="VIDEO"||i==="CANVAS"||i==="SOURCE")return!1}return Yi(t)&&ae(n)?!1:t in e}function lc(e,t){const n=e._def.props;if(!n)return!1;const s=Ee(t);return Array.isArray(n)?n.some(i=>Ee(i)===s):Object.keys(n).some(i=>Ee(i)===s)}const Xi=e=>{const t=e.props["onUpdate:modelValue"]||!1;return B(t)?n=>Un(t,n):t};function ac(e){e.target.composing=!0}function Zi(e){const t=e.target;t.composing&&(t.composing=!1,t.dispatchEvent(new Event("input")))}const ws=Symbol("_assign");function Qi(e,t,n){return t&&(e=e.trim()),n&&(e=Bs(e)),e}const cc={created(e,{modifiers:{lazy:t,trim:n,number:s}},i){e[ws]=Xi(i);const r=s||i.props&&i.props.type==="number";zt(e,t?"change":"input",l=>{l.target.composing||e[ws](Qi(e.value,n,r))}),(n||r)&&zt(e,"change",()=>{e.value=Qi(e.value,n,r)}),t||(zt(e,"compositionstart",ac),zt(e,"compositionend",Zi),zt(e,"change",Zi))},mounted(e,{value:t}){e.value=t??""},beforeUpdate(e,{value:t,oldValue:n,modifiers:{lazy:s,trim:i,number:r}},l){if(e[ws]=Xi(l),e.composing)return;const a=(r||e.type==="number")&&!/^0\d/.test(e.value)?Bs(e.value):e.value,f=t??"";a!==f&&(document.activeElement===e&&e.type!=="range"&&(s&&t===n||i&&e.value.trim()===f)||(e.value=f))}},uc=["ctrl","shift","alt","meta"],fc={stop:e=>e.stopPropagation(),prevent:e=>e.preventDefault(),self:e=>e.target!==e.currentTarget,ctrl:e=>!e.ctrlKey,shift:e=>!e.shiftKey,alt:e=>!e.altKey,meta:e=>!e.metaKey,left:e=>"button"in e&&e.button!==0,middle:e=>"button"in e&&e.button!==1,right:e=>"button"in e&&e.button!==2,exact:(e,t)=>uc.some(n=>e[`${n}Key`]&&!t.includes(n))},En=(e,t)=>{if(!e)return e;const n=e._withMods||(e._withMods={}),s=t.join(".");return n[s]||(n[s]=(i,...r)=>{for(let l=0;l{const n=e._withKeys||(e._withKeys={}),s=t.join(".");return n[s]||(n[s]=i=>{if(!("key"in i))return;const r=It(i.key);if(t.some(l=>l===r||dc[l]===r))return e(i)})},pc=Te({patchProp:rc},Ka);let er;function hc(){return er||(er=Ia(pc))}const mc=(...e)=>{const t=hc().createApp(...e),{mount:n}=t;return t.mount=s=>{const i=vc(s);if(!i)return;const r=t._component;!z(r)&&!r.render&&!r.template&&(r.template=i.innerHTML),i.nodeType===1&&(i.textContent="");const l=n(i,!1,gc(i));return i instanceof Element&&(i.removeAttribute("v-cloak"),i.setAttribute("data-v-app","")),l},t};function gc(e){if(e instanceof SVGElement)return"svg";if(typeof MathMLElement=="function"&&e instanceof MathMLElement)return"mathml"}function vc(e){return ae(e)?document.querySelector(e):e}const yc={class:"footer"},_c={__name:"AppFooter",emits:["request-admin"],setup(e,{emit:t}){const n=t,s=oe(0);let i=null;Xs(()=>{i&&window.clearTimeout(i)});function r(){s.value+=1,s.value===1&&(i=window.setTimeout(()=>{s.value=0,i=null},2e3)),s.value>=5&&(i&&(window.clearTimeout(i),i=null),s.value=0,n("request-admin"))}return(l,a)=>(L(),V("div",yc,[y("div",null,[a[0]||(a[0]=en(" © 2026 AirShare Pro. All rights reserved. ",-1)),a[1]||(a[1]=y("span",{class:"divider-line"},"|",-1)),y("span",{id:"admin-trigger",title:"点击 5 次进入后台",onClick:r},"V 1.0.0")]),a[2]||(a[2]=y("div",{style:{"font-size":"12px","margin-top":"4px"}},[y("a",{href:"https://beian.miit.gov.cn/",target:"_blank",rel:"noreferrer"}," 粤ICP备2026888888号-1 ")],-1))]))}},bc=["fill","stroke"],pe={__name:"LocalIcon",props:{name:{type:String,required:!0},size:{type:[Number,String],default:24}},setup(e){const t=e,n={light_mode:{type:"stroke",shapes:[{tag:"circle",attrs:{cx:"12",cy:"12",r:"4"}},{tag:"path",attrs:{d:"M12 2v2.2"}},{tag:"path",attrs:{d:"M12 19.8V22"}},{tag:"path",attrs:{d:"M4.93 4.93 6.5 6.5"}},{tag:"path",attrs:{d:"m17.5 17.5 1.57 1.57"}},{tag:"path",attrs:{d:"M2 12h2.2"}},{tag:"path",attrs:{d:"M19.8 12H22"}},{tag:"path",attrs:{d:"m4.93 19.07 1.57-1.57"}},{tag:"path",attrs:{d:"M17.5 6.5 19.07 4.93"}}]},dark_mode:{type:"fill",shapes:[{tag:"path",attrs:{d:"M21 12.8A9 9 0 1 1 11.2 3a7 7 0 0 0 9.8 9.8Z"}}]},add_circle:{type:"stroke",shapes:[{tag:"circle",attrs:{cx:"12",cy:"12",r:"9"}},{tag:"path",attrs:{d:"M12 8v8"}},{tag:"path",attrs:{d:"M8 12h8"}}]},sensors:{type:"stroke",shapes:[{tag:"path",attrs:{d:"M12 12h.01"}},{tag:"path",attrs:{d:"M9.2 14.8a4 4 0 0 1 0-5.6"}},{tag:"path",attrs:{d:"M14.8 9.2a4 4 0 0 1 0 5.6"}},{tag:"path",attrs:{d:"M6.4 17.6a8 8 0 0 1 0-11.2"}},{tag:"path",attrs:{d:"M17.6 6.4a8 8 0 0 1 0 11.2"}}]},smartphone:{type:"stroke",shapes:[{tag:"rect",attrs:{x:"7",y:"2.5",width:"10",height:"19",rx:"2.5"}},{tag:"path",attrs:{d:"M10 5h4"}},{tag:"circle",attrs:{cx:"12",cy:"18",r:"0.8"}}]},laptop_mac:{type:"stroke",shapes:[{tag:"rect",attrs:{x:"5",y:"4",width:"14",height:"10",rx:"1.5"}},{tag:"path",attrs:{d:"M3 18h18"}},{tag:"path",attrs:{d:"M8 18a2 2 0 0 0 2 2h4a2 2 0 0 0 2-2"}}]},close:{type:"stroke",shapes:[{tag:"path",attrs:{d:"M6 6l12 12"}},{tag:"path",attrs:{d:"M18 6 6 18"}}]},cloud_upload:{type:"stroke",shapes:[{tag:"path",attrs:{d:"M7 18a4 4 0 1 1 .7-7.94A5.5 5.5 0 0 1 18 11a3.5 3.5 0 1 1-.5 7"}},{tag:"path",attrs:{d:"M12 10v8"}},{tag:"path",attrs:{d:"m8.8 13.2 3.2-3.2 3.2 3.2"}}]},arrow_upward:{type:"stroke",shapes:[{tag:"path",attrs:{d:"M12 19V6"}},{tag:"path",attrs:{d:"m6.5 11.5 5.5-5.5 5.5 5.5"}}]},send_and_archive:{type:"stroke",shapes:[{tag:"path",attrs:{d:"M3 6h18l-2 4H5Z"}},{tag:"path",attrs:{d:"M5 10v8a2 2 0 0 0 2 2h10a2 2 0 0 0 2-2v-8"}},{tag:"path",attrs:{d:"M12 11v5"}},{tag:"path",attrs:{d:"m9.5 13.5 2.5 2.5 2.5-2.5"}}]},chat_bubble:{type:"stroke",shapes:[{tag:"path",attrs:{d:"M6 18.5 3.5 21v-5A7.5 7.5 0 0 1 11 4.5h2A7.5 7.5 0 0 1 20.5 12v.5A7.5 7.5 0 0 1 13 20H8.5"}}]},content_copy:{type:"stroke",shapes:[{tag:"rect",attrs:{x:"9",y:"9",width:"10",height:"10",rx:"2"}},{tag:"path",attrs:{d:"M7 15H6a2 2 0 0 1-2-2V6a2 2 0 0 1 2-2h7a2 2 0 0 1 2 2v1"}}]},check:{type:"stroke",shapes:[{tag:"path",attrs:{d:"m5 12 4.2 4.2L19 7.5"}}]},draft:{type:"stroke",shapes:[{tag:"rect",attrs:{x:"4",y:"5",width:"16",height:"14",rx:"2"}},{tag:"path",attrs:{d:"m5 7 7 5 7-5"}}]},save:{type:"stroke",shapes:[{tag:"path",attrs:{d:"M5 20h14a1 1 0 0 0 1-1V7.5L16.5 4H5a1 1 0 0 0-1 1v14a1 1 0 0 0 1 1Z"}},{tag:"path",attrs:{d:"M8 4v5h7"}},{tag:"rect",attrs:{x:"8",y:"14",width:"8",height:"4",rx:"1"}}]},download:{type:"stroke",shapes:[{tag:"path",attrs:{d:"M12 5v10"}},{tag:"path",attrs:{d:"m7.5 10.5 4.5 4.5 4.5-4.5"}},{tag:"path",attrs:{d:"M5 19h14"}}]}},s=Ft(()=>n[t.name]||n.close),i=Ft(()=>typeof t.size=="number"?`${t.size}px`:/^\d+(\.\d+)?$/.test(t.size)?`${t.size}px`:t.size);return(r,l)=>(L(),V("span",{class:"app-icon",style:Be({width:i.value,height:i.value}),"aria-hidden":"true"},[(L(),V("svg",{viewBox:"0 0 24 24",fill:s.value.type==="fill"?"currentColor":"none",stroke:s.value.type==="stroke"?"currentColor":"none","stroke-width":"1.8","stroke-linecap":"round","stroke-linejoin":"round"},[(L(!0),V(fe,null,Zt(s.value.shapes,(a,f)=>(L(),mt(Ql(a.tag),co({key:`${e.name}-${f}`},{ref_for:!0},a.attrs),null,16))),128))],8,bc))],4))}},wc={class:"header"},xc={__name:"AppHeader",props:{theme:{type:String,required:!0}},emits:["toggle-theme"],setup(e){return(t,n)=>(L(),V("div",wc,[n[1]||(n[1]=y("h1",null,"AirShare Pro",-1)),n[2]||(n[2]=y("p",null,"跨端局域网 & P2P 传输中心",-1)),y("button",{class:"theme-toggle",title:"切换日夜模式",onClick:n[0]||(n[0]=s=>t.$emit("toggle-theme"))},[q(pe,{id:"theme-icon",name:e.theme==="dark"?"dark_mode":"light_mode",size:"22"},null,8,["name"])])]))}},Ic={class:"card"},Sc={key:0,class:"section-title"},mn={__name:"GlassCard",props:{title:{type:String,default:""}},setup(e){return(t,n)=>(L(),V("div",Ic,[e.title?(L(),V("div",Sc,se(e.title),1)):We("",!0),ta(t.$slots,"default")]))}},Cc={class:"admin-panel active"},Tc={class:"card admin-header-card"},$c={class:"transfer-head transfer-head-compact"},Mc={class:"main-grid admin-summary-grid"},kc={class:"admin-stats-panel"},Rc={class:"admin-stats-row"},Ec={class:"admin-fluid-content"},Ac={class:"admin-fluid-icon"},Oc={class:"admin-fluid-copy"},Pc={key:0,class:"stat-suffix"},Dc={class:"admin-config-stack"},Fc={class:"text-input-group admin-config-row admin-config-row-field admin-config-row-last"},Nc={class:"admin-field-control-row"},Uc=["value"],Bc={class:"text-input-group admin-config-row admin-config-row-field admin-config-row-last"},Lc={class:"admin-field-control-row"},jc=["value"],Hc={class:"admin-config-insights"},zc={class:"admin-config-highlight"},Kc={class:"admin-config-highlight"},Wc={class:"admin-table-wrapper"},Vc={class:"admin-table"},qc={__name:"AdminPanel",props:{stats:{type:Array,required:!0},records:{type:Array,required:!0},fileLimit:{type:Number,required:!0},minioCapacity:{type:Number,required:!0}},emits:["exit","save-config","update:file-limit","update:minio-capacity"],setup(e){function t(l){const a=Number(l)||0;return a>=1024?`${(a/1024).toFixed(a%1024===0?0:1)} GB`:`${a} MB`}function n(l){const a=Number(l)||0;return a>=1024?`${(a/1024).toFixed(a%1024===0?0:1)} TB`:`${a} GB`}function s(l){return l==="blue"?{color:"var(--accent-blue)"}:l==="cyan"?{color:"var(--accent-cyan)"}:l==="success"?{color:"var(--success-green)"}:l==="danger"?{color:"var(--danger-red)"}:{color:"var(--text-main)"}}function i(l){return l==="success"?{color:"var(--success-green)",fontWeight:500}:l==="primary"?{color:"var(--accent-blue)",fontWeight:500}:{color:"var(--danger-red)",fontWeight:500}}function r(l){const a=Number(l)||0;return{"--fluid-level":`${Math.max(0,Math.min(a,100))}%`}}return(l,a)=>(L(),V("div",Cc,[y("div",Tc,[y("div",$c,[a[5]||(a[5]=y("div",{class:"connected-to"},[y("h2",{class:"admin-title"},"管理控制台"),y("p",{class:"admin-subtitle"},"AirShare Pro System Dashboard")],-1)),y("button",{class:"btn-small-primary",type:"button",onClick:a[0]||(a[0]=f=>l.$emit("exit"))},"退出管理")])]),y("div",Mc,[q(mn,{class:"admin-stats-card",title:"系统运行状态"},{default:Gt(()=>[y("div",kc,[y("div",Rc,[(L(!0),V(fe,null,Zt(e.stats,f=>(L(),V("div",{key:f.label,class:xt(["admin-stat-item",{"admin-stat-item-fluid":f.kind==="minio"}])},[f.kind==="minio"?(L(),V("div",{key:0,class:"admin-fluid-card",style:Be(r(f.percent))},[a[6]||(a[6]=y("div",{class:"admin-fluid-fill"},[y("span",{class:"admin-fluid-wave admin-fluid-wave-a"}),y("span",{class:"admin-fluid-wave admin-fluid-wave-b"})],-1)),y("div",Ec,[y("div",Ac,[q(pe,{name:"save",size:"18"})]),y("div",Oc,[y("h3",{style:Be(s(f.tone))},se(f.value),5),y("p",null,se(f.label),1),y("small",null,se(f.detail),1)])])],4)):(L(),V(fe,{key:1},[a[7]||(a[7]=y("span",{class:"admin-stat-kicker"},"实时指标",-1)),y("h3",{style:Be(s(f.tone))},[en(se(f.value),1),f.suffix?(L(),V("span",Pc,se(f.suffix),1)):We("",!0)],4),y("p",null,se(f.label),1)],64))],2))),128))])])]),_:1}),q(mn,{class:"admin-config-card",title:"核心参数配置"},{default:Gt(()=>[y("div",Dc,[y("div",Fc,[a[8]||(a[8]=y("div",{class:"admin-field-meta"},[y("label",{class:"admin-field-label",for:"admin-file-limit"},"单文件大小限制"),y("p",{class:"admin-field-hint"},"单位为 MB,超过该阈值的文件会按当前后端策略处理。")],-1)),y("div",Nc,[y("input",{id:"admin-file-limit",value:e.fileLimit,min:"1",placeholder:"10240",type:"number",onInput:a[1]||(a[1]=f=>l.$emit("update:file-limit",Number(f.target.value)||0))},null,40,Uc),y("button",{title:"保存配置",type:"button",onClick:a[2]||(a[2]=f=>l.$emit("save-config"))},[q(pe,{name:"save",size:"18"})])])]),y("div",Bc,[a[9]||(a[9]=y("div",{class:"admin-field-meta"},[y("label",{class:"admin-field-label",for:"admin-minio-capacity"},"MinIO 总容量"),y("p",{class:"admin-field-hint"},"单位为 GB,用于容量卡和液位比例计算。")],-1)),y("div",Lc,[y("input",{id:"admin-minio-capacity",value:e.minioCapacity,min:"1",placeholder:"120",type:"number",onInput:a[3]||(a[3]=f=>l.$emit("update:minio-capacity",Number(f.target.value)||0))},null,40,jc),y("button",{title:"保存配置",type:"button",onClick:a[4]||(a[4]=f=>l.$emit("save-config"))},[q(pe,{name:"save",size:"18"})])])]),y("div",Hc,[y("div",zc,[a[10]||(a[10]=y("span",{class:"admin-config-badge"},"ACTIVE POLICY",-1)),y("h3",null,se(t(e.fileLimit)),1),a[11]||(a[11]=y("p",null,"当前单文件阈值。超过该体积后,文件会按后端已设定的传输与存档策略处理。",-1))]),y("div",Kc,[a[12]||(a[12]=y("span",{class:"admin-config-badge"},"MINIO CAPACITY",-1)),y("h3",null,se(n(e.minioCapacity)),1),a[13]||(a[13]=y("p",null,"当前 MinIO 总容量基线,用于后台容量展示与液位占比计算。",-1))])])])]),_:1})]),q(mn,{class:"admin-table-card",title:"最近传输记录(Top 5)"},{default:Gt(()=>[y("div",Wc,[y("table",Vc,[a[14]||(a[14]=y("thead",null,[y("tr",null,[y("th",null,"时间"),y("th",null,"发送端特征"),y("th",null,"传输类型"),y("th",null,"数据量"),y("th",null,"状态")])],-1)),y("tbody",null,[(L(!0),V(fe,null,Zt(e.records,f=>(L(),V("tr",{key:`${f.time}-${f.peer}`},[y("td",null,se(f.time),1),y("td",null,se(f.peer),1),y("td",null,se(f.type),1),y("td",null,se(f.size),1),y("td",{style:Be(i(f.tone))},se(f.status),5)]))),128))])])])]),_:1})]))}},Gc={class:"local-device-name"},Jc={key:0,class:"radar-container"},Yc={class:"radar"},Xc={key:1,class:"device-list"},Zc=["onClick"],Qc={class:"device-icon"},eu={class:"device-info"},tu={key:2,class:"radar-container"},nu={class:"radar"},su={__name:"DeviceDiscoveryCard",props:{isScanning:{type:Boolean,required:!0},localDeviceName:{type:String,default:""},devices:{type:Array,required:!0}},emits:["select-device"],setup(e,{emit:t}){const n=t;function s(i){n("select-device",i)}return(i,r)=>(L(),mt(mn,{title:"局域网自动发现"},{default:Gt(()=>[y("p",Gc,[r[0]||(r[0]=en(" 本机:",-1)),y("strong",null,se(e.localDeviceName||"识别中"),1)]),e.isScanning?(L(),V("div",Jc,[y("div",Yc,[q(pe,{class:"radar-icon",name:"sensors",size:"36"})]),r[1]||(r[1]=y("p",{class:"scan-status"},"正在扫描附近设备...",-1))])):e.devices.length?(L(),V("div",Xc,[(L(!0),V(fe,null,Zt(e.devices,l=>(L(),V("button",{key:l.id,class:"device-item",type:"button",onClick:a=>s(l)},[y("div",Qc,[q(pe,{name:l.icon,size:"24"},null,8,["name"])]),y("div",eu,[y("h4",null,se(l.name),1),y("p",null,se(l.description),1)]),r[2]||(r[2]=y("div",{class:"device-status-beacon","aria-hidden":"true"},[y("span",{class:"device-status-dot"}),y("span",{class:"device-status-ring"}),y("span",{class:"device-status-ring device-status-ring-delay"})],-1))],8,Zc))),128))])):(L(),V("div",tu,[y("div",nu,[q(pe,{class:"radar-icon",name:"devices",size:"36"})]),r[3]||(r[3]=y("p",{class:"scan-status"},"暂未发现局域网设备,请保持页面开启后重试",-1))]))]),_:1}))}},iu={key:0,class:"room-action-area"},ru={class:"room-input-group"},ou=["value"],lu={key:0,class:"pending-downloads"},au={class:"pending-downloads-head"},cu=["href"],uu={class:"pending-download-copy"},fu=["title"],du={class:"pending-download-icon","aria-hidden":"true"},pu={key:1,class:"waiting-area"},hu={class:"huge-code"},mu={__name:"RemoteRoomCard",props:{roomCodeInput:{type:String,required:!0},isWaiting:{type:Boolean,required:!0},generatedCode:{type:String,required:!0},pendingDownloads:{type:Array,required:!0}},emits:["update-room-code","create-room","join-room","cancel-room"],setup(e,{emit:t}){const n=t;function s(r){n("update-room-code",r.target.value)}function i(){n("join-room")}return(r,l)=>(L(),mt(mn,{title:"远程直连"},{default:Gt(()=>[e.isWaiting?(L(),V("div",pu,[l[6]||(l[6]=y("p",{class:"waiting-subtitle"},"您的房间号码",-1)),y("div",hu,se(e.generatedCode),1),l[7]||(l[7]=y("div",{class:"spinner"},null,-1)),l[8]||(l[8]=y("p",{class:"waiting-tip"},"等待对方加入...",-1)),y("button",{class:"btn-cancel",type:"button",onClick:l[2]||(l[2]=a=>r.$emit("cancel-room"))},"取消建房")])):(L(),V("div",iu,[y("button",{class:"btn-create",type:"button",onClick:l[0]||(l[0]=a=>r.$emit("create-room"))},[q(pe,{name:"add_circle",size:"22"}),l[3]||(l[3]=en(" 创建专属传输房间 ",-1))]),l[5]||(l[5]=y("div",{class:"divider"},"或",-1)),y("div",ru,[y("input",{class:"room-code",inputmode:"numeric",maxlength:"4",pattern:"\\d*",placeholder:"输入4位房间号",type:"text",value:e.roomCodeInput,onInput:s,onKeyup:ho(i,["enter"])},null,40,ou),y("button",{class:"btn-primary",type:"button",onClick:l[1]||(l[1]=a=>r.$emit("join-room"))},"加入房间")]),e.pendingDownloads.length?(L(),V("div",lu,[y("div",au,[l[4]||(l[4]=y("span",null,"待领取文件",-1)),y("span",null,se(e.pendingDownloads.length),1)]),(L(!0),V(fe,null,Zt(e.pendingDownloads,a=>(L(),V("a",{key:a.transfer_id,class:"pending-download-item",href:a.download_path,target:"_blank",rel:"noopener noreferrer"},[y("div",uu,[y("strong",{title:a.name},se(a.name),9,fu),y("p",null,se(a.size_label)+" · "+se(a.created_label),1)]),y("span",du,[q(pe,{name:"download",size:"18"})])],8,cu))),128))])):We("",!0)]))]),_:1}))}},gu={class:"file-info"},vu=["title"],yu={class:"file-info-right"},_u=["download","href"],bu={key:0,class:"progress-bar-container"},wu={__name:"TransferQueueItem",props:{item:{type:Object,required:!0}},emits:["remove","start-upload","copy"],setup(e){const t=e,n=Ft(()=>t.item.tone==="success"?{color:"var(--success-green)"}:t.item.tone==="primary"?{color:"var(--accent-blue)"}:t.item.tone==="danger"?{color:"var(--danger-red)"}:{color:"var(--text-secondary)"}),s=Ft(()=>t.item.kind==="text"?{color:"var(--success-green)",background:"rgba(48, 209, 88, 0.1)"}:{});return(i,r)=>(L(),V("div",{class:xt(["batch-item",{"pending-file":e.item.kind==="file"&&e.item.pending}])},[y("div",gu,[y("div",{class:"file-info-left",style:Be(e.item.kind==="text"?{maxWidth:"70%"}:null)},[y("div",{class:"file-icon-wrapper",style:Be(s.value)},[q(pe,{name:e.item.kind==="text"?"chat_bubble":"draft",size:"18"},null,8,["name"])],4),y("span",{class:"file-name",title:e.item.kind==="text"?e.item.text:e.item.name},se(e.item.kind==="text"?e.item.text:e.item.name),9,vu)],4),y("div",yu,[y("span",{class:"file-status",style:Be(n.value)},se(e.item.kind==="text"&&e.item.copied?"已复制":e.item.status),5),e.item.kind==="text"?(L(),V("button",{key:0,class:"action-btn",title:"复制文本",type:"button",onClick:r[0]||(r[0]=l=>i.$emit("copy",e.item.id))},[q(pe,{name:e.item.copied?"check":"content_copy",size:"16"},null,8,["name"])])):We("",!0),e.item.kind==="file"&&e.item.pending?(L(),V("button",{key:1,class:"action-btn primary",title:"发送文件",type:"button",onClick:r[1]||(r[1]=l=>i.$emit("start-upload",e.item.id))},[q(pe,{name:"arrow_upward",size:"16"})])):We("",!0),e.item.kind==="file"&&e.item.downloadUrl?(L(),V("a",{key:2,class:"action-btn primary",download:e.item.name,href:e.item.downloadUrl,title:"保存文件"},[q(pe,{name:"download",size:"16"})],8,_u)):We("",!0),y("button",{class:"action-btn danger",title:"移除记录",type:"button",onClick:r[2]||(r[2]=l=>i.$emit("remove",e.item.id))},[q(pe,{name:"close",size:"16"})])])]),e.item.kind==="file"?(L(),V("div",bu,[y("div",{class:xt(["progress-bar-fill",{success:e.item.tone==="success"}]),style:Be({width:`${e.item.progress}%`})},null,6)])):We("",!0)],2))}},xu={class:"transfer-panel active"},Iu={class:"card"},Su={class:"transfer-head"},Cu={class:"connected-to"},Tu={class:"text-input-group"},$u={__name:"TransferPanel",props:{peerName:{type:String,required:!0},connectionType:{type:String,required:!0},items:{type:Array,required:!0},hasPendingItems:{type:Boolean,required:!0}},emits:["close","send-text","files-selected","send-all-pending","remove-item","start-upload","copy-item"],setup(e,{emit:t}){const n=e,s=t,i=oe(""),r=oe(!1),l=oe(null),a=oe(null);Dt(()=>n.items.length,async()=>{await Rr(),l.value&&(l.value.scrollTop=l.value.scrollHeight)});function f(){var T;(T=a.value)==null||T.click()}function m(){s("send-text",i.value),i.value=""}function p(T){const S=Array.from(T.target.files||[]);S.length&&s("files-selected",S),T.target.value=""}function _(T){var U;r.value=!1;const S=Array.from(((U=T.dataTransfer)==null?void 0:U.files)||[]);S.length&&s("files-selected",S)}return(T,S)=>(L(),V("div",xu,[y("div",Iu,[y("div",Su,[y("div",Cu,[y("h2",null,se(e.peerName),1),y("p",null,se(e.connectionType),1)]),y("button",{class:"close-btn",type:"button",onClick:S[0]||(S[0]=U=>T.$emit("close"))},[q(pe,{name:"close",size:"20"})])]),y("div",{class:xt(["drop-zone",{"drop-zone-active":r.value}]),onClick:f,onDragenter:S[1]||(S[1]=En(U=>r.value=!0,["prevent"])),onDragover:S[2]||(S[2]=En(U=>r.value=!0,["prevent"])),onDragleave:S[3]||(S[3]=En(U=>r.value=!1,["prevent"])),onDrop:En(_,["prevent"])},[q(pe,{class:"drop-zone-icon",name:"cloud_upload",size:"42"}),S[9]||(S[9]=y("p",{class:"drop-zone-text"},"点击或拖拽多个文件到这里",-1)),y("input",{ref_key:"fileInput",ref:a,class:"hidden",multiple:"",type:"file",onChange:p},null,544)],34),y("div",Tu,[Pl(y("input",{"onUpdate:modelValue":S[4]||(S[4]=U=>i.value=U),placeholder:"输入要发送的文本或链接...",type:"text",onKeyup:ho(m,["enter"])},null,544),[[cc,i.value]]),y("button",{title:"发送文本",type:"button",onClick:m},[q(pe,{name:"arrow_upward",size:"20"})])]),y("div",{class:xt(["batch-actions",{active:e.hasPendingItems}])},[y("button",{class:"btn-small-primary",type:"button",onClick:S[5]||(S[5]=U=>T.$emit("send-all-pending"))},[q(pe,{name:"send_and_archive",size:"16"}),S[10]||(S[10]=en(" 一键发送全部 ",-1))])],2),e.items.length?(L(),V("div",{key:0,ref_key:"batchContainer",ref:l,class:"batch-progress-container"},[(L(!0),V(fe,null,Zt(e.items,U=>(L(),mt(wu,{key:U.id,item:U,onCopy:S[6]||(S[6]=D=>T.$emit("copy-item",D)),onRemove:S[7]||(S[7]=D=>T.$emit("remove-item",D)),onStartUpload:S[8]||(S[8]=D=>T.$emit("start-upload",D))},null,8,["item"]))),128))],512)):We("",!0)])]))}};let Kt={deviceId:"",token:""};const Mu="filefast_device_id",ku="filefast_device_token";function mo(){return!Kt.deviceId||!Kt.token?{}:{"X-Device-ID":Kt.deviceId,"X-Device-Token":Kt.token}}function Ru(e={},t=!1){return{...t?{"Content-Type":"application/json"}:{},...mo(),...e}}function Eu(e,t){if(!t||Object.keys(t).length===0)return e;const n=new URLSearchParams;Object.entries(t).forEach(([i,r])=>{r!=null&&r!==""&&n.set(i,String(r))});const s=n.toString();return s?`${e}?${s}`:e}async function An(e,t={}){const n=t.body!==void 0,s=await fetch(Eu(e,t.query),{method:t.method||"GET",headers:Ru(t.headers,n),body:n?JSON.stringify(t.body):void 0}),i=await s.json().catch(()=>({}));if(!s.ok){const r=new Error(i.error||`Request failed: ${s.status}`);throw r.status=s.status,r}return i.data}const ve={get(e,t={}){return An(e,{...t,method:"GET"})},post(e,t,n={}){return An(e,{...n,method:"POST",body:t})},put(e,t,n={}){return An(e,{...n,method:"PUT",body:t})},patch(e,t,n={}){return An(e,{...n,method:"PATCH",body:t})}};function tr(e,t){Kt={deviceId:e||"",token:t||""},Pu(Kt)}function Au(){return mo()}function Ou(e){return{Authorization:`Bearer ${e}`}}function Pu(e){typeof document>"u"||(nr(Mu,e.deviceId),nr(ku,e.token))}function nr(e,t){if(!t){document.cookie=`${e}=; Path=/; Max-Age=0; SameSite=Lax`;return}document.cookie=`${e}=${encodeURIComponent(t)}; Path=/; SameSite=Lax`}function On(e){return{headers:Ou(e)}}const jt={login(e,t){return ve.post("/api/admin/login",{username:e,password:t})},stats(e){return ve.get("/api/admin/stats",On(e))},config(e){return ve.get("/api/admin/config",On(e))},updateConfig(e,t){return ve.put("/api/admin/config",t,On(e))},recentTransfers(e){return ve.get("/api/admin/transfers/recent",On(e))}},Pn={register(e){return ve.post("/api/devices/register",e)},heartbeat(e){return ve.post("/api/devices/heartbeat",{device_id:e})},listCandidates(e){return ve.get("/api/devices/candidates",{query:{deviceId:e}})},listPendingDownloads(e){return ve.get(`/api/devices/${encodeURIComponent(e)}/pending-downloads`)}},Dn={create(e){return ve.post("/api/rooms",{creator_device_id:e})},get(e){return ve.get(`/api/rooms/${encodeURIComponent(e)}`)},join(e,t){return ve.post("/api/rooms/join",{code:e,joiner_device_id:t})},cancel(e,t){return ve.post(`/api/rooms/${encodeURIComponent(e)}/cancel`,{requester_id:t})}},Du={config(){return ve.get("/api/runtime/config")}},ge={create(e){return ve.post("/api/transfers",e)},presignFallback(e){return ve.post(`/api/transfers/${encodeURIComponent(e)}/fallback/presign`,{})},uploadFallback(e,t,n){return Fu(`/api/transfers/${encodeURIComponent(e)}/fallback/upload`,t,n)},updateStatus(e,t){return ve.patch(`/api/transfers/${encodeURIComponent(e)}/status`,t)}};function Fu(e,t,n){return new Promise((s,i)=>{const r=new XMLHttpRequest;r.open("PUT",e),r.responseType="json",r.setRequestHeader("Content-Type",t.type||"application/octet-stream"),Object.entries(Au()).forEach(([l,a])=>{r.setRequestHeader(l,a)}),r.upload.onprogress=l=>{!l.lengthComputable||typeof n!="function"||n(Math.round(l.loaded/l.total*100))},r.onload=()=>{const l=r.response||Nu(r.responseText);if(r.status>=200&&r.status<300){s(l.data);return}i(new Error((l==null?void 0:l.error)||`Upload failed: ${r.status}`))},r.onerror=()=>i(new Error("Upload failed")),r.send(t)})}function Nu(e){try{return JSON.parse(e)}catch{return null}}const Uu={class:"container"},Bu={key:0,class:"main-grid"},xs="filefast-admin-token",Fn="filefast-admin-view",Nn="filefast-device-id",sr="filefast-device-name",Is="filefast-device-token",Lu=15e3,ju=5e3,Hu=2e3,zu=3e3,Ku=4*1024*1024,Wu=2e4,Vu=16*1024,ir=512*1024,qu={__name:"App",setup(e){const t=oe(localStorage.getItem("airshare-theme")||"light"),n=oe(localStorage.getItem(Fn)==="admin"?"admin":"main"),s=oe(!0),i=oe([]),r=oe(""),l=oe(!1),a=oe("----"),f=oe([]),m=oe({name:"--",type:"等待连接",deviceId:""}),p=oe([]),_=oe("/ws"),T=oe(10240),S=oe(120),U=oe([]),D=oe([]),Q=oe(null),K=oe(localStorage.getItem(xs)||""),R=oe({id:"",name:"",type:""}),ee=localStorage.getItem(Nn)||"",F=localStorage.getItem(Is)||"";ee&&F&&tr(ee,F);const le=new Map,he=new Map,$e=new Map,Me=new Map;let vt=null,je=null,Je=null,He=null,yt=null,de=null,st=null,H=null,j=null,G="",_e="p2p",ze=!1,be=!1,ce=!1,_t=null,St=null;const Sn=Ft(()=>p.value.filter(o=>o.kind==="file"&&o.pending)),Ct=Ft(()=>Sn.value.length>0);Dt(t,o=>{document.body.setAttribute("data-theme",o),localStorage.setItem("airshare-theme",o)},{immediate:!0}),Dt(n,o=>{if(o==="admin"&&K.value){localStorage.setItem(Fn,"admin");return}localStorage.removeItem(Fn)}),Dt([n,K],([o,c])=>{He&&(window.clearInterval(He),He=null),!(o!=="admin"||!c)&&(He=window.setInterval(()=>{is().catch(u=>{console.error(u)})},5e3))}),Lr(async()=>{_.value=ko(),await Nt(),n.value==="admin"&&K.value&&is().catch(o=>{console.error(o)}),je=window.setInterval(()=>{I()},Lu),vt=window.setInterval(()=>{h()},ju),yt=window.setInterval(()=>{v()},1e4)}),Xs(()=>{vt&&window.clearInterval(vt),je&&window.clearInterval(je),He&&window.clearInterval(He),yt&&window.clearInterval(yt),C(),bt(),pi(),A()});async function Nt(){try{await Cn(),await d(),await h()}catch(o){window.alert(`后端连接失败:${o.message}`)}}function tn(){t.value=t.value==="dark"?"light":"dark"}async function Cn(){try{it(await Du.config())}catch(o){console.error(o)}}function it(o){o&&(Q.value=o,T.value=Math.round((o.max_minio_fallback_size_bytes||0)/1024/1024),S.value=Math.max(0,Math.round((o.minio_capacity_bytes||0)/1024/1024/1024)))}function ni(o){r.value=o.replace(/\D/g,"").slice(0,4)}async function d(){const o=Co(),c=To(o),u=$o(),g=await Pn.register({device_id:o,name:c,type:u,network_group_key:window.location.hostname||"local"});localStorage.setItem(Nn,g.id),g.auth_token&&(localStorage.setItem(Is,g.auth_token),tr(g.id,g.auth_token)),R.value={id:g.id,name:g.name,type:g.type},await v(),di()}async function h(){if(R.value.id)try{const o=await ge.create({kind:"text",name:"text-message",content:value,sender_device_id:R.value.id,receiver_device_id:m.value.deviceId});try{await Pe(o,value)}catch(c){console.warn("realtime text send failed, fallback to relay",c),await sn(o,value)}p.value.push({id:Ue("text"),transferId:o.id,kind:"text",text:value,status:"已发送",tone:"success",copied:!1})}catch(o){window.alert(`发送文本失败:${o.message}`)}}async function v(){if(!R.value.id){f.value=[];return}try{const o=await Pn.listPendingDownloads(R.value.id);f.value=o.map(c=>({...c,download_path:c.download_path||`/api/transfers/${encodeURIComponent(c.transfer_id)}/fallback/download`,size_label:Bt(Number(c.size_bytes||0)),created_label:cs(c.created_at)}))}catch(o){if((o==null?void 0:o.status)===404){f.value=[];return}console.error(o)}}async function I(){if(R.value.id)try{await Pn.heartbeat(R.value.id)}catch(o){console.error(o)}}async function w(){if(!R.value.id){window.alert("当前设备尚未注册到后端");return}try{const o=await Dn.create(R.value.id);a.value=o.code,l.value=!0,$(o.code)}catch(o){window.alert(`创建房间失败:${o.message}`)}}async function x(){const o=a.value;C();try{l.value&&o!=="----"&&await Dn.cancel(o,R.value.id)}catch(c){console.error(c)}finally{l.value=!1,a.value="----"}}async function k(){if(!(r.value.length<4))try{const o=await Dn.join(r.value,R.value.id),c=$t(o.creator_device_id);r.value="",b({deviceId:o.creator_device_id,name:(c==null?void 0:c.name)||`房间 ${o.code} 创建者`,type:"房间配对成功"})}catch(o){window.alert(`加入房间失败:${o.message}`)}}function $(o){C(),Je=window.setInterval(async()=>{try{const c=await Dn.get(o);if(c.status==="joined"&&c.joiner_device_id){const u=$t(c.joiner_device_id);b({deviceId:c.joiner_device_id,name:(u==null?void 0:u.name)||`房间 ${o} 对端`,type:"房间配对成功"});return}(c.status==="expired"||c.status==="canceled")&&(C(),l.value=!1,a.value="----")}catch(c){console.error(c)}},Hu)}function C(){Je&&(window.clearInterval(Je),Je=null)}function b(o){const c=o.deviceId||o.id||"",u=o.connectionType||o.type||"点对点传输";C(),m.value.deviceId!==c&&(bt(),A()),m.value={name:o.name,type:o.connectionType||o.type||"点对点传输",deviceId:o.deviceId||o.id||""},l.value=!1,a.value="----",n.value="transfer",m.value.baseType=u,m.value.type=u,m.value.deviceId=c,Ke("正在建立实时通道"),rt(c,{initiate:!0})}function P(o,c=!1){const u=o.deviceId||o.id||"",g=o.connectionType||o.type||"点对点传输";m.value.deviceId===u&&n.value==="transfer"||(bt(),c||A()),m.value={name:o.name,type:o.connectionType||o.type||"点对点传输",deviceId:u},l.value=!1,a.value="----",n.value="transfer",m.value.baseType=g,m.value.type=g,m.value.deviceId=u,Ke("正在建立实时通道"),rt(u)}function E(){bt(),A(),m.value={name:"--",type:"等待连接",deviceId:""},n.value="main",m.value.baseType="等待连接",m.value.type="等待连接"}function A(){p.value.forEach(o=>nn(o)),p.value=[],he.clear()}async function N(o){const c=o.trim();if(c){if(!m.value.deviceId){window.alert("当前没有可用的接收端");return}try{const u=await ge.create({kind:"text",name:"text-message",content:c,sender_device_id:R.value.id,receiver_device_id:m.value.deviceId});ue("transfer.created",m.value.deviceId,{transfer_id:u.id,kind:"text",name:"text-message",content:c,sender_device_id:R.value.id,sender_name:R.value.name,sender_type:R.value.type,receiver_device_id:m.value.deviceId,final_status:"completed",current_channel:"p2p",transport_options:rs()}),await ge.updateStatus(u.id,{current_channel:"p2p",final_status:"completed"}),ue("transfer.updated",m.value.deviceId,{transfer_id:u.id,final_status:"completed",current_channel:"p2p"}),p.value.push({id:Ue("text"),transferId:u.id,kind:"text",text:c,status:"已发送",tone:"success",copied:!1})}catch(u){window.alert(`发送文本失败:${u.message}`)}}}function W(o){const c=o.filter(Boolean).map((u,g)=>({id:Ue(`file-${g}`),kind:"file",file:u,name:u.name,size:Bt(u.size),sizeBytes:u.size,status:"待发送",tone:"muted",progress:0,pending:!0,transferId:""}));c.length&&p.value.push(...c)}async function Y(o){const c=p.value.find(u=>u.id===o);if(!(!c||c.kind!=="file"||!c.pending)){if(!m.value.deviceId){window.alert("当前没有可用的接收端");return}c.pending=!1,c.status="创建传输中...",c.tone="primary";try{const u=await ge.create({kind:"file",name:c.name,size_bytes:c.sizeBytes,sender_device_id:R.value.id,receiver_device_id:m.value.deviceId});if(c.transferId=u.id,c.sizeBytes>Ku){await J(c,u);return}ue("transfer.created",m.value.deviceId,{transfer_id:u.id,kind:"file",name:c.name,size_bytes:c.sizeBytes,sender_device_id:R.value.id,sender_name:R.value.name,sender_type:R.value.type,receiver_device_id:m.value.deviceId,final_status:"connecting",current_channel:"p2p",transport_options:rs()});try{await rn(c,u)}catch(g){console.warn("realtime file send failed, fallback to relay",g),await si(c,u)}}catch(u){c.pending=!0,c.status=`发送失败:${u.message}`,c.tone="danger"}}}async function J(o,c){o.progress=0,o.status="上传准备中...";try{if(!c.fallback_allowed)throw new Error("当前文件过大,且未启用 MinIO 回退");await ge.presignFallback(o.transferId),ue("transfer.updated",m.value.deviceId,{transfer_id:o.transferId,final_status:"fallback_uploading",current_channel:"minio"}),o.status="上传中...";const u=await ge.uploadFallback(o.transferId,o.file,g=>{o.progress=Math.max(1,Math.min(g,99))});await ge.updateStatus(o.transferId,{current_channel:"minio",final_status:"completed"}),ue("transfer.updated",m.value.deviceId,{transfer_id:o.transferId,final_status:"completed",current_channel:"minio"}),ue("transfer.file",m.value.deviceId,{transfer_id:o.transferId,name:o.name,download_url:u.download_path||u.download_url}),o.progress=100,o.status="上传完成",o.tone="success"}catch(u){o.pending=!0,o.status=`上传失败:${u.message}`,o.tone="danger"}}async function we(){for(const o of Sn.value)await Y(o.id)}async function xe(o){const c=p.value.find(u=>u.id===o);if(c&&nn(c),p.value=p.value.filter(u=>u.id!==o),!(!(c!=null&&c.transferId)||c.tone==="success"))try{await ge.updateStatus(c.transferId,{final_status:"cancelled"}),ue("transfer.updated",m.value.deviceId,{transfer_id:c.transferId,final_status:"cancelled"})}catch(u){console.error(u)}}async function Fe(o){const c=p.value.find(u=>u.id===o);if(!(!c||c.kind!=="text"))try{await navigator.clipboard.writeText(c.text),c.copied=!0,window.setTimeout(()=>{const u=p.value.find(g=>g.id===o);u&&u.kind==="text"&&(u.copied=!1)},2e3)}catch{window.alert("复制失败")}}function Ne(o){const c=le.get(o);c&&(window.clearInterval(c),le.delete(o))}function Tt(o){return new Promise((c,u)=>{const g=new FileReader;g.onload=()=>c(String(g.result||"")),g.onerror=()=>u(new Error("Failed to read file")),g.readAsDataURL(o)})}function nn(o){if(Ne(o.id),o.ownedDownloadUrl&&o.downloadUrl)try{URL.revokeObjectURL(o.downloadUrl)}catch(c){console.error(c)}o.transferId&&he.delete(o.transferId)}function me(o,c,u=!1){if(o.ownedDownloadUrl&&o.downloadUrl&&o.downloadUrl!==c)try{URL.revokeObjectURL(o.downloadUrl)}catch(g){console.error(g)}o.downloadUrl=c,o.ownedDownloadUrl=u}async function Pe(o,c){const u=await li(m.value.deviceId);Tn(u,{type:"text",transfer_id:o.id,text:c,sender_device_id:R.value.id,sender_name:R.value.name,sender_type:R.value.type});const g=fi();await ge.updateStatus(o.id,{current_channel:g,final_status:"completed"})}async function sn(o,c){ue("transfer.created",m.value.deviceId,{transfer_id:o.id,kind:"text",name:"text-message",content:c,sender_device_id:R.value.id,sender_name:R.value.name,sender_type:R.value.type,receiver_device_id:m.value.deviceId,final_status:"completed",current_channel:"p2p"}),await ge.updateStatus(o.id,{current_channel:"p2p",final_status:"completed"})}async function rn(o,c){var O;const u=await li(m.value.deviceId);o.status="正在通过 WebRTC 发送...",o.progress=1,Tn(u,{type:"file-meta",transfer_id:c.id,name:o.name,mime_type:((O=o.file)==null?void 0:O.type)||"application/octet-stream",size_bytes:o.sizeBytes,sender_device_id:R.value.id,sender_name:R.value.name,sender_type:R.value.type});let g=0;for(;gir;)await bo(20)}function bo(o){return new Promise(c=>{window.setTimeout(c,o)})}function wo(o,c,u){return new Promise((g,M)=>{const O=window.setTimeout(()=>{M(new Error(u))},c);o.then(ne=>{window.clearTimeout(O),g(ne)}).catch(ne=>{window.clearTimeout(O),M(ne)})})}async function xo(){const o=window.prompt("管理员用户名","admin");if(o===null)return;const c=window.prompt("管理员密码");if(c!==null)try{const u=await jt.login(o.trim()||"admin",c);K.value=u.token,localStorage.setItem(xs,u.token),await is(),n.value="admin"}catch(u){window.alert(`管理员登录失败:${u.message}`)}}function Io(){n.value="main"}async function is(){if(K.value)try{const[o,c,u]=await Promise.all([jt.stats(K.value),jt.config(K.value),jt.recentTransfers(K.value)]);it(c),U.value=vi(o.stats||{},o.minio||{}),D.value=u.map(g=>jo(g))}catch(o){throw(o==null?void 0:o.status)===401&&(localStorage.removeItem(xs),localStorage.removeItem(Fn),K.value="",n.value="main"),o}}async function So(){if(!K.value||!Q.value){window.alert("当前没有可用的管理员会话");return}try{const o={...Q.value,max_minio_fallback_size_bytes:Math.max(0,T.value)*1024*1024,minio_capacity_bytes:Math.max(0,S.value)*1024*1024*1024},c=await jt.updateConfig(K.value,o);it(c);{const u=await jt.stats(K.value);U.value=vi(u.stats||{},u.minio||{})}window.alert("配置已保存")}catch(o){window.alert(`保存配置失败:${o.message}`)}}function Co(){let o=localStorage.getItem(Nn);return o||(o=typeof crypto<"u"&&crypto.randomUUID?crypto.randomUUID():`web-${Date.now()}-${Math.random().toString(36).slice(2,8)}`,localStorage.setItem(Nn,o)),o}function To(o){var u;let c=localStorage.getItem(sr);return c||(c=`${((u=navigator.userAgentData)==null?void 0:u.platform)||navigator.platform||"Web"} ${o.slice(0,4)}`,localStorage.setItem(sr,c)),c}function $o(){const o=`${navigator.userAgent} ${navigator.platform}`.toLowerCase();return o.includes("iphone")||o.includes("android")||o.includes("mobile")?"phone":o.includes("ipad")||o.includes("tablet")?"tablet":"desktop"}function Mo(o){return o==="phone"?"smartphone":o==="tablet"?"tablet_mac":"laptop_mac"}function Ut(o){return o==="phone"?"手机":o==="tablet"?"平板":"桌面端"}function ko(){return`${window.location.protocol==="https:"?"wss:":"ws:"}//${window.location.host}/ws`}function ii(o){const c=Array.isArray(o==null?void 0:o.turn_urls)?o.turn_urls.map(u=>String(u||"").trim()).filter(Boolean):[];return c.length?[{urls:c,username:(o==null?void 0:o.turn_username)||"",credential:(o==null?void 0:o.turn_password)||""}]:[]}function rs(){var o,c;return{ice_servers:ii(Q.value),p2p_connect_timeout_sec:((o=Q.value)==null?void 0:o.p2p_connect_timeout_sec)||15,turn_connect_timeout_sec:((c=Q.value)==null?void 0:c.turn_connect_timeout_sec)||20}}function ri(){return typeof RTCPeerConnection<"u"}function os(){_t=null,St=null}function Ro(){return _t||(_t=new Promise(o=>{St=o})),_t}function oi(o){St&&St(o),_t=Promise.resolve(o),St=null}function Ke(o=""){if(!m.value.deviceId)return;const c=m.value.baseType||m.value.type||"点对点传输";m.value={...m.value,type:o?`${c} · ${o}`:c}}async function rt(o,c={}){return!o||!ri()?null:((!H||G!==o)&&(bt(),Eo(o)),c.initiate&&H.signalingState==="stable"&&await Ao(o),H)}function Eo(o){G=o,_e="p2p",ze=!1,be=!1,ce=!1,Me.delete(o),os(),H=new RTCPeerConnection({iceServers:ii(Q.value)}),j=H.createDataChannel("filefast-control",{negotiated:!0,id:0,ordered:!0}),Oo(j),H.onicecandidate=({candidate:c})=>{if(c)try{ue("webrtc.candidate",o,{candidate:c})}catch(u){console.error(u)}},H.onconnectionstatechange=()=>{if(H){if(ls(),H.connectionState==="connected"){Ke(_e==="turn"?"TURN 中继已连接":"WebRTC 直连已连接");return}if(H.connectionState==="connecting"){Ke("实时通道连接中");return}if(H.connectionState==="failed"){Ke("实时通道连接失败");return}(H.connectionState==="disconnected"||H.connectionState==="closed")&&Ke("实时通道已断开")}},H.oniceconnectionstatechange=()=>{ls()}}async function Ao(o){if(H)try{ze=!0,await H.setLocalDescription(),ue("webrtc.description",o,{description:H.localDescription})}finally{ze=!1}}function Oo(o){j=o,o.bufferedAmountLowThreshold=ir/2,o.onopen=()=>{oi(o),Ke(_e==="turn"?"TURN 中继已连接":"WebRTC 直连已连接"),ls()},o.onclose=()=>{j===o&&(j=null,os(),Ke("实时通道已关闭"))},o.onerror=c=>{console.error(c)},o.onmessage=c=>{Po(c.data)},o.readyState==="open"&&oi(o)}function bt(){he.clear(),G&&Me.delete(G),j&&(j.onopen=null,j.onclose=null,j.onerror=null,j.onmessage=null,j.close(),j=null),H&&(H.onicecandidate=null,H.onconnectionstatechange=null,H.oniceconnectionstatechange=null,H.close(),H=null),G="",_e="p2p",ze=!1,be=!1,ce=!1,os()}async function li(o){if(!ri())throw new Error("当前浏览器不支持 WebRTC");if(await rt(o,{initiate:!0}),(j==null?void 0:j.readyState)==="open")return j;const c=await wo(Ro(),Wu,"WebRTC 连接超时");if(!c||c.readyState!=="open")throw new Error("实时通道未建立");return c}function Tn(o,c){if(!o||o.readyState!=="open")throw new Error("实时通道未就绪");o.send(JSON.stringify(c))}function Po(o){try{const c=JSON.parse(String(o||"{}"));if(c.type==="text"){Do(c);return}if(c.type==="file-meta"){Fo(c);return}if(c.type==="file-chunk"){No(c);return}c.type==="file-complete"&&Uo(c)}catch(c){console.error(c)}}function Do(o){var M;const c=o.sender_device_id||G,u={id:c,name:o.sender_name||((M=$t(c))==null?void 0:M.name)||`设备 ${Mt(c)}`,type:Ut(o.sender_type||"desktop"),connectionType:_e==="turn"?"TURN 中继已连接":"WebRTC 直连已连接"};P(u,!0);const g=p.value.find(O=>O.transferId===o.transfer_id);if(g&&g.kind==="text"){g.text=o.text||"",g.status="已接收",g.tone="success";return}p.value.push({id:Ue("incoming-text"),transferId:o.transfer_id,kind:"text",text:o.text||"",status:"已接收",tone:"success",copied:!1})}function Fo(o){var M;const c=o.sender_device_id||G,u={id:c,name:o.sender_name||((M=$t(c))==null?void 0:M.name)||`设备 ${Mt(c)}`,type:Ut(o.sender_type||"desktop"),connectionType:_e==="turn"?"TURN 中继已连接":"WebRTC 直连已连接"};P(u,!0),he.set(o.transfer_id,{name:o.name||"file",mimeType:o.mime_type||"application/octet-stream",sizeBytes:Number(o.size_bytes||0),receivedBytes:0,chunks:[]});let g=p.value.find(O=>O.transferId===o.transfer_id);g?(g.status="正在接收...",g.tone="primary",g.progress=0):(g={id:Ue("incoming-file"),transferId:o.transfer_id,kind:"file",name:o.name||"file",size:Bt(Number(o.size_bytes||0)),sizeBytes:Number(o.size_bytes||0),status:"正在接收...",tone:"primary",progress:0,pending:!1,downloadUrl:"",ownedDownloadUrl:!1},p.value.push(g))}function No(o){const c=he.get(o.transfer_id);if(!c)return;const u=yo(String(o.chunk_base64||""));c.receivedBytes+=Number(o.chunk_size||u.byteLength||0),c.chunks.push(u);const g=p.value.find(M=>M.transferId===o.transfer_id);if(g){const M=c.sizeBytes>0?c.receivedBytes/c.sizeBytes*100:0;g.progress=Math.max(1,Math.min(99,Math.round(M))),g.status="正在接收...",g.tone="primary"}}function Uo(o){const c=he.get(o.transfer_id);if(!c)return;const u=p.value.find(O=>O.transferId===o.transfer_id);if(!u){he.delete(o.transfer_id);return}const g=new Blob(c.chunks,{type:c.mimeType||"application/octet-stream"}),M=URL.createObjectURL(g);me(u,M,!0),u.progress=100,u.status="可保存",u.tone="success",he.delete(o.transfer_id)}function ai(o){return R.value.id.localeCompare(o)>0}function $n(o,c="等待实时数据"){const u=$t(o);return{id:o,deviceId:o,name:(u==null?void 0:u.name)||`设备 ${Mt(o)}`,type:Ut((u==null?void 0:u.type)||"desktop"),connectionType:c}}async function ci(o){const u=(o.payload||{}).description,g=o.device_id||"";if(!u||!g)return;P($n(g),!0);const M=await rt(g);if(!M)return;const O=ai(g),ne=!ze&&(M.signalingState==="stable"||ce),kt=u.type==="offer"&&!ne;be=!O&&kt,!be&&(ce=u.type==="answer",await M.setRemoteDescription(u),ce=!1,u.type==="offer"&&(await M.setLocalDescription(),ue("webrtc.description",g,{description:M.localDescription})))}async function ui(o){const c=o.payload||{},u=o.device_id||"";if(!c.candidate||!u)return;(n.value!=="transfer"||m.value.deviceId!==u)&&P($n(u),!0);const g=await rt(u);if(g)try{await g.addIceCandidate(c.candidate)}catch(M){be||console.error(M)}}async function ls(){if(!(!H||H.connectionState!=="connected"))try{const o=await H.getStats();let c=null;if(o.forEach(O=>{O.type==="transport"&&O.selectedCandidatePairId&&(c=o.get(O.selectedCandidatePairId)||c)}),c||o.forEach(O=>{O.type==="candidate-pair"&&O.state==="succeeded"&&(O.nominated||O.selected)&&(c=O)}),!c)return;const u=o.get(c.localCandidateId),g=o.get(c.remoteCandidateId),M=(u==null?void 0:u.candidateType)==="relay"||(g==null?void 0:g.candidateType)==="relay";_e=M?"turn":"p2p",(j==null?void 0:j.readyState)==="open"&&Ke(M?"TURN 中继已连接":"WebRTC 直连已连接")}catch(o){console.error(o)}}function fi(){return _e==="turn"?"turn":"p2p"}function di(){if(!R.value.id)return;const o=localStorage.getItem(Is)||"";o&&(pi(),de=new WebSocket(`${_.value}?deviceId=${encodeURIComponent(R.value.id)}&deviceToken=${encodeURIComponent(o)}`),de.addEventListener("message",c=>{Lo(c.data)}),de.addEventListener("close",()=>{de=null,Bo()}),de.addEventListener("error",()=>{de==null||de.close()}))}function pi(){if(st&&(window.clearTimeout(st),st=null),!de)return;const o=de;de=null,o.onclose=null,o.close()}function Bo(){st||!R.value.id||(st=window.setTimeout(()=>{st=null,di()},zu))}function ue(o,c,u){!de||de.readyState!==WebSocket.OPEN||!c||de.send(JSON.stringify({type:o,target_device_id:c,payload:u}))}function Lo(o){try{const c=JSON.parse(o);if(c.type==="presence.update"){h();return}if(c.type==="webrtc.description"){ci(c);return}if(c.type==="webrtc.candidate"){ui(c);return}if(c.type==="transfer.created"){hi(c);return}if(c.type==="transfer.updated"){mi(c);return}c.type==="transfer.file"&&gi(c)}catch(c){console.error(c)}}function hi(o){var O;const c=o.payload||{},u=o.device_id||c.sender_device_id||"",g={id:u,name:c.sender_name||((O=$t(u))==null?void 0:O.name)||`Device ${Mt(u)}`,type:Ut(c.sender_type||"desktop")};if(g.connectionType="等待实时数据",P(g,!0),!p.value.find(ne=>ne.transferId===c.transfer_id)){if(c.kind==="text"){p.value.push({id:Ue("incoming-text"),transferId:c.transfer_id,kind:"text",text:c.content||"",status:"已接收",tone:"success",copied:!1});return}p.value.push({id:Ue("incoming-file"),transferId:c.transfer_id,kind:"file",name:c.name||"file",size:Bt(Number(c.size_bytes||0)),sizeBytes:Number(c.size_bytes||0),status:"接收中...",tone:"primary",progress:35,pending:!1,downloadUrl:"",ownedDownloadUrl:!1})}}function mi(o){const c=o.payload||{},u=p.value.find(g=>g.transferId===c.transfer_id);if(u&&u.kind==="file"){if(c.final_status==="completed"){u.progress=100,u.status="已接收",u.tone="success",u.downloadUrl&&(u.status="可保存");return}c.final_status==="cancelled"&&(u.status="已取消",u.tone="danger")}}function gi(o){const c=o.payload||{};let u=p.value.find(g=>g.transferId===c.transfer_id);!u&&c.transfer_id&&(u={id:Ue("incoming-file"),transferId:c.transfer_id,kind:"file",name:c.name||"file",size:"",sizeBytes:0,status:"可保存",tone:"success",progress:100,pending:!1,downloadUrl:"",ownedDownloadUrl:!1},p.value.push(u)),!(!u||u.kind!=="file")&&(me(u,c.download_url||c.data_url||"",!1),u.status="可保存",u.progress=100,u.tone="success")}function $t(o){return i.value.find(c=>c.id===o)}function vi(o,c={}){return[{label:"在线设备",value:`${o.devices_online||0}`,tone:"blue"},{label:"待加入房间",value:`${o.rooms_waiting||0}`,tone:"cyan"},{label:"有效传输",value:`${o.transfers_total||0}`,tone:"default"},{label:"累计传输",value:`${o.transfers_cumulative||0}`,tone:"default"},{kind:"minio",label:"MinIO 剩余容量",value:as(c.remaining_bytes||0),tone:Number(c.usage_percent||0)>=85?"danger":Number(c.usage_percent||0)>=60?"cyan":"blue",percent:Math.max(0,100-Number(c.usage_percent||0)),detail:`已用 ${as(c.used_bytes||0)} / 总计 ${as(c.capacity_bytes||0)}`,kicker:`存档 ${c.object_count||0} 份`}]}function as(o){const c=Number(o||0);if(!c||c<=0)return"0 GB";const u=["B","KB","MB","GB","TB"],g=Math.min(Math.floor(Math.log(c)/Math.log(1024)),u.length-1),M=c/1024**g,O=g>=3?2:M>=10?1:2;return`${M.toFixed(O)} ${u[g]}`}function jo(o){const c=o.final_status==="completed",u=o.final_status==="failed"||o.final_status==="cancelled";return{time:cs(o.created_at),peer:`${Mt(o.sender_device_id)} -> ${Mt(o.receiver_device_id)}`,type:o.kind==="text"?"文本消息":`文件 ${o.name}`,size:Bt(Number(o.size_bytes||0)),status:c?`已完成 (${o.current_channel||"p2p"})`:u?`已结束 (${o.final_status})`:`进行中 (${o.final_status||"pending"})`,tone:c?"success":u?"danger":"primary"}}function Mt(o){return o?o.slice(0,8):"--"}function cs(o){if(!o)return"刚刚";const c=new Date(o),u=Date.now()-c.getTime();if(!Number.isFinite(u))return"刚刚";const g=Math.max(0,Math.floor(u/1e3));if(g<60)return`${g} 秒前`;const M=Math.floor(g/60);if(M<60)return`${M} 分钟前`;const O=Math.floor(M/60);return O<24?`${O} 小时前`:`${Math.floor(O/24)} 天前`}function Ue(o){return`${o}-${Date.now()}-${Math.random().toString(36).slice(2,8)}`}function Bt(o){if(!o||o<=0)return"0 B";const c=["B","KB","MB","GB","TB"],u=Math.min(Math.floor(Math.log(o)/Math.log(1024)),c.length-1),g=o/1024**u,M=g>=10||u===0?0:1;return`${g.toFixed(M)} ${c[u]}`}h=async function(){return R.value.id?Pn.listCandidates(R.value.id).then(c=>{i.value=c.map(u=>({...u,description:`${Ut(u.type)} · 最近活跃 ${cs(u.last_seen_at)}`,icon:Mo(u.type),connectionType:u.network_group_key&&u.network_group_key===window.location.hostname?"局域网直连优先":"跨网络实时传输"})),s.value=i.value.length===0}).catch(c=>{s.value=!1,console.error(c)}):Promise.resolve()},b=function(c){const u=c.deviceId||c.id||"",g=c.connectionType||c.type||"点对点传输";C(),m.value.deviceId!==u&&(bt(),A()),m.value={name:c.name,type:g,baseType:g,deviceId:u},l.value=!1,a.value="----",n.value="transfer",Ke("正在建立实时通道"),rt(u,{initiate:!0})},P=function(c,u=!1){const g=c.deviceId||c.id||"",M=c.connectionType||c.type||"点对点传输";m.value.deviceId===g&&n.value==="transfer"||(bt(),u||A()),m.value={name:c.name,type:M,baseType:M,deviceId:g},l.value=!1,a.value="----",n.value="transfer",g&&(Ke("正在建立实时通道"),rt(g))},E=function(){bt(),A(),m.value={name:"--",type:"等待连接",baseType:"等待连接",deviceId:""},n.value="main"},N=async function(c){const u=c.trim();if(u){if(!m.value.deviceId){window.alert("当前没有可用的接收端");return}try{const g=await ge.create({kind:"text",name:"text-message",content:u,sender_device_id:R.value.id,receiver_device_id:m.value.deviceId});try{await Pe(g,u)}catch(M){console.warn("realtime text send failed, fallback to relay",M),await sn(g,u)}p.value.push({id:Ue("text"),transferId:g.id,kind:"text",text:u,status:"已发送",tone:"success",copied:!1})}catch(g){window.alert(`发送文本失败:${g.message}`)}}};function yi(o,c){ue("transfer.file",m.value.deviceId,{transfer_id:o.transferId,name:o.name,download_url:c.download_path||c.download_url})}function _i(o,c,{onProgress:u}={}){if(!(o!=null&&o.file))return Promise.reject(new Error("未找到待上传文件"));if(!(c!=null&&c.fallback_allowed))return Promise.reject(new Error("MinIO 存档未启用"));const g=c.id;if($e.has(g))return $e.get(g);const M=(async()=>(await ge.presignFallback(g),ge.uploadFallback(g,o.file,O=>{typeof u=="function"&&u(O)})))().finally(()=>{$e.delete(g)});return $e.set(g,M),M}async function bi(o,c,u){await ge.updateStatus(c.id,{current_channel:"minio",final_status:"completed"}),ue("transfer.updated",m.value.deviceId,{transfer_id:c.id,final_status:"completed",current_channel:"minio"}),yi(o,u),o.progress=100,o.status="已上传到 MinIO,对方可直接领取",o.tone="success"}async function Ho(o,c){const u=Me.get(o);if(!(!(u!=null&&u.length)||!(c!=null&&c.remoteDescription))){Me.delete(o);for(const g of u)try{await c.addIceCandidate(g)}catch(M){console.error(M)}}}return J=async function(c,u){c.progress=Math.max(5,c.progress||0),c.status="正在切换到 MinIO...",c.tone="primary";try{ue("transfer.updated",m.value.deviceId,{transfer_id:c.transferId,final_status:"fallback_uploading",current_channel:"minio"});const g=await _i(c,u,{onProgress:M=>{c.progress=Math.max(5,Math.min(M,99))}});await bi(c,u,g)}catch(g){c.pending=!0,c.status=`上传失败:${g.message}`,c.tone="danger"}},ci=async function(c){const g=(c.payload||{}).description,M=c.device_id||"";if(!g||!M)return;P($n(M),!0);const O=await rt(M);if(!O)return;const ne=ai(M),kt=!ze&&(O.signalingState==="stable"||ce),zo=g.type==="offer"&&!kt;if(be=!ne&&zo,!be&&!(g.type==="answer"&&(O.signalingState!=="have-local-offer"||!O.localDescription))){try{ce=g.type==="answer",await O.setRemoteDescription(g),await Ho(M,O)}catch(us){console.error(us)}finally{ce=!1}if(g.type==="offer")try{await O.setLocalDescription(),ue("webrtc.description",M,{description:O.localDescription})}catch(us){console.error(us)}}},ui=async function(c){const u=c.payload||{},g=c.device_id||"",M=u.candidate;if(!M||!g)return;(n.value!=="transfer"||m.value.deviceId!==g)&&P($n(g),!0);const O=await rt(g);if(O){if(!O.remoteDescription){const ne=Me.get(g)||[];ne.push(M),Me.set(g,ne);return}try{await O.addIceCandidate(M)}catch(ne){be||console.error(ne)}}},Y=async function(c){const u=p.value.find(g=>g.id===c);if(!(!u||u.kind!=="file"||!u.pending)){if(!m.value.deviceId){window.alert("当前没有可用的接收端");return}u.pending=!1,u.status="创建传输中...",u.tone="primary";try{const g=await ge.create({kind:"file",name:u.name,size_bytes:u.sizeBytes,sender_device_id:R.value.id,receiver_device_id:m.value.deviceId});u.transferId=g.id;const M=g.fallback_allowed?_i(u,g).catch(O=>{throw console.warn("minio backup sync failed",O),O}):Promise.resolve(null);ue("transfer.created",m.value.deviceId,{transfer_id:g.id,kind:"file",name:u.name,size_bytes:u.sizeBytes,sender_device_id:R.value.id,sender_name:R.value.name,sender_type:R.value.type,receiver_device_id:m.value.deviceId,final_status:"connecting",current_channel:"p2p",transport_options:rs()});try{if(await rn(u,g),g.fallback_allowed){u.status="实时传输完成,正在同步云端备份...",u.tone="primary";try{const O=await M;O&&(yi(u,O),u.status="已发送,2 小时内可离线领取")}catch(O){u.status=`实时传输成功,但 MinIO 备份失败:${O.message}`,u.tone="danger";return}u.tone="success"}}catch(O){console.warn("realtime file send failed, fallback to minio",O);try{const ne=await M;if(ne){await bi(u,g,ne);return}}catch(ne){console.warn("minio backup sync failed after realtime failure",ne)}await si(u,g)}}catch(g){u.pending=!0,u.status=`发送失败:${g.message}`,u.tone="danger"}}},hi=function(c){var ne;const u=c.payload||{},g=c.device_id||u.sender_device_id||"",M={id:g,name:u.sender_name||((ne=$t(g))==null?void 0:ne.name)||`设备 ${Mt(g)}`,type:Ut(u.sender_type||"desktop"),connectionType:"等待实时数据"};if(P(M,!0),!p.value.find(kt=>kt.transferId===u.transfer_id)){if(u.kind==="text"){u.content&&p.value.push({id:Ue("incoming-text"),transferId:u.transfer_id,kind:"text",text:u.content||"",status:"已接收",tone:"success",copied:!1});return}p.value.push({id:Ue("incoming-file"),transferId:u.transfer_id,kind:"file",name:u.name||"file",size:Bt(Number(u.size_bytes||0)),sizeBytes:Number(u.size_bytes||0),status:"等待接收...",tone:"primary",progress:5,pending:!1,downloadUrl:"",ownedDownloadUrl:!1})}},mi=function(c){const u=c.payload||{},g=p.value.find(M=>M.transferId===u.transfer_id);if(g&&g.kind==="file"){if(u.final_status==="completed"){g.progress=100,g.status=g.downloadUrl?"可保存":"传输完成",g.tone="success";return}if(u.final_status==="cancelled"){g.status="已取消",g.tone="danger";return}u.final_status==="fallback_uploading"&&(g.status="发送端正在上传回退文件...",g.tone="primary")}},gi=function(c){const u=c.payload||{};let g=p.value.find(M=>M.transferId===u.transfer_id);!g&&u.transfer_id&&(g={id:Ue("incoming-file"),transferId:u.transfer_id,kind:"file",name:u.name||"file",size:"",sizeBytes:0,status:"可保存",tone:"success",progress:100,pending:!1,downloadUrl:"",ownedDownloadUrl:!1},p.value.push(g)),!(!g||g.kind!=="file")&&(me(g,u.download_url||u.data_url||"",!1),g.status="可保存",g.progress=100,g.tone="success")},(o,c)=>(L(),V("div",null,[y("div",Uu,[q(xc,{theme:t.value,onToggleTheme:tn},null,8,["theme"]),n.value==="main"?(L(),V("div",Bu,[q(su,{devices:i.value,"is-scanning":s.value,"local-device-name":R.value.name,onSelectDevice:b},null,8,["devices","is-scanning","local-device-name"]),q(mu,{"generated-code":a.value,"is-waiting":l.value,"pending-downloads":f.value,"room-code-input":r.value,onCancelRoom:x,onCreateRoom:w,onJoinRoom:k,onUpdateRoomCode:ni},null,8,["generated-code","is-waiting","pending-downloads","room-code-input"])])):We("",!0),n.value==="transfer"?(L(),mt($u,{key:1,"connection-type":m.value.type,"has-pending-items":Ct.value,items:p.value,"peer-name":m.value.name,onClose:E,onCopyItem:Fe,onFilesSelected:W,onRemoveItem:xe,onSendAllPending:we,onSendText:N,onStartUpload:Y},null,8,["connection-type","has-pending-items","items","peer-name"])):We("",!0),n.value==="admin"?(L(),mt(qc,{key:2,"file-limit":T.value,"minio-capacity":S.value,records:D.value,stats:U.value,onExit:Io,onSaveConfig:So,"onUpdate:fileLimit":c[0]||(c[0]=u=>T.value=u),"onUpdate:minioCapacity":c[1]||(c[1]=u=>S.value=u)},null,8,["file-limit","minio-capacity","records","stats"])):We("",!0)]),q(_c,{onRequestAdmin:xo})]))}};mc(qu).mount("#app"); diff --git a/frontend/dist/assets/index-DPzeYqvr.js b/frontend/dist/assets/index-DPzeYqvr.js new file mode 100644 index 0000000..39ef628 --- /dev/null +++ b/frontend/dist/assets/index-DPzeYqvr.js @@ -0,0 +1,17 @@ +(function(){const t=document.createElement("link").relList;if(t&&t.supports&&t.supports("modulepreload"))return;for(const r of document.querySelectorAll('link[rel="modulepreload"]'))s(r);new MutationObserver(r=>{for(const i of r)if(i.type==="childList")for(const l of i.addedNodes)l.tagName==="LINK"&&l.rel==="modulepreload"&&s(l)}).observe(document,{childList:!0,subtree:!0});function n(r){const i={};return r.integrity&&(i.integrity=r.integrity),r.referrerPolicy&&(i.referrerPolicy=r.referrerPolicy),r.crossOrigin==="use-credentials"?i.credentials="include":r.crossOrigin==="anonymous"?i.credentials="omit":i.credentials="same-origin",i}function s(r){if(r.ep)return;r.ep=!0;const i=n(r);fetch(r.href,i)}})();/** +* @vue/shared v3.5.30 +* (c) 2018-present Yuxi (Evan) You and Vue contributors +* @license MIT +**/function Ns(e){const t=Object.create(null);for(const n of e.split(","))t[n]=1;return n=>n in t}const re={},Vt=[],tt=()=>{},ci=()=>!1,Jn=e=>e.charCodeAt(0)===111&&e.charCodeAt(1)===110&&(e.charCodeAt(2)>122||e.charCodeAt(2)<97),Fs=e=>e.startsWith("onUpdate:"),Ce=Object.assign,Us=(e,t)=>{const n=e.indexOf(t);n>-1&&e.splice(n,1)},Zo=Object.prototype.hasOwnProperty,Z=(e,t)=>Zo.call(e,t),H=Array.isArray,qt=e=>bn(e)==="[object Map]",ui=e=>bn(e)==="[object Set]",Cr=e=>bn(e)==="[object Date]",K=e=>typeof e=="function",ae=e=>typeof e=="string",qe=e=>typeof e=="symbol",ne=e=>e!==null&&typeof e=="object",fi=e=>(ne(e)||K(e))&&K(e.then)&&K(e.catch),di=Object.prototype.toString,bn=e=>di.call(e),Qo=e=>bn(e).slice(8,-1),pi=e=>bn(e)==="[object Object]",Bs=e=>ae(e)&&e!=="NaN"&&e[0]!=="-"&&""+parseInt(e,10)===e,an=Ns(",key,ref,ref_for,ref_key,onVnodeBeforeMount,onVnodeMounted,onVnodeBeforeUpdate,onVnodeUpdated,onVnodeBeforeUnmount,onVnodeUnmounted"),Yn=e=>{const t=Object.create(null);return n=>t[n]||(t[n]=e(n))},el=/-\w/g,Ee=Yn(e=>e.replace(el,t=>t.slice(1).toUpperCase())),tl=/\B([A-Z])/g,It=Yn(e=>e.replace(tl,"-$1").toLowerCase()),Xn=Yn(e=>e.charAt(0).toUpperCase()+e.slice(1)),ds=Yn(e=>e?`on${Xn(e)}`:""),et=(e,t)=>!Object.is(e,t),Bn=(e,...t)=>{for(let n=0;n{Object.defineProperty(e,t,{configurable:!0,enumerable:!1,writable:s,value:n})},Ls=e=>{const t=parseFloat(e);return isNaN(t)?e:t};let Tr;const Zn=()=>Tr||(Tr=typeof globalThis<"u"?globalThis:typeof self<"u"?self:typeof window<"u"?window:typeof global<"u"?global:{});function Be(e){if(H(e)){const t={};for(let n=0;n{if(n){const s=n.split(sl);s.length>1&&(t[s[0].trim()]=s[1].trim())}}),t}function xt(e){let t="";if(ae(e))t=e;else if(H(e))for(let n=0;n!!(e&&e.__v_isRef===!0),te=e=>ae(e)?e:e==null?"":H(e)||ne(e)&&(e.toString===di||!K(e.toString))?gi(e)?te(e.value):JSON.stringify(e,vi,2):String(e),vi=(e,t)=>gi(t)?vi(e,t.value):qt(t)?{[`Map(${t.size})`]:[...t.entries()].reduce((n,[s,r],i)=>(n[ps(s,i)+" =>"]=r,n),{})}:ui(t)?{[`Set(${t.size})`]:[...t.values()].map(n=>ps(n))}:qe(t)?ps(t):ne(t)&&!H(t)&&!pi(t)?String(t):t,ps=(e,t="")=>{var n;return qe(e)?`Symbol(${(n=e.description)!=null?n:t})`:e};/** +* @vue/reactivity v3.5.30 +* (c) 2018-present Yuxi (Evan) You and Vue contributors +* @license MIT +**/let Oe;class cl{constructor(t=!1){this.detached=t,this._active=!0,this._on=0,this.effects=[],this.cleanups=[],this._isPaused=!1,this.__v_skip=!0,this.parent=Oe,!t&&Oe&&(this.index=(Oe.scopes||(Oe.scopes=[])).push(this)-1)}get active(){return this._active}pause(){if(this._active){this._isPaused=!0;let t,n;if(this.scopes)for(t=0,n=this.scopes.length;t0&&--this._on===0&&(Oe=this.prevScope,this.prevScope=void 0)}stop(t){if(this._active){this._active=!1;let n,s;for(n=0,s=this.effects.length;n0)return;if(un){let t=un;for(un=void 0;t;){const n=t.next;t.next=void 0,t.flags&=-9,t=n}}let e;for(;cn;){let t=cn;for(cn=void 0;t;){const n=t.next;if(t.next=void 0,t.flags&=-9,t.flags&1)try{t.trigger()}catch(s){e||(e=s)}t=n}}if(e)throw e}function wi(e){for(let t=e.deps;t;t=t.nextDep)t.version=-1,t.prevActiveLink=t.dep.activeLink,t.dep.activeLink=t}function xi(e){let t,n=e.depsTail,s=n;for(;s;){const r=s.prevDep;s.version===-1?(s===n&&(n=r),zs(s),fl(s)):t=s,s.dep.activeLink=s.prevActiveLink,s.prevActiveLink=void 0,s=r}e.deps=t,e.depsTail=n}function Cs(e){for(let t=e.deps;t;t=t.nextDep)if(t.dep.version!==t.version||t.dep.computed&&(Ii(t.dep.computed)||t.dep.version!==t.version))return!0;return!!e._dirty}function Ii(e){if(e.flags&4&&!(e.flags&16)||(e.flags&=-17,e.globalVersion===mn)||(e.globalVersion=mn,!e.isSSR&&e.flags&128&&(!e.deps&&!e._dirty||!Cs(e))))return;e.flags|=2;const t=e.dep,n=ie,s=Ve;ie=e,Ve=!0;try{wi(e);const r=e.fn(e._value);(t.version===0||et(r,e._value))&&(e.flags|=128,e._value=r,t.version++)}catch(r){throw t.version++,r}finally{ie=n,Ve=s,xi(e),e.flags&=-3}}function zs(e,t=!1){const{dep:n,prevSub:s,nextSub:r}=e;if(s&&(s.nextSub=r,e.prevSub=void 0),r&&(r.prevSub=s,e.nextSub=void 0),n.subs===e&&(n.subs=s,!s&&n.computed)){n.computed.flags&=-5;for(let i=n.computed.deps;i;i=i.nextDep)zs(i,!0)}!t&&!--n.sc&&n.map&&n.map.delete(n.key)}function fl(e){const{prevDep:t,nextDep:n}=e;t&&(t.nextDep=n,e.prevDep=void 0),n&&(n.prevDep=t,e.nextDep=void 0)}let Ve=!0;const Si=[];function ft(){Si.push(Ve),Ve=!1}function dt(){const e=Si.pop();Ve=e===void 0?!0:e}function kr(e){const{cleanup:t}=e;if(e.cleanup=void 0,t){const n=ie;ie=void 0;try{t()}finally{ie=n}}}let mn=0;class dl{constructor(t,n){this.sub=t,this.dep=n,this.version=n.version,this.nextDep=this.prevDep=this.nextSub=this.prevSub=this.prevActiveLink=void 0}}class Ws{constructor(t){this.computed=t,this.version=0,this.activeLink=void 0,this.subs=void 0,this.map=void 0,this.key=void 0,this.sc=0,this.__v_skip=!0}track(t){if(!ie||!Ve||ie===this.computed)return;let n=this.activeLink;if(n===void 0||n.sub!==ie)n=this.activeLink=new dl(ie,this),ie.deps?(n.prevDep=ie.depsTail,ie.depsTail.nextDep=n,ie.depsTail=n):ie.deps=ie.depsTail=n,Ci(n);else if(n.version===-1&&(n.version=this.version,n.nextDep)){const s=n.nextDep;s.prevDep=n.prevDep,n.prevDep&&(n.prevDep.nextDep=s),n.prevDep=ie.depsTail,n.nextDep=void 0,ie.depsTail.nextDep=n,ie.depsTail=n,ie.deps===n&&(ie.deps=s)}return n}trigger(t){this.version++,mn++,this.notify(t)}notify(t){Hs();try{for(let n=this.subs;n;n=n.prevSub)n.sub.notify()&&n.sub.dep.notify()}finally{Ks()}}}function Ci(e){if(e.dep.sc++,e.sub.flags&4){const t=e.dep.computed;if(t&&!e.dep.subs){t.flags|=20;for(let s=t.deps;s;s=s.nextDep)Ci(s)}const n=e.dep.subs;n!==e&&(e.prevSub=n,n&&(n.nextSub=e)),e.dep.subs=e}}const Ts=new WeakMap,Pt=Symbol(""),ks=Symbol(""),gn=Symbol("");function xe(e,t,n){if(Ve&&ie){let s=Ts.get(e);s||Ts.set(e,s=new Map);let r=s.get(n);r||(s.set(n,r=new Ws),r.map=s,r.key=n),r.track()}}function ct(e,t,n,s,r,i){const l=Ts.get(e);if(!l){mn++;return}const c=f=>{f&&f.trigger()};if(Hs(),t==="clear")l.forEach(c);else{const f=H(e),h=f&&Bs(n);if(f&&n==="length"){const p=Number(s);l.forEach((_,T)=>{(T==="length"||T===gn||!qe(T)&&T>=p)&&c(_)})}else switch((n!==void 0||l.has(void 0))&&c(l.get(n)),h&&c(l.get(gn)),t){case"add":f?h&&c(l.get("length")):(c(l.get(Pt)),qt(e)&&c(l.get(ks)));break;case"delete":f||(c(l.get(Pt)),qt(e)&&c(l.get(ks)));break;case"set":qt(e)&&c(l.get(Pt));break}}Ks()}function jt(e){const t=X(e);return t===e?t:(xe(t,"iterate",gn),je(e)?t:t.map(Ge))}function Qn(e){return xe(e=X(e),"iterate",gn),e}function Ze(e,t){return pt(e)?Zt(Dt(e)?Ge(t):t):Ge(t)}const pl={__proto__:null,[Symbol.iterator](){return ms(this,Symbol.iterator,e=>Ze(this,e))},concat(...e){return jt(this).concat(...e.map(t=>H(t)?jt(t):t))},entries(){return ms(this,"entries",e=>(e[1]=Ze(this,e[1]),e))},every(e,t){return ot(this,"every",e,t,void 0,arguments)},filter(e,t){return ot(this,"filter",e,t,n=>n.map(s=>Ze(this,s)),arguments)},find(e,t){return ot(this,"find",e,t,n=>Ze(this,n),arguments)},findIndex(e,t){return ot(this,"findIndex",e,t,void 0,arguments)},findLast(e,t){return ot(this,"findLast",e,t,n=>Ze(this,n),arguments)},findLastIndex(e,t){return ot(this,"findLastIndex",e,t,void 0,arguments)},forEach(e,t){return ot(this,"forEach",e,t,void 0,arguments)},includes(...e){return gs(this,"includes",e)},indexOf(...e){return gs(this,"indexOf",e)},join(e){return jt(this).join(e)},lastIndexOf(...e){return gs(this,"lastIndexOf",e)},map(e,t){return ot(this,"map",e,t,void 0,arguments)},pop(){return rn(this,"pop")},push(...e){return rn(this,"push",e)},reduce(e,...t){return $r(this,"reduce",e,t)},reduceRight(e,...t){return $r(this,"reduceRight",e,t)},shift(){return rn(this,"shift")},some(e,t){return ot(this,"some",e,t,void 0,arguments)},splice(...e){return rn(this,"splice",e)},toReversed(){return jt(this).toReversed()},toSorted(e){return jt(this).toSorted(e)},toSpliced(...e){return jt(this).toSpliced(...e)},unshift(...e){return rn(this,"unshift",e)},values(){return ms(this,"values",e=>Ze(this,e))}};function ms(e,t,n){const s=Qn(e),r=s[t]();return s!==e&&!je(e)&&(r._next=r.next,r.next=()=>{const i=r._next();return i.done||(i.value=n(i.value)),i}),r}const hl=Array.prototype;function ot(e,t,n,s,r,i){const l=Qn(e),c=l!==e&&!je(e),f=l[t];if(f!==hl[t]){const _=f.apply(e,i);return c?Ge(_):_}let h=n;l!==e&&(c?h=function(_,T){return n.call(this,Ze(e,_),T,e)}:n.length>2&&(h=function(_,T){return n.call(this,_,T,e)}));const p=f.call(l,h,s);return c&&r?r(p):p}function $r(e,t,n,s){const r=Qn(e),i=r!==e&&!je(e);let l=n,c=!1;r!==e&&(i?(c=s.length===0,l=function(h,p,_){return c&&(c=!1,h=Ze(e,h)),n.call(this,h,Ze(e,p),_,e)}):n.length>3&&(l=function(h,p,_){return n.call(this,h,p,_,e)}));const f=r[t](l,...s);return c?Ze(e,f):f}function gs(e,t,n){const s=X(e);xe(s,"iterate",gn);const r=s[t](...n);return(r===-1||r===!1)&&Js(n[0])?(n[0]=X(n[0]),s[t](...n)):r}function rn(e,t,n=[]){ft(),Hs();const s=X(e)[t].apply(e,n);return Ks(),dt(),s}const ml=Ns("__proto__,__v_isRef,__isVue"),Ti=new Set(Object.getOwnPropertyNames(Symbol).filter(e=>e!=="arguments"&&e!=="caller").map(e=>Symbol[e]).filter(qe));function gl(e){qe(e)||(e=String(e));const t=X(this);return xe(t,"has",e),t.hasOwnProperty(e)}class ki{constructor(t=!1,n=!1){this._isReadonly=t,this._isShallow=n}get(t,n,s){if(n==="__v_skip")return t.__v_skip;const r=this._isReadonly,i=this._isShallow;if(n==="__v_isReactive")return!r;if(n==="__v_isReadonly")return r;if(n==="__v_isShallow")return i;if(n==="__v_raw")return s===(r?i?Tl:Ei:i?Ri:Mi).get(t)||Object.getPrototypeOf(t)===Object.getPrototypeOf(s)?t:void 0;const l=H(t);if(!r){let f;if(l&&(f=pl[n]))return f;if(n==="hasOwnProperty")return gl}const c=Reflect.get(t,n,Se(t)?t:s);if((qe(n)?Ti.has(n):ml(n))||(r||xe(t,"get",n),i))return c;if(Se(c)){const f=l&&Bs(n)?c:c.value;return r&&ne(f)?Ms(f):f}return ne(c)?r?Ms(c):qs(c):c}}class $i extends ki{constructor(t=!1){super(!1,t)}set(t,n,s,r){let i=t[n];const l=H(t)&&Bs(n);if(!this._isShallow){const h=pt(i);if(!je(s)&&!pt(s)&&(i=X(i),s=X(s)),!l&&Se(i)&&!Se(s))return h||(i.value=s),!0}const c=l?Number(n)e,Mn=e=>Reflect.getPrototypeOf(e);function wl(e,t,n){return function(...s){const r=this.__v_raw,i=X(r),l=qt(i),c=e==="entries"||e===Symbol.iterator&&l,f=e==="keys"&&l,h=r[e](...s),p=n?$s:t?Zt:Ge;return!t&&xe(i,"iterate",f?ks:Pt),Ce(Object.create(h),{next(){const{value:_,done:T}=h.next();return T?{value:_,done:T}:{value:c?[p(_[0]),p(_[1])]:p(_),done:T}}})}}function Rn(e){return function(...t){return e==="delete"?!1:e==="clear"?void 0:this}}function xl(e,t){const n={get(r){const i=this.__v_raw,l=X(i),c=X(r);e||(et(r,c)&&xe(l,"get",r),xe(l,"get",c));const{has:f}=Mn(l),h=t?$s:e?Zt:Ge;if(f.call(l,r))return h(i.get(r));if(f.call(l,c))return h(i.get(c));i!==l&&i.get(r)},get size(){const r=this.__v_raw;return!e&&xe(X(r),"iterate",Pt),r.size},has(r){const i=this.__v_raw,l=X(i),c=X(r);return e||(et(r,c)&&xe(l,"has",r),xe(l,"has",c)),r===c?i.has(r):i.has(r)||i.has(c)},forEach(r,i){const l=this,c=l.__v_raw,f=X(c),h=t?$s:e?Zt:Ge;return!e&&xe(f,"iterate",Pt),c.forEach((p,_)=>r.call(i,h(p),h(_),l))}};return Ce(n,e?{add:Rn("add"),set:Rn("set"),delete:Rn("delete"),clear:Rn("clear")}:{add(r){const i=X(this),l=Mn(i),c=X(r),f=!t&&!je(r)&&!pt(r)?c:r;return l.has.call(i,f)||et(r,f)&&l.has.call(i,r)||et(c,f)&&l.has.call(i,c)||(i.add(f),ct(i,"add",f,f)),this},set(r,i){!t&&!je(i)&&!pt(i)&&(i=X(i));const l=X(this),{has:c,get:f}=Mn(l);let h=c.call(l,r);h||(r=X(r),h=c.call(l,r));const p=f.call(l,r);return l.set(r,i),h?et(i,p)&&ct(l,"set",r,i):ct(l,"add",r,i),this},delete(r){const i=X(this),{has:l,get:c}=Mn(i);let f=l.call(i,r);f||(r=X(r),f=l.call(i,r)),c&&c.call(i,r);const h=i.delete(r);return f&&ct(i,"delete",r,void 0),h},clear(){const r=X(this),i=r.size!==0,l=r.clear();return i&&ct(r,"clear",void 0,void 0),l}}),["keys","values","entries",Symbol.iterator].forEach(r=>{n[r]=wl(r,e,t)}),n}function Vs(e,t){const n=xl(e,t);return(s,r,i)=>r==="__v_isReactive"?!e:r==="__v_isReadonly"?e:r==="__v_raw"?s:Reflect.get(Z(n,r)&&r in s?n:s,r,i)}const Il={get:Vs(!1,!1)},Sl={get:Vs(!1,!0)},Cl={get:Vs(!0,!1)};const Mi=new WeakMap,Ri=new WeakMap,Ei=new WeakMap,Tl=new WeakMap;function kl(e){switch(e){case"Object":case"Array":return 1;case"Map":case"Set":case"WeakMap":case"WeakSet":return 2;default:return 0}}function $l(e){return e.__v_skip||!Object.isExtensible(e)?0:kl(Qo(e))}function qs(e){return pt(e)?e:Gs(e,!1,yl,Il,Mi)}function Ml(e){return Gs(e,!1,bl,Sl,Ri)}function Ms(e){return Gs(e,!0,_l,Cl,Ei)}function Gs(e,t,n,s,r){if(!ne(e)||e.__v_raw&&!(t&&e.__v_isReactive))return e;const i=$l(e);if(i===0)return e;const l=r.get(e);if(l)return l;const c=new Proxy(e,i===2?s:n);return r.set(e,c),c}function Dt(e){return pt(e)?Dt(e.__v_raw):!!(e&&e.__v_isReactive)}function pt(e){return!!(e&&e.__v_isReadonly)}function je(e){return!!(e&&e.__v_isShallow)}function Js(e){return e?!!e.__v_raw:!1}function X(e){const t=e&&e.__v_raw;return t?X(t):e}function Rl(e){return!Z(e,"__v_skip")&&Object.isExtensible(e)&&hi(e,"__v_skip",!0),e}const Ge=e=>ne(e)?qs(e):e,Zt=e=>ne(e)?Ms(e):e;function Se(e){return e?e.__v_isRef===!0:!1}function oe(e){return El(e,!1)}function El(e,t){return Se(e)?e:new Al(e,t)}class Al{constructor(t,n){this.dep=new Ws,this.__v_isRef=!0,this.__v_isShallow=!1,this._rawValue=n?t:X(t),this._value=n?t:Ge(t),this.__v_isShallow=n}get value(){return this.dep.track(),this._value}set value(t){const n=this._rawValue,s=this.__v_isShallow||je(t)||pt(t);t=s?t:X(t),et(t,n)&&(this._rawValue=t,this._value=s?t:Ge(t),this.dep.trigger())}}function Ol(e){return Se(e)?e.value:e}const Pl={get:(e,t,n)=>t==="__v_raw"?e:Ol(Reflect.get(e,t,n)),set:(e,t,n,s)=>{const r=e[t];return Se(r)&&!Se(n)?(r.value=n,!0):Reflect.set(e,t,n,s)}};function Ai(e){return Dt(e)?e:new Proxy(e,Pl)}class Dl{constructor(t,n,s){this.fn=t,this.setter=n,this._value=void 0,this.dep=new Ws(this),this.__v_isRef=!0,this.deps=void 0,this.depsTail=void 0,this.flags=16,this.globalVersion=mn-1,this.next=void 0,this.effect=this,this.__v_isReadonly=!n,this.isSSR=s}notify(){if(this.flags|=16,!(this.flags&8)&&ie!==this)return bi(this,!0),!0}get value(){const t=this.dep.track();return Ii(this),t&&(t.version=this.dep.version),this._value}set value(t){this.setter&&this.setter(t)}}function Nl(e,t,n=!1){let s,r;return K(e)?s=e:(s=e.get,r=e.set),new Dl(s,r,n)}const En={},Kn=new WeakMap;let Ot;function Fl(e,t=!1,n=Ot){if(n){let s=Kn.get(n);s||Kn.set(n,s=[]),s.push(e)}}function Ul(e,t,n=re){const{immediate:s,deep:r,once:i,scheduler:l,augmentJob:c,call:f}=n,h=F=>r?F:je(F)||r===!1||r===0?ut(F,1):ut(F);let p,_,T,S,U=!1,N=!1;if(Se(e)?(_=()=>e.value,U=je(e)):Dt(e)?(_=()=>h(e),U=!0):H(e)?(N=!0,U=e.some(F=>Dt(F)||je(F)),_=()=>e.map(F=>{if(Se(F))return F.value;if(Dt(F))return h(F);if(K(F))return f?f(F,2):F()})):K(e)?t?_=f?()=>f(e,2):e:_=()=>{if(T){ft();try{T()}finally{dt()}}const F=Ot;Ot=p;try{return f?f(e,3,[S]):e(S)}finally{Ot=F}}:_=tt,t&&r){const F=_,le=r===!0?1/0:r;_=()=>ut(F(),le)}const Y=ul(),z=()=>{p.stop(),Y&&Y.active&&Us(Y.effects,p)};if(i&&t){const F=t;t=(...le)=>{F(...le),z()}}let E=N?new Array(e.length).fill(En):En;const Q=F=>{if(!(!(p.flags&1)||!p.dirty&&!F))if(t){const le=p.run();if(r||U||(N?le.some((he,Te)=>et(he,E[Te])):et(le,E))){T&&T();const he=Ot;Ot=p;try{const Te=[le,E===En?void 0:N&&E[0]===En?[]:E,S];E=le,f?f(t,3,Te):t(...Te)}finally{Ot=he}}}else p.run()};return c&&c(Q),p=new yi(_),p.scheduler=l?()=>l(Q,!1):Q,S=F=>Fl(F,!1,p),T=p.onStop=()=>{const F=Kn.get(p);if(F){if(f)f(F,4);else for(const le of F)le();Kn.delete(p)}},t?s?Q(!0):E=p.run():l?l(Q.bind(null,!0),!0):p.run(),z.pause=p.pause.bind(p),z.resume=p.resume.bind(p),z.stop=z,z}function ut(e,t=1/0,n){if(t<=0||!ne(e)||e.__v_skip||(n=n||new Map,(n.get(e)||0)>=t))return e;if(n.set(e,t),t--,Se(e))ut(e.value,t,n);else if(H(e))for(let s=0;s{ut(s,t,n)});else if(pi(e)){for(const s in e)ut(e[s],t,n);for(const s of Object.getOwnPropertySymbols(e))Object.prototype.propertyIsEnumerable.call(e,s)&&ut(e[s],t,n)}return e}/** +* @vue/runtime-core v3.5.30 +* (c) 2018-present Yuxi (Evan) You and Vue contributors +* @license MIT +**/function wn(e,t,n,s){try{return s?e(...s):e()}catch(r){es(r,t,n)}}function nt(e,t,n,s){if(K(e)){const r=wn(e,t,n,s);return r&&fi(r)&&r.catch(i=>{es(i,t,n)}),r}if(H(e)){const r=[];for(let i=0;i>>1,r=Re[s],i=vn(r);i=vn(n)?Re.push(e):Re.splice(Ll(t),0,e),e.flags|=1,Di()}}function Di(){zn||(zn=Oi.then(Fi))}function jl(e){H(e)?Gt.push(...e):bt&&e.id===-1?bt.splice(Kt+1,0,e):e.flags&1||(Gt.push(e),e.flags|=1),Di()}function Mr(e,t,n=Xe+1){for(;nvn(n)-vn(s));if(Gt.length=0,bt){bt.push(...t);return}for(bt=t,Kt=0;Kte.id==null?e.flags&2?-1:1/0:e.id;function Fi(e){try{for(Xe=0;Xe{s._d&&jr(-1);const i=Wn(t);let l;try{l=e(...r)}finally{Wn(i),s._d&&jr(1)}return l};return s._n=!0,s._c=!0,s._d=!0,s}function Hl(e,t){if(ye===null)return e;const n=rs(ye),s=e.dirs||(e.dirs=[]);for(let r=0;r1)return n&&K(t)?t.call(s&&s.proxy):t}}const zl=Symbol.for("v-scx"),Wl=()=>Ln(zl);function Nt(e,t,n){return Bi(e,t,n)}function Bi(e,t,n=re){const{immediate:s,deep:r,flush:i,once:l}=n,c=Ce({},n),f=t&&s||!t&&i!=="post";let h;if(_n){if(i==="sync"){const S=Wl();h=S.__watcherHandles||(S.__watcherHandles=[])}else if(!f){const S=()=>{};return S.stop=tt,S.resume=tt,S.pause=tt,S}}const p=Ie;c.call=(S,U,N)=>nt(S,p,U,N);let _=!1;i==="post"?c.scheduler=S=>{Ae(S,p&&p.suspense)}:i!=="sync"&&(_=!0,c.scheduler=(S,U)=>{U?S():Ys(S)}),c.augmentJob=S=>{t&&(S.flags|=4),_&&(S.flags|=2,p&&(S.id=p.uid,S.i=p))};const T=Ul(e,t,c);return _n&&(h?h.push(T):f&&T()),T}function Vl(e,t,n){const s=this.proxy,r=ae(e)?e.includes(".")?Li(s,e):()=>s[e]:e.bind(s,s);let i;K(t)?i=t:(i=t.handler,n=t);const l=xn(this),c=Bi(r,i.bind(s),n);return l(),c}function Li(e,t){const n=t.split(".");return()=>{let s=e;for(let r=0;re.__isTeleport,Jl=Symbol("_leaveCb");function Xs(e,t){e.shapeFlag&6&&e.component?(e.transition=t,Xs(e.component.subTree,t)):e.shapeFlag&128?(e.ssContent.transition=t.clone(e.ssContent),e.ssFallback.transition=t.clone(e.ssFallback)):e.transition=t}function ji(e){e.ids=[e.ids[0]+e.ids[2]+++"-",0,0]}function Rr(e,t){let n;return!!((n=Object.getOwnPropertyDescriptor(e,t))&&!n.configurable)}const Vn=new WeakMap;function fn(e,t,n,s,r=!1){if(H(e)){e.forEach((N,Y)=>fn(N,t&&(H(t)?t[Y]:t),n,s,r));return}if(Yt(s)&&!r){s.shapeFlag&512&&s.type.__asyncResolved&&s.component.subTree.component&&fn(e,t,n,s.component.subTree);return}const i=s.shapeFlag&4?rs(s.component):s.el,l=r?null:i,{i:c,r:f}=e,h=t&&t.r,p=c.refs===re?c.refs={}:c.refs,_=c.setupState,T=X(_),S=_===re?ci:N=>Rr(p,N)?!1:Z(T,N),U=(N,Y)=>!(Y&&Rr(p,Y));if(h!=null&&h!==f){if(Er(t),ae(h))p[h]=null,S(h)&&(_[h]=null);else if(Se(h)){const N=t;U(h,N.k)&&(h.value=null),N.k&&(p[N.k]=null)}}if(K(f))wn(f,c,12,[l,p]);else{const N=ae(f),Y=Se(f);if(N||Y){const z=()=>{if(e.f){const E=N?S(f)?_[f]:p[f]:U()||!e.k?f.value:p[e.k];if(r)H(E)&&Us(E,i);else if(H(E))E.includes(i)||E.push(i);else if(N)p[f]=[i],S(f)&&(_[f]=p[f]);else{const Q=[i];U(f,e.k)&&(f.value=Q),e.k&&(p[e.k]=Q)}}else N?(p[f]=l,S(f)&&(_[f]=l)):Y&&(U(f,e.k)&&(f.value=l),e.k&&(p[e.k]=l))};if(l){const E=()=>{z(),Vn.delete(e)};E.id=-1,Vn.set(e,E),Ae(E,n)}else Er(e),z()}}}function Er(e){const t=Vn.get(e);t&&(t.flags|=8,Vn.delete(e))}Zn().requestIdleCallback;Zn().cancelIdleCallback;const Yt=e=>!!e.type.__asyncLoader,Hi=e=>e.type.__isKeepAlive;function Yl(e,t){Ki(e,"a",t)}function Xl(e,t){Ki(e,"da",t)}function Ki(e,t,n=Ie){const s=e.__wdc||(e.__wdc=()=>{let r=n;for(;r;){if(r.isDeactivated)return;r=r.parent}return e()});if(ts(t,s,n),n){let r=n.parent;for(;r&&r.parent;)Hi(r.parent.vnode)&&Zl(s,t,n,r),r=r.parent}}function Zl(e,t,n,s){const r=ts(t,e,s,!0);Wi(()=>{Us(s[t],r)},n)}function ts(e,t,n=Ie,s=!1){if(n){const r=n[e]||(n[e]=[]),i=t.__weh||(t.__weh=(...l)=>{ft();const c=xn(n),f=nt(t,n,e,l);return c(),dt(),f});return s?r.unshift(i):r.push(i),i}}const gt=e=>(t,n=Ie)=>{(!_n||e==="sp")&&ts(e,(...s)=>t(...s),n)},Ql=gt("bm"),zi=gt("m"),ea=gt("bu"),ta=gt("u"),Zs=gt("bum"),Wi=gt("um"),na=gt("sp"),sa=gt("rtg"),ra=gt("rtc");function ia(e,t=Ie){ts("ec",e,t)}const oa="components",Vi=Symbol.for("v-ndc");function la(e){return ae(e)?aa(oa,e,!1)||e:e||Vi}function aa(e,t,n=!0,s=!1){const r=ye||Ie;if(r){const i=r.type;{const c=qa(i,!1);if(c&&(c===t||c===Ee(t)||c===Xn(Ee(t))))return i}const l=Ar(r[e]||i[e],t)||Ar(r.appContext[e],t);return!l&&s?i:l}}function Ar(e,t){return e&&(e[t]||e[Ee(t)]||e[Xn(Ee(t))])}function Qt(e,t,n,s){let r;const i=n,l=H(e);if(l||ae(e)){const c=l&&Dt(e);let f=!1,h=!1;c&&(f=!je(e),h=pt(e),e=Qn(e)),r=new Array(e.length);for(let p=0,_=e.length;p<_;p++)r[p]=t(f?h?Zt(Ge(e[p])):Ge(e[p]):e[p],p,void 0,i)}else if(typeof e=="number"){r=new Array(e);for(let c=0;ct(c,f,void 0,i));else{const c=Object.keys(e);r=new Array(c.length);for(let f=0,h=c.length;f0;return j(),mt(fe,null,[q("slot",n,s)],h?-2:64)}let i=e[t];i&&i._c&&(i._d=!1),j();const l=i&&qi(i(n)),c=n.key||l&&l.key,f=mt(fe,{key:(c&&!qe(c)?c:`_${t}`)+(!l&&s?"_fb":"")},l||[],l&&e._===1?64:-2);return f.scopeId&&(f.slotScopeIds=[f.scopeId+"-s"]),i&&i._c&&(i._d=!0),f}function qi(e){return e.some(t=>tr(t)?!(t.type===ht||t.type===fe&&!qi(t.children)):!0)?e:null}const Rs=e=>e?mo(e)?rs(e):Rs(e.parent):null,dn=Ce(Object.create(null),{$:e=>e,$el:e=>e.vnode.el,$data:e=>e.data,$props:e=>e.props,$attrs:e=>e.attrs,$slots:e=>e.slots,$refs:e=>e.refs,$parent:e=>Rs(e.parent),$root:e=>Rs(e.root),$host:e=>e.ce,$emit:e=>e.emit,$options:e=>Ji(e),$forceUpdate:e=>e.f||(e.f=()=>{Ys(e.update)}),$nextTick:e=>e.n||(e.n=Pi.bind(e.proxy)),$watch:e=>Vl.bind(e)}),vs=(e,t)=>e!==re&&!e.__isScriptSetup&&Z(e,t),ua={get({_:e},t){if(t==="__v_skip")return!0;const{ctx:n,setupState:s,data:r,props:i,accessCache:l,type:c,appContext:f}=e;if(t[0]!=="$"){const T=l[t];if(T!==void 0)switch(T){case 1:return s[t];case 2:return r[t];case 4:return n[t];case 3:return i[t]}else{if(vs(s,t))return l[t]=1,s[t];if(r!==re&&Z(r,t))return l[t]=2,r[t];if(Z(i,t))return l[t]=3,i[t];if(n!==re&&Z(n,t))return l[t]=4,n[t];Es&&(l[t]=0)}}const h=dn[t];let p,_;if(h)return t==="$attrs"&&xe(e.attrs,"get",""),h(e);if((p=c.__cssModules)&&(p=p[t]))return p;if(n!==re&&Z(n,t))return l[t]=4,n[t];if(_=f.config.globalProperties,Z(_,t))return _[t]},set({_:e},t,n){const{data:s,setupState:r,ctx:i}=e;return vs(r,t)?(r[t]=n,!0):s!==re&&Z(s,t)?(s[t]=n,!0):Z(e.props,t)||t[0]==="$"&&t.slice(1)in e?!1:(i[t]=n,!0)},has({_:{data:e,setupState:t,accessCache:n,ctx:s,appContext:r,props:i,type:l}},c){let f;return!!(n[c]||e!==re&&c[0]!=="$"&&Z(e,c)||vs(t,c)||Z(i,c)||Z(s,c)||Z(dn,c)||Z(r.config.globalProperties,c)||(f=l.__cssModules)&&f[c])},defineProperty(e,t,n){return n.get!=null?e._.accessCache[t]=0:Z(n,"value")&&this.set(e,t,n.value,null),Reflect.defineProperty(e,t,n)}};function Or(e){return H(e)?e.reduce((t,n)=>(t[n]=null,t),{}):e}let Es=!0;function fa(e){const t=Ji(e),n=e.proxy,s=e.ctx;Es=!1,t.beforeCreate&&Pr(t.beforeCreate,e,"bc");const{data:r,computed:i,methods:l,watch:c,provide:f,inject:h,created:p,beforeMount:_,mounted:T,beforeUpdate:S,updated:U,activated:N,deactivated:Y,beforeDestroy:z,beforeUnmount:E,destroyed:Q,unmounted:F,render:le,renderTracked:he,renderTriggered:Te,errorCaptured:ke,serverPrefetch:vt,expose:He,inheritAttrs:Je,components:Ke,directives:yt,filters:de}=t;if(h&&da(h,s,null),l)for(const L in l){const G=l[L];K(G)&&(s[L]=G.bind(n))}if(r){const L=r.call(n,n);ne(L)&&(e.data=qs(L))}if(Es=!0,i)for(const L in i){const G=i[L],_e=K(G)?G.bind(n,n):K(G.get)?G.get.bind(n,n):tt,ze=!K(G)&&K(G.set)?G.set.bind(n):tt,be=wt({get:_e,set:ze});Object.defineProperty(s,L,{enumerable:!0,configurable:!0,get:()=>be.value,set:ue=>be.value=ue})}if(c)for(const L in c)Gi(c[L],s,n,L);if(f){const L=K(f)?f.call(n):f;Reflect.ownKeys(L).forEach(G=>{Kl(G,L[G])})}p&&Pr(p,e,"c");function B(L,G){H(G)?G.forEach(_e=>L(_e.bind(n))):G&&L(G.bind(n))}if(B(Ql,_),B(zi,T),B(ea,S),B(ta,U),B(Yl,N),B(Xl,Y),B(ia,ke),B(ra,he),B(sa,Te),B(Zs,E),B(Wi,F),B(na,vt),H(He))if(He.length){const L=e.exposed||(e.exposed={});He.forEach(G=>{Object.defineProperty(L,G,{get:()=>n[G],set:_e=>n[G]=_e,enumerable:!0})})}else e.exposed||(e.exposed={});le&&e.render===tt&&(e.render=le),Je!=null&&(e.inheritAttrs=Je),Ke&&(e.components=Ke),yt&&(e.directives=yt),vt&&ji(e)}function da(e,t,n=tt){H(e)&&(e=As(e));for(const s in e){const r=e[s];let i;ne(r)?"default"in r?i=Ln(r.from||s,r.default,!0):i=Ln(r.from||s):i=Ln(r),Se(i)?Object.defineProperty(t,s,{enumerable:!0,configurable:!0,get:()=>i.value,set:l=>i.value=l}):t[s]=i}}function Pr(e,t,n){nt(H(e)?e.map(s=>s.bind(t.proxy)):e.bind(t.proxy),t,n)}function Gi(e,t,n,s){let r=s.includes(".")?Li(n,s):()=>n[s];if(ae(e)){const i=t[e];K(i)&&Nt(r,i)}else if(K(e))Nt(r,e.bind(n));else if(ne(e))if(H(e))e.forEach(i=>Gi(i,t,n,s));else{const i=K(e.handler)?e.handler.bind(n):t[e.handler];K(i)&&Nt(r,i,e)}}function Ji(e){const t=e.type,{mixins:n,extends:s}=t,{mixins:r,optionsCache:i,config:{optionMergeStrategies:l}}=e.appContext,c=i.get(t);let f;return c?f=c:!r.length&&!n&&!s?f=t:(f={},r.length&&r.forEach(h=>qn(f,h,l,!0)),qn(f,t,l)),ne(t)&&i.set(t,f),f}function qn(e,t,n,s=!1){const{mixins:r,extends:i}=t;i&&qn(e,i,n,!0),r&&r.forEach(l=>qn(e,l,n,!0));for(const l in t)if(!(s&&l==="expose")){const c=pa[l]||n&&n[l];e[l]=c?c(e[l],t[l]):t[l]}return e}const pa={data:Dr,props:Nr,emits:Nr,methods:ln,computed:ln,beforeCreate:Me,created:Me,beforeMount:Me,mounted:Me,beforeUpdate:Me,updated:Me,beforeDestroy:Me,beforeUnmount:Me,destroyed:Me,unmounted:Me,activated:Me,deactivated:Me,errorCaptured:Me,serverPrefetch:Me,components:ln,directives:ln,watch:ma,provide:Dr,inject:ha};function Dr(e,t){return t?e?function(){return Ce(K(e)?e.call(this,this):e,K(t)?t.call(this,this):t)}:t:e}function ha(e,t){return ln(As(e),As(t))}function As(e){if(H(e)){const t={};for(let n=0;nt==="modelValue"||t==="model-value"?e.modelModifiers:e[`${t}Modifiers`]||e[`${Ee(t)}Modifiers`]||e[`${It(t)}Modifiers`];function _a(e,t,...n){if(e.isUnmounted)return;const s=e.vnode.props||re;let r=n;const i=t.startsWith("update:"),l=i&&ya(s,t.slice(7));l&&(l.trim&&(r=n.map(p=>ae(p)?p.trim():p)),l.number&&(r=n.map(Ls)));let c,f=s[c=ds(t)]||s[c=ds(Ee(t))];!f&&i&&(f=s[c=ds(It(t))]),f&&nt(f,e,6,r);const h=s[c+"Once"];if(h){if(!e.emitted)e.emitted={};else if(e.emitted[c])return;e.emitted[c]=!0,nt(h,e,6,r)}}const ba=new WeakMap;function Xi(e,t,n=!1){const s=n?ba:t.emitsCache,r=s.get(e);if(r!==void 0)return r;const i=e.emits;let l={},c=!1;if(!K(e)){const f=h=>{const p=Xi(h,t,!0);p&&(c=!0,Ce(l,p))};!n&&t.mixins.length&&t.mixins.forEach(f),e.extends&&f(e.extends),e.mixins&&e.mixins.forEach(f)}return!i&&!c?(ne(e)&&s.set(e,null),null):(H(i)?i.forEach(f=>l[f]=null):Ce(l,i),ne(e)&&s.set(e,l),l)}function ns(e,t){return!e||!Jn(t)?!1:(t=t.slice(2).replace(/Once$/,""),Z(e,t[0].toLowerCase()+t.slice(1))||Z(e,It(t))||Z(e,t))}function Fr(e){const{type:t,vnode:n,proxy:s,withProxy:r,propsOptions:[i],slots:l,attrs:c,emit:f,render:h,renderCache:p,props:_,data:T,setupState:S,ctx:U,inheritAttrs:N}=e,Y=Wn(e);let z,E;try{if(n.shapeFlag&4){const F=r||s,le=F;z=Qe(h.call(le,F,p,_,S,T,U)),E=c}else{const F=t;z=Qe(F.length>1?F(_,{attrs:c,slots:l,emit:f}):F(_,null)),E=t.props?c:wa(c)}}catch(F){pn.length=0,es(F,e,1),z=q(ht)}let Q=z;if(E&&N!==!1){const F=Object.keys(E),{shapeFlag:le}=Q;F.length&&le&7&&(i&&F.some(Fs)&&(E=xa(E,i)),Q=en(Q,E,!1,!0))}return n.dirs&&(Q=en(Q,null,!1,!0),Q.dirs=Q.dirs?Q.dirs.concat(n.dirs):n.dirs),n.transition&&Xs(Q,n.transition),z=Q,Wn(Y),z}const wa=e=>{let t;for(const n in e)(n==="class"||n==="style"||Jn(n))&&((t||(t={}))[n]=e[n]);return t},xa=(e,t)=>{const n={};for(const s in e)(!Fs(s)||!(s.slice(9)in t))&&(n[s]=e[s]);return n};function Ia(e,t,n){const{props:s,children:r,component:i}=e,{props:l,children:c,patchFlag:f}=t,h=i.emitsOptions;if(t.dirs||t.transition)return!0;if(n&&f>=0){if(f&1024)return!0;if(f&16)return s?Ur(s,l,h):!!l;if(f&8){const p=t.dynamicProps;for(let _=0;_Object.create(Qi),to=e=>Object.getPrototypeOf(e)===Qi;function Ca(e,t,n,s=!1){const r={},i=eo();e.propsDefaults=Object.create(null),no(e,t,r,i);for(const l in e.propsOptions[0])l in r||(r[l]=void 0);n?e.props=s?r:Ml(r):e.type.props?e.props=r:e.props=i,e.attrs=i}function Ta(e,t,n,s){const{props:r,attrs:i,vnode:{patchFlag:l}}=e,c=X(r),[f]=e.propsOptions;let h=!1;if((s||l>0)&&!(l&16)){if(l&8){const p=e.vnode.dynamicProps;for(let _=0;_{f=!0;const[T,S]=so(_,t,!0);Ce(l,T),S&&c.push(...S)};!n&&t.mixins.length&&t.mixins.forEach(p),e.extends&&p(e.extends),e.mixins&&e.mixins.forEach(p)}if(!i&&!f)return ne(e)&&s.set(e,Vt),Vt;if(H(i))for(let p=0;pe==="_"||e==="_ctx"||e==="$stable",er=e=>H(e)?e.map(Qe):[Qe(e)],$a=(e,t,n)=>{if(t._n)return t;const s=Jt((...r)=>er(t(...r)),n);return s._c=!1,s},ro=(e,t,n)=>{const s=e._ctx;for(const r in e){if(Qs(r))continue;const i=e[r];if(K(i))t[r]=$a(r,i,s);else if(i!=null){const l=er(i);t[r]=()=>l}}},io=(e,t)=>{const n=er(t);e.slots.default=()=>n},oo=(e,t,n)=>{for(const s in t)(n||!Qs(s))&&(e[s]=t[s])},Ma=(e,t,n)=>{const s=e.slots=eo();if(e.vnode.shapeFlag&32){const r=t._;r?(oo(s,t,n),n&&hi(s,"_",r,!0)):ro(t,s)}else t&&io(e,t)},Ra=(e,t,n)=>{const{vnode:s,slots:r}=e;let i=!0,l=re;if(s.shapeFlag&32){const c=t._;c?n&&c===1?i=!1:oo(r,t,n):(i=!t.$stable,ro(t,r)),l=t}else t&&(io(e,t),l={default:1});if(i)for(const c in r)!Qs(c)&&l[c]==null&&delete r[c]},Ae=Da;function Ea(e){return Aa(e)}function Aa(e,t){const n=Zn();n.__VUE__=!0;const{insert:s,remove:r,patchProp:i,createElement:l,createText:c,createComment:f,setText:h,setElementText:p,parentNode:_,nextSibling:T,setScopeId:S=tt,insertStaticContent:U}=e,N=(d,m,v,I=null,w=null,x=null,R=void 0,$=null,C=!!m.dynamicChildren)=>{if(d===m)return;d&&!on(d,m)&&(I=Ft(d),ue(d,w,x,!0),d=null),m.patchFlag===-2&&(C=!1,m.dynamicChildren=null);const{type:b,ref:P,shapeFlag:M}=m;switch(b){case ss:Y(d,m,v,I);break;case ht:z(d,m,v,I);break;case _s:d==null&&E(m,v,I,R);break;case fe:Ke(d,m,v,I,w,x,R,$,C);break;default:M&1?le(d,m,v,I,w,x,R,$,C):M&6?yt(d,m,v,I,w,x,R,$,C):(M&64||M&128)&&b.process(d,m,v,I,w,x,R,$,C,Tt)}P!=null&&w?fn(P,d&&d.ref,x,m||d,!m):P==null&&d&&d.ref!=null&&fn(d.ref,null,x,d,!0)},Y=(d,m,v,I)=>{if(d==null)s(m.el=c(m.children),v,I);else{const w=m.el=d.el;m.children!==d.children&&h(w,m.children)}},z=(d,m,v,I)=>{d==null?s(m.el=f(m.children||""),v,I):m.el=d.el},E=(d,m,v,I)=>{[d.el,d.anchor]=U(d.children,m,v,I,d.el,d.anchor)},Q=({el:d,anchor:m},v,I)=>{let w;for(;d&&d!==m;)w=T(d),s(d,v,I),d=w;s(m,v,I)},F=({el:d,anchor:m})=>{let v;for(;d&&d!==m;)v=T(d),r(d),d=v;r(m)},le=(d,m,v,I,w,x,R,$,C)=>{if(m.type==="svg"?R="svg":m.type==="math"&&(R="mathml"),d==null)he(m,v,I,w,x,R,$,C);else{const b=d.el&&d.el._isVueCE?d.el:null;try{b&&b._beginPatch(),vt(d,m,w,x,R,$,C)}finally{b&&b._endPatch()}}},he=(d,m,v,I,w,x,R,$)=>{let C,b;const{props:P,shapeFlag:M,transition:O,dirs:D}=d;if(C=d.el=l(d.type,x,P&&P.is,P),M&8?p(C,d.children):M&16&&ke(d.children,C,null,I,w,ys(d,x),R,$),D&&Et(d,null,I,"created"),Te(C,d,d.scopeId,R,I),P){for(const ee in P)ee!=="value"&&!an(ee)&&i(C,ee,null,P[ee],x,I);"value"in P&&i(C,"value",null,P.value,x),(b=P.onVnodeBeforeMount)&&Ye(b,I,d)}D&&Et(d,null,I,"beforeMount");const W=Oa(w,O);W&&O.beforeEnter(C),s(C,m,v),((b=P&&P.onVnodeMounted)||W||D)&&Ae(()=>{b&&Ye(b,I,d),W&&O.enter(C),D&&Et(d,null,I,"mounted")},w)},Te=(d,m,v,I,w)=>{if(v&&S(d,v),I)for(let x=0;x{for(let b=C;b{const $=m.el=d.el;let{patchFlag:C,dynamicChildren:b,dirs:P}=m;C|=d.patchFlag&16;const M=d.props||re,O=m.props||re;let D;if(v&&At(v,!1),(D=O.onVnodeBeforeUpdate)&&Ye(D,v,m,d),P&&Et(m,d,v,"beforeUpdate"),v&&At(v,!0),(M.innerHTML&&O.innerHTML==null||M.textContent&&O.textContent==null)&&p($,""),b?He(d.dynamicChildren,b,$,v,I,ys(m,w),x):R||G(d,m,$,null,v,I,ys(m,w),x,!1),C>0){if(C&16)Je($,M,O,v,w);else if(C&2&&M.class!==O.class&&i($,"class",null,O.class,w),C&4&&i($,"style",M.style,O.style,w),C&8){const W=m.dynamicProps;for(let ee=0;ee{D&&Ye(D,v,m,d),P&&Et(m,d,v,"updated")},I)},He=(d,m,v,I,w,x,R)=>{for(let $=0;${if(m!==v){if(m!==re)for(const x in m)!an(x)&&!(x in v)&&i(d,x,m[x],null,w,I);for(const x in v){if(an(x))continue;const R=v[x],$=m[x];R!==$&&x!=="value"&&i(d,x,$,R,w,I)}"value"in v&&i(d,"value",m.value,v.value,w)}},Ke=(d,m,v,I,w,x,R,$,C)=>{const b=m.el=d?d.el:c(""),P=m.anchor=d?d.anchor:c("");let{patchFlag:M,dynamicChildren:O,slotScopeIds:D}=m;D&&($=$?$.concat(D):D),d==null?(s(b,v,I),s(P,v,I),ke(m.children||[],v,P,w,x,R,$,C)):M>0&&M&64&&O&&d.dynamicChildren&&d.dynamicChildren.length===O.length?(He(d.dynamicChildren,O,v,w,x,R,$),(m.key!=null||w&&m===w.subTree)&&lo(d,m,!0)):G(d,m,v,P,w,x,R,$,C)},yt=(d,m,v,I,w,x,R,$,C)=>{m.slotScopeIds=$,d==null?m.shapeFlag&512?w.ctx.activate(m,v,I,R,C):de(m,v,I,w,x,R,C):st(d,m,C)},de=(d,m,v,I,w,x,R)=>{const $=d.component=ja(d,I,w);if(Hi(d)&&($.ctx.renderer=Tt),Ka($,!1,R),$.asyncDep){if(w&&w.registerDep($,B,R),!d.el){const C=$.subTree=q(ht);z(null,C,m,v),d.placeholder=C.el}}else B($,d,m,v,w,x,R)},st=(d,m,v)=>{const I=m.component=d.component;if(Ia(d,m,v))if(I.asyncDep&&!I.asyncResolved){L(I,m,v);return}else I.next=m,I.update();else m.el=d.el,I.vnode=m},B=(d,m,v,I,w,x,R)=>{const $=()=>{if(d.isMounted){let{next:M,bu:O,u:D,parent:W,vnode:ee}=d;{const Fe=ao(d);if(Fe){M&&(M.el=ee.el,L(d,M,R)),Fe.asyncDep.then(()=>{Ae(()=>{d.isUnmounted||b()},w)});return}}let J=M,me;At(d,!1),M?(M.el=ee.el,L(d,M,R)):M=ee,O&&Bn(O),(me=M.props&&M.props.onVnodeBeforeUpdate)&&Ye(me,W,M,ee),At(d,!0);const we=Fr(d),Ne=d.subTree;d.subTree=we,N(Ne,we,_(Ne.el),Ft(Ne),d,w,x),M.el=we.el,J===null&&Sa(d,we.el),D&&Ae(D,w),(me=M.props&&M.props.onVnodeUpdated)&&Ae(()=>Ye(me,W,M,ee),w)}else{let M;const{el:O,props:D}=m,{bm:W,m:ee,parent:J,root:me,type:we}=d,Ne=Yt(m);At(d,!1),W&&Bn(W),!Ne&&(M=D&&D.onVnodeBeforeMount)&&Ye(M,J,m),At(d,!0);{me.ce&&me.ce._hasShadowRoot()&&me.ce._injectChildStyle(we,d.parent?d.parent.type:void 0);const Fe=d.subTree=Fr(d);N(null,Fe,v,I,d,w,x),m.el=Fe.el}if(ee&&Ae(ee,w),!Ne&&(M=D&&D.onVnodeMounted)){const Fe=m;Ae(()=>Ye(M,J,Fe),w)}(m.shapeFlag&256||J&&Yt(J.vnode)&&J.vnode.shapeFlag&256)&&d.a&&Ae(d.a,w),d.isMounted=!0,m=v=I=null}};d.scope.on();const C=d.effect=new yi($);d.scope.off();const b=d.update=C.run.bind(C),P=d.job=C.runIfDirty.bind(C);P.i=d,P.id=d.uid,C.scheduler=()=>Ys(P),At(d,!0),b()},L=(d,m,v)=>{m.component=d;const I=d.vnode.props;d.vnode=m,d.next=null,Ta(d,m.props,I,v),Ra(d,m.children,v),ft(),Mr(d),dt()},G=(d,m,v,I,w,x,R,$,C=!1)=>{const b=d&&d.children,P=d?d.shapeFlag:0,M=m.children,{patchFlag:O,shapeFlag:D}=m;if(O>0){if(O&128){ze(b,M,v,I,w,x,R,$,C);return}else if(O&256){_e(b,M,v,I,w,x,R,$,C);return}}D&8?(P&16&&Ct(b,w,x),M!==b&&p(v,M)):P&16?D&16?ze(b,M,v,I,w,x,R,$,C):Ct(b,w,x,!0):(P&8&&p(v,""),D&16&&ke(M,v,I,w,x,R,$,C))},_e=(d,m,v,I,w,x,R,$,C)=>{d=d||Vt,m=m||Vt;const b=d.length,P=m.length,M=Math.min(b,P);let O;for(O=0;OP?Ct(d,w,x,!0,!1,M):ke(m,v,I,w,x,R,$,C,M)},ze=(d,m,v,I,w,x,R,$,C)=>{let b=0;const P=m.length;let M=d.length-1,O=P-1;for(;b<=M&&b<=O;){const D=d[b],W=m[b]=C?at(m[b]):Qe(m[b]);if(on(D,W))N(D,W,v,null,w,x,R,$,C);else break;b++}for(;b<=M&&b<=O;){const D=d[M],W=m[O]=C?at(m[O]):Qe(m[O]);if(on(D,W))N(D,W,v,null,w,x,R,$,C);else break;M--,O--}if(b>M){if(b<=O){const D=O+1,W=DO)for(;b<=M;)ue(d[b],w,x,!0),b++;else{const D=b,W=b,ee=new Map;for(b=W;b<=O;b++){const $e=m[b]=C?at(m[b]):Qe(m[b]);$e.key!=null&&ee.set($e.key,b)}let J,me=0;const we=O-W+1;let Ne=!1,Fe=0;const kt=new Array(we);for(b=0;b=we){ue($e,w,x,!0);continue}let Pe;if($e.key!=null)Pe=ee.get($e.key);else for(J=W;J<=O;J++)if(kt[J-W]===0&&on($e,m[J])){Pe=J;break}Pe===void 0?ue($e,w,x,!0):(kt[Pe-W]=b+1,Pe>=Fe?Fe=Pe:Ne=!0,N($e,m[Pe],v,null,w,x,R,$,C),me++)}const Tn=Ne?Pa(kt):Vt;for(J=Tn.length-1,b=we-1;b>=0;b--){const $e=W+b,Pe=m[$e],Ut=m[$e+1],sn=$e+1{const{el:x,type:R,transition:$,children:C,shapeFlag:b}=d;if(b&6){be(d.component.subTree,m,v,I);return}if(b&128){d.suspense.move(m,v,I);return}if(b&64){R.move(d,m,v,Tt);return}if(R===fe){s(x,m,v);for(let M=0;M$.enter(x),w);else{const{leave:M,delayLeave:O,afterLeave:D}=$,W=()=>{d.ctx.isUnmounted?r(x):s(x,m,v)},ee=()=>{x._isLeaving&&x[Jl](!0),M(x,()=>{W(),D&&D()})};O?O(x,W,ee):ee()}else s(x,m,v)},ue=(d,m,v,I=!1,w=!1)=>{const{type:x,props:R,ref:$,children:C,dynamicChildren:b,shapeFlag:P,patchFlag:M,dirs:O,cacheIndex:D}=d;if(M===-2&&(w=!1),$!=null&&(ft(),fn($,null,v,d,!0),dt()),D!=null&&(m.renderCache[D]=void 0),P&256){m.ctx.deactivate(d);return}const W=P&1&&O,ee=!Yt(d);let J;if(ee&&(J=R&&R.onVnodeBeforeUnmount)&&Ye(J,m,d),P&6)In(d.component,v,I);else{if(P&128){d.suspense.unmount(v,I);return}W&&Et(d,null,m,"beforeUnmount"),P&64?d.type.remove(d,m,v,Tt,I):b&&!b.hasOnce&&(x!==fe||M>0&&M&64)?Ct(b,m,v,!1,!0):(x===fe&&M&384||!w&&P&16)&&Ct(C,m,v),I&&_t(d)}(ee&&(J=R&&R.onVnodeUnmounted)||W)&&Ae(()=>{J&&Ye(J,m,d),W&&Et(d,null,m,"unmounted")},v)},_t=d=>{const{type:m,el:v,anchor:I,transition:w}=d;if(m===fe){St(v,I);return}if(m===_s){F(d);return}const x=()=>{r(v),w&&!w.persisted&&w.afterLeave&&w.afterLeave()};if(d.shapeFlag&1&&w&&!w.persisted){const{leave:R,delayLeave:$}=w,C=()=>R(v,x);$?$(d.el,x,C):C()}else x()},St=(d,m)=>{let v;for(;d!==m;)v=T(d),r(d),d=v;r(m)},In=(d,m,v)=>{const{bum:I,scope:w,job:x,subTree:R,um:$,m:C,a:b}=d;Lr(C),Lr(b),I&&Bn(I),w.stop(),x&&(x.flags|=8,ue(R,d,m,v)),$&&Ae($,m),Ae(()=>{d.isUnmounted=!0},m)},Ct=(d,m,v,I=!1,w=!1,x=0)=>{for(let R=x;R{if(d.shapeFlag&6)return Ft(d.component.subTree);if(d.shapeFlag&128)return d.suspense.next();const m=T(d.anchor||d.el),v=m&&m[ql];return v?T(v):m};let nn=!1;const Sn=(d,m,v)=>{let I;d==null?m._vnode&&(ue(m._vnode,null,null,!0),I=m._vnode.component):N(m._vnode||null,d,m,null,null,null,v),m._vnode=d,nn||(nn=!0,Mr(I),Ni(),nn=!1)},Tt={p:N,um:ue,m:be,r:_t,mt:de,mc:ke,pc:G,pbc:He,n:Ft,o:e};return{render:Sn,hydrate:void 0,createApp:va(Sn)}}function ys({type:e,props:t},n){return n==="svg"&&e==="foreignObject"||n==="mathml"&&e==="annotation-xml"&&t&&t.encoding&&t.encoding.includes("html")?void 0:n}function At({effect:e,job:t},n){n?(e.flags|=32,t.flags|=4):(e.flags&=-33,t.flags&=-5)}function Oa(e,t){return(!e||e&&!e.pendingBranch)&&t&&!t.persisted}function lo(e,t,n=!1){const s=e.children,r=t.children;if(H(s)&&H(r))for(let i=0;i>1,e[n[c]]0&&(t[s]=n[i-1]),n[i]=s)}}for(i=n.length,l=n[i-1];i-- >0;)n[i]=l,l=t[l];return n}function ao(e){const t=e.subTree.component;if(t)return t.asyncDep&&!t.asyncResolved?t:ao(t)}function Lr(e){if(e)for(let t=0;te.__isSuspense;function Da(e,t){t&&t.pendingBranch?H(e)?t.effects.push(...e):t.effects.push(e):jl(e)}const fe=Symbol.for("v-fgt"),ss=Symbol.for("v-txt"),ht=Symbol.for("v-cmt"),_s=Symbol.for("v-stc"),pn=[];let De=null;function j(e=!1){pn.push(De=e?null:[])}function Na(){pn.pop(),De=pn[pn.length-1]||null}let yn=1;function jr(e,t=!1){yn+=e,e<0&&De&&t&&(De.hasOnce=!0)}function fo(e){return e.dynamicChildren=yn>0?De||Vt:null,Na(),yn>0&&De&&De.push(e),e}function V(e,t,n,s,r,i){return fo(y(e,t,n,s,r,i,!0))}function mt(e,t,n,s,r){return fo(q(e,t,n,s,r,!0))}function tr(e){return e?e.__v_isVNode===!0:!1}function on(e,t){return e.type===t.type&&e.key===t.key}const po=({key:e})=>e??null,jn=({ref:e,ref_key:t,ref_for:n})=>(typeof e=="number"&&(e=""+e),e!=null?ae(e)||Se(e)||K(e)?{i:ye,r:e,k:t,f:!!n}:e:null);function y(e,t=null,n=null,s=0,r=null,i=e===fe?0:1,l=!1,c=!1){const f={__v_isVNode:!0,__v_skip:!0,type:e,props:t,key:t&&po(t),ref:t&&jn(t),scopeId:Ui,slotScopeIds:null,children:n,component:null,suspense:null,ssContent:null,ssFallback:null,dirs:null,transition:null,el:null,anchor:null,target:null,targetStart:null,targetAnchor:null,staticCount:0,shapeFlag:i,patchFlag:s,dynamicProps:r,dynamicChildren:null,appContext:null,ctx:ye};return c?(nr(f,n),i&128&&e.normalize(f)):n&&(f.shapeFlag|=ae(n)?8:16),yn>0&&!l&&De&&(f.patchFlag>0||i&6)&&f.patchFlag!==32&&De.push(f),f}const q=Fa;function Fa(e,t=null,n=null,s=0,r=null,i=!1){if((!e||e===Vi)&&(e=ht),tr(e)){const c=en(e,t,!0);return n&&nr(c,n),yn>0&&!i&&De&&(c.shapeFlag&6?De[De.indexOf(e)]=c:De.push(c)),c.patchFlag=-2,c}if(Ga(e)&&(e=e.__vccOpts),t){t=Ua(t);let{class:c,style:f}=t;c&&!ae(c)&&(t.class=xt(c)),ne(f)&&(Js(f)&&!H(f)&&(f=Ce({},f)),t.style=Be(f))}const l=ae(e)?1:uo(e)?128:Gl(e)?64:ne(e)?4:K(e)?2:0;return y(e,t,n,s,r,l,i,!0)}function Ua(e){return e?Js(e)||to(e)?Ce({},e):e:null}function en(e,t,n=!1,s=!1){const{props:r,ref:i,patchFlag:l,children:c,transition:f}=e,h=t?ho(r||{},t):r,p={__v_isVNode:!0,__v_skip:!0,type:e.type,props:h,key:h&&po(h),ref:t&&t.ref?n&&i?H(i)?i.concat(jn(t)):[i,jn(t)]:jn(t):i,scopeId:e.scopeId,slotScopeIds:e.slotScopeIds,children:c,target:e.target,targetStart:e.targetStart,targetAnchor:e.targetAnchor,staticCount:e.staticCount,shapeFlag:e.shapeFlag,patchFlag:t&&e.type!==fe?l===-1?16:l|16:l,dynamicProps:e.dynamicProps,dynamicChildren:e.dynamicChildren,appContext:e.appContext,dirs:e.dirs,transition:f,component:e.component,suspense:e.suspense,ssContent:e.ssContent&&en(e.ssContent),ssFallback:e.ssFallback&&en(e.ssFallback),placeholder:e.placeholder,el:e.el,anchor:e.anchor,ctx:e.ctx,ce:e.ce};return f&&s&&Xs(p,f.clone(p)),p}function tn(e=" ",t=0){return q(ss,null,e,t)}function Le(e="",t=!1){return t?(j(),mt(ht,null,e)):q(ht,null,e)}function Qe(e){return e==null||typeof e=="boolean"?q(ht):H(e)?q(fe,null,e.slice()):tr(e)?at(e):q(ss,null,String(e))}function at(e){return e.el===null&&e.patchFlag!==-1||e.memo?e:en(e)}function nr(e,t){let n=0;const{shapeFlag:s}=e;if(t==null)t=null;else if(H(t))n=16;else if(typeof t=="object")if(s&65){const r=t.default;r&&(r._c&&(r._d=!1),nr(e,r()),r._c&&(r._d=!0));return}else{n=32;const r=t._;!r&&!to(t)?t._ctx=ye:r===3&&ye&&(ye.slots._===1?t._=1:(t._=2,e.patchFlag|=1024))}else K(t)?(t={default:t,_ctx:ye},n=32):(t=String(t),s&64?(n=16,t=[tn(t)]):n=8);e.children=t,e.shapeFlag|=n}function ho(...e){const t={};for(let n=0;nIe||ye;let Gn,Ps;{const e=Zn(),t=(n,s)=>{let r;return(r=e[n])||(r=e[n]=[]),r.push(s),i=>{r.length>1?r.forEach(l=>l(i)):r[0](i)}};Gn=t("__VUE_INSTANCE_SETTERS__",n=>Ie=n),Ps=t("__VUE_SSR_SETTERS__",n=>_n=n)}const xn=e=>{const t=Ie;return Gn(e),e.scope.on(),()=>{e.scope.off(),Gn(t)}},Hr=()=>{Ie&&Ie.scope.off(),Gn(null)};function mo(e){return e.vnode.shapeFlag&4}let _n=!1;function Ka(e,t=!1,n=!1){t&&Ps(t);const{props:s,children:r}=e.vnode,i=mo(e);Ca(e,s,i,t),Ma(e,r,n||t);const l=i?za(e,t):void 0;return t&&Ps(!1),l}function za(e,t){const n=e.type;e.accessCache=Object.create(null),e.proxy=new Proxy(e.ctx,ua);const{setup:s}=n;if(s){ft();const r=e.setupContext=s.length>1?Va(e):null,i=xn(e),l=wn(s,e,0,[e.props,r]),c=fi(l);if(dt(),i(),(c||e.sp)&&!Yt(e)&&ji(e),c){if(l.then(Hr,Hr),t)return l.then(f=>{Kr(e,f)}).catch(f=>{es(f,e,0)});e.asyncDep=l}else Kr(e,l)}else go(e)}function Kr(e,t,n){K(t)?e.type.__ssrInlineRender?e.ssrRender=t:e.render=t:ne(t)&&(e.setupState=Ai(t)),go(e)}function go(e,t,n){const s=e.type;e.render||(e.render=s.render||tt);{const r=xn(e);ft();try{fa(e)}finally{dt(),r()}}}const Wa={get(e,t){return xe(e,"get",""),e[t]}};function Va(e){const t=n=>{e.exposed=n||{}};return{attrs:new Proxy(e.attrs,Wa),slots:e.slots,emit:e.emit,expose:t}}function rs(e){return e.exposed?e.exposeProxy||(e.exposeProxy=new Proxy(Ai(Rl(e.exposed)),{get(t,n){if(n in t)return t[n];if(n in dn)return dn[n](e)},has(t,n){return n in t||n in dn}})):e.proxy}function qa(e,t=!0){return K(e)?e.displayName||e.name:e.name||t&&e.__name}function Ga(e){return K(e)&&"__vccOpts"in e}const wt=(e,t)=>Nl(e,t,_n),Ja="3.5.30";/** +* @vue/runtime-dom v3.5.30 +* (c) 2018-present Yuxi (Evan) You and Vue contributors +* @license MIT +**/let Ds;const zr=typeof window<"u"&&window.trustedTypes;if(zr)try{Ds=zr.createPolicy("vue",{createHTML:e=>e})}catch{}const vo=Ds?e=>Ds.createHTML(e):e=>e,Ya="http://www.w3.org/2000/svg",Xa="http://www.w3.org/1998/Math/MathML",lt=typeof document<"u"?document:null,Wr=lt&<.createElement("template"),Za={insert:(e,t,n)=>{t.insertBefore(e,n||null)},remove:e=>{const t=e.parentNode;t&&t.removeChild(e)},createElement:(e,t,n,s)=>{const r=t==="svg"?lt.createElementNS(Ya,e):t==="mathml"?lt.createElementNS(Xa,e):n?lt.createElement(e,{is:n}):lt.createElement(e);return e==="select"&&s&&s.multiple!=null&&r.setAttribute("multiple",s.multiple),r},createText:e=>lt.createTextNode(e),createComment:e=>lt.createComment(e),setText:(e,t)=>{e.nodeValue=t},setElementText:(e,t)=>{e.textContent=t},parentNode:e=>e.parentNode,nextSibling:e=>e.nextSibling,querySelector:e=>lt.querySelector(e),setScopeId(e,t){e.setAttribute(t,"")},insertStaticContent(e,t,n,s,r,i){const l=n?n.previousSibling:t.lastChild;if(r&&(r===i||r.nextSibling))for(;t.insertBefore(r.cloneNode(!0),n),!(r===i||!(r=r.nextSibling)););else{Wr.innerHTML=vo(s==="svg"?`${e}`:s==="mathml"?`${e}`:e);const c=Wr.content;if(s==="svg"||s==="mathml"){const f=c.firstChild;for(;f.firstChild;)c.appendChild(f.firstChild);c.removeChild(f)}t.insertBefore(c,n)}return[l?l.nextSibling:t.firstChild,n?n.previousSibling:t.lastChild]}},Qa=Symbol("_vtc");function ec(e,t,n){const s=e[Qa];s&&(t=(t?[t,...s]:[...s]).join(" ")),t==null?e.removeAttribute("class"):n?e.setAttribute("class",t):e.className=t}const Vr=Symbol("_vod"),tc=Symbol("_vsh"),nc=Symbol(""),sc=/(?:^|;)\s*display\s*:/;function rc(e,t,n){const s=e.style,r=ae(n);let i=!1;if(n&&!r){if(t)if(ae(t))for(const l of t.split(";")){const c=l.slice(0,l.indexOf(":")).trim();n[c]==null&&Hn(s,c,"")}else for(const l in t)n[l]==null&&Hn(s,l,"");for(const l in n)l==="display"&&(i=!0),Hn(s,l,n[l])}else if(r){if(t!==n){const l=s[nc];l&&(n+=";"+l),s.cssText=n,i=sc.test(n)}}else t&&e.removeAttribute("style");Vr in e&&(e[Vr]=i?s.display:"",e[tc]&&(s.display="none"))}const qr=/\s*!important$/;function Hn(e,t,n){if(H(n))n.forEach(s=>Hn(e,t,s));else if(n==null&&(n=""),t.startsWith("--"))e.setProperty(t,n);else{const s=ic(e,t);qr.test(n)?e.setProperty(It(s),n.replace(qr,""),"important"):e[s]=n}}const Gr=["Webkit","Moz","ms"],bs={};function ic(e,t){const n=bs[t];if(n)return n;let s=Ee(t);if(s!=="filter"&&s in e)return bs[t]=s;s=Xn(s);for(let r=0;rws||(cc.then(()=>ws=0),ws=Date.now());function fc(e,t){const n=s=>{if(!s._vts)s._vts=Date.now();else if(s._vts<=n.attached)return;nt(dc(s,n.value),t,5,[s])};return n.value=e,n.attached=uc(),n}function dc(e,t){if(H(t)){const n=e.stopImmediatePropagation;return e.stopImmediatePropagation=()=>{n.call(e),e._stopped=!0},t.map(s=>r=>!r._stopped&&s&&s(r))}else return t}const ei=e=>e.charCodeAt(0)===111&&e.charCodeAt(1)===110&&e.charCodeAt(2)>96&&e.charCodeAt(2)<123,pc=(e,t,n,s,r,i)=>{const l=r==="svg";t==="class"?ec(e,s,l):t==="style"?rc(e,n,s):Jn(t)?Fs(t)||lc(e,t,n,s,i):(t[0]==="."?(t=t.slice(1),!0):t[0]==="^"?(t=t.slice(1),!1):hc(e,t,s,l))?(Xr(e,t,s),!e.tagName.includes("-")&&(t==="value"||t==="checked"||t==="selected")&&Yr(e,t,s,l,i,t!=="value")):e._isVueCE&&(mc(e,t)||e._def.__asyncLoader&&(/[A-Z]/.test(t)||!ae(s)))?Xr(e,Ee(t),s,i,t):(t==="true-value"?e._trueValue=s:t==="false-value"&&(e._falseValue=s),Yr(e,t,s,l))};function hc(e,t,n,s){if(s)return!!(t==="innerHTML"||t==="textContent"||t in e&&ei(t)&&K(n));if(t==="spellcheck"||t==="draggable"||t==="translate"||t==="autocorrect"||t==="sandbox"&&e.tagName==="IFRAME"||t==="form"||t==="list"&&e.tagName==="INPUT"||t==="type"&&e.tagName==="TEXTAREA")return!1;if(t==="width"||t==="height"){const r=e.tagName;if(r==="IMG"||r==="VIDEO"||r==="CANVAS"||r==="SOURCE")return!1}return ei(t)&&ae(n)?!1:t in e}function mc(e,t){const n=e._def.props;if(!n)return!1;const s=Ee(t);return Array.isArray(n)?n.some(r=>Ee(r)===s):Object.keys(n).some(r=>Ee(r)===s)}const ti=e=>{const t=e.props["onUpdate:modelValue"]||!1;return H(t)?n=>Bn(t,n):t};function gc(e){e.target.composing=!0}function ni(e){const t=e.target;t.composing&&(t.composing=!1,t.dispatchEvent(new Event("input")))}const xs=Symbol("_assign");function si(e,t,n){return t&&(e=e.trim()),n&&(e=Ls(e)),e}const vc={created(e,{modifiers:{lazy:t,trim:n,number:s}},r){e[xs]=ti(r);const i=s||r.props&&r.props.type==="number";zt(e,t?"change":"input",l=>{l.target.composing||e[xs](si(e.value,n,i))}),(n||i)&&zt(e,"change",()=>{e.value=si(e.value,n,i)}),t||(zt(e,"compositionstart",gc),zt(e,"compositionend",ni),zt(e,"change",ni))},mounted(e,{value:t}){e.value=t??""},beforeUpdate(e,{value:t,oldValue:n,modifiers:{lazy:s,trim:r,number:i}},l){if(e[xs]=ti(l),e.composing)return;const c=(i||e.type==="number")&&!/^0\d/.test(e.value)?Ls(e.value):e.value,f=t??"";c!==f&&(document.activeElement===e&&e.type!=="range"&&(s&&t===n||r&&e.value.trim()===f)||(e.value=f))}},yc=["ctrl","shift","alt","meta"],_c={stop:e=>e.stopPropagation(),prevent:e=>e.preventDefault(),self:e=>e.target!==e.currentTarget,ctrl:e=>!e.ctrlKey,shift:e=>!e.shiftKey,alt:e=>!e.altKey,meta:e=>!e.metaKey,left:e=>"button"in e&&e.button!==0,middle:e=>"button"in e&&e.button!==1,right:e=>"button"in e&&e.button!==2,exact:(e,t)=>yc.some(n=>e[`${n}Key`]&&!t.includes(n))},An=(e,t)=>{if(!e)return e;const n=e._withMods||(e._withMods={}),s=t.join(".");return n[s]||(n[s]=(r,...i)=>{for(let l=0;l{const n=e._withKeys||(e._withKeys={}),s=t.join(".");return n[s]||(n[s]=r=>{if(!("key"in r))return;const i=It(r.key);if(t.some(l=>l===i||bc[l]===i))return e(r)})},wc=Ce({patchProp:pc},Za);let ri;function xc(){return ri||(ri=Ea(wc))}const Ic=(...e)=>{const t=xc().createApp(...e),{mount:n}=t;return t.mount=s=>{const r=Cc(s);if(!r)return;const i=t._component;!K(i)&&!i.render&&!i.template&&(i.template=r.innerHTML),r.nodeType===1&&(r.textContent="");const l=n(r,!1,Sc(r));return r instanceof Element&&(r.removeAttribute("v-cloak"),r.setAttribute("data-v-app","")),l},t};function Sc(e){if(e instanceof SVGElement)return"svg";if(typeof MathMLElement=="function"&&e instanceof MathMLElement)return"mathml"}function Cc(e){return ae(e)?document.querySelector(e):e}const Tc={class:"footer"},kc={__name:"AppFooter",emits:["request-admin"],setup(e,{emit:t}){const n=t,s=oe(0);let r=null;Zs(()=>{r&&window.clearTimeout(r)});function i(){s.value+=1,s.value===1&&(r=window.setTimeout(()=>{s.value=0,r=null},2e3)),s.value>=5&&(r&&(window.clearTimeout(r),r=null),s.value=0,n("request-admin"))}return(l,c)=>(j(),V("div",Tc,[y("div",null,[c[0]||(c[0]=tn(" © 2026 AirShare Pro. All rights reserved. ",-1)),c[1]||(c[1]=y("span",{class:"divider-line"},"|",-1)),y("span",{id:"admin-trigger",title:"点击 5 次进入后台",onClick:i},"V 1.0.0")]),c[2]||(c[2]=y("div",{style:{"font-size":"12px","margin-top":"4px"}},[y("a",{href:"https://beian.miit.gov.cn/",target:"_blank",rel:"noreferrer"}," 粤ICP备2026888888号-1 ")],-1))]))}},$c=["fill","stroke"],pe={__name:"LocalIcon",props:{name:{type:String,required:!0},size:{type:[Number,String],default:24}},setup(e){const t=e,n={light_mode:{type:"stroke",shapes:[{tag:"circle",attrs:{cx:"12",cy:"12",r:"4"}},{tag:"path",attrs:{d:"M12 2v2.2"}},{tag:"path",attrs:{d:"M12 19.8V22"}},{tag:"path",attrs:{d:"M4.93 4.93 6.5 6.5"}},{tag:"path",attrs:{d:"m17.5 17.5 1.57 1.57"}},{tag:"path",attrs:{d:"M2 12h2.2"}},{tag:"path",attrs:{d:"M19.8 12H22"}},{tag:"path",attrs:{d:"m4.93 19.07 1.57-1.57"}},{tag:"path",attrs:{d:"M17.5 6.5 19.07 4.93"}}]},dark_mode:{type:"fill",shapes:[{tag:"path",attrs:{d:"M21 12.8A9 9 0 1 1 11.2 3a7 7 0 0 0 9.8 9.8Z"}}]},add_circle:{type:"stroke",shapes:[{tag:"circle",attrs:{cx:"12",cy:"12",r:"9"}},{tag:"path",attrs:{d:"M12 8v8"}},{tag:"path",attrs:{d:"M8 12h8"}}]},sensors:{type:"stroke",shapes:[{tag:"path",attrs:{d:"M12 12h.01"}},{tag:"path",attrs:{d:"M9.2 14.8a4 4 0 0 1 0-5.6"}},{tag:"path",attrs:{d:"M14.8 9.2a4 4 0 0 1 0 5.6"}},{tag:"path",attrs:{d:"M6.4 17.6a8 8 0 0 1 0-11.2"}},{tag:"path",attrs:{d:"M17.6 6.4a8 8 0 0 1 0 11.2"}}]},smartphone:{type:"stroke",shapes:[{tag:"rect",attrs:{x:"7",y:"2.5",width:"10",height:"19",rx:"2.5"}},{tag:"path",attrs:{d:"M10 5h4"}},{tag:"circle",attrs:{cx:"12",cy:"18",r:"0.8"}}]},laptop_mac:{type:"stroke",shapes:[{tag:"rect",attrs:{x:"5",y:"4",width:"14",height:"10",rx:"1.5"}},{tag:"path",attrs:{d:"M3 18h18"}},{tag:"path",attrs:{d:"M8 18a2 2 0 0 0 2 2h4a2 2 0 0 0 2-2"}}]},close:{type:"stroke",shapes:[{tag:"path",attrs:{d:"M6 6l12 12"}},{tag:"path",attrs:{d:"M18 6 6 18"}}]},cloud_upload:{type:"stroke",shapes:[{tag:"path",attrs:{d:"M7 18a4 4 0 1 1 .7-7.94A5.5 5.5 0 0 1 18 11a3.5 3.5 0 1 1-.5 7"}},{tag:"path",attrs:{d:"M12 10v8"}},{tag:"path",attrs:{d:"m8.8 13.2 3.2-3.2 3.2 3.2"}}]},arrow_upward:{type:"stroke",shapes:[{tag:"path",attrs:{d:"M12 19V6"}},{tag:"path",attrs:{d:"m6.5 11.5 5.5-5.5 5.5 5.5"}}]},send_and_archive:{type:"stroke",shapes:[{tag:"path",attrs:{d:"M3 6h18l-2 4H5Z"}},{tag:"path",attrs:{d:"M5 10v8a2 2 0 0 0 2 2h10a2 2 0 0 0 2-2v-8"}},{tag:"path",attrs:{d:"M12 11v5"}},{tag:"path",attrs:{d:"m9.5 13.5 2.5 2.5 2.5-2.5"}}]},chat_bubble:{type:"stroke",shapes:[{tag:"path",attrs:{d:"M6 18.5 3.5 21v-5A7.5 7.5 0 0 1 11 4.5h2A7.5 7.5 0 0 1 20.5 12v.5A7.5 7.5 0 0 1 13 20H8.5"}}]},content_copy:{type:"stroke",shapes:[{tag:"rect",attrs:{x:"9",y:"9",width:"10",height:"10",rx:"2"}},{tag:"path",attrs:{d:"M7 15H6a2 2 0 0 1-2-2V6a2 2 0 0 1 2-2h7a2 2 0 0 1 2 2v1"}}]},check:{type:"stroke",shapes:[{tag:"path",attrs:{d:"m5 12 4.2 4.2L19 7.5"}}]},draft:{type:"stroke",shapes:[{tag:"rect",attrs:{x:"4",y:"5",width:"16",height:"14",rx:"2"}},{tag:"path",attrs:{d:"m5 7 7 5 7-5"}}]},save:{type:"stroke",shapes:[{tag:"path",attrs:{d:"M5 20h14a1 1 0 0 0 1-1V7.5L16.5 4H5a1 1 0 0 0-1 1v14a1 1 0 0 0 1 1Z"}},{tag:"path",attrs:{d:"M8 4v5h7"}},{tag:"rect",attrs:{x:"8",y:"14",width:"8",height:"4",rx:"1"}}]},download:{type:"stroke",shapes:[{tag:"path",attrs:{d:"M12 5v10"}},{tag:"path",attrs:{d:"m7.5 10.5 4.5 4.5 4.5-4.5"}},{tag:"path",attrs:{d:"M5 19h14"}}]}},s=wt(()=>n[t.name]||n.close),r=wt(()=>typeof t.size=="number"?`${t.size}px`:/^\d+(\.\d+)?$/.test(t.size)?`${t.size}px`:t.size);return(i,l)=>(j(),V("span",{class:"app-icon",style:Be({width:r.value,height:r.value}),"aria-hidden":"true"},[(j(),V("svg",{viewBox:"0 0 24 24",fill:s.value.type==="fill"?"currentColor":"none",stroke:s.value.type==="stroke"?"currentColor":"none","stroke-width":"1.8","stroke-linecap":"round","stroke-linejoin":"round"},[(j(!0),V(fe,null,Qt(s.value.shapes,(c,f)=>(j(),mt(la(c.tag),ho({key:`${e.name}-${f}`},{ref_for:!0},c.attrs),null,16))),128))],8,$c))],4))}},Mc={class:"header"},Rc={__name:"AppHeader",props:{theme:{type:String,required:!0}},emits:["toggle-theme"],setup(e){return(t,n)=>(j(),V("div",Mc,[n[1]||(n[1]=y("h1",null,"AirShare Pro",-1)),n[2]||(n[2]=y("p",null,"跨端局域网 & P2P 传输中心",-1)),y("button",{class:"theme-toggle",title:"切换日夜模式",onClick:n[0]||(n[0]=s=>t.$emit("toggle-theme"))},[q(pe,{id:"theme-icon",name:e.theme==="dark"?"dark_mode":"light_mode",size:"22"},null,8,["name"])])]))}},Ec={class:"card"},Ac={key:0,class:"section-title"},hn={__name:"GlassCard",props:{title:{type:String,default:""}},setup(e){return(t,n)=>(j(),V("div",Ec,[e.title?(j(),V("div",Ac,te(e.title),1)):Le("",!0),ca(t.$slots,"default")]))}},Oc={class:"admin-panel active"},Pc={class:"card admin-header-card"},Dc={class:"transfer-head transfer-head-compact"},Nc={class:"main-grid admin-summary-grid"},Fc={class:"admin-stats-panel"},Uc={class:"admin-stats-row"},Bc={class:"admin-fluid-content"},Lc={class:"admin-fluid-icon"},jc={class:"admin-fluid-copy"},Hc={key:0,class:"stat-suffix"},Kc={class:"admin-config-stack"},zc={class:"text-input-group admin-config-row admin-config-row-field admin-config-row-last"},Wc={class:"admin-field-control-row"},Vc=["value"],qc={class:"text-input-group admin-config-row admin-config-row-field admin-config-row-last"},Gc={class:"admin-field-control-row"},Jc=["value"],Yc={class:"admin-config-insights"},Xc={class:"admin-config-highlight"},Zc={class:"admin-config-highlight"},Qc={class:"admin-table-wrapper"},eu={class:"admin-table"},tu={__name:"AdminPanel",props:{stats:{type:Array,required:!0},records:{type:Array,required:!0},fileLimit:{type:Number,required:!0},minioCapacity:{type:Number,required:!0}},emits:["exit","save-config","update:file-limit","update:minio-capacity"],setup(e){function t(l){const c=Number(l)||0;return c>=1024?`${(c/1024).toFixed(c%1024===0?0:1)} GB`:`${c} MB`}function n(l){const c=Number(l)||0;return c>=1024?`${(c/1024).toFixed(c%1024===0?0:1)} TB`:`${c} GB`}function s(l){return l==="blue"?{color:"var(--accent-blue)"}:l==="cyan"?{color:"var(--accent-cyan)"}:l==="success"?{color:"var(--success-green)"}:l==="danger"?{color:"var(--danger-red)"}:{color:"var(--text-main)"}}function r(l){return l==="success"?{color:"var(--success-green)",fontWeight:500}:l==="primary"?{color:"var(--accent-blue)",fontWeight:500}:{color:"var(--danger-red)",fontWeight:500}}function i(l){const c=Number(l)||0;return{"--fluid-level":`${Math.max(0,Math.min(c,100))}%`}}return(l,c)=>(j(),V("div",Oc,[y("div",Pc,[y("div",Dc,[c[5]||(c[5]=y("div",{class:"connected-to"},[y("h2",{class:"admin-title"},"管理控制台"),y("p",{class:"admin-subtitle"},"AirShare Pro System Dashboard")],-1)),y("button",{class:"btn-small-primary",type:"button",onClick:c[0]||(c[0]=f=>l.$emit("exit"))},"退出管理")])]),y("div",Nc,[q(hn,{class:"admin-stats-card",title:"系统运行状态"},{default:Jt(()=>[y("div",Fc,[y("div",Uc,[(j(!0),V(fe,null,Qt(e.stats,f=>(j(),V("div",{key:f.label,class:xt(["admin-stat-item",{"admin-stat-item-fluid":f.kind==="minio"}])},[f.kind==="minio"?(j(),V("div",{key:0,class:"admin-fluid-card",style:Be(i(f.percent))},[c[6]||(c[6]=y("div",{class:"admin-fluid-fill"},[y("span",{class:"admin-fluid-wave admin-fluid-wave-a"}),y("span",{class:"admin-fluid-wave admin-fluid-wave-b"})],-1)),y("div",Bc,[y("div",Lc,[q(pe,{name:"save",size:"18"})]),y("div",jc,[y("h3",{style:Be(s(f.tone))},te(f.value),5),y("p",null,te(f.label),1),y("small",null,te(f.detail),1)])])],4)):(j(),V(fe,{key:1},[c[7]||(c[7]=y("span",{class:"admin-stat-kicker"},"实时指标",-1)),y("h3",{style:Be(s(f.tone))},[tn(te(f.value),1),f.suffix?(j(),V("span",Hc,te(f.suffix),1)):Le("",!0)],4),y("p",null,te(f.label),1)],64))],2))),128))])])]),_:1}),q(hn,{class:"admin-config-card",title:"核心参数配置"},{default:Jt(()=>[y("div",Kc,[y("div",zc,[c[8]||(c[8]=y("div",{class:"admin-field-meta"},[y("label",{class:"admin-field-label",for:"admin-file-limit"},"单文件大小限制"),y("p",{class:"admin-field-hint"},"单位为 MB,超过该阈值的文件会按当前后端策略处理。")],-1)),y("div",Wc,[y("input",{id:"admin-file-limit",value:e.fileLimit,min:"1",placeholder:"10240",type:"number",onInput:c[1]||(c[1]=f=>l.$emit("update:file-limit",Number(f.target.value)||0))},null,40,Vc),y("button",{title:"保存配置",type:"button",onClick:c[2]||(c[2]=f=>l.$emit("save-config"))},[q(pe,{name:"save",size:"18"})])])]),y("div",qc,[c[9]||(c[9]=y("div",{class:"admin-field-meta"},[y("label",{class:"admin-field-label",for:"admin-minio-capacity"},"MinIO 总容量"),y("p",{class:"admin-field-hint"},"单位为 GB,用于容量卡和液位比例计算。")],-1)),y("div",Gc,[y("input",{id:"admin-minio-capacity",value:e.minioCapacity,min:"1",placeholder:"120",type:"number",onInput:c[3]||(c[3]=f=>l.$emit("update:minio-capacity",Number(f.target.value)||0))},null,40,Jc),y("button",{title:"保存配置",type:"button",onClick:c[4]||(c[4]=f=>l.$emit("save-config"))},[q(pe,{name:"save",size:"18"})])])]),y("div",Yc,[y("div",Xc,[c[10]||(c[10]=y("span",{class:"admin-config-badge"},"ACTIVE POLICY",-1)),y("h3",null,te(t(e.fileLimit)),1),c[11]||(c[11]=y("p",null,"当前单文件阈值。超过该体积后,文件会按后端已设定的传输与存档策略处理。",-1))]),y("div",Zc,[c[12]||(c[12]=y("span",{class:"admin-config-badge"},"MINIO CAPACITY",-1)),y("h3",null,te(n(e.minioCapacity)),1),c[13]||(c[13]=y("p",null,"当前 MinIO 总容量基线,用于后台容量展示与液位占比计算。",-1))])])])]),_:1})]),q(hn,{class:"admin-table-card",title:"最近传输记录(Top 5)"},{default:Jt(()=>[y("div",Qc,[y("table",eu,[c[14]||(c[14]=y("thead",null,[y("tr",null,[y("th",null,"时间"),y("th",null,"发送端特征"),y("th",null,"传输类型"),y("th",null,"数据量"),y("th",null,"状态")])],-1)),y("tbody",null,[(j(!0),V(fe,null,Qt(e.records,f=>(j(),V("tr",{key:`${f.time}-${f.peer}`},[y("td",null,te(f.time),1),y("td",null,te(f.peer),1),y("td",null,te(f.type),1),y("td",null,te(f.size),1),y("td",{style:Be(r(f.tone))},te(f.status),5)]))),128))])])])]),_:1})]))}},nu={class:"local-device-name"},su={key:0,class:"radar-container"},ru={class:"radar"},iu={key:1,class:"device-list"},ou=["onClick"],lu={class:"device-icon"},au={class:"device-info"},cu={key:2,class:"radar-container"},uu={class:"radar"},fu={__name:"DeviceDiscoveryCard",props:{isScanning:{type:Boolean,required:!0},localDeviceName:{type:String,default:""},devices:{type:Array,required:!0}},emits:["select-device"],setup(e,{emit:t}){const n=t;function s(r){n("select-device",r)}return(r,i)=>(j(),mt(hn,{title:"局域网自动发现"},{default:Jt(()=>[y("p",nu,[i[0]||(i[0]=tn(" 本机:",-1)),y("strong",null,te(e.localDeviceName||"识别中"),1)]),e.isScanning?(j(),V("div",su,[y("div",ru,[q(pe,{class:"radar-icon",name:"sensors",size:"36"})]),i[1]||(i[1]=y("p",{class:"scan-status"},"正在扫描附近设备...",-1))])):e.devices.length?(j(),V("div",iu,[(j(!0),V(fe,null,Qt(e.devices,l=>(j(),V("button",{key:l.id,class:"device-item",type:"button",onClick:c=>s(l)},[y("div",lu,[q(pe,{name:l.icon,size:"24"},null,8,["name"])]),y("div",au,[y("h4",null,te(l.name),1),y("p",null,te(l.description),1)]),i[2]||(i[2]=y("div",{class:"device-status-beacon","aria-hidden":"true"},[y("span",{class:"device-status-dot"}),y("span",{class:"device-status-ring"}),y("span",{class:"device-status-ring device-status-ring-delay"})],-1))],8,ou))),128))])):(j(),V("div",cu,[y("div",uu,[q(pe,{class:"radar-icon",name:"devices",size:"36"})]),i[3]||(i[3]=y("p",{class:"scan-status"},"暂未发现局域网设备,请保持页面开启后重试",-1))]))]),_:1}))}},du={key:0,class:"room-action-area"},pu={class:"room-input-group"},hu=["value"],mu={key:0,class:"pending-downloads"},gu={class:"pending-downloads-head"},vu=["href"],yu={class:"pending-download-copy"},_u=["title"],bu={class:"pending-download-icon","aria-hidden":"true"},wu={key:1,class:"waiting-area"},xu={class:"huge-code"},Iu={__name:"RemoteRoomCard",props:{roomCodeInput:{type:String,required:!0},isWaiting:{type:Boolean,required:!0},generatedCode:{type:String,required:!0},pendingDownloads:{type:Array,required:!0}},emits:["update-room-code","create-room","join-room","cancel-room"],setup(e,{emit:t}){const n=t;function s(i){n("update-room-code",i.target.value)}function r(){n("join-room")}return(i,l)=>(j(),mt(hn,{title:"远程直连"},{default:Jt(()=>[e.isWaiting?(j(),V("div",wu,[l[6]||(l[6]=y("p",{class:"waiting-subtitle"},"您的房间号码",-1)),y("div",xu,te(e.generatedCode),1),l[7]||(l[7]=y("div",{class:"spinner"},null,-1)),l[8]||(l[8]=y("p",{class:"waiting-tip"},"等待对方加入...",-1)),y("button",{class:"btn-cancel",type:"button",onClick:l[2]||(l[2]=c=>i.$emit("cancel-room"))},"取消建房")])):(j(),V("div",du,[y("button",{class:"btn-create",type:"button",onClick:l[0]||(l[0]=c=>i.$emit("create-room"))},[q(pe,{name:"add_circle",size:"22"}),l[3]||(l[3]=tn(" 创建专属传输房间 ",-1))]),l[5]||(l[5]=y("div",{class:"divider"},"或",-1)),y("div",pu,[y("input",{class:"room-code",inputmode:"numeric",maxlength:"4",pattern:"\\d*",placeholder:"输入4位房间号",type:"text",value:e.roomCodeInput,onInput:s,onKeyup:yo(r,["enter"])},null,40,hu),y("button",{class:"btn-primary",type:"button",onClick:l[1]||(l[1]=c=>i.$emit("join-room"))},"加入房间")]),e.pendingDownloads.length?(j(),V("div",mu,[y("div",gu,[l[4]||(l[4]=y("span",null,"待领取文件",-1)),y("span",null,te(e.pendingDownloads.length),1)]),(j(!0),V(fe,null,Qt(e.pendingDownloads,c=>(j(),V("a",{key:c.transfer_id,class:"pending-download-item",href:c.download_path,target:"_blank",rel:"noopener noreferrer"},[y("div",yu,[y("strong",{title:c.name},te(c.name),9,_u),y("p",null,te(c.size_label)+" · "+te(c.created_label),1)]),y("span",bu,[q(pe,{name:"download",size:"18"})])],8,vu))),128))])):Le("",!0)]))]),_:1}))}},Su={class:"file-info"},Cu=["title"],Tu={class:"file-info-right"},ku=["download","href"],$u={key:0,class:"progress-bar-container"},Mu={__name:"TransferQueueItem",props:{item:{type:Object,required:!0}},emits:["remove","start-upload","copy"],setup(e){const t=e,n=wt(()=>t.item.tone==="success"?{color:"var(--success-green)"}:t.item.tone==="primary"?{color:"var(--accent-blue)"}:t.item.tone==="danger"?{color:"var(--danger-red)"}:{color:"var(--text-secondary)"}),s=wt(()=>t.item.kind==="text"?{color:"var(--success-green)",background:"rgba(48, 209, 88, 0.1)"}:{});return(r,i)=>(j(),V("div",{class:xt(["batch-item",{"pending-file":e.item.kind==="file"&&e.item.pending}])},[y("div",Su,[y("div",{class:"file-info-left",style:Be(e.item.kind==="text"?{maxWidth:"70%"}:null)},[y("div",{class:"file-icon-wrapper",style:Be(s.value)},[q(pe,{name:e.item.kind==="text"?"chat_bubble":"draft",size:"18"},null,8,["name"])],4),y("span",{class:"file-name",title:e.item.kind==="text"?e.item.text:e.item.name},te(e.item.kind==="text"?e.item.text:e.item.name),9,Cu)],4),y("div",Tu,[y("span",{class:"file-status",style:Be(n.value)},te(e.item.kind==="text"&&e.item.copied?"已复制":e.item.status),5),e.item.kind==="text"?(j(),V("button",{key:0,class:"action-btn",title:"复制文本",type:"button",onClick:i[0]||(i[0]=l=>r.$emit("copy",e.item.id))},[q(pe,{name:e.item.copied?"check":"content_copy",size:"16"},null,8,["name"])])):Le("",!0),e.item.kind==="file"&&e.item.pending?(j(),V("button",{key:1,class:"action-btn primary",title:"发送文件",type:"button",onClick:i[1]||(i[1]=l=>r.$emit("start-upload",e.item.id))},[q(pe,{name:"arrow_upward",size:"16"})])):Le("",!0),e.item.kind==="file"&&e.item.downloadUrl?(j(),V("a",{key:2,class:"action-btn primary",download:e.item.name,href:e.item.downloadUrl,title:"保存文件"},[q(pe,{name:"download",size:"16"})],8,ku)):Le("",!0),y("button",{class:"action-btn danger",title:"移除记录",type:"button",onClick:i[2]||(i[2]=l=>r.$emit("remove",e.item.id))},[q(pe,{name:"close",size:"16"})])])]),e.item.kind==="file"?(j(),V("div",$u,[y("div",{class:xt(["progress-bar-fill",{success:e.item.tone==="success"}]),style:Be({width:`${e.item.progress}%`})},null,6)])):Le("",!0)],2))}},Ru={class:"transfer-panel active"},Eu={class:"card"},Au={class:"transfer-head"},Ou={class:"connected-to"},Pu={key:0,class:"connection-hint"},Du={class:"text-input-group"},Nu={__name:"TransferPanel",props:{peerName:{type:String,required:!0},connectionType:{type:String,required:!0},networkHint:{type:String,default:""},items:{type:Array,required:!0},hasPendingItems:{type:Boolean,required:!0}},emits:["close","send-text","files-selected","send-all-pending","remove-item","start-upload","copy-item"],setup(e,{emit:t}){const n=e,s=t,r=oe(""),i=oe(!1),l=oe(null),c=oe(null);Nt(()=>n.items.length,async()=>{await Pi(),l.value&&(l.value.scrollTop=l.value.scrollHeight)});function f(){var T;(T=c.value)==null||T.click()}function h(){s("send-text",r.value),r.value=""}function p(T){const S=Array.from(T.target.files||[]);S.length&&s("files-selected",S),T.target.value=""}function _(T){var U;i.value=!1;const S=Array.from(((U=T.dataTransfer)==null?void 0:U.files)||[]);S.length&&s("files-selected",S)}return(T,S)=>(j(),V("div",Ru,[y("div",Eu,[y("div",Au,[y("div",Ou,[y("h2",null,te(e.peerName),1),y("p",null,te(e.connectionType),1),e.networkHint?(j(),V("small",Pu,te(e.networkHint),1)):Le("",!0)]),y("button",{class:"close-btn",type:"button",onClick:S[0]||(S[0]=U=>T.$emit("close"))},[q(pe,{name:"close",size:"20"})])]),y("div",{class:xt(["drop-zone",{"drop-zone-active":i.value}]),onClick:f,onDragenter:S[1]||(S[1]=An(U=>i.value=!0,["prevent"])),onDragover:S[2]||(S[2]=An(U=>i.value=!0,["prevent"])),onDragleave:S[3]||(S[3]=An(U=>i.value=!1,["prevent"])),onDrop:An(_,["prevent"])},[q(pe,{class:"drop-zone-icon",name:"cloud_upload",size:"42"}),S[9]||(S[9]=y("p",{class:"drop-zone-text"},"点击或拖拽多个文件到这里",-1)),y("input",{ref_key:"fileInput",ref:c,class:"hidden",multiple:"",type:"file",onChange:p},null,544)],34),y("div",Du,[Hl(y("input",{"onUpdate:modelValue":S[4]||(S[4]=U=>r.value=U),placeholder:"输入要发送的文本或链接...",type:"text",onKeyup:yo(h,["enter"])},null,544),[[vc,r.value]]),y("button",{title:"发送文本",type:"button",onClick:h},[q(pe,{name:"arrow_upward",size:"20"})])]),y("div",{class:xt(["batch-actions",{active:e.hasPendingItems}])},[y("button",{class:"btn-small-primary",type:"button",onClick:S[5]||(S[5]=U=>T.$emit("send-all-pending"))},[q(pe,{name:"send_and_archive",size:"16"}),S[10]||(S[10]=tn(" 一键发送全部 ",-1))])],2),e.items.length?(j(),V("div",{key:0,ref_key:"batchContainer",ref:l,class:"batch-progress-container"},[(j(!0),V(fe,null,Qt(e.items,U=>(j(),mt(Mu,{key:U.id,item:U,onCopy:S[6]||(S[6]=N=>T.$emit("copy-item",N)),onRemove:S[7]||(S[7]=N=>T.$emit("remove-item",N)),onStartUpload:S[8]||(S[8]=N=>T.$emit("start-upload",N))},null,8,["item"]))),128))],512)):Le("",!0)])]))}};let Wt={deviceId:"",token:""};const Fu="filefast_device_id",Uu="filefast_device_token";function _o(){return!Wt.deviceId||!Wt.token?{}:{"X-Device-ID":Wt.deviceId,"X-Device-Token":Wt.token}}function Bu(e={},t=!1){return{...t?{"Content-Type":"application/json"}:{},..._o(),...e}}function Lu(e,t){if(!t||Object.keys(t).length===0)return e;const n=new URLSearchParams;Object.entries(t).forEach(([r,i])=>{i!=null&&i!==""&&n.set(r,String(i))});const s=n.toString();return s?`${e}?${s}`:e}async function On(e,t={}){const n=t.body!==void 0,s=await fetch(Lu(e,t.query),{method:t.method||"GET",headers:Bu(t.headers,n),body:n?JSON.stringify(t.body):void 0}),r=await s.json().catch(()=>({}));if(!s.ok){const i=new Error(r.error||`Request failed: ${s.status}`);throw i.status=s.status,i}return r.data}const ve={get(e,t={}){return On(e,{...t,method:"GET"})},post(e,t,n={}){return On(e,{...n,method:"POST",body:t})},put(e,t,n={}){return On(e,{...n,method:"PUT",body:t})},patch(e,t,n={}){return On(e,{...n,method:"PATCH",body:t})}};function ii(e,t){Wt={deviceId:e||"",token:t||""},Ku(Wt)}function ju(){return _o()}function Hu(e){return{Authorization:`Bearer ${e}`}}function Ku(e){typeof document>"u"||(oi(Fu,e.deviceId),oi(Uu,e.token))}function oi(e,t){if(!t){document.cookie=`${e}=; Path=/; Max-Age=0; SameSite=Lax`;return}document.cookie=`${e}=${encodeURIComponent(t)}; Path=/; SameSite=Lax`}function Pn(e){return{headers:Hu(e)}}const Ht={login(e,t){return ve.post("/api/admin/login",{username:e,password:t})},stats(e){return ve.get("/api/admin/stats",Pn(e))},config(e){return ve.get("/api/admin/config",Pn(e))},updateConfig(e,t){return ve.put("/api/admin/config",t,Pn(e))},recentTransfers(e){return ve.get("/api/admin/transfers/recent",Pn(e))}},Dn={register(e){return ve.post("/api/devices/register",e)},heartbeat(e){return ve.post("/api/devices/heartbeat",{device_id:e})},listCandidates(e){return ve.get("/api/devices/candidates",{query:{deviceId:e}})},listPendingDownloads(e){return ve.get(`/api/devices/${encodeURIComponent(e)}/pending-downloads`)}},Nn={create(e){return ve.post("/api/rooms",{creator_device_id:e})},get(e){return ve.get(`/api/rooms/${encodeURIComponent(e)}`)},join(e,t){return ve.post("/api/rooms/join",{code:e,joiner_device_id:t})},cancel(e,t){return ve.post(`/api/rooms/${encodeURIComponent(e)}/cancel`,{requester_id:t})}},zu={config(){return ve.get("/api/runtime/config")}},ge={create(e){return ve.post("/api/transfers",e)},presignFallback(e){return ve.post(`/api/transfers/${encodeURIComponent(e)}/fallback/presign`,{})},uploadFallback(e,t,n){return Wu(`/api/transfers/${encodeURIComponent(e)}/fallback/upload`,t,n)},updateStatus(e,t){return ve.patch(`/api/transfers/${encodeURIComponent(e)}/status`,t)}};function Wu(e,t,n){return new Promise((s,r)=>{const i=new XMLHttpRequest;i.open("PUT",e),i.responseType="json",i.setRequestHeader("Content-Type",t.type||"application/octet-stream"),Object.entries(ju()).forEach(([l,c])=>{i.setRequestHeader(l,c)}),i.upload.onprogress=l=>{!l.lengthComputable||typeof n!="function"||n(Math.round(l.loaded/l.total*100))},i.onload=()=>{const l=i.response||Vu(i.responseText);if(i.status>=200&&i.status<300){s(l.data);return}r(new Error((l==null?void 0:l.error)||`Upload failed: ${i.status}`))},i.onerror=()=>r(new Error("Upload failed")),i.send(t)})}function Vu(e){try{return JSON.parse(e)}catch{return null}}const qu={class:"container"},Gu={key:0,class:"main-grid"},Is="filefast-admin-token",Fn="filefast-admin-view",Un="filefast-device-id",li="filefast-device-name",Ss="filefast-device-token",Ju=15e3,Yu=5e3,Xu=2e3,Zu=3e3,Qu=4*1024*1024,ef=2e4,tf=16*1024,ai=512*1024,nf={__name:"App",setup(e){const t=oe(localStorage.getItem("airshare-theme")||"light"),n=oe(localStorage.getItem(Fn)==="admin"?"admin":"main"),s=oe(!0),r=oe([]),i=oe(""),l=oe(!1),c=oe("----"),f=oe([]),h=oe({name:"--",type:"等待连接",deviceId:"",networkGroupKey:""}),p=oe([]),_=oe("/ws"),T=oe(10240),S=oe(120),U=oe([]),N=oe([]),Y=oe(null),z=oe(localStorage.getItem(Is)||""),E=oe({id:"",name:"",type:""}),Q=localStorage.getItem(Un)||"",F=localStorage.getItem(Ss)||"";Q&&F&&ii(Q,F);const le=new Map,he=new Map,Te=new Map,ke=new Map;let vt=null,He=null,Je=null,Ke=null,yt=null,de=null,st=null,B=null,L=null,G="",_e="p2p",ze=!1,be=!1,ue=!1,_t=null,St=null;const In=wt(()=>p.value.filter(o=>o.kind==="file"&&o.pending)),Ct=wt(()=>In.value.length>0),Ft=wt(()=>!h.value.deviceId||lr(h.value.networkGroupKey)||Po()?"":"当前是跨网络访问,未配置 TURN 时实时通道可能失败。文本和小文件可回退中转,大文件建议使用 MinIO。");Nt(t,o=>{document.body.setAttribute("data-theme",o),localStorage.setItem("airshare-theme",o)},{immediate:!0}),Nt(n,o=>{if(o==="admin"&&z.value){localStorage.setItem(Fn,"admin");return}localStorage.removeItem(Fn)}),Nt([n,z],([o,a])=>{Ke&&(window.clearInterval(Ke),Ke=null),!(o!=="admin"||!a)&&(Ke=window.setInterval(()=>{is().catch(u=>{console.error(u)})},5e3))}),zi(async()=>{_.value=Oo(),await nn(),n.value==="admin"&&z.value&&is().catch(o=>{console.error(o)}),He=window.setInterval(()=>{w()},Ju),vt=window.setInterval(()=>{v()},Yu),yt=window.setInterval(()=>{I()},1e4)}),Zs(()=>{vt&&window.clearInterval(vt),He&&window.clearInterval(He),Ke&&window.clearInterval(Ke),yt&&window.clearInterval(yt),b(),it(),vr(),D()});async function nn(){try{await Tt(),await m(),await v()}catch(o){window.alert(`后端连接失败:${o.message}`)}}function Sn(){t.value=t.value==="dark"?"light":"dark"}async function Tt(){try{Cn(await zu.config())}catch(o){console.error(o)}}function Cn(o){o&&(Y.value=o,T.value=Math.round((o.max_minio_fallback_size_bytes||0)/1024/1024),S.value=Math.max(0,Math.round((o.minio_capacity_bytes||0)/1024/1024/1024)))}function d(o){i.value=o.replace(/\D/g,"").slice(0,4)}async function m(){const o=Mo(),a=Ro(o),u=Eo(),g=await Dn.register({device_id:o,name:a,type:u,network_group_key:or()});localStorage.setItem(Un,g.id),g.auth_token&&(localStorage.setItem(Ss,g.auth_token),ii(g.id,g.auth_token)),E.value={id:g.id,name:g.name,type:g.type},await I(),gr()}async function v(){if(E.value.id)try{const o=await ge.create({kind:"text",name:"text-message",content:value,sender_device_id:E.value.id,receiver_device_id:h.value.deviceId});try{await sn(o,value)}catch(a){console.warn("realtime text send failed, fallback to relay",a),await sr(o,value)}p.value.push({id:Ue("text"),transferId:o.id,kind:"text",text:value,status:"已发送",tone:"success",copied:!1})}catch(o){window.alert(`发送文本失败:${o.message}`)}}async function I(){if(!E.value.id){f.value=[];return}try{const o=await Dn.listPendingDownloads(E.value.id);f.value=o.map(a=>({...a,download_path:a.download_path||`/api/transfers/${encodeURIComponent(a.transfer_id)}/fallback/download`,size_label:Lt(Number(a.size_bytes||0)),created_label:us(a.created_at)}))}catch(o){if((o==null?void 0:o.status)===404){f.value=[];return}console.error(o)}}async function w(){if(E.value.id)try{await Dn.heartbeat(E.value.id)}catch(o){console.error(o)}}async function x(){if(!E.value.id){window.alert("当前设备尚未注册到后端");return}try{const o=await Nn.create(E.value.id);c.value=o.code,l.value=!0,C(o.code)}catch(o){window.alert(`创建房间失败:${o.message}`)}}async function R(){const o=c.value;b();try{l.value&&o!=="----"&&await Nn.cancel(o,E.value.id)}catch(a){console.error(a)}finally{l.value=!1,c.value="----"}}async function $(){if(!(i.value.length<4))try{const o=await Nn.join(i.value,E.value.id),a=$t(o.creator_device_id);i.value="",P({deviceId:o.creator_device_id,name:(a==null?void 0:a.name)||`房间 ${o.code} 创建者`,type:"房间配对成功"})}catch(o){window.alert(`加入房间失败:${o.message}`)}}function C(o){b(),Je=window.setInterval(async()=>{try{const a=await Nn.get(o);if(a.status==="joined"&&a.joiner_device_id){const u=$t(a.joiner_device_id);P({deviceId:a.joiner_device_id,name:(u==null?void 0:u.name)||`房间 ${o} 对端`,type:"房间配对成功"});return}(a.status==="expired"||a.status==="canceled")&&(b(),l.value=!1,c.value="----")}catch(a){console.error(a)}},Xu)}function b(){Je&&(window.clearInterval(Je),Je=null)}function P(o){const a=o.deviceId||o.id||"",u=o.connectionType||o.type||"点对点传输";b(),h.value.deviceId!==a&&(it(),D()),h.value={name:o.name,type:o.connectionType||o.type||"点对点传输",deviceId:o.deviceId||o.id||"",networkGroupKey:o.network_group_key||""},l.value=!1,c.value="----",n.value="transfer",h.value.baseType=u,h.value.type=u,h.value.deviceId=a,We("正在建立实时通道"),rt(a,{initiate:!0})}function M(o,a=!1){const u=o.deviceId||o.id||"",g=o.connectionType||o.type||"点对点传输";h.value.deviceId===u&&n.value==="transfer"||(it(),a||D()),h.value={name:o.name,type:o.connectionType||o.type||"点对点传输",deviceId:u,networkGroupKey:o.network_group_key||""},l.value=!1,c.value="----",n.value="transfer",h.value.baseType=g,h.value.type=g,h.value.deviceId=u,We("正在建立实时通道"),rt(u)}function O(){it(),D(),h.value={name:"--",type:"等待连接",deviceId:"",networkGroupKey:""},n.value="main",h.value.baseType="等待连接",h.value.type="等待连接"}function D(){p.value.forEach(o=>Pe(o)),p.value=[],he.clear()}async function W(o){const a=o.trim();if(a){if(!h.value.deviceId){window.alert("当前没有可用的接收端");return}try{const u=await ge.create({kind:"text",name:"text-message",content:a,sender_device_id:E.value.id,receiver_device_id:h.value.deviceId});ce("transfer.created",h.value.deviceId,{transfer_id:u.id,kind:"text",name:"text-message",content:a,sender_device_id:E.value.id,sender_name:E.value.name,sender_type:E.value.type,receiver_device_id:h.value.deviceId,final_status:"completed",current_channel:"p2p",transport_options:os()}),await ge.updateStatus(u.id,{current_channel:"p2p",final_status:"completed"}),ce("transfer.updated",h.value.deviceId,{transfer_id:u.id,final_status:"completed",current_channel:"p2p"}),p.value.push({id:Ue("text"),transferId:u.id,kind:"text",text:a,status:"已发送",tone:"success",copied:!1})}catch(u){window.alert(`发送文本失败:${u.message}`)}}}function ee(o){const a=o.filter(Boolean).map((u,g)=>({id:Ue(`file-${g}`),kind:"file",file:u,name:u.name,size:Lt(u.size),sizeBytes:u.size,status:"待发送",tone:"muted",progress:0,pending:!0,transferId:""}));a.length&&p.value.push(...a)}async function J(o){const a=p.value.find(u=>u.id===o);if(!(!a||a.kind!=="file"||!a.pending)){if(!h.value.deviceId){window.alert("当前没有可用的接收端");return}a.pending=!1,a.status="创建传输中...",a.tone="primary";try{const u=await ge.create({kind:"file",name:a.name,size_bytes:a.sizeBytes,sender_device_id:E.value.id,receiver_device_id:h.value.deviceId});if(a.transferId=u.id,a.sizeBytes>Qu){await me(a,u);return}ce("transfer.created",h.value.deviceId,{transfer_id:u.id,kind:"file",name:a.name,size_bytes:a.sizeBytes,sender_device_id:E.value.id,sender_name:E.value.name,sender_type:E.value.type,receiver_device_id:h.value.deviceId,final_status:"connecting",current_channel:"p2p",transport_options:os()});try{await rr(a,u)}catch(g){console.warn("realtime file send failed, fallback to relay",g),await ir(a,u)}}catch(u){a.pending=!0,a.status=`发送失败:${u.message}`,a.tone="danger"}}}async function me(o,a){o.progress=0,o.status="上传准备中...";try{if(!a.fallback_allowed)throw new Error("当前文件过大,且未启用 MinIO 回退");await ge.presignFallback(o.transferId),ce("transfer.updated",h.value.deviceId,{transfer_id:o.transferId,final_status:"fallback_uploading",current_channel:"minio"}),o.status="上传中...";const u=await ge.uploadFallback(o.transferId,o.file,g=>{o.progress=Math.max(1,Math.min(g,99))});await ge.updateStatus(o.transferId,{current_channel:"minio",final_status:"completed"}),ce("transfer.updated",h.value.deviceId,{transfer_id:o.transferId,final_status:"completed",current_channel:"minio"}),ce("transfer.file",h.value.deviceId,{transfer_id:o.transferId,name:o.name,download_url:u.download_path||u.download_url}),o.progress=100,o.status="上传完成",o.tone="success"}catch(u){o.pending=!0,o.status=`上传失败:${u.message}`,o.tone="danger"}}async function we(){for(const o of In.value)await J(o.id)}async function Ne(o){const a=p.value.find(u=>u.id===o);if(a&&Pe(a),p.value=p.value.filter(u=>u.id!==o),!(!(a!=null&&a.transferId)||a.tone==="success"))try{await ge.updateStatus(a.transferId,{final_status:"cancelled"}),ce("transfer.updated",h.value.deviceId,{transfer_id:a.transferId,final_status:"cancelled"})}catch(u){console.error(u)}}async function Fe(o){const a=p.value.find(u=>u.id===o);if(!(!a||a.kind!=="text"))try{await kt(a.text),a.copied=!0,window.setTimeout(()=>{const u=p.value.find(g=>g.id===o);u&&u.kind==="text"&&(u.copied=!1)},2e3)}catch{window.alert("复制失败")}}async function kt(o){var u;if((u=navigator.clipboard)!=null&&u.writeText&&window.isSecureContext){await navigator.clipboard.writeText(o);return}const a=document.createElement("textarea");a.value=o,a.setAttribute("readonly","readonly"),a.style.position="fixed",a.style.top="0",a.style.left="-9999px",a.style.opacity="0",document.body.appendChild(a),a.focus(),a.select(),a.setSelectionRange(0,a.value.length);try{if(!document.execCommand("copy"))throw new Error("copy command failed")}finally{document.body.removeChild(a)}}function Tn(o){const a=le.get(o);a&&(window.clearInterval(a),le.delete(o))}function $e(o){return new Promise((a,u)=>{const g=new FileReader;g.onload=()=>a(String(g.result||"")),g.onerror=()=>u(new Error("Failed to read file")),g.readAsDataURL(o)})}function Pe(o){if(Tn(o.id),o.ownedDownloadUrl&&o.downloadUrl)try{URL.revokeObjectURL(o.downloadUrl)}catch(a){console.error(a)}o.transferId&&he.delete(o.transferId)}function Ut(o,a,u=!1){if(o.ownedDownloadUrl&&o.downloadUrl&&o.downloadUrl!==a)try{URL.revokeObjectURL(o.downloadUrl)}catch(g){console.error(g)}o.downloadUrl=a,o.ownedDownloadUrl=u}async function sn(o,a){const u=await fr(h.value.deviceId);kn(u,{type:"text",transfer_id:o.id,text:a,sender_device_id:E.value.id,sender_name:E.value.name,sender_type:E.value.type});const g=mr();await ge.updateStatus(o.id,{current_channel:g,final_status:"completed"})}async function sr(o,a){ce("transfer.created",h.value.deviceId,{transfer_id:o.id,kind:"text",name:"text-message",content:a,sender_device_id:E.value.id,sender_name:E.value.name,sender_type:E.value.type,receiver_device_id:h.value.deviceId,final_status:"completed",current_channel:"p2p"}),await ge.updateStatus(o.id,{current_channel:"p2p",final_status:"completed"})}async function rr(o,a){var A;const u=await fr(h.value.deviceId);o.status="正在通过 WebRTC 发送...",o.progress=1,kn(u,{type:"file-meta",transfer_id:a.id,name:o.name,mime_type:((A=o.file)==null?void 0:A.type)||"application/octet-stream",size_bytes:o.sizeBytes,sender_device_id:E.value.id,sender_name:E.value.name,sender_type:E.value.type});let g=0;for(;gai;)await So(20)}function So(o){return new Promise(a=>{window.setTimeout(a,o)})}function Co(o,a,u){return new Promise((g,k)=>{const A=window.setTimeout(()=>{k(new Error(u))},a);o.then(se=>{window.clearTimeout(A),g(se)}).catch(se=>{window.clearTimeout(A),k(se)})})}async function To(){const o=window.prompt("管理员用户名","admin");if(o===null)return;const a=window.prompt("管理员密码");if(a!==null)try{const u=await Ht.login(o.trim()||"admin",a);z.value=u.token,localStorage.setItem(Is,u.token),await is(),n.value="admin"}catch(u){window.alert(`管理员登录失败:${u.message}`)}}function ko(){n.value="main"}async function is(){if(z.value)try{const[o,a,u]=await Promise.all([Ht.stats(z.value),Ht.config(z.value),Ht.recentTransfers(z.value)]);Cn(a),U.value=wr(o.stats||{},o.minio||{}),N.value=u.map(g=>Jo(g))}catch(o){throw(o==null?void 0:o.status)===401&&(localStorage.removeItem(Is),localStorage.removeItem(Fn),z.value="",n.value="main"),o}}async function $o(){if(!z.value||!Y.value){window.alert("当前没有可用的管理员会话");return}try{const o={...Y.value,max_minio_fallback_size_bytes:Math.max(0,T.value)*1024*1024,minio_capacity_bytes:Math.max(0,S.value)*1024*1024*1024},a=await Ht.updateConfig(z.value,o);Cn(a);{const u=await Ht.stats(z.value);U.value=wr(u.stats||{},u.minio||{})}window.alert("配置已保存")}catch(o){window.alert(`保存配置失败:${o.message}`)}}function Mo(){let o=localStorage.getItem(Un);return o||(o=typeof crypto<"u"&&crypto.randomUUID?crypto.randomUUID():`web-${Date.now()}-${Math.random().toString(36).slice(2,8)}`,localStorage.setItem(Un,o)),o}function Ro(o){var u;let a=localStorage.getItem(li);return a||(a=`${((u=navigator.userAgentData)==null?void 0:u.platform)||navigator.platform||"Web"} ${o.slice(0,4)}`,localStorage.setItem(li,a)),a}function Eo(){const o=`${navigator.userAgent} ${navigator.platform}`.toLowerCase();return o.includes("iphone")||o.includes("android")||o.includes("mobile")?"phone":o.includes("ipad")||o.includes("tablet")?"tablet":"desktop"}function Ao(o){return o==="phone"?"smartphone":o==="tablet"?"tablet_mac":"laptop_mac"}function Bt(o){return o==="phone"?"手机":o==="tablet"?"平板":"桌面端"}function Oo(){return`${window.location.protocol==="https:"?"wss:":"ws:"}//${window.location.host}/ws`}function Po(){var o;return Array.isArray((o=Y.value)==null?void 0:o.turn_urls)&&Y.value.turn_urls.some(a=>String(a||"").trim())}function or(){const o=String(window.location.hostname||"").trim().toLowerCase();return o?o==="localhost"||o==="127.0.0.1"||o==="::1"||o.endsWith(".local")||Do(o)?o:"":"local"}function lr(o){const a=or();return!!a&&o===a}function Do(o){const a=o.split(".");if(a.length!==4||a.some(k=>!/^\d+$/.test(k)))return!1;const[u,g]=a.map(k=>Number(k));return u===10||u===127||u===192&&g===168?!0:u===172&&g>=16&&g<=31}function ar(o){const a=Array.isArray(o==null?void 0:o.turn_urls)?o.turn_urls.map(u=>String(u||"").trim()).filter(Boolean):[];return a.length?[{urls:a,username:(o==null?void 0:o.turn_username)||"",credential:(o==null?void 0:o.turn_password)||""}]:[]}function os(){var o,a;return{ice_servers:ar(Y.value),p2p_connect_timeout_sec:((o=Y.value)==null?void 0:o.p2p_connect_timeout_sec)||15,turn_connect_timeout_sec:((a=Y.value)==null?void 0:a.turn_connect_timeout_sec)||20}}function cr(){return typeof RTCPeerConnection<"u"}function ls(){_t=null,St=null}function No(){return _t||(_t=new Promise(o=>{St=o})),_t}function ur(o){St&&St(o),_t=Promise.resolve(o),St=null}function We(o=""){if(!h.value.deviceId)return;const a=h.value.baseType||h.value.type||"点对点传输";h.value={...h.value,type:o?`${a} · ${o}`:a}}function Fo(o){return!B||G!==o||B.signalingState==="closed"||["failed","disconnected","closed"].includes(B.connectionState)||["failed","disconnected","closed"].includes(B.iceConnectionState)?!0:!L||L.readyState==="closed"}async function rt(o,a={}){return!o||!cr()?null:(Fo(o)&&(it(),Uo(o)),a.initiate&&B.signalingState==="stable"&&await Bo(o),B)}function Uo(o){G=o,_e="p2p",ze=!1,be=!1,ue=!1,ke.delete(o),ls(),B=new RTCPeerConnection({iceServers:ar(Y.value)}),L=B.createDataChannel("filefast-control",{negotiated:!0,id:0,ordered:!0}),Lo(L),B.onicecandidate=({candidate:a})=>{if(a)try{ce("webrtc.candidate",o,{candidate:a})}catch(u){console.error(u)}},B.onconnectionstatechange=()=>{if(B){if(as(),B.connectionState==="connected"){We(_e==="turn"?"TURN 中继已连接":"WebRTC 直连已连接");return}if(B.connectionState==="connecting"){We("实时通道连接中");return}if(B.connectionState==="failed"){We("实时通道连接失败");return}(B.connectionState==="disconnected"||B.connectionState==="closed")&&We("实时通道已断开")}},B.oniceconnectionstatechange=()=>{as()}}async function Bo(o){if(B)try{ze=!0,await B.setLocalDescription(),ce("webrtc.description",o,{description:B.localDescription})}finally{ze=!1}}function Lo(o){L=o,o.bufferedAmountLowThreshold=ai/2,o.onopen=()=>{ur(o),We(_e==="turn"?"TURN 中继已连接":"WebRTC 直连已连接"),as()},o.onclose=()=>{L===o&&(L=null,ls(),We("实时通道已关闭"))},o.onerror=a=>{console.error(a)},o.onmessage=a=>{jo(a.data)},o.readyState==="open"&&ur(o)}function it(){he.clear(),G&&ke.delete(G),L&&(L.onopen=null,L.onclose=null,L.onerror=null,L.onmessage=null,L.close(),L=null),B&&(B.onicecandidate=null,B.onconnectionstatechange=null,B.oniceconnectionstatechange=null,B.close(),B=null),G="",_e="p2p",ze=!1,be=!1,ue=!1,ls()}async function fr(o){if(!cr())throw new Error("当前浏览器不支持 WebRTC");if(await rt(o,{initiate:!0}),(L==null?void 0:L.readyState)==="open")return L;const a=await Co(No(),ef,"WebRTC 连接超时");if(!a||a.readyState!=="open")throw new Error("实时通道未建立");return a}function kn(o,a){if(!o||o.readyState!=="open")throw new Error("实时通道未就绪");o.send(JSON.stringify(a))}function jo(o){try{const a=JSON.parse(String(o||"{}"));if(a.type==="text"){Ho(a);return}if(a.type==="file-meta"){Ko(a);return}if(a.type==="file-chunk"){zo(a);return}a.type==="file-complete"&&Wo(a)}catch(a){console.error(a)}}function Ho(o){var k;const a=o.sender_device_id||G,u={id:a,name:o.sender_name||((k=$t(a))==null?void 0:k.name)||`设备 ${Mt(a)}`,type:Bt(o.sender_type||"desktop"),connectionType:_e==="turn"?"TURN 中继已连接":"WebRTC 直连已连接"};M(u,!0);const g=p.value.find(A=>A.transferId===o.transfer_id);if(g&&g.kind==="text"){g.text=o.text||"",g.status="已接收",g.tone="success";return}p.value.push({id:Ue("incoming-text"),transferId:o.transfer_id,kind:"text",text:o.text||"",status:"已接收",tone:"success",copied:!1})}function Ko(o){var k;const a=o.sender_device_id||G,u={id:a,name:o.sender_name||((k=$t(a))==null?void 0:k.name)||`设备 ${Mt(a)}`,type:Bt(o.sender_type||"desktop"),connectionType:_e==="turn"?"TURN 中继已连接":"WebRTC 直连已连接"};M(u,!0),he.set(o.transfer_id,{name:o.name||"file",mimeType:o.mime_type||"application/octet-stream",sizeBytes:Number(o.size_bytes||0),receivedBytes:0,chunks:[]});let g=p.value.find(A=>A.transferId===o.transfer_id);g?(g.status="正在接收...",g.tone="primary",g.progress=0):(g={id:Ue("incoming-file"),transferId:o.transfer_id,kind:"file",name:o.name||"file",size:Lt(Number(o.size_bytes||0)),sizeBytes:Number(o.size_bytes||0),status:"正在接收...",tone:"primary",progress:0,pending:!1,downloadUrl:"",ownedDownloadUrl:!1},p.value.push(g))}function zo(o){const a=he.get(o.transfer_id);if(!a)return;const u=xo(String(o.chunk_base64||""));a.receivedBytes+=Number(o.chunk_size||u.byteLength||0),a.chunks.push(u);const g=p.value.find(k=>k.transferId===o.transfer_id);if(g){const k=a.sizeBytes>0?a.receivedBytes/a.sizeBytes*100:0;g.progress=Math.max(1,Math.min(99,Math.round(k))),g.status="正在接收...",g.tone="primary"}}function Wo(o){const a=he.get(o.transfer_id);if(!a)return;const u=p.value.find(A=>A.transferId===o.transfer_id);if(!u){he.delete(o.transfer_id);return}const g=new Blob(a.chunks,{type:a.mimeType||"application/octet-stream"}),k=URL.createObjectURL(g);Ut(u,k,!0),u.progress=100,u.status="可保存",u.tone="success",he.delete(o.transfer_id)}function dr(o){return E.value.id.localeCompare(o)>0}function $n(o,a="等待实时数据"){const u=$t(o);return{id:o,deviceId:o,name:(u==null?void 0:u.name)||`设备 ${Mt(o)}`,type:Bt((u==null?void 0:u.type)||"desktop"),connectionType:a,network_group_key:(u==null?void 0:u.network_group_key)||""}}async function pr(o){const u=(o.payload||{}).description,g=o.device_id||"";if(!u||!g)return;M($n(g),!0);const k=await rt(g);if(!k)return;const A=dr(g),se=!ze&&(k.signalingState==="stable"||ue),Rt=u.type==="offer"&&!se;be=!A&&Rt,!be&&(ue=u.type==="answer",await k.setRemoteDescription(u),ue=!1,u.type==="offer"&&(await k.setLocalDescription(),ce("webrtc.description",g,{description:k.localDescription})))}async function hr(o){const a=o.payload||{},u=o.device_id||"";if(!a.candidate||!u)return;(n.value!=="transfer"||h.value.deviceId!==u)&&M($n(u),!0);const g=await rt(u);if(g)try{await g.addIceCandidate(a.candidate)}catch(k){be||console.error(k)}}async function as(){if(!(!B||B.connectionState!=="connected"))try{const o=await B.getStats();let a=null;if(o.forEach(A=>{A.type==="transport"&&A.selectedCandidatePairId&&(a=o.get(A.selectedCandidatePairId)||a)}),a||o.forEach(A=>{A.type==="candidate-pair"&&A.state==="succeeded"&&(A.nominated||A.selected)&&(a=A)}),!a)return;const u=o.get(a.localCandidateId),g=o.get(a.remoteCandidateId),k=(u==null?void 0:u.candidateType)==="relay"||(g==null?void 0:g.candidateType)==="relay";_e=k?"turn":"p2p",(L==null?void 0:L.readyState)==="open"&&We(k?"TURN 中继已连接":"WebRTC 直连已连接")}catch(o){console.error(o)}}function mr(){return _e==="turn"?"turn":"p2p"}function gr(){if(!E.value.id)return;const o=localStorage.getItem(Ss)||"";o&&(vr(),de=new WebSocket(`${_.value}?deviceId=${encodeURIComponent(E.value.id)}&deviceToken=${encodeURIComponent(o)}`),de.addEventListener("message",a=>{qo(a.data)}),de.addEventListener("close",()=>{de=null,Vo()}),de.addEventListener("error",()=>{de==null||de.close()}))}function vr(){if(st&&(window.clearTimeout(st),st=null),!de)return;const o=de;de=null,o.onclose=null,o.close()}function Vo(){st||!E.value.id||(st=window.setTimeout(()=>{st=null,gr()},Zu))}function ce(o,a,u){!de||de.readyState!==WebSocket.OPEN||!a||de.send(JSON.stringify({type:o,target_device_id:a,payload:u}))}function qo(o){try{const a=JSON.parse(o);if(a.type==="presence.update"){v();return}if(a.type==="webrtc.description"){pr(a);return}if(a.type==="webrtc.candidate"){hr(a);return}if(a.type==="transfer.created"){yr(a);return}if(a.type==="transfer.updated"){_r(a);return}if(a.type==="transfer.file"){br(a);return}a.type==="peer.session.closed"&&Go(a)}catch(a){console.error(a)}}function Go(o){const a=o.device_id||"";!a||h.value.deviceId!==a||(it(),D(),h.value={name:"--",type:"绛夊緟杩炴帴",baseType:"绛夊緟杩炴帴",deviceId:"",networkGroupKey:""},n.value="main")}function yr(o){var A;const a=o.payload||{},u=o.device_id||a.sender_device_id||"",g={id:u,name:a.sender_name||((A=$t(u))==null?void 0:A.name)||`Device ${Mt(u)}`,type:Bt(a.sender_type||"desktop")};if(g.connectionType="等待实时数据",M(g,!0),!p.value.find(se=>se.transferId===a.transfer_id)){if(a.kind==="text"){p.value.push({id:Ue("incoming-text"),transferId:a.transfer_id,kind:"text",text:a.content||"",status:"已接收",tone:"success",copied:!1});return}p.value.push({id:Ue("incoming-file"),transferId:a.transfer_id,kind:"file",name:a.name||"file",size:Lt(Number(a.size_bytes||0)),sizeBytes:Number(a.size_bytes||0),status:"接收中...",tone:"primary",progress:35,pending:!1,downloadUrl:"",ownedDownloadUrl:!1})}}function _r(o){const a=o.payload||{},u=p.value.find(g=>g.transferId===a.transfer_id);if(u&&u.kind==="file"){if(a.final_status==="completed"){u.progress=100,u.status="已接收",u.tone="success",u.downloadUrl&&(u.status="可保存");return}a.final_status==="cancelled"&&(u.status="已取消",u.tone="danger")}}function br(o){const a=o.payload||{};let u=p.value.find(g=>g.transferId===a.transfer_id);!u&&a.transfer_id&&(u={id:Ue("incoming-file"),transferId:a.transfer_id,kind:"file",name:a.name||"file",size:"",sizeBytes:0,status:"可保存",tone:"success",progress:100,pending:!1,downloadUrl:"",ownedDownloadUrl:!1},p.value.push(u)),!(!u||u.kind!=="file")&&(Ut(u,a.download_url||a.data_url||"",!1),u.status="可保存",u.progress=100,u.tone="success")}function $t(o){return r.value.find(a=>a.id===o)}function wr(o,a={}){return[{label:"在线设备",value:`${o.devices_online||0}`,tone:"blue"},{label:"待加入房间",value:`${o.rooms_waiting||0}`,tone:"cyan"},{label:"有效传输",value:`${o.transfers_total||0}`,tone:"default"},{label:"累计传输",value:`${o.transfers_cumulative||0}`,tone:"default"},{kind:"minio",label:"MinIO 剩余容量",value:cs(a.remaining_bytes||0),tone:Number(a.usage_percent||0)>=85?"danger":Number(a.usage_percent||0)>=60?"cyan":"blue",percent:Math.max(0,100-Number(a.usage_percent||0)),detail:`已用 ${cs(a.used_bytes||0)} / 总计 ${cs(a.capacity_bytes||0)}`,kicker:`存档 ${a.object_count||0} 份`}]}function cs(o){const a=Number(o||0);if(!a||a<=0)return"0 GB";const u=["B","KB","MB","GB","TB"],g=Math.min(Math.floor(Math.log(a)/Math.log(1024)),u.length-1),k=a/1024**g,A=g>=3?2:k>=10?1:2;return`${k.toFixed(A)} ${u[g]}`}function Jo(o){const a=o.final_status==="completed",u=o.final_status==="failed"||o.final_status==="cancelled";return{time:us(o.created_at),peer:`${Mt(o.sender_device_id)} -> ${Mt(o.receiver_device_id)}`,type:o.kind==="text"?"文本消息":`文件 ${o.name}`,size:Lt(Number(o.size_bytes||0)),status:a?`已完成 (${o.current_channel||"p2p"})`:u?`已结束 (${o.final_status})`:`进行中 (${o.final_status||"pending"})`,tone:a?"success":u?"danger":"primary"}}function Mt(o){return o?o.slice(0,8):"--"}function us(o){if(!o)return"刚刚";const a=new Date(o),u=Date.now()-a.getTime();if(!Number.isFinite(u))return"刚刚";const g=Math.max(0,Math.floor(u/1e3));if(g<60)return`${g} 秒前`;const k=Math.floor(g/60);if(k<60)return`${k} 分钟前`;const A=Math.floor(k/60);return A<24?`${A} 小时前`:`${Math.floor(A/24)} 天前`}function Ue(o){return`${o}-${Date.now()}-${Math.random().toString(36).slice(2,8)}`}function Lt(o){if(!o||o<=0)return"0 B";const a=["B","KB","MB","GB","TB"],u=Math.min(Math.floor(Math.log(o)/Math.log(1024)),a.length-1),g=o/1024**u,k=g>=10||u===0?0:1;return`${g.toFixed(k)} ${a[u]}`}v=async function(){return E.value.id?Dn.listCandidates(E.value.id).then(a=>{r.value=a.map(u=>({...u,description:`${Bt(u.type)} · 最近活跃 ${us(u.last_seen_at)}`,icon:Ao(u.type),connectionType:lr(u.network_group_key)?"局域网直连优先":"跨网络实时传输"})),s.value=r.value.length===0}).catch(a=>{s.value=!1,console.error(a)}):Promise.resolve()},P=function(a){const u=a.deviceId||a.id||"",g=a.connectionType||a.type||"点对点传输";b(),h.value.deviceId!==u&&(it(),D()),h.value={name:a.name,type:g,baseType:g,deviceId:u,networkGroupKey:a.network_group_key||""},l.value=!1,c.value="----",n.value="transfer",We("正在建立实时通道"),rt(u,{initiate:!0})},M=function(a,u=!1){const g=a.deviceId||a.id||"",k=a.connectionType||a.type||"点对点传输";h.value.deviceId===g&&n.value==="transfer"||(it(),u||D()),h.value={name:a.name,type:k,baseType:k,deviceId:g,networkGroupKey:a.network_group_key||""},l.value=!1,c.value="----",n.value="transfer",g&&(We("正在建立实时通道"),rt(g))},O=function(){h.value.deviceId&&ce("peer.session.closed",h.value.deviceId,{}),it(),D(),h.value={name:"--",type:"等待连接",baseType:"等待连接",deviceId:"",networkGroupKey:""},n.value="main"},W=async function(a){const u=a.trim();if(u){if(!h.value.deviceId){window.alert("当前没有可用的接收端");return}try{const g=await ge.create({kind:"text",name:"text-message",content:u,sender_device_id:E.value.id,receiver_device_id:h.value.deviceId});try{await sn(g,u)}catch(k){console.warn("realtime text send failed, fallback to relay",k),await sr(g,u)}p.value.push({id:Ue("text"),transferId:g.id,kind:"text",text:u,status:"已发送",tone:"success",copied:!1})}catch(g){window.alert(`发送文本失败:${g.message}`)}}};function xr(o,a){ce("transfer.file",h.value.deviceId,{transfer_id:o.transferId,name:o.name,download_url:a.download_path||a.download_url})}function Ir(o,a,{onProgress:u}={}){if(!(o!=null&&o.file))return Promise.reject(new Error("未找到待上传文件"));if(!(a!=null&&a.fallback_allowed))return Promise.reject(new Error("MinIO 存档未启用"));const g=a.id;if(Te.has(g))return Te.get(g);const k=(async()=>(await ge.presignFallback(g),ge.uploadFallback(g,o.file,A=>{typeof u=="function"&&u(A)})))().finally(()=>{Te.delete(g)});return Te.set(g,k),k}async function Sr(o,a,u){await ge.updateStatus(a.id,{current_channel:"minio",final_status:"completed"}),ce("transfer.updated",h.value.deviceId,{transfer_id:a.id,final_status:"completed",current_channel:"minio"}),xr(o,u),o.progress=100,o.status="已上传到 MinIO,对方可直接领取",o.tone="success"}async function Yo(o,a){const u=ke.get(o);if(!(!(u!=null&&u.length)||!(a!=null&&a.remoteDescription))){ke.delete(o);for(const g of u)try{await a.addIceCandidate(g)}catch(k){console.error(k)}}}return me=async function(a,u){a.progress=Math.max(5,a.progress||0),a.status="正在切换到 MinIO...",a.tone="primary";try{ce("transfer.updated",h.value.deviceId,{transfer_id:a.transferId,final_status:"fallback_uploading",current_channel:"minio"});const g=await Ir(a,u,{onProgress:k=>{a.progress=Math.max(5,Math.min(k,99))}});await Sr(a,u,g)}catch(g){a.pending=!0,a.status=`上传失败:${g.message}`,a.tone="danger"}},pr=async function(a){const g=(a.payload||{}).description,k=a.device_id||"";if(!g||!k)return;M($n(k),!0);const A=await rt(k);if(!A)return;const se=dr(k),Rt=!ze&&(A.signalingState==="stable"||ue),Xo=g.type==="offer"&&!Rt;if(be=!se&&Xo,!be&&!(g.type==="answer"&&(A.signalingState!=="have-local-offer"||!A.localDescription))){try{ue=g.type==="answer",await A.setRemoteDescription(g),await Yo(k,A)}catch(fs){console.error(fs)}finally{ue=!1}if(g.type==="offer")try{await A.setLocalDescription(),ce("webrtc.description",k,{description:A.localDescription})}catch(fs){console.error(fs)}}},hr=async function(a){const u=a.payload||{},g=a.device_id||"",k=u.candidate;if(!k||!g)return;(n.value!=="transfer"||h.value.deviceId!==g)&&M($n(g),!0);const A=await rt(g);if(A){if(!A.remoteDescription){const se=ke.get(g)||[];se.push(k),ke.set(g,se);return}try{await A.addIceCandidate(k)}catch(se){be||console.error(se)}}},J=async function(a){const u=p.value.find(g=>g.id===a);if(!(!u||u.kind!=="file"||!u.pending)){if(!h.value.deviceId){window.alert("当前没有可用的接收端");return}u.pending=!1,u.status="创建传输中...",u.tone="primary";try{const g=await ge.create({kind:"file",name:u.name,size_bytes:u.sizeBytes,sender_device_id:E.value.id,receiver_device_id:h.value.deviceId});u.transferId=g.id;const k=g.fallback_allowed?Ir(u,g).catch(A=>{throw console.warn("minio backup sync failed",A),A}):Promise.resolve(null);ce("transfer.created",h.value.deviceId,{transfer_id:g.id,kind:"file",name:u.name,size_bytes:u.sizeBytes,sender_device_id:E.value.id,sender_name:E.value.name,sender_type:E.value.type,receiver_device_id:h.value.deviceId,final_status:"connecting",current_channel:"p2p",transport_options:os()});try{if(await rr(u,g),g.fallback_allowed){u.status="实时传输完成,正在同步云端备份...",u.tone="primary";try{const A=await k;A&&(xr(u,A),u.status="已发送,2 小时内可离线领取")}catch(A){u.status=`实时传输成功,但 MinIO 备份失败:${A.message}`,u.tone="danger";return}u.tone="success"}}catch(A){console.warn("realtime file send failed, fallback to minio",A);try{const se=await k;if(se){await Sr(u,g,se);return}}catch(se){console.warn("minio backup sync failed after realtime failure",se)}await ir(u,g)}}catch(g){u.pending=!0,u.status=`发送失败:${g.message}`,u.tone="danger"}}},yr=function(a){var se;const u=a.payload||{},g=a.device_id||u.sender_device_id||"",k={id:g,name:u.sender_name||((se=$t(g))==null?void 0:se.name)||`设备 ${Mt(g)}`,type:Bt(u.sender_type||"desktop"),connectionType:"等待实时数据"};if(M(k,!0),!p.value.find(Rt=>Rt.transferId===u.transfer_id)){if(u.kind==="text"){u.content&&p.value.push({id:Ue("incoming-text"),transferId:u.transfer_id,kind:"text",text:u.content||"",status:"已接收",tone:"success",copied:!1});return}p.value.push({id:Ue("incoming-file"),transferId:u.transfer_id,kind:"file",name:u.name||"file",size:Lt(Number(u.size_bytes||0)),sizeBytes:Number(u.size_bytes||0),status:"等待接收...",tone:"primary",progress:5,pending:!1,downloadUrl:"",ownedDownloadUrl:!1})}},_r=function(a){const u=a.payload||{},g=p.value.find(k=>k.transferId===u.transfer_id);if(g&&g.kind==="file"){if(u.final_status==="completed"){g.progress=100,g.status=g.downloadUrl?"可保存":"传输完成",g.tone="success";return}if(u.final_status==="cancelled"){g.status="已取消",g.tone="danger";return}u.final_status==="fallback_uploading"&&(g.status="发送端正在上传回退文件...",g.tone="primary")}},br=function(a){const u=a.payload||{};let g=p.value.find(k=>k.transferId===u.transfer_id);!g&&u.transfer_id&&(g={id:Ue("incoming-file"),transferId:u.transfer_id,kind:"file",name:u.name||"file",size:"",sizeBytes:0,status:"可保存",tone:"success",progress:100,pending:!1,downloadUrl:"",ownedDownloadUrl:!1},p.value.push(g)),!(!g||g.kind!=="file")&&(Ut(g,u.download_url||u.data_url||"",!1),g.status="可保存",g.progress=100,g.tone="success")},(o,a)=>(j(),V("div",null,[y("div",qu,[q(Rc,{theme:t.value,onToggleTheme:Sn},null,8,["theme"]),n.value==="main"?(j(),V("div",Gu,[q(fu,{devices:r.value,"is-scanning":s.value,"local-device-name":E.value.name,onSelectDevice:P},null,8,["devices","is-scanning","local-device-name"]),q(Iu,{"generated-code":c.value,"is-waiting":l.value,"pending-downloads":f.value,"room-code-input":i.value,onCancelRoom:R,onCreateRoom:x,onJoinRoom:$,onUpdateRoomCode:d},null,8,["generated-code","is-waiting","pending-downloads","room-code-input"])])):Le("",!0),n.value==="transfer"?(j(),mt(Nu,{key:1,"connection-type":h.value.type,"has-pending-items":Ct.value,items:p.value,"network-hint":Ft.value,"peer-name":h.value.name,onClose:O,onCopyItem:Fe,onFilesSelected:ee,onRemoveItem:Ne,onSendAllPending:we,onSendText:W,onStartUpload:J},null,8,["connection-type","has-pending-items","items","network-hint","peer-name"])):Le("",!0),n.value==="admin"?(j(),mt(tu,{key:2,"file-limit":T.value,"minio-capacity":S.value,records:N.value,stats:U.value,onExit:ko,onSaveConfig:$o,"onUpdate:fileLimit":a[0]||(a[0]=u=>T.value=u),"onUpdate:minioCapacity":a[1]||(a[1]=u=>S.value=u)},null,8,["file-limit","minio-capacity","records","stats"])):Le("",!0)]),q(kc,{onRequestAdmin:To})]))}};Ic(nf).mount("#app"); diff --git a/frontend/dist/index.html b/frontend/dist/index.html index 6aa2830..6c8ad7f 100644 --- a/frontend/dist/index.html +++ b/frontend/dist/index.html @@ -4,8 +4,8 @@ AirShare Pro - - + +
diff --git a/frontend/src/App.vue b/frontend/src/App.vue index f3a9cb7..fcb53bd 100644 --- a/frontend/src/App.vue +++ b/frontend/src/App.vue @@ -35,6 +35,7 @@ const connectedPeer = ref({ name: '--', type: '等待连接', deviceId: '', + networkGroupKey: '', }) const transferItems = ref([]) const relayServer = ref('/ws') @@ -82,6 +83,17 @@ const pendingTransferItems = computed(() => ) const hasPendingItems = computed(() => pendingTransferItems.value.length > 0) +const currentTransferNetworkHint = computed(() => { + if (!connectedPeer.value.deviceId) { + return '' + } + + if (isSameLocalNetwork(connectedPeer.value.networkGroupKey) || hasTurnRelayConfigured()) { + return '' + } + + return '当前是跨网络访问,未配置 TURN 时实时通道可能失败。文本和小文件可回退中转,大文件建议使用 MinIO。' +}) watch( theme, @@ -208,7 +220,7 @@ async function registerCurrentDevice() { device_id: deviceId, name: deviceName, type: deviceType, - network_group_key: window.location.hostname || 'local', + network_group_key: deriveNetworkGroupKey(), }) localStorage.setItem(DEVICE_ID_KEY, device.id) @@ -269,7 +281,7 @@ async function refreshCandidates() { description: `${formatDeviceType(device.type)} · 最近活跃 ${formatRelativeTime(device.last_seen_at)}`, icon: mapDeviceIcon(device.type), connectionType: - device.network_group_key && device.network_group_key === window.location.hostname + isSameLocalNetwork(device.network_group_key) ? '局域网直连优先' : '点对点传输', })) @@ -412,6 +424,7 @@ function connectToPeer(device) { name: device.name, type: device.connectionType || device.type || '点对点传输', deviceId: device.deviceId || device.id || '', + networkGroupKey: device.network_group_key || '', } isWaitingRoom.value = false generatedRoomCode.value = '----' @@ -439,6 +452,7 @@ function ensurePeerSession(device, preserveItems = false) { name: device.name, type: device.connectionType || device.type || '点对点传输', deviceId: nextDeviceId, + networkGroupKey: device.network_group_key || '', } isWaitingRoom.value = false generatedRoomCode.value = '----' @@ -457,6 +471,7 @@ function closeTransfer() { name: '--', type: '等待连接', deviceId: '', + networkGroupKey: '', } viewMode.value = 'main' connectedPeer.value.baseType = '等待连接' @@ -733,7 +748,7 @@ async function copyTextItem(id) { } try { - await navigator.clipboard.writeText(item.text) + await copyToClipboard(item.text) item.copied = true window.setTimeout(() => { @@ -748,6 +763,33 @@ async function copyTextItem(id) { } } +async function copyToClipboard(value) { + if (navigator.clipboard?.writeText && window.isSecureContext) { + await navigator.clipboard.writeText(value) + return + } + + const textarea = document.createElement('textarea') + textarea.value = value + textarea.setAttribute('readonly', 'readonly') + textarea.style.position = 'fixed' + textarea.style.top = '0' + textarea.style.left = '-9999px' + textarea.style.opacity = '0' + document.body.appendChild(textarea) + textarea.focus() + textarea.select() + textarea.setSelectionRange(0, textarea.value.length) + + try { + if (!document.execCommand('copy')) { + throw new Error('copy command failed') + } + } finally { + document.body.removeChild(textarea) + } +} + function simulateUpload(id) { clearUploadTimer(id) @@ -1169,6 +1211,48 @@ function buildRelayServerPath() { return `${protocol}//${window.location.host}/ws` } +function hasTurnRelayConfigured() { + return Array.isArray(runtimeConfig.value?.turn_urls) && runtimeConfig.value.turn_urls.some((item) => String(item || '').trim()) +} + +function deriveNetworkGroupKey() { + const hostname = String(window.location.hostname || '').trim().toLowerCase() + if (!hostname) { + return 'local' + } + + if (hostname === 'localhost' || hostname === '127.0.0.1' || hostname === '::1' || hostname.endsWith('.local')) { + return hostname + } + + if (isPrivateIPv4Host(hostname)) { + return hostname + } + + return '' +} + +function isSameLocalNetwork(networkGroupKey) { + const localNetworkGroupKey = deriveNetworkGroupKey() + return !!localNetworkGroupKey && networkGroupKey === localNetworkGroupKey +} + +function isPrivateIPv4Host(hostname) { + const parts = hostname.split('.') + if (parts.length !== 4 || parts.some((part) => !/^\d+$/.test(part))) { + return false + } + + const [first, second] = parts.map((part) => Number(part)) + if (first === 10 || first === 127) { + return true + } + if (first === 192 && second === 168) { + return true + } + return first === 172 && second >= 16 && second <= 31 +} + function buildIceServers(configValue) { const urls = Array.isArray(configValue?.turn_urls) ? configValue.turn_urls.map((item) => String(item || '').trim()).filter(Boolean) @@ -1235,12 +1319,32 @@ function syncConnectionTypeLabel(status = '') { } } +function shouldRecreateRealtimeTransport(targetDeviceId) { + if (!peerConnection || currentRtcPeerId !== targetDeviceId) { + return true + } + + if (peerConnection.signalingState === 'closed') { + return true + } + + if (['failed', 'disconnected', 'closed'].includes(peerConnection.connectionState)) { + return true + } + + if (['failed', 'disconnected', 'closed'].includes(peerConnection.iceConnectionState)) { + return true + } + + return !controlChannel || controlChannel.readyState === 'closed' +} + async function ensureRealtimeTransport(targetDeviceId, options = {}) { if (!targetDeviceId || !supportsWebRTC()) { return null } - if (!peerConnection || currentRtcPeerId !== targetDeviceId) { + if (shouldRecreateRealtimeTransport(targetDeviceId)) { teardownRealtimeTransport() createPeerConnection(targetDeviceId) } @@ -1582,6 +1686,7 @@ function buildPeerSessionFromDevice(deviceId, fallbackConnectionType = '等待 name: knownDevice?.name || `设备 ${shortId(deviceId)}`, type: formatDeviceType(knownDevice?.type || 'desktop'), connectionType: fallbackConnectionType, + network_group_key: knownDevice?.network_group_key || '', } } @@ -1797,12 +1902,35 @@ function handleRelayMessage(raw) { if (envelope.type === 'transfer.file') { handleIncomingTransferFile(envelope) + return + } + + if (envelope.type === 'peer.session.closed') { + handlePeerSessionClosed(envelope) } } catch (error) { console.error(error) } } +function handlePeerSessionClosed(envelope) { + const senderDeviceId = envelope.device_id || '' + if (!senderDeviceId || connectedPeer.value.deviceId !== senderDeviceId) { + return + } + + teardownRealtimeTransport() + resetTransferItems() + connectedPeer.value = { + name: '--', + type: '绛夊緟杩炴帴', + baseType: '绛夊緟杩炴帴', + deviceId: '', + networkGroupKey: '', + } + viewMode.value = 'main' +} + function handleIncomingTransferCreated(envelope) { const payload = envelope.payload || {} const senderDeviceId = envelope.device_id || payload.sender_device_id || '' @@ -2178,7 +2306,7 @@ refreshCandidates = async function refreshCandidatesOverride() { description: `${formatDeviceType(device.type)} · 最近活跃 ${formatRelativeTime(device.last_seen_at)}`, icon: mapDeviceIcon(device.type), connectionType: - device.network_group_key && device.network_group_key === window.location.hostname + isSameLocalNetwork(device.network_group_key) ? '局域网直连优先' : '跨网络实时传输', })) @@ -2206,6 +2334,7 @@ connectToPeer = function connectToPeerOverride(device) { type: nextBaseType, baseType: nextBaseType, deviceId: nextDeviceId, + networkGroupKey: device.network_group_key || '', } isWaitingRoom.value = false generatedRoomCode.value = '----' @@ -2231,6 +2360,7 @@ ensurePeerSession = function ensurePeerSessionOverride(device, preserveItems = f type: nextBaseType, baseType: nextBaseType, deviceId: nextDeviceId, + networkGroupKey: device.network_group_key || '', } isWaitingRoom.value = false generatedRoomCode.value = '----' @@ -2243,6 +2373,10 @@ ensurePeerSession = function ensurePeerSessionOverride(device, preserveItems = f } closeTransfer = function closeTransferOverride() { + if (connectedPeer.value.deviceId) { + relayToPeer('peer.session.closed', connectedPeer.value.deviceId, {}) + } + teardownRealtimeTransport() resetTransferItems() connectedPeer.value = { @@ -2250,6 +2384,7 @@ closeTransfer = function closeTransferOverride() { type: '等待连接', baseType: '等待连接', deviceId: '', + networkGroupKey: '', } viewMode.value = 'main' } @@ -2709,6 +2844,7 @@ handleIncomingTransferFile = function handleIncomingTransferFileOverride(envelop :connection-type="connectedPeer.type" :has-pending-items="hasPendingItems" :items="transferItems" + :network-hint="currentTransferNetworkHint" :peer-name="connectedPeer.name" @close="closeTransfer" @copy-item="copyTextItem" diff --git a/frontend/src/components/TransferPanel.vue b/frontend/src/components/TransferPanel.vue index a334338..2ca57e9 100644 --- a/frontend/src/components/TransferPanel.vue +++ b/frontend/src/components/TransferPanel.vue @@ -12,6 +12,10 @@ const props = defineProps({ type: String, required: true, }, + networkHint: { + type: String, + default: '', + }, items: { type: Array, required: true, @@ -84,6 +88,7 @@ function handleDrop(event) {

{{ peerName }}

{{ connectionType }}

+ {{ networkHint }}