From 58a094b09af3aa122cc572587eeb986ae347b427 Mon Sep 17 00:00:00 2001 From: Christoph Weitkamp Date: Sat, 16 Oct 2021 07:27:27 +0200 Subject: [PATCH] [avmfritz] Added support for DECT500 and HAN-FUN bulbs (#11348) * Added support for DECT500 and HAN-FUN bulbs Signed-off-by: Christoph Weitkamp * Incorporated comment from review Signed-off-by: Christoph Weitkamp --- .../org.openhab.binding.avmfritz/README.md | 6 +- .../internal/AVMFritzBindingConstants.java | 15 +++-- .../internal/AVMFritzHandlerFactory.java | 3 + .../discovery/AVMFritzDiscoveryService.java | 11 ++-- .../internal/dto/AVMFritzBaseModel.java | 17 +++++- .../internal/dto/ColorControlModel.java | 43 ++++++++++++++ .../avmfritz/internal/dto/DeviceModel.java | 26 +++++++-- ...ntrolModel.java => LevelControlModel.java} | 6 +- .../handler/AVMFritzBaseThingHandler.java | 54 +++++++++++++++--- .../AVMFritzColorLightDeviceHandler.java | 30 ++++++++++ .../hardware/FritzAhaContentExchange.java | 5 +- .../hardware/FritzAhaWebInterface.java | 30 ++++++---- .../callbacks/FritzAhaSetColorCallback.java | 57 +++++++++++++++++++ ...> FritzAhaSetLevelPercentageCallback.java} | 10 ++-- .../resources/OH-INF/thing/thing-types.xml | 19 +++++++ .../dto/AVMFritzDeviceListModelTest.java | 20 +++---- .../AVMFritzDiscoveryServiceOSGiTest.java | 3 +- 17 files changed, 294 insertions(+), 61 deletions(-) create mode 100644 bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/dto/ColorControlModel.java rename bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/dto/{LevelcontrolModel.java => LevelControlModel.java} (87%) create mode 100644 bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/handler/AVMFritzColorLightDeviceHandler.java create mode 100644 bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/hardware/callbacks/FritzAhaSetColorCallback.java rename bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/hardware/callbacks/{FritzAhaSetBlindLevelCallback.java => FritzAhaSetLevelPercentageCallback.java} (78%) diff --git a/bundles/org.openhab.binding.avmfritz/README.md b/bundles/org.openhab.binding.avmfritz/README.md index ad2d4fc2d94..01537589994 100644 --- a/bundles/org.openhab.binding.avmfritz/README.md +++ b/bundles/org.openhab.binding.avmfritz/README.md @@ -67,6 +67,10 @@ The FRITZ!DECT 400 supports a configurable button to trigger short or long press Beside four customizable buttons the FRITZ!DECT 440 supports temperature readings. ** NOTE: ** FRITZ!DECT 440 now uses Channel Groups to group its Channels like `device#battery_level`, `device#battery_low` for device information, `sensors#temperature` for sensor data and `top-left`, `bottom-left`, `top-right` and `bottom-right` combined with `press` and `last_change` (see [Full Example](#full-example)) +### FRITZ!DECT 500 + +The [FRITZ!DECT 500](https://avm.de/produkte/fritzdect/fritzdect-500/) is a dimmable colorized light bulb. + #### Supported Channel Groups | Channel Group ID | Description | Available on thing | @@ -178,7 +182,7 @@ The AIN (actor identification number) can be found in the FRITZ!Box interface -> | power | Number:Power | Current power consumption | FRITZ!DECT 210, FRITZ!DECT 200, FRITZ!Powerline 546E | | voltage | Number:ElectricPotential | Current voltage - FRITZ!OS 7 | FRITZ!DECT 210, FRITZ!DECT 200, FRITZ!Powerline 546E | | outlet | Switch | Switchable outlet (ON/OFF) | FRITZ!DECT 210, FRITZ!DECT 200, FRITZ!Powerline 546E | -| on_off | Switch | Switchable device (ON/OFF) | HAN_FUN_ON_OFF | +| on_off | Switch | Switchable device (ON/OFF) | FRITZ!DECT 500, HAN_FUN_ON_OFF | | actual_temp | Number:Temperature | Current temperature of heating thermostat | FRITZ!DECT 301, FRITZ!DECT 300, Comet DECT | | set_temp | Number:Temperature | Set Temperature of heating thermostat | FRITZ!DECT 301, FRITZ!DECT 300, Comet DECT | | eco_temp | Number:Temperature | Eco Temperature of heating thermostat | FRITZ!DECT 301, FRITZ!DECT 300, Comet DECT | diff --git a/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/AVMFritzBindingConstants.java b/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/AVMFritzBindingConstants.java index bbdf5dc3d28..7abd3d26a96 100644 --- a/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/AVMFritzBindingConstants.java +++ b/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/AVMFritzBindingConstants.java @@ -12,7 +12,6 @@ */ package org.openhab.binding.avmfritz.internal; -import java.util.Collections; import java.util.Set; import java.util.stream.Collectors; import java.util.stream.Stream; @@ -41,6 +40,7 @@ public class AVMFritzBindingConstants { public static final String POWERLINE_MODEL_NAME = "FRITZ!Powerline"; // List of main device types + public static final String DEVICE_DECT500 = "FRITZ_DECT_500"; public static final String DEVICE_DECT400 = "FRITZ_DECT_400"; public static final String DEVICE_DECT440 = "FRITZ_DECT_440"; public static final String DEVICE_DECT301 = "FRITZ_DECT_301"; @@ -62,6 +62,7 @@ public class AVMFritzBindingConstants { // List of all Thing Type UIDs public static final ThingTypeUID BRIDGE_THING_TYPE = new ThingTypeUID(BINDING_ID, BRIDGE_FRITZBOX); + public static final ThingTypeUID DECT500_THING_TYPE = new ThingTypeUID(BINDING_ID, DEVICE_DECT500); public static final ThingTypeUID DECT400_THING_TYPE = new ThingTypeUID(BINDING_ID, DEVICE_DECT400); public static final ThingTypeUID DECT440_THING_TYPE = new ThingTypeUID(BINDING_ID, DEVICE_DECT440); public static final ThingTypeUID DECT301_THING_TYPE = new ThingTypeUID(BINDING_ID, DEVICE_DECT301); @@ -132,6 +133,8 @@ public class AVMFritzBindingConstants { public static final String CHANNEL_LAST_CHANGE = "last_change"; public static final String CHANNEL_ROLLERSHUTTER = "rollershutter"; public static final String CHANNEL_ON_OFF = "on_off"; + public static final String CHANNEL_COLOR = "color"; + public static final String CHANNEL_BRIGHTNESS = "brightness"; // List of all Channel config ids public static final String CONFIG_CHANNEL_TEMP_OFFSET = "offset"; @@ -164,6 +167,8 @@ public class AVMFritzBindingConstants { public static final String MODE_WINDOW_OPEN = "WINDOW_OPEN"; public static final String MODE_UNKNOWN = "UNKNOWN"; + public static final Set SUPPORTED_LIGHTING_THING_TYPES = Set.of(DECT500_THING_TYPE); + public static final Set SUPPORTED_BUTTON_THING_TYPES_UIDS = Set.of(DECT400_THING_TYPE, DECT440_THING_TYPE, HAN_FUN_SWITCH_THING_TYPE); @@ -180,8 +185,8 @@ public class AVMFritzBindingConstants { public static final Set SUPPORTED_BRIDGE_THING_TYPES_UIDS = Set.of(BRIDGE_THING_TYPE, PL546E_STANDALONE_THING_TYPE); - public static final Set SUPPORTED_THING_TYPES_UIDS = Collections.unmodifiableSet(Stream - .of(SUPPORTED_BUTTON_THING_TYPES_UIDS, SUPPORTED_HEATING_THING_TYPES, SUPPORTED_DEVICE_THING_TYPES_UIDS, - SUPPORTED_GROUP_THING_TYPES_UIDS, SUPPORTED_BRIDGE_THING_TYPES_UIDS) - .flatMap(Set::stream).collect(Collectors.toSet())); + public static final Set SUPPORTED_THING_TYPES_UIDS = Stream.of(SUPPORTED_LIGHTING_THING_TYPES, + SUPPORTED_BUTTON_THING_TYPES_UIDS, SUPPORTED_HEATING_THING_TYPES, SUPPORTED_DEVICE_THING_TYPES_UIDS, + SUPPORTED_GROUP_THING_TYPES_UIDS, SUPPORTED_BRIDGE_THING_TYPES_UIDS).flatMap(Set::stream) + .collect(Collectors.toUnmodifiableSet()); } diff --git a/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/AVMFritzHandlerFactory.java b/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/AVMFritzHandlerFactory.java index 10738a1541c..5d396a2c66a 100644 --- a/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/AVMFritzHandlerFactory.java +++ b/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/AVMFritzHandlerFactory.java @@ -18,6 +18,7 @@ import org.eclipse.jdt.annotation.NonNullByDefault; import org.eclipse.jdt.annotation.Nullable; import org.eclipse.jetty.client.HttpClient; import org.openhab.binding.avmfritz.internal.handler.AVMFritzButtonHandler; +import org.openhab.binding.avmfritz.internal.handler.AVMFritzColorLightDeviceHandler; import org.openhab.binding.avmfritz.internal.handler.AVMFritzHeatingDeviceHandler; import org.openhab.binding.avmfritz.internal.handler.AVMFritzHeatingGroupHandler; import org.openhab.binding.avmfritz.internal.handler.BoxHandler; @@ -76,6 +77,8 @@ public class AVMFritzHandlerFactory extends BaseThingHandlerFactory { return new BoxHandler((Bridge) thing, httpClient, commandDescriptionProvider); } else if (PL546E_STANDALONE_THING_TYPE.equals(thingTypeUID)) { return new Powerline546EHandler((Bridge) thing, httpClient, commandDescriptionProvider); + } else if (SUPPORTED_LIGHTING_THING_TYPES.contains(thingTypeUID)) { + return new AVMFritzColorLightDeviceHandler(thing); } else if (SUPPORTED_BUTTON_THING_TYPES_UIDS.contains(thingTypeUID)) { return new AVMFritzButtonHandler(thing); } else if (SUPPORTED_HEATING_THING_TYPES.contains(thingTypeUID)) { diff --git a/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/discovery/AVMFritzDiscoveryService.java b/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/discovery/AVMFritzDiscoveryService.java index c9534009481..2d92bb762d7 100644 --- a/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/discovery/AVMFritzDiscoveryService.java +++ b/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/discovery/AVMFritzDiscoveryService.java @@ -15,7 +15,6 @@ package org.openhab.binding.avmfritz.internal.discovery; import static org.openhab.binding.avmfritz.internal.AVMFritzBindingConstants.*; import static org.openhab.core.thing.Thing.*; -import java.util.Collections; import java.util.HashMap; import java.util.Map; import java.util.Set; @@ -56,12 +55,10 @@ public class AVMFritzDiscoveryService extends AbstractDiscoveryService private @NonNullByDefault({}) AVMFritzBaseBridgeHandler bridgeHandler; public AVMFritzDiscoveryService() { - super(Collections - .unmodifiableSet(Stream - .of(SUPPORTED_BUTTON_THING_TYPES_UIDS, SUPPORTED_HEATING_THING_TYPES, - SUPPORTED_DEVICE_THING_TYPES_UIDS, SUPPORTED_GROUP_THING_TYPES_UIDS) - .flatMap(Set::stream).collect(Collectors.toSet())), - 30); + super(Stream + .of(SUPPORTED_LIGHTING_THING_TYPES, SUPPORTED_BUTTON_THING_TYPES_UIDS, SUPPORTED_HEATING_THING_TYPES, + SUPPORTED_DEVICE_THING_TYPES_UIDS, SUPPORTED_GROUP_THING_TYPES_UIDS) + .flatMap(Set::stream).collect(Collectors.toUnmodifiableSet()), 30); } @Override diff --git a/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/dto/AVMFritzBaseModel.java b/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/dto/AVMFritzBaseModel.java index 82bdae0d630..ed01668ca24 100644 --- a/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/dto/AVMFritzBaseModel.java +++ b/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/dto/AVMFritzBaseModel.java @@ -26,6 +26,7 @@ import org.eclipse.jdt.annotation.Nullable; * *
    *
  1. Bit 0: HAN-FUN Gerät
  2. + *
  3. Bit 2: Licht/Lampe
  4. *
  5. Bit 3: HAN-FUN Button - undocumented
  6. *
  7. Bit 4: Alarm-Sensor
  8. *
  9. Bit 5: AVM-Button
  10. @@ -37,6 +38,8 @@ import org.eclipse.jdt.annotation.Nullable; *
  11. Bit 11: Mikrofon
  12. *
  13. Bit 13: HAN-FUN Unit
  14. *
  15. Bit 15: an-/ausschaltbares Gerät / Steckdose / Lampe / Aktor
  16. + *
  17. Bit 16: Gerät mit einstellbarem Dimm-, Höhen- bzw. Niveau-Level
  18. + *
  19. Bit 17: Lampe mit einstellbarer Farbe/Farbtemperatur
  20. *
  21. Bit 18: Rollladen - hoch, runter, stop und level 0% bis 100 %
  22. *
* @@ -47,6 +50,7 @@ import org.eclipse.jdt.annotation.Nullable; */ public abstract class AVMFritzBaseModel implements BatteryModel { protected static final int HAN_FUN_DEVICE_BIT = 1; // Bit 0 + protected static final int LIGHT_BIT = 1 << 2; // Bit 2 protected static final int HAN_FUN_BUTTON_BIT = 1 << 3; // Bit 3 - undocumented protected static final int HAN_FUN_ALARM_SENSOR_BIT = 1 << 4; // Bit 4 protected static final int BUTTON_BIT = 1 << 5; // Bit 5 @@ -58,6 +62,8 @@ public abstract class AVMFritzBaseModel implements BatteryModel { protected static final int MICROPHONE_BIT = 1 << 11; // Bit 11 protected static final int HAN_FUN_UNIT_BIT = 1 << 13; // Bit 13 protected static final int HAN_FUN_ON_OFF_BIT = 1 << 15; // Bit 15 + protected static final int DIMMABLE_LIGHT_BIT = 1 << 16; // Bit 16 + protected static final int COLOR_LIGHT_BIT = 1 << 17; // Bit 17 protected static final int HAN_FUN_BLINDS_BIT = 1 << 18; // Bit 18 protected static final int HUMIDITY_SENSOR_BIT = 1 << 20; // Bit 20 - undocumented @@ -195,6 +201,14 @@ public abstract class AVMFritzBaseModel implements BatteryModel { return (bitmask / HAN_FUN_ON_OFF_BIT) > 0; } + public boolean isDimmableLight() { + return (bitmask & DIMMABLE_LIGHT_BIT) > 0; + } + + public boolean isColorLight() { + return (bitmask & COLOR_LIGHT_BIT) > 0; + } + public boolean isHANFUNBlinds() { return (bitmask & HAN_FUN_BLINDS_BIT) > 0; } @@ -239,7 +253,8 @@ public abstract class AVMFritzBaseModel implements BatteryModel { .append(",isPowermeter=").append(isPowermeter()).append(",isDectRepeater=").append(isDectRepeater()) .append(",isHeatingThermostat=").append(isHeatingThermostat()).append(",hasMicrophone=") .append(hasMicrophone()).append(",isHANFUNUnit=").append(isHANFUNUnit()).append(",isHANFUNOnOff=") - .append(isHANFUNOnOff()).append(",isHANFUNBlind=").append(isHANFUNBlinds()).append(",id=") + .append(isHANFUNOnOff()).append(",isDimmableLight=").append(isDimmableLight()).append(",isColorLight=") + .append(isColorLight()).append(",isHANFUNBlind=").append(isHANFUNBlinds()).append(",id=") .append(deviceId).append(",manufacturer=").append(deviceManufacturer).append(",productname=") .append(productName).append(",fwversion=").append(firmwareVersion).append(",present=").append(present) .append(",name=").append(name).append(",battery=").append(getBattery()).append(",batterylow=") diff --git a/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/dto/ColorControlModel.java b/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/dto/ColorControlModel.java new file mode 100644 index 00000000000..37a9522df3e --- /dev/null +++ b/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/dto/ColorControlModel.java @@ -0,0 +1,43 @@ +/** + * Copyright (c) 2010-2021 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.avmfritz.internal.dto; + +import javax.xml.bind.annotation.XmlAccessType; +import javax.xml.bind.annotation.XmlAccessorType; +import javax.xml.bind.annotation.XmlAttribute; +import javax.xml.bind.annotation.XmlRootElement; + +/** + * See {@link DeviceListModel}. + * + * @author Joshua Bacher - Initial contribution + */ +@XmlAccessorType(XmlAccessType.FIELD) +@XmlRootElement(name = "colorcontrol") +public class ColorControlModel { + + @XmlAttribute(name = "supported_modes") + public int supportedModes; + @XmlAttribute(name = "current_mode") + public int currentMode; + public int hue; + public int saturation; + public int temperature; + + @Override + public String toString() { + return new StringBuilder("[supportedModes=").append(supportedModes).append(",currentMode=").append(currentMode) + .append(",hue=").append(hue).append(",saturation=").append(saturation).append(",temperature=") + .append(temperature).append("]").toString(); + } +} diff --git a/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/dto/DeviceModel.java b/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/dto/DeviceModel.java index 7596026529d..afcebc37c3c 100644 --- a/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/dto/DeviceModel.java +++ b/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/dto/DeviceModel.java @@ -33,7 +33,12 @@ public class DeviceModel extends AVMFritzBaseModel { private TemperatureModel temperature; private HumidityModel humidity; private AlertModel alert; - private LevelcontrolModel levelcontrol; + + @XmlElement(name = "levelcontrol") + private LevelControlModel levelControlModel; + + @XmlElement(name = "colorcontrol") + private ColorControlModel colorControlModel; @XmlElement(name = "button", type = ButtonModel.class) private List buttons; @@ -64,12 +69,20 @@ public class DeviceModel extends AVMFritzBaseModel { this.alert = alertModel; } - public LevelcontrolModel getLevelcontrol() { - return levelcontrol; + public LevelControlModel getLevelControlModel() { + return levelControlModel; } - public void setLevelcontrol(LevelcontrolModel levelcontrol) { - this.levelcontrol = levelcontrol; + public void setLevelControlModel(LevelControlModel levelControlModel) { + this.levelControlModel = levelControlModel; + } + + public ColorControlModel getColorControlModel() { + return colorControlModel; + } + + public void setColorControlModel(ColorControlModel colorControlModel) { + this.colorControlModel = colorControlModel; } public List getButtons() { @@ -91,7 +104,8 @@ public class DeviceModel extends AVMFritzBaseModel { @Override public String toString() { return new StringBuilder(super.toString()).append(temperature).append(",").append(humidity).append(",") - .append(alert).append(",").append(getButtons()).append(",").append(etsiunitinfo).append("]").toString(); + .append(alert).append(",").append(levelControlModel).append(",").append(colorControlModel).append(",") + .append(getButtons()).append(",").append(etsiunitinfo).append("]").toString(); } @XmlAccessorType(XmlAccessType.FIELD) diff --git a/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/dto/LevelcontrolModel.java b/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/dto/LevelControlModel.java similarity index 87% rename from bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/dto/LevelcontrolModel.java rename to bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/dto/LevelControlModel.java index 2d2198f6980..ea59711f60b 100644 --- a/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/dto/LevelcontrolModel.java +++ b/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/dto/LevelControlModel.java @@ -26,7 +26,7 @@ import javax.xml.bind.annotation.XmlRootElement; */ @XmlAccessorType(XmlAccessType.FIELD) @XmlRootElement(name = "levelcontrol") -public class LevelcontrolModel { +public class LevelControlModel { @XmlElement(name = "level") private BigDecimal level; @@ -52,7 +52,7 @@ public class LevelcontrolModel { @Override public String toString() { - return new StringBuilder().append("[level=").append(getLevel()).append(",levelpercentage=") - .append(getLevelPercentage()).append("]").toString(); + return new StringBuilder("[level=").append(getLevel()).append(",levelpercentage=").append(getLevelPercentage()) + .append("]").toString(); } } diff --git a/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/handler/AVMFritzBaseThingHandler.java b/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/handler/AVMFritzBaseThingHandler.java index 00debd0235e..9befbbd8ab7 100644 --- a/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/handler/AVMFritzBaseThingHandler.java +++ b/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/handler/AVMFritzBaseThingHandler.java @@ -29,11 +29,12 @@ import org.openhab.binding.avmfritz.internal.config.AVMFritzDeviceConfiguration; import org.openhab.binding.avmfritz.internal.dto.AVMFritzBaseModel; import org.openhab.binding.avmfritz.internal.dto.AlertModel; import org.openhab.binding.avmfritz.internal.dto.BatteryModel; +import org.openhab.binding.avmfritz.internal.dto.ColorControlModel; import org.openhab.binding.avmfritz.internal.dto.DeviceModel; import org.openhab.binding.avmfritz.internal.dto.HeatingModel; import org.openhab.binding.avmfritz.internal.dto.HeatingModel.NextChangeModel; import org.openhab.binding.avmfritz.internal.dto.HumidityModel; -import org.openhab.binding.avmfritz.internal.dto.LevelcontrolModel; +import org.openhab.binding.avmfritz.internal.dto.LevelControlModel; import org.openhab.binding.avmfritz.internal.dto.PowerMeterModel; import org.openhab.binding.avmfritz.internal.dto.SimpleOnOffModel; import org.openhab.binding.avmfritz.internal.dto.SwitchModel; @@ -44,6 +45,7 @@ import org.openhab.binding.avmfritz.internal.hardware.callbacks.FritzAhaSetBlind import org.openhab.core.config.core.Configuration; import org.openhab.core.library.types.DateTimeType; import org.openhab.core.library.types.DecimalType; +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.OpenClosedType; @@ -156,7 +158,12 @@ public abstract class AVMFritzBaseThingHandler extends BaseThingHandler implemen updateHANFUNAlarmSensor(deviceModel.getAlert()); } if (deviceModel.isHANFUNBlinds()) { - updateLevelcontrol(deviceModel.getLevelcontrol()); + updateLevelControl(deviceModel.getLevelControlModel()); + } + if (deviceModel.isColorLight()) { + updateColorLight(deviceModel.getColorControlModel(), deviceModel.getLevelControlModel()); + } else if (deviceModel.isDimmableLight()) { + updateDimmableLight(deviceModel.getLevelControlModel()); } } } @@ -185,9 +192,26 @@ public abstract class AVMFritzBaseThingHandler extends BaseThingHandler implemen } } - protected void updateLevelcontrol(@Nullable LevelcontrolModel levelcontrolModel) { - if (levelcontrolModel != null) { - updateThingChannelState(CHANNEL_ROLLERSHUTTER, new PercentType(levelcontrolModel.getLevelPercentage())); + protected void updateLevelControl(@Nullable LevelControlModel levelControlModel) { + if (levelControlModel != null) { + updateThingChannelState(CHANNEL_ROLLERSHUTTER, new PercentType(levelControlModel.getLevelPercentage())); + } + } + + private void updateDimmableLight(@Nullable LevelControlModel levelControlModel) { + if (levelControlModel != null) { + updateThingChannelState(CHANNEL_BRIGHTNESS, new PercentType(levelControlModel.getLevelPercentage())); + } + } + + private void updateColorLight(@Nullable ColorControlModel colorControlModel, + @Nullable LevelControlModel levelControlModel) { + if (colorControlModel != null && levelControlModel != null) { + DecimalType hue = new DecimalType(colorControlModel.hue); + PercentType saturation = new PercentType(colorControlModel.saturation); + PercentType brightness = new PercentType(levelControlModel.getLevelPercentage()); + updateThingChannelState(CHANNEL_COLOR, new HSBType(hue, saturation, brightness)); + updateThingChannelState(CHANNEL_BRIGHTNESS, brightness); } } @@ -384,6 +408,21 @@ public abstract class AVMFritzBaseThingHandler extends BaseThingHandler implemen fritzBox.setSwitch(ain, OnOffType.ON.equals(command)); } break; + case CHANNEL_COLOR: + case CHANNEL_BRIGHTNESS: + BigDecimal brightness = null; + if (command instanceof HSBType) { + HSBType hsbType = (HSBType) command; + brightness = hsbType.getBrightness().toBigDecimal(); + fritzBox.setHueAndSaturation(ain, hsbType.getHue().intValue(), hsbType.getSaturation().intValue(), + 0); + } else if (command instanceof PercentType) { + brightness = ((PercentType) command).toBigDecimal(); + } + if (brightness != null) { + fritzBox.setLevelPercentage(ain, brightness); + } + break; case CHANNEL_SETTEMP: BigDecimal temperature = null; if (command instanceof DecimalType) { @@ -462,9 +501,8 @@ public abstract class AVMFritzBaseThingHandler extends BaseThingHandler implemen fritzBox.setBlind(ain, BlindCommand.CLOSE); } } else if (command instanceof PercentType) { - PercentType rollershutterCommand = (PercentType) command; - BigDecimal levelpercentage = rollershutterCommand.toBigDecimal(); - fritzBox.setLevelpercentage(ain, levelpercentage); + BigDecimal levelPercentage = ((PercentType) command).toBigDecimal(); + fritzBox.setLevelPercentage(ain, levelPercentage); } else { logger.debug("Received unknown rollershutter command type '{}'", command.toString()); } diff --git a/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/handler/AVMFritzColorLightDeviceHandler.java b/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/handler/AVMFritzColorLightDeviceHandler.java new file mode 100644 index 00000000000..235f4516fec --- /dev/null +++ b/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/handler/AVMFritzColorLightDeviceHandler.java @@ -0,0 +1,30 @@ +/** + * Copyright (c) 2010-2021 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.avmfritz.internal.handler; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.openhab.core.thing.Thing; + +/** + * Handler for a FRITZ! color light device. Handles commands, which are sent to one of the channels. + * + * @author Joshua Bacher - Initial contribution + * @author Christoph Weitkamp - Initial contribution + */ +@NonNullByDefault +public class AVMFritzColorLightDeviceHandler extends DeviceHandler { + + public AVMFritzColorLightDeviceHandler(Thing thing) { + super(thing); + } +} diff --git a/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/hardware/FritzAhaContentExchange.java b/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/hardware/FritzAhaContentExchange.java index 1bbcffba5b5..1e4ba7b0da2 100644 --- a/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/hardware/FritzAhaContentExchange.java +++ b/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/hardware/FritzAhaContentExchange.java @@ -56,13 +56,14 @@ public class FritzAhaContentExchange extends BufferingResponseListener @Override public void onFailure(@NonNullByDefault({}) Response response, @NonNullByDefault({}) Throwable failure) { - logger.debug("response failed: {}", failure.getLocalizedMessage(), failure); + logger.debug("{} response failed: {}", response.getRequest().getMethod(), failure.getLocalizedMessage(), + failure); } @Override public void onComplete(@NonNullByDefault({}) Result result) { String content = getContentAsString(); - logger.debug("response complete: {}", content); + logger.debug("{} response complete: {}", result.getRequest().getMethod(), content); callback.execute(result.getResponse().getStatus(), content); } } diff --git a/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/hardware/FritzAhaWebInterface.java b/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/hardware/FritzAhaWebInterface.java index 4edc01fbec9..599bc8a3119 100644 --- a/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/hardware/FritzAhaWebInterface.java +++ b/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/hardware/FritzAhaWebInterface.java @@ -12,8 +12,6 @@ */ package org.openhab.binding.avmfritz.internal.hardware; -import static org.eclipse.jetty.http.HttpMethod.*; - import java.math.BigDecimal; import java.nio.charset.StandardCharsets; import java.security.MessageDigest; @@ -29,15 +27,17 @@ import org.eclipse.jdt.annotation.Nullable; import org.eclipse.jetty.client.HttpClient; import org.eclipse.jetty.client.api.ContentResponse; import org.eclipse.jetty.client.util.StringContentProvider; +import org.eclipse.jetty.http.HttpMethod; import org.openhab.binding.avmfritz.internal.config.AVMFritzBoxConfiguration; import org.openhab.binding.avmfritz.internal.handler.AVMFritzBaseBridgeHandler; import org.openhab.binding.avmfritz.internal.hardware.callbacks.FritzAhaApplyTemplateCallback; import org.openhab.binding.avmfritz.internal.hardware.callbacks.FritzAhaCallback; -import org.openhab.binding.avmfritz.internal.hardware.callbacks.FritzAhaSetBlindLevelCallback; import org.openhab.binding.avmfritz.internal.hardware.callbacks.FritzAhaSetBlindTargetCallback; import org.openhab.binding.avmfritz.internal.hardware.callbacks.FritzAhaSetBlindTargetCallback.BlindCommand; +import org.openhab.binding.avmfritz.internal.hardware.callbacks.FritzAhaSetColorCallback; import org.openhab.binding.avmfritz.internal.hardware.callbacks.FritzAhaSetHeatingModeCallback; import org.openhab.binding.avmfritz.internal.hardware.callbacks.FritzAhaSetHeatingTemperatureCallback; +import org.openhab.binding.avmfritz.internal.hardware.callbacks.FritzAhaSetLevelPercentageCallback; import org.openhab.binding.avmfritz.internal.hardware.callbacks.FritzAhaSetSwitchCallback; import org.openhab.core.thing.ThingStatus; import org.openhab.core.thing.ThingStatusDetail; @@ -239,18 +239,18 @@ public class FritzAhaWebInterface { * @param path Path of the requested resource * @return response */ - public @Nullable String syncGet(String url) { + public @Nullable String syncGet(String path) { try { - ContentResponse contentResponse = httpClient.newRequest(url) - .timeout(config.syncTimeout, TimeUnit.MILLISECONDS).method(GET).send(); + ContentResponse contentResponse = httpClient.newRequest(path) + .timeout(config.syncTimeout, TimeUnit.MILLISECONDS).method(HttpMethod.GET).send(); String content = contentResponse.getContentAsString(); logger.debug("GET response complete: {}", content); return content; } catch (ExecutionException | TimeoutException e) { - logger.debug("response failed: {}", e.getLocalizedMessage(), e); + logger.debug("GET response failed: {}", e.getLocalizedMessage(), e); return null; } catch (InterruptedException e) { - logger.debug("response interrupted: {}", e.getLocalizedMessage(), e); + logger.debug("GET response interrupted: {}", e.getLocalizedMessage(), e); Thread.currentThread().interrupt(); return null; } @@ -268,7 +268,7 @@ public class FritzAhaWebInterface { authenticate(); } FritzAhaContentExchange getExchange = new FritzAhaContentExchange(callback); - httpClient.newRequest(getURL(path, addSID(args))).method(GET).onResponseSuccess(getExchange) + httpClient.newRequest(getURL(path, addSID(args))).method(HttpMethod.GET).onResponseSuccess(getExchange) .onResponseFailure(getExchange).send(getExchange); return getExchange; } @@ -289,7 +289,7 @@ public class FritzAhaWebInterface { authenticate(); } FritzAhaContentExchange postExchange = new FritzAhaContentExchange(callback); - httpClient.newRequest(getURL(path)).timeout(config.asyncTimeout, TimeUnit.MILLISECONDS).method(POST) + httpClient.newRequest(getURL(path)).timeout(config.asyncTimeout, TimeUnit.MILLISECONDS).method(HttpMethod.POST) .onResponseSuccess(postExchange).onResponseFailure(postExchange) .content(new StringContentProvider(addSID(args), StandardCharsets.UTF_8)).send(postExchange); return postExchange; @@ -324,8 +324,14 @@ public class FritzAhaWebInterface { return asyncGet(callback); } - public FritzAhaContentExchange setLevelpercentage(String ain, BigDecimal levelpercentage) { - FritzAhaSetBlindLevelCallback callback = new FritzAhaSetBlindLevelCallback(this, ain, levelpercentage); + public FritzAhaContentExchange setLevelPercentage(String ain, BigDecimal levelPercentage) { + FritzAhaSetLevelPercentageCallback callback = new FritzAhaSetLevelPercentageCallback(this, ain, + levelPercentage); + return asyncGet(callback); + } + + public FritzAhaContentExchange setHueAndSaturation(String ain, int hue, int saturation, int duration) { + FritzAhaSetColorCallback callback = new FritzAhaSetColorCallback(this, ain, hue, saturation, duration); return asyncGet(callback); } diff --git a/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/hardware/callbacks/FritzAhaSetColorCallback.java b/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/hardware/callbacks/FritzAhaSetColorCallback.java new file mode 100644 index 00000000000..94dcec63933 --- /dev/null +++ b/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/hardware/callbacks/FritzAhaSetColorCallback.java @@ -0,0 +1,57 @@ +/** + * Copyright (c) 2010-2021 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.avmfritz.internal.hardware.callbacks; + +import static org.eclipse.jetty.http.HttpMethod.GET; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.openhab.binding.avmfritz.internal.hardware.FritzAhaWebInterface; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * Callback implementation for updating hue, saturation and duration. Supports reauthorization + * + * @author Christoph Weitkamp - Initial contribution + */ +@NonNullByDefault +public class FritzAhaSetColorCallback extends FritzAhaReauthCallback { + + private final Logger logger = LoggerFactory.getLogger(FritzAhaSetColorCallback.class); + + private final String ain; + + /** + * Constructor + * + * @param webIface Interface to FRITZ!Box + * @param ain AIN of the device that should be switched + * @param hue New hue + * @param saturation New saturation + * @param duration Duration of the change in 100ms. 0 immediately. + */ + public FritzAhaSetColorCallback(FritzAhaWebInterface webIface, String ain, int hue, int saturation, int duration) { + super(WEBSERVICE_PATH, + "switchcmd=setcolor&ain=" + ain + "&hue=" + hue + "&saturation=" + saturation + "&duration=" + duration, + webIface, GET, 1); + this.ain = ain; + } + + @Override + public void execute(int status, String response) { + super.execute(status, response); + if (isValidRequest()) { + logger.debug("Received response '{}' for item '{}'", response, ain); + } + } +} diff --git a/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/hardware/callbacks/FritzAhaSetBlindLevelCallback.java b/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/hardware/callbacks/FritzAhaSetLevelPercentageCallback.java similarity index 78% rename from bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/hardware/callbacks/FritzAhaSetBlindLevelCallback.java rename to bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/hardware/callbacks/FritzAhaSetLevelPercentageCallback.java index 430c0f26f42..2abc1c5cd79 100644 --- a/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/hardware/callbacks/FritzAhaSetBlindLevelCallback.java +++ b/bundles/org.openhab.binding.avmfritz/src/main/java/org/openhab/binding/avmfritz/internal/hardware/callbacks/FritzAhaSetLevelPercentageCallback.java @@ -22,14 +22,14 @@ import org.slf4j.Logger; import org.slf4j.LoggerFactory; /** - * Callback implementation for updating blind commands. Supports reauthorization + * Callback implementation for updating blind / levelpercentage commands. Supports reauthorization * * @author Ulrich Mertin - Initial contribution */ @NonNullByDefault -public class FritzAhaSetBlindLevelCallback extends FritzAhaReauthCallback { +public class FritzAhaSetLevelPercentageCallback extends FritzAhaReauthCallback { - private final Logger logger = LoggerFactory.getLogger(FritzAhaSetBlindLevelCallback.class); + private final Logger logger = LoggerFactory.getLogger(FritzAhaSetLevelPercentageCallback.class); private final String ain; @@ -40,8 +40,8 @@ public class FritzAhaSetBlindLevelCallback extends FritzAhaReauthCallback { * @param ain AIN of the device that should be switched * @param level Opening level percentage (0 ... 100) */ - public FritzAhaSetBlindLevelCallback(FritzAhaWebInterface webIface, String ain, BigDecimal levelpercentage) { - super(WEBSERVICE_PATH, "switchcmd=setlevelpercentage&level=" + levelpercentage + "&ain=" + ain, webIface, GET, + public FritzAhaSetLevelPercentageCallback(FritzAhaWebInterface webIface, String ain, BigDecimal levelPercentage) { + super(WEBSERVICE_PATH, "switchcmd=setlevelpercentage&level=" + levelPercentage + "&ain=" + ain, webIface, GET, 1); this.ain = ain; } diff --git a/bundles/org.openhab.binding.avmfritz/src/main/resources/OH-INF/thing/thing-types.xml b/bundles/org.openhab.binding.avmfritz/src/main/resources/OH-INF/thing/thing-types.xml index c4f11d5ab8f..291ca23f160 100644 --- a/bundles/org.openhab.binding.avmfritz/src/main/resources/OH-INF/thing/thing-types.xml +++ b/bundles/org.openhab.binding.avmfritz/src/main/resources/OH-INF/thing/thing-types.xml @@ -5,6 +5,25 @@ xsi:schemaLocation="https://openhab.org/schemas/thing-description/v1.0.0 https://openhab.org/schemas/thing-description-1.0.0.xsd"> + + + + + + + FRITZ!DECT500 color light. + + + + + + + + ain + + + + diff --git a/bundles/org.openhab.binding.avmfritz/src/test/java/org/openhab/binding/avmfritz/internal/dto/AVMFritzDeviceListModelTest.java b/bundles/org.openhab.binding.avmfritz/src/test/java/org/openhab/binding/avmfritz/internal/dto/AVMFritzDeviceListModelTest.java index 32fd63ab7d6..aeb65d7b3ef 100644 --- a/bundles/org.openhab.binding.avmfritz/src/test/java/org/openhab/binding/avmfritz/internal/dto/AVMFritzDeviceListModelTest.java +++ b/bundles/org.openhab.binding.avmfritz/src/test/java/org/openhab/binding/avmfritz/internal/dto/AVMFritzDeviceListModelTest.java @@ -137,7 +137,7 @@ public class AVMFritzDeviceListModelTest { assertNull(device.getHkr()); - assertNull(device.getLevelcontrol()); + assertNull(device.getLevelControlModel()); } @Test @@ -184,7 +184,7 @@ public class AVMFritzDeviceListModelTest { assertNull(device.getHkr()); - assertNull(device.getLevelcontrol()); + assertNull(device.getLevelControlModel()); } @Test @@ -231,7 +231,7 @@ public class AVMFritzDeviceListModelTest { assertNull(device.getHkr()); - assertNull(device.getLevelcontrol()); + assertNull(device.getLevelControlModel()); } @Test @@ -394,7 +394,7 @@ public class AVMFritzDeviceListModelTest { assertNull(device.getHkr()); - assertNull(device.getLevelcontrol()); + assertNull(device.getLevelControlModel()); } @Test @@ -464,7 +464,7 @@ public class AVMFritzDeviceListModelTest { assertNull(device.getHkr()); - assertNull(device.getLevelcontrol()); + assertNull(device.getLevelControlModel()); } @Test @@ -506,7 +506,7 @@ public class AVMFritzDeviceListModelTest { assertNull(device.getHkr()); - assertNull(device.getLevelcontrol()); + assertNull(device.getLevelControlModel()); } @Test @@ -549,7 +549,7 @@ public class AVMFritzDeviceListModelTest { assertNull(device.getHkr()); - assertNull(device.getLevelcontrol()); + assertNull(device.getLevelControlModel()); } @Test @@ -592,7 +592,7 @@ public class AVMFritzDeviceListModelTest { assertNull(device.getHkr()); - assertNull(device.getLevelcontrol()); + assertNull(device.getLevelControlModel()); } @Test @@ -635,7 +635,7 @@ public class AVMFritzDeviceListModelTest { assertNull(device.getHkr()); - LevelcontrolModel levelcontrol = device.getLevelcontrol(); + LevelControlModel levelcontrol = device.getLevelControlModel(); assertNotNull(levelcontrol); assertEquals(BigDecimal.valueOf(26L), levelcontrol.getLevel()); assertEquals(BigDecimal.valueOf(10L), levelcontrol.getLevelPercentage()); @@ -687,7 +687,7 @@ public class AVMFritzDeviceListModelTest { assertNull(device.getHkr()); - assertNull(device.getLevelcontrol()); + assertNull(device.getLevelControlModel()); } @Test diff --git a/itests/org.openhab.binding.avmfritz.tests/src/main/java/org/openhab/binding/avmfritz/internal/discovery/AVMFritzDiscoveryServiceOSGiTest.java b/itests/org.openhab.binding.avmfritz.tests/src/main/java/org/openhab/binding/avmfritz/internal/discovery/AVMFritzDiscoveryServiceOSGiTest.java index 3265dd8be5e..26d7dd3eb6c 100644 --- a/itests/org.openhab.binding.avmfritz.tests/src/main/java/org/openhab/binding/avmfritz/internal/discovery/AVMFritzDiscoveryServiceOSGiTest.java +++ b/itests/org.openhab.binding.avmfritz.tests/src/main/java/org/openhab/binding/avmfritz/internal/discovery/AVMFritzDiscoveryServiceOSGiTest.java @@ -89,7 +89,7 @@ public class AVMFritzDiscoveryServiceOSGiTest extends AVMFritzThingHandlerOSGiTe @Test public void correctSupportedTypes() { - assertEquals(15, discovery.getSupportedThingTypes().size()); + assertEquals(16, discovery.getSupportedThingTypes().size()); assertTrue(discovery.getSupportedThingTypes().contains(DECT100_THING_TYPE)); assertTrue(discovery.getSupportedThingTypes().contains(DECT200_THING_TYPE)); assertTrue(discovery.getSupportedThingTypes().contains(DECT210_THING_TYPE)); @@ -97,6 +97,7 @@ public class AVMFritzDiscoveryServiceOSGiTest extends AVMFritzThingHandlerOSGiTe assertTrue(discovery.getSupportedThingTypes().contains(DECT301_THING_TYPE)); assertTrue(discovery.getSupportedThingTypes().contains(DECT400_THING_TYPE)); assertTrue(discovery.getSupportedThingTypes().contains(DECT440_THING_TYPE)); + assertTrue(discovery.getSupportedThingTypes().contains(DECT500_THING_TYPE)); assertTrue(discovery.getSupportedThingTypes().contains(PL546E_THING_TYPE)); assertTrue(discovery.getSupportedThingTypes().contains(COMETDECT_THING_TYPE)); assertTrue(discovery.getSupportedThingTypes().contains(HAN_FUN_CONTACT_THING_TYPE));