diff --git a/CHANGELOG.md b/CHANGELOG.md index 3cf8863..a9bbbad 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,11 +1,19 @@ # Changelog +## [1.0.7] - 2020-08-07 +- Fix issues when waking up from sleep where dynamically rendered images may cause buttons to fail to work. + +## [1.0.6] - 2020-08-04 +- Add ability to show current volume on volume set button + +### Changed + ## [1.0.5] - 2020-06-27 ### Changed - Fixed race condition in cover art rendering -* Fixed an issue causing the cover art to not always be rendered -* Fixed an issue causing the Play button to not update correctly -* Fixed issue where previous cover art was not removed if the next track didn't have cover art available -* Fixed an issue when adding buttons and the default zone wasn't set -* Added ability to reconnect to Roon Core by pressing any key when disconnected +- Fixed an issue causing the cover art to not always be rendered +- Fixed an issue causing the Play button to not update correctly +- Fixed issue where previous cover art was not removed if the next track didn't have cover art available +- Fixed an issue when adding buttons and the default zone wasn't set +- Added ability to reconnect to Roon Core by pressing any key when disconnected diff --git a/Sources/manifest.json b/Sources/manifest.json index 0520708..14f037c 100644 --- a/Sources/manifest.json +++ b/Sources/manifest.json @@ -257,7 +257,7 @@ "CategoryIcon": "plug-in/images/roon-category", "PropertyInspectorPath": "property-inspector/index.html", "URL": "https://bliny.net/streamdeck-roon/", - "Version": "1.0.5", + "Version": "1.0.7", "OS": [ { "Platform": "mac", diff --git a/Sources/plug-in/js/Actions/PlayActionBase.js b/Sources/plug-in/js/Actions/PlayActionBase.js index 8e7829c..eb48e8e 100644 --- a/Sources/plug-in/js/Actions/PlayActionBase.js +++ b/Sources/plug-in/js/Actions/PlayActionBase.js @@ -26,12 +26,13 @@ export default class PlayActionBase extends ActionBase { width: 144, height: 144, }; + + // Trigger image updates based on this Action's custom settings once base image is ready + this.showCoverArt = config && config.settings && config.settings.showCoverArt; + this.showSeekPosition = config && config.settings && config.settings.showSeekPosition; }; - defaultPlayImage.src = PlayImage; - // Trigger image updates based on this Action's custom settings - this.showCoverArt = config && config.settings && config.settings.showCoverArt; - this.showSeekPosition = config && config.settings && config.settings.showSeekPosition; + defaultPlayImage.src = PlayImage; } // ******************************************** diff --git a/Sources/plug-in/js/Actions/VolumeSetAction.js b/Sources/plug-in/js/Actions/VolumeSetAction.js index 1a2940f..98882f0 100644 --- a/Sources/plug-in/js/Actions/VolumeSetAction.js +++ b/Sources/plug-in/js/Actions/VolumeSetAction.js @@ -1,6 +1,7 @@ import ActionBase from "./ActionBase"; import debug from "debug"; import { VolumeSetDisabled } from "../DataImages/volume-set-key-disabled"; +import { VolumeSetImage } from "../DataImages/volume-set-key"; const ACTION_NAME = "volume-set"; const ACTION_UUID = `net.bliny.roon.${ACTION_NAME}`; @@ -15,6 +16,21 @@ export default class VolumeSetAction extends ActionBase { super(config); log("ctor"); + + // Create image data objects for default images + const defaultVolumeImage = new Image(); + defaultVolumeImage.onload = () => { + this._defaultVolumeImageData = { + image: defaultVolumeImage, + width: 144, + height: 144, + }; + + // Trigger image updates based on this Action's custom settings once base image is ready + this.showCoverArt = config && config.settings && config.settings.showCoverArt; + this.showCurrentVolume = config && config.settings && config.settings.showCurrentVolume; + }; + defaultVolumeImage.src = VolumeSetImage; } // ******************************************** @@ -24,9 +40,32 @@ export default class VolumeSetAction extends ActionBase { return ACTION_UUID; } + get defaultVolumeImageData() { + return this._defaultVolumeImageData; + } + + get showCurrentVolume() { + return this._showCurrentVolume; + } + + set showCurrentVolume(value) { + if(value !== this._showCurrentVolume) { + this._showCurrentVolume = (value === true); + this.renderImage(); + } + } + // ******************************************** // * Private methods, event handlers // ******************************************** + onSettingsUpdated(settings) { + super.onSettingsUpdated(settings); + + if(settings) { + this.showCurrentVolume = settings.showCurrentVolume; + } + } + onKeyUp(data) { super.onKeyUp(data); @@ -50,16 +89,67 @@ export default class VolumeSetAction extends ActionBase { onRoonActiveOutputChanged(activeOutput) { super.onRoonActiveOutputChanged(activeOutput); - if(activeOutput !== null && activeOutput.volume) { - this.setImage(undefined); - } else { - this.setImage(this.getDisabledImageWhenRequested()); - } + this.renderImage(); } // ******************************************** // * Private methods // ******************************************** + formatVolume(volume) { + const type = volume.type === "db" ? " dB" : ""; + // return "0"; + return `${volume.value}${type}`; + } + + renderImage() { + if(this.showCurrentVolume === true && this.roonActiveOutput !== null && this.roonActiveOutput.volume) { + const { image, width, height } = this.defaultVolumeImageData; + const canvas = document.createElement("canvas"); + canvas.width = width; + canvas.height = height; + + const canvasContext = canvas.getContext("2d"); + const volumeText = this.formatVolume(this.roonActiveOutput.volume); + + // Render icon + canvasContext.drawImage(image, 0, 0); + + let font; + let left; + if(volumeText.length <= 3) { + left = 70; + font = "36px Arial"; + } else if(volumeText.length <= 6) { + left = 62; + font = "26px Arial"; + } else { + left = 62; + font = "20px Arial"; + } + + canvasContext.font = font; + canvasContext.textAlign = "left"; + canvasContext.textBaseline = "middle"; + canvasContext.shadowColor = "#000000"; + canvasContext.shadowBlur = 1; + canvasContext.lineWidth = 3; + canvasContext.strokeStyle = "#000000"; + canvasContext.strokeText(volumeText, left, height / 2 + 2); + canvasContext.fillStyle = "#ffffff"; + canvasContext.fillText(volumeText, left, height / 2 + 2); + + // Set the image on the button + const dataUri = canvas.toDataURL("image/png"); + this.setImage(dataUri); + } else { + if(this.roonActiveOutput !== null && this.roonActiveOutput.volume) { + this.setImage(undefined); + } else { + this.setImage(this.getDisabledImageWhenRequested()); + } + } + } + getDisabledImage() { return VolumeSetDisabled; } diff --git a/Sources/plug-in/js/DataImages/volume-set-key.js b/Sources/plug-in/js/DataImages/volume-set-key.js new file mode 100644 index 0000000..7777520 --- /dev/null +++ b/Sources/plug-in/js/DataImages/volume-set-key.js @@ -0,0 +1 @@ +export const VolumeSetImage = ""; diff --git a/Sources/plug-in/js/index.js b/Sources/plug-in/js/index.js index 3b4d1e4..75f905a 100644 --- a/Sources/plug-in/js/index.js +++ b/Sources/plug-in/js/index.js @@ -10,7 +10,7 @@ const roonLog = debug("roon"); const roonSubscribeLog = debug("roon:subscribe"); const roonUpdateLog = debug("roon:update"); -const VERSION_NUMBER = "1.0.5"; +const VERSION_NUMBER = "1.0.7"; // TODO: Adjust log level from config // debug.enable("plug-in,roon,roon:subscribe,roon:update,action:*"); diff --git a/Sources/property-inspector/index.html b/Sources/property-inspector/index.html index 0fb78cd..9f1b83a 100644 --- a/Sources/property-inspector/index.html +++ b/Sources/property-inspector/index.html @@ -60,16 +60,16 @@ -