From de418fe42c703cd3c7b34cb1e18240f414951727 Mon Sep 17 00:00:00 2001 From: Cody Cutrer Date: Sat, 16 Dec 2023 01:49:35 -0700 Subject: [PATCH] [mqtt.espmilighthub] Use system channel types for color temp and brightness (#13801) * [mqtt.espmilighthub] use system channel types for color temp and brightness closes #10096 also add absolute color temp channel. and remove level channel when we already have a color channel (you can link a DimmerItem to a color channel if your light doesn't happen to have color or you want direct control of brightness only for some reason) --------- Signed-off-by: Cody Cutrer --- .../README.md | 8 +- .../EspMilightHubBindingConstants.java | 3 + .../EspMilightStateDescriptionProvider.java | 69 ++++++++++++++++ .../handler/EspMilightHubHandler.java | 35 ++++++-- .../resources/OH-INF/i18n/mqtt.properties | 6 -- .../resources/OH-INF/thing/thing-types.xml | 61 ++++---------- .../resources/OH-INF/update/instructions.xml | 82 +++++++++++++++++++ 7 files changed, 200 insertions(+), 64 deletions(-) create mode 100644 bundles/org.openhab.binding.mqtt.espmilighthub/src/main/java/org/openhab/binding/mqtt/espmilighthub/internal/EspMilightStateDescriptionProvider.java create mode 100644 bundles/org.openhab.binding.mqtt.espmilighthub/src/main/resources/OH-INF/update/instructions.xml diff --git a/bundles/org.openhab.binding.mqtt.espmilighthub/README.md b/bundles/org.openhab.binding.mqtt.espmilighthub/README.md index 1a4e5587b02..b17a35670ec 100644 --- a/bundles/org.openhab.binding.mqtt.espmilighthub/README.md +++ b/bundles/org.openhab.binding.mqtt.espmilighthub/README.md @@ -118,9 +118,10 @@ Note that the group 0 (or ALL group) is not auto discovered as a thing and thus | Channel | Type | Description | |-|-|-| -| `level` | Dimmer | Level changes the brightness of the globe. | +| `level` | Dimmer | Level changes the brightness of the globe. Not present if the bulb supports the `colour` channel. | | `colourTemperature` | Dimmer | Change from cool to warm white with this control. | -| `colour` | Color | Allows you to change the colour, brightness and saturation of the globe. | +| `colourTemperatureAbs` | Number:Temperature | Colour temperature in mireds. | +| `colour` | Color | Allows you to change the colour, brightness and saturation of the globe. Can also be linked directly with a Dimmer item if you happen to have a bulb that doesn't support colour, but is controlled by a remote that normally does support colour. | | `discoMode` | String | Switch to a Disco mode directly from a drop down list. | | `bulbMode` | String (read only) | Displays the mode the bulb is currently in so that rules can determine if the globe is white, a color, disco modes or night mode are selected. | | `command` | String | Sends the raw commands that the buttons on a remote send. | @@ -192,8 +193,9 @@ Thing mqtt:rgb_cct:myBroker:0xE6C4 "Hallway" (mqtt:broker:myBroker) @ "MQTT" *.items ```java -Dimmer Hallway_Level "Front Hall" {channel="mqtt:rgb_cct:myBroker:0xE6C4:level"} +Dimmer Hallway_Level "Front Hall" {channel="mqtt:rgb_cct:myBroker:0xE6C4:colour"} Dimmer Hallway_ColourTemperature "White Color Temp" {channel="mqtt:rgb_cct:myBroker:0xE6C4:colourTemperature"} +Number:Temperature Hallway_ColourTemperatureK "White Color Temp [%d %unit%]" {channel="mqtt:rgb_cct:myBroker:0xE6C4:colourTemperatureAbs", unit="K"} Color Hallway_Colour "Front Hall" ["Lighting"] {channel="mqtt:rgb_cct:myBroker:0xE6C4:colour"} String Hallway_DiscoMode "Disco Mode" {channel="mqtt:rgb_cct:myBroker:0xE6C4:discoMode"} String Hallway_BulbCommand "Send Command" {channel="mqtt:rgb_cct:myBroker:0xE6C4:command"} diff --git a/bundles/org.openhab.binding.mqtt.espmilighthub/src/main/java/org/openhab/binding/mqtt/espmilighthub/internal/EspMilightHubBindingConstants.java b/bundles/org.openhab.binding.mqtt.espmilighthub/src/main/java/org/openhab/binding/mqtt/espmilighthub/internal/EspMilightHubBindingConstants.java index bd5bf8c73e3..2f538b7600e 100644 --- a/bundles/org.openhab.binding.mqtt.espmilighthub/src/main/java/org/openhab/binding/mqtt/espmilighthub/internal/EspMilightHubBindingConstants.java +++ b/bundles/org.openhab.binding.mqtt.espmilighthub/src/main/java/org/openhab/binding/mqtt/espmilighthub/internal/EspMilightHubBindingConstants.java @@ -33,6 +33,8 @@ public class EspMilightHubBindingConstants { public static final String COMMANDS_BASE_TOPIC = "milight/commands/"; public static final String STATUS_TOPIC = "milight/status"; public static final BigDecimal BIG_DECIMAL_100 = new BigDecimal(100); + public static final BigDecimal BIG_DECIMAL_153 = new BigDecimal(153); + public static final BigDecimal BIG_DECIMAL_370 = new BigDecimal(370); // List of all Thing Type UIDs public static final ThingTypeUID THING_TYPE_RGB_CCT = new ThingTypeUID(BINDING_ID, "rgb_cct"); public static final ThingTypeUID THING_TYPE_CCT = new ThingTypeUID(BINDING_ID, "cct"); @@ -48,6 +50,7 @@ public class EspMilightHubBindingConstants { public static final String CHANNEL_LEVEL = "level"; public static final String CHANNEL_COLOUR = "colour"; public static final String CHANNEL_COLOURTEMP = "colourTemperature"; + public static final String CHANNEL_COLOURTEMP_ABS = "colourTemperatureAbs"; public static final String CHANNEL_DISCO_MODE = "discoMode"; public static final String CHANNEL_BULB_MODE = "bulbMode"; public static final String CHANNEL_COMMAND = "command"; diff --git a/bundles/org.openhab.binding.mqtt.espmilighthub/src/main/java/org/openhab/binding/mqtt/espmilighthub/internal/EspMilightStateDescriptionProvider.java b/bundles/org.openhab.binding.mqtt.espmilighthub/src/main/java/org/openhab/binding/mqtt/espmilighthub/internal/EspMilightStateDescriptionProvider.java new file mode 100644 index 00000000000..a03a5d6809c --- /dev/null +++ b/bundles/org.openhab.binding.mqtt.espmilighthub/src/main/java/org/openhab/binding/mqtt/espmilighthub/internal/EspMilightStateDescriptionProvider.java @@ -0,0 +1,69 @@ +/** + * Copyright (c) 2010-2023 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ + +package org.openhab.binding.mqtt.espmilighthub.internal; + +import static org.openhab.binding.mqtt.espmilighthub.internal.EspMilightHubBindingConstants.*; + +import java.math.BigDecimal; +import java.util.Locale; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.eclipse.jdt.annotation.Nullable; +import org.openhab.core.thing.Channel; +import org.openhab.core.thing.ThingRegistry; +import org.openhab.core.thing.type.DynamicStateDescriptionProvider; +import org.openhab.core.types.StateDescription; +import org.openhab.core.types.StateDescriptionFragmentBuilder; +import org.osgi.service.component.annotations.Activate; +import org.osgi.service.component.annotations.Component; +import org.osgi.service.component.annotations.Reference; +import org.slf4j.LoggerFactory; + +/** + * Provides custom state descriptions for system.color-temperature-absolute channels. + * + * @author Cody Cutrer - Initial contribution + */ +@Component(service = DynamicStateDescriptionProvider.class) +@NonNullByDefault +public class EspMilightStateDescriptionProvider implements DynamicStateDescriptionProvider { + private final ThingRegistry thingRegistry; + + @Activate + public EspMilightStateDescriptionProvider(@Reference ThingRegistry thingRegistry) { + this.thingRegistry = thingRegistry; + } + + @Override + public @Nullable StateDescription getStateDescription(Channel channel, @Nullable StateDescription original, + @Nullable Locale locale) { + var logger = LoggerFactory.getLogger(EspMilightStateDescriptionProvider.class); + var channelUID = channel.getUID(); + var thing = thingRegistry.get(channelUID.getThingUID()); + if (thing == null || !SUPPORTED_THING_TYPES.contains(thing.getThingTypeUID()) + || !channelUID.getId().equals(CHANNEL_COLOURTEMP_ABS)) { + return null; + } + + StateDescriptionFragmentBuilder builder; + if (original != null) { + builder = StateDescriptionFragmentBuilder.create(original); + } else { + builder = StateDescriptionFragmentBuilder.create(); + } + builder.withMinimum(BIG_DECIMAL_153).withMaximum(BIG_DECIMAL_370).withStep(BigDecimal.ONE) + .withPattern("%d mired"); + return builder.build().toStateDescription(); + } +} diff --git a/bundles/org.openhab.binding.mqtt.espmilighthub/src/main/java/org/openhab/binding/mqtt/espmilighthub/internal/handler/EspMilightHubHandler.java b/bundles/org.openhab.binding.mqtt.espmilighthub/src/main/java/org/openhab/binding/mqtt/espmilighthub/internal/handler/EspMilightHubHandler.java index cfc64a3877e..58c1712abbe 100644 --- a/bundles/org.openhab.binding.mqtt.espmilighthub/src/main/java/org/openhab/binding/mqtt/espmilighthub/internal/handler/EspMilightHubHandler.java +++ b/bundles/org.openhab.binding.mqtt.espmilighthub/src/main/java/org/openhab/binding/mqtt/espmilighthub/internal/handler/EspMilightHubHandler.java @@ -35,7 +35,9 @@ 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.PercentType; +import org.openhab.core.library.types.QuantityType; import org.openhab.core.library.types.StringType; +import org.openhab.core.library.unit.Units; import org.openhab.core.thing.Bridge; import org.openhab.core.thing.ChannelUID; import org.openhab.core.thing.Thing; @@ -97,7 +99,6 @@ public class EspMilightHubHandler extends BaseThingHandler implements MqttMessag new BigDecimal("-0.5179722"), new BigDecimal("1.5317403"), new BigDecimal("-2.4243787"), new BigDecimal("1.925865"), new BigDecimal("-0.471106") }; - private static final BigDecimal BIG_DECIMAL_153 = new BigDecimal(153); private static final BigDecimal BIG_DECIMAL_217 = new BigDecimal(217); private final Logger logger = LoggerFactory.getLogger(this.getClass()); @@ -155,13 +156,20 @@ public class EspMilightHubHandler extends BaseThingHandler implements MqttMessag String bulbLevel = Helper.resolveJSON(messageJSON, "\"level\":", 3); if (!bulbLevel.isEmpty()) { if ("0".equals(bulbLevel) || "OFF".equals(bulbState)) { - changeChannel(CHANNEL_LEVEL, OnOffType.OFF); + if (!hasRGB()) { + changeChannel(CHANNEL_LEVEL, OnOffType.OFF); + } } else { tempBulbLevel = new PercentType(Integer.valueOf(bulbLevel)); - changeChannel(CHANNEL_LEVEL, tempBulbLevel); + savedLevel = tempBulbLevel.toBigDecimal(); + if (!hasRGB()) { + changeChannel(CHANNEL_LEVEL, tempBulbLevel); + } } } else if ("ON".equals(bulbState) || "OFF".equals(bulbState)) { // NOTE: Level is missing when this runs - changeChannel(CHANNEL_LEVEL, OnOffType.valueOf(bulbState)); + if (!hasRGB()) { + changeChannel(CHANNEL_LEVEL, OnOffType.valueOf(bulbState)); + } } bulbMode = Helper.resolveJSON(messageJSON, "\"bulb_mode\":\"", 5); switch (bulbMode) { @@ -174,6 +182,7 @@ public class EspMilightHubHandler extends BaseThingHandler implements MqttMessag if (!bulbCTempS.isEmpty()) { var bulbCTemp = Integer.valueOf(bulbCTempS); changeChannel(CHANNEL_COLOURTEMP, scaleMireds(bulbCTemp)); + changeChannel(CHANNEL_COLOURTEMP_ABS, new QuantityType(bulbCTemp, Units.MIRED)); if (hasRGB()) { changeChannel(CHANNEL_COLOUR, calculateHSBFromColorTemp(bulbCTemp, tempBulbLevel)); } @@ -198,7 +207,9 @@ public class EspMilightHubHandler extends BaseThingHandler implements MqttMessag new PercentType(Integer.valueOf(bulbSaturation)), tempBulbLevel); changeChannel(CHANNEL_COLOUR, hsb); if (hasCCT()) { - changeChannel(CHANNEL_COLOURTEMP, scaleMireds(calculateColorTempFromHSB(hsb))); + int mireds = calculateColorTempFromHSB(hsb); + changeChannel(CHANNEL_COLOURTEMP, scaleMireds(mireds)); + changeChannel(CHANNEL_COLOURTEMP_ABS, new QuantityType(mireds, Units.MIRED)); } } break; @@ -214,9 +225,6 @@ public class EspMilightHubHandler extends BaseThingHandler implements MqttMessag case "night": if (hasRGB()) { changeChannel(CHANNEL_BULB_MODE, new StringType("night")); - if (config.oneTriggersNightMode) { - changeChannel(CHANNEL_LEVEL, new PercentType("1")); - } } break; } @@ -430,6 +438,17 @@ public class EspMilightHubHandler extends BaseThingHandler implements MqttMessag int scaledCommand = (int) Math.round((370 - (2.17 * Float.valueOf(command.toString())))); sendMQTT("{\"state\":\"ON\",\"level\":" + savedLevel + ",\"color_temp\":" + scaledCommand + "}"); break; + case CHANNEL_COLOURTEMP_ABS: + int mireds; + QuantityType miredsQt; + if (command instanceof QuantityType + && (miredsQt = ((QuantityType) command).toInvertibleUnit(Units.MIRED)) != null) { + mireds = miredsQt.intValue(); + } else { + mireds = Integer.valueOf(command.toString()); + } + sendMQTT("{\"state\":\"ON\",\"level\":" + savedLevel + ",\"color_temp\":" + mireds + "}"); + break; case CHANNEL_COMMAND: sendMQTT("{\"command\":\"" + command + "\"}"); break; diff --git a/bundles/org.openhab.binding.mqtt.espmilighthub/src/main/resources/OH-INF/i18n/mqtt.properties b/bundles/org.openhab.binding.mqtt.espmilighthub/src/main/resources/OH-INF/i18n/mqtt.properties index 6b5bd216cfb..ebbdcb67791 100644 --- a/bundles/org.openhab.binding.mqtt.espmilighthub/src/main/resources/OH-INF/i18n/mqtt.properties +++ b/bundles/org.openhab.binding.mqtt.espmilighthub/src/main/resources/OH-INF/i18n/mqtt.properties @@ -56,10 +56,6 @@ channel-type.mqtt.bulbMode.state.option.white = white channel-type.mqtt.bulbMode.state.option.color = color channel-type.mqtt.bulbMode.state.option.scene = scene channel-type.mqtt.bulbMode.state.option.night = night -channel-type.mqtt.colour.label = Colour -channel-type.mqtt.colour.description = Allows you to change the colour, brightness and saturation of the globe. -channel-type.mqtt.colourTemperature.label = Colour Temperature -channel-type.mqtt.colourTemperature.description = Change from cool to warm white with this control. channel-type.mqtt.command.label = Command channel-type.mqtt.command.description = Send a raw command to the globe/s. channel-type.mqtt.command.state.option.next_mode = Next Mode @@ -83,5 +79,3 @@ channel-type.mqtt.discoMode.state.option.5 = Disco 5 channel-type.mqtt.discoMode.state.option.6 = Disco 6 channel-type.mqtt.discoMode.state.option.7 = Disco 7 channel-type.mqtt.discoMode.state.option.8 = Disco 8 -channel-type.mqtt.level.label = Level -channel-type.mqtt.level.description = Level changes the brightness of the globe. diff --git a/bundles/org.openhab.binding.mqtt.espmilighthub/src/main/resources/OH-INF/thing/thing-types.xml b/bundles/org.openhab.binding.mqtt.espmilighthub/src/main/resources/OH-INF/thing/thing-types.xml index 165320bb1ed..3ca8031feaa 100644 --- a/bundles/org.openhab.binding.mqtt.espmilighthub/src/main/resources/OH-INF/thing/thing-types.xml +++ b/bundles/org.openhab.binding.mqtt.espmilighthub/src/main/resources/OH-INF/thing/thing-types.xml @@ -12,9 +12,9 @@ LED globe with full Colour, and both cool and warm whites. Lightbulb - - - + + + @@ -30,9 +30,9 @@ Use this when your remote is the newer 8 group type called FUT089 and your globes are rgb_cct Lightbulb - - - + + + @@ -48,8 +48,9 @@ Use this when your remote is the newer fut091 and your globes are cct Lightbulb - - + + + @@ -63,8 +64,9 @@ LED globe with both cool and warm white controls Lightbulb - - + + + @@ -78,8 +80,7 @@ RGB Globe with a fixed white Lightbulb - - + @@ -95,8 +96,7 @@ RGB Globe with no white Lightbulb - - + @@ -104,39 +104,6 @@ - - Dimmer - - Level changes the brightness of the globe. - Slider - - Control - Light - - - - - Dimmer - - Change from cool to warm white with this control. - Slider - - Control - ColorTemperature - - - - - Color - - Allows you to change the colour, brightness and saturation of the globe. - ColorLight - - Control - Light - - - String diff --git a/bundles/org.openhab.binding.mqtt.espmilighthub/src/main/resources/OH-INF/update/instructions.xml b/bundles/org.openhab.binding.mqtt.espmilighthub/src/main/resources/OH-INF/update/instructions.xml new file mode 100644 index 00000000000..5c93553087a --- /dev/null +++ b/bundles/org.openhab.binding.mqtt.espmilighthub/src/main/resources/OH-INF/update/instructions.xml @@ -0,0 +1,82 @@ + + + + + + + + system:color-temperature + + + system:color-temperature-abs + + + system:color + + + + + + + + + system:color-temperature + + + system:color-temperature-abs + + + system:color + + + + + + + + system:brightness + + + system:color-temperature + + + system:color-temperature-abs + + + + + + + + system:brightness + + + system:color-temperature + + + system:color-temperature-abs + + + + + + + + + system:color + + + + + + + + + system:color + + + + +