A CSS-Only Star Rating Component And More! (Part 1)

A CSS-Only Star Rating Component And More! (Part 1)

Making a star score element is a basic train in net improvement. It has been done and re-done many instances utilizing completely different strategies. We normally want a small quantity of JavaScript to drag it collectively, however what a few CSS-only implementation? Sure, it’s attainable!

Here’s a demo of a CSS-only star rating component. You’ll be able to click on to replace the score.

Cool, proper? Along with being CSS-only, the HTML code is nothing however a single aspect:

<enter kind="vary" min="1" max="5">

An enter vary aspect is the right candidate right here because it permits a consumer to pick out a numeric worth between two boundaries (the min and max). Our aim is to type that native aspect and rework it right into a star score element with out further markup or any script! We may also create extra parts on the finish, so observe alongside.

Notice: This text will solely concentrate on the CSS half. Whereas I strive my greatest to contemplate UI, UX, and accessibility elements, my element is just not good. It could have some drawbacks (bugs, accessibility points, and many others), so please use it with warning.

The <enter> aspect

You most likely realize it however styling native components comparable to inputs is a bit difficult as a result of all of the default browser types and in addition the completely different inside buildings. If, for instance, you examine the code of an enter vary you will notice a distinct HTML between Chrome (or Safari, or Edge) and Firefox.

Fortunately, we have now some widespread components that I’ll depend on. I’ll goal two completely different components: the predominant aspect (the enter itself) and the thumb aspect (the one you slide along with your mouse to replace the worth).

Our CSS will primarily appear like this:

enter[type="range"] {
  /* styling the primary aspect */
}

enter[type="range" i]::-webkit-slider-thumb {
  /* styling the thumb for Chrome, Safari and Edge */
}

enter[type="range"]::-moz-range-thumb {
  /* styling the thumb for Firefox */
}

The one downside is that we have to repeat the types of the thumb aspect twice. Don’t attempt to do the next:

enter[type="range" i]::-webkit-slider-thumb,
enter[type="range"]::-moz-range-thumb {
  /* styling the thumb */
}

This doesn’t work as a result of the entire selector is invalid. Chrome & Co. don’t perceive the ::-moz-* half and Firefox doesn’t perceive the ::-webkit-* half. For the sake of simplicity, I’ll use the next selector for this text:

enter[type="range"]::thumb {
  /* styling the thumb */
}

However the demo incorporates the true selectors with the duplicated types. Sufficient introduction, let’s begin coding!

Styling the primary aspect (the star form)

We begin by defining the scale:

enter[type="range"] {
  --s: 100px; /* management the scale*/
  
  peak: var(--s);
  aspect-ratio: 5;
  
  look: none; /* take away the default browser types */
}

If we think about that every star is positioned inside a sq. space, then for a 5-star score we’d like a width equal to 5 instances the peak, therefore the usage of aspect-ratio: 5.

That 5 worth can be the worth outlined because the max attribute for the enter aspect.

<enter kind="vary" min="1" max="5">

So, we are able to depend on the newly enhanced attr() function (Chrome-only in the intervening time) to learn that worth as an alternative of manually defining it!

enter[type="range"] {
  --s: 100px; /* management the scale*/
  
  peak: var(--s);
  aspect-ratio: attr(max kind(<quantity>));
  
  look: none; /* take away the default browser types */
}

Now you may management the variety of stars by merely adjusting the max attribute. That is nice as a result of the max attribute can be utilized by the browser internally, so updating that worth will management our implementation in addition to the browser’s conduct.

This enhanced model of attr() is only available in Chrome for now so all my demos will comprise a fallback to assist with unsupported browsers.

The subsequent step is to make use of a CSS masks to create the celebs. We’d like the form to repeat 5 instances (or extra relying on the max worth) so the masks dimension needs to be equal to var(--s) var(--s) or var(--s) 100% or just var(--s) since by default the peak shall be equal to 100%.

enter[type="range"] {  
  --s: 100px; /* management the scale*/
  
  peak: var(--s);
  aspect-ratio: attr(max kind(<quantity>));
  
  look: none; /* take away the default browser types */

  mask-image: /* ... */;
  mask-size: var(--s);
}

What concerning the mask-image property you may ask? I feel it’s no shock that I inform you it’ll require a number of gradients, however it may be SVG as an alternative. This text is about making a star-rating element however I want to hold the star half type of generic so you may simply substitute it with any form you need. That’s why I say “and extra” within the title of this put up. We’ll see later how utilizing the identical code construction we are able to get quite a lot of completely different variations.

Here’s a demo displaying two completely different implementations for the star. One is utilizing gradients and the opposite is utilizing an SVG.

On this case, the SVG implementation seems to be cleaner and the code can be shorter however hold each approaches in your again pocket as a result of a gradient implementation can do a greater job in some conditions.

