这个是一个电商管理系统的前端项目的笔记,这个项目主要包括登录/退出功能、主页布局、用户管理模块、权限管理模块、分类管理模块、参数管理模块、商品管理模块、
订单管理模块等功能,我把里面部分的功能进行了整理,希望对大家有所帮助! 项目部分效果
管理系统功能
电商后台管理系统用于管理账号、商品分类、商品信息、订单、数据统计等业务功能
项目概述
前端技术栈
- vue
- vue-router
- Element-UI
- Axios
- Echarts
项目部分效果
这是一个登陆页面主要实现了用户登陆表单验证(登陆功能我使用的是线上功能)
这个是实现的首页的内容…
这个实现的用户列表页面的内容…
前端项目初始化
- 安装Vue脚手架
- 通过Vue脚手架创建项目
- 配置Vue路由
- 配置Element-UI组件库
- 配置axios库
登录页面布局
通过Element-UI组件实现布局
- el-form
- el-form-item
- el-input
- el-button
- 字体图标
这个是实现的一个登录页面和表单验证的代码…
<template>
<div class="login_container">
<div class="login_box">
<!-- 头像区 -->
<el-avatar src="https://cube.elemecdn.com/0/88/03b0d39583f48206768a7534e55bcpng.png"></el-avatar>
<!-- 登录表单区 -->
<el-form ref="loginFormRef" :rules="loginFormRules" label-width="0px" class="login_form" :model="loginForm">
<!-- 用户名 -->
<el-form-item prop="username"><el-input prefix-icon="el-icon-user-solid" v-model="loginForm.username"></el-input></el-form-item>
<!-- 密码 -->
<el-form-item prop="password"><el-input type="password" prefix-icon="el-icon-lock" v-model="loginForm.password"></el-input></el-form-item>
<!-- 按钮 -->
<el-form-item class="btns">
<el-button type="primary" @click="login">登录</el-button>
<el-button type="info" @click="resetLoginForm">重置</el-button>
</el-form-item>
</el-form>
</div>
</div>
</template>
<script>
export default {
data() {
return {
// 登录表单的数据绑定对象
loginForm: {
username: '',
password: ''
},
// 表单验证规则
loginFormRules: {
// 验证用户名是否合法
username: [{ required: true, message: '请输入用户名', trigger: 'blur' }, { min: 3, max: 10, message: '长度在3到10个字符', trigger: 'blur' }],
// 验证密码是否合法
password: [{ required: true, message: '请输入密码', trigger: 'blur' }, { min: 6, max: 15, message: '长度在6到15个字符', trigger: 'blur' }]
}
};
},
methods: {
// 重置按钮
resetLoginForm() {
// console.log(this);
this.$refs.loginFormRef.resetFields();
// this.loginForm.username='',
// this.loginForm.password=''
},
login() {
this.$refs.loginFormRef.validate(async valid => {
console.log(valid);
if (!valid) return;
const { data: res } = await this.$http.post('login', this.loginForm);
if (res.meta.status !== 200) return this.$message.error('登录失败!');
this.$message.success('登录成功!');
// 将登陆成功之后的token 保存到客户端的 sessionStorage中
// 项目中除了登录之外的其他API接口,必须在登陆成功之后才能访问
// token只应在当前网站打开期间生效,所以将token保存在sessionStorage中
window.sessionStorage.setItem('token', res.data.token);
//通过编程式导航跳转到后台主页,路由地址是/home
this.$router.push('/home');
});
}
}
};
</script>
<style scoped>
/* .login_container{
background: red;
} */
.login_container {
background: url('../assets/ac.jpg')no-repeat 0 0/cover ;
background-size: 100% 100%;
height: 100%;
}
.login_box {
width: 450px;
height: 300px;
background:rgba(0,0,0,.2);
border-radius: 3px;
position: absolute;
left: 50%;
top: 50%;
/* 位移——横轴移动-50%,纵轴移动-50% */
/* 减去自身一半距离为-50% */
transform: translate(-50%, -50%);
}
.avtar_box {
width: 130px;
height: 130px;
border: 1px solid #eee;
border-radius: 50%;
padding: 10px;
box-shadow: 0 0 10px #ddd;
position: absolute;
left: 50%;
transform: translate(-50%, -50%);
background-color: #fff;
}
.avtar_box img {
width: 100%;
height: 100%;
border-radius: 50%;
background-color: #eee;
}
.btns {
display: flex;
justify-content: flex-end;
}
.login_form {
position: absolute;
bottom: 0;
width: 100%;
padding: 0 20px;
box-sizing: border-box;
}
.el-avatar{
margin-left: 45%;
}
</style>
实现的一个登录页面的效果
实现面包屑导航功能.
<!-- 面包屑导航区 -->
<el-breadcrumb separator-class="el-icon-arrow-right">
<el-breadcrumb-item :to="{ path: '/home' }">首页</el-breadcrumb-item>
<el-breadcrumb-item>用户管理</el-breadcrumb-item>
<el-breadcrumb-item>用户列表</el-breadcrumb-item>
</el-breadcrumb>
实现商品列表分页功能
<!-- 分页区域 -->
<el-pagination
@size-change="handleSizeChange"
@current-change="handleCurrentChange"
:current-page="queryInfo.pagenum"
:page-sizes="[1, 2, 5, 10]"
:page-size="queryInfo.pagesize"
layout="total, sizes, prev, pager, next, jumper"
:total="total">
</el-pagination>
<script>
export default {
return{
// 获取用户列表的参数对象
queryInfo:{
query:'',
// 当前的页数
pagenum:1,
// 当前每页显示多少条数据
pagesize:5
},
}
}
</script>
左侧菜单布局
菜单分成二级,并且可以折叠
实现效果代码…
<el-container>
<!-- 左侧边栏 -->
<el-aside :width=" isCollapse ? '64px' : '200px' ">
<div class="toggle-button" @click="toggleCollapse">|||</div>
<!-- 侧边栏菜单区域 -->
<el-menu background-color="#373d41" text-color="#fff"
active-text-color="#409eff" unique-opened :collapse="isCollapse"
:collapse-transition="false" router :default-active="activePath">
<!-- 一级菜单 -->
<el-submenu :index="item.id + '' " v-for="item in menulist" :key="item.id">
<!-- 一级菜单的模板区域 -->
<template slot="title">
<!-- 图标 -->
<!-- <i class="el-icon-sugar"></i> -->
<i :class="iconsObj[item.id]"></i>
<!-- 文本 -->
<span>{{item.authName}}</span>
</template>
<!-- 二级菜单 -->
<el-menu-item :index="'/'+ subItem.path" v-for="subItem in item.children"
:key="subItem.id" @click="saveNavState('/'+ subItem.path)">
<template slot="title">
<!-- 图标 -->
<i class="el-icon-menu"></i>
<!-- 文本 -->
<span>{{subItem.authName}}</span>
</template>
</el-menu-item>
</el-submenu>
</el-menu>
</el-aside>
<!-- 右侧内容 -->
<el-main>
<router-view></router-view>
</el-main>
</el-container>
<script>
export default {
data(){
return{
// 左侧菜单数据
menulist:[],
iconsObj:{
'125':'el-icon-user-solid',
'103':'el-icon-s-operation',
'101':'el-icon-document',
'102':'el-icon-notebook-1',
'145':'el-icon-loading'
},
// 是否折叠
isCollapse:false,
// 被激活的链接地址
activePath:'',
dialogVisible: false,
}
},
created() {
this.getMenuList()
this.activePath = window.sessionStorage.getItem('activePath')
},
methods: {
logout() {
window.sessionStorage.clear();
this.$router.push('/login');
},
// 获取所有菜单
async getMenuList(){
const {data:res}=await this.$http.get('menus')
if(res.meta.status !== 200) return this.$message.error(res.meta.msg)
this.menulist=res.data
console.log(res)
},
// 点击按钮,切换菜单折叠与展开
toggleCollapse(){
this.isCollapse =! this.isCollapse
},
// 保存链接的激活状态
saveNavState(activePath){
window.sessionStorage.setItem('activePath',activePath)
this.activePath=activePath
}
}
};
</script>
实现用户管理用户列表全部功能…
<template>
<div>
<!-- 卡片视图区域 -->
<el-card>
<!-- 搜索与添加区 -->
<el-row :gutter="20">
<el-col :span="8">
<el-input placeholder="请输入内容" v-model="queryInfo.query"
clearable @clear="getUserList">
<el-button slot="append" icon="el-icon-search" @click="getUserList"></el-button>
</el-input>
</el-col>
<el-col :span="4">
<el-button type="primary" @click="addDialogVisible=true">添加用户</el-button>
</el-col>
</el-row>
<!-- 用户列表区 -->
<el-table :data="userlist" border stripe>
<el-table-column type="index"></el-table-column>
<el-table-column label="姓名" prop="username"></el-table-column>
<el-table-column label="邮箱" prop="email"></el-table-column>
<el-table-column label="电话" prop="mobile"></el-table-column>
<el-table-column label="角色" prop="role_name">
</el-table-column>
<el-table-column label="状态">
<template slot-scope="scope">
<el-switch v-model="scope.row.mg_state" @change="userStateChanged(scope.row)"></el-switch>
</template>
</el-table-column>
<el-table-column label="操作" width="182px">
<template slot-scope="scope">
<!-- 修改按钮 -->
<el-tooltip class="item" effect="dark" content="修改" placement="top" :enterable="false">
<el-button type="primary" icon="el-icon-edit" size="mini" @click="showEditDialog(scope.row.id)"></el-button>
</el-tooltip>
<!-- 删除按钮 -->
<el-tooltip class="item" effect="dark" content="删除用户" placement="top" :enterable="false">
<el-button type="danger" icon="el-icon-delete" size="mini" @click="removeUserByid(scope.row.id)"></el-button>
</el-tooltip>
<!-- 分配角色按钮 -->
<el-tooltip class="item" effect="dark" content="分配角色" placement="top" :enterable="false">
<el-button type="warning" icon="el-icon-setting" size="mini"></el-button>
</el-tooltip>
</template>
</el-table-column>
</el-table>
<!-- 分页区域 -->
<el-pagination
@size-change="handleSizeChange"
@current-change="handleCurrentChange"
:current-page="queryInfo.pagenum"
:page-sizes="[1, 2, 5, 10]"
:page-size="queryInfo.pagesize"
layout="total, sizes, prev, pager, next, jumper"
:total="total">
</el-pagination>
</el-card>
<!-- 添加用户的对话框 -->
<el-dialog title="添加用户" :visible.sync="addDialogVisible" width="50%" @close="addDialogClosed">
<!-- 内容主题区域 -->
<el-form :model="addForm" :rules="addFormRules" ref="addFormRef" label-width="70pxaddDialogClosed">
<el-form-item label="用户名" prop="username">
<el-input v-model="addForm.username"></el-input>
</el-form-item>
<el-form-item label="密码" prop="password" v-model="addForm.password">
<el-input v-model="addForm.password"></el-input>
</el-form-item>
<el-form-item label="邮箱" prop="email">
<el-input v-model="addForm.email"></el-input>
</el-form-item>
<el-form-item label="手机" prop="mobile">
<el-input v-model="addForm.mobile"></el-input>
</el-form-item>
</el-form>
<!-- 底部区域 -->
<span slot="footer" class="dialog-footer">
<el-button @click="addDialogVisible = false">取 消</el-button>
<el-button type="primary" @click="addUSer">确 定</el-button>
</span>
</el-dialog>
<!-- 修改用户的对话框 -->
<el-dialog
title="修改用户"
:visible.sync="editDiaglogVisible"
width="50%" @close="editDialogClosed">
<el-form :model="editForm" :rules="editFormRules" ref="editFormRef" label-width="70px">
<el-form-item label="用户名">
<el-input v-model="editForm.username" disabled></el-input>
</el-form-item>
<el-form-item label="邮箱" prop="email">
<el-input v-model="editForm.email"></el-input>
</el-form-item>
<el-form-item label="手机号" prop="mobile">
<el-input v-model="editForm.mobile"></el-input>
</el-form-item>
</el-form>
<span slot="footer" class="dialog-footer">
<el-button @click="editDiaglogVisible = false">取 消</el-button>
<el-button type="primary" @click="editUserInfo">确 定</el-button>
</span>
</el-dialog>
</div>
</template>
<script>
export default {
data() {
// 验证邮箱的正则
var checkEmail = (rule,value,cb) =>{
// 正则——邮箱
const regEmail = /^([a-zA-Z0-9_-])+@([a-zA-Z0-9_-])+(\.[a-zA-Z0-9_-])+/
if(regEmail.test(value)){
// 合法的邮箱
return cb()
}
cb(new Error('请输入合法的邮箱'))
}
// 验证手机号的正则
var checkMobile = (rule,value,cb) =>{
// 正则——手机号
const regMobile = /^(0|86|17951)?(13[0-9]|15[0123456789]|17[678]|18[0-9]|14[57])[0-9]{8}$/
if(regMobile.test(value)){
// 合法的手机号
return cb()
}
cb(new Error('请输入合法的手机号'))
}
return{
// 获取用户列表的参数对象
queryInfo:{
query:'',
// 当前的页数
pagenum:1,
// 当前每页显示多少条数据
pagesize:5
},
userlist:[],
total:0,
// 控制添加用户对话框的显示与隐藏
addDialogVisible:false,
// 添加用户的表单数据
addForm:{
username:'',
password:'',
email:'',
mobile:''
},
//添加表单的验证规则对象
addFormRules:{
username:[
{ required: true, message: '输入用户名', trigger: 'blur' },
{ min: 3, max: 10, message: '长度在 3 到 10 个字符', trigger: 'blur' }
],
password:[
{ required: true, message: '输入密码', trigger: 'blur' },
{ min: 6, max: 15, message: '长度在 6 到 15 个字符', trigger: 'blur' }
],
email:[
{ required: true, message: '输入邮箱', trigger: 'blur' },
{ validator: checkEmail, trigger: 'blur' }
],
mobile:[
{ required: true, message: '输入手机号', trigger: 'blur' },
{ validator: checkMobile, trigger: 'blur' }
]
},
// 控制修改用户对话框的显示与隐藏
editDiaglogVisible:false,
// 查询到的用户信息保存
editForm:{},
editFormRules:{
email:[
{ required: true, message: '请输入用户邮箱', trigger: 'blur'},
{ validator:checkEmail,trigger:'blur'}
],
mobile:[
{ required: true, message: '请输入用户手机号', trigger: 'blur'},
{ validator:checkMobile,trigger:'blur'}
]
}
}
},
created() {
this.getUserList()
},
methods: {
async getUserList(){
const{data:res} =await this.$http.get('users',{
params:this.queryInfo
})
if(res.meta.status !== 200){
return this.$message.error('获取用户列表失败!')
}
this.userlist=res.data.users
this.total=res.data.total
console.log(res)
},
// 监听pagesize改变的事件
handleSizeChange(newSize){
// console.log(newSize)
this.queryInfo.pagesize=newSize
this.getUserList()
},
// 监听页码值改变的事件
handleCurrentChange(newPage){
// console.log(newPage)
this.queryInfo.pagenum=newPage
this.getUserList()
},
// 监听switch状态的改变
async userStateChanged(userinfo){
console.log(userinfo)
await this.$http.put(`users/${userinfo.id}/state/state/${userinfo.mg_state}`)
if(res.meta.status !== 200){
userinfo.mg_state=!user.mg_state
return this.$message.error('更新用户状态!')
}
this.$message.success('更新用户状态成功')
},
// 监听添加用户对话框的关闭事件
addDialogClosed() {
this.$refs.addFormRef.resetFields()
},
// 点击按钮添加新用户
addUSer(){
this.$refs.addFormRef.validate(async valid => {
if(!valid) return
// 可以发起添加用户的网络请求
const {data:res} = await this.$http.post('users',this.addForm)
if(res.meta.status !== 201){
this.$message.error('添加用户失败!')
}
this.$message.success('添加用户成功!')
// 隐藏添加用户的对话框
this.addDialogVisible = false
// 重新获取用户列表数据
this.getUserList()
})
},
// 展示用户编辑的对话框
async showEditDialog(id){
const {data: res} = await this.$http.get('users/' + id)
console.log(res)
if(res.meta.status !== 200){
return this.$message.error('查询用户信息失败!')
}
this.editForm = res.data
this.editDiaglogVisible = true
},
// 监听修改用户对话框的关闭事件
editDialogClosed(){
this.$refs.editFormRef.resetFields()
},
// 修改用户信息并提交
editUserInfo(){
this.$refs.editFormRef.validate(async valid => {
if(!valid) return
// 发起修改用户信息的数据请求
const {data:res} = await this.$http.put('users/' + this.editForm.id,{
email:this.editForm.email,mobile:this.editForm.mobile
})
if(res.meta.status !== 200) {
return this.$message.error('更新用户信息失败')
}
// 关闭对话框
this.editDiaglogVisible=false
// 刷新数据列表
this.getUserList()
// 提示修改成功
this.$message.success('更新用户信息成功!')
})
},
// 根据id删除对应的用户信息
async removeUserByid(id){
// 弹框询问用户是否删除数据
const confirmResult = await this.$confirm('此操作将永久删除该用户, 是否继续?', '提示', {
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning'
}).catch(err => err)
// 如果用户确认删除,返回值为字符串confirm
// 如果用户取消删除,则返回值为字符串cancel
if(confirmResult !== 'confirm'){
return this.$message.info('已取消删除')
}
const {data:res} = await this.$http.delete('users/' + id)
if(res.meta.status !== 200){
return this.$message.error('删除用户失败!')
}
this.$message.success('删除用户成功!')
this.getUserList()
}
}
}
</script>
<style scoped>
.el-card{
margin-top: 10px;
}
.el-pagination{
margin-top: 10px;
}
.el-table{
margin-top: 15px;
}
</style>
总结
通过本次从零一行一行代码实现一个完整的常见的后台管理系统,其实绝大部分就是对Element
UI的使用。说是简单、实则坑很多,期间遇到了不少的bug,大多都是因为细节不注意,也让我更加体会 好记性不如烂笔头
这句话,实践才是真理啊,多动手,多探索。
最后
希望看到这篇文章的小伙伴都能有所收获~
欢迎大家加入,一起学习前端,共同进步!