/**
* Copyright (c) 2010-2016 by the respective copyright holders.
*
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/epl-v10.html
*/
package org.openhab.binding.rfxcom.internal.messages;
import java.util.Arrays;
import java.util.List;
import javax.xml.bind.DatatypeConverter;
import org.openhab.binding.rfxcom.RFXComValueSelector;
import org.openhab.binding.rfxcom.internal.RFXComException;
import org.openhab.core.library.items.ContactItem;
import org.openhab.core.library.items.DateTimeItem;
import org.openhab.core.library.items.NumberItem;
import org.openhab.core.library.items.StringItem;
import org.openhab.core.library.items.SwitchItem;
import org.openhab.core.library.types.DateTimeType;
import org.openhab.core.library.types.DecimalType;
import org.openhab.core.library.types.OnOffType;
import org.openhab.core.library.types.OpenClosedType;
import org.openhab.core.library.types.StringType;
import org.openhab.core.types.State;
import org.openhab.core.types.Type;
import org.openhab.core.types.UnDefType;
/**
* RFXCOM data class for Security1 message.
* (i.e. X10 Security, Visonic PowerCode, Meiantech, etc.)
*
* @author David Kalff
* @since 1.2.0
*/
public class RFXComSecurity1Message extends RFXComBaseMessage {
public enum SubType {
X10_SECURITY(0),
X10_SECURITY_MOTION(1),
X10_SECURITY_REMOTE(2),
KD101(3),
VISONIC_POWERCODE_SENSOR_PRIMARY_CONTACT(4),
VISONIC_POWERCODE_MOTION(5),
VISONIC_CODESECURE(6),
VISONIC_POWERCODE_SENSOR_AUX_CONTACT(7),
MEIANTECH(8),
SA30(9),
UNKNOWN(255);
private final int subType;
SubType(int subType) {
this.subType = subType;
}
SubType(byte subType) {
this.subType = subType;
}
public byte toByte() {
return (byte) subType;
}
public static SubType fromByte(int input) {
for (SubType c : SubType.values()) {
if (c.subType == input) {
return c;
}
}
return SubType.UNKNOWN;
}
}
public enum Status {
NORMAL(0),
NORMAL_DELAYED(1),
ALARM(2),
ALARM_DELAYED(3),
MOTION(4),
NO_MOTION(5),
PANIC(6),
END_PANIC(7),
IR(8),
ARM_AWAY(9),
ARM_AWAY_DELAYED(10),
ARM_HOME(11),
ARM_HOME_DELAYED(12),
DISARM(13),
LIGHT_1_OFF(16),
LIGHT_1_ON(17),
LIGHT_2_OFF(18),
LIGHT_2_ON(19),
DARK_DETECTED(20),
LIGHT_DETECTED(21),
BATLOW(22),
PAIR_KD101(23),
NORMAL_TAMPER(128),
NORMAL_DELAYED_TAMPER(129),
ALARM_TAMPER(130),
ALARM_DELAYED_TAMPER(131),
MOTION_TAMPER(132),
NO_MOTION_TAMPER(133),
UNKNOWN(255);
private final int status;
Status(int status) {
this.status = status;
}
Status(byte status) {
this.status = status;
}
public byte toByte() {
return (byte) status;
}
public static Status fromByte(int input) {
for (Status status : Status.values()) {
if (status.status == input) {
return status;
}
}
return Status.UNKNOWN;
}
}
/* Added item for ContactTypes */
public enum Contact {
NORMAL(0),
NORMAL_DELAYED(1),
ALARM(2),
ALARM_DELAYED(3),
NORMAL_TAMPER(128),
NORMAL_DELAYED_TAMPER(129),
ALARM_TAMPER(130),
ALARM_DELAYED_TAMPER(131),
UNKNOWN(255);
private final int contact;
Contact(int contact) {
this.contact = contact;
}
Contact(byte contact) {
this.contact = contact;
}
public byte toByte() {
return (byte) contact;
}
public static Contact fromByte(int input) {
for (Contact contact : Contact.values()) {
if (contact.contact == input) {
return contact;
}
}
return Contact.UNKNOWN;
}
}
/* Added item for MotionTypes */
public enum Motion {
MOTION(4),
NO_MOTION(5),
MOTION_TAMPER(132),
NO_MOTION_TAMPER(133),
UNKNOWN(255);
private final int motion;
Motion(int motion) {
this.motion = motion;
}
Motion(byte motion) {
this.motion = motion;
}
public byte toByte() {
return (byte) motion;
}
public static Motion fromByte(int input) {
for (Motion motion : Motion.values()) {
if (motion.motion == input) {
return motion;
}
}
return Motion.UNKNOWN;
}
}
private final static List<RFXComValueSelector> supportedValueSelectors = Arrays.asList(RFXComValueSelector.RAW_DATA,
RFXComValueSelector.SIGNAL_LEVEL, RFXComValueSelector.BATTERY_LEVEL, RFXComValueSelector.STATUS,
RFXComValueSelector.CONTACT, RFXComValueSelector.MOTION);
public SubType subType = SubType.UNKNOWN;
public int sensorId = 0;
public Status status = Status.UNKNOWN;
public byte batteryLevel = 0;
public byte signalLevel = 0;
public Contact contact = Contact.UNKNOWN;
public Motion motion = Motion.UNKNOWN;
public RFXComSecurity1Message() {
packetType = PacketType.SECURITY1;
}
public RFXComSecurity1Message(byte[] data) {
encodeMessage(data);
}
@Override
public String toString() {
String str = "";
str += super.toString();
str += "\n - Sub type = " + subType;
str += "\n - Id = " + sensorId;
str += "\n - Status = " + status;
str += "\n - Battery level = " + batteryLevel;
str += "\n - Signal level = " + signalLevel;
return str;
}
@Override
public void encodeMessage(byte[] data) {
super.encodeMessage(data);
subType = SubType.fromByte(super.subType);
sensorId = (data[4] & 0xFF) << 16 | (data[5] & 0xFF) << 8 | (data[6] & 0xFF);
status = Status.fromByte(data[7]);
batteryLevel = (byte) ((data[8] & 0xF0) >> 4);
signalLevel = (byte) (data[8] & 0x0F);
contact = Contact.fromByte(data[7]);
motion = Motion.fromByte(data[7]);
}
@Override
public byte[] decodeMessage() {
byte[] data = new byte[9];
data[0] = 0x08;
data[1] = RFXComBaseMessage.PacketType.SECURITY1.toByte();
data[2] = subType.toByte();
data[3] = seqNbr;
data[4] = (byte) ((sensorId >> 16) & 0xFF);
data[5] = (byte) ((sensorId >> 8) & 0xFF);
data[6] = (byte) (sensorId & 0xFF);
data[7] = status.toByte();
data[8] = (byte) (((batteryLevel & 0x0F) << 4) | (signalLevel & 0x0F));
return data;
}
@Override
public String generateDeviceId() {
return String.valueOf(sensorId);
}
@Override
public State convertToState(RFXComValueSelector valueSelector) throws RFXComException {
org.openhab.core.types.State state = UnDefType.UNDEF;
if (valueSelector.getItemClass() == SwitchItem.class) {
if (valueSelector == RFXComValueSelector.MOTION) {
switch (status) {
case MOTION:
state = OnOffType.ON;
break;
case NO_MOTION:
state = OnOffType.OFF;
break;
default:
break;
}
} else {
throw new RFXComException("Can't convert " + valueSelector + " to SwitchItem");
}
} else if (valueSelector.getItemClass() == ContactItem.class) {
if (valueSelector == RFXComValueSelector.CONTACT) {
switch (status) {
case NORMAL:
state = OpenClosedType.CLOSED;
break;
case NORMAL_DELAYED:
state = OpenClosedType.CLOSED;
break;
case ALARM:
state = OpenClosedType.OPEN;
break;
case ALARM_DELAYED:
state = OpenClosedType.OPEN;
break;
default:
break;
}
} else {
throw new RFXComException("Can't convert " + valueSelector + " to ContactItem");
}
} else if (valueSelector.getItemClass() == StringItem.class) {
if (valueSelector == RFXComValueSelector.RAW_DATA) {
state = new StringType(DatatypeConverter.printHexBinary(rawMessage));
} else if (valueSelector == RFXComValueSelector.STATUS) {
state = new StringType(status.toString());
} else {
throw new RFXComException("Can't convert " + valueSelector + " to StringItem");
}
} else if (valueSelector.getItemClass() == NumberItem.class) {
if (valueSelector == RFXComValueSelector.SIGNAL_LEVEL) {
state = new DecimalType(signalLevel);
} else if (valueSelector == RFXComValueSelector.BATTERY_LEVEL) {
state = new DecimalType(batteryLevel);
} else {
throw new RFXComException("Can't convert " + valueSelector + " to StringItem");
}
} else if (valueSelector.getItemClass() == DateTimeItem.class) {
state = new DateTimeType();
} else {
throw new RFXComException("Can't convert " + valueSelector + " to " + valueSelector.getItemClass());
}
return state;
}
@Override
public void convertFromState(RFXComValueSelector valueSelector, String id, Object subType, Type type,
byte seqNumber) throws RFXComException {
this.subType = ((SubType) subType);
seqNbr = seqNumber;
String ids = id;
sensorId = Integer.parseInt(ids);
switch (valueSelector) {
case COMMAND:
if ((type instanceof OnOffType) && (subType == SubType.X10_SECURITY_REMOTE)) {
status = (type == OnOffType.ON ? Status.ARM_AWAY_DELAYED : Status.DISARM);
} else {
throw new RFXComException("Can't convert " + type + " to Command");
}
break;
case STATUS:
if (type instanceof StringType) {
status = Status.valueOf(type.toString());
} else {
throw new RFXComException("Can't convert " + type + " to Status");
}
break;
default:
throw new RFXComException("Can't convert " + type + " to " + valueSelector);
}
}
@Override
public Object convertSubType(String subType) throws RFXComException {
for (SubType s : SubType.values()) {
if (s.toString().equals(subType)) {
return s;
}
}
throw new RFXComException("Unknown sub type " + subType);
}
@Override
public List<RFXComValueSelector> getSupportedValueSelectors() throws RFXComException {
return supportedValueSelectors;
}
}