Styling the thumb (the chosen worth)

Let’s now concentrate on the thumb aspect. Take the final demo then click on the celebs and see the place of the thumb.

The nice factor is that the thumb is at all times inside the space of a given star for all of the values (from min to max), however the place is completely different for every star. It will be good if the place is at all times the identical, whatever the worth. Ideally, the thumb ought to at all times be on the middle of the celebs for consistency.

Here’s a determine for example the place and the way to replace it.

A CSS-Only Star Rating Component And More! (Part 1)

The strains are the place of the thumb for every worth. On the left, we have now the default positions the place the thumb goes from the left edge to the appropriate fringe of the primary aspect. On the appropriate, if we prohibit the place of the thumb to a smaller space by including some areas on the perimeters, we get a lot better alignment. That house is the same as half the scale of 1 star, or var(--s)/2. We will use padding for this:

enter[type="range"] {  
  --s: 100px; /* management the scale */
  
  peak: var(--s);
  aspect-ratio: attr(max kind(<quantity>));
  padding-inline: calc(var(--s) / 2);
  box-sizing: border-box;
  
  look: none; /* take away the default browser types */

  mask-image: ...;
  mask-size: var(--s);
}

It’s higher however not good as a result of I’m not accounting for the thumb dimension, which suggests we don’t have true centering. It’s not a difficulty as a result of I’ll make the scale of the thumb very small with a width equal to 1px.

enter[type="range"]::thumb {
  width: 1px;
  peak: var(--s);  

  look: none; /* take away the default browser types */ 
}

The thumb is now a skinny line positioned on the middle of the celebs. I’m utilizing a pink colour to focus on the place however in actuality, I don’t want any colour as a result of it will likely be clear.

You might assume we’re nonetheless removed from the ultimate consequence however we’re nearly carried out! One property is lacking to finish the puzzle: border-image.

The border-image property permits us to attract decorations exterior a component due to its outset function. For that reason, I made the thumb small and clear. The coloration shall be carried out utilizing border-image. I’ll use a gradient with two stable colours because the supply:

linear-gradient(90deg, gold 50%, gray 0);

And we write the next:

border-image: linear-gradient(90deg, gold 50%, gray 0) fill 0 // 0 100px;

The above implies that we prolong the realm of the border-image from all sides of the aspect by 100px and the gradient will fill that space. In different phrases, every colour of the gradient will cowl half of that space, which is 100px.

Do you see the logic? We created a type of overflowing coloration on all sides of the thumb — a coloration that can logically observe the thumb so every time you click on a star it slides into place!

Now as an alternative of 100px let’s use a really huge worth:

We’re getting shut! The coloration is filling all the celebs however we don’t need it to be within the center however relatively throughout your entire chosen star. For this, we replace the gradient a bit and as an alternative of utilizing 50%, we use 50% + var(--s)/2. We add an offset equal to half the width of a star which suggests the primary colour will take extra space and our star score element is ideal!

We will nonetheless optimize the code a little bit the place as an alternative of defining a peak for the thumb, we hold it 0 and we think about the vertical outset of border-image to unfold the coloration.

enter[type="range"]::thumb{
  width: 1px;
  border-image: 
    linear-gradient(90deg, gold calc(50% + var(--s) / 2), gray 0)
    fill 0 // var(--s) 500px;
  look: none;
}

We will additionally write the gradient in another way utilizing a conic gradient as an alternative:

enter[type="range"]::thumb{
  width: 1px;
  border-image: 
    conic-gradient(at calc(50% + var(--s) / 2), gray 50%, gold 0)
    fill 0 // var(--s) 500px;
  look: none;
}

I do know that the syntax of border-image is just not straightforward to understand and I went a bit quick with the reason. But I have a very detailed article over at Smashing Magazine the place I dissect that property with a variety of examples that I invite you to learn for a deeper dive into how the property works.

The complete code of our element is that this:

<enter kind="vary" min="1" max="5">
enter[type="range"] {  
  --s: 100px; /* management the scale*/
  
  peak: var(--s);
  aspect-ratio: attr(max kind(<quantity>));
  padding-inline: calc(var(--s) / 2); 
  box-sizing: border-box; 
  look: none; 
  mask-image: /* ... */; /* both an SVG or gradients */
  mask-size: var(--s);
}

enter[type="range"]::thumb {
  width: 1px;
  border-image: 
    conic-gradient(at calc(50% + var(--s) / 2), gray 50%, gold 0)
    fill 0//var(--s) 500px;
  look: none;
}

That’s all! A number of strains of CSS code and we have now a pleasant score star element!

Half-Star Ranking

What about having a granularity of half a star as a score? It’s one thing widespread and we are able to do it with the earlier code by making a number of changes.

