6

I want to make rate stars progress bar with single div element and css pseudo elements like ::before.

Only thing what I can't do is manipulating width of ::before element with Javascript.

Here is my example:

.review-rate {
 background-image: url();
 background-repeat: repeat-x;
 width: 100px;
 height: 18px;
}
.review-rate::before {
 content: "";
 display: block;
 width: attr(data-width);
 height: 18px;
 width: 55%;
 background-image: url();
 background-repeat: repeat-x;
 background-position: 0 -18px;
}
<div class="review-rate" data-toggle="review" data-value="3.5" data-width="50" data-count="17"></div>

Is there any way to manipulate width of ::before element?

Maybe something like width: attr(data-width); ?

Any help will be appreciated.

Ivan Gerasimenko
  • 2,381
  • 3
  • 30
  • 46
zur4ik
  • 6,072
  • 9
  • 52
  • 78
  • 1
    I haven't tried the stuff with attr() for width, but you might need to add your units, e.g. `width: attr(data-width)px;` – Jon Snow Mar 05 '15 at 07:28
  • I tried this, but it's not working. Thanks anyway. – zur4ik Mar 05 '15 at 07:37
  • this thread has all the explanations [link](http://stackoverflow.com/questions/10061414/changing-width-property-of-a-before-css-selector-using-jquery) – Max Novich Mar 05 '15 at 07:40

2 Answers2

7

One way I know of changing :before & :after css dynamically is the one @Brett DeWoody mentioned, Adding style tags.

In your case you can simply switch the role of :before & div. Then use div width instead of :before. Like this:

.review-rate {
background-image: url("");
background-position: 0 -18px;
background-repeat: repeat-x;
height: 18px;
overflow: visible;
position: relative;
width: 50px;
}
.review-rate::before {
background-image: url("");
background-repeat: repeat-x;
content: "";
display: block;
height: 18px;
position: relative;
width: 100px;
z-index: -1;
}
 <div class="review-rate" data-toggle="review" data-value="3.5" data-width="50" data-count="17"></div>
Imran Bughio
  • 4,811
  • 2
  • 30
  • 53
5

Though it's not ideal, you can append a new <style> element to the page using jQuery with the new width value.

$('head').append('<style>.review-rate:before{width:75%}</style>');

Here's a demo.

Update
As Imran pointed out in the comment below, the above approach would result in many <style> tags over time. To prevent this a combo of append and replaceWith could be used to keep the number of <style> tags to a minimum.

if ($('head style[data-class="review-rate"]').length) {
    $('head style[data-class="review-rate"]').replaceWith('<style data-class="review-rate">.review-rate:before{width:75%}</style>')       
} else {
    $('head').append('<style data-class="review-rate">.review-rate:before{width:35%}</style>');
}
Brett DeWoody
  • 59,771
  • 29
  • 135
  • 184
  • 1
    True @ImranBughio. Some fancier jQuery could solve this however. For example, instead of using `append` you could use `append` with `replaceWith` to prevent too many style tags. The [demo](http://jsfiddle.net/brettdewoody/hktt43Ly/) has been updated with an example. – Brett DeWoody Mar 05 '15 at 07:46
  • 1
    Just want to note that you don't need jQuery, the DOM API has what you need: `document.head.querySelector`, `parentElement.appendChild` and `parentElement.replaceChild` – Max Heiber May 31 '17 at 18:03