fix: 增强vue3动态头部代码生成

This commit is contained in:
wintel
2025-05-10 22:00:32 +08:00
parent 9c0fa9d16a
commit eb3d8bb4b3

View File

@@ -1,62 +1,82 @@
<!-- <!--
* @Author: yubaolee <yubaolee@163.com> | ahfu~ <954478625@qq.com> * @Description: 代码生成器,多表模版,动态头部
* @Description: 代码生成界面,动态头部 * @Copyright (c) 2025 by yubaolee | ahfu~ , All Rights Reserved.
* @
* @Copyright (c) 2024 by yubaolee | ahfu~ , All Rights Reserved.
--> -->
<template> <template>
<div class="flex-column flex"> <div class="flex flex-column">
<sticky> <sticky>
<div class="search-box"> <div class="search-box">
<el-input @keyup.enter="handleFilter" size="small" class="w-200 custom-input" :placeholder="'名称'" <div class="flex items-center">
v-model="firstQuery.key"></el-input> <query-builder storageName="{FirstTableName}s_query_conditions" :columns="firstHeaderList"
<el-button class="filter-item custom-button" icon="el-icon-search" size="small" @click="handleFilter"> :query-options="firstQuery" @search="handleFilter" class="flex-1" />
搜索 <column-setting storageName="{FirstTableName}s_column_settings" :columns="firstHeaderList"
</el-button> @update:columns="handleColumnsUpdate" class="ml-2" />
</div>
</div> </div>
<permission-btn v-on:btn-event="onBtnClicked"></permission-btn> <permission-btn v-on:btn-event="onBtnClicked"></permission-btn>
</sticky> </sticky>
<div class="flex-item flex flex-column main-context"> <div class="flex-item flex flex-column main-context">
<el-card shadow="never" class="custom-card flex-item flex flex-column"> <!-- 主表 -->
<auth-table size="small" ref="mainTable" :select-type="'radio'" :table-fields="firstHeaderList" :data="mainList" <div class="flex-item flex flex-column b-w" v-show="showNotFullScreen && !showDetailInTable">
:v-loading="listLoading" @row-click="rowClickFirstTable"></auth-table> <auth-table size="small" ref="mainTable" id="mainTable" :table-fields="firstHeaderList" :data="mainList" :v-loading="listLoading" @row-click="rowClickFirstTable"
</el-card> @row-dblclick="rowDblClickFirstTable"></auth-table>
<div class="flex flex-direction-r b-w p-r-10 p-t-10 p-b-10"> <div class="flex flex-direction-r p-r-10 b-w p-b-5 p-t-5">
<el-pagination v-show="firstTotal > 0" :total="firstTotal" v-model:currentPage="firstQuery.page" <el-pagination size="small" background v-show="firstTotal > 0" v-model:currentPage="firstQuery.page"
v-model:page-size="firstQuery.limit" @current-change="handleCurrentChange" /> v-model:page-size="firstQuery.limit" :page-sizes="[10, 20, 50]"
layout="total, sizes, prev, pager, next, jumper" :total="firstTotal" @size-change="handleSizeChange"
@current-change="handleCurrentChange" />
</div> </div>
<div class="flex-item flex"> </div>
<div v-if="showTitleDialog" class="flex flex-column" style="width: 500px;">
<el-card shadow="never" class="demo-card custom-card fh"> <!-- 主表表单(双击显示在主表位置) -->
<div class="flex-item flex flex-column b-w" v-show="showNotFullScreen && showDetailInTable">
<el-card shadow="never" class="custom-card flex-item flex flex-column">
<template #header> <template #header>
<div> <div class="flex flex-center">
<span v-if="radio == ''" class="f-12 f-b">表信息</span>
<span v-else class="f-12 f-b">{{ tableName }}表信息</span> <el-button size="small" class="custom-button" :icon="ArrowLeft" @click="returnToList">
返回列表
</el-button>
<el-button size="small" v-if="editModel" :icon="Close" class="custom-button" @click="cancelEdit">取消</el-button>
<el-button size="small" class="custom-button" :icon="Check" v-if="editModel && dialogStatus == 'create'" type="primary"
@click="createData">
确认
</el-button>
<el-button size="small" class="custom-button" :icon="Check" v-else-if="editModel" type="primary"
@click="updateData">确认</el-button>
</div> </div>
</template> </template>
<auth-form ref="dataForm" :rules="mainRules" :edit-model="editModel" :form-fields="firstHeaderList" <auth-form ref="dataForm" :rules="mainRules" :edit-model="editModel" :form-fields="firstHeaderList"
v-model="firstTemp" :data="firstTemp" :col-num="2"></auth-form> v-model="firstTemp" :data="firstTemp" :col-num="4"></auth-form>
</el-card> </el-card>
</div> </div>
<div class="flex-item flex flex-column">
<el-card shadow="never" class="custom-card flex-item flex flex-column" id="secondCard"> <!-- 明细列表 -->
<div class="flex-item flex">
<div class="flex flex-column flex-item b-w" id="secondCard">
<el-card shadow="never" class="custom-card flex-item flex flex-column">
<template #header> <template #header>
<div class="flex flex-center"> <div class="flex flex-center">
<el-icon v-if="showTitleDialog" @click="showTitleDialog = !showTitleDialog" :size="14">
<ArrowLeftBold />
</el-icon>
<el-icon v-else @click="showTitleDialog = !showTitleDialog" :size="14">
<ArrowRightBold />
</el-icon>
<span v-if="radio == ''" class="flex-item f-12 f-b"> <span v-if="radio == ''" class="flex-item f-12 f-b">
字段信息(修改后,编辑框内回车生效) 明细列表(修改后,编辑框内回车生效)
</span> </span>
<span v-else class="flex-item f-12 f-b"> <span v-else class="flex-item f-12 f-b">
{{ tableName }}表字段信息(修改后,编辑框内回车生效) {{ tableName }}明细列表(修改后,编辑框内回车生效)
</span> </span>
<el-button size="small" v-if="editModel" :icon="Plus" @click="onBtnClicked('btnAddDetail')">
添加
</el-button>
<el-button size="small" v-if="editModel" :icon="Delete" @click="onBtnClicked('btnDelDetail')"> <el-button size="small" v-if="editModel" :icon="Delete" @click="onBtnClicked('btnDelDetail')">
删除 删除
</el-button> </el-button>
<column-setting buttonClass="filter-item" storageName="{SecondTableName}s_column_settings"
:columns="secondHeaderList" @update:columns="updateSecondColumns" class="ml-2" />
<el-button size="small" :icon="showNotFullScreen ? FullScreen : Close"
@click="showNotFullScreen = !showNotFullScreen">
{{ showNotFullScreen ? '全屏' : '退出全屏' }}
</el-button>
</div> </div>
</template> </template>
<auth-table size="small" ref="secondTable" :editModel="editModel" :table-fields="secondHeaderList" <auth-table size="small" ref="secondTable" :editModel="editModel" :table-fields="secondHeaderList"
@@ -64,39 +84,39 @@
@item-change="handleUpdateDetail"></auth-table> @item-change="handleUpdateDetail"></auth-table>
</el-card> </el-card>
<div class="flex flex-direction-r p-r-10 b-w p-b-5 p-t-5"> <div class="flex flex-direction-r p-r-10 b-w p-b-5 p-t-5">
<el-pagination v-show="secondTotal > 0" :total="secondTotal" v-model:page="secondQuery.page" <el-pagination size="small" background v-show="secondTotal > 0" :currentPage="secondQuery.page"
v-model:limit="secondQuery.limit" @current-change="handleSecondPage" /> :page-size="secondQuery.limit" :total="secondTotal" @current-change="handleSecondPage" />
</div> </div>
</div> </div>
</div> </div>
<div v-if="editModel" class="p-t-5 p-b-5 t-r b-w p-r-10 border-t-2">
<el-button @click="editModel = false">取消</el-button>
<el-button v-if="dialogStatus == 'create'" type="primary" @click="createData">
确认
</el-button>
<el-button v-else type="primary" @click="updateData">确认</el-button>
</div>
</div> </div>
<print-preview ref="preView" />
</div> </div>
</template> </template>
<script setup> <script setup>
//引入核心框架 //引入核心框架
import { ElMessage, ElNotification } from 'element-plus' import { ElMessage, ElNotification } from 'element-plus'
import { Refresh, Delete } from '@element-plus/icons-vue' import { Plus, Delete, FullScreen, Close, Check, ArrowLeft } from '@element-plus/icons-vue'
import { onMounted, ref, reactive, nextTick, computed } from 'vue' import { onMounted, ref, reactive, nextTick, computed, defineComponent, watch } from 'vue'
import { mapGetters, useStore } from 'vuex' import { useStore } from 'vuex'
import { useRouter } from 'vue-router'
//引入API共用方法 //引入API共用方法
import * as {FirstTableName}s from '@/api/{FirstTableName}s' import * as {FirstTableName}s from '@/api/{FirstTableName}s'
import * as {SecondTableName}s from '@/api/{SecondTableName}s' import * as {SecondTableName}s from '@/api/{SecondTableName}s'
import ColumnDefine from '@/interface/columnDefine'
import { typeLists } from '@/utils/const.js'
import { parseTime, defaultVal } from '@/utils/index' import { parseTime, defaultVal } from '@/utils/index'
import { delrows } from '@/utils/delRows.js' import common from "@/extensions/common.js"
import { getItem } from '@/utils/storage'
//引入组件 //引入组件
import Sticky from '@/components/Sticky/index.vue' import Sticky from '@/components/Sticky/index.vue'
import permissionBtn from '@/components/PermissionBtn/index.vue' import permissionBtn from '@/components/PermissionBtn/index.vue'
import AuthForm from '../../components/Base/AuthForm.vue' import AuthForm from '../../components/Base/AuthForm.vue'
import AuthTable from '../../components/Base/AuthTable.vue' import AuthTable from '../../components/Base/AuthTable.vue'
import QueryBuilder from '@/components/QueryBuilder/index.vue'
import ColumnSetting from '@/components/ColumnSetting/index.vue'
const firstHeaderList = ref([]) //列表头 const firstHeaderList = ref([]) //列表头
const secondHeaderList = ref([]) //明细列表头 const secondHeaderList = ref([]) //明细列表头
const multipleSelection = ref([]) //明细中checkbox选中的值 const multipleSelection = ref([]) //明细中checkbox选中的值
@@ -110,8 +130,20 @@
const dialogStatus = ref('') //主修改对话框状态create/update const dialogStatus = ref('') //主修改对话框状态create/update
const radio = ref('') //主列表选项值 const radio = ref('') //主列表选项值
const tableName = ref('') const tableName = ref('')
let firstTemp = reactive({}) //当前选中的头信息 const showDetailInTable = ref(false) // 是否在主表位置显示详情
let selectRow = reactive({}) let firstTemp = reactive({}) //当前选中的主表项
let selectRow = reactive({}) //当前选中的主表项
//组件refs
const mainTable = ref(null)
const dataForm = ref(null)
const secondTable = ref(null)
const preView = ref(null)
const store = useStore()
const router = useRouter()
const defaultorgid = computed(() => store.getters.defaultorgid)
const { showPreview, printPreview, createFlowInstance } = common()
const firstQuery = reactive({ const firstQuery = reactive({
// 查询条件 // 查询条件
page: 1, page: 1,
@@ -124,88 +156,89 @@
limit: 20, limit: 20,
customerKey: undefined, customerKey: undefined,
}) })
const showTitleDialog = ref(true) //是否显示左下主列表详情值
const showNotFullScreen = ref(true) //明细列表是否全屏
watch(showNotFullScreen, () => {
secondQuery.limit = showNotFullScreen.value ? 10 : 20
querySecondList(radio.value)
})
const mainRules = reactive({ const mainRules = reactive({
Id: [ Id: [
{ {
required: true, required: true,
message: 'id不能为空' message: '入库通知单号不能为空',
trigger: 'change',
}, },
], ],
}) })
//组件refs
const mainTable = ref(null)
const dataForm = ref(null)
const secondTable = ref(null)
const store = useStore()
const defaultorgid = computed(() => store.getters.defaultorgid)
onMounted(() => { onMounted(() => {
initCfg()
getList() getList()
}) })
const initCfg = function () {
firstHeaderList.value = [
{FirstHeaderList}
]
secondHeaderList.value = [
{SecondHeaderList}
]
}
// ------------------------通用处理函数------------------------------------- // ------------------------通用处理函数-------------------------------------
const onBtnClicked = function (domId, callback) { const onBtnClicked = (domId, args, callback) => {
switch (domId) { switch (domId) {
case 'btnAdd': // 添加新记录 case 'btnAdd': // 添加
resetFirstTemp() resetFirstTemp()
secondList.value = [] secondList.value = []
dialogStatus.value = 'create' dialogStatus.value = 'create'
editModel.value = true editModel.value = true
tableName.value = '新建' tableName.value = '新建'
editType.value = 'add' editType.value = 'add'
showDetailInTable.value = true // 显示详情编辑界面
nextTick(() => { nextTick(() => {
dataForm.value.clearValidate() dataForm.value.clearValidate()
}) })
break break
case 'btnEdit': // 编辑 case 'btnEdit': // 编辑
Object.assign(firstTemp, selectRow) Object.assign(firstTemp, selectRow)
if (firstTemp.id === '') { if (firstTemp.id === '') {
editModel.value = false editModel.value = false
ElMessage({ ElMessage.error('请选择要修改的项')
message: '请选择要修改的项',
type: 'error',
})
return return
} }
dialogStatus.value = 'update' dialogStatus.value = 'update'
editModel.value = true editModel.value = true
editType.value = 'edit' editType.value = 'edit'
showDetailInTable.value = true // 编辑时显示详情界面
nextTick(() => { nextTick(() => {
dataForm.value.clearValidate() dataForm.value.clearValidate()
}) })
break break
case 'btnDel': // 删除主表 case 'btnDel': // 删除
if (firstTemp.id === '') { if (firstTemp.id === '') {
ElMessage({ ElMessage.error('请选择要删除的项')
message: '请选择要删除的项',
type: 'error',
})
return return
} }
handleFirstDel(firstTemp) handleFirstDel(firstTemp)
break break
case 'btnPrint':
if (firstTemp.id === '') {
ElMessage.error('请选择要打印的内容')
return
}
showPreview(preView.value, args, firstTemp);
break
case 'btnFlowscheme':
if (firstTemp.id === '') {
ElMessage.error('请选择要送审的内容')
return
}
createFlowInstance(args, firstTemp);
break
case 'btnAddDetail': // 添加明细行 case 'btnAddDetail': // 添加明细行
handleAddOrderDetail() handleAddOrderDetail()
break break
case 'btnDelDetail': // 删除明细行 case 'btnDelDetail': // 删除明细行
if (multipleSelection.value.length < 1) { if (multipleSelection.value.length < 1) {
ElMessage({ ElMessage.error('至少删除一个')
message: '至少删除一个',
type: 'error',
})
return return
} }
handleSecondDel(multipleSelection.value) handleSecondDel(multipleSelection.value)
break break
case 'btnExport': case 'btnExport': //导出
mainTable.value.exportExcel(`表结构${parseTime(new Date())}`, callback) mainTable.value.exportExcel(`表结构${parseTime(new Date())}`, callback)
break break
default: default:
@@ -214,6 +247,7 @@
} }
// ------------------------主数据列表处理------------------------------------ // ------------------------主数据列表处理------------------------------------
const getList = () => { const getList = () => {
listLoading.value = true listLoading.value = true
{FirstTableName}s.getList(firstQuery).then(response => { {FirstTableName}s.getList(firstQuery).then(response => {
@@ -224,6 +258,12 @@
item.columnName.substring(1) item.columnName.substring(1)
}) })
firstHeaderList.value = response.columnFields firstHeaderList.value = response.columnFields
// 尝试从存储中加载列设置
const savedSettings = getItem('{FirstTableName}s_column_settings')
if (savedSettings) {
// 将保存的设置合并到当前列表
firstHeaderList.value = savedSettings
}
mainList.value = response.data mainList.value = response.data
firstTotal.value = response.count firstTotal.value = response.count
if (firstTotal.value > 0) { if (firstTotal.value > 0) {
@@ -232,34 +272,86 @@
listLoading.value = false listLoading.value = false
}) })
} }
const rowClickFirstTable = function (row) {
// 处理列设置更新
const handleColumnsUpdate = (columns) => {
if (columns) {
// 更新列显示配置
firstHeaderList.value = columns
} else {
// 如果是null则重置回默认配置
getList()
}
}
const updateSecondColumns = (columns) => {
if (columns) {
// 更新列显示配置
secondHeaderList.value = columns
} else {
// 如果是null则重置回默认配置
querySecondList(radio.value)
}
}
const rowClickFirstTable = row => {
// 点击行 // 点击行
mainTable.value.clearSelection()
mainTable.value.toggleRowSelection(row)
radio.value = row.id radio.value = row.id
tableName.value = row.tableName tableName.value = row.id
secondQuery.page = 1 secondQuery.page = 1
secondQuery.limit = 10 secondQuery.limit = 10
querySecondList(radio.value) querySecondList(radio.value)
showTitleDetail(row) showTitleDetail(row)
} }
const handleFilter = function () {
const handleFilter = () => {
firstQuery.page = 1 firstQuery.page = 1
getList() getList()
} }
const handleSizeChange = function (val) { const handleSizeChange = val => {
firstQuery.limit = val firstQuery.limit = val
getList() getList()
} }
const handleCurrentChange = function (val) {
const handleCurrentChange = val => {
firstQuery.page = val firstQuery.page = val
getList() getList()
} }
const resetFirstTemp = function () { const resetFirstTemp = () => {
firstHeaderList.value.forEach(item => { firstHeaderList.value.forEach(item => {
firstTemp[item.columnName] = defaultVal(item.entityType) firstTemp[item.columnName] = defaultVal(item.entityType)
}) })
firstTemp.namespace = 'OpenAuth.Repository.Domain'
} }
const createData = () => {
// 保存提交
dataForm.value.validate(() => {
let tempData = Object.assign({}, firstTemp)
tempData = setDetails(tempData)
tempData.OrgId = defaultorgid.value
{FirstTableName}s.add(tempData).then(resp => {
if (resp.result != undefined) {
//如果服务器返回生成的ID
tempData.id = resp.result
}
mainList.value.unshift(tempData)
editModel.value = false
showDetailInTable.value = false // 保存后返回列表视图
rowClickFirstTable(tempData)
ElNotification.success('创建成功')
})
})
}
const showTitleDetail = row => {
// 弹出编辑框
Object.assign(selectRow, row) // 新增订单时保存当前选中行
Object.assign(firstTemp, row) // copy obj
nextTick(() => {
dataForm.value.clearValidate()
})
}
const setDetails = tempData => { const setDetails = tempData => {
// 处理明细 // 处理明细
tempData.{SecondTableName}Reqs = [] tempData.{SecondTableName}Reqs = []
@@ -276,36 +368,7 @@
}) })
return tempData return tempData
} }
const updateData = () => {
const createData = function () {
// 保存提交
dataForm.value.validate(() => {
let tempData = Object.assign({}, firstTemp)
tempData = setDetails(tempData)
tempData.OrgId = defaultorgid.value
{FirstTableName}s.add(tempData).then(resp => {
tempData.id = resp.result
mainList.value.unshift(tempData)
editModel.value = false
rowClickFirstTable(tempData)
ElNotification({
title: '成功',
message: '创建成功',
type: 'success',
duration: 2000,
})
})
})
}
const showTitleDetail = function (row) {
// 弹出编辑框
Object.assign(selectRow, row) // 新增订单时保存当前选中行
Object.assign(firstTemp, row) // copy obj
nextTick(() => {
dataForm.value.clearValidate()
})
}
const updateData = function () {
// 更新提交 // 更新提交
dataForm.value.validate(() => { dataForm.value.validate(() => {
let tempData = Object.assign({}, firstTemp) let tempData = Object.assign({}, firstTemp)
@@ -319,24 +382,15 @@
} }
} }
editModel.value = false editModel.value = false
ElNotification({ showDetailInTable.value = false // 保存后返回列表视图
title: '成功', ElNotification.success('更新成功')
message: '更新成功',
type: 'success',
duration: 2000,
})
}) })
}) })
} }
const handleFirstDel = function (row) { const handleFirstDel = row => {
// 删除头 // 主表删除处理
{FirstTableName}s.del([row.id]).then(() => { {FirstTableName}s.del([row.id]).then(() => {
ElNotification({ ElNotification.success('删除成功')
title: '成功',
message: '删除成功',
type: 'success',
duration: 2000,
})
mainList.value = mainList.value.filter(item => item.id !== row.id) mainList.value = mainList.value.filter(item => item.id !== row.id)
if (mainList.value.length > 0) { if (mainList.value.length > 0) {
nextTick(() => { nextTick(() => {
@@ -348,11 +402,33 @@
showTitleDetail({}) showTitleDetail({})
}) })
} }
//返回列表
const returnToList = () => {
showDetailInTable.value = false
cancelEdit()
}
// 取消编辑
const cancelEdit = () => {
editModel.value = false
if (dialogStatus.value === 'create') {
showDetailInTable.value = false
}
}
// 主列表双击行事件
const rowDblClickFirstTable = row => {
rowClickFirstTable(row) // 先执行单击行的逻辑
showDetailInTable.value = true // 在主表位置显示详情
}
// ------------------------明细列表处理------------------------------------- // ------------------------明细列表处理-------------------------------------
const handleSecondPage = function (e) { const handleSecondPage = e => {
secondQuery.page = e secondQuery.page = e
querySecondList(radio.value) querySecondList(radio.value)
} }
const querySecondList = id => { const querySecondList = id => {
{SecondTableName}s {SecondTableName}s
.getList({ .getList({
@@ -369,11 +445,17 @@
item.columnName.substring(1) item.columnName.substring(1)
}) })
secondHeaderList.value = res.columnFields secondHeaderList.value = res.columnFields
// 尝试从存储中加载列设置
const savedSettings = getItem('{SecondTableName}s_column_settings')
if (savedSettings) {
// 将保存的设置合并到当前列表
secondHeaderList.value = savedSettings
}
secondTotal.value = res.count secondTotal.value = res.count
secondList.value = res.data secondList.value = res.data
}) })
} }
const rowClickSecondTable = function (row) { const rowClickSecondTable = row => {
// 行点击事件 // 行点击事件
secondTable.value.clearSelection() secondTable.value.clearSelection()
secondTable.value.toggleRowSelection(row) secondTable.value.toggleRowSelection(row)
@@ -383,27 +465,27 @@
rows.forEach(row => { rows.forEach(row => {
secondList.value = secondList.value.filter(item => item.id !== row.id) secondList.value = secondList.value.filter(item => item.id !== row.id)
}) })
ElNotification({ ElNotification.success('删除成功')
title: '成功',
message: '删除成功',
type: 'success',
duration: 2000,
})
}) })
} }
const selChangeSecondTable = function (val) { const selChangeSecondTable = val => {
// 明细选中事件 // 明细选中事件
multipleSelection.value = val multipleSelection.value = val
} }
const handleUpdateDetail = function (item) { const handleAddOrderDetail = () => {
// 同步表数据结构 // 添加明细
{SecondTableName}s.update(item).then(() => { const obj = {}
ElNotification({ secondHeaderList.value.forEach(header => {
title: '成功', obj[header.columnName] = defaultVal(header.entityType)
message: '更新成功',
type: 'success',
duration: 2000,
}) })
obj.orderId = firstTemp.id
secondList.value.push(Object.assign({}, obj))
}
const handleUpdateDetail = item => {
// 回车保存明细,如果不需要可以不用
{SecondTableName}s.update(item).then(() => {
ElNotification.success('成功')
}) })
} }
</script> </script>