Compare commits

...

7 Commits

Author SHA1 Message Date
dingchengus
e9b6a6531c MSA-2358: Bosch Pro-Grade ZigBee Wireless Motion Detector ISW-ZPR1-WP13 2017-11-11 15:49:59 -08:00
Vinay Rao
28b46e37ff Merge pull request #2505 from SmartThingsCommunity/staging
Rolling down staging to master
2017-11-10 14:54:36 -08:00
Ingvar Marstorp
0d08d7b201 DVCSMP-3374 Zen Thermostat has no settings for "Heat only" or "Cool only" (#2503)
A customer reported on PR 2296;
"My Zen can only Heat (10A setting I think). There is no option for "Off, Heat" in the thermostat modes settings UI."
Adding ["off", "heat"] and ["off", "cool"] to settings.
2017-11-10 11:47:51 -08:00
Vinay Rao
739418c12e Merge pull request #2501 from workingmonk/bug/multi_tiles
DVCSMP-3384 Add tile type to multi attribute tile for devices
2017-11-09 12:46:07 -08:00
Vinay Rao
9172abf71b DVCSMP-3384 Add tile type to multi attribute tile for devices 2017-11-09 12:42:35 -08:00
Vinay Rao
0ebed2894c DVCSMP-3283 Adding blacklisted DTH to catch bad devices (#2371)
* DVCSMP-3283 Adding blacklisted DTH to catch bad devices

* Adding correct support link
2017-11-07 16:23:12 -08:00
Vinay Rao
fd95a98f08 Merge pull request #2497 from SmartThingsCommunity/master
Rolling up master to staging
2017-11-07 13:38:27 -08:00
7 changed files with 282 additions and 13 deletions

View File

@@ -40,7 +40,7 @@ metadata {
}
tiles {
multiAttributeTile(name: "rich-control") {
multiAttributeTile(name: "rich-control", type: "generic", width: 6, height: 4, canChangeIcon: true) {
tileAttribute("device.button", key: "PRIMARY_CONTROL") {
attributeState "default", label: ' ', action: "", icon: "st.unknown.zwave.remote-controller", backgroundColor: "#ffffff"
}
@@ -201,4 +201,4 @@ private void createChildDevices() {
[completedSetup: true, label: "${device.displayName} button ${i}",
isComponent: true, componentName: "button$i", componentLabel: "Button $i"])
}
}
}

View File

@@ -39,7 +39,7 @@ metadata {
status "wakeup": "command: 8407, payload: "
}
tiles(scale: 2) {
multiAttributeTile(name: "rich-control") {
multiAttributeTile(name: "rich-control", type: "generic", width: 6, height: 4, canChangeIcon: true) {
tileAttribute("device.button", key: "PRIMARY_CONTROL") {
attributeState "default", label: ' ', action: "", icon: "st.unknown.zwave.remote-controller", backgroundColor: "#ffffff"
}

View File

@@ -0,0 +1,45 @@
/**
* Blacklisted Device
*
* Copyright 2017 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: "Blacklisted Device", namespace: "smartthings", author: "SmartThings") {
fingerprint profileId: "0104", inClusters: "0000, 0009, 000A, 0101, FC00, 0001", manufacturer:"Yale", model:"Cap", deviceJoinName: "Blacklisted Yale Lock"
fingerprint inClusters: "0000, 0003, 0101", manufacturer:"Kwikset", model:"Smartcode", deviceJoinName: "Blacklisted Kwikset Lock"
}
tiles(scale: 2) {
multiAttributeTile(name:"alarm", type: "generic", width: 3, height: 4){
tileAttribute ("device.alarm", key: "PRIMARY_CONTROL") {
attributeState "default", label:'blocked!', icon:"st.unknown.thing.thing-circle", backgroundColor:"#ff0000"
}
}
valueTile("info", "device.info", decoration: "flat", height: 3, width: 6, inactiveLabel: false) {
state "default", label: 'This device is known to be incompatible with SmartThings and may not function as expected or cause other devices to malfunction. For more information go to https://support.smartthings.com/hc/en-us/articles/115005123183'
}
main "alarm"
details(["info"])
}
}
def installed() {
log.warn "Blacklisted DTH device joined. DeviceId : ${device.id}, manufacturer: ${device.getDataValue('manufacturer')}, model: ${device.getDataValue('model')}"
}
def uninstalled() {
log.trace "Blacklisted DTH device deleted. DeviceId : ${device.id}, manufacturer: ${device.getDataValue('manufacturer')}, model: ${device.getDataValue('model')}"
}

View File

@@ -21,7 +21,7 @@ metadata {
}
tiles(scale: 2) {
multiAttributeTile(name: "rich-control") {
multiAttributeTile(name: "rich-control", type: "generic", width: 6, height: 4, canChangeIcon: true) {
tileAttribute("device.button", key: "PRIMARY_CONTROL") {
attributeState "default", label: ' ', action: "", icon: "st.unknown.zwave.remote-controller", backgroundColor: "#ffffff"
}

View File

@@ -26,7 +26,7 @@ metadata {
}
tiles(scale: 2) {
multiAttributeTile(name:"rich-control"){
multiAttributeTile(name: "rich-control", type: "generic", width: 6, height: 4, canChangeIcon: true) {
tileAttribute ("device.status", key: "PRIMARY_CONTROL") {
attributeState "Offline", label: '${currentValue}', action: "", icon: "st.Lighting.light99-hue", backgroundColor: "#ffffff"
attributeState "Online", label: '${currentValue}', action: "", icon: "st.Lighting.light99-hue", backgroundColor: "#00A0DC"
@@ -99,4 +99,3 @@ def parse(description) {
}
results
}

View File

@@ -0,0 +1,219 @@
/*
* Copyright 2017 Tomas Axerot
*
* 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.
*/
import physicalgraph.zigbee.clusters.iaszone.ZoneStatus
metadata {
definition(name: "Bosch Motion Detector", namespace: "tomasaxerot", author: "Tomas Axerot") {
capability "Motion Sensor"
capability "Configuration"
capability "Battery"
capability "Temperature Measurement"
capability "Refresh"
capability "Health Check"
capability "Sensor"
command "enrollResponse"
fingerprint inClusters: "0000,0001,0003,0402,0500,0020,0B05", outClusters: "0019", manufacturer: "Bosch", model: "ISW-ZDL1-WP11G", deviceJoinName: "Bosch TriTech Motion Detector"
fingerprint inClusters: "0000,0001,0003,0402,0500,0020,0B05", outClusters: "0019", manufacturer: "Bosch", model: "ISW-ZPR1-WP13", deviceJoinName: "Bosch PIR Motion Detector"
}
simulator {
status "active": "zone report :: type: 19 value: 0031"
status "inactive": "zone report :: type: 19 value: 0030"
}
preferences {
section {
image(name: 'educationalcontent', multiple: true, images: [
"http://cdn.device-gse.smartthings.com/Motion/Motion1.jpg",
"http://cdn.device-gse.smartthings.com/Motion/Motion2.jpg",
"http://cdn.device-gse.smartthings.com/Motion/Motion3.jpg"
])
}
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
}
}
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: "#00A0DC"
attributeState "inactive", label: 'no motion', icon: "st.motion.motion.inactive", backgroundColor: "#cccccc"
}
}
valueTile("temperature", "device.temperature", width: 2, height: 2) {
state("temperature", label: '${currentValue}°', unit: "F",
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: ""
}
standardTile("refresh", "device.refresh", inactiveLabel: false, decoration: "flat", width: 2, height: 2) {
state "default", action: "refresh.refresh", icon: "st.secondary.refresh"
}
main(["motion", "temperature"])
details(["motion", "temperature", "battery", "refresh"])
}
}
def parse(String description) {
log.debug "description: $description"
Map map = zigbee.getEvent(description)
if (!map) {
if (description?.startsWith('zone status')) {
map = parseIasMessage(description)
} else {
Map descMap = zigbee.parseDescriptionAsMap(description)
if (descMap?.clusterInt == 0x0001 && descMap.attrInt == 0x0020 && descMap.commandInt != 0x07 && descMap?.value) {
map = getBatteryResult(Integer.parseInt(descMap.value, 16))
} else if (descMap?.clusterInt == zigbee.TEMPERATURE_MEASUREMENT_CLUSTER && descMap.commandInt == 0x07) {
if (descMap.data[0] == "00") {
log.debug "TEMP REPORTING CONFIG RESPONSE: $descMap"
sendEvent(name: "checkInterval", value: 60 * 12, displayed: false, data: [protocol: "zigbee", hubHardwareId: device.hub.hardwareID])
} else {
log.warn "TEMP REPORTING CONFIG FAILED- error code: ${descMap.data[0]}"
}
} else if (descMap.clusterInt == 0x0406 && descMap.attrInt == 0x0000) {
def value = descMap.value.endsWith("01") ? "active" : "inactive"
log.debug "Doing a read attr motion event"
map = getMotionResult(value)
}
}
} else if (map.name == "temperature") {
if (tempOffset) {
map.value = (int) map.value + (int) tempOffset
}
map.descriptionText = temperatureScale == 'C' ? '{{ device.displayName }} was {{ value }}°C' : '{{ device.displayName }} was {{ value }}°F'
map.translatable = true
}
log.debug "Parse returned $map"
def result = map ? createEvent(map) : [:]
if (description?.startsWith('enroll request')) {
List cmds = zigbee.enrollResponse()
log.debug "enroll response: ${cmds}"
result = cmds?.collect { new physicalgraph.device.HubAction(it) }
}
return result
}
private Map parseIasMessage(String description) {
ZoneStatus zs = zigbee.parseZoneStatus(description)
// Some sensor models that use this DTH use alarm1 and some use alarm2 to signify motion
return (zs.isAlarm1Set() || zs.isAlarm2Set()) ? getMotionResult('active') : getMotionResult('inactive')
}
private Map getBatteryResult(rawValue) {
log.debug "Battery rawValue = ${rawValue}"
def linkText = getLinkText(device)
def result = [:]
//ISW-ZPR1-WP13 uses 4 batteries, 2 are used in measurement
//ISW-ZDL1-WP11G uses 6 batteries, 4 are used in measurement
if (!(rawValue == 0 || rawValue == 255)) {
result.name = 'battery'
result.translatable = true
result.descriptionText = "{{ device.displayName }} battery was {{ value }}%"
def model = device.getDataValue("model")
def volts = rawValue // For the batteryMap to work the key needs to be an int
def batteryMap = []
def minVolts = 0
def maxVolts = 0
if (model == "ISW-ZDL1-WP11G") {
batteryMap = [60: 100, 59: 100, 58: 100, 57: 100, 56: 100, 55: 100,
54: 100, 53: 100, 52: 100, 51: 100, 50: 90, 49: 90,
48: 90, 47: 90, 46: 70, 45: 70, 44: 70, 43: 70, 42: 50,
41: 50, 40: 50, 39: 50, 38: 30, 37: 30, 36: 30, 35: 30,
34: 15, 33: 15, 32: 1, 31: 1, 30: 0]
minVolts = 30
maxVolts = 60
} else if(model == "ISW-ZPR1-WP13") {
batteryMap = [30: 100, 29: 100, 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]
minVolts = 15
maxVolts = 30
} else {
result.value = 0
return result
}
if (volts < minVolts)
volts = minVolts
else if (volts > maxVolts)
volts = maxVolts
result.value = batteryMap[volts]
}
return result
}
private Map getMotionResult(value) {
log.debug 'motion'
String descriptionText = value == 'active' ? "{{ device.displayName }} detected motion" : "{{ device.displayName }} motion has stopped"
return [
name : 'motion',
value : value,
descriptionText: descriptionText,
translatable : true
]
}
/**
* PING is used by Device-Watch in attempt to reach the Device
* */
def ping() {
return zigbee.readAttribute(zigbee.POWER_CONFIGURATION_CLUSTER, 0x0020) // Read the Battery Level
}
def refresh() {
log.debug "refresh called"
def refreshCmds = zigbee.readAttribute(zigbee.POWER_CONFIGURATION_CLUSTER, 0x0020) +
zigbee.readAttribute(zigbee.TEMPERATURE_MEASUREMENT_CLUSTER, 0x0000)
return refreshCmds + zigbee.enrollResponse()
}
def configure() {
// Device-Watch allows 2 check-in misses from device + ping (plus 1 min lag time)
// enrolls with default periodic reporting until newer 5 min interval is confirmed
sendEvent(name: "checkInterval", value: 2 * 60 * 60 + 1 * 60, displayed: false, data: [protocol: "zigbee", hubHardwareId: device.hub.hardwareID])
// temperature minReportTime 30 seconds, maxReportTime 5 min. Reporting interval if no activity
// battery minReport 30 seconds, maxReportTime 6 hrs by default
return refresh() + zigbee.batteryConfig() + zigbee.temperatureConfig(30, 300) // send refresh cmds as part of config
}

View File

@@ -4,7 +4,7 @@
* Author: Zen Within
* Date: 2015-02-21
* Updated by SmartThings
* Date: 2017-10-12
* Date: 2017-11-12
*/
metadata {
definition (name: "Zen Thermostat", namespace: "zenwithin", author: "ZenWithin") {
@@ -23,7 +23,7 @@ metadata {
// To please some of the thermostat SmartApps
command "poll"
fingerprint profileId: "0104", endpointId: "01", inClusters: "0000,0001,0003,0004,0005,0020,0201,0202,0204,0B05", outClusters: "000A, 0019", manufacturer: "Zen Within", model: "Zen-01", deviceJoinName: "Zen Thermostat"
fingerprint profileId: "0104", endpointId: "01", inClusters: "0000,0001,0003,0004,0005,0020,0201,0202,0204,0B05", outClusters: "000A, 0019", manufacturer: "Zen Within", model: "Zen-01", deviceJoinName: "Zen Thermostat"
}
tiles {
@@ -89,8 +89,12 @@ metadata {
section {
input("systemModes", "enum",
title: "Thermostat configured modes\nSelect the modes the thermostat has been configured for, as displayed on the thermostat",
description: "off, heat, cool", defaultValue: "1", required: true, multiple: false,
options:["1":"off, heat, cool", "2":"off, auto, heat, cool", "3":"off, emergency heat, heat, cool"]
description: "off, heat, cool", defaultValue: "3", required: true, multiple: false,
options:["1":"off, heat",
"2":"off, cool",
"3":"off, heat, cool",
"4":"off, auto, heat, cool",
"5":"off, emergency heat, heat, cool"]
)
}
}
@@ -126,9 +130,11 @@ def getSupportedModes() {
def getSupportedModesMap() {
[
"1":["off", "heat", "cool"],
"2":["off", "auto", "heat", "cool"],
"3":["off", "emergency heat", "heat", "cool"]
"1":["off", "heat"],
"2":["off", "cool"],
"3":["off", "heat", "cool"],
"4":["off", "auto", "heat", "cool"],
"5":["off", "emergency heat", "heat", "cool"]
]
}