mirror of
https://github.com/mtan93/SmartThingsPublic.git
synced 2026-03-17 21:03:30 +00:00
Compare commits
17 Commits
PROD_2017.
...
MSA-2034-2
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
77f8fdeb2d | ||
|
|
0b81793b0f | ||
|
|
16f41bddae | ||
|
|
117adea586 | ||
|
|
783538e36d | ||
|
|
320c8918f8 | ||
|
|
0f3656cd12 | ||
|
|
6110aaa0fa | ||
|
|
c391ce4b43 | ||
|
|
073bac8dac | ||
|
|
6325101f52 | ||
|
|
945a972082 | ||
|
|
535fc112de | ||
|
|
a844e85c47 | ||
|
|
60f6e9c02c | ||
|
|
8aceef9be4 | ||
|
|
a1f39849ed |
@@ -1,3 +1,5 @@
|
|||||||
|
//DEPRECATED. INTEGRATION MOVED TO SUPER LAN CONNECT
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Bose SoundTouch
|
* Bose SoundTouch
|
||||||
*
|
*
|
||||||
|
|||||||
@@ -108,11 +108,20 @@ def updated(){
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
def getCommandClassVersions() {
|
||||||
|
[
|
||||||
|
0x20: 1, // Basic
|
||||||
|
0x26: 1, // SwitchMultilevel
|
||||||
|
0x56: 1, // Crc16Encap
|
||||||
|
0x70: 1, // Configuration
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
def parse(String description) {
|
def parse(String description) {
|
||||||
def result = null
|
def result = null
|
||||||
if (description != "updated") {
|
if (description != "updated") {
|
||||||
log.debug "parse() >> zwave.parse($description)"
|
log.debug "parse() >> zwave.parse($description)"
|
||||||
def cmd = zwave.parse(description, [0x20: 1, 0x26: 1, 0x70: 1])
|
def cmd = zwave.parse(description, commandClassVersions)
|
||||||
if (cmd) {
|
if (cmd) {
|
||||||
result = zwaveEvent(cmd)
|
result = zwaveEvent(cmd)
|
||||||
}
|
}
|
||||||
@@ -179,6 +188,16 @@ def zwaveEvent(physicalgraph.zwave.commands.switchmultilevelv1.SwitchMultilevelS
|
|||||||
[createEvent(name:"switch", value:"on"), response(zwave.switchMultilevelV1.switchMultilevelGet().format())]
|
[createEvent(name:"switch", value:"on"), response(zwave.switchMultilevelV1.switchMultilevelGet().format())]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
def zwaveEvent(physicalgraph.zwave.commands.crc16encapv1.Crc16Encap cmd) {
|
||||||
|
def versions = commandClassVersions
|
||||||
|
def version = versions[cmd.commandClass as Integer]
|
||||||
|
def ccObj = version ? zwave.commandClass(cmd.commandClass, version) : zwave.commandClass(cmd.commandClass)
|
||||||
|
def encapsulatedCommand = ccObj?.command(cmd.command)?.parse(cmd.data)
|
||||||
|
if (encapsulatedCommand) {
|
||||||
|
zwaveEvent(encapsulatedCommand)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
def zwaveEvent(physicalgraph.zwave.Command cmd) {
|
def zwaveEvent(physicalgraph.zwave.Command cmd) {
|
||||||
// Handles all Z-Wave commands we aren't interested in
|
// Handles all Z-Wave commands we aren't interested in
|
||||||
[:]
|
[:]
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
//DEPRECATED. INTEGRATION MOVED TO SUPER LAN CONNECT
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Hue Bloom
|
* Hue Bloom
|
||||||
*
|
*
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
//DEPRECATED. INTEGRATION MOVED TO SUPER LAN CONNECT
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Hue Bridge
|
* Hue Bridge
|
||||||
*
|
*
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
//DEPRECATED. INTEGRATION MOVED TO SUPER LAN CONNECT
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Hue Bulb
|
* Hue Bulb
|
||||||
*
|
*
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
//DEPRECATED. INTEGRATION MOVED TO SUPER LAN CONNECT
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Hue Lux Bulb
|
* Hue Lux Bulb
|
||||||
*
|
*
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
//DEPRECATED. INTEGRATION MOVED TO SUPER LAN CONNECT
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Hue White Ambiance Bulb
|
* Hue White Ambiance Bulb
|
||||||
*
|
*
|
||||||
|
|||||||
@@ -49,6 +49,6 @@ def arrived() {
|
|||||||
|
|
||||||
|
|
||||||
def departed() {
|
def departed() {
|
||||||
log.trace "Executing 'arrived'"
|
log.trace "Executing 'departed'"
|
||||||
sendEvent(name: "presence", value: "not present")
|
sendEvent(name: "presence", value: "not present")
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
//DEPRECATED. INTEGRATION MOVED TO SUPER LAN CONNECT
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Copyright 2015 SmartThings
|
* Copyright 2015 SmartThings
|
||||||
*
|
*
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
//DEPRECATED. INTEGRATION MOVED TO SUPER LAN CONNECT
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Copyright 2015 SmartThings
|
* Copyright 2015 SmartThings
|
||||||
*
|
*
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
//DEPRECATED. INTEGRATION MOVED TO SUPER LAN CONNECT
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Copyright 2015 SmartThings
|
* Copyright 2015 SmartThings
|
||||||
*
|
*
|
||||||
|
|||||||
@@ -66,22 +66,29 @@ def parse(String description) {
|
|||||||
else {
|
else {
|
||||||
sendEvent(event)
|
sendEvent(event)
|
||||||
}
|
}
|
||||||
}
|
} else {
|
||||||
else {
|
def descMap = zigbee.parseDescriptionAsMap(description)
|
||||||
def cluster = zigbee.parse(description)
|
if (descMap && descMap.clusterInt == 0x0006 && descMap.commandInt == 0x07) {
|
||||||
|
if (descMap.data[0] == "00") {
|
||||||
if (cluster && cluster.clusterId == 0x0006 && cluster.command == 0x07) {
|
|
||||||
if (cluster.data[0] == 0x00) {
|
|
||||||
log.debug "ON/OFF REPORTING CONFIG RESPONSE: " + cluster
|
log.debug "ON/OFF REPORTING CONFIG RESPONSE: " + cluster
|
||||||
sendEvent(name: "checkInterval", value: 60 * 12, displayed: false, data: [protocol: "zigbee", hubHardwareId: device.hub.hardwareID])
|
sendEvent(name: "checkInterval", value: 60 * 12, displayed: false, data: [protocol: "zigbee", hubHardwareId: device.hub.hardwareID])
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
log.warn "ON/OFF REPORTING CONFIG FAILED- error code:${cluster.data[0]}"
|
log.warn "ON/OFF REPORTING CONFIG FAILED- error code:${cluster.data[0]}"
|
||||||
}
|
}
|
||||||
}
|
} else if (device.getDataValue("manufacturer") == "sengled" && descMap && descMap.clusterInt == 0x0008 && descMap.attrInt == 0x0000) {
|
||||||
else {
|
// This is being done because the sengled element touch/classic incorrectly uses the value 0xFF for the max level.
|
||||||
|
// Per the ZCL spec for the UINT8 data type 0xFF is an invalid value, and 0xFE should be the max. Here we
|
||||||
|
// manually handle the invalid attribute value since it will be ignored by getEvent as an invalid value.
|
||||||
|
// We also set the level of the bulb to 0xFE so future level reports will be 0xFE until it is changed by
|
||||||
|
// something else.
|
||||||
|
if (descMap.value.toUpperCase() == "FF") {
|
||||||
|
descMap.value = "FE"
|
||||||
|
}
|
||||||
|
sendHubCommand(zigbee.command(zigbee.LEVEL_CONTROL_CLUSTER, 0x00, "FE0000").collect { new physicalgraph.device.HubAction(it) }, 0)
|
||||||
|
sendEvent(zigbee.getEventFromAttrData(descMap.clusterInt, descMap.attrInt, descMap.encoding, descMap.value))
|
||||||
|
} else {
|
||||||
log.warn "DID NOT PARSE MESSAGE for description : $description"
|
log.warn "DID NOT PARSE MESSAGE for description : $description"
|
||||||
log.debug "${cluster}"
|
log.debug "${descMap}"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -88,11 +88,20 @@ def updated(){
|
|||||||
sendEvent(name: "checkInterval", value: 2 * 15 * 60 + 2 * 60, displayed: false, data: [protocol: "zwave", hubHardwareId: device.hub.hardwareID])
|
sendEvent(name: "checkInterval", value: 2 * 15 * 60 + 2 * 60, displayed: false, data: [protocol: "zwave", hubHardwareId: device.hub.hardwareID])
|
||||||
}
|
}
|
||||||
|
|
||||||
|
def getCommandClassVersions() {
|
||||||
|
[
|
||||||
|
0x20: 1, // Basic
|
||||||
|
0x26: 1, // SwitchMultilevel
|
||||||
|
0x56: 1, // Crc16Encap
|
||||||
|
0x70: 1, // Configuration
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
def parse(String description) {
|
def parse(String description) {
|
||||||
def result = null
|
def result = null
|
||||||
if (description != "updated") {
|
if (description != "updated") {
|
||||||
log.debug "parse() >> zwave.parse($description)"
|
log.debug "parse() >> zwave.parse($description)"
|
||||||
def cmd = zwave.parse(description, [0x20: 1, 0x26: 1, 0x70: 1])
|
def cmd = zwave.parse(description, commandClassVersions)
|
||||||
if (cmd) {
|
if (cmd) {
|
||||||
result = zwaveEvent(cmd)
|
result = zwaveEvent(cmd)
|
||||||
}
|
}
|
||||||
@@ -159,6 +168,16 @@ def zwaveEvent(physicalgraph.zwave.commands.switchmultilevelv1.SwitchMultilevelS
|
|||||||
[createEvent(name:"switch", value:"on"), response(zwave.switchMultilevelV1.switchMultilevelGet().format())]
|
[createEvent(name:"switch", value:"on"), response(zwave.switchMultilevelV1.switchMultilevelGet().format())]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
def zwaveEvent(physicalgraph.zwave.commands.crc16encapv1.Crc16Encap cmd) {
|
||||||
|
def versions = commandClassVersions
|
||||||
|
def version = versions[cmd.commandClass as Integer]
|
||||||
|
def ccObj = version ? zwave.commandClass(cmd.commandClass, version) : zwave.commandClass(cmd.commandClass)
|
||||||
|
def encapsulatedCommand = ccObj?.command(cmd.command)?.parse(cmd.data)
|
||||||
|
if (encapsulatedCommand) {
|
||||||
|
zwaveEvent(encapsulatedCommand)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
def zwaveEvent(physicalgraph.zwave.Command cmd) {
|
def zwaveEvent(physicalgraph.zwave.Command cmd) {
|
||||||
// Handles all Z-Wave commands we aren't interested in
|
// Handles all Z-Wave commands we aren't interested in
|
||||||
[:]
|
[:]
|
||||||
|
|||||||
@@ -85,11 +85,21 @@ metadata {
|
|||||||
details(["switch", "power", "energy", "levelSliderControl", "refresh", "reset"])
|
details(["switch", "power", "energy", "levelSliderControl", "refresh", "reset"])
|
||||||
}
|
}
|
||||||
|
|
||||||
|
def getCommandClassVersions() {
|
||||||
|
[
|
||||||
|
0x20: 1, // Basic
|
||||||
|
0x26: 3, // SwitchMultilevel
|
||||||
|
0x56: 1, // Crc16Encap
|
||||||
|
0x70: 1, // Configuration
|
||||||
|
0x32: 3, // Meter
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
// parse events into attributes
|
// parse events into attributes
|
||||||
def parse(String description) {
|
def parse(String description) {
|
||||||
def result = null
|
def result = null
|
||||||
if (description != "updated") {
|
if (description != "updated") {
|
||||||
def cmd = zwave.parse(description, [0x20: 1, 0x26: 3, 0x70: 1, 0x32:3])
|
def cmd = zwave.parse(description, commandClassVersions)
|
||||||
if (cmd) {
|
if (cmd) {
|
||||||
result = zwaveEvent(cmd)
|
result = zwaveEvent(cmd)
|
||||||
log.debug("'$description' parsed to $result")
|
log.debug("'$description' parsed to $result")
|
||||||
@@ -124,6 +134,21 @@ def zwaveEvent(physicalgraph.zwave.commands.switchmultilevelv3.SwitchMultilevelR
|
|||||||
dimmerEvents(cmd)
|
dimmerEvents(cmd)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
def zwaveEvent(physicalgraph.zwave.commands.crc16encapv1.Crc16Encap cmd) {
|
||||||
|
def versions = commandClassVersions
|
||||||
|
def version = versions[cmd.commandClass as Integer]
|
||||||
|
def ccObj = version ? zwave.commandClass(cmd.commandClass, version) : zwave.commandClass(cmd.commandClass)
|
||||||
|
def encapsulatedCommand = ccObj?.command(cmd.command)?.parse(cmd.data)
|
||||||
|
if (encapsulatedCommand) {
|
||||||
|
zwaveEvent(encapsulatedCommand)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def zwaveEvent(physicalgraph.zwave.Command cmd) {
|
||||||
|
// Handles all Z-Wave commands we aren't interested in
|
||||||
|
[:]
|
||||||
|
}
|
||||||
|
|
||||||
def dimmerEvents(physicalgraph.zwave.Command cmd) {
|
def dimmerEvents(physicalgraph.zwave.Command cmd) {
|
||||||
def result = []
|
def result = []
|
||||||
def value = (cmd.value ? "on" : "off")
|
def value = (cmd.value ? "on" : "off")
|
||||||
|
|||||||
@@ -89,10 +89,19 @@ def updated() {
|
|||||||
} catch (e) { log.debug e }
|
} catch (e) { log.debug e }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
def getCommandClassVersions() {
|
||||||
|
[
|
||||||
|
0x20: 1, // Basic
|
||||||
|
0x32: 1, // SwitchMultilevel
|
||||||
|
0x56: 1, // Crc16Encap
|
||||||
|
0x72: 2, // ManufacturerSpecific
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
def parse(String description) {
|
def parse(String description) {
|
||||||
def result = null
|
def result = null
|
||||||
if(description == "updated") return
|
if(description == "updated") return
|
||||||
def cmd = zwave.parse(description, [0x20: 1, 0x32: 1, 0x72: 2])
|
def cmd = zwave.parse(description, commandClassVersions)
|
||||||
if (cmd) {
|
if (cmd) {
|
||||||
result = zwaveEvent(cmd)
|
result = zwaveEvent(cmd)
|
||||||
}
|
}
|
||||||
@@ -157,6 +166,16 @@ def zwaveEvent(physicalgraph.zwave.commands.manufacturerspecificv2.ManufacturerS
|
|||||||
result
|
result
|
||||||
}
|
}
|
||||||
|
|
||||||
|
def zwaveEvent(physicalgraph.zwave.commands.crc16encapv1.Crc16Encap cmd) {
|
||||||
|
def versions = commandClassVersions
|
||||||
|
def version = versions[cmd.commandClass as Integer]
|
||||||
|
def ccObj = version ? zwave.commandClass(cmd.commandClass, version) : zwave.commandClass(cmd.commandClass)
|
||||||
|
def encapsulatedCommand = ccObj?.command(cmd.command)?.parse(cmd.data)
|
||||||
|
if (encapsulatedCommand) {
|
||||||
|
zwaveEvent(encapsulatedCommand)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
def zwaveEvent(physicalgraph.zwave.Command cmd) {
|
def zwaveEvent(physicalgraph.zwave.Command cmd) {
|
||||||
log.debug "$device.displayName: Unhandled: $cmd"
|
log.debug "$device.displayName: Unhandled: $cmd"
|
||||||
[:]
|
[:]
|
||||||
|
|||||||
@@ -71,9 +71,17 @@ def updated(){
|
|||||||
sendEvent(name: "checkInterval", value: 2 * 15 * 60 + 2 * 60, displayed: false, data: [protocol: "zwave", hubHardwareId: device.hub.hardwareID])
|
sendEvent(name: "checkInterval", value: 2 * 15 * 60 + 2 * 60, displayed: false, data: [protocol: "zwave", hubHardwareId: device.hub.hardwareID])
|
||||||
}
|
}
|
||||||
|
|
||||||
|
def getCommandClassVersions() {
|
||||||
|
[
|
||||||
|
0x20: 1, // Basic
|
||||||
|
0x56: 1, // Crc16Encap
|
||||||
|
0x70: 1, // Configuration
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
def parse(String description) {
|
def parse(String description) {
|
||||||
def result = null
|
def result = null
|
||||||
def cmd = zwave.parse(description, [0x20: 1, 0x70: 1])
|
def cmd = zwave.parse(description, commandClassVersions)
|
||||||
if (cmd) {
|
if (cmd) {
|
||||||
result = createEvent(zwaveEvent(cmd))
|
result = createEvent(zwaveEvent(cmd))
|
||||||
}
|
}
|
||||||
@@ -120,6 +128,15 @@ def zwaveEvent(physicalgraph.zwave.commands.manufacturerspecificv2.ManufacturerS
|
|||||||
createEvent([descriptionText: "$device.displayName MSR: $msr", isStateChange: false])
|
createEvent([descriptionText: "$device.displayName MSR: $msr", isStateChange: false])
|
||||||
}
|
}
|
||||||
|
|
||||||
|
def zwaveEvent(physicalgraph.zwave.commands.crc16encapv1.Crc16Encap cmd) {
|
||||||
|
def versions = commandClassVersions
|
||||||
|
def version = versions[cmd.commandClass as Integer]
|
||||||
|
def ccObj = version ? zwave.commandClass(cmd.commandClass, version) : zwave.commandClass(cmd.commandClass)
|
||||||
|
def encapsulatedCommand = ccObj?.command(cmd.command)?.parse(cmd.data)
|
||||||
|
if (encapsulatedCommand) {
|
||||||
|
zwaveEvent(encapsulatedCommand)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
def zwaveEvent(physicalgraph.zwave.Command cmd) {
|
def zwaveEvent(physicalgraph.zwave.Command cmd) {
|
||||||
// Handles all Z-Wave commands we aren't interested in
|
// Handles all Z-Wave commands we aren't interested in
|
||||||
|
|||||||
@@ -89,9 +89,17 @@ def updated(){
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
def getCommandClassVersions() {
|
||||||
|
[
|
||||||
|
0x20: 1, // Basic
|
||||||
|
0x56: 1, // Crc16Encap
|
||||||
|
0x70: 1, // Configuration
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
def parse(String description) {
|
def parse(String description) {
|
||||||
def result = null
|
def result = null
|
||||||
def cmd = zwave.parse(description, [0x20: 1, 0x70: 1])
|
def cmd = zwave.parse(description, commandClassVersions)
|
||||||
if (cmd) {
|
if (cmd) {
|
||||||
result = createEvent(zwaveEvent(cmd))
|
result = createEvent(zwaveEvent(cmd))
|
||||||
}
|
}
|
||||||
@@ -138,6 +146,16 @@ def zwaveEvent(physicalgraph.zwave.commands.manufacturerspecificv2.ManufacturerS
|
|||||||
createEvent([descriptionText: "$device.displayName MSR: $msr", isStateChange: false])
|
createEvent([descriptionText: "$device.displayName MSR: $msr", isStateChange: false])
|
||||||
}
|
}
|
||||||
|
|
||||||
|
def zwaveEvent(physicalgraph.zwave.commands.crc16encapv1.Crc16Encap cmd) {
|
||||||
|
def versions = commandClassVersions
|
||||||
|
def version = versions[cmd.commandClass as Integer]
|
||||||
|
def ccObj = version ? zwave.commandClass(cmd.commandClass, version) : zwave.commandClass(cmd.commandClass)
|
||||||
|
def encapsulatedCommand = ccObj?.command(cmd.command)?.parse(cmd.data)
|
||||||
|
if (encapsulatedCommand) {
|
||||||
|
zwaveEvent(encapsulatedCommand)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
def zwaveEvent(physicalgraph.zwave.Command cmd) {
|
def zwaveEvent(physicalgraph.zwave.Command cmd) {
|
||||||
// Handles all Z-Wave commands we aren't interested in
|
// Handles all Z-Wave commands we aren't interested in
|
||||||
|
|||||||
412
smartapps/mujica/tcp-bulbs-connect.src/tcp-bulbs-connect.groovy
Normal file
412
smartapps/mujica/tcp-bulbs-connect.src/tcp-bulbs-connect.groovy
Normal file
@@ -0,0 +1,412 @@
|
|||||||
|
/**
|
||||||
|
* 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.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
definition(
|
||||||
|
name: "Tcp Bulbs (Connect)",
|
||||||
|
namespace: "mujica",
|
||||||
|
author: "SmartThings-Ule",
|
||||||
|
description: "Connect your TCP bulbs to SmartThings using local integration. You must have a geteway with firmware ver 2",
|
||||||
|
category: "SmartThings Labs",
|
||||||
|
iconUrl: "https://s3.amazonaws.com/smartapp-icons/Partner/tcp.png",
|
||||||
|
iconX2Url: "https://s3.amazonaws.com/smartapp-icons/Partner/tcp@2x.png",
|
||||||
|
singleInstance: true
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
preferences {
|
||||||
|
page(name: "iniSettings", title: "Connect Your TCP Lights to SmartThings", content: "iniSettings")
|
||||||
|
page(name: "chooseBulbs", title: "Choose Bulbs to Control With SmartThings", content: "bulbDiscovery")
|
||||||
|
}
|
||||||
|
|
||||||
|
def iniSettings(){
|
||||||
|
state.loadStatus = "Inactive"
|
||||||
|
log.trace "state.loadStatus ${state.loadStatus}"
|
||||||
|
return dynamicPage(name:"iniSettings", title:"Connect Your TCP Lights to SmartThings", nextPage:"chooseBulbs", install:false, uninstall: true) {
|
||||||
|
section("TCP Connected Remote Credentials") {
|
||||||
|
input "ipGateway", "text", title: "Enter TCP Gateway IP", required: true
|
||||||
|
paragraph "Tap 'Next' after you have entered the ip of your TCP Connected Gateway.\r\n\r\nOnce your ip are accepted, SmartThings will scan your TCP installation for Bulbs."
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def bulbDiscovery() {
|
||||||
|
debugOut "bulbDiscovery()"
|
||||||
|
//getToken()
|
||||||
|
state.token = "1234567890"
|
||||||
|
|
||||||
|
if (state.loadStatus == "Inactive"){
|
||||||
|
state.count = 0
|
||||||
|
state.loadStatus = "Loading"
|
||||||
|
log.trace "state.loadStatus ${state.loadStatus}"
|
||||||
|
deviceDiscovery()
|
||||||
|
}
|
||||||
|
log.trace "state.count ${state.count}"
|
||||||
|
state.count = state.count + 1
|
||||||
|
log.trace "state.count ${state.count}"
|
||||||
|
if(state.loadStatus == "Loaded" ){
|
||||||
|
def options = devicesDiscovered() ?: []
|
||||||
|
log.trace "state.loadStatus ${state.loadStatus}"
|
||||||
|
return dynamicPage(name:"chooseBulbs", title:"", nextPage:"", install:true, uninstall: true) {
|
||||||
|
section("Tap Below to View Device List") {
|
||||||
|
input "selectedBulbs", "enum", required:false, title:"Select Bulb", multiple:true, options:options
|
||||||
|
paragraph """Tap 'Done' after you have selected the desired devices."""
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}else{
|
||||||
|
if (state.count)
|
||||||
|
|
||||||
|
log.trace "state.loadStatus ${state.loadStatus}"
|
||||||
|
def msg = state.count >= 3 ? "The TCP Gateway is not responding, please verify the ip address" : "Please wait while we discover your devices. Discovery can take some minutes or more, so sit back and relax! Select your device below once discovered."
|
||||||
|
return dynamicPage(name:"chooseBulbs", title:"", nextPage:"", refreshInterval:5) {
|
||||||
|
section(msg) {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def installed() {
|
||||||
|
debugOut "Installed with settings: ${settings}"
|
||||||
|
|
||||||
|
unschedule()
|
||||||
|
unsubscribe()
|
||||||
|
|
||||||
|
setupBulbs()
|
||||||
|
|
||||||
|
def cron = "0 0/1 * * * ?"
|
||||||
|
log.debug "schedule('$cron', syncronizeDevices)"
|
||||||
|
schedule(cron, syncronizeDevices)
|
||||||
|
}
|
||||||
|
|
||||||
|
def updated() {
|
||||||
|
debugOut "Updated with settings: ${settings}"
|
||||||
|
unschedule()
|
||||||
|
setupBulbs()
|
||||||
|
def cron = "0 0/1 * * * ?"
|
||||||
|
log.debug "schedule('$cron', syncronizeDevices)"
|
||||||
|
schedule(cron, syncronizeDevices)
|
||||||
|
}
|
||||||
|
|
||||||
|
def uninstalled()
|
||||||
|
{
|
||||||
|
unschedule() //in case we have hanging runIn()'s
|
||||||
|
}
|
||||||
|
|
||||||
|
private removeChildDevices(delete)
|
||||||
|
{
|
||||||
|
debugOut "deleting ${delete.size()} bulbs"
|
||||||
|
debugOut "deleting ${delete}"
|
||||||
|
delete.each {
|
||||||
|
deleteChildDevice(it.device.deviceNetworkId)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def uninstallFromChildDevice(childDevice)
|
||||||
|
{
|
||||||
|
def errorMsg = "uninstallFromChildDevice was called and "
|
||||||
|
if (!settings.selectedBulbs) {
|
||||||
|
debugOut errorMsg += "had empty list passed in"
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
def dni = childDevice.device.deviceNetworkId
|
||||||
|
|
||||||
|
if ( !dni ) {
|
||||||
|
debugOut errorMsg += "could not find dni of device"
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
def newDeviceList = settings.selectedBulbs - dni
|
||||||
|
app.updateSetting("selectedBulbs", newDeviceList)
|
||||||
|
debugOut errorMsg += "completed succesfully"
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def setupBulbs() {
|
||||||
|
debugOut "setupBulbs()"
|
||||||
|
def bulbs = state.devices
|
||||||
|
def deviceFile = "TCP Bulb"
|
||||||
|
|
||||||
|
selectedBulbs.each { did ->
|
||||||
|
//see if this is a selected bulb and install it if not already
|
||||||
|
def d = getChildDevice(did)
|
||||||
|
|
||||||
|
if(!d) {
|
||||||
|
def newBulb = bulbs.find { (it.did) == did }
|
||||||
|
d = addChildDevice("mujica", deviceFile, did, null, [name: "${newBulb?.name}", label: "${newBulb?.name}", completedSetup: true,"data":["model":newBulb?.model,"nodetype":newBulb?.nodetype,"node":newBulb?.node,"dni":did]])
|
||||||
|
} else {
|
||||||
|
infoOut "Avoid add existent device ${did}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
def delete = getChildDevices().findAll { !selectedBulbs?.contains(it.deviceNetworkId) }
|
||||||
|
removeChildDevices(delete)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def deviceDiscovery() {
|
||||||
|
log.trace "deviceDiscovery()"
|
||||||
|
def data = "<gip><version>1</version><token>${state.token}</token></gip>"
|
||||||
|
|
||||||
|
def Params = [
|
||||||
|
cmd: "RoomGetCarousel",
|
||||||
|
data: "${data}",
|
||||||
|
fmt: "json"
|
||||||
|
]
|
||||||
|
|
||||||
|
def cmd = toQueryString(Params)
|
||||||
|
|
||||||
|
debugOut "deviceDiscovery()"
|
||||||
|
|
||||||
|
apiGet(cmd,"RoomGetCarouselHandler")
|
||||||
|
}
|
||||||
|
|
||||||
|
def apiGet(String data, String calledBackHandler) {
|
||||||
|
debugOut "apiGet($data, $calledBackHandler) $ipGateway"
|
||||||
|
sendHubCommand(new physicalgraph.device.HubAction([
|
||||||
|
method: "GET",
|
||||||
|
path: "/gwr/gop.php?$data",
|
||||||
|
headers: [
|
||||||
|
HOST: "$ipGateway:80"
|
||||||
|
]], getNetworkId("$ipGateway","80"), [callback: calledBackHandler]))
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void SendCommandHandler(physicalgraph.device.HubResponse hubResponse){
|
||||||
|
debugOut "SendCommandHandler($hubResponse)"
|
||||||
|
debugOut "hubResponse.body ${hubResponse.body}"
|
||||||
|
}
|
||||||
|
void RoomGetCarouselHandler(physicalgraph.device.HubResponse hubResponse){
|
||||||
|
debugOut "RoomGetCarouselHandler($hubResponse)"
|
||||||
|
def bodyXml
|
||||||
|
|
||||||
|
if (hubResponse?.body?.contains("<gip>"))
|
||||||
|
{ // description.xml response (application/xml)
|
||||||
|
debugOut "body contains xml"
|
||||||
|
bodyXml = new XmlSlurper().parseText(hubResponse.body)
|
||||||
|
debugOut "bodyXml $bodyXml"
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def rooms = ""
|
||||||
|
def devices = []
|
||||||
|
def deviceList = []
|
||||||
|
|
||||||
|
|
||||||
|
rooms = bodyXml.room
|
||||||
|
|
||||||
|
|
||||||
|
debugOut "rooms ${rooms[1]}"
|
||||||
|
|
||||||
|
rooms.each({
|
||||||
|
devices = it.device
|
||||||
|
debugOut "it.device ${it.device}"
|
||||||
|
def roomName = it.name
|
||||||
|
debugOut "roomName = ${it.name}"
|
||||||
|
debugOut "devices[1] ${devices[1]}"
|
||||||
|
debugOut "devices[1] != null"
|
||||||
|
def roomId = it?.rid
|
||||||
|
debugOut "Room Device Data: did:${roomId} roomName:${roomName}"
|
||||||
|
devices.each({
|
||||||
|
debugOut "Bulb Device Data: did:${it?.did} room:${roomName} BulbName:${it?.name}"
|
||||||
|
deviceList += ["name" : "${roomName} ${it?.name}", "did" : "${it?.did}", "type" : "${devices?.type}", "node" : "${devices?.node}", "nodetype" : "${devices?.nodetype}", "model" : "${devices?.prodmodel}"]
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
devices = ["devices" : deviceList]
|
||||||
|
debugOut "devices $devices"
|
||||||
|
state.devices = devices.devices
|
||||||
|
state.loadStatus = "Loaded"
|
||||||
|
}
|
||||||
|
|
||||||
|
def getDevices()
|
||||||
|
{
|
||||||
|
state.devices = state.devices ?: [:]
|
||||||
|
}
|
||||||
|
|
||||||
|
void RoomGetCarouselUpdateHandler(physicalgraph.device.HubResponse hubResponse){
|
||||||
|
debugOut "RoomGetCarouselUpdateHandler($hubResponse)"
|
||||||
|
|
||||||
|
|
||||||
|
debugOut "msg ${parseLanMessage(hubResponse.body)}"
|
||||||
|
|
||||||
|
def bodyXml
|
||||||
|
|
||||||
|
if (hubResponse?.body?.contains("<gip>"))
|
||||||
|
{
|
||||||
|
debugOut "body contains xml"
|
||||||
|
bodyXml = new XmlSlurper().parseText(hubResponse.body)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def rooms = ""
|
||||||
|
def devices = []
|
||||||
|
def deviceList = []
|
||||||
|
|
||||||
|
rooms = bodyXml.room
|
||||||
|
|
||||||
|
rooms.each({
|
||||||
|
devices = it.device
|
||||||
|
devices.each({
|
||||||
|
def dni = it.did.text()
|
||||||
|
def bulb = getChildDevice(dni)
|
||||||
|
if ( bulb ){
|
||||||
|
def power = it.power ? it.power.text() as float :0
|
||||||
|
sendEvent( dni, [name: "power", value: power*1000] )
|
||||||
|
if (( it.state.text() == "1" ) && ( bulb?.currentValue("switch") != "on" ))
|
||||||
|
sendEvent( dni, [name: "switch",value:"on"] )
|
||||||
|
|
||||||
|
if (( it.state.text() == "0" ) && ( bulb?.currentValue("switch") != "off" ))
|
||||||
|
sendEvent( dni, [name: "switch",value:"off"] )
|
||||||
|
|
||||||
|
if ( it.level.text() != bulb?.currentValue("level")) {
|
||||||
|
sendEvent( dni, [name: "level",value: "${it.level.text()}"] )
|
||||||
|
sendEvent( dni, [name: "setLevel",value: "${it.level.text()}"] )
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
Map devicesDiscovered() {
|
||||||
|
def devices = state.devices
|
||||||
|
def map = [:]
|
||||||
|
if (devices instanceof java.util.Map) {
|
||||||
|
devices.each {
|
||||||
|
def value = "${it?.name}"
|
||||||
|
def key = it?.did
|
||||||
|
map["${key}"] = value
|
||||||
|
}
|
||||||
|
} else { //backwards compatable
|
||||||
|
devices.each {
|
||||||
|
def value = "${it?.name}"
|
||||||
|
def key = it?.did
|
||||||
|
map["${key}"] = value
|
||||||
|
}
|
||||||
|
}
|
||||||
|
map
|
||||||
|
}
|
||||||
|
|
||||||
|
def getToken() {
|
||||||
|
|
||||||
|
state.token = "1234567890"
|
||||||
|
}
|
||||||
|
|
||||||
|
String toQueryString(Map m) {
|
||||||
|
return m.collect { k, v -> "${k}=${URLEncoder.encode(v.toString())}" }.sort().join("&")
|
||||||
|
}
|
||||||
|
|
||||||
|
def syncronizeDevices() {
|
||||||
|
poll(null)
|
||||||
|
}
|
||||||
|
|
||||||
|
def getNetworkId(ipaddr, port) {
|
||||||
|
"${ipaddr.tokenize('.').collect {String.format('%02X', it.toInteger())}.join()}:${String.format('%04X', port.toInteger())}"
|
||||||
|
}
|
||||||
|
|
||||||
|
/**************************************************************************
|
||||||
|
Child Device Call In Methods
|
||||||
|
**************************************************************************/
|
||||||
|
def on(childDevice) {
|
||||||
|
|
||||||
|
def dni = childDevice.device.deviceNetworkId
|
||||||
|
def data = ""
|
||||||
|
def cmd = ""
|
||||||
|
|
||||||
|
data = "<gip><version>1</version><token>$state.token</token><did>${dni}</did><type>power</type><value>1</value></gip>"
|
||||||
|
cmd = "DeviceSendCommand"
|
||||||
|
|
||||||
|
def qParams = [
|
||||||
|
cmd: cmd,
|
||||||
|
data: "${data}",
|
||||||
|
fmt: "json"
|
||||||
|
]
|
||||||
|
|
||||||
|
cmd = toQueryString(qParams)
|
||||||
|
apiGet(cmd,"SendCommandHandler" )
|
||||||
|
}
|
||||||
|
|
||||||
|
def off(childDevice) {
|
||||||
|
|
||||||
|
def dni = childDevice.device.deviceNetworkId
|
||||||
|
def data = ""
|
||||||
|
def cmd = ""
|
||||||
|
|
||||||
|
data = "<gip><version>1</version><token>$state.token</token><did>${dni}</did><type>power</type><value>0</value></gip>"
|
||||||
|
cmd = "DeviceSendCommand"
|
||||||
|
|
||||||
|
def qParams = [
|
||||||
|
cmd: cmd,
|
||||||
|
data: "${data}",
|
||||||
|
fmt: "json"
|
||||||
|
]
|
||||||
|
|
||||||
|
cmd = toQueryString(qParams)
|
||||||
|
apiGet(cmd,"SendCommandHandler")
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
def setLevel(childDevice, value) {
|
||||||
|
debugOut "setLevel request from child device"
|
||||||
|
|
||||||
|
def dni = childDevice.device.deviceNetworkId
|
||||||
|
def data = ""
|
||||||
|
def cmd = ""
|
||||||
|
|
||||||
|
data = "<gip><version>1</version><token>${state.token}</token><did>${dni}</did><type>level</type><value>${value}</value></gip>"
|
||||||
|
cmd = "DeviceSendCommand"
|
||||||
|
|
||||||
|
|
||||||
|
def qParams = [
|
||||||
|
cmd: cmd,
|
||||||
|
data: "${data}",
|
||||||
|
fmt: "json"
|
||||||
|
]
|
||||||
|
|
||||||
|
cmd = toQueryString(qParams)
|
||||||
|
|
||||||
|
apiGet(cmd,"SendCommandHandler")
|
||||||
|
}
|
||||||
|
|
||||||
|
def poll(childDevice) {
|
||||||
|
infoOut "poll()"
|
||||||
|
def eventTime = new Date().time
|
||||||
|
if ((state.lastPollTime ?:0) + 10000 <= eventTime ){
|
||||||
|
state.lastPollTime = new Date().time
|
||||||
|
def Params = [
|
||||||
|
cmd: "RoomGetCarousel",
|
||||||
|
data: "<gip><version>1</version><token>${state.token}</token></gip>",
|
||||||
|
fmt: "json"
|
||||||
|
]
|
||||||
|
def cmd = toQueryString(Params)
|
||||||
|
apiGet(cmd,"RoomGetCarouselUpdateHandler")
|
||||||
|
}else{
|
||||||
|
infoOut "Multiple poll requests avoided"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**************************************************************************
|
||||||
|
Msg Methods
|
||||||
|
**************************************************************************/
|
||||||
|
|
||||||
|
def debugOut(msg) {
|
||||||
|
//log.debug msg
|
||||||
|
}
|
||||||
|
|
||||||
|
def traceOut(msg) {
|
||||||
|
log.trace msg
|
||||||
|
}
|
||||||
|
|
||||||
|
def infoOut(msg) {
|
||||||
|
log.info msg
|
||||||
|
}
|
||||||
@@ -1,3 +1,5 @@
|
|||||||
|
//DEPRECATED. INTEGRATION MOVED TO SUPER LAN CONNECT
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Bose SoundTouch (Connect)
|
* Bose SoundTouch (Connect)
|
||||||
*
|
*
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
//DEPRECATED. INTEGRATION MOVED TO SUPER LAN CONNECT
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Hue Service Manager
|
* Hue Service Manager
|
||||||
*
|
*
|
||||||
@@ -1083,7 +1085,6 @@ def on(childDevice) {
|
|||||||
log.debug "Executing 'on'"
|
log.debug "Executing 'on'"
|
||||||
def id = getId(childDevice)
|
def id = getId(childDevice)
|
||||||
updateInProgress()
|
updateInProgress()
|
||||||
createSwitchEvent(childDevice, "on")
|
|
||||||
put("lights/$id/state", [on: true])
|
put("lights/$id/state", [on: true])
|
||||||
return "Bulb is turning On"
|
return "Bulb is turning On"
|
||||||
}
|
}
|
||||||
@@ -1092,7 +1093,6 @@ def off(childDevice) {
|
|||||||
log.debug "Executing 'off'"
|
log.debug "Executing 'off'"
|
||||||
def id = getId(childDevice)
|
def id = getId(childDevice)
|
||||||
updateInProgress()
|
updateInProgress()
|
||||||
createSwitchEvent(childDevice, "off")
|
|
||||||
put("lights/$id/state", [on: false])
|
put("lights/$id/state", [on: false])
|
||||||
return "Bulb is turning Off"
|
return "Bulb is turning Off"
|
||||||
}
|
}
|
||||||
@@ -1108,8 +1108,6 @@ def setLevel(childDevice, percent) {
|
|||||||
else
|
else
|
||||||
level = Math.min(Math.round(percent * 254 / 100), 254)
|
level = Math.min(Math.round(percent * 254 / 100), 254)
|
||||||
|
|
||||||
createSwitchEvent(childDevice, level > 0, percent)
|
|
||||||
|
|
||||||
// For Zigbee lights, if level is set to 0 ST just turns them off without changing level
|
// For Zigbee lights, if level is set to 0 ST just turns them off without changing level
|
||||||
// that means that the light will still be on when on is called next time
|
// that means that the light will still be on when on is called next time
|
||||||
// Lets emulate that here
|
// Lets emulate that here
|
||||||
@@ -1128,7 +1126,6 @@ def setSaturation(childDevice, percent) {
|
|||||||
// 0 - 254
|
// 0 - 254
|
||||||
def level = Math.min(Math.round(percent * 254 / 100), 254)
|
def level = Math.min(Math.round(percent * 254 / 100), 254)
|
||||||
// TODO should this be done by app only or should we default to on?
|
// TODO should this be done by app only or should we default to on?
|
||||||
createSwitchEvent(childDevice, "on")
|
|
||||||
put("lights/$id/state", [sat: level, on: true])
|
put("lights/$id/state", [sat: level, on: true])
|
||||||
return "Setting saturation to $percent"
|
return "Setting saturation to $percent"
|
||||||
}
|
}
|
||||||
@@ -1140,7 +1137,6 @@ def setHue(childDevice, percent) {
|
|||||||
// 0 - 65535
|
// 0 - 65535
|
||||||
def level = Math.min(Math.round(percent * 65535 / 100), 65535)
|
def level = Math.min(Math.round(percent * 65535 / 100), 65535)
|
||||||
// TODO should this be done by app only or should we default to on?
|
// TODO should this be done by app only or should we default to on?
|
||||||
createSwitchEvent(childDevice, "on")
|
|
||||||
put("lights/$id/state", [hue: level, on: true])
|
put("lights/$id/state", [hue: level, on: true])
|
||||||
return "Setting hue to $percent"
|
return "Setting hue to $percent"
|
||||||
}
|
}
|
||||||
@@ -1151,7 +1147,6 @@ def setColorTemperature(childDevice, huesettings) {
|
|||||||
updateInProgress()
|
updateInProgress()
|
||||||
// 153 (6500K) to 500 (2000K)
|
// 153 (6500K) to 500 (2000K)
|
||||||
def ct = hueSettings == 6500 ? 153 : Math.round(1000000 / huesettings)
|
def ct = hueSettings == 6500 ? 153 : Math.round(1000000 / huesettings)
|
||||||
createSwitchEvent(childDevice, "on")
|
|
||||||
put("lights/$id/state", [ct: ct, on: true])
|
put("lights/$id/state", [ct: ct, on: true])
|
||||||
return "Setting color temperature to $ct"
|
return "Setting color temperature to $ct"
|
||||||
}
|
}
|
||||||
@@ -1210,7 +1205,6 @@ def setColor(childDevice, huesettings) {
|
|||||||
if (huesettings.switch == "off")
|
if (huesettings.switch == "off")
|
||||||
value.on = false
|
value.on = false
|
||||||
|
|
||||||
createSwitchEvent(childDevice, value.on ? "on" : "off")
|
|
||||||
put("lights/$id/state", value)
|
put("lights/$id/state", value)
|
||||||
return "Setting color to $value"
|
return "Setting color to $value"
|
||||||
}
|
}
|
||||||
@@ -1322,32 +1316,6 @@ private List getRealHubFirmwareVersions() {
|
|||||||
return location.hubs*.firmwareVersionString.findAll { it }
|
return location.hubs*.firmwareVersionString.findAll { it }
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Sends appropriate turningOn/turningOff state events depending on switch or level changes.
|
|
||||||
*
|
|
||||||
* @param childDevice device to send event for
|
|
||||||
* @param setSwitch The new switch state, "on" or "off"
|
|
||||||
* @param setLevel Optional, switchLevel between 0-100, used if you set level to 0 for example since
|
|
||||||
* that should generate "off" instead of level change
|
|
||||||
*/
|
|
||||||
private void createSwitchEvent(childDevice, setSwitch, setLevel = null) {
|
|
||||||
|
|
||||||
if (setLevel == null) {
|
|
||||||
setLevel = childDevice.device?.currentValue("level")
|
|
||||||
}
|
|
||||||
// Create on, off, turningOn or turningOff event as necessary
|
|
||||||
def currentState = childDevice.device?.currentValue("switch")
|
|
||||||
if ((currentState == "off" || currentState == "turningOff")) {
|
|
||||||
if (setSwitch == "on" || setLevel > 0) {
|
|
||||||
childDevice.sendEvent(name: "switch", value: "turningOn", displayed: false)
|
|
||||||
}
|
|
||||||
} else if ((currentState == "on" || currentState == "turningOn")) {
|
|
||||||
if (setSwitch == "off" || setLevel == 0) {
|
|
||||||
childDevice.sendEvent(name: "switch", value: "turningOff", displayed: false)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return the supported color range for different Hue lights. If model is not specified
|
* Return the supported color range for different Hue lights. If model is not specified
|
||||||
* it defaults to the smallest Gamut (B) to ensure that colors set on a mix of devices
|
* it defaults to the smallest Gamut (B) to ensure that colors set on a mix of devices
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
//DEPRECATED. INTEGRATION MOVED TO SUPER LAN CONNECT
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Copyright 2015 SmartThings
|
* Copyright 2015 SmartThings
|
||||||
*
|
*
|
||||||
|
|||||||
Reference in New Issue
Block a user