vue2 + vue-router + Vuex + axios + elementui + echarts
目录
技术
serve
1.db数据库连接
2.user.js
3.app.js
router
1. permission.js路由拦截
store(vuex状态管理)
userModule.js
api和utils
1.request.js请求根路径
2.SessionStorage.js(数据持久化)
3.接口api文件,Goods的index.js
登录页面
1.LoginView..vue
Layout布局界面
1. MyContent.vue(动态面包屑)
相关知识点
项目准备
项目搭建
项目初始化
后台服务 (serve文件)
路由配置
设置代理proxy实现跨域
登录
数据可视化
mockjs (模拟数据)
商品管理界面
上传图片
富文本编译
国际化
vuei18n
element国际化
vue-pdf
vue项目实现表格导出Excel表格
项目完整代码:
// const mysql = require('mysql')
import mysql from 'mysql'
const client = mysql.createConnection({
host: 'localhost', // 数据域名
user: 'root',
password: 'admin123',
database: 'my_db_01',
port: '3306'
})
// 封装数据库操作语句
export function sqlFn (sql, arr, callback) {
client.query(sql, arr, function (error, result) {
if (error) {
console.log('数据库语句错误')
return
}
callback(result)
})
}
import express from 'express'
import { sqlFn } from '../db/index.js'
import Mock from 'mockjs'
import sercet from '../sercet.js'
import jwt from 'jsonwebtoken'
// 图片需要的模块
import fs from 'fs'
import path from 'path'
const router = new express.Router()
/* 配置路由 */
/* 登录接口: */
router.post('/login', (req, res) => {
const { username, password } = req.body
// console.log(username,password);
// 请求数据库
const sql = 'select * from ev_user where username=? and password=?'
const arr = [username, password]
sqlFn(sql, arr, result => {
if (result.length > 0) {
const token = jwt.sign({
username: result[0].username,
id: result[0].id
}, sercet.key, {
expiresIn: 20 * 1
})
res.send({
status: 200,
success: true,
message: '登录成功',
token: 'Bearer ' + token,
data: token
})
} else {
res.send({ status: 404, message: '信息错误' })
}
})
})
// 注册接口
router.post('/register', (req, res) => {
const {
username,
password
} = req.body
const sql = 'insert in ev_users values(null,?,?)'
const arr = [username, password]
sqlFn(sql, arr, (result) => {
if (result.attectedRows > 0) {
res.send({
msg: '注册成功'
})
} else {
res.status(401).json({
error: '用户名密码错误'
})
}
})
})
// 商品列表
router.get('/pojectList', (req, res) => {
const page = parseInt(req.query.pagesize || 1)
const sqlLen = 'select * from sp_goods where goods_id'
sqlFn(sqlLen, null, data => {
const len = data.length
const sql = 'select * from sp_goods order by goods_id desc limit 8 offset ' + (page - 1) * 8
sqlFn(sql, null, result => {
if (result.length > 0) {
res.send({
status: 200,
data: result,
pagesize: 8,
total: len
})
} else {
res.send({
status: 500,
msg: '暂无数据'
})
}
})
})
})
// 商品规格列表
router.get('/specificationsList', (req, res) => {
const page = parseInt(req.query.pagesize || 1)
const sqlLen = 'select * from sp_goods where goods_id'
sqlFn(sqlLen, null, data => {
const len = data.length
const sql = 'select * from sp_attribute order by attr_id desc limit 8 offset ' + (page - 1) * 8
sqlFn(sql, null, result => {
if (result.length > 0) {
res.send({
status: 200,
data: result,
pagesize: 8,
total: len
})
} else {
res.send({
status: 500,
msg: '暂无数据'
})
}
})
})
})
// 类目选择接口,参数cid:id
router.get('/category', (req, res) => {
const data = Mock.mock({
status: 200,
// 生成list字段:数组类型,内容是6个数据
'result|4': [
{
'id|+1': 1, // id自增
'name|+1': [// 一次获取一个数值,累次向下推
'电视',
'大屏',
'OLED',
'空调',
'冷风'
],
'cid|+1': 10001
}
]
})
console.log(req.query)
if (req.query.cid === 4) {
res.send({
status: 500
})
} else {
res.send(data)
}
})
router.post('/upload', (req, res) => {
const oldName = req.files[0].path// 获取名字
// 给新名字加上原来的后缀
const newName = req.files[0].path + path.parse(req.files[0].originalname).ext
fs.renameSync(oldName, newName)// 改图片的名字
res.send({
err: 0,
url:
'http://localhost:1314/upload/' +
req.files[0].filename +
path.parse(req.files[0].originalname).ext// 该图片的预览路径
})
})
// 测试mockjs数据
router.get('/test', (req, res) => {
// 使用mock生成数据
const data = Mock.mock({
info: '我是一个单纯的对象',
status: 200,
// 生成list字段:数组类型,内容是6个数据
'list|6': [
{
'id|+1': 1, // id自增
'flag|1-2': true,
'city|1': { // 获取城市数据
310000: '上海市',
320000: '江苏省',
330000: '浙江省',
340000: '安徽省',
110000: '北京市',
130000: '河北省'
},
'arr|+1': [// 一次获取一个数值,累次向下推
'AMD',
'CMD',
'UMD',
'BMD',
'aaa',
'ccc'
],
desc: '@cword(20,80)', // 随机汉字
imgUrl: '@image'
}
]
})
res.send(data)
})
// module.exports=router
export default router
import express from 'express'
import userRouter from './router/user.js'
// 图片需要的模块
import multer from 'multer'
const app = express()
/* post接受参数配置 配置解析表单数据的中间件 */
app.use(express.json()) // for parsing application/json
app.use(express.urlencoded({ extended: true })) // for parsing application/x-www-form-urlencoded
/* 设置跨域 */
app.use(function (req, res, next) {
res.header('Access-Control-Allow-Origin', '*')
res.header('Access-Control-Allow-Headers', 'Content-Type,Content-Length,Authorization,Accept,X-Requested-with,yourHeaderFeild')
res.header('Access-Control-Allow-Methods', 'PUT,POST,GET,DELETE,OPTIONS')
next()
})
const objMulter = multer({ dest: './public/upload/' })
// 实例化multer,传递的参数对象,dest表示上传文件的存储路径
app.use(objMulter.any())// any表示任意类型的文件
// app.use(objMulter.image())//仅允许上传图片类型
app.use(express.static('./public'))
app.use('/api', userRouter)
/* 配置错误级别中间件,捕获并处理Token认证失败的结果 */
app.use(function (err, req, res, next) {
if (err.name === 'UnauthorizedError') return res.send({ status: 1, message: err.message })
})
app.listen(1314, () => {
console.log('Express serve running at http://localhost:1314')
})
import store from '../store/index.js'
import router from './index.js'
// 路由拦截
router.beforeEach((to, from, next) => {
// 判断是否用户是否登录,用循环检查每一个路由元信息有无isLogin等true
if (to.matched.some(ele => ele.meta.isLogin)) {
// 2.判断当前用户是否已经登录
const token = store.state.userModule.userinfo.token
if (token) {
next()
} else {
next('/login')
}
} else {
next() // 不需要登录
}
})
// user模块数据
export default {
// 开启命名空间
namespaced: true,
state: () => ({
userinfo: {
username: '',
token: ''
}
}),
getters: {
},
mutations: {
// 设置用户信息
setUser (state, value) {
state.userinfo = value
},
// 清空信息
cleanUser (state) {
state.userinfo.username = ''
state.userinfo.token = ''
}
},
actions: {
}
}
import axios from 'axios'
const request = axios.create({
// 指定请求的根路径
baseURL: '/api',
timeout: 3000
})
export default request
import store from '../store/index.js'
// 持久化
let userinfo = sessionStorage.getItem('userinfo')
if (userinfo) {
userinfo = JSON.parse(userinfo)
store.commit('userModule/setUser', userinfo)
}
import request from '@/utils/request.js'
// 获取商品列表
export function getGoodsList (pagesize) {
return request.get('/api/pojectList', {
params: {
pagesize
}
})
}
// 获取类目选择
export function getCategory (cid) {
return request.get('/api/category', {
params: {
cid
}
})
}
欢迎光临 Pinia!
后台管理系统
欢迎回来
账号密码登录
提交
重置
{{item.meta.title}}
语言切换
中文
English
欢迎:{{userinfo.username}}
退出登录
iconfont图标
创建插件可以按需循环注册element.js,在main.js调用这个方法即可
reset.css(淘宝方案),在main.js导入
先下载,再在main.js引入
解析token: 安装jwt-decode
页面布局配置 同级登录页面
echarts5
1.安装 npm install echarts -S
2.使用方式
1.导入echarts在组件内使用
2.导入全局 挂载原型上 全局使用(该项目)
3.开发成vue插件
3.使用main.js
1.安装:npm i mockjs -S
2.引入:
node.js:const Mock =require('mockjs')
前端js:import Mock from 'mockjs'
3.语法:
Mock.mock({
})
1.element-ui upload组件
2.实现后台支持
上传图片接口 post请求
说明:
1.后台安装multer模块,同时引入fs模块
在app.js进行导入注册
import multer from 'multer'
let objMulter = multer({ dest: "./public/upload/" });
//实例化multer,传递的参数对象,dest表示上传文件的存储路径
app.use(objMulter.any())//any表示任意类型的文件
// app.use(objMulter.image())//仅允许上传图片类型
// 静态资源托管
app.use(express.static("./public"));
2.router.js入口文件导入模块
import fs from 'fs'
import path from 'path'
3.上传图片
1.百度编译器(较大)
2.wangEditor(轻量)
wangEditor使用步骤:
1.官网:https://www.wangeditor.com/doc/
2.基本使用
1.安装:npm i wangeditor -S
2.引入模块 :
import E from "wangeditor";//导入组件
3.使用
// 相当于js的变量设置
data() {
return {
editor: null,
}}
//methods里创建调用、或是mounted里面直接生产
this.editor = new E(_this.$refs.editorElem);//获取组件并构造编辑器
this.editor.create(); // 创建富文本实例
3.常用配置
(v5版本)
1.清空编辑器内容
editor.clear()
2.获取内容/获取非格式化的 html
editor.getHtml()
3.获取当前编辑器的纯文本内容
const text = editor.getText()
4.重置编辑器的 HTML 内容。(只能解析 editor.getHtml() 返回的 HTML 格式,不支持自定义 HTML 格式。)
editor.setHtml('hello
')
5.配置菜单
1.要配置哪个菜单,首先要知道这个菜单的 key 。执行 editor.getAllMenuKeys() 可获取编辑器所有菜单,从中找到自己想要的菜单 key 即可。
(v4版本)
1.清空内容
editor.txt.clear()
2.获取html内容
editor.txt.html()
3.设置内容
创建编辑器之后,使用 editor.txt.html(...) 设置编辑器内容。
4.配置菜单(编辑器创建之前)
editor.config.menus
使用 editor.config.menus 定义显示哪些菜单和菜单的顺序。
5.配置onchange回调---可获取内容
配置 onchange 函数之后,用户操作(鼠标点击、键盘打字等)导致的内容变化之后,会自动触发 onchange 函数执行。
vuei18n
1.介绍:Vue I18n 是 Vue.js 的国际化插件。它可以轻松地将一些本地化功能集成到你的 Vue.js 应用程序中。
2. 安装
npm install [email protected] -S
main.js导入或者单独的文件
import Vue from 'vue'
import VueI18n from 'vue-i18n'
Vue.use(VueI18n)
3.使用步骤
1. 如果使用模块系统 (例如通过 vue-cli),则需要导入 Vue 和 VueI18n ,然后调用 Vue.use(VueI18n)
import Vue from 'vue'
import VueI18n from 'vue-i18n'
Vue.use(VueI18n)
2.准备翻译的语言环境信息
const messages = {
en: {//英文
home: {
hello: 'hello world',
},
goods:{
}
},
zh: {//中文
home: {
hello: '你好 世界',
},
goods:{
}
}
}
3.通过选项创建 VueI18n 实例
const i18n = new VueI18n({
locale: 'en', // 设置地区
messages, // 设置地区信息
})
4.通过 `i18n` 选项创建 Vue 实例
new Vue({ i18n }).$mount('#app')
5.使用语法
{{ $t("home.hello") }}
element国际化
1.导入
import Element from 'element-ui'
2.导入语言环境
import enLocale from 'element-ui/lib/locale/lang/en'
import zhLocale from 'element-ui/lib/locale/lang/zh-CN'
目前 Element 内置了以下语言:
简体中文(zh-CN)
英语(en)
德语(de)
葡萄牙语(pt)
西班牙语(es)
丹麦语(da)
法语(fr)
挪威语(nb-NO)
繁体中文(zh-TW)
意大利语(it)
韩语(ko)
3.配置语言环境
const messages = {
en: {
message: 'hello',
...enLocale // 或者用 Object.assign({ message: 'hello' }, enLocale)
},
zh: {
message: '你好',
...zhLocale // 或者用 Object.assign({ message: '你好' }, zhLocale)
}
}
// Create VueI18n instance with options
const i18n = new VueI18n({
locale: 'en', // set locale
messages, // set locale messages
})
4.配置使用
Vue.use(Element, {
i18n: (key, value) => i18n.t(key, value)
})
vue-pdf
npm install vue-pdf