Browse Source

代码生成模板

CzRger 8 months ago
parent
commit
9122eac08f

+ 20 - 3
snowy-admin-web/src/views/gen/basic.vue

@@ -173,6 +173,11 @@
 						<a-input-number class="xn-wd" v-model:value="formData.sortCode" :max="100" />
 					</a-form-item>
 				</a-col>
+				<a-col :span="8">
+					<a-form-item label="表单模式:" name="formMode">
+						<a-radio-group v-model:value="formData.formMode" :options="formModeOptions" />
+					</a-form-item>
+				</a-col>
 			</a-row>
 		</a-form>
 	</a-card>
@@ -232,6 +237,16 @@
 			value: 'N'
 		}
 	])
+	const formModeOptions = ref([
+		{
+			label: '弹窗',
+			value: 'dialog'
+		},
+		{
+			label: '内嵌',
+			value: 'inside'
+		}
+	])
 	// 打开抽屉
 	const onOpen = (record) => {
 		// 加载默认的模块
@@ -274,10 +289,11 @@
 					packageName: 'vip.xiaonuo',
 					moduleName: 'biz',
 					sortCode: 99,
-					tablePrefix: 'Y',
-					generateType: 'ZIP',
+					tablePrefix: 'N',
+					generateType: 'PRO',
 					formLayout: 'vertical',
-					gridWhether: 'N'
+					gridWhether: 'Y',
+					formMode: 'inside',
 				}
 			}
 		})
@@ -314,6 +330,7 @@
 		packageName: [required('请输入包名')],
 		sortCode: [required('请选择排序')],
 		formLayout: [required('请选择表单布局')],
+		formMode: [required('请选择表单模式')],
 		gridWhether: [required('请选择是否使用栅格')],
 		authorName: [required('请输入作者名')]
 	}

+ 4 - 0
snowy-plugin/snowy-plugin-gen/src/main/java/vip/xiaonuo/gen/modular/basic/entity/GenBasic.java

@@ -103,4 +103,8 @@ public class GenBasic extends CommonEntity {
     /** 作者 */
     @Schema(description = "作者")
     private String authorName;
+
+    /** 表单模式 */
+    @Schema(description = "表单模式")
+    private String formMode;
 }

+ 16 - 2
snowy-plugin/snowy-plugin-gen/src/main/java/vip/xiaonuo/gen/modular/basic/service/impl/GenBasicServiceImpl.java

@@ -98,8 +98,12 @@ public class GenBasicServiceImpl extends ServiceImpl<GenBasicMapper, GenBasic> i
 
     private static final List<JSONObject> GEN_FRONT_FILE_LIST = CollectionUtil.newArrayList(
             JSONUtil.createObj().set("name", "Api.js.btl").set("path", "api"),
+            // formMode:dialog
             JSONUtil.createObj().set("name", "form.vue.btl").set("path",  "views"),
-            JSONUtil.createObj().set("name", "index.vue.btl").set("path",  "views"));
+            JSONUtil.createObj().set("name", "index.vue.btl").set("path",  "views"),
+            // formMode:inside
+            JSONUtil.createObj().set("name", "detail.vue.btl").set("path",  "views"),
+            JSONUtil.createObj().set("name", "index_inside.vue.btl").set("path",  "views"));
 
     private static final List<JSONObject> GEN_MOBILE_FILE_LIST = CollectionUtil.newArrayList(
             JSONUtil.createObj().set("name", "page.json.btl"),
@@ -515,11 +519,21 @@ public class GenBasicServiceImpl extends ServiceImpl<GenBasicMapper, GenBasic> i
                     genBasicCodeFrontResult.setCodeFileName(resultName);
                     genBasicCodeFrontResult.setCodeFileWithPathName(genFrontBasicPath + fileTemplatePath + File.separator + resultName);
                 } else {
+                    if ("index_inside.vue".equalsIgnoreCase(resultName)) {
+                        resultName = "index.vue";
+                    }
                     genBasicCodeFrontResult.setCodeFileName(resultName);
                     genBasicCodeFrontResult.setCodeFileWithPathName(genFrontBasicPath + fileTemplatePath + File.separator + genBasic.getBusName() + File.separator + resultName);
                 }
                 genBasicCodeFrontResult.setCodeFileContent(templateFront.render());
