vue实现动态展开与折叠内联块元素
功能需求
当多个内联块元素(比如div)在一个固定宽度的父元素内自动排列换行时,如果这些元素的行数超过四行,那么默认折叠超出的部分,并在第四行末尾显示一个按钮供用户切换展开或折叠状态。在折叠状态下,为了确保展开/折叠按钮能够显示在第四行末尾,会隐藏第四行的最后一个元素。在展开状态下,所有元素都会显示,并且展开/折叠按钮会显示在所有元素的末尾。
效果预览
代码实现
<template>
<div>
<div class="container" ref="container">
<!-- 遍历每一个项目并根据条件决定是否隐藏 -->
<div
v-for="(item, index) in items"
:key="index"
:class="{ hidden: shouldHideItem(index) }"
class="inline-block-item"
>
{{ item }}
</div>
<!-- 只有当需要显示切换按钮时,才渲染此按钮 -->
<button
v-if="showToggle"
@click="toggle"
class="inline-block-item"
ref="toggleButton"
>
{{ isCollapsed ? '展开' : '折叠' }}
</button>
</div>
</div>
</template>
<script>
export default {
data() {
return {
// 初始项目列表
items: Array.from({ length: 20 }, (_, i) => `Item ${i + 1}`),
// 默认为折叠状态
isCollapsed: true,
// 是否需要显示切换按钮
showToggle: false,
// 第四行后的第一个项目的索引
firstItemIndexAfterFourRows: Infinity,
};
},
mounted() {
// 当组件加载完成后,检查是否需要显示切换按钮
this.$nextTick(() => {
this.checkIfNeedToggle();
});
},
methods: {
// 切换展开/折叠状态
toggle() {
this.isCollapsed = !this.isCollapsed;
},
// 根据当前状态和项目索引判断是否应该隐藏项目
shouldHideItem(index) {
if (this.isCollapsed) {
// 折叠状态时,隐藏第四行的最后一个项目
if (index === this.firstItemIndexAfterFourRows - 1) return true;
// 以及所有在第四行之后的项目
return index >= this.firstItemIndexAfterFourRows;
}
// 展开状态时不隐藏任何项目
return false;
},
// 检查是否需要显示切换按钮
checkIfNeedToggle() {
const container = this.$refs.container;
const children = container.children;
if (!children.length) return;
let rowCount = 1;
let prevTop = children[0].getBoundingClientRect().top;
for (let i = 1; i < children.length; i++) {
const currTop = children[i].getBoundingClientRect().top;
// 如果当前项目的顶部位置大于前一个项目,说明它在一个新的行
if (currTop > prevTop) {
rowCount++;
// 如果行数超过4行,记录索引并设置 showToggle 为 true
if (rowCount > 4) {
this.firstItemIndexAfterFourRows = i;
this.showToggle = true;
break;
}
}
prevTop = currTop;
}
},
},
};
</script>
<style scoped>
.container {
width: 200px;
}
.inline-block-item {
display: inline-block;
margin: 2px;
white-space: nowrap;
}
.hidden {
display: none;
}
</style>