feat(audit): connect rule center to live asset APIs

This commit is contained in:
caoxiaozhu
2026-05-11 06:32:38 +00:00
parent 9b39df6277
commit e9eeb2e41d
4 changed files with 2099 additions and 839 deletions

View File

@@ -27,9 +27,11 @@
} }
.skill-list { .skill-list {
display: grid; display: flex;
grid-template-rows: auto auto auto minmax(0, 1fr); flex-direction: column;
min-height: 0;
padding: 18px 20px; padding: 18px 20px;
overflow: hidden;
} }
.status-tabs { .status-tabs {
@@ -73,13 +75,15 @@
.filter-set { .filter-set {
display: flex; display: flex;
align-items: center;
gap: 10px; gap: 10px;
flex: 1 1 auto;
flex-wrap: wrap; flex-wrap: wrap;
} }
.search-filter { .search-filter {
width: 260px; width: 280px;
min-height: 36px; min-height: 38px;
display: inline-flex; display: inline-flex;
align-items: center; align-items: center;
gap: 8px; gap: 8px;
@@ -109,23 +113,150 @@
color: #94a3b8; color: #94a3b8;
} }
.filter-btn, .search-filter:focus-within {
border-color: rgba(16, 185, 129, 0.48);
box-shadow: 0 0 0 3px rgba(16, 185, 129, 0.1);
}
.picker-trigger,
.ghost-filter-btn,
.create-btn, .create-btn,
.row-action { .row-action {
min-height: 36px; min-height: 38px;
border-radius: 8px; border-radius: 8px;
font-size: 13px; font-size: 13px;
font-weight: 760; font-weight: 760;
} }
.filter-btn { .picker-filter {
position: relative;
}
.picker-trigger {
min-width: 124px;
display: inline-flex; display: inline-flex;
align-items: center; align-items: center;
gap: 6px; justify-content: space-between;
padding: 0 12px; gap: 8px;
padding: 0 34px 0 12px;
border: 1px solid #d7e0ea; border: 1px solid #d7e0ea;
background: #fff; background: #fff;
color: #334155; color: #334155;
white-space: nowrap;
}
.picker-label {
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
}
.picker-trigger .mdi {
position: absolute;
right: 12px;
top: 50%;
transform: translateY(-50%);
color: #64748b;
pointer-events: none;
}
.picker-trigger:hover,
.picker-filter.open .picker-trigger {
border-color: rgba(16, 185, 129, 0.34);
background: #f6fffb;
color: #0f9f78;
}
.picker-popover {
position: absolute;
top: calc(100% + 8px);
left: 0;
width: 224px;
z-index: 40;
display: grid;
gap: 14px;
padding: 16px;
border: 1px solid #d7e0ea;
border-radius: 12px;
background: #fff;
box-shadow: 0 18px 42px rgba(15, 23, 42, 0.16);
}
.picker-popover header {
display: flex;
align-items: center;
justify-content: space-between;
gap: 12px;
}
.picker-popover header strong {
color: #0f172a;
font-size: 15px;
}
.picker-popover header button {
width: 30px;
height: 30px;
display: grid;
place-items: center;
border: 0;
border-radius: 8px;
background: transparent;
color: #64748b;
}
.picker-popover header button:hover {
background: #f1f5f9;
color: #0f172a;
}
.picker-option-list {
display: grid;
gap: 8px;
max-height: 240px;
overflow-y: auto;
}
.picker-option {
min-height: 36px;
display: inline-flex;
align-items: center;
padding: 0 12px;
border: 1px solid #d7e0ea;
border-radius: 8px;
background: #fff;
color: #334155;
font-size: 13px;
font-weight: 750;
text-align: left;
}
.picker-option:hover,
.picker-option.active {
border-color: rgba(16, 185, 129, 0.32);
background: rgba(16, 185, 129, 0.08);
color: #059669;
}
.toolbar-actions {
display: flex;
align-items: center;
gap: 10px;
}
.ghost-filter-btn {
display: inline-flex;
align-items: center;
gap: 8px;
padding: 0 12px;
border: 1px solid #d7e0ea;
background: #fff;
color: #475569;
}
.ghost-filter-btn:hover {
border-color: rgba(16, 185, 129, 0.28);
color: #059669;
} }
.create-btn { .create-btn {
@@ -139,6 +270,13 @@
box-shadow: 0 8px 18px rgba(5, 150, 105, 0.18); box-shadow: 0 8px 18px rgba(5, 150, 105, 0.18);
} }
.create-btn:disabled {
background: #cbd5e1;
color: #f8fafc;
box-shadow: none;
cursor: not-allowed;
}
.hint { .hint {
display: inline-flex; display: inline-flex;
align-items: center; align-items: center;
@@ -152,13 +290,113 @@
color: #94a3b8; color: #94a3b8;
} }
.active-filter-strip {
display: flex;
gap: 8px;
flex-wrap: wrap;
margin-bottom: 12px;
}
.active-filter-chip {
min-height: 28px;
display: inline-flex;
align-items: center;
padding: 0 10px;
border-radius: 999px;
background: rgba(16, 185, 129, 0.1);
color: #047857;
font-size: 12px;
font-weight: 800;
}
.table-wrap { .table-wrap {
flex: 1 1 auto;
position: relative;
min-height: 0; min-height: 0;
overflow: auto; overflow: auto;
border: 1px solid #edf2f7; border: 1px solid #edf2f7;
border-radius: 12px; border-radius: 12px;
} }
.table-state,
.detail-inline-state {
min-height: 220px;
display: grid;
place-items: center;
gap: 12px;
padding: 28px 24px;
text-align: center;
color: #64748b;
}
.detail-inline-state {
min-height: 180px;
display: flex;
align-items: center;
justify-content: space-between;
gap: 16px;
text-align: left;
}
.table-state i,
.detail-inline-state i {
font-size: 28px;
color: #94a3b8;
}
.table-state.error i,
.detail-inline-state.error i {
color: #dc2626;
}
.table-state.empty i {
color: #0ea5e9;
}
.table-state p,
.detail-inline-state p {
margin-top: 6px;
font-size: 13px;
line-height: 1.6;
}
.detail-inline-state strong {
color: #0f172a;
font-size: 14px;
font-weight: 850;
}
.detail-inline-state > div {
flex: 1 1 auto;
}
.state-action {
height: 36px;
display: inline-flex;
align-items: center;
justify-content: center;
padding: 0 14px;
border: 1px solid rgba(16, 185, 129, 0.28);
border-radius: 8px;
background: #fff;
color: #059669;
font-size: 13px;
font-weight: 760;
}
.list-foot {
display: flex;
align-items: center;
justify-content: flex-end;
padding-top: 12px;
}
.page-summary {
color: #64748b;
font-size: 12px;
font-weight: 700;
}
table { table {
width: 100%; width: 100%;
min-width: 1120px; min-width: 1120px;
@@ -268,6 +506,16 @@ tbody tr.spotlight {
color: #6366f1; color: #6366f1;
} }
.status-pill.danger {
background: #fee2e2;
color: #dc2626;
}
.status-pill.disabled {
background: #e2e8f0;
color: #475569;
}
.row-action { .row-action {
padding: 0 12px; padding: 0 12px;
border: 1px solid rgba(16, 185, 129, 0.32); border: 1px solid rgba(16, 185, 129, 0.32);
@@ -275,6 +523,10 @@ tbody tr.spotlight {
color: #059669; color: #059669;
} }
.row-action:hover {
background: rgba(16, 185, 129, 0.08);
}
.detail-scroll { .detail-scroll {
height: 100%; height: 100%;
overflow: auto; overflow: auto;
@@ -285,6 +537,8 @@ tbody tr.spotlight {
.detail-hero { .detail-hero {
display: grid; display: grid;
grid-template-columns: minmax(0, 1fr) 320px;
align-items: start;
gap: 10px; gap: 10px;
padding: 16px 20px; padding: 16px 20px;
} }
@@ -336,6 +590,34 @@ tbody tr.spotlight {
margin-top: 10px; margin-top: 10px;
} }
.review-note-block {
display: grid;
gap: 6px;
margin-top: 12px;
padding: 12px 14px;
border: 1px solid rgba(16, 185, 129, 0.16);
border-radius: 12px;
background: linear-gradient(180deg, #f8fffc, #ffffff);
}
.review-note-block strong {
color: #0f172a;
font-size: 13px;
font-weight: 850;
}
.review-note-block p,
.review-note-block span {
color: #64748b;
font-size: 12px;
line-height: 1.6;
}
.review-note-block.muted {
border-color: #e2e8f0;
background: #f8fafc;
}
.hero-review-meta span { .hero-review-meta span {
display: inline-flex; display: inline-flex;
align-items: center; align-items: center;
@@ -491,6 +773,13 @@ tbody tr.spotlight {
resize: vertical; resize: vertical;
} }
.field input[readonly],
.field textarea[readonly],
.prompt-block textarea[readonly] {
background: #f8fafc;
color: #334155;
}
.markdown-card { .markdown-card {
min-height: 620px; min-height: 620px;
display: grid; display: grid;
@@ -511,6 +800,39 @@ tbody tr.spotlight {
white-space: pre; white-space: pre;
} }
.markdown-editor.disabled {
background: #f8fafc;
color: #475569;
}
.subtle-banner,
.editor-foot {
display: flex;
align-items: center;
justify-content: space-between;
gap: 12px;
flex-wrap: wrap;
}
.subtle-banner {
min-height: 38px;
margin-bottom: 10px;
padding: 0 12px;
border: 1px solid #e0f2fe;
border-radius: 10px;
background: #f0f9ff;
color: #0369a1;
font-size: 12px;
font-weight: 700;
}
.editor-foot {
margin-top: 12px;
color: #64748b;
font-size: 12px;
line-height: 1.5;
}
.skill-review-side { .skill-review-side {
align-content: start; align-content: start;
padding-right: 8px; padding-right: 8px;
@@ -637,6 +959,21 @@ tbody tr.spotlight {
line-height: 1.5; line-height: 1.5;
} }
.empty-side-note {
min-height: 120px;
display: grid;
place-items: center;
gap: 8px;
color: #64748b;
font-size: 13px;
text-align: center;
}
.empty-side-note i {
font-size: 24px;
color: #94a3b8;
}
.modal-backdrop { .modal-backdrop {
position: fixed; position: fixed;
inset: 0; inset: 0;
@@ -830,6 +1167,26 @@ tbody tr.spotlight {
color: #ea580c; color: #ea580c;
} }
.test-state.danger,
.tool-state.danger {
background: #fee2e2;
color: #dc2626;
}
.review-action-strip {
display: flex;
gap: 10px;
flex-wrap: wrap;
margin-top: 16px;
}
.action-help {
margin-top: 12px;
color: #64748b;
font-size: 12px;
line-height: 1.6;
}
.tag-list { .tag-list {
display: flex; display: flex;
gap: 8px; gap: 8px;
@@ -919,6 +1276,16 @@ tbody tr.spotlight {
color: #334155; color: #334155;
} }
.minor-action.success-action {
border-color: rgba(5, 150, 105, 0.26);
color: #059669;
}
.minor-action.danger-action {
border-color: rgba(220, 38, 38, 0.2);
color: #dc2626;
}
.major-action { .major-action {
border: 1px solid #059669; border: 1px solid #059669;
background: #059669; background: #059669;
@@ -926,8 +1293,50 @@ tbody tr.spotlight {
box-shadow: 0 4px 12px rgba(5, 150, 105, .16); box-shadow: 0 4px 12px rgba(5, 150, 105, .16);
} }
.back-action:hover,
.minor-action:hover,
.major-action:hover,
.mini-btn:hover {
transform: translateY(-1px);
}
.back-action:disabled,
.minor-action:disabled,
.major-action:disabled,
.mini-btn:disabled {
opacity: 0.52;
cursor: not-allowed;
transform: none;
box-shadow: none;
}
.detail-meta-actions {
align-items: center;
}
.footer-note {
color: #64748b;
font-size: 12px;
font-weight: 700;
}
.mini-btn {
min-height: 36px;
display: inline-flex;
align-items: center;
justify-content: center;
gap: 6px;
padding: 0 12px;
border: 1px solid #d7e0ea;
border-radius: 8px;
background: #fff;
color: #334155;
font-size: 13px;
font-weight: 760;
}
.mini-btn.primary { .mini-btn.primary {
border-color: transparent; border-color: #059669;
background: #059669; background: #059669;
color: #fff; color: #fff;
} }
@@ -950,6 +1359,14 @@ tbody tr.spotlight {
.detail-grid.skill-md-detail-grid { .detail-grid.skill-md-detail-grid {
grid-template-columns: 1fr; grid-template-columns: 1fr;
} }
.skill-review-side {
padding-right: 0;
}
.review-card {
position: static;
}
} }
@media (max-width: 860px) { @media (max-width: 860px) {
@@ -963,7 +1380,9 @@ tbody tr.spotlight {
.list-toolbar, .list-toolbar,
.card-head, .card-head,
.detail-actions, .detail-actions,
.detail-action-group { .detail-action-group,
.toolbar-actions,
.detail-inline-state {
flex-direction: column; flex-direction: column;
align-items: stretch; align-items: stretch;
} }
@@ -973,12 +1392,36 @@ tbody tr.spotlight {
overflow-x: auto; overflow-x: auto;
} }
.search-filter,
.picker-trigger,
.picker-filter,
.toolbar-actions > * {
width: 100%;
}
.picker-popover {
width: min(100vw - 64px, 320px);
}
.hero-stats, .hero-stats,
.form-grid, .form-grid,
.contract-grid { .contract-grid {
grid-template-columns: 1fr; grid-template-columns: 1fr;
} }
.review-action-strip,
.modal-actions {
flex-direction: column;
}
.version-modal-summary {
grid-template-columns: 1fr;
}
.version-modal-summary i {
transform: rotate(90deg);
}
.field.span-2 { .field.span-2 {
grid-column: span 1; grid-column: span 1;
} }

View File

@@ -0,0 +1,116 @@
import { apiRequest } from './api.js'
const AUTH_USER_STORAGE_KEY = 'x-financial-auth-user'
function readActorName() {
if (typeof window === 'undefined') {
return 'system'
}
const raw = window.sessionStorage.getItem(AUTH_USER_STORAGE_KEY)
if (!raw) {
return 'system'
}
try {
const payload = JSON.parse(raw)
return String(payload?.name || payload?.username || 'system').trim() || 'system'
} catch {
return 'system'
}
}
function buildWriteHeaders(options = {}) {
const actor = String(options.actor || readActorName()).trim() || 'system'
const headers = {
'x-actor': actor
}
if (options.requestId) {
headers['x-request-id'] = String(options.requestId).trim()
}
return headers
}
function buildQuery(params = {}) {
const search = new URLSearchParams()
if (params.assetType) {
search.set('asset_type', params.assetType)
}
if (params.status) {
search.set('status', params.status)
}
if (params.domain) {
search.set('domain', params.domain)
}
if (params.keyword) {
search.set('keyword', params.keyword)
}
if (params.limit) {
search.set('limit', String(params.limit))
}
if (params.agent) {
search.set('agent', params.agent)
}
if (params.source) {
search.set('source', params.source)
}
const query = search.toString()
return query ? `?${query}` : ''
}
export function fetchAgentAssets(params = {}) {
return apiRequest(`/agent-assets${buildQuery(params)}`)
}
export function fetchAgentAssetDetail(assetId) {
return apiRequest(`/agent-assets/${assetId}`)
}
export function fetchAgentAssetVersions(assetId, limit = 5) {
return apiRequest(`/agent-assets/${assetId}/versions${buildQuery({ limit })}`)
}
export function updateAgentAsset(assetId, payload, options = {}) {
return apiRequest(`/agent-assets/${assetId}`, {
method: 'PATCH',
body: JSON.stringify(payload),
headers: buildWriteHeaders(options)
})
}
export function createAgentAssetVersion(assetId, payload, options = {}) {
return apiRequest(`/agent-assets/${assetId}/versions`, {
method: 'POST',
body: JSON.stringify(payload),
headers: buildWriteHeaders(options)
})
}
export function createAgentAssetReview(assetId, payload, options = {}) {
return apiRequest(`/agent-assets/${assetId}/reviews`, {
method: 'POST',
body: JSON.stringify(payload),
headers: buildWriteHeaders(options)
})
}
export function activateAgentAsset(assetId, options = {}) {
return apiRequest(`/agent-assets/${assetId}/activate`, {
method: 'POST',
headers: buildWriteHeaders(options)
})
}
export function fetchAgentRuns(params = {}) {
return apiRequest(`/agent-runs${buildQuery(params)}`)
}

View File

@@ -7,59 +7,144 @@
<div class="hero-title"> <div class="hero-title">
<div class="skill-badge" :class="selectedSkill.badgeTone">{{ selectedSkill.typeLabel }}</div> <div class="skill-badge" :class="selectedSkill.badgeTone">{{ selectedSkill.typeLabel }}</div>
<h2>{{ selectedSkill.name }}</h2> <h2>{{ selectedSkill.name }}</h2>
<p>{{ selectedSkill.summary }}</p> <p>{{ selectedSkill.summary || '当前资产尚未补充说明。' }}</p>
<div v-if="selectedSkill.type === 'rules'" class="hero-review-meta">
<div class="hero-review-meta">
<span>
<i class="mdi mdi-code-tags"></i>
{{ selectedSkill.code }}
</span>
<span>
<i class="mdi mdi-account-outline"></i>
负责人{{ selectedSkill.owner }}
</span>
<span> <span>
<i class="mdi mdi-account-check-outline"></i> <i class="mdi mdi-account-check-outline"></i>
审核人{{ selectedSkill.reviewer }} 审核人{{ selectedSkill.reviewer }}
</span> </span>
<b :class="['status-pill', selectedSkill.statusTone]">{{ selectedSkill.status }}</b> <b :class="['status-pill', selectedSkill.statusTone]">{{ selectedSkill.status }}</b>
<span>{{ selectedSkill.status === '已上线' ? '审核通过,可上线' : '待审核通过后上线' }}</span> <b
v-if="selectedSkillIsRule"
:class="['status-pill', selectedSkill.reviewStatusTone]"
>
{{ selectedSkill.reviewStatusLabel }}
</b>
</div>
<div v-if="selectedSkillIsRule" class="review-note-block">
<strong>上线约束</strong>
<p>{{ activateBlockedReason || '当前规则版本审核通过后可正式上线。' }}</p>
<span v-if="showReviewNote">
审核时间{{ selectedSkill.reviewTimeLabel }}
<template v-if="selectedSkill.reviewNote"> · 审核意见{{ selectedSkill.reviewNote }}</template>
</span>
</div>
</div>
<div class="hero-stats">
<div class="hero-stat">
<span>资产编码</span>
<strong>{{ selectedSkill.code }}</strong>
</div>
<div class="hero-stat">
<span>业务域</span>
<strong>{{ selectedSkill.category }}</strong>
</div>
<div class="hero-stat">
<span>{{ selectedSkillIsRule ? '当前展示版本' : '当前版本' }}</span>
<strong>{{ selectedSkill.displayVersion || selectedSkill.version }}</strong>
</div>
<div class="hero-stat">
<span>最近更新</span>
<strong>{{ selectedSkill.updatedAt }}</strong>
</div> </div>
</div> </div>
</section> </section>
<div class="detail-grid" :class="{ 'skill-md-detail-grid': selectedSkill.type === 'rules' }"> <section v-if="detailError" class="detail-inline-state panel error">
<i class="mdi mdi-alert-circle-outline"></i>
<div>
<strong>资产详情加载失败</strong>
<p>{{ detailError }}</p>
</div>
<button class="state-action" type="button" @click="openAssetDetail(selectedSkill)">重新加载</button>
</section>
<section v-else-if="detailLoading && selectedSkill.loading" class="detail-inline-state panel">
<i class="mdi mdi-loading mdi-spin"></i>
<div>
<strong>正在加载资产详情</strong>
<p>列表数据已就绪正在补充版本审核和运行信息</p>
</div>
</section>
<div
v-else
class="detail-grid"
:class="{ 'skill-md-detail-grid': selectedSkill.type === 'rules' }"
>
<section class="detail-main"> <section class="detail-main">
<article v-if="selectedSkill.type === 'rules'" class="detail-card panel markdown-card"> <article v-if="selectedSkill.type === 'rules'" class="detail-card panel markdown-card">
<div class="card-head"> <div class="card-head">
<div> <div>
<h3>Markdown 规则内容</h3> <h3>Markdown 规则内容</h3>
<p>管理员直接编辑该规则对应的 .md 审查规则文件内容</p> <p>当前展示版本{{ selectedSkill.displayVersion }}保存后会生成新的版本快照</p>
</div> </div>
<button class="mini-btn"> <button
class="mini-btn primary"
type="button"
:disabled="!canEditMarkdown || detailBusy"
@click="saveRuleMarkdown"
>
<i class="mdi mdi-content-save-outline"></i> <i class="mdi mdi-content-save-outline"></i>
<span>保存 .md</span> <span>{{ actionState === 'save-markdown' ? '保存中...' : '保存 Markdown' }}</span>
</button> </button>
</div> </div>
<div v-if="detailLoading" class="subtle-banner">
<i class="mdi mdi-loading mdi-spin"></i>
<span>正在刷新规则详情...</span>
</div>
<label class="field"> <label class="field">
<span>{{ selectedSkill.fields.find((field) => field.label === '文件路径')?.value }}</span> <span>{{ selectedSkill.code }}</span>
<textarea <textarea
v-model="selectedSkill.markdownContent" v-model="selectedSkill.markdownContent"
class="markdown-editor" class="markdown-editor"
:class="{ disabled: !canEditMarkdown }"
spellcheck="false" spellcheck="false"
:readonly="!canEditMarkdown || detailBusy"
></textarea> ></textarea>
</label> </label>
<div class="editor-foot">
<span>版本说明{{ selectedSkill.currentVersionChangeNote }}</span>
<span>最近保存{{ selectedSkill.updatedAt }}</span>
</div>
<div v-if="!canEditMarkdown" class="review-note-block muted">
<strong>只读模式</strong>
<p>当前账号没有规则编辑权限Markdown 仅可查看</p>
</div>
</article> </article>
<article v-else class="detail-card panel"> <article class="detail-card panel">
<div class="card-head"> <div class="card-head">
<div> <div>
<h3>{{ selectedSkill.configTitle }}</h3> <h3>{{ selectedSkill.configTitle }}</h3>
<p>{{ selectedSkill.configDesc }}</p> <p>{{ selectedSkill.configDesc }}</p>
</div> </div>
<button class="mini-btn">保存草稿</button> <span class="edit-badge">{{ selectedSkill.version }}</span>
</div> </div>
<div class="form-grid"> <div class="form-grid">
<label v-for="field in selectedSkill.fields" :key="field.label" class="field"> <label v-for="field in selectedSkill.fields" :key="field.label" class="field">
<span>{{ field.label }}</span> <span>{{ field.label }}</span>
<input :value="field.value" /> <input :value="field.value" readonly />
</label> </label>
<label class="field span-2"> <label class="field span-2">
<span>说明</span> <span>适用场景</span>
<textarea rows="3" :value="selectedSkill.summary"></textarea> <textarea rows="3" :value="selectedSkill.scope" readonly></textarea>
</label> </label>
</div> </div>
</article> </article>
@@ -74,23 +159,29 @@
</div> </div>
<div class="prompt-stack"> <div class="prompt-stack">
<section v-for="section in selectedSkill.promptSections" :key="section.title" class="prompt-block"> <section
v-for="section in selectedSkill.promptSections"
:key="section.title"
class="prompt-block"
>
<header> <header>
<strong>{{ section.title }}</strong> <strong>{{ section.title }}</strong>
<span>{{ section.intent }}</span> <span>{{ section.intent }}</span>
</header> </header>
<textarea rows="5" :value="section.content"></textarea> <textarea rows="5" :value="section.content" readonly></textarea>
</section> </section>
</div> </div>
</article> </article>
<article v-if="selectedSkill.type !== 'rules'" class="detail-card panel"> <article class="detail-card panel">
<div class="card-head"> <div class="card-head">
<div> <div>
<h3>{{ selectedSkill.outputTitle }}</h3> <h3>{{ selectedSkill.outputTitle }}</h3>
<p>{{ selectedSkill.outputDesc }}</p> <p>{{ selectedSkill.outputDesc }}</p>
</div> </div>
<button class="mini-btn primary">运行测试</button> <span class="edit-badge">
{{ selectedSkill.type === 'rules' ? selectedSkill.reviewStatusLabel : selectedSkill.publishState }}
</span>
</div> </div>
<div class="contract-grid"> <div class="contract-grid">
@@ -100,6 +191,7 @@
<li v-for="rule in selectedSkill.outputRules" :key="rule">{{ rule }}</li> <li v-for="rule in selectedSkill.outputRules" :key="rule">{{ rule }}</li>
</ul> </ul>
</div> </div>
<div class="contract-panel"> <div class="contract-panel">
<h4>{{ selectedSkill.checkListTitle }}</h4> <h4>{{ selectedSkill.checkListTitle }}</h4>
<div v-for="test in selectedSkill.tests" :key="test.name" class="test-row"> <div v-for="test in selectedSkill.tests" :key="test.name" class="test-row">
@@ -111,6 +203,40 @@
</div> </div>
</div> </div>
</div> </div>
<div v-if="selectedSkillIsRule" class="review-action-strip">
<button
class="minor-action"
type="button"
:disabled="!canManageSelected || detailBusy"
@click="reviewSelectedRule('pending')"
>
<i class="mdi mdi-send-outline"></i>
<span>{{ actionState === 'review-pending' ? '提交中...' : '提交审核' }}</span>
</button>
<button
class="minor-action success-action"
type="button"
:disabled="!canManageSelected || detailBusy"
@click="reviewSelectedRule('approved')"
>
<i class="mdi mdi-check-decagram-outline"></i>
<span>{{ actionState === 'review-approved' ? '处理中...' : '审核通过' }}</span>
</button>
<button
class="minor-action danger-action"
type="button"
:disabled="!canManageSelected || detailBusy"
@click="reviewSelectedRule('rejected')"
>
<i class="mdi mdi-close-octagon-outline"></i>
<span>{{ actionState === 'review-rejected' ? '处理中...' : '驳回版本' }}</span>
</button>
</div>
<p v-if="selectedSkillIsRule && activateBlockedReason" class="action-help">
{{ activateBlockedReason }}
</p>
</article> </article>
</section> </section>
@@ -119,29 +245,34 @@
<div class="card-head"> <div class="card-head">
<div> <div>
<h3>版本信息</h3> <h3>版本信息</h3>
<p>最近 5 个规则版本</p> <p>最近 5 个规则版本仅切换当前展示内容</p>
</div> </div>
</div> </div>
<div class="version-list"> <div v-if="selectedSkill.history.length" class="version-list">
<button <button
v-for="item in selectedSkill.history.slice(0, 5)" v-for="item in selectedSkill.history.slice(0, 5)"
:key="item.version + item.time" :key="item.version + item.time"
class="version-row" class="version-row"
:class="{ active: item.version === selectedSkill.version }" :class="{ active: item.version === selectedSkill.displayVersion }"
type="button" type="button"
@click="openVersionSwitch(item)" @click="openVersionSwitch(item)"
> >
<div class="version-row-head"> <div class="version-row-head">
<strong>{{ item.version }}</strong> <strong>{{ item.version }}</strong>
<span class="version-current-slot"> <span class="version-current-slot">
<b v-if="item.version === selectedSkill.version" class="current-version">当前</b> <b v-if="item.version === selectedSkill.currentVersion" class="current-version">当前</b>
</span> </span>
<span>{{ item.time }}</span> <span>{{ item.time }}</span>
</div> </div>
<p>{{ item.note }}</p> <p>{{ item.note }}</p>
</button> </button>
</div> </div>
<div v-else class="empty-side-note">
<i class="mdi mdi-history"></i>
<span>暂无版本历史</span>
</div>
</article> </article>
</aside> </aside>
@@ -165,8 +296,8 @@
<p>{{ selectedSkill.toolDesc }}</p> <p>{{ selectedSkill.toolDesc }}</p>
</div> </div>
</div> </div>
<div class="tool-list"> <div v-if="selectedSkill.tools.length" class="tool-list">
<div v-for="tool in selectedSkill.tools" :key="tool.name" class="tool-row"> <div v-for="tool in selectedSkill.tools" :key="tool.name + tool.scope" class="tool-row">
<div> <div>
<strong>{{ tool.name }}</strong> <strong>{{ tool.name }}</strong>
<span>{{ tool.scope }}</span> <span>{{ tool.scope }}</span>
@@ -174,6 +305,10 @@
<b :class="['tool-state', tool.tone]">{{ tool.mode }}</b> <b :class="['tool-state', tool.tone]">{{ tool.mode }}</b>
</div> </div>
</div> </div>
<div v-else class="empty-side-note">
<i class="mdi mdi-connection"></i>
<span>暂无依赖信息</span>
</div>
</article> </article>
<article class="side-card panel"> <article class="side-card panel">
@@ -183,13 +318,17 @@
<p>{{ selectedSkill.historyDesc }}</p> <p>{{ selectedSkill.historyDesc }}</p>
</div> </div>
</div> </div>
<div class="history-list"> <div v-if="selectedSkill.history.length" class="history-list">
<div v-for="item in selectedSkill.history" :key="item.version" class="history-row"> <div v-for="item in selectedSkill.history" :key="item.version + item.time" class="history-row">
<strong>{{ item.version }}</strong> <strong>{{ item.version }}</strong>
<span>{{ item.note }}</span> <span>{{ item.note }}</span>
<small>{{ item.time }}</small> <small>{{ item.time }}</small>
</div> </div>
</div> </div>
<div v-else class="empty-side-note">
<i class="mdi mdi-clock-outline"></i>
<span>暂无版本记录</span>
</div>
</article> </article>
<article class="side-card panel publish-card"> <article class="side-card panel publish-card">
@@ -207,25 +346,36 @@
</div> </div>
<footer class="detail-actions"> <footer class="detail-actions">
<button class="back-action" type="button" @click="selectedSkill = null"> <button class="back-action" type="button" @click="closeDetail">
<i class="mdi mdi-arrow-left"></i> <i class="mdi mdi-arrow-left"></i>
<span>返回能力列表</span> <span>返回能力列表</span>
</button> </button>
<div class="detail-action-group"> <div v-if="selectedSkillIsRule" class="detail-action-group">
<button class="minor-action" type="button"> <button
class="minor-action"
type="button"
:disabled="!canEditMarkdown || detailBusy"
@click="saveRuleMarkdown"
>
<i class="mdi mdi-content-save-outline"></i> <i class="mdi mdi-content-save-outline"></i>
<span>保存草稿</span> <span>{{ actionState === 'save-markdown' ? '保存中...' : '保存 Markdown' }}</span>
</button> </button>
<button class="minor-action" type="button"> <button
<i class="mdi mdi-flask-outline"></i> class="major-action"
<span>运行测试</span> type="button"
</button> :disabled="!canActivateSelected"
<button class="major-action" type="button"> :title="activateBlockedReason"
@click="activateSelectedRule"
>
<i class="mdi mdi-rocket-launch-outline"></i> <i class="mdi mdi-rocket-launch-outline"></i>
<span>正式上线</span> <span>{{ actionState === 'activate' ? '上线中...' : selectedSkill.statusValue === 'active' ? '已上线' : '正式上线' }}</span>
</button> </button>
</div> </div>
<div v-else class="detail-action-group detail-meta-actions">
<span class="footer-note">{{ selectedSkill.publishMeta }}</span>
</div>
</footer> </footer>
</article> </article>
@@ -252,21 +402,161 @@
:placeholder="searchPlaceholder" :placeholder="searchPlaceholder"
/> />
</label> </label>
<button v-for="filter in filters" :key="filter" type="button" class="filter-btn">
<span>{{ filter }}</span> <div class="picker-filter" :class="{ open: activeFilterPopover === 'domain' }">
<i class="mdi mdi-chevron-down"></i> <button
class="picker-trigger"
type="button"
:aria-expanded="activeFilterPopover === 'domain'"
aria-haspopup="dialog"
@click="toggleFilterPopover('domain')"
>
<span class="picker-label">{{ selectedDomainLabel }}</span>
<i class="mdi mdi-chevron-down"></i>
</button>
<div
v-if="activeFilterPopover === 'domain'"
class="picker-popover"
role="dialog"
aria-label="选择业务域"
>
<header>
<strong>选择业务域</strong>
<button type="button" aria-label="关闭业务域选择" @click="closeFilterPopover">
<i class="mdi mdi-close"></i>
</button>
</header>
<div class="picker-option-list">
<button
v-for="option in domainOptions"
:key="option.value || 'all-domain'"
type="button"
class="picker-option"
:class="{ active: selectedDomain === option.value }"
@click="selectFilter('domain', option.value)"
>
{{ option.label }}
</button>
</div>
</div>
</div>
<div class="picker-filter" :class="{ open: activeFilterPopover === 'owner' }">
<button
class="picker-trigger"
type="button"
:aria-expanded="activeFilterPopover === 'owner'"
aria-haspopup="dialog"
@click="toggleFilterPopover('owner')"
>
<span class="picker-label">{{ selectedOwnerLabel }}</span>
<i class="mdi mdi-chevron-down"></i>
</button>
<div
v-if="activeFilterPopover === 'owner'"
class="picker-popover"
role="dialog"
aria-label="选择负责人"
>
<header>
<strong>选择负责人</strong>
<button type="button" aria-label="关闭负责人选择" @click="closeFilterPopover">
<i class="mdi mdi-close"></i>
</button>
</header>
<div class="picker-option-list">
<button
v-for="option in ownerOptions"
:key="option.value || 'all-owner'"
type="button"
class="picker-option"
:class="{ active: selectedOwner === option.value }"
@click="selectFilter('owner', option.value)"
>
{{ option.label }}
</button>
</div>
</div>
</div>
<div class="picker-filter" :class="{ open: activeFilterPopover === 'status' }">
<button
class="picker-trigger"
type="button"
:aria-expanded="activeFilterPopover === 'status'"
aria-haspopup="dialog"
@click="toggleFilterPopover('status')"
>
<span class="picker-label">{{ selectedStatusLabel }}</span>
<i class="mdi mdi-chevron-down"></i>
</button>
<div
v-if="activeFilterPopover === 'status'"
class="picker-popover"
role="dialog"
aria-label="选择状态"
>
<header>
<strong>选择状态</strong>
<button type="button" aria-label="关闭状态选择" @click="closeFilterPopover">
<i class="mdi mdi-close"></i>
</button>
</header>
<div class="picker-option-list">
<button
v-for="option in statusOptions"
:key="option.value || 'all-status'"
type="button"
class="picker-option"
:class="{ active: selectedStatus === option.value }"
@click="selectFilter('status', option.value)"
>
{{ option.label }}
</button>
</div>
</div>
</div>
</div>
<div class="toolbar-actions">
<button v-if="activeFilterTokens.length" class="ghost-filter-btn" type="button" @click="resetFilters">
<i class="mdi mdi-filter-remove-outline"></i>
<span>清空筛选</span>
</button>
<button class="create-btn" type="button" disabled>
<i class="mdi mdi-plus"></i>
<span>{{ createButtonLabel }}</span>
</button> </button>
</div> </div>
<button class="create-btn" type="button">
<i class="mdi mdi-plus"></i>
<span>{{ createButtonLabel }}</span>
</button>
</div> </div>
<p class="hint"><i class="mdi mdi-information-outline"></i> {{ hintText }}</p> <p class="hint"><i class="mdi mdi-information-outline"></i> {{ hintText }}</p>
<div v-if="activeFilterTokens.length" class="active-filter-strip">
<span v-for="token in activeFilterTokens" :key="token" class="active-filter-chip">
{{ token }}
</span>
</div>
<div class="table-wrap"> <div class="table-wrap">
<table> <div v-if="loading" class="table-state">
<i class="mdi mdi-loading mdi-spin"></i>
<p>正在加载{{ activeTabLabel }}资产...</p>
</div>
<div v-else-if="errorMessage" class="table-state error">
<i class="mdi mdi-alert-circle-outline"></i>
<p>{{ errorMessage }}</p>
<button type="button" class="state-action" @click="loadAssets">重新加载</button>
</div>
<div v-else-if="!visibleSkills.length" class="table-state empty">
<i class="mdi mdi-database-search-outline"></i>
<p>没有匹配的资产数据</p>
</div>
<table v-else>
<thead> <thead>
<tr> <tr>
<th>{{ tableColumns.name }}</th> <th>{{ tableColumns.name }}</th>
@@ -286,7 +576,7 @@
v-for="skill in visibleSkills" v-for="skill in visibleSkills"
:key="skill.id" :key="skill.id"
:class="{ spotlight: skill.spotlight }" :class="{ spotlight: skill.spotlight }"
@click="selectedSkill = skill" @click="openAssetDetail(skill)"
> >
<td> <td>
<div class="skill-name-cell"> <div class="skill-name-cell">
@@ -306,14 +596,18 @@
<td>{{ skill.hitRate }}</td> <td>{{ skill.hitRate }}</td>
<td>{{ skill.updatedAt }}</td> <td>{{ skill.updatedAt }}</td>
<td> <td>
<button class="row-action" type="button" @click.stop="selectedSkill = skill"> <button class="row-action" type="button" @click.stop="openAssetDetail(skill)">
编辑 查看详情
</button> </button>
</td> </td>
</tr> </tr>
</tbody> </tbody>
</table> </table>
</div> </div>
<footer v-if="!loading && !errorMessage" class="list-foot">
<span class="page-summary">当前展示 {{ visibleSkills.length }} 条资产</span>
</footer>
</article> </article>
</Transition> </Transition>
@@ -323,14 +617,14 @@
<div class="card-head"> <div class="card-head">
<div> <div>
<h3 id="version-switch-title">切换规则版本</h3> <h3 id="version-switch-title">切换规则版本</h3>
<p>切换后编辑器会加载该版本的 .md 内容当前未保存内容不会自动发布</p> <p>切换后编辑器只会替换当前展示内容不会直接回滚后端当前版本</p>
</div> </div>
</div> </div>
<div class="version-modal-summary"> <div class="version-modal-summary">
<div> <div>
<span>当前版本</span> <span>当前展示版本</span>
<strong>{{ selectedSkill?.version }}</strong> <strong>{{ selectedSkill?.displayVersion }}</strong>
</div> </div>
<i class="mdi mdi-arrow-right"></i> <i class="mdi mdi-arrow-right"></i>
<div> <div>

File diff suppressed because it is too large Load Diff