Volume dial web component

How to get this working in DB2 template node
Also nice to have label and value inside knob...

Volume dial web component

Here's a 'Starter for 10'

[{"id":"04e299edc16ea9f5","type":"ui-template","z":"506727cd93754bdb","group":"a7a5b8170da2bea4","page":"","ui":"","name":"Knob","order":1,"width":"2","height":"1","head":"","format":"<template>\n    <div class = \"container\">\n        <p>Current volume: <span ref=\"volumeValue\" class=\"current-value\">0%</span></p>\n        <div class=\"knob-surround\">\n\n            <div ref=\"volumeKnob\" class=\"knob\"></div>\n\n            <span class=\"min\">Min</span>\n            <span class=\"max\">Max</span>\n\n            <div ref=\"tickContainer\" class=\"ticks\"></div>\n\n        </div>\n\n    </div>\n\n</template>\n\n\n<script>\n    /**\n     * \n     * Javascript written by Kevin Lam of zTransitions.com\n     * Original HTML/CSS code forked from Ed Hicks's original HTML/CSS example\n     * Original volume knob graphic design by Ricardo Salazar\n     * \n     */\n\n    export default {\n        data() {\n            // define variables available component-wide\n            // (in <template> and component functions)\n            return {\n                knobPositionX: null,\n                knobPositionY: null,\n                mouseX: null,\n                mouseY: null,\n                knobCenterX: null,\n                knobCenterY: null,\n                adjacentSide: null,\n                oppositeSide: null,\n                currentRadiansAngle: null,\n                getRadiansInDegrees: null,\n                finalAngleInDegrees: null,\n                volumeSetting: null,\n                tickHighlightPosition: null,\n                audio: new Audio(\"https://www.cineblueone.com/maskWall/audio/skylar.mp3\"),\n                startingTickAngle: -130,\n                tickCount: 27,\n\n            }\n        },\n\n        watch: {\n            // Watch for any changes of \"volumeSetting\"\n            volumeSetting: function() {\n                this.send(this.volumeSetting)\n\n            }\n\n        },\n\n        computed: {\n            // Get the knob bounding rectangle. Set here so that html element has been created\n            boundingRectangle: function() {\n                return this.$refs.volumeKnob.getBoundingClientRect()\n            }\n\n        },\n\n        methods: {\n            main: function() {\n                this.audio.volume = 0                                                   // Start at zero volume\n                // Listen for mouse button click\n                this.$refs.volumeKnob.addEventListener(this.getMouseDown(), this.onMouseDown)\n                document.addEventListener(this.getMouseUp(), this.onMouseUp)            // Listen for mouse button release\n\n                this.createTicks(this.tickCount, 0)\n\n            },\n\n            onMouseDown: function() {\n                // Start audio if not already playing\n                if (this.audio.paused == true) {\n                    // Mobile users must tap anywhere to start audio\n                    // https://developers.google.com/web/updates/2017/09/autoplay-policy-changes\n                    let promise = this.audio.play()\n                    if (promise !== undefined) {\n                        promise.then(function() { \n                            this.audio.play() \n                        }).catch(function(error) {})\n                    }\n    \n                }\n\n                document.addEventListener(this.getMouseMove(), this.onMouseMove)        // Start drag\n\n            },\n\n            // )n mouse button release\n            onMouseUp: function() {\n                document.removeEventListener(this.getMouseMove(), this.onMouseMove)     // Release drag\n            },\n\n            // Compute mouse angle relative to center of volume knob\n            onMouseMove: function(event) {\n                // Get knob's global x & y position\n                this.knobPositionX = this.boundingRectangle.left\n                this.knobPositionY = this.boundingRectangle.top\n\n                // Get mouse's or finger's x & y global position\n                if (this.detectMobile() == \"desktop\") {\n                    this.mouseX = event.pageX\n                    this.mouseY = event.pageY\n\n                } else {\n                    this.mouseX = event.touches[0].pageX\n                    this.mouseY = event.touches[0].pageY\n\n                }\n\n                // Get global horizontal center position of knob relative to mouse position\n                this.knobCenterX = this.boundingRectangle.width / 2 + this.knobPositionX\n                // Get global vertical center position of knob relative to mouse position\n                this.knobCenterY = this.boundingRectangle.height / 2 + this.knobPositionY\n\n                // Compute adjacent value of imaginary right angle triangle\n                this.adjacentSide = this.knobCenterX - this.mouseX\n                // Compute opposite value of imaginary right angle triangle\n                this.oppositeSide = this.knobCenterY - this.mouseY\n\n                /*\n                 * arc-tangent function returns circular angle in radians\n                 * use atan2() instead of atan() because atan() returns only 180 degree max (PI radians) \n                 * but atan2() returns four quadrant's 360 degree max (2PI radians)\n                 */\n                this.currentRadiansAngle = Math.atan2(this.adjacentSide, this.oppositeSide)\n\n                // Convert radians into degrees\n                this.getRadiansInDegrees = this.currentRadiansAngle * (180 / Math.PI)\n\n                /* \n                 * Knob is already starting at -130 degrees due to visual design so 130 degrees \n                 * needs to be subtracted to compensate for the angle offset, \n                 * negative value represents clockwise direction\n                */\n                this.finalAngleInDegrees = -(this.getRadiansInDegrees - 130)\n\n                // Only allow rotate if greater than zero degrees or lesser than 270 degrees\n                if (this.finalAngleInDegrees >= 0 && this.finalAngleInDegrees <= 270) { \n                    // Use dynamic CSS transform to rotate volume knob\n                    this.$refs.volumeKnob.style.transform = \"rotate(\" + this.finalAngleInDegrees + \"deg)\"\n\n                    // 270 degrees maximum freedom of rotation / 100% volume = 1% of volume difference per 2.7 degrees of rotation\n                    this.volumeSetting = Math.round(this.finalAngleInDegrees / (270 / 100))\n\n                    // Interpolate how many ticks need to be highlighted\n                    this.tickHighlightPosition = Math.round((this.volumeSetting * (this.tickCount / 10)) / 10)\n\n                    // Highlight ticks\n                    this.createTicks(this.tickCount, this.tickHighlightPosition)\n\n                    this.audio.volume = this.volumeSetting / 100\n    \n                    this.$refs.volumeValue.innerHTML = this.volumeSetting + \"%\"\n                }\n\n            },\n\n            createTicks: function(numTicks, highlightNumTicks) {\n                while (this.$refs.tickContainer.firstChild) {\n                    this.$refs.tickContainer.removeChild(this.$refs.tickContainer.firstChild)\n                }\n\n                for (let i = 0; i < numTicks; i++) { \n                    let tick = document.createElement(\"div\")\n                    if (i < highlightNumTicks) { \n                        tick.className = \"tick activetick\"\n\n                    } else {\n                        tick.className = \"tick\"\n\n                    } \n    \n                    this.$refs.tickContainer.appendChild(tick)\n                    tick.style.transform=\"rotate(\" + this.startingTickAngle + \"deg)\"\n                    this.startingTickAngle += 10\n                }\n\n                this.startingTickAngle = -130\n            },\n    \n            detectMobile: function() { \n                let result = navigator.userAgent.match(/(iphone)|(ipod)|(ipad)|(android)|(blackberry)|(windows phone)|(symbian)/i)\n                this.send(result)\n                if (result !==null) {\n                    return \"mobile\"\n                } else { \n                    return \"desktop\"\n                } \n    \n            },\n    \n            getMouseDown: function() { \n                if (this.detectMobile() == \"desktop\" ) { \n                    return \"mousedown\"\n                } else { \n                    return \"touchstart\" \n                } \n            }, \n    \n            getMouseUp: function() {\n                if (this.detectMobile() == \"desktop\" ) { \n                    return \"mouseup\"\n                } else { \n                    return \"touchend\"\n                } \n            }, \n    \n            getMouseMove: function() { \n                if (this.detectMobile() == \"desktop\" ) { \n                    return \"mousemove\"\n                } else { \n                    return \"touchmove\"\n                } \n            },\n\n        },\n\n    mounted() {\n            // listen for incoming msg's from Node-RED\n            // note our topic is \"msg-input\" + the node's unique ID\n            this.$socket.on('msg-input:' + this.id, (msg) => {\n\n            })\n            this.main()\n\n        },\n\n    }\n\n</script>\n\n<style>\n\t@font-face {\n      font-family: 'Open Sans';\n      font-style: normal;\n      font-weight: 300;\n      src: local('Open Sans Light'), local('OpenSans-Light'), url(https://fonts.gstatic.com/s/opensans/v18/mem5YaGs126MiZpBA-UN_r8OUuhs.ttf) format('truetype');\n    }\n    @font-face {\n      font-family: 'Varela Round';\n      font-style: normal;\n      font-weight: 400;\n      src: local('Varela Round Regular'), local('VarelaRound-Regular'), url(https://fonts.gstatic.com/s/varelaround/v13/w8gdH283Tvk__Lua32TysjIfp8uK.ttf) format('truetype');\n    }\n\n    .container {\n      background-color: #181818;\n      font-size: 100%;\n      font-family: \"Open Sans\", sans-serif;\n      color: #aaa;\n      text-align: center;\n      user-select: none;\n    }\n\n    .knob-surround {\n      position: relative;\n      background-color: grey;\n      width: 14em;\n      height: 14em;\n      border-radius: 50%;\n      border: solid 0.25em #0e0e0e;\n      margin: 5em auto;\n      background: #181818;\n      background: -webkit-gradient(linear, left bottom, left top, color-stop(0, #1d1d1d), color-stop(1, #131313));\n      background: -ms-linear-gradient(bottom, #1d1d1d, #131313);\n      background: -moz-linear-gradient(center bottom, #1d1d1d 0%, #131313 100%);\n      background: -o-linear-gradient(#131313, #1d1d1d);\n      filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#131313', endColorstr='#1d1d1d', GradientType=0);\n      -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);\n      -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);\n      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);\n    }\n\n    .knob {\n      position: absolute;\n      width: 100%;\n      height: 100%;\n      border-radius: 50%;\n      -webkit-transform: rotate(0deg);\n      -moz-transform: rotate(0deg);\n      -o-transform: rotate(0deg);\n      -ms-transform: rotate(0deg);\n      transform: rotate(0deg);\n      z-index: 10;\n    }\n\n    .knob:before {\n      content: \"\";\n      position: absolute;\n      bottom: 19%;\n      left: 19%;\n      width: 3%;\n      height: 3%;\n      background-color: #a8d8f8;\n      border-radius: 50%;\n      -webkit-box-shadow: 0 0 0.4em 0 #79c3f4;\n      -moz-box-shadow: 0 0 0.4em 0 #79c3f4;\n      box-shadow: 0 0 0.4em 0 #79c3f4;\n    }\n\n    .min,\n    .max {\n      display: block;\n      font-family: \"Varela Round\", sans-serif;\n      color: rgba(255, 255, 255, 0.4);\n      text-transform: uppercase;\n      -webkit-font-smoothing: antialiased;\n      font-size: 70%;\n      position: absolute;\n      opacity: 0.5;\n    }\n\n    .min {\n      bottom: 1em;\n      left: -2.5em;\n    }\n\n    .max {\n      bottom: 1em;\n      right: -2.5em;\n    }\n\n    .tick {\n      position: absolute;\n      width: 100%;\n      height: 100%;\n      top: 0;\n      left: 0;\n      z-index: 5;\n      overflow: visible;\n    }\n\n    .tick:after {\n      content: \"\";\n      width: 0.08em;\n      height: 0.6em;\n      background-color: rgba(255, 255, 255, 0.2);\n      position: absolute;\n      top: -1.5em;\n      left: 50%;\n      -webkit-transition: all 180ms ease-out;\n      -moz-transition: all 180ms ease-out;\n      -o-transition: all 180ms ease-out;\n      transition: all 180ms ease-out;\n    }\n\n    .activetick:after {\n      background-color: #a8d8f8;\n      -webkit-box-shadow: 0 0 0.3em 0.08em #79c3f4;\n      -moz-box-shadow: 0 0 0.3em 0.08em #79c3f4;\n      box-shadow: 0 0 0.3em 0.08em #79c3f4;\n      -webkit-transition: all 50ms ease-in;\n      -moz-transition: all 50ms ease-in;\n      -o-transition: all 50ms ease-in;\n      transition: all 50ms ease-in;\n    }\n\n    h1 {\n      font-weight: normal;\n      margin: 2em 0;\n    }\n\n    p {\n      line-height: 150%;\n      max-width: 36em;\n      margin: 1em auto;\n    }\n\n    a {\n      color: #aaa;\n      text-decoration: none;\n      border-bottom: 1px solid #444;\n      -webkit-transition: color 0.2s ease-in;\n      -moz-transition: color 0.2s ease-in;\n      -o-transition: color 0.2s ease-in;\n      transition: color 0.2s ease-in;\n    }\n\n    a:hover,\n    a:focus {\n      color: #eee;\n\n    }\n\n    .container,\n    .knob {\n      background-image: url();\n\n</style>","storeOutMessages":true,"passthru":true,"resendOnRefresh":true,"templateScope":"local","className":"","x":1510,"y":1260,"wires":[["d305099eaa9c6bbc"]]},{"id":"a7a5b8170da2bea4","type":"ui-group","name":"Knob","page":"17179e7821c4a787","width":"6","height":"1","order":1,"showTitle":true,"className":"","visible":"true","disabled":"false","groupType":"default"},{"id":"17179e7821c4a787","type":"ui-page","name":"Page 08","ui":"b810194ea14e3502","path":"/page8","icon":"home","layout":"grid","theme":"5075a7d8e4947586","breakpoints":[{"name":"Default","px":"0","cols":"3"},{"name":"Tablet","px":"576","cols":"6"},{"name":"Small Desktop","px":"768","cols":"9"},{"name":"Desktop","px":"1024","cols":"12"}],"order":14,"className":"","visible":"true","disabled":"false"},{"id":"b810194ea14e3502","type":"ui-base","name":"Dashboard 2 Examples","path":"/dashboard","includeClientData":true,"acceptsClientConfig":["ui-control","ui-notification"],"showPathInSidebar":false,"navigationStyle":"default","titleBarStyle":"default"},{"id":"5075a7d8e4947586","type":"ui-theme","name":"Default Theme","colors":{"surface":"#ffffff","primary":"#0094ce","bgPage":"#eeeeee","groupBg":"#ffffff","groupOutline":"#cccccc"},"sizes":{"pagePadding":"12px","groupGap":"12px","groupBorderRadius":"4px","widgetGap":"12px","density":"default"}}]
1 Like

This is the units as shown in uiBuilder

[{"id":"8ab1feb123d6d184","type":"uibuilder","z":"d2ae61b0cca41c05","name":"","topic":"","url":"Volume-Knob","okToGo":true,"fwdInMessages":false,"allowScripts":false,"allowStyles":false,"copyIndex":true,"templateFolder":"blank","extTemplate":"","showfolder":false,"reload":false,"sourceFolder":"src","deployedVersion":"7.2.0","showMsgUib":false,"title":"","descr":"","editurl":"vscode://vscode-remote/ssh-remote+192.168.1.23/home/pi/.node-red/uibuilder/Volume-Knob/?windowId=_blank","x":1520,"y":780,"wires":[[],[]]}]
1 Like