Browse Source

邮件登录

hm 1 month ago
parent
commit
4c55f7dc33

+ 2 - 2
.env.development

@@ -12,7 +12,7 @@ VUE_CLI_BABEL_TRANSPILE_MODULES = true
 
 # VUE_APP_BASE_API  = 'http://10.1.161.127:8080'
 # VUE_APP_BASE_API  = 'http://10.1.161.183:6661'
-VUE_APP_BASE_API  = 'http://192.168.136.24:6661'
-# VUE_APP_BASE_API2  = 'http://192.168.136.23:'
+VUE_APP_BASE_API  = 'http://10.1.161.48:6661'
+VUE_APP_BASE_API2  = 'http://10.1.161.48:5001'
 
 # VUE_APP_BASE_API  = 'http://10.1.162.158:6661'

+ 36 - 2
src/api/login.js

@@ -70,9 +70,43 @@ export function insertTabUser(data) {
 }
 
 //  catalog  token获取
-export function getSixCatalogToken() {
+export function getSixCatalogToken(data) {
   return request({
     url: '/web/getCatalogToken',
-    method: 'post'
+    method: 'post',
+    data: data    
+  })
+}
+
+
+// 获取邮箱验证码
+export function sendMail (data) {
+  return request({
+    url: '/web/mail/sendMail',
+    method: 'post',
+    data: data   
+  })
+}
+
+// 验证邮箱验证码
+export function verifyMail (data) {
+  return request({
+    url: '/web/mail/verifyMail',
+    method: 'post',
+    data: data   
+  })
+}
+
+export function loginByMail (data) {
+  return request({
+    url: '/loginByMail',
+    method: 'post',
+    headers: {
+      isToken: false,
+      repeatSubmit: false
+    },
+    data: data   
   })
 }
+
+

File diff suppressed because it is too large
+ 1 - 0
src/assets/icons/svg/mail.svg


+ 11 - 17
src/layout/components/Navbar.vue

@@ -1,17 +1,9 @@
 <template>
   <div class="navbar">
-    <hamburger
-      id="hamburger-container"
-      :is-active="sidebar.opened"
-      class="hamburger-container"
-      @toggleClick="toggleSideBar"
-    />
-
-    <breadcrumb
-      id="breadcrumb-container"
-      class="breadcrumb-container"
-      v-if="!topNav"
-    />
+    <hamburger id="hamburger-container" :is-active="sidebar.opened" class="hamburger-container"
+      @toggleClick="toggleSideBar" />
+
+    <breadcrumb id="breadcrumb-container" class="breadcrumb-container" v-if="!topNav" />
     <top-nav id="topmenu-container" class="topmenu-container" />
 
     <div class="right-menu">
@@ -33,14 +25,11 @@
         </el-tooltip> -->
       </template>
 
-      <el-dropdown
-        class="avatar-container right-menu-item hover-effect"
-        trigger="click"
-      >
+      <el-dropdown class="avatar-container right-menu-item hover-effect" trigger="click">
         <div class="avatar-wrapper">
           <!-- <img :src="avatar" class="user-avatar" /> -->
           <i class="el-icon-user" />
-          <span class="user-name">admin</span>
+          <span class="user-name">{{ username }}</span>
           <!-- <i class="el-icon-caret-bottom" color="#ffffff" icon="el-icon-search"/> -->
           <i class="el-icon-caret-bottom" />
         </div>
@@ -84,6 +73,11 @@ export default {
   },
   computed: {
     ...mapGetters(["sidebar", "avatar", "device"]),
+    username: {
+      get () {
+        return this.$store.state.user.name
+      }
+    },
     setting: {
       get() {
         return this.$store.state.settings.showSettings;

+ 2 - 2
src/permission.js

@@ -8,14 +8,14 @@ import { isRelogin } from '@/utils/request'
 
 NProgress.configure({ showSpinner: false })
 
-const whiteList = ['/login', '/register']
+const whiteList = ['/login', '/register','/mailLogin']
 
 router.beforeEach((to, from, next) => {
   NProgress.start()
   if (getToken()) {
     to.meta.title && store.dispatch('settings/setTitle', to.meta.title)
     /* has token*/
-    if (to.path === '/login') {
+    if (to.path === '/login' || to.path === '/mailLogin') {
       next({ path: '/' })
       NProgress.done()
     } else if (whiteList.indexOf(to.path) !== -1) {

+ 5 - 0
src/router/index.js

@@ -46,6 +46,11 @@ export const constantRoutes = [
     component: () => import('@/views/login'),
     hidden: true
   },
+  {
+    path: '/mailLogin',
+    component: () => import('@/views/mailLogin'),
+    hidden: true
+  },
   {
     path: '/register',
     component: () => import('@/views/register'),

+ 32 - 4
src/store/modules/user.js

@@ -4,6 +4,7 @@ import {
   getInfo,
   insertTabUser,
   getSixCatalogToken,
+  loginByMail
 } from "@/api/login";
 import {
   getToken,
@@ -62,10 +63,37 @@ const user = {
             commit("SET_TOKEN", res.token);
 
             // 获取 token 并发送给 iframe
-            getSixCatalogToken().then((result) => {
-              const token = result; // 假设 result 是整个 token 对象
-              setCatalogToken(token);
-            });
+            // getSixCatalogToken().then((result) => {
+            //   const token = result; // 假设 result 是整个 token 对象
+            //   setCatalogToken(token);
+            // });
+
+            resolve();
+          })
+          .catch((error) => {
+            reject(error);
+          });
+      });
+    },
+
+    // 邮件登录
+    MailLogin ({ commit }, userInfo) {
+      const toMail = userInfo.username.trim();
+      const password = userInfo.password;
+      const code = userInfo.code;
+      // const uuid = userInfo.uuid;
+      return new Promise((resolve, reject) => {
+        loginByMail({toMail, password, code})
+          .then((res) => {
+            console.log(res, "res==userID");
+            setToken(res.token);
+            commit("SET_TOKEN", res.token);
+
+            // 获取 token 并发送给 iframe
+            // getSixCatalogToken().then((result) => {
+            //   const token = result; // 假设 result 是整个 token 对象
+            //   setCatalogToken(token);
+            // });
 
             resolve();
           })

+ 2 - 2
src/utils/request2.js

@@ -8,7 +8,8 @@ import cache from '@/plugins/cache'
 import { saveAs } from 'file-saver'
 
 let downloadLoadingInstance;
-let baseURL = 'http://192.168.136.24:5001'
+// let baseURL = 'http://10.1.161.48:5001'
+let baseURL = process.env.VUE_APP_BASE_API2
 // 是否显示重新登录
 export let isRelogin = { show: false };
 
@@ -16,7 +17,6 @@ axios.defaults.headers['Content-Type'] = 'application/json;charset=utf-8'
 // 创建axios实例
 const service = axios.create({
   // axios中请求配置有baseURL选项,表示请求URL公共部分
-  // baseURL: process.env.VUE_APP_BASE_API2+'5001' ,
   baseURL: baseURL ,
   
   // 超时

+ 4 - 1
src/views/login.vue

@@ -62,7 +62,7 @@
         <el-form-item prop="">
           <div>
             <el-button type="text">数字登录证书</el-button><el-button type="text">短信验证码登录</el-button><el-button
-              type="text">邮件验证码登录</el-button><el-button type="text">微信登录</el-button>
+              type="text" @click="goMailLogin">邮件验证码登录</el-button><el-button type="text">微信登录</el-button>
           </div>
         </el-form-item>
       </el-form>
@@ -215,6 +215,9 @@ export default {
         }
       });
     },
+    goMailLogin () {
+      this.$router.push("/mailLogin").catch(() => { })
+    }
     
   },
 };

+ 1 - 1
src/views/login copy.vue

@@ -161,7 +161,7 @@ export default {
   justify-content: center;
   align-items: center;
   height: 100%;
-  background-image: url("../assets/images/login-background.jpg");
+  background-image: url("../assets/img/互联网门户登录页.jpg");
   background-size: cover;
 }
 .title {

+ 454 - 0
src/views/mailLogin.vue

@@ -0,0 +1,454 @@
+<template>
+  <div class="login">
+    <div class="login-content">
+      <!-- 公告栏 -->
+      <!-- <div class="announcement">
+        <div class="announcement-header">
+          <div class="announcement-bg">
+            公示公告 <img src="../assets/img/装饰.png" alt="" />
+          </div>
+        </div>
+        <div class="announcement-list">
+          <div v-for="item in announcements" :key="item.noticeId" class="announcement-item"
+            :class="{ highlighted: item.show === true }">
+            <div class="announcement-item-header" @click="announcementChange(item)">
+              <el-tooltip class="item" effect="dark" :content="item.noticeTitle" placement="left-start">
+                <div class="announcement-title" :class="{ highlighted: item.show === true }">
+                  {{ item.noticeTitle }}
+                </div>
+              </el-tooltip>
+              <div class="announcement-date">{{ item.updateTime }}</div>
+            </div>
+            <div class="announcement-qs" v-if="item.show">
+              <div v-html="item.noticeContent"></div>
+            </div>
+          </div>
+        </div>
+      </div> -->
+      <el-form
+        ref="loginForm"
+        :model="loginForm"
+        :rules="loginRules"
+        class="login-form"
+      >
+        <h3 class="title">邮箱登录</h3>
+        <el-form-item prop="username">
+          <el-input
+            v-model="loginForm.username"
+            type="text"
+            auto-complete="off"
+            placeholder="账号"
+          >
+            <svg-icon
+              slot="prefix"
+              icon-class="mail"
+              class="el-input__icon input-icon"
+            />
+          </el-input>
+        </el-form-item>
+        <el-form-item prop="password">
+          <el-input v-model="loginForm.password" type="password" auto-complete="off" placeholder="密码"
+            @keyup.enter.native="handleLogin">
+            <svg-icon slot="prefix" icon-class="password" class="el-input__icon input-icon" />
+          </el-input>
+        </el-form-item>
+        <el-form-item prop="code" v-if="captchaEnabled">
+          <el-input
+            v-model="loginForm.code"
+            auto-complete="off"
+            placeholder="验证码"
+            style="width: 63%"
+            @keyup.enter.native="handleLogin"
+          >
+            <svg-icon
+              slot="prefix"
+              icon-class="validCode"
+              class="el-input__icon input-icon"
+            />
+          </el-input>
+          <div class="login-code">
+            <!-- <img :src="codeUrl" @click="getCode" class="login-code-img" /> -->
+            <!-- <el-button type="text">发送邮件</el-button> -->
+            <el-button
+              @click="sendEmail"
+              :disabled="isCountingDown"
+              class="sendEmail_btn"
+              style=""
+            >
+              {{ isCountingDown ? `重新发送 (${countdown}s)` : "发送验证码" }}
+            </el-button>
+          </div>
+        </el-form-item>
+        <el-checkbox
+          v-model="loginForm.rememberMe"
+          style="margin: 0px 0px 25px 0px"
+          >记住密码</el-checkbox
+        >
+        <el-form-item style="width: 100%">
+          <el-button
+            round
+            :loading="loading"
+            size="medium"
+            type="primary"
+            style="width: 100%"
+            @click.native.prevent="handleLogin"
+          >
+            <span v-if="!loading">登 录</span>
+            <span v-else>登 录 中...</span>
+          </el-button>
+          <div style="float: right" v-if="register">
+            <router-link class="link-type" :to="'/register'"
+              >立即注册</router-link
+            >
+          </div>
+        </el-form-item>
+        <el-form-item prop="">
+          <div>
+            <el-button type="text" @click="goLogin">账号密码登录</el-button>
+            <el-button type="text">数字登录证书</el-button
+            ><el-button type="text">短信验证码登录</el-button
+            ><el-button type="text">微信登录</el-button>
+          </div>
+        </el-form-item>
+      </el-form>
+    </div>
+    <!--  底部  -->
+    <div class="el-login-footer">
+      <p>版权所有 上海市疾病预防控制中心</p>
+      <p>地址:上海市中申虹路1509号</p>
+      <p>电话:021-62758710 邮编:200336</p>
+      <p>
+        提醒:本系统建议采用IE11或 Google chrome(谷歌)或火狐或奇安信的测览雅
+      </p>
+    </div>
+  </div>
+</template>
+
+<script>
+// 公示公告
+import { listNotice } from "@/api/system/notice";
+import { getCodeImg, sendMail,verifyMail } from "@/api/login";
+import Cookies from "js-cookie";
+import { encrypt, decrypt } from "@/utils/jsencrypt";
+
+export default {
+  name: "Login",
+  data() {
+    return {
+      isCountingDown: false, // 控制按钮禁用和倒计时
+      countdown: 60, // 倒计时初始值
+      timer: null, // 存储计时器的引用
+      loginForm: {
+        username: "admin",
+        password: "admin123",
+        rememberMe: false,
+        code: "",
+        uuid: "",
+      },
+      loginRules: {
+        username: [
+          { required: true, trigger: "blur", message: "请输入您的账号" },
+        ],
+        password: [
+          { required: true, trigger: "blur", message: "请输入您的密码" },
+        ],
+        code: [{ required: true, trigger: "change", message: "请输入验证码" }],
+      },
+      loading: false,
+      // 验证码开关
+      captchaEnabled: true,
+      // 注册开关
+      register: false,
+      redirect: undefined,
+      announcements: [
+      ],
+    };
+  },
+  watch: {
+    $route: {
+      handler: function (route) {
+        this.redirect = route.query && route.query.redirect;
+      },
+      immediate: true,
+    },
+  },
+  created () {
+    this.loginForm={}
+    // this.getCode();
+    this.getCookie();
+    this.getListNotice();
+  },
+  methods: {
+    getListNotice() {
+      listNotice({ noticeType: 2 }).then((response) => {
+        response.rows.forEach((item) => {
+          item.show = false;
+        });
+        this.announcements = response.rows;
+      });
+    },
+    announcementChange(item) {
+      this.announcements.forEach((element) => {
+        if (element.noticeId === item.noticeId) {
+          element.show = !element.show;
+        }
+      });
+    },
+    // getCode() {
+    //   getCodeImg().then((res) => {
+    //     this.captchaEnabled =
+    //       res.captchaEnabled === undefined ? true : res.captchaEnabled;
+    //     if (this.captchaEnabled) {
+    //       this.codeUrl = "data:image/gif;base64," + res.img;
+    //       this.loginForm.uuid = res.uuid;
+    //     }
+    //   });
+    // },
+    getCookie() {
+      const username = Cookies.get("username");
+      const password = Cookies.get("password");
+      const rememberMe = Cookies.get("rememberMe");
+      this.loginForm = {
+        username: username === undefined ? this.loginForm.username : username,
+        password:
+          password === undefined ? this.loginForm.password : decrypt(password),
+        rememberMe: rememberMe === undefined ? false : Boolean(rememberMe),
+      };
+    },
+    handleLogin() {
+      this.$refs.loginForm.validate((valid) => {
+        if (valid) {
+          this.loading = true;
+          if (this.loginForm.rememberMe) {
+            Cookies.set("username", this.loginForm.username, { expires: 30 });
+            Cookies.set("password", encrypt(this.loginForm.password), {
+              expires: 30,
+            });
+            Cookies.set("rememberMe", this.loginForm.rememberMe, {
+              expires: 30,
+            });
+          } else {
+            Cookies.remove("username");
+            Cookies.remove("password");
+            Cookies.remove("rememberMe");
+          }
+          this.$store
+            .dispatch("MailLogin", this.loginForm)
+            .then(() => {
+              this.$router.push({ path: this.redirect || "/" }).catch(() => {});
+            })
+            .catch(() => {
+              this.loading = false;
+              if (this.captchaEnabled) {
+                // this.getCode();
+              }
+            });
+        }
+      });
+    },
+    goLogin() {
+      this.$router.push("/login").catch(() => {});
+    },
+    // 发送验证码
+    sendEmail() {
+      if (!this.isCountingDown) {
+        sendMail({ toMail: this.loginForm.username }).then((res) => {
+            if (+res.code===200) {
+              this.$message.success("验证码已发送到您的邮箱,请查收");
+              // 启动倒计时
+              this.isCountingDown = true;
+              this.startCountdown();
+            }
+          });
+      
+      }
+    },
+
+    // 倒计时逻辑
+    startCountdown() {
+      this.timer = setInterval(() => {
+        if (this.countdown > 0) {
+          this.countdown -= 1;
+        } else {
+          this.resetCountdown();
+        }
+      }, 1000);
+    },
+
+    // 重置倒计时
+    resetCountdown() {
+      clearInterval(this.timer);
+      this.countdown = 60;
+      this.isCountingDown = false;
+    },
+  },
+};
+</script>
+
+<style rel="stylesheet/scss" lang="scss" scoped>
+.login {
+  display: flex;
+  justify-content: center;
+  align-items: center;
+  height: 100%;
+  background-image: url("../assets/img/互联网门户登录页.jpg");
+  background-size: cover;
+}
+
+.title {
+  margin: 0px auto 30px auto;
+  text-align: center;
+  color: #707070;
+  font-weight: 700;
+}
+
+.login-form {
+  border-radius: 6px;
+  background: rgba(255, 255, 255, 0.7);
+  width: 440px;
+  padding: 25px 25px 5px 25px;
+
+  .el-input__inner {
+    border-radius: 19px;
+    border: 1px solid #2692ff;
+  }
+
+  .el-input {
+    height: 38px;
+
+    input {
+      height: 38px;
+    }
+  }
+
+  .input-icon {
+    height: 39px;
+    width: 14px;
+    margin-left: 2px;
+  }
+}
+
+.login-tip {
+  font-size: 13px;
+  text-align: center;
+  color: #bfbfbf;
+}
+
+.login-code {
+  width: 33%;
+  height: 38px;
+  float: right;
+  .sendEmail_btn {
+    width: 100%;
+    display: flex;
+    align-items: center;
+    justify-content: center;
+  }
+  img {
+    cursor: pointer;
+    vertical-align: middle;
+  }
+}
+
+.el-login-footer {
+  // height: 40px;
+  // line-height: 40px;
+  position: fixed;
+  bottom: 0;
+  width: 100%;
+  text-align: center;
+  color: #fff;
+  font-family: Arial;
+  font-size: 12px;
+  letter-spacing: 1px;
+}
+
+.login-code-img {
+  height: 38px;
+}
+
+.login-content {
+  width: 100%;
+  display: flex;
+  justify-content: space-evenly;
+}
+
+.announcement {
+  height: 400px;
+  width: 440px;
+  background: rgba(255, 255, 255, 0.7);
+  border-radius: 10px;
+  padding: 15px;
+  box-shadow: 0 4px 10px rgba(0, 0, 0, 0.1);
+}
+
+.announcement-header {
+  font-size: 16px;
+  height: 30px;
+  display: flex;
+  align-items: center;
+  justify-content: space-between;
+
+  .announcement-bg {
+    font-weight: 700;
+    color: #1a8ff9;
+
+    img {
+      margin-left: 5px;
+    }
+  }
+}
+
+.announcement-list {
+  margin-top: 10px;
+  padding-right: 20px;
+  height: 320px;
+  overflow-y: auto;
+  // overflow: hidden;
+}
+
+.announcement-item {
+  margin-bottom: 10px;
+  padding: 10px;
+
+  border-bottom: 1px dashed #f2f1f3;
+  // margin-bottom: 10px;
+
+  .announcement-item-header {
+    display: flex;
+    align-items: center;
+    justify-content: space-between;
+  }
+
+  .announcement-qs {
+    font-size: 12px;
+    color: #a3acb3;
+  }
+
+  &.highlighted {
+    background-color: rgba(#dbe7f5, 0.3);
+    /* 淡蓝色背景 */
+  }
+}
+
+.announcement-title {
+  cursor: pointer;
+  max-width: 260px;
+  font-size: 14px;
+  color: #828282;
+  white-space: nowrap; // 防止文本换行
+  overflow: hidden; // 超出部分隐藏
+  text-overflow: ellipsis; // 如果超出部分隐藏,则使用省略号显示
+
+  &.highlighted {
+    font-weight: 700;
+    color: #3981db;
+    /* 深蓝色标题 */
+  }
+}
+
+.announcement-date {
+  font-size: 12px;
+  color: #888;
+}
+</style>

+ 1 - 1
src/views/system/dept/index.vue

@@ -28,7 +28,7 @@
       <el-col :span="1.5">
         <el-button type="info" plain icon="el-icon-sort" size="mini" @click="toggleExpandAll">展开/折叠</el-button>
       </el-col>
-      <right-toolbar :showSearch.sync="showSearch" @queryTable="getList"></right-toolbar>
+      <!-- <right-toolbar :showSearch.sync="showSearch" @queryTable="getList"></right-toolbar> -->
     </el-row>
 
     <el-table v-if="refreshTable" v-loading="loading" :data="deptList" row-key="deptId"

+ 7 - 0
vue.config.js

@@ -43,6 +43,13 @@ module.exports = {
         pathRewrite: {
           ['^' + process.env.VUE_APP_BASE_API]: ''
         }
+      },
+      [process.env.VUE_APP_BASE_API2]: {
+        target: process.env.VUE_APP_BASE_API2 || 'http://10.1.161.48:5001',
+        changeOrigin: true,
+        pathRewrite: {
+          ['^' + process.env.VUE_APP_BASE_API2]: ''
+        }
       }
     },
     disableHostCheck: true