Compare commits

..

55 Commits

Author SHA1 Message Date
Paul Bottein
ee2ec00069 Bumped version to 20251001.4 2025-10-11 14:41:59 +02:00
Paul Bottein
0aa2941868 Fix unresolved merge conflict in core style 2025-10-11 14:41:42 +02:00
Paul Bottein
07a5c41fd4 Bumped version to 20251001.3 2025-10-11 14:27:11 +02:00
Paul Bottein
4ad3c553d5 Adjust yarn updates for rc (#27456) 2025-10-11 14:26:10 +02:00
Simon Lamon
d40cc448a5 Adjust yarn updates for rc 2025-10-11 12:08:06 +00:00
Bram Kragten
98d44950f8 Bumped version to 20251001.2 2025-10-10 11:36:46 +02:00
Paul Bottein
8ae9edb1ef Fix ha dialog default size (#27415)
* Don't hardcode width height on mobile for all dialogs

* Don't set min width on desktop
2025-10-10 11:35:45 +02:00
Paul Bottein
84c4396c13 Add tooltip instead of title for 'add' button (#27399) 2025-10-10 11:35:44 +02:00
Bram Kragten
b7815bfd86 Bumped version to 20251001.1 2025-10-10 10:38:26 +02:00
Wendelin
d94fa03411 Fix ha-button keyboard focus (#27437) 2025-10-10 10:38:08 +02:00
Petar Petrov
0a7007ef9e Escape device names in energy dashboard (#27425) 2025-10-10 10:31:09 +02:00
Paul Bottein
dd12136dee Use right variable for content color in tooltip (#27400) 2025-10-10 10:31:08 +02:00
Wendelin
6e2f89fe3d Fix android tap highlight border radius (#27382) 2025-10-10 10:31:07 +02:00
Jan-Philipp Benecke
092085b9af Fix media player more info title calculations (#27360) 2025-10-10 10:31:06 +02:00
Paulus Schoutsen
1c06eb8661 Add ESPHome to discovery sources (#27327) 2025-10-10 10:31:05 +02:00
Jan-Philipp Benecke
c7e87b06b5 Fix formatting of position slider tooltip in media player more info (#27326) 2025-10-10 10:31:04 +02:00
Jan-Philipp Benecke
38c738c199 Add "media_stop" action to media player controls in more info (#27325) 2025-10-10 10:31:03 +02:00
Wendelin
e899587307 Fix mobile ha-dialog height in Browser (#27298)
Enhance dialog responsiveness by adjusting min/max height to use svh units
2025-10-10 10:31:02 +02:00
Jan-Philipp Benecke
c9feb0b75f Add color tokens for slider thumb and indicator (#27295) 2025-10-10 10:31:00 +02:00
Jan-Philipp Benecke
10718c35d1 Make ha-slider not depend on font sizes (#27294) 2025-10-10 10:30:59 +02:00
Wendelin
4dc6a37bad Fix automation editor safe area (#27292) 2025-10-10 10:30:59 +02:00
Jan-Philipp Benecke
ac49fc7aba Support redo on Shift+CMD+Z (#27287)
* Support redo on Shift+CMD+Z

* Update redo shortcut for macOS to CMD+Shift+Z
2025-10-10 10:30:57 +02:00
Paul Bottein
e4f008800b Align bottom sheet border radius with resizable bottom sheet (#27280) 2025-10-10 10:30:56 +02:00
Bram Kragten
dfa77526a2 Bumped version to 20251001.0 2025-10-01 08:58:30 +02:00
Bram Kragten
9a3bd6c613 Fix intl polyfill loading (#27261) 2025-10-01 08:55:19 +02:00
Wendelin
1161de5746 Update WA to fix tab group scrolling (#27255) 2025-10-01 08:51:12 +02:00
Jan-Philipp Benecke
9df8e20391 Use local entity picture if available in media player more info (#27252) 2025-10-01 08:51:11 +02:00
Petar Petrov
11047a9c95 Make "loading next step" look like progress step in config flows (#27234) 2025-10-01 08:51:10 +02:00
Jan-Philipp Benecke
18fa66f61c Adjust media player cover image sizes in more info for smaller screens (#27232)
Adjust media player cover image sizes for smaller screens
2025-10-01 08:51:09 +02:00
Simon Lamon
758a048f34 Set explicit netlify version to fix workflows (#27229)
netlify set explicit version for fix
2025-10-01 08:51:08 +02:00
Jan-Philipp Benecke
ee0fc360b0 Add custom color token for control color (#27227) 2025-10-01 08:51:07 +02:00
Jan-Philipp Benecke
4012f95ec1 Add tooltips for undo/redo in automation & script editors (#27224) 2025-10-01 08:51:06 +02:00
Paul Bottein
9ba36ab7e2 Bumped version to 20250926.0 2025-09-26 15:38:51 +02:00
Paul Bottein
fe7a08a1b0 Don't display negative durations in media player more info (#27212)
Don't display negative value in media player more info
2025-09-26 15:38:21 +02:00
Paul Bottein
87a8f9cedc Fix slider ticks support for number selector (#27211) 2025-09-26 15:38:21 +02:00
Paul Bottein
01df7e20ca Fix try tts dialog max width (#27208) 2025-09-26 15:38:20 +02:00
Jan-Philipp Benecke
d181219522 Refactor media player slider to use slot for position and duration display (#27205)
* Refactor media player slider to use slot for position and duration display

* Fix variable naming
2025-09-26 15:38:19 +02:00
karwosts
6ae24b8135 Add validation issues to energy diagnostic (#27203) 2025-09-26 15:38:18 +02:00
karwosts
8e009f24f9 Add dropdown mode to water heater operation feature (#27201) 2025-09-26 15:38:17 +02:00
Simon Lamon
53031f44ac Fix typos in media player more info (#27198) 2025-09-26 15:38:16 +02:00
Jan-Philipp Benecke
af5a988457 Round seconds in media player more info before formatting (#27196) 2025-09-26 15:38:15 +02:00
Paul Bottein
444123c47e Bumped version to 20250925.1 2025-09-25 17:56:13 +02:00
Paul Bottein
f123d34046 Revert "Update dependency @types/chromecast-caf-receiver to v6.0.24" (#27188) 2025-09-25 17:53:59 +02:00
Paul Bottein
1b40f99f68 Fix storage bar not displayed (#27183) 2025-09-25 17:50:52 +02:00
Paul Bottein
b314b3ed2b Fix analytics switches (#27181) 2025-09-25 17:50:51 +02:00
Paul Bottein
59b8932969 Add icon option to common controls section strategy (#27180) 2025-09-25 17:50:50 +02:00
Wendelin
107af753ec Reduce default tab padding in tab-group (#27173) 2025-09-25 17:50:49 +02:00
Paul Bottein
1f0acb3046 Disabled config badge (#27172)
* Add disabled option for badge

* Add disabled to struct
2025-09-25 17:50:48 +02:00
Paul Bottein
02c845cbc6 Bumped version to 20250925.0 2025-09-25 10:47:41 +02:00
Paul Bottein
628111ed20 Bumped version to 20250924.1 2025-09-25 10:46:44 +02:00
Paul Bottein
e825a9c02f Smooth animation of the sidebar resizing handle (#27166) 2025-09-25 10:46:36 +02:00
Paul Bottein
7a35bddf36 Fix safe padding for bottom sheet and add scroll lock (#27165) 2025-09-25 10:46:35 +02:00
Norbert Rittel
ad69270af8 Use "Add (person)" instead of "New person" / "Create" (#27161)
* Update dialog-person-detail.ts

* Update en.json
2025-09-25 10:46:34 +02:00
Paulus Schoutsen
404edf9483 Avoid invalid entities in common controls (#27158) 2025-09-25 10:46:33 +02:00
Paul Bottein
a166b4e9b6 Do not show error message when action has no response in dev tools (#27156) 2025-09-25 10:46:32 +02:00
52 changed files with 572 additions and 367 deletions

View File

@@ -42,7 +42,7 @@ jobs:
- name: Deploy to Netlify - name: Deploy to Netlify
id: deploy id: deploy
run: | run: |
npx -y netlify-cli deploy --dir=cast/dist --alias dev npx -y netlify-cli@23.7.3 deploy --dir=cast/dist --alias dev
env: env:
NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }}
NETLIFY_SITE_ID: ${{ secrets.NETLIFY_CAST_SITE_ID }} NETLIFY_SITE_ID: ${{ secrets.NETLIFY_CAST_SITE_ID }}
@@ -77,7 +77,7 @@ jobs:
- name: Deploy to Netlify - name: Deploy to Netlify
id: deploy id: deploy
run: | run: |
npx -y netlify-cli deploy --dir=cast/dist --prod npx -y netlify-cli@23.7.3 deploy --dir=cast/dist --prod
env: env:
NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }}
NETLIFY_SITE_ID: ${{ secrets.NETLIFY_CAST_SITE_ID }} NETLIFY_SITE_ID: ${{ secrets.NETLIFY_CAST_SITE_ID }}

View File

@@ -43,7 +43,7 @@ jobs:
- name: Deploy to Netlify - name: Deploy to Netlify
id: deploy id: deploy
run: | run: |
npx -y netlify-cli deploy --dir=demo/dist --prod npx -y netlify-cli@23.7.3 deploy --dir=demo/dist --prod
env: env:
NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }}
NETLIFY_SITE_ID: ${{ secrets.NETLIFY_DEMO_DEV_SITE_ID }} NETLIFY_SITE_ID: ${{ secrets.NETLIFY_DEMO_DEV_SITE_ID }}
@@ -78,7 +78,7 @@ jobs:
- name: Deploy to Netlify - name: Deploy to Netlify
id: deploy id: deploy
run: | run: |
npx -y netlify-cli deploy --dir=demo/dist --prod npx -y netlify-cli@23.7.3 deploy --dir=demo/dist --prod
env: env:
NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }}
NETLIFY_SITE_ID: ${{ secrets.NETLIFY_DEMO_SITE_ID }} NETLIFY_SITE_ID: ${{ secrets.NETLIFY_DEMO_SITE_ID }}

View File

@@ -35,7 +35,7 @@ jobs:
- name: Deploy to Netlify - name: Deploy to Netlify
id: deploy id: deploy
run: | run: |
npx -y netlify-cli deploy --dir=gallery/dist --prod npx -y netlify-cli@23.7.3 deploy --dir=gallery/dist --prod
env: env:
NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }}
NETLIFY_SITE_ID: ${{ secrets.NETLIFY_GALLERY_SITE_ID }} NETLIFY_SITE_ID: ${{ secrets.NETLIFY_GALLERY_SITE_ID }}

View File

@@ -40,7 +40,7 @@ jobs:
- name: Deploy preview to Netlify - name: Deploy preview to Netlify
id: deploy id: deploy
run: | run: |
npx -y netlify-cli deploy --dir=gallery/dist --alias "deploy-preview-${{ github.event.number }}" \ npx -y netlify-cli@23.7.3 deploy --dir=gallery/dist --alias "deploy-preview-${{ github.event.number }}" \
--json > deploy_output.json --json > deploy_output.json
env: env:
NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }}

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

@@ -34,3 +34,5 @@ Check the [webawesome documentation](https://webawesome.com/docs/components/slid
**CSS Custom Properties** **CSS Custom Properties**
- `--ha-slider-track-size` - Height of the slider track. Defaults to `4px`. - `--ha-slider-track-size` - Height of the slider track. Defaults to `4px`.
- `--ha-slider-thumb-color` - Color of the slider thumb. Defaults to `var(--primary-color)`.
- `--ha-slider-indicator-color` - Color of the filled portion of the slider track. Defaults to `var(--primary-color)`.

View File

@@ -52,7 +52,7 @@
"@fullcalendar/list": "6.1.19", "@fullcalendar/list": "6.1.19",
"@fullcalendar/luxon3": "6.1.19", "@fullcalendar/luxon3": "6.1.19",
"@fullcalendar/timegrid": "6.1.19", "@fullcalendar/timegrid": "6.1.19",
"@home-assistant/webawesome": "3.0.0-beta.4.ha.3", "@home-assistant/webawesome": "3.0.0-beta.6.ha.4",
"@lezer/highlight": "1.2.1", "@lezer/highlight": "1.2.1",
"@lit-labs/motion": "1.0.9", "@lit-labs/motion": "1.0.9",
"@lit-labs/observers": "2.0.6", "@lit-labs/observers": "2.0.6",
@@ -161,7 +161,7 @@
"@rspack/core": "1.5.5", "@rspack/core": "1.5.5",
"@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.1",
@@ -203,7 +203,7 @@
"husky": "9.1.7", "husky": "9.1.7",
"jsdom": "27.0.0", "jsdom": "27.0.0",
"jszip": "3.10.1", "jszip": "3.10.1",
"lint-staged": "16.2.0", "lint-staged": "16.1.6",
"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",
@@ -213,7 +213,7 @@
"rspack-manifest-plugin": "5.1.0", "rspack-manifest-plugin": "5.1.0",
"serve": "14.2.5", "serve": "14.2.5",
"sinon": "21.0.0", "sinon": "21.0.0",
"tar": "7.4.4", "tar": "7.4.3",
"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",

View File

@@ -4,7 +4,7 @@ build-backend = "setuptools.build_meta"
[project] [project]
name = "home-assistant-frontend" name = "home-assistant-frontend"
version = "20250924.0" version = "20251001.4"
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

@@ -1,93 +0,0 @@
import { tinykeys } from "tinykeys";
import { canOverrideAlphanumericInput } from "../dom/can-override-input";
export type ShortcutHandler = (event: KeyboardEvent) => void;
interface ShortcutEntry {
/**
* The keys that the shortcut is registered to.
*/
keys: Set<string>;
/**
* A function to remove the shortcuts.
*/
disposer: () => void;
}
/**
* Register keyboard shortcuts using tinykeys.
*
* @param shortcuts - Key combinations mapped to handler functions.
* @returns A function to remove the shortcuts.
*/
function registerShortcuts(
shortcuts: Record<string, ShortcutHandler>
): () => void {
const wrappedShortcuts: Record<string, ShortcutHandler> = {};
Object.entries(shortcuts).forEach(([key, handler]) => {
wrappedShortcuts[key] = (event: KeyboardEvent) => {
// Don't capture the event if the user is not focused on an input
if (!canOverrideAlphanumericInput(event.composedPath())) {
return;
}
// Don't capture the event if the user is selecting text
if (window.getSelection()?.toString()) {
return;
}
handler(event);
};
});
return tinykeys(window, wrappedShortcuts);
}
/**
* A class that can add and remove keyboard shortcuts.
*/
export class ShortcutManager {
public shortcutEntries: ShortcutEntry[] = [];
/**
* Add a group of keyboard shortcuts to the manager.
*
* @param shortcuts - Key combinations mapped to handler functions.
* Uses tinykeys syntax. See https://github.com/jamiebuilds/tinykeys#usage.
*/
public add(shortcuts: Record<string, ShortcutHandler>) {
const disposer = registerShortcuts(shortcuts);
const keys = new Set(Object.keys(shortcuts));
const entry: ShortcutEntry = { keys, disposer };
this.shortcutEntries.push(entry);
}
/**
* Remove shortcuts from the manager.
*
* @param keys - Optional array of specific key combinations to remove. If provided,
* only shortcuts matching these keys will be removed. If omitted, all shortcuts
* from this manager will be removed.
*/
public remove(keys?: string[]) {
if (keys) {
const entriesToRemove: ShortcutEntry[] = [];
for (const entry of this.shortcutEntries) {
if (keys.some((key) => entry.keys.has(key))) {
entry.disposer();
entriesToRemove.push(entry);
}
}
entriesToRemove.forEach((entry) => {
const index = this.shortcutEntries.indexOf(entry);
if (index !== -1) {
this.shortcutEntries.splice(index, 1);
}
});
} else {
this.shortcutEntries.forEach((entry) => entry.disposer());
this.shortcutEntries.length = 0;
}
}
}

8
src/common/util/xss.ts Normal file
View File

@@ -0,0 +1,8 @@
import xss from "xss";
export const filterXSS = (html: string) =>
xss(html, {
whiteList: {},
stripIgnoreTag: true,
stripIgnoreTagBody: true,
});

View File

@@ -25,6 +25,7 @@ import type { ECOption } from "../../resources/echarts";
import type { HomeAssistant } from "../../types"; import type { HomeAssistant } from "../../types";
import { isMac } from "../../util/is_mac"; import { isMac } from "../../util/is_mac";
import "../ha-icon-button"; import "../ha-icon-button";
import { filterXSS } from "../../common/util/xss";
import { formatTimeLabel } from "./axis-label"; import { formatTimeLabel } from "./axis-label";
import { ensureArray } from "../../common/array/ensure-array"; import { ensureArray } from "../../common/array/ensure-array";
import "../chips/ha-assist-chip"; import "../chips/ha-assist-chip";
@@ -811,7 +812,8 @@ export class HaChartBase extends LitElement {
}; };
} }
} }
return { ...s, data }; const name = filterXSS(String(s.name ?? s.id ?? ""));
return { ...s, name, data };
}); });
return series as ECOption["series"]; return series as ECOption["series"];
} }

View File

@@ -9,6 +9,7 @@ import { ResizeController } from "@lit-labs/observers/resize-controller";
import type { HomeAssistant } from "../../types"; import type { HomeAssistant } from "../../types";
import type { ECOption } from "../../resources/echarts"; import type { ECOption } from "../../resources/echarts";
import { measureTextWidth } from "../../util/text"; import { measureTextWidth } from "../../util/text";
import { filterXSS } from "../../common/util/xss";
import "./ha-chart-base"; import "./ha-chart-base";
import { NODE_SIZE } from "../trace/hat-graph-const"; import { NODE_SIZE } from "../trace/hat-graph-const";
import "../ha-alert"; import "../ha-alert";
@@ -92,12 +93,12 @@ export class HaSankeyChart extends LitElement {
: data.value; : data.value;
if (data.id) { if (data.id) {
const node = this.data.nodes.find((n) => n.id === data.id); const node = this.data.nodes.find((n) => n.id === data.id);
return `${params.marker} ${node?.label ?? data.id}<br>${value}`; return `${params.marker} ${filterXSS(node?.label ?? data.id)}<br>${value}`;
} }
if (data.source && data.target) { if (data.source && data.target) {
const source = this.data.nodes.find((n) => n.id === data.source); const source = this.data.nodes.find((n) => n.id === data.source);
const target = this.data.nodes.find((n) => n.id === data.target); const target = this.data.nodes.find((n) => n.id === data.target);
return `${source?.label ?? data.source}${target?.label ?? data.target}<br>${value}`; return `${filterXSS(source?.label ?? data.source)}${filterXSS(target?.label ?? data.target)}<br>${value}`;
} }
return null; return null;
}; };

View File

@@ -1,5 +1,5 @@
import type { CSSResultGroup, TemplateResult } from "lit"; import type { CSSResultGroup, TemplateResult } from "lit";
import { css, html, LitElement } from "lit"; import { css, html, LitElement, nothing } 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 type { LocalizeFunc } from "../common/translations/localize"; import type { LocalizeFunc } from "../common/translations/localize";
@@ -73,14 +73,18 @@ export class HaAnalytics extends LitElement {
.checked=${this.analytics?.preferences[preference]} .checked=${this.analytics?.preferences[preference]}
.preference=${preference} .preference=${preference}
name=${preference} name=${preference}
?disabled=${baseEnabled}
> >
</ha-switch> </ha-switch>
<ha-tooltip .for="switch-${preference}" placement="right"> ${baseEnabled
${this.localize( ? nothing
`ui.panel.${this.translationKeyPanel}.analytics.need_base_enabled` : html`<ha-tooltip
)} .for="switch-${preference}"
</ha-tooltip> placement="right"
>
${this.localize(
`ui.panel.${this.translationKeyPanel}.analytics.need_base_enabled`
)}
</ha-tooltip>`}
</span> </span>
</ha-settings-row> </ha-settings-row>
` `

View File

@@ -42,8 +42,8 @@ export class HaBottomSheet extends LitElement {
static styles = css` static styles = css`
wa-drawer { wa-drawer {
--wa-color-surface-raised: var( --wa-color-surface-raised: var(
--ha-dialog-surface-background, --ha-bottom-sheet-surface-background,
var(--mdc-theme-surface, #fff) var(--ha-dialog-surface-background, var(--mdc-theme-surface, #fff)),
); );
--spacing: 0; --spacing: 0;
--size: auto; --size: auto;
@@ -51,8 +51,14 @@ export class HaBottomSheet extends LitElement {
--hide-duration: ${BOTTOM_SHEET_ANIMATION_DURATION_MS}ms; --hide-duration: ${BOTTOM_SHEET_ANIMATION_DURATION_MS}ms;
} }
wa-drawer::part(dialog) { wa-drawer::part(dialog) {
border-top-left-radius: var(--ha-border-radius-lg); border-top-left-radius: var(
border-top-right-radius: var(--ha-border-radius-lg); --ha-bottom-sheet-border-radius,
var(--ha-dialog-border-radius, var(--ha-border-radius-2xl))
);
border-top-right-radius: var(
--ha-bottom-sheet-border-radius,
var(--ha-dialog-border-radius, var(--ha-border-radius-2xl))
);
max-height: 90vh; max-height: 90vh;
padding-bottom: var(--safe-area-inset-bottom); padding-bottom: var(--safe-area-inset-bottom);
padding-left: var(--safe-area-inset-left); padding-left: var(--safe-area-inset-left);

View File

@@ -41,8 +41,7 @@ export class HaButton extends Button {
return [ return [
Button.styles, Button.styles,
css` css`
.button { :host {
/* set theme vars */
--wa-form-control-padding-inline: 16px; --wa-form-control-padding-inline: 16px;
--wa-font-weight-action: var(--ha-font-weight-medium); --wa-font-weight-action: var(--ha-font-weight-medium);
--wa-form-control-border-radius: var( --wa-form-control-border-radius: var(
@@ -54,7 +53,8 @@ export class HaButton extends Button {
--ha-button-height, --ha-button-height,
var(--button-height, 40px) var(--button-height, 40px)
); );
}
.button {
font-size: var(--ha-font-size-m); font-size: var(--ha-font-size-m);
line-height: 1; line-height: 1;

View File

@@ -121,7 +121,7 @@ export class HaDialog extends DialogBase {
position: var(--dialog-surface-position, relative); position: var(--dialog-surface-position, relative);
top: var(--dialog-surface-top); top: var(--dialog-surface-top);
margin-top: var(--dialog-surface-margin-top); margin-top: var(--dialog-surface-margin-top);
min-width: var(--mdc-dialog-min-width, 100vw); min-width: var(--mdc-dialog-min-width, auto);
min-height: var(--mdc-dialog-min-height, auto); min-height: var(--mdc-dialog-min-height, auto);
border-radius: var(--ha-dialog-border-radius, 24px); border-radius: var(--ha-dialog-border-radius, 24px);
-webkit-backdrop-filter: var(--ha-dialog-surface-backdrop-filter, none); -webkit-backdrop-filter: var(--ha-dialog-surface-backdrop-filter, none);
@@ -130,23 +130,13 @@ export class HaDialog extends DialogBase {
--ha-dialog-surface-background, --ha-dialog-surface-background,
var(--mdc-theme-surface, #fff) var(--mdc-theme-surface, #fff)
); );
padding: var(--dialog-surface-padding);
} }
:host([flexContent]) .mdc-dialog .mdc-dialog__content { :host([flexContent]) .mdc-dialog .mdc-dialog__content {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
} }
@media all and (max-width: 450px), all and (max-height: 500px) {
.mdc-dialog .mdc-dialog__surface {
min-height: 100vh;
max-height: 100vh;
padding-top: var(--safe-area-inset-top);
padding-bottom: var(--safe-area-inset-bottom);
padding-left: var(--safe-area-inset-left);
padding-right: var(--safe-area-inset-right);
}
}
.header_title { .header_title {
display: flex; display: flex;
align-items: center; align-items: center;

View File

@@ -1,6 +1,5 @@
import { css, html, LitElement } from "lit"; import { css, html, LitElement } from "lit";
import { customElement, query, state } from "lit/decorators"; import { customElement, query, state } from "lit/decorators";
import { styleMap } from "lit/directives/style-map";
import { fireEvent } from "../common/dom/fire_event"; import { fireEvent } from "../common/dom/fire_event";
import { BOTTOM_SHEET_ANIMATION_DURATION_MS } from "./ha-bottom-sheet"; import { BOTTOM_SHEET_ANIMATION_DURATION_MS } from "./ha-bottom-sheet";
@@ -37,13 +36,14 @@ export class HaResizableBottomSheet extends LitElement {
return html`<dialog return html`<dialog
open open
@transitionend=${this._handleTransitionEnd} @transitionend=${this._handleTransitionEnd}
style=${styleMap({ style=${`
height: this._dialogViewportHeight --height: ${this._dialogViewportHeight}vh;
? `${this._dialogViewportHeight}vh` --height: ${this._dialogViewportHeight}dvh;
: "auto", --max-height: ${this._dialogMaxViewpointHeight}vh;
maxHeight: `${this._dialogMaxViewpointHeight}vh`, --max-height: ${this._dialogMaxViewpointHeight}dvh;
minHeight: `${this._dialogMinViewpointHeight}vh`, --min-height: ${this._dialogMinViewpointHeight}vh;
})} --min-height: ${this._dialogMinViewpointHeight}dvh;
`}
> >
<div class="handle-wrapper"> <div class="handle-wrapper">
<div <div
@@ -213,12 +213,14 @@ export class HaResizableBottomSheet extends LitElement {
cursor: grabbing; cursor: grabbing;
} }
dialog { dialog {
height: auto; height: var(--height, auto);
max-height: 70vh; max-height: var(--max-height, 70vh);
min-height: 30vh; max-height: var(--max-height, 70dvh);
min-height: var(--min-height, 30vh);
min-height: var(--min-height, 30dvh);
background-color: var( background-color: var(
--ha-dialog-surface-background, --ha-bottom-sheet-surface-background,
var(--mdc-theme-surface, #fff) var(--ha-dialog-surface-background, var(--mdc-theme-surface, #fff)),
); );
display: flex; display: flex;
flex-direction: column; flex-direction: column;
@@ -239,12 +241,12 @@ export class HaResizableBottomSheet extends LitElement {
inset-inline-start: 0; inset-inline-start: 0;
box-shadow: 0px -8px 16px rgba(0, 0, 0, 0.2); box-shadow: 0px -8px 16px rgba(0, 0, 0, 0.2);
border-top-left-radius: var( border-top-left-radius: var(
--ha-dialog-border-radius, --ha-bottom-sheet-border-radius,
var(--ha-border-radius-2xl) var(--ha-dialog-border-radius, var(--ha-border-radius-2xl))
); );
border-top-right-radius: var( border-top-right-radius: var(
--ha-dialog-border-radius, --ha-bottom-sheet-border-radius,
var(--ha-border-radius-2xl) var(--ha-dialog-border-radius, var(--ha-border-radius-2xl))
); );
transform: translateY(100%); transform: translateY(100%);
transition: transform ${BOTTOM_SHEET_ANIMATION_DURATION_MS}ms ease; transition: transform ${BOTTOM_SHEET_ANIMATION_DURATION_MS}ms ease;
@@ -254,7 +256,6 @@ export class HaResizableBottomSheet extends LitElement {
border-bottom-width: 0; border-bottom-width: 0;
border-style: var(--ha-bottom-sheet-border-style); border-style: var(--ha-bottom-sheet-border-style);
border-color: var(--ha-bottom-sheet-border-color); border-color: var(--ha-bottom-sheet-border-color);
margin-bottom: var(--safe-area-inset-bottom);
margin-left: var(--safe-area-inset-left); margin-left: var(--safe-area-inset-left);
margin-right: var(--safe-area-inset-right); margin-right: var(--safe-area-inset-right);
} }

View File

@@ -39,22 +39,24 @@ class HaSegmentedBar extends LitElement {
<slot name="extra"></slot> <slot name="extra"></slot>
</div> </div>
<div class="bar"> <div class="bar">
${this.segments.map((segment) => { ${this.segments.map(
const bar = html`<div (segment, index) => html`
style=${styleMap({ ${this.hideTooltip || !segment.label
width: `${(segment.value / totalValue) * 100}%`, ? nothing
backgroundColor: segment.color, : html`
})} <ha-tooltip for="segment-${index}" placement="top">
></div>`; ${segment.label}
return this.hideTooltip && !segment.label </ha-tooltip>
? bar `}
: html` <div
<ha-tooltip> id="segment-${index}"
<span slot="content">${segment.label}</span> style=${styleMap({
${bar} width: `${(segment.value / totalValue) * 100}%`,
</ha-tooltip> backgroundColor: segment.color,
`; })}
})} ></div>
`
)}
</div> </div>
${this.hideLegend ${this.hideLegend
? nothing ? nothing

View File

@@ -82,12 +82,12 @@ export class HaNumberSelector extends LitElement {
labeled labeled
.min=${this.selector.number!.min} .min=${this.selector.number!.min}
.max=${this.selector.number!.max} .max=${this.selector.number!.max}
.value=${this.value ?? ""} .value=${this.value}
.step=${sliderStep} .step=${sliderStep}
.disabled=${this.disabled} .disabled=${this.disabled}
.required=${this.required} .required=${this.required}
@change=${this._handleSliderChange} @change=${this._handleSliderChange}
.ticks=${this.selector.number?.slider_ticks} .withMarkers=${this.selector.number?.slider_ticks || false}
> >
</ha-slider> </ha-slider>
` `

View File

@@ -19,7 +19,6 @@ export class HaSlider extends Slider {
Slider.styles, Slider.styles,
css` css`
:host { :host {
--wa-form-control-activated-color: var(--primary-color);
--track-size: var(--ha-slider-track-size, 4px); --track-size: var(--ha-slider-track-size, 4px);
--marker-height: calc(var(--ha-slider-track-size, 4px) / 2); --marker-height: calc(var(--ha-slider-track-size, 4px) / 2);
--marker-width: calc(var(--ha-slider-track-size, 4px) / 2); --marker-width: calc(var(--ha-slider-track-size, 4px) / 2);
@@ -54,6 +53,7 @@ export class HaSlider extends Slider {
#thumb { #thumb {
border: none; border: none;
background-color: var(--ha-slider-thumb-color, var(--primary-color));
} }
#slider:focus-visible:not(.disabled) #thumb, #slider:focus-visible:not(.disabled) #thumb,
@@ -62,14 +62,21 @@ export class HaSlider extends Slider {
outline: var(--wa-focus-ring); outline: var(--wa-focus-ring);
} }
#indicator {
background-color: var(
--ha-slider-indicator-color,
var(--primary-color)
);
}
:host([size="medium"]) { :host([size="medium"]) {
--thumb-width: var(--ha-font-size-l, 1.25em); --thumb-width: 20px;
--thumb-height: var(--ha-font-size-l, 1.25em); --thumb-height: 20px;
} }
:host([size="small"]) { :host([size="small"]) {
--thumb-width: var(--ha-font-size-m, 1em); --thumb-width: 16px;
--thumb-height: var(--ha-font-size-m, 1em); --thumb-height: 16px;
} }
`, `,
]; ];

View File

@@ -18,6 +18,8 @@ export class HaTabGroupTab extends Tab {
opacity: 0.8; opacity: 0.8;
color: inherit; color: inherit;
--wa-space-l: 16px;
} }
:host([active]:not([disabled])) { :host([active]:not([disabled])) {

View File

@@ -17,7 +17,7 @@ export class HaTooltip extends Tooltip {
css` css`
:host { :host {
--wa-tooltip-background-color: var(--secondary-background-color); --wa-tooltip-background-color: var(--secondary-background-color);
--wa-tooltip-color: var(--primary-text-color); --wa-tooltip-content-color: var(--primary-text-color);
--wa-tooltip-font-family: var( --wa-tooltip-font-family: var(
--ha-tooltip-font-family, --ha-tooltip-font-family,
var(--ha-font-family-body) var(--ha-font-family-body)

View File

@@ -12,6 +12,7 @@ export const DISCOVERY_SOURCES = [
"bluetooth", "bluetooth",
"dhcp", "dhcp",
"discovery", "discovery",
"esphome",
"hardware", "hardware",
"hassio", "hassio",
"homekit", "homekit",

View File

@@ -4,6 +4,7 @@ export interface LovelaceBadgeConfig {
type: string; type: string;
[key: string]: any; [key: string]: any;
visibility?: Condition[]; visibility?: Condition[];
disabled?: boolean;
} }
export const ensureBadgeConfig = ( export const ensureBadgeConfig = (

View File

@@ -26,20 +26,21 @@ class StepFlowLoading extends LitElement {
this.step this.step
); );
return html` return html`
<div class="init-spinner"> <div class="content">
<ha-spinner size="large"></ha-spinner>
${description ? html`<div>${description}</div>` : ""} ${description ? html`<div>${description}</div>` : ""}
<ha-spinner></ha-spinner>
</div> </div>
`; `;
} }
static styles = css` static styles = css`
.init-spinner { .content {
margin-top: 0;
padding: 50px 100px; padding: 50px 100px;
text-align: center; text-align: center;
} }
ha-spinner { ha-spinner {
margin-top: 16px; margin-bottom: 16px;
} }
`; `;
} }

View File

@@ -34,7 +34,7 @@ class StepFlowProgress extends LitElement {
)}%</ha-progress-ring )}%</ha-progress-ring
> >
` `
: html` <ha-spinner size="large"></ha-spinner> `} : html`<ha-spinner size="large"></ha-spinner>`}
${this.flowConfig.renderShowFormProgressDescription( ${this.flowConfig.renderShowFormProgressDescription(
this.hass, this.hass,
this.step this.step
@@ -48,6 +48,7 @@ class StepFlowProgress extends LitElement {
configFlowContentStyles, configFlowContentStyles,
css` css`
.content { .content {
margin-top: 0;
padding: 50px 100px; padding: 50px 100px;
text-align: center; text-align: center;
} }

View File

@@ -9,8 +9,9 @@ import {
mdiVolumeOff, mdiVolumeOff,
mdiVolumePlus, mdiVolumePlus,
} from "@mdi/js"; } from "@mdi/js";
import type { PropertyValues } from "lit";
import { css, html, LitElement, nothing } from "lit"; import { css, html, LitElement, nothing } from "lit";
import { customElement, property } from "lit/decorators"; import { customElement, property, query } from "lit/decorators";
import { ifDefined } from "lit/directives/if-defined"; import { ifDefined } from "lit/directives/if-defined";
import { classMap } from "lit/directives/class-map"; import { classMap } from "lit/directives/class-map";
import { stateActive } from "../../../common/entity/state_active"; import { stateActive } from "../../../common/entity/state_active";
@@ -19,7 +20,7 @@ import { formatDurationDigital } from "../../../common/datetime/format_duration"
import "../../../components/ha-icon-button"; import "../../../components/ha-icon-button";
import "../../../components/ha-list-item"; import "../../../components/ha-list-item";
import "../../../components/ha-select"; import "../../../components/ha-select";
import "../../../components/ha-slider"; import type { HaSlider } from "../../../components/ha-slider";
import "../../../components/ha-button"; import "../../../components/ha-button";
import "../../../components/ha-svg-icon"; import "../../../components/ha-svg-icon";
import { showMediaBrowserDialog } from "../../../components/media-player/show-media-browser-dialog"; import { showMediaBrowserDialog } from "../../../components/media-player/show-media-browser-dialog";
@@ -30,6 +31,8 @@ import type {
MediaPlayerEntity, MediaPlayerEntity,
} from "../../../data/media-player"; } from "../../../data/media-player";
import { import {
cleanupMediaTitle,
computeMediaDescription,
computeMediaControls, computeMediaControls,
handleMediaControlClick, handleMediaControlClick,
MediaPlayerEntityFeature, MediaPlayerEntityFeature,
@@ -48,10 +51,20 @@ class MoreInfoMediaPlayer extends LitElement {
@property({ attribute: false }) public stateObj?: MediaPlayerEntity; @property({ attribute: false }) public stateObj?: MediaPlayerEntity;
private _formateDuration(duration: number) { @query("#position-slider")
private _positionSlider?: HaSlider;
protected firstUpdated(_changedProperties: PropertyValues) {
if (this._positionSlider) {
this._positionSlider.valueFormatter = (value: number) =>
this._formatDuration(value);
}
}
private _formatDuration(duration: number) {
const hours = Math.floor(duration / 3600); const hours = Math.floor(duration / 3600);
const minutes = Math.floor((duration % 3600) / 60); const minutes = Math.floor((duration % 3600) / 60);
const seconds = duration % 60; const seconds = Math.floor(duration % 60);
return formatDurationDigital(this.hass.locale, { return formatDurationDigital(this.hass.locale, {
hours, hours,
minutes, minutes,
@@ -258,16 +271,19 @@ class MoreInfoMediaPlayer extends LitElement {
const stateObj = this.stateObj; const stateObj = this.stateObj;
const controls = computeMediaControls(stateObj, true); const controls = computeMediaControls(stateObj, true);
const coverUrl = stateObj.attributes.entity_picture || ""; const coverUrl =
stateObj.attributes.entity_picture_local ||
stateObj.attributes.entity_picture ||
"";
const playerObj = new HassMediaPlayerEntity(this.hass, this.stateObj); const playerObj = new HassMediaPlayerEntity(this.hass, this.stateObj);
const position = Math.floor(playerObj.currentProgress) || 0;
const duration = stateObj.attributes.media_duration || 0; const position = Math.max(Math.floor(playerObj.currentProgress || 0), 0);
const remaining = duration - position; const duration = Math.max(stateObj.attributes.media_duration || 0, 0);
const durationFormated = const remaining = Math.max(duration - position, 0);
remaining > 0 ? this._formateDuration(remaining) : 0; const remainingFormatted = this._formatDuration(remaining);
const postionFormated = this._formateDuration(position); const positionFormatted = this._formatDuration(position);
const primaryTitle = playerObj.primaryTitle; const primaryTitle = cleanupMediaTitle(stateObj.attributes.media_title);
const secondaryTitle = playerObj.secondaryTitle; const secondaryTitle = computeMediaDescription(stateObj);
const turnOn = controls?.find((c) => c.action === "turn_on"); const turnOn = controls?.find((c) => c.action === "turn_on");
const turnOff = controls?.find((c) => c.action === "turn_off"); const turnOff = controls?.find((c) => c.action === "turn_off");
@@ -313,6 +329,7 @@ class MoreInfoMediaPlayer extends LitElement {
? html` ? html`
<div class="position-bar"> <div class="position-bar">
<ha-slider <ha-slider
id="position-slider"
min="0" min="0"
max=${duration} max=${duration}
step="1" step="1"
@@ -323,11 +340,10 @@ class MoreInfoMediaPlayer extends LitElement {
@change=${this._handleMediaSeekChanged} @change=${this._handleMediaSeekChanged}
?disabled=${!stateActive(stateObj) || ?disabled=${!stateActive(stateObj) ||
!supportsFeature(stateObj, MediaPlayerEntityFeature.SEEK)} !supportsFeature(stateObj, MediaPlayerEntityFeature.SEEK)}
></ha-slider> >
<div class="position-info-row"> <span slot="reference">${positionFormatted}</span>
<span class="position-time">${postionFormated}</span> <span slot="reference">${remainingFormatted}</span>
<span class="duration-time">${durationFormated}</span> </ha-slider>
</div>
</div> </div>
` `
: nothing} : nothing}
@@ -348,28 +364,31 @@ class MoreInfoMediaPlayer extends LitElement {
</ha-icon-button>` </ha-icon-button>`
: html`<span class="spacer"></span>`; : html`<span class="spacer"></span>`;
})} })}
${["media_play_pause", "media_pause", "media_play"].map( ${[
(action) => { "media_play_pause",
const control = controls?.find((c) => c.action === action); "media_pause",
return control "media_play",
? html`<ha-button "media_stop",
variant="brand" ].map((action) => {
appearance="filled" const control = controls?.find((c) => c.action === action);
size="medium" return control
action=${action} ? html`<ha-button
@click=${this._handleClick} variant="brand"
class="center-control" appearance="filled"
> size="medium"
<ha-svg-icon action=${action}
.path=${control.icon} @click=${this._handleClick}
aria-label=${this.hass.localize( class="center-control"
`ui.card.media_player.${control.action}` >
)} <ha-svg-icon
></ha-svg-icon> .path=${control.icon}
</ha-button>` aria-label=${this.hass.localize(
: nothing; `ui.card.media_player.${control.action}`
} )}
)} ></ha-svg-icon>
</ha-button>`
: nothing;
})}
${["media_next_track", "shuffle_set"].map((action) => { ${["media_next_track", "shuffle_set"].map((action) => {
const control = controls?.find((c) => c.action === action); const control = controls?.find((c) => c.action === action);
return control return control
@@ -477,6 +496,22 @@ class MoreInfoMediaPlayer extends LitElement {
height: 320px; height: 320px;
} }
@media (max-height: 750px) {
.cover-container {
height: 120px;
}
.cover-image {
width: 100px;
height: 100px;
}
.cover-image--playing {
width: 120px;
height: 120px;
}
}
.empty-cover { .empty-cover {
background-color: var(--secondary-background-color); background-color: var(--secondary-background-color);
font-size: 1.5em; font-size: 1.5em;
@@ -548,13 +583,8 @@ class MoreInfoMediaPlayer extends LitElement {
flex-direction: column; flex-direction: column;
} }
.position-info-row { .position-bar ha-slider::part(references) {
display: flex;
flex-direction: row;
justify-content: space-between;
color: var(--secondary-text-color); color: var(--secondary-text-color);
padding: 0 8px;
font-size: var(--ha-font-size-s);
} }
.media-info-row { .media-info-row {

View File

@@ -8,6 +8,7 @@ import { createCloseHeading } from "../../components/ha-dialog";
import "../../components/ha-textarea"; import "../../components/ha-textarea";
import type { HaTextArea } from "../../components/ha-textarea"; import type { HaTextArea } from "../../components/ha-textarea";
import { convertTextToSpeech } from "../../data/tts"; import { convertTextToSpeech } from "../../data/tts";
import { haStyleDialog } from "../../resources/styles";
import type { HomeAssistant } from "../../types"; import type { HomeAssistant } from "../../types";
import { showAlertDialog } from "../generic/show-dialog-box"; import { showAlertDialog } from "../generic/show-dialog-box";
import type { TTSTryDialogParams } from "./show-dialog-tts-try"; import type { TTSTryDialogParams } from "./show-dialog-tts-try";
@@ -149,21 +150,24 @@ export class TTSTryDialog extends LitElement {
}); });
} }
static styles = css` static styles = [
ha-dialog { haStyleDialog,
--mdc-dialog-max-width: 500px; css`
} ha-dialog {
ha-textarea, --mdc-dialog-max-width: 500px;
ha-select { }
width: 100%; ha-textarea,
} ha-select {
ha-select { width: 100%;
margin-top: 8px; }
} ha-select {
.loading { margin-top: 8px;
height: 36px; }
} .loading {
`; height: 36px;
}
`,
];
} }
declare global { declare global {

View File

@@ -5,8 +5,8 @@ import { restoreScroll } from "../common/decorators/restore-scroll";
import { goBack } from "../common/navigate"; import { goBack } from "../common/navigate";
import "../components/ha-icon-button-arrow-prev"; import "../components/ha-icon-button-arrow-prev";
import "../components/ha-menu-button"; import "../components/ha-menu-button";
import type { HomeAssistant } from "../types";
import { haStyleScrollbar } from "../resources/styles"; import { haStyleScrollbar } from "../resources/styles";
import type { HomeAssistant } from "../types";
@customElement("hass-subpage") @customElement("hass-subpage")
class HassSubpage extends LitElement { class HassSubpage extends LitElement {
@@ -154,9 +154,15 @@ class HassSubpage extends LitElement {
1px - var(--header-height, 0px) - var( 1px - var(--header-height, 0px) - var(
--safe-area-inset-top, --safe-area-inset-top,
0px 0px
) - var(--safe-area-inset-bottom, 0px) ) - var(
--hass-subpage-bottom-inset,
var(--safe-area-inset-bottom, 0px)
)
);
margin-bottom: var(
--hass-subpage-bottom-inset,
var(--safe-area-inset-bottom)
); );
margin-bottom: var(--safe-area-inset-bottom);
margin-right: var(--safe-area-inset-right); margin-right: var(--safe-area-inset-right);
overflow-y: auto; overflow-y: auto;
overflow: auto; overflow: auto;

View File

@@ -12,11 +12,11 @@ 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();
const key = event.shiftKey ? event.key.toUpperCase() : event.key;
if ( if (
(event.ctrlKey || event.metaKey) && (event.ctrlKey || event.metaKey) &&
!event.shiftKey &&
!event.altKey && !event.altKey &&
event.key in supportedShortcuts key in supportedShortcuts
) { ) {
// Only capture the event if the user is not focused on an input // Only capture the event if the user is not focused on an input
if (!canOverrideAlphanumericInput(event.composedPath())) { if (!canOverrideAlphanumericInput(event.composedPath())) {
@@ -27,14 +27,14 @@ export const KeyboardShortcutMixin = <T extends Constructor<LitElement>>(
return; return;
} }
event.preventDefault(); event.preventDefault();
supportedShortcuts[event.key](); supportedShortcuts[key]();
return; return;
} }
const supportedSingleKeyShortcuts = this.supportedSingleKeyShortcuts(); const supportedSingleKeyShortcuts = this.supportedSingleKeyShortcuts();
if (event.key in supportedSingleKeyShortcuts) { if (key in supportedSingleKeyShortcuts) {
event.preventDefault(); event.preventDefault();
supportedSingleKeyShortcuts[event.key](); supportedSingleKeyShortcuts[key]();
} }
}; };

View File

@@ -1,5 +1,6 @@
import { consume } from "@lit/context"; import { consume } from "@lit/context";
import { import {
mdiAppleKeyboardCommand,
mdiCog, mdiCog,
mdiContentSave, mdiContentSave,
mdiDebugStepOver, mdiDebugStepOver,
@@ -73,8 +74,10 @@ import { showMoreInfoDialog } from "../../../dialogs/more-info/show-ha-more-info
import "../../../layouts/hass-subpage"; import "../../../layouts/hass-subpage";
import { KeyboardShortcutMixin } from "../../../mixins/keyboard-shortcut-mixin"; import { KeyboardShortcutMixin } from "../../../mixins/keyboard-shortcut-mixin";
import { PreventUnsavedMixin } from "../../../mixins/prevent-unsaved-mixin"; import { PreventUnsavedMixin } from "../../../mixins/prevent-unsaved-mixin";
import { UndoRedoMixin } from "../../../mixins/undo-redo-mixin";
import { haStyle } from "../../../resources/styles"; import { haStyle } from "../../../resources/styles";
import type { Entries, HomeAssistant, Route } from "../../../types"; import type { Entries, HomeAssistant, Route } from "../../../types";
import { isMac } from "../../../util/is_mac";
import { showToast } from "../../../util/toast"; import { showToast } from "../../../util/toast";
import { showAssignCategoryDialog } from "../category/show-dialog-assign-category"; import { showAssignCategoryDialog } from "../category/show-dialog-assign-category";
import "../ha-config-section"; import "../ha-config-section";
@@ -86,7 +89,6 @@ import {
import "./blueprint-automation-editor"; import "./blueprint-automation-editor";
import "./manual-automation-editor"; import "./manual-automation-editor";
import type { HaManualAutomationEditor } from "./manual-automation-editor"; import type { HaManualAutomationEditor } from "./manual-automation-editor";
import { UndoRedoMixin } from "../../../mixins/undo-redo-mixin";
declare global { declare global {
interface HTMLElementTagNameMap { interface HTMLElementTagNameMap {
@@ -215,6 +217,10 @@ export class HaAutomationEditor extends UndoRedoMixin<
: undefined; : undefined;
const useBlueprint = "use_blueprint" in this._config; const useBlueprint = "use_blueprint" in this._config;
const shortcutIcon = isMac
? html`<ha-svg-icon .path=${mdiAppleKeyboardCommand}></ha-svg-icon>`
: this.hass.localize("ui.panel.config.automation.editor.ctrl");
return html` return html`
<hass-subpage <hass-subpage
.hass=${this.hass} .hass=${this.hass}
@@ -231,16 +237,42 @@ export class HaAutomationEditor extends UndoRedoMixin<
.path=${mdiUndo} .path=${mdiUndo}
@click=${this.undo} @click=${this.undo}
.disabled=${!this.canUndo} .disabled=${!this.canUndo}
id="button-undo"
> >
</ha-icon-button> </ha-icon-button>
<ha-tooltip placement="bottom" for="button-undo">
${this.hass.localize("ui.common.undo")}
<span class="shortcut"
>(
<span>${shortcutIcon}</span>
<span>+</span>
<span>Z</span>)
</span>
</ha-tooltip>
<ha-icon-button <ha-icon-button
slot="toolbar-icon" slot="toolbar-icon"
.label=${this.hass.localize("ui.common.redo")} .label=${this.hass.localize("ui.common.redo")}
.path=${mdiRedo} .path=${mdiRedo}
@click=${this.redo} @click=${this.redo}
.disabled=${!this.canRedo} .disabled=${!this.canRedo}
id="button-redo"
> >
</ha-icon-button>` </ha-icon-button>
<ha-tooltip placement="bottom" for="button-redo">
${this.hass.localize("ui.common.redo")}
<span class="shortcut">
(
${isMac
? html`<span>${shortcutIcon}</span>
<span>+</span>
<span>Shift</span>
<span>+</span>
<span>Z</span>`
: html`<span>${shortcutIcon}</span>
<span>+</span>
<span>Y</span>`})
</span>
</ha-tooltip>`
: nothing} : nothing}
${this._config?.id && !this.narrow ${this._config?.id && !this.narrow
? html` ? html`
@@ -1171,6 +1203,7 @@ export class HaAutomationEditor extends UndoRedoMixin<
Delete: () => this._deleteSelectedRow(), Delete: () => this._deleteSelectedRow(),
Backspace: () => this._deleteSelectedRow(), Backspace: () => this._deleteSelectedRow(),
z: () => this.undo(), z: () => this.undo(),
Z: () => this.redo(),
y: () => this.redo(), y: () => this.redo(),
}; };
} }
@@ -1224,6 +1257,7 @@ export class HaAutomationEditor extends UndoRedoMixin<
--ha-automation-editor-width, --ha-automation-editor-width,
1540px 1540px
); );
--hass-subpage-bottom-inset: 0px;
} }
ha-fade-in { ha-fade-in {
display: flex; display: flex;
@@ -1292,6 +1326,15 @@ export class HaAutomationEditor extends UndoRedoMixin<
ha-fab.dirty { ha-fab.dirty {
bottom: calc(16px + var(--safe-area-inset-bottom, 0px)); bottom: calc(16px + var(--safe-area-inset-bottom, 0px));
} }
ha-tooltip ha-svg-icon {
width: 12px;
}
ha-tooltip .shortcut {
display: inline-flex;
flex-direction: row;
align-items: center;
gap: 2px;
}
`, `,
]; ];
} }

View File

@@ -292,9 +292,7 @@ export default class HaAutomationSidebar extends LitElement {
:host { :host {
z-index: 6; z-index: 6;
outline: none; outline: none;
height: calc( height: calc(100% - var(--safe-area-inset-top, 0px));
100% - var(--safe-area-inset-top) - var(--safe-area-inset-bottom)
);
--ha-card-border-radius: var( --ha-card-border-radius: var(
--ha-dialog-border-radius, --ha-dialog-border-radius,
var(--ha-border-radius-2xl) var(--ha-border-radius-2xl)
@@ -304,7 +302,6 @@ export default class HaAutomationSidebar extends LitElement {
--ha-bottom-sheet-border-style: solid; --ha-bottom-sheet-border-style: solid;
--ha-bottom-sheet-border-color: var(--primary-color); --ha-bottom-sheet-border-color: var(--primary-color);
margin-top: var(--safe-area-inset-top); margin-top: var(--safe-area-inset-top);
margin-bottom: var(--safe-area-inset-bottom);
} }
@media all and (max-width: 870px) { @media all and (max-width: 870px) {

View File

@@ -145,24 +145,19 @@ export const manualEditorStyles = css`
.content { .content {
padding-top: 24px; padding-top: 24px;
padding-bottom: 72px; padding-bottom: max(var(--safe-area-inset-bottom), 32px);
transition: padding-bottom 180ms ease-in-out; transition: padding-bottom 180ms ease-in-out;
} }
.content.has-bottom-sheet { .content.has-bottom-sheet {
padding-bottom: calc(90vh - 72px); padding-bottom: calc(90vh - max(var(--safe-area-inset-bottom), 32px));
} }
ha-automation-sidebar { ha-automation-sidebar {
position: fixed; position: fixed;
top: calc(var(--header-height) + 16px); top: calc(var(--header-height) + 16px);
height: calc( height: calc(-81px + 100vh - var(--safe-area-inset-top, 0px));
-81px + height: calc(-81px + 100dvh - var(--safe-area-inset-top, 0px));
100dvh - var(--safe-area-inset-top, 0px) - var(
--safe-area-inset-bottom,
0px
)
);
width: var(--sidebar-width); width: var(--sidebar-width);
display: block; display: block;
} }

View File

@@ -213,6 +213,7 @@ class HaConfigEnergy extends LitElement {
this.hass.states[key], this.hass.states[key],
]) ])
), ),
issues: this._validationResult,
}; };
const json = JSON.stringify(data, null, 2); const json = JSON.stringify(data, null, 2);
const blob = new Blob([json], { type: "application/json" }); const blob = new Blob([json], { type: "application/json" });

View File

@@ -1,5 +1,6 @@
import { consume } from "@lit/context"; import { consume } from "@lit/context";
import { import {
mdiAppleKeyboardCommand,
mdiCog, mdiCog,
mdiContentSave, mdiContentSave,
mdiDebugStepOver, mdiDebugStepOver,
@@ -64,8 +65,10 @@ import "../../../layouts/hass-subpage";
import { KeyboardShortcutMixin } from "../../../mixins/keyboard-shortcut-mixin"; import { KeyboardShortcutMixin } from "../../../mixins/keyboard-shortcut-mixin";
import { PreventUnsavedMixin } from "../../../mixins/prevent-unsaved-mixin"; import { PreventUnsavedMixin } from "../../../mixins/prevent-unsaved-mixin";
import { SubscribeMixin } from "../../../mixins/subscribe-mixin"; import { SubscribeMixin } from "../../../mixins/subscribe-mixin";
import { UndoRedoMixin } from "../../../mixins/undo-redo-mixin";
import { haStyle } from "../../../resources/styles"; import { haStyle } from "../../../resources/styles";
import type { Entries, HomeAssistant, Route } from "../../../types"; import type { Entries, HomeAssistant, Route } from "../../../types";
import { isMac } from "../../../util/is_mac";
import { showToast } from "../../../util/toast"; import { showToast } from "../../../util/toast";
import { showAutomationModeDialog } from "../automation/automation-mode-dialog/show-dialog-automation-mode"; import { showAutomationModeDialog } from "../automation/automation-mode-dialog/show-dialog-automation-mode";
import type { EntityRegistryUpdate } from "../automation/automation-save-dialog/show-dialog-automation-save"; import type { EntityRegistryUpdate } from "../automation/automation-save-dialog/show-dialog-automation-save";
@@ -74,7 +77,6 @@ import { showAssignCategoryDialog } from "../category/show-dialog-assign-categor
import "./blueprint-script-editor"; import "./blueprint-script-editor";
import "./manual-script-editor"; import "./manual-script-editor";
import type { HaManualScriptEditor } from "./manual-script-editor"; import type { HaManualScriptEditor } from "./manual-script-editor";
import { UndoRedoMixin } from "../../../mixins/undo-redo-mixin";
const baseEditorMixins = SubscribeMixin( const baseEditorMixins = SubscribeMixin(
PreventUnsavedMixin(KeyboardShortcutMixin(LitElement)) PreventUnsavedMixin(KeyboardShortcutMixin(LitElement))
@@ -168,6 +170,10 @@ export class HaScriptEditor extends UndoRedoMixin<
: undefined; : undefined;
const useBlueprint = "use_blueprint" in this._config; const useBlueprint = "use_blueprint" in this._config;
const shortcutIcon = isMac
? html`<ha-svg-icon .path=${mdiAppleKeyboardCommand}></ha-svg-icon>`
: this.hass.localize("ui.panel.config.automation.editor.ctrl");
return html` return html`
<hass-subpage <hass-subpage
.hass=${this.hass} .hass=${this.hass}
@@ -184,16 +190,41 @@ export class HaScriptEditor extends UndoRedoMixin<
.path=${mdiUndo} .path=${mdiUndo}
@click=${this.undo} @click=${this.undo}
.disabled=${!this.canUndo} .disabled=${!this.canUndo}
id="button-undo"
> >
</ha-icon-button> </ha-icon-button>
<ha-tooltip placement="bottom" for="button-undo">
${this.hass.localize("ui.common.undo")}
<span class="shortcut">
(<span>${shortcutIcon}</span>
<span>+</span>
<span>Z</span>)
</span>
</ha-tooltip>
<ha-icon-button <ha-icon-button
slot="toolbar-icon" slot="toolbar-icon"
.label=${this.hass.localize("ui.common.redo")} .label=${this.hass.localize("ui.common.redo")}
.path=${mdiRedo} .path=${mdiRedo}
@click=${this.redo} @click=${this.redo}
.disabled=${!this.canRedo} .disabled=${!this.canRedo}
id="button-redo"
> >
</ha-icon-button>` </ha-icon-button>
<ha-tooltip placement="bottom" for="button-redo">
${this.hass.localize("ui.common.redo")}
<span class="shortcut"
>(
${isMac
? html`<span>${shortcutIcon}</span>
<span>+</span>
<span>Shift</span>
<span>+</span>
<span>Z</span>`
: html`<span>${shortcutIcon}</span>
<span>+</span>
<span>Y</span>`})
</span>
</ha-tooltip>`
: nothing} : nothing}
${this.scriptId && !this.narrow ${this.scriptId && !this.narrow
? html` ? html`
@@ -1080,6 +1111,7 @@ export class HaScriptEditor extends UndoRedoMixin<
Delete: () => this._deleteSelectedRow(), Delete: () => this._deleteSelectedRow(),
Backspace: () => this._deleteSelectedRow(), Backspace: () => this._deleteSelectedRow(),
z: () => this.undo(), z: () => this.undo(),
Z: () => this.redo(),
y: () => this.redo(), y: () => this.redo(),
}; };
} }
@@ -1133,6 +1165,7 @@ export class HaScriptEditor extends UndoRedoMixin<
--ha-automation-editor-width, --ha-automation-editor-width,
1540px 1540px
); );
--hass-subpage-bottom-inset: 0px;
} }
.yaml-mode { .yaml-mode {
height: 100%; height: 100%;
@@ -1233,6 +1266,15 @@ export class HaScriptEditor extends UndoRedoMixin<
text-decoration: none; text-decoration: none;
color: var(--primary-color); color: var(--primary-color);
} }
ha-tooltip ha-svg-icon {
width: 12px;
}
ha-tooltip .shortcut {
display: inline-flex;
flex-direction: row;
align-items: center;
gap: 2px;
}
`, `,
]; ];
} }

View File

@@ -161,7 +161,7 @@ export class HuiBadge extends ReactiveElement {
); );
} }
private _updateVisibility(forceVisible?: boolean) { private _updateVisibility(ignoreConditions?: boolean) {
if (!this._element || !this.hass) { if (!this._element || !this.hass) {
return; return;
} }
@@ -171,9 +171,18 @@ export class HuiBadge extends ReactiveElement {
return; return;
} }
if (this.preview) {
this._setElementVisibility(true);
return;
}
if (this.config?.disabled) {
this._setElementVisibility(false);
return;
}
const visible = const visible =
forceVisible || ignoreConditions ||
this.preview ||
!this.config?.visibility || !this.config?.visibility ||
checkConditionsMet(this.config.visibility, this.hass); checkConditionsMet(this.config.visibility, this.hass);
this._setElementVisibility(visible); this._setElementVisibility(visible);

View File

@@ -1,30 +1,32 @@
import { mdiWaterBoiler } from "@mdi/js";
import type { PropertyValues, TemplateResult } from "lit"; import type { PropertyValues, TemplateResult } from "lit";
import { html, LitElement } from "lit"; import { html, LitElement } from "lit";
import { customElement, property, state } from "lit/decorators"; import { customElement, property, query, state } from "lit/decorators";
import { styleMap } from "lit/directives/style-map"; import { styleMap } from "lit/directives/style-map";
import { stopPropagation } from "../../../common/dom/stop_propagation";
import { computeDomain } from "../../../common/entity/compute_domain"; import { computeDomain } from "../../../common/entity/compute_domain";
import { stateColorCss } from "../../../common/entity/state_color"; import { stateColorCss } from "../../../common/entity/state_color";
import "../../../components/ha-control-button";
import "../../../components/ha-control-button-group";
import "../../../components/ha-control-select"; import "../../../components/ha-control-select";
import type { ControlSelectOption } from "../../../components/ha-control-select"; import type { ControlSelectOption } from "../../../components/ha-control-select";
import "../../../components/ha-control-slider"; import "../../../components/ha-control-select-menu";
import { UNAVAILABLE } from "../../../data/entity"; import type { HaControlSelectMenu } from "../../../components/ha-control-select-menu";
import "../../../components/ha-list-item";
import type { import type {
OperationMode, OperationMode,
WaterHeaterEntity, WaterHeaterEntity,
} from "../../../data/water_heater"; } from "../../../data/water_heater";
import { import {
compareWaterHeaterOperationMode,
computeOperationModeIcon, computeOperationModeIcon,
compareWaterHeaterOperationMode,
} from "../../../data/water_heater"; } from "../../../data/water_heater";
import { UNAVAILABLE } from "../../../data/entity";
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";
import { filterModes } from "./common/filter-modes"; import { filterModes } from "./common/filter-modes";
import type { import type {
LovelaceCardFeatureContext,
WaterHeaterOperationModesCardFeatureConfig, WaterHeaterOperationModesCardFeatureConfig,
LovelaceCardFeatureContext,
} from "./types"; } from "./types";
export const supportsWaterHeaterOperationModesCardFeature = ( export const supportsWaterHeaterOperationModesCardFeature = (
@@ -52,6 +54,9 @@ class HuiWaterHeaterOperationModeCardFeature
@state() _currentOperationMode?: OperationMode; @state() _currentOperationMode?: OperationMode;
@query("ha-control-select-menu", true)
private _haSelect?: HaControlSelectMenu;
private get _stateObj() { private get _stateObj() {
if (!this.hass || !this.context || !this.context.entity_id) { if (!this.hass || !this.context || !this.context.entity_id) {
return undefined; return undefined;
@@ -97,8 +102,23 @@ class HuiWaterHeaterOperationModeCardFeature
} }
} }
protected updated(changedProps: PropertyValues) {
super.updated(changedProps);
if (this._haSelect && changedProps.has("hass")) {
const oldHass = changedProps.get("hass") as HomeAssistant | undefined;
if (
this.hass &&
this.hass.formatEntityAttributeValue !==
oldHass?.formatEntityAttributeValue
) {
this._haSelect.layoutOptions();
}
}
}
private async _valueChanged(ev: CustomEvent) { private async _valueChanged(ev: CustomEvent) {
const mode = (ev.detail as any).value as OperationMode; const mode =
(ev.detail as any).value ?? ((ev.target as any).value as OperationMode);
if (mode === this._stateObj!.state) return; if (mode === this._stateObj!.state) return;
@@ -143,9 +163,48 @@ class HuiWaterHeaterOperationModeCardFeature
).map<ControlSelectOption>((mode) => ({ ).map<ControlSelectOption>((mode) => ({
value: mode, value: mode,
label: this.hass!.formatEntityState(this._stateObj!, mode), label: this.hass!.formatEntityState(this._stateObj!, mode),
path: computeOperationModeIcon(mode as OperationMode), icon: html`
<ha-svg-icon
slot="graphic"
.path=${computeOperationModeIcon(mode as OperationMode)}
></ha-svg-icon>
`,
})); }));
if (this._config.style === "dropdown") {
return html`
<ha-control-select-menu
show-arrow
hide-label
.label=${this.hass.localize("ui.card.water_heater.mode")}
.value=${this._currentOperationMode}
.disabled=${this._stateObj.state === UNAVAILABLE}
fixedMenuPosition
naturalMenuWidth
@selected=${this._valueChanged}
@closed=${stopPropagation}
>
${this._currentOperationMode
? html`
<ha-svg-icon
slot="icon"
.path=${computeOperationModeIcon(this._currentOperationMode)}
></ha-svg-icon>
`
: html`
<ha-svg-icon slot="icon" .path=${mdiWaterBoiler}></ha-svg-icon>
`}
${options.map(
(option) => html`
<ha-list-item .value=${option.value} graphic="icon">
${option.icon}${option.label}
</ha-list-item>
`
)}
</ha-control-select-menu>
`;
}
return html` return html`
<ha-control-select <ha-control-select
.options=${options} .options=${options}

View File

@@ -140,6 +140,7 @@ export interface ToggleCardFeatureConfig {
export interface WaterHeaterOperationModesCardFeatureConfig { export interface WaterHeaterOperationModesCardFeatureConfig {
type: "water-heater-operation-modes"; type: "water-heater-operation-modes";
style?: "dropdown" | "icons";
operation_modes?: OperationMode[]; operation_modes?: OperationMode[];
} }

View File

@@ -26,6 +26,7 @@ import {
} from "../../../../../common/datetime/format_date"; } from "../../../../../common/datetime/format_date";
import { formatTime } from "../../../../../common/datetime/format_time"; import { formatTime } from "../../../../../common/datetime/format_time";
import type { ECOption } from "../../../../../resources/echarts"; import type { ECOption } from "../../../../../resources/echarts";
import { filterXSS } from "../../../../../common/util/xss";
export function getSuggestedMax(dayDifference: number, end: Date): number { export function getSuggestedMax(dayDifference: number, end: Date): number {
let suggestedMax = new Date(end); let suggestedMax = new Date(end);
@@ -191,7 +192,7 @@ function formatTooltip(
countNegative++; countNegative++;
} }
} }
return `${param.marker} ${param.seriesName}: ${value} ${unit}`; return `${param.marker} ${filterXSS(param.seriesName!)}: ${value} ${unit}`;
}) })
.filter(Boolean); .filter(Boolean);
let footer = ""; let footer = "";

View File

@@ -6,6 +6,7 @@ import { classMap } from "lit/directives/class-map";
import memoizeOne from "memoize-one"; import memoizeOne from "memoize-one";
import type { BarSeriesOption } from "echarts/charts"; import type { BarSeriesOption } from "echarts/charts";
import type { ECElementEvent } from "echarts/types/dist/shared"; import type { ECElementEvent } from "echarts/types/dist/shared";
import { filterXSS } from "../../../../common/util/xss";
import { getGraphColorByIndex } from "../../../../common/color/colors"; import { getGraphColorByIndex } from "../../../../common/color/colors";
import { formatNumber } from "../../../../common/number/format_number"; import { formatNumber } from "../../../../common/number/format_number";
import "../../../../components/chart/ha-chart-base"; import "../../../../components/chart/ha-chart-base";
@@ -96,9 +97,8 @@ export class HuiEnergyDevicesGraphCard
} }
private _renderTooltip(params: any) { private _renderTooltip(params: any) {
const title = `<h4 style="text-align: center; margin: 0;">${this._getDeviceName( const deviceName = filterXSS(this._getDeviceName(params.value[1]));
params.value[1] const title = `<h4 style="text-align: center; margin: 0;">${deviceName}</h4>`;
)}</h4>`;
const value = `${formatNumber( const value = `${formatNumber(
params.value[0] as number, params.value[0] as number,
this.hass.locale, this.hass.locale,

View File

@@ -16,6 +16,7 @@ import type {
} from "../../card-features/types"; } from "../../card-features/types";
import type { LovelaceCardFeatureEditor } from "../../types"; import type { LovelaceCardFeatureEditor } from "../../types";
import { compareWaterHeaterOperationMode } from "../../../../data/water_heater"; import { compareWaterHeaterOperationMode } from "../../../../data/water_heater";
import type { LocalizeFunc } from "../../../../common/translations/localize";
type WaterHeaterOperationModesCardFeatureData = type WaterHeaterOperationModesCardFeatureData =
WaterHeaterOperationModesCardFeatureConfig & { WaterHeaterOperationModesCardFeatureConfig & {
@@ -39,11 +40,27 @@ export class HuiWaterHeaterOperationModesCardFeatureEditor
private _schema = memoizeOne( private _schema = memoizeOne(
( (
localize: LocalizeFunc,
formatEntityState: FormatEntityStateFunc, formatEntityState: FormatEntityStateFunc,
stateObj: HassEntity | undefined, stateObj: HassEntity | undefined,
customizeModes: boolean customizeModes: boolean
) => ) =>
[ [
{
name: "style",
selector: {
select: {
multiple: false,
mode: "list",
options: ["dropdown", "icons"].map((mode) => ({
value: mode,
label: localize(
`ui.panel.lovelace.editor.features.types.water-heater-operation-modes.style_list.${mode}`
),
})),
},
},
},
{ {
name: "customize_modes", name: "customize_modes",
selector: { selector: {
@@ -85,11 +102,13 @@ export class HuiWaterHeaterOperationModesCardFeatureEditor
: undefined; : undefined;
const data: WaterHeaterOperationModesCardFeatureData = { const data: WaterHeaterOperationModesCardFeatureData = {
style: "icons",
...this._config, ...this._config,
customize_modes: this._config.operation_modes !== undefined, customize_modes: this._config.operation_modes !== undefined,
}; };
const schema = this._schema( const schema = this._schema(
this.hass.localize,
this.hass.formatEntityState, this.hass.formatEntityState,
stateObj, stateObj,
data.customize_modes data.customize_modes
@@ -131,6 +150,7 @@ export class HuiWaterHeaterOperationModesCardFeatureEditor
) => { ) => {
switch (schema.name) { switch (schema.name) {
case "operation_modes": case "operation_modes":
case "style":
case "customize_modes": case "customize_modes":
return this.hass!.localize( return this.hass!.localize(
`ui.panel.lovelace.editor.features.types.water-heater-operation-modes.${schema.name}` `ui.panel.lovelace.editor.features.types.water-heater-operation-modes.${schema.name}`

View File

@@ -1,6 +1,7 @@
import { object, string, any } from "superstruct"; import { object, string, any, optional, boolean } from "superstruct";
export const baseLovelaceBadgeConfig = object({ export const baseLovelaceBadgeConfig = object({
type: string(), type: string(),
visibility: any(), visibility: any(),
disabled: optional(boolean()),
}); });

View File

@@ -1,4 +1,4 @@
import { object, string, any } from "superstruct"; import { object, string, any, optional, boolean } from "superstruct";
export const baseLovelaceCardConfig = object({ export const baseLovelaceCardConfig = object({
type: string(), type: string(),
@@ -6,4 +6,5 @@ export const baseLovelaceCardConfig = object({
layout_options: any(), layout_options: any(),
grid_options: any(), grid_options: any(),
visibility: any(), visibility: any(),
disabled: optional(boolean()),
}); });

View File

@@ -318,7 +318,7 @@ class HUIRoot extends LitElement {
menu-corner="END" menu-corner="END"
> >
<ha-icon-button <ha-icon-button
.label=${label} .id="button-${index}"
.path=${item.icon} .path=${item.icon}
slot="trigger" slot="trigger"
></ha-icon-button> ></ha-icon-button>
@@ -340,6 +340,9 @@ class HUIRoot extends LitElement {
` `
)} )}
</ha-button-menu> </ha-button-menu>
<ha-tooltip placement="bottom" .for="button-${index}">
${label}
</ha-tooltip>
` `
: html` : html`
<ha-icon-button <ha-icon-button

View File

@@ -4,13 +4,14 @@ import { isComponentLoaded } from "../../../../common/config/is_component_loaded
import type { LovelaceSectionConfig } from "../../../../data/lovelace/config/section"; import type { LovelaceSectionConfig } from "../../../../data/lovelace/config/section";
import { getCommonControlUsagePrediction } from "../../../../data/usage_prediction"; import { getCommonControlUsagePrediction } from "../../../../data/usage_prediction";
import type { HomeAssistant } from "../../../../types"; import type { HomeAssistant } from "../../../../types";
import type { TileCardConfig } from "../../cards/types"; import type { HeadingCardConfig, TileCardConfig } from "../../cards/types";
const DEFAULT_LIMIT = 8; const DEFAULT_LIMIT = 8;
export interface CommonControlSectionStrategyConfig { export interface CommonControlSectionStrategyConfig {
type: "common-controls"; type: "common-controls";
title?: string; title?: string;
icon?: string;
limit?: number; limit?: number;
exclude_entities?: string[]; exclude_entities?: string[];
hide_empty?: boolean; hide_empty?: boolean;
@@ -31,7 +32,8 @@ export class CommonControlsSectionStrategy extends ReactiveElement {
section.cards?.push({ section.cards?.push({
type: "heading", type: "heading",
heading: config.title, heading: config.title,
}); icon: config.icon,
} satisfies HeadingCardConfig);
} }
if (!isComponentLoaded(hass, "usage_prediction")) { if (!isComponentLoaded(hass, "usage_prediction")) {

View File

@@ -14,7 +14,15 @@ import {
polyfillTimeZoneData, polyfillTimeZoneData,
} from "./locale-data-polyfill"; } from "./locale-data-polyfill";
let polyfilled = false;
const _polyfillTimeZoneData = polyfillTimeZoneData;
const polyfillIntl = async () => { const polyfillIntl = async () => {
if (polyfilled) {
return;
}
polyfilled = true;
const locale = getLocalLanguage(); const locale = getLocalLanguage();
const polyfills: Promise<unknown>[] = []; const polyfills: Promise<unknown>[] = [];
if (shouldPolyfillGetCanonicalLocales()) { if (shouldPolyfillGetCanonicalLocales()) {
@@ -26,7 +34,7 @@ const polyfillIntl = async () => {
if (shouldPolyfillDateTimeFormat(locale)) { if (shouldPolyfillDateTimeFormat(locale)) {
polyfills.push( polyfills.push(
import("@formatjs/intl-datetimeformat/polyfill-force").then(() => import("@formatjs/intl-datetimeformat/polyfill-force").then(() =>
polyfillTimeZoneData() _polyfillTimeZoneData()
) )
); );
} }
@@ -58,7 +66,7 @@ const polyfillIntl = async () => {
if (polyfills.length === 0) { if (polyfills.length === 0) {
return; return;
} }
await Promise.all(polyfills).then(() => await Promise.allSettled(polyfills).then(() =>
// Load the default language // Load the default language
polyfillLocaleData(locale) polyfillLocaleData(locale)
); );

View File

@@ -157,8 +157,13 @@ export const haStyleDialog = css`
ha-dialog { ha-dialog {
--mdc-dialog-min-width: 100vw; --mdc-dialog-min-width: 100vw;
--mdc-dialog-max-width: 100vw; --mdc-dialog-max-width: 100vw;
--mdc-dialog-min-height: 100%; --mdc-dialog-min-height: 100vh;
--mdc-dialog-max-height: 100%; --mdc-dialog-min-height: 100svh;
--mdc-dialog-max-height: 100vh;
--mdc-dialog-max-height: 100svh;
--dialog-surface-padding: var(--safe-area-inset-top)
var(--safe-area-inset-right) var(--safe-area-inset-bottom)
var(--safe-area-inset-left);
--vertical-align-dialog: flex-end; --vertical-align-dialog: flex-end;
--ha-dialog-border-radius: 0; --ha-dialog-border-radius: 0;
} }

View File

@@ -19,6 +19,29 @@ export const coreStyles = css`
--ha-border-radius-pill: 9999px; --ha-border-radius-pill: 9999px;
--ha-border-radius-circle: 50%; --ha-border-radius-circle: 50%;
--ha-border-radius-square: 0; --ha-border-radius-square: 0;
/* Spacing */
--ha-space-0: 0px;
--ha-space-1: 4px;
--ha-space-2: 8px;
--ha-space-3: 12px;
--ha-space-4: 16px;
--ha-space-5: 20px;
--ha-space-6: 24px;
--ha-space-7: 28px;
--ha-space-8: 32px;
--ha-space-9: 36px;
--ha-space-10: 40px;
--ha-space-11: 44px;
--ha-space-12: 48px;
--ha-space-13: 52px;
--ha-space-14: 56px;
--ha-space-15: 60px;
--ha-space-16: 64px;
--ha-space-17: 68px;
--ha-space-18: 72px;
--ha-space-19: 76px;
--ha-space-20: 80px;
} }
`; `;

View File

@@ -1,4 +1,5 @@
import type { PropertyValues } from "lit"; import type { PropertyValues } from "lit";
import { tinykeys } from "tinykeys";
import memoizeOne from "memoize-one"; import memoizeOne from "memoize-one";
import { isComponentLoaded } from "../common/config/is_component_loaded"; import { isComponentLoaded } from "../common/config/is_component_loaded";
import { mainWindow } from "../common/dom/get_main_window"; import { mainWindow } from "../common/dom/get_main_window";
@@ -11,7 +12,6 @@ import type { Constructor, HomeAssistant } from "../types";
import { storeState } from "../util/ha-pref-storage"; import { storeState } from "../util/ha-pref-storage";
import { showToast } from "../util/toast"; import { showToast } from "../util/toast";
import type { HassElement } from "./hass-element"; import type { HassElement } from "./hass-element";
import { ShortcutManager } from "../common/keyboard/shortcuts";
import { extractSearchParamsObject } from "../common/url/search-params"; import { extractSearchParamsObject } from "../common/url/search-params";
import { showVoiceCommandDialog } from "../dialogs/voice-command-dialog/show-ha-voice-command-dialog"; import { showVoiceCommandDialog } from "../dialogs/voice-command-dialog/show-ha-voice-command-dialog";
import { canOverrideAlphanumericInput } from "../common/dom/can-override-input"; import { canOverrideAlphanumericInput } from "../common/dom/can-override-input";
@@ -62,8 +62,7 @@ export default <T extends Constructor<HassElement>>(superClass: T) =>
} }
private _registerShortcut() { private _registerShortcut() {
const shortcutManager = new ShortcutManager(); tinykeys(window, {
shortcutManager.add({
// Those are for latin keyboards that have e, c, m keys // Those are for latin keyboards that have e, c, m keys
e: (ev) => this._showQuickBar(ev), e: (ev) => this._showQuickBar(ev),
c: (ev) => this._showQuickBar(ev, QuickBarMode.Command), c: (ev) => this._showQuickBar(ev, QuickBarMode.Command),

View File

@@ -8217,7 +8217,12 @@
"water-heater-operation-modes": { "water-heater-operation-modes": {
"label": "Water heater operation modes", "label": "Water heater operation modes",
"operation_modes": "Operation modes", "operation_modes": "Operation modes",
"customize_modes": "Customize operation modes" "customize_modes": "Customize operation modes",
"style": "[%key:ui::panel::lovelace::editor::features::types::climate-preset-modes::style%]",
"style_list": {
"dropdown": "[%key:ui::panel::lovelace::editor::features::types::climate-preset-modes::style_list::dropdown%]",
"icons": "[%key:ui::panel::lovelace::editor::features::types::climate-preset-modes::style_list::icons%]"
}
}, },
"lawn-mower-commands": { "lawn-mower-commands": {
"label": "Lawn mower commands", "label": "Lawn mower commands",

130
yarn.lock
View File

@@ -1351,10 +1351,10 @@ __metadata:
languageName: node languageName: node
linkType: hard linkType: hard
"@ctrl/tinycolor@npm:^4.1.0": "@ctrl/tinycolor@npm:4.1.0":
version: 4.2.0 version: 4.1.0
resolution: "@ctrl/tinycolor@npm:4.2.0" resolution: "@ctrl/tinycolor@npm:4.1.0"
checksum: 10/1be14de7d7e8184c0bc5c8d7e3486cc8186e6702e8ca899c7239f328bb1df9a15d1575e2af7b4c6ba020727fa78f5a9f887555971f30a2890cece9e4253a9d3a checksum: 10/e64569399139ef0abd2eb0ec9fb7267dfd7820f7ad7d4567a63e5fc35e5cfdcb8ecdb3bad65cb9244b47ba6c77bc51085826c00e981acf263a3221dc89343adc
languageName: node languageName: node
linkType: hard linkType: hard
@@ -1940,11 +1940,11 @@ __metadata:
languageName: node languageName: node
linkType: hard linkType: hard
"@home-assistant/webawesome@npm:3.0.0-beta.4.ha.3": "@home-assistant/webawesome@npm:3.0.0-beta.6.ha.4":
version: 3.0.0-beta.4.ha.3 version: 3.0.0-beta.6.ha.4
resolution: "@home-assistant/webawesome@npm:3.0.0-beta.4.ha.3" resolution: "@home-assistant/webawesome@npm:3.0.0-beta.6.ha.4"
dependencies: dependencies:
"@ctrl/tinycolor": "npm:^4.1.0" "@ctrl/tinycolor": "npm:4.1.0"
"@floating-ui/dom": "npm:^1.6.13" "@floating-ui/dom": "npm:^1.6.13"
"@lit/react": "npm:^1.0.8" "@lit/react": "npm:^1.0.8"
"@shoelace-style/animations": "npm:^1.2.0" "@shoelace-style/animations": "npm:^1.2.0"
@@ -1953,8 +1953,7 @@ __metadata:
lit: "npm:^3.2.1" lit: "npm:^3.2.1"
nanoid: "npm:^5.1.5" nanoid: "npm:^5.1.5"
qr-creator: "npm:^1.0.0" qr-creator: "npm:^1.0.0"
style-observer: "npm:^0.0.7" checksum: 10/d9072b321126ef458468ed2cf040e0b04cb2aff73336c6e742c0cfb25d9fb674b7672e7c9abcf5bcb0aa0b2fe953c20186f0910f485024c827bfe4cf399f10a4
checksum: 10/b9241821ed471ccbad86b0ea4697a2d41395f05fdc26f46e5edbc7f6b5eeab5d248251ef702326312ded00d5bf850ce0dcdcf7cd5e2e542b9d9cb9a84f3726da
languageName: node languageName: node
linkType: hard linkType: hard
@@ -4491,10 +4490,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
@@ -6621,7 +6620,7 @@ __metadata:
languageName: node languageName: node
linkType: hard linkType: hard
"chalk@npm:^5.0.1": "chalk@npm:^5.0.1, chalk@npm:^5.6.0":
version: 5.6.2 version: 5.6.2
resolution: "chalk@npm:5.6.2" resolution: "chalk@npm:5.6.2"
checksum: 10/1b2f48f6fba1370670d5610f9cd54c391d6ede28f4b7062dd38244ea5768777af72e5be6b74fb6c6d54cb84c4a2dff3f3afa9b7cb5948f7f022cfd3d087989e0 checksum: 10/1b2f48f6fba1370670d5610f9cd54c391d6ede28f4b7062dd38244ea5768777af72e5be6b74fb6c6d54cb84c4a2dff3f3afa9b7cb5948f7f022cfd3d087989e0
@@ -6850,13 +6849,6 @@ __metadata:
languageName: node languageName: node
linkType: hard linkType: hard
"commander@npm:14.0.1":
version: 14.0.1
resolution: "commander@npm:14.0.1"
checksum: 10/783115e9403caeca29c0fcbd4e0358f70c67760e4e4933f3453fcdd5ddba2ec44173c8da5213d7ce5e404f51c7e71203a42c548164dbe27b668b32a8981577f1
languageName: node
linkType: hard
"commander@npm:^10.0.0": "commander@npm:^10.0.0":
version: 10.0.1 version: 10.0.1
resolution: "commander@npm:10.0.1" resolution: "commander@npm:10.0.1"
@@ -6864,6 +6856,13 @@ __metadata:
languageName: node languageName: node
linkType: hard linkType: hard
"commander@npm:^14.0.0":
version: 14.0.1
resolution: "commander@npm:14.0.1"
checksum: 10/783115e9403caeca29c0fcbd4e0358f70c67760e4e4933f3453fcdd5ddba2ec44173c8da5213d7ce5e404f51c7e71203a42c548164dbe27b668b32a8981577f1
languageName: node
linkType: hard
"commander@npm:^2.20.0, commander@npm:^2.20.3": "commander@npm:^2.20.0, commander@npm:^2.20.3":
version: 2.20.3 version: 2.20.3
resolution: "commander@npm:2.20.3" resolution: "commander@npm:2.20.3"
@@ -9397,7 +9396,7 @@ __metadata:
"@fullcalendar/list": "npm:6.1.19" "@fullcalendar/list": "npm:6.1.19"
"@fullcalendar/luxon3": "npm:6.1.19" "@fullcalendar/luxon3": "npm:6.1.19"
"@fullcalendar/timegrid": "npm:6.1.19" "@fullcalendar/timegrid": "npm:6.1.19"
"@home-assistant/webawesome": "npm:3.0.0-beta.4.ha.3" "@home-assistant/webawesome": "npm:3.0.0-beta.6.ha.4"
"@lezer/highlight": "npm:1.2.1" "@lezer/highlight": "npm:1.2.1"
"@lit-labs/motion": "npm:1.0.9" "@lit-labs/motion": "npm:1.0.9"
"@lit-labs/observers": "npm:2.0.6" "@lit-labs/observers": "npm:2.0.6"
@@ -9442,7 +9441,7 @@ __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.1"
@@ -9514,7 +9513,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.2.0" lint-staged: "npm:16.1.6"
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"
@@ -9539,7 +9538,7 @@ __metadata:
sortablejs: "patch:sortablejs@npm%3A1.15.6#~/.yarn/patches/sortablejs-npm-1.15.6-3235a8f83b.patch" sortablejs: "patch:sortablejs@npm%3A1.15.6#~/.yarn/patches/sortablejs-npm-1.15.6-3235a8f83b.patch"
stacktrace-js: "npm:2.0.2" stacktrace-js: "npm:2.0.2"
superstruct: "npm:2.0.2" superstruct: "npm:2.0.2"
tar: "npm:7.4.4" tar: "npm:7.4.3"
terser-webpack-plugin: "npm:5.3.14" terser-webpack-plugin: "npm:5.3.14"
tinykeys: "npm:3.0.0" tinykeys: "npm:3.0.0"
ts-lit-plugin: "npm:2.0.2" ts-lit-plugin: "npm:2.0.2"
@@ -10861,6 +10860,13 @@ __metadata:
languageName: node languageName: node
linkType: hard linkType: hard
"lilconfig@npm:^3.1.3":
version: 3.1.3
resolution: "lilconfig@npm:3.1.3"
checksum: 10/b932ce1af94985f0efbe8896e57b1f814a48c8dbd7fc0ef8469785c6303ed29d0090af3ccad7e36b626bfca3a4dc56cc262697e9a8dd867623cf09a39d54e4c3
languageName: node
linkType: hard
"lines-and-columns@npm:2.0.4": "lines-and-columns@npm:2.0.4":
version: 2.0.4 version: 2.0.4
resolution: "lines-and-columns@npm:2.0.4" resolution: "lines-and-columns@npm:2.0.4"
@@ -10868,24 +10874,27 @@ __metadata:
languageName: node languageName: node
linkType: hard linkType: hard
"lint-staged@npm:16.2.0": "lint-staged@npm:16.1.6":
version: 16.2.0 version: 16.1.6
resolution: "lint-staged@npm:16.2.0" resolution: "lint-staged@npm:16.1.6"
dependencies: dependencies:
commander: "npm:14.0.1" chalk: "npm:^5.6.0"
listr2: "npm:9.0.4" commander: "npm:^14.0.0"
micromatch: "npm:4.0.8" debug: "npm:^4.4.1"
nano-spawn: "npm:1.0.3" lilconfig: "npm:^3.1.3"
pidtree: "npm:0.6.0" listr2: "npm:^9.0.3"
string-argv: "npm:0.3.2" micromatch: "npm:^4.0.8"
yaml: "npm:2.8.1" nano-spawn: "npm:^1.0.2"
pidtree: "npm:^0.6.0"
string-argv: "npm:^0.3.2"
yaml: "npm:^2.8.1"
bin: bin:
lint-staged: bin/lint-staged.js lint-staged: bin/lint-staged.js
checksum: 10/809a42e21f2634c1a3e718dfb25786275a13b51c0cfaef6bb4bed509c656d31ee9b3e6231df55223b4b60cb37e4b5e3ebd958b239cabb529d2d07253cf7e1726 checksum: 10/922b4392ae5d3d56130e4eba706c2fa6151d5da5e21f57ab601b1d6ce9cc635ceb5e4c3dc00e7da83ba8f0cb244b82604469c7ea1470b1e6b6ea0fc12454aa08
languageName: node languageName: node
linkType: hard linkType: hard
"listr2@npm:9.0.4": "listr2@npm:^9.0.3":
version: 9.0.4 version: 9.0.4
resolution: "listr2@npm:9.0.4" resolution: "listr2@npm:9.0.4"
dependencies: dependencies:
@@ -11262,7 +11271,7 @@ __metadata:
languageName: node languageName: node
linkType: hard linkType: hard
"micromatch@npm:4.0.8, micromatch@npm:^4.0.2, micromatch@npm:^4.0.4, micromatch@npm:^4.0.8": "micromatch@npm:^4.0.2, micromatch@npm:^4.0.4, micromatch@npm:^4.0.8":
version: 4.0.8 version: 4.0.8
resolution: "micromatch@npm:4.0.8" resolution: "micromatch@npm:4.0.8"
dependencies: dependencies:
@@ -11467,12 +11476,21 @@ __metadata:
languageName: node languageName: node
linkType: hard linkType: hard
"minizlib@npm:^3.0.1, minizlib@npm:^3.1.0": "minizlib@npm:^3.0.1":
version: 3.1.0 version: 3.0.2
resolution: "minizlib@npm:3.1.0" resolution: "minizlib@npm:3.0.2"
dependencies: dependencies:
minipass: "npm:^7.1.2" minipass: "npm:^7.1.2"
checksum: 10/f47365cc2cb7f078cbe7e046eb52655e2e7e97f8c0a9a674f4da60d94fb0624edfcec9b5db32e8ba5a99a5f036f595680ae6fe02a262beaa73026e505cc52f99 checksum: 10/c075bed1594f68dcc8c35122333520112daefd4d070e5d0a228bd4cf5580e9eed3981b96c0ae1d62488e204e80fd27b2b9d0068ca9a5ef3993e9565faf63ca41
languageName: node
linkType: hard
"mkdirp@npm:^3.0.1":
version: 3.0.1
resolution: "mkdirp@npm:3.0.1"
bin:
mkdirp: dist/cjs/src/bin.js
checksum: 10/16fd79c28645759505914561e249b9a1f5fe3362279ad95487a4501e4467abeb714fd35b95307326b8fd03f3c7719065ef11a6f97b7285d7888306d1bd2232ba
languageName: node languageName: node
linkType: hard linkType: hard
@@ -11516,7 +11534,7 @@ __metadata:
languageName: node languageName: node
linkType: hard linkType: hard
"nano-spawn@npm:1.0.3": "nano-spawn@npm:^1.0.2":
version: 1.0.3 version: 1.0.3
resolution: "nano-spawn@npm:1.0.3" resolution: "nano-spawn@npm:1.0.3"
checksum: 10/72c56e68ae733c81c459a338fd51e2aa3be06b1cca746c2abe83df7acfac7eee008b01833f5a8781f4ac9fc1eafd23036a44755257a669dfcc2ff2453850822a checksum: 10/72c56e68ae733c81c459a338fd51e2aa3be06b1cca746c2abe83df7acfac7eee008b01833f5a8781f4ac9fc1eafd23036a44755257a669dfcc2ff2453850822a
@@ -12240,7 +12258,7 @@ __metadata:
languageName: node languageName: node
linkType: hard linkType: hard
"pidtree@npm:0.6.0": "pidtree@npm:^0.6.0":
version: 0.6.0 version: 0.6.0
resolution: "pidtree@npm:0.6.0" resolution: "pidtree@npm:0.6.0"
bin: bin:
@@ -13767,7 +13785,7 @@ __metadata:
languageName: node languageName: node
linkType: hard linkType: hard
"string-argv@npm:0.3.2": "string-argv@npm:^0.3.2":
version: 0.3.2 version: 0.3.2
resolution: "string-argv@npm:0.3.2" resolution: "string-argv@npm:0.3.2"
checksum: 10/f9d3addf887026b4b5f997a271149e93bf71efc8692e7dc0816e8807f960b18bcb9787b45beedf0f97ff459575ee389af3f189d8b649834cac602f2e857e75af checksum: 10/f9d3addf887026b4b5f997a271149e93bf71efc8692e7dc0816e8807f960b18bcb9787b45beedf0f97ff459575ee389af3f189d8b649834cac602f2e857e75af
@@ -13984,13 +14002,6 @@ __metadata:
languageName: node languageName: node
linkType: hard linkType: hard
"style-observer@npm:^0.0.7":
version: 0.0.7
resolution: "style-observer@npm:0.0.7"
checksum: 10/bb57f98bae4463c1e1b57234f8ffe72ec0de27fb08b032c1919910129c210aacd1ddd615432b9453d491e10d3b719cf6c2a68a97165ca55d6fc9b86c0fca37fb
languageName: node
linkType: hard
"style-observer@npm:^0.0.8": "style-observer@npm:^0.0.8":
version: 0.0.8 version: 0.0.8
resolution: "style-observer@npm:0.0.8" resolution: "style-observer@npm:0.0.8"
@@ -14079,16 +14090,17 @@ __metadata:
languageName: node languageName: node
linkType: hard linkType: hard
"tar@npm:7.4.4, tar@npm:^7.4.3": "tar@npm:7.4.3, tar@npm:^7.4.3":
version: 7.4.4 version: 7.4.3
resolution: "tar@npm:7.4.4" resolution: "tar@npm:7.4.3"
dependencies: dependencies:
"@isaacs/fs-minipass": "npm:^4.0.0" "@isaacs/fs-minipass": "npm:^4.0.0"
chownr: "npm:^3.0.0" chownr: "npm:^3.0.0"
minipass: "npm:^7.1.2" minipass: "npm:^7.1.2"
minizlib: "npm:^3.1.0" minizlib: "npm:^3.0.1"
mkdirp: "npm:^3.0.1"
yallist: "npm:^5.0.0" yallist: "npm:^5.0.0"
checksum: 10/be7d95e019b029ac507e7cd4b23c243ba896b67d0837c4f53d18c32a5014a24b7b247e982f4d47147b8d637c491b35cc122e19e29246137ecb2b88a495aaf1fb checksum: 10/12a2a4fc6dee23e07cc47f1aeb3a14a1afd3f16397e1350036a8f4cdfee8dcac7ef5978337a4e7b2ac2c27a9a6d46388fc2088ea7c80cb6878c814b1425f8ecf
languageName: node languageName: node
linkType: hard linkType: hard
@@ -16006,7 +16018,7 @@ __metadata:
languageName: node languageName: node
linkType: hard linkType: hard
"yaml@npm:2.8.1": "yaml@npm:^2.8.1":
version: 2.8.1 version: 2.8.1
resolution: "yaml@npm:2.8.1" resolution: "yaml@npm:2.8.1"
bin: bin: