查询消息详情信息
This commit is contained in:
@@ -20,4 +20,8 @@ public class QueryMessage {
|
|||||||
private long endTime;
|
private long endTime;
|
||||||
|
|
||||||
private long offset;
|
private long offset;
|
||||||
|
|
||||||
|
private String keyDeserializer;
|
||||||
|
|
||||||
|
private String valueDeserializer;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -23,6 +23,10 @@ public class QueryMessageDTO {
|
|||||||
|
|
||||||
private Long offset;
|
private Long offset;
|
||||||
|
|
||||||
|
private String keyDeserializer;
|
||||||
|
|
||||||
|
private String valueDeserializer;
|
||||||
|
|
||||||
public QueryMessage toQueryMessage() {
|
public QueryMessage toQueryMessage() {
|
||||||
QueryMessage queryMessage = new QueryMessage();
|
QueryMessage queryMessage = new QueryMessage();
|
||||||
queryMessage.setTopic(topic);
|
queryMessage.setTopic(topic);
|
||||||
@@ -38,6 +42,9 @@ public class QueryMessageDTO {
|
|||||||
queryMessage.setOffset(offset);
|
queryMessage.setOffset(offset);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
queryMessage.setKeyDeserializer(keyDeserializer);
|
||||||
|
queryMessage.setValueDeserializer(valueDeserializer);
|
||||||
|
|
||||||
return queryMessage;
|
return queryMessage;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,38 @@
|
|||||||
|
package com.xuxd.kafka.console.beans.vo;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
import lombok.Data;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* kafka-console-ui.
|
||||||
|
*
|
||||||
|
* @author xuxd
|
||||||
|
* @date 2021-12-12 12:45:23
|
||||||
|
**/
|
||||||
|
@Data
|
||||||
|
public class MessageDetailVO {
|
||||||
|
|
||||||
|
private String topic;
|
||||||
|
|
||||||
|
private int partition;
|
||||||
|
|
||||||
|
private long offset;
|
||||||
|
|
||||||
|
private long timestamp;
|
||||||
|
|
||||||
|
private String timestampType;
|
||||||
|
|
||||||
|
private List<HeaderVO> headers = new ArrayList<>();
|
||||||
|
|
||||||
|
private Object key;
|
||||||
|
|
||||||
|
private Object value;
|
||||||
|
|
||||||
|
@Data
|
||||||
|
public static class HeaderVO {
|
||||||
|
String key;
|
||||||
|
|
||||||
|
String value;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -3,6 +3,7 @@ package com.xuxd.kafka.console.controller;
|
|||||||
import com.xuxd.kafka.console.beans.dto.QueryMessageDTO;
|
import com.xuxd.kafka.console.beans.dto.QueryMessageDTO;
|
||||||
import com.xuxd.kafka.console.service.MessageService;
|
import com.xuxd.kafka.console.service.MessageService;
|
||||||
import org.springframework.beans.factory.annotation.Autowired;
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
import org.springframework.web.bind.annotation.GetMapping;
|
||||||
import org.springframework.web.bind.annotation.PostMapping;
|
import org.springframework.web.bind.annotation.PostMapping;
|
||||||
import org.springframework.web.bind.annotation.RequestBody;
|
import org.springframework.web.bind.annotation.RequestBody;
|
||||||
import org.springframework.web.bind.annotation.RequestMapping;
|
import org.springframework.web.bind.annotation.RequestMapping;
|
||||||
@@ -30,4 +31,14 @@ public class MessageController {
|
|||||||
public Object searchByOffset(@RequestBody QueryMessageDTO dto) {
|
public Object searchByOffset(@RequestBody QueryMessageDTO dto) {
|
||||||
return messageService.searchByOffset(dto.toQueryMessage());
|
return messageService.searchByOffset(dto.toQueryMessage());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@PostMapping("/search/detail")
|
||||||
|
public Object searchDetail(@RequestBody QueryMessageDTO dto) {
|
||||||
|
return messageService.searchDetail(dto.toQueryMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@GetMapping("/deserializer/list")
|
||||||
|
public Object deserializerList() {
|
||||||
|
return messageService.deserializerList();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -14,4 +14,8 @@ public interface MessageService {
|
|||||||
ResponseData searchByTime(QueryMessage queryMessage);
|
ResponseData searchByTime(QueryMessage queryMessage);
|
||||||
|
|
||||||
ResponseData searchByOffset(QueryMessage queryMessage);
|
ResponseData searchByOffset(QueryMessage queryMessage);
|
||||||
|
|
||||||
|
ResponseData searchDetail(QueryMessage queryMessage);
|
||||||
|
|
||||||
|
ResponseData deserializerList();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ package com.xuxd.kafka.console.service.impl;
|
|||||||
import com.xuxd.kafka.console.beans.QueryMessage;
|
import com.xuxd.kafka.console.beans.QueryMessage;
|
||||||
import com.xuxd.kafka.console.beans.ResponseData;
|
import com.xuxd.kafka.console.beans.ResponseData;
|
||||||
import com.xuxd.kafka.console.beans.vo.ConsumerRecordVO;
|
import com.xuxd.kafka.console.beans.vo.ConsumerRecordVO;
|
||||||
|
import com.xuxd.kafka.console.beans.vo.MessageDetailVO;
|
||||||
import com.xuxd.kafka.console.service.MessageService;
|
import com.xuxd.kafka.console.service.MessageService;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
@@ -14,9 +15,15 @@ import java.util.stream.Collectors;
|
|||||||
import kafka.console.MessageConsole;
|
import kafka.console.MessageConsole;
|
||||||
import kafka.console.TopicConsole;
|
import kafka.console.TopicConsole;
|
||||||
import org.apache.commons.collections.CollectionUtils;
|
import org.apache.commons.collections.CollectionUtils;
|
||||||
|
import org.apache.commons.lang3.StringUtils;
|
||||||
import org.apache.kafka.clients.admin.TopicDescription;
|
import org.apache.kafka.clients.admin.TopicDescription;
|
||||||
import org.apache.kafka.clients.consumer.ConsumerRecord;
|
import org.apache.kafka.clients.consumer.ConsumerRecord;
|
||||||
import org.apache.kafka.common.TopicPartition;
|
import org.apache.kafka.common.TopicPartition;
|
||||||
|
import org.apache.kafka.common.serialization.Deserializer;
|
||||||
|
import org.apache.kafka.common.serialization.DoubleDeserializer;
|
||||||
|
import org.apache.kafka.common.serialization.FloatDeserializer;
|
||||||
|
import org.apache.kafka.common.serialization.IntegerDeserializer;
|
||||||
|
import org.apache.kafka.common.serialization.StringDeserializer;
|
||||||
import org.springframework.beans.factory.annotation.Autowired;
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
import org.springframework.stereotype.Service;
|
import org.springframework.stereotype.Service;
|
||||||
|
|
||||||
@@ -35,6 +42,17 @@ public class MessageServiceImpl implements MessageService {
|
|||||||
@Autowired
|
@Autowired
|
||||||
private TopicConsole topicConsole;
|
private TopicConsole topicConsole;
|
||||||
|
|
||||||
|
private Map<String, Deserializer> deserializerDict = new HashMap<>();
|
||||||
|
|
||||||
|
{
|
||||||
|
deserializerDict.put("Integer", new IntegerDeserializer());
|
||||||
|
deserializerDict.put("String", new StringDeserializer());
|
||||||
|
deserializerDict.put("Float", new FloatDeserializer());
|
||||||
|
deserializerDict.put("Double", new DoubleDeserializer());
|
||||||
|
}
|
||||||
|
|
||||||
|
public static String defaultDeserializer = "String";
|
||||||
|
|
||||||
@Override public ResponseData searchByTime(QueryMessage queryMessage) {
|
@Override public ResponseData searchByTime(QueryMessage queryMessage) {
|
||||||
int maxNums = 10000;
|
int maxNums = 10000;
|
||||||
|
|
||||||
@@ -50,14 +68,68 @@ public class MessageServiceImpl implements MessageService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override public ResponseData searchByOffset(QueryMessage queryMessage) {
|
@Override public ResponseData searchByOffset(QueryMessage queryMessage) {
|
||||||
|
Map<TopicPartition, ConsumerRecord<byte[], byte[]>> recordMap = searchRecordByOffset(queryMessage);
|
||||||
|
|
||||||
|
return ResponseData.create().data(recordMap.values().stream().map(ConsumerRecordVO::fromConsumerRecord).collect(Collectors.toList())).success();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override public ResponseData searchDetail(QueryMessage queryMessage) {
|
||||||
|
if (queryMessage.getPartition() == -1) {
|
||||||
|
throw new IllegalArgumentException();
|
||||||
|
}
|
||||||
|
if (StringUtils.isBlank(queryMessage.getKeyDeserializer())) {
|
||||||
|
queryMessage.setKeyDeserializer(defaultDeserializer);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (StringUtils.isBlank(queryMessage.getValueDeserializer())) {
|
||||||
|
queryMessage.setValueDeserializer(defaultDeserializer);
|
||||||
|
}
|
||||||
|
|
||||||
|
Map<TopicPartition, ConsumerRecord<byte[], byte[]>> recordMap = searchRecordByOffset(queryMessage);
|
||||||
|
ConsumerRecord<byte[], byte[]> record = recordMap.get(new TopicPartition(queryMessage.getTopic(), queryMessage.getPartition()));
|
||||||
|
if (record != null) {
|
||||||
|
MessageDetailVO vo = new MessageDetailVO();
|
||||||
|
vo.setTopic(record.topic());
|
||||||
|
vo.setPartition(record.partition());
|
||||||
|
vo.setOffset(record.offset());
|
||||||
|
vo.setTimestamp(record.timestamp());
|
||||||
|
vo.setTimestampType(record.timestampType().name());
|
||||||
|
try {
|
||||||
|
vo.setKey(deserializerDict.get(queryMessage.getKeyDeserializer()).deserialize(queryMessage.getTopic(), record.key()));
|
||||||
|
} catch (Exception e) {
|
||||||
|
vo.setKey("KeyDeserializer Error: " + e.getMessage());
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
vo.setValue(deserializerDict.get(queryMessage.getValueDeserializer()).deserialize(queryMessage.getTopic(), record.value()));
|
||||||
|
} catch (Exception e) {
|
||||||
|
vo.setValue("ValueDeserializer Error: " + e.getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
record.headers().forEach(header -> {
|
||||||
|
MessageDetailVO.HeaderVO headerVO = new MessageDetailVO.HeaderVO();
|
||||||
|
headerVO.setKey(header.key());
|
||||||
|
headerVO.setValue(new String(header.value()));
|
||||||
|
vo.getHeaders().add(headerVO);
|
||||||
|
});
|
||||||
|
|
||||||
|
return ResponseData.create().data(vo).success();
|
||||||
|
}
|
||||||
|
return ResponseData.create().failed("Not found message detail.");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override public ResponseData deserializerList() {
|
||||||
|
return ResponseData.create().data(deserializerDict.keySet()).success();
|
||||||
|
}
|
||||||
|
|
||||||
|
private Map<TopicPartition, ConsumerRecord<byte[], byte[]>> searchRecordByOffset(QueryMessage queryMessage) {
|
||||||
Set<TopicPartition> partitions = getPartitions(queryMessage);
|
Set<TopicPartition> partitions = getPartitions(queryMessage);
|
||||||
|
|
||||||
Map<TopicPartition, Object> offsetTable = new HashMap<>();
|
Map<TopicPartition, Object> offsetTable = new HashMap<>();
|
||||||
partitions.forEach(tp -> {
|
partitions.forEach(tp -> {
|
||||||
offsetTable.put(tp, queryMessage.getOffset());
|
offsetTable.put(tp, queryMessage.getOffset());
|
||||||
});
|
});
|
||||||
Map<TopicPartition, ConsumerRecord<byte[], byte[]>> recordMap = messageConsole.searchBy(offsetTable);
|
Map<TopicPartition, ConsumerRecord<byte[], byte[]>> recordMap = messageConsole.searchBy(offsetTable);
|
||||||
|
return recordMap;
|
||||||
return ResponseData.create().data(recordMap.values().stream().map(ConsumerRecordVO::fromConsumerRecord).collect(Collectors.toList())).success();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private Set<TopicPartition> getPartitions(QueryMessage queryMessage) {
|
private Set<TopicPartition> getPartitions(QueryMessage queryMessage) {
|
||||||
|
|||||||
@@ -232,4 +232,12 @@ export const KafkaMessageApi = {
|
|||||||
url: "/message/search/offset",
|
url: "/message/search/offset",
|
||||||
method: "post",
|
method: "post",
|
||||||
},
|
},
|
||||||
|
searchDetail: {
|
||||||
|
url: "/message/search/detail",
|
||||||
|
method: "post",
|
||||||
|
},
|
||||||
|
deserializerList: {
|
||||||
|
url: "/message/deserializer/list",
|
||||||
|
method: "get",
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|||||||
204
ui/src/views/message/MessageDetail.vue
Normal file
204
ui/src/views/message/MessageDetail.vue
Normal file
@@ -0,0 +1,204 @@
|
|||||||
|
<template>
|
||||||
|
<a-modal
|
||||||
|
title="消息详情"
|
||||||
|
:visible="show"
|
||||||
|
:width="800"
|
||||||
|
:mask="false"
|
||||||
|
:destroyOnClose="true"
|
||||||
|
:footer="null"
|
||||||
|
:maskClosable="false"
|
||||||
|
@cancel="handleCancel"
|
||||||
|
>
|
||||||
|
<div>
|
||||||
|
<a-spin :spinning="loading">
|
||||||
|
<div>
|
||||||
|
<h4>消息信息</h4>
|
||||||
|
<hr />
|
||||||
|
<div class="message-detail" id="message-detail">
|
||||||
|
<p>
|
||||||
|
<label class="title">Topic: </label>
|
||||||
|
<span class="m-info">{{ data.topic }}</span>
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
<label class="title">分区: </label>
|
||||||
|
<span class="m-info">{{ data.partition }}</span>
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
<label class="title">偏移: </label>
|
||||||
|
<span class="m-info">{{ data.offset }}</span>
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
<label class="title">消息头: </label>
|
||||||
|
<span class="m-info">{{ data.headers }}</span>
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
<label class="title">时间类型: </label>
|
||||||
|
<span class="m-info"
|
||||||
|
>{{
|
||||||
|
data.timestampType
|
||||||
|
}}(表示下面的时间是哪种类型:消息创建、写入日志亦或其它)</span
|
||||||
|
>
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
<label class="title">时间: </label>
|
||||||
|
<span class="m-info">{{ formatTime(data.timestamp) }}</span>
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
<label class="title">Key反序列化: </label>
|
||||||
|
<a-select
|
||||||
|
style="width: 120px"
|
||||||
|
v-model="keyDeserializer"
|
||||||
|
@change="keyDeserializerChange"
|
||||||
|
>
|
||||||
|
<a-select-option
|
||||||
|
v-for="v in deserializerList"
|
||||||
|
:key="v"
|
||||||
|
:value="v"
|
||||||
|
>
|
||||||
|
{{ v }}
|
||||||
|
</a-select-option>
|
||||||
|
</a-select>
|
||||||
|
<span>选一个合适反序列化器,要不可能乱码了</span>
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
<label class="title">Key: </label>
|
||||||
|
<span class="m-info">{{ data.key }}</span>
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
<label class="title">消息体反序列化: </label>
|
||||||
|
<a-select
|
||||||
|
v-model="valueDeserializer"
|
||||||
|
style="width: 120px"
|
||||||
|
@change="valueDeserializerChange"
|
||||||
|
>
|
||||||
|
<a-select-option
|
||||||
|
v-for="v in deserializerList"
|
||||||
|
:key="v"
|
||||||
|
:value="v"
|
||||||
|
>
|
||||||
|
{{ v }}
|
||||||
|
</a-select-option>
|
||||||
|
</a-select>
|
||||||
|
<span>选一个合适反序列化器,要不可能乱码了</span>
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
<label class="title">消息体: </label>
|
||||||
|
<a-textarea
|
||||||
|
type="textarea"
|
||||||
|
:value="data.value"
|
||||||
|
:rows="5"
|
||||||
|
:read-only="true"
|
||||||
|
></a-textarea>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</a-spin>
|
||||||
|
</div>
|
||||||
|
</a-modal>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import request from "@/utils/request";
|
||||||
|
import { KafkaMessageApi } from "@/utils/api";
|
||||||
|
import notification from "ant-design-vue/lib/notification";
|
||||||
|
import moment from "moment";
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: "MessageDetail",
|
||||||
|
props: {
|
||||||
|
record: {},
|
||||||
|
visible: {
|
||||||
|
type: Boolean,
|
||||||
|
default: false,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
show: this.visible,
|
||||||
|
data: {},
|
||||||
|
loading: false,
|
||||||
|
deserializerList: [],
|
||||||
|
keyDeserializer: "String",
|
||||||
|
valueDeserializer: "String",
|
||||||
|
};
|
||||||
|
},
|
||||||
|
watch: {
|
||||||
|
visible(v) {
|
||||||
|
this.show = v;
|
||||||
|
if (this.show) {
|
||||||
|
this.getMessageDetail();
|
||||||
|
this.getDeserializerList();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
getMessageDetail() {
|
||||||
|
this.loading = false;
|
||||||
|
const params = Object.assign({}, this.record, {
|
||||||
|
keyDeserializer: this.keyDeserializer,
|
||||||
|
valueDeserializer: this.valueDeserializer,
|
||||||
|
});
|
||||||
|
request({
|
||||||
|
url: KafkaMessageApi.searchDetail.url,
|
||||||
|
method: KafkaMessageApi.searchDetail.method,
|
||||||
|
data: params,
|
||||||
|
}).then((res) => {
|
||||||
|
this.loading = false;
|
||||||
|
if (res.code != 0) {
|
||||||
|
notification.error({
|
||||||
|
message: "error",
|
||||||
|
description: res.msg,
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
this.data = res.data;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
getDeserializerList() {
|
||||||
|
request({
|
||||||
|
url: KafkaMessageApi.deserializerList.url,
|
||||||
|
method: KafkaMessageApi.deserializerList.method,
|
||||||
|
}).then((res) => {
|
||||||
|
if (res.code != 0) {
|
||||||
|
notification.error({
|
||||||
|
message: "error",
|
||||||
|
description: res.msg,
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
this.deserializerList = res.data;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
handleCancel() {
|
||||||
|
this.data = {};
|
||||||
|
this.$emit("closeDetailDialog", { refresh: false });
|
||||||
|
},
|
||||||
|
formatTime(time) {
|
||||||
|
return moment(time).format("YYYY-MM-DD HH:mm:ss:SSS");
|
||||||
|
},
|
||||||
|
keyDeserializerChange() {
|
||||||
|
this.getMessageDetail();
|
||||||
|
},
|
||||||
|
valueDeserializerChange() {
|
||||||
|
this.getMessageDetail();
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
.m-info {
|
||||||
|
/*text-decoration: underline;*/
|
||||||
|
}
|
||||||
|
.title {
|
||||||
|
width: 15%;
|
||||||
|
display: inline-block;
|
||||||
|
text-align: right;
|
||||||
|
margin-right: 2%;
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
.ant-spin-container #message-detail textarea {
|
||||||
|
max-width: 80% !important;
|
||||||
|
vertical-align: top !important;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -4,22 +4,36 @@
|
|||||||
:columns="columns"
|
:columns="columns"
|
||||||
:data-source="data"
|
:data-source="data"
|
||||||
bordered
|
bordered
|
||||||
row-key="(record,index)=>{return index}"
|
:row-key="
|
||||||
|
(record, index) => {
|
||||||
|
return index;
|
||||||
|
}
|
||||||
|
"
|
||||||
>
|
>
|
||||||
<div slot="operation" slot-scope="{}">
|
<div slot="operation" slot-scope="record">
|
||||||
<a-button size="small" href="javascript:;" class="operation-btn"
|
<a-button
|
||||||
|
size="small"
|
||||||
|
href="javascript:;"
|
||||||
|
class="operation-btn"
|
||||||
|
@click="openDetailDialog(record)"
|
||||||
>消息详情
|
>消息详情
|
||||||
</a-button>
|
</a-button>
|
||||||
</div>
|
</div>
|
||||||
</a-table>
|
</a-table>
|
||||||
|
<MessageDetail
|
||||||
|
:visible="showDetailDialog"
|
||||||
|
:record="record"
|
||||||
|
@closeDetailDialog="closeDetailDialog"
|
||||||
|
></MessageDetail>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import moment from "moment";
|
import moment from "moment";
|
||||||
|
import MessageDetail from "@/views/message/MessageDetail";
|
||||||
export default {
|
export default {
|
||||||
name: "MessageList",
|
name: "MessageList",
|
||||||
components: {},
|
components: { MessageDetail },
|
||||||
props: {
|
props: {
|
||||||
data: {
|
data: {
|
||||||
type: Array,
|
type: Array,
|
||||||
@@ -28,8 +42,19 @@ export default {
|
|||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
columns: columns,
|
columns: columns,
|
||||||
|
showDetailDialog: false,
|
||||||
|
record: {},
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
methods: {
|
||||||
|
openDetailDialog(record) {
|
||||||
|
this.record = record;
|
||||||
|
this.showDetailDialog = true;
|
||||||
|
},
|
||||||
|
closeDetailDialog() {
|
||||||
|
this.showDetailDialog = false;
|
||||||
|
},
|
||||||
|
},
|
||||||
};
|
};
|
||||||
const columns = [
|
const columns = [
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -147,7 +147,7 @@ export default {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
const defaultData = { realNum: 0, maxNum: 0 };
|
const defaultData = [];
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
|
|||||||
Reference in New Issue
Block a user