新增项目证据库一期功能

This commit is contained in:
wjj
2026-04-21 16:46:47 +08:00
parent d4ac165723
commit addea20fa1
26 changed files with 1489 additions and 13 deletions

View File

@@ -283,10 +283,23 @@
:visible.sync="detailVisible"
append-to-body
custom-class="detail-dialog"
title="流水详情"
width="980px"
@close="closeDetailDialog"
>
<template slot="title">
<div class="detail-dialog-title">
<span>流水详情</span>
<el-button
class="evidence-corner-btn"
size="mini"
plain
:disabled="detailLoading || !buildFlowEvidenceFingerprint(detailData)"
@click="handleAddEvidence"
>
加入证据库
</el-button>
</div>
</template>
<div v-loading="detailLoading" class="detail-dialog-body">
<div class="detail-overview-grid">
<div class="detail-field">
@@ -394,6 +407,7 @@ import {
getBankStatementOptions,
getBankStatementDetail,
} from "@/api/ccdiProjectBankStatement";
import { buildFlowEvidenceFingerprint, buildFlowEvidenceSnapshot } from "@/utils/ccdiEvidence";
const TAB_MAP = {
all: "all",
@@ -518,6 +532,7 @@ export default {
},
},
methods: {
buildFlowEvidenceFingerprint,
async getList() {
this.syncProjectId();
if (!this.queryParams.projectId) {
@@ -638,6 +653,29 @@ export default {
this.detailLoading = false;
this.detailData = createEmptyDetailData();
},
handleAddEvidence() {
const detail = this.detailData || {};
const sourceRecordId = buildFlowEvidenceFingerprint(detail);
const amountText = this.formatSignedAmount(detail.displayAmount);
const counterparty = this.formatCounterpartyName(detail);
const hitTagText = Array.isArray(detail.hitTags) && detail.hitTags.length
? `,命中${detail.hitTags.map((tag) => tag.ruleName).filter(Boolean).join("、")}标签`
: "";
this.$emit("evidence-confirm", {
evidenceType: "FLOW",
relatedPersonName: this.resolveFlowRelatedPerson(detail),
relatedPersonId: detail.cretNo || "",
evidenceTitle: `${this.resolveFlowRelatedPerson(detail)} / ${this.formatField(detail.leAccountNo)}`,
evidenceSummary: `${this.formatField(detail.trxDate)}${this.resolveFlowRelatedPerson(detail)}账户与${counterparty}发生交易,金额${amountText}${hitTagText}`,
sourceType: "BANK_STATEMENT",
sourceRecordId,
sourcePage: "流水详情",
snapshotJson: JSON.stringify(buildFlowEvidenceSnapshot(detail)),
});
},
resolveFlowRelatedPerson(detail) {
return this.formatField(detail.leAccountName) === "-" ? "关联人员" : this.formatField(detail.leAccountName);
},
handleExport() {
if (this.total === 0) {
return;
@@ -751,6 +789,26 @@ export default {
gap: 12px;
}
.detail-dialog-title {
display: inline-flex;
align-items: center;
gap: 10px;
}
.evidence-corner-btn {
padding: 4px 9px;
font-size: 12px;
color: #5b7fb8;
border-color: #d6e4f7;
background: #f8fbff;
&:hover {
color: #2474e8;
border-color: #9fc3ff;
background: #edf5ff;
}
}
.shell-sidebar,
.shell-main {
border: 1px solid #ebeef5;

View File

@@ -0,0 +1,136 @@
<template>
<el-dialog
:visible.sync="dialogVisible"
append-to-body
title="确认为证据"
width="560px"
:close-on-click-modal="false"
@close="handleClose"
>
<el-form label-position="top" class="evidence-confirm-form">
<el-form-item label="证据类型">
<el-input :value="typeLabel" readonly />
</el-form-item>
<el-form-item label="关联人员">
<el-input :value="payload.relatedPersonName || '-'" readonly />
</el-form-item>
<el-form-item label="证据摘要">
<el-input
:value="payload.evidenceSummary || '-'"
readonly
type="textarea"
:rows="3"
/>
</el-form-item>
<el-form-item label="确认理由/备注" required>
<el-input
v-model.trim="confirmReason"
type="textarea"
:rows="4"
maxlength="500"
show-word-limit
placeholder="请填写为什么将该详情确认为证据"
/>
</el-form-item>
</el-form>
<div slot="footer">
<el-button @click="handleClose">取消</el-button>
<el-button type="primary" :loading="submitting" @click="handleSubmit">
确认入库
</el-button>
</div>
</el-dialog>
</template>
<script>
import { saveEvidence } from "@/api/ccdiEvidence";
const TYPE_LABEL_MAP = {
FLOW: "流水证据",
MODEL: "模型证据",
ASSET: "资产证据",
};
export default {
name: "EvidenceConfirmDialog",
props: {
visible: {
type: Boolean,
default: false,
},
payload: {
type: Object,
default: () => ({}),
},
},
data() {
return {
confirmReason: "",
submitting: false,
};
},
computed: {
dialogVisible: {
get() {
return this.visible;
},
set(value) {
if (!value) {
this.handleClose();
}
},
},
typeLabel() {
return TYPE_LABEL_MAP[this.payload.evidenceType] || this.payload.evidenceType || "-";
},
},
watch: {
visible(value) {
if (value) {
this.confirmReason = "";
}
},
},
methods: {
handleClose() {
if (this.submitting) {
return;
}
this.$emit("update:visible", false);
},
async handleSubmit() {
if (!this.confirmReason) {
this.$message.warning("请填写确认理由/备注");
return;
}
this.submitting = true;
try {
const data = {
...this.payload,
confirmReason: this.confirmReason,
};
const response = await saveEvidence(data);
this.$message.success("证据入库成功");
this.$emit("saved", response.data);
this.$emit("update:visible", false);
} catch (error) {
this.$message.error("证据入库失败");
console.error("证据入库失败", error);
} finally {
this.submitting = false;
}
},
},
};
</script>
<style lang="scss" scoped>
.evidence-confirm-form {
:deep(.el-form-item__label) {
padding-bottom: 6px;
font-weight: 600;
color: #606266;
}
}
</style>

View File

@@ -0,0 +1,180 @@
<template>
<el-drawer
:visible.sync="drawerVisible"
append-to-body
title="证据线索"
size="420px"
custom-class="evidence-drawer"
@open="loadEvidence"
>
<div class="evidence-drawer-body">
<el-input
v-model.trim="keyword"
clearable
size="small"
prefix-icon="el-icon-search"
placeholder="搜索姓名、账号、证据编号"
@keyup.enter.native="loadEvidence"
@clear="loadEvidence"
>
<el-button slot="append" @click="loadEvidence">查询</el-button>
</el-input>
<div class="evidence-list" v-loading="loading">
<el-empty
v-if="!loading && evidenceList.length === 0"
:image-size="80"
description="暂无证据线索"
/>
<article
v-for="item in evidenceList"
:key="item.evidenceId"
class="evidence-card"
>
<div class="evidence-card__header">
<span class="evidence-code">EV-{{ formatEvidenceId(item.evidenceId) }} {{ formatType(item.evidenceType) }}</span>
<el-tag size="mini" type="danger" effect="plain">确认可疑</el-tag>
</div>
<div class="evidence-title">{{ item.evidenceTitle }}</div>
<div class="evidence-summary">{{ item.evidenceSummary }}</div>
<div class="evidence-meta">
来源{{ item.sourcePage || formatSource(item.sourceType) }}确认人{{ item.confirmBy || "-" }}
</div>
<div v-if="item.confirmReason" class="evidence-meta">
备注{{ item.confirmReason }}
</div>
</article>
</div>
</div>
</el-drawer>
</template>
<script>
import { listEvidence } from "@/api/ccdiEvidence";
const TYPE_LABEL_MAP = {
FLOW: "流水证据",
MODEL: "模型证据",
ASSET: "资产证据",
};
const SOURCE_LABEL_MAP = {
BANK_STATEMENT: "流水详情",
MODEL_DETAIL: "模型详情",
ASSET_DETAIL: "资产详情",
};
export default {
name: "EvidenceDrawer",
props: {
visible: {
type: Boolean,
default: false,
},
projectId: {
type: [String, Number],
default: null,
},
},
data() {
return {
keyword: "",
loading: false,
evidenceList: [],
};
},
computed: {
drawerVisible: {
get() {
return this.visible;
},
set(value) {
this.$emit("update:visible", value);
},
},
},
methods: {
async loadEvidence() {
if (!this.projectId) {
this.evidenceList = [];
return;
}
this.loading = true;
try {
const response = await listEvidence({
projectId: this.projectId,
keyword: this.keyword,
});
this.evidenceList = response.data || [];
} catch (error) {
this.evidenceList = [];
this.$message.error("加载证据线索失败");
console.error("加载证据线索失败", error);
} finally {
this.loading = false;
}
},
formatEvidenceId(value) {
return String(value || "").padStart(3, "0");
},
formatType(type) {
return TYPE_LABEL_MAP[type] || type || "证据";
},
formatSource(sourceType) {
return SOURCE_LABEL_MAP[sourceType] || sourceType || "-";
},
},
};
</script>
<style lang="scss" scoped>
.evidence-drawer-body {
padding: 0 16px 16px;
}
.evidence-list {
margin-top: 14px;
}
.evidence-card {
padding: 14px;
margin-bottom: 12px;
border: 1px solid #e5eaf2;
border-radius: 10px;
background: #fff;
}
.evidence-card__header {
display: flex;
align-items: center;
justify-content: space-between;
gap: 10px;
}
.evidence-code {
color: #2474e8;
font-size: 13px;
font-weight: 700;
}
.evidence-title {
margin-top: 8px;
font-size: 15px;
font-weight: 700;
color: #1f2937;
}
.evidence-summary {
margin-top: 8px;
font-size: 13px;
line-height: 1.6;
color: #475467;
}
.evidence-meta {
margin-top: 8px;
font-size: 12px;
color: #8a96a8;
line-height: 1.5;
}
</style>

View File

@@ -21,10 +21,24 @@
</template>
<el-table-column type="expand" width="1">
<template slot-scope="scope">
<family-asset-liability-detail
:detail="detailCache[scope.row.staffIdCard]"
:loading="Boolean(detailLoadingMap[scope.row.staffIdCard])"
/>
<div class="family-detail-wrapper">
<div class="family-detail-toolbar">
<span class="family-detail-title">资产详情</span>
<el-button
class="evidence-corner-btn"
size="mini"
plain
:disabled="Boolean(detailLoadingMap[scope.row.staffIdCard]) || !buildAssetEvidenceFingerprint(scope.row)"
@click="handleAddEvidence(scope.row)"
>
加入证据库
</el-button>
</div>
<family-asset-liability-detail
:detail="detailCache[scope.row.staffIdCard]"
:loading="Boolean(detailLoadingMap[scope.row.staffIdCard])"
/>
</div>
</template>
</el-table-column>
<el-table-column type="index" label="序号" width="60" />
@@ -67,6 +81,7 @@
<script>
import { getFamilyAssetLiabilityDetail } from "@/api/ccdi/projectSpecialCheck";
import { buildAssetEvidenceFingerprint, buildAssetEvidenceSnapshot } from "@/utils/ccdiEvidence";
import FamilyAssetLiabilityDetail from "./FamilyAssetLiabilityDetail";
export default {
@@ -114,6 +129,7 @@ export default {
},
},
methods: {
buildAssetEvidenceFingerprint,
resolveRiskTagType(riskLevelCode) {
const riskTagTypeMap = {
NORMAL: "success",
@@ -164,6 +180,30 @@ export default {
this.detailCache = {};
this.detailLoadingMap = {};
},
handleAddEvidence(row) {
if (!row || !row.staffIdCard) {
return;
}
const sourceRecordId = buildAssetEvidenceFingerprint(row);
if (!sourceRecordId) {
this.$message.warning("缺少人员身份证或资产字段,暂不能加入证据库");
return;
}
const detail = this.detailCache[row.staffIdCard] || {};
const summary = detail.summary || {};
const evidenceSummary = `${row.staffName}家庭资产负债核查:家庭总年收入${this.formatAmount(row.totalIncome)},家庭总负债${this.formatAmount(row.totalDebt)},家庭总资产${this.formatAmount(row.totalAsset)},风险情况${row.riskLevelName || "-" }`;
this.$emit("evidence-confirm", {
evidenceType: "ASSET",
relatedPersonName: row.staffName || "关联人员",
relatedPersonId: row.staffIdCard || "",
evidenceTitle: `${row.staffName || "关联人员"} / 家庭资产负债核查`,
evidenceSummary,
sourceType: "ASSET_DETAIL",
sourceRecordId,
sourcePage: "资产详情",
snapshotJson: JSON.stringify(buildAssetEvidenceSnapshot(row, detail, summary)),
});
},
},
};
</script>
@@ -204,6 +244,39 @@ export default {
overflow: hidden;
}
.family-detail-wrapper {
padding: 12px 0;
}
.family-detail-toolbar {
display: flex;
align-items: center;
justify-content: flex-end;
gap: 10px;
margin-bottom: 10px;
}
.family-detail-title {
margin-right: auto;
color: #64748b;
font-size: 13px;
font-weight: 600;
}
.evidence-corner-btn {
padding: 4px 9px;
font-size: 12px;
color: #5b7fb8;
border-color: #d6e4f7;
background: #f8fbff;
&:hover {
color: #2474e8;
border-color: #9fc3ff;
background: #edf5ff;
}
}
:deep(.family-table th) {
background: #f8fafc;
color: #64748b;

View File

@@ -33,7 +33,10 @@
@selection-change="handleRiskModelSelectionChange"
@view-project-analysis="handleRiskModelProjectAnalysis"
/>
<risk-detail-section :section-data="currentData.riskDetails" />
<risk-detail-section
:section-data="currentData.riskDetails"
@evidence-confirm="$emit('evidence-confirm', $event)"
/>
</div>
<project-analysis-dialog
:visible.sync="projectAnalysisDialogVisible"
@@ -43,6 +46,7 @@
:model-summary="projectAnalysisModelSummary"
:project-name="projectInfo.projectName"
@close="handleProjectAnalysisDialogClose"
@evidence-confirm="$emit('evidence-confirm', $event)"
/>
</div>
</template>

View File

@@ -75,15 +75,28 @@
<div v-else-if='group.groupType === "OBJECT"' class="object-card-grid">
<article
v-for="(item, index) in group.records || []"
:key="`${item.title || index}-object`"
:key="`${resolveGroupKey(group, groupIndex)}-${item.title || 'object'}-${index}`"
class="object-card"
>
<div class="object-card__title">{{ item.title || "-" }}</div>
<div class="object-card__subtitle">{{ item.subtitle || "-" }}</div>
<div class="object-card__header">
<div>
<div class="object-card__title">{{ item.title || "-" }}</div>
<div class="object-card__subtitle">{{ item.subtitle || "-" }}</div>
</div>
<el-button
class="evidence-corner-btn"
size="mini"
plain
:disabled="!buildModelEvidenceFingerprint(resolvePersonIdCard(), item.modelCode)"
@click="handleAddModelEvidence(item, group)"
>
加入证据库
</el-button>
</div>
<div v-if="item.riskTags && item.riskTags.length" class="tag-list">
<el-tag
v-for="(tag, tagIndex) in item.riskTags"
:key="`${item.title || index}-risk-${tagIndex}`"
:key="`${resolveGroupKey(group, groupIndex)}-${item.title || index}-risk-${tagIndex}`"
size="mini"
effect="plain"
>
@@ -97,7 +110,7 @@
<p class="object-card__summary">{{ item.summary || "-" }}</p>
<div
v-for="(field, fieldIndex) in item.extraFields || []"
:key="`${item.title || index}-field-${fieldIndex}`"
:key="`${resolveGroupKey(group, groupIndex)}-${item.title || index}-field-${fieldIndex}`"
class="summary-row"
>
<span class="summary-row__label">{{ field.label }}</span>
@@ -113,6 +126,8 @@
</template>
<script>
import { buildModelEvidenceFingerprint, MODEL_EVIDENCE_FINGERPRINT_RULE } from "@/utils/ccdiEvidence";
export default {
name: "ProjectAnalysisAbnormalTab",
props: {
@@ -122,6 +137,14 @@ export default {
groups: [],
}),
},
person: {
type: Object,
default: () => ({}),
},
projectId: {
type: [String, Number],
default: null,
},
},
data() {
return {
@@ -150,6 +173,7 @@ export default {
},
},
methods: {
buildModelEvidenceFingerprint,
resolveGroupKey(group, index = 0) {
return group.groupCode || group.groupName || `BANK_STATEMENT_${index}`;
},
@@ -170,6 +194,48 @@ export default {
const groupKey = this.resolveGroupKey(group);
this.$set(this.statementPageMap, groupKey, page);
},
handleAddModelEvidence(item, group) {
const safeItem = item || {};
const safeGroup = group || {};
const relatedPersonName = this.resolveRelatedPersonName(safeItem);
const personIdCard = this.resolvePersonIdCard();
const sourceRecordId = buildModelEvidenceFingerprint(personIdCard, safeItem.modelCode);
if (!sourceRecordId) {
this.$message.warning("缺少人员身份证或模型编码,暂不能加入证据库");
return;
}
const riskTags = Array.isArray(safeItem.riskTags) ? safeItem.riskTags.join("、") : "";
const reason = safeItem.reasonDetail || safeItem.summary || "-";
const payload = {
evidenceType: "MODEL",
relatedPersonName,
relatedPersonId: personIdCard,
evidenceTitle: `${relatedPersonName} / ${safeItem.title || safeGroup.groupName || "模型异常"}`,
evidenceSummary: `${safeItem.title || safeGroup.groupName || "模型异常"}${reason}`,
sourceType: "MODEL_DETAIL",
sourceRecordId,
sourcePage: "模型详情",
snapshotJson: JSON.stringify({
group: safeGroup,
item: safeItem,
person: this.person,
riskTags,
evidenceFingerprint: sourceRecordId,
evidenceFingerprintRule: MODEL_EVIDENCE_FINGERPRINT_RULE,
}),
};
this.$emit("evidence-confirm", payload);
this.$root.$emit("ccdi-evidence-confirm", payload);
},
resolvePersonIdCard() {
return (this.person && (this.person.idNo || this.person.staffIdCard)) || "";
},
resolveRelatedPersonName(item) {
if (this.person && (this.person.name || this.person.staffName)) {
return this.person.name || this.person.staffName;
}
return item.title || "关联人员";
},
},
};
</script>
@@ -252,12 +318,35 @@ export default {
background: #f8fafc;
}
.object-card__header {
display: flex;
align-items: flex-start;
justify-content: space-between;
gap: 12px;
margin-bottom: 8px;
}
.object-card__title {
font-size: 15px;
font-weight: 600;
color: #0f172a;
}
.evidence-corner-btn {
flex: 0 0 auto;
padding: 4px 9px;
font-size: 12px;
color: #5b7fb8;
border-color: #d6e4f7;
background: #f8fbff;
&:hover {
color: #2474e8;
border-color: #9fc3ff;
background: #edf5ff;
}
}
.object-card__subtitle {
margin-top: 6px;
font-size: 12px;

View File

@@ -46,7 +46,12 @@
</el-alert>
<el-tabs v-model="activeTab" class="project-analysis-tabs" stretch>
<el-tab-pane label="异常明细" name="abnormalDetail">
<project-analysis-abnormal-tab :detail-data="dialogData.abnormalDetail" />
<project-analysis-abnormal-tab
:detail-data="dialogData.abnormalDetail"
:person="person"
:project-id="projectId"
@evidence-confirm="$emit('evidence-confirm', $event)"
/>
</el-tab-pane>
<el-tab-pane label="资产分析" name="assetAnalysis">
<project-analysis-placeholder-tab :tab-data="getTabData('assetAnalysis')" />

View File

@@ -207,10 +207,23 @@
:visible.sync="detailVisible"
append-to-body
custom-class="detail-dialog"
title="流水详情"
width="980px"
@close="closeDetailDialog"
>
<template slot="title">
<div class="detail-dialog-title">
<span>流水详情</span>
<el-button
class="evidence-corner-btn"
size="mini"
plain
:disabled="detailLoading || !buildFlowEvidenceFingerprint(detailData)"
@click="handleAddEvidence"
>
加入证据库
</el-button>
</div>
</template>
<div v-loading="detailLoading" class="detail-dialog-body">
<div class="detail-overview-grid">
<div class="detail-field">
@@ -318,6 +331,7 @@ import {
getOverviewSuspiciousTransactions,
} from "@/api/ccdi/projectOverview";
import { getBankStatementDetail } from "@/api/ccdiProjectBankStatement";
import { buildFlowEvidenceFingerprint, buildFlowEvidenceSnapshot } from "@/utils/ccdiEvidence";
const SUSPICIOUS_TYPE_OPTIONS = [
{ value: "ALL", label: "全部可疑人员类型" },
@@ -428,6 +442,7 @@ export default {
},
},
methods: {
buildFlowEvidenceFingerprint,
async handleSuspiciousTypeChange(command) {
this.currentSuspiciousType = command;
this.suspiciousPageNum = 1;
@@ -586,6 +601,31 @@ export default {
this.detailLoading = false;
this.detailData = createEmptyDetailData();
},
handleAddEvidence() {
const detail = this.detailData || {};
const sourceRecordId = buildFlowEvidenceFingerprint(detail);
const amountText = this.formatSignedAmount(detail.displayAmount);
const counterparty = this.formatCounterpartyName(detail);
const relatedPersonName = this.resolveFlowRelatedPerson(detail);
const hitTagText = Array.isArray(detail.hitTags) && detail.hitTags.length
? `,命中${detail.hitTags.map((tag) => tag.ruleName).filter(Boolean).join("、")}标签`
: "";
this.$emit("evidence-confirm", {
evidenceType: "FLOW",
relatedPersonName,
relatedPersonId: detail.cretNo || "",
evidenceTitle: `${relatedPersonName} / ${this.formatField(detail.leAccountNo)}`,
evidenceSummary: `${this.formatField(detail.trxDate)}${relatedPersonName}账户与${counterparty}发生交易,金额${amountText}${hitTagText}`,
sourceType: "BANK_STATEMENT",
sourceRecordId,
sourcePage: "流水详情",
snapshotJson: JSON.stringify(buildFlowEvidenceSnapshot(detail)),
});
},
resolveFlowRelatedPerson(detail) {
const value = this.formatField(detail.leAccountName);
return value === "-" ? "关联人员" : value;
},
handleRiskDetailExport() {
if (!this.projectId) {
return;
@@ -960,6 +1000,26 @@ export default {
gap: 12px;
}
.detail-dialog-title {
display: inline-flex;
align-items: center;
gap: 10px;
}
.evidence-corner-btn {
padding: 4px 9px;
font-size: 12px;
color: #5b7fb8;
border-color: #d6e4f7;
background: #f8fbff;
&:hover {
color: #2474e8;
border-color: #9fc3ff;
background: #edf5ff;
}
}
:deep(.detail-dialog) {
border-radius: 8px;

View File

@@ -19,6 +19,7 @@
:project-id="projectId"
:title="sectionTitle"
:subtitle="sectionSubtitle"
@evidence-confirm="$emit('evidence-confirm', $event)"
/>
<section class="graph-placeholder-card">

View File

@@ -33,6 +33,15 @@
</div>
</div>
<div class="header-right">
<el-button
class="evidence-entry-btn"
size="mini"
plain
icon="el-icon-collection-tag"
@click="evidenceDrawerVisible = true"
>
证据线索
</el-button>
<el-menu
:default-active="activeTab"
mode="horizontal"
@@ -59,6 +68,18 @@
@name-selected="handleNameSelected"
@generate-report="handleGenerateReport"
@fetch-bank-info="handleFetchBankInfo"
@evidence-confirm="handleEvidenceConfirm"
/>
<evidence-confirm-dialog
:visible.sync="evidenceConfirmVisible"
:payload="evidencePayload"
@saved="handleEvidenceSaved"
/>
<evidence-drawer
ref="evidenceDrawer"
:visible.sync="evidenceDrawerVisible"
:project-id="projectId"
/>
</div>
</template>
@@ -69,6 +90,8 @@ import ParamConfig from "./components/detail/ParamConfig";
import PreliminaryCheck from "./components/detail/PreliminaryCheck";
import SpecialCheck from "./components/detail/SpecialCheck";
import DetailQuery from "./components/detail/DetailQuery";
import EvidenceConfirmDialog from "./components/detail/EvidenceConfirmDialog";
import EvidenceDrawer from "./components/detail/EvidenceDrawer";
import { getProject } from "@/api/ccdiProject";
export default {
@@ -79,6 +102,8 @@ export default {
PreliminaryCheck,
SpecialCheck,
DetailQuery,
EvidenceConfirmDialog,
EvidenceDrawer,
},
data() {
return {
@@ -102,6 +127,9 @@ export default {
warningThreshold: 60,
projectStatus: "0",
},
evidenceConfirmVisible: false,
evidenceDrawerVisible: false,
evidencePayload: {},
projectStatusPollingTimer: null,
projectStatusPollingInterval: 1000,
projectStatusPollingLoading: false,
@@ -139,8 +167,10 @@ export default {
// 初始化页面数据
this.initActiveTabFromRoute();
this.initPageData();
this.$root.$on("ccdi-evidence-confirm", this.handleEvidenceConfirm);
},
beforeDestroy() {
this.$root.$off("ccdi-evidence-confirm", this.handleEvidenceConfirm);
this.stopProjectStatusPolling();
},
methods: {
@@ -400,6 +430,21 @@ export default {
handleRefreshProject() {
this.initPageData();
},
handleEvidenceConfirm(payload) {
this.evidencePayload = {
projectId: this.projectId,
...(payload || {}),
};
this.evidenceConfirmVisible = true;
},
handleEvidenceSaved() {
this.evidenceDrawerVisible = true;
this.$nextTick(() => {
if (this.$refs.evidenceDrawer) {
this.$refs.evidenceDrawer.loadEvidence();
}
});
},
/** 导出报告 */
handleExport() {
console.log("导出报告");
@@ -496,6 +541,21 @@ export default {
.header-right {
display: flex;
align-items: center;
gap: 10px;
.evidence-entry-btn {
padding: 6px 10px;
font-size: 12px;
color: #5b7fb8;
border-color: #d6e4f7;
background: #f8fbff;
&:hover {
color: var(--ccdi-primary);
border-color: #9fc3ff;
background: #edf5ff;
}
}
.nav-menu {
// 移除默认背景色和边框