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;
33 import org.eclipse.jdt.annotation.NonNullByDefault;
34 import org.eclipse.jdt.annotation.Nullable;
35 import org.openhab.core.items.GenericItem;
36 import org.openhab.core.items.Item;
37 import org.openhab.core.library.items.ColorItem;
38 import org.openhab.core.library.items.DimmerItem;
39 import org.openhab.core.library.items.NumberItem;
40 import org.openhab.core.library.items.RollershutterItem;
41 import org.openhab.core.library.items.StringItem;
42 import org.openhab.core.library.items.SwitchItem;
43 import org.openhab.core.library.types.DecimalType;
44 import org.openhab.core.library.types.HSBType;
45 import org.openhab.core.library.types.IncreaseDecreaseType;
46 import org.openhab.core.library.types.OnOffType;
47 import org.openhab.core.library.types.OpenClosedType;
48 import org.openhab.core.library.types.PercentType;
49 import org.openhab.core.library.types.QuantityType;
50 import org.openhab.core.library.types.StopMoveType;
51 import org.openhab.core.library.types.StringType;
52 import org.openhab.core.library.unit.ImperialUnits;
53 import org.openhab.core.library.unit.SIUnits;
54 import org.openhab.core.library.unit.Units;
55 import org.openhab.core.types.State;
56 import org.openhab.core.types.UnDefType;
57 import org.openhab.io.homekit.Homekit;
58 import org.openhab.io.homekit.internal.HomekitAccessoryUpdater;
59 import org.openhab.io.homekit.internal.HomekitCharacteristicType;
60 import org.openhab.io.homekit.internal.HomekitCommandType;
61 import org.openhab.io.homekit.internal.HomekitException;
62 import org.openhab.io.homekit.internal.HomekitImpl;
63 import org.openhab.io.homekit.internal.HomekitTaggedItem;
64 import org.osgi.framework.FrameworkUtil;
65 import org.slf4j.Logger;
66 import org.slf4j.LoggerFactory;
68 import io.github.hapjava.characteristics.Characteristic;
69 import io.github.hapjava.characteristics.CharacteristicEnum;
70 import io.github.hapjava.characteristics.ExceptionalConsumer;
71 import io.github.hapjava.characteristics.HomekitCharacteristicChangeCallback;
72 import io.github.hapjava.characteristics.impl.accessoryinformation.FirmwareRevisionCharacteristic;
73 import io.github.hapjava.characteristics.impl.accessoryinformation.HardwareRevisionCharacteristic;
74 import io.github.hapjava.characteristics.impl.accessoryinformation.IdentifyCharacteristic;
75 import io.github.hapjava.characteristics.impl.accessoryinformation.ManufacturerCharacteristic;
76 import io.github.hapjava.characteristics.impl.accessoryinformation.ModelCharacteristic;
77 import io.github.hapjava.characteristics.impl.accessoryinformation.SerialNumberCharacteristic;
78 import io.github.hapjava.characteristics.impl.airquality.NitrogenDioxideDensityCharacteristic;
79 import io.github.hapjava.characteristics.impl.airquality.OzoneDensityCharacteristic;
80 import io.github.hapjava.characteristics.impl.airquality.PM10DensityCharacteristic;
81 import io.github.hapjava.characteristics.impl.airquality.PM25DensityCharacteristic;
82 import io.github.hapjava.characteristics.impl.airquality.SulphurDioxideDensityCharacteristic;
83 import io.github.hapjava.characteristics.impl.airquality.VOCDensityCharacteristic;
84 import io.github.hapjava.characteristics.impl.audio.MuteCharacteristic;
85 import io.github.hapjava.characteristics.impl.audio.VolumeCharacteristic;
86 import io.github.hapjava.characteristics.impl.battery.StatusLowBatteryCharacteristic;
87 import io.github.hapjava.characteristics.impl.battery.StatusLowBatteryEnum;
88 import io.github.hapjava.characteristics.impl.carbondioxidesensor.CarbonDioxideLevelCharacteristic;
89 import io.github.hapjava.characteristics.impl.carbondioxidesensor.CarbonDioxidePeakLevelCharacteristic;
90 import io.github.hapjava.characteristics.impl.carbonmonoxidesensor.CarbonMonoxideLevelCharacteristic;
91 import io.github.hapjava.characteristics.impl.carbonmonoxidesensor.CarbonMonoxidePeakLevelCharacteristic;
92 import io.github.hapjava.characteristics.impl.common.ActiveCharacteristic;
93 import io.github.hapjava.characteristics.impl.common.ActiveEnum;
94 import io.github.hapjava.characteristics.impl.common.ActiveIdentifierCharacteristic;
95 import io.github.hapjava.characteristics.impl.common.ConfiguredNameCharacteristic;
96 import io.github.hapjava.characteristics.impl.common.IdentifierCharacteristic;
97 import io.github.hapjava.characteristics.impl.common.IsConfiguredCharacteristic;
98 import io.github.hapjava.characteristics.impl.common.IsConfiguredEnum;
99 import io.github.hapjava.characteristics.impl.common.NameCharacteristic;
100 import io.github.hapjava.characteristics.impl.common.ObstructionDetectedCharacteristic;
101 import io.github.hapjava.characteristics.impl.common.StatusActiveCharacteristic;
102 import io.github.hapjava.characteristics.impl.common.StatusFaultCharacteristic;
103 import io.github.hapjava.characteristics.impl.common.StatusFaultEnum;
104 import io.github.hapjava.characteristics.impl.common.StatusTamperedCharacteristic;
105 import io.github.hapjava.characteristics.impl.common.StatusTamperedEnum;
106 import io.github.hapjava.characteristics.impl.fan.CurrentFanStateCharacteristic;
107 import io.github.hapjava.characteristics.impl.fan.CurrentFanStateEnum;
108 import io.github.hapjava.characteristics.impl.fan.LockPhysicalControlsCharacteristic;
109 import io.github.hapjava.characteristics.impl.fan.LockPhysicalControlsEnum;
110 import io.github.hapjava.characteristics.impl.fan.RotationDirectionCharacteristic;
111 import io.github.hapjava.characteristics.impl.fan.RotationDirectionEnum;
112 import io.github.hapjava.characteristics.impl.fan.RotationSpeedCharacteristic;
113 import io.github.hapjava.characteristics.impl.fan.SwingModeCharacteristic;
114 import io.github.hapjava.characteristics.impl.fan.SwingModeEnum;
115 import io.github.hapjava.characteristics.impl.fan.TargetFanStateCharacteristic;
116 import io.github.hapjava.characteristics.impl.fan.TargetFanStateEnum;
117 import io.github.hapjava.characteristics.impl.filtermaintenance.FilterLifeLevelCharacteristic;
118 import io.github.hapjava.characteristics.impl.filtermaintenance.ResetFilterIndicationCharacteristic;
119 import io.github.hapjava.characteristics.impl.humiditysensor.CurrentRelativeHumidityCharacteristic;
120 import io.github.hapjava.characteristics.impl.humiditysensor.TargetRelativeHumidityCharacteristic;
121 import io.github.hapjava.characteristics.impl.inputsource.CurrentVisibilityStateCharacteristic;
122 import io.github.hapjava.characteristics.impl.inputsource.CurrentVisibilityStateEnum;
123 import io.github.hapjava.characteristics.impl.inputsource.InputDeviceTypeCharacteristic;
124 import io.github.hapjava.characteristics.impl.inputsource.InputDeviceTypeEnum;
125 import io.github.hapjava.characteristics.impl.inputsource.InputSourceTypeCharacteristic;
126 import io.github.hapjava.characteristics.impl.inputsource.InputSourceTypeEnum;
127 import io.github.hapjava.characteristics.impl.inputsource.TargetVisibilityStateCharacteristic;
128 import io.github.hapjava.characteristics.impl.inputsource.TargetVisibilityStateEnum;
129 import io.github.hapjava.characteristics.impl.lightbulb.BrightnessCharacteristic;
130 import io.github.hapjava.characteristics.impl.lightbulb.ColorTemperatureCharacteristic;
131 import io.github.hapjava.characteristics.impl.lightbulb.HueCharacteristic;
132 import io.github.hapjava.characteristics.impl.lightbulb.SaturationCharacteristic;
133 import io.github.hapjava.characteristics.impl.slat.CurrentTiltAngleCharacteristic;
134 import io.github.hapjava.characteristics.impl.slat.TargetTiltAngleCharacteristic;
135 import io.github.hapjava.characteristics.impl.television.ClosedCaptionsCharacteristic;
136 import io.github.hapjava.characteristics.impl.television.ClosedCaptionsEnum;
137 import io.github.hapjava.characteristics.impl.television.CurrentMediaStateCharacteristic;
138 import io.github.hapjava.characteristics.impl.television.CurrentMediaStateEnum;
139 import io.github.hapjava.characteristics.impl.television.PictureModeCharacteristic;
140 import io.github.hapjava.characteristics.impl.television.PictureModeEnum;
141 import io.github.hapjava.characteristics.impl.television.PowerModeCharacteristic;
142 import io.github.hapjava.characteristics.impl.television.PowerModeEnum;
143 import io.github.hapjava.characteristics.impl.television.RemoteKeyCharacteristic;
144 import io.github.hapjava.characteristics.impl.television.RemoteKeyEnum;
145 import io.github.hapjava.characteristics.impl.television.SleepDiscoveryModeCharacteristic;
146 import io.github.hapjava.characteristics.impl.television.SleepDiscoveryModeEnum;
147 import io.github.hapjava.characteristics.impl.television.TargetMediaStateCharacteristic;
148 import io.github.hapjava.characteristics.impl.television.TargetMediaStateEnum;
149 import io.github.hapjava.characteristics.impl.televisionspeaker.VolumeControlTypeCharacteristic;
150 import io.github.hapjava.characteristics.impl.televisionspeaker.VolumeControlTypeEnum;
151 import io.github.hapjava.characteristics.impl.televisionspeaker.VolumeSelectorCharacteristic;
152 import io.github.hapjava.characteristics.impl.televisionspeaker.VolumeSelectorEnum;
153 import io.github.hapjava.characteristics.impl.thermostat.CoolingThresholdTemperatureCharacteristic;
154 import io.github.hapjava.characteristics.impl.thermostat.CurrentHeatingCoolingStateCharacteristic;
155 import io.github.hapjava.characteristics.impl.thermostat.CurrentHeatingCoolingStateEnum;
156 import io.github.hapjava.characteristics.impl.thermostat.CurrentTemperatureCharacteristic;
157 import io.github.hapjava.characteristics.impl.thermostat.HeatingThresholdTemperatureCharacteristic;
158 import io.github.hapjava.characteristics.impl.thermostat.TargetHeatingCoolingStateCharacteristic;
159 import io.github.hapjava.characteristics.impl.thermostat.TargetHeatingCoolingStateEnum;
160 import io.github.hapjava.characteristics.impl.thermostat.TargetTemperatureCharacteristic;
161 import io.github.hapjava.characteristics.impl.thermostat.TemperatureDisplayUnitCharacteristic;
162 import io.github.hapjava.characteristics.impl.thermostat.TemperatureDisplayUnitEnum;
163 import io.github.hapjava.characteristics.impl.valve.RemainingDurationCharacteristic;
164 import io.github.hapjava.characteristics.impl.valve.SetDurationCharacteristic;
165 import io.github.hapjava.characteristics.impl.windowcovering.CurrentHorizontalTiltAngleCharacteristic;
166 import io.github.hapjava.characteristics.impl.windowcovering.CurrentVerticalTiltAngleCharacteristic;
167 import io.github.hapjava.characteristics.impl.windowcovering.HoldPositionCharacteristic;
168 import io.github.hapjava.characteristics.impl.windowcovering.TargetHorizontalTiltAngleCharacteristic;
169 import io.github.hapjava.characteristics.impl.windowcovering.TargetVerticalTiltAngleCharacteristic;
172 * Creates an optional characteristics .
174 * @author Eugen Freiter - Initial contribution
177 public class HomekitCharacteristicFactory {
178 private static final Logger LOGGER = LoggerFactory.getLogger(HomekitCharacteristicFactory.class);
180 // List of optional characteristics and corresponding method to create them.
181 private static final Map<HomekitCharacteristicType, BiFunction<HomekitTaggedItem, HomekitAccessoryUpdater, Characteristic>> OPTIONAL = new HashMap<>() {
183 put(ACTIVE, HomekitCharacteristicFactory::createActiveCharacteristic);
184 put(ACTIVE_IDENTIFIER, HomekitCharacteristicFactory::createActiveIdentifierCharacteristic);
185 put(ACTIVE_STATUS, HomekitCharacteristicFactory::createStatusActiveCharacteristic);
186 put(BATTERY_LOW_STATUS, HomekitCharacteristicFactory::createStatusLowBatteryCharacteristic);
187 put(BRIGHTNESS, HomekitCharacteristicFactory::createBrightnessCharacteristic);
188 put(CARBON_DIOXIDE_LEVEL, HomekitCharacteristicFactory::createCarbonDioxideLevelCharacteristic);
189 put(CARBON_DIOXIDE_PEAK_LEVEL, HomekitCharacteristicFactory::createCarbonDioxidePeakLevelCharacteristic);
190 put(CARBON_MONOXIDE_LEVEL, HomekitCharacteristicFactory::createCarbonMonoxideLevelCharacteristic);
191 put(CARBON_MONOXIDE_PEAK_LEVEL, HomekitCharacteristicFactory::createCarbonMonoxidePeakLevelCharacteristic);
192 put(CLOSED_CAPTIONS, HomekitCharacteristicFactory::createClosedCaptionsCharacteristic);
193 put(COLOR_TEMPERATURE, HomekitCharacteristicFactory::createColorTemperatureCharacteristic);
194 put(CONFIGURED, HomekitCharacteristicFactory::createIsConfiguredCharacteristic);
195 put(CONFIGURED_NAME, HomekitCharacteristicFactory::createConfiguredNameCharacteristic);
196 put(COOLING_THRESHOLD_TEMPERATURE, HomekitCharacteristicFactory::createCoolingThresholdCharacteristic);
197 put(CURRENT_HEATING_COOLING_STATE,
198 HomekitCharacteristicFactory::createCurrentHeatingCoolingStateCharacteristic);
199 put(CURRENT_FAN_STATE, HomekitCharacteristicFactory::createCurrentFanStateCharacteristic);
200 put(CURRENT_HORIZONTAL_TILT_ANGLE,
201 HomekitCharacteristicFactory::createCurrentHorizontalTiltAngleCharacteristic);
202 put(CURRENT_MEDIA_STATE, HomekitCharacteristicFactory::createCurrentMediaStateCharacteristic);
203 put(CURRENT_TILT_ANGLE, HomekitCharacteristicFactory::createCurrentTiltAngleCharacteristic);
204 put(CURRENT_VERTICAL_TILT_ANGLE,
205 HomekitCharacteristicFactory::createCurrentVerticalTiltAngleCharacteristic);
206 put(CURRENT_VISIBILITY, HomekitCharacteristicFactory::createCurrentVisibilityStateCharacteristic);
207 put(CURRENT_TEMPERATURE, HomekitCharacteristicFactory::createCurrentTemperatureCharacteristic);
208 put(DURATION, HomekitCharacteristicFactory::createDurationCharacteristic);
209 put(FAULT_STATUS, HomekitCharacteristicFactory::createStatusFaultCharacteristic);
210 put(FIRMWARE_REVISION, HomekitCharacteristicFactory::createFirmwareRevisionCharacteristic);
211 put(FILTER_LIFE_LEVEL, HomekitCharacteristicFactory::createFilterLifeLevelCharacteristic);
212 put(FILTER_RESET_INDICATION, HomekitCharacteristicFactory::createFilterResetCharacteristic);
213 put(HARDWARE_REVISION, HomekitCharacteristicFactory::createHardwareRevisionCharacteristic);
214 put(HEATING_THRESHOLD_TEMPERATURE, HomekitCharacteristicFactory::createHeatingThresholdCharacteristic);
215 put(HOLD_POSITION, HomekitCharacteristicFactory::createHoldPositionCharacteristic);
216 put(HUE, HomekitCharacteristicFactory::createHueCharacteristic);
217 put(IDENTIFIER, HomekitCharacteristicFactory::createIdentifierCharacteristic);
218 put(IDENTIFY, HomekitCharacteristicFactory::createIdentifyCharacteristic);
219 put(INPUT_DEVICE_TYPE, HomekitCharacteristicFactory::createInputDeviceTypeCharacteristic);
220 put(INPUT_SOURCE_TYPE, HomekitCharacteristicFactory::createInputSourceTypeCharacteristic);
221 put(LOCK_CONTROL, HomekitCharacteristicFactory::createLockPhysicalControlsCharacteristic);
222 put(MANUFACTURER, HomekitCharacteristicFactory::createManufacturerCharacteristic);
223 put(MODEL, HomekitCharacteristicFactory::createModelCharacteristic);
224 put(MUTE, HomekitCharacteristicFactory::createMuteCharacteristic);
225 put(NAME, HomekitCharacteristicFactory::createNameCharacteristic);
226 put(NITROGEN_DIOXIDE_DENSITY, HomekitCharacteristicFactory::createNitrogenDioxideDensityCharacteristic);
227 put(OBSTRUCTION_STATUS, HomekitCharacteristicFactory::createObstructionDetectedCharacteristic);
228 put(OZONE_DENSITY, HomekitCharacteristicFactory::createOzoneDensityCharacteristic);
229 put(PICTURE_MODE, HomekitCharacteristicFactory::createPictureModeCharacteristic);
230 put(PM10_DENSITY, HomekitCharacteristicFactory::createPM10DensityCharacteristic);
231 put(PM25_DENSITY, HomekitCharacteristicFactory::createPM25DensityCharacteristic);
232 put(POWER_MODE, HomekitCharacteristicFactory::createPowerModeCharacteristic);
233 put(REMAINING_DURATION, HomekitCharacteristicFactory::createRemainingDurationCharacteristic);
234 put(REMOTE_KEY, HomekitCharacteristicFactory::createRemoteKeyCharacteristic);
235 put(RELATIVE_HUMIDITY, HomekitCharacteristicFactory::createRelativeHumidityCharacteristic);
236 put(ROTATION_DIRECTION, HomekitCharacteristicFactory::createRotationDirectionCharacteristic);
237 put(ROTATION_SPEED, HomekitCharacteristicFactory::createRotationSpeedCharacteristic);
238 put(SATURATION, HomekitCharacteristicFactory::createSaturationCharacteristic);
239 put(SERIAL_NUMBER, HomekitCharacteristicFactory::createSerialNumberCharacteristic);
240 put(SLEEP_DISCOVERY_MODE, HomekitCharacteristicFactory::createSleepDiscoveryModeCharacteristic);
241 put(SULPHUR_DIOXIDE_DENSITY, HomekitCharacteristicFactory::createSulphurDioxideDensityCharacteristic);
242 put(SWING_MODE, HomekitCharacteristicFactory::createSwingModeCharacteristic);
243 put(TAMPERED_STATUS, HomekitCharacteristicFactory::createStatusTamperedCharacteristic);
244 put(TARGET_FAN_STATE, HomekitCharacteristicFactory::createTargetFanStateCharacteristic);
245 put(TARGET_HEATING_COOLING_STATE,
246 HomekitCharacteristicFactory::createTargetHeatingCoolingStateCharacteristic);
247 put(TARGET_HORIZONTAL_TILT_ANGLE,
248 HomekitCharacteristicFactory::createTargetHorizontalTiltAngleCharacteristic);
249 put(TARGET_MEDIA_STATE, HomekitCharacteristicFactory::createTargetMediaStateCharacteristic);
250 put(TARGET_RELATIVE_HUMIDITY, HomekitCharacteristicFactory::createTargetRelativeHumidityCharacteristic);
251 put(TARGET_TEMPERATURE, HomekitCharacteristicFactory::createTargetTemperatureCharacteristic);
252 put(TARGET_TILT_ANGLE, HomekitCharacteristicFactory::createTargetTiltAngleCharacteristic);
253 put(TARGET_VERTICAL_TILT_ANGLE, HomekitCharacteristicFactory::createTargetVerticalTiltAngleCharacteristic);
254 put(TARGET_VISIBILITY_STATE, HomekitCharacteristicFactory::createTargetVisibilityStateCharacteristic);
255 put(TEMPERATURE_UNIT, HomekitCharacteristicFactory::createTemperatureDisplayUnitCharacteristic);
256 put(VOC_DENSITY, HomekitCharacteristicFactory::createVOCDensityCharacteristic);
257 put(VOLUME, HomekitCharacteristicFactory::createVolumeCharacteristic);
258 put(VOLUME_CONTROL_TYPE, HomekitCharacteristicFactory::createVolumeControlTypeCharacteristic);
259 put(VOLUME_SELECTOR, HomekitCharacteristicFactory::createVolumeSelectorCharacteristic);
263 public static @Nullable Characteristic createNullableCharacteristic(HomekitTaggedItem item,
264 HomekitAccessoryUpdater updater) {
265 final @Nullable HomekitCharacteristicType type = item.getCharacteristicType();
266 LOGGER.trace("Create characteristic {}", item);
267 if (OPTIONAL.containsKey(type)) {
268 return OPTIONAL.get(type).apply(item, updater);
274 * Create HomeKit characteristic
276 * @param item corresponding OH item
277 * @param updater update to keep OH item and HomeKit characteristic in sync
278 * @return HomeKit characteristic
280 public static Characteristic createCharacteristic(HomekitTaggedItem item, HomekitAccessoryUpdater updater)
281 throws HomekitException {
282 Characteristic characteristic = createNullableCharacteristic(item, updater);
283 if (characteristic != null) {
284 return characteristic;
286 final @Nullable HomekitCharacteristicType type = item.getCharacteristicType();
287 LOGGER.warn("Unsupported optional characteristic from item {}. Accessory type {}, characteristic type {}",
288 item.getName(), item.getAccessoryType(), type.getTag());
289 throw new HomekitException(
290 "Unsupported optional characteristic. Characteristic type \"" + type.getTag() + "\"");
294 * Create an EnumMap for a particular CharacteristicEnum.
296 * By default, the map will simply be from the Enum value to the string version of its value.
297 * If the item is a Number item, though, the values will the be underlying integer code
298 * for the item, as a String.
299 * Then the item's metadata will be inspected, applying any custom mappings.
300 * Finally, if customEnumList is supplied, it will be filled out with those mappings
301 * that are actually referenced in the metadata.
304 * @param klazz The HAP-Java Enum for the characteristic.
305 * @param customEnumList Optional output list of which enums are explicitly mentioned.
306 * @param inverted Default-invert the 0/1 values of the HAP enum when linked to a Switch or Contact item.
307 * This is set by the addon when creating mappings for specific characteristics where the 0 and 1
308 * values for the enum do not map naturally to 0/OFF/CLOSED and 1/ON/OPEN of openHAB items.
309 * Note that this is separate from the inverted item-level metadata configuration, which can be
310 * thought of independently as applying on top of this setting. It essentially "multiplies" out,
311 * but can also be thought of as simply swapping whichever value OFF/CLOSED and ON/OPEN are
312 * associated with, which has already been set.
315 public static <T extends Enum<T> & CharacteristicEnum> Map<T, String> createMapping(HomekitTaggedItem item,
316 Class<T> klazz, @Nullable List<T> customEnumList, boolean inverted) {
317 EnumMap<T, String> map = new EnumMap(klazz);
318 var dataTypes = item.getBaseItem().getAcceptedDataTypes();
319 boolean switchType = dataTypes.contains(OnOffType.class);
320 boolean contactType = dataTypes.contains(OpenClosedType.class);
321 boolean percentType = dataTypes.contains(PercentType.class);
322 boolean numberType = dataTypes.contains(DecimalType.class) || percentType || switchType || contactType;
324 if (item.isInverted()) {
325 inverted = !inverted;
327 String onValue = switchType ? OnOffType.ON.toString() : OpenClosedType.OPEN.toString();
328 String offValue = switchType ? OnOffType.OFF.toString() : OpenClosedType.CLOSED.toString();
330 for (var k : klazz.getEnumConstants()) {
332 int code = k.getCode();
333 if ((switchType || contactType) && code == 0) {
334 map.put(k, inverted ? onValue : offValue);
335 } else if ((switchType || contactType) && code == 1) {
336 map.put(k, inverted ? offValue : onValue);
337 } else if (percentType && code == 0) {
339 } else if (percentType && code == 1) {
342 map.put(k, Integer.toString(code));
345 map.put(k, k.toString());
348 var configuration = item.getConfiguration();
349 if (configuration != null) {
350 map.forEach((k, current_value) -> {
351 final Object newValue = configuration.get(k.toString());
352 if (newValue instanceof String || newValue instanceof Number) {
353 map.put(k, newValue.toString());
354 if (customEnumList != null) {
355 customEnumList.add(k);
360 if (customEnumList != null && customEnumList.isEmpty()) {
361 customEnumList.addAll(map.keySet());
363 LOGGER.debug("Created {} mapping for item {} ({}): {}", klazz.getSimpleName(), item.getName(),
364 item.getBaseItem().getClass().getSimpleName(), map);
368 public static <T extends Enum<T> & CharacteristicEnum> Map<T, String> createMapping(HomekitTaggedItem item,
370 return createMapping(item, klazz, null, false);
373 public static <T extends Enum<T> & CharacteristicEnum> Map<T, String> createMapping(HomekitTaggedItem item,
374 Class<T> klazz, @Nullable List<T> customEnumList) {
375 return createMapping(item, klazz, customEnumList, false);
378 public static <T extends Enum<T> & CharacteristicEnum> Map<T, String> createMapping(HomekitTaggedItem item,
379 Class<T> klazz, boolean inverted) {
380 return createMapping(item, klazz, null, inverted);
384 * Takes item state as value and retrieves the key for that value from mapping.
385 * E.g. used to map StringItem value to HomeKit Enum
388 * @param mapping mapping
389 * @param defaultValue default value if nothing found in mapping
390 * @param <T> type of the result derived from
391 * @return key for the value
393 public static <T> T getKeyFromMapping(HomekitTaggedItem item, Map<T, String> mapping, T defaultValue) {
394 final State state = item.getItem().getState();
395 LOGGER.trace("getKeyFromMapping: characteristic {}, state {}, mapping {}", item.getAccessoryType().getTag(),
399 if (state instanceof UnDefType) {
401 } else if (state instanceof StringType || state instanceof OnOffType || state instanceof OpenClosedType) {
402 value = state.toString();
403 } else if (state.getClass().equals(PercentType.class)) {
404 // We specifically want PercentType, but _not_ HSBType, so don't use instanceof
405 value = state.as(OnOffType.class).toString();
406 } else if (state.getClass().equals(DecimalType.class)) {
407 // We specifically want DecimalType, but _not_ PercentType or HSBType, so don't use instanceof
408 value = Integer.toString(((DecimalType) state).intValue());
411 "Wrong value type {} ({}) for {} characteristic of the item {}. Expected StringItem, NumberItem, or SwitchItem.",
412 state.toString(), state.getClass().getSimpleName(), item.getAccessoryType().getTag(),
417 return mapping.entrySet().stream().filter(entry -> value.equalsIgnoreCase(entry.getValue())).findAny()
418 .map(Map.Entry::getKey).orElseGet(() -> {
420 "Wrong value {} for {} characteristic of the item {}. Expected one of following {}. Returning {}.",
421 state.toString(), item.getAccessoryType().getTag(), item.getName(), mapping.values(),
427 // supporting methods
429 public static boolean useFahrenheit() {
430 return Boolean.TRUE.equals(FrameworkUtil.getBundle(HomekitImpl.class).getBundleContext()
431 .getServiceReference(Homekit.class.getName()).getProperty("useFahrenheitTemperature"));
434 public static TemperatureDisplayUnitCharacteristic createSystemTemperatureDisplayUnitCharacteristic() {
435 return new TemperatureDisplayUnitCharacteristic(() -> CompletableFuture
436 .completedFuture(HomekitCharacteristicFactory.useFahrenheit() ? TemperatureDisplayUnitEnum.FAHRENHEIT
437 : TemperatureDisplayUnitEnum.CELSIUS),
444 private static <T extends CharacteristicEnum> CompletableFuture<T> getEnumFromItem(HomekitTaggedItem item,
445 Map<T, String> mapping, T defaultValue) {
446 return CompletableFuture.completedFuture(getKeyFromMapping(item, mapping, defaultValue));
449 public static <T extends Enum<T>> void setValueFromEnum(HomekitTaggedItem taggedItem, T value, Map<T, String> map) {
450 if (taggedItem.getBaseItem() instanceof NumberItem) {
451 taggedItem.send(new DecimalType(Objects.requireNonNull(map.get(value))));
452 } else if (taggedItem.getBaseItem() instanceof SwitchItem) {
453 taggedItem.send(OnOffType.from(Objects.requireNonNull(map.get(value))));
455 taggedItem.send(new StringType(map.get(value)));
459 private static int getIntFromItem(HomekitTaggedItem taggedItem, int defaultValue) {
460 int value = defaultValue;
461 final State state = taggedItem.getItem().getState();
462 if (state instanceof PercentType stateAsPercentType) {
463 value = stateAsPercentType.intValue();
464 } else if (state instanceof DecimalType stateAsDecimalType) {
465 value = stateAsDecimalType.intValue();
466 } else if (state instanceof UnDefType) {
467 LOGGER.debug("Item state {} is UNDEF {}. Returning default value {}", state, taggedItem.getName(),
471 "Item state {} is not supported for {}. Only PercentType and DecimalType (0/100) are supported.",
472 state, taggedItem.getName());
477 /** special method for tilts. it converts percentage to angle */
478 private static int getAngleFromItem(HomekitTaggedItem taggedItem, int defaultValue) {
479 int value = defaultValue;
480 final State state = taggedItem.getItem().getState();
481 if (state instanceof PercentType stateAsPercentType) {
482 value = (int) ((stateAsPercentType.intValue() * 90.0) / 50.0 - 90.0);
484 value = getIntFromItem(taggedItem, defaultValue);
489 private static <T extends Quantity<T>> double convertAndRound(double value, Unit<T> from, Unit<T> to) {
490 double rawValue = from.equals(to) ? value : from.getConverterTo(to).convert(value);
491 return new BigDecimal(rawValue).setScale(1, RoundingMode.HALF_UP).doubleValue();
494 public static @Nullable Double stateAsTemperature(@Nullable State state) {
495 if (state == null || state instanceof UnDefType) {
499 if (state instanceof QuantityType<?> qt) {
500 if (qt.getDimension().equals(SIUnits.CELSIUS.getDimension())) {
501 return qt.toUnit(SIUnits.CELSIUS).doubleValue();
505 return convertToCelsius(state.as(DecimalType.class).doubleValue());
508 public static double convertToCelsius(double degrees) {
509 return convertAndRound(degrees, useFahrenheit() ? ImperialUnits.FAHRENHEIT : SIUnits.CELSIUS, SIUnits.CELSIUS);
512 public static double convertFromCelsius(double degrees) {
513 return convertAndRound(degrees, SIUnits.CELSIUS, useFahrenheit() ? ImperialUnits.FAHRENHEIT : SIUnits.CELSIUS);
516 public static double getTemperatureStep(HomekitTaggedItem taggedItem, double defaultValue) {
517 return taggedItem.getConfigurationAsQuantity(HomekitTaggedItem.STEP,
518 new QuantityType(defaultValue, SIUnits.CELSIUS), true).doubleValue();
521 private static Supplier<CompletableFuture<Integer>> getAngleSupplier(HomekitTaggedItem taggedItem,
523 return () -> CompletableFuture.completedFuture(getAngleFromItem(taggedItem, defaultValue));
526 private static Supplier<CompletableFuture<Integer>> getIntSupplier(HomekitTaggedItem taggedItem, int defaultValue) {
527 return () -> CompletableFuture.completedFuture(getIntFromItem(taggedItem, defaultValue));
530 private static ExceptionalConsumer<Integer> setIntConsumer(HomekitTaggedItem taggedItem) {
532 if (taggedItem.getBaseItem() instanceof NumberItem) {
533 taggedItem.send(new DecimalType(value));
535 LOGGER.warn("Item type {} is not supported for {}. Only NumberItem is supported.",
536 taggedItem.getBaseItem().getType(), taggedItem.getName());
541 private static ExceptionalConsumer<Integer> setPercentConsumer(HomekitTaggedItem taggedItem) {
543 if (taggedItem.getBaseItem() instanceof NumberItem) {
544 taggedItem.send(new DecimalType(value));
545 } else if (taggedItem.getBaseItem() instanceof DimmerItem) {
546 taggedItem.send(new PercentType(value));
548 LOGGER.warn("Item type {} is not supported for {}. Only DimmerItem and NumberItem are supported.",
549 taggedItem.getBaseItem().getType(), taggedItem.getName());
554 private static ExceptionalConsumer<Integer> setAngleConsumer(HomekitTaggedItem taggedItem) {
556 if (taggedItem.getBaseItem() instanceof NumberItem) {
557 taggedItem.send(new DecimalType(value));
558 } else if (taggedItem.getBaseItem() instanceof DimmerItem) {
559 value = (int) (value * 50.0 / 90.0 + 50.0);
560 taggedItem.send(new PercentType(value));
562 LOGGER.warn("Item type {} is not supported for {}. Only DimmerItem and NumberItem are supported.",
563 taggedItem.getBaseItem().getType(), taggedItem.getName());
568 private static Supplier<CompletableFuture<Double>> getDoubleSupplier(HomekitTaggedItem taggedItem,
569 double defaultValue) {
571 final State state = taggedItem.getItem().getState();
572 double value = defaultValue;
573 if (state instanceof PercentType stateAsPercentType) {
574 value = stateAsPercentType.doubleValue();
575 } else if (state instanceof DecimalType stateAsDecimalType) {
576 value = stateAsDecimalType.doubleValue();
577 } else if (state instanceof QuantityType stateAsQuantityType) {
578 value = stateAsQuantityType.doubleValue();
580 return CompletableFuture.completedFuture(value);
584 private static ExceptionalConsumer<Double> setDoubleConsumer(HomekitTaggedItem taggedItem) {
586 if (taggedItem.getBaseItem() instanceof NumberItem) {
587 taggedItem.send(new DecimalType(value.doubleValue()));
588 } else if (taggedItem.getBaseItem() instanceof DimmerItem) {
589 taggedItem.send(new PercentType(value.intValue()));
591 LOGGER.warn("Item type {} is not supported for {}. Only Number and Dimmer type are supported.",
592 taggedItem.getBaseItem().getType(), taggedItem.getName());
597 private static Supplier<CompletableFuture<Double>> getTemperatureSupplier(HomekitTaggedItem taggedItem,
598 double defaultValue) {
600 final @Nullable Double value = stateAsTemperature(taggedItem.getItem().getState());
601 return CompletableFuture.completedFuture(value != null ? value : defaultValue);
605 private static ExceptionalConsumer<Double> setTemperatureConsumer(HomekitTaggedItem taggedItem) {
607 if (taggedItem.getBaseItem() instanceof NumberItem) {
608 taggedItem.send(new DecimalType(convertFromCelsius(value)));
610 LOGGER.warn("Item type {} is not supported for {}. Only Number type is supported.",
611 taggedItem.getBaseItem().getType(), taggedItem.getName());
616 protected static Consumer<HomekitCharacteristicChangeCallback> getSubscriber(HomekitTaggedItem taggedItem,
617 HomekitCharacteristicType key, HomekitAccessoryUpdater updater) {
618 return (callback) -> updater.subscribe((GenericItem) taggedItem.getItem(), key.getTag(), callback);
621 protected static Runnable getUnsubscriber(HomekitTaggedItem taggedItem, HomekitCharacteristicType key,
622 HomekitAccessoryUpdater updater) {
623 return () -> updater.unsubscribe((GenericItem) taggedItem.getItem(), key.getTag());
626 // METHODS TO CREATE SINGLE CHARACTERISTIC FROM OPENHAB ITEM
628 private static ActiveCharacteristic createActiveCharacteristic(HomekitTaggedItem taggedItem,
629 HomekitAccessoryUpdater updater) {
630 var map = createMapping(taggedItem, ActiveEnum.class, false);
631 return new ActiveCharacteristic(() -> getEnumFromItem(taggedItem, map, ActiveEnum.INACTIVE),
632 (value) -> setValueFromEnum(taggedItem, value, map), getSubscriber(taggedItem, ACTIVE, updater),
633 getUnsubscriber(taggedItem, ACTIVE, updater));
636 private static ActiveIdentifierCharacteristic createActiveIdentifierCharacteristic(HomekitTaggedItem taggedItem,
637 HomekitAccessoryUpdater updater) {
638 return new ActiveIdentifierCharacteristic(getIntSupplier(taggedItem, 1), setIntConsumer(taggedItem),
639 getSubscriber(taggedItem, ACTIVE_IDENTIFIER, updater),
640 getUnsubscriber(taggedItem, ACTIVE_IDENTIFIER, updater));
643 private static BrightnessCharacteristic createBrightnessCharacteristic(HomekitTaggedItem taggedItem,
644 HomekitAccessoryUpdater updater) {
645 return new BrightnessCharacteristic(() -> {
647 final State state = taggedItem.getItem().getState();
648 if (state instanceof HSBType stateAsHSBType) {
649 value = stateAsHSBType.getBrightness().intValue();
650 } else if (state instanceof PercentType stateAsPercentType) {
651 value = stateAsPercentType.intValue();
653 return CompletableFuture.completedFuture(value);
655 if (taggedItem.getBaseItem() instanceof DimmerItem) {
656 taggedItem.sendCommandProxy(HomekitCommandType.BRIGHTNESS_COMMAND, new PercentType(brightness));
658 LOGGER.warn("Item type {} is not supported for {}. Only ColorItem and DimmerItem are supported.",
659 taggedItem.getBaseItem().getType(), taggedItem.getName());
661 }, getSubscriber(taggedItem, BRIGHTNESS, updater), getUnsubscriber(taggedItem, BRIGHTNESS, updater));
664 private static CarbonDioxideLevelCharacteristic createCarbonDioxideLevelCharacteristic(HomekitTaggedItem taggedItem,
665 HomekitAccessoryUpdater updater) {
666 return new CarbonDioxideLevelCharacteristic(
667 getDoubleSupplier(taggedItem,
668 taggedItem.getConfigurationAsDouble(HomekitTaggedItem.MIN_VALUE,
669 CarbonDioxideLevelCharacteristic.DEFAULT_MIN_VALUE)),
670 getSubscriber(taggedItem, CARBON_MONOXIDE_LEVEL, updater),
671 getUnsubscriber(taggedItem, CARBON_MONOXIDE_LEVEL, updater));
674 private static CarbonDioxidePeakLevelCharacteristic createCarbonDioxidePeakLevelCharacteristic(
675 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
676 return new CarbonDioxidePeakLevelCharacteristic(
677 getDoubleSupplier(taggedItem,
678 taggedItem.getConfigurationAsDouble(HomekitTaggedItem.MIN_VALUE,
679 CarbonDioxidePeakLevelCharacteristic.DEFAULT_MIN_VALUE)),
680 getSubscriber(taggedItem, CARBON_MONOXIDE_PEAK_LEVEL, updater),
681 getUnsubscriber(taggedItem, CARBON_MONOXIDE_PEAK_LEVEL, updater));
684 private static CarbonMonoxideLevelCharacteristic createCarbonMonoxideLevelCharacteristic(
685 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
686 return new CarbonMonoxideLevelCharacteristic(
687 getDoubleSupplier(taggedItem,
688 taggedItem.getConfigurationAsDouble(HomekitTaggedItem.MIN_VALUE,
689 CarbonMonoxideLevelCharacteristic.DEFAULT_MIN_VALUE)),
690 getSubscriber(taggedItem, CARBON_DIOXIDE_LEVEL, updater),
691 getUnsubscriber(taggedItem, CARBON_DIOXIDE_LEVEL, updater));
694 private static CarbonMonoxidePeakLevelCharacteristic createCarbonMonoxidePeakLevelCharacteristic(
695 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
696 return new CarbonMonoxidePeakLevelCharacteristic(
697 getDoubleSupplier(taggedItem,
698 taggedItem.getConfigurationAsDouble(HomekitTaggedItem.MIN_VALUE,
699 CarbonMonoxidePeakLevelCharacteristic.DEFAULT_MIN_VALUE)),
700 getSubscriber(taggedItem, CARBON_DIOXIDE_PEAK_LEVEL, updater),
701 getUnsubscriber(taggedItem, CARBON_DIOXIDE_PEAK_LEVEL, updater));
704 private static ClosedCaptionsCharacteristic createClosedCaptionsCharacteristic(HomekitTaggedItem taggedItem,
705 HomekitAccessoryUpdater updater) {
706 var map = createMapping(taggedItem, ClosedCaptionsEnum.class);
707 return new ClosedCaptionsCharacteristic(() -> getEnumFromItem(taggedItem, map, ClosedCaptionsEnum.DISABLED),
708 (value) -> setValueFromEnum(taggedItem, value, map),
709 getSubscriber(taggedItem, CLOSED_CAPTIONS, updater),
710 getUnsubscriber(taggedItem, CLOSED_CAPTIONS, updater));
713 private static ColorTemperatureCharacteristic createColorTemperatureCharacteristic(HomekitTaggedItem taggedItem,
714 HomekitAccessoryUpdater updater) {
715 final boolean inverted = taggedItem.isInverted();
717 int minValue = taggedItem
718 .getConfigurationAsQuantity(HomekitTaggedItem.MIN_VALUE,
719 new QuantityType(ColorTemperatureCharacteristic.DEFAULT_MIN_VALUE, Units.MIRED), false)
721 int maxValue = taggedItem
722 .getConfigurationAsQuantity(HomekitTaggedItem.MAX_VALUE,
723 new QuantityType(ColorTemperatureCharacteristic.DEFAULT_MAX_VALUE, Units.MIRED), false)
726 // It's common to swap these if you're providing in Kelvin instead of mired
727 if (minValue > maxValue) {
733 final int finalMinValue = minValue;
734 final int range = maxValue - minValue;
736 return new ColorTemperatureCharacteristic(minValue, maxValue, () -> {
737 int value = finalMinValue;
738 final State state = taggedItem.getItem().getState();
739 if (state instanceof QuantityType<?> qt) {
740 // Number:Temperature
741 qt = qt.toInvertibleUnit(Units.MIRED);
743 LOGGER.warn("Item {}'s state '{}' is not convertible to mireds.", taggedItem.getName(), state);
745 value = qt.intValue();
747 } else if (state instanceof PercentType stateAsPercentType) {
748 double percent = stateAsPercentType.doubleValue();
749 // invert so that 0% == coolest
751 percent = 100.0 - percent;
755 // scale to the originally configured range
756 value = (int) (percent * range / 100) + finalMinValue;
757 } else if (state instanceof DecimalType stateAsDecimalType) {
758 value = stateAsDecimalType.intValue();
760 return CompletableFuture.completedFuture(value);
762 if (taggedItem.getBaseItem() instanceof DimmerItem) {
763 // scale to a percent
764 double percent = (((double) value) - finalMinValue) * 100 / range;
766 percent = 100.0 - percent;
768 taggedItem.send(new PercentType(BigDecimal.valueOf(percent)));
769 } else if (taggedItem.getBaseItem() instanceof NumberItem) {
770 taggedItem.send(new QuantityType(value, Units.MIRED));
772 }, getSubscriber(taggedItem, COLOR_TEMPERATURE, updater),
773 getUnsubscriber(taggedItem, COLOR_TEMPERATURE, updater));
776 private static ConfiguredNameCharacteristic createConfiguredNameCharacteristic(HomekitTaggedItem taggedItem,
777 HomekitAccessoryUpdater updater) {
778 return new ConfiguredNameCharacteristic(() -> {
779 final State state = taggedItem.getItem().getState();
780 return CompletableFuture
781 .completedFuture(state instanceof UnDefType ? taggedItem.getName() : state.toString());
782 }, (value) -> ((StringItem) taggedItem.getItem()).send(new StringType(value)),
783 getSubscriber(taggedItem, CONFIGURED_NAME, updater),
784 getUnsubscriber(taggedItem, CONFIGURED_NAME, updater));
787 private static CoolingThresholdTemperatureCharacteristic createCoolingThresholdCharacteristic(
788 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
789 double minValue = HomekitCharacteristicFactory.convertToCelsius(taggedItem.getConfigurationAsDouble(
790 HomekitTaggedItem.MIN_VALUE, CoolingThresholdTemperatureCharacteristic.DEFAULT_MIN_VALUE));
791 double maxValue = HomekitCharacteristicFactory.convertToCelsius(taggedItem.getConfigurationAsDouble(
792 HomekitTaggedItem.MAX_VALUE, CoolingThresholdTemperatureCharacteristic.DEFAULT_MAX_VALUE));
793 double step = getTemperatureStep(taggedItem, CoolingThresholdTemperatureCharacteristic.DEFAULT_STEP);
794 return new CoolingThresholdTemperatureCharacteristic(minValue, maxValue, step,
795 getTemperatureSupplier(taggedItem, minValue), setTemperatureConsumer(taggedItem),
796 getSubscriber(taggedItem, COOLING_THRESHOLD_TEMPERATURE, updater),
797 getUnsubscriber(taggedItem, COOLING_THRESHOLD_TEMPERATURE, updater));
800 private static CurrentHeatingCoolingStateCharacteristic createCurrentHeatingCoolingStateCharacteristic(
801 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
802 List<CurrentHeatingCoolingStateEnum> validValues = new ArrayList<>();
803 var map = createMapping(taggedItem, CurrentHeatingCoolingStateEnum.class, validValues);
804 return new CurrentHeatingCoolingStateCharacteristic(validValues.toArray(new CurrentHeatingCoolingStateEnum[0]),
805 () -> getEnumFromItem(taggedItem, map, CurrentHeatingCoolingStateEnum.OFF),
806 getSubscriber(taggedItem, CURRENT_HEATING_COOLING_STATE, updater),
807 getUnsubscriber(taggedItem, CURRENT_HEATING_COOLING_STATE, updater));
810 private static CurrentFanStateCharacteristic createCurrentFanStateCharacteristic(HomekitTaggedItem taggedItem,
811 HomekitAccessoryUpdater updater) {
812 var map = createMapping(taggedItem, CurrentFanStateEnum.class);
813 return new CurrentFanStateCharacteristic(() -> getEnumFromItem(taggedItem, map, CurrentFanStateEnum.INACTIVE),
814 getSubscriber(taggedItem, CURRENT_FAN_STATE, updater),
815 getUnsubscriber(taggedItem, CURRENT_FAN_STATE, updater));
818 private static CurrentHorizontalTiltAngleCharacteristic createCurrentHorizontalTiltAngleCharacteristic(
819 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
820 return new CurrentHorizontalTiltAngleCharacteristic(getAngleSupplier(taggedItem, 0),
821 getSubscriber(taggedItem, CURRENT_HORIZONTAL_TILT_ANGLE, updater),
822 getUnsubscriber(taggedItem, CURRENT_HORIZONTAL_TILT_ANGLE, updater));
825 private static CurrentMediaStateCharacteristic createCurrentMediaStateCharacteristic(HomekitTaggedItem taggedItem,
826 HomekitAccessoryUpdater updater) {
827 var map = createMapping(taggedItem, CurrentMediaStateEnum.class);
828 return new CurrentMediaStateCharacteristic(
829 () -> getEnumFromItem(taggedItem, map, CurrentMediaStateEnum.UNKNOWN),
830 getSubscriber(taggedItem, CURRENT_MEDIA_STATE, updater),
831 getUnsubscriber(taggedItem, CURRENT_MEDIA_STATE, updater));
834 private static CurrentTemperatureCharacteristic createCurrentTemperatureCharacteristic(HomekitTaggedItem taggedItem,
835 HomekitAccessoryUpdater updater) {
836 double minValue = HomekitCharacteristicFactory.convertToCelsius(taggedItem.getConfigurationAsDouble(
837 HomekitTaggedItem.MIN_VALUE, CurrentTemperatureCharacteristic.DEFAULT_MIN_VALUE));
838 double maxValue = HomekitCharacteristicFactory.convertToCelsius(taggedItem.getConfigurationAsDouble(
839 HomekitTaggedItem.MAX_VALUE, CurrentTemperatureCharacteristic.DEFAULT_MAX_VALUE));
840 double step = getTemperatureStep(taggedItem, CurrentTemperatureCharacteristic.DEFAULT_STEP);
841 return new CurrentTemperatureCharacteristic(minValue, maxValue, step,
842 getTemperatureSupplier(taggedItem, minValue), getSubscriber(taggedItem, TARGET_TEMPERATURE, updater),
843 getUnsubscriber(taggedItem, TARGET_TEMPERATURE, updater));
846 private static CurrentTiltAngleCharacteristic createCurrentTiltAngleCharacteristic(HomekitTaggedItem taggedItem,
847 HomekitAccessoryUpdater updater) {
848 return new CurrentTiltAngleCharacteristic(getAngleSupplier(taggedItem, 0),
849 getSubscriber(taggedItem, CURRENT_TILT_ANGLE, updater),
850 getUnsubscriber(taggedItem, CURRENT_TILT_ANGLE, updater));
853 private static CurrentVerticalTiltAngleCharacteristic createCurrentVerticalTiltAngleCharacteristic(
854 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
855 return new CurrentVerticalTiltAngleCharacteristic(getAngleSupplier(taggedItem, 0),
856 getSubscriber(taggedItem, CURRENT_VERTICAL_TILT_ANGLE, updater),
857 getUnsubscriber(taggedItem, CURRENT_VERTICAL_TILT_ANGLE, updater));
860 private static CurrentVisibilityStateCharacteristic createCurrentVisibilityStateCharacteristic(
861 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
862 var map = createMapping(taggedItem, CurrentVisibilityStateEnum.class, true);
863 return new CurrentVisibilityStateCharacteristic(
864 () -> getEnumFromItem(taggedItem, map, CurrentVisibilityStateEnum.HIDDEN),
865 getSubscriber(taggedItem, CURRENT_VISIBILITY, updater),
866 getUnsubscriber(taggedItem, CURRENT_VISIBILITY, updater));
869 private static SetDurationCharacteristic createDurationCharacteristic(HomekitTaggedItem taggedItem,
870 HomekitAccessoryUpdater updater) {
871 return new SetDurationCharacteristic(() -> {
872 int value = getIntFromItem(taggedItem, 0);
873 final @Nullable Map<String, Object> itemConfiguration = taggedItem.getConfiguration();
874 if ((value == 0) && (itemConfiguration != null)) { // check for default duration
875 final Object duration = itemConfiguration.get(HomekitValveImpl.CONFIG_DEFAULT_DURATION);
876 if (duration instanceof BigDecimal durationAsBigDecimal) {
877 value = durationAsBigDecimal.intValue();
878 if (taggedItem.getItem() instanceof NumberItem taggedNumberItem) {
879 taggedNumberItem.setState(new DecimalType(value));
883 return CompletableFuture.completedFuture(value);
884 }, setIntConsumer(taggedItem), getSubscriber(taggedItem, DURATION, updater),
885 getUnsubscriber(taggedItem, DURATION, updater));
888 private static FilterLifeLevelCharacteristic createFilterLifeLevelCharacteristic(HomekitTaggedItem taggedItem,
889 HomekitAccessoryUpdater updater) {
890 return new FilterLifeLevelCharacteristic(getDoubleSupplier(taggedItem, 0),
891 getSubscriber(taggedItem, FILTER_LIFE_LEVEL, updater),
892 getUnsubscriber(taggedItem, FILTER_LIFE_LEVEL, updater));
895 private static ResetFilterIndicationCharacteristic createFilterResetCharacteristic(HomekitTaggedItem taggedItem,
896 HomekitAccessoryUpdater updater) {
897 return new ResetFilterIndicationCharacteristic(
898 (value) -> ((SwitchItem) taggedItem.getBaseItem()).send(OnOffType.ON));
901 private static FirmwareRevisionCharacteristic createFirmwareRevisionCharacteristic(HomekitTaggedItem taggedItem,
902 HomekitAccessoryUpdater updater) {
903 return new FirmwareRevisionCharacteristic(() -> {
904 final State state = taggedItem.getItem().getState();
905 return CompletableFuture.completedFuture(state instanceof UnDefType ? "" : state.toString());
909 private static HardwareRevisionCharacteristic createHardwareRevisionCharacteristic(HomekitTaggedItem taggedItem,
910 HomekitAccessoryUpdater updater) {
911 return new HardwareRevisionCharacteristic(() -> {
912 final State state = taggedItem.getItem().getState();
913 return CompletableFuture.completedFuture(state instanceof UnDefType ? "" : state.toString());
917 private static HeatingThresholdTemperatureCharacteristic createHeatingThresholdCharacteristic(
918 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
919 double minValue = HomekitCharacteristicFactory.convertToCelsius(taggedItem.getConfigurationAsDouble(
920 HomekitTaggedItem.MIN_VALUE, HeatingThresholdTemperatureCharacteristic.DEFAULT_MIN_VALUE));
921 double maxValue = HomekitCharacteristicFactory.convertToCelsius(taggedItem.getConfigurationAsDouble(
922 HomekitTaggedItem.MAX_VALUE, HeatingThresholdTemperatureCharacteristic.DEFAULT_MAX_VALUE));
923 double step = getTemperatureStep(taggedItem, HeatingThresholdTemperatureCharacteristic.DEFAULT_STEP);
924 return new HeatingThresholdTemperatureCharacteristic(minValue, maxValue, step,
925 getTemperatureSupplier(taggedItem, minValue), setTemperatureConsumer(taggedItem),
926 getSubscriber(taggedItem, HEATING_THRESHOLD_TEMPERATURE, updater),
927 getUnsubscriber(taggedItem, HEATING_THRESHOLD_TEMPERATURE, updater));
930 private static HoldPositionCharacteristic createHoldPositionCharacteristic(HomekitTaggedItem taggedItem,
931 HomekitAccessoryUpdater updater) {
932 final Item item = taggedItem.getBaseItem();
933 if (!(item instanceof SwitchItem || item instanceof RollershutterItem)) {
935 "Item {} cannot be used for the HoldPosition characteristic; only SwitchItem and RollershutterItem are supported. Hold requests will be ignored.",
939 return new HoldPositionCharacteristic(value -> {
944 if (item instanceof SwitchItem switchItem) {
945 switchItem.send(OnOffType.ON);
946 } else if (item instanceof RollershutterItem rollershutterItem) {
947 rollershutterItem.send(StopMoveType.STOP);
952 private static HueCharacteristic createHueCharacteristic(HomekitTaggedItem taggedItem,
953 HomekitAccessoryUpdater updater) {
954 return new HueCharacteristic(() -> {
956 State state = taggedItem.getItem().getState();
957 if (state instanceof HSBType stateAsHSBType) {
958 value = stateAsHSBType.getHue().doubleValue();
960 return CompletableFuture.completedFuture(value);
962 if (taggedItem.getBaseItem() instanceof ColorItem) {
963 taggedItem.sendCommandProxy(HomekitCommandType.HUE_COMMAND, new DecimalType(hue));
965 LOGGER.warn("Item type {} is not supported for {}. Only Color type is supported.",
966 taggedItem.getBaseItem().getType(), taggedItem.getName());
968 }, getSubscriber(taggedItem, HUE, updater), getUnsubscriber(taggedItem, HUE, updater));
971 private static IdentifierCharacteristic createIdentifierCharacteristic(HomekitTaggedItem taggedItem,
972 HomekitAccessoryUpdater updater) {
973 return new IdentifierCharacteristic(getIntSupplier(taggedItem, 1));
976 private static IdentifyCharacteristic createIdentifyCharacteristic(HomekitTaggedItem taggedItem,
977 HomekitAccessoryUpdater updater) {
978 return new IdentifyCharacteristic((value) -> ((SwitchItem) taggedItem.getBaseItem()).send(OnOffType.ON));
981 private static InputDeviceTypeCharacteristic createInputDeviceTypeCharacteristic(HomekitTaggedItem taggedItem,
982 HomekitAccessoryUpdater updater) {
983 var map = createMapping(taggedItem, InputDeviceTypeEnum.class);
984 return new InputDeviceTypeCharacteristic(() -> getEnumFromItem(taggedItem, map, InputDeviceTypeEnum.OTHER),
985 getSubscriber(taggedItem, INPUT_DEVICE_TYPE, updater),
986 getUnsubscriber(taggedItem, INPUT_DEVICE_TYPE, updater));
989 private static InputSourceTypeCharacteristic createInputSourceTypeCharacteristic(HomekitTaggedItem taggedItem,
990 HomekitAccessoryUpdater updater) {
991 var map = createMapping(taggedItem, InputSourceTypeEnum.class);
992 return new InputSourceTypeCharacteristic(() -> getEnumFromItem(taggedItem, map, InputSourceTypeEnum.OTHER),
993 getSubscriber(taggedItem, INPUT_SOURCE_TYPE, updater),
994 getUnsubscriber(taggedItem, INPUT_SOURCE_TYPE, updater));
997 private static IsConfiguredCharacteristic createIsConfiguredCharacteristic(HomekitTaggedItem taggedItem,
998 HomekitAccessoryUpdater updater) {
999 var map = createMapping(taggedItem, IsConfiguredEnum.class);
1000 return new IsConfiguredCharacteristic(() -> getEnumFromItem(taggedItem, map, IsConfiguredEnum.NOT_CONFIGURED),
1001 (value) -> setValueFromEnum(taggedItem, value, map), getSubscriber(taggedItem, CONFIGURED, updater),
1002 getUnsubscriber(taggedItem, CONFIGURED, updater));
1005 private static LockPhysicalControlsCharacteristic createLockPhysicalControlsCharacteristic(
1006 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
1007 var map = createMapping(taggedItem, LockPhysicalControlsEnum.class);
1008 return new LockPhysicalControlsCharacteristic(
1009 () -> getEnumFromItem(taggedItem, map, LockPhysicalControlsEnum.CONTROL_LOCK_DISABLED),
1010 (value) -> setValueFromEnum(taggedItem, value, map), getSubscriber(taggedItem, LOCK_CONTROL, updater),
1011 getUnsubscriber(taggedItem, LOCK_CONTROL, updater));
1014 private static ManufacturerCharacteristic createManufacturerCharacteristic(HomekitTaggedItem taggedItem,
1015 HomekitAccessoryUpdater updater) {
1016 return new ManufacturerCharacteristic(() -> {
1017 final State state = taggedItem.getItem().getState();
1018 return CompletableFuture.completedFuture(state instanceof UnDefType ? "" : state.toString());
1022 private static ModelCharacteristic createModelCharacteristic(HomekitTaggedItem taggedItem,
1023 HomekitAccessoryUpdater updater) {
1024 return new ModelCharacteristic(() -> {
1025 final State state = taggedItem.getItem().getState();
1026 return CompletableFuture.completedFuture(state instanceof UnDefType ? "" : state.toString());
1030 private static MuteCharacteristic createMuteCharacteristic(HomekitTaggedItem taggedItem,
1031 HomekitAccessoryUpdater updater) {
1032 BooleanItemReader muteReader = new BooleanItemReader(taggedItem.getItem(),
1033 OnOffType.from(!taggedItem.isInverted()),
1034 taggedItem.isInverted() ? OpenClosedType.CLOSED : OpenClosedType.OPEN);
1035 return new MuteCharacteristic(() -> CompletableFuture.completedFuture(muteReader.getValue()),
1036 (value) -> taggedItem.send(OnOffType.from(value)), getSubscriber(taggedItem, MUTE, updater),
1037 getUnsubscriber(taggedItem, MUTE, updater));
1040 private static NameCharacteristic createNameCharacteristic(HomekitTaggedItem taggedItem,
1041 HomekitAccessoryUpdater updater) {
1042 return new NameCharacteristic(() -> {
1043 final State state = taggedItem.getItem().getState();
1044 return CompletableFuture.completedFuture(state instanceof UnDefType ? "" : state.toString());
1048 private static NitrogenDioxideDensityCharacteristic createNitrogenDioxideDensityCharacteristic(
1049 final HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
1050 return new NitrogenDioxideDensityCharacteristic(
1051 getDoubleSupplier(taggedItem,
1052 taggedItem.getConfigurationAsDouble(HomekitTaggedItem.MIN_VALUE,
1053 NitrogenDioxideDensityCharacteristic.DEFAULT_MIN_VALUE)),
1054 getSubscriber(taggedItem, NITROGEN_DIOXIDE_DENSITY, updater),
1055 getUnsubscriber(taggedItem, NITROGEN_DIOXIDE_DENSITY, updater));
1058 private static ObstructionDetectedCharacteristic createObstructionDetectedCharacteristic(
1059 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
1060 return new ObstructionDetectedCharacteristic(
1061 () -> CompletableFuture.completedFuture(taggedItem.getItem().getState() == OnOffType.ON
1062 || taggedItem.getItem().getState() == OpenClosedType.OPEN),
1063 getSubscriber(taggedItem, OBSTRUCTION_STATUS, updater),
1064 getUnsubscriber(taggedItem, OBSTRUCTION_STATUS, updater));
1067 private static OzoneDensityCharacteristic createOzoneDensityCharacteristic(final HomekitTaggedItem taggedItem,
1068 HomekitAccessoryUpdater updater) {
1069 return new OzoneDensityCharacteristic(
1070 getDoubleSupplier(taggedItem,
1071 taggedItem.getConfigurationAsDouble(HomekitTaggedItem.MIN_VALUE,
1072 OzoneDensityCharacteristic.DEFAULT_MIN_VALUE)),
1073 getSubscriber(taggedItem, OZONE_DENSITY, updater), getUnsubscriber(taggedItem, OZONE_DENSITY, updater));
1076 private static PM10DensityCharacteristic createPM10DensityCharacteristic(final HomekitTaggedItem taggedItem,
1077 HomekitAccessoryUpdater updater) {
1078 return new PM10DensityCharacteristic(
1079 getDoubleSupplier(taggedItem,
1080 taggedItem.getConfigurationAsDouble(HomekitTaggedItem.MIN_VALUE,
1081 PM10DensityCharacteristic.DEFAULT_MIN_VALUE)),
1082 getSubscriber(taggedItem, PM10_DENSITY, updater), getUnsubscriber(taggedItem, PM10_DENSITY, updater));
1085 private static PM25DensityCharacteristic createPM25DensityCharacteristic(final HomekitTaggedItem taggedItem,
1086 HomekitAccessoryUpdater updater) {
1087 return new PM25DensityCharacteristic(
1088 getDoubleSupplier(taggedItem,
1089 taggedItem.getConfigurationAsDouble(HomekitTaggedItem.MIN_VALUE,
1090 PM25DensityCharacteristic.DEFAULT_MIN_VALUE)),
1091 getSubscriber(taggedItem, PM25_DENSITY, updater), getUnsubscriber(taggedItem, PM25_DENSITY, updater));
1094 private static CurrentRelativeHumidityCharacteristic createRelativeHumidityCharacteristic(
1095 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
1096 return new CurrentRelativeHumidityCharacteristic(getDoubleSupplier(taggedItem, 0.0),
1097 getSubscriber(taggedItem, RELATIVE_HUMIDITY, updater),
1098 getUnsubscriber(taggedItem, RELATIVE_HUMIDITY, updater));
1101 private static PictureModeCharacteristic createPictureModeCharacteristic(HomekitTaggedItem taggedItem,
1102 HomekitAccessoryUpdater updater) {
1103 var map = createMapping(taggedItem, PictureModeEnum.class);
1104 return new PictureModeCharacteristic(() -> getEnumFromItem(taggedItem, map, PictureModeEnum.OTHER),
1105 (value) -> setValueFromEnum(taggedItem, value, map), getSubscriber(taggedItem, PICTURE_MODE, updater),
1106 getUnsubscriber(taggedItem, PICTURE_MODE, updater));
1109 private static PowerModeCharacteristic createPowerModeCharacteristic(HomekitTaggedItem taggedItem,
1110 HomekitAccessoryUpdater updater) {
1111 var map = createMapping(taggedItem, PowerModeEnum.class, true);
1112 return new PowerModeCharacteristic((value) -> setValueFromEnum(taggedItem, value, map));
1115 private static RemainingDurationCharacteristic createRemainingDurationCharacteristic(HomekitTaggedItem taggedItem,
1116 HomekitAccessoryUpdater updater) {
1117 return new RemainingDurationCharacteristic(getIntSupplier(taggedItem, 0),
1118 getSubscriber(taggedItem, REMAINING_DURATION, updater),
1119 getUnsubscriber(taggedItem, REMAINING_DURATION, updater));
1122 private static RemoteKeyCharacteristic createRemoteKeyCharacteristic(HomekitTaggedItem taggedItem,
1123 HomekitAccessoryUpdater updater) {
1124 var map = createMapping(taggedItem, RemoteKeyEnum.class);
1125 return new RemoteKeyCharacteristic((value) -> setValueFromEnum(taggedItem, value, map));
1128 private static RotationDirectionCharacteristic createRotationDirectionCharacteristic(HomekitTaggedItem taggedItem,
1129 HomekitAccessoryUpdater updater) {
1130 var map = createMapping(taggedItem, RotationDirectionEnum.class);
1131 return new RotationDirectionCharacteristic(
1132 () -> getEnumFromItem(taggedItem, map, RotationDirectionEnum.CLOCKWISE),
1133 (value) -> setValueFromEnum(taggedItem, value, map),
1134 getSubscriber(taggedItem, ROTATION_DIRECTION, updater),
1135 getUnsubscriber(taggedItem, ROTATION_DIRECTION, updater));
1138 private static RotationSpeedCharacteristic createRotationSpeedCharacteristic(HomekitTaggedItem item,
1139 HomekitAccessoryUpdater updater) {
1140 return new RotationSpeedCharacteristic(
1141 item.getConfigurationAsDouble(HomekitTaggedItem.MIN_VALUE,
1142 RotationSpeedCharacteristic.DEFAULT_MIN_VALUE),
1143 item.getConfigurationAsDouble(HomekitTaggedItem.MAX_VALUE,
1144 RotationSpeedCharacteristic.DEFAULT_MAX_VALUE),
1145 item.getConfigurationAsDouble(HomekitTaggedItem.STEP, RotationSpeedCharacteristic.DEFAULT_STEP),
1146 getDoubleSupplier(item, 0), setDoubleConsumer(item), getSubscriber(item, ROTATION_SPEED, updater),
1147 getUnsubscriber(item, ROTATION_SPEED, updater));
1150 private static SaturationCharacteristic createSaturationCharacteristic(HomekitTaggedItem taggedItem,
1151 HomekitAccessoryUpdater updater) {
1152 return new SaturationCharacteristic(() -> {
1154 State state = taggedItem.getItem().getState();
1155 if (state instanceof HSBType stateAsHSBType) {
1156 value = stateAsHSBType.getSaturation().doubleValue();
1157 } else if (state instanceof PercentType stateAsPercentType) {
1158 value = stateAsPercentType.doubleValue();
1160 return CompletableFuture.completedFuture(value);
1161 }, (saturation) -> {
1162 if (taggedItem.getBaseItem() instanceof ColorItem) {
1163 taggedItem.sendCommandProxy(HomekitCommandType.SATURATION_COMMAND,
1164 new PercentType(saturation.intValue()));
1166 LOGGER.warn("Item type {} is not supported for {}. Only Color type is supported.",
1167 taggedItem.getBaseItem().getType(), taggedItem.getName());
1169 }, getSubscriber(taggedItem, SATURATION, updater), getUnsubscriber(taggedItem, SATURATION, updater));
1172 private static SerialNumberCharacteristic createSerialNumberCharacteristic(HomekitTaggedItem taggedItem,
1173 HomekitAccessoryUpdater updater) {
1174 return new SerialNumberCharacteristic(() -> {
1175 final State state = taggedItem.getItem().getState();
1176 return CompletableFuture.completedFuture(state instanceof UnDefType ? "" : state.toString());
1180 private static SleepDiscoveryModeCharacteristic createSleepDiscoveryModeCharacteristic(HomekitTaggedItem taggedItem,
1181 HomekitAccessoryUpdater updater) {
1182 var map = createMapping(taggedItem, SleepDiscoveryModeEnum.class);
1183 return new SleepDiscoveryModeCharacteristic(
1184 () -> getEnumFromItem(taggedItem, map, SleepDiscoveryModeEnum.ALWAYS_DISCOVERABLE),
1185 getSubscriber(taggedItem, SLEEP_DISCOVERY_MODE, updater),
1186 getUnsubscriber(taggedItem, SLEEP_DISCOVERY_MODE, updater));
1189 private static StatusActiveCharacteristic createStatusActiveCharacteristic(HomekitTaggedItem taggedItem,
1190 HomekitAccessoryUpdater updater) {
1191 return new StatusActiveCharacteristic(
1192 () -> CompletableFuture.completedFuture(taggedItem.getItem().getState() == OnOffType.ON
1193 || taggedItem.getItem().getState() == OpenClosedType.OPEN),
1194 getSubscriber(taggedItem, ACTIVE_STATUS, updater), getUnsubscriber(taggedItem, ACTIVE_STATUS, updater));
1197 private static StatusFaultCharacteristic createStatusFaultCharacteristic(HomekitTaggedItem taggedItem,
1198 HomekitAccessoryUpdater updater) {
1199 var map = createMapping(taggedItem, StatusFaultEnum.class);
1200 return new StatusFaultCharacteristic(() -> getEnumFromItem(taggedItem, map, StatusFaultEnum.NO_FAULT),
1201 getSubscriber(taggedItem, FAULT_STATUS, updater), getUnsubscriber(taggedItem, FAULT_STATUS, updater));
1204 private static StatusLowBatteryCharacteristic createStatusLowBatteryCharacteristic(HomekitTaggedItem taggedItem,
1205 HomekitAccessoryUpdater updater) {
1206 BigDecimal lowThreshold = taggedItem.getConfiguration(HomekitTaggedItem.BATTERY_LOW_THRESHOLD,
1207 BigDecimal.valueOf(20));
1208 BooleanItemReader lowBatteryReader = new BooleanItemReader(taggedItem.getItem(),
1209 OnOffType.from(!taggedItem.isInverted()),
1210 taggedItem.isInverted() ? OpenClosedType.CLOSED : OpenClosedType.OPEN, lowThreshold, true);
1211 return new StatusLowBatteryCharacteristic(
1212 () -> CompletableFuture.completedFuture(
1213 lowBatteryReader.getValue() ? StatusLowBatteryEnum.LOW : StatusLowBatteryEnum.NORMAL),
1214 getSubscriber(taggedItem, BATTERY_LOW_STATUS, updater),
1215 getUnsubscriber(taggedItem, BATTERY_LOW_STATUS, updater));
1218 private static StatusTamperedCharacteristic createStatusTamperedCharacteristic(HomekitTaggedItem taggedItem,
1219 HomekitAccessoryUpdater updater) {
1220 var map = createMapping(taggedItem, StatusTamperedEnum.class);
1221 return new StatusTamperedCharacteristic(() -> getEnumFromItem(taggedItem, map, StatusTamperedEnum.NOT_TAMPERED),
1222 getSubscriber(taggedItem, TAMPERED_STATUS, updater),
1223 getUnsubscriber(taggedItem, TAMPERED_STATUS, updater));
1226 private static SulphurDioxideDensityCharacteristic createSulphurDioxideDensityCharacteristic(
1227 final HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
1228 return new SulphurDioxideDensityCharacteristic(
1229 getDoubleSupplier(taggedItem,
1230 taggedItem.getConfigurationAsDouble(HomekitTaggedItem.MIN_VALUE,
1231 SulphurDioxideDensityCharacteristic.DEFAULT_MIN_VALUE)),
1232 getSubscriber(taggedItem, SULPHUR_DIOXIDE_DENSITY, updater),
1233 getUnsubscriber(taggedItem, SULPHUR_DIOXIDE_DENSITY, updater));
1236 private static SwingModeCharacteristic createSwingModeCharacteristic(HomekitTaggedItem taggedItem,
1237 HomekitAccessoryUpdater updater) {
1238 var map = createMapping(taggedItem, SwingModeEnum.class);
1239 return new SwingModeCharacteristic(() -> getEnumFromItem(taggedItem, map, SwingModeEnum.SWING_DISABLED),
1240 (value) -> setValueFromEnum(taggedItem, value, map), getSubscriber(taggedItem, SWING_MODE, updater),
1241 getUnsubscriber(taggedItem, SWING_MODE, updater));
1244 private static TargetFanStateCharacteristic createTargetFanStateCharacteristic(HomekitTaggedItem taggedItem,
1245 HomekitAccessoryUpdater updater) {
1246 var map = createMapping(taggedItem, TargetFanStateEnum.class);
1247 return new TargetFanStateCharacteristic(() -> getEnumFromItem(taggedItem, map, TargetFanStateEnum.AUTO),
1248 (targetState) -> setValueFromEnum(taggedItem, targetState, map),
1249 getSubscriber(taggedItem, TARGET_FAN_STATE, updater),
1250 getUnsubscriber(taggedItem, TARGET_FAN_STATE, updater));
1253 private static TargetHeatingCoolingStateCharacteristic createTargetHeatingCoolingStateCharacteristic(
1254 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
1255 List<TargetHeatingCoolingStateEnum> validValues = new ArrayList<>();
1256 var map = createMapping(taggedItem, TargetHeatingCoolingStateEnum.class, validValues);
1257 return new TargetHeatingCoolingStateCharacteristic(validValues.toArray(new TargetHeatingCoolingStateEnum[0]),
1258 () -> getEnumFromItem(taggedItem, map, TargetHeatingCoolingStateEnum.OFF),
1259 (value) -> setValueFromEnum(taggedItem, value, map),
1260 getSubscriber(taggedItem, TARGET_HEATING_COOLING_STATE, updater),
1261 getUnsubscriber(taggedItem, TARGET_HEATING_COOLING_STATE, updater));
1264 private static TargetHorizontalTiltAngleCharacteristic createTargetHorizontalTiltAngleCharacteristic(
1265 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
1266 return new TargetHorizontalTiltAngleCharacteristic(getAngleSupplier(taggedItem, 0),
1267 setAngleConsumer(taggedItem), getSubscriber(taggedItem, TARGET_HORIZONTAL_TILT_ANGLE, updater),
1268 getUnsubscriber(taggedItem, TARGET_HORIZONTAL_TILT_ANGLE, updater));
1271 private static TargetMediaStateCharacteristic createTargetMediaStateCharacteristic(HomekitTaggedItem taggedItem,
1272 HomekitAccessoryUpdater updater) {
1273 var map = createMapping(taggedItem, TargetMediaStateEnum.class);
1274 return new TargetMediaStateCharacteristic(() -> getEnumFromItem(taggedItem, map, TargetMediaStateEnum.STOP),
1275 (value) -> setValueFromEnum(taggedItem, value, map),
1276 getSubscriber(taggedItem, TARGET_MEDIA_STATE, updater),
1277 getUnsubscriber(taggedItem, TARGET_MEDIA_STATE, updater));
1280 private static TargetRelativeHumidityCharacteristic createTargetRelativeHumidityCharacteristic(
1281 HomekitTaggedItem item, HomekitAccessoryUpdater updater) {
1282 return new TargetRelativeHumidityCharacteristic(getDoubleSupplier(item, 0), setDoubleConsumer(item),
1283 getSubscriber(item, TARGET_RELATIVE_HUMIDITY, updater),
1284 getUnsubscriber(item, TARGET_RELATIVE_HUMIDITY, updater));
1287 private static TargetTemperatureCharacteristic createTargetTemperatureCharacteristic(HomekitTaggedItem taggedItem,
1288 HomekitAccessoryUpdater updater) {
1289 double minValue = HomekitCharacteristicFactory.convertToCelsius(taggedItem.getConfigurationAsDouble(
1290 HomekitTaggedItem.MIN_VALUE, TargetTemperatureCharacteristic.DEFAULT_MIN_VALUE));
1291 double maxValue = HomekitCharacteristicFactory.convertToCelsius(taggedItem.getConfigurationAsDouble(
1292 HomekitTaggedItem.MAX_VALUE, TargetTemperatureCharacteristic.DEFAULT_MAX_VALUE));
1293 double step = getTemperatureStep(taggedItem, TargetTemperatureCharacteristic.DEFAULT_STEP);
1294 return new TargetTemperatureCharacteristic(minValue, maxValue, step,
1295 getTemperatureSupplier(taggedItem, minValue), setTemperatureConsumer(taggedItem),
1296 getSubscriber(taggedItem, TARGET_TEMPERATURE, updater),
1297 getUnsubscriber(taggedItem, TARGET_TEMPERATURE, updater));
1300 private static TargetTiltAngleCharacteristic createTargetTiltAngleCharacteristic(HomekitTaggedItem taggedItem,
1301 HomekitAccessoryUpdater updater) {
1302 return new TargetTiltAngleCharacteristic(getAngleSupplier(taggedItem, 0), setAngleConsumer(taggedItem),
1303 getSubscriber(taggedItem, TARGET_TILT_ANGLE, updater),
1304 getUnsubscriber(taggedItem, TARGET_TILT_ANGLE, updater));
1307 private static TargetVerticalTiltAngleCharacteristic createTargetVerticalTiltAngleCharacteristic(
1308 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
1309 return new TargetVerticalTiltAngleCharacteristic(getAngleSupplier(taggedItem, 0), setAngleConsumer(taggedItem),
1310 getSubscriber(taggedItem, TARGET_HORIZONTAL_TILT_ANGLE, updater),
1311 getUnsubscriber(taggedItem, TARGET_HORIZONTAL_TILT_ANGLE, updater));
1314 private static TargetVisibilityStateCharacteristic createTargetVisibilityStateCharacteristic(
1315 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
1316 var map = createMapping(taggedItem, TargetVisibilityStateEnum.class, true);
1317 return new TargetVisibilityStateCharacteristic(
1318 () -> getEnumFromItem(taggedItem, map, TargetVisibilityStateEnum.HIDDEN),
1319 (value) -> setValueFromEnum(taggedItem, value, map),
1320 getSubscriber(taggedItem, TARGET_VISIBILITY_STATE, updater),
1321 getUnsubscriber(taggedItem, TARGET_VISIBILITY_STATE, updater));
1324 private static TemperatureDisplayUnitCharacteristic createTemperatureDisplayUnitCharacteristic(
1325 HomekitTaggedItem taggedItem, HomekitAccessoryUpdater updater) {
1326 var map = createMapping(taggedItem, TemperatureDisplayUnitEnum.class, true);
1327 return new TemperatureDisplayUnitCharacteristic(
1328 () -> getEnumFromItem(taggedItem, map,
1329 useFahrenheit() ? TemperatureDisplayUnitEnum.FAHRENHEIT : TemperatureDisplayUnitEnum.CELSIUS),
1330 (value) -> setValueFromEnum(taggedItem, value, map),
1331 getSubscriber(taggedItem, TEMPERATURE_UNIT, updater),
1332 getUnsubscriber(taggedItem, TEMPERATURE_UNIT, updater));
1335 private static VOCDensityCharacteristic createVOCDensityCharacteristic(final HomekitTaggedItem taggedItem,
1336 HomekitAccessoryUpdater updater) {
1337 return new VOCDensityCharacteristic(
1338 taggedItem.getConfigurationAsDouble(HomekitTaggedItem.MIN_VALUE,
1339 VOCDensityCharacteristic.DEFAULT_MIN_VALUE),
1340 taggedItem.getConfigurationAsDouble(HomekitTaggedItem.MAX_VALUE,
1341 VOCDensityCharacteristic.DEFAULT_MAX_VALUE),
1342 taggedItem.getConfigurationAsDouble(HomekitTaggedItem.STEP, VOCDensityCharacteristic.DEFAULT_STEP),
1343 getDoubleSupplier(taggedItem,
1344 taggedItem.getConfigurationAsDouble(HomekitTaggedItem.MIN_VALUE,
1345 VOCDensityCharacteristic.DEFAULT_MIN_VALUE)),
1346 getSubscriber(taggedItem, VOC_DENSITY, updater), getUnsubscriber(taggedItem, VOC_DENSITY, updater));
1349 private static VolumeCharacteristic createVolumeCharacteristic(HomekitTaggedItem taggedItem,
1350 HomekitAccessoryUpdater updater) {
1351 return new VolumeCharacteristic(getIntSupplier(taggedItem, 0),
1352 (volume) -> ((NumberItem) taggedItem.getItem()).send(new DecimalType(volume)),
1353 getSubscriber(taggedItem, DURATION, updater), getUnsubscriber(taggedItem, DURATION, updater));
1356 private static VolumeSelectorCharacteristic createVolumeSelectorCharacteristic(HomekitTaggedItem taggedItem,
1357 HomekitAccessoryUpdater updater) {
1358 if (taggedItem.getItem() instanceof DimmerItem) {
1359 return new VolumeSelectorCharacteristic((value) -> taggedItem
1360 .send(value.equals(VolumeSelectorEnum.INCREMENT) ? IncreaseDecreaseType.INCREASE
1361 : IncreaseDecreaseType.DECREASE));
1363 var map = createMapping(taggedItem, VolumeSelectorEnum.class);
1364 return new VolumeSelectorCharacteristic((value) -> setValueFromEnum(taggedItem, value, map));
1368 private static VolumeControlTypeCharacteristic createVolumeControlTypeCharacteristic(HomekitTaggedItem taggedItem,
1369 HomekitAccessoryUpdater updater) {
1370 var map = createMapping(taggedItem, VolumeControlTypeEnum.class);
1371 return new VolumeControlTypeCharacteristic(() -> getEnumFromItem(taggedItem, map, VolumeControlTypeEnum.NONE),
1372 getSubscriber(taggedItem, VOLUME_CONTROL_TYPE, updater),
1373 getUnsubscriber(taggedItem, VOLUME_CONTROL_TYPE, updater));