[hdpowerview] Improve color channel for Repeater and remove redundant brightness channel (#15880)

* Improve color channel
* Log warning for unexpected color command

---------

Signed-off-by: Jacob Laursen <jacob-github@vindvejr.dk>
This commit is contained in:
Jacob Laursen 2023-11-18 12:44:09 +01:00 committed by GitHub
parent 70095cdb49
commit 7d24cfbc9e
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
7 changed files with 88 additions and 58 deletions

View File

@ -129,12 +129,13 @@ On Generation 3 gateways the signal strength is displayed in dBm (deciBel-milliW
### Channels for Repeaters (Thing type `repeater`)<sup>[1/2]</sup> ### Channels for Repeaters (Thing type `repeater`)<sup>[1/2]</sup>
| Channel | Item Type | Description | | Channel | Item Type | Description |
|-----------------|-----------|---------------------------------------------------------------------------------------------| |-----------------|-----------|----------------------------------------------------------|
| color | Color | Controls the color of the LED ring. A switch item can be linked: ON = white, OFF = turn off | | color | Color | Controls the color of the LED ring. |
| brightness | Dimmer | Controls the brightness of the LED ring. | | color | Dimmer | Controls the brightness of the LED ring. |
| identify | String | Flash repeater to identify. Valid values are: `IDENTIFY` | | color | Switch | Switches the LED ring on or off. |
| blinkingEnabled | Switch | Blink during commands. | | identify | String | Flash repeater to identify. Valid values are: `IDENTIFY` |
| blinkingEnabled | Switch | Blink during commands. |
### Roller Shutter Up/Down Position vs. Open/Close State ### Roller Shutter Up/Down Position vs. Open/Close State
@ -292,7 +293,7 @@ Repeater items<sup>[1/2]</sup>:
```java ```java
Color Bedroom_Repeater_Color "Bedroom Repeater Color" {channel="hdpowerview:repeater:home:r16384:color"} Color Bedroom_Repeater_Color "Bedroom Repeater Color" {channel="hdpowerview:repeater:home:r16384:color"}
Dimmer Bedroom_Repeater_Brightness "Bedroom Repeater Brightness" {channel="hdpowerview:repeater:home:r16384:brightness"} Dimmer Bedroom_Repeater_Brightness "Bedroom Repeater Brightness" {channel="hdpowerview:repeater:home:r16384:color"}
String Bedroom_Repeater_Identify "Bedroom Repeater Identify" {channel="hdpowerview:repeater:home:r16384:identify"} String Bedroom_Repeater_Identify "Bedroom Repeater Identify" {channel="hdpowerview:repeater:home:r16384:identify"}
Switch Bedroom_Repeater_BlinkingEnabled "Bedroom Repeater Blinking Enabled [%s]" {channel="hdpowerview:repeater:home:r16384:blinkingEnabled"} Switch Bedroom_Repeater_BlinkingEnabled "Bedroom Repeater Blinking Enabled [%s]" {channel="hdpowerview:repeater:home:r16384:blinkingEnabled"}
``` ```

View File

@ -51,7 +51,6 @@ public class HDPowerViewBindingConstants {
public static final String CHANNEL_SHADE_REPEATER_RSSI = "repeaterRssi"; public static final String CHANNEL_SHADE_REPEATER_RSSI = "repeaterRssi";
public static final String CHANNEL_REPEATER_COLOR = "color"; public static final String CHANNEL_REPEATER_COLOR = "color";
public static final String CHANNEL_REPEATER_BRIGHTNESS = "brightness";
public static final String CHANNEL_REPEATER_IDENTIFY = "identify"; public static final String CHANNEL_REPEATER_IDENTIFY = "identify";
public static final String CHANNEL_REPEATER_BLINKING_ENABLED = "blinkingEnabled"; public static final String CHANNEL_REPEATER_BLINKING_ENABLED = "blinkingEnabled";

View File

@ -34,6 +34,10 @@ public class Color {
this(brightness, color.getRed(), color.getGreen(), color.getBlue()); this(brightness, color.getRed(), color.getGreen(), color.getBlue());
} }
public Color(int brightness, Color color) {
this(brightness, color.red, color.green, color.blue);
}
public Color(int brightness, int red, int green, int blue) { public Color(int brightness, int red, int green, int blue) {
this.brightness = brightness; this.brightness = brightness;
this.red = red; this.red = red;
@ -41,6 +45,10 @@ public class Color {
this.blue = blue; this.blue = blue;
} }
public boolean isBlack() {
return red == 0 && green == 0 && blue == 0;
}
@Override @Override
public String toString() { public String toString() {
return String.format("%d.%d.%d/%d%%", red, green, blue, brightness); return String.format("%d.%d.%d/%d%%", red, green, blue, brightness);

View File

@ -28,6 +28,7 @@ import org.openhab.binding.hdpowerview.internal.exceptions.HubException;
import org.openhab.binding.hdpowerview.internal.exceptions.HubInvalidResponseException; import org.openhab.binding.hdpowerview.internal.exceptions.HubInvalidResponseException;
import org.openhab.binding.hdpowerview.internal.exceptions.HubMaintenanceException; import org.openhab.binding.hdpowerview.internal.exceptions.HubMaintenanceException;
import org.openhab.core.library.types.HSBType; import org.openhab.core.library.types.HSBType;
import org.openhab.core.library.types.IncreaseDecreaseType;
import org.openhab.core.library.types.OnOffType; import org.openhab.core.library.types.OnOffType;
import org.openhab.core.library.types.PercentType; import org.openhab.core.library.types.PercentType;
import org.openhab.core.library.types.StringType; import org.openhab.core.library.types.StringType;
@ -37,6 +38,7 @@ import org.openhab.core.thing.Thing;
import org.openhab.core.thing.ThingStatus; import org.openhab.core.thing.ThingStatus;
import org.openhab.core.thing.ThingStatusDetail; import org.openhab.core.thing.ThingStatusDetail;
import org.openhab.core.types.Command; import org.openhab.core.types.Command;
import org.openhab.core.types.RefreshType;
import org.openhab.core.types.UnDefType; import org.openhab.core.types.UnDefType;
import org.openhab.core.util.ColorUtil; import org.openhab.core.util.ColorUtil;
import org.slf4j.Logger; import org.slf4j.Logger;
@ -54,6 +56,7 @@ public class HDPowerViewRepeaterHandler extends AbstractHubbedThingHandler {
private static final int REFRESH_INTERVAL_MINUTES = 5; private static final int REFRESH_INTERVAL_MINUTES = 5;
private static final int IDENTITY_PERIOD_SECONDS = 3; private static final int IDENTITY_PERIOD_SECONDS = 3;
private static final int BRIGHTNESS_STEP_PERCENT = 5;
private static final String COMMAND_IDENTIFY = "IDENTIFY"; private static final String COMMAND_IDENTIFY = "IDENTIFY";
private @Nullable ScheduledFuture<?> refreshStatusFuture = null; private @Nullable ScheduledFuture<?> refreshStatusFuture = null;
@ -94,40 +97,17 @@ public class HDPowerViewRepeaterHandler extends AbstractHubbedThingHandler {
logger.warn("Missing bridge handler"); logger.warn("Missing bridge handler");
return; return;
} }
if (command == RefreshType.REFRESH) {
scheduleRefreshJob();
return;
}
HDPowerViewWebTargets webTargets = bridge.getWebTargets(); HDPowerViewWebTargets webTargets = bridge.getWebTargets();
try { try {
RepeaterData repeaterData; RepeaterData repeaterData;
switch (channelUID.getId()) { switch (channelUID.getId()) {
case CHANNEL_REPEATER_COLOR: case CHANNEL_REPEATER_COLOR:
if (command instanceof HSBType hsbCommand) { handleColorCommand(command, webTargets);
Color currentColor = webTargets.getRepeater(repeaterId).color;
if (currentColor != null) {
var color = new Color(currentColor.brightness, ColorUtil.hsbTosRgb(hsbCommand));
repeaterData = webTargets.setRepeaterColor(repeaterId, color);
scheduler.submit(() -> updatePropertyAndStates(repeaterData));
}
} else if (command instanceof OnOffType) {
Color currentColor = webTargets.getRepeater(repeaterId).color;
if (currentColor != null) {
var color = command == OnOffType.ON
? new Color(currentColor.brightness, java.awt.Color.WHITE)
: new Color(currentColor.brightness, java.awt.Color.BLACK);
repeaterData = webTargets.setRepeaterColor(repeaterId, color);
scheduler.submit(() -> updatePropertyAndStates(repeaterData));
}
}
break;
case CHANNEL_REPEATER_BRIGHTNESS:
if (command instanceof PercentType brightnessCommand) {
Color currentColor = webTargets.getRepeater(repeaterId).color;
if (currentColor != null) {
var color = new Color(brightnessCommand.intValue(), currentColor.red, currentColor.green,
currentColor.blue);
repeaterData = webTargets.setRepeaterColor(repeaterId, color);
scheduler.submit(() -> updatePropertyAndStates(repeaterData));
}
}
break; break;
case CHANNEL_REPEATER_IDENTIFY: case CHANNEL_REPEATER_IDENTIFY:
if (command instanceof StringType stringCommand) { if (command instanceof StringType stringCommand) {
@ -163,6 +143,54 @@ public class HDPowerViewRepeaterHandler extends AbstractHubbedThingHandler {
} }
} }
private void handleColorCommand(Command command, HDPowerViewWebTargets webTargets) throws HubException {
if (command instanceof HSBType hsbCommand) {
var color = new Color(hsbCommand.getBrightness().intValue(), ColorUtil.hsbTosRgb(hsbCommand));
RepeaterData repeaterData = webTargets.setRepeaterColor(repeaterId, color);
scheduler.submit(() -> updatePropertyAndStates(repeaterData));
return;
}
Color currentColor = webTargets.getRepeater(repeaterId).color;
if (currentColor == null) {
return;
}
Color newColor;
if (command instanceof PercentType brightnessCommand) {
newColor = applyBrightnessToColor(currentColor, brightnessCommand.intValue());
} else if (command instanceof IncreaseDecreaseType increaseDecreaseCommand) {
int brightness = switch (increaseDecreaseCommand) {
case INCREASE -> currentColor.brightness + BRIGHTNESS_STEP_PERCENT;
case DECREASE -> currentColor.brightness - BRIGHTNESS_STEP_PERCENT;
};
brightness = brightness < 0 ? 0 : brightness > 100 ? 100 : brightness;
newColor = applyBrightnessToColor(currentColor, brightness);
} else if (command instanceof OnOffType) {
// Light is turned off either by RGB black or zero brightness.
int brightness;
if (command == OnOffType.ON) {
// Turn on with maximum brightness level per default,
// if no existing brightness level is available.
brightness = currentColor.brightness > 0 ? currentColor.brightness : 100;
} else {
// Turn off by zero brightness to preserve color.
brightness = 0;
}
newColor = applyBrightnessToColor(currentColor, brightness);
} else {
logger.warn("Unsupported command: {}", command);
return;
}
RepeaterData repeaterData = webTargets.setRepeaterColor(repeaterId, newColor);
scheduler.submit(() -> updatePropertyAndStates(repeaterData));
}
private Color applyBrightnessToColor(Color currentColor, int brightness) {
// If light is off by RGB black, then reset to white since otherwise brightness
// would have no effect; otherwise preserve color.
return currentColor.isBlack() ? new Color(brightness, java.awt.Color.WHITE)
: new Color(brightness, currentColor);
}
private void cancelResetIdentifyStateJob() { private void cancelResetIdentifyStateJob() {
ScheduledFuture<?> scheduledJob = resetIdentifyStateFuture; ScheduledFuture<?> scheduledJob = resetIdentifyStateFuture;
if (scheduledJob != null) { if (scheduledJob != null) {
@ -182,7 +210,7 @@ public class HDPowerViewRepeaterHandler extends AbstractHubbedThingHandler {
private void cancelRefreshJob() { private void cancelRefreshJob() {
ScheduledFuture<?> future = this.refreshStatusFuture; ScheduledFuture<?> future = this.refreshStatusFuture;
if (future != null) { if (future != null) {
future.cancel(false); future.cancel(true);
} }
this.refreshStatusFuture = null; this.refreshStatusFuture = null;
} }
@ -228,10 +256,17 @@ public class HDPowerViewRepeaterHandler extends AbstractHubbedThingHandler {
Color color = repeaterData.color; Color color = repeaterData.color;
if (color != null) { if (color != null) {
logger.debug("Repeater color data received: {}", color.toString()); logger.debug("Repeater color data received: {}", color.toString());
updateState(CHANNEL_REPEATER_COLOR, HSBType.fromRGB(color.red, color.green, color.red)); HSBType hsb;
updateState(CHANNEL_REPEATER_BRIGHTNESS, new PercentType(color.brightness)); if (color.isBlack()) {
// Light is off when RGB black, so discard brightness as otherwise it would appear on.
hsb = HSBType.BLACK;
} else {
hsb = HSBType.fromRGB(color.red, color.green, color.red);
hsb = new HSBType(hsb.getHue(), hsb.getSaturation(), new PercentType(color.brightness));
}
updateState(CHANNEL_REPEATER_COLOR, hsb);
} }
updateState(CHANNEL_REPEATER_BLINKING_ENABLED, repeaterData.blinkEnabled ? OnOffType.ON : OnOffType.OFF); updateState(CHANNEL_REPEATER_BLINKING_ENABLED, OnOffType.from(repeaterData.blinkEnabled));
} }
} }

View File

@ -11,8 +11,7 @@ thing-type.hdpowerview.hub.label = PowerView Hub
thing-type.hdpowerview.hub.description = Hunter Douglas (Luxaflex) PowerView Hub thing-type.hdpowerview.hub.description = Hunter Douglas (Luxaflex) PowerView Hub
thing-type.hdpowerview.repeater.label = PowerView Repeater thing-type.hdpowerview.repeater.label = PowerView Repeater
thing-type.hdpowerview.repeater.description = Hunter Douglas (Luxaflex) PowerView Repeater thing-type.hdpowerview.repeater.description = Hunter Douglas (Luxaflex) PowerView Repeater
thing-type.hdpowerview.repeater.channel.brightness.description = Controls the brightness of the LED ring thing-type.hdpowerview.repeater.channel.color.description = Controls the color and brightness of the LED ring
thing-type.hdpowerview.repeater.channel.color.description = Controls the color of the LED ring
thing-type.hdpowerview.shade.label = PowerView Shade thing-type.hdpowerview.shade.label = PowerView Shade
thing-type.hdpowerview.shade.description = Hunter Douglas (Luxaflex) PowerView Gen 1/2 Shade thing-type.hdpowerview.shade.description = Hunter Douglas (Luxaflex) PowerView Gen 1/2 Shade
thing-type.hdpowerview.shade.channel.hubRssi.label = Hub RSSI thing-type.hdpowerview.shade.channel.hubRssi.label = Hub RSSI

View File

@ -13,16 +13,14 @@
<channels> <channels>
<channel id="color" typeId="system.color"> <channel id="color" typeId="system.color">
<description>Controls the color of the LED ring</description> <description>Controls the color and brightness of the LED ring</description>
</channel>
<channel id="brightness" typeId="system.brightness">
<description>Controls the brightness of the LED ring</description>
</channel> </channel>
<channel id="identify" typeId="repeater-identify"/> <channel id="identify" typeId="repeater-identify"/>
<channel id="blinkingEnabled" typeId="repeater-blinking-enabled"/> <channel id="blinkingEnabled" typeId="repeater-blinking-enabled"/>
</channels> </channels>
<properties> <properties>
<property name="thingTypeVersion">1</property>
<property name="vendor">Hunter Douglas (Luxaflex)</property> <property name="vendor">Hunter Douglas (Luxaflex)</property>
<property name="modelId">PowerView Repeater</property> <property name="modelId">PowerView Repeater</property>
</properties> </properties>

View File

@ -3,19 +3,9 @@
xmlns:update="https://openhab.org/schemas/update-description/v1.0.0" xmlns:update="https://openhab.org/schemas/update-description/v1.0.0"
xsi:schemaLocation="https://openhab.org/schemas/update-description/v1.0.0 https://openhab.org/schemas/update-description-1.0.0.xsd"> xsi:schemaLocation="https://openhab.org/schemas/update-description/v1.0.0 https://openhab.org/schemas/update-description-1.0.0.xsd">
<thing-type uid="powerview:shade"> <thing-type uid="hdpowerview:repeater">
<instruction-set targetVersion="1"> <instruction-set targetVersion="1">
<update-channel id="position"> <remove-channel id="brightness"/>
<type>powerview:shade-position</type>
</update-channel>
<update-channel id="secondary">
<type>powerview:shade-position</type>
<label>Secondary Position</label>
<description>The secondary vertical position (on top-down/bottom-up shades)</description>
</update-channel>
<update-channel id="vane">
<type>powerview:shade-vane</type>
</update-channel>
</instruction-set> </instruction-set>
</thing-type> </thing-type>