First, we replace the enter aspect to increment in half steps as an alternative of full steps:

<enter kind="vary" min=".5" step=".5" max="5">

By default, the step is the same as 1 however we are able to replace it to .5 (or any worth) then we replace the min worth to .5 as nicely. On the CSS aspect, we alter the padding from var(--s)/2 to var(--s)/4, and we do the identical for the offset contained in the gradient.

enter[type="range"] {  
  --s: 100px; /* management the scale*/
  
  peak: var(--s);
  aspect-ratio: attr(max kind(<quantity>));
  padding-inline: calc(var(--s) / 4); 
  box-sizing: border-box; 
  look: none; 
  mask-image: ...; /* both SVG or gradients */
  mask-size: var(--s);
}

enter[type="range"]::thumb{
  width: 1px;
  border-image: 
    conic-gradient(at calc(50% + var(--s) / 4),gray 50%, gold 0)
    fill 0 // var(--s) 500px;
  look: none;
}

The distinction between the 2 implementations is an element of one-half which can be the step worth. Meaning we are able to use attr() and create a generic code that works for each instances.

enter[type="range"] {  
  --s: 100px; /* management the scale*/
  
  --_s: calc(attr(step kind(<quantity>),1) * var(--s) / 2);
  peak: var(--s);
  aspect-ratio: attr(max kind(<quantity>));
  padding-inline: var(--_s);
  box-sizing: border-box; 
  look: none; 
  mask-image: ...; /* both an SVG or gradients */
  mask-size: var(--s);
}

enter[type="range"]::thumb{
  width: 1px;
  border-image: 
    conic-gradient(at calc(50% + var(--_s)),gold 50%,gray 0)
    fill 0//var(--s) 500px;
  look: none;
}

Here’s a demo the place modifying the step is all that you should do to regulate the granularity. Don’t neglect that you may additionally management the variety of stars utilizing the max attribute.

Utilizing the keyboard to regulate the score

As it’s possible you’ll know, we are able to alter the worth of an enter vary slider utilizing a keyboard, so we are able to management the score utilizing the keyboard as nicely. That’s factor however there’s a caveat. Attributable to the usage of the masks property, we now not have the default define that signifies keyboard focus which is an accessibility concern for individuals who depend on keyboard enter.

For a greater consumer expertise and to make the element extra accessible, it’s good to show a top level view on focus. The best resolution is so as to add an additional wrapper:

<span>
  <enter kind="vary" min="1" max="5">
</span>

That can have a top level view when the enter inside has focus:

span:has(:focus-visible) {
  define: 2px stable;
}

Attempt to use your keyboard within the beneath instance to regulate each scores:

One other concept is to contemplate a extra complicated masks configuration that stops hiding the define (or any exterior ornament). The trick is to begin with the next:

masks:
  conic-gradient(#000 0 0) exclude,
  conic-gradient(#000 0 0) no-clip;

The no-clip key phrase implies that nothing from the aspect shall be clipped (together with outlines). Then we use an exclude composition with one other gradient. The exclusion will cover every part contained in the aspect whereas maintaining what’s exterior seen.

Lastly, we add again the masks that creates the star shapes:

masks: 
  /* ... */ 0/var(--s), 
  conic-gradient(#000 0 0) exclude,
  conic-gradient(#000 0 0) no-clip;

I desire utilizing this final technique as a result of it maintains the single-element implementation however perhaps your HTML construction means that you can add concentrate on an higher aspect and you’ll hold the masks configuration easy. It completely relies upon!

Credit to Ana Tudor for the final trick!

Extra examples!

As I mentioned earlier, what we’re making is greater than a star score element. You’ll be able to simply replace the masks worth to make use of any form you need.

Right here is an instance the place I’m utilizing an SVG of a coronary heart as an alternative of a star.

Why not butterflies?

This time I’m utilizing a PNG picture as a masks. In case you are not comfy utilizing SVG or gradients you should use a clear picture as an alternative. So long as you have got an SVG, a PNG, or gradients, there isn’t a restrict on what you are able to do with this so far as shapes go.

We will go even additional into the customization and create a volume control component like beneath:

I’m not repeating a selected form in that final instance, however am utilizing a fancy masks configuration to create a signal shape.

Conclusion

We began with a star score element and ended with a bunch of cool examples. The title might have been “How one can type an enter vary aspect” as a result of that is what we did. We upgraded a local element with none script or further markup, and with just a few strains of CSS.

What about you? Can you consider one other fancy element utilizing the identical code construction? Share your instance within the remark part!

Article sequence

  1. A CSS-Only Star Rating Component and More! (Part 1)
  2. A CSS-Solely Star Ranking Element and Extra! (Half 2) — Coming March 7!

Leave a Reply