2021-07-28 18:03:19 +08:00
|
|
|
|
<template>
|
|
|
|
|
|
<div>
|
2021-12-21 03:06:29 +00:00
|
|
|
|
<el-card>
|
2021-07-28 18:03:19 +08:00
|
|
|
|
<div style="float: left">
|
|
|
|
|
|
<el-row type="flex" justify="space-between">
|
|
|
|
|
|
<el-col :span="24">
|
|
|
|
|
|
<project-env-select @changeProjectEnv="changeProjectEnv" @clear="clearRedis">
|
|
|
|
|
|
<template #default>
|
|
|
|
|
|
<el-form-item label="redis" label-width="40px">
|
|
|
|
|
|
<el-select v-model="scanParam.id" placeholder="请选择redis" @change="changeRedis" @clear="clearRedis" clearable>
|
|
|
|
|
|
<el-option v-for="item in redisList" :key="item.id" :label="item.host" :value="item.id">
|
|
|
|
|
|
<span style="float: left">{{ item.host }}</span>
|
|
|
|
|
|
<span style="float: right; color: #8492a6; margin-left: 6px; font-size: 13px">{{
|
|
|
|
|
|
`库: [${item.db}]`
|
|
|
|
|
|
}}</span>
|
|
|
|
|
|
</el-option>
|
|
|
|
|
|
</el-select>
|
|
|
|
|
|
</el-form-item>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
</project-env-select>
|
|
|
|
|
|
</el-col>
|
2022-01-12 16:00:31 +08:00
|
|
|
|
<el-col class="mt10">
|
|
|
|
|
|
<el-form class="search-form" label-position="right" :inline="true" label-width="60px">
|
|
|
|
|
|
<el-form-item label="key" label-width="40px">
|
|
|
|
|
|
<el-input
|
2022-08-05 21:41:21 +08:00
|
|
|
|
placeholder="match 支持*模糊key"
|
2022-02-14 14:58:25 +08:00
|
|
|
|
style="width: 240px"
|
2022-01-12 16:00:31 +08:00
|
|
|
|
v-model="scanParam.match"
|
|
|
|
|
|
@clear="clear()"
|
|
|
|
|
|
clearable
|
|
|
|
|
|
></el-input>
|
|
|
|
|
|
</el-form-item>
|
|
|
|
|
|
<el-form-item label="count" label-width="60px">
|
2022-07-10 12:14:06 +08:00
|
|
|
|
<el-input placeholder="count" style="width: 62px" v-model.number="scanParam.count"></el-input>
|
2022-01-12 16:00:31 +08:00
|
|
|
|
</el-form-item>
|
|
|
|
|
|
<el-form-item>
|
|
|
|
|
|
<el-button @click="searchKey()" type="success" icon="search" plain></el-button>
|
|
|
|
|
|
<el-button @click="scan()" icon="bottom" plain>scan</el-button>
|
2022-08-05 21:41:21 +08:00
|
|
|
|
<el-popover placement="right" :width="200" trigger="click">
|
|
|
|
|
|
<template #reference>
|
|
|
|
|
|
<el-button type="primary" icon="plus" plain></el-button>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
<el-tag @click="onAddData('string')" :color="getTypeColor('string')" style="cursor: pointer">string</el-tag>
|
|
|
|
|
|
<el-tag @click="onAddData('hash')" :color="getTypeColor('hash')" class="ml5" style="cursor: pointer">hash</el-tag>
|
|
|
|
|
|
<el-tag @click="onAddData('set')" :color="getTypeColor('set')" class="ml5" style="cursor: pointer">set</el-tag>
|
2022-09-22 11:56:21 +08:00
|
|
|
|
<!-- <el-tag @click="onAddData('list')" :color="getTypeColor('list')" class="ml5" style="cursor: pointer">list</el-tag> -->
|
2022-08-05 21:41:21 +08:00
|
|
|
|
</el-popover>
|
2022-01-12 16:00:31 +08:00
|
|
|
|
</el-form-item>
|
|
|
|
|
|
<div style="float: right">
|
|
|
|
|
|
<span>keys: {{ dbsize }}</span>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</el-form>
|
|
|
|
|
|
</el-col>
|
2021-07-28 18:03:19 +08:00
|
|
|
|
</el-row>
|
|
|
|
|
|
</div>
|
2022-01-12 16:00:31 +08:00
|
|
|
|
|
2021-12-21 03:06:29 +00:00
|
|
|
|
<el-table v-loading="loading" :data="keys" stripe :highlight-current-row="true" style="cursor: pointer">
|
|
|
|
|
|
<el-table-column show-overflow-tooltip prop="key" label="key"></el-table-column>
|
2022-03-24 17:50:44 +08:00
|
|
|
|
<el-table-column prop="type" label="type" width="80">
|
|
|
|
|
|
<template #default="scope">
|
|
|
|
|
|
<el-tag :color="getTypeColor(scope.row.type)" size="small">{{ scope.row.type }}</el-tag>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
</el-table-column>
|
2022-01-21 11:28:55 +08:00
|
|
|
|
<el-table-column prop="ttl" label="ttl(过期时间)" width="130">
|
2021-12-21 03:06:29 +00:00
|
|
|
|
<template #default="scope">
|
|
|
|
|
|
{{ ttlConveter(scope.row.ttl) }}
|
|
|
|
|
|
</template>
|
|
|
|
|
|
</el-table-column>
|
|
|
|
|
|
<el-table-column label="操作">
|
|
|
|
|
|
<template #default="scope">
|
2022-01-12 16:00:31 +08:00
|
|
|
|
<el-button @click="getValue(scope.row)" type="success" icon="search" plain size="small">查看</el-button>
|
|
|
|
|
|
<el-button @click="del(scope.row.key)" type="danger" icon="delete" plain size="small">删除</el-button>
|
2021-12-21 03:06:29 +00:00
|
|
|
|
</template>
|
|
|
|
|
|
</el-table-column>
|
|
|
|
|
|
</el-table>
|
|
|
|
|
|
</el-card>
|
|
|
|
|
|
|
2021-07-28 18:03:19 +08:00
|
|
|
|
<div style="text-align: center; margin-top: 10px"></div>
|
|
|
|
|
|
|
2022-08-05 21:41:21 +08:00
|
|
|
|
<hash-value
|
|
|
|
|
|
v-model:visible="hashValueDialog.visible"
|
|
|
|
|
|
:operationType="dataEdit.operationType"
|
|
|
|
|
|
:title="dataEdit.title"
|
|
|
|
|
|
:keyInfo="dataEdit.keyInfo"
|
|
|
|
|
|
:redisId="scanParam.id"
|
|
|
|
|
|
@cancel="onCancelDataEdit"
|
|
|
|
|
|
@valChange="searchKey"
|
|
|
|
|
|
/>
|
|
|
|
|
|
|
|
|
|
|
|
<string-value
|
|
|
|
|
|
v-model:visible="stringValueDialog.visible"
|
|
|
|
|
|
:operationType="dataEdit.operationType"
|
|
|
|
|
|
:title="dataEdit.title"
|
|
|
|
|
|
:keyInfo="dataEdit.keyInfo"
|
|
|
|
|
|
:redisId="scanParam.id"
|
|
|
|
|
|
@cancel="onCancelDataEdit"
|
|
|
|
|
|
@valChange="searchKey"
|
|
|
|
|
|
/>
|
2022-01-12 16:00:31 +08:00
|
|
|
|
|
2022-08-05 21:41:21 +08:00
|
|
|
|
<set-value
|
|
|
|
|
|
v-model:visible="setValueDialog.visible"
|
2022-01-12 16:00:31 +08:00
|
|
|
|
:title="dataEdit.title"
|
|
|
|
|
|
:keyInfo="dataEdit.keyInfo"
|
|
|
|
|
|
:redisId="scanParam.id"
|
|
|
|
|
|
:operationType="dataEdit.operationType"
|
|
|
|
|
|
@valChange="searchKey"
|
|
|
|
|
|
@cancel="onCancelDataEdit"
|
|
|
|
|
|
/>
|
2022-09-22 11:56:21 +08:00
|
|
|
|
|
|
|
|
|
|
<list-value
|
|
|
|
|
|
v-model:visible="listValueDialog.visible"
|
|
|
|
|
|
:title="dataEdit.title"
|
|
|
|
|
|
:keyInfo="dataEdit.keyInfo"
|
|
|
|
|
|
:redisId="scanParam.id"
|
|
|
|
|
|
:operationType="dataEdit.operationType"
|
|
|
|
|
|
@valChange="searchKey"
|
|
|
|
|
|
@cancel="onCancelDataEdit"
|
|
|
|
|
|
/>
|
2021-07-28 18:03:19 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
|
|
<script lang="ts">
|
|
|
|
|
|
import { redisApi } from './api';
|
|
|
|
|
|
import { toRefs, reactive, defineComponent } from 'vue';
|
|
|
|
|
|
import { ElMessage, ElMessageBox } from 'element-plus';
|
|
|
|
|
|
import ProjectEnvSelect from '../component/ProjectEnvSelect.vue';
|
2022-08-05 21:41:21 +08:00
|
|
|
|
import HashValue from './HashValue.vue';
|
|
|
|
|
|
import StringValue from './StringValue.vue';
|
|
|
|
|
|
import SetValue from './SetValue.vue';
|
2022-09-22 11:56:21 +08:00
|
|
|
|
import ListValue from './ListValue.vue';
|
2022-07-10 12:14:06 +08:00
|
|
|
|
import { isTrue, notBlank, notNull } from '@/common/assert';
|
2021-07-28 18:03:19 +08:00
|
|
|
|
|
|
|
|
|
|
export default defineComponent({
|
|
|
|
|
|
name: 'DataOperation',
|
|
|
|
|
|
components: {
|
2022-08-05 21:41:21 +08:00
|
|
|
|
StringValue,
|
|
|
|
|
|
HashValue,
|
|
|
|
|
|
SetValue,
|
2022-09-22 11:56:21 +08:00
|
|
|
|
ListValue,
|
2021-07-28 18:03:19 +08:00
|
|
|
|
ProjectEnvSelect,
|
|
|
|
|
|
},
|
|
|
|
|
|
setup() {
|
|
|
|
|
|
const state = reactive({
|
|
|
|
|
|
loading: false,
|
|
|
|
|
|
redisList: [],
|
|
|
|
|
|
query: {
|
|
|
|
|
|
envId: 0,
|
|
|
|
|
|
},
|
|
|
|
|
|
scanParam: {
|
|
|
|
|
|
id: null,
|
|
|
|
|
|
match: null,
|
|
|
|
|
|
count: 10,
|
2022-07-10 12:14:06 +08:00
|
|
|
|
cursor: {},
|
2021-07-28 18:03:19 +08:00
|
|
|
|
},
|
2022-01-12 16:00:31 +08:00
|
|
|
|
dataEdit: {
|
|
|
|
|
|
visible: false,
|
|
|
|
|
|
title: '新增数据',
|
|
|
|
|
|
operationType: 1,
|
|
|
|
|
|
keyInfo: {
|
|
|
|
|
|
type: 'string',
|
|
|
|
|
|
timed: -1,
|
|
|
|
|
|
key: '',
|
|
|
|
|
|
},
|
2022-08-05 21:41:21 +08:00
|
|
|
|
},
|
|
|
|
|
|
hashValueDialog: {
|
|
|
|
|
|
visible: false,
|
|
|
|
|
|
},
|
|
|
|
|
|
stringValueDialog: {
|
|
|
|
|
|
visible: false,
|
|
|
|
|
|
},
|
|
|
|
|
|
setValueDialog: {
|
|
|
|
|
|
visible: false,
|
2022-01-12 16:00:31 +08:00
|
|
|
|
},
|
2022-09-22 11:56:21 +08:00
|
|
|
|
listValueDialog: {
|
|
|
|
|
|
visible: false,
|
|
|
|
|
|
},
|
2021-07-28 18:03:19 +08:00
|
|
|
|
keys: [],
|
|
|
|
|
|
dbsize: 0,
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
const searchRedis = async () => {
|
|
|
|
|
|
notNull(state.query.envId, '请先选择项目环境');
|
|
|
|
|
|
const res = await redisApi.redisList.request(state.query);
|
|
|
|
|
|
state.redisList = res.list;
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const changeProjectEnv = (projectId: any, envId: any) => {
|
|
|
|
|
|
clearRedis();
|
|
|
|
|
|
if (envId != null) {
|
|
|
|
|
|
state.query.envId = envId;
|
|
|
|
|
|
searchRedis();
|
|
|
|
|
|
}
|
|
|
|
|
|
};
|
|
|
|
|
|
|
2022-07-10 12:14:06 +08:00
|
|
|
|
const changeRedis = (id: number) => {
|
|
|
|
|
|
resetScanParam(id);
|
2021-07-28 18:03:19 +08:00
|
|
|
|
state.keys = [];
|
|
|
|
|
|
state.dbsize = 0;
|
|
|
|
|
|
searchKey();
|
|
|
|
|
|
};
|
|
|
|
|
|
|
2022-07-10 12:14:06 +08:00
|
|
|
|
const scan = async () => {
|
2021-07-28 18:03:19 +08:00
|
|
|
|
isTrue(state.scanParam.id != null, '请先选择redis');
|
2022-07-10 12:14:06 +08:00
|
|
|
|
notBlank(state.scanParam.count, 'count不能为空');
|
2021-07-28 18:03:19 +08:00
|
|
|
|
|
2022-08-05 21:41:21 +08:00
|
|
|
|
const match = state.scanParam.match;
|
|
|
|
|
|
if (!match || match == '*') {
|
|
|
|
|
|
isTrue(state.scanParam.count <= 200, 'match为空或者*时, count不能超过200');
|
|
|
|
|
|
} else {
|
|
|
|
|
|
isTrue(state.scanParam.count <= 20000, 'count不能超过20000');
|
|
|
|
|
|
}
|
2021-07-28 18:03:19 +08:00
|
|
|
|
|
2022-08-05 21:41:21 +08:00
|
|
|
|
state.loading = true;
|
2022-07-10 12:14:06 +08:00
|
|
|
|
try {
|
|
|
|
|
|
const res = await redisApi.scan.request(state.scanParam);
|
2021-07-28 18:03:19 +08:00
|
|
|
|
state.keys = res.keys;
|
|
|
|
|
|
state.dbsize = res.dbSize;
|
|
|
|
|
|
state.scanParam.cursor = res.cursor;
|
2022-07-10 12:14:06 +08:00
|
|
|
|
} finally {
|
2021-07-28 18:03:19 +08:00
|
|
|
|
state.loading = false;
|
2022-07-10 12:14:06 +08:00
|
|
|
|
}
|
2021-07-28 18:03:19 +08:00
|
|
|
|
};
|
|
|
|
|
|
|
2022-07-10 12:14:06 +08:00
|
|
|
|
const searchKey = async () => {
|
|
|
|
|
|
state.scanParam.cursor = {};
|
|
|
|
|
|
await scan();
|
2021-07-28 18:03:19 +08:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const clearRedis = () => {
|
|
|
|
|
|
state.redisList = [];
|
|
|
|
|
|
state.scanParam.id = null;
|
|
|
|
|
|
resetScanParam();
|
|
|
|
|
|
state.keys = [];
|
|
|
|
|
|
state.dbsize = 0;
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const clear = () => {
|
|
|
|
|
|
resetScanParam();
|
|
|
|
|
|
if (state.scanParam.id) {
|
|
|
|
|
|
scan();
|
|
|
|
|
|
}
|
|
|
|
|
|
};
|
|
|
|
|
|
|
2022-07-10 12:14:06 +08:00
|
|
|
|
const resetScanParam = (id: number = 0) => {
|
2021-07-28 18:03:19 +08:00
|
|
|
|
state.scanParam.count = 10;
|
2022-07-10 12:14:06 +08:00
|
|
|
|
if (id != 0) {
|
|
|
|
|
|
const redis: any = state.redisList.find((x: any) => x.id == id);
|
|
|
|
|
|
// 集群模式count设小点,因为后端会从所有master节点scan一遍然后合并结果
|
|
|
|
|
|
if (redis && redis.mode == 'cluster') {
|
|
|
|
|
|
state.scanParam.count = 5;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
state.scanParam.match = null;
|
|
|
|
|
|
state.scanParam.cursor = {};
|
2021-07-28 18:03:19 +08:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const getValue = async (row: any) => {
|
2022-01-12 16:00:31 +08:00
|
|
|
|
const type = row.type;
|
|
|
|
|
|
|
2022-08-05 21:41:21 +08:00
|
|
|
|
state.dataEdit.keyInfo.type = type;
|
|
|
|
|
|
state.dataEdit.keyInfo.timed = row.ttl;
|
|
|
|
|
|
state.dataEdit.keyInfo.key = row.key;
|
|
|
|
|
|
state.dataEdit.operationType = 2;
|
|
|
|
|
|
state.dataEdit.title = '查看数据';
|
2021-07-28 18:03:19 +08:00
|
|
|
|
|
2022-01-12 16:00:31 +08:00
|
|
|
|
if (type == 'hash') {
|
2022-08-05 21:41:21 +08:00
|
|
|
|
state.hashValueDialog.visible = true;
|
|
|
|
|
|
} else if (type == 'string') {
|
|
|
|
|
|
state.stringValueDialog.visible = true;
|
|
|
|
|
|
} else if (type == 'set') {
|
|
|
|
|
|
state.setValueDialog.visible = true;
|
2022-09-22 11:56:21 +08:00
|
|
|
|
} else if (type == 'list') {
|
|
|
|
|
|
state.listValueDialog.visible = true;
|
2022-08-05 21:41:21 +08:00
|
|
|
|
} else {
|
|
|
|
|
|
ElMessage.warning('暂不支持该类型');
|
2022-01-12 16:00:31 +08:00
|
|
|
|
}
|
2022-08-05 21:41:21 +08:00
|
|
|
|
};
|
2021-07-28 18:03:19 +08:00
|
|
|
|
|
2022-08-05 21:41:21 +08:00
|
|
|
|
const onAddData = (type: string) => {
|
|
|
|
|
|
notNull(state.scanParam.id, '请先选择redis');
|
|
|
|
|
|
state.dataEdit.operationType = 1;
|
|
|
|
|
|
state.dataEdit.title = '新增数据';
|
2022-01-12 16:00:31 +08:00
|
|
|
|
state.dataEdit.keyInfo.type = type;
|
2022-08-05 21:41:21 +08:00
|
|
|
|
state.dataEdit.keyInfo.timed = -1;
|
|
|
|
|
|
if (type == 'hash') {
|
|
|
|
|
|
state.hashValueDialog.visible = true;
|
|
|
|
|
|
} else if (type == 'string') {
|
|
|
|
|
|
state.stringValueDialog.visible = true;
|
|
|
|
|
|
} else if (type == 'set') {
|
|
|
|
|
|
state.setValueDialog.visible = true;
|
2022-09-22 11:56:21 +08:00
|
|
|
|
} else if (type == 'list') {
|
|
|
|
|
|
state.listValueDialog.visible = true;
|
2022-08-05 21:41:21 +08:00
|
|
|
|
} else {
|
|
|
|
|
|
ElMessage.warning('暂不支持该类型');
|
|
|
|
|
|
}
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const onCancelDataEdit = () => {
|
|
|
|
|
|
state.dataEdit.keyInfo = {} as any;
|
2022-01-12 16:00:31 +08:00
|
|
|
|
};
|
2021-07-28 18:03:19 +08:00
|
|
|
|
|
|
|
|
|
|
const del = (key: string) => {
|
2022-07-10 12:14:06 +08:00
|
|
|
|
ElMessageBox.confirm(`确定删除[ ${key} ] 该key?`, '提示', {
|
2021-07-28 18:03:19 +08:00
|
|
|
|
confirmButtonText: '确定',
|
|
|
|
|
|
cancelButtonText: '取消',
|
|
|
|
|
|
type: 'warning',
|
|
|
|
|
|
})
|
|
|
|
|
|
.then(() => {
|
|
|
|
|
|
redisApi.delKey
|
|
|
|
|
|
.request({
|
|
|
|
|
|
key,
|
2022-07-10 12:14:06 +08:00
|
|
|
|
id: state.scanParam.id,
|
2021-07-28 18:03:19 +08:00
|
|
|
|
})
|
2021-09-08 17:55:57 +08:00
|
|
|
|
.then(() => {
|
2021-07-28 18:03:19 +08:00
|
|
|
|
ElMessage.success('删除成功!');
|
2022-07-10 12:14:06 +08:00
|
|
|
|
searchKey();
|
2021-07-28 18:03:19 +08:00
|
|
|
|
});
|
|
|
|
|
|
})
|
2021-09-08 17:55:57 +08:00
|
|
|
|
.catch(() => {});
|
2021-07-28 18:03:19 +08:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const ttlConveter = (ttl: any) => {
|
2022-07-10 12:14:06 +08:00
|
|
|
|
if (ttl == -1 || ttl == 0) {
|
2021-07-28 18:03:19 +08:00
|
|
|
|
return '永久';
|
|
|
|
|
|
}
|
|
|
|
|
|
if (!ttl) {
|
|
|
|
|
|
ttl = 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
let second = parseInt(ttl); // 秒
|
|
|
|
|
|
let min = 0; // 分
|
|
|
|
|
|
let hour = 0; // 小时
|
|
|
|
|
|
let day = 0;
|
|
|
|
|
|
if (second > 60) {
|
|
|
|
|
|
min = parseInt(second / 60 + '');
|
|
|
|
|
|
second = second % 60;
|
|
|
|
|
|
if (min > 60) {
|
|
|
|
|
|
hour = parseInt(min / 60 + '');
|
|
|
|
|
|
min = min % 60;
|
|
|
|
|
|
if (hour > 24) {
|
|
|
|
|
|
day = parseInt(hour / 24 + '');
|
|
|
|
|
|
hour = hour % 24;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
let result = '' + second + 's';
|
|
|
|
|
|
if (min > 0) {
|
|
|
|
|
|
result = '' + min + 'm:' + result;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (hour > 0) {
|
|
|
|
|
|
result = '' + hour + 'h:' + result;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (day > 0) {
|
|
|
|
|
|
result = '' + day + 'd:' + result;
|
|
|
|
|
|
}
|
|
|
|
|
|
return result;
|
|
|
|
|
|
};
|
|
|
|
|
|
|
2022-03-24 17:50:44 +08:00
|
|
|
|
const getTypeColor = (type: string) => {
|
|
|
|
|
|
if (type == 'string') {
|
|
|
|
|
|
return '#E4F5EB';
|
|
|
|
|
|
}
|
|
|
|
|
|
if (type == 'hash') {
|
|
|
|
|
|
return '#F9E2AE';
|
|
|
|
|
|
}
|
|
|
|
|
|
if (type == 'set') {
|
|
|
|
|
|
return '#A8DEE0';
|
|
|
|
|
|
}
|
|
|
|
|
|
};
|
|
|
|
|
|
|
2021-07-28 18:03:19 +08:00
|
|
|
|
return {
|
|
|
|
|
|
...toRefs(state),
|
|
|
|
|
|
changeProjectEnv,
|
|
|
|
|
|
changeRedis,
|
|
|
|
|
|
clearRedis,
|
|
|
|
|
|
searchKey,
|
|
|
|
|
|
scan,
|
|
|
|
|
|
clear,
|
|
|
|
|
|
getValue,
|
|
|
|
|
|
del,
|
|
|
|
|
|
ttlConveter,
|
2022-03-24 17:50:44 +08:00
|
|
|
|
getTypeColor,
|
2022-01-12 16:00:31 +08:00
|
|
|
|
onAddData,
|
|
|
|
|
|
onCancelDataEdit,
|
2021-07-28 18:03:19 +08:00
|
|
|
|
};
|
|
|
|
|
|
},
|
|
|
|
|
|
});
|
|
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
|
|
<style>
|
|
|
|
|
|
</style>
|