From 5e85f5a150e9fcbd70c370b62cc5e33a5c930e93 Mon Sep 17 00:00:00 2001 From: Chris Vincent Date: Tue, 7 May 2024 00:01:15 -0600 Subject: [PATCH] Add sensor summation to area card --- src/panels/lovelace/cards/hui-area-card.ts | 192 ++++++++++++------ .../config-elements/hui-area-card-editor.ts | 11 +- 2 files changed, 135 insertions(+), 68 deletions(-) diff --git a/src/panels/lovelace/cards/hui-area-card.ts b/src/panels/lovelace/cards/hui-area-card.ts index d11cab94756e..0afe89baabc9 100644 --- a/src/panels/lovelace/cards/hui-area-card.ts +++ b/src/panels/lovelace/cards/hui-area-card.ts @@ -69,9 +69,61 @@ const TOGGLE_DOMAINS = ["light", "switch", "fan"]; const OTHER_DOMAINS = ["camera"]; -export const DEVICE_CLASSES = { - sensor: ["temperature", "humidity"], - binary_sensor: ["motion", "moisture"], +const ALL_DOMAINS = [ + ...SENSOR_DOMAINS, + ...ALERT_DOMAINS, + ...TOGGLE_DOMAINS, + ...OTHER_DOMAINS, +]; + +const AVG_SENSOR_CLASSES = ["temperature", "humidity"]; + +const SUM_SENSOR_CLASSES = ["power", "energy", "volume"]; + +const BINARY_SENSOR_CLASSES = ["motion", "moisture"]; + +const SENSOR_CLASSES = [ + ...AVG_SENSOR_CLASSES, + ...SUM_SENSOR_CLASSES, + ...BINARY_SENSOR_CLASSES, +]; + +type DeviceClassOptions = { + sensor_type: string; + default_display_function: (values: number[]) => number; +}; + +type DeviceClasses = { [key: string]: DeviceClassOptions }; + +export const deviceClassesByDomain = ( + deviceClasses: DeviceClasses, + domain: string +): string[] => + Object.entries(deviceClasses) + .filter(([_deviceClass, opts]) => opts.sensor_type === domain) + .map(([deviceClass, _opts]) => deviceClass); + +const sumValues = (values: number[]): number => + values.reduce((total, value) => total + value, 0); + +const avgValues = (values: number[]): number => + sumValues(values) / values.length; + +export const DEVICE_CLASSES: { [key: string]: DeviceClassOptions } = { + ...SENSOR_CLASSES.reduce( + (acc, dc) => ({ + ...acc, + [dc]: { + sensor_type: BINARY_SENSOR_CLASSES.includes(dc) + ? "binary_sensor" + : "sensor", + default_display_function: SUM_SENSOR_CLASSES.includes(dc) + ? sumValues + : avgValues, + }, + }), + {} + ), }; const DOMAIN_ICONS = { @@ -111,7 +163,7 @@ export class HuiAreaCard @state() private _areas?: AreaRegistryEntry[]; - private _deviceClasses: { [key: string]: string[] } = DEVICE_CLASSES; + private _deviceClasses: DeviceClasses = DEVICE_CLASSES; private _ratio: { w: number; @@ -123,7 +175,7 @@ export class HuiAreaCard areaId: string, devicesInArea: Set, registryEntities: EntityRegistryEntry[], - deviceClasses: { [key: string]: string[] }, + deviceClasses: DeviceClasses, states: HomeAssistant["states"] ) => { const entitiesInArea = registryEntities @@ -141,12 +193,7 @@ export class HuiAreaCard for (const entity of entitiesInArea) { const domain = computeDomain(entity); - if ( - !TOGGLE_DOMAINS.includes(domain) && - !SENSOR_DOMAINS.includes(domain) && - !ALERT_DOMAINS.includes(domain) && - !OTHER_DOMAINS.includes(domain) - ) { + if (!ALL_DOMAINS.includes(domain)) { continue; } const stateObj: HassEntity | undefined = states[entity]; @@ -156,8 +203,8 @@ export class HuiAreaCard } if ( - (SENSOR_DOMAINS.includes(domain) || ALERT_DOMAINS.includes(domain)) && - !deviceClasses[domain].includes( + [...SENSOR_DOMAINS, ...ALERT_DOMAINS].includes(domain) && + !deviceClassesByDomain(deviceClasses, domain).includes( stateObj.attributes.device_class || "" ) ) { @@ -197,7 +244,10 @@ export class HuiAreaCard ); } - private _average(domain: string, deviceClass?: string): string | undefined { + private _sensor_display_value( + domain: string, + deviceClass?: string + ): string | undefined { const entities = this._entitiesByDomain( this._config!.area, this._devicesInArea(this._config!.area, this._devices!), @@ -211,24 +261,25 @@ export class HuiAreaCard return undefined; } let uom; - const values = entities.filter((entity) => { - if (!isNumericState(entity) || isNaN(Number(entity.state))) { - return false; - } - if (!uom) { - uom = entity.attributes.unit_of_measurement; - return true; - } - return entity.attributes.unit_of_measurement === uom; - }); + const values = entities + .filter((entity) => { + if (!isNumericState(entity) || isNaN(Number(entity.state))) { + return false; + } + if (!uom) { + uom = entity.attributes.unit_of_measurement; + return true; + } + return entity.attributes.unit_of_measurement === uom; + }) + .map((entity) => Number(entity.state)); if (!values.length) { return undefined; } - const sum = values.reduce( - (total, entity) => total + Number(entity.state), - 0 - ); - return `${formatNumber(sum / values.length, this.hass!.locale, { + const displayValue = deviceClass + ? this._deviceClasses[deviceClass].default_display_function(values) + : ""; + return `${formatNumber(displayValue, this.hass!.locale, { maximumFractionDigits: 1, })}${uom ? blankBeforeUnit(uom, this.hass!.locale) : ""}${uom || ""}`; } @@ -274,13 +325,22 @@ export class HuiAreaCard this._config = config; - this._deviceClasses = { ...DEVICE_CLASSES }; - if (config.sensor_classes) { - this._deviceClasses.sensor = config.sensor_classes; - } - if (config.alert_classes) { - this._deviceClasses.binary_sensor = config.alert_classes; + if (!config.sensor_classes && !config.alert_classes) { + this._deviceClasses = { ...DEVICE_CLASSES }; + return; } + + this._deviceClasses = Object.entries(DEVICE_CLASSES) + .filter(([dc, _opts]) => + [...config.sensor_classes, ...config.alert_classes].includes(dc) + ) + .reduce( + (acc, [dc, opts]) => ({ + ...acc, + [dc]: opts, + }), + {} + ); } protected shouldUpdate(changedProps: PropertyValues): boolean { @@ -382,24 +442,26 @@ export class HuiAreaCard if (!(domain in entitiesByDomain)) { return; } - this._deviceClasses[domain].forEach((deviceClass) => { - if ( - entitiesByDomain[domain].some( - (entity) => entity.attributes.device_class === deviceClass - ) - ) { - sensors.push(html` -
- - ${this._average(domain, deviceClass)} -
- `); + deviceClassesByDomain(this._deviceClasses, domain).forEach( + (deviceClass) => { + if ( + entitiesByDomain[domain].some( + (entity) => entity.attributes.device_class === deviceClass + ) + ) { + sensors.push(html` +
+ + ${this._sensor_display_value(domain, deviceClass)} +
+ `); + } } - }); + ); }); let cameraEntityId: string | undefined; @@ -448,18 +510,20 @@ export class HuiAreaCard if (!(domain in entitiesByDomain)) { return nothing; } - return this._deviceClasses[domain].map((deviceClass) => { - const entity = this._isOn(domain, deviceClass); - return entity - ? html` - - ` - : nothing; - }); + return deviceClassesByDomain(this._deviceClasses, domain).map( + (deviceClass) => { + const entity = this._isOn(domain, deviceClass); + return entity + ? html` + + ` + : nothing; + } + ); })}
diff --git a/src/panels/lovelace/editor/config-elements/hui-area-card-editor.ts b/src/panels/lovelace/editor/config-elements/hui-area-card-editor.ts index f46f279b4580..714d20027d38 100644 --- a/src/panels/lovelace/editor/config-elements/hui-area-card-editor.ts +++ b/src/panels/lovelace/editor/config-elements/hui-area-card-editor.ts @@ -15,6 +15,7 @@ import "../../../../components/ha-form/ha-form"; import { DEFAULT_ASPECT_RATIO, DEVICE_CLASSES, + deviceClassesByDomain, } from "../../cards/hui-area-card"; import type { SchemaUnion } from "../../../../components/ha-form/types"; import type { HomeAssistant } from "../../../../types"; @@ -204,11 +205,13 @@ export class HuiAreaCardEditor ); const binarySelectOptions = this._buildBinaryOptions( possibleBinaryClasses, - this._config.alert_classes || DEVICE_CLASSES.binary_sensor + this._config.alert_classes || + deviceClassesByDomain(DEVICE_CLASSES, "binary_sensor") ); const sensorSelectOptions = this._buildSensorOptions( possibleSensorClasses, - this._config.sensor_classes || DEVICE_CLASSES.sensor + this._config.sensor_classes || + deviceClassesByDomain(DEVICE_CLASSES, "sensor") ); const schema = this._schema( @@ -219,8 +222,8 @@ export class HuiAreaCardEditor const data = { camera_view: "auto", - alert_classes: DEVICE_CLASSES.binary_sensor, - sensor_classes: DEVICE_CLASSES.sensor, + alert_classes: deviceClassesByDomain(DEVICE_CLASSES, "binary_sensor"), + sensor_classes: deviceClassesByDomain(DEVICE_CLASSES, "sensor"), ...this._config, };