2 * Copyright (c) 2010-2024 Contributors to the openHAB project
4 * See the NOTICE file(s) distributed with this work for additional
7 * This program and the accompanying materials are made available under the
8 * terms of the Eclipse Public License 2.0 which is available at
9 * http://www.eclipse.org/legal/epl-2.0
11 * SPDX-License-Identifier: EPL-2.0
13 package org.openhab.io.homekit.internal.accessories;
15 import static org.openhab.io.homekit.internal.HomekitCharacteristicType.*;
17 import java.math.BigDecimal;
18 import java.math.RoundingMode;
19 import java.util.ArrayList;
20 import java.util.EnumMap;
21 import java.util.HashMap;
22 import java.util.List;
24 import java.util.Objects;
25 import java.util.concurrent.CompletableFuture;
26 import java.util.function.BiFunction;
27 import java.util.function.Consumer;
28 import java.util.function.Supplier;
30 import javax.measure.Quantity;
31 import javax.measure.Unit;
32 import javax.measure.quantity.Temperature;
34 import org.eclipse.jdt.annotation.NonNullByDefault;
35 import org.eclipse.jdt.annotation.Nullable;
36 import org.openhab.core.items.GenericItem;
37 import org.openhab.core.items.Item;
38 import org.openhab.core.library.items.ColorItem;
39 import org.openhab.core.library.items.DimmerItem;
40 import org.openhab.core.library.items.NumberItem;
41 import org.openhab.core.library.items.RollershutterItem;
42 import org.openhab.core.library.items.StringItem;
43 import org.openhab.core.library.items.SwitchItem;
44 import org.openhab.core.library.types.DecimalType;
45 import org.openhab.core.library.types.HSBType;
46 import org.openhab.core.library.types.IncreaseDecreaseType;
47 import org.openhab.core.library.types.OnOffType;
48 import org.openhab.core.library.types.OpenClosedType;
49 import org.openhab.core.library.types.PercentType;
50 import org.openhab.core.library.types.QuantityType;
51 import org.openhab.core.library.types.StopMoveType;
52 import org.openhab.core.library.types.StringType;
53 import org.openhab.core.library.unit.ImperialUnits;
54 import org.openhab.core.library.unit.SIUnits;
55 import org.openhab.core.library.unit.Units;
56 import org.openhab.core.types.State;
57 import org.openhab.core.types.UnDefType;
58 import org.openhab.io.homekit.Homekit;
59 import org.openhab.io.homekit.internal.HomekitAccessoryUpdater;
60 import org.openhab.io.homekit.internal.HomekitCharacteristicType;
61 import org.openhab.io.homekit.internal.HomekitCommandType;
62 import org.openhab.io.homekit.internal.HomekitException;
63 import org.openhab.io.homekit.internal.HomekitImpl;
64 import org.openhab.io.homekit.internal.HomekitTaggedItem;
65 import org.osgi.framework.FrameworkUtil;
66 import org.slf4j.Logger;
67 import org.slf4j.LoggerFactory;
69 import io.github.hapjava.characteristics.Characteristic;
70 import io.github.hapjava.characteristics.CharacteristicEnum;
71 import io.github.hapjava.characteristics.ExceptionalConsumer;
72 import io.github.hapjava.characteristics.HomekitCharacteristicChangeCallback;
73 import io.github.hapjava.characteristics.impl.accessoryinformation.FirmwareRevisionCharacteristic;
74 import io.github.hapjava.characteristics.impl.accessoryinformation.HardwareRevisionCharacteristic;
75 import io.github.hapjava.characteristics.impl.accessoryinformation.IdentifyCharacteristic;
76 import io.github.hapjava.characteristics.impl.accessoryinformation.ManufacturerCharacteristic;
77 import io.github.hapjava.characteristics.impl.accessoryinformation.ModelCharacteristic;
78 import io.github.hapjava.characteristics.impl.accessoryinformation.SerialNumberCharacteristic;
79 import io.github.hapjava.characteristics.impl.airquality.NitrogenDioxideDensityCharacteristic;
80 import io.github.hapjava.characteristics.impl.airquality.OzoneDensityCharacteristic;
81 import io.github.hapjava.characteristics.impl.airquality.PM10DensityCharacteristic;
82 import io.github.hapjava.characteristics.impl.airquality.PM25DensityCharacteristic;
83 import io.github.hapjava.characteristics.impl.airquality.SulphurDioxideDensityCharacteristic;
84 import io.github.hapjava.characteristics.impl.airquality.VOCDensityCharacteristic;
85 import io.github.hapjava.characteristics.impl.audio.MuteCharacteristic;
86 import io.github.hapjava.characteristics.impl.audio.VolumeCharacteristic;
87 import io.github.hapjava.characteristics.impl.battery.StatusLowBatteryCharacteristic;
88 import io.github.hapjava.characteristics.impl.battery.StatusLowBatteryEnum;
89 import io.github.hapjava.characteristics.impl.carbondioxidesensor.CarbonDioxideLevelCharacteristic;
90 import io.github.hapjava.characteristics.impl.carbondioxidesensor.CarbonDioxidePeakLevelCharacteristic;
91 import io.github.hapjava.characteristics.impl.carbonmonoxidesensor.CarbonMonoxideLevelCharacteristic;
92 import io.github.hapjava.characteristics.impl.carbonmonoxidesensor.CarbonMonoxidePeakLevelCharacteristic;
93 import io.github.hapjava.characteristics.impl.common.ActiveCharacteristic;
94 import io.github.hapjava.characteristics.impl.common.ActiveEnum;
95 import io.github.hapjava.characteristics.impl.common.ActiveIdentifierCharacteristic;
96 import io.github.hapjava.characteristics.impl.common.ConfiguredNameCharacteristic;
97 import io.github.hapjava.characteristics.impl.common.IdentifierCharacteristic;
98 import io.github.hapjava.characteristics.impl.common.IsConfiguredCharacteristic;
99 import io.github.hapjava.characteristics.impl.common.IsConfiguredEnum;
100 import io.github.hapjava.characteristics.impl.common.NameCharacteristic;
101 import io.github.hapjava.characteristics.impl.common.ObstructionDetectedCharacteristic;
102 import io.github.hapjava.characteristics.impl.common.ProgrammableSwitchEnum;
103 import io.github.hapjava.characteristics.impl.common.ProgrammableSwitchEventCharacteristic;
104 import io.github.hapjava.characteristics.impl.common.StatusActiveCharacteristic;
105 import io.github.hapjava.characteristics.impl.common.StatusFaultCharacteristic;
106 import io.github.hapjava.characteristics.impl.common.StatusFaultEnum;
107 import io.github.hapjava.characteristics.impl.common.StatusTamperedCharacteristic;
108 import io.github.hapjava.characteristics.impl.common.StatusTamperedEnum;
109 import io.github.hapjava.characteristics.impl.fan.CurrentFanStateCharacteristic;
110 import io.github.hapjava.characteristics.impl.fan.CurrentFanStateEnum;
111 import io.github.hapjava.characteristics.impl.fan.LockPhysicalControlsCharacteristic;
112 import io.github.hapjava.characteristics.impl.fan.LockPhysicalControlsEnum;
113 import io.github.hapjava.characteristics.impl.fan.RotationDirectionCharacteristic;
114 import io.github.hapjava.characteristics.impl.fan.RotationDirectionEnum;
115 import io.github.hapjava.characteristics.impl.fan.RotationSpeedCharacteristic;
116 import io.github.hapjava.characteristics.impl.fan.SwingModeCharacteristic;
117 import io.github.hapjava.characteristics.impl.fan.SwingModeEnum;
118 import io.github.hapjava.characteristics.impl.fan.TargetFanStateCharacteristic;
119 import io.github.hapjava.characteristics.impl.fan.TargetFanStateEnum;
120 import io.github.hapjava.characteristics.impl.filtermaintenance.FilterLifeLevelCharacteristic;
121 import io.github.hapjava.characteristics.impl.filtermaintenance.ResetFilterIndicationCharacteristic;
122 import io.github.hapjava.characteristics.impl.humiditysensor.CurrentRelativeHumidityCharacteristic;
123 import io.github.hapjava.characteristics.impl.humiditysensor.TargetRelativeHumidityCharacteristic;
124 import io.github.hapjava.characteristics.impl.inputsource.CurrentVisibilityStateCharacteristic;
125 import io.github.hapjava.characteristics.impl.inputsource.CurrentVisibilityStateEnum;
126 import io.github.hapjava.characteristics.impl.inputsource.InputDeviceTypeCharacteristic;
127 import io.github.hapjava.characteristics.impl.inputsource.InputDeviceTypeEnum;
128 import io.github.hapjava.characteristics.impl.inputsource.InputSourceTypeCharacteristic;
129 import io.github.hapjava.characteristics.impl.inputsource.InputSourceTypeEnum;
130 import io.github.hapjava.characteristics.impl.inputsource.TargetVisibilityStateCharacteristic;
131 import io.github.hapjava.characteristics.impl.inputsource.TargetVisibilityStateEnum;
132 import io.github.hapjava.characteristics.impl.lightbulb.BrightnessCharacteristic;
133 import io.github.hapjava.characteristics.impl.lightbulb.ColorTemperatureCharacteristic;
134 import io.github.hapjava.characteristics.impl.lightbulb.HueCharacteristic;
135 import io.github.hapjava.characteristics.impl.lightbulb.SaturationCharacteristic;
136 import io.github.hapjava.characteristics.impl.slat.CurrentTiltAngleCharacteristic;
137 import io.github.hapjava.characteristics.impl.slat.TargetTiltAngleCharacteristic;
138 import io.github.hapjava.characteristics.impl.television.ClosedCaptionsCharacteristic;
139 import io.github.hapjava.characteristics.impl.television.ClosedCaptionsEnum;
140 import io.github.hapjava.characteristics.impl.television.CurrentMediaStateCharacteristic;
141 import io.github.hapjava.characteristics.impl.television.CurrentMediaStateEnum;
142 import io.github.hapjava.characteristics.impl.television.PictureModeCharacteristic;
143 import io.github.hapjava.characteristics.impl.television.PictureModeEnum;
144 import io.github.hapjava.characteristics.impl.television.PowerModeCharacteristic;
145 import io.github.hapjava.characteristics.impl.television.PowerModeEnum;
146 import io.github.hapjava.characteristics.impl.television.RemoteKeyCharacteristic;
147 import io.github.hapjava.characteristics.impl.television.RemoteKeyEnum;
148 import io.github.hapjava.characteristics.impl.television.SleepDiscoveryModeCharacteristic;
149 import io.github.hapjava.characteristics.impl.television.SleepDiscoveryModeEnum;
150 import io.github.hapjava.characteristics.impl.television.TargetMediaStateCharacteristic;
151 import io.github.hapjava.characteristics.impl.television.TargetMediaStateEnum;
152 import io.github.hapjava.characteristics.impl.televisionspeaker.VolumeControlTypeCharacteristic;
153 import io.github.hapjava.characteristics.impl.televisionspeaker.VolumeControlTypeEnum;
154 import io.github.hapjava.characteristics.impl.televisionspeaker.VolumeSelectorCharacteristic;
155 import io.github.hapjava.characteristics.impl.televisionspeaker.VolumeSelectorEnum;
156 import io.github.hapjava.characteristics.impl.thermostat.CoolingThresholdTemperatureCharacteristic;
157 import io.github.hapjava.characteristics.impl.thermostat.CurrentHeatingCoolingStateCharacteristic;
158 import io.github.hapjava.characteristics.impl.thermostat.CurrentHeatingCoolingStateEnum;
159 import io.github.hapjava.characteristics.impl.thermostat.CurrentTemperatureCharacteristic;
160 import io.github.hapjava.characteristics.impl.thermostat.HeatingThresholdTemperatureCharacteristic;
161 import io.github.hapjava.characteristics.impl.thermostat.TargetHeatingCoolingStateCharacteristic;
162 import io.github.hapjava.characteristics.impl.thermostat.TargetHeatingCoolingStateEnum;
163 import io.github.hapjava.characteristics.impl.thermostat.TargetTemperatureCharacteristic;
164 import io.github.hapjava.characteristics.impl.thermostat.TemperatureDisplayUnitCharacteristic;
165 import io.github.hapjava.characteristics.impl.thermostat.TemperatureDisplayUnitEnum;
166 import io.github.hapjava.characteristics.impl.valve.RemainingDurationCharacteristic;
167 import io.github.hapjava.characteristics.impl.valve.SetDurationCharacteristic;
168 import io.github.hapjava.characteristics.impl.windowcovering.CurrentHorizontalTiltAngleCharacteristic;
169 import io.github.hapjava.characteristics.impl.windowcovering.CurrentVerticalTiltAngleCharacteristic;
170 import io.github.hapjava.characteristics.impl.windowcovering.HoldPositionCharacteristic;
171 import io.github.hapjava.characteristics.impl.windowcovering.TargetHorizontalTiltAngleCharacteristic;
172 import io.github.hapjava.characteristics.impl.windowcovering.TargetVerticalTiltAngleCharacteristic;
175 * Creates an optional characteristics .
177 * @author Eugen Freiter - Initial contribution
180 public class HomekitCharacteristicFactory {
181 private static final Logger LOGGER = LoggerFactory.getLogger(HomekitCharacteristicFactory.class);
183 // List of optional characteristics and corresponding method to create them.
184 private static final Map<HomekitCharacteristicType, BiFunction<HomekitTaggedItem, HomekitAccessoryUpdater, Characteristic>> OPTIONAL = new HashMap<>() {
186 put(ACTIVE, HomekitCharacteristicFactory::createActiveCharacteristic);
187 put(ACTIVE_IDENTIFIER, HomekitCharacteristicFactory::createActiveIdentifierCharacteristic);
188 put(ACTIVE_STATUS, HomekitCharacteristicFactory::createStatusActiveCharacteristic);
189 put(BATTERY_LOW_STATUS, HomekitCharacteristicFactory::createStatusLowBatteryCharacteristic);
190 put(BRIGHTNESS, HomekitCharacteristicFactory::createBrightnessCharacteristic);
191 put(CARBON_DIOXIDE_LEVEL, HomekitCharacteristicFactory::createCarbonDioxideLevelCharacteristic);
192 put(CARBON_DIOXIDE_PEAK_LEVEL, HomekitCharacteristicFactory::createCarbonDioxidePeakLevelCharacteristic);
193 put(CARBON_MONOXIDE_LEVEL, HomekitCharacteristicFactory::createCarbonMonoxideLevelCharacteristic);
194 put(CARBON_MONOXIDE_PEAK_LEVEL, HomekitCharacteristicFactory::createCarbonMonoxidePeakLevelCharacteristic);
195 put(CLOSED_CAPTIONS, HomekitCharacteristicFactory::createClosedCaptionsCharacteristic);
196 put(COLOR_TEMPERATURE, HomekitCharacteristicFactory::createColorTemperatureCharacteristic);
197 put(CONFIGURED, HomekitCharacteristicFactory::createIsConfiguredCharacteristic);
198 put(CONFIGURED_NAME, HomekitCharacteristicFactory::createConfiguredNameCharacteristic);
199 put(COOLING_THRESHOLD_TEMPERATURE, HomekitCharacteristicFactory::createCoolingThresholdCharacteristic);
200 put(CURRENT_HEATING_COOLING_STATE,
201 HomekitCharacteristicFactory::createCurrentHeatingCoolingStateCharacteristic);
202 put(CURRENT_FAN_STATE, HomekitCharacteristicFactory::createCurrentFanStateCharacteristic);
203 put(CURRENT_HORIZONTAL_TILT_ANGLE,
204 HomekitCharacteristicFactory::createCurrentHorizontalTiltAngleCharacteristic);
205 put(CURRENT_MEDIA_STATE, HomekitCharacteristicFactory::createCurrentMediaStateCharacteristic);
206 put(CURRENT_TILT_ANGLE, HomekitCharacteristicFactory::createCurrentTiltAngleCharacteristic);
207 put(CURRENT_VERTICAL_TILT_ANGLE,
208 HomekitCharacteristicFactory::createCurrentVerticalTiltAngleCharacteristic);
209 put(CURRENT_VISIBILITY, HomekitCharacteristicFactory::createCurrentVisibilityStateCharacteristic);
210 put(CURRENT_TEMPERATURE, HomekitCharacteristicFactory::createCurrentTemperatureCharacteristic);
211 put(DURATION, HomekitCharacteristicFactory::createDurationCharacteristic);
212 put(FAULT_STATUS, HomekitCharacteristicFactory::createStatusFaultCharacteristic);
213 put(FIRMWARE_REVISION, HomekitCharacteristicFactory::createFirmwareRevisionCharacteristic);
214 put(FILTER_LIFE_LEVEL, HomekitCharacteristicFactory::createFilterLifeLevelCharacteristic);
215 put(FILTER_RESET_INDICATION, HomekitCharacteristicFactory::createFilterResetCharacteristic);
216 put(HARDWARE_REVISION, HomekitCharacteristicFactory::createHardwareRevisionCharacteristic);
217 put(HEATING_THRESHOLD_TEMPERATURE, HomekitCharacteristicFactory::createHeatingThresholdCharacteristic);
218 put(HOLD_POSITION, HomekitCharacteristicFactory::createHoldPositionCharacteristic);
219 put(HUE, HomekitCharacteristicFactory::createHueCharacteristic);
220 put(IDENTIFIER, HomekitCharacteristicFactory::createIdentifierCharacteristic);
221 put(IDENTIFY, HomekitCharacteristicFactory::createIdentifyCharacteristic);
222 put(INPUT_DEVICE_TYPE, HomekitCharacteristicFactory::createInputDeviceTypeCharacteristic);
223 put(INPUT_SOURCE_TYPE, HomekitCharacteristicFactory::createInputSourceTypeCharacteristic);
224 put(LOCK_CONTROL, HomekitCharacteristicFactory::createLockPhysicalControlsCharacteristic);
225 put(MANUFACTURER, HomekitCharacteristicFactory::createManufacturerCharacteristic);
226 put(MODEL, HomekitCharacteristicFactory::createModelCharacteristic);
227 put(MUTE, HomekitCharacteristicFactory::createMuteCharacteristic);
228 put(NAME, HomekitCharacteristicFactory::createNameCharacteristic);
229 put(NITROGEN_DIOXIDE_DENSITY, HomekitCharacteristicFactory::createNitrogenDioxideDensityCharacteristic);
230 put(OBSTRUCTION_STATUS, HomekitCharacteristicFactory::createObstructionDetectedCharacteristic);
231 put(OZONE_DENSITY, HomekitCharacteristicFactory::createOzoneDensityCharacteristic);
232 put(PICTURE_MODE, HomekitCharacteristicFactory::createPictureModeCharacteristic);
233 put(PM10_DENSITY, HomekitCharacteristicFactory::createPM10DensityCharacteristic);
234 put(PM25_DENSITY, HomekitCharacteristicFactory::createPM25DensityCharacteristic);
235 put(POWER_MODE, HomekitCharacteristicFactory::createPowerModeCharacteristic);
236 put(PROGRAMMABLE_SWITCH_EVENT, HomekitCharacteristicFactory::createProgrammableSwitchEventCharacteristic);
237 put(REMAINING_DURATION, HomekitCharacteristicFactory::createRemainingDurationCharacteristic);
238 put(REMOTE_KEY, HomekitCharacteristicFactory::createRemoteKeyCharacteristic);
239 put(RELATIVE_HUMIDITY, HomekitCharacteristicFactory::createRelativeHumidityCharacteristic);
240 put(ROTATION_DIRECTION, HomekitCharacteristicFactory::createRotationDirectionCharacteristic);
241 put(ROTATION_SPEED, HomekitCharacteristicFactory::createRotationSpeedCharacteristic);
242 put(SATURATION, HomekitCharacteristicFactory::createSaturationCharacteristic);
243 put(SERIAL_NUMBER, HomekitCharacteristicFactory::createSerialNumberCharacteristic);
244 put(SLEEP_DISCOVERY_MODE, HomekitCharacteristicFactory::createSleepDiscoveryModeCharacteristic);
245 put(SULPHUR_DIOXIDE_DENSITY, HomekitCharacteristicFactory::createSulphurDioxideDensityCharacteristic);
246 put(SWING_MODE, HomekitCharacteristicFactory::createSwingModeCharacteristic);
247 put(TAMPERED_STATUS, HomekitCharacteristicFactory::createStatusTamperedCharacteristic);
248 put(TARGET_FAN_STATE, HomekitCharacteristicFactory::createTargetFanStateCharacteristic);
249 put(TARGET_HEATING_COOLING_STATE,
250 HomekitCharacteristicFactory::createTargetHeatingCoolingStateCharacteristic);
251 put(TARGET_HORIZONTAL_TILT_ANGLE,
252 HomekitCharacteristicFactory::createTargetHorizontalTiltAngleCharacteristic);
253 put(TARGET_MEDIA_STATE, HomekitCharacteristicFactory::createTargetMediaStateCharacteristic);
254 put(TARGET_RELATIVE_HUMIDITY, HomekitCharacteristicFactory::createTargetRelativeHumidityCharacteristic);
255 put(TARGET_TEMPERATURE, HomekitCharacteristicFactory::createTargetTemperatureCharacteristic);
256 put(TARGET_TILT_ANGLE, HomekitCharacteristicFactory::createTargetTiltAngleCharacteristic);
257 put(TARGET_VERTICAL_TILT_ANGLE, HomekitCharacteristicFactory::createTargetVerticalTiltAngleCharacteristic);
258 put(TARGET_VISIBILITY_STATE, HomekitCharacteristicFactory::createTargetVisibilityStateCharacteristic);
259 put(TEMPERATURE_UNIT, HomekitCharacteristicFactory::createTemperatureDisplayUnitCharacteristic);
260 put(VOC_DENSITY, HomekitCharacteristicFactory::createVOCDensityCharacteristic);
261 put(VOLUME, HomekitCharacteristicFactory::createVolumeCharacteristic);
262 put(VOLUME_CONTROL_TYPE, HomekitCharacteristicFactory::createVolumeControlTypeCharacteristic);
263 put(VOLUME_SELECTOR, HomekitCharacteristicFactory::createVolumeSelectorCharacteristic);
267 public static @Nullable Characteristic createNullableCharacteristic(HomekitTaggedItem item,
268 HomekitAccessoryUpdater updater) {
269 final @Nullable HomekitCharacteristicType type = item.getCharacteristicType();
270 LOGGER.trace("Create characteristic {}", item);
271 if (OPTIONAL.containsKey(type)) {
272 return OPTIONAL.get(type).apply(item, updater);
278 * Create HomeKit characteristic
280 * @param item corresponding OH item
281 * @param updater update to keep OH item and HomeKit characteristic in sync
282 * @return HomeKit characteristic
284 public static Characteristic createCharacteristic(HomekitTaggedItem item, HomekitAccessoryUpdater updater)
285 throws HomekitException {
286 Characteristic characteristic = createNullableCharacteristic(item, updater);
287 if (characteristic != null) {
288 return characteristic;
290 final @Nullable HomekitCharacteristicType type = item.getCharacteristicType();
291 LOGGER.warn("Unsupported optional characteristic from item {}. Accessory type {}, characteristic type {}",
292 item.getName(), item.getAccessoryType(), type.getTag());
293 throw new HomekitException(
294 "Unsupported optional characteristic. Characteristic type \"" + type.getTag() + "\"");
298 * Create an EnumMap for a particular CharacteristicEnum.
300 * By default, the map will simply be from the Enum value to the string version of its value.
301 * If the item is a Number item, though, the values will the be underlying integer code
302 * for the item, as a String.
303 * Then the item's metadata will be inspected, applying any custom mappings.
304 * Finally, if customEnumList is supplied, it will be filled out with those mappings
305 * that are actually referenced in the metadata.
308 * @param klazz The HAP-Java Enum for the characteristic.
309 * @param customEnumList Optional output list of which enums are explicitly mentioned.
310 * @param inverted Default-invert the 0/1 values of the HAP enum when linked to a Switch or Contact item.
311 * This is set by the addon when creating mappings for specific characteristics where the 0 and 1
312 * values for the enum do not map naturally to 0/OFF/CLOSED and 1/ON/OPEN of openHAB items.
313 * Note that this is separate from the inverted item-level metadata configuration, which can be
314 * thought of independently as applying on top of this setting. It essentially "multiplies" out,
315 * but can also be thought of as simply swapping whichever value OFF/CLOSED and ON/OPEN are
316 * associated with, which has already been set.
319 public static <T extends Enum<T> & CharacteristicEnum> Map<T, String> createMapping(HomekitTaggedItem item,
320 Class<T> klazz, @Nullable List<T> customEnumList, boolean inverted) {
321 EnumMap<T, String> map = new EnumMap(klazz);
322 var dataTypes = item.getBaseItem().getAcceptedDataTypes();
323 boolean switchType = dataTypes.contains(OnOffType.class);
324 boolean contactType = dataTypes.contains(OpenClosedType.class);
325 boolean percentType = dataTypes.contains(PercentType.class);
326 boolean numberType = dataTypes.contains(DecimalType.class) || percentType || switchType || contactType;
328 if (item.isInverted()) {
329 inverted = !inverted;
331 String onValue = switchType ? OnOffType.ON.toString() : OpenClosedType.OPEN.toString();
332 String offValue = switchType ? OnOffType.OFF.toString() : OpenClosedType.CLOSED.toString();
334 for (var k : klazz.getEnumConstants()) {
336 int code = k.getCode();
337 if ((switchType || contactType) && code == 0) {
338 map.put(k, inverted ? onValue : offValue);
339 } else if ((switchType || contactType) && code == 1) {
340 map.put(k, inverted ? offValue : onValue);
341 } else if (percentType && code == 0) {
343 } else if (percentType && code == 1) {
346 map.put(k, Integer.toString(code));
349 map.put(k, k.toString());
352 var configuration = item.getConfiguration();
353 if (configuration != null) {
354 map.forEach((k, current_value) -> {
355 final Object newValue = configuration.get(k.toString());
356 if (newValue instanceof String || newValue instanceof Number) {
357 map.put(k, newValue.toString());
358 if (customEnumList != null) {
359 customEnumList.add(k);
364 if (customEnumList != null && customEnumList.isEmpty()) {
365 customEnumList.addAll(map.keySet());
367 LOGGER.debug("Created {} mapping for item {} ({}): {}", klazz.getSimpleName(), item.getName(),
368 item.getBaseItem().getClass().getSimpleName(), map);
372 public static <T extends Enum<T> & CharacteristicEnum> Map<T, String> createMapping(HomekitTaggedItem item,
374 return createMapping(item, klazz, null, false);
377 public static <T extends Enum<T> & CharacteristicEnum> Map<T, String> createMapping(HomekitTaggedItem item,
378 Class<T> klazz, @Nullable List<T> customEnumList) {
379 return createMapping(item, klazz, customEnumList, false);
382 public static <T extends Enum<T> & CharacteristicEnum> Map<T, String> createMapping(HomekitTaggedItem item,
383 Class<T> klazz, boolean inverted) {
384 return createMapping(item, klazz, null, inverted);
388 * Takes item state as value and retrieves the key for that value from mapping.
389 * E.g. used to map StringItem value to HomeKit Enum
392 * @param mapping mapping
393 * @param defaultValue default value if nothing found in mapping
394 * @param <T> type of the result derived from
395 * @return key for the value
397 public static <T> T getKeyFromMapping(HomekitTaggedItem item, State state, Map<T, String> mapping, T defaultValue) {
398 LOGGER.trace("getKeyFromMapping: characteristic {}, state {}, mapping {}", item.getAccessoryType().getTag(),
402 if (state instanceof UnDefType) {
404 } else if (state instanceof StringType || state instanceof OnOffType || state instanceof OpenClosedType) {
405 value = state.toString();
406 } else if (state.getClass().equals(PercentType.class)) {
407 // We specifically want PercentType, but _not_ HSBType, so don't use instanceof
408 value = state.as(OnOffType.class).toString();
409 } else if (state.getClass().equals(DecimalType.class)) {
410 // We specifically want DecimalType, but _not_ PercentType or HSBType, so don't use instanceof
411 value = Integer.toString(((DecimalType) state).intValue());
414 "Wrong value type {} ({}) for {} characteristic of the item {}. Expected StringItem, NumberItem, or SwitchItem.",
415 state.toString(), state.getClass().getSimpleName(), item.getAccessoryType().getTag(),
420 return mapping.entrySet().stream().filter(entry -> value.equalsIgnoreCase(entry.getValue())).findAny()
421 .map(Map.Entry::getKey).orElseGet(() -> {
423 "Wrong value {} for {} characteristic of the item {}. Expected one of following {}. Returning {}.",
424 state.toString(), item.getAccessoryType().getTag(), item.getName(), mapping.values(),
430 // supporting methods
432 public static boolean useFahrenheit() {
433 return Boolean.TRUE.equals(FrameworkUtil.getBundle(HomekitImpl.class).getBundleContext()
434 .getServiceReference(Homekit.class.getName()).getProperty("useFahrenheitTemperature"));
437 public static TemperatureDisplayUnitCharacteristic createSystemTemperatureDisplayUnitCharacteristic() {
438 return new TemperatureDisplayUnitCharacteristic(() -> CompletableFuture
439 .completedFuture(HomekitCharacteristicFactory.useFahrenheit() ? TemperatureDisplayUnitEnum.FAHRENHEIT
440 : TemperatureDisplayUnitEnum.CELSIUS),
447 public static Unit<Temperature> getSystemTemperatureUnit() {
448 return useFahrenheit() ? ImperialUnits.FAHRENHEIT : SIUnits.CELSIUS;
451 private static <T extends CharacteristicEnum> CompletableFuture<T> getEnumFromItem(HomekitTaggedItem item,
452 Map<T, String> mapping, T defaultValue) {
453 return CompletableFuture
454 .completedFuture(getKeyFromMapping(item, item.getItem().getState(), mapping, defaultValue));
457 public static <T extends Enum<T>> void setValueFromEnum(HomekitTaggedItem taggedItem, T value, Map<T, String> map) {
458 if (taggedItem.getBaseItem() instanceof NumberItem) {
459 taggedItem.send(new DecimalType(Objects.requireNonNull(map.get(value))));
460 } else if (taggedItem.getBaseItem() instanceof SwitchItem) {
461 taggedItem.send(OnOffType.from(Objects.requireNonNull(map.get(value))));
463 taggedItem.send(new StringType(map.get(value)));
467 private static int getIntFromItem(HomekitTaggedItem taggedItem, int defaultValue) {
468 int value = defaultValue;
469 final State state = taggedItem.getItem().getState();
470 if (state instanceof PercentType stateAsPercentType) {
471 value = stateAsPercentType.intValue();
472 } else if (state instanceof DecimalType stateAsDecimalType) {
473 value = stateAsDecimalType.intValue();
474 } else if (state instanceof UnDefType) {
475 LOGGER.debug("Item state {} is UNDEF {}. Returning default value {}", state, taggedItem.getName(),
479 "Item state {} is not supported for {}. Only PercentType and DecimalType (0/100) are supported.",
480 state, taggedItem.getName());
485 /** special method for tilts. it converts percentage to angle */
486 private static int getAngleFromItem(HomekitTaggedItem taggedItem, int defaultValue) {
487 int value = defaultValue;
488 final State state = taggedItem.getItem().getState();
489 if (state instanceof PercentType stateAsPercentType) {
490 value = (int) ((stateAsPercentType.intValue() * 90.0) / 50.0 - 90.0);
492 value = getIntFromItem(taggedItem, defaultValue);
497 private static <T extends Quantity<T>> double convertAndRound(double value, Unit<T> from, Unit<T> to) {
498 double rawValue = from.equals(to) ? value : from.getConverterTo(to).convert(value);
499 return new BigDecimal(rawValue).setScale(1, RoundingMode.HALF_UP).doubleValue();
502 public static @Nullable Double stateAsTemperature(@Nullable State state) {
503 if (state == null || state instanceof UnDefType) {
507 if (state instanceof QuantityType<?> qt) {
508 if (qt.getDimension().equals(SIUnits.CELSIUS.getDimension())) {
509 return qt.toUnit(SIUnits.CELSIUS).doubleValue();
513 return convertToCelsius(state.as(DecimalType.class).doubleValue());
516 public static double convertToCelsius(double degrees) {
517 return convertAndRound(degrees, getSystemTemperatureUnit(), SIUnits.CELSIUS);
520 public static double convertFromCelsius(double degrees) {
521 return convertAndRound(degrees, SIUnits.CELSIUS, getSystemTemperatureUnit());
524 public static double getTemperatureStep(HomekitTaggedItem taggedItem, double defaultValue) {
525 return taggedItem.getConfigurationAsQuantity(HomekitTaggedItem.STEP,
526 new QuantityType(defaultValue, SIUnits.CELSIUS), true).doubleValue();
529 private static Supplier<CompletableFuture<Integer>> getAngleSupplier(HomekitTaggedItem taggedItem,
531 return () -> CompletableFuture.completedFuture(getAngleFromItem(taggedItem, defaultValue));
534 private static Supplier<CompletableFuture<Integer>> getIntSupplier(HomekitTaggedItem taggedItem, int defaultValue) {
535 return () -> CompletableFuture.completedFuture(getIntFromItem(taggedItem, defaultValue));
538 private static ExceptionalConsumer<Integer> setIntConsumer(HomekitTaggedItem taggedItem) {
540 if (taggedItem.getBaseItem() instanceof NumberItem) {
541 taggedItem.send(new DecimalType(value));
543 LOGGER.warn("Item type {} is not supported for {}. Only NumberItem is supported.",
544 taggedItem.getBaseItem().getType(), taggedItem.getName());
549 private static ExceptionalConsumer<Integer> setPercentConsumer(HomekitTaggedItem taggedItem) {
551 if (taggedItem.getBaseItem() instanceof NumberItem) {
552 taggedItem.send(new DecimalType(value));
553 } else if (taggedItem.getBaseItem() instanceof DimmerItem) {
554 taggedItem.send(new PercentType(value));
556 LOGGER.warn("Item type {} is not supported for {}. Only DimmerItem and NumberItem are supported.",
557 taggedItem.getBaseItem().getType(), taggedItem.getName());
562 private static ExceptionalConsumer<Integer> setAngleConsumer(HomekitTaggedItem taggedItem) {
564 if (taggedItem.getBaseItem() instanceof NumberItem) {
565 taggedItem.send(new DecimalType(value));
566 } else if (taggedItem.getBaseItem() instanceof DimmerItem) {
567 value = (int) (value * 50.0 / 90.0 + 50.0);
568 taggedItem.send(new PercentType(value));
570 LOGGER.warn("Item type {} is not supported for {}. Only DimmerItem and NumberItem are supported.",
571 taggedItem.getBaseItem().getType(), taggedItem.getName());
576 private static Supplier<CompletableFuture<Double>> getDoubleSupplier(HomekitTaggedItem taggedItem,
577 double defaultValue) {
579 final State state = taggedItem.getItem().getState();
580 double value = defaultValue;
581 if (state instanceof PercentType stateAsPercentType) {
582 value = stateAsPercentType.doubleValue();
583 } else if (state instanceof DecimalType stateAsDecimalType) {
584 value = stateAsDecimalType.doubleValue();
585 } else if (state instanceof QuantityType stateAsQuantityType) {
586 value = stateAsQuantityType.doubleValue();
588 return CompletableFuture.completedFuture(value);
592 private static ExceptionalConsumer<Double> setDoubleConsumer(HomekitTaggedItem taggedItem) {
594 if (taggedItem.getBaseItem() instanceof NumberItem) {
595 taggedItem.send(new DecimalType(value.doubleValue()));
596 } else if (taggedItem.getBaseItem() instanceof DimmerItem) {
597 taggedItem.send(new PercentType(value.intValue()));
599 LOGGER.warn("Item type {} is not supported for {}. Only Number and Dimmer type are supported.",
600 taggedItem.getBaseItem().getType(), taggedItem.getName());
605 private static Supplier<CompletableFuture<Double>> getTemperatureSupplier(HomekitTaggedItem taggedItem,
606 double defaultValue) {
608 final @Nullable Double value = stateAsTemperature(taggedItem.getItem().getState());
609 return CompletableFuture.completedFuture(value != null ? value : defaultValue);
613 private static ExceptionalConsumer<Double> setTemperatureConsumer(HomekitTaggedItem taggedItem) {
615 Item baseItem = taggedItem.getBaseItem();
616 if (baseItem instanceof NumberItem baseAsNumberItem) {
617 if (baseAsNumberItem.getUnit() != null) {
618 taggedItem.send(new QuantityType(value, SIUnits.CELSIUS));
620 taggedItem.send(new DecimalType(convertFromCelsius(value)));
623 LOGGER.warn("Item type {} is not supported for {}. Only Number type is supported.",
624 taggedItem.getBaseItem().getType(), taggedItem.getName());
629 protected static Consumer<HomekitCharacteristicChangeCallback> getSubscriber(HomekitTaggedItem taggedItem,
630 HomekitCharacteristicType key, HomekitAccessoryUpdater updater) {
631 return (callback) -> updater.subscribe((GenericItem) taggedItem.getItem(), key.getTag(), callback);
634 protected static Runnable getUnsubscriber(HomekitTaggedItem taggedItem, HomekitCharacteristicType key,
635 HomekitAccessoryUpdater updater) {
636 return () -> updater.unsubscribe((GenericItem) taggedItem.getItem(), key.getTag());
639 // METHODS TO CREATE SINGLE CHARACTERISTIC FROM OPENHAB ITEM
641 private static ActiveCharacteristic createActiveCharacteristic(HomekitTaggedItem taggedItem,
642 HomekitAccessoryUpdater updater) {
643 var map = createMapping(taggedItem, ActiveEnum.class, false);
644 return new ActiveCharacteristic(() -> getEnumFromItem(taggedItem, map, ActiveEnum.INACTIVE),
645 (value) -> setValueFromEnum(taggedItem, value, map), getSubscriber(taggedItem, ACTIVE, updater),
646 getUnsubscriber(taggedItem, ACTIVE, updater));
649 private static ActiveIdentifierCharacteristic createActiveIdentifierCharacteristic(HomekitTaggedItem taggedItem,
650 HomekitAccessoryUpdater updater) {
651 return new ActiveIdentifierCharacteristic(getIntSupplier(taggedItem, 1), setIntConsumer(taggedItem),
652 getSubscriber(taggedItem, ACTIVE_IDENTIFIER, updater),
653 getUnsubscriber(taggedItem, ACTIVE_IDENTIFIER, updater));
656 private static BrightnessCharacteristic createBrightnessCharacteristic(HomekitTaggedItem taggedItem,
657 HomekitAccessoryUpdater updater) {
658 return new BrightnessCharacteristic(() -> {
660 final State state = taggedItem.getItem().getState();
661 if (state instanceof HSBType stateAsHSBType) {
662 value = stateAsHSBType.getBrightness().intValue();
663 } else if (state instanceof PercentType stateAsPercentType) {
664 value = stateAsPercentType.intValue();
666 return CompletableFuture.completedFuture(value);
668 if (taggedItem.getBaseItem() instanceof DimmerItem) {
669 taggedItem.sendCommandProxy(HomekitCommandType.BRIGHTNESS_COMMAND, new PercentType(brightness));
671 LOGGER.warn("Item type {} is not supported for {}. Only ColorItem and DimmerItem are supported.",
672 taggedItem.getBaseItem().getType(), taggedItem.getName());
674 }, getSubscriber(taggedItem, BRIGHTNESS, updater), getUnsubscriber(taggedItem, BRIGHTNESS, updater));
677 private static CarbonDioxideLevelCharacteristic createCarbonDioxideLevelCharacteristic(HomekitTaggedItem taggedItem,
678 HomekitAccessoryUpdater updater) {
679 return new CarbonDioxideLevelCharacteristic(
680 getDoubleSupplier(taggedItem,
681 taggedItem.getConfigurationAsDouble(HomekitTaggedItem.MIN_VALUE,
682 CarbonDioxideLevelCharacteristic.DEFAULT_MIN_VALUE)),
683 getSubscriber(taggedItem, CARBON_MONOXIDE_LEVEL, updater),
684 getUnsubscriber(taggedItem, CARBON_MONOXIDE_LEVEL, updater));
687 private static CarbonDioxidePeakLevelCharacteristic createCarbonDioxidePeakLevelCharacteristic(
688 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
689 return new CarbonDioxidePeakLevelCharacteristic(
690 getDoubleSupplier(taggedItem,
691 taggedItem.getConfigurationAsDouble(HomekitTaggedItem.MIN_VALUE,
692 CarbonDioxidePeakLevelCharacteristic.DEFAULT_MIN_VALUE)),
693 getSubscriber(taggedItem, CARBON_MONOXIDE_PEAK_LEVEL, updater),
694 getUnsubscriber(taggedItem, CARBON_MONOXIDE_PEAK_LEVEL, updater));
697 private static CarbonMonoxideLevelCharacteristic createCarbonMonoxideLevelCharacteristic(
698 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
699 return new CarbonMonoxideLevelCharacteristic(
700 getDoubleSupplier(taggedItem,
701 taggedItem.getConfigurationAsDouble(HomekitTaggedItem.MIN_VALUE,
702 CarbonMonoxideLevelCharacteristic.DEFAULT_MIN_VALUE)),
703 getSubscriber(taggedItem, CARBON_DIOXIDE_LEVEL, updater),
704 getUnsubscriber(taggedItem, CARBON_DIOXIDE_LEVEL, updater));
707 private static CarbonMonoxidePeakLevelCharacteristic createCarbonMonoxidePeakLevelCharacteristic(
708 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
709 return new CarbonMonoxidePeakLevelCharacteristic(
710 getDoubleSupplier(taggedItem,
711 taggedItem.getConfigurationAsDouble(HomekitTaggedItem.MIN_VALUE,
712 CarbonMonoxidePeakLevelCharacteristic.DEFAULT_MIN_VALUE)),
713 getSubscriber(taggedItem, CARBON_DIOXIDE_PEAK_LEVEL, updater),
714 getUnsubscriber(taggedItem, CARBON_DIOXIDE_PEAK_LEVEL, updater));
717 private static ClosedCaptionsCharacteristic createClosedCaptionsCharacteristic(HomekitTaggedItem taggedItem,
718 HomekitAccessoryUpdater updater) {
719 var map = createMapping(taggedItem, ClosedCaptionsEnum.class);
720 return new ClosedCaptionsCharacteristic(() -> getEnumFromItem(taggedItem, map, ClosedCaptionsEnum.DISABLED),
721 (value) -> setValueFromEnum(taggedItem, value, map),
722 getSubscriber(taggedItem, CLOSED_CAPTIONS, updater),
723 getUnsubscriber(taggedItem, CLOSED_CAPTIONS, updater));
726 private static ColorTemperatureCharacteristic createColorTemperatureCharacteristic(HomekitTaggedItem taggedItem,
727 HomekitAccessoryUpdater updater) {
728 final boolean inverted = taggedItem.isInverted();
730 int minValue = taggedItem
731 .getConfigurationAsQuantity(HomekitTaggedItem.MIN_VALUE,
732 new QuantityType(ColorTemperatureCharacteristic.DEFAULT_MIN_VALUE, Units.MIRED), false)
734 int maxValue = taggedItem
735 .getConfigurationAsQuantity(HomekitTaggedItem.MAX_VALUE,
736 new QuantityType(ColorTemperatureCharacteristic.DEFAULT_MAX_VALUE, Units.MIRED), false)
739 // It's common to swap these if you're providing in Kelvin instead of mired
740 if (minValue > maxValue) {
746 final int finalMinValue = minValue;
747 final int range = maxValue - minValue;
749 return new ColorTemperatureCharacteristic(minValue, maxValue, () -> {
750 int value = finalMinValue;
751 final State state = taggedItem.getItem().getState();
752 if (state instanceof QuantityType<?> qt) {
753 // Number:Temperature
754 qt = qt.toInvertibleUnit(Units.MIRED);
756 LOGGER.warn("Item {}'s state '{}' is not convertible to mireds.", taggedItem.getName(), state);
758 value = qt.intValue();
760 } else if (state instanceof PercentType stateAsPercentType) {
761 double percent = stateAsPercentType.doubleValue();
762 // invert so that 0% == coolest
764 percent = 100.0 - percent;
768 // scale to the originally configured range
769 value = (int) (percent * range / 100) + finalMinValue;
770 } else if (state instanceof DecimalType stateAsDecimalType) {
771 value = stateAsDecimalType.intValue();
773 return CompletableFuture.completedFuture(value);
775 if (taggedItem.getBaseItem() instanceof DimmerItem) {
776 // scale to a percent
777 double percent = (((double) value) - finalMinValue) * 100 / range;
779 percent = 100.0 - percent;
781 taggedItem.send(new PercentType(BigDecimal.valueOf(percent)));
782 } else if (taggedItem.getBaseItem() instanceof NumberItem) {
783 taggedItem.send(new QuantityType(value, Units.MIRED));
785 }, getSubscriber(taggedItem, COLOR_TEMPERATURE, updater),
786 getUnsubscriber(taggedItem, COLOR_TEMPERATURE, updater));
789 private static ConfiguredNameCharacteristic createConfiguredNameCharacteristic(HomekitTaggedItem taggedItem,
790 HomekitAccessoryUpdater updater) {
791 return new ConfiguredNameCharacteristic(() -> {
792 final State state = taggedItem.getItem().getState();
793 return CompletableFuture
794 .completedFuture(state instanceof UnDefType ? taggedItem.getName() : state.toString());
795 }, (value) -> ((StringItem) taggedItem.getItem()).send(new StringType(value)),
796 getSubscriber(taggedItem, CONFIGURED_NAME, updater),
797 getUnsubscriber(taggedItem, CONFIGURED_NAME, updater));
800 private static CoolingThresholdTemperatureCharacteristic createCoolingThresholdCharacteristic(
801 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
802 double minValue = taggedItem.getConfigurationAsQuantity(HomekitTaggedItem.MIN_VALUE,
803 Objects.requireNonNull(
804 new QuantityType(CoolingThresholdTemperatureCharacteristic.DEFAULT_MIN_VALUE, SIUnits.CELSIUS)
805 .toUnit(getSystemTemperatureUnit())),
806 false).toUnit(SIUnits.CELSIUS).doubleValue();
807 double maxValue = taggedItem.getConfigurationAsQuantity(HomekitTaggedItem.MAX_VALUE,
808 Objects.requireNonNull(
809 new QuantityType(CoolingThresholdTemperatureCharacteristic.DEFAULT_MAX_VALUE, SIUnits.CELSIUS)
810 .toUnit(getSystemTemperatureUnit())),
811 false).toUnit(SIUnits.CELSIUS).doubleValue();
812 double step = taggedItem
813 .getConfigurationAsQuantity(HomekitTaggedItem.STEP,
814 Objects.requireNonNull(new QuantityType(CoolingThresholdTemperatureCharacteristic.DEFAULT_STEP,
815 SIUnits.CELSIUS).toUnit(getSystemTemperatureUnit())),
817 .toUnit(SIUnits.CELSIUS).doubleValue();
818 return new CoolingThresholdTemperatureCharacteristic(minValue, maxValue, step,
819 getTemperatureSupplier(taggedItem, minValue), setTemperatureConsumer(taggedItem),
820 getSubscriber(taggedItem, COOLING_THRESHOLD_TEMPERATURE, updater),
821 getUnsubscriber(taggedItem, COOLING_THRESHOLD_TEMPERATURE, updater));
824 private static CurrentHeatingCoolingStateCharacteristic createCurrentHeatingCoolingStateCharacteristic(
825 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
826 List<CurrentHeatingCoolingStateEnum> validValues = new ArrayList<>();
827 var map = createMapping(taggedItem, CurrentHeatingCoolingStateEnum.class, validValues);
828 return new CurrentHeatingCoolingStateCharacteristic(validValues.toArray(new CurrentHeatingCoolingStateEnum[0]),
829 () -> getEnumFromItem(taggedItem, map, CurrentHeatingCoolingStateEnum.OFF),
830 getSubscriber(taggedItem, CURRENT_HEATING_COOLING_STATE, updater),
831 getUnsubscriber(taggedItem, CURRENT_HEATING_COOLING_STATE, updater));
834 private static CurrentFanStateCharacteristic createCurrentFanStateCharacteristic(HomekitTaggedItem taggedItem,
835 HomekitAccessoryUpdater updater) {
836 var map = createMapping(taggedItem, CurrentFanStateEnum.class);
837 return new CurrentFanStateCharacteristic(() -> getEnumFromItem(taggedItem, map, CurrentFanStateEnum.INACTIVE),
838 getSubscriber(taggedItem, CURRENT_FAN_STATE, updater),
839 getUnsubscriber(taggedItem, CURRENT_FAN_STATE, updater));
842 private static CurrentHorizontalTiltAngleCharacteristic createCurrentHorizontalTiltAngleCharacteristic(
843 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
844 return new CurrentHorizontalTiltAngleCharacteristic(getAngleSupplier(taggedItem, 0),
845 getSubscriber(taggedItem, CURRENT_HORIZONTAL_TILT_ANGLE, updater),
846 getUnsubscriber(taggedItem, CURRENT_HORIZONTAL_TILT_ANGLE, updater));
849 private static CurrentMediaStateCharacteristic createCurrentMediaStateCharacteristic(HomekitTaggedItem taggedItem,
850 HomekitAccessoryUpdater updater) {
851 var map = createMapping(taggedItem, CurrentMediaStateEnum.class);
852 return new CurrentMediaStateCharacteristic(
853 () -> getEnumFromItem(taggedItem, map, CurrentMediaStateEnum.UNKNOWN),
854 getSubscriber(taggedItem, CURRENT_MEDIA_STATE, updater),
855 getUnsubscriber(taggedItem, CURRENT_MEDIA_STATE, updater));
858 private static CurrentTemperatureCharacteristic createCurrentTemperatureCharacteristic(HomekitTaggedItem taggedItem,
859 HomekitAccessoryUpdater updater) {
860 double minValue = taggedItem
861 .getConfigurationAsQuantity(HomekitTaggedItem.MIN_VALUE,
862 Objects.requireNonNull(
863 new QuantityType(CurrentTemperatureCharacteristic.DEFAULT_MIN_VALUE, SIUnits.CELSIUS)
864 .toUnit(getSystemTemperatureUnit())),
866 .toUnit(SIUnits.CELSIUS).doubleValue();
867 double maxValue = taggedItem
868 .getConfigurationAsQuantity(HomekitTaggedItem.MAX_VALUE,
869 Objects.requireNonNull(
870 new QuantityType(CurrentTemperatureCharacteristic.DEFAULT_MAX_VALUE, SIUnits.CELSIUS)
871 .toUnit(getSystemTemperatureUnit())),
873 .toUnit(SIUnits.CELSIUS).doubleValue();
874 double step = taggedItem
875 .getConfigurationAsQuantity(HomekitTaggedItem.STEP,
876 Objects.requireNonNull(
877 new QuantityType(CurrentTemperatureCharacteristic.DEFAULT_STEP, SIUnits.CELSIUS)
878 .toUnit(getSystemTemperatureUnit())),
880 .toUnit(SIUnits.CELSIUS).doubleValue();
881 return new CurrentTemperatureCharacteristic(minValue, maxValue, step,
882 getTemperatureSupplier(taggedItem, minValue), getSubscriber(taggedItem, TARGET_TEMPERATURE, updater),
883 getUnsubscriber(taggedItem, TARGET_TEMPERATURE, updater));
886 private static CurrentTiltAngleCharacteristic createCurrentTiltAngleCharacteristic(HomekitTaggedItem taggedItem,
887 HomekitAccessoryUpdater updater) {
888 return new CurrentTiltAngleCharacteristic(getAngleSupplier(taggedItem, 0),
889 getSubscriber(taggedItem, CURRENT_TILT_ANGLE, updater),
890 getUnsubscriber(taggedItem, CURRENT_TILT_ANGLE, updater));
893 private static CurrentVerticalTiltAngleCharacteristic createCurrentVerticalTiltAngleCharacteristic(
894 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
895 return new CurrentVerticalTiltAngleCharacteristic(getAngleSupplier(taggedItem, 0),
896 getSubscriber(taggedItem, CURRENT_VERTICAL_TILT_ANGLE, updater),
897 getUnsubscriber(taggedItem, CURRENT_VERTICAL_TILT_ANGLE, updater));
900 private static CurrentVisibilityStateCharacteristic createCurrentVisibilityStateCharacteristic(
901 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
902 var map = createMapping(taggedItem, CurrentVisibilityStateEnum.class, true);
903 return new CurrentVisibilityStateCharacteristic(
904 () -> getEnumFromItem(taggedItem, map, CurrentVisibilityStateEnum.HIDDEN),
905 getSubscriber(taggedItem, CURRENT_VISIBILITY, updater),
906 getUnsubscriber(taggedItem, CURRENT_VISIBILITY, updater));
909 private static SetDurationCharacteristic createDurationCharacteristic(HomekitTaggedItem taggedItem,
910 HomekitAccessoryUpdater updater) {
911 return new SetDurationCharacteristic(() -> {
912 int value = getIntFromItem(taggedItem, 0);
913 final @Nullable Map<String, Object> itemConfiguration = taggedItem.getConfiguration();
914 if ((value == 0) && (itemConfiguration != null)) { // check for default duration
915 final Object duration = itemConfiguration.get(HomekitValveImpl.CONFIG_DEFAULT_DURATION);
916 if (duration instanceof BigDecimal durationAsBigDecimal) {
917 value = durationAsBigDecimal.intValue();
918 if (taggedItem.getItem() instanceof NumberItem taggedNumberItem) {
919 taggedNumberItem.setState(new DecimalType(value));
923 return CompletableFuture.completedFuture(value);
924 }, setIntConsumer(taggedItem), getSubscriber(taggedItem, DURATION, updater),
925 getUnsubscriber(taggedItem, DURATION, updater));
928 private static FilterLifeLevelCharacteristic createFilterLifeLevelCharacteristic(HomekitTaggedItem taggedItem,
929 HomekitAccessoryUpdater updater) {
930 return new FilterLifeLevelCharacteristic(getDoubleSupplier(taggedItem, 0),
931 getSubscriber(taggedItem, FILTER_LIFE_LEVEL, updater),
932 getUnsubscriber(taggedItem, FILTER_LIFE_LEVEL, updater));
935 private static ResetFilterIndicationCharacteristic createFilterResetCharacteristic(HomekitTaggedItem taggedItem,
936 HomekitAccessoryUpdater updater) {
937 return new ResetFilterIndicationCharacteristic(
938 (value) -> ((SwitchItem) taggedItem.getBaseItem()).send(OnOffType.ON));
941 private static FirmwareRevisionCharacteristic createFirmwareRevisionCharacteristic(HomekitTaggedItem taggedItem,
942 HomekitAccessoryUpdater updater) {
943 return new FirmwareRevisionCharacteristic(() -> {
944 final State state = taggedItem.getItem().getState();
945 return CompletableFuture.completedFuture(state instanceof UnDefType ? "" : state.toString());
949 private static HardwareRevisionCharacteristic createHardwareRevisionCharacteristic(HomekitTaggedItem taggedItem,
950 HomekitAccessoryUpdater updater) {
951 return new HardwareRevisionCharacteristic(() -> {
952 final State state = taggedItem.getItem().getState();
953 return CompletableFuture.completedFuture(state instanceof UnDefType ? "" : state.toString());
957 private static HeatingThresholdTemperatureCharacteristic createHeatingThresholdCharacteristic(
958 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
959 double minValue = taggedItem.getConfigurationAsQuantity(HomekitTaggedItem.MIN_VALUE,
960 Objects.requireNonNull(
961 new QuantityType(HeatingThresholdTemperatureCharacteristic.DEFAULT_MIN_VALUE, SIUnits.CELSIUS)
962 .toUnit(getSystemTemperatureUnit())),
963 false).toUnit(SIUnits.CELSIUS).doubleValue();
964 double maxValue = taggedItem.getConfigurationAsQuantity(HomekitTaggedItem.MAX_VALUE,
965 Objects.requireNonNull(
966 new QuantityType(HeatingThresholdTemperatureCharacteristic.DEFAULT_MAX_VALUE, SIUnits.CELSIUS)
967 .toUnit(getSystemTemperatureUnit())),
968 false).toUnit(SIUnits.CELSIUS).doubleValue();
969 double step = taggedItem
970 .getConfigurationAsQuantity(HomekitTaggedItem.STEP,
971 Objects.requireNonNull(new QuantityType(HeatingThresholdTemperatureCharacteristic.DEFAULT_STEP,
972 SIUnits.CELSIUS).toUnit(getSystemTemperatureUnit())),
974 .toUnit(SIUnits.CELSIUS).doubleValue();
975 return new HeatingThresholdTemperatureCharacteristic(minValue, maxValue, step,
976 getTemperatureSupplier(taggedItem, minValue), setTemperatureConsumer(taggedItem),
977 getSubscriber(taggedItem, HEATING_THRESHOLD_TEMPERATURE, updater),
978 getUnsubscriber(taggedItem, HEATING_THRESHOLD_TEMPERATURE, updater));
981 private static HoldPositionCharacteristic createHoldPositionCharacteristic(HomekitTaggedItem taggedItem,
982 HomekitAccessoryUpdater updater) {
983 final Item item = taggedItem.getBaseItem();
984 if (!(item instanceof SwitchItem || item instanceof RollershutterItem)) {
986 "Item {} cannot be used for the HoldPosition characteristic; only SwitchItem and RollershutterItem are supported. Hold requests will be ignored.",
990 return new HoldPositionCharacteristic(value -> {
995 if (item instanceof SwitchItem switchItem) {
996 switchItem.send(OnOffType.ON);
997 } else if (item instanceof RollershutterItem rollershutterItem) {
998 rollershutterItem.send(StopMoveType.STOP);
1003 private static HueCharacteristic createHueCharacteristic(HomekitTaggedItem taggedItem,
1004 HomekitAccessoryUpdater updater) {
1005 return new HueCharacteristic(() -> {
1007 State state = taggedItem.getItem().getState();
1008 if (state instanceof HSBType stateAsHSBType) {
1009 value = stateAsHSBType.getHue().doubleValue();
1011 return CompletableFuture.completedFuture(value);
1013 if (taggedItem.getBaseItem() instanceof ColorItem) {
1014 taggedItem.sendCommandProxy(HomekitCommandType.HUE_COMMAND, new DecimalType(hue));
1016 LOGGER.warn("Item type {} is not supported for {}. Only Color type is supported.",
1017 taggedItem.getBaseItem().getType(), taggedItem.getName());
1019 }, getSubscriber(taggedItem, HUE, updater), getUnsubscriber(taggedItem, HUE, updater));
1022 private static IdentifierCharacteristic createIdentifierCharacteristic(HomekitTaggedItem taggedItem,
1023 HomekitAccessoryUpdater updater) {
1024 return new IdentifierCharacteristic(getIntSupplier(taggedItem, 1));
1027 private static IdentifyCharacteristic createIdentifyCharacteristic(HomekitTaggedItem taggedItem,
1028 HomekitAccessoryUpdater updater) {
1029 return new IdentifyCharacteristic((value) -> ((SwitchItem) taggedItem.getBaseItem()).send(OnOffType.ON));
1032 private static InputDeviceTypeCharacteristic createInputDeviceTypeCharacteristic(HomekitTaggedItem taggedItem,
1033 HomekitAccessoryUpdater updater) {
1034 var map = createMapping(taggedItem, InputDeviceTypeEnum.class);
1035 return new InputDeviceTypeCharacteristic(() -> getEnumFromItem(taggedItem, map, InputDeviceTypeEnum.OTHER),
1036 getSubscriber(taggedItem, INPUT_DEVICE_TYPE, updater),
1037 getUnsubscriber(taggedItem, INPUT_DEVICE_TYPE, updater));
1040 private static InputSourceTypeCharacteristic createInputSourceTypeCharacteristic(HomekitTaggedItem taggedItem,
1041 HomekitAccessoryUpdater updater) {
1042 var map = createMapping(taggedItem, InputSourceTypeEnum.class);
1043 return new InputSourceTypeCharacteristic(() -> getEnumFromItem(taggedItem, map, InputSourceTypeEnum.OTHER),
1044 getSubscriber(taggedItem, INPUT_SOURCE_TYPE, updater),
1045 getUnsubscriber(taggedItem, INPUT_SOURCE_TYPE, updater));
1048 private static IsConfiguredCharacteristic createIsConfiguredCharacteristic(HomekitTaggedItem taggedItem,
1049 HomekitAccessoryUpdater updater) {
1050 var map = createMapping(taggedItem, IsConfiguredEnum.class);
1051 return new IsConfiguredCharacteristic(() -> getEnumFromItem(taggedItem, map, IsConfiguredEnum.NOT_CONFIGURED),
1052 (value) -> setValueFromEnum(taggedItem, value, map), getSubscriber(taggedItem, CONFIGURED, updater),
1053 getUnsubscriber(taggedItem, CONFIGURED, updater));
1056 private static LockPhysicalControlsCharacteristic createLockPhysicalControlsCharacteristic(
1057 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
1058 var map = createMapping(taggedItem, LockPhysicalControlsEnum.class);
1059 return new LockPhysicalControlsCharacteristic(
1060 () -> getEnumFromItem(taggedItem, map, LockPhysicalControlsEnum.CONTROL_LOCK_DISABLED),
1061 (value) -> setValueFromEnum(taggedItem, value, map), getSubscriber(taggedItem, LOCK_CONTROL, updater),
1062 getUnsubscriber(taggedItem, LOCK_CONTROL, updater));
1065 private static ManufacturerCharacteristic createManufacturerCharacteristic(HomekitTaggedItem taggedItem,
1066 HomekitAccessoryUpdater updater) {
1067 return new ManufacturerCharacteristic(() -> {
1068 final State state = taggedItem.getItem().getState();
1069 return CompletableFuture.completedFuture(state instanceof UnDefType ? "" : state.toString());
1073 private static ModelCharacteristic createModelCharacteristic(HomekitTaggedItem taggedItem,
1074 HomekitAccessoryUpdater updater) {
1075 return new ModelCharacteristic(() -> {
1076 final State state = taggedItem.getItem().getState();
1077 return CompletableFuture.completedFuture(state instanceof UnDefType ? "" : state.toString());
1081 private static MuteCharacteristic createMuteCharacteristic(HomekitTaggedItem taggedItem,
1082 HomekitAccessoryUpdater updater) {
1083 BooleanItemReader muteReader = new BooleanItemReader(taggedItem.getItem(),
1084 OnOffType.from(!taggedItem.isInverted()),
1085 taggedItem.isInverted() ? OpenClosedType.CLOSED : OpenClosedType.OPEN);
1086 return new MuteCharacteristic(() -> CompletableFuture.completedFuture(muteReader.getValue()),
1087 (value) -> taggedItem.send(OnOffType.from(value)), getSubscriber(taggedItem, MUTE, updater),
1088 getUnsubscriber(taggedItem, MUTE, updater));
1091 private static NameCharacteristic createNameCharacteristic(HomekitTaggedItem taggedItem,
1092 HomekitAccessoryUpdater updater) {
1093 return new NameCharacteristic(() -> {
1094 final State state = taggedItem.getItem().getState();
1095 return CompletableFuture.completedFuture(state instanceof UnDefType ? "" : state.toString());
1099 private static NitrogenDioxideDensityCharacteristic createNitrogenDioxideDensityCharacteristic(
1100 final HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
1101 return new NitrogenDioxideDensityCharacteristic(
1102 getDoubleSupplier(taggedItem,
1103 taggedItem.getConfigurationAsDouble(HomekitTaggedItem.MIN_VALUE,
1104 NitrogenDioxideDensityCharacteristic.DEFAULT_MIN_VALUE)),
1105 getSubscriber(taggedItem, NITROGEN_DIOXIDE_DENSITY, updater),
1106 getUnsubscriber(taggedItem, NITROGEN_DIOXIDE_DENSITY, updater));
1109 private static ObstructionDetectedCharacteristic createObstructionDetectedCharacteristic(
1110 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
1111 return new ObstructionDetectedCharacteristic(
1112 () -> CompletableFuture.completedFuture(taggedItem.getItem().getState() == OnOffType.ON
1113 || taggedItem.getItem().getState() == OpenClosedType.OPEN),
1114 getSubscriber(taggedItem, OBSTRUCTION_STATUS, updater),
1115 getUnsubscriber(taggedItem, OBSTRUCTION_STATUS, updater));
1118 private static OzoneDensityCharacteristic createOzoneDensityCharacteristic(final HomekitTaggedItem taggedItem,
1119 HomekitAccessoryUpdater updater) {
1120 return new OzoneDensityCharacteristic(
1121 getDoubleSupplier(taggedItem,
1122 taggedItem.getConfigurationAsDouble(HomekitTaggedItem.MIN_VALUE,
1123 OzoneDensityCharacteristic.DEFAULT_MIN_VALUE)),
1124 getSubscriber(taggedItem, OZONE_DENSITY, updater), getUnsubscriber(taggedItem, OZONE_DENSITY, updater));
1127 private static PM10DensityCharacteristic createPM10DensityCharacteristic(final HomekitTaggedItem taggedItem,
1128 HomekitAccessoryUpdater updater) {
1129 return new PM10DensityCharacteristic(
1130 getDoubleSupplier(taggedItem,
1131 taggedItem.getConfigurationAsDouble(HomekitTaggedItem.MIN_VALUE,
1132 PM10DensityCharacteristic.DEFAULT_MIN_VALUE)),
1133 getSubscriber(taggedItem, PM10_DENSITY, updater), getUnsubscriber(taggedItem, PM10_DENSITY, updater));
1136 private static PM25DensityCharacteristic createPM25DensityCharacteristic(final HomekitTaggedItem taggedItem,
1137 HomekitAccessoryUpdater updater) {
1138 return new PM25DensityCharacteristic(
1139 getDoubleSupplier(taggedItem,
1140 taggedItem.getConfigurationAsDouble(HomekitTaggedItem.MIN_VALUE,
1141 PM25DensityCharacteristic.DEFAULT_MIN_VALUE)),
1142 getSubscriber(taggedItem, PM25_DENSITY, updater), getUnsubscriber(taggedItem, PM25_DENSITY, updater));
1145 private static CurrentRelativeHumidityCharacteristic createRelativeHumidityCharacteristic(
1146 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
1147 return new CurrentRelativeHumidityCharacteristic(getDoubleSupplier(taggedItem, 0.0),
1148 getSubscriber(taggedItem, RELATIVE_HUMIDITY, updater),
1149 getUnsubscriber(taggedItem, RELATIVE_HUMIDITY, updater));
1152 private static PictureModeCharacteristic createPictureModeCharacteristic(HomekitTaggedItem taggedItem,
1153 HomekitAccessoryUpdater updater) {
1154 var map = createMapping(taggedItem, PictureModeEnum.class);
1155 return new PictureModeCharacteristic(() -> getEnumFromItem(taggedItem, map, PictureModeEnum.OTHER),
1156 (value) -> setValueFromEnum(taggedItem, value, map), getSubscriber(taggedItem, PICTURE_MODE, updater),
1157 getUnsubscriber(taggedItem, PICTURE_MODE, updater));
1160 private static PowerModeCharacteristic createPowerModeCharacteristic(HomekitTaggedItem taggedItem,
1161 HomekitAccessoryUpdater updater) {
1162 var map = createMapping(taggedItem, PowerModeEnum.class, true);
1163 return new PowerModeCharacteristic((value) -> setValueFromEnum(taggedItem, value, map));
1166 // this characteristic is unique in a few ways, so we can't use the "normal" helpers:
1167 // * you don't return a "current" value, just the value of the most recent event
1168 // * NULL/invalid values are very much expected, and should silently _not_ trigger an event
1169 // * every update to the item should trigger an event, not just changes
1171 private static ProgrammableSwitchEventCharacteristic createProgrammableSwitchEventCharacteristic(
1172 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
1173 // have to build the map custom, since SINGLE_PRESS starts at 0
1174 Map<ProgrammableSwitchEnum, String> map = new EnumMap(ProgrammableSwitchEnum.class);
1175 List<ProgrammableSwitchEnum> validValues = new ArrayList<>();
1177 if (taggedItem.getBaseItem().getAcceptedDataTypes().contains(OnOffType.class)) {
1178 map.put(ProgrammableSwitchEnum.SINGLE_PRESS, OnOffType.ON.toString());
1179 validValues.add(ProgrammableSwitchEnum.SINGLE_PRESS);
1180 } else if (taggedItem.getBaseItem().getAcceptedDataTypes().contains(OpenClosedType.class)) {
1181 map.put(ProgrammableSwitchEnum.SINGLE_PRESS, OpenClosedType.OPEN.toString());
1182 validValues.add(ProgrammableSwitchEnum.SINGLE_PRESS);
1184 map = createMapping(taggedItem, ProgrammableSwitchEnum.class, validValues, false);
1187 var helper = new ProgrammableSwitchEventCharacteristicHelper(taggedItem, updater, map);
1189 return new ProgrammableSwitchEventCharacteristic(validValues.toArray(new ProgrammableSwitchEnum[0]),
1190 helper::getValue, helper::subscribe, getUnsubscriber(taggedItem, PROGRAMMABLE_SWITCH_EVENT, updater));
1193 private static class ProgrammableSwitchEventCharacteristicHelper {
1194 private @Nullable ProgrammableSwitchEnum lastValue = null;
1195 private final HomekitTaggedItem taggedItem;
1196 private final Map<ProgrammableSwitchEnum, String> map;
1197 private final HomekitAccessoryUpdater updater;
1199 ProgrammableSwitchEventCharacteristicHelper(HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater,
1200 Map<ProgrammableSwitchEnum, String> map) {
1201 this.taggedItem = taggedItem;
1203 this.updater = updater;
1206 public CompletableFuture<ProgrammableSwitchEnum> getValue() {
1207 return CompletableFuture.completedFuture(lastValue);
1210 public void subscribe(HomekitCharacteristicChangeCallback cb) {
1211 updater.subscribeToUpdates((GenericItem) taggedItem.getItem(), PROGRAMMABLE_SWITCH_EVENT.getTag(),
1213 // perform inversion here, so logic below only needs to deal with the
1215 if (state instanceof OnOffType && taggedItem.isInverted()) {
1216 if (state.equals(OnOffType.ON)) {
1217 state = OnOffType.OFF;
1219 state = OnOffType.ON;
1221 } else if (state instanceof OpenClosedType && taggedItem.isInverted()) {
1222 if (state.equals(OpenClosedType.OPEN)) {
1223 state = OpenClosedType.CLOSED;
1225 state = OpenClosedType.OPEN;
1228 // if "not pressed", don't send an event
1229 if (state instanceof UnDefType || (state instanceof OnOffType && state.equals(OnOffType.OFF))
1230 || (state instanceof OpenClosedType && state.equals(OpenClosedType.CLOSED))) {
1234 lastValue = getKeyFromMapping(taggedItem, state, map, ProgrammableSwitchEnum.SINGLE_PRESS);
1240 private static RemainingDurationCharacteristic createRemainingDurationCharacteristic(HomekitTaggedItem taggedItem,
1241 HomekitAccessoryUpdater updater) {
1242 return new RemainingDurationCharacteristic(getIntSupplier(taggedItem, 0),
1243 getSubscriber(taggedItem, REMAINING_DURATION, updater),
1244 getUnsubscriber(taggedItem, REMAINING_DURATION, updater));
1247 private static RemoteKeyCharacteristic createRemoteKeyCharacteristic(HomekitTaggedItem taggedItem,
1248 HomekitAccessoryUpdater updater) {
1249 var map = createMapping(taggedItem, RemoteKeyEnum.class);
1250 return new RemoteKeyCharacteristic((value) -> setValueFromEnum(taggedItem, value, map));
1253 private static RotationDirectionCharacteristic createRotationDirectionCharacteristic(HomekitTaggedItem taggedItem,
1254 HomekitAccessoryUpdater updater) {
1255 var map = createMapping(taggedItem, RotationDirectionEnum.class);
1256 return new RotationDirectionCharacteristic(
1257 () -> getEnumFromItem(taggedItem, map, RotationDirectionEnum.CLOCKWISE),
1258 (value) -> setValueFromEnum(taggedItem, value, map),
1259 getSubscriber(taggedItem, ROTATION_DIRECTION, updater),
1260 getUnsubscriber(taggedItem, ROTATION_DIRECTION, updater));
1263 private static RotationSpeedCharacteristic createRotationSpeedCharacteristic(HomekitTaggedItem item,
1264 HomekitAccessoryUpdater updater) {
1265 return new RotationSpeedCharacteristic(
1266 item.getConfigurationAsDouble(HomekitTaggedItem.MIN_VALUE,
1267 RotationSpeedCharacteristic.DEFAULT_MIN_VALUE),
1268 item.getConfigurationAsDouble(HomekitTaggedItem.MAX_VALUE,
1269 RotationSpeedCharacteristic.DEFAULT_MAX_VALUE),
1270 item.getConfigurationAsDouble(HomekitTaggedItem.STEP, RotationSpeedCharacteristic.DEFAULT_STEP),
1271 getDoubleSupplier(item, 0), setDoubleConsumer(item), getSubscriber(item, ROTATION_SPEED, updater),
1272 getUnsubscriber(item, ROTATION_SPEED, updater));
1275 private static SaturationCharacteristic createSaturationCharacteristic(HomekitTaggedItem taggedItem,
1276 HomekitAccessoryUpdater updater) {
1277 return new SaturationCharacteristic(() -> {
1279 State state = taggedItem.getItem().getState();
1280 if (state instanceof HSBType stateAsHSBType) {
1281 value = stateAsHSBType.getSaturation().doubleValue();
1282 } else if (state instanceof PercentType stateAsPercentType) {
1283 value = stateAsPercentType.doubleValue();
1285 return CompletableFuture.completedFuture(value);
1286 }, (saturation) -> {
1287 if (taggedItem.getBaseItem() instanceof ColorItem) {
1288 taggedItem.sendCommandProxy(HomekitCommandType.SATURATION_COMMAND,
1289 new PercentType(saturation.intValue()));
1291 LOGGER.warn("Item type {} is not supported for {}. Only Color type is supported.",
1292 taggedItem.getBaseItem().getType(), taggedItem.getName());
1294 }, getSubscriber(taggedItem, SATURATION, updater), getUnsubscriber(taggedItem, SATURATION, updater));
1297 private static SerialNumberCharacteristic createSerialNumberCharacteristic(HomekitTaggedItem taggedItem,
1298 HomekitAccessoryUpdater updater) {
1299 return new SerialNumberCharacteristic(() -> {
1300 final State state = taggedItem.getItem().getState();
1301 return CompletableFuture.completedFuture(state instanceof UnDefType ? "" : state.toString());
1305 private static SleepDiscoveryModeCharacteristic createSleepDiscoveryModeCharacteristic(HomekitTaggedItem taggedItem,
1306 HomekitAccessoryUpdater updater) {
1307 var map = createMapping(taggedItem, SleepDiscoveryModeEnum.class);
1308 return new SleepDiscoveryModeCharacteristic(
1309 () -> getEnumFromItem(taggedItem, map, SleepDiscoveryModeEnum.ALWAYS_DISCOVERABLE),
1310 getSubscriber(taggedItem, SLEEP_DISCOVERY_MODE, updater),
1311 getUnsubscriber(taggedItem, SLEEP_DISCOVERY_MODE, updater));
1314 private static StatusActiveCharacteristic createStatusActiveCharacteristic(HomekitTaggedItem taggedItem,
1315 HomekitAccessoryUpdater updater) {
1316 return new StatusActiveCharacteristic(
1317 () -> CompletableFuture.completedFuture(taggedItem.getItem().getState() == OnOffType.ON
1318 || taggedItem.getItem().getState() == OpenClosedType.OPEN),
1319 getSubscriber(taggedItem, ACTIVE_STATUS, updater), getUnsubscriber(taggedItem, ACTIVE_STATUS, updater));
1322 private static StatusFaultCharacteristic createStatusFaultCharacteristic(HomekitTaggedItem taggedItem,
1323 HomekitAccessoryUpdater updater) {
1324 var map = createMapping(taggedItem, StatusFaultEnum.class);
1325 return new StatusFaultCharacteristic(() -> getEnumFromItem(taggedItem, map, StatusFaultEnum.NO_FAULT),
1326 getSubscriber(taggedItem, FAULT_STATUS, updater), getUnsubscriber(taggedItem, FAULT_STATUS, updater));
1329 private static StatusLowBatteryCharacteristic createStatusLowBatteryCharacteristic(HomekitTaggedItem taggedItem,
1330 HomekitAccessoryUpdater updater) {
1331 BigDecimal lowThreshold = taggedItem.getConfiguration(HomekitTaggedItem.BATTERY_LOW_THRESHOLD,
1332 BigDecimal.valueOf(20));
1333 BooleanItemReader lowBatteryReader = new BooleanItemReader(taggedItem.getItem(),
1334 OnOffType.from(!taggedItem.isInverted()),
1335 taggedItem.isInverted() ? OpenClosedType.CLOSED : OpenClosedType.OPEN, lowThreshold, true);
1336 return new StatusLowBatteryCharacteristic(
1337 () -> CompletableFuture.completedFuture(
1338 lowBatteryReader.getValue() ? StatusLowBatteryEnum.LOW : StatusLowBatteryEnum.NORMAL),
1339 getSubscriber(taggedItem, BATTERY_LOW_STATUS, updater),
1340 getUnsubscriber(taggedItem, BATTERY_LOW_STATUS, updater));
1343 private static StatusTamperedCharacteristic createStatusTamperedCharacteristic(HomekitTaggedItem taggedItem,
1344 HomekitAccessoryUpdater updater) {
1345 var map = createMapping(taggedItem, StatusTamperedEnum.class);
1346 return new StatusTamperedCharacteristic(() -> getEnumFromItem(taggedItem, map, StatusTamperedEnum.NOT_TAMPERED),
1347 getSubscriber(taggedItem, TAMPERED_STATUS, updater),
1348 getUnsubscriber(taggedItem, TAMPERED_STATUS, updater));
1351 private static SulphurDioxideDensityCharacteristic createSulphurDioxideDensityCharacteristic(
1352 final HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
1353 return new SulphurDioxideDensityCharacteristic(
1354 getDoubleSupplier(taggedItem,
1355 taggedItem.getConfigurationAsDouble(HomekitTaggedItem.MIN_VALUE,
1356 SulphurDioxideDensityCharacteristic.DEFAULT_MIN_VALUE)),
1357 getSubscriber(taggedItem, SULPHUR_DIOXIDE_DENSITY, updater),
1358 getUnsubscriber(taggedItem, SULPHUR_DIOXIDE_DENSITY, updater));
1361 private static SwingModeCharacteristic createSwingModeCharacteristic(HomekitTaggedItem taggedItem,
1362 HomekitAccessoryUpdater updater) {
1363 var map = createMapping(taggedItem, SwingModeEnum.class);
1364 return new SwingModeCharacteristic(() -> getEnumFromItem(taggedItem, map, SwingModeEnum.SWING_DISABLED),
1365 (value) -> setValueFromEnum(taggedItem, value, map), getSubscriber(taggedItem, SWING_MODE, updater),
1366 getUnsubscriber(taggedItem, SWING_MODE, updater));
1369 private static TargetFanStateCharacteristic createTargetFanStateCharacteristic(HomekitTaggedItem taggedItem,
1370 HomekitAccessoryUpdater updater) {
1371 var map = createMapping(taggedItem, TargetFanStateEnum.class);
1372 return new TargetFanStateCharacteristic(() -> getEnumFromItem(taggedItem, map, TargetFanStateEnum.AUTO),
1373 (targetState) -> setValueFromEnum(taggedItem, targetState, map),
1374 getSubscriber(taggedItem, TARGET_FAN_STATE, updater),
1375 getUnsubscriber(taggedItem, TARGET_FAN_STATE, updater));
1378 private static TargetHeatingCoolingStateCharacteristic createTargetHeatingCoolingStateCharacteristic(
1379 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
1380 List<TargetHeatingCoolingStateEnum> validValues = new ArrayList<>();
1381 var map = createMapping(taggedItem, TargetHeatingCoolingStateEnum.class, validValues);
1382 return new TargetHeatingCoolingStateCharacteristic(validValues.toArray(new TargetHeatingCoolingStateEnum[0]),
1383 () -> getEnumFromItem(taggedItem, map, TargetHeatingCoolingStateEnum.OFF),
1384 (value) -> setValueFromEnum(taggedItem, value, map),
1385 getSubscriber(taggedItem, TARGET_HEATING_COOLING_STATE, updater),
1386 getUnsubscriber(taggedItem, TARGET_HEATING_COOLING_STATE, updater));
1389 private static TargetHorizontalTiltAngleCharacteristic createTargetHorizontalTiltAngleCharacteristic(
1390 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
1391 return new TargetHorizontalTiltAngleCharacteristic(getAngleSupplier(taggedItem, 0),
1392 setAngleConsumer(taggedItem), getSubscriber(taggedItem, TARGET_HORIZONTAL_TILT_ANGLE, updater),
1393 getUnsubscriber(taggedItem, TARGET_HORIZONTAL_TILT_ANGLE, updater));
1396 private static TargetMediaStateCharacteristic createTargetMediaStateCharacteristic(HomekitTaggedItem taggedItem,
1397 HomekitAccessoryUpdater updater) {
1398 var map = createMapping(taggedItem, TargetMediaStateEnum.class);
1399 return new TargetMediaStateCharacteristic(() -> getEnumFromItem(taggedItem, map, TargetMediaStateEnum.STOP),
1400 (value) -> setValueFromEnum(taggedItem, value, map),
1401 getSubscriber(taggedItem, TARGET_MEDIA_STATE, updater),
1402 getUnsubscriber(taggedItem, TARGET_MEDIA_STATE, updater));
1405 private static TargetRelativeHumidityCharacteristic createTargetRelativeHumidityCharacteristic(
1406 HomekitTaggedItem item, HomekitAccessoryUpdater updater) {
1407 return new TargetRelativeHumidityCharacteristic(getDoubleSupplier(item, 0), setDoubleConsumer(item),
1408 getSubscriber(item, TARGET_RELATIVE_HUMIDITY, updater),
1409 getUnsubscriber(item, TARGET_RELATIVE_HUMIDITY, updater));
1412 private static TargetTemperatureCharacteristic createTargetTemperatureCharacteristic(HomekitTaggedItem taggedItem,
1413 HomekitAccessoryUpdater updater) {
1414 double minValue = taggedItem
1415 .getConfigurationAsQuantity(HomekitTaggedItem.MIN_VALUE,
1416 Objects.requireNonNull(
1417 new QuantityType(TargetTemperatureCharacteristic.DEFAULT_MIN_VALUE, SIUnits.CELSIUS)
1418 .toUnit(getSystemTemperatureUnit())),
1420 .toUnit(SIUnits.CELSIUS).doubleValue();
1421 double maxValue = taggedItem
1422 .getConfigurationAsQuantity(HomekitTaggedItem.MAX_VALUE,
1423 Objects.requireNonNull(
1424 new QuantityType(TargetTemperatureCharacteristic.DEFAULT_MAX_VALUE, SIUnits.CELSIUS)
1425 .toUnit(getSystemTemperatureUnit())),
1427 .toUnit(SIUnits.CELSIUS).doubleValue();
1428 double step = taggedItem
1429 .getConfigurationAsQuantity(HomekitTaggedItem.STEP,
1430 Objects.requireNonNull(
1431 new QuantityType(TargetTemperatureCharacteristic.DEFAULT_STEP, SIUnits.CELSIUS)
1432 .toUnit(getSystemTemperatureUnit())),
1434 .toUnit(SIUnits.CELSIUS).doubleValue();
1435 return new TargetTemperatureCharacteristic(minValue, maxValue, step,
1436 getTemperatureSupplier(taggedItem, minValue), setTemperatureConsumer(taggedItem),
1437 getSubscriber(taggedItem, TARGET_TEMPERATURE, updater),
1438 getUnsubscriber(taggedItem, TARGET_TEMPERATURE, updater));
1441 private static TargetTiltAngleCharacteristic createTargetTiltAngleCharacteristic(HomekitTaggedItem taggedItem,
1442 HomekitAccessoryUpdater updater) {
1443 return new TargetTiltAngleCharacteristic(getAngleSupplier(taggedItem, 0), setAngleConsumer(taggedItem),
1444 getSubscriber(taggedItem, TARGET_TILT_ANGLE, updater),
1445 getUnsubscriber(taggedItem, TARGET_TILT_ANGLE, updater));
1448 private static TargetVerticalTiltAngleCharacteristic createTargetVerticalTiltAngleCharacteristic(
1449 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
1450 return new TargetVerticalTiltAngleCharacteristic(getAngleSupplier(taggedItem, 0), setAngleConsumer(taggedItem),
1451 getSubscriber(taggedItem, TARGET_HORIZONTAL_TILT_ANGLE, updater),
1452 getUnsubscriber(taggedItem, TARGET_HORIZONTAL_TILT_ANGLE, updater));
1455 private static TargetVisibilityStateCharacteristic createTargetVisibilityStateCharacteristic(
1456 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
1457 var map = createMapping(taggedItem, TargetVisibilityStateEnum.class, true);
1458 return new TargetVisibilityStateCharacteristic(
1459 () -> getEnumFromItem(taggedItem, map, TargetVisibilityStateEnum.HIDDEN),
1460 (value) -> setValueFromEnum(taggedItem, value, map),
1461 getSubscriber(taggedItem, TARGET_VISIBILITY_STATE, updater),
1462 getUnsubscriber(taggedItem, TARGET_VISIBILITY_STATE, updater));
1465 private static TemperatureDisplayUnitCharacteristic createTemperatureDisplayUnitCharacteristic(
1466 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
1467 var map = createMapping(taggedItem, TemperatureDisplayUnitEnum.class, true);
1468 return new TemperatureDisplayUnitCharacteristic(
1469 () -> getEnumFromItem(taggedItem, map,
1470 useFahrenheit() ? TemperatureDisplayUnitEnum.FAHRENHEIT : TemperatureDisplayUnitEnum.CELSIUS),
1471 (value) -> setValueFromEnum(taggedItem, value, map),
1472 getSubscriber(taggedItem, TEMPERATURE_UNIT, updater),
1473 getUnsubscriber(taggedItem, TEMPERATURE_UNIT, updater));
1476 private static VOCDensityCharacteristic createVOCDensityCharacteristic(final HomekitTaggedItem taggedItem,
1477 HomekitAccessoryUpdater updater) {
1478 return new VOCDensityCharacteristic(
1479 taggedItem.getConfigurationAsDouble(HomekitTaggedItem.MIN_VALUE,
1480 VOCDensityCharacteristic.DEFAULT_MIN_VALUE),
1481 taggedItem.getConfigurationAsDouble(HomekitTaggedItem.MAX_VALUE,
1482 VOCDensityCharacteristic.DEFAULT_MAX_VALUE),
1483 taggedItem.getConfigurationAsDouble(HomekitTaggedItem.STEP, VOCDensityCharacteristic.DEFAULT_STEP),
1484 getDoubleSupplier(taggedItem,
1485 taggedItem.getConfigurationAsDouble(HomekitTaggedItem.MIN_VALUE,
1486 VOCDensityCharacteristic.DEFAULT_MIN_VALUE)),
1487 getSubscriber(taggedItem, VOC_DENSITY, updater), getUnsubscriber(taggedItem, VOC_DENSITY, updater));
1490 private static VolumeCharacteristic createVolumeCharacteristic(HomekitTaggedItem taggedItem,
1491 HomekitAccessoryUpdater updater) {
1492 return new VolumeCharacteristic(getIntSupplier(taggedItem, 0),
1493 (volume) -> ((NumberItem) taggedItem.getItem()).send(new DecimalType(volume)),
1494 getSubscriber(taggedItem, DURATION, updater), getUnsubscriber(taggedItem, DURATION, updater));
1497 private static VolumeSelectorCharacteristic createVolumeSelectorCharacteristic(HomekitTaggedItem taggedItem,
1498 HomekitAccessoryUpdater updater) {
1499 if (taggedItem.getItem() instanceof DimmerItem) {
1500 return new VolumeSelectorCharacteristic((value) -> taggedItem
1501 .send(value.equals(VolumeSelectorEnum.INCREMENT) ? IncreaseDecreaseType.INCREASE
1502 : IncreaseDecreaseType.DECREASE));
1504 var map = createMapping(taggedItem, VolumeSelectorEnum.class);
1505 return new VolumeSelectorCharacteristic((value) -> setValueFromEnum(taggedItem, value, map));
1509 private static VolumeControlTypeCharacteristic createVolumeControlTypeCharacteristic(HomekitTaggedItem taggedItem,
1510 HomekitAccessoryUpdater updater) {
1511 var map = createMapping(taggedItem, VolumeControlTypeEnum.class);
1512 return new VolumeControlTypeCharacteristic(() -> getEnumFromItem(taggedItem, map, VolumeControlTypeEnum.NONE),
1513 getSubscriber(taggedItem, VOLUME_CONTROL_TYPE, updater),
1514 getUnsubscriber(taggedItem, VOLUME_CONTROL_TYPE, updater));