3680 lines
139 KiB
YAML
3680 lines
139 KiB
YAML
blueprint:
|
|
name: 🔥 Advanced Heating Control V5
|
|
author: panhans
|
|
homeassistant:
|
|
min_version: "2024.10.0"
|
|
description: >
|
|
🔥 room based heating / ❄ based on
|
|
|
|
|
|
> 👥 people presence
|
|
|
|
🗓️ multiple schedulers
|
|
|
|
🚶 presence sensor
|
|
|
|
↔️ proximity aka geo fencing
|
|
|
|
|
|
🥶 frost protection
|
|
|
|
😡 adjustable aggressive mode
|
|
|
|
🌤️ activation based on weather, temperature or boolean entities
|
|
|
|
🎛️ granular schedule adjustments
|
|
|
|
🪟 multiple window open detection
|
|
|
|
🎈 party mode
|
|
|
|
🤝 guest mode
|
|
|
|
⚔️ liming protection
|
|
|
|
📍 dynamic valve positioning
|
|
|
|
🧭 thermostat calibration for the most common devices (Tado, Aqara, Popp / Danfoss / Hive, Tuya)
|
|
|
|
⚙️ several tweaks for fixing your thermostat issues
|
|
|
|
🎬 custom action
|
|
|
|
🤫 calm & 💪 reliable
|
|
|
|
|
|
**Ensure that you have installed the uptime integration:**
|
|
|
|
[](https://my.home-assistant.io/redirect/integration/?domain=uptime)
|
|
|
|
|
|
**Version**: 5.4.3
|
|
|
|
**Help & FAQ**: [Advanced Heating Control](https://community.home-assistant.io/t/advanced-heating-control/469873)
|
|
|
|
**Documentation:** [panhans.github.io/HomeAssistant/](https://panhans.github.io/HomeAssistant/)
|
|
|
|
|
|
[](https://ko-fi.com/Q5Q3QEH52)
|
|
source_url: https://github.com/panhans/HomeAssistant/blob/main/blueprints/automation/panhans/advanced_heating_control.yaml
|
|
domain: automation
|
|
input:
|
|
thermostat_section:
|
|
name: Thermostats & Sensors
|
|
icon: mdi:thermostat
|
|
input:
|
|
input_trvs:
|
|
name: 🔥 Thermostats / Climates
|
|
description: >
|
|
`thermostats` `climates`
|
|
|
|
|
|
[Thermostats / Climates](https://www.home-assistant.io/integrations/climate/) to be controlled.
|
|
selector:
|
|
entity:
|
|
filter:
|
|
- domain:
|
|
- climate
|
|
multiple: true
|
|
|
|
input_hvac_mode:
|
|
name: 🎛️ Operation / HVAC Mode
|
|
description: >
|
|
`hvac`
|
|
|
|
|
|
Select the hvac mode for your [thermostats](https://www.home-assistant.io/integrations/climate/). Be sure your selected thermostats support the hvac mode you've chosen.
|
|
AHC will log a warning if there is a miss match. For radiator [thermostats]((https://www.home-assistant.io/integrations/climate/)) the default is mostly *heat*.
|
|
If you own an air conditioner it will support *auto* or *cool*, too.
|
|
default: "heat"
|
|
selector:
|
|
select:
|
|
options:
|
|
- heat
|
|
- cool
|
|
- auto
|
|
- heat_cool
|
|
|
|
input_temperature_sensor:
|
|
name: 🌡️ Room Temperature Sensor
|
|
description: >
|
|
`calibration` `aggressive mode` `optional`
|
|
|
|
|
|
For some features an external temperature sensor is reqired, e.g. calibration.
|
|
|
|
Temperature calibration for your [thermostats](https://www.home-assistant.io/integrations/climate/). The following is supported:
|
|
|
|
* Tado, Aqara, Popp, Danfoss, Hive, Bosch, SONOFF, Tuya
|
|
* generic calibration
|
|
|
|
Note: This is an additional sensor inside your room usually next to your favourite spot. [Thermostats](https://www.home-assistant.io/integrations/climate/) or its integration (e.g. Z2M or ZHA) except Tado should provide a seperate calibration entity.
|
|
default: []
|
|
selector:
|
|
entity:
|
|
filter:
|
|
- domain:
|
|
- sensor
|
|
device_class:
|
|
- temperature
|
|
multiple: false
|
|
|
|
temperature_section:
|
|
name: Temperatures
|
|
icon: mdi:thermometer
|
|
collapsed: true
|
|
input:
|
|
input_temperature_comfort_static:
|
|
name: 🛋️ Static Comfort Temperature
|
|
description: >
|
|
`comfort temperature`
|
|
|
|
|
|
You can set a static comfort temperature here.
|
|
default: 22
|
|
selector:
|
|
number:
|
|
min: 12.0
|
|
max: 86.0
|
|
step: 0.5
|
|
mode: box
|
|
unit_of_measurement: °C / °F
|
|
|
|
input_temperature_eco_static:
|
|
name: 🌱 Static Eco Temperature
|
|
description: >
|
|
`eco temperature`
|
|
|
|
|
|
The temperature that is set when your heating schedule is not active.
|
|
default: 19
|
|
selector:
|
|
number:
|
|
min: 4.0
|
|
max: 75.0
|
|
step: 0.5
|
|
mode: box
|
|
unit_of_measurement: °C / °F
|
|
|
|
input_temperature_comfort:
|
|
name: 🛋️ Comfort Temperature
|
|
description: >
|
|
`comfort temperature` `optional`
|
|
|
|
|
|
To control your comfort temperature via automations or the UI, you can specify an *[input_number](https://www.home-assistant.io/integrations/input_number/)* entity here.
|
|
|
|
|
|
Create your helper [here](https://my.home-assistant.io/redirect/helpers/).
|
|
default: []
|
|
selector:
|
|
entity:
|
|
filter:
|
|
- domain:
|
|
- input_number
|
|
multiple: false
|
|
|
|
input_temperature_eco:
|
|
name: 🌱 Eco Temperature
|
|
description: >
|
|
`eco temperature` `optional`
|
|
|
|
|
|
To control your eco temperature via automations or the UI, you can specify an *[input_number](https://www.home-assistant.io/integrations/input_number/)* entity here.
|
|
|
|
|
|
Create your helper [here](https://my.home-assistant.io/redirect/helpers/).
|
|
default: []
|
|
selector:
|
|
entity:
|
|
filter:
|
|
- domain:
|
|
- input_number
|
|
multiple: false
|
|
|
|
adjustment_section:
|
|
name: Adjustments / Heating Plan
|
|
icon: mdi:sun-clock
|
|
collapsed: true
|
|
input:
|
|
input_adjustments:
|
|
name: 🎛️ Heating Schedule Adjustments
|
|
description: >
|
|
`optional`
|
|
|
|
|
|
Here you can setup some adjustments to your heating schedule.<br/><br/>
|
|
*Note*: Here you can set values for eco or comfort temperature. The switch between those target temperatures is controled by schedules, presence sensors, proximity, ect.
|
|
<br/>
|
|
|
|
<details>
|
|
<summary><code><strong>CLICK HERE:</strong> Modifiers</code></summary>
|
|
<br/>
|
|
|
|
|
|
> 🕔 **time**
|
|
|
|
> Timestamp when the adjustment should kick in. (required)
|
|
|
|
|
|
> 📆 **days**
|
|
|
|
> Select days where this setting shall be enabled.
|
|
|
|
> ['Mon','Tue','Wed','Thu','Fri','Sat','Sun']
|
|
|
|
|
|
> 🗓️ **scheduler**
|
|
|
|
> Only enable this entry if this string is part of the name of your active scheduler.
|
|
|
|
|
|
> 🛋️ **comfort**
|
|
|
|
> Adjust comfort temperature
|
|
|
|
|
|
> 🌱 **eco**
|
|
|
|
> Adjust eco temperature
|
|
|
|
|
|
> 🧭 **calibration**
|
|
|
|
> Toggle calibration
|
|
|
|
> on/off
|
|
|
|
|
|
> 🔘 **mode**
|
|
|
|
> Overwrite the operation mode
|
|
|
|
> comfort/eco/off/auto (auto means no overwrite)
|
|
|
|
|
|
</details>
|
|
|
|
<br/>
|
|
|
|
<details>
|
|
<summary><code><strong>CLICK HERE:</strong> Example</code></summary>
|
|
<br/>
|
|
|
|
|
|
```yaml
|
|
|
|
- time: "08:00"
|
|
comfort: "20"
|
|
calibration: "off"
|
|
- time: "16:00"
|
|
eco: "19"
|
|
calibration: "on"
|
|
- time: "20:00"
|
|
days: ['Sat','Sun']
|
|
scheduler: 'Holidays'
|
|
comfort: "24"
|
|
eco: "17"
|
|
```
|
|
|
|
</details>
|
|
|
|
selector:
|
|
object:
|
|
default: "[]"
|
|
|
|
# modes
|
|
mode_section:
|
|
name: Force Comfort/Eco Mode
|
|
icon: mdi:fire
|
|
collapsed: true
|
|
input:
|
|
input_mode_party:
|
|
name: 🎈 Party mode
|
|
description: >
|
|
`optional`
|
|
|
|
|
|
If on, all settings are ignored and heating takes place. You can define multiple [timers](https://www.home-assistant.io/integrations/timer/) or boolean entities.
|
|
If you put a number at the end of the friendly name like *Party Timer 20* this number will be taken as the desired comfort temperature for this [timer](https://www.home-assistant.io/integrations/timer/).
|
|
|
|
|
|
Create your timer [here](https://my.home-assistant.io/redirect/helpers/).
|
|
default: []
|
|
selector:
|
|
entity:
|
|
filter:
|
|
- domain:
|
|
- input_boolean
|
|
- binary_sensor
|
|
- timer
|
|
multiple: true
|
|
|
|
input_force_max_temperature:
|
|
name: 🥵 Force Max Temperature
|
|
description: >
|
|
`optional`
|
|
|
|
|
|
Set the maximum temperature of all [thermostats](https://www.home-assistant.io/integrations/climate/) regardless of any other settings.
|
|
|
|
|
|
**HINT:** Implemented by developer for maintenance reasons. Create your helper [here](https://my.home-assistant.io/redirect/helpers/).
|
|
default: []
|
|
selector:
|
|
entity:
|
|
filter:
|
|
- domain:
|
|
- input_boolean
|
|
- binary_sensor
|
|
multiple: false
|
|
|
|
input_force_eco_temperature:
|
|
name: 🌱 Force Eco Temperature
|
|
description: >
|
|
`optional`
|
|
|
|
|
|
If enabled *eco* temperature will be forced.
|
|
default: []
|
|
selector:
|
|
entity:
|
|
filter:
|
|
- domain:
|
|
- input_boolean
|
|
- binary_sensor
|
|
multiple: false
|
|
|
|
input_party_legacy_restore:
|
|
name: 🔄 Legacy Restore
|
|
description: >
|
|
`party`
|
|
|
|
|
|
Enable this if the temperatures after airing (closing windows) or party won't restore properly
|
|
default: false
|
|
selector:
|
|
boolean:
|
|
|
|
temperature_tweak_section:
|
|
name: Temperature Tweaks
|
|
icon: mdi:knob
|
|
collapsed: true
|
|
input:
|
|
input_off_instead_of_eco:
|
|
name: 🛑 Off Instead Of Eco
|
|
description: >
|
|
`optional` `temperature tweak`
|
|
|
|
|
|
Turn off your [thermostats](https://www.home-assistant.io/integrations/climate/) instead of lower the target temperature to eco temperature.
|
|
default: false
|
|
selector:
|
|
boolean:
|
|
|
|
input_min_instead_of_off:
|
|
name: ⬇️ Min Instead Of Off
|
|
description: >
|
|
`optional` `temperature tweak`
|
|
|
|
|
|
Lower the temperature instead of turning them *OFF*, e.g. during airing.
|
|
default: false
|
|
selector:
|
|
boolean:
|
|
|
|
input_fahrenheit:
|
|
name: 🇫 Fahrenheit
|
|
description: >
|
|
`optional` `temperature tweak`
|
|
|
|
|
|
Enable this if your unit of measurement is Fahrenheit (untested).
|
|
default: false
|
|
selector:
|
|
boolean:
|
|
|
|
input_reset_temperature:
|
|
name: ↩️ Reset Temperature
|
|
description: >
|
|
`optional` `temperature tweak`
|
|
|
|
|
|
Reset your temperature entities to the values of the static temperatures after [schedule](https://www.home-assistant.io/integrations/schedule/), [proximity](https://www.home-assistant.io/integrations/proximity/), [people](https://www.home-assistant.io/integrations/person/), presence ends.
|
|
The comfort entity is reset when eco takes place and vice versa.
|
|
default: false
|
|
selector:
|
|
boolean:
|
|
|
|
input_off_if_above_room_temperature:
|
|
name: ↕️ Off If Above/Below Room Temperature
|
|
description: >
|
|
|
|
Turns your [climate](https://www.home-assistant.io/integrations/climate/) entity *off* if the target temperature is below(cooling) / above(heating) the room temperature.
|
|
|
|
default: false
|
|
selector:
|
|
boolean:
|
|
|
|
input_off_if_nobody_home:
|
|
name: 🏠🚶➡️ Off If Nobody Home
|
|
description: >
|
|
|
|
Turns your [climate](https://www.home-assistant.io/integrations/climate/) entity *off* if persons are set and nobody is home.
|
|
|
|
default: false
|
|
selector:
|
|
boolean:
|
|
|
|
input_physical_change:
|
|
name: 🧪 Physical Temperature Change / Sync (experimental)
|
|
description: >
|
|
`optional` `temperature tweak`
|
|
|
|
|
|
Enable this if your want to adjust the temperature using your thermostat or thermostat card. Make sure aggressive mode and generic calibration is disabled for this feature. (experimental).
|
|
You also need to set entities for eco and comfort temperature for the moment.
|
|
default: false
|
|
selector:
|
|
boolean:
|
|
|
|
# persons
|
|
person_section:
|
|
name: Persons
|
|
icon: mdi:account-multiple
|
|
collapsed: true
|
|
input:
|
|
input_persons:
|
|
name: 👥 Persons
|
|
description: >
|
|
`person` `optional`
|
|
|
|
|
|
You can specify [persons](https://www.home-assistant.io/integrations/person/) to make your heating plan more dynamic. If you do not use [schedulers](https://www.home-assistant.io/integrations/schedule/) or presence sensors, heating is activated as soon as someone is at home.<br/>
|
|
With [schedulers](https://www.home-assistant.io/integrations/schedule/) or presence sensors, these are only active when someone is at home.
|
|
default: []
|
|
selector:
|
|
entity:
|
|
filter:
|
|
- domain:
|
|
- person
|
|
multiple: true
|
|
|
|
input_people_entering_home_duration:
|
|
name: 🏠 Enter Home Duration
|
|
description: >
|
|
`person`
|
|
|
|
|
|
Duration for which someone must be at home for heating to be activated.
|
|
default:
|
|
hours: 0
|
|
minutes: 0
|
|
seconds: 2
|
|
selector:
|
|
duration:
|
|
|
|
input_people_leaving_home_duration:
|
|
name: 💨 Leaving Home Duration
|
|
description: >
|
|
`person`
|
|
|
|
|
|
Duration for which someone must be out of the house for heating to be deactivated.
|
|
default:
|
|
hours: 0
|
|
minutes: 0
|
|
seconds: 2
|
|
selector:
|
|
duration:
|
|
|
|
input_mode_guest:
|
|
name: 🤝 Guest Mode
|
|
description: >
|
|
`person` `optional`
|
|
|
|
|
|
|
|
If an entity is specified here, it is treated like a [person](https://www.home-assistant.io/integrations/person/). It's usefull when you're leaving your guests alone in your home and you are not using presence detection.
|
|
|
|
* entity defined -> [person](https://www.home-assistant.io/integrations/person/) defined
|
|
* enitity is *on* -> simulates [person](https://www.home-assistant.io/integrations/person/) is home
|
|
* enitity is *off* -> simulates [person](https://www.home-assistant.io/integrations/person/) is away
|
|
default:
|
|
selector:
|
|
entity:
|
|
filter:
|
|
- domain:
|
|
- input_boolean
|
|
- binary_sensor
|
|
- timer
|
|
multiple: false
|
|
|
|
# scheduler
|
|
scheduling_section:
|
|
name: Scheduling
|
|
icon: mdi:clock-outline
|
|
collapsed: true
|
|
input:
|
|
input_schedulers:
|
|
name: ⏲️ Schedules
|
|
description: >
|
|
`schedules` `optional`
|
|
|
|
|
|
A [schedule](https://www.home-assistant.io/integrations/schedule/) specifies when heating to comfort temperature should take place. You can create it in the helper section of Home Assistant.<br/>
|
|
If you have also specified [people](https://www.home-assistant.io/integrations/person/), someone must also be at home for heating. This is the same behaviour with a [proximity](https://www.home-assistant.io/integrations/proximity/) entity.<br/>
|
|
You can create as many [schedules](https://www.home-assistant.io/integrations/schedule/) as you like. Make sure the names are clear.
|
|
default: []
|
|
selector:
|
|
entity:
|
|
filter:
|
|
- domain:
|
|
- schedule
|
|
multiple: true
|
|
|
|
input_scheduler_selector:
|
|
name: ☝🏻 Scheduler Selector
|
|
description: >
|
|
`schedule` `optional`
|
|
|
|
|
|
Define an entity to choose from your schedules. If you use one schedule only you can ignore this. If you use more than one schedule you have multiple possibilities to setup your selection.<br/>
|
|
|
|
<details>
|
|
<summary><code><strong>CLICK HERE:</strong> More information</code></summary>
|
|
<br/>
|
|
|
|
|
|
|
|
* toggle [input_boolean](https://www.home-assistant.io/integrations/input_boolean/) or [binary_sensor](https://www.home-assistant.io/integrations/binary_sensor/): If *off* the first defined [schedule](https://www.home-assistant.io/integrations/schedule/) is enabled. If *on* the second [schedule](https://www.home-assistant.io/integrations/schedule/) is active. More than two [schedules](https://www.home-assistant.io/integrations/schedule/) cannot be selected with binary inputs.
|
|
|
|
* text [input text](https://www.home-assistant.io/integrations/input_text/), drop down [input text](https://www.home-assistant.io/integrations/input_select/) or [sensor](https://www.home-assistant.io/integrations/sensor/):
|
|
* The value has to match the friendly name of the selected [schedule](https://www.home-assistant.io/integrations/schedule/) at least partially. Example: If you provide three [schedules](https://www.home-assistant.io/integrations/schedule/) called *work*, *holiday/sick*, *guest* you can select the holiday [schedule](https://www.home-assistant.io/integrations/schedule/) while setting the selection entity to *sick*, *holiday* or *holiday/sick*. This option is case insensitive.
|
|
* You also can go with numbers: if you want to choose the first [schedule](https://www.home-assistant.io/integrations/schedule/) the selector entity must return the number *1*. For the 2nd number *2* and so on.
|
|
|
|
</details>
|
|
default:
|
|
selector:
|
|
entity:
|
|
filter:
|
|
- domain:
|
|
- input_boolean
|
|
- binary_sensor
|
|
- input_text
|
|
- input_number
|
|
- input_select
|
|
multiple: false
|
|
|
|
# presence
|
|
presence_section:
|
|
name: Presence Detection
|
|
icon: mdi:location-enter
|
|
collapsed: true
|
|
input:
|
|
input_presence_sensor:
|
|
name: 🚶 Presence Sensor / On/Off-Entity
|
|
description: >
|
|
`presence detection` `optional`
|
|
|
|
|
|
If you specify a presence sensor, heating will take place if it detects presence.<br/>
|
|
If you have specified [persons](https://www.home-assistant.io/integrations/person/), at least one must also be at home. You also can select an [input boolean](https://www.home-assistant.io/integrations/input_boolean/) entity to realise a simple On/Off-Logic.
|
|
default:
|
|
selector:
|
|
entity:
|
|
filter:
|
|
- domain:
|
|
- binary_sensor
|
|
- input_boolean
|
|
multiple: false
|
|
|
|
input_scheduler_presence:
|
|
name: ⏲️ Presence Sensor Scheduler
|
|
description: >
|
|
`presence detection` `optional`
|
|
|
|
|
|
The presence [schedule](https://www.home-assistant.io/integrations/schedule/) specifies exactly when the presence sensor should be used during the day.
|
|
default:
|
|
selector:
|
|
entity:
|
|
filter:
|
|
- domain:
|
|
- schedule
|
|
multiple: false
|
|
|
|
input_presence_reaction_on_time:
|
|
name: ⏳ Presence Reaction On Time
|
|
description: >
|
|
`presence detection`
|
|
|
|
|
|
Specify the duration for which the presence sensor must detect any presence so that the comfort temperature is set.
|
|
default:
|
|
hours: 0
|
|
minutes: 5
|
|
seconds: 0
|
|
selector:
|
|
duration:
|
|
|
|
input_presence_reaction_off_time:
|
|
name: ⌛ Presence Reaction Off Time
|
|
description: >
|
|
`presence detection`
|
|
|
|
|
|
Specify the duration for which the presence sensor must not detect any presence so that the eco temperature is set.
|
|
default:
|
|
hours: 0
|
|
minutes: 5
|
|
seconds: 0
|
|
selector:
|
|
duration:
|
|
|
|
# proximity
|
|
proximity_section:
|
|
name: Proximity
|
|
icon: mdi:leak
|
|
collapsed: true
|
|
input:
|
|
input_proximity:
|
|
name: ↔️ Proximity
|
|
description: >
|
|
`proximity` `optional`
|
|
|
|
|
|
You can preheat your rooms with help of home assistant's [proximity integration](https://www.home-assistant.io/integrations/proximity/).<br/>
|
|
Just select your proxmity zone and take your adjustments to distance and duration.<br/>
|
|
If you're in range of your distance and towards to your home heating kicks in.<br/>
|
|
|
|
**Note**: The proximity entity is handles like a person. Comfort heating takes place when coming or beeing home. Combinations with [schedules](https://www.home-assistant.io/integrations/schedule/) are also possible.
|
|
default:
|
|
selector:
|
|
device:
|
|
filter:
|
|
integration: proximity
|
|
multiple: false
|
|
|
|
input_proximity_duration:
|
|
name: ⏰ Proximity Duration
|
|
description: >
|
|
`proximity`
|
|
|
|
|
|
Duration for which someone must be on way home before heating occurs.
|
|
default:
|
|
hours: 0
|
|
minutes: 2
|
|
seconds: 0
|
|
selector:
|
|
duration:
|
|
|
|
input_proximity_distance:
|
|
name: ↔️ Proximity Distance
|
|
description: >
|
|
`proximity`
|
|
|
|
|
|
The distance when [proximity](https://www.home-assistant.io/integrations/proximity/) sensor gets impact for this automation. Hint: Unit depends on the setup of your integration.
|
|
default: 500
|
|
selector:
|
|
number:
|
|
min: 0
|
|
max: 999999999
|
|
step: 1
|
|
mode: box
|
|
|
|
# away mode
|
|
away_section:
|
|
name: Away Mode
|
|
icon: mdi:walk
|
|
collapsed: true
|
|
input:
|
|
# AWAY OFFSET
|
|
input_away_offset:
|
|
name: 🏃 Away Temperature Offset
|
|
description: >
|
|
`scheduler` `persons` `presence` `away mode`
|
|
|
|
|
|
First: This feature only works for [schedule](https://www.home-assistant.io/integrations/schedule/) and/or presence based heating combined with [persons](https://www.home-assistant.io/integrations/person/). You can define an offset for your comfort temperature that will be subtracted (heating) from or added (cooling) to your comfort temperature.
|
|
|
|
If you enable this option for [schedules](https://www.home-assistant.io/integrations/schedule/) the away offset will be substracted from the comfort temperature if your schedule is *on* but nobody is at home.
|
|
For presence detection this is the case if you are at home but no presence is detected. For presence detection you can also ignoring [persons](https://www.home-assistant.io/integrations/person/). So the away temperature is set when no presence is detected but the presence [schedule](https://www.home-assistant.io/integrations/schedule/) is *on*.
|
|
default: 0
|
|
selector:
|
|
number:
|
|
min: 0
|
|
max: 10
|
|
step: 0.5
|
|
mode: slider
|
|
unit_of_measurement: °C / °F
|
|
|
|
input_away_scheduler_mode:
|
|
name: ⏲️ Scheduler Away Mode
|
|
description: >
|
|
`scheduler` `away mode`
|
|
|
|
|
|
Enable/Disable the Away Offset for [schedules](https://www.home-assistant.io/integrations/schedule/) based heating/cooling.
|
|
default: false
|
|
selector:
|
|
boolean:
|
|
|
|
input_away_presence_mode:
|
|
name: 🚶 Presence Away Mode
|
|
description: >
|
|
`presence` `away mode`
|
|
|
|
|
|
Enable/Disable the Away Offset for presence based heating/cooling.
|
|
default: false
|
|
selector:
|
|
boolean:
|
|
|
|
input_away_presence_ignor_people:
|
|
name: 🚶 Ignore People For Presence Away Mode
|
|
description: >
|
|
`presence` `away mode`
|
|
|
|
|
|
If you want to make away happen if your presence [schedule](https://www.home-assistant.io/integrations/schedule/) is on but no motion is detected regardless if somebody is at home enable this option.
|
|
default: false
|
|
selector:
|
|
boolean:
|
|
|
|
# windows
|
|
window_section:
|
|
name: Window & Door Detection
|
|
icon: mdi:door
|
|
collapsed: true
|
|
input:
|
|
input_windows:
|
|
name: 🪟 Windows & Doors
|
|
description: >
|
|
`airing` `optional`
|
|
|
|
|
|
If open during airing your [thermostats](https://www.home-assistant.io/integrations/climate/) will be set to *off* at least to their minimum temperature
|
|
if they don't support hvac mode *OFF* except you set a custom window open temperature.
|
|
default: []
|
|
selector:
|
|
entity:
|
|
filter:
|
|
- domain:
|
|
- binary_sensor
|
|
- sensor
|
|
multiple: true
|
|
|
|
input_windows_reaction_time_open:
|
|
name: ⏳ Window & Door Reaction Time Open
|
|
description: >
|
|
`airing`
|
|
|
|
|
|
Duration for which a window or door must be open for the [thermostats](https://www.home-assistant.io/integrations/climate/) to close.
|
|
default:
|
|
hours: 0
|
|
minutes: 0
|
|
seconds: 30
|
|
selector:
|
|
duration:
|
|
|
|
input_windows_reaction_time_close:
|
|
name: ⌛ Window & Door Reaction Time Close
|
|
description: >
|
|
`airing`
|
|
|
|
|
|
Duration for which a window or door must be closed for the [thermostats](https://www.home-assistant.io/integrations/climate/) to open.
|
|
default:
|
|
hours: 0
|
|
minutes: 0
|
|
seconds: 30
|
|
selector:
|
|
duration:
|
|
|
|
input_window_open_temperature:
|
|
name: Window Open Temperature
|
|
description: >
|
|
`airing`
|
|
|
|
|
|
If 0° your thermostat turns *off* or if not supported it turns to the minimum temperature of your thermostat.
|
|
default: 0
|
|
selector:
|
|
number:
|
|
min: 0
|
|
max: 15
|
|
step: 1
|
|
mode: slider
|
|
unit_of_measurement: °C / °F
|
|
|
|
input_window_legacy_restore:
|
|
name: 🏚️ Legacy Restore
|
|
description: >
|
|
`airing`
|
|
|
|
|
|
Enable this if the temperatures after airing (closing windows) won't restore properly.
|
|
default: false
|
|
selector:
|
|
boolean:
|
|
|
|
# calibration
|
|
calibration_section:
|
|
name: Calibration
|
|
icon: mdi:compass
|
|
description: ""
|
|
collapsed: true
|
|
input:
|
|
input_calibration_timeout:
|
|
name: ⏳ Calibration Timeout
|
|
description: >
|
|
`calibration`
|
|
|
|
|
|
Define a timeout if you want to decrease the amount of calibration calls if temperature changes too much.
|
|
At least the temperature of the external sensor or [thermostat](https://www.home-assistant.io/integrations/climate/) must stay for that duration before calibration gets triggered.
|
|
|
|
**HINT:** A minimum timeout of 2s is recommended.
|
|
default:
|
|
hours: 0
|
|
minutes: 1
|
|
seconds: 0
|
|
selector:
|
|
duration:
|
|
|
|
input_calibration_delta:
|
|
name: ↔️ Calibration Delta
|
|
description: >
|
|
`calibration`
|
|
|
|
|
|
If the difference between the [thermostat](https://www.home-assistant.io/integrations/climate/) temperature and the external sensor temperature is greater or less than the calibration delta the [thermostat](https://www.home-assistant.io/integrations/climate/) calibration will be triggered.<br/>
|
|
The lower the delta the often calibration gets triggered.
|
|
|
|
**HINT:** If your thermostat supports external temperature sensor values it is recommended to set this to a lower value like 0 - 0.2.
|
|
default: 0.5
|
|
selector:
|
|
number:
|
|
min: 0
|
|
max: 5
|
|
step: 0.1
|
|
mode: slider
|
|
unit_of_measurement: °C / °F
|
|
|
|
input_calibration_key_word:
|
|
name: 🗝️ Calibration Entity Key Word
|
|
description: >
|
|
`calibration`
|
|
|
|
|
|
Keyword for finding the calibration entity. This word must be part of the entity id.
|
|
As a rule, the entities with the word *offset*, *calibration* or *external* are marked by the integrations. Just have a look into your device overview, select your thermostat and check the naming of the *entity_ids* for the calibration.
|
|
default: "calibration"
|
|
selector:
|
|
text:
|
|
|
|
input_calibration_step_size:
|
|
name: 🦶 Step Size
|
|
description: >
|
|
`calibration`
|
|
|
|
|
|
Usually the step size is determined automatically. You can overwrite the step size by selecting another option if you know your thermostat handles the calibration not like the entities are exposed.
|
|
default: auto
|
|
selector:
|
|
select:
|
|
mode: dropdown
|
|
options:
|
|
- label: Auto
|
|
value: auto
|
|
- label: "0.1"
|
|
value: "0.1"
|
|
- label: "0.5"
|
|
value: "0.5"
|
|
- label: "Full Values"
|
|
value: "full"
|
|
|
|
input_calibration_generic:
|
|
name: 🧭 Generic Calibration
|
|
description: >
|
|
`generic` `calibration`
|
|
|
|
|
|
Adds the difference between room and [thermostat](https://www.home-assistant.io/integrations/climate/) temperature to the target temperature. This is useful if your thermostat integration doesn't provide a special entity for calibration.
|
|
Keep in mind the set temperatures for your thermostats will differ to the target temperature.
|
|
default: false
|
|
selector:
|
|
boolean:
|
|
|
|
input_generic_calibration_offset:
|
|
name: ↕️ Generic Calibration Offset
|
|
description: >
|
|
`generic` `calibration`
|
|
|
|
|
|
If the temperature difference between the thermostat and the temperature sensor is very high,
|
|
the offset, i.e. the correction temperature, can be limited to this value.
|
|
|
|
<details>
|
|
<summary><code><strong>CLICK HERE:</strong> Example</code></summary>
|
|
Generic Calibration Offset = 5°</br>
|
|
Thermostat Temperature = 28°</br>
|
|
Room Temperature = 18°</br>
|
|
</br>
|
|
Difference = Thermostat Temperature - Room Temperature = 10°</br>
|
|
Difference > Generic Calibration Offset -> Corrected Difference = 5°</br>
|
|
New Target Temperature = Thermostat Temperature + Corrected Difference = 33°
|
|
</details>
|
|
default: 5
|
|
selector:
|
|
number:
|
|
min: 0
|
|
max: 20
|
|
step: 1
|
|
mode: slider
|
|
unit_of_measurement: °C / °F
|
|
|
|
# aggressive mode
|
|
aggressive_mode_section:
|
|
name: Aggressive Mode
|
|
icon: mdi:emoticon-angry
|
|
collapsed: true
|
|
input:
|
|
input_aggressive_mode_range:
|
|
name: 😡 Aggressive Range
|
|
description: >
|
|
`aggressive mode` `tweak`
|
|
|
|
|
|
Activate this option if your [thermostats](https://www.home-assistant.io/integrations/climate/) react slowly or only start to react at a large temperature difference
|
|
between actual and set temperature.
|
|
Define a range when your real target temperature shall be set.
|
|
|
|
<details>
|
|
<summary><code><strong>CLICK HERE:</strong> More information</code></summary>
|
|
<br/>
|
|
|
|
E.g. you target temperature is 20°C and your room temperature is 19.5°C.
|
|
If your range is set to 0.5°C the real target temperature (20°C) will be set when room temperature is between 19.5°C and 20.5°C.
|
|
If the room temperature is above or lower than range, it gets some offset in order to force your [thermostat](https://www.home-assistant.io/integrations/climate/) to react.
|
|
(see Aggressive Mode - Offset)
|
|
|
|
</details>
|
|
|
|
default: 0
|
|
selector:
|
|
number:
|
|
min: 0
|
|
max: 5
|
|
step: 0.1
|
|
mode: slider
|
|
unit_of_measurement: °C / °F
|
|
|
|
input_aggressive_mode_offset:
|
|
name: ↕ Aggressive Offset
|
|
description: >
|
|
`aggressive mode` `tweak`
|
|
|
|
|
|
Here you can define the offset that will be added to your target temperature if the room temperature is not in range of your target temperature.
|
|
If your room temperature is not in the defined range, e.g. 19.5°C - 20.5°C this offset will be added to your target temperature. If range is 0, then offset is always added.
|
|
default: 0
|
|
selector:
|
|
number:
|
|
min: 0
|
|
max: 5
|
|
step: 0.5
|
|
mode: slider
|
|
unit_of_measurement: °C / °F
|
|
|
|
input_aggressive_mode_calibration:
|
|
name: 🌡️ Aggressive Calibration
|
|
description: >
|
|
`aggressive mode` `tweak` `experimental`
|
|
|
|
If you'd setup an temperature sensor and your thermostats allow calibration, you can enable this feature. If enabled the aggressive offset will be add
|
|
to the calibration value and not the target temperature.
|
|
|
|
*Note*: This feature is marked as experimental since not every calibration method could be tested. If you notice any problems simple open an issue or
|
|
post a message in the [AHC-Thread](https://community.home-assistant.io/t/advanced-heating-control/469873).
|
|
Enable this only if native calibration does NOT work when using generic calibration.
|
|
default: false
|
|
selector:
|
|
boolean:
|
|
|
|
# frost protection
|
|
frostprotection_section:
|
|
name: Frost Protection
|
|
icon: mdi:snowflake
|
|
collapsed: true
|
|
input:
|
|
input_frost_protection_temp:
|
|
name: ❄️ Frost Protection Temperature
|
|
description: >
|
|
`frost protection`
|
|
|
|
|
|
You can set the frost protection temperature here.
|
|
default: 5
|
|
selector:
|
|
number:
|
|
min: 5.0
|
|
max: 62.0
|
|
step: 0.5
|
|
mode: box
|
|
unit_of_measurement: °C / °F
|
|
|
|
input_frost_protection_duration:
|
|
name: ❄️ Frost Protection Fallback Duration
|
|
description: >
|
|
`frost protection`
|
|
|
|
|
|
If the defined [persons](https://www.home-assistant.io/integrations/person/) are not at home for a longer period of time or the presence sensor has no longer detected any presence, the frost protection temperature can be lowered after a this duration.
|
|
Note: If set to zero frost protection temperature never will be set.
|
|
default:
|
|
days: 0
|
|
hours: 0
|
|
minutes: 0
|
|
seconds: 0
|
|
selector:
|
|
duration:
|
|
enable_day: true
|
|
|
|
# liming protection
|
|
liming_protection_section:
|
|
name: Liming Protection
|
|
icon: mdi:pipe-valve
|
|
collapsed: true
|
|
input:
|
|
input_liming_protection:
|
|
name: 🎚️ Liming Protection
|
|
description: >
|
|
`liming protection`
|
|
|
|
|
|
Most smart thermostats come with that feature out of the box.
|
|
If your thermostat doesn't support this or you're using the generic thermostat integration this feature is maybe handy for you in order to prevent your valve against limescale.
|
|
The automation will set the thermostat to its max and open the valve for one minute.
|
|
|
|
default: off
|
|
selector:
|
|
boolean:
|
|
|
|
input_liming_protection_day:
|
|
name: 🗓️ Day
|
|
description: >
|
|
`liming protection`
|
|
|
|
|
|
Select the day of the week for the execution.
|
|
default: "Mon"
|
|
selector:
|
|
select:
|
|
options:
|
|
- label: Monday
|
|
value: Mon
|
|
- label: Tuesday
|
|
value: Tue
|
|
- label: Wednesday
|
|
value: Wed
|
|
- label: Thursday
|
|
value: Thu
|
|
- label: Friday
|
|
value: Fri
|
|
- label: Saturday
|
|
value: Sat
|
|
- label: Sunday
|
|
value: Sun
|
|
|
|
input_liming_protection_time:
|
|
name: 🕖 Time
|
|
description: >
|
|
`liming protection`
|
|
|
|
|
|
Select the time for the execution.
|
|
default: "12:00:00"
|
|
selector:
|
|
time:
|
|
|
|
input_liming_protection_duration:
|
|
name: 🕖 Liming Protection Duration
|
|
description: >
|
|
`liming protection`
|
|
|
|
|
|
Duration of liming protection before the thermostat is reset to its initial state.
|
|
default: 1
|
|
selector:
|
|
number:
|
|
min: 1
|
|
max: 30
|
|
step: 1
|
|
mode: slider
|
|
unit_of_measurement: min
|
|
|
|
input_liming_in_winter:
|
|
name: 🌨️ Liming In Winter / Liming If Automation is Disabled
|
|
description: >
|
|
`liming protection`
|
|
|
|
|
|
Enable this if you want liming protection even if the automation is active.
|
|
default: false
|
|
selector:
|
|
boolean:
|
|
|
|
# winter mode
|
|
toggle_section:
|
|
name: "On/Off Automation Options"
|
|
icon: mdi:light-switch
|
|
collapsed: true
|
|
input:
|
|
input_mode_winter:
|
|
name: ⛄ Winter Mode / Automation Toggle
|
|
description: >
|
|
`activation` `optional`
|
|
|
|
|
|
If *on* the automation is active. If *off* your valves will set to *off* and the automation is going to sleep.
|
|
You can set this up with:
|
|
|
|
* [input boolean](https://www.home-assistant.io/integrations/input_boolean/)
|
|
* [binary sensor](https://www.home-assistant.io/integrations/binary_sensor/)
|
|
|
|
|
|
Create your helper [here](https://my.home-assistant.io/redirect/helpers/).
|
|
default:
|
|
selector:
|
|
entity:
|
|
filter:
|
|
- domain:
|
|
- input_boolean
|
|
- binary_sensor
|
|
multiple: false
|
|
|
|
input_invert_winter_mode_value:
|
|
name: 🔄 Invert Winter Mode Value
|
|
description: >
|
|
`activation`
|
|
|
|
|
|
If enabled the the value of the binary winter mode entity will be inverted:
|
|
|
|
* off -> activates the automation
|
|
* on -> disables the automation
|
|
default: off
|
|
selector:
|
|
boolean:
|
|
|
|
input_mode_outside_temperature:
|
|
name: 🌤️ Outside Temperature Sensor
|
|
description: >
|
|
`activation` `optional`
|
|
|
|
|
|
You can control the switching on and off of your thermostats via the outside temperature.
|
|
To do this, select a temperature sensor or a weather entity and adjust the threshold below.
|
|
|
|
* [weather entity](https://www.home-assistant.io/integrations/weather/)
|
|
* [temperature sensor entity](https://www.home-assistant.io/integrations/sensor/)
|
|
|
|
default:
|
|
selector:
|
|
entity:
|
|
filter:
|
|
- domain:
|
|
- weather
|
|
- domain:
|
|
- sensor
|
|
device_class: temperature
|
|
multiple: false
|
|
|
|
input_mode_outside_temperature_threshold:
|
|
name: 🎚️ Outside Temperature Threshold
|
|
description: >
|
|
`activation`
|
|
|
|
|
|
If you'd select a temperature [sensor](https://www.home-assistant.io/integrations/sensor/) or a [weather entity](https://www.home-assistant.io/integrations/weather/)
|
|
for controlling heating you can adjust the temperature threshold here.
|
|
If the outside temperature falls below the threshold value, heating is activated.
|
|
default: 15
|
|
selector:
|
|
number:
|
|
min: 5
|
|
max: 68
|
|
step: 0.5
|
|
mode: box
|
|
unit_of_measurement: °C / °F
|
|
|
|
input_mode_room_temperature:
|
|
name: 🔘 Enable Room Temperature Threshold
|
|
description: >
|
|
`activation` `optional`
|
|
|
|
|
|
If you enable this option the value of the defined room temperature sensor and the value of the outside temperautre must be below / above
|
|
its threshold. That makes sense if you go with an A/C and the room is still heated up but it has already cooled down outside.
|
|
|
|
|
|
**Not recommendend for heating**
|
|
|
|
default: false
|
|
selector:
|
|
boolean:
|
|
|
|
input_mode_room_temperature_threshold:
|
|
name: 🎚️ Room Temperature Threshold
|
|
description: >
|
|
`activation`
|
|
|
|
|
|
Threshold for your room temperature sensor.
|
|
default: 18
|
|
selector:
|
|
number:
|
|
min: 5
|
|
max: 68
|
|
step: 0.5
|
|
mode: box
|
|
unit_of_measurement: °C / °F
|
|
|
|
# valve positioning
|
|
valve_positioning_section:
|
|
name: "Dynamic Valve Positioning"
|
|
icon: mdi:valve
|
|
collapsed: true
|
|
input:
|
|
input_valve_positioning_mode:
|
|
name: 🦶 Valve Positioning Mode
|
|
description: >
|
|
`valve positioning`
|
|
|
|
|
|
If your thermostat supports valve positioning you can enable this here. Everytime the autmation gets triggered the code checks if there is an adjustment needed.
|
|
|
|
|
|
📈 **regular**: means linear. The valve will open close proportional to the difference of the target and room temperature.
|
|
|
|
|
|
😊 **optimistic**: The valve opening is reduced earlier, as it is assumed that the radiator still has enough residual heat to heat the room.
|
|
|
|
|
|
🙁 **pessimistic**: The valve opening is initially left relatively open and only closes rapidly when the target temperature is almost reached.
|
|
default: "off"
|
|
selector:
|
|
select:
|
|
mode: dropdown
|
|
options:
|
|
- label: "off"
|
|
value: "off"
|
|
- label: "regular"
|
|
value: "regular"
|
|
- label: "optimistic"
|
|
value: "optimistic"
|
|
- label: "pessimistic"
|
|
value: "pessimistic"
|
|
|
|
input_fully_open_difference:
|
|
name: ↔️ Positioning Temperature Difference
|
|
description: >
|
|
`valve positioning`
|
|
|
|
|
|
The difference between target and set temperature when dynamic valve positioninig should happen.
|
|
|
|
|
|
<details>
|
|
<summary><code><strong>CLICK HERE:</strong> Example</code></summary>
|
|
<br/>
|
|
|
|
|
|
> Positioning Temperature Difference: 1°
|
|
|
|
> Target Temperature: 21°<br/>
|
|
|
|
|
|
> Positioning takes place in a range between 21° and 20° (21°-1°)
|
|
|
|
|
|
> If the local temperature is 21.5° the valve positioning is calculated and set.
|
|
|
|
> If the local temperture is below this range the valve is fully open.
|
|
|
|
</details>
|
|
|
|
default: 1
|
|
selector:
|
|
number:
|
|
min: 0.5
|
|
max: 20
|
|
step: 0.5
|
|
mode: box
|
|
unit_of_measurement: °C / °F
|
|
|
|
input_valve_positioning_step_size:
|
|
name: 🦶 Valve Positioning Step Size
|
|
description: >
|
|
`valve positioning`
|
|
|
|
|
|
The step size of for opening/closing the valve.
|
|
default: "10"
|
|
selector:
|
|
select:
|
|
mode: dropdown
|
|
options:
|
|
- label: "5%"
|
|
value: "5"
|
|
- label: "10%"
|
|
value: "10"
|
|
- label: "20%"
|
|
value: "20"
|
|
|
|
input_valve_positioning_max_opening:
|
|
name: 🎚️ Max Opening Valve Position
|
|
description: >
|
|
`valve positioning`
|
|
|
|
|
|
The maximal opening of the valve. Some thermostats have a maximum valve position of 80-90%. You can adjust the value here. *Force Max Temperature* still sets the value to 100%.
|
|
default: 100
|
|
selector:
|
|
number:
|
|
min: 1
|
|
max: 100
|
|
step: 1
|
|
mode: slider
|
|
unit_of_measurement: "%"
|
|
|
|
input_valve_positioning_timeout:
|
|
name: ⏱️ Valve Positioning Timeout
|
|
description: >
|
|
`valve positioning`
|
|
|
|
|
|
Timeout that must lie between two adjustments before the second is executed.
|
|
default:
|
|
hours: 0
|
|
minutes: 20
|
|
seconds: 0
|
|
selector:
|
|
duration:
|
|
|
|
input_valve_opening_keyword:
|
|
name: 🗝️ Positioning Entity Keyword
|
|
description: >
|
|
`valve positioning`
|
|
|
|
|
|
The key word for selecting the opening entity of your thermostats.
|
|
default: "valve_opening_degree"
|
|
selector:
|
|
text:
|
|
|
|
# tweaks
|
|
tweak_section:
|
|
name: Custom Settings
|
|
icon: mdi:cog-box
|
|
collapsed: true
|
|
input:
|
|
input_action_call_delay:
|
|
name: ⚙️ Action Call Delay
|
|
description: >
|
|
`tweak`
|
|
|
|
|
|
Some [thermostats](https://www.home-assistant.io/integrations/climate/) have problems with setting mode and temperature. You can try to increase the
|
|
delay between the action calls. This could fix your problems.
|
|
default:
|
|
hours: 0
|
|
minutes: 0
|
|
seconds: 2
|
|
selector:
|
|
duration:
|
|
|
|
input_startup_delay:
|
|
name: ⏲ Startup Delay
|
|
description: >
|
|
`tweak`
|
|
|
|
|
|
If your AHC automation is triggered directly after a Home Assistant restart, but the required integrations have not yet been loaded or certain sensors
|
|
have not yet been initialized, you can set an automation delay here.
|
|
|
|
*Note:* Make sure that you have set up the uptime integration for this purpose.
|
|
default:
|
|
hours: 0
|
|
minutes: 0
|
|
seconds: 0
|
|
selector:
|
|
duration:
|
|
|
|
# custom action
|
|
input_custom_action:
|
|
name: 🎬 Custom Action
|
|
description: >
|
|
`optional`
|
|
|
|
|
|
This custom action gets executed with every temperature / mode change except calibration. If you want to control other devices just check states before doing an action call.
|
|
Use the variable *is_heating* in your conditions. *True* means heating is active.
|
|
default:
|
|
selector:
|
|
action:
|
|
|
|
input_custom_condition:
|
|
name: ☑️ Temperature Change Custom Condition
|
|
description: >
|
|
`optional`
|
|
|
|
|
|
Define a custom condition that prevents / allows temperature changes to your thermostats. This has no impact to the rest of logic like calibration.
|
|
default:
|
|
selector:
|
|
condition:
|
|
|
|
input_custom_condition_calibration:
|
|
name: ☑️ Calibration Custom Condition
|
|
description: >
|
|
`optional`
|
|
|
|
|
|
Define a custom condition that prevents / allows calibration.
|
|
default:
|
|
selector:
|
|
condition:
|
|
|
|
input_log_level:
|
|
name: ✍️ Log Level
|
|
description: ""
|
|
default: debug
|
|
selector:
|
|
select:
|
|
mode: dropdown
|
|
options:
|
|
- info
|
|
- warning
|
|
- error
|
|
- debug
|
|
|
|
trigger_variables:
|
|
# thermostats / sensors
|
|
input_trvs: !input input_trvs
|
|
input_temperature_sensor: !input input_temperature_sensor
|
|
is_temperature_sensor_defined: "{{ input_temperature_sensor != [] }}"
|
|
|
|
# people
|
|
input_persons: !input input_persons
|
|
input_mode_guest: !input input_mode_guest
|
|
input_people_entering_home_duration: !input input_people_entering_home_duration
|
|
input_people_leaving_home_duration: !input input_people_leaving_home_duration
|
|
|
|
input_person_count: "{{ input_persons | count }}"
|
|
is_person_defined: "{{ input_person_count > 0 }}"
|
|
is_guest_mode_defined: "{{ input_mode_guest != none }}"
|
|
|
|
# scheduler
|
|
input_schedulers: !input input_schedulers
|
|
input_scheduler_selector: !input input_scheduler_selector
|
|
input_scheduler_presence: !input input_scheduler_presence
|
|
is_scheduler_presence_defined: "{{ input_scheduler_presence != none }}"
|
|
|
|
# temperatures
|
|
input_temperature_comfort: !input input_temperature_comfort
|
|
input_temperature_eco: !input input_temperature_eco
|
|
input_hvac_mode: !input input_hvac_mode
|
|
factor: "{{ iif(input_hvac_mode == 'cool', -1, 1) | int }}"
|
|
is_heat_only_if_below_real_temp: !input input_off_if_above_room_temperature
|
|
|
|
# on/ff
|
|
input_mode_winter: !input input_mode_winter
|
|
input_mode_outside_temperature: !input input_mode_outside_temperature
|
|
input_mode_outside_temperature_threshold: !input input_mode_outside_temperature_threshold
|
|
input_mode_room_temperature_threshold: !input input_mode_room_temperature_threshold
|
|
input_mode_room_temperature: !input input_mode_room_temperature
|
|
input_invert_winter_mode_value: !input input_invert_winter_mode_value
|
|
|
|
# party / force max
|
|
input_mode_party: !input input_mode_party
|
|
|
|
# adjustments / heating plan
|
|
input_adjustments: !input input_adjustments
|
|
|
|
# calibration
|
|
input_calibration_timeout: !input input_calibration_timeout
|
|
|
|
# windows
|
|
input_windows: !input input_windows
|
|
|
|
#presence
|
|
input_presence_sensor: !input input_presence_sensor
|
|
is_presence_sensor_defined: "{{ input_presence_sensor != none }}"
|
|
input_presence_reaction_on_time: !input input_presence_reaction_on_time
|
|
input_presence_reaction_off_time: !input input_presence_reaction_off_time
|
|
|
|
# proximity
|
|
input_proximity: !input input_proximity
|
|
input_proximity_duration: !input input_proximity_duration
|
|
input_proximity_distance: !input input_proximity_distance
|
|
|
|
# frost protection
|
|
input_frost_protection_duration: !input input_frost_protection_duration
|
|
|
|
# liming protection
|
|
input_liming_protection: !input input_liming_protection
|
|
input_liming_protection_day: !input input_liming_protection_day
|
|
input_liming_protection_time: !input input_liming_protection_time
|
|
input_liming_in_winter: !input input_liming_in_winter
|
|
input_liming_protection_duration: !input input_liming_protection_duration
|
|
|
|
trigger:
|
|
# system
|
|
- trigger: homeassistant
|
|
event: start
|
|
id: temperature_change_hastart
|
|
|
|
- trigger: event
|
|
event_type: automation_reloaded
|
|
id: temperature_change_reload
|
|
|
|
- trigger: event
|
|
event_type: ahc_delay_event
|
|
id: delayed_call_temperature_change
|
|
event_data:
|
|
automation: "{{ this.entity_id }}"
|
|
|
|
- trigger: event
|
|
event_type: ahc_positioning_event
|
|
id: positioning_event
|
|
event_data:
|
|
automation: "{{ this.entity_id }}"
|
|
|
|
# thermostats become available
|
|
- trigger: state
|
|
entity_id: !input input_trvs
|
|
from:
|
|
- unknown
|
|
- unavailable
|
|
for:
|
|
seconds: 5
|
|
id: temperature_change_available
|
|
|
|
# physical change
|
|
- trigger: state
|
|
entity_id: !input input_trvs
|
|
attribute: temperature
|
|
for:
|
|
seconds: 5
|
|
id: temperature_change_valve_target
|
|
|
|
# eco/comfort change
|
|
- trigger: state
|
|
entity_id: !input input_temperature_eco
|
|
for: !input input_action_call_delay
|
|
id: temperature_change_eco
|
|
|
|
- trigger: state
|
|
entity_id: !input input_temperature_comfort
|
|
for: !input input_action_call_delay
|
|
id: temperature_change_comfort
|
|
|
|
# persons
|
|
- trigger: template
|
|
value_template: >
|
|
{{ input_persons | expand
|
|
| selectattr('state', 'eq', 'home')
|
|
| list
|
|
| count > 0
|
|
|
|
or (is_guest_mode_defined and states(input_mode_guest) in ['on','active'] ) }}
|
|
id: temperature_change_person_on
|
|
for: !input input_people_entering_home_duration
|
|
|
|
- trigger: template
|
|
value_template: >
|
|
{{ input_persons | expand
|
|
| selectattr('state', 'eq', 'home')
|
|
| list
|
|
| count == 0
|
|
|
|
and (not is_guest_mode_defined or (is_guest_mode_defined and states(input_mode_guest) not in ['on','active'])) }}
|
|
id: temperature_change_person_off
|
|
for: !input input_people_leaving_home_duration
|
|
|
|
# scheduler
|
|
- trigger: template
|
|
id: temperature_change_scheduler_on
|
|
value_template: >
|
|
{% set selected_scheduler = none %}
|
|
{% set schedules_count = input_schedulers | count %}
|
|
|
|
{% if schedules_count == 0 %}
|
|
{% set selected_scheduler = none %}
|
|
{% elif schedules_count == 1 or input_scheduler_selector == none %}
|
|
{% set selected_scheduler = input_schedulers | first %}
|
|
{% elif schedules_count > 1 %}
|
|
{% set selector_value = states(input_scheduler_selector) %}
|
|
|
|
{% if is_number(selector_value) %}
|
|
{% set selector_value = iif(selector_value | int > schedules_count, schedules_count, selector_value) %}
|
|
{% set selector_value = iif(selector_value | int <= 0, 1, selector_value) %}
|
|
{% set selected_scheduler = input_schedulers[selector_value | int - 1] %}
|
|
{% elif selector_value in ['on','off'] %}
|
|
{% set selected_scheduler = iif(selector_value == 'off', input_schedulers[0], input_schedulers[1]) %}
|
|
{% else %}
|
|
{% set selected_scheduler = input_schedulers | expand | selectattr('attributes.friendly_name', 'eq', selector_value) | map(attribute='entity_id') | first | default(none) %}
|
|
{% if (selected_scheduler == none) %}
|
|
{% set selected_scheduler = input_schedulers | expand | selectattr('attributes.friendly_name', 'search', '(?i)' + selector_value) | map(attribute='entity_id') | first | default(none) %}
|
|
{% endif %}
|
|
{% endif %}
|
|
{% endif %}
|
|
|
|
{% if selected_scheduler == none %}
|
|
{{ false }}
|
|
{% else %}
|
|
{{ is_state(selected_scheduler, 'on') }}
|
|
{% endif %}
|
|
|
|
- trigger: template
|
|
id: temperature_change_scheduler_off
|
|
value_template: >
|
|
{% set selected_scheduler = none %}
|
|
{% set schedules_count = input_schedulers | count %}
|
|
|
|
{% if schedules_count == 0 %}
|
|
{% set selected_scheduler = none %}
|
|
{% elif schedules_count == 1 or input_scheduler_selector == none %}
|
|
{% set selected_scheduler = input_schedulers | first %}
|
|
{% elif schedules_count > 1 %}
|
|
{% set selector_value = states(input_scheduler_selector) %}
|
|
|
|
{% if is_number(selector_value) %}
|
|
{% set selector_value = iif(selector_value | int > schedules_count, schedules_count, selector_value) %}
|
|
{% set selector_value = iif(selector_value | int <= 0, 1, selector_value) %}
|
|
{% set selected_scheduler = input_schedulers[selector_value | int - 1] %}
|
|
{% elif selector_value in ['on','off'] %}
|
|
{% set selected_scheduler = iif(selector_value == 'off', input_schedulers[0], input_schedulers[1]) %}
|
|
{% else %}
|
|
{% set selected_scheduler = input_schedulers | expand | selectattr('attributes.friendly_name', 'eq', selector_value) | map(attribute='entity_id') | first | default(none) %}
|
|
{% if (selected_scheduler == none) %}
|
|
{% set selected_scheduler = input_schedulers | expand | selectattr('attributes.friendly_name', 'search', '(?i)' + selector_value) | map(attribute='entity_id') | first | default(none) %}
|
|
{% endif %}
|
|
{% endif %}
|
|
{% endif %}
|
|
|
|
{% if selected_scheduler == none %}
|
|
{{ false }}
|
|
{% else %}
|
|
{{ is_state(selected_scheduler, 'off') }}
|
|
{% endif %}
|
|
|
|
# presence sensor
|
|
- trigger: template
|
|
id: temperature_change_presence_on
|
|
value_template: "{{ input_presence_sensor != none and is_state(input_presence_sensor, 'on') }}"
|
|
for: !input input_presence_reaction_on_time
|
|
|
|
- trigger: template
|
|
id: temperature_change_presence_off
|
|
value_template: "{{ input_presence_sensor != none and is_state(input_presence_sensor, 'off') }}"
|
|
for: !input input_presence_reaction_off_time
|
|
|
|
# presence scheduler
|
|
- trigger: template
|
|
id: temperature_change_presence_scheduler_on
|
|
value_template: "{{ input_scheduler_presence != none and is_state(input_scheduler_presence, 'on') }}"
|
|
for: !input input_action_call_delay
|
|
|
|
- trigger: template
|
|
id: temperature_change_presence_scheduler_off
|
|
value_template: "{{ input_scheduler_presence != none and is_state(input_scheduler_presence, 'off') }}"
|
|
for: !input input_action_call_delay
|
|
|
|
# proximity
|
|
- trigger: template
|
|
id: temperature_change_person_proximity_on
|
|
value_template: >
|
|
{% set proximity_entities = device_entities(input_proximity) %}
|
|
|
|
{% set is_arrived = proximity_entities
|
|
| select('is_state','arrived')
|
|
| expand
|
|
| selectattr('attributes.device_class', 'eq', 'enum')
|
|
| list | count > 0 %}
|
|
|
|
{% set entities_towards = proximity_entities
|
|
| expand
|
|
| selectattr('attributes.device_class', 'eq', 'enum')
|
|
| map(attribute='entity_id') | select('is_state','towards')
|
|
| map('regex_replace','_(?=[^_]*$)(.*)', '')
|
|
| list %}
|
|
|
|
{% set distances = proximity_entities
|
|
| expand
|
|
| selectattr('attributes.device_class', 'eq', 'distance')
|
|
| map(attribute='state')
|
|
| reject('eq', 'unknown')
|
|
| map('int')
|
|
| select('<=', input_proximity_distance | int)
|
|
| map('string')
|
|
| list %}
|
|
|
|
{% set entities_distances = proximity_entities
|
|
| expand
|
|
| selectattr('attributes.device_class', 'eq', 'distance')
|
|
| selectattr('state', 'in', distances)
|
|
| map(attribute='entity_id')
|
|
| map('regex_replace','_(?=[^_]*$)(.*)', '')
|
|
| list %}
|
|
|
|
{% set entites_towards_and_in_distance = entities_towards | select('in', entities_distances) | list | count > 0 %}
|
|
|
|
{{ entites_towards_and_in_distance or is_arrived }}
|
|
for: !input input_proximity_duration
|
|
|
|
- trigger: template
|
|
id: temperature_change_person_proximity_off
|
|
value_template: >
|
|
{% set proximity_entities = device_entities(input_proximity) %}
|
|
{% set is_arrived = proximity_entities
|
|
| select('is_state','arrived')
|
|
| expand
|
|
| selectattr('attributes.device_class', 'eq', 'enum')
|
|
| list | count > 0 %}
|
|
|
|
{% set entities_towards = proximity_entities
|
|
| expand
|
|
| selectattr('attributes.device_class', 'eq', 'enum')
|
|
| map(attribute='entity_id') | select('is_state','towards')
|
|
| map('regex_replace','_(?=[^_]*$)(.*)', '')
|
|
| list %}
|
|
|
|
{% set distances = proximity_entities
|
|
| expand
|
|
| selectattr('attributes.device_class', 'eq', 'distance')
|
|
| map(attribute='state')
|
|
| reject('eq', 'unknown')
|
|
| map('int')
|
|
| select('<=', input_proximity_distance | int)
|
|
| map('string')
|
|
| list %}
|
|
|
|
{% set entities_distances = proximity_entities
|
|
| expand
|
|
| selectattr('attributes.device_class', 'eq', 'distance')
|
|
| selectattr('state', 'in', distances)
|
|
| map(attribute='entity_id')
|
|
| map('regex_replace','_(?=[^_]*$)(.*)', '')
|
|
| list %}
|
|
|
|
{% set entites_towards_and_in_distance = entities_towards | select('in', entities_distances) | list | count > 0 %}
|
|
|
|
{{ entites_towards_and_in_distance == false and is_arrived == false }}
|
|
for: !input input_proximity_duration
|
|
|
|
# window
|
|
- trigger: template
|
|
value_template: "{{ expand(input_windows) | selectattr('state', 'in', ['on','open','tilted']) | list | count > 0 }}"
|
|
for: !input input_windows_reaction_time_open
|
|
id: temperature_change_window_on
|
|
|
|
- trigger: template
|
|
value_template: "{{ expand(input_windows) | selectattr('state', 'in', ['on','open','tilted']) | list | count == 0 }}"
|
|
for: !input input_windows_reaction_time_close
|
|
id: temperature_change_window_off
|
|
|
|
# on/off winter mode
|
|
- trigger: template
|
|
id: temperature_change_winter_mode_on
|
|
value_template: >
|
|
{% if input_mode_winter != none %}
|
|
{% set activation_state = iif(input_invert_winter_mode_value, 'off', 'on') %}
|
|
{{ is_state(input_mode_winter, activation_state) }}
|
|
{% endif %}
|
|
for: !input input_action_call_delay
|
|
|
|
- trigger: template
|
|
id: temperature_change_winter_mode_off
|
|
value_template: >
|
|
{% if input_mode_winter != none %}
|
|
{% set activation_state = iif(input_invert_winter_mode_value, 'off', 'on') %}
|
|
{{ not is_state(input_mode_winter, activation_state) }}
|
|
{% endif %}
|
|
for: !input input_action_call_delay
|
|
|
|
# on/off temperature
|
|
- trigger: template
|
|
id: temperature_change_outside_on
|
|
value_template: >
|
|
{% if input_mode_outside_temperature == none %}
|
|
{{ false }}
|
|
{% else %}
|
|
{% set outside_state = false %}
|
|
{% set use_room_temp = input_mode_room_temperature and is_temperature_sensor_defined %}
|
|
{% set room_state = iif(use_room_temp, false, true) %}
|
|
|
|
{% set state = states(input_mode_outside_temperature) %}
|
|
{% set state = iif(is_number(state) == true, state, state_attr(input_mode_outside_temperature,'temperature'))%}
|
|
|
|
{% if is_number(state) %}
|
|
{% set outside_state = (state | float - input_mode_outside_temperature_threshold | float) * factor < 0 %}
|
|
{% endif %}
|
|
|
|
{% if use_room_temp %}
|
|
{% set state = states(input_temperature_sensor) %}
|
|
|
|
{% if is_number(state) %}
|
|
{% set room_state = (state | float - input_mode_room_temperature_threshold | float) * factor < 0 %}
|
|
{% endif %}
|
|
{% endif %}
|
|
|
|
{{ room_state and outside_state }}
|
|
{% endif %}
|
|
for: !input input_action_call_delay
|
|
|
|
- trigger: template
|
|
id: temperature_change_outside_off
|
|
value_template: >
|
|
{% if input_mode_outside_temperature == none %}
|
|
{{ false }}
|
|
{% else %}
|
|
{% set outside_state = false %}
|
|
{% set use_room_temp = input_mode_room_temperature and is_temperature_sensor_defined %}
|
|
{% set room_state = iif(use_room_temp, false, true) %}
|
|
|
|
{% set state = states(input_mode_outside_temperature) %}
|
|
{% set state = iif(is_number(state) == true, state, state_attr(input_mode_outside_temperature,'temperature'))%}
|
|
|
|
{% if is_number(state) %}
|
|
{% set outside_state = (state | float - input_mode_outside_temperature_threshold | float) * factor < 0 %}
|
|
{% endif %}
|
|
|
|
{% if use_room_temp %}
|
|
{% set state = states(input_temperature_sensor) %}
|
|
|
|
{% if is_number(state) %}
|
|
{% set room_state = (state | float - input_mode_room_temperature_threshold | float) * factor < 0 %}
|
|
{% endif %}
|
|
{% endif %}
|
|
|
|
{{ not (room_state and outside_state) }}
|
|
{% endif %}
|
|
for: !input input_action_call_delay
|
|
|
|
# force max temp
|
|
- trigger: state
|
|
id: temperature_change_force_max_temperature_on
|
|
entity_id: !input input_force_max_temperature
|
|
for: !input input_action_call_delay
|
|
|
|
# force eco temp
|
|
- trigger: state
|
|
id: temperature_change_force_eco_temperature_ds
|
|
entity_id: !input input_force_eco_temperature
|
|
for: !input input_action_call_delay
|
|
|
|
# party
|
|
- trigger: template
|
|
id: temperature_change_party_on
|
|
value_template: "{{ input_mode_party | expand | selectattr('state', 'in', ['active','on']) | list | count > 0 }}"
|
|
for: !input input_action_call_delay
|
|
|
|
- trigger: template
|
|
value_template: "{{ input_mode_party | expand | selectattr('state', 'in', ['active','on']) | list | count == 0 }}"
|
|
id: temperature_change_party_off
|
|
for: !input input_action_call_delay
|
|
|
|
# aggressive mode / heating above/below temp
|
|
- trigger: state
|
|
id: calibration_aggressive_mode_above_temp_thermostat_current_temp_change
|
|
entity_id: !input input_trvs
|
|
attribute: current_temperature
|
|
for: !input input_calibration_timeout
|
|
|
|
- trigger: state
|
|
id: calibration_aggressive_mode_thermostat_temp_change
|
|
entity_id: !input input_trvs
|
|
attribute: temperature
|
|
for:
|
|
seconds: 30
|
|
|
|
- trigger: state
|
|
id: aggressive_mode_above_temp_sensor_change
|
|
entity_id: !input input_temperature_sensor
|
|
for:
|
|
seconds: 30
|
|
|
|
# calibration trigger
|
|
- trigger: state
|
|
id: calibration_sensor_change
|
|
entity_id: !input input_temperature_sensor
|
|
for: !input input_calibration_timeout
|
|
|
|
- trigger: state
|
|
id: calibration_popp_change
|
|
entity_id: !input input_temperature_sensor
|
|
for:
|
|
seconds: 2
|
|
|
|
- trigger: template
|
|
id: calibration_popp_ping
|
|
value_template: >
|
|
{% set has_valves_danfoss = input_trvs | select('is_device_attr', 'manufacturer', 'Danfoss') | list %}
|
|
{% set has_valves_popp = input_trvs | select('is_device_attr', 'manufacturer', 'Popp') | list %}
|
|
{% set valves_hive = input_trvs | select('is_device_attr', 'manufacturer', 'Hive') | list %}
|
|
{% set valves_bosch = input_trvs | select('is_device_attr', 'manufacturer', 'Bosch') | list %}
|
|
|
|
{% set has_valves = (has_valves_danfoss + has_valves_popp + valves_hive + valves_bosch) | count > 0 %}
|
|
|
|
{{ has_valves and is_temperature_sensor_defined and now().strftime('%M') | int % 10 == 0 }}
|
|
|
|
# heating adjustments
|
|
- trigger: template
|
|
id: temperature_change_heating_adjustment
|
|
value_template: >
|
|
{% set timestamp = now() %}
|
|
|
|
{% set current_day = timestamp.strftime('%a') %}
|
|
{% set current_time = timestamp.strftime('%H:%M') %}
|
|
|
|
{% set plan = input_adjustments | rejectattr('time', 'undefined')
|
|
| selectattr('time','eq', current_time | string)
|
|
| sort(attribute='time', reverse = true)
|
|
| list %}
|
|
|
|
{{ plan | count > 0 and now() < now().replace(second=2) }}
|
|
|
|
# liming protection
|
|
- trigger: template
|
|
value_template: >
|
|
{% if not input_liming_protection%}
|
|
{{ false }}
|
|
{% else %}
|
|
{% set enable_liming = true %}
|
|
{% if input_mode_winter != none %}
|
|
{% set enable_liming = is_state(input_mode_winter,'on') or input_liming_in_winter %}
|
|
{% endif %}
|
|
|
|
{% set current_timestamp = now() %}
|
|
|
|
{% set is_liming_day = input_liming_protection_day == as_datetime(current_timestamp).strftime('%a') %}
|
|
|
|
{% set start_hour = input_liming_protection_time.split(':')[0] | int %}
|
|
{% set start_minute = input_liming_protection_time.split(':')[1] | int %}
|
|
|
|
{% set today_start = as_datetime(current_timestamp).replace(second=0,microsecond=0,hour=start_hour,minute=start_minute) %}
|
|
{% set today_end = as_datetime(current_timestamp).replace(second=0,microsecond=0,hour=start_hour,minute=start_minute) + timedelta(minutes=input_liming_protection_duration | int) %}
|
|
|
|
{% set is_liming_time = as_datetime(current_timestamp) >= today_start and as_datetime(current_timestamp) <= today_end %}
|
|
|
|
{{ enable_liming and is_liming_day and is_liming_time }}
|
|
{% endif %}
|
|
id: temperature_change_liming_protection_on
|
|
|
|
- trigger: template
|
|
value_template: >
|
|
{% if not input_liming_protection%}
|
|
{{ false }}
|
|
{% else %}
|
|
{% set enable_liming = true %}
|
|
{% if input_mode_winter != none %}
|
|
{% set enable_liming = is_state(input_mode_winter,'on') or input_liming_in_winter %}
|
|
{% endif %}
|
|
|
|
{% set current_timestamp = now() %}
|
|
|
|
{% set current_timestamp = now() %}
|
|
|
|
{% set is_liming_day = input_liming_protection_day == as_datetime(current_timestamp).strftime('%a') %}
|
|
|
|
{% set start_hour = input_liming_protection_time.split(':')[0] | int %}
|
|
{% set start_minute = input_liming_protection_time.split(':')[1] | int %}
|
|
|
|
{% set today_start = as_datetime(current_timestamp).replace(second=0,microsecond=0,hour=start_hour,minute=start_minute) %}
|
|
{% set today_end = as_datetime(current_timestamp).replace(second=0,microsecond=0,hour=start_hour,minute=start_minute) + timedelta(minutes=input_liming_protection_duration | int) %}
|
|
|
|
{% set is_liming_time = as_datetime(current_timestamp) >= today_start and as_datetime(current_timestamp) <= today_end %}
|
|
|
|
{{ not (enable_liming and is_liming_day and is_liming_time) }}
|
|
{% endif %}
|
|
id: temperature_change_liming_protection_off
|
|
|
|
# frost protection
|
|
- trigger: template
|
|
id: temperature_change_frost_protection_on
|
|
for: !input input_frost_protection_duration
|
|
value_template: >
|
|
{% set now_ts = now() %}
|
|
{% set frost_protection_timestamp = as_datetime(now_ts) - timedelta(**input_frost_protection_duration) %}
|
|
{% if frost_protection_timestamp == now_ts %}
|
|
{{ false }}
|
|
{% else %}
|
|
|
|
{% set relevant_entities = [input_presence_sensor] + [input_mode_guest] + input_persons %}
|
|
{% set relevant_entities_count = relevant_entities | reject('eq',none) | list | count %}
|
|
|
|
{% if relevant_entities_count > 0 %}
|
|
{% set presence_count = [input_presence_sensor]
|
|
| reject('eq',none)
|
|
| reject('is_state','on')
|
|
| list
|
|
| count %}
|
|
|
|
{% set guest_mode_count = [input_mode_guest]
|
|
| reject('eq',none)
|
|
| reject('is_state','on')
|
|
| list
|
|
| count %}
|
|
|
|
{% set person_count = input_persons
|
|
| reject('is_state','home')
|
|
| list
|
|
| count %}
|
|
|
|
{{ presence_count + guest_mode_count + person_count == relevant_entities_count }}
|
|
{% else %}
|
|
{{ false }}
|
|
{% endif %}
|
|
{% endif %}
|
|
|
|
variables:
|
|
#####################################################################################
|
|
###################################### INPUTS #######################################
|
|
#####################################################################################
|
|
|
|
# thermostats / sensors
|
|
input_trvs: !input input_trvs
|
|
input_hvac_mode: !input input_hvac_mode
|
|
input_temperature_sensor: !input input_temperature_sensor
|
|
|
|
# temperatures
|
|
input_temperature_comfort: !input input_temperature_comfort
|
|
input_temperature_comfort_entity: "{{ iif(input_temperature_comfort == [], none, input_temperature_comfort) }}"
|
|
input_temperature_comfort_static: !input input_temperature_comfort_static
|
|
input_temperature_eco: !input input_temperature_eco
|
|
input_temperature_eco_entity: "{{ iif(input_temperature_eco == [], none, input_temperature_eco) }}"
|
|
input_temperature_eco_static: !input input_temperature_eco_static
|
|
|
|
#frost protection
|
|
input_frost_protection_temp: !input input_frost_protection_temp
|
|
input_frost_protection_duration: !input input_frost_protection_duration
|
|
|
|
#liming protection
|
|
input_liming_protection: !input input_liming_protection
|
|
input_liming_protection_day: !input input_liming_protection_day
|
|
input_liming_protection_time: !input input_liming_protection_time
|
|
input_liming_in_winter: !input input_liming_in_winter
|
|
input_liming_protection_duration: !input input_liming_protection_duration
|
|
|
|
# heating scheduler
|
|
input_schedulers: !input input_schedulers
|
|
input_scheduler_selector: !input input_scheduler_selector
|
|
|
|
# presence
|
|
input_presence_sensor: !input input_presence_sensor
|
|
input_scheduler_presence: !input input_scheduler_presence
|
|
input_presence_reaction_off_time: !input input_presence_reaction_off_time
|
|
input_presence_reaction_on_time: !input input_presence_reaction_on_time
|
|
|
|
# window detection
|
|
input_windows: !input input_windows
|
|
input_windows_reaction_time_open: !input input_windows_reaction_time_open
|
|
input_windows_reaction_time_close: !input input_windows_reaction_time_close
|
|
input_window_open_temperature: !input input_window_open_temperature
|
|
input_party_legacy_restore: !input input_party_legacy_restore
|
|
input_window_legacy_restore: !input input_window_legacy_restore
|
|
is_legacy_restore: "{{ input_party_legacy_restore or input_window_legacy_restore }}"
|
|
|
|
# wintermode / on/off
|
|
input_mode_winter: !input input_mode_winter
|
|
input_invert_winter_mode_value: !input input_invert_winter_mode_value
|
|
input_mode_outside_temperature: !input input_mode_outside_temperature
|
|
input_mode_outside_temperature_threshold: !input input_mode_outside_temperature_threshold
|
|
input_mode_room_temperature: !input input_mode_room_temperature
|
|
input_mode_room_temperature_threshold: !input input_mode_room_temperature_threshold
|
|
|
|
# proximity
|
|
input_proximity: !input input_proximity
|
|
|
|
# people
|
|
input_persons: !input input_persons
|
|
input_mode_guest: !input input_mode_guest
|
|
input_people_entering_home_duration: !input input_people_entering_home_duration
|
|
input_people_leaving_home_duration: !input input_people_leaving_home_duration
|
|
|
|
# force comfort
|
|
input_mode_party: !input input_mode_party
|
|
input_force_max_temperature: !input input_force_max_temperature
|
|
input_force_eco_temperature: !input input_force_eco_temperature
|
|
|
|
# calibration
|
|
input_calibration_delta: !input input_calibration_delta
|
|
input_calibration_generic: !input input_calibration_generic
|
|
input_calibration_step_size: !input input_calibration_step_size
|
|
input_calibration_key_word: !input input_calibration_key_word
|
|
input_generic_calibration_offset: !input input_generic_calibration_offset
|
|
|
|
# Aggressive Mode
|
|
input_aggressive_mode_offset: !input input_aggressive_mode_offset
|
|
input_aggressive_mode_range: !input input_aggressive_mode_range
|
|
input_aggressive_mode_calibration: !input input_aggressive_mode_calibration
|
|
|
|
# away mode
|
|
input_away_offset: !input input_away_offset
|
|
is_scheduler_away_mode: !input input_away_scheduler_mode
|
|
is_presence_away_mode: !input input_away_presence_mode
|
|
presence_ignor_people: !input input_away_presence_ignor_people
|
|
|
|
# heating adjustments
|
|
input_adjustments: !input input_adjustments
|
|
|
|
# temperature tweaks
|
|
is_reset_temperature: !input input_reset_temperature
|
|
is_off_instead_min: !input input_off_instead_of_eco
|
|
is_not_off_but_min: !input input_min_instead_of_off
|
|
is_fahrenheit: !input input_fahrenheit
|
|
is_heat_only_if_below_real_temp: !input input_off_if_above_room_temperature
|
|
is_physical_change_enabled: !input input_physical_change
|
|
is_off_if_nobody_home: !input input_off_if_nobody_home
|
|
|
|
# custom tweaks
|
|
input_action_call_delay: !input input_action_call_delay
|
|
input_custom_action: !input input_custom_action
|
|
input_startup_delay: !input input_startup_delay
|
|
|
|
# valve positioning
|
|
input_fully_open_difference: !input input_fully_open_difference
|
|
input_valve_opening_keyword: !input input_valve_opening_keyword
|
|
input_valve_positioning_step_size: !input input_valve_positioning_step_size
|
|
input_valve_positioning_mode: !input input_valve_positioning_mode
|
|
input_valve_positioning_timeout: !input input_valve_positioning_timeout
|
|
input_valve_positioning_max_opening: !input input_valve_positioning_max_opening
|
|
|
|
#####################################################################################
|
|
#################################### EVALUATION #####################################
|
|
#####################################################################################
|
|
|
|
# global
|
|
is_temperature_sensor_defined: "{{ input_temperature_sensor != [] }}"
|
|
invalid_states: >
|
|
{{ ['unknown', 'unavailable'] }}
|
|
|
|
value_temperature_sensor: >
|
|
{% if is_temperature_sensor_defined %}
|
|
{{ states(input_temperature_sensor) }}
|
|
{% else %}
|
|
{{ 'unknown' }}
|
|
{% endif %}
|
|
|
|
valid_temperature_sensor: >
|
|
{{ value_temperature_sensor not in invalid_states }}
|
|
|
|
factor: "{{ iif(input_hvac_mode == 'cool', -1, 1) | int }}"
|
|
current_time_stamp: "{{ now() }}"
|
|
is_metric: "{{ not is_temperature_sensor_defined or (is_temperature_sensor_defined and state_attr(input_temperature_sensor,VAR_UNIT_OF_MEASUREMENT) == '°C') }}"
|
|
|
|
# uptime
|
|
up_time_sensor: "{{ integration_entities('uptime') | first | default(none) }}"
|
|
is_uptime_defined: "{{ up_time_sensor != none }}"
|
|
uptime: >
|
|
{% if is_uptime_defined %}
|
|
{{ states(up_time_sensor) | as_datetime }}
|
|
{% else %}
|
|
{{ current_time_stamp | as_datetime }}
|
|
{% endif %}
|
|
|
|
#startup delay
|
|
startup_delay: >
|
|
{% set start_delay_seconds = timedelta(**input_startup_delay).total_seconds() %}
|
|
|
|
{% if not is_uptime_defined or start_delay_seconds == 0 %}
|
|
{{ none }}
|
|
{% else %}
|
|
{% set difference = (current_time_stamp | as_datetime - uptime | as_datetime).total_seconds() %}
|
|
{% set real_start_delay = (start_delay_seconds - difference) %}
|
|
|
|
{{ iif(real_start_delay > 0, real_start_delay, none) }}
|
|
{% endif %}
|
|
|
|
# on/off
|
|
state_outside_temp: >
|
|
{% if input_mode_outside_temperature == none %}
|
|
{{ none }}
|
|
{% else %}
|
|
{% set outside_state = false %}
|
|
{% set use_room_temp = input_mode_room_temperature and valid_temperature_sensor %}
|
|
{% set room_state = iif(use_room_temp, false, true) %}
|
|
|
|
{% set state = states(input_mode_outside_temperature) %}
|
|
{% set state = iif(is_number(state) == true, state, state_attr(input_mode_outside_temperature,'temperature'))%}
|
|
|
|
{% if is_number(state) %}
|
|
{% set outside_state = (state | float - input_mode_outside_temperature_threshold | float) * factor < 0 %}
|
|
{% endif %}
|
|
|
|
{% if use_room_temp %}
|
|
{% set state = states(input_temperature_sensor) %}
|
|
|
|
{% if is_number(state) %}
|
|
{% set room_state = (state | float - input_mode_room_temperature_threshold | float) * factor < 0 %}
|
|
{% endif %}
|
|
{% endif %}
|
|
|
|
{{ room_state and outside_state }}
|
|
{% endif %}
|
|
|
|
state_ahc: >
|
|
{% set result = true %}
|
|
{% if input_mode_winter != none %}
|
|
{% set activation_state = iif(input_invert_winter_mode_value, 'off', 'on') %}
|
|
{% set result = is_state(input_mode_winter, activation_state) %}
|
|
{% endif %}
|
|
|
|
{{ iif(state_outside_temp == none, result, result and state_outside_temp) }}
|
|
|
|
# proximity
|
|
is_proximity_defined: "{{ input_proximity != none }}"
|
|
|
|
state_proximity_arrived: >
|
|
{% set proximity_entities = device_entities(input_proximity) %}
|
|
{% set is_arrived = proximity_entities
|
|
| select('is_state','arrived')
|
|
| expand
|
|
| selectattr('attributes.device_class', 'eq', 'enum')
|
|
| list | count > 0 %}
|
|
{{ is_arrived }}
|
|
|
|
state_proximity_way_home: >
|
|
{% set proximity_entities = device_entities(input_proximity) %}
|
|
|
|
{% set earliest_timestamp = current_time_stamp | as_datetime - timedelta(**input_proximity_duration) %}
|
|
{% set uptime_duration = as_datetime(uptime) + timedelta(**input_proximity_duration) %}
|
|
|
|
{% if uptime_duration > earliest_timestamp %}
|
|
{% set earliest_timestamp = uptime_duration%}
|
|
{% endif %}
|
|
|
|
{% set entities_towards = proximity_entities
|
|
| expand
|
|
| selectattr('attributes.device_class', 'eq', 'enum')
|
|
| selectattr('last_changed', '<=', earliest_timestamp)
|
|
| map(attribute='entity_id') | select('is_state','towards')
|
|
| map('regex_replace','_(?=[^_]*$)(.*)', '')
|
|
| list %}
|
|
|
|
{% set distances = proximity_entities
|
|
| expand
|
|
| selectattr('attributes.device_class', 'eq', 'distance')
|
|
| map(attribute='state')
|
|
| reject('eq', 'unknown')
|
|
| map('int')
|
|
| select('<=', input_proximity_distance | int)
|
|
| map('string')
|
|
| list %}
|
|
|
|
{% set entities_distances = proximity_entities
|
|
| expand
|
|
| selectattr('attributes.device_class', 'eq', 'distance')
|
|
| selectattr('state', 'in', distances)
|
|
| map(attribute='entity_id')
|
|
| map('regex_replace','_(?=[^_]*$)(.*)', '')
|
|
| list %}
|
|
|
|
{% set towards_and_in_distance = entities_towards | select('in', entities_distances) | list | count > 0 %}
|
|
|
|
{{ towards_and_in_distance }}
|
|
|
|
# persons
|
|
is_person_defined: "{{ input_persons | count > 0 or input_mode_guest != none }}"
|
|
is_guest_mode: "{{ input_mode_guest != none and is_state(input_mode_guest, 'on') }}"
|
|
is_anybody_home: >
|
|
{% if is_guest_mode %}
|
|
{{ true }}
|
|
{% elif not is_person_defined %}
|
|
{{ false }}
|
|
{% else %}
|
|
{% set on_time_delta = current_time_stamp | as_datetime - timedelta(**input_people_entering_home_duration) %}
|
|
{% set off_time_delta = current_time_stamp | as_datetime - timedelta(**input_people_leaving_home_duration) %}
|
|
|
|
{% set uptime_on = as_datetime(uptime) + timedelta(**input_people_entering_home_duration) %}
|
|
{% set uptime_off = as_datetime(uptime) + timedelta(**input_people_leaving_home_duration) %}
|
|
|
|
{% set result = false %}
|
|
|
|
{% if uptime_on > on_time_delta or uptime_off > off_time_delta %}
|
|
{{ input_persons | expand
|
|
| selectattr('state', 'eq', 'home')
|
|
| list
|
|
| count > 0 }}
|
|
{% else %}
|
|
{% set persons_home = state_attr('zone.home','persons') | select('in', input_persons) | list %}
|
|
|
|
{% set somebody_is_home = persons_home | expand
|
|
| selectattr('last_changed', '<=', on_time_delta)
|
|
| list
|
|
| count > 0 %}
|
|
|
|
{% set somebody_is_leaving = persons_home | count == 0 and ['zone.home'] | expand | map(attribute='last_changed') | first | default(off_time_delta) > off_time_delta %}
|
|
|
|
{{ somebody_is_home or somebody_is_leaving }}
|
|
{% endif %}
|
|
{% endif %}
|
|
|
|
is_anybody_home_or_proximity: "{{ is_anybody_home or state_proximity_way_home or state_proximity_arrived}}"
|
|
|
|
# schedules
|
|
active_scheduler: >
|
|
{% set selected_scheduler = none %}
|
|
{% set schedules_count = input_schedulers | count %}
|
|
|
|
{% if schedules_count == 0 %}
|
|
{% set selected_scheduler = none %}
|
|
{% elif schedules_count == 1 or input_scheduler_selector == none %}
|
|
{% set selected_scheduler = input_schedulers | first %}
|
|
{% elif schedules_count > 1 %}
|
|
{% set selector_value = states(input_scheduler_selector) %}
|
|
|
|
{% if is_number(selector_value) %}
|
|
{% set selector_value = iif(selector_value | int > schedules_count, schedules_count, selector_value) %}
|
|
{% set selector_value = iif(selector_value | int <= 0, 1, selector_value) %}
|
|
{% set selected_scheduler = input_schedulers[selector_value | int - 1] %}
|
|
{% elif selector_value in ['on','off'] %}
|
|
{% set selected_scheduler = iif(selector_value == 'off', input_schedulers[0], input_schedulers[1]) %}
|
|
{% else %}
|
|
{% set selected_scheduler = input_schedulers | expand | selectattr('attributes.friendly_name', 'eq', selector_value) | map(attribute='entity_id') | first | default(none) %}
|
|
{% if (selected_scheduler == none) %}
|
|
{% set selected_scheduler = input_schedulers | expand | selectattr('attributes.friendly_name', 'search', '(?i)' + selector_value) | map(attribute='entity_id') | first | default(none) %}
|
|
{% endif %}
|
|
{% endif %}
|
|
{% endif %}
|
|
|
|
{{ selected_scheduler }}
|
|
|
|
is_scheduler_defined: "{{ active_scheduler != none }}"
|
|
|
|
state_scheduler: "{{ active_scheduler != none and is_state(active_scheduler,'on') }}"
|
|
|
|
# presence
|
|
is_presence_sensor_defined: "{{ input_presence_sensor != none }}"
|
|
is_presence_scheduler_defined: "{{ input_scheduler_presence != none }}"
|
|
state_presence_scheduler: "{{ is_presence_scheduler_defined and is_state(input_scheduler_presence, 'on') }}"
|
|
|
|
state_presence_sensor: >
|
|
{% if not is_presence_sensor_defined %}
|
|
{{ false }}
|
|
{% else %}
|
|
{% set last_changed = [input_presence_sensor] | expand | map(attribute='last_changed') | first %}
|
|
{% set sensor_state = is_state(input_presence_sensor, 'on') %}
|
|
{% set reaction_time = iif(sensor_state, input_presence_reaction_on_time, input_presence_reaction_off_time) %}
|
|
{% set min_timestamp = last_changed + timedelta(**reaction_time) %}
|
|
{% set current_ts = current_time_stamp | as_datetime%}
|
|
|
|
{% if is_uptime_defined and as_datetime(uptime) + timedelta(**reaction_time) > current_ts - timedelta(**reaction_time) %}
|
|
{{ sensor_state }}
|
|
{% else %}
|
|
{% set is_limit = min_timestamp <= current_ts %}
|
|
|
|
{{ (sensor_state == true and is_limit) or (sensor_state == false and not is_limit) }}
|
|
{% endif %}
|
|
{% endif %}
|
|
|
|
state_presence: >
|
|
{{ iif(is_presence_scheduler_defined, state_presence_scheduler and state_presence_sensor, state_presence_sensor) }}
|
|
|
|
# force max temperature
|
|
is_force_max_temperature: "{{ input_force_max_temperature != [] and is_state(input_force_max_temperature, 'on') }}"
|
|
is_force_eco_temperature: "{{ input_force_eco_temperature != [] and is_state(input_force_eco_temperature, 'on') }}"
|
|
|
|
# party
|
|
active_party_entity: "{{ input_mode_party | expand | selectattr('state', 'in', ['active','on']) | map(attribute='entity_id') | first | default(none) }}"
|
|
state_party: "{{ active_party_entity != none }}"
|
|
party_temp: >
|
|
{% set pos_party_temp = none %}
|
|
{% if state_party == true %}
|
|
{% set name = state_attr(active_party_entity,'friendly_name') %}
|
|
{% set pos_temp = name.split(' ') | last %}
|
|
{% if is_number(pos_temp) %}
|
|
{% set pos_party_temp = pos_temp | float %}
|
|
{% endif %}
|
|
{% endif %}
|
|
{{ pos_party_temp }}
|
|
|
|
# away
|
|
is_away: >
|
|
{% if is_person_defined and not is_anybody_home_or_proximity %}
|
|
{{ (is_scheduler_away_mode and state_scheduler) or (is_presence_away_mode and state_presence_scheduler and not state_presence) }}
|
|
{% elif presence_ignor_people and is_presence_away_mode %}
|
|
{{ state_presence_scheduler and not state_presence }}
|
|
{% elif is_presence_away_mode and is_person_defined and is_anybody_home_or_proximity and not presence_ignor_people %}
|
|
{{ not state_presence }}
|
|
{% else %}
|
|
{{ false }}
|
|
{% endif %}
|
|
|
|
# windows & doors
|
|
state_window: >
|
|
{% set current_ts = current_time_stamp | as_datetime %}
|
|
{% set on_time_delta = current_ts - timedelta(**input_windows_reaction_time_open) %}
|
|
{% set off_time_delta = current_ts - timedelta(**input_windows_reaction_time_close) %}
|
|
|
|
{% set has_open_windows = input_windows
|
|
| expand
|
|
| selectattr('state', 'in', ['on','open','tilted'])
|
|
| selectattr('last_changed', '<=', on_time_delta)
|
|
| list
|
|
| count > 0 %}
|
|
|
|
{% set closed_but_not_in_duration = input_windows
|
|
| expand
|
|
| selectattr('state', 'in', ['off','closed'])
|
|
| selectattr('last_changed', '>=', off_time_delta)
|
|
| list
|
|
| count > 0 %}
|
|
|
|
{{ has_open_windows or closed_but_not_in_duration }}
|
|
|
|
# aggressive mode
|
|
is_aggressive_mode: "{{ input_aggressive_mode_offset > 0 }}"
|
|
is_aggressive_mode_calibration: "{{ is_aggressive_mode and input_aggressive_mode_calibration and valid_temperature_sensor }}"
|
|
|
|
# frost protection
|
|
is_frost_protection: >
|
|
{% set frost_protection_timestamp = as_datetime(current_time_stamp) - timedelta(**input_frost_protection_duration) %}
|
|
{% if frost_protection_timestamp == as_datetime(current_time_stamp) %}
|
|
{{ false }}
|
|
{% else %}
|
|
{% set relevant_entities = [input_presence_sensor] + [input_mode_guest] + input_persons %}
|
|
{% set relevant_entities_count = relevant_entities | reject('eq',none) | list | count %}
|
|
|
|
{% if relevant_entities_count > 0 %}
|
|
|
|
{% set presence_count = [input_presence_sensor]
|
|
| reject('eq',none)
|
|
| reject('is_state','on')
|
|
| expand
|
|
| selectattr('last_changed', '<=', frost_protection_timestamp)
|
|
| list | count %}
|
|
|
|
{% set persons_count = input_persons
|
|
| reject('eq',none)
|
|
| reject('is_state','home')
|
|
| expand
|
|
| selectattr('last_changed', '<=', frost_protection_timestamp)
|
|
| list | count %}
|
|
|
|
{% set guest_mode_count = [input_mode_guest]
|
|
| reject('eq',none)
|
|
| reject('is_state','on')
|
|
| expand
|
|
| selectattr('last_changed', '<=', frost_protection_timestamp)
|
|
| list | count %}
|
|
|
|
{{ presence_count + guest_mode_count + persons_count == relevant_entities_count }}
|
|
{% else %}
|
|
{{ false }}
|
|
{% endif %}
|
|
{% endif %}
|
|
|
|
# liming protection
|
|
is_liming_protection: >
|
|
{% if not input_liming_protection%}
|
|
{{ false }}
|
|
{% else %}
|
|
{% set enable_liming = true %}
|
|
{% if input_mode_winter != none %}
|
|
{% set enable_liming = is_state(input_mode_winter,'on') or input_liming_in_winter %}
|
|
{% endif %}
|
|
|
|
{% set current_timestamp = now() %}
|
|
|
|
{% set is_liming_day = input_liming_protection_day == as_datetime(current_timestamp).strftime('%a') %}
|
|
|
|
{% set start_hour = input_liming_protection_time.split(':')[0] | int %}
|
|
{% set start_minute = input_liming_protection_time.split(':')[1] | int %}
|
|
|
|
{% set today_start = as_datetime(current_timestamp).replace(second=0,microsecond=0,hour=start_hour,minute=start_minute) %}
|
|
{% set today_end = as_datetime(current_timestamp).replace(second=0,microsecond=0,hour=start_hour,minute=start_minute) + timedelta(minutes=input_liming_protection_duration | int) %}
|
|
|
|
{% set is_liming_time = as_datetime(current_timestamp) >= today_start and as_datetime(current_timestamp) <= today_end %}
|
|
|
|
{{ enable_liming and is_liming_day and is_liming_time }}
|
|
{% endif %}
|
|
|
|
# thermostat groups
|
|
valves: >
|
|
{{ input_trvs | expand
|
|
| selectattr('attributes.hvac_modes','search','(?i)'+input_hvac_mode)
|
|
| map(attribute='entity_id')
|
|
| list }}
|
|
|
|
valves_unsupported: >
|
|
{{ input_trvs | reject('in',valves) | list }}
|
|
|
|
valves_off_mode: >
|
|
{{ valves | expand | selectattr('attributes.hvac_modes','search','(?i)off')
|
|
| map(attribute='entity_id')
|
|
| list }}
|
|
|
|
valves_without_off_mode: >
|
|
{{ valves | reject('in',valves_off_mode) | list }}
|
|
|
|
# tado
|
|
valves_tado: "{{ valves | select('is_device_attr', 'manufacturer', 'Tado') | list }}"
|
|
|
|
# valves external thermometer support
|
|
valves_external: >
|
|
{% set result = namespace(r=[]) %}
|
|
{% for valve in valves %}
|
|
{% set select = device_entities(device_id(valve))
|
|
| expand
|
|
| selectattr('domain','in','select')
|
|
| selectattr('attributes.options', 'contains', 'external')
|
|
| map(attribute='entity_id') | list | first | default(none) %}
|
|
{% if select != none %}
|
|
{% set result.r = result.r + [valve] %}
|
|
{% endif %}
|
|
{% endfor %}
|
|
{{ result.r }}
|
|
|
|
# danfoss / popp / hive / Bosch
|
|
valves_danfoss: "{{ valves | select('is_device_attr', 'manufacturer', 'Danfoss') | list }}"
|
|
valves_popp: "{{ valves | select('is_device_attr', 'manufacturer', 'Popp') | list }}"
|
|
valves_hive: "{{ valves | select('is_device_attr', 'manufacturer', 'Hive') | list }}"
|
|
valves_bosch: "{{ valves | select('is_device_attr', 'manufacturer', 'Bosch') | list }}"
|
|
valves_dph: "{{ valves_danfoss + valves_popp + valves_hive + valves_bosch }}"
|
|
|
|
valves_calibration_common: "{{ valves | reject('in', valves_tado + valves_dph + valves_external) | list }}"
|
|
|
|
# global
|
|
last_comfort_entity_change: "{{ [input_temperature_comfort_entity] | expand | map(attribute='last_changed') | list | first | default(none) }}"
|
|
last_eco_entity_change: "{{ [input_temperature_eco_entity] | expand | map(attribute='last_changed') | list | first | default(none) }}"
|
|
|
|
#####################################################################################
|
|
################################## ADJUSTMENTS ######################################
|
|
#####################################################################################
|
|
|
|
latest_entry_today: >
|
|
{% set scheduler_name = none %}
|
|
{% if active_scheduler != none %}
|
|
{% set scheduler_name = state_attr(active_scheduler,'friendly_name') %}
|
|
{% endif %}
|
|
|
|
{% set current_ts = current_time_stamp | as_datetime %}
|
|
|
|
{% set current_day = current_ts.strftime('%a') %}
|
|
{% set current_time = current_ts.strftime('%H:%M') %}
|
|
|
|
{% set plan = input_adjustments | rejectattr('time', 'undefined')
|
|
| selectattr('time','<=', current_time| string)
|
|
| list %}
|
|
|
|
{% set selected_entries_days_and_schedule = plan | rejectattr('days','==',Undefined) | selectattr('days','search',current_day)
|
|
| rejectattr('scheduler','==',Undefined) | selectattr('scheduler','in',scheduler_name)
|
|
| list %}
|
|
|
|
{% set selected_entries_days = plan | rejectattr('days','==',Undefined) | selectattr('days','search',current_day)
|
|
| selectattr('scheduler','in',[Undefined])
|
|
| list %}
|
|
|
|
{% set selected_entries_schedule = plan | rejectattr('scheduler','==',Undefined) | selectattr('scheduler','in',scheduler_name)
|
|
| selectattr('days','in',[Undefined])
|
|
| list %}
|
|
|
|
{% set selected_entries_time_only = plan | selectattr('days','in',[Undefined])
|
|
| selectattr('scheduler','in',[Undefined])
|
|
| list %}
|
|
|
|
{% set selected_entries = selected_entries_days_and_schedule + selected_entries_days + selected_entries_schedule + selected_entries_time_only %}
|
|
|
|
{% if selected_entries | count > 0%}
|
|
{{ selected_entries | sort(attribute='time', reverse = true) | first }}
|
|
{% else %}
|
|
{{ none }}
|
|
{% endif %}
|
|
|
|
latest_entry_day_before: >
|
|
{% set timestamp = as_datetime(current_time_stamp).replace(hour=23,minute=59) + timedelta(days=-1) %}
|
|
|
|
{% set scheduler_name = none %}
|
|
{% if active_scheduler != none %}
|
|
{% set scheduler_name = state_attr(active_scheduler,'friendly_name') %}
|
|
{% endif %}
|
|
|
|
{% set current_day = timestamp.strftime('%a') %}
|
|
{% set current_time = timestamp.strftime('%H:%M') %}
|
|
|
|
{% set plan = input_adjustments | rejectattr('time', 'undefined')
|
|
| selectattr('time','<=', current_time| string)
|
|
| list %}
|
|
|
|
{% set selected_entries_days_and_schedule = plan | rejectattr('days','==',Undefined) | selectattr('days','search',current_day)
|
|
| rejectattr('scheduler','==',Undefined) | selectattr('scheduler','in',scheduler_name)
|
|
| list %}
|
|
|
|
{% set selected_entries_days = plan | rejectattr('days','==',Undefined) | selectattr('days','search',current_day)
|
|
| selectattr('scheduler','in',[Undefined])
|
|
| list %}
|
|
|
|
{% set selected_entries_schedule = plan | rejectattr('scheduler','==',Undefined) | selectattr('scheduler','in',scheduler_name)
|
|
| selectattr('days','in',[Undefined])
|
|
| list %}
|
|
|
|
{% set selected_entries_time_only = plan | selectattr('days','in',[Undefined])
|
|
| selectattr('scheduler','in',[Undefined])
|
|
| list %}
|
|
|
|
{% set selected_entries = selected_entries_days_and_schedule + selected_entries_days + selected_entries_schedule + selected_entries_time_only %}
|
|
|
|
{% if selected_entries | count > 0%}
|
|
{{ selected_entries | sort(attribute='time', reverse = true) | first }}
|
|
{% else %}
|
|
{{ none }}
|
|
{% endif %}
|
|
|
|
entry: "{{ iif(latest_entry_today != none, latest_entry_today, latest_entry_day_before) }}"
|
|
|
|
entry_time: >
|
|
{% if entry != none %}
|
|
{% set entry_hour = entry['time'].split(':')[0] | int %}
|
|
{% set entry_minute = entry['time'].split(':')[1] | int %}
|
|
{{ as_datetime(current_time_stamp).replace(hour=entry_hour, minute=entry_minute, second=0, microsecond=0) + timedelta(days=iif(latest_entry_today == none,-1,0)) }}
|
|
{% endif %}
|
|
|
|
entry_comfort_temp: >
|
|
{% if entry != none and 'comfort' in entry.keys() and (last_comfort_entity_change == none or as_datetime(entry_time) > as_datetime(last_comfort_entity_change)) %}
|
|
{% set entry_temp = entry['comfort']%}
|
|
{% if is_number(entry_temp) %}
|
|
{{ entry_temp }}
|
|
{% elif states[entry_temp] != none %}
|
|
{{ states(entry_temp) }}
|
|
{% endif %}
|
|
{% else %}
|
|
{{ none }}
|
|
{% endif %}
|
|
|
|
entry_eco_temp: >
|
|
{% if entry != none and 'eco' in entry.keys() and (last_eco_entity_change == none or as_datetime(entry_time) > as_datetime(last_eco_entity_change)) %}
|
|
{% set entry_temp = entry['eco']%}
|
|
{% if is_number(entry_temp) %}
|
|
{{ entry_temp }}
|
|
{% elif states[entry_temp] != none %}
|
|
{{ states(entry_temp) }}
|
|
{% endif %}
|
|
{% else %}
|
|
{{ none }}
|
|
{% endif %}
|
|
|
|
entry_calibration: >
|
|
{% if entry != none and 'calibration' in entry.keys() %}
|
|
{{ entry['calibration'] == 'on' }}
|
|
{% else %}
|
|
{{ true }}
|
|
{% endif %}
|
|
|
|
entry_mode: >
|
|
{% if entry != none and 'mode' in entry.keys() %}
|
|
{{ entry['mode'] }}
|
|
{% else %}
|
|
{{ 'auto' }}
|
|
{% endif %}
|
|
|
|
|
|
#####################################################################################
|
|
############################### TRIGGER EVALUATION ##################################
|
|
#####################################################################################
|
|
|
|
trigger_id_defined: "{{ trigger.id is defined }}"
|
|
|
|
# calibration
|
|
is_calibration_trigger: >
|
|
{% if valves_dph | count > 0 and trigger_id_defined and trigger.id in ['calibration_popp_ping','calibration_popp_change'] %}
|
|
{{ true }}
|
|
{% elif is_aggressive_mode_calibration and trigger_id_defined and 'aggressive_mode' in trigger.id %}
|
|
{{ true }}
|
|
{% else %}
|
|
{{ trigger_id_defined and 'calibration' in trigger.id and not trigger.id == 'calibration_aggressive_mode_thermostat_temp_change' }}
|
|
{% endif %}
|
|
|
|
# changes
|
|
is_generic_calibration_trigger: "{{ is_calibration_trigger and input_calibration_generic }}"
|
|
is_generic_calibration: "{{ is_generic_calibration_trigger and entry_calibration and valid_temperature_sensor }}"
|
|
|
|
is_aggressive_mode_trigger: "{{ is_aggressive_mode and trigger_id_defined and 'aggressive_mode' in trigger.id }}"
|
|
|
|
is_change_trigger: >
|
|
{{ trigger_id_defined and
|
|
'temperature_change' in trigger.id and
|
|
('presence' in trigger.id or
|
|
'scheduler' in trigger.id or
|
|
'proximity' in trigger.id or
|
|
'person' in trigger.id or
|
|
'_ds' in trigger.id)
|
|
and not trigger.id == 'temperature_change_valve_target' }}
|
|
|
|
set_max_temperature: "{{ is_force_max_temperature or is_liming_protection }}"
|
|
|
|
is_pysical_change: >
|
|
{{ trigger_id_defined
|
|
and trigger.id == 'temperature_change_valve_target'
|
|
and is_physical_change_enabled
|
|
and not state_window
|
|
and not set_max_temperature
|
|
and not is_away }}
|
|
|
|
is_adjustment_trigger:
|
|
"{{ trigger_id_defined and trigger.id == 'temperature_change_heating_adjustment' and
|
|
(entry_comfort_temp != none or entry_eco_temp != none) }}"
|
|
|
|
is_reset: >
|
|
{{ (is_reset_temperature and is_change_trigger) or
|
|
is_pysical_change or is_adjustment_trigger }}
|
|
|
|
is_changes_trigger: >
|
|
{% if state_window %}
|
|
{% if trigger_id_defined and 'temperature_change_window_on' in trigger.id %}
|
|
{{ true }}
|
|
{% elif trigger_id_defined and 'temperature_change_window_off' not in trigger.id %}
|
|
{{ false }}
|
|
{% endif %}
|
|
{% elif trigger.platform == none %}
|
|
{{ true }}
|
|
{% elif trigger_id_defined and trigger.id == 'temperature_change_valve_target' %}
|
|
{{ false }}
|
|
{% elif is_heat_only_if_below_real_temp and trigger_id_defined and 'above_temp' in trigger.id %}
|
|
{{ true }}
|
|
{% elif is_aggressive_mode_calibration and is_aggressive_mode_trigger %}
|
|
{{ false }}
|
|
{% elif is_aggressive_mode_trigger %}
|
|
{{ true }}
|
|
{% elif is_generic_calibration %}
|
|
{{ true }}
|
|
{% else %}
|
|
{{ trigger_id_defined and 'temperature_change' in trigger.id}}
|
|
{% endif %}
|
|
|
|
is_scene_create_trigger: >
|
|
{{ trigger_id_defined and (("window_on" in trigger.id and not state_party) or ("party_on" in trigger.id and not state_window)) }}
|
|
|
|
is_scene_apply_trigger: >
|
|
{{ trigger_id_defined and ("window_off" in trigger.id or "party_off" in trigger.id) and not is_legacy_restore and not (state_window or state_party) }}
|
|
|
|
is_scene_destroy_trigger: >
|
|
{{ (is_change_trigger or trigger.id == 'temperature_change_heating_adjustment') and (state_window or state_party) }}
|
|
|
|
# scene management
|
|
scene_entities: "{{ valves }}"
|
|
scene_window_id: "{{ 'scene.' + this.entity_id | replace('automation.','') | replace('.','_') + '_window' }}"
|
|
scene_party_id: "{{ 'scene.' + this.entity_id | replace('automation.','') | replace('.','_') + '_party' }}"
|
|
scenes_all: "{{ [scene_window_id, scene_party_id] }}"
|
|
|
|
scene_to_apply: >
|
|
{% if is_scene_apply_trigger and "window_off" in trigger.id %}
|
|
{{ scene_window_id }}
|
|
{% elif is_scene_apply_trigger and "party_off" in trigger.id %}
|
|
{{ scene_party_id }}
|
|
{% else %}
|
|
{{ none }}
|
|
{% endif %}
|
|
|
|
scenes_to_destroy: >
|
|
{% set scenes = [] %}
|
|
{% if is_scene_destroy_trigger %}
|
|
{% set scenes = iif(state_window, scenes + [scene_window_id], scenes) %}
|
|
{% set scenes = iif(state_party, scenes + [scene_party_id], scenes) %}
|
|
{% endif %}
|
|
{{ scenes }}
|
|
|
|
scene_to_create: >
|
|
{{ iif(is_scene_create_trigger and "window_on" in trigger.id, scene_window_id, scene_party_id) }}
|
|
|
|
#####################################################################################
|
|
#################################### CHANGES ########################################
|
|
#####################################################################################
|
|
|
|
set_comfort: >
|
|
{% if is_force_max_temperature %}
|
|
{{ true }}
|
|
{% elif entry_mode == 'eco' %}
|
|
{{ false }}
|
|
{% elif entry_mode == 'comfort' %}
|
|
{{ true }}
|
|
{% elif state_party %}
|
|
{{ true }}
|
|
{% elif is_force_eco_temperature %}
|
|
{{ false }}
|
|
{% elif is_away %}
|
|
{{ true }}
|
|
{% elif not is_scheduler_defined and not is_presence_sensor_defined %}
|
|
{{ is_anybody_home_or_proximity }}
|
|
{% else %}
|
|
{% set comfort_state = state_scheduler or state_presence %}
|
|
|
|
{% if is_person_defined or is_proximity_defined %}
|
|
{{ is_anybody_home_or_proximity and comfort_state }}
|
|
{% else %}
|
|
{{ comfort_state }}
|
|
{% endif %}
|
|
{% endif %}
|
|
|
|
mode: >
|
|
{% if not state_ahc %}
|
|
{{ 'off' }}
|
|
{% elif state_window and input_window_open_temperature | int == 0 and not set_max_temperature %}
|
|
{{ 'off' }}
|
|
{% elif entry_mode == 'off' %}
|
|
{{ 'off' }}
|
|
{% elif is_off_instead_min and not set_comfort %}
|
|
{{ 'off' }}
|
|
{% elif is_off_if_nobody_home and is_person_defined and not is_anybody_home_or_proximity and not set_comfort %}
|
|
{{ 'off' }}
|
|
{% else %}
|
|
{{ input_hvac_mode }}
|
|
{% endif %}
|
|
|
|
temperature_comfort_of_entity: >
|
|
{% if(input_temperature_comfort_entity != none) %}
|
|
{{ states(input_temperature_comfort_entity) | float }}
|
|
{% else %}
|
|
{{ none }}
|
|
{% endif %}
|
|
|
|
temperature_eco_of_entity: >
|
|
{% if(input_temperature_eco_entity != none) %}
|
|
{{ states(input_temperature_eco_entity) | float }}
|
|
{% else %}
|
|
{{ none }}
|
|
{% endif %}
|
|
|
|
temperature_comfort: "{{ [entry_comfort_temp, temperature_comfort_of_entity, input_temperature_comfort_static] | reject('==', none) | first }}"
|
|
temperature_away: "{{ temperature_comfort | float - input_away_offset }}"
|
|
temperature_eco: "{{ [entry_eco_temp, temperature_eco_of_entity, input_temperature_eco_static] | reject('==', none) | first }}"
|
|
target_temperature: >
|
|
{% if state_window and input_window_open_temperature > 0 %}
|
|
{{ input_window_open_temperature }}
|
|
{% elif state_party %}
|
|
{{ iif(party_temp != none, party_temp, temperature_comfort) }}
|
|
{% elif is_frost_protection %}
|
|
{{ input_frost_protection_temp }}
|
|
{% else %}
|
|
{{ iif(set_comfort, iif(is_away, temperature_away, temperature_comfort), temperature_eco) }}
|
|
{% endif %}
|
|
|
|
changes: >
|
|
{% set n = namespace(dict=[]) %}
|
|
|
|
{% set original_mode = mode %}
|
|
|
|
{% if not is_changes_trigger %}
|
|
{{ n.dict }}
|
|
{% else %}
|
|
{% for valve in input_trvs %}
|
|
|
|
{% set current_valve_temp = state_attr(valve, 'current_temperature') | float(20) %}
|
|
{% set current_valve_target_temp = state_attr(valve, 'temperature') | float(temperature) %}
|
|
{% set current_valve_mode = states(valve) %}
|
|
{% set min_temp = state_attr(valve, 'min_temp') | float(5) %}
|
|
{% set max_temp = state_attr(valve, 'max_temp') | float(30) %}
|
|
|
|
{% set valve_temp = target_temperature %}
|
|
|
|
{% set dont_turn_off =
|
|
valve in valves_without_off_mode or
|
|
is_not_off_but_min or
|
|
(state_window and input_window_open_temperature > 0) or
|
|
set_max_temperature %}
|
|
|
|
{% set ref_temp = current_valve_temp %}
|
|
{% if valid_temperature_sensor %}
|
|
{% set ref_temp = value_temperature_sensor | float(current_valve_temp) %}
|
|
{% endif %}
|
|
|
|
{% if is_heat_only_if_below_real_temp and iif(factor == 1, target_temperature <= ref_temp, target_temperature >= ref_temp) %}
|
|
{% set mode = 'off' %}
|
|
{% endif %}
|
|
|
|
{% set valve_mode = iif(mode == 'off' and dont_turn_off, current_valve_mode, mode) %}
|
|
|
|
{% if mode != 'off' %}
|
|
|
|
{% if is_aggressive_mode and not is_aggressive_mode_calibration %}
|
|
|
|
{% set temp_diff = valve_temp - ref_temp %}
|
|
|
|
{% if temp_diff * factor < input_aggressive_mode_range * -1 %}
|
|
{% set valve_temp = valve_temp - input_aggressive_mode_offset * factor %}
|
|
{% elif temp_diff * factor > input_aggressive_mode_range %}
|
|
{% set valve_temp = valve_temp + input_aggressive_mode_offset * factor %}
|
|
{% endif %}
|
|
|
|
{% endif %}
|
|
|
|
{% if input_calibration_generic %}
|
|
|
|
{% if current_valve_temp != ref_temp %}
|
|
{% set offset = current_valve_temp - ref_temp %}
|
|
|
|
{% set offset = iif(offset > float(input_generic_calibration_offset), input_generic_calibration_offset, offset) %}
|
|
{% set offset = iif(offset < float(input_generic_calibration_offset) * -1, input_generic_calibration_offset * -1, offset) %}
|
|
|
|
{% set temp_with_offset = float(valve_temp) + float(offset) %}
|
|
{% set step = state_attr(valve, 'target_temp_step') | float(0.5) %}
|
|
|
|
{% set temp_with_offset = (temp_with_offset | float(0) / float(step)) | round(0) * float(step) %}
|
|
|
|
{% set valve_temp = iif(input_calibration_step_size == 'full', float(temp_with_offset) | round(), temp_with_offset | round(1)) %}
|
|
|
|
{% endif %}
|
|
{% endif %}
|
|
|
|
{% endif %}
|
|
|
|
{% if mode == 'off' and dont_turn_off %}
|
|
{% set valve_temp = min_temp %}
|
|
{% endif %}
|
|
|
|
{% set valve_temp = iif(set_max_temperature, max_temp, valve_temp) %}
|
|
{% set valve_temp = iif(valve_temp > max_temp, max_temp, valve_temp) %}
|
|
{% set valve_temp = iif(valve_temp < min_temp, min_temp, valve_temp) %}
|
|
|
|
{% if current_valve_mode != valve_mode or current_valve_target_temp != valve_temp %}
|
|
{% set n.dict = n.dict + [(valve, [{'mode': valve_mode , 'temp': valve_temp}])] %}
|
|
{% endif %}
|
|
|
|
{% endfor %}
|
|
|
|
{% set mode = original_mode %}
|
|
|
|
{{ dict.from_keys(n.dict) }}
|
|
{% endif %}
|
|
|
|
positioning: >
|
|
{% set n = namespace(dict=[]) %}
|
|
|
|
{% if input_valve_positioning_mode == 'off' %}
|
|
{{ n.dict }}
|
|
{% else %}
|
|
{% for valve in input_trvs %}
|
|
|
|
{% set current_temp = state_attr(valve, 'current_temperature') | float(none) %}
|
|
{% if valid_temperature_sensor %}
|
|
{% set current_temp = value_temperature_sensor | float(none) %}
|
|
{% endif %}
|
|
|
|
{% set target_temp = state_attr(valve, 'temperature') | float(none) %}
|
|
|
|
{% set open_valve_entity = device_entities(device_id(valve)) | expand
|
|
| selectattr('domain','in','number')
|
|
| selectattr('entity_id', 'search', input_valve_opening_keyword)
|
|
| map(attribute='entity_id')
|
|
| list | first | default(none) %}
|
|
|
|
{% if open_valve_entity != none and current_temp != none and target_temp != none and
|
|
(
|
|
(trigger_id_defined and trigger.id == 'positioning_event') or
|
|
([open_valve_entity] | expand | map(attribute='last_changed') | first) + timedelta(**input_valve_positioning_timeout) <= now()
|
|
)
|
|
%}
|
|
|
|
{% set opening = 100 %}
|
|
{% set difference = target_temp - current_temp %}
|
|
{% set step_size = input_valve_positioning_step_size | int %}
|
|
|
|
{% if input_fully_open_difference > 0 and not is_force_max_temperature %}
|
|
|
|
{% set opening_regular = (100 / input_fully_open_difference) * difference %}
|
|
{% set opening_pessimistic = sqrt(((100 / input_fully_open_difference) * difference) | abs) * 10 %}
|
|
{% set opening_optimistic = ((100 / input_fully_open_difference) * difference)**2 / 100 %}
|
|
|
|
{% set opening = opening_regular %}
|
|
{% set opening = iif(input_valve_positioning_mode == 'pessimistic', opening_pessimistic, opening) %}
|
|
{% set opening = iif(input_valve_positioning_mode == 'optimistic', opening_optimistic, opening) %}
|
|
|
|
{% set opening = iif(difference >= input_fully_open_difference, 100, opening) %}
|
|
{% set opening = iif(difference < 0, 0, opening) %}
|
|
|
|
{% set opening = opening / 100 * input_valve_positioning_max_opening %}
|
|
|
|
{% set opening = ((opening + step_size / 2) // step_size * step_size) | int %}
|
|
{% endif %}
|
|
|
|
{% set open_valve_entity_value = states(open_valve_entity) | int %}
|
|
|
|
{% if open_valve_entity_value != opening %}
|
|
{% set n.dict = n.dict + [(valve, [{'entity': open_valve_entity , 'value': opening, 'current_temp': current_temp, 'target_temp': target_temp, 'difference': difference}])] %}
|
|
{% endif %}
|
|
|
|
{% endif %}
|
|
{% endfor %}
|
|
|
|
{{ dict.from_keys(n.dict) }}
|
|
{% endif %}
|
|
|
|
# reset temperature
|
|
reset_data: >
|
|
{% set result = [] %}
|
|
{% if is_adjustment_trigger %}
|
|
{% if entry_comfort_temp != none and input_temperature_comfort_entity != none %}
|
|
{% set result = result + [{'entity': input_temperature_comfort_entity, 'temp': entry_comfort_temp}] %}
|
|
{% endif %}
|
|
{% if entry_eco_temp != none and input_temperature_eco_entity != none %}
|
|
{% set result = result + [{'entity': input_temperature_eco_entity, 'temp': entry_eco_temp}] %}
|
|
{% endif %}
|
|
{% else %}
|
|
{% set entity = none %}
|
|
{% if is_reset and set_comfort %}
|
|
{% set entity = iif(is_pysical_change, input_temperature_comfort_entity, input_temperature_eco_entity) %}
|
|
{% elif is_reset and not set_comfort %}
|
|
{% set entity = iif(is_pysical_change, input_temperature_eco_entity, input_temperature_comfort_entity) %}
|
|
{% endif %}
|
|
|
|
{% set temp_r = none %}
|
|
{% if is_pysical_change %}
|
|
{% set temp_r = state_attr(trigger.to_state.entity_id,'temperature') %}
|
|
{% else %}
|
|
{% set temp_r = iif(is_reset and entity == input_temperature_eco_entity,
|
|
input_temperature_eco_static, input_temperature_comfort_static) %}
|
|
{% endif %}
|
|
|
|
{% if entity != none and temp_r != none %}
|
|
{% set result = result + [{'entity': entity, 'temp': temp_r}] %}
|
|
{% endif %}
|
|
{% endif %}
|
|
|
|
{{ result }}
|
|
|
|
is_reset_trigger: "{{ is_reset and reset_data | count > 0 }}"
|
|
|
|
#####################################################################################
|
|
################################## CALIBRATION ######################################
|
|
#####################################################################################
|
|
|
|
is_native_calibration: "{{ not input_calibration_generic and entry_calibration and valid_temperature_sensor }}"
|
|
is_native_calibration_trigger: "{{ is_calibration_trigger and is_native_calibration }}"
|
|
|
|
rounding_mode: >
|
|
{% if is_number(input_calibration_step_size) or input_calibration_step_size == 'full' %}
|
|
{{ 'manual' }}
|
|
{% else %}
|
|
{{ 'auto' }}
|
|
{% endif %}
|
|
|
|
# TADO
|
|
calibration_tado: >
|
|
{% set n = namespace(dict=[]) %}
|
|
|
|
{% if is_native_calibration_trigger %}
|
|
{% for valve in valves_tado %}
|
|
|
|
{% set offset_old = state_attr(valve, 'offset_celsius') | float(0) %}
|
|
{% set local_temperature = state_attr(valve, 'current_temperature') | float %}
|
|
{% set calibration_sensor_temperature = value_temperature_sensor | float %}
|
|
|
|
{% set offset_new = (-(local_temperature - calibration_sensor_temperature) + offset_old) %}
|
|
|
|
{% if is_aggressive_mode_calibration %}
|
|
{% set temp_diff = state_attr(valve,'temperature') | float(target_temperature) - calibration_sensor_temperature %}
|
|
|
|
{% if temp_diff * factor < input_aggressive_mode_range * -1 %}
|
|
{% set offset_new = offset_new + input_aggressive_mode_offset * factor %}
|
|
{% elif temp_diff * factor > input_aggressive_mode_range %}
|
|
{% set offset_new = offset_new - input_aggressive_mode_offset * factor %}
|
|
{% endif %}
|
|
{% endif %}
|
|
|
|
{% set t_min = -10.9 %}
|
|
{% set t_max = 10.9 %}
|
|
|
|
{% set offset_new = iif(offset_new > t_max, t_max, offset_new) %}
|
|
{% set offset_new = iif(offset_new < t_min, t_min, offset_new) %}
|
|
|
|
{% set offset_new = offset_new | round(1) %}
|
|
|
|
{% if (float(offset_old) - float(offset_new)) | abs >= float(input_calibration_delta) %}
|
|
{% set n.dict = n.dict + [(valve, [{'value': offset_new }])] %}
|
|
{% endif %}
|
|
|
|
{% endfor %}
|
|
{% endif %}
|
|
|
|
{{ dict.from_keys(n.dict) }}
|
|
|
|
# XIAOMI / AQARA / SONOFF
|
|
calibration_external: >
|
|
{% set n = namespace(dict=[]) %}
|
|
|
|
{% if is_native_calibration_trigger %}
|
|
{% for valve in valves_external %}
|
|
|
|
{% set calibration_entities = device_entities(device_id(valve)) |
|
|
expand | selectattr('domain','in','number') |
|
|
selectattr('entity_id', 'search', input_calibration_key_word) |
|
|
map(attribute='entity_id') | list %}
|
|
|
|
{% if calibration_entities | count > 0 %}
|
|
|
|
{% set calibration_entity = calibration_entities | first %}
|
|
{% set offset_old = states(calibration_entity) | float(0) %}
|
|
{% set offset_new = value_temperature_sensor | float %}
|
|
|
|
{% set step = state_attr(calibration_entity, 'step') | float(1) %}
|
|
{% if rounding_mode == 'manual' %}
|
|
{% set step = input_calibration_step_size | float(1) %}
|
|
{% endif %}
|
|
|
|
{% set min_val = state_attr(calibration_entity,'min') | float(0) %}
|
|
{% set max_val = state_attr(calibration_entity,'max') | float(55) %}
|
|
|
|
{% if is_aggressive_mode_calibration %}
|
|
{% set temp_diff = state_attr(valve,'temperature') | float(target_temperature) - value_temperature_sensor | float %}
|
|
|
|
{% if temp_diff * factor < input_aggressive_mode_range * -1 %}
|
|
{% set offset_new = offset_new + input_aggressive_mode_offset * factor %}
|
|
{% elif temp_diff * factor > input_aggressive_mode_range %}
|
|
{% set offset_new = offset_new - input_aggressive_mode_offset * factor %}
|
|
{% endif %}
|
|
{% endif %}
|
|
|
|
{% set round_size = iif('.' in (step | string), (step | string).split('.')[1] | length, 0) %}
|
|
{% set offset_new = ((offset_new | float(0) / step) | round(0) * step) | round(round_size) | float %}
|
|
|
|
{% set offset_new = iif(offset_new > max_val, max_val, offset_new) %}
|
|
{% set offset_new = iif(offset_new < min_val, min_val, offset_new) %}
|
|
|
|
{% if (float(offset_old) - float(offset_new)) | abs >= float(input_calibration_delta) %}
|
|
{% set n.dict = n.dict + [(calibration_entity, [{'value': offset_new, 'valve': valve}])] %}
|
|
{% endif %}
|
|
{% endif %}
|
|
|
|
{% endfor %}
|
|
{% endif %}
|
|
|
|
{{ dict.from_keys(n.dict) }}
|
|
|
|
# DANFOSS, POPP, HIVE, BOSCH
|
|
calibration_dph: >
|
|
{% set n = namespace(dict=[]) %}
|
|
{% if is_native_calibration_trigger %}
|
|
|
|
{% for valve in valves_dph %}
|
|
|
|
{% set calibration_entities = device_entities(device_id(valve)) |
|
|
expand | selectattr('domain','in','number') |
|
|
selectattr('entity_id', 'search', input_calibration_key_word) |
|
|
map(attribute='entity_id') | list %}
|
|
|
|
{% if calibration_entities | count > 0 %}
|
|
|
|
{% set calibration_entity = calibration_entities | first %}
|
|
|
|
{% set min_val = state_attr(calibration_entity,'min')%}
|
|
{% set max_val = state_attr(calibration_entity,'max')%}
|
|
|
|
{% set step = state_attr(calibration_entity, 'step') | float(1) %}
|
|
{% if rounding_mode == 'manual' %}
|
|
{% set step = input_calibration_step_size | float(1) %}
|
|
{% endif %}
|
|
|
|
{% set current_temp = state_attr(valve,'current_temperature') | float(20) %}
|
|
|
|
{% set new_state = value_temperature_sensor | float(current_temp) %}
|
|
{% set old_state = states(calibration_entity) | float %}
|
|
|
|
{% if is_aggressive_mode_calibration %}
|
|
{% set temp_diff = state_attr(valve,'temperature') | float(target_temperature) - value_temperature_sensor | float %}
|
|
|
|
{% if temp_diff * factor < input_aggressive_mode_range * -1 %}
|
|
{% set new_state = new_state + input_aggressive_mode_offset * factor %}
|
|
{% elif temp_diff * factor > input_aggressive_mode_range %}
|
|
{% set new_state = new_state - input_aggressive_mode_offset * factor %}
|
|
{% endif %}
|
|
{% endif %}
|
|
|
|
{% if step <= 1 and max_val | float < 1000 %}
|
|
{% set round_size = iif('.' in (step | string), (step | string).split('.')[1] | length, 0) %}
|
|
{% set new_state = ((new_state | float(0) / step) | round(0) * step) | round(round_size) | float %}
|
|
{% else %}
|
|
{% set new_state = new_state * 100 | int %}
|
|
{% set old_state = old_state | int %}
|
|
{% endif %}
|
|
|
|
{% set update_calibration = old_state != new_state %}
|
|
|
|
{% if is_calibration_trigger and not update_calibration %}
|
|
{% set last_updated = [calibration_entity] | expand | map(attribute='last_updated') | first %}
|
|
{% set update_calibration = as_datetime(current_time_stamp) - timedelta(minutes=20) >= last_updated %}
|
|
{% endif %}
|
|
|
|
{% if update_calibration %}
|
|
{% set n.dict = n.dict + [(calibration_entity, [{'value': new_state, 'valve': valve}])] %}
|
|
{% endif%}
|
|
|
|
{% endif %}
|
|
|
|
{% endfor %}
|
|
{% endif %}
|
|
|
|
{{ dict.from_keys(n.dict) }}
|
|
|
|
# COMMON CALIBRATION e.g. TUYA
|
|
calibration_common: >
|
|
{% set n = namespace(dict=[]) %}
|
|
|
|
{% if is_native_calibration_trigger %}
|
|
|
|
{% for valve in valves_calibration_common %}
|
|
|
|
{% set calibration_entities = device_entities(device_id(valve)) |
|
|
expand | selectattr('domain','in','number') |
|
|
selectattr('entity_id', 'search', input_calibration_key_word) |
|
|
map(attribute='entity_id') | list %}
|
|
|
|
{% if calibration_entities | count > 0%}
|
|
|
|
{% set calibration_entity = calibration_entities | first %}
|
|
|
|
{% set step = state_attr(calibration_entity, 'step') | float(1) %}
|
|
{% if rounding_mode == 'manual' %}
|
|
{% set step = input_calibration_step_size | float(1) %}
|
|
{% endif %}
|
|
|
|
{% set min_calibration_value = state_attr(calibration_entity,'min') | float %}
|
|
{% set max_calibration_value = state_attr(calibration_entity,'max') | float %}
|
|
{% set thermostat_temperature = state_attr(valve, 'current_temperature') | float %}
|
|
{% set offset_old = states(calibration_entity) | float(0) %}
|
|
|
|
{% set new_calibration_value = (-(thermostat_temperature - value_temperature_sensor) + offset_old) %}
|
|
|
|
{% if is_aggressive_mode_calibration %}
|
|
{% set temp_diff = state_attr(valve,'temperature') | float(target_temperature) - value_temperature_sensor %}
|
|
|
|
{% if temp_diff * factor < input_aggressive_mode_range * -1 %}
|
|
{% set new_calibration_value = new_calibration_value + input_aggressive_mode_offset * factor %}
|
|
{% elif temp_diff * factor > input_aggressive_mode_range %}
|
|
{% set new_calibration_value = new_calibration_value - input_aggressive_mode_offset * factor %}
|
|
{% endif %}
|
|
{% endif %}
|
|
|
|
{% set new_calibration_value = iif(new_calibration_value > max_calibration_value, max_calibration_value, new_calibration_value) %}
|
|
{% set new_calibration_value = iif(new_calibration_value < min_calibration_value, min_calibration_value, new_calibration_value) %}
|
|
|
|
{% set round_size = iif('.' in (step | string), (step | string).split('.')[1] | length, 0) %}
|
|
|
|
{% set offset_new = ((new_calibration_value | float(0) / step) | round(0) * step) | round(round_size) | float %}
|
|
|
|
{% if (float(offset_old) - float(offset_new)) | abs >= float(input_calibration_delta) %}
|
|
{% set n.dict = n.dict + [(calibration_entity, [{'value': offset_new, 'valve': valve}])] %}
|
|
{% endif %}
|
|
|
|
{% endif %}
|
|
|
|
{% endfor %}
|
|
{% endif %}
|
|
|
|
{{ dict.from_keys(n.dict) }}
|
|
|
|
calibration_value_set: "{{ dict(dict(calibration_external, **calibration_dph),**calibration_common) }}"
|
|
|
|
##############################################################################################
|
|
################################## CONDITIONS / BLOCKER ######################################
|
|
##############################################################################################
|
|
|
|
no_changes: >
|
|
{{
|
|
(input_persons | count == 0 and
|
|
input_mode_guest == none and
|
|
input_schedulers | count == 0 and
|
|
input_presence_sensor == none and
|
|
input_proximity == none) or
|
|
(is_temperature_sensor_defined and not valid_temperature_sensor)
|
|
}}
|
|
|
|
# conditions
|
|
scene_trigger: "{{ is_scene_create_trigger or is_scene_apply_trigger or is_scene_destroy_trigger }}"
|
|
change_trigger: "{{ is_changes_trigger and not scene_trigger and changes | count > 0 and not no_changes}}"
|
|
reset_trigger: "{{ is_reset_trigger and not no_changes }}"
|
|
calibration_trigger: "{{ is_calibration_trigger and not input_calibration_generic and (calibration_value_set | count > 0 or calibration_tado | count > 0) }}"
|
|
positioning_trigger: "{{ positioning | count > 0 }}"
|
|
|
|
# warnings
|
|
automation_name: "{{ state_attr(this.entity_id,'friendly_name') }}"
|
|
warnings: >
|
|
{% set messages = [] %}
|
|
{% if not is_uptime_defined %}
|
|
{% set messages = messages + ['To make Advance Heating Control work properly just setup the uptime integration (https://www.home-assistant.io/integrations/uptime/)'] %}
|
|
{% elif is_aggressive_mode and not input_aggressive_mode_calibration and is_physical_change_enabled %}
|
|
{% set messages = messages + ['Aggressive Mode in combination with physical change / sync feature is not recommended. Expect unwanted side effects.'] %}
|
|
{% elif is_generic_calibration and is_physical_change_enabled %}
|
|
{% set messages = messages + ['Generic Calibration in combination with physical change / sync feature is not recommended. Expect unwanted side effects.'] %}
|
|
{% elif valves_unsupported | count > 0 %}
|
|
{% set messages = messages + ['Unsupported climate entities: ' + valves_unsupported | join(',') | string ] %}
|
|
{% elif is_temperature_sensor_defined and not valid_temperature_sensor %}
|
|
{% set messages = messages + ['The temperature sensor' + input_temperature_sensor + ' has an invalid state: ' + states(input_temperature_sensor) ] %}
|
|
{% endif %}
|
|
|
|
{{ messages }}
|
|
|
|
climates_information: >
|
|
{% set n = namespace(dict=[]) %}
|
|
|
|
{% for valve in input_trvs %}
|
|
{% set temperature = state_attr(valve,'temperature') %}
|
|
{% set current_temperature = state_attr(valve,'current_temperature') %}
|
|
{% set state = states(valve) %}
|
|
{% set n.dict = n.dict + [{'entity_id': valve, 'state': state, 'temperature': temperature, 'current_temperature': current_temperature}] %}
|
|
{% endfor %}
|
|
|
|
{{ n.dict }}
|
|
|
|
conditions:
|
|
- condition: or
|
|
conditions:
|
|
- condition: template
|
|
value_template: "{{ calibration_trigger }}"
|
|
- condition: template
|
|
value_template: "{{ scene_trigger }}"
|
|
- condition: template
|
|
value_template: "{{ change_trigger }}"
|
|
- condition: template
|
|
value_template: "{{ reset_trigger }}"
|
|
- condition: template
|
|
value_template: "{{ positioning_trigger }}"
|
|
|
|
actions:
|
|
- variables:
|
|
is_delayed: "{{ not (not is_uptime_defined or (now() | as_datetime - states(up_time_sensor) | as_datetime) > timedelta(**input_startup_delay)) }}"
|
|
- action: system_log.write
|
|
data:
|
|
message: >
|
|
{{ 'AHC - ' + automation_name | string + ' \n ' +
|
|
'automation delayed: ' + is_delayed | string }}
|
|
level: !input input_log_level
|
|
logger: blueprints.panhans.heatingcontrol
|
|
- wait_template: >
|
|
{{ not is_uptime_defined or (now() | as_datetime - states(up_time_sensor) | as_datetime) > timedelta(**input_startup_delay) }}
|
|
- choose:
|
|
- conditions: "{{ is_delayed }}"
|
|
sequence:
|
|
- event: ahc_delay_event
|
|
event_data:
|
|
automation: "{{ this.entity_id }}"
|
|
|
|
default:
|
|
- if:
|
|
- condition: template
|
|
value_template: "{{ warnings | count > 0 }}"
|
|
then:
|
|
- action: system_log.write
|
|
data:
|
|
level: warning
|
|
logger: blueprints.panhans.heatingcontrol
|
|
message: >
|
|
{{ 'AHC-Warnings - ' + automation_name + ':\n' + warnings | join('\n') }}
|
|
|
|
- event: ahc_event
|
|
event_data:
|
|
state: "{{ state_ahc }}"
|
|
mode: "{{ iif(set_comfort == true, 'comfort', 'eco') }}"
|
|
automation: "{{ this.entity_id }}"
|
|
is_person_defined: "{{ is_person_defined }}"
|
|
is_anybody_home: "{{ is_anybody_home }}"
|
|
is_proximity_defined: "{{ is_proximity_defined }}"
|
|
is_anybody_home_or_proximity: "{{ is_anybody_home_or_proximity }}"
|
|
is_guest_mode: "{{ is_guest_mode }}"
|
|
active_scheduler: "{{ active_scheduler }}"
|
|
state_scheduler: "{{ state_scheduler }}"
|
|
state_presence_sensor: "{{ state_presence_sensor }}"
|
|
state_presence_scheduler: "{{ state_presence_scheduler }}"
|
|
state_presence: "{{ state_presence }}"
|
|
state_proximity_arrived: "{{ state_proximity_arrived }}"
|
|
state_proximity_way_home: "{{ state_proximity_way_home }}"
|
|
is_force_max_temperature: "{{ is_force_max_temperature }}"
|
|
is_force_eco_temperature: "{{ is_force_eco_temperature }}"
|
|
active_party_entity: "{{ active_party_entity }}"
|
|
party_temp: "{{ party_temp }}"
|
|
is_away: "{{ is_away }}"
|
|
state_window: "{{ state_window }}"
|
|
is_aggressive_mode: "{{ is_aggressive_mode }}"
|
|
is_frost_protection: "{{ is_frost_protection }}"
|
|
is_liming_protection: "{{ is_liming_protection }}"
|
|
state_outside_temp: "{{ state_outside_temp }}"
|
|
entry_time: "{{ entry_time }}"
|
|
thermostats: "{{ input_trvs }}"
|
|
hvac_mode: "{{ mode }}"
|
|
temperature_comfort: "{{ temperature_comfort }}"
|
|
temperature_eco: "{{ temperature_eco }}"
|
|
target_temperature: "{{ target_temperature }}"
|
|
set_max_temperature: "{{ set_max_temperature }}"
|
|
last_trigger_id: "{{ iif(trigger_id_defined, trigger.id, '') }}"
|
|
calibration_trigger: "{{ is_generic_calibration_trigger or calibration_trigger }}"
|
|
change_trigger: "{{ change_trigger }}"
|
|
warnings: "{{ warnings | count > 0 }}"
|
|
|
|
# calibration
|
|
- if:
|
|
- condition: template
|
|
value_template: "{{ calibration_trigger }}"
|
|
- condition: and
|
|
conditions: !input input_custom_condition_calibration
|
|
then:
|
|
- action: system_log.write
|
|
data:
|
|
message: >
|
|
{{ 'AHC - Calibration - ' + automation_name | string + ' \n ' +
|
|
'calibration data set: ' + calibration_value_set | string }}
|
|
level: !input input_log_level
|
|
logger: blueprints.panhans.heatingcontrol
|
|
- repeat:
|
|
count: "{{ calibration_value_set | count | int }}"
|
|
sequence:
|
|
- variables:
|
|
index: "{{ repeat.index-1 }}"
|
|
calibration_entity: "{{ (calibration_value_set.keys() | list) [index] }}"
|
|
thermostat: "{{ (((calibration_value_set.values() | list) [index]) | first) ['valve'] }}"
|
|
offset: "{{ (((calibration_value_set.values() | list) [index]) | first) ['value'] }}"
|
|
select_entity: "{{ device_entities(device_id(thermostat)) |
|
|
expand | selectattr('domain','in','select') |
|
|
selectattr('attributes.options', 'contains', 'external') |
|
|
map(attribute='entity_id') | list | first | default(none) }}"
|
|
is_external: "{{ select_entity != none and not is_state(select_entity, 'external') }}"
|
|
|
|
- action: system_log.write
|
|
data:
|
|
message: >
|
|
{{ 'AHC - Calibration - ' + automation_name | string + ' \n ' +
|
|
'calibration entity: ' + calibration_entity | string + ' \n ' +
|
|
'offset: ' + offset | string }}
|
|
level: !input input_log_level
|
|
logger: blueprints.panhans.heatingcontrol
|
|
|
|
- if:
|
|
- condition: template
|
|
value_template: "{{ is_external }}"
|
|
then:
|
|
- action: select.select_option
|
|
target:
|
|
entity_id: "{{ select_entity }}"
|
|
data:
|
|
option: external
|
|
- delay: !input input_action_call_delay
|
|
|
|
- action: number.set_value
|
|
data:
|
|
value: "{{ float(offset) }}"
|
|
target:
|
|
entity_id: "{{ calibration_entity }}"
|
|
- delay: !input input_action_call_delay
|
|
|
|
# TADO CALIBRATION
|
|
- repeat:
|
|
count: "{{ calibration_tado | count | int }}"
|
|
sequence:
|
|
- variables:
|
|
index: "{{ repeat.index-1 }}"
|
|
thermostat: "{{ (calibration_tado.keys() | list) [index] }}"
|
|
offset: "{{ (((calibration_tado.values() | list) [index]) | first) ['value'] }}"
|
|
- action: "{{ 'tado.set_climate_temperature_offset' }}"
|
|
data:
|
|
offset: "{{ offset }}"
|
|
entity_id: "{{ thermostat }}"
|
|
- delay: !input input_action_call_delay
|
|
|
|
# valve opening
|
|
- if:
|
|
- condition: template
|
|
value_template: "{{ positioning_trigger }}"
|
|
then:
|
|
- repeat:
|
|
count: "{{ positioning | count | int }}"
|
|
sequence:
|
|
- variables:
|
|
index: "{{ repeat.index-1 }}"
|
|
thermostat: "{{ (positioning.keys() | list) [index] }}"
|
|
positioning_value: "{{ (((positioning.values() | list) [index]) | first) ['value'] }}"
|
|
positioning_entity: "{{ (((positioning.values() | list) [index]) | first) ['entity'] }}"
|
|
|
|
- action: system_log.write
|
|
data:
|
|
message: >
|
|
{{ 'AHC - Positioning - ' + automation_name | string + ' \n ' +
|
|
'entity: ' + positioning_entity | string + ' \n ' +
|
|
'value: ' + positioning_value | string }}
|
|
level: !input input_log_level
|
|
logger: blueprints.panhans.heatingcontrol
|
|
|
|
- action: number.set_value
|
|
data:
|
|
value: "{{ positioning_value | int }}"
|
|
target:
|
|
entity_id: "{{ positioning_entity }}"
|
|
- delay: !input input_action_call_delay
|
|
|
|
# scenes
|
|
# scene create
|
|
- if:
|
|
- condition: template
|
|
value_template: "{{ is_scene_create_trigger }}"
|
|
- condition: template
|
|
value_template: "{{ states[scene_to_create] == none }}"
|
|
then:
|
|
- action: scene.create
|
|
data:
|
|
snapshot_entities: "{{ scene_entities }}"
|
|
scene_id: "{{ scene_to_create.split('.')[1] }}"
|
|
|
|
# scene destroy
|
|
- if:
|
|
- condition: template
|
|
value_template: "{{ is_scene_destroy_trigger }}"
|
|
- condition: template
|
|
value_template: "{{ scenes_to_destroy | count > 0 }}"
|
|
then:
|
|
- repeat:
|
|
count: "{{ scenes_to_destroy | count | int }}"
|
|
sequence:
|
|
- variables:
|
|
scene_to_destroy: "{{ scenes_to_destroy[repeat.index-1] }}"
|
|
- if:
|
|
- condition: template
|
|
value_template: "{{ states[scene_to_destroy] != none }}"
|
|
then:
|
|
- action: scene.delete
|
|
target:
|
|
entity_id: "{{ scene_to_destroy }}"
|
|
|
|
# scene apply
|
|
- variables:
|
|
scene_to_apply_tmp: >
|
|
{% if scene_to_apply != none and states[scene_to_apply] != none %}
|
|
{{ scene_to_apply }}
|
|
{% else %}
|
|
{{ scenes_all | expand | reject('==',none) | map(attribute="entity_id") | list | first | default(none) }}
|
|
{% endif %}
|
|
- if:
|
|
- condition: template
|
|
value_template: "{{ is_scene_apply_trigger }}"
|
|
- condition: template
|
|
value_template: "{{ scene_to_apply_tmp != none and states[scene_to_apply_tmp] != none }}"
|
|
then:
|
|
- action: system_log.write
|
|
data:
|
|
message: >
|
|
{{ 'AHC - Calibration - ' + automation_name | string + ' \n ' +
|
|
'apply scene: ' + scene_to_apply_tmp | string + ' state: ' + states[scene_to_apply_tmp] | string }}
|
|
level: !input input_log_level
|
|
logger: blueprints.panhans.heatingcontrol
|
|
- action: scene.turn_on
|
|
target:
|
|
entity_id: "{{ scene_to_apply_tmp }}"
|
|
- action: scene.delete
|
|
target:
|
|
entity_id: "{{ scene_to_apply_tmp }}"
|
|
- condition: template
|
|
value_template: "{{ false }}"
|
|
else:
|
|
# reset
|
|
- if:
|
|
- condition: template
|
|
value_template: "{{ is_reset_trigger }}"
|
|
then:
|
|
- repeat:
|
|
count: "{{ reset_data | count | int }}"
|
|
sequence:
|
|
- action: system_log.write
|
|
data:
|
|
message: >
|
|
{{ 'AHC - Calibration - ' + automation_name | string + ' \n ' +
|
|
'reset data: ' + reset_data | string }}
|
|
level: !input input_log_level
|
|
logger: blueprints.panhans.heatingcontrol
|
|
- variables:
|
|
index: "{{ repeat.index-1 }}"
|
|
reset_entity: "{{ reset_data[index]['entity'] }}"
|
|
reset_temp: >
|
|
{% set temp_r = reset_data[index]['temp'] %}
|
|
{% set t_min = state_attr(reset_entity,'min') %}
|
|
{% set t_max = state_attr(reset_entity,'max') %}
|
|
{% set step = state_attr(reset_entity,'step') %}
|
|
|
|
{% set temp_r = ((temp_r | float(0) / step) | round(0) * step) | float %}
|
|
{% set temp_r = iif(temp_r > t_max, t_max, temp_r) %}
|
|
{% set temp_r = iif(temp_r < t_min, t_min, temp_r) %}
|
|
{{ temp_r }}
|
|
- action: input_number.set_value
|
|
data:
|
|
value: "{{ reset_temp }}"
|
|
target:
|
|
entity_id: "{{ reset_entity }}"
|
|
|
|
- if:
|
|
- condition: and
|
|
conditions: !input input_custom_condition
|
|
- condition: template
|
|
value_template: "{{ changes | count | int > 0 and (not no_changes or (no_changes and state_window)) }}"
|
|
then:
|
|
- repeat:
|
|
count: "{{ changes | count | int }}"
|
|
sequence:
|
|
- variables:
|
|
index: "{{ repeat.index-1 }}"
|
|
thermostat: "{{ (changes.keys() | list) [index] }}"
|
|
mode: "{{ (((changes.values() | list) [index]) | first) ['mode'] }}"
|
|
temp_target: "{{ (((changes.values() | list) [index]) | first) ['temp'] }}"
|
|
- action: system_log.write
|
|
data:
|
|
message: >
|
|
AHC - Change - {{ automation_name }} {{" \n "}}
|
|
Trigger ID: {{ iif(trigger_id_defined, trigger.id, '') }}
|
|
Thermostat: {{ thermostat }} {{" \n "}}
|
|
Mode: {{ mode }} {{" \n "}}
|
|
New Target Temp: {{ temp_target }} {{" \n "}}
|
|
Current Target Temp: {{ state_attr(thermostat,'temperature') }}
|
|
level: !input input_log_level
|
|
logger: blueprints.panhans.heatingcontrol
|
|
|
|
- if:
|
|
- condition: template
|
|
value_template: "{{ states(thermostat) | lower != mode | lower }}"
|
|
then:
|
|
- action: climate.set_hvac_mode
|
|
data:
|
|
entity_id: "{{ thermostat }}"
|
|
hvac_mode: "{{ mode }}"
|
|
- delay: !input input_action_call_delay
|
|
|
|
- if:
|
|
- condition: template
|
|
value_template: "{{ state_attr(thermostat, 'temperature') != temp_target and mode != 'off' }}"
|
|
then:
|
|
- action: climate.set_temperature
|
|
data:
|
|
entity_id: "{{ thermostat }}"
|
|
temperature: "{{ temp_target | float }}"
|
|
- delay: !input input_action_call_delay
|
|
|
|
- if:
|
|
- condition: template
|
|
value_template: "{{ input_valve_positioning_mode != 'off' }}"
|
|
- condition: template
|
|
value_template: "{{ changes | count | int > 0 or is_scene_apply_trigger }}"
|
|
then:
|
|
- delay:
|
|
seconds: 10
|
|
- event: ahc_positioning_event
|
|
event_data:
|
|
automation: "{{ this.entity_id }}"
|
|
- delay: !input input_action_call_delay
|
|
|
|
# custom action
|
|
- if:
|
|
- condition: template
|
|
value_template: "{{ input_custom_action != none }}"
|
|
then: !input "input_custom_action"
|
|
|
|
mode: queued |