mirror of
https://github.com/mtan93/SmartThingsPublic.git
synced 2026-03-30 22:04:30 +01:00
Compare commits
16 Commits
MSA-1395-1
...
MSA-1399-1
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
9a2eb5fa5d | ||
|
|
2b3a4e1278 | ||
|
|
825e693efd | ||
|
|
686c8f7337 | ||
|
|
11f4e42fe9 | ||
|
|
bc459ae178 | ||
|
|
1392b6e1a5 | ||
|
|
3db96faa00 | ||
|
|
399fbcb676 | ||
|
|
eed1ced71b | ||
|
|
d080833d5c | ||
|
|
e998528e8e | ||
|
|
d85566bb98 | ||
|
|
e1a5b4dd27 | ||
|
|
a2baa37901 | ||
|
|
922ab45343 |
@@ -0,0 +1,350 @@
|
|||||||
|
/**
|
||||||
|
* LANnouncer Alerter (Formerly LANdroid - but Google didn't like that much.)
|
||||||
|
*
|
||||||
|
* Requires the LANnouncer android app; https://play.google.com/store/apps/details?id=com.keybounce.lannouncer
|
||||||
|
* See http://www.keybounce.com/LANdroidHowTo/LANdroid.html for full downloads and instructions.
|
||||||
|
* SmartThings thread: https://community.smartthings.com/t/android-as-a-speech-alarm-device-released/30282/12
|
||||||
|
*
|
||||||
|
* Note: Only Siren and Strobe from the U.I. or Alarm capabilities default to continuous.
|
||||||
|
*
|
||||||
|
* Version 1.24 16 July 2016
|
||||||
|
*
|
||||||
|
*
|
||||||
|
* Copyright 2015-2016 Tony McNamara
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*
|
||||||
|
* To Do: Add string return to Image Capture attribute
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
metadata {
|
||||||
|
definition (name: "LANnouncer Alerter", namespace: "KeyBounce", author: "Tony McNamara") {
|
||||||
|
capability "Alarm"
|
||||||
|
capability "Speech Synthesis"
|
||||||
|
capability "Notification"
|
||||||
|
capability "Tone"
|
||||||
|
capability "Image Capture"
|
||||||
|
attribute "LANdroidSMS","string"
|
||||||
|
/* Per http://docs.smartthings.com/en/latest/device-type-developers-guide/overview.html#actuator-and-sensor */
|
||||||
|
capability "Sensor"
|
||||||
|
capability "Actuator"
|
||||||
|
|
||||||
|
// Custom Commands
|
||||||
|
/** Retrieve image, formatted for SmartThings, from camera by name. */
|
||||||
|
command "chime"
|
||||||
|
command "doorbell"
|
||||||
|
command "ipCamSequence", ["number"]
|
||||||
|
command "retrieveAndWait", ["string"]
|
||||||
|
command "retrieveFirstAndWait"
|
||||||
|
command "retrieveSecondAndWait"
|
||||||
|
}
|
||||||
|
preferences {
|
||||||
|
input("DeviceLocalLan", "string", title:"Android IP Address", description:"Please enter your tablet's I.P. address", defaultValue:"" , required: false, displayDuringSetup: true)
|
||||||
|
input("DevicePort", "string", title:"Android Port", description:"Port the Android device listens on", defaultValue:"1035", required: false, displayDuringSetup: true)
|
||||||
|
input("ReplyOnEmpty", "bool", title:"Say Nothing", description:"When no speech is found, announce LANdroid? (Needed for the speech and notify tiles to work)", defaultValue: true, displayDuringSetup: true)
|
||||||
|
input("AlarmContinuous", "bool", title:"Continuous Alarm (vs 10 sec.)", description: "When on, the alarm will sound until Stop is issued.", defaultValue: false, displayDuringSetup: true)
|
||||||
|
}
|
||||||
|
|
||||||
|
simulator {
|
||||||
|
// reply messages
|
||||||
|
["strobe","siren","both","off"].each
|
||||||
|
{
|
||||||
|
reply "$it": "alarm:$it"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
tiles {
|
||||||
|
standardTile("alarm", "device.alarm", width: 2, height: 2) {
|
||||||
|
state "off", label:'off', action:'alarm.both', icon:"st.alarm.alarm.alarm", backgroundColor:"#ffffff"
|
||||||
|
state "strobe", label:'strobe!', action:'alarm.off', icon:"st.Lighting.light11", backgroundColor:"#e86d13"
|
||||||
|
state "siren", label:'siren!', action:'alarm.off', icon:"st.alarm.alarm.alarm", backgroundColor:"#e86d13"
|
||||||
|
state "both", label:'alarm!', action:'alarm.off', icon:"st.alarm.alarm.alarm", backgroundColor:"#e86d13"
|
||||||
|
}
|
||||||
|
standardTile("strobe", "device.alarm", inactiveLabel: false, decoration: "flat") {
|
||||||
|
state "default", label:'', action:"alarm.strobe", icon:"st.secondary.strobe"
|
||||||
|
}
|
||||||
|
|
||||||
|
standardTile("siren", "device.alarm", inactiveLabel: false, decoration: "flat") {
|
||||||
|
state "default", label:'', action:"alarm.siren", icon:"st.secondary.siren"
|
||||||
|
}
|
||||||
|
standardTile("off", "device.alarm", inactiveLabel: false, decoration: "flat") {
|
||||||
|
state "default", label:'Off', action:"alarm.off", icon:"st.quirky.spotter.quirky-spotter-sound-off"
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Apparently can't show image attributes on tiles. */
|
||||||
|
standardTile("take", "device.image", width: 1, height: 1, canChangeIcon: false, canChangeBackground: false)
|
||||||
|
{
|
||||||
|
state "take", label: "Take", action: "Image Capture.take", icon: "st.camera.camera", backgroundColor: "#FFFFFF", nextState:"taking"
|
||||||
|
state "taking", label:'Taking', action: "", icon: "st.camera.take-photo", backgroundColor: "#53a7c0"
|
||||||
|
state "image", label: "Take", action: "Image Capture.take", icon: "st.camera.camera", backgroundColor: "#FFFFFF", nextState:"taking"
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
standardTile("speak", "device.speech", inactiveLabel: false, decoration: "flat")
|
||||||
|
{
|
||||||
|
state "default", label:'Speak', action:"Speech Synthesis.speak", icon:"st.Electronics.electronics13"
|
||||||
|
}
|
||||||
|
standardTile("toast", "device.notification", inactiveLabel: false, decoration: "flat") {
|
||||||
|
state "default", label:'Notify', action:"notification.deviceNotification", icon:"st.Kids.kids1"
|
||||||
|
}
|
||||||
|
standardTile("beep", "device.tone", inactiveLabel: false, decoration: "flat") {
|
||||||
|
state "default", label:'Tone', action:"tone.beep", icon:"st.Entertainment.entertainment2"
|
||||||
|
}
|
||||||
|
carouselTile("cameraDetails", "device.image", width: 3, height: 2) { }
|
||||||
|
|
||||||
|
main (["alarm", "take"]);
|
||||||
|
details(["alarm","strobe","siren","off","speak", "take","toast","beep", "cameraDetails"]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Generally matches TTSServer/app/build.gradle */
|
||||||
|
String getVersion() {return "24 built July 2016";}
|
||||||
|
|
||||||
|
|
||||||
|
// handle commands
|
||||||
|
def off() {
|
||||||
|
log.debug "Executing 'off'"
|
||||||
|
sendEvent(name:"alarm", value:"off")
|
||||||
|
def command="&ALARM=STOP&FLASH=STOP&"+getDoneString();
|
||||||
|
sendCommands(command)
|
||||||
|
}
|
||||||
|
|
||||||
|
def strobe() {
|
||||||
|
log.debug "Executing 'strobe'"
|
||||||
|
// TODO: handle 'strobe' command
|
||||||
|
sendEvent(name:"alarm", value:"strobe")
|
||||||
|
def command="&FLASH=STROBE&"+getDoneString()
|
||||||
|
if (AlarmContinuous)
|
||||||
|
{
|
||||||
|
command="&FLASH=CONTINUOUS&"+getDoneString()
|
||||||
|
}
|
||||||
|
sendCommands(command)
|
||||||
|
}
|
||||||
|
|
||||||
|
def siren() {
|
||||||
|
log.debug "Executing 'siren'"
|
||||||
|
sendEvent(name:"alarm", value:"siren")
|
||||||
|
def command="&ALARM=SIREN&"+getDoneString()
|
||||||
|
if (AlarmContinuous)
|
||||||
|
{
|
||||||
|
command="&ALARM=SIREN:CONTINUOUS&"+getDoneString()
|
||||||
|
}
|
||||||
|
sendCommands(command)
|
||||||
|
}
|
||||||
|
|
||||||
|
def beep() {
|
||||||
|
log.debug "Executing 'beep'"
|
||||||
|
def command="&ALARM=CHIME&"+getDoneString()
|
||||||
|
sendCommands(command)
|
||||||
|
}
|
||||||
|
|
||||||
|
def both() {
|
||||||
|
log.debug "Executing 'both'"
|
||||||
|
sendEvent(name:"alarm", value:"both")
|
||||||
|
def command="&ALARM=ON&FLASH=ON&"+getDoneString()
|
||||||
|
if (AlarmContinuous)
|
||||||
|
{
|
||||||
|
command="&ALARM=SIREN:CONTINUOUS&FLASH=CONTINUOUS&"+getDoneString()
|
||||||
|
}
|
||||||
|
sendCommands(command)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def speak(toSay) {
|
||||||
|
log.debug "Executing 'speak'"
|
||||||
|
if (!toSay?.trim()) {
|
||||||
|
if (ReplyOnEmpty) {
|
||||||
|
toSay = "LANnouncer Version ${version}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (toSay?.trim()) {
|
||||||
|
def command="&SPEAK="+toSay+"&"+getDoneString()
|
||||||
|
sendCommands(command)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def deviceNotification(toToast) {
|
||||||
|
log.debug "Executing notification with "+toToast
|
||||||
|
if (!toToast?.trim()) {
|
||||||
|
if (ReplyOnEmpty) {
|
||||||
|
toToast = "LANnouncer Version ${version}";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (toToast?.trim()) {
|
||||||
|
def command="&TOAST="+toToast+"&"+getDoneString()
|
||||||
|
sendCommands(command)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def chime() {
|
||||||
|
log.debug "Executing 'chime'"
|
||||||
|
// TODO: handle 'siren' command
|
||||||
|
def command="&ALARM=CHIME&"+getDoneString()
|
||||||
|
sendCommands(command)
|
||||||
|
}
|
||||||
|
|
||||||
|
def doorbell() {
|
||||||
|
log.debug "Executing 'doorbell'"
|
||||||
|
// TODO: handle 'siren' command
|
||||||
|
def command="&ALARM=DOORBELL&"+getDoneString()
|
||||||
|
sendCommands(command)
|
||||||
|
}
|
||||||
|
|
||||||
|
def ipCamSequence(cameraNumber) {
|
||||||
|
if (cameraNumber == 1) {
|
||||||
|
def command="&RETRIEVESEQ=FIRST&"+getDoneString()
|
||||||
|
sendIPCommand(command, true)
|
||||||
|
|
||||||
|
} else {
|
||||||
|
def command="&RETRIEVESEQ=SECOND&"+getDoneString()
|
||||||
|
sendIPCommand(command, true)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def retrieveFirstAndWait() {
|
||||||
|
retrieveAndWait("FIRST");
|
||||||
|
}
|
||||||
|
def retrieveSecondAndWait() {
|
||||||
|
retrieveAndWait("SECOND");
|
||||||
|
}
|
||||||
|
def retrieveAndWait(cameraName) {
|
||||||
|
log.info("Requesting image from camera ${cameraName}");
|
||||||
|
def command="&RETRIEVE="+cameraName+"&STSHRINK=TRUE&"+getDoneString()
|
||||||
|
sendIPCommand(command, true)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def take() {
|
||||||
|
// This won't result in received file. Can't handle large or binaries in hub.
|
||||||
|
log.debug "Executing 'take'"
|
||||||
|
def command="&PHOTO=BACK&STSHRINK=TRUE&"+getDoneString()
|
||||||
|
sendIPCommand(command, true)
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Send to IP and to SMS as appropriate */
|
||||||
|
private sendCommands(command) {
|
||||||
|
log.info "Command request: "+command
|
||||||
|
sendSMSCommand(command)
|
||||||
|
sendIPCommand(command)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private sendIPCommand(commandString, sendToS3 = false) {
|
||||||
|
log.info "Sending command "+ commandString+" to "+DeviceLocalLan+":"+DevicePort
|
||||||
|
if (DeviceLocalLan?.trim()) {
|
||||||
|
def hosthex = convertIPtoHex(DeviceLocalLan)
|
||||||
|
def porthex = convertPortToHex(DevicePort)
|
||||||
|
device.deviceNetworkId = "$hosthex:$porthex"
|
||||||
|
|
||||||
|
def headers = [:]
|
||||||
|
headers.put("HOST", "$DeviceLocalLan:$DevicePort")
|
||||||
|
|
||||||
|
def method = "GET"
|
||||||
|
|
||||||
|
def hubAction = new physicalgraph.device.HubAction(
|
||||||
|
method: method,
|
||||||
|
path: "/"+commandString,
|
||||||
|
headers: headers
|
||||||
|
);
|
||||||
|
if (sendToS3 == true)
|
||||||
|
{
|
||||||
|
hubAction.options = [outputMsgToS3:true];
|
||||||
|
}
|
||||||
|
log.debug hubActionw
|
||||||
|
hubAction;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private sendSMSCommand(commandString) {
|
||||||
|
def preface = "+@TTSSMS@+"
|
||||||
|
def smsValue = preface+"&"+commandString
|
||||||
|
state.lastsmscommand = smsValue
|
||||||
|
sendEvent(name: "LANdroidSMS", value: smsValue, isStateChange: true)
|
||||||
|
/*
|
||||||
|
if (SMSPhone?.trim()) {
|
||||||
|
sendSmsMessage(SMSPhone, preface+"&"+commandString)
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
}
|
||||||
|
|
||||||
|
private String getDoneString() {
|
||||||
|
return "@DONE@"
|
||||||
|
}
|
||||||
|
|
||||||
|
def parse(String description) {
|
||||||
|
log.debug "Parsing '${description}'"
|
||||||
|
def map = parseLanMessage(description);
|
||||||
|
log.debug "As LAN: " + map;
|
||||||
|
if ((map.headers) && (map.headers.'Content-Type' != null) && (map.headers.'Content-Type'.contains("image/jpeg")) )
|
||||||
|
{ // Store the file
|
||||||
|
if(map.body)
|
||||||
|
{
|
||||||
|
storeImage(getPictureName(), map.body);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/* 'index:0F, mac:0073E023A13A, ip:C0A80114, port:040B, requestId:f9036fb2-9637-40b8-b2c5-71ba5a09fd3e, bucket:smartthings-device-conn-temp, key:fc8e3dfd-5035-40a2-8adc-a312926f9034' */
|
||||||
|
|
||||||
|
else if (map.bucket && map.key)
|
||||||
|
{ // S3 pointer; retrieve image from it to store.
|
||||||
|
try {
|
||||||
|
def s3ObjectContent; // Needed for scope of try-catch
|
||||||
|
def imageBytes = getS3Object(map.bucket, map.key + ".jpg")
|
||||||
|
|
||||||
|
if(imageBytes)
|
||||||
|
{
|
||||||
|
log.info ("Got image bytes; saving them.")
|
||||||
|
s3ObjectContent = imageBytes.getObjectContent()
|
||||||
|
def bytes = new ByteArrayInputStream(s3ObjectContent.bytes)
|
||||||
|
storeImage(getPictureName(), bytes)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch(Exception e)
|
||||||
|
{
|
||||||
|
log.error e
|
||||||
|
}
|
||||||
|
finally {
|
||||||
|
//explicitly close the stream
|
||||||
|
if (s3ObjectContent) { s3ObjectContent.close() }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// Image Capture handling
|
||||||
|
/* Note that images are stored in https://graph.api.smartthings.com/api/s3/smartthings-smartsense-camera/[IMAGE-ID4],
|
||||||
|
* where [IMAGE-ID] is listed in the IDE under My Devices > [Camera] > Current States: Image. That page is updated as pictures are taken.
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
private getPictureName() {
|
||||||
|
def pictureUuid = java.util.UUID.randomUUID().toString().replaceAll('-', '')
|
||||||
|
log.debug pictureUuid
|
||||||
|
def picName = device.deviceNetworkId.replaceAll(':', '') + "_$pictureUuid" + ".jpg"
|
||||||
|
return picName
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private String convertIPtoHex(ipAddress) {
|
||||||
|
String hex = ipAddress.tokenize( '.' ).collect { String.format( '%02X', it.toInteger() ) }.join()
|
||||||
|
log.debug "IP address entered is $ipAddress and the converted hex code is $hex"
|
||||||
|
return hex
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private String convertPortToHex(port) {
|
||||||
|
String hexport = port.toString().format( '%04X', port.toInteger() )
|
||||||
|
log.debug hexport
|
||||||
|
return hexport
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -255,7 +255,8 @@ private Map getBatteryResult(rawValue) {
|
|||||||
def minVolts = 2.1
|
def minVolts = 2.1
|
||||||
def maxVolts = 3.0
|
def maxVolts = 3.0
|
||||||
def pct = (volts - minVolts) / (maxVolts - minVolts)
|
def pct = (volts - minVolts) / (maxVolts - minVolts)
|
||||||
result.value = Math.min(100, (int) pct * 100)
|
def roundedPct = Math.round(pct * 100)
|
||||||
|
result.value = Math.min(100, roundedPct)
|
||||||
result.descriptionText = "{{ device.displayName }} battery was {{ value }}%"
|
result.descriptionText = "{{ device.displayName }} battery was {{ value }}%"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -271,7 +271,8 @@ private Map getBatteryResult(rawValue) {
|
|||||||
def minVolts = 2.1
|
def minVolts = 2.1
|
||||||
def maxVolts = 3.0
|
def maxVolts = 3.0
|
||||||
def pct = (volts - minVolts) / (maxVolts - minVolts)
|
def pct = (volts - minVolts) / (maxVolts - minVolts)
|
||||||
result.value = Math.min(100, (int) pct * 100)
|
def roundedPct = Math.round(pct * 100)
|
||||||
|
result.value = Math.min(100, roundedPct)
|
||||||
result.descriptionText = "{{ device.displayName }} battery was {{ value }}%"
|
result.descriptionText = "{{ device.displayName }} battery was {{ value }}%"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -240,7 +240,8 @@ private Map getBatteryResult(rawValue) {
|
|||||||
def minVolts = 2.1
|
def minVolts = 2.1
|
||||||
def maxVolts = 3.0
|
def maxVolts = 3.0
|
||||||
def pct = (volts - minVolts) / (maxVolts - minVolts)
|
def pct = (volts - minVolts) / (maxVolts - minVolts)
|
||||||
result.value = Math.min(100, (int) pct * 100)
|
def roundedPct = Math.round(pct * 100)
|
||||||
|
result.value = Math.min(100, roundedPct)
|
||||||
result.descriptionText = "${linkText} battery was ${result.value}%"
|
result.descriptionText = "${linkText} battery was ${result.value}%"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -338,7 +338,8 @@ private Map getBatteryResult(rawValue) {
|
|||||||
def minVolts = 2.1
|
def minVolts = 2.1
|
||||||
def maxVolts = 3.0
|
def maxVolts = 3.0
|
||||||
def pct = (volts - minVolts) / (maxVolts - minVolts)
|
def pct = (volts - minVolts) / (maxVolts - minVolts)
|
||||||
result.value = Math.min(100, (int) pct * 100)
|
def roundedPct = Math.round(pct * 100)
|
||||||
|
result.value = Math.min(100, roundedPct)
|
||||||
result.descriptionText = "{{ device.displayName }} battery was {{ value }}%"
|
result.descriptionText = "{{ device.displayName }} battery was {{ value }}%"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -234,7 +234,8 @@ def getTemperature(value) {
|
|||||||
def minVolts = 2.1
|
def minVolts = 2.1
|
||||||
def maxVolts = 3.0
|
def maxVolts = 3.0
|
||||||
def pct = (volts - minVolts) / (maxVolts - minVolts)
|
def pct = (volts - minVolts) / (maxVolts - minVolts)
|
||||||
result.value = Math.min(100, (int) pct * 100)
|
def roundedPct = Math.round(pct * 100)
|
||||||
|
result.value = Math.min(100, roundedPct)
|
||||||
result.descriptionText = "${linkText} battery was ${result.value}%"
|
result.descriptionText = "${linkText} battery was ${result.value}%"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -229,7 +229,8 @@ private Map getBatteryResult(rawValue) {
|
|||||||
def minVolts = 2.1
|
def minVolts = 2.1
|
||||||
def maxVolts = 3.0
|
def maxVolts = 3.0
|
||||||
def pct = (volts - minVolts) / (maxVolts - minVolts)
|
def pct = (volts - minVolts) / (maxVolts - minVolts)
|
||||||
result.value = Math.min(100, (int) pct * 100)
|
def roundedPct = Math.round(pct * 100)
|
||||||
|
result.value = Math.min(100, roundedPct)
|
||||||
result.descriptionText = "${linkText} battery was ${result.value}%"
|
result.descriptionText = "${linkText} battery was ${result.value}%"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -205,7 +205,8 @@ private Map getBatteryResult(rawValue) {
|
|||||||
def minVolts = 2.1
|
def minVolts = 2.1
|
||||||
def maxVolts = 3.0
|
def maxVolts = 3.0
|
||||||
def pct = (volts - minVolts) / (maxVolts - minVolts)
|
def pct = (volts - minVolts) / (maxVolts - minVolts)
|
||||||
result.value = Math.min(100, (int) pct * 100)
|
def roundedPct = Math.round(pct * 100)
|
||||||
|
result.value = Math.min(100, roundedPct)
|
||||||
result.descriptionText = "${linkText} battery was ${result.value}%"
|
result.descriptionText = "${linkText} battery was ${result.value}%"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -223,7 +223,8 @@ private Map getBatteryResult(rawValue) {
|
|||||||
def minVolts = 2.1
|
def minVolts = 2.1
|
||||||
def maxVolts = 3.0
|
def maxVolts = 3.0
|
||||||
def pct = (volts - minVolts) / (maxVolts - minVolts)
|
def pct = (volts - minVolts) / (maxVolts - minVolts)
|
||||||
result.value = Math.min(100, (int) pct * 100)
|
def roundedPct = Math.round(pct * 100)
|
||||||
|
result.value = Math.min(100, roundedPct)
|
||||||
result.descriptionText = "${linkText} battery was ${result.value}%"
|
result.descriptionText = "${linkText} battery was ${result.value}%"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -28,7 +28,6 @@ metadata {
|
|||||||
fingerprint deviceId: "0x0701", inClusters: "0x5E,0x98"
|
fingerprint deviceId: "0x0701", inClusters: "0x5E,0x98"
|
||||||
fingerprint deviceId: "0x0701", inClusters: "0x5E,0x86,0x72,0x98", outClusters: "0x5A,0x82"
|
fingerprint deviceId: "0x0701", inClusters: "0x5E,0x86,0x72,0x98", outClusters: "0x5A,0x82"
|
||||||
fingerprint deviceId: "0x0701", inClusters: "0x5E,0x80,0x71,0x85,0x70,0x72,0x86,0x30,0x31,0x84,0x59,0x73,0x5A,0x8F,0x98,0x7A", outClusters:"0x20" // Philio multi+
|
fingerprint deviceId: "0x0701", inClusters: "0x5E,0x80,0x71,0x85,0x70,0x72,0x86,0x30,0x31,0x84,0x59,0x73,0x5A,0x8F,0x98,0x7A", outClusters:"0x20" // Philio multi+
|
||||||
fingerprint deviceId: "0x0701", inClusters: "0x5E,0x72,0x5A,0x80,0x73,0x84,0x85,0x59,0x71,0x70,0x7A,0x98" // Vision door/window
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// simulator metadata
|
// simulator metadata
|
||||||
@@ -83,12 +82,12 @@ def updated() {
|
|||||||
cmds = [
|
cmds = [
|
||||||
command(zwave.manufacturerSpecificV2.manufacturerSpecificGet()),
|
command(zwave.manufacturerSpecificV2.manufacturerSpecificGet()),
|
||||||
"delay 1200",
|
"delay 1200",
|
||||||
zwave.wakeUpV1.wakeUpNoMoreInformation()
|
zwave.wakeUpV1.wakeUpNoMoreInformation().format()
|
||||||
]
|
]
|
||||||
} else if (!state.lastbat) {
|
} else if (!state.lastbat) {
|
||||||
cmds = []
|
cmds = []
|
||||||
} else {
|
} else {
|
||||||
cmds = [zwave.wakeUpV1.wakeUpNoMoreInformation()]
|
cmds = [zwave.wakeUpV1.wakeUpNoMoreInformation().format()]
|
||||||
}
|
}
|
||||||
response(cmds)
|
response(cmds)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,251 +0,0 @@
|
|||||||
/**
|
|
||||||
* 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
|
|
||||||
}
|
|
||||||
@@ -720,7 +720,7 @@ def completionPercentage() {
|
|||||||
|
|
||||||
def now = new Date().getTime()
|
def now = new Date().getTime()
|
||||||
def timeElapsed = now - atomicState.start
|
def timeElapsed = now - atomicState.start
|
||||||
def totalRunTime = totalRunTimeMillis()
|
def totalRunTime = totalRunTimeMillis() ?: 1
|
||||||
def percentComplete = timeElapsed / totalRunTime * 100
|
def percentComplete = timeElapsed / totalRunTime * 100
|
||||||
log.debug "percentComplete: ${percentComplete}"
|
log.debug "percentComplete: ${percentComplete}"
|
||||||
|
|
||||||
|
|||||||
@@ -689,7 +689,7 @@ def parse(childDevice, description) {
|
|||||||
log.warn "Parsing Body failed - trying again..."
|
log.warn "Parsing Body failed - trying again..."
|
||||||
poll()
|
poll()
|
||||||
}
|
}
|
||||||
if (body instanceof java.util.HashMap) {
|
if (body instanceof java.util.Map) {
|
||||||
//poll response
|
//poll response
|
||||||
def bulbs = getChildDevices()
|
def bulbs = getChildDevices()
|
||||||
for (bulb in body) {
|
for (bulb in body) {
|
||||||
|
|||||||
Reference in New Issue
Block a user