This commit is contained in:
2026-02-03 11:58:54 +08:00
3 changed files with 340 additions and 47 deletions

View File

@@ -26,20 +26,12 @@
</div> </div>
</el-form-item> </el-form-item>
<slot :visible="true" :expanded="isExpanded"></slot> <slot :visible="true" :expanded="isExpanded"></slot>
<!-- 收起时查询/重置在首行 --> <!-- 折叠区收起时隐藏display:contents 不破坏表单行内流 -->
<div v-show="!isExpanded" class="actions-outside"> <div v-if="hasCollapsibleItems" v-show="isExpanded" class="collapse-visibility">
<slot name="actions"></slot> <slot :visible="false" :expanded="isExpanded" @has-content="hasCollapsibleContent = true"></slot>
</div> </div>
<!-- 折叠区展开时内部再放一份查询/重置位置在更多筛选项之后 --> <div class="search-form__actions">
<div v-if="hasCollapsibleItems" class="collapse-grid" :class="{ 'is-expanded': isExpanded }"> <slot name="actions" :expanded="isExpanded"></slot>
<div class="collapse-grid-inner">
<div class="collapse-grid-cell">
<slot :visible="false" :expanded="isExpanded" @has-content="hasCollapsibleContent = true"></slot>
<div v-show="isExpanded" class="actions-inside">
<slot name="actions"></slot>
</div>
</div>
</div>
</div> </div>
<div v-show="false" ref="detectionWrapperRef" class="detection-wrapper"> <div v-show="false" ref="detectionWrapperRef" class="detection-wrapper">
<slot :visible="false" :expanded="true"></slot> <slot :visible="false" :expanded="true"></slot>
@@ -218,7 +210,6 @@ defineExpose({
:deep(.el-form-item__content) { :deep(.el-form-item__content) {
margin-left: 0 !important; margin-left: 0 !important;
} }
margin-right: 16px;
margin-bottom: 18px; margin-bottom: 18px;
} }
@@ -280,40 +271,21 @@ defineExpose({
} }
:deep(.el-form-item:not(:has(.el-button))) { :deep(.el-form-item:not(:has(.el-button))) {
margin-right: 16px; margin-right: 15px;
} }
/* grid 0fr→1fr 高度过渡ease-out 收尾不弹跳 */ /* 折叠区收起时隐藏,展开时 display:contents 不破坏表单行内流 */
.collapse-grid { .collapse-visibility {
display: grid;
grid-template-rows: 0fr;
transition: grid-template-rows 0.26s cubic-bezier(0.33, 1, 0.68, 1);
contain: layout;
&.is-expanded {
grid-template-rows: 1fr;
}
}
.collapse-grid-inner {
overflow: hidden;
min-height: 0;
backface-visibility: hidden;
}
.collapse-grid-cell {
overflow: hidden;
min-height: 0;
opacity: 0;
/* 收起时立即隐藏,不出现输入框慢慢消失 */
transition: opacity 0s;
}
.collapse-grid.is-expanded .collapse-grid-cell {
opacity: 1;
/* 仅展开时淡入 */
transition: opacity 0.12s ease-out;
}
.actions-outside,
.actions-inside {
display: contents; display: contents;
} }
/* 查询/重置按钮区:表单项 margin-right 为 0 */
.search-form__actions {
display: contents;
:deep(.el-form-item) {
margin-right: 0 !important;
}
}
} }
</style> </style>

View File

