Source code for aaa_modules.layout_model.actions.turn_on_switch
import time
from aaa_modules.layout_model.neighbor import Neighbor, NeighborType
from aaa_modules.layout_model.switch import Light, Switch
from aaa_modules.layout_model.motion_sensor import MotionSensor
from aaa_modules.layout_model.actions.action import Action
from aaa_modules.layout_model.actions.turn_off_adjacent_zones import TurnOffAdjacentZones
from org.slf4j import Logger, LoggerFactory
logger = LoggerFactory.getLogger("org.eclipse.smarthome.model.script.Rules")
DEBUG = False
[docs]class TurnOnSwitch(Action):
'''
Turns on a switch (fan, dimmer or regular light), after being triggered by
a motion event.
If the switch is a dimmer or light, it is only turned on if:
1. It is evening time, or
2. The illuminance is below a threshold.
A light/dimmer switch won't be turned on if:
1. The light has the flag set to ignore motion event, or
2. The adjacent zone is of type OPEN_SPACE_MASTER with the light on, or
3. The light was jsut turned off, or
4. The neighbor zone has a light switch that shares the same motion sensor,
and that light switch was just recently turned off.
No matter whether the switch is turned on or not (see the condition above),
any adjacent zones of type OPEN_SPACE, and OPEN_SPACE_SLAVE that currently
has the light on, will be sent a command to shut off the light.
'''
DELAY_AFTER_LAST_OFF_TIME_IN_SECONDS = 8
'''
The period of time in seconds (from the last timestamp a switch was
turned off) to ignore the motion sensor event. This takes care of the
scenario when the user manually turns off a light, but that physical
spot is covered by a motion sensor, which immediately turns on the light
again.
'''
[docs] def onAction(self, events, zone, getZoneByIdFn):
isProcessed = False
canTurnOffAdjacentZones = True
lightOnTime = zone.isLightOnTime()
zoneIlluminance = zone.getIlluminanceLevel()
for switch in zone.getDevicesByType(Switch):
if not switch.canBeTriggeredByMotionSensor():
# A special case: if a switch is configured not to be
# triggered by a motion sensor, it means there is already
# another switch sharing that motion sensor. In this case, we
# don't want to turn off the other switch.
canTurnOffAdjacentZones = False
if DEBUG:
logger.info("{}: rejected - can't be triggerred by motion sensor".format(
switch.getItemName()))
continue
# Break if switch was just turned off.
if None != switch.getLastOffTimestampInSeconds():
if (time.time() - switch.getLastOffTimestampInSeconds()) <= \
TurnOnSwitch.DELAY_AFTER_LAST_OFF_TIME_IN_SECONDS:
if DEBUG:
logger.info("{}: rejected - switch was just turned off".format(
switch.getItemName()))
continue
# Break if the switch of a neighbor sharing the motion sensor was
# just turned off.
openSpaceZones = [getZoneByIdFn(n.getZoneId()) \
for n in zone.getNeighbors() if n.isOpenSpace()]
sharedMotionSensorZones = [z for z in openSpaceZones
if zone.shareSensorWith(z, MotionSensor)]
theirSwitches = reduce(lambda a, b : a + b,
[z.getDevicesByType(Switch) for z in sharedMotionSensorZones],
[])
if any(time.time() - s.getLastOffTimestampInSeconds() <= \
TurnOnSwitch.DELAY_AFTER_LAST_OFF_TIME_IN_SECONDS \
for s in theirSwitches):
if DEBUG:
logger.info("{}: rejected - can't be triggerred by motion sensor".format(
switch.getItemName()))
continue
if isinstance(switch, Light):
if lightOnTime or switch.isLowIlluminance(zoneIlluminance):
isProcessed = True
if isProcessed and None != getZoneByIdFn:
masterZones = [getZoneByIdFn(n.getZoneId()) \
for n in zone.getNeighbors() \
if NeighborType.OPEN_SPACE_MASTER == n.getType()]
if any(z.isLightOn() for z in masterZones):
isProcessed = False
if DEBUG:
logger.info("{}: rejected - a master zone's light is on".format(
switch.getItemName()))
if isProcessed:
switch.turnOn(events)
else:
switch.turnOn(events)
isProcessed = True
# Now shut off the light in any shared space zones
if canTurnOffAdjacentZones:
if DEBUG:
logger.info("{}: turning off adjancent zone's light".format(
switch.getItemName()))
TurnOffAdjacentZones().onAction(events, zone, getZoneByIdFn)
return isProcessed