123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584 |
- <template>
- <div class="analysis-container">
- <el-row :gutter="20">
- <!-- 左侧主操作区 -->
- <el-col :span="16">
- <el-card class="upload-section">
- <div class="input-method-select" v-if="inputMethod != 'done'">
- <el-radio-group v-model="inputMethod">
- <el-radio-button value="upload">文件上传</el-radio-button>
- <el-radio-button value="input">在线输入</el-radio-button>
- </el-radio-group>
- </div>
- <div v-if="inputMethod === 'upload'" class="upload-area">
- <!-- 节点文件上传 -->
- <el-upload class="file-upload" :on-change="handleNodeFileChange" :auto-upload="false"
- :show-file-list="false">
- <el-button type="primary" style="width: 120px; margin-right: 20px" plain>
- <el-icon>
- <Upload />
- </el-icon>
- 选择节点文件
- </el-button>
- <div class="file-info">
- {{ nodeFile ? nodeFile.name : '未选择文件' }}
- <span v-if="nodeFile">({{ formatSize(nodeFile.size) }})</span>
- </div>
- </el-upload>
- <!-- 边文件上传 -->
- <el-upload class="file-upload" :on-change="handleEdgeFileChange" :auto-upload="false"
- :show-file-list="false">
- <el-button type="primary" style="width: 120px; margin-right: 20px" plain>
- <el-icon>
- <Upload />
- </el-icon>
- 选择边文件
- </el-button>
- <div class="file-info">
- {{ edgeFile ? edgeFile.name : '未选择文件' }}
- <span v-if="edgeFile">({{ formatSize(edgeFile.size) }})</span>
- </div>
- </el-upload>
- <!-- 上传按钮和进度 -->
- <el-button type="success" :disabled="!canUpload" @click="handleUpload" class="upload-button">
- 开始上传验证
- </el-button>
- </div>
- <!-- 在线输入节点与边 -->
- <div v-if="inputMethod === 'input'" class="upload-area">
- <div class="online-input">
- <el-row :gutter="20">
- <!-- 节点输入列 -->
- <el-col :span="12">
- <div class="input-section">
- <h4>输入节点</h4>
- <div v-for="(node, index) in nodes" :key="index" class="input-row">
- <el-row :gutter="10" align="middle">
- <el-col :span="4">
- <div class="node-id">ID:{{ node.id }}</div>
- </el-col>
- <el-col :span="6">
- <el-select v-model="node.type" placeholder="选择类型" @change="handleNodeTypeChange(index)">
- <el-option label="S" value="S" />
- <el-option label="D" value="D" />
- <el-option label="I" value="I" />
- </el-select>
- </el-col>
- <el-col :span="10">
- <el-input v-model="node.name" placeholder="输入节点名称" />
- </el-col>
- <el-col :span="4">
- <el-button v-if="index < nodes.length - 1" @click="deleteNode(index)" type="danger" plain
- size="small">
- 删除
- </el-button>
- </el-col>
- </el-row>
- </div>
- </div>
- </el-col>
- <!-- 边输入列 -->
- <el-col :span="12">
- <div class="input-section">
- <h4>输入边</h4>
- <div v-for="(edge, index) in edges" :key="index" class="input-row">
- <el-row :gutter="10" align="middle">
- <el-col :span="4">
- <div class="node-id">ID:{{ index + 1 }}</div>
- </el-col>
- <!-- 起始节点 -->
- <el-col :span="8">
- <el-select v-model="edge.from" placeholder="起始节点" :disabled="nodeOptions.length === 0"
- @change="handleEdgeChange(index)">
- <el-option v-for="node in nodeOptions" :key="node.id"
- :label="`ID:${node.id}-${node.name || '未命名'}`" :value="node.id" />
- </el-select>
- </el-col>
- <!-- 终止节点 -->
- <el-col :span="8">
- <el-select v-model="edge.to" placeholder="终止节点" :disabled="nodeOptions.length === 0"
- @change="handleEdgeChange(index)">
- <el-option v-for="node in nodeOptions" :key="node.id"
- :label="`ID:${node.id}-${node.name || '未命名'}`" :value="node.id" />
- </el-select>
- </el-col>
- <!-- 删除按钮 -->
- <el-col :span="4">
- <el-button v-if="index < edges.length - 1" @click="deleteEdge(index)" type="danger" plain
- size="small">
- 删除
- </el-button>
- </el-col>
- </el-row>
- </div>
- </div>
- </el-col>
- </el-row>
- </div>
- <el-button type="primary" @click="handleValidation" class="validate-button">
- 开始输入验证
- </el-button>
- </div>
- <div v-else >
- <router-view></router-view>
- </div>
- </el-card>
- </el-col>
- <!-- 右侧说明 -->
- <el-col v-if="inputMethod === 'input' || inputMethod === 'upload'" :span="8">
- <el-card class="instruction-section">
- <h3>文件格式说明</h3>
- <div class="instruction-content">
- <div class="file-format">
- <el-tag type="success" class="format-tag">节点文件格式</el-tag>
- <el-divider />
- <div class="format-example">
- <P>文件应使用CSV格式</P>
- <p>每一行按照:“节点编号,节点类型,节点名称” 顺序放置数据</p>
- <p class="example-text">示例:</p>
- <pre>
- 1,S,侦察节点
- 2,D,决策节点
- 3,I,打击节点</pre>
- </div>
- </div>
- <div class="file-format">
- <el-tag type="warning" class="format-tag">边文件格式</el-tag>
- <el-divider />
- <div class="format-example">
- <P>文件应使用CSV格式</P>
- <p>每一行按照:“起始节点,终止节点” 顺序放置数据</p>
- <p class="example-text">示例:</p>
- <pre>
- 1,2
- 2,3
- 1,3</pre>
- </div>
- </div>
- </div>
- </el-card>
- </el-col>
- </el-row>
- </div>
- </template>
- <script setup>
- import { ref, computed, onMounted, inject, watch } from 'vue'
- import { useRouter, useRoute } from 'vue-router'
- import { ElMessage, ElMessageBox } from 'element-plus'
- import { Upload, ArrowDown, Plus } from '@element-plus/icons-vue'
- import { getData, postData, deleteData, postFile } from '@/api/axios.js'
- // Store数据
- const useAnalyzeInfo = inject('analyzeInfo')
- // 响应式数据
- const inputMethod = ref('upload')
- // 上传的节点和边文件
- const nodeFile = ref(null)
- const edgeFile = ref(null)
- // 在线输入的节点和边
- const nodes = ref([{ id: 1, type: '', name: '' }])
- const edges = ref([{ from: '', to: '' }])
- const router = useRouter()
- // 计算属性
- // 是否允许上传文件
- const canUpload = computed(() => {
- return nodeFile.value && edgeFile.value
- })
- // 在线输入边时节点的可选项
- const nodeOptions = computed(() => {
- return nodes.value
- .filter(node => node.type !== '')
- .map(node => ({
- id: node.id,
- name: node.name || '未命名',
- type: node.type
- }))
- })
- // 方法
- // 在线输入自动添加节点
- const handleNodeTypeChange = (index) => {
- // 当最后一个节点选择类型后自动添加新行
- if (index === nodes.value.length - 1) {
- nodes.value.push({ id: nodes.value.length + 1, type: '', name: '' })
- }
- }
- // 删除节点
- const deleteNode = (index) => {
- // 保存原始ID列表
- const originalIds = nodes.value.map(n => n.id)
- const deletedId = originalIds[index]
- // 删除关联边
- edges.value = edges.value.filter(edge =>
- edge.from !== deletedId && edge.to !== deletedId
- )
- // 删除节点并创建新数组
- const newNodes = [...nodes.value]
- newNodes.splice(index, 1)
- // 创建ID映射表(旧ID -> 新ID)
- const idMap = new Map()
- newNodes.forEach((node, i) => {
- const newId = i + 1
- idMap.set(node.id, newId) // 记录原始ID到新ID的映射
- node.id = newId // 更新节点ID
- })
- // 更新节点数组
- nodes.value = newNodes
- // 更新边数据中的ID引用
- edges.value = edges.value.map(edge => ({
- from: idMap.get(edge.from),
- to: idMap.get(edge.to)
- }))
- // 确保至少保留一个空行
- if (nodes.value.length === 0) {
- nodes.value.push({ id: 1, type: '', name: '' })
- }
- }
- // 获取有效节点ID列表
- const validNodeIds = computed(() => {
- return nodes.value
- .filter(node => node.type !== '')
- .map(node => node.id)
- })
- // 在线输入自动添加边
- const handleEdgeChange = (index) => {
- const currentEdge = edges.value[index]
- // 当最后一个边填写完整后自动添加新行
- if (index === edges.value.length - 1 && currentEdge.from && currentEdge.to) {
- edges.value.push({ from: '', to: '' })
- }
- }
- // 删除边
- const deleteEdge = (index) => {
- edges.value.splice(index, 1)
- }
- // 验证在线输入
- const handleValidation = async () => {
- const errors = []
- const edgeSet = new Set()
- // 检查边数据
- // 因为始终有空行,所有没有输入的时候长度为1
- if (edges.value.length == 1) {
- errors.push(`没有输入边`)
- }
- if (nodes.value.length == 1) {
- errors.push(`没有输入节点`)
- }
- edges.value.slice(0, -1).forEach((edge, index) => {
- // 检查节点是否存在
- if (!validNodeIds.value.includes(edge.from)) {
- errors.push(`边 ${index + 1}: 起始节点 ${edge.from} 不存在`)
- }
- if (!validNodeIds.value.includes(edge.to)) {
- errors.push(`边 ${index + 1}: 终止节点 ${edge.to} 不存在`)
- }
- // 自环边检查
- if (edge.from === edge.to) {
- errors.push(`边 ${index + 1}: 不允许连接自身(自环边)`)
- }
- // 归一化两个方向的边
- const [min, max] = [edge.from, edge.to].sort()
- // 默认检查无向边
- const checkDirection = false
- const edgeKey = checkDirection
- ? `${edge.from},${edge.to}`
- : `${min},${max}`
- // 检查重复边
- if (edgeSet.has(edgeKey)) {
- errors.push(`发现重复边: ${edge.from} → ${edge.to}`)
- } else {
- edgeSet.add(edgeKey)
- }
- })
- // 处理验证结果
- if (errors.length > 0) {
- ElMessage.error({
- message: `发现 ${errors.length} 个错误:<br>${errors.join('<br>')}`,
- duration: 5000,
- dangerouslyUseHTMLString: true, // 启用 HTML 解析
- customClass: 'error-message' // 可选:添加自定义样式类
- })
- return
- }
- // 构造上传数据
- const uploadData = {
- nodes: nodes.value
- .filter(node => node.type !== '')
- .map(node => ({
- id: node.id,
- type: node.type,
- name: node.name
- })),
- edges: edges.value
- .filter(edge => edge.from && edge.to)
- .map(edge => ({
- source: edge.from,
- target: edge.to
- }))
- }
- // 执行上传
- try {
- const response = await postData('/inputFile/', uploadData)
- ElMessage.success('输入数据验证通过,上传成功')
- preparePlan(response)
- } catch (error) {
- ElMessage.error('输入数据上传失败: ' + error.message)
- }
- }
- // 上传文件处理
- const handleNodeFileChange = (file, fileList) => {
- if (fileList.length > 1) {
- fileList.splice(0, 1);
- }
- nodeFile.value = fileList[0].raw
- }
- const handleEdgeFileChange = (file, fileList) => {
- if (fileList.length > 1) {
- fileList.splice(0, 1);
- }
- edgeFile.value = fileList[0].raw
- }
- const formatSize = (bytes) => {
- if (bytes === 0) return '0 B'
- const k = 1024
- const sizes = ['B', 'KB', 'MB', 'GB']
- const i = Math.floor(Math.log(bytes) / Math.log(k))
- return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i]
- }
- const handleUpload = async () => {
- try {
- // 上传文件
- console.log(nodeFile)
- const formData = new FormData();
- formData.append('nodeFileName', nodeFile.value.name)
- formData.append('edgeFileName', edgeFile.value.name)
- formData.append('nodes', nodeFile.value)
- formData.append('edges', edgeFile.value)
- formData.append('type', 'csv')
- formData.append('usage', 'input')
- const response = await postFile('/uploadfile/', formData);
- if (response.status == 'success') {
- ElMessage.success('文件上传成功')
- nodeFile.value = null;
- edgeFile.value = null;
- preparePlan(response)
- } else {
- ElMessage.error("上传文件出错")
- console.log(response)
- }
- } catch (error) {
- console.log(error)
- ElMessage.error('文件验证失败: ' + error.response.data.message)
- nodeFile.value = null;
- edgeFile.value = null;
- }
- }
- // 上传文件和输入得到的响应结果相同,使用同一函数跳转到规划页面
- const preparePlan = (response) => {
- // 保存上传的文件信息
- response.data.forEach(file => {
- if (file.content === 'node') {
- useAnalyzeInfo.analyzeInfo.value.nodeFile.id = file.id
- useAnalyzeInfo.analyzeInfo.value.nodeFile.amount = file.ndoes
- useAnalyzeInfo.analyzeInfo.value.nodeFile.sNodes = file.sNodes
- useAnalyzeInfo.analyzeInfo.value.nodeFile.dNodes = file.dNodes
- useAnalyzeInfo.analyzeInfo.value.nodeFile.iNodes = file.iNodes
- }
- if (file.content === 'edge') {
- useAnalyzeInfo.analyzeInfo.value.edgeFile.id = file.id
- useAnalyzeInfo.analyzeInfo.value.edgeFile.amount = file.edges
- }
- // 获取创建的分析任务ID
- if (file.content === 'mission') {
- useAnalyzeInfo.analyzeInfo.value.mission.id = file.id
- useAnalyzeInfo.analyzeInfo.value.mission.name = file.name
- useAnalyzeInfo.analyzeInfo.value.mission.status = file.status
- }
- })
- // 将获取到的数据写入页面缓存,防止刷新丢失
- sessionStorage.setItem('analyze-info', JSON.stringify(useAnalyzeInfo.analyzeInfo.value))
- // 跳转到规划页面
- inputMethod.value = "done";
- console.log(useAnalyzeInfo.analyzeInfo.value)
- router.push(`/dashboard/analyze/plan`)
- updateUploadHistory()
- }
- const route = useRoute();
- // 监听路由变化
- watch(
- () => route.path,
- (newPath) => {
- // 路由变为plan时,修改显示内容
- if (newPath === '/dashboard/analyze/plan' || newPath === '/dashboard/analyze/plan/calculate') {
- inputMethod.value = "done"
- }
- if (newPath === '/dashboard/analyze'){
- inputMethod.value = "upload"
- }
- },
- { immediate: true }
- );
- </script>
- <style lang="scss" scoped>
- .analysis-container {
- padding: 20px;
- height: calc(100vh - 60px);
- overflow-y: auto;
- .upload-section {
- margin-bottom: 20px;
- .input-method-select {
- margin-bottom: 20px;
- }
- .upload-area {
- display: flex;
- flex-direction: column;
- gap: 15px;
- .file-upload {
- display: flex;
- align-items: center;
- gap: 10px;
- padding: 15px;
- border: 1px dashed var(--el-border-color);
- border-radius: 8px;
- .file-info {
- color: var(--el-text-color-secondary);
- font-size: 0.9em;
- }
- }
- .upload-button {
- margin-top: 15px;
- width: 200px;
- align-self: center;
- }
- .progress-bar {
- margin-top: 10px;
- }
- }
- }
- .instruction-section {
- height: 100%;
- h3 {
- margin-bottom: 15px;
- }
- .instruction-content {
- .file-format {
- margin-bottom: 25px;
- .format-tag {
- margin-bottom: 10px;
- }
- .format-example {
- background: var(--el-fill-color-light);
- padding: 10px;
- border-radius: 6px;
- pre {
- margin: 0;
- font-family: monospace;
- }
- .example-text {
- color: var(--el-text-color-secondary);
- margin: 8px 0;
- }
- }
- }
- }
- }
- }
- .online-input {
- padding: 15px;
- .input-section {
- border: 1px solid var(--el-border-color);
- border-radius: 8px;
- padding: 15px;
- margin-bottom: 20px;
- h4 {
- margin-bottom: 15px;
- color: var(--el-text-color-primary);
- }
- .input-row {
- margin-bottom: 12px;
- .node-id {
- padding: 8px 12px;
- background: var(--el-fill-color-light);
- border-radius: 4px;
- text-align: center;
- }
- }
- }
- }
- .el-message.error-message {
- white-space: pre-line;
- line-height: 1.6;
- padding: 15px 20px;
- }
- </style>
|