diff --git a/docs/LICENSE b/LICENSE similarity index 100% rename from docs/LICENSE rename to LICENSE diff --git a/README.md b/README.md index aa919f5..3cec786 100644 --- a/README.md +++ b/README.md @@ -47,6 +47,7 @@ The following is a list of configs for the card: | lux_dongle | list of strings | This is the LuxPower inverter's dongle number. It will later on be used to call the refresh service. (This requires the LuxPowerTek integration that supports this.) This or the inverter alias list are required if there are more than 1 inverter. | | inverter_alias | list of strings | This is used when there is more than 1 inverter. This will be the names used in the dropdown list. This or the lux dongle list is required. | | refresh_button | string | The location of the refresh button. Can be 'left', 'right' or 'both'. See below for more information. **NOTE:** the refresh button will only show if the *lux_dongle* is added. | +| title | string | A title for the card. | #### Sub-configs that are not a list of entities or values diff --git a/config-entity-functions.js b/config-entity-functions.js index 10fcbf4..f983bce 100755 --- a/config-entity-functions.js +++ b/config-entity-functions.js @@ -1,32 +1,27 @@ import * as constants from "./constants.js"; -const required_entries = [ - "inverter_count", - "battery_soc", - "battery_flow", - "home_consumption", - "grid_flow" -] +const required_entries = ["inverter_count", "battery_soc", "battery_flow", "home_consumption", "grid_flow"]; export function buildConfig(config) { function deepcopy(value) { - if (!(!!value && typeof value == 'object')) { + if (!(!!value && typeof value == "object")) { return value; } - if (Object.prototype.toString.call(value) == '[object Date]') { + if (Object.prototype.toString.call(value) == "[object Date]") { return new Date(value.getTime()); } if (Array.isArray(value)) { return value.map(deepcopy); } const result = {}; - Object.keys(value).forEach( - function(key) { result[key] = deepcopy(value[key]); }); + Object.keys(value).forEach(function (key) { + result[key] = deepcopy(value[key]); + }); return result; } config = deepcopy(config); - const new_config = deepcopy(constants.base_config) + const new_config = deepcopy(constants.base_config); new_config.title = config.title; @@ -244,9 +239,9 @@ export function getEntitiesUnit(config, hass, config_entity, index) { const handleEntityError = (config_entry, entity_name) => { if (required_entries.includes(config_entry)) { - throw new Error(`Invalid entity: ${entity_name} for config_entry`) ; + throw new Error(`Invalid entity: ${entity_name} for config_entry`); } -} +}; export function getStatusMessage(status_code, show_no_grid_as_warning) { var status_level = 0; diff --git a/constants.js b/constants.js index e538839..4562df7 100755 --- a/constants.js +++ b/constants.js @@ -107,7 +107,7 @@ export function getBase64Data(image_name) { case "inverter": return ``; case "parallel-inverter": - return ``; + return ``; case "solar": return ``; default: diff --git a/docs/changelog.md b/docs/changelog.md index 3880e44..1371eb6 100644 --- a/docs/changelog.md +++ b/docs/changelog.md @@ -1,3 +1,88 @@ -# 0.1.0 +## Changelog + +### v0.1.0 - Initial Release + +### v0.2.0 + +- Reworked the card with better practices. + +### v0.2.2 + +Enhancements + +- Implemented HACS validations actions. +- Updated README for HACS requirements. +- Moved README to root folder. + +Bugfixes + +- Fixed arrow-animation issus +- Cropped arrow image, final result is that the arrows are larger and closer together in the final card. + +### v0.3.0 + +Enhancements + +- The refresh button can be shows by adding the correct config, but unfortunately the service isn't called when pressing the button. When this issue is solved, the rest of the interactions should be simple +- Images stored as base64. Issue can be closed when v03.0 is made the latest release. +- Keeping the README up to date is a continuous process but it is sufficient for use and HACS. +- PR for adding to HACS is pending, but the card can be added as a custom repository. +- There are now 3 ways of showing grid status. The LuxPower integration will still require fine tuning. +- Added label to show when last the values were updated, and also how long ago that was (if the entity has a timestamp attribute) + +Bugfixes + +- Improved formatting and scaling but text is not at a point where I am satisfied with. + +### v0.4.0 + +Features + +- Added the functionality to see the entity history and the refresh button works. + +### v0.4.2 + +Bugfix + +- Changed styles to accommodate safari browsers. + +### v1.0.0 + +Features + +- Parallel inverters + - v1.0.0 implements parallel inverters. Adding a second inverter will allow you to choose which inverter's info you want to see. Blending the info is the next step +- Card now uses status codes directly from the integration and gives a short description based on that. + +Breaking changes + +- v1.0.0 implements a new config format. This will break existing cards until the new config is implemented. Please refer to the README file for information. + +### v1.1.0 + +Features + +- Added mixing between parallel inverters +- Refresh button on Parallel page will. refresh both inverters + +### v1.1.1 + +Bugfix + +- For parallel page, all values are added except for battery SOC and voltages. + +### v1.1.2 + +Bugfixes + +- Round values to max of 2 decimal places. +- Reworked code to fix issue with battery arrows not showing. +- Reverted LitElement to HTMLElement. + +### v1.2.0 + +Features + +- Updated the status indicator to allow for parallel inverters. diff --git a/html-functions.js b/html-functions.js index 8a82dc7..95b58eb 100755 --- a/html-functions.js +++ b/html-functions.js @@ -14,7 +14,6 @@ export function generateStyles(config) { grid-template-rows: repeat(${config.pv_power.is_used ? 5 : 4}, 1fr); padding-left: 5px; padding-right: 5px; - padding-top: 1px; } .status-grid { display: grid; @@ -22,7 +21,7 @@ export function generateStyles(config) { grid-template-rows: repeat(1, 1fr); padding-left: 5px; padding-right: 5px; - padding-top: 4px; + padding-top: ${config.title ? 0 : 30}px; } .diagram-grid img { max-width: 100%; @@ -53,7 +52,7 @@ export function generateStyles(config) { text-align: right; } */ .header-text { - font-size: min(4vw, 1.17em); + font-size: min(4vw, 1em); font-weight: bold; line-height: 1; margin: 0; @@ -64,6 +63,7 @@ export function generateStyles(config) { } .sub-text { font-size: min(2.5vw, 0.95em); + color: var(--secondary-text-color); line-height: 1; margin: 0; padding-left: 3px; @@ -181,7 +181,6 @@ export function generateStyles(config) { line-height: 1; padding-left: 5px; padding-right: 5px; - padding-bottom: 5px; } .grid-status { text-align: right; @@ -303,9 +302,7 @@ export function generateGrid(config) { cells += `
`; // Home info cells += `
`; // Home image cells += `
`; // Power allocation arrows - cells += `
`; // Power allocation image + cells += `
`; // Power allocation image cells += `
`; // Power allocation info } else { cells += `
${refresh_button_left}
`; @@ -322,7 +319,7 @@ export function generateGrid(config) { export function generateDateTime(config) { let date_time_info = ``; if (config.update_time.is_used) { - let date_time_info = ` + date_time_info = `

Last update at: -

`; if (config.update_time.show_last_update) { diff --git a/lux-power-distribution-card.js b/lux-power-distribution-card.js index d7dd4e8..8477f03 100755 --- a/lux-power-distribution-card.js +++ b/lux-power-distribution-card.js @@ -5,7 +5,7 @@ import * as constants from "./constants.js"; class LuxPowerDistributionCard extends HTMLElement { constructor() { super(); - this.attachShadow({ mode: 'open' }); + this.attachShadow({ mode: "open" }); } set hass(hass) { @@ -28,19 +28,18 @@ class LuxPowerDistributionCard extends HTMLElement { if (!this._hass || !this._config) { return; } - const shadowRoot = this.shadowRoot; - this.card = document.createElement('ha-card'); + this.card = document.createElement("ha-card"); if (this._config.title) { const header = document.createElement("h1"); - header.classList.add('card-header'); + header.classList.add("card-header"); header.appendChild(document.createTextNode(this._config.title)); this.card.appendChild(header); } - this.content = document.createElement('div'); - this.content.classList.add('card-content'); + this.content = document.createElement("div"); + this.content.classList.add("card-content"); this.content.innerHTML = ` @@ -80,63 +79,43 @@ class LuxPowerDistributionCard extends HTMLElement { this.updateHome(index); this.updateAllocatedPower(); this.updateDateTime(index); + this.updateInverterPic(index); } } bindRefresh(card, hass, config) { - let refresh_button_left = card.querySelector("#refresh-button-left"); - if (refresh_button_left) { - refresh_button_left.addEventListener("click", function (source) { - let index = 0; - if (config.inverter_count > 1) { - const inverter_selector_element = card.querySelector("#inverter-selector"); - if (inverter_selector_element) { - let select_value = inverter_selector_element.value; - let parsed_value = parseInt(select_value); - if (!isNaN(parsed_value)) { - index = parsed_value; + const refresh_list = ["#refresh-button-left", "#refresh-button-right"]; + + for (let i = 0; i < refresh_list.length; i++) { + let refresh_button = card.querySelector(refresh_list[i]); + if (refresh_button) { + refresh_button.addEventListener("click", function (source) { + let index = 0; + if (config.inverter_count > 1) { + const inverter_selector_element = card.querySelector("#inverter-selector"); + if (inverter_selector_element) { + let select_value = inverter_selector_element.value; + let parsed_value = parseInt(select_value); + if (!isNaN(parsed_value)) { + index = parsed_value; + } } } - } - if (index == this._config.inverter_count) { - for (let i = 0; i < this._config.inverter_count; i++) { - hass.callService("luxpower", "luxpower_refresh_registers", { - dongle: this._config.lux_dongle.values[i], - }); - } - } else { - hass.callService("luxpower", "luxpower_refresh_registers", { - dongle: this._config.lux_dongle.values[index], - }); - } - }); - } - let refresh_button_right = card.querySelector("#refresh-button-right"); - if (refresh_button_right) { - refresh_button_right.addEventListener("click", function (source) { - let index = 0; - if (config.inverter_count > 1) { - const inverter_selector_element = card.querySelector("#inverter-selector"); - if (inverter_selector_element) { - let select_value = inverter_selector_element.value; - let parsed_value = parseInt(select_value); - if (!isNaN(parsed_value)) { - index = parsed_value; + if (index == config.inverter_count) { + for (let j = 0; j < config.inverter_count; j++) { + console.log(config.lux_dongle.values[i]); + hass.callService("luxpower", "luxpower_refresh_registers", { + dongle: config.lux_dongle.values[i], + }); + console.log(config.lux_dongle.values[i]); } - } - } - if (index == this._config.inverter_count) { - for (let i = 0; i < this._config.inverter_count; i++) { + } else { hass.callService("luxpower", "luxpower_refresh_registers", { - dongle: this._config.lux_dongle.values[i], + dongle: config.lux_dongle.values[index], }); } - } else { - hass.callService("luxpower", "luxpower_refresh_registers", { - dongle: this._config.lux_dongle.values[index], - }); - } - }); + }); + } } } @@ -171,7 +150,7 @@ class LuxPowerDistributionCard extends HTMLElement { composed: true, }); event.detail = { - entityId: this._config[value].entities[index], + entityId: config[value].entities[index], }; card.dispatchEvent(event); return event; @@ -260,7 +239,7 @@ class LuxPowerDistributionCard extends HTMLElement { solar_info_element.innerHTML = `

${this.formatPowerStates("pv_power", pv_power, index)}

-

${pv_power > 0 ? "Solar Import" : ""}

+

${pv_power > 0 ? "PV Power" : ""}

`; } @@ -289,31 +268,22 @@ class LuxPowerDistributionCard extends HTMLElement { battery_charge_info_element.innerHTML = `

${this.formatPowerStates("battery_flow", battery_flow, index)}

-

${ - battery_flow > 0 ? "Battery Charging" : battery_flow < 0 ? "Battery Discharging" : "Idle" - }

+

${battery_flow > 0 ? "Charge Power" : battery_flow < 0 ? "Discharge Power" : "Idle"}

`; } var battery_voltage = ""; if (this._config.battery_voltage.is_used && index != -1) { battery_voltage = `${cef.getEntitiesState(this._config, this._hass, "battery_voltage", index, false)} Vdc`; - } else if (this._config.parallel.average_voltage) { - battery_voltage = `${cef.getEntitiesNumState( - this._config, - this._hass, - "battery_voltage", - index, - false, - true - )} Vdc (avg)`; + } else if (this._config.battery_voltage.is_used && this._config.parallel.average_voltage) { + battery_voltage = `${cef.getEntitiesNumState(this._config, this._hass, "battery_voltage", index, false, true)} Vdc (avg)`; } const battery_soc_info_element = this.shadowRoot.querySelector("#battery-soc-info"); if (battery_soc_info_element) { battery_soc_info_element.innerHTML = `
+ ${this._config.battery_voltage.is_used ? `

${battery_voltage}

` : ``}

${battery_soc}%

-

${battery_voltage}

`; } @@ -338,10 +308,7 @@ class LuxPowerDistributionCard extends HTMLElement { var grid_voltage = parseInt(cef.getEntitiesState(this._config, this._hass, "grid_voltage", index)); const grid_image_element = this.shadowRoot.querySelector("#grid-image"); if (this._config.grid_indicator.hue) { - grid_image_element.setAttribute( - "class", - grid_voltage == 0 ? `cell image-cell blend-overlay` : `cell image-cell` - ); + grid_image_element.setAttribute("class", grid_voltage == 0 ? `cell image-cell blend-overlay` : `cell image-cell`); } if (this._config.grid_indicator.dot) { grid_emoji = grid_voltage == 0 ? ` 🔴` : ``; @@ -356,14 +323,7 @@ class LuxPowerDistributionCard extends HTMLElement { if (index != -1) { grid_voltage = `${cef.getEntitiesState(this._config, this._hass, "grid_voltage", index)} Vac${grid_emoji}`; } else if (this._config.parallel.average_voltage) { - grid_voltage = `${cef.getEntitiesNumState( - this._config, - this._hass, - "grid_voltage", - index, - false, - true - )} Vac (avg)${grid_emoji}`; + grid_voltage = `${cef.getEntitiesNumState(this._config, this._hass, "grid_voltage", index, false, true)} Vac (avg)${grid_emoji}`; } } grid_info_element.innerHTML = ` @@ -403,8 +363,8 @@ class LuxPowerDistributionCard extends HTMLElement { home_info_element.innerHTML = `
-

${sub_text}

${value}

+

${sub_text}

`; } @@ -422,18 +382,15 @@ class LuxPowerDistributionCard extends HTMLElement { oldest_time = Date.parse(cef.getEntitiesState(this._config, this._hass, "update_time", i)); } } - update_time_element.innerHTML = `${ - this._config.inverter_alias.values[olderst_index] - } updated at: ${cef.getEntitiesState(this._config, this._hass, "update_time", olderst_index)}`; + update_time_element.innerHTML = `${this._config.inverter_alias.values[olderst_index]} updated at: ${cef.getEntitiesState( + this._config, + this._hass, + "update_time", + olderst_index + )}`; const since_time_element = this.shadowRoot.querySelector("#since-info"); if (since_time_element) { - var last_time_ts = cef.getEntitiesAttribute( - this._config, - this._hass, - "update_time", - "timestamp", - olderst_index - ); + var last_time_ts = cef.getEntitiesAttribute(this._config, this._hass, "update_time", "timestamp", olderst_index); var time_now = Date.now() / 1000; var diff = time_now - last_time_ts; @@ -458,12 +415,7 @@ class LuxPowerDistributionCard extends HTMLElement { } else { const update_time_element = this.shadowRoot.querySelector("#time-info"); if (update_time_element) { - update_time_element.innerHTML = `Last update at: ${cef.getEntitiesState( - this._config, - this._hass, - "update_time", - index - )}`; + update_time_element.innerHTML = `Last update at: ${cef.getEntitiesState(this._config, this._hass, "update_time", index)}`; } const since_time_element = this.shadowRoot.querySelector("#since-info"); if (since_time_element) { @@ -504,14 +456,25 @@ class LuxPowerDistributionCard extends HTMLElement { const power_allocation_info_element = this.shadowRoot.querySelector("#power-allocation-info"); power_allocation_info_element.innerHTML = `
+

${parseInt(this.getAllocatedPower())} W

Allocated Power

-

${parseInt(this.getAllocatedPower())} W

`; } } } + updateInverterPic(index) { + const inverter_pic_element = this.shadowRoot.querySelector("#inverter-image"); + if (inverter_pic_element) { + if (index == -1) { + inverter_pic_element.innerHTML = ``; + } else { + inverter_pic_element.innerHTML = ``; + } + } + } + getAllocatedPower() { let allocatedEnergy = 0; if (this._config.energy_allocations.is_used) {