mirror of
https://github.com/mtan93/SmartThingsPublic.git
synced 2026-03-08 05:31:56 +00:00
MSA-2158: Hue Motion Sensor device control
This commit is contained in:
@@ -0,0 +1,466 @@
|
|||||||
|
/**
|
||||||
|
* Copyright 2015 SmartThings
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except
|
||||||
|
* in compliance with the License. You may obtain a copy of the License at:
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software distributed under the License is distributed
|
||||||
|
* on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License
|
||||||
|
* for the specific language governing permissions and limitations under the License.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
metadata {
|
||||||
|
definition (name: "Hue Motion Sensor", namespace: "digitalgecko", author: "digitalgecko") {
|
||||||
|
|
||||||
|
|
||||||
|
capability "Motion Sensor"
|
||||||
|
capability "Configuration"
|
||||||
|
capability "Battery"
|
||||||
|
capability "Refresh"
|
||||||
|
capability "Temperature Measurement"
|
||||||
|
capability "Sensor"
|
||||||
|
capability "Illuminance Measurement" //0x0400
|
||||||
|
|
||||||
|
fingerprint profileId: "0104", inClusters: "0000,0001,0003,0406,0400,0402", outClusters: "0019", manufacturer: "Philips", model: "SML001", deviceJoinName: "Hue Motion Sensor"
|
||||||
|
}
|
||||||
|
|
||||||
|
preferences {
|
||||||
|
section {
|
||||||
|
input title: "Temperature Offset", description: "This feature allows you to correct any temperature variations by selecting an offset. Ex: If your sensor consistently reports a temp that's 5 degrees too warm, you'd enter '-5'. If 3 degrees too cold, enter '+3'.", displayDuringSetup: false, type: "paragraph", element: "paragraph"
|
||||||
|
input "tempOffset", "number", title: "Degrees", description: "Adjust temperature by this many degrees", range: "*..*", displayDuringSetup: false
|
||||||
|
}
|
||||||
|
section {
|
||||||
|
input title: "Luminance Offset", description: "This feature allows you to correct the luminance reading by selecting an offset. Enter a value such as 20 or -20 to adjust the luminance reading.", displayDuringSetup: false, type: "paragraph", element: "paragraph"
|
||||||
|
input "luxOffset", "number", title: "Lux", description: "Adjust luminance by this amount", range: "*..*", displayDuringSetup: false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
tiles(scale: 2) {
|
||||||
|
multiAttributeTile(name:"motion", type: "generic", width: 6, height: 4){
|
||||||
|
tileAttribute ("device.motion", key: "PRIMARY_CONTROL") {
|
||||||
|
attributeState "active", label:'motion', icon:"st.motion.motion.active", backgroundColor:"#53a7c0"
|
||||||
|
attributeState "inactive", label:'no motion', icon:"st.motion.motion.inactive", backgroundColor:"#ffffff"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
valueTile("temperature", "device.temperature", width: 2, height: 2) {
|
||||||
|
state("temperature", label:'${currentValue}°',
|
||||||
|
backgroundColors:[
|
||||||
|
[value: 31, color: "#153591"],
|
||||||
|
[value: 44, color: "#1e9cbb"],
|
||||||
|
[value: 59, color: "#90d2a7"],
|
||||||
|
[value: 74, color: "#44b621"],
|
||||||
|
[value: 84, color: "#f1d801"],
|
||||||
|
[value: 95, color: "#d04e00"],
|
||||||
|
[value: 96, color: "#bc2323"]
|
||||||
|
]
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
valueTile("battery", "device.battery", decoration: "flat", inactiveLabel: false, width: 2, height: 2) {
|
||||||
|
state "battery", label:'${currentValue}% battery', unit:""
|
||||||
|
}
|
||||||
|
valueTile("illuminance", "device.illuminance", width: 2, height: 2) {
|
||||||
|
state("illuminance", label:'${currentValue}', unit:"lux",
|
||||||
|
backgroundColors:[
|
||||||
|
[value: 9, color: "#767676"],
|
||||||
|
[value: 315, color: "#ffa81e"],
|
||||||
|
[value: 1000, color: "#fbd41b"]
|
||||||
|
]
|
||||||
|
)
|
||||||
|
}
|
||||||
|
standardTile("refresh", "device.refresh", inactiveLabel: false, decoration: "flat", width: 2, height: 2) {
|
||||||
|
state "default", label:"", action:"refresh.refresh", icon:"st.secondary.refresh"
|
||||||
|
}
|
||||||
|
standardTile("configure", "device.configure", inactiveLabel: false, decoration: "flat", width: 2, height: 2) {
|
||||||
|
state "default", label:"configure", action:"configure"
|
||||||
|
}
|
||||||
|
main "motion"
|
||||||
|
details(["motion","temperature","battery", "refresh","illuminance",'configure'])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// Parse incoming device messages to generate events
|
||||||
|
def parse(String description) {
|
||||||
|
def msg = zigbee.parse(description)
|
||||||
|
|
||||||
|
//log.warn "--"
|
||||||
|
//log.trace description
|
||||||
|
//log.debug msg
|
||||||
|
//def x = zigbee.parseDescriptionAsMap( description )
|
||||||
|
//log.error x
|
||||||
|
|
||||||
|
Map map = [:]
|
||||||
|
if (description?.startsWith('catchall:')) {
|
||||||
|
map = parseCatchAllMessage(description)
|
||||||
|
}
|
||||||
|
else if (description?.startsWith('temperature: ')) {
|
||||||
|
map = parseCustomMessage(description)
|
||||||
|
}
|
||||||
|
else if (description?.startsWith('illuminance: ')) {
|
||||||
|
map = parseCustomMessage(description)
|
||||||
|
}
|
||||||
|
// else if (description?.startsWith('zone status')) {
|
||||||
|
// //map = parseIasMessage(description)
|
||||||
|
// log.trace "zone status"
|
||||||
|
// }
|
||||||
|
|
||||||
|
def result = map ? createEvent(map) : null
|
||||||
|
|
||||||
|
if (description?.startsWith('enroll request')) {
|
||||||
|
List cmds = enrollResponse()
|
||||||
|
result = cmds?.collect { new physicalgraph.device.HubAction(it) }
|
||||||
|
}
|
||||||
|
else if (description?.startsWith('read attr -')) {
|
||||||
|
result = parseReportAttributeMessage(description).each { createEvent(it) }
|
||||||
|
}
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Refresh Function
|
||||||
|
*/
|
||||||
|
def refresh() {
|
||||||
|
log.debug "Refreshing Values"
|
||||||
|
|
||||||
|
def refreshCmds = []
|
||||||
|
refreshCmds +=zigbee.readAttribute(0x0001, 0x0020) // Read battery?
|
||||||
|
refreshCmds += zigbee.readAttribute(0x0402, 0x0000) // Read temp?
|
||||||
|
refreshCmds += zigbee.readAttribute(0x0400, 0x0000) // Read luminance?
|
||||||
|
refreshCmds += zigbee.readAttribute(0x0406, 0x0000) // Read motion?
|
||||||
|
|
||||||
|
return refreshCmds + enrollResponse()
|
||||||
|
|
||||||
|
}
|
||||||
|
/*
|
||||||
|
Configure Function
|
||||||
|
*/
|
||||||
|
def configure() {
|
||||||
|
|
||||||
|
// TODO : device watch?
|
||||||
|
|
||||||
|
String zigbeeId = swapEndianHex(device.hub.zigbeeId)
|
||||||
|
log.debug "Confuguring Reporting and Bindings."
|
||||||
|
|
||||||
|
|
||||||
|
def configCmds = []
|
||||||
|
configCmds += zigbee.batteryConfig()
|
||||||
|
configCmds += zigbee.temperatureConfig(60, 600) // Set temp reporting times // Confirmed
|
||||||
|
|
||||||
|
configCmds += zigbee.configureReporting(0x406,0x0000, 0x18, 30, 600, null) // motion // confirmed
|
||||||
|
|
||||||
|
|
||||||
|
// Data type is not 0x20 = 0x8D invalid data type Unsigned 8-bit integer
|
||||||
|
|
||||||
|
configCmds += zigbee.configureReporting(0x400,0x0000, 0x21, 60, 600, 0x20) // Set luminance reporting times?? maybe
|
||||||
|
return refresh() + configCmds
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
getMotionResult
|
||||||
|
*/
|
||||||
|
|
||||||
|
private Map getMotionResult(value) {
|
||||||
|
//log.trace "Motion : " + value
|
||||||
|
|
||||||
|
def descriptionText = value == "01" ? '{{ device.displayName }} detected motion':
|
||||||
|
'{{ device.displayName }} stopped detecting motion'
|
||||||
|
|
||||||
|
return [
|
||||||
|
name: 'motion',
|
||||||
|
value: value == "01" ? "active" : "inactive",
|
||||||
|
descriptionText: descriptionText,
|
||||||
|
translatable: true,
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
getTemperatureResult
|
||||||
|
*/
|
||||||
|
private Map getTemperatureResult(value) {
|
||||||
|
|
||||||
|
//log.trace "Temperature : " + value
|
||||||
|
if (tempOffset) {
|
||||||
|
def offset = tempOffset as int
|
||||||
|
def v = value as int
|
||||||
|
value = v + offset
|
||||||
|
}
|
||||||
|
def descriptionText = temperatureScale == 'C' ? '{{ device.displayName }} was {{ value }}°C':
|
||||||
|
'{{ device.displayName }} was {{ value }}°F'
|
||||||
|
|
||||||
|
return [
|
||||||
|
name: 'temperature',
|
||||||
|
value: value,
|
||||||
|
descriptionText: descriptionText,
|
||||||
|
translatable: true,
|
||||||
|
unit: temperatureScale
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
def getTemperature(value) {
|
||||||
|
def celsius = Integer.parseInt(value, 16).shortValue() / 100
|
||||||
|
if(getTemperatureScale() == "C"){
|
||||||
|
return Math.round(celsius)
|
||||||
|
} else {
|
||||||
|
return Math.round(celsiusToFahrenheit(celsius))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private Map getLuminanceResult(rawValue) {
|
||||||
|
log.debug "Luminance rawValue = ${rawValue}"
|
||||||
|
|
||||||
|
if (luxOffset) {
|
||||||
|
def offset = luxOffset as int
|
||||||
|
def v = rawValue as int
|
||||||
|
rawValue = v + offset
|
||||||
|
}
|
||||||
|
|
||||||
|
def result = [
|
||||||
|
name: 'illuminance',
|
||||||
|
value: '--',
|
||||||
|
translatable: true,
|
||||||
|
unit: 'lux'
|
||||||
|
]
|
||||||
|
|
||||||
|
result.value = rawValue as Integer
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
getBatteryResult
|
||||||
|
*/
|
||||||
|
//TODO: needs calibration
|
||||||
|
private Map getBatteryResult(rawValue) {
|
||||||
|
//log.debug "Battery rawValue = ${rawValue}"
|
||||||
|
|
||||||
|
def result = [
|
||||||
|
name: 'battery',
|
||||||
|
value: '--',
|
||||||
|
translatable: true
|
||||||
|
]
|
||||||
|
|
||||||
|
def volts = rawValue / 10
|
||||||
|
|
||||||
|
if (rawValue == 0 || rawValue == 255) {}
|
||||||
|
else {
|
||||||
|
if (volts > 3.5) {
|
||||||
|
result.descriptionText = "{{ device.displayName }} battery has too much power: (> 3.5) volts."
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
if (device.getDataValue("manufacturer") == "SmartThings") {
|
||||||
|
volts = rawValue // For the batteryMap to work the key needs to be an int
|
||||||
|
def batteryMap = [28:100, 27:100, 26:100, 25:90, 24:90, 23:70,
|
||||||
|
22:70, 21:50, 20:50, 19:30, 18:30, 17:15, 16:1, 15:0]
|
||||||
|
def minVolts = 15
|
||||||
|
def maxVolts = 28
|
||||||
|
|
||||||
|
if (volts < minVolts)
|
||||||
|
volts = minVolts
|
||||||
|
else if (volts > maxVolts)
|
||||||
|
volts = maxVolts
|
||||||
|
def pct = batteryMap[volts]
|
||||||
|
if (pct != null) {
|
||||||
|
result.value = pct
|
||||||
|
result.descriptionText = "{{ device.displayName }} battery was {{ value }}%"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
def minVolts = 2.1
|
||||||
|
def maxVolts = 3.0
|
||||||
|
def pct = (volts - minVolts) / (maxVolts - minVolts)
|
||||||
|
def roundedPct = Math.round(pct * 100)
|
||||||
|
if (roundedPct <= 0)
|
||||||
|
roundedPct = 1
|
||||||
|
result.value = Math.min(100, roundedPct)
|
||||||
|
result.descriptionText = "{{ device.displayName }} battery was {{ value }}%"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
/*
|
||||||
|
parseCustomMessage
|
||||||
|
*/
|
||||||
|
private Map parseCustomMessage(String description) {
|
||||||
|
Map resultMap = [:]
|
||||||
|
if (description?.startsWith('temperature: ')) {
|
||||||
|
def value = zigbee.parseHATemperatureValue(description, "temperature: ", getTemperatureScale())
|
||||||
|
resultMap = getTemperatureResult(value)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (description?.startsWith('illuminance: ')) {
|
||||||
|
log.warn "value: " + description.split(": ")[1]
|
||||||
|
log.warn "proc: " + value
|
||||||
|
|
||||||
|
def value = zigbee.lux( description.split(": ")[1] as Integer ) //zigbee.parseHAIlluminanceValue(description, "illuminance: ", getTemperatureScale())
|
||||||
|
resultMap = getLuminanceResult(value)
|
||||||
|
}
|
||||||
|
return resultMap
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
parseReportAttributeMessage
|
||||||
|
*/
|
||||||
|
private List parseReportAttributeMessage(String description) {
|
||||||
|
Map descMap = (description - "read attr - ").split(",").inject([:]) { map, param ->
|
||||||
|
def nameAndValue = param.split(":")
|
||||||
|
map += [(nameAndValue[0].trim()):nameAndValue[1].trim()]
|
||||||
|
}
|
||||||
|
|
||||||
|
List result = []
|
||||||
|
|
||||||
|
// Temperature
|
||||||
|
if (descMap.cluster == "0402" && descMap.attrId == "0000") {
|
||||||
|
def value = getTemperature(descMap.value)
|
||||||
|
result << getTemperatureResult(value)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Motion
|
||||||
|
else if (descMap.cluster == "0406" && descMap.attrId == "0000") {
|
||||||
|
result << getMotionResult(descMap.value)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Battery
|
||||||
|
else if (descMap.cluster == "0001" && descMap.attrId == "0020") {
|
||||||
|
result << getBatteryResult(Integer.parseInt(descMap.value, 16))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Luminance
|
||||||
|
else if (descMap.cluster == "0402" ) { //&& descMap.attrId == "0020") {
|
||||||
|
log.error "Luminance Response " + description
|
||||||
|
//result << getBatteryResult(Integer.parseInt(descMap.value, 16))
|
||||||
|
}
|
||||||
|
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
parseCatchAllMessage
|
||||||
|
*/
|
||||||
|
private Map parseCatchAllMessage(String description) {
|
||||||
|
Map resultMap = [:]
|
||||||
|
def cluster = zigbee.parse(description)
|
||||||
|
// log.debug cluster
|
||||||
|
if (shouldProcessMessage(cluster)) {
|
||||||
|
switch(cluster.clusterId) {
|
||||||
|
case 0x0001:
|
||||||
|
// 0x07 - configure reporting
|
||||||
|
if (cluster.command != 0x07) {
|
||||||
|
resultMap = getBatteryResult(cluster.data.last())
|
||||||
|
}
|
||||||
|
break
|
||||||
|
|
||||||
|
case 0x0400:
|
||||||
|
if (cluster.command == 0x07) { // Ignore Configure Reporting Response
|
||||||
|
if(cluster.data[0] == 0x00) {
|
||||||
|
log.trace "Luminance Reporting Configured"
|
||||||
|
sendEvent(name: "checkInterval", value: 60 * 12, displayed: false, data: [protocol: "zigbee", hubHardwareId: device.hub.hardwareID])
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
log.warn "Luminance REPORTING CONFIG FAILED- error code:${cluster.data[0]}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
log.debug "catchall : luminance" + cluster
|
||||||
|
resultMap = getLuminanceResult(cluster.data.last());
|
||||||
|
}
|
||||||
|
|
||||||
|
break
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
case 0x0402:
|
||||||
|
if (cluster.command == 0x07) {
|
||||||
|
if(cluster.data[0] == 0x00) {
|
||||||
|
log.trace "Temperature Reporting Configured"
|
||||||
|
sendEvent(name: "checkInterval", value: 60 * 12, displayed: false, data: [protocol: "zigbee", hubHardwareId: device.hub.hardwareID])
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
log.warn "TEMP REPORTING CONFIG FAILED- error code:${cluster.data[0]}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
// temp is last 2 data values. reverse to swap endian
|
||||||
|
String temp = cluster.data[-2..-1].reverse().collect { cluster.hex1(it) }.join()
|
||||||
|
def value = getTemperature(temp)
|
||||||
|
resultMap = getTemperatureResult(value)
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return resultMap
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean shouldProcessMessage(cluster) {
|
||||||
|
// 0x0B is default response indicating message got through
|
||||||
|
boolean ignoredMessage = cluster.profileId != 0x0104 ||
|
||||||
|
cluster.command == 0x0B ||
|
||||||
|
(cluster.data.size() > 0 && cluster.data.first() == 0x3e)
|
||||||
|
return !ignoredMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// This seems to be IAS Specific and not needed we are not really a motion sensor
|
||||||
|
def enrollResponse() {
|
||||||
|
// log.debug "Sending enroll response"
|
||||||
|
// String zigbeeEui = swapEndianHex(device.hub.zigbeeEui)
|
||||||
|
// [
|
||||||
|
// //Resending the CIE in case the enroll request is sent before CIE is written
|
||||||
|
// "zcl global write 0x500 0x10 0xf0 {${zigbeeEui}}", "delay 200",
|
||||||
|
// "send 0x${device.deviceNetworkId} 1 ${endpointId}", "delay 500",
|
||||||
|
// //Enroll Response
|
||||||
|
// "raw 0x500 {01 23 00 00 00}", "delay 200",
|
||||||
|
// "send 0x${device.deviceNetworkId} 1 1", "delay 200"
|
||||||
|
// ]
|
||||||
|
}
|
||||||
|
|
||||||
|
def configureHealthCheck() {
|
||||||
|
Integer hcIntervalMinutes = 12
|
||||||
|
refresh()
|
||||||
|
sendEvent(name: "checkInterval", value: hcIntervalMinutes * 60, displayed: false, data: [protocol: "zigbee", hubHardwareId: device.hub.hardwareID])
|
||||||
|
}
|
||||||
|
|
||||||
|
def updated() {
|
||||||
|
log.debug "in updated()"
|
||||||
|
configureHealthCheck()
|
||||||
|
}
|
||||||
|
|
||||||
|
def ping() {
|
||||||
|
return zigbee.onOffRefresh()
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
private getEndpointId() {
|
||||||
|
new BigInteger(device.endpointId, 16).toString()
|
||||||
|
}
|
||||||
|
|
||||||
|
private String swapEndianHex(String hex) {
|
||||||
|
reverseArray(hex.decodeHex()).encodeHex()
|
||||||
|
}
|
||||||
|
|
||||||
|
private byte[] reverseArray(byte[] array) {
|
||||||
|
int i = 0;
|
||||||
|
int j = array.length - 1;
|
||||||
|
byte tmp;
|
||||||
|
while (j > i) {
|
||||||
|
tmp = array[j];
|
||||||
|
array[j] = array[i];
|
||||||
|
array[i] = tmp;
|
||||||
|
j--;
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
return array
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user