#8

Closed
pgf2eulj7 wants to merge 0 commits from branch4 into branch1

@ -1,33 +1,48 @@
<template> <template>
<!-- 根元素应用的主容器 -->
<div id="app"> <div id="app">
<!-- 使用fade过渡效果来切换路由视图 -->
<transition name="fade"> <transition name="fade">
<router-view /> <router-view /> <!-- 路由出口用于渲染匹配的组件 -->
</transition> </transition>
</div> </div>
</template> </template>
<style> <style>
/* 样式定义 */
/* 定义小图片的样式 */
img.image-sm { img.image-sm {
max-width: 80px; max-width: 80px; /* 最大宽度 */
max-height: 80px; max-height: 80px; /* 最大高度 */
} }
/* 为el-col内的el-select和el-date-editor设置宽度 */
.el-col .el-select, .el-col .el-select,
.el-col .el-date-editor { .el-col .el-date-editor {
width: 100%; width: 100%; /* 宽度设置为父容器的100% */
} }
/* 设置表格展开行的样式 */
.demo-table-expand { .demo-table-expand {
font-size: 0; font-size: 0; /* 将字体大小设为0常用于清除子元素的默认间隙 */
} }
/* 设置表格展开行内label的样式 */
.demo-table-expand label { .demo-table-expand label {
width: 90px; width: 90px; /* 宽度 */
color: #99a9bf; color: #99a9bf; /* 文字颜色 */
} }
/* 设置el-form-item在表格展开行中的样式 */
.demo-table-expand .el-form-item { .demo-table-expand .el-form-item {
margin-right: 0; margin-right: 0; /* 右侧外边距 */
margin-bottom: 0; margin-bottom: 0; /* 底部外边距 */
width: 50%; width: 50%; /* 宽度设置为父容器的50% */
} }
/* 定义警告文本的样式 */
.text-warning { .text-warning {
color: #e6a23c; color: #e6a23c; /* 文字颜色 */
} }
</style> </style>

@ -1,32 +1,68 @@
import Vue from 'vue' // 导入Vue框架
import App from './App.vue' import Vue from 'vue';
import router from './router'
import store from './store' // 导入根组件
import VueCookie from 'vue-cookie' import App from './App.vue';
// 导入Vue Router实例用于页面路由管理
import router from './router';
// 导入Vuex状态管理实例
import store from './store';
// 导入vue-cookie插件用于处理cookie
import VueCookie from 'vue-cookie';
// 导入Element UI组件库
import ElementUI from 'element-ui'; import ElementUI from 'element-ui';
import moment from 'moment'
// 导入moment.js库用于日期和时间处理
import moment from 'moment';
// 导入Element UI的CSS样式
import 'element-ui/lib/theme-chalk/index.css'; import 'element-ui/lib/theme-chalk/index.css';
import './assets/css/common.css'
import './assets/scss/index.scss'
import httpRequest from '@/utils/httpRequest' // api: https://github.com/axios/axios
import { isAuth } from '@/utils'
import VueClipboard from 'vue-clipboard2'
// 导入全局CSS样式
import './assets/css/common.css';
// 导入全局SCSS样式
import './assets/scss/index.scss';
// 导入自定义的httpRequest工具基于axios的封装注意这里注释的api链接可能不准确因为httpRequest的具体实现未在代码中给出
import httpRequest from '@/utils/httpRequest';
// 导入自定义的权限验证工具
import { isAuth } from '@/utils';
// 导入vue-clipboard2插件用于复制文本到剪贴板
import VueClipboard from 'vue-clipboard2';
// 使用Element UI组件库
Vue.use(ElementUI); Vue.use(ElementUI);
Vue.use(VueClipboard)
Vue.use(VueCookie)
Vue.config.productionTip = false
// 挂载全局 // 使用vue-clipboard2插件
Vue.prototype.$http = httpRequest // ajax请求方法 Vue.use(VueClipboard);
Vue.prototype.isAuth = isAuth // 权限方法
// 使用vue-cookie插件
Vue.use(VueCookie);
// 禁止Vue在启动时生成生产提示
Vue.config.productionTip = false;
// 挂载全局属性和方法
// 全局ajax请求方法
Vue.prototype.$http = httpRequest;
// 全局权限验证方法
Vue.prototype.isAuth = isAuth;
// 设置moment.js的语言环境为中文
moment.locale('zh-cn'); moment.locale('zh-cn');
Vue.prototype.$moment = moment; //时间处理 // 将moment挂载到Vue原型上作为全局的时间处理方法
Vue.prototype.$moment = moment;
// 创建Vue实例并挂载到#app元素上
new Vue({ new Vue({
router, router, // 注入路由使得我们可以通过this.$router访问路由实例
store, store, // 注入store使得我们可以通过this.$store访问状态管理实例
render: h => h(App) render: h => h(App) // 渲染App组件
}).$mount('#app') }).$mount('#app'); // 挂载到DOM上的#app元素

@ -1,5 +1,7 @@
import Vue from 'vue' import Vue from 'vue'
import Vuex from 'vuex' import Vuex from 'vuex'
// 引入各个模块的 Vuex 状态管理文件
import common from './modules/common' import common from './modules/common'
import user from './modules/user' import user from './modules/user'
import article from './modules/article' import article from './modules/article'
@ -7,18 +9,26 @@ import message from './modules/message'
import wxUserTags from './modules/wxUserTags' import wxUserTags from './modules/wxUserTags'
import wxAccount from './modules/wxAccount' import wxAccount from './modules/wxAccount'
// 注册 Vuex 插件,使其可用于 Vue 中
Vue.use(Vuex) Vue.use(Vuex)
export default new Vuex.Store({ export default new Vuex.Store({
modules: { // 使用 modules 来组织不同的子模块
common, modules: {
user, // 引入并注册各个模块
article, common, // 公共模块,可能用于存储一些通用的状态
message, user, // 用户模块,存储用户信息相关的状态
wxUserTags, article, // 文章模块,存储与文章相关的状态
wxAccount message, // 消息模块,存储消息相关的状态
}, wxUserTags, // 微信用户标签模块,管理微信用户标签的状态
mutations: { wxAccount // 微信账号模块,管理微信账号相关的状态
}, },
strict: true
// mutations 用于同步修改状态,这里没有定义任何 mutations可根据需求进行扩展
mutations: {
// 这里可以添加全局的 mutation但目前没有定义
},
// 启用严格模式,开发环境下会对状态的修改进行检查,确保只能通过 mutation 修改状态
strict: true
}) })

@ -1,32 +1,59 @@
import Vue from 'vue' import Vue from 'vue';
export default { export default {
namespaced: true, // Vuex模块启用命名空间
state: { namespaced: true,
ACCOUNT_TYPES:{
1:'订阅号', // Vuex的state用来存储应用状态
2:'服务号' state: {
}, // 账户类型映射数字ID到账户类型名称
accountList:[], ACCOUNT_TYPES: {
selectedAppid:'' 1: '订阅号',
}, 2: '服务号'
mutations: { },
updateAccountList (state, list) {
state.accountList = list // 存储账户列表
if(!list.length)return accountList: [],
if(!state.selectedAppid){
let appidCookie = Vue.cookie.get('appid') // 当前选中的Appid用来标识选择的账号
let selectedAppid = appidCookie?appidCookie:list[0].appid selectedAppid: ''
this.commit('wxAccount/selectAccount',selectedAppid) },
}
}, // Vuex的mutations用来修改state
selectAccount (state, appid) { mutations: {
Vue.cookie.set('appid',appid) // 更新账户列表
let oldAppid = state.selectedAppid updateAccountList (state, list) {
state.selectedAppid = appid // 更新state中的accountList
if(oldAppid){//切换账号时刷新网页 state.accountList = list;
location.reload();
} // 如果列表为空,直接返回
}, if (!list.length) return;
}
} // 如果当前没有选中的Appid则从cookie或列表中选择一个默认Appid
if (!state.selectedAppid) {
let appidCookie = Vue.cookie.get('appid');
// 获取cookie中的appid如果有则使用cookie中的appid否则使用列表中的第一个Appid
let selectedAppid = appidCookie ? appidCookie : list[0].appid;
// 通过commit调用mutation更新选中的账号
this.commit('wxAccount/selectAccount', selectedAppid);
}
},
// 选择某个账号切换Appid
selectAccount (state, appid) {
// 更新cookie中的appid保存选中的Appid
Vue.cookie.set('appid', appid);
// 记录上一个选中的Appid
let oldAppid = state.selectedAppid;
// 更新当前选中的Appid
state.selectedAppid = appid;
// 如果选中的Appid发生变化则刷新页面
if (oldAppid) {
location.reload();
}
}
}
};

@ -1,12 +1,20 @@
export default { export default {
namespaced: true, // 启用 Vuex 模块的命名空间,避免命名冲突
state: { namespaced: true,
tags:[]
}, // state 存储模块的状态
mutations: { state: {
updateTags (state, tags) { // tags 用来存储标签数据的数组
state.tags = tags tags: []
} },
}
// mutations 用来修改 state 中的状态
mutations: {
// 更新 tags 数组的内容
updateTags (state, tags) {
// 将传入的 tags 更新到 state 中
state.tags = tags;
} }
}
};

@ -4,74 +4,81 @@ import router from '@/router'
import qs from 'qs' import qs from 'qs'
import merge from 'lodash/merge' import merge from 'lodash/merge'
import { clearLoginInfo } from '@/utils' import { clearLoginInfo } from '@/utils'
const baseUrl = '/wx'
const baseUrl = '/wx' // 设置请求的基础路径
// 创建axios实例
const http = axios.create({ const http = axios.create({
timeout: 1000 * 30, timeout: 1000 * 30, // 设置请求超时为30秒
withCredentials: true, withCredentials: true, // 允许携带跨域请求的cookie
headers: { headers: {
'Content-Type': 'application/json; charset=utf-8' 'Content-Type': 'application/json; charset=utf-8' // 默认请求头为json格式
} }
}) })
/** /**
* 请求拦截 * 请求拦截器
*/ * 在每个请求发送之前加入token从cookie中获取
*/
http.interceptors.request.use(config => { http.interceptors.request.use(config => {
config.headers['token'] = Vue.cookie.get('token') // 请求头带上token config.headers['token'] = Vue.cookie.get('token') // 在请求头中加入token
return config return config // 返回请求配置
}, error => { }, error => {
return Promise.reject(error) return Promise.reject(error) // 请求出错时返回Promise拒绝
}) })
/** /**
* 响应拦截 * 响应拦截器
*/ * 对响应数据进行拦截处理
* 如果返回的状态码为401未授权则清除登录信息并跳转到登录页
*/
http.interceptors.response.use(response => { http.interceptors.response.use(response => {
if (response.data && response.data.code === 401) { // 401, token失效 if (response.data && response.data.code === 401) { // 判断返回的code是否为401代表token失效
clearLoginInfo() clearLoginInfo() // 清除登录信息
router.push({ name: 'login' }) router.push({ name: 'login' }) // 跳转到登录页面
} }
return response return response // 返回响应数据
}, error => { }, error => {
return Promise.reject(error) return Promise.reject(error) // 响应出错时返回Promise拒绝
}) })
/** /**
* 请求地址处理 * 请求地址处理函数
* @param {*} actionName action方法名称 * @param {*} actionName 接口的名称拼接成完整的URL
*/ * @returns {string} 拼接后的完整URL
*/
http.adornUrl = (actionName) => { http.adornUrl = (actionName) => {
// 非生产环境 && 开启代理, 接口前缀统一使用[/proxyApi/]前缀做代理拦截! // 在开发环境下,如果开启了代理,则请求路径会带上代理前缀
return baseUrl + actionName return baseUrl + actionName // 返回完整的请求URL
} }
/** /**
* get请求参数处理 * get请求的参数处理
* @param {*} params 参数对象 * @param {*} params 请求的参数对象
* @param {*} openDefultParams 是否开启默认参数? * @param {*} openDefultParams 是否开启默认参数
*/ * @returns {object} 处理后的参数对象
*/
http.adornParams = (params = {}, openDefultParams = true) => { http.adornParams = (params = {}, openDefultParams = true) => {
var defaults = { const defaults = {
't': new Date().getTime() 't': new Date().getTime() // 添加时间戳参数,防止缓存
} }
return openDefultParams ? merge(defaults, params) : params return openDefultParams ? merge(defaults, params) : params // 合并默认参数和传入的参数
} }
/** /**
* post请求数据处理 * post请求的数据处理
* @param {*} data 数据对象 * @param {*} data 请求的数据对象
* @param {*} openDefultdata 是否开启默认数据? * @param {*} openDefultdata 是否开启默认数据
* @param {*} contentType 数据格式 * @param {*} contentType 数据格式类型'json''form'
* json: 'application/json; charset=utf-8' * @returns {string} 处理后的数据
* form: 'application/x-www-form-urlencoded; charset=utf-8' */
*/
http.adornData = (data = {}, openDefultdata = true, contentType = 'json') => { http.adornData = (data = {}, openDefultdata = true, contentType = 'json') => {
var defaults = { const defaults = {
't': new Date().getTime() 't': new Date().getTime() // 添加时间戳参数,防止缓存
} }
data = openDefultdata ? merge(defaults, data) : data data = openDefultdata ? merge(defaults, data) : data // 合并默认数据和传入的数据
return contentType === 'json' ? JSON.stringify(data) : qs.stringify(data) // 根据不同的contentType处理数据格式
return contentType === 'json' ? JSON.stringify(data) : qs.stringify(data)
} }
export default http export default http // 导出axios实例供其他模块使用

@ -3,56 +3,76 @@ import router from '@/router'
import store from '@/store' import store from '@/store'
/** /**
* 获取uuid * 获取UUID
*/ * 生成一个标准的UUID例如xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx
* 使用随机数和指定格式的规则生成UUID
*/
export function getUUID() { export function getUUID() {
return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, c => { return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, c => {
return (c === 'x' ? (Math.random() * 16 | 0) : ('r&0x3' | '0x8')).toString(16) return (c === 'x' ? (Math.random() * 16 | 0) : ('r&0x3' | '0x8')).toString(16)
}) })
} }
/** /**
* 是否有权限 * 检查是否有某个权限
* @param {*} key * @param {*} key 权限的标识符
*/ * @returns {boolean} 如果权限列表中包含该权限返回true否则返回false
*/
export function isAuth(key) { export function isAuth(key) {
return JSON.parse(sessionStorage.getItem('permissions') || '[]').indexOf(key) !== -1 || false // 从 sessionStorage 中获取权限列表,并转换为数组,如果没有权限列表,默认返回空数组
return JSON.parse(sessionStorage.getItem('permissions') || '[]').indexOf(key) !== -1 || false
} }
/** /**
* 树形数据转换 * 将平面数据转换为树形数据
* @param {*} data * @param {*} data 原始平面数据
* @param {*} id * @param {*} id 唯一标识符字段默认为'id'
* @param {*} pid * @param {*} pid 父级标识符字段默认为'parentId'
*/ * @returns {Array} 转换后的树形数据
*/
export function treeDataTranslate(data, id = 'id', pid = 'parentId') { export function treeDataTranslate(data, id = 'id', pid = 'parentId') {
var res = [] var res = [] // 存储最终的树形结构
var temp = {} var temp = {} // 临时存储每个节点,以便快速查找父节点
for (var i = 0; i < data.length; i++) {
temp[data[i][id]] = data[i] // 将数据转换为临时对象key为节点的id值为节点本身
} for (var i = 0; i < data.length; i++) {
for (var k = 0; k < data.length; k++) { temp[data[i][id]] = data[i]
if (temp[data[k][pid]] && data[k][id] !== data[k][pid]) { }
if (!temp[data[k][pid]]['children']) {
temp[data[k][pid]]['children'] = [] // 遍历数据根据pid将节点组织成树形结构
} for (var k = 0; k < data.length; k++) {
if (!temp[data[k][pid]]['_level']) { // 如果节点的父节点存在并且当前节点的id不等于父节点的id
temp[data[k][pid]]['_level'] = 1 if (temp[data[k][pid]] && data[k][id] !== data[k][pid]) {
} // 如果父节点没有'children'属性,则初始化为数组
data[k]['_level'] = temp[data[k][pid]]._level + 1 if (!temp[data[k][pid]]['children']) {
temp[data[k][pid]]['children'].push(data[k]) temp[data[k][pid]]['children'] = []
} else { }
res.push(data[k]) // 如果父节点没有'_level'属性则设置为1
} if (!temp[data[k][pid]]['_level']) {
} temp[data[k][pid]]['_level'] = 1
return res }
// 当前节点的级别为父节点的级别+1
data[k]['_level'] = temp[data[k][pid]]._level + 1
// 将当前节点推送到父节点的children数组中
temp[data[k][pid]]['children'].push(data[k])
} else {
// 如果当前节点是根节点,直接推送到结果数组中
res.push(data[k])
}
}
return res // 返回转换后的树形数据
} }
/** /**
* 清除登录信息 * 清除登录信息
*/ * 用于用户退出时清理本地存储的登录信息
*/
export function clearLoginInfo() { export function clearLoginInfo() {
Vue.cookie.delete('token') // 删除cookie中的'token'
//store.commit('resetStore') Vue.cookie.delete('token')
router.options.isAddDynamicMenuRoutes = false // 目前注释掉了重置store的操作若需要可以解除注释
// store.commit('resetStore')
// 重置动态菜单路由标志
router.options.isAddDynamicMenuRoutes = false
} }

@ -1,31 +1,36 @@
/** /**
* 邮箱 * 验证邮箱格式
* @param {*} s * @param {*} s - 需要验证的邮箱地址
*/ * @returns {boolean} - 返回是否是有效的邮箱地址
*/
export function isEmail(s) { export function isEmail(s) {
return /^([a-zA-Z0-9_-])+@([a-zA-Z0-9_-])+((.[a-zA-Z0-9_-]{2,3}){1,2})$/.test(s) return /^([a-zA-Z0-9_-])+@([a-zA-Z0-9_-])+((\.[a-zA-Z0-9_-]{2,3}){1,2})$/.test(s)
} }
/** /**
* 手机号码 * 验证手机号码格式中国手机号
* @param {*} s * @param {*} s - 需要验证的手机号
*/ * @returns {boolean} - 返回是否是有效的手机号码
export function isMobile(s) { */
export function isMobile(s) {
return /^1[0-9]{10}$/.test(s) return /^1[0-9]{10}$/.test(s)
} }
/** /**
* 电话号码 * 验证固定电话号码格式
* @param {*} s * @param {*} s - 需要验证的电话号码
*/ * @returns {boolean} - 返回是否是有效的电话号码包括区号和本地号码
export function isPhone(s) { */
export function isPhone(s) {
return /^([0-9]{3,4}-)?[0-9]{7,8}$/.test(s) return /^([0-9]{3,4}-)?[0-9]{7,8}$/.test(s)
} }
/** /**
* URL地址 * 验证URL地址格式
* @param {*} s * @param {*} s - 需要验证的URL地址
*/ * @returns {boolean} - 返回是否是有效的URL地址包括http或https协议
export function isURL(s) { */
export function isURL(s) {
return /^http[s]?:\/\/.*/.test(s) return /^http[s]?:\/\/.*/.test(s)
} }

@ -1,61 +1,81 @@
<template> <template>
<!-- 404 错误页面的模板 -->
<div class="site-wrapper site-page--not-found"> <div class="site-wrapper site-page--not-found">
<div class="site-content__wrapper"> <div class="site-content__wrapper">
<div class="site-content"> <div class="site-content">
<h2 class="not-found-title">400</h2> <!-- 错误代码 -->
<p class="not-found-desc">抱歉您访问的页面<em>失联</em> ...</p> <h2 class="not-found-title">400</h2>
<el-button @click="$router.go(-1)"></el-button> <!-- 错误描述 -->
<el-button type="primary" class="not-found-btn-gohome" @click="$router.push({ name: 'home' })">进入首页</el-button> <p class="not-found-desc">抱歉您访问的页面<em>失联</em> ...</p >
</div> <!-- 返回上一页按钮 -->
</div> <el-button @click="$router.go(-1)"></el-button>
<!-- 返回首页按钮 -->
<el-button type="primary" class="not-found-btn-gohome" @click="$router.push({ name: 'home' })">进入首页</el-button>
</div> </div>
</template> </div>
</div>
<script> </template>
export default {
} <script>
</script> export default {
// JavaScript
<style lang="scss"> }
.site-wrapper.site-page--not-found { </script>
position: absolute;
<style lang="scss">
/* 整个404页面的外部容器 */
.site-wrapper.site-page--not-found {
position: absolute; /* 定位到页面的绝对位置 */
top: 0; top: 0;
right: 0; right: 0;
bottom: 0; bottom: 0;
left: 0; left: 0;
overflow: hidden; overflow: hidden; /* 防止内容溢出 */
/* 内容包裹层 */
.site-content__wrapper { .site-content__wrapper {
padding: 0; padding: 0;
margin: 0; margin: 0;
background-color: #fff; background-color: #fff; /* 设置背景色为白色 */
} }
/* 页面内容容器 */
.site-content { .site-content {
position: fixed; position: fixed; /* 固定位置 */
top: 15%; top: 15%; /* 距离顶部15% */
left: 50%; left: 50%; /* 距离左边50% */
z-index: 2; z-index: 2; /* 确保内容在其他元素之上 */
padding: 30px; padding: 30px;
text-align: center; text-align: center; /* 内容居中 */
transform: translate(-50%, 0); transform: translate(-50%, 0); /* 将内容水平居中 */
} }
/* 错误标题 */
.not-found-title { .not-found-title {
margin: 20px 0 15px; margin: 20px 0 15px;
font-size: 10em; font-size: 10em; /* 大字体 */
font-weight: 400; font-weight: 400;
color: rgb(55, 71, 79); color: rgb(55, 71, 79); /* 文字颜色 */
} }
/* 错误描述 */
.not-found-desc { .not-found-desc {
margin: 0 0 30px; margin: 0 0 30px;
font-size: 26px; font-size: 26px;
text-transform: uppercase; text-transform: uppercase; /* 将文本转换为大写 */
color: rgb(118, 131, 143); color: rgb(118, 131, 143); /* 文字颜色 */
> em {
font-style: normal; /* 强调标签 <em> 样式 */
color: #ee8145; > em {
} font-style: normal;
color: #ee8145; /* 设置颜色 */
} }
}
/* 返回首页按钮的左边距 */
.not-found-btn-gohome { .not-found-btn-gohome {
margin-left: 30px; margin-left: 30px;
}
} }
} </style>
</style>

@ -1,12 +1,18 @@
<template> <template>
<!-- 模块容器 div -->
<div class="mod-home"> <div class="mod-home">
<h3>欢迎使用微信管理系统</h3> <!-- 欢迎标题 -->
<h3>欢迎使用微信管理系统</h3>
</div> </div>
</template> </template>
<style>
.mod-home { <style>
/* 样式部分 */
.mod-home {
/* 设置行高,增加文本的可读性 */
line-height: 2.5; line-height: 2.5;
/* 使文本水平居中对齐 */
text-align: center; text-align: center;
} }
</style> </style>

