diff --git a/flows.json b/flows.json index 5ff6c87..9a6b1d8 100644 --- a/flows.json +++ b/flows.json @@ -2391,12 +2391,16 @@ "15b160d6bfe30b15", "03c668f84022cc23", "a273dadf7a256af1", - "8bb8dc720af76714" + "8bb8dc720af76714", + "bb9595faa1592fd9", + "ddef9ad0ac0a98a0", + "71b1c6e934b85637", + "334a79e6f5f3a16a" ], "x": 14, "y": 1319, "w": 1072, - "h": 502 + "h": 562 }, { "id": "a09584861c2fe558", @@ -5364,6 +5368,39 @@ "resend": false, "debugEnabled": false }, + { + "id": "3ce4b28fb7c060b0", + "type": "ha-entity-config", + "server": "9e87348d.9c1c48", + "deviceConfig": "47b0f8d963c82cce", + "name": "Emma Bedroom Climate Watchdog", + "version": 6, + "entityType": "switch", + "haConfig": [ + { + "property": "name", + "value": "Emma Bedroom Climate Watchdog" + }, + { + "property": "icon", + "value": "mdi:lock-check" + }, + { + "property": "entity_picture", + "value": "" + }, + { + "property": "entity_category", + "value": "" + }, + { + "property": "device_class", + "value": "" + } + ], + "resend": false, + "debugEnabled": false + }, { "id": "afead614fabc01bb", "type": "function", @@ -23363,7 +23400,7 @@ "z": "72f99805df043603", "g": "5802ea32c1de36e8", "name": "Processing", - "func": "node.log(\"Emma Bedroom Climate: Processing Started\");\n// pull in the necessary information\n\nconst states = global.get('homeassistant.homeAssistant.states')\nconst allowed = states['input_boolean.emma_bedroom_climate_protocol'].state\nconst ac = global.get('emmaBedroom.aircon.installed', \"diskCon\")\nconst temp = global.get(\"outdoorTemp.tempStr\")\nconst payload = msg.payload\nconst vacation = states['input_boolean.vacation_mode'].state\nconst dayTemp = states['input_number.emma_bedroom_daytime_temp'].state\nconst nightTemp = states['input_number.emma_bedroom_night_temp'].state\nconst bedTemp = states['input_number.emma_bedroom_bedtime_temp'].state\nconst schedMode = states['input_select.scheduled_climate_mode_emma_bedroom'].state\nconst sleeping = states['input_boolean.emma_sleeping'].state\nconst hotDay = states['input_boolean.hot_day'].state\nconst heatWarning = states[\"binary_sensor.heat_warning\"].state\nconst earlyNight = states[\"binary_sensor.early_night_mode\"].state\nconst danger = states['binary_sensor.heat_warning'].attributes.danger\nconst meltdown = states['input_boolean.meltdown_protocol'].state\nconst coolingActive = states['input_boolean.emma_bedroom_cooling_on'].state\nnode.log(\"Emma Bedroom Climate: Constants Set\")\n\n// Define reusable constants\n\nconst emmaBedroomAircon = [\"climate.emma_bedroom_aircon\"]\nconst emmaBedroomWhiteNoise = [\"input_boolean.white_noise_emma_bedroom\"]\nconst validSchedModesAC = ['AC', 'Fan']\n\n// Helper function to convert a string to title case\n\nfunction convertToTitleCase(str) {\n if (!str) {\n return \"\";\n }\n return str.toLowerCase().replace(/\\b\\w/g, (s) => s.toUpperCase());\n}\n\n// init variables\n\nlet setTemp = []\nlet setEco = []\nlet setHvac = []\nlet setCool = []\nlet setSleep = []\nlet setPeople = []\nlet setDisplay = []\nlet setWhiteNoise = \"turn_off\"\nlet time = []\nlet type = msg.type\nlet topic = msg.topic\nlet isWakeup = context.get(\"isWakeup\")\nnode.log(\"Emma Bedroom Climate: Variables Defined\")\n\n// Sleep Switch Handling\nif (type === 'sleep' && payload === 'off') {\n setDisplay = 'turn_on'\n setPeople = 'turn_on'\n if (coolingActive === 'on') {\n time = 'night'\n } else {\n time = 'day'\n }\n} else if (type === 'sleep' && payload === 'on') {\n setDisplay = 'turn_off'\n time = 'bedtime'\n} else {\n time = msg.time\n}\n\nif (topic === 'emmabedroom-wakeup') {\n setSleep = 'turn_off'\n}\n\n// Day Time\nif (time === 'day') {\n if (type === 'auto') {\n setCool = 'turn_off'\n }\n if (earlyNight === 'off') {\n if (ac === 'on') {\n if (danger === 'Extreme') {\n setTemp = nightTemp\n setEco = \"eco\"\n setHvac = \"cool\"\n } else if (hotDay === 'on' || heatWarning === 'on') {\n setTemp = dayTemp\n setEco = \"eco\"\n setHvac = \"cool\"\n } else {\n setTemp = nightTemp\n setEco = \"eco\"\n setHvac = \"off\"\n }\n }\n } else if (earlyNight === 'on') {\n if (ac === 'on') {\n if (danger === 'Extreme') {\n setTemp === bedTemp\n } else {\n setTemp = nightTemp\n }\n if (schedMode === 'AC') {\n setHvac = 'cool'\n } else if (schedMode === 'fan') {\n setHvac = 'fan_only'\n } else {\n setHvac = 'off'\n }\n if (hotDay === 'on') {\n setEco = 'none'\n } else {\n setEco = 'eco'\n }\n }\n }\n// Night Time\n} else if (time === 'night') {\n // If this is being run at scheduled time, turn on input_boolean.emma_bedroom_cooling_on\n if (type === 'auto') {\n setCool = 'turn_on'\n }\n // Decide temperature\n if (type === 'sleep' && payload === 'off') {\n setTemp = dayTemp\n } else if (danger === 'Extreme') {\n setTemp = bedTemp\n } else {\n setTemp = nightTemp\n }\n // Decide eco mode\n if (sleeping === 'on') {\n setEco = 'none'\n } else {\n setEco = 'eco'\n }\n // Decide HVAC mode\n if (schedMode === 'AC') {\n setHvac = 'cool'\n } else if (schedMode === 'Fan') {\n setHvac = 'fan_only'\n } else {\n setHvac = 'off'\n }\n// Bed Time\n} else if (time === 'bedtime') {\n setPeople = 'turn_off'\n if (ac === 'on') {\n setTemp = bedTemp\n setEco = 'none'\n if (schedMode === 'AC') {\n setHvac = 'cool'\n } else if (schedMode === 'Fan') {\n setHvac = \"fan_only\"\n } else if (schedMode === 'White Noise') {\n setWhiteNoise = 'turn_on'\n if (ac === 'on') {\n setHvac = 'off'\n }\n } else {\n setHvac = \"off\"\n }\n }\n}\nnode.log(\"Emma Bedroom Climate: Decision Logic Complete\")\n\n// Define message payloads\n\nlet sendCool = {\n \"payload\": {\n \"action\": `input_boolean.${setCool}`,\n \"target\": {\n \"entity_id\": [\"input_boolean.emma_bedroom_cooling_on\"]\n },\n \"data\": {}\n }\n}\n\nlet sendSleep = {\n \"payload\": {\n \"action\": `input_boolean.${setSleep}`,\n \"target\": {\n \"entity_id\": [\"input_boolean.emma_sleeping\"]\n },\n \"data\": {}\n }\n}\n\nlet sendPeople = {\n \"payload\": {\n \"action\": `input_boolean.${setPeople}`,\n \"target\": {\n \"entity_id\": [\"input_boolean.emma_awake\"]\n },\n \"data\": {}\n }\n}\n\nlet sendDisplay = {\n \"payload\": {\n \"action\": `switch.${setDisplay}`,\n \"target\": {\n \"entity_id\": [\"switch.emma_bedroom_aircon_display\"]\n },\n \"data\": {}\n }\n}\n\nlet sendWhiteNoise = {\n \"payload\": {\n \"action\": `input_boolean.${setWhiteNoise}`,\n \"target\": {\n \"entity_id\": emmaBedroomWhiteNoise\n },\n \"data\": {}\n }\n}\n\nlet notify = {\n \"topic\": topic,\n \"nighttemp\": nightTemp,\n \"schedMode\": schedMode\n}\n\nlet sendHvac = {\n \"payload\": {\n \"action\": \"climate.set_hvac_mode\",\n \"target\": {\n \"entity_id\": emmaBedroomAircon\n },\n \"data\": {\n \"hvac_mode\": setHvac\n }\n }\n}\n\nlet sendTemp = {\n \"payload\": {\n \"action\": \"climate.set_temperature\",\n \"target\": {\n \"entity_id\": emmaBedroomAircon\n },\n \"data\": {\n \"temperature\": setTemp\n }\n }\n}\n\nlet sendEco = {\n \"payload\": {\n \"action\": \"climate.set_preset_mode\",\n \"target\": {\n \"entity_id\": emmaBedroomAircon\n },\n \"data\": {\n \"preset_mode\": setEco\n }\n }\n}\n\nlet sendAcFan = {\n \"payload\": {\n \"action\": \"climate.set_fan_mode\",\n \"target\": {\n \"entity_id\": emmaBedroomAircon\n },\n \"data\": {\n \"fan_mode\": \"auto\"\n }\n }\n}\n\nnode.log(\"Emma Bedroom Climate: Message Payloads Defined\")\n\n// Log the parameters that were chosen, for debugging purposes\n\nnode.log(\"----- Emma Bedroom Climate: Set Parameters -----\")\nnode.log(`setTemp: ${setTemp}`)\nnode.log(`setEco: ${setEco}`)\nnode.log(`setHvac: ${setHvac}`)\nnode.log(`setCool: ${setCool}`)\nnode.log(`setSleep: ${setSleep}`)\nnode.log(`setPeople: ${setPeople}`)\nnode.log(`setDisplay: ${setDisplay}`)\nnode.log(`time: ${time}`)\nnode.log(`type: ${type}`)\nnode.log(`topic: ${topic}`)\nnode.log(\"----- Emma Bedroom Climate: End Parameters -----\")\n\n// If this was an automated trigger, set the cooling context for the bedroom accordingly.\n\nif (type === 'auto' && time != 'bedtime') {\n node.send([null, sendCool, null])\n node.log(\"Emma Bedroom Climate: Cooling Context Set\")\n}\n\nif (type === 'auto' && allowed === 'on' && meltdown === 'off' && vacation === 'off') {\n node.log(\"Emma Bedroom Climate: Auto\")\n if (sleeping === 'on' && topic != 'emmabedroom-wakeup') {\n node.status({ fill: \"red\", shape: \"ring\", text: \"Blocked (sleep mode)\" })\n node.log(\"Emma Bedroom Climate: Blocked (sleep mode)\")\n } else {\n if (topic === 'emmabedroom-cooling' && ac === 'on') {\n node.status({ fill: \"green\", shape: \"dot\", text: \"Cooling Schedule\" })\n node.send([[sendHvac, sendTemp, sendEco, sendAcFan], null, null])\n node.log(\"Emma Bedroom Climate: Auto/Cooling\")\n } else if (topic === 'emmabedroom-bedtime') {\n node.send([null, sendPeople, null])\n node.status({ fill: \"green\", shape: \"dot\", text: \"Bedtime\" })\n node.log(\"Emma Bedroom Climate: Auto/Bedtime\")\n if (validSchedModesAC.includes(schedMode) && ac === 'on') {\n node.send([[sendHvac, sendTemp, sendEco, sendAcFan], null, null])\n node.log(\"Emma Bedroom Climate: Auto/Bedtime/AC\")\n } else if (schedMode === 'White Noise') {\n node.send([null, sendWhiteNoise, null])\n node.log(\"Emma Bedroom Climate: Auto/Bedtime/White Noise\")\n }\n } else if (topic === 'emmabedroom-wakeup') {\n node.status({ fill: \"green\", shape: \"dot\", text: \"Wakeup Schedule\" })\n node.log(\"Emma Bedroom Climate: Auto/Wakeup\")\n if (sleeping === 'off') {\n context.set(\"isWakeup\", false)\n node.send([null, sendWhiteNoise, null])\n node.log(\"Emma Bedroom Climate: Auto/Wakeup/Sleep Off\")\n if (ac === 'on') {\n node.send([[sendHvac, sendTemp, sendEco, sendAcFan], null, null])\n node.log(\"Emma Bedroom Climate: Auto/Wakeup/AC On\")\n }\n } else if (sleeping === 'on') {\n context.set(\"isWakeup\", true)\n node.send([null, sendSleep, null])\n node.log(\"Emma Bedroom Climate: Auto/Wakeup/Sleep On\")\n }\n }\n }\n// Manual Responses\n} else if (type === 'manual') {\n node.log(\"Emma Bedroom Climate: Manual\")\n if (time === 'night') {\n node.status({ fill: \"blue\", shape: \"dot\", text: \"Manual Night\" })\n node.log(\"Emma Bedroom Climate: Manual/Night\")\n if (ac === 'on') {\n node.send([[sendHvac, sendTemp, sendEco, sendAcFan], null, null])\n node.log(\"Emma Bedroom Climate: Manual/Night/AC\")\n }\n } else if (time === 'day') {\n node.status({ fill: \"blue\", shape: \"dot\", text: \"Manual Day\" })\n node.send([null, sendWhiteNoise, null])\n node.log(\"Emma Bedroom Climate: Manual/Day\")\n if (ac === 'on') {\n node.send([[sendHvac, sendTemp, sendEco, sendAcFan], null, null])\n node.log(\"Emma Bedroom Climate: Manual/Day/AC\")\n }\n } else if (time === 'bedtime') {\n node.status({ fill: \"blue\", shape: \"dot\", text: \"Manual Bedtime\" })\n node.log(\"Emma Bedroom Climate: Manual/Bedtime\")\n if (validSchedModesAC.includes(schedMode) && ac === 'on') {\n node.send([[sendHvac, sendTemp, sendEco, sendAcFan], null, null])\n node.log(\"Emma Bedroom Climate: Manual/Bedtime/AC\")\n } else if (schedMode === 'White Noise') {\n node.send([null, sendWhiteNoise, null])\n node.log(\"Emma Bedroom Climate: Manual/Bedtime/White Noise\")\n }\n }\n context.set(\"isWakeup\", false)\n// Sleep Switch Responses\n} else if (type === 'sleep') {\n if (payload === 'off') {\n node.send([null, [sendPeople, sendWhiteNoise], null])\n }\n node.log(\"Emma Bedroom Climate: Sleep\")\n if (time === 'bedtime') {\n node.status({ fill: \"blue\", shape: \"dot\", text: \"Sleep\" })\n node.send([null, [sendPeople, sendWhiteNoise], null])\n node.log(\"Emma Bedroom Climate: Sleep/Bedtime\")\n if (ac === 'on') {\n node.send([[sendDisplay, sendHvac, sendTemp, sendEco, sendAcFan], null, null])\n node.log(\"Emma Bedroom Climate: Sleep/Bedtime/AC\")\n }\n } else {\n node.status({ fill: \"blue\", shape: \"dot\", text: \"Wakeup\" })\n node.send([null, sendWhiteNoise, null])\n node.log(\"Emma Bedroom Climate: Sleep/Day\")\n if (ac === 'on') {\n node.send([[sendDisplay, sendHvac, sendTemp, sendEco, sendAcFan], null, null])\n node.log(`Emma Bedroom Climate: Sleep/${convertToTitleCase(time)}/AC`)\n }\n }\n context.set(\"isWakeup\", false)\n} else if (meltdown === 'on') {\n node.status({ fill: \"red\", shape: \"ring\", text: \"Blocked (Meltdown Protocol)\" })\n node.log(\"Emma Bedroom Climate: Blocked (Meltdown Protocol)\")\n} else if (vacation === 'on') {\n node.status({ fill: \"red\", shape: \"ring\", text: \"Blocked (Vacation Mode)\" })\n node.log(\"Emma Bedroom Climate: Blocked (Vacation Mode)\")\n} else {\n node.status({ fill: \"red\", shape: \"ring\", text: \"Blocked (Automation Disabled)\" })\n node.log(\"Emma Bedroom Climate: Blocked (Automation Disabled)\")\n}\n\nnode.log(\"Emma Bedroom Climate: Processing Complete\")\n", + "func": "node.log(\"Emma Bedroom Climate: Processing Started\");\n// pull in the necessary information\n\nconst states = global.get('homeassistant.homeAssistant.states')\nconst allowed = states['input_boolean.emma_bedroom_climate_protocol'].state\nconst ac = global.get('emmaBedroom.aircon.installed', \"diskCon\")\nconst temp = global.get(\"outdoorTemp.tempStr\")\nconst payload = msg.payload\nconst vacation = states['input_boolean.vacation_mode'].state\nconst dayTemp = states['input_number.emma_bedroom_daytime_temp'].state\nconst nightTemp = states['input_number.emma_bedroom_night_temp'].state\nconst bedTemp = states['input_number.emma_bedroom_bedtime_temp'].state\nconst schedMode = states['input_select.scheduled_climate_mode_emma_bedroom'].state\nconst sleeping = states['input_boolean.emma_sleeping'].state\nconst hotDay = states['input_boolean.hot_day'].state\nconst heatWarning = states[\"binary_sensor.heat_warning\"].state\nconst earlyNight = states[\"binary_sensor.early_night_mode\"].state\nconst danger = states['binary_sensor.heat_warning'].attributes.danger\nconst meltdown = states['input_boolean.meltdown_protocol'].state\nconst coolingActive = states['input_boolean.emma_bedroom_cooling_on'].state\nnode.log(\"Emma Bedroom Climate: Constants Set\")\n\n// Define reusable constants\n\nconst emmaBedroomAircon = [\"climate.emma_bedroom_aircon\"]\nconst emmaBedroomWhiteNoise = [\"input_boolean.white_noise_emma_bedroom\"]\nconst validSchedModesAC = ['AC', 'Fan']\n\n// Helper function to convert a string to title case\n\nfunction convertToTitleCase(str) {\n if (!str) {\n return \"\";\n }\n return str.toLowerCase().replace(/\\b\\w/g, (s) => s.toUpperCase());\n}\n\n// init variables\n\nlet setTemp = []\nlet setEco = []\nlet setHvac = []\nlet setCool = []\nlet setSleep = []\nlet setPeople = []\nlet setDisplay = []\nlet setWhiteNoise = \"turn_off\"\nlet time = []\nlet type = msg.type\nlet topic = msg.topic\nlet isWakeup = context.get(\"isWakeup\")\nnode.log(\"Emma Bedroom Climate: Variables Defined\")\n\n// Sleep Switch Handling\nif (type === 'sleep' && payload === 'off') {\n setDisplay = 'turn_on'\n setPeople = 'turn_on'\n if (coolingActive === 'on') {\n time = 'night'\n } else {\n time = 'day'\n }\n} else if (type === 'sleep' && payload === 'on') {\n setDisplay = 'turn_off'\n time = 'bedtime'\n} else {\n time = msg.time\n}\n\nif (topic === 'emmabedroom-wakeup') {\n setSleep = 'turn_off'\n}\n\n// Day Time\nif (time === 'day') {\n if (type === 'auto') {\n setCool = 'turn_off'\n }\n if (earlyNight === 'off') {\n if (ac === 'on') {\n if (danger === 'Extreme') {\n setTemp = nightTemp\n setEco = \"eco\"\n setHvac = \"cool\"\n } else if (hotDay === 'on' || heatWarning === 'on') {\n setTemp = dayTemp\n setEco = \"eco\"\n setHvac = \"cool\"\n } else {\n setTemp = nightTemp\n setEco = \"eco\"\n setHvac = \"off\"\n }\n }\n } else if (earlyNight === 'on') {\n if (ac === 'on') {\n if (danger === 'Extreme') {\n setTemp === bedTemp\n } else {\n setTemp = nightTemp\n }\n if (schedMode === 'AC') {\n setHvac = 'cool'\n } else if (schedMode === 'fan') {\n setHvac = 'fan_only'\n } else {\n setHvac = 'off'\n }\n if (hotDay === 'on') {\n setEco = 'none'\n } else {\n setEco = 'eco'\n }\n }\n }\n// Night Time\n} else if (time === 'night') {\n // If this is being run at scheduled time, turn on input_boolean.emma_bedroom_cooling_on\n if (type === 'auto') {\n setCool = 'turn_on'\n }\n // Decide temperature\n if (type === 'sleep' && payload === 'off') {\n setTemp = dayTemp\n } else if (danger === 'Extreme') {\n setTemp = bedTemp\n } else {\n setTemp = nightTemp\n }\n // Decide eco mode\n if (sleeping === 'on') {\n setEco = 'none'\n } else {\n setEco = 'eco'\n }\n // Decide HVAC mode\n if (schedMode === 'AC') {\n setHvac = 'cool'\n } else if (schedMode === 'Fan') {\n setHvac = 'fan_only'\n } else {\n setHvac = 'off'\n }\n// Bed Time\n} else if (time === 'bedtime') {\n setPeople = 'turn_off'\n if (ac === 'on') {\n setTemp = bedTemp\n setEco = 'none'\n if (schedMode === 'AC') {\n setHvac = 'cool'\n } else if (schedMode === 'Fan') {\n setHvac = \"fan_only\"\n } else if (schedMode === 'White Noise') {\n setWhiteNoise = 'turn_on'\n if (ac === 'on') {\n setHvac = 'off'\n }\n } else {\n setHvac = \"off\"\n }\n }\n}\nnode.log(\"Emma Bedroom Climate: Decision Logic Complete\")\n\n// Define message payloads\n\nlet sendCool = {\n \"payload\": {\n \"action\": `input_boolean.${setCool}`,\n \"target\": {\n \"entity_id\": [\"input_boolean.emma_bedroom_cooling_on\"]\n },\n \"data\": {}\n }\n}\n\nlet sendSleep = {\n \"payload\": {\n \"action\": `input_boolean.${setSleep}`,\n \"target\": {\n \"entity_id\": [\"input_boolean.emma_sleeping\"]\n },\n \"data\": {}\n }\n}\n\nlet sendPeople = {\n \"payload\": {\n \"action\": `input_boolean.${setPeople}`,\n \"target\": {\n \"entity_id\": [\"input_boolean.emma_awake\"]\n },\n \"data\": {}\n }\n}\n\nlet sendDisplay = {\n \"payload\": {\n \"action\": `switch.${setDisplay}`,\n \"target\": {\n \"entity_id\": [\"switch.emma_bedroom_aircon_display\"]\n },\n \"data\": {}\n }\n}\n\nlet sendWhiteNoise = {\n \"payload\": {\n \"action\": `input_boolean.${setWhiteNoise}`,\n \"target\": {\n \"entity_id\": emmaBedroomWhiteNoise\n },\n \"data\": {}\n }\n}\n\nlet notify = {\n \"topic\": topic,\n \"nighttemp\": nightTemp,\n \"schedMode\": schedMode\n}\n\nlet sendHvac = {\n \"payload\": {\n \"action\": \"climate.set_hvac_mode\",\n \"target\": {\n \"entity_id\": emmaBedroomAircon\n },\n \"data\": {\n \"hvac_mode\": setHvac\n }\n }\n}\n\nlet sendTemp = {\n \"payload\": {\n \"action\": \"climate.set_temperature\",\n \"target\": {\n \"entity_id\": emmaBedroomAircon\n },\n \"data\": {\n \"temperature\": setTemp\n }\n }\n}\n\nlet sendEco = {\n \"payload\": {\n \"action\": \"climate.set_preset_mode\",\n \"target\": {\n \"entity_id\": emmaBedroomAircon\n },\n \"data\": {\n \"preset_mode\": setEco\n }\n }\n}\n\nlet sendAcFan = {\n \"payload\": {\n \"action\": \"climate.set_fan_mode\",\n \"target\": {\n \"entity_id\": emmaBedroomAircon\n },\n \"data\": {\n \"fan_mode\": \"auto\"\n }\n }\n}\n\nnode.log(\"Emma Bedroom Climate: Message Payloads Defined\")\n\n// Send some parameters to flow context for use in other nodes\n\nif (setHvac.length > 0) {\n flow.set('emmaBedroom.airconHvacMode',setHvac,'diskCon')\n}\n\nif (setEco.length > 0) {\n flow.set('emmaBedroom.airconPreset',setEco,'diskCon')\n}\n\nif (setTemp.length > 0) {\n flow.set('emmaBedroom.airconTargetTemp',setTemp,'diskCon')\n}\n\nif (setDisplay.length > 0) {\n flow.set('emmaBedroom.airconDisplay',setDisplay,'diskCon')\n}\n\nflow.set('emmaBedroom.airconFanMode','auto','diskCon')\n\n// Log the parameters that were chosen, for debugging purposes\n\nnode.log(\"----- Emma Bedroom Climate: Set Parameters -----\")\nnode.log(`setTemp: ${setTemp}`)\nnode.log(`setEco: ${setEco}`)\nnode.log(`setHvac: ${setHvac}`)\nnode.log(`setCool: ${setCool}`)\nnode.log(`setSleep: ${setSleep}`)\nnode.log(`setPeople: ${setPeople}`)\nnode.log(`setDisplay: ${setDisplay}`)\nnode.log(`time: ${time}`)\nnode.log(`type: ${type}`)\nnode.log(`topic: ${topic}`)\nnode.log(\"----- Emma Bedroom Climate: End Parameters -----\")\n\n// If this was an automated trigger, set the cooling context for the bedroom accordingly.\n\nif (type === 'auto' && time != 'bedtime') {\n node.send([null, sendCool, null])\n node.log(\"Emma Bedroom Climate: Cooling Context Set\")\n}\n\nif (type === 'auto' && allowed === 'on' && meltdown === 'off' && vacation === 'off') {\n node.log(\"Emma Bedroom Climate: Auto\")\n if (sleeping === 'on' && topic != 'emmabedroom-wakeup') {\n node.status({ fill: \"red\", shape: \"ring\", text: \"Blocked (sleep mode)\" })\n node.log(\"Emma Bedroom Climate: Blocked (sleep mode)\")\n } else {\n if (topic === 'emmabedroom-cooling' && ac === 'on') {\n node.status({ fill: \"green\", shape: \"dot\", text: \"Cooling Schedule\" })\n node.send([[sendHvac, sendTemp, sendEco, sendAcFan], null, null])\n node.log(\"Emma Bedroom Climate: Auto/Cooling\")\n } else if (topic === 'emmabedroom-bedtime') {\n node.send([null, sendPeople, null])\n node.status({ fill: \"green\", shape: \"dot\", text: \"Bedtime\" })\n node.log(\"Emma Bedroom Climate: Auto/Bedtime\")\n if (validSchedModesAC.includes(schedMode) && ac === 'on') {\n node.send([[sendHvac, sendTemp, sendEco, sendAcFan], null, null])\n node.log(\"Emma Bedroom Climate: Auto/Bedtime/AC\")\n } else if (schedMode === 'White Noise') {\n node.send([null, sendWhiteNoise, null])\n node.log(\"Emma Bedroom Climate: Auto/Bedtime/White Noise\")\n }\n } else if (topic === 'emmabedroom-wakeup') {\n node.status({ fill: \"green\", shape: \"dot\", text: \"Wakeup Schedule\" })\n node.log(\"Emma Bedroom Climate: Auto/Wakeup\")\n if (sleeping === 'off') {\n context.set(\"isWakeup\", false)\n node.send([null, sendWhiteNoise, null])\n node.log(\"Emma Bedroom Climate: Auto/Wakeup/Sleep Off\")\n if (ac === 'on') {\n node.send([[sendHvac, sendTemp, sendEco, sendAcFan], null, null])\n node.log(\"Emma Bedroom Climate: Auto/Wakeup/AC On\")\n }\n } else if (sleeping === 'on') {\n context.set(\"isWakeup\", true)\n node.send([null, sendSleep, null])\n node.log(\"Emma Bedroom Climate: Auto/Wakeup/Sleep On\")\n }\n }\n }\n// Manual Responses\n} else if (type === 'manual') {\n node.log(\"Emma Bedroom Climate: Manual\")\n if (time === 'night') {\n node.status({ fill: \"blue\", shape: \"dot\", text: \"Manual Night\" })\n node.log(\"Emma Bedroom Climate: Manual/Night\")\n if (ac === 'on') {\n node.send([[sendHvac, sendTemp, sendEco, sendAcFan], null, null])\n node.log(\"Emma Bedroom Climate: Manual/Night/AC\")\n }\n } else if (time === 'day') {\n node.status({ fill: \"blue\", shape: \"dot\", text: \"Manual Day\" })\n node.send([null, sendWhiteNoise, null])\n node.log(\"Emma Bedroom Climate: Manual/Day\")\n if (ac === 'on') {\n node.send([[sendHvac, sendTemp, sendEco, sendAcFan], null, null])\n node.log(\"Emma Bedroom Climate: Manual/Day/AC\")\n }\n } else if (time === 'bedtime') {\n node.status({ fill: \"blue\", shape: \"dot\", text: \"Manual Bedtime\" })\n node.log(\"Emma Bedroom Climate: Manual/Bedtime\")\n if (validSchedModesAC.includes(schedMode) && ac === 'on') {\n node.send([[sendHvac, sendTemp, sendEco, sendAcFan], null, null])\n node.log(\"Emma Bedroom Climate: Manual/Bedtime/AC\")\n } else if (schedMode === 'White Noise') {\n node.send([null, sendWhiteNoise, null])\n node.log(\"Emma Bedroom Climate: Manual/Bedtime/White Noise\")\n }\n }\n context.set(\"isWakeup\", false)\n// Sleep Switch Responses\n} else if (type === 'sleep') {\n if (payload === 'off') {\n node.send([null, [sendPeople, sendWhiteNoise], null])\n }\n node.log(\"Emma Bedroom Climate: Sleep\")\n if (time === 'bedtime') {\n node.status({ fill: \"blue\", shape: \"dot\", text: \"Sleep\" })\n node.send([null, [sendPeople, sendWhiteNoise], null])\n node.log(\"Emma Bedroom Climate: Sleep/Bedtime\")\n if (ac === 'on') {\n node.send([[sendDisplay, sendHvac, sendTemp, sendEco, sendAcFan], null, null])\n node.log(\"Emma Bedroom Climate: Sleep/Bedtime/AC\")\n }\n } else {\n node.status({ fill: \"blue\", shape: \"dot\", text: \"Wakeup\" })\n node.send([null, sendWhiteNoise, null])\n node.log(\"Emma Bedroom Climate: Sleep/Day\")\n if (ac === 'on') {\n node.send([[sendDisplay, sendHvac, sendTemp, sendEco, sendAcFan], null, null])\n node.log(`Emma Bedroom Climate: Sleep/${convertToTitleCase(time)}/AC`)\n }\n }\n context.set(\"isWakeup\", false)\n} else if (meltdown === 'on') {\n node.status({ fill: \"red\", shape: \"ring\", text: \"Blocked (Meltdown Protocol)\" })\n node.log(\"Emma Bedroom Climate: Blocked (Meltdown Protocol)\")\n} else if (vacation === 'on') {\n node.status({ fill: \"red\", shape: \"ring\", text: \"Blocked (Vacation Mode)\" })\n node.log(\"Emma Bedroom Climate: Blocked (Vacation Mode)\")\n} else {\n node.status({ fill: \"red\", shape: \"ring\", text: \"Blocked (Automation Disabled)\" })\n node.log(\"Emma Bedroom Climate: Blocked (Automation Disabled)\")\n}\n\nnode.log(\"Emma Bedroom Climate: Processing Complete\")\n", "outputs": 3, "timeout": "", "noerr": 0, @@ -23417,7 +23454,9 @@ "x": 650, "y": 1480, "wires": [ - [] + [ + "bb9595faa1592fd9" + ] ] }, { @@ -23997,6 +24036,105 @@ "y": 660, "wires": [] }, + { + "id": "bb9595faa1592fd9", + "type": "debug", + "z": "72f99805df043603", + "g": "5802ea32c1de36e8", + "name": "AC Settings Debug", + "active": false, + "tosidebar": true, + "console": false, + "tostatus": false, + "complete": "true", + "targetType": "full", + "statusVal": "", + "statusType": "auto", + "x": 710, + "y": 1360, + "wires": [] + }, + { + "id": "ddef9ad0ac0a98a0", + "type": "poll-state", + "z": "72f99805df043603", + "g": "5802ea32c1de36e8", + "name": "Poll State", + "server": "9e87348d.9c1c48", + "version": 3, + "exposeAsEntityConfig": "3ce4b28fb7c060b0", + "updateInterval": "60", + "updateIntervalType": "num", + "updateIntervalUnits": "seconds", + "outputInitially": false, + "outputOnChanged": true, + "entityId": "climate.emma_bedroom_aircon", + "stateType": "str", + "ifState": "", + "ifStateType": "str", + "ifStateOperator": "is", + "outputs": 1, + "outputProperties": [ + { + "property": "payload", + "propertyType": "msg", + "value": "", + "valueType": "entity" + }, + { + "property": "topic", + "propertyType": "msg", + "value": "", + "valueType": "triggerId" + } + ], + "x": 200, + "y": 1840, + "wires": [ + [ + "71b1c6e934b85637" + ] + ] + }, + { + "id": "71b1c6e934b85637", + "type": "function", + "z": "72f99805df043603", + "g": "5802ea32c1de36e8", + "name": "Watchdog", + "func": "const airconState = msg.payload.state\nconst airconAttributes = msg.payload.attributes\nconst airconPreset = airconAttributes['preset_mode']\nconst airconEco = airconAttributes['eco_mode']\nconst airconTargetTemp = airconAttributes['temperature']\nconst airconFanMode = airconAttributes['fan_mode']\nconst airconDisplay = airconAttributes['screen_display']\n\nmsg.payload = {\n \"airconState\": airconState,\n \"airconPreset\": airconPreset,\n \"airconEco\": airconEco,\n \"airconTargetTemp\": airconTargetTemp,\n \"airconFanMode\": airconFanMode,\n \"airconDisplay\": airconDisplay\n}\n\nreturn msg", + "outputs": 1, + "timeout": 0, + "noerr": 0, + "initialize": "", + "finalize": "", + "libs": [], + "x": 480, + "y": 1840, + "wires": [ + [ + "334a79e6f5f3a16a" + ] + ] + }, + { + "id": "334a79e6f5f3a16a", + "type": "debug", + "z": "72f99805df043603", + "g": "5802ea32c1de36e8", + "name": "Watchdog Debug", + "active": false, + "tosidebar": true, + "console": false, + "tostatus": false, + "complete": "true", + "targetType": "full", + "statusVal": "", + "statusType": "auto", + "x": 670, + "y": 1840, + "wires": [] + }, { "id": "48b8af9419ed18d6", "type": "subflow:384eaf2f63dcb5db",