Compare commits

..

1 Commits

Author SHA1 Message Date
Bram Kragten
29a103e884 Prevent wrap of menu items 2024-04-02 15:09:08 +02:00
54 changed files with 649 additions and 2659 deletions

View File

@@ -187,7 +187,7 @@ export class DemoHaControlSelect extends LitElement {
--mdc-icon-size: 24px; --mdc-icon-size: 24px;
--control-select-color: var(--state-fan-active-color); --control-select-color: var(--state-fan-active-color);
--control-select-thickness: 130px; --control-select-thickness: 130px;
--control-select-border-radius: 36px; --control-select-border-radius: 48px;
} }
.vertical-selects { .vertical-selects {
height: 300px; height: 300px;

View File

@@ -151,7 +151,7 @@ export class DemoHaBarSlider extends LitElement {
--control-slider-background: #ffcf4c; --control-slider-background: #ffcf4c;
--control-slider-background-opacity: 0.2; --control-slider-background-opacity: 0.2;
--control-slider-thickness: 130px; --control-slider-thickness: 130px;
--control-slider-border-radius: 36px; --control-slider-border-radius: 48px;
} }
.vertical-sliders { .vertical-sliders {
height: 300px; height: 300px;

View File

@@ -118,7 +118,7 @@ export class DemoHaControlSwitch extends LitElement {
--control-switch-on-color: var(--green-color); --control-switch-on-color: var(--green-color);
--control-switch-off-color: var(--red-color); --control-switch-off-color: var(--red-color);
--control-switch-thickness: 130px; --control-switch-thickness: 130px;
--control-switch-border-radius: 36px; --control-switch-border-radius: 48px;
--control-switch-padding: 6px; --control-switch-padding: 6px;
--mdc-icon-size: 24px; --mdc-icon-size: 24px;
} }

View File

@@ -4,7 +4,7 @@ build-backend = "setuptools.build_meta"
[project] [project]
name = "home-assistant-frontend" name = "home-assistant-frontend"
version = "20240403.1" version = "20240402.0"
license = {text = "Apache-2.0"} license = {text = "Apache-2.0"}
description = "The Home Assistant frontend" description = "The Home Assistant frontend"
readme = "README.md" readme = "README.md"

View File

@@ -45,8 +45,8 @@ export class HaAssistChip extends MdAssistChip {
margin-inline-start: var(--_icon-label-space); margin-inline-start: var(--_icon-label-space);
} }
::before { ::before {
background: var(--ha-assist-chip-container-color, transparent); background: var(--ha-assist-chip-container-color);
opacity: var(--ha-assist-chip-container-opacity, 1); opacity: var(--ha-assist-chip-container-opacity);
} }
:where(.active)::before { :where(.active)::before {
background: var(--ha-assist-chip-active-container-color); background: var(--ha-assist-chip-active-container-color);

View File

@@ -33,7 +33,6 @@ import "../ha-svg-icon";
import "../search-input"; import "../search-input";
import { filterData, sortData } from "./sort-filter"; import { filterData, sortData } from "./sort-filter";
import { groupBy } from "../../common/util/group-by"; import { groupBy } from "../../common/util/group-by";
import { stringCompare } from "../../common/string/compare";
declare global { declare global {
// for fire event // for fire event
@@ -530,13 +529,7 @@ export class HaDataTable extends LitElement {
const sorted: { const sorted: {
[key: string]: DataTableRowData[]; [key: string]: DataTableRowData[];
} = Object.keys(grouped) } = Object.keys(grouped)
.sort((a, b) => .sort()
stringCompare(
["", "-", "—"].includes(a) ? "zzz" : a,
["", "-", "—"].includes(b) ? "zzz" : b,
this.hass.locale.language
)
)
.reduce((obj, key) => { .reduce((obj, key) => {
obj[key] = grouped[key]; obj[key] = grouped[key];
return obj; return obj;

View File

@@ -1,9 +1,8 @@
import { mdiTextureBox } from "@mdi/js"; import { mdiTextureBox } from "@mdi/js";
import { ComboBoxLitRenderer } from "@vaadin/combo-box/lit"; import { ComboBoxLitRenderer } from "@vaadin/combo-box/lit";
import { HassEntity, UnsubscribeFunc } from "home-assistant-js-websocket"; import { HassEntity, UnsubscribeFunc } from "home-assistant-js-websocket";
import { LitElement, PropertyValues, TemplateResult, html, nothing } from "lit"; import { LitElement, PropertyValues, TemplateResult, html } from "lit";
import { customElement, property, query, state } from "lit/decorators"; import { customElement, property, query, state } from "lit/decorators";
import { styleMap } from "lit/directives/style-map";
import memoizeOne from "memoize-one"; import memoizeOne from "memoize-one";
import { fireEvent } from "../common/dom/fire_event"; import { fireEvent } from "../common/dom/fire_event";
import { computeDomain } from "../common/entity/compute_domain"; import { computeDomain } from "../common/entity/compute_domain";
@@ -12,7 +11,6 @@ import {
ScorableTextItem, ScorableTextItem,
fuzzyFilterSort, fuzzyFilterSort,
} from "../common/string/filter/sequence-matching"; } from "../common/string/filter/sequence-matching";
import { computeRTL } from "../common/util/compute_rtl";
import { AreaRegistryEntry } from "../data/area_registry"; import { AreaRegistryEntry } from "../data/area_registry";
import { import {
DeviceEntityDisplayLookup, DeviceEntityDisplayLookup,
@@ -34,7 +32,6 @@ import "./ha-floor-icon";
import "./ha-icon-button"; import "./ha-icon-button";
import "./ha-list-item"; import "./ha-list-item";
import "./ha-svg-icon"; import "./ha-svg-icon";
import "./ha-tree-indicator";
type ScorableAreaFloorEntry = ScorableTextItem & FloorAreaEntry; type ScorableAreaFloorEntry = ScorableTextItem & FloorAreaEntry;
@@ -44,11 +41,28 @@ interface FloorAreaEntry {
icon: string | null; icon: string | null;
strings: string[]; strings: string[];
type: "floor" | "area"; type: "floor" | "area";
level: number | null;
hasFloor?: boolean; hasFloor?: boolean;
lastArea?: boolean; level: number | null;
} }
const rowRenderer: ComboBoxLitRenderer<FloorAreaEntry> = (item) =>
html`<ha-list-item
graphic="icon"
style=${item.type === "area" && item.hasFloor
? "--mdc-list-side-padding-left: 48px;"
: ""}
>
${item.type === "floor"
? html`<ha-floor-icon slot="graphic" .floor=${item}></ha-floor-icon>`
: item.icon
? html`<ha-icon slot="graphic" .icon=${item.icon}></ha-icon>`
: html`<ha-svg-icon
slot="graphic"
.path=${mdiTextureBox}
></ha-svg-icon>`}
${item.name}
</ha-list-item>`;
@customElement("ha-area-floor-picker") @customElement("ha-area-floor-picker")
export class HaAreaFloorPicker extends SubscribeMixin(LitElement) { export class HaAreaFloorPicker extends SubscribeMixin(LitElement) {
@property({ attribute: false }) public hass!: HomeAssistant; @property({ attribute: false }) public hass!: HomeAssistant;
@@ -137,44 +151,6 @@ export class HaAreaFloorPicker extends SubscribeMixin(LitElement) {
await this.comboBox?.focus(); await this.comboBox?.focus();
} }
private _rowRenderer: ComboBoxLitRenderer<FloorAreaEntry> = (item) => {
const rtl = computeRTL(this.hass);
return html`
<ha-list-item
graphic="icon"
style=${item.type === "area" && item.hasFloor
? rtl
? "--mdc-list-side-padding-right: 48px;"
: "--mdc-list-side-padding-left: 48px;"
: ""}
>
${item.type === "area" && item.hasFloor
? html`<ha-tree-indicator
style=${styleMap({
width: "48px",
position: "absolute",
top: "0px",
left: rtl ? undefined : "8px",
right: rtl ? "8px" : undefined,
transform: rtl ? "scaleX(-1)" : "",
})}
.end=${item.lastArea}
slot="graphic"
></ha-tree-indicator>`
: nothing}
${item.type === "floor"
? html`<ha-floor-icon slot="graphic" .floor=${item}></ha-floor-icon>`
: item.icon
? html`<ha-icon slot="graphic" .icon=${item.icon}></ha-icon>`
: html`<ha-svg-icon
slot="graphic"
.path=${mdiTextureBox}
></ha-svg-icon>`}
${item.name}
</ha-list-item>
`;
};
private _getAreas = memoizeOne( private _getAreas = memoizeOne(
( (
floors: FloorRegistryEntry[], floors: FloorRegistryEntry[],
@@ -388,7 +364,7 @@ export class HaAreaFloorPicker extends SubscribeMixin(LitElement) {
}); });
} }
output.push( output.push(
...floorAreas.map((area, index, array) => ({ ...floorAreas.map((area) => ({
id: area.area_id, id: area.area_id,
type: "area" as const, type: "area" as const,
name: area.name, name: area.name,
@@ -396,7 +372,6 @@ export class HaAreaFloorPicker extends SubscribeMixin(LitElement) {
strings: [area.area_id, ...area.aliases, area.name], strings: [area.area_id, ...area.aliases, area.name],
hasFloor: true, hasFloor: true,
level: null, level: null,
lastArea: index === array.length - 1,
})) }))
); );
}); });
@@ -470,7 +445,7 @@ export class HaAreaFloorPicker extends SubscribeMixin(LitElement) {
.placeholder=${this.placeholder .placeholder=${this.placeholder
? this.hass.areas[this.placeholder]?.name ? this.hass.areas[this.placeholder]?.name
: undefined} : undefined}
.renderer=${this._rowRenderer} .renderer=${rowRenderer}
@filter-changed=${this._filterChanged} @filter-changed=${this._filterChanged}
@opened-changed=${this._openedChanged} @opened-changed=${this._openedChanged}
@value-changed=${this._areaChanged} @value-changed=${this._areaChanged}

View File

@@ -428,8 +428,6 @@ export class HaAreaPicker extends LitElement {
(ev.target as any).value = this._value; (ev.target as any).value = this._value;
this.hass.loadFragmentTranslation("config");
showAreaRegistryDetailDialog(this, { showAreaRegistryDetailDialog(this, {
suggestedName: newValue === ADD_NEW_SUGGESTION_ID ? this._suggestion : "", suggestedName: newValue === ADD_NEW_SUGGESTION_ID ? this._suggestion : "",
createEntry: async (values) => { createEntry: async (values) => {

View File

@@ -1,13 +1,12 @@
import { SelectedDetail } from "@material/mwc-list"; import { SelectedDetail } from "@material/mwc-list";
import "@material/mwc-menu/mwc-menu-surface"; import "@material/mwc-menu/mwc-menu-surface";
import { mdiFilterVariantRemove } from "@mdi/js";
import { css, CSSResultGroup, html, LitElement, nothing } from "lit"; import { css, CSSResultGroup, html, LitElement, nothing } from "lit";
import { customElement, property, state } from "lit/decorators"; import { customElement, property, state } from "lit/decorators";
import { fireEvent } from "../common/dom/fire_event"; import { fireEvent } from "../common/dom/fire_event";
import { Blueprints, fetchBlueprints } from "../data/blueprint";
import { findRelated, RelatedResult } from "../data/search"; import { findRelated, RelatedResult } from "../data/search";
import { haStyleScrollbar } from "../resources/styles";
import type { HomeAssistant } from "../types"; import type { HomeAssistant } from "../types";
import { haStyleScrollbar } from "../resources/styles";
import { Blueprints, fetchBlueprints } from "../data/blueprint";
@customElement("ha-filter-blueprints") @customElement("ha-filter-blueprints")
export class HaFilterBlueprints extends LitElement { export class HaFilterBlueprints extends LitElement {
@@ -36,11 +35,7 @@ export class HaFilterBlueprints extends LitElement {
<div slot="header" class="header"> <div slot="header" class="header">
${this.hass.localize("ui.panel.config.blueprint.caption")} ${this.hass.localize("ui.panel.config.blueprint.caption")}
${this.value?.length ${this.value?.length
? html`<div class="badge">${this.value?.length}</div> ? html`<div class="badge">${this.value?.length}</div>`
<ha-icon-button
.path=${mdiFilterVariantRemove}
@click=${this._clearFilter}
></ha-icon-button>`
: nothing} : nothing}
</div> </div>
${this._blueprints && this._shouldRender ${this._blueprints && this._shouldRender
@@ -133,15 +128,6 @@ export class HaFilterBlueprints extends LitElement {
}); });
} }
private _clearFilter(ev) {
ev.preventDefault();
this.value = undefined;
fireEvent(this, "data-table-filter-changed", {
value: undefined,
items: undefined,
});
}
static get styles(): CSSResultGroup { static get styles(): CSSResultGroup {
return [ return [
haStyleScrollbar, haStyleScrollbar,
@@ -161,10 +147,6 @@ export class HaFilterBlueprints extends LitElement {
display: flex; display: flex;
align-items: center; align-items: center;
} }
.header ha-icon-button {
margin-inline-start: auto;
margin-inline-end: 8px;
}
.badge { .badge {
display: inline-block; display: inline-block;
margin-left: 8px; margin-left: 8px;

View File

@@ -2,7 +2,6 @@ import { ActionDetail, SelectedDetail } from "@material/mwc-list";
import { import {
mdiDelete, mdiDelete,
mdiDotsVertical, mdiDotsVertical,
mdiFilterVariantRemove,
mdiPencil, mdiPencil,
mdiPlus, mdiPlus,
mdiTag, mdiTag,
@@ -69,11 +68,7 @@ export class HaFilterCategories extends SubscribeMixin(LitElement) {
<div slot="header" class="header"> <div slot="header" class="header">
${this.hass.localize("ui.panel.config.category.caption")} ${this.hass.localize("ui.panel.config.category.caption")}
${this.value?.length ${this.value?.length
? html`<div class="badge">${this.value?.length}</div> ? html`<div class="badge">${this.value?.length}</div>`
<ha-icon-button
.path=${mdiFilterVariantRemove}
@click=${this._clearFilter}
></ha-icon-button>`
: nothing} : nothing}
</div> </div>
${this._shouldRender ${this._shouldRender
@@ -259,15 +254,6 @@ export class HaFilterCategories extends SubscribeMixin(LitElement) {
}); });
} }
private _clearFilter(ev) {
ev.preventDefault();
this.value = undefined;
fireEvent(this, "data-table-filter-changed", {
value: undefined,
items: undefined,
});
}
static get styles(): CSSResultGroup { static get styles(): CSSResultGroup {
return [ return [
haStyleScrollbar, haStyleScrollbar,
@@ -288,10 +274,6 @@ export class HaFilterCategories extends SubscribeMixin(LitElement) {
display: flex; display: flex;
align-items: center; align-items: center;
} }
.header ha-icon-button {
margin-inline-start: auto;
margin-inline-end: 8px;
}
.badge { .badge {
display: inline-block; display: inline-block;
margin-left: 8px; margin-left: 8px;

View File

@@ -1,4 +1,3 @@
import { mdiFilterVariantRemove } from "@mdi/js";
import { import {
css, css,
CSSResultGroup, CSSResultGroup,
@@ -14,11 +13,10 @@ import { stringCompare } from "../common/string/compare";
import { computeDeviceName } from "../data/device_registry"; import { computeDeviceName } from "../data/device_registry";
import { findRelated, RelatedResult } from "../data/search"; import { findRelated, RelatedResult } from "../data/search";
import { haStyleScrollbar } from "../resources/styles"; import { haStyleScrollbar } from "../resources/styles";
import { loadVirtualizer } from "../resources/virtualizer";
import type { HomeAssistant } from "../types"; import type { HomeAssistant } from "../types";
import "./ha-check-list-item";
import "./ha-expansion-panel"; import "./ha-expansion-panel";
import "./search-input-outlined"; import "./ha-check-list-item";
import { loadVirtualizer } from "../resources/virtualizer";
@customElement("ha-filter-devices") @customElement("ha-filter-devices")
export class HaFilterDevices extends LitElement { export class HaFilterDevices extends LitElement {
@@ -34,8 +32,6 @@ export class HaFilterDevices extends LitElement {
@state() private _shouldRender = false; @state() private _shouldRender = false;
@state() private _filter?: string;
public willUpdate(properties: PropertyValues) { public willUpdate(properties: PropertyValues) {
super.willUpdate(properties); super.willUpdate(properties);
@@ -55,33 +51,19 @@ export class HaFilterDevices extends LitElement {
<div slot="header" class="header"> <div slot="header" class="header">
${this.hass.localize("ui.panel.config.devices.caption")} ${this.hass.localize("ui.panel.config.devices.caption")}
${this.value?.length ${this.value?.length
? html`<div class="badge">${this.value?.length}</div> ? html`<div class="badge">${this.value?.length}</div>`
<ha-icon-button
.path=${mdiFilterVariantRemove}
@click=${this._clearFilter}
></ha-icon-button>`
: nothing} : nothing}
</div> </div>
${this._shouldRender ${this._shouldRender
? html`<search-input-outlined ? html`<mwc-list class="ha-scrollbar">
.hass=${this.hass} <lit-virtualizer
.filter=${this._filter} .items=${this._devices(this.hass.devices, this.value)}
@value-changed=${this._handleSearchChange} .keyFunction=${this._keyFunction}
.renderItem=${this._renderItem}
@click=${this._handleItemClick}
> >
</search-input-outlined> </lit-virtualizer>
<mwc-list class="ha-scrollbar"> </mwc-list>`
<lit-virtualizer
.items=${this._devices(
this.hass.devices,
this._filter || "",
this.value
)}
.keyFunction=${this._keyFunction}
.renderItem=${this._renderItem}
@click=${this._handleItemClick}
>
</lit-virtualizer>
</mwc-list>`
: nothing} : nothing}
</ha-expansion-panel> </ha-expansion-panel>
`; `;
@@ -90,14 +72,12 @@ export class HaFilterDevices extends LitElement {
private _keyFunction = (device) => device?.id; private _keyFunction = (device) => device?.id;
private _renderItem = (device) => private _renderItem = (device) =>
!device html`<ha-check-list-item
? nothing .value=${device.id}
: html`<ha-check-list-item .selected=${this.value?.includes(device.id)}
.value=${device.id} >
.selected=${this.value?.includes(device.id)} ${computeDeviceName(device, this.hass)}
> </ha-check-list-item>`;
${computeDeviceName(device, this.hass)}
</ha-check-list-item>`;
private _handleItemClick(ev) { private _handleItemClick(ev) {
const listItem = ev.target.closest("ha-check-list-item"); const listItem = ev.target.closest("ha-check-list-item");
@@ -119,7 +99,7 @@ export class HaFilterDevices extends LitElement {
setTimeout(() => { setTimeout(() => {
if (!this.expanded) return; if (!this.expanded) return;
this.renderRoot.querySelector("mwc-list")!.style.height = this.renderRoot.querySelector("mwc-list")!.style.height =
`${this.clientHeight - 49 - 32}px`; // 32px is the height of the search input `${this.clientHeight - 49}px`;
}, 300); }, 300);
} }
} }
@@ -132,28 +112,16 @@ export class HaFilterDevices extends LitElement {
this.expanded = ev.detail.expanded; this.expanded = ev.detail.expanded;
} }
private _handleSearchChange(ev: CustomEvent) { private _devices = memoizeOne((devices: HomeAssistant["devices"], _value) => {
this._filter = ev.detail.value.toLowerCase(); const values = Object.values(devices);
} return values.sort((a, b) =>
stringCompare(
private _devices = memoizeOne( a.name_by_user || a.name || "",
(devices: HomeAssistant["devices"], filter: string, _value) => { b.name_by_user || b.name || "",
const values = Object.values(devices); this.hass.locale.language
return values )
.filter( );
(device) => });
!filter ||
computeDeviceName(device, this.hass).toLowerCase().includes(filter)
)
.sort((a, b) =>
stringCompare(
computeDeviceName(a, this.hass),
computeDeviceName(b, this.hass),
this.hass.locale.language
)
);
}
);
private async _findRelated() { private async _findRelated() {
const relatedPromises: Promise<RelatedResult>[] = []; const relatedPromises: Promise<RelatedResult>[] = [];
@@ -190,15 +158,6 @@ export class HaFilterDevices extends LitElement {
}); });
} }
private _clearFilter(ev) {
ev.preventDefault();
this.value = undefined;
fireEvent(this, "data-table-filter-changed", {
value: undefined,
items: undefined,
});
}
static get styles(): CSSResultGroup { static get styles(): CSSResultGroup {
return [ return [
haStyleScrollbar, haStyleScrollbar,
@@ -219,10 +178,6 @@ export class HaFilterDevices extends LitElement {
display: flex; display: flex;
align-items: center; align-items: center;
} }
.header ha-icon-button {
margin-inline-start: auto;
margin-inline-end: 8px;
}
.badge { .badge {
display: inline-block; display: inline-block;
margin-left: 8px; margin-left: 8px;
@@ -242,10 +197,6 @@ export class HaFilterDevices extends LitElement {
ha-check-list-item { ha-check-list-item {
width: 100%; width: 100%;
} }
search-input-outlined {
display: block;
padding: 0 8px;
}
`, `,
]; ];
} }

View File

@@ -1,4 +1,3 @@
import { mdiFilterVariantRemove } from "@mdi/js";
import { import {
css, css,
CSSResultGroup, CSSResultGroup,
@@ -15,11 +14,10 @@ import { computeStateName } from "../common/entity/compute_state_name";
import { stringCompare } from "../common/string/compare"; import { stringCompare } from "../common/string/compare";
import { findRelated, RelatedResult } from "../data/search"; import { findRelated, RelatedResult } from "../data/search";
import { haStyleScrollbar } from "../resources/styles"; import { haStyleScrollbar } from "../resources/styles";
import { loadVirtualizer } from "../resources/virtualizer";
import type { HomeAssistant } from "../types"; import type { HomeAssistant } from "../types";
import "./ha-check-list-item";
import "./ha-state-icon"; import "./ha-state-icon";
import "./search-input-outlined"; import "./ha-check-list-item";
import { loadVirtualizer } from "../resources/virtualizer";
@customElement("ha-filter-entities") @customElement("ha-filter-entities")
export class HaFilterEntities extends LitElement { export class HaFilterEntities extends LitElement {
@@ -35,8 +33,6 @@ export class HaFilterEntities extends LitElement {
@state() private _shouldRender = false; @state() private _shouldRender = false;
@state() private _filter?: string;
public willUpdate(properties: PropertyValues) { public willUpdate(properties: PropertyValues) {
super.willUpdate(properties); super.willUpdate(properties);
@@ -56,27 +52,16 @@ export class HaFilterEntities extends LitElement {
<div slot="header" class="header"> <div slot="header" class="header">
${this.hass.localize("ui.panel.config.entities.caption")} ${this.hass.localize("ui.panel.config.entities.caption")}
${this.value?.length ${this.value?.length
? html`<div class="badge">${this.value?.length}</div> ? html`<div class="badge">${this.value?.length}</div>`
<ha-icon-button
.path=${mdiFilterVariantRemove}
@click=${this._clearFilter}
></ha-icon-button>`
: nothing} : nothing}
</div> </div>
${this._shouldRender ${this._shouldRender
? html` ? html`
<search-input-outlined
.hass=${this.hass}
.filter=${this._filter}
@value-changed=${this._handleSearchChange}
>
</search-input-outlined>
<mwc-list class="ha-scrollbar"> <mwc-list class="ha-scrollbar">
<lit-virtualizer <lit-virtualizer
.items=${this._entities( .items=${this._entities(
this.hass.states, this.hass.states,
this.type, this.type,
this._filter || "",
this.value this.value
)} )}
.keyFunction=${this._keyFunction} .keyFunction=${this._keyFunction}
@@ -96,7 +81,7 @@ export class HaFilterEntities extends LitElement {
setTimeout(() => { setTimeout(() => {
if (!this.expanded) return; if (!this.expanded) return;
this.renderRoot.querySelector("mwc-list")!.style.height = this.renderRoot.querySelector("mwc-list")!.style.height =
`${this.clientHeight - 49 - 32}px`; // 32px is the height of the search input `${this.clientHeight - 49}px`;
}, 300); }, 300);
} }
} }
@@ -104,20 +89,18 @@ export class HaFilterEntities extends LitElement {
private _keyFunction = (entity) => entity?.entity_id; private _keyFunction = (entity) => entity?.entity_id;
private _renderItem = (entity) => private _renderItem = (entity) =>
!entity html`<ha-check-list-item
? nothing .value=${entity.entity_id}
: html`<ha-check-list-item .selected=${this.value?.includes(entity.entity_id)}
.value=${entity.entity_id} graphic="icon"
.selected=${this.value?.includes(entity.entity_id)} >
graphic="icon" <ha-state-icon
> slot="graphic"
<ha-state-icon .hass=${this.hass}
slot="graphic" .stateObj=${entity}
.hass=${this.hass} ></ha-state-icon>
.stateObj=${entity} ${computeStateName(entity)}
></ha-state-icon> </ha-check-list-item>`;
${computeStateName(entity)}
</ha-check-list-item>`;
private _handleItemClick(ev) { private _handleItemClick(ev) {
const listItem = ev.target.closest("ha-check-list-item"); const listItem = ev.target.closest("ha-check-list-item");
@@ -142,27 +125,12 @@ export class HaFilterEntities extends LitElement {
this.expanded = ev.detail.expanded; this.expanded = ev.detail.expanded;
} }
private _handleSearchChange(ev: CustomEvent) {
this._filter = ev.detail.value.toLowerCase();
}
private _entities = memoizeOne( private _entities = memoizeOne(
( (states: HomeAssistant["states"], type: this["type"], _value) => {
states: HomeAssistant["states"],
type: this["type"],
filter: string,
_value
) => {
const values = Object.values(states); const values = Object.values(states);
return values return values
.filter( .filter(
(entityState) => (entityState) => !type || computeStateDomain(entityState) !== type
(!type || computeStateDomain(entityState) !== type) &&
(!filter ||
entityState.entity_id.toLowerCase().includes(filter) ||
entityState.attributes.friendly_name
?.toLowerCase()
.includes(filter))
) )
.sort((a, b) => .sort((a, b) =>
stringCompare( stringCompare(
@@ -209,15 +177,6 @@ export class HaFilterEntities extends LitElement {
}); });
} }
private _clearFilter(ev) {
ev.preventDefault();
this.value = undefined;
fireEvent(this, "data-table-filter-changed", {
value: undefined,
items: undefined,
});
}
static get styles(): CSSResultGroup { static get styles(): CSSResultGroup {
return [ return [
haStyleScrollbar, haStyleScrollbar,
@@ -237,10 +196,6 @@ export class HaFilterEntities extends LitElement {
display: flex; display: flex;
align-items: center; align-items: center;
} }
.header ha-icon-button {
margin-inline-start: auto;
margin-inline-end: 8px;
}
.badge { .badge {
display: inline-block; display: inline-block;
margin-left: 8px; margin-left: 8px;
@@ -261,10 +216,6 @@ export class HaFilterEntities extends LitElement {
--mdc-list-item-graphic-margin: 16px; --mdc-list-item-graphic-margin: 16px;
width: 100%; width: 100%;
} }
search-input-outlined {
display: block;
padding: 0 8px;
}
`, `,
]; ];
} }

View File

@@ -1,19 +1,17 @@
import "@material/mwc-menu/mwc-menu-surface"; import "@material/mwc-menu/mwc-menu-surface";
import { mdiFilterVariantRemove, mdiTextureBox } from "@mdi/js"; import { mdiTextureBox } from "@mdi/js";
import { UnsubscribeFunc } from "home-assistant-js-websocket"; import { UnsubscribeFunc } from "home-assistant-js-websocket";
import { CSSResultGroup, LitElement, css, html, nothing } from "lit"; import { css, CSSResultGroup, html, LitElement, nothing } from "lit";
import { customElement, property, state } from "lit/decorators"; import { customElement, property, state } from "lit/decorators";
import { classMap } from "lit/directives/class-map";
import { repeat } from "lit/directives/repeat"; import { repeat } from "lit/directives/repeat";
import memoizeOne from "memoize-one"; import memoizeOne from "memoize-one";
import { fireEvent } from "../common/dom/fire_event"; import { fireEvent } from "../common/dom/fire_event";
import { computeRTL } from "../common/util/compute_rtl";
import { import {
FloorRegistryEntry, FloorRegistryEntry,
getFloorAreaLookup, getFloorAreaLookup,
subscribeFloorRegistry, subscribeFloorRegistry,
} from "../data/floor_registry"; } from "../data/floor_registry";
import { RelatedResult, findRelated } from "../data/search"; import { findRelated, RelatedResult } from "../data/search";
import { SubscribeMixin } from "../mixins/subscribe-mixin"; import { SubscribeMixin } from "../mixins/subscribe-mixin";
import { haStyleScrollbar } from "../resources/styles"; import { haStyleScrollbar } from "../resources/styles";
import type { HomeAssistant } from "../types"; import type { HomeAssistant } from "../types";
@@ -21,7 +19,6 @@ import "./ha-check-list-item";
import "./ha-floor-icon"; import "./ha-floor-icon";
import "./ha-icon"; import "./ha-icon";
import "./ha-svg-icon"; import "./ha-svg-icon";
import "./ha-tree-indicator";
@customElement("ha-filter-floor-areas") @customElement("ha-filter-floor-areas")
export class HaFilterFloorAreas extends SubscribeMixin(LitElement) { export class HaFilterFloorAreas extends SubscribeMixin(LitElement) {
@@ -56,13 +53,9 @@ export class HaFilterFloorAreas extends SubscribeMixin(LitElement) {
${this.hass.localize("ui.panel.config.areas.caption")} ${this.hass.localize("ui.panel.config.areas.caption")}
${this.value?.areas?.length || this.value?.floors?.length ${this.value?.areas?.length || this.value?.floors?.length
? html`<div class="badge"> ? html`<div class="badge">
${(this.value?.areas?.length || 0) + ${(this.value?.areas?.length || 0) +
(this.value?.floors?.length || 0)} (this.value?.floors?.length || 0)}
</div> </div>`
<ha-icon-button
.path=${mdiFilterVariantRemove}
@click=${this._clearFilter}
></ha-icon-button>`
: nothing} : nothing}
</div> </div>
${this._shouldRender ${this._shouldRender
@@ -89,10 +82,8 @@ export class HaFilterFloorAreas extends SubscribeMixin(LitElement) {
</ha-check-list-item> </ha-check-list-item>
${repeat( ${repeat(
floor.areas, floor.areas,
(area, index) => (area) => area.area_id,
`${area.area_id}${index === floor.areas.length - 1 ? "___last" : ""}`, (area) => this._renderArea(area)
(area, index) =>
this._renderArea(area, index === floor.areas.length - 1)
)} )}
` `
)} )}
@@ -108,37 +99,23 @@ export class HaFilterFloorAreas extends SubscribeMixin(LitElement) {
`; `;
} }
private _renderArea(area, last: boolean = false) { private _renderArea(area) {
const hasFloor = !!area.floor_id; return html`<ha-check-list-item
return html` .value=${area.area_id}
<ha-check-list-item .selected=${this.value?.areas?.includes(area.area_id) || false}
.value=${area.area_id} .type=${"areas"}
.selected=${this.value?.areas?.includes(area.area_id) || false} graphic="icon"
.type=${"areas"} class=${area.floor_id ? "floor" : ""}
graphic="icon" @request-selected=${this._handleItemClick}
@request-selected=${this._handleItemClick} >
class=${classMap({ ${area.icon
rtl: computeRTL(this.hass), ? html`<ha-icon slot="graphic" .icon=${area.icon}></ha-icon>`
floor: hasFloor, : html`<ha-svg-icon
})} slot="graphic"
> .path=${mdiTextureBox}
${hasFloor ></ha-svg-icon>`}
? html` ${area.name}
<ha-tree-indicator </ha-check-list-item>`;
.end=${last}
slot="graphic"
></ha-tree-indicator>
`
: nothing}
${area.icon
? html`<ha-icon slot="graphic" .icon=${area.icon}></ha-icon>`
: html`<ha-svg-icon
slot="graphic"
.path=${mdiTextureBox}
></ha-svg-icon>`}
${area.name}
</ha-check-list-item>
`;
} }
private _handleItemClick(ev) { private _handleItemClick(ev) {
@@ -261,15 +238,6 @@ export class HaFilterFloorAreas extends SubscribeMixin(LitElement) {
}); });
} }
private _clearFilter(ev) {
ev.preventDefault();
this.value = undefined;
fireEvent(this, "data-table-filter-changed", {
value: undefined,
items: undefined,
});
}
static get styles(): CSSResultGroup { static get styles(): CSSResultGroup {
return [ return [
haStyleScrollbar, haStyleScrollbar,
@@ -289,10 +257,6 @@ export class HaFilterFloorAreas extends SubscribeMixin(LitElement) {
display: flex; display: flex;
align-items: center; align-items: center;
} }
.header ha-icon-button {
margin-inline-start: auto;
margin-inline-end: 8px;
}
.badge { .badge {
display: inline-block; display: inline-block;
margin-left: 8px; margin-left: 8px;
@@ -313,26 +277,9 @@ export class HaFilterFloorAreas extends SubscribeMixin(LitElement) {
--mdc-list-item-graphic-margin: 16px; --mdc-list-item-graphic-margin: 16px;
} }
.floor { .floor {
padding-left: 48px; padding-left: 32px;
padding-inline-start: 48px; padding-inline-start: 32px;
padding-inline-end: 16px;
} }
ha-tree-indicator {
width: 56px;
position: absolute;
top: 0px;
left: 0px;
}
.rtl ha-tree-indicator {
right: 0px;
left: initial;
transform: scaleX(-1);
}
.subdir {
margin-inline-end: 8px;
opacity: .6;
}
.
`, `,
]; ];
} }

View File

@@ -1,5 +1,4 @@
import { SelectedDetail } from "@material/mwc-list"; import { SelectedDetail } from "@material/mwc-list";
import { mdiFilterVariantRemove } from "@mdi/js";
import { css, CSSResultGroup, html, LitElement, nothing } from "lit"; import { css, CSSResultGroup, html, LitElement, nothing } from "lit";
import { customElement, property, state } from "lit/decorators"; import { customElement, property, state } from "lit/decorators";
import { repeat } from "lit/directives/repeat"; import { repeat } from "lit/directives/repeat";
@@ -13,7 +12,6 @@ import {
import { haStyleScrollbar } from "../resources/styles"; import { haStyleScrollbar } from "../resources/styles";
import type { HomeAssistant } from "../types"; import type { HomeAssistant } from "../types";
import "./ha-domain-icon"; import "./ha-domain-icon";
import "./search-input-outlined";
@customElement("ha-filter-integrations") @customElement("ha-filter-integrations")
export class HaFilterIntegrations extends LitElement { export class HaFilterIntegrations extends LitElement {
@@ -29,8 +27,6 @@ export class HaFilterIntegrations extends LitElement {
@state() private _shouldRender = false; @state() private _shouldRender = false;
@state() private _filter?: string;
protected render() { protected render() {
return html` return html`
<ha-expansion-panel <ha-expansion-panel
@@ -42,27 +38,18 @@ export class HaFilterIntegrations extends LitElement {
<div slot="header" class="header"> <div slot="header" class="header">
${this.hass.localize("ui.panel.config.integrations.caption")} ${this.hass.localize("ui.panel.config.integrations.caption")}
${this.value?.length ${this.value?.length
? html`<div class="badge">${this.value?.length}</div> ? html`<div class="badge">${this.value?.length}</div>`
<ha-icon-button
.path=${mdiFilterVariantRemove}
@click=${this._clearFilter}
></ha-icon-button>`
: nothing} : nothing}
</div> </div>
${this._manifests && this._shouldRender ${this._manifests && this._shouldRender
? html`<search-input-outlined ? html`
.hass=${this.hass}
.filter=${this._filter}
@value-changed=${this._handleSearchChange}
>
</search-input-outlined>
<mwc-list <mwc-list
@selected=${this._integrationsSelected} @selected=${this._integrationsSelected}
multi multi
class="ha-scrollbar" class="ha-scrollbar"
> >
${repeat( ${repeat(
this._integrations(this._manifests, this._filter, this.value), this._integrations(this._manifests, this.value),
(i) => i.domain, (i) => i.domain,
(integration) => (integration) =>
html`<ha-check-list-item html`<ha-check-list-item
@@ -81,7 +68,8 @@ export class HaFilterIntegrations extends LitElement {
${integration.name || integration.domain} ${integration.name || integration.domain}
</ha-check-list-item>` </ha-check-list-item>`
)} )}
</mwc-list> ` </mwc-list>
`
: nothing} : nothing}
</ha-expansion-panel> </ha-expansion-panel>
`; `;
@@ -110,17 +98,12 @@ export class HaFilterIntegrations extends LitElement {
} }
private _integrations = memoizeOne( private _integrations = memoizeOne(
(manifest: IntegrationManifest[], filter: string | undefined, _value) => (manifest: IntegrationManifest[], _value) =>
manifest manifest
.filter( .filter(
(mnfst) => (mnfst) =>
(!mnfst.integration_type || !mnfst.integration_type ||
!["entity", "system", "hardware"].includes( !["entity", "system", "hardware"].includes(mnfst.integration_type)
mnfst.integration_type
)) &&
(!filter ||
mnfst.name.toLowerCase().includes(filter) ||
mnfst.domain.toLowerCase().includes(filter))
) )
.sort((a, b) => .sort((a, b) =>
stringCompare( stringCompare(
@@ -134,11 +117,7 @@ export class HaFilterIntegrations extends LitElement {
private async _integrationsSelected( private async _integrationsSelected(
ev: CustomEvent<SelectedDetail<Set<number>>> ev: CustomEvent<SelectedDetail<Set<number>>>
) { ) {
const integrations = this._integrations( const integrations = this._integrations(this._manifests!, this.value);
this._manifests!,
this._filter,
this.value
);
if (!ev.detail.index.size) { if (!ev.detail.index.size) {
fireEvent(this, "data-table-filter-changed", { fireEvent(this, "data-table-filter-changed", {
@@ -163,19 +142,6 @@ export class HaFilterIntegrations extends LitElement {
}); });
} }
private _clearFilter(ev) {
ev.preventDefault();
this.value = undefined;
fireEvent(this, "data-table-filter-changed", {
value: undefined,
items: undefined,
});
}
private _handleSearchChange(ev: CustomEvent) {
this._filter = ev.detail.value.toLowerCase();
}
static get styles(): CSSResultGroup { static get styles(): CSSResultGroup {
return [ return [
haStyleScrollbar, haStyleScrollbar,
@@ -195,10 +161,6 @@ export class HaFilterIntegrations extends LitElement {
display: flex; display: flex;
align-items: center; align-items: center;
} }
.header ha-icon-button {
margin-inline-start: auto;
margin-inline-end: 8px;
}
.badge { .badge {
display: inline-block; display: inline-block;
margin-left: 8px; margin-left: 8px;
@@ -215,10 +177,6 @@ export class HaFilterIntegrations extends LitElement {
padding: 0px 2px; padding: 0px 2px;
color: var(--text-primary-color); color: var(--text-primary-color);
} }
search-input-outlined {
display: block;
padding: 0 8px;
}
`, `,
]; ];
} }

View File

@@ -1,18 +1,19 @@
import { SelectedDetail } from "@material/mwc-list"; import { SelectedDetail } from "@material/mwc-list";
import "@material/mwc-menu/mwc-menu-surface"; import "@material/mwc-menu/mwc-menu-surface";
import { mdiCog, mdiFilterVariantRemove } from "@mdi/js"; import { mdiPlus } from "@mdi/js";
import { UnsubscribeFunc } from "home-assistant-js-websocket"; import { UnsubscribeFunc } from "home-assistant-js-websocket";
import { CSSResultGroup, LitElement, css, html, nothing } from "lit"; import { CSSResultGroup, LitElement, css, html, nothing } from "lit";
import { customElement, property, state } from "lit/decorators"; import { customElement, property, state } from "lit/decorators";
import { repeat } from "lit/directives/repeat"; import { repeat } from "lit/directives/repeat";
import { computeCssColor } from "../common/color/compute-color"; import { computeCssColor } from "../common/color/compute-color";
import { fireEvent } from "../common/dom/fire_event"; import { fireEvent } from "../common/dom/fire_event";
import { navigate } from "../common/navigate";
import { import {
LabelRegistryEntry, LabelRegistryEntry,
createLabelRegistryEntry,
subscribeLabelRegistry, subscribeLabelRegistry,
} from "../data/label_registry"; } from "../data/label_registry";
import { SubscribeMixin } from "../mixins/subscribe-mixin"; import { SubscribeMixin } from "../mixins/subscribe-mixin";
import { showLabelDetailDialog } from "../panels/config/labels/show-dialog-label-detail";
import { haStyleScrollbar } from "../resources/styles"; import { haStyleScrollbar } from "../resources/styles";
import type { HomeAssistant } from "../types"; import type { HomeAssistant } from "../types";
import "./ha-check-list-item"; import "./ha-check-list-item";
@@ -53,11 +54,7 @@ export class HaFilterLabels extends SubscribeMixin(LitElement) {
<div slot="header" class="header"> <div slot="header" class="header">
${this.hass.localize("ui.panel.config.labels.caption")} ${this.hass.localize("ui.panel.config.labels.caption")}
${this.value?.length ${this.value?.length
? html`<div class="badge">${this.value?.length}</div> ? html`<div class="badge">${this.value?.length}</div>`
<ha-icon-button
.path=${mdiFilterVariantRemove}
@click=${this._clearFilter}
></ha-icon-button>`
: nothing} : nothing}
</div> </div>
${this._shouldRender ${this._shouldRender
@@ -98,11 +95,11 @@ export class HaFilterLabels extends SubscribeMixin(LitElement) {
${this.expanded ${this.expanded
? html`<ha-list-item ? html`<ha-list-item
graphic="icon" graphic="icon"
@click=${this._manageLabels} @click=${this._addLabel}
class="add" class="add"
> >
<ha-svg-icon slot="graphic" .path=${mdiCog}></ha-svg-icon> <ha-svg-icon slot="graphic" .path=${mdiPlus}></ha-svg-icon>
${this.hass.localize("ui.panel.config.labels.manage_labels")} ${this.hass.localize("ui.panel.config.labels.add_label")}
</ha-list-item>` </ha-list-item>`
: nothing} : nothing}
`; `;
@@ -118,8 +115,10 @@ export class HaFilterLabels extends SubscribeMixin(LitElement) {
} }
} }
private _manageLabels() { private _addLabel() {
navigate("/config/labels"); showLabelDetailDialog(this, {
createEntry: (values) => createLabelRegistryEntry(this.hass, values),
});
} }
private _expandedWillChange(ev) { private _expandedWillChange(ev) {
@@ -154,15 +153,6 @@ export class HaFilterLabels extends SubscribeMixin(LitElement) {
}); });
} }
private _clearFilter(ev) {
ev.preventDefault();
this.value = undefined;
fireEvent(this, "data-table-filter-changed", {
value: undefined,
items: undefined,
});
}
static get styles(): CSSResultGroup { static get styles(): CSSResultGroup {
return [ return [
haStyleScrollbar, haStyleScrollbar,
@@ -183,10 +173,6 @@ export class HaFilterLabels extends SubscribeMixin(LitElement) {
display: flex; display: flex;
align-items: center; align-items: center;
} }
.header ha-icon-button {
margin-inline-start: auto;
margin-inline-end: 8px;
}
.badge { .badge {
display: inline-block; display: inline-block;
margin-left: 8px; margin-left: 8px;

View File

@@ -1,12 +1,11 @@
import { SelectedDetail } from "@material/mwc-list"; import { SelectedDetail } from "@material/mwc-list";
import { mdiFilterVariantRemove } from "@mdi/js";
import { css, CSSResultGroup, html, LitElement, nothing } from "lit"; import { css, CSSResultGroup, html, LitElement, nothing } from "lit";
import { customElement, property, state } from "lit/decorators"; import { customElement, property, state } from "lit/decorators";
import { fireEvent } from "../common/dom/fire_event"; import { fireEvent } from "../common/dom/fire_event";
import { haStyleScrollbar } from "../resources/styles"; import { haStyleScrollbar } from "../resources/styles";
import type { HomeAssistant } from "../types"; import type { HomeAssistant } from "../types";
import "./ha-check-list-item";
import "./ha-expansion-panel"; import "./ha-expansion-panel";
import "./ha-check-list-item";
import "./ha-icon"; import "./ha-icon";
@customElement("ha-filter-states") @customElement("ha-filter-states")
@@ -44,11 +43,7 @@ export class HaFilterStates extends LitElement {
<div slot="header" class="header"> <div slot="header" class="header">
${this.label} ${this.label}
${this.value?.length ${this.value?.length
? html`<div class="badge">${this.value?.length}</div> ? html`<div class="badge">${this.value?.length}</div>`
<ha-icon-button
.path=${mdiFilterVariantRemove}
@click=${this._clearFilter}
></ha-icon-button>`
: nothing} : nothing}
</div> </div>
${this._shouldRender ${this._shouldRender
@@ -123,15 +118,6 @@ export class HaFilterStates extends LitElement {
}); });
} }
private _clearFilter(ev) {
ev.preventDefault();
this.value = undefined;
fireEvent(this, "data-table-filter-changed", {
value: undefined,
items: undefined,
});
}
static get styles(): CSSResultGroup { static get styles(): CSSResultGroup {
return [ return [
haStyleScrollbar, haStyleScrollbar,
@@ -151,10 +137,6 @@ export class HaFilterStates extends LitElement {
display: flex; display: flex;
align-items: center; align-items: center;
} }
.header ha-icon-button {
margin-inline-start: auto;
margin-inline-end: 8px;
}
.badge { .badge {
display: inline-block; display: inline-block;
margin-left: 8px; margin-left: 8px;

View File

@@ -10,10 +10,7 @@ import {
ScorableTextItem, ScorableTextItem,
fuzzyFilterSort, fuzzyFilterSort,
} from "../common/string/filter/sequence-matching"; } from "../common/string/filter/sequence-matching";
import { import { AreaRegistryEntry } from "../data/area_registry";
AreaRegistryEntry,
updateAreaRegistryEntry,
} from "../data/area_registry";
import { import {
DeviceEntityDisplayLookup, DeviceEntityDisplayLookup,
DeviceRegistryEntry, DeviceRegistryEntry,
@@ -440,18 +437,11 @@ export class HaFloorPicker extends SubscribeMixin(LitElement) {
(ev.target as any).value = this._value; (ev.target as any).value = this._value;
this.hass.loadFragmentTranslation("config");
showFloorRegistryDetailDialog(this, { showFloorRegistryDetailDialog(this, {
suggestedName: newValue === ADD_NEW_SUGGESTION_ID ? this._suggestion : "", suggestedName: newValue === ADD_NEW_SUGGESTION_ID ? this._suggestion : "",
createEntry: async (values, addedAreas) => { createEntry: async (values) => {
try { try {
const floor = await createFloorRegistryEntry(this.hass, values); const floor = await createFloorRegistryEntry(this.hass, values);
addedAreas.forEach((areaId) => {
updateAreaRegistryEntry(this.hass, areaId, {
floor_id: floor.floor_id,
});
});
const floors = [...this._floors!, floor]; const floors = [...this._floors!, floor];
this.comboBox.filteredItems = this._getFloors( this.comboBox.filteredItems = this._getFloors(
floors, floors,

View File

@@ -445,8 +445,6 @@ export class HaLabelPicker extends SubscribeMixin(LitElement) {
(ev.target as any).value = this._value; (ev.target as any).value = this._value;
this.hass.loadFragmentTranslation("config");
showLabelDetailDialog(this, { showLabelDetailDialog(this, {
entry: undefined, entry: undefined,
suggestedName: newValue === ADD_NEW_SUGGESTION_ID ? this._suggestion : "", suggestedName: newValue === ADD_NEW_SUGGESTION_ID ? this._suggestion : "",

View File

@@ -2,10 +2,8 @@ import { HassEntity, UnsubscribeFunc } from "home-assistant-js-websocket";
import { LitElement, TemplateResult, css, html, nothing } from "lit"; import { LitElement, TemplateResult, css, html, nothing } from "lit";
import { customElement, property, query, state } from "lit/decorators"; import { customElement, property, query, state } from "lit/decorators";
import { repeat } from "lit/directives/repeat"; import { repeat } from "lit/directives/repeat";
import memoizeOne from "memoize-one";
import { computeCssColor } from "../common/color/compute-color"; import { computeCssColor } from "../common/color/compute-color";
import { fireEvent } from "../common/dom/fire_event"; import { fireEvent } from "../common/dom/fire_event";
import { stringCompare } from "../common/string/compare";
import { import {
LabelRegistryEntry, LabelRegistryEntry,
subscribeLabelRegistry, subscribeLabelRegistry,
@@ -19,6 +17,7 @@ import "./chips/ha-input-chip";
import type { HaDevicePickerDeviceFilterFunc } from "./device/ha-device-picker"; import type { HaDevicePickerDeviceFilterFunc } from "./device/ha-device-picker";
import "./ha-label-picker"; import "./ha-label-picker";
import type { HaLabelPicker } from "./ha-label-picker"; import type { HaLabelPicker } from "./ha-label-picker";
import { stringCompare } from "../common/string/compare";
@customElement("ha-labels-picker") @customElement("ha-labels-picker")
export class HaLabelsPicker extends SubscribeMixin(LitElement) { export class HaLabelsPicker extends SubscribeMixin(LitElement) {
@@ -103,35 +102,25 @@ export class HaLabelsPicker extends SubscribeMixin(LitElement) {
]; ];
} }
private _sortedLabels = memoizeOne(
(
value: string[] | undefined,
labels: { [id: string]: LabelRegistryEntry } | undefined,
language: string
) =>
value
?.map((id) => labels?.[id])
.sort((a, b) => stringCompare(a?.name || "", b?.name || "", language))
);
protected render(): TemplateResult { protected render(): TemplateResult {
const labels = this._sortedLabels( const labels = this.value
this.value, ?.map((id) => this._labels?.[id])
this._labels, .sort((a, b) =>
this.hass.locale.language stringCompare(a?.name || "", b?.name || "", this.hass.locale.language)
); );
return html` return html`
${labels?.length ${labels?.length
? html`<ha-chip-set> ? html`<ha-chip-set>
${repeat( ${repeat(
labels, labels,
(label) => label?.label_id, (label) => label?.label_id,
(label) => { (label, idx) => {
const color = label?.color const color = label?.color
? computeCssColor(label.color) ? computeCssColor(label.color)
: undefined; : undefined;
return html` return html`
<ha-input-chip <ha-input-chip
.idx=${idx}
.item=${label} .item=${label}
@remove=${this._removeItem} @remove=${this._removeItem}
@click=${this._openDetail} @click=${this._openDetail}
@@ -172,12 +161,12 @@ export class HaLabelsPicker extends SubscribeMixin(LitElement) {
} }
private _removeItem(ev) { private _removeItem(ev) {
const label = ev.currentTarget.item; this._value.splice(ev.target.idx, 1);
this._setValue(this._value.filter((id) => id !== label.label_id)); this._setValue([...this._value]);
} }
private _openDetail(ev) { private _openDetail(ev) {
const label = ev.currentTarget.item; const label = ev.target.item;
showLabelDetailDialog(this, { showLabelDetailDialog(this, {
entry: label, entry: label,
updateEntry: async (values) => { updateEntry: async (values) => {

View File

@@ -1,7 +1,7 @@
import { MdMenuItem } from "@material/web/menu/menu-item"; import { customElement } from "lit/decorators";
import "element-internals-polyfill"; import "element-internals-polyfill";
import { CSSResult, css } from "lit"; import { CSSResult, css } from "lit";
import { customElement } from "lit/decorators"; import { MdMenuItem } from "@material/web/menu/menu-item";
@customElement("ha-menu-item") @customElement("ha-menu-item")
export class HaMenuItem extends MdMenuItem { export class HaMenuItem extends MdMenuItem {

View File

@@ -27,10 +27,6 @@ export class HaOutlinedTextField extends MdOutlinedTextField {
--md-outlined-field-focus-outline-width: 1px; --md-outlined-field-focus-outline-width: 1px;
--mdc-icon-size: var(--md-input-chip-icon-size, 18px); --mdc-icon-size: var(--md-input-chip-icon-size, 18px);
} }
md-outlined-field {
background: var(--ha-outlined-text-field-container-color, transparent);
opacity: var(--ha-outlined-text-field-container-opacity, 1);
}
.input { .input {
font-family: Roboto, sans-serif; font-family: Roboto, sans-serif;
} }

View File

@@ -30,7 +30,6 @@ export class HaLabelSelector extends LitElement {
if (this.selector.label.multiple) { if (this.selector.label.multiple) {
return html` return html`
<ha-labels-picker <ha-labels-picker
no-add
.hass=${this.hass} .hass=${this.hass}
.value=${ensureArray(this.value ?? [])} .value=${ensureArray(this.value ?? [])}
.disabled=${this.disabled} .disabled=${this.disabled}
@@ -42,7 +41,6 @@ export class HaLabelSelector extends LitElement {
} }
return html` return html`
<ha-label-picker <ha-label-picker
no-add
.hass=${this.hass} .hass=${this.hass}
.value=${this.value} .value=${this.value}
.disabled=${this.disabled} .disabled=${this.disabled}

View File

@@ -1,36 +0,0 @@
import { LitElement, TemplateResult, css, html } from "lit";
import { customElement, property } from "lit/decorators";
@customElement("ha-tree-indicator")
export class HaTreeIndicator extends LitElement {
@property({ type: Boolean, reflect: true })
public end?: boolean = false;
protected render(): TemplateResult {
return html`
<svg width="100%" height="100%" viewBox="0 0 48 48">
<line x1="24" y1="0" x2="24" y2=${this.end ? "24" : "48"}></line>
<line x1="24" y1="24" x2="36" y2="24"></line>
</svg>
`;
}
static styles = css`
:host {
display: block;
width: 48px;
height: 48px;
}
line {
stroke: var(--divider-color);
stroke-width: 2;
stroke-dasharray: 2;
}
`;
}
declare global {
interface HTMLElementTagNameMap {
"ha-tree-indicator": HaTreeIndicator;
}
}

View File

@@ -1,12 +1,5 @@
import { mdiClose, mdiMagnify } from "@mdi/js"; import { mdiMagnify } from "@mdi/js";
import { import { CSSResultGroup, LitElement, TemplateResult, css, html } from "lit";
CSSResultGroup,
LitElement,
TemplateResult,
css,
html,
nothing,
} from "lit";
import { customElement, property, query } from "lit/decorators"; import { customElement, property, query } from "lit/decorators";
import { fireEvent } from "../common/dom/fire_event"; import { fireEvent } from "../common/dom/fire_event";
import { HomeAssistant } from "../types"; import { HomeAssistant } from "../types";
@@ -61,15 +54,6 @@ class SearchInputOutlined extends LitElement {
.path=${mdiMagnify} .path=${mdiMagnify}
></ha-svg-icon> ></ha-svg-icon>
</slot> </slot>
${this.filter
? html`<ha-icon-button
aria-label="Clear input"
slot="trailing-icon"
@click=${this._clearSearch}
.path=${mdiClose}
>
</ha-icon-button>`
: nothing}
</ha-outlined-text-field> </ha-outlined-text-field>
`; `;
} }
@@ -82,22 +66,16 @@ class SearchInputOutlined extends LitElement {
this._filterChanged(e.target.value); this._filterChanged(e.target.value);
} }
private async _clearSearch() {
this._filterChanged("");
}
static get styles(): CSSResultGroup { static get styles(): CSSResultGroup {
return css` return css`
:host { :host {
display: inline-flex; display: inline-flex;
/* For iOS */ /* For iOS */
z-index: 0; z-index: 0;
--mdc-icon-button-size: 24px;
} }
ha-outlined-text-field { ha-outlined-text-field {
display: block; display: block;
width: 100%; width: 100%;
--ha-outlined-text-field-container-color: var(--card-background-color);
} }
ha-svg-icon, ha-svg-icon,
ha-icon-button { ha-icon-button {

View File

@@ -28,7 +28,6 @@ export type ItemType =
| "entity" | "entity"
| "floor" | "floor"
| "group" | "group"
| "label"
| "scene" | "scene"
| "script" | "script"
| "automation_blueprint" | "automation_blueprint"

View File

@@ -190,7 +190,7 @@ class LightColorTempPicker extends LitElement {
max-height: 320px; max-height: 320px;
min-height: 200px; min-height: 200px;
--control-slider-thickness: 130px; --control-slider-thickness: 130px;
--control-slider-border-radius: 36px; --control-slider-border-radius: 48px;
--control-slider-color: var(--primary-color); --control-slider-color: var(--primary-color);
--control-slider-background: -webkit-linear-gradient( --control-slider-background: -webkit-linear-gradient(
top, top,

View File

@@ -1,8 +1,9 @@
import { mdiShieldOff } from "@mdi/js";
import { css, CSSResultGroup, html, LitElement, nothing } from "lit"; import { css, CSSResultGroup, html, LitElement, nothing } from "lit";
import { customElement, property } from "lit/decorators"; import { customElement, property } from "lit/decorators";
import { styleMap } from "lit/directives/style-map"; import { styleMap } from "lit/directives/style-map";
import { stateColorCss } from "../../../common/entity/state_color"; import { stateColorCss } from "../../../common/entity/state_color";
import "../../../components/ha-control-button"; import "../../../components/ha-outlined-button";
import "../../../components/ha-state-icon"; import "../../../components/ha-state-icon";
import { AlarmControlPanelEntity } from "../../../data/alarm_control_panel"; import { AlarmControlPanelEntity } from "../../../data/alarm_control_panel";
import "../../../state-control/alarm_control_panel/ha-state-control-alarm_control_panel-modes"; import "../../../state-control/alarm_control_panel/ha-state-control-alarm_control_panel-modes";
@@ -56,10 +57,15 @@ class MoreInfoAlarmControlPanel extends LitElement {
${["triggered", "arming", "pending"].includes(this.stateObj.state) ${["triggered", "arming", "pending"].includes(this.stateObj.state)
? html` ? html`
<div class="status"> <div class="status">
<span></span>
<div class="icon"> <div class="icon">
<ha-state-icon .hass=${this.hass} .stateObj=${this.stateObj}> <ha-state-icon .hass=${this.hass} .stateObj=${this.stateObj}>
</ha-state-icon> </ha-state-icon>
</div> </div>
<ha-outlined-button @click=${this._disarm}>
${this.hass.localize("ui.card.alarm_control_panel.disarm")}
<ha-svg-icon slot="icon" .path=${mdiShieldOff}></ha-svg-icon>
</ha-outlined-button>
</div> </div>
` `
: html` : html`
@@ -70,15 +76,7 @@ class MoreInfoAlarmControlPanel extends LitElement {
</ha-state-control-alarm_control_panel-modes> </ha-state-control-alarm_control_panel-modes>
`} `}
</div> </div>
<div> <span></span>
${["triggered", "arming", "pending"].includes(this.stateObj.state)
? html`
<ha-control-button @click=${this._disarm} class="disarm">
${this.hass.localize("ui.card.alarm_control_panel.disarm")}
</ha-control-button>
`
: nothing}
</div>
`; `;
} }
@@ -129,12 +127,8 @@ class MoreInfoAlarmControlPanel extends LitElement {
transition: background-color 180ms ease-in-out; transition: background-color 180ms ease-in-out;
opacity: 0.2; opacity: 0.2;
} }
ha-control-button.disarm { .status ha-outlined-button {
height: 60px; margin-top: 32px;
min-width: 130px;
max-width: 200px;
margin: 0 auto;
--control-button-border-radius: 24px;
} }
`, `,
]; ];

View File

@@ -170,7 +170,7 @@ class MoreInfoLock extends LitElement {
--control-button-border-radius: 24px; --control-button-border-radius: 24px;
} }
.open-button { .open-button {
width: 130px; width: 100px;
--control-button-background-color: var(--state-color); --control-button-background-color: var(--state-color);
} }
.open-button.confirm { .open-button.confirm {

View File

@@ -321,28 +321,19 @@ export class HaTabsSubpageDataTable extends LitElement {
.path=${mdiMenuDown} .path=${mdiMenuDown}
></ha-svg-icon ></ha-svg-icon
></ha-assist-chip> ></ha-assist-chip>
<ha-menu-item .value=${undefined} @click=${this._selectAll}> <ha-menu-item .value=${undefined} @click=${this._selectAll}
<div slot="headline"> >${localize("ui.components.subpage-data-table.select_all")}
${localize("ui.components.subpage-data-table.select_all")}
</div>
</ha-menu-item> </ha-menu-item>
<ha-menu-item .value=${undefined} @click=${this._selectNone}> <ha-menu-item .value=${undefined} @click=${this._selectNone}
<div slot="headline"> >${localize("ui.components.subpage-data-table.select_none")}
${localize(
"ui.components.subpage-data-table.select_none"
)}
</div>
</ha-menu-item> </ha-menu-item>
<md-divider role="separator" tabindex="-1"></md-divider> <md-divider role="separator" tabindex="-1"></md-divider>
<ha-menu-item <ha-menu-item
.value=${undefined} .value=${undefined}
@click=${this._disableSelectMode} @click=${this._disableSelectMode}
> >${localize(
<div slot="headline"> "ui.components.subpage-data-table.close_select_mode"
${localize( )}
"ui.components.subpage-data-table.close_select_mode"
)}
</div>
</ha-menu-item> </ha-menu-item>
</ha-button-menu-new> </ha-button-menu-new>
<p> <p>
@@ -358,7 +349,37 @@ export class HaTabsSubpageDataTable extends LitElement {
: nothing} : nothing}
${this.showFilters ${this.showFilters
? !showPane ? !showPane
? nothing ? html`<ha-dialog
open
hideActions
.heading=${localize("ui.components.subpage-data-table.filters")}
>
<ha-dialog-header slot="heading">
<ha-icon-button
slot="navigationIcon"
.path=${mdiClose}
@click=${this._toggleFilters}
.label=${localize(
"ui.components.subpage-data-table.close_filter"
)}
></ha-icon-button>
<span slot="title"
>${localize(
"ui.components.subpage-data-table.filters"
)}</span
>
<ha-icon-button
slot="actionItems"
@click=${this._clearFilters}
.path=${mdiFilterVariantRemove}
.label=${localize(
"ui.components.subpage-data-table.clear_filter"
)}
></ha-icon-button>
</ha-dialog-header>
<div class="filter-dialog-content">
<slot name="filter-pane"></slot></div
></ha-dialog>`
: html`<div class="pane" slot="pane"> : html`<div class="pane" slot="pane">
<div class="table-header"> <div class="table-header">
<ha-assist-chip <ha-assist-chip
@@ -373,15 +394,13 @@ export class HaTabsSubpageDataTable extends LitElement {
.path=${mdiFilterVariant} .path=${mdiFilterVariant}
></ha-svg-icon> ></ha-svg-icon>
</ha-assist-chip> </ha-assist-chip>
${this.filters <ha-icon-button
? html`<ha-icon-button .path=${mdiFilterVariantRemove}
.path=${mdiFilterVariantRemove} @click=${this._clearFilters}
@click=${this._clearFilters} .label=${localize(
.label=${localize( "ui.components.subpage-data-table.clear_filter"
"ui.components.subpage-data-table.clear_filter" )}
)} ></ha-icon-button>
></ha-icon-button>`
: nothing}
</div> </div>
<div class="pane-content"> <div class="pane-content">
<slot name="filter-pane"></slot> <slot name="filter-pane"></slot>
@@ -493,39 +512,6 @@ export class HaTabsSubpageDataTable extends LitElement {
: nothing : nothing
)} )}
</ha-menu> </ha-menu>
${this.showFilters && !showPane
? html`<ha-dialog
open
hideActions
.heading=${localize("ui.components.subpage-data-table.filters")}
>
<ha-dialog-header slot="heading">
<ha-icon-button
slot="navigationIcon"
.path=${mdiClose}
@click=${this._toggleFilters}
.label=${localize(
"ui.components.subpage-data-table.close_filter"
)}
></ha-icon-button>
<span slot="title"
>${localize("ui.components.subpage-data-table.filters")}</span
>
${this.filters
? html`<ha-icon-button
slot="actionItems"
@click=${this._clearFilters}
.path=${mdiFilterVariantRemove}
.label=${localize(
"ui.components.subpage-data-table.clear_filter"
)}
></ha-icon-button>`
: nothing}
</ha-dialog-header>
<div class="filter-dialog-content">
<slot name="filter-pane"></slot></div
></ha-dialog>`
: nothing}
`; `;
} }
@@ -587,7 +573,6 @@ export class HaTabsSubpageDataTable extends LitElement {
return css` return css`
:host { :host {
display: block; display: block;
height: 100%;
} }
ha-data-table { ha-data-table {
@@ -743,7 +728,7 @@ export class HaTabsSubpageDataTable extends LitElement {
padding: 8px 12px; padding: 8px 12px;
box-sizing: border-box; box-sizing: border-box;
font-size: 14px; font-size: 14px;
--ha-assist-chip-container-color: var(--card-background-color); --ha-assist-chip-container-color: var(--primary-background-color);
} }
.selection-controls { .selection-controls {
@@ -770,7 +755,6 @@ export class HaTabsSubpageDataTable extends LitElement {
ha-assist-chip { ha-assist-chip {
--ha-assist-chip-container-shape: 10px; --ha-assist-chip-container-shape: 10px;
--ha-assist-chip-container-color: var(--card-background-color);
} }
.select-mode-chip { .select-mode-chip {
@@ -779,7 +763,6 @@ export class HaTabsSubpageDataTable extends LitElement {
} }
ha-dialog { ha-dialog {
--dialog-z-index: 100;
--mdc-dialog-min-width: calc( --mdc-dialog-min-width: calc(
100vw - env(safe-area-inset-right) - env(safe-area-inset-left) 100vw - env(safe-area-inset-right) - env(safe-area-inset-left)
); );

View File

@@ -1,13 +1,8 @@
import "@material/mwc-button"; import "@material/mwc-button";
import "@material/mwc-list/mwc-list"; import "@material/mwc-list/mwc-list";
import { mdiTextureBox } from "@mdi/js";
import { css, CSSResultGroup, html, LitElement, nothing } from "lit"; import { css, CSSResultGroup, html, LitElement, nothing } from "lit";
import { property, state } from "lit/decorators"; import { property, state } from "lit/decorators";
import { repeat } from "lit/directives/repeat";
import memoizeOne from "memoize-one";
import { fireEvent } from "../../../common/dom/fire_event"; import { fireEvent } from "../../../common/dom/fire_event";
import "../../../components/chips/ha-chip-set";
import "../../../components/chips/ha-input-chip";
import "../../../components/ha-alert"; import "../../../components/ha-alert";
import "../../../components/ha-aliases-editor"; import "../../../components/ha-aliases-editor";
import { createCloseHeading } from "../../../components/ha-dialog"; import { createCloseHeading } from "../../../components/ha-dialog";
@@ -16,15 +11,10 @@ import "../../../components/ha-picture-upload";
import "../../../components/ha-settings-row"; import "../../../components/ha-settings-row";
import "../../../components/ha-svg-icon"; import "../../../components/ha-svg-icon";
import "../../../components/ha-textfield"; import "../../../components/ha-textfield";
import { import { FloorRegistryEntryMutableParams } from "../../../data/floor_registry";
FloorRegistryEntry, import { haStyleDialog } from "../../../resources/styles";
FloorRegistryEntryMutableParams,
} from "../../../data/floor_registry";
import { haStyle, haStyleDialog } from "../../../resources/styles";
import { HomeAssistant } from "../../../types"; import { HomeAssistant } from "../../../types";
import { FloorRegistryDetailDialogParams } from "./show-dialog-floor-registry-detail"; import { FloorRegistryDetailDialogParams } from "./show-dialog-floor-registry-detail";
import { showAreaRegistryDetailDialog } from "./show-dialog-area-registry-detail";
import { updateAreaRegistryEntry } from "../../../data/area_registry";
class DialogFloorDetail extends LitElement { class DialogFloorDetail extends LitElement {
@property({ attribute: false }) public hass!: HomeAssistant; @property({ attribute: false }) public hass!: HomeAssistant;
@@ -43,11 +33,9 @@ class DialogFloorDetail extends LitElement {
@state() private _submitting?: boolean; @state() private _submitting?: boolean;
@state() private _addedAreas = new Set<string>(); public async showDialog(
params: FloorRegistryDetailDialogParams
@state() private _removedAreas = new Set<string>(); ): Promise<void> {
public showDialog(params: FloorRegistryDetailDialogParams): void {
this._params = params; this._params = params;
this._error = undefined; this._error = undefined;
this._name = this._params.entry this._name = this._params.entry
@@ -56,40 +44,16 @@ class DialogFloorDetail extends LitElement {
this._aliases = this._params.entry?.aliases || []; this._aliases = this._params.entry?.aliases || [];
this._icon = this._params.entry?.icon || null; this._icon = this._params.entry?.icon || null;
this._level = this._params.entry?.level ?? null; this._level = this._params.entry?.level ?? null;
this._addedAreas.clear(); await this.updateComplete;
this._removedAreas.clear();
} }
public closeDialog(): void { public closeDialog(): void {
this._error = ""; this._error = "";
this._params = undefined; this._params = undefined;
this._addedAreas.clear();
this._removedAreas.clear();
fireEvent(this, "dialog-closed", { dialog: this.localName }); fireEvent(this, "dialog-closed", { dialog: this.localName });
} }
private _floorAreas = memoizeOne(
(
entry: FloorRegistryEntry | undefined,
areas: HomeAssistant["areas"],
added: Set<string>,
removed: Set<string>
) =>
Object.values(areas).filter(
(area) =>
(area.floor_id === entry?.floor_id || added.has(area.area_id)) &&
!removed.has(area.area_id)
)
);
protected render() { protected render() {
const areas = this._floorAreas(
this._params?.entry,
this.hass.areas,
this._addedAreas,
this._removedAreas
);
if (!this._params) { if (!this._params) {
return nothing; return nothing;
} }
@@ -161,52 +125,6 @@ class DialogFloorDetail extends LitElement {
: nothing} : nothing}
</ha-icon-picker> </ha-icon-picker>
<h3 class="header">
${this.hass.localize(
"ui.panel.config.floors.editor.areas_section"
)}
</h3>
<p class="description">
${this.hass.localize(
"ui.panel.config.floors.editor.areas_description"
)}
</p>
${areas.length
? html`<ha-chip-set>
${repeat(
areas,
(area) => area.area_id,
(area) =>
html`<ha-input-chip
.area=${area}
@click=${this._openArea}
@remove=${this._removeArea}
.label=${area?.name}
>
${area.icon
? html`<ha-icon
slot="icon"
.icon=${area.icon}
></ha-icon>`
: html`<ha-svg-icon
slot="icon"
.path=${mdiTextureBox}
></ha-svg-icon>`}
</ha-input-chip>`
)}
</ha-chip-set>`
: nothing}
<ha-area-picker
no-add
.hass=${this.hass}
@value-changed=${this._addArea}
.excludeAreas=${areas.map((a) => a.area_id)}
.label=${this.hass.localize(
"ui.panel.config.floors.editor.add_area"
)}
></ha-area-picker>
<h3 class="header"> <h3 class="header">
${this.hass.localize( ${this.hass.localize(
"ui.panel.config.floors.editor.aliases_section" "ui.panel.config.floors.editor.aliases_section"
@@ -241,41 +159,6 @@ class DialogFloorDetail extends LitElement {
`; `;
} }
private _openArea(ev) {
const area = ev.target.area;
showAreaRegistryDetailDialog(this, {
entry: area,
updateEntry: (values) =>
updateAreaRegistryEntry(this.hass!, area.area_id, values),
});
}
private _removeArea(ev) {
const areaId = ev.target.area.area_id;
if (this._addedAreas.has(areaId)) {
this._addedAreas.delete(areaId);
this._addedAreas = new Set(this._addedAreas);
return;
}
this._removedAreas.add(areaId);
this._removedAreas = new Set(this._removedAreas);
}
private _addArea(ev) {
const areaId = ev.detail.value;
if (!areaId) {
return;
}
ev.target.value = "";
if (this._removedAreas.has(areaId)) {
this._removedAreas.delete(areaId);
this._removedAreas = new Set(this._removedAreas);
return;
}
this._addedAreas.add(areaId);
this._addedAreas = new Set(this._addedAreas);
}
private _isNameValid() { private _isNameValid() {
return this._name.trim() !== ""; return this._name.trim() !== "";
} }
@@ -306,13 +189,9 @@ class DialogFloorDetail extends LitElement {
aliases: this._aliases, aliases: this._aliases,
}; };
if (create) { if (create) {
await this._params!.createEntry!(values, this._addedAreas); await this._params!.createEntry!(values);
} else { } else {
await this._params!.updateEntry!( await this._params!.updateEntry!(values);
values,
this._addedAreas,
this._removedAreas
);
} }
this.closeDialog(); this.closeDialog();
} catch (err: any) { } catch (err: any) {
@@ -330,7 +209,6 @@ class DialogFloorDetail extends LitElement {
static get styles(): CSSResultGroup { static get styles(): CSSResultGroup {
return [ return [
haStyle,
haStyleDialog, haStyleDialog,
css` css`
ha-textfield { ha-textfield {
@@ -340,9 +218,6 @@ class DialogFloorDetail extends LitElement {
ha-floor-icon { ha-floor-icon {
color: var(--secondary-text-color); color: var(--secondary-text-color);
} }
ha-chip-set {
margin-bottom: 8px;
}
`, `,
]; ];
} }

View File

@@ -271,14 +271,7 @@ export class HaConfigAreasDashboard extends SubscribeMixin(LitElement) {
? html`<ha-icon .icon=${area.icon}></ha-icon>` ? html`<ha-icon .icon=${area.icon}></ha-icon>`
: ""} : ""}
</div> </div>
<div class="card-header"> <h1 class="card-header">${area.name}</h1>
${area.name}
<ha-icon-button
.area=${area}
.path=${mdiPencil}
@click=${this._openAreaDetails}
></ha-icon-button>
</div>
<div class="card-content"> <div class="card-content">
<div> <div>
${formatListWithAnds( ${formatListWithAnds(
@@ -312,16 +305,6 @@ export class HaConfigAreasDashboard extends SubscribeMixin(LitElement) {
loadAreaRegistryDetailDialog(); loadAreaRegistryDetailDialog();
} }
private _openAreaDetails(ev) {
ev.preventDefault();
const area = ev.currentTarget.area;
showAreaRegistryDetailDialog(this, {
entry: area,
updateEntry: async (values) =>
updateAreaRegistryEntry(this.hass!, area.area_id, values),
});
}
private async _areaMoved(ev) { private async _areaMoved(ev) {
const areasAndFloors = this._processAreas( const areasAndFloors = this._processAreas(
this.hass.areas, this.hass.areas,
@@ -414,31 +397,10 @@ export class HaConfigAreasDashboard extends SubscribeMixin(LitElement) {
private _openFloorDialog(entry?: FloorRegistryEntry) { private _openFloorDialog(entry?: FloorRegistryEntry) {
showFloorRegistryDetailDialog(this, { showFloorRegistryDetailDialog(this, {
entry, entry,
createEntry: async (values, addedAreas) => { createEntry: async (values) =>
const floor = await createFloorRegistryEntry(this.hass!, values); createFloorRegistryEntry(this.hass!, values),
addedAreas.forEach((areaId) => { updateEntry: async (values) =>
updateAreaRegistryEntry(this.hass, areaId, { updateFloorRegistryEntry(this.hass!, entry!.floor_id, values),
floor_id: floor.floor_id,
});
});
},
updateEntry: async (values, addedAreas, removedAreas) => {
const floor = await updateFloorRegistryEntry(
this.hass!,
entry!.floor_id,
values
);
addedAreas.forEach((areaId) => {
updateAreaRegistryEntry(this.hass, areaId, {
floor_id: floor.floor_id,
});
});
removedAreas.forEach((areaId) => {
updateAreaRegistryEntry(this.hass, areaId, {
floor_id: null,
});
});
},
}); });
} }
@@ -507,10 +469,8 @@ export class HaConfigAreasDashboard extends SubscribeMixin(LitElement) {
min-height: 16px; min-height: 16px;
color: var(--secondary-text-color); color: var(--secondary-text-color);
} }
.card-header { .floor {
display: flex; --primary-color: var(--secondary-text-color);
justify-content: space-between;
align-items: center;
} }
.warning { .warning {
color: var(--error-color); color: var(--error-color);

View File

@@ -7,14 +7,9 @@ import {
export interface FloorRegistryDetailDialogParams { export interface FloorRegistryDetailDialogParams {
entry?: FloorRegistryEntry; entry?: FloorRegistryEntry;
suggestedName?: string; suggestedName?: string;
createEntry?: ( createEntry?: (values: FloorRegistryEntryMutableParams) => Promise<unknown>;
values: FloorRegistryEntryMutableParams,
addedAreas: Set<string>
) => Promise<unknown>;
updateEntry?: ( updateEntry?: (
updates: Partial<FloorRegistryEntryMutableParams>, updates: Partial<FloorRegistryEntryMutableParams>
addedAreas: Set<string>,
removedAreas: Set<string>
) => Promise<unknown>; ) => Promise<unknown>;
} }

View File

@@ -1,5 +1,4 @@
import { consume } from "@lit-labs/context"; import { consume } from "@lit-labs/context";
import { ResizeController } from "@lit-labs/observers/resize-controller";
import "@lrnwebcomponents/simple-tooltip/simple-tooltip"; import "@lrnwebcomponents/simple-tooltip/simple-tooltip";
import "@material/web/divider/divider"; import "@material/web/divider/divider";
import { import {
@@ -74,7 +73,6 @@ import {
} from "../../../data/automation"; } from "../../../data/automation";
import { import {
CategoryRegistryEntry, CategoryRegistryEntry,
createCategoryRegistryEntry,
subscribeCategoryRegistry, subscribeCategoryRegistry,
} from "../../../data/category_registry"; } from "../../../data/category_registry";
import { fullEntitiesContext } from "../../../data/context"; import { fullEntitiesContext } from "../../../data/context";
@@ -86,7 +84,6 @@ import {
} from "../../../data/entity_registry"; } from "../../../data/entity_registry";
import { import {
LabelRegistryEntry, LabelRegistryEntry,
createLabelRegistryEntry,
subscribeLabelRegistry, subscribeLabelRegistry,
} from "../../../data/label_registry"; } from "../../../data/label_registry";
import { findRelated } from "../../../data/search"; import { findRelated } from "../../../data/search";
@@ -101,14 +98,11 @@ import { HomeAssistant, Route, ServiceCallResponse } from "../../../types";
import { documentationUrl } from "../../../util/documentation-url"; import { documentationUrl } from "../../../util/documentation-url";
import { turnOnOffEntity } from "../../lovelace/common/entity/turn-on-off-entity"; import { turnOnOffEntity } from "../../lovelace/common/entity/turn-on-off-entity";
import { showAssignCategoryDialog } from "../category/show-dialog-assign-category"; import { showAssignCategoryDialog } from "../category/show-dialog-assign-category";
import { showCategoryRegistryDetailDialog } from "../category/show-dialog-category-registry-detail";
import { configSections } from "../ha-panel-config"; import { configSections } from "../ha-panel-config";
import { showLabelDetailDialog } from "../labels/show-dialog-label-detail";
import { showNewAutomationDialog } from "./show-dialog-new-automation"; import { showNewAutomationDialog } from "./show-dialog-new-automation";
type AutomationItem = AutomationEntity & { type AutomationItem = AutomationEntity & {
name: string; name: string;
area: string | undefined;
last_triggered?: string | undefined; last_triggered?: string | undefined;
formatted_state: string; formatted_state: string;
category: string | undefined; category: string | undefined;
@@ -154,15 +148,10 @@ class HaAutomationPicker extends SubscribeMixin(LitElement) {
@query("#overflow-menu") private _overflowMenu!: HaMenu; @query("#overflow-menu") private _overflowMenu!: HaMenu;
private _sizeController = new ResizeController(this, {
callback: (entries) => entries[0]?.contentRect.width,
});
private _automations = memoizeOne( private _automations = memoizeOne(
( (
automations: AutomationEntity[], automations: AutomationEntity[],
entityReg: EntityRegistryEntry[], entityReg: EntityRegistryEntry[],
areas: HomeAssistant["areas"],
categoryReg?: CategoryRegistryEntry[], categoryReg?: CategoryRegistryEntry[],
labelReg?: LabelRegistryEntry[], labelReg?: LabelRegistryEntry[],
filteredAutomations?: string[] | null filteredAutomations?: string[] | null
@@ -185,9 +174,6 @@ class HaAutomationPicker extends SubscribeMixin(LitElement) {
return { return {
...automation, ...automation,
name: computeStateName(automation), name: computeStateName(automation),
area: entityRegEntry?.area_id
? areas[entityRegEntry?.area_id]?.name
: undefined,
last_triggered: automation.attributes.last_triggered || undefined, last_triggered: automation.attributes.last_triggered || undefined,
formatted_state: this.hass.formatEntityState(automation), formatted_state: this.hass.formatEntityState(automation),
category: category category: category
@@ -256,13 +242,6 @@ class HaAutomationPicker extends SubscribeMixin(LitElement) {
`; `;
}, },
}, },
area: {
title: localize("ui.panel.config.automation.picker.headers.area"),
hidden: true,
groupable: true,
filterable: true,
sortable: true,
},
category: { category: {
title: localize("ui.panel.config.automation.picker.headers.category"), title: localize("ui.panel.config.automation.picker.headers.category"),
hidden: true, hidden: true,
@@ -277,32 +256,33 @@ class HaAutomationPicker extends SubscribeMixin(LitElement) {
template: (automation) => template: (automation) =>
automation.labels.map((lbl) => lbl.name).join(" "), automation.labels.map((lbl) => lbl.name).join(" "),
}, },
last_triggered: { };
sortable: true, columns.last_triggered = {
width: "130px", sortable: true,
title: localize("ui.card.automation.last_triggered"), width: "130px",
hidden: narrow, title: localize("ui.card.automation.last_triggered"),
template: (automation) => { hidden: narrow,
if (!automation.last_triggered) { template: (automation) => {
return this.hass.localize("ui.components.relative_time.never"); if (!automation.last_triggered) {
} return this.hass.localize("ui.components.relative_time.never");
const date = new Date(automation.last_triggered); }
const now = new Date(); const date = new Date(automation.last_triggered);
const dayDifference = differenceInDays(now, date); const now = new Date();
return html` const dayDifference = differenceInDays(now, date);
${dayDifference > 3 return html`
? formatShortDateTime(date, locale, this.hass.config) ${dayDifference > 3
: relativeTime(date, locale)} ? formatShortDateTime(date, locale, this.hass.config)
`; : relativeTime(date, locale)}
}, `;
}, },
formatted_state: { };
if (!this.narrow) {
columns.formatted_state = {
width: "82px", width: "82px",
sortable: true, sortable: true,
groupable: true, groupable: true,
title: "", title: "",
type: "overflow",
hidden: narrow,
label: this.hass.localize("ui.panel.config.automation.picker.state"), label: this.hass.localize("ui.panel.config.automation.picker.state"),
template: (automation) => html` template: (automation) => html`
<ha-entity-toggle <ha-entity-toggle
@@ -310,20 +290,21 @@ class HaAutomationPicker extends SubscribeMixin(LitElement) {
.hass=${this.hass} .hass=${this.hass}
></ha-entity-toggle> ></ha-entity-toggle>
`, `,
}, };
actions: { }
title: "",
width: "64px", columns.actions = {
type: "icon-button", title: "",
template: (automation) => html` width: "64px",
<ha-icon-button type: "icon-button",
.automation=${automation} template: (automation) => html`
.label=${this.hass.localize("ui.common.overflow_menu")} <ha-icon-button
.path=${mdiDotsVertical} .automation=${automation}
@click=${this._showOverflowMenu} .label=${this.hass.localize("ui.common.overflow_menu")}
></ha-icon-button> .path=${mdiDotsVertical}
`, @click=${this._showOverflowMenu}
}, ></ha-icon-button>
`,
}; };
return columns; return columns;
} }
@@ -376,52 +357,22 @@ class HaAutomationPicker extends SubscribeMixin(LitElement) {
"ui.panel.config.automation.picker.bulk_actions.no_category" "ui.panel.config.automation.picker.bulk_actions.no_category"
)} )}
</div> </div>
</ha-menu-item>
<md-divider role="separator" tabindex="-1"></md-divider>
<ha-menu-item @click=${this._bulkCreateCategory}>
<div slot="headline">
${this.hass.localize("ui.panel.config.category.editor.add")}
</div>
</ha-menu-item>`; </ha-menu-item>`;
const labelItems = html`${this._labels?.map((label) => { const labelItems = html` ${this._labels?.map((label) => {
const color = label.color ? computeCssColor(label.color) : undefined; const color = label.color ? computeCssColor(label.color) : undefined;
const selected = this._selected.every((entityId) => return html`<ha-menu-item
this.hass.entities[entityId]?.labels.includes(label.label_id) .value=${label.label_id}
); @click=${this._handleBulkLabel}
const partial = >
!selected && <ha-label style=${color ? `--color: ${color}` : ""}>
this._selected.some((entityId) => ${label.icon
this.hass.entities[entityId]?.labels.includes(label.label_id) ? html`<ha-icon slot="icon" .icon=${label.icon}></ha-icon>`
); : nothing}
return html`<ha-menu-item ${label.name}
.value=${label.label_id} </ha-label>
.action=${selected ? "remove" : "add"} </ha-menu-item>`;
@click=${this._handleBulkLabel} })}`;
keep-open
>
<ha-checkbox
slot="start"
.checked=${selected}
.indeterminate=${partial}
reducedTouchTarget
></ha-checkbox>
<ha-label style=${color ? `--color: ${color}` : ""}>
${label.icon
? html`<ha-icon slot="icon" .icon=${label.icon}></ha-icon>`
: nothing}
${label.name}
</ha-label>
</ha-menu-item>`;
})}
<md-divider role="separator" tabindex="-1"></md-divider>
<ha-menu-item @click=${this._bulkCreateLabel}>
<div slot="headline">
${this.hass.localize("ui.panel.config.labels.add_label")}
</div></ha-menu-item
>`;
const labelsInOverflow =
(this._sizeController.value && this._sizeController.value < 700) ||
(!this._sizeController.value && this.hass.dockedSidebar === "docked");
return html` return html`
<hass-tabs-subpage-data-table <hass-tabs-subpage-data-table
.hass=${this.hass} .hass=${this.hass}
@@ -449,7 +400,6 @@ class HaAutomationPicker extends SubscribeMixin(LitElement) {
.data=${this._automations( .data=${this._automations(
this.automations, this.automations,
this._entityReg, this._entityReg,
this.hass.areas,
this._categories, this._categories,
this._labels, this._labels,
this._filteredAutomations this._filteredAutomations
@@ -545,7 +495,7 @@ class HaAutomationPicker extends SubscribeMixin(LitElement) {
</ha-assist-chip> </ha-assist-chip>
${categoryItems} ${categoryItems}
</ha-button-menu-new> </ha-button-menu-new>
${labelsInOverflow ${this.hass.dockedSidebar === "docked"
? nothing ? nothing
: html`<ha-button-menu-new slot="selection-bar"> : html`<ha-button-menu-new slot="selection-bar">
<ha-assist-chip <ha-assist-chip
@@ -607,7 +557,7 @@ class HaAutomationPicker extends SubscribeMixin(LitElement) {
: nothing : nothing
} }
${ ${
this.narrow || labelsInOverflow this.narrow || this.hass.dockedSidebar === "docked"
? html`<ha-sub-menu> ? html`<ha-sub-menu>
<ha-menu-item slot="item"> <ha-menu-item slot="item">
<div slot="headline"> <div slot="headline">
@@ -1089,10 +1039,6 @@ class HaAutomationPicker extends SubscribeMixin(LitElement) {
private async _handleBulkCategory(ev) { private async _handleBulkCategory(ev) {
const category = ev.currentTarget.value; const category = ev.currentTarget.value;
this._bulkAddCategory(category);
}
private async _bulkAddCategory(category: string) {
const promises: Promise<UpdateEntityRegistryEntryResult>[] = []; const promises: Promise<UpdateEntityRegistryEntryResult>[] = [];
this._selected.forEach((entityId) => { this._selected.forEach((entityId) => {
promises.push( promises.push(
@@ -1106,21 +1052,11 @@ class HaAutomationPicker extends SubscribeMixin(LitElement) {
private async _handleBulkLabel(ev) { private async _handleBulkLabel(ev) {
const label = ev.currentTarget.value; const label = ev.currentTarget.value;
const action = ev.currentTarget.action;
this._bulkLabel(label, action);
}
private async _bulkLabel(label: string, action: "add" | "remove") {
const promises: Promise<UpdateEntityRegistryEntryResult>[] = []; const promises: Promise<UpdateEntityRegistryEntryResult>[] = [];
this._selected.forEach((entityId) => { this._selected.forEach((entityId) => {
promises.push( promises.push(
updateEntityRegistryEntry(this.hass, entityId, { updateEntityRegistryEntry(this.hass, entityId, {
labels: labels: this.hass.entities[entityId].labels.concat(label),
action === "add"
? this.hass.entities[entityId].labels.concat(label)
: this.hass.entities[entityId].labels.filter(
(lbl) => lbl !== label
),
}) })
); );
}); });
@@ -1143,38 +1079,10 @@ class HaAutomationPicker extends SubscribeMixin(LitElement) {
await Promise.all(promises); await Promise.all(promises);
} }
private async _bulkCreateCategory() {
showCategoryRegistryDetailDialog(this, {
scope: "automation",
createEntry: async (values) => {
const category = await createCategoryRegistryEntry(
this.hass,
"automation",
values
);
this._bulkAddCategory(category.category_id);
return category;
},
});
}
private _bulkCreateLabel() {
showLabelDetailDialog(this, {
createEntry: async (values) => {
const label = await createLabelRegistryEntry(this.hass, values);
this._bulkLabel(label.label_id, "add");
return label;
},
});
}
static get styles(): CSSResultGroup { static get styles(): CSSResultGroup {
return [ return [
haStyle, haStyle,
css` css`
:host {
display: block;
}
hass-tabs-subpage-data-table { hass-tabs-subpage-data-table {
--data-table-row-height: 60px; --data-table-row-height: 60px;
} }

View File

@@ -237,8 +237,6 @@ export class HaCategoryPicker extends SubscribeMixin(LitElement) {
(ev.target as any).value = this._value; (ev.target as any).value = this._value;
this.hass.loadFragmentTranslation("config");
showCategoryRegistryDetailDialog(this, { showCategoryRegistryDetailDialog(this, {
scope: this.scope!, scope: this.scope!,
suggestedName: newValue === ADD_NEW_SUGGESTION_ID ? this._suggestion : "", suggestedName: newValue === ADD_NEW_SUGGESTION_ID ? this._suggestion : "",

View File

@@ -1,6 +1,6 @@
import { consume } from "@lit-labs/context"; import { consume } from "@lit-labs/context";
import "@lrnwebcomponents/simple-tooltip/simple-tooltip"; import "@lrnwebcomponents/simple-tooltip/simple-tooltip";
import { mdiChevronRight, mdiMenuDown, mdiPlus } from "@mdi/js"; import { mdiPlus } from "@mdi/js";
import { import {
CSSResultGroup, CSSResultGroup,
LitElement, LitElement,
@@ -13,7 +13,6 @@ import {
import { UnsubscribeFunc } from "home-assistant-js-websocket"; import { UnsubscribeFunc } from "home-assistant-js-websocket";
import { customElement, property, state } from "lit/decorators"; import { customElement, property, state } from "lit/decorators";
import memoizeOne from "memoize-one"; import memoizeOne from "memoize-one";
import { computeCssColor } from "../../../common/color/compute-color";
import { HASSDomEvent } from "../../../common/dom/fire_event"; import { HASSDomEvent } from "../../../common/dom/fire_event";
import { computeStateDomain } from "../../../common/entity/compute_state_domain"; import { computeStateDomain } from "../../../common/entity/compute_state_domain";
import { import {
@@ -25,7 +24,6 @@ import { LocalizeFunc } from "../../../common/translations/localize";
import { import {
DataTableColumnContainer, DataTableColumnContainer,
RowClickedEvent, RowClickedEvent,
SelectionChangedEvent,
} from "../../../components/data-table/ha-data-table"; } from "../../../components/data-table/ha-data-table";
import "../../../components/data-table/ha-data-table-labels"; import "../../../components/data-table/ha-data-table-labels";
import "../../../components/entity/ha-battery-icon"; import "../../../components/entity/ha-battery-icon";
@@ -39,15 +37,12 @@ import "../../../components/ha-filter-integrations";
import "../../../components/ha-filter-labels"; import "../../../components/ha-filter-labels";
import "../../../components/ha-filter-states"; import "../../../components/ha-filter-states";
import "../../../components/ha-icon-button"; import "../../../components/ha-icon-button";
import "../../../components/ha-menu-item";
import "../../../components/ha-sub-menu";
import { ConfigEntry, sortConfigEntries } from "../../../data/config_entries"; import { ConfigEntry, sortConfigEntries } from "../../../data/config_entries";
import { fullEntitiesContext } from "../../../data/context"; import { fullEntitiesContext } from "../../../data/context";
import { import {
DeviceEntityLookup, DeviceEntityLookup,
DeviceRegistryEntry, DeviceRegistryEntry,
computeDeviceName, computeDeviceName,
updateDeviceRegistryEntry,
} from "../../../data/device_registry"; } from "../../../data/device_registry";
import { import {
EntityRegistryEntry, EntityRegistryEntry,
@@ -57,7 +52,6 @@ import {
import { IntegrationManifest } from "../../../data/integration"; import { IntegrationManifest } from "../../../data/integration";
import { import {
LabelRegistryEntry, LabelRegistryEntry,
createLabelRegistryEntry,
subscribeLabelRegistry, subscribeLabelRegistry,
} from "../../../data/label_registry"; } from "../../../data/label_registry";
import "../../../layouts/hass-tabs-subpage-data-table"; import "../../../layouts/hass-tabs-subpage-data-table";
@@ -68,7 +62,6 @@ import { brandsUrl } from "../../../util/brands-url";
import { configSections } from "../ha-panel-config"; import { configSections } from "../ha-panel-config";
import "../integrations/ha-integration-overflow-menu"; import "../integrations/ha-integration-overflow-menu";
import { showAddIntegrationDialog } from "../integrations/show-add-integration-dialog"; import { showAddIntegrationDialog } from "../integrations/show-add-integration-dialog";
import { showLabelDetailDialog } from "../labels/show-dialog-label-detail";
interface DeviceRowData extends DeviceRegistryEntry { interface DeviceRowData extends DeviceRegistryEntry {
device?: DeviceRowData; device?: DeviceRowData;
@@ -98,8 +91,6 @@ export class HaConfigDeviceDashboard extends SubscribeMixin(LitElement) {
@state() private _searchParms = new URLSearchParams(window.location.search); @state() private _searchParms = new URLSearchParams(window.location.search);
@state() private _selected: string[] = [];
@state() private _filter: string = history.state?.filter || ""; @state() private _filter: string = history.state?.filter || "";
@state() private _filters: Record< @state() private _filters: Record<
@@ -544,43 +535,6 @@ export class HaConfigDeviceDashboard extends SubscribeMixin(LitElement) {
this._labels this._labels
); );
const labelItems = html`${this._labels?.map((label) => {
const color = label.color ? computeCssColor(label.color) : undefined;
const selected = this._selected.every((deviceId) =>
this.hass.devices[deviceId]?.labels.includes(label.label_id)
);
const partial =
!selected &&
this._selected.some((deviceId) =>
this.hass.devices[deviceId]?.labels.includes(label.label_id)
);
return html`<ha-menu-item
.value=${label.label_id}
.action=${selected ? "remove" : "add"}
@click=${this._handleBulkLabel}
keep-open
>
<ha-checkbox
slot="start"
.checked=${selected}
.indeterminate=${partial}
reducedTouchTarget
></ha-checkbox>
<ha-label style=${color ? `--color: ${color}` : ""}>
${label.icon
? html`<ha-icon slot="icon" .icon=${label.icon}></ha-icon>`
: nothing}
${label.name}
</ha-label>
</ha-menu-item>`;
})}
<md-divider role="separator" tabindex="-1"></md-divider>
<ha-menu-item @click=${this._bulkCreateLabel}>
<div slot="headline">
${this.hass.localize("ui.panel.config.labels.add_label")}
</div></ha-menu-item
>`;
return html` return html`
<hass-tabs-subpage-data-table <hass-tabs-subpage-data-table
.hass=${this.hass} .hass=${this.hass}
@@ -595,9 +549,6 @@ export class HaConfigDeviceDashboard extends SubscribeMixin(LitElement) {
)} )}
.columns=${this._columns(this.hass.localize, this.narrow)} .columns=${this._columns(this.hass.localize, this.narrow)}
.data=${devicesOutput} .data=${devicesOutput}
selectable
.selected=${this._selected.length}
@selection-changed=${this._handleSelectionChanged}
.filter=${this._filter} .filter=${this._filter}
hasFilters hasFilters
.filters=${Object.values(this._filters).filter( .filters=${Object.values(this._filters).filter(
@@ -670,49 +621,6 @@ export class HaConfigDeviceDashboard extends SubscribeMixin(LitElement) {
.narrow=${this.narrow} .narrow=${this.narrow}
@expanded-changed=${this._filterExpanded} @expanded-changed=${this._filterExpanded}
></ha-filter-labels> ></ha-filter-labels>
${!this.narrow
? html`<ha-button-menu-new slot="selection-bar">
<ha-assist-chip
slot="trigger"
.label=${this.hass.localize(
"ui.panel.config.automation.picker.bulk_actions.add_label"
)}
>
<ha-svg-icon
slot="trailing-icon"
.path=${mdiMenuDown}
></ha-svg-icon>
</ha-assist-chip>
${labelItems}
</ha-button-menu-new>`
: html` <ha-button-menu-new has-overflow slot="selection-bar"
><ha-assist-chip
.label=${this.hass.localize(
"ui.panel.config.automation.picker.bulk_action"
)}
slot="trigger"
>
<ha-svg-icon
slot="trailing-icon"
.path=${mdiMenuDown}
></ha-svg-icon>
</ha-assist-chip>
<ha-sub-menu>
<ha-menu-item slot="item">
<div slot="headline">
${this.hass.localize(
"ui.panel.config.automation.picker.bulk_actions.add_label"
)}
</div>
<ha-svg-icon
slot="end"
.path=${mdiChevronRight}
></ha-svg-icon>
</ha-menu-item>
<ha-menu slot="menu">${labelItems}</ha-menu>
</ha-sub-menu>
</ha-button-menu-new>`}
</hass-tabs-subpage-data-table> </hass-tabs-subpage-data-table>
`; `;
} }
@@ -792,45 +700,6 @@ export class HaConfigDeviceDashboard extends SubscribeMixin(LitElement) {
}); });
} }
private _handleSelectionChanged(
ev: HASSDomEvent<SelectionChangedEvent>
): void {
this._selected = ev.detail.value;
}
private async _handleBulkLabel(ev) {
const label = ev.currentTarget.value;
const action = ev.currentTarget.action;
this._bulkLabel(label, action);
}
private async _bulkLabel(label: string, action: "add" | "remove") {
const promises: Promise<DeviceRegistryEntry>[] = [];
this._selected.forEach((deviceId) => {
promises.push(
updateDeviceRegistryEntry(this.hass, deviceId, {
labels:
action === "add"
? this.hass.devices[deviceId].labels.concat(label)
: this.hass.devices[deviceId].labels.filter(
(lbl) => lbl !== label
),
})
);
});
await Promise.all(promises);
}
private _bulkCreateLabel() {
showLabelDetailDialog(this, {
createEntry: async (values) => {
const label = await createLabelRegistryEntry(this.hass, values);
this._bulkLabel(label.label_id, "add");
return label;
},
});
}
static get styles(): CSSResultGroup { static get styles(): CSSResultGroup {
return [ return [
css` css`
@@ -852,16 +721,6 @@ export class HaConfigDeviceDashboard extends SubscribeMixin(LitElement) {
text-transform: uppercase; text-transform: uppercase;
direction: var(--direction); direction: var(--direction);
} }
ha-assist-chip {
--ha-assist-chip-container-shape: 10px;
}
ha-button-menu-new ha-assist-chip {
--md-assist-chip-trailing-space: 8px;
}
ha-label {
--ha-label-background-color: var(--color, var(--grey-color));
--ha-label-background-opacity: 0.5;
}
`, `,
haStyle, haStyle,
]; ];

View File

@@ -3,17 +3,12 @@ import "@lrnwebcomponents/simple-tooltip/simple-tooltip";
import { import {
mdiAlertCircle, mdiAlertCircle,
mdiCancel, mdiCancel,
mdiChevronRight,
mdiDelete, mdiDelete,
mdiDotsVertical,
mdiEye,
mdiEyeOff, mdiEyeOff,
mdiMenuDown,
mdiPencilOff, mdiPencilOff,
mdiPlus, mdiPlus,
mdiRestoreAlert, mdiRestoreAlert,
mdiToggleSwitch, mdiUndo,
mdiToggleSwitchOffOutline,
} from "@mdi/js"; } from "@mdi/js";
import { HassEntity, UnsubscribeFunc } from "home-assistant-js-websocket"; import { HassEntity, UnsubscribeFunc } from "home-assistant-js-websocket";
import { import {
@@ -29,7 +24,6 @@ import { ifDefined } from "lit/directives/if-defined";
import { styleMap } from "lit/directives/style-map"; import { styleMap } from "lit/directives/style-map";
import { until } from "lit/directives/until"; import { until } from "lit/directives/until";
import memoize from "memoize-one"; import memoize from "memoize-one";
import { computeCssColor } from "../../../common/color/compute-color";
import type { HASSDomEvent } from "../../../common/dom/fire_event"; import type { HASSDomEvent } from "../../../common/dom/fire_event";
import { computeDomain } from "../../../common/entity/compute_domain"; import { computeDomain } from "../../../common/entity/compute_domain";
import { computeStateName } from "../../../common/entity/compute_state_name"; import { computeStateName } from "../../../common/entity/compute_state_name";
@@ -50,19 +44,16 @@ import "../../../components/ha-check-list-item";
import "../../../components/ha-filter-devices"; import "../../../components/ha-filter-devices";
import "../../../components/ha-filter-floor-areas"; import "../../../components/ha-filter-floor-areas";
import "../../../components/ha-filter-integrations"; import "../../../components/ha-filter-integrations";
import "../../../components/ha-filter-labels";
import "../../../components/ha-filter-states"; import "../../../components/ha-filter-states";
import "../../../components/ha-filter-labels";
import "../../../components/ha-icon"; import "../../../components/ha-icon";
import "../../../components/ha-icon-button"; import "../../../components/ha-icon-button";
import "../../../components/ha-menu-item";
import "../../../components/ha-sub-menu";
import "../../../components/ha-svg-icon"; import "../../../components/ha-svg-icon";
import { ConfigEntry, getConfigEntries } from "../../../data/config_entries"; import { ConfigEntry, getConfigEntries } from "../../../data/config_entries";
import { fullEntitiesContext } from "../../../data/context"; import { fullEntitiesContext } from "../../../data/context";
import { UNAVAILABLE } from "../../../data/entity"; import { UNAVAILABLE } from "../../../data/entity";
import { import {
EntityRegistryEntry, EntityRegistryEntry,
UpdateEntityRegistryEntryResult,
computeEntityRegistryName, computeEntityRegistryName,
removeEntityRegistryEntry, removeEntityRegistryEntry,
updateEntityRegistryEntry, updateEntityRegistryEntry,
@@ -70,7 +61,6 @@ import {
import { entryIcon } from "../../../data/icons"; import { entryIcon } from "../../../data/icons";
import { import {
LabelRegistryEntry, LabelRegistryEntry,
createLabelRegistryEntry,
subscribeLabelRegistry, subscribeLabelRegistry,
} from "../../../data/label_registry"; } from "../../../data/label_registry";
import { import {
@@ -87,11 +77,6 @@ import type { HomeAssistant, Route } from "../../../types";
import { configSections } from "../ha-panel-config"; import { configSections } from "../ha-panel-config";
import "../integrations/ha-integration-overflow-menu"; import "../integrations/ha-integration-overflow-menu";
import { showAddIntegrationDialog } from "../integrations/show-add-integration-dialog"; import { showAddIntegrationDialog } from "../integrations/show-add-integration-dialog";
import { showLabelDetailDialog } from "../labels/show-dialog-label-detail";
import {
EntitySources,
fetchEntitySourcesWithCache,
} from "../../../data/entity_sources";
export interface StateEntity export interface StateEntity
extends Omit<EntityRegistryEntry, "id" | "unique_id"> { extends Omit<EntityRegistryEntry, "id" | "unique_id"> {
@@ -138,15 +123,13 @@ export class HaConfigEntities extends SubscribeMixin(LitElement) {
{ value: string[] | undefined; items: Set<string> | undefined } { value: string[] | undefined; items: Set<string> | undefined }
> = {}; > = {};
@state() private _selected: string[] = []; @state() private _selectedEntities: string[] = [];
@state() private _expandedFilter?: string; @state() private _expandedFilter?: string;
@state() @state()
_labels!: LabelRegistryEntry[]; _labels!: LabelRegistryEntry[];
@state() private _entitySources?: EntitySources;
@query("hass-tabs-subpage-data-table", true) @query("hass-tabs-subpage-data-table", true)
private _dataTable!: HaTabsSubpageDataTable; private _dataTable!: HaTabsSubpageDataTable;
@@ -411,12 +394,10 @@ export class HaConfigEntities extends SubscribeMixin(LitElement) {
const entryIds = entries const entryIds = entries
.filter((entry) => filter.value!.includes(entry.domain)) .filter((entry) => filter.value!.includes(entry.domain))
.map((entry) => entry.entry_id); .map((entry) => entry.entry_id);
filteredEntities = filteredEntities.filter( filteredEntities = filteredEntities.filter(
(entity) => (entity) =>
filter.value?.includes(entity.platform) || entity.config_entry_id &&
(entity.config_entry_id && entryIds.includes(entity.config_entry_id)
entryIds.includes(entity.config_entry_id))
); );
filter.value!.forEach((domain) => filteredDomains.add(domain)); filter.value!.forEach((domain) => filteredDomains.add(domain));
} else if (key === "ha-filter-labels" && filter.value?.length) { } else if (key === "ha-filter-labels" && filter.value?.length) {
@@ -524,50 +505,13 @@ export class HaConfigEntities extends SubscribeMixin(LitElement) {
[...filteredDomains][0] [...filteredDomains][0]
); );
const labelItems = html` ${this._labels?.map((label) => {
const color = label.color ? computeCssColor(label.color) : undefined;
const selected = this._selected.every((entityId) =>
this.hass.entities[entityId]?.labels.includes(label.label_id)
);
const partial =
!selected &&
this._selected.some((entityId) =>
this.hass.entities[entityId]?.labels.includes(label.label_id)
);
return html`<ha-menu-item
.value=${label.label_id}
.action=${selected ? "remove" : "add"}
@click=${this._handleBulkLabel}
keep-open
>
<ha-checkbox
slot="start"
.checked=${selected}
.indeterminate=${partial}
reducedTouchTarget
></ha-checkbox>
<ha-label style=${color ? `--color: ${color}` : ""}>
${label.icon
? html`<ha-icon slot="icon" .icon=${label.icon}></ha-icon>`
: nothing}
${label.name}
</ha-label>
</ha-menu-item>`;
})}
<md-divider role="separator" tabindex="-1"></md-divider>
<ha-menu-item @click=${this._bulkCreateLabel}>
<div slot="headline">
${this.hass.localize("ui.panel.config.labels.add_label")}
</div></ha-menu-item
>`;
return html` return html`
<hass-tabs-subpage-data-table <hass-tabs-subpage-data-table
.hass=${this.hass} .hass=${this.hass}
.narrow=${this.narrow} .narrow=${this.narrow}
.backPath=${ .backPath=${this._searchParms.has("historyBack")
this._searchParms.has("historyBack") ? undefined : "/config" ? undefined
} : "/config"}
.route=${this.route} .route=${this.route}
.tabs=${configSections.devices} .tabs=${configSections.devices}
.columns=${this._columns( .columns=${this._columns(
@@ -580,13 +524,12 @@ export class HaConfigEntities extends SubscribeMixin(LitElement) {
"ui.panel.config.entities.picker.search" "ui.panel.config.entities.picker.search"
)} )}
hasFilters hasFilters
.filters=${ .filters=${Object.values(this._filters).filter(
Object.values(this._filters).filter((filter) => filter.value?.length) (filter) => filter.value?.length
.length ).length}
}
.filter=${this._filter} .filter=${this._filter}
selectable selectable
.selected=${this._selected.length} .selected=${this._selectedEntities.length}
@selection-changed=${this._handleSelectionChanged} @selection-changed=${this._handleSelectionChanged}
clickable clickable
@clear-filter=${this._clearFilter} @clear-filter=${this._clearFilter}
@@ -600,131 +543,100 @@ export class HaConfigEntities extends SubscribeMixin(LitElement) {
.hass=${this.hass} .hass=${this.hass}
slot="toolbar-icon" slot="toolbar-icon"
></ha-integration-overflow-menu> ></ha-integration-overflow-menu>
<div class="header-btns" slot="selection-bar">
${!this.narrow
${ ? html`
!this.narrow <mwc-button
? html`<ha-button-menu-new slot="selection-bar"> @click=${this._enableSelected}
<ha-assist-chip .disabled=${!this._selectedEntities.length}
slot="trigger" >${this.hass.localize(
.label=${this.hass.localize( "ui.panel.config.entities.picker.enable_selected.button"
"ui.panel.config.automation.picker.bulk_actions.add_label" )}</mwc-button
)} >
> <mwc-button
<ha-svg-icon slot="trailing-icon" .path=${mdiMenuDown}></ha-svg-icon> @click=${this._disableSelected}
</ha-assist-chip> .disabled=${!this._selectedEntities.length}
${labelItems} >${this.hass.localize(
</ha-button-menu-new>` "ui.panel.config.entities.picker.disable_selected.button"
: nothing )}</mwc-button
} >
<ha-button-menu-new has-overflow slot="selection-bar"> <mwc-button
${ @click=${this._hideSelected}
this.narrow .disabled=${!this._selectedEntities.length}
? html`<ha-assist-chip >${this.hass.localize(
.label=${this.hass.localize( "ui.panel.config.entities.picker.hide_selected.button"
"ui.panel.config.automation.picker.bulk_action" )}</mwc-button
)} >
slot="trigger" <mwc-button
> @click=${this._removeSelected}
<ha-svg-icon slot="trailing-icon" .path=${mdiMenuDown}></ha-svg-icon> .disabled=${!this._selectedEntities.length}
</ha-assist-chip>` class="warning"
: html`<ha-icon-button >${this.hass.localize(
.path=${mdiDotsVertical} "ui.panel.config.entities.picker.remove_selected.button"
.label=${"ui.panel.config.automation.picker.bulk_action"} )}</mwc-button
slot="trigger" >
></ha-icon-button>` `
} : html`
<ha-svg-icon <ha-icon-button
slot="trailing-icon" id="enable-btn"
.path=${mdiMenuDown} .disabled=${!this._selectedEntities.length}
></ha-svg-icon @click=${this._enableSelected}
></ha-assist-chip> .path=${mdiUndo}
${ .label=${this.hass.localize("ui.common.enable")}
this.narrow ></ha-icon-button>
? html`<ha-sub-menu> <simple-tooltip animation-delay="0" for="enable-btn">
<ha-menu-item slot="item"> ${this.hass.localize(
<div slot="headline"> "ui.panel.config.entities.picker.enable_selected.button"
${this.hass.localize( )}
"ui.panel.config.automation.picker.bulk_actions.add_label" </simple-tooltip>
)} <ha-icon-button
</div> id="disable-btn"
<ha-svg-icon slot="end" .path=${mdiChevronRight}></ha-svg-icon> .disabled=${!this._selectedEntities.length}
</ha-menu-item> @click=${this._disableSelected}
<ha-menu slot="menu">${labelItems}</ha-menu> .path=${mdiCancel}
</ha-sub-menu> .label=${this.hass.localize("ui.common.disable")}
<md-divider role="separator" tabindex="-1"></md-divider>` ></ha-icon-button>
: nothing <simple-tooltip animation-delay="0" for="disable-btn">
} ${this.hass.localize(
"ui.panel.config.entities.picker.disable_selected.button"
<ha-menu-item @click=${this._enableSelected}> )}
<ha-svg-icon slot="start" .path=${mdiToggleSwitch}></ha-svg-icon> </simple-tooltip>
<div slot="headline"> <ha-icon-button
${this.hass.localize( id="hide-btn"
"ui.panel.config.entities.picker.enable_selected.button" .disabled=${!this._selectedEntities.length}
)} @click=${this._hideSelected}
</div> .path=${mdiEyeOff}
</ha-menu-item> .label=${this.hass.localize("ui.common.hide")}
<ha-menu-item @click=${this._disableSelected}> ></ha-icon-button>
<ha-svg-icon <simple-tooltip animation-delay="0" for="hide-btn">
slot="start" ${this.hass.localize(
.path=${mdiToggleSwitchOffOutline} "ui.panel.config.entities.picker.hide_selected.button"
></ha-svg-icon> )}
<div slot="headline"> </simple-tooltip>
${this.hass.localize( <ha-icon-button
"ui.panel.config.entities.picker.disable_selected.button" class="warning"
)} id="remove-btn"
</div> .disabled=${!this._selectedEntities.length}
</ha-menu-item> @click=${this._removeSelected}
<md-divider role="separator" tabindex="-1"></md-divider> .path=${mdiDelete}
.label=${this.hass.localize("ui.common.remove")}
<ha-menu-item @click=${this._unhideSelected}> ></ha-icon-button>
<ha-svg-icon <simple-tooltip animation-delay="0" for="remove-btn">
slot="start" ${this.hass.localize(
.path=${mdiEye} "ui.panel.config.entities.picker.remove_selected.button"
></ha-svg-icon> )}
<div slot="headline"> </simple-tooltip>
${this.hass.localize( `}
"ui.panel.config.entities.picker.unhide_selected.button" </div>
)} ${this._filters.config_entry?.value?.length
</div> ? html`<ha-alert slot="filter-pane">
</ha-menu-item> Filtering by config entry
<ha-menu-item @click=${this._hideSelected}> ${this._entries?.find(
<ha-svg-icon (entry) =>
slot="start" entry.entry_id === this._filters.config_entry!.value![0]
.path=${mdiEyeOff} )?.title || this._filters.config_entry.value[0]}
></ha-svg-icon> </ha-alert>`
<div slot="headline"> : nothing}
${this.hass.localize(
"ui.panel.config.entities.picker.hide_selected.button"
)}
</div>
</ha-menu-item>
<md-divider role="separator" tabindex="-1"></md-divider>
<ha-menu-item @click=${this._removeSelected} class="warning">
<ha-svg-icon
slot="start"
.path=${mdiDelete}
></ha-svg-icon>
<div slot="headline">
${this.hass.localize(
"ui.panel.config.entities.picker.remove_selected.button"
)}
</div>
</ha-menu-item>
</ha-button-menu-new>
${
this._filters.config_entry?.value?.length
? html`<ha-alert slot="filter-pane">
Filtering by config entry
${this._entries?.find(
(entry) =>
entry.entry_id === this._filters.config_entry!.value![0]
)?.title || this._filters.config_entry.value[0]}
</ha-alert>`
: nothing
}
<ha-filter-floor-areas <ha-filter-floor-areas
.hass=${this.hass} .hass=${this.hass}
type="entity" type="entity"
@@ -776,20 +688,16 @@ ${
.narrow=${this.narrow} .narrow=${this.narrow}
@expanded-changed=${this._filterExpanded} @expanded-changed=${this._filterExpanded}
></ha-filter-labels> ></ha-filter-labels>
${ ${includeAddDeviceFab
includeAddDeviceFab ? html`<ha-fab
? html`<ha-fab .label=${this.hass.localize("ui.panel.config.devices.add_device")}
.label=${this.hass.localize( extended
"ui.panel.config.devices.add_device" @click=${this._addDevice}
)} slot="fab"
extended >
@click=${this._addDevice} <ha-svg-icon slot="icon" .path=${mdiPlus}></ha-svg-icon>
slot="fab" </ha-fab>`
> : nothing}
<ha-svg-icon slot="icon" .path=${mdiPlus}></ha-svg-icon>
</ha-fab>`
: nothing
}
</hass-tabs-subpage-data-table> </hass-tabs-subpage-data-table>
`; `;
} }
@@ -815,9 +723,6 @@ ${
}, },
}; };
this._setFiltersFromUrl(); this._setFiltersFromUrl();
fetchEntitySourcesWithCache(this.hass).then((sources) => {
this._entitySources = sources;
});
} }
private _setFiltersFromUrl() { private _setFiltersFromUrl() {
@@ -876,18 +781,14 @@ ${
this._filters = {}; this._filters = {};
} }
public willUpdate(changedProps: PropertyValues): void { public willUpdate(changedProps: PropertyValues<this>): void {
super.willUpdate(changedProps); super.willUpdate(changedProps);
const oldHass = changedProps.get("hass"); const oldHass = changedProps.get("hass");
let changed = false; let changed = false;
if (!this.hass || !this._entities) { if (!this.hass || !this._entities) {
return; return;
} }
if ( if (changedProps.has("hass") || changedProps.has("_entities")) {
changedProps.has("hass") ||
changedProps.has("_entities") ||
changedProps.has("_entitySources")
) {
const stateEntities: StateEntity[] = []; const stateEntities: StateEntity[] = [];
const regEntityIds = new Set( const regEntityIds = new Set(
this._entities.map((entity) => entity.entity_id) this._entities.map((entity) => entity.entity_id)
@@ -898,7 +799,6 @@ ${
} }
if ( if (
!oldHass || !oldHass ||
changedProps.has("_entitySources") ||
this.hass.states[entityId] !== oldHass.states[entityId] this.hass.states[entityId] !== oldHass.states[entityId]
) { ) {
changed = true; changed = true;
@@ -906,8 +806,7 @@ ${
stateEntities.push({ stateEntities.push({
name: computeStateName(this.hass.states[entityId]), name: computeStateName(this.hass.states[entityId]),
entity_id: entityId, entity_id: entityId,
platform: platform: computeDomain(entityId),
this._entitySources?.[entityId]?.domain || computeDomain(entityId),
disabled_by: null, disabled_by: null,
hidden_by: null, hidden_by: null,
area_id: null, area_id: null,
@@ -937,14 +836,14 @@ ${
private _handleSelectionChanged( private _handleSelectionChanged(
ev: HASSDomEvent<SelectionChangedEvent> ev: HASSDomEvent<SelectionChangedEvent>
): void { ): void {
this._selected = ev.detail.value; this._selectedEntities = ev.detail.value;
} }
private async _enableSelected() { private async _enableSelected() {
showConfirmationDialog(this, { showConfirmationDialog(this, {
title: this.hass.localize( title: this.hass.localize(
"ui.panel.config.entities.picker.enable_selected.confirm_title", "ui.panel.config.entities.picker.enable_selected.confirm_title",
{ number: this._selected.length } { number: this._selectedEntities.length }
), ),
text: this.hass.localize( text: this.hass.localize(
"ui.panel.config.entities.picker.enable_selected.confirm_text" "ui.panel.config.entities.picker.enable_selected.confirm_text"
@@ -955,7 +854,7 @@ ${
let require_restart = false; let require_restart = false;
let reload_delay = 0; let reload_delay = 0;
await Promise.all( await Promise.all(
this._selected.map(async (entity) => { this._selectedEntities.map(async (entity) => {
const result = await updateEntityRegistryEntry(this.hass, entity, { const result = await updateEntityRegistryEntry(this.hass, entity, {
disabled_by: null, disabled_by: null,
}); });
@@ -992,7 +891,7 @@ ${
showConfirmationDialog(this, { showConfirmationDialog(this, {
title: this.hass.localize( title: this.hass.localize(
"ui.panel.config.entities.picker.disable_selected.confirm_title", "ui.panel.config.entities.picker.disable_selected.confirm_title",
{ number: this._selected.length } { number: this._selectedEntities.length }
), ),
text: this.hass.localize( text: this.hass.localize(
"ui.panel.config.entities.picker.disable_selected.confirm_text" "ui.panel.config.entities.picker.disable_selected.confirm_text"
@@ -1000,7 +899,7 @@ ${
confirmText: this.hass.localize("ui.common.disable"), confirmText: this.hass.localize("ui.common.disable"),
dismissText: this.hass.localize("ui.common.cancel"), dismissText: this.hass.localize("ui.common.cancel"),
confirm: () => { confirm: () => {
this._selected.forEach((entity) => this._selectedEntities.forEach((entity) =>
updateEntityRegistryEntry(this.hass, entity, { updateEntityRegistryEntry(this.hass, entity, {
disabled_by: "user", disabled_by: "user",
}) })
@@ -1014,7 +913,7 @@ ${
showConfirmationDialog(this, { showConfirmationDialog(this, {
title: this.hass.localize( title: this.hass.localize(
"ui.panel.config.entities.picker.hide_selected.confirm_title", "ui.panel.config.entities.picker.hide_selected.confirm_title",
{ number: this._selected.length } { number: this._selectedEntities.length }
), ),
text: this.hass.localize( text: this.hass.localize(
"ui.panel.config.entities.picker.hide_selected.confirm_text" "ui.panel.config.entities.picker.hide_selected.confirm_text"
@@ -1022,7 +921,7 @@ ${
confirmText: this.hass.localize("ui.common.hide"), confirmText: this.hass.localize("ui.common.hide"),
dismissText: this.hass.localize("ui.common.cancel"), dismissText: this.hass.localize("ui.common.cancel"),
confirm: () => { confirm: () => {
this._selected.forEach((entity) => this._selectedEntities.forEach((entity) =>
updateEntityRegistryEntry(this.hass, entity, { updateEntityRegistryEntry(this.hass, entity, {
hidden_by: "user", hidden_by: "user",
}) })
@@ -1032,66 +931,22 @@ ${
}); });
} }
private _unhideSelected() {
this._selected.forEach((entity) =>
updateEntityRegistryEntry(this.hass, entity, {
hidden_by: null,
})
);
this._clearSelection();
}
private async _handleBulkLabel(ev) {
const label = ev.currentTarget.value;
const action = ev.currentTarget.action;
await this._bulkLabel(label, action);
}
private async _bulkLabel(label: string, action: "add" | "remove") {
const promises: Promise<UpdateEntityRegistryEntryResult>[] = [];
this._selected.forEach((entityId) => {
const entityReg =
this.hass.entities[entityId] ||
this._entities.find((entReg) => entReg.entity_id === entityId);
if (!entityReg) {
return;
}
promises.push(
updateEntityRegistryEntry(this.hass, entityId, {
labels:
action === "add"
? entityReg.labels.concat(label)
: entityReg.labels.filter((lbl) => lbl !== label),
})
);
});
await Promise.all(promises);
}
private _bulkCreateLabel() {
showLabelDetailDialog(this, {
createEntry: async (values) => {
const label = await createLabelRegistryEntry(this.hass, values);
this._bulkLabel(label.label_id, "add");
return label;
},
});
}
private _removeSelected() { private _removeSelected() {
const removeableEntities = this._selected.filter((entity) => { const removeableEntities = this._selectedEntities.filter((entity) => {
const stateObj = this.hass.states[entity]; const stateObj = this.hass.states[entity];
return stateObj?.attributes.restored; return stateObj?.attributes.restored;
}); });
showConfirmationDialog(this, { showConfirmationDialog(this, {
title: this.hass.localize( title: this.hass.localize(
`ui.panel.config.entities.picker.remove_selected.confirm_${ `ui.panel.config.entities.picker.remove_selected.confirm_${
removeableEntities.length !== this._selected.length ? "partly_" : "" removeableEntities.length !== this._selectedEntities.length
? "partly_"
: ""
}title`, }title`,
{ number: removeableEntities.length } { number: removeableEntities.length }
), ),
text: text:
removeableEntities.length === this._selected.length removeableEntities.length === this._selectedEntities.length
? this.hass.localize( ? this.hass.localize(
"ui.panel.config.entities.picker.remove_selected.confirm_text" "ui.panel.config.entities.picker.remove_selected.confirm_text"
) )
@@ -1099,7 +954,7 @@ ${
"ui.panel.config.entities.picker.remove_selected.confirm_partly_text", "ui.panel.config.entities.picker.remove_selected.confirm_partly_text",
{ {
removable: removeableEntities.length, removable: removeableEntities.length,
selected: this._selected.length, selected: this._selectedEntities.length,
} }
), ),
confirmText: this.hass.localize("ui.common.remove"), confirmText: this.hass.localize("ui.common.remove"),
@@ -1225,17 +1080,6 @@ ${
text-transform: uppercase; text-transform: uppercase;
direction: var(--direction); direction: var(--direction);
} }
ha-assist-chip {
--ha-assist-chip-container-shape: 10px;
}
ha-button-menu-new ha-assist-chip {
--md-assist-chip-trailing-space: 8px;
}
ha-label {
--ha-label-background-color: var(--color, var(--grey-color));
--ha-label-background-opacity: 0.5;
}
`, `,
]; ];
} }

View File

@@ -1,16 +1,5 @@
import { consume } from "@lit-labs/context";
import { ResizeController } from "@lit-labs/observers/resize-controller";
import "@lrnwebcomponents/simple-tooltip/simple-tooltip"; import "@lrnwebcomponents/simple-tooltip/simple-tooltip";
import { import { mdiAlertCircle, mdiPencilOff, mdiPlus } from "@mdi/js";
mdiAlertCircle,
mdiChevronRight,
mdiCog,
mdiDotsVertical,
mdiMenuDown,
mdiPencilOff,
mdiPlus,
mdiTag,
} from "@mdi/js";
import { HassEntity } from "home-assistant-js-websocket"; import { HassEntity } from "home-assistant-js-websocket";
import { import {
CSSResultGroup, CSSResultGroup,
@@ -22,9 +11,8 @@ import {
nothing, nothing,
} from "lit"; } from "lit";
import { customElement, property, state } from "lit/decorators"; import { customElement, property, state } from "lit/decorators";
import { consume } from "@lit-labs/context";
import memoizeOne from "memoize-one"; import memoizeOne from "memoize-one";
import { computeCssColor } from "../../../common/color/compute-color";
import { HASSDomEvent } from "../../../common/dom/fire_event";
import { computeStateDomain } from "../../../common/entity/compute_state_domain"; import { computeStateDomain } from "../../../common/entity/compute_state_domain";
import { navigate } from "../../../common/navigate"; import { navigate } from "../../../common/navigate";
import { import {
@@ -35,42 +23,22 @@ import { extractSearchParam } from "../../../common/url/search-params";
import { import {
DataTableColumnContainer, DataTableColumnContainer,
RowClickedEvent, RowClickedEvent,
SelectionChangedEvent,
} from "../../../components/data-table/ha-data-table"; } from "../../../components/data-table/ha-data-table";
import "../../../components/data-table/ha-data-table-labels"; import "../../../components/data-table/ha-data-table-labels";
import "../../../components/ha-fab"; import "../../../components/ha-fab";
import "../../../components/ha-filter-categories";
import "../../../components/ha-filter-devices";
import "../../../components/ha-filter-entities";
import "../../../components/ha-filter-floor-areas";
import "../../../components/ha-filter-labels";
import "../../../components/ha-icon"; import "../../../components/ha-icon";
import "../../../components/ha-icon-overflow-menu";
import "../../../components/ha-state-icon"; import "../../../components/ha-state-icon";
import "../../../components/ha-svg-icon"; import "../../../components/ha-svg-icon";
import {
CategoryRegistryEntry,
createCategoryRegistryEntry,
subscribeCategoryRegistry,
} from "../../../data/category_registry";
import { import {
ConfigEntry, ConfigEntry,
subscribeConfigEntries, subscribeConfigEntries,
} from "../../../data/config_entries"; } from "../../../data/config_entries";
import { getConfigFlowHandlers } from "../../../data/config_flow"; import { getConfigFlowHandlers } from "../../../data/config_flow";
import { fullEntitiesContext } from "../../../data/context";
import { import {
EntityRegistryEntry, EntityRegistryEntry,
UpdateEntityRegistryEntryResult,
subscribeEntityRegistry, subscribeEntityRegistry,
updateEntityRegistryEntry,
} from "../../../data/entity_registry"; } from "../../../data/entity_registry";
import { domainToName } from "../../../data/integration"; import { domainToName } from "../../../data/integration";
import {
LabelRegistryEntry,
createLabelRegistryEntry,
subscribeLabelRegistry,
} from "../../../data/label_registry";
import { showConfigFlowDialog } from "../../../dialogs/config-flow/show-dialog-config-flow"; import { showConfigFlowDialog } from "../../../dialogs/config-flow/show-dialog-config-flow";
import { showOptionsFlowDialog } from "../../../dialogs/config-flow/show-dialog-options-flow"; import { showOptionsFlowDialog } from "../../../dialogs/config-flow/show-dialog-options-flow";
import { import {
@@ -81,15 +49,18 @@ import { showMoreInfoDialog } from "../../../dialogs/more-info/show-ha-more-info
import "../../../layouts/hass-loading-screen"; import "../../../layouts/hass-loading-screen";
import "../../../layouts/hass-tabs-subpage-data-table"; import "../../../layouts/hass-tabs-subpage-data-table";
import { SubscribeMixin } from "../../../mixins/subscribe-mixin"; import { SubscribeMixin } from "../../../mixins/subscribe-mixin";
import { haStyle } from "../../../resources/styles";
import { HomeAssistant, Route } from "../../../types"; import { HomeAssistant, Route } from "../../../types";
import { showAssignCategoryDialog } from "../category/show-dialog-assign-category";
import { showCategoryRegistryDetailDialog } from "../category/show-dialog-category-registry-detail";
import { configSections } from "../ha-panel-config"; import { configSections } from "../ha-panel-config";
import "../integrations/ha-integration-overflow-menu"; import "../integrations/ha-integration-overflow-menu";
import { showLabelDetailDialog } from "../labels/show-dialog-label-detail";
import { isHelperDomain } from "./const"; import { isHelperDomain } from "./const";
import { showHelperDetailDialog } from "./show-dialog-helper-detail"; import { showHelperDetailDialog } from "./show-dialog-helper-detail";
import {
LabelRegistryEntry,
subscribeLabelRegistry,
} from "../../../data/label_registry";
import { fullEntitiesContext } from "../../../data/context";
import "../../../components/ha-filter-labels";
import { haStyle } from "../../../resources/styles";
type HelperItem = { type HelperItem = {
id: string; id: string;
@@ -100,7 +71,6 @@ type HelperItem = {
type: string; type: string;
configEntry?: ConfigEntry; configEntry?: ConfigEntry;
entity?: HassEntity; entity?: HassEntity;
category: string | undefined;
label_entries: LabelRegistryEntry[]; label_entries: LabelRegistryEntry[];
}; };
@@ -141,8 +111,6 @@ export class HaConfigHelpers extends SubscribeMixin(LitElement) {
@state() private _configEntries?: Record<string, ConfigEntry>; @state() private _configEntries?: Record<string, ConfigEntry>;
@state() private _selected: string[] = [];
@state() private _activeFilters?: string[]; @state() private _activeFilters?: string[];
@state() private _filters: Record< @state() private _filters: Record<
@@ -152,9 +120,6 @@ export class HaConfigHelpers extends SubscribeMixin(LitElement) {
@state() private _expandedFilter?: string; @state() private _expandedFilter?: string;
@state()
_categories!: CategoryRegistryEntry[];
@state() @state()
_labels!: LabelRegistryEntry[]; _labels!: LabelRegistryEntry[];
@@ -164,10 +129,6 @@ export class HaConfigHelpers extends SubscribeMixin(LitElement) {
@state() private _filteredStateItems?: string[] | null; @state() private _filteredStateItems?: string[] | null;
private _sizeController = new ResizeController(this, {
callback: (entries) => entries[0]?.contentRect.width,
});
public hassSubscribe() { public hassSubscribe() {
return [ return [
subscribeConfigEntries( subscribeConfigEntries(
@@ -195,86 +156,65 @@ export class HaConfigHelpers extends SubscribeMixin(LitElement) {
subscribeLabelRegistry(this.hass.connection, (labels) => { subscribeLabelRegistry(this.hass.connection, (labels) => {
this._labels = labels; this._labels = labels;
}), }),
subscribeCategoryRegistry(
this.hass.connection,
"helpers",
(categories) => {
this._categories = categories;
}
),
]; ];
} }
private _columns = memoizeOne( private _columns = memoizeOne(
( (narrow: boolean, localize: LocalizeFunc): DataTableColumnContainer => {
narrow: boolean, const columns: DataTableColumnContainer<HelperItem> = {
localize: LocalizeFunc icon: {
): DataTableColumnContainer<HelperItem> => ({ title: "",
icon: { label: localize("ui.panel.config.helpers.picker.headers.icon"),
title: "", type: "icon",
label: localize("ui.panel.config.helpers.picker.headers.icon"), template: (helper) =>
type: "icon", helper.entity
template: (helper) => ? html`<ha-state-icon
helper.entity .hass=${this.hass}
? html`<ha-state-icon .stateObj=${helper.entity}
.hass=${this.hass} ></ha-state-icon>`
.stateObj=${helper.entity} : html`<ha-svg-icon
></ha-state-icon>` .path=${helper.icon}
: html`<ha-svg-icon style="color: var(--error-color)"
.path=${helper.icon} ></ha-svg-icon>`,
style="color: var(--error-color)" },
></ha-svg-icon>`, name: {
}, title: localize("ui.panel.config.helpers.picker.headers.name"),
name: { main: true,
title: localize("ui.panel.config.helpers.picker.headers.name"), sortable: true,
main: true, filterable: true,
sortable: true, grows: true,
filterable: true, direction: "asc",
grows: true, template: (helper) => html`
direction: "asc", <div style="font-size: 14px;">${helper.name}</div>
template: (helper) => html` ${narrow
<div style="font-size: 14px;">${helper.name}</div> ? html`<div class="secondary">${helper.entity_id}</div> `
${narrow : nothing}
? html`<div class="secondary">${helper.entity_id}</div> ` ${helper.label_entries.length
: nothing} ? html`
${helper.label_entries.length <ha-data-table-labels
? html` .labels=${helper.label_entries}
<ha-data-table-labels ></ha-data-table-labels>
.labels=${helper.label_entries} `
></ha-data-table-labels> : nothing}
` `,
: nothing} },
`, };
}, if (!narrow) {
entity_id: { columns.entity_id = {
title: localize("ui.panel.config.helpers.picker.headers.entity_id"), title: localize("ui.panel.config.helpers.picker.headers.entity_id"),
hidden: this.narrow, sortable: true,
sortable: true, filterable: true,
filterable: true, width: "25%",
width: "25%", };
}, }
category: { columns.localized_type = {
title: localize("ui.panel.config.helpers.picker.headers.category"),
hidden: true,
groupable: true,
filterable: true,
sortable: true,
},
labels: {
title: "",
hidden: true,
filterable: true,
template: (helper) =>
helper.label_entries.map((lbl) => lbl.name).join(" "),
},
localized_type: {
title: localize("ui.panel.config.helpers.picker.headers.type"), title: localize("ui.panel.config.helpers.picker.headers.type"),
sortable: true, sortable: true,
width: "25%", width: "25%",
filterable: true, filterable: true,
groupable: true, groupable: true,
}, };
editable: { columns.editable = {
title: "", title: "",
label: this.hass.localize( label: this.hass.localize(
"ui.panel.config.helpers.picker.headers.editable" "ui.panel.config.helpers.picker.headers.editable"
@@ -297,36 +237,9 @@ export class HaConfigHelpers extends SubscribeMixin(LitElement) {
` `
: ""} : ""}
`, `,
}, };
actions: { return columns;
title: "", }
width: "64px",
type: "overflow-menu",
template: (helper) => html`
<ha-icon-overflow-menu
.hass=${this.hass}
narrow
.items=${[
{
path: mdiCog,
label: this.hass.localize(
"ui.panel.config.automation.picker.show_settings"
),
action: () => this._openSettings(helper),
},
{
path: mdiTag,
label: this.hass.localize(
`ui.panel.config.automation.picker.${helper.category ? "edit_category" : "assign_category"}`
),
action: () => this._editCategory(helper),
},
]}
>
</ha-icon-overflow-menu>
`,
},
})
); );
private _getItems = memoizeOne( private _getItems = memoizeOne(
@@ -336,7 +249,6 @@ export class HaConfigHelpers extends SubscribeMixin(LitElement) {
entityEntries: Record<string, EntityRegistryEntry>, entityEntries: Record<string, EntityRegistryEntry>,
configEntries: Record<string, ConfigEntry>, configEntries: Record<string, ConfigEntry>,
entityReg: EntityRegistryEntry[], entityReg: EntityRegistryEntry[],
categoryReg?: CategoryRegistryEntry[],
labelReg?: LabelRegistryEntry[], labelReg?: LabelRegistryEntry[],
filteredStateItems?: string[] | null filteredStateItems?: string[] | null
): HelperItem[] => { ): HelperItem[] => {
@@ -380,7 +292,6 @@ export class HaConfigHelpers extends SubscribeMixin(LitElement) {
type: configEntry.domain, type: configEntry.domain,
configEntry, configEntry,
entity: undefined, entity: undefined,
selectable: false,
})); }));
return [...states, ...entries] return [...states, ...entries]
@@ -394,7 +305,6 @@ export class HaConfigHelpers extends SubscribeMixin(LitElement) {
(reg) => reg.entity_id === item.entity_id (reg) => reg.entity_id === item.entity_id
); );
const labels = labelReg && entityRegEntry?.labels; const labels = labelReg && entityRegEntry?.labels;
const category = entityRegEntry?.categories.helpers;
return { return {
...item, ...item,
localized_type: item.configEntry localized_type: item.configEntry
@@ -405,9 +315,6 @@ export class HaConfigHelpers extends SubscribeMixin(LitElement) {
label_entries: (labels || []).map( label_entries: (labels || []).map(
(lbl) => labelReg!.find((label) => label.label_id === lbl)! (lbl) => labelReg!.find((label) => label.label_id === lbl)!
), ),
category: category
? categoryReg?.find((cat) => cat.category_id === category)?.name
: undefined,
}; };
}); });
} }
@@ -423,69 +330,6 @@ export class HaConfigHelpers extends SubscribeMixin(LitElement) {
return html` <hass-loading-screen></hass-loading-screen> `; return html` <hass-loading-screen></hass-loading-screen> `;
} }
const categoryItems = html`${this._categories?.map(
(category) =>
html`<ha-menu-item
.value=${category.category_id}
@click=${this._handleBulkCategory}
>
${category.icon
? html`<ha-icon slot="start" .icon=${category.icon}></ha-icon>`
: html`<ha-svg-icon slot="start" .path=${mdiTag}></ha-svg-icon>`}
<div slot="headline">${category.name}</div>
</ha-menu-item>`
)}
<ha-menu-item .value=${null} @click=${this._handleBulkCategory}>
<div slot="headline">
${this.hass.localize(
"ui.panel.config.automation.picker.bulk_actions.no_category"
)}
</div>
</ha-menu-item>
<md-divider role="separator" tabindex="-1"></md-divider>
<ha-menu-item @click=${this._bulkCreateCategory}>
<div slot="headline">
${this.hass.localize("ui.panel.config.category.editor.add")}
</div>
</ha-menu-item>`;
const labelItems = html`${this._labels?.map((label) => {
const color = label.color ? computeCssColor(label.color) : undefined;
const selected = this._selected.every((entityId) =>
this.hass.entities[entityId]?.labels.includes(label.label_id)
);
const partial =
!selected &&
this._selected.some((entityId) =>
this.hass.entities[entityId]?.labels.includes(label.label_id)
);
return html`<ha-menu-item
.value=${label.label_id}
.action=${selected ? "remove" : "add"}
@click=${this._handleBulkLabel}
keep-open
>
<ha-checkbox
slot="start"
.checked=${selected}
.indeterminate=${partial}
reducedTouchTarget
></ha-checkbox>
<ha-label style=${color ? `--color: ${color}` : ""}>
${label.icon
? html`<ha-icon slot="icon" .icon=${label.icon}></ha-icon>`
: nothing}
${label.name}
</ha-label>
</ha-menu-item> `;
})}<md-divider role="separator" tabindex="-1"></md-divider>
<ha-menu-item @click=${this._bulkCreateLabel}>
<div slot="headline">
${this.hass.localize("ui.panel.config.labels.add_label")}
</div>
</ha-menu-item>`;
const labelsInOverflow =
(this._sizeController.value && this._sizeController.value < 700) ||
(!this._sizeController.value && this.hass.dockedSidebar === "docked");
return html` return html`
<hass-tabs-subpage-data-table <hass-tabs-subpage-data-table
.hass=${this.hass} .hass=${this.hass}
@@ -493,9 +337,6 @@ export class HaConfigHelpers extends SubscribeMixin(LitElement) {
back-path="/config" back-path="/config"
.route=${this.route} .route=${this.route}
.tabs=${configSections.devices} .tabs=${configSections.devices}
selectable
.selected=${this._selected.length}
@selection-changed=${this._handleSelectionChanged}
hasFilters hasFilters
.filters=${Object.values(this._filters).filter( .filters=${Object.values(this._filters).filter(
(filter) => filter.value?.length (filter) => filter.value?.length
@@ -507,11 +348,9 @@ export class HaConfigHelpers extends SubscribeMixin(LitElement) {
this._entityEntries, this._entityEntries,
this._configEntries, this._configEntries,
this._entityReg, this._entityReg,
this._categories,
this._labels, this._labels,
this._filteredStateItems this._filteredStateItems
)} )}
initialGroupColumn="category"
.activeFilters=${this._activeFilters} .activeFilters=${this._activeFilters}
@clear-filter=${this._clearFilter} @clear-filter=${this._clearFilter}
@row-click=${this._openEditDialog} @row-click=${this._openEditDialog}
@@ -522,26 +361,6 @@ export class HaConfigHelpers extends SubscribeMixin(LitElement) {
)} )}
class=${this.narrow ? "narrow" : ""} class=${this.narrow ? "narrow" : ""}
> >
<ha-filter-floor-areas
.hass=${this.hass}
.type=${"entity"}
.value=${this._filters["ha-filter-floor-areas"]?.value}
@data-table-filter-changed=${this._filterChanged}
slot="filter-pane"
.expanded=${this._expandedFilter === "ha-filter-floor-areas"}
.narrow=${this.narrow}
@expanded-changed=${this._filterExpanded}
></ha-filter-floor-areas>
<ha-filter-devices
.hass=${this.hass}
.type=${"entity"}
.value=${this._filters["ha-filter-devices"]?.value}
@data-table-filter-changed=${this._filterChanged}
slot="filter-pane"
.expanded=${this._expandedFilter === "ha-filter-devices"}
.narrow=${this.narrow}
@expanded-changed=${this._filterExpanded}
></ha-filter-devices>
<ha-filter-labels <ha-filter-labels
.hass=${this.hass} .hass=${this.hass}
.value=${this._filters["ha-filter-labels"]?.value} .value=${this._filters["ha-filter-labels"]?.value}
@@ -551,114 +370,6 @@ export class HaConfigHelpers extends SubscribeMixin(LitElement) {
.narrow=${this.narrow} .narrow=${this.narrow}
@expanded-changed=${this._filterExpanded} @expanded-changed=${this._filterExpanded}
></ha-filter-labels> ></ha-filter-labels>
<ha-filter-categories
.hass=${this.hass}
scope="helpers"
.value=${this._filters["ha-filter-categories"]?.value}
@data-table-filter-changed=${this._filterChanged}
slot="filter-pane"
.expanded=${this._expandedFilter === "ha-filter-categories"}
.narrow=${this.narrow}
@expanded-changed=${this._filterExpanded}
></ha-filter-categories>
${!this.narrow
? html`<ha-button-menu-new slot="selection-bar">
<ha-assist-chip
slot="trigger"
.label=${this.hass.localize(
"ui.panel.config.automation.picker.bulk_actions.move_category"
)}
>
<ha-svg-icon
slot="trailing-icon"
.path=${mdiMenuDown}
></ha-svg-icon>
</ha-assist-chip>
${categoryItems}
</ha-button-menu-new>
${labelsInOverflow
? nothing
: html`<ha-button-menu-new slot="selection-bar">
<ha-assist-chip
slot="trigger"
.label=${this.hass.localize(
"ui.panel.config.automation.picker.bulk_actions.add_label"
)}
>
<ha-svg-icon
slot="trailing-icon"
.path=${mdiMenuDown}
></ha-svg-icon>
</ha-assist-chip>
${labelItems}
</ha-button-menu-new>`}`
: nothing}
${this.narrow || labelsInOverflow
? html`
<ha-button-menu-new has-overflow slot="selection-bar">
${
this.narrow
? html`<ha-assist-chip
.label=${this.hass.localize(
"ui.panel.config.automation.picker.bulk_action"
)}
slot="trigger"
>
<ha-svg-icon
slot="trailing-icon"
.path=${mdiMenuDown}
></ha-svg-icon>
</ha-assist-chip>`
: html`<ha-icon-button
.path=${mdiDotsVertical}
.label=${"ui.panel.config.automation.picker.bulk_action"}
slot="trigger"
></ha-icon-button>`
}
<ha-svg-icon
slot="trailing-icon"
.path=${mdiMenuDown}
></ha-svg-icon
></ha-assist-chip>
${
this.narrow
? html`<ha-sub-menu>
<ha-menu-item slot="item">
<div slot="headline">
${this.hass.localize(
"ui.panel.config.automation.picker.bulk_actions.move_category"
)}
</div>
<ha-svg-icon
slot="end"
.path=${mdiChevronRight}
></ha-svg-icon>
</ha-menu-item>
<ha-menu slot="menu">${categoryItems}</ha-menu>
</ha-sub-menu>`
: nothing
}
${
this.narrow || this.hass.dockedSidebar === "docked"
? html` <ha-sub-menu>
<ha-menu-item slot="item">
<div slot="headline">
${this.hass.localize(
"ui.panel.config.automation.picker.bulk_actions.add_label"
)}
</div>
<ha-svg-icon
slot="end"
.path=${mdiChevronRight}
></ha-svg-icon>
</ha-menu-item>
<ha-menu slot="menu">${labelItems}</ha-menu>
</ha-sub-menu>`
: nothing
}
</ha-button-menu-new>`
: nothing}
<ha-integration-overflow-menu <ha-integration-overflow-menu
.hass=${this.hass} .hass=${this.hass}
@@ -726,27 +437,6 @@ export class HaConfigHelpers extends SubscribeMixin(LitElement) {
items.intersection(labelItems) items.intersection(labelItems)
: new Set([...items].filter((x) => labelItems!.has(x))); : new Set([...items].filter((x) => labelItems!.has(x)));
} }
if (key === "ha-filter-categories" && filter.value?.length) {
const categoryItems: Set<string> = new Set();
this._stateItems
.filter(
(stateItem) =>
filter.value![0] ===
this._entityReg.find(
(reg) => reg.entity_id === stateItem.entity_id
)?.categories.helpers
)
.forEach((stateItem) => categoryItems.add(stateItem.entity_id));
if (!items) {
items = categoryItems;
continue;
}
items =
"intersection" in items
? // @ts-ignore
items.intersection(categoryItems)
: new Set([...items].filter((x) => categoryItems!.has(x)));
}
} }
this._filteredStateItems = items ? [...items] : undefined; this._filteredStateItems = items ? [...items] : undefined;
} }
@@ -756,73 +446,6 @@ export class HaConfigHelpers extends SubscribeMixin(LitElement) {
this._applyFilters(); this._applyFilters();
} }
private _editCategory(helper: any) {
const entityReg = this._entityReg.find(
(reg) => reg.entity_id === helper.entity_id
);
if (!entityReg) {
showAlertDialog(this, {
title: this.hass.localize(
"ui.panel.config.automation.picker.no_category_support"
),
text: this.hass.localize(
"ui.panel.config.automation.picker.no_category_entity_reg"
),
});
return;
}
showAssignCategoryDialog(this, {
scope: "helpers",
entityReg,
});
}
private async _handleBulkCategory(ev) {
const category = ev.currentTarget.value;
this._bulkAddCategory(category);
}
private async _bulkAddCategory(category: string) {
const promises: Promise<UpdateEntityRegistryEntryResult>[] = [];
this._selected.forEach((entityId) => {
promises.push(
updateEntityRegistryEntry(this.hass, entityId, {
categories: { helpers: category },
})
);
});
await Promise.all(promises);
}
private async _handleBulkLabel(ev) {
const label = ev.currentTarget.value;
const action = ev.currentTarget.action;
this._bulkLabel(label, action);
}
private async _bulkLabel(label: string, action: "add" | "remove") {
const promises: Promise<UpdateEntityRegistryEntryResult>[] = [];
this._selected.forEach((entityId) => {
promises.push(
updateEntityRegistryEntry(this.hass, entityId, {
labels:
action === "add"
? this.hass.entities[entityId].labels.concat(label)
: this.hass.entities[entityId].labels.filter(
(lbl) => lbl !== label
),
})
);
});
await Promise.all(promises);
}
private _handleSelectionChanged(
ev: HASSDomEvent<SelectionChangedEvent>
): void {
this._selected = ev.detail.value;
}
protected firstUpdated(changedProps: PropertyValues) { protected firstUpdated(changedProps: PropertyValues) {
super.firstUpdated(changedProps); super.firstUpdated(changedProps);
if (this.route.path === "/add") { if (this.route.path === "/add") {
@@ -940,69 +563,20 @@ export class HaConfigHelpers extends SubscribeMixin(LitElement) {
} }
} }
private _openSettings(helper: HelperItem) {
if (helper.entity) {
showMoreInfoDialog(this, {
entityId: helper.entity_id,
view: "settings",
});
} else {
showOptionsFlowDialog(this, helper.configEntry!);
}
}
private _createHelper() { private _createHelper() {
showHelperDetailDialog(this, {}); showHelperDetailDialog(this, {});
} }
private async _bulkCreateCategory() {
showCategoryRegistryDetailDialog(this, {
scope: "helpers",
createEntry: async (values) => {
const category = await createCategoryRegistryEntry(
this.hass,
"helpers",
values
);
this._bulkAddCategory(category.category_id);
return category;
},
});
}
private _bulkCreateLabel() {
showLabelDetailDialog(this, {
createEntry: async (values) => {
const label = await createLabelRegistryEntry(this.hass, values);
this._bulkLabel(label.label_id, "add");
return label;
},
});
}
static get styles(): CSSResultGroup { static get styles(): CSSResultGroup {
return [ return [
haStyle, haStyle,
css` css`
:host {
display: block;
}
hass-tabs-subpage-data-table { hass-tabs-subpage-data-table {
--data-table-row-height: 60px; --data-table-row-height: 60px;
} }
hass-tabs-subpage-data-table.narrow { hass-tabs-subpage-data-table.narrow {
--data-table-row-height: 72px; --data-table-row-height: 72px;
} }
ha-assist-chip {
--ha-assist-chip-container-shape: 10px;
}
ha-button-menu-new ha-assist-chip {
--md-assist-chip-trailing-space: 8px;
}
ha-label {
--ha-label-background-color: var(--color, var(--grey-color));
--ha-label-background-opacity: 0.5;
}
`, `,
]; ];
} }

View File

@@ -1,15 +1,10 @@
import { consume } from "@lit-labs/context"; import { consume } from "@lit-labs/context";
import { ResizeController } from "@lit-labs/observers/resize-controller";
import "@lrnwebcomponents/simple-tooltip/simple-tooltip"; import "@lrnwebcomponents/simple-tooltip/simple-tooltip";
import { import {
mdiChevronRight,
mdiCog,
mdiContentDuplicate, mdiContentDuplicate,
mdiDelete, mdiDelete,
mdiDotsVertical,
mdiHelpCircle, mdiHelpCircle,
mdiInformationOutline, mdiInformationOutline,
mdiMenuDown,
mdiPalette, mdiPalette,
mdiPencilOff, mdiPencilOff,
mdiPlay, mdiPlay,
@@ -29,7 +24,6 @@ import {
} from "lit"; } from "lit";
import { customElement, property, state } from "lit/decorators"; import { customElement, property, state } from "lit/decorators";
import memoizeOne from "memoize-one"; import memoizeOne from "memoize-one";
import { computeCssColor } from "../../../common/color/compute-color";
import { formatShortDateTime } from "../../../common/datetime/format_date_time"; import { formatShortDateTime } from "../../../common/datetime/format_date_time";
import { relativeTime } from "../../../common/datetime/relative_time"; import { relativeTime } from "../../../common/datetime/relative_time";
import { HASSDomEvent, fireEvent } from "../../../common/dom/fire_event"; import { HASSDomEvent, fireEvent } from "../../../common/dom/fire_event";
@@ -39,7 +33,6 @@ import { LocalizeFunc } from "../../../common/translations/localize";
import { import {
DataTableColumnContainer, DataTableColumnContainer,
RowClickedEvent, RowClickedEvent,
SelectionChangedEvent,
} from "../../../components/data-table/ha-data-table"; } from "../../../components/data-table/ha-data-table";
import "../../../components/data-table/ha-data-table-labels"; import "../../../components/data-table/ha-data-table-labels";
import "../../../components/ha-button"; import "../../../components/ha-button";
@@ -51,26 +44,18 @@ import "../../../components/ha-filter-floor-areas";
import "../../../components/ha-filter-labels"; import "../../../components/ha-filter-labels";
import "../../../components/ha-icon-button"; import "../../../components/ha-icon-button";
import "../../../components/ha-icon-overflow-menu"; import "../../../components/ha-icon-overflow-menu";
import "../../../components/ha-menu-item";
import "../../../components/ha-state-icon"; import "../../../components/ha-state-icon";
import "../../../components/ha-sub-menu";
import "../../../components/ha-svg-icon"; import "../../../components/ha-svg-icon";
import { import {
CategoryRegistryEntry, CategoryRegistryEntry,
createCategoryRegistryEntry,
subscribeCategoryRegistry, subscribeCategoryRegistry,
} from "../../../data/category_registry"; } from "../../../data/category_registry";
import { fullEntitiesContext } from "../../../data/context"; import { fullEntitiesContext } from "../../../data/context";
import { isUnavailableState } from "../../../data/entity"; import { isUnavailableState } from "../../../data/entity";
import { import { EntityRegistryEntry } from "../../../data/entity_registry";
EntityRegistryEntry,
UpdateEntityRegistryEntryResult,
updateEntityRegistryEntry,
} from "../../../data/entity_registry";
import { forwardHaptic } from "../../../data/haptics"; import { forwardHaptic } from "../../../data/haptics";
import { import {
LabelRegistryEntry, LabelRegistryEntry,
createLabelRegistryEntry,
subscribeLabelRegistry, subscribeLabelRegistry,
} from "../../../data/label_registry"; } from "../../../data/label_registry";
import { import {
@@ -84,7 +69,6 @@ import {
showAlertDialog, showAlertDialog,
showConfirmationDialog, showConfirmationDialog,
} from "../../../dialogs/generic/show-dialog-box"; } from "../../../dialogs/generic/show-dialog-box";
import { showMoreInfoDialog } from "../../../dialogs/more-info/show-ha-more-info-dialog";
import "../../../layouts/hass-tabs-subpage-data-table"; import "../../../layouts/hass-tabs-subpage-data-table";
import { SubscribeMixin } from "../../../mixins/subscribe-mixin"; import { SubscribeMixin } from "../../../mixins/subscribe-mixin";
import { haStyle } from "../../../resources/styles"; import { haStyle } from "../../../resources/styles";
@@ -92,13 +76,10 @@ import { HomeAssistant, Route } from "../../../types";
import { documentationUrl } from "../../../util/documentation-url"; import { documentationUrl } from "../../../util/documentation-url";
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 { showCategoryRegistryDetailDialog } from "../category/show-dialog-category-registry-detail";
import { configSections } from "../ha-panel-config"; import { configSections } from "../ha-panel-config";
import { showLabelDetailDialog } from "../labels/show-dialog-label-detail";
type SceneItem = SceneEntity & { type SceneItem = SceneEntity & {
name: string; name: string;
area: string | undefined;
category: string | undefined; category: string | undefined;
labels: LabelRegistryEntry[]; labels: LabelRegistryEntry[];
}; };
@@ -117,8 +98,6 @@ class HaSceneDashboard extends SubscribeMixin(LitElement) {
@state() private _searchParms = new URLSearchParams(window.location.search); @state() private _searchParms = new URLSearchParams(window.location.search);
@state() private _selected: string[] = [];
@state() private _activeFilters?: string[]; @state() private _activeFilters?: string[];
@state() private _filteredScenes?: string[] | null; @state() private _filteredScenes?: string[] | null;
@@ -140,15 +119,10 @@ class HaSceneDashboard extends SubscribeMixin(LitElement) {
@consume({ context: fullEntitiesContext, subscribe: true }) @consume({ context: fullEntitiesContext, subscribe: true })
_entityReg!: EntityRegistryEntry[]; _entityReg!: EntityRegistryEntry[];
private _sizeController = new ResizeController(this, {
callback: (entries) => entries[0]?.contentRect.width,
});
private _scenes = memoizeOne( private _scenes = memoizeOne(
( (
scenes: SceneEntity[], scenes: SceneEntity[],
entityReg: EntityRegistryEntry[], entityReg: EntityRegistryEntry[],
areas: HomeAssistant["areas"],
categoryReg?: CategoryRegistryEntry[], categoryReg?: CategoryRegistryEntry[],
labelReg?: LabelRegistryEntry[], labelReg?: LabelRegistryEntry[],
filteredScenes?: string[] | null filteredScenes?: string[] | null
@@ -169,9 +143,6 @@ class HaSceneDashboard extends SubscribeMixin(LitElement) {
return { return {
...scene, ...scene,
name: computeStateName(scene), name: computeStateName(scene),
area: entityRegEntry?.area_id
? areas[entityRegEntry?.area_id]?.name
: undefined,
category: category category: category
? categoryReg?.find((cat) => cat.category_id === category)?.name ? categoryReg?.find((cat) => cat.category_id === category)?.name
: undefined, : undefined,
@@ -214,13 +185,6 @@ class HaSceneDashboard extends SubscribeMixin(LitElement) {
: nothing} : nothing}
`, `,
}, },
area: {
title: localize("ui.panel.config.scene.picker.headers.area"),
hidden: true,
groupable: true,
filterable: true,
sortable: true,
},
category: { category: {
title: localize("ui.panel.config.scene.picker.headers.category"), title: localize("ui.panel.config.scene.picker.headers.category"),
hidden: true, hidden: true,
@@ -234,13 +198,14 @@ class HaSceneDashboard extends SubscribeMixin(LitElement) {
filterable: true, filterable: true,
template: (scene) => scene.labels.map((lbl) => lbl.name).join(" "), template: (scene) => scene.labels.map((lbl) => lbl.name).join(" "),
}, },
state: { };
if (!narrow) {
columns.state = {
title: localize( title: localize(
"ui.panel.config.scene.picker.headers.last_activated" "ui.panel.config.scene.picker.headers.last_activated"
), ),
sortable: true, sortable: true,
width: "30%", width: "30%",
hidden: narrow,
template: (scene) => { template: (scene) => {
const lastActivated = scene.state; const lastActivated = scene.state;
if (!lastActivated || isUnavailableState(lastActivated)) { if (!lastActivated || isUnavailableState(lastActivated)) {
@@ -255,87 +220,80 @@ class HaSceneDashboard extends SubscribeMixin(LitElement) {
: relativeTime(date, this.hass.locale)} : relativeTime(date, this.hass.locale)}
`; `;
}, },
}, };
only_editable: { }
title: "", columns.only_editable = {
width: "56px", title: "",
template: (scene) => width: "56px",
!scene.attributes.id template: (scene) =>
? html` !scene.attributes.id
<simple-tooltip animation-delay="0" position="left"> ? html`
${this.hass.localize( <simple-tooltip animation-delay="0" position="left">
"ui.panel.config.scene.picker.only_editable" ${this.hass.localize(
)} "ui.panel.config.scene.picker.only_editable"
</simple-tooltip> )}
<ha-svg-icon </simple-tooltip>
.path=${mdiPencilOff} <ha-svg-icon
style="color: var(--secondary-text-color)" .path=${mdiPencilOff}
></ha-svg-icon> style="color: var(--secondary-text-color)"
` ></ha-svg-icon>
: "", `
}, : "",
actions: { };
title: "", columns.actions = {
width: "64px", title: "",
type: "overflow-menu", width: "64px",
template: (scene) => html` type: "overflow-menu",
<ha-icon-overflow-menu template: (scene) => html`
.hass=${this.hass} <ha-icon-overflow-menu
narrow .hass=${this.hass}
.items=${[ narrow
{ .items=${[
path: mdiInformationOutline, {
label: this.hass.localize( path: mdiInformationOutline,
"ui.panel.config.scene.picker.show_info" label: this.hass.localize(
), "ui.panel.config.scene.picker.show_info"
action: () => this._showInfo(scene), ),
}, action: () => this._showInfo(scene),
{ },
path: mdiCog, {
label: this.hass.localize( path: mdiPlay,
"ui.panel.config.automation.picker.show_settings" label: this.hass.localize(
), "ui.panel.config.scene.picker.activate"
action: () => this._openSettings(scene), ),
}, action: () => this._activateScene(scene),
{ },
path: mdiPlay, {
label: this.hass.localize( path: mdiTag,
"ui.panel.config.scene.picker.activate" label: this.hass.localize(
), `ui.panel.config.scene.picker.${scene.category ? "edit_category" : "assign_category"}`
action: () => this._activateScene(scene), ),
}, action: () => this._editCategory(scene),
{ },
path: mdiTag, {
label: this.hass.localize( divider: true,
`ui.panel.config.scene.picker.${scene.category ? "edit_category" : "assign_category"}` },
), {
action: () => this._editCategory(scene), path: mdiContentDuplicate,
}, label: this.hass.localize(
{ "ui.panel.config.scene.picker.duplicate"
divider: true, ),
}, action: () => this._duplicate(scene),
{ disabled: !scene.attributes.id,
path: mdiContentDuplicate, },
label: this.hass.localize( {
"ui.panel.config.scene.picker.duplicate" label: this.hass.localize(
), "ui.panel.config.scene.picker.delete"
action: () => this._duplicate(scene), ),
disabled: !scene.attributes.id, path: mdiDelete,
}, action: () => this._deleteConfirm(scene),
{ warning: scene.attributes.id,
label: this.hass.localize( disabled: !scene.attributes.id,
"ui.panel.config.scene.picker.delete" },
), ]}
path: mdiDelete, >
action: () => this._deleteConfirm(scene), </ha-icon-overflow-menu>
warning: scene.attributes.id, `,
disabled: !scene.attributes.id,
},
]}
>
</ha-icon-overflow-menu>
`,
},
}; };
return columns; return columns;
@@ -361,70 +319,6 @@ class HaSceneDashboard extends SubscribeMixin(LitElement) {
} }
protected render(): TemplateResult { protected render(): TemplateResult {
const categoryItems = html`${this._categories?.map(
(category) =>
html`<ha-menu-item
.value=${category.category_id}
@click=${this._handleBulkCategory}
>
${category.icon
? html`<ha-icon slot="start" .icon=${category.icon}></ha-icon>`
: html`<ha-svg-icon slot="start" .path=${mdiTag}></ha-svg-icon>`}
<div slot="headline">${category.name}</div>
</ha-menu-item>`
)}
<ha-menu-item .value=${null} @click=${this._handleBulkCategory}>
<div slot="headline">
${this.hass.localize(
"ui.panel.config.automation.picker.bulk_actions.no_category"
)}
</div>
</ha-menu-item>
<md-divider role="separator" tabindex="-1"></md-divider>
<ha-menu-item @click=${this._bulkCreateCategory}>
<div slot="headline">
${this.hass.localize("ui.panel.config.category.editor.add")}
</div>
</ha-menu-item>`;
const labelItems = html` ${this._labels?.map((label) => {
const color = label.color ? computeCssColor(label.color) : undefined;
const selected = this._selected.every((entityId) =>
this.hass.entities[entityId]?.labels.includes(label.label_id)
);
const partial =
!selected &&
this._selected.some((entityId) =>
this.hass.entities[entityId]?.labels.includes(label.label_id)
);
return html`<ha-menu-item
.value=${label.label_id}
.action=${selected ? "remove" : "add"}
@click=${this._handleBulkLabel}
keep-open
>
<ha-checkbox
slot="start"
.checked=${selected}
.indeterminate=${partial}
reducedTouchTarget
></ha-checkbox>
<ha-label style=${color ? `--color: ${color}` : ""}>
${label.icon
? html`<ha-icon slot="icon" .icon=${label.icon}></ha-icon>`
: nothing}
${label.name}
</ha-label>
</ha-menu-item>`;
})}
<md-divider role="separator" tabindex="-1"></md-divider>
<ha-menu-item @click=${this._bulkCreateLabel}>
<div slot="headline">
${this.hass.localize("ui.panel.config.labels.add_label")}
</div></ha-menu-item
>`;
const labelsInOverflow =
(this._sizeController.value && this._sizeController.value < 700) ||
(!this._sizeController.value && this.hass.dockedSidebar === "docked");
return html` return html`
<hass-tabs-subpage-data-table <hass-tabs-subpage-data-table
.hass=${this.hass} .hass=${this.hass}
@@ -432,9 +326,6 @@ class HaSceneDashboard extends SubscribeMixin(LitElement) {
back-path="/config" back-path="/config"
.route=${this.route} .route=${this.route}
.tabs=${configSections.automations} .tabs=${configSections.automations}
selectable
.selected=${this._selected.length}
@selection-changed=${this._handleSelectionChanged}
hasFilters hasFilters
.filters=${Object.values(this._filters).filter( .filters=${Object.values(this._filters).filter(
(filter) => filter.value?.length (filter) => filter.value?.length
@@ -445,7 +336,6 @@ class HaSceneDashboard extends SubscribeMixin(LitElement) {
.data=${this._scenes( .data=${this._scenes(
this.scenes, this.scenes,
this._entityReg, this._entityReg,
this.hass.areas,
this._categories, this._categories,
this._labels, this._labels,
this._filteredScenes this._filteredScenes
@@ -517,103 +407,6 @@ class HaSceneDashboard extends SubscribeMixin(LitElement) {
@expanded-changed=${this._filterExpanded} @expanded-changed=${this._filterExpanded}
></ha-filter-categories> ></ha-filter-categories>
${!this.narrow
? html`<ha-button-menu-new slot="selection-bar">
<ha-assist-chip
slot="trigger"
.label=${this.hass.localize(
"ui.panel.config.automation.picker.bulk_actions.move_category"
)}
>
<ha-svg-icon
slot="trailing-icon"
.path=${mdiMenuDown}
></ha-svg-icon>
</ha-assist-chip>
${categoryItems}
</ha-button-menu-new>
${labelsInOverflow
? nothing
: html`<ha-button-menu-new slot="selection-bar">
<ha-assist-chip
slot="trigger"
.label=${this.hass.localize(
"ui.panel.config.automation.picker.bulk_actions.add_label"
)}
>
<ha-svg-icon
slot="trailing-icon"
.path=${mdiMenuDown}
></ha-svg-icon>
</ha-assist-chip>
${labelItems}
</ha-button-menu-new>`}`
: nothing}
${this.narrow || labelsInOverflow
? html`
<ha-button-menu-new has-overflow slot="selection-bar">
${
this.narrow
? html`<ha-assist-chip
.label=${this.hass.localize(
"ui.panel.config.automation.picker.bulk_action"
)}
slot="trigger"
>
<ha-svg-icon
slot="trailing-icon"
.path=${mdiMenuDown}
></ha-svg-icon>
</ha-assist-chip>`
: html`<ha-icon-button
.path=${mdiDotsVertical}
.label=${"ui.panel.config.automation.picker.bulk_action"}
slot="trigger"
></ha-icon-button>`
}
<ha-svg-icon
slot="trailing-icon"
.path=${mdiMenuDown}
></ha-svg-icon
></ha-assist-chip>
${
this.narrow
? html`<ha-sub-menu>
<ha-menu-item slot="item">
<div slot="headline">
${this.hass.localize(
"ui.panel.config.automation.picker.bulk_actions.move_category"
)}
</div>
<ha-svg-icon
slot="end"
.path=${mdiChevronRight}
></ha-svg-icon>
</ha-menu-item>
<ha-menu slot="menu">${categoryItems}</ha-menu>
</ha-sub-menu>`
: nothing
}
${
this.narrow || this.hass.dockedSidebar === "docked"
? html` <ha-sub-menu>
<ha-menu-item slot="item">
<div slot="headline">
${this.hass.localize(
"ui.panel.config.automation.picker.bulk_actions.add_label"
)}
</div>
<ha-svg-icon
slot="end"
.path=${mdiChevronRight}
></ha-svg-icon>
</ha-menu-item>
<ha-menu slot="menu">${labelItems}</ha-menu>
</ha-sub-menu>`
: nothing
}
</ha-button-menu-new>`
: nothing}
${!this.scenes.length ${!this.scenes.length
? html`<div class="empty" slot="empty"> ? html`<div class="empty" slot="empty">
<ha-svg-icon .path=${mdiPalette}></ha-svg-icon> <ha-svg-icon .path=${mdiPalette}></ha-svg-icon>
@@ -760,12 +553,6 @@ class HaSceneDashboard extends SubscribeMixin(LitElement) {
this._applyFilters(); this._applyFilters();
} }
private _handleSelectionChanged(
ev: HASSDomEvent<SelectionChangedEvent>
): void {
this._selected = ev.detail.value;
}
private _handleRowClicked(ev: HASSDomEvent<RowClickedEvent>) { private _handleRowClicked(ev: HASSDomEvent<RowClickedEvent>) {
const scene = this.scenes.find((a) => a.entity_id === ev.detail.id); const scene = this.scenes.find((a) => a.entity_id === ev.detail.id);
@@ -774,46 +561,6 @@ class HaSceneDashboard extends SubscribeMixin(LitElement) {
} }
} }
private async _handleBulkCategory(ev) {
const category = ev.currentTarget.value;
this._bulkAddCategory(category);
}
private async _bulkAddCategory(category: string) {
const promises: Promise<UpdateEntityRegistryEntryResult>[] = [];
this._selected.forEach((entityId) => {
promises.push(
updateEntityRegistryEntry(this.hass, entityId, {
categories: { scene: category },
})
);
});
await Promise.all(promises);
}
private async _handleBulkLabel(ev) {
const label = ev.currentTarget.value;
const action = ev.currentTarget.action;
this._bulkLabel(label, action);
}
private async _bulkLabel(label: string, action: "add" | "remove") {
const promises: Promise<UpdateEntityRegistryEntryResult>[] = [];
this._selected.forEach((entityId) => {
promises.push(
updateEntityRegistryEntry(this.hass, entityId, {
labels:
action === "add"
? this.hass.entities[entityId].labels.concat(label)
: this.hass.entities[entityId].labels.filter(
(lbl) => lbl !== label
),
})
);
});
await Promise.all(promises);
}
private _editCategory(scene: any) { private _editCategory(scene: any) {
const entityReg = this._entityReg.find( const entityReg = this._entityReg.find(
(reg) => reg.entity_id === scene.entity_id (reg) => reg.entity_id === scene.entity_id
@@ -839,13 +586,6 @@ class HaSceneDashboard extends SubscribeMixin(LitElement) {
fireEvent(this, "hass-more-info", { entityId: scene.entity_id }); fireEvent(this, "hass-more-info", { entityId: scene.entity_id });
} }
private _openSettings(scene: SceneEntity) {
showMoreInfoDialog(this, {
entityId: scene.entity_id,
view: "settings",
});
}
private _activateScene = async (scene: SceneEntity) => { private _activateScene = async (scene: SceneEntity) => {
await activateScene(this.hass, scene.entity_id); await activateScene(this.hass, scene.entity_id);
showToast(this, { showToast(this, {
@@ -909,38 +649,10 @@ class HaSceneDashboard extends SubscribeMixin(LitElement) {
}); });
} }
private async _bulkCreateCategory() {
showCategoryRegistryDetailDialog(this, {
scope: "scene",
createEntry: async (values) => {
const category = await createCategoryRegistryEntry(
this.hass,
"scene",
values
);
this._bulkAddCategory(category.category_id);
return category;
},
});
}
private _bulkCreateLabel() {
showLabelDetailDialog(this, {
createEntry: async (values) => {
const label = await createLabelRegistryEntry(this.hass, values);
this._bulkLabel(label.label_id, "add");
return label;
},
});
}
static get styles(): CSSResultGroup { static get styles(): CSSResultGroup {
return [ return [
haStyle, haStyle,
css` css`
:host {
display: block;
}
hass-tabs-subpage-data-table { hass-tabs-subpage-data-table {
--data-table-row-height: 60px; --data-table-row-height: 60px;
} }
@@ -952,16 +664,6 @@ class HaSceneDashboard extends SubscribeMixin(LitElement) {
--mdc-icon-size: 80px; --mdc-icon-size: 80px;
max-width: 500px; max-width: 500px;
} }
ha-assist-chip {
--ha-assist-chip-container-shape: 10px;
}
ha-button-menu-new ha-assist-chip {
--md-assist-chip-trailing-space: 8px;
}
ha-label {
--ha-label-background-color: var(--color, var(--grey-color));
--ha-label-background-opacity: 0.5;
}
`, `,
]; ];
} }

View File

@@ -1,14 +1,9 @@
import { consume } from "@lit-labs/context"; import { consume } from "@lit-labs/context";
import { ResizeController } from "@lit-labs/observers/resize-controller";
import { import {
mdiChevronRight,
mdiCog,
mdiContentDuplicate, mdiContentDuplicate,
mdiDelete, mdiDelete,
mdiDotsVertical,
mdiHelpCircle, mdiHelpCircle,
mdiInformationOutline, mdiInformationOutline,
mdiMenuDown,
mdiPlay, mdiPlay,
mdiPlus, mdiPlus,
mdiScriptText, mdiScriptText,
@@ -29,7 +24,6 @@ import {
import { customElement, property, state } from "lit/decorators"; import { customElement, property, state } from "lit/decorators";
import { styleMap } from "lit/directives/style-map"; import { styleMap } from "lit/directives/style-map";
import memoizeOne from "memoize-one"; import memoizeOne from "memoize-one";
import { computeCssColor } from "../../../common/color/compute-color";
import { isComponentLoaded } from "../../../common/config/is_component_loaded"; import { isComponentLoaded } from "../../../common/config/is_component_loaded";
import { formatShortDateTime } from "../../../common/datetime/format_date_time"; import { formatShortDateTime } from "../../../common/datetime/format_date_time";
import { relativeTime } from "../../../common/datetime/relative_time"; import { relativeTime } from "../../../common/datetime/relative_time";
@@ -40,7 +34,6 @@ import { LocalizeFunc } from "../../../common/translations/localize";
import { import {
DataTableColumnContainer, DataTableColumnContainer,
RowClickedEvent, RowClickedEvent,
SelectionChangedEvent,
} from "../../../components/data-table/ha-data-table"; } from "../../../components/data-table/ha-data-table";
import "../../../components/data-table/ha-data-table-labels"; import "../../../components/data-table/ha-data-table-labels";
import "../../../components/ha-fab"; import "../../../components/ha-fab";
@@ -52,24 +45,16 @@ import "../../../components/ha-filter-floor-areas";
import "../../../components/ha-filter-labels"; import "../../../components/ha-filter-labels";
import "../../../components/ha-icon-button"; import "../../../components/ha-icon-button";
import "../../../components/ha-icon-overflow-menu"; import "../../../components/ha-icon-overflow-menu";
import "../../../components/ha-menu-item";
import "../../../components/ha-sub-menu";
import "../../../components/ha-svg-icon"; import "../../../components/ha-svg-icon";
import { import {
CategoryRegistryEntry, CategoryRegistryEntry,
createCategoryRegistryEntry,
subscribeCategoryRegistry, subscribeCategoryRegistry,
} from "../../../data/category_registry"; } from "../../../data/category_registry";
import { fullEntitiesContext } from "../../../data/context"; import { fullEntitiesContext } from "../../../data/context";
import { UNAVAILABLE } from "../../../data/entity"; import { UNAVAILABLE } from "../../../data/entity";
import { import { EntityRegistryEntry } from "../../../data/entity_registry";
EntityRegistryEntry,
UpdateEntityRegistryEntryResult,
updateEntityRegistryEntry,
} from "../../../data/entity_registry";
import { import {
LabelRegistryEntry, LabelRegistryEntry,
createLabelRegistryEntry,
subscribeLabelRegistry, subscribeLabelRegistry,
} from "../../../data/label_registry"; } from "../../../data/label_registry";
import { import {
@@ -85,7 +70,6 @@ import {
showAlertDialog, showAlertDialog,
showConfirmationDialog, showConfirmationDialog,
} from "../../../dialogs/generic/show-dialog-box"; } from "../../../dialogs/generic/show-dialog-box";
import { showMoreInfoDialog } from "../../../dialogs/more-info/show-ha-more-info-dialog";
import "../../../layouts/hass-tabs-subpage-data-table"; import "../../../layouts/hass-tabs-subpage-data-table";
import { SubscribeMixin } from "../../../mixins/subscribe-mixin"; import { SubscribeMixin } from "../../../mixins/subscribe-mixin";
import { haStyle } from "../../../resources/styles"; import { haStyle } from "../../../resources/styles";
@@ -94,13 +78,10 @@ import { documentationUrl } from "../../../util/documentation-url";
import { showToast } from "../../../util/toast"; import { showToast } from "../../../util/toast";
import { showNewAutomationDialog } from "../automation/show-dialog-new-automation"; import { showNewAutomationDialog } from "../automation/show-dialog-new-automation";
import { showAssignCategoryDialog } from "../category/show-dialog-assign-category"; import { showAssignCategoryDialog } from "../category/show-dialog-assign-category";
import { showCategoryRegistryDetailDialog } from "../category/show-dialog-category-registry-detail";
import { configSections } from "../ha-panel-config"; import { configSections } from "../ha-panel-config";
import { showLabelDetailDialog } from "../labels/show-dialog-label-detail";
type ScriptItem = ScriptEntity & { type ScriptItem = ScriptEntity & {
name: string; name: string;
area: string | undefined;
category: string | undefined; category: string | undefined;
labels: LabelRegistryEntry[]; labels: LabelRegistryEntry[];
}; };
@@ -121,8 +102,6 @@ class HaScriptPicker extends SubscribeMixin(LitElement) {
@state() private _searchParms = new URLSearchParams(window.location.search); @state() private _searchParms = new URLSearchParams(window.location.search);
@state() private _selected: string[] = [];
@state() private _activeFilters?: string[]; @state() private _activeFilters?: string[];
@state() private _filteredScripts?: string[] | null; @state() private _filteredScripts?: string[] | null;
@@ -144,15 +123,10 @@ class HaScriptPicker extends SubscribeMixin(LitElement) {
@consume({ context: fullEntitiesContext, subscribe: true }) @consume({ context: fullEntitiesContext, subscribe: true })
_entityReg!: EntityRegistryEntry[]; _entityReg!: EntityRegistryEntry[];
private _sizeController = new ResizeController(this, {
callback: (entries) => entries[0]?.contentRect.width,
});
private _scripts = memoizeOne( private _scripts = memoizeOne(
( (
scripts: ScriptEntity[], scripts: ScriptEntity[],
entityReg: EntityRegistryEntry[], entityReg: EntityRegistryEntry[],
areas: HomeAssistant["areas"],
categoryReg?: CategoryRegistryEntry[], categoryReg?: CategoryRegistryEntry[],
labelReg?: LabelRegistryEntry[], labelReg?: LabelRegistryEntry[],
filteredScripts?: string[] | null filteredScripts?: string[] | null
@@ -175,9 +149,6 @@ class HaScriptPicker extends SubscribeMixin(LitElement) {
return { return {
...script, ...script,
name: computeStateName(script), name: computeStateName(script),
area: entityRegEntry?.area_id
? areas[entityRegEntry?.area_id]?.name
: undefined,
last_triggered: script.attributes.last_triggered || undefined, last_triggered: script.attributes.last_triggered || undefined,
category: category category: category
? categoryReg?.find((cat) => cat.category_id === category)?.name ? categoryReg?.find((cat) => cat.category_id === category)?.name
@@ -243,13 +214,6 @@ class HaScriptPicker extends SubscribeMixin(LitElement) {
`; `;
}, },
}, },
area: {
title: localize("ui.panel.config.script.picker.headers.area"),
hidden: true,
groupable: true,
filterable: true,
sortable: true,
},
category: { category: {
title: localize("ui.panel.config.script.picker.headers.category"), title: localize("ui.panel.config.script.picker.headers.category"),
hidden: true, hidden: true,
@@ -263,8 +227,9 @@ class HaScriptPicker extends SubscribeMixin(LitElement) {
filterable: true, filterable: true,
template: (script) => script.labels.map((lbl) => lbl.name).join(" "), template: (script) => script.labels.map((lbl) => lbl.name).join(" "),
}, },
last_triggered: { };
hidden: narrow, if (!narrow) {
columns.last_triggered = {
sortable: true, sortable: true,
width: "40%", width: "40%",
title: localize("ui.card.automation.last_triggered"), title: localize("ui.card.automation.last_triggered"),
@@ -284,74 +249,66 @@ class HaScriptPicker extends SubscribeMixin(LitElement) {
: this.hass.localize("ui.components.relative_time.never")} : this.hass.localize("ui.components.relative_time.never")}
`; `;
}, },
}, };
actions: { }
title: "",
width: "64px", columns.actions = {
type: "overflow-menu", title: "",
template: (script) => html` width: "64px",
<ha-icon-overflow-menu type: "overflow-menu",
.hass=${this.hass} template: (script) => html`
narrow <ha-icon-overflow-menu
.items=${[ .hass=${this.hass}
{ narrow
path: mdiInformationOutline, .items=${[
label: this.hass.localize( {
"ui.panel.config.script.picker.show_info" path: mdiInformationOutline,
), label: this.hass.localize(
action: () => this._showInfo(script), "ui.panel.config.script.picker.show_info"
}, ),
{ action: () => this._showInfo(script),
path: mdiCog, },
label: this.hass.localize( {
"ui.panel.config.automation.picker.show_settings" path: mdiTag,
), label: this.hass.localize(
action: () => this._openSettings(script), `ui.panel.config.script.picker.${script.category ? "edit_category" : "assign_category"}`
}, ),
{ action: () => this._editCategory(script),
path: mdiTag, },
label: this.hass.localize( {
`ui.panel.config.script.picker.${script.category ? "edit_category" : "assign_category"}` path: mdiPlay,
), label: this.hass.localize("ui.panel.config.script.picker.run"),
action: () => this._editCategory(script), action: () => this._runScript(script),
}, },
{ {
path: mdiPlay, path: mdiTransitConnection,
label: this.hass.localize( label: this.hass.localize(
"ui.panel.config.script.picker.run" "ui.panel.config.script.picker.show_trace"
), ),
action: () => this._runScript(script), action: () => this._showTrace(script),
}, },
{ {
path: mdiTransitConnection, divider: true,
label: this.hass.localize( },
"ui.panel.config.script.picker.show_trace" {
), path: mdiContentDuplicate,
action: () => this._showTrace(script), label: this.hass.localize(
}, "ui.panel.config.script.picker.duplicate"
{ ),
divider: true, action: () => this._duplicate(script),
}, },
{ {
path: mdiContentDuplicate, label: this.hass.localize(
label: this.hass.localize( "ui.panel.config.script.picker.delete"
"ui.panel.config.script.picker.duplicate" ),
), path: mdiDelete,
action: () => this._duplicate(script), action: () => this._deleteConfirm(script),
}, warning: true,
{ },
label: this.hass.localize( ]}
"ui.panel.config.script.picker.delete" >
), </ha-icon-overflow-menu>
path: mdiDelete, `,
action: () => this._deleteConfirm(script),
warning: true,
},
]}
>
</ha-icon-overflow-menu>
`,
},
}; };
return columns; return columns;
@@ -374,69 +331,6 @@ class HaScriptPicker extends SubscribeMixin(LitElement) {
} }
protected render(): TemplateResult { protected render(): TemplateResult {
const categoryItems = html`${this._categories?.map(
(category) =>
html`<ha-menu-item
.value=${category.category_id}
@click=${this._handleBulkCategory}
>
${category.icon
? html`<ha-icon slot="start" .icon=${category.icon}></ha-icon>`
: html`<ha-svg-icon slot="start" .path=${mdiTag}></ha-svg-icon>`}
<div slot="headline">${category.name}</div>
</ha-menu-item>`
)}
<ha-menu-item .value=${null} @click=${this._handleBulkCategory}>
<div slot="headline">
${this.hass.localize(
"ui.panel.config.automation.picker.bulk_actions.no_category"
)}
</div> </ha-menu-item
><md-divider role="separator" tabindex="-1"></md-divider>
<ha-menu-item @click=${this._bulkCreateCategory}>
<div slot="headline">
${this.hass.localize("ui.panel.config.category.editor.add")}
</div>
</ha-menu-item>`;
const labelItems = html`${this._labels?.map((label) => {
const color = label.color ? computeCssColor(label.color) : undefined;
const selected = this._selected.every((entityId) =>
this.hass.entities[entityId]?.labels.includes(label.label_id)
);
const partial =
!selected &&
this._selected.some((entityId) =>
this.hass.entities[entityId]?.labels.includes(label.label_id)
);
return html`<ha-menu-item
.value=${label.label_id}
.action=${selected ? "remove" : "add"}
@click=${this._handleBulkLabel}
keep-open
reducedTouchTarget
>
<ha-checkbox
slot="start"
.checked=${selected}
.indeterminate=${partial}
></ha-checkbox>
<ha-label style=${color ? `--color: ${color}` : ""}>
${label.icon
? html`<ha-icon slot="icon" .icon=${label.icon}></ha-icon>`
: nothing}
${label.name}
</ha-label>
</ha-menu-item>`;
})}
<md-divider role="separator" tabindex="-1"></md-divider>
<ha-menu-item @click=${this._bulkCreateLabel}>
<div slot="headline">
${this.hass.localize("ui.panel.config.labels.add_label")}
</div></ha-menu-item
>`;
const labelsInOverflow =
(this._sizeController.value && this._sizeController.value < 700) ||
(!this._sizeController.value && this.hass.dockedSidebar === "docked");
return html` return html`
<hass-tabs-subpage-data-table <hass-tabs-subpage-data-table
.hass=${this.hass} .hass=${this.hass}
@@ -446,9 +340,6 @@ class HaScriptPicker extends SubscribeMixin(LitElement) {
.tabs=${configSections.automations} .tabs=${configSections.automations}
hasFilters hasFilters
initialGroupColumn="category" initialGroupColumn="category"
selectable
.selected=${this._selected.length}
@selection-changed=${this._handleSelectionChanged}
.filters=${Object.values(this._filters).filter( .filters=${Object.values(this._filters).filter(
(filter) => filter.value?.length (filter) => filter.value?.length
).length} ).length}
@@ -460,7 +351,6 @@ class HaScriptPicker extends SubscribeMixin(LitElement) {
.data=${this._scripts( .data=${this._scripts(
this.scripts, this.scripts,
this._entityReg, this._entityReg,
this.hass.areas,
this._categories, this._categories,
this._labels, this._labels,
this._filteredScripts this._filteredScripts
@@ -542,104 +432,6 @@ class HaScriptPicker extends SubscribeMixin(LitElement) {
.narrow=${this.narrow} .narrow=${this.narrow}
@expanded-changed=${this._filterExpanded} @expanded-changed=${this._filterExpanded}
></ha-filter-blueprints> ></ha-filter-blueprints>
${!this.narrow
? html`<ha-button-menu-new slot="selection-bar">
<ha-assist-chip
slot="trigger"
.label=${this.hass.localize(
"ui.panel.config.automation.picker.bulk_actions.move_category"
)}
>
<ha-svg-icon
slot="trailing-icon"
.path=${mdiMenuDown}
></ha-svg-icon>
</ha-assist-chip>
${categoryItems}
</ha-button-menu-new>
${labelsInOverflow
? nothing
: html`<ha-button-menu-new slot="selection-bar">
<ha-assist-chip
slot="trigger"
.label=${this.hass.localize(
"ui.panel.config.automation.picker.bulk_actions.add_label"
)}
>
<ha-svg-icon
slot="trailing-icon"
.path=${mdiMenuDown}
></ha-svg-icon>
</ha-assist-chip>
${labelItems}
</ha-button-menu-new>`}`
: nothing}
${this.narrow || labelsInOverflow
? html`
<ha-button-menu-new has-overflow slot="selection-bar">
${
this.narrow
? html`<ha-assist-chip
.label=${this.hass.localize(
"ui.panel.config.automation.picker.bulk_action"
)}
slot="trigger"
>
<ha-svg-icon
slot="trailing-icon"
.path=${mdiMenuDown}
></ha-svg-icon>
</ha-assist-chip>`
: html`<ha-icon-button
.path=${mdiDotsVertical}
.label=${"ui.panel.config.automation.picker.bulk_action"}
slot="trigger"
></ha-icon-button>`
}
<ha-svg-icon
slot="trailing-icon"
.path=${mdiMenuDown}
></ha-svg-icon
></ha-assist-chip>
${
this.narrow
? html`<ha-sub-menu>
<ha-menu-item slot="item">
<div slot="headline">
${this.hass.localize(
"ui.panel.config.automation.picker.bulk_actions.move_category"
)}
</div>
<ha-svg-icon
slot="end"
.path=${mdiChevronRight}
></ha-svg-icon>
</ha-menu-item>
<ha-menu slot="menu">${categoryItems}</ha-menu>
</ha-sub-menu>`
: nothing
}
${
this.narrow || this.hass.dockedSidebar === "docked"
? html` <ha-sub-menu>
<ha-menu-item slot="item">
<div slot="headline">
${this.hass.localize(
"ui.panel.config.automation.picker.bulk_actions.add_label"
)}
</div>
<ha-svg-icon
slot="end"
.path=${mdiChevronRight}
></ha-svg-icon>
</ha-menu-item>
<ha-menu slot="menu">${labelItems}</ha-menu>
</ha-sub-menu>`
: nothing
}
</ha-button-menu-new>`
: nothing}
${!this.scripts.length ${!this.scripts.length
? html` <div class="empty" slot="empty"> ? html` <div class="empty" slot="empty">
<ha-svg-icon .path=${mdiScriptText}></ha-svg-icon> <ha-svg-icon .path=${mdiScriptText}></ha-svg-icon>
@@ -837,52 +629,6 @@ class HaScriptPicker extends SubscribeMixin(LitElement) {
}); });
} }
private _handleSelectionChanged(
ev: HASSDomEvent<SelectionChangedEvent>
): void {
this._selected = ev.detail.value;
}
private async _handleBulkCategory(ev) {
const category = ev.currentTarget.value;
this._bulkAddCategory(category);
}
private async _bulkAddCategory(category: string) {
const promises: Promise<UpdateEntityRegistryEntryResult>[] = [];
this._selected.forEach((entityId) => {
promises.push(
updateEntityRegistryEntry(this.hass, entityId, {
categories: { script: category },
})
);
});
await Promise.all(promises);
}
private async _handleBulkLabel(ev) {
const label = ev.currentTarget.value;
const action = ev.currentTarget.action;
this._bulkLabel(label, action);
}
private async _bulkLabel(label: string, action: "add" | "remove") {
const promises: Promise<UpdateEntityRegistryEntryResult>[] = [];
this._selected.forEach((entityId) => {
promises.push(
updateEntityRegistryEntry(this.hass, entityId, {
labels:
action === "add"
? this.hass.entities[entityId].labels.concat(label)
: this.hass.entities[entityId].labels.filter(
(lbl) => lbl !== label
),
})
);
});
await Promise.all(promises);
}
private _handleRowClicked(ev: HASSDomEvent<RowClickedEvent>) { private _handleRowClicked(ev: HASSDomEvent<RowClickedEvent>) {
const entry = this.entityRegistry.find((e) => e.entity_id === ev.detail.id); const entry = this.entityRegistry.find((e) => e.entity_id === ev.detail.id);
if (entry) { if (entry) {
@@ -919,13 +665,6 @@ class HaScriptPicker extends SubscribeMixin(LitElement) {
fireEvent(this, "hass-more-info", { entityId: script.entity_id }); fireEvent(this, "hass-more-info", { entityId: script.entity_id });
} }
private _openSettings(script: any) {
showMoreInfoDialog(this, {
entityId: script.entity_id,
view: "settings",
});
}
private _showTrace(script: any) { private _showTrace(script: any) {
const entry = this.entityRegistry.find( const entry = this.entityRegistry.find(
(e) => e.entity_id === script.entity_id (e) => e.entity_id === script.entity_id
@@ -1025,38 +764,10 @@ class HaScriptPicker extends SubscribeMixin(LitElement) {
} }
} }
private async _bulkCreateCategory() {
showCategoryRegistryDetailDialog(this, {
scope: "script",
createEntry: async (values) => {
const category = await createCategoryRegistryEntry(
this.hass,
"script",
values
);
this._bulkAddCategory(category.category_id);
return category;
},
});
}
private _bulkCreateLabel() {
showLabelDetailDialog(this, {
createEntry: async (values) => {
const label = await createLabelRegistryEntry(this.hass, values);
this._bulkLabel(label.label_id, "add");
return label;
},
});
}
static get styles(): CSSResultGroup { static get styles(): CSSResultGroup {
return [ return [
haStyle, haStyle,
css` css`
:host {
display: block;
}
hass-tabs-subpage-data-table { hass-tabs-subpage-data-table {
--data-table-row-height: 60px; --data-table-row-height: 60px;
} }
@@ -1071,16 +782,6 @@ class HaScriptPicker extends SubscribeMixin(LitElement) {
--mdc-icon-size: 80px; --mdc-icon-size: 80px;
max-width: 500px; max-width: 500px;
} }
ha-assist-chip {
--ha-assist-chip-container-shape: 10px;
}
ha-button-menu-new ha-assist-chip {
--md-assist-chip-trailing-space: 8px;
}
ha-label {
--ha-label-background-color: var(--color, var(--grey-color));
--ha-label-background-opacity: 0.5;
}
`, `,
]; ];
} }

View File

@@ -35,7 +35,6 @@ import { ButtonsHeaderFooterConfig } from "../header-footer/types";
const HIDE_DOMAIN = new Set([ const HIDE_DOMAIN = new Set([
"automation", "automation",
"configurator", "configurator",
"conversation",
"device_tracker", "device_tracker",
"geo_location", "geo_location",
"persistent_notification", "persistent_notification",

View File

@@ -58,12 +58,18 @@ export interface AndCondition extends BaseCondition {
function getValueFromEntityId( function getValueFromEntityId(
hass: HomeAssistant, hass: HomeAssistant,
value: string value: string | string[]
): string | undefined { ): string | string[] {
if (isValidEntityId(value) && hass.states[value]) { if (
return hass.states[value]?.state; typeof value === "string" &&
isValidEntityId(value) &&
hass.states[value]
) {
value = hass.states[value]?.state;
} else if (Array.isArray(value)) {
value = value.map((v) => getValueFromEntityId(hass, v) as string);
} }
return undefined; return value;
} }
function checkStateCondition( function checkStateCondition(
@@ -77,17 +83,8 @@ function checkStateCondition(
let value = condition.state ?? condition.state_not; let value = condition.state ?? condition.state_not;
// Handle entity_id, UI should be updated for conditionnal card (filters does not have UI for now) // Handle entity_id, UI should be updated for conditionnal card (filters does not have UI for now)
if (Array.isArray(value)) { if (Array.isArray(value) || typeof value === "string") {
const entityValues = value value = getValueFromEntityId(hass, value);
.map((v) => getValueFromEntityId(hass, v))
.filter((v): v is string => v !== undefined);
value = [...value, ...entityValues];
} else if (typeof value === "string") {
const entityValue = getValueFromEntityId(hass, value);
value = [value];
if (entityValue) {
value.push(entityValue);
}
} }
return condition.state != null return condition.state != null
@@ -106,10 +103,10 @@ function checkStateNumericCondition(
// Handle entity_id, UI should be updated for conditionnal card (filters does not have UI for now) // Handle entity_id, UI should be updated for conditionnal card (filters does not have UI for now)
if (typeof above === "string") { if (typeof above === "string") {
above = getValueFromEntityId(hass, above) ?? above; above = getValueFromEntityId(hass, above) as string;
} }
if (typeof below === "string") { if (typeof below === "string") {
below = getValueFromEntityId(hass, below) ?? below; below = getValueFromEntityId(hass, below) as string;
} }
const numericState = Number(state); const numericState = Number(state);

View File

@@ -172,14 +172,12 @@ class DialogDashboardStrategyEditor extends LitElement {
`; `;
} }
private _takeControl(ev) { private _takeControl() {
ev.stopPropagation();
this._params!.takeControl(); this._params!.takeControl();
this.closeDialog(); this.closeDialog();
} }
private _showRawConfigEditor(ev) { private _showRawConfigEditor() {
ev.stopPropagation();
this._params!.showRawConfigEditor(); this._params!.showRawConfigEditor();
this.closeDialog(); this.closeDialog();
} }

View File

@@ -116,9 +116,6 @@ export const getMyRedirects = (hasSupervisor: boolean): Redirects => ({
entities: { entities: {
redirect: "/config/entities", redirect: "/config/entities",
}, },
labels: {
redirect: "/config/labels",
},
energy: { energy: {
component: "energy", component: "energy",
redirect: "/energy", redirect: "/energy",

View File

@@ -129,7 +129,7 @@ export class HaStateControlAlarmControlPanelModes extends LitElement {
max-height: max(320px, var(--modes-count, 1) * 80px); max-height: max(320px, var(--modes-count, 1) * 80px);
min-height: max(200px, var(--modes-count, 1) * 80px); min-height: max(200px, var(--modes-count, 1) * 80px);
--control-select-thickness: 130px; --control-select-thickness: 130px;
--control-select-border-radius: 36px; --control-select-border-radius: 48px;
--control-select-color: var(--primary-color); --control-select-color: var(--primary-color);
--control-select-background: var(--disabled-color); --control-select-background: var(--disabled-color);
--control-select-background-opacity: 0.2; --control-select-background-opacity: 0.2;

View File

@@ -75,7 +75,7 @@ export class HaStateControlCoverPosition extends LitElement {
max-height: 320px; max-height: 320px;
min-height: 200px; min-height: 200px;
--control-slider-thickness: 130px; --control-slider-thickness: 130px;
--control-slider-border-radius: 36px; --control-slider-border-radius: 48px;
--control-slider-color: var(--primary-color); --control-slider-color: var(--primary-color);
--control-slider-background: var(--disabled-color); --control-slider-background: var(--disabled-color);
--control-slider-background-opacity: 0.2; --control-slider-background-opacity: 0.2;

View File

@@ -112,7 +112,7 @@ export class HaStateControlInfoCoverTiltPosition extends LitElement {
max-height: 320px; max-height: 320px;
min-height: 200px; min-height: 200px;
--control-slider-thickness: 130px; --control-slider-thickness: 130px;
--control-slider-border-radius: 36px; --control-slider-border-radius: 48px;
--control-slider-color: var(--primary-color); --control-slider-color: var(--primary-color);
--control-slider-background: var(--disabled-color); --control-slider-background: var(--disabled-color);
--control-slider-background-opacity: 0.2; --control-slider-background-opacity: 0.2;

View File

@@ -142,7 +142,7 @@ export class HaStateControlCoverToggle extends LitElement {
max-height: 320px; max-height: 320px;
min-height: 200px; min-height: 200px;
--control-switch-thickness: 130px; --control-switch-thickness: 130px;
--control-switch-border-radius: 36px; --control-switch-border-radius: 48px;
--control-switch-padding: 6px; --control-switch-padding: 6px;
--mdc-icon-size: 24px; --mdc-icon-size: 24px;
} }
@@ -159,7 +159,7 @@ export class HaStateControlCoverToggle extends LitElement {
ha-control-button { ha-control-button {
flex: 1; flex: 1;
width: 100%; width: 100%;
--control-button-border-radius: 36px; --control-button-border-radius: 48px;
--mdc-icon-size: 24px; --mdc-icon-size: 24px;
} }
ha-control-button.active { ha-control-button.active {

View File

@@ -142,7 +142,7 @@ export class HaStateControlFanSpeed extends LitElement {
max-height: 320px; max-height: 320px;
min-height: 200px; min-height: 200px;
--control-slider-thickness: 130px; --control-slider-thickness: 130px;
--control-slider-border-radius: 36px; --control-slider-border-radius: 48px;
--control-slider-color: var(--primary-color); --control-slider-color: var(--primary-color);
--control-slider-background: var(--disabled-color); --control-slider-background: var(--disabled-color);
--control-slider-background-opacity: 0.2; --control-slider-background-opacity: 0.2;
@@ -153,7 +153,7 @@ export class HaStateControlFanSpeed extends LitElement {
max-height: 320px; max-height: 320px;
min-height: 200px; min-height: 200px;
--control-select-thickness: 130px; --control-select-thickness: 130px;
--control-select-border-radius: 36px; --control-select-border-radius: 48px;
--control-select-color: var(--primary-color); --control-select-color: var(--primary-color);
--control-select-background: var(--disabled-color); --control-select-background: var(--disabled-color);
--control-select-background-opacity: 0.2; --control-select-background-opacity: 0.2;

View File

@@ -133,7 +133,7 @@ export class HaStateControlToggle extends LitElement {
max-height: 320px; max-height: 320px;
min-height: 200px; min-height: 200px;
--control-switch-thickness: 130px; --control-switch-thickness: 130px;
--control-switch-border-radius: 36px; --control-switch-border-radius: 48px;
--control-switch-padding: 6px; --control-switch-padding: 6px;
--mdc-icon-size: 24px; --mdc-icon-size: 24px;
} }
@@ -150,7 +150,7 @@ export class HaStateControlToggle extends LitElement {
ha-control-button { ha-control-button {
flex: 1; flex: 1;
width: 100%; width: 100%;
--control-button-border-radius: 36px; --control-button-border-radius: 48px;
--mdc-icon-size: 24px; --mdc-icon-size: 24px;
} }
ha-control-button.active { ha-control-button.active {

View File

@@ -89,7 +89,7 @@ export class HaStateControlLightBrightness extends LitElement {
max-height: 320px; max-height: 320px;
min-height: 200px; min-height: 200px;
--control-slider-thickness: 130px; --control-slider-thickness: 130px;
--control-slider-border-radius: 36px; --control-slider-border-radius: 48px;
--control-slider-color: var(--primary-color); --control-slider-color: var(--primary-color);
--control-slider-background: var(--disabled-color); --control-slider-background: var(--disabled-color);
--control-slider-background-opacity: 0.2; --control-slider-background-opacity: 0.2;

View File

@@ -167,7 +167,7 @@ export class HaStateControlLockToggle extends LitElement {
max-height: 320px; max-height: 320px;
min-height: 200px; min-height: 200px;
--control-switch-thickness: 130px; --control-switch-thickness: 130px;
--control-switch-border-radius: 36px; --control-switch-border-radius: 48px;
--control-switch-padding: 6px; --control-switch-padding: 6px;
--mdc-icon-size: 24px; --mdc-icon-size: 24px;
} }
@@ -187,7 +187,7 @@ export class HaStateControlLockToggle extends LitElement {
ha-control-button { ha-control-button {
flex: 1; flex: 1;
width: 100%; width: 100%;
--control-button-border-radius: 36px; --control-button-border-radius: 48px;
--mdc-icon-size: 24px; --mdc-icon-size: 24px;
} }
ha-control-button.active { ha-control-button.active {

View File

@@ -71,7 +71,7 @@ export class HaStateControlValvePosition extends LitElement {
max-height: 320px; max-height: 320px;
min-height: 200px; min-height: 200px;
--control-slider-thickness: 130px; --control-slider-thickness: 130px;
--control-slider-border-radius: 36px; --control-slider-border-radius: 48px;
--control-slider-color: var(--primary-color); --control-slider-color: var(--primary-color);
--control-slider-background: var(--disabled-color); --control-slider-background: var(--disabled-color);
--control-slider-background-opacity: 0.2; --control-slider-background-opacity: 0.2;

View File

@@ -142,7 +142,7 @@ export class HaStateControlValveToggle extends LitElement {
max-height: 320px; max-height: 320px;
min-height: 200px; min-height: 200px;
--control-switch-thickness: 130px; --control-switch-thickness: 130px;
--control-switch-border-radius: 36px; --control-switch-border-radius: 48px;
--control-switch-padding: 6px; --control-switch-padding: 6px;
--mdc-icon-size: 24px; --mdc-icon-size: 24px;
} }
@@ -159,7 +159,7 @@ export class HaStateControlValveToggle extends LitElement {
ha-control-button { ha-control-button {
flex: 1; flex: 1;
width: 100%; width: 100%;
--control-button-border-radius: 36px; --control-button-border-radius: 48px;
--mdc-icon-size: 24px; --mdc-icon-size: 24px;
} }
ha-control-button.active { ha-control-button.active {

View File

@@ -1927,10 +1927,7 @@
"aliases_section": "Aliases", "aliases_section": "Aliases",
"no_aliases": "No configured aliases", "no_aliases": "No configured aliases",
"configured_aliases": "{count} configured {count, plural,\n one {alias}\n other {aliases}\n}", "configured_aliases": "{count} configured {count, plural,\n one {alias}\n other {aliases}\n}",
"aliases_description": "Aliases are alternative names used in voice assistants to refer to this floor.", "aliases_description": "Aliases are alternative names used in voice assistants to refer to this floor."
"areas_section": "Areas",
"areas_description": "Specify the areas that are on this floor.",
"add_area": "Add area"
} }
}, },
"category": { "category": {
@@ -1965,7 +1962,6 @@
"color": "Color" "color": "Color"
}, },
"add_label": "Add label", "add_label": "Add label",
"manage_labels": "Manage labels",
"no_labels": "You don't have any labels", "no_labels": "You don't have any labels",
"introduction": "Labels can help you organize your areas, devices and entities. They can be used to filter in the UI, or use them as a target in automations.", "introduction": "Labels can help you organize your areas, devices and entities. They can be used to filter in the UI, or use them as a target in automations.",
"introduction2": "Go to the area, device or entity you want to add a label to, and click on the edit button to assign labels to them.", "introduction2": "Go to the area, device or entity you want to add a label to, and click on the edit button to assign labels to them.",
@@ -2266,8 +2262,7 @@
"name": "Name", "name": "Name",
"entity_id": "Entity ID", "entity_id": "Entity ID",
"type": "Type", "type": "Type",
"editable": "Editable", "editable": "Editable"
"category": "Category"
}, },
"create_helper": "Create helper", "create_helper": "Create helper",
"no_helpers": "Looks like you don't have any helpers yet!" "no_helpers": "Looks like you don't have any helpers yet!"
@@ -2690,8 +2685,7 @@
"trigger": "Trigger", "trigger": "Trigger",
"actions": "Actions", "actions": "Actions",
"state": "State", "state": "State",
"category": "Category", "category": "Category"
"area": "Area"
}, },
"bulk_action": "Action", "bulk_action": "Action",
"bulk_actions": { "bulk_actions": {
@@ -3565,8 +3559,7 @@
"headers": { "headers": {
"name": "Name", "name": "Name",
"state": "State", "state": "State",
"category": "Category", "category": "Category"
"area": "Area"
}, },
"edit_category": "[%key:ui::panel::config::automation::picker::edit_category%]", "edit_category": "[%key:ui::panel::config::automation::picker::edit_category%]",
"assign_category": "[%key:ui::panel::config::automation::picker::assign_category%]", "assign_category": "[%key:ui::panel::config::automation::picker::assign_category%]",
@@ -3675,8 +3668,7 @@
"state": "State", "state": "State",
"name": "Name", "name": "Name",
"last_activated": "Last activated", "last_activated": "Last activated",
"category": "Category", "category": "Category"
"area": "Area"
}, },
"edit_category": "[%key:ui::panel::config::automation::picker::edit_category%]", "edit_category": "[%key:ui::panel::config::automation::picker::edit_category%]",
"assign_category": "[%key:ui::panel::config::automation::picker::assign_category%]", "assign_category": "[%key:ui::panel::config::automation::picker::assign_category%]",
@@ -4060,9 +4052,6 @@
"button": "Hide selected", "button": "Hide selected",
"confirm_title": "Do you want to hide {number} {number, plural,\n one {entity}\n other {entities}\n}?", "confirm_title": "Do you want to hide {number} {number, plural,\n one {entity}\n other {entities}\n}?",
"confirm_text": "Hidden entities will not be shown on your dashboard. Their history is still tracked and you can still interact with them with services." "confirm_text": "Hidden entities will not be shown on your dashboard. Their history is still tracked and you can still interact with them with services."
},
"unhide_selected": {
"button": "Unhide selected"
} }
} }
}, },