JavaScript Knob Slider

JavaScript Knob Slider
Project: Javascript draggable volume knob
Author: Kevin Lam
Edit Online: View on CodePen
License: MIT

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.

Leave a Comment

Comments

No comments yet. Why don’t you start the discussion?

Leave a Reply