Problems With Volume Knob In Dashboard 2 Template

I found a volume knob on t'internet and managed to get it sort of working in a Dashboard 2 template node. The non-working part is the fact that this.$refs.volumeKnob.getBoundingClientRect() does not get a valid rectangle. I am hoping that someone with greater knowledge than me can work out why.

<template>
    <div class = "myBody">
        <p>Current volume: <span ref="volumeValue" class="current-value">0%</span></p>
        <div class="knob-surround">

            <div ref="volumeKnob" class="knob"></div>

            <span class="min">Min</span>
            <span class="max">Max</span>

            <div ref="tickContainer" class="ticks"></div>

        </div>

    </div>

</template>


<script>
    export default {
        data() {
            // define variables available component-wide
            // (in <template> and component functions)
            return {
                knobPositionX: null,
                knobPositionY: null,
                mouseX: null,
                mouseY: null,
                knobCenterX: null,
                knobCenterY: null,
                adjacentSide: null,
                oppositeSide: null,
                currentRadiansAngle: null,
                getRadiansInDegrees: null,
                finalAngleInDegrees: null,
                volumeSetting: null,
                tickHighlightPosition: null,
                audio: new Audio("https://www.cineblueone.com/maskWall/audio/skylar.mp3"),
                startingTickAngle: -135,
                //tickContainer: null,
                //volumeKnob: null,
                boundingRectangle: null,
            }
        },

        watch: {
            // watch for any changes of "btnState"
            volumeSetting: function() {
                this.send(this.volumeSetting)

            }

        },

        methods: {
            main: function() {
                this.audio.volume = 0

                this.boundingRectangle = this.$refs.volumeKnob.getBoundingClientRect()
                console.log(this.boundingRectangle.height)

                this.$refs.volumeKnob.addEventListener('mousedown', this.onMouseDown)
                document.addEventListener('mouseup', this.onMouseUp)

                this.createTicks(27, 0)

            },


            onMouseDown: function() {
                if (this.audio.paused == true) {
                    let promise = this.audio.play()
                    if (promise !== undefined) {
                        promise.then(function() { 
                            this.audio.play() 
                        }).catch(function(error) {})
                    }
    
                }

                document.addEventListener('mousemove', this.onMouseMove)

            },

            onMouseUp: function() {
                document.removeEventListener(this.getMouseMove(), this.onMouseMove)
            },

            onMouseMove: function(event) {
                this.knobPositionX = this.boundingRectangle.left
                this.knobPositionY = this.boundingRectangle.top

                //if (this.detectMobile() == "desktop") {
                    this.mouseX = event.pageX
                    this.mouseY = event.pageY
                //} else {
                    //this.mouseX = event.touches[0].pageX
                    //this.mouseY = event.touches[0].pageY
                //}

                this.knobCenterX = this.boundingRectangle.width / 2 + this.knobPositionX
                this.knobCenterY = this.boundingRectangle.height / 2 + this.knobPositionY

                this.adjacentSide = this.knobCenterX - this.mouseX
                this.oppositeSide = this.knobCenterY - this.mouseY
                
                this.currentRadiansAngle = Math.atan2(this.adjacentSide, this.oppositeSide)

                this.getRadiansInDegrees = this.currentRadiansAngle * (180 / Math.PI)
                this.finalAngleInDegrees = -(this.getRadiansInDegrees - 135)
                
                if (this.finalAngleInDegrees >= 0 && this.finalAngleInDegrees <= 270) { 
                    this.$refs.volumeKnob.style.transform = "rotate(" + this.finalAngleInDegrees + "deg)"

                    this.volumeSetting = Math.floor(this.finalAngleInDegrees / (270 / 100))

                    this.tickHighlightPosition = Math.round((this.volumeSetting * 2.7) / 10)
                    this.createTicks(27, this.tickHighlightPosition)

                    this.audio.volume = this.volumeSetting / 100
    
                    this.$refs.volumeValue.innerHTML = this.volumeSetting + "%"
                }

            },

            createTicks: function(numTicks, highlightNumTicks) {
                while (this.$refs.tickContainer.firstChild) {
                    this.$refs.tickContainer.removeChild(this.$refs.tickContainer.firstChild)
                }

                for (let i = 0; i < numTicks; i++) { 
                    let tick = document.createElement("div")
                    if (i < highlightNumTicks) { 
                        tick.className = "tick activetick"

                    } else {
                        tick.className = "tick"

                    } 
    
                    this.$refs.tickContainer.appendChild(tick)
                    tick.style.transform="rotate(" + this.startingTickAngle + "deg)"
                    this.startingTickAngle += 10
                }

                this.startingTickAngle = -135
            },
    
            detectMobile: function() { 
                let result = navigator.userAgent.match(/(iphone)|(ipod)|(ipad)|(android)|(blackberry)|(windows phone)|(symbian)/i)
                this.send(result)
                if (result !==null) {
                    return "mobile"
                } else { 
                    return "desktop"
                } 
    
            },
    
            getMouseDown: function() { 
                if (this.detectMobile() == "desktop" ) { 
                    return "mousedown"
                } else { 
                    return "touchstart" 
                } 
            }, 
    
            getMouseUp: function() {
                if (this.detectMobile() == "desktop" ) { 
                    return "mouseup"
                } else { 
                    return "touchend"
                } 
            }, 
    
            getMouseMove: function() { 
                if (this.detectMobile() == "desktop" ) { 
                    return "mousemove"
                } else { 
                    return "touchmove"
                } 
            },

        },

    mounted() {
            // listen for incoming msg's from Node-RED
            // note our topic is "msg-input" + the node's unique ID
            this.$socket.on('msg-input:' + this.id, (msg) => {

            })
            this.main()

        },

    }

