Compare commits

..

28 Commits

Author SHA1 Message Date
Jan-Philipp Benecke
e52cccd916 Implement core spacing tokens 2025-09-27 23:13:53 +02:00
Paul Bottein
c814b8e888 Align dashboard data table with other data tables (#27206)
* Align dashboard data table with other data tables

* Update ha-config-lovelace-dashboards.ts

Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com>

* Update ha-config-lovelace-dashboards.ts

Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com>

---------

Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com>
2025-09-27 18:07:01 +03:00
renovate[bot]
33a0b32cc5 Update vaadinWebComponents monorepo to v24.9.1 (#27220)
Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com>
2025-09-27 13:25:12 +02:00
renovate[bot]
7dae13bf57 Update dependency @rspack/core to v1.5.7 (#27219)
Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com>
2025-09-27 13:24:59 +02:00
renovate[bot]
0a3fe6e0fb Update dependency tar to v7.5.1 (#27216) 2025-09-26 19:21:46 +02:00
Paul Bottein
e0348e4da7 Fix slider ticks support for number selector (#27211) 2025-09-26 15:35:01 +02:00
Aidan Timson
d53f3ec898 Add missing translations for thread config panel (#27210) 2025-09-26 14:09:11 +01:00
renovate[bot]
e422547d93 Update Yarn to v4.10.3 (#27209)
Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com>
2025-09-26 15:33:12 +03:00
Paul Bottein
d91a3fbe85 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:28:59 +03:00
Paul Bottein
01d7130f22 Fix try tts dialog max width (#27208) 2025-09-26 13:36:05 +02:00
Aidan Timson
c57851e4df Migrate hex color helper functions to culori (#27184) 2025-09-26 11:31:18 +02:00
Aidan Timson
6f1f13acb0 Migrate rgb color helper functions to culori (#27185) 2025-09-26 11:00:08 +02:00
Jan-Philipp Benecke
a8abd00809 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 06:33:56 +00:00
karwosts
e053978dbe Add dropdown mode to water heater operation feature (#27201) 2025-09-26 08:51:03 +03:00
karwosts
6e57f726a3 Add validation issues to energy diagnostic (#27203) 2025-09-26 08:48:21 +03:00
renovate[bot]
b7cabadbe1 Update dependency typescript-eslint to v8.44.1 (#27197) 2025-09-25 22:09:08 +02:00
Simon Lamon
d920217374 Fix typos in media player more info (#27198) 2025-09-25 19:02:26 +00:00
Jan-Philipp Benecke
1630263276 Round seconds in media player more info before formatting (#27196) 2025-09-25 20:47:40 +02:00
Paul Bottein
5680c742be Revert "Update dependency @types/chromecast-caf-receiver to v6.0.24" (#27188) 2025-09-25 17:48:23 +02:00
Paul Bottein
2aeb9cf0ef Revert "Update dependency @types/chromecast-caf-receiver to v6.0.24 (#26500)"
This reverts commit 4a3ed62583.
2025-09-25 17:47:59 +02:00
Paul Bottein
c9931b3a3c Disabled config badge (#27172)
* Add disabled option for badge

* Add disabled to struct
2025-09-25 17:45:06 +02:00
Paul Bottein
fbf7ebdfe4 Add icon option to common controls section strategy (#27180) 2025-09-25 17:23:54 +02:00
renovate[bot]
52ccb03de5 Update dependency hls.js to v1.6.13 (#27187)
Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com>
2025-09-25 17:21:23 +02:00
Paul Bottein
900236ac07 Fix storage bar not displayed (#27183) 2025-09-25 15:56:26 +01:00
renovate[bot]
28940c930d Update dependency @codemirror/view to v6.38.3 (#27163)
Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com>
2025-09-25 15:55:35 +01:00
Paul Bottein
e278b463fd Fix analytics switches (#27181) 2025-09-25 15:54:11 +01:00
renovate[bot]
db2acd4e39 Update dependency @rspack/core to v1.5.6 (#27177)
Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com>
2025-09-25 13:10:01 +02:00
Wendelin
6dcc52cd44 Reduce default tab padding in tab-group (#27173) 2025-09-25 11:52:04 +01:00
127 changed files with 948 additions and 834 deletions

File diff suppressed because one or more lines are too long

View File

@@ -6,4 +6,4 @@ enableGlobalCache: false
nodeLinker: node-modules
yarnPath: .yarn/releases/yarn-4.10.2.cjs
yarnPath: .yarn/releases/yarn-4.10.3.cjs

View File

@@ -5,17 +5,17 @@ const castContext = framework.CastReceiverContext.getInstance();
const playerManager = castContext.getPlayerManager();
playerManager.setMessageInterceptor(
"LOAD" as framework.messages.MessageType.LOAD,
framework.messages.MessageType.LOAD,
(loadRequestData) => {
const media = loadRequestData.media;
// Special handling if it came from Google Assistant
if (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";
// @ts-ignore
media.hlsVideoSegmentFormat =
"fmp4" as framework.messages.HlsVideoSegmentFormat.FMP4;
framework.messages.HlsVideoSegmentFormat.FMP4;
}
return loadRequestData;
}

View File

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

View File

@@ -117,7 +117,7 @@ export class DemoHaBadge extends LitElement {
}
.card-content {
display: flex;
gap: 24px;
gap: var(--ha-space-6);
}
`;
}

View File

@@ -155,11 +155,11 @@ export class DemoHaButton extends LitElement {
.card-content {
display: flex;
flex-direction: column;
gap: 24px;
gap: var(--ha-space-6);
}
.card-content div {
display: flex;
gap: 8px;
gap: var(--ha-space-2);
}
`;
}

View File

@@ -123,11 +123,11 @@ export class DemoHaProgressButton extends LitElement {
.card-content {
display: flex;
flex-direction: column;
gap: 24px;
gap: var(--ha-space-6);
}
.card-content div {
display: flex;
gap: 8px;
gap: var(--ha-space-2);
}
`;
}

View File

@@ -88,7 +88,7 @@ export class DemoHaSlider extends LitElement {
display: flex;
flex-direction: column;
align-items: center;
gap: 24px;
gap: var(--ha-space-6);
}
`;
}

View File

@@ -70,7 +70,7 @@ export class DemoHaSpinner extends LitElement {
display: flex;
flex-direction: column;
align-items: center;
gap: 24px;
gap: var(--ha-space-6);
}
`;
}

View File

@@ -159,7 +159,7 @@ class HassioSystemManagedDialog extends LitElement {
display: flex;
justify-content: center;
align-items: center;
gap: 16px;
gap: var(--ha-space-4);
--mdc-icon-size: 48px;
margin-bottom: 32px;
}

View File

@@ -31,7 +31,7 @@ export const hassioStyle = css`
.card-group {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(300px, 1fr));
grid-gap: 8px;
grid-gap: var(--ha-space-2);
}
@media screen and (min-width: 640px) {
.card-group {

View File

@@ -213,7 +213,7 @@ class HaLandingPage extends LandingPageBaseElement {
ha-card .card-content {
display: flex;
flex-direction: column;
gap: 16px;
gap: var(--ha-space-4);
}
ha-alert p {
text-align: unset;

View File

@@ -34,7 +34,7 @@
"@codemirror/legacy-modes": "6.5.1",
"@codemirror/search": "6.5.11",
"@codemirror/state": "6.5.2",
"@codemirror/view": "6.38.2",
"@codemirror/view": "6.38.3",
"@date-fns/tz": "1.4.1",
"@egjs/hammerjs": "2.0.17",
"@formatjs/intl-datetimeformat": "6.18.0",
@@ -89,8 +89,8 @@
"@thomasloven/round-slider": "0.6.0",
"@tsparticles/engine": "3.9.1",
"@tsparticles/preset-links": "3.2.0",
"@vaadin/combo-box": "24.9.0",
"@vaadin/vaadin-themable-mixin": "24.9.0",
"@vaadin/combo-box": "24.9.1",
"@vaadin/vaadin-themable-mixin": "24.9.1",
"@vibrant/color": "4.0.0",
"@vue/web-component-wrapper": "1.3.0",
"@webcomponents/scoped-custom-element-registry": "0.0.10",
@@ -111,7 +111,7 @@
"fuse.js": "7.1.0",
"google-timezones-json": "1.2.0",
"gulp-zopfli-green": "6.0.2",
"hls.js": "1.6.12",
"hls.js": "1.6.13",
"home-assistant-js-websocket": "9.5.0",
"idb-keyval": "6.2.2",
"intl-messageformat": "10.7.16",
@@ -158,10 +158,10 @@
"@octokit/plugin-retry": "8.0.1",
"@octokit/rest": "22.0.0",
"@rsdoctor/rspack-plugin": "1.2.3",
"@rspack/core": "1.5.5",
"@rspack/core": "1.5.7",
"@rspack/dev-server": "1.1.4",
"@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/color-name": "2.0.0",
"@types/culori": "4.0.1",
@@ -213,11 +213,11 @@
"rspack-manifest-plugin": "5.1.0",
"serve": "14.2.5",
"sinon": "21.0.0",
"tar": "7.4.4",
"tar": "7.5.1",
"terser-webpack-plugin": "5.3.14",
"ts-lit-plugin": "2.0.2",
"typescript": "5.9.2",
"typescript-eslint": "8.44.0",
"typescript-eslint": "8.44.1",
"vite-tsconfig-paths": "5.1.4",
"vitest": "3.2.4",
"webpack-stats-plugin": "1.1.3",
@@ -235,5 +235,5 @@
"tslib": "2.8.1",
"@material/mwc-list@^0.27.0": "patch:@material/mwc-list@npm%3A0.27.0#~/.yarn/patches/@material-mwc-list-npm-0.27.0-5344fc9de4.patch"
},
"packageManager": "yarn@4.10.2"
"packageManager": "yarn@4.10.3"
}

View File

@@ -1,23 +1,40 @@
import { formatHex, parse } from "culori";
/**
* Expands a 3-digit hex color to a 6-digit hex color.
* @param hex - The hex color to expand.
* @returns The expanded hex color.
* @throws If the hex color is invalid.
*/
export const expandHex = (hex: string): string => {
hex = hex.replace("#", "");
if (hex.length === 6) return hex;
let result = "";
for (const val of hex) {
result += val + val;
const color = parse(hex);
if (!color) {
throw new Error(`Invalid hex color: ${hex}`);
}
return result;
const formattedColor = formatHex(color);
if (!formattedColor) {
throw new Error(`Could not format hex color: ${hex}`);
}
return formattedColor.replace("#", "");
};
// Blend 2 hex colors: c1 is placed over c2, blend is c1's opacity.
/**
* Blends two hex colors. c1 is placed over c2, blend is c1's opacity.
* @param c1 - The first hex color.
* @param c2 - The second hex color.
* @param blend - The blend percentage (0-100).
* @returns The blended hex color.
*/
export const hexBlend = (c1: string, c2: string, blend = 50): string => {
let color = "";
c1 = expandHex(c1);
c2 = expandHex(c2);
let color = "";
for (let i = 0; i <= 5; i += 2) {
const h1 = parseInt(c1.substring(i, i + 2), 16);
const h2 = parseInt(c2.substring(i, i + 2), 16);
let hex = Math.floor(h2 + (h1 - h2) * (blend / 100)).toString(16);
while (hex.length < 2) hex = "0" + hex;
const hex = Math.floor(h2 + (h1 - h2) * (blend / 100))
.toString(16)
.padStart(2, "0");
color += hex;
}
return `#${color}`;

View File

@@ -1,28 +1,49 @@
export const luminosity = (rgb: [number, number, number]): number => {
// http://www.w3.org/TR/WCAG20/#relativeluminancedef
const lum: [number, number, number] = [0, 0, 0];
for (let i = 0; i < rgb.length; i++) {
const chan = rgb[i] / 255;
lum[i] = chan <= 0.03928 ? chan / 12.92 : ((chan + 0.055) / 1.055) ** 2.4;
}
import { wcagLuminance, wcagContrast } from "culori";
return 0.2126 * lum[0] + 0.7152 * lum[1] + 0.0722 * lum[2];
};
/**
* Calculates the luminosity of an RGB color.
* @param rgb - The RGB color to calculate the luminosity of.
* @returns The luminosity of the color.
*/
export const luminosity = (rgb: [number, number, number]): number =>
wcagLuminance({
mode: "rgb",
r: rgb[0] / 255,
g: rgb[1] / 255,
b: rgb[2] / 255,
});
/**
* Calculates the contrast ratio between two RGB colors.
* @param color1 - The first color to calculate the contrast ratio of.
* @param color2 - The second color to calculate the contrast ratio of.
* @returns The contrast ratio between the two colors.
*/
export const rgbContrast = (
color1: [number, number, number],
color2: [number, number, number]
) => {
const lum1 = luminosity(color1);
const lum2 = luminosity(color2);
if (lum1 > lum2) {
return (lum1 + 0.05) / (lum2 + 0.05);
}
return (lum2 + 0.05) / (lum1 + 0.05);
};
) =>
wcagContrast(
{
mode: "rgb",
r: color1[0] / 255,
g: color1[1] / 255,
b: color1[2] / 255,
},
{
mode: "rgb",
r: color2[0] / 255,
g: color2[1] / 255,
b: color2[2] / 255,
}
);
/**
* Calculates the contrast ratio between two RGB colors.
* @param rgb1 - The first color to calculate the contrast ratio of.
* @param rgb2 - The second color to calculate the contrast ratio of.
* @returns The contrast ratio between the two colors.
*/
export const getRGBContrastRatio = (
rgb1: [number, number, number],
rgb2: [number, number, number]

View File

@@ -1,103 +0,0 @@
import { tinykeys } from "tinykeys";
import { canOverrideAlphanumericInput } from "../dom/can-override-input";
import type { HomeAssistant } from "../../types";
export type ShortcutHandler = (event: KeyboardEvent) => void;
export interface ShortcutManager {
/**
* 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.
*/
add: (shortcuts: Record<string, ShortcutHandler>) => void;
/**
* 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.
*/
remove: (keys?: string[]) => void;
}
interface ShortcutEntry {
keys: Set<string>;
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> = {};
for (const [key, handler] of Object.entries(shortcuts)) {
wrappedShortcuts[key] = (event: KeyboardEvent) => {
if (!canOverrideAlphanumericInput(event.composedPath())) {
return;
}
if (window.getSelection()?.toString()) {
return;
}
handler(event);
};
}
// Underlying implementation (tinykeys for now)
return tinykeys(window, wrappedShortcuts);
}
/**
* Create a shortcut manager that can add and dispose shortcuts.
*
* @param hass - Home Assistant context to check if shortcuts are enabled.
* @returns A shortcut manager containing the add and remove methods.
*/
export function createShortcutManager(hass?: HomeAssistant): ShortcutManager {
const shortcutEntries: ShortcutEntry[] = [];
return {
add(shortcuts: Record<string, ShortcutHandler>) {
// Skip registration if shortcuts are disabled
if (hass && !hass.enableShortcuts) {
return;
}
const disposer = registerShortcuts(shortcuts);
const keys = new Set(Object.keys(shortcuts));
const entry: ShortcutEntry = { keys, disposer };
shortcutEntries.push(entry);
},
remove(keys?: string[]) {
if (keys) {
const entriesToRemove: ShortcutEntry[] = [];
for (const entry of shortcutEntries) {
if (keys.some((key) => entry.keys.has(key))) {
entry.disposer();
entriesToRemove.push(entry);
}
}
for (const entry of entriesToRemove) {
const index = shortcutEntries.indexOf(entry);
if (index !== -1) {
shortcutEntries.splice(index, 1);
}
}
} else {
shortcutEntries.forEach((entry) => entry.disposer());
shortcutEntries.length = 0;
}
},
};
}

View File

@@ -974,7 +974,7 @@ export class HaChartBase extends LitElement {
right: 4px;
display: flex;
flex-direction: column;
gap: 4px;
gap: var(--ha-space-1);
}
.chart-controls.small {
top: 0;
@@ -1011,7 +1011,7 @@ export class HaChartBase extends LitElement {
flex-wrap: wrap;
justify-content: center;
align-items: center;
gap: 8px;
gap: var(--ha-space-2);
}
.chart-legend li {
height: 24px;

View File

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

View File

@@ -54,7 +54,7 @@ export class HaBadge extends LitElement {
flex-direction: row;
align-items: center;
justify-content: center;
gap: 8px;
gap: var(--ha-space-2);
height: var(--ha-badge-size, 36px);
min-width: var(--ha-badge-size, 36px);
padding: 0px 12px;

View File

@@ -76,7 +76,7 @@ export class HaCopyTextfield extends LitElement {
.container {
display: flex;
align-items: center;
gap: 8px;
gap: var(--ha-space-2);
margin-top: 8px;
}

View File

@@ -343,7 +343,7 @@ export class HaDateRangePicker extends LitElement {
.date-range-inputs {
display: flex;
align-items: center;
gap: 8px;
gap: var(--ha-space-2);
}
.date-range-ranges {

View File

@@ -77,8 +77,8 @@ export class HaFormGrid extends LitElement implements HaFormElement {
var(--form-grid-column-count, auto-fit),
minmax(var(--form-grid-min-width, 200px), 1fr)
);
grid-column-gap: 8px;
grid-row-gap: 24px;
grid-column-gap: var(--ha-space-2);
grid-row-gap: var(--ha-space-6);
}
:host > ha-form {
display: block;

View File

@@ -156,7 +156,7 @@ export class HaFormOptionalActions extends LitElement implements HaFormElement {
:host {
display: flex !important;
flex-direction: column;
gap: 24px;
gap: var(--ha-space-6);
}
:host ha-form {
display: block;

View File

@@ -57,7 +57,7 @@ export class HaFormfield extends FormfieldBase {
}
.mdc-form-field {
align-items: var(--ha-formfield-align-items, center);
gap: 4px;
gap: var(--ha-space-1);
}
.mdc-form-field > label {
direction: var(--direction);

View File

@@ -227,7 +227,7 @@ export class HaGridSizeEditor extends LitElement {
"row-slider preview";
grid-template-rows: auto auto;
grid-template-columns: auto 1fr;
gap: 8px;
gap: var(--ha-space-2);
}
#columns {
grid-area: column-slider;

View File

@@ -104,7 +104,7 @@ export class HaIconButtonToolbar extends LitElement {
background-color: transparent;
padding-right: 4px;
height: var(--icon-button-toolbar-height);
gap: 8px;
gap: var(--ha-space-2);
}
.icon-toolbar-button {

View File

@@ -105,7 +105,7 @@ export class HaPickerField extends LitElement {
--md-list-item-bottom-space: 0px;
--md-list-item-leading-space: 8px;
--md-list-item-trailing-space: 8px;
--ha-md-list-item-gap: 8px;
--ha-md-list-item-gap: var(--ha-space-2);
/* Remove the default focus ring */
--md-focus-ring-width: 0px;
--md-focus-ring-duration: 0s;

View File

@@ -39,22 +39,24 @@ class HaSegmentedBar extends LitElement {
<slot name="extra"></slot>
</div>
<div class="bar">
${this.segments.map((segment) => {
const bar = html`<div
style=${styleMap({
width: `${(segment.value / totalValue) * 100}%`,
backgroundColor: segment.color,
})}
></div>`;
return this.hideTooltip && !segment.label
? bar
: html`
<ha-tooltip>
<span slot="content">${segment.label}</span>
${bar}
</ha-tooltip>
`;
})}
${this.segments.map(
(segment, index) => html`
${this.hideTooltip || !segment.label
? nothing
: html`
<ha-tooltip for="segment-${index}" placement="top">
${segment.label}
</ha-tooltip>
`}
<div
id="segment-${index}"
style=${styleMap({
width: `${(segment.value / totalValue) * 100}%`,
backgroundColor: segment.color,
})}
></div>
`
)}
</div>
${this.hideLegend
? nothing
@@ -88,7 +90,7 @@ class HaSegmentedBar extends LitElement {
.heading {
display: flex;
flex-direction: row;
gap: 8px;
gap: var(--ha-space-2);
}
.heading .title {
flex: 1;
@@ -123,7 +125,7 @@ class HaSegmentedBar extends LitElement {
display: flex;
flex-direction: row;
flex-wrap: wrap;
gap: 12px;
gap: var(--ha-space-3);
margin: 12px 0;
padding: 0;
list-style: none;
@@ -131,7 +133,7 @@ class HaSegmentedBar extends LitElement {
.legend li {
display: flex;
align-items: center;
gap: 4px;
gap: var(--ha-space-1);
font-size: var(--ha-font-size-s);
}
.legend li .bullet {

View File

@@ -116,7 +116,7 @@ export class HaSelectBox extends LitElement {
.list {
display: grid;
grid-template-columns: repeat(var(--columns, 1), minmax(0, 1fr));
gap: 12px;
gap: var(--ha-space-3);
}
.option {
position: relative;
@@ -128,7 +128,7 @@ export class HaSelectBox extends LitElement {
align-items: center;
justify-content: space-between;
padding: 12px;
gap: 8px;
gap: var(--ha-space-2);
overflow: hidden;
cursor: pointer;
}
@@ -137,7 +137,7 @@ export class HaSelectBox extends LitElement {
position: relative;
display: flex;
flex-direction: row;
gap: 8px;
gap: var(--ha-space-2);
min-width: 0;
width: 100%;
}
@@ -148,7 +148,7 @@ export class HaSelectBox extends LitElement {
.option .content .text {
display: flex;
flex-direction: column;
gap: 4px;
gap: var(--ha-space-1);
min-width: 0;
flex: 1;
}

View File

@@ -89,7 +89,7 @@ export class HaButtonToggleSelector extends LitElement {
display: flex;
justify-content: space-between;
flex-wrap: wrap;
gap: 8px;
gap: var(--ha-space-2);
align-items: center;
}
@media all and (max-width: 600px) {

View File

@@ -321,7 +321,7 @@ export class HaMediaSelector extends LitElement {
display: flex;
align-items: center;
padding: 8px;
gap: 12px;
gap: var(--ha-space-3);
}
ha-card .thumbnail {
width: 40px;

View File

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

View File

@@ -297,7 +297,7 @@ export class HaObjectSelector extends LitElement {
return [
css`
ha-md-list {
gap: 8px;
gap: var(--ha-space-2);
}
ha-md-list-item {
border: 1px solid var(--divider-color);

View File

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

View File

@@ -217,7 +217,7 @@ class DialogJoinMediaPlayers extends LitElement {
.content {
display: flex;
flex-direction: column;
row-gap: 16px;
row-gap: var(--ha-space-4);
}
ha-dialog-header ha-button {

View File

@@ -1138,7 +1138,7 @@ export class HaMediaPlayerBrowse extends LitElement {
auto-fit,
minmax(var(--media-browse-item-size, 175px), 0.1fr)
);
grid-gap: 16px;
grid-gap: var(--ha-space-4);
padding: 16px;
}

View File

@@ -54,7 +54,7 @@ class HaMediaPlayerToggle extends LitElement {
.list-item {
display: grid;
grid-template-columns: auto 1fr auto;
column-gap: 16px;
column-gap: var(--ha-space-4);
align-items: center;
width: 100%;
}

View File

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

View File

@@ -325,7 +325,7 @@ class StepFlowCreateEntry extends LitElement {
.device-info {
display: flex;
align-items: center;
gap: 8px;
gap: var(--ha-space-2);
}
.device-info img {
width: 40px;

View File

@@ -209,7 +209,7 @@ export class DialogEnterCode
display: grid;
grid-template-columns: repeat(var(--keypad-columns), auto);
grid-auto-rows: auto;
grid-gap: 24px;
grid-gap: var(--ha-space-6);
justify-items: center;
align-items: center;
}

View File

@@ -32,7 +32,7 @@ export class HaMoreInfoControlSelectContainer extends LitElement {
display: flex;
flex-direction: row;
justify-content: flex-start;
gap: 12px;
gap: var(--ha-space-3);
margin: auto;
overflow: auto;
-ms-overflow-style: none; /* IE and Edge */

View File

@@ -202,13 +202,13 @@ class MoreInfoSirenAdvancedControls extends LitElement {
.options {
display: flex;
flex-direction: column;
gap: 16px;
gap: var(--ha-space-4);
}
.controls {
display: flex;
flex-direction: row;
justify-content: center;
gap: 16px;
gap: var(--ha-space-4);
margin-top: 16px;
}
ha-control-button {

View File

@@ -107,7 +107,7 @@ class MoreInfoCamera extends LitElement {
box-sizing: border-box;
padding: 16px;
z-index: 1;
gap: 8px;
gap: var(--ha-space-2);
}
`;
}

View File

@@ -181,7 +181,7 @@ class MoreInfoLock extends LitElement {
display: flex;
align-items: center;
flex-direction: row;
gap: 8px;
gap: var(--ha-space-2);
font-weight: var(--ha-font-weight-medium);
color: var(--success-color);
}

View File

@@ -48,10 +48,10 @@ class MoreInfoMediaPlayer extends LitElement {
@property({ attribute: false }) public stateObj?: MediaPlayerEntity;
private _formateDuration(duration: number) {
private _formatDuration(duration: number) {
const hours = Math.floor(duration / 3600);
const minutes = Math.floor((duration % 3600) / 60);
const seconds = duration % 60;
const seconds = Math.floor(duration % 60);
return formatDurationDigital(this.hass.locale, {
hours,
minutes,
@@ -260,12 +260,12 @@ class MoreInfoMediaPlayer extends LitElement {
const controls = computeMediaControls(stateObj, true);
const coverUrl = stateObj.attributes.entity_picture || "";
const playerObj = new HassMediaPlayerEntity(this.hass, this.stateObj);
const position = Math.floor(playerObj.currentProgress) || 0;
const duration = stateObj.attributes.media_duration || 0;
const remaining = duration - position;
const durationFormated =
remaining > 0 ? this._formateDuration(remaining) : 0;
const postionFormated = this._formateDuration(position);
const position = Math.max(Math.floor(playerObj.currentProgress || 0), 0);
const duration = Math.max(stateObj.attributes.media_duration || 0, 0);
const remaining = Math.max(duration - position, 0);
const remainingFormatted = this._formatDuration(remaining);
const positionFormatted = this._formatDuration(position);
const primaryTitle = playerObj.primaryTitle;
const secondaryTitle = playerObj.secondaryTitle;
const turnOn = controls?.find((c) => c.action === "turn_on");
@@ -323,11 +323,10 @@ class MoreInfoMediaPlayer extends LitElement {
@change=${this._handleMediaSeekChanged}
?disabled=${!stateActive(stateObj) ||
!supportsFeature(stateObj, MediaPlayerEntityFeature.SEEK)}
></ha-slider>
<div class="position-info-row">
<span class="position-time">${postionFormated}</span>
<span class="duration-time">${durationFormated}</span>
</div>
>
<span slot="reference">${positionFormatted}</span>
<span slot="reference">${remainingFormatted}</span>
</ha-slider>
</div>
`
: nothing}
@@ -443,7 +442,7 @@ class MoreInfoMediaPlayer extends LitElement {
:host {
display: flex;
flex-direction: column;
gap: 24px;
gap: var(--ha-space-6);
margin-top: 0;
}
@@ -511,7 +510,7 @@ class MoreInfoMediaPlayer extends LitElement {
.volume {
display: flex;
align-items: center;
gap: 12px;
gap: var(--ha-space-3);
margin-left: 8px;
}
@@ -548,13 +547,8 @@ class MoreInfoMediaPlayer extends LitElement {
flex-direction: column;
}
.position-info-row {
display: flex;
flex-direction: row;
justify-content: space-between;
.position-bar ha-slider::part(references) {
color: var(--secondary-text-color);
padding: 0 8px;
font-size: var(--ha-font-size-s);
}
.media-info-row {
@@ -597,7 +591,7 @@ class MoreInfoMediaPlayer extends LitElement {
.bottom-controls {
display: flex;
flex-direction: column;
gap: 24px;
gap: var(--ha-space-6);
align-self: center;
width: 320px;
}

View File

@@ -508,7 +508,7 @@ class MoreInfoUpdate extends LitElement {
box-sizing: border-box;
padding: 16px;
z-index: 1;
gap: 8px;
gap: var(--ha-space-2);
}
a {

View File

@@ -269,7 +269,7 @@ class DialogShortcuts extends LitElement {
display: flex;
flex-direction: row;
align-items: center;
gap: 8px;
gap: var(--ha-space-2);
margin: 4px 0;
}

View File

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

View File

@@ -127,7 +127,7 @@ export class CloudStepIntro extends LitElement {
.features {
display: flex;
flex-direction: column;
grid-gap: 16px;
grid-gap: var(--ha-space-4);
padding: 16px;
}
.feature {

View File

@@ -427,7 +427,7 @@ export class HaVoiceAssistantSetupStepPipeline extends LitElement {
width: 100%;
height: 10px;
display: flex;
gap: 4px;
gap: var(--ha-space-1);
margin: 8px 0;
}
.segment {

View File

@@ -368,7 +368,7 @@ export class HaVoiceAssistantSetupStepSuccess extends LitElement {
margin-top: 24px;
}
.rows {
gap: 16px;
gap: var(--ha-space-4);
display: flex;
flex-direction: column;
}

View File

@@ -182,7 +182,7 @@ class HassSubpage extends LitElement {
display: flex;
flex-wrap: wrap;
justify-content: flex-end;
gap: 8px;
gap: var(--ha-space-2);
}
:host([narrow]) #fab.tabs {
bottom: calc(84px + var(--safe-area-inset-bottom, 0px));

View File

@@ -740,7 +740,7 @@ export class HaTabsSubpageDataTable extends KeyboardShortcutMixin(LitElement) {
width: 100%;
justify-content: space-between;
padding: 0 16px;
gap: 16px;
gap: var(--ha-space-4);
box-sizing: border-box;
background: var(--primary-background-color);
border-bottom: 1px solid var(--divider-color);
@@ -823,7 +823,7 @@ export class HaTabsSubpageDataTable extends KeyboardShortcutMixin(LitElement) {
display: flex;
align-items: center;
min-width: 100%;
gap: 16px;
gap: var(--ha-space-4);
padding: 0 16px;
box-sizing: border-box;
overflow-x: scroll;
@@ -852,7 +852,7 @@ export class HaTabsSubpageDataTable extends KeyboardShortcutMixin(LitElement) {
.selection-controls {
display: flex;
align-items: center;
gap: 8px;
gap: var(--ha-space-2);
}
.selection-controls p {
@@ -864,7 +864,7 @@ export class HaTabsSubpageDataTable extends KeyboardShortcutMixin(LitElement) {
.center-vertical {
display: flex;
align-items: center;
gap: 8px;
gap: var(--ha-space-2);
}
.relative {

View File

@@ -362,7 +362,7 @@ class HassTabsSubpage extends LitElement {
display: flex;
flex-wrap: wrap;
justify-content: flex-end;
gap: 8px;
gap: var(--ha-space-2);
}
:host([narrow]) #fab.tabs {
bottom: calc(84px + var(--safe-area-inset-bottom, 0px));

View File

@@ -212,7 +212,7 @@ class OnboardingIntegrations extends SubscribeMixin(LitElement) {
margin-top: 24px;
display: grid;
grid-template-columns: repeat(auto-fill, minmax(106px, 1fr));
row-gap: 24px;
row-gap: var(--ha-space-6);
}
.more {
display: flex;

View File

@@ -64,8 +64,8 @@ class OnboardingWelcomeLinks extends LitElement {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(150px, 1fr));
margin-top: 16px;
column-gap: 16px;
row-gap: 16px;
column-gap: var(--ha-space-4);
row-gap: var(--ha-space-4);
}
@media (max-width: 550px) {
:host {

View File

@@ -123,7 +123,7 @@ class OnboardingRestoreBackupCloudLogin extends LitElement {
font-size: var(--ha-font-size-2xl);
display: flex;
align-items: center;
gap: 16px;
gap: var(--ha-space-4);
}
h2 img {
width: 48px;

View File

@@ -332,7 +332,7 @@ class OnboardingRestoreBackupRestore extends LitElement {
display: flex;
align-items: center;
flex-direction: row;
gap: 8px;
gap: var(--ha-space-2);
line-height: var(--ha-line-height-condensed);
}
h2 {

View File

@@ -736,7 +736,7 @@ class DialogAddAutomationElement
}
.shortcut-label {
display: flex;
gap: 12px;
gap: var(--ha-space-3);
justify-content: space-between;
}
.shortcut-label .supporting-text {

View File

@@ -92,7 +92,7 @@ class DialogPasteReplace extends LitElement implements HassDialog {
}
div[slot="primaryAction"] {
display: flex;
gap: 8px;
gap: var(--ha-space-2);
}
`,
];

View File

@@ -118,7 +118,7 @@ export const manualEditorStyles = css`
var(--ha-automation-editor-max-width) -
${CONTENT_MIN_WIDTH}px - var(--mdc-drawer-width, 0px)
);
--sidebar-gap: 16px;
--sidebar-gap: var(--ha-space-4);
}
.fab-positioner {
@@ -188,7 +188,7 @@ export const automationRowsStyles = css`
.rows {
display: flex;
flex-direction: column;
gap: 16px;
gap: var(--ha-space-4);
}
.rows.no-sidebar {
margin-inline-end: 0;
@@ -225,7 +225,7 @@ export const automationRowsStyles = css`
.buttons {
display: flex;
flex-wrap: wrap;
gap: 8px;
gap: var(--ha-space-2);
order: 1;
}
`;
@@ -244,7 +244,7 @@ export const overflowStyles = css`
.overflow-label {
display: flex;
justify-content: space-between;
gap: 12px;
gap: var(--ha-space-3);
white-space: nowrap;
}
.overflow-label .shortcut {

View File

@@ -806,7 +806,7 @@ export default class HaAutomationTriggerRow extends LitElement {
display: flex;
justify-content: center;
align-items: center;
gap: 4px;
gap: var(--ha-space-1);
line-height: 1;
padding: 0;
}

View File

@@ -313,13 +313,13 @@ class HaBackupConfigAgents extends LitElement {
display: flex;
align-items: center;
flex-direction: row;
gap: 8px;
gap: var(--ha-space-2);
line-height: var(--ha-line-height-condensed);
}
.unencrypted-warning {
display: flex;
align-items: center;
gap: 4px;
gap: var(--ha-space-1);
}
.dot {
display: block;
@@ -341,7 +341,7 @@ class HaBackupConfigAgents extends LitElement {
align-items: flex-start;
flex-direction: column;
justify-content: flex-start;
gap: 4px;
gap: var(--ha-space-1);
}
}
`;

View File

@@ -119,7 +119,7 @@ class HaBackupAgentsPicker extends LitElement {
display: flex;
flex-direction: row;
align-items: center;
gap: 16px;
gap: var(--ha-space-4);
font-size: var(--ha-font-size-l);
font-weight: var(--ha-font-weight-normal);
line-height: var(--ha-line-height-normal);

View File

@@ -110,7 +110,7 @@ class HaBackupDetailsRestore extends LitElement {
max-width: 690px;
width: 100%;
margin: 0 auto;
gap: 24px;
gap: var(--ha-space-6);
display: grid;
}
.card-content {
@@ -133,7 +133,7 @@ class HaBackupDetailsRestore extends LitElement {
display: flex;
align-items: center;
flex-direction: row;
gap: 8px;
gap: var(--ha-space-2);
line-height: var(--ha-line-height-condensed);
}
`;

View File

@@ -138,7 +138,7 @@ class HaBackupDetailsSummary extends LitElement {
max-width: 690px;
width: 100%;
margin: 0 auto;
gap: 24px;
gap: var(--ha-space-6);
display: grid;
}
.card-content {
@@ -168,7 +168,7 @@ class HaBackupDetailsSummary extends LitElement {
display: flex;
align-items: center;
flex-direction: row;
gap: 8px;
gap: var(--ha-space-2);
line-height: var(--ha-line-height-condensed);
}
`;

View File

@@ -35,7 +35,7 @@ class SupervisorFormfieldLabel extends LitElement {
:host {
display: flex;
flex-direction: row;
gap: 16px;
gap: var(--ha-space-4);
align-items: center;
}
.label {

View File

@@ -74,8 +74,8 @@ class HaBackupSummaryCard extends LitElement {
.summary {
display: flex;
flex-direction: row;
column-gap: 16px;
row-gap: 8px;
column-gap: var(--ha-space-4);
row-gap: var(--ha-space-2);
align-items: center;
padding: 16px;
padding-bottom: 8px;

View File

@@ -117,7 +117,7 @@ class HaBackupOverviewBackups extends LitElement {
padding: 28px 20px 0;
max-width: 690px;
margin: 0 auto;
gap: 24px;
gap: var(--ha-space-6);
display: flex;
flex-direction: column;
margin-bottom: calc(72px + var(--safe-area-inset-bottom));

View File

@@ -61,7 +61,7 @@ class HaBackupOverviewBackups extends LitElement {
display: flex;
flex-direction: row;
align-items: center;
gap: 16px;
gap: var(--ha-space-4);
}
.icon {
position: relative;

View File

@@ -303,7 +303,7 @@ class HaBackupBackupsSummary extends LitElement {
padding: 28px 20px 0;
max-width: 690px;
margin: 0 auto;
gap: 24px;
gap: var(--ha-space-6);
display: flex;
flex-direction: column;
margin-bottom: 24px;

View File

@@ -334,7 +334,7 @@ class HaBackupOverviewBackups extends LitElement {
display: flex;
flex-direction: row;
align-items: center;
gap: 16px;
gap: var(--ha-space-4);
}
p {
margin: 0;

View File

@@ -579,7 +579,7 @@ class DialogBackupOnboarding extends LitElement implements HassDialog {
display: flex;
flex-direction: row;
align-items: center;
gap: 24px;
gap: var(--ha-space-6);
}
.encryption-key p {
margin: 0;

View File

@@ -305,7 +305,7 @@ class DialogChangeBackupEncryptionKey extends LitElement implements HassDialog {
display: flex;
flex-direction: row;
align-items: center;
gap: 24px;
gap: var(--ha-space-6);
}
.encryption-key p {
margin: 0;

View File

@@ -231,7 +231,7 @@ class DialogSetBackupEncryptionKey extends LitElement implements HassDialog {
display: flex;
flex-direction: row;
align-items: center;
gap: 24px;
gap: var(--ha-space-6);
}
.encryption-key p {
margin: 0;

View File

@@ -153,7 +153,7 @@ class DialogShowBackupEncryptionKey extends LitElement implements HassDialog {
display: flex;
flex-direction: row;
align-items: center;
gap: 24px;
gap: var(--ha-space-6);
}
.encryption-key p {
margin: 0;

View File

@@ -189,7 +189,7 @@ class HaConfigBackupBackups extends SubscribeMixin(LitElement) {
0
);
return html`
<div style="display: flex; gap: 4px;">
<div style="display: flex; gap: var(--ha-space-1);">
${displayedAgentIds.map((agentId) => {
const name = computeBackupAgentName(
this.hass.localize,

View File

@@ -356,7 +356,7 @@ class HaConfigBackupDetails extends LitElement {
padding: 28px 20px 0;
max-width: 690px;
margin: 0 auto;
gap: 24px;
gap: var(--ha-space-6);
display: grid;
margin-bottom: 24px;
}
@@ -399,7 +399,7 @@ class HaConfigBackupDetails extends LitElement {
display: flex;
align-items: center;
flex-direction: row;
gap: 8px;
gap: var(--ha-space-2);
line-height: var(--ha-line-height-condensed);
}
.dot {

View File

@@ -326,7 +326,7 @@ class HaConfigBackupDetails extends LitElement {
padding: 28px 20px 0;
max-width: 690px;
margin: 0 auto;
gap: 24px;
gap: var(--ha-space-6);
display: grid;
margin-bottom: 24px;
}
@@ -376,7 +376,7 @@ class HaConfigBackupDetails extends LitElement {
display: flex;
align-items: center;
flex-direction: row;
gap: 8px;
gap: var(--ha-space-2);
line-height: var(--ha-line-height-condensed);
}
.dot {

View File

@@ -248,7 +248,7 @@ class HaConfigBackupOverview extends LitElement {
padding: 28px 20px 0;
max-width: 690px;
margin: 0 auto;
gap: 24px;
gap: var(--ha-space-6);
display: flex;
flex-direction: column;
margin-bottom: calc(var(--safe-area-inset-bottom) + 72px);

View File

@@ -510,7 +510,7 @@ class HaConfigBackupSettings extends LitElement {
padding: 28px 20px 0;
max-width: 690px;
margin: 0 auto;
gap: 24px;
gap: var(--ha-space-6);
display: flex;
flex-direction: column;
margin-bottom: 24px;
@@ -532,7 +532,7 @@ class HaConfigBackupSettings extends LitElement {
}
.cloud-info .cloud-header {
display: flex;
gap: 16px;
gap: var(--ha-space-4);
font-size: var(--ha-font-size-xl);
align-items: center;
padding: 16px;

View File

@@ -129,7 +129,7 @@ export class DialogSupportPackage extends LitElement {
}
.actions {
display: flex;
gap: 8px;
gap: var(--ha-space-2);
justify-content: flex-end;
}
hr {

View File

@@ -59,7 +59,7 @@ export class DashboardCard extends LitElement {
padding: 12px;
display: block;
text-align: var(--float-start);
gap: 8px;
gap: var(--ha-space-2);
}
.preview {
padding: 16px;

View File

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

View File

@@ -921,7 +921,7 @@ class HaConfigIntegrationPage extends SubscribeMixin(LitElement) {
.header {
display: flex;
flex-wrap: wrap;
gap: 24px;
gap: var(--ha-space-6);
align-items: center;
justify-content: space-between;
margin-bottom: 24px;
@@ -932,7 +932,7 @@ class HaConfigIntegrationPage extends SubscribeMixin(LitElement) {
}
.title {
display: flex;
gap: 4px;
gap: var(--ha-space-1);
flex-direction: column;
justify-content: space-between;
}
@@ -991,7 +991,7 @@ class HaConfigIntegrationPage extends SubscribeMixin(LitElement) {
.actions {
display: flex;
flex-wrap: wrap;
gap: 8px;
gap: var(--ha-space-2);
}
.section {
width: 100%;
@@ -1013,7 +1013,7 @@ class HaConfigIntegrationPage extends SubscribeMixin(LitElement) {
.integration-info {
display: flex;
align-items: center;
gap: 8px;
gap: var(--ha-space-2);
}
.integration-info ha-svg-icon {
min-width: 24px;

View File

@@ -1013,7 +1013,7 @@ class HaConfigIntegrationsDashboard extends KeyboardShortcutMixin(
z-index: 2;
background-color: var(--primary-background-color);
padding: 0 16px;
gap: 16px;
gap: var(--ha-space-4);
box-sizing: border-box;
border-bottom: 1px solid var(--divider-color);
}

View File

@@ -91,7 +91,7 @@ class MatterAddDeviceNew extends LitElement {
justify-content: space-between;
padding: 0 24px;
box-sizing: border-box;
gap: 16px;
gap: var(--ha-space-4);
width: 100%;
max-width: 400px;
}

View File

@@ -143,7 +143,9 @@ export class ThreadConfigPanel extends SubscribeMixin(LitElement) {
slot="fab"
@click=${this._importExternalThreadCredentials}
extended
label="Send credentials to Home Assistant"
.label=${this.hass.localize(
"ui.panel.config.thread.thread_network_send_credentials_ha"
)}
><ha-svg-icon slot="icon" .path=${mdiCellphoneKey}></ha-svg-icon
></ha-fab>`
: nothing}
@@ -310,7 +312,9 @@ export class ThreadConfigPanel extends SubscribeMixin(LitElement) {
<ha-button
.datasetId=${network.dataset.dataset_id}
@click=${this._setPreferred}
>Make preferred network</ha-button
>${this.hass.localize(
"ui.panel.config.thread.thread_network_make_preferred"
)}</ha-button
>
</div>`
: ""}
@@ -322,7 +326,9 @@ export class ThreadConfigPanel extends SubscribeMixin(LitElement) {
size="small"
.networkDataset=${network.dataset}
@click=${this._sendCredentials}
>Send credentials to phone</ha-button
>${this.hass.localize(
"ui.panel.config.thread.thread_network_send_credentials_phone"
)}</ha-button
>
</div>`
: ""}

View File

@@ -209,7 +209,7 @@ class ZWaveJSCapabilityThermostatSetback extends LitElement {
display: flex;
flex-direction: column;
align-items: flex-start;
gap: 16px;
gap: var(--ha-space-4);
}
:host > ha-select {
width: 100%;
@@ -225,7 +225,7 @@ class ZWaveJSCapabilityThermostatSetback extends LitElement {
.setback-state {
width: 100%;
display: flex;
gap: 16px;
gap: var(--ha-space-4);
}
.setback-state ha-select,
ha-textfield {

View File

@@ -362,7 +362,7 @@ class DialogZWaveJSRemoveNode extends LitElement {
display: flex;
align-items: center;
flex-direction: column;
gap: 16px;
gap: var(--ha-space-4);
text-align: center;
}

View File

@@ -997,7 +997,7 @@ class ZWaveJSConfigDashboard extends SubscribeMixin(LitElement) {
.right-buttons {
display: flex;
gap: 8px;
gap: var(--ha-space-2);
margin-left: auto;
}

View File

@@ -224,7 +224,7 @@ class ZWaveJSCustomParam extends LitElement {
.custom-config-form {
display: flex;
flex-wrap: wrap;
gap: 16px;
gap: var(--ha-space-4);
margin-bottom: 8px;
}

View File

@@ -141,7 +141,7 @@ class DownloadLogsDialog extends LitElement {
display: flex;
flex-direction: column;
align-items: center;
gap: 8px;
gap: var(--ha-space-2);
}
`,
];

View File

@@ -1,8 +1,9 @@
import {
mdiCheck,
mdiCheckCircleOutline,
mdiDelete,
mdiDotsVertical,
mdiOpenInNew,
mdiPencil,
mdiPlus,
} from "@mdi/js";
import type { PropertyValues } from "lit";
@@ -20,10 +21,11 @@ import type {
RowClickedEvent,
SortingChangedEvent,
} from "../../../../components/data-table/ha-data-table";
import "../../../../components/ha-button";
import "../../../../components/ha-fab";
import "../../../../components/ha-icon";
import "../../../../components/ha-button";
import "../../../../components/ha-icon-button";
import "../../../../components/ha-icon-overflow-menu";
import "../../../../components/ha-md-button-menu";
import "../../../../components/ha-md-list-item";
import "../../../../components/ha-svg-icon";
@@ -224,87 +226,94 @@ export class HaConfigLovelaceDashboards extends LitElement {
: html``,
};
columns.url_path = {
columns.actions = {
title: "",
label: localize(
"ui.panel.config.lovelace.dashboards.picker.headers.url"
),
filterable: true,
label: this.hass.localize("ui.panel.config.generic.headers.actions"),
type: "overflow-menu",
showNarrow: true,
template: (dashboard) =>
narrow
? html`
<ha-icon-button
.path=${mdiOpenInNew}
.urlPath=${dashboard.url_path}
@click=${this._navigate}
.label=${this.hass.localize(
"ui.panel.config.lovelace.dashboards.picker.open"
)}
></ha-icon-button>
`
: html`
<ha-button
href="/${dashboard.url_path}"
size="small"
appearance="plain"
>${this.hass.localize(
"ui.panel.config.lovelace.dashboards.picker.open"
)}</ha-button
>
`,
moveable: false,
hideable: false,
template: (dashboard) => html`
<ha-icon-overflow-menu
.hass=${this.hass}
narrow
.items=${[
{
path: mdiPencil,
label: this.hass.localize(
"ui.panel.config.lovelace.dashboards.picker.edit"
),
action: () => this._handleEdit(dashboard),
},
...(this._canDelete(dashboard.url_path)
? [
{
label: this.hass.localize(
"ui.panel.config.lovelace.dashboards.picker.delete"
),
path: mdiDelete,
action: () => this._handleDelete(dashboard),
warning: true,
},
]
: []),
]}
>
</ha-icon-overflow-menu>
`,
};
return columns;
}
);
private _getItems = memoize((dashboards: LovelaceDashboard[]) => {
const defaultMode = (
this.hass.panels?.lovelace?.config as LovelacePanelConfig
).mode;
const defaultUrlPath = this.hass.defaultPanel;
const isDefault = defaultUrlPath === "lovelace";
const result: DataTableItem[] = [
{
icon: "hass:view-dashboard",
title: this.hass.localize("panel.states"),
default: isDefault,
show_in_sidebar: isDefault,
require_admin: false,
url_path: "lovelace",
mode: defaultMode,
filename: defaultMode === "yaml" ? "ui-lovelace.yaml" : "",
iconColor: "var(--primary-color)",
},
];
if (isComponentLoaded(this.hass, "energy")) {
result.push({
icon: "hass:lightning-bolt",
title: this.hass.localize(`ui.panel.config.dashboard.energy.main`),
show_in_sidebar: true,
mode: "storage",
url_path: "energy",
filename: "",
iconColor: "var(--label-badge-yellow)",
default: false,
require_admin: false,
});
}
result.push(
...dashboards
.sort((a, b) =>
stringCompare(a.title, b.title, this.hass.locale.language)
)
.map((dashboard) => ({
private _getItems = memoize(
(dashboards: LovelaceDashboard[], defaultUrlPath: string) => {
const defaultMode = (
this.hass.panels?.lovelace?.config as LovelacePanelConfig
).mode;
const isDefault = defaultUrlPath === "lovelace";
const result: DataTableItem[] = [
{
icon: "hass:view-dashboard",
title: this.hass.localize("panel.states"),
default: isDefault,
show_in_sidebar: isDefault,
require_admin: false,
url_path: "lovelace",
mode: defaultMode,
filename: defaultMode === "yaml" ? "ui-lovelace.yaml" : "",
iconColor: "var(--primary-color)",
},
];
if (isComponentLoaded(this.hass, "energy")) {
result.push({
icon: "hass:lightning-bolt",
title: this.hass.localize(`ui.panel.config.dashboard.energy.main`),
show_in_sidebar: true,
mode: "storage",
url_path: "energy",
filename: "",
...dashboard,
default: defaultUrlPath === dashboard.url_path,
}))
);
return result;
});
iconColor: "var(--label-badge-yellow)",
default: false,
require_admin: false,
});
}
result.push(
...dashboards
.sort((a, b) =>
stringCompare(a.title, b.title, this.hass.locale.language)
)
.map((dashboard) => ({
filename: "",
...dashboard,
default: defaultUrlPath === dashboard.url_path,
}))
);
return result;
}
);
protected render() {
if (!this.hass || this._dashboards === undefined) {
@@ -324,7 +333,7 @@ export class HaConfigLovelaceDashboards extends LitElement {
this._dashboards,
this.hass.localize
)}
.data=${this._getItems(this._dashboards)}
.data=${this._getItems(this._dashboards, this.hass.defaultPanel)}
.initialSorting=${this._activeSorting}
.columnOrder=${this._activeColumnOrder}
.hiddenColumns=${this._activeHiddenColumns}
@@ -332,7 +341,7 @@ export class HaConfigLovelaceDashboards extends LitElement {
@sorting-changed=${this._handleSortingChanged}
.filter=${this._filter}
@search-changed=${this._handleSearchChange}
@row-click=${this._editDashboard}
@row-click=${this._handleRowClicked}
id="url_path"
has-fab
clickable
@@ -370,13 +379,14 @@ export class HaConfigLovelaceDashboards extends LitElement {
this._dashboards = await fetchDashboards(this.hass);
}
private _navigate(ev: Event) {
private _handleRowClicked(ev: CustomEvent) {
ev.stopPropagation();
navigate(`/${(ev.target as any).urlPath}`);
const urlPath = (ev.detail as RowClickedEvent).id;
navigate(`/${urlPath}`);
}
private _editDashboard(ev: CustomEvent) {
const urlPath = (ev.detail as RowClickedEvent).id;
private _handleEdit(item: DataTableItem) {
const urlPath = item.url_path;
if (urlPath === "energy") {
navigate("/config/energy");
@@ -386,6 +396,23 @@ export class HaConfigLovelaceDashboards extends LitElement {
this._openDetailDialog(dashboard, urlPath);
}
private _canDelete(urlPath: string) {
if (urlPath === "lovelace" || urlPath === "energy") {
return false;
}
return true;
}
private _handleDelete = async (item: DataTableItem) => {
const dashboard = this._dashboards.find(
(res) => res.url_path === item.url_path
);
if (!dashboard) {
return;
}
this._deleteDashboard(dashboard);
};
private async _addDashboard() {
showNewDashboardDialog(this, {
selectConfig: async (config) => {
@@ -445,33 +472,44 @@ export class HaConfigLovelaceDashboards extends LitElement {
);
},
removeDashboard: async () => {
const confirm = await showConfirmationDialog(this, {
title: this.hass!.localize(
"ui.panel.config.lovelace.dashboards.confirm_delete_title",
{ dashboard_title: dashboard!.title }
),
text: this.hass!.localize(
"ui.panel.config.lovelace.dashboards.confirm_delete_text"
),
confirmText: this.hass!.localize("ui.common.delete"),
destructive: true,
});
if (!confirm) {
return false;
}
try {
await deleteDashboard(this.hass!, dashboard!.id);
this._dashboards = this._dashboards!.filter(
(res) => res !== dashboard
);
return true;
} catch (_err: any) {
if (!dashboard) {
return false;
}
return this._deleteDashboard(dashboard);
},
});
}
private async _deleteDashboard(
dashboard: LovelaceDashboard
): Promise<boolean> {
if (!this._canDelete(dashboard.url_path)) {
return false;
}
const confirm = await showConfirmationDialog(this, {
title: this.hass!.localize(
"ui.panel.config.lovelace.dashboards.confirm_delete_title",
{ dashboard_title: dashboard.title }
),
text: this.hass!.localize(
"ui.panel.config.lovelace.dashboards.confirm_delete_text"
),
confirmText: this.hass!.localize("ui.common.delete"),
destructive: true,
});
if (!confirm) {
return false;
}
try {
await deleteDashboard(this.hass!, dashboard.id);
this._dashboards = this._dashboards.filter((res) => res !== dashboard);
return true;
} catch (_err: any) {
return false;
}
}
private _handleSortingChanged(ev: CustomEvent) {
this._activeSorting = ev.detail;
}

View File

@@ -444,7 +444,7 @@ class ConfigUrlForm extends SubscribeMixin(LitElement) {
.url-container {
display: flex;
align-items: center;
gap: 8px;
gap: var(--ha-space-2);
margin-top: 8px;
}
.textfield-container {

View File

@@ -803,7 +803,7 @@ export class HassioNetwork extends LitElement {
.address-row {
display: flex;
flex-direction: row;
gap: 8px;
gap: var(--ha-space-2);
align-items: center;
}
.address-row ha-textfield {

View File

@@ -1173,7 +1173,7 @@ export class HaScriptEditor extends UndoRedoMixin<
display: flex;
flex-direction: column;
align-items: center;
gap: 8px;
gap: var(--ha-space-2);
pointer-events: none;
}

View File

@@ -145,7 +145,7 @@ export class CloudDiscover extends LitElement {
.features {
display: grid;
grid-template-columns: auto;
grid-gap: 16px;
grid-gap: var(--ha-space-4);
padding: 16px;
}
@media (min-width: 600px) {

View File

@@ -470,7 +470,7 @@ export class AssistPipelineRunDebug extends LitElement {
.start-buttons {
display: flex;
flex-wrap: wrap;
gap: 8px;
gap: var(--ha-space-2);
align-items: center;
justify-content: center;
}

View File

@@ -146,7 +146,7 @@ class HaPanelDevEvent extends LitElement {
haStyle,
css`
.content {
gap: 16px;
gap: var(--ha-space-4);
padding: 16px;
max-width: 1200px;
margin: auto;

Some files were not shown because too many files have changed in this diff Show More