我正试着复制这个心形的"赞"按钮:

Screenshot of two buttons, one an outline of a heart for the inactive state and the other a filled heart for the hovering state.

(顶部处于非活动状态,底部处于悬停状态.)

悬停必须使用过渡设置动画,如下所示:

Animation showing hover transition.

我试过使用字体和material 图标,但我想不出如何使用它们来进行适当的过渡.如果为background-color,将保留图标的边框.

推荐答案

对于这样的图标,由于多种原因,使用可伸缩向量图形(Scalable Vector Graphics,SVG)是理想的,但在这种特定情况下尤其如此,因为您可以使用CSS来设置SVG的各个元素的样式.

.heart .heart-fill {
  /* notice the defined alpha component. you could also use opacity though, of course */
  fill: #00000000;
}

.heart .heart-outline {
  fill: #000000;
}

.heart .heart-outline,
.heart .heart-fill {
  transition-property: fill;
  transition-duration: 0.3s;
  transition-timing-function: ease-in-out;
}

#heart {
  width: 300px;
  height: auto;
}

#heart:hover {
  cursor: pointer;
}


/* this uses the bounding box of the whole svg for hovering, which is perfectly fine for small buttons, but may be slightly problematic for using this svg in a larger size (.heart-fill:hover + .heart-outline should work for that case though) */
#heart:hover .heart-fill,
#heart:hover .heart-outline {
  fill: #ff0000;
}
<!-- container -->
<div id="heart">
  <!-- source: https://www.svgrepo.com/svg/13666/heart -->
  <!-- edited with Adobe Illustrator to add the solid section -->
  <svg class="heart" version="1.1" id="Capa_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" viewBox="0 0 800 800" style="enable-background:new 0 0 800 800;" xml:space="preserve">
<path class="heart-fill" d="M735.4,113.6C693.5,71.7,638,48.8,578.8,48.8s-114.8,23.1-156.7,65l-21.9,21.9L378,113.5c-41.9-41.9-97.7-65.1-156.9-65.1
    c-59,0-114.6,23.1-156.4,64.8C22.9,155-0.2,210.6,0,269.8C0,329,23.2,384.5,65.1,426.4l318.5,318.5c4.4,4.4,10.3,6.8,16.1,6.8
    s11.7-2.2,16.1-6.6l319.2-318c41.9-41.9,65-97.5,65-156.7C800.2,211.2,777.3,155.5,735.4,113.6z"/>
<path class="heart-outline" d="M735.4,113.6C693.5,71.7,638,48.8,578.8,48.8c-59.2,0-114.8,23.1-156.7,65l-21.9,21.9L378,113.5
    c-41.9-41.9-97.7-65.1-156.9-65.1c-59,0-114.6,23.1-156.4,64.8C22.9,155-0.2,210.6,0,269.8C0,329,23.2,384.5,65.1,426.4l318.5,318.5
    c4.4,4.4,10.3,6.8,16.1,6.8c5.8,0,11.7-2.2,16.1-6.6l319.2-318c41.9-41.9,65-97.5,65-156.7C800.2,211.2,777.3,155.5,735.4,113.6z
     M702.8,394.7L399.7,696.5L97.4,394.1c-33.2-33.2-51.6-77.3-51.6-124.3s18.1-91.1,51.4-124.1c33.1-33.1,77.2-51.4,124-51.4
    c47,0,91.2,18.3,124.5,51.6l38.3,38.3c9,9,23.4,9,32.4,0l38-38c33.2-33.2,77.5-51.6,124.3-51.6c46.8,0,90.9,18.3,124.1,51.4
    c33.2,33.2,51.4,77.3,51.4,124.3C754.4,317.3,736.1,361.4,702.8,394.7z"/>
</svg>
</div>

这样做的唯一缺点是,您实际上需要在任何地方都包含svg标记,包括它的所有数据.不过,样式可以在外部样式表中.将SVG包含为imgobject将阻止您与其样式进行交互.

您还需要在SVG中有两个单独的元素(轮廓/边框和填充形状),因为CSSborderoutline不会应用于这些元素,而是它们的边界框.当然,stroke属性可以在填充的形状元素上很好地工作,but you have no control over the direction of it(就像在路径中应用笔触),所以它不是很理想.

另一种解决方案是使用堆叠在一起的普通光栅化图像,并设置其不透明度的样式:

.heart-fill {
  opacity: 0;
}

.heart-outline {
  opacity: 1;
}

.heart-fill, .heart-outline {
  transition-property: opacity;
  transition-duration: 0.3s;
  transition-timing-function: ease-in-out;
}

#heart {
  width: 300px;
  height: auto;

  position: relative;
}

#heart .heart-outline,
#heart .heart-fill {
  width: 100%;
  height: auto;
}

#heart .heart-fill {
  position: absolute;
  top: 0;
  left: 0;
  
  pointer-events: none;
  
  z-index: 1;
}

#heart:hover {
  cursor: pointer;
}

#heart:hover .heart-fill {
  opacity: 1;
}