</script>

<style>
	@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');
    }

    .myBody {
      background-color: #181818;
      font-size: 100%;
      font-family: "Open Sans", sans-serif;
      color: #aaa;
      text-align: center;
      user-select: none;
    }

    .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;
    }

    .myBody,
    .knob {
      background-image: url();
    }
</style>

Try putting this at the end of the function (after it's drawn) or make this.boundingRectangle a computed property.

Making it a computed property did the trick.

Final result. Although this is set as a volume control (complete with sample sound track) by amending the action in onMouseDown method it could be used for any number of things

<template>
    <div class = "container">
        <p>Current volume: <span ref="volumeValue" class="current-value">0%</span></p>
        <div class="knob-surround">

            <div ref="volumeKnob" class="knob"></div>

            <span class="min">Min</span>
            <span class="max">Max</span>

            <div ref="tickContainer" class="ticks"></div>

        </div>

    </div>

</template>


<script>
    /**
     * 
     * Javascript written by Kevin Lam of zTransitions.com
     * Original HTML/CSS code forked from Ed Hicks's original HTML/CSS example
     * Original volume knob graphic design by Ricardo Salazar
     * 
     */

    export default {
        data() {
            // define variables available component-wide
            // (in <template> and component functions)
            return {
                knobPositionX: null,
                knobPositionY: null,
                mouseX: null,
                mouseY: null,
                knobCenterX: null,
                knobCenterY: null,
                adjacentSide: null,
                oppositeSide: null,
                currentRadiansAngle: null,
                getRadiansInDegrees: null,
                finalAngleInDegrees: null,
                volumeSetting: null,
                tickHighlightPosition: null,
                audio: new Audio("https://www.cineblueone.com/maskWall/audio/skylar.mp3"),
                startingTickAngle: -130,
                tickCount: 27,

            }
        },

        watch: {
            // Watch for any changes of "volumeSetting" Note: this is NOT part of the original control
            //  I added it to show other options are available
            volumeSetting: function() {
                this.send(this.volumeSetting)

            }

        },

        computed: {
            // Get the knob bounding rectangle. Set here so that html element has been created
            boundingRectangle: function() {
                return this.$refs.volumeKnob.getBoundingClientRect()
            }

        },

        methods: {
            main: function() {
                this.audio.volume = 0                                                   // Start at zero volume
                // Listen for mouse button click
                this.$refs.volumeKnob.addEventListener(this.getMouseDown(), this.onMouseDown)
                document.addEventListener(this.getMouseUp(), this.onMouseUp)            // Listen for mouse button release

                this.createTicks(this.tickCount, 0)

            },

            onMouseDown: function() {
                // Start audio if not already playing
                if (this.audio.paused == true) {
                    // Mobile users must tap anywhere to start audio
                    // https://developers.google.com/web/updates/2017/09/autoplay-policy-changes
                    let promise = this.audio.play()
                    if (promise !== undefined) {
                        promise.then(function() { 
                            this.audio.play() 
                        }).catch(function(error) {})
                    }
    
                }

                document.addEventListener(this.getMouseMove(), this.onMouseMove)        // Start drag

            },

            // On mouse button release
            onMouseUp: function() {
                document.removeEventListener(this.getMouseMove(), this.onMouseMove)     // Release drag
            },

            // Compute mouse angle relative to center of volume knob
            onMouseMove: function(event) {
                // Get knob's global x & y position
                this.knobPositionX = this.boundingRectangle.left
                this.knobPositionY = this.boundingRectangle.top

                // Get mouse's or finger's x & y global position
                if (this.detectMobile() == "desktop") {
                    this.mouseX = event.pageX
                    this.mouseY = event.pageY

                } else {
                    this.mouseX = event.touches[0].pageX
                    this.mouseY = event.touches[0].pageY

                }

                // Get global horizontal center position of knob relative to mouse position
                this.knobCenterX = this.boundingRectangle.width / 2 + this.knobPositionX
                // Get global vertical center position of knob relative to mouse position
                this.knobCenterY = this.boundingRectangle.height / 2 + this.knobPositionY

                // Compute adjacent value of imaginary right angle triangle
                this.adjacentSide = this.knobCenterX - this.mouseX
                // Compute opposite value of imaginary right angle triangle
                this.oppositeSide = this.knobCenterY - this.mouseY

                /*
                 * 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)
                 */
                this.currentRadiansAngle = Math.atan2(this.adjacentSide, this.oppositeSide)

                // Convert radians into degrees
                this.getRadiansInDegrees = this.currentRadiansAngle * (180 / Math.PI)

                /* 
                 * Knob is already starting at -130 degrees due to visual design so 130 degrees 
                 * needs to be subtracted to compensate for the angle offset, 
                 * negative value represents clockwise direction
                */
                this.finalAngleInDegrees = -(this.getRadiansInDegrees - 130)

                // Only allow rotate if greater than zero degrees or lesser than 270 degrees
                if (this.finalAngleInDegrees >= 0 && this.finalAngleInDegrees <= 270) { 
                    // Use dynamic CSS transform to rotate volume knob
                    this.$refs.volumeKnob.style.transform = "rotate(" + this.finalAngleInDegrees + "deg)"

                    // 270 degrees maximum freedom of rotation / 100% volume = 1% of volume difference per 2.7 degrees of rotation
                    this.volumeSetting = Math.round(this.finalAngleInDegrees / (270 / 100))

                    // Interpolate how many ticks need to be highlighted
                    this.tickHighlightPosition = Math.round((this.volumeSetting * (this.tickCount / 10)) / 10)

                    // Highlight ticks
                    this.createTicks(this.tickCount, this.tickHighlightPosition)

                    this.audio.volume = this.volumeSetting / 100
    
                    this.$refs.volumeValue.innerHTML = this.volumeSetting + "%"
                }

            },

            createTicks: function(numTicks, highlightNumTicks) {
                while (this.$refs.tickContainer.firstChild) {
                    this.$refs.tickContainer.removeChild(this.$refs.tickContainer.firstChild)
                }

                for (let i = 0; i < numTicks; i++) { 
                    let tick = document.createElement("div")
                    if (i < highlightNumTicks) { 
                        tick.className = "tick activetick"

                    } else {
                        tick.className = "tick"

                    } 
    
                    this.$refs.tickContainer.appendChild(tick)
                    tick.style.transform="rotate(" + this.startingTickAngle + "deg)"
                    this.startingTickAngle += 10
                }

                this.startingTickAngle = -130
            },
    
            detectMobile: function() { 
                let result = navigator.userAgent.match(/(iphone)|(ipod)|(ipad)|(android)|(blackberry)|(windows phone)|(symbian)/i)
                this.send(result)
                if (result !==null) {
                    return "mobile"
                } else { 
                    return "desktop"
                } 
    
            },
    
            getMouseDown: function() { 
                if (this.detectMobile() == "desktop" ) { 
                    return "mousedown"
                } else { 
                    return "touchstart" 
                } 
            }, 
    
            getMouseUp: function() {
                if (this.detectMobile() == "desktop" ) { 
                    return "mouseup"
                } else { 
                    return "touchend"
                } 
            }, 
    
            getMouseMove: function() { 
                if (this.detectMobile() == "desktop" ) { 
                    return "mousemove"
                } else { 
                    return "touchmove"
                } 
            },

        },

    mounted() {
            // listen for incoming msg's from Node-RED
            // note our topic is "msg-input" + the node's unique ID
            this.$socket.on('msg-input:' + this.id, (msg) => {

            })
            this.main()

        },

    }

</script>

<style>
	@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');
    }

    .container {
      background-color: #181818;
      font-size: 100%;
      font-family: "Open Sans", sans-serif;
      color: #aaa;
      text-align: center;
      user-select: none;
    }

    .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;

    }

    .container,
    .knob {
      background-image: url();

</style>

This topic was automatically closed 14 days after the last reply. New replies are no longer allowed.