Skip to content

Instantly share code, notes, and snippets.

@keithchu
Created January 9, 2012 18:04
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 1 You must be signed in to fork a gist
  • Save keithchu/1584147 to your computer and use it in GitHub Desktop.
Save keithchu/1584147 to your computer and use it in GitHub Desktop.
Star Rating Using HTML <input type="range">
/**
* Star Rating Using HTML <input type="range">
*/
html {
min-height: 100%;
-webkit-tap-highlight-color: rgba(0,0,0,0);
tap-highlight-color: rgba(0,0,0,0);
}
body {
background-color: rgb(48,83,109);
background-image: -webkit-gradient(linear, from(rgb(48,83,109)), to(rgb(93,156,198))); /* Safari 4+, Chrome */
background-image: -webkit-linear-gradient(rgb(48,83,109), rgb(93,156,198)); /* Safari 5.1+, Chrome 10+ */
background-image: -moz-linear-gradient(rgb(48,83,109), rgb(93,156,198)); /* FF3.6+ */
background-image: -ms-linear-gradient(rgb(48,83,109), rgb(93,156,198)); /* IE10 */
background-image: -o-linear-gradient(rgb(48,83,109), rgb(93,156,198)); /* Opera 11.10+ */
background-image: linear-gradient(rgb(48,83,109), rgb(93,156,198));
background-repeat: no-repeat;
font-family: sans-serif;
height: 100%;
padding: 60px 0;
text-align: center;
}
a, a:link, a:visited {
color: rgba(80,0,0,.8);
text-decoration: none;
}
address {
color: #999;
font-size: .8em;
font-style: normal;
}
footer {
background-color: #f9f9f9;
border-top: 1px dashed rgba(200,200,200,.5);
color: rgba(150,150,150,.8);
font-size: .7em;
margin-top: 40px;
padding: 12px 15px;
}
footer ul {
margin: 0;
padding: 0;
}
footer li {
list-style: none;
}
header {
border-top: 5px solid #999;
padding: 4px 15px 35px;
text-align: left;
}
h1 {
color: #333;
font-size: 1em;
margin-bottom: 1px;
}
section {
padding: 0 30px;
position: relative;
text-align: center;
}
section ol {
display: block !important; /* Allows for CSS off scenario, JS turns it off, CSS turns it on */
font-size: 9px;
list-style: none;
margin: 0; padding: 0;
position: absolute;
top: 0; left: 132px;
width: 80px;
z-index: 1;
white-space: no-wrap;
}
section ol li {
display: inline-block;
line-height: 18px;
text-align: center;
width: 16px;
}
sup {
font-size: 75%;
line-height: 0;
position: relative; top: -.5em;
vertical-align: baseline;
}
input[name="rating"] {
background-color: transparent;
background-image: url(%3D%3D),
url(%3D);
/* background-image: url("css/images/ratings-stars-r.png"),
url("css/images/ratings.png");
*/
background-position: 0 15px, 0 0;
background-repeat: no-repeat, repeat-x;
position: relative; top: -1px;
vertical-align: top;
width: 80px;
z-index: 2;
-webkit-appearance: none;
}
input[name="rating"]::-webkit-slider-thumb {
background-color: rgba(255,203,4,.33); /* Highlight the selected item for the "Images Off" scenario */
background-image: url(%3D);
background-position: 0 -22px;
background-repeat: repeat-x;
border: none;
height: 16px;
width: 16px;
-webkit-border-radius: 8px;
-moz-border-radius: 8px;
border-radius: 8px;
/*-webkit-mask-image: url("css/images/ratings_mask.png"); /* Masks out the highlight in the "Images On" scenario */
-webkit-mask-image: url();
-webkit-appearance: none;
}
input[name="rating"].default::-webkit-slider-thumb {
background: none;
}
output {
border: 1px solid #999;
display: inline-block;
font-size: .8em;
height: 18px;
line-height: 1.4;
margin-left: 3px;
text-align: center;
vertical-align: bottom;
width: 18px;
}
#wrapper {
background-color: #fff;
border: 1px solid #7d7d7d;
box-shadow: 0 0 15px rgba(0,0,0,.3);
display: inline-block;
text-align: left;
}
<div id="wrapper">
<header>
<h1>Star Rating Using &lt;input&nbsp;type=range&gt;</h1>
<address class="author">by <a href="http://twitter.com/catharsis" rel="author">Keith Chu</a></address>
</header>
<section>
<label for="rating">Rating: </label>
<select data-type="range" name="rating">
<option disabled selected>Select one</option>
<option>1 star</option>
<option>2 stars</option>
<option>3 stars</option>
<option>4 stars</option>
<option>5 stars</option>
</select>
<!--<input type="range" name="rating" step="1" min="1" value="1" max="5" />-->
<!--<output for="rating">0</output>--> <!-- append to input via JS -->
</section>
<footer>
<ul>
<li><sup>1</sup><a href="https://github.com/Modernizr/Modernizr/blob/master/modernizr.js#L797">Modernizr</a> &mdash; range input test</li>
<li><sup>2</sup><a href="http://remysharp.com/2011/07/18/input-range-polyfill/">Remy Sharp</a> &mdash; range input polyfill</li>
</ul>
</footer>
</div>
<!-- Javascript at the bottom for fast page loading -->
<!-- Without adding any elements, and given JS on:
1. Images off (fallback) (added OL - sk)
2. CSS off (fallback) (Hid OL - sk)
3. Images/CSS off (fallback) (If CSS is off, images are off since all images are linked through CSS - sk)
-->
<script>
/* Test if input[type="range"] supported (via Modernizr) [footnote 1] */
var bool,
output = document.createElement("output"),
range = document.createElement("input"),
smile = ":)";
range.setAttribute("type", "range");
if (bool = range.type !== "text") {
range.value = smile;
range.style.cssText = "position:absolute;visibility:hidden;";
if (range.style.WebkitAppearance !== undefined ) {
document.body.appendChild(range);
bool = range.offsetHeight !== 0; // Mobile android web browser has false positive, so must check the height to see if the widget is actually there.
document.body.removeChild(range);
}
} else {
bool = range.value == smile;
}
if (bool) { initRange(); } // input[type="range"] is supported
function initRange() {
var listItemContent, listItem,
selects = document.getElementsByTagName("select"),
list = document.createElement("ol"),
i, j;
output.setAttribute("for","rating");
output.value = 0;
range.value = range.style.cssText = ""; // reset range from support testing
/* input["range"] polyfill (via @remysharp) [footnote 2] */
for (i = selects.length; i--;) {
if (selects[i].getAttribute("data-type") == "range") { (function(select) {
/* Create an ordered list based on the select element for use in the "images off" scenario*/
for (j = 0; j < select.length; j++) {
listItemContent = Number(select.options[j].value.substr(0,1));
if (listItemContent) {
listItem = document.createElement("li");
listItem.innerHTML = listItemContent.toString();
list.appendChild(listItem);
}
}
select.parentNode.appendChild(list);
/* Turn off display and show with CSS to account for CSS off scenario */
list.style.display = "none";
/* Replace Select element with Input Type = Range */
select = select.parentNode.replaceChild(range, select);
range.name = "rating";
range.value = range.step = range.min = 1;
range.max = 5;
})(selects[i]);}
}
/* end polyfill */
range.className = "default";
range.parentNode.appendChild(output);
bindEvent(range, "touchstart", function(e) { ratingSwap(this,e); });
bindEvent(range, "mousedown", function() { ratingSwap(this); });
bindEvent(range, "change", function() { ratingSwap(this); });
/* bindEvent(range, "click", function(e) { ratingSwap(this,e); }); */
}
/**
* @param el The element to be bound.
* @param eventName The event to attach to the element.
* @param eventHandler The function to trigger on event activation.
**/
function bindEvent(el, eventName, eventHandler) {
if (el.addEventListener) { el.addEventListener(eventName, eventHandler, false); }
else if (el.attachEvent) { el.attachEvent("on" + eventName, eventHandler); }
else { alert("Event binding failed due to lack of support for addEventListener/attachEvent"); }
}
/**
* @param el The input[type="range"] element.
* @param e The interaction event.
**/
function ratingSwap(el, e) {
var areaClicked = 0,
currentStep = 0,
areaStep = 16, /* step width of range (i.e. star width) */
areaSteps = 5, /* number of steps in range (i.e. # of stars) */
i;
if (el.className == "default") {
el.className = "";
};
/* Triggers on click (1) on smartphone tap, (2) on click of current star on desktop */
if (e) {
/**
* Get click position respective to the input
* position = (click) - (relatively positioned parent location) - (input location)
**/
areaClicked = e.pageX - e.target.parentNode.offsetLeft - e.target.offsetLeft;
/* Map click positions to input values; assign values */
for (i = areaSteps; i--;) {
if (Math.abs(areaClicked) <= (i * areaStep)) {
el.value = i;
break;
}
}
}
/* Get the correct background-position for the star sprite */
currentStep = ((el.value - 2) * -(areaStep) - 1) + "px";
/* Set the correct background-position for the star sprite */
el.style.backgroundPosition = "0 " + currentStep + ", 0 0";
/* Update the output value */
output.innerHTML = el.value;
}
</script>
{"view":"split-vertical","fontsize":"100","seethrough":"","prefixfree":"1","page":"html"}
@keithchu
Copy link
Author

keithchu commented Sep 7, 2012

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment