说明:

本项目使用了 mysql employees数据库,使用了vue + axois + element UI 2.0 ,演示了 单页程序 架构 ,vue router 的使用,axois 使用,以及 element UI 控件的使用。通过这几种技术的组合,实现了对 employee 的增,删。查,改 ,分页操作,展示了在实际项目中,Vue 结合 elementUI 如何在前端项目中使用。

路由

说白了就是,页面的跳转如何控制。

当用户点击了部门信息就需要展示部门信息的;点击了员工信息就需要展示员工的总体信息,点击员工列表中明细信息就需要跳转到该员工的明细信息。如下图所示:

在传统的web程序中,跳转是由连接来控制的,不同的连接可以跳转到具体的页面,也可以在mvc 结构中 ,不同的路由地址,由controller返回不同的view。

在SPA单页程序中,路由一般是由专门的Router 来控制,而且Router是前端的组件,而不是由后端来控制的。

在本项目中,Vue 路由组件 使用的是 Vue-Router,部门,员工列表信息,员工明细信息 ,都是一个个 活生生 vue 组件,是前端组件,而不是一个页面。这种方式也是把web前端开发带入了组件化开发模式,

相对传统的web开发模式,进步可不是一点点。

项目结构:

核心组件介绍:

  • main.js
import Vue from 'vue'

import ElementUI from 'element-ui'

import 'element-ui/lib/theme-chalk/index.css'

import App from './App.vue'

import VueRouter from 'vue-router'

import routerMap from './router.js'

// 引入axios以及element ui中的loading和message组件

import axios from 'axios';

import { Loading, Message } from 'element-ui'

Vue.use(VueRouter);

Vue.use(ElementUI);

Vue.prototype.$http = axios;

//axios 配置最好提出专门的页面

//axios.defaults.baseURL = "http://localhost:5001/api";

axios.defaults.baseURL = "http://localhost/CMS.API/api";

/**

* http配置

*/

// 引入axios以及element ui中的loading和message组件

// 超时时间

axios.defaults.timeout = 5000;

// http请求拦截器

var loadinginstace

axios.interceptors.request.use(config => {

// element ui Loading方法

console.log(config);

loadinginstace = Loading.service({ fullscreen: true })

return config

}, error => {

loadinginstace.close()

Message.error({

message: '加载超时'

})

return Promise.reject(error)

})

// http响应拦截器

axios.interceptors.response.use(data => {// 响应成功关闭loading

loadinginstace.close()

return data

}, error => {

loadinginstace.close()

Message.error({

message: '服务端发生错误'

})

return Promise.reject(error)

})

export default axios

const router = new VueRouter({ routes: routerMap })

const app = new Vue({

router

}).$mount('#app');

axios 拦截器中添加的方法说明:

  1. 当http请求发出后,响应为返回前,前端页面弹出遮罩层,显示loading,避免用户在请求未响应前误操作。
  2. 当http请求发出后,发生异常后,前端提示用户,后台发生错误。
  3. 拦截一次处理了这两种通用的操作,其它地方再也不用以上两种操作。
  • router.js
/*!

//Router Map 文件

//hbb0b0@163.com

*/

import Help from './components/help/Help.vue';

import Feedback from './components/feedback/Feedback.vue';

import UserInfo from './components/business/UserInfo.vue';

import DepartmentList from './components/business/DepartmentList.vue';

import EmployeeList from './components/business/Employee/EmployeeList.vue';

import EmployeeDetail from './components/business/Employee/EmployeeDetail.vue';

import EmployeeAdd from './components/business/Employee/EmployeeAdd.vue';

import EmployeeEdit from './components/business/Employee/EmployeeEdit.vue';

import App from './App.vue'

