How to use an SVG file for the shape of my text buttons?

124 Views Asked by At

I'm setting up a design system in NextJS14 (React). We are using CSS modules and have all colors stored in CSS variables (custom properties).

We have these nice-looking buttons in our Figma design. I want to use the exported SVG for the shape of our text buttons. preview of custom button shape svg

I made some progress by setting the background-image: to the url of the SVG. However, I have not found a way to dynamically change the fill and stroke colors of the SVG based on the props passed into the custom Button component.

2

There are 2 best solutions below

1
On BEST ANSWER

Here are some options. In your case I guess that the first two examples could be used. The last one is an alternative approach that only works in some cases.

Embed the SVG

Embed the SVG into the HTML and style the SVG elements using CSS.

button.orange rect {
  fill: orange;
}

button.lime rect {
  fill: lime;
}

button {
  background-color: transparent;
  border: none;
  display: grid;
}

button svg {
  grid-column: 1;
  grid-row: 1;
  align-self: center;
}

button span {
  grid-column: 1;
  grid-row: 1; 
  align-self: center;
  padding: 0 1em;
}
<button class="orange">
  <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 60 20">
    <rect width="60" height="20" rx="5" fill="white" />  
  </svg>
  <span>Button</span>
</button>

<button class="lime">
  <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 60 20">
    <rect width="60" height="20" rx="5" fill="white" />  
  </svg>
  <span>Button</span>
</button>

CSS background-image

Just repeat creating different CSS selectors with different background images.

button.orange {
  fill: orange;
  background-image: url('data:image/svg+xml,<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 60 20"><rect width="60" height="20" rx="5" fill="orange" /></svg>');
  background-repeat: no-repeat;
  background-size: contain;
  background-position: center;
}

button.lime {
  fill: orange;
  background-image: url('data:image/svg+xml,<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 60 20"><rect width="60" height="20" rx="5" fill="lime" /></svg>');
  background-repeat: no-repeat;
  background-size: contain;
  background-position: center;
}

button {
  background-color: transparent;
  border: none;
  display: grid;
}

button svg {
  grid-column: 1;
  grid-row: 1;
  align-self: center;
}

button span {
  grid-column: 1;
  grid-row: 1; 
  align-self: center;
  padding: .2em 1em;
}
<button class="orange">
  <span>Button</span>
</button>

<button class="lime">
  <span>Button</span>
</button>

CSS backgrounds with gradients only

Combine a background image (the SVG document) with a solid background color defined in CSS. The SVG image is just a gradient going from transparent to white.

button.gradient01 {
background-image: url('');
  background-repeat: no-repeat;
  background-size: contain;
  background-position: center;
}

button.orange {
  background-color: orange;
}

button.lime {
  background-color: lime;
}

button {
  background-color: transparent;
  border: none;
  display: grid;
  border-radius: 5px;
}

button svg {
  grid-column: 1;
  grid-row: 1;
  align-self: center;
}

button span {
  grid-column: 1;
  grid-row: 1; 
  align-self: center;
  padding: .2em 1em;
}
<button class="orange gradient01">
  <span>Button</span>
</button>

<button class="lime gradient01">
  <span>Button</span>
</button>

<p>The SVG used as background:</p> 
<svg xmlns="http://www.w3.org/2000/svg">
  <defs>
    <linearGradient id="g01" gradientTransform="rotate(90)">
      <stop offset="0%" stop-opacity="0" stop-color="white" />
      <stop offset="100%" stop-opacity="1" stop-color="white" />
    </linearGradient>
  </defs>
  <rect width="100%" height="100%" fill="url(#g01)" />
</svg>

1
On

End result here

I'm thrilled to share that I finally got it working to:

  • Use an SVG for the shape of my button
  • Change the scaling, width, colors and optional icons with props
  • Have the colors linked to my custom CSS variables for maintainability

I could not have gotten to this point without your suggestions @chrwahl. Thanks a million!