打通先後端全棧開發node+vue進階【課程學習系統項目實戰詳細講解】(2):後臺路由,導航,首頁,退出登陸

第二章 建議學習時間8小時      總項目預計10章javascript

 

學習方式:詳細閱讀,並手動實現相關代碼(若是沒有node和vue基礎,請學習前面的vue和node基礎博客【共10章】css

演示地址:後臺:demoback.lalalaweb.com  前臺:demo.lalalaweb.com
html

演示過程當中可能會發現bug,但願即時留言反饋,謝謝vue

源碼下載:https://github.com/sutianbinde/classweb               //不是所有的代碼,每次更新博客才更新代碼java

學習目標:此教程將教會你們 如何一步一步實現一個完整的課程學習系統(包括課程管理後臺/Node服務器/學習門戶三個模塊)。node

上次node基礎課程博客你們反響很好,時隔3個月,才更新項目部分,預計2~3天更新一章,我儘可能20天更新完畢,學完這個項目Nodejs和vue就基本熟悉了,如發現教程有誤的地方,請及時留言反饋git

視頻教程地址:www.lalalaweb.com,後期會上傳教學視頻,你們可前往視頻學習(暫時尚未視頻)github

 

首頁導航 路由配置web


 上一篇博客中,咱們已經實現了登陸功能,那麼接着我就須要寫登陸完成後跳轉的頁面ajax

 

咱們先把項目運行起來  Node端 npm start ,vue端npm run dev

 

首先咱們須要使用一個字體圖標庫 fontawesome,若是沒有用過的自行百度啊,這裏就不扯其餘知識了

下載地址:http://fontawesome.dashgame.com/    或者 到github下載的本項目中對應路徑去找也能夠

下載好之後把css和font放到static中,而後咱們在index.html中引入

    <link rel="stylesheet" type="text/css" href="../static/css/font-awesome.min.css"/>

注:1。我沒有找到能夠經過import引入的fontawesome包,因此就直接引入文件了,(網上有 vue-awesome,但貌似很差用,就沒有用)

這裏爲何是 ../static  這樣去找static,而不是 ./   ,由於當進入二級路由之後,在路由內部index和static就不在被認爲是同一級,就找不到了,因此就經過  ../往上再找了一級。

 

咱們要設置一些統一的全局樣式,咱們就直接寫在 index.html中,這裏原本不是一次就寫完這些樣式,但爲了不之後再回來添加樣式,這裏就一塊兒寫了,首先清楚了全局的margin等,而後定義了 .btn按鈕樣式   .myinput輸入框樣式,之後再使用

<style>
        *{
                margin: 0;
                padding: 0;
            }
            body{
                font-size: 14px;
                font-family: arial "microsoft yahei";
                background: #f0f2f5;
            }
            ul,li{
                list-style: none;
            }
            /*按鈕*/
            .btn{
                border:1px solid #4187db;
                color: #4187db;
                background: #fff;
                padding: 6px 14px 7px;
                border-radius: 3px;
                transition: all 0.5s ease;
                outline: none;
                margin-top: 14px;
                cursor: pointer;
            }
            .btn i{
                margin-right: 4px;
            }
            .btn:hover{
                background: #4187db;
                color: #fff;
            }
        
            /*輸入框*/
            .myinput{
                width: 65%;
                border: 1px solid #cad3de;
                height: 35px;
                line-height: 35px;
                margin: 5px 0 10px;
                border-radius: 3px;
                padding: 0 10px;
                outline: none;
                box-sizing: border-box;
            }
            .myinput:focus{
                border: 1px solid #4289dc;
            }
    </style>

 


在assets文件夾中建立 images文件夾,放入咱們backIndex.vue中須要的圖片  (圖片請到github下載的項目中對應路徑去找)

 

而後咱們去修改 路由文件 index.js的路由,在其中添加後臺首頁框架的路由,以下圖

 而且在components中建立 backIndex.vue組件

 

 

backIndex.vue組件中寫入後面代碼

基本功能以下圖,左側導航,頂部搜索欄和我的頭像 退出等操做

代碼解釋:中間大部分是Html+css代碼,代碼中註釋已經能夠幫助你們理解

這裏着重說一下路由部分:router-link表示點擊的時候url須要跳轉的地址,這些地址對應的路由文件尚未寫,這裏先寫上,下下步就配置這個,頁面很少,咱們就完成 首頁 用戶/學員管理 課程列表/課程編輯  幾個頁面,這樣就能夠算是一個基礎版的後臺管理系統了。

當點擊對應的link的時候,vue內部會自動添加router-link-active 類,咱們css中對這個類作了樣式設置。

 

<template>
  <div class="backlogin">
    <div class="header">
            <div class="search_box" :class="{search_box_fouce:search_box_fouce}">
                <i class="fa fa-search" aria-hidden="true"></i>
                <input @focus="focusFn" @blur="blurFn" type="text" name="" id="" value="" placeholder="搜索 . . . " />
            </div>
            <div class="handler">
                <div class="more" @click="toggleSlide">
                    <i class="fa fa-bars" aria-hidden="true"></i>
                    <ul :class="{showul:showExit}">
                        <li><a href="javascript:;" @click="logout"><i class="fa fa-sign-out" aria-hidden="true"></i>退出</a></li>
                        <li><a href="javascript:;" @click="">修改密碼</a></li>
                        <li><a href="javascript:;">意見反饋</a></li>
                    </ul>
                </div>
                <img src="../assets/images/teacherimg01.png" alt="" />
            </div>
        </div>
        
        <!--側面導航-->
        <div class="sidenav_box">
            <img class="logo" src="../assets/images/logo03.png" alt="" />
            <ul class="sidenav">
                <li class="now">
                    <router-link to="/backIndex/indexContent">
                        <i class="fa fa-home" aria-hidden="true"></i>
                        <span>網站首頁</span>
                    </router-link>
                </li>
                <li>
                    <router-link to="/backIndex/adminList">
                        <i class="fa fa-user-o" aria-hidden="true"></i>
                        <span>後臺人員</span>
                    </router-link>
                </li>
                <li>
                    <router-link to="/backIndex/studentList">
                        <i class="fa fa-user-circle-o" aria-hidden="true"></i>
                        <span>學員管理</span>
                    </router-link>
                </li>
                <li>
                    <router-link to="/backIndex/courseList">
                        <i class="fa fa-book" aria-hidden="true"></i>
                        <span>課程管理</span>
                    </router-link>
                </li>
            </ul>
        </div>
        
        <div class="content">
            
            <ul class="breadcrumb">
                <li><a href="#/backIndex/">首頁</a></li>
                <li>{{pageTitle}}</li>
            </ul>
            <router-view></router-view>
            
        </div>
        
  </div>
</template>

<script>
    var pageTitleObj = {
        indexContent:"網站首頁",
        adminList:"後臺人員",
        studentList:"學員管理",
        courseList:"課程管理",
        courseEdit:"課程編輯"
    };
    
    export default {
      name: 'backlogin',
      data () {
        return {
                    search_box_fouce:false,
                    showExit:false,
                    pageTitle: pageTitleObj[ this.$route.path.substr( this.$route.path.lastIndexOf("/")+1 ) ] || "網站首頁"
        }
      },
      methods:{
            focusFn(){  //搜索框獲取焦點,添加class
                this.search_box_fouce = true;
            },
            blurFn(){   //搜索框失去焦點,去掉class
                this.search_box_fouce = false;
            },
            toggleSlide(){ //這個是用來顯示和隱藏頭像旁的退出下拉框
                this.showExit = !this.showExit
            },
            logout(){ //退出系統
                
            }
        },
        watch:{ //監控路徑變化  當路徑發送變化的時候,改變麪包屑導航的顯示
            $route: {
                handler: function (val, oldVal) {
                    var path = val.path;
                    this.pageTitle = pageTitleObj[ path.substr( path.lastIndexOf("/")+1 ) ] || "網站首頁";
                }
            }
        }
    }
</script>

<!-- Add "scoped" attribute to limit CSS to this component only -->
<style scoped>
    ul, li{
        list-style: none;
    }
    .header{
        height: 60px;
        box-shadow: 0 1px 5px rgba(13,62,73,0.2) ;
        background: #fff;
        margin-left: 80px;
        min-width: 740px;
    }
    
    .sidenav_box{
        width: 80px;
        box-shadow: 0 1px 5px rgba(13,62,73,0.2) ;
        position: fixed;
        left: 0;
        top: 0;
        bottom: 0;
        background: #fff;
        z-index: 99;
    }
    .sidenav_box .logo{
        width: 46px;
        margin: 20px 0 0 17px;
    }
    .sidenav{
        margin-top: 30px;
    }
    .sidenav li{
        margin-bottom: 20px;
    }
    .sidenav a{
        display: block;
        width: 56px;
        height: 56px;
        margin: 0 auto;
        position: relative;
        cursor: pointer;
        opacity: 0.6;
        transition:all 0.5s ease;
        text-decoration: none;
    }
    .sidenav a i{
        font-size: 20px;
        line-height: 56px;
        text-align: center;
        display: block;
        color: #566a80;
    }
    .sidenav a:hover{
        background: #f0f2f5;
        opacity: 1;
    }
    .sidenav a span{
        position: absolute;
        left: 55px;
        top: 22px;
        background: #000;
        color: #fff;
        width: 0px;
        padding: 5px 0;
        border-radius: 3px;
        font-size: 12px;
        opacity: 0;
    }
    .sidenav a:hover span{
        opacity: 1;
        left: 65px;    
        width: 60px;
        padding: 5px 20px;
        transition:none 0.5s ease-out;
        transition-property: opacity,left;
    }
    .sidenav a span:after{
        content: "";
        position: absolute;
        top: 8px;
        left: -10px;
        border:5px solid transparent;
        border-right-color: #000;
    }
    .sidenav .router-link-active:after{
        content: "";
        position: absolute;
        left: -16px;
        top: 8px;
        height: 40px;
        width: 8px;
        border-radius: 3px;
        background: #566a80;
    }
    .sidenav .router-link-active{
        opacity: 1;
        background: #f0f2f5;
    }
    
    /*頂部欄*/
    .search_box{
        color: #979fa8;
        padding-top: 20px;
        float: left;
    }
    .search_box i{
        margin: 0 12px 0 70px;
        transition: all 0.5s ease;
    }
    .search_box input{
        border:none;
        outline: none;
    }
    .search_box_fouce i{
        margin-left: 55px;
        color: #2C3D50;
    }
    
    .handler > *{
        float: right;
        margin-right: 20px;
        cursor: pointer;
    }
    .handler .more{
        font-size: 20px;
        color: #566A80;
        margin: 15px 30px 0 0;
        position: relative;
    }
    .handler .more:hover{
        color: #2C3D50;
    }
    .handler .more ul{
        font-size: 14px;
        position: absolute;
        right: 0;
        top: 55px;
        width: 120px;
        box-shadow: 0 1px 5px rgba(13,62,73,0.2);
        transition: all 0.3s ease-out;
        height: 0;
        opacity: 0;
        overflow: hidden;
        text-align: center;
    }
    .handler .more .showul{
        height: auto;
        top: 45px;
        opacity: 1;
        border-top: 1px solid #979FA8;
    }
    .handler .more a{
        display: block;
        padding: 8px 10px;
        background: #fff;
        color: #566A80;
        text-decoration: none;
    }
    .handler .more a:hover{
        background: #f8f9fb;
    }
    .handler > img{
        width: 50px;
        border-radius: 50%;
        margin-top: 5px;
        margin-right: 30px;
    }
    
    .content{
        margin: 20px 30px 0px 100px;
        min-height: 300px;
        min-width: 700px;
    }
    
    .breadcrumb{
        border-radius: 4px;
        padding: 10px 15px;
        background: #fff;
    }
    .breadcrumb > li{
        display: inline-block;
        color: #777777;
    }
    .breadcrumb > li+li:before{
        padding: 0 5px;
        color: #ccc;
        content: "/\00a0";
    }
    .breadcrumb > li > a{
        color: #32475f;
        text-decoration: none;
    }
    .main{
        border-radius: 4px;
        background: #fff;
        margin-top: 10px;
    }
    
    
    
</style>

 

而後在地址欄輸入 http://localhost:8080/#/backIndex  就能夠看到首頁框架的效果了。  (這時候內部頁面尚未,因此點擊左側導航會找不到頁面,先不要點)

常見錯誤:  若是前面的圖片路徑沒放對,就會出現這個錯誤

 

 而後咱們繼續將全部的路由配置其餘頁面的路由

一共有 課程列表 / 編輯課程 / 首頁統計 /  後臺用戶  / 學員用戶  這些二級頁面,咱們都加入到  路由文件  index.js中

import Vue from 'vue'
import Router from 'vue-router'
import Login from '@/components/login'
import backIndex from '@/components/backIndex'
import courseList from '@/components/courseList'
import indexContent from '@/components/indexContent'
import adminList from '@/components/adminList'
import studentList from '@/components/studentList'
import courseEdit from '@/components/courseEdit'


Vue.use(Router)

export default new Router({
  routes: [
    {
      path: '/',
      name: 'Login',
      component: Login
    },{
        path: '/backIndex', //首頁框架
        name: 'backIndex',
        component: backIndex,
        children:[
            {
              path: 'courseList', //課程列表
                component: courseList
            },{
              path: 'courseEdit/:sysId', //編輯課程
                component: courseEdit
            },{
              path: 'indexContent', //首頁統計
                component: indexContent
            },{
              path: 'adminList', //後臺用戶
                component: adminList
            },{
              path: 'studentList', //學員用戶
                component: studentList
            },{
               path: '*',           //其餘路徑都跳轉到首頁
                redirect: 'indexContent'
            }
        ]
        }
  ]
})

添加這些路由後的須要創建對應的 vue 組件文件,在componets中創建上面路由對應的空vue文件

這些新建的vue文件中須要先放入一個空的template標籤,以避免報錯

<template>
</template>

 

 

 再刷新頁面的時候,左側導航就能夠點擊了(這些router-link對應的路徑和路由對應的path一致),能夠看到麪包屑導航對應的名字就修改了(這個修改的邏輯在backIndex.vue的97行左右的 watch方法中)。

 

 

首頁統計頁面


 

而後咱們爲 indexContent.vue 添加中間顯示的統計圖表,代碼在後面

代碼以下:因爲前臺頁面的統計還沒開始寫,當下咱們就只放一個靜態頁面在此,

canvas繪製提標使用的是原生js寫的,具體的原理步驟請查看我之前的博客  http://www.cnblogs.com/chengduxiaoc/p/7678967.html

<template>
  <div class="indexContent main">
    <h4>最新數據</h4>
    <ul class="number">
            <li>
                <div class="title">今日訪問</div>
                <p>12000</p>
                <a href="javascript:;">查看詳情<i class="fa fa-angle-right" aria-hidden="true"></i></a>
            </li>
            <li>
                <div class="title">學員總數</div>
                <p>3000000</p>
                <a href="javascript:;">查看詳情<i class="fa fa-angle-right" aria-hidden="true"></i></a>
            </li>
            <li>
                <div class="title">在學人數</div>
                <p>2000</p>
                <a href="javascript:;">查看詳情<i class="fa fa-angle-right" aria-hidden="true"></i></a>
            </li>
        </ul>
    <canvas id="barChart" height="400" width="600" style="margin:10px 0"> 你的瀏覽器不支持HTML5 canvas </canvas>
        
  </div>
</template>

<!-- Add "scoped" attribute to limit CSS to this component only -->
<style scoped>
    .main{
        border-radius: 4px;
        background: #fff;
        margin-top: 10px;
        overflow: hidden;
    }
    .main > h4{
        color: #51555a;
    padding:10px;
    border-bottom: 1px solid #DFE3EA;
    }
    .number{
        width: 30%;
        float: right;
        margin-right: 10%;
        margin-top: 10px;
        color: #566A80;
    }
    .number li{
        padding: 20px;
        border-top:1px solid #F0F2F5;
    }
    .number li:first-child{
        border: none 0;
    }
    .number p{
        font-size: 20px;
        font-family: arial;
        margin: 10px 0;
    }
    .number a{
        text-decoration: none;
        color: #4187db;
        font-size: 12px;
    }
    .number li:hover{
        color: #173859;
    }
    .number a:hover{
        
    }
    .number i{
        transition: all 0.3s ease-out;
        padding-left: 10px;
    }
    .number a:hover i{
        padding-left: 20px;
    }
    .number:hover li{
        border-color:#DFE3EA
    }
    canvas{
        max-width: 55%;
        min-width: 45%;
    }
</style>

<script>

    export default {
      name: 'indexContent',
      data () {
        return {

        }
      },
      methods:{
        },
        mounted:function(){
            var chartData = [["2017/01", 50], ["2017/02", 60], ["2017/03", 100], ["2017/04",200], ["2017/05",350], ["2017/06",600]];
            goBarChart(chartData);
            
        }
    }
    
    function goBarChart(dataArr){
        
        
    // 聲明所需變量
    var canvas,ctx;
    // 圖表屬性
    var cWidth, cHeight, cMargin, cSpace;
    var originX, originY;
    // 折線圖屬性
    var tobalDots, dotSpace, maxValue;
    var totalYNomber;
    // 運動相關變量
    var ctr, numctr, speed;

    // 得到canvas上下文
    canvas = document.getElementById("barChart");
    if(canvas && canvas.getContext){
        ctx = canvas.getContext("2d");
    }
    initChart(); // 圖表初始化
    drawLineLabelMarkers(); // 繪製圖表軸、標籤和標記
    drawBarAnimate(); // 繪製折線圖的動畫

    //點擊刷新圖表
    canvas.onclick = function(){
        initChart(); // 圖表初始化
        drawLineLabelMarkers(); // 繪製圖表軸、標籤和標記
        drawBarAnimate(); // 繪製折線圖的動畫
    };

    // 圖表初始化
    function initChart(){
        // 圖表信息
        cMargin = 60;
        cSpace = 80;
        canvas.width = Math.floor( (window.innerWidth-100)/2 ) * 2 ;
        canvas.height = 740;
        canvas.style.height = canvas.height/2 + "px";
        canvas.style.width = canvas.width/2 + "px";
        cHeight = canvas.height - cMargin - cSpace;
        cWidth = canvas.width - cMargin - cSpace;
        originX = cMargin + cSpace;
        originY = cMargin + cHeight;

        // 折線圖信息
        tobalDots = dataArr.length;
        dotSpace = parseInt( cWidth/tobalDots );
        maxValue = 0;
        for(var i=0; i<dataArr.length; i++){
            var dotVal = parseInt( dataArr[i][1] );
            if( dotVal > maxValue ){
                maxValue = dotVal;
            }
        }
        maxValue += 50;
        totalYNomber = 10;
        // 運動相關
        ctr = 1;
        numctr = 100;
        speed = 6;

        ctx.translate(0.5,0.5);  // 當只繪製1像素的線的時候,座標點須要偏移,這樣才能畫出1像素實線
    }

    // 繪製圖表軸、標籤和標記
    function drawLineLabelMarkers(){
        ctx.font = "24px Arial";
        ctx.lineWidth = 2;
        ctx.fillStyle = "#566a80";
        ctx.strokeStyle = "#566a80";
        // y軸
        drawLine(originX, originY, originX, cMargin);
        // x軸
        drawLine(originX, originY, originX+cWidth, originY);

        // 繪製標記
        drawMarkers();
    }

    // 畫線的方法
    function drawLine(x, y, X, Y){
        ctx.beginPath();
        ctx.moveTo(x, y);
        ctx.lineTo(X, Y);
        ctx.stroke();
        ctx.closePath();
    }

    // 繪製標記
    function drawMarkers(){
        ctx.strokeStyle = "#E0E0E0";
        // 繪製 y 軸 及中間橫線
        var oneVal = parseInt(maxValue/totalYNomber);
        ctx.textAlign = "right";
        for(var i=0; i<=totalYNomber; i++){
            var markerVal =  i*oneVal;
            var xMarker = originX-5;
            var yMarker = parseInt( cHeight*(1-markerVal/maxValue) ) + cMargin;
            //console.log(xMarker, yMarker+3,markerVal/maxValue,originY);
            ctx.fillText(markerVal, xMarker, yMarker+3, cSpace); // 文字
            if(i>0){
                drawLine(originX+2, yMarker, originX+cWidth, yMarker);
            }
        }
        // 繪製 x 軸 及中間豎線
        ctx.textAlign = "center";
        for(var i=0; i<tobalDots; i++){
            var markerVal = dataArr[i][0];
            var xMarker = originX+i*dotSpace;
            var yMarker = originY+30;
            ctx.fillText(markerVal, xMarker, yMarker, cSpace); // 文字
            if(i>0){
                drawLine(xMarker, originY-2, xMarker, cMargin    );
            }
        }
        // 繪製標題 y
        ctx.save();
        ctx.rotate(-Math.PI/2);
        ctx.fillText("訪問量", -canvas.height/2, cSpace-10);
        ctx.restore();
        // 繪製標題 x
        ctx.fillText("月份", originX+cWidth/2, originY+cSpace/2+20);
    };

    //繪製折線圖
    function drawBarAnimate(){
        ctx.strokeStyle = "#566a80";  //"#49FE79";

        //連線
        ctx.beginPath();
        for(var i=0; i<tobalDots; i++){
            var dotVal = dataArr[i][1];
            var barH = parseInt( cHeight*dotVal/maxValue* ctr/numctr );//
            var y = originY - barH;
            var x = originX + dotSpace*i;
            if(i==0){
                ctx.moveTo( x, y );
            }else{
                ctx.lineTo( x, y );
            }
        }
        ctx.stroke();

        //背景
        ctx.lineTo( originX+dotSpace*(tobalDots-1), originY);
        ctx.lineTo( originX, originY);
        //背景漸變色
        //柱狀圖漸變色
        var gradient = ctx.createLinearGradient(0, 0, 0, 300);
        gradient.addColorStop(0, 'rgba(133,171,212,0.6)');
        gradient.addColorStop(1, 'rgba(133,171,212,0.1)');
        ctx.fillStyle = gradient;
        ctx.fill();
        ctx.closePath();
        ctx.fillStyle = "#566a80";

        //繪製點
        for(var i=0; i<tobalDots; i++){
            var dotVal = dataArr[i][1];
            var barH = parseInt( cHeight*dotVal/maxValue * ctr/numctr );
            var y = originY - barH;
            var x = originX + dotSpace*i;
            drawArc( x, y );  //繪製點
            ctx.fillText(parseInt(dotVal*ctr/numctr), x+15, y-8); // 文字
        }

        if(ctr<numctr){
            ctr++;
            setTimeout(function(){
                ctx.clearRect(0,0,canvas.width, canvas.height);
                drawLineLabelMarkers();
                drawBarAnimate();
            }, speed);
        }
    }

    //繪製圓點
    function drawArc( x, y, X, Y ){
        ctx.beginPath();
        ctx.arc( x, y, 3, 0, Math.PI*2 );
        ctx.fill();
        ctx.closePath();
    }


    }

</script>

 

 而後咱們能夠看到,首頁的效果就出來啦

 

 

 登陸功能完善:

當登陸請求完成之後,若是出錯,就彈出錯誤(咱們本項目沒有封裝模態框,就直接用alert吧),若是正確,就跳轉到首頁

修改後的 login.vue中的 ajax請求代碼以下:

this.$reqs.post("/users/login",{
                        username:this.username,
                        password:this.password
                }).then(function(result){ 
                    //成功
                    if(result.data.err){
                        alert(result.data.err);
                    }else{
                        _this.$router.push({path:'/backIndex/indexContent'});
                    }
                    _this.disablebtn = false;
                    _this.loginText = "登陸";
                    
                }).catch(function (error) {
                    //失敗
                    _this.disablebtn = false;
                    _this.loginText = "登陸"
                });

注:咱們經過  router.push去修改url,做用和原生js的  window.location.href基本一致。

 

退出系統


 功能點:點擊退出,實現退出功能

在backIndex.vue中咱們有一個退出登陸的空方法,咱們在裏面寫退出登陸的請求的代碼,退出成功後跳轉到根目錄(以就算登陸頁面)

            logout(){ //退出系統
                var _this = this;
                this.$reqs.post("/users/logout",{
                    
                }).then(function(result){
                    //成功
                    _this.$router.push({path:'/'});
                }).catch(function (error) {
                    //失敗
                console.log(error)
              });
            }

 

而後在後臺寫接口,在user.js中 登陸的方法後面寫(修改完成後須要重啓node服務)

注:這裏直接清除登陸中設置的 session 就能夠了,(咱們後面會對全部的請求設置攔截,若是session中的用戶信息沒有,再提示用戶未登陸,跳轉到登陸頁面就能夠了)

//退出
router.post('/logout', function(req, res, next) {
    
    req.session.username = ""; //清除session
    req.session.password = "";
    res.end('{"success":"true"}');
});

 

 

到此,咱們就實現了登陸,顯示首頁,退出的基本功能

 

 

 

好啦,今天就講到這裏。下一篇將講解 用戶添加/修改/刪除,表格分頁

 

關注公衆號,博客更新便可收到推送

相關文章
相關標籤/搜索