export default [{

path: '/index',

component: App,

children: [

{

name: '部门信息',

path: 'departmentList',

component: DepartmentList

},

{

name: '员工信息',

path: 'employee/list',

component: EmployeeList

},

{

name: '帮助中心',

path: 'help',

component: Help

},

{

name: '意见反馈',

path: 'feedback',

component: Feedback

},

{

name:'员工详细信息',

path:'employee/detail/:id',

component:EmployeeDetail

},

{

name:'员工信息编辑',

path:'employee/edit/:id',

component:EmployeeEdit

},

{

name:'员工信息增加',

path:'employee/add/',

component:EmployeeAdd

}

]

},

{

path: '*',

redirect: '/index/departmentList'

}

]
  • EmployeeList.vue
<template>

<div class="testUser">

<div class="function">

<el-row>

<el-form :model="queryCondition" label-width="150px" class="common-margin common-form" ref="form" :rules="rules">

<el-form-item label="First Name" prop="param.first_Name">

<el-col :span="6">

<el-input placeholder="First Name" v-model="queryCondition.param.first_Name"></el-input>

</el-col>

</el-form-item>

<el-form-item label="Last Name" prop="param.last_Name">

<el-col :span="6">

<el-input placeholder="Last Name" v-model="queryCondition.param.last_Name"></el-input>

</el-col>

</el-form-item>

<el-form-item label="Gender">

<el-col :span="6">

<el-select placeholder="Gender" v-model="queryCondition.param.gender">

<el-option v-for="item in genderStatus" :key="item.value" :label="item.label" :value="item.value">

</el-option>

</el-select>

</el-col>

</el-form-item>

<el-form-item label="Hire Date">

<el-date-picker format="yyyy-MM-dd" value-format="yyyy-MM-dd" :editable="false" v-model="queryCondition.param.hire_date_range" type="daterange" start-placeholder="start " end-placeholder="end" default-value="1980-01-01">

</el-date-picker>

</el-form-item>

<el-form-item label="Birth Date">

<el-date-picker format="yyyy-MM-dd" value-format="yyyy-MM-dd" :editable="false" v-model="queryCondition.param.birth_date_range" type="daterange" start-placeholder="start" end-placeholder="end" default-value="1950-01-01"></el-date-picker>

</el-form-item>

<el-form-item label="">

<el-button type="primary" icon="el-icon-search" @click="getData()">查询</el-button>

<el-button type="primary" @click="addEmployeeInfo()" icon="el-icon-circle-plus">增加</el-button>

</el-form-item>

</el-form>

</el-row>

</div>

<div style="height: 10px; background-color: rgb(242, 242, 242);"></div>

<div id="table">

<el-table :data="pageList.items" stripe style="width: 100%" border>

<el-table-column prop="emp_No" sortable label="No">

</el-table-column>

<el-table-column prop="first_Name" sortable label="First Name">

</el-table-column>

<el-table-column prop="last_Name" sortable label="Last Name">

</el-table-column>

<el-table-column prop="gender" sortable label="Gender">

</el-table-column>

<el-table-column prop="hire_Date_Display" sortable label="Hire Date">

</el-table-column>

<el-table-column prop="birth_Date_Display" sortable label="Birth Date">

</el-table-column>

<el-table-column label="操作">

<template slot-scope="scope">

<el-button

@click="getDetail(scope.row)"

type="primary"

size="small" icon="el-icon-info">

</el-button>

<el-button

@click="editEmployeeInfo(scope.row)"

type="primary"

size="small" icon="el-icon-edit">

</el-button>

<el-button

@click="deleteEmployeeInfo(scope.row)"

type="primary"

size="small" icon="el-icon-delete">

</el-button>

</template>

</el-table-column>

</el-table>

<div class="block">

<el-pagination :data="pageList" @size-change="handleSizeChange" @current-change="handleCurrentChange" :current-page="queryCondition.pageInfo.pageIndex" :page-sizes="[10,100, 200, 300, 400]" :page-size="queryCondition.pageInfo.pageSize" layout="total, sizes, prev, pager, next, jumper"

:total="pageList.totalCount">

</el-pagination>

</div>

</div>

</div>

</template>

<script>

