打通Vue開發前後臺業務的任督二脈

Key.L發表於2020-03-03

打通Vue開發前後臺業務的任督二脈

前言

本文會在巨集觀視野上,講述從全棧開發到整棧開發的能力,會展示一個簡單的後臺開發專案片段,其中採用了element-ui作為後臺介面開發框架,會分享前後臺通訊之間的API設計經驗,以及vue3.0中使用proxy代理服務解決跨域問題。

專案目錄

先建立三個專案目錄,這裡以ele為例,分別建立前端, 後端, 後臺三個專案。

打通Vue開發前後臺業務的任督二脈

打理後臺頁面

  <div class="login_page fillcontain">
    <transition name="form-fade">
      <section class="form_container form-fade-enter" v-show="showLogin">
        <div class="manage_tip">
          <p>elm後臺管理系統</p>
        </div>
         <el-form :model="loginForm" :rules="rules" ref="loginForm">
          <el-form-item prop="username">
            <!-- 雙向繫結指令 -->
            <el-input v-model="loginForm.username" placeholder="使用者名稱"></el-input>
          </el-form-item>
          <el-form-item prop="password">
            <!-- 雙向繫結指令 -->
            <el-input v-model="loginForm.password" placeholder="密碼" type="password"></el-input>
          </el-form-item>
          <el-form-item>
            <el-button type="primary" @click="submitForm('loginForm')" class="submit_btn">登陸</el-button>
          </el-form-item>
        </el-form>
      </section>
    </transition>
  </div>
</template>
<script>
export default {
  data() {
    return {
      showLogin: false,
      loginForm: {
        username: "",
        password: ""
      },
      rules: {
        username: [
          { required: true, message: "請輸入使用者名稱", trigger: "blur" }
        ],
        // 漸進式
        password: [
          {
            required: true,
            message: "請輸入密碼",
            trigger: "blur"
          }
        ]
      }
    };
  },
mounted() {
    this.showLogin = true;
  }
};    
</script>
<style lang="stylus" scoped>
.login_page {
  height: 100vh;
  background-color: #324057;
}

.manage_tip {
  position: absolute;
  width: 100%;
  top: -100px;
  left: 0;

  p {
    font-size: 34px;
    color: #ffffff;
  }
}

.form_container {
  width: 320px;
  height: 240px;
  position: absolute;
  top: 50%;
  left: 50%;
  margin-top: -120px;
  margin-left: -160px;
  padding: 25px;
  box-sizing: border-box;
  border-radius: 5px;
  text-align: center;
  background-color: #ffffff;

  .submit_btn {
    width: 100%;
    font-size: 16px;
  }
}

// 動態的產生四個類
.form-fade-enter-active, .form-fade-leave-active {
  transition: all 1s;
}
.form-fade-enter, .form-fade-leave-active {
  transform: translate3d(0, -50px, 0);
  opacity: 0;
}
</style>
複製程式碼

transtion是 vue的內建元件,當它的子元素通過生命週期函式改變v-show = "showLogin" false->true的值,從不顯示到顯示,就會觸發,帶來特效。transtion的name屬性會幫我們動態產生四個類,transition-name-enter(一幀) -> transition-name-enter-active 子元素原來的樣式 transition-name-enter未進入的樣式 transition-name-enter-active 設定transtion-time all 1s

前後端聯調

methods: {
  async submitForm(formName) {
      this.$refs[formName].validate(async valid => {
        // console.log(valid);
        if (valid) {
          // 跟後端api通訊吧, 登入
        } else {
          this.$notify.error({
            title: "錯誤",
            message: "請輸入正確的使用者名稱和密碼",
            offset: 100
          });
        }
      });
    }
  },

複製程式碼

$notify 是element-ui中的訊息提示元件,相當於iview中的$message. 在Vue中ref相當與id屬性,基於domless的理念,vue中減少dom操作,讓開發更加簡單,用ref代替實現id的功能。

非同步ajax請求

 // api 非同步的ajax請求 
    const res = await login({
    user_name: this.loginForm.username,
    password: this.loginForm.password
複製程式碼

這個login,來自api模組,為了避免衝突,後端請示都應該在api裡, 這裡需要匯入一下

import { login } from "@/api/getData";
複製程式碼
/**
 * API定義集合檔案,
 * @param {*} data 
 */
export const login = (data) => {
  return new Promise((resolve, reject) => {
    // http://127.0.0.1:8080/  跨專案 
    fetch('/api/admin/login', {
      method: 'POST', //請求頭
      body: JSON.stringify(data) //請求體 request body
    })
    .then(data => data.json())
    .then(data => {
      console.log(data, '----------')
      resolve(data)
    })
    .catch(err => {
      console.log(err)
      reject(err)
    })
  })
}

複製程式碼

模擬mock

把能寫的程式碼寫完,減少不確定性

if (res.status == 1) {
    this.$message({
      type: "success",
      message: "登入成功"
    });
  } else {
    this.$message({
      type: "error",
      message: res.message
    });
  }
複製程式碼

這個status,是為了方便打通前後臺,約定好資料介面,約定後端一定返回一個status: 1|0,來驗證前後端通訊是否通暢。有了這個約定,就可以不用等到後端專案都寫完,處理前端業務。

後端路由模組化

const express = require('express');
const app = express();
// 路由的模組化
// api
const router = require('./routes/index.js');
router(app);

// app.use()
app.listen("3000", () => {
  console.log('api 伺服器上線了');
})
複製程式碼

子路由模組

const admin = require('./admin')
module.exports = app => {
  // 啟用路由中介軟體
  // 模組子路由
  app.use('/admin', admin)
}
複製程式碼

給路由新增方法

登入業務忽略

const express = require('express');
const router = express.Router();

router.post('/login', (req, res) => {
  res.send({
    status: 1,
    message: '登入成功'
  })
})

module.exports = router

複製程式碼

跨域解決方案

vue3.0提供了這個解決方案 在後臺專案根目錄新建一個vue.config.js

module.exports = {
    devServer: {
      // '127.0.0.1:8080/api/admin/login'
      proxy: {
        '/api': {
          target: 'http://127.0.0.1:3000',
          changeOrigin: true,
          // http://127.0.0.1:3000/admin/login
          pathRewrite: {
            '^/api': ''
          }
        }
      }
    }
  }
  
複製程式碼

通過proxy代理,使用正則速配,如果是/api請求則幫我們轉發到3000埠下,這樣就不存在跨域問題了。

寫在最後

  • 如果這篇文章有錯誤或不足之處,歡迎在評論區指出。
  • 歡迎大家點贊,關注,技術交流。

相關文章