mirror of
https://github.com/mtan93/SmartThingsPublic.git
synced 2026-04-15 06:13:06 +01:00
Initial commit
This commit is contained in:
@@ -0,0 +1,209 @@
|
||||
/**
|
||||
* 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.
|
||||
*
|
||||
* Ridiculously Automated Garage Door
|
||||
*
|
||||
* Author: SmartThings
|
||||
* Date: 2013-03-10
|
||||
*
|
||||
* Monitors arrival and departure of car(s) and
|
||||
*
|
||||
* 1) opens door when car arrives,
|
||||
* 2) closes door after car has departed (for N minutes),
|
||||
* 3) opens door when car door motion is detected,
|
||||
* 4) closes door when door was opened due to arrival and interior door is closed.
|
||||
*/
|
||||
|
||||
definition(
|
||||
name: "Ridiculously Automated Garage Door",
|
||||
namespace: "smartthings",
|
||||
author: "SmartThings",
|
||||
description: "Monitors arrival and departure of car(s) and 1) opens door when car arrives, 2) closes door after car has departed (for N minutes), 3) opens door when car door motion is detected, 4) closes door when door was opened due to arrival and interior door is closed.",
|
||||
category: "Convenience",
|
||||
iconUrl: "https://s3.amazonaws.com/smartapp-icons/Meta/garage_contact.png",
|
||||
iconX2Url: "https://s3.amazonaws.com/smartapp-icons/Meta/garage_contact@2x.png"
|
||||
)
|
||||
|
||||
preferences {
|
||||
|
||||
section("Garage door") {
|
||||
input "doorSensor", "capability.contactSensor", title: "Which sensor?"
|
||||
input "doorSwitch", "capability.momentary", title: "Which switch?"
|
||||
input "openThreshold", "number", title: "Warn when open longer than (optional)",description: "Number of minutes", required: false
|
||||
input("recipients", "contact", title: "Send notifications to") {
|
||||
input "phone", "phone", title: "Warn with text message (optional)", description: "Phone Number", required: false
|
||||
}
|
||||
}
|
||||
section("Car(s) using this garage door") {
|
||||
input "cars", "capability.presenceSensor", title: "Presence sensor", description: "Which car(s)?", multiple: true, required: false
|
||||
input "carDoorSensors", "capability.accelerationSensor", title: "Car door sensor(s)", description: "Which car(s)?", multiple: true, required: false
|
||||
}
|
||||
section("Interior door (optional)") {
|
||||
input "interiorDoorSensor", "capability.contactSensor", title: "Contact sensor?", required: false
|
||||
}
|
||||
section("False alarm threshold (defaults to 10 min)") {
|
||||
input "falseAlarmThreshold", "number", title: "Number of minutes", required: false
|
||||
}
|
||||
}
|
||||
|
||||
def installed() {
|
||||
log.trace "installed()"
|
||||
subscribe()
|
||||
}
|
||||
|
||||
def updated() {
|
||||
log.trace "updated()"
|
||||
unsubscribe()
|
||||
subscribe()
|
||||
}
|
||||
|
||||
def subscribe() {
|
||||
log.debug "present: ${cars.collect{it.displayName + ': ' + it.currentPresence}}"
|
||||
subscribe(doorSensor, "contact", garageDoorContact)
|
||||
|
||||
subscribe(cars, "presence", carPresence)
|
||||
subscribe(carDoorSensors, "acceleration", accelerationActive)
|
||||
|
||||
if (interiorDoorSensor) {
|
||||
subscribe(interiorDoorSensor, "contact.closed", interiorDoorClosed)
|
||||
}
|
||||
}
|
||||
|
||||
def doorOpenCheck()
|
||||
{
|
||||
final thresholdMinutes = openThreshold
|
||||
if (thresholdMinutes) {
|
||||
def currentState = doorSensor.contactState
|
||||
log.debug "doorOpenCheck"
|
||||
if (currentState?.value == "open") {
|
||||
log.debug "open for ${now() - currentState.date.time}, openDoorNotificationSent: ${state.openDoorNotificationSent}"
|
||||
if (!state.openDoorNotificationSent && now() - currentState.date.time > thresholdMinutes * 60 *1000) {
|
||||
def msg = "${doorSwitch.displayName} was been open for ${thresholdMinutes} minutes"
|
||||
log.info msg
|
||||
|
||||
if (location.contactBookEnabled) {
|
||||
sendNotificationToContacts(msg, recipients)
|
||||
}
|
||||
else {
|
||||
sendPush msg
|
||||
if (phone) {
|
||||
sendSms phone, msg
|
||||
}
|
||||
}
|
||||
state.openDoorNotificationSent = true
|
||||
}
|
||||
}
|
||||
else {
|
||||
state.openDoorNotificationSent = false
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
def carPresence(evt)
|
||||
{
|
||||
log.info "$evt.name: $evt.value"
|
||||
// time in which there must be no "not present" events in order to open the door
|
||||
final openDoorAwayInterval = falseAlarmThreshold ? falseAlarmThreshold * 60 : 600
|
||||
|
||||
if (evt.value == "present") {
|
||||
// A car comes home
|
||||
|
||||
def car = getCar(evt)
|
||||
def t0 = new Date(now() - (openDoorAwayInterval * 1000))
|
||||
def states = car.statesSince("presence", t0)
|
||||
def recentNotPresentState = states.find{it.value == "not present"}
|
||||
|
||||
if (recentNotPresentState) {
|
||||
log.debug "Not opening ${doorSwitch.displayName} since car was not present at ${recentNotPresentState.date}, less than ${openDoorAwayInterval} sec ago"
|
||||
}
|
||||
else {
|
||||
if (doorSensor.currentContact == "closed") {
|
||||
openDoor()
|
||||
sendPush "Opening garage door due to arrival of ${car.displayName}"
|
||||
state.appOpenedDoor = now()
|
||||
}
|
||||
else {
|
||||
log.debug "door already open"
|
||||
}
|
||||
}
|
||||
}
|
||||
else {
|
||||
// A car departs
|
||||
if (doorSensor.currentContact == "open") {
|
||||
closeDoor()
|
||||
log.debug "Closing ${doorSwitch.displayName} after departure"
|
||||
sendPush("Closing ${doorSwitch.displayName} after departure")
|
||||
|
||||
}
|
||||
else {
|
||||
log.debug "Not closing ${doorSwitch.displayName} because its already closed"
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
def garageDoorContact(evt)
|
||||
{
|
||||
log.info "garageDoorContact, $evt.name: $evt.value"
|
||||
if (evt.value == "open") {
|
||||
schedule("0 * * * * ?", "doorOpenCheck")
|
||||
}
|
||||
else {
|
||||
unschedule("doorOpenCheck")
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
def interiorDoorClosed(evt)
|
||||
{
|
||||
log.info "interiorContact, $evt.name: $evt.value"
|
||||
|
||||
// time during which closing the interior door will shut the garage door, if the app opened it
|
||||
final threshold = 15 * 60 * 1000
|
||||
if (state.appOpenedDoor && now() - state.appOpenedDoor < threshold) {
|
||||
state.appOpenedDoor = 0
|
||||
closeDoor()
|
||||
}
|
||||
else {
|
||||
log.debug "app didn't open door"
|
||||
}
|
||||
}
|
||||
|
||||
def accelerationActive(evt)
|
||||
{
|
||||
log.info "$evt.name: $evt.value"
|
||||
|
||||
if (doorSensor.currentContact == "closed") {
|
||||
log.debug "opening door when car door opened"
|
||||
openDoor()
|
||||
}
|
||||
}
|
||||
|
||||
private openDoor()
|
||||
{
|
||||
if (doorSensor.currentContact == "closed") {
|
||||
log.debug "opening door"
|
||||
doorSwitch.push()
|
||||
}
|
||||
}
|
||||
|
||||
private closeDoor()
|
||||
{
|
||||
if (doorSensor.currentContact == "open") {
|
||||
log.debug "closing door"
|
||||
doorSwitch.push()
|
||||
}
|
||||
}
|
||||
|
||||
private getCar(evt)
|
||||
{
|
||||
cars.find{it.id == evt.deviceId}
|
||||
}
|
||||
Reference in New Issue
Block a user