export default {

data() {

return {

input: "",

pageList: [],

genderStatus: [{

vale: "",

label: ""

},

{

value: "F",

label: "Female"

},

{

value: "M",

label: "Male"

}

],

queryCondition: {

param: {

first_Name: "",

last_Name: "",

gender: "",

hire_date_range: null,

birth_date_range: null,

},

pageInfo: {

pageIndex: 1,

pageSize: 10

}

},

rules: {

'param.first_Name': [{

required: false,

message: "只允许字母或数字",

pattern: /[a-zA-Z0-9]/

}]

}

}

},

mounted: function() {

//debugger;

this.getData();

},

methods: {

handleSizeChange(val) {

//debugger;

//console.log(`每页 ${val} 条`);

this.queryCondition.pageInfo.pageSize = val;

this.getData();

},

handleCurrentChange(val) {

//debugger;

this.queryCondition.pageInfo.pageIndex = val;

this.getData();

},

getData() {

let _self = this;

_self.$refs["form"].validate(function(isValid) {

if (isValid) {

let url = "/Employee/query";

//debugger;

_self.$http

.post(url, _self.queryCondition)

.then(function(response) {

//debugger;

//console.log(response.data.data);

_self.pageList = response.data.data;

})

.catch(function(error) {

console.log(error);

});

} else {

return false;

}

})

},

hire_date_pick(maxDate, minDate) {

//debugger;

alert(maxDate);

},

getDetail(currentRow) {

this.$router.push({

path: '/index/employee/detail/' + currentRow.emp_No

});

},

editEmployeeInfo(currentRow) {

this.$router.push({

path: '/index/employee/edit/' + currentRow.emp_No

});

},

deleteEmployeeInfo(currentRow) {

this.$confirm('此操作将永久删除该记录, 是否继续?', '提示', {

confirmButtonText: '确定',

cancelButtonText: '取消',

type: 'warning'

}).then(() => {

let _self = this;

let url = "/employee/delete/" + currentRow.emp_No;

//debugger;

_self.$http

.post(url)

.then(function(response) {

//debugger;

//console.log(response.data.data);

if (response.data.isSuccess) {

_self.$message({

type: 'success',

message: '删除成功!'

});

_self.getData();

} else {

_self.$message.error("删除失败");

}

})

.catch(function(error) {

console.log(error);

});

}).catch(() => {

this.$message({

type: 'info',

message: '已取消删除'

});

});

},

addEmployeeInfo() {

this.$router.push({

path: '/index/employee/add'

});

}

}

};

</script>

<style scoped>

@import '/static/default.css';

</style>

运行效果:

  • ElementUI table 排序

  • 分页

  • 设置分页大小

  • 提交验证功能

  • 异步验证功能

  • 多表格信息展示

  • 日期选择

  • 时间段选择

  • 确认提示

github 地址 https://github.com/hbb0b0/Hbb0b0.CMS/tree/master/hbb0b0.CMS.Portal

