mirror of
https://github.com/mtan93/SmartThingsPublic.git
synced 2026-03-22 21:03:51 +00:00
Compare commits
31 Commits
PROD_2016.
...
MSA-1626-5
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
27fc3983ae | ||
|
|
bbad6dfa7a | ||
|
|
df6c815aa4 | ||
|
|
d16ac00eb6 | ||
|
|
1941f56007 | ||
|
|
f34df19a65 | ||
|
|
af40246655 | ||
|
|
20239ca982 | ||
|
|
b20c0e371f | ||
|
|
60756e6dc6 | ||
|
|
ee2e1ee04f | ||
|
|
e443cb641c | ||
|
|
b9993a9bf2 | ||
|
|
092971c786 | ||
|
|
a8ee927893 | ||
|
|
51979f0030 | ||
|
|
61d1205e7d | ||
|
|
1dfd1818b4 | ||
|
|
1de73b643c | ||
|
|
fcb504f57e | ||
|
|
b9229c6ef8 | ||
|
|
918e9d9397 | ||
|
|
0c040120cc | ||
|
|
8e37b1fce7 | ||
|
|
439dd634bd | ||
|
|
1db5f75ec5 | ||
|
|
7978f45996 | ||
|
|
0ab657d5f2 | ||
|
|
0ae836b023 | ||
|
|
0b4d555d33 | ||
|
|
4b44460b0b |
@@ -1,10 +1,10 @@
|
|||||||
# SmartThings Public Github Repo
|
# SmartThings Public GitHub Repo
|
||||||
|
|
||||||
An official list of SmartApps and Device Types from SmartThings.
|
An official list of SmartApps and Device Types from SmartThings.
|
||||||
|
|
||||||
Here are some links to help you get started coding right away:
|
Here are some links to help you get started coding right away:
|
||||||
|
|
||||||
* [Github-specific Documentation](http://docs.smartthings.com/en/latest/tools-and-ide/github-integration.html)
|
* [GitHub-specific Documentation](http://docs.smartthings.com/en/latest/tools-and-ide/github-integration.html)
|
||||||
* [Full Documentation](http://docs.smartthings.com)
|
* [Full Documentation](http://docs.smartthings.com)
|
||||||
* [IDE & Simulator](http://ide.smartthings.com)
|
* [IDE & Simulator](http://ide.smartthings.com)
|
||||||
* [Community Forums](http://community.smartthings.com)
|
* [Community Forums](http://community.smartthings.com)
|
||||||
|
|||||||
@@ -0,0 +1,600 @@
|
|||||||
|
/**
|
||||||
|
* Copyright 2015 Stuart Buchanan
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*
|
||||||
|
* Tado AC Thermostat
|
||||||
|
*
|
||||||
|
* Author: Stuart Buchanan, Based on original work by Ian M with thanks. also source for icons was from @tonesto7's excellent Nest Manager.
|
||||||
|
* Date: 2016-11-28 v3.0 Moved all data collection functions into Tado (Connect) SmartApp, huge changes to device handler, existing devices and handler will need to be uninstalled before installing this version
|
||||||
|
* Date: 2016-07-13 v2.9 Quick dirty workaround to control zones with a single account.
|
||||||
|
* Date: 2016-05-07 v2.8 Corrected issue with Fan Speed commands not working.
|
||||||
|
* Date: 2016-04-25 v2.7 Minor changes to thermostatOperatingState to Show "idle" and "fan only" state
|
||||||
|
* Date: 2016-04-25 v2.6 Minor bug fix to correct issue with reading existing set point value.
|
||||||
|
* Date: 2016-04-09 v2.5 Major bug fix exercise, found lots and lots and lots.....now 100% conforms to ST Thermostat capability. main panel now shows colour of operating state. new attributes tadoMode and tadoFanSpeed created.
|
||||||
|
* Date: 2016-04-05 v2.4 Performed Testing with Thermostat Mode Director and found some deficiencies where this would not work correctly. i have now corrected, this now works fine and has been tested.
|
||||||
|
* Date: 2016-04-05 v2.3 added device preference for default temps for some commands as requested by @mitchell_lu66, also added some additional refreshes and error control for unsupported capabilities
|
||||||
|
* Date: 2016-04-05 v2.2 Added Fan Speed & Emergency Heat (1 Hour) Controls and also a manual Mode End function to fall back to Tado Control.
|
||||||
|
Also added preference for how long manual mode runs for either ends at Tado Mode Change (TADO_MODE) or User Control (MANUAL),
|
||||||
|
please ensure the default method is Set in the device properties
|
||||||
|
* Date: 2016-04-05 v2.1 Minor Bug Fixes & improved Icons
|
||||||
|
* Date: 2016-04-05 v2.0 Further Changes to MultiAttribute Tile
|
||||||
|
* Date: 2016-04-05 v1.9 Amended Device Handler Name
|
||||||
|
* Date: 2016-04-05 v1.8 Added all thermostat related capabilities
|
||||||
|
* Date: 2016-04-05 v1.7 Amended device to be capable of both Fahrenheit and celsius and amended the Device multiattribute tile
|
||||||
|
* Date: 2016-04-05 v1.6 switched API calls to new v2 calls as the old ones had been deprecated.
|
||||||
|
* Date: 2016-02-21 v1.5 switched around thermostatOperatingState & thermostatMode to get better compatibility with Home Remote
|
||||||
|
* Date: 2016-02-21 v1.4 added HeatingSetPoint & CoolingSetPoint to make compatible with SmartTiles
|
||||||
|
* Date: 2016-02-21 v1.3 amended the read thermostat properties to match the ST Thermostat Capability
|
||||||
|
* Date: 2016-02-14 v1.2 amended the thermostat properties to match the ST Capability.Thermostat values
|
||||||
|
* Date: 2016-01-23 v1.1 fixed error in Tado Mode detection
|
||||||
|
* Date: 2016-01-22 v1.1 Add Heating & Cooling Controls (initial offering, will need to look into adding all possible commands)
|
||||||
|
* Date: 2015-12-04 v1.0 Initial Release With Temperatures & Relative Humidity
|
||||||
|
*/
|
||||||
|
|
||||||
|
import groovy.json.JsonOutput
|
||||||
|
|
||||||
|
preferences {
|
||||||
|
}
|
||||||
|
|
||||||
|
metadata {
|
||||||
|
definition (name: "Tado Cooling Thermostat", namespace: "fuzzysb", author: "Stuart Buchanan") {
|
||||||
|
capability "Actuator"
|
||||||
|
capability "Temperature Measurement"
|
||||||
|
capability "Thermostat Cooling Setpoint"
|
||||||
|
capability "Thermostat Heating Setpoint"
|
||||||
|
capability "Thermostat Mode"
|
||||||
|
capability "Thermostat Fan Mode"
|
||||||
|
capability "Thermostat Setpoint"
|
||||||
|
capability "Thermostat Operating State"
|
||||||
|
capability "Thermostat"
|
||||||
|
capability "Relative Humidity Measurement"
|
||||||
|
capability "Polling"
|
||||||
|
capability "Refresh"
|
||||||
|
|
||||||
|
attribute "tadoMode", "string"
|
||||||
|
attribute "tadoFanSpeed", "string"
|
||||||
|
command "temperatureUp"
|
||||||
|
command "temperatureDown"
|
||||||
|
command "heatingSetpointUp"
|
||||||
|
command "heatingSetpointDown"
|
||||||
|
command "coolingSetpointUp"
|
||||||
|
command "coolingSetpointDown"
|
||||||
|
command "cmdFanSpeedAuto"
|
||||||
|
command "cmdFanSpeedHigh"
|
||||||
|
command "cmdFanSpeedMid"
|
||||||
|
command "cmdFanSpeedLow"
|
||||||
|
command "dry"
|
||||||
|
command "on"
|
||||||
|
command "fan"
|
||||||
|
command "endManualControl"
|
||||||
|
command "emergencyHeat"
|
||||||
|
}
|
||||||
|
|
||||||
|
// simulator metadata
|
||||||
|
simulator {
|
||||||
|
// status messages
|
||||||
|
|
||||||
|
// reply messages
|
||||||
|
}
|
||||||
|
|
||||||
|
tiles(scale: 2){
|
||||||
|
multiAttributeTile(name: "thermostat", type:"thermostat", width:6, height:4) {
|
||||||
|
tileAttribute("device.temperature", key:"PRIMARY_CONTROL", canChangeIcon: true, canChangeBackground: true){
|
||||||
|
attributeState "default", label:'${currentValue}°', backgroundColor:"#fab907", icon:"st.Home.home1"
|
||||||
|
}
|
||||||
|
tileAttribute("device.temperature", key: "VALUE_CONTROL") {
|
||||||
|
attributeState("VALUE_UP", action: "temperatureUp")
|
||||||
|
attributeState("VALUE_DOWN", action: "temperatureDown")
|
||||||
|
}
|
||||||
|
tileAttribute("device.humidity", key: "SECONDARY_CONTROL") {
|
||||||
|
attributeState("default", label:'${currentValue}%', unit:"%")
|
||||||
|
}
|
||||||
|
tileAttribute("device.thermostatOperatingState", key: "OPERATING_STATE") {
|
||||||
|
attributeState("idle", backgroundColor:"#666666")
|
||||||
|
attributeState("heating", backgroundColor:"#ff471a")
|
||||||
|
attributeState("cooling", backgroundColor:"#1a75ff")
|
||||||
|
attributeState("emergency heat", backgroundColor:"#ff471a")
|
||||||
|
attributeState("drying", backgroundColor:"#c68c53")
|
||||||
|
attributeState("fan only", backgroundColor:"#39e600")
|
||||||
|
attributeState("heating|cooling", backgroundColor:"#ff9900")
|
||||||
|
}
|
||||||
|
tileAttribute("device.thermostatMode", key: "THERMOSTAT_MODE") {
|
||||||
|
attributeState("off", label:'${name}')
|
||||||
|
attributeState("heat", label:'${name}')
|
||||||
|
attributeState("cool", label:'${name}')
|
||||||
|
attributeState("auto", label:'${name}')
|
||||||
|
attributeState("fan", label:'${name}')
|
||||||
|
attributeState("dry", label:'${name}')
|
||||||
|
}
|
||||||
|
tileAttribute("device.heatingSetpoint", key: "HEATING_SETPOINT") {
|
||||||
|
attributeState("default", label:'${currentValue}', unit:"dF")
|
||||||
|
}
|
||||||
|
tileAttribute("device.coolingSetpoint", key: "COOLING_SETPOINT") {
|
||||||
|
attributeState("default", label:'${currentValue}', unit:"dF")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
standardTile("tadoMode", "device.tadoMode", width: 2, height: 2, canChangeIcon: true, canChangeBackground: true) {
|
||||||
|
state("SLEEP", label:'${name}', backgroundColor:"#0164a8", icon:"st.Bedroom.bedroom2")
|
||||||
|
state("HOME", label:'${name}', backgroundColor:"#fab907", icon:"st.Home.home2")
|
||||||
|
state("AWAY", label:'${name}', backgroundColor:"#62aa12", icon:"st.Outdoor.outdoor18")
|
||||||
|
state("OFF", label:'${name}', backgroundColor:"#ffffff", icon:"st.switches.switch.off", defaultState: true)
|
||||||
|
state("MANUAL", label:'${name}', backgroundColor:"#804000", icon:"st.Weather.weather1")
|
||||||
|
}
|
||||||
|
|
||||||
|
standardTile("refresh", "device.switch", inactiveLabel: false, width: 2, height: 1, decoration: "flat") {
|
||||||
|
state "default", label:"", action:"refresh.refresh", icon:"st.secondary.refresh"
|
||||||
|
}
|
||||||
|
|
||||||
|
standardTile("thermostatMode", "device.thermostatMode", width: 2, height: 2, canChangeIcon: true, canChangeBackground: true) {
|
||||||
|
state("heat", label:'HEAT', backgroundColor:"#ea2a2a", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/heat_mode_icon.png")
|
||||||
|
state("emergency heat", label:'HEAT', backgroundColor:"#ea2a2a", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/heat_mode_icon.png")
|
||||||
|
state("cool", label:'COOL', backgroundColor:"#089afb", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/cool_mode_icon.png")
|
||||||
|
state("dry", label:'DRY', icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/dry_mode_icon.png")
|
||||||
|
state("fan", label:'FAN', backgroundColor:"#ffffff", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/fan_mode_icononly.png")
|
||||||
|
state("auto", label:'AUTO', icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/auto_mode_icon.png")
|
||||||
|
state("off", label:'', backgroundColor:"#ffffff", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/hvac_off.png", defaultState: true)
|
||||||
|
}
|
||||||
|
|
||||||
|
valueTile("thermostatSetpoint", "device.thermostatSetpoint", width: 2, height: 1, decoration: "flat") {
|
||||||
|
state "default", label: 'Set Point\r\n\${currentValue}°'
|
||||||
|
}
|
||||||
|
|
||||||
|
valueTile("heatingSetpoint", "device.heatingSetpoint", width: 2, height: 1, decoration: "flat") {
|
||||||
|
state "default", label: 'Set Point\r\n\${currentValue}°'
|
||||||
|
}
|
||||||
|
|
||||||
|
valueTile("coolingSetpoint", "device.coolingSetpoint", width: 2, height: 1, decoration: "flat") {
|
||||||
|
state "default", label: 'Set Point\r\n\${currentValue}°'
|
||||||
|
}
|
||||||
|
|
||||||
|
valueTile("outsidetemperature", "device.outsidetemperature", width: 2, height: 1, decoration: "flat") {
|
||||||
|
state "outsidetemperature", label: 'Outside Temp\r\n${currentValue}°'
|
||||||
|
}
|
||||||
|
|
||||||
|
standardTile("tadoFanSpeed", "device.tadoFanSpeed", width: 2, height: 2, canChangeIcon: true, canChangeBackground: true, decoration: "flat") {
|
||||||
|
state("OFF", label:'', icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/fan_off_icon.png", defaultState: true)
|
||||||
|
state("AUTO", label:'', icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/fan_auto_icon.png")
|
||||||
|
state("HIGH", label:'', icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/fan_high_icon.png")
|
||||||
|
state("MIDDLE", label:'', icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/fan_med_icon.png")
|
||||||
|
state("LOW", label:'', icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/fan_low_icon.png")
|
||||||
|
}
|
||||||
|
|
||||||
|
standardTile("setAuto", "device.thermostat", width: 2, height: 1, decoration: "flat") {
|
||||||
|
state "default", label:"", action:"thermostat.auto", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/hvac_auto.png"
|
||||||
|
}
|
||||||
|
standardTile("setDry", "device.thermostat", width: 2, height: 1, decoration: "flat") {
|
||||||
|
state "default", label:"", action:"dry", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/hvac_dry.png"
|
||||||
|
}
|
||||||
|
standardTile("setOn", "device.thermostat", width: 2, height: 1, decoration: "flat") {
|
||||||
|
state "default", label:"", action:"on", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/hvac_on.png"
|
||||||
|
}
|
||||||
|
standardTile("setOff", "device.thermostat", width: 2, height: 1, decoration: "flat") {
|
||||||
|
state "default", label:"", action:"thermostat.off", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/hvac_off.png"
|
||||||
|
}
|
||||||
|
standardTile("cool", "device.thermostat", width: 2, height: 1, decoration: "flat") {
|
||||||
|
state "default", label:"", action:"thermostat.cool", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/hvac_cool.png"
|
||||||
|
}
|
||||||
|
standardTile("heat", "device.thermostat", width: 2, height: 1, decoration: "flat") {
|
||||||
|
state "default", label:"", action:"thermostat.heat", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/hvac_heat.png"
|
||||||
|
}
|
||||||
|
standardTile("emergencyHeat", "device.thermostat", width: 2, height: 1, decoration: "flat") {
|
||||||
|
state "default", label:"", action:"thermostat.emergencyHeat", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/emergencyHeat.png"
|
||||||
|
}
|
||||||
|
standardTile("fan", "device.thermostat", width: 2, height: 1, decoration: "flat") {
|
||||||
|
state "default", label:"", action:"thermostat.fan", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/fan_mode_icon.png"
|
||||||
|
}
|
||||||
|
standardTile("coolingSetpointUp", "device.coolingSetpoint", canChangeIcon: false, decoration: "flat") {
|
||||||
|
state "coolingSetpointUp", label:' ', action:"coolingSetpointUp", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/cool_arrow_up.png"
|
||||||
|
}
|
||||||
|
standardTile("coolingSetpointDown", "device.coolingSetpoint", canChangeIcon: false, decoration: "flat") {
|
||||||
|
state "coolingSetpointDown", label:' ', action:"coolingSetpointDown", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/cool_arrow_down.png"
|
||||||
|
}
|
||||||
|
standardTile("heatingSetpointUp", "device.heatingSetpoint", canChangeIcon: false, decoration: "flat") {
|
||||||
|
state "heatingSetpointUp", label:' ', action:"heatingSetpointUp", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/heat_arrow_up.png"
|
||||||
|
}
|
||||||
|
standardTile("heatingSetpointDown", "device.heatingSetpoint", canChangeIcon: false, decoration: "flat") {
|
||||||
|
state "heatingSetpointDown", label:' ', action:"heatingSetpointDown", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/heat_arrow_down.png"
|
||||||
|
}
|
||||||
|
standardTile("cmdFanSpeedAuto", "device.thermostat", width: 2, height: 1, canChangeIcon: false, canChangeBackground: true, decoration: "flat") {
|
||||||
|
state("default", label:'', action:"cmdFanSpeedAuto", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/fan_auto_icon.png")
|
||||||
|
}
|
||||||
|
standardTile("cmdFanSpeedHigh", "device.thermostat", width: 2, height: 1, canChangeIcon: false, canChangeBackground: true, decoration: "flat") {
|
||||||
|
state("default", label:'', action:"cmdFanSpeedHigh", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/fan_high_icon.png")
|
||||||
|
}
|
||||||
|
standardTile("cmdFanSpeedMid", "device.thermostat", width: 2, height: 1, canChangeIcon: false, canChangeBackground: true, decoration: "flat") {
|
||||||
|
state("default", label:'', action:"cmdFanSpeedMid", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/fan_med_icon.png")
|
||||||
|
}
|
||||||
|
standardTile("cmdFanSpeedLow", "device.thermostat", width: 2, height: 1, canChangeIcon: false, canChangeBackground: true, decoration: "flat") {
|
||||||
|
state("default", label:'', action:"cmdFanSpeedLow", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/fan_low_icon.png")
|
||||||
|
}
|
||||||
|
standardTile("endManualControl", "device.thermostat", width: 2, height: 1, canChangeIcon: false, canChangeBackground: true, decoration: "flat") {
|
||||||
|
state("default", label:'', action:"endManualControl", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado-Cooling-AC.src/Images/endManual.png")
|
||||||
|
}
|
||||||
|
|
||||||
|
main(["thermostat"])
|
||||||
|
details(["thermostat","thermostatMode","coolingSetpointUp","coolingSetpointDown","autoOperation","heatingSetpointUp","heatingSetpointDown","outsidetemperature","thermostatSetpoint","tadoMode","refresh","tadoFanSpeed","setAuto","setOn","setOff","fan","cool","heat","setDry","cmdFanSpeedAuto","emergencyHeat","endManualControl","cmdFanSpeedLow","cmdFanSpeedMid","cmdFanSpeedHigh"])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def setCapabilitytadoType(value){
|
||||||
|
state.tadoType = value
|
||||||
|
log.debug("state.tadoType = ${state.tadoType}")
|
||||||
|
}
|
||||||
|
|
||||||
|
def getCapabilitytadoType() {
|
||||||
|
def map = null
|
||||||
|
map = [name: "capabilityTadoType", value: state.tadoType]
|
||||||
|
return map
|
||||||
|
}
|
||||||
|
|
||||||
|
def setCapabilitySupportsAuto(value){
|
||||||
|
state.supportsAuto = value
|
||||||
|
log.debug("state.supportsAuto = ${state.supportsAuto}")
|
||||||
|
}
|
||||||
|
|
||||||
|
def getCapabilitySupportsAuto() {
|
||||||
|
def map = null
|
||||||
|
map = [name: "capabilitySupportsAuto", value: state.supportsAuto]
|
||||||
|
return map
|
||||||
|
}
|
||||||
|
|
||||||
|
def setCapabilitySupportsCool(value){
|
||||||
|
state.supportsCool = value
|
||||||
|
log.debug("state.supportsCool = ${state.supportsCool}")
|
||||||
|
}
|
||||||
|
|
||||||
|
def getCapabilitySupportsCool() {
|
||||||
|
def map = null
|
||||||
|
map = [name: "capabilitySupportsCool", value: state.supportsCool]
|
||||||
|
return map
|
||||||
|
}
|
||||||
|
|
||||||
|
def setCapabilitySupportsCoolAutoFanSpeed(value){
|
||||||
|
state.SupportsCoolAutoFanSpeed = value
|
||||||
|
log.debug("state.SupportsCoolAutoFanSpeed = ${state.SupportsCoolAutoFanSpeed}")
|
||||||
|
}
|
||||||
|
|
||||||
|
def getCapabilitySupportsCoolAutoFanSpeed() {
|
||||||
|
def map = null
|
||||||
|
map = [name: "capabilitySupportsCoolAutoFanSpeed", value: state.supportsCoolAutoFanSpeed]
|
||||||
|
return map
|
||||||
|
}
|
||||||
|
|
||||||
|
def setCapabilitySupportsDry(value){
|
||||||
|
state.supportsDry = value
|
||||||
|
log.debug("state.supportsDry = ${state.supportsDry}")
|
||||||
|
}
|
||||||
|
|
||||||
|
def getCapabilitySupportsDry() {
|
||||||
|
def map = null
|
||||||
|
map = [name: "capabilitySupportsDry", value: state.supportsDry]
|
||||||
|
return map
|
||||||
|
}
|
||||||
|
|
||||||
|
def setCapabilitySupportsFan(value){
|
||||||
|
state.supportsFan = value
|
||||||
|
log.debug("state.supportsFan = ${state.supportsFan}")
|
||||||
|
}
|
||||||
|
|
||||||
|
def getCapabilitySupportsFan() {
|
||||||
|
def map = null
|
||||||
|
map = [name: "capabilitySupportsFan", value: state.supportsFan]
|
||||||
|
return map
|
||||||
|
}
|
||||||
|
|
||||||
|
def setCapabilitySupportsHeat(value){
|
||||||
|
state.supportsHeat = value
|
||||||
|
log.debug("state.supportsHeat = ${state.supportsHeat}")
|
||||||
|
}
|
||||||
|
|
||||||
|
def getCapabilitySupportsHeat() {
|
||||||
|
def map = null
|
||||||
|
map = [name: "capabilitySupportsHeat", value: state.supportsHeat]
|
||||||
|
return map
|
||||||
|
}
|
||||||
|
|
||||||
|
def setCapabilitySupportsHeatAutoFanSpeed(value){
|
||||||
|
state.SupportsHeatAutoFanSpeed = value
|
||||||
|
log.debug("state.SupportsHeatAutoFanSpeed = ${state.SupportsHeatAutoFanSpeed}")
|
||||||
|
}
|
||||||
|
|
||||||
|
def getCapabilitySupportsHeatAutoFanSpeed() {
|
||||||
|
def map = null
|
||||||
|
map = [name: "capabilitySupportsHeatAutoFanSpeed", value: state.SupportsHeatAutoFanSpeed]
|
||||||
|
return map
|
||||||
|
}
|
||||||
|
|
||||||
|
def setCapabilityMaxCoolTemp(value){
|
||||||
|
state.MaxCoolTemp = value
|
||||||
|
log.debug("set state.MaxCoolTemp to : " + state.MaxCoolTemp)
|
||||||
|
}
|
||||||
|
|
||||||
|
def getCapabilityMaxCoolTemp() {
|
||||||
|
def map = null
|
||||||
|
map = [name: "capabilityMaxCoolTemp", value: state.MaxCoolTemp]
|
||||||
|
return map
|
||||||
|
}
|
||||||
|
|
||||||
|
def setCapabilityMinCoolTemp(value){
|
||||||
|
state.MinCoolTemp = value
|
||||||
|
log.debug("set state.MinCoolTemp to : " + state.MinCoolTemp)
|
||||||
|
}
|
||||||
|
|
||||||
|
def getCapabilityMinCoolTemp() {
|
||||||
|
def map = null
|
||||||
|
map = [name: "capabilityMinCoolTemp", value: state.MinCoolTemp]
|
||||||
|
return map
|
||||||
|
}
|
||||||
|
|
||||||
|
def setCapabilityMaxHeatTemp(value){
|
||||||
|
state.MaxHeatTemp = value
|
||||||
|
log.debug("set state.MaxHeatTemp to : " + state.MaxHeatTemp)
|
||||||
|
}
|
||||||
|
|
||||||
|
def getCapabilityMaxHeatTemp() {
|
||||||
|
def map = null
|
||||||
|
map = [name: "capabilityMaxHeatTemp", value: state.MaxHeatTemp]
|
||||||
|
return map
|
||||||
|
}
|
||||||
|
|
||||||
|
def setCapabilityMinHeatTemp(value){
|
||||||
|
state.MinHeatTemp = value
|
||||||
|
log.debug("set state.MinHeatTemp to : " + state.MinHeatTemp)
|
||||||
|
}
|
||||||
|
|
||||||
|
def getCapabilityMinHeatTemp() {
|
||||||
|
def map = null
|
||||||
|
map = [name: "capabilityMinHeatTemp", value: state.MinHeatTemp]
|
||||||
|
return map
|
||||||
|
}
|
||||||
|
|
||||||
|
def updated(){
|
||||||
|
refresh()
|
||||||
|
}
|
||||||
|
|
||||||
|
def installed(){
|
||||||
|
refresh()
|
||||||
|
}
|
||||||
|
|
||||||
|
def poll() {
|
||||||
|
log.debug "Executing 'poll'"
|
||||||
|
refresh()
|
||||||
|
}
|
||||||
|
|
||||||
|
def refresh() {
|
||||||
|
log.debug "Executing 'refresh'"
|
||||||
|
parent.statusCommand(this)
|
||||||
|
getWeather()
|
||||||
|
}
|
||||||
|
|
||||||
|
def getWeather(){
|
||||||
|
parent.weatherStatusCommand(this)
|
||||||
|
}
|
||||||
|
|
||||||
|
def auto() {
|
||||||
|
log.debug "Executing 'auto'"
|
||||||
|
parent.autoCommand(this)
|
||||||
|
parent.statusCommand(this)
|
||||||
|
}
|
||||||
|
|
||||||
|
def on() {
|
||||||
|
log.debug "Executing 'on'"
|
||||||
|
parent.onCommand(this)
|
||||||
|
parent.statusCommand(this)
|
||||||
|
}
|
||||||
|
|
||||||
|
def off() {
|
||||||
|
log.debug "Executing 'off'"
|
||||||
|
parent.offCommand(this)
|
||||||
|
parent.statusCommand(this)
|
||||||
|
}
|
||||||
|
|
||||||
|
def dry() {
|
||||||
|
log.debug "Executing 'dry'"
|
||||||
|
parent.dryCommand(this)
|
||||||
|
parent.statusCommand(this)
|
||||||
|
}
|
||||||
|
|
||||||
|
def setThermostatMode(requiredMode){
|
||||||
|
switch (requiredMode) {
|
||||||
|
case "dry":
|
||||||
|
dry()
|
||||||
|
break
|
||||||
|
case "heat":
|
||||||
|
heat()
|
||||||
|
break
|
||||||
|
case "cool":
|
||||||
|
cool()
|
||||||
|
break
|
||||||
|
case "auto":
|
||||||
|
auto()
|
||||||
|
break
|
||||||
|
case "fan":
|
||||||
|
fan()
|
||||||
|
break
|
||||||
|
case "off":
|
||||||
|
off()
|
||||||
|
break
|
||||||
|
case "emergency heat":
|
||||||
|
emergencyHeat()
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def thermostatFanMode(requiredMode){
|
||||||
|
switch (requiredMode) {
|
||||||
|
case "auto":
|
||||||
|
fan()
|
||||||
|
break
|
||||||
|
case "on":
|
||||||
|
fan()
|
||||||
|
break
|
||||||
|
case "circulate":
|
||||||
|
fan()
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def setHeatingSetpoint(targetTemperature) {
|
||||||
|
log.debug "Executing 'setHeatingSetpoint'"
|
||||||
|
log.debug "Target Temperature ${targetTemperature}"
|
||||||
|
parent.setHeatingTempCommand(this,targetTemperature)
|
||||||
|
refresh()
|
||||||
|
}
|
||||||
|
|
||||||
|
def temperatureUp(){
|
||||||
|
if (device.currentValue("thermostatMode") == "heat") {
|
||||||
|
heatingSetpointUp()
|
||||||
|
} else if (device.currentValue("thermostatMode") == "cool") {
|
||||||
|
coolingSetpointUp()
|
||||||
|
} else {
|
||||||
|
log.debug ("temperature setpoint not supported in the current thermostat mode")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def temperatureDown(){
|
||||||
|
if (device.currentValue("thermostatMode") == "heat") {
|
||||||
|
heatingSetpointDown()
|
||||||
|
} else if (device.currentValue("thermostatMode") == "cool") {
|
||||||
|
coolingSetpointDown()
|
||||||
|
} else {
|
||||||
|
log.debug ("temperature setpoint not supported in the current thermostat mode")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def heatingSetpointUp(){
|
||||||
|
def capabilitysupported = state.supportsHeat
|
||||||
|
if (capabilitysupported == "true"){
|
||||||
|
log.debug "Current SetPoint Is " + (device.currentValue("thermostatSetpoint")).toString()
|
||||||
|
if ((device.currentValue("thermostatSetpoint").toInteger() - 1 ) < state.MinHeatTemp){
|
||||||
|
log.debug("cannot decrease heat setpoint, its already at the minimum level of " + state.MinHeatTemp)
|
||||||
|
} else {
|
||||||
|
int newSetpoint = (device.currentValue("thermostatSetpoint")).toInteger() + 1
|
||||||
|
log.debug "Setting heatingSetpoint up to: ${newSetpoint}"
|
||||||
|
setHeatingSetpoint(newSetpoint)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
log.debug("Sorry Heat Capability not supported by your HVAC Device")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def heatingSetpointDown(){
|
||||||
|
def capabilitysupported = state.supportsHeat
|
||||||
|
if (capabilitysupported == "true"){
|
||||||
|
log.debug "Current SetPoint Is " + (device.currentValue("thermostatSetpoint")).toString()
|
||||||
|
if ((device.currentValue("thermostatSetpoint").toInteger() + 1 ) > state.MaxHeatTemp){
|
||||||
|
log.debug("cannot increase heat setpoint, its already at the maximum level of " + state.MaxHeatTemp)
|
||||||
|
} else {
|
||||||
|
int newSetpoint = (device.currentValue("thermostatSetpoint")).toInteger() - 1
|
||||||
|
log.debug "Setting heatingSetpoint down to: ${newSetpoint}"
|
||||||
|
setHeatingSetpoint(newSetpoint)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
log.debug("Sorry Heat Capability not supported by your HVAC Device")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def setCoolingSetpoint(targetTemperature) {
|
||||||
|
log.debug "Executing 'setCoolingSetpoint'"
|
||||||
|
log.debug "Target Temperature ${targetTemperature}"
|
||||||
|
parent.setCoolingTempCommand(this,targetTemperature)
|
||||||
|
refresh()
|
||||||
|
}
|
||||||
|
|
||||||
|
def coolingSetpointUp(){
|
||||||
|
def capabilitysupported = state.supportsCool
|
||||||
|
if (capabilitysupported == "true"){
|
||||||
|
log.debug "Current SetPoint Is " + (device.currentValue("thermostatSetpoint")).toString()
|
||||||
|
if ((device.currentValue("thermostatSetpoint").toInteger() + 1 ) > state.MaxCoolTemp){
|
||||||
|
log.debug("cannot increase cool setpoint, its already at the maximum level of " + state.MaxCoolTemp)
|
||||||
|
} else {
|
||||||
|
int newSetpoint = (device.currentValue("thermostatSetpoint")).toInteger() + 1
|
||||||
|
log.debug "Setting coolingSetpoint up to: ${newSetpoint}"
|
||||||
|
setCoolingSetpoint(newSetpoint)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
log.debug("Sorry Cool Capability not supported by your HVAC Device")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def coolingSetpointDown(){
|
||||||
|
def capabilitysupported = state.supportsCool
|
||||||
|
if (capabilitysupported == "true"){
|
||||||
|
log.debug "Current SetPoint Is " + (device.currentValue("thermostatSetpoint")).toString()
|
||||||
|
if ((device.currentValue("thermostatSetpoint").toInteger() - 1 ) < state.MinCoolTemp){
|
||||||
|
log.debug("cannot decrease cool setpoint, its already at the minimum level of " + state.MinCoolTemp)
|
||||||
|
} else {
|
||||||
|
int newSetpoint = (device.currentValue("thermostatSetpoint")).toInteger() - 1
|
||||||
|
log.debug "Setting coolingSetpoint down to: ${newSetpoint}"
|
||||||
|
setCoolingSetpoint(newSetpoint)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
log.debug("Sorry Cool Capability not supported by your HVAC Device")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def fanOn(){
|
||||||
|
fan()
|
||||||
|
}
|
||||||
|
|
||||||
|
def fanCirculate(){
|
||||||
|
fan()
|
||||||
|
}
|
||||||
|
|
||||||
|
def cool(){
|
||||||
|
def capabilitysupported = state.supportsCool
|
||||||
|
if (capabilitysupported == "true"){
|
||||||
|
parent.coolCommand(this)
|
||||||
|
parent.statusCommand(this)
|
||||||
|
} else {
|
||||||
|
log.debug("Sorry Cool Capability not supported by your HVAC Device")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def heat(){
|
||||||
|
def capabilitysupported = state.supportsHeat
|
||||||
|
if (capabilitysupported == "true"){
|
||||||
|
parent.heatCommand(this)
|
||||||
|
parent.statusCommand(this)
|
||||||
|
} else {
|
||||||
|
log.debug("Sorry Heat Capability not supported by your HVAC Device")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def fan(){
|
||||||
|
parent.fanAuto(this)
|
||||||
|
refresh()
|
||||||
|
}
|
||||||
|
|
||||||
|
def emergencyHeat(){
|
||||||
|
parent.emergencyHeat(this)
|
||||||
|
}
|
||||||
|
|
||||||
|
def cmdFanSpeedAuto(){
|
||||||
|
parent.cmdFanSpeedAuto(this)
|
||||||
|
}
|
||||||
|
|
||||||
|
def cmdFanSpeedHigh(){
|
||||||
|
parent.cmdFanSpeedHigh(this)
|
||||||
|
}
|
||||||
|
|
||||||
|
def cmdFanSpeedMed(){
|
||||||
|
parent.cmdFanSpeedMed(this)
|
||||||
|
}
|
||||||
|
|
||||||
|
def cmdFanSpeedLow(){
|
||||||
|
parent.cmdFanSpeedLow(this)
|
||||||
|
}
|
||||||
|
|
||||||
|
def endManualControl(){
|
||||||
|
parent.endManualControl(this)
|
||||||
|
}
|
||||||
@@ -0,0 +1,270 @@
|
|||||||
|
/**
|
||||||
|
* 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.
|
||||||
|
*
|
||||||
|
* Tado Thermostat
|
||||||
|
*
|
||||||
|
* Author: Stuart Buchanan, Based on original work by Ian M with thanks. also source for icons was from @tonesto7's excellent Nest Manager.
|
||||||
|
*
|
||||||
|
* Updates:
|
||||||
|
* Date: 2016-11-28 v1.9 Moved all data collection functions into Tado (Connect) SmartApp, huge changes to device handler, existing devices and handler will need to be uninstalled before installing this version
|
||||||
|
* Date: 2016-07-13 v1.8 Quick dirty workaround to control zones with a single account.
|
||||||
|
* Date: 2016-04-25 v1.7 Finally found time to update this with the lessons learnt from the Tado Cooling Device Type. will bring better support for RM and Thermostat Director
|
||||||
|
* Date: 2016-04-08 v1.6 added statusCommand calls to refresh more frequently, also improved compatibility with Rule Machine and Thermostat Mode Director in addition also added default heating temperature where you can set the default temperature for the mode commands.
|
||||||
|
* Date: 2016-04-05 v1.5 added improved icons and also a manual Mode End function to fall back to Tado Control.
|
||||||
|
Also added preference for how long manual mode runs for either ends at Tado Mode Change (TADO_MODE) or User Control (MANUAL),
|
||||||
|
please ensure the default method is Set in the device properties
|
||||||
|
* Date: 2016-04-05 v1.4 rewrite of complete functions to support Tado API v2
|
||||||
|
* Date: 2016-01-20 v1.3 Updated hvacStatus to include include the correct HomeId for Humidity Value
|
||||||
|
* Date: 2016-01-15 v1.2 Refactored API request code and added querying/display of humidity
|
||||||
|
* Date: 2015-12-23 v1.1 Added functionality to change thermostat settings
|
||||||
|
* Date: 2015-12-04 v1.0 Initial release
|
||||||
|
*/
|
||||||
|
|
||||||
|
preferences {
|
||||||
|
input("username", "text", title: "Username", description: "Your Tado username")
|
||||||
|
input("password", "password", title: "Password", description: "Your Tado password")
|
||||||
|
input("tadoZoneId", "number", title: "Enter Tado Zone ID?", required: true)
|
||||||
|
input("manualmode", "enum", title: "Default Manual Overide Method", options: ["TADO_MODE","MANUAL"], required: false, defaultValue:"TADO_MODE")
|
||||||
|
input("defHeatingTemp", "number", title: "Default Heating Temperature?", required: false, defaultValue: 21)
|
||||||
|
}
|
||||||
|
|
||||||
|
metadata {
|
||||||
|
definition (name: "Tado Heating Thermostat", namespace: "fuzzysb", author: "Stuart Buchanan") {
|
||||||
|
capability "Actuator"
|
||||||
|
capability "Temperature Measurement"
|
||||||
|
capability "Thermostat Heating Setpoint"
|
||||||
|
capability "Thermostat Setpoint"
|
||||||
|
capability "Thermostat Mode"
|
||||||
|
capability "Thermostat Operating State"
|
||||||
|
capability "Thermostat"
|
||||||
|
capability "Relative Humidity Measurement"
|
||||||
|
capability "Polling"
|
||||||
|
capability "Refresh"
|
||||||
|
attribute "tadoMode", "string"
|
||||||
|
command "temperatureUp"
|
||||||
|
command "temperatureDown"
|
||||||
|
command "heatingSetpointUp"
|
||||||
|
command "heatingSetpointDown"
|
||||||
|
command "on"
|
||||||
|
command "endManualControl"
|
||||||
|
command "emergencyHeat"
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
// simulator metadata
|
||||||
|
simulator {
|
||||||
|
// status messages
|
||||||
|
|
||||||
|
// reply messages
|
||||||
|
}
|
||||||
|
|
||||||
|
tiles(scale: 2){
|
||||||
|
multiAttributeTile(name: "thermostat", type:"thermostat", width:6, height:4) {
|
||||||
|
tileAttribute("device.temperature", key:"PRIMARY_CONTROL", canChangeIcon: true, canChangeBackground: true){
|
||||||
|
attributeState "default", label:'${currentValue}°', backgroundColor:"#fab907", icon:"st.Home.home1"
|
||||||
|
}
|
||||||
|
tileAttribute("device.temperature", key: "VALUE_CONTROL") {
|
||||||
|
attributeState("VALUE_UP", action: "temperatureUp")
|
||||||
|
attributeState("VALUE_DOWN", action: "temperatureDown")
|
||||||
|
}
|
||||||
|
tileAttribute("device.humidity", key: "SECONDARY_CONTROL") {
|
||||||
|
attributeState("default", label:'${currentValue}%', unit:"%")
|
||||||
|
}
|
||||||
|
tileAttribute("device.thermostatOperatingState", key: "OPERATING_STATE") {
|
||||||
|
attributeState("idle", backgroundColor:"#666666")
|
||||||
|
attributeState("heating", backgroundColor:"#ff471a")
|
||||||
|
attributeState("emergency heat", backgroundColor:"#ff471a")
|
||||||
|
}
|
||||||
|
tileAttribute("device.thermostatMode", key: "THERMOSTAT_MODE") {
|
||||||
|
attributeState("off", label:'${name}')
|
||||||
|
attributeState("heat", label:'${name}')
|
||||||
|
}
|
||||||
|
tileAttribute("device.heatingSetpoint", key: "HEATING_SETPOINT") {
|
||||||
|
attributeState("default", label:'${currentValue}', unit:"dF")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
valueTile("heatingSetpoint", "device.heatingSetpoint", width: 2, height: 1, decoration: "flat") {
|
||||||
|
state "default", label: 'Set Point\r\n\${currentValue}°'
|
||||||
|
}
|
||||||
|
|
||||||
|
standardTile("tadoMode", "device.tadoMode", width: 2, height: 2, canChangeIcon: true, canChangeBackground: true) {
|
||||||
|
state("SLEEP", label:'${name}', backgroundColor:"#0164a8", icon:"st.Bedroom.bedroom2")
|
||||||
|
state("HOME", label:'${name}', backgroundColor:"#fab907", icon:"st.Home.home2")
|
||||||
|
state("AWAY", label:'${name}', backgroundColor:"#62aa12", icon:"st.Outdoor.outdoor18")
|
||||||
|
state("OFF", label:'', backgroundColor:"#ffffff", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado.Heating.src/Images/hvac_off.png", defaultState: true)
|
||||||
|
state("MANUAL", label:'${name}', backgroundColor:"#804000", icon:"st.Weather.weather1")
|
||||||
|
}
|
||||||
|
|
||||||
|
standardTile("thermostatMode", "device.thermostatMode", width: 2, height: 2, canChangeIcon: true, canChangeBackground: true) {
|
||||||
|
state("heat", label:'HEAT', backgroundColor:"#ea2a2a", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado.Heating.src/Images/heat_mode_icon.png")
|
||||||
|
state("off", label:'', backgroundColor:"#ffffff", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado.Heating.src/Images/hvac_off.png", defaultState: true)
|
||||||
|
}
|
||||||
|
|
||||||
|
standardTile("refresh", "device.switch", width: 2, height: 1, decoration: "flat") {
|
||||||
|
state "default", label:"", action:"refresh.refresh", icon:"st.secondary.refresh"
|
||||||
|
}
|
||||||
|
standardTile("Off", "device.thermostat", width: 2, height: 1, decoration: "flat") {
|
||||||
|
state "default", label:"", action:"thermostat.off", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado.Heating.src/Images/hvac_off.png"
|
||||||
|
}
|
||||||
|
standardTile("emergencyHeat", "device.thermostat", width: 2, height: 1, decoration: "flat") {
|
||||||
|
state "default", label:"", action:"thermostat.emergencyHeat", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado.Heating.src/Images/emergencyHeat.png"
|
||||||
|
}
|
||||||
|
valueTile("outsidetemperature", "device.outsidetemperature", width: 2, height: 1, decoration: "flat") {
|
||||||
|
state "outsidetemperature", label: 'Outside Temp\r\n${currentValue}°'
|
||||||
|
}
|
||||||
|
standardTile("heat", "device.thermostat", width: 2, height: 1, decoration: "flat") {
|
||||||
|
state "default", label:"", action:"thermostat.heat", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado.Heating.src/Images/hvac_heat.png"
|
||||||
|
}
|
||||||
|
standardTile("heatingSetpointUp", "device.heatingSetpoint", width: 1, height: 1, canChangeIcon: false, decoration: "flat") {
|
||||||
|
state "heatingSetpointUp", label:'', action:"heatingSetpointUp", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado.Heating.src/Images/heat_arrow_up.png"
|
||||||
|
}
|
||||||
|
standardTile("heatingSetpointDown", "device.heatingSetpoint", width: 1, height: 1, canChangeIcon: false, decoration: "flat") {
|
||||||
|
state "heatingSetpointDown", label:'', action:"heatingSetpointDown", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado.Heating.src/Images/heat_arrow_down.png"
|
||||||
|
}
|
||||||
|
standardTile("endManualControl", "device.thermostat", width: 2, height: 1, canChangeIcon: false, canChangeBackground: true, decoration: "flat") {
|
||||||
|
state("default", label:'', action:"endManualControl", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado.Heating.src/Images/endManual.png")
|
||||||
|
}
|
||||||
|
main "thermostat"
|
||||||
|
details (["thermostat","thermostatMode","outsidetemperature","heatingSetpoint","refresh","heatingSetpointUp","heatingSetpointDown","tadoMode","emergencyHeat","heat","Off","endManualControl"])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def getWeather(){
|
||||||
|
parent.weatherStatusCommand(this)
|
||||||
|
}
|
||||||
|
|
||||||
|
def setCapabilitySupportsHeat(value){
|
||||||
|
state.supportsHeat = value
|
||||||
|
log.debug("state.supportsHeat = ${state.supportsHeat}")
|
||||||
|
}
|
||||||
|
|
||||||
|
def getCapabilitySupportsHeat() {
|
||||||
|
def map = null
|
||||||
|
map = [name: "capabilitySupportsHeat", value: state.supportsHeat]
|
||||||
|
return map
|
||||||
|
}
|
||||||
|
|
||||||
|
def updated(){
|
||||||
|
refresh()
|
||||||
|
}
|
||||||
|
|
||||||
|
def installed(){
|
||||||
|
refresh()
|
||||||
|
}
|
||||||
|
|
||||||
|
def poll() {
|
||||||
|
log.debug "Executing 'poll'"
|
||||||
|
refresh()
|
||||||
|
}
|
||||||
|
|
||||||
|
def refresh() {
|
||||||
|
log.debug "Executing 'refresh'"
|
||||||
|
parent.statusCommand(this)
|
||||||
|
getWeather()
|
||||||
|
}
|
||||||
|
|
||||||
|
def auto() {
|
||||||
|
log.debug "Executing 'auto'"
|
||||||
|
parent.autoCommand(this)
|
||||||
|
parent.statusCommand(this)
|
||||||
|
}
|
||||||
|
|
||||||
|
def on() {
|
||||||
|
log.debug "Executing 'on'"
|
||||||
|
parent.onCommand(this)
|
||||||
|
parent.statusCommand(this)
|
||||||
|
}
|
||||||
|
|
||||||
|
def off() {
|
||||||
|
log.debug "Executing 'off'"
|
||||||
|
parent.offCommand(this)
|
||||||
|
parent.statusCommand(this)
|
||||||
|
}
|
||||||
|
|
||||||
|
def setHeatingSetpoint(targetTemperature) {
|
||||||
|
log.debug "Executing 'setHeatingSetpoint'"
|
||||||
|
log.debug "Target Temperature ${targetTemperature}"
|
||||||
|
parent.setHeatingTempCommand(this,targetTemperature)
|
||||||
|
parent.statusCommand(this)
|
||||||
|
}
|
||||||
|
|
||||||
|
def setThermostatMode(requiredMode){
|
||||||
|
switch (requiredMode) {
|
||||||
|
case "heat":
|
||||||
|
heat()
|
||||||
|
break
|
||||||
|
case "auto":
|
||||||
|
auto()
|
||||||
|
break
|
||||||
|
case "off":
|
||||||
|
off()
|
||||||
|
break
|
||||||
|
case "emergency heat":
|
||||||
|
emergencyHeat()
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def temperatureUp(){
|
||||||
|
if (device.currentValue("thermostatMode") == "heat") {
|
||||||
|
heatingSetpointUp()
|
||||||
|
} else {
|
||||||
|
log.debug ("temperature setpoint not supported in the current thermostat mode")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def temperatureDown(){
|
||||||
|
if (device.currentValue("thermostatMode") == "heat") {
|
||||||
|
heatingSetpointDown()
|
||||||
|
} else {
|
||||||
|
log.debug ("temperature setpoint not supported in the current thermostat mode")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def heatingSetpointUp(){
|
||||||
|
log.debug "Current SetPoint Is " + (device.currentValue("thermostatSetpoint")).toString()
|
||||||
|
if ((device.currentValue("thermostatSetpoint").toInteger() - 1 ) < state.MinHeatTemp){
|
||||||
|
log.debug("cannot decrease heat setpoint, its already at the minimum level of " + state.MinHeatTemp)
|
||||||
|
} else {
|
||||||
|
int newSetpoint = (device.currentValue("thermostatSetpoint")).toInteger() + 1
|
||||||
|
log.debug "Setting heatingSetpoint up to: ${newSetpoint}"
|
||||||
|
setHeatingSetpoint(newSetpoint)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def heatingSetpointDown(){
|
||||||
|
log.debug "Current SetPoint Is " + (device.currentValue("thermostatSetpoint")).toString()
|
||||||
|
if ((device.currentValue("thermostatSetpoint").toInteger() + 1 ) > state.MaxHeatTemp){
|
||||||
|
log.debug("cannot increase heat setpoint, its already at the maximum level of " + state.MaxHeatTemp)
|
||||||
|
} else {
|
||||||
|
int newSetpoint = (device.currentValue("thermostatSetpoint")).toInteger() - 1
|
||||||
|
log.debug "Setting heatingSetpoint down to: ${newSetpoint}"
|
||||||
|
setHeatingSetpoint(newSetpoint)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// Commands to device
|
||||||
|
def heat(){
|
||||||
|
parent.heatCommand(this)
|
||||||
|
parent.statusCommand(this)
|
||||||
|
}
|
||||||
|
|
||||||
|
def emergencyHeat(){
|
||||||
|
parent.emergencyHeat(this)
|
||||||
|
}
|
||||||
|
|
||||||
|
def endManualControl(){
|
||||||
|
parent.endManualControl(this)
|
||||||
|
}
|
||||||
@@ -0,0 +1,174 @@
|
|||||||
|
/**
|
||||||
|
* Copyright 2015 Stuart Buchanan
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*
|
||||||
|
* Tado Thermostat
|
||||||
|
*
|
||||||
|
* Author: Stuart Buchanan, Based on original work by Ian M with thanks
|
||||||
|
* Date: 2015-04-28 v1.3 changed Presence tile as this was reporting a bug
|
||||||
|
* Date: 2015-04-28 v1.2 updated API call found issue where session was closed and nothing else was returned, now add number generator to input noCache statement in the query
|
||||||
|
* Date: 2015-04-27 v1.1 updated API call and added refresh function
|
||||||
|
* Date: 2015-12-04 v1.0 Initial Release
|
||||||
|
*/
|
||||||
|
|
||||||
|
import groovy.json.JsonOutput
|
||||||
|
import groovy.json.JsonSlurper
|
||||||
|
import java.util.Random
|
||||||
|
|
||||||
|
preferences {
|
||||||
|
input("username", "text", title: "Username", description: "Your Tado username")
|
||||||
|
input("password", "password", title: "Password", description: "Your Tado password")
|
||||||
|
input("tadouser", "text", title: "Tado User", description: "Your Tado User")
|
||||||
|
}
|
||||||
|
|
||||||
|
metadata {
|
||||||
|
definition (name: "Tado Heating User Presence", namespace: "fuzzysb", author: "Stuart Buchanan") {
|
||||||
|
capability "Presence Sensor"
|
||||||
|
capability "Sensor"
|
||||||
|
capability "Polling"
|
||||||
|
capability "Refresh"
|
||||||
|
|
||||||
|
command "arrived"
|
||||||
|
command "departed"
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
// simulator metadata
|
||||||
|
simulator {
|
||||||
|
status "present": "presence: present"
|
||||||
|
status "not present": "presence: not present"
|
||||||
|
}
|
||||||
|
|
||||||
|
tiles {
|
||||||
|
standardTile("presence", "device.presence", width: 2, height: 2, canChangeBackground: true) {
|
||||||
|
state("present", labelIcon:"st.presence.tile.mobile-present", backgroundColor:"#53a7c0")
|
||||||
|
state("not present", labelIcon:"st.presence.tile.mobile-not-present", backgroundColor:"#ffffff")
|
||||||
|
}
|
||||||
|
standardTile("refresh", "device.refresh", width: 2, height: 1, decoration: "flat") {
|
||||||
|
state "default", label:"", action:"refresh.refresh", icon:"st.secondary.refresh"
|
||||||
|
}
|
||||||
|
main "presence"
|
||||||
|
details (["presence","refresh"])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parse incoming device messages to generate events
|
||||||
|
private parseResponse(resp) {
|
||||||
|
def result
|
||||||
|
log.debug("Executing parseResponse: "+resp.data)
|
||||||
|
log.debug("Output status: "+resp.status)
|
||||||
|
if(resp.status == 200) {
|
||||||
|
log.debug("Executing parseResponse.successTrue")
|
||||||
|
def evtJson = new groovy.json.JsonOutput().toJson(resp.data)
|
||||||
|
def json = new JsonSlurper().parseText(evtJson)
|
||||||
|
def appuserarray = new groovy.json.JsonOutput().toJson(json.appUsers)
|
||||||
|
def list = new JsonSlurper().parseText(appuserarray)
|
||||||
|
list.each {
|
||||||
|
if ((it.nickname).capitalize() == (settings.tadouser).capitalize()) {
|
||||||
|
log.debug("Found Tado User : " + it.nickname)
|
||||||
|
if (it.geoTrackingEnabled == true) {
|
||||||
|
log.debug("Users GeoTracking is Enabled")
|
||||||
|
if (it.geolocationIsStale == false){
|
||||||
|
log.debug("Users Current Relative Position is : " + it.relativePosition )
|
||||||
|
if (it.relativePosition == 0) {
|
||||||
|
result = arrived()
|
||||||
|
}else{
|
||||||
|
result = departed()
|
||||||
|
}
|
||||||
|
}else{
|
||||||
|
log.debug("Geolocation is Stale Skipping")
|
||||||
|
}
|
||||||
|
}else{
|
||||||
|
log.debug("Users GeoTracking Not Enabled")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}else if(resp.status == 201){
|
||||||
|
log.debug("Something was created/updated")
|
||||||
|
}
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
||||||
|
def installed() {
|
||||||
|
log.debug "Executing 'installed'"
|
||||||
|
statusCommand()
|
||||||
|
}
|
||||||
|
|
||||||
|
def updated() {
|
||||||
|
log.debug "Executing 'updated'"
|
||||||
|
statusCommand()
|
||||||
|
}
|
||||||
|
|
||||||
|
def poll() {
|
||||||
|
log.debug "Executing 'poll'"
|
||||||
|
refresh()
|
||||||
|
}
|
||||||
|
|
||||||
|
def refresh() {
|
||||||
|
log.debug "Executing 'refresh'"
|
||||||
|
statusCommand()
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private sendCommand(method, args = []) {
|
||||||
|
def methods = [
|
||||||
|
'status': [
|
||||||
|
uri: "https://my.tado.com",
|
||||||
|
path: "/mobile/1.6/getAppUsersRelativePositions",
|
||||||
|
requestContentType: "application/json",
|
||||||
|
query: [username:settings.username, password:settings.password, noCache:args[0], webapp:1]
|
||||||
|
],
|
||||||
|
]
|
||||||
|
|
||||||
|
def request = methods.getAt(method)
|
||||||
|
|
||||||
|
log.debug "Http Params ("+request+")"
|
||||||
|
|
||||||
|
try{
|
||||||
|
log.debug "Executing 'sendCommand'"
|
||||||
|
|
||||||
|
if (method == "status"){
|
||||||
|
httpGet(request) { resp ->
|
||||||
|
parseResponse(resp)
|
||||||
|
}
|
||||||
|
}else{
|
||||||
|
httpGet(request)
|
||||||
|
}
|
||||||
|
} catch(Exception e){
|
||||||
|
log.debug("___exception: " + e)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// Commands
|
||||||
|
def statusCommand(){
|
||||||
|
log.debug "Executing 'sendCommand.statusCommand'"
|
||||||
|
Random rand = new Random()
|
||||||
|
int min = 10000
|
||||||
|
int max = 99999
|
||||||
|
int randomNum = rand.nextInt((max - min) + 1) + min
|
||||||
|
sendCommand("status",[randomNum])
|
||||||
|
}
|
||||||
|
|
||||||
|
def arrived() {
|
||||||
|
log.trace "Executing 'arrived'"
|
||||||
|
def result = sendEvent(name: "presence", value: "present")
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def departed() {
|
||||||
|
log.trace "Executing 'departed'"
|
||||||
|
def result = sendEvent(name: "presence", value: "not present")
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
||||||
@@ -0,0 +1,280 @@
|
|||||||
|
/**
|
||||||
|
* 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.
|
||||||
|
*
|
||||||
|
* Tado Thermostat
|
||||||
|
*
|
||||||
|
* Author: Stuart Buchanan, Based on original work by Ian M with thanks. also source for icons was from @tonesto7's excellent Nest Manager.
|
||||||
|
* Date: 2016-11-28 v1.6 Moved all data collection functions into Tado (Connect) SmartApp, huge changes to device handler, existing devices and handler will need to be uninstalled before installing this version
|
||||||
|
* Date: 2016-07-13 v1.5 Quick dirty workaround to control zones with a single account.
|
||||||
|
* Date: 2016-04-25 v1.4 Tado Hot water does not actually return the current water temps, it only returns the Current set point temp. to get around this when the power is on for the hot water botht the temp and setpoint will both display the setpoint value, otherwise will display --
|
||||||
|
* Date: 2016-04-25 v1.3 Finally found time to update this with the lessons learnt from the Tado Cooling Device Type. will bring better support for RM and Thermostat Director
|
||||||
|
* Date: 2016-04-08 v1.2 added setThermostatMode(mode) function to work better with Rule Machine and Thermostat Mode Director
|
||||||
|
* Date: 2016-04-05 v1.1 change of default Water Heating Temps can now be defined in device preferences (default Value is 90C).
|
||||||
|
* Date: 2016-04-05 v1.0 Initial release
|
||||||
|
*/
|
||||||
|
|
||||||
|
preferences {
|
||||||
|
input("username", "text", title: "Username", description: "Your Tado username")
|
||||||
|
input("password", "password", title: "Password", description: "Your Tado password")
|
||||||
|
input("tadoZoneId", "number", title: "Enter Tado Zone ID?", required: true)
|
||||||
|
input("manualmode", "enum", title: "Default Manual Overide Method", options: ["TADO_MODE","MANUAL"], required: false, defaultValue:"TADO_MODE")
|
||||||
|
input("defWaterTemp", "number", title: "Default Water Heating Temperature", required: false, defaultValue: 90)
|
||||||
|
}
|
||||||
|
|
||||||
|
metadata {
|
||||||
|
definition (name: "Tado Hot Water Control", namespace: "fuzzysb", author: "Stuart Buchanan") {
|
||||||
|
capability "Actuator"
|
||||||
|
capability "Temperature Measurement"
|
||||||
|
capability "Thermostat Heating Setpoint"
|
||||||
|
capability "Thermostat Setpoint"
|
||||||
|
capability "Thermostat Mode"
|
||||||
|
capability "Thermostat Operating State"
|
||||||
|
capability "Thermostat"
|
||||||
|
capability "Polling"
|
||||||
|
capability "Refresh"
|
||||||
|
attribute "tadoMode", "string"
|
||||||
|
command "temperatureUp"
|
||||||
|
command "temperatureDown"
|
||||||
|
command "heatingSetpointUp"
|
||||||
|
command "heatingSetpointDown"
|
||||||
|
command "on"
|
||||||
|
command "endManualControl"
|
||||||
|
command "emergencyHeat"
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
// simulator metadata
|
||||||
|
simulator {
|
||||||
|
// status messages
|
||||||
|
|
||||||
|
// reply messages
|
||||||
|
}
|
||||||
|
|
||||||
|
tiles(scale: 2){
|
||||||
|
multiAttributeTile(name: "thermostat", type:"thermostat", width:6, height:4) {
|
||||||
|
tileAttribute("device.temperature", key:"PRIMARY_CONTROL", canChangeIcon: true, canChangeBackground: true){
|
||||||
|
attributeState "default", label:'${currentValue}°', backgroundColor:"#fab907", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado.Hot.Water.src/Images/tap_icon.png"
|
||||||
|
}
|
||||||
|
tileAttribute("device.temperature", key: "VALUE_CONTROL") {
|
||||||
|
attributeState("VALUE_UP", action: "temperatureUp")
|
||||||
|
attributeState("VALUE_DOWN", action: "temperatureDown")
|
||||||
|
}
|
||||||
|
tileAttribute("device.thermostatOperatingState", key: "OPERATING_STATE") {
|
||||||
|
attributeState("idle", backgroundColor:"#666666")
|
||||||
|
attributeState("heating", backgroundColor:"#ff471a")
|
||||||
|
attributeState("emergency heat", backgroundColor:"#ff471a")
|
||||||
|
}
|
||||||
|
tileAttribute("device.thermostatMode", key: "THERMOSTAT_MODE") {
|
||||||
|
attributeState("off", label:'${name}')
|
||||||
|
attributeState("heat", label:'${name}')
|
||||||
|
}
|
||||||
|
tileAttribute("device.heatingSetpoint", key: "HEATING_SETPOINT") {
|
||||||
|
attributeState("default", label:'${currentValue}', unit:"dF")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
valueTile("heatingSetpoint", "device.heatingSetpoint", width: 2, height: 1, decoration: "flat") {
|
||||||
|
state "default", label: 'Set Point\r\n\${currentValue}°'
|
||||||
|
}
|
||||||
|
|
||||||
|
standardTile("tadoMode", "device.tadoMode", width: 2, height: 2, canChangeIcon: true, canChangeBackground: true) {
|
||||||
|
state("SLEEP", label:'${name}', backgroundColor:"#0164a8", icon:"st.Bedroom.bedroom2")
|
||||||
|
state("HOME", label:'${name}', backgroundColor:"#fab907", icon:"st.Home.home2")
|
||||||
|
state("AWAY", label:'${name}', backgroundColor:"#62aa12", icon:"st.Outdoor.outdoor18")
|
||||||
|
state("OFF", label:'', backgroundColor:"#ffffff", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado.Hot.Water.src/Images/hvac_off.png", defaultState: true)
|
||||||
|
state("MANUAL", label:'${name}', backgroundColor:"#804000", icon:"st.Weather.weather1")
|
||||||
|
}
|
||||||
|
|
||||||
|
standardTile("thermostatMode", "device.thermostatMode", width: 2, height: 2, canChangeIcon: true, canChangeBackground: true) {
|
||||||
|
state("heat", label:'HEAT', backgroundColor:"#ea2a2a", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado.Hot.Water.src/Images/heat_mode_icon.png")
|
||||||
|
state("off", label:'', backgroundColor:"#ffffff", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado.Hot.Water.src/Images/hvac_off.png", defaultState: true)
|
||||||
|
}
|
||||||
|
|
||||||
|
standardTile("refresh", "device.switch", width: 2, height: 1, decoration: "flat") {
|
||||||
|
state "default", label:"", action:"refresh.refresh", icon:"st.secondary.refresh"
|
||||||
|
}
|
||||||
|
standardTile("Off", "device.thermostat", width: 2, height: 1, decoration: "flat") {
|
||||||
|
state "default", label:"", action:"thermostat.off", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado.Hot.Water.src/Images/hvac_off.png"
|
||||||
|
}
|
||||||
|
standardTile("emergencyHeat", "device.thermostat", width: 2, height: 1, decoration: "flat") {
|
||||||
|
state "default", label:"", action:"thermostat.emergencyHeat", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado.Hot.Water.src/Images/emergencyHeat.png"
|
||||||
|
}
|
||||||
|
valueTile("outsidetemperature", "device.outsidetemperature", width: 2, height: 1, decoration: "flat") {
|
||||||
|
state "outsidetemperature", label: 'Outside Temp\r\n${currentValue}°'
|
||||||
|
}
|
||||||
|
standardTile("heat", "device.thermostat", width: 2, height: 1, decoration: "flat") {
|
||||||
|
state "default", label:"", action:"thermostat.heat", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado.Hot.Water.src/Images/hvac_heat.png"
|
||||||
|
}
|
||||||
|
standardTile("heatingSetpointUp", "device.heatingSetpoint", width: 1, height: 1, canChangeIcon: false, decoration: "flat") {
|
||||||
|
state "heatingSetpointUp", label:'', action:"heatingSetpointUp", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado.Hot.Water.src/Images/heat_arrow_up.png"
|
||||||
|
}
|
||||||
|
standardTile("heatingSetpointDown", "device.heatingSetpoint", width: 1, height: 1, canChangeIcon: false, decoration: "flat") {
|
||||||
|
state "heatingSetpointDown", label:'', action:"heatingSetpointDown", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado.Hot.Water.src/Images/heat_arrow_down.png"
|
||||||
|
}
|
||||||
|
standardTile("endManualControl", "device.thermostat", width: 2, height: 1, canChangeIcon: false, canChangeBackground: true, decoration: "flat") {
|
||||||
|
state("default", label:'', action:"endManualControl", icon:"https://raw.githubusercontent.com/fuzzysb/SmartThings/master/DeviceTypes/fuzzysb/tado.Hot.Water.src/Images/endManual.png")
|
||||||
|
}
|
||||||
|
main "thermostat"
|
||||||
|
details (["thermostat","thermostatMode","outsidetemperature","heatingSetpoint","refresh","heatingSetpointUp","heatingSetpointDown","tadoMode","emergencyHeat","heat","Off","endManualControl"])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def getWeather(){
|
||||||
|
parent.weatherStatusCommand(this)
|
||||||
|
}
|
||||||
|
|
||||||
|
def setCapabilitySupportsWater(value){
|
||||||
|
state.supportsWater = value
|
||||||
|
log.debug("state.supportsWater = ${state.supportsWater}")
|
||||||
|
}
|
||||||
|
|
||||||
|
def getCapabilitySupportsWater() {
|
||||||
|
def map = null
|
||||||
|
map = [name: "capabilitySupportsWater", value: state.supportsWater]
|
||||||
|
return map
|
||||||
|
}
|
||||||
|
|
||||||
|
def setCapabilitySupportsWaterTempControl(value){
|
||||||
|
state.supportsWaterTempControl = value
|
||||||
|
log.debug("state.supportsWaterTempControl = ${state.supportsWaterTempControl}")
|
||||||
|
}
|
||||||
|
|
||||||
|
def getCapabilitySupportsWaterTempControl() {
|
||||||
|
def map = null
|
||||||
|
map = [name: "capabilitySupportsWaterTempControl", value: state.supportsWaterTempControl]
|
||||||
|
return map
|
||||||
|
}
|
||||||
|
|
||||||
|
def updated(){
|
||||||
|
refresh()
|
||||||
|
}
|
||||||
|
|
||||||
|
def installed(){
|
||||||
|
refresh()
|
||||||
|
}
|
||||||
|
|
||||||
|
def poll() {
|
||||||
|
log.debug "Executing 'poll'"
|
||||||
|
refresh()
|
||||||
|
}
|
||||||
|
|
||||||
|
def refresh() {
|
||||||
|
log.debug "Executing 'refresh'"
|
||||||
|
parent.statusCommand(this)
|
||||||
|
getWeather()
|
||||||
|
}
|
||||||
|
|
||||||
|
def auto() {
|
||||||
|
log.debug "Executing 'auto'"
|
||||||
|
parent.autoCommand(this)
|
||||||
|
parent.statusCommand(this)
|
||||||
|
}
|
||||||
|
|
||||||
|
def on() {
|
||||||
|
log.debug "Executing 'on'"
|
||||||
|
onCommand()
|
||||||
|
statusCommand()
|
||||||
|
}
|
||||||
|
|
||||||
|
def off() {
|
||||||
|
log.debug "Executing 'off'"
|
||||||
|
offCommand()
|
||||||
|
statusCommand()
|
||||||
|
}
|
||||||
|
|
||||||
|
def setHeatingSetpoint(targetTemperature) {
|
||||||
|
log.debug "Executing 'setHeatingSetpoint'"
|
||||||
|
log.debug "Target Temperature ${targetTemperature}"
|
||||||
|
setHeatingTempCommand(targetTemperature)
|
||||||
|
statusCommand()
|
||||||
|
}
|
||||||
|
|
||||||
|
def temperatureUp(){
|
||||||
|
if (device.currentValue("thermostatMode") == "heat") {
|
||||||
|
heatingSetpointUp()
|
||||||
|
} else {
|
||||||
|
log.debug ("temperature setpoint not supported in the current thermostat mode")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def temperatureDown(){
|
||||||
|
if (device.currentValue("thermostatMode") == "heat") {
|
||||||
|
heatingSetpointDown()
|
||||||
|
} else {
|
||||||
|
log.debug ("temperature setpoint not supported in the current thermostat mode")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def heatingSetpointUp(){
|
||||||
|
log.debug "Current SetPoint Is " + (device.currentValue("thermostatSetpoint")).toString()
|
||||||
|
if(state.supportsWaterTempControl == "true"){
|
||||||
|
if ((device.currentValue("thermostatSetpoint").toInteger() - 1 ) < state.MinHeatTemp){
|
||||||
|
log.debug("cannot decrease heat setpoint, its already at the minimum level of " + state.MinHeatTemp)
|
||||||
|
} else {
|
||||||
|
int newSetpoint = (device.currentValue("thermostatSetpoint")).toInteger() + 1
|
||||||
|
log.debug "Setting heatingSetpoint up to: ${newSetpoint}"
|
||||||
|
setHeatingSetpoint(newSetpoint)
|
||||||
|
statusCommand()
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
log.debug "Hot Water Temperature Capability Not Supported"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def heatingSetpointDown(){
|
||||||
|
log.debug "Current SetPoint Is " + (device.currentValue("thermostatSetpoint")).toString()
|
||||||
|
if(state.supportsWaterTempControl == "true"){
|
||||||
|
if ((device.currentValue("thermostatSetpoint").toInteger() + 1 ) > state.MaxHeatTemp){
|
||||||
|
log.debug("cannot increase heat setpoint, its already at the maximum level of " + state.MaxHeatTemp)
|
||||||
|
} else {
|
||||||
|
int newSetpoint = (device.currentValue("thermostatSetpoint")).toInteger() - 1
|
||||||
|
log.debug "Setting heatingSetpoint down to: ${newSetpoint}"
|
||||||
|
setHeatingSetpoint(newSetpoint)
|
||||||
|
statusCommand()
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
log.debug "Hot Water Temperature Capability Not Supported"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Commands to device
|
||||||
|
|
||||||
|
|
||||||
|
def setThermostatMode(requiredMode){
|
||||||
|
switch (requiredMode) {
|
||||||
|
case "heat":
|
||||||
|
heat()
|
||||||
|
break
|
||||||
|
case "auto":
|
||||||
|
auto()
|
||||||
|
break
|
||||||
|
case "off":
|
||||||
|
off()
|
||||||
|
break
|
||||||
|
case "emergency heat":
|
||||||
|
emergencyHeat()
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def heat(){
|
||||||
|
parent.heatCommand(this)
|
||||||
|
parent.statusCommand(this)
|
||||||
|
}
|
||||||
|
|
||||||
|
def emergencyHeat(){
|
||||||
|
parent.emergencyHeat(this)
|
||||||
|
}
|
||||||
|
|
||||||
|
def endManualControl(){
|
||||||
|
parent.endManualControl(this)
|
||||||
|
}
|
||||||
@@ -15,6 +15,7 @@ metadata {
|
|||||||
definition (name: "Aeon Key Fob", namespace: "smartthings", author: "SmartThings") {
|
definition (name: "Aeon Key Fob", namespace: "smartthings", author: "SmartThings") {
|
||||||
capability "Actuator"
|
capability "Actuator"
|
||||||
capability "Button"
|
capability "Button"
|
||||||
|
capability "Holdable Button"
|
||||||
capability "Configuration"
|
capability "Configuration"
|
||||||
capability "Sensor"
|
capability "Sensor"
|
||||||
capability "Battery"
|
capability "Battery"
|
||||||
@@ -118,3 +119,16 @@ def configure() {
|
|||||||
log.debug("Sending configuration: $cmd")
|
log.debug("Sending configuration: $cmd")
|
||||||
return cmd
|
return cmd
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def installed() {
|
||||||
|
initialize()
|
||||||
|
}
|
||||||
|
|
||||||
|
def updated() {
|
||||||
|
initialize()
|
||||||
|
}
|
||||||
|
|
||||||
|
def initialize() {
|
||||||
|
sendEvent(name: "numberOfButtons", value: 4)
|
||||||
|
}
|
||||||
|
|||||||
@@ -15,6 +15,7 @@ metadata {
|
|||||||
definition (name: "Aeon Minimote", namespace: "smartthings", author: "SmartThings") {
|
definition (name: "Aeon Minimote", namespace: "smartthings", author: "SmartThings") {
|
||||||
capability "Actuator"
|
capability "Actuator"
|
||||||
capability "Button"
|
capability "Button"
|
||||||
|
capability "Holdable Button"
|
||||||
capability "Configuration"
|
capability "Configuration"
|
||||||
capability "Sensor"
|
capability "Sensor"
|
||||||
|
|
||||||
@@ -107,3 +108,16 @@ def configure() {
|
|||||||
log.debug("Sending configuration: $cmds")
|
log.debug("Sending configuration: $cmds")
|
||||||
return cmds
|
return cmds
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def installed() {
|
||||||
|
initialize()
|
||||||
|
}
|
||||||
|
|
||||||
|
def updated() {
|
||||||
|
initialize()
|
||||||
|
}
|
||||||
|
|
||||||
|
def initialize() {
|
||||||
|
sendEvent(name: "numberOfButtons", value: 4)
|
||||||
|
}
|
||||||
|
|||||||
@@ -27,7 +27,7 @@ metadata {
|
|||||||
capability "Switch"
|
capability "Switch"
|
||||||
capability "Refresh"
|
capability "Refresh"
|
||||||
capability "Music Player"
|
capability "Music Player"
|
||||||
capability "Polling"
|
capability "Health Check"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Define all commands, ie, if you have a custom action not
|
* Define all commands, ie, if you have a custom action not
|
||||||
@@ -236,7 +236,33 @@ def parse(String event) {
|
|||||||
* @return action(s) to take or null
|
* @return action(s) to take or null
|
||||||
*/
|
*/
|
||||||
def installed() {
|
def installed() {
|
||||||
onAction("refresh")
|
// Notify health check about this device with timeout interval 12 minutes
|
||||||
|
sendEvent(name: "checkInterval", value: 12 * 60, data: [protocol: "lan", hubHardwareId: device.hub.hardwareID], displayed: false)
|
||||||
|
startPoll()
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Called by health check if no events been generated in the last 12 minutes
|
||||||
|
* If device doesn't respond it will be marked offline (not available)
|
||||||
|
*/
|
||||||
|
def ping() {
|
||||||
|
TRACE("ping")
|
||||||
|
boseSendGetNowPlaying()
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Schedule a 2 minute poll of the device to refresh the
|
||||||
|
* tiles so the user gets the correct information.
|
||||||
|
*/
|
||||||
|
def startPoll() {
|
||||||
|
TRACE("startPoll")
|
||||||
|
unschedule()
|
||||||
|
// Schedule 2 minute polling of speaker status (song average length is 3-4 minutes)
|
||||||
|
def sec = Math.round(Math.floor(Math.random() * 60))
|
||||||
|
//def cron = "$sec 0/5 * * * ?" // every 5 min
|
||||||
|
def cron = "$sec 0/2 * * * ?" // every 2 min
|
||||||
|
log.debug "schedule('$cron', boseSendGetNowPlaying)"
|
||||||
|
schedule(cron, boseSendGetNowPlaying)
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -316,14 +342,6 @@ def onAction(String user, data=null) {
|
|||||||
return actions
|
return actions
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Called every so often (every 5 minutes actually) to refresh the
|
|
||||||
* tiles so the user gets the correct information.
|
|
||||||
*/
|
|
||||||
def poll() {
|
|
||||||
return boseRefreshNowPlaying()
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Joins this speaker into the everywhere zone
|
* Joins this speaker into the everywhere zone
|
||||||
*/
|
*/
|
||||||
@@ -837,6 +855,10 @@ def boseRefreshNowPlaying(delay=0) {
|
|||||||
return boseGET("/now_playing")
|
return boseGET("/now_playing")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
def boseSendGetNowPlaying() {
|
||||||
|
sendHubCommand(boseGET("/now_playing"))
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Requests the list of presets
|
* Requests the list of presets
|
||||||
*
|
*
|
||||||
@@ -1014,4 +1036,8 @@ def boseGetDeviceID() {
|
|||||||
*/
|
*/
|
||||||
def getDeviceIP() {
|
def getDeviceIP() {
|
||||||
return parent.resolveDNI2Address(device.deviceNetworkId)
|
return parent.resolveDNI2Address(device.deviceNetworkId)
|
||||||
|
}
|
||||||
|
|
||||||
|
def TRACE(text) {
|
||||||
|
log.trace "${text}"
|
||||||
}
|
}
|
||||||
@@ -16,7 +16,7 @@ metadata {
|
|||||||
capability "Switch"
|
capability "Switch"
|
||||||
capability "Switch Level"
|
capability "Switch Level"
|
||||||
capability "Button"
|
capability "Button"
|
||||||
capability "Actuator"
|
capability "Actuator"
|
||||||
|
|
||||||
//fingerprint deviceId: "0x1200", inClusters: "0x77 0x86 0x75 0x73 0x85 0x72 0xEF", outClusters: "0x26"
|
//fingerprint deviceId: "0x1200", inClusters: "0x77 0x86 0x75 0x73 0x85 0x72 0xEF", outClusters: "0x26"
|
||||||
}
|
}
|
||||||
@@ -74,20 +74,20 @@ def off() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
def levelup() {
|
def levelup() {
|
||||||
def curlevel = device.currentValue('level') as Integer
|
def curlevel = device.currentValue('level') as Integer
|
||||||
if (curlevel <= 90)
|
if (curlevel <= 90)
|
||||||
setLevel(curlevel + 10);
|
setLevel(curlevel + 10);
|
||||||
}
|
}
|
||||||
|
|
||||||
def leveldown() {
|
def leveldown() {
|
||||||
def curlevel = device.currentValue('level') as Integer
|
def curlevel = device.currentValue('level') as Integer
|
||||||
if (curlevel >= 10)
|
if (curlevel >= 10)
|
||||||
setLevel(curlevel - 10)
|
setLevel(curlevel - 10)
|
||||||
}
|
}
|
||||||
|
|
||||||
def setLevel(value) {
|
def setLevel(value) {
|
||||||
log.trace "setLevel($value)"
|
log.trace "setLevel($value)"
|
||||||
sendEvent(name: "level", value: value)
|
sendEvent(name: "level", value: value)
|
||||||
}
|
}
|
||||||
|
|
||||||
def zwaveEvent(physicalgraph.zwave.commands.wakeupv1.WakeUpNotification cmd) {
|
def zwaveEvent(physicalgraph.zwave.commands.wakeupv1.WakeUpNotification cmd) {
|
||||||
@@ -106,11 +106,11 @@ def zwaveEvent(physicalgraph.zwave.commands.switchmultilevelv1.SwitchMultilevelS
|
|||||||
if (cmd.upDown == true) {
|
if (cmd.upDown == true) {
|
||||||
Integer buttonid = 2
|
Integer buttonid = 2
|
||||||
leveldown()
|
leveldown()
|
||||||
checkbuttonEvent(buttonid)
|
checkbuttonEvent(buttonid)
|
||||||
} else if (cmd.upDown == false) {
|
} else if (cmd.upDown == false) {
|
||||||
Integer buttonid = 3
|
Integer buttonid = 3
|
||||||
levelup()
|
levelup()
|
||||||
checkbuttonEvent(buttonid)
|
checkbuttonEvent(buttonid)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -140,12 +140,12 @@ def buttonEvent(button) {
|
|||||||
def result = []
|
def result = []
|
||||||
if (button == 1) {
|
if (button == 1) {
|
||||||
def mystate = device.currentValue('switch');
|
def mystate = device.currentValue('switch');
|
||||||
if (mystate == "on")
|
if (mystate == "on")
|
||||||
off()
|
off()
|
||||||
else
|
else
|
||||||
on()
|
on()
|
||||||
}
|
}
|
||||||
updateState("currentButton", "$button")
|
updateState("currentButton", "$button")
|
||||||
// update the device state, recording the button press
|
// update the device state, recording the button press
|
||||||
result << createEvent(name: "button", value: "pushed", data: [buttonNumber: button], descriptionText: "$device.displayName button $button was pushed", isStateChange: true)
|
result << createEvent(name: "button", value: "pushed", data: [buttonNumber: button], descriptionText: "$device.displayName button $button was pushed", isStateChange: true)
|
||||||
result
|
result
|
||||||
@@ -182,3 +182,16 @@ def updateState(String name, String value) {
|
|||||||
state[name] = value
|
state[name] = value
|
||||||
device.updateDataValue(name, value)
|
device.updateDataValue(name, value)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def installed() {
|
||||||
|
initialize()
|
||||||
|
}
|
||||||
|
|
||||||
|
def updated() {
|
||||||
|
initialize()
|
||||||
|
}
|
||||||
|
|
||||||
|
def initialize() {
|
||||||
|
sendEvent(name: "numberOfButtons", value: 3)
|
||||||
|
}
|
||||||
|
|||||||
@@ -236,17 +236,17 @@ def refresh() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void indicatorWhenOn() {
|
void indicatorWhenOn() {
|
||||||
sendEvent(name: "indicatorStatus", value: "when on", display: false)
|
sendEvent(name: "indicatorStatus", value: "when on", displayed: false)
|
||||||
sendHubCommand(new physicalgraph.device.HubAction(zwave.configurationV1.configurationSet(configurationValue: [1], parameterNumber: 3, size: 1).format()))
|
sendHubCommand(new physicalgraph.device.HubAction(zwave.configurationV1.configurationSet(configurationValue: [1], parameterNumber: 3, size: 1).format()))
|
||||||
}
|
}
|
||||||
|
|
||||||
void indicatorWhenOff() {
|
void indicatorWhenOff() {
|
||||||
sendEvent(name: "indicatorStatus", value: "when off", display: false)
|
sendEvent(name: "indicatorStatus", value: "when off", displayed: false)
|
||||||
sendHubCommand(new physicalgraph.device.HubAction(zwave.configurationV1.configurationSet(configurationValue: [0], parameterNumber: 3, size: 1).format()))
|
sendHubCommand(new physicalgraph.device.HubAction(zwave.configurationV1.configurationSet(configurationValue: [0], parameterNumber: 3, size: 1).format()))
|
||||||
}
|
}
|
||||||
|
|
||||||
void indicatorNever() {
|
void indicatorNever() {
|
||||||
sendEvent(name: "indicatorStatus", value: "never", display: false)
|
sendEvent(name: "indicatorStatus", value: "never", displayed: false)
|
||||||
sendHubCommand(new physicalgraph.device.HubAction(zwave.configurationV1.configurationSet(configurationValue: [2], parameterNumber: 3, size: 1).format()))
|
sendHubCommand(new physicalgraph.device.HubAction(zwave.configurationV1.configurationSet(configurationValue: [2], parameterNumber: 3, size: 1).format()))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -29,10 +29,10 @@
|
|||||||
capability "Polling"
|
capability "Polling"
|
||||||
capability "Refresh"
|
capability "Refresh"
|
||||||
capability "Sensor"
|
capability "Sensor"
|
||||||
capability "Configuration"
|
capability "Configuration"
|
||||||
capability "Color Control"
|
capability "Color Control"
|
||||||
capability "Power Meter"
|
capability "Power Meter"
|
||||||
|
|
||||||
command "getDeviceData"
|
command "getDeviceData"
|
||||||
command "softwhite"
|
command "softwhite"
|
||||||
command "daylight"
|
command "daylight"
|
||||||
@@ -54,12 +54,12 @@
|
|||||||
command "setAdjustedColor"
|
command "setAdjustedColor"
|
||||||
command "setWhiteLevel"
|
command "setWhiteLevel"
|
||||||
command "test"
|
command "test"
|
||||||
|
|
||||||
attribute "whiteLevel", "string"
|
attribute "whiteLevel", "string"
|
||||||
|
|
||||||
fingerprint deviceId: "0x1101", inClusters: "0x27,0x72,0x86,0x26,0x60,0x70,0x32,0x31,0x85,0x33"
|
fingerprint deviceId: "0x1101", inClusters: "0x27,0x72,0x86,0x26,0x60,0x70,0x32,0x31,0x85,0x33"
|
||||||
}
|
}
|
||||||
|
|
||||||
simulator {
|
simulator {
|
||||||
status "on": "command: 2003, payload: FF"
|
status "on": "command: 2003, payload: FF"
|
||||||
status "off": "command: 2003, payload: 00"
|
status "off": "command: 2003, payload: 00"
|
||||||
@@ -84,7 +84,7 @@
|
|||||||
}
|
}
|
||||||
controlTile("levelSliderControl", "device.level", "slider", height: 1, width: 2, inactiveLabel: false) {
|
controlTile("levelSliderControl", "device.level", "slider", height: 1, width: 2, inactiveLabel: false) {
|
||||||
state "level", action:"switch level.setLevel"
|
state "level", action:"switch level.setLevel"
|
||||||
}
|
}
|
||||||
controlTile("whiteSliderControl", "device.whiteLevel", "slider", height: 1, width: 3, inactiveLabel: false) {
|
controlTile("whiteSliderControl", "device.whiteLevel", "slider", height: 1, width: 3, inactiveLabel: false) {
|
||||||
state "whiteLevel", action:"setWhiteLevel", label:'White Level'
|
state "whiteLevel", action:"setWhiteLevel", label:'White Level'
|
||||||
}
|
}
|
||||||
@@ -183,24 +183,24 @@
|
|||||||
valueTile("hue", "device.hue", inactiveLabel: false, decoration: "flat") {
|
valueTile("hue", "device.hue", inactiveLabel: false, decoration: "flat") {
|
||||||
state "hue", label: 'Hue ${currentValue} '
|
state "hue", label: 'Hue ${currentValue} '
|
||||||
}
|
}
|
||||||
|
|
||||||
main(["switch"])
|
main(["switch"])
|
||||||
details(["switch",
|
details(["switch",
|
||||||
"levelSliderControl",
|
"levelSliderControl",
|
||||||
"rgbSelector",
|
"rgbSelector",
|
||||||
"whiteSliderControl",
|
"whiteSliderControl",
|
||||||
/*"softwhite",
|
/*"softwhite",
|
||||||
"daylight",
|
"daylight",
|
||||||
"warmwhite",
|
"warmwhite",
|
||||||
"red",
|
"red",
|
||||||
"green",
|
"green",
|
||||||
"blue",
|
"blue",
|
||||||
"white",
|
"white",
|
||||||
"cyan",
|
"cyan",
|
||||||
"magenta",
|
"magenta",
|
||||||
"orange",
|
"orange",
|
||||||
"purple",
|
"purple",
|
||||||
"yellow",
|
"yellow",
|
||||||
"fireplace",
|
"fireplace",
|
||||||
"storm",
|
"storm",
|
||||||
"deepfade",
|
"deepfade",
|
||||||
@@ -214,7 +214,7 @@
|
|||||||
|
|
||||||
def setAdjustedColor(value) {
|
def setAdjustedColor(value) {
|
||||||
log.debug "setAdjustedColor: ${value}"
|
log.debug "setAdjustedColor: ${value}"
|
||||||
|
|
||||||
toggleTiles("off") //turn off the hard color tiles
|
toggleTiles("off") //turn off the hard color tiles
|
||||||
|
|
||||||
def level = device.latestValue("level")
|
def level = device.latestValue("level")
|
||||||
@@ -223,19 +223,19 @@ def setAdjustedColor(value) {
|
|||||||
log.debug "level is: ${level}"
|
log.debug "level is: ${level}"
|
||||||
value.level = level
|
value.level = level
|
||||||
|
|
||||||
def c = hexToRgb(value.hex)
|
def c = hexToRgb(value.hex)
|
||||||
value.rh = hex(c.r * (level/100))
|
value.rh = hex(c.r * (level/100))
|
||||||
value.gh = hex(c.g * (level/100))
|
value.gh = hex(c.g * (level/100))
|
||||||
value.bh = hex(c.b * (level/100))
|
value.bh = hex(c.b * (level/100))
|
||||||
|
|
||||||
setColor(value)
|
setColor(value)
|
||||||
}
|
}
|
||||||
|
|
||||||
def setColor(value) {
|
def setColor(value) {
|
||||||
log.debug "setColor: ${value}"
|
log.debug "setColor: ${value}"
|
||||||
log.debug "hue is: ${value.hue}"
|
log.debug "hue is: ${value.hue}"
|
||||||
log.debug "saturation is: ${value.saturation}"
|
log.debug "saturation is: ${value.saturation}"
|
||||||
|
|
||||||
if (value.size() < 8)
|
if (value.size() < 8)
|
||||||
toggleTiles("off")
|
toggleTiles("off")
|
||||||
|
|
||||||
@@ -246,22 +246,22 @@ def setColor(value) {
|
|||||||
value.gh = hex(rgb.g)
|
value.gh = hex(rgb.g)
|
||||||
value.bh = hex(rgb.b)
|
value.bh = hex(rgb.b)
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((value.size() == 3) && (value.hue != null) && (value.saturation != null) && (value.level)) { //user passed in a level value too from outside (App)
|
if ((value.size() == 3) && (value.hue != null) && (value.saturation != null) && (value.level)) { //user passed in a level value too from outside (App)
|
||||||
def rgb = hslToRGB(value.hue, value.saturation, 0.5)
|
def rgb = hslToRGB(value.hue, value.saturation, 0.5)
|
||||||
value.hex = rgbToHex(rgb)
|
value.hex = rgbToHex(rgb)
|
||||||
value.rh = hex(rgb.r * value.level/100)
|
value.rh = hex(rgb.r * value.level/100)
|
||||||
value.gh = hex(rgb.g * value.level/100)
|
value.gh = hex(rgb.g * value.level/100)
|
||||||
value.bh = hex(rgb.b * value.level/100)
|
value.bh = hex(rgb.b * value.level/100)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (( value.size() == 1) && (value.hex)) { //being called from outside of device (App) with only hex
|
if (( value.size() == 1) && (value.hex)) { //being called from outside of device (App) with only hex
|
||||||
def rgbInt = hexToRgb(value.hex)
|
def rgbInt = hexToRgb(value.hex)
|
||||||
value.rh = hex(rgbInt.r)
|
value.rh = hex(rgbInt.r)
|
||||||
value.gh = hex(rgbInt.g)
|
value.gh = hex(rgbInt.g)
|
||||||
value.bh = hex(rgbInt.b)
|
value.bh = hex(rgbInt.b)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (( value.size() == 2) && (value.hex) && (value.level)) { //being called from outside of device (App) with only hex and level
|
if (( value.size() == 2) && (value.hex) && (value.level)) { //being called from outside of device (App) with only hex and level
|
||||||
|
|
||||||
def rgbInt = hexToRgb(value.hex)
|
def rgbInt = hexToRgb(value.hex)
|
||||||
@@ -269,7 +269,7 @@ def setColor(value) {
|
|||||||
value.gh = hex(rgbInt.g * value.level/100)
|
value.gh = hex(rgbInt.g * value.level/100)
|
||||||
value.bh = hex(rgbInt.b * value.level/100)
|
value.bh = hex(rgbInt.b * value.level/100)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (( value.size() == 1) && (value.colorName)) { //being called from outside of device (App) with only color name
|
if (( value.size() == 1) && (value.colorName)) { //being called from outside of device (App) with only color name
|
||||||
def colorData = getColorData(value.colorName)
|
def colorData = getColorData(value.colorName)
|
||||||
value.rh = colorData.rh
|
value.rh = colorData.rh
|
||||||
@@ -277,7 +277,7 @@ def setColor(value) {
|
|||||||
value.bh = colorData.bh
|
value.bh = colorData.bh
|
||||||
value.hex = "#${value.rh}${value.gh}${value.bh}"
|
value.hex = "#${value.rh}${value.gh}${value.bh}"
|
||||||
}
|
}
|
||||||
|
|
||||||
if (( value.size() == 2) && (value.colorName) && (value.level)) { //being called from outside of device (App) with only color name and level
|
if (( value.size() == 2) && (value.colorName) && (value.level)) { //being called from outside of device (App) with only color name and level
|
||||||
def colorData = getColorData(value.colorName)
|
def colorData = getColorData(value.colorName)
|
||||||
value.rh = hex(colorData.r * value.level/100)
|
value.rh = hex(colorData.r * value.level/100)
|
||||||
@@ -285,7 +285,7 @@ def setColor(value) {
|
|||||||
value.bh = hex(colorData.b * value.level/100)
|
value.bh = hex(colorData.b * value.level/100)
|
||||||
value.hex = "#${hex(colorData.r)}${hex(colorData.g)}${hex(colorData.b)}"
|
value.hex = "#${hex(colorData.r)}${hex(colorData.g)}${hex(colorData.b)}"
|
||||||
}
|
}
|
||||||
|
|
||||||
if (( value.size() == 3) && (value.red != null) && (value.green != null) && (value.blue != null)) { //being called from outside of device (App) with only color values (0-255)
|
if (( value.size() == 3) && (value.red != null) && (value.green != null) && (value.blue != null)) { //being called from outside of device (App) with only color values (0-255)
|
||||||
value.rh = hex(value.red)
|
value.rh = hex(value.red)
|
||||||
value.gh = hex(value.green)
|
value.gh = hex(value.green)
|
||||||
@@ -299,7 +299,7 @@ def setColor(value) {
|
|||||||
value.bh = hex(value.blue * value.level/100)
|
value.bh = hex(value.blue * value.level/100)
|
||||||
value.hex = "#${hex(value.red)}${hex(value.green)}${hex(value.blue)}"
|
value.hex = "#${hex(value.red)}${hex(value.green)}${hex(value.blue)}"
|
||||||
}
|
}
|
||||||
|
|
||||||
sendEvent(name: "hue", value: value.hue, displayed: false)
|
sendEvent(name: "hue", value: value.hue, displayed: false)
|
||||||
sendEvent(name: "saturation", value: value.saturation, displayed: false)
|
sendEvent(name: "saturation", value: value.saturation, displayed: false)
|
||||||
sendEvent(name: "color", value: value.hex, displayed: false)
|
sendEvent(name: "color", value: value.hex, displayed: false)
|
||||||
@@ -309,26 +309,26 @@ def setColor(value) {
|
|||||||
if (value.switch) {
|
if (value.switch) {
|
||||||
sendEvent(name: "switch", value: value.switch)
|
sendEvent(name: "switch", value: value.switch)
|
||||||
}
|
}
|
||||||
|
|
||||||
sendRGB(value.rh, value.gh, value.bh)
|
sendRGB(value.rh, value.gh, value.bh)
|
||||||
}
|
}
|
||||||
|
|
||||||
def setLevel(level) {
|
def setLevel(level) {
|
||||||
log.debug "setLevel($level)"
|
log.debug "setLevel($level)"
|
||||||
|
|
||||||
if (level == 0) { off() }
|
if (level == 0) { off() }
|
||||||
else if (device.latestValue("switch") == "off") { on() }
|
else if (device.latestValue("switch") == "off") { on() }
|
||||||
|
|
||||||
def colorHex = device.latestValue("color")
|
def colorHex = device.latestValue("color")
|
||||||
if (colorHex == null)
|
if (colorHex == null)
|
||||||
colorHex = "#FFFFFF"
|
colorHex = "#FFFFFF"
|
||||||
|
|
||||||
def c = hexToRgb(colorHex)
|
def c = hexToRgb(colorHex)
|
||||||
|
|
||||||
def r = hex(c.r * (level/100))
|
def r = hex(c.r * (level/100))
|
||||||
def g = hex(c.g * (level/100))
|
def g = hex(c.g * (level/100))
|
||||||
def b = hex(c.b * (level/100))
|
def b = hex(c.b * (level/100))
|
||||||
|
|
||||||
sendEvent(name: "level", value: level)
|
sendEvent(name: "level", value: level)
|
||||||
sendEvent(name: "setLevel", value: level, displayed: false)
|
sendEvent(name: "setLevel", value: level, displayed: false)
|
||||||
sendRGB(r, g, b)
|
sendRGB(r, g, b)
|
||||||
@@ -337,14 +337,14 @@ def setLevel(level) {
|
|||||||
|
|
||||||
def setWhiteLevel(value) {
|
def setWhiteLevel(value) {
|
||||||
log.debug "setWhiteLevel: ${value}"
|
log.debug "setWhiteLevel: ${value}"
|
||||||
def level = Math.min(value as Integer, 99)
|
def level = Math.min(value as Integer, 99)
|
||||||
level = 255 * level/99 as Integer
|
level = 255 * level/99 as Integer
|
||||||
def channel = 0
|
def channel = 0
|
||||||
|
|
||||||
if (device.latestValue("switch") == "off") { on() }
|
if (device.latestValue("switch") == "off") { on() }
|
||||||
|
|
||||||
sendEvent(name: "whiteLevel", value: value)
|
sendEvent(name: "whiteLevel", value: value)
|
||||||
sendWhite(channel, value)
|
sendWhite(channel, value)
|
||||||
}
|
}
|
||||||
|
|
||||||
def sendWhite(channel, value) {
|
def sendWhite(channel, value) {
|
||||||
@@ -367,20 +367,20 @@ def sendRGBW(redHex, greenHex, blueHex, whiteHex) {
|
|||||||
|
|
||||||
def configure() {
|
def configure() {
|
||||||
log.debug "Configuring Device For SmartThings Use"
|
log.debug "Configuring Device For SmartThings Use"
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def cmds = []
|
def cmds = []
|
||||||
|
|
||||||
// send associate to group 3 to get sensor data reported only to hub
|
// send associate to group 3 to get sensor data reported only to hub
|
||||||
cmds << zwave.associationV2.associationSet(groupingIdentifier:5, nodeId:[zwaveHubNodeId]).format()
|
cmds << zwave.associationV2.associationSet(groupingIdentifier:5, nodeId:[zwaveHubNodeId]).format()
|
||||||
|
|
||||||
|
|
||||||
//cmds << sendEvent(name: "level", value: 50)
|
//cmds << sendEvent(name: "level", value: 50)
|
||||||
//cmds << on()
|
//cmds << on()
|
||||||
//cmds << doColorButton("Green")
|
//cmds << doColorButton("Green")
|
||||||
delayBetween(cmds, 500)
|
delayBetween(cmds, 500)
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
def parse(String description) {
|
def parse(String description) {
|
||||||
@@ -411,11 +411,11 @@ def parse(String description) {
|
|||||||
|
|
||||||
def getDeviceData() {
|
def getDeviceData() {
|
||||||
def cmd = []
|
def cmd = []
|
||||||
|
|
||||||
cmd << response(zwave.manufacturerSpecificV2.manufacturerSpecificGet())
|
cmd << response(zwave.manufacturerSpecificV2.manufacturerSpecificGet())
|
||||||
cmd << response(zwave.versionV1.versionGet())
|
cmd << response(zwave.versionV1.versionGet())
|
||||||
cmd << response(zwave.firmwareUpdateMdV1.firmwareMdGet())
|
cmd << response(zwave.firmwareUpdateMdV1.firmwareMdGet())
|
||||||
|
|
||||||
delayBetween(cmd, 500)
|
delayBetween(cmd, 500)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -426,7 +426,7 @@ def createEvent(physicalgraph.zwave.commands.manufacturerspecificv2.Manufacturer
|
|||||||
log.debug "productTypeId: ${cmd.productTypeId}"
|
log.debug "productTypeId: ${cmd.productTypeId}"
|
||||||
}
|
}
|
||||||
|
|
||||||
def createEvent(physicalgraph.zwave.commands.versionv1.VersionReport cmd, Map item1) {
|
def createEvent(physicalgraph.zwave.commands.versionv1.VersionReport cmd, Map item1) {
|
||||||
updateDataValue("applicationVersion", "${cmd.applicationVersion}")
|
updateDataValue("applicationVersion", "${cmd.applicationVersion}")
|
||||||
log.debug "applicationVersion: ${cmd.applicationVersion}"
|
log.debug "applicationVersion: ${cmd.applicationVersion}"
|
||||||
log.debug "applicationSubVersion: ${cmd.applicationSubVersion}"
|
log.debug "applicationSubVersion: ${cmd.applicationSubVersion}"
|
||||||
@@ -435,13 +435,13 @@ def createEvent(physicalgraph.zwave.commands.versionv1.VersionReport cmd, Map it
|
|||||||
log.debug "zWaveProtocolSubVersion: ${cmd.zWaveProtocolSubVersion}"
|
log.debug "zWaveProtocolSubVersion: ${cmd.zWaveProtocolSubVersion}"
|
||||||
}
|
}
|
||||||
|
|
||||||
def createEvent(physicalgraph.zwave.commands.firmwareupdatemdv1.FirmwareMdReport cmd, Map item1) {
|
def createEvent(physicalgraph.zwave.commands.firmwareupdatemdv1.FirmwareMdReport cmd, Map item1) {
|
||||||
log.debug "checksum: ${cmd.checksum}"
|
log.debug "checksum: ${cmd.checksum}"
|
||||||
log.debug "firmwareId: ${cmd.firmwareId}"
|
log.debug "firmwareId: ${cmd.firmwareId}"
|
||||||
log.debug "manufacturerId: ${cmd.manufacturerId}"
|
log.debug "manufacturerId: ${cmd.manufacturerId}"
|
||||||
}
|
}
|
||||||
|
|
||||||
def zwaveEvent(physicalgraph.zwave.commands.colorcontrolv1.CapabilityReport cmd, Map item1) {
|
def zwaveEvent(physicalgraph.zwave.commands.colorcontrolv1.CapabilityReport cmd, Map item1) {
|
||||||
|
|
||||||
log.debug "In CapabilityReport"
|
log.debug "In CapabilityReport"
|
||||||
}
|
}
|
||||||
@@ -546,7 +546,7 @@ def zwaveEvent(physicalgraph.zwave.commands.configurationv1.ConfigurationReport
|
|||||||
def value = "when off"
|
def value = "when off"
|
||||||
if (cmd.configurationValue[0] == 1) {value = "when on"}
|
if (cmd.configurationValue[0] == 1) {value = "when on"}
|
||||||
if (cmd.configurationValue[0] == 2) {value = "never"}
|
if (cmd.configurationValue[0] == 2) {value = "never"}
|
||||||
[name: "indicatorStatus", value: value, display: false]
|
[name: "indicatorStatus", value: value, displayed: false]
|
||||||
}
|
}
|
||||||
*/
|
*/
|
||||||
def createEvent(physicalgraph.zwave.Command cmd, Map map) {
|
def createEvent(physicalgraph.zwave.Command cmd, Map map) {
|
||||||
@@ -557,7 +557,7 @@ def createEvent(physicalgraph.zwave.Command cmd, Map map) {
|
|||||||
def on() {
|
def on() {
|
||||||
log.debug "on()"
|
log.debug "on()"
|
||||||
sendEvent(name: "switch", value: "on")
|
sendEvent(name: "switch", value: "on")
|
||||||
delayBetween([zwave.basicV1.basicSet(value: 0xFF).format(),
|
delayBetween([zwave.basicV1.basicSet(value: 0xFF).format(),
|
||||||
zwave.switchMultilevelV1.switchMultilevelGet().format()], 5000)
|
zwave.switchMultilevelV1.switchMultilevelGet().format()], 5000)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -593,7 +593,7 @@ def refresh() {
|
|||||||
* @return none
|
* @return none
|
||||||
*/
|
*/
|
||||||
def updateZwaveParam(params) {
|
def updateZwaveParam(params) {
|
||||||
if ( params ) {
|
if ( params ) {
|
||||||
def pNumber = params.paramNumber
|
def pNumber = params.paramNumber
|
||||||
def pSize = params.size
|
def pSize = params.size
|
||||||
def pValue = [params.value]
|
def pValue = [params.value]
|
||||||
@@ -601,9 +601,9 @@ def updateZwaveParam(params) {
|
|||||||
|
|
||||||
def cmds = []
|
def cmds = []
|
||||||
cmds << zwave.configurationV1.configurationSet(configurationValue: pValue, parameterNumber: pNumber, size: pSize).format()
|
cmds << zwave.configurationV1.configurationSet(configurationValue: pValue, parameterNumber: pNumber, size: pSize).format()
|
||||||
|
|
||||||
cmds << zwave.configurationV1.configurationGet(parameterNumber: pNumber).format()
|
cmds << zwave.configurationV1.configurationGet(parameterNumber: pNumber).format()
|
||||||
delayBetween(cmds, 1500)
|
delayBetween(cmds, 1500)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -612,22 +612,22 @@ def test() {
|
|||||||
//value = [hue: 0, saturation: 100, level: 5]
|
//value = [hue: 0, saturation: 100, level: 5]
|
||||||
//value = [red: 255, green: 0, blue: 255, level: 60]
|
//value = [red: 255, green: 0, blue: 255, level: 60]
|
||||||
//setColor(value)
|
//setColor(value)
|
||||||
|
|
||||||
def cmd = []
|
def cmd = []
|
||||||
|
|
||||||
if ( !state.cnt ) {
|
if ( !state.cnt ) {
|
||||||
state.cnt = 6
|
state.cnt = 6
|
||||||
} else {
|
} else {
|
||||||
state.cnt = state.cnt + 1
|
state.cnt = state.cnt + 1
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( state.cnt > 10 )
|
if ( state.cnt > 10 )
|
||||||
state.cnt = 6
|
state.cnt = 6
|
||||||
|
|
||||||
// run programmed light show
|
// run programmed light show
|
||||||
cmd << zwave.configurationV1.configurationSet(configurationValue: [state.cnt], parameterNumber: 72, size: 1).format()
|
cmd << zwave.configurationV1.configurationSet(configurationValue: [state.cnt], parameterNumber: 72, size: 1).format()
|
||||||
cmd << zwave.configurationV1.configurationGet(parameterNumber: 72).format()
|
cmd << zwave.configurationV1.configurationGet(parameterNumber: 72).format()
|
||||||
|
|
||||||
delayBetween(cmd, 500)
|
delayBetween(cmd, 500)
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -638,23 +638,23 @@ def colorNameToRgb(color) {
|
|||||||
[name:"Soft White", r: 255, g: 241, b: 224 ],
|
[name:"Soft White", r: 255, g: 241, b: 224 ],
|
||||||
[name:"Daylight", r: 255, g: 255, b: 251 ],
|
[name:"Daylight", r: 255, g: 255, b: 251 ],
|
||||||
[name:"Warm White", r: 255, g: 244, b: 229 ],
|
[name:"Warm White", r: 255, g: 244, b: 229 ],
|
||||||
|
|
||||||
[name:"Red", r: 255, g: 0, b: 0 ],
|
[name:"Red", r: 255, g: 0, b: 0 ],
|
||||||
[name:"Green", r: 0, g: 255, b: 0 ],
|
[name:"Green", r: 0, g: 255, b: 0 ],
|
||||||
[name:"Blue", r: 0, g: 0, b: 255 ],
|
[name:"Blue", r: 0, g: 0, b: 255 ],
|
||||||
|
|
||||||
[name:"Cyan", r: 0, g: 255, b: 255 ],
|
[name:"Cyan", r: 0, g: 255, b: 255 ],
|
||||||
[name:"Magenta", r: 255, g: 0, b: 33 ],
|
[name:"Magenta", r: 255, g: 0, b: 33 ],
|
||||||
[name:"Orange", r: 255, g: 102, b: 0 ],
|
[name:"Orange", r: 255, g: 102, b: 0 ],
|
||||||
|
|
||||||
[name:"Purple", r: 170, g: 0, b: 255 ],
|
[name:"Purple", r: 170, g: 0, b: 255 ],
|
||||||
[name:"Yellow", r: 255, g: 255, b: 0 ],
|
[name:"Yellow", r: 255, g: 255, b: 0 ],
|
||||||
[name:"White", r: 255, g: 255, b: 255 ]
|
[name:"White", r: 255, g: 255, b: 255 ]
|
||||||
]
|
]
|
||||||
|
|
||||||
def colorData = [:]
|
def colorData = [:]
|
||||||
colorData = colors.find { it.name == color }
|
colorData = colors.find { it.name == color }
|
||||||
|
|
||||||
colorData
|
colorData
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -670,7 +670,7 @@ def hexToRgb(colorHex) {
|
|||||||
def rrInt = Integer.parseInt(colorHex.substring(1,3),16)
|
def rrInt = Integer.parseInt(colorHex.substring(1,3),16)
|
||||||
def ggInt = Integer.parseInt(colorHex.substring(3,5),16)
|
def ggInt = Integer.parseInt(colorHex.substring(3,5),16)
|
||||||
def bbInt = Integer.parseInt(colorHex.substring(5,7),16)
|
def bbInt = Integer.parseInt(colorHex.substring(5,7),16)
|
||||||
|
|
||||||
def colorData = [:]
|
def colorData = [:]
|
||||||
colorData = [r: rrInt, g: ggInt, b: bbInt]
|
colorData = [r: rrInt, g: ggInt, b: bbInt]
|
||||||
colorData
|
colorData
|
||||||
@@ -681,7 +681,7 @@ def rgbToHex(rgb) {
|
|||||||
def g = hex(rgb.g)
|
def g = hex(rgb.g)
|
||||||
def b = hex(rgb.b)
|
def b = hex(rgb.b)
|
||||||
def hexColor = "#${r}${g}${b}"
|
def hexColor = "#${r}${g}${b}"
|
||||||
|
|
||||||
hexColor
|
hexColor
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -689,11 +689,11 @@ def hslToRGB(float var_h, float var_s, float var_l) {
|
|||||||
float h = var_h / 100
|
float h = var_h / 100
|
||||||
float s = var_s / 100
|
float s = var_s / 100
|
||||||
float l = var_l
|
float l = var_l
|
||||||
|
|
||||||
def r = 0
|
def r = 0
|
||||||
def g = 0
|
def g = 0
|
||||||
def b = 0
|
def b = 0
|
||||||
|
|
||||||
if (s == 0) {
|
if (s == 0) {
|
||||||
r = l * 255
|
r = l * 255
|
||||||
g = l * 255
|
g = l * 255
|
||||||
@@ -705,26 +705,26 @@ def hslToRGB(float var_h, float var_s, float var_l) {
|
|||||||
} else {
|
} else {
|
||||||
var_2 = (l + s) - (s * l)
|
var_2 = (l + s) - (s * l)
|
||||||
}
|
}
|
||||||
|
|
||||||
float var_1 = 2 * l - var_2
|
float var_1 = 2 * l - var_2
|
||||||
|
|
||||||
r = 255 * hueToRgb(var_1, var_2, h + (1 / 3))
|
r = 255 * hueToRgb(var_1, var_2, h + (1 / 3))
|
||||||
g = 255 * hueToRgb(var_1, var_2, h)
|
g = 255 * hueToRgb(var_1, var_2, h)
|
||||||
b = 255 * hueToRgb(var_1, var_2, h - (1 / 3))
|
b = 255 * hueToRgb(var_1, var_2, h - (1 / 3))
|
||||||
}
|
}
|
||||||
|
|
||||||
def rgb = [:]
|
def rgb = [:]
|
||||||
rgb = [r: r, g: g, b: b]
|
rgb = [r: r, g: g, b: b]
|
||||||
|
|
||||||
rgb
|
rgb
|
||||||
}
|
}
|
||||||
|
|
||||||
def hueToRgb(v1, v2, vh) {
|
def hueToRgb(v1, v2, vh) {
|
||||||
if (vh < 0) { vh += 1 }
|
if (vh < 0) { vh += 1 }
|
||||||
if (vh > 1) { vh -= 1 }
|
if (vh > 1) { vh -= 1 }
|
||||||
if ((6 * vh) < 1) { return (v1 + (v2 - v1) * 6 * vh) }
|
if ((6 * vh) < 1) { return (v1 + (v2 - v1) * 6 * vh) }
|
||||||
if ((2 * vh) < 1) { return (v2) }
|
if ((2 * vh) < 1) { return (v2) }
|
||||||
if ((3 * vh) < 2) { return (v1 + (v2 - $v1) * ((2 / 3 - vh) * 6)) }
|
if ((3 * vh) < 2) { return (v1 + (v2 - $v1) * ((2 / 3 - vh) * 6)) }
|
||||||
return (v1)
|
return (v1)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -735,49 +735,49 @@ def rgbToHSL(rgb) {
|
|||||||
def h = 0
|
def h = 0
|
||||||
def s = 0
|
def s = 0
|
||||||
def l = 0
|
def l = 0
|
||||||
|
|
||||||
def var_min = [r,g,b].min()
|
def var_min = [r,g,b].min()
|
||||||
def var_max = [r,g,b].max()
|
def var_max = [r,g,b].max()
|
||||||
def del_max = var_max - var_min
|
def del_max = var_max - var_min
|
||||||
|
|
||||||
l = (var_max + var_min) / 2
|
l = (var_max + var_min) / 2
|
||||||
|
|
||||||
if (del_max == 0) {
|
if (del_max == 0) {
|
||||||
h = 0
|
h = 0
|
||||||
s = 0
|
s = 0
|
||||||
} else {
|
} else {
|
||||||
if (l < 0.5) { s = del_max / (var_max + var_min) }
|
if (l < 0.5) { s = del_max / (var_max + var_min) }
|
||||||
else { s = del_max / (2 - var_max - var_min) }
|
else { s = del_max / (2 - var_max - var_min) }
|
||||||
|
|
||||||
def del_r = (((var_max - r) / 6) + (del_max / 2)) / del_max
|
def del_r = (((var_max - r) / 6) + (del_max / 2)) / del_max
|
||||||
def del_g = (((var_max - g) / 6) + (del_max / 2)) / del_max
|
def del_g = (((var_max - g) / 6) + (del_max / 2)) / del_max
|
||||||
def del_b = (((var_max - b) / 6) + (del_max / 2)) / del_max
|
def del_b = (((var_max - b) / 6) + (del_max / 2)) / del_max
|
||||||
|
|
||||||
if (r == var_max) { h = del_b - del_g }
|
if (r == var_max) { h = del_b - del_g }
|
||||||
else if (g == var_max) { h = (1 / 3) + del_r - del_b }
|
else if (g == var_max) { h = (1 / 3) + del_r - del_b }
|
||||||
else if (b == var_max) { h = (2 / 3) + del_g - del_r }
|
else if (b == var_max) { h = (2 / 3) + del_g - del_r }
|
||||||
|
|
||||||
if (h < 0) { h += 1 }
|
if (h < 0) { h += 1 }
|
||||||
if (h > 1) { h -= 1 }
|
if (h > 1) { h -= 1 }
|
||||||
}
|
}
|
||||||
def hsl = [:]
|
def hsl = [:]
|
||||||
hsl = [h: h * 100, s: s * 100, l: l]
|
hsl = [h: h * 100, s: s * 100, l: l]
|
||||||
|
|
||||||
hsl
|
hsl
|
||||||
}
|
}
|
||||||
|
|
||||||
def getColorData(colorName) {
|
def getColorData(colorName) {
|
||||||
log.debug "getColorData: ${colorName}"
|
log.debug "getColorData: ${colorName}"
|
||||||
|
|
||||||
def colorRGB = colorNameToRgb(colorName)
|
def colorRGB = colorNameToRgb(colorName)
|
||||||
def colorHex = rgbToHex(colorRGB)
|
def colorHex = rgbToHex(colorRGB)
|
||||||
def colorHSL = rgbToHSL(colorRGB)
|
def colorHSL = rgbToHSL(colorRGB)
|
||||||
|
|
||||||
def colorData = [:]
|
def colorData = [:]
|
||||||
colorData = [h: colorHSL.h,
|
colorData = [h: colorHSL.h,
|
||||||
s: colorHSL.s,
|
s: colorHSL.s,
|
||||||
l: device.latestValue("level"),
|
l: device.latestValue("level"),
|
||||||
r: colorRGB.r,
|
r: colorRGB.r,
|
||||||
g: colorRGB.g,
|
g: colorRGB.g,
|
||||||
b: colorRGB.b,
|
b: colorRGB.b,
|
||||||
rh: hex(colorRGB.r),
|
rh: hex(colorRGB.r),
|
||||||
@@ -785,8 +785,8 @@ def getColorData(colorName) {
|
|||||||
bh: hex(colorRGB.b),
|
bh: hex(colorRGB.b),
|
||||||
hex: colorHex,
|
hex: colorHex,
|
||||||
alpha: 1]
|
alpha: 1]
|
||||||
|
|
||||||
colorData
|
colorData
|
||||||
}
|
}
|
||||||
|
|
||||||
def doColorButton(colorName) {
|
def doColorButton(colorName) {
|
||||||
@@ -798,7 +798,7 @@ def doColorButton(colorName) {
|
|||||||
def maxLevel = hex(99)
|
def maxLevel = hex(99)
|
||||||
|
|
||||||
toggleTiles(colorName.toLowerCase().replaceAll("\\s",""))
|
toggleTiles(colorName.toLowerCase().replaceAll("\\s",""))
|
||||||
|
|
||||||
if ( colorName == "Fire Place" ) { updateZwaveParam([paramNumber:72, value:6, size:1]) }
|
if ( colorName == "Fire Place" ) { updateZwaveParam([paramNumber:72, value:6, size:1]) }
|
||||||
else if ( colorName == "Storm" ) { updateZwaveParam([paramNumber:72, value:7, size:1]) }
|
else if ( colorName == "Storm" ) { updateZwaveParam([paramNumber:72, value:7, size:1]) }
|
||||||
else if ( colorName == "Deep Fade" ) { updateZwaveParam([paramNumber:72, value:8, size:1]) }
|
else if ( colorName == "Deep Fade" ) { updateZwaveParam([paramNumber:72, value:8, size:1]) }
|
||||||
@@ -808,8 +808,8 @@ def doColorButton(colorName) {
|
|||||||
else if ( colorName == "Daylight" ) { String.format("33050400${maxLevel}02${maxLevel}03${maxLevel}04${maxLevel}%02X", 100) }
|
else if ( colorName == "Daylight" ) { String.format("33050400${maxLevel}02${maxLevel}03${maxLevel}04${maxLevel}%02X", 100) }
|
||||||
else {
|
else {
|
||||||
def c = getColorData(colorName)
|
def c = getColorData(colorName)
|
||||||
def newValue = ["hue": c.h, "saturation": c.s, "level": level, "red": c.r, "green": c.g, "blue": c.b, "hex": c.hex, "alpha": c.alpha]
|
def newValue = ["hue": c.h, "saturation": c.s, "level": level, "red": c.r, "green": c.g, "blue": c.b, "hex": c.hex, "alpha": c.alpha]
|
||||||
setColor(newValue)
|
setColor(newValue)
|
||||||
def r = hex(c.r * (level/100))
|
def r = hex(c.r * (level/100))
|
||||||
def g = hex(c.g * (level/100))
|
def g = hex(c.g * (level/100))
|
||||||
def b = hex(c.b * (level/100))
|
def b = hex(c.b * (level/100))
|
||||||
@@ -823,19 +823,19 @@ def toggleTiles(color) {
|
|||||||
if ( !state.colorTiles ) {
|
if ( !state.colorTiles ) {
|
||||||
state.colorTiles = ["softwhite","daylight","warmwhite","red","green","blue","cyan","magenta","orange","purple","yellow","white","fireplace","storm","deepfade","litefade","police"]
|
state.colorTiles = ["softwhite","daylight","warmwhite","red","green","blue","cyan","magenta","orange","purple","yellow","white","fireplace","storm","deepfade","litefade","police"]
|
||||||
}
|
}
|
||||||
|
|
||||||
def cmds = []
|
def cmds = []
|
||||||
|
|
||||||
state.colorTiles.each({
|
state.colorTiles.each({
|
||||||
if ( it == color ) {
|
if ( it == color ) {
|
||||||
log.debug "Turning ${it} on"
|
log.debug "Turning ${it} on"
|
||||||
cmds << sendEvent(name: it, value: "on${it}", display: True, descriptionText: "${device.displayName} ${color} is 'ON'", isStateChange: true)
|
cmds << sendEvent(name: it, value: "on${it}", displayed: True, descriptionText: "${device.displayName} ${color} is 'ON'", isStateChange: true)
|
||||||
} else {
|
} else {
|
||||||
//log.debug "Turning ${it} off"
|
//log.debug "Turning ${it} off"
|
||||||
cmds << sendEvent(name: it, value: "off${it}", displayed: false)
|
cmds << sendEvent(name: it, value: "off${it}", displayed: false)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
delayBetween(cmds, 2500)
|
delayBetween(cmds, 2500)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -43,8 +43,8 @@ def parse(String description) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
def push() {
|
def push() {
|
||||||
sendEvent(name: "switch", value: "on", isStateChange: true, display: false)
|
sendEvent(name: "switch", value: "on", isStateChange: true, displayed: false)
|
||||||
sendEvent(name: "switch", value: "off", isStateChange: true, display: false)
|
sendEvent(name: "switch", value: "off", isStateChange: true, displayed: false)
|
||||||
sendEvent(name: "momentary", value: "pushed", isStateChange: true)
|
sendEvent(name: "momentary", value: "pushed", isStateChange: true)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -57,6 +57,7 @@ def parse(String description) {
|
|||||||
|
|
||||||
private Map parseBasicMessage(description) {
|
private Map parseBasicMessage(description) {
|
||||||
def name = parseName(description)
|
def name = parseName(description)
|
||||||
|
def results = [:]
|
||||||
if (name != null) {
|
if (name != null) {
|
||||||
def value = parseValue(description)
|
def value = parseValue(description)
|
||||||
def linkText = getLinkText(device)
|
def linkText = getLinkText(device)
|
||||||
@@ -64,7 +65,7 @@ private Map parseBasicMessage(description) {
|
|||||||
def handlerName = value
|
def handlerName = value
|
||||||
def isStateChange = isStateChange(device, name, value)
|
def isStateChange = isStateChange(device, name, value)
|
||||||
|
|
||||||
def results = [
|
results = [
|
||||||
name : name,
|
name : name,
|
||||||
value : value,
|
value : value,
|
||||||
linkText : linkText,
|
linkText : linkText,
|
||||||
@@ -73,8 +74,6 @@ private Map parseBasicMessage(description) {
|
|||||||
isStateChange : isStateChange,
|
isStateChange : isStateChange,
|
||||||
displayed : displayed(description, isStateChange)
|
displayed : displayed(description, isStateChange)
|
||||||
]
|
]
|
||||||
} else {
|
|
||||||
results = [:]
|
|
||||||
}
|
}
|
||||||
log.debug "Parse returned $results.descriptionText"
|
log.debug "Parse returned $results.descriptionText"
|
||||||
return results
|
return results
|
||||||
|
|||||||
@@ -15,9 +15,10 @@ metadata {
|
|||||||
definition (name: "Simulated Minimote", namespace: "smartthings/testing", author: "SmartThings") {
|
definition (name: "Simulated Minimote", namespace: "smartthings/testing", author: "SmartThings") {
|
||||||
capability "Actuator"
|
capability "Actuator"
|
||||||
capability "Button"
|
capability "Button"
|
||||||
|
capability "Holdable Button"
|
||||||
capability "Configuration"
|
capability "Configuration"
|
||||||
capability "Sensor"
|
capability "Sensor"
|
||||||
|
|
||||||
command "push1"
|
command "push1"
|
||||||
command "push2"
|
command "push2"
|
||||||
command "push3"
|
command "push3"
|
||||||
@@ -45,42 +46,42 @@ metadata {
|
|||||||
}
|
}
|
||||||
standardTile("push1", "device.button", width: 1, height: 1, decoration: "flat") {
|
standardTile("push1", "device.button", width: 1, height: 1, decoration: "flat") {
|
||||||
state "default", label: "Push 1", backgroundColor: "#ffffff", action: "push1"
|
state "default", label: "Push 1", backgroundColor: "#ffffff", action: "push1"
|
||||||
}
|
}
|
||||||
standardTile("push2", "device.button", width: 1, height: 1, decoration: "flat") {
|
standardTile("push2", "device.button", width: 1, height: 1, decoration: "flat") {
|
||||||
state "default", label: "Push 2", backgroundColor: "#ffffff", action: "push2"
|
state "default", label: "Push 2", backgroundColor: "#ffffff", action: "push2"
|
||||||
}
|
}
|
||||||
standardTile("push3", "device.button", width: 1, height: 1, decoration: "flat") {
|
standardTile("push3", "device.button", width: 1, height: 1, decoration: "flat") {
|
||||||
state "default", label: "Push 3", backgroundColor: "#ffffff", action: "push3"
|
state "default", label: "Push 3", backgroundColor: "#ffffff", action: "push3"
|
||||||
}
|
}
|
||||||
standardTile("push4", "device.button", width: 1, height: 1, decoration: "flat") {
|
standardTile("push4", "device.button", width: 1, height: 1, decoration: "flat") {
|
||||||
state "default", label: "Push 4", backgroundColor: "#ffffff", action: "push4"
|
state "default", label: "Push 4", backgroundColor: "#ffffff", action: "push4"
|
||||||
}
|
}
|
||||||
standardTile("dummy1", "device.button", width: 1, height: 1, decoration: "flat") {
|
standardTile("dummy1", "device.button", width: 1, height: 1, decoration: "flat") {
|
||||||
state "default", label: " ", backgroundColor: "#ffffff", action: "push4"
|
state "default", label: " ", backgroundColor: "#ffffff", action: "push4"
|
||||||
}
|
}
|
||||||
standardTile("hold1", "device.button", width: 1, height: 1, decoration: "flat") {
|
standardTile("hold1", "device.button", width: 1, height: 1, decoration: "flat") {
|
||||||
state "default", label: "Hold 1", backgroundColor: "#ffffff", action: "hold1"
|
state "default", label: "Hold 1", backgroundColor: "#ffffff", action: "hold1"
|
||||||
}
|
}
|
||||||
standardTile("hold2", "device.button", width: 1, height: 1, decoration: "flat") {
|
standardTile("hold2", "device.button", width: 1, height: 1, decoration: "flat") {
|
||||||
state "default", label: "Hold 2", backgroundColor: "#ffffff", action: "hold2"
|
state "default", label: "Hold 2", backgroundColor: "#ffffff", action: "hold2"
|
||||||
}
|
}
|
||||||
standardTile("dummy2", "device.button", width: 1, height: 1, decoration: "flat") {
|
standardTile("dummy2", "device.button", width: 1, height: 1, decoration: "flat") {
|
||||||
state "default", label: " ", backgroundColor: "#ffffff", action: "push4"
|
state "default", label: " ", backgroundColor: "#ffffff", action: "push4"
|
||||||
}
|
}
|
||||||
standardTile("hold3", "device.button", width: 1, height: 1, decoration: "flat") {
|
standardTile("hold3", "device.button", width: 1, height: 1, decoration: "flat") {
|
||||||
state "default", label: "Hold 3", backgroundColor: "#ffffff", action: "hold3"
|
state "default", label: "Hold 3", backgroundColor: "#ffffff", action: "hold3"
|
||||||
}
|
}
|
||||||
standardTile("hold4", "device.button", width: 1, height: 1, decoration: "flat") {
|
standardTile("hold4", "device.button", width: 1, height: 1, decoration: "flat") {
|
||||||
state "default", label: "Hold 4", backgroundColor: "#ffffff", action: "hold4"
|
state "default", label: "Hold 4", backgroundColor: "#ffffff", action: "hold4"
|
||||||
}
|
}
|
||||||
|
|
||||||
main "button"
|
main "button"
|
||||||
details(["push1","push2","button","push3","push4","dummy1","hold1","hold2","dummy2","hold3","hold4"])
|
details(["push1","push2","button","push3","push4","dummy1","hold1","hold2","dummy2","hold3","hold4"])
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
def parse(String description) {
|
def parse(String description) {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
def push1() {
|
def push1() {
|
||||||
@@ -125,3 +126,15 @@ private hold(button) {
|
|||||||
sendEvent(name: "button", value: "held", data: [buttonNumber: button], descriptionText: "$device.displayName button $button was held", isStateChange: true)
|
sendEvent(name: "button", value: "held", data: [buttonNumber: button], descriptionText: "$device.displayName button $button was held", isStateChange: true)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def installed() {
|
||||||
|
initialize()
|
||||||
|
}
|
||||||
|
|
||||||
|
def updated() {
|
||||||
|
initialize()
|
||||||
|
}
|
||||||
|
|
||||||
|
def initialize() {
|
||||||
|
sendEvent(name: "numberOfButtons", value: 4)
|
||||||
|
}
|
||||||
|
|||||||
@@ -19,6 +19,7 @@ metadata {
|
|||||||
capability "Actuator"
|
capability "Actuator"
|
||||||
capability "Battery"
|
capability "Battery"
|
||||||
capability "Button"
|
capability "Button"
|
||||||
|
capability "Holdable Button"
|
||||||
capability "Configuration"
|
capability "Configuration"
|
||||||
capability "Refresh"
|
capability "Refresh"
|
||||||
capability "Sensor"
|
capability "Sensor"
|
||||||
|
|||||||
@@ -23,9 +23,11 @@ metadata {
|
|||||||
|
|
||||||
|
|
||||||
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008"
|
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008"
|
||||||
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0B04, FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "LIGHTIFY A19 ON/OFF/DIM", deviceJoinName: "OSRAM LIGHTIFY LED Smart Connected Light"
|
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0B04, FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "LIGHTIFY A19 ON/OFF/DIM", deviceJoinName: "SYLVANIA Smart A19 Soft White"
|
||||||
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, FF00", outClusters: "0019", manufacturer: "MRVL", model: "MZ100", deviceJoinName: "Wemo Bulb"
|
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, FF00", outClusters: "0019", manufacturer: "MRVL", model: "MZ100", deviceJoinName: "Wemo Bulb"
|
||||||
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0B05", outClusters: "0019", manufacturer: "OSRAM SYLVANIA", model: "iQBR30", deviceJoinName: "Sylvania Ultra iQ"
|
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0B05", outClusters: "0019", manufacturer: "OSRAM SYLVANIA", model: "iQBR30", deviceJoinName: "Sylvania Ultra iQ"
|
||||||
|
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "LIGHTIFY PAR38 ON/OFF/DIM", deviceJoinName: "SYLVANIA Smart PAR38 Soft White"
|
||||||
|
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0B04, FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "LIGHTIFY BR ON/OFF/DIM", deviceJoinName: "SYLVANIA Smart BR30 Soft White"
|
||||||
}
|
}
|
||||||
|
|
||||||
tiles(scale: 2) {
|
tiles(scale: 2) {
|
||||||
|
|||||||
@@ -28,8 +28,8 @@ metadata {
|
|||||||
capability "Switch Level"
|
capability "Switch Level"
|
||||||
capability "Health Check"
|
capability "Health Check"
|
||||||
|
|
||||||
fingerprint profileId: "0104", inClusters: "0000,0003,0004,0005,0006,0008,0300,0B04,FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "Gardenspot RGB", deviceJoinName: "OSRAM LIGHTIFY Gardenspot mini RGB"
|
fingerprint profileId: "0104", inClusters: "0000,0003,0004,0005,0006,0008,0300,0B04,FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "Gardenspot RGB", deviceJoinName: "SYLVANIA Smart Gardenspot mini RGB"
|
||||||
fingerprint profileId: "0104", inClusters: "0000,0003,0004,0005,0006,0008,0300,0B04,FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "LIGHTIFY Gardenspot RGB", deviceJoinName: "OSRAM LIGHTIFY Gardenspot mini RGB"
|
fingerprint profileId: "0104", inClusters: "0000,0003,0004,0005,0006,0008,0300,0B04,FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "LIGHTIFY Gardenspot RGB", deviceJoinName: "SYLVANIA Smart Gardenspot mini RGB"
|
||||||
}
|
}
|
||||||
|
|
||||||
// UI tile definitions
|
// UI tile definitions
|
||||||
|
|||||||
@@ -32,11 +32,12 @@ metadata {
|
|||||||
attribute "colorName", "string"
|
attribute "colorName", "string"
|
||||||
command "setGenericName"
|
command "setGenericName"
|
||||||
|
|
||||||
fingerprint profileId: "0104", inClusters: "0000,0003,0004,0005,0006,0008,0300,0B04,FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "LIGHTIFY Flex RGBW", deviceJoinName: "OSRAM LIGHTIFY LED FLEXIBLE STRIP RGBW"
|
fingerprint profileId: "0104", inClusters: "0000,0003,0004,0005,0006,0008,0300,0B04,FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "LIGHTIFY Flex RGBW", deviceJoinName: "SYLVANIA Smart Flex RGBW"
|
||||||
fingerprint profileId: "0104", inClusters: "0000,0003,0004,0005,0006,0008,0300,0B04,FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "Flex RGBW", deviceJoinName: "OSRAM LIGHTIFY LED FLEXIBLE STRIP RGBW"
|
fingerprint profileId: "0104", inClusters: "0000,0003,0004,0005,0006,0008,0300,0B04,FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "Flex RGBW", deviceJoinName: "OSRAM LIGHTIFY Flex RGBW"
|
||||||
fingerprint profileId: "0104", inClusters: "0000,0003,0004,0005,0006,0008,0300,0B04,FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "LIGHTIFY A19 RGBW", deviceJoinName: "OSRAM LIGHTIFY LED A19 RGBW"
|
fingerprint profileId: "0104", inClusters: "0000,0003,0004,0005,0006,0008,0300,0B04,FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "LIGHTIFY A19 RGBW", deviceJoinName: "SYLVANIA Smart A19 RGBW"
|
||||||
fingerprint profileId: "0104", inClusters: "0000,0003,0004,0005,0006,0008,0300,0B04,FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "LIGHTIFY BR RGBW", deviceJoinName: "OSRAM LIGHTIFY LED BR30 RGBW"
|
fingerprint profileId: "0104", inClusters: "0000,0003,0004,0005,0006,0008,0300,0B04,FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "LIGHTIFY BR RGBW", deviceJoinName: "SYLVANIA Smart BR30 RGBW"
|
||||||
fingerprint profileId: "0104", inClusters: "0000,0003,0004,0005,0006,0008,0300,0B04,FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "LIGHTIFY RT RGBW", deviceJoinName: "OSRAM LIGHTIFY LED RT 5/6 RGBW"
|
fingerprint profileId: "0104", inClusters: "0000,0003,0004,0005,0006,0008,0300,0B04,FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "LIGHTIFY RT RGBW", deviceJoinName: "SYLVANIA Smart RT5/6 RGBW"
|
||||||
|
fingerprint profileId: "0104", inClusters: "0000,0003,0004,0005,0006,0008,0300,0B04,FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "LIGHTIFY FLEX OUTDOOR RGBW", deviceJoinName: "SYLVANIA Smart Outdoor RGBW Flex"
|
||||||
}
|
}
|
||||||
|
|
||||||
// UI tile definitions
|
// UI tile definitions
|
||||||
|
|||||||
@@ -32,11 +32,12 @@ metadata {
|
|||||||
|
|
||||||
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0300", outClusters: "0019"
|
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0300", outClusters: "0019"
|
||||||
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0300, 0B04", outClusters: "0019"
|
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0300, 0B04", outClusters: "0019"
|
||||||
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0300, 0B04, FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "LIGHTIFY BR Tunable White", deviceJoinName: "OSRAM LIGHTIFY LED Flood BR30 Tunable White"
|
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0300, 0B04, FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "LIGHTIFY BR Tunable White", deviceJoinName: "SYLVANIA Smart BR30 Tunable White"
|
||||||
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0300, 0B04, FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "LIGHTIFY RT Tunable White", deviceJoinName: "OSRAM LIGHTIFY LED Recessed Kit RT 5/6 Tunable White"
|
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0300, 0B04, FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "LIGHTIFY RT Tunable White", deviceJoinName: "SYLVANIA Smart RT5/6 Tunable White"
|
||||||
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0300, 0B04, FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "Classic A60 TW", deviceJoinName: "OSRAM LIGHTIFY LED Tunable White 60W"
|
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0300, 0B04, FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "Classic A60 TW", deviceJoinName: "OSRAM LIGHTIFY LED Classic A60 Tunable White"
|
||||||
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0300, 0B04, FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "LIGHTIFY A19 Tunable White", deviceJoinName: "OSRAM LIGHTIFY LED Tunable White 60W"
|
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0300, 0B04, FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "LIGHTIFY A19 Tunable White", deviceJoinName: "SYLVANIA Smart A19 Tunable White"
|
||||||
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0300, 0B04, FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "Classic B40 TW - LIGHTIFY", deviceJoinName: "OSRAM LIGHTIFY Classic B40 Tunable White"
|
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0300, 0B04, FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "Classic B40 TW - LIGHTIFY", deviceJoinName: "OSRAM LIGHTIFY Classic B40 Tunable White"
|
||||||
|
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0300, 0702, 0B05", outClusters: "0019", manufacturer: "sengled", model: "Z01-A19NAE26", deviceJoinName: "Sengled Element plus"
|
||||||
}
|
}
|
||||||
|
|
||||||
// UI tile definitions
|
// UI tile definitions
|
||||||
|
|||||||
@@ -127,8 +127,8 @@ def configureHealthCheck() {
|
|||||||
|
|
||||||
def configure() {
|
def configure() {
|
||||||
log.debug "configure()"
|
log.debug "configure()"
|
||||||
zigbee.onOffConfig() + zigbee.levelConfig() + zigbee.onOffRefresh() + zigbee.levelRefresh()
|
|
||||||
configureHealthCheck()
|
configureHealthCheck()
|
||||||
|
zigbee.onOffConfig() + zigbee.levelConfig() + zigbee.onOffRefresh() + zigbee.levelRefresh()
|
||||||
}
|
}
|
||||||
|
|
||||||
def updated() {
|
def updated() {
|
||||||
|
|||||||
@@ -27,6 +27,10 @@ metadata {
|
|||||||
fingerprint profileId: "C05E", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0300", outClusters: "0019"
|
fingerprint profileId: "C05E", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0300", outClusters: "0019"
|
||||||
fingerprint profileId: "C05E", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0300, 1000", outClusters: "0019"
|
fingerprint profileId: "C05E", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0300, 1000", outClusters: "0019"
|
||||||
fingerprint profileId: "C05E", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0300, 1000", outClusters: "0019", "manufacturer":"OSRAM", "model":"Classic A60 RGBW", deviceJoinName: "OSRAM LIGHTIFY LED Classic A60 RGBW"
|
fingerprint profileId: "C05E", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0300, 1000", outClusters: "0019", "manufacturer":"OSRAM", "model":"Classic A60 RGBW", deviceJoinName: "OSRAM LIGHTIFY LED Classic A60 RGBW"
|
||||||
|
fingerprint profileId: "C05E", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0300, 0B04, FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "PAR 16 50 RGBW - LIGHTIFY", deviceJoinName: "OSRAM LIGHTIFY RGBW PAR 16 50"
|
||||||
|
fingerprint profileId: "C05E", inClusters: "0000,0003,0004,0005,0006,0008,0300,1000,FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "Flex RGBW", deviceJoinName: "OSRAM LIGHTIFY Flex RGBW"
|
||||||
|
fingerprint profileId: "C05E", inClusters: "0000,0003,0004,0005,0006,0008,0300,1000,FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "Gardenpole RGBW-Lightify", deviceJoinName: "OSRAM LIGHTIFY Gardenpole RGBW"
|
||||||
|
fingerprint profileId: "C05E", inClusters: "0000,0003,0004,0005,0006,0008,0300,1000,FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "LIGHTIFY Outdoor Flex RGBW", deviceJoinName: "OSRAM LIGHTIFY Outdoor Flex RGBW"
|
||||||
}
|
}
|
||||||
|
|
||||||
// UI tile definitions
|
// UI tile definitions
|
||||||
|
|||||||
@@ -32,6 +32,7 @@ metadata {
|
|||||||
|
|
||||||
fingerprint profileId: "C05E", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0300, 1000, 0B04, FC0F", outClusters: "0019", "manufacturer":"OSRAM", "model":"Classic A60 TW", deviceJoinName: "OSRAM LIGHTIFY LED Classic A60 Tunable White"
|
fingerprint profileId: "C05E", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0300, 1000, 0B04, FC0F", outClusters: "0019", "manufacturer":"OSRAM", "model":"Classic A60 TW", deviceJoinName: "OSRAM LIGHTIFY LED Classic A60 Tunable White"
|
||||||
fingerprint profileId: "C05E", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0300, 1000, FC0F", outClusters: "0019", "manufacturer":"OSRAM", "model":"PAR16 50 TW", deviceJoinName: "OSRAM LIGHTIFY LED PAR16 50 Tunable White"
|
fingerprint profileId: "C05E", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0300, 1000, FC0F", outClusters: "0019", "manufacturer":"OSRAM", "model":"PAR16 50 TW", deviceJoinName: "OSRAM LIGHTIFY LED PAR16 50 Tunable White"
|
||||||
|
fingerprint profileId: "C05E", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0300, 1000, 0B04, FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "Classic B40 TW - LIGHTIFY", deviceJoinName: "OSRAM LIGHTIFY Classic B40 Tunable White"
|
||||||
}
|
}
|
||||||
|
|
||||||
// UI tile definitions
|
// UI tile definitions
|
||||||
|
|||||||
@@ -81,7 +81,7 @@ def zwaveEvent(physicalgraph.zwave.commands.configurationv1.ConfigurationReport
|
|||||||
def value = "when off"
|
def value = "when off"
|
||||||
if (cmd.configurationValue[0] == 1) {value = "when on"}
|
if (cmd.configurationValue[0] == 1) {value = "when on"}
|
||||||
if (cmd.configurationValue[0] == 2) {value = "never"}
|
if (cmd.configurationValue[0] == 2) {value = "never"}
|
||||||
[name: "indicatorStatus", value: value, display: false]
|
[name: "indicatorStatus", value: value, displayed: false]
|
||||||
}
|
}
|
||||||
|
|
||||||
def zwaveEvent(physicalgraph.zwave.commands.hailv1.Hail cmd) {
|
def zwaveEvent(physicalgraph.zwave.commands.hailv1.Hail cmd) {
|
||||||
|
|||||||
@@ -114,7 +114,7 @@ def zwaveEvent(physicalgraph.zwave.commands.configurationv1.ConfigurationReport
|
|||||||
def value = "when off"
|
def value = "when off"
|
||||||
if (cmd.configurationValue[0] == 1) {value = "when on"}
|
if (cmd.configurationValue[0] == 1) {value = "when on"}
|
||||||
if (cmd.configurationValue[0] == 2) {value = "never"}
|
if (cmd.configurationValue[0] == 2) {value = "never"}
|
||||||
[name: "indicatorStatus", value: value, display: false]
|
[name: "indicatorStatus", value: value, displayed: false]
|
||||||
}
|
}
|
||||||
|
|
||||||
def zwaveEvent(physicalgraph.zwave.commands.hailv1.Hail cmd) {
|
def zwaveEvent(physicalgraph.zwave.commands.hailv1.Hail cmd) {
|
||||||
@@ -174,17 +174,17 @@ def refresh() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void indicatorWhenOn() {
|
void indicatorWhenOn() {
|
||||||
sendEvent(name: "indicatorStatus", value: "when on", display: false)
|
sendEvent(name: "indicatorStatus", value: "when on", displayed: false)
|
||||||
sendHubCommand(new physicalgraph.device.HubAction(zwave.configurationV1.configurationSet(configurationValue: [1], parameterNumber: 3, size: 1).format()))
|
sendHubCommand(new physicalgraph.device.HubAction(zwave.configurationV1.configurationSet(configurationValue: [1], parameterNumber: 3, size: 1).format()))
|
||||||
}
|
}
|
||||||
|
|
||||||
void indicatorWhenOff() {
|
void indicatorWhenOff() {
|
||||||
sendEvent(name: "indicatorStatus", value: "when off", display: false)
|
sendEvent(name: "indicatorStatus", value: "when off", displayed: false)
|
||||||
sendHubCommand(new physicalgraph.device.HubAction(zwave.configurationV1.configurationSet(configurationValue: [0], parameterNumber: 3, size: 1).format()))
|
sendHubCommand(new physicalgraph.device.HubAction(zwave.configurationV1.configurationSet(configurationValue: [0], parameterNumber: 3, size: 1).format()))
|
||||||
}
|
}
|
||||||
|
|
||||||
void indicatorNever() {
|
void indicatorNever() {
|
||||||
sendEvent(name: "indicatorStatus", value: "never", display: false)
|
sendEvent(name: "indicatorStatus", value: "never", displayed: false)
|
||||||
sendHubCommand(new physicalgraph.device.HubAction(zwave.configurationV1.configurationSet(configurationValue: [2], parameterNumber: 3, size: 1).format()))
|
sendHubCommand(new physicalgraph.device.HubAction(zwave.configurationV1.configurationSet(configurationValue: [2], parameterNumber: 3, size: 1).format()))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
1545
smartapps/fuzzysb/tado-connect.src/tado-connect.groovy
Normal file
1545
smartapps/fuzzysb/tado-connect.src/tado-connect.groovy
Normal file
File diff suppressed because it is too large
Load Diff
@@ -1,9 +1,10 @@
|
|||||||
/**
|
/**
|
||||||
* Color Coordinator
|
* Color Coordinator
|
||||||
* Version 1.0.0 - 7/4/15
|
* Version 1.1.0 - 11/9/16
|
||||||
* By Michael Struck
|
* By Michael Struck
|
||||||
*
|
*
|
||||||
* 1.0.0 - Initial release
|
* 1.0.0 - Initial release
|
||||||
|
* 1.1.0 - Fixed issue where master can be part of slaves. This causes a loop that impacts SmartThings.
|
||||||
*
|
*
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except
|
* Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except
|
||||||
@@ -31,27 +32,35 @@ preferences {
|
|||||||
}
|
}
|
||||||
|
|
||||||
def mainPage() {
|
def mainPage() {
|
||||||
dynamicPage(name: "mainPage", title: "", install: true, uninstall: true) {
|
dynamicPage(name: "mainPage", title: "", install: true, uninstall: false) {
|
||||||
section("Master Light") {
|
def masterInList = slaves.id.find{it==master.id}
|
||||||
|
if (masterInList) {
|
||||||
|
section ("**WARNING**"){
|
||||||
|
paragraph "You have included the Master Light in the Slave Group. This will cause a loop in execution. Please remove this device from the Slave Group.", image: "https://raw.githubusercontent.com/MichaelStruck/SmartThingsPublic/master/img/caution.png"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
section("Master Light") {
|
||||||
input "master", "capability.colorControl", title: "Colored Light"
|
input "master", "capability.colorControl", title: "Colored Light"
|
||||||
}
|
}
|
||||||
section("Lights that follow the master settings") {
|
section("Lights that follow the master settings") {
|
||||||
input "slaves", "capability.colorControl", title: "Colored Lights", multiple: true, required: false
|
input "slaves", "capability.colorControl", title: "Colored Lights", multiple: true, required: false, submitOnChange: true
|
||||||
}
|
}
|
||||||
section([mobileOnly:true], "Options") {
|
section([mobileOnly:true], "Options") {
|
||||||
label(title: "Assign a name", required: false)
|
input "randomYes", "bool",title: "When Master Turned On, Randomize Color", defaultValue: false
|
||||||
href "pageAbout", title: "About ${textAppName()}", description: "Tap to get application version, license and instructions"
|
href "pageAbout", title: "About ${textAppName()}", description: "Tap to get application version, license and instructions"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
page(name: "pageAbout", title: "About ${textAppName()}") {
|
page(name: "pageAbout", title: "About ${textAppName()}", uninstall: true) {
|
||||||
section {
|
section {
|
||||||
paragraph "${textVersion()}\n${textCopyright()}\n\n${textLicense()}\n"
|
paragraph "${textVersion()}\n${textCopyright()}\n\n${textLicense()}\n"
|
||||||
}
|
}
|
||||||
section("Instructions") {
|
section("Instructions") {
|
||||||
paragraph textHelp()
|
paragraph textHelp()
|
||||||
}
|
}
|
||||||
|
section("Tap button below to remove application"){
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
def installed() {
|
def installed() {
|
||||||
@@ -72,27 +81,55 @@ def init() {
|
|||||||
}
|
}
|
||||||
//-----------------------------------
|
//-----------------------------------
|
||||||
def onOffHandler(evt){
|
def onOffHandler(evt){
|
||||||
if (master.currentValue("switch") == "on"){
|
if (!slaves.id.find{it==master.id}){
|
||||||
slaves?.on()
|
if (master.currentValue("switch") == "on"){
|
||||||
}
|
if (randomYes) getRandomColorMaster()
|
||||||
else {
|
else slaves?.on()
|
||||||
slaves?.off()
|
}
|
||||||
}
|
else {
|
||||||
|
slaves?.off()
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
def colorHandler(evt) {
|
def colorHandler(evt) {
|
||||||
def dimLevel = master.currentValue("level")
|
if (!slaves.id.find{it==master.id} && master.currentValue("switch") == "on"){
|
||||||
def hueLevel = master.currentValue("hue")
|
log.debug "Changing Slave units H,S,L"
|
||||||
def saturationLevel = master.currentValue("saturation")
|
def dimLevel = master.currentValue("level")
|
||||||
|
def hueLevel = master.currentValue("hue")
|
||||||
|
def saturationLevel = master.currentValue("saturation")
|
||||||
|
def newValue = [hue: hueLevel, saturation: saturationLevel, level: dimLevel as Integer]
|
||||||
|
slaves?.setColor(newValue)
|
||||||
|
try {
|
||||||
|
log.debug "Changing Slave color temp"
|
||||||
|
def tempLevel = master.currentValue("colorTemperature")
|
||||||
|
slaves?.setColorTemperature(tempLevel)
|
||||||
|
}
|
||||||
|
catch (e){
|
||||||
|
log.debug "Color temp for master --"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def getRandomColorMaster(){
|
||||||
|
def hueLevel = Math.floor(Math.random() *1000)
|
||||||
|
def saturationLevel = Math.floor(Math.random() * 100)
|
||||||
|
def dimLevel = master.currentValue("level")
|
||||||
def newValue = [hue: hueLevel, saturation: saturationLevel, level: dimLevel as Integer]
|
def newValue = [hue: hueLevel, saturation: saturationLevel, level: dimLevel as Integer]
|
||||||
slaves?.setColor(newValue)
|
log.debug hueLevel
|
||||||
|
log.debug saturationLevel
|
||||||
|
master.setColor(newValue)
|
||||||
|
slaves?.setColor(newValue)
|
||||||
}
|
}
|
||||||
|
|
||||||
def tempHandler(evt){
|
def tempHandler(evt){
|
||||||
if (evt.value != "--") {
|
if (!slaves.id.find{it==master.id} && master.currentValue("switch") == "on"){
|
||||||
def tempLevel = master.currentValue("colorTemperature")
|
if (evt.value != "--") {
|
||||||
slaves?.setColorTemperature(tempLevel)
|
log.debug "Changing Slave color temp based on Master change"
|
||||||
}
|
def tempLevel = master.currentValue("colorTemperature")
|
||||||
|
slaves?.setColorTemperature(tempLevel)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
//Version/Copyright/Information/Help
|
//Version/Copyright/Information/Help
|
||||||
@@ -102,11 +139,11 @@ private def textAppName() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private def textVersion() {
|
private def textVersion() {
|
||||||
def text = "Version 1.0.0 (07/04/2015)"
|
def text = "Version 1.1.0 (11/09/2016)"
|
||||||
}
|
}
|
||||||
|
|
||||||
private def textCopyright() {
|
private def textCopyright() {
|
||||||
def text = "Copyright © 2015 Michael Struck"
|
def text = "Copyright © 2016 Michael Struck"
|
||||||
}
|
}
|
||||||
|
|
||||||
private def textLicense() {
|
private def textLicense() {
|
||||||
@@ -128,5 +165,5 @@ private def textHelp() {
|
|||||||
def text =
|
def text =
|
||||||
"This application will allow you to control the settings of multiple colored lights with one control. " +
|
"This application will allow you to control the settings of multiple colored lights with one control. " +
|
||||||
"Simply choose a master control light, and then choose the lights that will follow the settings of the master, "+
|
"Simply choose a master control light, and then choose the lights that will follow the settings of the master, "+
|
||||||
"including on/off conditions, hue, saturation, level and color temperature."
|
"including on/off conditions, hue, saturation, level and color temperature. Also includes a random color feature."
|
||||||
}
|
}
|
||||||
@@ -19,9 +19,9 @@
|
|||||||
author: "SmartThings",
|
author: "SmartThings",
|
||||||
description: "Control your Bose SoundTouch speakers",
|
description: "Control your Bose SoundTouch speakers",
|
||||||
category: "SmartThings Labs",
|
category: "SmartThings Labs",
|
||||||
iconUrl: "https://s3.amazonaws.com/smartapp-icons/Convenience/Cat-Convenience.png",
|
iconUrl: "https://d3azp77rte0gip.cloudfront.net/smartapps/fcf1d93a-ba0b-4324-b96f-e5b5487dfaf5/images/BoseST_icon.png",
|
||||||
iconX2Url: "https://s3.amazonaws.com/smartapp-icons/Convenience/Cat-Convenience@2x.png",
|
iconX2Url: "https://d3azp77rte0gip.cloudfront.net/smartapps/fcf1d93a-ba0b-4324-b96f-e5b5487dfaf5/images/BoseST_icon@2x.png",
|
||||||
iconX3Url: "https://s3.amazonaws.com/smartapp-icons/Convenience/Cat-Convenience@2x.png",
|
iconX3Url: "https://d3azp77rte0gip.cloudfront.net/smartapps/fcf1d93a-ba0b-4324-b96f-e5b5487dfaf5/images/BoseST_icon@2x-1.png",
|
||||||
singleInstance: true
|
singleInstance: true
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -104,7 +104,7 @@ def deviceDiscovery()
|
|||||||
|
|
||||||
return dynamicPage(name:"deviceDiscovery", title:"Discovery Started!", nextPage:"", refreshInterval:refreshInterval, install:true, uninstall: true) {
|
return dynamicPage(name:"deviceDiscovery", title:"Discovery Started!", nextPage:"", refreshInterval:refreshInterval, install:true, uninstall: true) {
|
||||||
section("Please wait while we discover your ${getDeviceName()}. Discovery can take five minutes or more, so sit back and relax! Select your device below once discovered.") {
|
section("Please wait while we discover your ${getDeviceName()}. Discovery can take five minutes or more, so sit back and relax! Select your device below once discovered.") {
|
||||||
input "selecteddevice", "enum", required:false, title:"Select ${getDeviceName()} (${numFound} found)", multiple:true, options:devices
|
input "selecteddevice", "enum", required:false, title:"Select ${getDeviceName()} (${numFound} found)", multiple:true, options:devices, submitOnChange: true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -196,6 +196,8 @@ def addDevice(){
|
|||||||
d = addChildDevice(getNameSpace(), getDeviceName(), dni, newDevice?.value.hub, [label:"${deviceName}"])
|
d = addChildDevice(getNameSpace(), getDeviceName(), dni, newDevice?.value.hub, [label:"${deviceName}"])
|
||||||
d.boseSetDeviceID(newDevice.value.deviceID)
|
d.boseSetDeviceID(newDevice.value.deviceID)
|
||||||
log.trace "Created ${d.displayName} with id $dni"
|
log.trace "Created ${d.displayName} with id $dni"
|
||||||
|
// sync DTH with device, done here as it currently don't work from the DTH's installed() method
|
||||||
|
d.refresh()
|
||||||
} else {
|
} else {
|
||||||
log.trace "${d.displayName} with id $dni already exists"
|
log.trace "${d.displayName} with id $dni already exists"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -516,7 +516,7 @@ def pollResponse(response, data) {
|
|||||||
def hub = getChildDevice("harmony-${it.key}")
|
def hub = getChildDevice("harmony-${it.key}")
|
||||||
if (hub) {
|
if (hub) {
|
||||||
if (it.value.response.data.currentAvActivity == "-1") {
|
if (it.value.response.data.currentAvActivity == "-1") {
|
||||||
hub.sendEvent(name: "currentActivity", value: "--", descriptionText: "There isn't any activity running", display: false)
|
hub.sendEvent(name: "currentActivity", value: "--", descriptionText: "There isn't any activity running", displayed: false)
|
||||||
} else {
|
} else {
|
||||||
def currentActivity
|
def currentActivity
|
||||||
def activityDTH = getChildDevice("harmony-${it.key}-${it.value.response.data.currentAvActivity}")
|
def activityDTH = getChildDevice("harmony-${it.key}-${it.value.response.data.currentAvActivity}")
|
||||||
@@ -524,7 +524,7 @@ def pollResponse(response, data) {
|
|||||||
currentActivity = activityDTH.device.displayName
|
currentActivity = activityDTH.device.displayName
|
||||||
else
|
else
|
||||||
currentActivity = getActivityName(it.value.response.data.currentAvActivity,it.key)
|
currentActivity = getActivityName(it.value.response.data.currentAvActivity,it.key)
|
||||||
hub.sendEvent(name: "currentActivity", value: currentActivity, descriptionText: "Current activity is ${currentActivity}", display: false)
|
hub.sendEvent(name: "currentActivity", value: currentActivity, descriptionText: "Current activity is ${currentActivity}", displayed: false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@@ -576,7 +576,7 @@ def getActivityList() {
|
|||||||
}
|
}
|
||||||
activities += [id: "off", name: "Activity OFF", type: "0"]
|
activities += [id: "off", name: "Activity OFF", type: "0"]
|
||||||
}
|
}
|
||||||
hub.sendEvent(name: "activities", value: new groovy.json.JsonBuilder(activities).toString(), descriptionText: "Activities are ${activities.collect { it.name }?.join(', ')}", display: false)
|
hub.sendEvent(name: "activities", value: new groovy.json.JsonBuilder(activities).toString(), descriptionText: "Activities are ${activities.collect { it.name }?.join(', ')}", displayed: false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user