@@ -0,0 +1,321 @@
<template>
<div class="search-form__wrap" :class="{ 'search-form__wrap--with-title': filterTitle }">
<div class="search-form__bar">
<div class="search-form-container">
<el-form :model="formModel" ref="formRef" :inline="true" @keyup.enter="handleKeyupEnter" :label-width="labelWidth">
<!-- 筛选 + 展开更多 放在同一表单项内保证垂直对齐 -->
<el-form-item v-if="filterTitle || hasCollapsibleItems" class="search-form__left-group">
<div class="search-form__left-inner">
<span v-if="filterTitle" class="search-form__title">
<el-icon class="search-form__title-icon"><Search /></el-icon>
{{ filterTitle }}
</span>
<el-button
v-if="hasCollapsibleItems"
link
type="primary"
class="toggle-btn"
@click="toggleExpand"
>
<el-icon style="margin-right: 4px">
<ArrowUp v-if="isExpanded" />
<ArrowDown v-else />
</el-icon>
{{ isExpanded ? '收起' : '展开更多' }}
</el-button>
</div>
</el-form-item>
<slot :visible="true" :expanded="isExpanded"></slot>
<!-- 收起时查询/重置在首行 -->
<div v-show="!isExpanded" class="actions-outside">
<slot name="actions"></slot>
</div>
<!-- 折叠区展开时内部再放一份查询/重置位置在更多筛选项之后 -->
<div v-if="hasCollapsibleItems" class="collapse-grid" :class="{ 'is-expanded': isExpanded }">
<div class="collapse-grid-inner">
<div class="collapse-grid-cell">
<slot :visible="false" :expanded="isExpanded" @has-content="hasCollapsibleContent = true"></slot>
<div v-show="isExpanded" class="actions-inside">
<slot name="actions"></slot>
</div>
</div>
</div>
</div>
<div v-show="false" ref="detectionWrapperRef" class="detection-wrapper">
<slot :visible="false" :expanded="true"></slot>
</div>
</el-form>
</div>
</div>
</div>
</template>
<script setup lang="ts" name="search-form">
import { ref, watch, computed, onMounted, nextTick } from 'vue';
import { ArrowUp, ArrowDown, Search } from '@element-plus/icons-vue';
const props = defineProps({
/**
* 筛选标题(如「筛选」),传入时在表单左侧显示标题+竖线,与表单项同一行
*/
filterTitle: {
type: String,
default: '',
},
/**
* 表单数据模型
*/
model: {
type: Object,
required: true,
},
/**
* 是否默认展开折叠区域
*/
defaultExpanded: {
type: Boolean,
default: false,
},
/**
* 是否显示展开/折叠按钮
* 如果不设置将自动检测是否有可折叠内容visible=false 的插槽)
*/
showCollapse: {
type: Boolean,
default: undefined,
},
/**
* 表单项标签宽度
*/
labelWidth: {
type: String,
default: '',
},
});
const emit = defineEmits(['expand-change', 'keyup-enter']);
const formModel = props.model;
const { filterTitle } = props;
const formRef = ref();
const isExpanded = ref(props.defaultExpanded);
const hasCollapsibleContent = ref(false);
const detectionWrapperRef = ref<HTMLElement | null>(null);
// 检测是否有可折叠内容
const checkCollapsibleContent = () => {
// 如果 showCollapse 明确指定,则使用指定值
if (props.showCollapse !== undefined) {
hasCollapsibleContent.value = props.showCollapse
return
}
// 否则,通过检查隐藏的检测元素是否有内容来判断
// 需要等待 DOM 渲染完成,可能需要多次尝试以确保数据加载完成
let retryCount = 0
const maxRetries = 5
const check = () => {
nextTick(() => {
setTimeout(() => {
if (detectionWrapperRef.value) {
// 检查检测元素是否有子元素(排除文本节点)
// 检查是否有 el-form-item 元素(因为表单项会被渲染为 el-form-item
const hasContent = detectionWrapperRef.value.children.length > 0 ||
detectionWrapperRef.value.querySelector('.el-form-item') !== null ||
(!!detectionWrapperRef.value.textContent && detectionWrapperRef.value.textContent.trim() !== '')
if (hasContent || retryCount >= maxRetries) {
hasCollapsibleContent.value = hasContent
} else {
// 如果还没检测到内容且未达到最大重试次数,继续重试
retryCount++
setTimeout(check, 100)
}
} else {
if (retryCount < maxRetries) {
retryCount++
setTimeout(check, 100)
} else {
hasCollapsibleContent.value = false
}
}
}, 50)
})
}
check()
}
// 是否有需要折叠的项
const hasCollapsibleItems = computed(() => {
if (props.showCollapse !== undefined) {
return props.showCollapse
}
return hasCollapsibleContent.value
});
// 处理回车键
const handleKeyupEnter = () => {
emit('keyup-enter');
};
// 切换展开/收起状态
const toggleExpand = () => {
isExpanded.value = !isExpanded.value;
emit('expand-change', isExpanded.value);
};
// 监听 defaultExpanded 变化
watch(
() => props.defaultExpanded,
(newVal) => {
isExpanded.value = newVal;
}
);
// 监听 showCollapse 变化
watch(
() => props.showCollapse,
() => {
checkCollapsibleContent()
}
);
// 初始化时检测
onMounted(() => {
checkCollapsibleContent()
});
// 暴露方法供外部调用
defineExpose({
formRef,
toggleExpand,
expand: () => {
isExpanded.value = true;
emit('expand-change', true);
},
collapse: () => {
isExpanded.value = false;
emit('expand-change', false);
},
});
</script>
<style lang="scss" scoped>
.search-form__bar {
display: flex;
align-items: flex-start;
flex-wrap: wrap;
gap: 4px 0;
min-height: 32px;
}
.search-form__left-group {
:deep(.el-form-item__label) {
display: none;
}
:deep(.el-form-item__content) {
margin-left: 0 !important;
}
margin-bottom: 18px;
}
.search-form__left-inner {
display: flex;
align-items: center;
flex-wrap: wrap;
gap: 0;
}
.search-form__title {
position: relative;
display: inline-flex;
align-items: center;
gap: 6px;
flex-shrink: 0;
padding-right: 12px;
margin-right: 12px;
height: 32px;
line-height: 32px;
font-size: 14px;
font-weight: 500;
color: var(--el-text-color-secondary);
cursor: default;
user-select: none;
&::after {
content: '';
position: absolute;
right: 0;
top: 50%;
transform: translateY(-50%);
width: 1px;
height: 14px;
background: var(--el-border-color);
}
}
.search-form__title-icon {
font-size: 15px;
color: var(--el-text-color-secondary);
}
.search-form-container {
flex: 1;
min-width: 0;
:deep(.el-form--inline) {
align-items: center;
}
.search-form__left-inner :deep(.toggle-btn) {
font-size: 14px;
font-weight: 500;
height: 32px;
display: inline-flex;
align-items: center;
border-radius: 6px;
}
:deep(.el-form-item:not(:has(.el-button))) {
margin-right: 15px;
}
/* grid 0fr→1fr 高度过渡ease-out 收尾不弹跳 */
.collapse-grid {
display: grid;
grid-template-rows: 0fr;
transition: grid-template-rows 0.26s cubic-bezier(0.33, 1, 0.68, 1);
contain: layout;
&.is-expanded {
grid-template-rows: 1fr;
}
}
.collapse-grid-inner {
overflow: hidden;
min-height: 0;
backface-visibility: hidden;
}
.collapse-grid-cell {
overflow: hidden;
min-height: 0;
opacity: 0;
/* 收起时立即隐藏,不出现输入框慢慢消失 */
transition: opacity 0s;
}
.collapse-grid.is-expanded .collapse-grid-cell {
opacity: 1;
/* 仅展开时淡入 */
transition: opacity 0.12s ease-out;
}
.actions-outside,
.actions-inside {
display: contents;
:deep(.el-form-item) {
margin-right: 0!important;
}
}
}
</style>

