From 2b9540fe0344fd2d8bb96ff2c438429e111c4224 Mon Sep 17 00:00:00 2001 From: Bram Kragten Date: Mon, 23 Oct 2023 22:53:09 +0200 Subject: [PATCH] Add support for todo component (#18289) --- demo/src/configs/arsaboo/entities.ts | 9 + demo/src/configs/jimpower/entities.ts | 9 + demo/src/configs/kernehed/entities.ts | 9 + demo/src/configs/teachingbirds/entities.ts | 9 + demo/src/ha-demo.ts | 4 +- demo/src/stubs/shopping_list.ts | 44 --- demo/src/stubs/todo.ts | 24 ++ .../src/pages/lovelace/shopping-list-card.ts | 22 +- src/common/const.ts | 2 + src/components/ha-sidebar.ts | 4 +- src/data/shopping-list.ts | 58 ---- src/data/todo.ts | 104 ++++++ src/fake_data/demo_config.ts | 8 +- src/layouts/partial-panel-resolver.ts | 3 +- .../lovelace/cards/hui-shopping-list-card.ts | 305 ++++++++++++------ src/panels/lovelace/cards/types.ts | 1 + .../hui-shopping-list-editor.ts | 12 +- .../shopping-list/ha-panel-shopping-list.ts | 106 ------ src/panels/todo/ha-panel-todo.ts | 257 +++++++++++++++ src/translations/en.json | 7 +- 20 files changed, 670 insertions(+), 327 deletions(-) delete mode 100644 demo/src/stubs/shopping_list.ts create mode 100644 demo/src/stubs/todo.ts delete mode 100644 src/data/shopping-list.ts create mode 100644 src/data/todo.ts delete mode 100644 src/panels/shopping-list/ha-panel-shopping-list.ts create mode 100644 src/panels/todo/ha-panel-todo.ts diff --git a/demo/src/configs/arsaboo/entities.ts b/demo/src/configs/arsaboo/entities.ts index d90a9a8ff242..f656f4fca0ae 100644 --- a/demo/src/configs/arsaboo/entities.ts +++ b/demo/src/configs/arsaboo/entities.ts @@ -3,6 +3,15 @@ import { DemoConfig } from "../types"; export const demoEntitiesArsaboo: DemoConfig["entities"] = (localize) => convertEntities({ + "todo.shopping_list": { + entity_id: "todo.shopping_list", + state: "2", + attributes: { + supported_features: 15, + friendly_name: "Shopping List", + icon: "mdi:cart", + }, + }, "zone.home": { entity_id: "zone.home", state: "zoning", diff --git a/demo/src/configs/jimpower/entities.ts b/demo/src/configs/jimpower/entities.ts index 1752a682e920..e2c09636a3e5 100644 --- a/demo/src/configs/jimpower/entities.ts +++ b/demo/src/configs/jimpower/entities.ts @@ -3,6 +3,15 @@ import { DemoConfig } from "../types"; export const demoEntitiesJimpower: DemoConfig["entities"] = () => convertEntities({ + "todo.shopping_list": { + entity_id: "todo.shopping_list", + state: "2", + attributes: { + supported_features: 15, + friendly_name: "Shopping List", + icon: "mdi:cart", + }, + }, "zone.powertec": { entity_id: "zone.powertec", state: "zoning", diff --git a/demo/src/configs/kernehed/entities.ts b/demo/src/configs/kernehed/entities.ts index 193a05179d65..092498b9bb0f 100644 --- a/demo/src/configs/kernehed/entities.ts +++ b/demo/src/configs/kernehed/entities.ts @@ -3,6 +3,15 @@ import { DemoConfig } from "../types"; export const demoEntitiesKernehed: DemoConfig["entities"] = () => convertEntities({ + "todo.shopping_list": { + entity_id: "todo.shopping_list", + state: "2", + attributes: { + supported_features: 15, + friendly_name: "Shopping List", + icon: "mdi:cart", + }, + }, "zone.anna": { entity_id: "zone.anna", state: "zoning", diff --git a/demo/src/configs/teachingbirds/entities.ts b/demo/src/configs/teachingbirds/entities.ts index ca2d945ffa4e..4adaaf331c4b 100644 --- a/demo/src/configs/teachingbirds/entities.ts +++ b/demo/src/configs/teachingbirds/entities.ts @@ -3,6 +3,15 @@ import { DemoConfig } from "../types"; export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () => convertEntities({ + "todo.shopping_list": { + entity_id: "todo.shopping_list", + state: "2", + attributes: { + supported_features: 15, + friendly_name: "Shopping List", + icon: "mdi:cart", + }, + }, "sensor.pollen_grabo": { entity_id: "sensor.pollen_grabo", state: "", diff --git a/demo/src/ha-demo.ts b/demo/src/ha-demo.ts index d542477be96d..17550a6af68e 100644 --- a/demo/src/ha-demo.ts +++ b/demo/src/ha-demo.ts @@ -22,7 +22,7 @@ import { mockLovelace } from "./stubs/lovelace"; import { mockMediaPlayer } from "./stubs/media_player"; import { mockPersistentNotification } from "./stubs/persistent_notification"; import { mockRecorder } from "./stubs/recorder"; -import { mockShoppingList } from "./stubs/shopping_list"; +import { mockTodo } from "./stubs/todo"; import { mockSystemLog } from "./stubs/system_log"; import { mockTemplate } from "./stubs/template"; import { mockTranslations } from "./stubs/translations"; @@ -49,7 +49,7 @@ export class HaDemo extends HomeAssistantAppEl { mockTranslations(hass); mockHistory(hass); mockRecorder(hass); - mockShoppingList(hass); + mockTodo(hass); mockSystemLog(hass); mockTemplate(hass); mockEvents(hass); diff --git a/demo/src/stubs/shopping_list.ts b/demo/src/stubs/shopping_list.ts deleted file mode 100644 index 7b714a6e0199..000000000000 --- a/demo/src/stubs/shopping_list.ts +++ /dev/null @@ -1,44 +0,0 @@ -import { ShoppingListItem } from "../../../src/data/shopping-list"; -import { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; - -let items: ShoppingListItem[] = [ - { - id: 12, - name: "Milk", - complete: false, - }, - { - id: 13, - name: "Eggs", - complete: false, - }, - { - id: 14, - name: "Oranges", - complete: true, - }, -]; - -export const mockShoppingList = (hass: MockHomeAssistant) => { - hass.mockWS("shopping_list/items", () => items); - hass.mockWS("shopping_list/items/add", (msg) => { - const item: ShoppingListItem = { - id: new Date().getTime(), - complete: false, - name: msg.name, - }; - items.push(item); - hass.mockEvent("shopping_list_updated"); - return item; - }); - hass.mockWS("shopping_list/items/update", ({ type, item_id, ...updates }) => { - items = items.map((item) => - item.id === item_id ? { ...item, ...updates } : item - ); - hass.mockEvent("shopping_list_updated"); - }); - hass.mockWS("shopping_list/items/clear", () => { - items = items.filter((item) => !item.complete); - hass.mockEvent("shopping_list_updated"); - }); -}; diff --git a/demo/src/stubs/todo.ts b/demo/src/stubs/todo.ts new file mode 100644 index 000000000000..b0393f6c8800 --- /dev/null +++ b/demo/src/stubs/todo.ts @@ -0,0 +1,24 @@ +import { TodoItem, TodoItemStatus } from "../../../src/data/todo"; +import { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; + +export const mockTodo = (hass: MockHomeAssistant) => { + hass.mockWS("todo/item/list", () => ({ + items: [ + { + uid: "12", + summary: "Milk", + status: TodoItemStatus.NeedsAction, + }, + { + uid: "13", + summary: "Eggs", + status: TodoItemStatus.NeedsAction, + }, + { + uid: "14", + summary: "Oranges", + status: TodoItemStatus.Completed, + }, + ] as TodoItem[], + })); +}; diff --git a/gallery/src/pages/lovelace/shopping-list-card.ts b/gallery/src/pages/lovelace/shopping-list-card.ts index 68b08945165e..f7822be93715 100644 --- a/gallery/src/pages/lovelace/shopping-list-card.ts +++ b/gallery/src/pages/lovelace/shopping-list-card.ts @@ -2,12 +2,25 @@ import { html, LitElement, PropertyValues, TemplateResult } from "lit"; import { customElement, query } from "lit/decorators"; import { provideHass } from "../../../../src/fake_data/provide_hass"; import "../../components/demo-cards"; +import { getEntity } from "../../../../src/fake_data/entity"; +import { mockTodo } from "../../../../demo/src/stubs/todo"; + +const ENTITIES = [ + getEntity("todo", "shopping_list", "2", { + friendly_name: "Shopping List", + supported_features: 15, + }), + getEntity("todo", "read_only", "2", { + friendly_name: "Read only", + }), +]; const CONFIGS = [ { heading: "List example", config: ` - type: shopping-list + entity: todo.shopping_list `, }, { @@ -15,6 +28,7 @@ const CONFIGS = [ config: ` - type: shopping-list title: Shopping List + entity: todo.read_only `, }, ]; @@ -32,13 +46,9 @@ class DemoShoppingListEntity extends LitElement { const hass = provideHass(this._demoRoot); hass.updateTranslations(null, "en"); hass.updateTranslations("lovelace", "en"); + hass.addEntities(ENTITIES); - hass.mockAPI("shopping_list", () => [ - { name: "list", id: 1, complete: false }, - { name: "all", id: 2, complete: false }, - { name: "the", id: 3, complete: false }, - { name: "things", id: 4, complete: true }, - ]); + mockTodo(hass); } } diff --git a/src/common/const.ts b/src/common/const.ts index daead2ada95f..dbb3481e6c59 100644 --- a/src/common/const.ts +++ b/src/common/const.ts @@ -16,6 +16,7 @@ import { mdiCarCoolantLevel, mdiCash, mdiChatSleep, + mdiClipboardCheck, mdiClock, mdiCloudUpload, mdiCog, @@ -120,6 +121,7 @@ export const FIXED_DOMAIN_ICONS = { siren: mdiBullhorn, stt: mdiMicrophoneMessage, text: mdiFormTextbox, + todo: mdiClipboardCheck, time: mdiClock, timer: mdiTimerOutline, tts: mdiSpeakerMessage, diff --git a/src/components/ha-sidebar.ts b/src/components/ha-sidebar.ts index d8d4209f5bb9..98bd6a345870 100644 --- a/src/components/ha-sidebar.ts +++ b/src/components/ha-sidebar.ts @@ -2,9 +2,9 @@ import "@material/mwc-button/mwc-button"; import { mdiBell, mdiCalendar, - mdiCart, mdiCellphoneCog, mdiChartBox, + mdiClipboardList, mdiClose, mdiCog, mdiFormatListBulletedType, @@ -81,7 +81,7 @@ const PANEL_ICONS = { lovelace: mdiViewDashboard, map: mdiTooltipAccount, "media-browser": mdiPlayBoxMultiple, - "shopping-list": mdiCart, + todo: mdiClipboardList, }; const panelSorter = ( diff --git a/src/data/shopping-list.ts b/src/data/shopping-list.ts deleted file mode 100644 index 7a1d60a1ca32..000000000000 --- a/src/data/shopping-list.ts +++ /dev/null @@ -1,58 +0,0 @@ -import { HomeAssistant } from "../types"; - -export interface ShoppingListItem { - id: number; - name: string; - complete: boolean; -} - -export const fetchItems = (hass: HomeAssistant): Promise => - hass.callWS({ - type: "shopping_list/items", - }); - -export const updateItem = ( - hass: HomeAssistant, - itemId: number, - item: { - name?: string; - complete?: boolean; - } -): Promise => - hass.callWS({ - type: "shopping_list/items/update", - item_id: itemId, - ...item, - }); - -export const clearItems = (hass: HomeAssistant): Promise => - hass.callWS({ - type: "shopping_list/items/clear", - }); - -export const addItem = ( - hass: HomeAssistant, - name: string -): Promise => - hass.callWS({ - type: "shopping_list/items/add", - name, - }); - -export const removeItem = ( - hass: HomeAssistant, - item_id: string -): Promise => - hass.callWS({ - type: "shopping_list/items/remove", - item_id, - }); - -export const reorderItems = ( - hass: HomeAssistant, - itemIds: string[] -): Promise => - hass.callWS({ - type: "shopping_list/items/reorder", - item_ids: itemIds, - }); diff --git a/src/data/todo.ts b/src/data/todo.ts new file mode 100644 index 000000000000..54bd7e6e2133 --- /dev/null +++ b/src/data/todo.ts @@ -0,0 +1,104 @@ +import { HomeAssistant, ServiceCallResponse } from "../types"; +import { computeDomain } from "../common/entity/compute_domain"; +import { computeStateName } from "../common/entity/compute_state_name"; +import { isUnavailableState } from "./entity"; + +export interface TodoList { + entity_id: string; + name: string; +} + +export const enum TodoItemStatus { + NeedsAction = "needs-action", + Completed = "completed", +} + +export interface TodoItem { + uid?: string; + summary: string; + status: TodoItemStatus; +} + +export const enum TodoListEntityFeature { + CREATE_TODO_ITEM = 1, + DELETE_TODO_ITEM = 2, + UPDATE_TODO_ITEM = 4, + MOVE_TODO_ITEM = 8, +} + +export const getTodoLists = (hass: HomeAssistant): TodoList[] => + Object.keys(hass.states) + .filter( + (entityId) => + computeDomain(entityId) === "todo" && + !isUnavailableState(hass.states[entityId].state) + ) + .sort() + .map((entityId) => ({ + ...hass.states[entityId], + entity_id: entityId, + name: computeStateName(hass.states[entityId]), + })); + +export interface TodoItems { + items: TodoItem[]; +} + +export const fetchItems = async ( + hass: HomeAssistant, + entityId: string +): Promise => { + const result = await hass.callWS({ + type: "todo/item/list", + entity_id: entityId, + }); + return result.items; +}; + +export const updateItem = ( + hass: HomeAssistant, + entity_id: string, + item: TodoItem +): Promise => + hass.callService("todo", "update_item", item, { entity_id }); + +export const createItem = ( + hass: HomeAssistant, + entity_id: string, + summary: string +): Promise => + hass.callService( + "todo", + "create_item", + { + summary, + }, + { entity_id } + ); + +export const deleteItem = ( + hass: HomeAssistant, + entity_id: string, + uid: string +): Promise => + hass.callService( + "todo", + "delete_item", + { + uid, + }, + { entity_id } + ); + +export const moveItem = ( + hass: HomeAssistant, + entity_id: string, + uid: string, + pos: number +): Promise => + hass.callWS({ + type: "todo/item/move", + entity_id, + uid, + pos, + }); diff --git a/src/fake_data/demo_config.ts b/src/fake_data/demo_config.ts index e9efe63c65dd..b2663a50476f 100644 --- a/src/fake_data/demo_config.ts +++ b/src/fake_data/demo_config.ts @@ -14,13 +14,7 @@ export const demoConfig: HassConfig = { wind_speed: "m/s", accumulated_precipitation: "mm", }, - components: [ - "notify.html5", - "history", - "shopping_list", - "forecast_solar", - "energy", - ], + components: ["notify.html5", "history", "todo", "forecast_solar", "energy"], time_zone: "America/Los_Angeles", config_dir: "/config", version: "DEMO", diff --git a/src/layouts/partial-panel-resolver.ts b/src/layouts/partial-panel-resolver.ts index 48b44a0b2844..c2bfefd981f5 100644 --- a/src/layouts/partial-panel-resolver.ts +++ b/src/layouts/partial-panel-resolver.ts @@ -34,8 +34,7 @@ const COMPONENTS = { map: () => import("../panels/map/ha-panel-map"), my: () => import("../panels/my/ha-panel-my"), profile: () => import("../panels/profile/ha-panel-profile"), - "shopping-list": () => - import("../panels/shopping-list/ha-panel-shopping-list"), + todo: () => import("../panels/todo/ha-panel-todo"), "media-browser": () => import("../panels/media-browser/ha-panel-media-browser"), }; diff --git a/src/panels/lovelace/cards/hui-shopping-list-card.ts b/src/panels/lovelace/cards/hui-shopping-list-card.ts index 6c7243b7c50c..2f0aa58e73a0 100644 --- a/src/panels/lovelace/cards/hui-shopping-list-card.ts +++ b/src/panels/lovelace/cards/hui-shopping-list-card.ts @@ -1,11 +1,12 @@ import { mdiDrag, mdiNotificationClearAll, mdiPlus, mdiSort } from "@mdi/js"; import { UnsubscribeFunc } from "home-assistant-js-websocket"; import { - css, CSSResultGroup, - html, LitElement, + PropertyValueMap, PropertyValues, + css, + html, nothing, } from "lit"; import { customElement, property, query, state } from "lit/decorators"; @@ -13,25 +14,31 @@ import { classMap } from "lit/directives/class-map"; import { guard } from "lit/directives/guard"; import { repeat } from "lit/directives/repeat"; import { applyThemesOnElement } from "../../../common/dom/apply_themes_on_element"; +import { supportsFeature } from "../../../common/entity/supports-feature"; import "../../../components/ha-card"; import "../../../components/ha-checkbox"; +import "../../../components/ha-list-item"; +import "../../../components/ha-select"; import "../../../components/ha-svg-icon"; import "../../../components/ha-textfield"; import type { HaTextField } from "../../../components/ha-textfield"; import { - addItem, - clearItems, + TodoItem, + TodoItemStatus, + TodoListEntityFeature, + createItem, + deleteItem, fetchItems, - removeItem, - reorderItems, - ShoppingListItem, + getTodoLists, + moveItem, updateItem, -} from "../../../data/shopping-list"; +} from "../../../data/todo"; import { SubscribeMixin } from "../../../mixins/subscribe-mixin"; import type { SortableInstance } from "../../../resources/sortable"; import { HomeAssistant } from "../../../types"; +import { findEntities } from "../common/find-entities"; import { LovelaceCard, LovelaceCardEditor } from "../types"; -import { SensorCardConfig, ShoppingListCardConfig } from "./types"; +import { ShoppingListCardConfig } from "./types"; @customElement("hui-shopping-list-card") class HuiShoppingListCard @@ -43,17 +50,35 @@ class HuiShoppingListCard return document.createElement("hui-shopping-list-card-editor"); } - public static getStubConfig(): ShoppingListCardConfig { - return { type: "shopping-list" }; + public static getStubConfig( + hass: HomeAssistant, + entities: string[], + entitiesFallback: string[] + ): ShoppingListCardConfig { + const includeDomains = ["todo"]; + const maxEntities = 1; + const foundEntities = findEntities( + hass, + maxEntities, + entities, + entitiesFallback, + includeDomains + ); + + return { type: "shopping-list", entity: foundEntities[0] || "" }; } @property({ attribute: false }) public hass?: HomeAssistant; @state() private _config?: ShoppingListCardConfig; - @state() private _uncheckedItems?: ShoppingListItem[]; + @state() private _entityId?: string; - @state() private _checkedItems?: ShoppingListItem[]; + @state() private _items: Record = {}; + + @state() private _uncheckedItems?: TodoItem[]; + + @state() private _checkedItems?: TodoItem[]; @state() private _reordering = false; @@ -71,10 +96,39 @@ class HuiShoppingListCard this._config = config; this._uncheckedItems = []; this._checkedItems = []; + + if (this._config!.entity) { + this._entityId = this._config!.entity; + } + } + + public willUpdate( + _changedProperties: PropertyValueMap | Map + ): void { + if (!this.hasUpdated) { + if (!this._entityId) { + const todoLists = getTodoLists(this.hass!); + if (todoLists.length) { + if (todoLists.length > 1) { + // find first entity provided by "shopping_list" + for (const list of todoLists) { + const entityReg = this.hass?.entities[list.entity_id]; + if (entityReg?.platform === "shopping_list") { + this._entityId = list.entity_id; + break; + } + } + } + if (!this._entityId) { + this._entityId = todoLists[0].entity_id; + } + } + } + this._fetchData(); + } } public hassSubscribe(): Promise[] { - this._fetchData(); return [ this.hass!.connection.subscribeEvents( () => this._fetchData(), @@ -91,7 +145,7 @@ class HuiShoppingListCard const oldHass = changedProps.get("hass") as HomeAssistant | undefined; const oldConfig = changedProps.get("_config") as - | SensorCardConfig + | ShoppingListCardConfig | undefined; if ( @@ -103,7 +157,7 @@ class HuiShoppingListCard } protected render() { - if (!this._config || !this.hass) { + if (!this._config || !this.hass || !this._entityId) { return nothing; } @@ -115,31 +169,39 @@ class HuiShoppingListCard })} >
- - - - - + ${this.todoListSupportsFeature(TodoListEntityFeature.CREATE_TODO_ITEM) + ? html` + + + + ` + : nothing} + ${this.todoListSupportsFeature(TodoListEntityFeature.MOVE_TODO_ITEM) + ? html` + + + ` + : nothing}
${this._reordering ? html` @@ -163,32 +225,43 @@ class HuiShoppingListCard "ui.panel.lovelace.cards.shopping-list.checked_items" )} - - + ${this.todoListSupportsFeature( + TodoListEntityFeature.DELETE_TODO_ITEM + ) + ? html` + ` + : nothing} ${repeat( this._checkedItems!, - (item) => item.id, + (item) => item.uid, (item) => html`
- + ${this.todoListSupportsFeature( + TodoListEntityFeature.UPDATE_TODO_ITEM + ) + ? html` ` + : nothing}
@@ -200,23 +273,30 @@ class HuiShoppingListCard `; } - private _renderItems(items: ShoppingListItem[]) { + private _renderItems(items: TodoItem[]) { return html` ${repeat( items, - (item) => item.id, + (item) => item.uid, (item) => html` -
- +
+ ${this.todoListSupportsFeature( + TodoListEntityFeature.UPDATE_TODO_ITEM + ) + ? html` ` + : nothing} ${this._reordering @@ -237,47 +317,70 @@ class HuiShoppingListCard `; } + private todoListSupportsFeature(feature: number): boolean { + const entityStateObj = this.hass!.states[this._entityId!]; + return entityStateObj && supportsFeature(entityStateObj, feature); + } + private async _fetchData(): Promise { - if (!this.hass) { + if (!this.hass || !this._entityId) { return; } - const checkedItems: ShoppingListItem[] = []; - const uncheckedItems: ShoppingListItem[] = []; - const items = await fetchItems(this.hass); - for (const key in items) { - if (items[key].complete) { - checkedItems.push(items[key]); + const checkedItems: TodoItem[] = []; + const uncheckedItems: TodoItem[] = []; + const items = await fetchItems(this.hass!, this._entityId!); + const records: Record = {}; + items.forEach((item) => { + records[item.uid!] = item; + if (item.status === TodoItemStatus.Completed) { + checkedItems.push(item); } else { - uncheckedItems.push(items[key]); + uncheckedItems.push(item); } - } + }); + this._items = records; this._checkedItems = checkedItems; this._uncheckedItems = uncheckedItems; } private _completeItem(ev): void { - updateItem(this.hass!, ev.target.itemId, { - complete: ev.target.checked, - }).catch(() => this._fetchData()); + const item = this._items[ev.target.itemId]; + updateItem(this.hass!, this._entityId!, { + ...item, + status: ev.target.checked + ? TodoItemStatus.Completed + : TodoItemStatus.NeedsAction, + }).finally(() => this._fetchData()); } private _saveEdit(ev): void { // If name is not empty, update the item otherwise remove it if (ev.target.value) { - updateItem(this.hass!, ev.target.itemId, { - name: ev.target.value, - }).catch(() => this._fetchData()); - } else { - removeItem(this.hass!, ev.target.itemId).catch(() => this._fetchData()); + const item = this._items[ev.target.itemId]; + updateItem(this.hass!, this._entityId!, { + ...item, + summary: ev.target.value, + }).finally(() => this._fetchData()); + } else if ( + this.todoListSupportsFeature(TodoListEntityFeature.DELETE_TODO_ITEM) + ) { + deleteItem(this.hass!, this._entityId!, ev.target.itemId).finally(() => + this._fetchData() + ); } ev.target.blur(); } - private _clearItems(): void { - if (this.hass) { - clearItems(this.hass).catch(() => this._fetchData()); + private async _clearCompletedItems(): Promise { + if (!this.hass) { + return; } + const deleteActions: Array> = []; + this._checkedItems!.forEach((item: TodoItem) => { + deleteActions.push(deleteItem(this.hass!, this._entityId!, item.uid!)); + }); + await Promise.all(deleteActions).finally(() => this._fetchData()); } private get _newItem(): HaTextField { @@ -286,9 +389,10 @@ class HuiShoppingListCard private _addItem(ev): void { const newItem = this._newItem; - if (newItem.value!.length > 0) { - addItem(this.hass!, newItem.value!).catch(() => this._fetchData()); + createItem(this.hass!, this._entityId!, newItem.value!).finally(() => + this._fetchData() + ); } newItem.value = ""; @@ -329,9 +433,13 @@ class HuiShoppingListCard // Since this is `onEnd` event, it's possible that // an item wa dragged away and was put back to its original position. if (evt.oldIndex !== evt.newIndex) { - reorderItems(this.hass!, this._sortable!.toArray()).catch(() => - this._fetchData() - ); + const item = this._uncheckedItems![evt.oldIndex]; + moveItem( + this.hass!, + this._entityId!, + item.uid!, + evt.newIndex + ).finally(() => this._fetchData()); // Move the shopping list item in memory. this._uncheckedItems!.splice( evt.newIndex, @@ -416,6 +524,11 @@ class HuiShoppingListCard .clearall { cursor: pointer; } + + .todoList { + display: block; + padding: 8px; + } `; } } diff --git a/src/panels/lovelace/cards/types.ts b/src/panels/lovelace/cards/types.ts index 48d94d3758f5..6a1737fd97cf 100644 --- a/src/panels/lovelace/cards/types.ts +++ b/src/panels/lovelace/cards/types.ts @@ -430,6 +430,7 @@ export interface SensorCardConfig extends LovelaceCardConfig { export interface ShoppingListCardConfig extends LovelaceCardConfig { title?: string; theme?: string; + entity?: string; } export interface StackCardConfig extends LovelaceCardConfig { diff --git a/src/panels/lovelace/editor/config-elements/hui-shopping-list-editor.ts b/src/panels/lovelace/editor/config-elements/hui-shopping-list-editor.ts index ed0be3e4a89f..13cdbbbdb02b 100644 --- a/src/panels/lovelace/editor/config-elements/hui-shopping-list-editor.ts +++ b/src/panels/lovelace/editor/config-elements/hui-shopping-list-editor.ts @@ -3,6 +3,7 @@ import { customElement, property, state } from "lit/decorators"; import { assert, assign, object, optional, string } from "superstruct"; import { isComponentLoaded } from "../../../../common/config/is_component_loaded"; import { fireEvent } from "../../../../common/dom/fire_event"; +import "../../../../components/entity/ha-entity-picker"; import "../../../../components/ha-textfield"; import "../../../../components/ha-theme-picker"; import { HomeAssistant } from "../../../../types"; @@ -16,6 +17,7 @@ const cardConfigStruct = assign( object({ title: optional(string()), theme: optional(string()), + entity: optional(string()), }) ); @@ -48,7 +50,7 @@ export class HuiShoppingListEditor return html`
- ${!isComponentLoaded(this.hass, "shopping_list") + ${!isComponentLoaded(this.hass, "todo") ? html`
${this.hass.localize( @@ -67,6 +69,14 @@ export class HuiShoppingListEditor .configValue=${"title"} @input=${this._valueChanged} > + + - isComponentLoaded(this.hass, "conversation") - ); - - protected firstUpdated(changedProperties: PropertyValues): void { - super.firstUpdated(changedProperties); - - this._card = createCardElement({ type: "shopping-list" }) as LovelaceCard; - this._card.hass = this.hass; - } - - protected updated(changedProperties: PropertyValues): void { - super.updated(changedProperties); - - if (changedProperties.has("hass")) { - this._card.hass = this.hass; - } - } - - protected render(): TemplateResult { - return html` - - -
${this.hass.localize("panel.shopping_list")}
- ${this._conversation(this.hass.config.components) - ? html` - - ` - : ""} -
-
${this._card}
-
-
- `; - } - - private _showVoiceCommandDialog(): void { - showVoiceCommandDialog(this, this.hass, { pipeline_id: "last_used" }); - } - - static get styles(): CSSResultGroup { - return [ - haStyle, - css` - #columns { - display: flex; - flex-direction: row; - justify-content: center; - margin: 8px; - } - .column { - flex: 1 0 0; - max-width: 500px; - min-width: 0; - } - `, - ]; - } -} - -declare global { - interface HTMLElementTagNameMap { - "ha-panel-shopping-list": PanelShoppingList; - } -} diff --git a/src/panels/todo/ha-panel-todo.ts b/src/panels/todo/ha-panel-todo.ts new file mode 100644 index 000000000000..5e79de67f5ba --- /dev/null +++ b/src/panels/todo/ha-panel-todo.ts @@ -0,0 +1,257 @@ +import { ResizeController } from "@lit-labs/observers/resize-controller"; +import "@material/mwc-list"; +import { mdiChevronDown, mdiDotsVertical, mdiMicrophone } from "@mdi/js"; +import { + CSSResultGroup, + LitElement, + PropertyValues, + TemplateResult, + css, + html, + nothing, +} from "lit"; +import { customElement, property, state } from "lit/decorators"; +import memoizeOne from "memoize-one"; +import { isComponentLoaded } from "../../common/config/is_component_loaded"; +import { storage } from "../../common/decorators/storage"; +import { computeDomain } from "../../common/entity/compute_domain"; +import { computeStateName } from "../../common/entity/compute_state_name"; +import "../../components/ha-button"; +import "../../components/ha-icon-button"; +import "../../components/ha-list-item"; +import "../../components/ha-menu-button"; +import "../../components/ha-state-icon"; +import "../../components/ha-svg-icon"; +import "../../components/ha-two-pane-top-app-bar-fixed"; +import { getTodoLists } from "../../data/todo"; +import { showVoiceCommandDialog } from "../../dialogs/voice-command-dialog/show-ha-voice-command-dialog"; +import { haStyle } from "../../resources/styles"; +import { HomeAssistant } from "../../types"; +import { HuiErrorCard } from "../lovelace/cards/hui-error-card"; +import { createCardElement } from "../lovelace/create-element/create-card-element"; +import { LovelaceCard } from "../lovelace/types"; + +@customElement("ha-panel-todo") +class PanelTodo extends LitElement { + @property({ attribute: false }) public hass!: HomeAssistant; + + @property({ type: Boolean, reflect: true }) public narrow!: boolean; + + @property({ type: Boolean, reflect: true }) public mobile = false; + + @state() private _card?: LovelaceCard | HuiErrorCard; + + @storage({ + key: "selectedTodoEntity", + state: true, + }) + private _entityId?: string; + + private _headerHeight = 56; + + private _showPaneController = new ResizeController(this, { + callback: (entries: ResizeObserverEntry[]) => + entries[0]?.contentRect.width > 750, + }); + + private _mql?: MediaQueryList; + + private _conversation = memoizeOne((_components) => + isComponentLoaded(this.hass, "conversation") + ); + + public connectedCallback() { + super.connectedCallback(); + this._mql = window.matchMedia( + "(max-width: 450px), all and (max-height: 500px)" + ); + this._mql.addListener(this._setIsMobile); + this.mobile = this._mql.matches; + const computedStyles = getComputedStyle(this); + this._headerHeight = Number( + computedStyles.getPropertyValue("--header-height").replace("px", "") + ); + } + + public disconnectedCallback() { + super.disconnectedCallback(); + this._mql?.removeListener(this._setIsMobile!); + this._mql = undefined; + } + + private _setIsMobile = (ev: MediaQueryListEvent) => { + this.mobile = ev.matches; + }; + + protected willUpdate(changedProperties: PropertyValues): void { + super.willUpdate(changedProperties); + + if (!this.hasUpdated && !this._entityId) { + this._entityId = Object.keys(this.hass.states).find( + (entityId) => computeDomain(entityId) === "todo" + ); + } else if (!this.hasUpdated) { + this._createCard(); + } + } + + protected updated(changedProperties: PropertyValues): void { + super.updated(changedProperties); + + if (changedProperties.has("_entityId")) { + this._createCard(); + } + + if (changedProperties.has("hass") && this._card) { + this._card.hass = this.hass; + } + } + + private _createCard(): void { + if (!this._entityId) { + this._card = undefined; + return; + } + this._card = createCardElement({ + type: "shopping-list", + entity: this._entityId, + }) as LovelaceCard; + this._card.hass = this.hass; + } + + protected render(): TemplateResult { + const showPane = this._showPaneController.value ?? !this.narrow; + const listItems = getTodoLists(this.hass).map( + (list) => + html` + ${list.name} + ` + ); + return html` + + +
+ ${!showPane + ? html` + + ${this._entityId + ? computeStateName(this.hass.states[this._entityId]) + : ""} + + + ${listItems} +
  • +
    ` + : "Lists"} +
    + ${listItems} + + + ${this._conversation(this.hass.config.components) + ? html` + + + ${this.hass.localize("ui.panel.todo.start_conversation")} + ` + : nothing} + +
    +
    ${this._card}
    +
    +
    + `; + } + + private _handleEntityPicked(ev) { + this._entityId = ev.currentTarget.entityId; + } + + private _showVoiceCommandDialog(): void { + showVoiceCommandDialog(this, this.hass, { pipeline_id: "last_used" }); + } + + static get styles(): CSSResultGroup { + return [ + haStyle, + css` + :host { + display: block; + } + #columns { + display: flex; + flex-direction: row; + justify-content: center; + margin: 8px; + } + .column { + flex: 1 0 0; + max-width: 500px; + min-width: 0; + } + :host([mobile]) .lists { + --mdc-menu-min-width: 100vw; + } + :host([mobile]) ha-button-menu { + --mdc-shape-medium: 0 0 var(--mdc-shape-medium) + var(--mdc-shape-medium); + } + ha-button-menu ha-button { + --mdc-theme-primary: currentColor; + --mdc-typography-button-text-transform: none; + --mdc-typography-button-font-size: var( + --mdc-typography-headline6-font-size, + 1.25rem + ); + --mdc-typography-button-font-weight: var( + --mdc-typography-headline6-font-weight, + 500 + ); + --mdc-typography-button-letter-spacing: var( + --mdc-typography-headline6-letter-spacing, + 0.0125em + ); + --mdc-typography-button-line-height: var( + --mdc-typography-headline6-line-height, + 2rem + ); + --button-height: 40px; + } + `, + ]; + } +} + +declare global { + interface HTMLElementTagNameMap { + "ha-panel-todo": PanelTodo; + } +} diff --git a/src/translations/en.json b/src/translations/en.json index e5f75b58f467..7c3e392ffd65 100644 --- a/src/translations/en.json +++ b/src/translations/en.json @@ -8,7 +8,7 @@ "logbook": "Logbook", "history": "History", "mailbox": "Mailbox", - "shopping_list": "Shopping list", + "todo": "To-do Lists", "developer_tools": "Developer tools", "media_browser": "Media", "profile": "Profile" @@ -4471,6 +4471,7 @@ "never_triggered": "Never triggered" }, "shopping-list": { + "lists": "To-do Lists", "checked_items": "Checked items", "clear_items": "Clear checked items", "add_item": "Add item", @@ -5046,7 +5047,7 @@ "shopping-list": { "name": "Shopping list", "description": "The Shopping list card allows you to add, edit, check-off, and clear items from your shopping list.", - "integration_not_loaded": "This card requires the `shopping_list` integration to be set up." + "integration_not_loaded": "This card requires the `todo` integration to be set up." }, "thermostat": { "name": "Thermostat", @@ -5458,7 +5459,7 @@ "qr_code_image": "QR code for token {name}" } }, - "shopping_list": { + "todo": { "start_conversation": "Start conversation" }, "page-authorize": {