Vue电商后台管理系统部分效果

这个是一个电商管理系统的前端项目的笔记,这个项目主要包括登录/退出功能、主页布局、用户管理模块、权限管理模块、分类管理模块、参数管理模块、商品管理模块、
订单管理模块等功能,我把里面部分的功能进行了整理,希望对大家有所帮助! 项目部分效果

管理系统功能

电商后台管理系统用于管理账号、商品分类、商品信息、订单、数据统计等业务功能

项目概述

前端技术栈
  • vue
  • vue-router
  • Element-UI
  • Axios
  • Echarts
项目部分效果

这是一个登陆页面主要实现了用户登陆表单验证(登陆功能我使用的是线上功能)

Vue电商后台管理系统部分效果

这个是实现的首页的内容…

Vue电商后台管理系统部分效果

这个实现的用户列表页面的内容…

Vue电商后台管理系统部分效果

前端项目初始化
  • 安装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>

实现的一个登录页面的效果

Vue电商后台管理系统部分效果

实现面包屑导航功能.

Vue电商后台管理系统部分效果

<!-- 面包屑导航区 -->
<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>

实现商品列表分页功能

Vue电商后台管理系统部分效果

<!-- 分页区域 -->
<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>

左侧菜单布局

菜单分成二级,并且可以折叠

Vue电商后台管理系统部分效果

实现效果代码…

<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>

实现用户管理用户列表全部功能…

Vue电商后台管理系统部分效果

<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,大多都是因为细节不注意,也让我更加体会 好记性不如烂笔头
这句话,实践才是真理啊,多动手,多探索。

最后

希望看到这篇文章的小伙伴都能有所收获~

欢迎大家加入,一起学习前端,共同进步!

上一篇:Spring Boot项目中使用Quartz完成定时任务


下一篇:用python创建一个最简单的触发器trigger