javascript技巧

关注公众号 jb51net

关闭
首页 > 网络编程 > JavaScript > javascript技巧 > JavaScript文本中间缩略

JavaScript实现文本中间缩略的两种方案

作者:碎_浪

项目中经常会遇到缩略展示文字的场景,即要求文字在一行不换行展示,超出自动展示...,常用的展示效果有两种,文字中间缩略以及文字末尾缩略,本文将通过代码示例给大家详细的讲一下这两种方案,需要的朋友可以参考下

前言

项目中经常会遇到缩略展示文字的场景,即要求文字在一行不换行展示,超出自动展示...

常用的展示效果有两种,文字中间缩略以及文字末尾缩略,效果如下所示

// 中间缩略
这是一段超长...超长文字如何展示

// 末尾缩略
这是一段超长的文字,看看超长....

对于描述之类的文本,一般选择末尾缩略,这种场景仅靠css即可快速实现。但是对于文件名之类的场景,会期望中间缩略,保留末尾的文件后缀。下面分别介绍下两种方案的实现。

末尾缩略

使用css需实现3个效果:

如果在实现过程中发现样式不生效,一定是上述3个条件有不满足的。

对应css代码如下:

    overflow: hidden;
    text-overflow: ellipsis;
    white-space: nowrap;

中间缩略

中间缩略的样式,只能手动切分字符串,并在中间拼接..., 因此关键在与,如何获得要切分的起止位置。

思路整理

本例使用React实现。之所以只用js实现,一个是因为简单,第二个则是使用css的大部分方式,需要重叠两个div,用户在选中的时候,体验不太好。

<div className="auto-ellipsis-text">
    <span className="auto-ellipsis-text-inner">{str}</span>
 </div>
.auto-ellipsis-text{
    width: 100%;
    overflow: hidden;
    white-space: nowrap;
}

div.auto-ellipsis-text上增加ref,获取对应的dom句柄

const handle = useCallback(() => {
    // 获取渲染节点的句柄
    const cu = textRef.current;
    if (!cu) {
        return;
    }
    // 获取此时父元素宽度
    const width = cu.clientWidth;

    // 获取文本节点宽度
    const textChild = cu.firstElementChild as HTMLElement;
    const innerWidth = textChild.offsetWidth;
    
    // 没有超出则不处理
    if (width >= innerWidth) {
        return;
    }

    // 计算切分位置
    const splitWidth = width / 2;

    const range = document.createRange();
    const length = textChild.innerText.length;
    const arr = [];
    let countWidth = 0;
    let startIndex = 0;
    let endIndex = 0;
    // 步进为2,逐个获取对应的宽度,并在首次累计宽度大于 splitWidth 时记录起始位置,
    // 在首次剩余宽度小于 splitWidth 时,记录截止位置,停止for循环
    for (let i = 0; i < length - 2; i += 2) {
        range.setStart(textChild.childNodes[0], i);
        range.setEnd(textChild.childNodes[0], i + 2);
        const rs = range.getBoundingClientRect();
        arr.push({ index: i, width: rs.width });
        countWidth += rs.width;
        if (countWidth > splitWidth && !startIndex) {
            startIndex = i;
        }
        if (innerWidth - countWidth < splitWidth) {
            endIndex = i + 2;
            break;
        }
    }
    // 切分字符串并补充缩略符
    const newStr = text.slice(0, startIndex) + '...' + str.slice(endIndex);

    // 更新渲染节点
    setStr(newStr);
}, []);

最终实现

export default function AutoEllipsisText({
    text,
    className,
    version,
}: {
    text: string;
    className?: string;
    version?: number;
}) {
    const [str, setStr] = useState(text);
    const textRef = useRef<HTMLDivElement>(null);

    const handle = useCallback(() => {
        /* 功能函数...*/
    }, []);

    useEffect(() => {
        setStr(text);
        Promise.resolve().then(() => {
            handle();
        });
    }, [text, version]);

    return (
        <div className={`auto-ellipsis-text ${className || ''}`} ref={textRef}>
            <span className="auto-ellipsis-text-inner">{str}</span>
        </div>
    );
}

使用

调用示例

<AutoEllipsisText text="这是一个超长的文件名超长的文件名超长的文件名文件末尾在这里.txt" version={width} />

待优化项

写在最后

以上就是JavaScript实现文本中间缩略的两种方案的详细内容,更多关于JavaScript文本中间缩略的资料请关注脚本之家其它相关文章!

您可能感兴趣的文章:
阅读全文