#heart:hover .heart-outline {
  opacity: 0;
}
<!-- container -->
<div id="heart">
  <!-- source: https://static.thenounproject.com/png/2175990-200.png -->
  <!-- edited with Adobe Photoshop to create the border version, then encoded as inline Base64 png -->
  <img class="heart-fill" type="image/png" src="" />
  <img class="heart-outline" type="image/png" src="" />
</div>

不过,我绝对推荐使用SVG方法,因为它更灵活,而且还具有矢量图像的优势.向SVG添加渐变(如您的示例所示)就像将.heart-fill类的background添加一样简单.当然,您也可以将其设置为动画.

编辑:嗯,不完全是.我记错了.在SVG的defs中定义一个SVG linearGradient element,然后将其用作要应用渐变的形状的fill样式(通过CSSurl()和渐变的ID).

/* need to use opacity for the transitions now, as fill transitions don't play nice with gradients */
.heart-fill {
  opacity: 0;
  fill: url(#heart-gradient);
}

.heart-outline {
  opacity: 1;
  fill: #000000;
}

.heart-outline,
.heart-fill {
  transition-property: opacity;
  transition-duration: 0.3s;
  transition-timing-function: ease-in-out;
}

.heart-fill {
  background:
}

#heart {
  width: 300px;
  height: auto;
}

#heart:hover {
  cursor: pointer;
}

#heart:hover .heart-fill {
  opacity: 1;
}

#heart:hover .heart-outline {
  opacity: 0;
}
<div id="heart">
  <!-- source: https://www.svgrepo.com/svg/13666/heart -->
  <!-- edited with Adobe Illustrator to add the solid section -->
  <svg class="heart" version="1.1" id="Capa_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" viewBox="0 0 800 800" style="enable-background:new 0 0 800 800;" xml:space="preserve">
<defs>
    <linearGradient id="heart-gradient" gradientTransform="rotate(90)">
      <stop offset="5%" stop-color="#ff7f00" />
      <stop offset="95%" stop-color="#ff0000" />
    </linearGradient>
</defs>
<path class="heart-fill" d="M735.4,113.6C693.5,71.7,638,48.8,578.8,48.8s-114.8,23.1-156.7,65l-21.9,21.9L378,113.5c-41.9-41.9-97.7-65.1-156.9-65.1
    c-59,0-114.6,23.1-156.4,64.8C22.9,155-0.2,210.6,0,269.8C0,329,23.2,384.5,65.1,426.4l318.5,318.5c4.4,4.4,10.3,6.8,16.1,6.8
    s11.7-2.2,16.1-6.6l319.2-318c41.9-41.9,65-97.5,65-156.7C800.2,211.2,777.3,155.5,735.4,113.6z"/>
<path class="heart-outline" d="M735.4,113.6C693.5,71.7,638,48.8,578.8,48.8c-59.2,0-114.8,23.1-156.7,65l-21.9,21.9L378,113.5
    c-41.9-41.9-97.7-65.1-156.9-65.1c-59,0-114.6,23.1-156.4,64.8C22.9,155-0.2,210.6,0,269.8C0,329,23.2,384.5,65.1,426.4l318.5,318.5
    c4.4,4.4,10.3,6.8,16.1,6.8c5.8,0,11.7-2.2,16.1-6.6l319.2-318c41.9-41.9,65-97.5,65-156.7C800.2,211.2,777.3,155.5,735.4,113.6z
     M702.8,394.7L399.7,696.5L97.4,394.1c-33.2-33.2-51.6-77.3-51.6-124.3s18.1-91.1,51.4-124.1c33.1-33.1,77.2-51.4,124-51.4
    c47,0,91.2,18.3,124.5,51.6l38.3,38.3c9,9,23.4,9,32.4,0l38-38c33.2-33.2,77.5-51.6,124.3-51.6c46.8,0,90.9,18.3,124.1,51.4
    c33.2,33.2,51.4,77.3,51.4,124.3C754.4,317.3,736.1,361.4,702.8,394.7z"/>
</svg>
</div>

Html相关问答推荐

如何通过将项目包裹在迪夫中来保持最右列的对齐方式来对齐项目?

springBoot + Thymeleaf:基于Locale设置页面语言

将多个内联元素置于中心,而无需访问父级上的CSS

如何将grid—template—column应用于元素中的子元素

如何获得一个背景图像,不是模糊的内部容器,但模糊的外部'

仅包含名为X的子元素的元素的XPath?

天使17:令人惊叹的动画

仅向上扩展并以最大高度滚动的Div

如何使用Reaction Js读取html文件中的代码

旧文本淡出,但新文本不会淡入,而是突然出现

div居中碰撞问题

如何在CSS中延迟触发2个转换?

如何在css中在span中绘制圆弧?

带有图像的虚线边框

使用shinyjqui拖放到网格表中

如何居中此按钮,即使它已经在计算机分辨率中居中

Django 如何从文件系统下载文件?

当我将有序列表居中时,数字会跑到列表的另一行

如何更改 Quarto 中标签crossref-def-title的标题 colored颜色 ?

从垫分页器中删除输入边框