mirror of
https://github.com/hyperion-project/hyperion.ng.git
synced 2025-03-01 10:33:28 +00:00
Add Home Assistant Lights support (#1763)
* New HomeAssistant LEDDevice * Fix typos * Ping Qt for Windows to 6.7 until aqtinstaller is fixed * Fix HA default port handling * HA - Update default latchtime and range * Add HA Wizard and light selection * Naming consistency * Fix "Selected Hyperion instance is not running" * CodeQL findings * HA - allow to overwrite brightness by HA yes or no * HA - Support switch off on black * HA - Add transition time
This commit is contained in:
parent
df2b2b2372
commit
4f1b95ec83
1
.github/workflows/qt5_6.yml
vendored
1
.github/workflows/qt5_6.yml
vendored
@ -194,7 +194,6 @@ jobs:
|
|||||||
version: ${{ inputs.qt_version == '6' && '6.7' || '5.15.*' }}
|
version: ${{ inputs.qt_version == '6' && '6.7' || '5.15.*' }}
|
||||||
target: 'desktop'
|
target: 'desktop'
|
||||||
modules: ${{ inputs.qt_version == '6' && 'qtserialport' || '' }}
|
modules: ${{ inputs.qt_version == '6' && 'qtserialport' || '' }}
|
||||||
arch: 'win64_msvc2019_64'
|
|
||||||
cache: 'true'
|
cache: 'true'
|
||||||
cache-key-prefix: 'cache-qt-windows'
|
cache-key-prefix: 'cache-qt-windows'
|
||||||
|
|
||||||
|
@ -85,6 +85,7 @@
|
|||||||
"conf_leds_layout_cl_bottomleft": "Bottom Left (Corner)",
|
"conf_leds_layout_cl_bottomleft": "Bottom Left (Corner)",
|
||||||
"conf_leds_layout_cl_bottomright": "Bottom Right (Corner)",
|
"conf_leds_layout_cl_bottomright": "Bottom Right (Corner)",
|
||||||
"conf_leds_layout_cl_cornergap": "Corner Gap",
|
"conf_leds_layout_cl_cornergap": "Corner Gap",
|
||||||
|
"conf_leds_layout_cl_disabled": "Deactivated",
|
||||||
"conf_leds_layout_cl_edgegap": "Edge Gap",
|
"conf_leds_layout_cl_edgegap": "Edge Gap",
|
||||||
"conf_leds_layout_cl_entertainment": "Entertainment Area",
|
"conf_leds_layout_cl_entertainment": "Entertainment Area",
|
||||||
"conf_leds_layout_cl_entertainment_center": "Entertainment Area Center",
|
"conf_leds_layout_cl_entertainment_center": "Entertainment Area Center",
|
||||||
@ -103,6 +104,7 @@
|
|||||||
"conf_leds_layout_cl_lightPosBottomLeft112": "Bottom: 0 - 50% from Left",
|
"conf_leds_layout_cl_lightPosBottomLeft112": "Bottom: 0 - 50% from Left",
|
||||||
"conf_leds_layout_cl_lightPosBottomLeft121": "Bottom: 50 - 100% from Left",
|
"conf_leds_layout_cl_lightPosBottomLeft121": "Bottom: 50 - 100% from Left",
|
||||||
"conf_leds_layout_cl_lightPosBottomLeftNewMid": "Bottom: 25 - 75% from Left",
|
"conf_leds_layout_cl_lightPosBottomLeftNewMid": "Bottom: 25 - 75% from Left",
|
||||||
|
"conf_leds_layout_cl_lightPosEntire": "Whole picture",
|
||||||
"conf_leds_layout_cl_lightPosTopLeft112": "Top: 0 - 50% from Left",
|
"conf_leds_layout_cl_lightPosTopLeft112": "Top: 0 - 50% from Left",
|
||||||
"conf_leds_layout_cl_lightPosTopLeft121": "Top: 50 - 100% from Left",
|
"conf_leds_layout_cl_lightPosTopLeft121": "Top: 50 - 100% from Left",
|
||||||
"conf_leds_layout_cl_lightPosTopLeftNewMid": "Top: 25 - 75% from Left",
|
"conf_leds_layout_cl_lightPosTopLeftNewMid": "Top: 25 - 75% from Left",
|
||||||
@ -661,13 +663,14 @@
|
|||||||
"edt_dev_spec_colorComponent_title": "Colour component",
|
"edt_dev_spec_colorComponent_title": "Colour component",
|
||||||
"edt_dev_spec_debugLevel_title": "Debug Level",
|
"edt_dev_spec_debugLevel_title": "Debug Level",
|
||||||
"edt_dev_spec_delayAfterConnect_title": "Delay after connect",
|
"edt_dev_spec_delayAfterConnect_title": "Delay after connect",
|
||||||
"edt_dev_spec_devices_discovered_none": "No Devices Discovered",
|
"edt_dev_spec_devices_discovered_none": "No Devices discovered",
|
||||||
"edt_dev_spec_devices_discovered_title": "Devices Discovered",
|
"edt_dev_spec_devices_discovered_title": "Devices discovered",
|
||||||
"edt_dev_spec_devices_discovered_title_info": "Select your LED-Device discovered",
|
"edt_dev_spec_devices_discovered_title_info": "Select your LED-Device discovered",
|
||||||
"edt_dev_spec_devices_discovered_title_info_custom": "Select your LED-Device discovered or configure a custome one",
|
"edt_dev_spec_devices_discovered_title_info_custom": "Select your LED-Device discovered or configure a custome one",
|
||||||
"edt_dev_spec_devices_discovery_inprogress": "Discovery in progress",
|
"edt_dev_spec_devices_discovery_inprogress": "Discovery in progress",
|
||||||
"edt_dev_spec_dithering_title": "Dithering",
|
"edt_dev_spec_dithering_title": "Dithering",
|
||||||
"edt_dev_spec_dmaNumber_title": "DMA channel",
|
"edt_dev_spec_dmaNumber_title": "DMA channel",
|
||||||
|
"edt_dev_spec_fullBrightnessAtStart_title": "Full brightness at start",
|
||||||
"edt_dev_spec_gamma_title": "Gamma",
|
"edt_dev_spec_gamma_title": "Gamma",
|
||||||
"edt_dev_spec_globalBrightnessControlMaxLevel_title": "Max Current Level",
|
"edt_dev_spec_globalBrightnessControlMaxLevel_title": "Max Current Level",
|
||||||
"edt_dev_spec_globalBrightnessControlThreshold_title": "Adaptive Current Threshold",
|
"edt_dev_spec_globalBrightnessControlThreshold_title": "Adaptive Current Threshold",
|
||||||
@ -685,6 +688,7 @@
|
|||||||
"edt_dev_spec_ledType_title": "LED Type",
|
"edt_dev_spec_ledType_title": "LED Type",
|
||||||
"edt_dev_spec_lightid_itemtitle": "ID",
|
"edt_dev_spec_lightid_itemtitle": "ID",
|
||||||
"edt_dev_spec_lightid_title": "Light ID(s)",
|
"edt_dev_spec_lightid_title": "Light ID(s)",
|
||||||
|
"edt_dev_spec_lights_discovered_none": "No Lights discovered",
|
||||||
"edt_dev_spec_lights_itemtitle": "Light",
|
"edt_dev_spec_lights_itemtitle": "Light",
|
||||||
"edt_dev_spec_lights_name": "Name",
|
"edt_dev_spec_lights_name": "Name",
|
||||||
"edt_dev_spec_lights_title": "Light(s)",
|
"edt_dev_spec_lights_title": "Light(s)",
|
||||||
@ -1184,9 +1188,10 @@
|
|||||||
"wiz_identify_tip": "Identify configured device by lighting it up",
|
"wiz_identify_tip": "Identify configured device by lighting it up",
|
||||||
"wiz_identify_light": "Identify $1",
|
"wiz_identify_light": "Identify $1",
|
||||||
"wiz_layout": "Generate Layout",
|
"wiz_layout": "Generate Layout",
|
||||||
|
"wiz_layout_led_position_title": "LED position",
|
||||||
|
"wiz_layout_led_positions_title": "LED position layout wizard",
|
||||||
|
"wiz_layout_led_positions_expl": "Select the LED position for the $1 controller lights.",
|
||||||
"wiz_layout_tip": "Generate a layout for the configured device",
|
"wiz_layout_tip": "Generate a layout for the configured device",
|
||||||
"wiz_ids_disabled": "Deactivated",
|
|
||||||
"wiz_ids_entire": "Whole picture",
|
|
||||||
"wiz_nanoleaf_failure_auth_token": "Please press the Nanoleaf Power On/Off button within 30 seconds",
|
"wiz_nanoleaf_failure_auth_token": "Please press the Nanoleaf Power On/Off button within 30 seconds",
|
||||||
"wiz_nanoleaf_failure_auth_token_t": "User authorization token generating timeout",
|
"wiz_nanoleaf_failure_auth_token_t": "User authorization token generating timeout",
|
||||||
"wiz_nanoleaf_press_onoff_button": "Please press the Power On/Off button on your Nanoleaf device for 5-7 seconds",
|
"wiz_nanoleaf_press_onoff_button": "Please press the Power On/Off button on your Nanoleaf device for 5-7 seconds",
|
||||||
|
@ -197,7 +197,7 @@ $(document).ready(function () {
|
|||||||
removeStorage("loginToken");
|
removeStorage("loginToken");
|
||||||
requestRequiresDefaultPasswortChange();
|
requestRequiresDefaultPasswortChange();
|
||||||
}
|
}
|
||||||
else if (event.reason == "Selected Hyperion instance isn't running") {
|
else if (event.reason == "Selected Hyperion instance is not running") {
|
||||||
//Switch to default instance
|
//Switch to default instance
|
||||||
instanceSwitch(0);
|
instanceSwitch(0);
|
||||||
} else {
|
} else {
|
||||||
|
@ -22,7 +22,7 @@ var devSPI = ['apa102', 'apa104', 'ws2801', 'lpd6803', 'lpd8806', 'p9813', 'sk68
|
|||||||
var devFTDI = ['apa102_ftdi', 'sk6812_ftdi', 'ws2812_ftdi'];
|
var devFTDI = ['apa102_ftdi', 'sk6812_ftdi', 'ws2812_ftdi'];
|
||||||
var devRPiPWM = ['ws281x'];
|
var devRPiPWM = ['ws281x'];
|
||||||
var devRPiGPIO = ['piblaster'];
|
var devRPiGPIO = ['piblaster'];
|
||||||
var devNET = ['atmoorb', 'cololight', 'fadecandy', 'philipshue', 'nanoleaf', 'razer', 'tinkerforge', 'tpm2net', 'udpe131', 'udpartnet', 'udpddp', 'udph801', 'udpraw', 'wled', 'yeelight'];
|
var devNET = ['atmoorb', 'cololight', 'fadecandy', 'homeassistant', 'philipshue', 'nanoleaf', 'razer', 'tinkerforge', 'tpm2net', 'udpe131', 'udpartnet', 'udpddp', 'udph801', 'udpraw', 'wled', 'yeelight'];
|
||||||
var devSerial = ['adalight', 'dmx', 'atmo', 'sedu', 'tpm2', 'karate'];
|
var devSerial = ['adalight', 'dmx', 'atmo', 'sedu', 'tpm2', 'karate'];
|
||||||
var devHID = ['hyperionusbasp', 'lightpack', 'paintpack', 'rawhid'];
|
var devHID = ['hyperionusbasp', 'lightpack', 'paintpack', 'rawhid'];
|
||||||
|
|
||||||
@ -1100,6 +1100,7 @@ $(document).ready(function () {
|
|||||||
switch (ledType) {
|
switch (ledType) {
|
||||||
case "wled":
|
case "wled":
|
||||||
case "cololight":
|
case "cololight":
|
||||||
|
case "homeassistant":
|
||||||
case "nanoleaf":
|
case "nanoleaf":
|
||||||
showAllDeviceInputOptions("hostList", false);
|
showAllDeviceInputOptions("hostList", false);
|
||||||
case "apa102":
|
case "apa102":
|
||||||
@ -1279,7 +1280,21 @@ $(document).ready(function () {
|
|||||||
if (hostList !== "SELECT") {
|
if (hostList !== "SELECT") {
|
||||||
const host = conf_editor.getEditor("root.specificOptions.host").getValue();
|
const host = conf_editor.getEditor("root.specificOptions.host").getValue();
|
||||||
const token = conf_editor.getEditor("root.specificOptions.token").getValue();
|
const token = conf_editor.getEditor("root.specificOptions.token").getValue();
|
||||||
if (host !== "" && token !== "") {
|
if (host !== "" && token !== "" && entityIds) {
|
||||||
|
canIdentify = true;
|
||||||
|
canSave = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case "homeassistant": {
|
||||||
|
const hostList = conf_editor.getEditor("root.specificOptions.hostList").getValue();
|
||||||
|
if (hostList !== "SELECT") {
|
||||||
|
const host = conf_editor.getEditor("root.specificOptions.host").getValue();
|
||||||
|
const token = conf_editor.getEditor("root.specificOptions.token").getValue();
|
||||||
|
const entityIds = conf_editor.getEditor("root.specificOptions.entityIds").getValue();
|
||||||
|
if (host !== "" && token !== "" && entityIds) {
|
||||||
canIdentify = true;
|
canIdentify = true;
|
||||||
canSave = true;
|
canSave = true;
|
||||||
}
|
}
|
||||||
@ -1387,6 +1402,16 @@ $(document).ready(function () {
|
|||||||
getProperties_device(ledType, host, params);
|
getProperties_device(ledType, host, params);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case "homeassistant":
|
||||||
|
var token = conf_editor.getEditor("root.specificOptions.token").getValue();
|
||||||
|
if (token === "") {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
params = { host: host, token: token, filter: "states" };
|
||||||
|
getProperties_device(ledType, host, params);
|
||||||
|
break;
|
||||||
|
|
||||||
case "nanoleaf":
|
case "nanoleaf":
|
||||||
$('#btn_wiz_holder').show();
|
$('#btn_wiz_holder').show();
|
||||||
|
|
||||||
@ -1552,6 +1577,14 @@ $(document).ready(function () {
|
|||||||
|
|
||||||
var host = "";
|
var host = "";
|
||||||
switch (ledType) {
|
switch (ledType) {
|
||||||
|
case "homeassistant":
|
||||||
|
host = conf_editor.getEditor("root.specificOptions.host").getValue();
|
||||||
|
if (host === "") {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
params = { host: host, token: token, filter: "states" };
|
||||||
|
break;
|
||||||
|
|
||||||
case "nanoleaf":
|
case "nanoleaf":
|
||||||
host = conf_editor.getEditor("root.specificOptions.host").getValue();
|
host = conf_editor.getEditor("root.specificOptions.host").getValue();
|
||||||
if (host === "") {
|
if (host === "") {
|
||||||
@ -1654,6 +1687,16 @@ $(document).ready(function () {
|
|||||||
default:
|
default:
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
conf_editor.watch('root.specificOptions.entityIds', () => {
|
||||||
|
var entityIds = conf_editor.getEditor("root.specificOptions.entityIds").getValue();
|
||||||
|
if (entityIds.length > 0) {
|
||||||
|
$('#btn_test_controller').prop('disabled', false);
|
||||||
|
} else {
|
||||||
|
$('#btn_test_controller').prop('disabled', true);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|
||||||
//philipshueentertainment backward fix
|
//philipshueentertainment backward fix
|
||||||
@ -1684,7 +1727,7 @@ $(document).ready(function () {
|
|||||||
else if ($.inArray(ledDevices[idx], devHID) != -1)
|
else if ($.inArray(ledDevices[idx], devHID) != -1)
|
||||||
optArr[4].push(ledDevices[idx]);
|
optArr[4].push(ledDevices[idx]);
|
||||||
else if (ledDevices[idx].endsWith("_ftdi")) {
|
else if (ledDevices[idx].endsWith("_ftdi")) {
|
||||||
var title = ledDevices[idx].replace('_ftdi','');
|
var title = ledDevices[idx].replace('_ftdi', '');
|
||||||
optArr[5].push(ledDevices[idx] + ":" + title);
|
optArr[5].push(ledDevices[idx] + ":" + title);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@ -1744,6 +1787,13 @@ $(document).ready(function () {
|
|||||||
params = { host: host };
|
params = { host: host };
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case "homeassistant":
|
||||||
|
var host = conf_editor.getEditor("root.specificOptions.host").getValue();
|
||||||
|
var token = conf_editor.getEditor("root.specificOptions.token").getValue();
|
||||||
|
const entityIds = conf_editor.getEditor("root.specificOptions.entityIds").getValue();
|
||||||
|
params = { host: host, token: token, entity_id: entityIds };
|
||||||
|
break;
|
||||||
|
|
||||||
case "nanoleaf":
|
case "nanoleaf":
|
||||||
var host = conf_editor.getEditor("root.specificOptions.host").getValue();
|
var host = conf_editor.getEditor("root.specificOptions.host").getValue();
|
||||||
var token = conf_editor.getEditor("root.specificOptions.token").getValue();
|
var token = conf_editor.getEditor("root.specificOptions.token").getValue();
|
||||||
@ -1878,6 +1928,7 @@ function saveLedConfig(genDefLayout = false) {
|
|||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case "homeassistant":
|
||||||
case "nanoleaf":
|
case "nanoleaf":
|
||||||
case "wled":
|
case "wled":
|
||||||
case "yeelight":
|
case "yeelight":
|
||||||
@ -2311,6 +2362,12 @@ function updateElements(ledType, key) {
|
|||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case "homeassistant":
|
||||||
|
updateElementsHomeAssistant(ledType, key);
|
||||||
|
hardwareLedCount = 1;
|
||||||
|
conf_editor.getEditor("root.generalOptions.hardwareLedCount").setValue(hardwareLedCount);
|
||||||
|
break;
|
||||||
|
|
||||||
case "atmo":
|
case "atmo":
|
||||||
case "karate":
|
case "karate":
|
||||||
var ledProperties = devicesProperties[ledType][key];
|
var ledProperties = devicesProperties[ledType][key];
|
||||||
@ -2438,6 +2495,63 @@ function validateWledLedCount(hardwareLedCount) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function updateElementsHomeAssistant(ledType, key) {
|
||||||
|
|
||||||
|
// Get configured device's details
|
||||||
|
var configuredDeviceType = window.serverConfig.device.type;
|
||||||
|
var configuredHost = window.serverConfig.device.host;
|
||||||
|
var host = conf_editor.getEditor("root.specificOptions.host").getValue();
|
||||||
|
|
||||||
|
// New light selection list values
|
||||||
|
var enumVals = [];
|
||||||
|
var enumTitleVals = [];
|
||||||
|
var enumDefaultVal = [];
|
||||||
|
|
||||||
|
if (devicesProperties[ledType] && devicesProperties[ledType][key]) {
|
||||||
|
var ledDeviceProperties = devicesProperties[ledType][key];
|
||||||
|
|
||||||
|
if (!jQuery.isEmptyObject(ledDeviceProperties)) {
|
||||||
|
if (ledDeviceProperties && ledDeviceProperties.lightEntities) {
|
||||||
|
|
||||||
|
|
||||||
|
for (const light of ledDeviceProperties.lightEntities) {
|
||||||
|
enumVals.push(light.entity_id);
|
||||||
|
enumTitleVals.push(light.attributes.friendly_name);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Select configured device
|
||||||
|
if (configuredDeviceType == ledType && configuredHost == host) {
|
||||||
|
let configuredEntityIds = window.serverConfig.device.entityIds;
|
||||||
|
for (const light of configuredEntityIds) {
|
||||||
|
if ($.inArray(enumVals, light) != -1) {
|
||||||
|
enumVals.push(light);
|
||||||
|
}
|
||||||
|
enumDefaultVal.push(light);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (enumVals.length < 1) {
|
||||||
|
enumVals.push("NONE");
|
||||||
|
enumTitleVals.push($.i18n('edt_dev_spec_lights_discovered_none'));
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
$('#btn_wiz_holder').show();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
let addSchemaElements = {
|
||||||
|
"uniqueItems": true,
|
||||||
|
"minItems": 1,
|
||||||
|
"required": true
|
||||||
|
};
|
||||||
|
|
||||||
|
updateJsonEditorMultiSelection(conf_editor, 'root.specificOptions', 'entityIds', addSchemaElements, enumVals, enumTitleVals, enumDefaultVal);
|
||||||
|
}
|
||||||
|
|
||||||
function updateElementsWled(ledType, key) {
|
function updateElementsWled(ledType, key) {
|
||||||
|
|
||||||
// Get configured device's details
|
// Get configured device's details
|
||||||
@ -2533,6 +2647,7 @@ function updateElementsWled(ledType, key) {
|
|||||||
}
|
}
|
||||||
showInputOptionForItem(conf_editor, "root.specificOptions.segments", "switchOffOtherSegments", showAdditionalOptions);
|
showInputOptionForItem(conf_editor, "root.specificOptions.segments", "switchOffOtherSegments", showAdditionalOptions);
|
||||||
}
|
}
|
||||||
|
|
||||||
function sortByPanelCoordinates(arr, topToBottom, leftToRight) {
|
function sortByPanelCoordinates(arr, topToBottom, leftToRight) {
|
||||||
arr.sort((a, b) => {
|
arr.sort((a, b) => {
|
||||||
//Nanoleaf corodinates start at bottom left, therefore reverse topToBottom
|
//Nanoleaf corodinates start at bottom left, therefore reverse topToBottom
|
||||||
@ -2591,7 +2706,7 @@ function nanoleafGeneratelayout(panelLayout, panelOrderTopDown, panelOrderLeftRi
|
|||||||
29: { name: "4DLightstrip", led: true, sideLengthX: 50, sideLengthY: 50 },
|
29: { name: "4DLightstrip", led: true, sideLengthX: 50, sideLengthY: 50 },
|
||||||
30: { name: "Skylight Panel", led: true, sideLengthX: 180, sideLengthY: 180 },
|
30: { name: "Skylight Panel", led: true, sideLengthX: 180, sideLengthY: 180 },
|
||||||
31: { name: "SkylightControllerPrimary", led: true, sideLengthX: 180, sideLengthY: 180 },
|
31: { name: "SkylightControllerPrimary", led: true, sideLengthX: 180, sideLengthY: 180 },
|
||||||
32: { name: "SkylightControllerPassive", led: true, sideLengthX: 180, sideLengthY: 180 },
|
32: { name: "SkylightControllerPassive", led: true, sideLengthX: 180, sideLengthY: 180 },
|
||||||
999: { name: "Unknown", led: true, sideLengthX: 100, sideLengthY: 100 }
|
999: { name: "Unknown", led: true, sideLengthX: 100, sideLengthY: 100 }
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -321,7 +321,7 @@ function showInfoDialog(type, header, message) {
|
|||||||
$(document).on('click', '[data-dismiss-modal]', function () {
|
$(document).on('click', '[data-dismiss-modal]', function () {
|
||||||
var target = $(this).data('dismiss-modal');
|
var target = $(this).data('dismiss-modal');
|
||||||
$($.find(target)).modal('hide');
|
$($.find(target)).modal('hide');
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function createHintH(type, text, container) {
|
function createHintH(type, text, container) {
|
||||||
@ -478,7 +478,7 @@ function createJsonEditor(container, schema, setconfig, usePanel, arrayre) {
|
|||||||
return editor;
|
return editor;
|
||||||
}
|
}
|
||||||
|
|
||||||
function updateJsonEditorSelection(rootEditor, path, key, addElements, newEnumVals, newTitelVals, newDefaultVal, addSelect, addCustom, addCustomAsFirst, customText) {
|
function updateJsonEditorSelection(rootEditor, path, key, addElements, newEnumVals, newTitleVals, newDefaultVal, addSelect, addCustom, addCustomAsFirst, customText) {
|
||||||
var editor = rootEditor.getEditor(path);
|
var editor = rootEditor.getEditor(path);
|
||||||
var orginalProperties = editor.schema.properties[key];
|
var orginalProperties = editor.schema.properties[key];
|
||||||
|
|
||||||
@ -516,8 +516,8 @@ function updateJsonEditorSelection(rootEditor, path, key, addElements, newEnumVa
|
|||||||
|
|
||||||
if (addCustom) {
|
if (addCustom) {
|
||||||
|
|
||||||
if (newTitelVals.length === 0) {
|
if (newTitleVals.length === 0) {
|
||||||
newTitelVals = [...newEnumVals];
|
newTitleVals = [...newEnumVals];
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!!!customText) {
|
if (!!!customText) {
|
||||||
@ -526,10 +526,10 @@ function updateJsonEditorSelection(rootEditor, path, key, addElements, newEnumVa
|
|||||||
|
|
||||||
if (addCustomAsFirst) {
|
if (addCustomAsFirst) {
|
||||||
newEnumVals.unshift("CUSTOM");
|
newEnumVals.unshift("CUSTOM");
|
||||||
newTitelVals.unshift(customText);
|
newTitleVals.unshift(customText);
|
||||||
} else {
|
} else {
|
||||||
newEnumVals.push("CUSTOM");
|
newEnumVals.push("CUSTOM");
|
||||||
newTitelVals.push(customText);
|
newTitleVals.push(customText);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (newSchema[key].options.infoText) {
|
if (newSchema[key].options.infoText) {
|
||||||
@ -540,7 +540,7 @@ function updateJsonEditorSelection(rootEditor, path, key, addElements, newEnumVa
|
|||||||
|
|
||||||
if (addSelect) {
|
if (addSelect) {
|
||||||
newEnumVals.unshift("SELECT");
|
newEnumVals.unshift("SELECT");
|
||||||
newTitelVals.unshift("edt_conf_enum_please_select");
|
newTitleVals.unshift("edt_conf_enum_please_select");
|
||||||
newDefaultVal = "SELECT";
|
newDefaultVal = "SELECT";
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -548,8 +548,8 @@ function updateJsonEditorSelection(rootEditor, path, key, addElements, newEnumVa
|
|||||||
newSchema[key]["enum"] = newEnumVals;
|
newSchema[key]["enum"] = newEnumVals;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (newTitelVals) {
|
if (newTitleVals) {
|
||||||
newSchema[key]["options"]["enum_titles"] = newTitelVals;
|
newSchema[key]["options"]["enum_titles"] = newTitleVals;
|
||||||
}
|
}
|
||||||
if (newDefaultVal) {
|
if (newDefaultVal) {
|
||||||
newSchema[key]["default"] = newDefaultVal;
|
newSchema[key]["default"] = newDefaultVal;
|
||||||
@ -572,7 +572,7 @@ function updateJsonEditorSelection(rootEditor, path, key, addElements, newEnumVa
|
|||||||
rootEditor.notifyWatchers(path + "." + key);
|
rootEditor.notifyWatchers(path + "." + key);
|
||||||
}
|
}
|
||||||
|
|
||||||
function updateJsonEditorMultiSelection(rootEditor, path, key, addElements, newEnumVals, newTitelVals, newDefaultVal) {
|
function updateJsonEditorMultiSelection(rootEditor, path, key, addElements, newEnumVals, newTitleVals, newDefaultVal) {
|
||||||
var editor = rootEditor.getEditor(path);
|
var editor = rootEditor.getEditor(path);
|
||||||
var orginalProperties = editor.schema.properties[key];
|
var orginalProperties = editor.schema.properties[key];
|
||||||
|
|
||||||
@ -617,8 +617,8 @@ function updateJsonEditorMultiSelection(rootEditor, path, key, addElements, newE
|
|||||||
newSchema[key]["items"]["enum"] = newEnumVals;
|
newSchema[key]["items"]["enum"] = newEnumVals;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (newTitelVals) {
|
if (newTitleVals) {
|
||||||
newSchema[key]["items"]["options"]["enum_titles"] = newTitelVals;
|
newSchema[key]["items"]["options"]["enum_titles"] = newTitleVals;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (newDefaultVal) {
|
if (newDefaultVal) {
|
||||||
@ -923,8 +923,8 @@ function createTableRow(list, head, align) {
|
|||||||
el.style.verticalAlign = "middle";
|
el.style.verticalAlign = "middle";
|
||||||
|
|
||||||
var purifyConfig = {
|
var purifyConfig = {
|
||||||
ADD_TAGS: ['button'],
|
ADD_TAGS: ['button'],
|
||||||
ADD_ATTR: ['onclick']
|
ADD_ATTR: ['onclick']
|
||||||
};
|
};
|
||||||
el.innerHTML = DOMPurify.sanitize(list[i], purifyConfig);
|
el.innerHTML = DOMPurify.sanitize(list[i], purifyConfig);
|
||||||
row.appendChild(el);
|
row.appendChild(el);
|
||||||
@ -1403,7 +1403,7 @@ function loadScript(src, callback, ...params) {
|
|||||||
if (isScriptLoaded(src)) {
|
if (isScriptLoaded(src)) {
|
||||||
debugMessage('Script ' + src + ' already loaded');
|
debugMessage('Script ' + src + ' already loaded');
|
||||||
if (callback && typeof callback === 'function') {
|
if (callback && typeof callback === 'function') {
|
||||||
callback( ...params);
|
callback(...params);
|
||||||
}
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -37,27 +37,37 @@ function createLedDeviceWizards(ledType) {
|
|||||||
$('#btn_led_device_wiz').off();
|
$('#btn_led_device_wiz').off();
|
||||||
if (ledType == "philipshue") {
|
if (ledType == "philipshue") {
|
||||||
$('#btn_wiz_holder').show();
|
$('#btn_wiz_holder').show();
|
||||||
data = { ledType };
|
wizardName = ledType;
|
||||||
|
data = { wizardName };
|
||||||
title = 'wiz_hue_title';
|
title = 'wiz_hue_title';
|
||||||
}
|
}
|
||||||
else if (ledType == "nanoleaf") {
|
else if (ledType == "nanoleaf") {
|
||||||
$('#btn_wiz_holder').hide();
|
$('#btn_wiz_holder').hide();
|
||||||
data = { ledType };
|
wizardName = ledType;
|
||||||
|
data = { wizardName };
|
||||||
title = 'wiz_nanoleaf_user_auth_title';
|
title = 'wiz_nanoleaf_user_auth_title';
|
||||||
}
|
}
|
||||||
|
else if (ledType == "homeassistant") {
|
||||||
|
$('#btn_wiz_holder').hide();
|
||||||
|
wizardName = "layoutLedPositions";
|
||||||
|
data = { wizardName, ledType };
|
||||||
|
title = 'wiz_layout_led_positions_title';
|
||||||
|
}
|
||||||
else if (ledType == "atmoorb") {
|
else if (ledType == "atmoorb") {
|
||||||
$('#btn_wiz_holder').show();
|
$('#btn_wiz_holder').show();
|
||||||
data = { ledType };
|
wizardName = ledType;
|
||||||
|
data = { wizardName };
|
||||||
title = 'wiz_atmoorb_title';
|
title = 'wiz_atmoorb_title';
|
||||||
}
|
}
|
||||||
else if (ledType == "yeelight") {
|
else if (ledType == "yeelight") {
|
||||||
$('#btn_wiz_holder').show();
|
$('#btn_wiz_holder').show();
|
||||||
data = { ledType };
|
wizardName = ledType;
|
||||||
|
data = { wizardName };
|
||||||
title = 'wiz_yeelight_title';
|
title = 'wiz_yeelight_title';
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Object.keys(data).length !== 0) {
|
if (Object.keys(data).length !== 0) {
|
||||||
startLedDeviceWizard(data, title, ledType + "Wizard");
|
startLedDeviceWizard(data, title, wizardName + "Wizard");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -66,8 +76,7 @@ function startLedDeviceWizard(data, hint, wizardName) {
|
|||||||
createHint("wizard", $.i18n(hint), "btn_wiz_holder", "btn_led_device_wiz");
|
createHint("wizard", $.i18n(hint), "btn_wiz_holder", "btn_led_device_wiz");
|
||||||
$('#btn_led_device_wiz').off();
|
$('#btn_led_device_wiz').off();
|
||||||
$('#btn_led_device_wiz').on('click', async (e) => {
|
$('#btn_led_device_wiz').on('click', async (e) => {
|
||||||
const { [wizardName]: winzardObject } = await import('./wizards/LedDevice_' + data.ledType + '.js');
|
const { [wizardName]: winzardObject } = await import('./wizards/LedDevice_' + data.wizardName + '.js');
|
||||||
winzardObject.start(e);
|
winzardObject.start(e, data);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -151,17 +151,7 @@ const atmoorbWizard = (() => {
|
|||||||
$('#wh_topcontainer').toggle(false);
|
$('#wh_topcontainer').toggle(false);
|
||||||
$('#orb_ids_t, #btn_wiz_save').toggle(true);
|
$('#orb_ids_t, #btn_wiz_save').toggle(true);
|
||||||
|
|
||||||
const lightOptions = [
|
const lightOptions = utils.getLayoutPositions();
|
||||||
"top", "topleft", "topright",
|
|
||||||
"bottom", "bottomleft", "bottomright",
|
|
||||||
"left", "lefttop", "leftmiddle", "leftbottom",
|
|
||||||
"right", "righttop", "rightmiddle", "rightbottom",
|
|
||||||
"entire",
|
|
||||||
"lightPosTopLeft112", "lightPosTopLeftNewMid", "lightPosTopLeft121",
|
|
||||||
"lightPosBottomLeft14", "lightPosBottomLeft12", "lightPosBottomLeft34", "lightPosBottomLeft11",
|
|
||||||
"lightPosBottomLeft112", "lightPosBottomLeftNewMid", "lightPosBottomLeft121"
|
|
||||||
];
|
|
||||||
|
|
||||||
lightOptions.unshift("disabled");
|
lightOptions.unshift("disabled");
|
||||||
|
|
||||||
$('.lidsb').html("");
|
$('.lidsb').html("");
|
||||||
@ -178,10 +168,9 @@ const atmoorbWizard = (() => {
|
|||||||
let options = "";
|
let options = "";
|
||||||
for (const opt in lightOptions) {
|
for (const opt in lightOptions) {
|
||||||
const val = lightOptions[opt];
|
const val = lightOptions[opt];
|
||||||
const txt = (val !== 'entire' && val !== 'disabled') ? 'conf_leds_layout_cl_' : 'wiz_ids_';
|
|
||||||
options += '<option value="' + val + '"';
|
options += '<option value="' + val + '"';
|
||||||
if (pos === val) options += ' selected="selected"';
|
if (pos === val) options += ' selected="selected"';
|
||||||
options += '>' + $.i18n(txt + val) + '</option>';
|
options += '>' + $.i18n('conf_leds_layout_cl_' + val) + '</option>';
|
||||||
}
|
}
|
||||||
|
|
||||||
let enabled = 'enabled';
|
let enabled = 'enabled';
|
||||||
|
74
assets/webconfig/js/wizards/LedDevice_layoutLedPositions.js
Normal file
74
assets/webconfig/js/wizards/LedDevice_layoutLedPositions.js
Normal file
@ -0,0 +1,74 @@
|
|||||||
|
//****************************
|
||||||
|
// Wizard LED Layout
|
||||||
|
//****************************
|
||||||
|
|
||||||
|
import { ledDeviceWizardUtils as utils } from './LedDevice_utils.js';
|
||||||
|
|
||||||
|
const layoutLedPositionsWizard = (() => {
|
||||||
|
|
||||||
|
let wiz_editor;
|
||||||
|
|
||||||
|
function createEditor() {
|
||||||
|
wiz_editor = createJsonEditor('editor_container_wiz', {
|
||||||
|
layoutPosition: {
|
||||||
|
"type": "string",
|
||||||
|
"title": "wiz_layout_led_position_title",
|
||||||
|
"enum": utils.getLayoutPositions(),
|
||||||
|
"options": {
|
||||||
|
"enum_titles": utils.getLayoutPositionsTitles()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, true, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
function stopWizardLedLayout(reload) {
|
||||||
|
resetWizard(reload);
|
||||||
|
}
|
||||||
|
|
||||||
|
function beginWizardLayoutLedPositions() {
|
||||||
|
createEditor();
|
||||||
|
setStorage("wizardactive", true);
|
||||||
|
|
||||||
|
$('#btn_wiz_abort').off().on('click', function () {
|
||||||
|
stopWizardLedLayout(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
$('#btn_wiz_ok').off().on('click', function () {
|
||||||
|
const layoutPosition = wiz_editor.getEditor("root.layoutPosition").getValue();
|
||||||
|
const layoutObject = utils.assignLightPos(layoutPosition);
|
||||||
|
|
||||||
|
var layoutObjects = [];
|
||||||
|
layoutObjects.push(JSON.parse(JSON.stringify(layoutObject)));
|
||||||
|
aceEdt.set(layoutObjects);
|
||||||
|
|
||||||
|
stopWizardLedLayout(true);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
start: function (e, data) {
|
||||||
|
$('#wiz_header').html('<i class="fa fa-magic fa-fw"></i>' + $.i18n('wiz_layout_led_positions_title'));
|
||||||
|
$('#wizp1_body').html('<div <p style="font-weight:bold">' + $.i18n('wiz_layout_led_positions_expl', data.ledType) + '</p></div>' +
|
||||||
|
'<div id="editor_container_wiz"></div>'
|
||||||
|
);
|
||||||
|
$('#wizp1_footer').html('<button type="button" class="btn btn-primary" id="btn_wiz_ok"><i class="fa fa-fw fa-check"></i>' + $.i18n('general_btn_ok') +
|
||||||
|
'</button><button type="button" class="btn btn-danger" id="btn_wiz_abort"><i class="fa fa-fw fa-close"></i>' + $.i18n('general_btn_cancel') + '</button>'
|
||||||
|
);
|
||||||
|
|
||||||
|
if (getStorage("darkMode") == "on")
|
||||||
|
$('#wizard_logo').attr("src", 'img/hyperion/logo_negativ.png');
|
||||||
|
|
||||||
|
//open modal
|
||||||
|
$("#wizard_modal").modal({
|
||||||
|
backdrop: "static",
|
||||||
|
keyboard: false,
|
||||||
|
show: true
|
||||||
|
});
|
||||||
|
|
||||||
|
beginWizardLayoutLedPositions();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
})();
|
||||||
|
|
||||||
|
export { layoutLedPositionsWizard };
|
||||||
|
|
@ -794,17 +794,7 @@ const philipshueWizard = (() => {
|
|||||||
}
|
}
|
||||||
$('#hue_ids_t, #btn_wiz_save').toggle(true);
|
$('#hue_ids_t, #btn_wiz_save').toggle(true);
|
||||||
|
|
||||||
const lightOptions = [
|
const lightOptions = utils.getLayoutPositions();
|
||||||
"top", "topleft", "topright",
|
|
||||||
"bottom", "bottomleft", "bottomright",
|
|
||||||
"left", "lefttop", "leftmiddle", "leftbottom",
|
|
||||||
"right", "righttop", "rightmiddle", "rightbottom",
|
|
||||||
"entire",
|
|
||||||
"lightPosTopLeft112", "lightPosTopLeftNewMid", "lightPosTopLeft121",
|
|
||||||
"lightPosBottomLeft14", "lightPosBottomLeft12", "lightPosBottomLeft34", "lightPosBottomLeft11",
|
|
||||||
"lightPosBottomLeft112", "lightPosBottomLeftNewMid", "lightPosBottomLeft121"
|
|
||||||
];
|
|
||||||
|
|
||||||
if (isEntertainmentReady && hueEntertainmentConfigs.length > 0) {
|
if (isEntertainmentReady && hueEntertainmentConfigs.length > 0) {
|
||||||
lightOptions.unshift("entertainment_center");
|
lightOptions.unshift("entertainment_center");
|
||||||
lightOptions.unshift("entertainment");
|
lightOptions.unshift("entertainment");
|
||||||
@ -866,10 +856,9 @@ const philipshueWizard = (() => {
|
|||||||
let options = "";
|
let options = "";
|
||||||
for (const opt in lightOptions) {
|
for (const opt in lightOptions) {
|
||||||
const val = lightOptions[opt];
|
const val = lightOptions[opt];
|
||||||
const txt = (val != 'entire' && val != 'disabled') ? 'conf_leds_layout_cl_' : 'wiz_ids_';
|
|
||||||
options += '<option value="' + val + '"';
|
options += '<option value="' + val + '"';
|
||||||
if (pos == val) options += ' selected="selected"';
|
if (pos == val) options += ' selected="selected"';
|
||||||
options += '>' + $.i18n(txt + val) + '</option>';
|
options += '>' + $.i18n('conf_leds_layout_cl_' + val) + '</option>';
|
||||||
}
|
}
|
||||||
|
|
||||||
$('.lidsb').append(createTableRow([id + ' (' + lightName + ')',
|
$('.lidsb').append(createTableRow([id + ' (' + lightName + ')',
|
||||||
|
@ -52,6 +52,17 @@ const ledDeviceWizardUtils = (() => {
|
|||||||
const i = positionMap[pos] || positionMap["lightPosEntire"];
|
const i = positionMap[pos] || positionMap["lightPosEntire"];
|
||||||
i.name = name;
|
i.name = name;
|
||||||
return i;
|
return i;
|
||||||
|
},
|
||||||
|
getLayoutPositions: function () {
|
||||||
|
return Object.keys(positionMap);
|
||||||
|
},
|
||||||
|
getLayoutPositionsTitles: function () {
|
||||||
|
|
||||||
|
let layoutPositionTitles = [];
|
||||||
|
for (const layoutPosition of Object.keys(positionMap)) {
|
||||||
|
layoutPositionTitles.push($.i18n('conf_leds_layout_cl_' + layoutPosition));
|
||||||
|
}
|
||||||
|
return layoutPositionTitles;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -173,17 +173,7 @@ const yeelightWizard = (() => {
|
|||||||
$('#wh_topcontainer').toggle(false);
|
$('#wh_topcontainer').toggle(false);
|
||||||
$('#yee_ids_t, #btn_wiz_save').toggle(true);
|
$('#yee_ids_t, #btn_wiz_save').toggle(true);
|
||||||
|
|
||||||
const lightOptions = [
|
const lightOptions = utils.getLayoutPositions();
|
||||||
"top", "topleft", "topright",
|
|
||||||
"bottom", "bottomleft", "bottomright",
|
|
||||||
"left", "lefttop", "leftmiddle", "leftbottom",
|
|
||||||
"right", "righttop", "rightmiddle", "rightbottom",
|
|
||||||
"entire",
|
|
||||||
"lightPosTopLeft112", "lightPosTopLeftNewMid", "lightPosTopLeft121",
|
|
||||||
"lightPosBottomLeft14", "lightPosBottomLeft12", "lightPosBottomLeft34", "lightPosBottomLeft11",
|
|
||||||
"lightPosBottomLeft112", "lightPosBottomLeftNewMid", "lightPosBottomLeft121"
|
|
||||||
];
|
|
||||||
|
|
||||||
lightOptions.unshift("disabled");
|
lightOptions.unshift("disabled");
|
||||||
|
|
||||||
$('.lidsb').html("");
|
$('.lidsb').html("");
|
||||||
@ -200,10 +190,9 @@ const yeelightWizard = (() => {
|
|||||||
let options = "";
|
let options = "";
|
||||||
for (const opt in lightOptions) {
|
for (const opt in lightOptions) {
|
||||||
const val = lightOptions[opt];
|
const val = lightOptions[opt];
|
||||||
const txt = (val !== 'entire' && val !== 'disabled') ? 'conf_leds_layout_cl_' : 'wiz_ids_';
|
|
||||||
options += '<option value="' + val + '"';
|
options += '<option value="' + val + '"';
|
||||||
if (pos === val) options += ' selected="selected"';
|
if (pos === val) options += ' selected="selected"';
|
||||||
options += '>' + $.i18n(txt + val) + '</option>';
|
options += '>' + $.i18n('conf_leds_layout_cl_' + val) + '</option>';
|
||||||
}
|
}
|
||||||
|
|
||||||
let enabled = 'enabled';
|
let enabled = 'enabled';
|
||||||
|
@ -22,6 +22,7 @@ const MdnsServiceMap mDnsServiceMap = {
|
|||||||
|
|
||||||
//LED Devices
|
//LED Devices
|
||||||
{"cololight" , {"_hap._tcp.local.", "ColoLight.*"}},
|
{"cololight" , {"_hap._tcp.local.", "ColoLight.*"}},
|
||||||
|
{"homeassistant", {"_home-assistant._tcp.local.", ".*"}},
|
||||||
{"nanoleaf" , {"_nanoleafapi._tcp.local.", ".*"}},
|
{"nanoleaf" , {"_nanoleafapi._tcp.local.", ".*"}},
|
||||||
{"philipshue" , {"_hue._tcp.local.", ".*"}},
|
{"philipshue" , {"_hue._tcp.local.", ".*"}},
|
||||||
{"wled" , {"_wled._tcp.local.", ".*"}},
|
{"wled" , {"_wled._tcp.local.", ".*"}},
|
||||||
|
@ -735,7 +735,7 @@ void JsonAPI::handleConfigSetCommand(const QJsonObject &message, const JsonApiCo
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
sendErrorReply("Saving configuration while Hyperion is disabled isn't possible", cmd);
|
sendErrorReply("It is not possible saving a configuration while Hyperion is disabled", cmd);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -7,6 +7,7 @@
|
|||||||
<file alias="schema-dmx">schemas/schema-dmx.json</file>
|
<file alias="schema-dmx">schemas/schema-dmx.json</file>
|
||||||
<file alias="schema-fadecandy">schemas/schema-fadecandy.json</file>
|
<file alias="schema-fadecandy">schemas/schema-fadecandy.json</file>
|
||||||
<file alias="schema-file">schemas/schema-file.json</file>
|
<file alias="schema-file">schemas/schema-file.json</file>
|
||||||
|
<file alias="schema-homeassistant">schemas/schema-homeassistant.json</file>
|
||||||
<file alias="schema-hyperionusbasp">schemas/schema-hyperionusbasp.json</file>
|
<file alias="schema-hyperionusbasp">schemas/schema-hyperionusbasp.json</file>
|
||||||
<file alias="schema-lightpack">schemas/schema-lightpack.json</file>
|
<file alias="schema-lightpack">schemas/schema-lightpack.json</file>
|
||||||
<file alias="schema-lpd6803">schemas/schema-lpd6803.json</file>
|
<file alias="schema-lpd6803">schemas/schema-lpd6803.json</file>
|
||||||
|
446
libsrc/leddevice/dev_net/LedDeviceHomeAssistant.cpp
Normal file
446
libsrc/leddevice/dev_net/LedDeviceHomeAssistant.cpp
Normal file
@ -0,0 +1,446 @@
|
|||||||
|
// Local-Hyperion includes
|
||||||
|
#include "LedDeviceHomeAssistant.h"
|
||||||
|
|
||||||
|
#include <ssdp/SSDPDiscover.h>
|
||||||
|
// mDNS discover
|
||||||
|
#ifdef ENABLE_MDNS
|
||||||
|
#include <mdns/MdnsBrowser.h>
|
||||||
|
#include <mdns/MdnsServiceRegister.h>
|
||||||
|
#endif
|
||||||
|
#include <utils/NetUtils.h>
|
||||||
|
#include <utils/ColorRgb.h>
|
||||||
|
|
||||||
|
#include <algorithm>
|
||||||
|
|
||||||
|
// Constants
|
||||||
|
namespace {
|
||||||
|
const bool verbose = false;
|
||||||
|
|
||||||
|
// Configuration settings
|
||||||
|
const char CONFIG_HOST[] = "host";
|
||||||
|
const char CONFIG_PORT[] = "port";
|
||||||
|
const char CONFIG_AUTH_TOKEN[] = "token";
|
||||||
|
const char CONFIG_ENITYIDS[] = "entityIds";
|
||||||
|
const char CONFIG_BRIGHTNESS[] = "brightness";
|
||||||
|
const char CONFIG_BRIGHTNESS_OVERWRITE[] = "overwriteBrightness";
|
||||||
|
const char CONFIG_FULL_BRIGHTNESS_AT_START[] = "fullBrightnessAtStart";
|
||||||
|
const char CONFIG_ON_OFF_BLACK[] = "switchOffOnBlack";
|
||||||
|
const char CONFIG_TRANSITIONTIME[] = "transitionTime";
|
||||||
|
|
||||||
|
const bool DEFAULT_IS_BRIGHTNESS_OVERWRITE = true;
|
||||||
|
const bool DEFAULT_IS_FULL_BRIGHTNESS_AT_START = true;
|
||||||
|
const int BRI_MAX = 255;
|
||||||
|
const bool DEFAULT_IS_SWITCH_OFF_ON_BLACK = false;
|
||||||
|
|
||||||
|
// Home Assistant API
|
||||||
|
const int API_DEFAULT_PORT = 8123;
|
||||||
|
const char API_BASE_PATH[] = "/api/";
|
||||||
|
const char API_STATES[] = "states";
|
||||||
|
const char API_LIGHT_TURN_ON[] = "services/light/turn_on";
|
||||||
|
const char API_LIGHT_TURN_OFF[] = "services/light/turn_off";
|
||||||
|
|
||||||
|
const char ENTITY_ID[] = "entity_id";
|
||||||
|
const char RGB_COLOR[] = "rgb_color";
|
||||||
|
const char BRIGHTNESS[] = "brightness";
|
||||||
|
const char TRANSITION[] = "transition";
|
||||||
|
const char FLASH[] = "flash";
|
||||||
|
|
||||||
|
// // Home Assistant ssdp services
|
||||||
|
const char SSDP_ID[] = "ssdp:all";
|
||||||
|
const char SSDP_FILTER_HEADER[] = "ST";
|
||||||
|
const char SSDP_FILTER[] = "(.*)home-assistant.io(.*)";
|
||||||
|
|
||||||
|
} //End of constants
|
||||||
|
|
||||||
|
LedDeviceHomeAssistant::LedDeviceHomeAssistant(const QJsonObject& deviceConfig)
|
||||||
|
: LedDevice(deviceConfig)
|
||||||
|
, _restApi(nullptr)
|
||||||
|
, _apiPort(API_DEFAULT_PORT)
|
||||||
|
, _isBrightnessOverwrite(DEFAULT_IS_BRIGHTNESS_OVERWRITE)
|
||||||
|
, _isFullBrightnessAtStart(DEFAULT_IS_FULL_BRIGHTNESS_AT_START)
|
||||||
|
, _brightness (BRI_MAX)
|
||||||
|
{
|
||||||
|
#ifdef ENABLE_MDNS
|
||||||
|
QMetaObject::invokeMethod(MdnsBrowser::getInstance().data(), "browseForServiceType",
|
||||||
|
Qt::QueuedConnection, Q_ARG(QByteArray, MdnsServiceRegister::getServiceType(_activeDeviceType)));
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
LedDevice* LedDeviceHomeAssistant::construct(const QJsonObject& deviceConfig)
|
||||||
|
{
|
||||||
|
return new LedDeviceHomeAssistant(deviceConfig);
|
||||||
|
}
|
||||||
|
|
||||||
|
LedDeviceHomeAssistant::~LedDeviceHomeAssistant()
|
||||||
|
{
|
||||||
|
delete _restApi;
|
||||||
|
_restApi = nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool LedDeviceHomeAssistant::init(const QJsonObject& deviceConfig)
|
||||||
|
{
|
||||||
|
bool isInitOK{ false };
|
||||||
|
|
||||||
|
if ( LedDevice::init(deviceConfig) )
|
||||||
|
{
|
||||||
|
// Overwrite non supported/required features
|
||||||
|
if (deviceConfig["rewriteTime"].toInt(0) > 0)
|
||||||
|
{
|
||||||
|
Info(_log, "Home Assistant lights do not require rewrites. Refresh time is ignored.");
|
||||||
|
setRewriteTime(0);
|
||||||
|
}
|
||||||
|
DebugIf(verbose, _log, "deviceConfig: [%s]", QString(QJsonDocument(_devConfig).toJson(QJsonDocument::Compact)).toUtf8().constData());
|
||||||
|
|
||||||
|
//Set hostname as per configuration and default port
|
||||||
|
_hostName = deviceConfig[CONFIG_HOST].toString();
|
||||||
|
_apiPort = deviceConfig[CONFIG_PORT].toInt(API_DEFAULT_PORT);
|
||||||
|
_bearerToken = deviceConfig[CONFIG_AUTH_TOKEN].toString();
|
||||||
|
|
||||||
|
_isBrightnessOverwrite = _devConfig[CONFIG_BRIGHTNESS_OVERWRITE].toBool(DEFAULT_IS_BRIGHTNESS_OVERWRITE);
|
||||||
|
_isFullBrightnessAtStart = _devConfig[CONFIG_FULL_BRIGHTNESS_AT_START].toBool(DEFAULT_IS_FULL_BRIGHTNESS_AT_START);
|
||||||
|
_brightness = _devConfig[CONFIG_BRIGHTNESS].toInt(BRI_MAX);
|
||||||
|
_switchOffOnBlack = _devConfig[CONFIG_ON_OFF_BLACK].toBool(DEFAULT_IS_SWITCH_OFF_ON_BLACK);
|
||||||
|
int transitionTimeMs = _devConfig[CONFIG_TRANSITIONTIME].toInt(0);
|
||||||
|
_transitionTime = transitionTimeMs / 1000.0;
|
||||||
|
|
||||||
|
Debug(_log, "Hostname/IP : %s", QSTRING_CSTR(_hostName));
|
||||||
|
Debug(_log, "Port : %d", _apiPort );
|
||||||
|
|
||||||
|
Debug(_log, "Overwrite Brightn.: %s", _isBrightnessOverwrite ? "Yes" : "No" );
|
||||||
|
Debug(_log, "Set Brightness to : %d", _brightness);
|
||||||
|
Debug(_log, "Full Bri. at start: %s", _isFullBrightnessAtStart ? "Yes" : "No" );
|
||||||
|
Debug(_log, "Off on Black : %s", _switchOffOnBlack ? "Yes" : "No" );
|
||||||
|
Debug(_log, "Transition Time : %d ms", transitionTimeMs );
|
||||||
|
|
||||||
|
_lightEntityIds = _devConfig[ CONFIG_ENITYIDS ].toVariant().toStringList();
|
||||||
|
int configuredLightsCount = _lightEntityIds.size();
|
||||||
|
|
||||||
|
if ( configuredLightsCount == 0 )
|
||||||
|
{
|
||||||
|
this->setInError( "No light entity-ids configured" );
|
||||||
|
isInitOK = false;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Debug(_log, "Lights configured : %d", configuredLightsCount );
|
||||||
|
isInitOK = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return isInitOK;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool LedDeviceHomeAssistant::initLedsConfiguration()
|
||||||
|
{
|
||||||
|
bool isInitOK = false;
|
||||||
|
|
||||||
|
//Currently on one light is supported
|
||||||
|
QString lightEntityId = _lightEntityIds[0];
|
||||||
|
|
||||||
|
//Get properties for configured light entitiy to check availability
|
||||||
|
_restApi->setPath({ API_STATES, lightEntityId});
|
||||||
|
httpResponse response = _restApi->get();
|
||||||
|
if (response.error())
|
||||||
|
{
|
||||||
|
QString errorReason = QString("%1 get properties failed with error: '%2'").arg(_activeDeviceType,response.getErrorReason());
|
||||||
|
this->setInError(errorReason);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
QJsonObject propertiesDetails = response.getBody().object();
|
||||||
|
if (propertiesDetails.isEmpty())
|
||||||
|
{
|
||||||
|
QString errorReason = QString("Light [%1] does not exist").arg(lightEntityId);
|
||||||
|
this->setInError(errorReason);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (propertiesDetails.value("state").toString().compare("unavailable") == 0)
|
||||||
|
{
|
||||||
|
Warning(_log, "Light [%s] is currently unavailable", QSTRING_CSTR(lightEntityId));
|
||||||
|
}
|
||||||
|
isInitOK = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return isInitOK;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool LedDeviceHomeAssistant::openRestAPI()
|
||||||
|
{
|
||||||
|
bool isInitOK{ true };
|
||||||
|
|
||||||
|
if (_restApi == nullptr)
|
||||||
|
{
|
||||||
|
if (_apiPort == 0)
|
||||||
|
{
|
||||||
|
_apiPort = API_DEFAULT_PORT;
|
||||||
|
}
|
||||||
|
|
||||||
|
_restApi = new ProviderRestApi(_address.toString(), _apiPort);
|
||||||
|
_restApi->setLogger(_log);
|
||||||
|
|
||||||
|
_restApi->setHeader(QNetworkRequest::ContentTypeHeader, "application/json");
|
||||||
|
_restApi->setHeader("Authorization", QByteArrayLiteral("Bearer ") + _bearerToken.toUtf8());
|
||||||
|
|
||||||
|
//Base-path is api-path
|
||||||
|
_restApi->setBasePath(API_BASE_PATH);
|
||||||
|
}
|
||||||
|
return isInitOK;
|
||||||
|
}
|
||||||
|
|
||||||
|
int LedDeviceHomeAssistant::open()
|
||||||
|
{
|
||||||
|
int retval = -1;
|
||||||
|
_isDeviceReady = false;
|
||||||
|
|
||||||
|
if (NetUtils::resolveHostToAddress(_log, _hostName, _address, _apiPort))
|
||||||
|
{
|
||||||
|
if (openRestAPI())
|
||||||
|
{
|
||||||
|
// Read LedDevice configuration and validate against device configuration
|
||||||
|
if (initLedsConfiguration())
|
||||||
|
{
|
||||||
|
// Everything is OK, device is ready
|
||||||
|
_isDeviceReady = true;
|
||||||
|
retval = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_restApi->setHost(_address.toString());
|
||||||
|
_restApi->setPort(_apiPort);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return retval;
|
||||||
|
}
|
||||||
|
|
||||||
|
QJsonArray LedDeviceHomeAssistant::discoverSsdp() const
|
||||||
|
{
|
||||||
|
QJsonArray deviceList;
|
||||||
|
SSDPDiscover ssdpDiscover;
|
||||||
|
ssdpDiscover.skipDuplicateKeys(true);
|
||||||
|
ssdpDiscover.setSearchFilter(SSDP_FILTER, SSDP_FILTER_HEADER);
|
||||||
|
QString searchTarget = SSDP_ID;
|
||||||
|
|
||||||
|
if (ssdpDiscover.discoverServices(searchTarget) > 0)
|
||||||
|
{
|
||||||
|
deviceList = ssdpDiscover.getServicesDiscoveredJson();
|
||||||
|
}
|
||||||
|
return deviceList;
|
||||||
|
}
|
||||||
|
|
||||||
|
QJsonObject LedDeviceHomeAssistant::discover(const QJsonObject& /*params*/)
|
||||||
|
{
|
||||||
|
QJsonObject devicesDiscovered;
|
||||||
|
devicesDiscovered.insert("ledDeviceType", _activeDeviceType);
|
||||||
|
|
||||||
|
QJsonArray deviceList;
|
||||||
|
|
||||||
|
#ifdef ENABLE_MDNS
|
||||||
|
QString discoveryMethod("mDNS");
|
||||||
|
deviceList = MdnsBrowser::getInstance().data()->getServicesDiscoveredJson(
|
||||||
|
MdnsServiceRegister::getServiceType(_activeDeviceType),
|
||||||
|
MdnsServiceRegister::getServiceNameFilter(_activeDeviceType),
|
||||||
|
DEFAULT_DISCOVER_TIMEOUT
|
||||||
|
);
|
||||||
|
#else
|
||||||
|
QString discoveryMethod("ssdp");
|
||||||
|
deviceList = discoverSsdp();
|
||||||
|
#endif
|
||||||
|
|
||||||
|
devicesDiscovered.insert("discoveryMethod", discoveryMethod);
|
||||||
|
devicesDiscovered.insert("devices", deviceList);
|
||||||
|
|
||||||
|
DebugIf(verbose, _log, "devicesDiscovered: [%s]", QString(QJsonDocument(devicesDiscovered).toJson(QJsonDocument::Compact)).toUtf8().constData());
|
||||||
|
|
||||||
|
return devicesDiscovered;
|
||||||
|
}
|
||||||
|
|
||||||
|
QJsonObject LedDeviceHomeAssistant::getProperties(const QJsonObject& params)
|
||||||
|
{
|
||||||
|
DebugIf(verbose, _log, "params: [%s]", QString(QJsonDocument(params).toJson(QJsonDocument::Compact)).toUtf8().constData());
|
||||||
|
QJsonObject properties;
|
||||||
|
|
||||||
|
_hostName = params[CONFIG_HOST].toString("");
|
||||||
|
_apiPort = API_DEFAULT_PORT;
|
||||||
|
_bearerToken = params[CONFIG_AUTH_TOKEN].toString("");
|
||||||
|
|
||||||
|
Info(_log, "Get properties for %s, hostname (%s)", QSTRING_CSTR(_activeDeviceType), QSTRING_CSTR(_hostName));
|
||||||
|
|
||||||
|
if (NetUtils::resolveHostToAddress(_log, _hostName, _address, _apiPort))
|
||||||
|
{
|
||||||
|
if (openRestAPI())
|
||||||
|
{
|
||||||
|
QString filter = params["filter"].toString("");
|
||||||
|
_restApi->setPath(filter);
|
||||||
|
|
||||||
|
// Perform request
|
||||||
|
httpResponse response = _restApi->get();
|
||||||
|
if (response.error())
|
||||||
|
{
|
||||||
|
Warning(_log, "%s get properties failed with error: '%s'", QSTRING_CSTR(_activeDeviceType), QSTRING_CSTR(response.getErrorReason()));
|
||||||
|
}
|
||||||
|
|
||||||
|
QJsonObject propertiesDetails;
|
||||||
|
const QJsonDocument jsonDoc = response.getBody();
|
||||||
|
if (jsonDoc.isArray()) {
|
||||||
|
const QJsonArray jsonArray = jsonDoc.array();
|
||||||
|
QVector<QJsonValue> filteredVector;
|
||||||
|
|
||||||
|
// Iterate over the array and filter objects with entity_id starting with "light."
|
||||||
|
for (const QJsonValue &value : jsonArray)
|
||||||
|
{
|
||||||
|
QJsonObject obj = value.toObject();
|
||||||
|
QString entityId = obj[ENTITY_ID].toString();
|
||||||
|
|
||||||
|
if (entityId.startsWith("light."))
|
||||||
|
{
|
||||||
|
filteredVector.append(obj);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Sort the filtered vector by "friendly_name" in ascending order
|
||||||
|
std::sort(filteredVector.begin(), filteredVector.end(), [](const QJsonValue &a, const QJsonValue &b) {
|
||||||
|
QString nameA = a.toObject()["attributes"].toObject()["friendly_name"].toString();
|
||||||
|
QString nameB = b.toObject()["attributes"].toObject()["friendly_name"].toString();
|
||||||
|
return nameA < nameB; // Ascending order
|
||||||
|
});
|
||||||
|
// Convert the sorted vector back to a QJsonArray
|
||||||
|
QJsonArray sortedArray;
|
||||||
|
for (const QJsonValue &value : filteredVector) {
|
||||||
|
sortedArray.append(value);
|
||||||
|
}
|
||||||
|
|
||||||
|
propertiesDetails.insert("lightEntities", sortedArray);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!propertiesDetails.isEmpty())
|
||||||
|
{
|
||||||
|
propertiesDetails.insert("ledCount", 1);
|
||||||
|
}
|
||||||
|
properties.insert("properties", propertiesDetails);
|
||||||
|
}
|
||||||
|
|
||||||
|
DebugIf(verbose, _log, "properties: [%s]", QString(QJsonDocument(properties).toJson(QJsonDocument::Compact)).toUtf8().constData());
|
||||||
|
}
|
||||||
|
return properties;
|
||||||
|
}
|
||||||
|
|
||||||
|
void LedDeviceHomeAssistant::identify(const QJsonObject& params)
|
||||||
|
{
|
||||||
|
DebugIf(verbose, _log, "params: [%s]", QString(QJsonDocument(params).toJson(QJsonDocument::Compact)).toUtf8().constData());
|
||||||
|
|
||||||
|
_hostName = params[CONFIG_HOST].toString("");
|
||||||
|
_apiPort = API_DEFAULT_PORT;
|
||||||
|
_bearerToken = params[CONFIG_AUTH_TOKEN].toString("");
|
||||||
|
|
||||||
|
Info(_log, "Identify %s, hostname (%s)", QSTRING_CSTR(_activeDeviceType), QSTRING_CSTR(_hostName));
|
||||||
|
|
||||||
|
if (NetUtils::resolveHostToAddress(_log, _hostName, _address, _apiPort))
|
||||||
|
{
|
||||||
|
if (openRestAPI())
|
||||||
|
{
|
||||||
|
QJsonArray lightEntityIds = params[ ENTITY_ID ].toArray();
|
||||||
|
|
||||||
|
_restApi->setPath(API_LIGHT_TURN_ON);
|
||||||
|
QJsonObject serviceAttributes{{ENTITY_ID, lightEntityIds}};
|
||||||
|
serviceAttributes.insert(FLASH, "short");
|
||||||
|
|
||||||
|
httpResponse response = _restApi->post(serviceAttributes);
|
||||||
|
if (response.error())
|
||||||
|
{
|
||||||
|
Warning(_log, "%s identification failed with error: '%s'", QSTRING_CSTR(_activeDeviceType), QSTRING_CSTR(response.getErrorReason()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool LedDeviceHomeAssistant::powerOn()
|
||||||
|
{
|
||||||
|
bool isOn = false;
|
||||||
|
if (_isDeviceReady)
|
||||||
|
{
|
||||||
|
_restApi->setPath(API_LIGHT_TURN_ON);
|
||||||
|
QJsonObject serviceAttributes {{ENTITY_ID, QJsonArray::fromStringList(_lightEntityIds)}};
|
||||||
|
|
||||||
|
if (_isFullBrightnessAtStart)
|
||||||
|
{
|
||||||
|
serviceAttributes.insert(BRIGHTNESS, BRI_MAX);
|
||||||
|
}
|
||||||
|
|
||||||
|
httpResponse response = _restApi->post(serviceAttributes);
|
||||||
|
if (response.error())
|
||||||
|
{
|
||||||
|
QString errorReason = QString("Power-on request failed with error: '%1'").arg(response.getErrorReason());
|
||||||
|
this->setInError(errorReason);
|
||||||
|
isOn = false;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
isOn = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return isOn;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool LedDeviceHomeAssistant::powerOff()
|
||||||
|
{
|
||||||
|
bool isOff = true;
|
||||||
|
if (_isDeviceReady)
|
||||||
|
{
|
||||||
|
_restApi->setPath(API_LIGHT_TURN_OFF);
|
||||||
|
QJsonObject serviceAttributes {{ENTITY_ID, QJsonArray::fromStringList(_lightEntityIds)}};
|
||||||
|
httpResponse response = _restApi->post(serviceAttributes);
|
||||||
|
if (response.error())
|
||||||
|
{
|
||||||
|
QString errorReason = QString("Power-off request failed with error: '%1'").arg(response.getErrorReason());
|
||||||
|
this->setInError(errorReason);
|
||||||
|
isOff = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return isOff;
|
||||||
|
}
|
||||||
|
|
||||||
|
int LedDeviceHomeAssistant::write(const std::vector<ColorRgb>& ledValues)
|
||||||
|
{
|
||||||
|
int retVal = 0;
|
||||||
|
|
||||||
|
QJsonObject serviceAttributes {{ENTITY_ID, QJsonArray::fromStringList(_lightEntityIds)}};
|
||||||
|
ColorRgb ledValue = ledValues.at(0);
|
||||||
|
|
||||||
|
if (_switchOffOnBlack && ledValue == ColorRgb::BLACK)
|
||||||
|
{
|
||||||
|
_restApi->setPath(API_LIGHT_TURN_OFF);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// http://hostname:port/api/services/light/turn_on
|
||||||
|
// {
|
||||||
|
// "entity_id": [ entity-IDs ],
|
||||||
|
// "rgb_color": [R,G,B]
|
||||||
|
// }
|
||||||
|
|
||||||
|
_restApi->setPath(API_LIGHT_TURN_ON);
|
||||||
|
QJsonArray rgbColor {ledValue.red, ledValue.green, ledValue.blue};
|
||||||
|
serviceAttributes.insert(RGB_COLOR, rgbColor);
|
||||||
|
|
||||||
|
if (_isBrightnessOverwrite)
|
||||||
|
{
|
||||||
|
serviceAttributes.insert(BRIGHTNESS, _brightness);
|
||||||
|
}
|
||||||
|
if (_transitionTime > 0)
|
||||||
|
{
|
||||||
|
// Transition time in seconds
|
||||||
|
serviceAttributes.insert(TRANSITION, _transitionTime);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
httpResponse response = _restApi->post(serviceAttributes);
|
||||||
|
if (response.error())
|
||||||
|
{
|
||||||
|
Warning(_log,"Updating lights failed with error: '%s'", QSTRING_CSTR(response.getErrorReason()) );
|
||||||
|
retVal = -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return retVal;
|
||||||
|
}
|
181
libsrc/leddevice/dev_net/LedDeviceHomeAssistant.h
Normal file
181
libsrc/leddevice/dev_net/LedDeviceHomeAssistant.h
Normal file
@ -0,0 +1,181 @@
|
|||||||
|
#ifndef LEDEVICEHOMEASSISTANT_H
|
||||||
|
#define LEDEVICEHOMEASSISTANT_H
|
||||||
|
|
||||||
|
// LedDevice includes
|
||||||
|
#include <leddevice/LedDevice.h>
|
||||||
|
#include "ProviderRestApi.h"
|
||||||
|
|
||||||
|
// Qt includes
|
||||||
|
#include <QString>
|
||||||
|
#include <QHostAddress>
|
||||||
|
#include <QNetworkAccessManager>
|
||||||
|
|
||||||
|
///
|
||||||
|
/// Implementation of the LedDevice interface for sending to
|
||||||
|
/// lights made available via the Home Assistant platform.
|
||||||
|
///
|
||||||
|
class LedDeviceHomeAssistant : LedDevice
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
///
|
||||||
|
/// @brief Constructs LED-device for Home Assistant Lights
|
||||||
|
///
|
||||||
|
/// following code shows all configuration options
|
||||||
|
/// @code
|
||||||
|
/// "device" :
|
||||||
|
/// {
|
||||||
|
/// "type" : "homeassistant"
|
||||||
|
/// "host" : "hostname or IP",
|
||||||
|
/// "port" : port
|
||||||
|
/// "token": "bearer token",
|
||||||
|
/// },
|
||||||
|
///@endcode
|
||||||
|
///
|
||||||
|
/// @param deviceConfig Device's configuration as JSON-Object
|
||||||
|
///
|
||||||
|
explicit LedDeviceHomeAssistant(const QJsonObject& deviceConfig);
|
||||||
|
|
||||||
|
///
|
||||||
|
/// @brief Destructor of the LED-device
|
||||||
|
///
|
||||||
|
~LedDeviceHomeAssistant() override;
|
||||||
|
|
||||||
|
///
|
||||||
|
/// @brief Constructs the LED-device
|
||||||
|
///
|
||||||
|
/// @param[in] deviceConfig Device's configuration as JSON-Object
|
||||||
|
/// @return LedDevice constructed
|
||||||
|
static LedDevice* construct(const QJsonObject& deviceConfig);
|
||||||
|
|
||||||
|
///
|
||||||
|
/// @brief Discover Home Assistant lights available (for configuration).
|
||||||
|
///
|
||||||
|
/// @param[in] params Parameters used to overwrite discovery default behaviour
|
||||||
|
///
|
||||||
|
/// @return A JSON structure holding a list of devices found
|
||||||
|
///
|
||||||
|
QJsonObject discover(const QJsonObject& params) override;
|
||||||
|
|
||||||
|
///
|
||||||
|
/// @brief Get the Home Assistant light's resource properties
|
||||||
|
///
|
||||||
|
/// Following parameters are required
|
||||||
|
/// @code
|
||||||
|
/// {
|
||||||
|
/// "host" : "hostname or IP",
|
||||||
|
/// "port" : port
|
||||||
|
/// "token" : "bearer token",
|
||||||
|
/// "filter": "resource to query", root "/" is used, if empty
|
||||||
|
/// }
|
||||||
|
///@endcode
|
||||||
|
///
|
||||||
|
/// @param[in] params Parameters to query device
|
||||||
|
/// @return A JSON structure holding the device's properties
|
||||||
|
///
|
||||||
|
QJsonObject getProperties(const QJsonObject& params) override;
|
||||||
|
|
||||||
|
///
|
||||||
|
/// @brief Send an update to the Nanoleaf device to identify it.
|
||||||
|
///
|
||||||
|
/// Following parameters are required
|
||||||
|
/// @code
|
||||||
|
/// {
|
||||||
|
/// "host" : "hostname or IP",
|
||||||
|
/// "port" : port
|
||||||
|
/// "token" : "bearer token",
|
||||||
|
/// "entity_id": array of lightIds
|
||||||
|
/// }
|
||||||
|
///@endcode
|
||||||
|
///
|
||||||
|
/// @param[in] params Parameters to address device
|
||||||
|
///
|
||||||
|
void identify(const QJsonObject& params) override;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
|
||||||
|
///
|
||||||
|
/// @brief Initialise the Home Assistant light's configuration and network address details
|
||||||
|
///
|
||||||
|
/// @param[in] deviceConfig the JSON device configuration
|
||||||
|
/// @return True, if success
|
||||||
|
///
|
||||||
|
bool init(const QJsonObject& deviceConfig) override;
|
||||||
|
|
||||||
|
///
|
||||||
|
/// @brief Opens the output device.
|
||||||
|
///
|
||||||
|
/// @return Zero on success (i.e. device is ready), else negative
|
||||||
|
///
|
||||||
|
int open() override;
|
||||||
|
|
||||||
|
///
|
||||||
|
/// @brief Writes the RGB-Color values to the Home Assistant light.
|
||||||
|
///
|
||||||
|
/// @param[in] ledValues The RGB-color
|
||||||
|
/// @return Zero on success, else negative
|
||||||
|
//////
|
||||||
|
int write(const std::vector<ColorRgb>& ledValues) override;
|
||||||
|
|
||||||
|
///
|
||||||
|
/// @brief Power-/turn on the Home Assistant light.
|
||||||
|
///
|
||||||
|
/// @brief Store the device's original state.
|
||||||
|
///
|
||||||
|
bool powerOn() override;
|
||||||
|
|
||||||
|
///
|
||||||
|
/// @brief Power-/turn off the Home Assistant light.
|
||||||
|
///
|
||||||
|
/// @return True if success
|
||||||
|
///
|
||||||
|
bool powerOff() override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
|
||||||
|
///
|
||||||
|
/// @brief Initialise the access to the REST-API wrapper
|
||||||
|
///
|
||||||
|
/// @return True, if success
|
||||||
|
///
|
||||||
|
bool openRestAPI();
|
||||||
|
|
||||||
|
///
|
||||||
|
/// @brief Get Nanoleaf device details and configuration
|
||||||
|
///
|
||||||
|
/// @return True, if Nanoleaf device capabilities fit configuration
|
||||||
|
///
|
||||||
|
bool initLedsConfiguration();
|
||||||
|
|
||||||
|
///
|
||||||
|
/// @brief Discover Home Assistant lights available (for configuration).
|
||||||
|
///
|
||||||
|
/// @return A JSON structure holding a list of devices found
|
||||||
|
///
|
||||||
|
QJsonArray discoverSsdp() const;
|
||||||
|
|
||||||
|
// ///
|
||||||
|
// /// @brief Get number of panels that can be used as LEds.
|
||||||
|
// ///
|
||||||
|
// /// @return Number of usable LED panels
|
||||||
|
// ///
|
||||||
|
// int getHwLedCount(const QJsonObject& jsonLayout) const;
|
||||||
|
|
||||||
|
QString _hostName;
|
||||||
|
QHostAddress _address;
|
||||||
|
ProviderRestApi* _restApi;
|
||||||
|
int _apiPort;
|
||||||
|
QString _bearerToken;
|
||||||
|
|
||||||
|
/// List of the HA light entity_ids.
|
||||||
|
QStringList _lightEntityIds;
|
||||||
|
|
||||||
|
bool _isBrightnessOverwrite;
|
||||||
|
bool _isFullBrightnessAtStart;
|
||||||
|
int _brightness;
|
||||||
|
bool _switchOffOnBlack;
|
||||||
|
/// Transition time in seconds
|
||||||
|
double _transitionTime;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // LEDEVICEHOMEASSISTANT_H
|
@ -31,7 +31,7 @@ const char CONFIG_TRANSITIONTIME[] = "transitiontime";
|
|||||||
const char CONFIG_BLACK_LIGHTS_TIMEOUT[] = "blackLightsTimeout";
|
const char CONFIG_BLACK_LIGHTS_TIMEOUT[] = "blackLightsTimeout";
|
||||||
const char CONFIG_ON_OFF_BLACK[] = "switchOffOnBlack";
|
const char CONFIG_ON_OFF_BLACK[] = "switchOffOnBlack";
|
||||||
const char CONFIG_RESTORE_STATE[] = "restoreOriginalState";
|
const char CONFIG_RESTORE_STATE[] = "restoreOriginalState";
|
||||||
const char CONFIG_lightIdS[] = "lightIds";
|
const char CONFIG_LIGHTIDS[] = "lightIds";
|
||||||
const char CONFIG_USE_HUE_API_V2[] = "useAPIv2";
|
const char CONFIG_USE_HUE_API_V2[] = "useAPIv2";
|
||||||
const char CONFIG_USE_HUE_ENTERTAINMENT_API[] = "useEntertainmentAPI";
|
const char CONFIG_USE_HUE_ENTERTAINMENT_API[] = "useEntertainmentAPI";
|
||||||
const char CONFIG_groupId[] = "groupId";
|
const char CONFIG_groupId[] = "groupId";
|
||||||
@ -1849,7 +1849,7 @@ bool LedDevicePhilipsHue::setLights()
|
|||||||
_useEntertainmentAPI = false;
|
_useEntertainmentAPI = false;
|
||||||
Error(_log, "Group-ID [%s] is not usable - Entertainment API usage was disabled!", QSTRING_CSTR(_groupId) );
|
Error(_log, "Group-ID [%s] is not usable - Entertainment API usage was disabled!", QSTRING_CSTR(_groupId) );
|
||||||
}
|
}
|
||||||
lights = _devConfig[ CONFIG_lightIdS ].toVariant().toStringList();
|
lights = _devConfig[ CONFIG_LIGHTIDS ].toVariant().toStringList();
|
||||||
}
|
}
|
||||||
|
|
||||||
_lightIds = lights;
|
_lightIds = lights;
|
||||||
|
@ -23,7 +23,7 @@ namespace {
|
|||||||
const char CONFIG_RAZER_DEVICE_TYPE[] = "subType";
|
const char CONFIG_RAZER_DEVICE_TYPE[] = "subType";
|
||||||
const char CONFIG_SINGLE_COLOR[] = "singleColor";
|
const char CONFIG_SINGLE_COLOR[] = "singleColor";
|
||||||
|
|
||||||
// WLED JSON-API elements
|
// API elements
|
||||||
const char API_DEFAULT_HOST[] = "localhost";
|
const char API_DEFAULT_HOST[] = "localhost";
|
||||||
const int API_DEFAULT_PORT = 54235;
|
const int API_DEFAULT_PORT = 54235;
|
||||||
|
|
||||||
|
135
libsrc/leddevice/schemas/schema-homeassistant.json
Normal file
135
libsrc/leddevice/schemas/schema-homeassistant.json
Normal file
@ -0,0 +1,135 @@
|
|||||||
|
{
|
||||||
|
"type": "object",
|
||||||
|
"required": true,
|
||||||
|
"properties": {
|
||||||
|
"hostList": {
|
||||||
|
"type": "string",
|
||||||
|
"title": "edt_dev_spec_devices_discovered_title",
|
||||||
|
"enum": [ "NONE" ],
|
||||||
|
"options": {
|
||||||
|
"enum_titles": [ "edt_dev_spec_devices_discovery_inprogress" ],
|
||||||
|
"infoText": "edt_dev_spec_devices_discovered_title_info"
|
||||||
|
},
|
||||||
|
"required": true,
|
||||||
|
"propertyOrder": 1
|
||||||
|
},
|
||||||
|
"host": {
|
||||||
|
"type": "string",
|
||||||
|
"format": "hostname_or_ip",
|
||||||
|
"title": "edt_dev_spec_targetIpHost_title",
|
||||||
|
"options": {
|
||||||
|
"infoText": "edt_dev_spec_targetIpHost_title_info"
|
||||||
|
},
|
||||||
|
"required": true,
|
||||||
|
"propertyOrder": 2
|
||||||
|
},
|
||||||
|
"port": {
|
||||||
|
"type": "integer",
|
||||||
|
"title": "edt_dev_spec_port_title",
|
||||||
|
"default": 8123,
|
||||||
|
"minimum": 0,
|
||||||
|
"maximum": 65535,
|
||||||
|
"access": "expert",
|
||||||
|
"propertyOrder": 3
|
||||||
|
},
|
||||||
|
"token": {
|
||||||
|
"type": "string",
|
||||||
|
"title": "edt_dev_auth_key_title",
|
||||||
|
"options": {
|
||||||
|
"infoText": "edt_dev_auth_key_title_info"
|
||||||
|
},
|
||||||
|
"propertyOrder": 4
|
||||||
|
},
|
||||||
|
"restoreOriginalState": {
|
||||||
|
"type": "boolean",
|
||||||
|
"format": "checkbox",
|
||||||
|
"title": "edt_dev_spec_restoreOriginalState_title",
|
||||||
|
"default": true,
|
||||||
|
"required": true,
|
||||||
|
"options": {
|
||||||
|
"hidden": true,
|
||||||
|
"infoText": "edt_dev_spec_restoreOriginalState_title_info"
|
||||||
|
},
|
||||||
|
"propertyOrder": 5
|
||||||
|
},
|
||||||
|
"overwriteBrightness": {
|
||||||
|
"type": "boolean",
|
||||||
|
"format": "checkbox",
|
||||||
|
"title": "edt_dev_spec_brightnessOverwrite_title",
|
||||||
|
"default": true,
|
||||||
|
"required": true,
|
||||||
|
"access": "advanced",
|
||||||
|
"propertyOrder": 5
|
||||||
|
},
|
||||||
|
"brightness": {
|
||||||
|
"type": "integer",
|
||||||
|
"title": "edt_dev_spec_brightness_title",
|
||||||
|
"default": 255,
|
||||||
|
"minimum": 1,
|
||||||
|
"maximum": 255,
|
||||||
|
"options": {
|
||||||
|
"dependencies": {
|
||||||
|
"overwriteBrightness": true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"access": "advanced",
|
||||||
|
"propertyOrder": 6
|
||||||
|
},
|
||||||
|
"fullBrightnessAtStart": {
|
||||||
|
"type": "boolean",
|
||||||
|
"format": "checkbox",
|
||||||
|
"title": "edt_dev_spec_fullBrightnessAtStart_title",
|
||||||
|
"default": true,
|
||||||
|
"required": true,
|
||||||
|
"access": "advanced",
|
||||||
|
"propertyOrder": 7
|
||||||
|
},
|
||||||
|
"switchOffOnBlack": {
|
||||||
|
"type": "boolean",
|
||||||
|
"format": "checkbox",
|
||||||
|
"title": "edt_dev_spec_switchOffOnBlack_title",
|
||||||
|
"default": false,
|
||||||
|
"access": "advanced",
|
||||||
|
"propertyOrder": 8
|
||||||
|
},
|
||||||
|
"transitionTime": {
|
||||||
|
"type": "integer",
|
||||||
|
"title": "edt_dev_spec_transistionTime_title",
|
||||||
|
"default": 0,
|
||||||
|
"append": "ms",
|
||||||
|
"minimum": 0,
|
||||||
|
"maximum": 2000,
|
||||||
|
"required": false,
|
||||||
|
"access": "advanced",
|
||||||
|
"propertyOrder": 9
|
||||||
|
},
|
||||||
|
"entityIds": {
|
||||||
|
"title": "edt_dev_spec_lightid_title",
|
||||||
|
"type": "array",
|
||||||
|
"required": true,
|
||||||
|
"format": "select",
|
||||||
|
"options": {
|
||||||
|
"hidden": true
|
||||||
|
},
|
||||||
|
"items": {
|
||||||
|
"type": "string",
|
||||||
|
"title": "edt_dev_spec_lights_itemtitle"
|
||||||
|
},
|
||||||
|
"propertyOrder": 10
|
||||||
|
},
|
||||||
|
"latchTime": {
|
||||||
|
"type": "integer",
|
||||||
|
"title": "edt_dev_spec_latchtime_title",
|
||||||
|
"default": 250,
|
||||||
|
"append": "edt_append_ms",
|
||||||
|
"minimum": 100,
|
||||||
|
"maximum": 2000,
|
||||||
|
"access": "expert",
|
||||||
|
"options": {
|
||||||
|
"infoText": "edt_dev_spec_latchtime_title_info"
|
||||||
|
},
|
||||||
|
"propertyOrder": 11
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"additionalProperties": true
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user