slider-captcha-sdk 1.0.0 → 1.0.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +2 -2
- package/dist/README.md +2 -2
- package/dist/index.cjs.js +1 -0
- package/dist/index.cjs.js.map +1 -0
- package/dist/index.d.ts +50 -41
- package/dist/index.esm.js +1 -0
- package/dist/index.esm.js.map +1 -0
- package/dist/index.js +1 -0
- package/dist/index.js.map +1 -0
- package/dist/index.min.js +59 -0
- package/dist/index.min.js.map +1 -0
- package/dist/password-validator.cjs.js +1 -0
- package/dist/password-validator.cjs.js.map +1 -0
- package/dist/password-validator.esm.js +1 -0
- package/dist/password-validator.esm.js.map +1 -0
- package/dist/password-validator.js +1 -0
- package/dist/password-validator.js.map +1 -0
- package/dist/password-validator.min.js +17 -0
- package/dist/password-validator.min.js.map +1 -0
- package/dist/slider-captcha.cjs.js +1 -1
- package/dist/slider-captcha.cjs.js.map +1 -0
- package/dist/slider-captcha.esm.js +1 -1
- package/dist/slider-captcha.esm.js.map +1 -0
- package/dist/slider-captcha.js +1 -1
- package/dist/slider-captcha.js.map +1 -0
- package/dist/slider-captcha.min.js +42 -1
- package/dist/slider-captcha.min.js.map +1 -1
- package/dist/slider-captcha.ultra.min.js +50 -0
- package/package.json +31 -11
package/dist/index.js
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
!function(t,e){"object"==typeof exports&&"undefined"!=typeof module?e(exports):"function"==typeof define&&define.amd?define(["exports"],e):e((t="undefined"!=typeof globalThis?globalThis:t||self).SliderCaptchaSDK={})}(this,function(t){"use strict";class PopupSliderCaptcha{static DEFAULTS={width:320,height:155,sliderSize:38,maxRetries:3,timeout:3e4,apiUrl:"/api/captcha",verifyUrl:"/api/captcha/verify",throttleDelay:16,clickMaskClose:!1};static CSS_CLASSES={overlay:"slider-captcha-overlay",modal:"slider-captcha-modal",header:"slider-captcha-header",container:"slider-captcha-container",track:"slider-captcha-track",btn:"slider-captcha-btn",hint:"slider-captcha-hint",loading:"slider-captcha-loading",error:"slider-captcha-error"};static getStyles(){return":root{--sc-primary:#409eff;--sc-success:#67c23a;--sc-danger:#f56c6c;--sc-border:#e4e7eb;--sc-bg:linear-gradient(90deg, #f7f9fa 0%, #e8f4fd 100%);--sc-text:#333;--sc-text-light:#999;--sc-shadow:0 4px 20px rgba(0,0,0,.3);--sc-radius:8px;--sc-transition:.3s ease}.slider-captcha-overlay{position:fixed;top:0;left:0;width:100%;height:100%;background:rgba(0,0,0,.5);z-index:9999;display:none;justify-content:center;align-items:center;opacity:0;transition:opacity var(--sc-transition)}.slider-captcha-overlay.show{opacity:1}.slider-captcha-modal{background:#fff;border-radius:var(--sc-radius);padding:20px;box-shadow:var(--sc-shadow);position:relative;max-width:90vw;max-height:90vh;transform:scale(.8) translateY(-20px);opacity:0;transition:all var(--sc-transition)}.slider-captcha-modal.show{transform:scale(1) translateY(0);opacity:1}.slider-captcha-header{display:flex;justify-content:space-between;align-items:center;margin-bottom:15px;padding-bottom:10px;border-bottom:1px solid var(--sc-border)}.slider-captcha-container{display:flex;align-items:center;position:relative;border-radius:4px;overflow:hidden;margin-bottom:15px;background:#837a7a;justify-content:center}.slider-captcha-track{width:100%;height:40px;line-height:40px;background:var(--sc-bg);border:1px solid var(--sc-border);border-radius:20px;position:relative;margin-bottom:15px;overflow:hidden}.slider-captcha-btn{width:38px;height:38px;background:#fff;border:1px solid #ccc;border-radius:50%;position:absolute;top:0;left:0;cursor:pointer;display:flex;align-items:center;justify-content:center;box-shadow:0 2px 4px rgba(0,0,0,.1);transition:all var(--sc-transition);user-select:none;z-index:1}.slider-captcha-loading{position:absolute;top:0;left:0;width:100%;height:100%;background:rgba(255,255,255,.6);display:flex;align-items:center;justify-content:center;flex-direction:column;color:#666;font-size:14px;z-index:10;border-radius:4px}.slider-captcha-error{color:var(--sc-danger);font-size:12px;text-align:center;margin-top:10px;display:none}.slider-captcha-title{margin:0;font-size:16px;color:var(--sc-text)}.slider-captcha-close,.slider-captcha-refresh{background:none;border:none;cursor:pointer;color:var(--sc-text-light);padding:0;width:30px;height:30px;display:flex;align-items:center;justify-content:center;border-radius:50%;transition:all var(--sc-transition);position:relative;font-size:0}.slider-captcha-close::before,.slider-captcha-close::after{content:'';position:absolute;width:16px;height:2px;background-color:var(--sc-text-light);border-radius:1px;transition:all var(--sc-transition)}.slider-captcha-close::before{transform:rotate(45deg)}.slider-captcha-close::after{transform:rotate(-45deg)}.slider-captcha-close:hover{background:#f5f5f5;transform:scale(1.1)}.slider-captcha-close:hover::before,.slider-captcha-close:hover::after{background-color:var(--sc-danger)}.slider-captcha-refresh{margin-left:10px}.slider-captcha-refresh svg{width:20px;height:20px;fill:var(--sc-text-light);transition:all var(--sc-transition)}.slider-captcha-refresh:hover{background:#f5f5f5;transform:scale(1.1)}.slider-captcha-refresh:hover svg{fill:var(--sc-primary);transform:rotate(180deg)}.slider-captcha-floating-time{position:absolute;bottom:-40px;left:50%;transform:translateX(-50%);color:#fff;font-size:12px;white-space:nowrap;opacity:0;pointer-events:none;z-index:10;transition:all var(--sc-transition);background:#fff;padding:2px 15px;border-radius:10px}.slider-captcha-floating-time.show{opacity:1;transform:translateX(-50%) translateY(-45px)}.slider-captcha-floating-time.success{color:var(--sc-success)}.slider-captcha-floating-time.fail{color:var(--sc-danger)}.slider-captcha-bg{width:100%;height:100%;object-fit:cover;display:block}.slider-captcha-piece{position:absolute;top:0;left:0;cursor:pointer;transition:none;z-index:2}.slider-captcha-finger{position:absolute;top:50%;left:10px;transform:translateY(-50%);font-size:20px;animation:fingerSlide 2s ease-in-out infinite;pointer-events:none;z-index:1;opacity:.6}.slider-captcha-hint{position:absolute;top:50%;left:50%;transform:translate(-50%,-50%);color:var(--sc-text-light);font-size:14px;pointer-events:none;z-index:1;transition:all var(--sc-transition)}.slider-captcha-header-buttons{display:flex;align-items:center}@keyframes fingerSlide{0%{left:10px;opacity:.6}50%{opacity:1}100%{left:calc(50% - 10px);opacity:.6}}"}constructor(t={}){this.options={...PopupSliderCaptcha.DEFAULTS,...t},this.elements={},this.state=this.createInitialState(),this.captchaData=null,this.times=[],this.startTime=null,this.eventListeners=[],this.timers=new Set,this.rafId=null,this.cachedDimensions=null,this.imageCache=new Map,this.throttledHandleMove=this.throttle(t=>this.handleMove(t),this.options.throttleDelay),this.init()}createInitialState(){return{isVisible:!1,isDragging:!1,currentX:0,startX:0,retryCount:0,isLoading:!1}}init(){this.injectStyles(),this.createElements(),this.bindEvents()}injectStyles(){if(document.querySelector("#slider-captcha-styles"))return;const t=document.createElement("style");t.id="slider-captcha-styles",t.textContent=PopupSliderCaptcha.getStyles(),document.head.appendChild(t)}createElements(){const{elements:t,options:e}=this;[["overlay","div",PopupSliderCaptcha.CSS_CLASSES.overlay],["modal","div",PopupSliderCaptcha.CSS_CLASSES.modal],["header","div",PopupSliderCaptcha.CSS_CLASSES.header],["title","h3","slider-captcha-title","安全验证"],["closeBtn","button","slider-captcha-close"],["refreshBtn","button","slider-captcha-refresh"],["container","div",PopupSliderCaptcha.CSS_CLASSES.container],["backgroundImg","img","slider-captcha-bg"],["sliderImg","img","slider-captcha-piece"],["loadingText","div",PopupSliderCaptcha.CSS_CLASSES.loading,"加载中..."],["floatingTime","div","slider-captcha-floating-time"],["track","div",PopupSliderCaptcha.CSS_CLASSES.track],["fingerAnimation","div","slider-captcha-finger","👉"],["btn","div",PopupSliderCaptcha.CSS_CLASSES.btn],["icon","div","","→"],["hint","div",PopupSliderCaptcha.CSS_CLASSES.hint,"向右滑动完成验证"],["error","div",PopupSliderCaptcha.CSS_CLASSES.error]].forEach(([e,i,s,a])=>{t[e]=this.createElement(i,s,a)}),t.container.style.cssText=`width:${e.width}px;height:${e.height}px`,t.refreshBtn.innerHTML='<svg viewBox="0 0 24 24"><path d="M17.65,6.35C16.2,4.9 14.21,4 12,4A8,8 0 0,0 4,12A8,8 0 0,0 12,20C15.73,20 18.84,17.45 19.73,14H17.65C16.83,16.33 14.61,18 12,18A6,6 0 0,1 6,12A6,6 0 0,1 12,6C13.66,6 15.14,6.69 16.22,7.78L13,11H20V4L17.65,6.35Z"/></svg>',this.assembleDOM(),this.setInitialState()}createElement(t,e="",i=""){const s=document.createElement(t);return e&&(s.className=e),i&&(s.textContent=i),s}assembleDOM(){const{elements:t}=this,e=this.createElement("div","slider-captcha-header-buttons");e.append(t.refreshBtn,t.closeBtn),t.header.append(t.title,e),t.container.append(t.backgroundImg,t.sliderImg,t.loadingText,t.floatingTime),t.btn.appendChild(t.icon),t.track.append(t.fingerAnimation,t.btn,t.hint),t.modal.append(t.header,t.container,t.track,t.error),t.overlay.appendChild(t.modal),document.body.appendChild(t.overlay)}setInitialState(){Object.assign(this.elements.container.style,{display:"none"}),Object.assign(this.elements.track.style,{display:"none"})}bindEvents(){const{elements:t}=this;this.addEventListener(t.closeBtn,"click",()=>this.hide()),this.addEventListener(t.refreshBtn,"click",()=>this.refresh()),this.addEventListener(t.overlay,"click",e=>{e.target===t.overlay&&this.options.clickMaskClose&&this.hide()}),this.addEventListener(document,"keydown",t=>{"Escape"===t.key&&this.state.isVisible&&this.hide()}),this.addEventListener(document,"visibilitychange",()=>this.handleVisibilityChange()),this.bindSliderEvents()}bindSliderEvents(){const{elements:t}=this,e={start:t=>this.handleStart(t),move:this.throttledHandleMove,end:()=>this.handleEnd()};this.addEventListener(t.btn,"mousedown",e.start),this.addEventListener(t.btn,"touchstart",e.start),this.addEventListener(t.sliderImg,"mousedown",e.start),this.addEventListener(t.sliderImg,"touchstart",e.start),this.addEventListener(document,"mousemove",e.move,{passive:!1}),this.addEventListener(document,"touchmove",e.move,{passive:!1}),this.addEventListener(document,"mouseup",e.end),this.addEventListener(document,"touchend",e.end)}addEventListener(t,e,i,s={}){t&&"function"==typeof i&&(t.addEventListener(e,i,s),this.eventListeners.push({element:t,event:e,handler:i,options:s}))}removeAllEventListeners(){this.eventListeners.forEach(({element:t,event:e,handler:i,options:s})=>{try{t?.removeEventListener?.(e,i,s)}catch(t){}}),this.eventListeners.length=0}getDimensions(){if(!this.cachedDimensions){const t=this.elements.track.offsetWidth,e=this.elements.btn.offsetWidth;this.cachedDimensions={trackWidth:t,btnWidth:e,maxX:t-e}}return this.cachedDimensions}getPosition(){const{maxX:t}=this.getDimensions(),e=this.state.currentX/t;return Math.round(e*(this.options.width-this.options.sliderSize))}handleStart(t){!this.captchaData||this.state.isDragging||this.state.isLoading||(t.preventDefault(),this.state.isDragging=!0,this.state.startX=this.getClientX(t)-this.state.currentX,this.startTime=this.startTime||Date.now(),this.times=[{time:Date.now(),position:this.getPosition()}],this.setTransition(!1),this.updateUIState("dragging"),this.cachedDimensions=null)}handleMove(t){if(!this.state.isDragging)return;t.preventDefault();const e=this.getClientX(t)-this.state.startX,{maxX:i}=this.getDimensions();this.state.currentX=Math.max(0,Math.min(e,i)),this.times.push({time:Date.now(),position:this.getPosition()}),this.rafId&&cancelAnimationFrame(this.rafId),this.rafId=requestAnimationFrame(()=>this.updateSliderPosition())}handleEnd(){this.state.isDragging&&(this.times.push({time:Date.now(),position:this.getPosition()}),this.state.isDragging=!1,this.rafId&&(cancelAnimationFrame(this.rafId),this.rafId=null),this.verify())}handleVisibilityChange(){const t=document.hidden?"paused":"running";this.elements.fingerAnimation&&(this.elements.fingerAnimation.style.animationPlayState=t)}getClientX(t){return t.type.includes("touch")?t.touches[0].clientX:t.clientX}setTransition(t){const e=t?"all 0.3s ease":"none";requestAnimationFrame(()=>{this.elements.btn.style.transition=e,this.elements.sliderImg.style.transition=e})}updateUIState(t){const{elements:e}=this,i={dragging:()=>{e.hint.style.opacity="0",e.fingerAnimation.style.display="none"},success:()=>{Object.assign(e.btn.style,{background:"var(--sc-success)"}),Object.assign(e.icon.style,{innerHTML:"✓",color:"white"}),e.icon.innerHTML="✓"},fail:()=>{Object.assign(e.btn.style,{background:"var(--sc-danger)"}),Object.assign(e.icon.style,{innerHTML:"✗",color:"white"}),e.icon.innerHTML="✗"},reset:()=>{Object.assign(e.btn.style,{background:"white"}),Object.assign(e.icon.style,{color:"#666"}),e.icon.innerHTML="→",e.fingerAnimation.style.display="block",this.updateHintText("向右滑动完成验证","var(--sc-text-light)")},loading:()=>{e.hint.style.opacity="0",e.fingerAnimation.style.display="none",Object.assign(e.track.style,{pointerEvents:"none",opacity:"0.6"})}};i[t]&&requestAnimationFrame(()=>{i[t](),"loading"!==t&&Object.assign(e.track.style,{pointerEvents:"auto",opacity:"1"})})}updateHintText(t,e){requestAnimationFrame(()=>{Object.assign(this.elements.hint,{textContent:t}),Object.assign(this.elements.hint.style,{color:e,opacity:"1"})})}updateSliderPosition(){const{elements:t,options:e,state:i}=this,{maxX:s}=this.getDimensions(),a=i.currentX/s*(e.width-e.sliderSize),n=i.currentX/s;requestAnimationFrame(()=>{t.btn.style.transform=`translateX(${i.currentX}px)`,t.sliderImg.style.transform=`translateX(${a}px)`,t.fingerAnimation.style.opacity=n>=.8?"0":"0.6"})}show(){this.state.isVisible=!0,this.elements.overlay.style.display="flex",this.elements.overlay.offsetHeight,requestAnimationFrame(()=>{this.elements.overlay.classList.add("show"),this.elements.modal.classList.add("show")}),this.loadCaptcha()}hide(){this.state.isVisible=!1,this.elements.overlay.classList.remove("show"),this.elements.modal.classList.remove("show"),this.safeSetTimeout(()=>{this.elements.overlay.style.display="none",this.reset(),this.options.onClose?.()},300)}async loadCaptcha(){try{this.showLoading(),this.startTime=Date.now();const t=new AbortController,e=this.safeSetTimeout(()=>t.abort(),this.options.timeout),i=await fetch(this.options.apiUrl,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({place:2,timestamp:Date.now()}),signal:t.signal});if(this.safeClearTimeout(e),!i.ok)throw new Error(`HTTP ${i.status}`);const s=await i.json();if(!s.data?.canvasSrc||!s.data?.blockSrc)throw new Error("验证码数据格式错误");this.captchaData=s.data,this.showCaptcha(),await this.renderCaptcha()}catch(t){const e="AbortError"===t.name?"请求超时,请重试":`加载验证码失败: ${t.message}`;this.showError(e)}}async renderCaptcha(){return new Promise((t,e)=>{let i=0;const s=()=>2===++i&&(this.hideLoading(),t()),a=()=>e(new Error("图片加载失败"));this.loadImage(this.elements.backgroundImg,this.captchaData.canvasSrc,{width:this.captchaData.canvasWidth,height:this.captchaData.canvasHeight},s,a),this.loadImage(this.elements.sliderImg,this.captchaData.blockSrc,{width:this.captchaData.blockWidth,height:this.captchaData.blockHeight,top:this.captchaData.blockY},s,a)})}loadImage(t,e,i,s,a){if(this.imageCache.has(e)){const a=this.imageCache.get(e);return t.src=a.src,this.applyStyles(t,i),void s()}t.onload=()=>{this.imageCache.set(e,t.cloneNode()),s()},t.onerror=a,t.src=e,this.applyStyles(t,i)}applyStyles(t,e){Object.entries(e).forEach(([e,i])=>{t.style[e]="number"==typeof i?i+"px":i})}showLoading(){this.state.isLoading=!0,this.batchUpdateStyles({container:{display:"block"},loadingText:{display:"flex"},error:{display:"none"}}),this.updateUIState("loading")}hideLoading(){this.state.isLoading=!1,this.batchUpdateStyles({loadingText:{display:"none"}}),this.updateUIState("reset")}showCaptcha(){this.batchUpdateStyles({container:{display:"block"},track:{display:"block"},error:{display:"none"}})}showError(t){this.hideLoading(),this.batchUpdateStyles({error:{display:"block",textContent:t}})}batchUpdateStyles(t){requestAnimationFrame(()=>{Object.entries(t).forEach(([t,e])=>{const i=this.elements[t];i&&Object.entries(e).forEach(([t,e])=>{"textContent"===t?i.textContent=e:i.style[t]=e})})})}async verify(){if(this.captchaData)try{const t=new AbortController,e=this.safeSetTimeout(()=>t.abort(),this.options.timeout),i=await fetch(this.options.verifyUrl,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({loginVo:{nonceStr:this.captchaData.nonceStr,value:this.getPosition()},dragEventList:[...this.times]}),signal:t.signal});if(this.safeClearTimeout(e),!i.ok)throw new Error(`HTTP ${i.status}`);const s=await i.json();"0"===s.code||!0===s.success?this.onVerifySuccess():this.onVerifyFail(s.message||"验证失败,请重试!")}catch(t){const e="AbortError"===t.name?"验证超时,请重试":"网络错误";this.onVerifyFail(e)}else this.onVerifyFail("验证码数据丢失")}onVerifySuccess(){const t=Date.now()-this.startTime,e=`验证成功!耗时:${(t/1e3).toFixed(2)}s`;this.updateUIState("success"),this.showFloatingTime(e,"success"),this.safeSetTimeout(()=>{this.options.onSuccess?.({captchaId:this.captchaData.captchaId,timestamp:Date.now(),duration:t}),this.hide()},2e3)}showFloatingTime(t,e="success"){const{elements:i}=this;i.floatingTime.textContent=t,i.floatingTime.className=`slider-captcha-floating-time ${e}`,this.safeSetTimeout(()=>i.floatingTime.classList.add("show"),100),this.safeSetTimeout(()=>{i.floatingTime.className="slider-captcha-floating-time"},2500)}onVerifyFail(t){this.state.retryCount++,this.updateUIState("fail"),this.showFloatingTime(t,"fail"),this.safeSetTimeout(()=>{this.state.retryCount>=this.options.maxRetries?this.refresh():this.reset()},2500)}reset(){this.clearAllTimers(),Object.assign(this.state,{isDragging:!1,currentX:0,startX:0,isLoading:!1}),this.times=[],this.startTime=null,this.cachedDimensions=null,requestAnimationFrame(()=>{this.setTransition(!0),this.elements.btn.style.transform="translateX(0px)",this.elements.sliderImg.style.transform="translateX(0px)",this.updateUIState("reset"),this.elements.error.style.display="none"})}refresh(){this.reset(),this.state.retryCount=0,this.loadCaptcha()}safeSetTimeout(t,e){const i=setTimeout(()=>{this.timers.delete(i),t()},e);return this.timers.add(i),i}safeClearTimeout(t){t&&(clearTimeout(t),this.timers.delete(t))}clearAllTimers(){this.timers.forEach(t=>{clearTimeout(t),clearInterval(t)}),this.timers.clear(),this.rafId&&(cancelAnimationFrame(this.rafId),this.rafId=null)}cleanupImages(){this.elements.backgroundImg&&(this.elements.backgroundImg.src="",this.elements.backgroundImg.onload=null,this.elements.backgroundImg.onerror=null),this.elements.sliderImg&&(this.elements.sliderImg.src="",this.elements.sliderImg.onload=null,this.elements.sliderImg.onerror=null),this.imageCache.clear()}throttle(t,e){let i=0;return function(...s){const a=Date.now();if(a-i>=e)return i=a,t.apply(this,s)}}debounce(t,e){let i;return function(...s){clearTimeout(i),i=setTimeout(()=>t.apply(this,s),e)}}destroy(){document.body.style.userSelect="",document.body.style.cursor="",this.clearAllTimers(),this.removeAllEventListeners(),this.cleanupImages(),this.elements.overlay?.parentNode&&this.elements.overlay.parentNode.removeChild(this.elements.overlay);const t=document.getElementById("slider-captcha-styles");t&&t.remove(),Object.keys(this).forEach(t=>{"constructor"!==t&&(this[t]=null)})}static create(t){return new PopupSliderCaptcha(t)}static show(t){const e=new PopupSliderCaptcha(t);return e.show(),e}}"undefined"!=typeof module&&module.exports?(module.exports=PopupSliderCaptcha,module.exports.default=PopupSliderCaptcha):"function"==typeof define&&define.amd?define([],()=>PopupSliderCaptcha):"undefined"!=typeof window&&(window.PopupSliderCaptcha=PopupSliderCaptcha,window.SliderCaptcha=PopupSliderCaptcha);class PasswordValidator{constructor(t={}){this.options={publicKeyUrl:t.publicKeyUrl||"/microservice/strongPassword/getPublicKey",validateUrl:t.validateUrl||"/microservice/strongPassword/checkPassword",timeout:t.timeout||1e4,headers:t.headers||{},...t},this.publicKeyCache=null,this.publicKeyExpiry=null}async getPublicKey(){if(this.publicKeyCache&&this.publicKeyExpiry&&Date.now()<this.publicKeyExpiry)return this.publicKeyCache;try{const t=await this.makeRequest(this.options.publicKeyUrl,{method:"GET"});if("0"!==t.code)throw new Error("获取公钥失败:"+(t.message||"未知错误"));return this.publicKeyCache=t.data,this.publicKeyExpiry=Date.now()+3e5,this.publicKeyCache}catch(t){throw new Error("获取公钥失败: "+t.message)}}encryptPassword(t,e){try{if("undefined"==typeof JSEncrypt)throw new Error("JSEncrypt库未加载,请先引入jsencrypt库");const i=new JSEncrypt;i.setPublicKey(e);const s=i.encrypt(t);if(!s)throw new Error("密码加密失败");return s}catch(t){throw new Error("密码加密失败: "+t.message)}}async validatePassword(t,e={}){try{const i=await this.getPublicKey(),s=this.encryptPassword(t,i),a=await this.makeRequest(this.options.validateUrl,{method:"POST",body:JSON.stringify({userName:"13212345678",password:s,timestamp:Date.now(),...e})});return{success:a.success,message:a.message,data:a.data,code:a.code}}catch(t){return{success:!1,message:t.message,code:"VALIDATION_ERROR"}}}async makeRequest(t,e={}){const i=new AbortController,s=setTimeout(()=>i.abort(),this.options.timeout);try{const a=await fetch(t,{...e,headers:{"Content-Type":"application/json",...this.options.headers,...e.headers},signal:i.signal});if(clearTimeout(s),!a.ok)throw new Error(`HTTP错误: ${a.status} ${a.statusText}`);return await a.json()}catch(t){if(clearTimeout(s),"AbortError"===t.name)throw new Error("请求超时");throw t}}clearCache(){this.publicKeyCache=null,this.publicKeyExpiry=null}updateOptions(t){this.options={...this.options,...t},this.clearCache()}}function e(t){return new PasswordValidator(t)}async function i(t,e={},i={}){const s=new PasswordValidator(e);return await s.validatePassword(t,i)}var s={PopupSliderCaptcha:PopupSliderCaptcha,PasswordValidator:PasswordValidator,createPasswordValidator:e,validatePassword:i};"undefined"!=typeof window&&(window.SliderCaptcha=PopupSliderCaptcha,window.PopupSliderCaptcha=PopupSliderCaptcha,window.PasswordValidator=PasswordValidator,window.createPasswordValidator=e,window.validatePassword=i),t.PasswordValidator=PasswordValidator,t.PopupSliderCaptcha=PopupSliderCaptcha,t.createPasswordValidator=e,t.default=s,t.validatePassword=i,Object.defineProperty(t,"__esModule",{value:!0})});
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sources":["../src/slider-captcha.js","../src/password-validator.js","../src/index.js"],"sourcesContent":["/**\n * 纯JavaScript弹窗滑块验证码组件\n */\nclass PopupSliderCaptcha {\n static DEFAULTS = {\n width: 320,\n height: 155,\n sliderSize: 38,\n maxRetries: 3,\n timeout: 30000,\n apiUrl: \"/api/captcha\",\n verifyUrl: \"/api/captcha/verify\",\n throttleDelay: 16,\n clickMaskClose: false\n }\n\n static CSS_CLASSES = {\n overlay: \"slider-captcha-overlay\",\n modal: \"slider-captcha-modal\",\n header: \"slider-captcha-header\",\n container: \"slider-captcha-container\",\n track: \"slider-captcha-track\",\n btn: \"slider-captcha-btn\",\n hint: \"slider-captcha-hint\",\n loading: \"slider-captcha-loading\",\n error: \"slider-captcha-error\"\n }\n\n // 优化:提取CSS样式为独立方法,减少主体代码长度\n static getStyles() {\n return `:root{--sc-primary:#409eff;--sc-success:#67c23a;--sc-danger:#f56c6c;--sc-border:#e4e7eb;--sc-bg:linear-gradient(90deg, #f7f9fa 0%, #e8f4fd 100%);--sc-text:#333;--sc-text-light:#999;--sc-shadow:0 4px 20px rgba(0,0,0,.3);--sc-radius:8px;--sc-transition:.3s ease}.slider-captcha-overlay{position:fixed;top:0;left:0;width:100%;height:100%;background:rgba(0,0,0,.5);z-index:9999;display:none;justify-content:center;align-items:center;opacity:0;transition:opacity var(--sc-transition)}.slider-captcha-overlay.show{opacity:1}.slider-captcha-modal{background:#fff;border-radius:var(--sc-radius);padding:20px;box-shadow:var(--sc-shadow);position:relative;max-width:90vw;max-height:90vh;transform:scale(.8) translateY(-20px);opacity:0;transition:all var(--sc-transition)}.slider-captcha-modal.show{transform:scale(1) translateY(0);opacity:1}.slider-captcha-header{display:flex;justify-content:space-between;align-items:center;margin-bottom:15px;padding-bottom:10px;border-bottom:1px solid var(--sc-border)}.slider-captcha-container{display:flex;align-items:center;position:relative;border-radius:4px;overflow:hidden;margin-bottom:15px;background:#837a7a;justify-content:center}.slider-captcha-track{width:100%;height:40px;line-height:40px;background:var(--sc-bg);border:1px solid var(--sc-border);border-radius:20px;position:relative;margin-bottom:15px;overflow:hidden}.slider-captcha-btn{width:38px;height:38px;background:#fff;border:1px solid #ccc;border-radius:50%;position:absolute;top:0;left:0;cursor:pointer;display:flex;align-items:center;justify-content:center;box-shadow:0 2px 4px rgba(0,0,0,.1);transition:all var(--sc-transition);user-select:none;z-index:1}.slider-captcha-loading{position:absolute;top:0;left:0;width:100%;height:100%;background:rgba(255,255,255,.6);display:flex;align-items:center;justify-content:center;flex-direction:column;color:#666;font-size:14px;z-index:10;border-radius:4px}.slider-captcha-error{color:var(--sc-danger);font-size:12px;text-align:center;margin-top:10px;display:none}.slider-captcha-title{margin:0;font-size:16px;color:var(--sc-text)}.slider-captcha-close,.slider-captcha-refresh{background:none;border:none;cursor:pointer;color:var(--sc-text-light);padding:0;width:30px;height:30px;display:flex;align-items:center;justify-content:center;border-radius:50%;transition:all var(--sc-transition);position:relative;font-size:0}.slider-captcha-close::before,.slider-captcha-close::after{content:'';position:absolute;width:16px;height:2px;background-color:var(--sc-text-light);border-radius:1px;transition:all var(--sc-transition)}.slider-captcha-close::before{transform:rotate(45deg)}.slider-captcha-close::after{transform:rotate(-45deg)}.slider-captcha-close:hover{background:#f5f5f5;transform:scale(1.1)}.slider-captcha-close:hover::before,.slider-captcha-close:hover::after{background-color:var(--sc-danger)}.slider-captcha-refresh{margin-left:10px}.slider-captcha-refresh svg{width:20px;height:20px;fill:var(--sc-text-light);transition:all var(--sc-transition)}.slider-captcha-refresh:hover{background:#f5f5f5;transform:scale(1.1)}.slider-captcha-refresh:hover svg{fill:var(--sc-primary);transform:rotate(180deg)}.slider-captcha-floating-time{position:absolute;bottom:-40px;left:50%;transform:translateX(-50%);color:#fff;font-size:12px;white-space:nowrap;opacity:0;pointer-events:none;z-index:10;transition:all var(--sc-transition);background:#fff;padding:2px 15px;border-radius:10px}.slider-captcha-floating-time.show{opacity:1;transform:translateX(-50%) translateY(-45px)}.slider-captcha-floating-time.success{color:var(--sc-success)}.slider-captcha-floating-time.fail{color:var(--sc-danger)}.slider-captcha-bg{width:100%;height:100%;object-fit:cover;display:block}.slider-captcha-piece{position:absolute;top:0;left:0;cursor:pointer;transition:none;z-index:2}.slider-captcha-finger{position:absolute;top:50%;left:10px;transform:translateY(-50%);font-size:20px;animation:fingerSlide 2s ease-in-out infinite;pointer-events:none;z-index:1;opacity:.6}.slider-captcha-hint{position:absolute;top:50%;left:50%;transform:translate(-50%,-50%);color:var(--sc-text-light);font-size:14px;pointer-events:none;z-index:1;transition:all var(--sc-transition)}.slider-captcha-header-buttons{display:flex;align-items:center}@keyframes fingerSlide{0%{left:10px;opacity:.6}50%{opacity:1}100%{left:calc(50% - 10px);opacity:.6}}`\n }\n\n constructor(options = {}) {\n this.options = { ...PopupSliderCaptcha.DEFAULTS, ...options }\n this.elements = {}\n this.state = this.createInitialState()\n this.captchaData = null\n this.times = []\n this.startTime = null\n this.eventListeners = []\n this.timers = new Set()\n this.rafId = null\n this.cachedDimensions = null\n this.imageCache = new Map()\n\n // 优化:使用箭头函数绑定this,避免重复bind调用\n this.throttledHandleMove = this.throttle((e) => this.handleMove(e), this.options.throttleDelay)\n\n this.init()\n }\n\n // 优化:提取初始状态创建为独立方法\n createInitialState() {\n return {\n isVisible: false,\n isDragging: false,\n currentX: 0,\n startX: 0,\n retryCount: 0,\n isLoading: false\n }\n }\n\n init() {\n this.injectStyles()\n this.createElements()\n this.bindEvents()\n }\n\n injectStyles() {\n if (document.querySelector(\"#slider-captcha-styles\")) return\n\n const style = document.createElement(\"style\")\n style.id = \"slider-captcha-styles\"\n style.textContent = PopupSliderCaptcha.getStyles()\n document.head.appendChild(style)\n }\n\n // 优化:简化元素创建逻辑\n createElements() {\n const { elements, options } = this\n\n // 批量创建元素配置\n const elementConfigs = [\n ['overlay', 'div', PopupSliderCaptcha.CSS_CLASSES.overlay],\n ['modal', 'div', PopupSliderCaptcha.CSS_CLASSES.modal],\n ['header', 'div', PopupSliderCaptcha.CSS_CLASSES.header],\n ['title', 'h3', 'slider-captcha-title', '安全验证'],\n ['closeBtn', 'button', 'slider-captcha-close'],\n ['refreshBtn', 'button', 'slider-captcha-refresh'],\n ['container', 'div', PopupSliderCaptcha.CSS_CLASSES.container],\n ['backgroundImg', 'img', 'slider-captcha-bg'],\n ['sliderImg', 'img', 'slider-captcha-piece'],\n ['loadingText', 'div', PopupSliderCaptcha.CSS_CLASSES.loading, '加载中...'],\n ['floatingTime', 'div', 'slider-captcha-floating-time'],\n ['track', 'div', PopupSliderCaptcha.CSS_CLASSES.track],\n ['fingerAnimation', 'div', 'slider-captcha-finger', '👉'],\n ['btn', 'div', PopupSliderCaptcha.CSS_CLASSES.btn],\n ['icon', 'div', '', '→'],\n ['hint', 'div', PopupSliderCaptcha.CSS_CLASSES.hint, '向右滑动完成验证'],\n ['error', 'div', PopupSliderCaptcha.CSS_CLASSES.error]\n ]\n\n // 批量创建元素\n elementConfigs.forEach(([key, tag, className, textContent]) => {\n elements[key] = this.createElement(tag, className, textContent)\n })\n\n // 设置容器尺寸\n elements.container.style.cssText = `width:${options.width}px;height:${options.height}px`\n\n // 添加刷新按钮图标\n elements.refreshBtn.innerHTML = `<svg viewBox=\"0 0 24 24\"><path d=\"M17.65,6.35C16.2,4.9 14.21,4 12,4A8,8 0 0,0 4,12A8,8 0 0,0 12,20C15.73,20 18.84,17.45 19.73,14H17.65C16.83,16.33 14.61,18 12,18A6,6 0 0,1 6,12A6,6 0 0,1 12,6C13.66,6 15.14,6.69 16.22,7.78L13,11H20V4L17.65,6.35Z\"/></svg>`\n\n this.assembleDOM()\n this.setInitialState()\n }\n\n createElement(tag, className = \"\", textContent = \"\") {\n const element = document.createElement(tag)\n if (className) element.className = className\n if (textContent) element.textContent = textContent\n return element\n }\n\n // 优化:简化DOM组装逻辑\n assembleDOM() {\n const { elements } = this\n\n // 组装头部\n const headerButtons = this.createElement(\"div\", \"slider-captcha-header-buttons\")\n headerButtons.append(elements.refreshBtn, elements.closeBtn)\n elements.header.append(elements.title, headerButtons)\n\n // 组装验证码容器\n elements.container.append(\n elements.backgroundImg,\n elements.sliderImg,\n elements.loadingText,\n elements.floatingTime\n )\n\n // 组装滑块轨道\n elements.btn.appendChild(elements.icon)\n elements.track.append(\n elements.fingerAnimation,\n elements.btn,\n elements.hint\n )\n\n // 组装模态框\n elements.modal.append(\n elements.header,\n elements.container,\n elements.track,\n elements.error\n )\n\n // 组装到覆盖层\n elements.overlay.appendChild(elements.modal)\n document.body.appendChild(elements.overlay)\n }\n\n setInitialState() {\n // 批量设置初始状态\n Object.assign(this.elements.container.style, { display: \"none\" })\n Object.assign(this.elements.track.style, { display: \"none\" })\n }\n\n // 优化:简化事件绑定\n bindEvents() {\n const { elements } = this\n\n // 基础事件\n this.addEventListener(elements.closeBtn, \"click\", () => this.hide())\n this.addEventListener(elements.refreshBtn, \"click\", () => this.refresh())\n this.addEventListener(elements.overlay, \"click\", (e) => {\n if (e.target === elements.overlay && this.options.clickMaskClose) this.hide()\n })\n this.addEventListener(document, \"keydown\", (e) => {\n if (e.key === \"Escape\" && this.state.isVisible) this.hide()\n })\n this.addEventListener(document, \"visibilitychange\", () => this.handleVisibilityChange())\n\n this.bindSliderEvents()\n }\n\n bindSliderEvents() {\n const { elements } = this\n const handlers = {\n start: (e) => this.handleStart(e),\n move: this.throttledHandleMove,\n end: () => this.handleEnd()\n }\n\n // 滑块事件\n this.addEventListener(elements.btn, \"mousedown\", handlers.start)\n this.addEventListener(elements.btn, \"touchstart\", handlers.start)\n this.addEventListener(elements.sliderImg, \"mousedown\", handlers.start)\n this.addEventListener(elements.sliderImg, \"touchstart\", handlers.start)\n this.addEventListener(document, \"mousemove\", handlers.move, { passive: false })\n this.addEventListener(document, \"touchmove\", handlers.move, { passive: false })\n this.addEventListener(document, \"mouseup\", handlers.end)\n this.addEventListener(document, \"touchend\", handlers.end)\n }\n\n // 优化:改进事件管理\n addEventListener(element, event, handler, options = {}) {\n if (!element || typeof handler !== 'function') return\n\n element.addEventListener(event, handler, options)\n this.eventListeners.push({ element, event, handler, options })\n }\n\n removeAllEventListeners() {\n this.eventListeners.forEach(({ element, event, handler, options }) => {\n try {\n element?.removeEventListener?.(event, handler, options)\n } catch (error) {\n console.warn('Failed to remove event listener:', error)\n }\n })\n this.eventListeners.length = 0\n }\n\n // 优化:缓存尺寸计算\n getDimensions() {\n if (!this.cachedDimensions) {\n const trackWidth = this.elements.track.offsetWidth\n const btnWidth = this.elements.btn.offsetWidth\n this.cachedDimensions = {\n trackWidth,\n btnWidth,\n maxX: trackWidth - btnWidth\n }\n }\n return this.cachedDimensions\n }\n\n getPosition() {\n const { maxX } = this.getDimensions()\n const percentage = this.state.currentX / maxX\n return Math.round(percentage * (this.options.width - this.options.sliderSize))\n }\n\n // 优化:简化拖拽处理\n handleStart(e) {\n if (!this.captchaData || this.state.isDragging || this.state.isLoading) return\n\n e.preventDefault()\n this.state.isDragging = true\n this.state.startX = this.getClientX(e) - this.state.currentX\n this.startTime = this.startTime || Date.now()\n this.times = [{ time: Date.now(), position: this.getPosition() }]\n\n this.setTransition(false)\n this.updateUIState(\"dragging\")\n this.cachedDimensions = null // 清除缓存\n }\n\n handleMove(e) {\n if (!this.state.isDragging) return\n e.preventDefault()\n\n const clientX = this.getClientX(e)\n const deltaX = clientX - this.state.startX\n const { maxX } = this.getDimensions()\n\n this.state.currentX = Math.max(0, Math.min(deltaX, maxX))\n this.times.push({ time: Date.now(), position: this.getPosition() })\n\n // 优化:使用RAF批量更新\n this.rafId && cancelAnimationFrame(this.rafId)\n this.rafId = requestAnimationFrame(() => this.updateSliderPosition())\n }\n\n handleEnd() {\n if (!this.state.isDragging) return\n\n this.times.push({ time: Date.now(), position: this.getPosition() })\n this.state.isDragging = false\n\n if (this.rafId) {\n cancelAnimationFrame(this.rafId)\n this.rafId = null\n }\n\n this.verify()\n }\n\n handleVisibilityChange() {\n const animationState = document.hidden ? 'paused' : 'running'\n if (this.elements.fingerAnimation) {\n this.elements.fingerAnimation.style.animationPlayState = animationState\n }\n }\n\n getClientX(e) {\n return e.type.includes(\"touch\") ? e.touches[0].clientX : e.clientX\n }\n\n setTransition(enabled) {\n const transition = enabled ? \"all 0.3s ease\" : \"none\"\n requestAnimationFrame(() => {\n this.elements.btn.style.transition = transition\n this.elements.sliderImg.style.transition = transition\n })\n }\n\n // 优化:简化UI状态更新\n updateUIState(state) {\n const { elements } = this\n const updates = {\n dragging: () => {\n elements.hint.style.opacity = \"0\"\n elements.fingerAnimation.style.display = \"none\"\n },\n success: () => {\n Object.assign(elements.btn.style, { background: \"var(--sc-success)\" })\n Object.assign(elements.icon.style, { innerHTML: \"✓\", color: \"white\" })\n elements.icon.innerHTML = \"✓\"\n },\n fail: () => {\n Object.assign(elements.btn.style, { background: \"var(--sc-danger)\" })\n Object.assign(elements.icon.style, { innerHTML: \"✗\", color: \"white\" })\n elements.icon.innerHTML = \"✗\"\n },\n reset: () => {\n Object.assign(elements.btn.style, { background: \"white\" })\n Object.assign(elements.icon.style, { color: \"#666\" })\n elements.icon.innerHTML = \"→\"\n elements.fingerAnimation.style.display = \"block\"\n this.updateHintText(\"向右滑动完成验证\", \"var(--sc-text-light)\")\n },\n loading: () => {\n elements.hint.style.opacity = \"0\"\n elements.fingerAnimation.style.display = \"none\"\n Object.assign(elements.track.style, { pointerEvents: \"none\", opacity: \"0.6\" })\n }\n }\n\n if (updates[state]) {\n requestAnimationFrame(() => {\n updates[state]()\n if (state !== \"loading\") {\n Object.assign(elements.track.style, { pointerEvents: \"auto\", opacity: \"1\" })\n }\n })\n }\n }\n\n updateHintText(text, color) {\n requestAnimationFrame(() => {\n Object.assign(this.elements.hint, { textContent: text })\n Object.assign(this.elements.hint.style, { color, opacity: \"1\" })\n })\n }\n\n updateSliderPosition() {\n const { elements, options, state } = this\n const { maxX } = this.getDimensions()\n const pieceX = (state.currentX / maxX) * (options.width - options.sliderSize)\n const progress = state.currentX / maxX\n\n requestAnimationFrame(() => {\n elements.btn.style.transform = `translateX(${state.currentX}px)`\n elements.sliderImg.style.transform = `translateX(${pieceX}px)`\n elements.fingerAnimation.style.opacity = progress >= 0.8 ? \"0\" : \"0.6\"\n })\n }\n\n // 优化:简化显示/隐藏逻辑\n show() {\n this.state.isVisible = true\n this.elements.overlay.style.display = \"flex\"\n this.elements.overlay.offsetHeight // 强制重绘\n\n requestAnimationFrame(() => {\n this.elements.overlay.classList.add(\"show\")\n this.elements.modal.classList.add(\"show\")\n })\n\n this.loadCaptcha()\n }\n\n hide() {\n this.state.isVisible = false\n this.elements.overlay.classList.remove(\"show\")\n this.elements.modal.classList.remove(\"show\")\n\n this.safeSetTimeout(() => {\n this.elements.overlay.style.display = \"none\"\n this.reset()\n this.options.onClose?.()\n }, 300)\n }\n\n // 优化:简化加载逻辑\n async loadCaptcha() {\n try {\n this.showLoading()\n this.startTime = Date.now()\n\n const controller = new AbortController()\n const timeoutId = this.safeSetTimeout(() => controller.abort(), this.options.timeout)\n\n const response = await fetch(this.options.apiUrl, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({ place: 2, timestamp: Date.now() }),\n signal: controller.signal\n })\n\n this.safeClearTimeout(timeoutId)\n\n if (!response.ok) throw new Error(`HTTP ${response.status}`)\n const data = await response.json()\n\n if (data.data?.canvasSrc && data.data?.blockSrc) {\n this.captchaData = data.data\n this.showCaptcha()\n await this.renderCaptcha()\n } else {\n throw new Error(\"验证码数据格式错误\")\n }\n } catch (error) {\n const message = error.name === 'AbortError' ? \"请求超时,请重试\" : `加载验证码失败: ${error.message}`\n this.showError(message)\n }\n }\n\n async renderCaptcha() {\n return new Promise((resolve, reject) => {\n let loadedCount = 0\n const onLoad = () => ++loadedCount === 2 && (this.hideLoading(), resolve())\n const onError = () => reject(new Error(\"图片加载失败\"))\n\n this.loadImage(this.elements.backgroundImg, this.captchaData.canvasSrc, {\n width: this.captchaData.canvasWidth,\n height: this.captchaData.canvasHeight\n }, onLoad, onError)\n\n this.loadImage(this.elements.sliderImg, this.captchaData.blockSrc, {\n width: this.captchaData.blockWidth,\n height: this.captchaData.blockHeight,\n top: this.captchaData.blockY\n }, onLoad, onError)\n })\n }\n\n loadImage(imgElement, src, styles, onLoad, onError) {\n // 检查缓存\n if (this.imageCache.has(src)) {\n const cachedImg = this.imageCache.get(src)\n imgElement.src = cachedImg.src\n this.applyStyles(imgElement, styles)\n onLoad()\n return\n }\n\n imgElement.onload = () => {\n this.imageCache.set(src, imgElement.cloneNode())\n onLoad()\n }\n imgElement.onerror = onError\n imgElement.src = src\n this.applyStyles(imgElement, styles)\n }\n\n // 优化:提取样式应用逻辑\n applyStyles(element, styles) {\n Object.entries(styles).forEach(([key, value]) => {\n element.style[key] = typeof value === \"number\" ? value + \"px\" : value\n })\n }\n\n // 优化:简化状态显示方法\n showLoading() {\n this.state.isLoading = true\n this.batchUpdateStyles({\n container: { display: \"block\" },\n loadingText: { display: \"flex\" },\n error: { display: \"none\" }\n })\n this.updateUIState(\"loading\")\n }\n\n hideLoading() {\n this.state.isLoading = false\n this.batchUpdateStyles({ loadingText: { display: \"none\" } })\n this.updateUIState(\"reset\")\n }\n\n showCaptcha() {\n this.batchUpdateStyles({\n container: { display: \"block\" },\n track: { display: \"block\" },\n error: { display: \"none\" }\n })\n }\n\n showError(message) {\n this.hideLoading()\n this.batchUpdateStyles({\n error: { display: \"block\", textContent: message }\n })\n }\n\n // 优化:批量样式更新\n batchUpdateStyles(updates) {\n requestAnimationFrame(() => {\n Object.entries(updates).forEach(([elementKey, styles]) => {\n const element = this.elements[elementKey]\n if (element) {\n Object.entries(styles).forEach(([prop, value]) => {\n if (prop === 'textContent') {\n element.textContent = value\n } else {\n element.style[prop] = value\n }\n })\n }\n })\n })\n }\n\n async verify() {\n if (!this.captchaData) {\n this.onVerifyFail(\"验证码数据丢失\")\n return\n }\n\n try {\n const controller = new AbortController()\n const timeoutId = this.safeSetTimeout(() => controller.abort(), this.options.timeout)\n\n const response = await fetch(this.options.verifyUrl, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({\n loginVo: {\n nonceStr: this.captchaData.nonceStr,\n value: this.getPosition()\n },\n dragEventList: [...this.times]\n }),\n signal: controller.signal\n })\n\n this.safeClearTimeout(timeoutId)\n\n if (!response.ok) throw new Error(`HTTP ${response.status}`)\n const data = await response.json()\n\n if (data.code === \"0\" || data.success === true) {\n this.onVerifySuccess()\n } else {\n this.onVerifyFail(data.message || \"验证失败,请重试!\")\n }\n } catch (error) {\n const message = error.name === 'AbortError' ? \"验证超时,请重试\" : \"网络错误\"\n this.onVerifyFail(message)\n }\n }\n\n onVerifySuccess() {\n const duration = Date.now() - this.startTime\n const durationText = `验证成功!耗时:${(duration / 1000).toFixed(2)}s`\n\n this.updateUIState(\"success\")\n this.showFloatingTime(durationText, \"success\")\n\n this.safeSetTimeout(() => {\n this.options.onSuccess?.({\n captchaId: this.captchaData.captchaId,\n timestamp: Date.now(),\n duration\n })\n this.hide()\n }, 2000)\n }\n\n showFloatingTime(text, type = \"success\") {\n const { elements } = this\n elements.floatingTime.textContent = text\n elements.floatingTime.className = `slider-captcha-floating-time ${type}`\n\n this.safeSetTimeout(() => elements.floatingTime.classList.add(\"show\"), 100)\n this.safeSetTimeout(() => {\n elements.floatingTime.className = \"slider-captcha-floating-time\"\n }, 2500) // 优化:延长显示时间,避免被reset清除\n }\n\n onVerifyFail(message) {\n this.state.retryCount++\n this.updateUIState(\"fail\")\n this.showFloatingTime(message, \"fail\")\n\n this.safeSetTimeout(() => {\n if (this.state.retryCount >= this.options.maxRetries) {\n this.refresh()\n } else {\n this.reset()\n }\n }, 2500) // 优化:延长等待时间,确保浮动提示完整显示\n }\n\n reset() {\n this.clearAllTimers()\n\n // 重置状态\n Object.assign(this.state, {\n isDragging: false,\n currentX: 0,\n startX: 0,\n isLoading: false\n })\n\n this.times = []\n this.startTime = null\n this.cachedDimensions = null\n\n // 重置UI\n requestAnimationFrame(() => {\n this.setTransition(true)\n this.elements.btn.style.transform = \"translateX(0px)\"\n this.elements.sliderImg.style.transform = \"translateX(0px)\"\n this.updateUIState(\"reset\")\n this.elements.error.style.display = \"none\"\n })\n }\n\n refresh() {\n this.reset()\n this.state.retryCount = 0\n this.loadCaptcha()\n }\n\n // 安全的定时器管理\n safeSetTimeout(callback, delay) {\n const timerId = setTimeout(() => {\n this.timers.delete(timerId)\n callback()\n }, delay)\n this.timers.add(timerId)\n return timerId\n }\n\n safeClearTimeout(timerId) {\n if (timerId) {\n clearTimeout(timerId)\n this.timers.delete(timerId)\n }\n }\n\n clearAllTimers() {\n this.timers.forEach(timer => {\n clearTimeout(timer)\n clearInterval(timer)\n })\n this.timers.clear()\n\n if (this.rafId) {\n cancelAnimationFrame(this.rafId)\n this.rafId = null\n }\n }\n\n // 清理图片资源\n cleanupImages() {\n if (this.elements.backgroundImg) {\n this.elements.backgroundImg.src = ''\n this.elements.backgroundImg.onload = null\n this.elements.backgroundImg.onerror = null\n }\n if (this.elements.sliderImg) {\n this.elements.sliderImg.src = ''\n this.elements.sliderImg.onload = null\n this.elements.sliderImg.onerror = null\n }\n this.imageCache.clear()\n }\n\n // 工具函数:节流\n throttle(func, delay) {\n let lastCall = 0\n return function (...args) {\n const now = Date.now()\n if (now - lastCall >= delay) {\n lastCall = now\n return func.apply(this, args)\n }\n }\n }\n\n // 工具函数:防抖\n debounce(func, delay) {\n let timeoutId\n return function (...args) {\n clearTimeout(timeoutId)\n timeoutId = setTimeout(() => func.apply(this, args), delay)\n }\n }\n\n destroy() {\n // 恢复body样式\n document.body.style.userSelect = \"\"\n document.body.style.cursor = \"\"\n\n // 清理所有定时器\n this.clearAllTimers()\n\n // 移除所有事件监听器\n this.removeAllEventListeners()\n\n // 清理图片资源\n this.cleanupImages()\n\n // 移除DOM元素\n if (this.elements.overlay?.parentNode) {\n this.elements.overlay.parentNode.removeChild(this.elements.overlay)\n }\n\n // 清理样式表\n const styleElement = document.getElementById('slider-captcha-styles')\n if (styleElement) {\n styleElement.remove()\n }\n\n // 清空所有属性\n Object.keys(this).forEach(key => {\n if (key !== 'constructor') {\n this[key] = null\n }\n })\n }\n\n static create(options) {\n return new PopupSliderCaptcha(options)\n }\n\n static show(options) {\n const instance = new PopupSliderCaptcha(options)\n instance.show()\n return instance\n }\n}\n\n// 模块导出\nif (typeof module !== \"undefined\" && module.exports) {\n module.exports = PopupSliderCaptcha\n module.exports.default = PopupSliderCaptcha\n} else if (typeof define === \"function\" && define.amd) {\n define([], () => PopupSliderCaptcha)\n} else if (typeof window !== \"undefined\") {\n window.PopupSliderCaptcha = PopupSliderCaptcha\n window.SliderCaptcha = PopupSliderCaptcha\n}\n\n// Add ES6 export for modern module systems\nexport default PopupSliderCaptcha\n","// import JSEncrypt from 'jsencrypt'\n\n/**\n * 密码校验工具类\n * 提供密码加密和校验功能\n */\nclass PasswordValidator {\n constructor(options = {}) {\n this.options = {\n // 获取公钥的接口地址\n publicKeyUrl: options.publicKeyUrl || '/microservice/strongPassword/getPublicKey',\n // 密码校验接口地址\n validateUrl: options.validateUrl || '/microservice/strongPassword/checkPassword',\n // 请求超时时间\n timeout: options.timeout || 10000,\n // 自定义请求头\n headers: options.headers || {},\n ...options\n }\n\n // 缓存公钥,避免重复请求\n this.publicKeyCache = null\n this.publicKeyExpiry = null\n }\n\n /**\n * 获取公钥\n * @returns {Promise<string>} 公钥字符串\n */\n async getPublicKey() {\n // 检查缓存是否有效(5分钟有效期)\n if (this.publicKeyCache && this.publicKeyExpiry && Date.now() < this.publicKeyExpiry) {\n return this.publicKeyCache\n }\n\n try {\n const response = await this.makeRequest(this.options.publicKeyUrl, {\n method: 'GET'\n })\n\n if (response.code !== '0' ) {\n throw new Error('获取公钥失败:' + (response.message || '未知错误'))\n }\n\n // 缓存公钥,设置5分钟过期时间\n this.publicKeyCache = response.data\n this.publicKeyExpiry = Date.now() + 5 * 60 * 1000\n\n return this.publicKeyCache\n } catch (error) {\n console.error('获取公钥失败:', error)\n throw new Error('获取公钥失败: ' + error.message)\n }\n }\n\n /**\n * 使用RSA公钥加密密码\n * @param {string} password 原始密码\n * @param {string} publicKey 公钥字符串\n * @returns {string} 加密后的密码\n */\n encryptPassword(password, publicKey) {\n try {\n // 这里使用JSEncrypt库进行RSA加密\n // 需要先安装: npm install jsencrypt\n if (typeof JSEncrypt === 'undefined') {\n throw new Error('JSEncrypt库未加载,请先引入jsencrypt库')\n }\n\n const encrypt = new JSEncrypt()\n encrypt.setPublicKey(publicKey)\n const encrypted = encrypt.encrypt(password)\n\n if (!encrypted) {\n throw new Error('密码加密失败')\n }\n\n return encrypted\n } catch (error) {\n console.error('密码加密失败:', error)\n throw new Error('密码加密失败: ' + error.message)\n }\n }\n\n /**\n * 校验密码\n * @param {string} password 原始密码\n * @param {Object} additionalData 额外的校验数据\n * @returns {Promise<Object>} 校验结果\n */\n async validatePassword(password, additionalData = {}) {\n try {\n // 1. 获取公钥\n const publicKey = await this.getPublicKey()\n\n // 2. 加密密码\n const encryptedPassword = this.encryptPassword(password, publicKey)\n\n // 3. 调用校验接口\n const response = await this.makeRequest(this.options.validateUrl, {\n method: 'POST',\n body: JSON.stringify({\n userName: '13212345678',\n password: encryptedPassword,\n timestamp: Date.now(),\n ...additionalData\n })\n })\n\n return {\n success: response.success,\n message: response.message,\n data: response.data,\n code: response.code\n }\n } catch (error) {\n console.error('密码校验失败:', error)\n return {\n success: false,\n message: error.message,\n code: 'VALIDATION_ERROR'\n }\n }\n }\n\n /**\n * 发送HTTP请求的通用方法\n * @param {string} url 请求地址\n * @param {Object} options 请求选项\n * @returns {Promise<Object>} 响应结果\n */\n async makeRequest(url, options = {}) {\n const controller = new AbortController()\n const timeoutId = setTimeout(() => controller.abort(), this.options.timeout)\n\n try {\n const response = await fetch(url, {\n ...options,\n headers: {\n 'Content-Type': 'application/json',\n ...this.options.headers,\n ...options.headers\n },\n signal: controller.signal\n })\n\n clearTimeout(timeoutId)\n\n if (!response.ok) {\n throw new Error(`HTTP错误: ${response.status} ${response.statusText}`)\n }\n\n return await response.json()\n } catch (error) {\n clearTimeout(timeoutId)\n\n if (error.name === 'AbortError') {\n throw new Error('请求超时')\n }\n\n throw error\n }\n }\n\n /**\n * 清除公钥缓存\n */\n clearCache() {\n this.publicKeyCache = null\n this.publicKeyExpiry = null\n }\n\n /**\n * 更新配置\n * @param {Object} newOptions 新的配置选项\n */\n updateOptions(newOptions) {\n this.options = { ...this.options, ...newOptions }\n // 清除缓存,使用新配置重新获取\n this.clearCache()\n }\n}\n\n// 导出类和创建实例的工厂函数\nexport default PasswordValidator\n\n/**\n * 创建密码校验器实例的工厂函数\n * @param {Object} options 配置选项\n * @returns {PasswordValidator} 密码校验器实例\n */\nexport function createPasswordValidator(options) {\n return new PasswordValidator(options)\n}\n\n/**\n * 快速校验密码的便捷函数\n * @param {string} password 密码\n * @param {Object} options 配置选项\n * @param {Object} additionalData 额外数据\n * @returns {Promise<Object>} 校验结果\n */\nexport async function validatePassword(password, options = {}, additionalData = {}) {\n const validator = new PasswordValidator(options)\n return await validator.validatePassword(password, additionalData)\n}\n","// 导入滑块验证码组件(使用默认导入)\nimport PopupSliderCaptcha from './slider-captcha.js'\n\n// 导入密码校验器\nimport PasswordValidator, { createPasswordValidator, validatePassword } from './password-validator.js'\n\n// 导出所有模块\nexport { \n PopupSliderCaptcha,\n PasswordValidator,\n createPasswordValidator,\n validatePassword\n}\n\n// 默认导出(向后兼容)\nexport default {\n PopupSliderCaptcha,\n PasswordValidator,\n createPasswordValidator,\n validatePassword\n}\n\n// 全局注册(用于UMD构建)\nif (typeof window !== 'undefined') {\n window.SliderCaptcha = PopupSliderCaptcha\n window.PopupSliderCaptcha = PopupSliderCaptcha\n window.PasswordValidator = PasswordValidator\n window.createPasswordValidator = createPasswordValidator\n window.validatePassword = validatePassword\n}\n"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
(function(t,i){"object"==typeof exports&&"undefined"!=typeof module?i(exports):"function"==typeof define&&define.amd?define(["exports"],i):i((t="undefined"!=typeof globalThis?globalThis:t||self).SliderCaptchaSDK={})})(this,(function(t){"use strict"
|
|
2
|
+
function i(t){return new PasswordValidator(t)}async function e(t,i={},e={}){const s=new PasswordValidator(i)
|
|
3
|
+
return await s.validatePassword(t,e)}class PopupSliderCaptcha{static DEFAULTS={width:320,height:155,sliderSize:38,maxRetries:3,timeout:3e4,apiUrl:"/api/captcha",verifyUrl:"/api/captcha/verify",throttleDelay:16,clickMaskClose:!1}
|
|
4
|
+
static CSS_CLASSES={overlay:"slider-captcha-overlay",modal:"slider-captcha-modal",header:"slider-captcha-header",container:"slider-captcha-container",track:"slider-captcha-track",btn:"slider-captcha-btn",hint:"slider-captcha-hint",loading:"slider-captcha-loading",error:"slider-captcha-error"}
|
|
5
|
+
static getStyles(){return":root{--sc-primary:#409eff;--sc-success:#67c23a;--sc-danger:#f56c6c;--sc-border:#e4e7eb;--sc-bg:linear-gradient(90deg, #f7f9fa 0%, #e8f4fd 100%);--sc-text:#333;--sc-text-light:#999;--sc-shadow:0 4px 20px rgba(0,0,0,.3);--sc-radius:8px;--sc-transition:.3s ease}.slider-captcha-overlay{position:fixed;top:0;left:0;width:100%;height:100%;background:rgba(0,0,0,.5);z-index:9999;display:none;justify-content:center;align-items:center;opacity:0;transition:opacity var(--sc-transition)}.slider-captcha-overlay.show{opacity:1}.slider-captcha-modal{background:#fff;border-radius:var(--sc-radius);padding:20px;box-shadow:var(--sc-shadow);position:relative;max-width:90vw;max-height:90vh;transform:scale(.8) translateY(-20px);opacity:0;transition:all var(--sc-transition)}.slider-captcha-modal.show{transform:scale(1) translateY(0);opacity:1}.slider-captcha-header{display:flex;justify-content:space-between;align-items:center;margin-bottom:15px;padding-bottom:10px;border-bottom:1px solid var(--sc-border)}.slider-captcha-container{display:flex;align-items:center;position:relative;border-radius:4px;overflow:hidden;margin-bottom:15px;background:#837a7a;justify-content:center}.slider-captcha-track{width:100%;height:40px;line-height:40px;background:var(--sc-bg);border:1px solid var(--sc-border);border-radius:20px;position:relative;margin-bottom:15px;overflow:hidden}.slider-captcha-btn{width:38px;height:38px;background:#fff;border:1px solid #ccc;border-radius:50%;position:absolute;top:0;left:0;cursor:pointer;display:flex;align-items:center;justify-content:center;box-shadow:0 2px 4px rgba(0,0,0,.1);transition:all var(--sc-transition);user-select:none;z-index:1}.slider-captcha-loading{position:absolute;top:0;left:0;width:100%;height:100%;background:rgba(255,255,255,.6);display:flex;align-items:center;justify-content:center;flex-direction:column;color:#666;font-size:14px;z-index:10;border-radius:4px}.slider-captcha-error{color:var(--sc-danger);font-size:12px;text-align:center;margin-top:10px;display:none}.slider-captcha-title{margin:0;font-size:16px;color:var(--sc-text)}.slider-captcha-close,.slider-captcha-refresh{background:none;border:none;cursor:pointer;color:var(--sc-text-light);padding:0;width:30px;height:30px;display:flex;align-items:center;justify-content:center;border-radius:50%;transition:all var(--sc-transition);position:relative;font-size:0}.slider-captcha-close::before,.slider-captcha-close::after{content:'';position:absolute;width:16px;height:2px;background-color:var(--sc-text-light);border-radius:1px;transition:all var(--sc-transition)}.slider-captcha-close::before{transform:rotate(45deg)}.slider-captcha-close::after{transform:rotate(-45deg)}.slider-captcha-close:hover{background:#f5f5f5;transform:scale(1.1)}.slider-captcha-close:hover::before,.slider-captcha-close:hover::after{background-color:var(--sc-danger)}.slider-captcha-refresh{margin-left:10px}.slider-captcha-refresh svg{width:20px;height:20px;fill:var(--sc-text-light);transition:all var(--sc-transition)}.slider-captcha-refresh:hover{background:#f5f5f5;transform:scale(1.1)}.slider-captcha-refresh:hover svg{fill:var(--sc-primary);transform:rotate(180deg)}.slider-captcha-floating-time{position:absolute;bottom:-40px;left:50%;transform:translateX(-50%);color:#fff;font-size:12px;white-space:nowrap;opacity:0;pointer-events:none;z-index:10;transition:all var(--sc-transition);background:#fff;padding:2px 15px;border-radius:10px}.slider-captcha-floating-time.show{opacity:1;transform:translateX(-50%) translateY(-45px)}.slider-captcha-floating-time.success{color:var(--sc-success)}.slider-captcha-floating-time.fail{color:var(--sc-danger)}.slider-captcha-bg{width:100%;height:100%;object-fit:cover;display:block}.slider-captcha-piece{position:absolute;top:0;left:0;cursor:pointer;transition:none;z-index:2}.slider-captcha-finger{position:absolute;top:50%;left:10px;transform:translateY(-50%);font-size:20px;animation:fingerSlide 2s ease-in-out infinite;pointer-events:none;z-index:1;opacity:.6}.slider-captcha-hint{position:absolute;top:50%;left:50%;transform:translate(-50%,-50%);color:var(--sc-text-light);font-size:14px;pointer-events:none;z-index:1;transition:all var(--sc-transition)}.slider-captcha-header-buttons{display:flex;align-items:center}@keyframes fingerSlide{0%{left:10px;opacity:.6}50%{opacity:1}100%{left:calc(50% - 10px);opacity:.6}}"}constructor(t={}){this.options={...PopupSliderCaptcha.DEFAULTS,...t},this.elements={},this.state=this.createInitialState(),this.captchaData=null,this.times=[],this.startTime=null,this.eventListeners=[],this.timers=new Set,this.rafId=null,this.cachedDimensions=null,this.imageCache=new Map,this.throttledHandleMove=this.throttle((t=>this.handleMove(t)),this.options.throttleDelay),this.init()}createInitialState(){return{isVisible:!1,isDragging:!1,currentX:0,startX:0,retryCount:0,isLoading:!1}}init(){this.injectStyles(),this.createElements(),this.bindEvents()}injectStyles(){if(document.querySelector("#slider-captcha-styles"))return
|
|
6
|
+
const t=document.createElement("style")
|
|
7
|
+
t.id="slider-captcha-styles",t.textContent=PopupSliderCaptcha.getStyles(),document.head.appendChild(t)}createElements(){const{elements:t,options:i}=this;[["overlay","div",PopupSliderCaptcha.CSS_CLASSES.overlay],["modal","div",PopupSliderCaptcha.CSS_CLASSES.modal],["header","div",PopupSliderCaptcha.CSS_CLASSES.header],["title","h3","slider-captcha-title","安全验证"],["closeBtn","button","slider-captcha-close"],["refreshBtn","button","slider-captcha-refresh"],["container","div",PopupSliderCaptcha.CSS_CLASSES.container],["backgroundImg","img","slider-captcha-bg"],["sliderImg","img","slider-captcha-piece"],["loadingText","div",PopupSliderCaptcha.CSS_CLASSES.loading,"加载中..."],["floatingTime","div","slider-captcha-floating-time"],["track","div",PopupSliderCaptcha.CSS_CLASSES.track],["fingerAnimation","div","slider-captcha-finger","👉"],["btn","div",PopupSliderCaptcha.CSS_CLASSES.btn],["icon","div","","→"],["hint","div",PopupSliderCaptcha.CSS_CLASSES.hint,"向右滑动完成验证"],["error","div",PopupSliderCaptcha.CSS_CLASSES.error]].forEach((([i,e,s,a])=>{t[i]=this.createElement(e,s,a)})),t.container.style.cssText=`width:${i.width}px;height:${i.height}px`,t.refreshBtn.innerHTML='<svg viewBox="0 0 24 24"><path d="M17.65,6.35C16.2,4.9 14.21,4 12,4A8,8 0 0,0 4,12A8,8 0 0,0 12,20C15.73,20 18.84,17.45 19.73,14H17.65C16.83,16.33 14.61,18 12,18A6,6 0 0,1 6,12A6,6 0 0,1 12,6C13.66,6 15.14,6.69 16.22,7.78L13,11H20V4L17.65,6.35Z"/></svg>',this.assembleDOM(),this.setInitialState()}createElement(t,i="",e=""){const s=document.createElement(t)
|
|
8
|
+
return i&&(s.className=i),e&&(s.textContent=e),s}assembleDOM(){const{elements:t}=this,i=this.createElement("div","slider-captcha-header-buttons")
|
|
9
|
+
i.append(t.refreshBtn,t.closeBtn),t.header.append(t.title,i),t.container.append(t.backgroundImg,t.sliderImg,t.loadingText,t.floatingTime),t.btn.appendChild(t.icon),t.track.append(t.fingerAnimation,t.btn,t.hint),t.modal.append(t.header,t.container,t.track,t.error),t.overlay.appendChild(t.modal),document.body.appendChild(t.overlay)}setInitialState(){Object.assign(this.elements.container.style,{display:"none"}),Object.assign(this.elements.track.style,{display:"none"})}bindEvents(){const{elements:t}=this
|
|
10
|
+
this.addEventListener(t.closeBtn,"click",(()=>this.hide())),this.addEventListener(t.refreshBtn,"click",(()=>this.refresh())),this.addEventListener(t.overlay,"click",(i=>{i.target===t.overlay&&this.options.clickMaskClose&&this.hide()})),this.addEventListener(document,"keydown",(t=>{"Escape"===t.key&&this.state.isVisible&&this.hide()})),this.addEventListener(document,"visibilitychange",(()=>this.handleVisibilityChange())),this.bindSliderEvents()}bindSliderEvents(){const{elements:t}=this,i={start:t=>this.handleStart(t),move:this.throttledHandleMove,end:()=>this.handleEnd()}
|
|
11
|
+
this.addEventListener(t.btn,"mousedown",i.start),this.addEventListener(t.btn,"touchstart",i.start),this.addEventListener(t.sliderImg,"mousedown",i.start),this.addEventListener(t.sliderImg,"touchstart",i.start),this.addEventListener(document,"mousemove",i.move,{passive:!1}),this.addEventListener(document,"touchmove",i.move,{passive:!1}),this.addEventListener(document,"mouseup",i.end),this.addEventListener(document,"touchend",i.end)}addEventListener(t,i,e,s={}){t&&"function"==typeof e&&(t.addEventListener(i,e,s),this.eventListeners.push({element:t,event:i,handler:e,options:s}))}removeAllEventListeners(){this.eventListeners.forEach((({element:t,event:i,handler:e,options:s})=>{try{t?.removeEventListener?.(i,e,s)}catch(t){}})),this.eventListeners.length=0}getDimensions(){if(!this.cachedDimensions){const t=this.elements.track.offsetWidth,i=this.elements.btn.offsetWidth
|
|
12
|
+
this.cachedDimensions={trackWidth:t,btnWidth:i,maxX:t-i}}return this.cachedDimensions}getPosition(){const{maxX:t}=this.getDimensions(),i=this.state.currentX/t
|
|
13
|
+
return Math.round(i*(this.options.width-this.options.sliderSize))}handleStart(t){!this.captchaData||this.state.isDragging||this.state.isLoading||(t.preventDefault(),this.state.isDragging=!0,this.state.startX=this.getClientX(t)-this.state.currentX,this.startTime=this.startTime||Date.now(),this.times=[{time:Date.now(),position:this.getPosition()}],this.setTransition(!1),this.updateUIState("dragging"),this.cachedDimensions=null)}handleMove(t){if(!this.state.isDragging)return
|
|
14
|
+
t.preventDefault()
|
|
15
|
+
const i=this.getClientX(t)-this.state.startX,{maxX:e}=this.getDimensions()
|
|
16
|
+
this.state.currentX=Math.max(0,Math.min(i,e)),this.times.push({time:Date.now(),position:this.getPosition()}),this.rafId&&cancelAnimationFrame(this.rafId),this.rafId=requestAnimationFrame((()=>this.updateSliderPosition()))}handleEnd(){this.state.isDragging&&(this.times.push({time:Date.now(),position:this.getPosition()}),this.state.isDragging=!1,this.rafId&&(cancelAnimationFrame(this.rafId),this.rafId=null),this.verify())}handleVisibilityChange(){const t=document.hidden?"paused":"running"
|
|
17
|
+
this.elements.fingerAnimation&&(this.elements.fingerAnimation.style.animationPlayState=t)}getClientX(t){return t.type.includes("touch")?t.touches[0].clientX:t.clientX}setTransition(t){const i=t?"all 0.3s ease":"none"
|
|
18
|
+
requestAnimationFrame((()=>{this.elements.btn.style.transition=i,this.elements.sliderImg.style.transition=i}))}updateUIState(t){const{elements:i}=this,e={dragging:()=>{i.hint.style.opacity="0",i.fingerAnimation.style.display="none"},success:()=>{Object.assign(i.btn.style,{background:"var(--sc-success)"}),Object.assign(i.icon.style,{innerHTML:"✓",color:"white"}),i.icon.innerHTML="✓"},fail:()=>{Object.assign(i.btn.style,{background:"var(--sc-danger)"}),Object.assign(i.icon.style,{innerHTML:"✗",color:"white"}),i.icon.innerHTML="✗"},reset:()=>{Object.assign(i.btn.style,{background:"white"}),Object.assign(i.icon.style,{color:"#666"}),i.icon.innerHTML="→",i.fingerAnimation.style.display="block",this.updateHintText("向右滑动完成验证","var(--sc-text-light)")},loading:()=>{i.hint.style.opacity="0",i.fingerAnimation.style.display="none",Object.assign(i.track.style,{pointerEvents:"none",opacity:"0.6"})}}
|
|
19
|
+
e[t]&&requestAnimationFrame((()=>{e[t](),"loading"!==t&&Object.assign(i.track.style,{pointerEvents:"auto",opacity:"1"})}))}updateHintText(t,i){requestAnimationFrame((()=>{Object.assign(this.elements.hint,{textContent:t}),Object.assign(this.elements.hint.style,{color:i,opacity:"1"})}))}updateSliderPosition(){const{elements:t,options:i,state:e}=this,{maxX:s}=this.getDimensions(),a=e.currentX/s*(i.width-i.sliderSize),r=e.currentX/s
|
|
20
|
+
requestAnimationFrame((()=>{t.btn.style.transform=`translateX(${e.currentX}px)`,t.sliderImg.style.transform=`translateX(${a}px)`,t.fingerAnimation.style.opacity=.8>r?"0.6":"0"}))}show(){this.state.isVisible=!0,this.elements.overlay.style.display="flex",this.elements.overlay.offsetHeight,requestAnimationFrame((()=>{this.elements.overlay.classList.add("show"),this.elements.modal.classList.add("show")})),this.loadCaptcha()}hide(){this.state.isVisible=!1,this.elements.overlay.classList.remove("show"),this.elements.modal.classList.remove("show"),this.safeSetTimeout((()=>{this.elements.overlay.style.display="none",this.reset(),this.options.onClose?.()}),300)}async loadCaptcha(){try{this.showLoading(),this.startTime=Date.now()
|
|
21
|
+
const t=new AbortController,i=this.safeSetTimeout((()=>t.abort()),this.options.timeout),e=await fetch(this.options.apiUrl,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({place:2,timestamp:Date.now()}),signal:t.signal})
|
|
22
|
+
if(this.safeClearTimeout(i),!e.ok)throw Error("HTTP "+e.status)
|
|
23
|
+
const s=await e.json()
|
|
24
|
+
if(!s.data?.canvasSrc||!s.data?.blockSrc)throw Error("验证码数据格式错误")
|
|
25
|
+
this.captchaData=s.data,this.showCaptcha(),await this.renderCaptcha()}catch(t){const i="AbortError"===t.name?"请求超时,请重试":"加载验证码失败: "+t.message
|
|
26
|
+
this.showError(i)}}async renderCaptcha(){return new Promise(((t,i)=>{let e=0
|
|
27
|
+
const s=()=>2===++e&&(this.hideLoading(),t()),a=()=>i(Error("图片加载失败"))
|
|
28
|
+
this.loadImage(this.elements.backgroundImg,this.captchaData.canvasSrc,{width:this.captchaData.canvasWidth,height:this.captchaData.canvasHeight},s,a),this.loadImage(this.elements.sliderImg,this.captchaData.blockSrc,{width:this.captchaData.blockWidth,height:this.captchaData.blockHeight,top:this.captchaData.blockY},s,a)}))}loadImage(t,i,e,s,a){if(this.imageCache.has(i)){const a=this.imageCache.get(i)
|
|
29
|
+
return t.src=a.src,this.applyStyles(t,e),void s()}t.onload=()=>{this.imageCache.set(i,t.cloneNode()),s()},t.onerror=a,t.src=i,this.applyStyles(t,e)}applyStyles(t,i){Object.entries(i).forEach((([i,e])=>{t.style[i]="number"==typeof e?e+"px":e}))}showLoading(){this.state.isLoading=!0,this.batchUpdateStyles({container:{display:"block"},loadingText:{display:"flex"},error:{display:"none"}}),this.updateUIState("loading")}hideLoading(){this.state.isLoading=!1,this.batchUpdateStyles({loadingText:{display:"none"}}),this.updateUIState("reset")}showCaptcha(){this.batchUpdateStyles({container:{display:"block"},track:{display:"block"},error:{display:"none"}})}showError(t){this.hideLoading(),this.batchUpdateStyles({error:{display:"block",textContent:t}})}batchUpdateStyles(t){requestAnimationFrame((()=>{Object.entries(t).forEach((([t,i])=>{const e=this.elements[t]
|
|
30
|
+
e&&Object.entries(i).forEach((([t,i])=>{"textContent"===t?e.textContent=i:e.style[t]=i}))}))}))}async verify(){if(this.captchaData)try{const t=new AbortController,i=this.safeSetTimeout((()=>t.abort()),this.options.timeout),e=await fetch(this.options.verifyUrl,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({loginVo:{nonceStr:this.captchaData.nonceStr,value:this.getPosition()},dragEventList:[...this.times]}),signal:t.signal})
|
|
31
|
+
if(this.safeClearTimeout(i),!e.ok)throw Error("HTTP "+e.status)
|
|
32
|
+
const s=await e.json()
|
|
33
|
+
"0"===s.code||!0===s.success?this.onVerifySuccess():this.onVerifyFail(s.message||"验证失败,请重试!")}catch(t){const i="AbortError"===t.name?"验证超时,请重试":"网络错误"
|
|
34
|
+
this.onVerifyFail(i)}else this.onVerifyFail("验证码数据丢失")}onVerifySuccess(){const t=Date.now()-this.startTime,i=`验证成功!耗时:${(t/1e3).toFixed(2)}s`
|
|
35
|
+
this.updateUIState("success"),this.showFloatingTime(i,"success"),this.safeSetTimeout((()=>{this.options.onSuccess?.({captchaId:this.captchaData.captchaId,timestamp:Date.now(),duration:t}),this.hide()}),2e3)}showFloatingTime(t,i="success"){const{elements:e}=this
|
|
36
|
+
e.floatingTime.textContent=t,e.floatingTime.className="slider-captcha-floating-time "+i,this.safeSetTimeout((()=>e.floatingTime.classList.add("show")),100),this.safeSetTimeout((()=>{e.floatingTime.className="slider-captcha-floating-time"}),2500)}onVerifyFail(t){this.state.retryCount++,this.updateUIState("fail"),this.showFloatingTime(t,"fail"),this.safeSetTimeout((()=>{this.state.retryCount<this.options.maxRetries?this.reset():this.refresh()}),2500)}reset(){this.clearAllTimers(),Object.assign(this.state,{isDragging:!1,currentX:0,startX:0,isLoading:!1}),this.times=[],this.startTime=null,this.cachedDimensions=null,requestAnimationFrame((()=>{this.setTransition(!0),this.elements.btn.style.transform="translateX(0px)",this.elements.sliderImg.style.transform="translateX(0px)",this.updateUIState("reset"),this.elements.error.style.display="none"}))}refresh(){this.reset(),this.state.retryCount=0,this.loadCaptcha()}safeSetTimeout(t,i){const e=setTimeout((()=>{this.timers.delete(e),t()}),i)
|
|
37
|
+
return this.timers.add(e),e}safeClearTimeout(t){t&&(clearTimeout(t),this.timers.delete(t))}clearAllTimers(){this.timers.forEach((t=>{clearTimeout(t),clearInterval(t)})),this.timers.clear(),this.rafId&&(cancelAnimationFrame(this.rafId),this.rafId=null)}cleanupImages(){this.elements.backgroundImg&&(this.elements.backgroundImg.src="",this.elements.backgroundImg.onload=null,this.elements.backgroundImg.onerror=null),this.elements.sliderImg&&(this.elements.sliderImg.src="",this.elements.sliderImg.onload=null,this.elements.sliderImg.onerror=null),this.imageCache.clear()}throttle(t,i){let e=0
|
|
38
|
+
return function(...s){const a=Date.now()
|
|
39
|
+
if(a-e>=i)return e=a,t.apply(this,s)}}debounce(t,i){let e
|
|
40
|
+
return function(...s){clearTimeout(e),e=setTimeout((()=>t.apply(this,s)),i)}}destroy(){document.body.style.userSelect="",document.body.style.cursor="",this.clearAllTimers(),this.removeAllEventListeners(),this.cleanupImages(),this.elements.overlay?.parentNode&&this.elements.overlay.parentNode.removeChild(this.elements.overlay)
|
|
41
|
+
const t=document.getElementById("slider-captcha-styles")
|
|
42
|
+
t&&t.remove(),Object.keys(this).forEach((t=>{"constructor"!==t&&(this[t]=null)}))}static create(t){return new PopupSliderCaptcha(t)}static show(t){const i=new PopupSliderCaptcha(t)
|
|
43
|
+
return i.show(),i}}"undefined"!=typeof module&&module.exports?(module.exports=PopupSliderCaptcha,module.exports.default=PopupSliderCaptcha):"function"==typeof define&&define.amd?define([],(()=>PopupSliderCaptcha)):"undefined"!=typeof window&&(window.PopupSliderCaptcha=PopupSliderCaptcha,window.SliderCaptcha=PopupSliderCaptcha)
|
|
44
|
+
class PasswordValidator{constructor(t={}){this.options={publicKeyUrl:t.publicKeyUrl||"/microservice/strongPassword/getPublicKey",validateUrl:t.validateUrl||"/microservice/strongPassword/checkPassword",timeout:t.timeout||1e4,headers:t.headers||{},...t},this.publicKeyCache=null,this.publicKeyExpiry=null}async getPublicKey(){if(this.publicKeyCache&&this.publicKeyExpiry&&Date.now()<this.publicKeyExpiry)return this.publicKeyCache
|
|
45
|
+
try{const t=await this.makeRequest(this.options.publicKeyUrl,{method:"GET"})
|
|
46
|
+
if("0"!==t.code)throw Error("获取公钥失败:"+(t.message||"未知错误"))
|
|
47
|
+
return this.publicKeyCache=t.data,this.publicKeyExpiry=Date.now()+3e5,this.publicKeyCache}catch(t){throw Error("获取公钥失败: "+t.message)}}encryptPassword(t,i){try{if("undefined"==typeof JSEncrypt)throw Error("JSEncrypt库未加载,请先引入jsencrypt库")
|
|
48
|
+
const e=new JSEncrypt
|
|
49
|
+
e.setPublicKey(i)
|
|
50
|
+
const s=e.encrypt(t)
|
|
51
|
+
if(!s)throw Error("密码加密失败")
|
|
52
|
+
return s}catch(t){throw Error("密码加密失败: "+t.message)}}async validatePassword(t,i={}){try{const e=await this.getPublicKey(),s=this.encryptPassword(t,e),a=await this.makeRequest(this.options.validateUrl,{method:"POST",body:JSON.stringify({userName:"13212345678",password:s,timestamp:Date.now(),...i})})
|
|
53
|
+
return{success:a.success,message:a.message,data:a.data,code:a.code}}catch(t){return{success:!1,message:t.message,code:"VALIDATION_ERROR"}}}async makeRequest(t,i={}){const e=new AbortController,s=setTimeout((()=>e.abort()),this.options.timeout)
|
|
54
|
+
try{const a=await fetch(t,{...i,headers:{"Content-Type":"application/json",...this.options.headers,...i.headers},signal:e.signal})
|
|
55
|
+
if(clearTimeout(s),!a.ok)throw Error(`HTTP错误: ${a.status} ${a.statusText}`)
|
|
56
|
+
return await a.json()}catch(t){if(clearTimeout(s),"AbortError"===t.name)throw Error("请求超时")
|
|
57
|
+
throw t}}clearCache(){this.publicKeyCache=null,this.publicKeyExpiry=null}updateOptions(t){this.options={...this.options,...t},this.clearCache()}}var s={PopupSliderCaptcha:PopupSliderCaptcha,PasswordValidator:PasswordValidator,createPasswordValidator:i,validatePassword:e}
|
|
58
|
+
"undefined"!=typeof window&&(window.SliderCaptcha=PopupSliderCaptcha,window.PopupSliderCaptcha=PopupSliderCaptcha,window.PasswordValidator=PasswordValidator,window.createPasswordValidator=i,window.validatePassword=e),t.PasswordValidator=PasswordValidator,t.PopupSliderCaptcha=PopupSliderCaptcha,t.createPasswordValidator=i,t.default=s,t.validatePassword=e,Object.defineProperty(t,"t",{value:!0})}))
|
|
59
|
+
//# sourceMappingURL=index.min.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.min.js","sources":["../src/password-validator.js","../src/slider-captcha.js","../src/index.js"],"sourcesContent":["// import JSEncrypt from 'jsencrypt'\n\n/**\n * 密码校验工具类\n * 提供密码加密和校验功能\n */\nclass PasswordValidator {\n constructor(options = {}) {\n this.options = {\n // 获取公钥的接口地址\n publicKeyUrl: options.publicKeyUrl || '/microservice/strongPassword/getPublicKey',\n // 密码校验接口地址\n validateUrl: options.validateUrl || '/microservice/strongPassword/checkPassword',\n // 请求超时时间\n timeout: options.timeout || 10000,\n // 自定义请求头\n headers: options.headers || {},\n ...options\n }\n\n // 缓存公钥,避免重复请求\n this.publicKeyCache = null\n this.publicKeyExpiry = null\n }\n\n /**\n * 获取公钥\n * @returns {Promise<string>} 公钥字符串\n */\n async getPublicKey() {\n // 检查缓存是否有效(5分钟有效期)\n if (this.publicKeyCache && this.publicKeyExpiry && Date.now() < this.publicKeyExpiry) {\n return this.publicKeyCache\n }\n\n try {\n const response = await this.makeRequest(this.options.publicKeyUrl, {\n method: 'GET'\n })\n\n if (response.code !== '0' ) {\n throw new Error('获取公钥失败:' + (response.message || '未知错误'))\n }\n\n // 缓存公钥,设置5分钟过期时间\n this.publicKeyCache = response.data\n this.publicKeyExpiry = Date.now() + 5 * 60 * 1000\n\n return this.publicKeyCache\n } catch (error) {\n console.error('获取公钥失败:', error)\n throw new Error('获取公钥失败: ' + error.message)\n }\n }\n\n /**\n * 使用RSA公钥加密密码\n * @param {string} password 原始密码\n * @param {string} publicKey 公钥字符串\n * @returns {string} 加密后的密码\n */\n encryptPassword(password, publicKey) {\n try {\n // 这里使用JSEncrypt库进行RSA加密\n // 需要先安装: npm install jsencrypt\n if (typeof JSEncrypt === 'undefined') {\n throw new Error('JSEncrypt库未加载,请先引入jsencrypt库')\n }\n\n const encrypt = new JSEncrypt()\n encrypt.setPublicKey(publicKey)\n const encrypted = encrypt.encrypt(password)\n\n if (!encrypted) {\n throw new Error('密码加密失败')\n }\n\n return encrypted\n } catch (error) {\n console.error('密码加密失败:', error)\n throw new Error('密码加密失败: ' + error.message)\n }\n }\n\n /**\n * 校验密码\n * @param {string} password 原始密码\n * @param {Object} additionalData 额外的校验数据\n * @returns {Promise<Object>} 校验结果\n */\n async validatePassword(password, additionalData = {}) {\n try {\n // 1. 获取公钥\n const publicKey = await this.getPublicKey()\n\n // 2. 加密密码\n const encryptedPassword = this.encryptPassword(password, publicKey)\n\n // 3. 调用校验接口\n const response = await this.makeRequest(this.options.validateUrl, {\n method: 'POST',\n body: JSON.stringify({\n userName: '13212345678',\n password: encryptedPassword,\n timestamp: Date.now(),\n ...additionalData\n })\n })\n\n return {\n success: response.success,\n message: response.message,\n data: response.data,\n code: response.code\n }\n } catch (error) {\n console.error('密码校验失败:', error)\n return {\n success: false,\n message: error.message,\n code: 'VALIDATION_ERROR'\n }\n }\n }\n\n /**\n * 发送HTTP请求的通用方法\n * @param {string} url 请求地址\n * @param {Object} options 请求选项\n * @returns {Promise<Object>} 响应结果\n */\n async makeRequest(url, options = {}) {\n const controller = new AbortController()\n const timeoutId = setTimeout(() => controller.abort(), this.options.timeout)\n\n try {\n const response = await fetch(url, {\n ...options,\n headers: {\n 'Content-Type': 'application/json',\n ...this.options.headers,\n ...options.headers\n },\n signal: controller.signal\n })\n\n clearTimeout(timeoutId)\n\n if (!response.ok) {\n throw new Error(`HTTP错误: ${response.status} ${response.statusText}`)\n }\n\n return await response.json()\n } catch (error) {\n clearTimeout(timeoutId)\n\n if (error.name === 'AbortError') {\n throw new Error('请求超时')\n }\n\n throw error\n }\n }\n\n /**\n * 清除公钥缓存\n */\n clearCache() {\n this.publicKeyCache = null\n this.publicKeyExpiry = null\n }\n\n /**\n * 更新配置\n * @param {Object} newOptions 新的配置选项\n */\n updateOptions(newOptions) {\n this.options = { ...this.options, ...newOptions }\n // 清除缓存,使用新配置重新获取\n this.clearCache()\n }\n}\n\n// 导出类和创建实例的工厂函数\nexport default PasswordValidator\n\n/**\n * 创建密码校验器实例的工厂函数\n * @param {Object} options 配置选项\n * @returns {PasswordValidator} 密码校验器实例\n */\nexport function createPasswordValidator(options) {\n return new PasswordValidator(options)\n}\n\n/**\n * 快速校验密码的便捷函数\n * @param {string} password 密码\n * @param {Object} options 配置选项\n * @param {Object} additionalData 额外数据\n * @returns {Promise<Object>} 校验结果\n */\nexport async function validatePassword(password, options = {}, additionalData = {}) {\n const validator = new PasswordValidator(options)\n return await validator.validatePassword(password, additionalData)\n}\n","/**\n * 纯JavaScript弹窗滑块验证码组件\n */\nclass PopupSliderCaptcha {\n static DEFAULTS = {\n width: 320,\n height: 155,\n sliderSize: 38,\n maxRetries: 3,\n timeout: 30000,\n apiUrl: \"/api/captcha\",\n verifyUrl: \"/api/captcha/verify\",\n throttleDelay: 16,\n clickMaskClose: false\n }\n\n static CSS_CLASSES = {\n overlay: \"slider-captcha-overlay\",\n modal: \"slider-captcha-modal\",\n header: \"slider-captcha-header\",\n container: \"slider-captcha-container\",\n track: \"slider-captcha-track\",\n btn: \"slider-captcha-btn\",\n hint: \"slider-captcha-hint\",\n loading: \"slider-captcha-loading\",\n error: \"slider-captcha-error\"\n }\n\n // 优化:提取CSS样式为独立方法,减少主体代码长度\n static getStyles() {\n return `:root{--sc-primary:#409eff;--sc-success:#67c23a;--sc-danger:#f56c6c;--sc-border:#e4e7eb;--sc-bg:linear-gradient(90deg, #f7f9fa 0%, #e8f4fd 100%);--sc-text:#333;--sc-text-light:#999;--sc-shadow:0 4px 20px rgba(0,0,0,.3);--sc-radius:8px;--sc-transition:.3s ease}.slider-captcha-overlay{position:fixed;top:0;left:0;width:100%;height:100%;background:rgba(0,0,0,.5);z-index:9999;display:none;justify-content:center;align-items:center;opacity:0;transition:opacity var(--sc-transition)}.slider-captcha-overlay.show{opacity:1}.slider-captcha-modal{background:#fff;border-radius:var(--sc-radius);padding:20px;box-shadow:var(--sc-shadow);position:relative;max-width:90vw;max-height:90vh;transform:scale(.8) translateY(-20px);opacity:0;transition:all var(--sc-transition)}.slider-captcha-modal.show{transform:scale(1) translateY(0);opacity:1}.slider-captcha-header{display:flex;justify-content:space-between;align-items:center;margin-bottom:15px;padding-bottom:10px;border-bottom:1px solid var(--sc-border)}.slider-captcha-container{display:flex;align-items:center;position:relative;border-radius:4px;overflow:hidden;margin-bottom:15px;background:#837a7a;justify-content:center}.slider-captcha-track{width:100%;height:40px;line-height:40px;background:var(--sc-bg);border:1px solid var(--sc-border);border-radius:20px;position:relative;margin-bottom:15px;overflow:hidden}.slider-captcha-btn{width:38px;height:38px;background:#fff;border:1px solid #ccc;border-radius:50%;position:absolute;top:0;left:0;cursor:pointer;display:flex;align-items:center;justify-content:center;box-shadow:0 2px 4px rgba(0,0,0,.1);transition:all var(--sc-transition);user-select:none;z-index:1}.slider-captcha-loading{position:absolute;top:0;left:0;width:100%;height:100%;background:rgba(255,255,255,.6);display:flex;align-items:center;justify-content:center;flex-direction:column;color:#666;font-size:14px;z-index:10;border-radius:4px}.slider-captcha-error{color:var(--sc-danger);font-size:12px;text-align:center;margin-top:10px;display:none}.slider-captcha-title{margin:0;font-size:16px;color:var(--sc-text)}.slider-captcha-close,.slider-captcha-refresh{background:none;border:none;cursor:pointer;color:var(--sc-text-light);padding:0;width:30px;height:30px;display:flex;align-items:center;justify-content:center;border-radius:50%;transition:all var(--sc-transition);position:relative;font-size:0}.slider-captcha-close::before,.slider-captcha-close::after{content:'';position:absolute;width:16px;height:2px;background-color:var(--sc-text-light);border-radius:1px;transition:all var(--sc-transition)}.slider-captcha-close::before{transform:rotate(45deg)}.slider-captcha-close::after{transform:rotate(-45deg)}.slider-captcha-close:hover{background:#f5f5f5;transform:scale(1.1)}.slider-captcha-close:hover::before,.slider-captcha-close:hover::after{background-color:var(--sc-danger)}.slider-captcha-refresh{margin-left:10px}.slider-captcha-refresh svg{width:20px;height:20px;fill:var(--sc-text-light);transition:all var(--sc-transition)}.slider-captcha-refresh:hover{background:#f5f5f5;transform:scale(1.1)}.slider-captcha-refresh:hover svg{fill:var(--sc-primary);transform:rotate(180deg)}.slider-captcha-floating-time{position:absolute;bottom:-40px;left:50%;transform:translateX(-50%);color:#fff;font-size:12px;white-space:nowrap;opacity:0;pointer-events:none;z-index:10;transition:all var(--sc-transition);background:#fff;padding:2px 15px;border-radius:10px}.slider-captcha-floating-time.show{opacity:1;transform:translateX(-50%) translateY(-45px)}.slider-captcha-floating-time.success{color:var(--sc-success)}.slider-captcha-floating-time.fail{color:var(--sc-danger)}.slider-captcha-bg{width:100%;height:100%;object-fit:cover;display:block}.slider-captcha-piece{position:absolute;top:0;left:0;cursor:pointer;transition:none;z-index:2}.slider-captcha-finger{position:absolute;top:50%;left:10px;transform:translateY(-50%);font-size:20px;animation:fingerSlide 2s ease-in-out infinite;pointer-events:none;z-index:1;opacity:.6}.slider-captcha-hint{position:absolute;top:50%;left:50%;transform:translate(-50%,-50%);color:var(--sc-text-light);font-size:14px;pointer-events:none;z-index:1;transition:all var(--sc-transition)}.slider-captcha-header-buttons{display:flex;align-items:center}@keyframes fingerSlide{0%{left:10px;opacity:.6}50%{opacity:1}100%{left:calc(50% - 10px);opacity:.6}}`\n }\n\n constructor(options = {}) {\n this.options = { ...PopupSliderCaptcha.DEFAULTS, ...options }\n this.elements = {}\n this.state = this.createInitialState()\n this.captchaData = null\n this.times = []\n this.startTime = null\n this.eventListeners = []\n this.timers = new Set()\n this.rafId = null\n this.cachedDimensions = null\n this.imageCache = new Map()\n\n // 优化:使用箭头函数绑定this,避免重复bind调用\n this.throttledHandleMove = this.throttle((e) => this.handleMove(e), this.options.throttleDelay)\n\n this.init()\n }\n\n // 优化:提取初始状态创建为独立方法\n createInitialState() {\n return {\n isVisible: false,\n isDragging: false,\n currentX: 0,\n startX: 0,\n retryCount: 0,\n isLoading: false\n }\n }\n\n init() {\n this.injectStyles()\n this.createElements()\n this.bindEvents()\n }\n\n injectStyles() {\n if (document.querySelector(\"#slider-captcha-styles\")) return\n\n const style = document.createElement(\"style\")\n style.id = \"slider-captcha-styles\"\n style.textContent = PopupSliderCaptcha.getStyles()\n document.head.appendChild(style)\n }\n\n // 优化:简化元素创建逻辑\n createElements() {\n const { elements, options } = this\n\n // 批量创建元素配置\n const elementConfigs = [\n ['overlay', 'div', PopupSliderCaptcha.CSS_CLASSES.overlay],\n ['modal', 'div', PopupSliderCaptcha.CSS_CLASSES.modal],\n ['header', 'div', PopupSliderCaptcha.CSS_CLASSES.header],\n ['title', 'h3', 'slider-captcha-title', '安全验证'],\n ['closeBtn', 'button', 'slider-captcha-close'],\n ['refreshBtn', 'button', 'slider-captcha-refresh'],\n ['container', 'div', PopupSliderCaptcha.CSS_CLASSES.container],\n ['backgroundImg', 'img', 'slider-captcha-bg'],\n ['sliderImg', 'img', 'slider-captcha-piece'],\n ['loadingText', 'div', PopupSliderCaptcha.CSS_CLASSES.loading, '加载中...'],\n ['floatingTime', 'div', 'slider-captcha-floating-time'],\n ['track', 'div', PopupSliderCaptcha.CSS_CLASSES.track],\n ['fingerAnimation', 'div', 'slider-captcha-finger', '👉'],\n ['btn', 'div', PopupSliderCaptcha.CSS_CLASSES.btn],\n ['icon', 'div', '', '→'],\n ['hint', 'div', PopupSliderCaptcha.CSS_CLASSES.hint, '向右滑动完成验证'],\n ['error', 'div', PopupSliderCaptcha.CSS_CLASSES.error]\n ]\n\n // 批量创建元素\n elementConfigs.forEach(([key, tag, className, textContent]) => {\n elements[key] = this.createElement(tag, className, textContent)\n })\n\n // 设置容器尺寸\n elements.container.style.cssText = `width:${options.width}px;height:${options.height}px`\n\n // 添加刷新按钮图标\n elements.refreshBtn.innerHTML = `<svg viewBox=\"0 0 24 24\"><path d=\"M17.65,6.35C16.2,4.9 14.21,4 12,4A8,8 0 0,0 4,12A8,8 0 0,0 12,20C15.73,20 18.84,17.45 19.73,14H17.65C16.83,16.33 14.61,18 12,18A6,6 0 0,1 6,12A6,6 0 0,1 12,6C13.66,6 15.14,6.69 16.22,7.78L13,11H20V4L17.65,6.35Z\"/></svg>`\n\n this.assembleDOM()\n this.setInitialState()\n }\n\n createElement(tag, className = \"\", textContent = \"\") {\n const element = document.createElement(tag)\n if (className) element.className = className\n if (textContent) element.textContent = textContent\n return element\n }\n\n // 优化:简化DOM组装逻辑\n assembleDOM() {\n const { elements } = this\n\n // 组装头部\n const headerButtons = this.createElement(\"div\", \"slider-captcha-header-buttons\")\n headerButtons.append(elements.refreshBtn, elements.closeBtn)\n elements.header.append(elements.title, headerButtons)\n\n // 组装验证码容器\n elements.container.append(\n elements.backgroundImg,\n elements.sliderImg,\n elements.loadingText,\n elements.floatingTime\n )\n\n // 组装滑块轨道\n elements.btn.appendChild(elements.icon)\n elements.track.append(\n elements.fingerAnimation,\n elements.btn,\n elements.hint\n )\n\n // 组装模态框\n elements.modal.append(\n elements.header,\n elements.container,\n elements.track,\n elements.error\n )\n\n // 组装到覆盖层\n elements.overlay.appendChild(elements.modal)\n document.body.appendChild(elements.overlay)\n }\n\n setInitialState() {\n // 批量设置初始状态\n Object.assign(this.elements.container.style, { display: \"none\" })\n Object.assign(this.elements.track.style, { display: \"none\" })\n }\n\n // 优化:简化事件绑定\n bindEvents() {\n const { elements } = this\n\n // 基础事件\n this.addEventListener(elements.closeBtn, \"click\", () => this.hide())\n this.addEventListener(elements.refreshBtn, \"click\", () => this.refresh())\n this.addEventListener(elements.overlay, \"click\", (e) => {\n if (e.target === elements.overlay && this.options.clickMaskClose) this.hide()\n })\n this.addEventListener(document, \"keydown\", (e) => {\n if (e.key === \"Escape\" && this.state.isVisible) this.hide()\n })\n this.addEventListener(document, \"visibilitychange\", () => this.handleVisibilityChange())\n\n this.bindSliderEvents()\n }\n\n bindSliderEvents() {\n const { elements } = this\n const handlers = {\n start: (e) => this.handleStart(e),\n move: this.throttledHandleMove,\n end: () => this.handleEnd()\n }\n\n // 滑块事件\n this.addEventListener(elements.btn, \"mousedown\", handlers.start)\n this.addEventListener(elements.btn, \"touchstart\", handlers.start)\n this.addEventListener(elements.sliderImg, \"mousedown\", handlers.start)\n this.addEventListener(elements.sliderImg, \"touchstart\", handlers.start)\n this.addEventListener(document, \"mousemove\", handlers.move, { passive: false })\n this.addEventListener(document, \"touchmove\", handlers.move, { passive: false })\n this.addEventListener(document, \"mouseup\", handlers.end)\n this.addEventListener(document, \"touchend\", handlers.end)\n }\n\n // 优化:改进事件管理\n addEventListener(element, event, handler, options = {}) {\n if (!element || typeof handler !== 'function') return\n\n element.addEventListener(event, handler, options)\n this.eventListeners.push({ element, event, handler, options })\n }\n\n removeAllEventListeners() {\n this.eventListeners.forEach(({ element, event, handler, options }) => {\n try {\n element?.removeEventListener?.(event, handler, options)\n } catch (error) {\n console.warn('Failed to remove event listener:', error)\n }\n })\n this.eventListeners.length = 0\n }\n\n // 优化:缓存尺寸计算\n getDimensions() {\n if (!this.cachedDimensions) {\n const trackWidth = this.elements.track.offsetWidth\n const btnWidth = this.elements.btn.offsetWidth\n this.cachedDimensions = {\n trackWidth,\n btnWidth,\n maxX: trackWidth - btnWidth\n }\n }\n return this.cachedDimensions\n }\n\n getPosition() {\n const { maxX } = this.getDimensions()\n const percentage = this.state.currentX / maxX\n return Math.round(percentage * (this.options.width - this.options.sliderSize))\n }\n\n // 优化:简化拖拽处理\n handleStart(e) {\n if (!this.captchaData || this.state.isDragging || this.state.isLoading) return\n\n e.preventDefault()\n this.state.isDragging = true\n this.state.startX = this.getClientX(e) - this.state.currentX\n this.startTime = this.startTime || Date.now()\n this.times = [{ time: Date.now(), position: this.getPosition() }]\n\n this.setTransition(false)\n this.updateUIState(\"dragging\")\n this.cachedDimensions = null // 清除缓存\n }\n\n handleMove(e) {\n if (!this.state.isDragging) return\n e.preventDefault()\n\n const clientX = this.getClientX(e)\n const deltaX = clientX - this.state.startX\n const { maxX } = this.getDimensions()\n\n this.state.currentX = Math.max(0, Math.min(deltaX, maxX))\n this.times.push({ time: Date.now(), position: this.getPosition() })\n\n // 优化:使用RAF批量更新\n this.rafId && cancelAnimationFrame(this.rafId)\n this.rafId = requestAnimationFrame(() => this.updateSliderPosition())\n }\n\n handleEnd() {\n if (!this.state.isDragging) return\n\n this.times.push({ time: Date.now(), position: this.getPosition() })\n this.state.isDragging = false\n\n if (this.rafId) {\n cancelAnimationFrame(this.rafId)\n this.rafId = null\n }\n\n this.verify()\n }\n\n handleVisibilityChange() {\n const animationState = document.hidden ? 'paused' : 'running'\n if (this.elements.fingerAnimation) {\n this.elements.fingerAnimation.style.animationPlayState = animationState\n }\n }\n\n getClientX(e) {\n return e.type.includes(\"touch\") ? e.touches[0].clientX : e.clientX\n }\n\n setTransition(enabled) {\n const transition = enabled ? \"all 0.3s ease\" : \"none\"\n requestAnimationFrame(() => {\n this.elements.btn.style.transition = transition\n this.elements.sliderImg.style.transition = transition\n })\n }\n\n // 优化:简化UI状态更新\n updateUIState(state) {\n const { elements } = this\n const updates = {\n dragging: () => {\n elements.hint.style.opacity = \"0\"\n elements.fingerAnimation.style.display = \"none\"\n },\n success: () => {\n Object.assign(elements.btn.style, { background: \"var(--sc-success)\" })\n Object.assign(elements.icon.style, { innerHTML: \"✓\", color: \"white\" })\n elements.icon.innerHTML = \"✓\"\n },\n fail: () => {\n Object.assign(elements.btn.style, { background: \"var(--sc-danger)\" })\n Object.assign(elements.icon.style, { innerHTML: \"✗\", color: \"white\" })\n elements.icon.innerHTML = \"✗\"\n },\n reset: () => {\n Object.assign(elements.btn.style, { background: \"white\" })\n Object.assign(elements.icon.style, { color: \"#666\" })\n elements.icon.innerHTML = \"→\"\n elements.fingerAnimation.style.display = \"block\"\n this.updateHintText(\"向右滑动完成验证\", \"var(--sc-text-light)\")\n },\n loading: () => {\n elements.hint.style.opacity = \"0\"\n elements.fingerAnimation.style.display = \"none\"\n Object.assign(elements.track.style, { pointerEvents: \"none\", opacity: \"0.6\" })\n }\n }\n\n if (updates[state]) {\n requestAnimationFrame(() => {\n updates[state]()\n if (state !== \"loading\") {\n Object.assign(elements.track.style, { pointerEvents: \"auto\", opacity: \"1\" })\n }\n })\n }\n }\n\n updateHintText(text, color) {\n requestAnimationFrame(() => {\n Object.assign(this.elements.hint, { textContent: text })\n Object.assign(this.elements.hint.style, { color, opacity: \"1\" })\n })\n }\n\n updateSliderPosition() {\n const { elements, options, state } = this\n const { maxX } = this.getDimensions()\n const pieceX = (state.currentX / maxX) * (options.width - options.sliderSize)\n const progress = state.currentX / maxX\n\n requestAnimationFrame(() => {\n elements.btn.style.transform = `translateX(${state.currentX}px)`\n elements.sliderImg.style.transform = `translateX(${pieceX}px)`\n elements.fingerAnimation.style.opacity = progress >= 0.8 ? \"0\" : \"0.6\"\n })\n }\n\n // 优化:简化显示/隐藏逻辑\n show() {\n this.state.isVisible = true\n this.elements.overlay.style.display = \"flex\"\n this.elements.overlay.offsetHeight // 强制重绘\n\n requestAnimationFrame(() => {\n this.elements.overlay.classList.add(\"show\")\n this.elements.modal.classList.add(\"show\")\n })\n\n this.loadCaptcha()\n }\n\n hide() {\n this.state.isVisible = false\n this.elements.overlay.classList.remove(\"show\")\n this.elements.modal.classList.remove(\"show\")\n\n this.safeSetTimeout(() => {\n this.elements.overlay.style.display = \"none\"\n this.reset()\n this.options.onClose?.()\n }, 300)\n }\n\n // 优化:简化加载逻辑\n async loadCaptcha() {\n try {\n this.showLoading()\n this.startTime = Date.now()\n\n const controller = new AbortController()\n const timeoutId = this.safeSetTimeout(() => controller.abort(), this.options.timeout)\n\n const response = await fetch(this.options.apiUrl, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({ place: 2, timestamp: Date.now() }),\n signal: controller.signal\n })\n\n this.safeClearTimeout(timeoutId)\n\n if (!response.ok) throw new Error(`HTTP ${response.status}`)\n const data = await response.json()\n\n if (data.data?.canvasSrc && data.data?.blockSrc) {\n this.captchaData = data.data\n this.showCaptcha()\n await this.renderCaptcha()\n } else {\n throw new Error(\"验证码数据格式错误\")\n }\n } catch (error) {\n const message = error.name === 'AbortError' ? \"请求超时,请重试\" : `加载验证码失败: ${error.message}`\n this.showError(message)\n }\n }\n\n async renderCaptcha() {\n return new Promise((resolve, reject) => {\n let loadedCount = 0\n const onLoad = () => ++loadedCount === 2 && (this.hideLoading(), resolve())\n const onError = () => reject(new Error(\"图片加载失败\"))\n\n this.loadImage(this.elements.backgroundImg, this.captchaData.canvasSrc, {\n width: this.captchaData.canvasWidth,\n height: this.captchaData.canvasHeight\n }, onLoad, onError)\n\n this.loadImage(this.elements.sliderImg, this.captchaData.blockSrc, {\n width: this.captchaData.blockWidth,\n height: this.captchaData.blockHeight,\n top: this.captchaData.blockY\n }, onLoad, onError)\n })\n }\n\n loadImage(imgElement, src, styles, onLoad, onError) {\n // 检查缓存\n if (this.imageCache.has(src)) {\n const cachedImg = this.imageCache.get(src)\n imgElement.src = cachedImg.src\n this.applyStyles(imgElement, styles)\n onLoad()\n return\n }\n\n imgElement.onload = () => {\n this.imageCache.set(src, imgElement.cloneNode())\n onLoad()\n }\n imgElement.onerror = onError\n imgElement.src = src\n this.applyStyles(imgElement, styles)\n }\n\n // 优化:提取样式应用逻辑\n applyStyles(element, styles) {\n Object.entries(styles).forEach(([key, value]) => {\n element.style[key] = typeof value === \"number\" ? value + \"px\" : value\n })\n }\n\n // 优化:简化状态显示方法\n showLoading() {\n this.state.isLoading = true\n this.batchUpdateStyles({\n container: { display: \"block\" },\n loadingText: { display: \"flex\" },\n error: { display: \"none\" }\n })\n this.updateUIState(\"loading\")\n }\n\n hideLoading() {\n this.state.isLoading = false\n this.batchUpdateStyles({ loadingText: { display: \"none\" } })\n this.updateUIState(\"reset\")\n }\n\n showCaptcha() {\n this.batchUpdateStyles({\n container: { display: \"block\" },\n track: { display: \"block\" },\n error: { display: \"none\" }\n })\n }\n\n showError(message) {\n this.hideLoading()\n this.batchUpdateStyles({\n error: { display: \"block\", textContent: message }\n })\n }\n\n // 优化:批量样式更新\n batchUpdateStyles(updates) {\n requestAnimationFrame(() => {\n Object.entries(updates).forEach(([elementKey, styles]) => {\n const element = this.elements[elementKey]\n if (element) {\n Object.entries(styles).forEach(([prop, value]) => {\n if (prop === 'textContent') {\n element.textContent = value\n } else {\n element.style[prop] = value\n }\n })\n }\n })\n })\n }\n\n async verify() {\n if (!this.captchaData) {\n this.onVerifyFail(\"验证码数据丢失\")\n return\n }\n\n try {\n const controller = new AbortController()\n const timeoutId = this.safeSetTimeout(() => controller.abort(), this.options.timeout)\n\n const response = await fetch(this.options.verifyUrl, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({\n loginVo: {\n nonceStr: this.captchaData.nonceStr,\n value: this.getPosition()\n },\n dragEventList: [...this.times]\n }),\n signal: controller.signal\n })\n\n this.safeClearTimeout(timeoutId)\n\n if (!response.ok) throw new Error(`HTTP ${response.status}`)\n const data = await response.json()\n\n if (data.code === \"0\" || data.success === true) {\n this.onVerifySuccess()\n } else {\n this.onVerifyFail(data.message || \"验证失败,请重试!\")\n }\n } catch (error) {\n const message = error.name === 'AbortError' ? \"验证超时,请重试\" : \"网络错误\"\n this.onVerifyFail(message)\n }\n }\n\n onVerifySuccess() {\n const duration = Date.now() - this.startTime\n const durationText = `验证成功!耗时:${(duration / 1000).toFixed(2)}s`\n\n this.updateUIState(\"success\")\n this.showFloatingTime(durationText, \"success\")\n\n this.safeSetTimeout(() => {\n this.options.onSuccess?.({\n captchaId: this.captchaData.captchaId,\n timestamp: Date.now(),\n duration\n })\n this.hide()\n }, 2000)\n }\n\n showFloatingTime(text, type = \"success\") {\n const { elements } = this\n elements.floatingTime.textContent = text\n elements.floatingTime.className = `slider-captcha-floating-time ${type}`\n\n this.safeSetTimeout(() => elements.floatingTime.classList.add(\"show\"), 100)\n this.safeSetTimeout(() => {\n elements.floatingTime.className = \"slider-captcha-floating-time\"\n }, 2500) // 优化:延长显示时间,避免被reset清除\n }\n\n onVerifyFail(message) {\n this.state.retryCount++\n this.updateUIState(\"fail\")\n this.showFloatingTime(message, \"fail\")\n\n this.safeSetTimeout(() => {\n if (this.state.retryCount >= this.options.maxRetries) {\n this.refresh()\n } else {\n this.reset()\n }\n }, 2500) // 优化:延长等待时间,确保浮动提示完整显示\n }\n\n reset() {\n this.clearAllTimers()\n\n // 重置状态\n Object.assign(this.state, {\n isDragging: false,\n currentX: 0,\n startX: 0,\n isLoading: false\n })\n\n this.times = []\n this.startTime = null\n this.cachedDimensions = null\n\n // 重置UI\n requestAnimationFrame(() => {\n this.setTransition(true)\n this.elements.btn.style.transform = \"translateX(0px)\"\n this.elements.sliderImg.style.transform = \"translateX(0px)\"\n this.updateUIState(\"reset\")\n this.elements.error.style.display = \"none\"\n })\n }\n\n refresh() {\n this.reset()\n this.state.retryCount = 0\n this.loadCaptcha()\n }\n\n // 安全的定时器管理\n safeSetTimeout(callback, delay) {\n const timerId = setTimeout(() => {\n this.timers.delete(timerId)\n callback()\n }, delay)\n this.timers.add(timerId)\n return timerId\n }\n\n safeClearTimeout(timerId) {\n if (timerId) {\n clearTimeout(timerId)\n this.timers.delete(timerId)\n }\n }\n\n clearAllTimers() {\n this.timers.forEach(timer => {\n clearTimeout(timer)\n clearInterval(timer)\n })\n this.timers.clear()\n\n if (this.rafId) {\n cancelAnimationFrame(this.rafId)\n this.rafId = null\n }\n }\n\n // 清理图片资源\n cleanupImages() {\n if (this.elements.backgroundImg) {\n this.elements.backgroundImg.src = ''\n this.elements.backgroundImg.onload = null\n this.elements.backgroundImg.onerror = null\n }\n if (this.elements.sliderImg) {\n this.elements.sliderImg.src = ''\n this.elements.sliderImg.onload = null\n this.elements.sliderImg.onerror = null\n }\n this.imageCache.clear()\n }\n\n // 工具函数:节流\n throttle(func, delay) {\n let lastCall = 0\n return function (...args) {\n const now = Date.now()\n if (now - lastCall >= delay) {\n lastCall = now\n return func.apply(this, args)\n }\n }\n }\n\n // 工具函数:防抖\n debounce(func, delay) {\n let timeoutId\n return function (...args) {\n clearTimeout(timeoutId)\n timeoutId = setTimeout(() => func.apply(this, args), delay)\n }\n }\n\n destroy() {\n // 恢复body样式\n document.body.style.userSelect = \"\"\n document.body.style.cursor = \"\"\n\n // 清理所有定时器\n this.clearAllTimers()\n\n // 移除所有事件监听器\n this.removeAllEventListeners()\n\n // 清理图片资源\n this.cleanupImages()\n\n // 移除DOM元素\n if (this.elements.overlay?.parentNode) {\n this.elements.overlay.parentNode.removeChild(this.elements.overlay)\n }\n\n // 清理样式表\n const styleElement = document.getElementById('slider-captcha-styles')\n if (styleElement) {\n styleElement.remove()\n }\n\n // 清空所有属性\n Object.keys(this).forEach(key => {\n if (key !== 'constructor') {\n this[key] = null\n }\n })\n }\n\n static create(options) {\n return new PopupSliderCaptcha(options)\n }\n\n static show(options) {\n const instance = new PopupSliderCaptcha(options)\n instance.show()\n return instance\n }\n}\n\n// 模块导出\nif (typeof module !== \"undefined\" && module.exports) {\n module.exports = PopupSliderCaptcha\n module.exports.default = PopupSliderCaptcha\n} else if (typeof define === \"function\" && define.amd) {\n define([], () => PopupSliderCaptcha)\n} else if (typeof window !== \"undefined\") {\n window.PopupSliderCaptcha = PopupSliderCaptcha\n window.SliderCaptcha = PopupSliderCaptcha\n}\n\n// Add ES6 export for modern module systems\nexport default PopupSliderCaptcha\n","// 导入滑块验证码组件(使用默认导入)\nimport PopupSliderCaptcha from './slider-captcha.js'\n\n// 导入密码校验器\nimport PasswordValidator, { createPasswordValidator, validatePassword } from './password-validator.js'\n\n// 导出所有模块\nexport { \n PopupSliderCaptcha,\n PasswordValidator,\n createPasswordValidator,\n validatePassword\n}\n\n// 默认导出(向后兼容)\nexport default {\n PopupSliderCaptcha,\n PasswordValidator,\n createPasswordValidator,\n validatePassword\n}\n\n// 全局注册(用于UMD构建)\nif (typeof window !== 'undefined') {\n window.SliderCaptcha = PopupSliderCaptcha\n window.PopupSliderCaptcha = PopupSliderCaptcha\n window.PasswordValidator = PasswordValidator\n window.createPasswordValidator = createPasswordValidator\n window.validatePassword = validatePassword\n}\n"],"names":["createPasswordValidator","options","PasswordValidator","async","validatePassword","password","additionalData","validator","PopupSliderCaptcha","static","width","height","sliderSize","maxRetries","timeout","apiUrl","verifyUrl","throttleDelay","clickMaskClose","overlay","modal","header","container","track","btn","hint","loading","error","getStyles","constructor","this","DEFAULTS","elements","state","createInitialState","captchaData","times","startTime","eventListeners","timers","Set","rafId","cachedDimensions","imageCache","Map","throttledHandleMove","throttle","e","handleMove","init","isVisible","isDragging","currentX","startX","retryCount","isLoading","injectStyles","createElements","bindEvents","document","querySelector","style","createElement","id","textContent","head","appendChild","CSS_CLASSES","forEach","key","tag","className","cssText","refreshBtn","innerHTML","assembleDOM","setInitialState","element","headerButtons","append","closeBtn","title","backgroundImg","sliderImg","loadingText","floatingTime","icon","fingerAnimation","body","Object","assign","display","addEventListener","hide","refresh","target","handleVisibilityChange","bindSliderEvents","handlers","start","handleStart","move","end","handleEnd","passive","event","handler","push","removeAllEventListeners","removeEventListener","length","getDimensions","trackWidth","offsetWidth","btnWidth","maxX","getPosition","percentage","Math","round","preventDefault","getClientX","Date","now","time","position","setTransition","updateUIState","deltaX","max","min","cancelAnimationFrame","requestAnimationFrame","updateSliderPosition","verify","animationState","hidden","animationPlayState","type","includes","touches","clientX","enabled","transition","updates","dragging","opacity","success","background","color","fail","reset","updateHintText","pointerEvents","text","pieceX","progress","transform","show","offsetHeight","classList","add","loadCaptcha","remove","safeSetTimeout","onClose","showLoading","controller","AbortController","timeoutId","abort","response","fetch","method","headers","JSON","stringify","place","timestamp","signal","safeClearTimeout","ok","Error","status","data","json","canvasSrc","blockSrc","showCaptcha","renderCaptcha","message","name","showError","Promise","resolve","reject","loadedCount","onLoad","hideLoading","onError","loadImage","canvasWidth","canvasHeight","blockWidth","blockHeight","top","blockY","imgElement","src","styles","has","cachedImg","get","applyStyles","onload","set","cloneNode","onerror","entries","value","batchUpdateStyles","elementKey","prop","loginVo","nonceStr","dragEventList","code","onVerifySuccess","onVerifyFail","duration","durationText","toFixed","showFloatingTime","onSuccess","captchaId","clearAllTimers","callback","delay","timerId","setTimeout","delete","clearTimeout","timer","clearInterval","clear","cleanupImages","func","lastCall","args","apply","debounce","destroy","userSelect","cursor","parentNode","removeChild","styleElement","getElementById","keys","create","instance","module","exports","default","define","amd","window","SliderCaptcha","publicKeyUrl","validateUrl","publicKeyCache","publicKeyExpiry","getPublicKey","makeRequest","encryptPassword","publicKey","JSEncrypt","encrypt","setPublicKey","encrypted","encryptedPassword","userName","url","statusText","clearCache","updateOptions","newOptions","index"],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
"use strict";Object.defineProperty(exports,"__esModule",{value:!0});class PasswordValidator{constructor(e={}){this.options={publicKeyUrl:e.publicKeyUrl||"/microservice/strongPassword/getPublicKey",validateUrl:e.validateUrl||"/microservice/strongPassword/checkPassword",timeout:e.timeout||1e4,headers:e.headers||{},...e},this.publicKeyCache=null,this.publicKeyExpiry=null}async getPublicKey(){if(this.publicKeyCache&&this.publicKeyExpiry&&Date.now()<this.publicKeyExpiry)return this.publicKeyCache;try{const e=await this.makeRequest(this.options.publicKeyUrl,{method:"GET"});if("0"!==e.code)throw new Error("获取公钥失败:"+(e.message||"未知错误"));return this.publicKeyCache=e.data,this.publicKeyExpiry=Date.now()+3e5,this.publicKeyCache}catch(e){throw new Error("获取公钥失败: "+e.message)}}encryptPassword(e,t){try{if("undefined"==typeof JSEncrypt)throw new Error("JSEncrypt库未加载,请先引入jsencrypt库");const s=new JSEncrypt;s.setPublicKey(t);const r=s.encrypt(e);if(!r)throw new Error("密码加密失败");return r}catch(e){throw new Error("密码加密失败: "+e.message)}}async validatePassword(e,t={}){try{const s=await this.getPublicKey(),r=this.encryptPassword(e,s),a=await this.makeRequest(this.options.validateUrl,{method:"POST",body:JSON.stringify({userName:"13212345678",password:r,timestamp:Date.now(),...t})});return{success:a.success,message:a.message,data:a.data,code:a.code}}catch(e){return{success:!1,message:e.message,code:"VALIDATION_ERROR"}}}async makeRequest(e,t={}){const s=new AbortController,r=setTimeout(()=>s.abort(),this.options.timeout);try{const a=await fetch(e,{...t,headers:{"Content-Type":"application/json",...this.options.headers,...t.headers},signal:s.signal});if(clearTimeout(r),!a.ok)throw new Error(`HTTP错误: ${a.status} ${a.statusText}`);return await a.json()}catch(e){if(clearTimeout(r),"AbortError"===e.name)throw new Error("请求超时");throw e}}clearCache(){this.publicKeyCache=null,this.publicKeyExpiry=null}updateOptions(e){this.options={...this.options,...e},this.clearCache()}}exports.createPasswordValidator=function(e){return new PasswordValidator(e)},exports.default=PasswordValidator,exports.validatePassword=async function(e,t={},s={}){const r=new PasswordValidator(t);return await r.validatePassword(e,s)};
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"password-validator.cjs.js","sources":["../src/password-validator.js"],"sourcesContent":["// import JSEncrypt from 'jsencrypt'\n\n/**\n * 密码校验工具类\n * 提供密码加密和校验功能\n */\nclass PasswordValidator {\n constructor(options = {}) {\n this.options = {\n // 获取公钥的接口地址\n publicKeyUrl: options.publicKeyUrl || '/microservice/strongPassword/getPublicKey',\n // 密码校验接口地址\n validateUrl: options.validateUrl || '/microservice/strongPassword/checkPassword',\n // 请求超时时间\n timeout: options.timeout || 10000,\n // 自定义请求头\n headers: options.headers || {},\n ...options\n }\n\n // 缓存公钥,避免重复请求\n this.publicKeyCache = null\n this.publicKeyExpiry = null\n }\n\n /**\n * 获取公钥\n * @returns {Promise<string>} 公钥字符串\n */\n async getPublicKey() {\n // 检查缓存是否有效(5分钟有效期)\n if (this.publicKeyCache && this.publicKeyExpiry && Date.now() < this.publicKeyExpiry) {\n return this.publicKeyCache\n }\n\n try {\n const response = await this.makeRequest(this.options.publicKeyUrl, {\n method: 'GET'\n })\n\n if (response.code !== '0' ) {\n throw new Error('获取公钥失败:' + (response.message || '未知错误'))\n }\n\n // 缓存公钥,设置5分钟过期时间\n this.publicKeyCache = response.data\n this.publicKeyExpiry = Date.now() + 5 * 60 * 1000\n\n return this.publicKeyCache\n } catch (error) {\n console.error('获取公钥失败:', error)\n throw new Error('获取公钥失败: ' + error.message)\n }\n }\n\n /**\n * 使用RSA公钥加密密码\n * @param {string} password 原始密码\n * @param {string} publicKey 公钥字符串\n * @returns {string} 加密后的密码\n */\n encryptPassword(password, publicKey) {\n try {\n // 这里使用JSEncrypt库进行RSA加密\n // 需要先安装: npm install jsencrypt\n if (typeof JSEncrypt === 'undefined') {\n throw new Error('JSEncrypt库未加载,请先引入jsencrypt库')\n }\n\n const encrypt = new JSEncrypt()\n encrypt.setPublicKey(publicKey)\n const encrypted = encrypt.encrypt(password)\n\n if (!encrypted) {\n throw new Error('密码加密失败')\n }\n\n return encrypted\n } catch (error) {\n console.error('密码加密失败:', error)\n throw new Error('密码加密失败: ' + error.message)\n }\n }\n\n /**\n * 校验密码\n * @param {string} password 原始密码\n * @param {Object} additionalData 额外的校验数据\n * @returns {Promise<Object>} 校验结果\n */\n async validatePassword(password, additionalData = {}) {\n try {\n // 1. 获取公钥\n const publicKey = await this.getPublicKey()\n\n // 2. 加密密码\n const encryptedPassword = this.encryptPassword(password, publicKey)\n\n // 3. 调用校验接口\n const response = await this.makeRequest(this.options.validateUrl, {\n method: 'POST',\n body: JSON.stringify({\n userName: '13212345678',\n password: encryptedPassword,\n timestamp: Date.now(),\n ...additionalData\n })\n })\n\n return {\n success: response.success,\n message: response.message,\n data: response.data,\n code: response.code\n }\n } catch (error) {\n console.error('密码校验失败:', error)\n return {\n success: false,\n message: error.message,\n code: 'VALIDATION_ERROR'\n }\n }\n }\n\n /**\n * 发送HTTP请求的通用方法\n * @param {string} url 请求地址\n * @param {Object} options 请求选项\n * @returns {Promise<Object>} 响应结果\n */\n async makeRequest(url, options = {}) {\n const controller = new AbortController()\n const timeoutId = setTimeout(() => controller.abort(), this.options.timeout)\n\n try {\n const response = await fetch(url, {\n ...options,\n headers: {\n 'Content-Type': 'application/json',\n ...this.options.headers,\n ...options.headers\n },\n signal: controller.signal\n })\n\n clearTimeout(timeoutId)\n\n if (!response.ok) {\n throw new Error(`HTTP错误: ${response.status} ${response.statusText}`)\n }\n\n return await response.json()\n } catch (error) {\n clearTimeout(timeoutId)\n\n if (error.name === 'AbortError') {\n throw new Error('请求超时')\n }\n\n throw error\n }\n }\n\n /**\n * 清除公钥缓存\n */\n clearCache() {\n this.publicKeyCache = null\n this.publicKeyExpiry = null\n }\n\n /**\n * 更新配置\n * @param {Object} newOptions 新的配置选项\n */\n updateOptions(newOptions) {\n this.options = { ...this.options, ...newOptions }\n // 清除缓存,使用新配置重新获取\n this.clearCache()\n }\n}\n\n// 导出类和创建实例的工厂函数\nexport default PasswordValidator\n\n/**\n * 创建密码校验器实例的工厂函数\n * @param {Object} options 配置选项\n * @returns {PasswordValidator} 密码校验器实例\n */\nexport function createPasswordValidator(options) {\n return new PasswordValidator(options)\n}\n\n/**\n * 快速校验密码的便捷函数\n * @param {string} password 密码\n * @param {Object} options 配置选项\n * @param {Object} additionalData 额外数据\n * @returns {Promise<Object>} 校验结果\n */\nexport async function validatePassword(password, options = {}, additionalData = {}) {\n const validator = new PasswordValidator(options)\n return await validator.validatePassword(password, additionalData)\n}\n"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
class PasswordValidator{constructor(e={}){this.options={publicKeyUrl:e.publicKeyUrl||"/microservice/strongPassword/getPublicKey",validateUrl:e.validateUrl||"/microservice/strongPassword/checkPassword",timeout:e.timeout||1e4,headers:e.headers||{},...e},this.publicKeyCache=null,this.publicKeyExpiry=null}async getPublicKey(){if(this.publicKeyCache&&this.publicKeyExpiry&&Date.now()<this.publicKeyExpiry)return this.publicKeyCache;try{const e=await this.makeRequest(this.options.publicKeyUrl,{method:"GET"});if("0"!==e.code)throw new Error("获取公钥失败:"+(e.message||"未知错误"));return this.publicKeyCache=e.data,this.publicKeyExpiry=Date.now()+3e5,this.publicKeyCache}catch(e){throw new Error("获取公钥失败: "+e.message)}}encryptPassword(e,t){try{if("undefined"==typeof JSEncrypt)throw new Error("JSEncrypt库未加载,请先引入jsencrypt库");const s=new JSEncrypt;s.setPublicKey(t);const r=s.encrypt(e);if(!r)throw new Error("密码加密失败");return r}catch(e){throw new Error("密码加密失败: "+e.message)}}async validatePassword(e,t={}){try{const s=await this.getPublicKey(),r=this.encryptPassword(e,s),i=await this.makeRequest(this.options.validateUrl,{method:"POST",body:JSON.stringify({userName:"13212345678",password:r,timestamp:Date.now(),...t})});return{success:i.success,message:i.message,data:i.data,code:i.code}}catch(e){return{success:!1,message:e.message,code:"VALIDATION_ERROR"}}}async makeRequest(e,t={}){const s=new AbortController,r=setTimeout(()=>s.abort(),this.options.timeout);try{const i=await fetch(e,{...t,headers:{"Content-Type":"application/json",...this.options.headers,...t.headers},signal:s.signal});if(clearTimeout(r),!i.ok)throw new Error(`HTTP错误: ${i.status} ${i.statusText}`);return await i.json()}catch(e){if(clearTimeout(r),"AbortError"===e.name)throw new Error("请求超时");throw e}}clearCache(){this.publicKeyCache=null,this.publicKeyExpiry=null}updateOptions(e){this.options={...this.options,...e},this.clearCache()}}function e(e){return new PasswordValidator(e)}async function t(e,t={},s={}){const r=new PasswordValidator(t);return await r.validatePassword(e,s)}export{e as createPasswordValidator,PasswordValidator as default,t as validatePassword};
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"password-validator.esm.js","sources":["../src/password-validator.js"],"sourcesContent":["// import JSEncrypt from 'jsencrypt'\n\n/**\n * 密码校验工具类\n * 提供密码加密和校验功能\n */\nclass PasswordValidator {\n constructor(options = {}) {\n this.options = {\n // 获取公钥的接口地址\n publicKeyUrl: options.publicKeyUrl || '/microservice/strongPassword/getPublicKey',\n // 密码校验接口地址\n validateUrl: options.validateUrl || '/microservice/strongPassword/checkPassword',\n // 请求超时时间\n timeout: options.timeout || 10000,\n // 自定义请求头\n headers: options.headers || {},\n ...options\n }\n\n // 缓存公钥,避免重复请求\n this.publicKeyCache = null\n this.publicKeyExpiry = null\n }\n\n /**\n * 获取公钥\n * @returns {Promise<string>} 公钥字符串\n */\n async getPublicKey() {\n // 检查缓存是否有效(5分钟有效期)\n if (this.publicKeyCache && this.publicKeyExpiry && Date.now() < this.publicKeyExpiry) {\n return this.publicKeyCache\n }\n\n try {\n const response = await this.makeRequest(this.options.publicKeyUrl, {\n method: 'GET'\n })\n\n if (response.code !== '0' ) {\n throw new Error('获取公钥失败:' + (response.message || '未知错误'))\n }\n\n // 缓存公钥,设置5分钟过期时间\n this.publicKeyCache = response.data\n this.publicKeyExpiry = Date.now() + 5 * 60 * 1000\n\n return this.publicKeyCache\n } catch (error) {\n console.error('获取公钥失败:', error)\n throw new Error('获取公钥失败: ' + error.message)\n }\n }\n\n /**\n * 使用RSA公钥加密密码\n * @param {string} password 原始密码\n * @param {string} publicKey 公钥字符串\n * @returns {string} 加密后的密码\n */\n encryptPassword(password, publicKey) {\n try {\n // 这里使用JSEncrypt库进行RSA加密\n // 需要先安装: npm install jsencrypt\n if (typeof JSEncrypt === 'undefined') {\n throw new Error('JSEncrypt库未加载,请先引入jsencrypt库')\n }\n\n const encrypt = new JSEncrypt()\n encrypt.setPublicKey(publicKey)\n const encrypted = encrypt.encrypt(password)\n\n if (!encrypted) {\n throw new Error('密码加密失败')\n }\n\n return encrypted\n } catch (error) {\n console.error('密码加密失败:', error)\n throw new Error('密码加密失败: ' + error.message)\n }\n }\n\n /**\n * 校验密码\n * @param {string} password 原始密码\n * @param {Object} additionalData 额外的校验数据\n * @returns {Promise<Object>} 校验结果\n */\n async validatePassword(password, additionalData = {}) {\n try {\n // 1. 获取公钥\n const publicKey = await this.getPublicKey()\n\n // 2. 加密密码\n const encryptedPassword = this.encryptPassword(password, publicKey)\n\n // 3. 调用校验接口\n const response = await this.makeRequest(this.options.validateUrl, {\n method: 'POST',\n body: JSON.stringify({\n userName: '13212345678',\n password: encryptedPassword,\n timestamp: Date.now(),\n ...additionalData\n })\n })\n\n return {\n success: response.success,\n message: response.message,\n data: response.data,\n code: response.code\n }\n } catch (error) {\n console.error('密码校验失败:', error)\n return {\n success: false,\n message: error.message,\n code: 'VALIDATION_ERROR'\n }\n }\n }\n\n /**\n * 发送HTTP请求的通用方法\n * @param {string} url 请求地址\n * @param {Object} options 请求选项\n * @returns {Promise<Object>} 响应结果\n */\n async makeRequest(url, options = {}) {\n const controller = new AbortController()\n const timeoutId = setTimeout(() => controller.abort(), this.options.timeout)\n\n try {\n const response = await fetch(url, {\n ...options,\n headers: {\n 'Content-Type': 'application/json',\n ...this.options.headers,\n ...options.headers\n },\n signal: controller.signal\n })\n\n clearTimeout(timeoutId)\n\n if (!response.ok) {\n throw new Error(`HTTP错误: ${response.status} ${response.statusText}`)\n }\n\n return await response.json()\n } catch (error) {\n clearTimeout(timeoutId)\n\n if (error.name === 'AbortError') {\n throw new Error('请求超时')\n }\n\n throw error\n }\n }\n\n /**\n * 清除公钥缓存\n */\n clearCache() {\n this.publicKeyCache = null\n this.publicKeyExpiry = null\n }\n\n /**\n * 更新配置\n * @param {Object} newOptions 新的配置选项\n */\n updateOptions(newOptions) {\n this.options = { ...this.options, ...newOptions }\n // 清除缓存,使用新配置重新获取\n this.clearCache()\n }\n}\n\n// 导出类和创建实例的工厂函数\nexport default PasswordValidator\n\n/**\n * 创建密码校验器实例的工厂函数\n * @param {Object} options 配置选项\n * @returns {PasswordValidator} 密码校验器实例\n */\nexport function createPasswordValidator(options) {\n return new PasswordValidator(options)\n}\n\n/**\n * 快速校验密码的便捷函数\n * @param {string} password 密码\n * @param {Object} options 配置选项\n * @param {Object} additionalData 额外数据\n * @returns {Promise<Object>} 校验结果\n */\nexport async function validatePassword(password, options = {}, additionalData = {}) {\n const validator = new PasswordValidator(options)\n return await validator.validatePassword(password, additionalData)\n}\n"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
!function(e,t){"object"==typeof exports&&"undefined"!=typeof module?t(exports):"function"==typeof define&&define.amd?define(["exports"],t):t((e="undefined"!=typeof globalThis?globalThis:e||self).PasswordValidator={})}(this,function(e){"use strict";class PasswordValidator{constructor(e={}){this.options={publicKeyUrl:e.publicKeyUrl||"/microservice/strongPassword/getPublicKey",validateUrl:e.validateUrl||"/microservice/strongPassword/checkPassword",timeout:e.timeout||1e4,headers:e.headers||{},...e},this.publicKeyCache=null,this.publicKeyExpiry=null}async getPublicKey(){if(this.publicKeyCache&&this.publicKeyExpiry&&Date.now()<this.publicKeyExpiry)return this.publicKeyCache;try{const e=await this.makeRequest(this.options.publicKeyUrl,{method:"GET"});if("0"!==e.code)throw new Error("获取公钥失败:"+(e.message||"未知错误"));return this.publicKeyCache=e.data,this.publicKeyExpiry=Date.now()+3e5,this.publicKeyCache}catch(e){throw new Error("获取公钥失败: "+e.message)}}encryptPassword(e,t){try{if("undefined"==typeof JSEncrypt)throw new Error("JSEncrypt库未加载,请先引入jsencrypt库");const s=new JSEncrypt;s.setPublicKey(t);const r=s.encrypt(e);if(!r)throw new Error("密码加密失败");return r}catch(e){throw new Error("密码加密失败: "+e.message)}}async validatePassword(e,t={}){try{const s=await this.getPublicKey(),r=this.encryptPassword(e,s),i=await this.makeRequest(this.options.validateUrl,{method:"POST",body:JSON.stringify({userName:"13212345678",password:r,timestamp:Date.now(),...t})});return{success:i.success,message:i.message,data:i.data,code:i.code}}catch(e){return{success:!1,message:e.message,code:"VALIDATION_ERROR"}}}async makeRequest(e,t={}){const s=new AbortController,r=setTimeout(()=>s.abort(),this.options.timeout);try{const i=await fetch(e,{...t,headers:{"Content-Type":"application/json",...this.options.headers,...t.headers},signal:s.signal});if(clearTimeout(r),!i.ok)throw new Error(`HTTP错误: ${i.status} ${i.statusText}`);return await i.json()}catch(e){if(clearTimeout(r),"AbortError"===e.name)throw new Error("请求超时");throw e}}clearCache(){this.publicKeyCache=null,this.publicKeyExpiry=null}updateOptions(e){this.options={...this.options,...e},this.clearCache()}}e.createPasswordValidator=function(e){return new PasswordValidator(e)},e.default=PasswordValidator,e.validatePassword=async function(e,t={},s={}){const r=new PasswordValidator(t);return await r.validatePassword(e,s)},Object.defineProperty(e,"__esModule",{value:!0})});
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"password-validator.js","sources":["../src/password-validator.js"],"sourcesContent":["// import JSEncrypt from 'jsencrypt'\n\n/**\n * 密码校验工具类\n * 提供密码加密和校验功能\n */\nclass PasswordValidator {\n constructor(options = {}) {\n this.options = {\n // 获取公钥的接口地址\n publicKeyUrl: options.publicKeyUrl || '/microservice/strongPassword/getPublicKey',\n // 密码校验接口地址\n validateUrl: options.validateUrl || '/microservice/strongPassword/checkPassword',\n // 请求超时时间\n timeout: options.timeout || 10000,\n // 自定义请求头\n headers: options.headers || {},\n ...options\n }\n\n // 缓存公钥,避免重复请求\n this.publicKeyCache = null\n this.publicKeyExpiry = null\n }\n\n /**\n * 获取公钥\n * @returns {Promise<string>} 公钥字符串\n */\n async getPublicKey() {\n // 检查缓存是否有效(5分钟有效期)\n if (this.publicKeyCache && this.publicKeyExpiry && Date.now() < this.publicKeyExpiry) {\n return this.publicKeyCache\n }\n\n try {\n const response = await this.makeRequest(this.options.publicKeyUrl, {\n method: 'GET'\n })\n\n if (response.code !== '0' ) {\n throw new Error('获取公钥失败:' + (response.message || '未知错误'))\n }\n\n // 缓存公钥,设置5分钟过期时间\n this.publicKeyCache = response.data\n this.publicKeyExpiry = Date.now() + 5 * 60 * 1000\n\n return this.publicKeyCache\n } catch (error) {\n console.error('获取公钥失败:', error)\n throw new Error('获取公钥失败: ' + error.message)\n }\n }\n\n /**\n * 使用RSA公钥加密密码\n * @param {string} password 原始密码\n * @param {string} publicKey 公钥字符串\n * @returns {string} 加密后的密码\n */\n encryptPassword(password, publicKey) {\n try {\n // 这里使用JSEncrypt库进行RSA加密\n // 需要先安装: npm install jsencrypt\n if (typeof JSEncrypt === 'undefined') {\n throw new Error('JSEncrypt库未加载,请先引入jsencrypt库')\n }\n\n const encrypt = new JSEncrypt()\n encrypt.setPublicKey(publicKey)\n const encrypted = encrypt.encrypt(password)\n\n if (!encrypted) {\n throw new Error('密码加密失败')\n }\n\n return encrypted\n } catch (error) {\n console.error('密码加密失败:', error)\n throw new Error('密码加密失败: ' + error.message)\n }\n }\n\n /**\n * 校验密码\n * @param {string} password 原始密码\n * @param {Object} additionalData 额外的校验数据\n * @returns {Promise<Object>} 校验结果\n */\n async validatePassword(password, additionalData = {}) {\n try {\n // 1. 获取公钥\n const publicKey = await this.getPublicKey()\n\n // 2. 加密密码\n const encryptedPassword = this.encryptPassword(password, publicKey)\n\n // 3. 调用校验接口\n const response = await this.makeRequest(this.options.validateUrl, {\n method: 'POST',\n body: JSON.stringify({\n userName: '13212345678',\n password: encryptedPassword,\n timestamp: Date.now(),\n ...additionalData\n })\n })\n\n return {\n success: response.success,\n message: response.message,\n data: response.data,\n code: response.code\n }\n } catch (error) {\n console.error('密码校验失败:', error)\n return {\n success: false,\n message: error.message,\n code: 'VALIDATION_ERROR'\n }\n }\n }\n\n /**\n * 发送HTTP请求的通用方法\n * @param {string} url 请求地址\n * @param {Object} options 请求选项\n * @returns {Promise<Object>} 响应结果\n */\n async makeRequest(url, options = {}) {\n const controller = new AbortController()\n const timeoutId = setTimeout(() => controller.abort(), this.options.timeout)\n\n try {\n const response = await fetch(url, {\n ...options,\n headers: {\n 'Content-Type': 'application/json',\n ...this.options.headers,\n ...options.headers\n },\n signal: controller.signal\n })\n\n clearTimeout(timeoutId)\n\n if (!response.ok) {\n throw new Error(`HTTP错误: ${response.status} ${response.statusText}`)\n }\n\n return await response.json()\n } catch (error) {\n clearTimeout(timeoutId)\n\n if (error.name === 'AbortError') {\n throw new Error('请求超时')\n }\n\n throw error\n }\n }\n\n /**\n * 清除公钥缓存\n */\n clearCache() {\n this.publicKeyCache = null\n this.publicKeyExpiry = null\n }\n\n /**\n * 更新配置\n * @param {Object} newOptions 新的配置选项\n */\n updateOptions(newOptions) {\n this.options = { ...this.options, ...newOptions }\n // 清除缓存,使用新配置重新获取\n this.clearCache()\n }\n}\n\n// 导出类和创建实例的工厂函数\nexport default PasswordValidator\n\n/**\n * 创建密码校验器实例的工厂函数\n * @param {Object} options 配置选项\n * @returns {PasswordValidator} 密码校验器实例\n */\nexport function createPasswordValidator(options) {\n return new PasswordValidator(options)\n}\n\n/**\n * 快速校验密码的便捷函数\n * @param {string} password 密码\n * @param {Object} options 配置选项\n * @param {Object} additionalData 额外数据\n * @returns {Promise<Object>} 校验结果\n */\nexport async function validatePassword(password, options = {}, additionalData = {}) {\n const validator = new PasswordValidator(options)\n return await validator.validatePassword(password, additionalData)\n}\n"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
(function(t,e){"object"==typeof exports&&"undefined"!=typeof module?e(exports):"function"==typeof define&&define.amd?define(["exports"],e):e((t="undefined"!=typeof globalThis?globalThis:t||self).PasswordValidator={})})(this,(function(t){"use strict"
|
|
2
|
+
class PasswordValidator{constructor(t={}){this.options={publicKeyUrl:t.publicKeyUrl||"/microservice/strongPassword/getPublicKey",validateUrl:t.validateUrl||"/microservice/strongPassword/checkPassword",timeout:t.timeout||1e4,headers:t.headers||{},...t},this.publicKeyCache=null,this.publicKeyExpiry=null}async getPublicKey(){if(this.publicKeyCache&&this.publicKeyExpiry&&Date.now()<this.publicKeyExpiry)return this.publicKeyCache
|
|
3
|
+
try{const t=await this.makeRequest(this.options.publicKeyUrl,{method:"GET"})
|
|
4
|
+
if("0"!==t.code)throw Error("获取公钥失败:"+(t.message||"未知错误"))
|
|
5
|
+
return this.publicKeyCache=t.data,this.publicKeyExpiry=Date.now()+3e5,this.publicKeyCache}catch(t){throw Error("获取公钥失败: "+t.message)}}encryptPassword(t,e){try{if("undefined"==typeof JSEncrypt)throw Error("JSEncrypt库未加载,请先引入jsencrypt库")
|
|
6
|
+
const r=new JSEncrypt
|
|
7
|
+
r.setPublicKey(e)
|
|
8
|
+
const s=r.encrypt(t)
|
|
9
|
+
if(!s)throw Error("密码加密失败")
|
|
10
|
+
return s}catch(t){throw Error("密码加密失败: "+t.message)}}async validatePassword(t,e={}){try{const r=await this.getPublicKey(),s=this.encryptPassword(t,r),o=await this.makeRequest(this.options.validateUrl,{method:"POST",body:JSON.stringify({userName:"13212345678",password:s,timestamp:Date.now(),...e})})
|
|
11
|
+
return{success:o.success,message:o.message,data:o.data,code:o.code}}catch(t){return{success:!1,message:t.message,code:"VALIDATION_ERROR"}}}async makeRequest(t,e={}){const r=new AbortController,s=setTimeout((()=>r.abort()),this.options.timeout)
|
|
12
|
+
try{const o=await fetch(t,{...e,headers:{"Content-Type":"application/json",...this.options.headers,...e.headers},signal:r.signal})
|
|
13
|
+
if(clearTimeout(s),!o.ok)throw Error(`HTTP错误: ${o.status} ${o.statusText}`)
|
|
14
|
+
return await o.json()}catch(t){if(clearTimeout(s),"AbortError"===t.name)throw Error("请求超时")
|
|
15
|
+
throw t}}clearCache(){this.publicKeyCache=null,this.publicKeyExpiry=null}updateOptions(t){this.options={...this.options,...t},this.clearCache()}}t.createPasswordValidator=function(t){return new PasswordValidator(t)},t.default=PasswordValidator,t.validatePassword=async function(t,e={},r={}){const s=new PasswordValidator(e)
|
|
16
|
+
return await s.validatePassword(t,r)},Object.defineProperty(t,"t",{value:!0})}))
|
|
17
|
+
//# sourceMappingURL=password-validator.min.js.map
|