1

I'm trying to reproduce this heart-shaped "like" button:

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

(Inactive state at the top, hovering state at the bottom.)

The hover has to be animated with a transition, like this:

Animation showing hover transition.

I've tried using Font Awesome and Material Icons, but I can't figure out how to do a proper transition with them. background-color leaves the borders of the icon.

Darryl Noakes
  • 2,207
  • 1
  • 9
  • 27
8bitmd
  • 29
  • 8

2 Answers2

2

For icons like these, using Scalable Vector Graphics (SVGs) is ideal for a multitude of reasons, but in this specific case especially so because of the fact you can use CSS to style individual elements of the 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>

The only downside to this is that you need to actually include the svg tag everywhere, including all it's data. The styles can be in an external stylesheet though. Including the SVG as an img or an object would prevent you from interacting with it's styles.

You also need to have two separate elements (the outline/border, and the filled in shape) inside the SVG, as CSS border or outline do not get applied to those elements, but their bounding box instead. There is of course the stroke attribute that would work fine on the filled shape element, but you have no control over the direction of it (like applying the stroke inside the path), so it's not ideal.

The other solution would be using normal rasterized images stacked on top of one another and styling their opacity:

.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>

I'd definitely recommend using the SVG approach though, as it is just so much more flexible, and also has the advantage of being a vector image. Adding a gradient (like in your example) to the SVG would be as simple as adding a CSS linear-gradient to the background of the .heart-fill class. And you could of course animate that too.

EDIT: Well, not quite. I misremembered that. You define an SVG linearGradient element inside the defs of the SVG, then use it as the fill style of the shape you want to apply the gradient on (by way of CSS url() and the id of the gradient).

/* 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>
TARN4T1ON
  • 522
  • 2
  • 12
  • 1
    That was incredibly helpful, thank you. I've been trying to implement this for at least 5 hours... Small problem: I can't seem to be able to add a gradient to `.heart-fill` with `linear-gradient`, I'm using the SVG method you mentioned. – 8bitmd Aug 23 '23 at 15:22
  • 1
    @8bitmd Oop, that's because I misremembered how linear gradients work for SVGs haha. You have to define a ```linearGradient``` element inside the SVG, give it an id, and then use ```url(#{id})``` as the value for the ```fill``` property of the shape you want to use it on. I'll add an example to my answer in a moment. – TARN4T1ON Aug 23 '23 at 15:37
  • 1
    @8bitmd Alright, I fixed the description and added an example. – TARN4T1ON Aug 23 '23 at 15:52
  • Done! Honestly, thank you so much. There's an extra issue I'm facing since apparently you can't transition gradients... I'm thinking about adding a second SVG with the gradient below the first SVG and transitioning the opacity instead, thoughts? – 8bitmd Aug 23 '23 at 15:53
  • 1
    @8bitmd I actually already did that in the gradient example. Since you have the outline and filled shape as 2 separate shapes in the SVG already, you can change their opacities independently too. Instead of transitioning the fill, you transition the opacity, and set the fill in the base class. No need for a second SVG, although that would work too of course. – TARN4T1ON Aug 23 '23 at 15:57
  • 1
    Oh, sorry, I didn't see your third example at first. Thank you so much again, truly appreciate it. I can finally take a break haha – 8bitmd Aug 23 '23 at 16:09
1

I figure 2 ways of doing it, the first one:

In this way, you have to "draw" the heart manually with SVG, and then work on hover with css.

.icon-container {
      display: inline-block;
        position: relative;
        overflow: hidden;
        width: 150px;
        height: 150px;
  }

  .icon {
    width: 100%;
    height: 100%;
    fill: white; 
    stroke: black; 
    stroke-width: 0.5; 
    transition: fill 0.3s ease;
   }
  .icon-container:hover .icon path {
fill: red;
stroke: red;
  }
<body>
  <div class="icon-container">
   <svg class="icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24">
      <path d="M12 21.35l-1.45-1.32C5.4 16.36 2 13.28 2 9.5 2 6.42 4.42 4 7.5 4c1.74 0 3.41.81 4.5 2.09C15.09 4.81 16.76 4 18.5 4 21.58 4 24 6.42 24 9.5c0 3.78-3.4 6.86-8.55 10.54L12 21.35z"/>
    </svg>
  </div>
</body>

The second one:

That's the one i prefer, but you have to work with a PNG who have the transparency inside the heart, or you'll get (as the example) the entire image container coloured.

.icon-container {
    display: inline-block;
    position: relative;
    overflow: hidden;
    width: 150px;
    height: 150px;
  }

  .icon {
    width: 100%;
    height: 100%;
    background-image: url('https://cdn-icons-png.flaticon.com/512/1077/1077035.png');
    background-size: cover;
    transition: background-color 0.3s ease;
  }

  .icon-container:hover .icon {
    background-color: red;
  }
<body>
  <div class="icon-container">
    <div class="icon"></div>
  </div>
</body>

Hope it helps!

DarioS
  • 111
  • 1
  • 13
  • 1
    The PNG way really isn't nice... It won't work with icon libraries, it's hard to keep the background inside the outline (yours doesn't), and the outline doesn't change color. – Darryl Noakes Aug 23 '23 at 16:20
  • @DarrylNoakes you're right, wanted just to give some options :D – DarioS Aug 23 '23 at 16:23