mirror of
				https://github.com/pure-admin/pure-admin-thin.git
				synced 2025-11-04 17:44:48 +08:00 
			
		
		
		
	feat: 新增登录页面
This commit is contained in:
		
							parent
							
								
									e23191f6ad
								
							
						
					
					
						commit
						292d760130
					
				
							
								
								
									
										3
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							@ -27,5 +27,6 @@
 | 
			
		||||
  "editor.codeActionsOnSave": {
 | 
			
		||||
    "source.fixAll.eslint": true
 | 
			
		||||
  },
 | 
			
		||||
  "iconify.excludes": ["el"]
 | 
			
		||||
  "iconify.excludes": ["el"],
 | 
			
		||||
  "cSpell.words": ["iconify", "Qrcode"]
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@ -42,6 +42,7 @@
 | 
			
		||||
## 开发环境
 | 
			
		||||
 | 
			
		||||
node 版本应不小于 16 ,pnpm 版本应不小于 6  
 | 
			
		||||
版本请勿过新,有先选择 node=16, pnpm=6
 | 
			
		||||
如果您还没安装 pnpm,请执行下面命令进行安装(mac 用户遇到安装报错请在命令前加上 sudo) 如果是 windows 用户 用使用管理员 power shell 来执行
 | 
			
		||||
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
@ -8,7 +8,7 @@
 | 
			
		||||
      name="viewport"
 | 
			
		||||
      content="width=device-width,initial-scale=1.0,minimum-scale=1.0,maximum-scale=1.0,user-scalable=0"
 | 
			
		||||
    />
 | 
			
		||||
    <title>pure-admin-thin</title>
 | 
			
		||||
    <title>Agileboot管理系统</title>
 | 
			
		||||
    <link rel="icon" href="/favicon.ico" />
 | 
			
		||||
    <script>
 | 
			
		||||
      window.process = {};
 | 
			
		||||
 | 
			
		||||
@ -46,7 +46,9 @@
 | 
			
		||||
    "nprogress": "^0.2.0",
 | 
			
		||||
    "path": "^0.12.7",
 | 
			
		||||
    "pinia": "^2.1.3",
 | 
			
		||||
    "qrcode": "^1.5.3",
 | 
			
		||||
    "qs": "^6.11.2",
 | 
			
		||||
    "typeit": "^8.7.1",
 | 
			
		||||
    "responsive-storage": "^2.2.0",
 | 
			
		||||
    "sortablejs": "^1.15.0",
 | 
			
		||||
    "vue": "^3.3.4",
 | 
			
		||||
 | 
			
		||||
