This code snippet helps you to create a knob slider. It defines a function called main
that is executed when the page is loaded. The function sets the audio volume to zero, attaches event listeners to the volume knob for mouse button clicks and releases, and creates 27 ticks.
When the mouse button is pressed, a function called onMouseDown
is executed. If the audio is not already playing, it starts to play, and event listener is attached for mouse movement.
When the mouse button is released, a function called onMouseUp
is executed, which removes the event listener for mouse movement.
When the mouse is moved while the button is held down, a function called onMouseMove
is executed. This function computes the angle of the mouse relative to the center of the volume knob, sets the volume knob’s rotation to the angle, and updates the audio volume and tick highlights accordingly.
There are also utility functions for detecting mobile devices and for getting the appropriate event names based on whether the device is mobile or desktop.
How to Create JavaScript Knob Slider
Create the HTML structure for the knob slider as follows:
<div class="body"> <h1>Click anywhere to begin playing audio first, then drag volume knob with mouse or finger to control volume</h1> <p>Current volume: <span id="volumeValue" class="current-value">0%</span></p> <div class="knob-surround"> <div id="knob" class="knob"></div> <span class="min">Min</span> <span class="max">Max</span> <div id="tickContainer" class="ticks"></div> </div> <p>Javascript written by <a href="https://www.quora.com/profile/Kevin-Lam-15">Kevin Lam</a> of <a href="https://www.ztransitions.com">zTransitions.com</a><br><br> Original HTML/CSS code forked from <a href="https://twitter.com/blucube">Ed Hicks's</a> - <a href="https://codepen.io/blucube/pen/cudAz">original HTML/CSS example</a><br><br><a href="http://dribbble.com/shots/753124-Volume-Knob">Original volume knob graphic design </a><a href="https://twitter.com/rickss">by Ricardo Salazar</a></p> </div>
Now, style the knob slider using the following CSS styles:
@font-face { font-family: 'Open Sans'; font-style: normal; font-weight: 300; src: local('Open Sans Light'), local('OpenSans-Light'), url(https://fonts.gstatic.com/s/opensans/v18/mem5YaGs126MiZpBA-UN_r8OUuhs.ttf) format('truetype'); } @font-face { font-family: 'Varela Round'; font-style: normal; font-weight: 400; src: local('Varela Round Regular'), local('VarelaRound-Regular'), url(https://fonts.gstatic.com/s/varelaround/v13/w8gdH283Tvk__Lua32TysjIfp8uK.ttf) format('truetype'); } body { background-color: #181818; font-size: 100%; font-family: "Open Sans", sans-serif; color: #aaa; text-align: center; user-select: none; } .cd__main{ display: block !important; } .body{ width: 100%; height: 100%; background-color: black; } .knob-surround { position: relative; background-color: grey; width: 14em; height: 14em; border-radius: 50%; border: solid 0.25em #0e0e0e; margin: 5em auto; background: #181818; background: -webkit-gradient(linear, left bottom, left top, color-stop(0, #1d1d1d), color-stop(1, #131313)); background: -ms-linear-gradient(bottom, #1d1d1d, #131313); background: -moz-linear-gradient(center bottom, #1d1d1d 0%, #131313 100%); background: -o-linear-gradient(#131313, #1d1d1d); filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#131313', endColorstr='#1d1d1d', GradientType=0); -webkit-box-shadow: 0 0.2em 0.1em 0.05em rgba(255, 255, 255, 0.1) inset, 0 -0.2em 0.1em 0.05em rgba(0, 0, 0, 0.5) inset, 0 0.5em 0.65em 0 rgba(0, 0, 0, 0.3); -moz-box-shadow: 0 0.2em 0.1em 0.05em rgba(255, 255, 255, 0.1) inset, 0 -0.2em 0.1em 0.05em rgba(0, 0, 0, 0.5) inset, 0 0.5em 0.65em 0 rgba(0, 0, 0, 0.3); box-shadow: 0 0.2em 0.1em 0.05em rgba(255, 255, 255, 0.1) inset, 0 -0.2em 0.1em 0.05em rgba(0, 0, 0, 0.5) inset, 0 0.5em 0.65em 0 rgba(0, 0, 0, 0.3); } .knob { position: absolute; width: 100%; height: 100%; border-radius: 50%; -webkit-transform: rotate(0deg); -moz-transform: rotate(0deg); -o-transform: rotate(0deg); -ms-transform: rotate(0deg); transform: rotate(0deg); z-index: 10; } .knob:before { content: ""; position: absolute; bottom: 19%; left: 19%; width: 3%; height: 3%; background-color: #a8d8f8; border-radius: 50%; -webkit-box-shadow: 0 0 0.4em 0 #79c3f4; -moz-box-shadow: 0 0 0.4em 0 #79c3f4; box-shadow: 0 0 0.4em 0 #79c3f4; } .min, .max { display: block; font-family: "Varela Round", sans-serif; color: rgba(255, 255, 255, 0.4); text-transform: uppercase; -webkit-font-smoothing: antialiased; font-size: 70%; position: absolute; opacity: 0.5; } .min { bottom: 1em; left: -2.5em; } .max { bottom: 1em; right: -2.5em; } .tick { position: absolute; width: 100%; height: 100%; top: 0; left: 0; z-index: 5; overflow: visible; } .tick:after { content: ""; width: 0.08em; height: 0.6em; background-color: rgba(255, 255, 255, 0.2); position: absolute; top: -1.5em; left: 50%; -webkit-transition: all 180ms ease-out; -moz-transition: all 180ms ease-out; -o-transition: all 180ms ease-out; transition: all 180ms ease-out; } .activetick:after { background-color: #a8d8f8; -webkit-box-shadow: 0 0 0.3em 0.08em #79c3f4; -moz-box-shadow: 0 0 0.3em 0.08em #79c3f4; box-shadow: 0 0 0.3em 0.08em #79c3f4; -webkit-transition: all 50ms ease-in; -moz-transition: all 50ms ease-in; -o-transition: all 50ms ease-in; transition: all 50ms ease-in; } h1 { font-weight: normal; margin: 2em 0; } p { line-height: 150%; max-width: 36em; margin: 1em auto; } a { color: #aaa; text-decoration: none; border-bottom: 1px solid #444; -webkit-transition: color 0.2s ease-in; -moz-transition: color 0.2s ease-in; -o-transition: color 0.2s ease-in; transition: color 0.2s ease-in; } a:hover, a:focus { color: #eee; } body, .knob { background-image: url(); }
Finally, add the following JavaScript function for the functionality:
var knobPositionX; var knobPositionY; var mouseX; var mouseY; var knobCenterX; var knobCenterY; var adjacentSide; var oppositeSide; var currentRadiansAngle; var getRadiansInDegrees; var finalAngleInDegrees; var volumeSetting; var tickHighlightPosition; var audio = new Audio("https://www.cineblueone.com/maskWall/audio/skylar.mp3"); //Celine Dion's "Ashes" var startingTickAngle = -135; var tickContainer = document.getElementById("tickContainer"); var volumeKnob = document.getElementById("knob"); var boundingRectangle = volumeKnob.getBoundingClientRect(); //get rectangular geometric data of knob (x, y, width, height) function main() { audio.volume = 0; //start at zero volume volumeKnob.addEventListener(getMouseDown(), onMouseDown); //listen for mouse button click document.addEventListener(getMouseUp(), onMouseUp); //listen for mouse button release createTicks(27, 0); } //on mouse button down function onMouseDown() { //start audio if not already playing if(audio.paused == true) { //mobile users must tap anywhere to start audio //https://developers.google.com/web/updates/2017/09/autoplay-policy-changes var promise = audio.play(); if (promise !== undefined) { promise.then(_ => { audio.play(); }).catch(error => { }); } } document.addEventListener(getMouseMove(), onMouseMove); //start drag } //on mouse button release function onMouseUp() { document.removeEventListener(getMouseMove(), onMouseMove); //stop drag } //compute mouse angle relative to center of volume knob //For clarification, see my basic trig explanation at: //https://www.quora.com/What-is-the-significance-of-the-number-pi-to-the-universe/answer/Kevin-Lam-15 function onMouseMove(event) { knobPositionX = boundingRectangle.left; //get knob's global x position knobPositionY = boundingRectangle.top; //get knob's global y position if(detectMobile() == "desktop") { mouseX = event.pageX; //get mouse's x global position mouseY = event.pageY; //get mouse's y global position } else { mouseX = event.touches[0].pageX; //get finger's x global position mouseY = event.touches[0].pageY; //get finger's y global position } knobCenterX = boundingRectangle.width / 2 + knobPositionX; //get global horizontal center position of knob relative to mouse position knobCenterY = boundingRectangle.height / 2 + knobPositionY; //get global vertical center position of knob relative to mouse position adjacentSide = knobCenterX - mouseX; //compute adjacent value of imaginary right angle triangle oppositeSide = knobCenterY - mouseY; //compute opposite value of imaginary right angle triangle //arc-tangent function returns circular angle in radians //use atan2() instead of atan() because atan() returns only 180 degree max (PI radians) but atan2() returns four quadrant's 360 degree max (2PI radians) currentRadiansAngle = Math.atan2(adjacentSide, oppositeSide); getRadiansInDegrees = currentRadiansAngle * 180 / Math.PI; //convert radians into degrees finalAngleInDegrees = -(getRadiansInDegrees - 135); //knob is already starting at -135 degrees due to visual design so 135 degrees needs to be subtracted to compensate for the angle offset, negative value represents clockwise direction //only allow rotate if greater than zero degrees or lesser than 270 degrees if(finalAngleInDegrees >= 0 && finalAngleInDegrees <= 270) { volumeKnob.style.transform = "rotate(" + finalAngleInDegrees + "deg)"; //use dynamic CSS transform to rotate volume knob //270 degrees maximum freedom of rotation / 100% volume = 1% of volume difference per 2.7 degrees of rotation volumeSetting = Math.floor(finalAngleInDegrees / (270 / 100)); tickHighlightPosition = Math.round((volumeSetting * 2.7) / 10); //interpolate how many ticks need to be highlighted createTicks(27, tickHighlightPosition); //highlight ticks audio.volume = volumeSetting / 100; //set audio volume document.getElementById("volumeValue").innerHTML = volumeSetting + "%"; //update volume text } } //dynamically create volume knob "ticks" function createTicks(numTicks, highlightNumTicks) { //reset first by deleting all existing ticks while(tickContainer.firstChild) { tickContainer.removeChild(tickContainer.firstChild); } //create ticks for(var i=0;i<numTicks;i++) { var tick = document.createElement("div"); //highlight only the appropriate ticks using dynamic CSS if(i < highlightNumTicks) { tick.className = "tick activetick"; } else { tick.className = "tick"; } tickContainer.appendChild(tick); tick.style.transform = "rotate(" + startingTickAngle + "deg)"; startingTickAngle += 10; } startingTickAngle = -135; //reset } //detect for mobile devices from https://www.sitepoint.com/navigator-useragent-mobiles-including-ipad/ function detectMobile() { var result = (navigator.userAgent.match(/(iphone)|(ipod)|(ipad)|(android)|(blackberry)|(windows phone)|(symbian)/i)); if(result !== null) { return "mobile"; } else { return "desktop"; } } function getMouseDown() { if(detectMobile() == "desktop") { return "mousedown"; } else { return "touchstart"; } } function getMouseUp() { if(detectMobile() == "desktop") { return "mouseup"; } else { return "touchend"; } } function getMouseMove() { if(detectMobile() == "desktop") { return "mousemove"; } else { return "touchmove"; } } main();
That’s all! hopefully, you have successfully created the JavaScript knob slider. If you have any questions or suggestions, feel free to comment below.