fetch(modify):修改bug
All checks were successful
构建并部署前端到测试环境 / build-and-deploy (push) Successful in 5m45s

This commit is contained in:
sexygoat
2026-02-05 17:22:41 +08:00
parent d97dc5f007
commit b94c043a56
24 changed files with 2734 additions and 446 deletions

View File

@@ -0,0 +1,159 @@
<template>
<div class="detail-container">
<ElCard v-for="(section, index) in sections" :key="index" class="detail-section">
<template #header>
<div class="section-title">{{ section.title }}</div>
</template>
<div :class="['section-fields', section.columns === 1 ? 'single-column' : 'double-column']">
<div
v-for="(field, fieldIndex) in section.fields"
:key="fieldIndex"
class="field-item"
:class="{ 'full-width': field.fullWidth }"
>
<div class="field-label">{{ field.label }}:</div>
<div class="field-value">
<!-- 自定义渲染 -->
<template v-if="field.render">
<component :is="field.render(data)" />
</template>
<!-- 默认渲染 -->
<template v-else>
{{ formatFieldValue(field, data) }}
</template>
</div>
</div>
</div>
</ElCard>
</div>
</template>
<script setup lang="ts">
import { ElCard } from 'element-plus'
export interface DetailField {
label: string // 字段标签
prop?: string // 数据属性路径,支持点号分隔的嵌套路径,如 'one_time_commission_config.enable'
formatter?: (value: any, data: any) => string // 自定义格式化函数
render?: (data: any) => any // 自定义渲染函数,返回 VNode
fullWidth?: boolean // 是否占据整行
}
export interface DetailSection {
title: string // 分组标题
fields: DetailField[] // 字段列表
columns?: 1 | 2 // 列数默认2列
}
interface Props {
sections: DetailSection[] // 详情页分组配置
data: any // 详情数据
}
const props = defineProps<Props>()
/**
* 根据点号分隔的路径获取嵌套对象的值
*/
const getNestedValue = (obj: any, path: string): any => {
if (!path) return obj
return path.split('.').reduce((acc, part) => acc?.[part], obj)
}
/**
* 格式化字段值
*/
const formatFieldValue = (field: DetailField, data: any): string => {
const value = field.prop ? getNestedValue(data, field.prop) : data
// 如果有自定义格式化函数
if (field.formatter) {
return field.formatter(value, data)
}
// 默认处理
if (value === null || value === undefined || value === '') {
return '-'
}
return String(value)
}
</script>
<style scoped lang="scss">
.detail-container {
max-height: 70vh;
overflow-y: auto;
padding: 4px;
}
.detail-section {
margin-bottom: 20px;
&:last-child {
margin-bottom: 0;
}
.section-title {
font-size: 16px;
font-weight: 600;
color: var(--el-text-color-primary);
}
}
.section-fields {
display: grid;
gap: 16px 24px;
&.single-column {
grid-template-columns: 1fr;
}
&.double-column {
grid-template-columns: repeat(2, 1fr);
}
}
.field-item {
display: flex;
align-items: flex-start;
min-height: 32px;
&.full-width {
grid-column: 1 / -1;
}
.field-label {
flex-shrink: 0;
width: 140px;
font-weight: 500;
color: var(--el-text-color-regular);
line-height: 32px;
}
.field-value {
flex: 1;
color: var(--el-text-color-primary);
line-height: 32px;
word-break: break-word;
}
}
@media (max-width: 768px) {
.section-fields {
&.double-column {
grid-template-columns: 1fr;
}
}
.field-item {
flex-direction: column;
.field-label {
width: 100%;
margin-bottom: 4px;
}
}
}
</style>

View File

@@ -13,7 +13,7 @@
const props = withDefaults(
defineProps<{
text?: string
type?: 'add' | 'edit' | 'delete' | 'more'
type?: 'add' | 'edit' | 'delete' | 'more' | 'view'
icon?: string // 自定义图标
iconClass?: BgColorEnum // 自定义按钮背景色、文字颜色
iconColor?: string // 外部传入的图标文字颜色
@@ -31,7 +31,8 @@
{ type: 'add', icon: '&#xe602;', color: BgColorEnum.PRIMARY },
{ type: 'edit', icon: '&#xe642;', color: BgColorEnum.SECONDARY },
{ type: 'delete', icon: '&#xe783;', color: BgColorEnum.ERROR },
{ type: 'more', icon: '&#xe6df;', color: '' }
{ type: 'more', icon: '&#xe6df;', color: '' },
{ type: 'view', icon: '&#xe6df;', color: BgColorEnum.SECONDARY }
] as const
// 计算最终使用的图标:优先使用外部传入的 icon否则根据 type 获取默认图标