-                genBasicCodeFrontendResultList.add(genBasicCodeFrontResult);
+                if (!(
+                    "dialog".equals(genBasic.getFormMode()) && "index_inside.vue.btl".equalsIgnoreCase(fileTemplateName) ||
+                    "dialog".equals(genBasic.getFormMode()) && "detail.vue.btl".equalsIgnoreCase(fileTemplateName) ||
+                    "inside".equals(genBasic.getFormMode()) && "index.vue.btl".equalsIgnoreCase(fileTemplateName) ||
+                    "inside".equals(genBasic.getFormMode()) && "form.vue.btl".equalsIgnoreCase(fileTemplateName)
+                )) {
+                    genBasicCodeFrontendResultList.add(genBasicCodeFrontResult);
+                }
             });
             genBasicPreviewResult.setGenBasicCodeFrontendResultList(genBasicCodeFrontendResultList);
 

+ 184 - 0
snowy-plugin/snowy-plugin-gen/src/main/resources/frontend/detail.vue.btl

@@ -0,0 +1,184 @@
+<template>
+    <a-card :bordered="false">
+		<a-page-header
+			:title="formData.${configList[i].fieldNameCamelCase} ? '编辑${functionName}' : '增加${functionName}'"
+			@back="onClose"
+		>
+			<template #extra>
+				<a-button style="margin-right: 8px" @click="onClose">关闭</a-button>
+				<a-button type="primary" @click="onSubmit" :loading="submitLoading">保存</a-button>
+			</template>
+		</a-page-header>
+		<a-form ref="formRef" :model="formData" :rules="formRules" layout="${formLayout}">
+        		<% if(gridWhether) { %>
+        			<a-row :gutter="16">
+        			<% for(var i = 0; i < configList.~size; i++) { %>
+        			<% if(!configList[i].needTableId && configList[i].whetherAddUpdate) { %>
+        				<a-col :span="12">
+        					<a-form-item label="${configList[i].fieldRemark}:" name="${configList[i].fieldNameCamelCase}">
+        						<% if(configList[i].effectType == 'input') { %>
+        						<a-input v-model:value="formData.${configList[i].fieldNameCamelCase}" placeholder="请输入${configList[i].fieldRemark}" allow-clear />
+        						<% } else if (configList[i].effectType == 'textarea') {%>
+        						<a-textarea v-model:value="formData.${configList[i].fieldNameCamelCase}" placeholder="请输入${configList[i].fieldRemark}" :auto-size="{ minRows: 3, maxRows: 5 }" />
+        						<% } else if (configList[i].effectType == 'select') {%>
+        						<a-select v-model:value="formData.${configList[i].fieldNameCamelCase}" placeholder="请选择${configList[i].fieldRemark}" :options="${configList[i].fieldNameCamelCase}Options" />
+        						<% } else if (configList[i].effectType == 'radio') {%>
+        						<a-radio-group v-model:value="formData.${configList[i].fieldNameCamelCase}" placeholder="请选择${configList[i].fieldRemark}" :options="${configList[i].fieldNameCamelCase}Options" />
+        						<% } else if (configList[i].effectType == 'checkbox') {%>
+        						<a-checkbox-group v-model:value="formData.${configList[i].fieldNameCamelCase}" placeholder="请选择${configList[i].fieldRemark}" :options="${configList[i].fieldNameCamelCase}Options" />
+        						<% } else if (configList[i].effectType == 'datepicker') {%>
+        						<a-date-picker v-model:value="formData.${configList[i].fieldNameCamelCase}" value-format="YYYY-MM-DD HH:mm:ss" show-time placeholder="请选择${configList[i].fieldRemark}" style="width: 100%" />
+        						<% } else if (configList[i].effectType == 'timepicker') {%>
+        						<a-time-picker v-model:value="formData.${configList[i].fieldNameCamelCase}" placeholder="请选择${configList[i].fieldRemark}" style="width: 100%" />
+        						<% } else if (configList[i].effectType == 'inputNumber') {%>
+        						<a-input-number v-model:value="formData.${configList[i].fieldNameCamelCase}" :min="1" :max="10000" style="width: 100%" />
+        						<% } else if (configList[i].effectType == 'slider') {%>
+        						<a-slider v-model:value="formData.${configList[i].fieldNameCamelCase}" :max="1000" style="width: 100%" />
+        						<% } else if (configList[i].effectType == 'fileUpload') {%>
+        						<xn-upload v-model:value="formData.${configList[i].fieldNameCamelCase}" />
+        						<% } else if (configList[i].effectType == 'imageUpload') {%>
+        						<xn-upload v-model:value="formData.${configList[i].fieldNameCamelCase}" uploadMode="image" />
+        						<% } else if (configList[i].effectType == 'editor') {%>
+        						<xn-editor v-model:value="formData.${configList[i].fieldNameCamelCase}" placeholder="请输入${configList[i].fieldRemark}" />
+        						<% } %>
+        					</a-form-item>
+        				</a-col>
+        			<% } %>
+        			<% } %>
+        			</a-row>
+        		<% } else { %>
+        		<% for(var i = 0; i < configList.~size; i++) { %>
+        		<% if(!configList[i].needTableId && configList[i].whetherAddUpdate) { %>
+        			<a-form-item label="${configList[i].fieldRemark}:" name="${configList[i].fieldNameCamelCase}">
+        				<% if(configList[i].effectType == 'input') { %>
+        				<a-input v-model:value="formData.${configList[i].fieldNameCamelCase}" placeholder="请输入${configList[i].fieldRemark}" allow-clear />
+        				<% } else if (configList[i].effectType == 'textarea') {%>
+        				<a-textarea v-model:value="formData.${configList[i].fieldNameCamelCase}" placeholder="请输入${configList[i].fieldRemark}" :auto-size="{ minRows: 3, maxRows: 5 }" />
+        				<% } else if (configList[i].effectType == 'select') {%>
+        				<a-select v-model:value="formData.${configList[i].fieldNameCamelCase}" placeholder="请选择${configList[i].fieldRemark}" :options="${configList[i].fieldNameCamelCase}Options" />
+        				<% } else if (configList[i].effectType == 'radio') {%>
+        				<a-radio-group v-model:value="formData.${configList[i].fieldNameCamelCase}" placeholder="请选择${configList[i].fieldRemark}" :options="${configList[i].fieldNameCamelCase}Options" />
+        				<% } else if (configList[i].effectType == 'checkbox') {%>
+        				<a-checkbox-group v-model:value="formData.${configList[i].fieldNameCamelCase}" placeholder="请选择${configList[i].fieldRemark}" :options="${configList[i].fieldNameCamelCase}Options" />
+        				<% } else if (configList[i].effectType == 'datepicker') {%>
+        				<a-date-picker v-model:value="formData.${configList[i].fieldNameCamelCase}" value-format="YYYY-MM-DD HH:mm:ss" show-time placeholder="请选择${configList[i].fieldRemark}" style="width: 100%" />
+        				<% } else if (configList[i].effectType == 'timepicker') {%>
+        				<a-time-picker v-model:value="formData.${configList[i].fieldNameCamelCase}" value-format="YYYY-MM-DD HH:mm:ss" show-time placeholder="请选择${configList[i].fieldRemark}" style="width: 100%" />
+        				<% } else if (configList[i].effectType == 'inputNumber') {%>
+        				<a-input-number v-model:value="formData.${configList[i].fieldNameCamelCase}" placeholder="请输入${configList[i].fieldRemark}" :min="1" :max="10000" style="width: 100%" />
+        				<% } else if (configList[i].effectType == 'slider') {%>
+        				<a-slider v-model:value="formData.${configList[i].fieldNameCamelCase}" placeholder="请滑动${configList[i].fieldRemark}" :max="1000" style="width: 100%" />
+        				<% } else if (configList[i].effectType == 'fileUpload') {%>
+        				<xn-upload v-model:value="formData.${configList[i].fieldNameCamelCase}" />
+        				<% } else if (configList[i].effectType == 'imageUpload') {%>
+        				<xn-upload v-model:value="formData.${configList[i].fieldNameCamelCase}" uploadMode="image" />
+        				<% } else if (configList[i].effectType == 'editor') {%>
+        				<xn-editor v-model:value="formData.${configList[i].fieldNameCamelCase}" placeholder="请输入${configList[i].fieldRemark}" />
+        				<% } %>
+        			</a-form-item>
+        		<% } %>
+        		<% } %>
+        		<% } %>
+        		</a-form>
+	</a-card>
+</template>
+
+<script setup name="${classNameFirstLower}Detail">
+	<%
+	var iptTool = 0;
+	for(var i = 0; i < configList.~size; i++) {
+		if(!configList[i].needTableId) {
+		if(configList[i].effectType == 'select' || configList[i].effectType == 'radio' || configList[i].effectType == 'checkbox') {
+			iptTool++;
+		}
+		}
+	}
+	%>
+	<% if(iptTool > 0) { %>
+	import tool from '@/utils/tool'
+	<% } %>
+	import { cloneDeep } from 'lodash-es'
+	import { required } from '@/utils/formRules'
+	import ${classNameFirstLower}Api from '@/api/${moduleName}/${classNameFirstLower}Api'
+	// 抽屉状态
+	const emit = defineEmits({ successful: null })
+	const formRef = ref()
+	// 表单数据
+	const formData = ref({})
+	const submitLoading = ref(false)
+	<% for(var i = 0; i < configList.~size; i++) { %>
+	<% if(!configList[i].needTableId) { %>
+	<% if(configList[i].effectType == 'select' || configList[i].effectType == 'radio' || configList[i].effectType == 'checkbox') { %>
+	const ${configList[i].fieldNameCamelCase}Options = ref([])
+	<% } %>
+	<% } %>
+	<% } %>
+
+	// 打开抽屉
+	const onOpen = (record) => {
+		if (record) {
+			let recordData = cloneDeep(record)
+			<% for(var i = 0; i < configList.~size; i++) { %>
+			<% if(!configList[i].needTableId && configList[i].whetherAddUpdate && configList[i].effectType == 'checkbox') { %>
+			recordData.${configList[i].fieldNameCamelCase} = JSON.parse(recordData.${configList[i].fieldNameCamelCase})
+			<% } %>
+			<% } %>
+			formData.value = Object.assign({}, recordData)
+		}
+		<% for(var i = 0; i < configList.~size; i++) { %>
+		<% if(!configList[i].needTableId) { %>
+		<% if(configList[i].effectType == 'select' || configList[i].effectType == 'radio' || configList[i].effectType == 'checkbox') { %>
+		${configList[i].fieldNameCamelCase}Options.value = tool.dictList('${configList[i].dictTypeCode}')
+		<% } %>
+		<% } %>
+		<% } %>
+	}
+	// 关闭抽屉
+	const onClose = () => {
+		formRef.value.resetFields()
+		formData.value = {}
+		emit('onClose')
+	}
+	// 默认要校验的
+	const formRules = {
+		<% for(var i = 0; i < configList.~size; i++) { %>
+		<% if(!configList[i].needTableId) { %>
+		<% if(configList[i].required) { %>
+		${configList[i].fieldNameCamelCase}: [required('请输入${configList[i].fieldRemark}')],
+		<% } %>
+		<% } %>
+		<% } %>
+	}
+	// 验证并提交数据
+	const onSubmit = () => {
+		formRef.value
+			.validate()
+			.then(() => {
+				submitLoading.value = true
+				const formDataParam = cloneDeep(formData.value)
+				<% for(var i = 0; i < configList.~size; i++) { %>
+				<% if(configList[i].whetherAddUpdate && configList[i].effectType == 'checkbox') { %>
+				formDataParam.${configList[i].fieldNameCamelCase} = JSON.stringify(formDataParam.${configList[i].fieldNameCamelCase})
+				<% } %>
+				<% } %>
+				${classNameFirstLower}Api
+					<% for(var i = 0; i < configList.~size; i++) { %>
+					<% if(configList[i].needTableId) { %>
+					.${classNameFirstLower}SubmitForm(formDataParam, formDataParam.${configList[i].fieldNameCamelCase})
+					<% } %>
+					<% } %>
+					.then(() => {
+						onClose()
+						emit('successful')
+					})
+					.finally(() => {
+						submitLoading.value = false
+					})
+			})
+			.catch(() => {})
+	}
+	// 抛出函数
+	defineExpose({
+		onOpen
+	})
+</script>

