mirror of
https://github.com/openhab/openhab-addons.git
synced 2025-01-25 14:55:55 +01:00
[mqtt.generic] Send ON/OFF for dimmer channels when so configured (#15929)
Similar to how UP/DOWN are processed for rollershutters. Signed-off-by: Cody Cutrer <cody@cutrer.us> Signed-off-by: Ciprian Pascu <contact@ciprianpascu.ro>
This commit is contained in:
parent
a1ca069b4d
commit
09ba1bb41d
@ -72,7 +72,7 @@ public class PercentageValue extends Value {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public PercentType parseCommand(Command command) throws IllegalArgumentException {
|
public Command parseCommand(Command command) throws IllegalArgumentException {
|
||||||
PercentType oldvalue = (state instanceof UnDefType) ? new PercentType() : (PercentType) state;
|
PercentType oldvalue = (state instanceof UnDefType) ? new PercentType() : (PercentType) state;
|
||||||
// Nothing do to -> We have received a percentage
|
// Nothing do to -> We have received a percentage
|
||||||
if (command instanceof PercentType percent) {
|
if (command instanceof PercentType percent) {
|
||||||
@ -105,8 +105,8 @@ public class PercentageValue extends Value {
|
|||||||
}
|
}
|
||||||
} else //
|
} else //
|
||||||
// On/Off equals 100 or 0 percent
|
// On/Off equals 100 or 0 percent
|
||||||
if (command instanceof OnOffType increaseDecreaseCommand) {
|
if (command instanceof OnOffType) {
|
||||||
return increaseDecreaseCommand == OnOffType.ON ? PercentType.HUNDRED : PercentType.ZERO;
|
return command;
|
||||||
} else//
|
} else//
|
||||||
// Increase or decrease by "step"
|
// Increase or decrease by "step"
|
||||||
if (command instanceof UpDownType upDownCommand) {
|
if (command instanceof UpDownType upDownCommand) {
|
||||||
@ -121,9 +121,9 @@ public class PercentageValue extends Value {
|
|||||||
// Check against custom on/off values
|
// Check against custom on/off values
|
||||||
if (command instanceof StringType) {
|
if (command instanceof StringType) {
|
||||||
if (onValue != null && command.toString().equals(onValue)) {
|
if (onValue != null && command.toString().equals(onValue)) {
|
||||||
return new PercentType(max);
|
return OnOffType.ON;
|
||||||
} else if (offValue != null && command.toString().equals(offValue)) {
|
} else if (offValue != null && command.toString().equals(offValue)) {
|
||||||
return new PercentType(min);
|
return OnOffType.OFF;
|
||||||
} else {
|
} else {
|
||||||
throw new IllegalStateException("Unable to parse " + command.toString() + " as a percent.");
|
throw new IllegalStateException("Unable to parse " + command.toString() + " as a percent.");
|
||||||
}
|
}
|
||||||
@ -135,17 +135,36 @@ public class PercentageValue extends Value {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String getMQTTpublishValue(Command command, @Nullable String pattern) {
|
public String getMQTTpublishValue(Command command, @Nullable String pattern) {
|
||||||
|
String formatPattern = pattern;
|
||||||
|
if (formatPattern == null) {
|
||||||
|
formatPattern = "%s";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (command instanceof OnOffType onOffCommand) {
|
||||||
|
if (onOffCommand == OnOffType.ON) {
|
||||||
|
if (onValue != null) {
|
||||||
|
command = new StringType(onValue);
|
||||||
|
} else {
|
||||||
|
command = PercentType.HUNDRED;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (offValue != null) {
|
||||||
|
command = new StringType(offValue);
|
||||||
|
} else {
|
||||||
|
command = PercentType.ZERO;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (command instanceof StringType) {
|
||||||
|
return command.format(formatPattern);
|
||||||
|
}
|
||||||
|
|
||||||
// Formula: From percentage to custom min/max: value*span/100+min
|
// Formula: From percentage to custom min/max: value*span/100+min
|
||||||
// Calculation need to happen with big decimals to either return a straight integer or a decimal depending on
|
// Calculation need to happen with big decimals to either return a straight integer or a decimal depending on
|
||||||
// the value.
|
// the value.
|
||||||
BigDecimal value = ((PercentType) command).toBigDecimal().multiply(span).divide(HUNDRED, MathContext.DECIMAL128)
|
BigDecimal value = ((PercentType) command).toBigDecimal().multiply(span).divide(HUNDRED, MathContext.DECIMAL128)
|
||||||
.add(min).stripTrailingZeros();
|
.add(min).stripTrailingZeros();
|
||||||
|
|
||||||
String formatPattern = pattern;
|
|
||||||
if (formatPattern == null) {
|
|
||||||
formatPattern = "%s";
|
|
||||||
}
|
|
||||||
|
|
||||||
return new DecimalType(value).format(formatPattern);
|
return new DecimalType(value).format(formatPattern);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -97,16 +97,12 @@
|
|||||||
<advanced>true</advanced>
|
<advanced>true</advanced>
|
||||||
</parameter>
|
</parameter>
|
||||||
<parameter name="on" type="text">
|
<parameter name="on" type="text">
|
||||||
<label>Custom On/Open Value</label>
|
<label>Custom ON Command</label>
|
||||||
<description>A number (like 1, 10) or a string (like "enabled") that is additionally recognised as on/open state. You
|
<description>A string (like "ON") that is sent to MQTT when an ON command is received, instead of an explicit 100%.</description>
|
||||||
can use this parameter for a second keyword, next to ON (OPEN respectively on a Contact).</description>
|
|
||||||
<default>1</default>
|
|
||||||
</parameter>
|
</parameter>
|
||||||
<parameter name="off" type="text">
|
<parameter name="off" type="text">
|
||||||
<label>Custom Off/Closed Value</label>
|
<label>Custom OFF Command</label>
|
||||||
<description>A number (like 0, -10) or a string (like "disabled") that is additionally recognised as off/closed
|
<description>A string (like "OFF") that is sent to MQTT when an OFF command is received, instead of an explicit 0%.</description>
|
||||||
state. You can use this parameter for a second keyword, next to OFF (CLOSED respectively on a Contact).</description>
|
|
||||||
<default>0</default>
|
|
||||||
</parameter>
|
</parameter>
|
||||||
</config-description>
|
</config-description>
|
||||||
</config-description:config-descriptions>
|
</config-description:config-descriptions>
|
||||||
|
@ -77,10 +77,10 @@ thing-type.config.mqtt.dimmer_channel.max.label = Absolute Maximum
|
|||||||
thing-type.config.mqtt.dimmer_channel.max.description = This configuration represents the maximum of the allowed range. For a percentage channel that equals one-hundred percent.
|
thing-type.config.mqtt.dimmer_channel.max.description = This configuration represents the maximum of the allowed range. For a percentage channel that equals one-hundred percent.
|
||||||
thing-type.config.mqtt.dimmer_channel.min.label = Absolute Minimum
|
thing-type.config.mqtt.dimmer_channel.min.label = Absolute Minimum
|
||||||
thing-type.config.mqtt.dimmer_channel.min.description = This configuration represents the minimum of the allowed range. For a percentage channel that equals zero percent.
|
thing-type.config.mqtt.dimmer_channel.min.description = This configuration represents the minimum of the allowed range. For a percentage channel that equals zero percent.
|
||||||
thing-type.config.mqtt.dimmer_channel.off.label = Custom Off/Closed Value
|
thing-type.config.mqtt.dimmer_channel.off.label = Custom OFF Command
|
||||||
thing-type.config.mqtt.dimmer_channel.off.description = A number (like 0, -10) or a string (like "disabled") that is additionally recognised as off/closed state. You can use this parameter for a second keyword, next to OFF (CLOSED respectively on a Contact).
|
thing-type.config.mqtt.dimmer_channel.off.description = A string (like "OFF") that is sent to MQTT when an OFF command is received, instead of an explicit 0%.
|
||||||
thing-type.config.mqtt.dimmer_channel.on.label = Custom On/Open Value
|
thing-type.config.mqtt.dimmer_channel.on.label = Custom ON Command
|
||||||
thing-type.config.mqtt.dimmer_channel.on.description = A number (like 1, 10) or a string (like "enabled") that is additionally recognised as on/open state. You can use this parameter for a second keyword, next to ON (OPEN respectively on a Contact).
|
thing-type.config.mqtt.dimmer_channel.on.description = A string (like "ON") that is sent to MQTT when an ON command is received, instead of an explicit 100%.
|
||||||
thing-type.config.mqtt.dimmer_channel.postCommand.label = Is Command
|
thing-type.config.mqtt.dimmer_channel.postCommand.label = Is Command
|
||||||
thing-type.config.mqtt.dimmer_channel.postCommand.description = If the received MQTT value should not only update the state of linked items, but command them, enable this option.
|
thing-type.config.mqtt.dimmer_channel.postCommand.description = If the received MQTT value should not only update the state of linked items, but command them, enable this option.
|
||||||
thing-type.config.mqtt.dimmer_channel.qos.label = QoS
|
thing-type.config.mqtt.dimmer_channel.qos.label = QoS
|
||||||
@ -170,6 +170,8 @@ thing-type.config.mqtt.string_channel.formatBeforePublish.label = Outgoing Value
|
|||||||
thing-type.config.mqtt.string_channel.formatBeforePublish.description = Format a value before it is published to the MQTT broker. The default is to just pass the channel/item state. If you want to apply a prefix, say "MYCOLOR,", you would use "MYCOLOR,%s". If you want to adjust the precision of a number to for example 4 digits, you would use "%.4f".
|
thing-type.config.mqtt.string_channel.formatBeforePublish.description = Format a value before it is published to the MQTT broker. The default is to just pass the channel/item state. If you want to apply a prefix, say "MYCOLOR,", you would use "MYCOLOR,%s". If you want to adjust the precision of a number to for example 4 digits, you would use "%.4f".
|
||||||
thing-type.config.mqtt.string_channel.group.transformations.label = Transform Values
|
thing-type.config.mqtt.string_channel.group.transformations.label = Transform Values
|
||||||
thing-type.config.mqtt.string_channel.group.transformations.description = These configuration parameters allow you to alter a value before it is published to MQTT or before a received value is assigned to an item.
|
thing-type.config.mqtt.string_channel.group.transformations.description = These configuration parameters allow you to alter a value before it is published to MQTT or before a received value is assigned to an item.
|
||||||
|
thing-type.config.mqtt.string_channel.nullValue.label = NULL Value
|
||||||
|
thing-type.config.mqtt.string_channel.nullValue.description = If the received MQTT value matches this, treat it as NULL.
|
||||||
thing-type.config.mqtt.string_channel.postCommand.label = Is Command
|
thing-type.config.mqtt.string_channel.postCommand.label = Is Command
|
||||||
thing-type.config.mqtt.string_channel.postCommand.description = If the received MQTT value should not only update the state of linked items, but command them, enable this option.
|
thing-type.config.mqtt.string_channel.postCommand.description = If the received MQTT value should not only update the state of linked items, but command them, enable this option.
|
||||||
thing-type.config.mqtt.string_channel.qos.label = QoS
|
thing-type.config.mqtt.string_channel.qos.label = QoS
|
||||||
@ -185,8 +187,6 @@ thing-type.config.mqtt.string_channel.transformationPattern.label = Incoming Val
|
|||||||
thing-type.config.mqtt.string_channel.transformationPattern.description = Applies transformations to an incoming MQTT topic value. A transformation example for a received JSON would be "JSONPATH:$.device.status.temperature" for a json {device: {status: { temperature: 23.2 }}}. You can chain transformations by separating them with the intersection character ∩.
|
thing-type.config.mqtt.string_channel.transformationPattern.description = Applies transformations to an incoming MQTT topic value. A transformation example for a received JSON would be "JSONPATH:$.device.status.temperature" for a json {device: {status: { temperature: 23.2 }}}. You can chain transformations by separating them with the intersection character ∩.
|
||||||
thing-type.config.mqtt.string_channel.transformationPatternOut.label = Outgoing Value Transformation
|
thing-type.config.mqtt.string_channel.transformationPatternOut.label = Outgoing Value Transformation
|
||||||
thing-type.config.mqtt.string_channel.transformationPatternOut.description = Applies a transformation before publishing a MQTT topic value. Transformations are specialised in extracting a value, but some transformations like the MAP one could be useful.
|
thing-type.config.mqtt.string_channel.transformationPatternOut.description = Applies a transformation before publishing a MQTT topic value. Transformations are specialised in extracting a value, but some transformations like the MAP one could be useful.
|
||||||
thing-type.config.mqtt.string_channel.nullValue.label = NULL Value
|
|
||||||
thing-type.config.mqtt.string_channel.nullValue.description = If the received MQTT value matches this, treat it as NULL.
|
|
||||||
thing-type.config.mqtt.switch_channel.commandTopic.label = MQTT Command Topic
|
thing-type.config.mqtt.switch_channel.commandTopic.label = MQTT Command Topic
|
||||||
thing-type.config.mqtt.switch_channel.commandTopic.description = An MQTT topic that this thing will send a command to. If not set, this will be a read-only switch.
|
thing-type.config.mqtt.switch_channel.commandTopic.description = An MQTT topic that this thing will send a command to. If not set, this will be a read-only switch.
|
||||||
thing-type.config.mqtt.switch_channel.formatBeforePublish.label = Outgoing Value Format
|
thing-type.config.mqtt.switch_channel.formatBeforePublish.label = Outgoing Value Format
|
||||||
|
@ -310,8 +310,10 @@ public class ValueTests {
|
|||||||
assertThat(v.parseCommand(new DecimalType(10.0)), is(PercentType.ZERO));
|
assertThat(v.parseCommand(new DecimalType(10.0)), is(PercentType.ZERO));
|
||||||
assertThat(v.getMQTTpublishValue(PercentType.ZERO, null), is("10"));
|
assertThat(v.getMQTTpublishValue(PercentType.ZERO, null), is("10"));
|
||||||
|
|
||||||
assertThat(v.parseCommand(OnOffType.ON), is(PercentType.HUNDRED));
|
assertThat(v.parseCommand(OnOffType.ON), is(OnOffType.ON));
|
||||||
assertThat(v.parseCommand(OnOffType.OFF), is(PercentType.ZERO));
|
assertThat(v.getMQTTpublishValue(OnOffType.ON, null), is("110"));
|
||||||
|
assertThat(v.parseCommand(OnOffType.OFF), is(OnOffType.OFF));
|
||||||
|
assertThat(v.getMQTTpublishValue(OnOffType.OFF, null), is("10"));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@ -332,8 +334,10 @@ public class ValueTests {
|
|||||||
public void percentCustomOnOff() {
|
public void percentCustomOnOff() {
|
||||||
PercentageValue v = new PercentageValue(new BigDecimal("0.0"), new BigDecimal("100.0"), new BigDecimal("1.0"),
|
PercentageValue v = new PercentageValue(new BigDecimal("0.0"), new BigDecimal("100.0"), new BigDecimal("1.0"),
|
||||||
"on", "off");
|
"on", "off");
|
||||||
assertThat(v.parseCommand(new StringType("on")), is(PercentType.HUNDRED));
|
assertThat(v.parseCommand(new StringType("on")), is(OnOffType.ON));
|
||||||
assertThat(v.parseCommand(new StringType("off")), is(PercentType.ZERO));
|
assertThat(v.getMQTTpublishValue(OnOffType.ON, "%s"), is("on"));
|
||||||
|
assertThat(v.parseCommand(new StringType("off")), is(OnOffType.OFF));
|
||||||
|
assertThat(v.getMQTTpublishValue(OnOffType.OFF, "%s"), is("off"));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@ -342,7 +346,7 @@ public class ValueTests {
|
|||||||
null, null);
|
null, null);
|
||||||
assertThat(v.parseCommand(new DecimalType(1.0)), is(PercentType.HUNDRED));
|
assertThat(v.parseCommand(new DecimalType(1.0)), is(PercentType.HUNDRED));
|
||||||
assertThat(v.parseCommand(new DecimalType(0.1)), is(PercentType.ZERO));
|
assertThat(v.parseCommand(new DecimalType(0.1)), is(PercentType.ZERO));
|
||||||
PercentType command = v.parseCommand(new DecimalType(0.2));
|
PercentType command = (PercentType) v.parseCommand(new DecimalType(0.2));
|
||||||
assertEquals(command.floatValue(), 11.11f, 0.01f);
|
assertEquals(command.floatValue(), 11.11f, 0.01f);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -352,26 +356,26 @@ public class ValueTests {
|
|||||||
null, null);
|
null, null);
|
||||||
|
|
||||||
// Normal operation.
|
// Normal operation.
|
||||||
PercentType command = v.parseCommand(new DecimalType("6.0"));
|
PercentType command = (PercentType) v.parseCommand(new DecimalType("6.0"));
|
||||||
assertEquals(command.floatValue(), 50.0f, 0.01f);
|
assertEquals(command.floatValue(), 50.0f, 0.01f);
|
||||||
v.update(command);
|
v.update(command);
|
||||||
command = v.parseCommand(IncreaseDecreaseType.INCREASE);
|
command = (PercentType) v.parseCommand(IncreaseDecreaseType.INCREASE);
|
||||||
assertEquals(command.floatValue(), 55.0f, 0.01f);
|
assertEquals(command.floatValue(), 55.0f, 0.01f);
|
||||||
command = v.parseCommand(IncreaseDecreaseType.DECREASE);
|
command = (PercentType) v.parseCommand(IncreaseDecreaseType.DECREASE);
|
||||||
assertEquals(command.floatValue(), 45.0f, 0.01f);
|
assertEquals(command.floatValue(), 45.0f, 0.01f);
|
||||||
|
|
||||||
// Lower limit.
|
// Lower limit.
|
||||||
command = v.parseCommand(new DecimalType("1.1"));
|
command = (PercentType) v.parseCommand(new DecimalType("1.1"));
|
||||||
assertEquals(command.floatValue(), 1.0f, 0.01f);
|
assertEquals(command.floatValue(), 1.0f, 0.01f);
|
||||||
v.update(command);
|
v.update(command);
|
||||||
command = v.parseCommand(IncreaseDecreaseType.DECREASE);
|
command = (PercentType) v.parseCommand(IncreaseDecreaseType.DECREASE);
|
||||||
assertEquals(command.floatValue(), 0.0f, 0.01f);
|
assertEquals(command.floatValue(), 0.0f, 0.01f);
|
||||||
|
|
||||||
// Upper limit.
|
// Upper limit.
|
||||||
command = v.parseCommand(new DecimalType("10.8"));
|
command = (PercentType) v.parseCommand(new DecimalType("10.8"));
|
||||||
assertEquals(command.floatValue(), 98.0f, 0.01f);
|
assertEquals(command.floatValue(), 98.0f, 0.01f);
|
||||||
v.update(command);
|
v.update(command);
|
||||||
command = v.parseCommand(IncreaseDecreaseType.INCREASE);
|
command = (PercentType) v.parseCommand(IncreaseDecreaseType.INCREASE);
|
||||||
assertEquals(command.floatValue(), 100.0f, 0.01f);
|
assertEquals(command.floatValue(), 100.0f, 0.01f);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -381,26 +385,26 @@ public class ValueTests {
|
|||||||
null, null);
|
null, null);
|
||||||
|
|
||||||
// Normal operation.
|
// Normal operation.
|
||||||
PercentType command = v.parseCommand(new DecimalType("6.0"));
|
PercentType command = (PercentType) v.parseCommand(new DecimalType("6.0"));
|
||||||
assertEquals(command.floatValue(), 50.0f, 0.01f);
|
assertEquals(command.floatValue(), 50.0f, 0.01f);
|
||||||
v.update(command);
|
v.update(command);
|
||||||
command = v.parseCommand(UpDownType.UP);
|
command = (PercentType) v.parseCommand(UpDownType.UP);
|
||||||
assertEquals(command.floatValue(), 55.0f, 0.01f);
|
assertEquals(command.floatValue(), 55.0f, 0.01f);
|
||||||
command = v.parseCommand(UpDownType.DOWN);
|
command = (PercentType) v.parseCommand(UpDownType.DOWN);
|
||||||
assertEquals(command.floatValue(), 45.0f, 0.01f);
|
assertEquals(command.floatValue(), 45.0f, 0.01f);
|
||||||
|
|
||||||
// Lower limit.
|
// Lower limit.
|
||||||
command = v.parseCommand(new DecimalType("1.1"));
|
command = (PercentType) v.parseCommand(new DecimalType("1.1"));
|
||||||
assertEquals(command.floatValue(), 1.0f, 0.01f);
|
assertEquals(command.floatValue(), 1.0f, 0.01f);
|
||||||
v.update(command);
|
v.update(command);
|
||||||
command = v.parseCommand(UpDownType.DOWN);
|
command = (PercentType) v.parseCommand(UpDownType.DOWN);
|
||||||
assertEquals(command.floatValue(), 0.0f, 0.01f);
|
assertEquals(command.floatValue(), 0.0f, 0.01f);
|
||||||
|
|
||||||
// Upper limit.
|
// Upper limit.
|
||||||
command = v.parseCommand(new DecimalType("10.8"));
|
command = (PercentType) v.parseCommand(new DecimalType("10.8"));
|
||||||
assertEquals(command.floatValue(), 98.0f, 0.01f);
|
assertEquals(command.floatValue(), 98.0f, 0.01f);
|
||||||
v.update(command);
|
v.update(command);
|
||||||
command = v.parseCommand(UpDownType.UP);
|
command = (PercentType) v.parseCommand(UpDownType.UP);
|
||||||
assertEquals(command.floatValue(), 100.0f, 0.01f);
|
assertEquals(command.floatValue(), 100.0f, 0.01f);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -255,13 +255,15 @@ public class JSONSchemaLight extends AbstractRawSchemaLight {
|
|||||||
|
|
||||||
boolean off = false;
|
boolean off = false;
|
||||||
if (jsonState.state != null) {
|
if (jsonState.state != null) {
|
||||||
onOffValue.update(onOffValue.parseCommand(new StringType(jsonState.state)));
|
onOffValue.update((State) onOffValue.parseMessage(new StringType(jsonState.state)));
|
||||||
off = onOffValue.getChannelState().equals(OnOffType.OFF);
|
off = onOffValue.getChannelState().equals(OnOffType.OFF);
|
||||||
if (brightnessValue.getChannelState() instanceof UnDefType) {
|
if (onOffValue.getChannelState() instanceof OnOffType onOffState) {
|
||||||
brightnessValue.update(off ? PercentType.ZERO : PercentType.HUNDRED);
|
if (brightnessValue.getChannelState() instanceof UnDefType) {
|
||||||
}
|
brightnessValue.update(Objects.requireNonNull(onOffState.as(PercentType.class)));
|
||||||
if (colorValue.getChannelState() instanceof UnDefType) {
|
}
|
||||||
colorValue.update(off ? HSBType.BLACK : HSBType.WHITE);
|
if (colorValue.getChannelState() instanceof UnDefType) {
|
||||||
|
colorValue.update(Objects.requireNonNull(onOffState.as(PercentType.class)));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user