View File

@@ -154,7 +154,7 @@
{label:'高校教师资格证',value:'highCer_高校教师资格证'}, {label:'高校教师资格证',value:'highCer_高校教师资格证'},
{label:'中等教师资格证',value:'midCer_中等教师资格证'}, {label:'中等教师资格证',value:'midCer_中等教师资格证'},
{label:'教师上岗证',value:'priCer_教师上岗证'}, {label:'教师上岗证',value:'priCer_教师上岗证'},
{label:'共同居住人',value:'livewith_共同居住人'}, // {label:'共同居住人',value:'livewith_共同居住人'},
] ]
// 导出文件函数 // 导出文件函数
@@ -169,7 +169,7 @@
'Content-Type': 'application/json' 'Content-Type': 'application/json'
} }
}) })
const blob = new Blob([res.data]) const blob = new Blob([res])
const fileName = fileNameStr const fileName = fileNameStr
const elink = document.createElement('a') const elink = document.createElement('a')
elink.download = fileName elink.download = fileName
@@ -239,7 +239,7 @@
"teacherBaseDTO": searchData "teacherBaseDTO": searchData
} }
await exportForModel(params, "教职工信息.xls", "/professional/teacherbase/exportTeacherInfoBySelf") await exportForModel(params, "教职工信息.xls", "/professional/file/exportTeacherInfoBySelf")
setTimeout(() => { setTimeout(() => {
exportLoading.value = false exportLoading.value = false