标签页以tab形式展示
This commit is contained in:
5
zyplayer-doc-ui/db-ui/package-lock.json
generated
5
zyplayer-doc-ui/db-ui/package-lock.json
generated
@@ -7374,6 +7374,11 @@
|
|||||||
"integrity": "sha1-HuO8mhbsv1EYvjNLsV+cRvgvWCU=",
|
"integrity": "sha1-HuO8mhbsv1EYvjNLsV+cRvgvWCU=",
|
||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
|
"vuex": {
|
||||||
|
"version": "3.3.0",
|
||||||
|
"resolved": "https://registry.npm.taobao.org/vuex/download/vuex-3.3.0.tgz",
|
||||||
|
"integrity": "sha1-ZltGMOoTRzFxOfzFy0laqz7F5RM="
|
||||||
|
},
|
||||||
"wangeditor": {
|
"wangeditor": {
|
||||||
"version": "3.1.1",
|
"version": "3.1.1",
|
||||||
"resolved": "https://registry.npm.taobao.org/wangeditor/download/wangeditor-3.1.1.tgz",
|
"resolved": "https://registry.npm.taobao.org/wangeditor/download/wangeditor-3.1.1.tgz",
|
||||||
|
|||||||
@@ -13,6 +13,7 @@
|
|||||||
"vue": "^2.5.16",
|
"vue": "^2.5.16",
|
||||||
"vue-axios": "^2.1.4",
|
"vue-axios": "^2.1.4",
|
||||||
"vue-router": "^3.0.6",
|
"vue-router": "^3.0.6",
|
||||||
|
"vuex": "^3.3.0",
|
||||||
"wangeditor": "^3.1.1"
|
"wangeditor": "^3.1.1"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
|
|||||||
@@ -5,14 +5,14 @@ var href = window.location.href;
|
|||||||
var _fn = {
|
var _fn = {
|
||||||
href: href,
|
href: href,
|
||||||
// 本地启动时使用本地接口调试
|
// 本地启动时使用本地接口调试
|
||||||
// HOST: 'http://local.zyplayer.com:8083/zyplayer-doc-manage',
|
HOST: 'http://local.zyplayer.com:8083/zyplayer-doc-manage',
|
||||||
// HOST1: 'http://local.zyplayer.com:8083/zyplayer-doc-manage',
|
HOST1: 'http://local.zyplayer.com:8083/zyplayer-doc-manage',
|
||||||
// 也可以直接使用线上的服务调试
|
// 也可以直接使用线上的服务调试
|
||||||
// HOST: 'http://doc.zyplayer.com/zyplayer-doc-manage',
|
// HOST: 'http://doc.zyplayer.com/zyplayer-doc-manage',
|
||||||
// HOST1: 'http://doc.zyplayer.com/zyplayer-doc-manage',
|
// HOST1: 'http://doc.zyplayer.com/zyplayer-doc-manage',
|
||||||
// 打包时使用下面这两行,文件就放在根目录下,所以当前路劲就好
|
// 打包时使用下面这两行,文件就放在根目录下,所以当前路劲就好
|
||||||
HOST: './',
|
// HOST: './',
|
||||||
HOST1: './',
|
// HOST1: './',
|
||||||
|
|
||||||
mixUrl: function (host, url) {
|
mixUrl: function (host, url) {
|
||||||
var p;
|
var p;
|
||||||
|
|||||||
141
zyplayer-doc-ui/db-ui/src/components/layouts/PageTableView.vue
Normal file
141
zyplayer-doc-ui/db-ui/src/components/layouts/PageTableView.vue
Normal file
@@ -0,0 +1,141 @@
|
|||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<el-tabs v-model="activePage" type="card" closable @tab-click="changePage" @tab-remove="removePageTab" style="padding: 5px 10px 0;">
|
||||||
|
<el-tab-pane :label="pageTabNameMap[item.fullPath]||item.name" :name="item.fullPath" v-for="item in pageList"/>
|
||||||
|
</el-tabs>
|
||||||
|
<keep-alive>
|
||||||
|
<router-view/>
|
||||||
|
</keep-alive>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
name: 'MenuView',
|
||||||
|
components: {},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
pageList: [],
|
||||||
|
linkList: [],
|
||||||
|
activePage: '',
|
||||||
|
multiPage: true,
|
||||||
|
menuVisible: false,
|
||||||
|
menuItemList: [
|
||||||
|
{key: '1', icon: 'arrow-left', text: '关闭左侧'},
|
||||||
|
{key: '2', icon: 'arrow-right', text: '关闭右侧'},
|
||||||
|
{key: '3', icon: 'close', text: '关闭其它'}
|
||||||
|
],
|
||||||
|
}
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
pageTabNameMap () {
|
||||||
|
return this.$store.state.global.pageTabNameMap;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
created() {
|
||||||
|
this.pageList.push(this.$route);
|
||||||
|
this.linkList.push(this.$route.fullPath);
|
||||||
|
this.activePage = this.$route.fullPath;
|
||||||
|
},
|
||||||
|
watch: {
|
||||||
|
'$route': function (newRoute, oldRoute) {
|
||||||
|
this.activePage = newRoute.fullPath;
|
||||||
|
if (!this.multiPage) {
|
||||||
|
this.linkList = [newRoute.fullPath];
|
||||||
|
this.pageList = [newRoute];
|
||||||
|
} else if (this.linkList.indexOf(newRoute.fullPath) < 0) {
|
||||||
|
this.linkList.push(newRoute.fullPath);
|
||||||
|
this.pageList.push(newRoute);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
'activePage': function (key) {
|
||||||
|
this.$router.push(key)
|
||||||
|
},
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
changePage(key) {
|
||||||
|
this.activePage = key.name;
|
||||||
|
},
|
||||||
|
editPage(key, action) {
|
||||||
|
this[action](key);
|
||||||
|
},
|
||||||
|
removePageTab(key) {
|
||||||
|
if (this.pageList.length === 1) {
|
||||||
|
this.$message.warning('这是最后一页,不能再关闭了啦');
|
||||||
|
return
|
||||||
|
}
|
||||||
|
this.pageList = this.pageList.filter(item => item.fullPath !== key);
|
||||||
|
let index = this.linkList.indexOf(key);
|
||||||
|
this.linkList = this.linkList.filter(item => item !== key);
|
||||||
|
index = index >= this.linkList.length ? this.linkList.length - 1 : index;
|
||||||
|
this.activePage = this.linkList[index];
|
||||||
|
},
|
||||||
|
onContextmenu(e) {
|
||||||
|
const pagekey = this.getPageKey(e.target)
|
||||||
|
if (pagekey !== null) {
|
||||||
|
e.preventDefault()
|
||||||
|
this.menuVisible = true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
/**
|
||||||
|
* 由于ant-design-vue组件库的TabPane组件暂不支持自定义监听器,无法直接获取到右键target所在标签页的 pagekey 。故增加此方法用于
|
||||||
|
* 查询右键target所在标签页的标识 pagekey ,以用于自定义右键菜单的事件处理。
|
||||||
|
* 注:TabPane组件支持自定义监听器后可去除该方法并重构 ‘自定义右键菜单的事件处理’
|
||||||
|
* @param target 查询开始目标
|
||||||
|
* @param count 查询层级深度 (查找层级最多不超过3层,超过3层深度直接返回 null)
|
||||||
|
* @returns {String}
|
||||||
|
*/
|
||||||
|
getPageKey(target, depth) {
|
||||||
|
depth = depth || 0
|
||||||
|
if (depth > 2) {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
let pageKey = target.getAttribute('pagekey')
|
||||||
|
pageKey = pageKey || (target.previousElementSibling ? target.previousElementSibling.getAttribute('pagekey') : null)
|
||||||
|
return pageKey || (target.firstElementChild ? this.getPageKey(target.firstElementChild, ++depth) : null)
|
||||||
|
},
|
||||||
|
onMenuSelect(key, target) {
|
||||||
|
let pageKey = this.getPageKey(target)
|
||||||
|
switch (key) {
|
||||||
|
case '1':
|
||||||
|
this.closeLeft(pageKey)
|
||||||
|
break
|
||||||
|
case '2':
|
||||||
|
this.closeRight(pageKey)
|
||||||
|
break
|
||||||
|
case '3':
|
||||||
|
this.closeOthers(pageKey)
|
||||||
|
break
|
||||||
|
default:
|
||||||
|
break
|
||||||
|
}
|
||||||
|
},
|
||||||
|
closeOthers(pageKey) {
|
||||||
|
let index = this.linkList.indexOf(pageKey)
|
||||||
|
this.linkList = this.linkList.slice(index, index + 1)
|
||||||
|
this.pageList = this.pageList.slice(index, index + 1)
|
||||||
|
this.activePage = this.linkList[0]
|
||||||
|
},
|
||||||
|
closeLeft(pageKey) {
|
||||||
|
let index = this.linkList.indexOf(pageKey)
|
||||||
|
this.linkList = this.linkList.slice(index)
|
||||||
|
this.pageList = this.pageList.slice(index)
|
||||||
|
if (this.linkList.indexOf(this.activePage) < 0) {
|
||||||
|
this.activePage = this.linkList[0]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
closeRight(pageKey) {
|
||||||
|
let index = this.linkList.indexOf(pageKey)
|
||||||
|
this.linkList = this.linkList.slice(0, index + 1)
|
||||||
|
this.pageList = this.pageList.slice(0, index + 1)
|
||||||
|
if (this.linkList.indexOf(this.activePage < 0)) {
|
||||||
|
this.activePage = this.linkList[this.linkList.length - 1]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
|
||||||
|
</style>
|
||||||
@@ -10,6 +10,7 @@ import toast from './common/lib/common/toast'
|
|||||||
|
|
||||||
import VueRouter from 'vue-router'
|
import VueRouter from 'vue-router'
|
||||||
import routes from './routes'
|
import routes from './routes'
|
||||||
|
import store from './store/index'
|
||||||
import axios from 'axios'
|
import axios from 'axios'
|
||||||
import VueAxios from 'vue-axios'
|
import VueAxios from 'vue-axios'
|
||||||
|
|
||||||
@@ -25,6 +26,7 @@ Vue.prototype.apilist2 = apimix.apilist1;
|
|||||||
// 公用方法
|
// 公用方法
|
||||||
Vue.prototype.common = common;
|
Vue.prototype.common = common;
|
||||||
Vue.prototype.toast = toast;
|
Vue.prototype.toast = toast;
|
||||||
|
Vue.prototype.$store = store;
|
||||||
|
|
||||||
const router = new VueRouter({routes});
|
const router = new VueRouter({routes});
|
||||||
// 路由跳转时判断处理
|
// 路由跳转时判断处理
|
||||||
|
|||||||
@@ -4,6 +4,8 @@ import UserLogin from './views/user/Login.vue'
|
|||||||
import UserMyInfo from './views/user/MyInfo.vue'
|
import UserMyInfo from './views/user/MyInfo.vue'
|
||||||
import UserRouterView from './views/user/RouterView.vue'
|
import UserRouterView from './views/user/RouterView.vue'
|
||||||
|
|
||||||
|
import PageTableView from './components/layouts/PageTableView'
|
||||||
|
|
||||||
import TableInfo from './views/table/Info.vue'
|
import TableInfo from './views/table/Info.vue'
|
||||||
import TableDatabase from './views/table/Database.vue'
|
import TableDatabase from './views/table/Database.vue'
|
||||||
import TableRouterView from './views/table/RouterView.vue'
|
import TableRouterView from './views/table/RouterView.vue'
|
||||||
@@ -18,14 +20,29 @@ import CommonNoAuth from './views/common/NoAuth.vue'
|
|||||||
|
|
||||||
let routes = [
|
let routes = [
|
||||||
{
|
{
|
||||||
path: '/home',
|
path: '/home1',
|
||||||
component: Home,
|
component: Home,
|
||||||
name: '主页',
|
name: '主页',
|
||||||
meta: {
|
meta: {
|
||||||
requireAuth: true,
|
requireAuth: true,
|
||||||
}
|
}
|
||||||
}, {
|
}, {
|
||||||
path: '/user',
|
path: '/',
|
||||||
|
name: 'Tab标签页',
|
||||||
|
component: PageTableView,
|
||||||
|
children: [
|
||||||
|
{path: '/home', name: '主页',component: Home},
|
||||||
|
{path: '/user/login', name: '系统登录',component: UserLogin, meta: {fullscreen: true}},
|
||||||
|
{path: '/user/myInfo', name: '我的信息',component: UserMyInfo},
|
||||||
|
{path: '/table/info', name: '表信息',component: TableInfo},
|
||||||
|
{path: '/table/database', name: '库信息',component: TableDatabase},
|
||||||
|
{path: '/data/datasourceManage', name: '数据源管理',component: DataDatasourceManage},
|
||||||
|
{path: '/data/export', name: '数据库导出',component: DataExport},
|
||||||
|
{path: '/data/executor', name: 'SQL执行器',component: DataExecutor},
|
||||||
|
{path: '/data/transferData', name: '数据互导工具',component: DataTransferData},
|
||||||
|
]
|
||||||
|
}, {
|
||||||
|
path: '/user1',
|
||||||
name: '用户管理',
|
name: '用户管理',
|
||||||
component: UserRouterView,
|
component: UserRouterView,
|
||||||
children: [
|
children: [
|
||||||
@@ -33,7 +50,7 @@ let routes = [
|
|||||||
{path: 'myInfo', name: '我的信息',component: UserMyInfo},
|
{path: 'myInfo', name: '我的信息',component: UserMyInfo},
|
||||||
]
|
]
|
||||||
}, {
|
}, {
|
||||||
path: '/table',
|
path: '/table1',
|
||||||
name: '表信息',
|
name: '表信息',
|
||||||
component: TableRouterView,
|
component: TableRouterView,
|
||||||
children: [
|
children: [
|
||||||
@@ -41,7 +58,7 @@ let routes = [
|
|||||||
{path: 'database', name: '库信息',component: TableDatabase},
|
{path: 'database', name: '库信息',component: TableDatabase},
|
||||||
]
|
]
|
||||||
}, {
|
}, {
|
||||||
path: '/data',
|
path: '/data1',
|
||||||
name: '数据信息',
|
name: '数据信息',
|
||||||
component: DataRouterView,
|
component: DataRouterView,
|
||||||
children: [
|
children: [
|
||||||
|
|||||||
11
zyplayer-doc-ui/db-ui/src/store/index.js
Normal file
11
zyplayer-doc-ui/db-ui/src/store/index.js
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
import Vue from 'vue'
|
||||||
|
import Vuex from 'vuex'
|
||||||
|
import global from './modules/global'
|
||||||
|
|
||||||
|
Vue.use(Vuex);
|
||||||
|
|
||||||
|
export default new Vuex.Store({
|
||||||
|
modules: {
|
||||||
|
global,
|
||||||
|
}
|
||||||
|
});
|
||||||
18
zyplayer-doc-ui/db-ui/src/store/modules/global.js
Normal file
18
zyplayer-doc-ui/db-ui/src/store/modules/global.js
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
export default {
|
||||||
|
namespaced: true,
|
||||||
|
state: {
|
||||||
|
pageTabNameMap: {},
|
||||||
|
},
|
||||||
|
getters: {
|
||||||
|
getPageTabNameMap(state) {
|
||||||
|
return state.pageTabNameMap;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
mutations: {
|
||||||
|
addTableName(state, item) {
|
||||||
|
let sameObj = Object.assign({}, state.pageTabNameMap);
|
||||||
|
sameObj[item.key] = item.val;
|
||||||
|
state.pageTabNameMap = sameObj;
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,13 +1,8 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="data-executor-vue">
|
<div class="data-executor-vue">
|
||||||
<div style="padding: 10px;height: 100%;box-sizing: border-box;background: #fafafa;">
|
<div style="padding: 0 10px;height: 100%;box-sizing: border-box;">
|
||||||
<el-card style="margin-bottom: 10px;">
|
<el-card style="margin-bottom: 10px;">
|
||||||
<div>
|
<pre id="sqlExecutorEditor" style="width: 100%;height: 500px;margin-top: 0;"></pre>
|
||||||
<!-- <el-select v-model="choiceDatabase" @change="databaseChangeEvents" filterable placeholder="请选择数据库">-->
|
|
||||||
<!-- <el-option v-for="item in databaseList" :key="item.dbName" :label="item.dbName" :value="item.dbName"></el-option>-->
|
|
||||||
<!-- </el-select>-->
|
|
||||||
</div>
|
|
||||||
<pre id="sqlExecutorEditor" style="width: 100%;height: 500px;"></pre>
|
|
||||||
<div>
|
<div>
|
||||||
<el-button v-if="sqlExecuting" v-on:click="cancelExecutorSql" type="primary" plain size="small" icon="el-icon-video-pause">取消执行</el-button>
|
<el-button v-if="sqlExecuting" v-on:click="cancelExecutorSql" type="primary" plain size="small" icon="el-icon-video-pause">取消执行</el-button>
|
||||||
<el-tooltip v-else effect="dark" content="Ctrl+R、Ctrl+Enter" placement="top">
|
<el-tooltip v-else effect="dark" content="Ctrl+R、Ctrl+Enter" placement="top">
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
<template>
|
<template>
|
||||||
<div id="app">
|
<div id="app">
|
||||||
<div style="padding: 10px;height: 100%;box-sizing: border-box;background: #fafafa;">
|
<div style="padding: 10px;height: 100%;box-sizing: border-box;">
|
||||||
<el-card style="margin: 10px;">
|
<el-card style="margin: 10px;">
|
||||||
<div slot="header" class="clearfix">
|
<div slot="header" class="clearfix">
|
||||||
<span>数据库表导出</span>
|
<span>数据库表导出</span>
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="data-transfer-vue">
|
<div class="data-transfer-vue">
|
||||||
<div style="padding: 10px;height: 100%;box-sizing: border-box;background: #fafafa;">
|
<div style="padding: 0 10px;height: 100%;box-sizing: border-box;">
|
||||||
<el-card style="margin: 10px;">
|
<el-card style="margin: 10px;">
|
||||||
<div slot="header" class="clearfix">
|
<div slot="header" class="clearfix">
|
||||||
<span>数据互导工具</span>
|
<span>数据互导工具</span>
|
||||||
|
|||||||
@@ -115,6 +115,11 @@
|
|||||||
tableInfo.newDesc = tableInfo.description;
|
tableInfo.newDesc = tableInfo.description;
|
||||||
app.tableInfo = tableInfo;
|
app.tableInfo = tableInfo;
|
||||||
app.columnListLoading = false;
|
app.columnListLoading = false;
|
||||||
|
var newName = {key: app.$route.fullPath, val: tableInfo.tableName};
|
||||||
|
app.$store.commit('global/addTableName', newName);
|
||||||
|
app.$forceUpdate();
|
||||||
|
console.log(newName)
|
||||||
|
// app.$store.state.global.pageTabNameMap
|
||||||
});
|
});
|
||||||
this.common.post(this.apilist1.tableStatus, this.vueQueryParam, function (json) {
|
this.common.post(this.apilist1.tableStatus, this.vueQueryParam, function (json) {
|
||||||
app.tableStatusInfo = json.data || {};
|
app.tableStatusInfo = json.data || {};
|
||||||
|
|||||||
Reference in New Issue
Block a user