关于多tag搜索换行场景的展开折叠实现

近期做了一个多行tag搜索的需求,要求是最多展示两行,超过两行则隐藏,并展示一个 expand all 的按钮,具体效果如下

注意,这里并不是文字的展开收缩,所以css自带的展开收缩样式是不太适合的

核心要点

定位第二行最后一个元素的下标

首先,布局肯定是用flex布局,并且支持换行,核心部分就是需要找到第二行最后一个元素的下标

将第二行最后一个元素设置为 visibility: hidden;

目的是避免用expand all替换最后一个元素之后造成布局的改变,避免第三行的元素被挤到第二行expand all 后面

expand all 按钮绝对定位到第二行最后一个子元素旁边

在不影响原本tags列表布局情况的同时将expand all加上去

子元素宽度相加计算【不建议】

经过一番查找,在社区里学到的一种解决方案是通过计算子元素的宽度以及视图宽度来确定子元素是否换行,并且记录当前行数,当前行数进行到第三行时,就可以知道第二行最后一个元素的下标

这种方式理论上是可以的,但是实践过程中,常常出现expand all后面多出一个tag的情况,这种情况的原因我猜测是因为宽度计算的准确性没法保证

通过子元素的定位计算【推荐】

首先,子元素是横向排布,理论上每一行的子元素的y坐标是一样的【事实证明也会有一点点偏差,比如border之类样式的影响,所以最好有个5px的容错比较】

只需要循环子元素,通过 offsetTop 比较,就可以计算出第二行的最后一个元素

expand all 元素后面增加 <div style={{ width: "100%" }}></div> 可以避免后面的元素被挤上来,强制flex换行

核心代码

js 复制代码
  function getLastElementOfSecondRow() {
    const container = document.querySelector<HTMLElement>(".tags");
    if (!container) return;
    const containerWidth = container.offsetWidth;

    let lastIndex = 100;
    let currnetY = 0;
    let row = 0;

  
    if (containerWidth !== null) {
      const items = container.querySelectorAll<HTMLElement>(".tag-item");

      for (let i = 0; i < items.length; i++) {
        const item = items[i];
        if (Math.abs(currnetY - item.offsetTop) > 20) {
          currnetY = item.offsetTop;
          row++;
        }

        if (row > 2) {
          lastIndex = i - 1;
          break;
        }
      }
    }
    if (row <= 2) setExpand(false);

    setSectionRowIndex(lastIndex - 1);
  }

组件渲染

js 复制代码
<div className="tags flex flex-row flex-wrap">
 {list.map((item, index) => {
          if (index === secondRowIndex)
            return (
              <>
                <div
                  className="tag-item"
                  style={{
                    position: "relative",
                    minWidth: "112px",
                    height: "40px",
                  }}
                >
                  <button
                    key="expand"
                    style={{ position: "absolute", left: 0 }}
                    onClick={() => setExpand(!expand)}
                  >
                      + Expand all
                  </button>
                  <button
                    key={index}
                    style={{ visibility: "hidden" }}
                  >
                      {item.nickname}
                  </button>
                </div>
                <div style={{ width: "100%" }}></div>
              </>
            );
          return (
            <button
              className="tag-item"
              key={index}
            >
                {item.nickname}
            </button>
          );
        })}
</div>
相关推荐
YL有搞头4 分钟前
webpack的构建流程以及loader和plugin
前端·webpack·node.js
2503_928411569 分钟前
11.20 vue项目搭建-单页面应用
前端·javascript·vue.js
BUG创建者9 分钟前
项目中使用script-ext-html-webpack-plugin
前端·webpack·html
极光代码工作室20 分钟前
基于SpringBoot的校园招聘信息管理系统的设计与实现
java·前端·spring
打小就很皮...36 分钟前
React VideoPlay 组件封装与使用指南
前端·react.js·video
Ace_317508877639 分钟前
微店平台关键字搜索接口深度解析:从 Token 动态生成到多维度数据挖掘
java·前端·javascript
苏小画1 小时前
Vue 组件库从创建到发布全流程
前端·javascript·vue.js
月小满1 小时前
DataV轮播时其他组件的内容也一起滚动 修复bug的方法
前端·vue.js·bug·大屏端
小莫分享1 小时前
Github Action 一键部署HTML 静态服务
前端·html·github
星释1 小时前
Rust 练习册 66:密码方块与文本加密
java·前端·rust