HTML文本溢出省略号检测


175

我在页面上有一组块元素。它们都设置了CSS规则的空白,溢出,文本溢出,以便修剪掉溢出的文本并使用省略号。

但是,并非所有元素都会溢出。

无论如何,我可以使用JavaScript来检测哪些元素正在溢出吗?

谢谢。

补充:我正在使用的示例HTML结构。

<td><span>Normal text</span></td>
<td><span>Long text that will be trimmed text</span></td>

SPAN元素始终适合单元格,它们应用了省略号规则。我想检测何时将省略号应用于SPAN的文本内容。



10
不能重复!这个问题是关于另一个父元素中的一个元素的。我在谈论单个元素内的文本。就我而言,TD中的SPAN永远不会溢出TD,这是SPAN中的文本溢出并被修剪。这就是我要检测的!对不起-我本可以更好地提出这个问题。
deanoj 2011年

哦,我忘了补充-仅在有帮助的情况下才需要在Webkit上工作...
deanoj 2011年

我做了Ghommey只是为了看看它是否有效……没有。
deanoj 2011年

省略号方面无关紧要;您需要检测的只是它是否溢出。
Spudley

Answers:


114

曾几何时,我需要这样做,而我遇到的唯一跨浏览器可靠的解决方案是黑客工作。我不是这种解决方案的最大拥护者,但是它肯定会一次又一次地产生正确的结果。

这个想法是您克隆元素,删除所有边界宽度,然后测试克隆的元素是否比原始元素宽。如果是这样,您将知道它会被截断。

例如,使用jQuery:

var $element = $('#element-to-test');
var $c = $element
           .clone()
           .css({display: 'inline', width: 'auto', visibility: 'hidden'})
           .appendTo('body');

if( $c.width() > $element.width() ) {
    // text was truncated. 
    // do what you need to do
}

$c.remove();

我做了一个jsFiddle来演示这一点,http://jsfiddle.net/cgzW8/2/

您甚至可以为jQuery创建自己的自定义伪选择器:

$.expr[':'].truncated = function(obj) {
  var $this = $(obj);
  var $c = $this
             .clone()
             .css({display: 'inline', width: 'auto', visibility: 'hidden'})
             .appendTo('body');

  var c_width = $c.width();
  $c.remove();

  if ( c_width > $this.width() )
    return true;
  else
    return false;
};

然后用它来查找元素

$truncated_elements = $('.my-selector:truncated');

演示:http//jsfiddle.net/cgzW8/293/

希望这会有所帮助,确实如此。


1
@劳里不; CSS截断不会更改框中的实际文本,因此,无论截断,可见还是隐藏,内容始终是相同的。仍然没有办法以编程方式获取被截断的文本,如果有的话,您将不需要首先克隆该元素!
基督教徒

1
似乎在没有的情况下这是行不通的white-space: nowrap
Jakub Hampl

2
我必须说,在互联网上搜索很多商品并尝试实施许多解决方案之后,这是我找到的最可靠的解决方案。这种解决方案在element.innerWidth或element.OffsetWidth这样的浏览器之间不会给出不同的结果,在使用边距或填充时会出现问题。
Script14

1
对我来说,这在任何地方都行不通。我不确定这如何取决于CSS(我曾尝试在输入控件上使用它,以下解决方案至少在Chrome和Firefox中有效(但在IE11中无效))...
亚历山大

3
很棒的解决方案,尤其是使用jQuery伪选择器。但是有时可能不起作用,因为如果元素文本具有不同的样式(字体大小,字母间距,内部元素的边距),则宽度计算不正确。在这种情况下,我建议将克隆元素附加到$ this.parent()而不是'body'。这将提供元素的精确副本,并且计算将是正确的。还是谢谢你
亚历克斯

284

试试这个JS函数,将span元素作为参数传递:

function isEllipsisActive(e) {
     return (e.offsetWidth < e.scrollWidth);
}

10
这应该是答案-非常优雅,可在Chrome和IE(9)上使用
Roy Truelove

14
此答案和Alex的答案在IE8中不起作用。在某些奇怪的情况下,滚动宽度和外部宽度是相同的...但是有省略号,在某些情况下它们是相同的...并且没有省略号。换句话说,第一个解决方案是唯一可在所有浏览器上使用的解决方案。

3
对于那些需要了解offsetWidth,scrollWidth和clientWidth的人来说,这是一个很好的解释:stackoverflow.com/a/21064102/1815779
Linh Dam

4
text-overflow:ellipsis它应该是e.offsetWidth <= e.scrollWidth
oriadam

4
在flex子代上,它们始终彼此相等,因此对它们不起作用。
凯文·比尔

14

除了italo的答案外,您还可以使用jQuery完成此操作。

function isEllipsisActive($jQueryObject) {
    return ($jQueryObject.width() < $jQueryObject[0].scrollWidth);
}

另外,正如Smoky所指出的那样,您可能要使用jQuery outsideWidth()而不是width()。

function isEllipsisActive($jQueryObject) {
    return ($jQueryObject.outerWidth() < $jQueryObject[0].scrollWidth);
}

9

对于使用(或计划使用)Christian Varga接受的答案的用户,请注意性能问题。

以这种方式克隆/操作DOM会导致DOM重排(请参阅此处有关DOM重的说明),这是非常耗费资源的。

在页面上的100多个元素上使用Christian Varga的解决方案会导致4秒钟的重排延迟,在此期间JS线程被锁定。考虑到JS是单线程的,这意味着最终用户会有大量的UX延迟。

Italo Borssatto的答案应该是被接受的答案,在我进行分析时,它的速度大约要快10倍。