+ 281 - 0
snowy-plugin/snowy-plugin-gen/src/main/resources/frontend/index_inside.vue.btl

@@ -0,0 +1,281 @@
+<template>
+	<%
+	var searchCount = 0;
+	var row = 0;
+	%>
+	<% for(var i = 0; i < configList.~size; i++) { %>
+		<% if(!configList[i].needTableId && configList[i].needPage) { searchCount ++; }%>
+	<% } %>
+	<a-card :bordered="false" v-if="indexShow">
+		<% if (searchCount > 0) { %>
+		<a-form ref="searchFormRef" name="advanced_search" :model="searchFormState" class="ant-advanced-search-form">
+			<a-row :gutter="24">
+				<% for(var i = 0; i < configList.~size; i++) { %>
+				<% if(!configList[i].needTableId && configList[i].needPage) { row ++; %>
+				<% if(row <= 3) { %>
+				<a-col :span="6">
+				<% if(configList[i].effectType == 'input' || configList[i].effectType == 'textarea') { %>
+					<a-form-item label="${configList[i].fieldRemark}" name="${configList[i].fieldNameCamelCase}">
+						<a-input v-model:value="searchFormState.${configList[i].fieldNameCamelCase}" placeholder="请输入${configList[i].fieldRemark}" />
+					</a-form-item>
+				<% } else if (configList[i].effectType == 'select' || configList[i].effectType == 'radio' || configList[i].effectType == 'checkbox') {%>
+					<a-form-item label="${configList[i].fieldRemark}" name="${configList[i].fieldNameCamelCase}">
+						<a-select v-model:value="searchFormState.${configList[i].fieldNameCamelCase}" placeholder="请选择${configList[i].fieldRemark}" :options="${configList[i].fieldNameCamelCase}Options" />
+					</a-form-item>
+				<% } else if (configList[i].effectType == 'inputNumber' || configList[i].effectType == 'slider') {%>
+					<a-form-item label="${configList[i].fieldRemark}" name="${configList[i].fieldNameCamelCase}">
+						<a-input-number v-model:value="searchFormState.${configList[i].fieldNameCamelCase}" placeholder="请输入${configList[i].fieldRemark}" style="width: 100%" />
+					</a-form-item>
+				<% } else if (configList[i].effectType == 'datepicker') {%>
+					<a-form-item label="${configList[i].fieldRemark}" name="${configList[i].fieldNameCamelCase}">
+						<a-range-picker v-model:value="searchFormState.${configList[i].fieldNameCamelCase}" value-format="YYYY-MM-DD HH:mm:ss" show-time />
+					</a-form-item>
+				<% } else {%>
+					<a-form-item label="${configList[i].fieldRemark}" name="${configList[i].fieldNameCamelCase}">
+						<a-input v-model:value="searchFormState.${configList[i].fieldNameCamelCase}" placeholder="请输入${configList[i].fieldRemark}" />
+					</a-form-item>
+				<% } %>
+				</a-col>
+				<% } else { %>
+				<a-col :span="6" v-show="advanced">
+				<% if(configList[i].effectType == 'input' || configList[i].effectType == 'textarea') { %>
+					<a-form-item label="${configList[i].fieldRemark}" name="${configList[i].fieldNameCamelCase}">
+						<a-input v-model:value="searchFormState.${configList[i].fieldNameCamelCase}" placeholder="请输入${configList[i].fieldRemark}" />
+					</a-form-item>
+				<% } else if (configList[i].effectType == 'select' || configList[i].effectType == 'radio' || configList[i].effectType == 'checkbox') {%>
+					<a-form-item label="${configList[i].fieldRemark}" name="${configList[i].fieldNameCamelCase}">
+						<a-select v-model:value="searchFormState.${configList[i].fieldNameCamelCase}" placeholder="请选择${configList[i].fieldRemark}" :options="${configList[i].fieldNameCamelCase}Options" />
+					</a-form-item>
+				<% } else if (configList[i].effectType == 'inputNumber' || configList[i].effectType == 'slider') {%>
+					<a-form-item label="${configList[i].fieldRemark}" name="${configList[i].fieldNameCamelCase}">
+						<a-input-number v-model:value="searchFormState.${configList[i].fieldNameCamelCase}" placeholder="请输入${configList[i].fieldRemark}" style="width: 100%" />
+					</a-form-item>
+				<% } else if (configList[i].effectType == 'datepicker') {%>
+					<a-form-item label="${configList[i].fieldRemark}" name="${configList[i].fieldNameCamelCase}">
+						<a-range-picker v-model:value="searchFormState.${configList[i].fieldNameCamelCase}" show-time />
+					</a-form-item>
+				<% } else {%>
+					<a-form-item label="${configList[i].fieldRemark}" name="${configList[i].fieldNameCamelCase}">
+						<a-input v-model:value="searchFormState.${configList[i].fieldNameCamelCase}" placeholder="请输入${configList[i].fieldRemark}" />
+					</a-form-item>
+				<% } %>
+				</a-col>
+				<% } %>
+				<% } %>
+				<% } %>
+				<a-col :span="6">
+					<a-button type="primary" @click="tableRef.refresh()">查询</a-button>
+					<a-button style="margin: 0 8px" @click="reset">重置</a-button>
+					<% if(searchCount > 3) { %>
+					<a @click="toggleAdvanced" style="margin-left: 8px">
+						{{ advanced ? '收起' : '展开' }}
+						<component :is="advanced ? 'up-outlined' : 'down-outlined'"/>
+					</a>
+					<% }%>
+				</a-col>
+			</a-row>
+		</a-form>
+		<% } %>
+		<s-table
+			ref="tableRef"
+			:columns="columns"
+			:data="loadData"
+			:alert="options.alert.show"
+			bordered
+			<% for(var i = 0; i < configList.~size; i++) { %>
+			<% if(configList[i].needTableId) { %>
+			:row-key="(record) => record.${configList[i].fieldNameCamelCase}"
+			<% } %>
+			<% } %>
+			:tool-config="toolConfig"
+			:row-selection="options.rowSelection"
+		>
+			<template #operator class="table-operator">
+				<a-space>
+					<a-button type="primary" @click="onDetail()" v-if="hasPerm('${classNameFirstLower}Add')">
+						<template #icon><plus-outlined /></template>
+						新增
+					</a-button>
+					<xn-batch-delete
+						v-if="hasPerm('${classNameFirstLower}BatchDelete')"
+						:selectedRowKeys="selectedRowKeys"
+						@batchDelete="deleteBatch${className}"
+					/>
+				</a-space>
+			</template>
+			<template #bodyCell="{ column, record }">
+				<% for(var i = 0; i < configList.~size; i++) { %>
+				<% if(!configList[i].needTableId && configList[i].whetherTable) { %>
+				<% if (configList[i].effectType == 'select' || configList[i].effectType == 'radio') { %>
+				<template v-if="column.dataIndex === '${configList[i].fieldNameCamelCase}'">
+					{{ $TOOL.dictTypeData('${configList[i].dictTypeCode}', record.${configList[i].fieldNameCamelCase}) }}
+				</template>
+				<% } else if (configList[i].effectType == 'checkbox') { %>
+				<template v-if="column.dataIndex === '${configList[i].fieldNameCamelCase}'">
+					<a-tag v-for="textValue in JSON.parse(record.${configList[i].fieldNameCamelCase})" :key="textValue" color="green">{{ $TOOL.dictTypeData('${configList[i].dictTypeCode}', textValue) }}</a-tag>
+				</template>
+				<% } else if (configList[i].effectType == 'imageUpload') { %>
+				<template v-if="column.dataIndex === '${configList[i].fieldNameCamelCase}'">
+					<a-image :src="record.${configList[i].fieldNameCamelCase}" style="width: 30px; height: 30px;" />
+				</template>
+				<% } else if (configList[i].effectType == 'fileUpload') { %>
+				<template v-if="column.dataIndex === '${configList[i].fieldNameCamelCase}'">
+					<a :href="record.${configList[i].fieldNameCamelCase}" :target="record.${configList[i].fieldNameCamelCase}">{{ record.${configList[i].fieldNameCamelCase} }}</a>
+				</template>
+				<% } %>
+				<% } %>
+				<% } %>
+				<template v-if="column.dataIndex === 'action'">
+					<a-space>
+						<a @click="onDetail(record)" v-if="hasPerm('${classNameFirstLower}Edit')">编辑</a>
+						<a-divider type="vertical" v-if="hasPerm(['${classNameFirstLower}Edit', '${classNameFirstLower}Delete'], 'and')" />
+						<a-popconfirm title="确定要删除吗?" @confirm="delete${className}(record)">
+							<a-button type="link" danger size="small" v-if="hasPerm('${classNameFirstLower}Delete')">删除</a-button>
+						</a-popconfirm>
+					</a-space>
+				</template>
+			</template>
+		</s-table>
+	</a-card>
+	<Detail v-else ref="detailRef" @onClose="indexShow = true" @successful="tableRef.refresh()" />
+</template>
+
+<script setup name="${busName}">
+	<%
+	var iptTool = 0;
+	if (searchCount > 0) {
+		for(var i = 0; i < configList.~size; i++) {
+			if(!configList[i].needTableId) {
+			if(configList[i].effectType == 'select' || configList[i].effectType == 'radio' || configList[i].effectType == 'checkbox') {
+				iptTool++;
+			}
+			}
+		}
+	}
+	%>
+	<% if(iptTool > 0) { %>
+	import tool from '@/utils/tool'
+	<% } %>
+	import { cloneDeep } from 'lodash-es'
+	import Detail from './detail.vue'
+	import ${classNameFirstLower}Api from '@/api/${moduleName}/${classNameFirstLower}Api'
+	<% if (searchCount > 0) { %>
+	const searchFormState = ref({})
+	const searchFormRef = ref()
+	<% } %>
+	const tableRef = ref()
+	const detailRef = ref()
+	const indexShow = ref(true)
+	const toolConfig = { refresh: true, height: true, columnSetting: true, striped: false }
+	<% if(searchCount > 3) { %>
+	// 查询区域显示更多控制
+	const advanced = ref(false)
+	const toggleAdvanced = () => {
+		advanced.value = !advanced.value
+	}
+	<% } %>
+	const columns = [
+	<% for(var i = 0; i < configList.~size; i++) { %>
+	<% if(!configList[i].needTableId && configList[i].whetherTable) { %>
+		{
+			title: '${configList[i].fieldRemark}',
+			dataIndex: '${configList[i].fieldNameCamelCase}'<% if(configList[i].whetherRetract) { %>,<% } %>
+			<% if(configList[i].whetherRetract) { %>
+			ellipsis: true
+			<% } %>
+		},
+	<% } %>
+	<% } %>
+	]
+	// 操作栏通过权限判断是否显示
+	if (hasPerm(['${classNameFirstLower}Edit', '${classNameFirstLower}Delete'])) {
+		columns.push({
+			title: '操作',
+			dataIndex: 'action',
+			align: 'center',
+			width: 150
+		})
+	}
+	const selectedRowKeys = ref([])
+	// 列表选择配置
+	const options = {
+		// columns数字类型字段加入 needTotal: true 可以勾选自动算账
+		alert: {
+			show: true,
+			clear: () => {
+				selectedRowKeys.value = ref([])
+			}
+		},
+		rowSelection: {
+			onChange: (selectedRowKey, selectedRows) => {
+				selectedRowKeys.value = selectedRowKey
+			}
+		}
+	}
+	const loadData = (parameter) => {
+		<% if (searchCount > 0) { %>
+		const searchFormParam = cloneDeep(searchFormState.value)
+		<% for(var i = 0; i < configList.~size; i++) { %>
+		<% if(!configList[i].needTableId && configList[i].needPage) { %>
+		<% if (configList[i].effectType == 'datepicker') {%>
+		// ${configList[i].fieldNameCamelCase}范围查询条件重载
+		if (searchFormParam.${configList[i].fieldNameCamelCase}) {
+			searchFormParam.start${configList[i].fieldNameCamelCaseFirstUpper} = searchFormParam.${configList[i].fieldNameCamelCase}[0]
+			searchFormParam.end${configList[i].fieldNameCamelCaseFirstUpper} = searchFormParam.${configList[i].fieldNameCamelCase}[1]
+			delete searchFormParam.${configList[i].fieldNameCamelCase}
+		}
+		<% } %>
+		<% } %>
+		<% } %>
+		return ${classNameFirstLower}Api.${classNameFirstLower}Page(Object.assign(parameter, searchFormParam)).then((data) => {
+		<% } else { %>
+		return ${classNameFirstLower}Api.${classNameFirstLower}Page(parameter).then((data) => {
+		<% } %>
+			return data
+		})
+	}
+	// 重置
+	const reset = () => {
+		searchFormRef.value.resetFields()
+		tableRef.value.refresh(true)
+	}
+	// 删除
+	const delete${className} = (record) => {
+		let params = [
+			{
+				<% for(var i = 0; i < configList.~size; i++) { %>
+				<% if(configList[i].needTableId) { %>
+				${configList[i].fieldNameCamelCase}: record.${configList[i].fieldNameCamelCase}
+				<% } %>
+				<% } %>
+			}
+		]
+		${classNameFirstLower}Api.${classNameFirstLower}Delete(params).then(() => {
+			tableRef.value.refresh(true)
+		})
+	}
+	// 批量删除
+	const deleteBatch${className} = (params) => {
+		${classNameFirstLower}Api.${classNameFirstLower}Delete(params).then(() => {
+			tableRef.value.clearRefreshSelected()
+		})
+	}
+	// 切换至表单
+    const onDetail = (record = null) => {
+    	indexShow.value = false
+    	nextTick(() => {
+    		if (record) {
+    			detailRef.value.onOpen(record)
+    		}
+    	})
+    }
+	<% if (searchCount > 0) { %>
+	<% for(var i = 0; i < configList.~size; i++) { %>
+	<% if(!configList[i].needTableId && configList[i].needPage) { %>
+	<% if (configList[i].effectType == 'select' || configList[i].effectType == 'radio' || configList[i].effectType == 'checkbox') { %>
+	const ${configList[i].fieldNameCamelCase}Options = tool.dictList('${configList[i].dictTypeCode}')
+	<% } %>
+	<% } %>
+	<% } %>
+	<% } %>
+</script>