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

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

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

  • 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. <div class="wrap"> 
  2.  
  3.   <div class="prop">1.prop<br>float:left</div> 
  4.  
  5.   <div class="main">2.main<br>float:right<br>Fairly short text</div> 
  6.  
  7.   <div class="end">3.end<br>float:right</div> 
  8.  
  9. </div> 
  10.  
  11.   
  12.  
  13. .wrap { 
  14.  
  15.   width: 400px; height: 200px; 
  16.  
  17.     margin: 20px 20px 50px; 
  18.  
  19.     border: 5px solid #AAA; 
  20.  
  21.     line-height: 25px; 
  22.  
  23.  
  24.   
  25.  
  26. .prop { 
  27.  
  28.     float: left; 
  29.  
  30.     width: 100px; height: 200px; 
  31.  
  32.     background: #FAF; } 
  33.  
  34. .main { 
  35.  
  36.     float: right; 
  37.  
  38.     width: 300px; 
  39.  
  40.     background: #AFF; } 
  41.  
  42. .end { 
  43.  
  44.     float: right; 
  45.  
  46.     width: 100px; 
  47.  
  48.     background: #FFA; }  

2nd 模拟场景

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


  1. <div class="wrap"> 
  2.  
  3.   <div class="prop"> 
  4.  
  5.    1.prop<br> 
  6.  
  7.    float:right</div> 
  8.  
  9.   <div class="main"> 
  10.  
  11.    2.main<br> 
  12.  
  13.    float:left<br> 
  14.  
  15.    Fairly short text</div> 
  16.  
  17.   <div class="end"> 
  18.  
  19.    <div class="realend"> 
  20.  
  21.      4.realend<br> 
  22.  
  23.      position:absolute</div> 
  24.  
  25.   3.end<br>float:right 
  26.  
  27.   </div> 
  28.  
  29. </div> 
  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. <div class="wrap"> 
  2.  
  3.   <div class="prop">1.prop<br>float:right</div> 
  4.  
  5.   <div class="main">2.main<br>float:left<br>Fairly short text</div> 
  6.  
  7.   <div class="realend"> 
  8.  
  9.   3.realend<br>position:relative</div> 
  10.  
  11. </div> 
  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. <div class="wrap"> 
  2.  
  3.   <div class="prop">1.prop<br>float:right</div> 
  4.  
  5.   <div class="main">2.main<br>float:left<br>Fairly short text</div> 
  6.  
  7.   <div class="realend"> 
  8.  
  9.   3.realend<br>position:relative</div> 
  10.  
  11. </div> 
  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. <div class="ellipsis"> 
  2.  
  3.   <div>2.main<br>float:left<br>Fairly short text 
  4.  
  5.   </div> 
  6.  
  7. </div> 
  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(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAgAAAAABCAMAAACfZeZEAAAABGdBTUEAALGPC/xhBQAAAwBQTFRF////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////AAAA////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////wDWRdwAAAP90Uk5TgsRjMZXhS30YrvDUP3Emow1YibnM9+ggOZxrBtpRRo94gxItwLOoX/vsHdA2yGgL8+TdKUK8VFufmHSGgAQWJNc9tk+rb5KMCA8aM0iwpWV6dwP9+fXuFerm3yMs0jDOysY8wr5FTldeoWKabgEJ8RATG+IeIdsn2NUqLjQ3OgBDumC3SbRMsVKsValZplydZpZpbJOQco2KdYeEe36BDAL8/vgHBfr2CvTyDu8R7esU6RcZ5ecc4+Af3iLcJSjZ1ivT0S/PMs3LNck4x8U7wz7Bv0G9RLtHuEq1TbJQr1OtVqqnWqRdoqBhnmSbZ5mXapRtcJGOc4t2eYiFfH9AS7qYlgAAARlJREFUKM9jqK9fEGS7VNrDI2+F/nyB1Z4Fa5UKN4TbbeLY7FW0Tatkp3jp7mj7vXzl+4yrDsYoVx+JYz7mXXNSp/a0RN25JMcLPP8umzRcTZW77tNyk63tdprzXdmO+2ZdD9MFe56Y9z3LUG96mcX02n/CW71JH6Qmf8px/cw77ZvVzB+BCj8D5vxhn/vXZh6D4uzf1rN+Cc347j79q/zUL25TPrJMfG/5LvuNZP8rixeZz/mf+vU+Vut+5NL5gPOeb/sd1dZbTs03hBuvmV5JuaRyMfk849nEM7qnEk6IHI8/qn049hB35QGHiv0yZXuMdkXtYC3ebrglcqvYxoj1muvC1nDlrzJYGbpcdHHIMo2FwYv+j3QAAOBSfkZYITwUAAAAAElFTkSuQmCC); 
  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规范中的浮动+定位+盒模型宽度计算,唯一存在兼容性问题的在于无关痛痒的渐变实现,因此可以在大多数浏览器下进行尝试。

作者:伯乐在线

来源:51CTO

时间: 2024-09-30 04:06:00

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

如何用CSS控制网页背景

在符合标准的网页设计中如何用CSS控制网页的背景呢?包括背景的颜色,背景的图片等一些问题,在本教程中一次给你讲清楚. ·背景颜色 background-color 我想这个我就不用多做介绍了,颜色代码我想大家都知道的,不是用英文来代替就是用指定的代码来表示的.这个的默认值是transparent(透明色). 例:body{background-color:yellow} H1{background-color:#000000} ·背景图片 background-image 背景图片和背景颜色在H

如何用CSS实现大背景网页设计

自从我发布了大背景网页设计合集之后,我收到了很多电子邮件询问如何用css实现大背景的网页设计.因此我决定和大家分享大背景网站的设计技巧.在此教程中,我会用一些实例来说明如何用一张或者两张图片实现大背景网站的设计. 经常会犯的错误:背景被裁减(查看示例) 查看示例文件,在小于1280分辨率时,是没有问题的.但是如果你的显示器的分辨率大于1280像素,你会看到背景图片以外的部分. 实例1:一张图片(查看示例) 简单的解决问题的方法:将图片边缘的颜色设置成和网页背景色相同的颜色.这里我用Web Des

CSS技巧分享:如何用css制作横排二级下拉菜单

原文:CSS技巧分享:如何用css制作横排二级下拉菜单 导航菜单是每个网站所必备的功能,也是每个学习制作网站的朋友所必须接触的,如何用css样式制作一个简单漂亮的二级下拉菜单呢,下来小编就一步一步教大家来制作横排二级下拉菜单,让我共同来学习吧. 首页我们打看dreamweaver或其它编辑器,创建一个名为nav的导航菜单 <div class="nav"> <ul> <li><a href="#">栏目一</a&

js-如何用CSS或者JS隐藏这样一个标签,&amp;amp;lt;label for=&amp;amp;quot;notebutton&amp;amp;quot;&amp;amp;gt;VJSP批注模式&amp;amp;lt;/label&amp;amp;gt;

问题描述 如何用CSS或者JS隐藏这样一个标签,<label for="notebutton">VJSP批注模式</label> 完整形式 VJSP批注模式 解决方案 没人知道吗,真心求教啊

效果 显示 css jquery-请问如何用css/Jquery实现大的鼠标,以及实现点击表哥高亮度显示

问题描述 请问如何用css/Jquery实现大的鼠标,以及实现点击表哥高亮度显示 最近上面给个任务,要求做一个layout如下图所示 如图所示,点击时要显示这个大的箭头光标,并且点击的那个表格要高亮显示.而原来的表格就像它下面两个,是看上去模糊的,我也不知道这种模糊的效果在网页上是怎么实现的? 盼望园子里的高手不吝赐教,谢谢 解决方案 模糊效果是用滤镜做的...不同浏览器有不同的关键字.. 解决方案二: css就可以..需要注意添加xhtml申明,要不cursor不支持多个候选资源设置 <!DO

如何用CSS把flash添加到网页中?

问题描述 如何用CSS把flash添加到网页中? 解决方案 解决方案二:CSS不知道javascript直接操作document对象添加即可解决方案三:该回复于2008-06-30 09:26:34被版主删除

用CSS实现标题过长部分用省略号显示

使用方法:在CSS中同时应用width=""; overflow:hidden; text-overflow:ellipsis; white-space:nowrap;才能达到效果. width="";定义显示宽度 overflow:hidden; 溢出隐藏 text-overflow:ellipsis;省略号显示溢出文本内容 white-space:nowrap; 强制文本在一行内显示 用width指它好宽度.overflow:hidden;将超出的内容隐藏就行了

如果文字多了,想让超出元素外面的文字以省略号显示的css样式

请看下面的demo <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>省略号显示</title> <style type="text/css"> .box{ width: 200px; height: 20px; line-height: 20px; background:

Js与css实现文本溢出自动添加省略号方法总结

1.常规css方法--可以实现IE,Safari,chrome,opera浏览器下文字溢出省略号表示 这是一段测试文字,主要是用来测试文字溢出后是否会用省略号显示.  代码如下 复制代码 .zxx_text_overflow_1{width:27em; white-space:nowrap; text-overflow:ellipsis; -o-text-overflow:ellipsis; overflow:hidden;} 2.使用ellipsis.xml文件使Firefox支持文字溢出后点