Compare commits

..

1 Commits

Author SHA1 Message Date
Paulus Schoutsen
f8ecc7827e Simplify automation conditions 2025-08-31 13:02:03 -03:00
96 changed files with 1405 additions and 3478 deletions

View File

@@ -5,17 +5,17 @@ const castContext = framework.CastReceiverContext.getInstance();
const playerManager = castContext.getPlayerManager(); const playerManager = castContext.getPlayerManager();
playerManager.setMessageInterceptor( playerManager.setMessageInterceptor(
"LOAD" as framework.messages.MessageType.LOAD, framework.messages.MessageType.LOAD,
(loadRequestData) => { (loadRequestData) => {
const media = loadRequestData.media; const media = loadRequestData.media;
// Special handling if it came from Google Assistant // Special handling if it came from Google Assistant
if (media.entity) { if (media.entity) {
media.contentId = media.entity; media.contentId = media.entity;
media.streamType = "LIVE" as framework.messages.StreamType.LIVE; media.streamType = framework.messages.StreamType.LIVE;
media.contentType = "application/vnd.apple.mpegurl"; media.contentType = "application/vnd.apple.mpegurl";
// @ts-ignore // @ts-ignore
media.hlsVideoSegmentFormat = media.hlsVideoSegmentFormat =
"fmp4" as framework.messages.HlsVideoSegmentFormat.FMP4; framework.messages.HlsVideoSegmentFormat.FMP4;
} }
return loadRequestData; return loadRequestData;
} }

View File

@@ -40,8 +40,7 @@ const playDummyMedia = (viewTitle?: string) => {
loadRequestData.media.contentId = loadRequestData.media.contentId =
"https://cast.home-assistant.io/images/google-nest-hub.png"; "https://cast.home-assistant.io/images/google-nest-hub.png";
loadRequestData.media.contentType = "image/jpeg"; loadRequestData.media.contentType = "image/jpeg";
loadRequestData.media.streamType = loadRequestData.media.streamType = framework.messages.StreamType.NONE;
"NONE" as framework.messages.StreamType.NONE;
const metadata = new framework.messages.GenericMediaMetadata(); const metadata = new framework.messages.GenericMediaMetadata();
metadata.title = viewTitle; metadata.title = viewTitle;
loadRequestData.media.metadata = metadata; loadRequestData.media.metadata = metadata;
@@ -90,7 +89,7 @@ const showMediaPlayer = () => {
const options = new framework.CastReceiverOptions(); const options = new framework.CastReceiverOptions();
options.disableIdleTimeout = true; options.disableIdleTimeout = true;
options.customNamespaces = { options.customNamespaces = {
[CAST_NS]: "json" as framework.system.MessageType.JSON, [CAST_NS]: framework.system.MessageType.JSON,
}; };
castContext.addCustomMessageListener( castContext.addCustomMessageListener(
@@ -98,7 +97,9 @@ castContext.addCustomMessageListener(
// @ts-ignore // @ts-ignore
(ev: ReceivedMessage<HassMessage>) => { (ev: ReceivedMessage<HassMessage>) => {
// We received a show Lovelace command, stop media from playing, hide media player and show Lovelace controller // We received a show Lovelace command, stop media from playing, hide media player and show Lovelace controller
if (playerManager.getPlayerState() !== "IDLE") { if (
playerManager.getPlayerState() !== framework.messages.PlayerState.IDLE
) {
playerManager.stop(); playerManager.stop();
} else { } else {
showLovelaceController(); showLovelaceController();
@@ -112,7 +113,7 @@ castContext.addCustomMessageListener(
const playerManager = castContext.getPlayerManager(); const playerManager = castContext.getPlayerManager();
playerManager.setMessageInterceptor( playerManager.setMessageInterceptor(
"LOAD" as framework.messages.MessageType.LOAD, framework.messages.MessageType.LOAD,
(loadRequestData) => { (loadRequestData) => {
if ( if (
loadRequestData.media.contentId === loadRequestData.media.contentId ===
@@ -126,23 +127,24 @@ playerManager.setMessageInterceptor(
// Special handling if it came from Google Assistant // Special handling if it came from Google Assistant
if (media.entity) { if (media.entity) {
media.contentId = media.entity; media.contentId = media.entity;
media.streamType = "LIVE" as framework.messages.StreamType.LIVE; media.streamType = framework.messages.StreamType.LIVE;
media.contentType = "application/vnd.apple.mpegurl"; media.contentType = "application/vnd.apple.mpegurl";
// @ts-ignore // @ts-ignore
media.hlsVideoSegmentFormat = media.hlsVideoSegmentFormat =
"fmp4" as framework.messages.HlsVideoSegmentFormat.FMP4; framework.messages.HlsVideoSegmentFormat.FMP4;
} }
return loadRequestData; return loadRequestData;
} }
); );
playerManager.addEventListener( playerManager.addEventListener(
"MEDIA_STATUS" as framework.events.EventType.MEDIA_STATUS, framework.events.EventType.MEDIA_STATUS,
(event) => { (event) => {
if ( if (
event.mediaStatus?.playerState === "IDLE" && event.mediaStatus?.playerState === framework.messages.PlayerState.IDLE &&
event.mediaStatus?.idleReason && event.mediaStatus?.idleReason &&
event.mediaStatus?.idleReason !== "INTERRUPTED" event.mediaStatus?.idleReason !==
framework.messages.IdleReason.INTERRUPTED
) { ) {
// media finished or stopped, return to default Lovelace // media finished or stopped, return to default Lovelace
showLovelaceController(); showLovelaceController();

View File

@@ -68,7 +68,7 @@
} }
#ha-launch-screen .ha-launch-screen-spacer-top { #ha-launch-screen .ha-launch-screen-spacer-top {
flex: 1; flex: 1;
margin-top: calc( 2 * max(var(--safe-area-inset-top, 0px), 48px) + 46px ); margin-top: calc( 2 * max(var(--safe-area-inset-bottom, 0px), 48px) + 46px );
padding-top: 48px; padding-top: 48px;
} }
#ha-launch-screen .ha-launch-screen-spacer-bottom { #ha-launch-screen .ha-launch-screen-spacer-bottom {

View File

@@ -1,11 +1,10 @@
import { LitElement, css, html, nothing } from "lit"; import { LitElement, css, html } from "lit";
import { customElement, property } from "lit/decorators"; import { customElement, property } from "lit/decorators";
import "../../../src/components/ha-card"; import "../../../src/components/ha-card";
import "../../../src/dialogs/more-info/more-info-content"; import "../../../src/dialogs/more-info/more-info-content";
import "../../../src/state-summary/state-card-content"; import "../../../src/state-summary/state-card-content";
import "../ha-demo-options"; import "../ha-demo-options";
import type { HomeAssistant } from "../../../src/types"; import type { HomeAssistant } from "../../../src/types";
import { computeShowNewMoreInfo } from "../../../src/dialogs/more-info/const";
@customElement("demo-more-info") @customElement("demo-more-info")
class DemoMoreInfo extends LitElement { class DemoMoreInfo extends LitElement {
@@ -22,13 +21,11 @@ class DemoMoreInfo extends LitElement {
<div class="root"> <div class="root">
<div id="card"> <div id="card">
<ha-card> <ha-card>
${!computeShowNewMoreInfo(state) <state-card-content
? html`<state-card-content
.stateObj=${state} .stateObj=${state}
.hass=${this.hass} .hass=${this.hass}
in-dialog in-dialog
></state-card-content>` ></state-card-content>
: nothing}
<more-info-content <more-info-content
.hass=${this.hass} .hass=${this.hass}

View File

@@ -1106,7 +1106,7 @@ export default {
friendly_name: "Philips Hue", friendly_name: "Philips Hue",
entity_picture: null, entity_picture: null,
description: description:
"Press the button on the bridge to register Philips Hue with Home Assistant.", "Press the button on the bridge to register Philips Hue with Home Assistant.\n\n![Description image](/static/images/config_philips_hue.jpg)",
submit_caption: "I have pressed the button", submit_caption: "I have pressed the button",
}, },
last_changed: "2018-07-19T10:44:46.515160+00:00", last_changed: "2018-07-19T10:44:46.515160+00:00",

View File

@@ -19,9 +19,8 @@
height: auto; height: auto;
padding: 32px 0; padding: 32px 0;
} }
.content { .content {
max-width: min(560px, calc(100vw - var(--safe-area-inset-right, 0px) - var(--safe-area-inset-left, 0px))); max-width: 560px;
margin: 0 auto; margin: 0 auto;
padding: 0 16px; padding: 0 16px;
box-sizing: content-box; box-sizing: content-box;

View File

@@ -29,13 +29,13 @@
"@awesome.me/webawesome": "3.0.0-beta.4", "@awesome.me/webawesome": "3.0.0-beta.4",
"@babel/runtime": "7.28.3", "@babel/runtime": "7.28.3",
"@braintree/sanitize-url": "7.1.1", "@braintree/sanitize-url": "7.1.1",
"@codemirror/autocomplete": "6.18.7", "@codemirror/autocomplete": "6.18.6",
"@codemirror/commands": "6.8.1", "@codemirror/commands": "6.8.1",
"@codemirror/language": "6.11.3", "@codemirror/language": "6.11.3",
"@codemirror/legacy-modes": "6.5.1", "@codemirror/legacy-modes": "6.5.1",
"@codemirror/search": "6.5.11", "@codemirror/search": "6.5.11",
"@codemirror/state": "6.5.2", "@codemirror/state": "6.5.2",
"@codemirror/view": "6.38.2", "@codemirror/view": "6.38.1",
"@egjs/hammerjs": "2.0.17", "@egjs/hammerjs": "2.0.17",
"@formatjs/intl-datetimeformat": "6.18.0", "@formatjs/intl-datetimeformat": "6.18.0",
"@formatjs/intl-displaynames": "6.8.11", "@formatjs/intl-displaynames": "6.8.11",
@@ -159,18 +159,18 @@
"@octokit/plugin-retry": "8.0.1", "@octokit/plugin-retry": "8.0.1",
"@octokit/rest": "22.0.0", "@octokit/rest": "22.0.0",
"@rsdoctor/rspack-plugin": "1.2.3", "@rsdoctor/rspack-plugin": "1.2.3",
"@rspack/core": "1.5.2", "@rspack/core": "1.5.1",
"@rspack/dev-server": "1.1.4", "@rspack/dev-server": "1.1.4",
"@types/babel__plugin-transform-runtime": "7.9.5", "@types/babel__plugin-transform-runtime": "7.9.5",
"@types/chromecast-caf-receiver": "6.0.24", "@types/chromecast-caf-receiver": "6.0.22",
"@types/chromecast-caf-sender": "1.0.11", "@types/chromecast-caf-sender": "1.0.11",
"@types/color-name": "2.0.0", "@types/color-name": "2.0.0",
"@types/culori": "4.0.1", "@types/culori": "4.0.0",
"@types/html-minifier-terser": "7.0.2", "@types/html-minifier-terser": "7.0.2",
"@types/js-yaml": "4.0.9", "@types/js-yaml": "4.0.9",
"@types/leaflet": "1.9.20", "@types/leaflet": "1.9.20",
"@types/leaflet-draw": "1.0.13", "@types/leaflet-draw": "1.0.12",
"@types/leaflet.markercluster": "1.5.6", "@types/leaflet.markercluster": "1.5.5",
"@types/lodash.merge": "4.6.9", "@types/lodash.merge": "4.6.9",
"@types/luxon": "3.7.1", "@types/luxon": "3.7.1",
"@types/mocha": "10.0.10", "@types/mocha": "10.0.10",
@@ -204,7 +204,7 @@
"husky": "9.1.7", "husky": "9.1.7",
"jsdom": "26.1.0", "jsdom": "26.1.0",
"jszip": "3.10.1", "jszip": "3.10.1",
"lint-staged": "16.1.6", "lint-staged": "16.1.5",
"lit-analyzer": "2.0.3", "lit-analyzer": "2.0.3",
"lodash.merge": "4.6.2", "lodash.merge": "4.6.2",
"lodash.template": "4.5.0", "lodash.template": "4.5.0",
@@ -218,7 +218,7 @@
"terser-webpack-plugin": "5.3.14", "terser-webpack-plugin": "5.3.14",
"ts-lit-plugin": "2.0.2", "ts-lit-plugin": "2.0.2",
"typescript": "5.9.2", "typescript": "5.9.2",
"typescript-eslint": "8.42.0", "typescript-eslint": "8.41.0",
"vite-tsconfig-paths": "5.1.4", "vite-tsconfig-paths": "5.1.4",
"vitest": "3.2.4", "vitest": "3.2.4",
"webpack-stats-plugin": "1.1.3", "webpack-stats-plugin": "1.1.3",

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.4 KiB

View File

@@ -4,7 +4,7 @@ build-backend = "setuptools.build_meta"
[project] [project]
name = "home-assistant-frontend" name = "home-assistant-frontend"
version = "20250903.0" version = "20250827.0"
license = "Apache-2.0" license = "Apache-2.0"
license-files = ["LICENSE*"] license-files = ["LICENSE*"]
description = "The Home Assistant frontend" description = "The Home Assistant frontend"

View File

@@ -67,7 +67,10 @@ export const generateEntityFilter = (
} }
if (floors) { if (floors) {
if (!floor || !floors.has(floor.floor_id)) { if (!floor) {
return false;
}
if (!floors) {
return false; return false;
} }
} }

View File

@@ -25,8 +25,6 @@ export class HaAutomationRow extends LitElement {
@property({ type: Boolean, reflect: true, attribute: "building-block" }) @property({ type: Boolean, reflect: true, attribute: "building-block" })
public buildingBlock = false; public buildingBlock = false;
@property({ type: Boolean, reflect: true }) public highlight?: boolean;
@query(".row") @query(".row")
private _rowElement?: HTMLDivElement; private _rowElement?: HTMLDivElement;
@@ -80,18 +78,7 @@ export class HaAutomationRow extends LitElement {
ev.key !== " " && ev.key !== " " &&
!( !(
(this.sortSelected || ev.altKey) && (this.sortSelected || ev.altKey) &&
!(ev.ctrlKey || ev.metaKey) &&
!ev.shiftKey &&
(ev.key === "ArrowUp" || ev.key === "ArrowDown") (ev.key === "ArrowUp" || ev.key === "ArrowDown")
) &&
!(
(ev.ctrlKey || ev.metaKey) &&
!ev.shiftKey &&
!ev.altKey &&
(ev.key === "c" ||
ev.key === "x" ||
ev.key === "Delete" ||
ev.key === "Backspace")
) )
) { ) {
return; return;
@@ -112,22 +99,6 @@ export class HaAutomationRow extends LitElement {
return; return;
} }
if (ev.ctrlKey || ev.metaKey) {
if (ev.key === "c") {
fireEvent(this, "copy-row");
return;
}
if (ev.key === "x") {
fireEvent(this, "cut-row");
return;
}
if (ev.key === "Delete" || ev.key === "Backspace") {
fireEvent(this, "delete-row");
return;
}
}
this.click(); this.click();
} }
@@ -159,7 +130,6 @@ export class HaAutomationRow extends LitElement {
.expand-button { .expand-button {
transition: transform 150ms cubic-bezier(0.4, 0, 0.2, 1); transition: transform 150ms cubic-bezier(0.4, 0, 0.2, 1);
color: var(--ha-color-on-neutral-quiet); color: var(--ha-color-on-neutral-quiet);
margin-left: -8px;
} }
:host([building-block]) .leading-icon-wrapper { :host([building-block]) .leading-icon-wrapper {
background-color: var(--ha-color-fill-neutral-loud-resting); background-color: var(--ha-color-fill-neutral-loud-resting);
@@ -198,20 +168,9 @@ export class HaAutomationRow extends LitElement {
margin: 0 12px; margin: 0 12px;
} }
:host([sort-selected]) .row { :host([sort-selected]) .row {
outline: solid; box-shadow:
outline-color: rgba(var(--rgb-accent-color), 0.6); 0px 0px 8px 4px rgba(var(--rgb-accent-color), 0.8),
outline-offset: -2px; inset 0px 2px 8px 4px rgba(var(--rgb-accent-color), 0.4);
outline-width: 2px;
background-color: rgba(var(--rgb-accent-color), 0.08);
}
.row:hover {
background-color: rgba(var(--rgb-primary-text-color), 0.04);
}
:host([highlight]) .row {
background-color: rgba(var(--rgb-primary-color), 0.08);
}
:host([highlight]) .row:hover {
background-color: rgba(var(--rgb-primary-color), 0.16);
} }
`; `;
} }
@@ -224,8 +183,5 @@ declare global {
interface HASSDomEvents { interface HASSDomEvents {
"toggle-collapsed": undefined; "toggle-collapsed": undefined;
"stop-sort-selection": undefined; "stop-sort-selection": undefined;
"copy-row": undefined;
"cut-row": undefined;
"delete-row": undefined;
} }
} }

View File

@@ -30,8 +30,6 @@ export class HaBottomSheet extends LitElement {
@state() private _dialogMaxViewpointHeight = 70; @state() private _dialogMaxViewpointHeight = 70;
@state() private _dialogMinViewpointHeight = 55;
@state() private _dialogViewportHeight?: number; @state() private _dialogViewportHeight?: number;
render() { render() {
@@ -43,7 +41,6 @@ export class HaBottomSheet extends LitElement {
? `${this._dialogViewportHeight}vh` ? `${this._dialogViewportHeight}vh`
: "auto", : "auto",
maxHeight: `${this._dialogMaxViewpointHeight}vh`, maxHeight: `${this._dialogMaxViewpointHeight}vh`,
minHeight: `${this._dialogMinViewpointHeight}vh`,
})} })}
> >
<div class="handle-wrapper"> <div class="handle-wrapper">
@@ -83,7 +80,6 @@ export class HaBottomSheet extends LitElement {
this._dialogViewportHeight = this._dialogViewportHeight =
(this._dialog.offsetHeight / window.innerHeight) * 100; (this._dialog.offsetHeight / window.innerHeight) * 100;
this._dialogMaxViewpointHeight = 90; this._dialogMaxViewpointHeight = 90;
this._dialogMinViewpointHeight = 20;
} else { } else {
// after close animation is done close dialog element and fire closed event // after close animation is done close dialog element and fire closed event
this._dialog.close(); this._dialog.close();
@@ -232,6 +228,7 @@ export class HaBottomSheet extends LitElement {
box-shadow: var(--wa-shadow-l); box-shadow: var(--wa-shadow-l);
padding: 0; padding: 0;
margin: 0; margin: 0;
top: auto; top: auto;
inset-inline-end: auto; inset-inline-end: auto;
bottom: 0; bottom: 0;

View File

@@ -393,13 +393,10 @@ export class HaItemDisplayEditor extends LitElement {
--md-list-item-one-line-container-height: 48px; --md-list-item-one-line-container-height: 48px;
} }
ha-md-list-item.drag-selected { ha-md-list-item.drag-selected {
--md-focus-ring-color: rgba(var(--rgb-accent-color), 0.6); box-shadow:
0px 0px 8px 4px rgba(var(--rgb-accent-color), 0.8),
inset 0px 2px 8px 4px rgba(var(--rgb-accent-color), 0.4);
border-radius: 8px; border-radius: 8px;
outline: solid;
outline-color: rgba(var(--rgb-accent-color), 0.6);
outline-offset: -2px;
outline-width: 2px;
background-color: rgba(var(--rgb-accent-color), 0.08);
} }
ha-md-list-item ha-icon-button { ha-md-list-item ha-icon-button {
margin-left: -12px; margin-left: -12px;

View File

@@ -159,7 +159,6 @@ export class HaMdDialog extends Dialog {
--md-dialog-headline-size: var(--ha-font-size-xl); --md-dialog-headline-size: var(--ha-font-size-xl);
--md-dialog-supporting-text-size: var(--ha-font-size-m); --md-dialog-supporting-text-size: var(--ha-font-size-m);
--md-dialog-supporting-text-line-height: var(--ha-line-height-normal); --md-dialog-supporting-text-line-height: var(--ha-line-height-normal);
--md-divider-color: var(--divider-color);
} }
:host([type="alert"]) { :host([type="alert"]) {

View File

@@ -3,7 +3,6 @@ import {
mdiDevices, mdiDevices,
mdiPaletteSwatch, mdiPaletteSwatch,
mdiTextureBox, mdiTextureBox,
mdiTransitConnectionVariant,
} from "@mdi/js"; } from "@mdi/js";
import type { CSSResultGroup, PropertyValues } from "lit"; import type { CSSResultGroup, PropertyValues } from "lit";
import { LitElement, css, html, nothing } from "lit"; import { LitElement, css, html, nothing } from "lit";
@@ -267,9 +266,7 @@ export class HaRelatedItems extends LitElement {
<a href="/config/devices/device/${relatedDeviceId}"> <a href="/config/devices/device/${relatedDeviceId}">
<ha-list-item hasMeta graphic="icon"> <ha-list-item hasMeta graphic="icon">
<ha-svg-icon <ha-svg-icon
.path=${device.entry_type === "service" .path=${mdiDevices}
? mdiTransitConnectionVariant
: mdiDevices}
slot="graphic" slot="graphic"
></ha-svg-icon> ></ha-svg-icon>
${device.name_by_user || device.name} ${device.name_by_user || device.name}

View File

@@ -15,16 +15,21 @@ declare global {
"item-added": { "item-added": {
index: number; index: number;
data: any; data: any;
item: any;
}; };
"item-removed": { "item-removed": {
index: number; index: number;
}; };
"drag-start": undefined; "drag-start": undefined;
"drag-end": undefined; "drag-end": undefined;
"item-cloned": HaSortableClonedEventData;
} }
} }
export interface HaSortableClonedEventData {
item: any;
clone: any;
}
export type HaSortableOptions = Omit< export type HaSortableOptions = Omit<
SortableInstance.SortableOptions, SortableInstance.SortableOptions,
"onStart" | "onChoose" | "onEnd" | "onUpdate" | "onAdd" | "onRemove" "onStart" | "onChoose" | "onEnd" | "onUpdate" | "onAdd" | "onRemove"
@@ -149,6 +154,7 @@ export class HaSortable extends LitElement {
onUpdate: this._handleUpdate, onUpdate: this._handleUpdate,
onAdd: this._handleAdd, onAdd: this._handleAdd,
onRemove: this._handleRemove, onRemove: this._handleRemove,
onClone: this._handleClone,
}; };
if (this.draggableSelector) { if (this.draggableSelector) {
@@ -181,7 +187,6 @@ export class HaSortable extends LitElement {
fireEvent(this, "item-added", { fireEvent(this, "item-added", {
index: evt.newIndex, index: evt.newIndex,
data: evt.item.sortableData, data: evt.item.sortableData,
item: evt.item,
}); });
}; };
@@ -189,6 +194,10 @@ export class HaSortable extends LitElement {
fireEvent(this, "item-removed", { index: evt.oldIndex }); fireEvent(this, "item-removed", { index: evt.oldIndex });
}; };
private _handleClone = (evt) => {
fireEvent(this, "item-cloned", evt);
};
private _handleEnd = async (evt) => { private _handleEnd = async (evt) => {
fireEvent(this, "drag-end"); fireEvent(this, "drag-end");
// put back in original location // put back in original location

View File

@@ -387,6 +387,18 @@ export const normalizeAutomationConfig = <
} }
} }
// We move all conditions into the action for display
if (config.conditions) {
if (config.actions) {
(config.actions as Action[]).unshift(
...(config.conditions as Condition[])
);
} else {
config.actions = config.conditions;
}
delete config.conditions;
}
return config; return config;
}; };
@@ -556,6 +568,7 @@ export interface AutomationClipboard {
} }
export interface BaseSidebarConfig { export interface BaseSidebarConfig {
toggleYamlMode: () => boolean;
delete: () => void; delete: () => void;
close: (focus?: boolean) => void; close: (focus?: boolean) => void;
} }
@@ -567,7 +580,6 @@ export interface TriggerSidebarConfig extends BaseSidebarConfig {
duplicate: () => void; duplicate: () => void;
cut: () => void; cut: () => void;
copy: () => void; copy: () => void;
toggleYamlMode: () => void;
config: Trigger; config: Trigger;
yamlMode: boolean; yamlMode: boolean;
uiSupported: boolean; uiSupported: boolean;
@@ -581,7 +593,6 @@ export interface ConditionSidebarConfig extends BaseSidebarConfig {
duplicate: () => void; duplicate: () => void;
cut: () => void; cut: () => void;
copy: () => void; copy: () => void;
toggleYamlMode: () => void;
config: Condition; config: Condition;
yamlMode: boolean; yamlMode: boolean;
uiSupported: boolean; uiSupported: boolean;
@@ -595,7 +606,6 @@ export interface ActionSidebarConfig extends BaseSidebarConfig {
cut: () => void; cut: () => void;
copy: () => void; copy: () => void;
run: () => void; run: () => void;
toggleYamlMode: () => void;
config: { config: {
action: Action; action: Action;
}; };
@@ -617,7 +627,6 @@ export interface ScriptFieldSidebarConfig extends BaseSidebarConfig {
key: string; key: string;
excludeKeys: string[]; excludeKeys: string[];
}; };
toggleYamlMode: () => void;
yamlMode: boolean; yamlMode: boolean;
} }

View File

@@ -24,14 +24,11 @@ export interface BluetoothConnectionData extends DataTableRowData {
source: string; source: string;
} }
export type HaScannerType = "usb" | "uart" | "remote" | "unknown";
export interface BluetoothScannerDetails { export interface BluetoothScannerDetails {
source: string; source: string;
connectable: boolean; connectable: boolean;
name: string; name: string;
adapter: string; adapter: string;
scanner_type?: HaScannerType;
} }
export type BluetoothScannersDetails = Record<string, BluetoothScannerDetails>; export type BluetoothScannersDetails = Record<string, BluetoothScannerDetails>;
@@ -58,13 +55,6 @@ export interface BluetoothAllocationsData {
allocated: string[]; allocated: string[];
} }
export interface BluetoothScannerState {
source: string;
adapter: string;
current_mode: "active" | "passive" | null;
requested_mode: "active" | "passive" | null;
}
export const subscribeBluetoothScannersDetailsUpdates = ( export const subscribeBluetoothScannersDetailsUpdates = (
conn: Connection, conn: Connection,
store: Store<BluetoothScannersDetails> store: Store<BluetoothScannersDetails>
@@ -180,20 +170,3 @@ export const subscribeBluetoothConnectionAllocations = (
params params
); );
}; };
export const subscribeBluetoothScannerState = (
conn: Connection,
callbackFunction: (scannerState: BluetoothScannerState) => void,
configEntryId?: string
): Promise<() => Promise<void>> => {
const params: { type: string; config_entry_id?: string } = {
type: "bluetooth/subscribe_scanner_state",
};
if (configEntryId) {
params.config_entry_id = configEntryId;
}
return conn.subscribeMessage<BluetoothScannerState>(
(scannerState) => callbackFunction(scannerState),
params
);
};

View File

@@ -97,7 +97,6 @@ export interface DataEntryFlowStepMenu {
step_id: string; step_id: string;
/** If array, use value to lookup translations in strings.json */ /** If array, use value to lookup translations in strings.json */
menu_options: string[] | Record<string, string>; menu_options: string[] | Record<string, string>;
sort?: boolean;
description_placeholders?: Record<string, string>; description_placeholders?: Record<string, string>;
translation_domain?: string; translation_domain?: string;
} }

View File

@@ -1 +1 @@
export const strokeWidth = 2; export const strokeWidth = 5;

View File

@@ -2,7 +2,6 @@ import type { Connection } from "home-assistant-js-websocket";
import { computeStateName } from "../common/entity/compute_state_name"; import { computeStateName } from "../common/entity/compute_state_name";
import type { HaDurationData } from "../components/ha-duration-input"; import type { HaDurationData } from "../components/ha-duration-input";
import type { HomeAssistant } from "../types"; import type { HomeAssistant } from "../types";
import { firstWeekday } from "../common/datetime/first_weekday";
export interface RecorderInfo { export interface RecorderInfo {
backlog: number | null; backlog: number | null;
@@ -109,7 +108,7 @@ export interface StatisticsValidationResultMeanTypeChanged {
}; };
} }
export const VOLUME_UNITS = ["L", "gal", "ft³", "m³", "CCF", "MCF"] as const; export const VOLUME_UNITS = ["L", "gal", "ft³", "m³", "CCF"] as const;
export interface StatisticsUnitConfiguration { export interface StatisticsUnitConfiguration {
energy?: "Wh" | "kWh" | "MWh" | "GJ"; energy?: "Wh" | "kWh" | "MWh" | "GJ";
@@ -212,14 +211,7 @@ export const fetchStatistic = (
: period.fixed_period.end, : period.fixed_period.end,
} }
: undefined, : undefined,
calendar: period.calendar calendar: period.calendar,
? {
...(period.calendar.period === "week"
? { first_weekday: firstWeekday(hass.locale).substring(0, 3) }
: {}),
...period.calendar,
}
: undefined,
rolling_window: period.rolling_window, rolling_window: period.rolling_window,
}); });

View File

@@ -256,13 +256,6 @@ export const showConfigFlowDialog = (
); );
}, },
renderMenuOptionDescription(hass, step, option) {
return hass.localize(
`component.${step.translation_domain || step.handler}.config.step.${step.step_id}.menu_option_descriptions.${option}`,
step.description_placeholders
);
},
renderLoadingDescription(hass, reason, handler, step) { renderLoadingDescription(hass, reason, handler, step) {
if (reason !== "loading_flow" && reason !== "loading_step") { if (reason !== "loading_flow" && reason !== "loading_step") {
return ""; return "";

View File

@@ -137,12 +137,6 @@ export interface FlowConfig {
option: string option: string
): string; ): string;
renderMenuOptionDescription(
hass: HomeAssistant,
step: DataEntryFlowStepMenu,
option: string
): string;
renderLoadingDescription( renderLoadingDescription(
hass: HomeAssistant, hass: HomeAssistant,
loadingReason: LoadingReason, loadingReason: LoadingReason,

View File

@@ -225,13 +225,6 @@ export const showOptionsFlowDialog = (
); );
}, },
renderMenuOptionDescription(hass, step, option) {
return hass.localize(
`component.${step.translation_domain || configEntry.domain}.options.step.${step.step_id}.menu_option_descriptions.${option}`,
step.description_placeholders
);
},
renderLoadingDescription(hass, reason) { renderLoadingDescription(hass, reason) {
return ( return (
hass.localize(`component.${configEntry.domain}.options.loading`) || hass.localize(`component.${configEntry.domain}.options.loading`) ||

View File

@@ -252,13 +252,6 @@ export const showSubConfigFlowDialog = (
); );
}, },
renderMenuOptionDescription(hass, step, option) {
return hass.localize(
`component.${step.translation_domain || configEntry.domain}.config_subentries.${flowType}.step.${step.step_id}.menu_option_descriptions.${option}`,
step.description_placeholders
);
},
renderLoadingDescription(hass, reason, handler, step) { renderLoadingDescription(hass, reason, handler, step) {
if (reason !== "loading_flow" && reason !== "loading_step") { if (reason !== "loading_flow" && reason !== "loading_step") {
return ""; return "";

View File

@@ -1,5 +1,5 @@
import type { PropertyValues, TemplateResult } from "lit"; import type { TemplateResult } from "lit";
import { css, html, LitElement, nothing } from "lit"; import { css, html, LitElement } from "lit";
import { customElement, property } from "lit/decorators"; import { customElement, property } from "lit/decorators";
import { fireEvent } from "../../common/dom/fire_event"; import { fireEvent } from "../../common/dom/fire_event";
import "../../components/ha-icon-next"; import "../../components/ha-icon-next";
@@ -8,7 +8,6 @@ import type { DataEntryFlowStepMenu } from "../../data/data_entry_flow";
import type { HomeAssistant } from "../../types"; import type { HomeAssistant } from "../../types";
import type { FlowConfig } from "./show-dialog-data-entry-flow"; import type { FlowConfig } from "./show-dialog-data-entry-flow";
import { configFlowContentStyles } from "./styles"; import { configFlowContentStyles } from "./styles";
import { stringCompare } from "../../common/string/compare";
@customElement("step-flow-menu") @customElement("step-flow-menu")
class StepFlowMenu extends LitElement { class StepFlowMenu extends LitElement {
@@ -18,18 +17,9 @@ class StepFlowMenu extends LitElement {
@property({ attribute: false }) public step!: DataEntryFlowStepMenu; @property({ attribute: false }) public step!: DataEntryFlowStepMenu;
protected shouldUpdate(changedProps: PropertyValues): boolean {
return (
changedProps.size > 1 ||
!changedProps.has("hass") ||
this.hass.localize !== changedProps.get("hass")?.localize
);
}
protected render(): TemplateResult { protected render(): TemplateResult {
let options: string[]; let options: string[];
let translations: Record<string, string>; let translations: Record<string, string>;
let optionDescriptions: Record<string, string> = {};
if (Array.isArray(this.step.menu_options)) { if (Array.isArray(this.step.menu_options)) {
options = this.step.menu_options; options = this.step.menu_options;
@@ -40,36 +30,10 @@ class StepFlowMenu extends LitElement {
this.step, this.step,
option option
); );
optionDescriptions[option] =
this.flowConfig.renderMenuOptionDescription(
this.hass,
this.step,
option
);
} }
} else { } else {
options = Object.keys(this.step.menu_options); options = Object.keys(this.step.menu_options);
translations = this.step.menu_options; translations = this.step.menu_options;
optionDescriptions = Object.fromEntries(
options.map((key) => [
key,
this.flowConfig.renderMenuOptionDescription(
this.hass,
this.step,
key
),
])
);
}
if (this.step.sort) {
options = options.sort((a, b) =>
stringCompare(
translations[a]!,
translations[b]!,
this.hass.locale.language
)
);
} }
const description = this.flowConfig.renderMenuDescription( const description = this.flowConfig.renderMenuDescription(
@@ -82,18 +46,8 @@ class StepFlowMenu extends LitElement {
<div class="options"> <div class="options">
${options.map( ${options.map(
(option) => html` (option) => html`
<ha-list-item <ha-list-item hasMeta .step=${option} @click=${this._handleStep}>
hasMeta
.step=${option}
@click=${this._handleStep}
?twoline=${optionDescriptions[option]}
>
<span>${translations[option]}</span> <span>${translations[option]}</span>
${optionDescriptions[option]
? html`<span slot="secondary">
${optionDescriptions[option]}
</span>`
: nothing}
<ha-icon-next slot="meta"></ha-icon-next> <ha-icon-next slot="meta"></ha-icon-next>
</ha-list-item> </ha-list-item>
` `
@@ -119,10 +73,11 @@ class StepFlowMenu extends LitElement {
css` css`
.options { .options {
margin-top: 20px; margin-top: 20px;
margin-bottom: 16px; margin-bottom: 8px;
} }
.content { .content {
padding-bottom: 16px; padding-bottom: 16px;
border-bottom: 1px solid var(--divider-color);
} }
.content + .options { .content + .options {
margin-top: 8px; margin-top: 8px;

View File

@@ -3,11 +3,11 @@ import { css, html, LitElement, nothing } from "lit";
import { customElement, property, query, state } from "lit/decorators"; import { customElement, property, query, state } from "lit/decorators";
import { ifDefined } from "lit/directives/if-defined"; import { ifDefined } from "lit/directives/if-defined";
import { fireEvent } from "../../common/dom/fire_event"; import { fireEvent } from "../../common/dom/fire_event";
import "../../components/ha-button";
import "../../components/ha-dialog-header";
import "../../components/ha-md-dialog"; import "../../components/ha-md-dialog";
import type { HaMdDialog } from "../../components/ha-md-dialog"; import type { HaMdDialog } from "../../components/ha-md-dialog";
import "../../components/ha-dialog-header";
import "../../components/ha-svg-icon"; import "../../components/ha-svg-icon";
import "../../components/ha-button";
import "../../components/ha-textfield"; import "../../components/ha-textfield";
import type { HaTextField } from "../../components/ha-textfield"; import type { HaTextField } from "../../components/ha-textfield";
import type { HomeAssistant } from "../../types"; import type { HomeAssistant } from "../../types";
@@ -52,7 +52,7 @@ class DialogBox extends LitElement {
return nothing; return nothing;
} }
const confirmPrompt = this._params.confirmation || !!this._params.prompt; const confirmPrompt = this._params.confirmation || this._params.prompt;
const dialogTitle = const dialogTitle =
this._params.title || this._params.title ||
@@ -62,7 +62,7 @@ class DialogBox extends LitElement {
return html` return html`
<ha-md-dialog <ha-md-dialog
open open
.disableCancelAction=${confirmPrompt} .disableCancelAction=${confirmPrompt || false}
@closed=${this._dialogClosed} @closed=${this._dialogClosed}
type="alert" type="alert"
aria-labelledby="dialog-box-title" aria-labelledby="dialog-box-title"
@@ -100,22 +100,23 @@ class DialogBox extends LitElement {
: ""} : ""}
</div> </div>
<div slot="actions"> <div slot="actions">
${confirmPrompt ${confirmPrompt &&
? html` html`
<ha-button <ha-button
@click=${this._dismiss} @click=${this._dismiss}
?autofocus=${!this._params.prompt && this._params.destructive} ?dialogInitialFocus=${!this._params.prompt &&
this._params.destructive}
appearance="plain" appearance="plain"
> >
${this._params.dismissText ${this._params.dismissText
? this._params.dismissText ? this._params.dismissText
: this.hass.localize("ui.common.cancel")} : this.hass.localize("ui.common.cancel")}
</ha-button> </ha-button>
` `}
: nothing}
<ha-button <ha-button
@click=${this._confirm} @click=${this._confirm}
?autofocus=${!this._params.prompt && !this._params.destructive} ?dialogInitialFocus=${!this._params.prompt &&
!this._params.destructive}
variant=${this._params.destructive ? "danger" : "brand"} variant=${this._params.destructive ? "danger" : "brand"}
> >
${this._params.confirmText ${this._params.confirmText

View File

@@ -6,9 +6,7 @@ import memoizeOne from "memoize-one";
import { formatDateWeekdayShort } from "../../../common/datetime/format_date"; import { formatDateWeekdayShort } from "../../../common/datetime/format_date";
import { formatTime } from "../../../common/datetime/format_time"; import { formatTime } from "../../../common/datetime/format_time";
import { formatNumber } from "../../../common/number/format_number"; import { formatNumber } from "../../../common/number/format_number";
import "../../../components/ha-alert";
import "../../../components/ha-relative-time"; import "../../../components/ha-relative-time";
import "../../../components/ha-spinner";
import "../../../components/ha-state-icon"; import "../../../components/ha-state-icon";
import "../../../components/ha-svg-icon"; import "../../../components/ha-svg-icon";
import "../../../components/ha-tooltip"; import "../../../components/ha-tooltip";
@@ -294,12 +292,15 @@ class MoreInfoWeather extends LitElement {
</div> </div>
` `
: nothing} : nothing}
${forecast
? html`
<div class="section"> <div class="section">
${this.hass.localize("ui.card.weather.forecast")}: ${this.hass.localize("ui.card.weather.forecast")}:
</div> </div>
${supportedForecasts?.length > 1 ${supportedForecasts.length > 1
? html`<sl-tab-group @sl-tab-show=${this._handleForecastTypeChanged}> ? html`<sl-tab-group
@sl-tab-show=${this._handleForecastTypeChanged}
>
${supportedForecasts.map( ${supportedForecasts.map(
(forecastType) => (forecastType) =>
html`<sl-tab html`<sl-tab
@@ -307,15 +308,17 @@ class MoreInfoWeather extends LitElement {
.panel=${forecastType} .panel=${forecastType}
.active=${this._forecastType === forecastType} .active=${this._forecastType === forecastType}
> >
${this.hass!.localize(`ui.card.weather.${forecastType}`)} ${this.hass!.localize(
`ui.card.weather.${forecastType}`
)}
</sl-tab>` </sl-tab>`
)} )}
</sl-tab-group>` </sl-tab-group>`
: nothing} : nothing}
<div class="forecast"> <div class="forecast">
${forecast?.length ${forecast.map((item) =>
? forecast.map((item) => this._showValue(item.templow) ||
this._showValue(item.templow) || this._showValue(item.temperature) this._showValue(item.temperature)
? html` ? html`
<div> <div>
<div> <div>
@@ -385,10 +388,10 @@ class MoreInfoWeather extends LitElement {
</div> </div>
` `
: nothing : nothing
) )}
: html`<ha-spinner size="medium"></ha-spinner>`}
</div> </div>
`
: nothing}
${this.stateObj.attributes.attribution ${this.stateObj.attributes.attribution
? html` ? html`
<div class="attribution"> <div class="attribution">
@@ -586,10 +589,6 @@ class MoreInfoWeather extends LitElement {
.forecast-icon { .forecast-icon {
--mdc-icon-size: 40px; --mdc-icon-size: 40px;
} }
.forecast ha-spinner {
height: 120px;
}
`, `,
]; ];
} }

View File

@@ -8,7 +8,6 @@ import {
mdiPencil, mdiPencil,
mdiPencilOff, mdiPencilOff,
mdiPencilOutline, mdiPencilOutline,
mdiTransitConnectionVariant,
} from "@mdi/js"; } from "@mdi/js";
import type { HassEntity } from "home-assistant-js-websocket"; import type { HassEntity } from "home-assistant-js-websocket";
import type { PropertyValues } from "lit"; import type { PropertyValues } from "lit";
@@ -312,8 +311,6 @@ export class MoreInfoDialog extends LitElement {
const isAdmin = this.hass.user!.is_admin; const isAdmin = this.hass.user!.is_admin;
const deviceId = this._getDeviceId(); const deviceId = this._getDeviceId();
const deviceType =
(deviceId && this.hass.devices[deviceId].entry_type) || "device";
const isDefaultView = this._currView === DEFAULT_VIEW && !this._childView; const isDefaultView = this._currView === DEFAULT_VIEW && !this._childView;
const isSpecificInitialView = const isSpecificInitialView =
@@ -437,18 +434,11 @@ export class MoreInfoDialog extends LitElement {
@request-selected=${this._goToDevice} @request-selected=${this._goToDevice}
> >
${this.hass.localize( ${this.hass.localize(
"ui.dialogs.more_info_control.device_or_service_info", "ui.dialogs.more_info_control.device_info"
{
type: this.hass.localize(
`ui.dialogs.more_info_control.device_type.${deviceType}`
),
}
)} )}
<ha-svg-icon <ha-svg-icon
slot="graphic" slot="graphic"
.path=${deviceType === "service" .path=${mdiDevices}
? mdiTransitConnectionVariant
: mdiDevices}
></ha-svg-icon> ></ha-svg-icon>
</ha-list-item> </ha-list-item>
` `

View File

@@ -1,148 +1,121 @@
import { mdiAppleKeyboardCommand } from "@mdi/js";
import { css, html, LitElement, nothing } from "lit"; import { css, html, LitElement, nothing } from "lit";
import { customElement, property, state } from "lit/decorators"; import { customElement, property, state } from "lit/decorators";
import { fireEvent } from "../../common/dom/fire_event"; import { fireEvent } from "../../common/dom/fire_event";
import type { LocalizeKeys } from "../../common/translations/localize"; import "../../components/ha-button";
import "../../components/ha-alert";
import { createCloseHeading } from "../../components/ha-dialog"; import { createCloseHeading } from "../../components/ha-dialog";
import "../../components/ha-svg-icon";
import { haStyleDialog } from "../../resources/styles";
import type { HomeAssistant } from "../../types"; import type { HomeAssistant } from "../../types";
import { isMac } from "../../util/is_mac"; import { haStyleDialog } from "../../resources/styles";
import "../../components/ha-alert";
import "../../components/chips/ha-assist-chip";
import type { LocalizeKeys } from "../../common/translations/localize";
interface Text { interface Text {
textTranslationKey: LocalizeKeys; type: "text";
key: LocalizeKeys;
} }
interface LocalizedShortcut { type ShortcutString = string | { key: LocalizeKeys };
shortcutTranslationKey: LocalizeKeys;
}
type ShortcutString = string | LocalizedShortcut;
interface Shortcut { interface Shortcut {
type: "shortcut";
shortcut: ShortcutString[]; shortcut: ShortcutString[];
descriptionTranslationKey: LocalizeKeys; key: LocalizeKeys;
} }
interface Section { interface Section {
titleTranslationKey: LocalizeKeys; key: LocalizeKeys;
items: (Text | Shortcut)[]; items: (Text | Shortcut)[];
} }
const CTRL_CMD = "__CTRL_CMD__";
const _SHORTCUTS: Section[] = [ const _SHORTCUTS: Section[] = [
{ {
titleTranslationKey: "ui.dialogs.shortcuts.searching.title", key: "ui.dialogs.shortcuts.searching.title",
items: [ items: [
{ type: "text", key: "ui.dialogs.shortcuts.searching.on_any_page" },
{ {
textTranslationKey: "ui.dialogs.shortcuts.searching.on_any_page", type: "shortcut",
},
{
shortcut: ["C"], shortcut: ["C"],
descriptionTranslationKey: key: "ui.dialogs.shortcuts.searching.search_command",
"ui.dialogs.shortcuts.searching.search_command",
}, },
{ {
type: "shortcut",
shortcut: ["E"], shortcut: ["E"],
descriptionTranslationKey: key: "ui.dialogs.shortcuts.searching.search_entities",
"ui.dialogs.shortcuts.searching.search_entities",
}, },
{ {
type: "shortcut",
shortcut: ["D"], shortcut: ["D"],
descriptionTranslationKey: key: "ui.dialogs.shortcuts.searching.search_devices",
"ui.dialogs.shortcuts.searching.search_devices",
}, },
{ {
textTranslationKey: type: "text",
"ui.dialogs.shortcuts.searching.on_pages_with_tables", key: "ui.dialogs.shortcuts.searching.on_pages_with_tables",
}, },
{ {
shortcut: [CTRL_CMD, "F"], type: "shortcut",
descriptionTranslationKey: shortcut: [{ key: "ui.dialogs.shortcuts.shortcuts.ctrl_cmd" }, "F"],
"ui.dialogs.shortcuts.searching.search_in_table", key: "ui.dialogs.shortcuts.searching.search_in_table",
}, },
], ],
}, },
{ {
titleTranslationKey: "ui.dialogs.shortcuts.assist.title", key: "ui.dialogs.shortcuts.assist.title",
items: [ items: [
{ {
type: "shortcut",
shortcut: ["A"], shortcut: ["A"],
descriptionTranslationKey: "ui.dialogs.shortcuts.assist.open_assist", key: "ui.dialogs.shortcuts.assist.open_assist",
}, },
], ],
}, },
{ {
titleTranslationKey: "ui.dialogs.shortcuts.automation_script.title", key: "ui.dialogs.shortcuts.automation_script.title",
items: [ items: [
{ {
shortcut: [CTRL_CMD, "C"], type: "shortcut",
descriptionTranslationKey: shortcut: [{ key: "ui.dialogs.shortcuts.shortcuts.ctrl_cmd" }, "V"],
"ui.dialogs.shortcuts.automation_script.copy", key: "ui.dialogs.shortcuts.automation_script.paste",
}, },
{ {
shortcut: [CTRL_CMD, "X"], type: "shortcut",
descriptionTranslationKey: "ui.dialogs.shortcuts.automation_script.cut", shortcut: [{ key: "ui.dialogs.shortcuts.shortcuts.ctrl_cmd" }, "S"],
}, key: "ui.dialogs.shortcuts.automation_script.save",
{
shortcut: [
CTRL_CMD,
{ shortcutTranslationKey: "ui.dialogs.shortcuts.keys.del" },
],
descriptionTranslationKey:
"ui.dialogs.shortcuts.automation_script.delete",
},
{
shortcut: [CTRL_CMD, "V"],
descriptionTranslationKey:
"ui.dialogs.shortcuts.automation_script.paste",
},
{
shortcut: [CTRL_CMD, "S"],
descriptionTranslationKey:
"ui.dialogs.shortcuts.automation_script.save",
}, },
], ],
}, },
{ {
titleTranslationKey: "ui.dialogs.shortcuts.charts.title", key: "ui.dialogs.shortcuts.charts.title",
items: [ items: [
{ {
type: "shortcut",
shortcut: [ shortcut: [
CTRL_CMD, { key: "ui.dialogs.shortcuts.shortcuts.ctrl_cmd" },
{ shortcutTranslationKey: "ui.dialogs.shortcuts.shortcuts.drag" }, { key: "ui.dialogs.shortcuts.shortcuts.drag" },
], ],
descriptionTranslationKey: "ui.dialogs.shortcuts.charts.drag_to_zoom", key: "ui.dialogs.shortcuts.charts.drag_to_zoom",
}, },
{ {
type: "shortcut",
shortcut: [ shortcut: [
CTRL_CMD, { key: "ui.dialogs.shortcuts.shortcuts.ctrl_cmd" },
{ { key: "ui.dialogs.shortcuts.shortcuts.scroll_wheel" },
shortcutTranslationKey:
"ui.dialogs.shortcuts.shortcuts.scroll_wheel",
},
], ],
descriptionTranslationKey: "ui.dialogs.shortcuts.charts.scroll_to_zoom", key: "ui.dialogs.shortcuts.charts.scroll_to_zoom",
}, },
{ {
shortcut: [ type: "shortcut",
{ shortcut: [{ key: "ui.dialogs.shortcuts.shortcuts.double_click" }],
shortcutTranslationKey: key: "ui.dialogs.shortcuts.charts.double_click",
"ui.dialogs.shortcuts.shortcuts.double_click",
},
],
descriptionTranslationKey: "ui.dialogs.shortcuts.charts.double_click",
}, },
], ],
}, },
{ {
titleTranslationKey: "ui.dialogs.shortcuts.other.title", key: "ui.dialogs.shortcuts.other.title",
items: [ items: [
{ {
type: "shortcut",
shortcut: ["M"], shortcut: ["M"],
descriptionTranslationKey: "ui.dialogs.shortcuts.other.my_link", key: "ui.dialogs.shortcuts.other.my_link",
}, },
], ],
}, },
@@ -164,28 +137,17 @@ class DialogShortcuts extends LitElement {
} }
private _renderShortcut( private _renderShortcut(
shortcutKeys: ShortcutString[], shortcuts: ShortcutString[],
descriptionKey: LocalizeKeys translationKey: LocalizeKeys
) { ) {
const keys = shortcuts.map((shortcut) =>
typeof shortcut === "string" ? shortcut : this.hass.localize(shortcut.key)
);
return html` return html`
<div class="shortcut"> <div class="shortcut">
${shortcutKeys.map( ${keys.map((key) => html` <span>${key.toUpperCase()}</span>`)}
(shortcutKey) => ${this.hass.localize(translationKey)}
html`<span
>${shortcutKey === CTRL_CMD
? isMac
? html`<ha-svg-icon
.path=${mdiAppleKeyboardCommand}
></ha-svg-icon>`
: this.hass.localize("ui.panel.config.automation.editor.ctrl")
: typeof shortcutKey === "string"
? shortcutKey
: this.hass.localize(
shortcutKey.shortcutTranslationKey
)}</span
>`
)}
${this.hass.localize(descriptionKey)}
</div> </div>
`; `;
} }
@@ -209,18 +171,16 @@ class DialogShortcuts extends LitElement {
<div class="content"> <div class="content">
${_SHORTCUTS.map( ${_SHORTCUTS.map(
(section) => html` (section) => html`
<h3>${this.hass.localize(section.titleTranslationKey)}</h3> <h3>${this.hass.localize(section.key)}</h3>
<div class="items"> <div class="items">
${section.items.map((item) => { ${section.items.map((item) => {
if ("shortcut" in item) { if (item.type === "text") {
return this._renderShortcut( return html`<p>${this.hass.localize(item.key)}</p>`;
(item as Shortcut).shortcut,
(item as Shortcut).descriptionTranslationKey
);
} }
return html`<p> if (item.type === "shortcut") {
${this.hass.localize((item as Text).textTranslationKey)} return this._renderShortcut(item.shortcut, item.key);
</p>`; }
return nothing;
})} })}
</div> </div>
` `
@@ -272,10 +232,6 @@ class DialogShortcuts extends LitElement {
.items p { .items p {
margin-bottom: 8px; margin-bottom: 8px;
} }
ha-svg-icon {
width: 12px;
}
`, `,
]; ];
} }

View File

@@ -35,7 +35,6 @@
align-items: center; align-items: center;
justify-content: center; justify-content: center;
margin-bottom: 32px; margin-bottom: 32px;
padding-top: var(--safe-area-inset-top);
} }
.header img { .header img {

View File

@@ -44,7 +44,7 @@
} }
#ha-launch-screen .ha-launch-screen-spacer-top { #ha-launch-screen .ha-launch-screen-spacer-top {
flex: 1; flex: 1;
margin-top: calc( 2 * max(var(--safe-area-inset-top, 0px), 48px) + 46px ); margin-top: calc( 2 * max(var(--safe-area-inset-bottom, 0px), 48px) + 46px );
padding-top: 48px; padding-top: 48px;
} }
#ha-launch-screen .ha-launch-screen-spacer-bottom { #ha-launch-screen .ha-launch-screen-spacer-bottom {

View File

@@ -19,9 +19,8 @@
height: auto; height: auto;
padding: 32px 0; padding: 32px 0;
} }
.content { .content {
max-width: min(560px, calc(100vw - var(--safe-area-inset-right, 0px) - var(--safe-area-inset-left, 0px))); max-width: 560px;
margin: 0 auto; margin: 0 auto;
padding: 0 16px; padding: 0 16px;
box-sizing: content-box; box-sizing: content-box;
@@ -33,7 +32,6 @@
justify-content: flex-start; justify-content: flex-start;
margin-bottom: 32px; margin-bottom: 32px;
margin-left: 32px; margin-left: 32px;
padding-top: var(--safe-area-inset-top);
} }
.header img { .header img {

View File

@@ -146,8 +146,6 @@ export class HomeAssistantMain extends LitElement {
-webkit-tap-highlight-color: rgba(0, 0, 0, 0); -webkit-tap-highlight-color: rgba(0, 0, 0, 0);
--mdc-drawer-width: 56px; --mdc-drawer-width: 56px;
--mdc-top-app-bar-width: calc(100% - var(--mdc-drawer-width)); --mdc-top-app-bar-width: calc(100% - var(--mdc-drawer-width));
--safe-area-content-inset-left: 0px;
--safe-area-content-inset-right: var(--safe-area-inset-right);
} }
:host([expanded]) { :host([expanded]) {
--mdc-drawer-width: calc(256px + var(--safe-area-inset-left)); --mdc-drawer-width: calc(256px + var(--safe-area-inset-left));
@@ -155,7 +153,6 @@ export class HomeAssistantMain extends LitElement {
:host([modal]) { :host([modal]) {
--mdc-drawer-width: unset; --mdc-drawer-width: unset;
--mdc-top-app-bar-width: unset; --mdc-top-app-bar-width: unset;
--safe-area-content-inset-left: var(--safe-area-inset-left);
} }
partial-panel-resolver, partial-panel-resolver,
ha-sidebar { ha-sidebar {

View File

@@ -11,54 +11,23 @@ export const KeyboardShortcutMixin = <T extends Constructor<LitElement>>(
class extends superClass { class extends superClass {
private _keydownEvent = (event: KeyboardEvent) => { private _keydownEvent = (event: KeyboardEvent) => {
const supportedShortcuts = this.supportedShortcuts(); const supportedShortcuts = this.supportedShortcuts();
if ( if ((event.ctrlKey || event.metaKey) && event.key in supportedShortcuts) {
(event.ctrlKey || event.metaKey) &&
!event.shiftKey &&
!event.altKey &&
event.key in supportedShortcuts
) {
event.preventDefault(); event.preventDefault();
supportedShortcuts[event.key](); supportedShortcuts[event.key]();
return;
}
const supportedSingleKeyShortcuts = this.supportedSingleKeyShortcuts();
if (event.key in supportedSingleKeyShortcuts) {
event.preventDefault();
supportedSingleKeyShortcuts[event.key]();
} }
}; };
private _listenersAdded = false;
public connectedCallback() { public connectedCallback() {
super.connectedCallback(); super.connectedCallback();
this.addKeyboardShortcuts();
}
public disconnectedCallback() {
this.removeKeyboardShortcuts();
super.disconnectedCallback();
}
public addKeyboardShortcuts() {
if (this._listenersAdded) {
return;
}
this._listenersAdded = true;
window.addEventListener("keydown", this._keydownEvent); window.addEventListener("keydown", this._keydownEvent);
} }
public removeKeyboardShortcuts() { public disconnectedCallback() {
this._listenersAdded = false;
window.removeEventListener("keydown", this._keydownEvent); window.removeEventListener("keydown", this._keydownEvent);
super.disconnectedCallback();
} }
protected supportedShortcuts(): SupportedShortcuts { protected supportedShortcuts(): SupportedShortcuts {
return {}; return {};
} }
protected supportedSingleKeyShortcuts(): SupportedShortcuts {
return {};
}
}; };

View File

@@ -97,7 +97,7 @@ export default class HaAutomationActionEditor extends LitElement {
if (!ev.detail.isValid) { if (!ev.detail.isValid) {
return; return;
} }
fireEvent(this, "yaml-changed", { fireEvent(this, "value-changed", {
value: migrateAutomationAction(ev.detail.value), value: migrateAutomationAction(ev.detail.value),
}); });
} }

View File

@@ -5,12 +5,12 @@ import {
mdiArrowUp, mdiArrowUp,
mdiContentCopy, mdiContentCopy,
mdiContentCut, mdiContentCut,
mdiContentDuplicate,
mdiDelete, mdiDelete,
mdiDotsVertical, mdiDotsVertical,
mdiPlay, mdiPlay,
mdiPlayCircleOutline, mdiPlayCircleOutline,
mdiPlaylistEdit, mdiPlaylistEdit,
mdiPlusCircleMultipleOutline,
mdiRenameBox, mdiRenameBox,
mdiStopCircleOutline, mdiStopCircleOutline,
} from "@mdi/js"; } from "@mdi/js";
@@ -151,8 +151,6 @@ export default class HaAutomationActionRow extends LitElement {
@property({ type: Boolean }) public last?: boolean; @property({ type: Boolean }) public last?: boolean;
@property({ type: Boolean }) public highlight?: boolean;
@property({ type: Boolean, attribute: "sidebar" }) @property({ type: Boolean, attribute: "sidebar" })
public optionsInSidebar = false; public optionsInSidebar = false;
@@ -195,10 +193,6 @@ export default class HaAutomationActionRow extends LitElement {
@query("ha-automation-row") @query("ha-automation-row")
private _automationRowElement?: HaAutomationRow; private _automationRowElement?: HaAutomationRow;
get selected() {
return this._selected;
}
protected firstUpdated(changedProperties: PropertyValues): void { protected firstUpdated(changedProperties: PropertyValues): void {
super.firstUpdated(changedProperties); super.firstUpdated(changedProperties);
@@ -309,7 +303,7 @@ export default class HaAutomationActionRow extends LitElement {
)} )}
<ha-svg-icon <ha-svg-icon
slot="start" slot="start"
.path=${mdiPlusCircleMultipleOutline} .path=${mdiContentDuplicate}
></ha-svg-icon> ></ha-svg-icon>
</ha-md-menu-item> </ha-md-menu-item>
@@ -442,6 +436,7 @@ export default class HaAutomationActionRow extends LitElement {
${this.optionsInSidebar ${this.optionsInSidebar
? html`<ha-automation-row ? html`<ha-automation-row
.disabled=${this.action.enabled === false} .disabled=${this.action.enabled === false}
@click=${this._toggleSidebar}
.leftChevron=${[ .leftChevron=${[
...ACTION_BUILDING_BLOCKS, ...ACTION_BUILDING_BLOCKS,
...ACTION_COMBINED_BLOCKS, ...ACTION_COMBINED_BLOCKS,
@@ -452,17 +447,12 @@ export default class HaAutomationActionRow extends LitElement {
))} ))}
.collapsed=${this._collapsed} .collapsed=${this._collapsed}
.selected=${this._selected} .selected=${this._selected}
.highlight=${this.highlight} @toggle-collapsed=${this._toggleCollapse}
.buildingBlock=${[ .buildingBlock=${[
...ACTION_BUILDING_BLOCKS, ...ACTION_BUILDING_BLOCKS,
...ACTION_COMBINED_BLOCKS, ...ACTION_COMBINED_BLOCKS,
].includes(blockType!)} ].includes(blockType!)}
.sortSelected=${this.sortSelected} .sortSelected=${this.sortSelected}
@click=${this._toggleSidebar}
@toggle-collapsed=${this._toggleCollapse}
@copy-row=${this._copyAction}
@cut-row=${this._cutAction}
@delete-row=${this._onDelete}
>${this._renderRow()}</ha-automation-row >${this._renderRow()}</ha-automation-row
>` >`
: html` : html`
@@ -521,15 +511,6 @@ export default class HaAutomationActionRow extends LitElement {
}; };
private _runAction = async () => { private _runAction = async () => {
requestAnimationFrame(() => {
// @ts-ignore is supported in all browsers except firefox
if (this.scrollIntoViewIfNeeded) {
// @ts-ignore is supported in all browsers except firefox
this.scrollIntoViewIfNeeded();
return;
}
this.scrollIntoView();
});
const validated = await validateConfig(this.hass, { const validated = await validateConfig(this.hass, {
actions: this.action, actions: this.action,
}); });
@@ -639,12 +620,6 @@ export default class HaAutomationActionRow extends LitElement {
private _copyAction = () => { private _copyAction = () => {
this._setClipboard(); this._setClipboard();
showToast(this, {
message: this.hass.localize(
"ui.panel.config.automation.editor.actions.copied_to_clipboard"
),
duration: 2000,
});
}; };
private _cutAction = () => { private _cutAction = () => {
@@ -653,12 +628,6 @@ export default class HaAutomationActionRow extends LitElement {
if (this._selected) { if (this._selected) {
fireEvent(this, "close-sidebar"); fireEvent(this, "close-sidebar");
} }
showToast(this, {
message: this.hass.localize(
"ui.panel.config.automation.editor.actions.cut_to_clipboard"
),
duration: 2000,
});
}; };
private _moveUp = () => { private _moveUp = () => {
@@ -692,7 +661,8 @@ export default class HaAutomationActionRow extends LitElement {
ev?.stopPropagation(); ev?.stopPropagation();
if (this._selected) { if (this._selected) {
fireEvent(this, "request-close-sidebar"); this._selected = false;
fireEvent(this, "close-sidebar");
return; return;
} }
this.openSidebar(); this.openSidebar();
@@ -718,7 +688,7 @@ export default class HaAutomationActionRow extends LitElement {
}, },
toggleYamlMode: () => { toggleYamlMode: () => {
this._toggleYamlMode(); this._toggleYamlMode();
this.openSidebar(); return this._yamlMode;
}, },
disable: this._onDisable, disable: this._onDisable,
delete: this._onDelete, delete: this._onDelete,
@@ -736,12 +706,12 @@ export default class HaAutomationActionRow extends LitElement {
this._collapsed = false; this._collapsed = false;
if (this.narrow) { if (this.narrow) {
window.setTimeout(() => { requestAnimationFrame(() => {
this.scrollIntoView({ this.scrollIntoView({
block: "start", block: "start",
behavior: "smooth", behavior: "smooth",
}); });
}, 180); // duration of transition of added padding for bottom sheet });
} }
} }
@@ -788,6 +758,10 @@ export default class HaAutomationActionRow extends LitElement {
this._collapsed = !this._collapsed; this._collapsed = !this._collapsed;
} }
public isSelected() {
return this._selected;
}
public focus() { public focus() {
this._automationRowElement?.focus(); this._automationRowElement?.focus();
} }

View File

@@ -6,9 +6,11 @@ import { customElement, property, queryAll, state } from "lit/decorators";
import { repeat } from "lit/directives/repeat"; import { repeat } from "lit/directives/repeat";
import { storage } from "../../../../common/decorators/storage"; import { storage } from "../../../../common/decorators/storage";
import { fireEvent } from "../../../../common/dom/fire_event"; import { fireEvent } from "../../../../common/dom/fire_event";
import { listenMediaQuery } from "../../../../common/dom/media_query";
import { nextRender } from "../../../../common/util/render-status"; import { nextRender } from "../../../../common/util/render-status";
import "../../../../components/ha-button"; import "../../../../components/ha-button";
import "../../../../components/ha-sortable"; import "../../../../components/ha-sortable";
import type { HaSortableClonedEventData } from "../../../../components/ha-sortable";
import "../../../../components/ha-svg-icon"; import "../../../../components/ha-svg-icon";
import { import {
ACTION_BUILDING_BLOCKS, ACTION_BUILDING_BLOCKS,
@@ -44,6 +46,8 @@ export default class HaAutomationAction extends LitElement {
@property({ type: Boolean, attribute: "sidebar" }) public optionsInSidebar = @property({ type: Boolean, attribute: "sidebar" }) public optionsInSidebar =
false; false;
@state() private _showReorder = false;
@state() private _rowSortSelected?: number; @state() private _rowSortSelected?: number;
@state() @state()
@@ -64,17 +68,33 @@ export default class HaAutomationAction extends LitElement {
private _actionKeys = new WeakMap<Action, string>(); private _actionKeys = new WeakMap<Action, string>();
private _unsubMql?: () => void;
public connectedCallback() {
super.connectedCallback();
this._unsubMql = listenMediaQuery("(min-width: 600px)", (matches) => {
this._showReorder = matches;
});
}
public disconnectedCallback() {
super.disconnectedCallback();
this._unsubMql?.();
this._unsubMql = undefined;
}
protected render() { protected render() {
return html` return html`
<ha-sortable <ha-sortable
handle-selector=".handle" handle-selector=".handle"
draggable-selector="ha-automation-action-row" draggable-selector="ha-automation-action-row"
.disabled=${this.disabled} .disabled=${!this._showReorder || this.disabled}
group="actions" group="actions"
invert-swap invert-swap
@item-moved=${this._actionMoved} @item-moved=${this._actionMoved}
@item-added=${this._actionAdded} @item-added=${this._actionAdded}
@item-removed=${this._actionRemoved} @item-removed=${this._actionRemoved}
@item-cloned=${this._actionCloned}
> >
<div class="rows ${!this.optionsInSidebar ? "no-sidebar" : ""}"> <div class="rows ${!this.optionsInSidebar ? "no-sidebar" : ""}">
${repeat( ${repeat(
@@ -95,12 +115,12 @@ export default class HaAutomationAction extends LitElement {
@move-up=${this._moveUp} @move-up=${this._moveUp}
@value-changed=${this._actionChanged} @value-changed=${this._actionChanged}
.hass=${this.hass} .hass=${this.hass}
.highlight=${this.highlightedActions?.includes(action)} ?highlight=${this.highlightedActions?.includes(action)}
.optionsInSidebar=${this.optionsInSidebar} .optionsInSidebar=${this.optionsInSidebar}
.sortSelected=${this._rowSortSelected === idx} .sortSelected=${this._rowSortSelected === idx}
@stop-sort-selection=${this._stopSortSelection} @stop-sort-selection=${this._stopSortSelection}
> >
${!this.disabled ${this._showReorder && !this.disabled
? html` ? html`
<div <div
tabindex="0" tabindex="0"
@@ -299,8 +319,11 @@ export default class HaAutomationAction extends LitElement {
private async _actionAdded(ev: CustomEvent): Promise<void> { private async _actionAdded(ev: CustomEvent): Promise<void> {
ev.stopPropagation(); ev.stopPropagation();
const { index, data } = ev.detail; const { index, data } = ev.detail;
const item = ev.detail.item as HaAutomationActionRow; let selected = false;
const selected = item.selected; if (data?.["ha-automation-row-selected"]) {
selected = true;
delete data["ha-automation-row-selected"];
}
let actions = [ let actions = [
...this.actions.slice(0, index), ...this.actions.slice(0, index),
@@ -358,6 +381,12 @@ export default class HaAutomationAction extends LitElement {
fireEvent(this, "value-changed", { value: actions }); fireEvent(this, "value-changed", { value: actions });
} }
private _actionCloned(ev: CustomEvent<HaSortableClonedEventData>) {
if (ev.detail.item.action && ev.detail.item.isSelected()) {
ev.detail.item.action["ha-automation-row-selected"] = true;
}
}
private _duplicateAction(ev: CustomEvent) { private _duplicateAction(ev: CustomEvent) {
ev.stopPropagation(); ev.stopPropagation();
const index = (ev.target as any).index; const index = (ev.target as any).index;

View File

@@ -1,9 +1,4 @@
import { import { mdiClose, mdiContentPaste, mdiPlus } from "@mdi/js";
mdiAppleKeyboardCommand,
mdiClose,
mdiContentPaste,
mdiPlus,
} from "@mdi/js";
import Fuse from "fuse.js"; import Fuse from "fuse.js";
import type { CSSResultGroup, PropertyValues, TemplateResult } from "lit"; import type { CSSResultGroup, PropertyValues, TemplateResult } from "lit";
import { LitElement, css, html, nothing } from "lit"; import { LitElement, css, html, nothing } from "lit";
@@ -46,14 +41,11 @@ import {
} from "../../../data/integration"; } from "../../../data/integration";
import { TRIGGER_GROUPS, TRIGGER_ICONS } from "../../../data/trigger"; import { TRIGGER_GROUPS, TRIGGER_ICONS } from "../../../data/trigger";
import type { HassDialog } from "../../../dialogs/make-dialog-manager"; import type { HassDialog } from "../../../dialogs/make-dialog-manager";
import { KeyboardShortcutMixin } from "../../../mixins/keyboard-shortcut-mixin";
import { HaFuse } from "../../../resources/fuse";
import { haStyle, haStyleDialog } from "../../../resources/styles"; import { haStyle, haStyleDialog } from "../../../resources/styles";
import type { HomeAssistant } from "../../../types"; import type { HomeAssistant } from "../../../types";
import { isMac } from "../../../util/is_mac";
import { showToast } from "../../../util/toast";
import type { AddAutomationElementDialogParams } from "./show-add-automation-element-dialog"; import type { AddAutomationElementDialogParams } from "./show-add-automation-element-dialog";
import { PASTE_VALUE } from "./show-add-automation-element-dialog"; import { PASTE_VALUE } from "./show-add-automation-element-dialog";
import { HaFuse } from "../../../resources/fuse";
const TYPES = { const TYPES = {
trigger: { groups: TRIGGER_GROUPS, icons: TRIGGER_ICONS }, trigger: { groups: TRIGGER_GROUPS, icons: TRIGGER_ICONS },
@@ -93,10 +85,7 @@ const ENTITY_DOMAINS_OTHER = new Set([
const ENTITY_DOMAINS_MAIN = new Set(["notify"]); const ENTITY_DOMAINS_MAIN = new Set(["notify"]);
@customElement("add-automation-element-dialog") @customElement("add-automation-element-dialog")
class DialogAddAutomationElement class DialogAddAutomationElement extends LitElement implements HassDialog {
extends KeyboardShortcutMixin(LitElement)
implements HassDialog
{
@property({ attribute: false }) public hass!: HomeAssistant; @property({ attribute: false }) public hass!: HomeAssistant;
@state() private _params?: AddAutomationElementDialogParams; @state() private _params?: AddAutomationElementDialogParams;
@@ -119,14 +108,9 @@ class DialogAddAutomationElement
@state() private _height?: number; @state() private _height?: number;
@state() private _narrow = false;
public showDialog(params): void { public showDialog(params): void {
this._params = params; this._params = params;
this._group = params.group; this._group = params.group;
this.addKeyboardShortcuts();
if (this._params?.type === "action") { if (this._params?.type === "action") {
this.hass.loadBackendTranslation("services"); this.hass.loadBackendTranslation("services");
this._fetchManifests(); this._fetchManifests();
@@ -136,12 +120,9 @@ class DialogAddAutomationElement
this._fullScreen = matchMedia( this._fullScreen = matchMedia(
"all and (max-width: 450px), all and (max-height: 500px)" "all and (max-width: 450px), all and (max-height: 500px)"
).matches; ).matches;
this._narrow = matchMedia("(max-width: 870px)").matches;
} }
public closeDialog() { public closeDialog() {
this.removeKeyboardShortcuts();
if (this._params) { if (this._params) {
fireEvent(this, "dialog-closed", { dialog: this.localName }); fireEvent(this, "dialog-closed", { dialog: this.localName });
} }
@@ -574,37 +555,15 @@ class DialogAddAutomationElement
.value=${PASTE_VALUE} .value=${PASTE_VALUE}
@click=${this._selected} @click=${this._selected}
> >
<div class="shortcut-label">
<div class="label">
<div>
${this.hass.localize( ${this.hass.localize(
`ui.panel.config.automation.editor.${this._params.type}s.paste` `ui.panel.config.automation.editor.${this._params.type}s.paste`
)} )}
</div> <span slot="supporting-text"
<div class="supporting-text"> >${this.hass.localize(
${this.hass.localize(
// @ts-ignore // @ts-ignore
`ui.panel.config.automation.editor.${this._params.type}s.type.${this._params.clipboardItem}.label` `ui.panel.config.automation.editor.${this._params.type}s.type.${this._params.clipboardItem}.label`
)}
</div>
</div>
${!this._narrow
? html`<span class="shortcut">
<span
>${isMac
? html`<ha-svg-icon
slot="start"
.path=${mdiAppleKeyboardCommand}
></ha-svg-icon>`
: this.hass.localize(
"ui.panel.config.automation.editor.ctrl"
)}</span )}</span
> >
<span>+</span>
<span>V</span>
</span>`
: nothing}
</div>
<ha-svg-icon <ha-svg-icon
slot="start" slot="start"
.path=${mdiContentPaste} .path=${mdiContentPaste}
@@ -612,7 +571,7 @@ class DialogAddAutomationElement
><ha-svg-icon slot="end" .path=${mdiPlus}></ha-svg-icon> ><ha-svg-icon slot="end" .path=${mdiPlus}></ha-svg-icon>
</ha-md-list-item> </ha-md-list-item>
<ha-md-divider role="separator" tabindex="-1"></ha-md-divider>` <ha-md-divider role="separator" tabindex="-1"></ha-md-divider>`
: nothing} : ""}
${repeat( ${repeat(
items, items,
(item) => item.key, (item) => item.key,
@@ -678,30 +637,6 @@ class DialogAddAutomationElement
this._filter = ev.detail.value; this._filter = ev.detail.value;
} }
private _addClipboard = () => {
if (this._params?.clipboardItem) {
this._params!.add(PASTE_VALUE);
showToast(this, {
message: this.hass.localize(
"ui.panel.config.automation.editor.item_pasted",
{
item: this.hass.localize(
// @ts-ignore
`ui.panel.config.automation.editor.${this._params.type}s.type.${this._params.clipboardItem}.label`
),
}
),
});
this.closeDialog();
}
};
protected supportedShortcuts(): SupportedShortcuts {
return {
v: () => this._addClipboard(),
};
}
static get styles(): CSSResultGroup { static get styles(): CSSResultGroup {
return [ return [
haStyle, haStyle,
@@ -725,7 +660,6 @@ class DialogAddAutomationElement
max-width: 100vw; max-width: 100vw;
--md-list-item-leading-space: 24px; --md-list-item-leading-space: 24px;
--md-list-item-trailing-space: 24px; --md-list-item-trailing-space: 24px;
--md-list-item-supporting-text-font: var(--ha-font-size-s);
} }
ha-md-list-item img { ha-md-list-item img {
width: 24px; width: 24px;
@@ -734,27 +668,6 @@ class DialogAddAutomationElement
display: block; display: block;
margin: 0 16px; margin: 0 16px;
} }
.shortcut-label {
display: flex;
gap: 12px;
justify-content: space-between;
}
.shortcut-label .supporting-text {
color: var(--secondary-text-color);
font-size: var(--ha-font-size-s);
}
.shortcut-label .shortcut {
--mdc-icon-size: 12px;
display: inline-flex;
flex-direction: row;
align-items: center;
gap: 2px;
}
.shortcut-label .shortcut span {
font-size: var(--ha-font-size-s);
font-family: var(--ha-font-family-code);
color: var(--ha-color-text-secondary);
}
`, `,
]; ];
} }

View File

@@ -103,7 +103,8 @@ export default class HaAutomationConditionEditor extends LitElement {
if (!ev.detail.isValid) { if (!ev.detail.isValid) {
return; return;
} }
fireEvent(this, "yaml-changed", { value: ev.detail.value }); // @ts-ignore
fireEvent(this, "value-changed", { value: ev.detail.value, yaml: true });
} }
private _onUiChanged(ev: CustomEvent) { private _onUiChanged(ev: CustomEvent) {

View File

@@ -4,12 +4,12 @@ import {
mdiArrowUp, mdiArrowUp,
mdiContentCopy, mdiContentCopy,
mdiContentCut, mdiContentCut,
mdiContentDuplicate,
mdiDelete, mdiDelete,
mdiDotsVertical, mdiDotsVertical,
mdiFlask, mdiFlask,
mdiPlayCircleOutline, mdiPlayCircleOutline,
mdiPlaylistEdit, mdiPlaylistEdit,
mdiPlusCircleMultipleOutline,
mdiRenameBox, mdiRenameBox,
mdiStopCircleOutline, mdiStopCircleOutline,
} from "@mdi/js"; } from "@mdi/js";
@@ -53,7 +53,6 @@ import {
showPromptDialog, showPromptDialog,
} from "../../../../dialogs/generic/show-dialog-box"; } from "../../../../dialogs/generic/show-dialog-box";
import type { HomeAssistant } from "../../../../types"; import type { HomeAssistant } from "../../../../types";
import { showToast } from "../../../../util/toast";
import "../ha-automation-editor-warning"; import "../ha-automation-editor-warning";
import { rowStyles } from "../styles"; import { rowStyles } from "../styles";
import "./ha-automation-condition-editor"; import "./ha-automation-condition-editor";
@@ -117,8 +116,6 @@ export default class HaAutomationConditionRow extends LitElement {
@property({ type: Boolean }) public narrow = false; @property({ type: Boolean }) public narrow = false;
@property({ type: Boolean }) public highlight?: boolean;
@property({ type: Boolean, attribute: "sort-selected" }) @property({ type: Boolean, attribute: "sort-selected" })
public sortSelected = false; public sortSelected = false;
@@ -155,10 +152,6 @@ export default class HaAutomationConditionRow extends LitElement {
@query("ha-automation-row") @query("ha-automation-row")
private _automationRowElement?: HaAutomationRow; private _automationRowElement?: HaAutomationRow;
get selected() {
return this._selected;
}
private _renderRow() { private _renderRow() {
return html` return html`
<ha-svg-icon <ha-svg-icon
@@ -219,7 +212,7 @@ export default class HaAutomationConditionRow extends LitElement {
)} )}
<ha-svg-icon <ha-svg-icon
slot="start" slot="start"
.path=${mdiPlusCircleMultipleOutline} .path=${mdiContentDuplicate}
></ha-svg-icon> ></ha-svg-icon>
</ha-md-menu-item> </ha-md-menu-item>
@@ -362,16 +355,12 @@ export default class HaAutomationConditionRow extends LitElement {
)} )}
.collapsed=${this._collapsed} .collapsed=${this._collapsed}
.selected=${this._selected} .selected=${this._selected}
.highlight=${this.highlight} @click=${this._toggleSidebar}
@toggle-collapsed=${this._toggleCollapse}
.buildingBlock=${CONDITION_BUILDING_BLOCKS.includes( .buildingBlock=${CONDITION_BUILDING_BLOCKS.includes(
this.condition.condition this.condition.condition
)} )}
.sortSelected=${this.sortSelected} .sortSelected=${this.sortSelected}
@click=${this._toggleSidebar}
@toggle-collapsed=${this._toggleCollapse}
@copy-row=${this._copyCondition}
@cut-row=${this._cutCondition}
@delete-row=${this._onDelete}
>${this._renderRow()}</ha-automation-row >${this._renderRow()}</ha-automation-row
>` >`
: html` : html`
@@ -488,15 +477,6 @@ export default class HaAutomationConditionRow extends LitElement {
this._testingResult = undefined; this._testingResult = undefined;
this._testing = true; this._testing = true;
const condition = this.condition; const condition = this.condition;
requestAnimationFrame(() => {
// @ts-ignore is supported in all browsers expect firefox
if (this.scrollIntoViewIfNeeded) {
// @ts-ignore is supported in all browsers expect firefox
this.scrollIntoViewIfNeeded();
return;
}
this.scrollIntoView();
});
try { try {
const validateResult = await validateConfig(this.hass, { const validateResult = await validateConfig(this.hass, {
@@ -587,12 +567,6 @@ export default class HaAutomationConditionRow extends LitElement {
private _copyCondition = () => { private _copyCondition = () => {
this._setClipboard(); this._setClipboard();
showToast(this, {
message: this.hass.localize(
"ui.panel.config.automation.editor.conditions.copied_to_clipboard"
),
duration: 2000,
});
}; };
private _cutCondition = () => { private _cutCondition = () => {
@@ -601,12 +575,6 @@ export default class HaAutomationConditionRow extends LitElement {
if (this._selected) { if (this._selected) {
fireEvent(this, "close-sidebar"); fireEvent(this, "close-sidebar");
} }
showToast(this, {
message: this.hass.localize(
"ui.panel.config.automation.editor.conditions.cut_to_clipboard"
),
duration: 2000,
});
}; };
private _moveUp = () => { private _moveUp = () => {
@@ -667,7 +635,8 @@ export default class HaAutomationConditionRow extends LitElement {
ev?.stopPropagation(); ev?.stopPropagation();
if (this._selected) { if (this._selected) {
fireEvent(this, "request-close-sidebar"); this._selected = false;
fireEvent(this, "close-sidebar");
return; return;
} }
this.openSidebar(); this.openSidebar();
@@ -691,7 +660,7 @@ export default class HaAutomationConditionRow extends LitElement {
}, },
toggleYamlMode: () => { toggleYamlMode: () => {
this._toggleYamlMode(); this._toggleYamlMode();
this.openSidebar(); return this._yamlMode;
}, },
disable: this._onDisable, disable: this._onDisable,
delete: this._onDelete, delete: this._onDelete,
@@ -707,12 +676,12 @@ export default class HaAutomationConditionRow extends LitElement {
this._collapsed = false; this._collapsed = false;
if (this.narrow) { if (this.narrow) {
window.setTimeout(() => { requestAnimationFrame(() => {
this.scrollIntoView({ this.scrollIntoView({
block: "start", block: "start",
behavior: "smooth", behavior: "smooth",
}); });
}, 180); // duration of transition of added padding for bottom sheet });
} }
} }
@@ -725,6 +694,10 @@ export default class HaAutomationConditionRow extends LitElement {
this._collapsed = !this._collapsed; this._collapsed = !this._collapsed;
} }
public isSelected() {
return this._selected;
}
public focus() { public focus() {
this._automationRowElement?.focus(); this._automationRowElement?.focus();
} }

View File

@@ -6,10 +6,12 @@ import { customElement, property, queryAll, state } from "lit/decorators";
import { repeat } from "lit/directives/repeat"; import { repeat } from "lit/directives/repeat";
import { storage } from "../../../../common/decorators/storage"; import { storage } from "../../../../common/decorators/storage";
import { fireEvent } from "../../../../common/dom/fire_event"; import { fireEvent } from "../../../../common/dom/fire_event";
import { listenMediaQuery } from "../../../../common/dom/media_query";
import { nextRender } from "../../../../common/util/render-status"; import { nextRender } from "../../../../common/util/render-status";
import "../../../../components/ha-button"; import "../../../../components/ha-button";
import "../../../../components/ha-button-menu"; import "../../../../components/ha-button-menu";
import "../../../../components/ha-sortable"; import "../../../../components/ha-sortable";
import type { HaSortableClonedEventData } from "../../../../components/ha-sortable";
import "../../../../components/ha-svg-icon"; import "../../../../components/ha-svg-icon";
import type { import type {
AutomationClipboard, AutomationClipboard,
@@ -42,6 +44,8 @@ export default class HaAutomationCondition extends LitElement {
@property({ type: Boolean, attribute: "sidebar" }) public optionsInSidebar = @property({ type: Boolean, attribute: "sidebar" }) public optionsInSidebar =
false; false;
@state() private _showReorder = false;
@state() private _rowSortSelected?: number; @state() private _rowSortSelected?: number;
@state() @state()
@@ -62,6 +66,21 @@ export default class HaAutomationCondition extends LitElement {
private _conditionKeys = new WeakMap<Condition, string>(); private _conditionKeys = new WeakMap<Condition, string>();
private _unsubMql?: () => void;
public connectedCallback() {
super.connectedCallback();
this._unsubMql = listenMediaQuery("(min-width: 600px)", (matches) => {
this._showReorder = matches;
});
}
public disconnectedCallback() {
super.disconnectedCallback();
this._unsubMql?.();
this._unsubMql = undefined;
}
protected updated(changedProperties: PropertyValues) { protected updated(changedProperties: PropertyValues) {
if (!changedProperties.has("conditions")) { if (!changedProperties.has("conditions")) {
return; return;
@@ -146,12 +165,13 @@ export default class HaAutomationCondition extends LitElement {
<ha-sortable <ha-sortable
handle-selector=".handle" handle-selector=".handle"
draggable-selector="ha-automation-condition-row" draggable-selector="ha-automation-condition-row"
.disabled=${this.disabled} .disabled=${!this._showReorder || this.disabled}
group="conditions" group="conditions"
invert-swap invert-swap
@item-moved=${this._conditionMoved} @item-moved=${this._conditionMoved}
@item-added=${this._conditionAdded} @item-added=${this._conditionAdded}
@item-removed=${this._conditionRemoved} @item-removed=${this._conditionRemoved}
@item-cloned=${this._conditionCloned}
> >
<div class="rows ${!this.optionsInSidebar ? "no-sidebar" : ""}"> <div class="rows ${!this.optionsInSidebar ? "no-sidebar" : ""}">
${repeat( ${repeat(
@@ -173,12 +193,12 @@ export default class HaAutomationCondition extends LitElement {
@move-up=${this._moveUp} @move-up=${this._moveUp}
@value-changed=${this._conditionChanged} @value-changed=${this._conditionChanged}
.hass=${this.hass} .hass=${this.hass}
.highlight=${this.highlightedConditions?.includes(cond)} ?highlight=${this.highlightedConditions?.includes(cond)}
.optionsInSidebar=${this.optionsInSidebar} .optionsInSidebar=${this.optionsInSidebar}
.sortSelected=${this._rowSortSelected === idx} .sortSelected=${this._rowSortSelected === idx}
@stop-sort-selection=${this._stopSortSelection} @stop-sort-selection=${this._stopSortSelection}
> >
${!this.disabled ${this._showReorder && !this.disabled
? html` ? html`
<div <div
tabindex="0" tabindex="0"
@@ -317,8 +337,11 @@ export default class HaAutomationCondition extends LitElement {
private async _conditionAdded(ev: CustomEvent): Promise<void> { private async _conditionAdded(ev: CustomEvent): Promise<void> {
ev.stopPropagation(); ev.stopPropagation();
const { index, data } = ev.detail; const { index, data } = ev.detail;
const item = ev.detail.item as HaAutomationConditionRow; let selected = false;
const selected = item.selected; if (data?.["ha-automation-row-selected"]) {
selected = true;
delete data["ha-automation-row-selected"];
}
let conditions = [ let conditions = [
...this.conditions.slice(0, index), ...this.conditions.slice(0, index),
data, data,
@@ -356,6 +379,12 @@ export default class HaAutomationCondition extends LitElement {
fireEvent(this, "value-changed", { value: conditions }); fireEvent(this, "value-changed", { value: conditions });
} }
private _conditionCloned(ev: CustomEvent<HaSortableClonedEventData>) {
if (ev.detail.item.isSelected()) {
ev.detail.item.condition["ha-automation-row-selected"] = true;
}
}
private _conditionChanged(ev: CustomEvent) { private _conditionChanged(ev: CustomEvent) {
ev.stopPropagation(); ev.stopPropagation();
const conditions = [...this.conditions]; const conditions = [...this.conditions];

View File

@@ -1,6 +1,7 @@
import { consume } from "@lit/context"; import { consume } from "@lit/context";
import { import {
mdiCog, mdiCog,
mdiContentDuplicate,
mdiContentSave, mdiContentSave,
mdiDebugStepOver, mdiDebugStepOver,
mdiDelete, mdiDelete,
@@ -10,18 +11,20 @@ import {
mdiPlay, mdiPlay,
mdiPlayCircleOutline, mdiPlayCircleOutline,
mdiPlaylistEdit, mdiPlaylistEdit,
mdiPlusCircleMultipleOutline,
mdiRenameBox, mdiRenameBox,
mdiRobotConfused, mdiRobotConfused,
mdiStopCircleOutline, mdiStopCircleOutline,
mdiTag, mdiTag,
mdiTransitConnection, mdiTransitConnection,
mdiUnfoldLessHorizontal,
mdiUnfoldMoreHorizontal,
} from "@mdi/js"; } from "@mdi/js";
import type { UnsubscribeFunc } from "home-assistant-js-websocket"; import type { UnsubscribeFunc } from "home-assistant-js-websocket";
import type { CSSResultGroup, PropertyValues, TemplateResult } from "lit"; import type { CSSResultGroup, PropertyValues, TemplateResult } from "lit";
import { css, html, LitElement, nothing } from "lit"; import { css, html, LitElement, nothing } from "lit";
import { property, query, state } from "lit/decorators"; import { property, query, state } from "lit/decorators";
import { classMap } from "lit/directives/class-map"; import { classMap } from "lit/directives/class-map";
import { isArray } from "@tsparticles/engine";
import { transform } from "../../../common/decorators/transform"; import { transform } from "../../../common/decorators/transform";
import { fireEvent } from "../../../common/dom/fire_event"; import { fireEvent } from "../../../common/dom/fire_event";
import { navigate } from "../../../common/navigate"; import { navigate } from "../../../common/navigate";
@@ -41,6 +44,7 @@ import type {
AutomationConfig, AutomationConfig,
AutomationEntity, AutomationEntity,
BlueprintAutomationConfig, BlueprintAutomationConfig,
Condition,
} from "../../../data/automation"; } from "../../../data/automation";
import { import {
deleteAutomation, deleteAutomation,
@@ -335,7 +339,7 @@ export class HaAutomationEditor extends PreventUnsavedMixin(
)} )}
<ha-svg-icon <ha-svg-icon
slot="graphic" slot="graphic"
.path=${mdiPlusCircleMultipleOutline} .path=${mdiContentDuplicate}
></ha-svg-icon> ></ha-svg-icon>
</ha-list-item> </ha-list-item>
@@ -369,6 +373,30 @@ export class HaAutomationEditor extends PreventUnsavedMixin(
<ha-svg-icon slot="graphic" .path=${mdiPlaylistEdit}></ha-svg-icon> <ha-svg-icon slot="graphic" .path=${mdiPlaylistEdit}></ha-svg-icon>
</ha-list-item> </ha-list-item>
${!useBlueprint
? html`
<ha-list-item graphic="icon" @click=${this._collapseAll}>
<ha-svg-icon
slot="graphic"
.path=${mdiUnfoldLessHorizontal}
></ha-svg-icon>
${this.hass.localize(
"ui.panel.config.automation.editor.collapse_all"
)}
</ha-list-item>
<ha-list-item graphic="icon" @click=${this._expandAll}>
<ha-svg-icon
slot="graphic"
.path=${mdiUnfoldMoreHorizontal}
></ha-svg-icon>
${this.hass.localize(
"ui.panel.config.automation.editor.expand_all"
)}
</ha-list-item>
`
: nothing}
<li divider role="separator"></li> <li divider role="separator"></li>
<ha-list-item <ha-list-item
@@ -1036,8 +1064,28 @@ export class HaAutomationEditor extends PreventUnsavedMixin(
}); });
} }
// Move conditions at top of action to automation condition key
const configToSave = { ...this._config! };
configToSave.conditions = !configToSave.conditions
? []
: isArray(configToSave.conditions)
? [...configToSave.conditions]
: [configToSave.conditions];
configToSave.actions = !configToSave.actions
? []
: isArray(configToSave.actions)
? [...configToSave.actions]
: [configToSave.actions];
while (
configToSave.actions.length > 0 &&
"condition" in configToSave.actions[0]
) {
configToSave.conditions.push(configToSave.actions.shift() as Condition);
}
try { try {
await saveAutomationConfig(this.hass, id, this._config!); await saveAutomationConfig(this.hass, id, configToSave);
if (this._entityRegistryUpdate !== undefined) { if (this._entityRegistryUpdate !== undefined) {
let entityId = this._entityId; let entityId = this._entityId;
@@ -1112,10 +1160,6 @@ export class HaAutomationEditor extends PreventUnsavedMixin(
protected supportedShortcuts(): SupportedShortcuts { protected supportedShortcuts(): SupportedShortcuts {
return { return {
s: () => this._handleSaveAutomation(), s: () => this._handleSaveAutomation(),
c: () => this._copySelectedRow(),
x: () => this._cutSelectedRow(),
Delete: () => this._deleteSelectedRow(),
Backspace: () => this._deleteSelectedRow(),
}; };
} }
@@ -1127,28 +1171,14 @@ export class HaAutomationEditor extends PreventUnsavedMixin(
return this._confirmUnsavedChanged(); return this._confirmUnsavedChanged();
} }
// @ts-ignore
private _collapseAll() { private _collapseAll() {
this._manualEditor?.collapseAll(); this._manualEditor?.collapseAll();
} }
// @ts-ignore
private _expandAll() { private _expandAll() {
this._manualEditor?.expandAll(); this._manualEditor?.expandAll();
} }
private _copySelectedRow() {
this._manualEditor?.copySelectedRow();
}
private _cutSelectedRow() {
this._manualEditor?.cutSelectedRow();
}
private _deleteSelectedRow() {
this._manualEditor?.deleteSelectedRow();
}
static get styles(): CSSResultGroup { static get styles(): CSSResultGroup {
return [ return [
haStyle, haStyle,

View File

@@ -292,7 +292,7 @@ class HaAutomationPicker extends SubscribeMixin(LitElement) {
extraTemplate: (automation) => extraTemplate: (automation) =>
automation.labels.length automation.labels.length
? html`<ha-data-table-labels ? html`<ha-data-table-labels
@label-clicked=${narrow ? undefined : this._labelClicked} @label-clicked=${this._labelClicked}
.labels=${automation.labels} .labels=${automation.labels}
></ha-data-table-labels>` ></ha-data-table-labels>`
: nothing, : nothing,

View File

@@ -1,5 +1,6 @@
import { css, html, LitElement, nothing } from "lit"; import { css, html, LitElement, nothing } from "lit";
import { customElement, property, query, state } from "lit/decorators"; import { customElement, property, query, state } from "lit/decorators";
import { fireEvent } from "../../../common/dom/fire_event";
import "../../../components/ha-bottom-sheet"; import "../../../components/ha-bottom-sheet";
import type { HaBottomSheet } from "../../../components/ha-bottom-sheet"; import type { HaBottomSheet } from "../../../components/ha-bottom-sheet";
import { import {
@@ -33,8 +34,6 @@ export default class HaAutomationSidebar extends LitElement {
@property({ type: Boolean }) public narrow = false; @property({ type: Boolean }) public narrow = false;
@property({ attribute: "sidebar-key" }) public sidebarKey?: string;
@state() private _yamlMode = false; @state() private _yamlMode = false;
@query("ha-bottom-sheet") private _bottomSheetElement?: HaBottomSheet; @query("ha-bottom-sheet") private _bottomSheetElement?: HaBottomSheet;
@@ -53,9 +52,8 @@ export default class HaAutomationSidebar extends LitElement {
.narrow=${this.narrow} .narrow=${this.narrow}
.disabled=${this.disabled} .disabled=${this.disabled}
.yamlMode=${this._yamlMode} .yamlMode=${this._yamlMode}
.sidebarKey=${this.sidebarKey}
@toggle-yaml-mode=${this._toggleYamlMode} @toggle-yaml-mode=${this._toggleYamlMode}
@close-sidebar=${this.triggerCloseSidebar} @close-sidebar=${this._handleCloseSidebar}
></ha-automation-sidebar-trigger> ></ha-automation-sidebar-trigger>
`; `;
} }
@@ -69,9 +67,8 @@ export default class HaAutomationSidebar extends LitElement {
.narrow=${this.narrow} .narrow=${this.narrow}
.disabled=${this.disabled} .disabled=${this.disabled}
.yamlMode=${this._yamlMode} .yamlMode=${this._yamlMode}
.sidebarKey=${this.sidebarKey}
@toggle-yaml-mode=${this._toggleYamlMode} @toggle-yaml-mode=${this._toggleYamlMode}
@close-sidebar=${this.triggerCloseSidebar} @close-sidebar=${this._handleCloseSidebar}
></ha-automation-sidebar-condition> ></ha-automation-sidebar-condition>
`; `;
} }
@@ -85,9 +82,8 @@ export default class HaAutomationSidebar extends LitElement {
.narrow=${this.narrow} .narrow=${this.narrow}
.disabled=${this.disabled} .disabled=${this.disabled}
.yamlMode=${this._yamlMode} .yamlMode=${this._yamlMode}
.sidebarKey=${this.sidebarKey}
@toggle-yaml-mode=${this._toggleYamlMode} @toggle-yaml-mode=${this._toggleYamlMode}
@close-sidebar=${this.triggerCloseSidebar} @close-sidebar=${this._handleCloseSidebar}
></ha-automation-sidebar-action> ></ha-automation-sidebar-action>
`; `;
} }
@@ -100,7 +96,7 @@ export default class HaAutomationSidebar extends LitElement {
.isWide=${this.isWide} .isWide=${this.isWide}
.narrow=${this.narrow} .narrow=${this.narrow}
.disabled=${this.disabled} .disabled=${this.disabled}
@close-sidebar=${this.triggerCloseSidebar} @close-sidebar=${this._handleCloseSidebar}
></ha-automation-sidebar-option> ></ha-automation-sidebar-option>
`; `;
} }
@@ -114,9 +110,8 @@ export default class HaAutomationSidebar extends LitElement {
.narrow=${this.narrow} .narrow=${this.narrow}
.disabled=${this.disabled} .disabled=${this.disabled}
.yamlMode=${this._yamlMode} .yamlMode=${this._yamlMode}
.sidebarKey=${this.sidebarKey}
@toggle-yaml-mode=${this._toggleYamlMode} @toggle-yaml-mode=${this._toggleYamlMode}
@close-sidebar=${this.triggerCloseSidebar} @close-sidebar=${this._handleCloseSidebar}
></ha-automation-sidebar-script-field-selector> ></ha-automation-sidebar-script-field-selector>
`; `;
} }
@@ -130,9 +125,8 @@ export default class HaAutomationSidebar extends LitElement {
.narrow=${this.narrow} .narrow=${this.narrow}
.disabled=${this.disabled} .disabled=${this.disabled}
.yamlMode=${this._yamlMode} .yamlMode=${this._yamlMode}
.sidebarKey=${this.sidebarKey}
@toggle-yaml-mode=${this._toggleYamlMode} @toggle-yaml-mode=${this._toggleYamlMode}
@close-sidebar=${this.triggerCloseSidebar} @close-sidebar=${this._handleCloseSidebar}
></ha-automation-sidebar-script-field> ></ha-automation-sidebar-script-field>
`; `;
} }
@@ -188,8 +182,8 @@ export default class HaAutomationSidebar extends LitElement {
return undefined; return undefined;
} }
public triggerCloseSidebar(ev?: CustomEvent) { private _handleCloseSidebar(ev: CustomEvent) {
ev?.stopPropagation(); ev.stopPropagation();
if (this.narrow) { if (this.narrow) {
this._bottomSheetElement?.closeSheet(); this._bottomSheetElement?.closeSheet();
return; return;
@@ -203,12 +197,17 @@ export default class HaAutomationSidebar extends LitElement {
} }
private _toggleYamlMode = () => { private _toggleYamlMode = () => {
(this.config as ActionSidebarConfig)?.toggleYamlMode(); this._yamlMode = this.config!.toggleYamlMode();
fireEvent(this, "value-changed", {
value: {
...this.config,
yamlMode: this._yamlMode,
},
});
}; };
static styles = css` static styles = css`
:host { :host {
z-index: 6;
outline: none; outline: none;
height: 100%; height: 100%;
--ha-card-border-radius: var( --ha-card-border-radius: var(
@@ -236,8 +235,5 @@ declare global {
interface HASSDomEvents { interface HASSDomEvents {
"toggle-yaml-mode": undefined; "toggle-yaml-mode": undefined;
"yaml-changed": {
value: unknown;
};
} }
} }

View File

@@ -3,13 +3,7 @@ import type { HassEntity } from "home-assistant-js-websocket";
import { load } from "js-yaml"; import { load } from "js-yaml";
import type { CSSResultGroup, PropertyValues } from "lit"; import type { CSSResultGroup, PropertyValues } from "lit";
import { css, html, LitElement, nothing } from "lit"; import { css, html, LitElement, nothing } from "lit";
import { import { customElement, property, query, state } from "lit/decorators";
customElement,
property,
query,
queryAll,
state,
} from "lit/decorators";
import { classMap } from "lit/directives/class-map"; import { classMap } from "lit/directives/class-map";
import { import {
any, any,
@@ -34,7 +28,6 @@ import "../../../components/ha-fab";
import "../../../components/ha-icon-button"; import "../../../components/ha-icon-button";
import "../../../components/ha-markdown"; import "../../../components/ha-markdown";
import type { import type {
ActionSidebarConfig,
AutomationConfig, AutomationConfig,
Condition, Condition,
ManualAutomationConfig, ManualAutomationConfig,
@@ -52,8 +45,6 @@ import { documentationUrl } from "../../../util/documentation-url";
import { showToast } from "../../../util/toast"; import { showToast } from "../../../util/toast";
import "./action/ha-automation-action"; import "./action/ha-automation-action";
import type HaAutomationAction from "./action/ha-automation-action"; import type HaAutomationAction from "./action/ha-automation-action";
import "./condition/ha-automation-condition";
import type HaAutomationCondition from "./condition/ha-automation-condition";
import "./ha-automation-sidebar"; import "./ha-automation-sidebar";
import type HaAutomationSidebar from "./ha-automation-sidebar"; import type HaAutomationSidebar from "./ha-automation-sidebar";
import { showPasteReplaceDialog } from "./paste-replace-dialog/show-dialog-paste-replace"; import { showPasteReplaceDialog } from "./paste-replace-dialog/show-dialog-paste-replace";
@@ -99,15 +90,8 @@ export class HaManualAutomationEditor extends LitElement {
@state() private _sidebarConfig?: SidebarConfig; @state() private _sidebarConfig?: SidebarConfig;
@state() private _sidebarKey?: string;
@query("ha-automation-sidebar") private _sidebarElement?: HaAutomationSidebar; @query("ha-automation-sidebar") private _sidebarElement?: HaAutomationSidebar;
@queryAll("ha-automation-action, ha-automation-condition")
private _collapsableElements?: NodeListOf<
HaAutomationAction | HaAutomationCondition
>;
private _previousConfig?: ManualAutomationConfig; private _previousConfig?: ManualAutomationConfig;
public connectedCallback() { public connectedCallback() {
@@ -166,59 +150,12 @@ export class HaManualAutomationEditor extends LitElement {
.disabled=${this.disabled || this.saving} .disabled=${this.disabled || this.saving}
.narrow=${this.narrow} .narrow=${this.narrow}
@open-sidebar=${this._openSidebar} @open-sidebar=${this._openSidebar}
@request-close-sidebar=${this._triggerCloseSidebar} @request-close-sidebar=${this._closeSidebar}
@close-sidebar=${this._handleCloseSidebar} @close-sidebar=${this._handleCloseSidebar}
root root
sidebar sidebar
></ha-automation-trigger> ></ha-automation-trigger>
<div class="header">
<h2 id="conditions-heading" class="name">
${this.hass.localize(
"ui.panel.config.automation.editor.conditions.header"
)}
<span class="small"
>(${this.hass.localize("ui.common.optional")})</span
>
</h2>
<a
href=${documentationUrl(this.hass, "/docs/automation/condition/")}
target="_blank"
rel="noreferrer"
>
<ha-icon-button
.path=${mdiHelpCircle}
.label=${this.hass.localize(
"ui.panel.config.automation.editor.conditions.learn_more"
)}
></ha-icon-button>
</a>
</div>
${!ensureArray(this.config.conditions)?.length
? html`<p>
${this.hass.localize(
"ui.panel.config.automation.editor.conditions.description",
{ user: this.hass.user?.name || "Alice" }
)}
</p>`
: nothing}
<ha-automation-condition
role="region"
aria-labelledby="conditions-heading"
.conditions=${this.config.conditions || []}
.highlightedConditions=${this._pastedConfig?.conditions || []}
@value-changed=${this._conditionChanged}
.hass=${this.hass}
.disabled=${this.disabled || this.saving}
.narrow=${this.narrow}
@open-sidebar=${this._openSidebar}
@request-close-sidebar=${this._triggerCloseSidebar}
@close-sidebar=${this._handleCloseSidebar}
root
sidebar
></ha-automation-condition>
<div class="header"> <div class="header">
<h2 id="actions-heading" class="name"> <h2 id="actions-heading" class="name">
${this.hass.localize( ${this.hass.localize(
@@ -255,7 +192,7 @@ export class HaManualAutomationEditor extends LitElement {
.highlightedActions=${this._pastedConfig?.actions || []} .highlightedActions=${this._pastedConfig?.actions || []}
@value-changed=${this._actionChanged} @value-changed=${this._actionChanged}
@open-sidebar=${this._openSidebar} @open-sidebar=${this._openSidebar}
@request-close-sidebar=${this._triggerCloseSidebar} @request-close-sidebar=${this._closeSidebar}
@close-sidebar=${this._handleCloseSidebar} @close-sidebar=${this._handleCloseSidebar}
.hass=${this.hass} .hass=${this.hass}
.narrow=${this.narrow} .narrow=${this.narrow}
@@ -274,11 +211,7 @@ export class HaManualAutomationEditor extends LitElement {
})} })}
> >
<div class="content-wrapper"> <div class="content-wrapper">
<div <div class="content">
class="content ${this._sidebarConfig && this.narrow
? "has-bottom-sheet"
: ""}"
>
<slot name="alerts"></slot> <slot name="alerts"></slot>
${this._renderContent()} ${this._renderContent()}
</div> </div>
@@ -305,7 +238,6 @@ export class HaManualAutomationEditor extends LitElement {
.config=${this._sidebarConfig} .config=${this._sidebarConfig}
@value-changed=${this._sidebarConfigChanged} @value-changed=${this._sidebarConfigChanged}
.disabled=${this.disabled} .disabled=${this.disabled}
.sidebarKey=${this._sidebarKey}
></ha-automation-sidebar> ></ha-automation-sidebar>
</div> </div>
</div> </div>
@@ -333,7 +265,6 @@ export class HaManualAutomationEditor extends LitElement {
// deselect previous selected row // deselect previous selected row
this._sidebarConfig?.close?.(); this._sidebarConfig?.close?.();
this._sidebarConfig = ev.detail; this._sidebarConfig = ev.detail;
this._sidebarKey = JSON.stringify(this._sidebarConfig);
await this._sidebarElement?.updateComplete; await this._sidebarElement?.updateComplete;
this._sidebarElement?.focus(); this._sidebarElement?.focus();
@@ -351,12 +282,8 @@ export class HaManualAutomationEditor extends LitElement {
}; };
} }
private _triggerCloseSidebar() { private _closeSidebar() {
if (this._sidebarConfig) { if (this._sidebarConfig) {
if (this._sidebarElement) {
this._sidebarElement.triggerCloseSidebar();
return;
}
this._sidebarConfig?.close(); this._sidebarConfig?.close();
} }
} }
@@ -393,7 +320,7 @@ export class HaManualAutomationEditor extends LitElement {
} }
private _saveAutomation() { private _saveAutomation() {
this._triggerCloseSidebar(); this._closeSidebar();
fireEvent(this, "save-automation"); fireEvent(this, "save-automation");
} }
@@ -498,12 +425,7 @@ export class HaManualAutomationEditor extends LitElement {
if (normalized) { if (normalized) {
ev.preventDefault(); ev.preventDefault();
if ( if (this.dirty) {
this.dirty ||
ensureArray(this.config.triggers)?.length ||
ensureArray(this.config.conditions)?.length ||
ensureArray(this.config.actions)?.length
) {
const result = await new Promise<boolean>((resolve) => { const result = await new Promise<boolean>((resolve) => {
showPasteReplaceDialog(this, { showPasteReplaceDialog(this, {
domain: "automation", domain: "automation",
@@ -612,36 +534,24 @@ export class HaManualAutomationEditor extends LitElement {
}); });
} }
private _getCollapsableElements() {
return this.shadowRoot!.querySelectorAll<HaAutomationAction>(
"ha-automation-action"
);
}
public expandAll() { public expandAll() {
this._collapsableElements?.forEach((element) => { this._getCollapsableElements().forEach((element) => {
element.expandAll(); element.expandAll();
}); });
} }
public collapseAll() { public collapseAll() {
this._collapsableElements?.forEach((element) => { this._getCollapsableElements().forEach((element) => {
element.collapseAll(); element.collapseAll();
}); });
} }
public copySelectedRow() {
if ((this._sidebarConfig as ActionSidebarConfig)?.copy) {
(this._sidebarConfig as ActionSidebarConfig).copy();
}
}
public cutSelectedRow() {
if ((this._sidebarConfig as ActionSidebarConfig)?.cut) {
(this._sidebarConfig as ActionSidebarConfig).cut();
}
}
public deleteSelectedRow() {
if ((this._sidebarConfig as ActionSidebarConfig)?.delete) {
(this._sidebarConfig as ActionSidebarConfig).delete();
}
}
static get styles(): CSSResultGroup { static get styles(): CSSResultGroup {
return [ return [
saveFabStyles, saveFabStyles,

View File

@@ -2,9 +2,9 @@ import { consume } from "@lit/context";
import { import {
mdiArrowDown, mdiArrowDown,
mdiArrowUp, mdiArrowUp,
mdiContentDuplicate,
mdiDelete, mdiDelete,
mdiDotsVertical, mdiDotsVertical,
mdiPlusCircleMultipleOutline,
mdiRenameBox, mdiRenameBox,
} from "@mdi/js"; } from "@mdi/js";
import type { CSSResultGroup } from "lit"; import type { CSSResultGroup } from "lit";
@@ -86,10 +86,6 @@ export default class HaAutomationOptionRow extends LitElement {
@query("ha-automation-row") @query("ha-automation-row")
private _automationRowElement?: HaAutomationRow; private _automationRowElement?: HaAutomationRow;
get selected() {
return this._selected;
}
private _expandedChanged(ev) { private _expandedChanged(ev) {
if (ev.currentTarget.id !== "option") { if (ev.currentTarget.id !== "option") {
return; return;
@@ -171,7 +167,7 @@ export default class HaAutomationOptionRow extends LitElement {
)} )}
<ha-svg-icon <ha-svg-icon
slot="start" slot="start"
.path=${mdiPlusCircleMultipleOutline} .path=${mdiContentDuplicate}
></ha-svg-icon> ></ha-svg-icon>
</ha-md-menu-item> </ha-md-menu-item>
@@ -275,10 +271,9 @@ export default class HaAutomationOptionRow extends LitElement {
left-chevron left-chevron
.collapsed=${this._collapsed} .collapsed=${this._collapsed}
.selected=${this._selected} .selected=${this._selected}
.sortSelected=${this.sortSelected}
@click=${this._toggleSidebar} @click=${this._toggleSidebar}
@toggle-collapsed=${this._toggleCollapse} @toggle-collapsed=${this._toggleCollapse}
@delete-row=${this._removeOption} .sortSelected=${this.sortSelected}
>${this._renderRow()}</ha-automation-row >${this._renderRow()}</ha-automation-row
>` >`
: html` : html`
@@ -381,7 +376,8 @@ export default class HaAutomationOptionRow extends LitElement {
ev?.stopPropagation(); ev?.stopPropagation();
if (this._selected) { if (this._selected) {
fireEvent(this, "request-close-sidebar"); this._selected = false;
fireEvent(this, "close-sidebar");
return; return;
} }
this.openSidebar(); this.openSidebar();
@@ -399,6 +395,7 @@ export default class HaAutomationOptionRow extends LitElement {
rename: () => { rename: () => {
this._renameOption(); this._renameOption();
}, },
toggleYamlMode: () => false, // no yaml mode for options
delete: this._removeOption, delete: this._removeOption,
duplicate: this._duplicateOption, duplicate: this._duplicateOption,
defaultOption: !!this.defaultActions, defaultOption: !!this.defaultActions,
@@ -407,12 +404,12 @@ export default class HaAutomationOptionRow extends LitElement {
this._collapsed = false; this._collapsed = false;
if (this.narrow) { if (this.narrow) {
window.setTimeout(() => { requestAnimationFrame(() => {
this.scrollIntoView({ this.scrollIntoView({
block: "start", block: "start",
behavior: "smooth", behavior: "smooth",
}); });
}, 180); // duration of transition of added padding for bottom sheet });
} }
} }
@@ -449,6 +446,10 @@ export default class HaAutomationOptionRow extends LitElement {
this._collapsed = !this._collapsed; this._collapsed = !this._collapsed;
} }
public isSelected() {
return this._selected;
}
public focus() { public focus() {
this._automationRowElement?.focus(); this._automationRowElement?.focus();
} }

View File

@@ -6,9 +6,11 @@ import { customElement, property, queryAll, state } from "lit/decorators";
import { repeat } from "lit/directives/repeat"; import { repeat } from "lit/directives/repeat";
import { storage } from "../../../../common/decorators/storage"; import { storage } from "../../../../common/decorators/storage";
import { fireEvent } from "../../../../common/dom/fire_event"; import { fireEvent } from "../../../../common/dom/fire_event";
import { listenMediaQuery } from "../../../../common/dom/media_query";
import { nextRender } from "../../../../common/util/render-status"; import { nextRender } from "../../../../common/util/render-status";
import "../../../../components/ha-button"; import "../../../../components/ha-button";
import "../../../../components/ha-sortable"; import "../../../../components/ha-sortable";
import type { HaSortableClonedEventData } from "../../../../components/ha-sortable";
import "../../../../components/ha-svg-icon"; import "../../../../components/ha-svg-icon";
import type { AutomationClipboard } from "../../../../data/automation"; import type { AutomationClipboard } from "../../../../data/automation";
import type { Option } from "../../../../data/script"; import type { Option } from "../../../../data/script";
@@ -33,6 +35,8 @@ export default class HaAutomationOption extends LitElement {
@property({ type: Boolean, attribute: "show-default" }) @property({ type: Boolean, attribute: "show-default" })
public showDefaultActions = false; public showDefaultActions = false;
@state() private _showReorder = false;
@state() private _rowSortSelected?: number; @state() private _rowSortSelected?: number;
@state() @state()
@@ -53,17 +57,33 @@ export default class HaAutomationOption extends LitElement {
private _optionsKeys = new WeakMap<Option, string>(); private _optionsKeys = new WeakMap<Option, string>();
private _unsubMql?: () => void;
public connectedCallback() {
super.connectedCallback();
this._unsubMql = listenMediaQuery("(min-width: 600px)", (matches) => {
this._showReorder = matches;
});
}
public disconnectedCallback() {
super.disconnectedCallback();
this._unsubMql?.();
this._unsubMql = undefined;
}
protected render() { protected render() {
return html` return html`
<ha-sortable <ha-sortable
handle-selector=".handle" handle-selector=".handle"
draggable-selector="ha-automation-option-row" draggable-selector="ha-automation-option-row"
.disabled=${this.disabled} .disabled=${!this._showReorder || this.disabled}
group="options" group="options"
invert-swap invert-swap
@item-moved=${this._optionMoved} @item-moved=${this._optionMoved}
@item-added=${this._optionAdded} @item-added=${this._optionAdded}
@item-removed=${this._optionRemoved} @item-removed=${this._optionRemoved}
@item-cloned=${this._optionCloned}
> >
<div class="rows ${!this.optionsInSidebar ? "no-sidebar" : ""}"> <div class="rows ${!this.optionsInSidebar ? "no-sidebar" : ""}">
${repeat( ${repeat(
@@ -87,7 +107,7 @@ export default class HaAutomationOption extends LitElement {
.sortSelected=${this._rowSortSelected === idx} .sortSelected=${this._rowSortSelected === idx}
@stop-sort-selection=${this._stopSortSelection} @stop-sort-selection=${this._stopSortSelection}
> >
${!this.disabled ${this._showReorder && !this.disabled
? html` ? html`
<div <div
tabindex="0" tabindex="0"
@@ -239,8 +259,11 @@ export default class HaAutomationOption extends LitElement {
private async _optionAdded(ev: CustomEvent): Promise<void> { private async _optionAdded(ev: CustomEvent): Promise<void> {
ev.stopPropagation(); ev.stopPropagation();
const { index, data } = ev.detail; const { index, data } = ev.detail;
const item = ev.detail.item as HaAutomationOptionRow; let selected = false;
const selected = item.selected; if (data?.["ha-automation-row-selected"]) {
selected = true;
delete data["ha-automation-row-selected"];
}
const options = [ const options = [
...this.options.slice(0, index), ...this.options.slice(0, index),
@@ -268,6 +291,12 @@ export default class HaAutomationOption extends LitElement {
fireEvent(this, "value-changed", { value: options }); fireEvent(this, "value-changed", { value: options });
} }
private _optionCloned(ev: CustomEvent<HaSortableClonedEventData>) {
if (ev.detail.item.isSelected()) {
ev.detail.item.option["ha-automation-row-selected"] = true;
}
}
private _optionChanged(ev: CustomEvent) { private _optionChanged(ev: CustomEvent) {
ev.stopPropagation(); ev.stopPropagation();
const options = [...this.options]; const options = [...this.options];

View File

@@ -1,12 +1,11 @@
import { css, type CSSResultGroup, html, LitElement, nothing } from "lit";
import { customElement, property, state } from "lit/decorators"; import { customElement, property, state } from "lit/decorators";
import { fireEvent } from "../../../../common/dom/fire_event"; import { css, type CSSResultGroup, html, LitElement, nothing } from "lit";
import "../../../../components/ha-button";
import { createCloseHeading } from "../../../../components/ha-dialog";
import "../../../../components/ha-yaml-editor";
import type { HassDialog } from "../../../../dialogs/make-dialog-manager"; import type { HassDialog } from "../../../../dialogs/make-dialog-manager";
import { haStyle, haStyleDialog } from "../../../../resources/styles";
import type { HomeAssistant } from "../../../../types"; import type { HomeAssistant } from "../../../../types";
import { fireEvent } from "../../../../common/dom/fire_event";
import { haStyle, haStyleDialog } from "../../../../resources/styles";
import { createCloseHeading } from "../../../../components/ha-dialog";
import "../trigger/ha-automation-trigger-row";
import type { PasteReplaceDialogParams } from "./show-dialog-paste-replace"; import type { PasteReplaceDialogParams } from "./show-dialog-paste-replace";
@customElement("ha-dialog-paste-replace") @customElement("ha-dialog-paste-replace")

View File

@@ -1,18 +1,16 @@
import { import {
mdiAppleKeyboardCommand,
mdiContentCopy, mdiContentCopy,
mdiContentCut, mdiContentCut,
mdiContentDuplicate,
mdiDelete, mdiDelete,
mdiPlay, mdiPlay,
mdiPlayCircleOutline, mdiPlayCircleOutline,
mdiPlaylistEdit, mdiPlaylistEdit,
mdiPlusCircleMultipleOutline,
mdiRenameBox, mdiRenameBox,
mdiStopCircleOutline, mdiStopCircleOutline,
} from "@mdi/js"; } from "@mdi/js";
import { html, LitElement, nothing } from "lit"; import { html, LitElement } from "lit";
import { customElement, property, query, state } from "lit/decorators"; import { customElement, property, query, state } from "lit/decorators";
import { keyed } from "lit/directives/keyed";
import { fireEvent } from "../../../../common/dom/fire_event"; import { fireEvent } from "../../../../common/dom/fire_event";
import { handleStructError } from "../../../../common/structs/handle-errors"; import { handleStructError } from "../../../../common/structs/handle-errors";
import type { LocalizeKeys } from "../../../../common/translations/localize"; import type { LocalizeKeys } from "../../../../common/translations/localize";
@@ -22,7 +20,6 @@ import { ACTION_BUILDING_BLOCKS } from "../../../../data/action";
import type { ActionSidebarConfig } from "../../../../data/automation"; import type { ActionSidebarConfig } from "../../../../data/automation";
import type { RepeatAction } from "../../../../data/script"; import type { RepeatAction } from "../../../../data/script";
import type { HomeAssistant } from "../../../../types"; import type { HomeAssistant } from "../../../../types";
import { isMac } from "../../../../util/is_mac";
import type HaAutomationConditionEditor from "../action/ha-automation-action-editor"; import type HaAutomationConditionEditor from "../action/ha-automation-action-editor";
import { getAutomationActionType } from "../action/ha-automation-action-row"; import { getAutomationActionType } from "../action/ha-automation-action-row";
import { getRepeatType } from "../action/types/ha-automation-action-repeat"; import { getRepeatType } from "../action/types/ha-automation-action-repeat";
@@ -44,8 +41,6 @@ export default class HaAutomationSidebarAction extends LitElement {
@property({ type: Boolean }) public narrow = false; @property({ type: Boolean }) public narrow = false;
@property({ attribute: "sidebar-key" }) public sidebarKey?: string;
@state() private _warnings?: string[]; @state() private _warnings?: string[];
@query(".sidebar-editor") @query(".sidebar-editor")
@@ -105,24 +100,18 @@ export default class HaAutomationSidebarAction extends LitElement {
<span slot="subtitle">${subtitle}</span> <span slot="subtitle">${subtitle}</span>
<ha-md-menu-item slot="menu-items" .clickAction=${this.config.run}> <ha-md-menu-item slot="menu-items" .clickAction=${this.config.run}>
<ha-svg-icon slot="start" .path=${mdiPlay}></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize("ui.panel.config.automation.editor.actions.run")} ${this.hass.localize("ui.panel.config.automation.editor.actions.run")}
<span class="shortcut-placeholder ${isMac ? "mac" : ""}"></span> <ha-svg-icon slot="start" .path=${mdiPlay}></ha-svg-icon>
</div>
</ha-md-menu-item> </ha-md-menu-item>
<ha-md-menu-item <ha-md-menu-item
slot="menu-items" slot="menu-items"
.clickAction=${this.config.rename} .clickAction=${this.config.rename}
.disabled=${!!disabled} .disabled=${!!disabled}
> >
<ha-svg-icon slot="start" .path=${mdiRenameBox}></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize( ${this.hass.localize(
"ui.panel.config.automation.editor.triggers.rename" "ui.panel.config.automation.editor.triggers.rename"
)} )}
<span class="shortcut-placeholder ${isMac ? "mac" : ""}"></span> <ha-svg-icon slot="start" .path=${mdiRenameBox}></ha-svg-icon>
</div>
</ha-md-menu-item> </ha-md-menu-item>
<ha-md-divider <ha-md-divider
slot="menu-items" slot="menu-items"
@@ -134,85 +123,36 @@ export default class HaAutomationSidebarAction extends LitElement {
.clickAction=${this.config.duplicate} .clickAction=${this.config.duplicate}
.disabled=${this.disabled} .disabled=${this.disabled}
> >
<ha-svg-icon
slot="start"
.path=${mdiPlusCircleMultipleOutline}
></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize( ${this.hass.localize(
"ui.panel.config.automation.editor.actions.duplicate" "ui.panel.config.automation.editor.actions.duplicate"
)} )}
<span class="shortcut-placeholder ${isMac ? "mac" : ""}"></span> <ha-svg-icon slot="start" .path=${mdiContentDuplicate}></ha-svg-icon>
</div>
</ha-md-menu-item> </ha-md-menu-item>
<ha-md-menu-item <ha-md-menu-item
slot="menu-items" slot="menu-items"
.clickAction=${this.config.copy} .clickAction=${this.config.copy}
.disabled=${this.disabled} .disabled=${this.disabled}
> >
${this.hass.localize("ui.panel.config.automation.editor.triggers.copy")}
<ha-svg-icon slot="start" .path=${mdiContentCopy}></ha-svg-icon> <ha-svg-icon slot="start" .path=${mdiContentCopy}></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize(
"ui.panel.config.automation.editor.triggers.copy"
)}
${!this.narrow
? html`<span class="shortcut">
<span
>${isMac
? html`<ha-svg-icon
slot="start"
.path=${mdiAppleKeyboardCommand}
></ha-svg-icon>`
: this.hass.localize(
"ui.panel.config.automation.editor.ctrl"
)}</span
>
<span>+</span>
<span>C</span>
</span>`
: nothing}
</div>
</ha-md-menu-item> </ha-md-menu-item>
<ha-md-menu-item <ha-md-menu-item
slot="menu-items" slot="menu-items"
.clickAction=${this.config.cut} .clickAction=${this.config.cut}
.disabled=${this.disabled} .disabled=${this.disabled}
> >
${this.hass.localize("ui.panel.config.automation.editor.triggers.cut")}
<ha-svg-icon slot="start" .path=${mdiContentCut}></ha-svg-icon> <ha-svg-icon slot="start" .path=${mdiContentCut}></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize(
"ui.panel.config.automation.editor.triggers.cut"
)}
${!this.narrow
? html`<span class="shortcut">
<span
>${isMac
? html`<ha-svg-icon
slot="start"
.path=${mdiAppleKeyboardCommand}
></ha-svg-icon>`
: this.hass.localize(
"ui.panel.config.automation.editor.ctrl"
)}</span
>
<span>+</span>
<span>X</span>
</span>`
: nothing}
</div>
</ha-md-menu-item> </ha-md-menu-item>
<ha-md-menu-item <ha-md-menu-item
slot="menu-items" slot="menu-items"
.clickAction=${this._toggleYamlMode} .clickAction=${this._toggleYamlMode}
.disabled=${!this.config.uiSupported || !!this._warnings} .disabled=${!this.config.uiSupported || !!this._warnings}
> >
<ha-svg-icon slot="start" .path=${mdiPlaylistEdit}></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize( ${this.hass.localize(
`ui.panel.config.automation.editor.edit_${!this.yamlMode ? "yaml" : "ui"}` `ui.panel.config.automation.editor.edit_${!this.yamlMode ? "yaml" : "ui"}`
)} )}
<span class="shortcut-placeholder ${isMac ? "mac" : ""}"></span> <ha-svg-icon slot="start" .path=${mdiPlaylistEdit}></ha-svg-icon>
</div>
</ha-md-menu-item> </ha-md-menu-item>
<ha-md-divider <ha-md-divider
slot="menu-items" slot="menu-items"
@@ -220,16 +160,13 @@ export default class HaAutomationSidebarAction extends LitElement {
tabindex="-1" tabindex="-1"
></ha-md-divider> ></ha-md-divider>
<ha-md-menu-item slot="menu-items" .clickAction=${this.config.disable}> <ha-md-menu-item slot="menu-items" .clickAction=${this.config.disable}>
${this.hass.localize(
`ui.panel.config.automation.editor.actions.${disabled ? "enable" : "disable"}`
)}
<ha-svg-icon <ha-svg-icon
slot="start" slot="start"
.path=${this.disabled ? mdiPlayCircleOutline : mdiStopCircleOutline} .path=${this.disabled ? mdiPlayCircleOutline : mdiStopCircleOutline}
></ha-svg-icon> ></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize(
`ui.panel.config.automation.editor.actions.${disabled ? "enable" : "disable"}`
)}
<span class="shortcut-placeholder ${isMac ? "mac" : ""}"></span>
</div>
</ha-md-menu-item> </ha-md-menu-item>
<ha-md-menu-item <ha-md-menu-item
slot="menu-items" slot="menu-items"
@@ -237,51 +174,25 @@ export default class HaAutomationSidebarAction extends LitElement {
.disabled=${this.disabled} .disabled=${this.disabled}
class="warning" class="warning"
> >
<ha-svg-icon slot="start" .path=${mdiDelete}></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize( ${this.hass.localize(
"ui.panel.config.automation.editor.actions.delete" "ui.panel.config.automation.editor.actions.delete"
)} )}
${!this.narrow <ha-svg-icon slot="start" .path=${mdiDelete}></ha-svg-icon>
? html`<span class="shortcut">
<span
>${isMac
? html`<ha-svg-icon
slot="start"
.path=${mdiAppleKeyboardCommand}
></ha-svg-icon>`
: this.hass.localize(
"ui.panel.config.automation.editor.ctrl"
)}</span
>
<span>+</span>
<span
>${this.hass.localize(
"ui.panel.config.automation.editor.del"
)}</span
>
</span>`
: nothing}
</div>
</ha-md-menu-item> </ha-md-menu-item>
${description && !this.yamlMode ${description && !this.yamlMode
? html`<div class="description">${description}</div>` ? html`<div class="description">${description}</div>`
: keyed( : html`<ha-automation-action-editor
this.sidebarKey,
html`<ha-automation-action-editor
class="sidebar-editor" class="sidebar-editor"
.hass=${this.hass} .hass=${this.hass}
.action=${actionConfig} .action=${actionConfig}
.yamlMode=${this.yamlMode} .yamlMode=${this.yamlMode}
.uiSupported=${this.config.uiSupported} .uiSupported=${this.config.uiSupported}
@value-changed=${this._valueChangedSidebar} @value-changed=${this._valueChangedSidebar}
@yaml-changed=${this._yamlChangedSidebar}
sidebar sidebar
narrow narrow
.disabled=${this.disabled} .disabled=${this.disabled}
@ui-mode-not-available=${this._handleUiModeNotAvailable} @ui-mode-not-available=${this._handleUiModeNotAvailable}
></ha-automation-action-editor>` ></ha-automation-action-editor>`}
)}
</ha-automation-sidebar-card>`; </ha-automation-sidebar-card>`;
} }
@@ -309,12 +220,6 @@ export default class HaAutomationSidebarAction extends LitElement {
} }
} }
private _yamlChangedSidebar(ev: CustomEvent) {
ev.stopPropagation();
this.config?.save?.(ev.detail.value);
}
private _toggleYamlMode = () => { private _toggleYamlMode = () => {
fireEvent(this, "toggle-yaml-mode"); fireEvent(this, "toggle-yaml-mode");
}; };

View File

@@ -1,6 +1,4 @@
import { ResizeController } from "@lit-labs/observers/resize-controller";
import { mdiClose, mdiDotsVertical } from "@mdi/js"; import { mdiClose, mdiDotsVertical } from "@mdi/js";
import type { PropertyValues } from "lit";
import { css, html, LitElement, nothing } from "lit"; import { css, html, LitElement, nothing } from "lit";
import { import {
customElement, customElement,
@@ -45,22 +43,7 @@ export default class HaAutomationSidebarCard extends LitElement {
@state() private _contentScrolled = false; @state() private _contentScrolled = false;
@state() private _contentScrollable = false; @query(".card-content") private _contentElement?: HTMLDivElement;
@query(".card-content") private _contentElement!: HTMLDivElement;
private _contentSize = new ResizeController(this, {
target: null,
callback: (entries) => {
if (entries[0]?.target) {
this._canScrollDown(entries[0].target);
}
},
});
protected firstUpdated(_changedProperties: PropertyValues): void {
this._contentSize.observe(this._contentElement);
}
protected render() { protected render() {
return html` return html`
@@ -111,29 +94,14 @@ export default class HaAutomationSidebarCard extends LitElement {
<div class="card-content" @scroll=${this._onScroll}> <div class="card-content" @scroll=${this._onScroll}>
<slot></slot> <slot></slot>
</div> </div>
${this.narrow
? html`
<div
class="fade ${this._contentScrollable ? "scrollable" : ""}"
></div>
`
: nothing}
</ha-card> </ha-card>
`; `;
} }
@eventOptions({ passive: true }) @eventOptions({ passive: true })
private _onScroll(ev) { private _onScroll() {
const top = ev.target.scrollTop ?? 0; const top = this._contentElement?.scrollTop ?? 0;
this._contentScrolled = top > 0; this._contentScrolled = top > 0;
this._canScrollDown(ev.target);
}
private _canScrollDown(element: HTMLElement) {
this._contentScrollable =
(element.scrollHeight ?? 0) - (element.clientHeight ?? 0) >
(element.scrollTop ?? 0);
} }
private _closeSidebar() { private _closeSidebar() {
@@ -157,7 +125,6 @@ export default class HaAutomationSidebarCard extends LitElement {
@media all and (max-width: 870px) { @media all and (max-width: 870px) {
ha-card.mobile { ha-card.mobile {
border: none; border: none;
box-shadow: none;
} }
ha-card.mobile { ha-card.mobile {
border-bottom-right-radius: var(--ha-border-radius-square); border-bottom-right-radius: var(--ha-border-radius-square);
@@ -171,6 +138,7 @@ export default class HaAutomationSidebarCard extends LitElement {
transition: box-shadow 180ms ease-in-out; transition: box-shadow 180ms ease-in-out;
border-bottom-left-radius: 0; border-bottom-left-radius: 0;
border-bottom-right-radius: 0; border-bottom-right-radius: 0;
z-index: 6;
position: relative; position: relative;
background-color: var( background-color: var(
--ha-dialog-surface-background, --ha-dialog-surface-background,
@@ -179,28 +147,12 @@ export default class HaAutomationSidebarCard extends LitElement {
} }
ha-dialog-header.scrolled { ha-dialog-header.scrolled {
box-shadow: var(--bar-box-shadow); box-shadow: 0 2px 8px rgba(0, 0, 0, 0.12);
}
.fade {
position: fixed;
bottom: -12px;
left: 0;
right: 0;
height: 12px;
pointer-events: none;
transition: box-shadow 180ms ease-in-out;
}
.fade.scrollable {
box-shadow: var(--bar-box-shadow);
transform: rotate(180deg);
} }
.card-content { .card-content {
max-height: calc(100% - 80px); max-height: calc(100% - 80px);
overflow: auto; overflow: auto;
margin-top: 0;
} }
@media (min-width: 450px) and (min-height: 500px) { @media (min-width: 450px) and (min-height: 500px) {

View File

@@ -1,30 +1,21 @@
import { import {
mdiAppleKeyboardCommand,
mdiContentCopy, mdiContentCopy,
mdiContentCut, mdiContentCut,
mdiContentDuplicate,
mdiDelete, mdiDelete,
mdiFlask, mdiFlask,
mdiPlayCircleOutline, mdiPlayCircleOutline,
mdiPlaylistEdit, mdiPlaylistEdit,
mdiPlusCircleMultipleOutline,
mdiRenameBox, mdiRenameBox,
mdiStopCircleOutline, mdiStopCircleOutline,
} from "@mdi/js"; } from "@mdi/js";
import { css, html, LitElement, nothing } from "lit"; import { html, LitElement } from "lit";
import { customElement, property, query, state } from "lit/decorators"; import { customElement, property, query, state } from "lit/decorators";
import { classMap } from "lit/directives/class-map";
import { keyed } from "lit/directives/keyed";
import { fireEvent } from "../../../../common/dom/fire_event"; import { fireEvent } from "../../../../common/dom/fire_event";
import { handleStructError } from "../../../../common/structs/handle-errors"; import { handleStructError } from "../../../../common/structs/handle-errors";
import { import type { ConditionSidebarConfig } from "../../../../data/automation";
testCondition,
type ConditionSidebarConfig,
} from "../../../../data/automation";
import { CONDITION_BUILDING_BLOCKS } from "../../../../data/condition"; import { CONDITION_BUILDING_BLOCKS } from "../../../../data/condition";
import { validateConfig } from "../../../../data/config";
import type { HomeAssistant } from "../../../../types"; import type { HomeAssistant } from "../../../../types";
import { isMac } from "../../../../util/is_mac";
import { showAlertDialog } from "../../../lovelace/custom-card-helpers";
import "../condition/ha-automation-condition-editor"; import "../condition/ha-automation-condition-editor";
import type HaAutomationConditionEditor from "../condition/ha-automation-condition-editor"; import type HaAutomationConditionEditor from "../condition/ha-automation-condition-editor";
import { sidebarEditorStyles } from "../styles"; import { sidebarEditorStyles } from "../styles";
@@ -44,14 +35,8 @@ export default class HaAutomationSidebarCondition extends LitElement {
@property({ type: Boolean }) public narrow = false; @property({ type: Boolean }) public narrow = false;
@property({ attribute: "sidebar-key" }) public sidebarKey?: string;
@state() private _warnings?: string[]; @state() private _warnings?: string[];
@state() private _testing = false;
@state() private _testingResult?: boolean;
@query(".sidebar-editor") @query(".sidebar-editor")
public editor?: HaAutomationConditionEditor; public editor?: HaAutomationConditionEditor;
@@ -65,10 +50,6 @@ export default class HaAutomationSidebarCondition extends LitElement {
} }
} }
} }
// Reset testing state when condition changes
if (changedProperties.has("sidebarKey")) {
this._testing = false;
}
} }
protected render() { protected render() {
@@ -104,27 +85,21 @@ export default class HaAutomationSidebarCondition extends LitElement {
> >
<span slot="title">${title}</span> <span slot="title">${title}</span>
<span slot="subtitle">${subtitle}</span> <span slot="subtitle">${subtitle}</span>
<ha-md-menu-item slot="menu-items" .clickAction=${this._testCondition}> <ha-md-menu-item slot="menu-items" .clickAction=${this.config.test}>
<ha-svg-icon slot="start" .path=${mdiFlask}></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize( ${this.hass.localize(
"ui.panel.config.automation.editor.conditions.test" "ui.panel.config.automation.editor.conditions.test"
)} )}
<span class="shortcut-placeholder ${isMac ? "mac" : ""}"></span> <ha-svg-icon slot="start" .path=${mdiFlask}></ha-svg-icon>
</div>
</ha-md-menu-item> </ha-md-menu-item>
<ha-md-menu-item <ha-md-menu-item
slot="menu-items" slot="menu-items"
.clickAction=${this.config.rename} .clickAction=${this.config.rename}
.disabled=${!!disabled} .disabled=${!!disabled}
> >
<ha-svg-icon slot="start" .path=${mdiRenameBox}></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize( ${this.hass.localize(
"ui.panel.config.automation.editor.triggers.rename" "ui.panel.config.automation.editor.triggers.rename"
)} )}
<span class="shortcut-placeholder ${isMac ? "mac" : ""}"></span> <ha-svg-icon slot="start" .path=${mdiRenameBox}></ha-svg-icon>
</div>
</ha-md-menu-item> </ha-md-menu-item>
<ha-md-divider <ha-md-divider
@@ -138,16 +113,10 @@ export default class HaAutomationSidebarCondition extends LitElement {
.clickAction=${this.config.duplicate} .clickAction=${this.config.duplicate}
.disabled=${this.disabled} .disabled=${this.disabled}
> >
<ha-svg-icon
slot="start"
.path=${mdiPlusCircleMultipleOutline}
></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize( ${this.hass.localize(
"ui.panel.config.automation.editor.actions.duplicate" "ui.panel.config.automation.editor.actions.duplicate"
)} )}
<span class="shortcut-placeholder ${isMac ? "mac" : ""}"></span> <ha-svg-icon slot="start" .path=${mdiContentDuplicate}></ha-svg-icon>
</div>
</ha-md-menu-item> </ha-md-menu-item>
<ha-md-menu-item <ha-md-menu-item
@@ -155,28 +124,8 @@ export default class HaAutomationSidebarCondition extends LitElement {
.clickAction=${this.config.copy} .clickAction=${this.config.copy}
.disabled=${this.disabled} .disabled=${this.disabled}
> >
${this.hass.localize("ui.panel.config.automation.editor.triggers.copy")}
<ha-svg-icon slot="start" .path=${mdiContentCopy}></ha-svg-icon> <ha-svg-icon slot="start" .path=${mdiContentCopy}></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize(
"ui.panel.config.automation.editor.triggers.copy"
)}
${!this.narrow
? html`<span class="shortcut">
<span
>${isMac
? html`<ha-svg-icon
slot="start"
.path=${mdiAppleKeyboardCommand}
></ha-svg-icon>`
: this.hass.localize(
"ui.panel.config.automation.editor.ctrl"
)}</span
>
<span>+</span>
<span>C</span>
</span>`
: nothing}
</div>
</ha-md-menu-item> </ha-md-menu-item>
<ha-md-menu-item <ha-md-menu-item
@@ -184,41 +133,18 @@ export default class HaAutomationSidebarCondition extends LitElement {
.clickAction=${this.config.cut} .clickAction=${this.config.cut}
.disabled=${this.disabled} .disabled=${this.disabled}
> >
${this.hass.localize("ui.panel.config.automation.editor.triggers.cut")}
<ha-svg-icon slot="start" .path=${mdiContentCut}></ha-svg-icon> <ha-svg-icon slot="start" .path=${mdiContentCut}></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize(
"ui.panel.config.automation.editor.triggers.cut"
)}
${!this.narrow
? html`<span class="shortcut">
<span
>${isMac
? html`<ha-svg-icon
slot="start"
.path=${mdiAppleKeyboardCommand}
></ha-svg-icon>`
: this.hass.localize(
"ui.panel.config.automation.editor.ctrl"
)}</span
>
<span>+</span>
<span>X</span>
</span>`
: nothing}
</div>
</ha-md-menu-item> </ha-md-menu-item>
<ha-md-menu-item <ha-md-menu-item
slot="menu-items" slot="menu-items"
.clickAction=${this._toggleYamlMode} .clickAction=${this._toggleYamlMode}
.disabled=${!this.config.uiSupported || !!this._warnings} .disabled=${!this.config.uiSupported || !!this._warnings}
> >
<ha-svg-icon slot="start" .path=${mdiPlaylistEdit}></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize( ${this.hass.localize(
`ui.panel.config.automation.editor.edit_${!this.yamlMode ? "yaml" : "ui"}` `ui.panel.config.automation.editor.edit_${!this.yamlMode ? "yaml" : "ui"}`
)} )}
<span class="shortcut-placeholder ${isMac ? "mac" : ""}"></span> <ha-svg-icon slot="start" .path=${mdiPlaylistEdit}></ha-svg-icon>
</div>
</ha-md-menu-item> </ha-md-menu-item>
<ha-md-divider <ha-md-divider
slot="menu-items" slot="menu-items"
@@ -226,16 +152,13 @@ export default class HaAutomationSidebarCondition extends LitElement {
tabindex="-1" tabindex="-1"
></ha-md-divider> ></ha-md-divider>
<ha-md-menu-item slot="menu-items" .clickAction=${this.config.disable}> <ha-md-menu-item slot="menu-items" .clickAction=${this.config.disable}>
${this.hass.localize(
`ui.panel.config.automation.editor.actions.${disabled ? "enable" : "disable"}`
)}
<ha-svg-icon <ha-svg-icon
slot="start" slot="start"
.path=${this.disabled ? mdiPlayCircleOutline : mdiStopCircleOutline} .path=${this.disabled ? mdiPlayCircleOutline : mdiStopCircleOutline}
></ha-svg-icon> ></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize(
`ui.panel.config.automation.editor.actions.${disabled ? "enable" : "disable"}`
)}
<span class="shortcut-placeholder ${isMac ? "mac" : ""}"></span>
</div>
</ha-md-menu-item> </ha-md-menu-item>
<ha-md-menu-item <ha-md-menu-item
slot="menu-items" slot="menu-items"
@@ -243,128 +166,27 @@ export default class HaAutomationSidebarCondition extends LitElement {
.disabled=${this.disabled} .disabled=${this.disabled}
class="warning" class="warning"
> >
<ha-svg-icon slot="start" .path=${mdiDelete}></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize( ${this.hass.localize(
"ui.panel.config.automation.editor.actions.delete" "ui.panel.config.automation.editor.actions.delete"
)} )}
${!this.narrow <ha-svg-icon slot="start" .path=${mdiDelete}></ha-svg-icon>
? html`<span class="shortcut">
<span
>${isMac
? html`<ha-svg-icon
slot="start"
.path=${mdiAppleKeyboardCommand}
></ha-svg-icon>`
: this.hass.localize(
"ui.panel.config.automation.editor.ctrl"
)}</span
>
<span>+</span>
<span
>${this.hass.localize(
"ui.panel.config.automation.editor.del"
)}</span
>
</span>`
: nothing}
</div>
</ha-md-menu-item> </ha-md-menu-item>
${description && !this.yamlMode ${description && !this.yamlMode
? html`<div class="description">${description}</div>` ? html`<div class="description">${description}</div>`
: keyed( : html`<ha-automation-condition-editor
this.sidebarKey,
html`<ha-automation-condition-editor
class="sidebar-editor" class="sidebar-editor"
.hass=${this.hass} .hass=${this.hass}
.condition=${this.config.config} .condition=${this.config.config}
.yamlMode=${this.yamlMode} .yamlMode=${this.yamlMode}
.uiSupported=${this.config.uiSupported} .uiSupported=${this.config.uiSupported}
@value-changed=${this._valueChangedSidebar} @value-changed=${this._valueChangedSidebar}
@yaml-changed=${this._yamlChangedSidebar}
.disabled=${this.disabled} .disabled=${this.disabled}
@ui-mode-not-available=${this._handleUiModeNotAvailable} @ui-mode-not-available=${this._handleUiModeNotAvailable}
sidebar sidebar
></ha-automation-condition-editor>` ></ha-automation-condition-editor> `}
)}
<div class="testing-wrapper">
<div
class="testing ${classMap({
active: this._testing,
pass: this._testingResult === true,
error: this._testingResult === false,
narrow: this.narrow,
})}"
>
${this._testingResult
? this.hass.localize(
"ui.panel.config.automation.editor.conditions.testing_pass"
)
: this.hass.localize(
"ui.panel.config.automation.editor.conditions.testing_error"
)}
</div>
</div>
</ha-automation-sidebar-card>`; </ha-automation-sidebar-card>`;
} }
private _testCondition = async () => {
if (this._testing) {
return;
}
this._testingResult = undefined;
this._testing = true;
const condition = this.config.config;
try {
const validateResult = await validateConfig(this.hass, {
conditions: condition,
});
// Abort if condition changed.
if (this.config.config !== condition) {
this._testing = false;
return;
}
if (!validateResult.conditions.valid) {
showAlertDialog(this, {
title: this.hass.localize(
"ui.panel.config.automation.editor.conditions.invalid_condition"
),
text: validateResult.conditions.error,
});
this._testing = false;
return;
}
let result: { result: boolean };
try {
result = await testCondition(this.hass, condition);
} catch (err: any) {
if (this.config.config !== condition) {
this._testing = false;
return;
}
showAlertDialog(this, {
title: this.hass.localize(
"ui.panel.config.automation.editor.conditions.test_failed"
),
text: err.message,
});
this._testing = false;
return;
}
this._testingResult = result.result;
} finally {
setTimeout(() => {
this._testing = false;
}, 2500);
}
};
private _handleUiModeNotAvailable(ev: CustomEvent) { private _handleUiModeNotAvailable(ev: CustomEvent) {
this._warnings = handleStructError(this.hass, ev.detail).warnings; this._warnings = handleStructError(this.hass, ev.detail).warnings;
if (!this.yamlMode) { if (!this.yamlMode) {
@@ -387,67 +209,11 @@ export default class HaAutomationSidebarCondition extends LitElement {
} }
} }
private _yamlChangedSidebar(ev: CustomEvent) {
ev.stopPropagation();
this.config?.save?.(ev.detail.value);
}
private _toggleYamlMode = () => { private _toggleYamlMode = () => {
fireEvent(this, "toggle-yaml-mode"); fireEvent(this, "toggle-yaml-mode");
}; };
static styles = [ static styles = sidebarEditorStyles;
sidebarEditorStyles,
css`
ha-automation-sidebar-card {
position: relative;
}
.testing-wrapper {
position: absolute;
top: 0px;
right: 0px;
left: 0px;
margin: -1px;
overflow: hidden;
border-top-right-radius: var(
--ha-card-border-radius,
var(--ha-border-radius-lg)
);
border-top-left-radius: var(
--ha-card-border-radius,
var(--ha-border-radius-lg)
);
pointer-events: none;
height: 100px;
}
.testing {
--testing-color: var(--divider-color, #e0e0e0);
text-transform: uppercase;
font-size: var(--ha-font-size-m);
font-weight: var(--ha-font-weight-bold);
background-color: var(--testing-color);
color: var(--text-primary-color);
max-height: 0px;
transition:
max-height 0.3s ease-in-out,
padding-top 0.3s ease-in-out;
text-align: center;
}
.testing.active.narrow {
padding-top: 16px;
}
.testing.active {
max-height: 100%;
}
.testing.error {
--testing-color: var(--accent-color);
}
.testing.pass {
--testing-color: var(--success-color);
}
`,
];
} }
declare global { declare global {

View File

@@ -1,14 +1,8 @@
import { import { mdiContentDuplicate, mdiDelete, mdiRenameBox } from "@mdi/js";
mdiAppleKeyboardCommand, import { html, LitElement } from "lit";
mdiDelete,
mdiPlusCircleMultipleOutline,
mdiRenameBox,
} from "@mdi/js";
import { html, LitElement, nothing } from "lit";
import { customElement, property, query } from "lit/decorators"; import { customElement, property, query } from "lit/decorators";
import type { OptionSidebarConfig } from "../../../../data/automation"; import type { OptionSidebarConfig } from "../../../../data/automation";
import type { HomeAssistant } from "../../../../types"; import type { HomeAssistant } from "../../../../types";
import { isMac } from "../../../../util/is_mac";
import type HaAutomationConditionEditor from "../action/ha-automation-action-editor"; import type HaAutomationConditionEditor from "../action/ha-automation-action-editor";
import { sidebarEditorStyles } from "../styles"; import { sidebarEditorStyles } from "../styles";
import "./ha-automation-sidebar-card"; import "./ha-automation-sidebar-card";
@@ -58,13 +52,10 @@ export default class HaAutomationSidebarOption extends LitElement {
.clickAction=${this.config.rename} .clickAction=${this.config.rename}
.disabled=${!!disabled} .disabled=${!!disabled}
> >
<ha-svg-icon slot="start" .path=${mdiRenameBox}></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize( ${this.hass.localize(
"ui.panel.config.automation.editor.triggers.rename" "ui.panel.config.automation.editor.triggers.rename"
)} )}
<span class="shortcut-placeholder ${isMac ? "mac" : ""}"></span> <ha-svg-icon slot="start" .path=${mdiRenameBox}></ha-svg-icon>
</div>
</ha-md-menu-item> </ha-md-menu-item>
<ha-md-menu-item <ha-md-menu-item
@@ -72,16 +63,13 @@ export default class HaAutomationSidebarOption extends LitElement {
@click=${this.config.duplicate} @click=${this.config.duplicate}
.disabled=${this.disabled} .disabled=${this.disabled}
> >
<ha-svg-icon
slot="start"
.path=${mdiPlusCircleMultipleOutline}
></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize( ${this.hass.localize(
"ui.panel.config.automation.editor.actions.duplicate" "ui.panel.config.automation.editor.actions.duplicate"
)} )}
<span class="shortcut-placeholder ${isMac ? "mac" : ""}"></span> <ha-svg-icon
</div> slot="start"
.path=${mdiContentDuplicate}
></ha-svg-icon>
</ha-md-menu-item> </ha-md-menu-item>
<ha-md-divider <ha-md-divider
slot="menu-items" slot="menu-items"
@@ -94,32 +82,10 @@ export default class HaAutomationSidebarOption extends LitElement {
.disabled=${this.disabled} .disabled=${this.disabled}
class="warning" class="warning"
> >
<ha-svg-icon slot="start" .path=${mdiDelete}></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize( ${this.hass.localize(
"ui.panel.config.automation.editor.actions.type.choose.remove_option" "ui.panel.config.automation.editor.actions.type.choose.remove_option"
)} )}
${!this.narrow <ha-svg-icon slot="start" .path=${mdiDelete}></ha-svg-icon>
? html`<span class="shortcut">
<span
>${isMac
? html`<ha-svg-icon
slot="start"
.path=${mdiAppleKeyboardCommand}
></ha-svg-icon>`
: this.hass.localize(
"ui.panel.config.automation.editor.ctrl"
)}</span
>
<span>+</span>
<span
>${this.hass.localize(
"ui.panel.config.automation.editor.del"
)}</span
>
</span>`
: nothing}
</div>
</ha-md-menu-item> </ha-md-menu-item>
`} `}

View File

@@ -1,12 +1,10 @@
import { mdiAppleKeyboardCommand, mdiDelete, mdiPlaylistEdit } from "@mdi/js"; import { mdiDelete, mdiPlaylistEdit } from "@mdi/js";
import { html, LitElement, nothing } from "lit"; import { html, LitElement } from "lit";
import { customElement, property, query, state } from "lit/decorators"; import { customElement, property, query, state } from "lit/decorators";
import { keyed } from "lit/directives/keyed";
import { fireEvent } from "../../../../common/dom/fire_event"; import { fireEvent } from "../../../../common/dom/fire_event";
import type { LocalizeKeys } from "../../../../common/translations/localize"; import type { LocalizeKeys } from "../../../../common/translations/localize";
import type { ScriptFieldSidebarConfig } from "../../../../data/automation"; import type { ScriptFieldSidebarConfig } from "../../../../data/automation";
import type { HomeAssistant } from "../../../../types"; import type { HomeAssistant } from "../../../../types";
import { isMac } from "../../../../util/is_mac";
import "../../script/ha-script-field-selector-editor"; import "../../script/ha-script-field-selector-editor";
import type HaAutomationConditionEditor from "../action/ha-automation-action-editor"; import type HaAutomationConditionEditor from "../action/ha-automation-action-editor";
import { sidebarEditorStyles } from "../styles"; import { sidebarEditorStyles } from "../styles";
@@ -26,8 +24,6 @@ export default class HaAutomationSidebarScriptFieldSelector extends LitElement {
@property({ type: Boolean }) public narrow = false; @property({ type: Boolean }) public narrow = false;
@property({ attribute: "sidebar-key" }) public sidebarKey?: string;
@state() private _warnings?: string[]; @state() private _warnings?: string[];
@query(".sidebar-editor") @query(".sidebar-editor")
@@ -69,13 +65,10 @@ export default class HaAutomationSidebarScriptFieldSelector extends LitElement {
.clickAction=${this._toggleYamlMode} .clickAction=${this._toggleYamlMode}
.disabled=${!!this._warnings} .disabled=${!!this._warnings}
> >
<ha-svg-icon slot="start" .path=${mdiPlaylistEdit}></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize( ${this.hass.localize(
`ui.panel.config.automation.editor.edit_${!this.yamlMode ? "yaml" : "ui"}` `ui.panel.config.automation.editor.edit_${!this.yamlMode ? "yaml" : "ui"}`
)} )}
<span class="shortcut-placeholder ${isMac ? "mac" : ""}"></span> <ha-svg-icon slot="start" .path=${mdiPlaylistEdit}></ha-svg-icon>
</div>
</ha-md-menu-item> </ha-md-menu-item>
<ha-md-menu-item <ha-md-menu-item
slot="menu-items" slot="menu-items"
@@ -83,45 +76,19 @@ export default class HaAutomationSidebarScriptFieldSelector extends LitElement {
.disabled=${this.disabled} .disabled=${this.disabled}
class="warning" class="warning"
> >
<ha-svg-icon slot="start" .path=${mdiDelete}></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize( ${this.hass.localize(
"ui.panel.config.automation.editor.actions.delete" "ui.panel.config.automation.editor.actions.delete"
)} )}
${!this.narrow <ha-svg-icon slot="start" .path=${mdiDelete}></ha-svg-icon>
? html`<span class="shortcut">
<span
>${isMac
? html`<ha-svg-icon
slot="start"
.path=${mdiAppleKeyboardCommand}
></ha-svg-icon>`
: this.hass.localize(
"ui.panel.config.automation.editor.ctrl"
)}</span
>
<span>+</span>
<span
>${this.hass.localize(
"ui.panel.config.automation.editor.del"
)}</span
>
</span>`
: nothing}
</div>
</ha-md-menu-item> </ha-md-menu-item>
${keyed( <ha-script-field-selector-editor
this.sidebarKey,
html`<ha-script-field-selector-editor
class="sidebar-editor" class="sidebar-editor"
.hass=${this.hass} .hass=${this.hass}
.field=${this.config.config.field} .field=${this.config.config.field}
.disabled=${this.disabled} .disabled=${this.disabled}
@value-changed=${this._valueChangedSidebar} @value-changed=${this._valueChangedSidebar}
@yaml-changed=${this._yamlChangedSidebar}
.yamlMode=${this.yamlMode} .yamlMode=${this.yamlMode}
></ha-script-field-selector-editor>` ></ha-script-field-selector-editor>
)}
</ha-automation-sidebar-card>`; </ha-automation-sidebar-card>`;
} }
@@ -149,12 +116,6 @@ export default class HaAutomationSidebarScriptFieldSelector extends LitElement {
} }
} }
private _yamlChangedSidebar(ev: CustomEvent) {
ev.stopPropagation();
this.config?.save?.(ev.detail.value);
}
private _toggleYamlMode = () => { private _toggleYamlMode = () => {
fireEvent(this, "toggle-yaml-mode"); fireEvent(this, "toggle-yaml-mode");
}; };

View File

@@ -1,11 +1,9 @@
import { mdiAppleKeyboardCommand, mdiDelete, mdiPlaylistEdit } from "@mdi/js"; import { mdiDelete, mdiPlaylistEdit } from "@mdi/js";
import { html, LitElement, nothing } from "lit"; import { html, LitElement } from "lit";
import { customElement, property, query, state } from "lit/decorators"; import { customElement, property, query, state } from "lit/decorators";
import { keyed } from "lit/directives/keyed";
import { fireEvent } from "../../../../common/dom/fire_event"; import { fireEvent } from "../../../../common/dom/fire_event";
import type { ScriptFieldSidebarConfig } from "../../../../data/automation"; import type { ScriptFieldSidebarConfig } from "../../../../data/automation";
import type { HomeAssistant } from "../../../../types"; import type { HomeAssistant } from "../../../../types";
import { isMac } from "../../../../util/is_mac";
import "../../script/ha-script-field-editor"; import "../../script/ha-script-field-editor";
import type HaAutomationConditionEditor from "../action/ha-automation-action-editor"; import type HaAutomationConditionEditor from "../action/ha-automation-action-editor";
import { sidebarEditorStyles } from "../styles"; import { sidebarEditorStyles } from "../styles";
@@ -25,8 +23,6 @@ export default class HaAutomationSidebarScriptField extends LitElement {
@property({ type: Boolean }) public narrow = false; @property({ type: Boolean }) public narrow = false;
@property({ attribute: "sidebar-key" }) public sidebarKey?: string;
@state() private _warnings?: string[]; @state() private _warnings?: string[];
@query(".sidebar-editor") @query(".sidebar-editor")
@@ -62,13 +58,10 @@ export default class HaAutomationSidebarScriptField extends LitElement {
.clickAction=${this._toggleYamlMode} .clickAction=${this._toggleYamlMode}
.disabled=${!!this._warnings} .disabled=${!!this._warnings}
> >
<ha-svg-icon slot="start" .path=${mdiPlaylistEdit}></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize( ${this.hass.localize(
`ui.panel.config.automation.editor.edit_${!this.yamlMode ? "yaml" : "ui"}` `ui.panel.config.automation.editor.edit_${!this.yamlMode ? "yaml" : "ui"}`
)} )}
<span class="shortcut-placeholder ${isMac ? "mac" : ""}"></span> <ha-svg-icon slot="start" .path=${mdiPlaylistEdit}></ha-svg-icon>
</div>
</ha-md-menu-item> </ha-md-menu-item>
<ha-md-menu-item <ha-md-menu-item
slot="menu-items" slot="menu-items"
@@ -76,36 +69,12 @@ export default class HaAutomationSidebarScriptField extends LitElement {
.disabled=${this.disabled} .disabled=${this.disabled}
class="warning" class="warning"
> >
<ha-svg-icon slot="start" .path=${mdiDelete}></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize( ${this.hass.localize(
"ui.panel.config.automation.editor.actions.delete" "ui.panel.config.automation.editor.actions.delete"
)} )}
${!this.narrow <ha-svg-icon slot="start" .path=${mdiDelete}></ha-svg-icon>
? html`<span class="shortcut">
<span
>${isMac
? html`<ha-svg-icon
slot="start"
.path=${mdiAppleKeyboardCommand}
></ha-svg-icon>`
: this.hass.localize(
"ui.panel.config.automation.editor.ctrl"
)}</span
>
<span>+</span>
<span
>${this.hass.localize(
"ui.panel.config.automation.editor.del"
)}</span
>
</span>`
: nothing}
</div>
</ha-md-menu-item> </ha-md-menu-item>
${keyed( <ha-script-field-editor
this.sidebarKey,
html`<ha-script-field-editor
class="sidebar-editor" class="sidebar-editor"
.hass=${this.hass} .hass=${this.hass}
.field=${this.config.config.field} .field=${this.config.config.field}
@@ -114,9 +83,7 @@ export default class HaAutomationSidebarScriptField extends LitElement {
.disabled=${this.disabled} .disabled=${this.disabled}
.yamlMode=${this.yamlMode} .yamlMode=${this.yamlMode}
@value-changed=${this._valueChangedSidebar} @value-changed=${this._valueChangedSidebar}
@yaml-changed=${this._yamlChangedSidebar} ></ha-script-field-editor>
></ha-script-field-editor>`
)}
</ha-automation-sidebar-card>`; </ha-automation-sidebar-card>`;
} }
@@ -143,12 +110,6 @@ export default class HaAutomationSidebarScriptField extends LitElement {
} }
} }
private _yamlChangedSidebar(ev: CustomEvent) {
ev.stopPropagation();
this.config?.save?.(ev.detail.value);
}
private _toggleYamlMode = () => { private _toggleYamlMode = () => {
fireEvent(this, "toggle-yaml-mode"); fireEvent(this, "toggle-yaml-mode");
}; };

View File

@@ -1,24 +1,21 @@
import { import {
mdiAppleKeyboardCommand,
mdiContentCopy, mdiContentCopy,
mdiContentCut, mdiContentCut,
mdiContentDuplicate,
mdiDelete, mdiDelete,
mdiIdentifier, mdiIdentifier,
mdiPlayCircleOutline, mdiPlayCircleOutline,
mdiPlaylistEdit, mdiPlaylistEdit,
mdiPlusCircleMultipleOutline,
mdiRenameBox, mdiRenameBox,
mdiStopCircleOutline, mdiStopCircleOutline,
} from "@mdi/js"; } from "@mdi/js";
import { html, LitElement, nothing } from "lit"; import { html, LitElement, nothing } from "lit";
import { customElement, property, query, state } from "lit/decorators"; import { customElement, property, query, state } from "lit/decorators";
import { keyed } from "lit/directives/keyed";
import { fireEvent } from "../../../../common/dom/fire_event"; import { fireEvent } from "../../../../common/dom/fire_event";
import { handleStructError } from "../../../../common/structs/handle-errors"; import { handleStructError } from "../../../../common/structs/handle-errors";
import type { TriggerSidebarConfig } from "../../../../data/automation"; import type { TriggerSidebarConfig } from "../../../../data/automation";
import { isTriggerList } from "../../../../data/trigger"; import { isTriggerList } from "../../../../data/trigger";
import type { HomeAssistant } from "../../../../types"; import type { HomeAssistant } from "../../../../types";
import { isMac } from "../../../../util/is_mac";
import { sidebarEditorStyles } from "../styles"; import { sidebarEditorStyles } from "../styles";
import "../trigger/ha-automation-trigger-editor"; import "../trigger/ha-automation-trigger-editor";
import type HaAutomationTriggerEditor from "../trigger/ha-automation-trigger-editor"; import type HaAutomationTriggerEditor from "../trigger/ha-automation-trigger-editor";
@@ -38,8 +35,6 @@ export default class HaAutomationSidebarTrigger extends LitElement {
@property({ type: Boolean }) public narrow = false; @property({ type: Boolean }) public narrow = false;
@property({ attribute: "sidebar-key" }) public sidebarKey?: string;
@state() private _requestShowId = false; @state() private _requestShowId = false;
@state() private _warnings?: string[]; @state() private _warnings?: string[];
@@ -91,13 +86,10 @@ export default class HaAutomationSidebarTrigger extends LitElement {
.clickAction=${this.config.rename} .clickAction=${this.config.rename}
.disabled=${disabled || type === "list"} .disabled=${disabled || type === "list"}
> >
<ha-svg-icon slot="start" .path=${mdiRenameBox}></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize( ${this.hass.localize(
"ui.panel.config.automation.editor.triggers.rename" "ui.panel.config.automation.editor.triggers.rename"
)} )}
<span class="shortcut-placeholder ${isMac ? "mac" : ""}"></span> <ha-svg-icon slot="start" .path=${mdiRenameBox}></ha-svg-icon>
</div>
</ha-md-menu-item> </ha-md-menu-item>
${!this.yamlMode && ${!this.yamlMode &&
!("id" in this.config.config) && !("id" in this.config.config) &&
@@ -107,13 +99,10 @@ export default class HaAutomationSidebarTrigger extends LitElement {
.clickAction=${this._showTriggerId} .clickAction=${this._showTriggerId}
.disabled=${disabled || type === "list"} .disabled=${disabled || type === "list"}
> >
<ha-svg-icon slot="start" .path=${mdiIdentifier}></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize( ${this.hass.localize(
"ui.panel.config.automation.editor.triggers.edit_id" "ui.panel.config.automation.editor.triggers.edit_id"
)} )}
<span class="shortcut-placeholder ${isMac ? "mac" : ""}"></span> <ha-svg-icon slot="start" .path=${mdiIdentifier}></ha-svg-icon>
</div>
</ha-md-menu-item>` </ha-md-menu-item>`
: nothing} : nothing}
@@ -131,10 +120,7 @@ export default class HaAutomationSidebarTrigger extends LitElement {
${this.hass.localize( ${this.hass.localize(
"ui.panel.config.automation.editor.triggers.duplicate" "ui.panel.config.automation.editor.triggers.duplicate"
)} )}
<ha-svg-icon <ha-svg-icon slot="start" .path=${mdiContentDuplicate}></ha-svg-icon>
slot="start"
.path=${mdiPlusCircleMultipleOutline}
></ha-svg-icon>
</ha-md-menu-item> </ha-md-menu-item>
<ha-md-menu-item <ha-md-menu-item
@@ -142,28 +128,10 @@ export default class HaAutomationSidebarTrigger extends LitElement {
.clickAction=${this.config.copy} .clickAction=${this.config.copy}
.disabled=${this.disabled} .disabled=${this.disabled}
> >
<ha-svg-icon slot="start" .path=${mdiContentCopy}></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize( ${this.hass.localize(
"ui.panel.config.automation.editor.triggers.copy" "ui.panel.config.automation.editor.triggers.copy"
)} )}
${!this.narrow <ha-svg-icon slot="start" .path=${mdiContentCopy}></ha-svg-icon>
? html`<span class="shortcut">
<span
>${isMac
? html`<ha-svg-icon
slot="start"
.path=${mdiAppleKeyboardCommand}
></ha-svg-icon>`
: this.hass.localize(
"ui.panel.config.automation.editor.ctrl"
)}</span
>
<span>+</span>
<span>C</span>
</span>`
: nothing}
</div>
</ha-md-menu-item> </ha-md-menu-item>
<ha-md-menu-item <ha-md-menu-item
@@ -171,41 +139,20 @@ export default class HaAutomationSidebarTrigger extends LitElement {
.clickAction=${this.config.cut} .clickAction=${this.config.cut}
.disabled=${this.disabled} .disabled=${this.disabled}
> >
<ha-svg-icon slot="start" .path=${mdiContentCut}></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize( ${this.hass.localize(
"ui.panel.config.automation.editor.triggers.cut" "ui.panel.config.automation.editor.triggers.cut"
)} )}
${!this.narrow <ha-svg-icon slot="start" .path=${mdiContentCut}></ha-svg-icon>
? html`<span class="shortcut">
<span
>${isMac
? html`<ha-svg-icon
slot="start"
.path=${mdiAppleKeyboardCommand}
></ha-svg-icon>`
: this.hass.localize(
"ui.panel.config.automation.editor.ctrl"
)}</span
>
<span>+</span>
<span>X</span>
</span>`
: nothing}
</div>
</ha-md-menu-item> </ha-md-menu-item>
<ha-md-menu-item <ha-md-menu-item
slot="menu-items" slot="menu-items"
.clickAction=${this._toggleYamlMode} .clickAction=${this._toggleYamlMode}
.disabled=${!this.config.uiSupported || !!this._warnings} .disabled=${!this.config.uiSupported || !!this._warnings}
> >
<ha-svg-icon slot="start" .path=${mdiPlaylistEdit}></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize( ${this.hass.localize(
`ui.panel.config.automation.editor.edit_${!this.yamlMode ? "yaml" : "ui"}` `ui.panel.config.automation.editor.edit_${!this.yamlMode ? "yaml" : "ui"}`
)} )}
<span class="shortcut-placeholder ${isMac ? "mac" : ""}"></span> <ha-svg-icon slot="start" .path=${mdiPlaylistEdit}></ha-svg-icon>
</div>
</ha-md-menu-item> </ha-md-menu-item>
<ha-md-divider <ha-md-divider
slot="menu-items" slot="menu-items"
@@ -217,16 +164,13 @@ export default class HaAutomationSidebarTrigger extends LitElement {
.clickAction=${this.config.disable} .clickAction=${this.config.disable}
.disabled=${type === "list"} .disabled=${type === "list"}
> >
${this.hass.localize(
`ui.panel.config.automation.editor.actions.${disabled ? "enable" : "disable"}`
)}
<ha-svg-icon <ha-svg-icon
slot="start" slot="start"
.path=${this.disabled ? mdiPlayCircleOutline : mdiStopCircleOutline} .path=${this.disabled ? mdiPlayCircleOutline : mdiStopCircleOutline}
></ha-svg-icon> ></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize(
`ui.panel.config.automation.editor.actions.${disabled ? "enable" : "disable"}`
)}
<span class="shortcut-placeholder ${isMac ? "mac" : ""}"></span>
</div>
</ha-md-menu-item> </ha-md-menu-item>
<ha-md-menu-item <ha-md-menu-item
slot="menu-items" slot="menu-items"
@@ -234,49 +178,23 @@ export default class HaAutomationSidebarTrigger extends LitElement {
.disabled=${this.disabled} .disabled=${this.disabled}
class="warning" class="warning"
> >
<ha-svg-icon slot="start" .path=${mdiDelete}></ha-svg-icon>
<div class="overflow-label">
${this.hass.localize( ${this.hass.localize(
"ui.panel.config.automation.editor.actions.delete" "ui.panel.config.automation.editor.actions.delete"
)} )}
${!this.narrow <ha-svg-icon slot="start" .path=${mdiDelete}></ha-svg-icon>
? html`<span class="shortcut">
<span
>${isMac
? html`<ha-svg-icon
slot="start"
.path=${mdiAppleKeyboardCommand}
></ha-svg-icon>`
: this.hass.localize(
"ui.panel.config.automation.editor.ctrl"
)}</span
>
<span>+</span>
<span
>${this.hass.localize(
"ui.panel.config.automation.editor.del"
)}</span
>
</span>`
: nothing}
</div>
</ha-md-menu-item> </ha-md-menu-item>
${keyed( <ha-automation-trigger-editor
this.sidebarKey,
html`<ha-automation-trigger-editor
class="sidebar-editor" class="sidebar-editor"
.hass=${this.hass} .hass=${this.hass}
.trigger=${this.config.config} .trigger=${this.config.config}
@value-changed=${this._valueChangedSidebar} @value-changed=${this._valueChangedSidebar}
@yaml-changed=${this._yamlChangedSidebar}
.uiSupported=${this.config.uiSupported} .uiSupported=${this.config.uiSupported}
.showId=${this._requestShowId} .showId=${this._requestShowId}
.yamlMode=${this.yamlMode} .yamlMode=${this.yamlMode}
.disabled=${this.disabled} .disabled=${this.disabled}
@ui-mode-not-available=${this._handleUiModeNotAvailable} @ui-mode-not-available=${this._handleUiModeNotAvailable}
sidebar sidebar
></ha-automation-trigger-editor>` ></ha-automation-trigger-editor>
)}
</ha-automation-sidebar-card> </ha-automation-sidebar-card>
`; `;
} }
@@ -303,12 +221,6 @@ export default class HaAutomationSidebarTrigger extends LitElement {
} }
} }
private _yamlChangedSidebar(ev: CustomEvent) {
ev.stopPropagation();
this.config?.save?.(ev.detail.value);
}
private _toggleYamlMode = () => { private _toggleYamlMode = () => {
fireEvent(this, "toggle-yaml-mode"); fireEvent(this, "toggle-yaml-mode");
}; };

View File

@@ -37,6 +37,12 @@ export const rowStyles = css`
ha-tooltip { ha-tooltip {
cursor: default; cursor: default;
} }
:host([highlight]) ha-card {
--shadow-default: var(--ha-card-box-shadow, 0 0 0 0 transparent);
--shadow-focus: 0 0 0 1px var(--state-inactive-color);
border-color: var(--state-inactive-color);
box-shadow: var(--shadow-default), var(--shadow-focus);
}
.hidden { .hidden {
display: none; display: none;
} }
@@ -136,11 +142,6 @@ export const manualEditorStyles = css`
.content { .content {
padding-top: 24px; padding-top: 24px;
padding-bottom: 72px; padding-bottom: 72px;
transition: padding-bottom 180ms ease-in-out;
}
.content.has-bottom-sheet {
padding-bottom: calc(90vh - 72px);
} }
ha-automation-sidebar { ha-automation-sidebar {
@@ -189,7 +190,8 @@ export const automationRowsStyles = css`
scroll-margin-top: 48px; scroll-margin-top: 48px;
} }
.handle { .handle {
padding: 4px; margin: 4px;
padding: 8px;
cursor: move; /* fallback if grab cursor is unsupported */ cursor: move; /* fallback if grab cursor is unsupported */
cursor: grab; cursor: grab;
border-radius: var(--ha-border-radius-pill); border-radius: var(--ha-border-radius-pill);
@@ -217,39 +219,9 @@ export const automationRowsStyles = css`
export const sidebarEditorStyles = css` export const sidebarEditorStyles = css`
.sidebar-editor { .sidebar-editor {
display: block; display: block;
padding-top: 8px; padding-top: 16px;
} }
.description { .description {
padding-top: 16px; padding-top: 16px;
} }
.overflow-label {
display: flex;
justify-content: space-between;
gap: 12px;
white-space: nowrap;
}
.overflow-label .shortcut {
--mdc-icon-size: 12px;
display: inline-flex;
flex-direction: row;
align-items: center;
gap: 2px;
}
.overflow-label .shortcut span {
font-size: var(--ha-font-size-s);
font-family: var(--ha-font-family-code);
color: var(--ha-color-text-secondary);
}
.shortcut-placeholder {
display: inline-block;
width: 60px;
}
.shortcut-placeholder.mac {
width: 46px;
}
@media all and (max-width: 870px) {
.shortcut-placeholder {
display: none;
}
}
`; `;

View File

@@ -121,7 +121,7 @@ export default class HaAutomationTriggerEditor extends LitElement {
if (!ev.detail.isValid) { if (!ev.detail.isValid) {
return; return;
} }
fireEvent(this, "yaml-changed", { fireEvent(this, "value-changed", {
value: migrateAutomationTrigger(ev.detail.value), value: migrateAutomationTrigger(ev.detail.value),
}); });
} }

View File

@@ -4,12 +4,12 @@ import {
mdiArrowUp, mdiArrowUp,
mdiContentCopy, mdiContentCopy,
mdiContentCut, mdiContentCut,
mdiContentDuplicate,
mdiDelete, mdiDelete,
mdiDotsVertical, mdiDotsVertical,
mdiIdentifier, mdiIdentifier,
mdiPlayCircleOutline, mdiPlayCircleOutline,
mdiPlaylistEdit, mdiPlaylistEdit,
mdiPlusCircleMultipleOutline,
mdiRenameBox, mdiRenameBox,
mdiStopCircleOutline, mdiStopCircleOutline,
} from "@mdi/js"; } from "@mdi/js";
@@ -52,7 +52,6 @@ import {
showPromptDialog, showPromptDialog,
} from "../../../../dialogs/generic/show-dialog-box"; } from "../../../../dialogs/generic/show-dialog-box";
import type { HomeAssistant } from "../../../../types"; import type { HomeAssistant } from "../../../../types";
import { showToast } from "../../../../util/toast";
import "../ha-automation-editor-warning"; import "../ha-automation-editor-warning";
import { rowStyles } from "../styles"; import { rowStyles } from "../styles";
import "./ha-automation-trigger-editor"; import "./ha-automation-trigger-editor";
@@ -114,8 +113,6 @@ export default class HaAutomationTriggerRow extends LitElement {
@property({ type: Boolean }) public last?: boolean; @property({ type: Boolean }) public last?: boolean;
@property({ type: Boolean }) public highlight?: boolean;
@property({ type: Boolean, attribute: "sidebar" }) @property({ type: Boolean, attribute: "sidebar" })
public optionsInSidebar = false; public optionsInSidebar = false;
@@ -154,10 +151,6 @@ export default class HaAutomationTriggerRow extends LitElement {
@consume({ context: fullEntitiesContext, subscribe: true }) @consume({ context: fullEntitiesContext, subscribe: true })
_entityReg!: EntityRegistryEntry[]; _entityReg!: EntityRegistryEntry[];
get selected() {
return this._selected;
}
private _triggerUnsub?: Promise<UnsubscribeFunc>; private _triggerUnsub?: Promise<UnsubscribeFunc>;
private _renderRow() { private _renderRow() {
@@ -226,7 +219,7 @@ export default class HaAutomationTriggerRow extends LitElement {
)} )}
<ha-svg-icon <ha-svg-icon
slot="start" slot="start"
.path=${mdiPlusCircleMultipleOutline} .path=${mdiContentDuplicate}
></ha-svg-icon> ></ha-svg-icon>
</ha-md-menu-item> </ha-md-menu-item>
@@ -354,13 +347,9 @@ export default class HaAutomationTriggerRow extends LitElement {
? html`<ha-automation-row ? html`<ha-automation-row
.disabled=${"enabled" in this.trigger && .disabled=${"enabled" in this.trigger &&
this.trigger.enabled === false} this.trigger.enabled === false}
.selected=${this._selected}
.highlight=${this.highlight}
.sortSelected=${this.sortSelected}
@click=${this._toggleSidebar} @click=${this._toggleSidebar}
@copy-row=${this._copyTrigger} .selected=${this._selected}
@cut-row=${this._cutTrigger} .sortSelected=${this.sortSelected}
@delete-row=${this._onDelete}
>${this._selected >${this._selected
? "selected" ? "selected"
: nothing}${this._renderRow()}</ha-automation-row : nothing}${this._renderRow()}</ha-automation-row
@@ -481,7 +470,8 @@ export default class HaAutomationTriggerRow extends LitElement {
ev?.stopPropagation(); ev?.stopPropagation();
if (this._selected) { if (this._selected) {
fireEvent(this, "request-close-sidebar"); this._selected = false;
fireEvent(this, "close-sidebar");
return; return;
} }
this.openSidebar(); this.openSidebar();
@@ -504,7 +494,7 @@ export default class HaAutomationTriggerRow extends LitElement {
}, },
toggleYamlMode: () => { toggleYamlMode: () => {
this._toggleYamlMode(); this._toggleYamlMode();
this.openSidebar(); return this._yamlMode;
}, },
disable: this._onDisable, disable: this._onDisable,
delete: this._onDelete, delete: this._onDelete,
@@ -518,12 +508,12 @@ export default class HaAutomationTriggerRow extends LitElement {
this._selected = true; this._selected = true;
if (this.narrow) { if (this.narrow) {
window.setTimeout(() => { requestAnimationFrame(() => {
this.scrollIntoView({ this.scrollIntoView({
block: "start", block: "start",
behavior: "smooth", behavior: "smooth",
}); });
}, 180); });
} }
} }
@@ -641,12 +631,6 @@ export default class HaAutomationTriggerRow extends LitElement {
private _copyTrigger = () => { private _copyTrigger = () => {
this._setClipboard(); this._setClipboard();
showToast(this, {
message: this.hass.localize(
"ui.panel.config.automation.editor.triggers.copied_to_clipboard"
),
duration: 2000,
});
}; };
private _cutTrigger = () => { private _cutTrigger = () => {
@@ -655,12 +639,6 @@ export default class HaAutomationTriggerRow extends LitElement {
if (this._selected) { if (this._selected) {
fireEvent(this, "close-sidebar"); fireEvent(this, "close-sidebar");
} }
showToast(this, {
message: this.hass.localize(
"ui.panel.config.automation.editor.triggers.cut_to_clipboard"
),
duration: 2000,
});
}; };
private _moveUp = () => { private _moveUp = () => {
@@ -698,6 +676,10 @@ export default class HaAutomationTriggerRow extends LitElement {
customElements.get(`ha-automation-trigger-${type}`) !== undefined customElements.get(`ha-automation-trigger-${type}`) !== undefined
); );
public isSelected() {
return this._selected;
}
public focus() { public focus() {
this._automationRowElement?.focus(); this._automationRowElement?.focus();
} }

View File

@@ -6,10 +6,12 @@ import { customElement, property, state } from "lit/decorators";
import { repeat } from "lit/directives/repeat"; import { repeat } from "lit/directives/repeat";
import { storage } from "../../../../common/decorators/storage"; import { storage } from "../../../../common/decorators/storage";
import { fireEvent } from "../../../../common/dom/fire_event"; import { fireEvent } from "../../../../common/dom/fire_event";
import { listenMediaQuery } from "../../../../common/dom/media_query";
import { nextRender } from "../../../../common/util/render-status"; import { nextRender } from "../../../../common/util/render-status";
import "../../../../components/ha-button"; import "../../../../components/ha-button";
import "../../../../components/ha-button-menu"; import "../../../../components/ha-button-menu";
import "../../../../components/ha-sortable"; import "../../../../components/ha-sortable";
import type { HaSortableClonedEventData } from "../../../../components/ha-sortable";
import "../../../../components/ha-svg-icon"; import "../../../../components/ha-svg-icon";
import type { import type {
AutomationClipboard, AutomationClipboard,
@@ -43,6 +45,8 @@ export default class HaAutomationTrigger extends LitElement {
@property({ type: Boolean }) public root = false; @property({ type: Boolean }) public root = false;
@state() private _showReorder = false;
@state() private _rowSortSelected?: number; @state() private _rowSortSelected?: number;
@state() @state()
@@ -60,17 +64,33 @@ export default class HaAutomationTrigger extends LitElement {
private _triggerKeys = new WeakMap<Trigger, string>(); private _triggerKeys = new WeakMap<Trigger, string>();
private _unsubMql?: () => void;
public connectedCallback() {
super.connectedCallback();
this._unsubMql = listenMediaQuery("(min-width: 600px)", (matches) => {
this._showReorder = matches;
});
}
public disconnectedCallback() {
super.disconnectedCallback();
this._unsubMql?.();
this._unsubMql = undefined;
}
protected render() { protected render() {
return html` return html`
<ha-sortable <ha-sortable
handle-selector=".handle" handle-selector=".handle"
draggable-selector="ha-automation-trigger-row" draggable-selector="ha-automation-trigger-row"
.disabled=${this.disabled} .disabled=${!this._showReorder || this.disabled}
group="triggers" group="triggers"
invert-swap invert-swap
@item-moved=${this._triggerMoved} @item-moved=${this._triggerMoved}
@item-added=${this._triggerAdded} @item-added=${this._triggerAdded}
@item-removed=${this._triggerRemoved} @item-removed=${this._triggerRemoved}
@item-cloned=${this._triggerCloned}
> >
<div class="rows ${!this.optionsInSidebar ? "no-sidebar" : ""}"> <div class="rows ${!this.optionsInSidebar ? "no-sidebar" : ""}">
${repeat( ${repeat(
@@ -90,12 +110,12 @@ export default class HaAutomationTrigger extends LitElement {
.hass=${this.hass} .hass=${this.hass}
.disabled=${this.disabled} .disabled=${this.disabled}
.narrow=${this.narrow} .narrow=${this.narrow}
.highlight=${this.highlightedTriggers?.includes(trg)} ?highlight=${this.highlightedTriggers?.includes(trg)}
.optionsInSidebar=${this.optionsInSidebar} .optionsInSidebar=${this.optionsInSidebar}
.sortSelected=${this._rowSortSelected === idx} .sortSelected=${this._rowSortSelected === idx}
@stop-sort-selection=${this._stopSortSelection} @stop-sort-selection=${this._stopSortSelection}
> >
${!this.disabled ${this._showReorder && !this.disabled
? html` ? html`
<div <div
tabindex="0" tabindex="0"
@@ -258,8 +278,11 @@ export default class HaAutomationTrigger extends LitElement {
private async _triggerAdded(ev: CustomEvent): Promise<void> { private async _triggerAdded(ev: CustomEvent): Promise<void> {
ev.stopPropagation(); ev.stopPropagation();
const { index, data } = ev.detail; const { index, data } = ev.detail;
const item = ev.detail.item as HaAutomationTriggerRow; let selected = false;
const selected = item.selected; if (data?.["ha-automation-row-selected"]) {
selected = true;
delete data["ha-automation-row-selected"];
}
let triggers = [ let triggers = [
...this.triggers.slice(0, index), ...this.triggers.slice(0, index),
@@ -298,6 +321,12 @@ export default class HaAutomationTrigger extends LitElement {
fireEvent(this, "value-changed", { value: triggers }); fireEvent(this, "value-changed", { value: triggers });
} }
private _triggerCloned(ev: CustomEvent<HaSortableClonedEventData>) {
if (ev.detail.item.isSelected()) {
ev.detail.item.trigger["ha-automation-row-selected"] = true;
}
}
private _triggerChanged(ev: CustomEvent) { private _triggerChanged(ev: CustomEvent) {
ev.stopPropagation(); ev.stopPropagation();
const triggers = [...this.triggers]; const triggers = [...this.triggers];

View File

@@ -42,8 +42,6 @@ export class HaTagTrigger extends LitElement implements TriggerElement {
.disabled=${this.disabled || this._tags.length === 0} .disabled=${this.disabled || this._tags.length === 0}
.value=${this.trigger.tag_id} .value=${this.trigger.tag_id}
@selected=${this._tagChanged} @selected=${this._tagChanged}
fixedMenuPosition
naturalMenuWidth
> >
${this._tags.map( ${this._tags.map(
(tag) => html` (tag) => html`

View File

@@ -773,11 +773,6 @@ export class HaConfigDevicePage extends LitElement {
appearance="plain" appearance="plain"
target=${ifDefined(firstDeviceAction!.target)} target=${ifDefined(firstDeviceAction!.target)}
class=${ifDefined(firstDeviceAction!.classes)} class=${ifDefined(firstDeviceAction!.classes)}
.variant=${firstDeviceAction!.classes?.includes(
"warning"
)
? "danger"
: "brand"}
.action=${firstDeviceAction!.action} .action=${firstDeviceAction!.action}
@click=${this._deviceActionClicked} @click=${this._deviceActionClicked}
> >

View File

@@ -1,5 +1,5 @@
import type { CSSResultGroup, TemplateResult } from "lit"; import type { CSSResultGroup, TemplateResult } from "lit";
import { css, html, LitElement, nothing } from "lit"; import { css, html, LitElement } from "lit";
import { customElement, property, state } from "lit/decorators"; import { customElement, property, state } from "lit/decorators";
import "../../../../../components/ha-card"; import "../../../../../components/ha-card";
import "../../../../../components/ha-code-editor"; import "../../../../../components/ha-code-editor";
@@ -11,22 +11,13 @@ import { showOptionsFlowDialog } from "../../../../../dialogs/config-flow/show-d
import "../../../../../layouts/hass-subpage"; import "../../../../../layouts/hass-subpage";
import { haStyle } from "../../../../../resources/styles"; import { haStyle } from "../../../../../resources/styles";
import type { HomeAssistant } from "../../../../../types"; import type { HomeAssistant } from "../../../../../types";
import { import { subscribeBluetoothConnectionAllocations } from "../../../../../data/bluetooth";
subscribeBluetoothConnectionAllocations,
subscribeBluetoothScannerState,
subscribeBluetoothScannersDetails,
} from "../../../../../data/bluetooth";
import type {
BluetoothAllocationsData,
BluetoothScannerState,
BluetoothScannersDetails,
HaScannerType,
} from "../../../../../data/bluetooth";
import { import {
getValueInPercentage, getValueInPercentage,
roundWithOneDecimal, roundWithOneDecimal,
} from "../../../../../util/calculate"; } from "../../../../../util/calculate";
import "../../../../../components/ha-metric"; import "../../../../../components/ha-metric";
import type { BluetoothAllocationsData } from "../../../../../data/bluetooth";
@customElement("bluetooth-config-dashboard") @customElement("bluetooth-config-dashboard")
export class BluetoothConfigDashboard extends LitElement { export class BluetoothConfigDashboard extends LitElement {
@@ -38,26 +29,16 @@ export class BluetoothConfigDashboard extends LitElement {
@state() private _connectionAllocationsError?: string; @state() private _connectionAllocationsError?: string;
@state() private _scannerState?: BluetoothScannerState;
@state() private _scannerDetails?: BluetoothScannersDetails;
private _configEntry = new URLSearchParams(window.location.search).get( private _configEntry = new URLSearchParams(window.location.search).get(
"config_entry" "config_entry"
); );
private _unsubConnectionAllocations?: (() => Promise<void>) | undefined; private _unsubConnectionAllocations?: (() => Promise<void>) | undefined;
private _unsubScannerState?: (() => Promise<void>) | undefined;
private _unsubScannerDetails?: (() => void) | undefined;
public connectedCallback(): void { public connectedCallback(): void {
super.connectedCallback(); super.connectedCallback();
if (this.hass) { if (this.hass) {
this._subscribeBluetoothConnectionAllocations(); this._subscribeBluetoothConnectionAllocations();
this._subscribeBluetoothScannerState();
this._subscribeScannerDetails();
} }
} }
@@ -80,45 +61,12 @@ export class BluetoothConfigDashboard extends LitElement {
} }
} }
private async _subscribeBluetoothScannerState(): Promise<void> {
if (this._unsubScannerState || !this._configEntry) {
return;
}
this._unsubScannerState = await subscribeBluetoothScannerState(
this.hass.connection,
(scannerState) => {
this._scannerState = scannerState;
},
this._configEntry
);
}
private _subscribeScannerDetails(): void {
if (this._unsubScannerDetails) {
return;
}
this._unsubScannerDetails = subscribeBluetoothScannersDetails(
this.hass.connection,
(details) => {
this._scannerDetails = details;
}
);
}
public disconnectedCallback() { public disconnectedCallback() {
super.disconnectedCallback(); super.disconnectedCallback();
if (this._unsubConnectionAllocations) { if (this._unsubConnectionAllocations) {
this._unsubConnectionAllocations(); this._unsubConnectionAllocations();
this._unsubConnectionAllocations = undefined; this._unsubConnectionAllocations = undefined;
} }
if (this._unsubScannerState) {
this._unsubScannerState();
this._unsubScannerState = undefined;
}
if (this._unsubScannerDetails) {
this._unsubScannerDetails();
this._unsubScannerDetails = undefined;
}
} }
protected render(): TemplateResult { protected render(): TemplateResult {
@@ -130,7 +78,6 @@ export class BluetoothConfigDashboard extends LitElement {
"ui.panel.config.bluetooth.settings_title" "ui.panel.config.bluetooth.settings_title"
)} )}
> >
<div class="card-content">${this._renderScannerState()}</div>
<div class="card-actions"> <div class="card-actions">
<ha-button @click=${this._openOptionFlow} <ha-button @click=${this._openOptionFlow}
>${this.hass.localize( >${this.hass.localize(
@@ -195,118 +142,6 @@ export class BluetoothConfigDashboard extends LitElement {
private _getUsedAllocations = (used: number, total: number) => private _getUsedAllocations = (used: number, total: number) =>
roundWithOneDecimal(getValueInPercentage(used, 0, total)); roundWithOneDecimal(getValueInPercentage(used, 0, total));
private _renderScannerMismatchWarning(
scannerState: BluetoothScannerState,
scannerType: HaScannerType,
formatMode: (mode: string | null) => string
) {
const instructions: string[] = [];
if (scannerType === "remote" || scannerType === "unknown") {
instructions.push(
this.hass.localize(
"ui.panel.config.bluetooth.scanner_mode_mismatch_remote"
)
);
}
if (scannerType === "usb" || scannerType === "unknown") {
instructions.push(
this.hass.localize(
"ui.panel.config.bluetooth.scanner_mode_mismatch_usb"
)
);
}
if (scannerType === "uart" || scannerType === "unknown") {
instructions.push(
this.hass.localize(
"ui.panel.config.bluetooth.scanner_mode_mismatch_uart"
)
);
}
return html`<ha-alert alert-type="warning">
<div>
${this.hass.localize(
"ui.panel.config.bluetooth.scanner_mode_mismatch",
{
requested: formatMode(scannerState.requested_mode),
current: formatMode(scannerState.current_mode),
}
)}
</div>
<ul>
${instructions.map((instruction) => html`<li>${instruction}</li>`)}
</ul>
</ha-alert>`;
}
private _renderScannerState() {
if (!this._configEntry || !this._scannerState) {
return html`<div>
${this.hass.localize(
"ui.panel.config.bluetooth.no_scanner_state_available"
)}
</div>`;
}
const scannerState = this._scannerState;
// Find the scanner details for this source
const scannerDetails = this._scannerDetails?.[scannerState.source];
const scannerType: HaScannerType =
scannerDetails?.scanner_type ?? "unknown";
const formatMode = (mode: string | null) => {
switch (mode) {
case null:
return this.hass.localize(
"ui.panel.config.bluetooth.scanning_mode_none"
);
case "active":
return this.hass.localize(
"ui.panel.config.bluetooth.scanning_mode_active"
);
case "passive":
return this.hass.localize(
"ui.panel.config.bluetooth.scanning_mode_passive"
);
default:
return mode; // Fallback for unknown modes
}
};
return html`
<div class="scanner-state">
<div class="state-row">
<span
>${this.hass.localize(
"ui.panel.config.bluetooth.current_scanning_mode"
)}:</span
>
<span class="state-value"
>${formatMode(scannerState.current_mode)}</span
>
</div>
<div class="state-row">
<span
>${this.hass.localize(
"ui.panel.config.bluetooth.requested_scanning_mode"
)}:</span
>
<span class="state-value"
>${formatMode(scannerState.requested_mode)}</span
>
</div>
${scannerState.current_mode !== scannerState.requested_mode
? this._renderScannerMismatchWarning(
scannerState,
scannerType,
formatMode
)
: nothing}
</div>
`;
}
private _renderConnectionAllocations() { private _renderConnectionAllocations() {
if (this._connectionAllocationsError) { if (this._connectionAllocationsError) {
return html`<ha-alert alert-type="error" return html`<ha-alert alert-type="error"
@@ -385,18 +220,6 @@ export class BluetoothConfigDashboard extends LitElement {
display: flex; display: flex;
justify-content: flex-end; justify-content: flex-end;
} }
.scanner-state {
margin-bottom: 16px;
}
.state-row {
display: flex;
justify-content: space-between;
align-items: center;
padding: 4px 0;
}
.state-value {
font-weight: 500;
}
`, `,
]; ];
} }

View File

@@ -289,15 +289,6 @@ export const showRepairsFlowDialog = (
); );
}, },
renderMenuOptionDescription(hass, step, option) {
return hass.localize(
`component.${issue.domain}.issues.${
issue.translation_key || issue.issue_id
}.fix_flow.step.${step.step_id}.menu_option_descriptions.${option}`,
mergePlaceholders(issue, step)
);
},
renderLoadingDescription(hass, reason) { renderLoadingDescription(hass, reason) {
return ( return (
hass.localize( hass.localize(

View File

@@ -1,6 +1,7 @@
import { consume } from "@lit/context"; import { consume } from "@lit/context";
import { import {
mdiCog, mdiCog,
mdiContentDuplicate,
mdiContentSave, mdiContentSave,
mdiDebugStepOver, mdiDebugStepOver,
mdiDelete, mdiDelete,
@@ -10,11 +11,12 @@ import {
mdiInformationOutline, mdiInformationOutline,
mdiPlay, mdiPlay,
mdiPlaylistEdit, mdiPlaylistEdit,
mdiPlusCircleMultipleOutline,
mdiRenameBox, mdiRenameBox,
mdiRobotConfused, mdiRobotConfused,
mdiTag, mdiTag,
mdiTransitConnection, mdiTransitConnection,
mdiUnfoldLessHorizontal,
mdiUnfoldMoreHorizontal,
} from "@mdi/js"; } from "@mdi/js";
import type { CSSResultGroup, PropertyValues, TemplateResult } from "lit"; import type { CSSResultGroup, PropertyValues, TemplateResult } from "lit";
import { LitElement, css, html, nothing } from "lit"; import { LitElement, css, html, nothing } from "lit";
@@ -306,7 +308,7 @@ export class HaScriptEditor extends SubscribeMixin(
)} )}
<ha-svg-icon <ha-svg-icon
slot="graphic" slot="graphic"
.path=${mdiPlusCircleMultipleOutline} .path=${mdiContentDuplicate}
></ha-svg-icon> ></ha-svg-icon>
</ha-list-item> </ha-list-item>
@@ -340,6 +342,30 @@ export class HaScriptEditor extends SubscribeMixin(
<ha-svg-icon slot="graphic" .path=${mdiPlaylistEdit}></ha-svg-icon> <ha-svg-icon slot="graphic" .path=${mdiPlaylistEdit}></ha-svg-icon>
</ha-list-item> </ha-list-item>
${!useBlueprint
? html`
<ha-list-item graphic="icon" @click=${this._collapseAll}>
<ha-svg-icon
slot="graphic"
.path=${mdiUnfoldLessHorizontal}
></ha-svg-icon>
${this.hass.localize(
"ui.panel.config.automation.editor.collapse_all"
)}
</ha-list-item>
<ha-list-item graphic="icon" @click=${this._expandAll}>
<ha-svg-icon
slot="graphic"
.path=${mdiUnfoldMoreHorizontal}
></ha-svg-icon>
${this.hass.localize(
"ui.panel.config.automation.editor.expand_all"
)}
</ha-list-item>
`
: nothing}
<li divider role="separator"></li> <li divider role="separator"></li>
<ha-list-item <ha-list-item
@@ -1021,9 +1047,6 @@ export class HaScriptEditor extends SubscribeMixin(
protected supportedShortcuts(): SupportedShortcuts { protected supportedShortcuts(): SupportedShortcuts {
return { return {
s: () => this._handleSaveScript(), s: () => this._handleSaveScript(),
c: () => this._copySelectedRow(),
x: () => this._cutSelectedRow(),
Delete: () => this._deleteSelectedRow(),
}; };
} }
@@ -1035,28 +1058,14 @@ export class HaScriptEditor extends SubscribeMixin(
return this._confirmUnsavedChanged(); return this._confirmUnsavedChanged();
} }
// @ts-ignore
private _collapseAll() { private _collapseAll() {
this._manualEditor?.collapseAll(); this._manualEditor?.collapseAll();
} }
// @ts-ignore
private _expandAll() { private _expandAll() {
this._manualEditor?.expandAll(); this._manualEditor?.expandAll();
} }
private _copySelectedRow() {
this._manualEditor?.copySelectedRow();
}
private _cutSelectedRow() {
this._manualEditor?.cutSelectedRow();
}
private _deleteSelectedRow() {
this._manualEditor?.deleteSelectedRow();
}
static get styles(): CSSResultGroup { static get styles(): CSSResultGroup {
return [ return [
haStyle, haStyle,

View File

@@ -152,12 +152,7 @@ export default class HaScriptFieldEditor extends LitElement {
ev.stopPropagation(); ev.stopPropagation();
const value = { ...ev.detail.value }; const value = { ...ev.detail.value };
if ( if (typeof value !== "object" || Object.keys(value).length !== 1) {
typeof value !== "object" ||
Object.keys(value).length !== 1 ||
!value[Object.keys(value)[0]] ||
!value[Object.keys(value)[0]].selector
) {
this._yamlError = "yaml_error"; this._yamlError = "yaml_error";
return; return;
} }
@@ -170,7 +165,7 @@ export default class HaScriptFieldEditor extends LitElement {
const newValue = { ...value[key], key }; const newValue = { ...value[key], key };
fireEvent(this, "yaml-changed", { value: newValue }); fireEvent(this, "value-changed", { value: newValue });
} }
private _computeLabelCallback = ( private _computeLabelCallback = (

View File

@@ -32,8 +32,6 @@ export default class HaScriptFieldRow extends LitElement {
@property({ type: Boolean }) public narrow = false; @property({ type: Boolean }) public narrow = false;
@property({ type: Boolean }) public highlight?: boolean;
@state() private _yamlMode = false; @state() private _yamlMode = false;
@state() private _selected = false; @state() private _selected = false;
@@ -63,8 +61,6 @@ export default class HaScriptFieldRow extends LitElement {
left-chevron left-chevron
@toggle-collapsed=${this._toggleCollapse} @toggle-collapsed=${this._toggleCollapse}
.collapsed=${this._collapsed} .collapsed=${this._collapsed}
.highlight=${this.highlight}
@delete-row=${this._onDelete}
> >
<h3 slot="header">${this.key}</h3> <h3 slot="header">${this.key}</h3>
@@ -87,7 +83,6 @@ export default class HaScriptFieldRow extends LitElement {
.leftChevron=${SELECTOR_SELECTOR_BUILDING_BLOCKS.includes( .leftChevron=${SELECTOR_SELECTOR_BUILDING_BLOCKS.includes(
Object.keys(this.field.selector)[0] Object.keys(this.field.selector)[0]
)} )}
.highlight=${this.highlight}
> >
<h3 slot="header"> <h3 slot="header">
${this.hass.localize( ${this.hass.localize(
@@ -162,7 +157,8 @@ export default class HaScriptFieldRow extends LitElement {
ev?.stopPropagation(); ev?.stopPropagation();
if (this._selected) { if (this._selected) {
fireEvent(this, "request-close-sidebar"); this._selected = false;
fireEvent(this, "close-sidebar");
return; return;
} }
@@ -175,7 +171,8 @@ export default class HaScriptFieldRow extends LitElement {
ev?.stopPropagation(); ev?.stopPropagation();
if (this._selectorRowSelected) { if (this._selectorRowSelected) {
fireEvent(this, "request-close-sidebar"); this._selectorRowSelected = false;
fireEvent(this, "close-sidebar");
return; return;
} }
@@ -221,7 +218,7 @@ export default class HaScriptFieldRow extends LitElement {
}, },
toggleYamlMode: () => { toggleYamlMode: () => {
this._toggleYamlMode(); this._toggleYamlMode();
this.openSidebar(); return this._yamlMode;
}, },
delete: this._onDelete, delete: this._onDelete,
config: { config: {
@@ -234,12 +231,12 @@ export default class HaScriptFieldRow extends LitElement {
} satisfies ScriptFieldSidebarConfig); } satisfies ScriptFieldSidebarConfig);
if (this.narrow) { if (this.narrow) {
window.setTimeout(() => { requestAnimationFrame(() => {
this.scrollIntoView({ this.scrollIntoView({
block: "start", block: "start",
behavior: "smooth", behavior: "smooth",
}); });
}, 180); // duration of transition of added padding for bottom sheet });
} }
} }
@@ -334,6 +331,13 @@ export default class HaScriptFieldRow extends LitElement {
li[role="separator"] { li[role="separator"] {
border-bottom-color: var(--divider-color); border-bottom-color: var(--divider-color);
} }
:host([highlight]) ha-card {
--shadow-default: var(--ha-card-box-shadow, 0 0 0 0 transparent);
--shadow-focus: 0 0 0 1px var(--state-inactive-color);
border-color: var(--state-inactive-color);
box-shadow: var(--shadow-default), var(--shadow-focus);
}
.selector-row { .selector-row {
padding: 12px 0 16px 16px; padding: 12px 0 16px 16px;
} }

View File

@@ -132,7 +132,7 @@ export default class HaScriptFieldSelectorEditor extends LitElement {
return; return;
} }
fireEvent(this, "yaml-changed", { value }); fireEvent(this, "value-changed", { value });
} }
private _computeLabelCallback = ( private _computeLabelCallback = (

View File

@@ -43,7 +43,7 @@ export default class HaScriptFields extends LitElement {
.disabled=${this.disabled} .disabled=${this.disabled}
@value-changed=${this._fieldChanged} @value-changed=${this._fieldChanged}
.hass=${this.hass} .hass=${this.hass}
.highlight=${this.highlightedFields?.[key] !== undefined} ?highlight=${this.highlightedFields?.[key] !== undefined}
.narrow=${this.narrow} .narrow=${this.narrow}
> >
</ha-script-field-row> </ha-script-field-row>
@@ -76,12 +76,10 @@ export default class HaScriptFields extends LitElement {
row.focus(); row.focus();
if (this.narrow) { if (this.narrow) {
window.setTimeout(() => {
row.scrollIntoView({ row.scrollIntoView({
block: "start", block: "start",
behavior: "smooth", behavior: "smooth",
}); });
}, 180); // duration of transition of added padding for bottom sheet
} }
}); });
} }

View File

@@ -2,13 +2,7 @@ import { mdiContentSave, mdiHelpCircle } from "@mdi/js";
import { load } from "js-yaml"; import { load } from "js-yaml";
import type { CSSResultGroup, PropertyValues } from "lit"; import type { CSSResultGroup, PropertyValues } from "lit";
import { css, html, LitElement, nothing } from "lit"; import { css, html, LitElement, nothing } from "lit";
import { import { customElement, property, query, state } from "lit/decorators";
customElement,
property,
query,
queryAll,
state,
} from "lit/decorators";
import { classMap } from "lit/directives/class-map"; import { classMap } from "lit/directives/class-map";
import { import {
any, any,
@@ -30,10 +24,7 @@ import {
} from "../../../common/url/search-params"; } from "../../../common/url/search-params";
import "../../../components/ha-icon-button"; import "../../../components/ha-icon-button";
import "../../../components/ha-markdown"; import "../../../components/ha-markdown";
import type { import type { SidebarConfig } from "../../../data/automation";
ActionSidebarConfig,
SidebarConfig,
} from "../../../data/automation";
import type { Action, Fields, ScriptConfig } from "../../../data/script"; import type { Action, Fields, ScriptConfig } from "../../../data/script";
import { import {
getActionType, getActionType,
@@ -82,18 +73,11 @@ export class HaManualScriptEditor extends LitElement {
@state() private _sidebarConfig?: SidebarConfig; @state() private _sidebarConfig?: SidebarConfig;
@state() private _sidebarKey?: string;
@query("ha-script-fields") @query("ha-script-fields")
private _scriptFields?: HaScriptFields; private _scriptFields?: HaScriptFields;
@query("ha-automation-sidebar") private _sidebarElement?: HaAutomationSidebar; @query("ha-automation-sidebar") private _sidebarElement?: HaAutomationSidebar;
@queryAll("ha-automation-action, ha-script-fields")
private _collapsableElements?: NodeListOf<
HaAutomationAction | HaScriptFields
>;
private _previousConfig?: ScriptConfig; private _previousConfig?: ScriptConfig;
private _openFields = false; private _openFields = false;
@@ -170,7 +154,6 @@ export class HaManualScriptEditor extends LitElement {
.disabled=${this.disabled} .disabled=${this.disabled}
.narrow=${this.narrow} .narrow=${this.narrow}
@open-sidebar=${this._openSidebar} @open-sidebar=${this._openSidebar}
@request-close-sidebar=${this._triggerCloseSidebar}
@close-sidebar=${this._handleCloseSidebar} @close-sidebar=${this._handleCloseSidebar}
></ha-script-fields>` ></ha-script-fields>`
: nothing : nothing
@@ -201,7 +184,6 @@ export class HaManualScriptEditor extends LitElement {
.highlightedActions=${this._pastedConfig?.sequence || []} .highlightedActions=${this._pastedConfig?.sequence || []}
@value-changed=${this._sequenceChanged} @value-changed=${this._sequenceChanged}
@open-sidebar=${this._openSidebar} @open-sidebar=${this._openSidebar}
@request-close-sidebar=${this._triggerCloseSidebar}
@close-sidebar=${this._handleCloseSidebar} @close-sidebar=${this._handleCloseSidebar}
.hass=${this.hass} .hass=${this.hass}
.narrow=${this.narrow} .narrow=${this.narrow}
@@ -220,11 +202,7 @@ export class HaManualScriptEditor extends LitElement {
})} })}
> >
<div class="content-wrapper"> <div class="content-wrapper">
<div <div class="content">
class="content ${this._sidebarConfig && this.narrow
? "has-bottom-sheet"
: ""}"
>
<slot name="alerts"></slot> <slot name="alerts"></slot>
${this._renderContent()} ${this._renderContent()}
</div> </div>
@@ -245,7 +223,6 @@ export class HaManualScriptEditor extends LitElement {
</div> </div>
<div class="sidebar-positioner"> <div class="sidebar-positioner">
<ha-automation-sidebar <ha-automation-sidebar
.sidebarKey=${this._sidebarKey}
tabindex="-1" tabindex="-1"
class=${classMap({ hidden: !this._sidebarConfig })} class=${classMap({ hidden: !this._sidebarConfig })}
.narrow=${this.narrow} .narrow=${this.narrow}
@@ -377,11 +354,7 @@ export class HaManualScriptEditor extends LitElement {
if (normalized) { if (normalized) {
ev.preventDefault(); ev.preventDefault();
if ( if (this.dirty) {
this.dirty ||
ensureArray(this.config.sequence)?.length ||
Object.keys(this.config.fields || {}).length
) {
const result = await new Promise<boolean>((resolve) => { const result = await new Promise<boolean>((resolve) => {
showPasteReplaceDialog(this, { showPasteReplaceDialog(this, {
domain: "script", domain: "script",
@@ -490,7 +463,6 @@ export class HaManualScriptEditor extends LitElement {
// deselect previous selected row // deselect previous selected row
this._sidebarConfig?.close?.(); this._sidebarConfig?.close?.();
this._sidebarConfig = ev.detail; this._sidebarConfig = ev.detail;
this._sidebarKey = JSON.stringify(this._sidebarConfig);
await this._sidebarElement?.updateComplete; await this._sidebarElement?.updateComplete;
this._sidebarElement?.focus(); this._sidebarElement?.focus();
@@ -508,13 +480,11 @@ export class HaManualScriptEditor extends LitElement {
}; };
} }
private _triggerCloseSidebar() { private _closeSidebar() {
if (this._sidebarConfig) { if (this._sidebarConfig) {
if (this._sidebarElement) { const closeRow = this._sidebarConfig?.close;
this._sidebarElement.triggerCloseSidebar(); this._sidebarConfig = undefined;
return; closeRow?.();
}
this._sidebarConfig?.close();
} }
} }
@@ -523,40 +493,28 @@ export class HaManualScriptEditor extends LitElement {
} }
private _saveScript() { private _saveScript() {
this._triggerCloseSidebar(); this._closeSidebar();
fireEvent(this, "save-script"); fireEvent(this, "save-script");
} }
private _getCollapsableElements() {
return this.shadowRoot!.querySelectorAll<
HaAutomationAction | HaScriptFields
>("ha-automation-action, ha-script-fields");
}
public expandAll() { public expandAll() {
this._collapsableElements?.forEach((element) => { this._getCollapsableElements().forEach((element) => {
element.expandAll(); element.expandAll();
}); });
} }
public collapseAll() { public collapseAll() {
this._collapsableElements?.forEach((element) => { this._getCollapsableElements().forEach((element) => {
element.collapseAll(); element.collapseAll();
}); });
} }
public copySelectedRow() {
if ((this._sidebarConfig as ActionSidebarConfig)?.copy) {
(this._sidebarConfig as ActionSidebarConfig).copy();
}
}
public cutSelectedRow() {
if ((this._sidebarConfig as ActionSidebarConfig)?.cut) {
(this._sidebarConfig as ActionSidebarConfig).cut();
}
}
public deleteSelectedRow() {
if ((this._sidebarConfig as ActionSidebarConfig)?.delete) {
(this._sidebarConfig as ActionSidebarConfig).delete();
}
}
static get styles(): CSSResultGroup { static get styles(): CSSResultGroup {
return [ return [
saveFabStyles, saveFabStyles,

View File

@@ -4,8 +4,6 @@ import { customElement, property, state } from "lit/decorators";
import { computeDomain } from "../../../common/entity/compute_domain"; import { computeDomain } from "../../../common/entity/compute_domain";
import "../../../components/ha-control-button"; import "../../../components/ha-control-button";
import "../../../components/ha-control-button-group"; import "../../../components/ha-control-button-group";
import { hasScriptFields } from "../../../data/script";
import { showMoreInfoDialog } from "../../../dialogs/more-info/show-ha-more-info-dialog";
import type { HomeAssistant } from "../../../types"; import type { HomeAssistant } from "../../../types";
import type { LovelaceCardFeature, LovelaceCardFeatureEditor } from "../types"; import type { LovelaceCardFeature, LovelaceCardFeatureEditor } from "../types";
import { cardFeatureStyles } from "./common/card-feature-styles"; import { cardFeatureStyles } from "./common/card-feature-styles";
@@ -48,14 +46,6 @@ class HuiButtonCardFeature extends LitElement implements LovelaceCardFeature {
const service = const service =
domain === "button" || domain === "input_button" ? "press" : "turn_on"; domain === "button" || domain === "input_button" ? "press" : "turn_on";
if (domain === "script") {
const entityId = this._stateObj.entity_id;
if (hasScriptFields(this.hass!, entityId)) {
showMoreInfoDialog(this, { entityId: entityId });
return;
}
}
this.hass.callService(domain, service, { this.hass.callService(domain, service, {
entity_id: this._stateObj.entity_id, entity_id: this._stateObj.entity_id,
}); });

View File

@@ -0,0 +1,301 @@
import { css, html, LitElement, nothing, svg } from "lit";
import { customElement, property, state } from "lit/decorators";
import { computeDomain } from "../../../common/entity/compute_domain";
import {
computeHistory,
subscribeHistoryStatesTimeWindow,
} from "../../../data/history";
import type {
HistoryResult,
LineChartUnit,
TimelineEntity,
} from "../../../data/history";
import type { HomeAssistant } from "../../../types";
import type { LovelaceCardFeature } from "../types";
import type {
LovelaceCardFeatureContext,
HistoryChartCardFeatureConfig,
} from "./types";
import { getSensorNumericDeviceClasses } from "../../../data/sensor";
import { isComponentLoaded } from "../../../common/config/is_component_loaded";
import { SubscribeMixin } from "../../../mixins/subscribe-mixin";
import { getGraphColorByIndex } from "../../../common/color/colors";
import { computeTimelineColor } from "../../../components/chart/timeline-color";
import { downSampleLineData } from "../../../components/chart/down-sample";
import { fireEvent } from "../../../common/dom/fire_event";
export const supportsHistoryChartCardFeature = (
_hass: HomeAssistant,
context: LovelaceCardFeatureContext
) =>
!!context.entity_id &&
["sensor", "binary_sensor"].includes(computeDomain(context.entity_id));
@customElement("hui-history-chart-card-feature")
class HuiHistoryChartCardFeature
extends SubscribeMixin(LitElement)
implements LovelaceCardFeature
{
@property({ attribute: false, hasChanged: () => false })
public hass?: HomeAssistant;
@property({ attribute: false }) public context?: LovelaceCardFeatureContext;
@state() private _config?: HistoryChartCardFeatureConfig;
@state() private _stateHistory?: HistoryResult;
private _interval?: number;
static getStubConfig(): HistoryChartCardFeatureConfig {
return {
type: "history-chart",
hours_to_show: 24,
};
}
public setConfig(config: HistoryChartCardFeatureConfig): void {
if (!config) {
throw new Error("Invalid configuration");
}
this._config = config;
}
public connectedCallback() {
super.connectedCallback();
// redraw the graph every minute to update the time axis
clearInterval(this._interval);
this._interval = window.setInterval(() => this.requestUpdate(), 1000 * 60);
}
public disconnectedCallback() {
super.disconnectedCallback();
clearInterval(this._interval);
}
protected hassSubscribe() {
return [this._subscribeHistory()];
}
protected render() {
if (
!this._config ||
!this.hass ||
!this.context ||
!this._stateHistory ||
!supportsHistoryChartCardFeature(this.hass, this.context)
) {
return nothing;
}
const line = this._stateHistory.line[0];
const timeline = this._stateHistory.timeline[0];
const width = this.clientWidth;
const height = this.clientHeight;
if (line) {
const points = this._generateLinePoints(line);
const { paths, filledPaths } = this._getLinePaths(points);
const color = getGraphColorByIndex(0, this.style);
return html`
<div class="line" @click=${this._handleClick}>
${svg`<svg width="${width}" height="${height}" viewBox="0 0 ${width} ${height}">
${paths.map(
(path) =>
svg`<path d="${path}" stroke="${color}" stroke-width="1" stroke-linecap="round" fill="none" />`
)}
${filledPaths.map(
(path) =>
svg`<path d="${path}" stroke="none" stroke-linecap="round" fill="${color}" fill-opacity="0.2" />`
)}
</svg>`}
</div>
`;
}
if (timeline) {
const ranges = this._generateTimelineRanges(timeline);
return html`
<div class="timeline" @click=${this._handleClick}>
${svg`<svg width="${width}" height="${height}" viewBox="0 0 ${width} ${height}">
<g>
${ranges.map((r) => svg`<rect x="${r.startX}" y="0" width="${r.endX - r.startX}" height="${height}" fill="${r.color}" />`)}
</g>
</svg>`}
</div>
`;
}
return nothing;
}
private _handleClick() {
// open more info dialog to show more detailed history
fireEvent(this, "hass-more-info", { entityId: this.context!.entity_id! });
}
private async _subscribeHistory(): Promise<() => Promise<void>> {
if (
!isComponentLoaded(this.hass!, "history") ||
!this.context?.entity_id ||
!this._config
) {
return () => Promise.resolve();
}
const { numeric_device_classes: sensorNumericDeviceClasses } =
await getSensorNumericDeviceClasses(this.hass!);
return subscribeHistoryStatesTimeWindow(
this.hass!,
(historyStates) => {
this._stateHistory = computeHistory(
this.hass!,
historyStates,
[this.context!.entity_id!],
this.hass!.localize,
sensorNumericDeviceClasses,
false
);
},
this._config!.hours_to_show ?? 24,
[this.context!.entity_id!]
);
}
private _generateLinePoints(line: LineChartUnit): { x: number; y: number }[] {
const width = this.clientWidth;
const height = this.clientHeight;
let minY = Number(line.data[0].states[0].state);
let maxY = Number(line.data[0].states[0].state);
const minX = line.data[0].states[0].last_changed;
const maxX = Date.now();
line.data[0].states.forEach((stateData) => {
const stateValue = Number(stateData.state);
if (stateValue < minY) {
minY = stateValue;
}
if (stateValue > maxY) {
maxY = stateValue;
}
});
const rangeY = maxY - minY || minY * 0.1;
const sampledData = downSampleLineData(
line.data[0].states.map((stateData) => [
stateData.last_changed,
Number(stateData.state),
]),
width,
minX,
maxX
);
// add margin to the min and max
minY -= rangeY * 0.1;
maxY += rangeY * 0.1;
const yDenom = maxY - minY || 1;
const xDenom = maxX - minX || 1;
const points = sampledData!.map((point) => {
const x = ((point![0] - minX) / xDenom) * width;
const y = height - ((Number(point![1]) - minY) / yDenom) * height;
return { x, y };
});
points.push({ x: width, y: points[points.length - 1].y });
return points;
}
private _generateTimelineRanges(timeline: TimelineEntity) {
if (timeline.data.length === 0) {
return [];
}
const computedStyles = getComputedStyle(this);
const width = this.clientWidth;
const minX = timeline.data[0].last_changed;
const maxX = Date.now();
let prevEndX = 0;
let prevStateColor = "";
const ranges = timeline.data.map((t) => {
const x = ((t.last_changed - minX) / (maxX - minX)) * width;
const range = {
startX: prevEndX,
endX: x,
color: prevStateColor,
};
prevStateColor = computeTimelineColor(
t.state,
computedStyles,
this.hass!.states[timeline.entity_id]
);
prevEndX = x;
return range;
});
ranges.push({
startX: prevEndX,
endX: width,
color: prevStateColor,
});
return ranges;
}
private _getLinePaths(points: { x: number; y: number }[]) {
const paths: string[] = [];
const filledPaths: string[] = [];
if (!points.length) {
return { paths, filledPaths };
}
// path can interupted by missing data, so we need to split the path into segments
const pathSegments: { x: number; y: number }[][] = [[]];
points.forEach((point) => {
if (!isNaN(point.y)) {
pathSegments[pathSegments.length - 1].push(point);
} else if (pathSegments[pathSegments.length - 1].length > 0) {
pathSegments.push([]);
}
});
pathSegments.forEach((pathPoints) => {
// create a smoothed path
let next: { x: number; y: number };
let path = "";
let last = pathPoints[0];
path += `M ${last.x},${last.y}`;
pathPoints.forEach((coord) => {
next = coord;
path += ` ${(next.x + last.x) / 2},${(next.y + last.y) / 2}`;
path += ` Q${next.x},${next.y}`;
last = next;
});
path += ` ${next!.x},${next!.y}`;
paths.push(path);
filledPaths.push(
path +
` L ${next!.x},${this.clientHeight} L ${pathPoints[0].x},${this.clientHeight} Z`
);
});
return { paths, filledPaths };
}
static styles = css`
:host {
display: block;
width: 100%;
height: var(--feature-height);
}
:host > div {
width: 100%;
height: 100%;
cursor: pointer;
}
.timeline {
border-radius: 4px;
overflow: hidden;
}
`;
}
declare global {
interface HTMLElementTagNameMap {
"hui-history-chart-card-feature": HuiHistoryChartCardFeature;
}
}

View File

@@ -1,164 +0,0 @@
import { css, html, LitElement, nothing } from "lit";
import { customElement, property, state } from "lit/decorators";
import { isComponentLoaded } from "../../../common/config/is_component_loaded";
import { computeDomain } from "../../../common/entity/compute_domain";
import { isNumericFromAttributes } from "../../../common/number/format_number";
import "../../../components/ha-spinner";
import { subscribeHistoryStatesTimeWindow } from "../../../data/history";
import { SubscribeMixin } from "../../../mixins/subscribe-mixin";
import type { HomeAssistant } from "../../../types";
import { coordinatesMinimalResponseCompressedState } from "../common/graph/coordinates";
import "../components/hui-graph-base";
import type { LovelaceCardFeature, LovelaceCardFeatureEditor } from "../types";
import type {
TrendGraphCardFeatureConfig,
LovelaceCardFeatureContext,
} from "./types";
export const supportsTrendGraphCardFeature = (
hass: HomeAssistant,
context: LovelaceCardFeatureContext
) => {
const stateObj = context.entity_id
? hass.states[context.entity_id]
: undefined;
if (!stateObj) return false;
const domain = computeDomain(stateObj.entity_id);
return domain === "sensor" && isNumericFromAttributes(stateObj.attributes);
};
export const DEFAULT_HOURS_TO_SHOW = 24;
@customElement("hui-trend-graph-card-feature")
class HuiHistoryChartCardFeature
extends SubscribeMixin(LitElement)
implements LovelaceCardFeature
{
@property({ attribute: false, hasChanged: () => false })
public hass?: HomeAssistant;
@property({ attribute: false }) public context?: LovelaceCardFeatureContext;
@state() private _config?: TrendGraphCardFeatureConfig;
@state() private _coordinates?: [number, number][];
private _interval?: number;
static getStubConfig(): TrendGraphCardFeatureConfig {
return {
type: "trend-graph",
};
}
public static async getConfigElement(): Promise<LovelaceCardFeatureEditor> {
await import(
"../editor/config-elements/hui-trend-graph-card-feature-editor"
);
return document.createElement("hui-trend-graph-card-feature-editor");
}
public setConfig(config: TrendGraphCardFeatureConfig): void {
if (!config) {
throw new Error("Invalid configuration");
}
this._config = config;
}
public connectedCallback() {
super.connectedCallback();
// redraw the graph every minute to update the time axis
clearInterval(this._interval);
this._interval = window.setInterval(() => this.requestUpdate(), 1000 * 60);
}
public disconnectedCallback() {
super.disconnectedCallback();
clearInterval(this._interval);
}
protected hassSubscribe() {
return [this._subscribeHistory()];
}
protected render() {
if (
!this._config ||
!this.hass ||
!this.context ||
!supportsTrendGraphCardFeature(this.hass, this.context)
) {
return nothing;
}
if (!this._coordinates) {
return html`
<div class="container">
<ha-spinner size="small"></ha-spinner>
</div>
`;
}
if (!this._coordinates.length) {
return html`
<div class="container">
<div class="info">No state history found.</div>
</div>
`;
}
return html`
<hui-graph-base .coordinates=${this._coordinates}></hui-graph-base>
`;
}
private async _subscribeHistory(): Promise<() => Promise<void>> {
if (
!isComponentLoaded(this.hass!, "history") ||
!this.context?.entity_id ||
!this._config
) {
return () => Promise.resolve();
}
const hourToShow = this._config.hours_to_show ?? DEFAULT_HOURS_TO_SHOW;
return subscribeHistoryStatesTimeWindow(
this.hass!,
(historyStates) => {
this._coordinates =
coordinatesMinimalResponseCompressedState(
historyStates[this.context!.entity_id!],
hourToShow,
500,
2,
undefined
) || [];
},
hourToShow,
[this.context!.entity_id!]
);
}
static styles = css`
:host {
display: flex;
width: 100%;
height: var(--feature-height);
flex-direction: column;
justify-content: flex-end;
align-items: flex-end;
pointer-events: none !important;
}
hui-graph-base {
width: 100%;
--accent-color: var(--feature-color);
border-bottom-right-radius: 8px;
border-bottom-left-radius: 8px;
overflow: hidden;
}
`;
}
declare global {
interface HTMLElementTagNameMap {
"hui-trend-graph-card-feature": HuiHistoryChartCardFeature;
}
}

View File

@@ -187,9 +187,9 @@ export interface UpdateActionsCardFeatureConfig {
backup?: "yes" | "no" | "ask"; backup?: "yes" | "no" | "ask";
} }
export interface TrendGraphCardFeatureConfig { export interface HistoryChartCardFeatureConfig {
type: "trend-graph"; type: "history-chart";
hours_to_show?: number; hours_to_show: number;
} }
export const AREA_CONTROLS = [ export const AREA_CONTROLS = [
@@ -239,7 +239,7 @@ export type LovelaceCardFeatureConfig =
| FanOscillateCardFeatureConfig | FanOscillateCardFeatureConfig
| FanPresetModesCardFeatureConfig | FanPresetModesCardFeatureConfig
| FanSpeedCardFeatureConfig | FanSpeedCardFeatureConfig
| TrendGraphCardFeatureConfig | HistoryChartCardFeatureConfig
| HumidifierToggleCardFeatureConfig | HumidifierToggleCardFeatureConfig
| HumidifierModesCardFeatureConfig | HumidifierModesCardFeatureConfig
| LawnMowerCommandsCardFeatureConfig | LawnMowerCommandsCardFeatureConfig
@@ -251,7 +251,7 @@ export type LovelaceCardFeatureConfig =
| MediaPlayerVolumeSliderCardFeatureConfig | MediaPlayerVolumeSliderCardFeatureConfig
| NumericInputCardFeatureConfig | NumericInputCardFeatureConfig
| SelectOptionsCardFeatureConfig | SelectOptionsCardFeatureConfig
| TrendGraphCardFeatureConfig | HistoryChartCardFeatureConfig
| TargetHumidityCardFeatureConfig | TargetHumidityCardFeatureConfig
| TargetTemperatureCardFeatureConfig | TargetTemperatureCardFeatureConfig
| ToggleCardFeatureConfig | ToggleCardFeatureConfig

View File

@@ -109,7 +109,7 @@ class HuiEnergySankeyCard
"ui.panel.lovelace.cards.energy.energy_distribution.home" "ui.panel.lovelace.cards.energy.energy_distribution.home"
), ),
value: Math.max(0, consumption.total.used_total), value: Math.max(0, consumption.total.used_total),
color: computedStyle.getPropertyValue("--primary-color").trim(), color: computedStyle.getPropertyValue("--primary-color"),
index: 1, index: 1,
}; };
nodes.push(homeNode); nodes.push(homeNode);
@@ -125,9 +125,7 @@ class HuiEnergySankeyCard
"ui.panel.lovelace.cards.energy.energy_distribution.battery" "ui.panel.lovelace.cards.energy.energy_distribution.battery"
), ),
value: totalBatteryOut, value: totalBatteryOut,
color: computedStyle color: computedStyle.getPropertyValue("--energy-battery-out-color"),
.getPropertyValue("--energy-battery-out-color")
.trim(),
index: 0, index: 0,
}); });
links.push({ links.push({
@@ -143,9 +141,7 @@ class HuiEnergySankeyCard
"ui.panel.lovelace.cards.energy.energy_distribution.battery" "ui.panel.lovelace.cards.energy.energy_distribution.battery"
), ),
value: totalBatteryIn, value: totalBatteryIn,
color: computedStyle color: computedStyle.getPropertyValue("--energy-battery-in-color"),
.getPropertyValue("--energy-battery-in-color")
.trim(),
index: 1, index: 1,
}); });
if (consumption.total.grid_to_battery > 0) { if (consumption.total.grid_to_battery > 0) {
@@ -173,9 +169,9 @@ class HuiEnergySankeyCard
"ui.panel.lovelace.cards.energy.energy_distribution.grid" "ui.panel.lovelace.cards.energy.energy_distribution.grid"
), ),
value: totalFromGrid, value: totalFromGrid,
color: computedStyle color: computedStyle.getPropertyValue(
.getPropertyValue("--energy-grid-consumption-color") "--energy-grid-consumption-color"
.trim(), ),
index: 0, index: 0,
}); });
@@ -196,7 +192,7 @@ class HuiEnergySankeyCard
"ui.panel.lovelace.cards.energy.energy_distribution.solar" "ui.panel.lovelace.cards.energy.energy_distribution.solar"
), ),
value: totalSolarProduction, value: totalSolarProduction,
color: computedStyle.getPropertyValue("--energy-solar-color").trim(), color: computedStyle.getPropertyValue("--energy-solar-color"),
index: 0, index: 0,
}); });
@@ -217,9 +213,7 @@ class HuiEnergySankeyCard
"ui.panel.lovelace.cards.energy.energy_distribution.grid" "ui.panel.lovelace.cards.energy.energy_distribution.grid"
), ),
value: totalToGrid, value: totalToGrid,
color: computedStyle color: computedStyle.getPropertyValue("--energy-grid-return-color"),
.getPropertyValue("--energy-grid-return-color")
.trim(),
index: 1, index: 1,
}); });
if (consumption.total.battery_to_grid > 0) { if (consumption.total.battery_to_grid > 0) {
@@ -301,7 +295,7 @@ class HuiEnergySankeyCard
label: this.hass.floors[floorId].name, label: this.hass.floors[floorId].name,
value: floors[floorId].value, value: floors[floorId].value,
index: 2, index: 2,
color: computedStyle.getPropertyValue("--primary-color").trim(), color: computedStyle.getPropertyValue("--primary-color"),
}); });
links.push({ links.push({
source: "home", source: "home",
@@ -322,7 +316,7 @@ class HuiEnergySankeyCard
label: this.hass.areas[areaId]!.name, label: this.hass.areas[areaId]!.name,
value: areas[areaId].value, value: areas[areaId].value,
index: 3, index: 3,
color: computedStyle.getPropertyValue("--primary-color").trim(), color: computedStyle.getPropertyValue("--primary-color"),
}); });
links.push({ links.push({
source: floorNodeId, source: floorNodeId,
@@ -366,9 +360,7 @@ class HuiEnergySankeyCard
"ui.panel.lovelace.cards.energy.energy_devices_detail_graph.untracked_consumption" "ui.panel.lovelace.cards.energy.energy_devices_detail_graph.untracked_consumption"
), ),
value: untrackedConsumption, value: untrackedConsumption,
color: computedStyle color: computedStyle.getPropertyValue("--state-unavailable-color"),
.getPropertyValue("--state-unavailable-color")
.trim(),
index: 3 + deviceSections.length, index: 3 + deviceSections.length,
}); });
links.push({ links.push({

View File

@@ -138,7 +138,6 @@ export class HuiHeadingCard extends LitElement implements LovelaceCard {
flex-direction: column; flex-direction: column;
justify-content: flex-end; justify-content: flex-end;
height: 100%; height: 100%;
min-height: 24px;
} }
[role="button"] { [role="button"] {
cursor: pointer; cursor: pointer;
@@ -148,7 +147,7 @@ export class HuiHeadingCard extends LitElement implements LovelaceCard {
transition: transform 180ms ease-in-out; transition: transform 180ms ease-in-out;
} }
.container { .container {
padding: 0 4px; padding: 2px 4px;
display: flex; display: flex;
flex-direction: row; flex-direction: row;
justify-content: space-between; justify-content: space-between;

View File

@@ -1,372 +0,0 @@
import { css, html, LitElement, nothing } from "lit";
import { customElement, property, state } from "lit/decorators";
import { classMap } from "lit/directives/class-map";
import { ifDefined } from "lit/directives/if-defined";
import { styleMap } from "lit/directives/style-map";
import { computeCssColor } from "../../../common/color/compute-color";
import { computeDomain } from "../../../common/entity/compute_domain";
import { generateEntityFilter } from "../../../common/entity/entity_filter";
import { formatNumber } from "../../../common/number/format_number";
import "../../../components/ha-card";
import "../../../components/ha-icon";
import "../../../components/ha-ripple";
import "../../../components/tile/ha-tile-icon";
import "../../../components/tile/ha-tile-info";
import type { ActionHandlerEvent } from "../../../data/lovelace/action_handler";
import "../../../state-display/state-display";
import type { HomeAssistant } from "../../../types";
import { actionHandler } from "../common/directives/action-handler-directive";
import { handleAction } from "../common/handle-action";
import { hasAction } from "../common/has-action";
import {
findEntities,
getSummaryLabel,
HOME_SUMMARIES_FILTERS,
HOME_SUMMARIES_ICONS,
type HomeSummary,
} from "../strategies/home/helpers/home-summaries";
import type { LovelaceCard, LovelaceGridOptions } from "../types";
import type { HomeSummaryCard } from "./types";
const COLORS: Record<HomeSummary, string> = {
lights: "amber",
climate: "deep-orange",
security: "blue-grey",
media_players: "blue",
};
@customElement("hui-home-summary-card")
export class HuiHomeSummaryCard extends LitElement implements LovelaceCard {
@property({ attribute: false }) public hass?: HomeAssistant;
@state() private _config?: HomeSummaryCard;
public setConfig(config: HomeSummaryCard): void {
this._config = config;
}
public getCardSize(): number {
return this._config?.vertical ? 2 : 1;
}
public getGridOptions(): LovelaceGridOptions {
const columns = 6;
let min_columns = 6;
let rows = 1;
if (this._config?.vertical) {
rows++;
min_columns = 3;
}
return {
columns,
rows,
min_columns,
min_rows: rows,
};
}
private _handleAction(ev: ActionHandlerEvent) {
handleAction(this, this.hass!, this._config!, ev.detail.action!);
}
private get _hasCardAction() {
return (
hasAction(this._config?.tap_action) ||
hasAction(this._config?.hold_action) ||
hasAction(this._config?.double_tap_action)
);
}
private _computeSummaryState(): string {
if (!this._config || !this.hass) {
return "";
}
const allEntities = Object.keys(this.hass!.states);
const areas = Object.values(this.hass.areas);
const areasFilter = generateEntityFilter(this.hass, {
area: areas.map((area) => area.area_id),
});
const entitiesInsideArea = allEntities.filter(areasFilter);
switch (this._config.summary) {
case "lights": {
// Number of lights on
const lightsFilters = HOME_SUMMARIES_FILTERS.lights.map((filter) =>
generateEntityFilter(this.hass!, filter)
);
const lightEntities = findEntities(entitiesInsideArea, lightsFilters);
const onLights = lightEntities.filter((entityId) => {
const s = this.hass!.states[entityId]?.state;
return s === "on";
});
return onLights.length
? this.hass.localize("ui.card.home-summary.count_lights_on", {
count: onLights.length,
})
: this.hass.localize("ui.card.home-summary.all_lights_off");
}
case "climate": {
// Min/Max temperature of the areas
const areaSensors = areas
.map((area) => area.temperature_entity_id)
.filter(Boolean);
const sensorsValues = areaSensors
.map(
(entityId) => parseFloat(this.hass!.states[entityId!]?.state) || NaN
)
.filter((value) => !isNaN(value));
if (sensorsValues.length === 0) {
return "";
}
const minTemp = Math.min(...sensorsValues);
const maxTemp = Math.max(...sensorsValues);
if (isNaN(minTemp) || isNaN(maxTemp)) {
return "";
}
const formattedMinTemp = formatNumber(minTemp, this.hass?.locale, {
minimumFractionDigits: 1,
maximumFractionDigits: 1,
});
const formattedMaxTemp = formatNumber(maxTemp, this.hass?.locale, {
minimumFractionDigits: 1,
maximumFractionDigits: 1,
});
return formattedMinTemp === formattedMaxTemp
? `${formattedMinTemp}°`
: `${formattedMinTemp} - ${formattedMaxTemp}°`;
}
case "security": {
// Alarm and lock status
const securityFilters = HOME_SUMMARIES_FILTERS.security.map((filter) =>
generateEntityFilter(this.hass!, filter)
);
const securityEntities = findEntities(
entitiesInsideArea,
securityFilters
);
const locks = securityEntities.filter((entityId) => {
const domain = computeDomain(entityId);
return domain === "lock";
});
const alarms = securityEntities.filter((entityId) => {
const domain = computeDomain(entityId);
return domain === "alarm_control_panel";
});
const disarmedAlarms = alarms.filter((entityId) => {
const s = this.hass!.states[entityId]?.state;
return s === "disarmed";
});
if (!locks.length && !alarms.length) {
return "";
}
const unlockedLocks = locks.filter((entityId) => {
const s = this.hass!.states[entityId]?.state;
return s === "unlocked" || s === "jammed" || s === "open";
});
if (unlockedLocks.length) {
return this.hass.localize(
"ui.card.home-summary.count_locks_unlocked",
{
count: unlockedLocks.length,
}
);
}
if (disarmedAlarms.length) {
return this.hass.localize(
"ui.card.home-summary.count_alarms_disarmed",
{
count: disarmedAlarms.length,
}
);
}
return this.hass.localize("ui.card.home-summary.all_secure");
}
case "media_players": {
// Playing media
const mediaPlayerFilters = HOME_SUMMARIES_FILTERS.media_players.map(
(filter) => generateEntityFilter(this.hass!, filter)
);
const mediaPlayerEntities = findEntities(
entitiesInsideArea,
mediaPlayerFilters
);
const playingMedia = mediaPlayerEntities.filter((entityId) => {
const s = this.hass!.states[entityId]?.state;
return s === "playing";
});
return playingMedia.length
? this.hass.localize("ui.card.home-summary.count_media_playing", {
count: playingMedia.length,
})
: this.hass.localize("ui.card.home-summary.no_media_playing");
}
}
return "";
}
protected render() {
if (!this._config || !this.hass) {
return nothing;
}
const contentClasses = { vertical: Boolean(this._config.vertical) };
const color = computeCssColor(COLORS[this._config.summary]);
const style = {
"--tile-color": color,
};
const secondary = this._computeSummaryState();
const label = getSummaryLabel(this.hass.localize, this._config.summary);
const icon = HOME_SUMMARIES_ICONS[this._config.summary];
return html`
<ha-card style=${styleMap(style)}>
<div
class="background"
@action=${this._handleAction}
.actionHandler=${actionHandler({
hasHold: hasAction(this._config!.hold_action),
hasDoubleClick: hasAction(this._config!.double_tap_action),
})}
role=${ifDefined(this._hasCardAction ? "button" : undefined)}
tabindex=${ifDefined(this._hasCardAction ? "0" : undefined)}
aria-labelledby="info"
>
<ha-ripple .disabled=${!this._hasCardAction}></ha-ripple>
</div>
<div class="container">
<div class="content ${classMap(contentClasses)}">
<ha-tile-icon>
<ha-icon slot="icon" .icon=${icon}></ha-icon>
</ha-tile-icon>
<ha-tile-info
id="info"
.primary=${label}
.secondary=${secondary}
></ha-tile-info>
</div>
</div>
</ha-card>
`;
}
static styles = css`
:host {
--tile-color: var(--state-inactive-color);
-webkit-tap-highlight-color: transparent;
}
ha-card:has(.background:focus-visible) {
--shadow-default: var(--ha-card-box-shadow, 0 0 0 0 transparent);
--shadow-focus: 0 0 0 1px var(--tile-color);
border-color: var(--tile-color);
box-shadow: var(--shadow-default), var(--shadow-focus);
}
ha-card {
--ha-ripple-color: var(--tile-color);
--ha-ripple-hover-opacity: 0.04;
--ha-ripple-pressed-opacity: 0.12;
height: 100%;
transition:
box-shadow 180ms ease-in-out,
border-color 180ms ease-in-out;
display: flex;
flex-direction: column;
justify-content: space-between;
}
ha-card.active {
--tile-color: var(--state-icon-color);
}
[role="button"] {
cursor: pointer;
pointer-events: auto;
}
[role="button"]:focus {
outline: none;
}
.background {
position: absolute;
top: 0;
left: 0;
bottom: 0;
right: 0;
border-radius: var(--ha-card-border-radius, 12px);
margin: calc(-1 * var(--ha-card-border-width, 1px));
overflow: hidden;
}
.container {
margin: calc(-1 * var(--ha-card-border-width, 1px));
display: flex;
flex-direction: column;
flex: 1;
}
.container.horizontal {
flex-direction: row;
}
.content {
position: relative;
display: flex;
flex-direction: row;
align-items: center;
padding: 10px;
flex: 1;
min-width: 0;
box-sizing: border-box;
pointer-events: none;
gap: 10px;
}
.vertical {
flex-direction: column;
text-align: center;
justify-content: center;
}
.vertical ha-tile-info {
width: 100%;
flex: none;
}
ha-tile-icon {
--tile-icon-color: var(--tile-color);
position: relative;
padding: 6px;
margin: -6px;
}
ha-tile-info {
position: relative;
min-width: 0;
transition: background-color 180ms ease-in-out;
box-sizing: border-box;
}
`;
}
declare global {
interface HTMLElementTagNameMap {
"hui-home-summary-card": HuiHomeSummaryCard;
}
}

View File

@@ -26,7 +26,6 @@ import type {
import type { LovelaceHeaderFooterConfig } from "../header-footer/types"; import type { LovelaceHeaderFooterConfig } from "../header-footer/types";
import type { LovelaceHeadingBadgeConfig } from "../heading-badges/types"; import type { LovelaceHeadingBadgeConfig } from "../heading-badges/types";
import type { TimeFormat } from "../../../data/translation"; import type { TimeFormat } from "../../../data/translation";
import type { HomeSummary } from "../strategies/home/helpers/home-summaries";
export type AlarmPanelCardConfigState = export type AlarmPanelCardConfigState =
| "arm_away" | "arm_away"
@@ -589,11 +588,3 @@ export interface HeadingCardConfig extends LovelaceCardConfig {
/** @deprecated Use `badges` instead */ /** @deprecated Use `badges` instead */
entities?: LovelaceHeadingBadgeConfig[]; entities?: LovelaceHeadingBadgeConfig[];
} }
export interface HomeSummaryCard extends LovelaceCardConfig {
summary: HomeSummary;
vertical?: boolean;
tap_action?: ActionConfig;
hold_action?: ActionConfig;
double_tap_action?: ActionConfig;
}

View File

@@ -14,8 +14,8 @@ const calcPoints = (
detail: number, detail: number,
min: number, min: number,
max: number max: number
): [number, number][] => { ): number[][] => {
const coords = [] as [number, number][]; const coords = [] as number[][];
const height = 80; const height = 80;
let yRatio = (max - min) / height; let yRatio = (max - min) / height;
yRatio = yRatio !== 0 ? yRatio : height; yRatio = yRatio !== 0 ? yRatio : height;
@@ -61,7 +61,7 @@ export const coordinates = (
width: number, width: number,
detail: number, detail: number,
limits?: { min?: number; max?: number } limits?: { min?: number; max?: number }
): [number, number][] | undefined => { ): number[][] | undefined => {
history.forEach((item) => { history.forEach((item) => {
item.state = Number(item.state); item.state = Number(item.state);
}); });
@@ -119,7 +119,7 @@ export const coordinatesMinimalResponseCompressedState = (
width: number, width: number,
detail: number, detail: number,
limits?: { min?: number; max?: number } limits?: { min?: number; max?: number }
): [number, number][] | undefined => { ): number[][] | undefined => {
if (!history) { if (!history) {
return undefined; return undefined;
} }

View File

@@ -13,7 +13,7 @@ export class HuiGraphBase extends LitElement {
protected render(): TemplateResult { protected render(): TemplateResult {
return html` return html`
${this._path ${this._path
? svg`<svg width="100%" height="100%" viewBox="0 0 500 100" preserveAspectRatio="none"> ? svg`<svg width="100%" height="100%" viewBox="0 0 500 100">
<g> <g>
<mask id="fill"> <mask id="fill">
<path <path
@@ -25,10 +25,8 @@ export class HuiGraphBase extends LitElement {
<rect height="100%" width="100%" id="fill-rect" fill="var(--accent-color)" mask="url(#fill)"></rect> <rect height="100%" width="100%" id="fill-rect" fill="var(--accent-color)" mask="url(#fill)"></rect>
<mask id="line"> <mask id="line">
<path <path
vector-effect="non-scaling-stroke"
class='line'
fill="none" fill="none"
stroke="white" stroke="var(--accent-color)"
stroke-width="${strokeWidth}" stroke-width="${strokeWidth}"
stroke-linecap="round" stroke-linecap="round"
stroke-linejoin="round" stroke-linejoin="round"
@@ -56,10 +54,6 @@ export class HuiGraphBase extends LitElement {
:host { :host {
display: flex; display: flex;
width: 100%; width: 100%;
height: 100%;
}
.line {
opacity: 0.8;
} }
.fill { .fill {
opacity: 0.1; opacity: 0.1;

View File

@@ -68,7 +68,6 @@ const LAZY_LOAD_TYPES = {
"energy-sankey": () => import("../cards/energy/hui-energy-sankey-card"), "energy-sankey": () => import("../cards/energy/hui-energy-sankey-card"),
"entity-filter": () => import("../cards/hui-entity-filter-card"), "entity-filter": () => import("../cards/hui-entity-filter-card"),
error: () => import("../cards/hui-error-card"), error: () => import("../cards/hui-error-card"),
"home-summary": () => import("../cards/hui-home-summary-card"),
gauge: () => import("../cards/hui-gauge-card"), gauge: () => import("../cards/hui-gauge-card"),
"history-graph": () => import("../cards/hui-history-graph-card"), "history-graph": () => import("../cards/hui-history-graph-card"),
"horizontal-stack": () => import("../cards/hui-horizontal-stack-card"), "horizontal-stack": () => import("../cards/hui-horizontal-stack-card"),

View File

@@ -36,7 +36,7 @@ import "../card-features/hui-valve-position-card-feature";
import "../card-features/hui-water-heater-operation-modes-card-feature"; import "../card-features/hui-water-heater-operation-modes-card-feature";
import "../card-features/hui-area-controls-card-feature"; import "../card-features/hui-area-controls-card-feature";
import "../card-features/hui-bar-gauge-card-feature"; import "../card-features/hui-bar-gauge-card-feature";
import "../card-features/hui-trend-graph-card-feature"; import "../card-features/hui-history-chart-card-feature";
import type { LovelaceCardFeatureConfig } from "../card-features/types"; import type { LovelaceCardFeatureConfig } from "../card-features/types";
import { import {
@@ -75,7 +75,7 @@ const TYPES = new Set<LovelaceCardFeatureConfig["type"]>([
"media-player-volume-slider", "media-player-volume-slider",
"numeric-input", "numeric-input",
"select-options", "select-options",
"trend-graph", "history-chart",
"target-humidity", "target-humidity",
"target-temperature", "target-temperature",
"toggle", "toggle",

View File

@@ -46,7 +46,7 @@ import { supportsMediaPlayerPlaybackCardFeature } from "../../card-features/hui-
import { supportsMediaPlayerVolumeSliderCardFeature } from "../../card-features/hui-media-player-volume-slider-card-feature"; import { supportsMediaPlayerVolumeSliderCardFeature } from "../../card-features/hui-media-player-volume-slider-card-feature";
import { supportsNumericInputCardFeature } from "../../card-features/hui-numeric-input-card-feature"; import { supportsNumericInputCardFeature } from "../../card-features/hui-numeric-input-card-feature";
import { supportsSelectOptionsCardFeature } from "../../card-features/hui-select-options-card-feature"; import { supportsSelectOptionsCardFeature } from "../../card-features/hui-select-options-card-feature";
import { supportsTrendGraphCardFeature } from "../../card-features/hui-trend-graph-card-feature"; import { supportsHistoryChartCardFeature } from "../../card-features/hui-history-chart-card-feature";
import { supportsTargetHumidityCardFeature } from "../../card-features/hui-target-humidity-card-feature"; import { supportsTargetHumidityCardFeature } from "../../card-features/hui-target-humidity-card-feature";
import { supportsTargetTemperatureCardFeature } from "../../card-features/hui-target-temperature-card-feature"; import { supportsTargetTemperatureCardFeature } from "../../card-features/hui-target-temperature-card-feature";
import { supportsToggleCardFeature } from "../../card-features/hui-toggle-card-feature"; import { supportsToggleCardFeature } from "../../card-features/hui-toggle-card-feature";
@@ -100,7 +100,7 @@ const UI_FEATURE_TYPES = [
"media-player-volume-slider", "media-player-volume-slider",
"numeric-input", "numeric-input",
"select-options", "select-options",
"trend-graph", "history-chart",
"target-humidity", "target-humidity",
"target-temperature", "target-temperature",
"toggle", "toggle",
@@ -128,7 +128,6 @@ const EDITABLES_FEATURE_TYPES = new Set<UiFeatureTypes>([
"lawn-mower-commands", "lawn-mower-commands",
"numeric-input", "numeric-input",
"select-options", "select-options",
"trend-graph",
"update-actions", "update-actions",
"vacuum-commands", "vacuum-commands",
"water-heater-operation-modes", "water-heater-operation-modes",
@@ -169,7 +168,7 @@ const SUPPORTS_FEATURE_TYPES: Record<
"media-player-volume-slider": supportsMediaPlayerVolumeSliderCardFeature, "media-player-volume-slider": supportsMediaPlayerVolumeSliderCardFeature,
"numeric-input": supportsNumericInputCardFeature, "numeric-input": supportsNumericInputCardFeature,
"select-options": supportsSelectOptionsCardFeature, "select-options": supportsSelectOptionsCardFeature,
"trend-graph": supportsTrendGraphCardFeature, "history-chart": supportsHistoryChartCardFeature,
"target-humidity": supportsTargetHumidityCardFeature, "target-humidity": supportsTargetHumidityCardFeature,
"target-temperature": supportsTargetTemperatureCardFeature, "target-temperature": supportsTargetTemperatureCardFeature,
toggle: supportsToggleCardFeature, toggle: supportsToggleCardFeature,

View File

@@ -1,82 +0,0 @@
import { html, LitElement, nothing } from "lit";
import { customElement, property, state } from "lit/decorators";
import { fireEvent } from "../../../../common/dom/fire_event";
import "../../../../components/ha-form/ha-form";
import type {
HaFormSchema,
SchemaUnion,
} from "../../../../components/ha-form/types";
import type { HomeAssistant } from "../../../../types";
import { DEFAULT_HOURS_TO_SHOW } from "../../card-features/hui-trend-graph-card-feature";
import type {
LovelaceCardFeatureContext,
TrendGraphCardFeatureConfig,
} from "../../card-features/types";
import type { LovelaceCardFeatureEditor } from "../../types";
const SCHEMA = [
{
name: "hours_to_show",
default: DEFAULT_HOURS_TO_SHOW,
selector: { number: { min: 1, mode: "box" } },
},
] as const satisfies HaFormSchema[];
@customElement("hui-trend-graph-card-feature-editor")
export class HuiTrendGraphCardFeatureEditor
extends LitElement
implements LovelaceCardFeatureEditor
{
@property({ attribute: false }) public hass?: HomeAssistant;
@property({ attribute: false }) public context?: LovelaceCardFeatureContext;
@state() private _config?: TrendGraphCardFeatureConfig;
public setConfig(config: TrendGraphCardFeatureConfig): void {
this._config = config;
}
protected render() {
if (!this.hass || !this._config) {
return nothing;
}
const data = { ...this._config };
if (!this._config.hours_to_show) {
data.hours_to_show = DEFAULT_HOURS_TO_SHOW;
}
return html`
<ha-form
.hass=${this.hass}
.data=${data}
.schema=${SCHEMA}
.computeLabel=${this._computeLabelCallback}
@value-changed=${this._valueChanged}
></ha-form>
`;
}
private _valueChanged(ev: CustomEvent): void {
fireEvent(this, "config-changed", { config: ev.detail.value });
}
private _computeLabelCallback = (schema: SchemaUnion<typeof SCHEMA>) => {
switch (schema.name) {
case "hours_to_show":
return this.hass!.localize(
`ui.panel.lovelace.editor.card.generic.${schema.name}`
);
default:
return "";
}
};
}
declare global {
interface HTMLElementTagNameMap {
"hui-trend-graph-card-feature-editor": HuiTrendGraphCardFeatureEditor;
}
}

View File

@@ -61,7 +61,7 @@ export class HuiGraphHeaderFooter
@state() protected _config?: GraphHeaderFooterConfig; @state() protected _config?: GraphHeaderFooterConfig;
@state() private _coordinates?: [number, number][]; @state() private _coordinates?: number[][];
private _error?: string; private _error?: string;

View File

@@ -355,10 +355,7 @@ class HUIRoot extends LitElement {
overflowItems.forEach((i) => { overflowItems.forEach((i) => {
const title = [this.hass!.localize(i.key), i.suffix].join(" "); const title = [this.hass!.localize(i.key), i.suffix].join(" ");
const action = i.subItems const action = i.subItems
? (e) => { ? () => {
if (!shouldHandleRequestSelectedEvent(e)) {
return;
}
showListItemsDialog(this, { showListItemsDialog(this, {
title: title, title: title,
items: i.subItems!.map((si) => ({ items: i.subItems!.map((si) => ({

View File

@@ -12,11 +12,30 @@ import {
} from "../areas/helpers/areas-strategy-helper"; } from "../areas/helpers/areas-strategy-helper";
import { getHomeStructure } from "./helpers/home-structure"; import { getHomeStructure } from "./helpers/home-structure";
import { findEntities, HOME_SUMMARIES_FILTERS } from "./helpers/home-summaries"; import { findEntities, HOME_SUMMARIES_FILTERS } from "./helpers/home-summaries";
import { computeStateName } from "../../../../common/entity/compute_state_name";
import { computeObjectId } from "../../../../common/entity/compute_object_id";
export interface HomeClimateViewStrategyConfig { export interface HomeClimateViewStrategyConfig {
type: "home-climate"; type: "home-climate";
} }
const createTempHumidBadge = (hass: HomeAssistant, entityId: string) => {
const stateObj = hass.states[entityId];
return {
type: "tile",
entity: entityId,
name: stateObj
? computeStateName(stateObj)
: computeObjectId(entityId).replace(/_/g, " "),
features: [
{
type: "history-chart",
hours_to_show: 3,
},
],
};
};
const processAreasForClimate = ( const processAreasForClimate = (
areaIds: string[], areaIds: string[],
hass: HomeAssistant, hass: HomeAssistant,
@@ -33,28 +52,8 @@ const processAreasForClimate = (
area: area.area_id, area: area.area_id,
}); });
const areaEntities = entities.filter(areaFilter); const areaEntities = entities.filter(areaFilter);
const areaCards: LovelaceCardConfig[] = [];
const temperatureEntityId = area.temperature_entity_id; if (areaEntities.length > 0) {
if (temperatureEntityId && hass.states[temperatureEntityId]) {
areaCards.push({
...computeTileCard(temperatureEntityId),
features: [{ type: "trend-graph" }],
});
}
const humidityEntityId = area.humidity_entity_id;
if (humidityEntityId && hass.states[humidityEntityId]) {
areaCards.push({
...computeTileCard(humidityEntityId),
features: [{ type: "trend-graph" }],
});
}
for (const entityId of areaEntities) {
areaCards.push(computeTileCard(entityId));
}
if (areaCards.length > 0) {
cards.push({ cards.push({
heading_style: "subtitle", heading_style: "subtitle",
type: "heading", type: "heading",
@@ -64,7 +63,21 @@ const processAreasForClimate = (
navigation_path: `areas-${area.area_id}`, navigation_path: `areas-${area.area_id}`,
}, },
}); });
cards.push(...areaCards);
if (hass.areas[areaId].temperature_entity_id) {
cards.push(
createTempHumidBadge(hass, hass.areas[areaId].temperature_entity_id)
);
}
if (hass.areas[areaId].humidity_entity_id) {
cards.push(
createTempHumidBadge(hass, hass.areas[areaId].humidity_entity_id)
);
}
for (const entityId of areaEntities) {
cards.push(computeTileCard(entityId));
}
} }
} }

View File

@@ -32,15 +32,10 @@ const processAreasForLights = (
area: area.area_id, area: area.area_id,
}); });
const areaLights = entities.filter(areaFilter); const areaLights = entities.filter(areaFilter);
const areaCards: LovelaceCardConfig[] = [];
const computeTileCard = computeAreaTileCardConfig(hass, "", false); const computeTileCard = computeAreaTileCardConfig(hass, "", false);
for (const entityId of areaLights) { if (areaLights.length > 0) {
areaCards.push(computeTileCard(entityId));
}
if (areaCards.length > 0) {
cards.push({ cards.push({
heading_style: "subtitle", heading_style: "subtitle",
type: "heading", type: "heading",
@@ -50,7 +45,10 @@ const processAreasForLights = (
navigation_path: `areas-${area.area_id}`, navigation_path: `areas-${area.area_id}`,
}, },
}); });
cards.push(...areaCards);
for (const entityId of areaLights) {
cards.push(computeTileCard(entityId));
}
} }
} }

View File

@@ -9,12 +9,16 @@ import type { LovelaceViewConfig } from "../../../../data/lovelace/config/view";
import type { HomeAssistant } from "../../../../types"; import type { HomeAssistant } from "../../../../types";
import type { import type {
AreaCardConfig, AreaCardConfig,
HomeSummaryCard, ButtonCardConfig,
MarkdownCardConfig, MarkdownCardConfig,
TileCardConfig, TileCardConfig,
WeatherForecastCardConfig, WeatherForecastCardConfig,
} from "../../cards/types"; } from "../../cards/types";
import { getAreas } from "../areas/helpers/areas-strategy-helper"; import { getAreas } from "../areas/helpers/areas-strategy-helper";
import {
getSummaryLabel,
HOME_SUMMARIES_ICONS,
} from "./helpers/home-summaries";
export interface HomeMainViewStrategyConfig { export interface HomeMainViewStrategyConfig {
type: "home-main"; type: "home-main";
@@ -110,57 +114,61 @@ export class HomeMainViewStrategy extends ReactiveElement {
heading: hass.localize("ui.panel.lovelace.strategy.home.summaries"), heading: hass.localize("ui.panel.lovelace.strategy.home.summaries"),
}, },
{ {
type: "home-summary", type: "button",
summary: "lights", icon: HOME_SUMMARIES_ICONS.lights,
vertical: true, name: getSummaryLabel(hass.localize, "lights"),
icon_height: "24px",
grid_options: {
rows: 2,
columns: 4,
},
tap_action: { tap_action: {
action: "navigate", action: "navigate",
navigation_path: "lights", navigation_path: "lights",
}, },
} satisfies ButtonCardConfig,
{
type: "button",
icon: HOME_SUMMARIES_ICONS.climate,
name: getSummaryLabel(hass.localize, "climate"),
icon_height: "30px",
grid_options: { grid_options: {
rows: 2, rows: 2,
columns: 4, columns: 4,
}, },
} satisfies HomeSummaryCard,
{
type: "home-summary",
summary: "climate",
vertical: true,
tap_action: { tap_action: {
action: "navigate", action: "navigate",
navigation_path: "climate", navigation_path: "climate",
}, },
} satisfies ButtonCardConfig,
{
type: "button",
icon: HOME_SUMMARIES_ICONS.security,
name: getSummaryLabel(hass.localize, "security"),
icon_height: "30px",
grid_options: { grid_options: {
rows: 2, rows: 2,
columns: 4, columns: 4,
}, },
} satisfies HomeSummaryCard,
{
type: "home-summary",
summary: "security",
vertical: true,
tap_action: { tap_action: {
action: "navigate", action: "navigate",
navigation_path: "security", navigation_path: "security",
}, },
} satisfies ButtonCardConfig,
{
type: "button",
icon: HOME_SUMMARIES_ICONS.media_players,
name: getSummaryLabel(hass.localize, "media_players"),
icon_height: "30px",
grid_options: { grid_options: {
rows: 2, rows: 2,
columns: 4, columns: 4,
}, },
} satisfies HomeSummaryCard,
{
type: "home-summary",
summary: "media_players",
vertical: true,
tap_action: { tap_action: {
action: "navigate", action: "navigate",
navigation_path: "media-players", navigation_path: "media-players",
}, },
grid_options: { } satisfies ButtonCardConfig,
rows: 2,
columns: 4,
},
} satisfies HomeSummaryCard,
], ],
}; };
@@ -207,7 +215,6 @@ export class HomeMainViewStrategy extends ReactiveElement {
"ui.panel.lovelace.cards.energy.energy_distribution.title_today" "ui.panel.lovelace.cards.energy.energy_distribution.title_today"
), ),
type: "energy-distribution", type: "energy-distribution",
collection_key: "energy_home_dashboard",
link_dashboard: true, link_dashboard: true,
}); });
} }

View File

@@ -29,14 +29,6 @@ const processAreasForMediaPlayers = (
area: area.area_id, area: area.area_id,
}); });
const areaEntities = entities.filter(areaFilter); const areaEntities = entities.filter(areaFilter);
const areaCards: LovelaceCardConfig[] = [];
for (const entityId of areaEntities) {
cards.push({
type: "media-control",
entity: entityId,
} satisfies MediaControlCardConfig);
}
if (areaEntities.length > 0) { if (areaEntities.length > 0) {
cards.push({ cards.push({
@@ -48,7 +40,13 @@ const processAreasForMediaPlayers = (
navigation_path: `areas-${area.area_id}`, navigation_path: `areas-${area.area_id}`,
}, },
}); });
cards.push(...areaCards);
for (const entityId of areaEntities) {
cards.push({
type: "media-control",
entity: entityId,
} satisfies MediaControlCardConfig);
}
} }
} }

View File

@@ -12,7 +12,9 @@ import {
} from "../areas/helpers/areas-strategy-helper"; } from "../areas/helpers/areas-strategy-helper";
import { getHomeStructure } from "./helpers/home-structure"; import { getHomeStructure } from "./helpers/home-structure";
import { findEntities, HOME_SUMMARIES_FILTERS } from "./helpers/home-summaries"; import { findEntities, HOME_SUMMARIES_FILTERS } from "./helpers/home-summaries";
import type { LogbookCardConfig } from "../../cards/types"; import { computeDomain } from "../../../../common/entity/compute_domain";
import { computeStateName } from "../../../../common/entity/compute_state_name";
import { computeObjectId } from "../../../../common/entity/compute_object_id";
export interface HomeSecurityViewStrategyConfig { export interface HomeSecurityViewStrategyConfig {
type: "home-security"; type: "home-security";
@@ -33,13 +35,7 @@ const processAreasForSecurity = (
const areaFilter = generateEntityFilter(hass, { const areaFilter = generateEntityFilter(hass, {
area: area.area_id, area: area.area_id,
}); });
const areaEntities = entities.filter(areaFilter); const areaEntities = entities.filter(areaFilter);
const areaCards: LovelaceCardConfig[] = [];
for (const entityId of areaEntities) {
areaCards.push(computeTileCard(entityId));
}
if (areaEntities.length > 0) { if (areaEntities.length > 0) {
cards.push({ cards.push({
@@ -51,7 +47,28 @@ const processAreasForSecurity = (
navigation_path: `areas-${area.area_id}`, navigation_path: `areas-${area.area_id}`,
}, },
}); });
cards.push(...areaCards);
for (const entityId of areaEntities) {
const stateObj = hass.states[entityId];
cards.push(
computeDomain(entityId) === "binary_sensor" &&
stateObj?.attributes.device_class === "motion"
? {
type: "tile",
entity: entityId,
name: stateObj
? computeStateName(stateObj)
: computeObjectId(entityId).replace(/_/g, " "),
features: [
{
type: "history-chart",
hours_to_show: 6,
},
],
}
: computeTileCard(entityId)
);
}
} }
} }
@@ -132,24 +149,6 @@ export class HomeSecurityViewStrategy extends ReactiveElement {
} }
} }
sections.push({
type: "grid",
cards: [
{
type: "heading",
heading: hass.localize("panel.logbook"),
tap_action: {
action: "navigate",
navigation_path: `/logbook?entity_id=${entities.join(",")}`,
},
},
{
type: "logbook",
target: { entity_id: entities },
} satisfies LogbookCardConfig,
],
});
return { return {
type: "sections", type: "sections",
max_columns: 2, max_columns: 2,

View File

@@ -218,7 +218,6 @@ export const colorStyles = css`
--table-row-alternative-background-color: var(--secondary-background-color); --table-row-alternative-background-color: var(--secondary-background-color);
--data-table-background-color: var(--card-background-color); --data-table-background-color: var(--card-background-color);
--markdown-code-background-color: var(--primary-background-color); --markdown-code-background-color: var(--primary-background-color);
--bar-box-shadow: 0 2px 12px rgba(0, 0, 0, 0.16);
/* https://github.com/material-components/material-web/blob/master/docs/theming.md */ /* https://github.com/material-components/material-web/blob/master/docs/theming.md */
--mdc-theme-primary: var(--primary-color); --mdc-theme-primary: var(--primary-color);
@@ -247,7 +246,6 @@ export const colorStyles = css`
--mdc-dialog-scroll-divider-color: var(--divider-color); --mdc-dialog-scroll-divider-color: var(--divider-color);
--mdc-dialog-heading-ink-color: var(--primary-text-color); --mdc-dialog-heading-ink-color: var(--primary-text-color);
--mdc-dialog-content-ink-color: var(--primary-text-color); --mdc-dialog-content-ink-color: var(--primary-text-color);
--mdc-top-app-bar-fixed-box-shadow: var(--bar-box-shadow);
--mdc-text-field-idle-line-color: var(--input-idle-line-color); --mdc-text-field-idle-line-color: var(--input-idle-line-color);
--mdc-text-field-hover-line-color: var(--input-hover-line-color); --mdc-text-field-hover-line-color: var(--input-hover-line-color);
@@ -362,8 +360,6 @@ export const darkColorStyles = css`
--ha-button-warning-light-color: #917b54c1; --ha-button-warning-light-color: #917b54c1;
--ha-button-neutral-color: #d9dae0; --ha-button-neutral-color: #d9dae0;
--ha-button-neutral-light-color: #6a7081; --ha-button-neutral-light-color: #6a7081;
--bar-box-shadow: 0 2px 12px rgba(0, 0, 0, 0.48);
} }
`; `;

View File

@@ -5,7 +5,7 @@
"config": "Settings", "config": "Settings",
"states": "Overview", "states": "Overview",
"map": "Map", "map": "Map",
"logbook": "Logbook", "logbook": "Activity",
"history": "History", "history": "History",
"todo": "To-do lists", "todo": "To-do lists",
"developer_tools": "Developer tools", "developer_tools": "Developer tools",
@@ -201,15 +201,6 @@
"open_door_confirm": "Really open?", "open_door_confirm": "Really open?",
"open_door_done": "Done" "open_door_done": "Done"
}, },
"home-summary": {
"all_lights_off": "All off",
"count_lights_on": "{count} {count, plural,\n one {on}\n other {on}\n}",
"count_locks_unlocked": "{count} {count, plural,\n one {unlocked}\n other {unlocked}\n}",
"count_alarms_disarmed": "{count} {count, plural,\n one {disarmed}\n other {disarmed}\n}",
"all_secure": "All secure",
"no_media_playing": "No media playing",
"count_media_playing": "{count} {count, plural,\n one {playing}\n other {playing}\n}"
},
"media_player": { "media_player": {
"source": "Source", "source": "Source",
"sound_mode": "Sound mode", "sound_mode": "Sound mode",
@@ -526,7 +517,7 @@
} }
}, },
"logbook": { "logbook": {
"entries_not_found": "No logbook events found.", "entries_not_found": "No activity found.",
"triggered_by": "triggered by", "triggered_by": "triggered by",
"triggered_by_automation": "triggered by automation", "triggered_by_automation": "triggered by automation",
"triggered_by_script": "triggered by script", "triggered_by_script": "triggered by script",
@@ -539,7 +530,7 @@
"triggered_by_homeassistant_stopping": "triggered by Home Assistant stopping", "triggered_by_homeassistant_stopping": "triggered by Home Assistant stopping",
"triggered_by_homeassistant_starting": "triggered by Home Assistant starting", "triggered_by_homeassistant_starting": "triggered by Home Assistant starting",
"show_trace": "[%key:ui::panel::config::automation::editor::show_trace%]", "show_trace": "[%key:ui::panel::config::automation::editor::show_trace%]",
"retrieval_error": "Could not load logbook", "retrieval_error": "Could not load activity",
"not_loaded": "[%key:ui::dialogs::helper_settings::platform_not_loaded%]", "not_loaded": "[%key:ui::dialogs::helper_settings::platform_not_loaded%]",
"messages": { "messages": {
"was_away": "was detected away", "was_away": "was detected away",
@@ -678,7 +669,7 @@
}, },
"subpage-data-table": { "subpage-data-table": {
"filters": "Filters", "filters": "Filters",
"show_results": "Show {number} results", "show_results": "show {number} results",
"clear_filter": "Clear filter", "clear_filter": "Clear filter",
"close_filter": "Close filters", "close_filter": "Close filters",
"exit_selection_mode": "Exit selection mode", "exit_selection_mode": "Exit selection mode",
@@ -1382,12 +1373,8 @@
"info": "Information", "info": "Information",
"related": "Related", "related": "Related",
"history": "History", "history": "History",
"logbook": "Logbook", "logbook": "Activity",
"device_or_service_info": "[%key:ui::panel::config::devices::device_info%]", "device_info": "Device info",
"device_type": {
"device": "[%key:ui::panel::config::devices::type::device_heading%]",
"service": "[%key:ui::panel::config::devices::type::service_heading%]"
},
"last_changed": "Last changed", "last_changed": "Last changed",
"last_updated": "Last updated", "last_updated": "Last updated",
"show_more": "Show more", "show_more": "Show more",
@@ -1618,7 +1605,7 @@
"preload_stream": "Preload camera stream", "preload_stream": "Preload camera stream",
"preload_stream_description": "This keeps the camera stream open in the background so it shows quicker. Warning! This is device intensive.", "preload_stream_description": "This keeps the camera stream open in the background so it shows quicker. Warning! This is device intensive.",
"stream_orientation": "Camera stream orientation", "stream_orientation": "Camera stream orientation",
"stream_orientation_description": "The orientation transformation to use for the camera stream.\nWarning: Stream orientation processing occurs on the Home Assistant device and may impact system performance. When possible, configure this setting directly on your camera instead.", "stream_orientation_description": "The orientation transformation to use for the camera stream.",
"stream_orientation_1": "No orientation transform", "stream_orientation_1": "No orientation transform",
"stream_orientation_2": "Mirror", "stream_orientation_2": "Mirror",
"stream_orientation_3": "Rotate 180", "stream_orientation_3": "Rotate 180",
@@ -2039,13 +2026,11 @@
"title": "Shortcuts", "title": "Shortcuts",
"enable_shortcuts_hint": "For keyboard shortcuts to work, make sure you have them enabled in your {user_profile}.", "enable_shortcuts_hint": "For keyboard shortcuts to work, make sure you have them enabled in your {user_profile}.",
"enable_shortcuts_hint_user_profile": "user profile", "enable_shortcuts_hint_user_profile": "user profile",
"keys": {
"del": "Del"
},
"shortcuts": { "shortcuts": {
"double_click": "Double-click", "double_click": "Double-click",
"scroll_wheel": "Scroll", "scroll_wheel": "Scroll",
"drag": "Drag" "drag": "Drag",
"ctrl_cmd": "Ctrl/Cmd"
}, },
"searching": { "searching": {
"title": "Searching", "title": "Searching",
@@ -2062,9 +2047,6 @@
}, },
"automation_script": { "automation_script": {
"title": "Automations / Scripts", "title": "Automations / Scripts",
"copy": "to copy the selected item to clipboard",
"cut": "to cut the selected item and place it on the clipboard",
"delete": "to delete the selected item",
"paste": "to paste automation/script YAML from clipboard to editor", "paste": "to paste automation/script YAML from clipboard to editor",
"save": "to save automation/script" "save": "to save automation/script"
}, },
@@ -3865,9 +3847,6 @@
"type_script_plural": "[%key:ui::panel::config::blueprint::overview::types_plural::script%]", "type_script_plural": "[%key:ui::panel::config::blueprint::overview::types_plural::script%]",
"new_automation_setup_failed_title": "New {type} setup failed", "new_automation_setup_failed_title": "New {type} setup failed",
"new_automation_setup_failed_text": "Your new {type} has saved, but waiting for it to setup has timed out. This could be due to errors parsing your configuration.yaml, please check the configuration in developer tools. Your {type} will not be visible until this is corrected, and {types} are reloaded. Changes to area, category, or labels were not saved and must be reapplied.", "new_automation_setup_failed_text": "Your new {type} has saved, but waiting for it to setup has timed out. This could be due to errors parsing your configuration.yaml, please check the configuration in developer tools. Your {type} will not be visible until this is corrected, and {types} are reloaded. Changes to area, category, or labels were not saved and must be reapplied.",
"item_pasted": "{item} pasted",
"ctrl": "Ctrl",
"del": "Del",
"triggers": { "triggers": {
"name": "Triggers", "name": "Triggers",
"header": "When", "header": "When",
@@ -3894,8 +3873,6 @@
"unknown_trigger": "[%key:ui::panel::config::devices::automation::triggers::unknown_trigger%]", "unknown_trigger": "[%key:ui::panel::config::devices::automation::triggers::unknown_trigger%]",
"triggering_event_detail": "Triggering event detail", "triggering_event_detail": "Triggering event detail",
"trigger": "Trigger", "trigger": "Trigger",
"copied_to_clipboard": "Trigger copied to clipboard",
"cut_to_clipboard": "Trigger cut to clipboard",
"groups": { "groups": {
"entity": { "entity": {
"label": "Entity", "label": "Entity",
@@ -4132,9 +4109,6 @@
}, },
"conditions": { "conditions": {
"name": "Conditions", "name": "Conditions",
"header": "And if",
"description": "All conditions added here need to be satisfied for the automation to run. A condition can be satisfied or not at any given time, for example: ''If {user} is home''. You can use building blocks to create more complex conditions.",
"learn_more": "Learn more about conditions",
"add": "Add condition", "add": "Add condition",
"search": "Search condition", "search": "Search condition",
"add_building_block": "Add building block", "add_building_block": "Add building block",
@@ -4158,8 +4132,6 @@
"type_select": "Condition type", "type_select": "Condition type",
"unknown_condition": "[%key:ui::panel::config::devices::automation::conditions::unknown_condition%]", "unknown_condition": "[%key:ui::panel::config::devices::automation::conditions::unknown_condition%]",
"condition": "Condition", "condition": "Condition",
"copied_to_clipboard": "Condition copied to clipboard",
"cut_to_clipboard": "Condition cut to clipboard",
"groups": { "groups": {
"entity": { "entity": {
"label": "Entity", "label": "Entity",
@@ -4329,8 +4301,6 @@
"type_select": "Action type", "type_select": "Action type",
"continue_on_error": "Continue on error", "continue_on_error": "Continue on error",
"action": "Action", "action": "Action",
"copied_to_clipboard": "Action copied to clipboard",
"cut_to_clipboard": "Action cut to clipboard",
"groups": { "groups": {
"helpers": { "helpers": {
"label": "Helpers" "label": "Helpers"
@@ -4595,7 +4565,7 @@
"tabs": { "tabs": {
"details": "Step details", "details": "Step details",
"timeline": "Trace timeline", "timeline": "Trace timeline",
"logbook": "Related logbook entries", "logbook": "Related activity",
"automation_config": "Automation config", "automation_config": "Automation config",
"step_config": "Step config", "step_config": "Step config",
"changed_variables": "Changed variables", "changed_variables": "Changed variables",
@@ -4612,7 +4582,7 @@
"error": "Error: {error}", "error": "Error: {error}",
"result": "Result:", "result": "Result:",
"step_not_executed": "This step was not executed.", "step_not_executed": "This step was not executed.",
"no_logbook_entries": "No logbook entries found for this step.", "no_logbook_entries": "No activity found for this step.",
"no_variables_changed": "No variables changed", "no_variables_changed": "No variables changed",
"unable_to_find_config": "Unable to find config" "unable_to_find_config": "Unable to find config"
}, },
@@ -4638,8 +4608,8 @@
"disabled": "(disabled)", "disabled": "(disabled)",
"triggered_by": "{triggeredBy, select, \n alias {{alias} triggered}\n other {Triggered} \n} {triggeredPath, select, \n trigger {by the {trigger}}\n other {manually} \n} at {time}", "triggered_by": "{triggeredBy, select, \n alias {{alias} triggered}\n other {Triggered} \n} {triggeredPath, select, \n trigger {by the {trigger}}\n other {manually} \n} at {time}",
"path_error": "Unable to extract path {path}. Download trace and report as bug.", "path_error": "Unable to extract path {path}. Download trace and report as bug.",
"not_all_entries_are_related_automation_note": "Not all shown logbook entries might be related to this automation.", "not_all_entries_are_related_automation_note": "Not all shown activity might be related to this automation.",
"not_all_entries_are_related_script_note": "Not all shown logbook entries might be related to this script." "not_all_entries_are_related_script_note": "Not all shown activity might be related to this script."
} }
} }
}, },
@@ -5145,7 +5115,7 @@
"integration": "Integration", "integration": "Integration",
"config_entry": "Config entry" "config_entry": "Config entry"
}, },
"enabled_description": "Disabled devices and services will not be shown and entities belonging to them will be disabled, too.", "enabled_description": "Disabled devices will not be shown and entities belonging to the device will be disabled and not added to Home Assistant.",
"open_configuration_url": "Visit", "open_configuration_url": "Visit",
"set_up_voice_assistant": "Set up voice assistant", "set_up_voice_assistant": "Set up voice assistant",
"download_diagnostics": "Download diagnostics", "download_diagnostics": "Download diagnostics",
@@ -5742,16 +5712,6 @@
"no_advertisements_found": "No matching Bluetooth advertisements found", "no_advertisements_found": "No matching Bluetooth advertisements found",
"no_connection_slot_allocations": "No connection slot allocations information available", "no_connection_slot_allocations": "No connection slot allocations information available",
"no_active_connection_support": "This adapter does not support making active (GATT) connections.", "no_active_connection_support": "This adapter does not support making active (GATT) connections.",
"no_scanner_state_available": "No scanner state available",
"current_scanning_mode": "Current scanning mode",
"requested_scanning_mode": "Requested scanning mode",
"scanning_mode_none": "none",
"scanning_mode_active": "active",
"scanning_mode_passive": "passive",
"scanner_mode_mismatch": "Scanner requested {requested} mode but is operating in {current} mode. The scanner is in a bad state and needs to be power cycled.",
"scanner_mode_mismatch_remote": "For proxies: reboot the device",
"scanner_mode_mismatch_usb": "For USB adapters: unplug and plug back in",
"scanner_mode_mismatch_uart": "For UART/onboard adapters: power down the system completely and power it back up",
"address": "Address", "address": "Address",
"name": "Name", "name": "Name",
"source": "Source", "source": "Source",
@@ -7566,8 +7526,8 @@
"square": "Render cards as squares" "square": "Render cards as squares"
}, },
"logbook": { "logbook": {
"name": "Logbook", "name": "Activity",
"description": "The Logbook card shows a list of events for entities." "description": "The Activity card shows a list of events for entities."
}, },
"history-graph": { "history-graph": {
"name": "History graph", "name": "History graph",
@@ -8208,8 +8168,8 @@
"bar-gauge": { "bar-gauge": {
"label": "Bar gauge" "label": "Bar gauge"
}, },
"trend-graph": { "history-chart": {
"label": "Trend graph" "label": "History chart"
} }
} }
}, },

View File

@@ -1,124 +0,0 @@
import type { HassEntity } from "home-assistant-js-websocket";
import { describe, expect, it } from "vitest";
import { deviceTrackerIcon } from "../../../src/common/entity/device_tracker_icon";
describe("deviceTrackerIcon", () => {
const createMockStateObj = (
source_type: string,
state = "home"
): HassEntity => ({
entity_id: "device_tracker.test",
state,
attributes: { source_type },
context: { id: "test", parent_id: null, user_id: null },
last_changed: "2023-01-01T00:00:00Z",
last_updated: "2023-01-01T00:00:00Z",
});
describe("router source type", () => {
it("should return lan-connect icon when home", () => {
const stateObj = createMockStateObj("router", "home");
expect(deviceTrackerIcon(stateObj)).toBe("mdi:lan-connect");
});
it("should return lan-disconnect icon when not home", () => {
const stateObj = createMockStateObj("router", "not_home");
expect(deviceTrackerIcon(stateObj)).toBe("mdi:lan-disconnect");
});
it("should return lan-disconnect icon for any other state", () => {
const stateObj = createMockStateObj("router", "office");
expect(deviceTrackerIcon(stateObj)).toBe("mdi:lan-disconnect");
});
it("should use explicit state parameter over state object state", () => {
const stateObj = createMockStateObj("router", "not_home");
expect(deviceTrackerIcon(stateObj, "home")).toBe("mdi:lan-connect");
});
});
describe("bluetooth source type", () => {
it("should return bluetooth-connect icon when home for bluetooth", () => {
const stateObj = createMockStateObj("bluetooth", "home");
expect(deviceTrackerIcon(stateObj)).toBe("mdi:bluetooth-connect");
});
it("should return bluetooth icon when not home for bluetooth", () => {
const stateObj = createMockStateObj("bluetooth", "not_home");
expect(deviceTrackerIcon(stateObj)).toBe("mdi:bluetooth");
});
it("should return bluetooth-connect icon when home for bluetooth_le", () => {
const stateObj = createMockStateObj("bluetooth_le", "home");
expect(deviceTrackerIcon(stateObj)).toBe("mdi:bluetooth-connect");
});
it("should return bluetooth icon when not home for bluetooth_le", () => {
const stateObj = createMockStateObj("bluetooth_le", "not_home");
expect(deviceTrackerIcon(stateObj)).toBe("mdi:bluetooth");
});
it("should use explicit state parameter for bluetooth", () => {
const stateObj = createMockStateObj("bluetooth", "not_home");
expect(deviceTrackerIcon(stateObj, "home")).toBe("mdi:bluetooth-connect");
});
});
describe("other source types", () => {
it("should return account icon when home for gps", () => {
const stateObj = createMockStateObj("gps", "home");
expect(deviceTrackerIcon(stateObj)).toBe("mdi:account");
});
it("should return account-arrow-right icon when not home for gps", () => {
const stateObj = createMockStateObj("gps", "not_home");
expect(deviceTrackerIcon(stateObj)).toBe("mdi:account-arrow-right");
});
it("should return account icon for unknown location with gps", () => {
const stateObj = createMockStateObj("gps", "office");
expect(deviceTrackerIcon(stateObj)).toBe("mdi:account");
});
it("should handle unknown source type", () => {
const stateObj = createMockStateObj("unknown", "home");
expect(deviceTrackerIcon(stateObj)).toBe("mdi:account");
});
it("should handle unknown source type when not home", () => {
const stateObj = createMockStateObj("unknown", "not_home");
expect(deviceTrackerIcon(stateObj)).toBe("mdi:account-arrow-right");
});
});
describe("edge cases", () => {
it("should handle missing source_type attribute", () => {
const stateObj: HassEntity = {
entity_id: "device_tracker.test",
state: "home",
attributes: {},
context: { id: "test", parent_id: null, user_id: null },
last_changed: "2023-01-01T00:00:00Z",
last_updated: "2023-01-01T00:00:00Z",
};
expect(deviceTrackerIcon(stateObj)).toBe("mdi:account");
});
it("should handle undefined state object attributes", () => {
const stateObj: HassEntity = {
entity_id: "device_tracker.test",
state: "not_home",
attributes: {},
context: { id: "test", parent_id: null, user_id: null },
last_changed: "2023-01-01T00:00:00Z",
last_updated: "2023-01-01T00:00:00Z",
};
expect(deviceTrackerIcon(stateObj)).toBe("mdi:account-arrow-right");
});
it("should handle empty string state", () => {
const stateObj = createMockStateObj("router", "");
expect(deviceTrackerIcon(stateObj)).toBe("mdi:lan-disconnect");
});
});
});

View File

@@ -1,391 +0,0 @@
import { describe, expect, it } from "vitest";
import { generateEntityFilter } from "../../../src/common/entity/entity_filter";
import type { HomeAssistant } from "../../../src/types";
// Mock HomeAssistant with comprehensive data
const mockHass: HomeAssistant = {
states: {
"light.living_room": {
entity_id: "light.living_room",
state: "on",
attributes: { device_class: "light" },
},
"switch.kitchen": {
entity_id: "switch.kitchen",
state: "off",
attributes: { device_class: "switch" },
},
"sensor.temperature": {
entity_id: "sensor.temperature",
state: "22.5",
attributes: { device_class: "temperature" },
},
"binary_sensor.motion": {
entity_id: "binary_sensor.motion",
state: "off",
attributes: { device_class: "motion" },
},
"climate.thermostat": {
entity_id: "climate.thermostat",
state: "heat",
attributes: {},
},
"media_player.tv": {
entity_id: "media_player.tv",
state: "off",
attributes: {},
},
"light.bedroom": {
entity_id: "light.bedroom",
state: "off",
attributes: { device_class: "light" },
},
"switch.basement": {
entity_id: "switch.basement",
state: "on",
attributes: { device_class: "switch" },
},
"sensor.humidity": {
entity_id: "sensor.humidity",
state: "45",
attributes: { device_class: "humidity", entity_category: "diagnostic" },
},
"light.no_area": {
entity_id: "light.no_area",
state: "off",
attributes: { device_class: "light" },
},
} as any,
entities: {
"light.living_room": {
entity_id: "light.living_room",
device_id: "device1",
area_id: "living_room",
labels: [],
},
"switch.kitchen": {
entity_id: "switch.kitchen",
device_id: "device2",
area_id: "kitchen",
labels: [],
},
"sensor.temperature": {
entity_id: "sensor.temperature",
device_id: "device3",
area_id: "living_room",
labels: [],
},
"binary_sensor.motion": {
entity_id: "binary_sensor.motion",
device_id: "device4",
area_id: "hallway",
labels: [],
},
"climate.thermostat": {
entity_id: "climate.thermostat",
device_id: "device5",
area_id: "living_room",
labels: [],
},
"media_player.tv": {
entity_id: "media_player.tv",
device_id: "device6",
area_id: "living_room",
labels: [],
},
"light.bedroom": {
entity_id: "light.bedroom",
device_id: "device7",
area_id: "bedroom",
labels: [],
},
"switch.basement": {
entity_id: "switch.basement",
device_id: "device8",
area_id: "basement",
labels: [],
},
"sensor.humidity": {
entity_id: "sensor.humidity",
device_id: "device9",
area_id: "living_room",
entity_category: "diagnostic",
labels: ["climate", "monitoring"],
},
"light.no_area": {
entity_id: "light.no_area",
device_id: "device10",
labels: [],
},
} as any,
devices: {
device1: { id: "device1", area_id: "living_room" },
device2: { id: "device2", area_id: "kitchen" },
device3: { id: "device3", area_id: "living_room" },
device4: { id: "device4", area_id: "hallway" },
device5: { id: "device5", area_id: "living_room" },
device6: { id: "device6", area_id: "living_room" },
device7: { id: "device7", area_id: "bedroom" },
device8: { id: "device8", area_id: "basement" },
device9: { id: "device9", area_id: "living_room" },
device10: { id: "device10" }, // no area_id
} as any,
areas: {
living_room: {
area_id: "living_room",
name: "Living Room",
floor_id: "main_floor",
},
kitchen: { area_id: "kitchen", name: "Kitchen", floor_id: "main_floor" },
bedroom: { area_id: "bedroom", name: "Bedroom", floor_id: "upper_floor" },
basement: {
area_id: "basement",
name: "Basement",
floor_id: "basement_floor",
},
hallway: { area_id: "hallway", name: "Hallway", floor_id: "main_floor" },
} as any,
floors: {
main_floor: { floor_id: "main_floor", name: "Main Floor" },
upper_floor: { floor_id: "upper_floor", name: "Upper Floor" },
basement_floor: { floor_id: "basement_floor", name: "Basement Floor" },
} as any,
} as HomeAssistant;
describe("generateEntityFilter", () => {
describe("domain filtering", () => {
it("should filter entities by single domain", () => {
const filter = generateEntityFilter(mockHass, { domain: "light" });
expect(filter("light.living_room")).toBe(true);
expect(filter("switch.kitchen")).toBe(false);
});
it("should filter entities by multiple domains", () => {
const filter = generateEntityFilter(mockHass, {
domain: ["light", "switch"],
});
expect(filter("light.living_room")).toBe(true);
expect(filter("switch.kitchen")).toBe(true);
// Non-existent entities return false
expect(filter("switch.fan")).toBe(false);
expect(filter("sensor.temperature")).toBe(false);
});
it("should handle domain as string vs array", () => {
const singleFilter = generateEntityFilter(mockHass, { domain: "sensor" });
const arrayFilter = generateEntityFilter(mockHass, {
domain: ["sensor"],
});
expect(singleFilter("sensor.temperature")).toBe(true);
expect(arrayFilter("sensor.temperature")).toBe(true);
expect(singleFilter("light.living_room")).toBe(false);
expect(arrayFilter("light.living_room")).toBe(false);
});
});
describe("device class filtering", () => {
it("should filter entities by single device class", () => {
const filter = generateEntityFilter(mockHass, {
device_class: "temperature",
});
expect(filter("sensor.temperature")).toBe(true);
expect(filter("sensor.humidity")).toBe(false);
});
it("should filter entities by multiple device classes", () => {
const filter = generateEntityFilter(mockHass, {
device_class: ["temperature", "humidity"],
});
expect(filter("sensor.temperature")).toBe(true);
expect(filter("sensor.humidity")).toBe(true);
expect(filter("light.living_room")).toBe(false);
});
it("should handle entities without device class", () => {
const filter = generateEntityFilter(mockHass, { device_class: "test" });
expect(filter("climate.thermostat")).toBe(false);
expect(filter("media_player.tv")).toBe(false);
});
});
describe("area filtering", () => {
it("should filter entities by single area", () => {
const filter = generateEntityFilter(mockHass, { area: "living_room" });
expect(filter("light.living_room")).toBe(true);
expect(filter("sensor.temperature")).toBe(true);
expect(filter("switch.kitchen")).toBe(false);
});
it("should filter entities by multiple areas", () => {
const filter = generateEntityFilter(mockHass, {
area: ["living_room", "kitchen"],
});
expect(filter("light.living_room")).toBe(true);
expect(filter("switch.kitchen")).toBe(true);
expect(filter("light.bedroom")).toBe(false);
});
});
describe("floor filtering", () => {
// NOTE: The current implementation has a bug where it checks `if (!floors)` instead of `if (!floors.has(floor.floor_id))`
// So floor filtering will never actually filter by floor - it only checks if the entity has a floor at all
it("should filter entities by floor (tests current buggy behavior)", () => {
const filter = generateEntityFilter(mockHass, { floor: "main_floor" });
// Due to bug, all entities with floors pass (not just main_floor)
expect(filter("light.living_room")).toBe(true); // has floor
expect(filter("switch.kitchen")).toBe(true); // has floor
expect(filter("binary_sensor.motion")).toBe(true); // has floor
expect(filter("light.bedroom")).toBe(false); // wrong floor
expect(filter("switch.basement")).toBe(false); // wrong floor
// Entities without floors should fail
expect(filter("light.no_area")).toBe(false); // no area = no floor
});
it("should handle multiple floors (tests current buggy behavior)", () => {
const filter = generateEntityFilter(mockHass, {
floor: ["main_floor", "upper_floor"],
});
expect(filter("light.living_room")).toBe(true);
expect(filter("light.bedroom")).toBe(true);
expect(filter("switch.basement")).toBe(false);
// Entities without floors should fail
expect(filter("light.no_area")).toBe(false);
});
});
describe("device filtering", () => {
it("should filter entities by single device", () => {
const filter = generateEntityFilter(mockHass, { device: "device1" });
expect(filter("light.living_room")).toBe(true);
expect(filter("switch.kitchen")).toBe(false);
});
it("should filter entities by multiple devices", () => {
const filter = generateEntityFilter(mockHass, {
device: ["device1", "device2"],
});
expect(filter("light.living_room")).toBe(true);
expect(filter("switch.kitchen")).toBe(true);
expect(filter("sensor.temperature")).toBe(false);
});
});
describe("entity category filtering", () => {
it("should filter entities by entity category", () => {
const filter = generateEntityFilter(mockHass, {
entity_category: "diagnostic",
});
expect(filter("sensor.humidity")).toBe(true);
expect(filter("sensor.temperature")).toBe(false);
});
it("should filter entities with no entity category", () => {
const filter = generateEntityFilter(mockHass, {
entity_category: "none",
});
expect(filter("light.living_room")).toBe(true);
expect(filter("sensor.humidity")).toBe(false);
});
});
describe("label filtering", () => {
it("should filter entities by single label", () => {
const filter = generateEntityFilter(mockHass, { label: "climate" });
expect(filter("sensor.humidity")).toBe(true);
expect(filter("sensor.temperature")).toBe(false);
});
it("should filter entities by multiple labels", () => {
const filter = generateEntityFilter(mockHass, {
label: ["climate", "monitoring"],
});
expect(filter("sensor.humidity")).toBe(true);
expect(filter("light.living_room")).toBe(false);
});
});
describe("combined filtering", () => {
it("should combine multiple filter criteria with AND logic", () => {
const filter = generateEntityFilter(mockHass, {
domain: "light",
area: "living_room",
});
expect(filter("light.living_room")).toBe(true);
expect(filter("light.bedroom")).toBe(false);
expect(filter("sensor.temperature")).toBe(false);
});
it("should handle complex combinations", () => {
const filter = generateEntityFilter(mockHass, {
domain: ["sensor", "light"],
area: "living_room",
device_class: ["temperature", "light"],
});
expect(filter("sensor.temperature")).toBe(true);
expect(filter("light.living_room")).toBe(true);
expect(filter("sensor.humidity")).toBe(false); // wrong device class
expect(filter("light.bedroom")).toBe(false); // wrong area
});
});
describe("empty filter criteria", () => {
it("should handle empty filter criteria", () => {
const filter = generateEntityFilter(mockHass, {});
// Empty filter should pass all entities that exist in hass.states
expect(filter("light.living_room")).toBe(true);
expect(filter("switch.kitchen")).toBe(true);
expect(filter("nonexistent.entity")).toBe(false);
});
it("should handle empty domain array", () => {
const filter = generateEntityFilter(mockHass, { domain: [] });
// Empty domain array means no entities should pass domain filter
expect(filter("light.living_room")).toBe(false);
expect(filter("switch.kitchen")).toBe(false);
});
});
describe("edge cases", () => {
it("should handle non-existent entities", () => {
const filter = generateEntityFilter(mockHass, { domain: "light" });
expect(filter("light.nonexistent")).toBe(false);
expect(filter("invalid_entity_id")).toBe(false);
});
it("should handle entities without device or area assignments", () => {
const filter = generateEntityFilter(mockHass, { area: "living_room" });
expect(filter("light.no_area")).toBe(false);
});
it("should handle entities with device but no area", () => {
const filter = generateEntityFilter(mockHass, { area: "living_room" });
// light.no_area has device10 which has no area_id
expect(filter("light.no_area")).toBe(false);
});
});
});

336
yarn.lock
View File

@@ -1216,15 +1216,15 @@ __metadata:
languageName: node languageName: node
linkType: hard linkType: hard
"@codemirror/autocomplete@npm:6.18.7": "@codemirror/autocomplete@npm:6.18.6":
version: 6.18.7 version: 6.18.6
resolution: "@codemirror/autocomplete@npm:6.18.7" resolution: "@codemirror/autocomplete@npm:6.18.6"
dependencies: dependencies:
"@codemirror/language": "npm:^6.0.0" "@codemirror/language": "npm:^6.0.0"
"@codemirror/state": "npm:^6.0.0" "@codemirror/state": "npm:^6.0.0"
"@codemirror/view": "npm:^6.17.0" "@codemirror/view": "npm:^6.17.0"
"@lezer/common": "npm:^1.0.0" "@lezer/common": "npm:^1.0.0"
checksum: 10/e50e3345d7d33e762d9abd2e6b1ea4ff54afe1630310464a5ddb42cab52fd5bac783ec0dc8a328cb746be6a7f9f711b6fcd8ef311af123511e8307b4c056cb9d checksum: 10/0574d96fd04ccf2d3b7ae3c4efe0a72f423fa81658876ec50865ce3371cea038aeddf026976ec0d0ccbee72ac66bdf7deec9106dee251ad49019ae7e1a871663
languageName: node languageName: node
linkType: hard linkType: hard
@@ -1283,15 +1283,15 @@ __metadata:
languageName: node languageName: node
linkType: hard linkType: hard
"@codemirror/view@npm:6.38.2, @codemirror/view@npm:^6.0.0, @codemirror/view@npm:^6.17.0, @codemirror/view@npm:^6.23.0, @codemirror/view@npm:^6.27.0": "@codemirror/view@npm:6.38.1, @codemirror/view@npm:^6.0.0, @codemirror/view@npm:^6.17.0, @codemirror/view@npm:^6.23.0, @codemirror/view@npm:^6.27.0":
version: 6.38.2 version: 6.38.1
resolution: "@codemirror/view@npm:6.38.2" resolution: "@codemirror/view@npm:6.38.1"
dependencies: dependencies:
"@codemirror/state": "npm:^6.5.0" "@codemirror/state": "npm:^6.5.0"
crelt: "npm:^1.0.6" crelt: "npm:^1.0.6"
style-mod: "npm:^4.1.0" style-mod: "npm:^4.1.0"
w3c-keyname: "npm:^2.2.4" w3c-keyname: "npm:^2.2.4"
checksum: 10/300608850a29215d7b47fe8ade183fc2241457a924335bd127e29e1af11da9314369c65ec0da968177086f3529abbcd71a609c1af673ea8951c32a523cab358c checksum: 10/e0c5a365608749dd096ba7a930c8393f316bf4c2cacd1465a47a057d0a9f9868ff372a0bb6eb696c926f88411139f79a97a05f8c884bcc380145445cc61e68c8
languageName: node languageName: node
linkType: hard linkType: hard
@@ -3990,92 +3990,92 @@ __metadata:
languageName: node languageName: node
linkType: hard linkType: hard
"@rspack/binding-darwin-arm64@npm:1.5.2": "@rspack/binding-darwin-arm64@npm:1.5.1":
version: 1.5.2 version: 1.5.1
resolution: "@rspack/binding-darwin-arm64@npm:1.5.2" resolution: "@rspack/binding-darwin-arm64@npm:1.5.1"
conditions: os=darwin & cpu=arm64 conditions: os=darwin & cpu=arm64
languageName: node languageName: node
linkType: hard linkType: hard
"@rspack/binding-darwin-x64@npm:1.5.2": "@rspack/binding-darwin-x64@npm:1.5.1":
version: 1.5.2 version: 1.5.1
resolution: "@rspack/binding-darwin-x64@npm:1.5.2" resolution: "@rspack/binding-darwin-x64@npm:1.5.1"
conditions: os=darwin & cpu=x64 conditions: os=darwin & cpu=x64
languageName: node languageName: node
linkType: hard linkType: hard
"@rspack/binding-linux-arm64-gnu@npm:1.5.2": "@rspack/binding-linux-arm64-gnu@npm:1.5.1":
version: 1.5.2 version: 1.5.1
resolution: "@rspack/binding-linux-arm64-gnu@npm:1.5.2" resolution: "@rspack/binding-linux-arm64-gnu@npm:1.5.1"
conditions: os=linux & cpu=arm64 & libc=glibc conditions: os=linux & cpu=arm64 & libc=glibc
languageName: node languageName: node
linkType: hard linkType: hard
"@rspack/binding-linux-arm64-musl@npm:1.5.2": "@rspack/binding-linux-arm64-musl@npm:1.5.1":
version: 1.5.2 version: 1.5.1
resolution: "@rspack/binding-linux-arm64-musl@npm:1.5.2" resolution: "@rspack/binding-linux-arm64-musl@npm:1.5.1"
conditions: os=linux & cpu=arm64 & libc=musl conditions: os=linux & cpu=arm64 & libc=musl
languageName: node languageName: node
linkType: hard linkType: hard
"@rspack/binding-linux-x64-gnu@npm:1.5.2": "@rspack/binding-linux-x64-gnu@npm:1.5.1":
version: 1.5.2 version: 1.5.1
resolution: "@rspack/binding-linux-x64-gnu@npm:1.5.2" resolution: "@rspack/binding-linux-x64-gnu@npm:1.5.1"
conditions: os=linux & cpu=x64 & libc=glibc conditions: os=linux & cpu=x64 & libc=glibc
languageName: node languageName: node
linkType: hard linkType: hard
"@rspack/binding-linux-x64-musl@npm:1.5.2": "@rspack/binding-linux-x64-musl@npm:1.5.1":
version: 1.5.2 version: 1.5.1
resolution: "@rspack/binding-linux-x64-musl@npm:1.5.2" resolution: "@rspack/binding-linux-x64-musl@npm:1.5.1"
conditions: os=linux & cpu=x64 & libc=musl conditions: os=linux & cpu=x64 & libc=musl
languageName: node languageName: node
linkType: hard linkType: hard
"@rspack/binding-wasm32-wasi@npm:1.5.2": "@rspack/binding-wasm32-wasi@npm:1.5.1":
version: 1.5.2 version: 1.5.1
resolution: "@rspack/binding-wasm32-wasi@npm:1.5.2" resolution: "@rspack/binding-wasm32-wasi@npm:1.5.1"
dependencies: dependencies:
"@napi-rs/wasm-runtime": "npm:^1.0.1" "@napi-rs/wasm-runtime": "npm:^1.0.1"
conditions: cpu=wasm32 conditions: cpu=wasm32
languageName: node languageName: node
linkType: hard linkType: hard
"@rspack/binding-win32-arm64-msvc@npm:1.5.2": "@rspack/binding-win32-arm64-msvc@npm:1.5.1":
version: 1.5.2 version: 1.5.1
resolution: "@rspack/binding-win32-arm64-msvc@npm:1.5.2" resolution: "@rspack/binding-win32-arm64-msvc@npm:1.5.1"
conditions: os=win32 & cpu=arm64 conditions: os=win32 & cpu=arm64
languageName: node languageName: node
linkType: hard linkType: hard
"@rspack/binding-win32-ia32-msvc@npm:1.5.2": "@rspack/binding-win32-ia32-msvc@npm:1.5.1":
version: 1.5.2 version: 1.5.1
resolution: "@rspack/binding-win32-ia32-msvc@npm:1.5.2" resolution: "@rspack/binding-win32-ia32-msvc@npm:1.5.1"
conditions: os=win32 & cpu=ia32 conditions: os=win32 & cpu=ia32
languageName: node languageName: node
linkType: hard linkType: hard
"@rspack/binding-win32-x64-msvc@npm:1.5.2": "@rspack/binding-win32-x64-msvc@npm:1.5.1":
version: 1.5.2 version: 1.5.1
resolution: "@rspack/binding-win32-x64-msvc@npm:1.5.2" resolution: "@rspack/binding-win32-x64-msvc@npm:1.5.1"
conditions: os=win32 & cpu=x64 conditions: os=win32 & cpu=x64
languageName: node languageName: node
linkType: hard linkType: hard
"@rspack/binding@npm:1.5.2": "@rspack/binding@npm:1.5.1":
version: 1.5.2 version: 1.5.1
resolution: "@rspack/binding@npm:1.5.2" resolution: "@rspack/binding@npm:1.5.1"
dependencies: dependencies:
"@rspack/binding-darwin-arm64": "npm:1.5.2" "@rspack/binding-darwin-arm64": "npm:1.5.1"
"@rspack/binding-darwin-x64": "npm:1.5.2" "@rspack/binding-darwin-x64": "npm:1.5.1"
"@rspack/binding-linux-arm64-gnu": "npm:1.5.2" "@rspack/binding-linux-arm64-gnu": "npm:1.5.1"
"@rspack/binding-linux-arm64-musl": "npm:1.5.2" "@rspack/binding-linux-arm64-musl": "npm:1.5.1"
"@rspack/binding-linux-x64-gnu": "npm:1.5.2" "@rspack/binding-linux-x64-gnu": "npm:1.5.1"
"@rspack/binding-linux-x64-musl": "npm:1.5.2" "@rspack/binding-linux-x64-musl": "npm:1.5.1"
"@rspack/binding-wasm32-wasi": "npm:1.5.2" "@rspack/binding-wasm32-wasi": "npm:1.5.1"
"@rspack/binding-win32-arm64-msvc": "npm:1.5.2" "@rspack/binding-win32-arm64-msvc": "npm:1.5.1"
"@rspack/binding-win32-ia32-msvc": "npm:1.5.2" "@rspack/binding-win32-ia32-msvc": "npm:1.5.1"
"@rspack/binding-win32-x64-msvc": "npm:1.5.2" "@rspack/binding-win32-x64-msvc": "npm:1.5.1"
dependenciesMeta: dependenciesMeta:
"@rspack/binding-darwin-arm64": "@rspack/binding-darwin-arm64":
optional: true optional: true
@@ -4097,23 +4097,23 @@ __metadata:
optional: true optional: true
"@rspack/binding-win32-x64-msvc": "@rspack/binding-win32-x64-msvc":
optional: true optional: true
checksum: 10/71c41c6c878445ea561b7a02d9f75ec13ce170f5d63053debd72dee82a07d23c491a55526cfe9e0aceb5ee1154a07bbe69121deb2821d1a3ac5021eea75d9114 checksum: 10/a6756a35bda55fd9e21b1ce142ca18e228d92832dc213027a19314981f8f12e6510dd862a9724ee96dee61755b3dd30ce73b2bb117d150e9f5ce73ba8fe4b57a
languageName: node languageName: node
linkType: hard linkType: hard
"@rspack/core@npm:1.5.2": "@rspack/core@npm:1.5.1":
version: 1.5.2 version: 1.5.1
resolution: "@rspack/core@npm:1.5.2" resolution: "@rspack/core@npm:1.5.1"
dependencies: dependencies:
"@module-federation/runtime-tools": "npm:0.18.0" "@module-federation/runtime-tools": "npm:0.18.0"
"@rspack/binding": "npm:1.5.2" "@rspack/binding": "npm:1.5.1"
"@rspack/lite-tapable": "npm:1.0.1" "@rspack/lite-tapable": "npm:1.0.1"
peerDependencies: peerDependencies:
"@swc/helpers": ">=0.5.1" "@swc/helpers": ">=0.5.1"
peerDependenciesMeta: peerDependenciesMeta:
"@swc/helpers": "@swc/helpers":
optional: true optional: true
checksum: 10/e72023c8eea0ed351d950a28b6897ca7143ad749a65380ab855e12f96f8ce692ab044c14acf9b030bca740b722c197ad3075eaadac4fe480389e3131c519ac0e checksum: 10/b7a6269d5bdbcad140d172ebe951f4693711573d4f38e4c676c250a9cc6c1bdf602ad5187eeacc07ff12b74d510b746c92e3f112c8ab4dca46846c595d2876b0
languageName: node languageName: node
linkType: hard linkType: hard
@@ -4445,10 +4445,10 @@ __metadata:
languageName: node languageName: node
linkType: hard linkType: hard
"@types/chromecast-caf-receiver@npm:6.0.24": "@types/chromecast-caf-receiver@npm:6.0.22":
version: 6.0.24 version: 6.0.22
resolution: "@types/chromecast-caf-receiver@npm:6.0.24" resolution: "@types/chromecast-caf-receiver@npm:6.0.22"
checksum: 10/1f2b95e8a15dbb36d5328895229d4a5cb255b33e62d46335bd6ed75e16aa9ea6a7d765a64ae120d19b3134fb3e51e9547d2544c7277f7bffe0bf0b3999f026da checksum: 10/6c51cb52527776ddfa187a261b88184c98bdd61c129dd8719cba213894d565cf69073734d6473696ffd60a768f6fb5a3fe9932693f43174fbc5e7af201db8a90
languageName: node languageName: node
linkType: hard linkType: hard
@@ -4496,10 +4496,10 @@ __metadata:
languageName: node languageName: node
linkType: hard linkType: hard
"@types/culori@npm:4.0.1": "@types/culori@npm:4.0.0":
version: 4.0.1 version: 4.0.0
resolution: "@types/culori@npm:4.0.1" resolution: "@types/culori@npm:4.0.0"
checksum: 10/34240fce795cdcbeefbbb4ec1fd6adec1c7edafa949131586176649c21d912236d151bf7af53de161454a6c2fa259a4ddd54f204c66e67e8b9ecfd90b9021c68 checksum: 10/62a9058d6125fe489ca1e7df27ac9837ea7a34c772b8bed8e5e00177b141574830efaa0c93363e9532878490d3245a9c9c8183ebee181a450097584af0cfefc1
languageName: node languageName: node
linkType: hard linkType: hard
@@ -4685,25 +4685,25 @@ __metadata:
languageName: node languageName: node
linkType: hard linkType: hard
"@types/leaflet-draw@npm:1.0.13": "@types/leaflet-draw@npm:1.0.12":
version: 1.0.13 version: 1.0.12
resolution: "@types/leaflet-draw@npm:1.0.13" resolution: "@types/leaflet-draw@npm:1.0.12"
dependencies: dependencies:
"@types/leaflet": "npm:^1.9" "@types/leaflet": "npm:*"
checksum: 10/1a6c3a8b3011f15362108b522fa6d17cca888a7f866e2e582dac921e2c748d9e24685d83b2a5ed1d97e3a1448b0f5b1879a42f1143ffdeb36b71c7fb9b94e9f5 checksum: 10/2a73a152e6a9405502789d7b2d8ffe18d679da03533e17c2a1fe722e78c8ed8cf3daf6a56aae5572c4dd86257811286b5e06b0cb307141a241618f33a360618a
languageName: node languageName: node
linkType: hard linkType: hard
"@types/leaflet.markercluster@npm:1.5.6": "@types/leaflet.markercluster@npm:1.5.5":
version: 1.5.6 version: 1.5.5
resolution: "@types/leaflet.markercluster@npm:1.5.6" resolution: "@types/leaflet.markercluster@npm:1.5.5"
dependencies: dependencies:
"@types/leaflet": "npm:^1.9" "@types/leaflet": "npm:*"
checksum: 10/6ddc628fa6d8a3735f154418115b8b0225fefc74d1d472d0aa987a945a92ed6e0dcc0bcaad5a65d104f38e7445cddbf91d75de97b970b6d173e43afa373d8761 checksum: 10/17647d187ed8c9c38124005c3c45c0c7998c6359d8783e2ea162f9649b151862750c813eba2373054e90156a11a37af2b220429f937b302889b9d6e2105bf2ca
languageName: node languageName: node
linkType: hard linkType: hard
"@types/leaflet@npm:1.9.20, @types/leaflet@npm:^1.9": "@types/leaflet@npm:*, @types/leaflet@npm:1.9.20":
version: 1.9.20 version: 1.9.20
resolution: "@types/leaflet@npm:1.9.20" resolution: "@types/leaflet@npm:1.9.20"
dependencies: dependencies:
@@ -4964,106 +4964,106 @@ __metadata:
languageName: node languageName: node
linkType: hard linkType: hard
"@typescript-eslint/eslint-plugin@npm:8.42.0": "@typescript-eslint/eslint-plugin@npm:8.41.0":
version: 8.42.0 version: 8.41.0
resolution: "@typescript-eslint/eslint-plugin@npm:8.42.0" resolution: "@typescript-eslint/eslint-plugin@npm:8.41.0"
dependencies: dependencies:
"@eslint-community/regexpp": "npm:^4.10.0" "@eslint-community/regexpp": "npm:^4.10.0"
"@typescript-eslint/scope-manager": "npm:8.42.0" "@typescript-eslint/scope-manager": "npm:8.41.0"
"@typescript-eslint/type-utils": "npm:8.42.0" "@typescript-eslint/type-utils": "npm:8.41.0"
"@typescript-eslint/utils": "npm:8.42.0" "@typescript-eslint/utils": "npm:8.41.0"
"@typescript-eslint/visitor-keys": "npm:8.42.0" "@typescript-eslint/visitor-keys": "npm:8.41.0"
graphemer: "npm:^1.4.0" graphemer: "npm:^1.4.0"
ignore: "npm:^7.0.0" ignore: "npm:^7.0.0"
natural-compare: "npm:^1.4.0" natural-compare: "npm:^1.4.0"
ts-api-utils: "npm:^2.1.0" ts-api-utils: "npm:^2.1.0"
peerDependencies: peerDependencies:
"@typescript-eslint/parser": ^8.42.0 "@typescript-eslint/parser": ^8.41.0
eslint: ^8.57.0 || ^9.0.0 eslint: ^8.57.0 || ^9.0.0
typescript: ">=4.8.4 <6.0.0" typescript: ">=4.8.4 <6.0.0"
checksum: 10/fb5b0e0785f9fa9d5ef88e78ff189334b2d1c558efd7b5063508d50275224a8aa38d4af0478228b90d6be6620289384a8d814f05e0af8c952c204515c0f3514e checksum: 10/b96e3fd9e8ae2c289aa7f1c0d2fbf89c608d37f54162a893bac5895318b05d21d3fd456cf7a6adf165915a8212f773f1bae9b4d83f732441864f6d92d083ed99
languageName: node languageName: node
linkType: hard linkType: hard
"@typescript-eslint/parser@npm:8.42.0": "@typescript-eslint/parser@npm:8.41.0":
version: 8.42.0 version: 8.41.0
resolution: "@typescript-eslint/parser@npm:8.42.0" resolution: "@typescript-eslint/parser@npm:8.41.0"
dependencies: dependencies:
"@typescript-eslint/scope-manager": "npm:8.42.0" "@typescript-eslint/scope-manager": "npm:8.41.0"
"@typescript-eslint/types": "npm:8.42.0" "@typescript-eslint/types": "npm:8.41.0"
"@typescript-eslint/typescript-estree": "npm:8.42.0" "@typescript-eslint/typescript-estree": "npm:8.41.0"
"@typescript-eslint/visitor-keys": "npm:8.42.0" "@typescript-eslint/visitor-keys": "npm:8.41.0"
debug: "npm:^4.3.4" debug: "npm:^4.3.4"
peerDependencies: peerDependencies:
eslint: ^8.57.0 || ^9.0.0 eslint: ^8.57.0 || ^9.0.0
typescript: ">=4.8.4 <6.0.0" typescript: ">=4.8.4 <6.0.0"
checksum: 10/25eb2d08c118742dc01c2aa279ea4ba2d277e2d9a042ffd4f9bda9e94d7ff2aa90b63aad1204a82617a5c63ddd3dd553d927944cd9c8345826484d0d523cf7ad checksum: 10/d4ba418aa62e08d49a5b953c9debd52674c30b9b2bb7bf2efc173a22ad3942df72cd83072beac06d98dad82741baf502a55fc648925ca407b01abdc908675f67
languageName: node languageName: node
linkType: hard linkType: hard
"@typescript-eslint/project-service@npm:8.42.0": "@typescript-eslint/project-service@npm:8.41.0":
version: 8.42.0 version: 8.41.0
resolution: "@typescript-eslint/project-service@npm:8.42.0" resolution: "@typescript-eslint/project-service@npm:8.41.0"
dependencies: dependencies:
"@typescript-eslint/tsconfig-utils": "npm:^8.42.0" "@typescript-eslint/tsconfig-utils": "npm:^8.41.0"
"@typescript-eslint/types": "npm:^8.42.0" "@typescript-eslint/types": "npm:^8.41.0"
debug: "npm:^4.3.4" debug: "npm:^4.3.4"
peerDependencies: peerDependencies:
typescript: ">=4.8.4 <6.0.0" typescript: ">=4.8.4 <6.0.0"
checksum: 10/3e91fd4b4d60edd6fe3e108e8e75947de8aa060aab1de63c23017e8afeca72ef405faa6fcdd17e8aa0023261a81135d095072dc31343c57395e50450258d9fa5 checksum: 10/ff8315de005ea7072ecd208b50b35fa01db034f110f30f415faa9c9441648494e5322723a0a4267beb28524babd6b04b349c32f2a2821f4ae0e9c4d503e1e8f0
languageName: node languageName: node
linkType: hard linkType: hard
"@typescript-eslint/scope-manager@npm:8.42.0": "@typescript-eslint/scope-manager@npm:8.41.0":
version: 8.42.0 version: 8.41.0
resolution: "@typescript-eslint/scope-manager@npm:8.42.0" resolution: "@typescript-eslint/scope-manager@npm:8.41.0"
dependencies: dependencies:
"@typescript-eslint/types": "npm:8.42.0" "@typescript-eslint/types": "npm:8.41.0"
"@typescript-eslint/visitor-keys": "npm:8.42.0" "@typescript-eslint/visitor-keys": "npm:8.41.0"
checksum: 10/81be2d908a9d2d83bc9fe5e9219b04277b9fa466bfa7faf45dc076e4b33b39db2fb99b34b8832e329c7db48ddfdc7b78f6c92b564cd6eec99e124d3feaad8645 checksum: 10/4fc1dd6b3390d3a770c228dac227f35ff1126034fce484ab5e5a4fdbe2dab5dca1c8de3c528708320fee021adec1a1260ee45ed2aef9f7e3fdfbb1faf2191f9f
languageName: node languageName: node
linkType: hard linkType: hard
"@typescript-eslint/tsconfig-utils@npm:8.42.0, @typescript-eslint/tsconfig-utils@npm:^8.42.0": "@typescript-eslint/tsconfig-utils@npm:8.41.0, @typescript-eslint/tsconfig-utils@npm:^8.41.0":
version: 8.42.0 version: 8.41.0
resolution: "@typescript-eslint/tsconfig-utils@npm:8.42.0" resolution: "@typescript-eslint/tsconfig-utils@npm:8.41.0"
peerDependencies: peerDependencies:
typescript: ">=4.8.4 <6.0.0" typescript: ">=4.8.4 <6.0.0"
checksum: 10/927aa127983a62ddcbfbcd18806fd278e0bf18fade3cca658946f9ff4915e6a5c5cc85926afaa490512c88dd2950b2059f22b50b6d1f4461c9dbd755a4c71c1c checksum: 10/522d54252f9647d22e46f963df6bafe98aa0572b021e6acf7474c40f1a68afa6753f23a0a125abb1d792a89a1b1cc654d918553a03d08f769139f2f40b0d026c
languageName: node languageName: node
linkType: hard linkType: hard
"@typescript-eslint/type-utils@npm:8.42.0": "@typescript-eslint/type-utils@npm:8.41.0":
version: 8.42.0 version: 8.41.0
resolution: "@typescript-eslint/type-utils@npm:8.42.0" resolution: "@typescript-eslint/type-utils@npm:8.41.0"
dependencies: dependencies:
"@typescript-eslint/types": "npm:8.42.0" "@typescript-eslint/types": "npm:8.41.0"
"@typescript-eslint/typescript-estree": "npm:8.42.0" "@typescript-eslint/typescript-estree": "npm:8.41.0"
"@typescript-eslint/utils": "npm:8.42.0" "@typescript-eslint/utils": "npm:8.41.0"
debug: "npm:^4.3.4" debug: "npm:^4.3.4"
ts-api-utils: "npm:^2.1.0" ts-api-utils: "npm:^2.1.0"
peerDependencies: peerDependencies:
eslint: ^8.57.0 || ^9.0.0 eslint: ^8.57.0 || ^9.0.0
typescript: ">=4.8.4 <6.0.0" typescript: ">=4.8.4 <6.0.0"
checksum: 10/8d876bbd23c956b604d973c49720060c251f4d8cab255f1fd04826a9a1e3ab7c1310400d49d9ec6cdac3288d7a23cd9fb48d42777651ba53c02b5e1a34efd6e9 checksum: 10/6c4c693c1ee3d1a1a3635898d59f1a3bcdf224be84284ea95a21fa68a3206bae32ce04d371df366fcad250a3eca3af723ed6ca1b4aefba238d4e553797c2dc9d
languageName: node languageName: node
linkType: hard linkType: hard
"@typescript-eslint/types@npm:8.42.0, @typescript-eslint/types@npm:^8.42.0": "@typescript-eslint/types@npm:8.41.0, @typescript-eslint/types@npm:^8.41.0":
version: 8.42.0 version: 8.41.0
resolution: "@typescript-eslint/types@npm:8.42.0" resolution: "@typescript-eslint/types@npm:8.41.0"
checksum: 10/7c39a35e5bb7083070872edc797ea60a3d6ceff0e3bdf85701919b71da83a51963562053a4b35c9e2a2b08c138fb595e14bc0b5c450e671a26059b58f8d8b4f4 checksum: 10/e2fe5d9125264a1b1310fff7ac65e827da9885219d7f910dba090dcf7d4242830cb96695c7257634b22e1947943a2e890f9740536d95612452e5752385ab6a5b
languageName: node languageName: node
linkType: hard linkType: hard
"@typescript-eslint/typescript-estree@npm:8.42.0": "@typescript-eslint/typescript-estree@npm:8.41.0":
version: 8.42.0 version: 8.41.0
resolution: "@typescript-eslint/typescript-estree@npm:8.42.0" resolution: "@typescript-eslint/typescript-estree@npm:8.41.0"
dependencies: dependencies:
"@typescript-eslint/project-service": "npm:8.42.0" "@typescript-eslint/project-service": "npm:8.41.0"
"@typescript-eslint/tsconfig-utils": "npm:8.42.0" "@typescript-eslint/tsconfig-utils": "npm:8.41.0"
"@typescript-eslint/types": "npm:8.42.0" "@typescript-eslint/types": "npm:8.41.0"
"@typescript-eslint/visitor-keys": "npm:8.42.0" "@typescript-eslint/visitor-keys": "npm:8.41.0"
debug: "npm:^4.3.4" debug: "npm:^4.3.4"
fast-glob: "npm:^3.3.2" fast-glob: "npm:^3.3.2"
is-glob: "npm:^4.0.3" is-glob: "npm:^4.0.3"
@@ -5072,32 +5072,32 @@ __metadata:
ts-api-utils: "npm:^2.1.0" ts-api-utils: "npm:^2.1.0"
peerDependencies: peerDependencies:
typescript: ">=4.8.4 <6.0.0" typescript: ">=4.8.4 <6.0.0"
checksum: 10/9bb5df97a2ac31e6e3ee6941e10702498a76d23235ba28a23d93e09aa75a2cbcd40dc74935d86706c8e2e55e1a8b6a34bb9fb234461920ed3d8a5abed68ba36b checksum: 10/e039815d2ee03727fadb32c460e0c7df71a35b6c93a87e019c63836c53e51ce41f1975b32c9e5bcc840f4cd49c7bf7715c95df149f915379ec4c559d02436623
languageName: node languageName: node
linkType: hard linkType: hard
"@typescript-eslint/utils@npm:8.42.0": "@typescript-eslint/utils@npm:8.41.0":
version: 8.42.0 version: 8.41.0
resolution: "@typescript-eslint/utils@npm:8.42.0" resolution: "@typescript-eslint/utils@npm:8.41.0"
dependencies: dependencies:
"@eslint-community/eslint-utils": "npm:^4.7.0" "@eslint-community/eslint-utils": "npm:^4.7.0"
"@typescript-eslint/scope-manager": "npm:8.42.0" "@typescript-eslint/scope-manager": "npm:8.41.0"
"@typescript-eslint/types": "npm:8.42.0" "@typescript-eslint/types": "npm:8.41.0"
"@typescript-eslint/typescript-estree": "npm:8.42.0" "@typescript-eslint/typescript-estree": "npm:8.41.0"
peerDependencies: peerDependencies:
eslint: ^8.57.0 || ^9.0.0 eslint: ^8.57.0 || ^9.0.0
typescript: ">=4.8.4 <6.0.0" typescript: ">=4.8.4 <6.0.0"
checksum: 10/41c6c0d01c414c94d7109e21deee73b416547b3be26240d0237a3004c6198f146afefc75feee5333bc957ece6a0856518750655e794fd68c96feec1001edbfe8 checksum: 10/863565c0891d89ee27497571092783a7fa90e281a7643f1bda5d9e8b94aea2acbc851e81141ce7a53ddea3638a0527ea165801dd9611f5532940e4d413c955a8
languageName: node languageName: node
linkType: hard linkType: hard
"@typescript-eslint/visitor-keys@npm:8.42.0": "@typescript-eslint/visitor-keys@npm:8.41.0":
version: 8.42.0 version: 8.41.0
resolution: "@typescript-eslint/visitor-keys@npm:8.42.0" resolution: "@typescript-eslint/visitor-keys@npm:8.41.0"
dependencies: dependencies:
"@typescript-eslint/types": "npm:8.42.0" "@typescript-eslint/types": "npm:8.41.0"
eslint-visitor-keys: "npm:^4.2.1" eslint-visitor-keys: "npm:^4.2.1"
checksum: 10/ef3aeabf7b01eb72e176053a4fe7a4c4f0769a9f58d1f7a920c97d365305b950c402ad34227209781996ae187652ccf0f47c31015f992c502b5fa898a9d44bd5 checksum: 10/3c764be2f0d3b212c2cb7d0cc8a7b0ed378feb58883654471fd8ee943f1e124c0b78df92fe14368ceb46016b0e3ae1c47e2630ec3599aa7b4bd54f7793747657
languageName: node languageName: node
linkType: hard linkType: hard
@@ -6561,7 +6561,7 @@ __metadata:
languageName: node languageName: node
linkType: hard linkType: hard
"chalk@npm:^5.0.1, chalk@npm:^5.6.0": "chalk@npm:^5.0.1, chalk@npm:^5.5.0":
version: 5.6.0 version: 5.6.0
resolution: "chalk@npm:5.6.0" resolution: "chalk@npm:5.6.0"
checksum: 10/f0e0646a72adbd0f6e73441d3872d7f2f40ba98052924f08a30c10634ec6b1e2cd19cc3c40cc21081dad640e2a1a2749030418571690b89bd7782babf7f89866 checksum: 10/f0e0646a72adbd0f6e73441d3872d7f2f40ba98052924f08a30c10634ec6b1e2cd19cc3c40cc21081dad640e2a1a2749030418571690b89bd7782babf7f89866
@@ -9317,13 +9317,13 @@ __metadata:
"@babel/runtime": "npm:7.28.3" "@babel/runtime": "npm:7.28.3"
"@braintree/sanitize-url": "npm:7.1.1" "@braintree/sanitize-url": "npm:7.1.1"
"@bundle-stats/plugin-webpack-filter": "npm:4.21.3" "@bundle-stats/plugin-webpack-filter": "npm:4.21.3"
"@codemirror/autocomplete": "npm:6.18.7" "@codemirror/autocomplete": "npm:6.18.6"
"@codemirror/commands": "npm:6.8.1" "@codemirror/commands": "npm:6.8.1"
"@codemirror/language": "npm:6.11.3" "@codemirror/language": "npm:6.11.3"
"@codemirror/legacy-modes": "npm:6.5.1" "@codemirror/legacy-modes": "npm:6.5.1"
"@codemirror/search": "npm:6.5.11" "@codemirror/search": "npm:6.5.11"
"@codemirror/state": "npm:6.5.2" "@codemirror/state": "npm:6.5.2"
"@codemirror/view": "npm:6.38.2" "@codemirror/view": "npm:6.38.1"
"@egjs/hammerjs": "npm:2.0.17" "@egjs/hammerjs": "npm:2.0.17"
"@formatjs/intl-datetimeformat": "npm:6.18.0" "@formatjs/intl-datetimeformat": "npm:6.18.0"
"@formatjs/intl-displaynames": "npm:6.8.11" "@formatjs/intl-displaynames": "npm:6.8.11"
@@ -9377,7 +9377,7 @@ __metadata:
"@octokit/rest": "npm:22.0.0" "@octokit/rest": "npm:22.0.0"
"@replit/codemirror-indentation-markers": "npm:6.5.3" "@replit/codemirror-indentation-markers": "npm:6.5.3"
"@rsdoctor/rspack-plugin": "npm:1.2.3" "@rsdoctor/rspack-plugin": "npm:1.2.3"
"@rspack/core": "npm:1.5.2" "@rspack/core": "npm:1.5.1"
"@rspack/dev-server": "npm:1.1.4" "@rspack/dev-server": "npm:1.1.4"
"@shoelace-style/shoelace": "npm:2.20.1" "@shoelace-style/shoelace": "npm:2.20.1"
"@swc/helpers": "npm:0.5.17" "@swc/helpers": "npm:0.5.17"
@@ -9385,15 +9385,15 @@ __metadata:
"@tsparticles/engine": "npm:3.9.1" "@tsparticles/engine": "npm:3.9.1"
"@tsparticles/preset-links": "npm:3.2.0" "@tsparticles/preset-links": "npm:3.2.0"
"@types/babel__plugin-transform-runtime": "npm:7.9.5" "@types/babel__plugin-transform-runtime": "npm:7.9.5"
"@types/chromecast-caf-receiver": "npm:6.0.24" "@types/chromecast-caf-receiver": "npm:6.0.22"
"@types/chromecast-caf-sender": "npm:1.0.11" "@types/chromecast-caf-sender": "npm:1.0.11"
"@types/color-name": "npm:2.0.0" "@types/color-name": "npm:2.0.0"
"@types/culori": "npm:4.0.1" "@types/culori": "npm:4.0.0"
"@types/html-minifier-terser": "npm:7.0.2" "@types/html-minifier-terser": "npm:7.0.2"
"@types/js-yaml": "npm:4.0.9" "@types/js-yaml": "npm:4.0.9"
"@types/leaflet": "npm:1.9.20" "@types/leaflet": "npm:1.9.20"
"@types/leaflet-draw": "npm:1.0.13" "@types/leaflet-draw": "npm:1.0.12"
"@types/leaflet.markercluster": "npm:1.5.6" "@types/leaflet.markercluster": "npm:1.5.5"
"@types/lodash.merge": "npm:4.6.9" "@types/lodash.merge": "npm:4.6.9"
"@types/luxon": "npm:3.7.1" "@types/luxon": "npm:3.7.1"
"@types/mocha": "npm:10.0.10" "@types/mocha": "npm:10.0.10"
@@ -9458,7 +9458,7 @@ __metadata:
leaflet: "npm:1.9.4" leaflet: "npm:1.9.4"
leaflet-draw: "patch:leaflet-draw@npm%3A1.0.4#./.yarn/patches/leaflet-draw-npm-1.0.4-0ca0ebcf65.patch" leaflet-draw: "patch:leaflet-draw@npm%3A1.0.4#./.yarn/patches/leaflet-draw-npm-1.0.4-0ca0ebcf65.patch"
leaflet.markercluster: "npm:1.5.3" leaflet.markercluster: "npm:1.5.3"
lint-staged: "npm:16.1.6" lint-staged: "npm:16.1.5"
lit: "npm:3.3.1" lit: "npm:3.3.1"
lit-analyzer: "npm:2.0.3" lit-analyzer: "npm:2.0.3"
lit-html: "npm:3.3.1" lit-html: "npm:3.3.1"
@@ -9488,7 +9488,7 @@ __metadata:
tinykeys: "npm:3.0.0" tinykeys: "npm:3.0.0"
ts-lit-plugin: "npm:2.0.2" ts-lit-plugin: "npm:2.0.2"
typescript: "npm:5.9.2" typescript: "npm:5.9.2"
typescript-eslint: "npm:8.42.0" typescript-eslint: "npm:8.41.0"
ua-parser-js: "npm:2.0.4" ua-parser-js: "npm:2.0.4"
vite-tsconfig-paths: "npm:5.1.4" vite-tsconfig-paths: "npm:5.1.4"
vitest: "npm:3.2.4" vitest: "npm:3.2.4"
@@ -10826,15 +10826,15 @@ __metadata:
languageName: node languageName: node
linkType: hard linkType: hard
"lint-staged@npm:16.1.6": "lint-staged@npm:16.1.5":
version: 16.1.6 version: 16.1.5
resolution: "lint-staged@npm:16.1.6" resolution: "lint-staged@npm:16.1.5"
dependencies: dependencies:
chalk: "npm:^5.6.0" chalk: "npm:^5.5.0"
commander: "npm:^14.0.0" commander: "npm:^14.0.0"
debug: "npm:^4.4.1" debug: "npm:^4.4.1"
lilconfig: "npm:^3.1.3" lilconfig: "npm:^3.1.3"
listr2: "npm:^9.0.3" listr2: "npm:^9.0.1"
micromatch: "npm:^4.0.8" micromatch: "npm:^4.0.8"
nano-spawn: "npm:^1.0.2" nano-spawn: "npm:^1.0.2"
pidtree: "npm:^0.6.0" pidtree: "npm:^0.6.0"
@@ -10842,13 +10842,13 @@ __metadata:
yaml: "npm:^2.8.1" yaml: "npm:^2.8.1"
bin: bin:
lint-staged: bin/lint-staged.js lint-staged: bin/lint-staged.js
checksum: 10/922b4392ae5d3d56130e4eba706c2fa6151d5da5e21f57ab601b1d6ce9cc635ceb5e4c3dc00e7da83ba8f0cb244b82604469c7ea1470b1e6b6ea0fc12454aa08 checksum: 10/02b284f89d7b8118e1b27b1f2068017ed84407e57a1166463789caa65f3429f206372c483bc37304ce03dcb30bd1dd3e624f0502ae4973d440fe73cdd04e0747
languageName: node languageName: node
linkType: hard linkType: hard
"listr2@npm:^9.0.3": "listr2@npm:^9.0.1":
version: 9.0.3 version: 9.0.1
resolution: "listr2@npm:9.0.3" resolution: "listr2@npm:9.0.1"
dependencies: dependencies:
cli-truncate: "npm:^4.0.0" cli-truncate: "npm:^4.0.0"
colorette: "npm:^2.0.20" colorette: "npm:^2.0.20"
@@ -10856,7 +10856,7 @@ __metadata:
log-update: "npm:^6.1.0" log-update: "npm:^6.1.0"
rfdc: "npm:^1.4.1" rfdc: "npm:^1.4.1"
wrap-ansi: "npm:^9.0.0" wrap-ansi: "npm:^9.0.0"
checksum: 10/8cb7cd1cec0f4360502c14cd54af948f831134811d84d3fd2b38b2fa11ea66ee0b15ca8b00b8088d28d7381031afbe755ee3f46bc2c03c2c96c433f04296bd44 checksum: 10/ac5f98317fe17588d304bb4dce47ea22892f223511948656f588c5ab47b99d5d97ca4b812b4fb1237db8ec8d86a504875d8d6a0bb24c877553537eab44941983
languageName: node languageName: node
linkType: hard linkType: hard
@@ -14527,18 +14527,18 @@ __metadata:
languageName: node languageName: node
linkType: hard linkType: hard
"typescript-eslint@npm:8.42.0": "typescript-eslint@npm:8.41.0":
version: 8.42.0 version: 8.41.0
resolution: "typescript-eslint@npm:8.42.0" resolution: "typescript-eslint@npm:8.41.0"
dependencies: dependencies:
"@typescript-eslint/eslint-plugin": "npm:8.42.0" "@typescript-eslint/eslint-plugin": "npm:8.41.0"
"@typescript-eslint/parser": "npm:8.42.0" "@typescript-eslint/parser": "npm:8.41.0"
"@typescript-eslint/typescript-estree": "npm:8.42.0" "@typescript-eslint/typescript-estree": "npm:8.41.0"
"@typescript-eslint/utils": "npm:8.42.0" "@typescript-eslint/utils": "npm:8.41.0"
peerDependencies: peerDependencies:
eslint: ^8.57.0 || ^9.0.0 eslint: ^8.57.0 || ^9.0.0
typescript: ">=4.8.4 <6.0.0" typescript: ">=4.8.4 <6.0.0"
checksum: 10/7f71501823b2c1e87e89ff00d6d8eb40c7514630dbb6b7b44c4dd830c95709357270763df2d711a8ea7bb0b58bd69534f15b01db4550dc6e745df8fec8f6a3ae checksum: 10/a398a367b3a674bcdb74f060e0b06aacb9e8bd0637079c5079ff66a43a35286098b97d71fca1b81b738c0df840fda4b53aeee03ed0aacef03f9644c61a68960e
languageName: node languageName: node
linkType: hard linkType: hard