CSS
探讨一下,在多行文本情形下的一些有意思的文字动效。
多行文本,相对于单行文本,场景会复杂一些,但是在实际业务中,多行文本也是非常之多的,但是其效果处理比起单行文本会更困难。

单行与多行文本的渐隐

首先,来看这样一个例子,要实现这样一个单行文本的渐隐:
多行文本下的文字渐隐消失术 - 图1
使用 mask,可以轻松实现这样的效果,只需要:

  1. <p>Lorem ipsum dolor sit amet consectetur.</p>
  1. p {
  2. mask: linear-gradient(90deg, #fff, transparent);
  3. }

但是,如果,场景变成了多行呢?需要将多行文本最后一行,实现渐隐消失,并且适配不同的多行场景:
多行文本下的文字渐隐消失术 - 图2
这个就会稍微复杂一点点,但是也是有多种方式可以实现的。
首先来看一下使用 background 的方式。

使用 background 实现

这里会运用到一个技巧,就是 display: inline 内联元素的 background 展现形式与 display: block 块级元素(或者 inline-blockflexgrid)不一致。
简单看个例子:

  1. <p>Lorem .....</p>
  2. <a>Lorem .....</a>

这里需要注意,<p> 元素是块级元素,而 <a>内联元素
给它们统一添加上一个从绿色到蓝色的渐变背景色:

  1. p, a {
  2. background: linear-gradient(90deg, blue, green);
  3. }

看看效果:
多行文本下的文字渐隐消失术 - 图3
什么意思呢?区别很明显,块级元素的背景整体是一个渐变整体,而内联元素的每一行都是会有不一样的效果,整体连起来串联成一个整体。
基于这个特性,可以构造这样一种布局:

  1. <p><a>Mollitia nostrum placeat consequatur deserunt velit ducimus possimus commodi temporibus debitis quam</a></p>
  1. p {
  2. position: relative;
  3. width: 400px;
  4. }
  5. a {
  6. background: linear-gradient(90deg, transparent, transparent 70%, #fff);
  7. background-repeat: no-repeat;
  8. cursor: pointer;
  9. color: transparent;
  10. &::before {
  11. content: "Mollitia nostrum placeat consequatur deserunt velit ducimus possimus commodi temporibus debitis quam";
  12. position: absolute;
  13. top: 0;
  14. left: 0;
  15. color: #000;
  16. z-index: -1;
  17. }
  18. }

这里需要解释一下:

  1. 为了利用到实际的内联元素的 **background** 的特性,需要将实际的文本包裹在内联元素 **<a>**
  2. 实际的文本,利用了 **opacity: 0** 进行隐藏,实际展示的文本使用了 **<a>** 元素的伪元素,并且将它的层级设置为 **-1**,目的是让父元素的背景可以盖过它
  3. **<a>** 元素的渐变为从透明到白色,利用它去遮住下面的实际用伪元素展示的文字,实现文字的渐隐

这样,就能得到这样一种效果:
多行文本下的文字渐隐消失术 - 图4
这里,<a> 元素的渐变为从透明到白色,利用后面的白色逐渐遮住文字。
如果将渐变改为从黑色到白色(为了方便理解,渐变的黑色和白色都带上了一些透明),能很快的明白这是怎么回事:

  1. a {
  2. background: linear-gradient(90deg, rgba(0,0,0, .8), rgba(0,0,0, .9) 70%, rgba(255, 255, 255, .9));
  3. }

多行文本下的文字渐隐消失术 - 图5
完整的代码,可以戳这里:CodePen Demo — Text fades away
点击查看【codepen】
当然,这个方案有很多问题,譬如利用了 z-index: -1,如果父容器设置了背景色,则会失效,同时不容易准确定位最后一行。因此,更好的方式是使用 mask 来解决。

使用 mask 实现

那么,如果使用 mask 的话,问题,就会变得简单一些,只需要在一个 mask 中,实现两块 mask 区域,一块用于准确控制最后一行,一块用于控制剩余部分的透明。
也不需要特殊构造 HTML:

  1. <p>Lorem ipsum dolor sit amet ....</p>
  1. p {
  2. width: 300px;
  3. padding: 10px;
  4. line-height: 36px;
  5. mask:
  6. linear-gradient(270deg, transparent, transparent 30%, #000),
  7. linear-gradient(270deg, #000, #000);
  8. mask-size: 100% 46px, 100% calc(100% - 46px);
  9. mask-position: bottom, top;
  10. mask-repeat: no-repeat;
  11. }

效果如下:
多行文本下的文字渐隐消失术 - 图6
核心在于整个 mask 相关的代码,正如上面而言的,mask 将整个区域分成了两块进行控制:
多行文本下的文字渐隐消失术 - 图7
在下部分这块,利用 mask 做了从右向左的渐隐效果。并且利用了 mask-position 定位,以及 calc 的计算,无论文本都多少行,都是适用的!需要说明的是,这里的 46px 的意思是单行文本的行高加上 padding-bottom 的距离。可以适配任意行数的文本:
多行文本下的文字渐隐消失术 - 图8
完整的代码,可以戳这里:CodePen Demo — Text fades away 2
点击查看【codepen】

添加动画效果

好,看完静态的,再来实现一种动态的文字渐隐消失。
整体的效果是当鼠标 Hover 到文字的时候,整个文本逐行逐渐消失。像是这样:
多行文本下的文字渐隐消失术 - 图9
这里的核心在于,需要去适配不同的行数,不同的宽度,而且文字是一行一行的进行消失。
这里核心还是会运用上内联元素 background 的特性。
实现整段文字的渐现,从一种颜色到另外一种颜色**:

  1. <div class="button">Button</div>
  2. <p><a>Lorem ipsum dolor sit amet consectetur adipisicing elit. Mollitia nostrum placeat consequatur deserunt velit ducimus possimus commodi temporibus debitis quam, molestiae laboriosam sit repellendus sed sapiente quidem quod accusantium vero.</a></p>
  1. a {
  2. background:
  3. linear-gradient(90deg, #999, #999),
  4. linear-gradient(90deg, #fc0, #fc0);
  5. background-size: 100% 100%, 0 100px;
  6. background-repeat: no-repeat;
  7. background-position: 100% 100%, 0 100%;
  8. color: transparent;
  9. background-clip: text;
  10. }
  11. .button:hover ~ p a {
  12. transition: .8s all linear;
  13. background-size: 0 100px, 100% 100%;
  14. }

这里需要解释一下,虽然设置了 color: transparent,但是文字默认还是有颜色的,默认的文字颜色,是由第一层渐变赋予的 background: linear-gradient(90deg, #999, #999), linear-gradient(90deg, #fc0, #fc0),也就是这一层:linear-gradient(90deg, #999, #999)
多行文本下的文字渐隐消失术 - 图10
当 hover 触发时,linear-gradient(90deg, #999, #999) 这一层渐变逐渐消失,而另外一层 linear-gradient(90deg, #fc0, #fc0)`` 逐渐出现,借此实现上述效果。<br />[**CodePen -- background-clip 文字渐现效果**](https://codepen.io/Fcant/pen/OJvzYzV)<br />[点击查看【codepen】](https://codepen.io/Fcant/embed/OJvzYzV)<br />可以借鉴这个技巧,去实现文字的渐隐消失。一层为实际的文本,而另外一层是进行动画的遮罩,进行动画的这一层,本身的文字设置为color: transparent`,这样,就只能看到背景颜色的变化。
大致的代码如下:

  1. <p>
  2. <a class="word">Mollitia nostrum placeat consequatur deserunt.</a>
  3. <a class="pesudo">Mollitia nostrum placeat consequatur deserunt.</a>
  4. </p>
  1. p {
  2. width: 500px;
  3. }
  4. .word {
  5. position: absolute;
  6. top: 0;
  7. left: 0;
  8. color: transparent;
  9. color: #000;
  10. }
  11. .pesudo {
  12. position: relative;
  13. background: linear-gradient(90deg, transparent, #fff 20%, #fff);
  14. background-size: 0 100%;
  15. background-repeat: no-repeat;
  16. background-position: 100% 100%;
  17. transition: all 3s linear;
  18. color: transparent;
  19. }
  20. p:hover .pesudo,
  21. p:active .pesudo{
  22. background-size: 500% 100%;
  23. }

其中,.word 为实际在底部,展示的文字层,而 pesudo 为叠在上方的背景层,hover 的时候,触发上方元素的背景变化,逐渐遮挡住下方的文字,并且,能适用于不同长度的文本。
多行文本下的文字渐隐消失术 - 图11
当然,上述方案会有一点瑕疵,无法让不同长度的文本整体的动画时间一致。当文案数量相差不大时,整体可以接受,文案相差数量较大时,需要分别设定下 transition-duration 的时长。
完整的 DEMO可以看:CodePen — Text fades away Animation
点击查看【codepen】

参考资料

CodePen Demo — Text fades away:https://codepen.io/Fcant/pen/ZExvNXX
CodePen Demo — Text fades away 2:https://codepen.io/Fcant/pen/vYRpwew
CodePen — background-clip 文字渐现效果:https://codepen.io/Fcant/pen/OJvzYzV
CodePen — Text fades away Animation:https://codepen.io/Fcant/pen/QWmaRvE