From b538270704258cd593149961dfb2469e2b72e5c0 Mon Sep 17 00:00:00 2001 From: Joseph Abbey Date: Tue, 7 Nov 2023 13:57:54 +0000 Subject: [PATCH 1/5] hui-number-tile-feature --- .../create-tile-feature-element.ts | 2 + .../hui-number-tile-feature-editor.ts | 90 +++++++++++++ .../hui-tile-card-features-editor.ts | 4 + .../tile-features/hui-number-tile-feature.ts | 121 ++++++++++++++++++ src/panels/lovelace/tile-features/types.ts | 8 +- src/translations/en.json | 8 ++ 6 files changed, 232 insertions(+), 1 deletion(-) create mode 100644 src/panels/lovelace/editor/config-elements/hui-number-tile-feature-editor.ts create mode 100644 src/panels/lovelace/tile-features/hui-number-tile-feature.ts diff --git a/src/panels/lovelace/create-element/create-tile-feature-element.ts b/src/panels/lovelace/create-element/create-tile-feature-element.ts index b0cc259cb2ac..0f608494c9c5 100644 --- a/src/panels/lovelace/create-element/create-tile-feature-element.ts +++ b/src/panels/lovelace/create-element/create-tile-feature-element.ts @@ -13,6 +13,7 @@ import "../tile-features/hui-select-options-tile-feature"; import "../tile-features/hui-target-temperature-tile-feature"; import "../tile-features/hui-vacuum-commands-tile-feature"; import "../tile-features/hui-water-heater-operation-modes-tile-feature"; +import "../tile-features/hui-number-tile-feature"; import { LovelaceTileFeatureConfig } from "../tile-features/types"; import { createLovelaceElement, @@ -35,6 +36,7 @@ const TYPES: Set = new Set([ "target-temperature", "vacuum-commands", "water-heater-operation-modes", + "number", ]); export const createTileFeatureElement = (config: LovelaceTileFeatureConfig) => diff --git a/src/panels/lovelace/editor/config-elements/hui-number-tile-feature-editor.ts b/src/panels/lovelace/editor/config-elements/hui-number-tile-feature-editor.ts new file mode 100644 index 000000000000..ac0b5d78cec1 --- /dev/null +++ b/src/panels/lovelace/editor/config-elements/hui-number-tile-feature-editor.ts @@ -0,0 +1,90 @@ +import { html, LitElement, nothing } from "lit"; +import { customElement, property, state } from "lit/decorators"; +import memoizeOne from "memoize-one"; +import { fireEvent } from "../../../../common/dom/fire_event"; +import "../../../../components/ha-form/ha-form"; +import type { SchemaUnion } from "../../../../components/ha-form/types"; +import type { HomeAssistant } from "../../../../types"; +import { + NumberTileFeatureConfig, + LovelaceTileFeatureContext, +} from "../../tile-features/types"; +import type { LovelaceTileFeatureEditor } from "../../types"; +import { LocalizeFunc } from "../../../../common/translations/localize"; + +@customElement("hui-number-tile-feature-editor") +export class HuiNumberTileFeatureEditor + extends LitElement + implements LovelaceTileFeatureEditor +{ + @property({ attribute: false }) public hass?: HomeAssistant; + + @property({ attribute: false }) public context?: LovelaceTileFeatureContext; + + @state() private _config?: NumberTileFeatureConfig; + + public setConfig(config: NumberTileFeatureConfig): void { + this._config = config; + } + + private _schema = memoizeOne( + (localize: LocalizeFunc) => + [ + { + name: "style", + selector: { + select: { + multiple: false, + mode: "list", + options: ["slider", "buttons"].map((mode) => ({ + value: mode, + label: localize( + `ui.panel.lovelace.editor.card.tile.features.types.number.style_list.${mode}` + ), + })), + }, + }, + }, + ] as const + ); + + protected render() { + if (!this.hass || !this._config) { + return nothing; + } + + const data: NumberTileFeatureConfig = { + style: "buttons", + ...this._config, + }; + + const schema = this._schema(this.hass.localize); + + return html` + + `; + } + + private _valueChanged(ev: CustomEvent): void { + fireEvent(this, "config-changed", { config: ev.detail.value }); + } + + private _computeLabelCallback = ( + schema: SchemaUnion> + ) => + this.hass!.localize( + `ui.panel.lovelace.editor.card.tile.features.types.number.${schema.name}` + ); +} + +declare global { + interface HTMLElementTagNameMap { + "hui-number-tile-feature-editor": HuiNumberTileFeatureEditor; + } +} diff --git a/src/panels/lovelace/editor/config-elements/hui-tile-card-features-editor.ts b/src/panels/lovelace/editor/config-elements/hui-tile-card-features-editor.ts index a4b3ea2ad26a..2c4a60b8b52c 100644 --- a/src/panels/lovelace/editor/config-elements/hui-tile-card-features-editor.ts +++ b/src/panels/lovelace/editor/config-elements/hui-tile-card-features-editor.ts @@ -38,6 +38,7 @@ import { supportsVacuumCommandTileFeature } from "../../tile-features/hui-vacuum import { supportsWaterHeaterOperationModesTileFeature } from "../../tile-features/hui-water-heater-operation-modes-tile-feature"; import { LovelaceTileFeatureConfig } from "../../tile-features/types"; import { supportsClimatePresetModesTileFeature } from "../../tile-features/hui-climate-preset-modes-tile-feature"; +import { supportsNumberTileFeature } from "../../tile-features/hui-number-tile-feature"; type FeatureType = LovelaceTileFeatureConfig["type"]; type SupportsFeature = (stateObj: HassEntity) => boolean; @@ -58,6 +59,7 @@ const UI_FEATURE_TYPES = [ "target-temperature", "vacuum-commands", "water-heater-operation-modes", + "number", ] as const satisfies readonly FeatureType[]; type UiFeatureTypes = (typeof UI_FEATURE_TYPES)[number]; @@ -69,6 +71,7 @@ const EDITABLES_FEATURE_TYPES = new Set([ "water-heater-operation-modes", "lawn-mower-commands", "climate-preset-modes", + "number", ]); const SUPPORTS_FEATURE_TYPES: Record< @@ -90,6 +93,7 @@ const SUPPORTS_FEATURE_TYPES: Record< "vacuum-commands": supportsVacuumCommandTileFeature, "water-heater-operation-modes": supportsWaterHeaterOperationModesTileFeature, "select-options": supportsSelectOptionTileFeature, + number: supportsNumberTileFeature, }; const CUSTOM_FEATURE_ENTRIES: Record< diff --git a/src/panels/lovelace/tile-features/hui-number-tile-feature.ts b/src/panels/lovelace/tile-features/hui-number-tile-feature.ts new file mode 100644 index 000000000000..ae787665ea82 --- /dev/null +++ b/src/panels/lovelace/tile-features/hui-number-tile-feature.ts @@ -0,0 +1,121 @@ +import { HassEntity } from "home-assistant-js-websocket"; +import { css, html, LitElement, nothing, PropertyValues } from "lit"; +import { customElement, property, state } from "lit/decorators"; +import { computeDomain } from "../../../common/entity/compute_domain"; +import { isUnavailableState } from "../../../data/entity"; +import { HomeAssistant } from "../../../types"; +import { LovelaceTileFeature, LovelaceTileFeatureEditor } from "../types"; +import { NumberTileFeatureConfig } from "./types"; +import "../../../components/ha-control-button"; +import "../../../components/ha-control-button-group"; +import "../../../components/ha-control-number-buttons"; +import "../../../components/ha-control-slider"; +import "../../../components/ha-icon"; + +export const supportsNumberTileFeature = (stateObj: HassEntity) => { + const domain = computeDomain(stateObj.entity_id); + return domain === "input_number"; +}; + +@customElement("hui-number-tile-feature") +class HuiNumberTileFeature extends LitElement implements LovelaceTileFeature { + @property({ attribute: false }) public hass?: HomeAssistant; + + @property({ attribute: false }) public stateObj?: HassEntity; + + @state() private _config?: NumberTileFeatureConfig; + + @state() _currentState?: string; + + static getStubConfig(): NumberTileFeatureConfig { + return { + type: "number", + style: "buttons", + }; + } + + public static async getConfigElement(): Promise { + await import("../editor/config-elements/hui-number-tile-feature-editor"); + return document.createElement("hui-number-tile-feature-editor"); + } + + public setConfig(config: NumberTileFeatureConfig): void { + if (!config) { + throw new Error("Invalid configuration"); + } + this._config = config; + } + + protected willUpdate(changedProp: PropertyValues): void { + super.willUpdate(changedProp); + if (changedProp.has("stateObj") && this.stateObj) { + this._currentState = this.stateObj.state; + } + } + + private async _setValue(ev: CustomEvent) { + const stateObj = this.stateObj!; + + await this.hass!.callService("input_number", "set_value", { + entity_id: stateObj.entity_id, + value: ev.detail.value, + }); + } + + protected render() { + if ( + !this._config || + !this.hass || + !this.stateObj || + !supportsNumberTileFeature(this.stateObj) + ) { + return nothing; + } + + const stateObj = this.stateObj; + + return html` +
+ ${this._config.style === "buttons" + ? html`` + : html`test + `} +
+ `; + } + + static get styles() { + return css` + ha-control-number-buttons { + width: auto; + } + ha-control-slider { + --control-slider-color: var(--tile-color); + } + .container { + padding: 0 12px 12px 12px; + width: auto; + } + `; + } +} + +declare global { + interface HTMLElementTagNameMap { + "hui-number-tile-feature": HuiNumberTileFeature; + } +} diff --git a/src/panels/lovelace/tile-features/types.ts b/src/panels/lovelace/tile-features/types.ts index 893791f235d9..8eb4fa556ef9 100644 --- a/src/panels/lovelace/tile-features/types.ts +++ b/src/panels/lovelace/tile-features/types.ts @@ -50,6 +50,11 @@ export interface SelectOptionsTileFeatureConfig { type: "select-options"; } +export interface NumberTileFeatureConfig { + type: "number"; + style?: "buttons" | "slider"; +} + export interface TargetTemperatureTileFeatureConfig { type: "target-temperature"; } @@ -98,7 +103,8 @@ export type LovelaceTileFeatureConfig = | VacuumCommandsTileFeatureConfig | TargetTemperatureTileFeatureConfig | WaterHeaterOperationModesTileFeatureConfig - | SelectOptionsTileFeatureConfig; + | SelectOptionsTileFeatureConfig + | NumberTileFeatureConfig; export type LovelaceTileFeatureContext = { entity_id?: string; diff --git a/src/translations/en.json b/src/translations/en.json index 907445f5b42a..841c29f59ab1 100644 --- a/src/translations/en.json +++ b/src/translations/en.json @@ -5207,6 +5207,14 @@ "select-options": { "label": "Select options" }, + "number": { + "label": "Number Input", + "style": "Style", + "style_list": { + "buttons": "Buttons", + "slider": "Slider" + } + }, "target-temperature": { "label": "Target temperature" }, From c3487a30e12a465df5c0e71f91fe703572524add Mon Sep 17 00:00:00 2001 From: Joseph Abbey Date: Wed, 8 Nov 2023 13:05:30 +0000 Subject: [PATCH 2/5] Added support for number domain --- src/panels/lovelace/tile-features/hui-number-tile-feature.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/panels/lovelace/tile-features/hui-number-tile-feature.ts b/src/panels/lovelace/tile-features/hui-number-tile-feature.ts index ae787665ea82..74957283ed7b 100644 --- a/src/panels/lovelace/tile-features/hui-number-tile-feature.ts +++ b/src/panels/lovelace/tile-features/hui-number-tile-feature.ts @@ -14,7 +14,7 @@ import "../../../components/ha-icon"; export const supportsNumberTileFeature = (stateObj: HassEntity) => { const domain = computeDomain(stateObj.entity_id); - return domain === "input_number"; + return domain === "input_number" || domain === "number"; }; @customElement("hui-number-tile-feature") From 0522cff0448bc4020a1146fe580aab0fab0862e9 Mon Sep 17 00:00:00 2001 From: Joseph Abbey Date: Wed, 8 Nov 2023 13:06:18 +0000 Subject: [PATCH 3/5] Apply suggestions from code review (thanks @piitaya) Co-authored-by: Paul Bottein --- src/panels/lovelace/tile-features/hui-number-tile-feature.ts | 2 +- src/translations/en.json | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/panels/lovelace/tile-features/hui-number-tile-feature.ts b/src/panels/lovelace/tile-features/hui-number-tile-feature.ts index 74957283ed7b..6e360c30b072 100644 --- a/src/panels/lovelace/tile-features/hui-number-tile-feature.ts +++ b/src/panels/lovelace/tile-features/hui-number-tile-feature.ts @@ -92,7 +92,7 @@ class HuiNumberTileFeature extends LitElement implements LovelaceTileFeature { step=${stateObj.attributes.step} @value-changed=${this._setValue} .disabled=${isUnavailableState(stateObj.state)} - >test + > `} `; diff --git a/src/translations/en.json b/src/translations/en.json index 841c29f59ab1..566fedce9640 100644 --- a/src/translations/en.json +++ b/src/translations/en.json @@ -5208,7 +5208,7 @@ "label": "Select options" }, "number": { - "label": "Number Input", + "label": "Number", "style": "Style", "style_list": { "buttons": "Buttons", From 9100ade9686ec19df4c1ae1b642e11c1c0d698ac Mon Sep 17 00:00:00 2001 From: Joseph Abbey Date: Wed, 8 Nov 2023 13:36:26 +0000 Subject: [PATCH 4/5] Fixed the callback to use the correct service --- src/panels/lovelace/tile-features/hui-number-tile-feature.ts | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/src/panels/lovelace/tile-features/hui-number-tile-feature.ts b/src/panels/lovelace/tile-features/hui-number-tile-feature.ts index 6e360c30b072..cb7cdda78437 100644 --- a/src/panels/lovelace/tile-features/hui-number-tile-feature.ts +++ b/src/panels/lovelace/tile-features/hui-number-tile-feature.ts @@ -56,7 +56,9 @@ class HuiNumberTileFeature extends LitElement implements LovelaceTileFeature { private async _setValue(ev: CustomEvent) { const stateObj = this.stateObj!; - await this.hass!.callService("input_number", "set_value", { + const domain = computeDomain(stateObj.entity_id); + + await this.hass!.callService(domain, "set_value", { entity_id: stateObj.entity_id, value: ev.detail.value, }); From 3549376239f941a4ce9bf5c93f726edcac49524a Mon Sep 17 00:00:00 2001 From: Joseph Abbey Date: Thu, 9 Nov 2023 13:00:16 +0000 Subject: [PATCH 5/5] fix formatting --- src/panels/lovelace/tile-features/hui-number-tile-feature.ts | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/src/panels/lovelace/tile-features/hui-number-tile-feature.ts b/src/panels/lovelace/tile-features/hui-number-tile-feature.ts index cb7cdda78437..768d7a45b490 100644 --- a/src/panels/lovelace/tile-features/hui-number-tile-feature.ts +++ b/src/panels/lovelace/tile-features/hui-number-tile-feature.ts @@ -57,7 +57,7 @@ class HuiNumberTileFeature extends LitElement implements LovelaceTileFeature { const stateObj = this.stateObj!; const domain = computeDomain(stateObj.entity_id); - + await this.hass!.callService(domain, "set_value", { entity_id: stateObj.entity_id, value: ev.detail.value, @@ -94,8 +94,7 @@ class HuiNumberTileFeature extends LitElement implements LovelaceTileFeature { step=${stateObj.attributes.step} @value-changed=${this._setValue} .disabled=${isUnavailableState(stateObj.state)} - > - `} + >`} `; }