@ -1,184 +1,220 @@
<template> <template>
<div class="site-wrapper site-page--login"> <div class="site-wrapper site-page--login">
<div class="site-content__wrapper"> <!-- 页面容器背景层 -->
<div class="site-content"> <div class="site-content__wrapper">
<div class="brand-info"> <!-- 页面内容容器 -->
<h2 class="brand-info__text">微信后台管理系统</h2> <div class="site-content">
<p class="brand-info__intro">微信公众号后台管理系统</p> <!-- 品牌信息部分 -->
</div> <div class="brand-info">
<div class="login-main"> <h2 class="brand-info__text">微信后台管理系统</h2>
<h3 class="login-title">管理员登录</h3> <p class="brand-info__intro">微信公众号后台管理系统</p >
<el-form :model="dataForm" :rules="dataRule" ref="dataForm" @keyup.enter.native="dataFormSubmit()" status-icon>
<el-form-item prop="userName">
<el-input v-model="dataForm.userName" placeholder="帐号"></el-input>
</el-form-item>
<el-form-item prop="password">
<el-input v-model="dataForm.password" type="password" placeholder="密码"></el-input>
</el-form-item>
<el-form-item prop="captcha">
<el-row :gutter="20">
<el-col :span="14">
<el-input v-model="dataForm.captcha" placeholder="验证码">
</el-input>
</el-col>
<el-col :span="10" class="login-captcha">
<img :src="captchaPath" @click="getCaptcha()" alt="">
</el-col>
</el-row>
</el-form-item>
<el-form-item>
<el-button class="login-btn-submit" type="primary" @click="dataFormSubmit()"></el-button>
</el-form-item>
</el-form>
</div>
</div>
</div>
</div> </div>
</template>
<!-- 登录表单 -->
<div class="login-main">
<h3 class="login-title">管理员登录</h3>
<!-- 表单绑定了 model validation 规则 -->
<el-form :model="dataForm" :rules="dataRule" ref="dataForm" @keyup.enter.native="dataFormSubmit()" status-icon>
<!-- 用户名输入框 -->
<el-form-item prop="userName">
<el-input v-model="dataForm.userName" placeholder="帐号"></el-input>
</el-form-item>
<!-- 密码输入框 -->
<el-form-item prop="password">
<el-input v-model="dataForm.password" type="password" placeholder="密码"></el-input>
</el-form-item>
<!-- 验证码输入框 -->
<el-form-item prop="captcha">
<el-row :gutter="20">
<el-col :span="14">
<el-input v-model="dataForm.captcha" placeholder="验证码"></el-input>
</el-col>
<el-col :span="10" class="login-captcha">
<!-- 验证码图片点击刷新验证码 -->
< img :src="captchaPath" @click="getCaptcha()" alt="">
</el-col>
</el-row>
</el-form-item>
<!-- 登录按钮 -->
<el-form-item>
<el-button class="login-btn-submit" type="primary" @click="dataFormSubmit()"></el-button>
</el-form-item>
</el-form>
</div>
</div>
</div>
</div>
</template>
<script> <script>
import { getUUID } from '@/utils' import { getUUID } from '@/utils' // UUID
export default { export default {
data() { data() {
return { return {
dataForm: { //
userName: '', dataForm: {
password: '', userName: '', //
uuid: '', password: '', //
captcha: '' uuid: '', // UUID
}, captcha: '' //
dataRule: { },
userName: [ //
{ required: true, message: '帐号不能为空', trigger: 'blur' } dataRule: {
], userName: [
password: [ { required: true, message: '帐号不能为空', trigger: 'blur' }
{ required: true, message: '密码不能为空', trigger: 'blur' } ],
], password: [
captcha: [ { required: true, message: '密码不能为空', trigger: 'blur' }
{ required: true, message: '验证码不能为空', trigger: 'blur' } ],
] captcha: [
}, { required: true, message: '验证码不能为空', trigger: 'blur' }
captchaPath: '' ]
} },
}, captchaPath: '' //
created() { }
this.getCaptcha() },
}, created() {
methods: { //
// this.getCaptcha()
dataFormSubmit() { },
this.$refs['dataForm'].validate((valid) => { methods: {
if (valid) { //
this.$http({ dataFormSubmit() {
url: this.$http.adornUrl('/sys/login'), //
method: 'post', this.$refs['dataForm'].validate((valid) => {
data: this.$http.adornData({ if (valid) {
'username': this.dataForm.userName, //
'password': this.dataForm.password, this.$http({
'uuid': this.dataForm.uuid, url: this.$http.adornUrl('/sys/login'), //
'captcha': this.dataForm.captcha method: 'post',
}) data: this.$http.adornData({
}).then(({ data }) => { 'username': this.dataForm.userName,
if (data && data.code === 200) { 'password': this.dataForm.password,
this.$cookie.set('token', data.token) 'uuid': this.dataForm.uuid,
this.$router.replace({ name: 'home' }) 'captcha': this.dataForm.captcha
} else { })
this.getCaptcha() }).then(({ data }) => {
this.$message.error(data.msg) if (data && data.code === 200) {
} // token
}) this.$cookie.set('token', data.token)
} this.$router.replace({ name: 'home' })
}) } else {
}, //
// this.getCaptcha()
getCaptcha() { this.$message.error(data.msg)
this.dataForm.uuid = getUUID() }
this.captchaPath = this.$http.adornUrl(`/captcha?uuid=${this.dataForm.uuid}`) })
} }
} })
},
// UUID
getCaptcha() {
this.dataForm.uuid = getUUID() // UUID
//
this.captchaPath = this.$http.adornUrl(`/captcha?uuid=${this.dataForm.uuid}`)
}
}
} }
</script> </script>
<style lang="scss"> <style lang="scss">
.site-wrapper.site-page--login { .site-wrapper.site-page--login {
position: absolute; position: absolute;
top: 0; top: 0;
right: 0; right: 0;
bottom: 0; bottom: 0;
left: 0; left: 0;
background-color: rgba(38, 50, 56, 0.5); background-color: rgba(38, 50, 56, 0.5); /* 半透明背景 */
overflow: hidden; overflow: hidden;
&:before {
position: fixed; /* 页面背景渐变效果 */
top: 0; &:before {
left: 0; position: fixed;
z-index: -1; top: 0;
width: 100%; left: 0;
height: 100%; z-index: -1; /* 确保背景层在最底层 */
content: ""; width: 100%;
background-color: #fa8bff; height: 100%;
background-image: linear-gradient( content: "";
45deg, background-color: #fa8bff;
#fa8bff 0%, background-image: linear-gradient(
#2bd2ff 52%, 45deg,
#2bff88 90% #fa8bff 0%,
); #2bd2ff 52%,
background-size: cover; #2bff88 90%
} );
.site-content__wrapper { background-size: cover;
position: absolute; }
top: 0;
right: 0; .site-content__wrapper {
bottom: 0; position: absolute;
left: 0; top: 0;
padding: 0; right: 0;
margin: 0; bottom: 0;
overflow-x: hidden; left: 0;
overflow-y: auto; padding: 0;
background-color: transparent; margin: 0;
} overflow-x: hidden;
.site-content { overflow-y: auto;
min-height: 100%; background-color: transparent;
padding: 30px 500px 30px 30px; }
}
.brand-info { .site-content {
margin: 220px 100px 0 90px; min-height: 100%;
color: #fff; padding: 30px 500px 30px 30px;
} }
.brand-info__text {
margin: 0 0 22px 0; /* 品牌信息样式 */
font-size: 48px; .brand-info {
font-weight: 400; margin: 220px 100px 0 90px;
text-transform: uppercase; color: #fff;
} }
.brand-info__intro {
margin: 10px 0; .brand-info__text {
font-size: 16px; margin: 0 0 22px 0;
line-height: 1.58; font-size: 48px;
opacity: 0.6; font-weight: 400;
} text-transform: uppercase;
.login-main { }
position: absolute;
top: 0; .brand-info__intro {
right: 0; margin: 10px 0;
padding: 150px 60px 180px; font-size: 16px;
width: 470px; line-height: 1.58;
min-height: 100%; opacity: 0.6;
background-color: #fff; }
}
.login-title { /* 登录表单样式 */
font-size: 16px; .login-main {
} position: absolute;
.login-captcha { top: 0;
overflow: hidden; right: 0;
> img { padding: 150px 60px 180px;
width: 100%; width: 470px;
cursor: pointer; min-height: 100%;
} background-color: #fff;
} }
.login-btn-submit {
width: 100%; .login-title {
margin-top: 38px; font-size: 16px;
} }
/* 验证码图片样式 */
.login-captcha {
overflow: hidden;
> img {
width: 100%;
cursor: pointer; /* 鼠标指针变成手形,表示可以点击 */
}
}
/* 登录按钮样式 */
.login-btn-submit {
width: 100%;
margin-top: 38px;
}
} }
</style> </style>

@ -1,33 +1,53 @@
<template> <template>
<el-form> <el-form>
<h2>布局设置</h2> <!-- 布局设置表单标题 -->
<el-form-item label="导航条类型"> <h2>布局设置</h2>
<el-radio-group v-model="navbarLayoutType">
<el-radio label="default" border>default</el-radio> <!-- 导航条类型设置项 -->
<el-radio label="inverse" border>inverse</el-radio> <el-form-item label="导航条类型">
</el-radio-group> <!-- 导航条类型选择框使用 radio 按钮选择 'default' 'inverse' -->
</el-form-item> <el-radio-group v-model="navbarLayoutType">
<el-form-item label="侧边栏皮肤"> <el-radio label="default" border>default</el-radio>
<el-radio-group v-model="sidebarLayoutSkin"> <el-radio label="inverse" border>inverse</el-radio>
<el-radio label="light" border>light</el-radio> </el-radio-group>
<el-radio label="dark" border>dark</el-radio> </el-form-item>
</el-radio-group>
</el-form-item> <!-- 侧边栏皮肤设置项 -->
<el-form-item label="侧边栏皮肤">
<!-- 侧边栏皮肤选择框使用 radio 按钮选择 'light' 'dark' -->
<el-radio-group v-model="sidebarLayoutSkin">
<el-radio label="light" border>light</el-radio>
<el-radio label="dark" border>dark</el-radio>
</el-radio-group>
</el-form-item>
</el-form> </el-form>
</template> </template>
<script> <script>
export default { export default {
computed: { computed: {
navbarLayoutType: { //
get() { return this.$store.state.common.navbarLayoutType }, navbarLayoutType: {
set(val) { this.$store.commit('common/updateNavbarLayoutType', val) } // getter Vuex navbarLayoutType
}, get() {
sidebarLayoutSkin: { return this.$store.state.common.navbarLayoutType
get() { return this.$store.state.common.sidebarLayoutSkin }, },
set(val) { this.$store.commit('common/updateSidebarLayoutSkin', val) } // setter Vuex navbarLayoutType
} set(val) {
} this.$store.commit('common/updateNavbarLayoutType', val)
}
},
//
sidebarLayoutSkin: {
// getter Vuex sidebarLayoutSkin
get() {
return this.$store.state.common.sidebarLayoutSkin
},
// setter Vuex sidebarLayoutSkin
set(val) {
this.$store.commit('common/updateSidebarLayoutSkin', val)
}
}
}
} }
</script> </script>

@ -1,7 +1,9 @@
<template> <template>
<main class="site-content" :class="{ 'site-content--tabs': $route.meta.isTab }"> <main class="site-content" :class="{ 'site-content--tabs': $route.meta.isTab }">
<!-- 主入口标签页 s --> <!-- 如果当前路由元数据中包含isTab则显示标签页组件 -->
<el-tabs v-if="$route.meta.isTab" v-model="mainTabsActiveName" :closable="true" @tab-click="selectedTabHandle" @tab-remove="removeTabHandle"> <el-tabs v-if="$route.meta.isTab" v-model="mainTabsActiveName" :closable="true"
@tab-click="selectedTabHandle" @tab-remove="removeTabHandle">
<!-- 标签页操作菜单包括关闭当前关闭其他关闭全部和刷新 -->
<el-dropdown class="site-tabs__tools" :show-timeout="0"> <el-dropdown class="site-tabs__tools" :show-timeout="0">
<i class="el-icon-arrow-down el-icon--right"></i> <i class="el-icon-arrow-down el-icon--right"></i>
<el-dropdown-menu slot="dropdown"> <el-dropdown-menu slot="dropdown">
@ -11,17 +13,19 @@
<el-dropdown-item @click.native="refresh()">刷新当前标签页</el-dropdown-item> <el-dropdown-item @click.native="refresh()">刷新当前标签页</el-dropdown-item>
</el-dropdown-menu> </el-dropdown-menu>
</el-dropdown> </el-dropdown>
<!-- 循环渲染标签页 -->
<el-tab-pane v-for="item in mainTabs" :key="item.name" :label="item.title" :name="item.name"> <el-tab-pane v-for="item in mainTabs" :key="item.name" :label="item.title" :name="item.name">
<el-card :body-style="siteContentViewHeight"> <el-card :body-style="siteContentViewHeight">
<iframe v-if="item.type === 'iframe'" :src="item.iframeUrl" width="100%" height="100%" frameborder="0" scrolling="yes"> <!-- 如果标签页类型为iframe则显示iframe -->
</iframe> <iframe v-if="item.type === 'iframe'" :src="item.iframeUrl" width="100%" height="100%" frameborder="0" scrolling="yes"></iframe>
<!-- 否则使用keep-alive缓存组件并显示当前激活的标签页内容 -->
<keep-alive v-else> <keep-alive v-else>
<router-view v-if="item.name === mainTabsActiveName" /> <router-view v-if="item.name === mainTabsActiveName" />
</keep-alive> </keep-alive>
</el-card> </el-card>
</el-tab-pane> </el-tab-pane>
</el-tabs> </el-tabs>
<!-- 主入口标签页 e --> <!-- 如果当前路由不包含isTab则显示单个页面 -->
<el-card v-else :body-style="siteContentViewHeight"> <el-card v-else :body-style="siteContentViewHeight">
<keep-alive> <keep-alive>
<router-view /> <router-view />
@ -31,73 +35,80 @@
</template> </template>
<script> <script>
import { isURL } from '@/utils/validate' import { isURL } from '@/utils/validate' // URL
export default { export default {
inject: ['refresh'], inject: ['refresh'], //
data() { data() {
return { return {
//
} }
}, },
computed: { computed: {
//
documentClientHeight: { documentClientHeight: {
get() { return this.$store.state.common.documentClientHeight } get() { return this.$store.state.common.documentClientHeight }
}, },
//
menuActiveName: { menuActiveName: {
get() { return this.$store.state.common.menuActiveName }, get() { return this.$store.state.common.menuActiveName },
set(val) { this.$store.commit('common/updateMenuActiveName', val) } set(val) { this.$store.commit('common/updateMenuActiveName', val) }
}, },
//
mainTabs: { mainTabs: {
get() { return this.$store.state.common.mainTabs }, get() { return this.$store.state.common.mainTabs },
set(val) { this.$store.commit('common/updateMainTabs', val) } set(val) { this.$store.commit('common/updateMainTabs', val) }
}, },
//
mainTabsActiveName: { mainTabsActiveName: {
get() { return this.$store.state.common.mainTabsActiveName }, get() { return this.$store.state.common.mainTabsActiveName },
set(val) { this.$store.commit('common/updateMainTabsActiveName', val) } set(val) { this.$store.commit('common/updateMainTabsActiveName', val) }
}, },
//
siteContentViewHeight() { siteContentViewHeight() {
var height = this.documentClientHeight - 50 - 30 - 2 var height = this.documentClientHeight - 50 - 30 - 2
if (this.$route.meta.isTab) { if (this.$route.meta.isTab) {
height -= 40 height -= 40
// metaiframeUrlURL
return isURL(this.$route.meta.iframeUrl) ? { height: height + 'px' } : { minHeight: height + 'px' } return isURL(this.$route.meta.iframeUrl) ? { height: height + 'px' } : { minHeight: height + 'px' }
} }
return { minHeight: height + 'px' } return { minHeight: height + 'px' }
} }
}, },
methods: { methods: {
// tabs, tab //
selectedTabHandle(tab) { selectedTabHandle(tab) {
tab = this.mainTabs.filter(item => item.name === tab.name) tab = this.mainTabs.filter(item => item.name === tab.name)
if (tab.length >= 1) { if (tab.length >= 1) {
this.$router.push({ name: tab[0].name, query: tab[0].query, params: tab[0].params }) this.$router.push({ name: tab[0].name, query: tab[0].query, params: tab[0].params })
} }
}, },
// tabs, tab //
removeTabHandle(tabName) { removeTabHandle(tabName) {
this.$store.commit('common/removeTab', tabName) this.$store.commit('common/removeTab', tabName)
}, },
// tabs, //
tabsCloseCurrentHandle() { tabsCloseCurrentHandle() {
this.removeTabHandle(this.mainTabsActiveName) this.removeTabHandle(this.mainTabsActiveName)
}, },
// tabs, //
tabsCloseOtherHandle() { tabsCloseOtherHandle() {
this.mainTabs = this.mainTabs.filter(item => item.name === this.mainTabsActiveName) this.mainTabs = this.mainTabs.filter(item => item.name === this.mainTabsActiveName)
}, },
// tabs, //
tabsCloseAllHandle() { tabsCloseAllHandle() {
this.mainTabs = [] this.mainTabs = []
this.menuActiveName = '' this.menuActiveName = ''
this.$router.push({ name: 'home' }) this.$router.push({ name: 'home' })
}, },
// tabs, // methodstemplate
tabsRefreshCurrentHandle() { // tabsRefreshCurrentHandle refresh() methodstabsRefreshCurrentHandle
var tab = this.$route // tabsRefreshCurrentHandle() {
this.removeTabHandle(tab.name) // var tab = this.$route
this.$nextTick(() => { // this.removeTabHandle(tab.name)
this.$router.push({ name: tab.name, query: tab.query, params: tab.params }) // this.$nextTick(() => {
}) // this.$router.push({ name: tab.name, query: tab.query, params: tab.params })
} // })
// }
} }
} }
</script> </script>

