I am trying to style an SVG within a pseudo-element.
p::before {
content: url("data:image/svg+xml;charset=UTF-8,<svg xmlns='http://www.w3.org/2000/svg' class='svg-check' viewBox='0 0 20 20'><path fill-rule='evenodd' d='M10 18a8 8 0 100-16 8 8 0 000 16zm3.707-9.293a1 1 0 00-1.414-1.414L9 10.586 7.707 9.293a1 1 0 00-1.414 1.414l2 2a1 1 0 001.414 0l4-4z' clip-rule='evenodd' /></svg>");
display: inline-block;
height: 2rem;
width: 2rem;
}
.svg-check {
fill: green;
}
<div>
<p>First</p>
<p>Second</p>
</div>
My question is :
The following are three ways to color SVG in pseudo-elements:
As per this comment, the mask
property can be assigned the url()
function instead of content
or background
. The background-color
is then assigned the desired color.
Reference
If the url()
function is assigned to the content
property, filter
functions can determine the SVG color.
Reference
Tool
If background
(specifically background-image
) has the url()
function, certain steps can be done manually or by JavaScript to hardcode a [fill]
attribute into the url()
by encoding it into base64 ASCII.
Reference
Tool
Details are commented in the example. Also refer to this article about SVG data URI.
/* #1
|| Define SVG elements as a string. Include a [fill]
|| attribute and assign it a color. This example
|| has [fill='#D22578'] on <path>.
*/
const svgString = `<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 20 20'><path fill='#D22578' fill-rule='evenodd' d='M10 18a8 8 0 100-16 8 8 0 000 16zm3.707-9.293a1 1 0 00-1.414-1.414L9 10.586 7.707 9.293a1 1 0 00-1.414 1.414l2 2a1 1 0 001.414 0l4-4z' clip-rule='evenodd' /></svg>`;
/* #2
|| Next, encode the string into base64 ASCII.
*/
const bin2ASCII = btoa(svgString);
/* #3
|| Then interpolate the base64 encoding into a CSS url()
|| function as a string.
*/
const base64 = `url("data:image/svg+xml;base64,${bin2ASCII}")`;
/* #4
|| Finally, assign the value to a CSS property on
|| the :root (aka <html>). Make sure the CSS property is
|| assigned to the element/pseudo-element as a background.
|| ex. .icon::before { background: var(--base64) }
*/
document.documentElement.style.setProperty("--base64", base64);
/*
|| This is the value of variable base64:
url("")
|| Rather than use JavaScript, we can paste SVG into
|| an online tool like:
|| https://www.base64encode.org/
|| and paste the results to our CSS.
|| Set the first <select> form control to: "ASCII".
*/
:root {
--base64;
--utf8: url("data:image/svg+xml;charset=UTF-8,<svg xmlns='http://www.w3.org/2000/svg' class='svg-check' viewBox='0 0 20 20'><path fill-rule='evenodd' d='M10 18a8 8 0 100-16 8 8 0 000 16zm3.707-9.293a1 1 0 00-1.414-1.414L9 10.586 7.707 9.293a1 1 0 00-1.414 1.414l2 2a1 1 0 001.414 0l4-4z' clip-rule='evenodd' /></svg>");
font: small-caps 2ch/1.5 "Segoe UI";
}
code {
padding: 0.25rem 0.5rem;
font-family: "Source Code Pro";
font-variant: normal;
color: #FFB41F;
background: #000;
}
dd+dt,
dd {
margin-top: 0.5rem;
}
dd::before {
content: '';
display: inline-block;
height: 2rem;
width: 2rem;
margin: -0.25rem 0.25rem 0 0;
vertical-align: middle;
background-size: contain;
}
/*
|| Assign url() function to mask. Assign color to
|| background-color
*/
.mask::before {
mask: var(--utf8) no-repeat;
mask-size: contain;
background-color: #FE0000;
}
/*
|| Use the online tool to get the filter values:
|| https://isotropic.co/tool/hex-color-to-css-filter/
|| Assign the url() function to content
*/
.filter::before {
content: var(--utf8);
filter: invert(7%) sepia(96%) saturate(7103%) hue-rotate(248deg) brightness(111%) contrast(144%);
}
/*
|| Assign url() function to background.
*/
.base64::before {
background: var(--base64) no-repeat;
}
<dl>
<dt><code>mask</code> Shorthand</dt>
<dd class="mask">Alpha</dd>
<dt><code>filter</code> Functions</dt>
<dd class="filter">Beta</dd>
<dt>Base64 Encoding</dt>
<dd class="base64">Gamma</dd>
</dl>