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