如何用CSS实现多行文本的省略号显示

本文翻译自CSS Ellipsis: How to Manage Multi-Line Ellipsis in Pure CSS,文中某些部分有些许改动,并添加译者的一些感想,请各位读者谅解。

我们提供的服务有:网站设计、成都做网站、微信公众号开发、网站优化、网站认证、织金ssl等。为千余家企事业单位解决了网站和推广的问题。提供周到的售前咨询和贴心的售后服务,是有科学管理、有技术的织金网站制作公司

合理的截断多行文本是件不容易的事情,我们通常采用几种方法解决:

  • overflow: hidden直接隐藏多余的文本
  • text-overflow: ellipsis只适用于单行文本的处理
  • 各种比较脆弱的javascript实现。之所以说这种实现比较脆弱是由于需要文本长度的变化时刻得到回流(relayout)后的布局信息,如宽度

英文原文写作时间是2012.9.18号,比较有意义的一天。不过作者忽略了WebKit提供的一个扩展属性-webkit-line-clamp,它并不是CSS规范中的属性。利用该属性实现多行文本的省略号显示需要配合其他三个属性:display: -webkit-box、-webkit-box-orient、text-overflow: ellipsis;。其中,-webkit-line-clamp设置块元素包含的文本行数;display: -webkit-box设置块元素的布局为伸缩布局;-webkit-box-orient设置伸缩项的布局方向;text-overflow: ellipsis;则表示超出盒子的部分使用省略号表示。

不过本文将要介绍的方法是采用CSS规范中的属性,并结合特殊的实现技巧完成的。这意味着在实现CSS2.1规范的浏览器中都是可以兼容的,不将仅仅是纯粹的移动端领域,在传统的PC浏览器(你们懂得我指的是哪些浏览器)中仍是可行的。好吧,让我们一起见识下。

CSS实现多行文本溢出的省略号显示

我们把实现的细节划分为7个步骤,在这个实现过程中最简单的就是截断文本,而最难的部分则是让一个元素处在其父包含块溢出时的右下方,并且当父元素未溢出时该元素消失不可见。为了去难避易,我们先从比较简单的地方开始–当父包含框比较小时,将子元素布局到父包含框的右下角。

1st 引子 

  

