第二章 建議學習時間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"}'); });
到此,咱們就實現了登陸,顯示首頁,退出的基本功能
好啦,今天就講到這裏。下一篇將講解 用戶添加/修改/刪除,表格分頁
關注公衆號,博客更新便可收到推送