mirror of
https://github.com/mtan93/SmartThingsPublic.git
synced 2026-03-09 13:21:53 +00:00
Compare commits
16 Commits
PROD_2016.
...
MSA-1395-1
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
966bd2c1df | ||
|
|
3472ee329d | ||
|
|
577b127287 | ||
|
|
5f41af35e2 | ||
|
|
962774996e | ||
|
|
d79594cbcb | ||
|
|
bf8fe4cad7 | ||
|
|
65752ce378 | ||
|
|
95f08aeb3d | ||
|
|
cd7bc1b262 | ||
|
|
be7f6a76a9 | ||
|
|
10e5b7e9d7 | ||
|
|
fc38c534f9 | ||
|
|
fd47bcb8a8 | ||
|
|
972599b1b5 | ||
|
|
f5d3cca6a0 |
@@ -22,6 +22,7 @@ metadata {
|
||||
capability "Configuration"
|
||||
capability "Sensor"
|
||||
capability "Battery"
|
||||
capability "Health Check"
|
||||
|
||||
attribute "tamper", "enum", ["detected", "clear"]
|
||||
attribute "batteryStatus", "string"
|
||||
@@ -326,6 +327,9 @@ def zwaveEvent(physicalgraph.zwave.Command cmd) {
|
||||
}
|
||||
|
||||
def configure() {
|
||||
// allow device user configured or default 16 min to check in; double the periodic reporting interval
|
||||
sendEvent(name: "checkInterval", value: 2* timeOptionValueMap[reportInterval] ?: 2*8*60, displayed: false)
|
||||
|
||||
// This sensor joins as a secure device if you double-click the button to include it
|
||||
log.debug "${device.displayName} is configuring its settings"
|
||||
def request = []
|
||||
|
||||
@@ -20,6 +20,9 @@ metadata {
|
||||
capability "Configuration"
|
||||
capability "Sensor"
|
||||
capability "Battery"
|
||||
capability "Health Check"
|
||||
|
||||
command "configureAfterSecure"
|
||||
|
||||
fingerprint deviceId: "0x0701", inClusters: "0x5E,0x86,0x72,0x59,0x85,0x73,0x71,0x84,0x80,0x30,0x31,0x70,0x98,0x7A", outClusters:"0x5A"
|
||||
}
|
||||
@@ -245,6 +248,8 @@ def configureAfterSecure() {
|
||||
def configure() {
|
||||
// log.debug "configure()"
|
||||
//["delay 30000"] + secure(zwave.securityV1.securityCommandsSupportedGet())
|
||||
// allow device 16 min to check in; double the periodic reporting interval
|
||||
sendEvent(name: "checkInterval", value: 2*8*60, displayed: false)
|
||||
}
|
||||
|
||||
private setConfigured() {
|
||||
|
||||
@@ -20,6 +20,7 @@ metadata {
|
||||
capability "Illuminance Measurement"
|
||||
capability "Sensor"
|
||||
capability "Battery"
|
||||
capability "Health Check"
|
||||
|
||||
fingerprint deviceId: "0x2001", inClusters: "0x30,0x31,0x80,0x84,0x70,0x85,0x72,0x86"
|
||||
}
|
||||
@@ -180,6 +181,9 @@ def zwaveEvent(physicalgraph.zwave.Command cmd) {
|
||||
}
|
||||
|
||||
def configure() {
|
||||
// allow device 10 min to check in; double the periodic reporting interval
|
||||
sendEvent(name: "checkInterval", value: 2*5*60, displayed: false)
|
||||
|
||||
delayBetween([
|
||||
// send binary sensor report instead of basic set for motion
|
||||
zwave.configurationV1.configurationSet(parameterNumber: 5, size: 1, scaledConfigurationValue: 2).format(),
|
||||
|
||||
@@ -0,0 +1,251 @@
|
||||
/**
|
||||
* Smart Family Presence
|
||||
*
|
||||
* Copyright 2016 Darin Spivey
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except
|
||||
* in compliance with the License. You may obtain a copy of the License at:
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software distributed under the License is distributed
|
||||
* on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License
|
||||
* for the specific language governing permissions and limitations under the License.
|
||||
*
|
||||
*/
|
||||
definition(
|
||||
name: "Smart Family Presence",
|
||||
namespace: "ddspivey",
|
||||
author: "Darin Spivey",
|
||||
description: "Smart arrival and departure push messages for couples/families that are traveling together. When family members arrive and depart together, there is no need to send an individual push alert for each.",
|
||||
category: "Family",
|
||||
iconUrl: "http://cdn.device-icons.smartthings.com/Home/home4-icn@2x.png",
|
||||
iconX2Url: "http://cdn.device-icons.smartthings.com/Home/home4-icn@2x.png",
|
||||
iconX3Url: "http://cdn.device-icons.smartthings.com/Home/home4-icn@2x.png")
|
||||
|
||||
|
||||
preferences {
|
||||
section("Family Members") {
|
||||
input "familySensors", "capability.presenceSensor", required: true, title: "Who's in your family?", multiple: true
|
||||
}
|
||||
section("Threshold") {
|
||||
paragraph "Set the time in seconds to allow for group arrival/departure"
|
||||
input "timeThreshold", "text", required: false, title: "Default is ${defaultThreshold}."
|
||||
}
|
||||
section("Smart departure alerts") {
|
||||
paragraph "When family members are home together, departure push alerts may not be necessary because most of the time, people are aware when their family members are leaving. This feature will only send a push alert if the the entire family was previously apart."
|
||||
paragraph "For example, when my wife and I are home together, I know when she's leaving; I don't need an alert for that. If this feature is off, it will send a departure alert when she leaves."
|
||||
input("smartDepartureFeature", "enum", title: "Default is On.", default:"On", options: ["On","Off"])
|
||||
}
|
||||
section("Verbose logging") {
|
||||
paragraph "For debugging, you may log all the app's decisions to the notifications log."
|
||||
input("logToNotifications", "enum", title: "Default is No.", default:"No", options: ["Yes", "No" ])
|
||||
}
|
||||
}
|
||||
|
||||
/****************************
|
||||
Auto-getters and setters
|
||||
*****************************/
|
||||
|
||||
def getDefaultThreshold() {
|
||||
60
|
||||
}
|
||||
|
||||
def setInProgress(value) {
|
||||
state.inProgress = value
|
||||
}
|
||||
|
||||
def getInProgress() {
|
||||
state.inProgress == true
|
||||
}
|
||||
|
||||
def getSmartDeparture() {
|
||||
settings.smartDepartureFeature == 'On'
|
||||
}
|
||||
|
||||
def getLogToNotifications() {
|
||||
settings.logToNotifications == 'Yes'
|
||||
}
|
||||
|
||||
def getThreshold() {
|
||||
settings.timeThreshold ? settings.timeThreshold.toInteger() : defaultThreshold
|
||||
}
|
||||
|
||||
/****************************
|
||||
Framework methods
|
||||
*****************************/
|
||||
|
||||
def installed() {
|
||||
initialize()
|
||||
}
|
||||
|
||||
def logit(msg) {
|
||||
log.debug msg
|
||||
if (logToNotifications) {
|
||||
sendNotificationEvent("[Smart Family Presense] $msg")
|
||||
}
|
||||
}
|
||||
|
||||
def updated() {
|
||||
unsubscribe()
|
||||
initialize()
|
||||
}
|
||||
|
||||
def initialize() {
|
||||
subscribe(familySensors, "presence", presenceHandler)
|
||||
log.debug("Subscribed ${familySensors.toString()} to presenceHandler")
|
||||
|
||||
/*
|
||||
Regular usage shows that, during certain cases such as the hub going offline,
|
||||
or power outages, the app may lose state and not send alerts. This will ensure that
|
||||
it re-evaluates its state at least once per hour.
|
||||
*/
|
||||
|
||||
logit "Scheduling a re-calibration at the top of every hour."
|
||||
schedule("0 0 0/1 1/1 * ? *", reset)
|
||||
reset()
|
||||
}
|
||||
|
||||
/*****************************
|
||||
Smart Family Presence methods
|
||||
******************************/
|
||||
|
||||
def reset() {
|
||||
if (inProgress) {
|
||||
logit "Skipping re-calibration, execution in progress!"
|
||||
return
|
||||
}
|
||||
logit "Re-calibrating."
|
||||
state.baseCase = null
|
||||
state.changedThisTime = []
|
||||
wasApart()
|
||||
}
|
||||
|
||||
def isFamilyTogether() {
|
||||
// Check to see if the entire family has arrived/departed together
|
||||
|
||||
if (state.changedThisTime.size() == 0) {
|
||||
logit "No changes."
|
||||
reset()
|
||||
return
|
||||
}
|
||||
|
||||
logit "People who changed presence: ${state.changedThisTime}"
|
||||
|
||||
def theirState = state.baseCase
|
||||
def notTogether = statusNotEquals(theirState)
|
||||
|
||||
if (notTogether) {
|
||||
// The family is not together, send an alert as normal
|
||||
|
||||
logit "${notTogether.join(", ")} is not with the rest of the family (who are $theirState)"
|
||||
sendPushAlert()
|
||||
}
|
||||
else {
|
||||
/*
|
||||
Special case - When everyone is gone, but they *previously* weren't together,
|
||||
then technically they were apart to begin with and are still apart upon leaving
|
||||
*/
|
||||
|
||||
if (state.wasApart && theirState == 'not present') {
|
||||
logit "Family was previously apart and now all gone. Alert."
|
||||
sendPushAlert()
|
||||
}
|
||||
else {
|
||||
logit "OK! Everyone has arrived/departed together. The family is $theirState"
|
||||
}
|
||||
}
|
||||
inProgress = false
|
||||
reset()
|
||||
}
|
||||
|
||||
def wasApart() {
|
||||
// This is true if everyone is gone, or some were home
|
||||
def allGone = statusEquals('not present') == familySensors
|
||||
def someHome = statusEquals('present') != familySensors
|
||||
if (allGone || someHome) {
|
||||
state.wasApart = true
|
||||
}
|
||||
else {
|
||||
state.wasApart = false
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
def presenceHandler(evt) {
|
||||
def person = evt.displayName
|
||||
|
||||
logit "Presence Event: $person is $evt.value"
|
||||
|
||||
if (! inProgress) {
|
||||
inProgress = true
|
||||
state.baseCase = evt.value
|
||||
logit "First person sensed. Checking for others to be $evt.value in ${threshold} seconds"
|
||||
runIn(threshold, isFamilyTogether, [overwrite: false])
|
||||
}
|
||||
|
||||
if (! state.changedThisTime.contains(person)) {
|
||||
state.changedThisTime.push person
|
||||
}
|
||||
else {
|
||||
// Special case - presence has changed within the threshold. Remove this person.
|
||||
state.changedThisTime = state.changedThisTime - person
|
||||
logit "Ignoring flapping presence event for $person"
|
||||
}
|
||||
}
|
||||
|
||||
def statusEquals(status) {
|
||||
if (status == null) return
|
||||
|
||||
familySensors.findAll {
|
||||
it.currentPresence == status
|
||||
}
|
||||
}
|
||||
|
||||
def statusNotEquals(status) {
|
||||
if (status == null) return
|
||||
|
||||
familySensors.findAll {
|
||||
it.currentPresence != status
|
||||
}
|
||||
}
|
||||
|
||||
def sendPushAlert() {
|
||||
def baseCase = state.baseCase
|
||||
def changedPeople = state.changedThisTime
|
||||
|
||||
if (baseCase == 'not present' && state.wasApart == false && smartDeparture) {
|
||||
logit "Not sending departure alert because smartDeparture is $smartDeparture"
|
||||
return
|
||||
}
|
||||
|
||||
def statuses = [ 'present':[], 'not present':[] ]
|
||||
|
||||
for (sensor in familySensors) {
|
||||
def person = sensor.toString()
|
||||
if (changedPeople.contains(person)) {
|
||||
def currentState = sensor.currentPresence
|
||||
log.debug "$person is now $currentState"
|
||||
statuses[currentState].push person
|
||||
}
|
||||
}
|
||||
logit "Statuses: $statuses"
|
||||
|
||||
// Construct the message payload
|
||||
|
||||
def pushMsg = ""
|
||||
def home = statuses.present
|
||||
def notHome = statuses['not present']
|
||||
String adVerb;
|
||||
|
||||
if (home.size() > 0) {
|
||||
adVerb = home.size > 1 ? "have" : "has"
|
||||
pushMsg += "${home.join(", ")} $adVerb arrived $location.name. "
|
||||
}
|
||||
if (notHome.size() > 0) {
|
||||
adVerb = notHome.size > 1 ? "have" : "has"
|
||||
pushMsg += "${notHome.join(", ")} $adVerb left $location.name"
|
||||
}
|
||||
|
||||
sendPush pushMsg
|
||||
}
|
||||
@@ -39,6 +39,7 @@ preferences {
|
||||
page(name: "completionPage")
|
||||
page(name: "numbersPage")
|
||||
page(name: "controllerExplanationPage")
|
||||
page(name: "unsupportedDevicesPage")
|
||||
}
|
||||
|
||||
def rootPage() {
|
||||
@@ -47,6 +48,9 @@ def rootPage() {
|
||||
section("What to dim") {
|
||||
input(name: "dimmers", type: "capability.switchLevel", title: "Dimmers", description: null, multiple: true, required: true, submitOnChange: true)
|
||||
if (dimmers) {
|
||||
if (dimmersContainUnsupportedDevices()) {
|
||||
href(name: "toUnsupportedDevicesPage", page: "unsupportedDevicesPage", title: "Some of your selected dimmers don't seem to be supported", description: "Tap here to fix it", required: true)
|
||||
}
|
||||
href(name: "toNumbersPage", page: "numbersPage", title: "Duration & Direction", description: numbersPageHrefDescription(), state: "complete")
|
||||
}
|
||||
}
|
||||
@@ -71,6 +75,31 @@ def rootPage() {
|
||||
}
|
||||
}
|
||||
|
||||
def unsupportedDevicesPage() {
|
||||
|
||||
def unsupportedDimmers = dimmers.findAll { !hasSetLevelCommand(it) }
|
||||
|
||||
dynamicPage(name: "unsupportedDevicesPage") {
|
||||
if (unsupportedDimmers) {
|
||||
section("These devices do not support the setLevel command") {
|
||||
unsupportedDimmers.each {
|
||||
paragraph deviceLabel(it)
|
||||
}
|
||||
}
|
||||
section {
|
||||
input(name: "dimmers", type: "capability.sensor", title: "Please remove the above devices from this list.", submitOnChange: true, multiple: true)
|
||||
}
|
||||
section {
|
||||
paragraph "If you think there is a mistake here, please contact support."
|
||||
}
|
||||
} else {
|
||||
section {
|
||||
paragraph "You're all set. You can hit the back button, now. Thanks for cleaning up your settings :)"
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
def controllerExplanationPage() {
|
||||
dynamicPage(name: "controllerExplanationPage", title: "How To Control Gentle Wake Up") {
|
||||
|
||||
@@ -528,14 +557,16 @@ def updateDimmers(percentComplete) {
|
||||
} else {
|
||||
|
||||
def shouldChangeColors = (colorize && colorize != "false")
|
||||
def canChangeColors = hasSetColorCommand(dimmer)
|
||||
|
||||
log.debug "Setting ${deviceLabel(dimmer)} to ${nextLevel}"
|
||||
|
||||
if (shouldChangeColors && canChangeColors) {
|
||||
dimmer.setColor([hue: getHue(dimmer, nextLevel), saturation: 100, level: nextLevel])
|
||||
} else {
|
||||
if (shouldChangeColors && hasSetColorCommand(dimmer)) {
|
||||
def hue = getHue(dimmer, nextLevel)
|
||||
log.debug "Setting ${deviceLabel(dimmer)} level to ${nextLevel} and hue to ${hue}"
|
||||
dimmer.setColor([hue: hue, saturation: 100, level: nextLevel])
|
||||
} else if (hasSetLevelCommand(dimmer)) {
|
||||
log.debug "Setting ${deviceLabel(dimmer)} level to ${nextLevel}"
|
||||
dimmer.setLevel(nextLevel)
|
||||
} else {
|
||||
log.warn "${deviceLabel(dimmer)} does not have setColor or setLevel commands."
|
||||
}
|
||||
|
||||
}
|
||||
@@ -817,24 +848,21 @@ private getRedHue(level) {
|
||||
if (level >= 96) return 17
|
||||
}
|
||||
|
||||
private dimmersContainUnsupportedDevices() {
|
||||
def found = dimmers.find { hasSetLevelCommand(it) == false }
|
||||
return found != null
|
||||
}
|
||||
|
||||
private hasSetLevelCommand(device) {
|
||||
def isDimmer = false
|
||||
device.supportedCommands.each {
|
||||
if (it.name.contains("setLevel")) {
|
||||
isDimmer = true
|
||||
}
|
||||
}
|
||||
return isDimmer
|
||||
return hasCommand(device, "setLevel")
|
||||
}
|
||||
|
||||
private hasSetColorCommand(device) {
|
||||
def hasColor = false
|
||||
device.supportedCommands.each {
|
||||
if (it.name.contains("setColor")) {
|
||||
hasColor = true
|
||||
}
|
||||
}
|
||||
return hasColor
|
||||
return hasCommand(device, "setColor")
|
||||
}
|
||||
|
||||
private hasCommand(device, String command) {
|
||||
return (device.supportedCommands.find { it.name == command } != null)
|
||||
}
|
||||
|
||||
private dimmersWithSetColorCommand() {
|
||||
@@ -1073,4 +1101,4 @@ def hasStartLevel() {
|
||||
|
||||
def hasEndLevel() {
|
||||
return (endLevel != null && endLevel != "")
|
||||
}
|
||||
}
|
||||
@@ -688,7 +688,7 @@ def validateCommand(device, command) {
|
||||
def capabilityCommands = getDeviceCapabilityCommands(device.capabilities)
|
||||
def currentDeviceCapability = getCapabilityName(device)
|
||||
if (currentDeviceCapability != "" && capabilityCommands[currentDeviceCapability]) {
|
||||
return command in capabilityCommands[currentDeviceCapability] ? true : false
|
||||
return (command in capabilityCommands[currentDeviceCapability] || (currentDeviceCapability == "Switch" && command == "setLevel" && device.hasCommand("setLevel"))) ? true : false
|
||||
} else {
|
||||
// Handling other device types here, which don't accept commands
|
||||
httpError(400, "Bad request.")
|
||||
@@ -823,8 +823,8 @@ def deviceHandler(evt) {
|
||||
}
|
||||
|
||||
def sendToHarmony(evt, String callbackUrl) {
|
||||
def callback = new URI(callbackUrl)
|
||||
if(isIP(callback.host)){
|
||||
def callback = new URI(callbackUrl)
|
||||
if (callback.port != -1) {
|
||||
def host = callback.port != -1 ? "${callback.host}:${callback.port}" : callback.host
|
||||
def path = callback.query ? "${callback.path}?${callback.query}".toString() : callback.path
|
||||
sendHubCommand(new physicalgraph.device.HubAction(
|
||||
@@ -852,25 +852,6 @@ def sendToHarmony(evt, String callbackUrl) {
|
||||
}
|
||||
}
|
||||
|
||||
public static boolean isIP(String str) {
|
||||
try {
|
||||
String[] parts = str.split("\\.");
|
||||
if (parts.length != 4) return false;
|
||||
for (int i = 0; i < 4; ++i) {
|
||||
int p
|
||||
try {
|
||||
p = Integer.parseInt(parts[i]);
|
||||
} catch (Exception e) {
|
||||
return false;
|
||||
}
|
||||
if (p > 255 || p < 0) return false;
|
||||
}
|
||||
return true;
|
||||
} catch (Exception e) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
def listHubs() {
|
||||
location.hubs?.findAll { it.type.toString() == "PHYSICAL" }?.collect { hubItem(it) }
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user