mirror of
				https://github.com/home-assistant/frontend.git
				synced 2025-10-26 03:59:43 +00:00 
			
		
		
		
	Compare commits
	
		
			19 Commits
		
	
	
		
			sec_pypi_p
			...
			power-char
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 8b3b7e701a | ||
|   | 299c08c0ea | ||
|   | 2fd33069c1 | ||
|   | a322182f45 | ||
|   | 9926730497 | ||
|   | daec91bbe2 | ||
|   | 8dde4c9a21 | ||
|   | 24388e924f | ||
|   | 7a2826a580 | ||
|   | c32e5a049a | ||
|   | f294c1bae6 | ||
|   | 4aadd7ec71 | ||
|   | 0b71ae51b3 | ||
|   | 3301c00471 | ||
|   | f68d885dd8 | ||
|   | 2d602a13e0 | ||
|   | 9d78043915 | ||
|   | 18bcc1c4f9 | ||
|   | 14133e28ad | 
							
								
								
									
										13
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										13
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -19,11 +19,8 @@ jobs: | |||||||
|   release: |   release: | ||||||
|     name: Release |     name: Release | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     environment: pypi |  | ||||||
|     permissions: |     permissions: | ||||||
|       contents: write # Required to upload release assets |       contents: write # Required to upload release assets | ||||||
|       id-token: write # For "Trusted Publisher" to PyPi |  | ||||||
|     if: github.repository_owner == 'home-assistant' |  | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout the repository |       - name: Checkout the repository | ||||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
| @@ -49,18 +46,14 @@ jobs: | |||||||
|         run: ./script/translations_download |         run: ./script/translations_download | ||||||
|         env: |         env: | ||||||
|           LOKALISE_TOKEN: ${{ secrets.LOKALISE_TOKEN }} |           LOKALISE_TOKEN: ${{ secrets.LOKALISE_TOKEN }} | ||||||
|  |  | ||||||
|       - name: Build and release package |       - name: Build and release package | ||||||
|         run: | |         run: | | ||||||
|           python3 -m pip install build |           python3 -m pip install twine build | ||||||
|  |           export TWINE_USERNAME="__token__" | ||||||
|  |           export TWINE_PASSWORD="${{ secrets.TWINE_TOKEN }}" | ||||||
|           export SKIP_FETCH_NIGHTLY_TRANSLATIONS=1 |           export SKIP_FETCH_NIGHTLY_TRANSLATIONS=1 | ||||||
|           script/release |           script/release | ||||||
|  |  | ||||||
|       - name: Publish to PyPI |  | ||||||
|         uses: pypa/gh-action-pypi-publish@ed0c53931b1dc9bd32cbe73a98c7f6766f8a527e # v1.13.0 |  | ||||||
|         with: |  | ||||||
|           skip-existing: true |  | ||||||
|  |  | ||||||
|       - name: Upload release assets |       - name: Upload release assets | ||||||
|         uses: softprops/action-gh-release@6da8fa9354ddfdc4aeace5fc48d7f679b5214090 # v2.4.1 |         uses: softprops/action-gh-release@6da8fa9354ddfdc4aeace5fc48d7f679b5214090 # v2.4.1 | ||||||
|         with: |         with: | ||||||
|   | |||||||
							
								
								
									
										14
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										14
									
								
								package.json
									
									
									
									
									
								
							| @@ -34,7 +34,7 @@ | |||||||
|     "@codemirror/legacy-modes": "6.5.2", |     "@codemirror/legacy-modes": "6.5.2", | ||||||
|     "@codemirror/search": "6.5.11", |     "@codemirror/search": "6.5.11", | ||||||
|     "@codemirror/state": "6.5.2", |     "@codemirror/state": "6.5.2", | ||||||
|     "@codemirror/view": "6.38.6", |     "@codemirror/view": "6.38.5", | ||||||
|     "@date-fns/tz": "1.4.1", |     "@date-fns/tz": "1.4.1", | ||||||
|     "@egjs/hammerjs": "2.0.17", |     "@egjs/hammerjs": "2.0.17", | ||||||
|     "@formatjs/intl-datetimeformat": "6.18.2", |     "@formatjs/intl-datetimeformat": "6.18.2", | ||||||
| @@ -52,7 +52,7 @@ | |||||||
|     "@fullcalendar/list": "6.1.19", |     "@fullcalendar/list": "6.1.19", | ||||||
|     "@fullcalendar/luxon3": "6.1.19", |     "@fullcalendar/luxon3": "6.1.19", | ||||||
|     "@fullcalendar/timegrid": "6.1.19", |     "@fullcalendar/timegrid": "6.1.19", | ||||||
|     "@home-assistant/webawesome": "3.0.0-beta.6.ha.5", |     "@home-assistant/webawesome": "3.0.0-beta.6.ha.4", | ||||||
|     "@lezer/highlight": "1.2.1", |     "@lezer/highlight": "1.2.1", | ||||||
|     "@lit-labs/motion": "1.0.9", |     "@lit-labs/motion": "1.0.9", | ||||||
|     "@lit-labs/observers": "2.0.6", |     "@lit-labs/observers": "2.0.6", | ||||||
| @@ -153,11 +153,11 @@ | |||||||
|     "@babel/plugin-transform-runtime": "7.28.3", |     "@babel/plugin-transform-runtime": "7.28.3", | ||||||
|     "@babel/preset-env": "7.28.3", |     "@babel/preset-env": "7.28.3", | ||||||
|     "@bundle-stats/plugin-webpack-filter": "4.21.5", |     "@bundle-stats/plugin-webpack-filter": "4.21.5", | ||||||
|     "@lokalise/node-api": "15.3.1", |     "@lokalise/node-api": "15.3.0", | ||||||
|     "@octokit/auth-oauth-device": "8.0.2", |     "@octokit/auth-oauth-device": "8.0.2", | ||||||
|     "@octokit/plugin-retry": "8.0.2", |     "@octokit/plugin-retry": "8.0.2", | ||||||
|     "@octokit/rest": "22.0.0", |     "@octokit/rest": "22.0.0", | ||||||
|     "@rsdoctor/rspack-plugin": "1.3.3", |     "@rsdoctor/rspack-plugin": "1.3.1", | ||||||
|     "@rspack/core": "1.5.8", |     "@rspack/core": "1.5.8", | ||||||
|     "@rspack/dev-server": "1.1.4", |     "@rspack/dev-server": "1.1.4", | ||||||
|     "@types/babel__plugin-transform-runtime": "7.9.5", |     "@types/babel__plugin-transform-runtime": "7.9.5", | ||||||
| @@ -167,7 +167,7 @@ | |||||||
|     "@types/culori": "4.0.1", |     "@types/culori": "4.0.1", | ||||||
|     "@types/html-minifier-terser": "7.0.2", |     "@types/html-minifier-terser": "7.0.2", | ||||||
|     "@types/js-yaml": "4.0.9", |     "@types/js-yaml": "4.0.9", | ||||||
|     "@types/leaflet": "1.9.21", |     "@types/leaflet": "1.9.20", | ||||||
|     "@types/leaflet-draw": "1.0.13", |     "@types/leaflet-draw": "1.0.13", | ||||||
|     "@types/leaflet.markercluster": "1.5.6", |     "@types/leaflet.markercluster": "1.5.6", | ||||||
|     "@types/lodash.merge": "4.6.9", |     "@types/lodash.merge": "4.6.9", | ||||||
| @@ -203,7 +203,7 @@ | |||||||
|     "husky": "9.1.7", |     "husky": "9.1.7", | ||||||
|     "jsdom": "27.0.0", |     "jsdom": "27.0.0", | ||||||
|     "jszip": "3.10.1", |     "jszip": "3.10.1", | ||||||
|     "lint-staged": "16.2.4", |     "lint-staged": "16.2.3", | ||||||
|     "lit-analyzer": "2.0.3", |     "lit-analyzer": "2.0.3", | ||||||
|     "lodash.merge": "4.6.2", |     "lodash.merge": "4.6.2", | ||||||
|     "lodash.template": "4.5.0", |     "lodash.template": "4.5.0", | ||||||
| @@ -217,7 +217,7 @@ | |||||||
|     "terser-webpack-plugin": "5.3.14", |     "terser-webpack-plugin": "5.3.14", | ||||||
|     "ts-lit-plugin": "2.0.2", |     "ts-lit-plugin": "2.0.2", | ||||||
|     "typescript": "5.9.3", |     "typescript": "5.9.3", | ||||||
|     "typescript-eslint": "8.46.1", |     "typescript-eslint": "8.46.0", | ||||||
|     "vite-tsconfig-paths": "5.1.4", |     "vite-tsconfig-paths": "5.1.4", | ||||||
|     "vitest": "3.2.4", |     "vitest": "3.2.4", | ||||||
|     "webpack-stats-plugin": "1.1.3", |     "webpack-stats-plugin": "1.1.3", | ||||||
|   | |||||||
| @@ -1,4 +1,5 @@ | |||||||
| #!/bin/sh | #!/bin/sh | ||||||
|  | # Pushes a new version to PyPi. | ||||||
|  |  | ||||||
| # Stop on errors | # Stop on errors | ||||||
| set -e | set -e | ||||||
| @@ -11,4 +12,5 @@ yarn install | |||||||
| script/build_frontend | script/build_frontend | ||||||
|  |  | ||||||
| rm -rf dist home_assistant_frontend.egg-info | rm -rf dist home_assistant_frontend.egg-info | ||||||
| python3 -m build -q | python3 -m build | ||||||
|  | python3 -m twine upload dist/*.whl --skip-existing | ||||||
|   | |||||||
| @@ -9,11 +9,6 @@ import { getEntityContext } from "./context/get_entity_context"; | |||||||
|  |  | ||||||
| const DEFAULT_SEPARATOR = " "; | const DEFAULT_SEPARATOR = " "; | ||||||
|  |  | ||||||
| export const DEFAULT_ENTITY_NAME = [ |  | ||||||
|   { type: "device" }, |  | ||||||
|   { type: "entity" }, |  | ||||||
| ] satisfies EntityNameItem[]; |  | ||||||
|  |  | ||||||
| export type EntityNameItem = | export type EntityNameItem = | ||||||
|   | { |   | { | ||||||
|       type: "entity" | "device" | "area" | "floor"; |       type: "entity" | "device" | "area" | "floor"; | ||||||
| @@ -29,14 +24,14 @@ export interface EntityNameOptions { | |||||||
|  |  | ||||||
| export const computeEntityNameDisplay = ( | export const computeEntityNameDisplay = ( | ||||||
|   stateObj: HassEntity, |   stateObj: HassEntity, | ||||||
|   name: EntityNameItem | EntityNameItem[] | undefined, |   name: EntityNameItem | EntityNameItem[], | ||||||
|   entities: HomeAssistant["entities"], |   entities: HomeAssistant["entities"], | ||||||
|   devices: HomeAssistant["devices"], |   devices: HomeAssistant["devices"], | ||||||
|   areas: HomeAssistant["areas"], |   areas: HomeAssistant["areas"], | ||||||
|   floors: HomeAssistant["floors"], |   floors: HomeAssistant["floors"], | ||||||
|   options?: EntityNameOptions |   options?: EntityNameOptions | ||||||
| ) => { | ) => { | ||||||
|   let items = ensureArray(name || DEFAULT_ENTITY_NAME); |   let items = ensureArray(name); | ||||||
|  |  | ||||||
|   const separator = options?.separator ?? DEFAULT_SEPARATOR; |   const separator = options?.separator ?? DEFAULT_SEPARATOR; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -8,10 +8,10 @@ interface AreaContext { | |||||||
| } | } | ||||||
| export const getAreaContext = ( | export const getAreaContext = ( | ||||||
|   area: AreaRegistryEntry, |   area: AreaRegistryEntry, | ||||||
|   hassFloors: HomeAssistant["floors"] |   hass: HomeAssistant | ||||||
| ): AreaContext => { | ): AreaContext => { | ||||||
|   const floorId = area.floor_id; |   const floorId = area.floor_id; | ||||||
|   const floor = floorId ? hassFloors[floorId] : undefined; |   const floor = floorId ? hass.floors[floorId] : undefined; | ||||||
|  |  | ||||||
|   return { |   return { | ||||||
|     area: area, |     area: area, | ||||||
|   | |||||||
| @@ -6,7 +6,6 @@ import { showConfirmationDialog } from "../../dialogs/generic/show-dialog-box"; | |||||||
| import "./ha-progress-button"; | import "./ha-progress-button"; | ||||||
| import type { HomeAssistant } from "../../types"; | import type { HomeAssistant } from "../../types"; | ||||||
| import { fireEvent } from "../../common/dom/fire_event"; | import { fireEvent } from "../../common/dom/fire_event"; | ||||||
| import type { Appearance } from "../ha-button"; |  | ||||||
|  |  | ||||||
| @customElement("ha-call-service-button") | @customElement("ha-call-service-button") | ||||||
| class HaCallServiceButton extends LitElement { | class HaCallServiceButton extends LitElement { | ||||||
| @@ -26,14 +25,12 @@ class HaCallServiceButton extends LitElement { | |||||||
|  |  | ||||||
|   @property() public confirmation?; |   @property() public confirmation?; | ||||||
|  |  | ||||||
|   @property() public appearance: Appearance = "plain"; |  | ||||||
|  |  | ||||||
|   public render(): TemplateResult { |   public render(): TemplateResult { | ||||||
|     return html` |     return html` | ||||||
|       <ha-progress-button |       <ha-progress-button | ||||||
|         .progress=${this.progress} |         .progress=${this.progress} | ||||||
|         .disabled=${this.disabled} |         .disabled=${this.disabled} | ||||||
|         .appearance=${this.appearance} |         appearance="plain" | ||||||
|         @click=${this._buttonTapped} |         @click=${this._buttonTapped} | ||||||
|         tabindex="0" |         tabindex="0" | ||||||
|       > |       > | ||||||
|   | |||||||
| @@ -1,22 +1,21 @@ | |||||||
| import type { LineSeriesOption } from "echarts"; | import type { LineSeriesOption } from "echarts"; | ||||||
|  |  | ||||||
| export function downSampleLineData< | export function downSampleLineData( | ||||||
|   T extends [number, number] | NonNullable<LineSeriesOption["data"]>[number], |   data: LineSeriesOption["data"], | ||||||
| >( |   chartWidth: number, | ||||||
|   data: T[] | undefined, |  | ||||||
|   maxDetails: number, |  | ||||||
|   minX?: number, |   minX?: number, | ||||||
|   maxX?: number |   maxX?: number | ||||||
| ): T[] { | ) { | ||||||
|   if (!data) { |   if (!data || data.length < 10) { | ||||||
|     return []; |     return data; | ||||||
|   } |   } | ||||||
|   if (data.length <= maxDetails) { |   const width = chartWidth * window.devicePixelRatio; | ||||||
|  |   if (data.length <= width) { | ||||||
|     return data; |     return data; | ||||||
|   } |   } | ||||||
|   const min = minX ?? getPointData(data[0]!)[0]; |   const min = minX ?? getPointData(data[0]!)[0]; | ||||||
|   const max = maxX ?? getPointData(data[data.length - 1]!)[0]; |   const max = maxX ?? getPointData(data[data.length - 1]!)[0]; | ||||||
|   const step = Math.ceil((max - min) / Math.floor(maxDetails)); |   const step = Math.floor((max - min) / width); | ||||||
|   const frames = new Map< |   const frames = new Map< | ||||||
|     number, |     number, | ||||||
|     { |     { | ||||||
| @@ -48,7 +47,7 @@ export function downSampleLineData< | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   // Convert frames back to points |   // Convert frames back to points | ||||||
|   const result: T[] = []; |   const result: typeof data = []; | ||||||
|   for (const [_i, frame] of frames) { |   for (const [_i, frame] of frames) { | ||||||
|     // Use min/max points to preserve visual accuracy |     // Use min/max points to preserve visual accuracy | ||||||
|     // The order of the data must be preserved so max may be before min |     // The order of the data must be preserved so max may be before min | ||||||
|   | |||||||
| @@ -22,7 +22,7 @@ import { fireEvent } from "../../common/dom/fire_event"; | |||||||
| import { listenMediaQuery } from "../../common/dom/media_query"; | import { listenMediaQuery } from "../../common/dom/media_query"; | ||||||
| import { themesContext } from "../../data/context"; | import { themesContext } from "../../data/context"; | ||||||
| import type { Themes } from "../../data/ws-themes"; | import type { Themes } from "../../data/ws-themes"; | ||||||
| import type { ECOption } from "../../resources/echarts/echarts"; | import type { ECOption } from "../../resources/echarts"; | ||||||
| import type { HomeAssistant } from "../../types"; | import type { HomeAssistant } from "../../types"; | ||||||
| import { isMac } from "../../util/is_mac"; | import { isMac } from "../../util/is_mac"; | ||||||
| import "../chips/ha-assist-chip"; | import "../chips/ha-assist-chip"; | ||||||
| @@ -346,7 +346,7 @@ export class HaChartBase extends LitElement { | |||||||
|       if (this.chart) { |       if (this.chart) { | ||||||
|         this.chart.dispose(); |         this.chart.dispose(); | ||||||
|       } |       } | ||||||
|       const echarts = (await import("../../resources/echarts/echarts")).default; |       const echarts = (await import("../../resources/echarts")).default; | ||||||
|  |  | ||||||
|       if (this.extraComponents?.length) { |       if (this.extraComponents?.length) { | ||||||
|         echarts.use(this.extraComponents); |         echarts.use(this.extraComponents); | ||||||
| @@ -805,7 +805,7 @@ export class HaChartBase extends LitElement { | |||||||
|             sampling: undefined, |             sampling: undefined, | ||||||
|             data: downSampleLineData( |             data: downSampleLineData( | ||||||
|               data as LineSeriesOption["data"], |               data as LineSeriesOption["data"], | ||||||
|               this.clientWidth * window.devicePixelRatio, |               this.clientWidth, | ||||||
|               minX, |               minX, | ||||||
|               maxX |               maxX | ||||||
|             ), |             ), | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ import type { TopLevelFormatterParams } from "echarts/types/dist/shared"; | |||||||
| import { mdiFormatTextVariant, mdiGoogleCirclesGroup } from "@mdi/js"; | import { mdiFormatTextVariant, mdiGoogleCirclesGroup } from "@mdi/js"; | ||||||
| import memoizeOne from "memoize-one"; | import memoizeOne from "memoize-one"; | ||||||
| import { listenMediaQuery } from "../../common/dom/media_query"; | import { listenMediaQuery } from "../../common/dom/media_query"; | ||||||
| import type { ECOption } from "../../resources/echarts/echarts"; | import type { ECOption } from "../../resources/echarts"; | ||||||
| import "./ha-chart-base"; | import "./ha-chart-base"; | ||||||
| import type { HaChartBase } from "./ha-chart-base"; | import type { HaChartBase } from "./ha-chart-base"; | ||||||
| import type { HomeAssistant } from "../../types"; | import type { HomeAssistant } from "../../types"; | ||||||
|   | |||||||
| @@ -1,13 +1,13 @@ | |||||||
| import { customElement, property, state } from "lit/decorators"; | import { customElement, property, state } from "lit/decorators"; | ||||||
| import { LitElement, html, css } from "lit"; | import { LitElement, html, css } from "lit"; | ||||||
| import type { EChartsType } from "echarts/core"; | import type { EChartsType } from "echarts/core"; | ||||||
|  | import type { CallbackDataParams } from "echarts/types/dist/shared"; | ||||||
| import type { SankeySeriesOption } from "echarts/types/dist/echarts"; | import type { SankeySeriesOption } from "echarts/types/dist/echarts"; | ||||||
| import type { CallbackDataParams } from "echarts/types/src/util/types"; | import { SankeyChart } from "echarts/charts"; | ||||||
| import memoizeOne from "memoize-one"; | import memoizeOne from "memoize-one"; | ||||||
| import { ResizeController } from "@lit-labs/observers/resize-controller"; | import { ResizeController } from "@lit-labs/observers/resize-controller"; | ||||||
| import SankeyChart from "../../resources/echarts/components/sankey/install"; |  | ||||||
| import type { HomeAssistant } from "../../types"; | import type { HomeAssistant } from "../../types"; | ||||||
| import type { ECOption } from "../../resources/echarts/echarts"; | import type { ECOption } from "../../resources/echarts"; | ||||||
| import { measureTextWidth } from "../../util/text"; | import { measureTextWidth } from "../../util/text"; | ||||||
| import { filterXSS } from "../../common/util/xss"; | import { filterXSS } from "../../common/util/xss"; | ||||||
| import "./ha-chart-base"; | import "./ha-chart-base"; | ||||||
| @@ -39,7 +39,7 @@ type ProcessedLink = Link & { | |||||||
|  |  | ||||||
| const OVERFLOW_MARGIN = 5; | const OVERFLOW_MARGIN = 5; | ||||||
| const FONT_SIZE = 12; | const FONT_SIZE = 12; | ||||||
| const NODE_GAP = 6; | const NODE_GAP = 8; | ||||||
| const LABEL_DISTANCE = 5; | const LABEL_DISTANCE = 5; | ||||||
|  |  | ||||||
| @customElement("ha-sankey-chart") | @customElement("ha-sankey-chart") | ||||||
| @@ -164,7 +164,6 @@ export class HaSankeyChart extends LitElement { | |||||||
|       lineStyle: { |       lineStyle: { | ||||||
|         color: "gradient", |         color: "gradient", | ||||||
|         opacity: 0.4, |         opacity: 0.4, | ||||||
|         curveness: 0.5, |  | ||||||
|       }, |       }, | ||||||
|       layoutIterations: 0, |       layoutIterations: 0, | ||||||
|       label: { |       label: { | ||||||
|   | |||||||
| @@ -11,7 +11,7 @@ import { computeRTL } from "../../common/util/compute_rtl"; | |||||||
| import type { LineChartEntity, LineChartState } from "../../data/history"; | import type { LineChartEntity, LineChartState } from "../../data/history"; | ||||||
| import type { HomeAssistant } from "../../types"; | import type { HomeAssistant } from "../../types"; | ||||||
| import { MIN_TIME_BETWEEN_UPDATES } from "./ha-chart-base"; | import { MIN_TIME_BETWEEN_UPDATES } from "./ha-chart-base"; | ||||||
| import type { ECOption } from "../../resources/echarts/echarts"; | import type { ECOption } from "../../resources/echarts"; | ||||||
| import { formatDateTimeWithSeconds } from "../../common/datetime/format_date_time"; | import { formatDateTimeWithSeconds } from "../../common/datetime/format_date_time"; | ||||||
| import { | import { | ||||||
|   getNumberFormatOptions, |   getNumberFormatOptions, | ||||||
|   | |||||||
| @@ -15,8 +15,8 @@ import type { TimelineEntity } from "../../data/history"; | |||||||
| import type { HomeAssistant } from "../../types"; | import type { HomeAssistant } from "../../types"; | ||||||
| import { MIN_TIME_BETWEEN_UPDATES } from "./ha-chart-base"; | import { MIN_TIME_BETWEEN_UPDATES } from "./ha-chart-base"; | ||||||
| import { computeTimelineColor } from "./timeline-color"; | import { computeTimelineColor } from "./timeline-color"; | ||||||
| import type { ECOption } from "../../resources/echarts/echarts"; | import type { ECOption } from "../../resources/echarts"; | ||||||
| import echarts from "../../resources/echarts/echarts"; | import echarts from "../../resources/echarts"; | ||||||
| import { luminosity } from "../../common/color/rgb"; | import { luminosity } from "../../common/color/rgb"; | ||||||
| import { hex2rgb } from "../../common/color/convert-color"; | import { hex2rgb } from "../../common/color/convert-color"; | ||||||
| import { measureTextWidth } from "../../util/text"; | import { measureTextWidth } from "../../util/text"; | ||||||
|   | |||||||
| @@ -29,7 +29,7 @@ import { | |||||||
|   getStatisticMetadata, |   getStatisticMetadata, | ||||||
|   statisticsHaveType, |   statisticsHaveType, | ||||||
| } from "../../data/recorder"; | } from "../../data/recorder"; | ||||||
| import type { ECOption } from "../../resources/echarts/echarts"; | import type { ECOption } from "../../resources/echarts"; | ||||||
| import type { HomeAssistant } from "../../types"; | import type { HomeAssistant } from "../../types"; | ||||||
| import type { CustomLegendOption } from "./ha-chart-base"; | import type { CustomLegendOption } from "./ha-chart-base"; | ||||||
| import "./ha-chart-base"; | import "./ha-chart-base"; | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| import { mdiDragHorizontalVariant, mdiEye, mdiEyeOff } from "@mdi/js"; | import { mdiDrag, mdiEye, mdiEyeOff } from "@mdi/js"; | ||||||
| import type { CSSResultGroup } from "lit"; | import type { CSSResultGroup } from "lit"; | ||||||
| import { LitElement, css, html, nothing } from "lit"; | import { LitElement, css, html, nothing } from "lit"; | ||||||
| import { customElement, property, state } from "lit/decorators"; | import { customElement, property, state } from "lit/decorators"; | ||||||
| @@ -129,7 +129,7 @@ export class DialogDataTableSettings extends LitElement { | |||||||
|                   ${canMove && isVisible |                   ${canMove && isVisible | ||||||
|                     ? html`<ha-svg-icon |                     ? html`<ha-svg-icon | ||||||
|                         class="handle" |                         class="handle" | ||||||
|                         .path=${mdiDragHorizontalVariant} |                         .path=${mdiDrag} | ||||||
|                         slot="graphic" |                         slot="graphic" | ||||||
|                       ></ha-svg-icon>` |                       ></ha-svg-icon>` | ||||||
|                     : nothing} |                     : nothing} | ||||||
|   | |||||||
| @@ -5,18 +5,24 @@ import { customElement, property, query, state } from "lit/decorators"; | |||||||
| 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 { computeAreaName } from "../../common/entity/compute_area_name"; | import { computeAreaName } from "../../common/entity/compute_area_name"; | ||||||
| import { computeDeviceName } from "../../common/entity/compute_device_name"; | import { | ||||||
|  |   computeDeviceName, | ||||||
|  |   computeDeviceNameDisplay, | ||||||
|  | } from "../../common/entity/compute_device_name"; | ||||||
|  | import { computeDomain } from "../../common/entity/compute_domain"; | ||||||
| import { getDeviceContext } from "../../common/entity/context/get_device_context"; | import { getDeviceContext } from "../../common/entity/context/get_device_context"; | ||||||
| import { getConfigEntries, type ConfigEntry } from "../../data/config_entries"; | import { getConfigEntries, type ConfigEntry } from "../../data/config_entries"; | ||||||
| import { | import { | ||||||
|   getDevices, |   getDeviceEntityDisplayLookup, | ||||||
|   type DevicePickerItem, |   type DeviceEntityDisplayLookup, | ||||||
|   type DeviceRegistryEntry, |   type DeviceRegistryEntry, | ||||||
| } from "../../data/device_registry"; | } from "../../data/device_registry"; | ||||||
|  | import { domainToName } from "../../data/integration"; | ||||||
| import type { HomeAssistant } from "../../types"; | import type { HomeAssistant } from "../../types"; | ||||||
| import { brandsUrl } from "../../util/brands-url"; | import { brandsUrl } from "../../util/brands-url"; | ||||||
| import "../ha-generic-picker"; | import "../ha-generic-picker"; | ||||||
| import type { HaGenericPicker } from "../ha-generic-picker"; | import type { HaGenericPicker } from "../ha-generic-picker"; | ||||||
|  | import type { PickerComboBoxItem } from "../ha-picker-combo-box"; | ||||||
|  |  | ||||||
| export type HaDevicePickerDeviceFilterFunc = ( | export type HaDevicePickerDeviceFilterFunc = ( | ||||||
|   device: DeviceRegistryEntry |   device: DeviceRegistryEntry | ||||||
| @@ -24,6 +30,11 @@ export type HaDevicePickerDeviceFilterFunc = ( | |||||||
|  |  | ||||||
| export type HaDevicePickerEntityFilterFunc = (entity: HassEntity) => boolean; | export type HaDevicePickerEntityFilterFunc = (entity: HassEntity) => boolean; | ||||||
|  |  | ||||||
|  | interface DevicePickerItem extends PickerComboBoxItem { | ||||||
|  |   domain?: string; | ||||||
|  |   domain_name?: string; | ||||||
|  | } | ||||||
|  |  | ||||||
| @customElement("ha-device-picker") | @customElement("ha-device-picker") | ||||||
| export class HaDevicePicker extends LitElement { | export class HaDevicePicker extends LitElement { | ||||||
|   @property({ attribute: false }) public hass!: HomeAssistant; |   @property({ attribute: false }) public hass!: HomeAssistant; | ||||||
| @@ -93,8 +104,6 @@ export class HaDevicePicker extends LitElement { | |||||||
|  |  | ||||||
|   @state() private _configEntryLookup: Record<string, ConfigEntry> = {}; |   @state() private _configEntryLookup: Record<string, ConfigEntry> = {}; | ||||||
|  |  | ||||||
|   private _getDevicesMemoized = memoizeOne(getDevices); |  | ||||||
|  |  | ||||||
|   protected firstUpdated(_changedProperties: PropertyValues): void { |   protected firstUpdated(_changedProperties: PropertyValues): void { | ||||||
|     super.firstUpdated(_changedProperties); |     super.firstUpdated(_changedProperties); | ||||||
|     this._loadConfigEntries(); |     this._loadConfigEntries(); | ||||||
| @@ -108,18 +117,162 @@ export class HaDevicePicker extends LitElement { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   private _getItems = () => |   private _getItems = () => | ||||||
|     this._getDevicesMemoized( |     this._getDevices( | ||||||
|       this.hass, |       this.hass.devices, | ||||||
|  |       this.hass.entities, | ||||||
|       this._configEntryLookup, |       this._configEntryLookup, | ||||||
|       this.includeDomains, |       this.includeDomains, | ||||||
|       this.excludeDomains, |       this.excludeDomains, | ||||||
|       this.includeDeviceClasses, |       this.includeDeviceClasses, | ||||||
|       this.deviceFilter, |       this.deviceFilter, | ||||||
|       this.entityFilter, |       this.entityFilter, | ||||||
|       this.excludeDevices, |       this.excludeDevices | ||||||
|       this.value |  | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|  |   private _getDevices = memoizeOne( | ||||||
|  |     ( | ||||||
|  |       haDevices: HomeAssistant["devices"], | ||||||
|  |       haEntities: HomeAssistant["entities"], | ||||||
|  |       configEntryLookup: Record<string, ConfigEntry>, | ||||||
|  |       includeDomains: this["includeDomains"], | ||||||
|  |       excludeDomains: this["excludeDomains"], | ||||||
|  |       includeDeviceClasses: this["includeDeviceClasses"], | ||||||
|  |       deviceFilter: this["deviceFilter"], | ||||||
|  |       entityFilter: this["entityFilter"], | ||||||
|  |       excludeDevices: this["excludeDevices"] | ||||||
|  |     ): DevicePickerItem[] => { | ||||||
|  |       const devices = Object.values(haDevices); | ||||||
|  |       const entities = Object.values(haEntities); | ||||||
|  |  | ||||||
|  |       let deviceEntityLookup: DeviceEntityDisplayLookup = {}; | ||||||
|  |  | ||||||
|  |       if ( | ||||||
|  |         includeDomains || | ||||||
|  |         excludeDomains || | ||||||
|  |         includeDeviceClasses || | ||||||
|  |         entityFilter | ||||||
|  |       ) { | ||||||
|  |         deviceEntityLookup = getDeviceEntityDisplayLookup(entities); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       let inputDevices = devices.filter( | ||||||
|  |         (device) => device.id === this.value || !device.disabled_by | ||||||
|  |       ); | ||||||
|  |  | ||||||
|  |       if (includeDomains) { | ||||||
|  |         inputDevices = inputDevices.filter((device) => { | ||||||
|  |           const devEntities = deviceEntityLookup[device.id]; | ||||||
|  |           if (!devEntities || !devEntities.length) { | ||||||
|  |             return false; | ||||||
|  |           } | ||||||
|  |           return deviceEntityLookup[device.id].some((entity) => | ||||||
|  |             includeDomains.includes(computeDomain(entity.entity_id)) | ||||||
|  |           ); | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (excludeDomains) { | ||||||
|  |         inputDevices = inputDevices.filter((device) => { | ||||||
|  |           const devEntities = deviceEntityLookup[device.id]; | ||||||
|  |           if (!devEntities || !devEntities.length) { | ||||||
|  |             return true; | ||||||
|  |           } | ||||||
|  |           return entities.every( | ||||||
|  |             (entity) => | ||||||
|  |               !excludeDomains.includes(computeDomain(entity.entity_id)) | ||||||
|  |           ); | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (excludeDevices) { | ||||||
|  |         inputDevices = inputDevices.filter( | ||||||
|  |           (device) => !excludeDevices!.includes(device.id) | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (includeDeviceClasses) { | ||||||
|  |         inputDevices = inputDevices.filter((device) => { | ||||||
|  |           const devEntities = deviceEntityLookup[device.id]; | ||||||
|  |           if (!devEntities || !devEntities.length) { | ||||||
|  |             return false; | ||||||
|  |           } | ||||||
|  |           return deviceEntityLookup[device.id].some((entity) => { | ||||||
|  |             const stateObj = this.hass.states[entity.entity_id]; | ||||||
|  |             if (!stateObj) { | ||||||
|  |               return false; | ||||||
|  |             } | ||||||
|  |             return ( | ||||||
|  |               stateObj.attributes.device_class && | ||||||
|  |               includeDeviceClasses.includes(stateObj.attributes.device_class) | ||||||
|  |             ); | ||||||
|  |           }); | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (entityFilter) { | ||||||
|  |         inputDevices = inputDevices.filter((device) => { | ||||||
|  |           const devEntities = deviceEntityLookup[device.id]; | ||||||
|  |           if (!devEntities || !devEntities.length) { | ||||||
|  |             return false; | ||||||
|  |           } | ||||||
|  |           return devEntities.some((entity) => { | ||||||
|  |             const stateObj = this.hass.states[entity.entity_id]; | ||||||
|  |             if (!stateObj) { | ||||||
|  |               return false; | ||||||
|  |             } | ||||||
|  |             return entityFilter(stateObj); | ||||||
|  |           }); | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (deviceFilter) { | ||||||
|  |         inputDevices = inputDevices.filter( | ||||||
|  |           (device) => | ||||||
|  |             // We always want to include the device of the current value | ||||||
|  |             device.id === this.value || deviceFilter!(device) | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       const outputDevices = inputDevices.map<DevicePickerItem>((device) => { | ||||||
|  |         const deviceName = computeDeviceNameDisplay( | ||||||
|  |           device, | ||||||
|  |           this.hass, | ||||||
|  |           deviceEntityLookup[device.id] | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |         const { area } = getDeviceContext(device, this.hass); | ||||||
|  |  | ||||||
|  |         const areaName = area ? computeAreaName(area) : undefined; | ||||||
|  |  | ||||||
|  |         const configEntry = device.primary_config_entry | ||||||
|  |           ? configEntryLookup?.[device.primary_config_entry] | ||||||
|  |           : undefined; | ||||||
|  |  | ||||||
|  |         const domain = configEntry?.domain; | ||||||
|  |         const domainName = domain | ||||||
|  |           ? domainToName(this.hass.localize, domain) | ||||||
|  |           : undefined; | ||||||
|  |  | ||||||
|  |         return { | ||||||
|  |           id: device.id, | ||||||
|  |           label: "", | ||||||
|  |           primary: | ||||||
|  |             deviceName || | ||||||
|  |             this.hass.localize("ui.components.device-picker.unnamed_device"), | ||||||
|  |           secondary: areaName, | ||||||
|  |           domain: configEntry?.domain, | ||||||
|  |           domain_name: domainName, | ||||||
|  |           search_labels: [deviceName, areaName, domain, domainName].filter( | ||||||
|  |             Boolean | ||||||
|  |           ) as string[], | ||||||
|  |           sorting_label: deviceName || "zzz", | ||||||
|  |         }; | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       return outputDevices; | ||||||
|  |     } | ||||||
|  |   ); | ||||||
|  |  | ||||||
|   private _valueRenderer = memoizeOne( |   private _valueRenderer = memoizeOne( | ||||||
|     (configEntriesLookup: Record<string, ConfigEntry>) => (value: string) => { |     (configEntriesLookup: Record<string, ConfigEntry>) => (value: string) => { | ||||||
|       const deviceId = value; |       const deviceId = value; | ||||||
|   | |||||||
| @@ -1,13 +1,13 @@ | |||||||
| import { mdiDragHorizontalVariant } from "@mdi/js"; | import { mdiDrag } from "@mdi/js"; | ||||||
| import { css, html, LitElement, nothing } from "lit"; | import { css, html, LitElement, nothing } from "lit"; | ||||||
| import { customElement, property } from "lit/decorators"; | import { customElement, property } from "lit/decorators"; | ||||||
| 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 { isValidEntityId } from "../../common/entity/valid_entity_id"; | import { isValidEntityId } from "../../common/entity/valid_entity_id"; | ||||||
| import type { HaEntityPickerEntityFilterFunc } from "../../data/entity"; |  | ||||||
| import type { HomeAssistant, ValueChangedEvent } from "../../types"; | import type { HomeAssistant, ValueChangedEvent } from "../../types"; | ||||||
| import "../ha-sortable"; | import "../ha-sortable"; | ||||||
| import "./ha-entity-picker"; | import "./ha-entity-picker"; | ||||||
|  | import type { HaEntityPickerEntityFilterFunc } from "./ha-entity-picker"; | ||||||
|  |  | ||||||
| @customElement("ha-entities-picker") | @customElement("ha-entities-picker") | ||||||
| class HaEntitiesPicker extends LitElement { | class HaEntitiesPicker extends LitElement { | ||||||
| @@ -118,7 +118,7 @@ class HaEntitiesPicker extends LitElement { | |||||||
|                   ? html` |                   ? html` | ||||||
|                       <ha-svg-icon |                       <ha-svg-icon | ||||||
|                         class="entity-handle" |                         class="entity-handle" | ||||||
|                         .path=${mdiDragHorizontalVariant} |                         .path=${mdiDrag} | ||||||
|                       ></ha-svg-icon> |                       ></ha-svg-icon> | ||||||
|                     ` |                     ` | ||||||
|                   : nothing} |                   : nothing} | ||||||
|   | |||||||
| @@ -1,3 +1,4 @@ | |||||||
|  | import type { HassEntity } from "home-assistant-js-websocket"; | ||||||
| import type { PropertyValues } from "lit"; | import type { PropertyValues } from "lit"; | ||||||
| import { LitElement, html, nothing } from "lit"; | import { LitElement, html, nothing } from "lit"; | ||||||
| import { customElement, property, query, state } from "lit/decorators"; | import { customElement, property, query, state } from "lit/decorators"; | ||||||
| @@ -7,6 +8,8 @@ import type { HomeAssistant, ValueChangedEvent } from "../../types"; | |||||||
| import "../ha-combo-box"; | import "../ha-combo-box"; | ||||||
| import type { HaComboBox } from "../ha-combo-box"; | import type { HaComboBox } from "../ha-combo-box"; | ||||||
|  |  | ||||||
|  | export type HaEntityPickerEntityFilterFunc = (entityId: HassEntity) => boolean; | ||||||
|  |  | ||||||
| interface AttributeOption { | interface AttributeOption { | ||||||
|   value: string; |   value: string; | ||||||
|   label: string; |   label: string; | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| import "@material/mwc-menu/mwc-menu-surface"; | import "@material/mwc-menu/mwc-menu-surface"; | ||||||
| import { mdiDragHorizontalVariant, mdiPlus } from "@mdi/js"; | import { mdiDrag, mdiPlus } from "@mdi/js"; | ||||||
| import type { ComboBoxLitRenderer } from "@vaadin/combo-box/lit"; | import type { ComboBoxLitRenderer } from "@vaadin/combo-box/lit"; | ||||||
| import type { IFuseOptions } from "fuse.js"; | import type { IFuseOptions } from "fuse.js"; | ||||||
| import Fuse from "fuse.js"; | import Fuse from "fuse.js"; | ||||||
| @@ -25,7 +25,6 @@ import "../ha-sortable"; | |||||||
| interface EntityNameOption { | interface EntityNameOption { | ||||||
|   primary: string; |   primary: string; | ||||||
|   secondary?: string; |   secondary?: string; | ||||||
|   field_label: string; |  | ||||||
|   value: string; |   value: string; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -42,23 +41,6 @@ const KNOWN_TYPES = new Set(["entity", "device", "area", "floor"]); | |||||||
|  |  | ||||||
| const UNIQUE_TYPES = new Set(["entity", "device", "area", "floor"]); | const UNIQUE_TYPES = new Set(["entity", "device", "area", "floor"]); | ||||||
|  |  | ||||||
| const formatOptionValue = (item: EntityNameItem) => { |  | ||||||
|   if (item.type === "text" && item.text) { |  | ||||||
|     return item.text; |  | ||||||
|   } |  | ||||||
|   return `___${item.type}___`; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const parseOptionValue = (value: string): EntityNameItem => { |  | ||||||
|   if (value.startsWith("___") && value.endsWith("___")) { |  | ||||||
|     const type = value.slice(3, -3); |  | ||||||
|     if (KNOWN_TYPES.has(type)) { |  | ||||||
|       return { type: type as EntityNameType }; |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|   return { type: "text", text: value }; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| @customElement("ha-entity-name-picker") | @customElement("ha-entity-name-picker") | ||||||
| export class HaEntityNamePicker extends LitElement { | export class HaEntityNamePicker extends LitElement { | ||||||
|   @property({ attribute: false }) public hass!: HomeAssistant; |   @property({ attribute: false }) public hass!: HomeAssistant; | ||||||
| @@ -86,8 +68,8 @@ export class HaEntityNamePicker extends LitElement { | |||||||
|  |  | ||||||
|   private _editIndex?: number; |   private _editIndex?: number; | ||||||
|  |  | ||||||
|   private _validTypes = memoizeOne((entityId?: string) => { |   private _validOptions = memoizeOne((entityId?: string) => { | ||||||
|     const options = new Set<string>(["text"]); |     const options = new Set<string>(); | ||||||
|     if (!entityId) { |     if (!entityId) { | ||||||
|       return options; |       return options; | ||||||
|     } |     } | ||||||
| @@ -119,43 +101,33 @@ export class HaEntityNamePicker extends LitElement { | |||||||
|       return []; |       return []; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     const types = this._validTypes(entityId); |     const options = this._validOptions(entityId); | ||||||
|  |  | ||||||
|     const items = ( |     const items = ( | ||||||
|       ["entity", "device", "area", "floor"] as const |       ["entity", "device", "area", "floor"] as const | ||||||
|     ).map<EntityNameOption>((name) => { |     ).map<EntityNameOption>((name) => { | ||||||
|       const stateObj = this.hass.states[entityId]; |       const stateObj = this.hass.states[entityId]; | ||||||
|       const isValid = types.has(name); |       const isValid = options.has(name); | ||||||
|       const primary = this.hass.localize( |       const primary = this.hass.localize( | ||||||
|         `ui.components.entity.entity-name-picker.types.${name}` |         `ui.components.entity.entity-name-picker.types.${name}` | ||||||
|       ); |       ); | ||||||
|       const secondary = |       const secondary = | ||||||
|         (stateObj && isValid |         stateObj && isValid | ||||||
|           ? this.hass.formatEntityName(stateObj, { type: name }) |           ? this.hass.formatEntityName(stateObj, { type: name }) | ||||||
|           : this.hass.localize( |           : this.hass.localize( | ||||||
|               `ui.components.entity.entity-name-picker.types.${name}_missing` as LocalizeKeys |               `ui.components.entity.entity-name-picker.types.${name}_missing` as LocalizeKeys | ||||||
|             )) || "-"; |             ) || "-"; | ||||||
|  |  | ||||||
|       return { |       return { | ||||||
|         primary, |         primary, | ||||||
|         secondary, |         secondary, | ||||||
|         field_label: primary, |         value: name, | ||||||
|         value: formatOptionValue({ type: name }), |  | ||||||
|       }; |       }; | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     return items; |     return items; | ||||||
|   }); |   }); | ||||||
|  |  | ||||||
|   private _customNameOption = memoizeOne((text: string) => ({ |  | ||||||
|     primary: this.hass.localize( |  | ||||||
|       "ui.components.entity.entity-name-picker.custom_name" |  | ||||||
|     ), |  | ||||||
|     secondary: `"${text}"`, |  | ||||||
|     field_label: text, |  | ||||||
|     value: formatOptionValue({ type: "text", text }), |  | ||||||
|   })); |  | ||||||
|  |  | ||||||
|   private _formatItem = (item: EntityNameItem) => { |   private _formatItem = (item: EntityNameItem) => { | ||||||
|     if (item.type === "text") { |     if (item.type === "text") { | ||||||
|       return `"${item.text}"`; |       return `"${item.text}"`; | ||||||
| @@ -169,9 +141,9 @@ export class HaEntityNamePicker extends LitElement { | |||||||
|   }; |   }; | ||||||
|  |  | ||||||
|   protected render() { |   protected render() { | ||||||
|     const value = this._items; |     const value = this._value; | ||||||
|     const options = this._getOptions(this.entityId); |     const options = this._getOptions(this.entityId); | ||||||
|     const validTypes = this._validTypes(this.entityId); |     const validOptions = this._validOptions(this.entityId); | ||||||
|  |  | ||||||
|     return html` |     return html` | ||||||
|       ${this.label ? html`<label>${this.label}</label>` : nothing} |       ${this.label ? html`<label>${this.label}</label>` : nothing} | ||||||
| @@ -185,11 +157,12 @@ export class HaEntityNamePicker extends LitElement { | |||||||
|         > |         > | ||||||
|           <ha-chip-set> |           <ha-chip-set> | ||||||
|             ${repeat( |             ${repeat( | ||||||
|               this._items, |               this._value, | ||||||
|               (item) => item, |               (item) => item, | ||||||
|               (item: EntityNameItem, idx) => { |               (item: EntityNameItem, idx) => { | ||||||
|                 const label = this._formatItem(item); |                 const label = this._formatItem(item); | ||||||
|                 const isValid = validTypes.has(item.type); |                 const isValid = | ||||||
|  |                   item.type === "text" || validOptions.has(item.type); | ||||||
|                 return html` |                 return html` | ||||||
|                   <ha-input-chip |                   <ha-input-chip | ||||||
|                     data-idx=${idx} |                     data-idx=${idx} | ||||||
| @@ -200,10 +173,7 @@ export class HaEntityNamePicker extends LitElement { | |||||||
|                     .disabled=${this.disabled} |                     .disabled=${this.disabled} | ||||||
|                     class=${!isValid ? "invalid" : ""} |                     class=${!isValid ? "invalid" : ""} | ||||||
|                   > |                   > | ||||||
|                     <ha-svg-icon |                     <ha-svg-icon slot="icon" .path=${mdiDrag}></ha-svg-icon> | ||||||
|                       slot="icon" |  | ||||||
|                       .path=${mdiDragHorizontalVariant} |  | ||||||
|                     ></ha-svg-icon> |  | ||||||
|                     <span>${label}</span> |                     <span>${label}</span> | ||||||
|                   </ha-input-chip> |                   </ha-input-chip> | ||||||
|                 `; |                 `; | ||||||
| @@ -237,14 +207,14 @@ export class HaEntityNamePicker extends LitElement { | |||||||
|             .hass=${this.hass} |             .hass=${this.hass} | ||||||
|             .value=${""} |             .value=${""} | ||||||
|             .autofocus=${this.autofocus} |             .autofocus=${this.autofocus} | ||||||
|             .disabled=${this.disabled} |             .disabled=${this.disabled || !this.entityId} | ||||||
|             .required=${this.required && !value.length} |             .required=${this.required && !value.length} | ||||||
|             .helper=${this.helper} |             .helper=${this.helper} | ||||||
|             .items=${options} |             .items=${options} | ||||||
|             allow-custom-value |             allow-custom-value | ||||||
|             item-id-path="value" |             item-id-path="value" | ||||||
|             item-value-path="value" |             item-value-path="value" | ||||||
|             item-label-path="field_label" |             item-label-path="primary" | ||||||
|             .renderer=${rowRenderer} |             .renderer=${rowRenderer} | ||||||
|             @opened-changed=${this._openedChanged} |             @opened-changed=${this._openedChanged} | ||||||
|             @value-changed=${this._comboBoxValueChanged} |             @value-changed=${this._comboBoxValueChanged} | ||||||
| @@ -284,16 +254,13 @@ export class HaEntityNamePicker extends LitElement { | |||||||
|     this._opened = true; |     this._opened = true; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private get _items(): EntityNameItem[] { |   private get _value(): EntityNameItem[] { | ||||||
|     return this._toItems(this.value); |     return this._toItems(this.value); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private _toItems = memoizeOne((value?: typeof this.value) => { |   private _toItems = memoizeOne((value?: typeof this.value) => { | ||||||
|     if (typeof value === "string") { |     if (typeof value === "string") { | ||||||
|       if (value === "") { |       return [{ type: "text", text: value } as const]; | ||||||
|         return []; |  | ||||||
|       } |  | ||||||
|       return [{ type: "text", text: value } satisfies EntityNameItem]; |  | ||||||
|     } |     } | ||||||
|     return value ? ensureArray(value) : []; |     return value ? ensureArray(value) : []; | ||||||
|   }); |   }); | ||||||
| @@ -301,7 +268,7 @@ export class HaEntityNamePicker extends LitElement { | |||||||
|   private _toValue = memoizeOne( |   private _toValue = memoizeOne( | ||||||
|     (items: EntityNameItem[]): typeof this.value => { |     (items: EntityNameItem[]): typeof this.value => { | ||||||
|       if (items.length === 0) { |       if (items.length === 0) { | ||||||
|         return ""; |         return []; | ||||||
|       } |       } | ||||||
|       if (items.length === 1) { |       if (items.length === 1) { | ||||||
|         const item = items[0]; |         const item = items[0]; | ||||||
| @@ -317,21 +284,20 @@ export class HaEntityNamePicker extends LitElement { | |||||||
|       const options = this._comboBox.items || []; |       const options = this._comboBox.items || []; | ||||||
|  |  | ||||||
|       const initialItem = |       const initialItem = | ||||||
|         this._editIndex != null ? this._items[this._editIndex] : undefined; |         this._editIndex != null ? this._value[this._editIndex] : undefined; | ||||||
|  |  | ||||||
|       const initialValue = initialItem ? formatOptionValue(initialItem) : ""; |       const initialValue = initialItem | ||||||
|  |         ? initialItem.type === "text" | ||||||
|  |           ? initialItem.text | ||||||
|  |           : initialItem.type | ||||||
|  |         : ""; | ||||||
|  |  | ||||||
|       const filteredItems = this._filterSelectedOptions(options, initialValue); |       const filteredItems = this._filterSelectedOptions(options, initialValue); | ||||||
|  |  | ||||||
|       if (initialItem?.type === "text" && initialItem.text) { |  | ||||||
|         filteredItems.push(this._customNameOption(initialItem.text)); |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       this._comboBox.filteredItems = filteredItems; |       this._comboBox.filteredItems = filteredItems; | ||||||
|       this._comboBox.setInputValue(initialValue); |       this._comboBox.setInputValue(initialValue); | ||||||
|     } else { |     } else { | ||||||
|       this._opened = false; |       this._opened = false; | ||||||
|       this._comboBox.setInputValue(""); |  | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -339,16 +305,15 @@ export class HaEntityNamePicker extends LitElement { | |||||||
|     options: EntityNameOption[], |     options: EntityNameOption[], | ||||||
|     current?: string |     current?: string | ||||||
|   ) => { |   ) => { | ||||||
|     const items = this._items; |     const value = this._value; | ||||||
|  |  | ||||||
|     const excludedValues = new Set( |     const types = value.map((item) => item.type) as string[]; | ||||||
|       items |  | ||||||
|         .filter((item) => UNIQUE_TYPES.has(item.type)) |  | ||||||
|         .map((item) => formatOptionValue(item)) |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     const filteredOptions = options.filter( |     const filteredOptions = options.filter( | ||||||
|       (option) => !excludedValues.has(option.value) || option.value === current |       (option) => | ||||||
|  |         !UNIQUE_TYPES.has(option.value) || | ||||||
|  |         !types.includes(option.value) || | ||||||
|  |         option.value === current | ||||||
|     ); |     ); | ||||||
|     return filteredOptions; |     return filteredOptions; | ||||||
|   }; |   }; | ||||||
| @@ -359,14 +324,20 @@ export class HaEntityNamePicker extends LitElement { | |||||||
|     const options = this._comboBox.items || []; |     const options = this._comboBox.items || []; | ||||||
|  |  | ||||||
|     const currentItem = |     const currentItem = | ||||||
|       this._editIndex != null ? this._items[this._editIndex] : undefined; |       this._editIndex != null ? this._value[this._editIndex] : undefined; | ||||||
|  |  | ||||||
|     const currentValue = currentItem ? formatOptionValue(currentItem) : ""; |     const currentValue = currentItem | ||||||
|  |       ? currentItem.type === "text" | ||||||
|  |         ? currentItem.text | ||||||
|  |         : currentItem.type | ||||||
|  |       : ""; | ||||||
|  |  | ||||||
|     let filteredItems = this._filterSelectedOptions(options, currentValue); |     this._comboBox.filteredItems = this._filterSelectedOptions( | ||||||
|  |       options, | ||||||
|  |       currentValue | ||||||
|  |     ); | ||||||
|  |  | ||||||
|     if (!filter) { |     if (!filter) { | ||||||
|       this._comboBox.filteredItems = filteredItems; |  | ||||||
|       return; |       return; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -378,16 +349,16 @@ export class HaEntityNamePicker extends LitElement { | |||||||
|       ignoreDiacritics: true, |       ignoreDiacritics: true, | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     const fuse = new Fuse(filteredItems, fuseOptions); |     const fuse = new Fuse(this._comboBox.filteredItems, fuseOptions); | ||||||
|     filteredItems = fuse.search(filter).map((result) => result.item); |     const filteredItems = fuse.search(filter).map((result) => result.item); | ||||||
|     filteredItems.push(this._customNameOption(input)); |  | ||||||
|     this._comboBox.filteredItems = filteredItems; |     this._comboBox.filteredItems = filteredItems; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private async _moveItem(ev: CustomEvent) { |   private async _moveItem(ev: CustomEvent) { | ||||||
|     ev.stopPropagation(); |     ev.stopPropagation(); | ||||||
|     const { oldIndex, newIndex } = ev.detail; |     const { oldIndex, newIndex } = ev.detail; | ||||||
|     const value = this._items; |     const value = this._value; | ||||||
|     const newValue = value.concat(); |     const newValue = value.concat(); | ||||||
|     const element = newValue.splice(oldIndex, 1)[0]; |     const element = newValue.splice(oldIndex, 1)[0]; | ||||||
|     newValue.splice(newIndex, 0, element); |     newValue.splice(newIndex, 0, element); | ||||||
| @@ -398,7 +369,7 @@ export class HaEntityNamePicker extends LitElement { | |||||||
|  |  | ||||||
|   private async _removeItem(ev) { |   private async _removeItem(ev) { | ||||||
|     ev.stopPropagation(); |     ev.stopPropagation(); | ||||||
|     const value = [...this._items]; |     const value = [...this._value]; | ||||||
|     const idx = parseInt(ev.target.dataset.idx, 10); |     const idx = parseInt(ev.target.dataset.idx, 10); | ||||||
|     value.splice(idx, 1); |     value.splice(idx, 1); | ||||||
|     this._setValue(value); |     this._setValue(value); | ||||||
| @@ -414,9 +385,11 @@ export class HaEntityNamePicker extends LitElement { | |||||||
|       return; |       return; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     const item: EntityNameItem = parseOptionValue(value); |     const item: EntityNameItem = KNOWN_TYPES.has(value as any) | ||||||
|  |       ? { type: value as EntityNameType } | ||||||
|  |       : { type: "text", text: value }; | ||||||
|  |  | ||||||
|     const newValue = [...this._items]; |     const newValue = [...this._value]; | ||||||
|  |  | ||||||
|     if (this._editIndex != null) { |     if (this._editIndex != null) { | ||||||
|       newValue[this._editIndex] = item; |       newValue[this._editIndex] = item; | ||||||
|   | |||||||
| @@ -1,17 +1,15 @@ | |||||||
| import { mdiPlus, mdiShape } from "@mdi/js"; | import { mdiPlus, mdiShape } from "@mdi/js"; | ||||||
| import type { ComboBoxLitRenderer } from "@vaadin/combo-box/lit"; | import type { ComboBoxLitRenderer } from "@vaadin/combo-box/lit"; | ||||||
|  | import type { HassEntity } from "home-assistant-js-websocket"; | ||||||
| import { html, LitElement, nothing, type PropertyValues } from "lit"; | import { html, LitElement, nothing, type PropertyValues } from "lit"; | ||||||
| import { customElement, property, query } from "lit/decorators"; | import { customElement, property, query } from "lit/decorators"; | ||||||
| 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 { computeEntityNameList } from "../../common/entity/compute_entity_name_display"; | import { computeEntityNameList } from "../../common/entity/compute_entity_name_display"; | ||||||
|  | import { computeStateName } from "../../common/entity/compute_state_name"; | ||||||
| import { isValidEntityId } from "../../common/entity/valid_entity_id"; | import { isValidEntityId } from "../../common/entity/valid_entity_id"; | ||||||
| import { computeRTL } from "../../common/util/compute_rtl"; | import { computeRTL } from "../../common/util/compute_rtl"; | ||||||
| import type { HaEntityPickerEntityFilterFunc } from "../../data/entity"; |  | ||||||
| import { |  | ||||||
|   getEntities, |  | ||||||
|   type EntityComboBoxItem, |  | ||||||
| } from "../../data/entity_registry"; |  | ||||||
| import { domainToName } from "../../data/integration"; | import { domainToName } from "../../data/integration"; | ||||||
| import { | import { | ||||||
|   isHelperDomain, |   isHelperDomain, | ||||||
| @@ -22,11 +20,21 @@ import type { HomeAssistant } from "../../types"; | |||||||
| import "../ha-combo-box-item"; | import "../ha-combo-box-item"; | ||||||
| import "../ha-generic-picker"; | import "../ha-generic-picker"; | ||||||
| import type { HaGenericPicker } from "../ha-generic-picker"; | import type { HaGenericPicker } from "../ha-generic-picker"; | ||||||
| import type { PickerComboBoxSearchFn } from "../ha-picker-combo-box"; | import type { | ||||||
|  |   PickerComboBoxItem, | ||||||
|  |   PickerComboBoxSearchFn, | ||||||
|  | } from "../ha-picker-combo-box"; | ||||||
| import type { PickerValueRenderer } from "../ha-picker-field"; | import type { PickerValueRenderer } from "../ha-picker-field"; | ||||||
| import "../ha-svg-icon"; | import "../ha-svg-icon"; | ||||||
| import "./state-badge"; | import "./state-badge"; | ||||||
|  |  | ||||||
|  | interface EntityComboBoxItem extends PickerComboBoxItem { | ||||||
|  |   domain_name?: string; | ||||||
|  |   stateObj?: HassEntity; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export type HaEntityPickerEntityFilterFunc = (entity: HassEntity) => boolean; | ||||||
|  |  | ||||||
| const CREATE_ID = "___create-new-entity___"; | const CREATE_ID = "___create-new-entity___"; | ||||||
|  |  | ||||||
| @customElement("ha-entity-picker") | @customElement("ha-entity-picker") | ||||||
| @@ -247,10 +255,8 @@ export class HaEntityPicker extends LitElement { | |||||||
|     } |     } | ||||||
|   ); |   ); | ||||||
|  |  | ||||||
|   private _getEntitiesMemoized = memoizeOne(getEntities); |  | ||||||
|  |  | ||||||
|   private _getItems = () => |   private _getItems = () => | ||||||
|     this._getEntitiesMemoized( |     this._getEntities( | ||||||
|       this.hass, |       this.hass, | ||||||
|       this.includeDomains, |       this.includeDomains, | ||||||
|       this.excludeDomains, |       this.excludeDomains, | ||||||
| @@ -258,10 +264,128 @@ export class HaEntityPicker extends LitElement { | |||||||
|       this.includeDeviceClasses, |       this.includeDeviceClasses, | ||||||
|       this.includeUnitOfMeasurement, |       this.includeUnitOfMeasurement, | ||||||
|       this.includeEntities, |       this.includeEntities, | ||||||
|       this.excludeEntities, |       this.excludeEntities | ||||||
|       this.value |  | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|  |   private _getEntities = memoizeOne( | ||||||
|  |     ( | ||||||
|  |       hass: this["hass"], | ||||||
|  |       includeDomains: this["includeDomains"], | ||||||
|  |       excludeDomains: this["excludeDomains"], | ||||||
|  |       entityFilter: this["entityFilter"], | ||||||
|  |       includeDeviceClasses: this["includeDeviceClasses"], | ||||||
|  |       includeUnitOfMeasurement: this["includeUnitOfMeasurement"], | ||||||
|  |       includeEntities: this["includeEntities"], | ||||||
|  |       excludeEntities: this["excludeEntities"] | ||||||
|  |     ): EntityComboBoxItem[] => { | ||||||
|  |       let items: EntityComboBoxItem[] = []; | ||||||
|  |  | ||||||
|  |       let entityIds = Object.keys(hass.states); | ||||||
|  |  | ||||||
|  |       if (includeEntities) { | ||||||
|  |         entityIds = entityIds.filter((entityId) => | ||||||
|  |           includeEntities.includes(entityId) | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (excludeEntities) { | ||||||
|  |         entityIds = entityIds.filter( | ||||||
|  |           (entityId) => !excludeEntities.includes(entityId) | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (includeDomains) { | ||||||
|  |         entityIds = entityIds.filter((eid) => | ||||||
|  |           includeDomains.includes(computeDomain(eid)) | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (excludeDomains) { | ||||||
|  |         entityIds = entityIds.filter( | ||||||
|  |           (eid) => !excludeDomains.includes(computeDomain(eid)) | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       const isRTL = computeRTL(hass); | ||||||
|  |  | ||||||
|  |       items = entityIds.map<EntityComboBoxItem>((entityId) => { | ||||||
|  |         const stateObj = hass.states[entityId]; | ||||||
|  |  | ||||||
|  |         const friendlyName = computeStateName(stateObj); // Keep this for search | ||||||
|  |  | ||||||
|  |         const [entityName, deviceName, areaName] = computeEntityNameList( | ||||||
|  |           stateObj, | ||||||
|  |           [{ type: "entity" }, { type: "device" }, { type: "area" }], | ||||||
|  |           hass.entities, | ||||||
|  |           hass.devices, | ||||||
|  |           hass.areas, | ||||||
|  |           hass.floors | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |         const domainName = domainToName(hass.localize, computeDomain(entityId)); | ||||||
|  |  | ||||||
|  |         const primary = entityName || deviceName || entityId; | ||||||
|  |         const secondary = [areaName, entityName ? deviceName : undefined] | ||||||
|  |           .filter(Boolean) | ||||||
|  |           .join(isRTL ? " ◂ " : " ▸ "); | ||||||
|  |         const a11yLabel = [deviceName, entityName].filter(Boolean).join(" - "); | ||||||
|  |  | ||||||
|  |         return { | ||||||
|  |           id: entityId, | ||||||
|  |           primary: primary, | ||||||
|  |           secondary: secondary, | ||||||
|  |           domain_name: domainName, | ||||||
|  |           sorting_label: [deviceName, entityName].filter(Boolean).join("_"), | ||||||
|  |           search_labels: [ | ||||||
|  |             entityName, | ||||||
|  |             deviceName, | ||||||
|  |             areaName, | ||||||
|  |             domainName, | ||||||
|  |             friendlyName, | ||||||
|  |             entityId, | ||||||
|  |           ].filter(Boolean) as string[], | ||||||
|  |           a11y_label: a11yLabel, | ||||||
|  |           stateObj: stateObj, | ||||||
|  |         }; | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       if (includeDeviceClasses) { | ||||||
|  |         items = items.filter( | ||||||
|  |           (item) => | ||||||
|  |             // We always want to include the entity of the current value | ||||||
|  |             item.id === this.value || | ||||||
|  |             (item.stateObj?.attributes.device_class && | ||||||
|  |               includeDeviceClasses.includes( | ||||||
|  |                 item.stateObj.attributes.device_class | ||||||
|  |               )) | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (includeUnitOfMeasurement) { | ||||||
|  |         items = items.filter( | ||||||
|  |           (item) => | ||||||
|  |             // We always want to include the entity of the current value | ||||||
|  |             item.id === this.value || | ||||||
|  |             (item.stateObj?.attributes.unit_of_measurement && | ||||||
|  |               includeUnitOfMeasurement.includes( | ||||||
|  |                 item.stateObj.attributes.unit_of_measurement | ||||||
|  |               )) | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (entityFilter) { | ||||||
|  |         items = items.filter( | ||||||
|  |           (item) => | ||||||
|  |             // We always want to include the entity of the current value | ||||||
|  |             item.id === this.value || | ||||||
|  |             (item.stateObj && entityFilter!(item.stateObj)) | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       return items; | ||||||
|  |     } | ||||||
|  |   ); | ||||||
|  |  | ||||||
|   protected render() { |   protected render() { | ||||||
|     const placeholder = |     const placeholder = | ||||||
|       this.placeholder ?? |       this.placeholder ?? | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| import { mdiDragHorizontalVariant } from "@mdi/js"; | import { mdiDrag } from "@mdi/js"; | ||||||
| import type { HassEntity } from "home-assistant-js-websocket"; | import type { HassEntity } from "home-assistant-js-websocket"; | ||||||
| import type { PropertyValues } from "lit"; | import type { PropertyValues } from "lit"; | ||||||
| import { LitElement, css, html, nothing } from "lit"; | import { LitElement, css, html, nothing } from "lit"; | ||||||
| @@ -195,10 +195,7 @@ class HaEntityStatePicker extends LitElement { | |||||||
|                         .label=${label} |                         .label=${label} | ||||||
|                         selected |                         selected | ||||||
|                       > |                       > | ||||||
|                         <ha-svg-icon |                         <ha-svg-icon slot="icon" .path=${mdiDrag}></ha-svg-icon> | ||||||
|                           slot="icon" |  | ||||||
|                           .path=${mdiDragHorizontalVariant} |  | ||||||
|                         ></ha-svg-icon> |  | ||||||
|                         ${label} |                         ${label} | ||||||
|                       </ha-input-chip> |                       </ha-input-chip> | ||||||
|                     `; |                     `; | ||||||
|   | |||||||
| @@ -1,3 +1,4 @@ | |||||||
|  | import type { HassEntity } from "home-assistant-js-websocket"; | ||||||
| import type { PropertyValues } from "lit"; | import type { PropertyValues } from "lit"; | ||||||
| import { LitElement, html, nothing } from "lit"; | import { LitElement, html, nothing } from "lit"; | ||||||
| import { customElement, property, query, state } from "lit/decorators"; | import { customElement, property, query, state } from "lit/decorators"; | ||||||
| @@ -8,6 +9,8 @@ import type { HomeAssistant, ValueChangedEvent } from "../../types"; | |||||||
| import "../ha-combo-box"; | import "../ha-combo-box"; | ||||||
| import type { HaComboBox } from "../ha-combo-box"; | import type { HaComboBox } from "../ha-combo-box"; | ||||||
|  |  | ||||||
|  | export type HaEntityPickerEntityFilterFunc = (entityId: HassEntity) => boolean; | ||||||
|  |  | ||||||
| interface StateOption { | interface StateOption { | ||||||
|   value: string; |   value: string; | ||||||
|   label: string; |   label: string; | ||||||
|   | |||||||
| @@ -21,7 +21,6 @@ import "../ha-combo-box-item"; | |||||||
| import "../ha-generic-picker"; | import "../ha-generic-picker"; | ||||||
| import type { HaGenericPicker } from "../ha-generic-picker"; | import type { HaGenericPicker } from "../ha-generic-picker"; | ||||||
| import "../ha-icon-button"; | import "../ha-icon-button"; | ||||||
| import "../ha-input-helper-text"; |  | ||||||
| import type { | import type { | ||||||
|   PickerComboBoxItem, |   PickerComboBoxItem, | ||||||
|   PickerComboBoxSearchFn, |   PickerComboBoxSearchFn, | ||||||
| @@ -477,6 +476,7 @@ export class HaStatisticPicker extends LitElement { | |||||||
|         .hideClearIcon=${this.hideClearIcon} |         .hideClearIcon=${this.hideClearIcon} | ||||||
|         .searchFn=${this._searchFn} |         .searchFn=${this._searchFn} | ||||||
|         .valueRenderer=${this._valueRenderer} |         .valueRenderer=${this._valueRenderer} | ||||||
|  |         .helper=${this.helper} | ||||||
|         @value-changed=${this._valueChanged} |         @value-changed=${this._valueChanged} | ||||||
|       > |       > | ||||||
|       </ha-generic-picker> |       </ha-generic-picker> | ||||||
|   | |||||||
| @@ -8,13 +8,21 @@ 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 { computeAreaName } from "../common/entity/compute_area_name"; | import { computeAreaName } from "../common/entity/compute_area_name"; | ||||||
|  | import { computeDomain } from "../common/entity/compute_domain"; | ||||||
| import { computeFloorName } from "../common/entity/compute_floor_name"; | import { computeFloorName } from "../common/entity/compute_floor_name"; | ||||||
|  | import { stringCompare } from "../common/string/compare"; | ||||||
| import { computeRTL } from "../common/util/compute_rtl"; | import { computeRTL } from "../common/util/compute_rtl"; | ||||||
|  | import type { AreaRegistryEntry } from "../data/area_registry"; | ||||||
|  | import type { | ||||||
|  |   DeviceEntityDisplayLookup, | ||||||
|  |   DeviceRegistryEntry, | ||||||
|  | } from "../data/device_registry"; | ||||||
|  | import { getDeviceEntityDisplayLookup } from "../data/device_registry"; | ||||||
|  | import type { EntityRegistryDisplayEntry } from "../data/entity_registry"; | ||||||
| import { | import { | ||||||
|   getAreasAndFloors, |   getFloorAreaLookup, | ||||||
|   type AreaFloorValue, |   type FloorRegistryEntry, | ||||||
|   type FloorComboBoxItem, | } from "../data/floor_registry"; | ||||||
| } from "../data/area_floor"; |  | ||||||
| import type { HomeAssistant, ValueChangedEvent } from "../types"; | import type { HomeAssistant, ValueChangedEvent } from "../types"; | ||||||
| import type { HaDevicePickerDeviceFilterFunc } from "./device/ha-device-picker"; | import type { HaDevicePickerDeviceFilterFunc } from "./device/ha-device-picker"; | ||||||
| import "./ha-combo-box-item"; | import "./ha-combo-box-item"; | ||||||
| @@ -22,12 +30,24 @@ import "./ha-floor-icon"; | |||||||
| import "./ha-generic-picker"; | import "./ha-generic-picker"; | ||||||
| import type { HaGenericPicker } from "./ha-generic-picker"; | import type { HaGenericPicker } from "./ha-generic-picker"; | ||||||
| import "./ha-icon-button"; | import "./ha-icon-button"; | ||||||
|  | import type { PickerComboBoxItem } from "./ha-picker-combo-box"; | ||||||
| import type { PickerValueRenderer } from "./ha-picker-field"; | import type { PickerValueRenderer } from "./ha-picker-field"; | ||||||
| import "./ha-svg-icon"; | import "./ha-svg-icon"; | ||||||
| import "./ha-tree-indicator"; | import "./ha-tree-indicator"; | ||||||
|  |  | ||||||
| const SEPARATOR = "________"; | const SEPARATOR = "________"; | ||||||
|  |  | ||||||
|  | interface FloorComboBoxItem extends PickerComboBoxItem { | ||||||
|  |   type: "floor" | "area"; | ||||||
|  |   floor?: FloorRegistryEntry; | ||||||
|  |   area?: AreaRegistryEntry; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | interface AreaFloorValue { | ||||||
|  |   id: string; | ||||||
|  |   type: "floor" | "area"; | ||||||
|  | } | ||||||
|  |  | ||||||
| @customElement("ha-area-floor-picker") | @customElement("ha-area-floor-picker") | ||||||
| export class HaAreaFloorPicker extends LitElement { | export class HaAreaFloorPicker extends LitElement { | ||||||
|   @property({ attribute: false }) public hass!: HomeAssistant; |   @property({ attribute: false }) public hass!: HomeAssistant; | ||||||
| @@ -134,6 +154,243 @@ export class HaAreaFloorPicker extends LitElement { | |||||||
|     `; |     `; | ||||||
|   }; |   }; | ||||||
|  |  | ||||||
|  |   private _getAreasAndFloors = memoizeOne( | ||||||
|  |     ( | ||||||
|  |       haFloors: HomeAssistant["floors"], | ||||||
|  |       haAreas: HomeAssistant["areas"], | ||||||
|  |       haDevices: HomeAssistant["devices"], | ||||||
|  |       haEntities: HomeAssistant["entities"], | ||||||
|  |       includeDomains: this["includeDomains"], | ||||||
|  |       excludeDomains: this["excludeDomains"], | ||||||
|  |       includeDeviceClasses: this["includeDeviceClasses"], | ||||||
|  |       deviceFilter: this["deviceFilter"], | ||||||
|  |       entityFilter: this["entityFilter"], | ||||||
|  |       excludeAreas: this["excludeAreas"], | ||||||
|  |       excludeFloors: this["excludeFloors"] | ||||||
|  |     ): FloorComboBoxItem[] => { | ||||||
|  |       const floors = Object.values(haFloors); | ||||||
|  |       const areas = Object.values(haAreas); | ||||||
|  |       const devices = Object.values(haDevices); | ||||||
|  |       const entities = Object.values(haEntities); | ||||||
|  |  | ||||||
|  |       let deviceEntityLookup: DeviceEntityDisplayLookup = {}; | ||||||
|  |       let inputDevices: DeviceRegistryEntry[] | undefined; | ||||||
|  |       let inputEntities: EntityRegistryDisplayEntry[] | undefined; | ||||||
|  |  | ||||||
|  |       if ( | ||||||
|  |         includeDomains || | ||||||
|  |         excludeDomains || | ||||||
|  |         includeDeviceClasses || | ||||||
|  |         deviceFilter || | ||||||
|  |         entityFilter | ||||||
|  |       ) { | ||||||
|  |         deviceEntityLookup = getDeviceEntityDisplayLookup(entities); | ||||||
|  |         inputDevices = devices; | ||||||
|  |         inputEntities = entities.filter((entity) => entity.area_id); | ||||||
|  |  | ||||||
|  |         if (includeDomains) { | ||||||
|  |           inputDevices = inputDevices!.filter((device) => { | ||||||
|  |             const devEntities = deviceEntityLookup[device.id]; | ||||||
|  |             if (!devEntities || !devEntities.length) { | ||||||
|  |               return false; | ||||||
|  |             } | ||||||
|  |             return deviceEntityLookup[device.id].some((entity) => | ||||||
|  |               includeDomains.includes(computeDomain(entity.entity_id)) | ||||||
|  |             ); | ||||||
|  |           }); | ||||||
|  |           inputEntities = inputEntities!.filter((entity) => | ||||||
|  |             includeDomains.includes(computeDomain(entity.entity_id)) | ||||||
|  |           ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (excludeDomains) { | ||||||
|  |           inputDevices = inputDevices!.filter((device) => { | ||||||
|  |             const devEntities = deviceEntityLookup[device.id]; | ||||||
|  |             if (!devEntities || !devEntities.length) { | ||||||
|  |               return true; | ||||||
|  |             } | ||||||
|  |             return entities.every( | ||||||
|  |               (entity) => | ||||||
|  |                 !excludeDomains.includes(computeDomain(entity.entity_id)) | ||||||
|  |             ); | ||||||
|  |           }); | ||||||
|  |           inputEntities = inputEntities!.filter( | ||||||
|  |             (entity) => | ||||||
|  |               !excludeDomains.includes(computeDomain(entity.entity_id)) | ||||||
|  |           ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (includeDeviceClasses) { | ||||||
|  |           inputDevices = inputDevices!.filter((device) => { | ||||||
|  |             const devEntities = deviceEntityLookup[device.id]; | ||||||
|  |             if (!devEntities || !devEntities.length) { | ||||||
|  |               return false; | ||||||
|  |             } | ||||||
|  |             return deviceEntityLookup[device.id].some((entity) => { | ||||||
|  |               const stateObj = this.hass.states[entity.entity_id]; | ||||||
|  |               if (!stateObj) { | ||||||
|  |                 return false; | ||||||
|  |               } | ||||||
|  |               return ( | ||||||
|  |                 stateObj.attributes.device_class && | ||||||
|  |                 includeDeviceClasses.includes(stateObj.attributes.device_class) | ||||||
|  |               ); | ||||||
|  |             }); | ||||||
|  |           }); | ||||||
|  |           inputEntities = inputEntities!.filter((entity) => { | ||||||
|  |             const stateObj = this.hass.states[entity.entity_id]; | ||||||
|  |             return ( | ||||||
|  |               stateObj.attributes.device_class && | ||||||
|  |               includeDeviceClasses.includes(stateObj.attributes.device_class) | ||||||
|  |             ); | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (deviceFilter) { | ||||||
|  |           inputDevices = inputDevices!.filter((device) => | ||||||
|  |             deviceFilter!(device) | ||||||
|  |           ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (entityFilter) { | ||||||
|  |           inputDevices = inputDevices!.filter((device) => { | ||||||
|  |             const devEntities = deviceEntityLookup[device.id]; | ||||||
|  |             if (!devEntities || !devEntities.length) { | ||||||
|  |               return false; | ||||||
|  |             } | ||||||
|  |             return deviceEntityLookup[device.id].some((entity) => { | ||||||
|  |               const stateObj = this.hass.states[entity.entity_id]; | ||||||
|  |               if (!stateObj) { | ||||||
|  |                 return false; | ||||||
|  |               } | ||||||
|  |               return entityFilter(stateObj); | ||||||
|  |             }); | ||||||
|  |           }); | ||||||
|  |           inputEntities = inputEntities!.filter((entity) => { | ||||||
|  |             const stateObj = this.hass.states[entity.entity_id]; | ||||||
|  |             if (!stateObj) { | ||||||
|  |               return false; | ||||||
|  |             } | ||||||
|  |             return entityFilter!(stateObj); | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       let outputAreas = areas; | ||||||
|  |  | ||||||
|  |       let areaIds: string[] | undefined; | ||||||
|  |  | ||||||
|  |       if (inputDevices) { | ||||||
|  |         areaIds = inputDevices | ||||||
|  |           .filter((device) => device.area_id) | ||||||
|  |           .map((device) => device.area_id!); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (inputEntities) { | ||||||
|  |         areaIds = (areaIds ?? []).concat( | ||||||
|  |           inputEntities | ||||||
|  |             .filter((entity) => entity.area_id) | ||||||
|  |             .map((entity) => entity.area_id!) | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (areaIds) { | ||||||
|  |         outputAreas = outputAreas.filter((area) => | ||||||
|  |           areaIds!.includes(area.area_id) | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (excludeAreas) { | ||||||
|  |         outputAreas = outputAreas.filter( | ||||||
|  |           (area) => !excludeAreas!.includes(area.area_id) | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (excludeFloors) { | ||||||
|  |         outputAreas = outputAreas.filter( | ||||||
|  |           (area) => !area.floor_id || !excludeFloors!.includes(area.floor_id) | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       const floorAreaLookup = getFloorAreaLookup(outputAreas); | ||||||
|  |       const unassisgnedAreas = Object.values(outputAreas).filter( | ||||||
|  |         (area) => !area.floor_id || !floorAreaLookup[area.floor_id] | ||||||
|  |       ); | ||||||
|  |  | ||||||
|  |       // @ts-ignore | ||||||
|  |       const floorAreaEntries: [ | ||||||
|  |         FloorRegistryEntry | undefined, | ||||||
|  |         AreaRegistryEntry[], | ||||||
|  |       ][] = Object.entries(floorAreaLookup) | ||||||
|  |         .map(([floorId, floorAreas]) => { | ||||||
|  |           const floor = floors.find((fl) => fl.floor_id === floorId)!; | ||||||
|  |           return [floor, floorAreas] as const; | ||||||
|  |         }) | ||||||
|  |         .sort(([floorA], [floorB]) => { | ||||||
|  |           if (floorA.level !== floorB.level) { | ||||||
|  |             return (floorA.level ?? 0) - (floorB.level ?? 0); | ||||||
|  |           } | ||||||
|  |           return stringCompare(floorA.name, floorB.name); | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |       const items: FloorComboBoxItem[] = []; | ||||||
|  |  | ||||||
|  |       floorAreaEntries.forEach(([floor, floorAreas]) => { | ||||||
|  |         if (floor) { | ||||||
|  |           const floorName = computeFloorName(floor); | ||||||
|  |  | ||||||
|  |           const areaSearchLabels = floorAreas | ||||||
|  |             .map((area) => { | ||||||
|  |               const areaName = computeAreaName(area) || area.area_id; | ||||||
|  |               return [area.area_id, areaName, ...area.aliases]; | ||||||
|  |             }) | ||||||
|  |             .flat(); | ||||||
|  |  | ||||||
|  |           items.push({ | ||||||
|  |             id: this._formatValue({ id: floor.floor_id, type: "floor" }), | ||||||
|  |             type: "floor", | ||||||
|  |             primary: floorName, | ||||||
|  |             floor: floor, | ||||||
|  |             search_labels: [ | ||||||
|  |               floor.floor_id, | ||||||
|  |               floorName, | ||||||
|  |               ...floor.aliases, | ||||||
|  |               ...areaSearchLabels, | ||||||
|  |             ], | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |         items.push( | ||||||
|  |           ...floorAreas.map((area) => { | ||||||
|  |             const areaName = computeAreaName(area) || area.area_id; | ||||||
|  |             return { | ||||||
|  |               id: this._formatValue({ id: area.area_id, type: "area" }), | ||||||
|  |               type: "area" as const, | ||||||
|  |               primary: areaName, | ||||||
|  |               area: area, | ||||||
|  |               icon: area.icon || undefined, | ||||||
|  |               search_labels: [area.area_id, areaName, ...area.aliases], | ||||||
|  |             }; | ||||||
|  |           }) | ||||||
|  |         ); | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       items.push( | ||||||
|  |         ...unassisgnedAreas.map((area) => { | ||||||
|  |           const areaName = computeAreaName(area) || area.area_id; | ||||||
|  |           return { | ||||||
|  |             id: this._formatValue({ id: area.area_id, type: "area" }), | ||||||
|  |             type: "area" as const, | ||||||
|  |             primary: areaName, | ||||||
|  |             icon: area.icon || undefined, | ||||||
|  |             search_labels: [area.area_id, areaName, ...area.aliases], | ||||||
|  |           }; | ||||||
|  |         }) | ||||||
|  |       ); | ||||||
|  |  | ||||||
|  |       return items; | ||||||
|  |     } | ||||||
|  |   ); | ||||||
|  |  | ||||||
|   private _rowRenderer: ComboBoxLitRenderer<FloorComboBoxItem> = ( |   private _rowRenderer: ComboBoxLitRenderer<FloorComboBoxItem> = ( | ||||||
|     item, |     item, | ||||||
|     { index }, |     { index }, | ||||||
| @@ -188,16 +445,12 @@ export class HaAreaFloorPicker extends LitElement { | |||||||
|     `; |     `; | ||||||
|   }; |   }; | ||||||
|  |  | ||||||
|   private _getAreasAndFloorsMemoized = memoizeOne(getAreasAndFloors); |  | ||||||
|  |  | ||||||
|   private _getItems = () => |   private _getItems = () => | ||||||
|     this._getAreasAndFloorsMemoized( |     this._getAreasAndFloors( | ||||||
|       this.hass.states, |  | ||||||
|       this.hass.floors, |       this.hass.floors, | ||||||
|       this.hass.areas, |       this.hass.areas, | ||||||
|       this.hass.devices, |       this.hass.devices, | ||||||
|       this.hass.entities, |       this.hass.entities, | ||||||
|       this._formatValue, |  | ||||||
|       this.includeDomains, |       this.includeDomains, | ||||||
|       this.excludeDomains, |       this.excludeDomains, | ||||||
|       this.includeDeviceClasses, |       this.includeDeviceClasses, | ||||||
|   | |||||||
| @@ -107,7 +107,7 @@ export class HaAreaPicker extends LitElement { | |||||||
|           `; |           `; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         const { floor } = getAreaContext(area, this.hass.floors); |         const { floor } = getAreaContext(area, this.hass); | ||||||
|  |  | ||||||
|         const areaName = area ? computeAreaName(area) : undefined; |         const areaName = area ? computeAreaName(area) : undefined; | ||||||
|         const floorName = floor ? computeFloorName(floor) : undefined; |         const floorName = floor ? computeFloorName(floor) : undefined; | ||||||
| @@ -279,7 +279,7 @@ export class HaAreaPicker extends LitElement { | |||||||
|       } |       } | ||||||
|  |  | ||||||
|       const items = outputAreas.map<PickerComboBoxItem>((area) => { |       const items = outputAreas.map<PickerComboBoxItem>((area) => { | ||||||
|         const { floor } = getAreaContext(area, this.hass.floors); |         const { floor } = getAreaContext(area, this.hass); | ||||||
|         const floorName = floor ? computeFloorName(floor) : undefined; |         const floorName = floor ? computeFloorName(floor) : undefined; | ||||||
|         const areaName = computeAreaName(area); |         const areaName = computeAreaName(area); | ||||||
|         return { |         return { | ||||||
|   | |||||||
| @@ -44,7 +44,7 @@ export class HaAreasDisplayEditor extends LitElement { | |||||||
|     ); |     ); | ||||||
|  |  | ||||||
|     const items: DisplayItem[] = areas.map((area) => { |     const items: DisplayItem[] = areas.map((area) => { | ||||||
|       const { floor } = getAreaContext(area, this.hass.floors); |       const { floor } = getAreaContext(area, this.hass!); | ||||||
|       return { |       return { | ||||||
|         value: area.area_id, |         value: area.area_id, | ||||||
|         label: area.name, |         label: area.name, | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| import { mdiDragHorizontalVariant, mdiTextureBox } from "@mdi/js"; | import { mdiDrag, mdiTextureBox } from "@mdi/js"; | ||||||
| import type { TemplateResult } from "lit"; | import type { TemplateResult } from "lit"; | ||||||
| import { LitElement, css, html, nothing } from "lit"; | import { LitElement, css, html, nothing } from "lit"; | ||||||
| import { customElement, property } from "lit/decorators"; | import { customElement, property } from "lit/decorators"; | ||||||
| @@ -105,7 +105,7 @@ export class HaAreasFloorsDisplayEditor extends LitElement { | |||||||
|                       <ha-svg-icon |                       <ha-svg-icon | ||||||
|                         class="handle" |                         class="handle" | ||||||
|                         slot="icons" |                         slot="icons" | ||||||
|                         .path=${mdiDragHorizontalVariant} |                         .path=${mdiDrag} | ||||||
|                       ></ha-svg-icon> |                       ></ha-svg-icon> | ||||||
|                     `} |                     `} | ||||||
|                 <ha-items-display-editor |                 <ha-items-display-editor | ||||||
| @@ -138,7 +138,7 @@ export class HaAreasFloorsDisplayEditor extends LitElement { | |||||||
|       ); |       ); | ||||||
|       const groupedItems: Record<string, DisplayItem[]> = areas.reduce( |       const groupedItems: Record<string, DisplayItem[]> = areas.reduce( | ||||||
|         (acc, area) => { |         (acc, area) => { | ||||||
|           const { floor } = getAreaContext(area, this.hass.floors); |           const { floor } = getAreaContext(area, this.hass!); | ||||||
|           const floorId = floor?.floor_id ?? UNASSIGNED_FLOOR; |           const floorId = floor?.floor_id ?? UNASSIGNED_FLOOR; | ||||||
|  |  | ||||||
|           if (!acc[floorId]) { |           if (!acc[floorId]) { | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| import "@home-assistant/webawesome/dist/components/drawer/drawer"; |  | ||||||
| import { css, html, LitElement, type PropertyValues } from "lit"; | import { css, html, LitElement, type PropertyValues } from "lit"; | ||||||
|  | import "@home-assistant/webawesome/dist/components/drawer/drawer"; | ||||||
| import { customElement, property, state } from "lit/decorators"; | import { customElement, property, state } from "lit/decorators"; | ||||||
|  |  | ||||||
| export const BOTTOM_SHEET_ANIMATION_DURATION_MS = 300; | export const BOTTOM_SHEET_ANIMATION_DURATION_MS = 300; | ||||||
| @@ -8,9 +8,6 @@ export const BOTTOM_SHEET_ANIMATION_DURATION_MS = 300; | |||||||
| export class HaBottomSheet extends LitElement { | export class HaBottomSheet extends LitElement { | ||||||
|   @property({ type: Boolean }) public open = false; |   @property({ type: Boolean }) public open = false; | ||||||
|  |  | ||||||
|   @property({ type: Boolean, reflect: true, attribute: "flexcontent" }) |  | ||||||
|   public flexContent = false; |  | ||||||
|  |  | ||||||
|   @state() private _drawerOpen = false; |   @state() private _drawerOpen = false; | ||||||
|  |  | ||||||
|   private _handleAfterHide() { |   private _handleAfterHide() { | ||||||
| @@ -44,19 +41,16 @@ export class HaBottomSheet extends LitElement { | |||||||
|  |  | ||||||
|   static styles = css` |   static styles = css` | ||||||
|     wa-drawer { |     wa-drawer { | ||||||
|       --wa-color-surface-raised: transparent; |       --wa-color-surface-raised: var( | ||||||
|  |         --ha-bottom-sheet-surface-background, | ||||||
|  |         var(--ha-dialog-surface-background, var(--mdc-theme-surface, #fff)), | ||||||
|  |       ); | ||||||
|       --spacing: 0; |       --spacing: 0; | ||||||
|       --size: var(--ha-bottom-sheet-height, auto); |       --size: auto; | ||||||
|       --show-duration: ${BOTTOM_SHEET_ANIMATION_DURATION_MS}ms; |       --show-duration: ${BOTTOM_SHEET_ANIMATION_DURATION_MS}ms; | ||||||
|       --hide-duration: ${BOTTOM_SHEET_ANIMATION_DURATION_MS}ms; |       --hide-duration: ${BOTTOM_SHEET_ANIMATION_DURATION_MS}ms; | ||||||
|     } |     } | ||||||
|     wa-drawer::part(dialog) { |     wa-drawer::part(dialog) { | ||||||
|       max-height: var(--ha-bottom-sheet-max-height, 90vh); |  | ||||||
|       align-items: center; |  | ||||||
|     } |  | ||||||
|     wa-drawer::part(body) { |  | ||||||
|       max-width: var(--ha-bottom-sheet-max-width); |  | ||||||
|       width: 100%; |  | ||||||
|       border-top-left-radius: var( |       border-top-left-radius: var( | ||||||
|         --ha-bottom-sheet-border-radius, |         --ha-bottom-sheet-border-radius, | ||||||
|         var(--ha-dialog-border-radius, var(--ha-border-radius-2xl)) |         var(--ha-dialog-border-radius, var(--ha-border-radius-2xl)) | ||||||
| @@ -65,19 +59,10 @@ export class HaBottomSheet extends LitElement { | |||||||
|         --ha-bottom-sheet-border-radius, |         --ha-bottom-sheet-border-radius, | ||||||
|         var(--ha-dialog-border-radius, var(--ha-border-radius-2xl)) |         var(--ha-dialog-border-radius, var(--ha-border-radius-2xl)) | ||||||
|       ); |       ); | ||||||
|       background-color: var( |       max-height: 90vh; | ||||||
|         --ha-bottom-sheet-surface-background, |       padding-bottom: var(--safe-area-inset-bottom); | ||||||
|         var(--ha-dialog-surface-background, var(--mdc-theme-surface, #fff)), |       padding-left: var(--safe-area-inset-left); | ||||||
|       ); |       padding-right: var(--safe-area-inset-right); | ||||||
|       padding: var( |  | ||||||
|         --ha-bottom-sheet-padding, |  | ||||||
|         0 var(--safe-area-inset-right) var(--safe-area-inset-bottom) |  | ||||||
|           var(--safe-area-inset-left) |  | ||||||
|       ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     :host([flexcontent]) wa-drawer::part(body) { |  | ||||||
|       display: flex; |  | ||||||
|     } |     } | ||||||
|   `; |   `; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -86,8 +86,7 @@ export class HaCameraStream extends LitElement { | |||||||
|     const streams = this._streams( |     const streams = this._streams( | ||||||
|       this._capabilities?.frontend_stream_types, |       this._capabilities?.frontend_stream_types, | ||||||
|       this._hlsStreams, |       this._hlsStreams, | ||||||
|       this._webRtcStreams, |       this._webRtcStreams | ||||||
|       this.muted |  | ||||||
|     ); |     ); | ||||||
|     return html`${repeat( |     return html`${repeat( | ||||||
|       streams, |       streams, | ||||||
| @@ -191,8 +190,7 @@ export class HaCameraStream extends LitElement { | |||||||
|     ( |     ( | ||||||
|       supportedTypes?: StreamType[], |       supportedTypes?: StreamType[], | ||||||
|       hlsStreams?: { hasAudio: boolean; hasVideo: boolean }, |       hlsStreams?: { hasAudio: boolean; hasVideo: boolean }, | ||||||
|       webRtcStreams?: { hasAudio: boolean; hasVideo: boolean }, |       webRtcStreams?: { hasAudio: boolean; hasVideo: boolean } | ||||||
|       muted?: boolean |  | ||||||
|     ): Stream[] => { |     ): Stream[] => { | ||||||
|       if (__DEMO__) { |       if (__DEMO__) { | ||||||
|         return [{ type: MJPEG_STREAM, visible: true }]; |         return [{ type: MJPEG_STREAM, visible: true }]; | ||||||
| @@ -222,10 +220,9 @@ export class HaCameraStream extends LitElement { | |||||||
|         if ( |         if ( | ||||||
|           hlsStreams.hasVideo && |           hlsStreams.hasVideo && | ||||||
|           hlsStreams.hasAudio && |           hlsStreams.hasAudio && | ||||||
|           !webRtcStreams.hasAudio && |           !webRtcStreams.hasAudio | ||||||
|           !muted |  | ||||||
|         ) { |         ) { | ||||||
|           // webRTC stream is missing audio and audio is not muted, use HLS |           // webRTC stream is missing audio, use HLS | ||||||
|           return [{ type: STREAM_TYPE_HLS, visible: true }]; |           return [{ type: STREAM_TYPE_HLS, visible: true }]; | ||||||
|         } |         } | ||||||
|         if (webRtcStreams.hasVideo) { |         if (webRtcStreams.hasVideo) { | ||||||
|   | |||||||
| @@ -49,16 +49,12 @@ export class HaDialogHeader extends LitElement { | |||||||
|           display: flex; |           display: flex; | ||||||
|           flex-direction: row; |           flex-direction: row; | ||||||
|           align-items: center; |           align-items: center; | ||||||
|           padding: 0 var(--ha-space-1); |           padding: 4px; | ||||||
|           box-sizing: border-box; |           box-sizing: border-box; | ||||||
|         } |         } | ||||||
|         .header-content { |         .header-content { | ||||||
|           flex: 1; |           flex: 1; | ||||||
|           padding: 10px var(--ha-space-1); |           padding: 10px 4px; | ||||||
|           display: flex; |  | ||||||
|           flex-direction: column; |  | ||||||
|           justify-content: center; |  | ||||||
|           min-height: var(--ha-space-12); |  | ||||||
|           min-width: 0; |           min-width: 0; | ||||||
|           overflow: hidden; |           overflow: hidden; | ||||||
|           text-overflow: ellipsis; |           text-overflow: ellipsis; | ||||||
| @@ -67,7 +63,7 @@ export class HaDialogHeader extends LitElement { | |||||||
|         .header-title { |         .header-title { | ||||||
|           height: var( |           height: var( | ||||||
|             --ha-dialog-header-title-height, |             --ha-dialog-header-title-height, | ||||||
|             calc(var(--ha-font-size-xl) + var(--ha-space-1)) |             calc(var(--ha-font-size-xl) + 4px) | ||||||
|           ); |           ); | ||||||
|           font-size: var(--ha-font-size-xl); |           font-size: var(--ha-font-size-xl); | ||||||
|           line-height: var(--ha-line-height-condensed); |           line-height: var(--ha-line-height-condensed); | ||||||
| @@ -80,19 +76,19 @@ export class HaDialogHeader extends LitElement { | |||||||
|         } |         } | ||||||
|         @media all and (min-width: 450px) and (min-height: 500px) { |         @media all and (min-width: 450px) and (min-height: 500px) { | ||||||
|           .header-bar { |           .header-bar { | ||||||
|             padding: 0 var(--ha-space-2); |             padding: 16px; | ||||||
|           } |           } | ||||||
|         } |         } | ||||||
|         .header-navigation-icon { |         .header-navigation-icon { | ||||||
|           flex: none; |           flex: none; | ||||||
|           min-width: var(--ha-space-2); |           min-width: 8px; | ||||||
|           height: 100%; |           height: 100%; | ||||||
|           display: flex; |           display: flex; | ||||||
|           flex-direction: row; |           flex-direction: row; | ||||||
|         } |         } | ||||||
|         .header-action-items { |         .header-action-items { | ||||||
|           flex: none; |           flex: none; | ||||||
|           min-width: var(--ha-space-2); |           min-width: 8px; | ||||||
|           height: 100%; |           height: 100%; | ||||||
|           display: flex; |           display: flex; | ||||||
|           flex-direction: row; |           flex-direction: row; | ||||||
|   | |||||||
| @@ -49,7 +49,6 @@ export class HaExpansionPanel extends LitElement { | |||||||
|           tabindex=${this.noCollapse ? -1 : 0} |           tabindex=${this.noCollapse ? -1 : 0} | ||||||
|           aria-expanded=${this.expanded} |           aria-expanded=${this.expanded} | ||||||
|           aria-controls="sect1" |           aria-controls="sect1" | ||||||
|           part="summary" |  | ||||||
|         > |         > | ||||||
|           ${this.leftChevron ? chevronIcon : nothing} |           ${this.leftChevron ? chevronIcon : nothing} | ||||||
|           <slot name="leading-icon"></slot> |           <slot name="leading-icon"></slot> | ||||||
| @@ -171,11 +170,6 @@ export class HaExpansionPanel extends LitElement { | |||||||
|       margin-left: 8px; |       margin-left: 8px; | ||||||
|       margin-inline-start: 8px; |       margin-inline-start: 8px; | ||||||
|       margin-inline-end: initial; |       margin-inline-end: initial; | ||||||
|       border-radius: var(--ha-border-radius-circle); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     #summary:focus-visible ha-svg-icon.summary-icon { |  | ||||||
|       background-color: var(--ha-color-fill-neutral-normal-active); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     :host([left-chevron]) .summary-icon, |     :host([left-chevron]) .summary-icon, | ||||||
|   | |||||||
| @@ -79,7 +79,6 @@ export class HaGenericPicker extends LitElement { | |||||||
|         ${!this._opened |         ${!this._opened | ||||||
|           ? html` |           ? html` | ||||||
|               <ha-picker-field |               <ha-picker-field | ||||||
|                 id="picker" |  | ||||||
|                 type="button" |                 type="button" | ||||||
|                 compact |                 compact | ||||||
|                 aria-label=${ifDefined(this.label)} |                 aria-label=${ifDefined(this.label)} | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| import { ResizeController } from "@lit-labs/observers/resize-controller"; | import { ResizeController } from "@lit-labs/observers/resize-controller"; | ||||||
| import { mdiDragHorizontalVariant, mdiEye, mdiEyeOff } from "@mdi/js"; | import { mdiDrag, mdiEye, mdiEyeOff } from "@mdi/js"; | ||||||
| import type { TemplateResult } from "lit"; | import type { TemplateResult } from "lit"; | ||||||
| import { LitElement, css, html, nothing } from "lit"; | import { LitElement, css, html, nothing } from "lit"; | ||||||
| import { customElement, property, state } from "lit/decorators"; | import { customElement, property, state } from "lit/decorators"; | ||||||
| @@ -178,7 +178,7 @@ export class HaItemDisplayEditor extends LitElement { | |||||||
|                             ? this._dragHandleKeydown |                             ? this._dragHandleKeydown | ||||||
|                             : undefined} |                             : undefined} | ||||||
|                           class="handle" |                           class="handle" | ||||||
|                           .path=${mdiDragHorizontalVariant} |                           .path=${mdiDrag} | ||||||
|                           slot="end" |                           slot="end" | ||||||
|                         ></ha-svg-icon> |                         ></ha-svg-icon> | ||||||
|                       ` |                       ` | ||||||
|   | |||||||
| @@ -5,10 +5,16 @@ import { LitElement, html } from "lit"; | |||||||
| import { customElement, property, query, state } from "lit/decorators"; | import { customElement, property, query, state } from "lit/decorators"; | ||||||
| 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 type { | ||||||
|  |   DeviceEntityDisplayLookup, | ||||||
|  |   DeviceRegistryEntry, | ||||||
|  | } from "../data/device_registry"; | ||||||
|  | import { getDeviceEntityDisplayLookup } from "../data/device_registry"; | ||||||
|  | import type { EntityRegistryDisplayEntry } from "../data/entity_registry"; | ||||||
| import type { LabelRegistryEntry } from "../data/label_registry"; | import type { LabelRegistryEntry } from "../data/label_registry"; | ||||||
| import { | import { | ||||||
|   createLabelRegistryEntry, |   createLabelRegistryEntry, | ||||||
|   getLabels, |  | ||||||
|   subscribeLabelRegistry, |   subscribeLabelRegistry, | ||||||
| } from "../data/label_registry"; | } from "../data/label_registry"; | ||||||
| import { showAlertDialog } from "../dialogs/generic/show-dialog-box"; | import { showAlertDialog } from "../dialogs/generic/show-dialog-box"; | ||||||
| @@ -131,22 +137,201 @@ export class HaLabelPicker extends SubscribeMixin(LitElement) { | |||||||
|       } |       } | ||||||
|   ); |   ); | ||||||
|  |  | ||||||
|   private _getLabelsMemoized = memoizeOne(getLabels); |   private _getLabels = memoizeOne( | ||||||
|  |     ( | ||||||
|  |       labels: LabelRegistryEntry[] | undefined, | ||||||
|  |       haAreas: HomeAssistant["areas"], | ||||||
|  |       haDevices: HomeAssistant["devices"], | ||||||
|  |       haEntities: HomeAssistant["entities"], | ||||||
|  |       includeDomains: this["includeDomains"], | ||||||
|  |       excludeDomains: this["excludeDomains"], | ||||||
|  |       includeDeviceClasses: this["includeDeviceClasses"], | ||||||
|  |       deviceFilter: this["deviceFilter"], | ||||||
|  |       entityFilter: this["entityFilter"], | ||||||
|  |       excludeLabels: this["excludeLabels"] | ||||||
|  |     ): PickerComboBoxItem[] => { | ||||||
|  |       if (!labels || labels.length === 0) { | ||||||
|  |         return [ | ||||||
|  |           { | ||||||
|  |             id: NO_LABELS, | ||||||
|  |             primary: this.hass.localize("ui.components.label-picker.no_labels"), | ||||||
|  |             icon_path: mdiLabel, | ||||||
|  |           }, | ||||||
|  |         ]; | ||||||
|  |       } | ||||||
|  |  | ||||||
|   private _getItems = () => { |       const devices = Object.values(haDevices); | ||||||
|     if (!this._labels || this._labels.length === 0) { |       const entities = Object.values(haEntities); | ||||||
|       return [ |  | ||||||
|         { |       let deviceEntityLookup: DeviceEntityDisplayLookup = {}; | ||||||
|           id: NO_LABELS, |       let inputDevices: DeviceRegistryEntry[] | undefined; | ||||||
|           primary: this.hass.localize("ui.components.label-picker.no_labels"), |       let inputEntities: EntityRegistryDisplayEntry[] | undefined; | ||||||
|           icon_path: mdiLabel, |  | ||||||
|         }, |       if ( | ||||||
|       ]; |         includeDomains || | ||||||
|  |         excludeDomains || | ||||||
|  |         includeDeviceClasses || | ||||||
|  |         deviceFilter || | ||||||
|  |         entityFilter | ||||||
|  |       ) { | ||||||
|  |         deviceEntityLookup = getDeviceEntityDisplayLookup(entities); | ||||||
|  |         inputDevices = devices; | ||||||
|  |         inputEntities = entities.filter((entity) => entity.labels.length > 0); | ||||||
|  |  | ||||||
|  |         if (includeDomains) { | ||||||
|  |           inputDevices = inputDevices!.filter((device) => { | ||||||
|  |             const devEntities = deviceEntityLookup[device.id]; | ||||||
|  |             if (!devEntities || !devEntities.length) { | ||||||
|  |               return false; | ||||||
|  |             } | ||||||
|  |             return deviceEntityLookup[device.id].some((entity) => | ||||||
|  |               includeDomains.includes(computeDomain(entity.entity_id)) | ||||||
|  |             ); | ||||||
|  |           }); | ||||||
|  |           inputEntities = inputEntities!.filter((entity) => | ||||||
|  |             includeDomains.includes(computeDomain(entity.entity_id)) | ||||||
|  |           ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (excludeDomains) { | ||||||
|  |           inputDevices = inputDevices!.filter((device) => { | ||||||
|  |             const devEntities = deviceEntityLookup[device.id]; | ||||||
|  |             if (!devEntities || !devEntities.length) { | ||||||
|  |               return true; | ||||||
|  |             } | ||||||
|  |             return entities.every( | ||||||
|  |               (entity) => | ||||||
|  |                 !excludeDomains.includes(computeDomain(entity.entity_id)) | ||||||
|  |             ); | ||||||
|  |           }); | ||||||
|  |           inputEntities = inputEntities!.filter( | ||||||
|  |             (entity) => | ||||||
|  |               !excludeDomains.includes(computeDomain(entity.entity_id)) | ||||||
|  |           ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (includeDeviceClasses) { | ||||||
|  |           inputDevices = inputDevices!.filter((device) => { | ||||||
|  |             const devEntities = deviceEntityLookup[device.id]; | ||||||
|  |             if (!devEntities || !devEntities.length) { | ||||||
|  |               return false; | ||||||
|  |             } | ||||||
|  |             return deviceEntityLookup[device.id].some((entity) => { | ||||||
|  |               const stateObj = this.hass.states[entity.entity_id]; | ||||||
|  |               if (!stateObj) { | ||||||
|  |                 return false; | ||||||
|  |               } | ||||||
|  |               return ( | ||||||
|  |                 stateObj.attributes.device_class && | ||||||
|  |                 includeDeviceClasses.includes(stateObj.attributes.device_class) | ||||||
|  |               ); | ||||||
|  |             }); | ||||||
|  |           }); | ||||||
|  |           inputEntities = inputEntities!.filter((entity) => { | ||||||
|  |             const stateObj = this.hass.states[entity.entity_id]; | ||||||
|  |             return ( | ||||||
|  |               stateObj.attributes.device_class && | ||||||
|  |               includeDeviceClasses.includes(stateObj.attributes.device_class) | ||||||
|  |             ); | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (deviceFilter) { | ||||||
|  |           inputDevices = inputDevices!.filter((device) => | ||||||
|  |             deviceFilter!(device) | ||||||
|  |           ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (entityFilter) { | ||||||
|  |           inputDevices = inputDevices!.filter((device) => { | ||||||
|  |             const devEntities = deviceEntityLookup[device.id]; | ||||||
|  |             if (!devEntities || !devEntities.length) { | ||||||
|  |               return false; | ||||||
|  |             } | ||||||
|  |             return deviceEntityLookup[device.id].some((entity) => { | ||||||
|  |               const stateObj = this.hass.states[entity.entity_id]; | ||||||
|  |               if (!stateObj) { | ||||||
|  |                 return false; | ||||||
|  |               } | ||||||
|  |               return entityFilter(stateObj); | ||||||
|  |             }); | ||||||
|  |           }); | ||||||
|  |           inputEntities = inputEntities!.filter((entity) => { | ||||||
|  |             const stateObj = this.hass.states[entity.entity_id]; | ||||||
|  |             if (!stateObj) { | ||||||
|  |               return false; | ||||||
|  |             } | ||||||
|  |             return entityFilter!(stateObj); | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       let outputLabels = labels; | ||||||
|  |       const usedLabels = new Set<string>(); | ||||||
|  |  | ||||||
|  |       let areaIds: string[] | undefined; | ||||||
|  |  | ||||||
|  |       if (inputDevices) { | ||||||
|  |         areaIds = inputDevices | ||||||
|  |           .filter((device) => device.area_id) | ||||||
|  |           .map((device) => device.area_id!); | ||||||
|  |  | ||||||
|  |         inputDevices.forEach((device) => { | ||||||
|  |           device.labels.forEach((label) => usedLabels.add(label)); | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (inputEntities) { | ||||||
|  |         areaIds = (areaIds ?? []).concat( | ||||||
|  |           inputEntities | ||||||
|  |             .filter((entity) => entity.area_id) | ||||||
|  |             .map((entity) => entity.area_id!) | ||||||
|  |         ); | ||||||
|  |         inputEntities.forEach((entity) => { | ||||||
|  |           entity.labels.forEach((label) => usedLabels.add(label)); | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (areaIds) { | ||||||
|  |         areaIds.forEach((areaId) => { | ||||||
|  |           const area = haAreas[areaId]; | ||||||
|  |           area.labels.forEach((label) => usedLabels.add(label)); | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (excludeLabels) { | ||||||
|  |         outputLabels = outputLabels.filter( | ||||||
|  |           (label) => !excludeLabels!.includes(label.label_id) | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (inputDevices || inputEntities) { | ||||||
|  |         outputLabels = outputLabels.filter((label) => | ||||||
|  |           usedLabels.has(label.label_id) | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       const items = outputLabels.map<PickerComboBoxItem>((label) => ({ | ||||||
|  |         id: label.label_id, | ||||||
|  |         primary: label.name, | ||||||
|  |         icon: label.icon || undefined, | ||||||
|  |         icon_path: label.icon ? undefined : mdiLabel, | ||||||
|  |         sorting_label: label.name, | ||||||
|  |         search_labels: [label.name, label.label_id, label.description].filter( | ||||||
|  |           (v): v is string => Boolean(v) | ||||||
|  |         ), | ||||||
|  |       })); | ||||||
|  |  | ||||||
|  |       return items; | ||||||
|     } |     } | ||||||
|  |   ); | ||||||
|  |  | ||||||
|     return this._getLabelsMemoized( |   private _getItems = () => | ||||||
|       this.hass, |     this._getLabels( | ||||||
|       this._labels, |       this._labels, | ||||||
|  |       this.hass.areas, | ||||||
|  |       this.hass.devices, | ||||||
|  |       this.hass.entities, | ||||||
|       this.includeDomains, |       this.includeDomains, | ||||||
|       this.excludeDomains, |       this.excludeDomains, | ||||||
|       this.includeDeviceClasses, |       this.includeDeviceClasses, | ||||||
| @@ -154,7 +339,6 @@ export class HaLabelPicker extends SubscribeMixin(LitElement) { | |||||||
|       this.entityFilter, |       this.entityFilter, | ||||||
|       this.excludeLabels |       this.excludeLabels | ||||||
|     ); |     ); | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   private _allLabelNames = memoizeOne((labels?: LabelRegistryEntry[]) => { |   private _allLabelNames = memoizeOne((labels?: LabelRegistryEntry[]) => { | ||||||
|     if (!labels) { |     if (!labels) { | ||||||
|   | |||||||
							
								
								
									
										152
									
								
								src/components/ha-selector/ha-selector-image.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										152
									
								
								src/components/ha-selector/ha-selector-image.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,152 @@ | |||||||
|  | import { css, html, LitElement } from "lit"; | ||||||
|  | import { customElement, property, state } from "lit/decorators"; | ||||||
|  | import { fireEvent } from "../../common/dom/fire_event"; | ||||||
|  | import type { ImageSelector } from "../../data/selector"; | ||||||
|  | import type { HomeAssistant } from "../../types"; | ||||||
|  | import "../ha-icon-button"; | ||||||
|  | import "../ha-textarea"; | ||||||
|  | import "../ha-textfield"; | ||||||
|  | import "../ha-picture-upload"; | ||||||
|  | import "../ha-radio"; | ||||||
|  | import "../ha-formfield"; | ||||||
|  | import type { HaPictureUpload } from "../ha-picture-upload"; | ||||||
|  | import { URL_PREFIX } from "../../data/image_upload"; | ||||||
|  |  | ||||||
|  | @customElement("ha-selector-image") | ||||||
|  | export class HaImageSelector extends LitElement { | ||||||
|  |   @property({ attribute: false }) public hass!: HomeAssistant; | ||||||
|  |  | ||||||
|  |   @property() public value?: any; | ||||||
|  |  | ||||||
|  |   @property() public name?: string; | ||||||
|  |  | ||||||
|  |   @property() public label?: string; | ||||||
|  |  | ||||||
|  |   @property() public placeholder?: string; | ||||||
|  |  | ||||||
|  |   @property() public helper?: string; | ||||||
|  |  | ||||||
|  |   @property({ attribute: false }) public selector!: ImageSelector; | ||||||
|  |  | ||||||
|  |   @property({ type: Boolean }) public disabled = false; | ||||||
|  |  | ||||||
|  |   @property({ type: Boolean }) public required = true; | ||||||
|  |  | ||||||
|  |   @state() private showUpload = false; | ||||||
|  |  | ||||||
|  |   protected firstUpdated(changedProps): void { | ||||||
|  |     super.firstUpdated(changedProps); | ||||||
|  |  | ||||||
|  |     if (!this.value || this.value.startsWith(URL_PREFIX)) { | ||||||
|  |       this.showUpload = true; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   protected render() { | ||||||
|  |     return html` | ||||||
|  |       <div> | ||||||
|  |         <label> | ||||||
|  |           ${this.hass.localize( | ||||||
|  |             "ui.components.selectors.image.select_image_with_label", | ||||||
|  |             { | ||||||
|  |               label: | ||||||
|  |                 this.label || | ||||||
|  |                 this.hass.localize("ui.components.selectors.image.image"), | ||||||
|  |             } | ||||||
|  |           )} | ||||||
|  |           <ha-formfield | ||||||
|  |             .label=${this.hass.localize("ui.components.selectors.image.upload")} | ||||||
|  |           > | ||||||
|  |             <ha-radio | ||||||
|  |               name="mode" | ||||||
|  |               value="upload" | ||||||
|  |               .checked=${this.showUpload} | ||||||
|  |               @change=${this._radioGroupPicked} | ||||||
|  |             ></ha-radio> | ||||||
|  |           </ha-formfield> | ||||||
|  |           <ha-formfield | ||||||
|  |             .label=${this.hass.localize("ui.components.selectors.image.url")} | ||||||
|  |           > | ||||||
|  |             <ha-radio | ||||||
|  |               name="mode" | ||||||
|  |               value="url" | ||||||
|  |               .checked=${!this.showUpload} | ||||||
|  |               @change=${this._radioGroupPicked} | ||||||
|  |             ></ha-radio> | ||||||
|  |           </ha-formfield> | ||||||
|  |         </label> | ||||||
|  |         ${!this.showUpload | ||||||
|  |           ? html` | ||||||
|  |               <ha-textfield | ||||||
|  |                 .name=${this.name} | ||||||
|  |                 .value=${this.value || ""} | ||||||
|  |                 .placeholder=${this.placeholder || ""} | ||||||
|  |                 .helper=${this.helper} | ||||||
|  |                 helperPersistent | ||||||
|  |                 .disabled=${this.disabled} | ||||||
|  |                 @input=${this._handleChange} | ||||||
|  |                 .label=${this.label || ""} | ||||||
|  |                 .required=${this.required} | ||||||
|  |               ></ha-textfield> | ||||||
|  |             ` | ||||||
|  |           : html` | ||||||
|  |               <ha-picture-upload | ||||||
|  |                 .hass=${this.hass} | ||||||
|  |                 .value=${this.value?.startsWith(URL_PREFIX) ? this.value : null} | ||||||
|  |                 .original=${this.selector.image?.original} | ||||||
|  |                 .cropOptions=${this.selector.image?.crop} | ||||||
|  |                 select-media | ||||||
|  |                 @change=${this._pictureChanged} | ||||||
|  |               ></ha-picture-upload> | ||||||
|  |             `} | ||||||
|  |       </div> | ||||||
|  |     `; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private _radioGroupPicked(ev): void { | ||||||
|  |     this.showUpload = ev.target.value === "upload"; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private _pictureChanged(ev) { | ||||||
|  |     const value = (ev.target as HaPictureUpload).value; | ||||||
|  |  | ||||||
|  |     fireEvent(this, "value-changed", { value: value ?? undefined }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private _handleChange(ev) { | ||||||
|  |     let value = ev.target.value; | ||||||
|  |     if (this.value === value) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     if (value === "" && !this.required) { | ||||||
|  |       value = undefined; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fireEvent(this, "value-changed", { value }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   static styles = css` | ||||||
|  |     :host { | ||||||
|  |       display: block; | ||||||
|  |       position: relative; | ||||||
|  |     } | ||||||
|  |     div { | ||||||
|  |       display: flex; | ||||||
|  |       flex-direction: column; | ||||||
|  |     } | ||||||
|  |     label { | ||||||
|  |       display: flex; | ||||||
|  |       flex-direction: column; | ||||||
|  |     } | ||||||
|  |     ha-textarea, | ||||||
|  |     ha-textfield { | ||||||
|  |       width: 100%; | ||||||
|  |     } | ||||||
|  |   `; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | declare global { | ||||||
|  |   interface HTMLElementTagNameMap { | ||||||
|  |     "ha-selector-image": HaImageSelector; | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -107,15 +107,14 @@ export class HaMediaSelector extends LitElement { | |||||||
|         supportsFeature(stateObj, MediaPlayerEntityFeature.BROWSE_MEDIA)); |         supportsFeature(stateObj, MediaPlayerEntityFeature.BROWSE_MEDIA)); | ||||||
|  |  | ||||||
|     if (this.selector.media?.image_upload && !this.value) { |     if (this.selector.media?.image_upload && !this.value) { | ||||||
|       return html`${this.label ? html`<label>${this.label}</label>` : nothing} |       return html`<ha-picture-upload | ||||||
|         <ha-picture-upload |         .hass=${this.hass} | ||||||
|           .hass=${this.hass} |         .value=${null} | ||||||
|           .value=${null} |         .contentIdHelper=${this.selector.media?.content_id_helper} | ||||||
|           .contentIdHelper=${this.selector.media?.content_id_helper} |         select-media | ||||||
|           select-media |         full-media | ||||||
|           full-media |         @media-picked=${this._pictureUploadMediaPicked} | ||||||
|           @media-picked=${this._pictureUploadMediaPicked} |       ></ha-picture-upload>`; | ||||||
|         ></ha-picture-upload>`; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     return html` |     return html` | ||||||
| @@ -142,7 +141,6 @@ export class HaMediaSelector extends LitElement { | |||||||
|           `} |           `} | ||||||
|       ${!supportsBrowse |       ${!supportsBrowse | ||||||
|         ? html` |         ? html` | ||||||
|             ${this.label ? html`<label>${this.label}</label>` : nothing} |  | ||||||
|             <ha-alert> |             <ha-alert> | ||||||
|               ${this.hass.localize( |               ${this.hass.localize( | ||||||
|                 "ui.components.selectors.media.browse_not_supported" |                 "ui.components.selectors.media.browse_not_supported" | ||||||
| @@ -156,8 +154,7 @@ export class HaMediaSelector extends LitElement { | |||||||
|               .computeHelper=${this._computeHelperCallback} |               .computeHelper=${this._computeHelperCallback} | ||||||
|             ></ha-form> |             ></ha-form> | ||||||
|           ` |           ` | ||||||
|         : html`${this.label ? html`<label>${this.label}</label>` : nothing} |         : html`<ha-card | ||||||
|             <ha-card |  | ||||||
|               outlined |               outlined | ||||||
|               tabindex="0" |               tabindex="0" | ||||||
|               role="button" |               role="button" | ||||||
|   | |||||||
| @@ -1,9 +1,4 @@ | |||||||
| import { | import { mdiClose, mdiDelete, mdiDrag, mdiPencil } from "@mdi/js"; | ||||||
|   mdiClose, |  | ||||||
|   mdiDelete, |  | ||||||
|   mdiDragHorizontalVariant, |  | ||||||
|   mdiPencil, |  | ||||||
| } from "@mdi/js"; |  | ||||||
| import { css, html, LitElement, nothing, type PropertyValues } from "lit"; | import { css, html, LitElement, nothing, type PropertyValues } from "lit"; | ||||||
| import { customElement, property, query } from "lit/decorators"; | import { customElement, property, query } from "lit/decorators"; | ||||||
| import memoizeOne from "memoize-one"; | import memoizeOne from "memoize-one"; | ||||||
| @@ -97,7 +92,7 @@ export class HaObjectSelector extends LitElement { | |||||||
|           ? html` |           ? html` | ||||||
|               <ha-svg-icon |               <ha-svg-icon | ||||||
|                 class="handle" |                 class="handle" | ||||||
|                 .path=${mdiDragHorizontalVariant} |                 .path=${mdiDrag} | ||||||
|                 slot="start" |                 slot="start" | ||||||
|               ></ha-svg-icon> |               ></ha-svg-icon> | ||||||
|             ` |             ` | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| import { mdiDragHorizontalVariant } from "@mdi/js"; | import { mdiDrag } from "@mdi/js"; | ||||||
| import { LitElement, css, html, nothing } from "lit"; | import { LitElement, css, html, nothing } from "lit"; | ||||||
| import { customElement, property, query } from "lit/decorators"; | import { customElement, property, query } from "lit/decorators"; | ||||||
| import { repeat } from "lit/directives/repeat"; | import { repeat } from "lit/directives/repeat"; | ||||||
| @@ -197,7 +197,7 @@ export class HaSelectSelector extends LitElement { | |||||||
|                             ? html` |                             ? html` | ||||||
|                                 <ha-svg-icon |                                 <ha-svg-icon | ||||||
|                                   slot="icon" |                                   slot="icon" | ||||||
|                                   .path=${mdiDragHorizontalVariant} |                                   .path=${mdiDrag} | ||||||
|                                 ></ha-svg-icon> |                                 ></ha-svg-icon> | ||||||
|                               ` |                               ` | ||||||
|                             : nothing} |                             : nothing} | ||||||
|   | |||||||
| @@ -34,6 +34,7 @@ const LOAD_ELEMENTS = { | |||||||
|   file: () => import("./ha-selector-file"), |   file: () => import("./ha-selector-file"), | ||||||
|   floor: () => import("./ha-selector-floor"), |   floor: () => import("./ha-selector-floor"), | ||||||
|   label: () => import("./ha-selector-label"), |   label: () => import("./ha-selector-label"), | ||||||
|  |   image: () => import("./ha-selector-image"), | ||||||
|   background: () => import("./ha-selector-background"), |   background: () => import("./ha-selector-background"), | ||||||
|   language: () => import("./ha-selector-language"), |   language: () => import("./ha-selector-language"), | ||||||
|   navigation: () => import("./ha-selector-navigation"), |   navigation: () => import("./ha-selector-navigation"), | ||||||
|   | |||||||
| @@ -59,17 +59,6 @@ export class HaSlider extends Slider { | |||||||
|           background-color: var(--ha-slider-thumb-color, var(--primary-color)); |           background-color: var(--ha-slider-thumb-color, var(--primary-color)); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         #thumb:after { |  | ||||||
|           content: ""; |  | ||||||
|           border-radius: 50%; |  | ||||||
|           position: absolute; |  | ||||||
|           width: calc(var(--thumb-width) * 2 + 8px); |  | ||||||
|           height: calc(var(--thumb-height) * 2 + 8px); |  | ||||||
|           left: calc(-50% - 4px); |  | ||||||
|           top: calc(-50% - 4px); |  | ||||||
|           cursor: pointer; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         #slider:focus-visible:not(.disabled) #thumb, |         #slider:focus-visible:not(.disabled) #thumb, | ||||||
|         #slider:focus-visible:not(.disabled) #thumb-min, |         #slider:focus-visible:not(.disabled) #thumb-min, | ||||||
|         #slider:focus-visible:not(.disabled) #thumb-max { |         #slider:focus-visible:not(.disabled) #thumb-max { | ||||||
|   | |||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -247,7 +247,10 @@ export class HaWaDialog extends LitElement { | |||||||
|       .header-title { |       .header-title { | ||||||
|         margin: 0; |         margin: 0; | ||||||
|         margin-bottom: 0; |         margin-bottom: 0; | ||||||
|         color: var(--ha-dialog-header-title-color, var(--primary-text-color)); |         color: var( | ||||||
|  |           --ha-dialog-header-title-color, | ||||||
|  |           var(--ha-color-on-surface-default, var(--primary-text-color)) | ||||||
|  |         ); | ||||||
|         font-size: var( |         font-size: var( | ||||||
|           --ha-dialog-header-title-font-size, |           --ha-dialog-header-title-font-size, | ||||||
|           var(--ha-font-size-2xl) |           var(--ha-font-size-2xl) | ||||||
|   | |||||||
| @@ -321,10 +321,6 @@ class HaWebRtcPlayer extends LitElement { | |||||||
|     if (!this._remoteStream) { |     if (!this._remoteStream) { | ||||||
|       return; |       return; | ||||||
|     } |     } | ||||||
|     // If the track is audio and the player is muted, we do not add it to the stream. |  | ||||||
|     if (event.track.kind === "audio" && this.muted) { |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
|     this._remoteStream.addTrack(event.track); |     this._remoteStream.addTrack(event.track); | ||||||
|     if (!this.hasUpdated) { |     if (!this.hasUpdated) { | ||||||
|       await this.updateComplete; |       await this.updateComplete; | ||||||
|   | |||||||
| @@ -1,104 +0,0 @@ | |||||||
| import { mdiClose } from "@mdi/js"; |  | ||||||
| import { css, html, LitElement, nothing } from "lit"; |  | ||||||
| import { customElement, property, query, state } from "lit/decorators"; |  | ||||||
| import { fireEvent } from "../../../common/dom/fire_event"; |  | ||||||
| import type { HassDialog } from "../../../dialogs/make-dialog-manager"; |  | ||||||
| import type { HomeAssistant } from "../../../types"; |  | ||||||
| import "../../ha-dialog-header"; |  | ||||||
| import "../../ha-icon-button"; |  | ||||||
| import "../../ha-icon-next"; |  | ||||||
| import "../../ha-md-dialog"; |  | ||||||
| import type { HaMdDialog } from "../../ha-md-dialog"; |  | ||||||
| import "../../ha-md-list"; |  | ||||||
| import "../../ha-md-list-item"; |  | ||||||
| import "../../ha-svg-icon"; |  | ||||||
| import "../ha-target-picker-item-row"; |  | ||||||
| import type { TargetDetailsDialogParams } from "./show-dialog-target-details"; |  | ||||||
|  |  | ||||||
| @customElement("ha-dialog-target-details") |  | ||||||
| class DialogTargetDetails extends LitElement implements HassDialog { |  | ||||||
|   @property({ attribute: false }) public hass!: HomeAssistant; |  | ||||||
|  |  | ||||||
|   @state() private _params?: TargetDetailsDialogParams; |  | ||||||
|  |  | ||||||
|   @query("ha-md-dialog") private _dialog?: HaMdDialog; |  | ||||||
|  |  | ||||||
|   public showDialog(params: TargetDetailsDialogParams): void { |  | ||||||
|     this._params = params; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   public closeDialog() { |  | ||||||
|     this._dialog?.close(); |  | ||||||
|     return true; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private _dialogClosed() { |  | ||||||
|     fireEvent(this, "dialog-closed", { dialog: this.localName }); |  | ||||||
|     this._params = undefined; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   protected render() { |  | ||||||
|     if (!this._params) { |  | ||||||
|       return nothing; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     return html` |  | ||||||
|       <ha-md-dialog open @closed=${this._dialogClosed}> |  | ||||||
|         <ha-dialog-header slot="headline"> |  | ||||||
|           <ha-icon-button |  | ||||||
|             slot="navigationIcon" |  | ||||||
|             @click=${this.closeDialog} |  | ||||||
|             .label=${this.hass.localize("ui.common.close")} |  | ||||||
|             .path=${mdiClose} |  | ||||||
|           ></ha-icon-button> |  | ||||||
|           <span slot="title" |  | ||||||
|             >${this.hass.localize( |  | ||||||
|               "ui.components.target-picker.target_details" |  | ||||||
|             )}</span |  | ||||||
|           > |  | ||||||
|           <span slot="subtitle" |  | ||||||
|             >${this.hass.localize( |  | ||||||
|               `ui.components.target-picker.type.${this._params.type}` |  | ||||||
|             )}: |  | ||||||
|             ${this._params.title}</span |  | ||||||
|           > |  | ||||||
|         </ha-dialog-header> |  | ||||||
|         <div slot="content"> |  | ||||||
|           <ha-target-picker-item-row |  | ||||||
|             .hass=${this.hass} |  | ||||||
|             .type=${this._params.type} |  | ||||||
|             .itemId=${this._params.itemId} |  | ||||||
|             .deviceFilter=${this._params.deviceFilter} |  | ||||||
|             .entityFilter=${this._params.entityFilter} |  | ||||||
|             .includeDomains=${this._params.includeDomains} |  | ||||||
|             .includeDeviceClasses=${this._params.includeDeviceClasses} |  | ||||||
|             expand |  | ||||||
|           ></ha-target-picker-item-row> |  | ||||||
|         </div> |  | ||||||
|       </ha-md-dialog> |  | ||||||
|     `; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   static styles = css` |  | ||||||
|     ha-md-dialog { |  | ||||||
|       min-width: 400px; |  | ||||||
|       max-height: 90%; |  | ||||||
|       --dialog-content-padding: var(--ha-space-2) var(--ha-space-6) |  | ||||||
|         max(var(--safe-area-inset-bottom, var(--ha-space-0)), var(--ha-space-8)); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @media all and (max-width: 600px), all and (max-height: 500px) { |  | ||||||
|       ha-md-dialog { |  | ||||||
|         --md-dialog-container-shape: var(--ha-space-0); |  | ||||||
|         min-width: 100%; |  | ||||||
|         min-height: 100%; |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   `; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| declare global { |  | ||||||
|   interface HTMLElementTagNameMap { |  | ||||||
|     "ha-dialog-target-details": DialogTargetDetails; |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| @@ -1,28 +0,0 @@ | |||||||
| import { fireEvent } from "../../../common/dom/fire_event"; |  | ||||||
| import type { HaEntityPickerEntityFilterFunc } from "../../../data/entity"; |  | ||||||
| import type { TargetType } from "../../../data/target"; |  | ||||||
| import type { HaDevicePickerDeviceFilterFunc } from "../../device/ha-device-picker"; |  | ||||||
|  |  | ||||||
| export type NewBackupType = "automatic" | "manual"; |  | ||||||
|  |  | ||||||
| export interface TargetDetailsDialogParams { |  | ||||||
|   title: string; |  | ||||||
|   type: TargetType; |  | ||||||
|   itemId: string; |  | ||||||
|   deviceFilter?: HaDevicePickerDeviceFilterFunc; |  | ||||||
|   entityFilter?: HaEntityPickerEntityFilterFunc; |  | ||||||
|   includeDomains?: string[]; |  | ||||||
|   includeDeviceClasses?: string[]; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export const loadTargetDetailsDialog = () => import("./dialog-target-details"); |  | ||||||
|  |  | ||||||
| export const showTargetDetailsDialog = ( |  | ||||||
|   element: HTMLElement, |  | ||||||
|   params: TargetDetailsDialogParams |  | ||||||
| ) => |  | ||||||
|   fireEvent(element, "show-dialog", { |  | ||||||
|     dialogTag: "ha-dialog-target-details", |  | ||||||
|     dialogImport: loadTargetDetailsDialog, |  | ||||||
|     dialogParams: params, |  | ||||||
|   }); |  | ||||||
| @@ -1,113 +0,0 @@ | |||||||
| import { css, html, LitElement, nothing } from "lit"; |  | ||||||
| import { customElement, property } from "lit/decorators"; |  | ||||||
| import type { HaEntityPickerEntityFilterFunc } from "../../data/entity"; |  | ||||||
| import type { TargetType, TargetTypeFloorless } from "../../data/target"; |  | ||||||
| import type { HomeAssistant } from "../../types"; |  | ||||||
| import type { HaDevicePickerDeviceFilterFunc } from "../device/ha-device-picker"; |  | ||||||
| import "../ha-expansion-panel"; |  | ||||||
| import "../ha-md-list"; |  | ||||||
| import "./ha-target-picker-item-row"; |  | ||||||
|  |  | ||||||
| @customElement("ha-target-picker-item-group") |  | ||||||
| export class HaTargetPickerItemGroup extends LitElement { |  | ||||||
|   @property({ attribute: false }) public hass!: HomeAssistant; |  | ||||||
|  |  | ||||||
|   @property() public type!: TargetTypeFloorless; |  | ||||||
|  |  | ||||||
|   @property({ attribute: false }) public items!: Partial< |  | ||||||
|     Record<TargetType, string[]> |  | ||||||
|   >; |  | ||||||
|  |  | ||||||
|   @property({ type: Boolean, reflect: true }) public collapsed = false; |  | ||||||
|  |  | ||||||
|   @property({ attribute: false }) |  | ||||||
|   public deviceFilter?: HaDevicePickerDeviceFilterFunc; |  | ||||||
|  |  | ||||||
|   @property({ attribute: false }) |  | ||||||
|   public entityFilter?: HaEntityPickerEntityFilterFunc; |  | ||||||
|  |  | ||||||
|   /** |  | ||||||
|    * Show only targets with entities from specific domains. |  | ||||||
|    * @type {Array} |  | ||||||
|    * @attr include-domains |  | ||||||
|    */ |  | ||||||
|   @property({ type: Array, attribute: "include-domains" }) |  | ||||||
|   public includeDomains?: string[]; |  | ||||||
|  |  | ||||||
|   /** |  | ||||||
|    * Show only targets with entities of these device classes. |  | ||||||
|    * @type {Array} |  | ||||||
|    * @attr include-device-classes |  | ||||||
|    */ |  | ||||||
|   @property({ type: Array, attribute: "include-device-classes" }) |  | ||||||
|   public includeDeviceClasses?: string[]; |  | ||||||
|  |  | ||||||
|   protected render() { |  | ||||||
|     let count = 0; |  | ||||||
|     Object.values(this.items).forEach((items) => { |  | ||||||
|       if (items) { |  | ||||||
|         count += items.length; |  | ||||||
|       } |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     return html`<ha-expansion-panel |  | ||||||
|       .expanded=${!this.collapsed} |  | ||||||
|       left-chevron |  | ||||||
|       @expanded-changed=${this._expandedChanged} |  | ||||||
|     > |  | ||||||
|       <div slot="header" class="heading"> |  | ||||||
|         ${this.hass.localize( |  | ||||||
|           `ui.components.target-picker.selected.${this.type}`, |  | ||||||
|           { |  | ||||||
|             count, |  | ||||||
|           } |  | ||||||
|         )} |  | ||||||
|       </div> |  | ||||||
|       ${Object.entries(this.items).map(([type, items]) => |  | ||||||
|         items |  | ||||||
|           ? items.map( |  | ||||||
|               (item) => |  | ||||||
|                 html`<ha-target-picker-item-row |  | ||||||
|                   .hass=${this.hass} |  | ||||||
|                   .type=${type as TargetTypeFloorless} |  | ||||||
|                   .itemId=${item} |  | ||||||
|                   .deviceFilter=${this.deviceFilter} |  | ||||||
|                   .entityFilter=${this.entityFilter} |  | ||||||
|                   .includeDomains=${this.includeDomains} |  | ||||||
|                   .includeDeviceClasses=${this.includeDeviceClasses} |  | ||||||
|                 ></ha-target-picker-item-row>` |  | ||||||
|             ) |  | ||||||
|           : nothing |  | ||||||
|       )} |  | ||||||
|     </ha-expansion-panel>`; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private _expandedChanged(ev: CustomEvent) { |  | ||||||
|     this.collapsed = !ev.detail.expanded; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   static styles = css` |  | ||||||
|     :host { |  | ||||||
|       display: block; |  | ||||||
|       --expansion-panel-content-padding: var(--ha-space-0); |  | ||||||
|     } |  | ||||||
|     ha-expansion-panel::part(summary) { |  | ||||||
|       background-color: var(--ha-color-fill-neutral-quiet-resting); |  | ||||||
|       padding: var(--ha-space-1) var(--ha-space-2); |  | ||||||
|       font-weight: var(--ha-font-weight-bold); |  | ||||||
|       color: var(--secondary-text-color); |  | ||||||
|       display: flex; |  | ||||||
|       justify-content: space-between; |  | ||||||
|       min-height: unset; |  | ||||||
|     } |  | ||||||
|     ha-md-list { |  | ||||||
|       padding: var(--ha-space-0); |  | ||||||
|     } |  | ||||||
|   `; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| declare global { |  | ||||||
|   interface HTMLElementTagNameMap { |  | ||||||
|     "ha-target-picker-item-group": HaTargetPickerItemGroup; |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| @@ -1,694 +0,0 @@ | |||||||
| import { consume } from "@lit/context"; |  | ||||||
| import { |  | ||||||
|   mdiClose, |  | ||||||
|   mdiDevices, |  | ||||||
|   mdiHome, |  | ||||||
|   mdiLabel, |  | ||||||
|   mdiTextureBox, |  | ||||||
| } from "@mdi/js"; |  | ||||||
| import { css, html, LitElement, nothing, type PropertyValues } from "lit"; |  | ||||||
| import { customElement, property, query, state } from "lit/decorators"; |  | ||||||
| import memoizeOne from "memoize-one"; |  | ||||||
| import { fireEvent } from "../../common/dom/fire_event"; |  | ||||||
| import { computeAreaName } from "../../common/entity/compute_area_name"; |  | ||||||
| import { |  | ||||||
|   computeDeviceName, |  | ||||||
|   computeDeviceNameDisplay, |  | ||||||
| } from "../../common/entity/compute_device_name"; |  | ||||||
| import { computeDomain } from "../../common/entity/compute_domain"; |  | ||||||
| import { computeEntityName } from "../../common/entity/compute_entity_name"; |  | ||||||
| import { getEntityContext } from "../../common/entity/context/get_entity_context"; |  | ||||||
| import { computeRTL } from "../../common/util/compute_rtl"; |  | ||||||
| import { getConfigEntry } from "../../data/config_entries"; |  | ||||||
| import { labelsContext } from "../../data/context"; |  | ||||||
| import type { HaEntityPickerEntityFilterFunc } from "../../data/entity"; |  | ||||||
| import { domainToName } from "../../data/integration"; |  | ||||||
| import type { LabelRegistryEntry } from "../../data/label_registry"; |  | ||||||
| import { |  | ||||||
|   areaMeetsFilter, |  | ||||||
|   deviceMeetsFilter, |  | ||||||
|   entityRegMeetsFilter, |  | ||||||
|   extractFromTarget, |  | ||||||
|   type ExtractFromTargetResult, |  | ||||||
|   type ExtractFromTargetResultReferenced, |  | ||||||
|   type TargetType, |  | ||||||
| } from "../../data/target"; |  | ||||||
| import { buttonLinkStyle } from "../../resources/styles"; |  | ||||||
| import type { HomeAssistant } from "../../types"; |  | ||||||
| import { brandsUrl } from "../../util/brands-url"; |  | ||||||
| import type { HaDevicePickerDeviceFilterFunc } from "../device/ha-device-picker"; |  | ||||||
| import { floorDefaultIconPath } from "../ha-floor-icon"; |  | ||||||
| import "../ha-icon-button"; |  | ||||||
| import "../ha-md-list"; |  | ||||||
| import type { HaMdList } from "../ha-md-list"; |  | ||||||
| import "../ha-md-list-item"; |  | ||||||
| import type { HaMdListItem } from "../ha-md-list-item"; |  | ||||||
| import "../ha-state-icon"; |  | ||||||
| import "../ha-svg-icon"; |  | ||||||
| import { showTargetDetailsDialog } from "./dialog/show-dialog-target-details"; |  | ||||||
|  |  | ||||||
| @customElement("ha-target-picker-item-row") |  | ||||||
| export class HaTargetPickerItemRow extends LitElement { |  | ||||||
|   @property({ attribute: false }) public hass!: HomeAssistant; |  | ||||||
|  |  | ||||||
|   @property({ reflect: true }) public type!: TargetType; |  | ||||||
|  |  | ||||||
|   @property({ attribute: "item-id" }) public itemId!: string; |  | ||||||
|  |  | ||||||
|   @property({ type: Boolean }) public expand = false; |  | ||||||
|  |  | ||||||
|   @property({ type: Boolean, attribute: "sub-entry", reflect: true }) |  | ||||||
|   public subEntry = false; |  | ||||||
|  |  | ||||||
|   @property({ type: Boolean, attribute: "hide-context" }) |  | ||||||
|   public hideContext = false; |  | ||||||
|  |  | ||||||
|   @property({ attribute: false }) |  | ||||||
|   public parentEntries?: ExtractFromTargetResultReferenced; |  | ||||||
|  |  | ||||||
|   @property({ attribute: false }) |  | ||||||
|   public deviceFilter?: HaDevicePickerDeviceFilterFunc; |  | ||||||
|  |  | ||||||
|   @property({ attribute: false }) |  | ||||||
|   public entityFilter?: HaEntityPickerEntityFilterFunc; |  | ||||||
|  |  | ||||||
|   /** |  | ||||||
|    * Show only targets with entities from specific domains. |  | ||||||
|    * @type {Array} |  | ||||||
|    * @attr include-domains |  | ||||||
|    */ |  | ||||||
|   @property({ type: Array, attribute: "include-domains" }) |  | ||||||
|   public includeDomains?: string[]; |  | ||||||
|  |  | ||||||
|   /** |  | ||||||
|    * Show only targets with entities of these device classes. |  | ||||||
|    * @type {Array} |  | ||||||
|    * @attr include-device-classes |  | ||||||
|    */ |  | ||||||
|   @property({ type: Array, attribute: "include-device-classes" }) |  | ||||||
|   public includeDeviceClasses?: string[]; |  | ||||||
|  |  | ||||||
|   @state() private _iconImg?: string; |  | ||||||
|  |  | ||||||
|   @state() private _domainName?: string; |  | ||||||
|  |  | ||||||
|   @state() private _entries?: ExtractFromTargetResult; |  | ||||||
|  |  | ||||||
|   @state() |  | ||||||
|   @consume({ context: labelsContext, subscribe: true }) |  | ||||||
|   _labelRegistry!: LabelRegistryEntry[]; |  | ||||||
|  |  | ||||||
|   @query("ha-md-list-item") public item?: HaMdListItem; |  | ||||||
|  |  | ||||||
|   @query("ha-md-list") public list?: HaMdList; |  | ||||||
|  |  | ||||||
|   @query("ha-target-picker-item-row") public itemRow?: HaTargetPickerItemRow; |  | ||||||
|  |  | ||||||
|   protected willUpdate(changedProps: PropertyValues) { |  | ||||||
|     if (!this.subEntry && changedProps.has("itemId")) { |  | ||||||
|       this._updateItemData(); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   protected render() { |  | ||||||
|     const { name, context, iconPath, fallbackIconPath, stateObject } = |  | ||||||
|       this._itemData(this.type, this.itemId); |  | ||||||
|  |  | ||||||
|     const showDevices = ["floor", "area", "label"].includes(this.type); |  | ||||||
|     const showEntities = this.type !== "entity"; |  | ||||||
|  |  | ||||||
|     const entries = this.parentEntries || this._entries; |  | ||||||
|  |  | ||||||
|     // Don't show sub entries that have no entities |  | ||||||
|     if ( |  | ||||||
|       this.subEntry && |  | ||||||
|       this.type !== "entity" && |  | ||||||
|       (!entries || entries.referenced_entities.length === 0) |  | ||||||
|     ) { |  | ||||||
|       return nothing; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     return html` |  | ||||||
|       <ha-md-list-item type="text"> |  | ||||||
|         <div class="icon" slot="start"> |  | ||||||
|           ${this.subEntry |  | ||||||
|             ? html` |  | ||||||
|                 <div class="horizontal-line-wrapper"> |  | ||||||
|                   <div class="horizontal-line"></div> |  | ||||||
|                 </div> |  | ||||||
|               ` |  | ||||||
|             : nothing} |  | ||||||
|           ${iconPath |  | ||||||
|             ? html`<ha-icon .icon=${iconPath}></ha-icon>` |  | ||||||
|             : this._iconImg |  | ||||||
|               ? html`<img |  | ||||||
|                   alt=${this._domainName || ""} |  | ||||||
|                   crossorigin="anonymous" |  | ||||||
|                   referrerpolicy="no-referrer" |  | ||||||
|                   src=${this._iconImg} |  | ||||||
|                 />` |  | ||||||
|               : fallbackIconPath |  | ||||||
|                 ? html`<ha-svg-icon .path=${fallbackIconPath}></ha-svg-icon>` |  | ||||||
|                 : stateObject |  | ||||||
|                   ? html` |  | ||||||
|                       <ha-state-icon |  | ||||||
|                         .hass=${this.hass} |  | ||||||
|                         .stateObj=${stateObject} |  | ||||||
|                       > |  | ||||||
|                       </ha-state-icon> |  | ||||||
|                     ` |  | ||||||
|                   : nothing} |  | ||||||
|         </div> |  | ||||||
|  |  | ||||||
|         <div slot="headline">${name}</div> |  | ||||||
|         ${context && !this.hideContext |  | ||||||
|           ? html`<span slot="supporting-text">${context}</span>` |  | ||||||
|           : this._domainName && this.subEntry |  | ||||||
|             ? html`<span slot="supporting-text" class="domain" |  | ||||||
|                 >${this._domainName}</span |  | ||||||
|               >` |  | ||||||
|             : nothing} |  | ||||||
|         ${!this.subEntry && |  | ||||||
|         ((entries && (showEntities || showDevices)) || this._domainName) |  | ||||||
|           ? html` |  | ||||||
|               <div slot="end" class="summary"> |  | ||||||
|                 ${showEntities && |  | ||||||
|                 !this.expand && |  | ||||||
|                 entries?.referenced_entities.length |  | ||||||
|                   ? html`<button class="main link" @click=${this._openDetails}> |  | ||||||
|                       ${this.hass.localize( |  | ||||||
|                         "ui.components.target-picker.entities_count", |  | ||||||
|                         { |  | ||||||
|                           count: entries?.referenced_entities.length, |  | ||||||
|                         } |  | ||||||
|                       )} |  | ||||||
|                     </button>` |  | ||||||
|                   : showEntities |  | ||||||
|                     ? html`<span class="main"> |  | ||||||
|                         ${this.hass.localize( |  | ||||||
|                           "ui.components.target-picker.entities_count", |  | ||||||
|                           { |  | ||||||
|                             count: entries?.referenced_entities.length, |  | ||||||
|                           } |  | ||||||
|                         )} |  | ||||||
|                       </span>` |  | ||||||
|                     : nothing} |  | ||||||
|                 ${showDevices |  | ||||||
|                   ? html`<span class="secondary" |  | ||||||
|                       >${this.hass.localize( |  | ||||||
|                         "ui.components.target-picker.devices_count", |  | ||||||
|                         { |  | ||||||
|                           count: entries?.referenced_devices.length, |  | ||||||
|                         } |  | ||||||
|                       )}</span |  | ||||||
|                     >` |  | ||||||
|                   : nothing} |  | ||||||
|                 ${this._domainName && !showDevices |  | ||||||
|                   ? html`<span class="secondary domain" |  | ||||||
|                       >${this._domainName}</span |  | ||||||
|                     >` |  | ||||||
|                   : nothing} |  | ||||||
|               </div> |  | ||||||
|             ` |  | ||||||
|           : nothing} |  | ||||||
|         ${!this.expand && !this.subEntry |  | ||||||
|           ? html` |  | ||||||
|               <ha-icon-button |  | ||||||
|                 .path=${mdiClose} |  | ||||||
|                 slot="end" |  | ||||||
|                 @click=${this._removeItem} |  | ||||||
|               ></ha-icon-button> |  | ||||||
|             ` |  | ||||||
|           : nothing} |  | ||||||
|       </ha-md-list-item> |  | ||||||
|       ${this.expand && entries && entries.referenced_entities |  | ||||||
|         ? this._renderEntries() |  | ||||||
|         : nothing} |  | ||||||
|     `; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private _renderEntries() { |  | ||||||
|     const entries = this.parentEntries || this._entries; |  | ||||||
|  |  | ||||||
|     let nextType: TargetType = |  | ||||||
|       this.type === "floor" |  | ||||||
|         ? "area" |  | ||||||
|         : this.type === "area" |  | ||||||
|           ? "device" |  | ||||||
|           : "entity"; |  | ||||||
|  |  | ||||||
|     if (this.type === "label") { |  | ||||||
|       if (entries?.referenced_areas.length) { |  | ||||||
|         nextType = "area"; |  | ||||||
|       } else if (entries?.referenced_devices.length) { |  | ||||||
|         nextType = "device"; |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     const rows1 = |  | ||||||
|       (nextType === "area" |  | ||||||
|         ? entries?.referenced_areas |  | ||||||
|         : nextType === "device" |  | ||||||
|           ? entries?.referenced_devices |  | ||||||
|           : entries?.referenced_entities) || []; |  | ||||||
|  |  | ||||||
|     const devicesInAreas = [] as string[]; |  | ||||||
|  |  | ||||||
|     const rows1Entries = |  | ||||||
|       nextType === "entity" |  | ||||||
|         ? undefined |  | ||||||
|         : rows1.map((rowItem) => { |  | ||||||
|             const nextEntries = { |  | ||||||
|               referenced_areas: [] as string[], |  | ||||||
|               referenced_devices: [] as string[], |  | ||||||
|               referenced_entities: [] as string[], |  | ||||||
|             }; |  | ||||||
|  |  | ||||||
|             if (nextType === "area") { |  | ||||||
|               nextEntries.referenced_devices = |  | ||||||
|                 entries?.referenced_devices.filter( |  | ||||||
|                   (device_id) => |  | ||||||
|                     this.hass.devices?.[device_id]?.area_id === rowItem && |  | ||||||
|                     entries?.referenced_entities.some( |  | ||||||
|                       (entity_id) => |  | ||||||
|                         this.hass.entities?.[entity_id]?.device_id === device_id |  | ||||||
|                     ) |  | ||||||
|                 ) || ([] as string[]); |  | ||||||
|  |  | ||||||
|               devicesInAreas.push(...nextEntries.referenced_devices); |  | ||||||
|  |  | ||||||
|               nextEntries.referenced_entities = |  | ||||||
|                 entries?.referenced_entities.filter((entity_id) => { |  | ||||||
|                   const entity = this.hass.entities[entity_id]; |  | ||||||
|                   return ( |  | ||||||
|                     entity.area_id === rowItem || |  | ||||||
|                     !entity.device_id || |  | ||||||
|                     nextEntries.referenced_devices.includes(entity.device_id) |  | ||||||
|                   ); |  | ||||||
|                 }) || ([] as string[]); |  | ||||||
|  |  | ||||||
|               return nextEntries; |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             nextEntries.referenced_entities = |  | ||||||
|               entries?.referenced_entities.filter( |  | ||||||
|                 (entity_id) => |  | ||||||
|                   this.hass.entities?.[entity_id]?.device_id === rowItem |  | ||||||
|               ) || ([] as string[]); |  | ||||||
|  |  | ||||||
|             return nextEntries; |  | ||||||
|           }); |  | ||||||
|  |  | ||||||
|     const entityRows = |  | ||||||
|       this.type === "label" && entries |  | ||||||
|         ? entries.referenced_entities.filter((entity_id) => |  | ||||||
|             this.hass.entities[entity_id].labels.includes(this.itemId) |  | ||||||
|           ) |  | ||||||
|         : nextType === "device" && entries |  | ||||||
|           ? entries.referenced_entities.filter( |  | ||||||
|               (entity_id) => |  | ||||||
|                 this.hass.entities[entity_id].area_id === this.itemId |  | ||||||
|             ) |  | ||||||
|           : []; |  | ||||||
|  |  | ||||||
|     const deviceRows = |  | ||||||
|       this.type === "label" && entries |  | ||||||
|         ? entries.referenced_devices.filter( |  | ||||||
|             (device_id) => |  | ||||||
|               !devicesInAreas.includes(device_id) && |  | ||||||
|               this.hass.devices[device_id].labels.includes(this.itemId) |  | ||||||
|           ) |  | ||||||
|         : []; |  | ||||||
|  |  | ||||||
|     const deviceRowsEntries = |  | ||||||
|       deviceRows.length === 0 |  | ||||||
|         ? undefined |  | ||||||
|         : deviceRows.map((device_id) => ({ |  | ||||||
|             referenced_areas: [] as string[], |  | ||||||
|             referenced_devices: [] as string[], |  | ||||||
|             referenced_entities: |  | ||||||
|               entries?.referenced_entities.filter( |  | ||||||
|                 (entity_id) => |  | ||||||
|                   this.hass.entities?.[entity_id]?.device_id === device_id |  | ||||||
|               ) || ([] as string[]), |  | ||||||
|           })); |  | ||||||
|  |  | ||||||
|     return html` |  | ||||||
|       <div class="entries-tree"> |  | ||||||
|         <div class="line-wrapper"> |  | ||||||
|           <div class="line"></div> |  | ||||||
|         </div> |  | ||||||
|         <ha-md-list class="entries"> |  | ||||||
|           ${rows1.map( |  | ||||||
|             (itemId, index) => html` |  | ||||||
|               <ha-target-picker-item-row |  | ||||||
|                 sub-entry |  | ||||||
|                 .hass=${this.hass} |  | ||||||
|                 .type=${nextType} |  | ||||||
|                 .itemId=${itemId} |  | ||||||
|                 .parentEntries=${rows1Entries?.[index]} |  | ||||||
|                 .hideContext=${this.hideContext || this.type !== "label"} |  | ||||||
|                 expand |  | ||||||
|               ></ha-target-picker-item-row> |  | ||||||
|             ` |  | ||||||
|           )} |  | ||||||
|           ${deviceRows.map( |  | ||||||
|             (itemId, index) => html` |  | ||||||
|               <ha-target-picker-item-row |  | ||||||
|                 sub-entry |  | ||||||
|                 .hass=${this.hass} |  | ||||||
|                 type="device" |  | ||||||
|                 .itemId=${itemId} |  | ||||||
|                 .parentEntries=${deviceRowsEntries?.[index]} |  | ||||||
|                 .hideContext=${this.hideContext || this.type !== "label"} |  | ||||||
|                 expand |  | ||||||
|               ></ha-target-picker-item-row> |  | ||||||
|             ` |  | ||||||
|           )} |  | ||||||
|           ${entityRows.map( |  | ||||||
|             (itemId) => html` |  | ||||||
|               <ha-target-picker-item-row |  | ||||||
|                 sub-entry |  | ||||||
|                 .hass=${this.hass} |  | ||||||
|                 type="entity" |  | ||||||
|                 .itemId=${itemId} |  | ||||||
|                 .hideContext=${this.hideContext || this.type !== "label"} |  | ||||||
|               ></ha-target-picker-item-row> |  | ||||||
|             ` |  | ||||||
|           )} |  | ||||||
|         </ha-md-list> |  | ||||||
|       </div> |  | ||||||
|     `; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private async _updateItemData() { |  | ||||||
|     if (this.type === "entity") { |  | ||||||
|       this._entries = undefined; |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
|     try { |  | ||||||
|       const entries = await extractFromTarget(this.hass, { |  | ||||||
|         [`${this.type}_id`]: [this.itemId], |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       const hiddenAreaIds: string[] = []; |  | ||||||
|       if (this.type === "floor" || this.type === "label") { |  | ||||||
|         entries.referenced_areas = entries.referenced_areas.filter( |  | ||||||
|           (area_id) => { |  | ||||||
|             const area = this.hass.areas[area_id]; |  | ||||||
|             if ( |  | ||||||
|               (this.type === "floor" || area.labels.includes(this.itemId)) && |  | ||||||
|               areaMeetsFilter( |  | ||||||
|                 area, |  | ||||||
|                 this.hass.devices, |  | ||||||
|                 this.hass.entities, |  | ||||||
|                 this.deviceFilter, |  | ||||||
|                 this.includeDomains, |  | ||||||
|                 this.includeDeviceClasses, |  | ||||||
|                 this.hass.states, |  | ||||||
|                 this.entityFilter |  | ||||||
|               ) |  | ||||||
|             ) { |  | ||||||
|               return true; |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             hiddenAreaIds.push(area_id); |  | ||||||
|             return false; |  | ||||||
|           } |  | ||||||
|         ); |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       const hiddenDeviceIds: string[] = []; |  | ||||||
|       if ( |  | ||||||
|         this.type === "floor" || |  | ||||||
|         this.type === "area" || |  | ||||||
|         this.type === "label" |  | ||||||
|       ) { |  | ||||||
|         entries.referenced_devices = entries.referenced_devices.filter( |  | ||||||
|           (device_id) => { |  | ||||||
|             const device = this.hass.devices[device_id]; |  | ||||||
|             if ( |  | ||||||
|               !hiddenAreaIds.includes(device.area_id || "") && |  | ||||||
|               (this.type !== "label" || device.labels.includes(this.itemId)) && |  | ||||||
|               deviceMeetsFilter( |  | ||||||
|                 device, |  | ||||||
|                 this.hass.entities, |  | ||||||
|                 this.deviceFilter, |  | ||||||
|                 this.includeDomains, |  | ||||||
|                 this.includeDeviceClasses, |  | ||||||
|                 this.hass.states, |  | ||||||
|                 this.entityFilter |  | ||||||
|               ) |  | ||||||
|             ) { |  | ||||||
|               return true; |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             hiddenDeviceIds.push(device_id); |  | ||||||
|             return false; |  | ||||||
|           } |  | ||||||
|         ); |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       entries.referenced_entities = entries.referenced_entities.filter( |  | ||||||
|         (entity_id) => { |  | ||||||
|           const entity = this.hass.entities[entity_id]; |  | ||||||
|           if (hiddenDeviceIds.includes(entity.device_id || "")) { |  | ||||||
|             return false; |  | ||||||
|           } |  | ||||||
|           if ( |  | ||||||
|             (this.type === "area" && entity.area_id === this.itemId) || |  | ||||||
|             (this.type === "label" && entity.labels.includes(this.itemId)) || |  | ||||||
|             entries.referenced_devices.includes(entity.device_id || "") |  | ||||||
|           ) { |  | ||||||
|             return entityRegMeetsFilter( |  | ||||||
|               entity, |  | ||||||
|               this.type === "label", |  | ||||||
|               this.includeDomains, |  | ||||||
|               this.includeDeviceClasses, |  | ||||||
|               this.hass.states, |  | ||||||
|               this.entityFilter |  | ||||||
|             ); |  | ||||||
|           } |  | ||||||
|           return false; |  | ||||||
|         } |  | ||||||
|       ); |  | ||||||
|  |  | ||||||
|       this._entries = entries; |  | ||||||
|     } catch (e) { |  | ||||||
|       // eslint-disable-next-line no-console |  | ||||||
|       console.error("Failed to extract target", e); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private _itemData = memoizeOne((type: TargetType, item: string) => { |  | ||||||
|     if (type === "floor") { |  | ||||||
|       const floor = this.hass.floors?.[item]; |  | ||||||
|       return { |  | ||||||
|         name: floor?.name || item, |  | ||||||
|         iconPath: floor?.icon, |  | ||||||
|         fallbackIconPath: floor ? floorDefaultIconPath(floor) : mdiHome, |  | ||||||
|       }; |  | ||||||
|     } |  | ||||||
|     if (type === "area") { |  | ||||||
|       const area = this.hass.areas?.[item]; |  | ||||||
|       return { |  | ||||||
|         name: area?.name || item, |  | ||||||
|         context: area.floor_id && this.hass.floors?.[area.floor_id]?.name, |  | ||||||
|         iconPath: area?.icon, |  | ||||||
|         fallbackIconPath: mdiTextureBox, |  | ||||||
|       }; |  | ||||||
|     } |  | ||||||
|     if (type === "device") { |  | ||||||
|       const device = this.hass.devices?.[item]; |  | ||||||
|  |  | ||||||
|       if (device.primary_config_entry) { |  | ||||||
|         this._getDeviceDomain(device.primary_config_entry); |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       return { |  | ||||||
|         name: device ? computeDeviceNameDisplay(device, this.hass) : item, |  | ||||||
|         context: device?.area_id && this.hass.areas?.[device.area_id]?.name, |  | ||||||
|         fallbackIconPath: mdiDevices, |  | ||||||
|       }; |  | ||||||
|     } |  | ||||||
|     if (type === "entity") { |  | ||||||
|       this._setDomainName(computeDomain(item)); |  | ||||||
|  |  | ||||||
|       const stateObject = this.hass.states[item]; |  | ||||||
|       const entityName = computeEntityName( |  | ||||||
|         stateObject, |  | ||||||
|         this.hass.entities, |  | ||||||
|         this.hass.devices |  | ||||||
|       ); |  | ||||||
|       const { area, device } = getEntityContext( |  | ||||||
|         stateObject, |  | ||||||
|         this.hass.entities, |  | ||||||
|         this.hass.devices, |  | ||||||
|         this.hass.areas, |  | ||||||
|         this.hass.floors |  | ||||||
|       ); |  | ||||||
|       const deviceName = device ? computeDeviceName(device) : undefined; |  | ||||||
|       const areaName = area ? computeAreaName(area) : undefined; |  | ||||||
|       const context = [areaName, entityName ? deviceName : undefined] |  | ||||||
|         .filter(Boolean) |  | ||||||
|         .join(computeRTL(this.hass) ? " ◂ " : " ▸ "); |  | ||||||
|       return { |  | ||||||
|         name: entityName || deviceName || item, |  | ||||||
|         context, |  | ||||||
|         stateObject, |  | ||||||
|       }; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // type label |  | ||||||
|     const label = this._labelRegistry.find((lab) => lab.label_id === item); |  | ||||||
|     return { |  | ||||||
|       name: label?.name || item, |  | ||||||
|       iconPath: label?.icon, |  | ||||||
|       fallbackIconPath: mdiLabel, |  | ||||||
|     }; |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   private _setDomainName(domain: string) { |  | ||||||
|     this._domainName = domainToName(this.hass.localize, domain); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private _removeItem(ev) { |  | ||||||
|     ev.stopPropagation(); |  | ||||||
|     fireEvent(this, "remove-target-item", { |  | ||||||
|       type: this.type, |  | ||||||
|       id: this.itemId, |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private async _getDeviceDomain(configEntryId: string) { |  | ||||||
|     try { |  | ||||||
|       const data = await getConfigEntry(this.hass, configEntryId); |  | ||||||
|       const domain = data.config_entry.domain; |  | ||||||
|       this._iconImg = brandsUrl({ |  | ||||||
|         domain: domain, |  | ||||||
|         type: "icon", |  | ||||||
|         darkOptimized: this.hass.themes?.darkMode, |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       this._setDomainName(domain); |  | ||||||
|     } catch { |  | ||||||
|       // failed to load config entry -> ignore |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private _openDetails() { |  | ||||||
|     showTargetDetailsDialog(this, { |  | ||||||
|       title: this._itemData(this.type, this.itemId).name, |  | ||||||
|       type: this.type, |  | ||||||
|       itemId: this.itemId, |  | ||||||
|       deviceFilter: this.deviceFilter, |  | ||||||
|       entityFilter: this.entityFilter, |  | ||||||
|       includeDomains: this.includeDomains, |  | ||||||
|       includeDeviceClasses: this.includeDeviceClasses, |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   static styles = [ |  | ||||||
|     buttonLinkStyle, |  | ||||||
|     css` |  | ||||||
|       :host { |  | ||||||
|         --md-list-item-top-space: var(--ha-space-0); |  | ||||||
|         --md-list-item-bottom-space: var(--ha-space-0); |  | ||||||
|         --md-list-item-leading-space: var(--ha-space-2); |  | ||||||
|         --md-list-item-trailing-space: var(--ha-space-2); |  | ||||||
|         --md-list-item-two-line-container-height: 56px; |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       :host([expand]:not([sub-entry])) ha-md-list-item { |  | ||||||
|         border: 2px solid var(--ha-color-border-neutral-loud); |  | ||||||
|         background-color: var(--ha-color-fill-neutral-quiet-resting); |  | ||||||
|         border-radius: var(--ha-card-border-radius, var(--ha-border-radius-lg)); |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       state-badge { |  | ||||||
|         color: var(--ha-color-on-neutral-quiet); |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       .icon { |  | ||||||
|         display: flex; |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       img { |  | ||||||
|         width: 24px; |  | ||||||
|         height: 24px; |  | ||||||
|       } |  | ||||||
|       ha-icon-button { |  | ||||||
|         --mdc-icon-button-size: 32px; |  | ||||||
|       } |  | ||||||
|       .summary { |  | ||||||
|         display: flex; |  | ||||||
|         flex-direction: column; |  | ||||||
|         align-items: flex-end; |  | ||||||
|         line-height: var(--ha-line-height-condensed); |  | ||||||
|       } |  | ||||||
|       :host([sub-entry]) .summary { |  | ||||||
|         margin-right: var(--ha-space-12); |  | ||||||
|       } |  | ||||||
|       .summary .main { |  | ||||||
|         font-weight: var(--ha-font-weight-medium); |  | ||||||
|       } |  | ||||||
|       .summary .secondary { |  | ||||||
|         font-size: var(--ha-font-size-s); |  | ||||||
|         color: var(--secondary-text-color); |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       .entries-tree { |  | ||||||
|         display: flex; |  | ||||||
|         position: relative; |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       .entries-tree .line-wrapper { |  | ||||||
|         padding: var(--ha-space-5); |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       .entries-tree .line-wrapper .line { |  | ||||||
|         border-left: 2px dashed var(--divider-color); |  | ||||||
|         height: calc(100% - 28px); |  | ||||||
|         position: absolute; |  | ||||||
|         top: 0; |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       :host([sub-entry]) .entries-tree .line-wrapper .line { |  | ||||||
|         height: calc(100% - 12px); |  | ||||||
|         top: -18px; |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       .entries { |  | ||||||
|         padding: 0; |  | ||||||
|         --md-item-overflow: visible; |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       .horizontal-line-wrapper { |  | ||||||
|         position: relative; |  | ||||||
|       } |  | ||||||
|       .horizontal-line-wrapper .horizontal-line { |  | ||||||
|         position: absolute; |  | ||||||
|         top: 11px; |  | ||||||
|         margin-inline-start: -28px; |  | ||||||
|         width: 29px; |  | ||||||
|         border-top: 2px dashed var(--divider-color); |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       button.link { |  | ||||||
|         text-decoration: none; |  | ||||||
|         color: var(--primary-color); |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       button.link:hover, |  | ||||||
|       button.link:focus { |  | ||||||
|         text-decoration: underline; |  | ||||||
|       } |  | ||||||
|     `, |  | ||||||
|   ]; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| declare global { |  | ||||||
|   interface HTMLElementTagNameMap { |  | ||||||
|     "ha-target-picker-item-row": HaTargetPickerItemRow; |  | ||||||
|   } |  | ||||||
| } |  | ||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,354 +0,0 @@ | |||||||
| import { consume } from "@lit/context"; |  | ||||||
| // @ts-ignore |  | ||||||
| import chipStyles from "@material/chips/dist/mdc.chips.min.css"; |  | ||||||
| import { |  | ||||||
|   mdiClose, |  | ||||||
|   mdiDevices, |  | ||||||
|   mdiHome, |  | ||||||
|   mdiLabel, |  | ||||||
|   mdiTextureBox, |  | ||||||
|   mdiUnfoldMoreVertical, |  | ||||||
| } from "@mdi/js"; |  | ||||||
| import { css, html, LitElement, nothing, unsafeCSS } from "lit"; |  | ||||||
| import { customElement, property, state } from "lit/decorators"; |  | ||||||
| import { classMap } from "lit/directives/class-map"; |  | ||||||
| import memoizeOne from "memoize-one"; |  | ||||||
| import { computeCssColor } from "../../common/color/compute-color"; |  | ||||||
| import { hex2rgb } from "../../common/color/convert-color"; |  | ||||||
| import { fireEvent } from "../../common/dom/fire_event"; |  | ||||||
| import { |  | ||||||
|   computeDeviceName, |  | ||||||
|   computeDeviceNameDisplay, |  | ||||||
| } from "../../common/entity/compute_device_name"; |  | ||||||
| import { computeDomain } from "../../common/entity/compute_domain"; |  | ||||||
| import { computeEntityName } from "../../common/entity/compute_entity_name"; |  | ||||||
| import { getEntityContext } from "../../common/entity/context/get_entity_context"; |  | ||||||
| import { getConfigEntry } from "../../data/config_entries"; |  | ||||||
| import { labelsContext } from "../../data/context"; |  | ||||||
| import { domainToName } from "../../data/integration"; |  | ||||||
| import type { LabelRegistryEntry } from "../../data/label_registry"; |  | ||||||
| import type { TargetType } from "../../data/target"; |  | ||||||
| import type { HomeAssistant } from "../../types"; |  | ||||||
| import { brandsUrl } from "../../util/brands-url"; |  | ||||||
| import { floorDefaultIconPath } from "../ha-floor-icon"; |  | ||||||
| import "../ha-icon"; |  | ||||||
| import "../ha-icon-button"; |  | ||||||
| import "../ha-md-list"; |  | ||||||
| import "../ha-md-list-item"; |  | ||||||
| import "../ha-state-icon"; |  | ||||||
| import "../ha-tooltip"; |  | ||||||
|  |  | ||||||
| @customElement("ha-target-picker-value-chip") |  | ||||||
| export class HaTargetPickerValueChip extends LitElement { |  | ||||||
|   @property({ attribute: false }) public hass!: HomeAssistant; |  | ||||||
|  |  | ||||||
|   @property() public type!: TargetType; |  | ||||||
|  |  | ||||||
|   @property({ attribute: "item-id" }) public itemId!: string; |  | ||||||
|  |  | ||||||
|   @state() private _domainName?: string; |  | ||||||
|  |  | ||||||
|   @state() private _iconImg?: string; |  | ||||||
|  |  | ||||||
|   @state() |  | ||||||
|   @consume({ context: labelsContext, subscribe: true }) |  | ||||||
|   _labelRegistry!: LabelRegistryEntry[]; |  | ||||||
|  |  | ||||||
|   protected render() { |  | ||||||
|     const { name, iconPath, fallbackIconPath, stateObject, color } = |  | ||||||
|       this._itemData(this.type, this.itemId); |  | ||||||
|  |  | ||||||
|     return html` |  | ||||||
|       <div |  | ||||||
|         class="mdc-chip ${classMap({ |  | ||||||
|           [this.type]: true, |  | ||||||
|         })}" |  | ||||||
|         style=${color |  | ||||||
|           ? `--color: rgb(${color}); --background-color: rgba(${color}, .5)` |  | ||||||
|           : ""} |  | ||||||
|       > |  | ||||||
|         ${iconPath |  | ||||||
|           ? html`<ha-icon |  | ||||||
|               class="mdc-chip__icon mdc-chip__icon--leading" |  | ||||||
|               .icon=${iconPath} |  | ||||||
|             ></ha-icon>` |  | ||||||
|           : this._iconImg |  | ||||||
|             ? html`<img |  | ||||||
|                 class="mdc-chip__icon mdc-chip__icon--leading" |  | ||||||
|                 alt=${this._domainName || ""} |  | ||||||
|                 crossorigin="anonymous" |  | ||||||
|                 referrerpolicy="no-referrer" |  | ||||||
|                 src=${this._iconImg} |  | ||||||
|               />` |  | ||||||
|             : fallbackIconPath |  | ||||||
|               ? html`<ha-svg-icon |  | ||||||
|                   class="mdc-chip__icon mdc-chip__icon--leading" |  | ||||||
|                   .path=${fallbackIconPath} |  | ||||||
|                 ></ha-svg-icon>` |  | ||||||
|               : stateObject |  | ||||||
|                 ? html`<ha-state-icon |  | ||||||
|                     class="mdc-chip__icon mdc-chip__icon--leading" |  | ||||||
|                     .hass=${this.hass} |  | ||||||
|                     .stateObj=${stateObject} |  | ||||||
|                   ></ha-state-icon>` |  | ||||||
|                 : nothing} |  | ||||||
|         <span role="gridcell"> |  | ||||||
|           <span role="button" tabindex="0" class="mdc-chip__primary-action"> |  | ||||||
|             <span id="title-${this.itemId}" class="mdc-chip__text" |  | ||||||
|               >${name}</span |  | ||||||
|             > |  | ||||||
|           </span> |  | ||||||
|         </span> |  | ||||||
|         ${this.type === "entity" |  | ||||||
|           ? nothing |  | ||||||
|           : html`<span role="gridcell"> |  | ||||||
|               <ha-tooltip .for="expand-${this.itemId}" |  | ||||||
|                 >${this.hass.localize( |  | ||||||
|                   `ui.components.target-picker.expand_${this.type}_id` |  | ||||||
|                 )} |  | ||||||
|               </ha-tooltip> |  | ||||||
|               <ha-icon-button |  | ||||||
|                 class="expand-btn mdc-chip__icon mdc-chip__icon--trailing" |  | ||||||
|                 .label=${this.hass.localize( |  | ||||||
|                   "ui.components.target-picker.expand" |  | ||||||
|                 )} |  | ||||||
|                 .path=${mdiUnfoldMoreVertical} |  | ||||||
|                 hide-title |  | ||||||
|                 .id="expand-${this.itemId}" |  | ||||||
|                 .type=${this.type} |  | ||||||
|                 @click=${this._handleExpand} |  | ||||||
|               ></ha-icon-button> |  | ||||||
|             </span>`} |  | ||||||
|         <span role="gridcell"> |  | ||||||
|           <ha-tooltip .for="remove-${this.itemId}"> |  | ||||||
|             ${this.hass.localize( |  | ||||||
|               `ui.components.target-picker.remove_${this.type}_id` |  | ||||||
|             )} |  | ||||||
|           </ha-tooltip> |  | ||||||
|           <ha-icon-button |  | ||||||
|             class="mdc-chip__icon mdc-chip__icon--trailing" |  | ||||||
|             .label=${this.hass.localize("ui.components.target-picker.remove")} |  | ||||||
|             .path=${mdiClose} |  | ||||||
|             hide-title |  | ||||||
|             .id="remove-${this.itemId}" |  | ||||||
|             .type=${this.type} |  | ||||||
|             @click=${this._removeItem} |  | ||||||
|           ></ha-icon-button> |  | ||||||
|         </span> |  | ||||||
|       </div> |  | ||||||
|     `; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private _itemData = memoizeOne((type: TargetType, itemId: string) => { |  | ||||||
|     if (type === "floor") { |  | ||||||
|       const floor = this.hass.floors?.[itemId]; |  | ||||||
|       return { |  | ||||||
|         name: floor?.name || itemId, |  | ||||||
|         iconPath: floor?.icon, |  | ||||||
|         fallbackIconPath: floor ? floorDefaultIconPath(floor) : mdiHome, |  | ||||||
|       }; |  | ||||||
|     } |  | ||||||
|     if (type === "area") { |  | ||||||
|       const area = this.hass.areas?.[itemId]; |  | ||||||
|       return { |  | ||||||
|         name: area?.name || itemId, |  | ||||||
|         iconPath: area?.icon, |  | ||||||
|         fallbackIconPath: mdiTextureBox, |  | ||||||
|       }; |  | ||||||
|     } |  | ||||||
|     if (type === "device") { |  | ||||||
|       const device = this.hass.devices?.[itemId]; |  | ||||||
|  |  | ||||||
|       if (device.primary_config_entry) { |  | ||||||
|         this._getDeviceDomain(device.primary_config_entry); |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       return { |  | ||||||
|         name: device ? computeDeviceNameDisplay(device, this.hass) : itemId, |  | ||||||
|         fallbackIconPath: mdiDevices, |  | ||||||
|       }; |  | ||||||
|     } |  | ||||||
|     if (type === "entity") { |  | ||||||
|       this._setDomainName(computeDomain(itemId)); |  | ||||||
|  |  | ||||||
|       const stateObject = this.hass.states[itemId]; |  | ||||||
|       const entityName = computeEntityName( |  | ||||||
|         stateObject, |  | ||||||
|         this.hass.entities, |  | ||||||
|         this.hass.devices |  | ||||||
|       ); |  | ||||||
|       const { device } = getEntityContext( |  | ||||||
|         stateObject, |  | ||||||
|         this.hass.entities, |  | ||||||
|         this.hass.devices, |  | ||||||
|         this.hass.areas, |  | ||||||
|         this.hass.floors |  | ||||||
|       ); |  | ||||||
|       const deviceName = device ? computeDeviceName(device) : undefined; |  | ||||||
|       return { |  | ||||||
|         name: entityName || deviceName || itemId, |  | ||||||
|         stateObject, |  | ||||||
|       }; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // type label |  | ||||||
|     const label = this._labelRegistry.find((lab) => lab.label_id === itemId); |  | ||||||
|     let color = label?.color ? computeCssColor(label.color) : undefined; |  | ||||||
|     if (color?.startsWith("var(")) { |  | ||||||
|       const computedStyles = getComputedStyle(this); |  | ||||||
|       color = computedStyles.getPropertyValue( |  | ||||||
|         color.substring(4, color.length - 1) |  | ||||||
|       ); |  | ||||||
|     } |  | ||||||
|     if (color?.startsWith("#")) { |  | ||||||
|       color = hex2rgb(color).join(","); |  | ||||||
|     } |  | ||||||
|     return { |  | ||||||
|       name: label?.name || itemId, |  | ||||||
|       iconPath: label?.icon, |  | ||||||
|       fallbackIconPath: mdiLabel, |  | ||||||
|       color, |  | ||||||
|     }; |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   private _setDomainName(domain: string) { |  | ||||||
|     this._domainName = domainToName(this.hass.localize, domain); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private async _getDeviceDomain(configEntryId: string) { |  | ||||||
|     try { |  | ||||||
|       const data = await getConfigEntry(this.hass, configEntryId); |  | ||||||
|       const domain = data.config_entry.domain; |  | ||||||
|       this._iconImg = brandsUrl({ |  | ||||||
|         domain: domain, |  | ||||||
|         type: "icon", |  | ||||||
|         darkOptimized: this.hass.themes?.darkMode, |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       this._setDomainName(domain); |  | ||||||
|     } catch { |  | ||||||
|       // failed to load config entry -> ignore |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private _removeItem(ev) { |  | ||||||
|     ev.stopPropagation(); |  | ||||||
|     fireEvent(this, "remove-target-item", { |  | ||||||
|       type: this.type, |  | ||||||
|       id: this.itemId, |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private _handleExpand(ev) { |  | ||||||
|     ev.stopPropagation(); |  | ||||||
|     fireEvent(this, "expand-target-item", { |  | ||||||
|       type: this.type, |  | ||||||
|       id: this.itemId, |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   static styles = css` |  | ||||||
|     ${unsafeCSS(chipStyles)} |  | ||||||
|     .mdc-chip { |  | ||||||
|       color: var(--primary-text-color); |  | ||||||
|     } |  | ||||||
|     .mdc-chip.add { |  | ||||||
|       color: rgba(0, 0, 0, 0.87); |  | ||||||
|     } |  | ||||||
|     .add-container { |  | ||||||
|       position: relative; |  | ||||||
|       display: inline-flex; |  | ||||||
|     } |  | ||||||
|     .mdc-chip:not(.add) { |  | ||||||
|       cursor: default; |  | ||||||
|     } |  | ||||||
|     .mdc-chip ha-icon-button { |  | ||||||
|       --mdc-icon-button-size: 24px; |  | ||||||
|       display: flex; |  | ||||||
|       align-items: center; |  | ||||||
|       outline: none; |  | ||||||
|     } |  | ||||||
|     .mdc-chip ha-icon-button ha-svg-icon { |  | ||||||
|       border-radius: 50%; |  | ||||||
|       background: var(--secondary-text-color); |  | ||||||
|     } |  | ||||||
|     .mdc-chip__icon.mdc-chip__icon--trailing { |  | ||||||
|       width: var(--ha-space-4); |  | ||||||
|       height: var(--ha-space-4); |  | ||||||
|       --mdc-icon-size: 14px; |  | ||||||
|       color: var(--secondary-text-color); |  | ||||||
|       margin-inline-start: var(--ha-space-1) !important; |  | ||||||
|       margin-inline-end: calc(-1 * var(--ha-space-1)) !important; |  | ||||||
|       direction: var(--direction); |  | ||||||
|     } |  | ||||||
|     .mdc-chip__icon--leading { |  | ||||||
|       display: flex; |  | ||||||
|       align-items: center; |  | ||||||
|       justify-content: center; |  | ||||||
|       --mdc-icon-size: 20px; |  | ||||||
|       border-radius: var(--ha-border-radius-circle); |  | ||||||
|       padding: 6px; |  | ||||||
|       margin-left: -13px !important; |  | ||||||
|       margin-inline-start: -13px !important; |  | ||||||
|       margin-inline-end: var(--ha-space-1) !important; |  | ||||||
|       direction: var(--direction); |  | ||||||
|     } |  | ||||||
|     .expand-btn { |  | ||||||
|       margin-right: var(--ha-space-0); |  | ||||||
|       margin-inline-end: var(--ha-space-0); |  | ||||||
|       margin-inline-start: initial; |  | ||||||
|     } |  | ||||||
|     .mdc-chip.area:not(.add), |  | ||||||
|     .mdc-chip.floor:not(.add) { |  | ||||||
|       border: 1px solid #fed6a4; |  | ||||||
|       background: var(--card-background-color); |  | ||||||
|     } |  | ||||||
|     .mdc-chip.area:not(.add) .mdc-chip__icon--leading, |  | ||||||
|     .mdc-chip.area.add, |  | ||||||
|     .mdc-chip.floor:not(.add) .mdc-chip__icon--leading, |  | ||||||
|     .mdc-chip.floor.add { |  | ||||||
|       background: #fed6a4; |  | ||||||
|     } |  | ||||||
|     .mdc-chip.device:not(.add) { |  | ||||||
|       border: 1px solid #a8e1fb; |  | ||||||
|       background: var(--card-background-color); |  | ||||||
|     } |  | ||||||
|     .mdc-chip.device:not(.add) .mdc-chip__icon--leading, |  | ||||||
|     .mdc-chip.device.add { |  | ||||||
|       background: #a8e1fb; |  | ||||||
|     } |  | ||||||
|     .mdc-chip.entity:not(.add) { |  | ||||||
|       border: 1px solid #d2e7b9; |  | ||||||
|       background: var(--card-background-color); |  | ||||||
|     } |  | ||||||
|     .mdc-chip.entity:not(.add) .mdc-chip__icon--leading, |  | ||||||
|     .mdc-chip.entity.add { |  | ||||||
|       background: #d2e7b9; |  | ||||||
|     } |  | ||||||
|     .mdc-chip.label:not(.add) { |  | ||||||
|       border: 1px solid var(--color, #e0e0e0); |  | ||||||
|       background: var(--card-background-color); |  | ||||||
|     } |  | ||||||
|     .mdc-chip.label:not(.add) .mdc-chip__icon--leading, |  | ||||||
|     .mdc-chip.label.add { |  | ||||||
|       background: var(--background-color, #e0e0e0); |  | ||||||
|     } |  | ||||||
|     .mdc-chip:hover { |  | ||||||
|       z-index: 5; |  | ||||||
|     } |  | ||||||
|     :host([disabled]) .mdc-chip { |  | ||||||
|       opacity: var(--light-disabled-opacity); |  | ||||||
|       pointer-events: none; |  | ||||||
|     } |  | ||||||
|     .tooltip-icon-img { |  | ||||||
|       width: 24px; |  | ||||||
|       height: 24px; |  | ||||||
|     } |  | ||||||
|   `; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| declare global { |  | ||||||
|   interface HTMLElementTagNameMap { |  | ||||||
|     "ha-target-picker-value-chip": HaTargetPickerValueChip; |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| @@ -1,260 +0,0 @@ | |||||||
| import { computeAreaName } from "../common/entity/compute_area_name"; |  | ||||||
| import { computeDomain } from "../common/entity/compute_domain"; |  | ||||||
| import { computeFloorName } from "../common/entity/compute_floor_name"; |  | ||||||
| import type { HaDevicePickerDeviceFilterFunc } from "../components/device/ha-device-picker"; |  | ||||||
| import type { PickerComboBoxItem } from "../components/ha-picker-combo-box"; |  | ||||||
| import type { HomeAssistant } from "../types"; |  | ||||||
| import type { AreaRegistryEntry } from "./area_registry"; |  | ||||||
| import { |  | ||||||
|   getDeviceEntityDisplayLookup, |  | ||||||
|   type DeviceEntityDisplayLookup, |  | ||||||
|   type DeviceRegistryEntry, |  | ||||||
| } from "./device_registry"; |  | ||||||
| import type { HaEntityPickerEntityFilterFunc } from "./entity"; |  | ||||||
| import type { EntityRegistryDisplayEntry } from "./entity_registry"; |  | ||||||
| import { |  | ||||||
|   floorCompare, |  | ||||||
|   getFloorAreaLookup, |  | ||||||
|   type FloorRegistryEntry, |  | ||||||
| } from "./floor_registry"; |  | ||||||
|  |  | ||||||
| export interface FloorComboBoxItem extends PickerComboBoxItem { |  | ||||||
|   type: "floor" | "area"; |  | ||||||
|   floor?: FloorRegistryEntry; |  | ||||||
|   area?: AreaRegistryEntry; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export interface AreaFloorValue { |  | ||||||
|   id: string; |  | ||||||
|   type: "floor" | "area"; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export const getAreasAndFloors = ( |  | ||||||
|   states: HomeAssistant["states"], |  | ||||||
|   haFloors: HomeAssistant["floors"], |  | ||||||
|   haAreas: HomeAssistant["areas"], |  | ||||||
|   haDevices: HomeAssistant["devices"], |  | ||||||
|   haEntities: HomeAssistant["entities"], |  | ||||||
|   formatId: (value: AreaFloorValue) => string, |  | ||||||
|   includeDomains?: string[], |  | ||||||
|   excludeDomains?: string[], |  | ||||||
|   includeDeviceClasses?: string[], |  | ||||||
|   deviceFilter?: HaDevicePickerDeviceFilterFunc, |  | ||||||
|   entityFilter?: HaEntityPickerEntityFilterFunc, |  | ||||||
|   excludeAreas?: string[], |  | ||||||
|   excludeFloors?: string[] |  | ||||||
| ): FloorComboBoxItem[] => { |  | ||||||
|   const floors = Object.values(haFloors); |  | ||||||
|   const areas = Object.values(haAreas); |  | ||||||
|   const devices = Object.values(haDevices); |  | ||||||
|   const entities = Object.values(haEntities); |  | ||||||
|  |  | ||||||
|   let deviceEntityLookup: DeviceEntityDisplayLookup = {}; |  | ||||||
|   let inputDevices: DeviceRegistryEntry[] | undefined; |  | ||||||
|   let inputEntities: EntityRegistryDisplayEntry[] | undefined; |  | ||||||
|  |  | ||||||
|   if ( |  | ||||||
|     includeDomains || |  | ||||||
|     excludeDomains || |  | ||||||
|     includeDeviceClasses || |  | ||||||
|     deviceFilter || |  | ||||||
|     entityFilter |  | ||||||
|   ) { |  | ||||||
|     deviceEntityLookup = getDeviceEntityDisplayLookup(entities); |  | ||||||
|     inputDevices = devices; |  | ||||||
|     inputEntities = entities.filter((entity) => entity.area_id); |  | ||||||
|  |  | ||||||
|     if (includeDomains) { |  | ||||||
|       inputDevices = inputDevices!.filter((device) => { |  | ||||||
|         const devEntities = deviceEntityLookup[device.id]; |  | ||||||
|         if (!devEntities || !devEntities.length) { |  | ||||||
|           return false; |  | ||||||
|         } |  | ||||||
|         return deviceEntityLookup[device.id].some((entity) => |  | ||||||
|           includeDomains.includes(computeDomain(entity.entity_id)) |  | ||||||
|         ); |  | ||||||
|       }); |  | ||||||
|       inputEntities = inputEntities!.filter((entity) => |  | ||||||
|         includeDomains.includes(computeDomain(entity.entity_id)) |  | ||||||
|       ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (excludeDomains) { |  | ||||||
|       inputDevices = inputDevices!.filter((device) => { |  | ||||||
|         const devEntities = deviceEntityLookup[device.id]; |  | ||||||
|         if (!devEntities || !devEntities.length) { |  | ||||||
|           return true; |  | ||||||
|         } |  | ||||||
|         return entities.every( |  | ||||||
|           (entity) => !excludeDomains.includes(computeDomain(entity.entity_id)) |  | ||||||
|         ); |  | ||||||
|       }); |  | ||||||
|       inputEntities = inputEntities!.filter( |  | ||||||
|         (entity) => !excludeDomains.includes(computeDomain(entity.entity_id)) |  | ||||||
|       ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (includeDeviceClasses) { |  | ||||||
|       inputDevices = inputDevices!.filter((device) => { |  | ||||||
|         const devEntities = deviceEntityLookup[device.id]; |  | ||||||
|         if (!devEntities || !devEntities.length) { |  | ||||||
|           return false; |  | ||||||
|         } |  | ||||||
|         return deviceEntityLookup[device.id].some((entity) => { |  | ||||||
|           const stateObj = states[entity.entity_id]; |  | ||||||
|           if (!stateObj) { |  | ||||||
|             return false; |  | ||||||
|           } |  | ||||||
|           return ( |  | ||||||
|             stateObj.attributes.device_class && |  | ||||||
|             includeDeviceClasses.includes(stateObj.attributes.device_class) |  | ||||||
|           ); |  | ||||||
|         }); |  | ||||||
|       }); |  | ||||||
|       inputEntities = inputEntities!.filter((entity) => { |  | ||||||
|         const stateObj = states[entity.entity_id]; |  | ||||||
|         return ( |  | ||||||
|           stateObj.attributes.device_class && |  | ||||||
|           includeDeviceClasses.includes(stateObj.attributes.device_class) |  | ||||||
|         ); |  | ||||||
|       }); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (deviceFilter) { |  | ||||||
|       inputDevices = inputDevices!.filter((device) => deviceFilter!(device)); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (entityFilter) { |  | ||||||
|       inputDevices = inputDevices!.filter((device) => { |  | ||||||
|         const devEntities = deviceEntityLookup[device.id]; |  | ||||||
|         if (!devEntities || !devEntities.length) { |  | ||||||
|           return false; |  | ||||||
|         } |  | ||||||
|         return deviceEntityLookup[device.id].some((entity) => { |  | ||||||
|           const stateObj = states[entity.entity_id]; |  | ||||||
|           if (!stateObj) { |  | ||||||
|             return false; |  | ||||||
|           } |  | ||||||
|           return entityFilter(stateObj); |  | ||||||
|         }); |  | ||||||
|       }); |  | ||||||
|       inputEntities = inputEntities!.filter((entity) => { |  | ||||||
|         const stateObj = states[entity.entity_id]; |  | ||||||
|         if (!stateObj) { |  | ||||||
|           return false; |  | ||||||
|         } |  | ||||||
|         return entityFilter!(stateObj); |  | ||||||
|       }); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   let outputAreas = areas; |  | ||||||
|  |  | ||||||
|   let areaIds: string[] | undefined; |  | ||||||
|  |  | ||||||
|   if (inputDevices) { |  | ||||||
|     areaIds = inputDevices |  | ||||||
|       .filter((device) => device.area_id) |  | ||||||
|       .map((device) => device.area_id!); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (inputEntities) { |  | ||||||
|     areaIds = (areaIds ?? []).concat( |  | ||||||
|       inputEntities |  | ||||||
|         .filter((entity) => entity.area_id) |  | ||||||
|         .map((entity) => entity.area_id!) |  | ||||||
|     ); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (areaIds) { |  | ||||||
|     outputAreas = outputAreas.filter((area) => areaIds!.includes(area.area_id)); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (excludeAreas) { |  | ||||||
|     outputAreas = outputAreas.filter( |  | ||||||
|       (area) => !excludeAreas!.includes(area.area_id) |  | ||||||
|     ); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (excludeFloors) { |  | ||||||
|     outputAreas = outputAreas.filter( |  | ||||||
|       (area) => !area.floor_id || !excludeFloors!.includes(area.floor_id) |  | ||||||
|     ); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   const floorAreaLookup = getFloorAreaLookup(outputAreas); |  | ||||||
|   const unassignedAreas = Object.values(outputAreas).filter( |  | ||||||
|     (area) => !area.floor_id || !floorAreaLookup[area.floor_id] |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   const compare = floorCompare(haFloors); |  | ||||||
|  |  | ||||||
|   // @ts-ignore |  | ||||||
|   const floorAreaEntries: [ |  | ||||||
|     FloorRegistryEntry | undefined, |  | ||||||
|     AreaRegistryEntry[], |  | ||||||
|   ][] = Object.entries(floorAreaLookup) |  | ||||||
|     .map(([floorId, floorAreas]) => { |  | ||||||
|       const floor = floors.find((fl) => fl.floor_id === floorId)!; |  | ||||||
|       return [floor, floorAreas] as const; |  | ||||||
|     }) |  | ||||||
|     .sort(([floorA], [floorB]) => compare(floorA.floor_id, floorB.floor_id)); |  | ||||||
|  |  | ||||||
|   const items: FloorComboBoxItem[] = []; |  | ||||||
|  |  | ||||||
|   floorAreaEntries.forEach(([floor, floorAreas]) => { |  | ||||||
|     if (floor) { |  | ||||||
|       const floorName = computeFloorName(floor); |  | ||||||
|  |  | ||||||
|       const areaSearchLabels = floorAreas |  | ||||||
|         .map((area) => { |  | ||||||
|           const areaName = computeAreaName(area) || area.area_id; |  | ||||||
|           return [area.area_id, areaName, ...area.aliases]; |  | ||||||
|         }) |  | ||||||
|         .flat(); |  | ||||||
|  |  | ||||||
|       items.push({ |  | ||||||
|         id: formatId({ id: floor.floor_id, type: "floor" }), |  | ||||||
|         type: "floor", |  | ||||||
|         primary: floorName, |  | ||||||
|         floor: floor, |  | ||||||
|         icon: floor.icon || undefined, |  | ||||||
|         search_labels: [ |  | ||||||
|           floor.floor_id, |  | ||||||
|           floorName, |  | ||||||
|           ...floor.aliases, |  | ||||||
|           ...areaSearchLabels, |  | ||||||
|         ], |  | ||||||
|       }); |  | ||||||
|     } |  | ||||||
|     items.push( |  | ||||||
|       ...floorAreas.map((area) => { |  | ||||||
|         const areaName = computeAreaName(area) || area.area_id; |  | ||||||
|         return { |  | ||||||
|           id: formatId({ id: area.area_id, type: "area" }), |  | ||||||
|           type: "area" as const, |  | ||||||
|           primary: areaName, |  | ||||||
|           area: area, |  | ||||||
|           icon: area.icon || undefined, |  | ||||||
|           search_labels: [area.area_id, areaName, ...area.aliases], |  | ||||||
|         }; |  | ||||||
|       }) |  | ||||||
|     ); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   items.push( |  | ||||||
|     ...unassignedAreas.map((area) => { |  | ||||||
|       const areaName = computeAreaName(area) || area.area_id; |  | ||||||
|       return { |  | ||||||
|         id: formatId({ id: area.area_id, type: "area" }), |  | ||||||
|         type: "area" as const, |  | ||||||
|         primary: areaName, |  | ||||||
|         area: area, |  | ||||||
|         icon: area.icon || undefined, |  | ||||||
|         search_labels: [area.area_id, areaName, ...area.aliases], |  | ||||||
|       }; |  | ||||||
|     }) |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   return items; |  | ||||||
| }; |  | ||||||
| @@ -79,7 +79,6 @@ export interface DataEntryFlowStepAbort { | |||||||
|   reason: string; |   reason: string; | ||||||
|   description_placeholders?: Record<string, string>; |   description_placeholders?: Record<string, string>; | ||||||
|   translation_domain?: string; |   translation_domain?: string; | ||||||
|   next_flow?: [FlowType, string]; // [flow_type, flow_id] |  | ||||||
| } | } | ||||||
|  |  | ||||||
| export interface DataEntryFlowStepProgress { | export interface DataEntryFlowStepProgress { | ||||||
|   | |||||||
| @@ -1,20 +1,12 @@ | |||||||
| import { computeAreaName } from "../common/entity/compute_area_name"; |  | ||||||
| import { computeDeviceNameDisplay } from "../common/entity/compute_device_name"; |  | ||||||
| import { computeDomain } from "../common/entity/compute_domain"; |  | ||||||
| import { computeStateName } from "../common/entity/compute_state_name"; | import { computeStateName } from "../common/entity/compute_state_name"; | ||||||
| import { getDeviceContext } from "../common/entity/context/get_device_context"; |  | ||||||
| import { caseInsensitiveStringCompare } from "../common/string/compare"; | import { caseInsensitiveStringCompare } from "../common/string/compare"; | ||||||
| import type { HaDevicePickerDeviceFilterFunc } from "../components/device/ha-device-picker"; |  | ||||||
| import type { PickerComboBoxItem } from "../components/ha-picker-combo-box"; |  | ||||||
| import type { HomeAssistant } from "../types"; | import type { HomeAssistant } from "../types"; | ||||||
| import type { ConfigEntry } from "./config_entries"; | import type { ConfigEntry } from "./config_entries"; | ||||||
| import type { HaEntityPickerEntityFilterFunc } from "./entity"; |  | ||||||
| import type { | import type { | ||||||
|   EntityRegistryDisplayEntry, |   EntityRegistryDisplayEntry, | ||||||
|   EntityRegistryEntry, |   EntityRegistryEntry, | ||||||
| } from "./entity_registry"; | } from "./entity_registry"; | ||||||
| import type { EntitySources } from "./entity_sources"; | import type { EntitySources } from "./entity_sources"; | ||||||
| import { domainToName } from "./integration"; |  | ||||||
| import type { RegistryEntry } from "./registry"; | import type { RegistryEntry } from "./registry"; | ||||||
|  |  | ||||||
| export { | export { | ||||||
| @@ -171,147 +163,3 @@ export const getDeviceIntegrationLookup = ( | |||||||
|   } |   } | ||||||
|   return deviceIntegrations; |   return deviceIntegrations; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| export interface DevicePickerItem extends PickerComboBoxItem { |  | ||||||
|   domain?: string; |  | ||||||
|   domain_name?: string; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export const getDevices = ( |  | ||||||
|   hass: HomeAssistant, |  | ||||||
|   configEntryLookup: Record<string, ConfigEntry>, |  | ||||||
|   includeDomains?: string[], |  | ||||||
|   excludeDomains?: string[], |  | ||||||
|   includeDeviceClasses?: string[], |  | ||||||
|   deviceFilter?: HaDevicePickerDeviceFilterFunc, |  | ||||||
|   entityFilter?: HaEntityPickerEntityFilterFunc, |  | ||||||
|   excludeDevices?: string[], |  | ||||||
|   value?: string |  | ||||||
| ): DevicePickerItem[] => { |  | ||||||
|   const devices = Object.values(hass.devices); |  | ||||||
|   const entities = Object.values(hass.entities); |  | ||||||
|  |  | ||||||
|   let deviceEntityLookup: DeviceEntityDisplayLookup = {}; |  | ||||||
|  |  | ||||||
|   if ( |  | ||||||
|     includeDomains || |  | ||||||
|     excludeDomains || |  | ||||||
|     includeDeviceClasses || |  | ||||||
|     entityFilter |  | ||||||
|   ) { |  | ||||||
|     deviceEntityLookup = getDeviceEntityDisplayLookup(entities); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   let inputDevices = devices.filter( |  | ||||||
|     (device) => device.id === value || !device.disabled_by |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   if (includeDomains) { |  | ||||||
|     inputDevices = inputDevices.filter((device) => { |  | ||||||
|       const devEntities = deviceEntityLookup[device.id]; |  | ||||||
|       if (!devEntities || !devEntities.length) { |  | ||||||
|         return false; |  | ||||||
|       } |  | ||||||
|       return deviceEntityLookup[device.id].some((entity) => |  | ||||||
|         includeDomains.includes(computeDomain(entity.entity_id)) |  | ||||||
|       ); |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (excludeDomains) { |  | ||||||
|     inputDevices = inputDevices.filter((device) => { |  | ||||||
|       const devEntities = deviceEntityLookup[device.id]; |  | ||||||
|       if (!devEntities || !devEntities.length) { |  | ||||||
|         return true; |  | ||||||
|       } |  | ||||||
|       return entities.every( |  | ||||||
|         (entity) => !excludeDomains.includes(computeDomain(entity.entity_id)) |  | ||||||
|       ); |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (excludeDevices) { |  | ||||||
|     inputDevices = inputDevices.filter( |  | ||||||
|       (device) => !excludeDevices!.includes(device.id) |  | ||||||
|     ); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (includeDeviceClasses) { |  | ||||||
|     inputDevices = inputDevices.filter((device) => { |  | ||||||
|       const devEntities = deviceEntityLookup[device.id]; |  | ||||||
|       if (!devEntities || !devEntities.length) { |  | ||||||
|         return false; |  | ||||||
|       } |  | ||||||
|       return deviceEntityLookup[device.id].some((entity) => { |  | ||||||
|         const stateObj = hass.states[entity.entity_id]; |  | ||||||
|         if (!stateObj) { |  | ||||||
|           return false; |  | ||||||
|         } |  | ||||||
|         return ( |  | ||||||
|           stateObj.attributes.device_class && |  | ||||||
|           includeDeviceClasses.includes(stateObj.attributes.device_class) |  | ||||||
|         ); |  | ||||||
|       }); |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (entityFilter) { |  | ||||||
|     inputDevices = inputDevices.filter((device) => { |  | ||||||
|       const devEntities = deviceEntityLookup[device.id]; |  | ||||||
|       if (!devEntities || !devEntities.length) { |  | ||||||
|         return false; |  | ||||||
|       } |  | ||||||
|       return devEntities.some((entity) => { |  | ||||||
|         const stateObj = hass.states[entity.entity_id]; |  | ||||||
|         if (!stateObj) { |  | ||||||
|           return false; |  | ||||||
|         } |  | ||||||
|         return entityFilter(stateObj); |  | ||||||
|       }); |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (deviceFilter) { |  | ||||||
|     inputDevices = inputDevices.filter( |  | ||||||
|       (device) => |  | ||||||
|         // We always want to include the device of the current value |  | ||||||
|         device.id === value || deviceFilter!(device) |  | ||||||
|     ); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   const outputDevices = inputDevices.map<DevicePickerItem>((device) => { |  | ||||||
|     const deviceName = computeDeviceNameDisplay( |  | ||||||
|       device, |  | ||||||
|       hass, |  | ||||||
|       deviceEntityLookup[device.id] |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     const { area } = getDeviceContext(device, hass); |  | ||||||
|  |  | ||||||
|     const areaName = area ? computeAreaName(area) : undefined; |  | ||||||
|  |  | ||||||
|     const configEntry = device.primary_config_entry |  | ||||||
|       ? configEntryLookup?.[device.primary_config_entry] |  | ||||||
|       : undefined; |  | ||||||
|  |  | ||||||
|     const domain = configEntry?.domain; |  | ||||||
|     const domainName = domain ? domainToName(hass.localize, domain) : undefined; |  | ||||||
|  |  | ||||||
|     return { |  | ||||||
|       id: device.id, |  | ||||||
|       label: "", |  | ||||||
|       primary: |  | ||||||
|         deviceName || |  | ||||||
|         hass.localize("ui.components.device-picker.unnamed_device"), |  | ||||||
|       secondary: areaName, |  | ||||||
|       domain: configEntry?.domain, |  | ||||||
|       domain_name: domainName, |  | ||||||
|       search_labels: [deviceName, areaName, domain, domainName].filter( |  | ||||||
|         Boolean |  | ||||||
|       ) as string[], |  | ||||||
|       sorting_label: deviceName || "zzz", |  | ||||||
|     }; |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   return outputDevices; |  | ||||||
| }; |  | ||||||
|   | |||||||
| @@ -102,6 +102,7 @@ export type EnergySolarForecasts = Record<string, EnergySolarForecast>; | |||||||
| export interface DeviceConsumptionEnergyPreference { | export interface DeviceConsumptionEnergyPreference { | ||||||
|   // This is an ever increasing value |   // This is an ever increasing value | ||||||
|   stat_consumption: string; |   stat_consumption: string; | ||||||
|  |   stat_power?: string; | ||||||
|   name?: string; |   name?: string; | ||||||
|   included_in_stat?: string; |   included_in_stat?: string; | ||||||
| } | } | ||||||
| @@ -130,11 +131,17 @@ export interface FlowToGridSourceEnergyPreference { | |||||||
|   number_energy_price: number | null; |   number_energy_price: number | null; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | export interface GridPowerSourceEnergyPreference { | ||||||
|  |   // W meter | ||||||
|  |   stat_power: string; | ||||||
|  | } | ||||||
|  |  | ||||||
| export interface GridSourceTypeEnergyPreference { | export interface GridSourceTypeEnergyPreference { | ||||||
|   type: "grid"; |   type: "grid"; | ||||||
|  |  | ||||||
|   flow_from: FlowFromGridSourceEnergyPreference[]; |   flow_from: FlowFromGridSourceEnergyPreference[]; | ||||||
|   flow_to: FlowToGridSourceEnergyPreference[]; |   flow_to: FlowToGridSourceEnergyPreference[]; | ||||||
|  |   power?: GridPowerSourceEnergyPreference[]; | ||||||
|  |  | ||||||
|   cost_adjustment_day: number; |   cost_adjustment_day: number; | ||||||
| } | } | ||||||
| @@ -143,6 +150,7 @@ export interface SolarSourceTypeEnergyPreference { | |||||||
|   type: "solar"; |   type: "solar"; | ||||||
|  |  | ||||||
|   stat_energy_from: string; |   stat_energy_from: string; | ||||||
|  |   stat_power?: string; | ||||||
|   config_entry_solar_forecast: string[] | null; |   config_entry_solar_forecast: string[] | null; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -150,6 +158,7 @@ export interface BatterySourceTypeEnergyPreference { | |||||||
|   type: "battery"; |   type: "battery"; | ||||||
|   stat_energy_from: string; |   stat_energy_from: string; | ||||||
|   stat_energy_to: string; |   stat_energy_to: string; | ||||||
|  |   stat_power?: string; | ||||||
| } | } | ||||||
| export interface GasSourceTypeEnergyPreference { | export interface GasSourceTypeEnergyPreference { | ||||||
|   type: "gas"; |   type: "gas"; | ||||||
| @@ -351,6 +360,35 @@ export const getReferencedStatisticIds = ( | |||||||
|   return statIDs; |   return statIDs; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | export const getReferencedStatisticIdsPower = ( | ||||||
|  |   prefs: EnergyPreferences | ||||||
|  | ): string[] => { | ||||||
|  |   const statIDs: (string | undefined)[] = []; | ||||||
|  |  | ||||||
|  |   for (const source of prefs.energy_sources) { | ||||||
|  |     if (source.type === "gas" || source.type === "water") { | ||||||
|  |       continue; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (source.type === "solar") { | ||||||
|  |       statIDs.push(source.stat_power); | ||||||
|  |       continue; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (source.type === "battery") { | ||||||
|  |       statIDs.push(source.stat_power); | ||||||
|  |       continue; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (source.power) { | ||||||
|  |       statIDs.push(...source.power.map((p) => p.stat_power)); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   statIDs.push(...prefs.device_consumption.map((d) => d.stat_power)); | ||||||
|  |  | ||||||
|  |   return statIDs.filter(Boolean) as string[]; | ||||||
|  | }; | ||||||
|  |  | ||||||
| export const enum CompareMode { | export const enum CompareMode { | ||||||
|   NONE = "", |   NONE = "", | ||||||
|   PREVIOUS = "previous", |   PREVIOUS = "previous", | ||||||
| @@ -398,9 +436,10 @@ const getEnergyData = async ( | |||||||
|     "gas", |     "gas", | ||||||
|     "device", |     "device", | ||||||
|   ]); |   ]); | ||||||
|  |   const powerStatIds = getReferencedStatisticIdsPower(prefs); | ||||||
|   const waterStatIds = getReferencedStatisticIds(prefs, info, ["water"]); |   const waterStatIds = getReferencedStatisticIds(prefs, info, ["water"]); | ||||||
|  |  | ||||||
|   const allStatIDs = [...energyStatIds, ...waterStatIds]; |   const allStatIDs = [...energyStatIds, ...waterStatIds, ...powerStatIds]; | ||||||
|  |  | ||||||
|   const dayDifference = differenceInDays(end || new Date(), start); |   const dayDifference = differenceInDays(end || new Date(), start); | ||||||
|   const period = |   const period = | ||||||
| @@ -411,6 +450,8 @@ const getEnergyData = async ( | |||||||
|       : dayDifference > 2 |       : dayDifference > 2 | ||||||
|         ? "day" |         ? "day" | ||||||
|         : "hour"; |         : "hour"; | ||||||
|  |   const finePeriod = | ||||||
|  |     dayDifference > 64 ? "day" : dayDifference > 8 ? "hour" : "5minute"; | ||||||
|  |  | ||||||
|   const statsMetadata: Record<string, StatisticsMetaData> = {}; |   const statsMetadata: Record<string, StatisticsMetaData> = {}; | ||||||
|   const statsMetadataArray = allStatIDs.length |   const statsMetadataArray = allStatIDs.length | ||||||
| @@ -432,6 +473,9 @@ const getEnergyData = async ( | |||||||
|       ? (gasUnit as (typeof VOLUME_UNITS)[number]) |       ? (gasUnit as (typeof VOLUME_UNITS)[number]) | ||||||
|       : undefined, |       : undefined, | ||||||
|   }; |   }; | ||||||
|  |   const powerUnits: StatisticsUnitConfiguration = { | ||||||
|  |     power: "kW", | ||||||
|  |   }; | ||||||
|   const waterUnit = getEnergyWaterUnit(hass, prefs, statsMetadata); |   const waterUnit = getEnergyWaterUnit(hass, prefs, statsMetadata); | ||||||
|   const waterUnits: StatisticsUnitConfiguration = { |   const waterUnits: StatisticsUnitConfiguration = { | ||||||
|     volume: waterUnit, |     volume: waterUnit, | ||||||
| @@ -442,6 +486,12 @@ const getEnergyData = async ( | |||||||
|         "change", |         "change", | ||||||
|       ]) |       ]) | ||||||
|     : {}; |     : {}; | ||||||
|  |   const _powerStats: Statistics | Promise<Statistics> = powerStatIds.length | ||||||
|  |     ? fetchStatistics(hass!, start, end, powerStatIds, finePeriod, powerUnits, [ | ||||||
|  |         "mean", | ||||||
|  |       ]) | ||||||
|  |     : {}; | ||||||
|  |  | ||||||
|   const _waterStats: Statistics | Promise<Statistics> = waterStatIds.length |   const _waterStats: Statistics | Promise<Statistics> = waterStatIds.length | ||||||
|     ? fetchStatistics(hass!, start, end, waterStatIds, period, waterUnits, [ |     ? fetchStatistics(hass!, start, end, waterStatIds, period, waterUnits, [ | ||||||
|         "change", |         "change", | ||||||
| @@ -548,6 +598,7 @@ const getEnergyData = async ( | |||||||
|  |  | ||||||
|   const [ |   const [ | ||||||
|     energyStats, |     energyStats, | ||||||
|  |     powerStats, | ||||||
|     waterStats, |     waterStats, | ||||||
|     energyStatsCompare, |     energyStatsCompare, | ||||||
|     waterStatsCompare, |     waterStatsCompare, | ||||||
| @@ -555,13 +606,14 @@ const getEnergyData = async ( | |||||||
|     fossilEnergyConsumptionCompare, |     fossilEnergyConsumptionCompare, | ||||||
|   ] = await Promise.all([ |   ] = await Promise.all([ | ||||||
|     _energyStats, |     _energyStats, | ||||||
|  |     _powerStats, | ||||||
|     _waterStats, |     _waterStats, | ||||||
|     _energyStatsCompare, |     _energyStatsCompare, | ||||||
|     _waterStatsCompare, |     _waterStatsCompare, | ||||||
|     _fossilEnergyConsumption, |     _fossilEnergyConsumption, | ||||||
|     _fossilEnergyConsumptionCompare, |     _fossilEnergyConsumptionCompare, | ||||||
|   ]); |   ]); | ||||||
|   const stats = { ...energyStats, ...waterStats }; |   const stats = { ...energyStats, ...waterStats, ...powerStats }; | ||||||
|   if (compare) { |   if (compare) { | ||||||
|     statsCompare = { ...energyStatsCompare, ...waterStatsCompare }; |     statsCompare = { ...energyStatsCompare, ...waterStatsCompare }; | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -1,4 +1,3 @@ | |||||||
| import type { HassEntity } from "home-assistant-js-websocket"; |  | ||||||
| import { arrayLiteralIncludes } from "../common/array/literal-includes"; | import { arrayLiteralIncludes } from "../common/array/literal-includes"; | ||||||
|  |  | ||||||
| export const UNAVAILABLE = "unavailable"; | export const UNAVAILABLE = "unavailable"; | ||||||
| @@ -11,5 +10,3 @@ export const OFF_STATES = [UNAVAILABLE, UNKNOWN, OFF] as const; | |||||||
|  |  | ||||||
| export const isUnavailableState = arrayLiteralIncludes(UNAVAILABLE_STATES); | export const isUnavailableState = arrayLiteralIncludes(UNAVAILABLE_STATES); | ||||||
| export const isOffState = arrayLiteralIncludes(OFF_STATES); | export const isOffState = arrayLiteralIncludes(OFF_STATES); | ||||||
|  |  | ||||||
| export type HaEntityPickerEntityFilterFunc = (entityId: HassEntity) => boolean; |  | ||||||
|   | |||||||
| @@ -1,17 +1,12 @@ | |||||||
| import type { Connection, HassEntity } from "home-assistant-js-websocket"; | import type { Connection } from "home-assistant-js-websocket"; | ||||||
| import { createCollection } from "home-assistant-js-websocket"; | import { createCollection } from "home-assistant-js-websocket"; | ||||||
| import type { Store } from "home-assistant-js-websocket/dist/store"; | import type { Store } from "home-assistant-js-websocket/dist/store"; | ||||||
| import memoizeOne from "memoize-one"; | import memoizeOne from "memoize-one"; | ||||||
| import { computeDomain } from "../common/entity/compute_domain"; | import { computeDomain } from "../common/entity/compute_domain"; | ||||||
| import { computeEntityNameList } from "../common/entity/compute_entity_name_display"; |  | ||||||
| import { computeStateName } from "../common/entity/compute_state_name"; | import { computeStateName } from "../common/entity/compute_state_name"; | ||||||
| import { caseInsensitiveStringCompare } from "../common/string/compare"; | import { caseInsensitiveStringCompare } from "../common/string/compare"; | ||||||
| import { computeRTL } from "../common/util/compute_rtl"; |  | ||||||
| import { debounce } from "../common/util/debounce"; | import { debounce } from "../common/util/debounce"; | ||||||
| import type { PickerComboBoxItem } from "../components/ha-picker-combo-box"; |  | ||||||
| import type { HomeAssistant } from "../types"; | import type { HomeAssistant } from "../types"; | ||||||
| import type { HaEntityPickerEntityFilterFunc } from "./entity"; |  | ||||||
| import { domainToName } from "./integration"; |  | ||||||
| import type { LightColor } from "./light"; | import type { LightColor } from "./light"; | ||||||
| import type { RegistryEntry } from "./registry"; | import type { RegistryEntry } from "./registry"; | ||||||
|  |  | ||||||
| @@ -329,122 +324,3 @@ export const getAutomaticEntityIds = ( | |||||||
|     type: "config/entity_registry/get_automatic_entity_ids", |     type: "config/entity_registry/get_automatic_entity_ids", | ||||||
|     entity_ids, |     entity_ids, | ||||||
|   }); |   }); | ||||||
|  |  | ||||||
| export interface EntityComboBoxItem extends PickerComboBoxItem { |  | ||||||
|   domain_name?: string; |  | ||||||
|   stateObj?: HassEntity; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export const getEntities = ( |  | ||||||
|   hass: HomeAssistant, |  | ||||||
|   includeDomains?: string[], |  | ||||||
|   excludeDomains?: string[], |  | ||||||
|   entityFilter?: HaEntityPickerEntityFilterFunc, |  | ||||||
|   includeDeviceClasses?: string[], |  | ||||||
|   includeUnitOfMeasurement?: string[], |  | ||||||
|   includeEntities?: string[], |  | ||||||
|   excludeEntities?: string[], |  | ||||||
|   value?: string |  | ||||||
| ): EntityComboBoxItem[] => { |  | ||||||
|   let items: EntityComboBoxItem[] = []; |  | ||||||
|  |  | ||||||
|   let entityIds = Object.keys(hass.states); |  | ||||||
|  |  | ||||||
|   if (includeEntities) { |  | ||||||
|     entityIds = entityIds.filter((entityId) => |  | ||||||
|       includeEntities.includes(entityId) |  | ||||||
|     ); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (excludeEntities) { |  | ||||||
|     entityIds = entityIds.filter( |  | ||||||
|       (entityId) => !excludeEntities.includes(entityId) |  | ||||||
|     ); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (includeDomains) { |  | ||||||
|     entityIds = entityIds.filter((eid) => |  | ||||||
|       includeDomains.includes(computeDomain(eid)) |  | ||||||
|     ); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (excludeDomains) { |  | ||||||
|     entityIds = entityIds.filter( |  | ||||||
|       (eid) => !excludeDomains.includes(computeDomain(eid)) |  | ||||||
|     ); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   items = entityIds.map<EntityComboBoxItem>((entityId) => { |  | ||||||
|     const stateObj = hass.states[entityId]; |  | ||||||
|  |  | ||||||
|     const friendlyName = computeStateName(stateObj); // Keep this for search |  | ||||||
|     const [entityName, deviceName, areaName] = computeEntityNameList( |  | ||||||
|       stateObj, |  | ||||||
|       [{ type: "entity" }, { type: "device" }, { type: "area" }], |  | ||||||
|       hass.entities, |  | ||||||
|       hass.devices, |  | ||||||
|       hass.areas, |  | ||||||
|       hass.floors |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     const domainName = domainToName(hass.localize, computeDomain(entityId)); |  | ||||||
|  |  | ||||||
|     const isRTL = computeRTL(hass); |  | ||||||
|  |  | ||||||
|     const primary = entityName || deviceName || entityId; |  | ||||||
|     const secondary = [areaName, entityName ? deviceName : undefined] |  | ||||||
|       .filter(Boolean) |  | ||||||
|       .join(isRTL ? " ◂ " : " ▸ "); |  | ||||||
|     const a11yLabel = [deviceName, entityName].filter(Boolean).join(" - "); |  | ||||||
|  |  | ||||||
|     return { |  | ||||||
|       id: entityId, |  | ||||||
|       primary: primary, |  | ||||||
|       secondary: secondary, |  | ||||||
|       domain_name: domainName, |  | ||||||
|       sorting_label: [deviceName, entityName].filter(Boolean).join("_"), |  | ||||||
|       search_labels: [ |  | ||||||
|         entityName, |  | ||||||
|         deviceName, |  | ||||||
|         areaName, |  | ||||||
|         domainName, |  | ||||||
|         friendlyName, |  | ||||||
|         entityId, |  | ||||||
|       ].filter(Boolean) as string[], |  | ||||||
|       a11y_label: a11yLabel, |  | ||||||
|       stateObj: stateObj, |  | ||||||
|     }; |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   if (includeDeviceClasses) { |  | ||||||
|     items = items.filter( |  | ||||||
|       (item) => |  | ||||||
|         // We always want to include the entity of the current value |  | ||||||
|         item.id === value || |  | ||||||
|         (item.stateObj?.attributes.device_class && |  | ||||||
|           includeDeviceClasses.includes(item.stateObj.attributes.device_class)) |  | ||||||
|     ); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (includeUnitOfMeasurement) { |  | ||||||
|     items = items.filter( |  | ||||||
|       (item) => |  | ||||||
|         // We always want to include the entity of the current value |  | ||||||
|         item.id === value || |  | ||||||
|         (item.stateObj?.attributes.unit_of_measurement && |  | ||||||
|           includeUnitOfMeasurement.includes( |  | ||||||
|             item.stateObj.attributes.unit_of_measurement |  | ||||||
|           )) |  | ||||||
|     ); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (entityFilter) { |  | ||||||
|     items = items.filter( |  | ||||||
|       (item) => |  | ||||||
|         // We always want to include the entity of the current value |  | ||||||
|         item.id === value || (item.stateObj && entityFilter!(item.stateObj)) |  | ||||||
|     ); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   return items; |  | ||||||
| }; |  | ||||||
|   | |||||||
| @@ -68,18 +68,13 @@ export const getFloorAreaLookup = ( | |||||||
| }; | }; | ||||||
|  |  | ||||||
| export const floorCompare = | export const floorCompare = | ||||||
|   (entries?: HomeAssistant["floors"], order?: string[]) => |   (entries?: FloorRegistryEntry[], order?: string[]) => | ||||||
|   (a: string, b: string) => { |   (a: string, b: string) => { | ||||||
|     const indexA = order ? order.indexOf(a) : -1; |     const indexA = order ? order.indexOf(a) : -1; | ||||||
|     const indexB = order ? order.indexOf(b) : -1; |     const indexB = order ? order.indexOf(b) : -1; | ||||||
|     if (indexA === -1 && indexB === -1) { |     if (indexA === -1 && indexB === -1) { | ||||||
|       const floorA = entries?.[a]; |       const nameA = entries?.[a]?.name ?? a; | ||||||
|       const floorB = entries?.[b]; |       const nameB = entries?.[b]?.name ?? b; | ||||||
|       if (floorA && floorB && floorA.level !== floorB.level) { |  | ||||||
|         return (floorA.level ?? 0) - (floorB.level ?? 0); |  | ||||||
|       } |  | ||||||
|       const nameA = floorA?.name ?? a; |  | ||||||
|       const nameB = floorB?.name ?? b; |  | ||||||
|       return stringCompare(nameA, nameB); |       return stringCompare(nameA, nameB); | ||||||
|     } |     } | ||||||
|     if (indexA === -1) { |     if (indexA === -1) { | ||||||
|   | |||||||
| @@ -1,20 +1,9 @@ | |||||||
| import { mdiLabel } from "@mdi/js"; |  | ||||||
| import type { Connection } from "home-assistant-js-websocket"; | import type { Connection } from "home-assistant-js-websocket"; | ||||||
| import { createCollection } from "home-assistant-js-websocket"; | import { createCollection } from "home-assistant-js-websocket"; | ||||||
| import type { Store } from "home-assistant-js-websocket/dist/store"; | import type { Store } from "home-assistant-js-websocket/dist/store"; | ||||||
| import { computeDomain } from "../common/entity/compute_domain"; |  | ||||||
| import { stringCompare } from "../common/string/compare"; | import { stringCompare } from "../common/string/compare"; | ||||||
| import { debounce } from "../common/util/debounce"; | import { debounce } from "../common/util/debounce"; | ||||||
| import type { HaDevicePickerDeviceFilterFunc } from "../components/device/ha-device-picker"; |  | ||||||
| import type { PickerComboBoxItem } from "../components/ha-picker-combo-box"; |  | ||||||
| import type { HomeAssistant } from "../types"; | import type { HomeAssistant } from "../types"; | ||||||
| import { |  | ||||||
|   getDeviceEntityDisplayLookup, |  | ||||||
|   type DeviceEntityDisplayLookup, |  | ||||||
|   type DeviceRegistryEntry, |  | ||||||
| } from "./device_registry"; |  | ||||||
| import type { HaEntityPickerEntityFilterFunc } from "./entity"; |  | ||||||
| import type { EntityRegistryDisplayEntry } from "./entity_registry"; |  | ||||||
| import type { RegistryEntry } from "./registry"; | import type { RegistryEntry } from "./registry"; | ||||||
|  |  | ||||||
| export interface LabelRegistryEntry extends RegistryEntry { | export interface LabelRegistryEntry extends RegistryEntry { | ||||||
| @@ -99,178 +88,3 @@ export const deleteLabelRegistryEntry = ( | |||||||
|     type: "config/label_registry/delete", |     type: "config/label_registry/delete", | ||||||
|     label_id: labelId, |     label_id: labelId, | ||||||
|   }); |   }); | ||||||
|  |  | ||||||
| export const getLabels = ( |  | ||||||
|   hass: HomeAssistant, |  | ||||||
|   labels?: LabelRegistryEntry[], |  | ||||||
|   includeDomains?: string[], |  | ||||||
|   excludeDomains?: string[], |  | ||||||
|   includeDeviceClasses?: string[], |  | ||||||
|   deviceFilter?: HaDevicePickerDeviceFilterFunc, |  | ||||||
|   entityFilter?: HaEntityPickerEntityFilterFunc, |  | ||||||
|   excludeLabels?: string[] |  | ||||||
| ): PickerComboBoxItem[] => { |  | ||||||
|   if (!labels || labels.length === 0) { |  | ||||||
|     return []; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   const devices = Object.values(hass.devices); |  | ||||||
|   const entities = Object.values(hass.entities); |  | ||||||
|  |  | ||||||
|   let deviceEntityLookup: DeviceEntityDisplayLookup = {}; |  | ||||||
|   let inputDevices: DeviceRegistryEntry[] | undefined; |  | ||||||
|   let inputEntities: EntityRegistryDisplayEntry[] | undefined; |  | ||||||
|  |  | ||||||
|   if ( |  | ||||||
|     includeDomains || |  | ||||||
|     excludeDomains || |  | ||||||
|     includeDeviceClasses || |  | ||||||
|     deviceFilter || |  | ||||||
|     entityFilter |  | ||||||
|   ) { |  | ||||||
|     deviceEntityLookup = getDeviceEntityDisplayLookup(entities); |  | ||||||
|     inputDevices = devices; |  | ||||||
|     inputEntities = entities.filter((entity) => entity.labels.length > 0); |  | ||||||
|  |  | ||||||
|     if (includeDomains) { |  | ||||||
|       inputDevices = inputDevices!.filter((device) => { |  | ||||||
|         const devEntities = deviceEntityLookup[device.id]; |  | ||||||
|         if (!devEntities || !devEntities.length) { |  | ||||||
|           return false; |  | ||||||
|         } |  | ||||||
|         return deviceEntityLookup[device.id].some((entity) => |  | ||||||
|           includeDomains.includes(computeDomain(entity.entity_id)) |  | ||||||
|         ); |  | ||||||
|       }); |  | ||||||
|       inputEntities = inputEntities!.filter((entity) => |  | ||||||
|         includeDomains.includes(computeDomain(entity.entity_id)) |  | ||||||
|       ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (excludeDomains) { |  | ||||||
|       inputDevices = inputDevices!.filter((device) => { |  | ||||||
|         const devEntities = deviceEntityLookup[device.id]; |  | ||||||
|         if (!devEntities || !devEntities.length) { |  | ||||||
|           return true; |  | ||||||
|         } |  | ||||||
|         return entities.every( |  | ||||||
|           (entity) => !excludeDomains.includes(computeDomain(entity.entity_id)) |  | ||||||
|         ); |  | ||||||
|       }); |  | ||||||
|       inputEntities = inputEntities!.filter( |  | ||||||
|         (entity) => !excludeDomains.includes(computeDomain(entity.entity_id)) |  | ||||||
|       ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (includeDeviceClasses) { |  | ||||||
|       inputDevices = inputDevices!.filter((device) => { |  | ||||||
|         const devEntities = deviceEntityLookup[device.id]; |  | ||||||
|         if (!devEntities || !devEntities.length) { |  | ||||||
|           return false; |  | ||||||
|         } |  | ||||||
|         return deviceEntityLookup[device.id].some((entity) => { |  | ||||||
|           const stateObj = hass.states[entity.entity_id]; |  | ||||||
|           if (!stateObj) { |  | ||||||
|             return false; |  | ||||||
|           } |  | ||||||
|           return ( |  | ||||||
|             stateObj.attributes.device_class && |  | ||||||
|             includeDeviceClasses.includes(stateObj.attributes.device_class) |  | ||||||
|           ); |  | ||||||
|         }); |  | ||||||
|       }); |  | ||||||
|       inputEntities = inputEntities!.filter((entity) => { |  | ||||||
|         const stateObj = hass.states[entity.entity_id]; |  | ||||||
|         return ( |  | ||||||
|           stateObj.attributes.device_class && |  | ||||||
|           includeDeviceClasses.includes(stateObj.attributes.device_class) |  | ||||||
|         ); |  | ||||||
|       }); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (deviceFilter) { |  | ||||||
|       inputDevices = inputDevices!.filter((device) => deviceFilter!(device)); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (entityFilter) { |  | ||||||
|       inputDevices = inputDevices!.filter((device) => { |  | ||||||
|         const devEntities = deviceEntityLookup[device.id]; |  | ||||||
|         if (!devEntities || !devEntities.length) { |  | ||||||
|           return false; |  | ||||||
|         } |  | ||||||
|         return deviceEntityLookup[device.id].some((entity) => { |  | ||||||
|           const stateObj = hass.states[entity.entity_id]; |  | ||||||
|           if (!stateObj) { |  | ||||||
|             return false; |  | ||||||
|           } |  | ||||||
|           return entityFilter(stateObj); |  | ||||||
|         }); |  | ||||||
|       }); |  | ||||||
|       inputEntities = inputEntities!.filter((entity) => { |  | ||||||
|         const stateObj = hass.states[entity.entity_id]; |  | ||||||
|         if (!stateObj) { |  | ||||||
|           return false; |  | ||||||
|         } |  | ||||||
|         return entityFilter!(stateObj); |  | ||||||
|       }); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   let outputLabels = labels; |  | ||||||
|   const usedLabels = new Set<string>(); |  | ||||||
|  |  | ||||||
|   let areaIds: string[] | undefined; |  | ||||||
|  |  | ||||||
|   if (inputDevices) { |  | ||||||
|     areaIds = inputDevices |  | ||||||
|       .filter((device) => device.area_id) |  | ||||||
|       .map((device) => device.area_id!); |  | ||||||
|  |  | ||||||
|     inputDevices.forEach((device) => { |  | ||||||
|       device.labels.forEach((label) => usedLabels.add(label)); |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (inputEntities) { |  | ||||||
|     areaIds = (areaIds ?? []).concat( |  | ||||||
|       inputEntities |  | ||||||
|         .filter((entity) => entity.area_id) |  | ||||||
|         .map((entity) => entity.area_id!) |  | ||||||
|     ); |  | ||||||
|     inputEntities.forEach((entity) => { |  | ||||||
|       entity.labels.forEach((label) => usedLabels.add(label)); |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (areaIds) { |  | ||||||
|     areaIds.forEach((areaId) => { |  | ||||||
|       const area = hass.areas[areaId]; |  | ||||||
|       area.labels.forEach((label) => usedLabels.add(label)); |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (excludeLabels) { |  | ||||||
|     outputLabels = outputLabels.filter( |  | ||||||
|       (label) => !excludeLabels!.includes(label.label_id) |  | ||||||
|     ); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (inputDevices || inputEntities) { |  | ||||||
|     outputLabels = outputLabels.filter((label) => |  | ||||||
|       usedLabels.has(label.label_id) |  | ||||||
|     ); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   const items = outputLabels.map<PickerComboBoxItem>((label) => ({ |  | ||||||
|     id: label.label_id, |  | ||||||
|     primary: label.name, |  | ||||||
|     icon: label.icon || undefined, |  | ||||||
|     icon_path: label.icon ? undefined : mdiLabel, |  | ||||||
|     sorting_label: label.name, |  | ||||||
|     search_labels: [label.name, label.label_id, label.description].filter( |  | ||||||
|       (v): v is string => Boolean(v) |  | ||||||
|     ), |  | ||||||
|   })); |  | ||||||
|  |  | ||||||
|   return items; |  | ||||||
| }; |  | ||||||
|   | |||||||
| @@ -1,164 +0,0 @@ | |||||||
| import type { HassServiceTarget } from "home-assistant-js-websocket"; |  | ||||||
| import { computeDomain } from "../common/entity/compute_domain"; |  | ||||||
| import type { HaDevicePickerDeviceFilterFunc } from "../components/device/ha-device-picker"; |  | ||||||
| import type { HomeAssistant } from "../types"; |  | ||||||
| import type { AreaRegistryEntry } from "./area_registry"; |  | ||||||
| import type { DeviceRegistryEntry } from "./device_registry"; |  | ||||||
| import type { HaEntityPickerEntityFilterFunc } from "./entity"; |  | ||||||
| import type { EntityRegistryDisplayEntry } from "./entity_registry"; |  | ||||||
|  |  | ||||||
| export type TargetType = "entity" | "device" | "area" | "label" | "floor"; |  | ||||||
| export type TargetTypeFloorless = Exclude<TargetType, "floor">; |  | ||||||
|  |  | ||||||
| export interface ExtractFromTargetResult { |  | ||||||
|   missing_areas: string[]; |  | ||||||
|   missing_devices: string[]; |  | ||||||
|   missing_floors: string[]; |  | ||||||
|   missing_labels: string[]; |  | ||||||
|   referenced_areas: string[]; |  | ||||||
|   referenced_devices: string[]; |  | ||||||
|   referenced_entities: string[]; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export interface ExtractFromTargetResultReferenced { |  | ||||||
|   referenced_areas: string[]; |  | ||||||
|   referenced_devices: string[]; |  | ||||||
|   referenced_entities: string[]; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export const extractFromTarget = async ( |  | ||||||
|   hass: HomeAssistant, |  | ||||||
|   target: HassServiceTarget |  | ||||||
| ) => |  | ||||||
|   hass.callWS<ExtractFromTargetResult>({ |  | ||||||
|     type: "extract_from_target", |  | ||||||
|     target, |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
| export const areaMeetsFilter = ( |  | ||||||
|   area: AreaRegistryEntry, |  | ||||||
|   devices: HomeAssistant["devices"], |  | ||||||
|   entities: HomeAssistant["entities"], |  | ||||||
|   deviceFilter?: HaDevicePickerDeviceFilterFunc, |  | ||||||
|   includeDomains?: string[], |  | ||||||
|   includeDeviceClasses?: string[], |  | ||||||
|   states?: HomeAssistant["states"], |  | ||||||
|   entityFilter?: HaEntityPickerEntityFilterFunc |  | ||||||
| ): boolean => { |  | ||||||
|   const areaDevices = Object.values(devices).filter( |  | ||||||
|     (device) => device.area_id === area.area_id |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   if ( |  | ||||||
|     areaDevices.some((device) => |  | ||||||
|       deviceMeetsFilter( |  | ||||||
|         device, |  | ||||||
|         entities, |  | ||||||
|         deviceFilter, |  | ||||||
|         includeDomains, |  | ||||||
|         includeDeviceClasses, |  | ||||||
|         states, |  | ||||||
|         entityFilter |  | ||||||
|       ) |  | ||||||
|     ) |  | ||||||
|   ) { |  | ||||||
|     return true; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   const areaEntities = Object.values(entities).filter( |  | ||||||
|     (entity) => entity.area_id === area.area_id |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   if ( |  | ||||||
|     areaEntities.some((entity) => |  | ||||||
|       entityRegMeetsFilter( |  | ||||||
|         entity, |  | ||||||
|         false, |  | ||||||
|         includeDomains, |  | ||||||
|         includeDeviceClasses, |  | ||||||
|         states, |  | ||||||
|         entityFilter |  | ||||||
|       ) |  | ||||||
|     ) |  | ||||||
|   ) { |  | ||||||
|     return true; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   return false; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export const deviceMeetsFilter = ( |  | ||||||
|   device: DeviceRegistryEntry, |  | ||||||
|   entities: HomeAssistant["entities"], |  | ||||||
|   deviceFilter?: HaDevicePickerDeviceFilterFunc, |  | ||||||
|   includeDomains?: string[], |  | ||||||
|   includeDeviceClasses?: string[], |  | ||||||
|   states?: HomeAssistant["states"], |  | ||||||
|   entityFilter?: HaEntityPickerEntityFilterFunc |  | ||||||
| ): boolean => { |  | ||||||
|   const devEntities = Object.values(entities).filter( |  | ||||||
|     (entity) => entity.device_id === device.id |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   if ( |  | ||||||
|     !devEntities.some((entity) => |  | ||||||
|       entityRegMeetsFilter( |  | ||||||
|         entity, |  | ||||||
|         false, |  | ||||||
|         includeDomains, |  | ||||||
|         includeDeviceClasses, |  | ||||||
|         states, |  | ||||||
|         entityFilter |  | ||||||
|       ) |  | ||||||
|     ) |  | ||||||
|   ) { |  | ||||||
|     return false; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (deviceFilter) { |  | ||||||
|     return deviceFilter(device); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   return true; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export const entityRegMeetsFilter = ( |  | ||||||
|   entity: EntityRegistryDisplayEntry, |  | ||||||
|   includeSecondary = false, |  | ||||||
|   includeDomains?: string[], |  | ||||||
|   includeDeviceClasses?: string[], |  | ||||||
|   states?: HomeAssistant["states"], |  | ||||||
|   entityFilter?: HaEntityPickerEntityFilterFunc |  | ||||||
| ): boolean => { |  | ||||||
|   if (entity.hidden || (entity.entity_category && !includeSecondary)) { |  | ||||||
|     return false; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if ( |  | ||||||
|     includeDomains && |  | ||||||
|     !includeDomains.includes(computeDomain(entity.entity_id)) |  | ||||||
|   ) { |  | ||||||
|     return false; |  | ||||||
|   } |  | ||||||
|   if (includeDeviceClasses) { |  | ||||||
|     const stateObj = states?.[entity.entity_id]; |  | ||||||
|     if (!stateObj) { |  | ||||||
|       return false; |  | ||||||
|     } |  | ||||||
|     if ( |  | ||||||
|       !stateObj.attributes.device_class || |  | ||||||
|       !includeDeviceClasses!.includes(stateObj.attributes.device_class) |  | ||||||
|     ) { |  | ||||||
|       return false; |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (entityFilter) { |  | ||||||
|     const stateObj = states?.[entity.entity_id]; |  | ||||||
|     if (!stateObj) { |  | ||||||
|       return false; |  | ||||||
|     } |  | ||||||
|     return entityFilter!(stateObj); |  | ||||||
|   } |  | ||||||
|   return true; |  | ||||||
| }; |  | ||||||
| @@ -1,13 +1,18 @@ | |||||||
| import type { Connection } from "home-assistant-js-websocket"; | import type { Connection } from "home-assistant-js-websocket"; | ||||||
| import { createCollection } from "home-assistant-js-websocket"; | import { createCollection } from "home-assistant-js-websocket"; | ||||||
| import type { Store } from "home-assistant-js-websocket/dist/store"; | import type { Store } from "home-assistant-js-websocket/dist/store"; | ||||||
|  | import { stringCompare } from "../common/string/compare"; | ||||||
| import { debounce } from "../common/util/debounce"; | import { debounce } from "../common/util/debounce"; | ||||||
| import type { AreaRegistryEntry } from "./area_registry"; | import type { AreaRegistryEntry } from "./area_registry"; | ||||||
|  |  | ||||||
| const fetchAreaRegistry = (conn: Connection) => | const fetchAreaRegistry = (conn: Connection) => | ||||||
|   conn.sendMessagePromise<AreaRegistryEntry[]>({ |   conn | ||||||
|     type: "config/area_registry/list", |     .sendMessagePromise<AreaRegistryEntry[]>({ | ||||||
|   }); |       type: "config/area_registry/list", | ||||||
|  |     }) | ||||||
|  |     .then((areas) => | ||||||
|  |       areas.sort((ent1, ent2) => stringCompare(ent1.name, ent2.name)) | ||||||
|  |     ); | ||||||
|  |  | ||||||
| const subscribeAreaRegistryUpdates = ( | const subscribeAreaRegistryUpdates = ( | ||||||
|   conn: Connection, |   conn: Connection, | ||||||
|   | |||||||
| @@ -1,13 +1,23 @@ | |||||||
| import type { Connection } from "home-assistant-js-websocket"; | import type { Connection } from "home-assistant-js-websocket"; | ||||||
| import { createCollection } from "home-assistant-js-websocket"; | import { createCollection } from "home-assistant-js-websocket"; | ||||||
| import type { Store } from "home-assistant-js-websocket/dist/store"; | import type { Store } from "home-assistant-js-websocket/dist/store"; | ||||||
|  | import { stringCompare } from "../common/string/compare"; | ||||||
| import { debounce } from "../common/util/debounce"; | import { debounce } from "../common/util/debounce"; | ||||||
| import type { FloorRegistryEntry } from "./floor_registry"; | import type { FloorRegistryEntry } from "./floor_registry"; | ||||||
|  |  | ||||||
| const fetchFloorRegistry = (conn: Connection) => | const fetchFloorRegistry = (conn: Connection) => | ||||||
|   conn.sendMessagePromise<FloorRegistryEntry[]>({ |   conn | ||||||
|     type: "config/floor_registry/list", |     .sendMessagePromise({ | ||||||
|   }); |       type: "config/floor_registry/list", | ||||||
|  |     }) | ||||||
|  |     .then((floors) => | ||||||
|  |       (floors as FloorRegistryEntry[]).sort((ent1, ent2) => { | ||||||
|  |         if (ent1.level !== ent2.level) { | ||||||
|  |           return (ent1.level ?? 9999) - (ent2.level ?? 9999); | ||||||
|  |         } | ||||||
|  |         return stringCompare(ent1.name, ent2.name); | ||||||
|  |       }) | ||||||
|  |     ); | ||||||
|  |  | ||||||
| const subscribeFloorRegistryUpdates = ( | const subscribeFloorRegistryUpdates = ( | ||||||
|   conn: Connection, |   conn: Connection, | ||||||
|   | |||||||
| @@ -472,10 +472,7 @@ class DataEntryFlowDialog extends LitElement { | |||||||
|     this._step = undefined; |     this._step = undefined; | ||||||
|     await this.updateComplete; |     await this.updateComplete; | ||||||
|     this._step = _step; |     this._step = _step; | ||||||
|     if ( |     if (_step.type === "create_entry" && _step.next_flow) { | ||||||
|       (_step.type === "create_entry" || _step.type === "abort") && |  | ||||||
|       _step.next_flow |  | ||||||
|     ) { |  | ||||||
|       // skip device rename if there is a chained flow |       // skip device rename if there is a chained flow | ||||||
|       this._step = undefined; |       this._step = undefined; | ||||||
|       this._handler = undefined; |       this._handler = undefined; | ||||||
| @@ -489,36 +486,32 @@ class DataEntryFlowDialog extends LitElement { | |||||||
|           carryOverDevices: this._devices( |           carryOverDevices: this._devices( | ||||||
|             this._params!.flowConfig.showDevices, |             this._params!.flowConfig.showDevices, | ||||||
|             Object.values(this.hass.devices), |             Object.values(this.hass.devices), | ||||||
|             _step.type === "create_entry" ? _step.result?.entry_id : undefined, |             _step.result?.entry_id, | ||||||
|             this._params!.carryOverDevices |             this._params!.carryOverDevices | ||||||
|           ).map((device) => device.id), |           ).map((device) => device.id), | ||||||
|           dialogClosedCallback: this._params!.dialogClosedCallback, |           dialogClosedCallback: this._params!.dialogClosedCallback, | ||||||
|         }); |         }); | ||||||
|       } else if (_step.next_flow[0] === "options_flow") { |       } else if (_step.next_flow[0] === "options_flow") { | ||||||
|         if (_step.type === "create_entry") { |         showOptionsFlowDialog( | ||||||
|           showOptionsFlowDialog( |           this._params!.dialogParentElement!, | ||||||
|             this._params!.dialogParentElement!, |           _step.result!, | ||||||
|             _step.result!, |           { | ||||||
|             { |             continueFlowId: _step.next_flow[1], | ||||||
|               continueFlowId: _step.next_flow[1], |             navigateToResult: this._params!.navigateToResult, | ||||||
|               navigateToResult: this._params!.navigateToResult, |             dialogClosedCallback: this._params!.dialogClosedCallback, | ||||||
|               dialogClosedCallback: this._params!.dialogClosedCallback, |           } | ||||||
|             } |         ); | ||||||
|           ); |  | ||||||
|         } |  | ||||||
|       } else if (_step.next_flow[0] === "config_subentries_flow") { |       } else if (_step.next_flow[0] === "config_subentries_flow") { | ||||||
|         if (_step.type === "create_entry") { |         showSubConfigFlowDialog( | ||||||
|           showSubConfigFlowDialog( |           this._params!.dialogParentElement!, | ||||||
|             this._params!.dialogParentElement!, |           _step.result!, | ||||||
|             _step.result!, |           _step.next_flow[0], | ||||||
|             _step.next_flow[0], |           { | ||||||
|             { |             continueFlowId: _step.next_flow[1], | ||||||
|               continueFlowId: _step.next_flow[1], |             navigateToResult: this._params!.navigateToResult, | ||||||
|               navigateToResult: this._params!.navigateToResult, |             dialogClosedCallback: this._params!.dialogClosedCallback, | ||||||
|               dialogClosedCallback: this._params!.dialogClosedCallback, |           } | ||||||
|             } |         ); | ||||||
|           ); |  | ||||||
|         } |  | ||||||
|       } else { |       } else { | ||||||
|         this.closeDialog(); |         this.closeDialog(); | ||||||
|         showAlertDialog(this._params!.dialogParentElement!, { |         showAlertDialog(this._params!.dialogParentElement!, { | ||||||
|   | |||||||
| @@ -77,80 +77,84 @@ class MoreInfoMediaPlayer extends LitElement { | |||||||
|       return nothing; |       return nothing; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     if (!stateActive(this.stateObj)) { | ||||||
|  |       return nothing; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     const supportsMute = supportsFeature( |     const supportsMute = supportsFeature( | ||||||
|       this.stateObj, |       this.stateObj, | ||||||
|       MediaPlayerEntityFeature.VOLUME_MUTE |       MediaPlayerEntityFeature.VOLUME_MUTE | ||||||
|     ); |     ); | ||||||
|     const supportsSliding = supportsFeature( |     const supportsSet = supportsFeature( | ||||||
|       this.stateObj, |       this.stateObj, | ||||||
|       MediaPlayerEntityFeature.VOLUME_SET |       MediaPlayerEntityFeature.VOLUME_SET | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|     return html`${(supportsFeature( |     const supportsStep = supportsFeature( | ||||||
|       this.stateObj!, |       this.stateObj, | ||||||
|       MediaPlayerEntityFeature.VOLUME_SET |       MediaPlayerEntityFeature.VOLUME_STEP | ||||||
|     ) || |     ); | ||||||
|       supportsFeature(this.stateObj!, MediaPlayerEntityFeature.VOLUME_STEP)) && |  | ||||||
|     stateActive(this.stateObj!) |     if (!supportsMute && !supportsSet && !supportsStep) { | ||||||
|       ? html` |       return nothing; | ||||||
|           <div class="volume"> |     } | ||||||
|             ${supportsMute |  | ||||||
|               ? html` |     return html` | ||||||
|                   <ha-icon-button |       <div class="volume"> | ||||||
|                     .path=${this.stateObj.attributes.is_volume_muted |         ${supportsMute | ||||||
|                       ? mdiVolumeOff |           ? html` | ||||||
|                       : mdiVolumeHigh} |               <ha-icon-button | ||||||
|                     .label=${this.hass.localize( |                 .path=${this.stateObj.attributes.is_volume_muted | ||||||
|                       `ui.card.media_player.${ |                   ? mdiVolumeOff | ||||||
|                         this.stateObj.attributes.is_volume_muted |                   : mdiVolumeHigh} | ||||||
|                           ? "media_volume_unmute" |                 .label=${this.hass.localize( | ||||||
|                           : "media_volume_mute" |                   `ui.card.media_player.${ | ||||||
|                       }` |                     this.stateObj.attributes.is_volume_muted | ||||||
|                     )} |                       ? "media_volume_unmute" | ||||||
|                     @click=${this._toggleMute} |                       : "media_volume_mute" | ||||||
|                   ></ha-icon-button> |                   }` | ||||||
|                 ` |                 )} | ||||||
|               : ""} |                 @click=${this._toggleMute} | ||||||
|             ${supportsFeature( |               ></ha-icon-button> | ||||||
|               this.stateObj, |             ` | ||||||
|               MediaPlayerEntityFeature.VOLUME_STEP |           : nothing} | ||||||
|             ) && !supportsSliding |         ${supportsStep | ||||||
|               ? html` |           ? html` <ha-icon-button | ||||||
|                   <ha-icon-button |               action="volume_down" | ||||||
|                     action="volume_down" |               .path=${mdiVolumeMinus} | ||||||
|                     .path=${mdiVolumeMinus} |               .label=${this.hass.localize( | ||||||
|                     .label=${this.hass.localize( |                 "ui.card.media_player.media_volume_down" | ||||||
|                       "ui.card.media_player.media_volume_down" |               )} | ||||||
|                     )} |               @click=${this._handleClick} | ||||||
|                     @click=${this._handleClick} |             ></ha-icon-button>` | ||||||
|                   ></ha-icon-button> |           : nothing} | ||||||
|                   <ha-icon-button |         ${supportsSet | ||||||
|                     action="volume_up" |           ? html` | ||||||
|                     .path=${mdiVolumePlus} |               ${!supportsMute && !supportsStep | ||||||
|                     .label=${this.hass.localize( |                 ? html`<ha-svg-icon .path=${mdiVolumeHigh}></ha-svg-icon>` | ||||||
|                       "ui.card.media_player.media_volume_up" |                 : nothing} | ||||||
|                     )} |               <ha-slider | ||||||
|                     @click=${this._handleClick} |                 labeled | ||||||
|                   ></ha-icon-button> |                 id="input" | ||||||
|                 ` |                 .value=${Number(this.stateObj.attributes.volume_level) * 100} | ||||||
|               : nothing} |                 @change=${this._selectedValueChanged} | ||||||
|             ${supportsSliding |               ></ha-slider> | ||||||
|               ? html` |             ` | ||||||
|                   ${!supportsMute |           : nothing} | ||||||
|                     ? html`<ha-svg-icon .path=${mdiVolumeHigh}></ha-svg-icon>` |         ${supportsStep | ||||||
|                     : nothing} |           ? html` | ||||||
|                   <ha-slider |               <ha-icon-button | ||||||
|                     labeled |                 action="volume_up" | ||||||
|                     id="input" |                 .path=${mdiVolumePlus} | ||||||
|                     .value=${Number(this.stateObj.attributes.volume_level) * |                 .label=${this.hass.localize( | ||||||
|                     100} |                   "ui.card.media_player.media_volume_up" | ||||||
|                     @change=${this._selectedValueChanged} |                 )} | ||||||
|                   ></ha-slider> |                 @click=${this._handleClick} | ||||||
|                 ` |               ></ha-icon-button> | ||||||
|               : nothing} |             ` | ||||||
|           </div> |           : nothing} | ||||||
|         ` |       </div> | ||||||
|       : nothing}`; |     `; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   protected _renderSourceControl() { |   protected _renderSourceControl() { | ||||||
|   | |||||||
| @@ -15,6 +15,7 @@ import type { PropertyValues } from "lit"; | |||||||
| import { LitElement, css, html, nothing } from "lit"; | import { LitElement, css, html, nothing } from "lit"; | ||||||
| import { customElement, property, state } from "lit/decorators"; | import { customElement, property, state } from "lit/decorators"; | ||||||
| import { cache } from "lit/directives/cache"; | import { cache } from "lit/directives/cache"; | ||||||
|  | import { join } from "lit/directives/join"; | ||||||
| import { keyed } from "lit/directives/keyed"; | import { keyed } from "lit/directives/keyed"; | ||||||
| import { dynamicElement } from "../../common/dom/dynamic-element-directive"; | import { dynamicElement } from "../../common/dom/dynamic-element-directive"; | ||||||
| import { fireEvent } from "../../common/dom/fire_event"; | import { fireEvent } from "../../common/dom/fire_event"; | ||||||
| @@ -32,7 +33,6 @@ import { | |||||||
| } from "../../common/entity/context/get_entity_context"; | } from "../../common/entity/context/get_entity_context"; | ||||||
| import { shouldHandleRequestSelectedEvent } from "../../common/mwc/handle-request-selected-event"; | import { shouldHandleRequestSelectedEvent } from "../../common/mwc/handle-request-selected-event"; | ||||||
| import { navigate } from "../../common/navigate"; | import { navigate } from "../../common/navigate"; | ||||||
| import { computeRTL } from "../../common/util/compute_rtl"; |  | ||||||
| import "../../components/ha-button-menu"; | import "../../components/ha-button-menu"; | ||||||
| import "../../components/ha-dialog"; | import "../../components/ha-dialog"; | ||||||
| import "../../components/ha-dialog-header"; | import "../../components/ha-dialog-header"; | ||||||
| @@ -361,8 +361,6 @@ export class MoreInfoDialog extends LitElement { | |||||||
|     ); |     ); | ||||||
|     const title = this._childView?.viewTitle || breadcrumb.pop() || entityId; |     const title = this._childView?.viewTitle || breadcrumb.pop() || entityId; | ||||||
|  |  | ||||||
|     const isRTL = computeRTL(this.hass); |  | ||||||
|  |  | ||||||
|     return html` |     return html` | ||||||
|       <ha-dialog |       <ha-dialog | ||||||
|         open |         open | ||||||
| @@ -396,13 +394,17 @@ export class MoreInfoDialog extends LitElement { | |||||||
|             ${breadcrumb.length > 0 |             ${breadcrumb.length > 0 | ||||||
|               ? !__DEMO__ && isAdmin |               ? !__DEMO__ && isAdmin | ||||||
|                 ? html` |                 ? html` | ||||||
|                     <button class="breadcrumb" @click=${this._breadcrumbClick}> |                     <button | ||||||
|                       ${breadcrumb.join(isRTL ? " ◂ " : " ▸ ")} |                       class="breadcrumb" | ||||||
|  |                       @click=${this._breadcrumbClick} | ||||||
|  |                       aria-label=${breadcrumb.join(" > ")} | ||||||
|  |                     > | ||||||
|  |                       ${join(breadcrumb, html`<ha-icon-next></ha-icon-next>`)} | ||||||
|                     </button> |                     </button> | ||||||
|                   ` |                   ` | ||||||
|                 : html` |                 : html` | ||||||
|                     <p class="breadcrumb"> |                     <p class="breadcrumb"> | ||||||
|                       ${breadcrumb.join(isRTL ? " ◂ " : " ▸ ")} |                       ${join(breadcrumb, html`<ha-icon-next></ha-icon-next>`)} | ||||||
|                     </p> |                     </p> | ||||||
|                   ` |                   ` | ||||||
|               : nothing} |               : nothing} | ||||||
|   | |||||||
| @@ -1,10 +1,14 @@ | |||||||
|  | import { mdiClose } from "@mdi/js"; | ||||||
| import type { UnsubscribeFunc } from "home-assistant-js-websocket"; | import type { UnsubscribeFunc } from "home-assistant-js-websocket"; | ||||||
| import type { CSSResultGroup } from "lit"; | import type { CSSResultGroup } from "lit"; | ||||||
| import { LitElement, css, html } from "lit"; | import { LitElement, css, html, nothing } from "lit"; | ||||||
| import { customElement, property, state } from "lit/decorators"; | import { customElement, property, query, state } from "lit/decorators"; | ||||||
| import { fireEvent } from "../../common/dom/fire_event"; | import { fireEvent } from "../../common/dom/fire_event"; | ||||||
| import "../../components/ha-alert"; | import "../../components/ha-alert"; | ||||||
| import "../../components/ha-wa-dialog"; | import "../../components/ha-dialog-header"; | ||||||
|  | import "../../components/ha-icon-button"; | ||||||
|  | import "../../components/ha-md-dialog"; | ||||||
|  | import type { HaMdDialog } from "../../components/ha-md-dialog"; | ||||||
| import "../../components/ha-spinner"; | import "../../components/ha-spinner"; | ||||||
| import { | import { | ||||||
|   subscribeBackupEvents, |   subscribeBackupEvents, | ||||||
| @@ -33,6 +37,8 @@ class DialogRestartWait extends LitElement { | |||||||
|  |  | ||||||
|   private _backupEventsSubscription?: Promise<UnsubscribeFunc>; |   private _backupEventsSubscription?: Promise<UnsubscribeFunc>; | ||||||
|  |  | ||||||
|  |   @query("ha-md-dialog") private _dialog?: HaMdDialog; | ||||||
|  |  | ||||||
|   public async showDialog(params: RestartWaitDialogParams): Promise<void> { |   public async showDialog(params: RestartWaitDialogParams): Promise<void> { | ||||||
|     this._open = true; |     this._open = true; | ||||||
|     this._loadBackupState(); |     this._loadBackupState(); | ||||||
| @@ -43,11 +49,9 @@ class DialogRestartWait extends LitElement { | |||||||
|     this._actionOnIdle = params.action; |     this._actionOnIdle = params.action; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   public closeDialog(): void { |  | ||||||
|     this._open = false; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private _dialogClosed(): void { |   private _dialogClosed(): void { | ||||||
|  |     this._open = false; | ||||||
|  |  | ||||||
|     if (this._backupEventsSubscription) { |     if (this._backupEventsSubscription) { | ||||||
|       this._backupEventsSubscription.then((unsub) => { |       this._backupEventsSubscription.then((unsub) => { | ||||||
|         unsub(); |         unsub(); | ||||||
| @@ -58,6 +62,10 @@ class DialogRestartWait extends LitElement { | |||||||
|     fireEvent(this, "dialog-closed", { dialog: this.localName }); |     fireEvent(this, "dialog-closed", { dialog: this.localName }); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   public closeDialog(): void { | ||||||
|  |     this._dialog?.close(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   private _getWaitMessage() { |   private _getWaitMessage() { | ||||||
|     switch (this._backupState) { |     switch (this._backupState) { | ||||||
|       case "create_backup": |       case "create_backup": | ||||||
| @@ -72,17 +80,28 @@ class DialogRestartWait extends LitElement { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   protected render() { |   protected render() { | ||||||
|  |     if (!this._open) { | ||||||
|  |       return nothing; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     const waitMessage = this._getWaitMessage(); |     const waitMessage = this._getWaitMessage(); | ||||||
|  |  | ||||||
|     return html` |     return html` | ||||||
|       <ha-wa-dialog |       <ha-md-dialog | ||||||
|         .hass=${this.hass} |         open | ||||||
|         .open=${this._open} |  | ||||||
|         .headerTitle=${this._title} |  | ||||||
|         width="medium" |  | ||||||
|         @closed=${this._dialogClosed} |         @closed=${this._dialogClosed} | ||||||
|  |         .disableCancelAction=${true} | ||||||
|       > |       > | ||||||
|         <div class="content"> |         <ha-dialog-header slot="headline"> | ||||||
|  |           <ha-icon-button | ||||||
|  |             slot="navigationIcon" | ||||||
|  |             .label=${this.hass.localize("ui.common.cancel")} | ||||||
|  |             .path=${mdiClose} | ||||||
|  |             @click=${this.closeDialog} | ||||||
|  |           ></ha-icon-button> | ||||||
|  |           <span slot="title" .title=${this._title}> ${this._title} </span> | ||||||
|  |         </ha-dialog-header> | ||||||
|  |         <div slot="content" class="content"> | ||||||
|           ${this._error |           ${this._error | ||||||
|             ? html`<ha-alert alert-type="error" |             ? html`<ha-alert alert-type="error" | ||||||
|                 >${this.hass.localize("ui.dialogs.restart.error_backup_state", { |                 >${this.hass.localize("ui.dialogs.restart.error_backup_state", { | ||||||
| @@ -94,7 +113,7 @@ class DialogRestartWait extends LitElement { | |||||||
|                 ${waitMessage} |                 ${waitMessage} | ||||||
|               `} |               `} | ||||||
|         </div> |         </div> | ||||||
|       </ha-wa-dialog> |       </ha-md-dialog> | ||||||
|     `; |     `; | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -120,9 +139,15 @@ class DialogRestartWait extends LitElement { | |||||||
|       haStyle, |       haStyle, | ||||||
|       haStyleDialog, |       haStyleDialog, | ||||||
|       css` |       css` | ||||||
|         ha-wa-dialog { |         ha-md-dialog { | ||||||
|           --dialog-content-padding: 0; |           --dialog-content-padding: 0; | ||||||
|         } |         } | ||||||
|  |         @media all and (min-width: 550px) { | ||||||
|  |           ha-md-dialog { | ||||||
|  |             min-width: 500px; | ||||||
|  |             max-width: 500px; | ||||||
|  |           } | ||||||
|  |         } | ||||||
|         .content { |         .content { | ||||||
|           display: flex; |           display: flex; | ||||||
|           flex-direction: column; |           flex-direction: column; | ||||||
|   | |||||||
| @@ -33,7 +33,7 @@ const COMPONENTS = { | |||||||
|   "media-browser": () => |   "media-browser": () => | ||||||
|     import("../panels/media-browser/ha-panel-media-browser"), |     import("../panels/media-browser/ha-panel-media-browser"), | ||||||
|   light: () => import("../panels/light/ha-panel-light"), |   light: () => import("../panels/light/ha-panel-light"), | ||||||
|   safety: () => import("../panels/safety/ha-panel-safety"), |   security: () => import("../panels/security/ha-panel-security"), | ||||||
|   climate: () => import("../panels/climate/ha-panel-climate"), |   climate: () => import("../panels/climate/ha-panel-climate"), | ||||||
| }; | }; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| import { mdiDragHorizontalVariant, mdiPlus } from "@mdi/js"; | import { mdiDrag, mdiPlus } from "@mdi/js"; | ||||||
| import deepClone from "deep-clone-simple"; | import deepClone from "deep-clone-simple"; | ||||||
| import type { PropertyValues } from "lit"; | import type { PropertyValues } from "lit"; | ||||||
| import { LitElement, html, nothing } from "lit"; | import { LitElement, html, nothing } from "lit"; | ||||||
| @@ -115,9 +115,7 @@ export default class HaAutomationAction extends LitElement { | |||||||
|                         @click=${stopPropagation} |                         @click=${stopPropagation} | ||||||
|                         .index=${idx} |                         .index=${idx} | ||||||
|                       > |                       > | ||||||
|                         <ha-svg-icon |                         <ha-svg-icon .path=${mdiDrag}></ha-svg-icon> | ||||||
|                           .path=${mdiDragHorizontalVariant} |  | ||||||
|                         ></ha-svg-icon> |  | ||||||
|                       </div> |                       </div> | ||||||
|                     ` |                     ` | ||||||
|                   : nothing} |                   : nothing} | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| import { mdiDragHorizontalVariant, mdiPlus } from "@mdi/js"; | import { mdiDrag, mdiPlus } from "@mdi/js"; | ||||||
| import deepClone from "deep-clone-simple"; | import deepClone from "deep-clone-simple"; | ||||||
| import type { PropertyValues } from "lit"; | import type { PropertyValues } from "lit"; | ||||||
| import { html, LitElement, nothing } from "lit"; | import { html, LitElement, nothing } from "lit"; | ||||||
| @@ -193,9 +193,7 @@ export default class HaAutomationCondition extends LitElement { | |||||||
|                         @click=${stopPropagation} |                         @click=${stopPropagation} | ||||||
|                         .index=${idx} |                         .index=${idx} | ||||||
|                       > |                       > | ||||||
|                         <ha-svg-icon |                         <ha-svg-icon .path=${mdiDrag}></ha-svg-icon> | ||||||
|                           .path=${mdiDragHorizontalVariant} |  | ||||||
|                         ></ha-svg-icon> |  | ||||||
|                       </div> |                       </div> | ||||||
|                     ` |                     ` | ||||||
|                   : nothing} |                   : nothing} | ||||||
|   | |||||||
| @@ -1,6 +1,5 @@ | |||||||
| import { css, html, LitElement, nothing, type PropertyValues } from "lit"; | import { css, html, LitElement, nothing } from "lit"; | ||||||
| import { customElement, property, query, state } from "lit/decorators"; | import { customElement, property, query, state } from "lit/decorators"; | ||||||
| import { tinykeys } from "tinykeys"; |  | ||||||
| import { fireEvent } from "../../../common/dom/fire_event"; | import { fireEvent } from "../../../common/dom/fire_event"; | ||||||
| import { computeRTL } from "../../../common/util/compute_rtl"; | import { computeRTL } from "../../../common/util/compute_rtl"; | ||||||
| import "../../../components/ha-resizable-bottom-sheet"; | import "../../../components/ha-resizable-bottom-sheet"; | ||||||
| @@ -45,27 +44,11 @@ export default class HaAutomationSidebar extends LitElement { | |||||||
|   @query("ha-resizable-bottom-sheet") |   @query("ha-resizable-bottom-sheet") | ||||||
|   private _bottomSheetElement?: HaResizableBottomSheet; |   private _bottomSheetElement?: HaResizableBottomSheet; | ||||||
|  |  | ||||||
|   @query(".handle") |  | ||||||
|   private _handleElement?: HTMLDivElement; |  | ||||||
|  |  | ||||||
|   private _resizeStartX = 0; |   private _resizeStartX = 0; | ||||||
|  |  | ||||||
|   private _tinykeysUnsub?: () => void; |  | ||||||
|  |  | ||||||
|   protected updated(changedProperties: PropertyValues) { |  | ||||||
|     super.updated(changedProperties); |  | ||||||
|     if (changedProperties.has("config") || changedProperties.has("narrow")) { |  | ||||||
|       if (!this.config || this.narrow) { |  | ||||||
|         this._tinykeysUnsub?.(); |  | ||||||
|         this._tinykeysUnsub = undefined; |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   disconnectedCallback() { |   disconnectedCallback() { | ||||||
|     super.disconnectedCallback(); |     super.disconnectedCallback(); | ||||||
|     this._unregisterResizeHandlers(); |     this._unregisterResizeHandlers(); | ||||||
|     this._tinykeysUnsub?.(); |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private _renderContent() { |   private _renderContent() { | ||||||
| @@ -187,9 +170,6 @@ export default class HaAutomationSidebar extends LitElement { | |||||||
|         class="handle ${this._resizing ? "resizing" : ""}" |         class="handle ${this._resizing ? "resizing" : ""}" | ||||||
|         @mousedown=${this._handleMouseDown} |         @mousedown=${this._handleMouseDown} | ||||||
|         @touchstart=${this._handleMouseDown} |         @touchstart=${this._handleMouseDown} | ||||||
|         @focus=${this._startKeyboardResizing} |  | ||||||
|         @blur=${this._stopKeyboardResizing} |  | ||||||
|         tabindex="0" |  | ||||||
|       > |       > | ||||||
|         <div class="indicator ${this._resizing ? "" : "hidden"}"></div> |         <div class="indicator ${this._resizing ? "" : "hidden"}"></div> | ||||||
|       </div> |       </div> | ||||||
| @@ -308,44 +288,6 @@ export default class HaAutomationSidebar extends LitElement { | |||||||
|     document.removeEventListener("touchcancel", this._endResizing); |     document.removeEventListener("touchcancel", this._endResizing); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private _startKeyboardResizing = (ev: KeyboardEvent) => { |  | ||||||
|     ev.stopPropagation(); |  | ||||||
|     this._resizing = true; |  | ||||||
|     this._resizeStartX = 0; |  | ||||||
|     this._tinykeysUnsub = tinykeys(this._handleElement!, { |  | ||||||
|       ArrowLeft: this._increaseSize, |  | ||||||
|       ArrowRight: this._decreaseSize, |  | ||||||
|     }); |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   private _stopKeyboardResizing = (ev: KeyboardEvent) => { |  | ||||||
|     ev.stopPropagation(); |  | ||||||
|     this._resizing = false; |  | ||||||
|     fireEvent(this, "sidebar-resizing-stopped"); |  | ||||||
|     this._tinykeysUnsub?.(); |  | ||||||
|     this._tinykeysUnsub = undefined; |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   private _increaseSize = (ev: KeyboardEvent) => { |  | ||||||
|     ev.stopPropagation(); |  | ||||||
|  |  | ||||||
|     this._resizeStartX -= computeRTL(this.hass) ? 10 : -10; |  | ||||||
|     this._keyboardResize(); |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   private _decreaseSize = (ev: KeyboardEvent) => { |  | ||||||
|     ev.stopPropagation(); |  | ||||||
|  |  | ||||||
|     this._resizeStartX += computeRTL(this.hass) ? 10 : -10; |  | ||||||
|     this._keyboardResize(); |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   private _keyboardResize() { |  | ||||||
|     fireEvent(this, "sidebar-resized", { |  | ||||||
|       deltaInPx: this._resizeStartX, |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   static styles = css` |   static styles = css` | ||||||
|     :host { |     :host { | ||||||
|       z-index: 6; |       z-index: 6; | ||||||
| @@ -400,10 +342,6 @@ export default class HaAutomationSidebar extends LitElement { | |||||||
|       transform: scale3d(0, 1, 1); |       transform: scale3d(0, 1, 1); | ||||||
|       opacity: 0; |       opacity: 0; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     .handle:focus-visible { |  | ||||||
|       outline: none; |  | ||||||
|     } |  | ||||||
|   `; |   `; | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| import { mdiDragHorizontalVariant, mdiPlus } from "@mdi/js"; | import { mdiDrag, mdiPlus } from "@mdi/js"; | ||||||
| import deepClone from "deep-clone-simple"; | import deepClone from "deep-clone-simple"; | ||||||
| import type { PropertyValues } from "lit"; | import type { PropertyValues } from "lit"; | ||||||
| import { LitElement, css, html, nothing } from "lit"; | import { LitElement, css, html, nothing } from "lit"; | ||||||
| @@ -100,9 +100,7 @@ export default class HaAutomationOption extends LitElement { | |||||||
|                         @click=${stopPropagation} |                         @click=${stopPropagation} | ||||||
|                         .index=${idx} |                         .index=${idx} | ||||||
|                       > |                       > | ||||||
|                         <ha-svg-icon |                         <ha-svg-icon .path=${mdiDrag}></ha-svg-icon> | ||||||
|                           .path=${mdiDragHorizontalVariant} |  | ||||||
|                         ></ha-svg-icon> |  | ||||||
|                       </div> |                       </div> | ||||||
|                     ` |                     ` | ||||||
|                   : nothing} |                   : nothing} | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| import { mdiDragHorizontalVariant, mdiPlus } from "@mdi/js"; | import { mdiDrag, mdiPlus } from "@mdi/js"; | ||||||
| import deepClone from "deep-clone-simple"; | import deepClone from "deep-clone-simple"; | ||||||
| import type { PropertyValues } from "lit"; | import type { PropertyValues } from "lit"; | ||||||
| import { html, LitElement, nothing } from "lit"; | import { html, LitElement, nothing } from "lit"; | ||||||
| @@ -110,9 +110,7 @@ export default class HaAutomationTrigger extends LitElement { | |||||||
|                         @click=${stopPropagation} |                         @click=${stopPropagation} | ||||||
|                         .index=${idx} |                         .index=${idx} | ||||||
|                       > |                       > | ||||||
|                         <ha-svg-icon |                         <ha-svg-icon .path=${mdiDrag}></ha-svg-icon> | ||||||
|                           .path=${mdiDragHorizontalVariant} |  | ||||||
|                         ></ha-svg-icon> |  | ||||||
|                       </div> |                       </div> | ||||||
|                     ` |                     ` | ||||||
|                   : nothing} |                   : nothing} | ||||||
|   | |||||||
| @@ -2,7 +2,6 @@ import { | |||||||
|   mdiChartBox, |   mdiChartBox, | ||||||
|   mdiCog, |   mdiCog, | ||||||
|   mdiFolder, |   mdiFolder, | ||||||
|   mdiInformation, |  | ||||||
|   mdiPlayBoxMultiple, |   mdiPlayBoxMultiple, | ||||||
|   mdiPuzzle, |   mdiPuzzle, | ||||||
| } from "@mdi/js"; | } from "@mdi/js"; | ||||||
| @@ -12,7 +11,6 @@ import { customElement, property, state } from "lit/decorators"; | |||||||
| import memoizeOne from "memoize-one"; | import memoizeOne from "memoize-one"; | ||||||
| import { isComponentLoaded } from "../../../../../common/config/is_component_loaded"; | import { isComponentLoaded } from "../../../../../common/config/is_component_loaded"; | ||||||
| import { fireEvent } from "../../../../../common/dom/fire_event"; | import { fireEvent } from "../../../../../common/dom/fire_event"; | ||||||
| import "../../../../../components/ha-alert"; |  | ||||||
| import "../../../../../components/ha-button"; | import "../../../../../components/ha-button"; | ||||||
| import "../../../../../components/ha-expansion-panel"; | import "../../../../../components/ha-expansion-panel"; | ||||||
| import "../../../../../components/ha-md-list"; | import "../../../../../components/ha-md-list"; | ||||||
| @@ -20,15 +18,10 @@ import "../../../../../components/ha-md-list-item"; | |||||||
| import "../../../../../components/ha-md-select"; | import "../../../../../components/ha-md-select"; | ||||||
| import type { HaMdSelect } from "../../../../../components/ha-md-select"; | import type { HaMdSelect } from "../../../../../components/ha-md-select"; | ||||||
| import "../../../../../components/ha-md-select-option"; | import "../../../../../components/ha-md-select-option"; | ||||||
| import "../../../../../components/ha-spinner"; |  | ||||||
| import "../../../../../components/ha-switch"; | import "../../../../../components/ha-switch"; | ||||||
| import type { HaSwitch } from "../../../../../components/ha-switch"; | import type { HaSwitch } from "../../../../../components/ha-switch"; | ||||||
| import "../../../../../components/ha-tooltip"; |  | ||||||
| import { fetchHassioAddonsInfo } from "../../../../../data/hassio/addon"; | import { fetchHassioAddonsInfo } from "../../../../../data/hassio/addon"; | ||||||
| import type { HostDisksUsage } from "../../../../../data/hassio/host"; |  | ||||||
| import { fetchHostDisksUsage } from "../../../../../data/hassio/host"; |  | ||||||
| import type { HomeAssistant } from "../../../../../types"; | import type { HomeAssistant } from "../../../../../types"; | ||||||
| import { bytesToString } from "../../../../../util/bytes-to-string"; |  | ||||||
| import "../ha-backup-addons-picker"; | import "../ha-backup-addons-picker"; | ||||||
| import type { BackupAddonItem } from "../ha-backup-addons-picker"; | import type { BackupAddonItem } from "../ha-backup-addons-picker"; | ||||||
| import { getRecorderInfo } from "../../../../../data/recorder"; | import { getRecorderInfo } from "../../../../../data/recorder"; | ||||||
| @@ -85,14 +78,11 @@ class HaBackupConfigData extends LitElement { | |||||||
|  |  | ||||||
|   @state() private _showDbOption = true; |   @state() private _showDbOption = true; | ||||||
|  |  | ||||||
|   @state() private _storageInfo?: HostDisksUsage | null; |  | ||||||
|  |  | ||||||
|   protected firstUpdated(changedProperties: PropertyValues): void { |   protected firstUpdated(changedProperties: PropertyValues): void { | ||||||
|     super.firstUpdated(changedProperties); |     super.firstUpdated(changedProperties); | ||||||
|     this._checkDbOption(); |     this._checkDbOption(); | ||||||
|     if (isComponentLoaded(this.hass, "hassio")) { |     if (isComponentLoaded(this.hass, "hassio")) { | ||||||
|       this._fetchAddons(); |       this._fetchAddons(); | ||||||
|       this._fetchStorageInfo(); |  | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -124,68 +114,10 @@ class HaBackupConfigData extends LitElement { | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private async _fetchStorageInfo() { |  | ||||||
|     try { |  | ||||||
|       this._storageInfo = await fetchHostDisksUsage(this.hass); |  | ||||||
|     } catch (_err: any) { |  | ||||||
|       this._storageInfo = null; |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private _hasLocalAddons(addons: BackupAddonItem[]): boolean { |   private _hasLocalAddons(addons: BackupAddonItem[]): boolean { | ||||||
|     return addons.some((addon) => addon.slug === "local"); |     return addons.some((addon) => addon.slug === "local"); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private _estimateBackupSize = memoizeOne( |  | ||||||
|     ( |  | ||||||
|       data: FormData, |  | ||||||
|       storageInfo: HostDisksUsage | null | undefined, |  | ||||||
|       addonsLength: number |  | ||||||
|     ): { |  | ||||||
|       compressedBytes: number; |  | ||||||
|       addonsNotAccurate: boolean; |  | ||||||
|     } | null => { |  | ||||||
|       if (!storageInfo?.children) { |  | ||||||
|         return null; |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       let totalBytes = 0; |  | ||||||
|  |  | ||||||
|       const segments: Record<string, number> = {}; |  | ||||||
|       storageInfo.children.forEach((child) => { |  | ||||||
|         segments[child.id] = child.used_bytes; |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       if (data.homeassistant) { |  | ||||||
|         totalBytes += segments.homeassistant ?? 0; |  | ||||||
|       } |  | ||||||
|       if (data.media) { |  | ||||||
|         totalBytes += segments.media ?? 0; |  | ||||||
|       } |  | ||||||
|       if (data.share) { |  | ||||||
|         totalBytes += segments.share ?? 0; |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       if ( |  | ||||||
|         data.addons_mode === "all" || |  | ||||||
|         (data.addons_mode === "custom" && data.addons.length > 0) |  | ||||||
|       ) { |  | ||||||
|         // It would be better if we could receive individual addon sizes in the WS request instead |  | ||||||
|         totalBytes += |  | ||||||
|           (segments.addons_data ?? 0) + (segments.addons_config ?? 0); |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       return { |  | ||||||
|         // Estimate compressed size (40% reduction typical for gzip) |  | ||||||
|         compressedBytes: Math.round(totalBytes * 0.6), |  | ||||||
|         addonsNotAccurate: |  | ||||||
|           data.addons_mode === "custom" && |  | ||||||
|           data.addons.length > 0 && |  | ||||||
|           data.addons.length !== addonsLength, |  | ||||||
|       }; |  | ||||||
|     } |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   private _getData = memoizeOne( |   private _getData = memoizeOne( | ||||||
|     (value: BackupConfigData | undefined, showAddon: boolean): FormData => { |     (value: BackupConfigData | undefined, showAddon: boolean): FormData => { | ||||||
|       if (!value) { |       if (!value) { | ||||||
| @@ -239,7 +171,6 @@ class HaBackupConfigData extends LitElement { | |||||||
|     const isHassio = isComponentLoaded(this.hass, "hassio"); |     const isHassio = isComponentLoaded(this.hass, "hassio"); | ||||||
|  |  | ||||||
|     return html` |     return html` | ||||||
|       ${this._renderSizeEstimate()} |  | ||||||
|       <ha-md-list> |       <ha-md-list> | ||||||
|         <ha-md-list-item> |         <ha-md-list-item> | ||||||
|           <ha-svg-icon slot="start" .path=${mdiCog}></ha-svg-icon> |           <ha-svg-icon slot="start" .path=${mdiCog}></ha-svg-icon> | ||||||
| @@ -450,103 +381,7 @@ class HaBackupConfigData extends LitElement { | |||||||
|     }); |     }); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private _renderSizeEstimate() { |  | ||||||
|     if (!isComponentLoaded(this.hass, "hassio")) { |  | ||||||
|       return nothing; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     const data = this._getData(this.value, this._showAddons); |  | ||||||
|  |  | ||||||
|     if ( |  | ||||||
|       !( |  | ||||||
|         data.homeassistant || |  | ||||||
|         data.database || |  | ||||||
|         data.media || |  | ||||||
|         data.share || |  | ||||||
|         data.local_addons || |  | ||||||
|         data.addons_mode === "all" || |  | ||||||
|         (data.addons_mode === "custom" && data.addons.length > 0) |  | ||||||
|       ) |  | ||||||
|     ) { |  | ||||||
|       return nothing; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (this._storageInfo === undefined) { |  | ||||||
|       return html` |  | ||||||
|         <ha-alert alert-type="info"> |  | ||||||
|           <ha-spinner slot="icon"></ha-spinner> |  | ||||||
|           ${this.hass.localize( |  | ||||||
|             "ui.panel.config.backup.data.estimated_size_loading" |  | ||||||
|           )} |  | ||||||
|         </ha-alert> |  | ||||||
|       `; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (!this._storageInfo || !this._storageInfo.children) { |  | ||||||
|       return nothing; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     const result = this._estimateBackupSize( |  | ||||||
|       data, |  | ||||||
|       this._storageInfo, |  | ||||||
|       this._addons.length |  | ||||||
|     ); |  | ||||||
|     if (result === null) { |  | ||||||
|       return nothing; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     const { compressedBytes, addonsNotAccurate } = result; |  | ||||||
|  |  | ||||||
|     return html` |  | ||||||
|       <span class="estimated-size"> |  | ||||||
|         <span class="estimated-size-heading"> |  | ||||||
|           ${this.hass.localize("ui.panel.config.backup.data.estimated_size")} |  | ||||||
|           <ha-svg-icon |  | ||||||
|             id="estimated-size-info" |  | ||||||
|             .path=${mdiInformation} |  | ||||||
|           ></ha-svg-icon> |  | ||||||
|           <ha-tooltip for="estimated-size-info" placement="right"> |  | ||||||
|             ${this.hass.localize( |  | ||||||
|               "ui.panel.config.backup.data.estimated_size_disclaimer" |  | ||||||
|             )} |  | ||||||
|             ${addonsNotAccurate |  | ||||||
|               ? html`<br /><br />${this.hass.localize( |  | ||||||
|                     "ui.panel.config.backup.data.estimated_size_disclaimer_addons_custom" |  | ||||||
|                   )}` |  | ||||||
|               : nothing} |  | ||||||
|           </ha-tooltip> |  | ||||||
|         </span> |  | ||||||
|         <span class="estimated-size-value"> |  | ||||||
|           ${bytesToString(compressedBytes)} |  | ||||||
|         </span> |  | ||||||
|       </span> |  | ||||||
|     `; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   static styles = css` |   static styles = css` | ||||||
|     .estimated-size { |  | ||||||
|       display: block; |  | ||||||
|       margin-top: var(--ha-space-2); |  | ||||||
|       font-size: var(--ha-font-size-m); |  | ||||||
|     } |  | ||||||
|     .estimated-size-heading { |  | ||||||
|       font-size: var(--ha-font-size-m); |  | ||||||
|       line-height: var(--ha-line-height-expanded); |  | ||||||
|     } |  | ||||||
|     .estimated-size-heading ha-svg-icon { |  | ||||||
|       --mdc-icon-size: 16px; |  | ||||||
|       color: var(--secondary-text-color); |  | ||||||
|       margin-inline-start: var(--ha-space-1); |  | ||||||
|       vertical-align: middle; |  | ||||||
|     } |  | ||||||
|     .estimated-size-value { |  | ||||||
|       display: block; |  | ||||||
|       font-size: var(--ha-font-size-s); |  | ||||||
|       color: var(--secondary-text-color); |  | ||||||
|     } |  | ||||||
|     ha-spinner { |  | ||||||
|       --ha-spinner-size: 24px; |  | ||||||
|     } |  | ||||||
|     ha-md-list { |     ha-md-list { | ||||||
|       background: none; |       background: none; | ||||||
|       --md-list-item-leading-space: 0; |       --md-list-item-leading-space: 0; | ||||||
|   | |||||||
| @@ -1,11 +1,14 @@ | |||||||
| import { mdiCalendarSync, mdiGestureTap } from "@mdi/js"; | import { mdiCalendarSync, mdiClose, mdiGestureTap } from "@mdi/js"; | ||||||
| import type { CSSResultGroup } from "lit"; | import type { CSSResultGroup } from "lit"; | ||||||
| import { LitElement, css, html, nothing } from "lit"; | import { LitElement, css, html, nothing } from "lit"; | ||||||
| import { customElement, property, state } from "lit/decorators"; | import { customElement, property, query, state } from "lit/decorators"; | ||||||
| import { fireEvent } from "../../../../common/dom/fire_event"; | import { fireEvent } from "../../../../common/dom/fire_event"; | ||||||
|  | import "../../../../components/ha-dialog-header"; | ||||||
|  | import "../../../../components/ha-icon-button"; | ||||||
| import "../../../../components/ha-icon-next"; | import "../../../../components/ha-icon-next"; | ||||||
|  | import "../../../../components/ha-md-dialog"; | ||||||
|  | import type { HaMdDialog } from "../../../../components/ha-md-dialog"; | ||||||
| import "../../../../components/ha-md-list"; | import "../../../../components/ha-md-list"; | ||||||
| import "../../../../components/ha-wa-dialog"; |  | ||||||
| import "../../../../components/ha-md-list-item"; | import "../../../../components/ha-md-list-item"; | ||||||
| import "../../../../components/ha-svg-icon"; | import "../../../../components/ha-svg-icon"; | ||||||
| import type { HassDialog } from "../../../../dialogs/make-dialog-manager"; | import type { HassDialog } from "../../../../dialogs/make-dialog-manager"; | ||||||
| @@ -21,80 +24,92 @@ class DialogNewBackup extends LitElement implements HassDialog { | |||||||
|  |  | ||||||
|   @state() private _params?: NewBackupDialogParams; |   @state() private _params?: NewBackupDialogParams; | ||||||
|  |  | ||||||
|  |   @query("ha-md-dialog") private _dialog?: HaMdDialog; | ||||||
|  |  | ||||||
|   public showDialog(params: NewBackupDialogParams): void { |   public showDialog(params: NewBackupDialogParams): void { | ||||||
|     this._opened = true; |     this._opened = true; | ||||||
|     this._params = params; |     this._params = params; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   public closeDialog() { |   public closeDialog() { | ||||||
|     this._opened = false; |     this._dialog?.close(); | ||||||
|     return true; |     return true; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private _dialogClosed() { |   private _dialogClosed() { | ||||||
|     if (this._params?.cancel) { |     if (this._params!.cancel) { | ||||||
|       this._params.cancel(); |       this._params!.cancel(); | ||||||
|     } |     } | ||||||
|  |     if (this._opened) { | ||||||
|  |       fireEvent(this, "dialog-closed", { dialog: this.localName }); | ||||||
|  |     } | ||||||
|  |     this._opened = false; | ||||||
|     this._params = undefined; |     this._params = undefined; | ||||||
|     fireEvent(this, "dialog-closed", { dialog: this.localName }); |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   protected render() { |   protected render() { | ||||||
|     if (!this._params) { |     if (!this._opened || !this._params) { | ||||||
|       return nothing; |       return nothing; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     return html` |     return html` | ||||||
|       <ha-wa-dialog |       <ha-md-dialog open @closed=${this._dialogClosed}> | ||||||
|         .hass=${this.hass} |         <ha-dialog-header slot="headline"> | ||||||
|         .open=${this._opened} |           <ha-icon-button | ||||||
|         header-title=${this.hass.localize( |             slot="navigationIcon" | ||||||
|           "ui.panel.config.backup.dialogs.new.title" |             @click=${this.closeDialog} | ||||||
|         )} |             .label=${this.hass.localize("ui.common.close")} | ||||||
|         @closed=${this._dialogClosed} |             .path=${mdiClose} | ||||||
|       > |           ></ha-icon-button> | ||||||
|         <ha-md-list |           <span slot="title"> | ||||||
|           innerRole="listbox" |             ${this.hass.localize("ui.panel.config.backup.dialogs.new.title")} | ||||||
|           itemRoles="option" |           </span> | ||||||
|           .innerAriaLabel=${this.hass.localize( |         </ha-dialog-header> | ||||||
|             "ui.panel.config.backup.dialogs.new.options" |         <div slot="content"> | ||||||
|           )} |           <ha-md-list | ||||||
|           rootTabbable |             innerRole="listbox" | ||||||
|         > |             itemRoles="option" | ||||||
|           <ha-md-list-item |             .innerAriaLabel=${this.hass.localize( | ||||||
|             @click=${this._automatic} |               "ui.panel.config.backup.dialogs.new.options" | ||||||
|             type="button" |             )} | ||||||
|             .disabled=${!this._params.config.create_backup.password} |             rootTabbable | ||||||
|  |             dialogInitialFocus | ||||||
|           > |           > | ||||||
|             <ha-svg-icon slot="start" .path=${mdiCalendarSync}></ha-svg-icon> |             <ha-md-list-item | ||||||
|             <span slot="headline"> |               @click=${this._automatic} | ||||||
|               ${this.hass.localize( |               type="button" | ||||||
|                 "ui.panel.config.backup.dialogs.new.automatic.title" |               .disabled=${!this._params.config.create_backup.password} | ||||||
|               )} |             > | ||||||
|             </span> |               <ha-svg-icon slot="start" .path=${mdiCalendarSync}></ha-svg-icon> | ||||||
|             <span slot="supporting-text"> |               <span slot="headline"> | ||||||
|               ${this.hass.localize( |                 ${this.hass.localize( | ||||||
|                 "ui.panel.config.backup.dialogs.new.automatic.description" |                   "ui.panel.config.backup.dialogs.new.automatic.title" | ||||||
|               )} |                 )} | ||||||
|             </span> |               </span> | ||||||
|             <ha-icon-next slot="end"></ha-icon-next> |               <span slot="supporting-text"> | ||||||
|           </ha-md-list-item> |                 ${this.hass.localize( | ||||||
|           <ha-md-list-item @click=${this._manual} type="button"> |                   "ui.panel.config.backup.dialogs.new.automatic.description" | ||||||
|             <ha-svg-icon slot="start" .path=${mdiGestureTap}></ha-svg-icon> |                 )} | ||||||
|             <span slot="headline"> |               </span> | ||||||
|               ${this.hass.localize( |               <ha-icon-next slot="end"></ha-icon-next> | ||||||
|                 "ui.panel.config.backup.dialogs.new.manual.title" |             </ha-md-list-item> | ||||||
|               )} |             <ha-md-list-item @click=${this._manual} type="button"> | ||||||
|             </span> |               <ha-svg-icon slot="start" .path=${mdiGestureTap}></ha-svg-icon> | ||||||
|             <span slot="supporting-text"> |               <span slot="headline"> | ||||||
|               ${this.hass.localize( |                 ${this.hass.localize( | ||||||
|                 "ui.panel.config.backup.dialogs.new.manual.description" |                   "ui.panel.config.backup.dialogs.new.manual.title" | ||||||
|               )} |                 )} | ||||||
|             </span> |               </span> | ||||||
|             <ha-icon-next slot="end"></ha-icon-next> |               <span slot="supporting-text"> | ||||||
|           </ha-md-list-item> |                 ${this.hass.localize( | ||||||
|         </ha-md-list> |                   "ui.panel.config.backup.dialogs.new.manual.description" | ||||||
|       </ha-wa-dialog> |                 )} | ||||||
|  |               </span> | ||||||
|  |               <ha-icon-next slot="end"></ha-icon-next> | ||||||
|  |             </ha-md-list-item> | ||||||
|  |           </ha-md-list> | ||||||
|  |         </div> | ||||||
|  |       </ha-md-dialog> | ||||||
|     `; |     `; | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -113,13 +128,24 @@ class DialogNewBackup extends LitElement implements HassDialog { | |||||||
|       haStyle, |       haStyle, | ||||||
|       haStyleDialog, |       haStyleDialog, | ||||||
|       css` |       css` | ||||||
|         ha-wa-dialog { |         ha-md-dialog { | ||||||
|           --dialog-content-padding: 0; |           --dialog-content-padding: 0; | ||||||
|  |           max-width: 500px; | ||||||
|  |         } | ||||||
|  |         @media all and (max-width: 450px), all and (max-height: 500px) { | ||||||
|  |           ha-md-dialog { | ||||||
|  |             max-width: none; | ||||||
|  |           } | ||||||
|  |           div[slot="content"] { | ||||||
|  |             margin-top: 0; | ||||||
|  |           } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         ha-md-list { |         ha-md-list { | ||||||
|           background: none; |           background: none; | ||||||
|         } |         } | ||||||
|  |         ha-md-list-item { | ||||||
|  |         } | ||||||
|         ha-icon-next { |         ha-icon-next { | ||||||
|           width: 24px; |           width: 24px; | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -244,8 +244,7 @@ class HaConfigBackupSettings extends LitElement { | |||||||
|                   ` |                   ` | ||||||
|                 : nothing} |                 : nothing} | ||||||
|             </div> |             </div> | ||||||
|             ${!this.cloudStatus?.logged_in && |             ${!this.cloudStatus?.logged_in | ||||||
|             isComponentLoaded(this.hass, "cloud") |  | ||||||
|               ? html`<ha-card class="cloud-info"> |               ? html`<ha-card class="cloud-info"> | ||||||
|                   <div class="cloud-header"> |                   <div class="cloud-header"> | ||||||
|                     <img |                     <img | ||||||
| @@ -280,10 +279,7 @@ class HaConfigBackupSettings extends LitElement { | |||||||
|                         "ui.panel.config.voice_assistants.assistants.cloud.sign_in" |                         "ui.panel.config.voice_assistants.assistants.cloud.sign_in" | ||||||
|                       )} |                       )} | ||||||
|                     </ha-button> |                     </ha-button> | ||||||
|                     <ha-button |                     <ha-button href="/config/cloud/register"> | ||||||
|                       href="/config/cloud/register" |  | ||||||
|                       appearance="filled" |  | ||||||
|                     > |  | ||||||
|                       ${this.hass.localize( |                       ${this.hass.localize( | ||||||
|                         "ui.panel.config.voice_assistants.assistants.cloud.try_one_month" |                         "ui.panel.config.voice_assistants.assistants.cloud.try_one_month" | ||||||
|                       )} |                       )} | ||||||
|   | |||||||
| @@ -1,10 +1,4 @@ | |||||||
| import { | import { mdiDelete, mdiDevices, mdiDrag, mdiPencil, mdiPlus } from "@mdi/js"; | ||||||
|   mdiDelete, |  | ||||||
|   mdiDevices, |  | ||||||
|   mdiDragHorizontalVariant, |  | ||||||
|   mdiPencil, |  | ||||||
|   mdiPlus, |  | ||||||
| } from "@mdi/js"; |  | ||||||
| import type { CSSResultGroup, TemplateResult } from "lit"; | import type { CSSResultGroup, TemplateResult } from "lit"; | ||||||
| import { css, html, LitElement } from "lit"; | import { css, html, LitElement } from "lit"; | ||||||
| import { repeat } from "lit/directives/repeat"; | import { repeat } from "lit/directives/repeat"; | ||||||
| @@ -95,9 +89,7 @@ export class EnergyDeviceSettings extends LitElement { | |||||||
|                 (device) => html` |                 (device) => html` | ||||||
|                   <div class="row" .device=${device}> |                   <div class="row" .device=${device}> | ||||||
|                     <div class="handle"> |                     <div class="handle"> | ||||||
|                       <ha-svg-icon |                       <ha-svg-icon .path=${mdiDrag}></ha-svg-icon> | ||||||
|                         .path=${mdiDragHorizontalVariant} |  | ||||||
|                       ></ha-svg-icon> |  | ||||||
|                     </div> |                     </div> | ||||||
|                     <span class="content" |                     <span class="content" | ||||||
|                       >${device.name || |                       >${device.name || | ||||||
|   | |||||||
| @@ -26,6 +26,7 @@ import type { | |||||||
|   EnergySource, |   EnergySource, | ||||||
|   FlowFromGridSourceEnergyPreference, |   FlowFromGridSourceEnergyPreference, | ||||||
|   FlowToGridSourceEnergyPreference, |   FlowToGridSourceEnergyPreference, | ||||||
|  |   GridPowerSourceEnergyPreference, | ||||||
|   GridSourceTypeEnergyPreference, |   GridSourceTypeEnergyPreference, | ||||||
| } from "../../../../data/energy"; | } from "../../../../data/energy"; | ||||||
| import { | import { | ||||||
| @@ -47,6 +48,7 @@ import { documentationUrl } from "../../../../util/documentation-url"; | |||||||
| import { | import { | ||||||
|   showEnergySettingsGridFlowFromDialog, |   showEnergySettingsGridFlowFromDialog, | ||||||
|   showEnergySettingsGridFlowToDialog, |   showEnergySettingsGridFlowToDialog, | ||||||
|  |   showEnergySettingsGridPowerDialog, | ||||||
| } from "../dialogs/show-dialogs-energy"; | } from "../dialogs/show-dialogs-energy"; | ||||||
| import "./ha-energy-validation-result"; | import "./ha-energy-validation-result"; | ||||||
| import { energyCardStyles } from "./styles"; | import { energyCardStyles } from "./styles"; | ||||||
| @@ -226,6 +228,58 @@ export class EnergyGridSettings extends LitElement { | |||||||
|             > |             > | ||||||
|           </div> |           </div> | ||||||
|  |  | ||||||
|  |           <h3> | ||||||
|  |             ${this.hass.localize("ui.panel.config.energy.grid.grid_power")} | ||||||
|  |           </h3> | ||||||
|  |           ${gridSource.power?.map((power) => { | ||||||
|  |             const entityState = this.hass.states[power.stat_power]; | ||||||
|  |             return html` | ||||||
|  |               <div class="row" .source=${power}> | ||||||
|  |                 ${entityState?.attributes.icon | ||||||
|  |                   ? html`<ha-icon | ||||||
|  |                       .icon=${entityState.attributes.icon} | ||||||
|  |                     ></ha-icon>` | ||||||
|  |                   : html`<ha-svg-icon | ||||||
|  |                       .path=${mdiTransmissionTower} | ||||||
|  |                     ></ha-svg-icon>`} | ||||||
|  |                 <span class="content" | ||||||
|  |                   >${getStatisticLabel( | ||||||
|  |                     this.hass, | ||||||
|  |                     power.stat_power, | ||||||
|  |                     this.statsMetadata?.[power.stat_power] | ||||||
|  |                   )}</span | ||||||
|  |                 > | ||||||
|  |                 <ha-icon-button | ||||||
|  |                   .label=${this.hass.localize( | ||||||
|  |                     "ui.panel.config.energy.grid.edit_power" | ||||||
|  |                   )} | ||||||
|  |                   @click=${this._editPowerSource} | ||||||
|  |                   .path=${mdiPencil} | ||||||
|  |                 ></ha-icon-button> | ||||||
|  |                 <ha-icon-button | ||||||
|  |                   .label=${this.hass.localize( | ||||||
|  |                     "ui.panel.config.energy.grid.delete_power" | ||||||
|  |                   )} | ||||||
|  |                   @click=${this._deletePowerSource} | ||||||
|  |                   .path=${mdiDelete} | ||||||
|  |                 ></ha-icon-button> | ||||||
|  |               </div> | ||||||
|  |             `; | ||||||
|  |           })} | ||||||
|  |           <div class="row border-bottom"> | ||||||
|  |             <ha-svg-icon .path=${mdiTransmissionTower}></ha-svg-icon> | ||||||
|  |             <ha-button | ||||||
|  |               @click=${this._addPowerSource} | ||||||
|  |               appearance="filled" | ||||||
|  |               size="small" | ||||||
|  |             > | ||||||
|  |               <ha-svg-icon .path=${mdiPlus} slot="start"></ha-svg-icon | ||||||
|  |               >${this.hass.localize( | ||||||
|  |                 "ui.panel.config.energy.grid.add_power" | ||||||
|  |               )}</ha-button | ||||||
|  |             > | ||||||
|  |           </div> | ||||||
|  |  | ||||||
|           <h3> |           <h3> | ||||||
|             ${this.hass.localize( |             ${this.hass.localize( | ||||||
|               "ui.panel.config.energy.grid.grid_carbon_footprint" |               "ui.panel.config.energy.grid.grid_carbon_footprint" | ||||||
| @@ -499,6 +553,97 @@ export class EnergyGridSettings extends LitElement { | |||||||
|     await this._savePreferences(cleanedPreferences); |     await this._savePreferences(cleanedPreferences); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   private _addPowerSource() { | ||||||
|  |     const gridSource = this.preferences.energy_sources.find( | ||||||
|  |       (src) => src.type === "grid" | ||||||
|  |     ) as GridSourceTypeEnergyPreference | undefined; | ||||||
|  |     showEnergySettingsGridPowerDialog(this, { | ||||||
|  |       grid_source: gridSource, | ||||||
|  |       saveCallback: async (power) => { | ||||||
|  |         let preferences: EnergyPreferences; | ||||||
|  |         if (!gridSource) { | ||||||
|  |           preferences = { | ||||||
|  |             ...this.preferences, | ||||||
|  |             energy_sources: [ | ||||||
|  |               ...this.preferences.energy_sources, | ||||||
|  |               { | ||||||
|  |                 ...emptyGridSourceEnergyPreference(), | ||||||
|  |                 power: [power], | ||||||
|  |               }, | ||||||
|  |             ], | ||||||
|  |           }; | ||||||
|  |         } else { | ||||||
|  |           preferences = { | ||||||
|  |             ...this.preferences, | ||||||
|  |             energy_sources: this.preferences.energy_sources.map((src) => | ||||||
|  |               src.type === "grid" | ||||||
|  |                 ? { ...src, power: [...(gridSource.power || []), power] } | ||||||
|  |                 : src | ||||||
|  |             ), | ||||||
|  |           }; | ||||||
|  |         } | ||||||
|  |         await this._savePreferences(preferences); | ||||||
|  |       }, | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private _editPowerSource(ev) { | ||||||
|  |     const origSource: GridPowerSourceEnergyPreference = | ||||||
|  |       ev.currentTarget.closest(".row").source; | ||||||
|  |     const gridSource = this.preferences.energy_sources.find( | ||||||
|  |       (src) => src.type === "grid" | ||||||
|  |     ) as GridSourceTypeEnergyPreference | undefined; | ||||||
|  |     showEnergySettingsGridPowerDialog(this, { | ||||||
|  |       source: { ...origSource }, | ||||||
|  |       grid_source: gridSource, | ||||||
|  |       saveCallback: async (source) => { | ||||||
|  |         const power = | ||||||
|  |           energySourcesByType(this.preferences).grid![0].power || []; | ||||||
|  |  | ||||||
|  |         const preferences: EnergyPreferences = { | ||||||
|  |           ...this.preferences, | ||||||
|  |           energy_sources: this.preferences.energy_sources.map((src) => | ||||||
|  |             src.type === "grid" | ||||||
|  |               ? { | ||||||
|  |                   ...src, | ||||||
|  |                   power: power.map((p) => (p === origSource ? source : p)), | ||||||
|  |                 } | ||||||
|  |               : src | ||||||
|  |           ), | ||||||
|  |         }; | ||||||
|  |         await this._savePreferences(preferences); | ||||||
|  |       }, | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private async _deletePowerSource(ev) { | ||||||
|  |     const sourceToDelete: GridPowerSourceEnergyPreference = | ||||||
|  |       ev.currentTarget.closest(".row").source; | ||||||
|  |  | ||||||
|  |     if ( | ||||||
|  |       !(await showConfirmationDialog(this, { | ||||||
|  |         title: this.hass.localize("ui.panel.config.energy.delete_source"), | ||||||
|  |       })) | ||||||
|  |     ) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const power = | ||||||
|  |       energySourcesByType(this.preferences).grid![0].power?.filter( | ||||||
|  |         (p) => p !== sourceToDelete | ||||||
|  |       ) || []; | ||||||
|  |  | ||||||
|  |     const preferences: EnergyPreferences = { | ||||||
|  |       ...this.preferences, | ||||||
|  |       energy_sources: this.preferences.energy_sources.map((source) => | ||||||
|  |         source.type === "grid" ? { ...source, power } : source | ||||||
|  |       ), | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     const cleanedPreferences = this._removeEmptySources(preferences); | ||||||
|  |     await this._savePreferences(cleanedPreferences); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   private _removeEmptySources(preferences: EnergyPreferences) { |   private _removeEmptySources(preferences: EnergyPreferences) { | ||||||
|     // Check if grid sources became an empty type and remove if so |     // Check if grid sources became an empty type and remove if so | ||||||
|     preferences.energy_sources = preferences.energy_sources.reduce< |     preferences.energy_sources = preferences.energy_sources.reduce< | ||||||
| @@ -507,7 +652,8 @@ export class EnergyGridSettings extends LitElement { | |||||||
|       if ( |       if ( | ||||||
|         source.type !== "grid" || |         source.type !== "grid" || | ||||||
|         source.flow_from.length > 0 || |         source.flow_from.length > 0 || | ||||||
|         source.flow_to.length > 0 |         source.flow_to.length > 0 || | ||||||
|  |         (source.power && source.power.length > 0) | ||||||
|       ) { |       ) { | ||||||
|         acc.push(source); |         acc.push(source); | ||||||
|       } |       } | ||||||
|   | |||||||
| @@ -18,6 +18,7 @@ import type { HomeAssistant } from "../../../../types"; | |||||||
| import type { EnergySettingsBatteryDialogParams } from "./show-dialogs-energy"; | import type { EnergySettingsBatteryDialogParams } from "./show-dialogs-energy"; | ||||||
|  |  | ||||||
| const energyUnitClasses = ["energy"]; | const energyUnitClasses = ["energy"]; | ||||||
|  | const powerUnitClasses = ["power"]; | ||||||
|  |  | ||||||
| @customElement("dialog-energy-battery-settings") | @customElement("dialog-energy-battery-settings") | ||||||
| export class DialogEnergyBatterySettings | export class DialogEnergyBatterySettings | ||||||
| @@ -32,10 +33,14 @@ export class DialogEnergyBatterySettings | |||||||
|  |  | ||||||
|   @state() private _energy_units?: string[]; |   @state() private _energy_units?: string[]; | ||||||
|  |  | ||||||
|  |   @state() private _power_units?: string[]; | ||||||
|  |  | ||||||
|   @state() private _error?: string; |   @state() private _error?: string; | ||||||
|  |  | ||||||
|   private _excludeList?: string[]; |   private _excludeList?: string[]; | ||||||
|  |  | ||||||
|  |   private _excludeListPower?: string[]; | ||||||
|  |  | ||||||
|   public async showDialog( |   public async showDialog( | ||||||
|     params: EnergySettingsBatteryDialogParams |     params: EnergySettingsBatteryDialogParams | ||||||
|   ): Promise<void> { |   ): Promise<void> { | ||||||
| @@ -46,6 +51,9 @@ export class DialogEnergyBatterySettings | |||||||
|     this._energy_units = ( |     this._energy_units = ( | ||||||
|       await getSensorDeviceClassConvertibleUnits(this.hass, "energy") |       await getSensorDeviceClassConvertibleUnits(this.hass, "energy") | ||||||
|     ).units; |     ).units; | ||||||
|  |     this._power_units = ( | ||||||
|  |       await getSensorDeviceClassConvertibleUnits(this.hass, "power") | ||||||
|  |     ).units; | ||||||
|     const allSources: string[] = []; |     const allSources: string[] = []; | ||||||
|     this._params.battery_sources.forEach((entry) => { |     this._params.battery_sources.forEach((entry) => { | ||||||
|       allSources.push(entry.stat_energy_from); |       allSources.push(entry.stat_energy_from); | ||||||
| @@ -56,6 +64,9 @@ export class DialogEnergyBatterySettings | |||||||
|         id !== this._source?.stat_energy_from && |         id !== this._source?.stat_energy_from && | ||||||
|         id !== this._source?.stat_energy_to |         id !== this._source?.stat_energy_to | ||||||
|     ); |     ); | ||||||
|  |     this._excludeListPower = this._params.battery_sources | ||||||
|  |       .map((entry) => entry.stat_power) | ||||||
|  |       .filter((id) => id && id !== this._source?.stat_power) as string[]; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   public closeDialog() { |   public closeDialog() { | ||||||
| @@ -72,8 +83,6 @@ export class DialogEnergyBatterySettings | |||||||
|       return nothing; |       return nothing; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     const pickableUnit = this._energy_units?.join(", ") || ""; |  | ||||||
|  |  | ||||||
|     return html` |     return html` | ||||||
|       <ha-dialog |       <ha-dialog | ||||||
|         open |         open | ||||||
| @@ -85,12 +94,6 @@ export class DialogEnergyBatterySettings | |||||||
|         @closed=${this.closeDialog} |         @closed=${this.closeDialog} | ||||||
|       > |       > | ||||||
|         ${this._error ? html`<p class="error">${this._error}</p>` : ""} |         ${this._error ? html`<p class="error">${this._error}</p>` : ""} | ||||||
|         <div> |  | ||||||
|           ${this.hass.localize( |  | ||||||
|             "ui.panel.config.energy.battery.dialog.entity_para", |  | ||||||
|             { unit: pickableUnit } |  | ||||||
|           )} |  | ||||||
|         </div> |  | ||||||
|  |  | ||||||
|         <ha-statistic-picker |         <ha-statistic-picker | ||||||
|           .hass=${this.hass} |           .hass=${this.hass} | ||||||
| @@ -105,6 +108,10 @@ export class DialogEnergyBatterySettings | |||||||
|             this._source.stat_energy_from, |             this._source.stat_energy_from, | ||||||
|           ]} |           ]} | ||||||
|           @value-changed=${this._statisticToChanged} |           @value-changed=${this._statisticToChanged} | ||||||
|  |           .helper=${this.hass.localize( | ||||||
|  |             "ui.panel.config.energy.battery.dialog.energy_helper_into", | ||||||
|  |             { unit: this._energy_units?.join(", ") || "" } | ||||||
|  |           )} | ||||||
|           dialogInitialFocus |           dialogInitialFocus | ||||||
|         ></ha-statistic-picker> |         ></ha-statistic-picker> | ||||||
|  |  | ||||||
| @@ -121,6 +128,25 @@ export class DialogEnergyBatterySettings | |||||||
|             this._source.stat_energy_to, |             this._source.stat_energy_to, | ||||||
|           ]} |           ]} | ||||||
|           @value-changed=${this._statisticFromChanged} |           @value-changed=${this._statisticFromChanged} | ||||||
|  |           .helper=${this.hass.localize( | ||||||
|  |             "ui.panel.config.energy.battery.dialog.energy_helper_out", | ||||||
|  |             { unit: this._energy_units?.join(", ") || "" } | ||||||
|  |           )} | ||||||
|  |         ></ha-statistic-picker> | ||||||
|  |  | ||||||
|  |         <ha-statistic-picker | ||||||
|  |           .hass=${this.hass} | ||||||
|  |           .includeUnitClass=${powerUnitClasses} | ||||||
|  |           .value=${this._source.stat_power} | ||||||
|  |           .label=${this.hass.localize( | ||||||
|  |             "ui.panel.config.energy.battery.dialog.power" | ||||||
|  |           )} | ||||||
|  |           .excludeStatistics=${this._excludeListPower} | ||||||
|  |           @value-changed=${this._powerChanged} | ||||||
|  |           .helper=${this.hass.localize( | ||||||
|  |             "ui.panel.config.energy.battery.dialog.power_helper", | ||||||
|  |             { unit: this._power_units?.join(", ") || "" } | ||||||
|  |           )} | ||||||
|         ></ha-statistic-picker> |         ></ha-statistic-picker> | ||||||
|  |  | ||||||
|         <ha-button |         <ha-button | ||||||
| @@ -150,6 +176,10 @@ export class DialogEnergyBatterySettings | |||||||
|     this._source = { ...this._source!, stat_energy_from: ev.detail.value }; |     this._source = { ...this._source!, stat_energy_from: ev.detail.value }; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   private _powerChanged(ev: CustomEvent<{ value: string }>) { | ||||||
|  |     this._source = { ...this._source!, stat_power: ev.detail.value }; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   private async _save() { |   private async _save() { | ||||||
|     try { |     try { | ||||||
|       await this._params!.saveCallback(this._source!); |       await this._params!.saveCallback(this._source!); | ||||||
| @@ -168,7 +198,11 @@ export class DialogEnergyBatterySettings | |||||||
|           --mdc-dialog-max-width: 430px; |           --mdc-dialog-max-width: 430px; | ||||||
|         } |         } | ||||||
|         ha-statistic-picker { |         ha-statistic-picker { | ||||||
|           width: 100%; |           display: block; | ||||||
|  |           margin-bottom: var(--ha-space-4); | ||||||
|  |         } | ||||||
|  |         ha-statistic-picker:last-of-type { | ||||||
|  |           margin-bottom: 0; | ||||||
|         } |         } | ||||||
|       `, |       `, | ||||||
|     ]; |     ]; | ||||||
|   | |||||||
| @@ -21,6 +21,7 @@ import type { HomeAssistant } from "../../../../types"; | |||||||
| import type { EnergySettingsDeviceDialogParams } from "./show-dialogs-energy"; | import type { EnergySettingsDeviceDialogParams } from "./show-dialogs-energy"; | ||||||
|  |  | ||||||
| const energyUnitClasses = ["energy"]; | const energyUnitClasses = ["energy"]; | ||||||
|  | const powerUnitClasses = ["power"]; | ||||||
|  |  | ||||||
| @customElement("dialog-energy-device-settings") | @customElement("dialog-energy-device-settings") | ||||||
| export class DialogEnergyDeviceSettings | export class DialogEnergyDeviceSettings | ||||||
| @@ -35,10 +36,14 @@ export class DialogEnergyDeviceSettings | |||||||
|  |  | ||||||
|   @state() private _energy_units?: string[]; |   @state() private _energy_units?: string[]; | ||||||
|  |  | ||||||
|  |   @state() private _power_units?: string[]; | ||||||
|  |  | ||||||
|   @state() private _error?: string; |   @state() private _error?: string; | ||||||
|  |  | ||||||
|   private _excludeList?: string[]; |   private _excludeList?: string[]; | ||||||
|  |  | ||||||
|  |   private _excludeListPower?: string[]; | ||||||
|  |  | ||||||
|   private _possibleParents: DeviceConsumptionEnergyPreference[] = []; |   private _possibleParents: DeviceConsumptionEnergyPreference[] = []; | ||||||
|  |  | ||||||
|   public async showDialog( |   public async showDialog( | ||||||
| @@ -50,9 +55,15 @@ export class DialogEnergyDeviceSettings | |||||||
|     this._energy_units = ( |     this._energy_units = ( | ||||||
|       await getSensorDeviceClassConvertibleUnits(this.hass, "energy") |       await getSensorDeviceClassConvertibleUnits(this.hass, "energy") | ||||||
|     ).units; |     ).units; | ||||||
|  |     this._power_units = ( | ||||||
|  |       await getSensorDeviceClassConvertibleUnits(this.hass, "power") | ||||||
|  |     ).units; | ||||||
|     this._excludeList = this._params.device_consumptions |     this._excludeList = this._params.device_consumptions | ||||||
|       .map((entry) => entry.stat_consumption) |       .map((entry) => entry.stat_consumption) | ||||||
|       .filter((id) => id !== this._device?.stat_consumption); |       .filter((id) => id !== this._device?.stat_consumption); | ||||||
|  |     this._excludeListPower = this._params.device_consumptions | ||||||
|  |       .map((entry) => entry.stat_power) | ||||||
|  |       .filter((id) => id && id !== this._device?.stat_power) as string[]; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private _computePossibleParents() { |   private _computePossibleParents() { | ||||||
| @@ -93,8 +104,6 @@ export class DialogEnergyDeviceSettings | |||||||
|       return nothing; |       return nothing; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     const pickableUnit = this._energy_units?.join(", ") || ""; |  | ||||||
|  |  | ||||||
|     return html` |     return html` | ||||||
|       <ha-dialog |       <ha-dialog | ||||||
|         open |         open | ||||||
| @@ -108,12 +117,6 @@ export class DialogEnergyDeviceSettings | |||||||
|         @closed=${this.closeDialog} |         @closed=${this.closeDialog} | ||||||
|       > |       > | ||||||
|         ${this._error ? html`<p class="error">${this._error}</p>` : ""} |         ${this._error ? html`<p class="error">${this._error}</p>` : ""} | ||||||
|         <div> |  | ||||||
|           ${this.hass.localize( |  | ||||||
|             "ui.panel.config.energy.device_consumption.dialog.selected_stat_intro", |  | ||||||
|             { unit: pickableUnit } |  | ||||||
|           )} |  | ||||||
|         </div> |  | ||||||
|  |  | ||||||
|         <ha-statistic-picker |         <ha-statistic-picker | ||||||
|           .hass=${this.hass} |           .hass=${this.hass} | ||||||
| @@ -125,9 +128,28 @@ export class DialogEnergyDeviceSettings | |||||||
|           )} |           )} | ||||||
|           .excludeStatistics=${this._excludeList} |           .excludeStatistics=${this._excludeList} | ||||||
|           @value-changed=${this._statisticChanged} |           @value-changed=${this._statisticChanged} | ||||||
|  |           .helper=${this.hass.localize( | ||||||
|  |             "ui.panel.config.energy.device_consumption.dialog.selected_stat_intro", | ||||||
|  |             { unit: this._energy_units?.join(", ") || "" } | ||||||
|  |           )} | ||||||
|           dialogInitialFocus |           dialogInitialFocus | ||||||
|         ></ha-statistic-picker> |         ></ha-statistic-picker> | ||||||
|  |  | ||||||
|  |         <ha-statistic-picker | ||||||
|  |           .hass=${this.hass} | ||||||
|  |           .includeUnitClass=${powerUnitClasses} | ||||||
|  |           .value=${this._device?.stat_power} | ||||||
|  |           .label=${this.hass.localize( | ||||||
|  |             "ui.panel.config.energy.device_consumption.dialog.device_consumption_power" | ||||||
|  |           )} | ||||||
|  |           .excludeStatistics=${this._excludeListPower} | ||||||
|  |           @value-changed=${this._powerStatisticChanged} | ||||||
|  |           .helper=${this.hass.localize( | ||||||
|  |             "ui.panel.config.energy.device_consumption.dialog.selected_stat_intro", | ||||||
|  |             { unit: this._power_units?.join(", ") || "" } | ||||||
|  |           )} | ||||||
|  |         ></ha-statistic-picker> | ||||||
|  |  | ||||||
|         <ha-textfield |         <ha-textfield | ||||||
|           .label=${this.hass.localize( |           .label=${this.hass.localize( | ||||||
|             "ui.panel.config.energy.device_consumption.dialog.display_name" |             "ui.panel.config.energy.device_consumption.dialog.display_name" | ||||||
| @@ -210,6 +232,20 @@ export class DialogEnergyDeviceSettings | |||||||
|     this._computePossibleParents(); |     this._computePossibleParents(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   private _powerStatisticChanged(ev: CustomEvent<{ value: string }>) { | ||||||
|  |     if (!this._device) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     const newDevice = { | ||||||
|  |       ...this._device, | ||||||
|  |       stat_power: ev.detail.value, | ||||||
|  |     } as DeviceConsumptionEnergyPreference; | ||||||
|  |     if (!newDevice.stat_power) { | ||||||
|  |       delete newDevice.stat_power; | ||||||
|  |     } | ||||||
|  |     this._device = newDevice; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   private _nameChanged(ev) { |   private _nameChanged(ev) { | ||||||
|     const newDevice = { |     const newDevice = { | ||||||
|       ...this._device!, |       ...this._device!, | ||||||
| @@ -245,15 +281,19 @@ export class DialogEnergyDeviceSettings | |||||||
|     return [ |     return [ | ||||||
|       haStyleDialog, |       haStyleDialog, | ||||||
|       css` |       css` | ||||||
|  |         ha-statistic-picker { | ||||||
|  |           display: block; | ||||||
|  |           margin-bottom: var(--ha-space-2); | ||||||
|  |         } | ||||||
|         ha-statistic-picker { |         ha-statistic-picker { | ||||||
|           width: 100%; |           width: 100%; | ||||||
|         } |         } | ||||||
|         ha-select { |         ha-select { | ||||||
|           margin-top: 16px; |           margin-top: var(--ha-space-4); | ||||||
|           width: 100%; |           width: 100%; | ||||||
|         } |         } | ||||||
|         ha-textfield { |         ha-textfield { | ||||||
|           margin-top: 16px; |           margin-top: var(--ha-space-4); | ||||||
|           width: 100%; |           width: 100%; | ||||||
|         } |         } | ||||||
|       `, |       `, | ||||||
|   | |||||||
| @@ -115,8 +115,6 @@ export class DialogEnergyGridFlowSettings | |||||||
|       return nothing; |       return nothing; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     const pickableUnit = this._energy_units?.join(", ") || ""; |  | ||||||
|  |  | ||||||
|     const unitPriceSensor = this._pickedDisplayUnit |     const unitPriceSensor = this._pickedDisplayUnit | ||||||
|       ? `${this.hass.config.currency}/${this._pickedDisplayUnit}` |       ? `${this.hass.config.currency}/${this._pickedDisplayUnit}` | ||||||
|       : undefined; |       : undefined; | ||||||
| @@ -150,19 +148,11 @@ export class DialogEnergyGridFlowSettings | |||||||
|         @closed=${this.closeDialog} |         @closed=${this.closeDialog} | ||||||
|       > |       > | ||||||
|         ${this._error ? html`<p class="error">${this._error}</p>` : ""} |         ${this._error ? html`<p class="error">${this._error}</p>` : ""} | ||||||
|         <div> |         <p> | ||||||
|           <p> |           ${this.hass.localize( | ||||||
|             ${this.hass.localize( |             `ui.panel.config.energy.grid.flow_dialog.${this._params.direction}.paragraph` | ||||||
|               `ui.panel.config.energy.grid.flow_dialog.${this._params.direction}.paragraph` |           )} | ||||||
|             )} |         </p> | ||||||
|           </p> |  | ||||||
|           <p> |  | ||||||
|             ${this.hass.localize( |  | ||||||
|               `ui.panel.config.energy.grid.flow_dialog.${this._params.direction}.entity_para`, |  | ||||||
|               { unit: pickableUnit } |  | ||||||
|             )} |  | ||||||
|           </p> |  | ||||||
|         </div> |  | ||||||
|  |  | ||||||
|         <ha-statistic-picker |         <ha-statistic-picker | ||||||
|           .hass=${this.hass} |           .hass=${this.hass} | ||||||
| @@ -178,6 +168,10 @@ export class DialogEnergyGridFlowSettings | |||||||
|           )} |           )} | ||||||
|           .excludeStatistics=${this._excludeList} |           .excludeStatistics=${this._excludeList} | ||||||
|           @value-changed=${this._statisticChanged} |           @value-changed=${this._statisticChanged} | ||||||
|  |           .helper=${this.hass.localize( | ||||||
|  |             `ui.panel.config.energy.grid.flow_dialog.${this._params.direction}.entity_para`, | ||||||
|  |             { unit: this._energy_units?.join(", ") || "" } | ||||||
|  |           )} | ||||||
|           dialogInitialFocus |           dialogInitialFocus | ||||||
|         ></ha-statistic-picker> |         ></ha-statistic-picker> | ||||||
|  |  | ||||||
| @@ -380,6 +374,10 @@ export class DialogEnergyGridFlowSettings | |||||||
|         ha-dialog { |         ha-dialog { | ||||||
|           --mdc-dialog-max-width: 430px; |           --mdc-dialog-max-width: 430px; | ||||||
|         } |         } | ||||||
|  |         ha-statistic-picker { | ||||||
|  |           display: block; | ||||||
|  |           margin: var(--ha-space-4) 0; | ||||||
|  |         } | ||||||
|         ha-formfield { |         ha-formfield { | ||||||
|           display: block; |           display: block; | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -0,0 +1,153 @@ | |||||||
|  | import { mdiTransmissionTower } from "@mdi/js"; | ||||||
|  | import type { CSSResultGroup } from "lit"; | ||||||
|  | import { css, html, LitElement, nothing } from "lit"; | ||||||
|  | import { customElement, property, state } from "lit/decorators"; | ||||||
|  | import { fireEvent } from "../../../../common/dom/fire_event"; | ||||||
|  | import "../../../../components/entity/ha-statistic-picker"; | ||||||
|  | import "../../../../components/ha-dialog"; | ||||||
|  | import "../../../../components/ha-button"; | ||||||
|  | import type { GridPowerSourceEnergyPreference } from "../../../../data/energy"; | ||||||
|  | import { energyStatisticHelpUrl } from "../../../../data/energy"; | ||||||
|  | import { getSensorDeviceClassConvertibleUnits } from "../../../../data/sensor"; | ||||||
|  | import type { HassDialog } from "../../../../dialogs/make-dialog-manager"; | ||||||
|  | import { haStyleDialog } from "../../../../resources/styles"; | ||||||
|  | import type { HomeAssistant } from "../../../../types"; | ||||||
|  | import type { EnergySettingsGridPowerDialogParams } from "./show-dialogs-energy"; | ||||||
|  |  | ||||||
|  | const powerUnitClasses = ["power"]; | ||||||
|  |  | ||||||
|  | @customElement("dialog-energy-grid-power-settings") | ||||||
|  | export class DialogEnergyGridPowerSettings | ||||||
|  |   extends LitElement | ||||||
|  |   implements HassDialog<EnergySettingsGridPowerDialogParams> | ||||||
|  | { | ||||||
|  |   @property({ attribute: false }) public hass!: HomeAssistant; | ||||||
|  |  | ||||||
|  |   @state() private _params?: EnergySettingsGridPowerDialogParams; | ||||||
|  |  | ||||||
|  |   @state() private _source?: GridPowerSourceEnergyPreference; | ||||||
|  |  | ||||||
|  |   @state() private _power_units?: string[]; | ||||||
|  |  | ||||||
|  |   @state() private _error?: string; | ||||||
|  |  | ||||||
|  |   private _excludeListPower?: string[]; | ||||||
|  |  | ||||||
|  |   public async showDialog( | ||||||
|  |     params: EnergySettingsGridPowerDialogParams | ||||||
|  |   ): Promise<void> { | ||||||
|  |     this._params = params; | ||||||
|  |     this._source = params.source ? { ...params.source } : { stat_power: "" }; | ||||||
|  |  | ||||||
|  |     const initialSourceIdPower = this._source.stat_power; | ||||||
|  |  | ||||||
|  |     this._power_units = ( | ||||||
|  |       await getSensorDeviceClassConvertibleUnits(this.hass, "power") | ||||||
|  |     ).units; | ||||||
|  |  | ||||||
|  |     this._excludeListPower = [ | ||||||
|  |       ...(this._params.grid_source?.power?.map((entry) => entry.stat_power) || | ||||||
|  |         []), | ||||||
|  |     ].filter((id) => id && id !== initialSourceIdPower) as string[]; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public closeDialog() { | ||||||
|  |     this._params = undefined; | ||||||
|  |     this._source = undefined; | ||||||
|  |     this._error = undefined; | ||||||
|  |     this._excludeListPower = undefined; | ||||||
|  |     fireEvent(this, "dialog-closed", { dialog: this.localName }); | ||||||
|  |     return true; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   protected render() { | ||||||
|  |     if (!this._params || !this._source) { | ||||||
|  |       return nothing; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return html` | ||||||
|  |       <ha-dialog | ||||||
|  |         open | ||||||
|  |         .heading=${html`<ha-svg-icon | ||||||
|  |             .path=${mdiTransmissionTower} | ||||||
|  |             style="--mdc-icon-size: 32px;" | ||||||
|  |           ></ha-svg-icon | ||||||
|  |           >${this.hass.localize( | ||||||
|  |             "ui.panel.config.energy.grid.power_dialog.header" | ||||||
|  |           )}`} | ||||||
|  |         @closed=${this.closeDialog} | ||||||
|  |       > | ||||||
|  |         ${this._error ? html`<p class="error">${this._error}</p>` : ""} | ||||||
|  |  | ||||||
|  |         <ha-statistic-picker | ||||||
|  |           .hass=${this.hass} | ||||||
|  |           .helpMissingEntityUrl=${energyStatisticHelpUrl} | ||||||
|  |           .includeUnitClass=${powerUnitClasses} | ||||||
|  |           .value=${this._source.stat_power} | ||||||
|  |           .label=${this.hass.localize( | ||||||
|  |             "ui.panel.config.energy.grid.power_dialog.power_stat" | ||||||
|  |           )} | ||||||
|  |           .excludeStatistics=${this._excludeListPower} | ||||||
|  |           @value-changed=${this._powerStatisticChanged} | ||||||
|  |           .helper=${this.hass.localize( | ||||||
|  |             "ui.panel.config.energy.grid.power_dialog.power_helper", | ||||||
|  |             { unit: this._power_units?.join(", ") || "" } | ||||||
|  |           )} | ||||||
|  |           dialogInitialFocus | ||||||
|  |         ></ha-statistic-picker> | ||||||
|  |  | ||||||
|  |         <ha-button | ||||||
|  |           appearance="plain" | ||||||
|  |           @click=${this.closeDialog} | ||||||
|  |           slot="primaryAction" | ||||||
|  |         > | ||||||
|  |           ${this.hass.localize("ui.common.cancel")} | ||||||
|  |         </ha-button> | ||||||
|  |         <ha-button | ||||||
|  |           @click=${this._save} | ||||||
|  |           .disabled=${!this._source.stat_power} | ||||||
|  |           slot="primaryAction" | ||||||
|  |         > | ||||||
|  |           ${this.hass.localize("ui.common.save")} | ||||||
|  |         </ha-button> | ||||||
|  |       </ha-dialog> | ||||||
|  |     `; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private _powerStatisticChanged(ev: CustomEvent<{ value: string }>) { | ||||||
|  |     this._source = { | ||||||
|  |       ...this._source!, | ||||||
|  |       stat_power: ev.detail.value, | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private async _save() { | ||||||
|  |     try { | ||||||
|  |       await this._params!.saveCallback(this._source!); | ||||||
|  |       this.closeDialog(); | ||||||
|  |     } catch (err: any) { | ||||||
|  |       this._error = err.message; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   static get styles(): CSSResultGroup { | ||||||
|  |     return [ | ||||||
|  |       haStyleDialog, | ||||||
|  |       css` | ||||||
|  |         ha-dialog { | ||||||
|  |           --mdc-dialog-max-width: 430px; | ||||||
|  |         } | ||||||
|  |         ha-statistic-picker { | ||||||
|  |           display: block; | ||||||
|  |           margin: var(--ha-space-4) 0; | ||||||
|  |         } | ||||||
|  |       `, | ||||||
|  |     ]; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | declare global { | ||||||
|  |   interface HTMLElementTagNameMap { | ||||||
|  |     "dialog-energy-grid-power-settings": DialogEnergyGridPowerSettings; | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -28,6 +28,7 @@ import { brandsUrl } from "../../../../util/brands-url"; | |||||||
| import type { EnergySettingsSolarDialogParams } from "./show-dialogs-energy"; | import type { EnergySettingsSolarDialogParams } from "./show-dialogs-energy"; | ||||||
|  |  | ||||||
| const energyUnitClasses = ["energy"]; | const energyUnitClasses = ["energy"]; | ||||||
|  | const powerUnitClasses = ["power"]; | ||||||
|  |  | ||||||
| @customElement("dialog-energy-solar-settings") | @customElement("dialog-energy-solar-settings") | ||||||
| export class DialogEnergySolarSettings | export class DialogEnergySolarSettings | ||||||
| @@ -46,10 +47,14 @@ export class DialogEnergySolarSettings | |||||||
|  |  | ||||||
|   @state() private _energy_units?: string[]; |   @state() private _energy_units?: string[]; | ||||||
|  |  | ||||||
|  |   @state() private _power_units?: string[]; | ||||||
|  |  | ||||||
|   @state() private _error?: string; |   @state() private _error?: string; | ||||||
|  |  | ||||||
|   private _excludeList?: string[]; |   private _excludeList?: string[]; | ||||||
|  |  | ||||||
|  |   private _excludeListPower?: string[]; | ||||||
|  |  | ||||||
|   public async showDialog( |   public async showDialog( | ||||||
|     params: EnergySettingsSolarDialogParams |     params: EnergySettingsSolarDialogParams | ||||||
|   ): Promise<void> { |   ): Promise<void> { | ||||||
| @@ -62,9 +67,15 @@ export class DialogEnergySolarSettings | |||||||
|     this._energy_units = ( |     this._energy_units = ( | ||||||
|       await getSensorDeviceClassConvertibleUnits(this.hass, "energy") |       await getSensorDeviceClassConvertibleUnits(this.hass, "energy") | ||||||
|     ).units; |     ).units; | ||||||
|  |     this._power_units = ( | ||||||
|  |       await getSensorDeviceClassConvertibleUnits(this.hass, "power") | ||||||
|  |     ).units; | ||||||
|     this._excludeList = this._params.solar_sources |     this._excludeList = this._params.solar_sources | ||||||
|       .map((entry) => entry.stat_energy_from) |       .map((entry) => entry.stat_energy_from) | ||||||
|       .filter((id) => id !== this._source?.stat_energy_from); |       .filter((id) => id !== this._source?.stat_energy_from); | ||||||
|  |     this._excludeListPower = this._params.solar_sources | ||||||
|  |       .map((entry) => entry.stat_power) | ||||||
|  |       .filter((id) => id && id !== this._source?.stat_power) as string[]; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   public closeDialog() { |   public closeDialog() { | ||||||
| @@ -81,8 +92,6 @@ export class DialogEnergySolarSettings | |||||||
|       return nothing; |       return nothing; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     const pickableUnit = this._energy_units?.join(", ") || ""; |  | ||||||
|  |  | ||||||
|     return html` |     return html` | ||||||
|       <ha-dialog |       <ha-dialog | ||||||
|         open |         open | ||||||
| @@ -94,12 +103,6 @@ export class DialogEnergySolarSettings | |||||||
|         @closed=${this.closeDialog} |         @closed=${this.closeDialog} | ||||||
|       > |       > | ||||||
|         ${this._error ? html`<p class="error">${this._error}</p>` : ""} |         ${this._error ? html`<p class="error">${this._error}</p>` : ""} | ||||||
|         <div> |  | ||||||
|           ${this.hass.localize( |  | ||||||
|             "ui.panel.config.energy.solar.dialog.entity_para", |  | ||||||
|             { unit: pickableUnit } |  | ||||||
|           )} |  | ||||||
|         </div> |  | ||||||
|  |  | ||||||
|         <ha-statistic-picker |         <ha-statistic-picker | ||||||
|           .hass=${this.hass} |           .hass=${this.hass} | ||||||
| @@ -111,9 +114,28 @@ export class DialogEnergySolarSettings | |||||||
|           )} |           )} | ||||||
|           .excludeStatistics=${this._excludeList} |           .excludeStatistics=${this._excludeList} | ||||||
|           @value-changed=${this._statisticChanged} |           @value-changed=${this._statisticChanged} | ||||||
|  |           .helper=${this.hass.localize( | ||||||
|  |             "ui.panel.config.energy.solar.dialog.entity_para", | ||||||
|  |             { unit: this._energy_units?.join(", ") || "" } | ||||||
|  |           )} | ||||||
|           dialogInitialFocus |           dialogInitialFocus | ||||||
|         ></ha-statistic-picker> |         ></ha-statistic-picker> | ||||||
|  |  | ||||||
|  |         <ha-statistic-picker | ||||||
|  |           .hass=${this.hass} | ||||||
|  |           .includeUnitClass=${powerUnitClasses} | ||||||
|  |           .value=${this._source.stat_power} | ||||||
|  |           .label=${this.hass.localize( | ||||||
|  |             "ui.panel.config.energy.solar.dialog.solar_production_power" | ||||||
|  |           )} | ||||||
|  |           .excludeStatistics=${this._excludeListPower} | ||||||
|  |           @value-changed=${this._powerStatisticChanged} | ||||||
|  |           .helper=${this.hass.localize( | ||||||
|  |             "ui.panel.config.energy.solar.dialog.entity_para", | ||||||
|  |             { unit: this._power_units?.join(", ") || "" } | ||||||
|  |           )} | ||||||
|  |         ></ha-statistic-picker> | ||||||
|  |  | ||||||
|         <h3> |         <h3> | ||||||
|           ${this.hass.localize( |           ${this.hass.localize( | ||||||
|             "ui.panel.config.energy.solar.dialog.solar_production_forecast" |             "ui.panel.config.energy.solar.dialog.solar_production_forecast" | ||||||
| @@ -267,6 +289,10 @@ export class DialogEnergySolarSettings | |||||||
|     this._source = { ...this._source!, stat_energy_from: ev.detail.value }; |     this._source = { ...this._source!, stat_energy_from: ev.detail.value }; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   private _powerStatisticChanged(ev: CustomEvent<{ value: string }>) { | ||||||
|  |     this._source = { ...this._source!, stat_power: ev.detail.value }; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   private async _save() { |   private async _save() { | ||||||
|     try { |     try { | ||||||
|       if (!this._forecast) { |       if (!this._forecast) { | ||||||
| @@ -287,6 +313,10 @@ export class DialogEnergySolarSettings | |||||||
|         ha-dialog { |         ha-dialog { | ||||||
|           --mdc-dialog-max-width: 430px; |           --mdc-dialog-max-width: 430px; | ||||||
|         } |         } | ||||||
|  |         ha-statistic-picker { | ||||||
|  |           display: block; | ||||||
|  |           margin-bottom: var(--ha-space-4); | ||||||
|  |         } | ||||||
|         img { |         img { | ||||||
|           height: 24px; |           height: 24px; | ||||||
|           margin-right: 16px; |           margin-right: 16px; | ||||||
|   | |||||||
| @@ -7,6 +7,7 @@ import type { | |||||||
|   FlowFromGridSourceEnergyPreference, |   FlowFromGridSourceEnergyPreference, | ||||||
|   FlowToGridSourceEnergyPreference, |   FlowToGridSourceEnergyPreference, | ||||||
|   GasSourceTypeEnergyPreference, |   GasSourceTypeEnergyPreference, | ||||||
|  |   GridPowerSourceEnergyPreference, | ||||||
|   GridSourceTypeEnergyPreference, |   GridSourceTypeEnergyPreference, | ||||||
|   SolarSourceTypeEnergyPreference, |   SolarSourceTypeEnergyPreference, | ||||||
|   WaterSourceTypeEnergyPreference, |   WaterSourceTypeEnergyPreference, | ||||||
| @@ -41,6 +42,12 @@ export interface EnergySettingsGridFlowToDialogParams { | |||||||
|   saveCallback: (source: FlowToGridSourceEnergyPreference) => Promise<void>; |   saveCallback: (source: FlowToGridSourceEnergyPreference) => Promise<void>; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | export interface EnergySettingsGridPowerDialogParams { | ||||||
|  |   source?: GridPowerSourceEnergyPreference; | ||||||
|  |   grid_source?: GridSourceTypeEnergyPreference; | ||||||
|  |   saveCallback: (source: GridPowerSourceEnergyPreference) => Promise<void>; | ||||||
|  | } | ||||||
|  |  | ||||||
| export interface EnergySettingsSolarDialogParams { | export interface EnergySettingsSolarDialogParams { | ||||||
|   info: EnergyInfo; |   info: EnergyInfo; | ||||||
|   source?: SolarSourceTypeEnergyPreference; |   source?: SolarSourceTypeEnergyPreference; | ||||||
| @@ -152,3 +159,14 @@ export const showEnergySettingsGridFlowToDialog = ( | |||||||
|     dialogParams: { ...dialogParams, direction: "to" }, |     dialogParams: { ...dialogParams, direction: "to" }, | ||||||
|   }); |   }); | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | export const showEnergySettingsGridPowerDialog = ( | ||||||
|  |   element: HTMLElement, | ||||||
|  |   dialogParams: EnergySettingsGridPowerDialogParams | ||||||
|  | ): void => { | ||||||
|  |   fireEvent(element, "show-dialog", { | ||||||
|  |     dialogTag: "dialog-energy-grid-power-settings", | ||||||
|  |     dialogImport: () => import("./dialog-energy-grid-power-settings"), | ||||||
|  |     dialogParams: dialogParams, | ||||||
|  |   }); | ||||||
|  | }; | ||||||
|   | |||||||
| @@ -31,7 +31,7 @@ import { showOptionsFlowDialog } from "../../../dialogs/config-flow/show-dialog- | |||||||
| import { showRestartDialog } from "../../../dialogs/restart/show-dialog-restart"; | import { showRestartDialog } from "../../../dialogs/restart/show-dialog-restart"; | ||||||
| import "../../../layouts/hass-subpage"; | import "../../../layouts/hass-subpage"; | ||||||
| import { SubscribeMixin } from "../../../mixins/subscribe-mixin"; | import { SubscribeMixin } from "../../../mixins/subscribe-mixin"; | ||||||
| import type { ECOption } from "../../../resources/echarts/echarts"; | import type { ECOption } from "../../../resources/echarts"; | ||||||
| import { haStyle } from "../../../resources/styles"; | import { haStyle } from "../../../resources/styles"; | ||||||
| import { DefaultPrimaryColor } from "../../../resources/theme/color/color.globals"; | import { DefaultPrimaryColor } from "../../../resources/theme/color/color.globals"; | ||||||
| import type { HomeAssistant } from "../../../types"; | import type { HomeAssistant } from "../../../types"; | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| import { mdiDelete, mdiDragHorizontalVariant } from "@mdi/js"; | import { mdiDelete, mdiDrag } from "@mdi/js"; | ||||||
| import type { CSSResultGroup } from "lit"; | import type { CSSResultGroup } from "lit"; | ||||||
| import { LitElement, css, html, nothing } from "lit"; | import { LitElement, css, html, nothing } from "lit"; | ||||||
| import { customElement, property, query, state } from "lit/decorators"; | import { customElement, property, query, state } from "lit/decorators"; | ||||||
| @@ -111,9 +111,7 @@ class HaInputSelectForm extends LitElement { | |||||||
|                     <ha-list-item class="option" hasMeta> |                     <ha-list-item class="option" hasMeta> | ||||||
|                       <div class="optioncontent"> |                       <div class="optioncontent"> | ||||||
|                         <div class="handle"> |                         <div class="handle"> | ||||||
|                           <ha-svg-icon |                           <ha-svg-icon .path=${mdiDrag}></ha-svg-icon> | ||||||
|                             .path=${mdiDragHorizontalVariant} |  | ||||||
|                           ></ha-svg-icon> |  | ||||||
|                         </div> |                         </div> | ||||||
|                         ${option} |                         ${option} | ||||||
|                       </div> |                       </div> | ||||||
|   | |||||||
| @@ -2,7 +2,6 @@ import { | |||||||
|   mdiAlertCircle, |   mdiAlertCircle, | ||||||
|   mdiChevronDown, |   mdiChevronDown, | ||||||
|   mdiCogOutline, |   mdiCogOutline, | ||||||
|   mdiContentCopy, |  | ||||||
|   mdiDelete, |   mdiDelete, | ||||||
|   mdiDevices, |   mdiDevices, | ||||||
|   mdiDotsVertical, |   mdiDotsVertical, | ||||||
| @@ -72,8 +71,6 @@ import { | |||||||
| import "./ha-config-entry-device-row"; | import "./ha-config-entry-device-row"; | ||||||
| import { renderConfigEntryError } from "./ha-config-integration-page"; | import { renderConfigEntryError } from "./ha-config-integration-page"; | ||||||
| import "./ha-config-sub-entry-row"; | import "./ha-config-sub-entry-row"; | ||||||
| import { copyToClipboard } from "../../../common/util/copy-clipboard"; |  | ||||||
| import { showToast } from "../../../util/toast"; |  | ||||||
|  |  | ||||||
| @customElement("ha-config-entry-row") | @customElement("ha-config-entry-row") | ||||||
| class HaConfigEntryRow extends LitElement { | class HaConfigEntryRow extends LitElement { | ||||||
| @@ -318,13 +315,6 @@ class HaConfigEntryRow extends LitElement { | |||||||
|             )} |             )} | ||||||
|           </ha-md-menu-item> |           </ha-md-menu-item> | ||||||
|  |  | ||||||
|           <ha-md-menu-item @click=${this._handleCopy} graphic="icon"> |  | ||||||
|             <ha-svg-icon slot="start" .path=${mdiContentCopy}></ha-svg-icon> |  | ||||||
|             ${this.hass.localize( |  | ||||||
|               "ui.panel.config.integrations.config_entry.copy" |  | ||||||
|             )} |  | ||||||
|           </ha-md-menu-item> |  | ||||||
|  |  | ||||||
|           ${Object.keys(item.supported_subentry_types).map( |           ${Object.keys(item.supported_subentry_types).map( | ||||||
|             (flowType) => |             (flowType) => | ||||||
|               html`<ha-md-menu-item |               html`<ha-md-menu-item | ||||||
| @@ -633,15 +623,6 @@ class HaConfigEntryRow extends LitElement { | |||||||
|     }); |     }); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private async _handleCopy() { |  | ||||||
|     await copyToClipboard(this.entry.entry_id); |  | ||||||
|     showToast(this, { |  | ||||||
|       message: |  | ||||||
|         this.hass?.localize("ui.common.copied_clipboard") || |  | ||||||
|         "Copied to clipboard", |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private async _handleRename() { |   private async _handleRename() { | ||||||
|     const newName = await showPromptDialog(this, { |     const newName = await showPromptDialog(this, { | ||||||
|       title: this.hass.localize("ui.panel.config.integrations.rename_dialog"), |       title: this.hass.localize("ui.panel.config.integrations.rename_dialog"), | ||||||
|   | |||||||
| @@ -220,9 +220,6 @@ class DialogZHAManageZigbeeDevice extends LitElement { | |||||||
|  |  | ||||||
|         .content { |         .content { | ||||||
|           outline: none; |           outline: none; | ||||||
|           display: flex; |  | ||||||
|           flex-direction: column; |  | ||||||
|           gap: var(--ha-space-2); |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         @media all and (min-width: 600px) and (min-height: 501px) { |         @media all and (min-width: 600px) and (min-height: 501px) { | ||||||
|   | |||||||
| @@ -128,10 +128,11 @@ class ZHAAddDevicesPage extends LitElement { | |||||||
|                               this.hass, |                               this.hass, | ||||||
|                               "/integrations/zha#adding-devices" |                               "/integrations/zha#adding-devices" | ||||||
|                             )} |                             )} | ||||||
|                             >${this.hass.localize( |  | ||||||
|                               "ui.panel.config.zha.add_device_page.pairing_mode_link" |  | ||||||
|                             )}</a |  | ||||||
|                           > |                           > | ||||||
|  |                             ${this.hass.localize( | ||||||
|  |                               "ui.panel.config.zha.add_device_page.pairing_mode_link" | ||||||
|  |                             )} | ||||||
|  |                           </a> | ||||||
|                         `, |                         `, | ||||||
|                       } |                       } | ||||||
|                     )} |                     )} | ||||||
|   | |||||||
| @@ -117,6 +117,15 @@ export class ZHAClusterAttributes extends LitElement { | |||||||
|         ></ha-textfield> |         ></ha-textfield> | ||||||
|       </div> |       </div> | ||||||
|       <div class="card-actions"> |       <div class="card-actions"> | ||||||
|  |         <ha-progress-button | ||||||
|  |           @click=${this._onGetZigbeeAttributeClick} | ||||||
|  |           .progress=${this._readingAttribute} | ||||||
|  |           .disabled=${this._readingAttribute} | ||||||
|  |         > | ||||||
|  |           ${this.hass!.localize( | ||||||
|  |             "ui.panel.config.zha.cluster_attributes.read_zigbee_attribute" | ||||||
|  |           )} | ||||||
|  |         </ha-progress-button> | ||||||
|         <ha-call-service-button |         <ha-call-service-button | ||||||
|           .hass=${this.hass} |           .hass=${this.hass} | ||||||
|           domain="zha" |           domain="zha" | ||||||
| @@ -127,15 +136,6 @@ export class ZHAClusterAttributes extends LitElement { | |||||||
|             "ui.panel.config.zha.cluster_attributes.write_zigbee_attribute" |             "ui.panel.config.zha.cluster_attributes.write_zigbee_attribute" | ||||||
|           )} |           )} | ||||||
|         </ha-call-service-button> |         </ha-call-service-button> | ||||||
|         <ha-progress-button |  | ||||||
|           @click=${this._onGetZigbeeAttributeClick} |  | ||||||
|           .progress=${this._readingAttribute} |  | ||||||
|           .disabled=${this._readingAttribute} |  | ||||||
|         > |  | ||||||
|           ${this.hass!.localize( |  | ||||||
|             "ui.panel.config.zha.cluster_attributes.read_zigbee_attribute" |  | ||||||
|           )} |  | ||||||
|         </ha-progress-button> |  | ||||||
|       </div> |       </div> | ||||||
|     `; |     `; | ||||||
|   } |   } | ||||||
| @@ -230,10 +230,6 @@ export class ZHAClusterAttributes extends LitElement { | |||||||
|     return [ |     return [ | ||||||
|       haStyle, |       haStyle, | ||||||
|       css` |       css` | ||||||
|         ha-card { |  | ||||||
|           border: none; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         ha-select { |         ha-select { | ||||||
|           margin-top: 16px; |           margin-top: 16px; | ||||||
|         } |         } | ||||||
| @@ -267,12 +263,6 @@ export class ZHAClusterAttributes extends LitElement { | |||||||
|         .header { |         .header { | ||||||
|           flex-grow: 1; |           flex-grow: 1; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         .card-actions { |  | ||||||
|           display: flex; |  | ||||||
|           justify-content: flex-end; |  | ||||||
|           gap: var(--ha-space-1); |  | ||||||
|         } |  | ||||||
|       `, |       `, | ||||||
|     ]; |     ]; | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -108,7 +108,6 @@ export class ZHAClusterCommands extends LitElement { | |||||||
|                   service="issue_zigbee_cluster_command" |                   service="issue_zigbee_cluster_command" | ||||||
|                   .data=${this._issueClusterCommandServiceData} |                   .data=${this._issueClusterCommandServiceData} | ||||||
|                   .disabled=${!this._canIssueCommand} |                   .disabled=${!this._canIssueCommand} | ||||||
|                   appearance="accent" |  | ||||||
|                 > |                 > | ||||||
|                   ${this.hass!.localize( |                   ${this.hass!.localize( | ||||||
|                     "ui.panel.config.zha.cluster_commands.issue_zigbee_command" |                     "ui.panel.config.zha.cluster_commands.issue_zigbee_command" | ||||||
| @@ -188,10 +187,6 @@ export class ZHAClusterCommands extends LitElement { | |||||||
|     return [ |     return [ | ||||||
|       haStyle, |       haStyle, | ||||||
|       css` |       css` | ||||||
|         ha-card { |  | ||||||
|           border: none; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         ha-select { |         ha-select { | ||||||
|           margin-top: 16px; |           margin-top: 16px; | ||||||
|         } |         } | ||||||
| @@ -244,11 +239,6 @@ export class ZHAClusterCommands extends LitElement { | |||||||
|           padding-inline-start: initial; |           padding-inline-start: initial; | ||||||
|           color: var(--primary-color); |           color: var(--primary-color); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         .card-actions { |  | ||||||
|           display: flex; |  | ||||||
|           justify-content: flex-end; |  | ||||||
|         } |  | ||||||
|       `, |       `, | ||||||
|     ]; |     ]; | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -9,7 +9,7 @@ import { | |||||||
| } from "@mdi/js"; | } from "@mdi/js"; | ||||||
| import type { CSSResultGroup, PropertyValues, TemplateResult } from "lit"; | import type { CSSResultGroup, PropertyValues, TemplateResult } from "lit"; | ||||||
| import { css, html, LitElement, nothing } from "lit"; | import { css, html, LitElement, nothing } from "lit"; | ||||||
| import { customElement, property, query, state } from "lit/decorators"; | import { customElement, property, state } from "lit/decorators"; | ||||||
| import "../../../../../components/buttons/ha-progress-button"; | import "../../../../../components/buttons/ha-progress-button"; | ||||||
| import "../../../../../components/ha-alert"; | import "../../../../../components/ha-alert"; | ||||||
| import "../../../../../components/ha-button"; | import "../../../../../components/ha-button"; | ||||||
| @@ -43,7 +43,6 @@ import type { HomeAssistant, Route } from "../../../../../types"; | |||||||
| import { fileDownload } from "../../../../../util/file_download"; | import { fileDownload } from "../../../../../util/file_download"; | ||||||
| import "../../../ha-config-section"; | import "../../../ha-config-section"; | ||||||
| import { showZHAChangeChannelDialog } from "./show-dialog-zha-change-channel"; | import { showZHAChangeChannelDialog } from "./show-dialog-zha-change-channel"; | ||||||
| import type { HaProgressButton } from "../../../../../components/buttons/ha-progress-button"; |  | ||||||
|  |  | ||||||
| const MULTIPROTOCOL_ADDON_URL = "socket://core-silabs-multiprotocol:9999"; | const MULTIPROTOCOL_ADDON_URL = "socket://core-silabs-multiprotocol:9999"; | ||||||
|  |  | ||||||
| @@ -89,8 +88,6 @@ class ZHAConfigDashboard extends LitElement { | |||||||
|  |  | ||||||
|   @state() private _generatingBackup = false; |   @state() private _generatingBackup = false; | ||||||
|  |  | ||||||
|   @query("#config-save-button") private _configSaveButton?: HaProgressButton; |  | ||||||
|  |  | ||||||
|   protected firstUpdated(changedProperties: PropertyValues) { |   protected firstUpdated(changedProperties: PropertyValues) { | ||||||
|     super.firstUpdated(changedProperties); |     super.firstUpdated(changedProperties); | ||||||
|     if (this.hass) { |     if (this.hass) { | ||||||
| @@ -113,201 +110,191 @@ class ZHAConfigDashboard extends LitElement { | |||||||
|         back-path="/config/integrations" |         back-path="/config/integrations" | ||||||
|         has-fab |         has-fab | ||||||
|       > |       > | ||||||
|         <div class="container"> |         <ha-card class="content network-status"> | ||||||
|           <ha-card class="content network-status"> |           ${this._error | ||||||
|             ${this._error |             ? html`<ha-alert alert-type="error">${this._error}</ha-alert>` | ||||||
|               ? html`<ha-alert alert-type="error">${this._error}</ha-alert>` |             : nothing} | ||||||
|               : nothing} |           <div class="card-content"> | ||||||
|             <div class="card-content"> |             <div class="heading"> | ||||||
|               <div class="heading"> |               <div class="icon"> | ||||||
|                 <div class="icon"> |                 <ha-svg-icon | ||||||
|                   <ha-svg-icon |                   .path=${deviceOnline ? mdiCheckCircle : mdiAlertCircle} | ||||||
|                     .path=${deviceOnline ? mdiCheckCircle : mdiAlertCircle} |                   class=${deviceOnline ? "online" : "offline"} | ||||||
|                     class=${deviceOnline ? "online" : "offline"} |                 ></ha-svg-icon> | ||||||
|                   ></ha-svg-icon> |               </div> | ||||||
|                 </div> |               <div class="details"> | ||||||
|                 <div class="details"> |                 ZHA | ||||||
|                   ZHA |                 ${this.hass.localize( | ||||||
|  |                   "ui.panel.config.zha.configuration_page.status_title" | ||||||
|  |                 )}: | ||||||
|  |                 ${this.hass.localize( | ||||||
|  |                   `ui.panel.config.zha.configuration_page.status_${deviceOnline ? "online" : "offline"}` | ||||||
|  |                 )}<br /> | ||||||
|  |                 <small> | ||||||
|                   ${this.hass.localize( |                   ${this.hass.localize( | ||||||
|                     "ui.panel.config.zha.configuration_page.status_title" |                     "ui.panel.config.zha.configuration_page.devices", | ||||||
|                   )}: |                     { count: this._totalDevices } | ||||||
|                   ${this.hass.localize( |                   )} | ||||||
|                     `ui.panel.config.zha.configuration_page.status_${deviceOnline ? "online" : "offline"}` |                 </small> | ||||||
|                   )}<br /> |                 <small class="offline"> | ||||||
|                   <small> |                   ${this._offlineDevices > 0 | ||||||
|                     ${this.hass.localize( |                     ? html`(${this.hass.localize( | ||||||
|                       "ui.panel.config.zha.configuration_page.devices", |                         "ui.panel.config.zha.configuration_page.devices_offline", | ||||||
|                       { count: this._totalDevices } |                         { count: this._offlineDevices } | ||||||
|                     )} |                       )})` | ||||||
|                   </small> |                     : nothing} | ||||||
|                   <small class="offline"> |                 </small> | ||||||
|                     ${this._offlineDevices > 0 |  | ||||||
|                       ? html`(${this.hass.localize( |  | ||||||
|                           "ui.panel.config.zha.configuration_page.devices_offline", |  | ||||||
|                           { count: this._offlineDevices } |  | ||||||
|                         )})` |  | ||||||
|                       : nothing} |  | ||||||
|                   </small> |  | ||||||
|                 </div> |  | ||||||
|               </div> |               </div> | ||||||
|             </div> |             </div> | ||||||
|             ${this.configEntryId |           </div> | ||||||
|               ? html`<div class="card-actions"> |           ${this.configEntryId | ||||||
|                   <ha-button |             ? html`<div class="card-actions"> | ||||||
|                     href=${`/config/devices/dashboard?historyBack=1&config_entry=${this.configEntryId}`} |                 <ha-button | ||||||
|                     appearance="plain" |                   href=${`/config/devices/dashboard?historyBack=1&config_entry=${this.configEntryId}`} | ||||||
|                     size="small" |                   appearance="plain" | ||||||
|  |                   size="small" | ||||||
|  |                 > | ||||||
|  |                   ${this.hass.localize( | ||||||
|  |                     "ui.panel.config.devices.caption" | ||||||
|  |                   )}</ha-button | ||||||
|  |                 > | ||||||
|  |                 <ha-button | ||||||
|  |                   appearance="plain" | ||||||
|  |                   size="small" | ||||||
|  |                   href=${`/config/entities/dashboard?historyBack=1&config_entry=${this.configEntryId}`} | ||||||
|  |                 > | ||||||
|  |                   ${this.hass.localize( | ||||||
|  |                     "ui.panel.config.entities.caption" | ||||||
|  |                   )}</ha-button | ||||||
|  |                 > | ||||||
|  |               </div>` | ||||||
|  |             : ""} | ||||||
|  |         </ha-card> | ||||||
|  |         <ha-card | ||||||
|  |           class="network-settings" | ||||||
|  |           header=${this.hass.localize( | ||||||
|  |             "ui.panel.config.zha.configuration_page.network_settings_title" | ||||||
|  |           )} | ||||||
|  |         > | ||||||
|  |           ${this._networkSettings | ||||||
|  |             ? html`<div class="card-content"> | ||||||
|  |                 <ha-settings-row> | ||||||
|  |                   <span slot="description">PAN ID</span> | ||||||
|  |                   <span slot="heading" | ||||||
|  |                     >${this._networkSettings.settings.network_info.pan_id}</span | ||||||
|                   > |                   > | ||||||
|                     ${this.hass.localize( |                 </ha-settings-row> | ||||||
|                       "ui.panel.config.devices.caption" |  | ||||||
|                     )}</ha-button |                 <ha-settings-row> | ||||||
|  |                   <span slot="heading" | ||||||
|  |                     >${this._networkSettings.settings.network_info | ||||||
|  |                       .extended_pan_id}</span | ||||||
|                   > |                   > | ||||||
|                   <ha-button |                   <span slot="description">Extended PAN ID</span> | ||||||
|                     appearance="plain" |                 </ha-settings-row> | ||||||
|                     size="small" |  | ||||||
|                     href=${`/config/entities/dashboard?historyBack=1&config_entry=${this.configEntryId}`} |                 <ha-settings-row> | ||||||
|  |                   <span slot="description">Channel</span> | ||||||
|  |                   <span slot="heading" | ||||||
|  |                     >${this._networkSettings.settings.network_info | ||||||
|  |                       .channel}</span | ||||||
|                   > |                   > | ||||||
|                     ${this.hass.localize( |  | ||||||
|                       "ui.panel.config.entities.caption" |  | ||||||
|                     )}</ha-button |  | ||||||
|                   > |  | ||||||
|                 </div>` |  | ||||||
|               : ""} |  | ||||||
|           </ha-card> |  | ||||||
|           <ha-card |  | ||||||
|             class="network-settings" |  | ||||||
|             header=${this.hass.localize( |  | ||||||
|               "ui.panel.config.zha.configuration_page.network_settings_title" |  | ||||||
|             )} |  | ||||||
|           > |  | ||||||
|             ${this._networkSettings |  | ||||||
|               ? html`<div class="card-content"> |  | ||||||
|                   <ha-settings-row> |  | ||||||
|                     <span slot="description">PAN ID</span> |  | ||||||
|                     <span slot="heading" |  | ||||||
|                       >${this._networkSettings.settings.network_info |  | ||||||
|                         .pan_id}</span |  | ||||||
|                     > |  | ||||||
|                   </ha-settings-row> |  | ||||||
|  |  | ||||||
|                   <ha-settings-row> |                   <ha-icon-button | ||||||
|                     <span slot="heading" |                     .label=${this.hass.localize( | ||||||
|                       >${this._networkSettings.settings.network_info |                       "ui.panel.config.zha.configuration_page.change_channel" | ||||||
|                         .extended_pan_id}</span |  | ||||||
|                     > |  | ||||||
|                     <span slot="description">Extended PAN ID</span> |  | ||||||
|                   </ha-settings-row> |  | ||||||
|  |  | ||||||
|                   <ha-settings-row> |  | ||||||
|                     <span slot="description">Channel</span> |  | ||||||
|                     <span slot="heading" |  | ||||||
|                       >${this._networkSettings.settings.network_info |  | ||||||
|                         .channel}</span |  | ||||||
|                     > |  | ||||||
|  |  | ||||||
|                     <ha-icon-button |  | ||||||
|                       .label=${this.hass.localize( |  | ||||||
|                         "ui.panel.config.zha.configuration_page.change_channel" |  | ||||||
|                       )} |  | ||||||
|                       .path=${mdiPencil} |  | ||||||
|                       @click=${this._showChannelMigrationDialog} |  | ||||||
|                     > |  | ||||||
|                     </ha-icon-button> |  | ||||||
|                   </ha-settings-row> |  | ||||||
|  |  | ||||||
|                   <ha-settings-row> |  | ||||||
|                     <span slot="description">Coordinator IEEE</span> |  | ||||||
|                     <span slot="heading" |  | ||||||
|                       >${this._networkSettings.settings.node_info.ieee}</span |  | ||||||
|                     > |  | ||||||
|                   </ha-settings-row> |  | ||||||
|  |  | ||||||
|                   <ha-settings-row> |  | ||||||
|                     <span slot="description">Radio type</span> |  | ||||||
|                     <span slot="heading" |  | ||||||
|                       >${this._networkSettings.radio_type}</span |  | ||||||
|                     > |  | ||||||
|                   </ha-settings-row> |  | ||||||
|  |  | ||||||
|                   <ha-settings-row> |  | ||||||
|                     <span slot="description">Serial port</span> |  | ||||||
|                     <span slot="heading" |  | ||||||
|                       >${this._networkSettings.device.path}</span |  | ||||||
|                     > |  | ||||||
|                   </ha-settings-row> |  | ||||||
|  |  | ||||||
|                   ${this._networkSettings.device.baudrate && |  | ||||||
|                   !this._networkSettings.device.path.startsWith("socket://") |  | ||||||
|                     ? html` |  | ||||||
|                         <ha-settings-row> |  | ||||||
|                           <span slot="description">Baudrate</span> |  | ||||||
|                           <span slot="heading" |  | ||||||
|                             >${this._networkSettings.device.baudrate}</span |  | ||||||
|                           > |  | ||||||
|                         </ha-settings-row> |  | ||||||
|                       ` |  | ||||||
|                     : nothing} |  | ||||||
|                 </div>` |  | ||||||
|               : nothing} |  | ||||||
|             <div class="card-actions"> |  | ||||||
|               <ha-progress-button |  | ||||||
|                 appearance="plain" |  | ||||||
|                 @click=${this._createAndDownloadBackup} |  | ||||||
|                 .progress=${this._generatingBackup} |  | ||||||
|                 .disabled=${!this._networkSettings || this._generatingBackup} |  | ||||||
|               > |  | ||||||
|                 ${this.hass.localize( |  | ||||||
|                   "ui.panel.config.zha.configuration_page.download_backup" |  | ||||||
|                 )} |  | ||||||
|               </ha-progress-button> |  | ||||||
|               <ha-button |  | ||||||
|                 appearance="filled" |  | ||||||
|                 variant="brand" |  | ||||||
|                 @click=${this._openOptionFlow} |  | ||||||
|               > |  | ||||||
|                 ${this.hass.localize( |  | ||||||
|                   "ui.panel.config.zha.configuration_page.migrate_radio" |  | ||||||
|                 )} |  | ||||||
|               </ha-button> |  | ||||||
|             </div> |  | ||||||
|           </ha-card> |  | ||||||
|           ${this._configuration |  | ||||||
|             ? Object.entries(this._configuration.schemas).map( |  | ||||||
|                 ([section, schema]) => |  | ||||||
|                   html`<ha-card |  | ||||||
|                     header=${this.hass.localize( |  | ||||||
|                       `component.zha.config_panel.${section}.title` |  | ||||||
|                     )} |                     )} | ||||||
|  |                     .path=${mdiPencil} | ||||||
|  |                     @click=${this._showChannelMigrationDialog} | ||||||
|                   > |                   > | ||||||
|                     <div class="card-content"> |                   </ha-icon-button> | ||||||
|                       <ha-form |                 </ha-settings-row> | ||||||
|                         .hass=${this.hass} |  | ||||||
|                         .schema=${schema} |                 <ha-settings-row> | ||||||
|                         .data=${this._configuration!.data[section]} |                   <span slot="description">Coordinator IEEE</span> | ||||||
|                         @value-changed=${this._dataChanged} |                   <span slot="heading" | ||||||
|                         .section=${section} |                     >${this._networkSettings.settings.node_info.ieee}</span | ||||||
|                         .computeLabel=${this._computeLabelCallback( |                   > | ||||||
|                           this.hass.localize, |                 </ha-settings-row> | ||||||
|                           section |  | ||||||
|                         )} |                 <ha-settings-row> | ||||||
|                       ></ha-form> |                   <span slot="description">Radio type</span> | ||||||
|                     </div> |                   <span slot="heading" | ||||||
|                     <div class="card-actions"> |                     >${this._networkSettings.radio_type}</span | ||||||
|                       <ha-progress-button |                   > | ||||||
|                         id="config-save-button" |                 </ha-settings-row> | ||||||
|                         appearance="filled" |  | ||||||
|                         variant="brand" |                 <ha-settings-row> | ||||||
|                         @click=${this._updateConfiguration} |                   <span slot="description">Serial port</span> | ||||||
|                       > |                   <span slot="heading" | ||||||
|                         ${this.hass.localize( |                     >${this._networkSettings.device.path}</span | ||||||
|                           "ui.panel.config.zha.configuration_page.update_button" |                   > | ||||||
|                         )} |                 </ha-settings-row> | ||||||
|                       </ha-progress-button> |  | ||||||
|                     </div> |                 ${this._networkSettings.device.baudrate && | ||||||
|                   </ha-card>` |                 !this._networkSettings.device.path.startsWith("socket://") | ||||||
|               ) |                   ? html` | ||||||
|             : nothing} |                       <ha-settings-row> | ||||||
|         </div> |                         <span slot="description">Baudrate</span> | ||||||
|  |                         <span slot="heading" | ||||||
|  |                           >${this._networkSettings.device.baudrate}</span | ||||||
|  |                         > | ||||||
|  |                       </ha-settings-row> | ||||||
|  |                     ` | ||||||
|  |                   : ""} | ||||||
|  |               </div>` | ||||||
|  |             : ""} | ||||||
|  |           <div class="card-actions"> | ||||||
|  |             <ha-progress-button | ||||||
|  |               appearance="plain" | ||||||
|  |               @click=${this._createAndDownloadBackup} | ||||||
|  |               .progress=${this._generatingBackup} | ||||||
|  |               .disabled=${!this._networkSettings || this._generatingBackup} | ||||||
|  |             > | ||||||
|  |               ${this.hass.localize( | ||||||
|  |                 "ui.panel.config.zha.configuration_page.download_backup" | ||||||
|  |               )} | ||||||
|  |             </ha-progress-button> | ||||||
|  |             <ha-button variant="danger" @click=${this._openOptionFlow}> | ||||||
|  |               ${this.hass.localize( | ||||||
|  |                 "ui.panel.config.zha.configuration_page.migrate_radio" | ||||||
|  |               )} | ||||||
|  |             </ha-button> | ||||||
|  |           </div> | ||||||
|  |         </ha-card> | ||||||
|  |         ${this._configuration | ||||||
|  |           ? Object.entries(this._configuration.schemas).map( | ||||||
|  |               ([section, schema]) => | ||||||
|  |                 html`<ha-card | ||||||
|  |                   header=${this.hass.localize( | ||||||
|  |                     `component.zha.config_panel.${section}.title` | ||||||
|  |                   )} | ||||||
|  |                 > | ||||||
|  |                   <div class="card-content"> | ||||||
|  |                     <ha-form | ||||||
|  |                       .hass=${this.hass} | ||||||
|  |                       .schema=${schema} | ||||||
|  |                       .data=${this._configuration!.data[section]} | ||||||
|  |                       @value-changed=${this._dataChanged} | ||||||
|  |                       .section=${section} | ||||||
|  |                       .computeLabel=${this._computeLabelCallback( | ||||||
|  |                         this.hass.localize, | ||||||
|  |                         section | ||||||
|  |                       )} | ||||||
|  |                     ></ha-form> | ||||||
|  |                   </div> | ||||||
|  |                 </ha-card>` | ||||||
|  |             ) | ||||||
|  |           : ""} | ||||||
|  |         <ha-card> | ||||||
|  |           <div class="card-actions"> | ||||||
|  |             <ha-button @click=${this._updateConfiguration}> | ||||||
|  |               ${this.hass.localize( | ||||||
|  |                 "ui.panel.config.zha.configuration_page.update_button" | ||||||
|  |               )} | ||||||
|  |             </ha-button> | ||||||
|  |           </div> | ||||||
|  |         </ha-card> | ||||||
|  |  | ||||||
|         <a href="/config/zha/add" slot="fab"> |         <a href="/config/zha/add" slot="fab"> | ||||||
|           <ha-fab |           <ha-fab | ||||||
| @@ -420,15 +407,7 @@ class ZHAConfigDashboard extends LitElement { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   private async _updateConfiguration(): Promise<any> { |   private async _updateConfiguration(): Promise<any> { | ||||||
|     this._configSaveButton!.progress = true; |     await updateZHAConfiguration(this.hass!, this._configuration!.data); | ||||||
|     try { |  | ||||||
|       await updateZHAConfiguration(this.hass!, this._configuration!.data); |  | ||||||
|       this._configSaveButton!.actionSuccess(); |  | ||||||
|     } catch (_err: any) { |  | ||||||
|       this._configSaveButton!.actionError(); |  | ||||||
|     } finally { |  | ||||||
|       this._configSaveButton!.progress = false; |  | ||||||
|     } |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private _computeLabelCallback(localize, section: string) { |   private _computeLabelCallback(localize, section: string) { | ||||||
| @@ -510,10 +489,6 @@ class ZHAConfigDashboard extends LitElement { | |||||||
|         .network-status .offline { |         .network-status .offline { | ||||||
|           color: var(--error-color, var(--error-color)); |           color: var(--error-color, var(--error-color)); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         .container { |  | ||||||
|           padding: var(--ha-space-2) var(--ha-space-4) var(--ha-space-4); |  | ||||||
|         } |  | ||||||
|       `, |       `, | ||||||
|     ]; |     ]; | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -60,15 +60,6 @@ export class ZHADeviceBindingControl extends LitElement { | |||||||
|           </ha-select> |           </ha-select> | ||||||
|         </div> |         </div> | ||||||
|         <div class="card-actions"> |         <div class="card-actions"> | ||||||
|           <ha-progress-button |  | ||||||
|             @click=${this._onUnbindDevicesClick} |  | ||||||
|             .disabled=${!(this._deviceToBind && this.device) || |  | ||||||
|             this._bindingOperationInProgress} |  | ||||||
|             variant="danger" |  | ||||||
|             appearance="plain" |  | ||||||
|           > |  | ||||||
|             ${this.hass!.localize("ui.panel.config.zha.device_binding.unbind")} |  | ||||||
|           </ha-progress-button> |  | ||||||
|           <ha-progress-button |           <ha-progress-button | ||||||
|             @click=${this._onBindDevicesClick} |             @click=${this._onBindDevicesClick} | ||||||
|             .disabled=${!(this._deviceToBind && this.device) || |             .disabled=${!(this._deviceToBind && this.device) || | ||||||
| @@ -76,6 +67,13 @@ export class ZHADeviceBindingControl extends LitElement { | |||||||
|           > |           > | ||||||
|             ${this.hass!.localize("ui.panel.config.zha.device_binding.bind")} |             ${this.hass!.localize("ui.panel.config.zha.device_binding.bind")} | ||||||
|           </ha-progress-button> |           </ha-progress-button> | ||||||
|  |           <ha-progress-button | ||||||
|  |             @click=${this._onUnbindDevicesClick} | ||||||
|  |             .disabled=${!(this._deviceToBind && this.device) || | ||||||
|  |             this._bindingOperationInProgress} | ||||||
|  |           > | ||||||
|  |             ${this.hass!.localize("ui.panel.config.zha.device_binding.unbind")} | ||||||
|  |           </ha-progress-button> | ||||||
|         </div> |         </div> | ||||||
|       </ha-card> |       </ha-card> | ||||||
|     `; |     `; | ||||||
| @@ -135,10 +133,6 @@ export class ZHADeviceBindingControl extends LitElement { | |||||||
|           width: 100%; |           width: 100%; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         .content { |  | ||||||
|           padding-top: var(--ha-space-2); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         .command-picker { |         .command-picker { | ||||||
|           align-items: center; |           align-items: center; | ||||||
|           padding-left: 28px; |           padding-left: 28px; | ||||||
| @@ -151,11 +145,6 @@ export class ZHADeviceBindingControl extends LitElement { | |||||||
|         .header { |         .header { | ||||||
|           flex-grow: 1; |           flex-grow: 1; | ||||||
|         } |         } | ||||||
|         .card-actions { |  | ||||||
|           display: flex; |  | ||||||
|           justify-content: flex-end; |  | ||||||
|           gap: var(--ha-space-1); |  | ||||||
|         } |  | ||||||
|       `, |       `, | ||||||
|     ]; |     ]; | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -85,24 +85,23 @@ export class ZHAGroupBindingControl extends LitElement { | |||||||
|             ></zha-clusters-data-table> |             ></zha-clusters-data-table> | ||||||
|           </div> |           </div> | ||||||
|           <div class="card-actions"> |           <div class="card-actions"> | ||||||
|  |           <ha-progress-button | ||||||
|  |             @click=${this._onBindGroupClick} | ||||||
|  |             .disabled=${!this._canBind || this._bindingOperationInProgress} | ||||||
|  |           > | ||||||
|  |             ${this.hass!.localize( | ||||||
|  |               "ui.panel.config.zha.group_binding.bind_button_label" | ||||||
|  |             )} | ||||||
|  |           </ha-progress-button> | ||||||
|  |  | ||||||
|           <ha-progress-button |           <ha-progress-button | ||||||
|             @click=${this._onUnbindGroupClick} |             @click=${this._onUnbindGroupClick} | ||||||
|             .disabled=${!this._canBind || this._bindingOperationInProgress} |             .disabled=${!this._canBind || this._bindingOperationInProgress} | ||||||
|             variant="danger" |  | ||||||
|             appearance="plain" |  | ||||||
|           > |           > | ||||||
|             ${this.hass!.localize( |             ${this.hass!.localize( | ||||||
|               "ui.panel.config.zha.group_binding.unbind_button_label" |               "ui.panel.config.zha.group_binding.unbind_button_label" | ||||||
|             )} |             )} | ||||||
|           </ha-progress-button> |           </ha-progress-button> | ||||||
|               <ha-progress-button |  | ||||||
|                       @click=${this._onBindGroupClick} |  | ||||||
|                       .disabled=${!this._canBind || this._bindingOperationInProgress} |  | ||||||
|               > |  | ||||||
|                   ${this.hass!.localize( |  | ||||||
|                     "ui.panel.config.zha.group_binding.bind_button_label" |  | ||||||
|                   )} |  | ||||||
|               </ha-progress-button> |  | ||||||
|           </div> |           </div> | ||||||
|         </ha-card> |         </ha-card> | ||||||
|       </ha-config-section> |       </ha-config-section> | ||||||
| @@ -206,10 +205,6 @@ export class ZHAGroupBindingControl extends LitElement { | |||||||
|           width: 100%; |           width: 100%; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         .content { |  | ||||||
|           padding-top: var(--ha-space-2); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         .command-picker { |         .command-picker { | ||||||
|           align-items: center; |           align-items: center; | ||||||
|           padding-left: 28px; |           padding-left: 28px; | ||||||
| @@ -230,12 +225,6 @@ export class ZHAGroupBindingControl extends LitElement { | |||||||
|         .sectionHeader { |         .sectionHeader { | ||||||
|           flex-grow: 1; |           flex-grow: 1; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         .card-actions { |  | ||||||
|           display: flex; |  | ||||||
|           justify-content: flex-end; |  | ||||||
|           gap: var(--ha-space-1); |  | ||||||
|         } |  | ||||||
|       `, |       `, | ||||||
|     ]; |     ]; | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -999,7 +999,6 @@ class ZWaveJSConfigDashboard extends SubscribeMixin(LitElement) { | |||||||
|           display: flex; |           display: flex; | ||||||
|           gap: var(--ha-space-2); |           gap: var(--ha-space-2); | ||||||
|           margin-left: auto; |           margin-left: auto; | ||||||
|           flex-wrap: wrap; |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         .container { |         .container { | ||||||
|   | |||||||
| @@ -9,6 +9,7 @@ import { | |||||||
| import type { PropertyValues } from "lit"; | import type { PropertyValues } from "lit"; | ||||||
| import { LitElement, html, nothing } from "lit"; | import { LitElement, html, nothing } from "lit"; | ||||||
| import { customElement, property, state } from "lit/decorators"; | import { customElement, property, state } from "lit/decorators"; | ||||||
|  | import { ifDefined } from "lit/directives/if-defined"; | ||||||
| import memoize from "memoize-one"; | import memoize from "memoize-one"; | ||||||
| import { isComponentLoaded } from "../../../../common/config/is_component_loaded"; | import { isComponentLoaded } from "../../../../common/config/is_component_loaded"; | ||||||
| import { storage } from "../../../../common/decorators/storage"; | import { storage } from "../../../../common/decorators/storage"; | ||||||
| @@ -61,7 +62,7 @@ type DataTableItem = Pick< | |||||||
| > & { | > & { | ||||||
|   default: boolean; |   default: boolean; | ||||||
|   filename: string; |   filename: string; | ||||||
|   type: string; |   iconColor?: string; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| @customElement("ha-config-lovelace-dashboards") | @customElement("ha-config-lovelace-dashboards") | ||||||
| @@ -106,20 +107,6 @@ export class HaConfigLovelaceDashboards extends LitElement { | |||||||
|   }) |   }) | ||||||
|   private _activeHiddenColumns?: string[]; |   private _activeHiddenColumns?: string[]; | ||||||
|  |  | ||||||
|   @storage({ |  | ||||||
|     key: "lovelace-dashboards-table-grouping", |  | ||||||
|     state: false, |  | ||||||
|     subscribe: false, |  | ||||||
|   }) |  | ||||||
|   private _activeGrouping?: string = "type"; |  | ||||||
|  |  | ||||||
|   @storage({ |  | ||||||
|     key: "lovelace-dashboards-table-collapsed", |  | ||||||
|     state: false, |  | ||||||
|     subscribe: false, |  | ||||||
|   }) |  | ||||||
|   private _activeCollapsed: string[] = []; |  | ||||||
|  |  | ||||||
|   public willUpdate() { |   public willUpdate() { | ||||||
|     if (!this.hasUpdated) { |     if (!this.hasUpdated) { | ||||||
|       this.hass.loadFragmentTranslation("lovelace"); |       this.hass.loadFragmentTranslation("lovelace"); | ||||||
| @@ -145,7 +132,15 @@ export class HaConfigLovelaceDashboards extends LitElement { | |||||||
|           template: (dashboard) => |           template: (dashboard) => | ||||||
|             dashboard.icon |             dashboard.icon | ||||||
|               ? html` |               ? html` | ||||||
|                   <ha-icon slot="item-icon" .icon=${dashboard.icon}></ha-icon> |                   <ha-icon | ||||||
|  |                     slot="item-icon" | ||||||
|  |                     .icon=${dashboard.icon} | ||||||
|  |                     style=${ifDefined( | ||||||
|  |                       dashboard.iconColor | ||||||
|  |                         ? `color: ${dashboard.iconColor}` | ||||||
|  |                         : undefined | ||||||
|  |                     )} | ||||||
|  |                   ></ha-icon> | ||||||
|                 ` |                 ` | ||||||
|               : nothing, |               : nothing, | ||||||
|         }, |         }, | ||||||
| @@ -182,15 +177,6 @@ export class HaConfigLovelaceDashboards extends LitElement { | |||||||
|         }, |         }, | ||||||
|       }; |       }; | ||||||
|  |  | ||||||
|       columns.type = { |  | ||||||
|         title: localize( |  | ||||||
|           "ui.panel.config.lovelace.dashboards.picker.headers.type" |  | ||||||
|         ), |  | ||||||
|         sortable: true, |  | ||||||
|         groupable: true, |  | ||||||
|         filterable: true, |  | ||||||
|       }; |  | ||||||
|  |  | ||||||
|       columns.mode = { |       columns.mode = { | ||||||
|         title: localize( |         title: localize( | ||||||
|           "ui.panel.config.lovelace.dashboards.picker.headers.conf_mode" |           "ui.panel.config.lovelace.dashboards.picker.headers.conf_mode" | ||||||
| @@ -301,7 +287,7 @@ export class HaConfigLovelaceDashboards extends LitElement { | |||||||
|           url_path: "lovelace", |           url_path: "lovelace", | ||||||
|           mode: defaultMode, |           mode: defaultMode, | ||||||
|           filename: defaultMode === "yaml" ? "ui-lovelace.yaml" : "", |           filename: defaultMode === "yaml" ? "ui-lovelace.yaml" : "", | ||||||
|           type: this._localizeType("built_in"), |           iconColor: "var(--primary-color)", | ||||||
|         }, |         }, | ||||||
|       ]; |       ]; | ||||||
|       if (isComponentLoaded(this.hass, "energy")) { |       if (isComponentLoaded(this.hass, "energy")) { | ||||||
| @@ -312,9 +298,9 @@ export class HaConfigLovelaceDashboards extends LitElement { | |||||||
|           mode: "storage", |           mode: "storage", | ||||||
|           url_path: "energy", |           url_path: "energy", | ||||||
|           filename: "", |           filename: "", | ||||||
|  |           iconColor: "var(--orange-color)", | ||||||
|           default: false, |           default: false, | ||||||
|           require_admin: false, |           require_admin: false, | ||||||
|           type: this._localizeType("built_in"), |  | ||||||
|         }); |         }); | ||||||
|       } |       } | ||||||
|  |  | ||||||
| @@ -326,23 +312,23 @@ export class HaConfigLovelaceDashboards extends LitElement { | |||||||
|           mode: "storage", |           mode: "storage", | ||||||
|           url_path: "light", |           url_path: "light", | ||||||
|           filename: "", |           filename: "", | ||||||
|  |           iconColor: "var(--amber-color)", | ||||||
|           default: false, |           default: false, | ||||||
|           require_admin: false, |           require_admin: false, | ||||||
|           type: this._localizeType("built_in"), |  | ||||||
|         }); |         }); | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       if (this.hass.panels.safety) { |       if (this.hass.panels.security) { | ||||||
|         result.push({ |         result.push({ | ||||||
|           icon: "mdi:security", |           icon: "mdi:security", | ||||||
|           title: this.hass.localize("panel.safety"), |           title: this.hass.localize("panel.security"), | ||||||
|           show_in_sidebar: false, |           show_in_sidebar: false, | ||||||
|           mode: "storage", |           mode: "storage", | ||||||
|           url_path: "safety", |           url_path: "security", | ||||||
|           filename: "", |           filename: "", | ||||||
|  |           iconColor: "var(--blue-grey-color)", | ||||||
|           default: false, |           default: false, | ||||||
|           require_admin: false, |           require_admin: false, | ||||||
|           type: this._localizeType("built_in"), |  | ||||||
|         }); |         }); | ||||||
|       } |       } | ||||||
|  |  | ||||||
| @@ -354,9 +340,9 @@ export class HaConfigLovelaceDashboards extends LitElement { | |||||||
|           mode: "storage", |           mode: "storage", | ||||||
|           url_path: "climate", |           url_path: "climate", | ||||||
|           filename: "", |           filename: "", | ||||||
|  |           iconColor: "var(--deep-orange-color)", | ||||||
|           default: false, |           default: false, | ||||||
|           require_admin: false, |           require_admin: false, | ||||||
|           type: this._localizeType("built_in"), |  | ||||||
|         }); |         }); | ||||||
|       } |       } | ||||||
|  |  | ||||||
| @@ -365,25 +351,16 @@ export class HaConfigLovelaceDashboards extends LitElement { | |||||||
|           .sort((a, b) => |           .sort((a, b) => | ||||||
|             stringCompare(a.title, b.title, this.hass.locale.language) |             stringCompare(a.title, b.title, this.hass.locale.language) | ||||||
|           ) |           ) | ||||||
|           .map( |           .map((dashboard) => ({ | ||||||
|             (dashboard) => |             filename: "", | ||||||
|               ({ |             ...dashboard, | ||||||
|                 filename: "", |             default: defaultUrlPath === dashboard.url_path, | ||||||
|                 ...dashboard, |           })) | ||||||
|                 default: defaultUrlPath === dashboard.url_path, |  | ||||||
|                 type: this._localizeType("user_created"), |  | ||||||
|               }) satisfies DataTableItem |  | ||||||
|           ) |  | ||||||
|       ); |       ); | ||||||
|       return result; |       return result; | ||||||
|     } |     } | ||||||
|   ); |   ); | ||||||
|  |  | ||||||
|   private _localizeType = (type: "user_created" | "built_in") => |  | ||||||
|     this.hass.localize( |  | ||||||
|       `ui.panel.config.lovelace.dashboards.picker.type.${type}` |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|   protected render() { |   protected render() { | ||||||
|     if (!this.hass || this._dashboards === undefined) { |     if (!this.hass || this._dashboards === undefined) { | ||||||
|       return html` <hass-loading-screen></hass-loading-screen> `; |       return html` <hass-loading-screen></hass-loading-screen> `; | ||||||
| @@ -403,13 +380,9 @@ export class HaConfigLovelaceDashboards extends LitElement { | |||||||
|           this.hass.localize |           this.hass.localize | ||||||
|         )} |         )} | ||||||
|         .data=${this._getItems(this._dashboards, this.hass.defaultPanel)} |         .data=${this._getItems(this._dashboards, this.hass.defaultPanel)} | ||||||
|         .initialGroupColumn=${this._activeGrouping} |  | ||||||
|         .initialCollapsedGroups=${this._activeCollapsed} |  | ||||||
|         .initialSorting=${this._activeSorting} |         .initialSorting=${this._activeSorting} | ||||||
|         .columnOrder=${this._activeColumnOrder} |         .columnOrder=${this._activeColumnOrder} | ||||||
|         .hiddenColumns=${this._activeHiddenColumns} |         .hiddenColumns=${this._activeHiddenColumns} | ||||||
|         @grouping-changed=${this._handleGroupingChanged} |  | ||||||
|         @collapsed-changed=${this._handleCollapseChanged} |  | ||||||
|         @columns-changed=${this._handleColumnsChanged} |         @columns-changed=${this._handleColumnsChanged} | ||||||
|         @sorting-changed=${this._handleSortingChanged} |         @sorting-changed=${this._handleSortingChanged} | ||||||
|         .filter=${this._filter} |         .filter=${this._filter} | ||||||
| @@ -470,13 +443,13 @@ export class HaConfigLovelaceDashboards extends LitElement { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   private _canDelete(urlPath: string) { |   private _canDelete(urlPath: string) { | ||||||
|     return !["lovelace", "energy", "light", "safety", "climate"].includes( |     return !["lovelace", "energy", "light", "security", "climate"].includes( | ||||||
|       urlPath |       urlPath | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private _canEdit(urlPath: string) { |   private _canEdit(urlPath: string) { | ||||||
|     return !["light", "safety", "climate"].includes(urlPath); |     return !["light", "security", "climate"].includes(urlPath); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private _handleDelete = async (item: DataTableItem) => { |   private _handleDelete = async (item: DataTableItem) => { | ||||||
| @@ -598,14 +571,6 @@ export class HaConfigLovelaceDashboards extends LitElement { | |||||||
|     this._activeColumnOrder = ev.detail.columnOrder; |     this._activeColumnOrder = ev.detail.columnOrder; | ||||||
|     this._activeHiddenColumns = ev.detail.hiddenColumns; |     this._activeHiddenColumns = ev.detail.hiddenColumns; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private _handleGroupingChanged(ev: CustomEvent) { |  | ||||||
|     this._activeGrouping = ev.detail.value; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private _handleCollapseChanged(ev: CustomEvent) { |  | ||||||
|     this._activeCollapsed = ev.detail.value; |  | ||||||
|   } |  | ||||||
| } | } | ||||||
|  |  | ||||||
| declare global { | declare global { | ||||||
|   | |||||||
| @@ -1,10 +1,10 @@ | |||||||
| import type { ActionDetail } from "@material/mwc-list"; |  | ||||||
| import { | import { | ||||||
|   mdiDotsVertical, |   mdiDotsVertical, | ||||||
|   mdiDownload, |   mdiDownload, | ||||||
|   mdiFilterRemove, |   mdiFilterRemove, | ||||||
|   mdiImagePlus, |   mdiImagePlus, | ||||||
| } from "@mdi/js"; | } from "@mdi/js"; | ||||||
|  | import type { ActionDetail } from "@material/mwc-list"; | ||||||
| import { differenceInHours } from "date-fns"; | import { differenceInHours } from "date-fns"; | ||||||
| import type { | import type { | ||||||
|   HassServiceTarget, |   HassServiceTarget, | ||||||
| @@ -27,21 +27,21 @@ import { | |||||||
| import { MIN_TIME_BETWEEN_UPDATES } from "../../components/chart/ha-chart-base"; | import { MIN_TIME_BETWEEN_UPDATES } from "../../components/chart/ha-chart-base"; | ||||||
| import "../../components/chart/state-history-charts"; | import "../../components/chart/state-history-charts"; | ||||||
| import type { StateHistoryCharts } from "../../components/chart/state-history-charts"; | import type { StateHistoryCharts } from "../../components/chart/state-history-charts"; | ||||||
| import "../../components/ha-button-menu"; | import "../../components/ha-spinner"; | ||||||
| import "../../components/ha-date-range-picker"; | import "../../components/ha-date-range-picker"; | ||||||
| import "../../components/ha-icon-button"; | import "../../components/ha-icon-button"; | ||||||
| import "../../components/ha-icon-button-arrow-prev"; | import "../../components/ha-button-menu"; | ||||||
| import "../../components/ha-list-item"; | import "../../components/ha-list-item"; | ||||||
|  | import "../../components/ha-icon-button-arrow-prev"; | ||||||
| import "../../components/ha-menu-button"; | import "../../components/ha-menu-button"; | ||||||
| import "../../components/ha-spinner"; |  | ||||||
| import "../../components/ha-target-picker"; | import "../../components/ha-target-picker"; | ||||||
| import "../../components/ha-top-app-bar-fixed"; | import "../../components/ha-top-app-bar-fixed"; | ||||||
| import type { HistoryResult } from "../../data/history"; | import type { HistoryResult } from "../../data/history"; | ||||||
| import { | import { | ||||||
|   computeHistory, |   computeHistory, | ||||||
|   convertStatisticsToHistory, |  | ||||||
|   mergeHistoryResults, |  | ||||||
|   subscribeHistory, |   subscribeHistory, | ||||||
|  |   mergeHistoryResults, | ||||||
|  |   convertStatisticsToHistory, | ||||||
| } from "../../data/history"; | } from "../../data/history"; | ||||||
| import { fetchStatistics } from "../../data/recorder"; | import { fetchStatistics } from "../../data/recorder"; | ||||||
| import { resolveEntityIDs } from "../../data/selector"; | import { resolveEntityIDs } from "../../data/selector"; | ||||||
| @@ -182,7 +182,6 @@ class HaPanelHistory extends LitElement { | |||||||
|               .disabled=${this._isLoading} |               .disabled=${this._isLoading} | ||||||
|               add-on-top |               add-on-top | ||||||
|               @value-changed=${this._targetsChanged} |               @value-changed=${this._targetsChanged} | ||||||
|               compact |  | ||||||
|             ></ha-target-picker> |             ></ha-target-picker> | ||||||
|           </div> |           </div> | ||||||
|           ${this._isLoading |           ${this._isLoading | ||||||
| @@ -650,10 +649,6 @@ class HaPanelHistory extends LitElement { | |||||||
|           direction: var(--direction); |           direction: var(--direction); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         ha-target-picker { |  | ||||||
|           flex: 1; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         @media all and (max-width: 1025px) { |         @media all and (max-width: 1025px) { | ||||||
|           .filters { |           .filters { | ||||||
|             flex-direction: column; |             flex-direction: column; | ||||||
|   | |||||||
| @@ -1,11 +1,9 @@ | |||||||
| import { mdiRefresh } from "@mdi/js"; | import { mdiRefresh } from "@mdi/js"; | ||||||
| import type { HassServiceTarget } from "home-assistant-js-websocket"; |  | ||||||
| import type { PropertyValues } from "lit"; | import type { PropertyValues } from "lit"; | ||||||
| import { css, html, LitElement } from "lit"; | import { css, html, LitElement } from "lit"; | ||||||
| import { customElement, property, state } from "lit/decorators"; | import { customElement, property, state } from "lit/decorators"; | ||||||
|  | import type { HassServiceTarget } from "home-assistant-js-websocket"; | ||||||
| import memoizeOne from "memoize-one"; | import memoizeOne from "memoize-one"; | ||||||
| import { ensureArray } from "../../common/array/ensure-array"; |  | ||||||
| import { storage } from "../../common/decorators/storage"; |  | ||||||
| import { goBack, navigate } from "../../common/navigate"; | import { goBack, navigate } from "../../common/navigate"; | ||||||
| import { constructUrlCurrentPath } from "../../common/url/construct-url"; | import { constructUrlCurrentPath } from "../../common/url/construct-url"; | ||||||
| import { | import { | ||||||
| @@ -18,15 +16,17 @@ import "../../components/ha-date-range-picker"; | |||||||
| import "../../components/ha-icon-button"; | import "../../components/ha-icon-button"; | ||||||
| import "../../components/ha-icon-button-arrow-prev"; | import "../../components/ha-icon-button-arrow-prev"; | ||||||
| import "../../components/ha-menu-button"; | import "../../components/ha-menu-button"; | ||||||
| import "../../components/ha-target-picker"; |  | ||||||
| import "../../components/ha-top-app-bar-fixed"; | import "../../components/ha-top-app-bar-fixed"; | ||||||
| import type { HaEntityPickerEntityFilterFunc } from "../../data/entity"; | import "../../components/ha-target-picker"; | ||||||
| import { filterLogbookCompatibleEntities } from "../../data/logbook"; | import { filterLogbookCompatibleEntities } from "../../data/logbook"; | ||||||
| import { resolveEntityIDs } from "../../data/selector"; |  | ||||||
| import { getSensorNumericDeviceClasses } from "../../data/sensor"; |  | ||||||
| import { haStyle } from "../../resources/styles"; | import { haStyle } from "../../resources/styles"; | ||||||
| import type { HomeAssistant } from "../../types"; | import type { HomeAssistant } from "../../types"; | ||||||
| import "./ha-logbook"; | import "./ha-logbook"; | ||||||
|  | import { storage } from "../../common/decorators/storage"; | ||||||
|  | import { ensureArray } from "../../common/array/ensure-array"; | ||||||
|  | import { resolveEntityIDs } from "../../data/selector"; | ||||||
|  | import { getSensorNumericDeviceClasses } from "../../data/sensor"; | ||||||
|  | import type { HaEntityPickerEntityFilterFunc } from "../../components/entity/ha-entity-picker"; | ||||||
|  |  | ||||||
| @customElement("ha-panel-logbook") | @customElement("ha-panel-logbook") | ||||||
| export class HaPanelLogbook extends LitElement { | export class HaPanelLogbook extends LitElement { | ||||||
| @@ -108,7 +108,6 @@ export class HaPanelLogbook extends LitElement { | |||||||
|               .value=${this._targetPickerValue} |               .value=${this._targetPickerValue} | ||||||
|               add-on-top |               add-on-top | ||||||
|               @value-changed=${this._targetsChanged} |               @value-changed=${this._targetsChanged} | ||||||
|               compact |  | ||||||
|             ></ha-target-picker> |             ></ha-target-picker> | ||||||
|           </div> |           </div> | ||||||
|  |  | ||||||
| @@ -364,10 +363,6 @@ export class HaPanelLogbook extends LitElement { | |||||||
|           max-width: 400px; |           max-width: 400px; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         ha-target-picker { |  | ||||||
|           flex: 1; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         :host([narrow]) ha-entity-picker { |         :host([narrow]) ha-entity-picker { | ||||||
|           max-width: none; |           max-width: none; | ||||||
|           width: 100%; |           width: 100%; | ||||||
|   | |||||||
| @@ -9,6 +9,7 @@ import { computeCssColor } from "../../../common/color/compute-color"; | |||||||
| import { hsv2rgb, rgb2hex, rgb2hsv } from "../../../common/color/convert-color"; | import { hsv2rgb, rgb2hex, rgb2hsv } from "../../../common/color/convert-color"; | ||||||
| import { computeDomain } from "../../../common/entity/compute_domain"; | import { computeDomain } from "../../../common/entity/compute_domain"; | ||||||
| import { computeStateDomain } from "../../../common/entity/compute_state_domain"; | import { computeStateDomain } from "../../../common/entity/compute_state_domain"; | ||||||
|  | import { computeStateName } from "../../../common/entity/compute_state_name"; | ||||||
| import { stateActive } from "../../../common/entity/state_active"; | import { stateActive } from "../../../common/entity/state_active"; | ||||||
| import { stateColorCss } from "../../../common/entity/state_color"; | import { stateColorCss } from "../../../common/entity/state_color"; | ||||||
| import "../../../components/ha-badge"; | import "../../../components/ha-badge"; | ||||||
| @@ -19,7 +20,6 @@ import { cameraUrlWithWidthHeight } from "../../../data/camera"; | |||||||
| import type { ActionHandlerEvent } from "../../../data/lovelace/action_handler"; | import type { ActionHandlerEvent } from "../../../data/lovelace/action_handler"; | ||||||
| import type { HomeAssistant } from "../../../types"; | import type { HomeAssistant } from "../../../types"; | ||||||
| import { actionHandler } from "../common/directives/action-handler-directive"; | import { actionHandler } from "../common/directives/action-handler-directive"; | ||||||
| import { computeLovelaceEntityName } from "../common/entity/compute-lovelace-entity-name"; |  | ||||||
| import { findEntities } from "../common/find-entities"; | import { findEntities } from "../common/find-entities"; | ||||||
| import { handleAction } from "../common/handle-action"; | import { handleAction } from "../common/handle-action"; | ||||||
| import { hasAction } from "../common/has-action"; | import { hasAction } from "../common/has-action"; | ||||||
| @@ -162,7 +162,11 @@ export class HuiEntityBadge extends LitElement implements LovelaceBadge { | |||||||
|     if (!stateObj) { |     if (!stateObj) { | ||||||
|       return html` |       return html` | ||||||
|         <ha-badge .label=${entityId} class="error"> |         <ha-badge .label=${entityId} class="error"> | ||||||
|           <ha-svg-icon slot="icon" .path=${mdiAlertCircle}></ha-svg-icon> |           <ha-svg-icon | ||||||
|  |             slot="icon" | ||||||
|  |             .hass=${this.hass} | ||||||
|  |             .path=${mdiAlertCircle} | ||||||
|  |           ></ha-svg-icon> | ||||||
|           ${this.hass.localize("ui.badge.entity.not_found")} |           ${this.hass.localize("ui.badge.entity.not_found")} | ||||||
|         </ha-badge> |         </ha-badge> | ||||||
|       `; |       `; | ||||||
| @@ -175,22 +179,18 @@ export class HuiEntityBadge extends LitElement implements LovelaceBadge { | |||||||
|       "--badge-color": color, |       "--badge-color": color, | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     const name = computeLovelaceEntityName( |  | ||||||
|       this.hass, |  | ||||||
|       stateObj, |  | ||||||
|       this._config.name |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     const stateDisplay = html` |     const stateDisplay = html` | ||||||
|       <state-display |       <state-display | ||||||
|         .stateObj=${stateObj} |         .stateObj=${stateObj} | ||||||
|         .hass=${this.hass} |         .hass=${this.hass} | ||||||
|         .content=${this._config.state_content} |         .content=${this._config.state_content} | ||||||
|         .name=${name} |         .name=${this._config.name} | ||||||
|       > |       > | ||||||
|       </state-display> |       </state-display> | ||||||
|     `; |     `; | ||||||
|  |  | ||||||
|  |     const name = this._config.name || computeStateName(stateObj); | ||||||
|  |  | ||||||
|     const showState = this._config.show_state; |     const showState = this._config.show_state; | ||||||
|     const showName = this._config.show_name; |     const showName = this._config.show_name; | ||||||
|     const showIcon = this._config.show_icon; |     const showIcon = this._config.show_icon; | ||||||
|   | |||||||
| @@ -1,4 +1,3 @@ | |||||||
| import type { EntityNameItem } from "../../../common/entity/compute_entity_name_display"; |  | ||||||
| import type { ActionConfig } from "../../../data/lovelace/config/action"; | import type { ActionConfig } from "../../../data/lovelace/config/action"; | ||||||
| import type { LovelaceBadgeConfig } from "../../../data/lovelace/config/badge"; | import type { LovelaceBadgeConfig } from "../../../data/lovelace/config/badge"; | ||||||
| import type { LegacyStateFilter } from "../common/evaluate-filter"; | import type { LegacyStateFilter } from "../common/evaluate-filter"; | ||||||
| @@ -32,7 +31,7 @@ export interface StateLabelBadgeConfig extends LovelaceBadgeConfig { | |||||||
| export interface EntityBadgeConfig extends LovelaceBadgeConfig { | export interface EntityBadgeConfig extends LovelaceBadgeConfig { | ||||||
|   type: "entity"; |   type: "entity"; | ||||||
|   entity?: string; |   entity?: string; | ||||||
|   name?: string | EntityNameItem | EntityNameItem[]; |   name?: string; | ||||||
|   icon?: string; |   icon?: string; | ||||||
|   color?: string; |   color?: string; | ||||||
|   show_name?: boolean; |   show_name?: boolean; | ||||||
|   | |||||||
| @@ -43,8 +43,6 @@ class HuiHistoryChartCardFeature | |||||||
|  |  | ||||||
|   @state() private _coordinates?: [number, number][]; |   @state() private _coordinates?: [number, number][]; | ||||||
|  |  | ||||||
|   @state() private _yAxisOrigin?: number; |  | ||||||
|  |  | ||||||
|   private _interval?: number; |   private _interval?: number; | ||||||
|  |  | ||||||
|   static getStubConfig(): TrendGraphCardFeatureConfig { |   static getStubConfig(): TrendGraphCardFeatureConfig { | ||||||
| @@ -107,10 +105,7 @@ class HuiHistoryChartCardFeature | |||||||
|       `; |       `; | ||||||
|     } |     } | ||||||
|     return html` |     return html` | ||||||
|       <hui-graph-base |       <hui-graph-base .coordinates=${this._coordinates}></hui-graph-base> | ||||||
|         .coordinates=${this._coordinates} |  | ||||||
|         .yAxisOrigin=${this._yAxisOrigin} |  | ||||||
|       ></hui-graph-base> |  | ||||||
|     `; |     `; | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -128,15 +123,14 @@ class HuiHistoryChartCardFeature | |||||||
|     return subscribeHistoryStatesTimeWindow( |     return subscribeHistoryStatesTimeWindow( | ||||||
|       this.hass!, |       this.hass!, | ||||||
|       (historyStates) => { |       (historyStates) => { | ||||||
|         const { points, yAxisOrigin } = |         this._coordinates = | ||||||
|           coordinatesMinimalResponseCompressedState( |           coordinatesMinimalResponseCompressedState( | ||||||
|             historyStates[this.context!.entity_id!], |             historyStates[this.context!.entity_id!], | ||||||
|             this.clientWidth, |             hourToShow, | ||||||
|             this.clientHeight, |             500, | ||||||
|             this.clientWidth / 5 // sample to 1 point per 5 pixels |             2, | ||||||
|           ); |             undefined | ||||||
|         this._coordinates = points; |           ) || []; | ||||||
|         this._yAxisOrigin = yAxisOrigin; |  | ||||||
|       }, |       }, | ||||||
|       hourToShow, |       hourToShow, | ||||||
|       [this.context!.entity_id!] |       [this.context!.entity_id!] | ||||||
|   | |||||||
| @@ -16,6 +16,7 @@ import { | |||||||
| import type { | import type { | ||||||
|   BarSeriesOption, |   BarSeriesOption, | ||||||
|   CallbackDataParams, |   CallbackDataParams, | ||||||
|  |   LineSeriesOption, | ||||||
|   TopLevelFormatterParams, |   TopLevelFormatterParams, | ||||||
| } from "echarts/types/dist/shared"; | } from "echarts/types/dist/shared"; | ||||||
| import type { FrontendLocaleData } from "../../../../../data/translation"; | import type { FrontendLocaleData } from "../../../../../data/translation"; | ||||||
| @@ -26,7 +27,7 @@ import { | |||||||
|   formatDateVeryShort, |   formatDateVeryShort, | ||||||
| } from "../../../../../common/datetime/format_date"; | } from "../../../../../common/datetime/format_date"; | ||||||
| import { formatTime } from "../../../../../common/datetime/format_time"; | import { formatTime } from "../../../../../common/datetime/format_time"; | ||||||
| import type { ECOption } from "../../../../../resources/echarts/echarts"; | import type { ECOption } from "../../../../../resources/echarts"; | ||||||
| import { filterXSS } from "../../../../../common/util/xss"; | import { filterXSS } from "../../../../../common/util/xss"; | ||||||
|  |  | ||||||
| export function getSuggestedMax(dayDifference: number, end: Date): number { | export function getSuggestedMax(dayDifference: number, end: Date): number { | ||||||
| @@ -170,11 +171,10 @@ function formatTooltip( | |||||||
|       compare |       compare | ||||||
|         ? `${(showCompareYear ? formatDateShort : formatDateVeryShort)(date, locale, config)}: ` |         ? `${(showCompareYear ? formatDateShort : formatDateVeryShort)(date, locale, config)}: ` | ||||||
|         : "" |         : "" | ||||||
|     }${formatTime(date, locale, config)} – ${formatTime( |     }${formatTime(date, locale, config)}`; | ||||||
|       addHours(date, 1), |     if (params[0].componentSubType === "bar") { | ||||||
|       locale, |       period += ` – ${formatTime(addHours(date, 1), locale, config)}`; | ||||||
|       config |     } | ||||||
|     )}`; |  | ||||||
|   } |   } | ||||||
|   const title = `<h4 style="text-align: center; margin: 0;">${period}</h4>`; |   const title = `<h4 style="text-align: center; margin: 0;">${period}</h4>`; | ||||||
|  |  | ||||||
| @@ -281,6 +281,35 @@ export function fillDataGapsAndRoundCaps(datasets: BarSeriesOption[]) { | |||||||
|   }); |   }); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | export function fillLineGaps(datasets: LineSeriesOption[]) { | ||||||
|  |   const buckets = Array.from( | ||||||
|  |     new Set( | ||||||
|  |       datasets | ||||||
|  |         .map((dataset) => | ||||||
|  |           dataset.data!.map((datapoint) => Number(datapoint![0])) | ||||||
|  |         ) | ||||||
|  |         .flat() | ||||||
|  |     ) | ||||||
|  |   ).sort((a, b) => a - b); | ||||||
|  |   buckets.forEach((bucket, index) => { | ||||||
|  |     for (let i = datasets.length - 1; i >= 0; i--) { | ||||||
|  |       const dataPoint = datasets[i].data![index]; | ||||||
|  |       const item: any = | ||||||
|  |         dataPoint && typeof dataPoint === "object" && "value" in dataPoint | ||||||
|  |           ? dataPoint | ||||||
|  |           : { value: dataPoint }; | ||||||
|  |       const x = item.value?.[0]; | ||||||
|  |       if (x === undefined) { | ||||||
|  |         continue; | ||||||
|  |       } | ||||||
|  |       if (Number(x) !== bucket) { | ||||||
|  |         datasets[i].data?.splice(index, 0, [bucket, 0]); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |   return datasets; | ||||||
|  | } | ||||||
|  |  | ||||||
| export function getCompareTransform(start: Date, compareStart?: Date) { | export function getCompareTransform(start: Date, compareStart?: Date) { | ||||||
|   if (!compareStart) { |   if (!compareStart) { | ||||||
|     return (ts: Date) => ts; |     return (ts: Date) => ts; | ||||||
|   | |||||||
| @@ -36,7 +36,7 @@ import { | |||||||
|   getCompareTransform, |   getCompareTransform, | ||||||
| } from "./common/energy-chart-options"; | } from "./common/energy-chart-options"; | ||||||
| import { storage } from "../../../../common/decorators/storage"; | import { storage } from "../../../../common/decorators/storage"; | ||||||
| import type { ECOption } from "../../../../resources/echarts/echarts"; | import type { ECOption } from "../../../../resources/echarts"; | ||||||
| import { formatNumber } from "../../../../common/number/format_number"; | import { formatNumber } from "../../../../common/number/format_number"; | ||||||
| import type { CustomLegendOption } from "../../../../components/chart/ha-chart-base"; | import type { CustomLegendOption } from "../../../../components/chart/ha-chart-base"; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -2,22 +2,16 @@ import type { UnsubscribeFunc } from "home-assistant-js-websocket"; | |||||||
| import type { PropertyValues } from "lit"; | import type { PropertyValues } from "lit"; | ||||||
| import { css, html, LitElement, nothing } from "lit"; | import { css, html, LitElement, nothing } from "lit"; | ||||||
| import { customElement, property, state } from "lit/decorators"; | import { customElement, property, state } from "lit/decorators"; | ||||||
| import { mdiChartDonut, mdiChartBar } from "@mdi/js"; |  | ||||||
| import { classMap } from "lit/directives/class-map"; | import { classMap } from "lit/directives/class-map"; | ||||||
| import memoizeOne from "memoize-one"; | import memoizeOne from "memoize-one"; | ||||||
| import type { BarSeriesOption, PieSeriesOption } from "echarts/charts"; | import type { BarSeriesOption } from "echarts/charts"; | ||||||
| import { PieChart } from "echarts/charts"; |  | ||||||
| import type { ECElementEvent } from "echarts/types/dist/shared"; | import type { ECElementEvent } from "echarts/types/dist/shared"; | ||||||
| import { filterXSS } from "../../../../common/util/xss"; | import { filterXSS } from "../../../../common/util/xss"; | ||||||
| import { getGraphColorByIndex } from "../../../../common/color/colors"; | import { getGraphColorByIndex } from "../../../../common/color/colors"; | ||||||
| import { formatNumber } from "../../../../common/number/format_number"; | import { formatNumber } from "../../../../common/number/format_number"; | ||||||
| import "../../../../components/chart/ha-chart-base"; | import "../../../../components/chart/ha-chart-base"; | ||||||
| import type { EnergyData } from "../../../../data/energy"; | import type { EnergyData } from "../../../../data/energy"; | ||||||
| import { | import { getEnergyDataCollection } from "../../../../data/energy"; | ||||||
|   computeConsumptionData, |  | ||||||
|   getEnergyDataCollection, |  | ||||||
|   getSummedData, |  | ||||||
| } from "../../../../data/energy"; |  | ||||||
| import { | import { | ||||||
|   calculateStatisticSumGrowth, |   calculateStatisticSumGrowth, | ||||||
|   getStatisticLabel, |   getStatisticLabel, | ||||||
| @@ -28,12 +22,10 @@ import type { HomeAssistant } from "../../../../types"; | |||||||
| import type { LovelaceCard } from "../../types"; | import type { LovelaceCard } from "../../types"; | ||||||
| import type { EnergyDevicesGraphCardConfig } from "../types"; | import type { EnergyDevicesGraphCardConfig } from "../types"; | ||||||
| import { hasConfigChanged } from "../../common/has-changed"; | import { hasConfigChanged } from "../../common/has-changed"; | ||||||
| import type { ECOption } from "../../../../resources/echarts/echarts"; | import type { ECOption } from "../../../../resources/echarts"; | ||||||
| import "../../../../components/ha-card"; | import "../../../../components/ha-card"; | ||||||
| import { fireEvent } from "../../../../common/dom/fire_event"; | import { fireEvent } from "../../../../common/dom/fire_event"; | ||||||
| import { measureTextWidth } from "../../../../util/text"; | import { measureTextWidth } from "../../../../util/text"; | ||||||
| import "../../../../components/ha-icon-button"; |  | ||||||
| import { storage } from "../../../../common/decorators/storage"; |  | ||||||
|  |  | ||||||
| @customElement("hui-energy-devices-graph-card") | @customElement("hui-energy-devices-graph-card") | ||||||
| export class HuiEnergyDevicesGraphCard | export class HuiEnergyDevicesGraphCard | ||||||
| @@ -44,20 +36,10 @@ export class HuiEnergyDevicesGraphCard | |||||||
|  |  | ||||||
|   @state() private _config?: EnergyDevicesGraphCardConfig; |   @state() private _config?: EnergyDevicesGraphCardConfig; | ||||||
|  |  | ||||||
|   @state() private _chartData: (BarSeriesOption | PieSeriesOption)[] = []; |   @state() private _chartData: BarSeriesOption[] = []; | ||||||
|  |  | ||||||
|   @state() private _data?: EnergyData; |   @state() private _data?: EnergyData; | ||||||
|  |  | ||||||
|   @state() |  | ||||||
|   @storage({ |  | ||||||
|     key: "energy-devices-graph-chart-type", |  | ||||||
|     state: true, |  | ||||||
|     subscribe: false, |  | ||||||
|   }) |  | ||||||
|   private _chartType: "bar" | "pie" = "bar"; |  | ||||||
|  |  | ||||||
|   private _compoundStats: string[] = []; |  | ||||||
|  |  | ||||||
|   protected hassSubscribeRequiredHostProps = ["_config"]; |   protected hassSubscribeRequiredHostProps = ["_config"]; | ||||||
|  |  | ||||||
|   public hassSubscribe(): UnsubscribeFunc[] { |   public hassSubscribe(): UnsubscribeFunc[] { | ||||||
| @@ -94,16 +76,9 @@ export class HuiEnergyDevicesGraphCard | |||||||
|  |  | ||||||
|     return html` |     return html` | ||||||
|       <ha-card> |       <ha-card> | ||||||
|         <div class="card-header"> |         ${this._config.title | ||||||
|           <span>${this._config.title ? this._config.title : nothing}</span> |           ? html`<h1 class="card-header">${this._config.title}</h1>` | ||||||
|           <ha-icon-button |           : ""} | ||||||
|             .path=${this._chartType === "pie" ? mdiChartBar : mdiChartDonut} |  | ||||||
|             .label=${this.hass.localize( |  | ||||||
|               "ui.panel.lovelace.cards.energy.energy_devices_graph.change_chart_type" |  | ||||||
|             )} |  | ||||||
|             @click=${this._handleChartTypeChange} |  | ||||||
|           ></ha-icon-button> |  | ||||||
|         </div> |  | ||||||
|         <div |         <div | ||||||
|           class="content ${classMap({ |           class="content ${classMap({ | ||||||
|             "has-header": !!this._config.title, |             "has-header": !!this._config.title, | ||||||
| @@ -112,10 +87,9 @@ export class HuiEnergyDevicesGraphCard | |||||||
|           <ha-chart-base |           <ha-chart-base | ||||||
|             .hass=${this.hass} |             .hass=${this.hass} | ||||||
|             .data=${this._chartData} |             .data=${this._chartData} | ||||||
|             .options=${this._createOptions(this._chartData, this._chartType)} |             .options=${this._createOptions(this._chartData)} | ||||||
|             .height=${`${Math.max(300, (this._chartData[0]?.data?.length || 0) * 28 + 50)}px`} |             .height=${`${(this._chartData[0]?.data?.length || 0) * 28 + 50}px`} | ||||||
|             @chart-click=${this._handleChartClick} |             @chart-click=${this._handleChartClick} | ||||||
|             .extraComponents=${[PieChart]} |  | ||||||
|           ></ha-chart-base> |           ></ha-chart-base> | ||||||
|         </div> |         </div> | ||||||
|       </ha-card> |       </ha-card> | ||||||
| @@ -123,86 +97,71 @@ export class HuiEnergyDevicesGraphCard | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   private _renderTooltip(params: any) { |   private _renderTooltip(params: any) { | ||||||
|     const deviceName = filterXSS(this._getDeviceName(params.name)); |     const deviceName = filterXSS(this._getDeviceName(params.value[1])); | ||||||
|     const title = `<h4 style="text-align: center; margin: 0;">${deviceName}</h4>`; |     const title = `<h4 style="text-align: center; margin: 0;">${deviceName}</h4>`; | ||||||
|     const value = `${formatNumber( |     const value = `${formatNumber( | ||||||
|       params.value[0] as number, |       params.value[0] as number, | ||||||
|       this.hass.locale, |       this.hass.locale, | ||||||
|       params.value < 0.1 ? { maximumFractionDigits: 3 } : undefined |       params.value[0] < 0.1 ? { maximumFractionDigits: 3 } : undefined | ||||||
|     )} kWh`; |     )} kWh`; | ||||||
|     return `${title}${params.marker} ${params.seriesName}: ${value}`; |     return `${title}${params.marker} ${params.seriesName}: ${value}`; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private _createOptions = memoizeOne( |   private _createOptions = memoizeOne((data: BarSeriesOption[]): ECOption => { | ||||||
|     ( |     const isMobile = window.matchMedia( | ||||||
|       data: (BarSeriesOption | PieSeriesOption)[], |       "all and (max-width: 450px), all and (max-height: 500px)" | ||||||
|       chartType: "bar" | "pie" |     ).matches; | ||||||
|     ): ECOption => { |     return { | ||||||
|       const options: ECOption = { |       xAxis: { | ||||||
|         grid: { |         type: "value", | ||||||
|           top: 5, |         name: "kWh", | ||||||
|           left: 5, |       }, | ||||||
|           right: 40, |       yAxis: { | ||||||
|           bottom: 0, |         type: "category", | ||||||
|           containLabel: true, |         inverse: true, | ||||||
|  |         triggerEvent: true, | ||||||
|  |         // take order from data | ||||||
|  |         data: data[0]?.data?.map((d: any) => d.value[1]), | ||||||
|  |         axisLabel: { | ||||||
|  |           formatter: this._getDeviceName.bind(this), | ||||||
|  |           overflow: "truncate", | ||||||
|  |           fontSize: 12, | ||||||
|  |           margin: 5, | ||||||
|  |           width: Math.min( | ||||||
|  |             isMobile ? 100 : 200, | ||||||
|  |             Math.max( | ||||||
|  |               ...(data[0]?.data?.map( | ||||||
|  |                 (d: any) => | ||||||
|  |                   measureTextWidth(this._getDeviceName(d.value[1]), 12) + 5 | ||||||
|  |               ) || []) | ||||||
|  |             ) | ||||||
|  |           ), | ||||||
|         }, |         }, | ||||||
|         tooltip: { |       }, | ||||||
|           show: true, |       grid: { | ||||||
|           formatter: this._renderTooltip.bind(this), |         top: 5, | ||||||
|         }, |         left: 5, | ||||||
|         xAxis: { show: false }, |         right: 40, | ||||||
|         yAxis: { show: false }, |         bottom: 0, | ||||||
|       }; |         containLabel: true, | ||||||
|       if (chartType === "bar") { |       }, | ||||||
|         const isMobile = window.matchMedia( |       tooltip: { | ||||||
|           "all and (max-width: 450px), all and (max-height: 500px)" |         show: true, | ||||||
|         ).matches; |         formatter: this._renderTooltip.bind(this), | ||||||
|         options.xAxis = { |       }, | ||||||
|           show: true, |     }; | ||||||
|           type: "value", |   }); | ||||||
|           name: "kWh", |  | ||||||
|         }; |  | ||||||
|         options.yAxis = { |  | ||||||
|           show: true, |  | ||||||
|           type: "category", |  | ||||||
|           inverse: true, |  | ||||||
|           triggerEvent: true, |  | ||||||
|           // take order from data |  | ||||||
|           data: data[0]?.data?.map((d: any) => d.name), |  | ||||||
|           axisLabel: { |  | ||||||
|             formatter: this._getDeviceName.bind(this), |  | ||||||
|             overflow: "truncate", |  | ||||||
|             fontSize: 12, |  | ||||||
|             margin: 5, |  | ||||||
|             width: Math.min( |  | ||||||
|               isMobile ? 100 : 200, |  | ||||||
|               Math.max( |  | ||||||
|                 ...(data[0]?.data?.map( |  | ||||||
|                   (d: any) => |  | ||||||
|                     measureTextWidth(this._getDeviceName(d.name), 12) + 5 |  | ||||||
|                 ) || []) |  | ||||||
|               ) |  | ||||||
|             ), |  | ||||||
|           }, |  | ||||||
|         }; |  | ||||||
|       } |  | ||||||
|       return options; |  | ||||||
|     } |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   private _getDeviceName(statisticId: string): string { |   private _getDeviceName(statisticId: string): string { | ||||||
|     const suffix = this._compoundStats.includes(statisticId) |  | ||||||
|       ? ` (${this.hass.localize("ui.panel.lovelace.cards.energy.energy_devices_graph.untracked")})` |  | ||||||
|       : ""; |  | ||||||
|     return ( |     return ( | ||||||
|       (this._data?.prefs.device_consumption.find( |       this._data?.prefs.device_consumption.find( | ||||||
|         (d) => d.stat_consumption === statisticId |         (d) => d.stat_consumption === statisticId | ||||||
|       )?.name || |       )?.name || | ||||||
|         getStatisticLabel( |       getStatisticLabel( | ||||||
|           this.hass, |         this.hass, | ||||||
|           statisticId, |         statisticId, | ||||||
|           this._data?.statsMetadata[statisticId] |         this._data?.statsMetadata[statisticId] | ||||||
|         )) + suffix |       ) | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -210,105 +169,60 @@ export class HuiEnergyDevicesGraphCard | |||||||
|     const data = energyData.stats; |     const data = energyData.stats; | ||||||
|     const compareData = energyData.statsCompare; |     const compareData = energyData.statsCompare; | ||||||
|  |  | ||||||
|     const chartData: NonNullable<(BarSeriesOption | PieSeriesOption)["data"]> = |     const chartData: NonNullable<BarSeriesOption["data"]> = []; | ||||||
|       []; |     const chartDataCompare: NonNullable<BarSeriesOption["data"]> = []; | ||||||
|     const chartDataCompare: NonNullable< |  | ||||||
|       (BarSeriesOption | PieSeriesOption)["data"] |  | ||||||
|     > = []; |  | ||||||
|  |  | ||||||
|     const datasets: (BarSeriesOption | PieSeriesOption)[] = [ |     const datasets: BarSeriesOption[] = [ | ||||||
|       { |       { | ||||||
|         type: this._chartType, |         type: "bar", | ||||||
|         radius: [compareData ? "50%" : "40%", "70%"], |  | ||||||
|         universalTransition: true, |  | ||||||
|         name: this.hass.localize( |         name: this.hass.localize( | ||||||
|           "ui.panel.lovelace.cards.energy.energy_devices_graph.energy_usage" |           "ui.panel.lovelace.cards.energy.energy_devices_graph.energy_usage" | ||||||
|         ), |         ), | ||||||
|         itemStyle: { |         itemStyle: { | ||||||
|           borderRadius: this._chartType === "bar" ? [0, 4, 4, 0] : 4, |           borderRadius: [0, 4, 4, 0], | ||||||
|         }, |         }, | ||||||
|         data: chartData, |         data: chartData, | ||||||
|         barWidth: compareData ? 10 : 20, |         barWidth: compareData ? 10 : 20, | ||||||
|         cursor: "default", |         cursor: "default", | ||||||
|         minShowLabelAngle: 15, |       }, | ||||||
|         label: |  | ||||||
|           this._chartType === "pie" |  | ||||||
|             ? { |  | ||||||
|                 formatter: ({ name }) => this._getDeviceName(name), |  | ||||||
|               } |  | ||||||
|             : undefined, |  | ||||||
|       } as BarSeriesOption | PieSeriesOption, |  | ||||||
|     ]; |     ]; | ||||||
|  |  | ||||||
|     if (compareData) { |     if (compareData) { | ||||||
|       datasets.push({ |       datasets.push({ | ||||||
|         type: this._chartType, |         type: "bar", | ||||||
|         radius: ["30%", "50%"], |  | ||||||
|         universalTransition: true, |  | ||||||
|         name: this.hass.localize( |         name: this.hass.localize( | ||||||
|           "ui.panel.lovelace.cards.energy.energy_devices_graph.previous_energy_usage" |           "ui.panel.lovelace.cards.energy.energy_devices_graph.previous_energy_usage" | ||||||
|         ), |         ), | ||||||
|         itemStyle: { |         itemStyle: { | ||||||
|           borderRadius: this._chartType === "bar" ? [0, 4, 4, 0] : 4, |           borderRadius: [0, 4, 4, 0], | ||||||
|         }, |         }, | ||||||
|         data: chartDataCompare, |         data: chartDataCompare, | ||||||
|         barWidth: 10, |         barWidth: 10, | ||||||
|         cursor: "default", |         cursor: "default", | ||||||
|         label: this._chartType === "pie" ? { show: false } : undefined, |       }); | ||||||
|         emphasis: |  | ||||||
|           this._chartType === "pie" |  | ||||||
|             ? { |  | ||||||
|                 focus: "series", |  | ||||||
|                 blurScope: "global", |  | ||||||
|               } |  | ||||||
|             : undefined, |  | ||||||
|       } as BarSeriesOption | PieSeriesOption); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     const computedStyle = getComputedStyle(this); |     const computedStyle = getComputedStyle(this); | ||||||
|  |  | ||||||
|     this._compoundStats = energyData.prefs.device_consumption |     const exclude = this._config?.hide_compound_stats | ||||||
|       .map((d) => d.included_in_stat) |       ? energyData.prefs.device_consumption | ||||||
|       .filter(Boolean) as string[]; |           .map((d) => d.included_in_stat) | ||||||
|  |           .filter(Boolean) | ||||||
|  |       : []; | ||||||
|  |  | ||||||
|     const devices = energyData.prefs.device_consumption; |     energyData.prefs.device_consumption.forEach((device, id) => { | ||||||
|     const devicesTotals: Record<string, number> = {}; |       if (exclude.includes(device.stat_consumption)) { | ||||||
|     devices.forEach((device) => { |         return; | ||||||
|       devicesTotals[device.stat_consumption] = |       } | ||||||
|  |       const value = | ||||||
|         device.stat_consumption in data |         device.stat_consumption in data | ||||||
|           ? calculateStatisticSumGrowth(data[device.stat_consumption]) || 0 |           ? calculateStatisticSumGrowth(data[device.stat_consumption]) || 0 | ||||||
|           : 0; |           : 0; | ||||||
|     }); |       const color = getGraphColorByIndex(id, computedStyle); | ||||||
|     const devicesTotalsCompare: Record<string, number> = {}; |  | ||||||
|     if (compareData) { |  | ||||||
|       devices.forEach((device) => { |  | ||||||
|         devicesTotalsCompare[device.stat_consumption] = |  | ||||||
|           device.stat_consumption in compareData |  | ||||||
|             ? calculateStatisticSumGrowth( |  | ||||||
|                 compareData[device.stat_consumption] |  | ||||||
|               ) || 0 |  | ||||||
|             : 0; |  | ||||||
|       }); |  | ||||||
|     } |  | ||||||
|     devices.forEach((device, idx) => { |  | ||||||
|       let value = devicesTotals[device.stat_consumption]; |  | ||||||
|       if (!this._config?.hide_compound_stats) { |  | ||||||
|         const childSum = devices.reduce((acc, d) => { |  | ||||||
|           if (d.included_in_stat === device.stat_consumption) { |  | ||||||
|             return acc + devicesTotals[d.stat_consumption]; |  | ||||||
|           } |  | ||||||
|           return acc; |  | ||||||
|         }, 0); |  | ||||||
|         value -= Math.min(value, childSum); |  | ||||||
|       } else if (this._compoundStats.includes(device.stat_consumption)) { |  | ||||||
|         return; |  | ||||||
|       } |  | ||||||
|       const color = getGraphColorByIndex(idx, computedStyle); |  | ||||||
|  |  | ||||||
|       chartData.push({ |       chartData.push({ | ||||||
|         id: device.stat_consumption, |         id, | ||||||
|         value: [value, device.stat_consumption] as any, |         value: [value, device.stat_consumption], | ||||||
|         name: device.stat_consumption, |  | ||||||
|         itemStyle: { |         itemStyle: { | ||||||
|           color: color + "7F", |           color: color + "7F", | ||||||
|           borderColor: color, |           borderColor: color, | ||||||
| @@ -316,24 +230,16 @@ export class HuiEnergyDevicesGraphCard | |||||||
|       }); |       }); | ||||||
|  |  | ||||||
|       if (compareData) { |       if (compareData) { | ||||||
|         let compareValue = |         const compareValue = | ||||||
|           device.stat_consumption in compareData |           device.stat_consumption in compareData | ||||||
|             ? calculateStatisticSumGrowth( |             ? calculateStatisticSumGrowth( | ||||||
|                 compareData[device.stat_consumption] |                 compareData[device.stat_consumption] | ||||||
|               ) || 0 |               ) || 0 | ||||||
|             : 0; |             : 0; | ||||||
|         const compareChildSum = devices.reduce((acc, d) => { |  | ||||||
|           if (d.included_in_stat === device.stat_consumption) { |  | ||||||
|             return acc + devicesTotalsCompare[d.stat_consumption]; |  | ||||||
|           } |  | ||||||
|           return acc; |  | ||||||
|         }, 0); |  | ||||||
|         compareValue -= Math.min(compareValue, compareChildSum); |  | ||||||
|  |  | ||||||
|         chartDataCompare.push({ |         chartDataCompare.push({ | ||||||
|           id: device.stat_consumption, |           id, | ||||||
|           value: [compareValue, device.stat_consumption] as any, |           value: [compareValue, device.stat_consumption], | ||||||
|           name: device.stat_consumption, |  | ||||||
|           itemStyle: { |           itemStyle: { | ||||||
|             color: color + "32", |             color: color + "32", | ||||||
|             borderColor: color + "7F", |             borderColor: color + "7F", | ||||||
| @@ -343,62 +249,11 @@ export class HuiEnergyDevicesGraphCard | |||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     chartData.sort((a: any, b: any) => b.value[0] - a.value[0]); |     chartData.sort((a: any, b: any) => b.value[0] - a.value[0]); | ||||||
|     if (compareData) { |  | ||||||
|       datasets[1].data = chartData.map((d) => |  | ||||||
|         chartDataCompare.find((d2) => (d2 as any).id === d.id) |  | ||||||
|       ) as typeof chartDataCompare; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     datasets.forEach((dataset) => { |     chartData.length = Math.min( | ||||||
|       dataset.data!.length = Math.min( |       this._config?.max_devices || Infinity, | ||||||
|         this._config?.max_devices || Infinity, |       chartData.length | ||||||
|         dataset.data!.length |     ); | ||||||
|       ); |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     if (this._chartType === "pie") { |  | ||||||
|       const { summedData } = getSummedData(energyData); |  | ||||||
|       const { consumption } = computeConsumptionData(summedData); |  | ||||||
|       const totalUsed = consumption.total.used_total; |  | ||||||
|       const showUntracked = |  | ||||||
|         "from_grid" in summedData || |  | ||||||
|         "solar" in summedData || |  | ||||||
|         "from_battery" in summedData; |  | ||||||
|       const untracked = showUntracked |  | ||||||
|         ? totalUsed - |  | ||||||
|           chartData.reduce((acc: number, d: any) => acc + d.value[0], 0) |  | ||||||
|         : 0; |  | ||||||
|       datasets.push({ |  | ||||||
|         type: "pie", |  | ||||||
|         radius: ["0%", compareData ? "30%" : "40%"], |  | ||||||
|         name: this.hass.localize( |  | ||||||
|           "ui.panel.lovelace.cards.energy.energy_devices_graph.total_energy_usage" |  | ||||||
|         ), |  | ||||||
|         data: [totalUsed], |  | ||||||
|         label: { |  | ||||||
|           show: true, |  | ||||||
|           position: "center", |  | ||||||
|           color: computedStyle.getPropertyValue("--secondary-text-color"), |  | ||||||
|           fontSize: computedStyle.getPropertyValue("--ha-font-size-l"), |  | ||||||
|           lineHeight: 24, |  | ||||||
|           fontWeight: "bold", |  | ||||||
|           formatter: `{a}\n${formatNumber(totalUsed, this.hass.locale)} kWh`, |  | ||||||
|         }, |  | ||||||
|         cursor: "default", |  | ||||||
|         itemStyle: { |  | ||||||
|           color: "rgba(0, 0, 0, 0)", |  | ||||||
|         }, |  | ||||||
|         tooltip: { |  | ||||||
|           formatter: () => |  | ||||||
|             untracked > 0 |  | ||||||
|               ? this.hass.localize( |  | ||||||
|                   "ui.panel.lovelace.cards.energy.energy_devices_graph.includes_untracked", |  | ||||||
|                   { num: formatNumber(untracked, this.hass.locale) } |  | ||||||
|                 ) |  | ||||||
|               : "", |  | ||||||
|         }, |  | ||||||
|       }); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     this._chartData = datasets; |     this._chartData = datasets; | ||||||
|     await this.updateComplete; |     await this.updateComplete; | ||||||
| @@ -413,26 +268,11 @@ export class HuiEnergyDevicesGraphCard | |||||||
|       fireEvent(this, "hass-more-info", { |       fireEvent(this, "hass-more-info", { | ||||||
|         entityId: e.detail.value as string, |         entityId: e.detail.value as string, | ||||||
|       }); |       }); | ||||||
|     } else if ( |  | ||||||
|       e.detail.seriesType === "pie" && |  | ||||||
|       e.detail.event?.target?.type === "tspan" // label |  | ||||||
|     ) { |  | ||||||
|       fireEvent(this, "hass-more-info", { |  | ||||||
|         entityId: (e.detail.data as any).id as string, |  | ||||||
|       }); |  | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private _handleChartTypeChange(): void { |  | ||||||
|     this._chartType = this._chartType === "pie" ? "bar" : "pie"; |  | ||||||
|     this._getStatistics(this._data!); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   static styles = css` |   static styles = css` | ||||||
|     .card-header { |     .card-header { | ||||||
|       display: flex; |  | ||||||
|       justify-content: space-between; |  | ||||||
|       align-items: center; |  | ||||||
|       padding-bottom: 0; |       padding-bottom: 0; | ||||||
|     } |     } | ||||||
|     .content { |     .content { | ||||||
| @@ -444,11 +284,6 @@ export class HuiEnergyDevicesGraphCard | |||||||
|     ha-chart-base { |     ha-chart-base { | ||||||
|       --chart-max-height: none; |       --chart-max-height: none; | ||||||
|     } |     } | ||||||
|     ha-icon-button { |  | ||||||
|       transform: rotate(90deg); |  | ||||||
|       color: var(--secondary-text-color); |  | ||||||
|       cursor: pointer; |  | ||||||
|     } |  | ||||||
|   `; |   `; | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user