@ -1,19 +1,26 @@
<template> <template>
<!-- 使用Element UI的对话框组件 -->
<el-dialog title="修改密码" :visible.sync="visible" :append-to-body="true"> <el-dialog title="修改密码" :visible.sync="visible" :append-to-body="true">
<!-- 表单用于输入原密码和新密码 -->
<el-form :model="dataForm" :rules="dataRule" ref="dataForm" @keyup.enter.native="dataFormSubmit()" label-width="80px"> <el-form :model="dataForm" :rules="dataRule" ref="dataForm" @keyup.enter.native="dataFormSubmit()" label-width="80px">
<!-- 显示用户名不可编辑 -->
<el-form-item label="账号"> <el-form-item label="账号">
<span>{{ userName }}</span> <span>{{ userName }}</span>
</el-form-item> </el-form-item>
<!-- 输入原密码 -->
<el-form-item label="原密码" prop="password"> <el-form-item label="原密码" prop="password">
<el-input type="password" v-model="dataForm.password"></el-input> <el-input type="password" v-model="dataForm.password"></el-input>
</el-form-item> </el-form-item>
<!-- 输入新密码 -->
<el-form-item label="新密码" prop="newPassword"> <el-form-item label="新密码" prop="newPassword">
<el-input type="password" v-model="dataForm.newPassword"></el-input> <el-input type="password" v-model="dataForm.newPassword"></el-input>
</el-form-item> </el-form-item>
<!-- 确认新密码 -->
<el-form-item label="确认密码" prop="confirmPassword"> <el-form-item label="确认密码" prop="confirmPassword">
<el-input type="password" v-model="dataForm.confirmPassword"></el-input> <el-input type="password" v-model="dataForm.confirmPassword"></el-input>
</el-form-item> </el-form-item>
</el-form> </el-form>
<!-- 对话框底部按钮 -->
<span slot="footer" class="dialog-footer"> <span slot="footer" class="dialog-footer">
<el-button @click="visible = false">取消</el-button> <el-button @click="visible = false">取消</el-button>
<el-button type="primary" @click="dataFormSubmit()"></el-button> <el-button type="primary" @click="dataFormSubmit()"></el-button>
@ -22,9 +29,10 @@
</template> </template>
<script> <script>
import { clearLoginInfo } from '@/utils' import { clearLoginInfo } from '@/utils' //
export default { export default {
data() { data() {
//
var validateConfirmPassword = (rule, value, callback) => { var validateConfirmPassword = (rule, value, callback) => {
if (this.dataForm.newPassword !== value) { if (this.dataForm.newPassword !== value) {
callback(new Error('确认密码与新密码不一致')) callback(new Error('确认密码与新密码不一致'))
@ -33,12 +41,15 @@ export default {
} }
} }
return { return {
//
visible: false, visible: false,
//
dataForm: { dataForm: {
password: '', password: '', //
newPassword: '', newPassword: '', //
confirmPassword: '' confirmPassword: '' //
}, },
//
dataRule: { dataRule: {
password: [ password: [
{ required: true, message: '原密码不能为空', trigger: 'blur' } { required: true, message: '原密码不能为空', trigger: 'blur' }
@ -48,55 +59,65 @@ export default {
], ],
confirmPassword: [ confirmPassword: [
{ required: true, message: '确认密码不能为空', trigger: 'blur' }, { required: true, message: '确认密码不能为空', trigger: 'blur' },
{ validator: validateConfirmPassword, trigger: 'blur' } { validator: validateConfirmPassword, trigger: 'blur' } //
] ]
} }
} }
}, },
computed: { computed: {
//
userName: { userName: {
get() { return this.$store.state.user.name } get() { return this.$store.state.user.name }
}, },
//
mainTabs: { mainTabs: {
get() { return this.$store.state.common.mainTabs }, get() { return this.$store.state.common.mainTabs },
set(val) { this.$store.commit('common/updateMainTabs', val) } set(val) { this.$store.commit('common/updateMainTabs', val) }
} }
}, },
methods: { methods: {
// //
init() { init() {
this.visible = true this.visible = true
this.$nextTick(() => { this.$nextTick(() => {
this.$refs['dataForm'].resetFields() this.$refs['dataForm'].resetFields()
}) })
}, },
// //
dataFormSubmit() { dataFormSubmit() {
//
this.$refs['dataForm'].validate((valid) => { this.$refs['dataForm'].validate((valid) => {
if (valid) { if (valid) {
//
this.$http({ this.$http({
url: this.$http.adornUrl('/sys/user/password'), url: this.$http.adornUrl('/sys/user/password'), //
method: 'post', method: 'post', //
data: this.$http.adornData({ data: this.$http.adornData({ //
'password': this.dataForm.password, 'password': this.dataForm.password,
'newPassword': this.dataForm.newPassword 'newPassword': this.dataForm.newPassword
}) })
}).then(({ data }) => { }).then(({ data }) => {
if (data && data.code === 200) { if (data && data.code === 200) {
//
this.$message({ this.$message({
message: '操作成功', message: '操作成功',
type: 'success', type: 'success',
duration: 1500, duration: 1500,
onClose: () => { onClose: () => {
//
this.visible = false this.visible = false
this.$nextTick(() => { this.$nextTick(() => {
//
this.mainTabs = [] this.mainTabs = []
//
clearLoginInfo() clearLoginInfo()
//
this.$router.replace({ name: 'login' }) this.$router.replace({ name: 'login' })
}) })
} }
}) })
} else { } else {
//
this.$message.error(data.msg) this.$message.error(data.msg)
} }
}) })
@ -105,5 +126,4 @@ export default {
} }
} }
} }
</script> </script>

@ -1,60 +1,76 @@
<template> <template>
<!-- 导航栏组件类名根据navbarLayoutType动态设置 -->
<nav class="site-navbar" :class="'site-navbar--' + navbarLayoutType"> <nav class="site-navbar" :class="'site-navbar--' + navbarLayoutType">
<!-- 导航栏头部包含品牌标识 -->
<div class="site-navbar__header"> <div class="site-navbar__header">
<h1 class="site-navbar__brand" @click="$router.push({ name: 'home' })"> <h1 class="site-navbar__brand" @click="$router.push({ name: 'home' })">
<!-- 大尺寸品牌标识点击返回首页 -->
<a class="site-navbar__brand-lg" href="javascript:;">微信管理系统</a> <a class="site-navbar__brand-lg" href="javascript:;">微信管理系统</a>
<!-- 小尺寸品牌标识点击无效果 -->
<a class="site-navbar__brand-mini" href="javascript:;">W</a> <a class="site-navbar__brand-mini" href="javascript:;">W</a>
</h1> </h1>
</div> </div>
<!-- 导航栏主体包含菜单 -->
<div class="site-navbar__body clearfix"> <div class="site-navbar__body clearfix">
<!-- 左侧菜单包含侧边栏折叠/展开按钮 -->
<el-menu class="site-navbar__menu" mode="horizontal"> <el-menu class="site-navbar__menu" mode="horizontal">
<el-menu-item class="site-navbar__switch" index="0" @click="sidebarFold = !sidebarFold"> <el-menu-item class="site-navbar__switch" index="0" @click="sidebarFold = !sidebarFold">
<!-- 侧边栏折叠/展开图标根据sidebarFold动态切换 -->
<i :class="sidebarFold?'el-icon-s-unfold':'el-icon-s-fold'"></i> <i :class="sidebarFold?'el-icon-s-unfold':'el-icon-s-fold'"></i>
</el-menu-item> </el-menu-item>
</el-menu> </el-menu>
<!-- 右侧菜单包含设置微信账号选择用户信息等 -->
<el-menu class="site-navbar__menu site-navbar__menu--right" mode="horizontal"> <el-menu class="site-navbar__menu site-navbar__menu--right" mode="horizontal">
<!-- 设置菜单项 -->
<el-menu-item index="1" @click="$router.push({ name: 'theme' })"> <el-menu-item index="1" @click="$router.push({ name: 'theme' })">
<template slot="title"> <template slot="title">
<i class="el-icon-setting"></i> <i class="el-icon-setting"></i>
</template> </template>
</el-menu-item> </el-menu-item>
<!-- 微信账号选择器需要权限验证 -->
<el-menu-item index="2" v-if="isAuth('wx:wxaccount:list')"> <el-menu-item index="2" v-if="isAuth('wx:wxaccount:list')">
<template slot="title"> <template slot="title">
<wx-account-selector></wx-account-selector> <wx-account-selector></wx-account-selector>
</template> </template>
</el-menu-item> </el-menu-item>
<!-- 用户信息菜单项包含下拉菜单 -->
<el-menu-item class="site-navbar__avatar" index="3"> <el-menu-item class="site-navbar__avatar" index="3">
<el-dropdown :show-timeout="0" placement="bottom"> <el-dropdown :show-timeout="0" placement="bottom">
<span class="el-dropdown-link"> <span class="el-dropdown-link">
{{ userName }} {{ userName }}
</span> </span>
<!-- 下拉菜单内容 -->
<el-dropdown-menu slot="dropdown"> <el-dropdown-menu slot="dropdown">
<!-- 修改密码菜单项 -->
<el-dropdown-item @click.native="updatePasswordHandle()">修改密码</el-dropdown-item> <el-dropdown-item @click.native="updatePasswordHandle()">修改密码</el-dropdown-item>
<!-- 退出菜单项 -->
<el-dropdown-item @click.native="logoutHandle()">退出</el-dropdown-item> <el-dropdown-item @click.native="logoutHandle()">退出</el-dropdown-item>
</el-dropdown-menu> </el-dropdown-menu>
</el-dropdown> </el-dropdown>
</el-menu-item> </el-menu-item>
</el-menu> </el-menu>
</div> </div>
<!-- 弹窗, 修改密码 --> <!-- 修改密码弹窗组件根据updatePassowrdVisible显示/隐藏 -->
<update-password v-if="updatePassowrdVisible" ref="updatePassowrd"></update-password> <update-password v-if="updatePassowrdVisible" ref="updatePassowrd"></update-password>
</nav> </nav>
</template> </template>
<script> <script>
import UpdatePassword from './main-navbar-update-password' import UpdatePassword from './main-navbar-update-password' //
import WxAccountSelector from '@/components/wx-account-selector' import WxAccountSelector from '@/components/wx-account-selector' //
import { clearLoginInfo } from '@/utils' import { clearLoginInfo } from '@/utils' //
export default { export default {
data() { data() {
return { return {
updatePassowrdVisible: false updatePassowrdVisible: false // /
} }
}, },
components: { components: {
UpdatePassword,WxAccountSelector UpdatePassword, //
WxAccountSelector //
}, },
computed: { computed: {
// navbarLayoutTypesidebarFoldmainTabsuserName
navbarLayoutType: { navbarLayoutType: {
get() { return this.$store.state.common.navbarLayoutType } get() { return this.$store.state.common.navbarLayoutType }
}, },
@ -71,26 +87,28 @@ export default {
} }
}, },
methods: { methods: {
// //
updatePasswordHandle() { updatePasswordHandle() {
this.updatePassowrdVisible = true this.updatePassowrdVisible = true
this.$nextTick(() => { this.$nextTick(() => {
this.$refs.updatePassowrd.init() this.$refs.updatePassowrd.init()
}) })
}, },
// 退 // 退
logoutHandle() { logoutHandle() {
this.$confirm(`确定进行[退出]操作?`, '提示', { this.$confirm(`确定进行[退出]操作?`, '提示', {
confirmButtonText: '确定', confirmButtonText: '确定',
cancelButtonText: '取消', cancelButtonText: '取消',
type: 'warning' type: 'warning'
}).then(() => { }).then(() => {
// 退
this.$http({ this.$http({
url: this.$http.adornUrl('/sys/logout'), url: this.$http.adornUrl('/sys/logout'),
method: 'post', method: 'post',
data: this.$http.adornData() data: this.$http.adornData()
}).then(({ data }) => { }).then(({ data }) => {
if (data && data.code === 200) { if (data && data.code === 200) {
//
clearLoginInfo() clearLoginInfo()
this.$router.push({ name: 'login' }) this.$router.push({ name: 'login' })
} }
@ -99,4 +117,4 @@ export default {
} }
} }
} }
</script> </script>

@ -1,50 +1,38 @@
<template>
<el-submenu v-if="menu.list && menu.list.length >= 1" :index="menu.menuId + ''" :popper-class="'site-sidebar--' + sidebarLayoutSkin + '-popper'">
<template slot="title">
<i class="site-sidebar__menu-icon" :class="menu.icon"></i>
<!-- <icon-svg :name="menu.icon || ''" class="site-sidebar__menu-icon"></icon-svg> -->
<span>{{ menu.name }}</span>
</template>
<sub-menu v-for="item in menu.list" :key="item.menuId" :menu="item" :dynamicMenuRoutes="dynamicMenuRoutes">
</sub-menu>
</el-submenu>
<el-menu-item v-else :index="menu.menuId + ''" @click="gotoRouteHandle(menu)">
<!-- <icon-svg :name="menu.icon || ''" class="site-sidebar__menu-icon"></icon-svg> -->
<i class="site-sidebar__menu-icon fa" :class="menu.icon"></i>
<span>{{ menu.name }}</span>
</el-menu-item>
</template>
<script> <script>
// SubMenu
import SubMenu from './main-sidebar-sub-menu' import SubMenu from './main-sidebar-sub-menu'
export default { export default {
name: 'sub-menu', name: 'sub-menu', //
props: { props: {
menu: { menu: {
type: Object, type: Object, //
required: true required: true //
}, },
dynamicMenuRoutes: { dynamicMenuRoutes: {
type: Array, type: Array, //
required: true required: true //
} }
}, },
components: { components: {
SubMenu SubMenu // SubMenu
}, },
computed: { computed: {
sidebarLayoutSkin: { sidebarLayoutSkin() {
get() { return this.$store.state.common.sidebarLayoutSkin } // Vuex storesidebarLayoutSkin
return this.$store.state.common.sidebarLayoutSkin
} }
}, },
methods: { methods: {
// menuId() // menuId
gotoRouteHandle(menu) { gotoRouteHandle(menu) {
// dynamicMenuRoutesmenuIdmenumenuId
var route = this.dynamicMenuRoutes.filter(item => item.meta.menuId === menu.menuId) var route = this.dynamicMenuRoutes.filter(item => item.meta.menuId === menu.menuId)
// 使$router.push
if (route.length >= 1) { if (route.length >= 1) {
this.$router.push({ name: route[0].name }) this.$router.push({ name: route[0].name })
} }
} }
} }
} }
</script> </script>

@ -1,69 +1,78 @@
<template> <template>
<!-- 侧边栏容器根据sidebarLayoutSkin动态设置类名 -->
<aside class="site-sidebar" :class="'site-sidebar--' + sidebarLayoutSkin"> <aside class="site-sidebar" :class="'site-sidebar--' + sidebarLayoutSkin">
<div class="site-sidebar__inner"> <div class="site-sidebar__inner">
<!-- 侧边栏菜单默认激活项为menuActiveName或'home'可折叠无折叠动画 -->
<el-menu :default-active="menuActiveName || 'home'" :collapse="sidebarFold" :collapseTransition="false" class="site-sidebar__menu"> <el-menu :default-active="menuActiveName || 'home'" :collapse="sidebarFold" :collapseTransition="false" class="site-sidebar__menu">
<!-- 首页菜单项点击后跳转到'home'路由 -->
<el-menu-item index="home" @click="$router.push({ name: 'home' })"> <el-menu-item index="home" @click="$router.push({ name: 'home' })">
<i class="site-sidebar__menu-icon el-icon-s-home"></i> <i class="site-sidebar__menu-icon el-icon-s-home"></i>
<span slot="title">首页</span> <!-- 菜单项标题使用slot="title"指定但在这里直接写了Element UI内部会处理 -->
<span>首页</span>
</el-menu-item> </el-menu-item>
<sub-menu v-for="menu in menuList" :key="menu.menuId" :menu="menu" :dynamicMenuRoutes="dynamicMenuRoutes"> <!-- 遍历menuList为每个菜单项渲染一个sub-menu组件 -->
</sub-menu> <sub-menu v-for="menu in menuList" :key="menu.menuId" :menu="menu" :dynamicMenuRoutes="dynamicMenuRoutes"></sub-menu>
</el-menu> </el-menu>
</div> </div>
</aside> </aside>
</template> </template>
<script> <script>
import SubMenu from './main-sidebar-sub-menu' import SubMenu from './main-sidebar-sub-menu' // SubMenu
import { isURL } from '@/utils/validate' import { isURL } from '@/utils/validate' // isURL
export default { export default {
data() { data() {
return { return {
dynamicMenuRoutes: [] dynamicMenuRoutes: [] //
} }
}, },
components: { components: {
SubMenu SubMenu // SubMenu
}, },
computed: { computed: {
// Vuex store
sidebarLayoutSkin: { sidebarLayoutSkin: {
get() { return this.$store.state.common.sidebarLayoutSkin } get() { return this.$store.state.common.sidebarLayoutSkin } //
}, },
sidebarFold: { sidebarFold: {
get() { return this.$store.state.common.sidebarFold } get() { return this.$store.state.common.sidebarFold } //
}, },
menuList: { menuList: {
get() { return this.$store.state.common.menuList }, get() { return this.$store.state.common.menuList }, //
set(val) { this.$store.commit('common/updateMenuList', val) } set(val) { this.$store.commit('common/updateMenuList', val) } //
}, },
menuActiveName: { menuActiveName: {
get() { return this.$store.state.common.menuActiveName }, get() { return this.$store.state.common.menuActiveName }, //
set(val) { this.$store.commit('common/updateMenuActiveName', val) } set(val) { this.$store.commit('common/updateMenuActiveName', val) } //
}, },
mainTabs: { mainTabs: {
get() { return this.$store.state.common.mainTabs }, get() { return this.$store.state.common.mainTabs }, //
set(val) { this.$store.commit('common/updateMainTabs', val) } set(val) { this.$store.commit('common/updateMainTabs', val) } //
}, },
mainTabsActiveName: { mainTabsActiveName: {
get() { return this.$store.state.common.mainTabsActiveName }, get() { return this.$store.state.common.mainTabsActiveName }, //
set(val) { this.$store.commit('common/updateMainTabsActiveName', val) } set(val) { this.$store.commit('common/updateMainTabsActiveName', val) } //
} }
}, },
watch: { watch: {
// routeHandle
$route: 'routeHandle' $route: 'routeHandle'
}, },
created() { created() {
// sessionStoragerouteHandle
this.menuList = JSON.parse(sessionStorage.getItem('menuList') || '[]') this.menuList = JSON.parse(sessionStorage.getItem('menuList') || '[]')
this.dynamicMenuRoutes = JSON.parse(sessionStorage.getItem('dynamicMenuRoutes') || '[]') this.dynamicMenuRoutes = JSON.parse(sessionStorage.getItem('dynamicMenuRoutes') || '[]')
this.routeHandle(this.$route) this.routeHandle(this.$route)
}, },
methods: { methods: {
// //
routeHandle(route) { routeHandle(route) {
if (route.meta.isTab) { if (route.meta.isTab) {
// tab, // tabtab
var tab = this.mainTabs.filter(item => item.name === route.name)[0] var tab = this.mainTabs.filter(item => item.name === route.name)[0]
if (!tab) { if (!tab) {
// tabtab
if (route.meta.isDynamic) { if (route.meta.isDynamic) {
route = this.dynamicMenuRoutes.filter(item => item.name === route.name)[0] route = this.dynamicMenuRoutes.filter(item => item.name === route.name)[0]
if (!route) { if (!route) {
@ -74,17 +83,18 @@ export default {
menuId: route.meta.menuId || route.name, menuId: route.meta.menuId || route.name,
name: route.name, name: route.name,
title: route.meta.title, title: route.meta.title,
type: isURL(route.meta.iframeUrl) ? 'iframe' : 'module', type: isURL(route.meta.iframeUrl) ? 'iframe' : 'module', // iframeUrltab
iframeUrl: route.meta.iframeUrl || '', iframeUrl: route.meta.iframeUrl || '',
params: route.params, params: route.params,
query: route.query query: route.query
} }
this.mainTabs = this.mainTabs.concat(tab) this.mainTabs = this.mainTabs.concat(tab) // tabmainTabs
} }
//
this.menuActiveName = tab.menuId + '' this.menuActiveName = tab.menuId + ''
this.mainTabsActiveName = tab.name this.mainTabsActiveName = tab.name
} }
} }
} }
} }
</script> </script>

@ -1,9 +1,16 @@
<template> <template>
<div class="site-wrapper" :class="{ 'site-sidebar--fold': sidebarFold }" v-loading.fullscreen.lock="loading" element-loading-text=""> <!-- 站点包装器根据sidebarFold状态添加类名同时绑定全屏加载指示器 -->
<div class="site-wrapper" :class="{ 'site-sidebar--fold': sidebarFold }"
v-loading.fullscreen.lock="loading" element-loading-text="拼命加载中">
<!-- 当loading为false时显示以下内容 -->
<template v-if="!loading"> <template v-if="!loading">
<!-- 导航栏组件 -->
<main-navbar /> <main-navbar />
<!-- 侧边栏组件 -->
<main-sidebar /> <main-sidebar />
<!-- 站点内容包装器动态设置最小高度 -->
<div class="site-content__wrapper" :style="{ 'min-height': documentClientHeight + 'px' }"> <div class="site-content__wrapper" :style="{ 'min-height': documentClientHeight + 'px' }">
<!-- 如果内容不需要刷新则显示主要内容组件 -->
<main-content v-if="!$store.state.common.contentIsNeedRefresh" /> <main-content v-if="!$store.state.common.contentIsNeedRefresh" />
</div> </div>
</template> </template>
@ -11,31 +18,38 @@
</template> </template>
<script> <script>
//
import MainNavbar from './main-navbar' import MainNavbar from './main-navbar'
import MainSidebar from './main-sidebar' import MainSidebar from './main-sidebar'
import MainContent from './main-content' import MainContent from './main-content'
export default { export default {
// refresh
provide() { provide() {
return { return {
//
refresh() { refresh() {
//
this.$store.commit('common/updateContentIsNeedRefresh', true) this.$store.commit('common/updateContentIsNeedRefresh', true)
// DOM
this.$nextTick(() => { this.$nextTick(() => {
this.$store.commit('common/updateContentIsNeedRefresh', false) this.$store.commit('common/updateContentIsNeedRefresh', false)
}) })
} }
} }
}, },
//
data() { data() {
return { return {
loading: true loading: true //
} }
}, },
//
components: { components: {
MainNavbar, MainNavbar,
MainSidebar, MainSidebar,
MainContent MainContent
}, },
// Vuex store
computed: { computed: {
documentClientHeight: { documentClientHeight: {
get() { return this.$store.state.common.documentClientHeight }, get() { return this.$store.state.common.documentClientHeight },
@ -53,29 +67,38 @@ export default {
set(val) { this.$store.commit('user/updateName', val) } set(val) { this.$store.commit('user/updateName', val) }
} }
}, },
//
created() { created() {
this.getUserInfo() this.getUserInfo() //
}, },
//
mounted() { mounted() {
this.resetDocumentClientHeight() this.resetDocumentClientHeight() //
}, },
//
methods: { methods: {
// //
resetDocumentClientHeight() { resetDocumentClientHeight() {
//
this.documentClientHeight = document.documentElement['clientHeight'] this.documentClientHeight = document.documentElement['clientHeight']
//
window.onresize = () => { window.onresize = () => {
this.documentClientHeight = document.documentElement['clientHeight'] this.documentClientHeight = document.documentElement['clientHeight']
} }
}, },
// //
getUserInfo() { getUserInfo() {
// HTTP GET
this.$http({ this.$http({
url: this.$http.adornUrl('/sys/user/info'), url: this.$http.adornUrl('/sys/user/info'),
method: 'get', method: 'get',
params: this.$http.adornParams() params: this.$http.adornParams()
}).then(({ data }) => { }).then(({ data }) => {
// 200
if (data && data.code === 200) { if (data && data.code === 200) {
//
this.loading = false this.loading = false
// ID
this.userId = data.user.userId this.userId = data.user.userId
this.userName = data.user.username this.userName = data.user.username
} }
@ -83,4 +106,4 @@ export default {
} }
} }
} }
</script> </script>

@ -1,127 +1,150 @@
<template> <template>
<!-- Dialog弹框用于配置云存储 -->
<el-dialog title="云存储配置" :close-on-click-modal="false" :visible.sync="visible"> <el-dialog title="云存储配置" :close-on-click-modal="false" :visible.sync="visible">
<el-form :model="dataForm" :rules="dataRule" ref="dataForm" @keyup.enter.native="dataFormSubmit()" label-width="120px">
<el-form-item size="mini" label="存储类型"> <!-- 表单组件model 绑定 dataFormrules 绑定 dataRule -->
<el-radio-group v-model="dataForm.type"> <el-form :model="dataForm" :rules="dataRule" ref="dataForm" @keyup.enter.native="dataFormSubmit()" label-width="120px">
<el-radio :label="1">七牛</el-radio>
<el-radio :label="2">阿里云</el-radio> <!-- 存储类型选择项 -->
<el-radio :label="3">腾讯云</el-radio> <el-form-item size="mini" label="存储类型">
</el-radio-group> <el-radio-group v-model="dataForm.type">
</el-form-item> <el-radio :label="1">七牛</el-radio>
<template v-if="dataForm.type === 1"> <el-radio :label="2">阿里云</el-radio>
<el-form-item label="域名"> <el-radio :label="3">腾讯云</el-radio>
<el-input v-model="dataForm.qiniuDomain" placeholder="七牛绑定的域名"></el-input> </el-radio-group>
</el-form-item> </el-form-item>
<el-form-item label="路径前缀">
<el-input v-model="dataForm.qiniuPrefix" placeholder="不设置默认为空"></el-input> <!-- 当选择七牛云时显示七牛相关配置项 -->
</el-form-item> <template v-if="dataForm.type === 1">
<el-form-item label="AccessKey"> <el-form-item label="域名">
<el-input v-model="dataForm.qiniuAccessKey" placeholder="七牛AccessKey"></el-input> <el-input v-model="dataForm.qiniuDomain" placeholder="七牛绑定的域名"></el-input>
</el-form-item> </el-form-item>
<el-form-item label="SecretKey"> <el-form-item label="路径前缀">
<el-input v-model="dataForm.qiniuSecretKey" placeholder="七牛SecretKey"></el-input> <el-input v-model="dataForm.qiniuPrefix" placeholder="不设置默认为空"></el-input>
</el-form-item> </el-form-item>
<el-form-item label="空间名"> <el-form-item label="AccessKey">
<el-input v-model="dataForm.qiniuBucketName" placeholder="七牛存储空间名"></el-input> <el-input v-model="dataForm.qiniuAccessKey" placeholder="七牛AccessKey"></el-input>
</el-form-item> </el-form-item>
</template> <el-form-item label="SecretKey">
<template v-else-if="dataForm.type === 2"> <el-input v-model="dataForm.qiniuSecretKey" placeholder="七牛SecretKey"></el-input>
<el-form-item label="域名"> </el-form-item>
<el-input v-model="dataForm.aliyunDomain" placeholder="阿里云绑定的域名"></el-input> <el-form-item label="空间名">
</el-form-item> <el-input v-model="dataForm.qiniuBucketName" placeholder="七牛存储空间名"></el-input>
<el-form-item label="路径前缀"> </el-form-item>
<el-input v-model="dataForm.aliyunPrefix" placeholder="不设置默认为空"></el-input> </template>
</el-form-item>
<el-form-item label="EndPoint"> <!-- 当选择阿里云时显示阿里云相关配置项 -->
<el-input v-model="dataForm.aliyunEndPoint" placeholder="阿里云EndPoint"></el-input> <template v-else-if="dataForm.type === 2">
</el-form-item> <el-form-item label="域名">
<el-form-item label="AccessKeyId"> <el-input v-model="dataForm.aliyunDomain" placeholder="阿里云绑定的域名"></el-input>
<el-input v-model="dataForm.aliyunAccessKeyId" placeholder="阿里云AccessKeyId"></el-input> </el-form-item>
</el-form-item> <el-form-item label="路径前缀">
<el-form-item label="AccessKeySecret"> <el-input v-model="dataForm.aliyunPrefix" placeholder="不设置默认为空"></el-input>
<el-input v-model="dataForm.aliyunAccessKeySecret" placeholder="阿里云AccessKeySecret"></el-input> </el-form-item>
</el-form-item> <el-form-item label="EndPoint">
<el-form-item label="BucketName"> <el-input v-model="dataForm.aliyunEndPoint" placeholder="阿里云EndPoint"></el-input>
<el-input v-model="dataForm.aliyunBucketName" placeholder="阿里云BucketName"></el-input> </el-form-item>
</el-form-item> <el-form-item label="AccessKeyId">
</template> <el-input v-model="dataForm.aliyunAccessKeyId" placeholder="阿里云AccessKeyId"></el-input>
<template v-else-if="dataForm.type === 3"> </el-form-item>
<el-form-item label="域名"> <el-form-item label="AccessKeySecret">
<el-input v-model="dataForm.qcloudDomain" placeholder="腾讯云绑定的域名"></el-input> <el-input v-model="dataForm.aliyunAccessKeySecret" placeholder="阿里云AccessKeySecret"></el-input>
</el-form-item> </el-form-item>
<el-form-item label="路径前缀"> <el-form-item label="BucketName">
<el-input v-model="dataForm.qcloudPrefix" placeholder="不设置默认为空"></el-input> <el-input v-model="dataForm.aliyunBucketName" placeholder="阿里云BucketName"></el-input>
</el-form-item> </el-form-item>
<el-form-item label="AppId"> </template>
<el-input v-model="dataForm.qcloudAppId" placeholder="腾讯云AppId"></el-input>
</el-form-item> <!-- 当选择腾讯云时显示腾讯云相关配置项 -->
<el-form-item label="SecretId"> <template v-else-if="dataForm.type === 3">
<el-input v-model="dataForm.qcloudSecretId" placeholder="腾讯云SecretId"></el-input> <el-form-item label="域名">
</el-form-item> <el-input v-model="dataForm.qcloudDomain" placeholder="腾讯云绑定的域名"></el-input>
<el-form-item label="SecretKey"> </el-form-item>
<el-input v-model="dataForm.qcloudSecretKey" placeholder="腾讯云SecretKey"></el-input> <el-form-item label="路径前缀">
</el-form-item> <el-input v-model="dataForm.qcloudPrefix" placeholder="不设置默认为空"></el-input>
<el-form-item label="BucketName"> </el-form-item>
<el-input v-model="dataForm.qcloudBucketName" placeholder="腾讯云BucketName"></el-input> <el-form-item label="AppId">
</el-form-item> <el-input v-model="dataForm.qcloudAppId" placeholder="腾讯云AppId"></el-input>
<el-form-item label="Bucket所属地区"> </el-form-item>
<el-input v-model="dataForm.qcloudRegion" placeholder="如sh可选值 华南gz 华北tj 华东sh"></el-input> <el-form-item label="SecretId">
</el-form-item> <el-input v-model="dataForm.qcloudSecretId" placeholder="腾讯云SecretId"></el-input>
</template> </el-form-item>
</el-form> <el-form-item label="SecretKey">
<span slot="footer" class="dialog-footer"> <el-input v-model="dataForm.qcloudSecretKey" placeholder="腾讯云SecretKey"></el-input>
<el-button @click="visible = false">取消</el-button> </el-form-item>
<el-button type="primary" @click="dataFormSubmit()"></el-button> <el-form-item label="BucketName">
</span> <el-input v-model="dataForm.qcloudBucketName" placeholder="腾讯云BucketName"></el-input>
</el-form-item>
<el-form-item label="Bucket所属地区">
<el-input v-model="dataForm.qcloudRegion" placeholder="如sh可选值 华南gz 华北tj 华东sh"></el-input>
</el-form-item>
</template>
</el-form>
<!-- 弹窗底部按钮 -->
<span slot="footer" class="dialog-footer">
<el-button @click="visible = false">取消</el-button>
<el-button type="primary" @click="dataFormSubmit()"></el-button>
</span>
</el-dialog> </el-dialog>
</template> </template>
<script> <script>
export default { export default {
data() { //
return { data() {
visible: false, return {
dataForm: {}, // /
dataRule: {} visible: false,
} //
}, dataForm: {},
methods: { //
init(id) { dataRule: {}
this.visible = true }
this.$http({ },
url: this.$http.adornUrl('/sys/oss/config'), methods: {
method: 'get', // id
params: this.$http.adornParams() init(id) {
}).then(({ data }) => { this.visible = true //
this.dataForm = data && data.code === 200 ? data.config : [] //
}) this.$http({
}, url: this.$http.adornUrl('/sys/oss/config'), //
// method: 'get', // GET
dataFormSubmit() { params: this.$http.adornParams() //
this.$refs['dataForm'].validate((valid) => { }).then(({ data }) => {
if (valid) { //
this.$http({ this.dataForm = data && data.code === 200 ? data.config : []
url: this.$http.adornUrl('/sys/oss/saveConfig'), })
method: 'post', },
data: this.$http.adornData(this.dataForm) //
}).then(({ data }) => { dataFormSubmit() {
if (data && data.code === 200) { //
this.$message({ this.$refs['dataForm'].validate((valid) => {
message: '操作成功', if (valid) {
type: 'success', //
duration: 1500, this.$http({
onClose: () => { url: this.$http.adornUrl('/sys/oss/saveConfig'), //
this.visible = false method: 'post', // POST
} data: this.$http.adornData(this.dataForm) //
}) }).then(({ data }) => {
} else { if (data && data.code === 200) {
this.$message.error(data.msg) //
} this.$message({
}) message: '操作成功',
} type: 'success',
}) duration: 1500,
} onClose: () => {
} this.visible = false //
}
})
} else {
//
this.$message.error(data.msg)
}
})
}
})
}
}
} }
</script> </script>

@ -1,87 +1,105 @@
<template> <template>
<div @click="selectFile"> <!-- 触发文件选择的区域点击时触发 selectFile 方法 -->
<input type="file" ref="fileInput" v-show="false" @change="onFileChange" /> <div @click="selectFile">
<div>{{uploading?infoText:'上传文件'}}</div> <!-- 隐藏的文件输入框通过 click 触发文件选择 -->
</div> <input type="file" ref="fileInput" v-show="false" @change="onFileChange" />
</template> <!-- 显示上传状态文本如果正在上传则显示 infoText否则显示 "上传文件" -->
<div>{{uploading ? infoText : '上传文件'}}</div>
<script> </div>
// 使 </template>
// 使 <script src="https://unpkg.com/cos-js-sdk-v5@0.5.23/dist/cos-js-sdk-v5.min.js" async></script>
<script>
// 使
// 使 SDKhttps://unpkg.com/cos-js-sdk-v5@0.5.23/dist/cos-js-sdk-v5.min.js
var cos; var cos;
export default { export default {
name: "oss-uploader", name: "oss-uploader", //
data() { data() {
return { return {
uploading: false, uploading: false, //
infoText:"上传中...", infoText: "上传中...", //
cosConfig:[] cosConfig: [] // COS
}
},
mounted(){
this.$http({
url: this.$http.adornUrl('/sys/oss/config'),
method: 'get',
params: this.$http.adornParams()
}).then(({data}) => {
if(data && data.code === 200){
this.cosConfig = data.config
cos=new COS({
SecretId: data.config.qcloudSecretId,
SecretKey: data.config.qcloudSecretKey,
});
}else{
this.$message.error('请先配置云存储相关信息!')
}
})
},
methods: {
selectFile() {//
if (!this.uploading) {
this.$refs.fileInput.click();
}
},
onFileChange() {
let file = this.$refs.fileInput.files[0];
this.uploading = true;
let now = new Date();
let path=now.toISOString().slice(0,10)+'/'+now.getTime()+file.name.substr(file.name.lastIndexOf('.'))
cos.putObject({
Bucket: this.cosConfig.qcloudBucketName, /* 必须 */
Region: this.cosConfig.qcloudRegion, /* 必须 */
Key: path, /* 必须 */
Body: file, //
onProgress: (progressData)=> {
this.infoText='上传中:'+progressData.percent*100+'%'
}
}, (err, data)=> {
console.log(err || data);
this.uploading = false;
if(data){
this.infoText='上传文件'
let fileUrl='https://'+this.cosConfig.qcloudBucketName+'.cos.'+this.cosConfig.qcloudRegion+'.myqcloud.com/'+path;
this.saveUploadResult(fileUrl)
}else {
this.$message.error('文件上传失败',err)
}
});
},
saveUploadResult(url){
this.$http({
url: this.$http.adornUrl('/sys/oss/upload'),
method: 'post',
data:{
url:url
}
}).then(({data})=>{
this.$emit('uploaded', url)
})
}
}
} }
</script> },
mounted() {
<style scoped> // COS
</style> this.$http({
url: this.$http.adornUrl('/sys/oss/config'),
method: 'get',
params: this.$http.adornParams() //
}).then(({ data }) => {
if (data && data.code === 200) {
//
this.cosConfig = data.config;
// COS
cos = new COS({
SecretId: data.config.qcloudSecretId,
SecretKey: data.config.qcloudSecretKey,
});
} else {
//
this.$message.error('请先配置云存储相关信息!');
}
})
},
methods: {
//
selectFile() {
//
if (!this.uploading) {
this.$refs.fileInput.click();
}
},
//
onFileChange() {
let file = this.$refs.fileInput.files[0]; //
this.uploading = true; // true
let now = new Date();
//
let path = now.toISOString().slice(0, 10) + '/' + now.getTime() + file.name.substr(file.name.lastIndexOf('.'));
// COS
cos.putObject({
Bucket: this.cosConfig.qcloudBucketName, //
Region: this.cosConfig.qcloudRegion, //
Key: path, //
Body: file, //
onProgress: (progressData) => { //
//
this.infoText = '上传中:' + (progressData.percent * 100).toFixed(2) + '%';
}
}, (err, data) => {
//
console.log(err || data);
this.uploading = false; // false
if (data) {
//
this.infoText = '上传文件';
// 访 URL
let fileUrl = 'https://' + this.cosConfig.qcloudBucketName + '.cos.' + this.cosConfig.qcloudRegion + '.myqcloud.com/' + path;
this.saveUploadResult(fileUrl); //
} else {
//
this.$message.error('文件上传失败', err);
}
});
},
// 访 URL
saveUploadResult(url) {
this.$http({
url: this.$http.adornUrl('/sys/oss/upload'),
method: 'post',
data: { url: url } // URL
}).then(({ data }) => {
// `uploaded` URL
this.$emit('uploaded', url);
})
}
}
}
</script>
<style scoped>
/* 样式部分为空 */
</style>

@ -1,59 +1,73 @@
<template> <template>
<!-- 点击外部div触发文件选择操作 -->
<div @click="selectFile"> <div @click="selectFile">
<input type="file" ref="fileInput" v-show="false" @change="onFileChange" /> <!-- 隐藏的文件上传input选择文件后触发onFileChange方法 -->
<div>{{uploading?infoText:'上传文件'}}</div> <input type="file" ref="fileInput" v-show="false" @change="onFileChange" />
<!-- 显示上传中的状态或默认的上传文件文本 -->
<div>{{uploading ? infoText : '上传文件'}}</div>
</div> </div>
</template> </template>
<script> <script>
export default { export default {
name: "oss-uploader", name: "oss-uploader", // "oss-uploader"
data() { data() {
return { return {
uploading: false, uploading: false, // false
infoText: "上传中...", infoText: "上传中...", //
cosConfig: [] cosConfig: [] //
} }
}, },
mounted() { mounted() {
this.$http({ //
url: this.$http.adornUrl('/sys/oss/config'), this.$http({
method: 'get', url: this.$http.adornUrl('/sys/oss/config'), // API
params: this.$http.adornParams() method: 'get', // 使GET
}).then(({ data }) => { params: this.$http.adornParams() //
if (data && data.code === 200 && data.config.type) { }).then(({ data }) => {
this.cosConfig = data.config //
} else { if (data && data.code === 200 && data.config.type) {
this.$message.error('请先配置云存储相关信息!') this.cosConfig = data.config //
} } else {
//
}) this.$message.error('请先配置云存储相关信息!')
}
})
}, },
methods: { methods: {
selectFile() {// //
if (!this.uploading) { selectFile() {
this.$refs.fileInput.click(); if (!this.uploading) {
} // input
}, this.$refs.fileInput.click();
onFileChange() { }
let file = this.$refs.fileInput.files[0]; },
this.uploading = true; //
let formData = new FormData(); onFileChange() {
formData.append("file", file) let file = this.$refs.fileInput.files[0]; //
this.$http({ this.uploading = true; // true
url: this.$http.adornUrl('/sys/oss/upload'), let formData = new FormData();
method: 'post', formData.append("file", file); // formData
data: formData
}).then(({ data }) => { // POST
console.log(data) this.$http({
if (data && data.code === 200) { url: this.$http.adornUrl('/sys/oss/upload'), // API
this.$emit('uploaded', data.url) method: 'post', // 使POST
} else { data: formData //
this.$message.error("文件上传失败:" + data.msg) }).then(({ data }) => {
} //
this.uploading = false; console.log(data); // 便
}) if (data && data.code === 200) {
} // 'uploaded'url
this.$emit('uploaded', data.url)
} else {
//
this.$message.error("文件上传失败:" + data.msg)
}
this.uploading = false; // false
})
}
}
} }
} </script>
</script>

@ -39,108 +39,79 @@
<script> <script>
export default { export default {
data() { data() {
return { return {
dataForm: {}, //
dataList: [], dataForm: {},
pageIndex: 1, //
pageSize: 10, dataList: [],
totalCount: 0, //
dataListLoading: false, pageIndex: 1,
dataListSelections: [], //
configVisible: false, pageSize: 10,
uploadVisible: false //
} totalCount: 0,
}, //
components: { dataListLoading: false,
Config: () => import('./oss-config'), //
OssUploader: () => import('./oss-uploader') dataListSelections: [],
}, //
activated() { configVisible: false,
this.getDataList() //
}, uploadVisible: false
methods: { }
// },
getDataList() { components: {
this.dataListLoading = true //
this.$http({ Config: () => import('./oss-config'),
url: this.$http.adornUrl('/sys/oss/list'), //
method: 'get', OssUploader: () => import('./oss-uploader')
params: this.$http.adornParams({ },
'page': this.pageIndex, //
'limit': this.pageSize, activated() {
'sidx': 'id', this.getDataList();
'order': 'desc' },
}) methods: {
}).then(({ data }) => { //
if (data && data.code === 200) { getDataList() {
this.dataList = data.page.list this.dataListLoading = true;
this.totalCount = data.page.totalCount this.$http({
} else { url: this.$http.adornUrl('/sys/oss/list'),
this.dataList = [] method: 'get',
this.totalCount = 0 params: this.$http.adornParams({
} 'page': this.pageIndex, //
this.dataListLoading = false 'limit': this.pageSize, //
}) 'sidx': 'id', //
}, 'order': 'desc' //
// })
sizeChangeHandle(val) { }).then(({ data }) => {
this.pageSize = val // dataListtotalCount
this.pageIndex = 1 if (data && data.code === 200) {
this.getDataList() this.dataList = data.page.list;
}, this.totalCount = data.page.totalCount;
// } else {
currentChangeHandle(val) { //
this.pageIndex = val this.dataList = [];
this.getDataList() this.totalCount = 0;
}, }
// this.dataListLoading = false; //
selectionChangeHandle(val) { });
this.dataListSelections = val },
}, //
// sizeChangeHandle(val) {
configHandle() { this.pageSize = val;
this.configVisible = true this.pageIndex = 1; //
this.$nextTick(() => { this.getDataList(); //
this.$refs.config.init() },
}) //
}, currentChangeHandle(val) {
// this.pageIndex = val;
uploadHandle() { this.getDataList(); //
this.uploadVisible = true },
this.$nextTick(() => { //
this.$refs.upload.init() selectionChangeHandle(val) {
}) this.dataListSelections = val;
}, },
// }
deleteHandle(id) {
var ids = id ? [id] : this.dataListSelections.map(item => item.id)
this.$confirm(`确定对[id=${ids.join(',')}]进行[${id ? '删除' : '批量删除'}]操作?`, '提示', {
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning'
}).then(() => {
this.$http({
url: this.$http.adornUrl('/sys/oss/delete'),
method: 'post',
data: this.$http.adornData(ids, false)
}).then(({ data }) => {
if (data && data.code === 200) {
this.$message({
message: '操作成功',
type: 'success',
duration: 1500,
onClose: () => this.getDataList()
})
} else {
this.$message.error(data.msg)
}
})
}).catch(() => { })
},
isImageUrl(url) {
return url && /.*\.(gif|jpg|jpeg|png|GIF|JPEG|JPG|PNG)/.test(url)
}
}
} }
</script> </script>

@ -1,96 +1,117 @@
<template> <template>
<el-dialog :title="!dataForm.id ? '新增' : '修改'" :close-on-click-modal="false" :visible.sync="visible"> <el-dialog :title="!dataForm.id ? '新增' : '修改'" :close-on-click-modal="false" :visible.sync="visible">
<el-form :model="dataForm" :rules="dataRule" ref="dataForm" @keyup.enter.native="dataFormSubmit()" label-width="80px"> <!-- 弹窗组件显示标题是新增修改取决于dataForm.id -->
<el-form-item label="参数名" prop="paramKey"> <el-form :model="dataForm" :rules="dataRule" ref="dataForm" @keyup.enter.native="dataFormSubmit()" label-width="80px">
<el-input v-model="dataForm.paramKey" placeholder="参数名"></el-input> <!-- 表单组件绑定数据模型为dataForm表单校验规则为dataRule按回车键提交表单 -->
</el-form-item>
<el-form-item label="参数值" prop="paramValue"> <el-form-item label="参数名" prop="paramKey">
<el-input v-model="dataForm.paramValue" placeholder="参数值"></el-input> <!-- 参数名输入框验证参数名 -->
</el-form-item> <el-input v-model="dataForm.paramKey" placeholder="参数名"></el-input>
<el-form-item label="备注" prop="remark"> </el-form-item>
<el-input v-model="dataForm.remark" placeholder="备注"></el-input>
</el-form-item> <el-form-item label="参数值" prop="paramValue">
</el-form> <!-- 参数值输入框验证参数值 -->
<span slot="footer" class="dialog-footer"> <el-input v-model="dataForm.paramValue" placeholder="参数值"></el-input>
<el-button @click="visible = false">取消</el-button> </el-form-item>
<el-button type="primary" @click="dataFormSubmit()"></el-button>
</span> <el-form-item label="备注" prop="remark">
</el-dialog> <!-- 备注输入框 -->
<el-input v-model="dataForm.remark" placeholder="备注"></el-input>
</el-form-item>
</el-form>
<span slot="footer" class="dialog-footer">
<!-- 弹窗底部按钮 -->
<el-button @click="visible = false">取消</el-button>
<!-- 取消按钮点击时关闭弹窗 -->
<el-button type="primary" @click="dataFormSubmit()"></el-button>
<!-- 确定按钮点击时提交表单 -->
</span>
</el-dialog>
</template> </template>
<script> <script>
export default { export default {
data() { data() {
return { return {
visible: false, visible: false, //
dataForm: { dataForm: {
id: 0, id: 0, // ID0ID
paramKey: '', paramKey: '', //
paramValue: '', paramValue: '',//
remark: '' remark: '' //
}, },
dataRule: { dataRule: {
paramKey: [ paramKey: [
{ required: true, message: '参数名不能为空', trigger: 'blur' } { required: true, message: '参数名不能为空', trigger: 'blur' }
], ],
paramValue: [ paramValue: [
{ required: true, message: '参数值不能为空', trigger: 'blur' } { required: true, message: '参数值不能为空', trigger: 'blur' }
] ]
} }
} }
}, },
methods: { methods: {
init(id) { // ID
this.dataForm.id = id || 0 init(id) {
this.visible = true this.dataForm.id = id || 0
this.$nextTick(() => { this.visible = true
this.$refs['dataForm'].resetFields() this.$nextTick(() => {
if (this.dataForm.id) { this.$refs['dataForm'].resetFields() //
this.$http({ if (this.dataForm.id) {
url: this.$http.adornUrl(`/sys/config/info/${this.dataForm.id}`), //
method: 'get', this.$http({
params: this.$http.adornParams() url: this.$http.adornUrl(`/sys/config/info/${this.dataForm.id}`),
}).then(({ data }) => { method: 'get',
if (data && data.code === 200) { params: this.$http.adornParams()
this.dataForm.paramKey = data.config.paramKey }).then(({ data }) => {
this.dataForm.paramValue = data.config.paramValue if (data && data.code === 200) {
this.dataForm.remark = data.config.remark this.dataForm.paramKey = data.config.paramKey
} this.dataForm.paramValue = data.config.paramValue
}) this.dataForm.remark = data.config.remark
}
})
},
//
dataFormSubmit() {
this.$refs['dataForm'].validate((valid) => {
if (valid) {
this.$http({
url: this.$http.adornUrl(`/sys/config/${!this.dataForm.id ? 'save' : 'update'}`),
method: 'post',
data: this.$http.adornData({
'id': this.dataForm.id || undefined,
'paramKey': this.dataForm.paramKey,
'paramValue': this.dataForm.paramValue,
'remark': this.dataForm.remark
})
}).then(({ data }) => {
if (data && data.code === 200) {
this.$message({
message: '操作成功',
type: 'success',
duration: 1500,
onClose: () => {
this.visible = false
this.$emit('refreshDataList')
}
})
} else {
this.$message.error(data.msg)
}
})
}
})
}
}
} }
})
}
})
},
//
dataFormSubmit() {
//
this.$refs['dataForm'].validate((valid) => {
if (valid) {
//
this.$http({
url: this.$http.adornUrl(`/sys/config/${!this.dataForm.id ? 'save' : 'update'}`),
method: 'post',
data: this.$http.adornData({
'id': this.dataForm.id || undefined, // IDID
'paramKey': this.dataForm.paramKey,
'paramValue': this.dataForm.paramValue,
'remark': this.dataForm.remark
})
}).then(({ data }) => {
if (data && data.code === 200) {
//
this.$message({
message: '操作成功',
type: 'success',
duration: 1500,
onClose: () => {
this.visible = false
this.$emit('refreshDataList') //
}
})
} else {
//
this.$message.error(data.msg)
}
})
}
})
}
}
}
</script> </script>

@ -1,134 +1,172 @@
<template> <template>
<div class="mod-config"> <div class="mod-config">
<el-form :inline="true" :model="dataForm" @keyup.enter.native="getDataList()"> <!-- 表单部分包含参数名的查询框和操作按钮 -->
<el-form-item> <el-form :inline="true" :model="dataForm" @keyup.enter.native="getDataList()">
<el-input v-model="dataForm.paramKey" placeholder="参数名" clearable></el-input> <!-- 表单项使用 inline 布局按回车触发查询 -->
</el-form-item> <el-form-item>
<el-form-item> <el-input v-model="dataForm.paramKey" placeholder="参数名" clearable></el-input>
<el-button @click="getDataList()"></el-button> <!-- 输入框绑定 model dataForm.paramKey允许清除内容 -->
<el-button type="primary" @click="addOrUpdateHandle()"></el-button> </el-form-item>
<el-button type="danger" @click="deleteHandle()" :disabled="dataListSelections.length <= 0">批量删除</el-button> <el-form-item>
</el-form-item> <!-- 查询按钮点击时触发 getDataList 方法 -->
</el-form> <el-button @click="getDataList()"></el-button>
<el-table :data="dataList" border v-loading="dataListLoading" @selection-change="selectionChangeHandle" style="width: 100%;"> <!-- 新增按钮点击时触发 addOrUpdateHandle 方法 -->
<el-table-column type="selection" header-align="center" align="center" width="50"> <el-button type="primary" @click="addOrUpdateHandle()"></el-button>
</el-table-column> <!-- 批量删除按钮点击时触发 deleteHandle 方法只有当有选中的数据时可用 -->
<el-table-column prop="id" header-align="center" align="center" width="80" label="ID"> <el-button type="danger" @click="deleteHandle()" :disabled="dataListSelections.length <= 0">批量删除</el-button>
</el-table-column> </el-form-item>
<el-table-column prop="paramKey" header-align="center" align="center" label="参数名"> </el-form>
</el-table-column>
<el-table-column prop="paramValue" header-align="center" align="center" label="参数值"> <!-- 表格部分用于显示数据列表 -->
</el-table-column> <el-table :data="dataList" border v-loading="dataListLoading" @selection-change="selectionChangeHandle" style="width: 100%;">
<el-table-column prop="remark" header-align="center" align="center" label="备注"> <!-- 多选列 -->
</el-table-column> <el-table-column type="selection" header-align="center" align="center" width="50"></el-table-column>
<el-table-column fixed="right" header-align="center" align="center" width="150" label="操作">
<template slot-scope="scope"> <!-- ID列 -->
<el-button type="text" size="small" @click="addOrUpdateHandle(scope.row.id)"></el-button> <el-table-column prop="id" header-align="center" align="center" width="80" label="ID"></el-table-column>
<el-button type="text" size="small" @click="deleteHandle(scope.row.id)"></el-button>
</template> <!-- 参数名列 -->
</el-table-column> <el-table-column prop="paramKey" header-align="center" align="center" label="参数名"></el-table-column>
</el-table>
<el-pagination @size-change="sizeChangeHandle" @current-change="currentChangeHandle" :current-page="pageIndex" :page-sizes="[10, 20, 50, 100]" :page-size="pageSize" :total="totalCount" layout="total, sizes, prev, pager, next, jumper"> <!-- 参数值列 -->
</el-pagination> <el-table-column prop="paramValue" header-align="center" align="center" label="参数值"></el-table-column>
<!-- 弹窗, 新增 / 修改 -->
<add-or-update v-if="addOrUpdateVisible" ref="addOrUpdate" @refreshDataList="getDataList"></add-or-update> <!-- 备注列 -->
</div> <el-table-column prop="remark" header-align="center" align="center" label="备注"></el-table-column>
<!-- 操作列包含修改和删除按钮 -->
<el-table-column fixed="right" header-align="center" align="center" width="150" label="操作">
<template slot-scope="scope">
<!-- 修改按钮点击时触发 addOrUpdateHandle 方法传入当前行的 id -->
<el-button type="text" size="small" @click="addOrUpdateHandle(scope.row.id)"></el-button>
<!-- 删除按钮点击时触发 deleteHandle 方法传入当前行的 id -->
<el-button type="text" size="small" @click="deleteHandle(scope.row.id)"></el-button>
</template>
</el-table-column>
</el-table>
<!-- 分页组件 -->
<el-pagination @size-change="sizeChangeHandle" @current-change="currentChangeHandle" :current-page="pageIndex" :page-sizes="[10, 20, 50, 100]" :page-size="pageSize" :total="totalCount" layout="total, sizes, prev, pager, next, jumper">
</el-pagination>
<!-- 弹窗组件用于新增或修改数据 -->
<add-or-update v-if="addOrUpdateVisible" ref="addOrUpdate" @refreshDataList="getDataList"></add-or-update>
</div>
</template> </template>
<script> <script>
import AddOrUpdate from './config-add-or-update' import AddOrUpdate from './config-add-or-update'
export default { export default {
data() { data() {
return { return {
dataForm: { //
paramKey: '' dataForm: {
}, paramKey: ''
dataList: [], },
pageIndex: 1, //
pageSize: 10, dataList: [],
totalCount: 0, //
dataListLoading: false, pageIndex: 1,
dataListSelections: [], //
addOrUpdateVisible: false pageSize: 10,
} //
}, totalCount: 0,
components: { //
AddOrUpdate dataListLoading: false,
}, //
activated() { dataListSelections: [],
this.getDataList() // /
}, addOrUpdateVisible: false
methods: {
//
getDataList() {
this.dataListLoading = true
this.$http({
url: this.$http.adornUrl('/sys/config/list'),
method: 'get',
params: this.$http.adornParams({
'page': this.pageIndex,
'limit': this.pageSize,
'paramKey': this.dataForm.paramKey
})
}).then(({ data }) => {
if (data && data.code === 200) {
this.dataList = data.page.list
this.totalCount = data.page.totalCount
} else {
this.dataList = []
this.totalCount = 0
}
this.dataListLoading = false
})
},
//
sizeChangeHandle(val) {
this.pageSize = val
this.pageIndex = 1
this.getDataList()
},
//
currentChangeHandle(val) {
this.pageIndex = val
this.getDataList()
},
//
selectionChangeHandle(val) {
this.dataListSelections = val
},
// /
addOrUpdateHandle(id) {
this.addOrUpdateVisible = true
this.$nextTick(() => {
this.$refs.addOrUpdate.init(id)
})
},
//
deleteHandle(id) {
var ids = id ? [id] : this.dataListSelections.map(item => item.id)
this.$confirm(`确定对[id=${ids.join(',')}]进行[${id ? '删除' : '批量删除'}]操作?`, '提示', {
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning'
}).then(() => {
this.$http({
url: this.$http.adornUrl('/sys/config/delete'),
method: 'post',
data: this.$http.adornData(ids, false)
}).then(({ data }) => {
if (data && data.code === 200) {
this.$message({
message: '操作成功',
type: 'success',
duration: 1500,
onClose: () => this.getDataList()
})
} else {
this.$message.error(data.msg)
}
})
}).catch(() => { })
}
}
} }
},
components: {
AddOrUpdate
},
activated() {
//
this.getDataList()
},
methods: {
//
getDataList() {
this.dataListLoading = true
this.$http({
url: this.$http.adornUrl('/sys/config/list'),
method: 'get',
params: this.$http.adornParams({
'page': this.pageIndex,
'limit': this.pageSize,
'paramKey': this.dataForm.paramKey
})
}).then(({ data }) => {
if (data && data.code === 200) {
this.dataList = data.page.list
this.totalCount = data.page.totalCount
} else {
this.dataList = []
this.totalCount = 0
}
this.dataListLoading = false
})
},
//
sizeChangeHandle(val) {
this.pageSize = val
this.pageIndex = 1
this.getDataList()
},
//
currentChangeHandle(val) {
this.pageIndex = val
this.getDataList()
},
//
selectionChangeHandle(val) {
this.dataListSelections = val
},
// /
addOrUpdateHandle(id) {
this.addOrUpdateVisible = true
this.$nextTick(() => {
// /
this.$refs.addOrUpdate.init(id)
})
},
//
deleteHandle(id) {
var ids = id ? [id] : this.dataListSelections.map(item => item.id)
//
this.$confirm(`确定对[id=${ids.join(',')}]进行[${id ? '删除' : '批量删除'}]操作?`, '提示', {
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning'
}).then(() => {
this.$http({
url: this.$http.adornUrl('/sys/config/delete'),
method: 'post',
data: this.$http.adornData(ids, false)
}).then(({ data }) => {
if (data && data.code === 200) {
this.$message({
message: '操作成功',
type: 'success',
duration: 1500,
onClose: () => this.getDataList()
})
} else {
this.$message.error(data.msg)
}
})
}).catch(() => { })
}
}
}
</script> </script>

@ -1,90 +1,107 @@
<template> <template>
<div class="mod-log"> <div class="mod-log">
<el-form :inline="true" :model="dataForm" @keyup.enter.native="getDataList()"> <!-- 搜索表单部分 -->
<el-form-item> <el-form :inline="true" :model="dataForm" @keyup.enter.native="getDataList()">
<el-input v-model="dataForm.key" placeholder="用户名/用户操作" clearable></el-input> <!-- 输入框绑定到 dataForm.key用于搜索用户操作或者用户名 -->
</el-form-item> <el-form-item>
<el-form-item> <el-input v-model="dataForm.key" placeholder="用户名/用户操作" clearable></el-input>
<el-button @click="getDataList()"></el-button> </el-form-item>
</el-form-item> <el-form-item>
</el-form> <!-- 查询按钮点击时触发 getDataList 方法 -->
<el-table :data="dataList" border v-loading="dataListLoading" style="width: 100%"> <el-button @click="getDataList()"></el-button>
<el-table-column prop="id" header-align="center" align="center" width="80" label="ID"> </el-form-item>
</el-table-column> </el-form>
<el-table-column prop="username" header-align="center" align="center" label="用户名">
</el-table-column> <!-- 表格显示数据 -->
<el-table-column prop="operation" header-align="center" align="center" label="用户操作"> <el-table :data="dataList" border v-loading="dataListLoading" style="width: 100%">
</el-table-column> <!-- 表格的列定义 -->
<el-table-column prop="method" header-align="center" align="center" width="150" :show-overflow-tooltip="true" label="请求方法"> <el-table-column prop="id" header-align="center" align="center" width="80" label="ID"></el-table-column>
</el-table-column> <el-table-column prop="username" header-align="center" align="center" label="用户名"></el-table-column>
<el-table-column prop="params" header-align="center" align="center" width="150" :show-overflow-tooltip="true" label="请求参数"> <el-table-column prop="operation" header-align="center" align="center" label="用户操作"></el-table-column>
</el-table-column> <el-table-column prop="method" header-align="center" align="center" width="150" :show-overflow-tooltip="true" label="请求方法"></el-table-column>
<el-table-column prop="time" header-align="center" align="center" label="执行时长(毫秒)"> <el-table-column prop="params" header-align="center" align="center" width="150" :show-overflow-tooltip="true" label="请求参数"></el-table-column>
</el-table-column> <el-table-column prop="time" header-align="center" align="center" label="执行时长(毫秒)"></el-table-column>
<el-table-column prop="ip" header-align="center" align="center" width="150" label="IP地址"> <el-table-column prop="ip" header-align="center" align="center" width="150" label="IP地址"></el-table-column>
</el-table-column> <el-table-column prop="createDate" header-align="center" align="center" width="180" label="创建时间"></el-table-column>
<el-table-column prop="createDate" header-align="center" align="center" width="180" label="创建时间"> </el-table>
</el-table-column>
</el-table> <!-- 分页组件 -->
<el-pagination @size-change="sizeChangeHandle" @current-change="currentChangeHandle" :current-page="pageIndex" :page-sizes="[10, 20, 50, 100]" :page-size="pageSize" :total="totalCount" layout="total, sizes, prev, pager, next, jumper"> <el-pagination @size-change="sizeChangeHandle" @current-change="currentChangeHandle" :current-page="pageIndex" :page-sizes="[10, 20, 50, 100]" :page-size="pageSize" :total="totalCount" layout="total, sizes, prev, pager, next, jumper">
</el-pagination> </el-pagination>
</div> </div>
</template> </template>
<script> <script>
export default { export default {
data() { data() {
return { return {
dataForm: { // 'key'
key: '' dataForm: {
}, key: ''
dataList: [], },
pageIndex: 1, //
pageSize: 10, dataList: [],
totalCount: 0, //
dataListLoading: false, pageIndex: 1,
selectionDataList: [] //
} pageSize: 10,
}, //
created() { totalCount: 0,
this.getDataList() //
}, dataListLoading: false,
methods: { // 使
// selectionDataList: []
getDataList() { }
this.dataListLoading = true },
this.$http({ created() {
url: this.$http.adornUrl('/sys/log/list'), //
method: 'get', this.getDataList()
params: this.$http.adornParams({ },
'page': this.pageIndex, methods: {
'limit': this.pageSize, //
'key': this.dataForm.key, getDataList() {
'sidx': 'id', //
'order': 'desc' this.dataListLoading = true
}) this.$http({
}).then(({ data }) => { url: this.$http.adornUrl('/sys/log/list'), //
if (data && data.code === 200) { method: 'get',
this.dataList = data.page.list params: this.$http.adornParams({
this.totalCount = data.page.totalCount //
} else { 'page': this.pageIndex,
this.dataList = [] 'limit': this.pageSize,
this.totalCount = 0 'key': this.dataForm.key, //
} 'sidx': 'id', // id
this.dataListLoading = false 'order': 'desc' //
}) })
}, }).then(({ data }) => {
// //
sizeChangeHandle(val) { if (data && data.code === 200) {
this.pageSize = val this.dataList = data.page.list //
this.pageIndex = 1 this.totalCount = data.page.totalCount //
this.getDataList() } else {
}, //
// this.dataList = []
currentChangeHandle(val) { this.totalCount = 0
this.pageIndex = val }
this.getDataList() //
} this.dataListLoading = false
} })
},
//
sizeChangeHandle(val) {
this.pageSize = val //
this.pageIndex = 1 //
this.getDataList() //
},
//
currentChangeHandle(val) {
this.pageIndex = val //
this.getDataList() //
}
} }
}
</script> </script>

@ -1,218 +1,253 @@
<template> <template>
<!-- 弹窗组件标题根据dataForm.id判断是新增还是修改 -->
<el-dialog :title="!dataForm.id ? '新增' : '修改'" :close-on-click-modal="false" :visible.sync="visible"> <el-dialog :title="!dataForm.id ? '新增' : '修改'" :close-on-click-modal="false" :visible.sync="visible">
<el-form :model="dataForm" :rules="dataRule" ref="dataForm" @keyup.enter.native="dataFormSubmit()" label-width="80px"> <!-- 表单组件绑定数据模型和验证规则 -->
<el-form-item label="类型" prop="type"> <el-form :model="dataForm" :rules="dataRule" ref="dataForm" @keyup.enter.native="dataFormSubmit()" label-width="80px">
<el-radio-group v-model="dataForm.type">
<el-radio v-for="(type, index) in dataForm.typeList" :label="index" :key="index">{{ type }}</el-radio> <!-- 菜单类型选择 -->
</el-radio-group> <el-form-item label="类型" prop="type">
</el-form-item> <el-radio-group v-model="dataForm.type">
<el-form-item :label="dataForm.typeList[dataForm.type] + '名称'" prop="name"> <!-- 循环渲染可选择的菜单类型 -->
<el-input v-model="dataForm.name" :placeholder="dataForm.typeList[dataForm.type] + '名称'"></el-input> <el-radio v-for="(type, index) in dataForm.typeList" :label="index" :key="index">{{ type }}</el-radio>
</el-form-item> </el-radio-group>
<el-form-item label="上级菜单" prop="parentName"> </el-form-item>
<el-popover ref="menuListPopover" placement="bottom-start" trigger="click">
<el-tree :data="menuList" :props="menuListTreeProps" node-key="menuId" ref="menuListTree" @current-change="menuListTreeCurrentChangeHandle" :default-expand-all="true" :highlight-current="true" :expand-on-click-node="false"> <!-- 菜单名称输入框 -->
</el-tree> <el-form-item :label="dataForm.typeList[dataForm.type] + '名称'" prop="name">
</el-popover> <el-input v-model="dataForm.name" :placeholder="dataForm.typeList[dataForm.type] + '名称'"></el-input>
<el-input v-model="dataForm.parentName" v-popover:menuListPopover :readonly="true" placeholder="点击选择上级菜单" class="menu-list__input"></el-input> </el-form-item>
</el-form-item>
<el-form-item v-if="dataForm.type === 1" label="菜单路由" prop="url"> <!-- 上级菜单选择 -->
<el-input v-model="dataForm.url" placeholder="菜单路由"></el-input> <el-form-item label="上级菜单" prop="parentName">
</el-form-item> <!-- 弹出菜单树选择 -->
<el-form-item v-if="dataForm.type !== 0" label="授权标识" prop="perms"> <el-popover ref="menuListPopover" placement="bottom-start" trigger="click">
<el-input v-model="dataForm.perms" placeholder="多个用逗号分隔, 如: user:list,user:create"></el-input> <el-tree :data="menuList" :props="menuListTreeProps" node-key="menuId" ref="menuListTree"
</el-form-item> @current-change="menuListTreeCurrentChangeHandle" :default-expand-all="true"
:highlight-current="true" :expand-on-click-node="false">
<el-form-item v-if="dataForm.type !== 2" label="菜单图标" prop="icon"> </el-tree>
<el-row> </el-popover>
<el-col :span="12"> <!-- 显示上级菜单选择 -->
<el-input v-model="dataForm.icon" placeholder="菜单图标名称" class="icon-list__input"></el-input> <el-input v-model="dataForm.parentName" v-popover:menuListPopover :readonly="true"
</el-col> placeholder="点击选择上级菜单" class="menu-list__input"></el-input>
<el-col :span="12" class="icon-list__tips"> </el-form-item>
<el-form-item v-if="dataForm.type !== 2" label="排序号" prop="orderNum">
<el-input-number v-model="dataForm.orderNum" controls-position="right" :min="0" label="排序号"></el-input-number> <!-- 菜单路由仅在选择菜单类型时显示 -->
</el-form-item> <el-form-item v-if="dataForm.type === 1" label="菜单路由" prop="url">
</el-col> <el-input v-model="dataForm.url" placeholder="菜单路由"></el-input>
</el-row> </el-form-item>
</el-form-item>
<div>参考ElementUI图标库, <a href="https://element.eleme.cn/#/zh-CN/component/icon" target="_blank">找图标</a></div> <!-- 授权标识仅在类型不为目录时显示 -->
</el-form> <el-form-item v-if="dataForm.type !== 0" label="授权标识" prop="perms">
<span slot="footer" class="dialog-footer"> <el-input v-model="dataForm.perms" placeholder="多个用逗号分隔, 如: user:list,user:create"></el-input>
<el-button @click="visible = false">取消</el-button> </el-form-item>
<el-button type="primary" @click="dataFormSubmit()"></el-button>
</span> <!-- 菜单图标仅在类型不为按钮时显示 -->
<el-form-item v-if="dataForm.type !== 2" label="菜单图标" prop="icon">
<el-row>
<el-col :span="12">
<el-input v-model="dataForm.icon" placeholder="菜单图标名称" class="icon-list__input"></el-input>
</el-col>
<el-col :span="12" class="icon-list__tips">
<!-- 排序号输入框仅在类型不为按钮时显示 -->
<el-form-item v-if="dataForm.type !== 2" label="排序号" prop="orderNum">
<el-input-number v-model="dataForm.orderNum" controls-position="right" :min="0" label="排序号"></el-input-number>
</el-form-item>
</el-col>
</el-row>
</el-form-item>
<!-- 提示用户参考图标库 -->
<div>参考ElementUI图标库, 找图标</div>
</el-form>
<!-- 弹窗底部按钮 -->
<span slot="footer" class="dialog-footer">
<el-button @click="visible = false">取消</el-button>
<el-button type="primary" @click="dataFormSubmit()"></el-button>
</span>
</el-dialog> </el-dialog>
</template> </template>
<script> <script>
import { treeDataTranslate } from '@/utils' //
export default { import { treeDataTranslate } from '@/utils'
export default {
data() { data() {
var validateUrl = (rule, value, callback) => { // URL
if (this.dataForm.type === 1 && !/\S/.test(value)) { var validateUrl = (rule, value, callback) => {
callback(new Error('菜单URL不能为空')) // URL
} else { if (this.dataForm.type === 1 && !/\S/.test(value)) {
callback() callback(new Error('菜单URL不能为空'))
} } else {
} callback() //
return { }
visible: false, }
dataForm: { return {
id: 0, visible: false, //
type: 1, dataForm: {
typeList: ['目录', '菜单', '按钮'], id: 0, // ID0
name: '', type: 1, // 0-1-2-
parentId: 0, typeList: ['目录', '菜单', '按钮'], //
parentName: '', name: '', //
url: '', parentId: 0, // ID
perms: '', parentName: '', //
orderNum: 0, url: '', //
icon: '', perms: '', //
}, orderNum: 0, //
dataRule: { icon: '', //
name: [ },
{ required: true, message: '菜单名称不能为空', trigger: 'blur' } //
], dataRule: {
parentName: [ name: [
{ required: true, message: '上级菜单不能为空', trigger: 'change' } { required: true, message: '菜单名称不能为空', trigger: 'blur' } //
], ],
url: [ parentName: [
{ validator: validateUrl, trigger: 'blur' } { required: true, message: '上级菜单不能为空', trigger: 'change' } //
] ],
}, url: [
menuList: [], { validator: validateUrl, trigger: 'blur' } // URL
menuListTreeProps: { ]
label: 'name', },
children: 'children' menuList: [], // 使
} menuListTreeProps: {
} label: 'name', //
children: 'children' //
}
}
}, },
methods: { methods: {
init(id) { //
this.dataForm.id = id || 0 init(id) {
this.$http({ this.dataForm.id = id || 0 // ID
url: this.$http.adornUrl('/sys/menu/select'), this.$http({
method: 'get', url: this.$http.adornUrl('/sys/menu/select'),
params: this.$http.adornParams() method: 'get',
}).then(({ data }) => { params: this.$http.adornParams()
this.menuList = treeDataTranslate(data.menuList, 'menuId') }).then(({ data }) => {
}).then(() => { //
this.visible = true this.menuList = treeDataTranslate(data.menuList, 'menuId')
this.$nextTick(() => { }).then(() => {
this.$refs['dataForm'].resetFields() this.visible = true //
}) this.$nextTick(() => {
}).then(() => { this.$refs['dataForm'].resetFields() //
if (!this.dataForm.id) { })
// }).then(() => {
this.menuListTreeSetCurrentNode() if (!this.dataForm.id) {
} else { //
// this.menuListTreeSetCurrentNode()
this.$http({ } else {
url: this.$http.adornUrl(`/sys/menu/info/${this.dataForm.id}`), //
method: 'get', this.$http({
params: this.$http.adornParams() url: this.$http.adornUrl(`/sys/menu/info/${this.dataForm.id}`),
}).then(({ data }) => { method: 'get',
this.dataForm.id = data.menu.menuId params: this.$http.adornParams()
this.dataForm.type = data.menu.type }).then(({ data }) => {
this.dataForm.name = data.menu.name //
this.dataForm.parentId = data.menu.parentId this.dataForm.id = data.menu.menuId
this.dataForm.url = data.menu.url this.dataForm.type = data.menu.type
this.dataForm.perms = data.menu.perms this.dataForm.name = data.menu.name
this.dataForm.orderNum = data.menu.orderNum this.dataForm.parentId = data.menu.parentId
this.dataForm.icon = data.menu.icon this.dataForm.url = data.menu.url
this.menuListTreeSetCurrentNode() this.dataForm.perms = data.menu.perms
}) this.dataForm.orderNum = data.menu.orderNum
} this.dataForm.icon = data.menu.icon
}) this.menuListTreeSetCurrentNode() //
}, })
// }
menuListTreeCurrentChangeHandle(data, node) { })
this.dataForm.parentId = data.menuId },
this.dataForm.parentName = data.name //
}, menuListTreeCurrentChangeHandle(data, node) {
// this.dataForm.parentId = data.menuId // ID
menuListTreeSetCurrentNode() { this.dataForm.parentName = data.name //
this.$refs.menuListTree.setCurrentKey(this.dataForm.parentId) },
this.dataForm.parentName = (this.$refs.menuListTree.getCurrentNode() || {})['name'] //
}, menuListTreeSetCurrentNode() {
// this.$refs.menuListTree.setCurrentKey(this.dataForm.parentId) //
dataFormSubmit() { this.dataForm.parentName = (this.$refs.menuListTree.getCurrentNode() || {})['name'] //
this.$refs['dataForm'].validate((valid) => { },
if (valid) { //
this.$http({ dataFormSubmit() {
url: this.$http.adornUrl(`/sys/menu/${!this.dataForm.id ? 'save' : 'update'}`), this.$refs['dataForm'].validate((valid) => {
method: 'post', if (valid) {
data: this.$http.adornData({ //
'menuId': this.dataForm.id || undefined, this.$http({
'type': this.dataForm.type, url: this.$http.adornUrl(`/sys/menu/${!this.dataForm.id ? 'save' : 'update'}`),
'name': this.dataForm.name, method: 'post',
'parentId': this.dataForm.parentId, data: this.$http.adornData({
'url': this.dataForm.url, 'menuId': this.dataForm.id || undefined,
'perms': this.dataForm.perms, 'type': this.dataForm.type,
'orderNum': this.dataForm.orderNum, 'name': this.dataForm.name,
'icon': this.dataForm.icon 'parentId': this.dataForm.parentId,
}) 'url': this.dataForm.url,
}).then(({ data }) => { 'perms': this.dataForm.perms,
if (data && data.code === 200) { 'orderNum': this.dataForm.orderNum,
this.$message({ 'icon': this.dataForm.icon
message: '操作成功', })
type: 'success', }).then(({ data }) => {
duration: 1500, if (data && data.code === 200) {
onClose: () => { //
this.visible = false this.$message({
this.$emit('refreshDataList') message: '操作成功',
} type: 'success',
}) duration: 1500,
} else { onClose: () => {
this.$message.error(data.msg) this.visible = false //
} this.$emit('refreshDataList') //
}) }
} })
}) } else {
} //
} this.$message.error(data.msg)
} }
</script> })
}
<style lang="scss"> })
.mod-menu { }
}
}
</script>
<style lang="scss">
.mod-menu {
.menu-list__input, .menu-list__input,
.icon-list__input { .icon-list__input {
> .el-input__inner { > .el-input__inner {
cursor: pointer; cursor: pointer; //
} }
} }
&__icon-popover { &__icon-popover {
width: 458px; width: 458px; //
overflow: hidden; overflow: hidden; //
} }
&__icon-inner { &__icon-inner {
width: 478px; width: 478px; //
max-height: 258px; max-height: 258px; //
overflow-x: hidden; overflow-x: hidden; //
overflow-y: auto; overflow-y: auto; //
} }
&__icon-list { &__icon-list {
width: 458px; width: 458px; //
padding: 0; padding: 0; //
margin: -8px 0 0 -8px; margin: -8px 0 0 -8px; //
> .el-button { > .el-button {
padding: 8px; padding: 8px; //
margin: 8px 0 0 8px; margin: 8px 0 0 8px; //
> span { > span {
display: inline-block; display: inline-block; // 使
vertical-align: middle; vertical-align: middle; //
width: 18px; width: 18px; //
height: 18px; height: 18px; //
font-size: 18px; font-size: 18px; //
} }
} }
} }
.icon-list__tips { .icon-list__tips {
font-size: 18px; font-size: 18px; //
text-align: center; text-align: center; //
color: #e6a23c; color: #e6a23c; //
cursor: pointer; cursor: pointer; //
}
} }
} </style>
</style>

@ -1,109 +1,148 @@
<template> <template>
<!-- 菜单管理模块 -->
<div class="mod-menu"> <div class="mod-menu">
<el-form :inline="true" :model="dataForm"> <!-- 表单组件内联布局 -->
<el-form-item> <el-form :inline="true" :model="dataForm">
<el-button v-if="isAuth('sys:menu:save')" type="primary" @click="addOrUpdateHandle()"></el-button> <el-form-item>
</el-form-item> <!-- 如果有新增权限显示新增按钮 -->
</el-form> <el-button v-if="isAuth('sys:menu:save')" type="primary" @click="addOrUpdateHandle()"></el-button>
</el-form-item>
<el-table :data="dataList" row-key="menuId" border style="width: 100%; "> </el-form>
<el-table-column prop="name" header-align="center" min-width="150" label="名称">
</el-table-column> <!-- 菜单数据表格 -->
<el-table-column prop="parentName" header-align="center" align="center" width="120" label="上级菜单"> <el-table :data="dataList" row-key="menuId" border style="width: 100%;">
</el-table-column> <!-- 菜单名称列 -->
<el-table-column header-align="center" align="center" label="图标"> <el-table-column prop="name" header-align="center" min-width="150" label="名称">
<template slot-scope="scope"> </el-table-column>
<i :class="scope.row.icon"></i>
</template> <!-- 上级菜单列 -->
</el-table-column> <el-table-column prop="parentName" header-align="center" align="center" width="120" label="上级菜单">
<el-table-column prop="type" header-align="center" align="center" label="类型"> </el-table-column>
<template slot-scope="scope">
<el-tag v-if="scope.row.type === 0" size="small"></el-tag> <!-- 菜单图标列 -->
<el-tag v-else-if="scope.row.type === 1" size="small" type="success">菜单</el-tag> <el-table-column header-align="center" align="center" label="图标">
<el-tag v-else-if="scope.row.type === 2" size="small" type="info">按钮</el-tag> <template slot-scope="scope">
</template> <!-- 根据图标类渲染图标 -->
</el-table-column> <i :class="scope.row.icon"></i>
<el-table-column prop="orderNum" header-align="center" align="center" label="排序号"> </template>
</el-table-column> </el-table-column>
<el-table-column prop="url" header-align="center" align="center" width="150" :show-overflow-tooltip="true" label="菜单URL">
</el-table-column> <!-- 菜单类型列 -->
<el-table-column prop="perms" header-align="center" align="center" width="150" :show-overflow-tooltip="true" label="授权标识"> <el-table-column prop="type" header-align="center" align="center" label="类型">
</el-table-column> <template slot-scope="scope">
<el-table-column fixed="right" header-align="center" align="center" width="150" label="操作"> <!-- 根据菜单类型渲染不同的标签 -->
<template slot-scope="scope"> <el-tag v-if="scope.row.type === 0" size="small"></el-tag>
<el-button v-if="isAuth('sys:menu:update')" type="text" size="small" @click="addOrUpdateHandle(scope.row.menuId)"></el-button> <el-tag v-else-if="scope.row.type === 1" size="small" type="success">菜单</el-tag>
<el-button v-if="isAuth('sys:menu:delete')" type="text" size="small" @click="deleteHandle(scope.row.menuId)"></el-button> <el-tag v-else-if="scope.row.type === 2" size="small" type="info">按钮</el-tag>
</template> </template>
</el-table-column> </el-table-column>
</el-table>
<!-- 弹窗, 新增 / 修改 --> <!-- 排序号列 -->
<add-or-update v-if="addOrUpdateVisible" ref="addOrUpdate" @refreshDataList="getDataList"></add-or-update> <el-table-column prop="orderNum" header-align="center" align="center" label="排序号">
</el-table-column>
<!-- 菜单URL列支持溢出提示 -->
<el-table-column prop="url" header-align="center" align="center" width="150" :show-overflow-tooltip="true" label="菜单URL">
</el-table-column>
<!-- 授权标识列支持溢出提示 -->
<el-table-column prop="perms" header-align="center" align="center" width="150" :show-overflow-tooltip="true" label="授权标识">
</el-table-column>
<!-- 操作列显示修改和删除按钮 -->
<el-table-column fixed="right" header-align="center" align="center" width="150" label="操作">
<template slot-scope="scope">
<!-- 如果有修改权限显示修改按钮 -->
<el-button v-if="isAuth('sys:menu:update')" type="text" size="small" @click="addOrUpdateHandle(scope.row.menuId)"></el-button>
<!-- 如果有删除权限显示删除按钮 -->
<el-button v-if="isAuth('sys:menu:delete')" type="text" size="small" @click="deleteHandle(scope.row.menuId)"></el-button>
</template>
</el-table-column>
</el-table>
<!-- 弹窗组件用于新增或修改菜单 -->
<add-or-update v-if="addOrUpdateVisible" ref="addOrUpdate" @refreshDataList="getDataList"></add-or-update>
</div> </div>
</template> </template>
<script> <script>
import AddOrUpdate from './menu-add-or-update' // /
import { treeDataTranslate } from '@/utils' import AddOrUpdate from './menu-add-or-update'
export default { import { treeDataTranslate } from '@/utils'
export default {
data() { data() {
return { return {
dataForm: {}, dataForm: {}, //
dataList: [], dataList: [], //
dataListLoading: false, dataListLoading: false, //
addOrUpdateVisible: false addOrUpdateVisible: false //
} }
}, },
components: { components: {
AddOrUpdate AddOrUpdate // /
}, },
activated() { activated() {
this.getDataList() //
this.getDataList()
}, },
methods: { methods: {
// //
getDataList() { getDataList() {
this.dataListLoading = true this.dataListLoading = true //
this.$http({ // HTTP
url: this.$http.adornUrl('/sys/menu/list'), this.$http({
method: 'get', url: this.$http.adornUrl('/sys/menu/list'),
params: this.$http.adornParams() method: 'get',
}).then(({ data }) => { params: this.$http.adornParams()
this.dataList = treeDataTranslate(data, 'menuId') }).then(({ data }) => {
this.dataListLoading = false // 使
}) this.dataList = treeDataTranslate(data, 'menuId')
}, this.dataListLoading = false //
// / })
addOrUpdateHandle(id) { },
this.addOrUpdateVisible = true
this.$nextTick(() => { //
this.$refs.addOrUpdate.init(id) addOrUpdateHandle(id) {
}) this.addOrUpdateVisible = true // /
}, this.$nextTick(() => {
// //
deleteHandle(id) { this.$refs.addOrUpdate.init(id)
this.$confirm(`确定对[id=${id}]进行[删除]操作?`, '提示', { })
confirmButtonText: '确定', },
cancelButtonText: '取消',
type: 'warning' //
}).then(() => { deleteHandle(id) {
this.$http({ //
url: this.$http.adornUrl(`/sys/menu/delete/${id}`), this.$confirm(`确定对[id=${id}]进行[删除]操作?`, '提示', {
method: 'post', confirmButtonText: '确定',
data: this.$http.adornData() cancelButtonText: '取消',
}).then(({ data }) => { type: 'warning' //
if (data && data.code === 200) { }).then(() => {
this.$message({ //
message: '操作成功', this.$http({
type: 'success', url: this.$http.adornUrl(`/sys/menu/delete/${id}`),
duration: 1500, method: 'post',
onClose: () => this.getDataList() data: this.$http.adornData()
}) }).then(({ data }) => {
} else { if (data && data.code === 200) {
this.$message.error(data.msg) //
} this.$message({
}) message: '操作成功',
}).catch(() => { }) type: 'success',
} duration: 1500,
onClose: () => this.getDataList() //
})
} else {
//
this.$message.error(data.msg)
}
})
}).catch(() => {
//
})
}
}
} }
} </script>
</script>

@ -1,111 +1,148 @@
<template> <template>
<!-- 弹窗组件显示角色的新增或修改表单 -->
<el-dialog :title="!dataForm.id ? '新增' : '修改'" :close-on-click-modal="false" :visible.sync="visible"> <el-dialog :title="!dataForm.id ? '新增' : '修改'" :close-on-click-modal="false" :visible.sync="visible">
<el-form :model="dataForm" :rules="dataRule" ref="dataForm" @keyup.enter.native="dataFormSubmit()" label-width="80px"> <!-- 表单组件绑定数据和验证规则 -->
<el-form-item label="角色名称" prop="roleName"> <el-form :model="dataForm" :rules="dataRule" ref="dataForm" @keyup.enter.native="dataFormSubmit()" label-width="80px">
<el-input v-model="dataForm.roleName" placeholder="角色名称"></el-input>
</el-form-item> <!-- 角色名称输入框 -->
<el-form-item label="备注" prop="remark"> <el-form-item label="角色名称" prop="roleName">
<el-input v-model="dataForm.remark" placeholder="备注"></el-input> <el-input v-model="dataForm.roleName" placeholder="角色名称"></el-input>
</el-form-item> </el-form-item>
<el-form-item size="mini" label="授权">
<el-tree :data="menuList" :props="menuListTreeProps" node-key="menuId" ref="menuListTree" :default-expand-all="true" show-checkbox> <!-- 备注输入框 -->
</el-tree> <el-form-item label="备注" prop="remark">
</el-form-item> <el-input v-model="dataForm.remark" placeholder="备注"></el-input>
</el-form> </el-form-item>
<span slot="footer" class="dialog-footer">
<el-button @click="visible = false">取消</el-button> <!-- 授权菜单树 -->
<el-button type="primary" @click="dataFormSubmit()"></el-button> <el-form-item size="mini" label="授权">
</span> <el-tree :data="menuList" :props="menuListTreeProps" node-key="menuId" ref="menuListTree" :default-expand-all="true" show-checkbox>
</el-tree>
</el-form-item>
</el-form>
<!-- 弹窗底部按钮取消和确认按钮 -->
<span slot="footer" class="dialog-footer">
<el-button @click="visible = false">取消</el-button>
<el-button type="primary" @click="dataFormSubmit()"></el-button>
</span>
</el-dialog> </el-dialog>
</template> </template>
<script> <script>
import { treeDataTranslate } from '@/utils' //
export default { import { treeDataTranslate } from '@/utils'
export default {
data() { data() {
return { return {
visible: false, //
menuList: [], visible: false,
menuListTreeProps: { //
label: 'name', menuList: [],
children: 'children' // el-tree
}, menuListTreeProps: {
dataForm: { label: 'name', //
id: 0, children: 'children' //
roleName: '', },
remark: '' //
}, dataForm: {
dataRule: { id: 0, // ID0
roleName: [ roleName: '', //
{ required: true, message: '角色名称不能为空', trigger: 'blur' } remark: '' //
] },
} //
} dataRule: {
roleName: [
{ required: true, message: '角色名称不能为空', trigger: 'blur' } //
]
}
}
}, },
methods: { methods: {
init(id) { // ID
this.dataForm.id = id || 0 init(id) {
this.$http({ this.dataForm.id = id || 0 // ID0
url: this.$http.adornUrl('/sys/menu/list'),
method: 'get', //
params: this.$http.adornParams() this.$http({
}).then(({ data }) => { url: this.$http.adornUrl('/sys/menu/list'),
this.menuList = treeDataTranslate(data, 'menuId') method: 'get',
}).then(() => { params: this.$http.adornParams()
this.visible = true }).then(({ data }) => {
this.$nextTick(() => { this.menuList = treeDataTranslate(data, 'menuId') //
this.$refs['dataForm'].resetFields() }).then(() => {
this.$refs.menuListTree.setCheckedKeys([]) //
}) this.visible = true
}).then(() => {
if (this.dataForm.id) { // DOM
this.$http({ this.$nextTick(() => {
url: this.$http.adornUrl(`/sys/role/info/${this.dataForm.id}`), //
method: 'get', this.$refs['dataForm'].resetFields()
params: this.$http.adornParams() //
}).then(({ data }) => { this.$refs.menuListTree.setCheckedKeys([])
if (data && data.code === 200) { })
this.dataForm.roleName = data.role.roleName }).then(() => {
this.dataForm.remark = data.role.remark // ID
data.role.menuIdList.forEach(item => { if (this.dataForm.id) {
this.$refs.menuListTree.setChecked(item, true); this.$http({
}); url: this.$http.adornUrl(`/sys/role/info/${this.dataForm.id}`),
} method: 'get',
}) params: this.$http.adornParams()
} }).then(({ data }) => {
}) if (data && data.code === 200) {
}, //
// this.dataForm.roleName = data.role.roleName
dataFormSubmit() { this.dataForm.remark = data.role.remark
this.$refs['dataForm'].validate((valid) => { //
if (valid) { data.role.menuIdList.forEach(item => {
this.$http({ this.$refs.menuListTree.setChecked(item, true)
url: this.$http.adornUrl(`/sys/role/${!this.dataForm.id ? 'save' : 'update'}`), })
method: 'post', }
data: this.$http.adornData({ })
'roleId': this.dataForm.id || undefined, }
'roleName': this.dataForm.roleName, })
'remark': this.dataForm.remark, },
'menuIdList': [].concat(this.$refs.menuListTree.getCheckedKeys(), this.$refs.menuListTree.getHalfCheckedKeys())
}) //
}).then(({ data }) => { dataFormSubmit() {
if (data && data.code === 200) { //
this.$message({ this.$refs['dataForm'].validate((valid) => {
message: '操作成功', if (valid) {
type: 'success', //
duration: 1500, this.$http({
onClose: () => { url: this.$http.adornUrl(`/sys/role/${!this.dataForm.id ? 'save' : 'update'}`), // ID
this.visible = false method: 'post',
this.$emit('refreshDataList') data: this.$http.adornData({
} 'roleId': this.dataForm.id || undefined, // ID
}) 'roleName': this.dataForm.roleName, //
} else { 'remark': this.dataForm.remark, //
this.$message.error(data.msg) // ID
} 'menuIdList': [].concat(this.$refs.menuListTree.getCheckedKeys(), this.$refs.menuListTree.getHalfCheckedKeys())
}) })
} }).then(({ data }) => {
}) if (data && data.code === 200) {
} //
this.$message({
message: '操作成功',
type: 'success',
duration: 1500,
onClose: () => {
//
this.visible = false
this.$emit('refreshDataList')
}
})
} else {
//
this.$message.error(data.msg)
}
})
}
})
}
}
} }
} </script>
</script>

@ -1,132 +1,148 @@
<template> <template>
<div class="mod-role"> <!-- 弹窗组件显示角色的新增或修改表单 -->
<el-form :inline="true" :model="dataForm" @keyup.enter.native="getDataList()"> <el-dialog :title="!dataForm.id ? '新增' : '修改'" :close-on-click-modal="false" :visible.sync="visible">
<el-form-item> <!-- 表单组件绑定数据和验证规则 -->
<el-input v-model="dataForm.roleName" placeholder="角色名称" clearable></el-input> <el-form :model="dataForm" :rules="dataRule" ref="dataForm" @keyup.enter.native="dataFormSubmit()" label-width="80px">
</el-form-item>
<el-form-item> <!-- 角色名称输入框 -->
<el-button @click="getDataList()"></el-button> <el-form-item label="角色名称" prop="roleName">
<el-button v-if="isAuth('sys:role:save')" type="primary" @click="addOrUpdateHandle()"></el-button> <el-input v-model="dataForm.roleName" placeholder="角色名称"></el-input>
<el-button v-if="isAuth('sys:role:delete')" type="danger" @click="deleteHandle()" :disabled="dataListSelections.length <= 0"></el-button> </el-form-item>
</el-form-item>
</el-form> <!-- 备注输入框 -->
<el-table :data="dataList" border v-loading="dataListLoading" @selection-change="selectionChangeHandle" style="width: 100%;"> <el-form-item label="备注" prop="remark">
<el-table-column type="selection" header-align="center" align="center" width="50"> <el-input v-model="dataForm.remark" placeholder="备注"></el-input>
</el-table-column> </el-form-item>
<el-table-column prop="roleId" header-align="center" align="center" width="80" label="ID">
</el-table-column> <!-- 授权菜单树 -->
<el-table-column prop="roleName" header-align="center" align="center" label="角色名称"> <el-form-item size="mini" label="授权">
</el-table-column> <el-tree :data="menuList" :props="menuListTreeProps" node-key="menuId" ref="menuListTree" :default-expand-all="true" show-checkbox>
<el-table-column prop="remark" header-align="center" align="center" label="备注"> </el-tree>
</el-table-column> </el-form-item>
<el-table-column fixed="right" header-align="center" align="center" width="150" label="操作">
<template slot-scope="scope"> </el-form>
<el-button v-if="isAuth('sys:role:update')" type="text" size="small" @click="addOrUpdateHandle(scope.row.roleId)"></el-button>
<el-button v-if="isAuth('sys:role:delete')" type="text" size="small" @click="deleteHandle(scope.row.roleId)"></el-button> <!-- 弹窗底部按钮取消和确认按钮 -->
</template> <span slot="footer" class="dialog-footer">
</el-table-column> <el-button @click="visible = false">取消</el-button>
</el-table> <el-button type="primary" @click="dataFormSubmit()"></el-button>
<el-pagination @size-change="sizeChangeHandle" @current-change="currentChangeHandle" :current-page="pageIndex" :page-sizes="[10, 20, 50, 100]" :page-size="pageSize" :total="totalCount" layout="total, sizes, prev, pager, next, jumper"> </span>
</el-pagination> </el-dialog>
<!-- 弹窗, 新增 / 修改 --> </template>
<add-or-update v-if="addOrUpdateVisible" ref="addOrUpdate" @refreshDataList="getDataList"></add-or-update>
</div> <script>
</template> //
import { treeDataTranslate } from '@/utils'
<script>
import AddOrUpdate from './role-add-or-update' export default {
export default {
data() { data() {
return { return {
dataForm: { //
roleName: '' visible: false,
}, //
dataList: [], menuList: [],
pageIndex: 1, // el-tree
pageSize: 10, menuListTreeProps: {
totalCount: 0, label: 'name', //
dataListLoading: false, children: 'children' //
dataListSelections: [],
addOrUpdateVisible: false
}
}, },
components: { //
AddOrUpdate dataForm: {
id: 0, // ID0
roleName: '', //
remark: '' //
}, },
activated() { //
this.getDataList() dataRule: {
roleName: [
{ required: true, message: '角色名称不能为空', trigger: 'blur' } //
]
}
}
}, },
methods: { methods: {
// // ID
getDataList() { init(id) {
this.dataListLoading = true this.dataForm.id = id || 0 // ID0
this.$http({
url: this.$http.adornUrl('/sys/role/list'), //
method: 'get', this.$http({
params: this.$http.adornParams({ url: this.$http.adornUrl('/sys/menu/list'),
'page': this.pageIndex, method: 'get',
'limit': this.pageSize, params: this.$http.adornParams()
'roleName': this.dataForm.roleName }).then(({ data }) => {
}) this.menuList = treeDataTranslate(data, 'menuId') //
}).then(({ data }) => { }).then(() => {
if (data && data.code === 200) { //
this.dataList = data.page.list this.visible = true
this.totalCount = data.page.totalCount
} else { // DOM
this.dataList = [] this.$nextTick(() => {
this.totalCount = 0 //
} this.$refs['dataForm'].resetFields()
this.dataListLoading = false //
}) this.$refs.menuListTree.setCheckedKeys([])
}, })
// }).then(() => {
sizeChangeHandle(val) { // ID
this.pageSize = val if (this.dataForm.id) {
this.pageIndex = 1 this.$http({
this.getDataList() url: this.$http.adornUrl(`/sys/role/info/${this.dataForm.id}`),
}, method: 'get',
// params: this.$http.adornParams()
currentChangeHandle(val) { }).then(({ data }) => {
this.pageIndex = val if (data && data.code === 200) {
this.getDataList() //
}, this.dataForm.roleName = data.role.roleName
// this.dataForm.remark = data.role.remark
selectionChangeHandle(val) { //
this.dataListSelections = val data.role.menuIdList.forEach(item => {
}, this.$refs.menuListTree.setChecked(item, true)
// / })
addOrUpdateHandle(id) { }
this.addOrUpdateVisible = true })
this.$nextTick(() => { }
this.$refs.addOrUpdate.init(id) })
}) },
},
// //
deleteHandle(id) { dataFormSubmit() {
var ids = id ? [id] : this.dataListSelections.map(item => item.roleId) //
this.$confirm(`确定对[id=${ids.join(',')}]进行[${id ? '删除' : '批量删除'}]操作?`, '提示', { this.$refs['dataForm'].validate((valid) => {
confirmButtonText: '确定', if (valid) {
cancelButtonText: '取消', //
type: 'warning' this.$http({
}).then(() => { url: this.$http.adornUrl(`/sys/role/${!this.dataForm.id ? 'save' : 'update'}`), // ID
this.$http({ method: 'post',
url: this.$http.adornUrl('/sys/role/delete'), data: this.$http.adornData({
method: 'post', 'roleId': this.dataForm.id || undefined, // ID
data: this.$http.adornData(ids, false) 'roleName': this.dataForm.roleName, //
}).then(({ data }) => { 'remark': this.dataForm.remark, //
if (data && data.code === 200) { // ID
this.$message({ 'menuIdList': [].concat(this.$refs.menuListTree.getCheckedKeys(), this.$refs.menuListTree.getHalfCheckedKeys())
message: '操作成功', })
type: 'success', }).then(({ data }) => {
duration: 1500, if (data && data.code === 200) {
onClose: () => this.getDataList() //
}) this.$message({
} else { message: '操作成功',
this.$message.error(data.msg) type: 'success',
} duration: 1500,
}) onClose: () => {
}).catch(() => { }) //
} this.visible = false
this.$emit('refreshDataList')
}
})
} else {
//
this.$message.error(data.msg)
}
})
}
})
}
}
} }
} </script>
</script>

@ -6,37 +6,54 @@
<div class="weixin-bd"> <div class="weixin-bd">
<div class="weixin-header">公众号菜单</div> <div class="weixin-header">公众号菜单</div>
<ul class="weixin-menu" id="weixin-menu"> <ul class="weixin-menu" id="weixin-menu">
<li v-for="(btn,i) in menu.buttons" :key="i" class="menu-item" :class="{'current':selectedMenuIndex===i&&selectedMenuLevel==1}" @click="selectMenu(i)"> <!-- 遍历主菜单按钮 -->
<li v-for="(btn, i) in menu.buttons" :key="i" class="menu-item"
:class="{'current': selectedMenuIndex === i && selectedMenuLevel == 1}"
@click="selectMenu(i)">
<div class="menu-item-title"> <div class="menu-item-title">
<span>{{ btn.name }}</span> <span>{{ btn.name }}</span>
</div> </div>
<ul class="weixin-sub-menu"> <ul class="weixin-sub-menu">
<li v-for="(sub,i2) in btn.subButtons" :key="i2" class="menu-sub-item" :class="{'current':selectedMenuIndex===i&&selectedSubMenuIndex===i2&&selectedMenuLevel==2,'on-drag-over':onDragOverMenu==(i+'_'+i2)}" @click.stop="selectSubMenu(i,i2)" draggable="true" @dragstart="selectSubMenu(i,i2)" @dragover.prevent="onDragOverMenu=(i+'_'+i2)" @drop="onDrop(i,i2)"> <!-- 遍历子菜单按钮 -->
<li v-for="(sub, i2) in btn.subButtons" :key="i2" class="menu-sub-item"
:class="{'current': selectedMenuIndex === i && selectedSubMenuIndex === i2 && selectedMenuLevel == 2, 'on-drag-over': onDragOverMenu === (i + '_' + i2)}"
@click.stop="selectSubMenu(i, i2)" draggable="true"
@dragstart="selectSubMenu(i, i2)" @dragover.prevent="onDragOverMenu = (i + '_' + i2)" @drop="onDrop(i, i2)">
<div class="menu-item-title"> <div class="menu-item-title">
<span>{{sub.name}}</span> <span>{{ sub.name }}</span>
</div> </div>
</li> </li>
<li v-if="btn.subButtons.length<5" class="menu-sub-item" :class="{'on-drag-over':onDragOverMenu==(i+'_'+btn.subButtons.length)}" @click.stop="addMenu(2,i)" @dragover.prevent="onDragOverMenu=(i+'_'+btn.subButtons.length)" @drop="onDrop(i,btn.subButtons.length)"> <!-- 如果子菜单数量少于5个显示添加按钮 -->
<li v-if="btn.subButtons.length < 5" class="menu-sub-item"
:class="{'on-drag-over': onDragOverMenu === (i + '_' + btn.subButtons.length)}"
@click.stop="addMenu(2, i)" @dragover.prevent="onDragOverMenu = (i + '_' + btn.subButtons.length)" @drop="onDrop(i, btn.subButtons.length)">
<div class="menu-item-title"> <div class="menu-item-title">
<i class="el-icon-plus"></i> <i class="el-icon-plus"></i>
</div> </div>
</li> </li>
<!-- 菜单展开/收起图标 -->
<i class="menu-arrow arrow_out"></i> <i class="menu-arrow arrow_out"></i>
<i class="menu-arrow arrow_in"></i> <i class="menu-arrow arrow_in"></i>
</ul> </ul>
</li> </li>
<li class="menu-item" v-if="menu.buttons.length<3" @click="addMenu(1)"> <i class="el-icon-plus"></i></li> <!-- 如果主菜单数量少于3个显示添加按钮 -->
<li class="menu-item" v-if="menu.buttons.length < 3" @click="addMenu(1)">
<i class="el-icon-plus"></i>
</li>
</ul> </ul>
</div> </div>
</div> </div>
<!-- 菜单编辑器 --> <!-- 菜单编辑器 -->
<div class="weixin-menu-detail" v-if="selectedMenuLevel>0"> <div class="weixin-menu-detail" v-if="selectedMenuLevel > 0">
<wx-menu-button-editor :button="selectedButton" :selectedMenuLevel="selectedMenuLevel" @delMenu="delMenu"></wx-menu-button-editor> <wx-menu-button-editor :button="selectedButton" :selectedMenuLevel="selectedMenuLevel" @delMenu="delMenu"></wx-menu-button-editor>
</div> </div>
</div> </div>
<!-- 发布和清空菜单的按钮组 -->
<div class="weixin-btn-group" v-if="isAuth('wx:menu:save')" @click="updateWxMenu"> <div class="weixin-btn-group" v-if="isAuth('wx:menu:save')" @click="updateWxMenu">
<el-button type="success" icon="el-icon-upload">发布</el-button> <el-button type="success" icon="el-icon-upload">发布</el-button>
<el-button type="warning" icon="el-icon-delete" @click="delMenu"></el-button> <el-button type="warning" icon="el-icon-delete" @click="delMenuAll"></el-button>
<!-- 注意这里可能是一个小错误因为delMenu方法通常用于删除选中的菜单项而不是清空所有菜单 -->
<!-- 如果要清空所有菜单可能需要一个单独的方法比如delMenuAll -->
</div> </div>
</div> </div>
</template> </template>
@ -56,6 +73,7 @@ export default {
} }
}, },
mounted() { mounted() {
//
this.getWxMenu(); this.getWxMenu();
}, },
methods: { methods: {
@ -88,7 +106,7 @@ export default {
this.selectedSubMenuIndex = i2 this.selectedSubMenuIndex = i2
this.selectedButton = this.menu.buttons[i].subButtons[i2] this.selectedButton = this.menu.buttons[i].subButtons[i2]
}, },
// //
addMenu(level,i) { addMenu(level,i) {
if (level == 1 && this.menu.buttons.length < 3) { if (level == 1 && this.menu.buttons.length < 3) {
this.menu.buttons.push({ this.menu.buttons.push({
@ -108,7 +126,7 @@ export default {
this.selectSubMenu(i,this.menu.buttons[i].subButtons.length - 1) this.selectSubMenu(i,this.menu.buttons[i].subButtons.length - 1)
} }
}, },
// //
delMenu() { delMenu() {
if (this.selectedMenuLevel == 1 && confirm('删除后菜单下设置的内容将被删除')) { if (this.selectedMenuLevel == 1 && confirm('删除后菜单下设置的内容将被删除')) {
this.menu.buttons.splice(this.selectedMenuIndex, 1); this.menu.buttons.splice(this.selectedMenuIndex, 1);
@ -118,12 +136,13 @@ export default {
this.unSelectMenu() this.unSelectMenu()
} }
}, },
unSelectMenu(){// unSelectMenu(){//
this.selectedMenuLevel = 0 this.selectedMenuLevel = 0
this.selectedMenuIndex = '' this.selectedMenuIndex = ''
this.selectedSubMenuIndex = '' this.selectedSubMenuIndex = ''
this.selectedButton = '' this.selectedButton = ''
}, },
//
updateWxMenu() { updateWxMenu() {
this.$http({ this.$http({
url: this.$http.adornUrl('/manage/wxMenu/updateMenu'), url: this.$http.adornUrl('/manage/wxMenu/updateMenu'),
@ -138,7 +157,7 @@ export default {
}); });
}, },
onDrop(i,i2){// onDrop(i,i2){//
this.onDragOverMenu=''; this.onDragOverMenu='';
if(i==this.selectedMenuIndex && i2==this.selectedSubMenuIndex) // if(i==this.selectedMenuIndex && i2==this.selectedSubMenuIndex) //
return return

@ -1,14 +1,22 @@
<template> <template>
<!-- 消息回复对话框 -->
<el-dialog title="消息回复" :close-on-click-modal="false" :visible.sync="visible"> <el-dialog title="消息回复" :close-on-click-modal="false" :visible.sync="visible">
<!-- 表单区域 -->
<el-form :model="dataForm" :rules="dataRule" ref="dataForm"> <el-form :model="dataForm" :rules="dataRule" ref="dataForm">
<!-- 表单项回复内容 -->
<el-form-item prop="replyContent"> <el-form-item prop="replyContent">
<!-- 文本区域输入框用于输入回复内容 -->
<el-input v-model="dataForm.replyContent" type="textarea" :rows="5" placeholder="回复内容" maxlength="600" show-word-limit :autosize="{ minRows: 5, maxRows: 30 }" autocomplete></el-input> <el-input v-model="dataForm.replyContent" type="textarea" :rows="5" placeholder="回复内容" maxlength="600" show-word-limit :autosize="{ minRows: 5, maxRows: 30 }" autocomplete></el-input>
<!-- 当回复类型为文本时显示点击插入链接 -->
<el-button type="text" v-show="'text'==dataForm.replyType" @click="addLink"></el-button> <el-button type="text" v-show="'text'==dataForm.replyType" @click="addLink"></el-button>
</el-form-item> </el-form-item>
</el-form> </el-form>
<!-- 对话框底部操作区域 -->
<span slot="footer" class="dialog-footer"> <span slot="footer" class="dialog-footer">
<!-- 取消按钮点击关闭对话框 -->
<el-button @click="visible = false">取消</el-button> <el-button @click="visible = false">取消</el-button>
<el-button type="success" @click="dataFormSubmit()" :disabled="uploading">{{uploading?'发送中...':'发送'}}</el-button> <!-- 发送按钮根据uploading状态显示不同文本并控制是否禁用 -->
<el-button type="success" @click="dataFormSubmit()" :disabled="uploading">{{uploading ? '发送中...' : '发送'}}</el-button>
</span> </span>
</el-dialog> </el-dialog>
</template> </template>
@ -17,68 +25,83 @@
export default { export default {
data() { data() {
return { return {
//
visible: false, visible: false,
//
uploading: false, uploading: false,
//
dataForm: { dataForm: {
openid:'', //
replyType:'text', openid: '',
replyContent:'' //
replyType: 'text',
//
replyContent: ''
}, },
//
dataRule: { dataRule: {
//
replyContent: [ replyContent: [
{ required: true, message: "回复内容不能为空", trigger: "blur" } { required: true, message: "回复内容不能为空", trigger: "blur" }
] ]
} }
} }
}, },
components:{ // 使
WxMsgPreview:()=>import('@/components/wx-msg-preview') components: {
WxMsgPreview: () => import('@/components/wx-msg-preview')
}, },
methods: { methods: {
// openid
init(openid) { init(openid) {
if(!openid)throw '参数异常' if (!openid) throw '参数异常'; // openid
this.dataForm.openid=openid this.dataForm.openid = openid; // openid
this.visible = true this.visible = true; //
}, },
// //
dataFormSubmit() { dataFormSubmit() {
if(this.uploading)return if (this.uploading) return; //
this.uploading=true this.uploading = true; // true
this.$refs['dataForm'].validate((valid) => { this.$refs['dataForm'].validate((valid) => { //
if (valid) { if (valid) { //
// $http
this.$http({ this.$http({
url: this.$http.adornUrl(`/manage/wxMsg/reply`), url: this.$http.adornUrl(`/manage/wxMsg/reply`), //
method: 'post', method: 'post', //
data: this.$http.adornData(this.dataForm) data: this.$http.adornData(this.dataForm) //
}).then(({ data }) => { }).then(({ data }) => { //
if (data && data.code === 200) { if (data && data.code === 200) { // 200
//
this.$message({ this.$message({
message: '回复成功', message: '回复成功',
type: 'success', type: 'success',
duration: 1500, duration: 1500,
onClose: () => { onClose: () => {
this.visible = false this.visible = false;
} }
}) });
this.$emit("success",{...this.dataForm}); this.$emit("success", { ...this.dataForm }); // success
this.dataForm.replyContent='' this.dataForm.replyContent = ''; //
} else { } else { //
this.$message.error(data.msg) //
this.$message.error(data.msg);
} }
this.uploading=false this.uploading = false; //
}) });
} }
}) });
}, },
//
addLink() { addLink() {
this.dataForm.replyContent += '<a href="链接地址">链接文字</a>' this.dataForm.replyContent += '<a href="链接地址">链接文字</a>';
} }
} }
} }
</script> </script>
<style scoped> <style scoped>
.msg-container{ /* 样式部分,但.msg-container类在模板中未使用 */
.msg-container {
background: #eee; background: #eee;
} }
</style> </style>

@ -1,12 +1,16 @@
<template> <template>
<div class="mod-config"> <div class="mod-config">
<!-- 搜索表单 -->
<el-form :inline="true" :model="dataForm" @keyup.enter.native="getDataList()"> <el-form :inline="true" :model="dataForm" @keyup.enter.native="getDataList()">
<el-form-item> <el-form-item>
<!-- 时间选择器 -->
<el-select v-model="dataForm.startTime" placeholder="时间"> <el-select v-model="dataForm.startTime" placeholder="时间">
<el-option v-for="(name,key) in timeSelections" :key="key" :value="name" :label="key"></el-option> <el-option v-for="(name,key) in timeSelections" :key="key" :value="name" :label="key"></el-option>
</el-select> </el-select>
</el-form-item> </el-form-item>
<el-form-item> <el-form-item>
<!-- 消息类型选择器 -->
<el-select v-model="dataForm.msgTypes" placeholder="消息类型"> <el-select v-model="dataForm.msgTypes" placeholder="消息类型">
<el-option value="" label="不限类型"></el-option> <el-option value="" label="不限类型"></el-option>
<el-option value="text,image,voice,shortvideo,video,news,music,location,link" label="消息"></el-option> <el-option value="text,image,voice,shortvideo,video,news,music,location,link" label="消息"></el-option>
@ -14,63 +18,80 @@
</el-select> </el-select>
</el-form-item> </el-form-item>
<el-form-item> <el-form-item>
<!-- 查询按钮 -->
<el-button @click="getDataList()"></el-button> <el-button @click="getDataList()"></el-button>
</el-form-item> </el-form-item>
</el-form> </el-form>
<!-- 消息回复提示 -->
<div class="text-gray"> <div class="text-gray">
24小时内消息可回复此后台展示消息有一分钟左右延迟如需畅聊请使用 24小时内消息可回复此后台展示消息有一分钟左右延迟如需畅聊请使用
<a href="https://mpkf.weixin.qq.com/" target="_blank">公众平台客服</a> <a href="https://mpkf.weixin.qq.com/" target="_blank">公众平台客服</a>
</div> </div>
<!-- 消息列表使用v-loading显示加载状态 -->
<div v-loading="dataListLoading"> <div v-loading="dataListLoading">
<div class="msg-item" v-for="(msg,index) in dataList" :key="index"> <div class="msg-item" v-for="(msg,index) in dataList" :key="index">
<!-- 用户头像 -->
<div class="avatar"><el-avatar shape="square" :size="60" :src="getUserInfo(msg.openid).headimgurl"></el-avatar></div> <div class="avatar"><el-avatar shape="square" :size="60" :src="getUserInfo(msg.openid).headimgurl"></el-avatar></div>
<!-- 消息内容 -->
<div class="item-content"> <div class="item-content">
<div class="flex justify-between margin-bottom"> <div class="flex justify-between margin-bottom">
<div class="text-cut">{{getUserInfo(msg.openid).nickname || '--'}}</div> <div class="text-cut">{{getUserInfo(msg.openid).nickname || '--'}}</div>
<div>{{$moment(msg.createTime).calendar()}}</div> <div>{{$moment(msg.createTime).calendar()}}</div>
<div class="reply-btn"> <div class="reply-btn">
<!-- 回复按钮如果消息在24小时内可点击 -->
<div v-if="canReply(msg.createTime)" @click="replyHandle(msg.openid)" class="el-icon-s-promotion"></div> <div v-if="canReply(msg.createTime)" @click="replyHandle(msg.openid)" class="el-icon-s-promotion"></div>
</div> </div>
</div> </div>
<!-- 消息预览组件 -->
<wx-msg-preview :msg="msg" singleLine></wx-msg-preview> <wx-msg-preview :msg="msg" singleLine></wx-msg-preview>
</div> </div>
</div> </div>
</div> </div>
<!-- 分页组件 -->
<el-pagination @size-change="sizeChangeHandle" @current-change="currentChangeHandle" :current-page="pageIndex" :page-sizes="[10, 20, 50, 100]" :page-size="pageSize" :total="totalCount" layout="total, sizes, prev, pager, next, jumper"> <el-pagination @size-change="sizeChangeHandle" @current-change="currentChangeHandle" :current-page="pageIndex" :page-sizes="[10, 20, 50, 100]" :page-size="pageSize" :total="totalCount" layout="total, sizes, prev, pager, next, jumper">
</el-pagination> </el-pagination>
<!-- 弹窗, 消息回复 --> <!-- 弹窗, 消息回复 -->
<wx-msg-reply ref="wxMsgReply" @success="onReplyed"></wx-msg-reply> <wx-msg-reply ref="wxMsgReply" @success="onReplyed"></wx-msg-reply>
</div> </div>
</template> </template>
<script> <script>
const TIME_FORMAT = 'YYYY/MM/DD hh:mm:ss' const TIME_FORMAT = 'YYYY/MM/DD hh:mm:ss'
export default { export default {
data() { data() {
return { return {
//
timeSelections:{ timeSelections:{
'近24小时':this.$moment().subtract(1, 'days').format(TIME_FORMAT), '近24小时':this.$moment().subtract(1, 'days').format(TIME_FORMAT),
'近3天': this.$moment().subtract(3, 'days').format(TIME_FORMAT), '近3天': this.$moment().subtract(3, 'days').format(TIME_FORMAT),
'近7天': this.$moment().subtract(7, 'days').format(TIME_FORMAT), '近7天': this.$moment().subtract(7, 'days').format(TIME_FORMAT),
'近30天': this.$moment().subtract(30, 'days').format(TIME_FORMAT), '近30天': this.$moment().subtract(30, 'days').format(TIME_FORMAT),
}, },
//
dataForm: { dataForm: {
startTime: this.$moment().subtract(1, 'days').format(TIME_FORMAT), startTime: this.$moment().subtract(1, 'days').format(TIME_FORMAT),
msgTypes: '' msgTypes: ''
}, },
//
dataList: [], dataList: [],
//
userDataList:[], userDataList:[],
//
pageIndex: 1, pageIndex: 1,
pageSize: 20, pageSize: 20,
totalCount: 0, totalCount: 0,
//
dataListLoading: false, dataListLoading: false,
// 使
dataListSelections: [] dataListSelections: []
} }
}, },
components: { components: {
//
WxMsgReply:()=>import('./wx-msg-reply'), WxMsgReply:()=>import('./wx-msg-reply'),
WxMsgPreview:()=>import('@/components/wx-msg-preview') WxMsgPreview:()=>import('@/components/wx-msg-preview')
}, },
//
activated() { activated() {
this.getDataList() this.getDataList()
}, },
@ -101,6 +122,7 @@ export default {
this.dataListLoading = false this.dataListLoading = false
}) })
}, },
//
refreshUserList(msgList){ refreshUserList(msgList){
let openidList=msgList.map(msg=>msg.openid).filter(openid=>!this.userDataList.some(u=>u.openid==openid)) let openidList=msgList.map(msg=>msg.openid).filter(openid=>!this.userDataList.some(u=>u.openid==openid))
if(!openidList.length)return if(!openidList.length)return
@ -115,25 +137,26 @@ export default {
} }
}) })
}, },
// openid
getUserInfo(openid){ getUserInfo(openid){
return this.userDataList.find(u=>u.openid==openid) || {nickname:'--',headimgurl:''} return this.userDataList.find(u=>u.openid==openid) || {nickname:'--',headimgurl:''}
}, },
// 24 // 24
canReply(time){ canReply(time){
return new Date(time).getTime()>new Date().getTime()-24*60*60*1000 return new Date(time).getTime()>new Date().getTime()-24*60*60*1000
}, },
// //
sizeChangeHandle(val) { sizeChangeHandle(val) {
this.pageSize = val this.pageSize = val
this.pageIndex = 1 this.pageIndex = 1
this.getDataList() this.getDataList()
}, },
// //
currentChangeHandle(val) { currentChangeHandle(val) {
this.pageIndex = val this.pageIndex = val
this.getDataList() this.getDataList()
}, },
// // 使
selectionChangeHandle(val) { selectionChangeHandle(val) {
this.dataListSelections = val this.dataListSelections = val
}, },
@ -143,6 +166,7 @@ export default {
this.$refs.wxMsgReply.init(openid) this.$refs.wxMsgReply.init(openid)
}) })
}, },
//
onReplyed(replyMsg){ onReplyed(replyMsg){
this.dataList.unshift({ this.dataList.unshift({
openid : replyMsg.openid, openid : replyMsg.openid,

@ -1,22 +1,43 @@
<template> <template>
<el-dialog :title="!dataForm.id ? '新增' : '修改'" :close-on-click-modal="false" :visible.sync="visible"> <!-- Element UI的对话框组件根据dataForm.id是否存在来设置标题为新增修改 -->
<el-form :model="dataForm" :rules="dataRule" ref="dataForm" @keyup.enter.native="dataFormSubmit()" label-width="100px"> <el-dialog
:title="!dataForm.id ? '新增' : '修改'"
:close-on-click-modal="false" <!-- 点击遮罩层不关闭对话框 -->
:visible.sync="visible"> <!-- 双向绑定控制对话框的显示隐藏 -->
<!-- 表单部分绑定dataForm作为数据模型dataRule作为验证规则 -->
<el-form
:model="dataForm"
:rules="dataRule"
ref="dataForm"
@keyup.enter.native="dataFormSubmit()" <!-- 按下Enter键提交表单 -->
label-width="100px">
<!-- 表单项二维码类型 -->
<el-form-item label="二维码类型" prop="isTemp"> <el-form-item label="二维码类型" prop="isTemp">
<el-radio v-model="dataForm.isTemp" :label="true"></el-radio> <el-radio v-model="dataForm.isTemp" :label="true"></el-radio>
<el-radio v-model="dataForm.isTemp" :label="false"></el-radio> <el-radio v-model="dataForm.isTemp" :label="false"></el-radio>
<div> <div>
<a class="text-warning" v-show="!dataForm.isTemp" target="_blank" href="https://developers.weixin.qq.com/doc/offiaccount/Account_Management/Generating_a_Parametric_QR_Code.html">10</a> <!-- 当不是临时二维码时显示链接 -->
<a class="text-warning" v-show="!dataForm.isTemp" target="_blank" href="https://developers.weixin.qq.com/doc/offiaccount/Account_Management/Generating_a_Parametric_QR_Code.html">
注意永久二维码上限10万个且暂时无法删除旧的二维码
</a>
</div> </div>
</el-form-item> </el-form-item>
<!-- 表单项场景值 -->
<el-form-item label="场景值" prop="sceneStr"> <el-form-item label="场景值" prop="sceneStr">
<el-input v-model="dataForm.sceneStr" placeholder="任意字符串" maxlength="64"></el-input> <el-input v-model="dataForm.sceneStr" placeholder="任意字符串" maxlength="64"></el-input>
</el-form-item> </el-form-item>
<!-- 表单项失效时间仅临时二维码显示 -->
<el-form-item label="失效时间/秒" prop="expireSeconds" v-if="dataForm.isTemp"> <el-form-item label="失效时间/秒" prop="expireSeconds" v-if="dataForm.isTemp">
<el-input v-model="dataForm.expireSeconds" placeholder="单位最大259200030天"></el-input> <el-input v-model="dataForm.expireSeconds" placeholder="单位最大259200030天"></el-input>
<div>最大30天当前设置<span class="text-warning">{{dataForm.expireSeconds/(24*3600)}}</span></div> <div>最大30天当前设置<span class="text-warning">{{dataForm.expireSeconds/(24*3600)}}</span></div>
</el-form-item> </el-form-item>
</el-form> </el-form>
<!-- 对话框底部按钮 -->
<span slot="footer" class="dialog-footer"> <span slot="footer" class="dialog-footer">
<el-button @click="visible = false">取消</el-button> <el-button @click="visible = false">取消</el-button>
<el-button type="primary" @click="dataFormSubmit()"></el-button> <el-button type="primary" @click="dataFormSubmit()"></el-button>
@ -28,54 +49,55 @@
export default { export default {
data() { data() {
return { return {
visible: false, visible: false, //
dataForm: { dataForm: {
isTemp: true, isTemp: true, //
sceneStr: '', sceneStr: '', //
expireSeconds: 2592000 expireSeconds: 2592000 // 30
}, },
dataRule: { dataRule: {
isTemp: [ isTemp: [
{ required: true, message: '二维码类型不能为空', trigger: 'blur' } { required: true, message: '二维码类型不能为空', trigger: 'blur' } //
], ],
sceneStr: [ sceneStr: [
{ required: true, message: '场景值ID不能为空', trigger: 'blur' } { required: true, message: '场景值ID不能为空', trigger: 'blur' } //
], ],
expireSeconds: [ expireSeconds: [
{ required: true, message: '该二维码失效时间不能为空', trigger: 'blur' } { required: true, message: '该二维码失效时间不能为空', trigger: 'blur' } //
] ]
} }
} }
}, },
methods: { methods: {
// id
init(id) { init(id) {
this.dataForm.id = id || 0 this.dataForm.id = id || 0 // dataFormid
this.visible = true this.visible = true //
this.$nextTick(() => { this.$nextTick(() => {
this.$refs['dataForm'].resetFields() this.$refs['dataForm'].resetFields() //
}) })
}, },
// //
dataFormSubmit() { dataFormSubmit() {
this.$refs['dataForm'].validate((valid) => { this.$refs['dataForm'].validate((valid) => {
if (valid) { if (valid) { //
this.$http({ this.$http({ //
url: this.$http.adornUrl(`/manage/wxQrCode/createTicket`), url: this.$http.adornUrl(`/manage/wxQrCode/createTicket`), // URL
method: 'post', method: 'post', //
data: this.$http.adornData(this.dataForm) data: this.$http.adornData(this.dataForm) //
}).then(({ data }) => { }).then(({ data }) => {
if (data && data.code === 200) { if (data && data.code === 200) { //
this.$message({ this.$message({
message: '操作成功', message: '操作成功',
type: 'success', type: 'success',
duration: 1500, duration: 1500,
onClose: () => { onClose: () => {
this.visible = false this.visible = false //
this.$emit('refreshDataList') this.$emit('refreshDataList') //
} }
}) })
} else { } else { //
this.$message.error(data.msg) this.$message.error(data.msg) //
} }
}) })
} }
@ -83,4 +105,4 @@ export default {
} }
} }
} }
</script> </script>

@ -1,67 +1,86 @@
<template> <template>
<div class="mod-config"> <div class="mod-config">
<!-- 内联表单用于输入查询条件和执行操作 -->
<el-form :inline="true" :model="dataForm" @keyup.enter.native="getDataList()"> <el-form :inline="true" :model="dataForm" @keyup.enter.native="getDataList()">
<el-form-item> <el-form-item>
<!-- 输入框用于输入场景值进行查询 -->
<el-input v-model="dataForm.sceneStr" placeholder="场景值" clearable></el-input> <el-input v-model="dataForm.sceneStr" placeholder="场景值" clearable></el-input>
</el-form-item> </el-form-item>
<el-form-item> <el-form-item>
<!-- 查询按钮触发查询操作 -->
<el-button @click="getDataList()"></el-button> <el-button @click="getDataList()"></el-button>
<!-- 新增按钮仅当有权限时显示触发新增操作 -->
<el-button v-if="isAuth('wx:wxqrcode:save')" type="primary" @click="addOrUpdateHandle()"></el-button> <el-button v-if="isAuth('wx:wxqrcode:save')" type="primary" @click="addOrUpdateHandle()"></el-button>
<!-- 批量删除按钮仅当有权限且至少选中一项时显示 -->
<el-button v-if="isAuth('wx:wxqrcode:delete')" type="danger" @click="deleteHandle()" :disabled="dataListSelections.length <= 0"></el-button> <el-button v-if="isAuth('wx:wxqrcode:delete')" type="danger" @click="deleteHandle()" :disabled="dataListSelections.length <= 0"></el-button>
</el-form-item> </el-form-item>
</el-form> </el-form>
<!-- 表格显示数据列表 -->
<el-table :data="dataList" border v-loading="dataListLoading" @selection-change="selectionChangeHandle" style="width: 100%;"> <el-table :data="dataList" border v-loading="dataListLoading" @selection-change="selectionChangeHandle" style="width: 100%;">
<el-table-column type="selection" header-align="center" align="center" width="50"> <el-table-column type="selection" header-align="center" align="center" width="50"></el-table-column>
</el-table-column> <el-table-column prop="id" header-align="center" align="center" label="ID"></el-table-column>
<el-table-column prop="id" header-align="center" align="center" label="ID">
</el-table-column>
<el-table-column prop="isTemp" header-align="center" align="center" label="类型"> <el-table-column prop="isTemp" header-align="center" align="center" label="类型">
<span slot-scope="scope">{{scope.row.isTemp?'临时':'永久'}}</span> <!-- 根据isTemp的值显示临时永久 -->
</el-table-column> <span slot-scope="scope">{{scope.row.isTemp ? '临时' : '永久'}}</span>
<el-table-column prop="sceneStr" header-align="center" align="center" label="场景值">
</el-table-column> </el-table-column>
<el-table-column prop="sceneStr" header-align="center" align="center" label="场景值"></el-table-column>
<el-table-column prop="ticket" header-align="center" align="center" show-overflow-tooltip label="二维码图片"> <el-table-column prop="ticket" header-align="center" align="center" show-overflow-tooltip label="二维码图片">
<!-- 点击链接查看二维码图片 -->
<a :href="'https://mp.weixin.qq.com/cgi-bin/showqrcode?ticket='+scope.row.ticket" slot-scope="scope">{{scope.row.ticket}}</a> <a :href="'https://mp.weixin.qq.com/cgi-bin/showqrcode?ticket='+scope.row.ticket" slot-scope="scope">{{scope.row.ticket}}</a>
</el-table-column> </el-table-column>
<el-table-column prop="url" header-align="center" align="center" show-overflow-tooltip label="解析后的地址"> <el-table-column prop="url" header-align="center" align="center" show-overflow-tooltip label="解析后的地址">
<!-- 点击链接访问解析后的地址 -->
<a :href="scope.row.url" slot-scope="scope">{{scope.row.url}}</a> <a :href="scope.row.url" slot-scope="scope">{{scope.row.url}}</a>
</el-table-column> </el-table-column>
<el-table-column prop="expireTime" header-align="center" align="center" width="100" label="失效时间"> <el-table-column prop="expireTime" header-align="center" align="center" width="100" label="失效时间"></el-table-column>
</el-table-column>
<el-table-column fixed="right" header-align="center" align="center" width="150" label="操作"> <el-table-column fixed="right" header-align="center" align="center" width="150" label="操作">
<!-- 单行删除按钮 -->
<template slot-scope="scope"> <template slot-scope="scope">
<el-button type="text" size="small" @click="deleteHandle(scope.row.id)"></el-button> <el-button type="text" size="small" @click="deleteHandle(scope.row.id)"></el-button>
</template> </template>
</el-table-column> </el-table-column>
</el-table> </el-table>
<el-pagination @size-change="sizeChangeHandle" @current-change="currentChangeHandle" :current-page="pageIndex" :page-sizes="[10, 20, 50, 100]" :page-size="pageSize" :total="totalPage" layout="total, sizes, prev, pager, next, jumper">
</el-pagination> <!-- 分页组件 -->
<!-- 弹窗, 新增 / 修改 --> <el-pagination @size-change="sizeChangeHandle" @current-change="currentChangeHandle" :current-page="pageIndex" :page-sizes="[10, 20, 50, 100]" :page-size="pageSize" :total="totalPage" layout="total, sizes, prev, pager, next, jumper"></el-pagination>
<!-- 弹窗组件用于新增或修改操作 -->
<add-or-update v-if="addOrUpdateVisible" ref="addOrUpdate" @refreshDataList="getDataList"></add-or-update> <add-or-update v-if="addOrUpdateVisible" ref="addOrUpdate" @refreshDataList="getDataList"></add-or-update>
</div> </div>
</template> </template>
<script> <script>
import AddOrUpdate from './wx-qrcode-add-or-update' import AddOrUpdate from './wx-qrcode-add-or-update'
export default { export default {
data() { data() {
return { return {
//
dataForm: { dataForm: {
sceneStr: '' sceneStr: ''
}, },
//
dataList: [], dataList: [],
//
pageIndex: 1, pageIndex: 1,
//
pageSize: 10, pageSize: 10,
//
totalPage: 0, totalPage: 0,
//
dataListLoading: false, dataListLoading: false,
//
dataListSelections: [], dataListSelections: [],
// /
addOrUpdateVisible: false addOrUpdateVisible: false
} }
}, },
components: { components: {
AddOrUpdate AddOrUpdate // /
}, },
activated() { activated() {
//
this.getDataList() this.getDataList()
}, },
methods: { methods: {
@ -69,7 +88,7 @@ export default {
getDataList() { getDataList() {
this.dataListLoading = true this.dataListLoading = true
this.$http({ this.$http({
url: this.$http.adornUrl('/manage/wxQrCode/list'), url: this.$http.adornUrl('/manage/wxQrCode/list'), //
method: 'get', method: 'get',
params: this.$http.adornParams({ params: this.$http.adornParams({
'page': this.pageIndex, 'page': this.pageIndex,
@ -80,63 +99,63 @@ export default {
}) })
}).then(({ data }) => { }).then(({ data }) => {
if (data && data.code === 200) { if (data && data.code === 200) {
this.dataList = data.page.list this.dataList = data.page.list //
this.totalPage = data.page.totalCount this.totalPage = data.page.totalCount //
} else { } else {
this.dataList = [] this.dataList = [] //
this.totalPage = 0 this.totalPage = 0 //
} }
this.dataListLoading = false this.dataListLoading = false //
}) })
}, },
// //
sizeChangeHandle(val) { sizeChangeHandle(val) {
this.pageSize = val this.pageSize = val //
this.pageIndex = 1 this.pageIndex = 1 //
this.getDataList() this.getDataList() //
}, },
// //
currentChangeHandle(val) { currentChangeHandle(val) {
this.pageIndex = val this.pageIndex = val //
this.getDataList() this.getDataList() //
}, },
// //
selectionChangeHandle(val) { selectionChangeHandle(val) {
this.dataListSelections = val this.dataListSelections = val //
}, },
// / //
addOrUpdateHandle(id) { addOrUpdateHandle(id) {
this.addOrUpdateVisible = true this.addOrUpdateVisible = true //
this.$nextTick(() => { this.$nextTick(() => {
this.$refs.addOrUpdate.init(id) this.$refs.addOrUpdate.init(id) // id
}) })
}, },
// //
deleteHandle(id) { deleteHandle(id) {
var ids = id ? [id] : this.dataListSelections.map(item => item.id) var ids = id ? [id] : this.dataListSelections.map(item => item.id) // id
this.$confirm(`确定对[id=${ids.join(',')}]进行[${id ? '删除' : '批量删除'}]操作?(仅删存档)`, '提示', { this.$confirm(`确定对[id=${ids.join(',')}]进行[${id ? '删除' : '批量删除'}]操作?(仅删存档)`, '提示', {
confirmButtonText: '确定', confirmButtonText: '确定',
cancelButtonText: '取消', cancelButtonText: '取消',
type: 'warning' type: 'warning'
}).then(() => { }).then(() => {
this.$http({ this.$http({
url: this.$http.adornUrl('/manage/wxQrCode/delete'), url: this.$http.adornUrl('/manage/wxQrCode/delete'), //
method: 'post', method: 'post',
data: this.$http.adornData(ids, false) data: this.$http.adornData(ids, false) // id
}).then(({ data }) => { }).then(({ data }) => {
if (data && data.code === 200) { if (data && data.code === 200) {
this.$message({ this.$message({
message: '操作成功', message: '操作成功',
type: 'success', type: 'success',
duration: 1500, duration: 1500,
onClose: () => this.getDataList() onClose: () => this.getDataList() //
}) })
} else { } else {
this.$message.error(data.msg) this.$message.error(data.msg) //
} }
}) })
}) })
} }
} }
} }
</script> </script>

@ -1,101 +1,120 @@
<template> <template>
<el-dialog :title="modeDesc[mode]+'用户标签'" :close-on-click-modal="false" :visible.sync="dialogVisible"> <el-dialog :title="modeDesc[mode]+'用户标签'" :close-on-click-modal="false" :visible.sync="dialogVisible">
<!-- 对话框内容开始 -->
<div> <div>
<!-- 标签选择器用户可以选择一个标签 -->
<el-select v-model="selectedTagid" filterable placeholder="请选择标签" style="width:100%"> <el-select v-model="selectedTagid" filterable placeholder="请选择标签" style="width:100%">
<!-- 遍历所有可选标签生成下拉选项 -->
<el-option v-for="tagid in tagidsInOption" :key="tagid" :label="getTagName(tagid)" :value="tagid"></el-option> <el-option v-for="tagid in tagidsInOption" :key="tagid" :label="getTagName(tagid)" :value="tagid"></el-option>
</el-select> </el-select>
<!-- 显示已选择用户的数量 -->
<div style="margin-top:20px;">已选择用户数{{wxUsers.length}}</div> <div style="margin-top:20px;">已选择用户数{{wxUsers.length}}</div>
</div> </div>
<!-- 对话框底部按钮 -->
<span slot="footer" class="dialog-footer"> <span slot="footer" class="dialog-footer">
<!-- 关闭按钮 -->
<el-button @click="dialogVisible=false"></el-button> <el-button @click="dialogVisible=false"></el-button>
<!-- 提交按钮根据提交状态显示不同文字 -->
<el-button type="primary" @click="dataFormSubmit()" :disabled="submitting">{{submitting?'保存中...':'确定'}}</el-button> <el-button type="primary" @click="dataFormSubmit()" :disabled="submitting">{{submitting?'保存中...':'确定'}}</el-button>
</span> </span>
</el-dialog> </el-dialog>
</template> </template>
<script> <script>
import { mapState } from 'vuex' import { mapState } from 'vuex'
export default { export default {
name:'wx-user-tagging', name: 'wx-user-tagging',
props:{ //
wxUsers:Array, props: {
wxUsers: Array,
}, },
data(){ data() {
return{ return {
mode:'tagging',//tagging | untagging //
modeDesc:{ mode: 'tagging',
'tagging':'绑定', //
'untagging':'解绑' modeDesc: {
'tagging': '绑定',
'untagging': '解绑'
}, },
selectedTagid:'', // ID
dialogVisible:false, selectedTagid: '',
submitting:false //
dialogVisible: false,
//
submitting: false
} }
}, },
// Vuexstate
computed: mapState({ computed: mapState({
wxUserTags:state=>state.wxUserTags.tags, //
/** wxUserTags: state => state.wxUserTags.tags,
* 返回下拉选择框中的选项列 //
* 假设 all= 全部标签intersection = 用户标签交集即所有用户都有的 union=用户标签并集即至少一个用户的 tagidsInOption() {
* 那么绑定时可选all-intersection的差集即所有用户都有的就不列出来了 // ID
* 解绑时可选union 即用户有的标签都列出来 let userTags = this.wxUsers.map(u => u.tagidList || []);
*/ //
tagidsInOption(){ if (this.mode === 'tagging') {
let userTags=this.wxUsers.map(u=>u.tagidList || [])//[[1,2],[],[1,3]] let all = this.wxUserTags.map(item => item.id);
if(this.mode=='tagging'){// - return all.filter(tagid => !userTags.every(tagsIdArray => tagsIdArray.indexOf(tagid) > -1));
let all = this.wxUserTags.map(item=>item.id) }
return all.filter(tagid=>!userTags.every(tagsIdArray=>tagsIdArray.indexOf(tagid)>-1)) //
}else if(this.mode=='untagging'){// else if (this.mode === 'untagging') {
let unionSet = new Set(); let unionSet = new Set();
userTags.forEach(tagsIdArray=>{ userTags.forEach(tagsIdArray => {
tagsIdArray.forEach(tagid => unionSet.add(tagid)) tagsIdArray.forEach(tagid => unionSet.add(tagid));
});//unionSet });
return Array.from(unionSet);//unionSet return Array.from(unionSet);
} }
return [] return [];
} }
}), }),
methods:{ methods: {
init(mode){ //
if('tagging'==mode || 'untagging'==mode){ init(mode) {
this.mode=mode; if (mode === 'tagging' || mode === 'untagging') {
this.dialogVisible=true this.mode = mode;
}else{ this.dialogVisible = true;
throw('mode参数有误') } else {
throw ('mode参数有误');
} }
}, },
getTagName(tagid){ // ID
let tag = this.wxUserTags.find(item=>item.id==tagid) getTagName(tagid) {
return tag?tag.name : "?" let tag = this.wxUserTags.find(item => item.id == tagid);
return tag ? tag.name : "?";
}, },
dataFormSubmit(){ //
if(this.submitting)return dataFormSubmit() {
if(!this.selectedTagid){ if (this.submitting) return;
this.$message.error('未选择标签') if (!this.selectedTagid) {
return this.$message.error('未选择标签');
return;
} }
this.submitting=true this.submitting = true;
let openidList=this.wxUsers.map(u=>u.openid) let openidList = this.wxUsers.map(u => u.openid);
//
this.$http({ this.$http({
url: this.$http.adornUrl(`/manage/wxUserTags/${this.mode=='tagging'?'batchTagging':'batchUnTagging'}`), url: this.$http.adornUrl(`/manage/wxUserTags/${this.mode === 'tagging' ? 'batchTagging' : 'batchUnTagging'}`),
method: 'post', method: 'post',
data:this.$http.adornData({ data: this.$http.adornData({
tagid : this.selectedTagid, tagid: this.selectedTagid,
openidList : openidList openidList: openidList
}) })
}).then(({ data }) => { }).then(({ data }) => {
this.submitting=false this.submitting = false;
if (data && data.code === 200) { if (data && data.code === 200) {
//
this.$message({ this.$message({
message: '操作成功,列表数据需稍后刷新查看', message: '操作成功,列表数据需稍后刷新查看',
type: 'success', type: 'success',
onClose: () =>this.dialogVisible=false onClose: () => this.dialogVisible = false
}) });
} else { } else {
this.$message.error(data.msg) //
this.$message.error(data.msg);
} }
});
})
} }
} }
} }

@ -1,110 +1,144 @@
<template> <template>
<div class="mod-config"> <div class="mod-config">
<!-- 使用内联表单绑定数据模型并在按下Enter键时调用获取数据列表的方法 -->
<el-form :inline="true" :model="dataForm" @keyup.enter.native="getDataList()"> <el-form :inline="true" :model="dataForm" @keyup.enter.native="getDataList()">
<!-- 用户标签选择器 -->
<el-form-item> <el-form-item>
<el-select v-model="dataForm.tagid" filterable clearable placeholder="用户标签"> <el-select v-model="dataForm.tagid" filterable clearable placeholder="用户标签">
<!-- 循环遍历wxUserTags数组生成选项 -->
<el-option v-for="item in wxUserTags" :key="item.id" :label="item.name" :value="item.id"></el-option> <el-option v-for="item in wxUserTags" :key="item.id" :label="item.name" :value="item.id"></el-option>
</el-select> </el-select>
</el-form-item> </el-form-item>
<!-- 昵称输入框 -->
<el-form-item> <el-form-item>
<el-input v-model="dataForm.nickname" placeholder="昵称" clearable></el-input> <el-input v-model="dataForm.nickname" placeholder="昵称" clearable></el-input>
</el-form-item> </el-form-item>
<!-- 城市输入框 -->
<el-form-item> <el-form-item>
<el-input v-model="dataForm.city" placeholder="城市" clearable></el-input> <el-input v-model="dataForm.city" placeholder="城市" clearable></el-input>
</el-form-item> </el-form-item>
<!-- 关注场景值输入框 -->
<el-form-item> <el-form-item>
<el-input v-model="dataForm.qrSceneStr" placeholder="关注场景值" clearable></el-input> <el-input v-model="dataForm.qrSceneStr" placeholder="关注场景值" clearable></el-input>
</el-form-item> </el-form-item>
<!-- 操作按钮区域 -->
<el-form-item> <el-form-item>
<el-button @click="getDataList()"></el-button> <el-button @click="getDataList()"></el-button>
<el-button v-if="isAuth('wx:wxuser:save')" type="primary" @click="$refs.wxUserTagging.init('tagging')" :disabled="dataListSelections.length <= 0"></el-button> <!-- 根据权限显示绑定标签按钮如果选中的用户数为0则禁用 -->
<el-button v-if="isAuth('wx:wxuser:save')" type="primary" @click="$refs.wxUserTagging.init('untagging')" :disabled="dataListSelections.length <= 0"></el-button> <el-button v-if="isAuth('wx:wxuser:save')" type="primary" @click="$refs.wxUserTagging.init('tagging')" :disabled="dataListSelections.length <= 0"></el-button>
<!-- 根据权限显示解绑标签按钮如果选中的用户数为0则禁用 -->
<el-button v-if="isAuth('wx:wxuser:save')" type="primary" @click="$refs.wxUserTagging.init('untagging')" :disabled="dataListSelections.length <= 0"></el-button>
<!-- 根据权限显示批量删除按钮如果选中的用户数为0则禁用 -->
<el-button v-if="isAuth('wx:wxuser:delete')" type="danger" @click="deleteHandle()" :disabled="dataListSelections.length <= 0"></el-button> <el-button v-if="isAuth('wx:wxuser:delete')" type="danger" @click="deleteHandle()" :disabled="dataListSelections.length <= 0"></el-button>
</el-form-item> </el-form-item>
<!-- 其他操作按钮 -->
<el-form-item class="fr"> <el-form-item class="fr">
<el-button icon="el-icon-price-tag" type="success" @click="$refs.wxUserTagsEditor.show()"></el-button> <!-- 打开标签管理界面 -->
<el-button icon="el-icon-sort" type="success" @click="syncWxUsers()"></el-button> <el-button icon="el-icon-price-tag" type="success" @click="$refs.wxUserTagsEditor.show()"></el-button>
<!-- 同步粉丝数据 -->
<el-button icon="el-icon-sort" type="success" @click="syncWxUsers()"></el-button>
</el-form-item> </el-form-item>
</el-form> </el-form>
<!-- 数据表格 -->
<el-table :data="dataList" border v-loading="dataListLoading" @selection-change="selectionChangeHandle" style="width: 100%;"> <el-table :data="dataList" border v-loading="dataListLoading" @selection-change="selectionChangeHandle" style="width: 100%;">
<el-table-column type="selection" header-align="center" align="center" width="50"> <!-- 多选框 -->
</el-table-column> <el-table-column type="selection" header-align="center" align="center" width="50"></el-table-column>
<el-table-column prop="openid" header-align="center" align="center" label="openid"> <!-- openid列 -->
</el-table-column> <el-table-column prop="openid" header-align="center" align="center" label="openid"></el-table-column>
<el-table-column prop="nickname" header-align="center" align="center" label="昵称"> <!-- 昵称列 -->
</el-table-column> <el-table-column prop="nickname" header-align="center" align="center" label="昵称"></el-table-column>
<el-table-column prop="sex" header-align="center" align="center" label="性别" :formatter="sexFormat"> <!-- 性别列使用formatter格式化显示 -->
</el-table-column> <el-table-column prop="sex" header-align="center" align="center" label="性别" :formatter="sexFormat"></el-table-column>
<el-table-column prop="city" header-align="center" align="center" label="城市"> <!-- 城市列 -->
</el-table-column> <el-table-column prop="city" header-align="center" align="center" label="城市"></el-table-column>
<!-- 头像列使用插槽自定义显示 -->
<el-table-column prop="headimgurl" header-align="center" align="center" label="头像"> <el-table-column prop="headimgurl" header-align="center" align="center" label="头像">
<img class="headimg" slot-scope="scope" v-if="scope.row.headimgurl" :src="scope.row.headimgurl" /> <img class="headimg" slot-scope="scope" v-if="scope.row.headimgurl" :src="scope.row.headimgurl" />
</el-table-column> </el-table-column>
<!-- 标签列使用插槽和循环显示多个标签 -->
<el-table-column prop="tagidList" header-align="center" align="center" label="标签" show-overflow-tooltip> <el-table-column prop="tagidList" header-align="center" align="center" label="标签" show-overflow-tooltip>
<template slot-scope="scope"> <template slot-scope="scope">
<span v-for="tagid in scope.row.tagidList" :key="tagid">{{getTagName(tagid)}} </span> <span v-for="tagid in scope.row.tagidList" :key="tagid">{{getTagName(tagid)}} </span>
</template> </template>
</el-table-column> </el-table-column>
<!-- 订阅时间列使用moment格式化显示 -->
<el-table-column prop="subscribeTime" header-align="center" align="center" label="订阅时间"> <el-table-column prop="subscribeTime" header-align="center" align="center" label="订阅时间">
<template slot-scope="scope">{{$moment(scope.row.subscribeTime).calendar()}}</template> <template slot-scope="scope">{{$moment(scope.row.subscribeTime).calendar()}}</template>
</el-table-column> </el-table-column>
<el-table-column prop="qrSceneStr" header-align="center" align="center" label="场景值"> <!-- 场景值列 -->
</el-table-column> <el-table-column prop="qrSceneStr" header-align="center" align="center" label="场景值"></el-table-column>
<!-- 是否关注列使用插槽自定义显示 -->
<el-table-column prop="subscribe" header-align="center" align="center" label="是否关注"> <el-table-column prop="subscribe" header-align="center" align="center" label="是否关注">
<span slot-scope="scope">{{scope.row.subscribe?"是":"否"}}</span> <span slot-scope="scope">{{scope.row.subscribe?"是":"否"}}</span>
</el-table-column> </el-table-column>
<!-- 操作列包含删除按钮 -->
<el-table-column fixed="right" header-align="center" align="center" width="150" label="操作"> <el-table-column fixed="right" header-align="center" align="center" width="150" label="操作">
<template slot-scope="scope"> <template slot-scope="scope">
<el-button type="text" size="small" @click="deleteHandle(scope.row.openid)"></el-button> <el-button type="text" size="small" @click="deleteHandle(scope.row.openid)"></el-button>
</template> </template>
</el-table-column> </el-table-column>
</el-table> </el-table>
<el-pagination @size-change="sizeChangeHandle" @current-change="currentChangeHandle" :current-page="pageIndex" :page-sizes="[10, 20, 50, 100]" :page-size="pageSize" :total="totalPage" layout="total, sizes, prev, pager, next, jumper"> <!-- 分页组件 -->
</el-pagination> <el-pagination @size-change="sizeChangeHandle" @current-change="currentChangeHandle" :current-page="pageIndex" :page-sizes="[10, 20, 50, 100]" :page-size="pageSize" :total="totalPage" layout="total, sizes, prev, pager, next, jumper"></el-pagination>
<!-- 标签管理组件 -->
<wx-user-tags-manager ref="wxUserTagsEditor" :visible="showWxUserTagsEditor" @close="showWxUserTagsEditor=false"></wx-user-tags-manager> <wx-user-tags-manager ref="wxUserTagsEditor" :visible="showWxUserTagsEditor" @close="showWxUserTagsEditor=false"></wx-user-tags-manager>
<!-- 用户标签绑定/解绑组件 -->
<wx-user-tagging ref="wxUserTagging" :wxUsers="dataListSelections"></wx-user-tagging> <wx-user-tagging ref="wxUserTagging" :wxUsers="dataListSelections"></wx-user-tagging>
</div> </div>
</template> </template>
<script> <script>
//
import WxUserTagsManager from '@/components/wx-user-tags-manager' import WxUserTagsManager from '@/components/wx-user-tags-manager'
import WxUserTagging from './wx-user-tagging' import WxUserTagging from './wx-user-tagging'
// vuexmapState
import { mapState } from 'vuex' import { mapState } from 'vuex'
export default { export default {
data() { data() {
return { return {
//
dataForm: { dataForm: {
tagid:'', tagid:'',
nickname: '', nickname: '',
city:'', city:'',
qrSceneStr:'' qrSceneStr:''
}, },
//
dataList: [], dataList: [],
//
pageIndex: 1, pageIndex: 1,
//
pageSize: 10, pageSize: 10,
//
totalPage: 0, totalPage: 0,
//
showWxUserTagsEditor:false, showWxUserTagsEditor:false,
//
dataListLoading: false, dataListLoading: false,
//
dataListSelections: [], dataListSelections: [],
} }
}, },
//
components: { components: {
WxUserTagsManager,WxUserTagging WxUserTagsManager, WxUserTagging
}, },
// keep-alive
activated() { activated() {
this.getDataList() this.getDataList() //
}, },
// vuexstatewxUserTags
computed: mapState({ computed: mapState({
wxUserTags:state=>state.wxUserTags.tags wxUserTags: state => state.wxUserTags.tags
}), }),
methods: { methods: {
// //
getDataList() { getDataList() {
this.dataListLoading = true this.dataListLoading = true //
this.$http({ this.$http({ // HTTP
url: this.$http.adornUrl('/manage/wxUser/list'), url: this.$http.adornUrl('/manage/wxUser/list'), // URL
method: 'get', method: 'get', //
params: this.$http.adornParams({ params: this.$http.adornParams({ //
'page': this.pageIndex, 'page': this.pageIndex,
'limit': this.pageSize, 'limit': this.pageSize,
'nickname': this.dataForm.nickname, 'nickname': this.dataForm.nickname,
@ -114,7 +148,7 @@ export default {
'sidx': 'subscribe_time', 'sidx': 'subscribe_time',
'order': 'desc' 'order': 'desc'
}) })
}).then(({ data }) => { }).then(({ data }) => { //
if (data && data.code === 200) { if (data && data.code === 200) {
this.dataList = data.page.list this.dataList = data.page.list
this.totalPage = data.page.totalCount this.totalPage = data.page.totalCount
@ -122,33 +156,34 @@ export default {
this.dataList = [] this.dataList = []
this.totalPage = 0 this.totalPage = 0
} }
this.dataListLoading = false this.dataListLoading = false //
}) })
}, },
// //
sizeChangeHandle(val) { sizeChangeHandle(val) {
this.pageSize = val this.pageSize = val
this.pageIndex = 1 this.pageIndex = 1
this.getDataList() this.getDataList()
}, },
// //
currentChangeHandle(val) { currentChangeHandle(val) {
this.pageIndex = val this.pageIndex = val
this.getDataList() this.getDataList()
}, },
// //
selectionChangeHandle(val) { selectionChangeHandle(val) {
this.dataListSelections = val this.dataListSelections = val
}, },
// //
deleteHandle(id) { deleteHandle(id) {
// ID
var ids = id ? [id] : this.dataListSelections.map(item => item.openid) var ids = id ? [id] : this.dataListSelections.map(item => item.openid)
this.$confirm(`确定对[id=${ids.join(',')}]进行[${id ? '删除' : '批量删除'}]操作?`, '提示', { this.$confirm(`确定对[id=${ids.join(',')}]进行[${id ? '删除' : '批量删除'}]操作?`, '提示', {
confirmButtonText: '确定', confirmButtonText: '确定',
cancelButtonText: '取消', cancelButtonText: '取消',
type: 'warning' type: 'warning'
}).then(() => { }).then(() => {
this.$http({ this.$http({ //
url: this.$http.adornUrl('/manage/wxUser/delete'), url: this.$http.adornUrl('/manage/wxUser/delete'),
method: 'post', method: 'post',
data: this.$http.adornData(ids, false) data: this.$http.adornData(ids, false)
@ -159,7 +194,7 @@ export default {
type: 'success', type: 'success',
duration: 1500, duration: 1500,
onClose: () => { onClose: () => {
this.getDataList() this.getDataList() //
} }
}) })
} else { } else {
@ -168,8 +203,9 @@ export default {
}) })
}) })
}, },
//
syncWxUsers(){ syncWxUsers(){
this.$http({ this.$http({ //
url: this.$http.adornUrl('/manage/wxUser/syncWxUsers'), url: this.$http.adornUrl('/manage/wxUser/syncWxUsers'),
method: 'post', method: 'post',
}).then(({ data }) => { }).then(({ data }) => {
@ -185,6 +221,7 @@ export default {
}) })
}, },
//
sexFormat(row, column, cellValue) { sexFormat(row, column, cellValue) {
let sexType = { let sexType = {
0: '未知', 0: '未知',
@ -193,6 +230,7 @@ export default {
} }
return sexType[cellValue]; return sexType[cellValue];
}, },
// tagid
getTagName(tagid){ getTagName(tagid){
let tag = this.wxUserTags.find(item=>item.id==tagid) let tag = this.wxUserTags.find(item=>item.id==tagid)
return tag?tag.name : "?" return tag?tag.name : "?"
@ -202,8 +240,11 @@ export default {
</script> </script>
<style scoped> <style scoped>
.headimg{ .headimg{
width: 50px; width: 50px;
height: 50px; /* // 头像宽度 */
border-radius: 8px; height: 50px;
/* // 头像高度 */
border-radius: 8px;
/* // 头像圆角 */
} }
</style> </style>
Loading…
Cancel
Save