其实这个实现完全利用了元素浮动的基本规则。在这里不详细讲解CSS2.1规范中的几种情形,不明白的读者自行查阅。这段代码实现很简单,就是三个子元素和包含块的高度及浮动设置:

 
 
 
 
  1.  
  2.  
  3.   1.prop
    float:left
 
  •  
  •   2.main
    float:right
    Fairly short text
  •  
  •  
  •   3.end
    float:right
  •  
  •  
  •  
  •  
  •   
  •  
  • .wrap { 
  •  
  •   width: 400px; height: 200px; 
  •  
  •     margin: 20px 20px 50px; 
  •  
  •     border: 5px solid #AAA; 
  •  
  •     line-height: 25px; 
  •  
  •  
  •   
  •  
  • .prop { 
  •  
  •     float: left; 
  •  
  •     width: 100px; height: 200px; 
  •  
  •     background: #FAF; } 
  •  
  • .main { 
  •  
  •     float: right; 
  •  
  •     width: 300px; 
  •  
  •     background: #AFF; } 
  •  
  • .end { 
  •  
  •     float: right; 
  •  
  •     width: 100px; 
  •  
  •     background: #FFA; }  
  • 2nd 模拟场景

    我们通过创建一个子元素来替代将要显示的省略号,当文本溢出的情形下该元素显示在正确的位置上。在接下来的实现中,我们创建了一个realend元素,并利用上一节end元素浮动后的位置来实现realend元素的定位。

     
     
     
     
    1.  
    2.  
    3.    
    4.  
    5.    1.prop
       
    6.  
    7.    float:right 
    8.  
    9.    
    10.  
    11.    2.main
       
    12.  
    13.    float:left
       
    14.  
    15.    Fairly short text 
    16.  
    17.    
    18.  
    19.     
    20.  
    21.      4.realend
       
    22.  
    23.      position:absolute 
    24.  
    25.   3.end
      float:right 
    26.  
    27.    
    28.  
    29.  
    30.  
    31.   
    32.  
    33. .end { 
    34.  
    35.     float: right; position: relative; 
    36.  
    37.     width: 100px; 
    38.  
    39.     background: #FFA; } 
    40.  
    41. .realend { 
    42.  
    43.     position: absolute; 
    44.  
    45.     width: 100%; 
    46.  
    47.     top: -50px; 
    48.  
    49.     left: 300px; 
    50.  
    51.     background: #FAA; font-size: 13px; }  

    这一步中,我们主要关心的是realend元素的定位,基于浮动后的end元素设置偏移量,当end元素浮动到***节第二章图的位置时(即在prop元素的下方),此时realend元素正好处在end元素的上方50px,右侧300px-100px=200px处,而该位置正是父包含框wrap元素的右下角,此时正是我们期待的结果: 

    若父元素并没有溢出,那么realend元素会出现在其右侧 

    这种情况解决很简单,请看下文之第七节,此处仅作实例说明。

    3rd 优化定位模型

    在第二节中,我们针对end元素设置了相对定位,对realend元素设置绝对定位。但是我们可以采用更为简单的代码来实现,即只使用相对定位。熟悉定位模型的同学应该知道,相对定位的元素仍然占据文本流,同时仍可针对元素设置偏移。这样,就可以去掉end元素,仅针对realend元素设置相对定位。

     
     
     
     
    1.  
    2.  
    3.   1.prop
      float:right 
    4.  
    5.   2.main
      float:left
      Fairly short text 
    6.  
    7.    
    8.  
    9.   3.realend
      position:relative 
    10.  
    11.  
    12.  
    13.   
    14.  
    15. .realend { 
    16.  
    17.     float: right; 
    18.  
    19.         position: relative; 
    20.  
    21.     width: 100px; 
    22.  
    23.     top: -50px; left: 300px; 
    24.  
    25.     background: #FAA; font-size: 14px; }  

    其他的属性并不改变,效果一样。

    4th 削窄prop元素

    目前,最左侧的prop元素的作用在于让realend元素在文本溢出时处在其正下方,在前几节的示例代码中为了直观的演示,设置prop元素的宽度为100px,那么现在为了更好的模拟实际的效果,我们缩小逐渐缩小prop元素的宽度。

     
     
     
     
    1.  
    2.  
    3.   1.prop
      float:right 
    4.  
    5.   2.main
      float:left
      Fairly short text 
    6.  
    7.    
    8.  
    9.   3.realend
      position:relative 
    10.  
    11.  
    12.  
    13.   
    14.  
    15.   
    16.  
    17. .prop { 
    18.  
    19.   float: left; 
    20.  
    21.   width: 5px; 
    22.  
    23.   height: 200px; 
    24.  
    25.   background: #F0F; } 
    26.  
    27. .main { 
    28.  
    29.     float: right; 
    30.  
    31.     width: 300px; 
    32.  
    33.     margin-left: -5px; 
    34.  
    35.     background: #AFF; } 
    36.  
    37. .realend { 
    38.  
    39.     float: right; 
    40.  
    41.         position: relative; 
    42.  
    43.     top: -50px; 
    44.  
    45.         left: 300px; 
    46.  
    47.     width: 100px; 
    48.  
    49.         margin-left: -100px; 
    50.  
    51.     padding-right: 5px; 
    52.  
    53.     background: #FAA; font-size: 14px; }  

    针对prop元素,缩小宽度为5px,其余属性不变;

    针对main元素,设置margin-left:5px,让main元素左移5px,这样main元素在宽度上就完全占满了父元素;

    对于realend元素,top、left和width的值不变。而设置margin-left: -100px、padding-right: 5px则是为了让realend元素的盒模型的最终宽度计算为5px。

     
     
     
     
    1. BoxWidth = ChildMarginLeft + ChildBorderLeftWidth + ChildPaddingLeft + ChildWidth + ChildPaddingLeft + ChildBorderRightWidth + ChildMarginRightWidth; 

    具体可参考我之前的文章负margin的原理以及应用一文。

    由于CSS规范规定padding的值不可以为负数,因此只有设置margind-left为负值,且等于其宽度。这样做的最终目的就是保证realend元素处在prop元素的下方,保证在文本溢出的情况下定位准确性: 

    5th 继续优化:流式布局+伪元素

    目前,realend元素的相关属性仍采用px度量,为了更好的扩展性,可以改用%替代。

    同时,prop元素和realend元素可以采用伪元素来实现,减少额外标签的使用。

     
     
     
     
    1.  
    2.  
    3.   
      2.main
      float:left
      Fairly short text 
    4.  
    5.    
    6.  
    7.  
    8.  
    9.   
    10.  
    11. /*相当于之前的prop元素*/ 
    12.  
    13. .ellipsis:before { 
    14.  
    15.     content: ""; 
    16.  
    17.     float: left; 
    18.  
    19.     width: 5px; height: 200px; 
    20.  
    21.     background: #F0F; } 
    22.  
    23. /*相当于之前的main元素*/ 
    24.  
    25. .ellipsis > *:first-child { 
    26.  
    27.     float: right; 
    28.  
    29.     width: 100%; 
    30.  
    31.     margin-left: -5px; 
    32.  
    33.     background: #AFF; } 
    34.  
    35. /*相当于之前的realend元素*/ 
    36.  
    37. .ellipsis:after { 
    38.  
    39.     content: "realend"; 
    40.  
    41.     float: right; position: relative; 
    42.  
    43.     top: -25px; left: 100%; 
    44.  
    45.     width: 100px; margin-left: -100px; 
    46.  
    47.     padding-right: 5px; 
    48.  
    49.     background: #FAA; font-size: 14px; }  

    效果图和上节一样。

    6th 隐藏

    之前的实现中在文本未溢出的情况下,realend元素会出现在父元素的右侧,正如。 

    解决此问题很简单,急需要设置:

     
     
     
     
    1. .ellipsis{ 
    2.  
    3.     overflow:hidden; 
    4.  
    5. }  

    即可解决问题。

    7th 大功告成

    现在我们离完结就差一步了,即去掉各元素的背景色,并且用“…”替换文本。***为了优化体验,采用渐变来隐藏“…”覆盖的文本,并设置了一些兼容性的属性。

    到了此处,相信现在关心的只是CSS的代码了:

     
     
     
     
    1. .ellipsis { 
    2.  
    3.   overflow: hidden; 
    4.  
    5.   height: 200px; 
    6.  
    7.     line-height: 25px; 
    8.  
    9.     margin: 20px; 
    10.  
    11.     border: 5px solid #AAA; } 
    12.  
    13.   
    14.  
    15. .ellipsis:before { 
    16.  
    17.     content:""; 
    18.  
    19.     float: left; 
    20.  
    21.     width: 5px; height: 200px; } 
    22.  
    23.   
    24.  
    25. .ellipsis > *:first-child { 
    26.  
    27.     float: right; 
    28.  
    29.     width: 100%; 
    30.  
    31.     margin-left: -5px; }         
    32.  
    33.   
    34.  
    35. .ellipsis:after { 
    36.  
    37.     content: "\02026";   
    38.  
    39.   
    40.  
    41.     box-sizing: content-box; 
    42.  
    43.     -webkit-box-sizing: content-box; 
    44.  
    45.     -moz-box-sizing: content-box; 
    46.  
    47.   
    48.  
    49.     float: right; position: relative; 
    50.  
    51.     top: -25px; left: 100%; 
    52.  
    53.     width: 3em; margin-left: -3em; 
    54.  
    55.     padding-right: 5px; 
    56.  
    57.      
    58.  
    59.     text-align: right; 
    60.  
    61.   
    62.  
    63.    
    64.  
    65.         background-size: 100% 100%; 
    66.  
    67.   /* 512x1 image, gradient for IE9. Transparent at 0% -> white at 50% -> white at 100%.*/ 
    68.  
    69. background-image: url(); 
    70.  
    71.    
    72.  
    73.     background: -webkit-gradient(linear, left top, right top, 
    74.  
    75.         from(rgba(255, 255, 255, 0)), to(white), color-stop(50%, white)); 
    76.  
    77.     background: -moz-linear-gradient(to right, rgba(255, 255, 255, 0), white 50%, white);           
    78.  
    79.     background: -o-linear-gradient(to right, rgba(255, 255, 255, 0), white 50%, white); 
    80.  
    81.     background: -ms-linear-gradient(to right, rgba(255, 255, 255, 0), white 50%, white); 
    82.  
    83.     background: linear-gradient(to right, rgba(255, 255, 255, 0), white 50%, white); 
    84.  

     总结之兼容性

    从上文的实现细节来看,我们利用的技巧完全是CSS规范中的浮动+定位+盒模型宽度计算,唯一存在兼容性问题的在于无关痛痒的渐变实现,因此可以在大多数浏览器下进行尝试。 

    网站栏目:如何用CSS实现多行文本的省略号显示
    链接地址:http://www.mswzjz.cn/qtweb/news45/556595.html

    攀枝花网站建设、攀枝花网站运维推广公司-贝锐智能,是专注品牌与效果的网络营销公司;服务项目有等

    广告

    声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 贝锐智能

    贝锐智能技术为您推荐以下文章

    定制网站知识

    分类信息网站