diff --git a/continew-admin-common/pom.xml b/continew-admin-common/pom.xml index 40bcbbbb..07ccca22 100644 --- a/continew-admin-common/pom.xml +++ b/continew-admin-common/pom.xml @@ -52,6 +52,12 @@ <artifactId>continew-starter-captcha-graphic</artifactId> </dependency> + <!-- ContiNew Starter 验证码模块 - 行为验证码 --> + <dependency> + <groupId>top.charles7c.continew</groupId> + <artifactId>continew-starter-captcha-behavior</artifactId> + </dependency> + <!-- ContiNew Starter 文件处理模块 - Excel --> <dependency> <groupId>top.charles7c.continew</groupId> diff --git a/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/enums/MessageTemplateEnum.java b/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/enums/MessageTemplateEnum.java index 156bf02f..d07a0add 100644 --- a/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/enums/MessageTemplateEnum.java +++ b/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/enums/MessageTemplateEnum.java @@ -22,7 +22,7 @@ import lombok.RequiredArgsConstructor; /** * 消息模板枚举 * - * @author BULL_BCLS + * @author Bull-BCLS * @since 2023/10/15 19:51 */ @Getter diff --git a/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/mapper/MessageMapper.java b/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/mapper/MessageMapper.java index b0700bc1..93f8668d 100644 --- a/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/mapper/MessageMapper.java +++ b/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/mapper/MessageMapper.java @@ -29,7 +29,7 @@ import top.charles7c.continew.starter.data.mybatis.plus.base.BaseMapper; /** * 消息 Mapper * - * @author BULL_BCLS + * @author Bull-BCLS * @since 2023/10/15 19:05 */ public interface MessageMapper extends BaseMapper<MessageDO> { diff --git a/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/mapper/MessageUserMapper.java b/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/mapper/MessageUserMapper.java index ee1b34c2..c9ee3041 100644 --- a/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/mapper/MessageUserMapper.java +++ b/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/mapper/MessageUserMapper.java @@ -24,7 +24,7 @@ import top.charles7c.continew.starter.data.mybatis.plus.base.BaseMapper; /** * 消息和用户 Mapper * - * @author BULL_BCLS + * @author Bull-BCLS * @since 2023/10/15 20:25 */ public interface MessageUserMapper extends BaseMapper<MessageUserDO> { diff --git a/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/model/entity/MessageDO.java b/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/model/entity/MessageDO.java index 5cac1853..e2b55a5e 100644 --- a/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/model/entity/MessageDO.java +++ b/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/model/entity/MessageDO.java @@ -32,7 +32,7 @@ import top.charles7c.continew.admin.common.enums.MessageTypeEnum; /** * 消息实体 * - * @author BULL_BCLS + * @author Bull-BCLS * @since 2023/10/15 19:05 */ @Data diff --git a/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/model/entity/MessageUserDO.java b/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/model/entity/MessageUserDO.java index 1d51a196..58b1ece5 100644 --- a/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/model/entity/MessageUserDO.java +++ b/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/model/entity/MessageUserDO.java @@ -26,7 +26,7 @@ import com.baomidou.mybatisplus.annotation.TableName; /** * 消息和用户关联实体 * - * @author BULL_BCLS + * @author Bull-BCLS * @since 2023/10/15 20:25 */ @Data diff --git a/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/model/query/MessageQuery.java b/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/model/query/MessageQuery.java index 1ddd2d2c..99857f71 100644 --- a/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/model/query/MessageQuery.java +++ b/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/model/query/MessageQuery.java @@ -28,7 +28,7 @@ import top.charles7c.continew.starter.data.mybatis.plus.query.QueryType; /** * 消息查询条件 * - * @author BULL_BCLS + * @author Bull-BCLS * @since 2023/10/15 19:05 */ @Data diff --git a/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/model/req/MessageReq.java b/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/model/req/MessageReq.java index d44d5a99..a1c90ce8 100644 --- a/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/model/req/MessageReq.java +++ b/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/model/req/MessageReq.java @@ -33,7 +33,7 @@ import top.charles7c.continew.starter.extension.crud.base.BaseReq; /** * 创建消息信息 * - * @author BULL_BCLS + * @author Bull-BCLS * @since 2023/10/15 19:05 */ @Data diff --git a/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/model/resp/MessageResp.java b/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/model/resp/MessageResp.java index b6f8c166..5c02028f 100644 --- a/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/model/resp/MessageResp.java +++ b/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/model/resp/MessageResp.java @@ -31,7 +31,7 @@ import top.charles7c.continew.admin.common.enums.MessageTypeEnum; /** * 消息信息 * - * @author BULL_BCLS + * @author Bull-BCLS * @since 2023/10/15 19:05 */ @Data diff --git a/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/service/MessageService.java b/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/service/MessageService.java index 39904d3c..2ed355d9 100644 --- a/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/service/MessageService.java +++ b/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/service/MessageService.java @@ -27,7 +27,7 @@ import top.charles7c.continew.starter.extension.crud.model.resp.PageDataResp; /** * 消息业务接口 * - * @author BULL_BCLS + * @author Bull-BCLS * @since 2023/10/15 19:05 */ public interface MessageService { diff --git a/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/service/MessageUserService.java b/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/service/MessageUserService.java index 06487b33..493a29f2 100644 --- a/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/service/MessageUserService.java +++ b/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/service/MessageUserService.java @@ -23,7 +23,7 @@ import top.charles7c.continew.admin.system.model.resp.MessageUnreadResp; /** * 消息和用户关联业务接口 * - * @author BULL_BCLS + * @author Bull-BCLS * @since 2023/10/15 19:05 */ public interface MessageUserService { diff --git a/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/service/impl/MessageServiceImpl.java b/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/service/impl/MessageServiceImpl.java index 1451479f..13830e72 100644 --- a/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/service/impl/MessageServiceImpl.java +++ b/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/service/impl/MessageServiceImpl.java @@ -47,7 +47,7 @@ import top.charles7c.continew.starter.extension.crud.model.resp.PageDataResp; /** * 消息业务实现 * - * @author BULL_BCLS + * @author Bull-BCLS * @since 2023/10/15 19:05 */ @Service diff --git a/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/service/impl/MessageUserServiceImpl.java b/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/service/impl/MessageUserServiceImpl.java index 847d3935..0b53ccd3 100644 --- a/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/service/impl/MessageUserServiceImpl.java +++ b/continew-admin-system/src/main/java/top/charles7c/continew/admin/system/service/impl/MessageUserServiceImpl.java @@ -38,7 +38,7 @@ import top.charles7c.continew.starter.core.util.validate.CheckUtils; /** * 消息和用户关联业务实现 * - * @author BULL_BCLS + * @author Bull-BCLS * @since 2023/10/15 19:05 */ @Service diff --git a/continew-admin-ui/src/api/common/captcha.ts b/continew-admin-ui/src/api/common/captcha.ts index d8efb0f5..8e426792 100644 --- a/continew-admin-ui/src/api/common/captcha.ts +++ b/continew-admin-ui/src/api/common/captcha.ts @@ -1,11 +1,35 @@ import axios from 'axios'; +import qs from 'query-string'; -const BASE_URL = '/common/captcha'; +const BASE_URL = '/captcha'; export interface ImageCaptchaRes { uuid: string; img: string; } + +export interface BehaviorCaptchaRes { + originalImageBase64: string; + point: { + x: number; + y: number; + }; + jigsawImageBase64: string; + token: string; + secretKey: string; +} + +export interface BehaviorCaptchaReq { + captchaType?: string; + captchaVerification?: string; + clientUid?: string; +} + +export interface CheckBehaviorCaptchaRes { + repCode: string; + repMsg: string; +} + export function getImageCaptcha() { return axios.get<ImageCaptchaRes>(`${BASE_URL}/img`); } @@ -14,6 +38,26 @@ export function getMailCaptcha(email: string) { return axios.get(`${BASE_URL}/mail?email=${email}`); } -export function getSmsCaptcha(phone: string) { - return axios.get(`${BASE_URL}/sms?phone=${phone}`); +export function getSmsCaptcha( + phone: string, + behaviorCaptcha: BehaviorCaptchaReq, +) { + return axios.get( + `${BASE_URL}/sms?phone=${phone}&captchaVerification=${encodeURIComponent( + behaviorCaptcha.captchaVerification || '', + )}`, + ); +} + +export function getBehaviorCaptcha(params: any) { + return axios.get<BehaviorCaptchaRes>(`${BASE_URL}/behavior`, { + params, + paramsSerializer: (obj) => { + return qs.stringify(obj); + }, + }); +} + +export function checkBehaviorCaptcha(params: any) { + return axios.post<CheckBehaviorCaptchaRes>(`${BASE_URL}/behavior`, params); } diff --git a/continew-admin-ui/src/components/index.ts b/continew-admin-ui/src/components/index.ts index 09d54c0a..b10eaeb7 100644 --- a/continew-admin-ui/src/components/index.ts +++ b/continew-admin-ui/src/components/index.ts @@ -17,6 +17,7 @@ import RightToolbar from './right-toolbar/index.vue'; import SvgIcon from './svg-icon/index.vue'; import IconSelect from './icon-select/index.vue'; import download from './crud'; +import Verify from './verifition/Verify.vue'; // Manually introduce ECharts modules to reduce packing size @@ -46,5 +47,6 @@ export default { Vue.component('RightToolbar', RightToolbar); Vue.component('SvgIcon', SvgIcon); Vue.component('IconSelect', IconSelect); + Vue.component('Verify', Verify); }, }; diff --git a/continew-admin-ui/src/components/verifition/Verify.vue b/continew-admin-ui/src/components/verifition/Verify.vue new file mode 100644 index 00000000..9c9871b6 --- /dev/null +++ b/continew-admin-ui/src/components/verifition/Verify.vue @@ -0,0 +1,431 @@ +<template> + <div v-show="showBox" :class="mode === 'pop' ? 'mask' : ''"> + <div + :class="mode === 'pop' ? 'verifybox' : ''" + :style="{ 'max-width': parseInt(imgSize.width) + 30 + 'px' }" + > + <div v-if="mode === 'pop'" class="verifybox-top"> + 请完成安全验证 + <span class="verifybox-close" @click="closeBox"> + <i class="iconfont icon-close"></i> + </span> + </div> + <div + class="verifybox-bottom" + :style="{ padding: mode === 'pop' ? '15px' : '0' }" + > + <!-- 验证码容器 --> + <component + :is="componentType" + v-if="componentType" + ref="instance" + :captcha-type="captchaType" + :type="verifyType" + :figure="figure" + :arith="arith" + :mode="mode" + :v-space="vSpace" + :explain="explain" + :img-size="imgSize" + :block-size="blockSize" + :bar-size="barSize" + ></component> + </div> + </div> + </div> +</template> + +<script type="text/babel"> + import { computed, ref, toRefs, watchEffect } from 'vue'; + import VerifySlide from './Verify/VerifySlide.vue'; + import VerifyPoints from './Verify/VerifyPoints.vue'; + + export default { + name: 'Vue2Verify', + components: { + VerifySlide, + VerifyPoints, + }, + props: { + captchaType: { + type: String, + required: true, + }, + figure: { + type: Number, + }, + arith: { + type: Number, + }, + mode: { + type: String, + default: 'pop', + }, + vSpace: { + type: Number, + }, + explain: { + type: String, + }, + imgSize: { + type: Object, + default() { + return { + width: '310px', + height: '155px', + }; + }, + }, + blockSize: { + type: Object, + }, + barSize: { + type: Object, + }, + }, + setup(props) { + const { captchaType, mode } = toRefs(props); + const clickShow = ref(false); + const verifyType = ref(undefined); + const componentType = ref(undefined); + + const instance = ref({}); + + const showBox = computed(() => { + if (mode.value === 'pop') { + return clickShow.value; + } + return true; + }); + /** + * refresh + * @description 刷新 + * */ + const refresh = () => { + if (instance.value.refresh) { + instance.value.refresh(); + } + }; + const closeBox = () => { + clickShow.value = false; + refresh(); + }; + const show = () => { + if (mode.value === 'pop') { + clickShow.value = true; + } + }; + watchEffect(() => { + switch (captchaType.value) { + case 'blockPuzzle': + verifyType.value = '2'; + componentType.value = 'VerifySlide'; + break; + case 'clickWord': + verifyType.value = ''; + componentType.value = 'VerifyPoints'; + break; + default: + break; + } + }); + + return { + clickShow, + verifyType, + componentType, + instance, + showBox, + closeBox, + show, + }; + }, + }; +</script> + +<style> + .verifybox { + position: relative; + box-sizing: border-box; + border-radius: 2px; + border: 1px solid #e4e7eb; + background-color: #fff; + box-shadow: 0 0 10px rgba(0, 0, 0, 0.3); + left: 50%; + top: 50%; + transform: translate(-50%, -50%); + } + .verifybox-top { + padding: 0 15px; + height: 50px; + line-height: 50px; + text-align: left; + font-size: 16px; + color: #45494c; + border-bottom: 1px solid #e4e7eb; + box-sizing: border-box; + } + .verifybox-bottom { + padding: 15px; + box-sizing: border-box; + } + .verifybox-close { + position: absolute; + top: 13px; + right: 9px; + width: 24px; + height: 24px; + text-align: center; + cursor: pointer; + } + .mask { + position: fixed; + top: 0; + left: 0; + z-index: 1001; + width: 100%; + height: 100vh; + background: rgba(0, 0, 0, 0.3); + /* display: none; */ + transition: all 0.5s; + } + .verify-tips { + position: absolute; + left: 0; + bottom: 0; + width: 100%; + height: 30px; + line-height: 30px; + color: #fff; + } + .suc-bg { + background-color: rgba(92, 184, 92, 0.5); + filter: progid:DXImageTransform.Microsoft.gradient(startcolorstr=#7f5CB85C, endcolorstr=#7f5CB85C); + } + .err-bg { + background-color: rgba(217, 83, 79, 0.5); + filter: progid:DXImageTransform.Microsoft.gradient(startcolorstr=#7fD9534F, endcolorstr=#7fD9534F); + } + .tips-enter, + .tips-leave-to { + bottom: -30px; + } + .tips-enter-active, + .tips-leave-active { + transition: bottom 0.5s; + } + /* ---------------------------- */ + /*常规验证码*/ + .verify-code { + font-size: 20px; + text-align: center; + cursor: pointer; + margin-bottom: 5px; + border: 1px solid #ddd; + } + + .cerify-code-panel { + height: 100%; + overflow: hidden; + } + + .verify-code-area { + float: left; + } + + .verify-input-area { + float: left; + width: 60%; + padding-right: 10px; + } + + .verify-change-area { + line-height: 30px; + float: left; + } + + .varify-input-code { + display: inline-block; + width: 100%; + height: 25px; + } + + .verify-change-code { + color: #337ab7; + cursor: pointer; + } + + .verify-btn { + width: 200px; + height: 30px; + background-color: #337ab7; + color: #ffffff; + border: none; + margin-top: 10px; + } + + /*滑动验证码*/ + .verify-bar-area { + position: relative; + background: #ffffff; + text-align: center; + -webkit-box-sizing: content-box; + -moz-box-sizing: content-box; + box-sizing: content-box; + border: 1px solid #ddd; + -webkit-border-radius: 4px; + } + + .verify-bar-area .verify-move-block { + position: absolute; + top: 0; + left: 0; + background: #fff; + cursor: pointer; + -webkit-box-sizing: content-box; + -moz-box-sizing: content-box; + box-sizing: content-box; + box-shadow: 0 0 2px #888888; + -webkit-border-radius: 1px; + } + + .verify-bar-area .verify-move-block:hover { + background-color: #337ab7; + color: #ffffff; + } + + .verify-bar-area .verify-left-bar { + position: absolute; + top: -1px; + left: -1px; + background: #f0fff0; + cursor: pointer; + -webkit-box-sizing: content-box; + -moz-box-sizing: content-box; + box-sizing: content-box; + border: 1px solid #ddd; + } + + .verify-img-panel { + margin: 0; + -webkit-box-sizing: content-box; + -moz-box-sizing: content-box; + box-sizing: content-box; + border-top: 1px solid #ddd; + border-bottom: 1px solid #ddd; + border-radius: 3px; + position: relative; + } + + .verify-img-panel .verify-refresh { + width: 25px; + height: 25px; + text-align: center; + padding: 5px; + cursor: pointer; + position: absolute; + top: 0; + right: 0; + z-index: 2; + } + + .verify-img-panel .icon-refresh { + font-size: 20px; + color: #fff; + } + + .verify-img-panel .verify-gap { + background-color: #fff; + position: relative; + z-index: 2; + border: 1px solid #fff; + } + + .verify-bar-area .verify-move-block .verify-sub-block { + position: absolute; + text-align: center; + z-index: 3; + /* border: 1px solid #fff; */ + } + + .verify-bar-area .verify-move-block .verify-icon { + font-size: 18px; + } + + .verify-bar-area .verify-msg { + z-index: 3; + } + + .iconfont { + font-family: 'iconfont', serif !important; + font-size: 16px; + font-style: normal; + -webkit-font-smoothing: antialiased; + -moz-osx-font-smoothing: grayscale; + } + + .icon-check:before { + content: ' '; + display: block; + width: 16px; + height: 16px; + position: absolute; + margin: auto; + left: 0; + right: 0; + top: 0; + bottom: 0; + z-index: 9999; + background-image: url(''); + background-size: contain; + } + + .icon-close:before { + content: ' '; + display: block; + width: 16px; + height: 16px; + position: absolute; + margin: auto; + left: 0; + right: 0; + top: 0; + bottom: 0; + z-index: 9999; + background-image: url(''); + background-size: contain; + } + + .icon-right:before { + content: ' '; + display: block; + width: 16px; + height: 16px; + position: absolute; + margin: auto; + left: 0; + right: 0; + top: 0; + bottom: 0; + background-size: cover; + z-index: 9999; + background-image: url(''); + background-size: contain; + } + + .icon-refresh:before { + content: ' '; + display: block; + width: 16px; + height: 16px; + position: absolute; + margin: auto; + left: 0; + right: 0; + top: 0; + bottom: 0; + z-index: 9999; + background-image: url(''); + background-size: contain; + } +</style> diff --git a/continew-admin-ui/src/components/verifition/Verify/VerifyPoints.vue b/continew-admin-ui/src/components/verifition/Verify/VerifyPoints.vue new file mode 100644 index 00000000..add38991 --- /dev/null +++ b/continew-admin-ui/src/components/verifition/Verify/VerifyPoints.vue @@ -0,0 +1,296 @@ +<template> + <div style="position: relative"> + <div class="verify-img-out"> + <div + class="verify-img-panel" + :style="{ + 'width': setSize.imgWidth, + 'height': setSize.imgHeight, + 'background-size': setSize.imgWidth + ' ' + setSize.imgHeight, + 'margin-bottom': vSpace + 'px', + }" + > + <div + v-show="showRefresh" + class="verify-refresh" + style="z-index: 3" + @click="refresh" + > + <i class="iconfont icon-refresh"></i> + </div> + <img + ref="canvas" + :src="'data:image/png;base64,' + pointBackImgBase" + alt="" + style="width: 100%; height: 100%; display: block" + @click="bindingClick ? canvasClick($event) : undefined" + /> + + <div + v-for="(tempPoint, index) in tempPoints" + :key="index" + class="point-area" + :style="{ + 'background-color': '#1abd6c', + 'color': '#fff', + 'z-index': 9999, + 'width': '20px', + 'height': '20px', + 'text-align': 'center', + 'line-height': '20px', + 'border-radius': '50%', + 'position': 'absolute', + 'top': parseInt(tempPoint.y - 10) + 'px', + 'left': parseInt(tempPoint.x - 10) + 'px', + }" + > + {{ index + 1 }} + </div> + </div> + </div> + + <div + class="verify-bar-area" + :style="{ + 'width': setSize.imgWidth, + 'color': barAreaColor, + 'border-color': barAreaBorderColor, + 'line-height': barSize.height, + }" + > + <span class="verify-msg">{{ text }}</span> + </div> + </div> +</template> + +<script type="text/babel"> + import { + checkBehaviorCaptcha, + getBehaviorCaptcha, + } from '@/api/common/captcha'; + import { + getCurrentInstance, + nextTick, + onMounted, + reactive, + ref, + toRefs, + } from 'vue'; + import { resetSize } from '../utils/util'; + import { aesEncrypt } from '../utils/ase'; + + export default { + name: 'VerifyPoints', + props: { + // 弹出式pop,固定fixed + mode: { + type: String, + default: '', + }, + captchaType: { + type: String, + }, + // 间隔 + vSpace: { + type: Number, + default: 5, + }, + imgSize: { + type: Object, + default() { + return { + width: '310px', + height: '155px', + }; + }, + }, + barSize: { + type: Object, + default() { + return { + width: '310px', + height: '40px', + }; + }, + }, + }, + setup(props) { + const { mode, captchaType } = toRefs(props); + const { proxy } = getCurrentInstance(); + const secretKey = ref(''); // 后端返回的ase加密秘钥 + const checkNum = ref(3); // 默认需要点击的字数 + const fontPos = reactive([]); // 选中的坐标信息 + const checkPosArr = reactive([]); // 用户点击的坐标 + const num = ref(1); // 点击的记数 + const pointBackImgBase = ref(''); // 后端获取到的背景图片 + const poinTextList = reactive([]); // 后端返回的点击字体顺序 + const backToken = ref(''); // 后端返回的token值 + const setSize = reactive({ + imgHeight: 0, + imgWidth: 0, + barHeight: 0, + barWidth: 0, + }); + const tempPoints = reactive([]); + const text = ref(''); + const barAreaColor = ref(undefined); + const barAreaBorderColor = ref(undefined); + const showRefresh = ref(true); + const bindingClick = ref(true); + + // 请求背景图片和验证图片 + function getPictrue() { + const data = { + captchaType: captchaType.value, + }; + getBehaviorCaptcha(data).then((res) => { + pointBackImgBase.value = res.data.originalImageBase64; + backToken.value = res.data.token; + secretKey.value = res.data.secretKey; + poinTextList.value = res.data.wordList; + text.value = `请依次点击【${poinTextList.value.join(',')}】`; + }); + } + + // 获取坐标 + const getMousePos = function (obj, e) { + const x = e.offsetX; + const y = e.offsetY; + return { x, y }; + }; + + // 创建坐标点 + const createPoint = function (pos) { + tempPoints.push({ ...pos }); + return num.value + 1; + }; + + // 坐标转换函数 + const pointTransfrom = function (pointArr, imgSize) { + return pointArr.map((p) => { + const x = Math.round((310 * p.x) / parseInt(imgSize.imgWidth, 10)); + const y = Math.round((155 * p.y) / parseInt(imgSize.imgHeight, 10)); + return { x, y }; + }); + }; + + const init = () => { + // 加载页面 + fontPos.splice(0, fontPos.length); + checkPosArr.splice(0, checkPosArr.length); + num.value = 1; + getPictrue(); + nextTick(() => { + const { imgHeight, imgWidth, barHeight, barWidth } = resetSize(proxy); + setSize.imgHeight = imgHeight; + setSize.imgWidth = imgWidth; + setSize.barHeight = barHeight; + setSize.barWidth = barWidth; + proxy.$parent.$emit('ready', proxy); + }); + }; + onMounted(() => { + // 禁止拖拽 + init(); + proxy.$el.onselectstart = function () { + return false; + }; + }); + + const refresh = function () { + tempPoints.splice(0, tempPoints.length); + barAreaColor.value = '#000'; + barAreaBorderColor.value = '#ddd'; + bindingClick.value = true; + fontPos.splice(0, fontPos.length); + checkPosArr.splice(0, checkPosArr.length); + num.value = 1; + getPictrue(); + text.value = '验证失败'; + showRefresh.value = true; + }; + + const canvas = ref(null); + const canvasClick = (e) => { + checkPosArr.push(getMousePos(canvas, e)); + if (num.value === checkNum.value) { + num.value = createPoint(getMousePos(canvas, e)); + // 按比例转换坐标值 + const arr = pointTransfrom(checkPosArr, setSize); + checkPosArr.length = 0; + checkPosArr.push(...arr); + // 等创建坐标执行完 + setTimeout(() => { + // var flag = this.comparePos(this.fontPos, this.checkPosArr); + // 发送后端请求 + const captchaVerification = secretKey.value + ? aesEncrypt( + `${backToken.value}---${JSON.stringify(checkPosArr)}`, + secretKey.value, + ) + : `${backToken.value}---${JSON.stringify(checkPosArr)}`; + const data = { + captchaType: captchaType.value, + pointJson: secretKey.value + ? aesEncrypt(JSON.stringify(checkPosArr), secretKey.value) + : JSON.stringify(checkPosArr), + token: backToken.value, + }; + checkBehaviorCaptcha(data).then((res) => { + if (res.success && res.data.repCode === '0000') { + barAreaColor.value = '#4cae4c'; + barAreaBorderColor.value = '#5cb85c'; + text.value = '验证成功'; + bindingClick.value = false; + if (mode.value === 'pop') { + setTimeout(() => { + proxy.$parent.clickShow = false; + refresh(); + }, 1500); + } + proxy.$parent.$emit('success', { captchaVerification }); + } else { + proxy.$parent.$emit('error', proxy); + barAreaColor.value = '#d9534f'; + barAreaBorderColor.value = '#d9534f'; + text.value = res.data.repMsg; + setTimeout(() => { + refresh(); + }, 700); + } + }); + }, 400); + } + if (num.value < checkNum.value) { + num.value = createPoint(getMousePos(canvas, e)); + } + }; + + return { + secretKey, + checkNum, + fontPos, + checkPosArr, + num, + pointBackImgBase, + poinTextList, + backToken, + setSize, + tempPoints, + text, + barAreaColor, + barAreaBorderColor, + showRefresh, + bindingClick, + init, + canvas, + canvasClick, + getMousePos, + createPoint, + refresh, + getPictrue, + pointTransfrom, + }; + }, + }; +</script> diff --git a/continew-admin-ui/src/components/verifition/Verify/VerifySlide.vue b/continew-admin-ui/src/components/verifition/Verify/VerifySlide.vue new file mode 100644 index 00000000..c648fbe1 --- /dev/null +++ b/continew-admin-ui/src/components/verifition/Verify/VerifySlide.vue @@ -0,0 +1,462 @@ +<template> + <div style="position: relative"> + <div + v-if="type === '2'" + class="verify-img-out" + :style="{ height: parseInt(setSize.imgHeight) + vSpace + 'px' }" + > + <div + class="verify-img-panel" + :style="{ width: setSize.imgWidth, height: setSize.imgHeight }" + > + <img + :src="'data:image/png;base64,' + backImgBase" + alt="" + style="width: 100%; height: 100%; display: block" + /> + <div v-show="showRefresh" class="verify-refresh" @click="refresh" + ><i class="iconfont icon-refresh"></i + ></div> + <transition name="tips"> + <span + v-if="tipWords" + class="verify-tips" + :class="passFlag ? 'suc-bg' : 'err-bg'" + >{{ tipWords }}</span + > + </transition> + </div> + </div> + <!-- 公共部分 --> + <div + class="verify-bar-area" + :style="{ + 'width': setSize.imgWidth, + 'height': barSize.height, + 'line-height': barSize.height, + }" + > + <span class="verify-msg" v-text="text"></span> + <div + class="verify-left-bar" + :style="{ + 'width': leftBarWidth !== undefined ? leftBarWidth : barSize.height, + 'height': barSize.height, + 'border-color': leftBarBorderColor, + 'transaction': transitionWidth, + }" + > + <span class="verify-msg" v-text="finishText"></span> + <div + class="verify-move-block" + :style="{ + 'width': barSize.height, + 'height': barSize.height, + 'background-color': moveBlockBackgroundColor, + 'left': moveBlockLeft, + 'transition': transitionLeft, + }" + @touchstart="start" + @mousedown="start" + > + <i + :class="['verify-icon iconfont', iconClass]" + :style="{ color: iconColor }" + ></i> + <div + v-if="type === '2'" + class="verify-sub-block" + :style="{ + 'width': + Math.floor((parseInt(setSize.imgWidth) * 47) / 310) + 'px', + 'height': setSize.imgHeight, + 'top': '-' + (parseInt(setSize.imgHeight) + vSpace) + 'px', + 'background-size': setSize.imgWidth + ' ' + setSize.imgHeight, + }" + > + <img + :src="'data:image/png;base64,' + blockBackImgBase" + alt="" + style=" + width: 100%; + height: 100%; + display: block; + -webkit-user-drag: none; + " + /> + </div> + </div> + </div> + </div> + </div> +</template> + +<script type="text/babel"> + import { + computed, + onMounted, + reactive, + ref, + watch, + nextTick, + toRefs, + getCurrentInstance, + } from 'vue'; + import { + checkBehaviorCaptcha, + getBehaviorCaptcha, + } from '@/api/common/captcha'; + import { aesEncrypt } from '../utils/ase'; + import { resetSize } from '../utils/util'; + + export default { + name: 'VerifySlide', + props: { + captchaType: { + type: String, + }, + type: { + type: String, + default: '1', + }, + // 弹出式pop,固定fixed + mode: { + type: String, + default: 'fixed', + }, + vSpace: { + type: Number, + default: 5, + }, + explain: { + type: String, + default: '向右滑动完成验证', + }, + imgSize: { + type: Object, + default() { + return { + width: '310px', + height: '155px', + }; + }, + }, + blockSize: { + type: Object, + default() { + return { + width: '50px', + height: '50px', + }; + }, + }, + barSize: { + type: Object, + default() { + return { + width: '310px', + height: '40px', + }; + }, + }, + }, + setup(props) { + const { mode, captchaType, type, blockSize, explain } = toRefs(props); + const { proxy } = getCurrentInstance(); + const secretKey = ref(''); // 后端返回的ase加密秘钥 + const passFlag = ref(''); // 是否通过的标识 + const backImgBase = ref(''); // 验证码背景图片 + const blockBackImgBase = ref(''); // 验证滑块的背景图片 + const backToken = ref(''); // 后端返回的唯一token值 + const startMoveTime = ref(''); // 移动开始的时间 + const endMovetime = ref(''); // 移动结束的时间 + const tipsBackColor = ref(''); // 提示词的背景颜色 + const tipWords = ref(''); + const text = ref(''); + const finishText = ref(''); + const setSize = reactive({ + imgHeight: 0, + imgWidth: 0, + barHeight: 0, + barWidth: 0, + }); + const top = ref(0); + const left = ref(0); + const moveBlockLeft = ref(undefined); + const leftBarWidth = ref(undefined); + // 移动中样式 + const moveBlockBackgroundColor = ref(undefined); + const leftBarBorderColor = ref('#ddd'); + const iconColor = ref(undefined); + const iconClass = ref('icon-right'); + const status = ref(false); // 鼠标状态 + const isEnd = ref(false); // 是够验证完成 + const showRefresh = ref(true); + const transitionLeft = ref(''); + const transitionWidth = ref(''); + const startLeft = ref(0); + + // 请求背景图片和验证图片 + function getPictrue() { + const data = { + captchaType: captchaType.value, + }; + getBehaviorCaptcha(data).then((res) => { + backImgBase.value = res.data.originalImageBase64; + blockBackImgBase.value = res.data.jigsawImageBase64; + backToken.value = res.data.token; + secretKey.value = res.data.secretKey; + }); + } + const barArea = computed(() => { + return proxy.$el.querySelector('.verify-bar-area'); + }); + // 鼠标移动 + function move(e) { + e = e || window.event; + if (status.value && isEnd.value === false) { + let x; + if (!e.touches) { + // 兼容PC端 + x = e.clientX; + } else { + // 兼容移动端 + x = e.touches[0].pageX; + } + const bar_area_left = barArea.value.getBoundingClientRect().left; + let move_block_left = x - bar_area_left; // 小方块相对于父元素的left值 + if ( + move_block_left >= + barArea.value.offsetWidth - + parseInt(parseInt(blockSize.value.width, 10) / 2, 10) - + 2 + ) { + move_block_left = + barArea.value.offsetWidth - + parseInt(parseInt(blockSize.value.width, 10) / 2, 10) - + 2; + } + if (move_block_left <= 0) { + move_block_left = parseInt( + parseInt(blockSize.value.width, 10) / 2, + 10, + ); + } + // 拖动后小方块的left值 + moveBlockLeft.value = `${move_block_left - startLeft.value}px`; + leftBarWidth.value = `${move_block_left - startLeft.value}px`; + } + } + + const refresh = () => { + showRefresh.value = true; + finishText.value = ''; + + transitionLeft.value = 'left .3s'; + moveBlockLeft.value = 0; + + leftBarWidth.value = undefined; + transitionWidth.value = 'width .3s'; + + leftBarBorderColor.value = '#ddd'; + moveBlockBackgroundColor.value = '#fff'; + iconColor.value = '#000'; + iconClass.value = 'icon-right'; + isEnd.value = false; + + getPictrue(); + setTimeout(() => { + transitionWidth.value = ''; + transitionLeft.value = ''; + text.value = explain.value; + }, 300); + }; + + // 鼠标松开 + function end() { + endMovetime.value = +new Date(); + // 判断是否重合 + if (status.value && isEnd.value === false) { + let moveLeftDistance = parseInt( + (moveBlockLeft.value || '').replace('px', ''), + 10, + ); + moveLeftDistance = + (moveLeftDistance * 310) / parseInt(setSize.imgWidth, 10); + const data = { + captchaType: captchaType.value, + pointJson: secretKey.value + ? aesEncrypt( + JSON.stringify({ x: moveLeftDistance, y: 5.0 }), + secretKey.value, + ) + : JSON.stringify({ x: moveLeftDistance, y: 5.0 }), + token: backToken.value, + }; + checkBehaviorCaptcha(data).then((res) => { + if (res.success && res.data.repCode === '0000') { + moveBlockBackgroundColor.value = '#5cb85c'; + leftBarBorderColor.value = '#5cb85c'; + iconColor.value = '#fff'; + iconClass.value = 'icon-check'; + showRefresh.value = false; + isEnd.value = true; + if (mode.value === 'pop') { + setTimeout(() => { + proxy.$parent.clickShow = false; + refresh(); + }, 1500); + } + passFlag.value = true; + tipWords.value = `${( + (endMovetime.value - startMoveTime.value) / + 1000 + ).toFixed(2)}s验证成功`; + const captchaVerification = secretKey.value + ? aesEncrypt( + `${backToken.value}---${JSON.stringify({ + x: moveLeftDistance, + y: 5.0, + })}`, + secretKey.value, + ) + : `${backToken.value}---${JSON.stringify({ + x: moveLeftDistance, + y: 5.0, + })}`; + setTimeout(() => { + tipWords.value = ''; + proxy.$parent.closeBox(); + proxy.$parent.$emit('success', { captchaVerification }); + }, 1000); + } else { + moveBlockBackgroundColor.value = '#d9534f'; + leftBarBorderColor.value = '#d9534f'; + iconColor.value = '#fff'; + iconClass.value = 'icon-close'; + passFlag.value = false; + setTimeout(function () { + refresh(); + }, 1000); + proxy.$parent.$emit('error', proxy); + tipWords.value = res.data.repMsg; + setTimeout(() => { + tipWords.value = ''; + }, 1000); + } + }); + status.value = false; + } + } + + function init() { + text.value = explain.value; + getPictrue(); + nextTick(() => { + const { imgHeight, imgWidth, barHeight, barWidth } = resetSize(proxy); + setSize.imgHeight = imgHeight; + setSize.imgWidth = imgWidth; + setSize.barHeight = barHeight; + setSize.barWidth = barWidth; + proxy.$parent.$emit('ready', proxy); + }); + + window.removeEventListener('touchmove', function (e) { + move(e); + }); + window.removeEventListener('mousemove', function (e) { + move(e); + }); + + // 鼠标松开 + window.removeEventListener('touchend', function () { + end(); + }); + window.removeEventListener('mouseup', function () { + end(); + }); + + window.addEventListener('touchmove', function (e) { + move(e); + }); + window.addEventListener('mousemove', function (e) { + move(e); + }); + + // 鼠标松开 + window.addEventListener('touchend', function () { + end(); + }); + window.addEventListener('mouseup', function () { + end(); + }); + } + watch(type, () => { + init(); + }); + onMounted(() => { + // 禁止拖拽 + init(); + proxy.$el.onselectstart = function () { + return false; + }; + }); + // 鼠标按下 + function start(e) { + e = e || window.event; + let x; + if (!e.touches) { + // 兼容PC端 + x = e.clientX; + } else { + // 兼容移动端 + x = e.touches[0].pageX; + } + startLeft.value = Math.floor( + x - barArea.value.getBoundingClientRect().left, + ); + startMoveTime.value = +new Date(); // 开始滑动的时间 + if (isEnd.value === false) { + text.value = ''; + moveBlockBackgroundColor.value = '#337ab7'; + leftBarBorderColor.value = '#337AB7'; + iconColor.value = '#fff'; + e.stopPropagation(); + status.value = true; + } + } + + return { + secretKey, // 后端返回的ase加密秘钥 + passFlag, // 是否通过的标识 + backImgBase, // 验证码背景图片 + blockBackImgBase, // 验证滑块的背景图片 + backToken, // 后端返回的唯一token值 + startMoveTime, // 移动开始的时间 + endMovetime, // 移动结束的时间 + tipsBackColor, // 提示词的背景颜色 + tipWords, + text, + finishText, + setSize, + top, + left, + moveBlockLeft, + leftBarWidth, + // 移动中样式 + moveBlockBackgroundColor, + leftBarBorderColor, + iconColor, + iconClass, + status, // 鼠标状态 + isEnd, // 是够验证完成 + showRefresh, + transitionLeft, + transitionWidth, + barArea, + refresh, + start, + }; + }, + }; +</script> diff --git a/continew-admin-ui/src/components/verifition/utils/ase.js b/continew-admin-ui/src/components/verifition/utils/ase.js new file mode 100644 index 00000000..9ed83769 --- /dev/null +++ b/continew-admin-ui/src/components/verifition/utils/ase.js @@ -0,0 +1,14 @@ +import CryptoJS from 'crypto-js'; +/** + * @word 要加密的内容 + * @keyWord String 服务器随机返回的关键字 + * */ +export function aesEncrypt(word, keyWord = 'XwKsGlMcdPMEhR1B') { + const key = CryptoJS.enc.Utf8.parse(keyWord); + const arcs = CryptoJS.enc.Utf8.parse(word); + const encrypted = CryptoJS.AES.encrypt(arcs, key, { + mode: CryptoJS.mode.ECB, + padding: CryptoJS.pad.Pkcs7, + }); + return encrypted.toString(); +} diff --git a/continew-admin-ui/src/components/verifition/utils/util.js b/continew-admin-ui/src/components/verifition/utils/util.js new file mode 100644 index 00000000..901e307f --- /dev/null +++ b/continew-admin-ui/src/components/verifition/utils/util.js @@ -0,0 +1,39 @@ +export function resetSize(vm) { + let img_width; + let img_height; + let bar_width; + let bar_height; // 图片的宽度、高度,移动条的宽度、高度 + + const parentWidth = vm.$el.parentNode.offsetWidth || window.offsetWidth; + const parentHeight = vm.$el.parentNode.offsetHeight || window.offsetHeight; + if (vm.imgSize.width.indexOf('%') !== -1) { + img_width = `${(parseInt(vm.imgSize.width, 10) / 100) * parentWidth}px`; + } else { + img_width = vm.imgSize.width; + } + + if (vm.imgSize.height.indexOf('%') !== -1) { + img_height = `${(parseInt(vm.imgSize.height, 10) / 100) * parentHeight}px`; + } else { + img_height = vm.imgSize.height; + } + + if (vm.barSize.width.indexOf('%') !== -1) { + bar_width = `${(parseInt(vm.barSize.width, 10) / 100) * parentWidth}px`; + } else { + bar_width = vm.barSize.width; + } + + if (vm.barSize.height.indexOf('%') !== -1) { + bar_height = `${(parseInt(vm.barSize.height, 10) / 100) * parentHeight}px`; + } else { + bar_height = vm.barSize.height; + } + + return { + imgWidth: img_width, + imgHeight: img_height, + barWidth: bar_width, + barHeight: bar_height, + }; +} diff --git a/continew-admin-ui/src/views/login/components/phone-login.vue b/continew-admin-ui/src/views/login/components/phone-login.vue index 71b634a7..621225d6 100644 --- a/continew-admin-ui/src/views/login/components/phone-login.vue +++ b/continew-admin-ui/src/views/login/components/phone-login.vue @@ -29,7 +29,7 @@ class="captcha-btn" :loading="captchaLoading" :disabled="captchaDisable" - @click="handleSendCaptcha" + @click="handleOpenBehaviorCaptcha" > {{ captchaBtnName }} </a-button> @@ -43,6 +43,13 @@ >{{ $t('login.button') }}(演示不开放) </a-button> </a-form> + <Verify + ref="verifyRef" + :mode="captchaMode" + :captcha-type="captchaType" + :img-size="{ width: '330px', height: '155px' }" + @success="handleSendCaptcha" + ></Verify> </template> <script lang="ts" setup> @@ -52,7 +59,7 @@ import { ValidatedError } from '@arco-design/web-vue'; import { useUserStore } from '@/store'; import { PhoneLoginReq } from '@/api/auth'; - import { getSmsCaptcha } from '@/api/common/captcha'; + import { BehaviorCaptchaReq, getSmsCaptcha } from '@/api/common/captcha'; const { proxy } = getCurrentInstance() as any; const { t } = useI18n(); @@ -63,6 +70,8 @@ const captchaDisable = ref(true); const captchaTime = ref(60); const captchaTimer = ref(); + const captchaType = ref('blockPuzzle'); + const captchaMode = ref('pop'); const captchaBtnNameKey = ref('login.captcha.get'); const captchaBtnName = computed(() => t(captchaBtnNameKey.value)); const data = reactive({ @@ -85,6 +94,18 @@ }); const { form, rules } = toRefs(data); + /** + * 弹出行为验证码 + */ + const handleOpenBehaviorCaptcha = () => { + if (captchaLoading.value) return; + proxy.$refs.formRef.validateField('phone', (valid: any) => { + if (!valid) { + proxy.$refs.verifyRef.show(); + } + }); + }; + /** * 重置验证码 */ @@ -98,13 +119,13 @@ /** * 发送验证码 */ - const handleSendCaptcha = () => { + const handleSendCaptcha = (captchaParam: BehaviorCaptchaReq) => { if (captchaLoading.value) return; proxy.$refs.formRef.validateField('phone', (valid: any) => { if (!valid) { captchaLoading.value = true; captchaBtnNameKey.value = 'login.captcha.ing'; - getSmsCaptcha(form.value.phone) + getSmsCaptcha(form.value.phone, captchaParam) .then((res) => { captchaLoading.value = false; captchaDisable.value = true; diff --git a/continew-admin-ui/src/views/system/user/center/components/security-settings/update-phone.vue b/continew-admin-ui/src/views/system/user/center/components/security-settings/update-phone.vue index 0a9b352b..11577c71 100644 --- a/continew-admin-ui/src/views/system/user/center/components/security-settings/update-phone.vue +++ b/continew-admin-ui/src/views/system/user/center/components/security-settings/update-phone.vue @@ -47,7 +47,7 @@ v-model="form.newPhone" :placeholder=" $t( - 'userCenter.securitySettings.updatePhone.form.placeholder.newPhone' + 'userCenter.securitySettings.updatePhone.form.placeholder.newPhone', ) " allow-clear @@ -73,7 +73,7 @@ type="primary" :disabled="captchaDisable" class="captcha-btn" - @click="handleSendCaptcha" + @click="handleOpenBehaviorCaptcha" > {{ captchaBtnName }} </a-button> @@ -81,7 +81,7 @@ <a-form-item :label=" $t( - 'userCenter.securitySettings.updatePhone.form.label.currentPassword' + 'userCenter.securitySettings.updatePhone.form.label.currentPassword', ) " field="currentPassword" @@ -90,7 +90,7 @@ v-model="form.currentPassword" :placeholder=" $t( - 'userCenter.securitySettings.updatePhone.form.placeholder.currentPassword' + 'userCenter.securitySettings.updatePhone.form.placeholder.currentPassword', ) " :max-length="32" @@ -98,13 +98,20 @@ /> </a-form-item> </a-form> + <Verify + ref="verifyRef" + :mode="captchaMode" + :captcha-type="captchaType" + :img-size="{ width: '330px', height: '155px' }" + @success="handleSendCaptcha" + ></Verify> </a-modal> </template> <script lang="ts" setup> import { getCurrentInstance, ref, reactive, computed } from 'vue'; import { FieldRule } from '@arco-design/web-vue'; - import { getSmsCaptcha } from '@/api/common/captcha'; + import { BehaviorCaptchaReq, getSmsCaptcha } from '@/api/common/captcha'; import { UserPhoneUpdateReq, updatePhone } from '@/api/system/user-center'; import { useI18n } from 'vue-i18n'; import { useUserStore } from '@/store'; @@ -117,6 +124,8 @@ const captchaTimer = ref(); const captchaLoading = ref(false); const captchaDisable = ref(true); + const captchaType = ref('blockPuzzle'); + const captchaMode = ref('pop'); const visible = ref(false); const captchaBtnNameKey = ref('userCenter.securitySettings.captcha.get'); const captchaBtnName = computed(() => t(captchaBtnNameKey.value)); @@ -134,13 +143,13 @@ { required: true, message: t( - 'userCenter.securitySettings.updatePhone.form.error.required.newPhone' + 'userCenter.securitySettings.updatePhone.form.error.required.newPhone', ), }, { match: /^1[3-9]\d{9}$/, message: t( - 'userCenter.securitySettings.updatePhone.form.error.match.newPhone' + 'userCenter.securitySettings.updatePhone.form.error.match.newPhone', ), }, ], @@ -154,7 +163,7 @@ { required: true, message: t( - 'userCenter.securitySettings.updatePhone.form.error.required.currentPassword' + 'userCenter.securitySettings.updatePhone.form.error.required.currentPassword', ), }, ], @@ -171,26 +180,38 @@ captchaDisable.value = false; }; + /** + * 弹出行为验证码 + */ + const handleOpenBehaviorCaptcha = () => { + if (captchaLoading.value) return; + proxy.$refs.formRef.validateField('newPhone', (valid: any) => { + if (!valid) { + proxy.$refs.verifyRef.show(); + } + }); + }; + /** * 发送验证码 */ - const handleSendCaptcha = () => { + const handleSendCaptcha = (captchaParam: BehaviorCaptchaReq) => { if (captchaLoading.value) return; proxy.$refs.formRef.validateField('newPhone', (valid: any) => { if (!valid) { captchaLoading.value = true; captchaBtnNameKey.value = 'userCenter.securitySettings.captcha.ing'; - getSmsCaptcha(form.newPhone) + getSmsCaptcha(form.newPhone, captchaParam) .then((res) => { captchaLoading.value = false; captchaDisable.value = true; captchaBtnNameKey.value = `${t( - 'userCenter.securitySettings.captcha.get' + 'userCenter.securitySettings.captcha.get', )}(${(captchaTime.value -= 1)}s)`; captchaTimer.value = window.setInterval(() => { captchaTime.value -= 1; captchaBtnNameKey.value = `${t( - 'userCenter.securitySettings.captcha.get' + 'userCenter.securitySettings.captcha.get', )}(${captchaTime.value}s)`; if (captchaTime.value <= 0) { resetCaptcha(); diff --git a/continew-admin-ui/tsconfig.json b/continew-admin-ui/tsconfig.json index c025998c..1186c8db 100644 --- a/continew-admin-ui/tsconfig.json +++ b/continew-admin-ui/tsconfig.json @@ -13,7 +13,8 @@ "@/*": ["src/*"] }, "lib": ["es2020", "dom"], - "skipLibCheck": true + "skipLibCheck": true, + "allowJs": true }, "include": ["src/**/*", "src/**/*.vue"], "exclude": ["node_modules"] diff --git a/continew-admin-webapi/src/main/java/top/charles7c/continew/admin/webapi/common/CaptchaController.java b/continew-admin-webapi/src/main/java/top/charles7c/continew/admin/webapi/common/CaptchaController.java index 5b5de9db..e91b3841 100644 --- a/continew-admin-webapi/src/main/java/top/charles7c/continew/admin/webapi/common/CaptchaController.java +++ b/continew-admin-webapi/src/main/java/top/charles7c/continew/admin/webapi/common/CaptchaController.java @@ -35,11 +35,14 @@ import org.dromara.sms4j.api.entity.SmsResponse; import org.dromara.sms4j.comm.constant.SupplierConstant; import org.dromara.sms4j.core.factory.SmsFactory; import org.redisson.api.RateType; +import org.springframework.http.HttpHeaders; import org.springframework.validation.annotation.Validated; -import org.springframework.web.bind.annotation.GetMapping; -import org.springframework.web.bind.annotation.RequestMapping; -import org.springframework.web.bind.annotation.RestController; +import org.springframework.web.bind.annotation.*; +import com.anji.captcha.model.common.RepCodeEnum; +import com.anji.captcha.model.common.ResponseModel; +import com.anji.captcha.model.vo.CaptchaVO; +import com.anji.captcha.service.CaptchaService; import com.wf.captcha.base.Captcha; import cn.dev33.satoken.annotation.SaIgnore; @@ -58,6 +61,7 @@ import top.charles7c.continew.starter.captcha.graphic.autoconfigure.GraphicCaptc import top.charles7c.continew.starter.core.autoconfigure.project.ProjectProperties; import top.charles7c.continew.starter.core.util.TemplateUtils; import top.charles7c.continew.starter.core.util.validate.CheckUtils; +import top.charles7c.continew.starter.core.util.validate.ValidationUtils; import top.charles7c.continew.starter.extension.crud.model.resp.R; import top.charles7c.continew.starter.messaging.mail.util.MailUtils; @@ -72,13 +76,27 @@ import top.charles7c.continew.starter.messaging.mail.util.MailUtils; @Validated @RestController @RequiredArgsConstructor -@RequestMapping("/common/captcha") +@RequestMapping("/captcha") public class CaptchaController { + private final CaptchaService captchaService; private final CaptchaProperties captchaProperties; private final ProjectProperties projectProperties; private final GraphicCaptchaProperties graphicCaptchaProperties; + @Operation(summary = "获取行为验证码", description = "获取行为验证码(Base64编码)") + @GetMapping("/behavior") + public R<Object> getBehaviorCaptcha(CaptchaVO captchaReq, HttpServletRequest request) { + captchaReq.setBrowserInfo(JakartaServletUtil.getClientIP(request) + request.getHeader(HttpHeaders.USER_AGENT)); + return R.ok(captchaService.get(captchaReq).getRepData()); + } + + @Operation(summary = "校验行为验证码", description = "校验行为验证码") + @PostMapping("/behavior") + public R<Object> checkBehaviorCaptcha(@RequestBody CaptchaVO captchaReq) { + return R.ok(captchaService.check(captchaReq)); + } + @Operation(summary = "获取图片验证码", description = "获取图片验证码(Base64编码,带图片格式:data:image/gif;base64)") @GetMapping("/img") public R<CaptchaResp> getImageCaptcha() { @@ -118,7 +136,11 @@ public class CaptchaController { @GetMapping("/sms") public R getSmsCaptcha( @NotBlank(message = "手机号不能为空") @Pattern(regexp = RegexConstants.MOBILE, message = "手机号格式错误") String phone, - HttpServletRequest request) { + CaptchaVO captchaReq, HttpServletRequest request) { + // 行为验证码校验 + ResponseModel verificationRes = captchaService.verification(captchaReq); + ValidationUtils.throwIfNotEqual(verificationRes.getRepCode(), RepCodeEnum.SUCCESS.getCode(), + verificationRes.getRepMsg()); CaptchaProperties.CaptchaSms captchaSms = captchaProperties.getSms(); String templateId = captchaSms.getTemplateId(); String limitKeyPrefix = CacheConstants.LIMIT_KEY_PREFIX; diff --git a/continew-admin-webapi/src/main/java/top/charles7c/continew/admin/webapi/system/MessageController.java b/continew-admin-webapi/src/main/java/top/charles7c/continew/admin/webapi/system/MessageController.java index a590e3e9..d8d06569 100644 --- a/continew-admin-webapi/src/main/java/top/charles7c/continew/admin/webapi/system/MessageController.java +++ b/continew-admin-webapi/src/main/java/top/charles7c/continew/admin/webapi/system/MessageController.java @@ -42,7 +42,7 @@ import top.charles7c.continew.starter.log.common.annotation.Log; /** * 消息管理 API * - * @author BULL_BCLS + * @author Bull-BCLS * @since 2023/10/15 19:05 */ @Tag(name = "消息管理 API") diff --git a/continew-admin-webapi/src/main/resources/config/application-dev.yml b/continew-admin-webapi/src/main/resources/config/application-dev.yml index 6b4a739a..930870c9 100644 --- a/continew-admin-webapi/src/main/resources/config/application-dev.yml +++ b/continew-admin-webapi/src/main/resources/config/application-dev.yml @@ -88,8 +88,13 @@ spring.cache: # 是否允许缓存空值(默认 true,表示允许,可以解决缓存穿透问题) cache-null-values: true ---- ### 图形验证码配置 +--- ### 验证码配置 continew-starter.captcha: + ## 行为验证码配置 + behavior: + enabled: true + cache-type: REDIS + water-mark: ${project.app-name} ## 图形验证码配置 graphic: enabled: true diff --git a/continew-admin-webapi/src/main/resources/config/application-prod.yml b/continew-admin-webapi/src/main/resources/config/application-prod.yml index 629ddd31..9407db58 100644 --- a/continew-admin-webapi/src/main/resources/config/application-prod.yml +++ b/continew-admin-webapi/src/main/resources/config/application-prod.yml @@ -90,8 +90,13 @@ spring.cache: # 是否允许缓存空值(默认 true,表示允许,可以解决缓存穿透问题) cache-null-values: true ---- ### 图形验证码配置 +--- ### 验证码配置 continew-starter.captcha: + ## 行为验证码配置 + behavior: + enabled: true + cache-type: REDIS + water-mark: ${project.app-name} ## 图形验证码配置 graphic: enabled: true