@ -1,6 +1,6 @@
 | 
			
		||||
{
 | 
			
		||||
  "Version": "4.3.0",
 | 
			
		||||
  "Title": "PureAdmin",
 | 
			
		||||
  "Title": "Agileboot",
 | 
			
		||||
  "FixedHeader": true,
 | 
			
		||||
  "HiddenSideBar": false,
 | 
			
		||||
  "MultiTagsCache": false,
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										7
									
								
								src/components/ReImageVerify/index.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								src/components/ReImageVerify/index.ts
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,7 @@
 | 
			
		||||
import reImageVerify from "./src/index.vue";
 | 
			
		||||
import { withInstall } from "@pureadmin/utils";
 | 
			
		||||
 | 
			
		||||
/** 图形验证码组件 */
 | 
			
		||||
export const ReImageVerify = withInstall(reImageVerify);
 | 
			
		||||
 | 
			
		||||
export default ReImageVerify;
 | 
			
		||||
							
								
								
									
										85
									
								
								src/components/ReImageVerify/src/hooks.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										85
									
								
								src/components/ReImageVerify/src/hooks.ts
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,85 @@
 | 
			
		||||
import { ref, onMounted } from "vue";
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * 绘制图形验证码
 | 
			
		||||
 * @param width - 图形宽度
 | 
			
		||||
 * @param height - 图形高度
 | 
			
		||||
 */
 | 
			
		||||
export const useImageVerify = (width = 120, height = 40) => {
 | 
			
		||||
  const domRef = ref<HTMLCanvasElement>();
 | 
			
		||||
  const imgCode = ref("");
 | 
			
		||||
 | 
			
		||||
  function setImgCode(code: string) {
 | 
			
		||||
    imgCode.value = code;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  function getImgCode() {
 | 
			
		||||
    if (!domRef.value) return;
 | 
			
		||||
    imgCode.value = draw(domRef.value, width, height);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  onMounted(() => {
 | 
			
		||||
    getImgCode();
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  return {
 | 
			
		||||
    domRef,
 | 
			
		||||
    imgCode,
 | 
			
		||||
    setImgCode,
 | 
			
		||||
    getImgCode
 | 
			
		||||
  };
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
function randomNum(min: number, max: number) {
 | 
			
		||||
  const num = Math.floor(Math.random() * (max - min) + min);
 | 
			
		||||
  return num;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function randomColor(min: number, max: number) {
 | 
			
		||||
  const r = randomNum(min, max);
 | 
			
		||||
  const g = randomNum(min, max);
 | 
			
		||||
  const b = randomNum(min, max);
 | 
			
		||||
  return `rgb(${r},${g},${b})`;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function draw(dom: HTMLCanvasElement, width: number, height: number) {
 | 
			
		||||
  let imgCode = "";
 | 
			
		||||
 | 
			
		||||
  const NUMBER_STRING = "0123456789";
 | 
			
		||||
 | 
			
		||||
  const ctx = dom.getContext("2d");
 | 
			
		||||
  if (!ctx) return imgCode;
 | 
			
		||||
 | 
			
		||||
  ctx.fillStyle = randomColor(180, 230);
 | 
			
		||||
  ctx.fillRect(0, 0, width, height);
 | 
			
		||||
  for (let i = 0; i < 4; i += 1) {
 | 
			
		||||
    const text = NUMBER_STRING[randomNum(0, NUMBER_STRING.length)];
 | 
			
		||||
    imgCode += text;
 | 
			
		||||
    const fontSize = randomNum(18, 41);
 | 
			
		||||
    const deg = randomNum(-30, 30);
 | 
			
		||||
    ctx.font = `${fontSize}px Simhei`;
 | 
			
		||||
    ctx.textBaseline = "top";
 | 
			
		||||
    ctx.fillStyle = randomColor(80, 150);
 | 
			
		||||
    ctx.save();
 | 
			
		||||
    ctx.translate(30 * i + 15, 15);
 | 
			
		||||
    ctx.rotate((deg * Math.PI) / 180);
 | 
			
		||||
    ctx.fillText(text, -15 + 5, -15);
 | 
			
		||||
    ctx.restore();
 | 
			
		||||
  }
 | 
			
		||||
  for (let i = 0; i < 5; i += 1) {
 | 
			
		||||
    ctx.beginPath();
 | 
			
		||||
    ctx.moveTo(randomNum(0, width), randomNum(0, height));
 | 
			
		||||
    ctx.lineTo(randomNum(0, width), randomNum(0, height));
 | 
			
		||||
    ctx.strokeStyle = randomColor(180, 230);
 | 
			
		||||
    ctx.closePath();
 | 
			
		||||
    ctx.stroke();
 | 
			
		||||
  }
 | 
			
		||||
  for (let i = 0; i < 41; i += 1) {
 | 
			
		||||
    ctx.beginPath();
 | 
			
		||||
    ctx.arc(randomNum(0, width), randomNum(0, height), 1, 0, 2 * Math.PI);
 | 
			
		||||
    ctx.closePath();
 | 
			
		||||
    ctx.fillStyle = randomColor(150, 200);
 | 
			
		||||
    ctx.fill();
 | 
			
		||||
  }
 | 
			
		||||
  return imgCode;
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										46
									
								
								src/components/ReImageVerify/src/index.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								src/components/ReImageVerify/src/index.vue
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,46 @@
 | 
			
		||||
<script setup lang="ts">
 | 
			
		||||
import { watch } from "vue";
 | 
			
		||||
import { useImageVerify } from "./hooks";
 | 
			
		||||
 | 
			
		||||
defineOptions({
 | 
			
		||||
  name: "ReImageVerify"
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
interface Props {
 | 
			
		||||
  code?: string;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
interface Emits {
 | 
			
		||||
  (e: "update:code", code: string): void;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const props = withDefaults(defineProps<Props>(), {
 | 
			
		||||
  code: ""
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
const emit = defineEmits<Emits>();
 | 
			
		||||
 | 
			
		||||
const { domRef, imgCode, setImgCode, getImgCode } = useImageVerify();
 | 
			
		||||
 | 
			
		||||
watch(
 | 
			
		||||
  () => props.code,
 | 
			
		||||
  newValue => {
 | 
			
		||||
    setImgCode(newValue);
 | 
			
		||||
  }
 | 
			
		||||
);
 | 
			
		||||
watch(imgCode, newValue => {
 | 
			
		||||
  emit("update:code", newValue);
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
defineExpose({ getImgCode });
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<template>
 | 
			
		||||
  <canvas
 | 
			
		||||
    ref="domRef"
 | 
			
		||||
    width="120"
 | 
			
		||||
    height="40"
 | 
			
		||||
    class="cursor-pointer"
 | 
			
		||||
    @click="getImgCode"
 | 
			
		||||
  />
 | 
			
		||||
</template>
 | 
			
		||||
							
								
								
									
										7
									
								
								src/components/ReQrcode/index.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								src/components/ReQrcode/index.ts
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,7 @@
 | 
			
		||||
import reQrcode from "./src/index";
 | 
			
		||||
import { withInstall } from "@pureadmin/utils";
 | 
			
		||||
 | 
			
		||||
/** 二维码组件 */
 | 
			
		||||
export const ReQrcode = withInstall(reQrcode);
 | 
			
		||||
 | 
			
		||||
export default ReQrcode;
 | 
			
		||||
							
								
								
									
										9
									
								
								src/components/ReQrcode/src/index.scss
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								src/components/ReQrcode/src/index.scss
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,9 @@
 | 
			
		||||
.qrcode {
 | 
			
		||||
  &--disabled {
 | 
			
		||||
    background: rgb(255 255 255 / 95%);
 | 
			
		||||
 | 
			
		||||
    & > div {
 | 
			
		||||
      transform: translate(-50%, -50%);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										261
									
								
								src/components/ReQrcode/src/index.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										261
									
								
								src/components/ReQrcode/src/index.tsx
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,261 @@
 | 
			
		||||
import {
 | 
			
		||||
  ref,
 | 
			
		||||
  unref,
 | 
			
		||||
  watch,
 | 
			
		||||
  nextTick,
 | 
			
		||||
  computed,
 | 
			
		||||
  PropType,
 | 
			
		||||
  defineComponent
 | 
			
		||||
} from "vue";
 | 
			
		||||
import "./index.scss";
 | 
			
		||||
import propTypes from "@/utils/propTypes";
 | 
			
		||||
import { isString, cloneDeep } from "@pureadmin/utils";
 | 
			
		||||
import QRCode, { QRCodeRenderersOptions } from "qrcode";
 | 
			
		||||
import RefreshRight from "@iconify-icons/ep/refresh-right";
 | 
			
		||||
 | 
			
		||||
interface QrcodeLogo {
 | 
			
		||||
  src?: string;
 | 
			
		||||
  logoSize?: number;
 | 
			
		||||
  bgColor?: string;
 | 
			
		||||
  borderSize?: number;
 | 
			
		||||
  crossOrigin?: string;
 | 
			
		||||
  borderRadius?: number;
 | 
			
		||||
  logoRadius?: number;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const props = {
 | 
			
		||||
  // img 或者 canvas,img不支持logo嵌套
 | 
			
		||||
  tag: propTypes.string
 | 
			
		||||
    .validate((v: string) => ["canvas", "img"].includes(v))
 | 
			
		||||
    .def("canvas"),
 | 
			
		||||
  // 二维码内容
 | 
			
		||||
  text: {
 | 
			
		||||
    type: [String, Array] as PropType<string | Recordable[]>,
 | 
			
		||||
    default: null
 | 
			
		||||
  },
 | 
			
		||||
  // qrcode.js配置项
 | 
			
		||||
  options: {
 | 
			
		||||
    type: Object as PropType<QRCodeRenderersOptions>,
 | 
			
		||||
    default: (): QRCodeRenderersOptions => ({})
 | 
			
		||||
  },
 | 
			
		||||
  // 宽度
 | 
			
		||||
  width: propTypes.number.def(200),
 | 
			
		||||
  // logo
 | 
			
		||||
  logo: {
 | 
			
		||||
    type: [String, Object] as PropType<Partial<QrcodeLogo> | string>,
 | 
			
		||||
    default: (): QrcodeLogo | string => ""
 | 
			
		||||
  },
 | 
			
		||||
  // 是否过期
 | 
			
		||||
  disabled: propTypes.bool.def(false),
 | 
			
		||||
  // 过期提示内容
 | 
			
		||||
  disabledText: propTypes.string.def("")
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
export default defineComponent({
 | 
			
		||||
  name: "ReQrcode",
 | 
			
		||||
  props,
 | 
			
		||||
  emits: ["done", "click", "disabled-click"],
 | 
			
		||||
  setup(props, { emit }) {
 | 
			
		||||
    const { toCanvas, toDataURL } = QRCode;
 | 
			
		||||
    const loading = ref(true);
 | 
			
		||||
    const wrapRef = ref<Nullable<HTMLCanvasElement | HTMLImageElement>>(null);
 | 
			
		||||
    const renderText = computed(() => String(props.text));
 | 
			
		||||
    const wrapStyle = computed(() => {
 | 
			
		||||
      return {
 | 
			
		||||
        width: props.width + "px",
 | 
			
		||||
        height: props.width + "px"
 | 
			
		||||
      };
 | 
			
		||||
    });
 | 
			
		||||
    const initQrcode = async () => {
 | 
			
		||||
      await nextTick();
 | 
			
		||||
      const options = cloneDeep(props.options || {});
 | 
			
		||||
      if (props.tag === "canvas") {
 | 
			
		||||
        // 容错率,默认对内容少的二维码采用高容错率,内容多的二维码采用低容错率
 | 
			
		||||
        options.errorCorrectionLevel =
 | 
			
		||||
          options.errorCorrectionLevel ||
 | 
			
		||||
          getErrorCorrectionLevel(unref(renderText));
 | 
			
		||||
        const _width: number = await getOriginWidth(unref(renderText), options);
 | 
			
		||||
        options.scale =
 | 
			
		||||
          props.width === 0 ? undefined : (props.width / _width) * 4;
 | 
			
		||||
        const canvasRef: any = await toCanvas(
 | 
			
		||||
          unref(wrapRef) as HTMLCanvasElement,
 | 
			
		||||
          unref(renderText),
 | 
			
		||||
          options
 | 
			
		||||
        );
 | 
			
		||||
        if (props.logo) {
 | 
			
		||||
          const url = await createLogoCode(canvasRef);
 | 
			
		||||
          emit("done", url);
 | 
			
		||||
          loading.value = false;
 | 
			
		||||
        } else {
 | 
			
		||||
          emit("done", canvasRef.toDataURL());
 | 
			
		||||
          loading.value = false;
 | 
			
		||||
        }
 | 
			
		||||
      } else {
 | 
			
		||||
        const url = await toDataURL(renderText.value, {
 | 
			
		||||
          errorCorrectionLevel: "H",
 | 
			
		||||
          width: props.width,
 | 
			
		||||
          ...options
 | 
			
		||||
        });
 | 
			
		||||
        (unref(wrapRef) as any).src = url;
 | 
			
		||||
        emit("done", url);
 | 
			
		||||
        loading.value = false;
 | 
			
		||||
      }
 | 
			
		||||
    };
 | 
			
		||||
    watch(
 | 
			
		||||
      () => renderText.value,
 | 
			
		||||
      val => {
 | 
			
		||||
        if (!val) return;
 | 
			
		||||
        initQrcode();
 | 
			
		||||
      },
 | 
			
		||||
      {
 | 
			
		||||
        deep: true,
 | 
			
		||||
        immediate: true
 | 
			
		||||
      }
 | 
			
		||||
    );
 | 
			
		||||
    const createLogoCode = (canvasRef: HTMLCanvasElement) => {
 | 
			
		||||
      const canvasWidth = canvasRef.width;
 | 
			
		||||
      const logoOptions: QrcodeLogo = Object.assign(
 | 
			
		||||
        {
 | 
			
		||||
          logoSize: 0.15,
 | 
			
		||||
          bgColor: "#ffffff",
 | 
			
		||||
          borderSize: 0.05,
 | 
			
		||||
          crossOrigin: "anonymous",
 | 
			
		||||
          borderRadius: 8,
 | 
			
		||||
          logoRadius: 0
 | 
			
		||||
        },
 | 
			
		||||
        isString(props.logo) ? {} : props.logo
 | 
			
		||||
      );
 | 
			
		||||
      const {
 | 
			
		||||
        logoSize = 0.15,
 | 
			
		||||
        bgColor = "#ffffff",
 | 
			
		||||
        borderSize = 0.05,
 | 
			
		||||
        crossOrigin = "anonymous",
 | 
			
		||||
        borderRadius = 8,
 | 
			
		||||
        logoRadius = 0
 | 
			
		||||
      } = logoOptions;
 | 
			
		||||
      const logoSrc = isString(props.logo) ? props.logo : props.logo.src;
 | 
			
		||||
      const logoWidth = canvasWidth * logoSize;
 | 
			
		||||
      const logoXY = (canvasWidth * (1 - logoSize)) / 2;
 | 
			
		||||
      const logoBgWidth = canvasWidth * (logoSize + borderSize);
 | 
			
		||||
      const logoBgXY = (canvasWidth * (1 - logoSize - borderSize)) / 2;
 | 
			
		||||
      const ctx = canvasRef.getContext("2d");
 | 
			
		||||
      if (!ctx) return;
 | 
			
		||||
      // logo 底色
 | 
			
		||||
      canvasRoundRect(ctx)(
 | 
			
		||||
        logoBgXY,
 | 
			
		||||
        logoBgXY,
 | 
			
		||||
        logoBgWidth,
 | 
			
		||||
        logoBgWidth,
 | 
			
		||||
        borderRadius
 | 
			
		||||
      );
 | 
			
		||||
      ctx.fillStyle = bgColor;
 | 
			
		||||
      ctx.fill();
 | 
			
		||||
      // logo
 | 
			
		||||
      const image = new Image();
 | 
			
		||||
      if (crossOrigin || logoRadius) {
 | 
			
		||||
        image.setAttribute("crossOrigin", crossOrigin);
 | 
			
		||||
      }
 | 
			
		||||
      (image as any).src = logoSrc;
 | 
			
		||||
      // 使用image绘制可以避免某些跨域情况
 | 
			
		||||
      const drawLogoWithImage = (image: HTMLImageElement) => {
 | 
			
		||||
        ctx.drawImage(image, logoXY, logoXY, logoWidth, logoWidth);
 | 
			
		||||
      };
 | 
			
		||||
      // 使用canvas绘制以获得更多的功能
 | 
			
		||||
      const drawLogoWithCanvas = (image: HTMLImageElement) => {
 | 
			
		||||
        const canvasImage = document.createElement("canvas");
 | 
			
		||||
        canvasImage.width = logoXY + logoWidth;
 | 
			
		||||
        canvasImage.height = logoXY + logoWidth;
 | 
			
		||||
        const imageCanvas = canvasImage.getContext("2d");
 | 
			
		||||
        if (!imageCanvas || !ctx) return;
 | 
			
		||||
        imageCanvas.drawImage(image, logoXY, logoXY, logoWidth, logoWidth);
 | 
			
		||||
        canvasRoundRect(ctx)(logoXY, logoXY, logoWidth, logoWidth, logoRadius);
 | 
			
		||||
        if (!ctx) return;
 | 
			
		||||
        const fillStyle = ctx.createPattern(canvasImage, "no-repeat");
 | 
			
		||||
        if (fillStyle) {
 | 
			
		||||
          ctx.fillStyle = fillStyle;
 | 
			
		||||
          ctx.fill();
 | 
			
		||||
        }
 | 
			
		||||
      };
 | 
			
		||||
      // 将 logo绘制到 canvas上
 | 
			
		||||
      return new Promise((resolve: any) => {
 | 
			
		||||
        image.onload = () => {
 | 
			
		||||
          logoRadius ? drawLogoWithCanvas(image) : drawLogoWithImage(image);
 | 
			
		||||
          resolve(canvasRef.toDataURL());
 | 
			
		||||
        };
 | 
			
		||||
      });
 | 
			
		||||
    };
 | 
			
		||||
    // 得到原QrCode的大小,以便缩放得到正确的QrCode大小
 | 
			
		||||
    const getOriginWidth = async (
 | 
			
		||||
      content: string,
 | 
			
		||||
      options: QRCodeRenderersOptions
 | 
			
		||||
    ) => {
 | 
			
		||||
      const _canvas = document.createElement("canvas");
 | 
			
		||||
      await toCanvas(_canvas, content, options);
 | 
			
		||||
      return _canvas.width;
 | 
			
		||||
    };
 | 
			
		||||
    // 对于内容少的QrCode,增大容错率
 | 
			
		||||
    const getErrorCorrectionLevel = (content: string) => {
 | 
			
		||||
      if (content.length > 36) {
 | 
			
		||||
        return "M";
 | 
			
		||||
      } else if (content.length > 16) {
 | 
			
		||||
        return "Q";
 | 
			
		||||
      } else {
 | 
			
		||||
        return "H";
 | 
			
		||||
      }
 | 
			
		||||
    };
 | 
			
		||||
    // 用于绘制圆角
 | 
			
		||||
    const canvasRoundRect = (ctx: CanvasRenderingContext2D) => {
 | 
			
		||||
      return (x: number, y: number, w: number, h: number, r: number) => {
 | 
			
		||||
        const minSize = Math.min(w, h);
 | 
			
		||||
        if (r > minSize / 2) {
 | 
			
		||||
          r = minSize / 2;
 | 
			
		||||
        }
 | 
			
		||||
        ctx.beginPath();
 | 
			
		||||
        ctx.moveTo(x + r, y);
 | 
			
		||||
        ctx.arcTo(x + w, y, x + w, y + h, r);
 | 
			
		||||
        ctx.arcTo(x + w, y + h, x, y + h, r);
 | 
			
		||||
        ctx.arcTo(x, y + h, x, y, r);
 | 
			
		||||
        ctx.arcTo(x, y, x + w, y, r);
 | 
			
		||||
        ctx.closePath();
 | 
			
		||||
        return ctx;
 | 
			
		||||
      };
 | 
			
		||||
    };
 | 
			
		||||
    const clickCode = () => {
 | 
			
		||||
      emit("click");
 | 
			
		||||
    };
 | 
			
		||||
    const disabledClick = () => {
 | 
			
		||||
      emit("disabled-click");
 | 
			
		||||
    };
 | 
			
		||||
    return () => (
 | 
			
		||||
      <>
 | 
			
		||||
        <div
 | 
			
		||||
          v-loading={unref(loading)}
 | 
			
		||||
          class="qrcode relative inline-block"
 | 
			
		||||
          style={unref(wrapStyle)}
 | 
			
		||||
        >
 | 
			
		||||
          {props.tag === "canvas" ? (
 | 
			
		||||
            <canvas ref={wrapRef} onClick={clickCode}></canvas>
 | 
			
		||||
          ) : (
 | 
			
		||||
            <img ref={wrapRef} onClick={clickCode}></img>
 | 
			
		||||
          )}
 | 
			
		||||
          {props.disabled && (
 | 
			
		||||
            <div
 | 
			
		||||
              class="qrcode--disabled absolute top-0 left-0 flex w-full h-full items-center justify-center"
 | 
			
		||||
              onClick={disabledClick}
 | 
			
		||||
            >
 | 
			
		||||
              <div class="absolute top-[50%] left-[50%] font-bold">
 | 
			
		||||
                <iconify-icon-offline
 | 
			
		||||
                  class="cursor-pointer"
 | 
			
		||||
                  icon={RefreshRight}
 | 
			
		||||
                  width="30"
 | 
			
		||||
                  color="var(--el-color-primary)"
 | 
			
		||||
                />
 | 
			
		||||
                <div>{props.disabledText}</div>
 | 
			
		||||
              </div>
 | 
			
		||||
            </div>
 | 
			
		||||
          )}
 | 
			
		||||
        </div>
 | 
			
		||||
      </>
 | 
			
		||||
    );
 | 
			
		||||
  }
 | 
			
		||||
});
 | 
			
		||||
							
								
								
									
										44
									
								
								src/components/ReTypeit/index.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								src/components/ReTypeit/index.ts
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,44 @@
 | 
			
		||||
import { h, defineComponent } from "vue";
 | 
			
		||||
import TypeIt from "typeit";
 | 
			
		||||
 | 
			
		||||
// 打字机效果组件(只是简单的封装下,更多配置项参考 https://www.typeitjs.com/docs/vanilla/usage#options)
 | 
			
		||||
export default defineComponent({
 | 
			
		||||
  name: "TypeIt",
 | 
			
		||||
  props: {
 | 
			
		||||
    /** 打字速度,以每一步之间的毫秒数为单位,默认`200` */
 | 
			
		||||
    speed: {
 | 
			
		||||
      type: Number,
 | 
			
		||||
      default: 200
 | 
			
		||||
    },
 | 
			
		||||
    values: {
 | 
			
		||||
      type: Array,
 | 
			
		||||
      defalut: []
 | 
			
		||||
    },
 | 
			
		||||
    className: {
 | 
			
		||||
      type: String,
 | 
			
		||||
      default: "type-it"
 | 
			
		||||
    },
 | 
			
		||||
    cursor: {
 | 
			
		||||
      type: Boolean,
 | 
			
		||||
      default: true
 | 
			
		||||
    }
 | 
			
		||||
  },
 | 
			
		||||
  render() {
 | 
			
		||||
    return h(
 | 
			
		||||
      "span",
 | 
			
		||||
      {
 | 
			
		||||
        class: this.className
 | 
			
		||||
      },
 | 
			
		||||
      {
 | 
			
		||||
        default: () => []
 | 
			
		||||
      }
 | 
			
		||||
    );
 | 
			
		||||
  },
 | 
			
		||||
  mounted() {
 | 
			
		||||
    new TypeIt(`.${this.className}`, {
 | 
			
		||||
      strings: this.values,
 | 
			
		||||
      speed: this.speed,
 | 
			
		||||
      cursor: this.cursor
 | 
			
		||||
    }).go();
 | 
			
		||||
  }
 | 
			
		||||
});
 | 
			
		||||
@ -39,4 +39,5 @@ export type setType = {
 | 
			
		||||
export type userType = {
 | 
			
		||||
  username?: string;
 | 
			
		||||
  roles?: Array<string>;
 | 
			
		||||
  currentPage?: number;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
@ -10,13 +10,15 @@ import { useMultiTagsStoreHook } from "@/store/modules/multiTags";
 | 
			
		||||
import { type DataInfo, setToken, removeToken, sessionKey } from "@/utils/auth";
 | 
			
		||||
 | 
			
		||||
export const useUserStore = defineStore({
 | 
			
		||||
  id: "pure-user",
 | 
			
		||||
  id: "ag-user",
 | 
			
		||||
  state: (): userType => ({
 | 
			
		||||
    // 用户名
 | 
			
		||||
    username:
 | 
			
		||||
      storageSession().getItem<DataInfo<number>>(sessionKey)?.username ?? "",
 | 
			
		||||
    // 页面级别权限
 | 
			
		||||
    roles: storageSession().getItem<DataInfo<number>>(sessionKey)?.roles ?? []
 | 
			
		||||
    roles: storageSession().getItem<DataInfo<number>>(sessionKey)?.roles ?? [],
 | 
			
		||||
    // 判断登录页面显示哪个组件(0:登录(默认)、1:手机登录、2:二维码登录、3:注册、4:忘记密码)
 | 
			
		||||
    currentPage: 0
 | 
			
		||||
  }),
 | 
			
		||||
  actions: {
 | 
			
		||||
    /** 存储用户名 */
 | 
			
		||||
@ -27,6 +29,10 @@ export const useUserStore = defineStore({
 | 
			
		||||
    SET_ROLES(roles: Array<string>) {
 | 
			
		||||
      this.roles = roles;
 | 
			
		||||
    },
 | 
			
		||||
    /** 存储登录页面显示哪个组件 */
 | 
			
		||||
    SET_CURRENTPAGE(value: number) {
 | 
			
		||||
      this.currentPage = value;
 | 
			
		||||
    },
 | 
			
		||||
    /** 登入 */
 | 
			
		||||
    async loginByUsername(data) {
 | 
			
		||||
      return new Promise<UserResult>((resolve, reject) => {
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										98
									
								
								src/views/login/components/phone.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										98
									
								
								src/views/login/components/phone.vue
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,98 @@
 | 
			
		||||
<script setup lang="ts">
 | 
			
		||||
import { ref, reactive } from "vue";
 | 
			
		||||
import Motion from "../utils/motion";
 | 
			
		||||
import { message } from "@/utils/message";
 | 
			
		||||
import { phoneRules } from "../utils/rule";
 | 
			
		||||
import type { FormInstance } from "element-plus";
 | 
			
		||||
import { useVerifyCode } from "../utils/verifyCode";
 | 
			
		||||
import { useUserStoreHook } from "@/store/modules/user";
 | 
			
		||||
import { useRenderIcon } from "@/components/ReIcon/src/hooks";
 | 
			
		||||
import Iphone from "@iconify-icons/ep/iphone";
 | 
			
		||||
 | 
			
		||||
const loading = ref(false);
 | 
			
		||||
const ruleForm = reactive({
 | 
			
		||||
  phone: "",
 | 
			
		||||
  verifyCode: ""
 | 
			
		||||
});
 | 
			
		||||
const ruleFormRef = ref<FormInstance>();
 | 
			
		||||
const { isDisabled, text } = useVerifyCode();
 | 
			
		||||
 | 
			
		||||
const onLogin = async (formEl: FormInstance | undefined) => {
 | 
			
		||||
  loading.value = true;
 | 
			
		||||
  if (!formEl) return;
 | 
			
		||||
  await formEl.validate((valid, fields) => {
 | 
			
		||||
    if (valid) {
 | 
			
		||||
      // 模拟登录请求,需根据实际开发进行修改
 | 
			
		||||
      setTimeout(() => {
 | 
			
		||||
        message("登录成功", { type: "success" });
 | 
			
		||||
        loading.value = false;
 | 
			
		||||
      }, 2000);
 | 
			
		||||
    } else {
 | 
			
		||||
      loading.value = false;
 | 
			
		||||
      return fields;
 | 
			
		||||
    }
 | 
			
		||||
  });
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
function onBack() {
 | 
			
		||||
  useVerifyCode().end();
 | 
			
		||||
  useUserStoreHook().SET_CURRENTPAGE(0);
 | 
			
		||||
}
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<template>
 | 
			
		||||
  <el-form ref="ruleFormRef" :model="ruleForm" :rules="phoneRules" size="large">
 | 
			
		||||
    <Motion>
 | 
			
		||||
      <el-form-item prop="phone">
 | 
			
		||||
        <el-input
 | 
			
		||||
          clearable
 | 
			
		||||
          v-model="ruleForm.phone"
 | 
			
		||||
          placeholder="手机号码"
 | 
			
		||||
          :prefix-icon="useRenderIcon(Iphone)"
 | 
			
		||||
        />
 | 
			
		||||
      </el-form-item>
 | 
			
		||||
    </Motion>
 | 
			
		||||
 | 
			
		||||
    <Motion :delay="100">
 | 
			
		||||
      <el-form-item prop="verifyCode">
 | 
			
		||||
        <div class="flex justify-between w-full">
 | 
			
		||||
          <el-input
 | 
			
		||||
            clearable
 | 
			
		||||
            v-model="ruleForm.verifyCode"
 | 
			
		||||
            placeholder="短信验证码"
 | 
			
		||||
            :prefix-icon="useRenderIcon('ri:shield-keyhole-line')"
 | 
			
		||||
          />
 | 
			
		||||
          <el-button
 | 
			
		||||
            :disabled="isDisabled"
 | 
			
		||||
            class="ml-2"
 | 
			
		||||
            @click="useVerifyCode().start(ruleFormRef, 'phone')"
 | 
			
		||||
          >
 | 
			
		||||
            {{ text.length > 0 ? text + "秒后重新获取" : "获取验证码" }}
 | 
			
		||||
          </el-button>
 | 
			
		||||
        </div>
 | 
			
		||||
      </el-form-item>
 | 
			
		||||
    </Motion>
 | 
			
		||||
 | 
			
		||||
    <Motion :delay="150">
 | 
			
		||||
      <el-form-item>
 | 
			
		||||
        <el-button
 | 
			
		||||
          class="w-full"
 | 
			
		||||
          size="default"
 | 
			
		||||
          type="primary"
 | 
			
		||||
          :loading="loading"
 | 
			
		||||
          @click="onLogin(ruleFormRef)"
 | 
			
		||||
        >
 | 
			
		||||
          {{ "登录" }}
 | 
			
		||||
        </el-button>
 | 
			
		||||
      </el-form-item>
 | 
			
		||||
    </Motion>
 | 
			
		||||
 | 
			
		||||
    <Motion :delay="200">
 | 
			
		||||
      <el-form-item>
 | 
			
		||||
        <el-button class="w-full" size="default" @click="onBack">
 | 
			
		||||
          {{ "返回" }}
 | 
			
		||||
        </el-button>
 | 
			
		||||
      </el-form-item>
 | 
			
		||||
    </Motion>
 | 
			
		||||
  </el-form>
 | 
			
		||||
</template>
 | 
			
		||||
							
								
								
									
										22
									
								
								src/views/login/components/qrCode.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								src/views/login/components/qrCode.vue
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,22 @@
 | 
			
		||||
<script setup lang="ts">
 | 
			
		||||
import Motion from "../utils/motion";
 | 
			
		||||
import ReQrcode from "@/components/ReQrcode";
 | 
			
		||||
import { useUserStoreHook } from "@/store/modules/user";
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<template>
 | 
			
		||||
  <Motion class="-mt-2 -mb-2"> <ReQrcode text="模拟测试" /> </Motion>
 | 
			
		||||
  <Motion :delay="100">
 | 
			
		||||
    <el-divider>
 | 
			
		||||
      <p class="text-xs text-gray-500">{{ '扫码后点击"确认",即可完成登录' }}</p>
 | 
			
		||||
    </el-divider>
 | 
			
		||||
  </Motion>
 | 
			
		||||
  <Motion :delay="150">
 | 
			
		||||
    <el-button
 | 
			
		||||
      class="w-full mt-4"
 | 
			
		||||
      @click="useUserStoreHook().SET_CURRENTPAGE(0)"
 | 
			
		||||
    >
 | 
			
		||||
      {{ "返回" }}
 | 
			
		||||
    </el-button>
 | 
			
		||||
  </Motion>
 | 
			
		||||
</template>
 | 
			
		||||
							
								
								
									
										185
									
								
								src/views/login/components/regist.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										185
									
								
								src/views/login/components/regist.vue
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,185 @@
 | 
			
		||||
<script setup lang="ts">
 | 
			
		||||
import { ref, reactive } from "vue";
 | 
			
		||||
import Motion from "../utils/motion";
 | 
			
		||||
import { message } from "@/utils/message";
 | 
			
		||||
import { updateRules } from "../utils/rule";
 | 
			
		||||
import type { FormInstance } from "element-plus";
 | 
			
		||||
import { useVerifyCode } from "../utils/verifyCode";
 | 
			
		||||
import { useUserStoreHook } from "@/store/modules/user";
 | 
			
		||||
import { useRenderIcon } from "@/components/ReIcon/src/hooks";
 | 
			
		||||
import Lock from "@iconify-icons/ri/lock-fill";
 | 
			
		||||
import Iphone from "@iconify-icons/ep/iphone";
 | 
			
		||||
import User from "@iconify-icons/ri/user-3-fill";
 | 
			
		||||
 | 
			
		||||
const checked = ref(false);
 | 
			
		||||
const loading = ref(false);
 | 
			
		||||
const ruleForm = reactive({
 | 
			
		||||
  username: "",
 | 
			
		||||
  phone: "",
 | 
			
		||||
  verifyCode: "",
 | 
			
		||||
  password: "",
 | 
			
		||||
  repeatPassword: ""
 | 
			
		||||
});
 | 
			
		||||
const ruleFormRef = ref<FormInstance>();
 | 
			
		||||
const { isDisabled, text } = useVerifyCode();
 | 
			
		||||
const repeatPasswordRule = [
 | 
			
		||||
  {
 | 
			
		||||
    validator: (rule, value, callback) => {
 | 
			
		||||
      if (value === "") {
 | 
			
		||||
        callback(new Error("请输入确认密码"));
 | 
			
		||||
      } else if (ruleForm.password !== value) {
 | 
			
		||||
        callback(new Error("两次密码不一致"));
 | 
			
		||||
      } else {
 | 
			
		||||
        callback();
 | 
			
		||||
      }
 | 
			
		||||
    },
 | 
			
		||||
    trigger: "blur"
 | 
			
		||||
  }
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
const onUpdate = async (formEl: FormInstance | undefined) => {
 | 
			
		||||
  loading.value = true;
 | 
			
		||||
  if (!formEl) return;
 | 
			
		||||
  await formEl.validate((valid, fields) => {
 | 
			
		||||
    if (valid) {
 | 
			
		||||
      if (checked.value) {
 | 
			
		||||
        // 模拟请求,需根据实际开发进行修改
 | 
			
		||||
        setTimeout(() => {
 | 
			
		||||
          message("注册成功", {
 | 
			
		||||
            type: "success"
 | 
			
		||||
          });
 | 
			
		||||
          loading.value = false;
 | 
			
		||||
        }, 2000);
 | 
			
		||||
      } else {
 | 
			
		||||
        loading.value = false;
 | 
			
		||||
        message("请勾选隐私政策", { type: "warning" });
 | 
			
		||||
      }
 | 
			
		||||
    } else {
 | 
			
		||||
      loading.value = false;
 | 
			
		||||
      return fields;
 | 
			
		||||
    }
 | 
			
		||||
  });
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
function onBack() {
 | 
			
		||||
  useVerifyCode().end();
 | 
			
		||||
  useUserStoreHook().SET_CURRENTPAGE(0);
 | 
			
		||||
}
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<template>
 | 
			
		||||
  <el-form
 | 
			
		||||
    ref="ruleFormRef"
 | 
			
		||||
    :model="ruleForm"
 | 
			
		||||
    :rules="updateRules"
 | 
			
		||||
    size="large"
 | 
			
		||||
  >
 | 
			
		||||
    <Motion>
 | 
			
		||||
      <el-form-item
 | 
			
		||||
        :rules="[
 | 
			
		||||
          {
 | 
			
		||||
            required: true,
 | 
			
		||||
            message: '请输入账号',
 | 
			
		||||
            trigger: 'blur'
 | 
			
		||||
          }
 | 
			
		||||
        ]"
 | 
			
		||||
        prop="username"
 | 
			
		||||
      >
 | 
			
		||||
        <el-input
 | 
			
		||||
          clearable
 | 
			
		||||
          v-model="ruleForm.username"
 | 
			
		||||
          placeholder="账号"
 | 
			
		||||
          :prefix-icon="useRenderIcon(User)"
 | 
			
		||||
        />
 | 
			
		||||
      </el-form-item>
 | 
			
		||||
    </Motion>
 | 
			
		||||
 | 
			
		||||
    <Motion :delay="100">
 | 
			
		||||
      <el-form-item prop="phone">
 | 
			
		||||
        <el-input
 | 
			
		||||
          clearable
 | 
			
		||||
          v-model="ruleForm.phone"
 | 
			
		||||
          placeholder="手机号码"
 | 
			
		||||
          :prefix-icon="useRenderIcon(Iphone)"
 | 
			
		||||
        />
 | 
			
		||||
      </el-form-item>
 | 
			
		||||
    </Motion>
 | 
			
		||||
 | 
			
		||||
    <Motion :delay="150">
 | 
			
		||||
      <el-form-item prop="verifyCode">
 | 
			
		||||
        <div class="flex justify-between w-full">
 | 
			
		||||
          <el-input
 | 
			
		||||
            clearable
 | 
			
		||||
            v-model="ruleForm.verifyCode"
 | 
			
		||||
            placeholder="短信验证码"
 | 
			
		||||
            :prefix-icon="useRenderIcon('ri:shield-keyhole-line')"
 | 
			
		||||
          />
 | 
			
		||||
          <el-button
 | 
			
		||||
            :disabled="isDisabled"
 | 
			
		||||
            class="ml-2"
 | 
			
		||||
            @click="useVerifyCode().start(ruleFormRef, 'phone')"
 | 
			
		||||
          >
 | 
			
		||||
            {{ text.length > 0 ? text + "秒后重新获取" : "获取验证码" }}
 | 
			
		||||
          </el-button>
 | 
			
		||||
        </div>
 | 
			
		||||
      </el-form-item>
 | 
			
		||||
    </Motion>
 | 
			
		||||
 | 
			
		||||
    <Motion :delay="200">
 | 
			
		||||
      <el-form-item prop="password">
 | 
			
		||||
        <el-input
 | 
			
		||||
          clearable
 | 
			
		||||
          show-password
 | 
			
		||||
          v-model="ruleForm.password"
 | 
			
		||||
          placeholder="密码"
 | 
			
		||||
          :prefix-icon="useRenderIcon(Lock)"
 | 
			
		||||
        />
 | 
			
		||||
      </el-form-item>
 | 
			
		||||
    </Motion>
 | 
			
		||||
 | 
			
		||||
    <Motion :delay="250">
 | 
			
		||||
      <el-form-item :rules="repeatPasswordRule" prop="repeatPassword">
 | 
			
		||||
        <el-input
 | 
			
		||||
          clearable
 | 
			
		||||
          show-password
 | 
			
		||||
          v-model="ruleForm.repeatPassword"
 | 
			
		||||
          placeholder="确认密码"
 | 
			
		||||
          :prefix-icon="useRenderIcon(Lock)"
 | 
			
		||||
        />
 | 
			
		||||
      </el-form-item>
 | 
			
		||||
    </Motion>
 | 
			
		||||
 | 
			
		||||
    <Motion :delay="300">
 | 
			
		||||
      <el-form-item>
 | 
			
		||||
        <el-checkbox v-model="checked">
 | 
			
		||||
          {{ "我已仔细阅读并接受" }}
 | 
			
		||||
        </el-checkbox>
 | 
			
		||||
        <el-button link type="primary">
 | 
			
		||||
          {{ "隐私政策" }}
 | 
			
		||||
        </el-button>
 | 
			
		||||
      </el-form-item>
 | 
			
		||||
    </Motion>
 | 
			
		||||
 | 
			
		||||
    <Motion :delay="350">
 | 
			
		||||
      <el-form-item>
 | 
			
		||||
        <el-button
 | 
			
		||||
          class="w-full"
 | 
			
		||||
          size="default"
 | 
			
		||||
          type="primary"
 | 
			
		||||
          :loading="loading"
 | 
			
		||||
          @click="onUpdate(ruleFormRef)"
 | 
			
		||||
        >
 | 
			
		||||
          {{ "确定" }}
 | 
			
		||||
        </el-button>
 | 
			
		||||
      </el-form-item>
 | 
			
		||||
    </Motion>
 | 
			
		||||
 | 
			
		||||
    <Motion :delay="400">
 | 
			
		||||
      <el-form-item>
 | 
			
		||||
        <el-button class="w-full" size="default" @click="onBack">
 | 
			
		||||
          {{ "返回" }}
 | 
			
		||||
        </el-button>
 | 
			
		||||
      </el-form-item>
 | 
			
		||||
    </Motion>
 | 
			
		||||
  </el-form>
 | 
			
		||||
</template>
 | 
			
		||||
							
								
								
									
										146
									
								
								src/views/login/components/update.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										146
									
								
								src/views/login/components/update.vue
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,146 @@
 | 
			
		||||
<script setup lang="ts">
 | 
			
		||||
import { ref, reactive } from "vue";
 | 
			
		||||
import Motion from "../utils/motion";
 | 
			
		||||
import { message } from "@/utils/message";
 | 
			
		||||
import { updateRules } from "../utils/rule";
 | 
			
		||||
import type { FormInstance } from "element-plus";
 | 
			
		||||
import { useVerifyCode } from "../utils/verifyCode";
 | 
			
		||||
import { useUserStoreHook } from "@/store/modules/user";
 | 
			
		||||
import { useRenderIcon } from "@/components/ReIcon/src/hooks";
 | 
			
		||||
import Lock from "@iconify-icons/ri/lock-fill";
 | 
			
		||||
import Iphone from "@iconify-icons/ep/iphone";
 | 
			
		||||
 | 
			
		||||
const loading = ref(false);
 | 
			
		||||
const ruleForm = reactive({
 | 
			
		||||
  phone: "",
 | 
			
		||||
  verifyCode: "",
 | 
			
		||||
  password: "",
 | 
			
		||||
  repeatPassword: ""
 | 
			
		||||
});
 | 
			
		||||
const ruleFormRef = ref<FormInstance>();
 | 
			
		||||
const { isDisabled, text } = useVerifyCode();
 | 
			
		||||
const repeatPasswordRule = [
 | 
			
		||||
  {
 | 
			
		||||
    validator: (rule, value, callback) => {
 | 
			
		||||
      if (value === "") {
 | 
			
		||||
        callback(new Error("请输入确认密码"));
 | 
			
		||||
      } else if (ruleForm.password !== value) {
 | 
			
		||||
        callback(new Error("两次密码不一致"));
 | 
			
		||||
      } else {
 | 
			
		||||
        callback();
 | 
			
		||||
      }
 | 
			
		||||
    },
 | 
			
		||||
    trigger: "blur"
 | 
			
		||||
  }
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
const onUpdate = async (formEl: FormInstance | undefined) => {
 | 
			
		||||
  loading.value = true;
 | 
			
		||||
  if (!formEl) return;
 | 
			
		||||
  await formEl.validate((valid, fields) => {
 | 
			
		||||
    if (valid) {
 | 
			
		||||
      // 模拟请求,需根据实际开发进行修改
 | 
			
		||||
      setTimeout(() => {
 | 
			
		||||
        message("修改密码成功", {
 | 
			
		||||
          type: "success"
 | 
			
		||||
        });
 | 
			
		||||
        loading.value = false;
 | 
			
		||||
      }, 2000);
 | 
			
		||||
    } else {
 | 
			
		||||
      loading.value = false;
 | 
			
		||||
      return fields;
 | 
			
		||||
    }
 | 
			
		||||
  });
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
function onBack() {
 | 
			
		||||
  useVerifyCode().end();
 | 
			
		||||
  useUserStoreHook().SET_CURRENTPAGE(0);
 | 
			
		||||
}
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<template>
 | 
			
		||||
  <el-form
 | 
			
		||||
    ref="ruleFormRef"
 | 
			
		||||
    :model="ruleForm"
 | 
			
		||||
    :rules="updateRules"
 | 
			
		||||
    size="large"
 | 
			
		||||
  >
 | 
			
		||||
    <Motion>
 | 
			
		||||
      <el-form-item prop="phone">
 | 
			
		||||
        <el-input
 | 
			
		||||
          clearable
 | 
			
		||||
          v-model="ruleForm.phone"
 | 
			
		||||
          placeholder="手机号码"
 | 
			
		||||
          :prefix-icon="useRenderIcon(Iphone)"
 | 
			
		||||
        />
 | 
			
		||||
      </el-form-item>
 | 
			
		||||
    </Motion>
 | 
			
		||||
 | 
			
		||||
    <Motion :delay="100">
 | 
			
		||||
      <el-form-item prop="verifyCode">
 | 
			
		||||
        <div class="flex justify-between w-full">
 | 
			
		||||
          <el-input
 | 
			
		||||
            clearable
 | 
			
		||||
            v-model="ruleForm.verifyCode"
 | 
			
		||||
            placeholder="短信验证码"
 | 
			
		||||
            :prefix-icon="useRenderIcon('ri:shield-keyhole-line')"
 | 
			
		||||
          />
 | 
			
		||||
          <el-button
 | 
			
		||||
            :disabled="isDisabled"
 | 
			
		||||
            class="ml-2"
 | 
			
		||||
            @click="useVerifyCode().start(ruleFormRef, 'phone')"
 | 
			
		||||
          >
 | 
			
		||||
            {{ text.length > 0 ? text + "秒后重新获取" : "获取验证码" }}
 | 
			
		||||
          </el-button>
 | 
			
		||||
        </div>
 | 
			
		||||
      </el-form-item>
 | 
			
		||||
    </Motion>
 | 
			
		||||
 | 
			
		||||
    <Motion :delay="150">
 | 
			
		||||
      <el-form-item prop="password">
 | 
			
		||||
        <el-input
 | 
			
		||||
          clearable
 | 
			
		||||
          show-password
 | 
			
		||||
          v-model="ruleForm.password"
 | 
			
		||||
          placeholder="密码"
 | 
			
		||||
          :prefix-icon="useRenderIcon(Lock)"
 | 
			
		||||
        />
 | 
			
		||||
      </el-form-item>
 | 
			
		||||
    </Motion>
 | 
			
		||||
 | 
			
		||||
    <Motion :delay="200">
 | 
			
		||||
      <el-form-item :rules="repeatPasswordRule" prop="repeatPassword">
 | 
			
		||||
        <el-input
 | 
			
		||||
          clearable
 | 
			
		||||
          show-password
 | 
			
		||||
          v-model="ruleForm.repeatPassword"
 | 
			
		||||
          placeholder="确认密码"
 | 
			
		||||
          :prefix-icon="useRenderIcon(Lock)"
 | 
			
		||||
        />
 | 
			
		||||
      </el-form-item>
 | 
			
		||||
    </Motion>
 | 
			
		||||
 | 
			
		||||
    <Motion :delay="250">
 | 
			
		||||
      <el-form-item>
 | 
			
		||||
        <el-button
 | 
			
		||||
          class="w-full"
 | 
			
		||||
          size="default"
 | 
			
		||||
          type="primary"
 | 
			
		||||
          :loading="loading"
 | 
			
		||||
          @click="onUpdate(ruleFormRef)"
 | 
			
		||||
        >
 | 
			
		||||
          {{ "确定" }}
 | 
			
		||||
        </el-button>
 | 
			
		||||
      </el-form-item>
 | 
			
		||||
    </Motion>
 | 
			
		||||
 | 
			
		||||
    <Motion :delay="300">
 | 
			
		||||
      <el-form-item>
 | 
			
		||||
        <el-button class="w-full" size="default" @click="onBack">
 | 
			
		||||
          {{ "返回" }}
 | 
			
		||||
        </el-button>
 | 
			
		||||
      </el-form-item>
 | 
			
		||||
    </Motion>
 | 
			
		||||
  </el-form>
 | 
			
		||||
</template>
 | 
			
		||||
@ -1,16 +1,30 @@
 | 
			
		||||
<script setup lang="ts">
 | 
			
		||||
import {
 | 
			
		||||
  ref,
 | 
			
		||||
  toRaw,
 | 
			
		||||
  reactive,
 | 
			
		||||
  computed,
 | 
			
		||||
  onMounted,
 | 
			
		||||
  onBeforeUnmount
 | 
			
		||||
} from "vue";
 | 
			
		||||
import Motion from "./utils/motion";
 | 
			
		||||
import { useRouter } from "vue-router";
 | 
			
		||||
import { message } from "@/utils/message";
 | 
			
		||||
import { loginRules } from "./utils/rule";
 | 
			
		||||
import phone from "./components/phone.vue";
 | 
			
		||||
import TypeIt from "@/components/ReTypeit";
 | 
			
		||||
import qrCode from "./components/qrCode.vue";
 | 
			
		||||
import regist from "./components/regist.vue";
 | 
			
		||||
import update from "./components/update.vue";
 | 
			
		||||
import { useNav } from "@/layout/hooks/useNav";
 | 
			
		||||
import type { FormInstance } from "element-plus";
 | 
			
		||||
import { operates, thirdParty } from "./utils/enums";
 | 
			
		||||
import { useLayout } from "@/layout/hooks/useLayout";
 | 
			
		||||
import { useUserStoreHook } from "@/store/modules/user";
 | 
			
		||||
import { initRouter, getTopMenu } from "@/router/utils";
 | 
			
		||||
import { bg, avatar, illustration } from "./utils/static";
 | 
			
		||||
import { ReImageVerify } from "@/components/ReImageVerify";
 | 
			
		||||
import { useRenderIcon } from "@/components/ReIcon/src/hooks";
 | 
			
		||||
import { ref, reactive, toRaw, onMounted, onBeforeUnmount } from "vue";
 | 
			
		||||
import { useDataThemeChange } from "@/layout/hooks/useDataThemeChange";
 | 
			
		||||
 | 
			
		||||
import dayIcon from "@/assets/svg/day.svg?component";
 | 
			
		||||
@ -21,20 +35,27 @@ import User from "@iconify-icons/ri/user-3-fill";
 | 
			
		||||
defineOptions({
 | 
			
		||||
  name: "Login"
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
const imgCode = ref("");
 | 
			
		||||
const router = useRouter();
 | 
			
		||||
const loading = ref(false);
 | 
			
		||||
const checked = ref(false);
 | 
			
		||||
const ruleFormRef = ref<FormInstance>();
 | 
			
		||||
const currentPage = computed(() => {
 | 
			
		||||
  return useUserStoreHook().currentPage;
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
const { initStorage } = useLayout();
 | 
			
		||||
initStorage();
 | 
			
		||||
 | 
			
		||||
const { dataTheme, dataThemeChange } = useDataThemeChange();
 | 
			
		||||
dataThemeChange();
 | 
			
		||||
// const { title, getDropdownItemStyle, getDropdownItemClass } = useNav();
 | 
			
		||||
const { title } = useNav();
 | 
			
		||||
 | 
			
		||||
const ruleForm = reactive({
 | 
			
		||||
  username: "admin",
 | 
			
		||||
  password: "admin123"
 | 
			
		||||
  password: "admin123",
 | 
			
		||||
  verifyCode: ""
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
const onLogin = async (formEl: FormInstance | undefined) => {
 | 
			
		||||
@ -43,10 +64,7 @@ const onLogin = async (formEl: FormInstance | undefined) => {
 | 
			
		||||
  await formEl.validate((valid, fields) => {
 | 
			
		||||
    if (valid) {
 | 
			
		||||
      useUserStoreHook()
 | 
			
		||||
        .loginByUsername({
 | 
			
		||||
          username: ruleForm.username,
 | 
			
		||||
          password: ruleForm.password
 | 
			
		||||
        })
 | 
			
		||||
        .loginByUsername({ username: ruleForm.username, password: "admin123" })
 | 
			
		||||
        .then(res => {
 | 
			
		||||
          if (res.success) {
 | 
			
		||||
            // 获取后端路由
 | 
			
		||||
@ -77,12 +95,16 @@ onMounted(() => {
 | 
			
		||||
onBeforeUnmount(() => {
 | 
			
		||||
  window.document.removeEventListener("keypress", onkeypress);
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
// watch(imgCode, value => {
 | 
			
		||||
//   useUserStoreHook().SET_VERIFYCODE(value);
 | 
			
		||||
// });
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<template>
 | 
			
		||||
  <div class="select-none">
 | 
			
		||||
    <img :src="bg" class="wave" />
 | 
			
		||||
    <div class="flex-c absolute right-5 top-3">
 | 
			
		||||
    <div class="absolute flex-c right-5 top-3">
 | 
			
		||||
      <!-- 主题 -->
 | 
			
		||||
      <el-switch
 | 
			
		||||
        v-model="dataTheme"
 | 
			
		||||
@ -94,16 +116,21 @@ onBeforeUnmount(() => {
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class="login-container">
 | 
			
		||||
      <div class="img">
 | 
			
		||||
        <!-- 登录页面的背景图 -->
 | 
			
		||||
        <component :is="toRaw(illustration)" />
 | 
			
		||||
      </div>
 | 
			
		||||
      <div class="login-box">
 | 
			
		||||
        <div class="login-form">
 | 
			
		||||
          <!-- 登录窗口上面的LOGO -->
 | 
			
		||||
          <avatar class="avatar" />
 | 
			
		||||
          <Motion>
 | 
			
		||||
            <h2 class="outline-none">{{ title }}</h2>
 | 
			
		||||
            <h2 class="outline-none">
 | 
			
		||||
              <TypeIt :values="[title]" :cursor="false" :speed="150" />
 | 
			
		||||
            </h2>
 | 
			
		||||
          </Motion>
 | 
			
		||||
 | 
			
		||||
          <el-form
 | 
			
		||||
            v-if="currentPage === 0"
 | 
			
		||||
            ref="ruleFormRef"
 | 
			
		||||
            :model="ruleForm"
 | 
			
		||||
            :rules="loginRules"
 | 
			
		||||
@ -141,18 +168,92 @@ onBeforeUnmount(() => {
 | 
			
		||||
              </el-form-item>
 | 
			
		||||
            </Motion>
 | 
			
		||||
 | 
			
		||||
            <Motion :delay="200">
 | 
			
		||||
              <el-form-item prop="verifyCode">
 | 
			
		||||
                <el-input
 | 
			
		||||
                  clearable
 | 
			
		||||
                  v-model="ruleForm.verifyCode"
 | 
			
		||||
                  placeholder="验证码"
 | 
			
		||||
                  :prefix-icon="useRenderIcon('ri:shield-keyhole-line')"
 | 
			
		||||
                >
 | 
			
		||||
                  <template v-slot:append>
 | 
			
		||||
                    <ReImageVerify v-model:code="imgCode" />
 | 
			
		||||
                  </template>
 | 
			
		||||
                </el-input>
 | 
			
		||||
              </el-form-item>
 | 
			
		||||
            </Motion>
 | 
			
		||||
 | 
			
		||||
            <Motion :delay="250">
 | 
			
		||||
              <el-button
 | 
			
		||||
                class="w-full mt-4"
 | 
			
		||||
                size="default"
 | 
			
		||||
                type="primary"
 | 
			
		||||
                :loading="loading"
 | 
			
		||||
                @click="onLogin(ruleFormRef)"
 | 
			
		||||
              >
 | 
			
		||||
                登录
 | 
			
		||||
              </el-button>
 | 
			
		||||
              <el-form-item>
 | 
			
		||||
                <div class="w-full h-[20px] flex justify-between items-center">
 | 
			
		||||
                  <el-checkbox v-model="checked">
 | 
			
		||||
                    {{ "记住密码" }}
 | 
			
		||||
                  </el-checkbox>
 | 
			
		||||
                  <el-button
 | 
			
		||||
                    link
 | 
			
		||||
                    type="primary"
 | 
			
		||||
                    @click="useUserStoreHook().SET_CURRENTPAGE(4)"
 | 
			
		||||
                  >
 | 
			
		||||
                    {{ "忘记密码" }}
 | 
			
		||||
                  </el-button>
 | 
			
		||||
                </div>
 | 
			
		||||
                <el-button
 | 
			
		||||
                  class="w-full mt-4"
 | 
			
		||||
                  size="default"
 | 
			
		||||
                  type="primary"
 | 
			
		||||
                  :loading="loading"
 | 
			
		||||
                  @click="onLogin(ruleFormRef)"
 | 
			
		||||
                >
 | 
			
		||||
                  {{ "登录" }}
 | 
			
		||||
                </el-button>
 | 
			
		||||
              </el-form-item>
 | 
			
		||||
            </Motion>
 | 
			
		||||
 | 
			
		||||
            <Motion :delay="300">
 | 
			
		||||
              <el-form-item>
 | 
			
		||||
                <div class="w-full h-[20px] flex justify-between items-center">
 | 
			
		||||
                  <el-button
 | 
			
		||||
                    v-for="(item, index) in operates"
 | 
			
		||||
                    :key="index"
 | 
			
		||||
                    class="w-full mt-4"
 | 
			
		||||
                    size="default"
 | 
			
		||||
                    @click="useUserStoreHook().SET_CURRENTPAGE(index + 1)"
 | 
			
		||||
                  >
 | 
			
		||||
                    {{ item.title }}
 | 
			
		||||
                  </el-button>
 | 
			
		||||
                </div>
 | 
			
		||||
              </el-form-item>
 | 
			
		||||
            </Motion>
 | 
			
		||||
          </el-form>
 | 
			
		||||
 | 
			
		||||
          <Motion v-if="currentPage === 0" :delay="350">
 | 
			
		||||
            <el-form-item>
 | 
			
		||||
              <el-divider>
 | 
			
		||||
                <p class="text-xs text-gray-500">{{ "第三方登录" }}</p>
 | 
			
		||||
              </el-divider>
 | 
			
		||||
              <div class="flex w-full justify-evenly">
 | 
			
		||||
                <span
 | 
			
		||||
                  v-for="(item, index) in thirdParty"
 | 
			
		||||
                  :key="index"
 | 
			
		||||
                  :title="item.title"
 | 
			
		||||
                >
 | 
			
		||||
                  <IconifyIconOnline
 | 
			
		||||
                    :icon="`ri:${item.icon}-fill`"
 | 
			
		||||
                    width="20"
 | 
			
		||||
                    class="text-gray-500 cursor-pointer hover:text-blue-400"
 | 
			
		||||
                  />
 | 
			
		||||
                </span>
 | 
			
		||||
              </div>
 | 
			
		||||
            </el-form-item>
 | 
			
		||||
          </Motion>
 | 
			
		||||
          <!-- 手机号登录 -->
 | 
			
		||||
          <phone v-if="currentPage === 1" />
 | 
			
		||||
          <!-- 二维码登录 -->
 | 
			
		||||
          <qrCode v-if="currentPage === 2" />
 | 
			
		||||
          <!-- 注册 -->
 | 
			
		||||
          <regist v-if="currentPage === 3" />
 | 
			
		||||
          <!-- 忘记密码 -->
 | 
			
		||||
          <update v-if="currentPage === 4" />
 | 
			
		||||
        </div>
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
@ -167,4 +268,20 @@ onBeforeUnmount(() => {
 | 
			
		||||
:deep(.el-input-group__append, .el-input-group__prepend) {
 | 
			
		||||
  padding: 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.translation {
 | 
			
		||||
  ::v-deep(.el-dropdown-menu__item) {
 | 
			
		||||
    padding: 5px 40px;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .check-zh {
 | 
			
		||||
    position: absolute;
 | 
			
		||||
    left: 20px;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .check-en {
 | 
			
		||||
    position: absolute;
 | 
			
		||||
    left: 20px;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
</style>
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										32
									
								
								src/views/login/utils/enums.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								src/views/login/utils/enums.ts
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,32 @@
 | 
			
		||||
const operates = [
 | 
			
		||||
  {
 | 
			
		||||
    title: "手机登录"
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    title: "二维码登录"
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    title: "注册"
 | 
			
		||||
  }
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
const thirdParty = [
 | 
			
		||||
  {
 | 
			
		||||
    title: "微信登录",
 | 
			
		||||
    icon: "wechat"
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    title: "支付宝登录",
 | 
			
		||||
    icon: "alipay"
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    title: "QQ登录",
 | 
			
		||||
    icon: "qq"
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    title: "微博登录",
 | 
			
		||||
    icon: "weibo"
 | 
			
		||||
  }
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
export { operates, thirdParty };
 | 
			
		||||
@ -1,12 +1,111 @@
 | 
			
		||||
import { reactive } from "vue";
 | 
			
		||||
import { isPhone } from "@pureadmin/utils";
 | 
			
		||||
import type { FormRules } from "element-plus";
 | 
			
		||||
import { useUserStoreHook } from "@/store/modules/user";
 | 
			
		||||
 | 
			
		||||
/** 6位数字验证码正则 */
 | 
			
		||||
export const REGEXP_SIX = /^\d{6}$/;
 | 
			
		||||
 | 
			
		||||
/** 密码正则(密码格式应为8-18位数字、字母、符号的任意两种组合) */
 | 
			
		||||
export const REGEXP_PWD =
 | 
			
		||||
  /^(?![0-9]+$)(?![a-z]+$)(?![A-Z]+$)(?!([^(0-9a-zA-Z)]|[()])+$)(?!^.*[\u4E00-\u9FA5].*$)([^(0-9a-zA-Z)]|[()]|[a-z]|[A-Z]|[0-9]){8,18}$/;
 | 
			
		||||
 | 
			
		||||
/** 登录校验 */
 | 
			
		||||
const loginRules = reactive(<FormRules>{
 | 
			
		||||
const loginRules = reactive<FormRules>({
 | 
			
		||||
  password: [
 | 
			
		||||
    {
 | 
			
		||||
      validator: (rule, value, callback) => {
 | 
			
		||||
        if (value === "") {
 | 
			
		||||
          callback(new Error("请输入密码"));
 | 
			
		||||
        } else if (!REGEXP_PWD.test(value)) {
 | 
			
		||||
          callback(
 | 
			
		||||
            new Error("密码格式应为8-18位数字、字母、符号的任意两种组合")
 | 
			
		||||
          );
 | 
			
		||||
        } else {
 | 
			
		||||
          callback();
 | 
			
		||||
        }
 | 
			
		||||
      },
 | 
			
		||||
      trigger: "blur"
 | 
			
		||||
    }
 | 
			
		||||
  ],
 | 
			
		||||
  verifyCode: [
 | 
			
		||||
    {
 | 
			
		||||
      validator: (rule, value, callback) => {
 | 
			
		||||
        if (value === "") {
 | 
			
		||||
          callback(new Error("请输入验证码"));
 | 
			
		||||
        } else if (useUserStoreHook().verifyCode !== value) {
 | 
			
		||||
          callback(new Error("请输入正确的验证码"));
 | 
			
		||||
        } else {
 | 
			
		||||
          callback();
 | 
			
		||||
        }
 | 
			
		||||
      },
 | 
			
		||||
      trigger: "blur"
 | 
			
		||||
    }
 | 
			
		||||
  ]
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
/** 手机登录校验 */
 | 
			
		||||
const phoneRules = reactive<FormRules>({
 | 
			
		||||
  phone: [
 | 
			
		||||
    {
 | 
			
		||||
      validator: (rule, value, callback) => {
 | 
			
		||||
        if (value === "") {
 | 
			
		||||
          callback(new Error("请输入手机号码"));
 | 
			
		||||
        } else if (!isPhone(value)) {
 | 
			
		||||
          callback(new Error("请输入正确的手机号码格式"));
 | 
			
		||||
        } else {
 | 
			
		||||
          callback();
 | 
			
		||||
        }
 | 
			
		||||
      },
 | 
			
		||||
      trigger: "blur"
 | 
			
		||||
    }
 | 
			
		||||
  ],
 | 
			
		||||
  verifyCode: [
 | 
			
		||||
    {
 | 
			
		||||
      validator: (rule, value, callback) => {
 | 
			
		||||
        if (value === "") {
 | 
			
		||||
          callback(new Error("请输入验证码"));
 | 
			
		||||
        } else if (!REGEXP_SIX.test(value)) {
 | 
			
		||||
          callback(new Error("请输入6位数字验证码"));
 | 
			
		||||
        } else {
 | 
			
		||||
          callback();
 | 
			
		||||
        }
 | 
			
		||||
      },
 | 
			
		||||
      trigger: "blur"
 | 
			
		||||
    }
 | 
			
		||||
  ]
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
/** 忘记密码校验 */
 | 
			
		||||
const updateRules = reactive<FormRules>({
 | 
			
		||||
  phone: [
 | 
			
		||||
    {
 | 
			
		||||
      validator: (rule, value, callback) => {
 | 
			
		||||
        if (value === "") {
 | 
			
		||||
          callback(new Error("请输入手机号码"));
 | 
			
		||||
        } else if (!isPhone(value)) {
 | 
			
		||||
          callback(new Error("请输入正确的手机号码格式"));
 | 
			
		||||
        } else {
 | 
			
		||||
          callback();
 | 
			
		||||
        }
 | 
			
		||||
      },
 | 
			
		||||
      trigger: "blur"
 | 
			
		||||
    }
 | 
			
		||||
  ],
 | 
			
		||||
  verifyCode: [
 | 
			
		||||
    {
 | 
			
		||||
      validator: (rule, value, callback) => {
 | 
			
		||||
        if (value === "") {
 | 
			
		||||
          callback(new Error("请输入验证码"));
 | 
			
		||||
        } else if (!REGEXP_SIX.test(value)) {
 | 
			
		||||
          callback(new Error("请输入6位数字验证码"));
 | 
			
		||||
        } else {
 | 
			
		||||
          callback();
 | 
			
		||||
        }
 | 
			
		||||
      },
 | 
			
		||||
      trigger: "blur"
 | 
			
		||||
    }
 | 
			
		||||
  ],
 | 
			
		||||
  password: [
 | 
			
		||||
    {
 | 
			
		||||
      validator: (rule, value, callback) => {
 | 
			
		||||
@ -25,4 +124,4 @@ const loginRules = reactive(<FormRules>{
 | 
			
		||||
  ]
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
export { loginRules };
 | 
			
		||||
export { loginRules, phoneRules, updateRules };
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										50
									
								
								src/views/login/utils/verifyCode.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								src/views/login/utils/verifyCode.ts
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,50 @@
 | 
			
		||||
import type { FormInstance, FormItemProp } from "element-plus";
 | 
			
		||||
import { clone } from "@pureadmin/utils";
 | 
			
		||||
import { ref } from "vue";
 | 
			
		||||
 | 
			
		||||
const isDisabled = ref(false);
 | 
			
		||||
const timer = ref(null);
 | 
			
		||||
const text = ref("");
 | 
			
		||||
 | 
			
		||||
export const useVerifyCode = () => {
 | 
			
		||||
  const start = async (
 | 
			
		||||
    formEl: FormInstance | undefined,
 | 
			
		||||
    props: FormItemProp,
 | 
			
		||||
    time = 60
 | 
			
		||||
  ) => {
 | 
			
		||||
    if (!formEl) return;
 | 
			
		||||
    const initTime = clone(time, true);
 | 
			
		||||
    await formEl.validateField(props, isValid => {
 | 
			
		||||
      if (isValid) {
 | 
			
		||||
        clearInterval(timer.value);
 | 
			
		||||
        isDisabled.value = true;
 | 
			
		||||
        text.value = `${time}`;
 | 
			
		||||
        timer.value = setInterval(() => {
 | 
			
		||||
          if (time > 0) {
 | 
			
		||||
            time -= 1;
 | 
			
		||||
            text.value = `${time}`;
 | 
			
		||||
          } else {
 | 
			
		||||
            text.value = "";
 | 
			
		||||
            isDisabled.value = false;
 | 
			
		||||
            clearInterval(timer.value);
 | 
			
		||||
            time = initTime;
 | 
			
		||||
          }
 | 
			
		||||
        }, 1000);
 | 
			
		||||
      }
 | 
			
		||||
    });
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const end = () => {
 | 
			
		||||
    text.value = "";
 | 
			
		||||
    isDisabled.value = false;
 | 
			
		||||
    clearInterval(timer.value);
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  return {
 | 
			
		||||
    isDisabled,
 | 
			
		||||
    timer,
 | 
			
		||||
    text,
 | 
			
		||||
    start,
 | 
			
		||||
    end
 | 
			
		||||
  };
 | 
			
		||||
};
 | 
			
		||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user