2
不幸的是,它并非在每种情况下都有效(我希望如此)。通过仅在需要检查的情况下运行它,就可以抵消您在此处提到的性能损失-就像仅在mouseenter查看是否需要显示工具提示时一样。
雅各布

5

italo的回答非常好!但是,让我对其进行一些改进:

function isEllipsisActive(e) {
   var tolerance = 2; // In px. Depends on the font you are using
   return e.offsetWidth + tolerance < e.scrollWidth;
}

跨浏览器兼容性

如果,事实上,你试试上面的代码和使用console.log打印出的价值观e.offsetWidthe.scrollWidth,你会发现,在IE浏览器,即,即使你有没有文字截断的值差1px或者2px是经验丰富。

因此,根据您使用的字体大小,请允许一定的公差!


在IE10上不起作用-offsetWidth与scrollWidth相同,都给了我截短的宽度。
SsjCosty

1
我在Chrome 60(最新版)上也需要此公差。
JanŻankowski17年


4

此示例显示单元格表上的工具提示,其中文本被截断。根据表格宽度是动态的:

$.expr[':'].truncated = function (obj) {
    var element = $(obj);

    return (element[0].scrollHeight > (element.innerHeight() + 1)) || (element[0].scrollWidth > (element.innerWidth() + 1));
};

$(document).ready(function () {
    $("td").mouseenter(function () {
        var cella = $(this);
        var isTruncated = cella.filter(":truncated").length > 0;
        if (isTruncated) 
            cella.attr("title", cella.text());
        else 
            cella.attr("title", null);
    });
});

演示:https//jsfiddle.net/t4qs3tqs/

它适用于所有版本的jQuery


1

我认为更好的检测方法是use getClientRects(),似乎每个rect都具有相同的高度,因此我们可以使用不同top值的数量来计算行数。

getClientRects这样工作

function getRowRects(element) {
    var rects = [],
        clientRects = element.getClientRects(),
        len = clientRects.length,
        clientRect, top, rectsLen, rect, i;

    for(i=0; i<len; i++) {
        has = false;
        rectsLen = rects.length;
        clientRect = clientRects[i];
        top = clientRect.top;
        while(rectsLen--) {
            rect = rects[rectsLen];
            if (rect.top == top) {
                has = true;
                break;
            }
        }
        if(has) {
            rect.right = rect.right > clientRect.right ? rect.right : clientRect.right;
            rect.width = rect.right - rect.left;
        }
        else {
            rects.push({
                top: clientRect.top,
                right: clientRect.right,
                bottom: clientRect.bottom,
                left: clientRect.left,
                width: clientRect.width,
                height: clientRect.height
            });
        }
    }
    return rects;
}

getRowRects这样工作

你可以发现像这样


1

所有的解决方案并没有真正为我工作,怎么工作的要素比较scrollWidthscrollWidth其父(或子女,这取决于哪个元素具有触发)。

当孩子的scrollWidth父母高于其父母时,则表示.text-ellipsis是活跃的。


event父元素何时是

function isEllipsisActive(event) {
    let el          = event.currentTarget;
    let width       = el.offsetWidth;
    let widthChild  = el.firstChild.offsetWidth;
    return (widthChild >= width);
}

什么时候event是子元素

function isEllipsisActive(event) {
    let el          = event.currentTarget;
    let width       = el.offsetWidth;
    let widthParent = el.parentElement.scrollWidth;
    return (width >= widthParent);
}

0

e.offsetWidth < e.scrollWidth解决方案并不总是工作。

而且,如果您要使用纯JavaScript,我建议使用以下代码:

(打字稿)

public isEllipsisActive(element: HTMLElement): boolean {
    element.style.overflow = 'initial';
    const noEllipsisWidth = element.offsetWidth;
    element.style.overflow = 'hidden';
    const ellipsisWidth = element.offsetWidth;

    if (ellipsisWidth < noEllipsisWidth) {
      return true;
    } else {
      return false;
    }
}

0

@ItaloBorssatto解决方案是完美的。但是在看SO之前-我做了决定。这里是 :)

const elems = document.querySelectorAll('span');
elems.forEach(elem => {
  checkEllipsis(elem);
});

function checkEllipsis(elem){
  const canvas = document.createElement('canvas');
  const ctx = canvas.getContext('2d');
  const styles = getComputedStyle(elem);
  ctx.font = `${styles.fontWeight} ${styles.fontSize} ${styles.fontFamily}`;
  const widthTxt = ctx.measureText(elem.innerText).width;
  if (widthTxt > parseFloat(styles.width)){
    elem.style.color = 'red'
  }
}
span.cat {
    display: block;
    border: 1px solid black;
    white-space: nowrap;
    width: 100px;
    overflow: hidden;
    text-overflow: ellipsis;
}
 <span class="cat">Small Cat</span>
      <span class="cat">Looooooooooooooong Cat</span>


0

我的实现)

const items = Array.from(document.querySelectorAll('.item'));
items.forEach(item =>{
    item.style.color = checkEllipsis(item) ? 'red': 'black'
})

function checkEllipsis(el){
  const styles = getComputedStyle(el);
  const widthEl = parseFloat(styles.width);
  const ctx = document.createElement('canvas').getContext('2d');
  ctx.font = `${styles.fontSize} ${styles.fontFamily}`;
  const text = ctx.measureText(el.innerText);
  return text.width > widthEl;
}
.item{
  width: 60px;
  overflow: hidden;
  text-overflow: ellipsis;
}
      <div class="item">Short</div>
      <div class="item">Loooooooooooong</div>


By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.