dotnetcore+vue+elementUI 前后端分离 三(前端篇)的更多相关文章

  1. .netcore+vue+elementUI 前后端分离---支持前端、后台业务代码扩展的快速开发框架

    框架采用.NetCore + Vue前后端分离,并且支持前端.后台代码业务动态扩展,框架内置了一套有着20多种属性配置的代码生成器,可灵活配置生成的代码,代码生成器界面配置完成即可生成单表(主表)的增 ...

  2. dotnetcore vue+elementUI 前后端分离架二(后端篇)

    前言 最近几年前后端分离架构大行其道,而且各种框架也是层出不穷.本文通过dotnetcore +vue 来介绍 前后端分离架构实战. 涉及的技术栈 服务端技术 mysql 本项目使用mysql 作为持 ...

  3. 解决vue+springboot前后端分离项目,前端跨域访问sessionID不一致导致的session为null问题

    问题: 前端跨域访问后端接口, 在浏览器的安全策略下默认是不携带cookie的, 所以每次请求都开启了一次新的会话. 在后台打印sessionID我们会发现, 每次请求的sessionID都是不同的, ...

  4. Vue .Net 前后端分离框架搭建

    [参考]IntellIJ IDEA 配置 Vue 支持 打开Vue项目 一.前端开发环境搭建 1.零基础 Vue 开发环境搭建 打开运行Vue项目 2.nodejs http-proxy-middle ...

  5. gin+vue的前后端分离开源项目

    该项目是gin+vue的前后端分离项目,使用gorm访问MySQL,其中vue前端是使用vue-element-admin框架简单实现的: go后台使用jwt,对API接口进行权限控制.此外,Web页 ...

  6. 一套基于SpringBoot+Vue+Shiro 前后端分离 开发的代码生成器

    一.前言 最近花了一个月时间完成了一套基于Spring Boot+Vue+Shiro前后端分离的代码生成器,目前项目代码已基本完成 止步传统CRUD,进阶代码优化: 该项目可根据数据库字段动态生成 c ...

  7. docker-compose 部署 Vue+SpringBoot 前后端分离项目

    一.前言 本文将通过docker-compose来部署前端Vue项目到Nginx中,和运行后端SpringBoot项目 服务器基本环境: CentOS7.3 Dokcer MySQL 二.docker ...

  8. vue+springboot前后端分离实现单点登录跨域问题处理

    最近在做一个后台管理系统,前端是用时下火热的vue.js,后台是基于springboot的.因为后台系统没有登录功能,但是公司要求统一登录,登录认证统一使用.net项目组的认证系统.那就意味着做单点登 ...

  9. Flask & Vue 构建前后端分离的应用

    Flask & Vue 构建前后端分离的应用 最近在使用 Flask 制作基于 HTML5 的桌面应用,前面写过<用 Python 构建 web 应用>,借助于完善的 Flask ...

随机推荐

  1. understand试用笔记一阅读VS2010项目

    一.查看vs2010项目 打开understand,File—New—Project...—Next—Next [向导第三步,选“Import Visual Sudio project files”] ...

  2. shell脚本之正则表达式

    具体参考: www.jb51.net/tools/shell_regex.html 正则表达式常用于grep AWK 等工具中

  3. SQL常用增删改查

    转 http://www.cnblogs.com/daxueshan/p/6687521.html 1增 1.1[插入单行]insert [into] <表名> (列名) values ( ...

  4. Kindeditor图片粘贴上传(chrome)

    kindeditor4.1.x版本已支持图片批量上传,不过传统的选文件上传的方式依然效率低下. 很多时候,编辑人员可能需要将一个文档中图片上传到网上,那么,按照传统的上传方法,他必须先将图片另存为到本 ...

  5. x86_64汇编调试程序初步

    寄存器说明: rdi 存第1个参数(值或地址) rsi 存第2个参数 rdx 存第3个参数 rcx 存第4个参数 r8 存第5个参数 r9 存第6个参数 rax 第1个返回值 rdx 第2个返回值 r ...

  6. 深入浅出javascript(六)对象

    2.为什么一切皆对象? <Javascript权威指南>解释了这个问题,问题的起源在于,如果typeof(字符串)返回的是string,并非object,那么为什么字符串也是对象呢? 简单 ...

  7. _ZNote_Qt_对话框_模态非模态

    QDialog(及其子类,以及所有Qt::Dialog类型的类)的对于其 parent 指针都有额外的解释:如果 parent 为 NULL,则该对话框会作为一个顶层窗口,否则则作为其父组件的子对话框 ...

  8. linux 下载文件

    工作流程 1.tar pczvf file.tar.gz file 2.sz file.tar.gz:下载. 3.rm -i file.tar.gz: 删除.

  9. node linux

    在linux下安装nodejs 教程:http://my.oschina.net/blogshi/blog/260953 连接linux服务器,supervisor bin/www,断开连接,服务器还 ...

  10. ASP.NET Web API 入门 (API接口、寄宿方式、HttpClient调用)

    一.ASP.NET Web API接口定义 ASP.NET Web API默认实现了Action方法和HTTP方法的映射,Action方法方法名体现了其能处理的请求必须采用的HTTP方法 二.寄宿方式 ...