HTML文本溢出省略号检测

dea*_*noj 161 html javascript css ellipsis

我在页面上有一组块元素.它们都具有CSS规则white-space,overflow,text-overflow set,以便修剪溢出的文本并使用省略号.

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

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

谢谢.

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

<td><span>Normal text</span></td>
<td><span>Long text that will be trimmed text</span></td>
Run Code Online (Sandbox Code Playgroud)

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

Ita*_*tto 251

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

function isEllipsisActive(e) {
     return (e.offsetWidth < e.scrollWidth);
}
Run Code Online (Sandbox Code Playgroud)

  • 这个答案和Alex的答案在IE8中不起作用; 有一些奇怪的情况,滚动宽度和外部宽度是相同的...但它有省略号,然后它们是相同的一些情况......并且没有省略号.换句话说,第一个解决方案是唯一适用于所有浏览器的解决方案. (14认同)
  • 他们在 flex 孩子上总是平等的,因此,对他们不起作用。 (11认同)
  • 这应该是答案 - 非常优雅,适用于Chrome和IE(9) (9认同)
  • 在`text-overflow:ellipsis`上应该是`e.offsetWidth <= e.scrollWidth` (4认同)
  • 我正在调查检测似乎偏离一个字符的问题。我检测到溢出比我应该晚一个字符。我认为当文本实际上可以适合时,省略号的显示略有差异。我尝试暂时禁用“文本溢出:省略号”,果然文本适合没有任何溢出。可能是浏览器问题。我在 WINdows 10 上使用 Firefox 62。 (4认同)
  • 对于那些需要了解offsetWidth,scrollWidth和clientWidth的人来说,这是一个非常好的解释:http://stackoverflow.com/a/21064102/1815779 (3认同)
  • 就我而言,“offsetWidth”和“scrollWidth”始终相等。使用“offsetHeight”和“scrollHeight”来检测溢出。请参阅 /sf/ask/3656241851/ (3认同)
  • 我还用 `clientWidth` 而不是 `offsetWidth` 进行了测试,同样的问题也很明显。 (2认同)

Chr*_*rga 109

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

我们的想法是克隆元素,删除任何边界宽度,并测试克隆元素是否比原始元素宽.如果是这样,你知道它会被截断.

例如,使用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();
Run Code Online (Sandbox Code Playgroud)

我做了一个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;
};
Run Code Online (Sandbox Code Playgroud)

然后用它来查找元素

$truncated_elements = $('.my-selector:truncated');
Run Code Online (Sandbox Code Playgroud)

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

希望这有助于实现hacky.

  • 很好的解决方案,特别是使用jQuery伪选择器.但有时可能不起作用,因为如果元素文本具有不同的样式(字体大小,字母间距,内部元素的边距),则宽度计算不正确.在这种情况下,我建议将克隆元素追加到$ this.parent()而不是'body'.这将给出元素的精确副本,并且计算将是正确的.不管怎么说,还是要谢谢你 (3认同)
  • 我必须说,在网上搜索了很多并尝试实施了许多解决方案,这是迄今为止我发现的最可靠的解决方案.这个解决方案不会在像meta.innerWidth或element.OffsetWidth这样的浏览器之间给出不同的结果,它们在使用边距或填充时会有问题.很好的解决方案,做得好. (2认同)

Ale*_*x K 13

添加到italo的答案,你也可以使用jQuery来做到这一点.

function isEllipsisActive($jQueryObject) {
    return ($jQueryObject.width() < $jQueryObject[0].scrollWidth);
}
Run Code Online (Sandbox Code Playgroud)

另外,正如Smoky指出的那样,您可能希望使用jQuery outerWidth()而不是width().

function isEllipsisActive($jQueryObject) {
    return ($jQueryObject.outerWidth() < $jQueryObject[0].scrollWidth);
}
Run Code Online (Sandbox Code Playgroud)


小智 7

此示例在单元格表格上显示工具提示,并截断文本。根据表格宽度动态变化:

$.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);
    });
});
Run Code Online (Sandbox Code Playgroud)

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

它适用于所有版本的 jQuery


Rya*_*led 7

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

以这种方式克隆/操作DOM会导致DOM Reflow(请参阅此处有关DOM reflow的解释),这非常耗费资源.

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

Italo Borssatto的答案应该是被接受的答案,在我的分析中它快了大约10倍.

  • 不幸的是,它并不适用于所有场景(我希望如此)。您在这里提到的性能损失可以通过仅在需要检查的情况下运行它来抵消 - 例如仅在“mouseenter”上查看是否需要显示工具提示。 (2认同)

Jef*_*aal 6

所有的解决方案并不真正适合我,有效的将元素scrollWidthscrollWidth其父元素(或子元素,取决于哪个元素具有触发器)进行比较。

当孩子的scrollWidth比父母高时,就意味着.text-ellipsis活跃。


什么时候el是父元素

function isEllipsisActive(el) {
    let width       = el.offsetWidth;
    let widthChild  = el.firstChild.offsetWidth;
    return (widthChild >= width);
}
Run Code Online (Sandbox Code Playgroud)

el子元素是什么时候

function isEllipsisActive(event) {
    let width       = el.offsetWidth;
    let widthParent = el.parentElement.scrollWidth;
    return (width >= widthParent);
}
Run Code Online (Sandbox Code Playgroud)


And*_*dry 5

来自italo 的回答非常好!不过让我稍微细化一下:

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

跨浏览器兼容性

事实上,如果你尝试上面的代码并使用console.log打印出e.offsetWidthand的值e.scrollWidth,你会注意到,在 IE 上,即使你没有文本截断,也经历了1px或的值差异2px

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


小智 5

elem.offsetWdith VS ele.scrollWidth这对我有用! https://jsfiddle.net/gustavojuan/210to9p1/

$(function() {
  $('.endtext').each(function(index, elem) {
    debugger;
    if(elem.offsetWidth !== elem.scrollWidth){
      $(this).css({color: '#FF0000'})
    }
  });
});
Run Code Online (Sandbox Code Playgroud)


Nic*_*ier 5

最简单(也是跨浏览器)的解决方案实际上是将scrollWidth与clientWidth进行比较

此处的工作代码:https : //stackoverflow.com/a/19156627/1213445


归档时间:

查看次数:

77629 次

最近记录:

5 年,11 月 前