diff --git a/css/733.ce2542a6.css b/css/733.ce2542a6.css
new file mode 100644
index 0000000..e17518e
--- /dev/null
+++ b/css/733.ce2542a6.css
@@ -0,0 +1 @@
+.tm-viewport[data-v-5c291694]{width:100%;height:100%;display:flex;align-items:center;justify-content:center;touch-action:none;-webkit-user-select:none;-moz-user-select:none;user-select:none}.tm-media[data-v-5c291694]{max-width:100%;max-height:100%;-o-object-fit:contain;object-fit:contain;-webkit-user-drag:none}.public-browse[data-v-7aa9aaa3]{min-height:100vh;background:#0a0a0a;color:#fff}.header[data-v-7aa9aaa3]{position:sticky;top:0;z-index:100;display:flex;align-items:center;justify-content:space-between;padding:16px 32px;background:hsla(0,0%,6%,.95);backdrop-filter:blur(10px);border-bottom:1px solid #1a1a1a}.logo[data-v-7aa9aaa3]{font-size:20px;font-weight:600;color:#fff}.breadcrumb[data-v-7aa9aaa3]{display:flex;align-items:center;gap:4px;font-size:14px}.breadcrumb-item[data-v-7aa9aaa3]{padding:6px 12px;border-radius:6px;cursor:pointer;transition:background .2s;color:#ccc}.breadcrumb-item[data-v-7aa9aaa3]:hover{background:#252525;color:#fff}.breadcrumb-sep[data-v-7aa9aaa3]{color:#444}.file-count[data-v-7aa9aaa3]{color:#666;font-size:14px}.error-container[data-v-7aa9aaa3],.loading-container[data-v-7aa9aaa3]{display:flex;flex-direction:column;align-items:center;justify-content:center;min-height:60vh;color:#666}.loading-spinner[data-v-7aa9aaa3]{width:48px;height:48px;border:3px solid #222;border-top-color:#3b82f6;border-radius:50%;animation:spin-7aa9aaa3 1s linear infinite}.loading-spinner-small[data-v-7aa9aaa3]{width:24px;height:24px;border:2px solid #222;border-top-color:#3b82f6;border-radius:50%;animation:spin-7aa9aaa3 1s linear infinite}@keyframes spin-7aa9aaa3{to{transform:rotate(1turn)}}.retry-btn[data-v-7aa9aaa3]{margin-top:20px;padding:10px 32px;background:#3b82f6;border:none;border-radius:8px;color:#fff;font-size:14px;cursor:pointer;transition:background .2s}.retry-btn[data-v-7aa9aaa3]:hover{background:#2563eb}.gallery-container[data-v-7aa9aaa3]{padding:8px}@media (min-width:1200px){.gallery-container[data-v-7aa9aaa3]{max-width:1400px;margin:0 auto;padding:24px}}.folders-section[data-v-7aa9aaa3]{margin-bottom:24px}.folders-grid[data-v-7aa9aaa3]{display:grid;grid-template-columns:repeat(auto-fill,minmax(160px,1fr));gap:16px}.folder-card[data-v-7aa9aaa3]{display:flex;flex-direction:column;align-items:center;justify-content:center;padding:24px 16px;background:#141414;border-radius:12px;border:1px solid #1a1a1a;cursor:pointer;transition:all .2s}.folder-card[data-v-7aa9aaa3]:hover{background:#1a1a1a;border-color:#333;transform:translateY(-2px)}.folder-icon[data-v-7aa9aaa3]{width:48px;height:48px;margin-bottom:12px;color:#555}.folder-icon svg[data-v-7aa9aaa3]{width:100%;height:100%}.folder-name[data-v-7aa9aaa3]{font-size:14px;color:#999;text-align:center;word-break:break-all}.waterfall[data-v-7aa9aaa3]{display:flex;gap:16px}.waterfall-column[data-v-7aa9aaa3]{flex:1;display:flex;flex-direction:column;gap:16px}.waterfall-item[data-v-7aa9aaa3]{cursor:pointer}.image-wrapper[data-v-7aa9aaa3]{position:relative;border-radius:12px;overflow:hidden;background:#141414;border:1px solid #1a1a1a;min-height:180px}.image-wrapper[data-v-7aa9aaa3]:before{content:"";position:absolute;inset:0;background:linear-gradient(90deg,#141414 25%,#1a1a1a 50%,#141414 75%);background-size:200% 100%;animation:shimmer-7aa9aaa3 1.5s infinite;z-index:1;pointer-events:none}.image-wrapper.loaded[data-v-7aa9aaa3]:before{display:none}.image-wrapper.loaded[data-v-7aa9aaa3]{min-height:auto}.image-wrapper img[data-v-7aa9aaa3],.image-wrapper video[data-v-7aa9aaa3]{width:100%;display:block;position:relative;z-index:2}@keyframes shimmer-7aa9aaa3{0%{background-position:200% 0}to{background-position:-200% 0}}.image-wrapper[data-v-7aa9aaa3]:hover{border-color:#333}.overlay[data-v-7aa9aaa3]{position:absolute;inset:0;background:linear-gradient(transparent 50%,rgba(0,0,0,.85));opacity:0;transition:opacity .2s;display:flex;align-items:flex-end;justify-content:flex-end;padding:12px;z-index:10}.image-wrapper:hover .overlay[data-v-7aa9aaa3]{opacity:1}.file-placeholder[data-v-7aa9aaa3]{width:100%;height:200px;display:flex;align-items:center;justify-content:center;background:#141414;color:#333}.file-placeholder svg[data-v-7aa9aaa3]{width:56px;height:56px}.overlay-actions[data-v-7aa9aaa3]{display:flex;gap:8px}.action-btn[data-v-7aa9aaa3]{width:32px;height:32px;border:none;border-radius:50%;background:hsla(0,0%,100%,.08);backdrop-filter:blur(8px);cursor:pointer;transition:all .2s;display:flex;align-items:center;justify-content:center;color:hsla(0,0%,100%,.6)}.action-btn svg[data-v-7aa9aaa3]{width:16px;height:16px}.action-btn[data-v-7aa9aaa3]:hover{background:hsla(0,0%,100%,.2);color:#fff;transform:scale(1.1)}.load-trigger[data-v-7aa9aaa3]{display:flex;justify-content:center;align-items:center;padding:48px;min-height:100px}.loading-more[data-v-7aa9aaa3]{display:flex;align-items:center;gap:12px;color:#666;font-size:14px}.no-more[data-v-7aa9aaa3]{color:#444;font-size:14px}.preview-modal[data-v-7aa9aaa3]{position:fixed;inset:0;z-index:1000;background:rgba(0,0,0,.97)}.preview-content[data-v-7aa9aaa3],.preview-modal[data-v-7aa9aaa3]{display:flex;align-items:center;justify-content:center;overflow:hidden}.preview-content[data-v-7aa9aaa3]{width:100%;height:100%;padding:60px 80px;box-sizing:border-box}.swipe-viewport[data-v-7aa9aaa3]{width:100%;height:100%;overflow:hidden;position:relative}.swipe-track[data-v-7aa9aaa3]{width:300%;height:100%;display:flex;will-change:transform}.swipe-slide[data-v-7aa9aaa3]{width:33.333%;flex-shrink:0;height:100%;display:flex;align-items:center;justify-content:center}.preview-image[data-v-7aa9aaa3],.preview-video[data-v-7aa9aaa3]{max-width:100%;max-height:100%;-o-object-fit:contain;object-fit:contain;-webkit-user-select:none;-moz-user-select:none;user-select:none;-webkit-user-drag:none}.preview-close[data-v-7aa9aaa3]{position:fixed;top:20px;right:20px;background:hsla(0,0%,100%,.1);border:none;color:#fff;cursor:pointer;width:48px;height:48px;border-radius:50%;display:flex;align-items:center;justify-content:center;transition:background .2s;z-index:1010}.preview-close[data-v-7aa9aaa3]:hover{background:hsla(0,0%,100%,.2)}.preview-close svg[data-v-7aa9aaa3]{width:28px;height:28px}.page-indicator[data-v-7aa9aaa3]{position:fixed;bottom:30px;left:50%;transform:translateX(-50%);background:rgba(0,0,0,.6);color:hsla(0,0%,100%,.8);padding:8px 16px;border-radius:20px;font-size:14px;z-index:1010}.preview-next[data-v-7aa9aaa3],.preview-prev[data-v-7aa9aaa3]{position:fixed;top:50%;transform:translateY(-50%);background:hsla(0,0%,100%,.1);border:none;color:#fff;padding:16px;cursor:pointer;border-radius:50%;width:56px;height:56px;display:flex;align-items:center;justify-content:center;transition:background .2s;z-index:1010}.preview-next[data-v-7aa9aaa3]:hover,.preview-prev[data-v-7aa9aaa3]:hover{background:hsla(0,0%,100%,.2)}.preview-next svg[data-v-7aa9aaa3],.preview-prev svg[data-v-7aa9aaa3]{width:32px;height:32px}.preview-prev[data-v-7aa9aaa3]{left:20px}.preview-next[data-v-7aa9aaa3]{right:20px}.rotate-btn[data-v-7aa9aaa3]{position:fixed;bottom:30px;right:20px;background:hsla(0,0%,100%,.1);border:none;color:#fff;cursor:pointer;width:48px;height:48px;border-radius:50%;display:flex;align-items:center;justify-content:center;transition:background .2s;z-index:1010}.rotate-btn[data-v-7aa9aaa3]:hover{background:hsla(0,0%,100%,.2)}.rotate-btn svg[data-v-7aa9aaa3]{width:24px;height:24px}@media (max-width:768px){.desktop-only[data-v-7aa9aaa3]{display:none!important}.preview-content.mobile-only[data-v-7aa9aaa3]{display:block;padding:0;width:100%;height:100%;position:absolute;top:0;left:0}.page-indicator[data-v-7aa9aaa3]{bottom:40px}}@media (min-width:769px){.mobile-only[data-v-7aa9aaa3]{display:none!important}}@media (max-width:1199px) and (min-width:601px){.gallery-container[data-v-7aa9aaa3]{padding:12px}.waterfall-column[data-v-7aa9aaa3],.waterfall[data-v-7aa9aaa3]{gap:10px}.image-wrapper[data-v-7aa9aaa3]{border-radius:8px}}@media (max-width:600px){.header[data-v-7aa9aaa3]{padding:10px 12px}.header-left .logo[data-v-7aa9aaa3]{font-size:16px}.breadcrumb[data-v-7aa9aaa3]{font-size:12px}.breadcrumb-item[data-v-7aa9aaa3]{padding:4px 8px}.file-count[data-v-7aa9aaa3]{font-size:12px}.gallery-container[data-v-7aa9aaa3]{padding:6px}.waterfall-column[data-v-7aa9aaa3],.waterfall[data-v-7aa9aaa3]{gap:6px}.image-wrapper[data-v-7aa9aaa3]{border-radius:6px;min-height:120px}.folders-section[data-v-7aa9aaa3]{margin-bottom:12px}.folders-grid[data-v-7aa9aaa3]{grid-template-columns:repeat(2,1fr);gap:8px}.folder-card[data-v-7aa9aaa3]{padding:16px 12px;border-radius:8px}.folder-icon[data-v-7aa9aaa3]{width:36px;height:36px;margin-bottom:8px}.folder-name[data-v-7aa9aaa3]{font-size:12px}.load-trigger[data-v-7aa9aaa3]{padding:24px;min-height:60px}}.copy-toast{position:fixed;bottom:80px;left:50%;transform:translateX(-50%) translateY(20px);background:rgba(0,0,0,.8);color:#fff;padding:10px 24px;border-radius:20px;font-size:14px;opacity:0;transition:all .3s ease;z-index:9999;pointer-events:none}.copy-toast.show{opacity:1;transform:translateX(-50%) translateY(0)}
\ No newline at end of file
diff --git a/css/733.ce2542a6.css.gz b/css/733.ce2542a6.css.gz
new file mode 100644
index 0000000..12990e1
Binary files /dev/null and b/css/733.ce2542a6.css.gz differ
diff --git a/index.html b/index.html
index f1f88d7..9821f91 100644
--- a/index.html
+++ b/index.html
@@ -1 +1 @@
-
Sanyue ImgHub
\ No newline at end of file
+Sanyue ImgHub
\ No newline at end of file
diff --git a/index.html.gz b/index.html.gz
index 311e145..39ebc9c 100644
Binary files a/index.html.gz and b/index.html.gz differ
diff --git a/js/733.0149a1b8.js b/js/733.0149a1b8.js
new file mode 100644
index 0000000..849f3f5
--- /dev/null
+++ b/js/733.0149a1b8.js
@@ -0,0 +1,2 @@
+"use strict";(self["webpackChunksanyue_imghub"]=self["webpackChunksanyue_imghub"]||[]).push([[733],{6733:function(t,e,i){i.r(e),i.d(e,{default:function(){return q}});var s=i(6768),r=i(5130),o=i(4232);const n=["src"],a=["src"];function l(t,e,i,l,h,c){return(0,s.uX)(),(0,s.CE)("div",{class:"tm-viewport",ref:"viewport",onPointerdown:e[1]||(e[1]=(...t)=>c.onPointerDown&&c.onPointerDown(...t)),onPointermove:e[2]||(e[2]=(...t)=>c.onPointerMove&&c.onPointerMove(...t)),onPointerup:e[3]||(e[3]=(...t)=>c.onPointerUp&&c.onPointerUp(...t)),onPointercancel:e[4]||(e[4]=(...t)=>c.onPointerUp&&c.onPointerUp(...t)),onDblclick:e[5]||(e[5]=(0,r.D$)((...t)=>c.onDblClick&&c.onDblClick(...t),["prevent"]))},[i.isImage?((0,s.uX)(),(0,s.CE)("img",{key:0,class:"tm-media",src:i.src,draggable:"false",style:(0,o.Tr)(c.mediaStyle),onLoad:e[0]||(e[0]=(...t)=>c.onLoad&&c.onLoad(...t))},null,44,n)):i.isVideo?((0,s.uX)(),(0,s.CE)("video",{key:1,class:"tm-media",src:i.src,controls:"",autoplay:"",playsinline:"",style:(0,o.Tr)(c.mediaStyle)},null,12,a)):(0,s.Q3)("",!0)],544)}var h={name:"TransformMedia",props:{file:{type:Object,required:!0},src:{type:String,required:!0},isImage:{type:Boolean,default:!0},isVideo:{type:Boolean,default:!1}},data(){return{pointers:new Map,scale:1,rotation:0,rotatePreview:0,tx:0,ty:0,naturalWidth:0,naturalHeight:0,startScale:1,startRotation:0,startTx:0,startTy:0,startCenter:null,startDist:0,startAngle:0,dragging:!1,dragStart:null,viewportRect:null,minScale:1,maxScale:4,gestureMode:null}},computed:{isActiveTransform(){return this.scale>1.001||this.pointers.size>=2||this.dragging},displayRotation(){return this.rotation+this.rotatePreview},rotateShrink(){const t=Math.min(1,Math.abs(this.rotatePreview)/90),e=Math.sin(Math.PI*t);return 1-.12*e},mediaStyle(){const t=this.scale*this.rotateShrink,e=this.pointers.size>0;return{transform:`translate3d(${this.tx}px, ${this.ty}px, 0) scale(${t}) rotate(${this.displayRotation}deg)`,transition:e?"none":"transform 0.25s ease",transformOrigin:"center center"}}},watch:{isActiveTransform(t){this.$emit(t?"lock":"unlock")}},methods:{onLoad(t){const e=t.target;this.naturalWidth=e.naturalWidth,this.naturalHeight=e.naturalHeight},reset(){this.scale=1,this.rotation=0,this.rotatePreview=0,this.tx=0,this.ty=0,this.pointers.clear(),this.dragging=!1,this.$emit("unlock")},clamp(t,e,i){return Math.max(e,Math.min(i,t))},rubberBand(t,e,i=.55){return t*e*i/(e+i*t)},getViewportRect(){return this.$refs.viewport?.getBoundingClientRect()},getPanBounds(){const t=this.$refs.viewport?.getBoundingClientRect();if(!t)return{maxX:0,maxY:0,vw:0,vh:0};const e=t.width,i=t.height,s=this.$el.querySelector("img, video"),r=s?.clientWidth||e,o=s?.clientHeight||i,n=r*this.scale,a=o*this.scale,l=Math.max(0,(n-e)/2),h=Math.max(0,(a-i)/2);return{maxX:l,maxY:h,vw:e,vh:i}},applyBoundWithRubber(t,e,i){return t>e?e+this.rubberBand(t-e,i,.55):t<-e?-e-this.rubberBand(-e-t,i,.55):t},calcTwoPointer(){const t=Array.from(this.pointers.values()),e=t[0],i=t[1],s=i.x-e.x,r=i.y-e.y,o=Math.hypot(s,r),n=Math.atan2(r,s)*(180/Math.PI),a={x:(e.x+i.x)/2,y:(e.y+i.y)/2};return{dist:o,angle:n,center:a}},normalizeAngle(t){return t=(t%360+360)%360,t>180?t-360:t},onPointerDown(t){if(t.currentTarget.setPointerCapture?.(t.pointerId),this.viewportRect=this.getViewportRect(),this.pointers.set(t.pointerId,{x:t.clientX,y:t.clientY}),2===this.pointers.size){const{dist:t,angle:e,center:i}=this.calcTwoPointer();return this.startDist=t,this.startAngle=e,this.startCenter=i,this.startScale=this.scale,this.startRotation=this.rotation,this.startTx=this.tx,this.startTy=this.ty,this.dragging=!1,this.gestureMode=null,void(this.rotatePreview=0)}this.scale>1.001&&(this.dragging=!0,this.dragStart={x:t.clientX,y:t.clientY},this.startTx=this.tx,this.startTy=this.ty)},onPointerMove(t){if(this.pointers.has(t.pointerId)){if(this.pointers.set(t.pointerId,{x:t.clientX,y:t.clientY}),2===this.pointers.size){t.preventDefault();const{dist:e,angle:i,center:s}=this.calcTwoPointer(),r=e/(this.startDist||e),o=Math.abs(r-1),n=this.normalizeAngle(i-this.startAngle),a=Math.abs(n),l=8,h=.08;if(!this.gestureMode)if(a>=l)this.gestureMode="rotate";else{if(!(o>=h))return;this.gestureMode="pinch"}if("rotate"===this.gestureMode)return this.scale=this.startScale,void(this.rotatePreview=this.clamp(n,-90,90));if("pinch"===this.gestureMode&&(this.scale=this.clamp(this.startScale*r,this.minScale,this.maxScale),this.rotatePreview=0),this.startCenter&&this.viewportRect){const t=this.startCenter.x-this.viewportRect.left-this.viewportRect.width/2,e=this.startCenter.y-this.viewportRect.top-this.viewportRect.height/2,i=s.x-this.viewportRect.left-this.viewportRect.width/2,r=s.y-this.viewportRect.top-this.viewportRect.height/2;this.tx=this.startTx+(i-t),this.ty=this.startTy+(r-e)}return}if(this.dragging&&this.scale>1.001){t.preventDefault();const e=t.clientX-this.dragStart.x,i=t.clientY-this.dragStart.y,s=this.startTx+e,r=this.startTy+i,{maxX:o,maxY:n,vw:a,vh:l}=this.getPanBounds();this.tx=this.applyBoundWithRubber(s,o,a),this.ty=this.applyBoundWithRubber(r,n,l)}}},onPointerUp(t){if(this.pointers.has(t.pointerId)&&this.pointers.delete(t.pointerId),this.pointers.size<2&&"rotate"===this.gestureMode&&(this.finishRotate(),this.gestureMode=null),this.pointers.size<2&&(this.startCenter=null,this.startDist=0,this.startAngle=0,this.gestureMode=null),0===this.pointers.size)if(this.dragging=!1,this.scale<=1.001)this.scale=1,this.tx=0,this.ty=0;else{const{maxX:t,maxY:e}=this.getPanBounds();this.tx=Math.max(-t,Math.min(t,this.tx)),this.ty=Math.max(-e,Math.min(e,this.ty))}},finishRotate(){const t=this.rotatePreview,e=30;let i=0;Math.abs(t)>=e&&(i=t>0?90:-90);const s=((this.rotation+i)%360+360)%360;this.rotation=s,this.rotatePreview=0,this.updateFillScale()},updateFillScale(){const t=this.$refs.viewport?.getBoundingClientRect();if(!t||!this.naturalWidth||!this.naturalHeight)return;const e=t.width,i=t.height,s=this.naturalWidth,r=this.naturalHeight,o=this.rotation%360,n=90===o||270===o,a=n?r:s,l=n?s:r,h=Math.min(e/a,i/l),c=Math.max(e/a,i/l);if(n){const t=c/h;this.scale=Math.min(t,this.maxScale),this.tx=0,this.ty=0}else this.scale=1,this.tx=0,this.ty=0},onDblClick(){this.scale>1.001?(this.scale=1,this.tx=0,this.ty=0):this.scale=2}}},c=i(1241);const d=(0,c.A)(h,[["render",l],["__scopeId","data-v-5c291694"]]);var u=d;const p={class:"public-browse"},m={class:"header"},g={class:"header-left"},v={class:"logo"},w={class:"header-center"},f={class:"breadcrumb"},y=["onClick"],k={class:"header-right"},C={class:"file-count"},x={key:0,class:"loading-container"},b={key:1,class:"error-container"},L={key:2,class:"gallery-container",ref:"galleryContainer"},I={key:0,class:"folders-section"},M={class:"folders-grid"},F=["onClick"],P={class:"folder-name"},T={class:"waterfall",ref:"waterfall"},S=["onClick"],X=["src","alt","onLoad"],R=["src","onLoadedmetadata"],$={key:2,class:"file-placeholder"},E={class:"overlay"},D={class:"overlay-actions"},B=["onClick"],A=["onClick"],z={ref:"loadTrigger",class:"load-trigger"},H={key:0,class:"loading-more"},V={key:1,class:"no-more"},W=["src"],U=["src"],_={class:"page-indicator"};function Y(t,e,i,n,a,l){const h=u;return(0,s.uX)(),(0,s.CE)("div",p,[(0,s.Lk)("header",m,[(0,s.Lk)("div",g,[(0,s.Lk)("span",v,(0,o.v_)(l.siteName),1)]),(0,s.Lk)("div",w,[(0,s.Lk)("div",f,[(0,s.Lk)("span",{class:"breadcrumb-item",onClick:e[0]||(e[0]=(...t)=>l.goToRoot&&l.goToRoot(...t))},(0,o.v_)(l.rootDirName),1),((0,s.uX)(!0),(0,s.CE)(s.FK,null,(0,s.pI)(l.pathParts,(t,i)=>((0,s.uX)(),(0,s.CE)(s.FK,{key:i},[e[18]||(e[18]=(0,s.Lk)("span",{class:"breadcrumb-sep"},"/",-1)),(0,s.Lk)("span",{class:"breadcrumb-item",onClick:t=>l.goToPath(i)},(0,o.v_)(t),9,y)],64))),128))])]),(0,s.Lk)("div",k,[(0,s.Lk)("span",C,(0,o.v_)(a.totalCount)+" 个文件",1)])]),a.loading&&0===a.files.length?((0,s.uX)(),(0,s.CE)("div",x,[...e[19]||(e[19]=[(0,s.Lk)("div",{class:"loading-spinner"},null,-1),(0,s.Lk)("p",null,"加载中...",-1)])])):a.error?((0,s.uX)(),(0,s.CE)("div",b,[(0,s.Lk)("p",null,(0,o.v_)(a.error),1),a.canRetry?((0,s.uX)(),(0,s.CE)("button",{key:0,onClick:e[1]||(e[1]=(...t)=>l.loadFiles&&l.loadFiles(...t)),class:"retry-btn"},"重试")):(0,s.Q3)("",!0)])):((0,s.uX)(),(0,s.CE)("div",L,[l.folders.length>0?((0,s.uX)(),(0,s.CE)("div",I,[(0,s.Lk)("div",M,[((0,s.uX)(!0),(0,s.CE)(s.FK,null,(0,s.pI)(l.folders,t=>((0,s.uX)(),(0,s.CE)("div",{key:t.name,class:"folder-card",onClick:e=>l.enterFolder(t.name)},[e[20]||(e[20]=(0,s.Lk)("div",{class:"folder-icon"},[(0,s.Lk)("svg",{viewBox:"0 0 24 24",fill:"currentColor"},[(0,s.Lk)("path",{d:"M10 4H4c-1.1 0-2 .9-2 2v12c0 1.1.9 2 2 2h16c1.1 0 2-.9 2-2V8c0-1.1-.9-2-2-2h-8l-2-2z"})])],-1)),(0,s.Lk)("span",P,(0,o.v_)(l.getFolderName(t.name)),1)],8,F))),128))])])):(0,s.Q3)("",!0),(0,s.Lk)("div",T,[((0,s.uX)(!0),(0,s.CE)(s.FK,null,(0,s.pI)(l.columns,(t,i)=>((0,s.uX)(),(0,s.CE)("div",{key:i,class:"waterfall-column"},[((0,s.uX)(!0),(0,s.CE)(s.FK,null,(0,s.pI)(t,t=>((0,s.uX)(),(0,s.CE)("div",{key:t.name,class:"waterfall-item",onClick:e=>l.openPreview(t)},[(0,s.Lk)("div",{class:(0,o.C4)(["image-wrapper",{loaded:t.loaded}])},[l.isImage(t)?((0,s.uX)(),(0,s.CE)("img",{key:0,src:l.getFileUrl(t.name),alt:t.name,loading:"lazy",onLoad:e=>l.onImageLoad(e,t),onError:e[2]||(e[2]=(...t)=>l.handleImageError&&l.handleImageError(...t))},null,40,X)):l.isVideo(t)?((0,s.uX)(),(0,s.CE)("video",{key:1,src:l.getFileUrl(t.name),muted:"",loop:"",preload:"metadata",onLoadedmetadata:e=>l.onVideoLoad(e,t),onMouseenter:e[3]||(e[3]=t=>t.target.play()),onMouseleave:e[4]||(e[4]=t=>t.target.pause())},null,40,R)):((0,s.uX)(),(0,s.CE)("div",$,[...e[21]||(e[21]=[(0,s.Lk)("svg",{viewBox:"0 0 24 24",fill:"currentColor"},[(0,s.Lk)("path",{d:"M14 2H6c-1.1 0-2 .9-2 2v16c0 1.1.9 2 2 2h12c1.1 0 2-.9 2-2V8l-6-6zm-1 7V3.5L18.5 9H13z"})],-1)])])),(0,s.Lk)("div",E,[(0,s.Lk)("div",D,[(0,s.Lk)("button",{class:"action-btn",onClick:(0,r.D$)(e=>l.copyLink(t.name),["stop"]),title:"复制链接"},[...e[22]||(e[22]=[(0,s.Lk)("svg",{viewBox:"0 0 24 24",fill:"currentColor"},[(0,s.Lk)("path",{d:"M3.9 12c0-1.71 1.39-3.1 3.1-3.1h4V7H7c-2.76 0-5 2.24-5 5s2.24 5 5 5h4v-1.9H7c-1.71 0-3.1-1.39-3.1-3.1zM8 13h8v-2H8v2zm9-6h-4v1.9h4c1.71 0 3.1 1.39 3.1 3.1s-1.39 3.1-3.1 3.1h-4V17h4c2.76 0 5-2.24 5-5s-2.24-5-5-5z"})],-1)])],8,B),(0,s.Lk)("button",{class:"action-btn",onClick:(0,r.D$)(e=>l.downloadFile(t.name),["stop"]),title:"下载"},[...e[23]||(e[23]=[(0,s.Lk)("svg",{viewBox:"0 0 24 24",fill:"currentColor"},[(0,s.Lk)("path",{d:"M19 9h-4V3H9v6H5l7 7 7-7zM5 18v2h14v-2H5z"})],-1)])],8,A)])])],2)],8,S))),128))]))),128))],512),(0,s.Lk)("div",z,[a.loading&&a.files.length>0?((0,s.uX)(),(0,s.CE)("div",H,[...e[24]||(e[24]=[(0,s.Lk)("div",{class:"loading-spinner-small"},null,-1),(0,s.Lk)("span",null,"加载中...",-1)])])):!a.hasMore&&l.mediaFiles.length>0?((0,s.uX)(),(0,s.CE)("div",V," 已加载全部 ")):(0,s.Q3)("",!0)],512)],512)),a.previewVisible?((0,s.uX)(),(0,s.CE)("div",{key:3,class:"preview-modal",onClick:e[17]||(e[17]=(0,r.D$)((...t)=>l.closePreview&&l.closePreview(...t),["self"]))},[(0,s.Lk)("button",{class:"preview-close",onClick:e[5]||(e[5]=(0,r.D$)((...t)=>l.closePreview&&l.closePreview(...t),["stop"]))},[...e[25]||(e[25]=[(0,s.Lk)("svg",{viewBox:"0 0 24 24",fill:"currentColor"},[(0,s.Lk)("path",{d:"M19 6.41L17.59 5 12 10.59 6.41 5 5 6.41 10.59 12 5 17.59 6.41 19 12 13.41 17.59 19 19 17.59 13.41 12z"})],-1)])]),(0,s.Lk)("div",{class:"preview-content desktop-only",onClick:e[6]||(e[6]=(0,r.D$)(()=>{},["stop"]))},[l.currentPreviewFile&&l.isImage(l.currentPreviewFile)?((0,s.uX)(),(0,s.CE)("img",{key:0,src:l.getFileUrl(l.currentPreviewFile.name),class:"preview-image",style:(0,o.Tr)(l.desktopImageStyle),draggable:"false"},null,12,W)):l.currentPreviewFile&&l.isVideo(l.currentPreviewFile)?((0,s.uX)(),(0,s.CE)("video",{key:1,src:l.getFileUrl(l.currentPreviewFile.name),controls:"",autoplay:"",class:"preview-video",style:(0,o.Tr)(l.desktopImageStyle)},null,12,U)):(0,s.Q3)("",!0)]),(0,s.Lk)("div",{class:"preview-content mobile-only",onClick:e[13]||(e[13]=(0,r.D$)(()=>{},["stop"]))},[(0,s.Lk)("div",{class:"swipe-viewport",ref:"mobileViewport",onTouchstart:e[10]||(e[10]=(...t)=>l.onSwipeStart&&l.onSwipeStart(...t)),onTouchmove:e[11]||(e[11]=(...t)=>l.onSwipeMove&&l.onSwipeMove(...t)),onTouchend:e[12]||(e[12]=(...t)=>l.onSwipeEnd&&l.onSwipeEnd(...t))},[(0,s.Lk)("div",{class:"swipe-track",style:(0,o.Tr)(l.swipeTrackStyle),onTransitionend:e[9]||(e[9]=(...t)=>l.onSwipeTransitionEnd&&l.onSwipeTransitionEnd(...t))},[((0,s.uX)(!0),(0,s.CE)(s.FK,null,(0,s.pI)(l.swipeWindow,(t,i)=>((0,s.uX)(),(0,s.CE)("div",{class:"swipe-slide",key:l.getSlideKey(t,i)},[t?((0,s.uX)(),(0,s.Wv)(h,{key:0,file:t,src:l.getFileUrl(t.name),"is-image":l.isImage(t),"is-video":l.isVideo(t),onLock:e[7]||(e[7]=t=>a.gestureLocked=!0),onUnlock:e[8]||(e[8]=t=>a.gestureLocked=!1)},null,8,["file","src","is-image","is-video"])):(0,s.Q3)("",!0)]))),128))],36)],544)]),a.previewIndex>0?((0,s.uX)(),(0,s.CE)("button",{key:0,class:"preview-prev desktop-only",onClick:e[14]||(e[14]=(0,r.D$)((...t)=>l.prevImage&&l.prevImage(...t),["stop"]))},[...e[26]||(e[26]=[(0,s.Lk)("svg",{viewBox:"0 0 24 24",fill:"currentColor"},[(0,s.Lk)("path",{d:"M15.41 7.41L14 6l-6 6 6 6 1.41-1.41L10.83 12z"})],-1)])])):(0,s.Q3)("",!0),a.previewIndexl.nextImage&&l.nextImage(...t),["stop"]))},[...e[27]||(e[27]=[(0,s.Lk)("svg",{viewBox:"0 0 24 24",fill:"currentColor"},[(0,s.Lk)("path",{d:"M10 6L8.59 7.41 13.17 12l-4.58 4.59L10 18l6-6z"})],-1)])])):(0,s.Q3)("",!0),(0,s.Lk)("button",{class:"rotate-btn desktop-only",onClick:e[16]||(e[16]=(0,r.D$)((...t)=>l.rotateImage&&l.rotateImage(...t),["stop"])),title:"旋转90°"},[...e[28]||(e[28]=[(0,s.Lk)("svg",{viewBox:"0 0 24 24",fill:"currentColor"},[(0,s.Lk)("path",{d:"M7.11 8.53L5.7 7.11C4.8 8.27 4.24 9.61 4.07 11h2.02c.14-.87.49-1.72 1.02-2.47zM6.09 13H4.07c.17 1.39.72 2.73 1.62 3.89l1.41-1.42c-.52-.75-.87-1.59-1.01-2.47zm1.01 5.32c1.16.9 2.51 1.44 3.9 1.61V17.9c-.87-.15-1.71-.49-2.46-1.03L7.1 18.32zM13 4.07V1L8.45 5.55 13 10V6.09c2.84.48 5 2.94 5 5.91s-2.16 5.43-5 5.91v2.02c3.95-.49 7-3.85 7-7.93s-3.05-7.44-7-7.93z"})],-1)])]),(0,s.Lk)("div",_,(0,o.v_)(a.previewIndex+1)+" / "+(0,o.v_)(l.mediaFiles.length),1)])):(0,s.Q3)("",!0)])}i(4114),i(8111),i(2489),i(7588),i(1701);var Q=i(4373),K=i(8401),N={name:"PublicBrowse",components:{TransformMedia:u},data(){return{files:[],allowedDirs:[],rootDir:"",currentPath:"",totalCount:0,loading:!1,error:null,canRetry:!0,hasMore:!0,previewVisible:!1,previewIndex:0,observer:null,pageSize:24,columnCount:4,columnHeights:[0,0,0,0],imageRotation:0,swipeX:0,swipeStartX:0,swipeStartY:0,swipeStartT:0,swipeActive:!1,swipeAnimating:!1,swipeDir:0,viewportW:0,gestureLocked:!1}},computed:{...(0,K.L8)(["userConfig"]),siteName(){return this.userConfig?.siteTitle||"公开相册"},rootDirName(){return this.rootDir.split("/").filter(Boolean).pop()||"根目录"},pathParts(){if(!this.currentPath||!this.rootDir)return[];const t=this.currentPath.replace(this.rootDir,"").replace(/^\/+/,"");return t.split("/").filter(Boolean)},folders(){return this.files.filter(t=>t.isFolder)},mediaFiles(){return this.files.filter(t=>!t.isFolder)},columns(){const t=Array.from({length:this.columnCount},()=>[]);for(const e of this.mediaFiles){const i=e.columnIndex??0;i0?this.mediaFiles[this.previewIndex-1]:null},nextPreviewFile(){return this.previewIndex{t.columnIndex=void 0,this.assignToColumn(t)}))},getShortestColumn(){let t=0,e=this.columnHeights[0];for(let i=1;i{const e=t[0];e.isIntersecting&&this.hasMore&&!this.loading&&this.loadMore()},{rootMargin:"200px"})},observeLoadTrigger(){this.$nextTick(()=>{this.$refs.loadTrigger&&this.observer&&this.observer.observe(this.$refs.loadTrigger)})},async initFromRoute(){const t=this.$route.params.dir||"",e=Array.isArray(t)?t.join("/"):t;if(!e)return this.error="请指定要浏览的目录,例如: /browse/landscape",void(this.canRetry=!1);const i=e.split("/").filter(Boolean);this.rootDir=i[0],this.currentPath=e,this.files=[],this.hasMore=!0,this.columnHeights=new Array(this.columnCount).fill(0),await this.loadFiles(),this.observeLoadTrigger()},async loadFiles(){this.loading=!0,this.error=null,this.canRetry=!0;try{const t=await Q.A.get(`/api/public/list?dir=${encodeURIComponent(this.currentPath)}&count=${this.pageSize}`);t.data.allowedDirs&&(this.allowedDirs=t.data.allowedDirs);const e=(t.data.directories||[]).map(t=>({name:t,isFolder:!0})),i=(t.data.files||[]).map(t=>({name:t.name,isFolder:!1,metadata:t.metadata,columnIndex:void 0}));i.forEach(t=>this.assignToColumn(t)),this.files=[...e,...i],this.totalCount=t.data.totalCount||this.files.length,this.hasMore=this.mediaFiles.length({name:t.name,isFolder:!1,metadata:t.metadata,columnIndex:void 0}));i.forEach(t=>this.assignToColumn(t)),this.files.push(...i),this.hasMore=this.mediaFiles.length{this.showToast("已复制")}).catch(()=>{const t=document.createElement("input");t.value=e,document.body.appendChild(t),t.select(),document.execCommand("copy"),document.body.removeChild(t),this.showToast("已复制")})},showToast(t){const e=document.querySelector(".copy-toast");e&&e.remove();const i=document.createElement("div");i.className="copy-toast",i.textContent=t,document.body.appendChild(i),setTimeout(()=>i.classList.add("show"),10),setTimeout(()=>{i.classList.remove("show"),setTimeout(()=>i.remove(),300)},1500)},downloadFile(t){const e=document.createElement("a");e.href=this.getFileUrl(t),e.download=t.split("/").pop(),e.click()},openPreview(t){if(t.isFolder)return;const e=this.mediaFiles.findIndex(e=>e.name===t.name);e>=0&&(this.previewIndex=e,this.previewVisible=!0,this.imageRotation=0,this.gestureLocked=!1,document.body.style.overflow="hidden",this.$nextTick(()=>{this.viewportW=this.$refs.mobileViewport?.getBoundingClientRect().width||window.innerWidth}))},closePreview(){this.previewVisible=!1,this.imageRotation=0,this.gestureLocked=!1,document.body.style.overflow=""},prevImage(){this.previewIndex>0&&(this.previewIndex--,this.imageRotation=0)},nextImage(){this.previewIndex=360&&setTimeout(()=>{const t=this.$el.querySelector(".preview-image, .preview-video");t?(t.style.transition="none",this.imageRotation=0,t.offsetHeight,t.style.transition=""):this.imageRotation=0},300)},onSwipeStart(t){if(this.gestureLocked)return;if(this.swipeAnimating)return;const e=t.touches[0];this.swipeStartX=e.clientX,this.swipeStartY=e.clientY,this.swipeStartT=performance.now(),this.swipeX=0,this.swipeActive=!1,this.viewportW=this.$refs.mobileViewport?.getBoundingClientRect().width||window.innerWidth},onSwipeMove(t){if(this.gestureLocked)return;if(this.swipeAnimating)return;const e=t.touches[0],i=e.clientX-this.swipeStartX,s=e.clientY-this.swipeStartY;if(!this.swipeActive){if(Math.abs(i)<8)return;if(Math.abs(i)<=Math.abs(s))return;this.swipeActive=!0}t.preventDefault();let r=i;0===this.previewIndex&&r>0?r=this.rubberBand(r,this.viewportW,.55):this.previewIndex===this.mediaFiles.length-1&&r<0&&(r=-this.rubberBand(-r,this.viewportW,.55)),this.swipeX=r},onSwipeEnd(){if(this.gestureLocked)return;if(this.swipeAnimating)return;if(!this.swipeActive)return void(this.swipeX=0);const t=Math.max(1,performance.now()-this.swipeStartT),e=this.swipeX/t,i=.2*this.viewportW;let s=0;(this.swipeX<=-i||e<=-.8)&&(s=1),(this.swipeX>=i||e>=.8)&&(s=-1),(-1===s&&0===this.previewIndex||1===s&&this.previewIndex===this.mediaFiles.length-1)&&(s=0),this.swipeDir=s,this.swipeAnimating=!0,this.swipeX=1===s?-this.viewportW:-1===s?+this.viewportW:0},onSwipeTransitionEnd(){this.swipeAnimating&&(1===this.swipeDir&&this.previewIndex++,-1===this.swipeDir&&this.previewIndex--,this.swipeAnimating=!1,this.swipeDir=0,this.swipeX=0)},rubberBand(t,e,i=.55){return t*e*i/(e+i*t)}}};const j=(0,c.A)(N,[["render",Y],["__scopeId","data-v-7aa9aaa3"]]);var q=j}}]);
+//# sourceMappingURL=733.0149a1b8.js.map
\ No newline at end of file
diff --git a/js/733.0149a1b8.js.gz b/js/733.0149a1b8.js.gz
new file mode 100644
index 0000000..cfd45cf
Binary files /dev/null and b/js/733.0149a1b8.js.gz differ
diff --git a/js/733.0149a1b8.js.map b/js/733.0149a1b8.js.map
new file mode 100644
index 0000000..d81f882
--- /dev/null
+++ b/js/733.0149a1b8.js.map
@@ -0,0 +1 @@
+{"version":3,"file":"js/733.0149a1b8.js","mappings":"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","sources":["webpack://sanyue_imghub/./src/components/TransformMedia.vue?50a7","webpack://sanyue_imghub/./src/components/TransformMedia.vue","webpack://sanyue_imghub/./src/components/TransformMedia.vue?c392","webpack://sanyue_imghub/./src/views/PublicBrowse.vue?eb47","webpack://sanyue_imghub/./src/views/PublicBrowse.vue","webpack://sanyue_imghub/./src/views/PublicBrowse.vue?bab5"],"sourcesContent":["import { normalizeStyle as _normalizeStyle, openBlock as _openBlock, createElementBlock as _createElementBlock, createCommentVNode as _createCommentVNode, withModifiers as _withModifiers } from \"vue\";\nconst _hoisted_1 = [\"src\"];\nconst _hoisted_2 = [\"src\"];\nexport function render(_ctx, _cache, $props, $setup, $data, $options) {\n return _openBlock(), _createElementBlock(\"div\", {\n class: \"tm-viewport\",\n ref: \"viewport\",\n onPointerdown: _cache[1] || (_cache[1] = (...args) => $options.onPointerDown && $options.onPointerDown(...args)),\n onPointermove: _cache[2] || (_cache[2] = (...args) => $options.onPointerMove && $options.onPointerMove(...args)),\n onPointerup: _cache[3] || (_cache[3] = (...args) => $options.onPointerUp && $options.onPointerUp(...args)),\n onPointercancel: _cache[4] || (_cache[4] = (...args) => $options.onPointerUp && $options.onPointerUp(...args)),\n onDblclick: _cache[5] || (_cache[5] = _withModifiers((...args) => $options.onDblClick && $options.onDblClick(...args), [\"prevent\"]))\n }, [$props.isImage ? (_openBlock(), _createElementBlock(\"img\", {\n key: 0,\n class: \"tm-media\",\n src: $props.src,\n draggable: \"false\",\n style: _normalizeStyle($options.mediaStyle),\n onLoad: _cache[0] || (_cache[0] = (...args) => $options.onLoad && $options.onLoad(...args))\n }, null, 44, _hoisted_1)) : $props.isVideo ? (_openBlock(), _createElementBlock(\"video\", {\n key: 1,\n class: \"tm-media\",\n src: $props.src,\n controls: \"\",\n autoplay: \"\",\n playsinline: \"\",\n style: _normalizeStyle($options.mediaStyle)\n }, null, 12, _hoisted_2)) : _createCommentVNode(\"\", true)], 544);\n}","export default {\n name: \"TransformMedia\",\n props: {\n file: {\n type: Object,\n required: true\n },\n src: {\n type: String,\n required: true\n },\n isImage: {\n type: Boolean,\n default: true\n },\n isVideo: {\n type: Boolean,\n default: false\n }\n },\n data() {\n return {\n pointers: new Map(),\n // transform state\n scale: 1,\n rotation: 0,\n // 落地的旋转角度:0, 90, 180, 270\n rotatePreview: 0,\n // 旋转预览角度(-90~+90,跟手)\n tx: 0,\n ty: 0,\n // 图片原始尺寸\n naturalWidth: 0,\n naturalHeight: 0,\n // gesture start snapshots\n startScale: 1,\n startRotation: 0,\n startTx: 0,\n startTy: 0,\n startCenter: null,\n startDist: 0,\n startAngle: 0,\n // drag\n dragging: false,\n dragStart: null,\n viewportRect: null,\n // bounds\n minScale: 1,\n maxScale: 4,\n // 手势模式锁定:'pinch' | 'rotate' | null\n gestureMode: null\n };\n },\n computed: {\n isActiveTransform() {\n return this.scale > 1.001 || this.pointers.size >= 2 || this.dragging;\n },\n // 实际显示的旋转角度 = 落地角度 + 预览角度\n displayRotation() {\n return this.rotation + this.rotatePreview;\n },\n // 旋转时缩小系数(中间最小,两端恢复)\n rotateShrink() {\n const p = Math.min(1, Math.abs(this.rotatePreview) / 90);\n const k = Math.sin(Math.PI * p);\n return 1 - 0.12 * k; // 最多缩小12%\n },\n mediaStyle() {\n const finalScale = this.scale * this.rotateShrink;\n const inGesture = this.pointers.size > 0;\n return {\n transform: `translate3d(${this.tx}px, ${this.ty}px, 0) scale(${finalScale}) rotate(${this.displayRotation}deg)`,\n transition: inGesture ? \"none\" : \"transform 0.25s ease\",\n transformOrigin: \"center center\"\n };\n }\n },\n watch: {\n isActiveTransform(v) {\n this.$emit(v ? \"lock\" : \"unlock\");\n }\n },\n methods: {\n onLoad(e) {\n // 记录图片原始尺寸\n const img = e.target;\n this.naturalWidth = img.naturalWidth;\n this.naturalHeight = img.naturalHeight;\n },\n reset() {\n this.scale = 1;\n this.rotation = 0;\n this.rotatePreview = 0;\n this.tx = 0;\n this.ty = 0;\n this.pointers.clear();\n this.dragging = false;\n this.$emit(\"unlock\");\n },\n clamp(v, min, max) {\n return Math.max(min, Math.min(max, v));\n },\n // iOS 风格橡皮筋阻尼函数\n rubberBand(distance, dimension, constant = 0.55) {\n return distance * dimension * constant / (dimension + constant * distance);\n },\n getViewportRect() {\n return this.$refs.viewport?.getBoundingClientRect();\n },\n // 计算放大后允许的最大平移范围\n getPanBounds() {\n const rect = this.$refs.viewport?.getBoundingClientRect();\n if (!rect) return {\n maxX: 0,\n maxY: 0,\n vw: 0,\n vh: 0\n };\n const vw = rect.width,\n vh = rect.height;\n const img = this.$el.querySelector('img, video');\n const iw = img?.clientWidth || vw;\n const ih = img?.clientHeight || vh;\n const sw = iw * this.scale;\n const sh = ih * this.scale;\n const maxX = Math.max(0, (sw - vw) / 2);\n const maxY = Math.max(0, (sh - vh) / 2);\n return {\n maxX,\n maxY,\n vw,\n vh\n };\n },\n // 应用边界阻尼\n applyBoundWithRubber(value, max, dimension) {\n if (value > max) {\n return max + this.rubberBand(value - max, dimension, 0.55);\n }\n if (value < -max) {\n return -max - this.rubberBand(-max - value, dimension, 0.55);\n }\n return value;\n },\n calcTwoPointer() {\n const pts = Array.from(this.pointers.values());\n const p0 = pts[0],\n p1 = pts[1];\n const dx = p1.x - p0.x;\n const dy = p1.y - p0.y;\n const dist = Math.hypot(dx, dy);\n const angle = Math.atan2(dy, dx) * (180 / Math.PI);\n const center = {\n x: (p0.x + p1.x) / 2,\n y: (p0.y + p1.y) / 2\n };\n return {\n dist,\n angle,\n center\n };\n },\n // 角度归一化到 -180~180\n normalizeAngle(deg) {\n deg = (deg % 360 + 360) % 360;\n return deg > 180 ? deg - 360 : deg;\n },\n onPointerDown(e) {\n e.currentTarget.setPointerCapture?.(e.pointerId);\n this.viewportRect = this.getViewportRect();\n this.pointers.set(e.pointerId, {\n x: e.clientX,\n y: e.clientY\n });\n\n // 2指开始:初始化 pinch/rotate 基准\n if (this.pointers.size === 2) {\n const {\n dist,\n angle,\n center\n } = this.calcTwoPointer();\n this.startDist = dist;\n this.startAngle = angle;\n this.startCenter = center;\n this.startScale = this.scale;\n this.startRotation = this.rotation;\n this.startTx = this.tx;\n this.startTy = this.ty;\n this.dragging = false;\n this.gestureMode = null;\n this.rotatePreview = 0;\n return;\n }\n\n // 1指:如果已放大,则进入拖拽\n if (this.scale > 1.001) {\n this.dragging = true;\n this.dragStart = {\n x: e.clientX,\n y: e.clientY\n };\n this.startTx = this.tx;\n this.startTy = this.ty;\n }\n },\n onPointerMove(e) {\n if (!this.pointers.has(e.pointerId)) return;\n this.pointers.set(e.pointerId, {\n x: e.clientX,\n y: e.clientY\n });\n\n // 2指:缩放 + 旋转(带死区锁定)\n if (this.pointers.size === 2) {\n e.preventDefault();\n const {\n dist,\n angle,\n center\n } = this.calcTwoPointer();\n const scaleFactor = dist / (this.startDist || dist);\n const scaleChange = Math.abs(scaleFactor - 1);\n const deltaAngle = this.normalizeAngle(angle - this.startAngle);\n const angleChange = Math.abs(deltaAngle);\n\n // 阈值\n const rotateStartDeg = 8; // 8° 开始进入旋转模式\n const pinchStartScale = 0.08; // 8% 缩放变化开始进入缩放模式\n\n // 判断手势模式(只在第一次超过死区时锁定)\n // 优先判断旋转:角度变化超过8°就进入旋转模式(不管缩放)\n if (!this.gestureMode) {\n if (angleChange >= rotateStartDeg) {\n this.gestureMode = 'rotate';\n } else if (scaleChange >= pinchStartScale) {\n this.gestureMode = 'pinch';\n } else {\n return; // 还在死区内\n }\n }\n\n // 旋转模式:预览角跟手,限制在 -90~+90\n if (this.gestureMode === 'rotate') {\n this.scale = this.startScale; // 锁定缩放\n this.rotatePreview = this.clamp(deltaAngle, -90, 90);\n return;\n }\n\n // 缩放模式\n if (this.gestureMode === 'pinch') {\n this.scale = this.clamp(this.startScale * scaleFactor, this.minScale, this.maxScale);\n this.rotatePreview = 0;\n }\n\n // 跟随双指中心移动\n if (this.startCenter && this.viewportRect) {\n const cx0 = this.startCenter.x - this.viewportRect.left - this.viewportRect.width / 2;\n const cy0 = this.startCenter.y - this.viewportRect.top - this.viewportRect.height / 2;\n const cx1 = center.x - this.viewportRect.left - this.viewportRect.width / 2;\n const cy1 = center.y - this.viewportRect.top - this.viewportRect.height / 2;\n this.tx = this.startTx + (cx1 - cx0);\n this.ty = this.startTy + (cy1 - cy0);\n }\n return;\n }\n\n // 1指:拖拽(只在 scale>1 时)+ 边界阻尼\n if (this.dragging && this.scale > 1.001) {\n e.preventDefault();\n const dx = e.clientX - this.dragStart.x;\n const dy = e.clientY - this.dragStart.y;\n const rawX = this.startTx + dx;\n const rawY = this.startTy + dy;\n const {\n maxX,\n maxY,\n vw,\n vh\n } = this.getPanBounds();\n this.tx = this.applyBoundWithRubber(rawX, maxX, vw);\n this.ty = this.applyBoundWithRubber(rawY, maxY, vh);\n }\n },\n onPointerUp(e) {\n if (this.pointers.has(e.pointerId)) this.pointers.delete(e.pointerId);\n\n // 两指结束:处理旋转吸附\n if (this.pointers.size < 2 && this.gestureMode === 'rotate') {\n this.finishRotate();\n this.gestureMode = null;\n }\n if (this.pointers.size < 2) {\n this.startCenter = null;\n this.startDist = 0;\n this.startAngle = 0;\n this.gestureMode = null;\n }\n if (this.pointers.size === 0) {\n this.dragging = false;\n // 缩放回到1附近,自动归位\n if (this.scale <= 1.001) {\n this.scale = 1;\n this.tx = 0;\n this.ty = 0;\n } else {\n // 放大状态:回弹到合法范围\n const {\n maxX,\n maxY\n } = this.getPanBounds();\n this.tx = Math.max(-maxX, Math.min(maxX, this.tx));\n this.ty = Math.max(-maxY, Math.min(maxY, this.ty));\n }\n }\n },\n // 松手后吸附到 0° 或 ±90°\n finishRotate() {\n const d = this.rotatePreview;\n const commitDeg = 30; // 超过30°就翻到90°\n\n let target = 0;\n if (Math.abs(d) >= commitDeg) {\n target = d > 0 ? 90 : -90;\n }\n\n // 计算新的落地角度\n const newRot = ((this.rotation + target) % 360 + 360) % 360;\n\n // 落地并重置预览角(transition 会自动处理动画)\n this.rotation = newRot;\n this.rotatePreview = 0;\n\n // 计算旋转后的铺满缩放\n this.updateFillScale();\n },\n // 计算旋转后让图片铺满屏幕的缩放比例\n updateFillScale() {\n const rect = this.$refs.viewport?.getBoundingClientRect();\n if (!rect || !this.naturalWidth || !this.naturalHeight) return;\n const vw = rect.width;\n const vh = rect.height;\n const nw = this.naturalWidth;\n const nh = this.naturalHeight;\n\n // 旋转 90° 或 270° 时宽高交换\n const rot = this.rotation % 360;\n const isRotated = rot === 90 || rot === 270;\n const effW = isRotated ? nh : nw;\n const effH = isRotated ? nw : nh;\n\n // 计算 contain 模式下的基础缩放(让图片完全显示)\n const containScale = Math.min(vw / effW, vh / effH);\n // 计算 cover 模式下的缩放(让图片铺满,可能裁切)\n const coverScale = Math.max(vw / effW, vh / effH);\n\n // 旋转后使用 cover 模式铺满屏幕\n if (isRotated) {\n // 计算相对于 contain 的放大倍数\n const fillScale = coverScale / containScale;\n this.scale = Math.min(fillScale, this.maxScale);\n this.tx = 0;\n this.ty = 0;\n } else {\n // 0° 或 180° 恢复正常\n this.scale = 1;\n this.tx = 0;\n this.ty = 0;\n }\n },\n onDblClick() {\n if (this.scale > 1.001) {\n this.scale = 1;\n this.tx = 0;\n this.ty = 0;\n } else {\n this.scale = 2;\n }\n }\n }\n};","/* unplugin-vue-components disabled */import { render } from \"./TransformMedia.vue?vue&type=template&id=5c291694&scoped=true\"\nimport script from \"./TransformMedia.vue?vue&type=script&lang=js\"\nexport * from \"./TransformMedia.vue?vue&type=script&lang=js\"\n\nimport \"./TransformMedia.vue?vue&type=style&index=0&id=5c291694&scoped=true&lang=css\"\n\nimport exportComponent from \"../../node_modules/vue-loader/dist/exportHelper.js\"\nconst __exports__ = /*#__PURE__*/exportComponent(script, [['render',render],['__scopeId',\"data-v-5c291694\"]])\n\nexport default __exports__","import { toDisplayString as _toDisplayString, createElementVNode as _createElementVNode, renderList as _renderList, Fragment as _Fragment, openBlock as _openBlock, createElementBlock as _createElementBlock, createCommentVNode as _createCommentVNode, withModifiers as _withModifiers, normalizeClass as _normalizeClass, normalizeStyle as _normalizeStyle, resolveComponent as _resolveComponent, createBlock as _createBlock } from \"vue\";\nconst _hoisted_1 = {\n class: \"public-browse\"\n};\nconst _hoisted_2 = {\n class: \"header\"\n};\nconst _hoisted_3 = {\n class: \"header-left\"\n};\nconst _hoisted_4 = {\n class: \"logo\"\n};\nconst _hoisted_5 = {\n class: \"header-center\"\n};\nconst _hoisted_6 = {\n class: \"breadcrumb\"\n};\nconst _hoisted_7 = [\"onClick\"];\nconst _hoisted_8 = {\n class: \"header-right\"\n};\nconst _hoisted_9 = {\n class: \"file-count\"\n};\nconst _hoisted_10 = {\n key: 0,\n class: \"loading-container\"\n};\nconst _hoisted_11 = {\n key: 1,\n class: \"error-container\"\n};\nconst _hoisted_12 = {\n key: 2,\n class: \"gallery-container\",\n ref: \"galleryContainer\"\n};\nconst _hoisted_13 = {\n key: 0,\n class: \"folders-section\"\n};\nconst _hoisted_14 = {\n class: \"folders-grid\"\n};\nconst _hoisted_15 = [\"onClick\"];\nconst _hoisted_16 = {\n class: \"folder-name\"\n};\nconst _hoisted_17 = {\n class: \"waterfall\",\n ref: \"waterfall\"\n};\nconst _hoisted_18 = [\"onClick\"];\nconst _hoisted_19 = [\"src\", \"alt\", \"onLoad\"];\nconst _hoisted_20 = [\"src\", \"onLoadedmetadata\"];\nconst _hoisted_21 = {\n key: 2,\n class: \"file-placeholder\"\n};\nconst _hoisted_22 = {\n class: \"overlay\"\n};\nconst _hoisted_23 = {\n class: \"overlay-actions\"\n};\nconst _hoisted_24 = [\"onClick\"];\nconst _hoisted_25 = [\"onClick\"];\nconst _hoisted_26 = {\n ref: \"loadTrigger\",\n class: \"load-trigger\"\n};\nconst _hoisted_27 = {\n key: 0,\n class: \"loading-more\"\n};\nconst _hoisted_28 = {\n key: 1,\n class: \"no-more\"\n};\nconst _hoisted_29 = [\"src\"];\nconst _hoisted_30 = [\"src\"];\nconst _hoisted_31 = {\n class: \"page-indicator\"\n};\nexport function render(_ctx, _cache, $props, $setup, $data, $options) {\n const _component_TransformMedia = _resolveComponent(\"TransformMedia\");\n return _openBlock(), _createElementBlock(\"div\", _hoisted_1, [_createElementVNode(\"header\", _hoisted_2, [_createElementVNode(\"div\", _hoisted_3, [_createElementVNode(\"span\", _hoisted_4, _toDisplayString($options.siteName), 1)]), _createElementVNode(\"div\", _hoisted_5, [_createElementVNode(\"div\", _hoisted_6, [_createElementVNode(\"span\", {\n class: \"breadcrumb-item\",\n onClick: _cache[0] || (_cache[0] = (...args) => $options.goToRoot && $options.goToRoot(...args))\n }, _toDisplayString($options.rootDirName), 1), (_openBlock(true), _createElementBlock(_Fragment, null, _renderList($options.pathParts, (part, index) => {\n return _openBlock(), _createElementBlock(_Fragment, {\n key: index\n }, [_cache[18] || (_cache[18] = _createElementVNode(\"span\", {\n class: \"breadcrumb-sep\"\n }, \"/\", -1)), _createElementVNode(\"span\", {\n class: \"breadcrumb-item\",\n onClick: $event => $options.goToPath(index)\n }, _toDisplayString(part), 9, _hoisted_7)], 64);\n }), 128))])]), _createElementVNode(\"div\", _hoisted_8, [_createElementVNode(\"span\", _hoisted_9, _toDisplayString($data.totalCount) + \" 个文件\", 1)])]), $data.loading && $data.files.length === 0 ? (_openBlock(), _createElementBlock(\"div\", _hoisted_10, [...(_cache[19] || (_cache[19] = [_createElementVNode(\"div\", {\n class: \"loading-spinner\"\n }, null, -1), _createElementVNode(\"p\", null, \"加载中...\", -1)]))])) : $data.error ? (_openBlock(), _createElementBlock(\"div\", _hoisted_11, [_createElementVNode(\"p\", null, _toDisplayString($data.error), 1), $data.canRetry ? (_openBlock(), _createElementBlock(\"button\", {\n key: 0,\n onClick: _cache[1] || (_cache[1] = (...args) => $options.loadFiles && $options.loadFiles(...args)),\n class: \"retry-btn\"\n }, \"重试\")) : _createCommentVNode(\"\", true)])) : (_openBlock(), _createElementBlock(\"div\", _hoisted_12, [$options.folders.length > 0 ? (_openBlock(), _createElementBlock(\"div\", _hoisted_13, [_createElementVNode(\"div\", _hoisted_14, [(_openBlock(true), _createElementBlock(_Fragment, null, _renderList($options.folders, folder => {\n return _openBlock(), _createElementBlock(\"div\", {\n key: folder.name,\n class: \"folder-card\",\n onClick: $event => $options.enterFolder(folder.name)\n }, [_cache[20] || (_cache[20] = _createElementVNode(\"div\", {\n class: \"folder-icon\"\n }, [_createElementVNode(\"svg\", {\n viewBox: \"0 0 24 24\",\n fill: \"currentColor\"\n }, [_createElementVNode(\"path\", {\n d: \"M10 4H4c-1.1 0-2 .9-2 2v12c0 1.1.9 2 2 2h16c1.1 0 2-.9 2-2V8c0-1.1-.9-2-2-2h-8l-2-2z\"\n })])], -1)), _createElementVNode(\"span\", _hoisted_16, _toDisplayString($options.getFolderName(folder.name)), 1)], 8, _hoisted_15);\n }), 128))])])) : _createCommentVNode(\"\", true), _createElementVNode(\"div\", _hoisted_17, [(_openBlock(true), _createElementBlock(_Fragment, null, _renderList($options.columns, (column, colIndex) => {\n return _openBlock(), _createElementBlock(\"div\", {\n key: colIndex,\n class: \"waterfall-column\"\n }, [(_openBlock(true), _createElementBlock(_Fragment, null, _renderList(column, file => {\n return _openBlock(), _createElementBlock(\"div\", {\n key: file.name,\n class: \"waterfall-item\",\n onClick: $event => $options.openPreview(file)\n }, [_createElementVNode(\"div\", {\n class: _normalizeClass([\"image-wrapper\", {\n loaded: file.loaded\n }])\n }, [$options.isImage(file) ? (_openBlock(), _createElementBlock(\"img\", {\n key: 0,\n src: $options.getFileUrl(file.name),\n alt: file.name,\n loading: \"lazy\",\n onLoad: $event => $options.onImageLoad($event, file),\n onError: _cache[2] || (_cache[2] = (...args) => $options.handleImageError && $options.handleImageError(...args))\n }, null, 40, _hoisted_19)) : $options.isVideo(file) ? (_openBlock(), _createElementBlock(\"video\", {\n key: 1,\n src: $options.getFileUrl(file.name),\n muted: \"\",\n loop: \"\",\n preload: \"metadata\",\n onLoadedmetadata: $event => $options.onVideoLoad($event, file),\n onMouseenter: _cache[3] || (_cache[3] = $event => $event.target.play()),\n onMouseleave: _cache[4] || (_cache[4] = $event => $event.target.pause())\n }, null, 40, _hoisted_20)) : (_openBlock(), _createElementBlock(\"div\", _hoisted_21, [...(_cache[21] || (_cache[21] = [_createElementVNode(\"svg\", {\n viewBox: \"0 0 24 24\",\n fill: \"currentColor\"\n }, [_createElementVNode(\"path\", {\n d: \"M14 2H6c-1.1 0-2 .9-2 2v16c0 1.1.9 2 2 2h12c1.1 0 2-.9 2-2V8l-6-6zm-1 7V3.5L18.5 9H13z\"\n })], -1)]))])), _createElementVNode(\"div\", _hoisted_22, [_createElementVNode(\"div\", _hoisted_23, [_createElementVNode(\"button\", {\n class: \"action-btn\",\n onClick: _withModifiers($event => $options.copyLink(file.name), [\"stop\"]),\n title: \"复制链接\"\n }, [...(_cache[22] || (_cache[22] = [_createElementVNode(\"svg\", {\n viewBox: \"0 0 24 24\",\n fill: \"currentColor\"\n }, [_createElementVNode(\"path\", {\n d: \"M3.9 12c0-1.71 1.39-3.1 3.1-3.1h4V7H7c-2.76 0-5 2.24-5 5s2.24 5 5 5h4v-1.9H7c-1.71 0-3.1-1.39-3.1-3.1zM8 13h8v-2H8v2zm9-6h-4v1.9h4c1.71 0 3.1 1.39 3.1 3.1s-1.39 3.1-3.1 3.1h-4V17h4c2.76 0 5-2.24 5-5s-2.24-5-5-5z\"\n })], -1)]))], 8, _hoisted_24), _createElementVNode(\"button\", {\n class: \"action-btn\",\n onClick: _withModifiers($event => $options.downloadFile(file.name), [\"stop\"]),\n title: \"下载\"\n }, [...(_cache[23] || (_cache[23] = [_createElementVNode(\"svg\", {\n viewBox: \"0 0 24 24\",\n fill: \"currentColor\"\n }, [_createElementVNode(\"path\", {\n d: \"M19 9h-4V3H9v6H5l7 7 7-7zM5 18v2h14v-2H5z\"\n })], -1)]))], 8, _hoisted_25)])])], 2)], 8, _hoisted_18);\n }), 128))]);\n }), 128))], 512), _createElementVNode(\"div\", _hoisted_26, [$data.loading && $data.files.length > 0 ? (_openBlock(), _createElementBlock(\"div\", _hoisted_27, [...(_cache[24] || (_cache[24] = [_createElementVNode(\"div\", {\n class: \"loading-spinner-small\"\n }, null, -1), _createElementVNode(\"span\", null, \"加载中...\", -1)]))])) : !$data.hasMore && $options.mediaFiles.length > 0 ? (_openBlock(), _createElementBlock(\"div\", _hoisted_28, \" 已加载全部 \")) : _createCommentVNode(\"\", true)], 512)], 512)), $data.previewVisible ? (_openBlock(), _createElementBlock(\"div\", {\n key: 3,\n class: \"preview-modal\",\n onClick: _cache[17] || (_cache[17] = _withModifiers((...args) => $options.closePreview && $options.closePreview(...args), [\"self\"]))\n }, [_createElementVNode(\"button\", {\n class: \"preview-close\",\n onClick: _cache[5] || (_cache[5] = _withModifiers((...args) => $options.closePreview && $options.closePreview(...args), [\"stop\"]))\n }, [...(_cache[25] || (_cache[25] = [_createElementVNode(\"svg\", {\n viewBox: \"0 0 24 24\",\n fill: \"currentColor\"\n }, [_createElementVNode(\"path\", {\n d: \"M19 6.41L17.59 5 12 10.59 6.41 5 5 6.41 10.59 12 5 17.59 6.41 19 12 13.41 17.59 19 19 17.59 13.41 12z\"\n })], -1)]))]), _createElementVNode(\"div\", {\n class: \"preview-content desktop-only\",\n onClick: _cache[6] || (_cache[6] = _withModifiers(() => {}, [\"stop\"]))\n }, [$options.currentPreviewFile && $options.isImage($options.currentPreviewFile) ? (_openBlock(), _createElementBlock(\"img\", {\n key: 0,\n src: $options.getFileUrl($options.currentPreviewFile.name),\n class: \"preview-image\",\n style: _normalizeStyle($options.desktopImageStyle),\n draggable: \"false\"\n }, null, 12, _hoisted_29)) : $options.currentPreviewFile && $options.isVideo($options.currentPreviewFile) ? (_openBlock(), _createElementBlock(\"video\", {\n key: 1,\n src: $options.getFileUrl($options.currentPreviewFile.name),\n controls: \"\",\n autoplay: \"\",\n class: \"preview-video\",\n style: _normalizeStyle($options.desktopImageStyle)\n }, null, 12, _hoisted_30)) : _createCommentVNode(\"\", true)]), _createElementVNode(\"div\", {\n class: \"preview-content mobile-only\",\n onClick: _cache[13] || (_cache[13] = _withModifiers(() => {}, [\"stop\"]))\n }, [_createElementVNode(\"div\", {\n class: \"swipe-viewport\",\n ref: \"mobileViewport\",\n onTouchstart: _cache[10] || (_cache[10] = (...args) => $options.onSwipeStart && $options.onSwipeStart(...args)),\n onTouchmove: _cache[11] || (_cache[11] = (...args) => $options.onSwipeMove && $options.onSwipeMove(...args)),\n onTouchend: _cache[12] || (_cache[12] = (...args) => $options.onSwipeEnd && $options.onSwipeEnd(...args))\n }, [_createElementVNode(\"div\", {\n class: \"swipe-track\",\n style: _normalizeStyle($options.swipeTrackStyle),\n onTransitionend: _cache[9] || (_cache[9] = (...args) => $options.onSwipeTransitionEnd && $options.onSwipeTransitionEnd(...args))\n }, [(_openBlock(true), _createElementBlock(_Fragment, null, _renderList($options.swipeWindow, (f, i) => {\n return _openBlock(), _createElementBlock(\"div\", {\n class: \"swipe-slide\",\n key: $options.getSlideKey(f, i)\n }, [f ? (_openBlock(), _createBlock(_component_TransformMedia, {\n key: 0,\n file: f,\n src: $options.getFileUrl(f.name),\n \"is-image\": $options.isImage(f),\n \"is-video\": $options.isVideo(f),\n onLock: _cache[7] || (_cache[7] = $event => $data.gestureLocked = true),\n onUnlock: _cache[8] || (_cache[8] = $event => $data.gestureLocked = false)\n }, null, 8, [\"file\", \"src\", \"is-image\", \"is-video\"])) : _createCommentVNode(\"\", true)]);\n }), 128))], 36)], 544)]), $data.previewIndex > 0 ? (_openBlock(), _createElementBlock(\"button\", {\n key: 0,\n class: \"preview-prev desktop-only\",\n onClick: _cache[14] || (_cache[14] = _withModifiers((...args) => $options.prevImage && $options.prevImage(...args), [\"stop\"]))\n }, [...(_cache[26] || (_cache[26] = [_createElementVNode(\"svg\", {\n viewBox: \"0 0 24 24\",\n fill: \"currentColor\"\n }, [_createElementVNode(\"path\", {\n d: \"M15.41 7.41L14 6l-6 6 6 6 1.41-1.41L10.83 12z\"\n })], -1)]))])) : _createCommentVNode(\"\", true), $data.previewIndex < $options.mediaFiles.length - 1 ? (_openBlock(), _createElementBlock(\"button\", {\n key: 1,\n class: \"preview-next desktop-only\",\n onClick: _cache[15] || (_cache[15] = _withModifiers((...args) => $options.nextImage && $options.nextImage(...args), [\"stop\"]))\n }, [...(_cache[27] || (_cache[27] = [_createElementVNode(\"svg\", {\n viewBox: \"0 0 24 24\",\n fill: \"currentColor\"\n }, [_createElementVNode(\"path\", {\n d: \"M10 6L8.59 7.41 13.17 12l-4.58 4.59L10 18l6-6z\"\n })], -1)]))])) : _createCommentVNode(\"\", true), _createElementVNode(\"button\", {\n class: \"rotate-btn desktop-only\",\n onClick: _cache[16] || (_cache[16] = _withModifiers((...args) => $options.rotateImage && $options.rotateImage(...args), [\"stop\"])),\n title: \"旋转90°\"\n }, [...(_cache[28] || (_cache[28] = [_createElementVNode(\"svg\", {\n viewBox: \"0 0 24 24\",\n fill: \"currentColor\"\n }, [_createElementVNode(\"path\", {\n d: \"M7.11 8.53L5.7 7.11C4.8 8.27 4.24 9.61 4.07 11h2.02c.14-.87.49-1.72 1.02-2.47zM6.09 13H4.07c.17 1.39.72 2.73 1.62 3.89l1.41-1.42c-.52-.75-.87-1.59-1.01-2.47zm1.01 5.32c1.16.9 2.51 1.44 3.9 1.61V17.9c-.87-.15-1.71-.49-2.46-1.03L7.1 18.32zM13 4.07V1L8.45 5.55 13 10V6.09c2.84.48 5 2.94 5 5.91s-2.16 5.43-5 5.91v2.02c3.95-.49 7-3.85 7-7.93s-3.05-7.44-7-7.93z\"\n })], -1)]))]), _createElementVNode(\"div\", _hoisted_31, _toDisplayString($data.previewIndex + 1) + \" / \" + _toDisplayString($options.mediaFiles.length), 1)])) : _createCommentVNode(\"\", true)]);\n}","import \"core-js/modules/es.array.push.js\";\nimport \"core-js/modules/es.iterator.constructor.js\";\nimport \"core-js/modules/es.iterator.filter.js\";\nimport \"core-js/modules/es.iterator.for-each.js\";\nimport \"core-js/modules/es.iterator.map.js\";\nimport axios from 'axios';\nimport { mapGetters } from 'vuex';\nimport TransformMedia from '@/components/TransformMedia.vue';\nexport default {\n name: 'PublicBrowse',\n components: {\n TransformMedia\n },\n data() {\n return {\n files: [],\n allowedDirs: [],\n rootDir: '',\n currentPath: '',\n totalCount: 0,\n loading: false,\n error: null,\n canRetry: true,\n hasMore: true,\n previewVisible: false,\n previewIndex: 0,\n observer: null,\n pageSize: 24,\n columnCount: 4,\n columnHeights: [0, 0, 0, 0],\n // 桌面端旋转\n imageRotation: 0,\n // 手机端滑动\n swipeX: 0,\n swipeStartX: 0,\n swipeStartY: 0,\n swipeStartT: 0,\n swipeActive: false,\n swipeAnimating: false,\n swipeDir: 0,\n viewportW: 0,\n // 手势锁定(子组件缩放/旋转时锁住轮播)\n gestureLocked: false\n };\n },\n computed: {\n ...mapGetters(['userConfig']),\n siteName() {\n return this.userConfig?.siteTitle || '公开相册';\n },\n rootDirName() {\n return this.rootDir.split('/').filter(Boolean).pop() || '根目录';\n },\n pathParts() {\n if (!this.currentPath || !this.rootDir) return [];\n const relative = this.currentPath.replace(this.rootDir, '').replace(/^\\/+/, '');\n return relative.split('/').filter(Boolean);\n },\n folders() {\n return this.files.filter(f => f.isFolder);\n },\n mediaFiles() {\n return this.files.filter(f => !f.isFolder);\n },\n columns() {\n const cols = Array.from({\n length: this.columnCount\n }, () => []);\n for (const file of this.mediaFiles) {\n const idx = file.columnIndex ?? 0;\n if (idx < this.columnCount) {\n cols[idx].push(file);\n } else {\n cols[0].push(file);\n }\n }\n return cols;\n },\n currentPreviewFile() {\n return this.mediaFiles[this.previewIndex];\n },\n prevPreviewFile() {\n return this.previewIndex > 0 ? this.mediaFiles[this.previewIndex - 1] : null;\n },\n nextPreviewFile() {\n return this.previewIndex < this.mediaFiles.length - 1 ? this.mediaFiles[this.previewIndex + 1] : null;\n },\n desktopImageStyle() {\n return {\n transform: `rotate(${this.imageRotation}deg)`,\n transition: 'transform 0.3s ease'\n };\n },\n swipeWindow() {\n return [this.prevPreviewFile, this.currentPreviewFile, this.nextPreviewFile];\n },\n swipeTrackStyle() {\n // 默认停在中间那页(-viewportW)\n const base = -this.viewportW;\n const x = base + this.swipeX;\n return {\n transform: `translate3d(${x}px, 0, 0)`,\n transition: this.swipeAnimating ? 'transform 0.28s ease' : 'none'\n };\n }\n },\n watch: {\n '$route.params.dir': {\n handler() {\n this.initFromRoute();\n }\n }\n },\n mounted() {\n this.initFromRoute();\n this.setupIntersectionObserver();\n this.updateColumnCount();\n window.addEventListener('resize', this.updateColumnCount);\n },\n beforeUnmount() {\n if (this.observer) {\n this.observer.disconnect();\n }\n window.removeEventListener('resize', this.updateColumnCount);\n },\n methods: {\n // 生成 slide key,切换时让子组件重新挂载以重置 transform\n getSlideKey(f, i) {\n if (!f) return `empty-${i}`;\n // 中间那张用 previewIndex 作为 key 的一部分,确保切换时重新挂载\n if (i === 1) return `${f.name}-${this.previewIndex}`;\n return f.name;\n },\n updateColumnCount() {\n const width = window.innerWidth;\n let newCount;\n if (width < 600) {\n newCount = 2;\n } else if (width < 900) {\n newCount = 3;\n } else {\n newCount = 4;\n }\n if (newCount !== this.columnCount) {\n this.columnCount = newCount;\n this.columnHeights = new Array(this.columnCount).fill(0);\n this.mediaFiles.forEach(f => {\n f.columnIndex = undefined;\n this.assignToColumn(f);\n });\n }\n },\n getShortestColumn() {\n let minIndex = 0;\n let minHeight = this.columnHeights[0];\n for (let i = 1; i < this.columnCount; i++) {\n if (this.columnHeights[i] < minHeight) {\n minHeight = this.columnHeights[i];\n minIndex = i;\n }\n }\n return minIndex;\n },\n assignToColumn(file, height = 200) {\n const colIndex = this.getShortestColumn();\n file.columnIndex = colIndex;\n this.columnHeights[colIndex] += height;\n },\n onImageLoad(event, file) {\n const img = event.target;\n const ratio = img.naturalHeight / img.naturalWidth;\n const height = 280 * ratio;\n if (file.columnIndex === undefined) {\n this.assignToColumn(file, height);\n }\n file.loaded = true;\n },\n onVideoLoad(event, file) {\n const video = event.target;\n const ratio = video.videoHeight / video.videoWidth;\n const height = 280 * ratio;\n if (file.columnIndex === undefined) {\n this.assignToColumn(file, height);\n }\n file.loaded = true;\n },\n setupIntersectionObserver() {\n this.observer = new IntersectionObserver(entries => {\n const entry = entries[0];\n if (entry.isIntersecting && this.hasMore && !this.loading) {\n this.loadMore();\n }\n }, {\n rootMargin: '200px'\n });\n },\n observeLoadTrigger() {\n this.$nextTick(() => {\n if (this.$refs.loadTrigger && this.observer) {\n this.observer.observe(this.$refs.loadTrigger);\n }\n });\n },\n async initFromRoute() {\n const dirParam = this.$route.params.dir || '';\n const dirPath = Array.isArray(dirParam) ? dirParam.join('/') : dirParam;\n if (!dirPath) {\n this.error = '请指定要浏览的目录,例如: /browse/landscape';\n this.canRetry = false;\n return;\n }\n const parts = dirPath.split('/').filter(Boolean);\n this.rootDir = parts[0];\n this.currentPath = dirPath;\n this.files = [];\n this.hasMore = true;\n this.columnHeights = new Array(this.columnCount).fill(0);\n await this.loadFiles();\n this.observeLoadTrigger();\n },\n async loadFiles() {\n this.loading = true;\n this.error = null;\n this.canRetry = true;\n try {\n const res = await axios.get(`/api/public/list?dir=${encodeURIComponent(this.currentPath)}&count=${this.pageSize}`);\n if (res.data.allowedDirs) {\n this.allowedDirs = res.data.allowedDirs;\n }\n const dirs = (res.data.directories || []).map(d => ({\n name: d,\n isFolder: true\n }));\n const files = (res.data.files || []).map(f => ({\n name: f.name,\n isFolder: false,\n metadata: f.metadata,\n columnIndex: undefined\n }));\n files.forEach(f => this.assignToColumn(f));\n this.files = [...dirs, ...files];\n this.totalCount = res.data.totalCount || this.files.length;\n this.hasMore = this.mediaFiles.length < this.totalCount;\n } catch (err) {\n if (err.response?.status === 403) {\n const msg = err.response?.data?.error || '';\n if (msg.includes('disabled')) {\n this.error = '公开浏览功能未启用';\n } else if (msg.includes('not allowed') || msg.includes('No public')) {\n this.error = '该目录不允许公开访问';\n } else {\n this.error = '访问被拒绝';\n }\n this.canRetry = false;\n } else {\n this.error = '加载失败,请重试';\n }\n } finally {\n this.loading = false;\n }\n },\n async loadMore() {\n if (this.loading || !this.hasMore) return;\n this.loading = true;\n try {\n const start = this.mediaFiles.length;\n const res = await axios.get(`/api/public/list?dir=${encodeURIComponent(this.currentPath)}&start=${start}&count=${this.pageSize}`);\n const moreFiles = (res.data.files || []).map(f => ({\n name: f.name,\n isFolder: false,\n metadata: f.metadata,\n columnIndex: undefined\n }));\n moreFiles.forEach(f => this.assignToColumn(f));\n this.files.push(...moreFiles);\n this.hasMore = this.mediaFiles.length < this.totalCount;\n } catch (err) {\n console.error('加载更多失败', err);\n } finally {\n this.loading = false;\n }\n },\n enterFolder(folderPath) {\n const newPath = folderPath.replace(/\\/+$/, '');\n this.$router.push(`/browse/${newPath}`);\n },\n goToRoot() {\n this.$router.push(`/browse/${this.rootDir}`);\n },\n goToPath(index) {\n const parts = this.pathParts.slice(0, index + 1);\n const newPath = this.rootDir + (parts.length ? '/' + parts.join('/') : '');\n this.$router.push(`/browse/${newPath}`);\n },\n getFolderName(path) {\n return path.split('/').filter(Boolean).pop() || path;\n },\n getFileUrl(name) {\n return `${window.location.origin}/file/${name}`;\n },\n isImage(file) {\n const ext = file.name.split('.').pop().toLowerCase();\n return ['jpg', 'jpeg', 'png', 'gif', 'webp', 'bmp', 'svg'].includes(ext);\n },\n isVideo(file) {\n const ext = file.name.split('.').pop().toLowerCase();\n return ['mp4', 'webm', 'ogg', 'mov'].includes(ext);\n },\n handleImageError(e) {\n e.target.style.display = 'none';\n },\n copyLink(name) {\n const url = this.getFileUrl(name);\n navigator.clipboard?.writeText(url).then(() => {\n this.showToast('已复制');\n }).catch(() => {\n const input = document.createElement('input');\n input.value = url;\n document.body.appendChild(input);\n input.select();\n document.execCommand('copy');\n document.body.removeChild(input);\n this.showToast('已复制');\n });\n },\n showToast(msg) {\n const existing = document.querySelector('.copy-toast');\n if (existing) existing.remove();\n const toast = document.createElement('div');\n toast.className = 'copy-toast';\n toast.textContent = msg;\n document.body.appendChild(toast);\n setTimeout(() => toast.classList.add('show'), 10);\n setTimeout(() => {\n toast.classList.remove('show');\n setTimeout(() => toast.remove(), 300);\n }, 1500);\n },\n downloadFile(name) {\n const link = document.createElement('a');\n link.href = this.getFileUrl(name);\n link.download = name.split('/').pop();\n link.click();\n },\n openPreview(file) {\n if (file.isFolder) return;\n const mediaIndex = this.mediaFiles.findIndex(f => f.name === file.name);\n if (mediaIndex >= 0) {\n this.previewIndex = mediaIndex;\n this.previewVisible = true;\n this.imageRotation = 0;\n this.gestureLocked = false;\n document.body.style.overflow = 'hidden';\n this.$nextTick(() => {\n this.viewportW = this.$refs.mobileViewport?.getBoundingClientRect().width || window.innerWidth;\n });\n }\n },\n closePreview() {\n this.previewVisible = false;\n this.imageRotation = 0;\n this.gestureLocked = false;\n document.body.style.overflow = '';\n },\n prevImage() {\n if (this.previewIndex > 0) {\n this.previewIndex--;\n this.imageRotation = 0;\n }\n },\n nextImage() {\n if (this.previewIndex < this.mediaFiles.length - 1) {\n this.previewIndex++;\n this.imageRotation = 0;\n }\n },\n rotateImage() {\n this.imageRotation += 90;\n // 动画结束后归一化(无动画)\n if (this.imageRotation >= 360) {\n setTimeout(() => {\n // 临时禁用 transition\n const el = this.$el.querySelector('.preview-image, .preview-video');\n if (el) {\n el.style.transition = 'none';\n this.imageRotation = 0;\n // 强制重绘后恢复 transition\n el.offsetHeight;\n el.style.transition = '';\n } else {\n this.imageRotation = 0;\n }\n }, 300);\n }\n },\n // 手机端滑动:开始\n onSwipeStart(e) {\n if (this.gestureLocked) return;\n if (this.swipeAnimating) return;\n const t = e.touches[0];\n this.swipeStartX = t.clientX;\n this.swipeStartY = t.clientY;\n this.swipeStartT = performance.now();\n this.swipeX = 0;\n this.swipeActive = false;\n this.viewportW = this.$refs.mobileViewport?.getBoundingClientRect().width || window.innerWidth;\n },\n // 手机端滑动:移动\n onSwipeMove(e) {\n if (this.gestureLocked) return;\n if (this.swipeAnimating) return;\n const t = e.touches[0];\n const dx = t.clientX - this.swipeStartX;\n const dy = t.clientY - this.swipeStartY;\n if (!this.swipeActive) {\n if (Math.abs(dx) < 8) return;\n if (Math.abs(dx) <= Math.abs(dy)) return;\n this.swipeActive = true;\n }\n e.preventDefault();\n let x = dx;\n // 边界阻尼:用 rubberBand 代替线性 *0.3\n if (this.previewIndex === 0 && x > 0) {\n x = this.rubberBand(x, this.viewportW, 0.55);\n } else if (this.previewIndex === this.mediaFiles.length - 1 && x < 0) {\n x = -this.rubberBand(-x, this.viewportW, 0.55);\n }\n this.swipeX = x;\n },\n // 手机端滑动:结束\n onSwipeEnd() {\n if (this.gestureLocked) return;\n if (this.swipeAnimating) return;\n if (!this.swipeActive) {\n this.swipeX = 0;\n return;\n }\n const dt = Math.max(1, performance.now() - this.swipeStartT);\n const vx = this.swipeX / dt;\n const threshold = this.viewportW * 0.2;\n let dir = 0;\n if (this.swipeX <= -threshold || vx <= -0.8) dir = +1;\n if (this.swipeX >= threshold || vx >= 0.8) dir = -1;\n if (dir === -1 && this.previewIndex === 0 || dir === +1 && this.previewIndex === this.mediaFiles.length - 1) {\n dir = 0;\n }\n this.swipeDir = dir;\n this.swipeAnimating = true;\n if (dir === +1) this.swipeX = -this.viewportW;else if (dir === -1) this.swipeX = +this.viewportW;else this.swipeX = 0;\n },\n onSwipeTransitionEnd() {\n if (!this.swipeAnimating) return;\n if (this.swipeDir === +1) this.previewIndex++;\n if (this.swipeDir === -1) this.previewIndex--;\n this.swipeAnimating = false;\n this.swipeDir = 0;\n this.swipeX = 0;\n },\n // iOS 风格橡皮筋阻尼函数\n rubberBand(distance, dimension, constant = 0.55) {\n return distance * dimension * constant / (dimension + constant * distance);\n }\n }\n};","/* unplugin-vue-components disabled */import { render } from \"./PublicBrowse.vue?vue&type=template&id=7aa9aaa3&scoped=true\"\nimport script from \"./PublicBrowse.vue?vue&type=script&lang=js\"\nexport * from \"./PublicBrowse.vue?vue&type=script&lang=js\"\n\nimport \"./PublicBrowse.vue?vue&type=style&index=0&id=7aa9aaa3&scoped=true&lang=css\"\n\nimport exportComponent from \"../../node_modules/vue-loader/dist/exportHelper.js\"\nconst __exports__ = /*#__PURE__*/exportComponent(script, [['render',render],['__scopeId',\"data-v-7aa9aaa3\"]])\n\nexport default __exports__"],"names":["_hoisted_1","_hoisted_2","render","_ctx","_cache","$props","$setup","$data","$options","class","ref","onPointerdown","args","onPointerDown","onPointermove","onPointerMove","onPointerup","onPointerUp","onPointercancel","onDblclick","onDblClick","isImage","key","src","draggable","style","mediaStyle","onLoad","isVideo","controls","autoplay","playsinline","name","props","file","type","Object","required","String","Boolean","default","data","pointers","Map","scale","rotation","rotatePreview","tx","ty","naturalWidth","naturalHeight","startScale","startRotation","startTx","startTy","startCenter","startDist","startAngle","dragging","dragStart","viewportRect","minScale","maxScale","gestureMode","computed","isActiveTransform","this","size","displayRotation","rotateShrink","p","Math","min","abs","k","sin","PI","finalScale","inGesture","transform","transition","transformOrigin","watch","v","$emit","methods","e","img","target","reset","clear","clamp","max","rubberBand","distance","dimension","constant","getViewportRect","$refs","viewport","getBoundingClientRect","getPanBounds","rect","maxX","maxY","vw","vh","width","height","$el","querySelector","iw","clientWidth","ih","clientHeight","sw","sh","applyBoundWithRubber","value","calcTwoPointer","pts","Array","from","values","p0","p1","dx","x","dy","y","dist","hypot","angle","atan2","center","normalizeAngle","deg","currentTarget","setPointerCapture","pointerId","set","clientX","clientY","has","preventDefault","scaleFactor","scaleChange","deltaAngle","angleChange","rotateStartDeg","pinchStartScale","cx0","left","cy0","top","cx1","cy1","rawX","rawY","delete","finishRotate","d","commitDeg","newRot","updateFillScale","nw","nh","rot","isRotated","effW","effH","containScale","coverScale","fillScale","__exports__","_hoisted_3","_hoisted_4","_hoisted_5","_hoisted_6","_hoisted_7","_hoisted_8","_hoisted_9","_hoisted_10","_hoisted_11","_hoisted_12","_hoisted_13","_hoisted_14","_hoisted_15","_hoisted_16","_hoisted_17","_hoisted_18","_hoisted_19","_hoisted_20","_hoisted_21","_hoisted_22","_hoisted_23","_hoisted_24","_hoisted_25","_hoisted_26","_hoisted_27","_hoisted_28","_hoisted_29","_hoisted_30","_hoisted_31","_component_TransformMedia","siteName","onClick","goToRoot","rootDirName","pathParts","part","index","$event","goToPath","totalCount","loading","files","length","error","canRetry","loadFiles","folders","folder","enterFolder","viewBox","fill","getFolderName","columns","column","colIndex","openPreview","loaded","getFileUrl","alt","onImageLoad","onError","handleImageError","muted","loop","preload","onLoadedmetadata","onVideoLoad","onMouseenter","play","onMouseleave","pause","copyLink","title","downloadFile","hasMore","mediaFiles","previewVisible","closePreview","currentPreviewFile","desktopImageStyle","onTouchstart","onSwipeStart","onTouchmove","onSwipeMove","onTouchend","onSwipeEnd","swipeTrackStyle","onTransitionend","onSwipeTransitionEnd","swipeWindow","f","i","getSlideKey","onLock","gestureLocked","onUnlock","previewIndex","prevImage","nextImage","rotateImage","components","TransformMedia","allowedDirs","rootDir","currentPath","observer","pageSize","columnCount","columnHeights","imageRotation","swipeX","swipeStartX","swipeStartY","swipeStartT","swipeActive","swipeAnimating","swipeDir","viewportW","userConfig","siteTitle","split","filter","pop","relative","replace","isFolder","cols","idx","columnIndex","push","prevPreviewFile","nextPreviewFile","base","handler","initFromRoute","mounted","setupIntersectionObserver","updateColumnCount","window","addEventListener","beforeUnmount","disconnect","removeEventListener","innerWidth","newCount","forEach","undefined","assignToColumn","getShortestColumn","minIndex","minHeight","event","ratio","video","videoHeight","videoWidth","IntersectionObserver","entries","entry","isIntersecting","loadMore","rootMargin","observeLoadTrigger","$nextTick","loadTrigger","observe","dirParam","$route","params","dir","dirPath","isArray","join","parts","res","axios","get","encodeURIComponent","dirs","directories","map","metadata","err","response","status","msg","includes","start","moreFiles","console","folderPath","newPath","$router","slice","path","location","origin","ext","toLowerCase","display","url","navigator","clipboard","writeText","then","showToast","catch","input","document","createElement","body","appendChild","select","execCommand","removeChild","existing","remove","toast","className","textContent","setTimeout","classList","add","link","href","download","click","mediaIndex","findIndex","overflow","mobileViewport","el","offsetHeight","t","touches","performance","now","dt","vx","threshold"],"ignoreList":[],"sourceRoot":""}
\ No newline at end of file
diff --git a/js/733.0149a1b8.js.map.gz b/js/733.0149a1b8.js.map.gz
new file mode 100644
index 0000000..c22adfb
Binary files /dev/null and b/js/733.0149a1b8.js.map.gz differ
diff --git a/js/app.c7b1facf.js b/js/app.c7b1facf.js
new file mode 100644
index 0000000..44d7a7e
--- /dev/null
+++ b/js/app.c7b1facf.js
@@ -0,0 +1,2 @@
+(function(){"use strict";var e={457:function(e,t,o){o(4114);var n=o(4373),r=o(4570),a=o.n(r),s=o(6915),i=o(1219);const l=n.A.create({baseURL:"/"});l.interceptors.request.use(e=>{if(e.withAuthCode){const t=a().get("authCode");t&&(e.headers["authCode"]=t)}return e},e=>Promise.reject(e)),l.interceptors.response.use(e=>e,e=>(e.config?.withAuthCode&&401===e.response?.status&&(i.nk.error("认证失败,请重新登录!"),s.A.push("/login")),Promise.reject(e))),t.A=l},3354:function(e,t,o){o(8111),o(7588);var n=o(5130),r=o(3888),a=o(4068),s=(o(9436),o(372),o(8950)),i=o(2353),l=o(292),u=o(6768);function c(e,t,o,n,r,a){const s=(0,u.g2)("router-view");return(0,u.uX)(),(0,u.Wv)(s)}var d=o(8401),p=o(3785),m={computed:{...(0,d.L8)(["userConfig","useDarkMode"])},mounted(){this.$nextTick(()=>{this.initOverlayScrollbars()})},watch:{useDarkMode(){this.setSiteIcon()}},methods:{initOverlayScrollbars(){try{if(p.ae.valid(document.body))return;(0,p.ae)(document.body,{scrollbars:{theme:"os-theme-dark",visibility:"auto",autoHide:"scroll",autoHideDelay:600,dragScroll:!0,clickScroll:!0},overflow:{x:"hidden",y:"scroll"}}),console.log("OverlayScrollbars initialized successfully")}catch(e){console.error("Failed to initialize OverlayScrollbars:",e)}},setSiteIcon(){const e=document.querySelectorAll('link[rel="icon"], link[rel="apple-touch-icon"], link[rel="mask-icon"]');e.forEach(e=>e.remove());const t=document.createElement("link"),o=document.createElement("link"),n=document.createElement("link");t.rel="icon",o.rel="apple-touch-icon",n.rel="mask-icon",this.useDarkMode?(t.href=this.userConfig?.siteIcon||"/logo-dark.png",o.href=this.userConfig?.siteIcon||"/logo-dark.png",n.href=this.userConfig?.siteIcon||"/logo-dark.png"):(t.href=this.userConfig?.siteIcon||"/logo.png",o.href=this.userConfig?.siteIcon||"/logo.png",n.href=this.userConfig?.siteIcon||"/logo.png"),document.head.appendChild(t),document.head.appendChild(o),document.head.appendChild(n)}}},f=o(1241);const h=(0,f.A)(m,[["render",c]]);var g=h,b=o(6915),k=o(5507),y=o(7477);o(1862),o(635);s.Yv.add(i.X7I);const C=(0,n.Ef)(g),v=(0,r.Zf)();C.component("font-awesome-icon",l.gc);for(const[M,E]of Object.entries(y))C.component(M,E);const A=()=>{const e=document.documentElement;let t;if(k.A.state.cusDarkMode&&null!==k.A.state.useDarkMode)t=k.A.state.useDarkMode;else{if(t=window.matchMedia("(prefers-color-scheme: dark)").matches,!t){const e=new Date,o=e.getHours();t=o>=22||o<6}k.A.commit("setUseDarkMode",t)}t?e.classList.add("dark"):e.classList.remove("dark")},U=e=>{const t=document.documentElement;e?t.classList.add("dark"):t.classList.remove("dark")},w=e=>{document.title=e?.siteTitle||"Sanyue ImgHub"},S=(e,t)=>{const o=document.querySelectorAll('link[rel="icon"], link[rel="apple-touch-icon"], link[rel="mask-icon"]');o.forEach(e=>e.remove());const n=document.createElement("link"),r=document.createElement("link"),a=document.createElement("link");n.rel="icon",r.rel="apple-touch-icon",a.rel="mask-icon",e?(n.href=t?.siteIcon||"/logo-dark.png",r.href=t?.siteIcon||"/logo-dark.png",a.href=t?.siteIcon||"/logo-dark.png"):(n.href=t?.siteIcon||"/logo.png",r.href=t?.siteIcon||"/logo.png",a.href=t?.siteIcon||"/logo.png"),document.head.appendChild(n),document.head.appendChild(r),document.head.appendChild(a)};k.A.dispatch("fetchUserConfig").then(()=>{A(),w(k.A.getters.userConfig),S(k.A.state.useDarkMode,k.A.getters.userConfig),k.A.subscribe((e,t)=>{"setUseDarkMode"===e.type&&k.A.state.cusDarkMode&&(U(t.useDarkMode),S(t.useDarkMode,k.A.getters.userConfig)),"setCusDarkMode"!==e.type||e.payload||(A(),S(k.A.state.useDarkMode,k.A.getters.userConfig))}),C.use(k.A).use(b.A).use(a.A).mount("#app")}).catch(e=>{console.error("Failed to load user configuration:",e),C.use(k.A).use(b.A).use(a.A).use(v).mount("#app")})},5507:function(e,t,o){o(8111),o(1701);var n=o(8401),r=o(457),a=o(5131);t.A=(0,n.y$)({state:{userConfig:null,bingWallPapers:[],credentials:null,uploadMethod:"default",uploadCopyUrlForm:"",compressConfig:{customerCompress:!0,compressQuality:4,compressBar:5,serverCompress:!0},storeUploadChannel:"",storeAutoRetry:!0,storeUploadNameType:"",uploadFolder:"",customUrlSettings:{useCustomUrl:"false",customUrlPrefix:""},adminUrlSettings:{useCustomUrl:"false",customUrlPrefix:""},autoReUpload:!0,useDarkMode:null,cusDarkMode:!1},getters:{userConfig:e=>e.userConfig,bingWallPapers:e=>e.bingWallPapers,credentials:e=>e.credentials,storeUploadMethod:e=>e.uploadMethod,uploadCopyUrlForm:e=>e.uploadCopyUrlForm,compressConfig:e=>e.compressConfig,storeUploadChannel:e=>e.storeUploadChannel,storeUploadNameType:e=>e.storeUploadNameType,customUrlSettings:e=>e.customUrlSettings,storeAutoRetry:e=>e.storeAutoRetry,adminUrlSettings:e=>e.adminUrlSettings,storeUploadFolder:e=>e.uploadFolder||localStorage.getItem("uploadFolder")||"",useDarkMode:e=>e.useDarkMode,cusDarkMode:e=>e.cusDarkMode,storeAutoReUpload:e=>e.autoReUpload},mutations:{setUserConfig(e,t){e.userConfig=t},setBingWallPapers(e,t){e.bingWallPapers=t},setCredentials(e,t){e.credentials=t},setUploadMethod(e,t){e.uploadMethod=t},setUploadCopyUrlForm(e,t){e.uploadCopyUrlForm=t},setCompressConfig(e,{key:t,value:o}){e.compressConfig[t]=o},setStoreUploadChannel(e,t){e.storeUploadChannel=t},setStoreUploadNameType(e,t){e.storeUploadNameType=t},setCustomUrlSettings(e,{key:t,value:o}){e.customUrlSettings[t]=o},setStoreAutoRetry(e,t){e.storeAutoRetry=t},setAdminUrlSettings(e,{key:t,value:o}){e.adminUrlSettings[t]=o},setUseDarkMode(e,t){e.useDarkMode=t},setCusDarkMode(e,t){e.cusDarkMode=t},setStoreUploadFolder(e,t){e.uploadFolder=t,localStorage.setItem("uploadFolder",t)},setStoreAutoReUpload(e,t){e.autoReUpload=t}},actions:{async fetchUserConfig({commit:e}){try{const t=await r.A.get("/api/userConfig");e("setUserConfig",t.data)}catch(t){console.log(t)}},async fetchBingWallPapers({commit:e}){try{const t=await r.A.get("/api/bing/wallpaper"),o=t.data.data,n=o.map(e=>({url:"https://www.bing.com"+e.url}));await Promise.all(n.map(e=>new Promise((t,o)=>{const n=new Image;n.onload=t,n.onerror=o,n.src=e.url}))),e("setBingWallPapers",n)}catch(t){console.log(t)}}},modules:{},plugins:[(0,a.A)()]})},6915:function(e,t,o){o(4979);var n=o(8512),r=o(1219),a=o(4570),s=o.n(a),i=o(5507),l=o(457);const u=(e,t,o)=>{const n=i.A.getters.credentials;if(null===n&&"adminLogin"!==e.name){const e=btoa("unset:unset");l.A.get("/api/manage/check",{headers:{Authorization:"Basic "+e},withCredentials:!0}).then(t=>{if(200!==t.status)throw new Error("认证失败!");i.A.commit("setCredentials",e),o()}).catch(e=>{r.nk.error("请先认证!"),o({name:"adminLogin"})})}else o()},c=(e,t,o)=>{let n=s().get("authCode");null===n&&"login"!==e.name?l.A.post("/api/login",{authCode:"unset"}).then(e=>{if(200!==e.status)throw new Error("认证失败!");s().set("authCode","unset","14d"),n="unset",o()}).catch(e=>{r.nk.error("请先认证!"),o({name:"login"})}):o()},d=[{path:"/",name:"home",component:()=>Promise.all([o.e(171),o.e(747)]).then(o.bind(o,3695)),beforeEnter:c},{path:"/login",name:"login",component:()=>Promise.all([o.e(171),o.e(672)]).then(o.bind(o,2560))},{path:"/dashboard",name:"dashboard",component:()=>Promise.all([o.e(239),o.e(482)]).then(o.bind(o,1482)),beforeEnter:u},{path:"/customerConfig",name:"customerConfig",component:()=>Promise.all([o.e(239),o.e(419)]).then(o.bind(o,8419)),beforeEnter:u},{path:"/systemConfig",name:"systemConfig",component:()=>Promise.all([o.e(239),o.e(643)]).then(o.bind(o,6643)),beforeEnter:u},{path:"/adminLogin",name:"adminLogin",component:()=>Promise.all([o.e(171),o.e(698)]).then(o.bind(o,3950))},{path:"/blockimg",name:"blockimg",component:()=>o.e(917).then(o.bind(o,4917))},{path:"/whiteliston",name:"whiteliston",component:()=>o.e(845).then(o.bind(o,7226))},{path:"/browse/:dir*",name:"publicBrowse",component:()=>o.e(733).then(o.bind(o,6733))},{path:"/:pathMatch(.*)*",name:"notFound",component:()=>o.e(226).then(o.bind(o,8226))}],p=(0,n.aE)({history:(0,n.LA)("/"),routes:d});t.A=p}},t={};function o(n){var r=t[n];if(void 0!==r)return r.exports;var a=t[n]={exports:{}};return e[n].call(a.exports,a,a.exports,o),a.exports}o.m=e,function(){var e=[];o.O=function(t,n,r,a){if(!n){var s=1/0;for(c=0;c=a)&&Object.keys(o.O).every(function(e){return o.O[e](n[l])})?n.splice(l--,1):(i=!1,a0&&e[c-1][2]>a;c--)e[c]=e[c-1];e[c]=[n,r,a]}}(),function(){o.n=function(e){var t=e&&e.__esModule?function(){return e["default"]}:function(){return e};return o.d(t,{a:t}),t}}(),function(){o.d=function(e,t){for(var n in t)o.o(t,n)&&!o.o(e,n)&&Object.defineProperty(e,n,{enumerable:!0,get:t[n]})}}(),function(){o.f={},o.e=function(e){return Promise.all(Object.keys(o.f).reduce(function(t,n){return o.f[n](e,t),t},[]))}}(),function(){o.u=function(e){return"js/"+e+"."+{171:"833208d0",226:"577ea134",239:"32572dba",419:"99e00aec",482:"e302adaf",643:"d1f24e26",672:"11bec4e1",698:"77f859a8",733:"0149a1b8",747:"51eee506",845:"6b4cb4a5",917:"5bf3db27"}[e]+".js"}}(),function(){o.miniCssF=function(e){return"css/"+e+"."+{226:"6548e7b4",239:"6563616d",419:"1d235d6e",482:"f42e1415",643:"74c1c29d",672:"aed3581b",698:"aed3581b",733:"ce2542a6",747:"e115ba5f",845:"6b45e8e3",917:"8529ad1e"}[e]+".css"}}(),function(){o.g=function(){if("object"===typeof globalThis)return globalThis;try{return this||new Function("return this")()}catch(e){if("object"===typeof window)return window}}()}(),function(){o.o=function(e,t){return Object.prototype.hasOwnProperty.call(e,t)}}(),function(){var e={},t="sanyue_imghub:";o.l=function(n,r,a,s){if(e[n])e[n].push(r);else{var i,l;if(void 0!==a)for(var u=document.getElementsByTagName("script"),c=0;c {\r\n // 如果配置中标记了withAuthCode,则添加authCode到header\r\n if (config.withAuthCode) {\r\n const authCode = cookies.get('authCode');\r\n if (authCode) {\r\n config.headers['authCode'] = authCode;\r\n }\r\n }\r\n return config;\r\n },\r\n (error) => {\r\n return Promise.reject(error);\r\n }\r\n);\r\n\r\n// 响应拦截器\r\ninstance.interceptors.response.use(\r\n (response) => {\r\n return response;\r\n },\r\n (error) => {\r\n // 如果请求配置了withAuthCode且返回401,则跳转到登录页\r\n if (error.config?.withAuthCode && error.response?.status === 401) {\r\n ElMessage.error('认证失败,请重新登录!');\r\n router.push('/login');\r\n }\r\n return Promise.reject(error);\r\n }\r\n);\r\n\r\nexport default instance;\r\n","import { resolveComponent as _resolveComponent, openBlock as _openBlock, createBlock as _createBlock } from \"vue\";\nexport function render(_ctx, _cache, $props, $setup, $data, $options) {\n const _component_router_view = _resolveComponent(\"router-view\");\n return _openBlock(), _createBlock(_component_router_view);\n}","import \"core-js/modules/es.iterator.constructor.js\";\nimport \"core-js/modules/es.iterator.for-each.js\";\nimport { mapGetters } from 'vuex';\nimport { OverlayScrollbars } from 'overlayscrollbars';\nexport default {\n computed: {\n ...mapGetters(['userConfig', 'useDarkMode'])\n },\n mounted() {\n // 初始化 OverlayScrollbars 悬浮滚动条\n this.$nextTick(() => {\n this.initOverlayScrollbars();\n });\n },\n watch: {\n useDarkMode() {\n this.setSiteIcon();\n }\n },\n methods: {\n initOverlayScrollbars() {\n try {\n // 检查是否已经初始化\n if (OverlayScrollbars.valid(document.body)) {\n return;\n }\n\n // 应用到 body 实现全局悬浮滚动条\n OverlayScrollbars(document.body, {\n scrollbars: {\n theme: 'os-theme-dark',\n visibility: 'auto',\n autoHide: 'scroll',\n autoHideDelay: 600,\n dragScroll: true,\n clickScroll: true\n },\n overflow: {\n x: 'hidden',\n y: 'scroll'\n }\n });\n console.log('OverlayScrollbars initialized successfully');\n } catch (error) {\n console.error('Failed to initialize OverlayScrollbars:', error);\n }\n },\n setSiteIcon() {\n // 同时更改 icon apple-touch-icon 和 mask-icon\n const existingIcons = document.querySelectorAll('link[rel=\"icon\"], link[rel=\"apple-touch-icon\"], link[rel=\"mask-icon\"]');\n existingIcons.forEach(icon => icon.remove());\n const iconLink = document.createElement('link');\n const appleIconLink = document.createElement('link');\n const maskIconLink = document.createElement('link');\n iconLink.rel = 'icon';\n appleIconLink.rel = 'apple-touch-icon';\n maskIconLink.rel = 'mask-icon';\n if (this.useDarkMode) {\n iconLink.href = this.userConfig?.siteIcon || '/logo-dark.png';\n appleIconLink.href = this.userConfig?.siteIcon || '/logo-dark.png';\n maskIconLink.href = this.userConfig?.siteIcon || '/logo-dark.png';\n } else {\n iconLink.href = this.userConfig?.siteIcon || '/logo.png';\n appleIconLink.href = this.userConfig?.siteIcon || '/logo.png';\n maskIconLink.href = this.userConfig?.siteIcon || '/logo.png';\n }\n document.head.appendChild(iconLink);\n document.head.appendChild(appleIconLink);\n document.head.appendChild(maskIconLink);\n }\n }\n};","/* unplugin-vue-components disabled */import { render } from \"./App.vue?vue&type=template&id=7d6c298a\"\nimport script from \"./App.vue?vue&type=script&lang=js\"\nexport * from \"./App.vue?vue&type=script&lang=js\"\n\nimport \"./App.vue?vue&type=style&index=0&id=7d6c298a&lang=css\"\nimport \"./App.vue?vue&type=style&index=1&id=7d6c298a&lang=css\"\n\nimport exportComponent from \"../node_modules/vue-loader/dist/exportHelper.js\"\nconst __exports__ = /*#__PURE__*/exportComponent(script, [['render',render]])\n\nexport default __exports__","import { createApp } from 'vue'\r\nimport { createHead } from '@vueuse/head'; // 导入 createHead\r\nimport ElementPlus from 'element-plus'\r\nimport 'element-plus/es/components/message/style/css'\r\nimport 'element-plus/es/components/message-box/style/css'\r\n\r\nimport { library } from '@fortawesome/fontawesome-svg-core';\r\nimport { fas } from '@fortawesome/free-solid-svg-icons'; // 引入所有 solid 图标\r\nimport { FontAwesomeIcon } from '@fortawesome/vue-fontawesome';\r\n\r\nimport App from './App.vue'\r\nimport router from './router'\r\nimport store from './store'\r\n\r\nimport * as ElementPlusIconsVue from '@element-plus/icons-vue'\r\nimport 'element-plus/theme-chalk/dark/css-vars.css'\r\nimport './styles/global.css'\r\n\r\n// OverlayScrollbars 悬浮滚动条\r\nimport 'overlayscrollbars/overlayscrollbars.css'\r\n\r\n\r\nlibrary.add(fas);\r\n\r\nconst app = createApp(App);\r\nconst head = createHead(); // 创建 head 对象\r\n\r\napp.component('font-awesome-icon', FontAwesomeIcon);\r\nfor (const [key, component] of Object.entries(ElementPlusIconsVue)) {\r\n app.component(key, component)\r\n}\r\n\r\n// 根据 useDarkMode 的值添加或移除 dark 类\r\nconst initDarkModeClass = () => {\r\n const htmlElement = document.documentElement;\r\n let isDarkMode;\r\n\r\n // 判断用户是否是自定义模式\r\n if (store.state.cusDarkMode && store.state.useDarkMode !== null) {\r\n // 用户手动设置了暗色模式\r\n isDarkMode = store.state.useDarkMode;\r\n } else {\r\n // 跟随系统模式或时间\r\n isDarkMode = window.matchMedia('(prefers-color-scheme: dark)').matches;\r\n if (!isDarkMode) {\r\n const now = new Date();\r\n const hour = now.getHours();\r\n isDarkMode = hour >= 22 || hour < 6;\r\n }\r\n // 更新 useDarkMode 的值\r\n store.commit('setUseDarkMode', isDarkMode);\r\n }\r\n\r\n if (isDarkMode) {\r\n htmlElement.classList.add('dark');\r\n } else {\r\n htmlElement.classList.remove('dark');\r\n }\r\n};\r\n\r\nconst applyDarkModeClass = (isDarkMode) => {\r\n const htmlElement = document.documentElement;\r\n if (isDarkMode) {\r\n htmlElement.classList.add('dark');\r\n } else {\r\n htmlElement.classList.remove('dark');\r\n }\r\n};\r\n\r\n// 预设网站标题的函数\r\nconst presetSiteTitle = (userConfig) => {\r\n document.title = userConfig?.siteTitle || 'Sanyue ImgHub';\r\n};\r\n\r\n// 预设网站图标的函数\r\nconst presetSiteIcon = (isDarkMode, userConfig) => {\r\n // 同时更改 icon apple-touch-icon 和 mask-icon\r\n const existingIcons = document.querySelectorAll('link[rel=\"icon\"], link[rel=\"apple-touch-icon\"], link[rel=\"mask-icon\"]');\r\n existingIcons.forEach(icon => icon.remove());\r\n\r\n const iconLink = document.createElement('link');\r\n const appleIconLink = document.createElement('link');\r\n const maskIconLink = document.createElement('link');\r\n iconLink.rel = 'icon';\r\n appleIconLink.rel = 'apple-touch-icon';\r\n maskIconLink.rel = 'mask-icon';\r\n\r\n if (isDarkMode) {\r\n iconLink.href = userConfig?.siteIcon || '/logo-dark.png';\r\n appleIconLink.href = userConfig?.siteIcon || '/logo-dark.png';\r\n maskIconLink.href = userConfig?.siteIcon || '/logo-dark.png';\r\n } else {\r\n iconLink.href = userConfig?.siteIcon || '/logo.png';\r\n appleIconLink.href = userConfig?.siteIcon || '/logo.png';\r\n maskIconLink.href = userConfig?.siteIcon || '/logo.png';\r\n }\r\n\r\n document.head.appendChild(iconLink);\r\n document.head.appendChild(appleIconLink);\r\n document.head.appendChild(maskIconLink);\r\n};\r\n\r\nstore.dispatch('fetchUserConfig').then(() => {\r\n // 初始化时应用 dark 模式\r\n initDarkModeClass();\r\n\r\n // 预设网站标题和图标\r\n presetSiteTitle(store.getters.userConfig);\r\n presetSiteIcon(store.state.useDarkMode, store.getters.userConfig);\r\n\r\n // 监听 useDarkMode 和 cusDarkMode 的变化\r\n store.subscribe((mutation, state) => {\r\n if (mutation.type === 'setUseDarkMode' && store.state.cusDarkMode) {\r\n applyDarkModeClass(state.useDarkMode);\r\n // 同时更新网站图标\r\n presetSiteIcon(state.useDarkMode, store.getters.userConfig);\r\n }\r\n\r\n // 监听 cusDarkMode 变化,当设置为 false 时重新初始化\r\n if (mutation.type === 'setCusDarkMode' && !mutation.payload) {\r\n // 切换到跟随系统模式,重新初始化\r\n initDarkModeClass();\r\n // 同时更新网站图标\r\n presetSiteIcon(store.state.useDarkMode, store.getters.userConfig);\r\n }\r\n });\r\n\r\n app.use(store).use(router).use(ElementPlus).mount('#app');\r\n}).catch(error => {\r\n console.error('Failed to load user configuration:', error);\r\n app.use(store).use(router).use(ElementPlus).use(head).mount('#app');\r\n})\r\n","import { createStore } from 'vuex'\r\nimport axios from '@/utils/axios';\r\nimport createPersistedState from 'vuex-persistedstate';\r\n\r\nexport default createStore({\r\n state: {\r\n userConfig: null,\r\n bingWallPapers: [],\r\n credentials: null,\r\n uploadMethod: 'default',\r\n uploadCopyUrlForm: '',\r\n compressConfig: {\r\n customerCompress: true,\r\n compressQuality: 4,\r\n compressBar: 5,\r\n serverCompress: true,\r\n },\r\n storeUploadChannel: '',\r\n storeAutoRetry: true,\r\n storeUploadNameType: '',\r\n uploadFolder: '',\r\n customUrlSettings: {\r\n useCustomUrl: 'false',\r\n customUrlPrefix: '',\r\n },\r\n adminUrlSettings: {\r\n useCustomUrl: 'false',\r\n customUrlPrefix: '',\r\n },\r\n autoReUpload: true,\r\n // 深色模式\r\n useDarkMode: null,\r\n cusDarkMode: false,\r\n },\r\n getters: {\r\n userConfig: state => state.userConfig,\r\n bingWallPapers: state => state.bingWallPapers,\r\n credentials: state => state.credentials,\r\n storeUploadMethod: state => state.uploadMethod,\r\n uploadCopyUrlForm: state => state.uploadCopyUrlForm,\r\n compressConfig: state => state.compressConfig,\r\n storeUploadChannel: state => state.storeUploadChannel,\r\n storeUploadNameType: state => state.storeUploadNameType,\r\n customUrlSettings: state => state.customUrlSettings,\r\n storeAutoRetry: state => state.storeAutoRetry,\r\n adminUrlSettings: state => state.adminUrlSettings,\r\n storeUploadFolder: (state) => {\r\n return state.uploadFolder || localStorage.getItem('uploadFolder') || ''\r\n },\r\n useDarkMode: state => state.useDarkMode,\r\n cusDarkMode: state => state.cusDarkMode,\r\n storeAutoReUpload: state => state.autoReUpload,\r\n },\r\n mutations: {\r\n setUserConfig(state, userConfig) {\r\n state.userConfig = userConfig;\r\n },\r\n setBingWallPapers(state, bingWallPapers) {\r\n state.bingWallPapers = bingWallPapers;\r\n },\r\n setCredentials(state, credentials) {\r\n state.credentials = credentials;\r\n },\r\n setUploadMethod(state, uploadMethod) {\r\n state.uploadMethod = uploadMethod;\r\n },\r\n setUploadCopyUrlForm(state, uploadCopyUrlForm) {\r\n state.uploadCopyUrlForm = uploadCopyUrlForm;\r\n },\r\n setCompressConfig(state, { key, value }) {\r\n state.compressConfig[key] = value;\r\n },\r\n setStoreUploadChannel(state, uploadChannel) {\r\n state.storeUploadChannel = uploadChannel;\r\n },\r\n setStoreUploadNameType(state, storeUploadNameType) {\r\n state.storeUploadNameType = storeUploadNameType;\r\n },\r\n setCustomUrlSettings(state, { key, value }) {\r\n state.customUrlSettings[key] = value;\r\n },\r\n setStoreAutoRetry(state, storeAutoRetry) {\r\n state.storeAutoRetry = storeAutoRetry;\r\n },\r\n setAdminUrlSettings(state, { key, value }) {\r\n state.adminUrlSettings[key] = value;\r\n },\r\n setUseDarkMode(state, useDarkMode) {\r\n state.useDarkMode = useDarkMode;\r\n },\r\n setCusDarkMode(state, cusDarkMode) {\r\n state.cusDarkMode = cusDarkMode;\r\n },\r\n setStoreUploadFolder(state, folder) {\r\n state.uploadFolder = folder\r\n localStorage.setItem('uploadFolder', folder)\r\n },\r\n setStoreAutoReUpload(state, autoReUpload) {\r\n state.autoReUpload = autoReUpload;\r\n }\r\n },\r\n actions: {\r\n async fetchUserConfig({ commit }) {\r\n try {\r\n const response = await axios.get('/api/userConfig');\r\n commit('setUserConfig', response.data);\r\n } catch (error) {\r\n console.log(error);\r\n }\r\n },\r\n async fetchBingWallPapers({ commit }) {\r\n try {\r\n const response = await axios.get('/api/bing/wallpaper');\r\n const wallpapers = response.data.data;\r\n const bingWallPapers = wallpapers.map(wallpaper => {\r\n return {\r\n url: 'https://www.bing.com' + wallpaper.url,\r\n };\r\n }\r\n );\r\n\r\n //预加载图片,阻塞直到图片加载完成\r\n await Promise.all(bingWallPapers.map(wallpaper => {\r\n return new Promise((resolve, reject) => {\r\n const img = new Image();\r\n img.onload = resolve;\r\n img.onerror = reject;\r\n img.src = wallpaper.url;\r\n });\r\n }));\r\n commit('setBingWallPapers', bingWallPapers);\r\n } catch (error) {\r\n console.log(error);\r\n }\r\n }\r\n },\r\n modules: {\r\n },\r\n plugins: [createPersistedState()]\r\n})\r\n","import { createRouter, createWebHistory } from 'vue-router'\r\nimport { ElMessage } from 'element-plus'\r\nimport cookies from 'vue-cookies'\r\nimport store from '../store'\r\nimport axios from '@/utils/axios'\r\n\r\n// 通用的管理员认证守卫\r\nconst adminAuthGuard = (to, from, next) => {\r\n // 从store中获取凭据\r\n const credentials = store.getters.credentials\r\n if (credentials === null && to.name !== 'adminLogin') {\r\n // 尝试未设置密码的情况\r\n const credentials = btoa('unset:unset')\r\n axios.get('/api/manage/check', {\r\n headers: {\r\n 'Authorization': 'Basic ' + credentials\r\n },\r\n withCredentials: true\r\n }).then(res => {\r\n if (res.status !== 200) {\r\n throw new Error('认证失败!')\r\n }\r\n store.commit('setCredentials', credentials)\r\n next()\r\n }).catch(err => {\r\n ElMessage.error('请先认证!')\r\n next({ name: 'adminLogin' })\r\n })\r\n } else {\r\n next()\r\n }\r\n}\r\n\r\n// 通用的用户认证守卫\r\nconst userAuthGuard = (to, from, next) => {\r\n let authCode = cookies.get('authCode');\r\n if (authCode === null && to.name !== 'login') {\r\n // 尝试未设置密码的情况\r\n axios.post('/api/login', {\r\n authCode: 'unset'\r\n }).then(res => {\r\n if (res.status !== 200) {\r\n throw new Error('认证失败!')\r\n }\r\n cookies.set('authCode', 'unset', '14d')\r\n authCode = 'unset'\r\n next()\r\n }).catch(err => {\r\n ElMessage.error('请先认证!')\r\n next({ name: 'login' })\r\n })\r\n } else {\r\n next()\r\n }\r\n}\r\n\r\nconst routes = [\r\n {\r\n path: '/',\r\n name: 'home',\r\n component: () => import('../views/UploadHome.vue'),\r\n beforeEnter: userAuthGuard\r\n },\r\n {\r\n path: '/login',\r\n name: 'login',\r\n component: () => import('../views/Login.vue')\r\n },\r\n {\r\n path: '/dashboard',\r\n name: 'dashboard',\r\n component: () => import('../views/AdminDashBoard.vue'),\r\n beforeEnter: adminAuthGuard\r\n },\r\n {\r\n path: '/customerConfig',\r\n name: 'customerConfig',\r\n component: () => import('../views/CustomerConfig.vue'),\r\n beforeEnter: adminAuthGuard\r\n },\r\n {\r\n path: '/systemConfig',\r\n name: 'systemConfig',\r\n component: () => import('../views/SystemConfig.vue'),\r\n beforeEnter: adminAuthGuard\r\n },\r\n {\r\n path: '/adminLogin',\r\n name: 'adminLogin',\r\n component: () => import('../views/AdminLogin.vue'),\r\n },\r\n {\r\n path: '/blockimg',\r\n name: 'blockimg',\r\n component: () => import('../views/BlockImage.vue'),\r\n },\r\n {\r\n path: '/whiteliston',\r\n name: 'whiteliston',\r\n component: () => import('../views/WhiteListOn.vue'),\r\n },\r\n {\r\n path: '/browse/:dir*',\r\n name: 'publicBrowse',\r\n component: () => import('../views/PublicBrowse.vue'),\r\n },\r\n {\r\n path: '/:pathMatch(.*)*',\r\n name: 'notFound',\r\n component: () => import('../views/NotFound.vue'),\r\n },\r\n]\r\n\r\nconst router = createRouter({\r\n history: createWebHistory(process.env.BASE_URL),\r\n routes\r\n})\r\n\r\nexport default router\r\n","// The module cache\nvar __webpack_module_cache__ = {};\n\n// The require function\nfunction __webpack_require__(moduleId) {\n\t// Check if module is in cache\n\tvar cachedModule = __webpack_module_cache__[moduleId];\n\tif (cachedModule !== undefined) {\n\t\treturn cachedModule.exports;\n\t}\n\t// Create a new module (and put it into the cache)\n\tvar module = __webpack_module_cache__[moduleId] = {\n\t\t// no module.id needed\n\t\t// no module.loaded needed\n\t\texports: {}\n\t};\n\n\t// Execute the module function\n\t__webpack_modules__[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n\t// Return the exports of the module\n\treturn module.exports;\n}\n\n// expose the modules object (__webpack_modules__)\n__webpack_require__.m = __webpack_modules__;\n\n","var deferred = [];\n__webpack_require__.O = function(result, chunkIds, fn, priority) {\n\tif(chunkIds) {\n\t\tpriority = priority || 0;\n\t\tfor(var i = deferred.length; i > 0 && deferred[i - 1][2] > priority; i--) deferred[i] = deferred[i - 1];\n\t\tdeferred[i] = [chunkIds, fn, priority];\n\t\treturn;\n\t}\n\tvar notFulfilled = Infinity;\n\tfor (var i = 0; i < deferred.length; i++) {\n\t\tvar chunkIds = deferred[i][0];\n\t\tvar fn = deferred[i][1];\n\t\tvar priority = deferred[i][2];\n\t\tvar fulfilled = true;\n\t\tfor (var j = 0; j < chunkIds.length; j++) {\n\t\t\tif ((priority & 1 === 0 || notFulfilled >= priority) && Object.keys(__webpack_require__.O).every(function(key) { return __webpack_require__.O[key](chunkIds[j]); })) {\n\t\t\t\tchunkIds.splice(j--, 1);\n\t\t\t} else {\n\t\t\t\tfulfilled = false;\n\t\t\t\tif(priority < notFulfilled) notFulfilled = priority;\n\t\t\t}\n\t\t}\n\t\tif(fulfilled) {\n\t\t\tdeferred.splice(i--, 1)\n\t\t\tvar r = fn();\n\t\t\tif (r !== undefined) result = r;\n\t\t}\n\t}\n\treturn result;\n};","// getDefaultExport function for compatibility with non-harmony modules\n__webpack_require__.n = function(module) {\n\tvar getter = module && module.__esModule ?\n\t\tfunction() { return module['default']; } :\n\t\tfunction() { return module; };\n\t__webpack_require__.d(getter, { a: getter });\n\treturn getter;\n};","// define getter functions for harmony exports\n__webpack_require__.d = function(exports, definition) {\n\tfor(var key in definition) {\n\t\tif(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {\n\t\t\tObject.defineProperty(exports, key, { enumerable: true, get: definition[key] });\n\t\t}\n\t}\n};","__webpack_require__.f = {};\n// This file contains only the entry chunk.\n// The chunk loading function for additional chunks\n__webpack_require__.e = function(chunkId) {\n\treturn Promise.all(Object.keys(__webpack_require__.f).reduce(function(promises, key) {\n\t\t__webpack_require__.f[key](chunkId, promises);\n\t\treturn promises;\n\t}, []));\n};","// This function allow to reference async chunks\n__webpack_require__.u = function(chunkId) {\n\t// return url for filenames based on template\n\treturn \"js/\" + chunkId + \".\" + {\"171\":\"833208d0\",\"226\":\"577ea134\",\"239\":\"32572dba\",\"419\":\"99e00aec\",\"482\":\"e302adaf\",\"643\":\"d1f24e26\",\"672\":\"11bec4e1\",\"698\":\"77f859a8\",\"733\":\"0149a1b8\",\"747\":\"51eee506\",\"845\":\"6b4cb4a5\",\"917\":\"5bf3db27\"}[chunkId] + \".js\";\n};","// This function allow to reference async chunks\n__webpack_require__.miniCssF = function(chunkId) {\n\t// return url for filenames based on template\n\treturn \"css/\" + chunkId + \".\" + {\"226\":\"6548e7b4\",\"239\":\"6563616d\",\"419\":\"1d235d6e\",\"482\":\"f42e1415\",\"643\":\"74c1c29d\",\"672\":\"aed3581b\",\"698\":\"aed3581b\",\"733\":\"ce2542a6\",\"747\":\"e115ba5f\",\"845\":\"6b45e8e3\",\"917\":\"8529ad1e\"}[chunkId] + \".css\";\n};","__webpack_require__.g = (function() {\n\tif (typeof globalThis === 'object') return globalThis;\n\ttry {\n\t\treturn this || new Function('return this')();\n\t} catch (e) {\n\t\tif (typeof window === 'object') return window;\n\t}\n})();","__webpack_require__.o = function(obj, prop) { return Object.prototype.hasOwnProperty.call(obj, prop); }","var inProgress = {};\nvar dataWebpackPrefix = \"sanyue_imghub:\";\n// loadScript function to load a script via script tag\n__webpack_require__.l = function(url, done, key, chunkId) {\n\tif(inProgress[url]) { inProgress[url].push(done); return; }\n\tvar script, needAttach;\n\tif(key !== undefined) {\n\t\tvar scripts = document.getElementsByTagName(\"script\");\n\t\tfor(var i = 0; i < scripts.length; i++) {\n\t\t\tvar s = scripts[i];\n\t\t\tif(s.getAttribute(\"src\") == url || s.getAttribute(\"data-webpack\") == dataWebpackPrefix + key) { script = s; break; }\n\t\t}\n\t}\n\tif(!script) {\n\t\tneedAttach = true;\n\t\tscript = document.createElement('script');\n\n\t\tscript.charset = 'utf-8';\n\t\tif (__webpack_require__.nc) {\n\t\t\tscript.setAttribute(\"nonce\", __webpack_require__.nc);\n\t\t}\n\t\tscript.setAttribute(\"data-webpack\", dataWebpackPrefix + key);\n\n\t\tscript.src = url;\n\t}\n\tinProgress[url] = [done];\n\tvar onScriptComplete = function(prev, event) {\n\t\t// avoid mem leaks in IE.\n\t\tscript.onerror = script.onload = null;\n\t\tclearTimeout(timeout);\n\t\tvar doneFns = inProgress[url];\n\t\tdelete inProgress[url];\n\t\tscript.parentNode && script.parentNode.removeChild(script);\n\t\tdoneFns && doneFns.forEach(function(fn) { return fn(event); });\n\t\tif(prev) return prev(event);\n\t}\n\tvar timeout = setTimeout(onScriptComplete.bind(null, undefined, { type: 'timeout', target: script }), 120000);\n\tscript.onerror = onScriptComplete.bind(null, script.onerror);\n\tscript.onload = onScriptComplete.bind(null, script.onload);\n\tneedAttach && document.head.appendChild(script);\n};","// define __esModule on exports\n__webpack_require__.r = function(exports) {\n\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n\t}\n\tObject.defineProperty(exports, '__esModule', { value: true });\n};","__webpack_require__.p = \"/\";","if (typeof document === \"undefined\") return;\nvar createStylesheet = function(chunkId, fullhref, oldTag, resolve, reject) {\n\tvar linkTag = document.createElement(\"link\");\n\n\tlinkTag.rel = \"stylesheet\";\n\tlinkTag.type = \"text/css\";\n\tif (__webpack_require__.nc) {\n\t\tlinkTag.nonce = __webpack_require__.nc;\n\t}\n\tvar onLinkComplete = function(event) {\n\t\t// avoid mem leaks.\n\t\tlinkTag.onerror = linkTag.onload = null;\n\t\tif (event.type === 'load') {\n\t\t\tresolve();\n\t\t} else {\n\t\t\tvar errorType = event && event.type;\n\t\t\tvar realHref = event && event.target && event.target.href || fullhref;\n\t\t\tvar err = new Error(\"Loading CSS chunk \" + chunkId + \" failed.\\n(\" + errorType + \": \" + realHref + \")\");\n\t\t\terr.name = \"ChunkLoadError\";\n\t\t\terr.code = \"CSS_CHUNK_LOAD_FAILED\";\n\t\t\terr.type = errorType;\n\t\t\terr.request = realHref;\n\t\t\tif (linkTag.parentNode) linkTag.parentNode.removeChild(linkTag)\n\t\t\treject(err);\n\t\t}\n\t}\n\tlinkTag.onerror = linkTag.onload = onLinkComplete;\n\tlinkTag.href = fullhref;\n\n\n\tif (oldTag) {\n\t\toldTag.parentNode.insertBefore(linkTag, oldTag.nextSibling);\n\t} else {\n\t\tdocument.head.appendChild(linkTag);\n\t}\n\treturn linkTag;\n};\nvar findStylesheet = function(href, fullhref) {\n\tvar existingLinkTags = document.getElementsByTagName(\"link\");\n\tfor(var i = 0; i < existingLinkTags.length; i++) {\n\t\tvar tag = existingLinkTags[i];\n\t\tvar dataHref = tag.getAttribute(\"data-href\") || tag.getAttribute(\"href\");\n\t\tif(tag.rel === \"stylesheet\" && (dataHref === href || dataHref === fullhref)) return tag;\n\t}\n\tvar existingStyleTags = document.getElementsByTagName(\"style\");\n\tfor(var i = 0; i < existingStyleTags.length; i++) {\n\t\tvar tag = existingStyleTags[i];\n\t\tvar dataHref = tag.getAttribute(\"data-href\");\n\t\tif(dataHref === href || dataHref === fullhref) return tag;\n\t}\n};\nvar loadStylesheet = function(chunkId) {\n\treturn new Promise(function(resolve, reject) {\n\t\tvar href = __webpack_require__.miniCssF(chunkId);\n\t\tvar fullhref = __webpack_require__.p + href;\n\t\tif(findStylesheet(href, fullhref)) return resolve();\n\t\tcreateStylesheet(chunkId, fullhref, null, resolve, reject);\n\t});\n}\n// object to store loaded CSS chunks\nvar installedCssChunks = {\n\t524: 0\n};\n\n__webpack_require__.f.miniCss = function(chunkId, promises) {\n\tvar cssChunks = {\"226\":1,\"239\":1,\"419\":1,\"482\":1,\"643\":1,\"672\":1,\"698\":1,\"733\":1,\"747\":1,\"845\":1,\"917\":1};\n\tif(installedCssChunks[chunkId]) promises.push(installedCssChunks[chunkId]);\n\telse if(installedCssChunks[chunkId] !== 0 && cssChunks[chunkId]) {\n\t\tpromises.push(installedCssChunks[chunkId] = loadStylesheet(chunkId).then(function() {\n\t\t\tinstalledCssChunks[chunkId] = 0;\n\t\t}, function(e) {\n\t\t\tdelete installedCssChunks[chunkId];\n\t\t\tthrow e;\n\t\t}));\n\t}\n};\n\n// no hmr\n\n// no prefetching\n\n// no preloaded","// no baseURI\n\n// object to store loaded and loading chunks\n// undefined = chunk not loaded, null = chunk preloaded/prefetched\n// [resolve, reject, Promise] = chunk loading, 0 = chunk loaded\nvar installedChunks = {\n\t524: 0\n};\n\n__webpack_require__.f.j = function(chunkId, promises) {\n\t\t// JSONP chunk loading for javascript\n\t\tvar installedChunkData = __webpack_require__.o(installedChunks, chunkId) ? installedChunks[chunkId] : undefined;\n\t\tif(installedChunkData !== 0) { // 0 means \"already installed\".\n\n\t\t\t// a Promise means \"currently loading\".\n\t\t\tif(installedChunkData) {\n\t\t\t\tpromises.push(installedChunkData[2]);\n\t\t\t} else {\n\t\t\t\tif(true) { // all chunks have JS\n\t\t\t\t\t// setup Promise in chunk cache\n\t\t\t\t\tvar promise = new Promise(function(resolve, reject) { installedChunkData = installedChunks[chunkId] = [resolve, reject]; });\n\t\t\t\t\tpromises.push(installedChunkData[2] = promise);\n\n\t\t\t\t\t// start chunk loading\n\t\t\t\t\tvar url = __webpack_require__.p + __webpack_require__.u(chunkId);\n\t\t\t\t\t// create error before stack unwound to get useful stacktrace later\n\t\t\t\t\tvar error = new Error();\n\t\t\t\t\tvar loadingEnded = function(event) {\n\t\t\t\t\t\tif(__webpack_require__.o(installedChunks, chunkId)) {\n\t\t\t\t\t\t\tinstalledChunkData = installedChunks[chunkId];\n\t\t\t\t\t\t\tif(installedChunkData !== 0) installedChunks[chunkId] = undefined;\n\t\t\t\t\t\t\tif(installedChunkData) {\n\t\t\t\t\t\t\t\tvar errorType = event && (event.type === 'load' ? 'missing' : event.type);\n\t\t\t\t\t\t\t\tvar realSrc = event && event.target && event.target.src;\n\t\t\t\t\t\t\t\terror.message = 'Loading chunk ' + chunkId + ' failed.\\n(' + errorType + ': ' + realSrc + ')';\n\t\t\t\t\t\t\t\terror.name = 'ChunkLoadError';\n\t\t\t\t\t\t\t\terror.type = errorType;\n\t\t\t\t\t\t\t\terror.request = realSrc;\n\t\t\t\t\t\t\t\tinstalledChunkData[1](error);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t};\n\t\t\t\t\t__webpack_require__.l(url, loadingEnded, \"chunk-\" + chunkId, chunkId);\n\t\t\t\t}\n\t\t\t}\n\t\t}\n};\n\n// no prefetching\n\n// no preloaded\n\n// no HMR\n\n// no HMR manifest\n\n__webpack_require__.O.j = function(chunkId) { return installedChunks[chunkId] === 0; };\n\n// install a JSONP callback for chunk loading\nvar webpackJsonpCallback = function(parentChunkLoadingFunction, data) {\n\tvar chunkIds = data[0];\n\tvar moreModules = data[1];\n\tvar runtime = data[2];\n\t// add \"moreModules\" to the modules object,\n\t// then flag all \"chunkIds\" as loaded and fire callback\n\tvar moduleId, chunkId, i = 0;\n\tif(chunkIds.some(function(id) { return installedChunks[id] !== 0; })) {\n\t\tfor(moduleId in moreModules) {\n\t\t\tif(__webpack_require__.o(moreModules, moduleId)) {\n\t\t\t\t__webpack_require__.m[moduleId] = moreModules[moduleId];\n\t\t\t}\n\t\t}\n\t\tif(runtime) var result = runtime(__webpack_require__);\n\t}\n\tif(parentChunkLoadingFunction) parentChunkLoadingFunction(data);\n\tfor(;i < chunkIds.length; i++) {\n\t\tchunkId = chunkIds[i];\n\t\tif(__webpack_require__.o(installedChunks, chunkId) && installedChunks[chunkId]) {\n\t\t\tinstalledChunks[chunkId][0]();\n\t\t}\n\t\tinstalledChunks[chunkId] = 0;\n\t}\n\treturn __webpack_require__.O(result);\n}\n\nvar chunkLoadingGlobal = self[\"webpackChunksanyue_imghub\"] = self[\"webpackChunksanyue_imghub\"] || [];\nchunkLoadingGlobal.forEach(webpackJsonpCallback.bind(null, 0));\nchunkLoadingGlobal.push = webpackJsonpCallback.bind(null, chunkLoadingGlobal.push.bind(chunkLoadingGlobal));","// startup\n// Load entry module and return exports\n// This entry module depends on other loaded chunks and execution need to be delayed\nvar __webpack_exports__ = __webpack_require__.O(undefined, [504], function() { return __webpack_require__(3354); })\n__webpack_exports__ = __webpack_require__.O(__webpack_exports__);\n"],"names":["instance","axios","create","baseURL","interceptors","request","use","config","withAuthCode","authCode","cookies","headers","error","Promise","reject","response","status","ElMessage","router","push","render","_ctx","_cache","$props","$setup","$data","$options","_component_router_view","computed","mounted","this","$nextTick","initOverlayScrollbars","watch","useDarkMode","setSiteIcon","methods","valid","document","body","scrollbars","theme","visibility","autoHide","autoHideDelay","dragScroll","clickScroll","overflow","x","y","console","log","existingIcons","querySelectorAll","forEach","icon","remove","iconLink","createElement","appleIconLink","maskIconLink","rel","href","userConfig","siteIcon","head","appendChild","__exports__","library","add","fas","app","createApp","App","createHead","component","FontAwesomeIcon","key","Object","entries","ElementPlusIconsVue","initDarkModeClass","htmlElement","documentElement","isDarkMode","store","state","cusDarkMode","window","matchMedia","matches","now","Date","hour","getHours","commit","classList","applyDarkModeClass","presetSiteTitle","title","siteTitle","presetSiteIcon","dispatch","then","getters","subscribe","mutation","type","payload","ElementPlus","mount","catch","createStore","bingWallPapers","credentials","uploadMethod","uploadCopyUrlForm","compressConfig","customerCompress","compressQuality","compressBar","serverCompress","storeUploadChannel","storeAutoRetry","storeUploadNameType","uploadFolder","customUrlSettings","useCustomUrl","customUrlPrefix","adminUrlSettings","autoReUpload","storeUploadMethod","storeUploadFolder","localStorage","getItem","storeAutoReUpload","mutations","setUserConfig","setBingWallPapers","setCredentials","setUploadMethod","setUploadCopyUrlForm","setCompressConfig","value","setStoreUploadChannel","uploadChannel","setStoreUploadNameType","setCustomUrlSettings","setStoreAutoRetry","setAdminUrlSettings","setUseDarkMode","setCusDarkMode","setStoreUploadFolder","folder","setItem","setStoreAutoReUpload","actions","fetchUserConfig","get","data","fetchBingWallPapers","wallpapers","map","wallpaper","url","all","resolve","img","Image","onload","onerror","src","modules","plugins","createPersistedState","adminAuthGuard","to","from","next","name","btoa","withCredentials","res","Error","err","userAuthGuard","post","routes","path","beforeEnter","createRouter","history","createWebHistory","process","__webpack_module_cache__","__webpack_require__","moduleId","cachedModule","undefined","exports","module","__webpack_modules__","call","m","deferred","O","result","chunkIds","fn","priority","notFulfilled","Infinity","i","length","fulfilled","j","keys","every","splice","r","n","getter","__esModule","d","a","definition","o","defineProperty","enumerable","f","e","chunkId","reduce","promises","u","miniCssF","g","globalThis","Function","obj","prop","prototype","hasOwnProperty","inProgress","dataWebpackPrefix","l","done","script","needAttach","scripts","getElementsByTagName","s","getAttribute","charset","nc","setAttribute","onScriptComplete","prev","event","clearTimeout","timeout","doneFns","parentNode","removeChild","setTimeout","bind","target","Symbol","toStringTag","p","createStylesheet","fullhref","oldTag","linkTag","nonce","onLinkComplete","errorType","realHref","code","insertBefore","nextSibling","findStylesheet","existingLinkTags","tag","dataHref","existingStyleTags","loadStylesheet","installedCssChunks","miniCss","cssChunks","installedChunks","installedChunkData","promise","loadingEnded","realSrc","message","webpackJsonpCallback","parentChunkLoadingFunction","moreModules","runtime","some","id","chunkLoadingGlobal","self","__webpack_exports__"],"ignoreList":[],"sourceRoot":""}
\ No newline at end of file
diff --git a/js/app.c7b1facf.js.map.gz b/js/app.c7b1facf.js.map.gz
new file mode 100644
index 0000000..7e68a3c
Binary files /dev/null and b/js/app.c7b1facf.js.map.gz differ