mirror of
				https://github.com/home-assistant/frontend.git
				synced 2025-10-26 03:59:43 +00:00 
			
		
		
		
	Compare commits
	
		
			309 Commits
		
	
	
		
			update-sta
			...
			sensor_dev
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | c7e9ee785d | ||
|   | 079cc39a6e | ||
|   | d6a1d5af79 | ||
|   | c0dce08e19 | ||
|   | a7a347ed05 | ||
|   | 2d9b50defc | ||
|   | 840858b18c | ||
|   | afd2e71f6c | ||
|   | 88af0aa788 | ||
|   | 49124f6f09 | ||
|   | 73f5580555 | ||
|   | bdde5268c6 | ||
|   | 15e972c158 | ||
|   | 0fc4c24f5a | ||
|   | 9eba50df0c | ||
|   | 0e0e07437f | ||
|   | 6ac51ede52 | ||
|   | ccf1fb573a | ||
|   | fa537968c4 | ||
|   | 6bf2111a3c | ||
|   | ddf1cc0733 | ||
|   | 9c1d1cb6f6 | ||
|   | 470225abde | ||
|   | ee230b86c1 | ||
|   | f927fc64a9 | ||
|   | 03677c33f7 | ||
|   | bc36a206da | ||
|   | af06ab1e2d | ||
|   | 3e2135a485 | ||
|   | 2e7f8fb46f | ||
|   | 102568c4bd | ||
|   | 4fcdae842e | ||
|   | ea19740f5a | ||
|   | 3e0942b631 | ||
|   | 0261cea796 | ||
|   | 5247b2813f | ||
|   | 8a5090684e | ||
|   | 1784ba5e68 | ||
|   | 4fbe9a7b10 | ||
|   | 1ca9c7838a | ||
|   | 4fc2c3ef05 | ||
|   | 73ff8e28a8 | ||
|   | dde1c5e03c | ||
|   | 01eed22592 | ||
|   | 94ebb63589 | ||
|   | 29119db5ce | ||
|   | 9908162ac2 | ||
|   | 1e929ae78a | ||
|   | ab5df0fe6e | ||
|   | d5010dda9e | ||
|   | 4ac097f32b | ||
|   | 5d3d15072f | ||
|   | 5c53bc4225 | ||
|   | d5a307f8f4 | ||
|   | a27dd1e7f1 | ||
|   | c86ed1fb3e | ||
|   | 7fa7a48072 | ||
|   | 4e0fc8ee08 | ||
|   | 5f6490e54e | ||
|   | db78b046a2 | ||
|   | c37fe1e7ff | ||
|   | f1ec479d41 | ||
|   | e01cb3ca82 | ||
|   | b8d3c68a7a | ||
|   | 641003bb2a | ||
|   | 3358fc2b18 | ||
|   | dcf50e055b | ||
|   | 1fa04baa16 | ||
|   | 84ffa2369a | ||
|   | cc27ddb362 | ||
|   | c4dc6bfb0d | ||
|   | 4fbcc30a37 | ||
|   | 4916527e5f | ||
|   | fad8a27232 | ||
|   | a993d3a753 | ||
|   | 5dfe17a43a | ||
|   | 9b6c935ffb | ||
|   | f4e28da0a3 | ||
|   | 294a69d7e4 | ||
|   | f89b8cffcf | ||
|   | 99fd3a1b6f | ||
|   | 246e426182 | ||
|   | 9f1e9b43fe | ||
|   | 8301ae262c | ||
|   | d968fe41ee | ||
|   | db830e9014 | ||
|   | fc6b594a27 | ||
|   | 86dbf99ebe | ||
|   | 68e7ce1883 | ||
|   | e9003ac35e | ||
|   | 1dd5214b42 | ||
|   | 96738350bb | ||
|   | 5bdecf57cf | ||
|   | ec12282f8c | ||
|   | 552dbca201 | ||
|   | 0bbc0ebb3c | ||
|   | ac7acc5802 | ||
|   | 64e1d160d1 | ||
|   | 8e51878b6d | ||
|   | 7c94ced303 | ||
|   | a040e1d5e0 | ||
|   | 87c7407857 | ||
|   | d0d0c44ec7 | ||
|   | 4cdff3faea | ||
|   | 0dac10aa23 | ||
|   | 4b8b14a69d | ||
|   | 9d28df31bd | ||
|   | 8258641443 | ||
|   | dfcb0f6ba0 | ||
|   | 2e10eb04b6 | ||
|   | b4b52d3872 | ||
|   | 3873203721 | ||
|   | ccb91e0b49 | ||
|   | bd20c15a55 | ||
|   | 0936fd9ae4 | ||
|   | adefc7a4e2 | ||
|   | 8f8017ecff | ||
|   | 604b79696e | ||
|   | 8c445f6409 | ||
|   | 797c871137 | ||
|   | 24829bd903 | ||
|   | add92a559d | ||
|   | 7f086c0900 | ||
|   | 17018c0f26 | ||
|   | cd6a478130 | ||
|   | 4f6d7ca5c9 | ||
|   | c2994343b4 | ||
|   | e5f77c35d4 | ||
|   | a9e5a5dd44 | ||
|   | 1159798b8d | ||
|   | 437de42c55 | ||
|   | 89e0bb3f16 | ||
|   | 28c9631b6c | ||
|   | 8882624618 | ||
|   | a769f84755 | ||
|   | 7abf9c2473 | ||
|   | 298296a81f | ||
|   | 6907fa5c8e | ||
|   | 546461b70f | ||
|   | 4031009c26 | ||
|   | 91e4557625 | ||
|   | f0c4b92dbb | ||
|   | ffac3d055e | ||
|   | 04ae8c9d14 | ||
|   | 0158610d42 | ||
|   | 5ab6121581 | ||
|   | 3d9c31aef9 | ||
|   | acfeea5c92 | ||
|   | 75e8e17073 | ||
|   | 976fd4b32d | ||
|   | 49beafbe5f | ||
|   | 151f8d5524 | ||
|   | 48355aa98e | ||
|   | fc31929f41 | ||
|   | b7c149fcc1 | ||
|   | 02d058561b | ||
|   | 4e57fb1ec1 | ||
|   | 30f79c5a46 | ||
|   | 30f7252d84 | ||
|   | 8af795a7ce | ||
|   | 8576eeae41 | ||
|   | cd740ed135 | ||
|   | 892f774792 | ||
|   | aa504fe1f8 | ||
|   | be491451d5 | ||
|   | bad184210d | ||
|   | a43b3b64b3 | ||
|   | aa831a9adf | ||
|   | 43d4f55392 | ||
|   | 130c66fb24 | ||
|   | 684c232c8c | ||
|   | 09f8f816d1 | ||
|   | 1719d062b3 | ||
|   | 87290c4330 | ||
|   | fec0dc0032 | ||
|   | 70ca27c8c9 | ||
|   | 9ae1f01ad6 | ||
|   | 0113cc3cf6 | ||
|   | 2a98ace0b3 | ||
|   | 5f69a4c165 | ||
|   | 8db22d4f88 | ||
|   | 3204dbfc4d | ||
|   | 430b47fc4a | ||
|   | 5d8b3227f3 | ||
|   | b341ee9d38 | ||
|   | e6dbbc31a8 | ||
|   | 0010bf5a8f | ||
|   | 6e2e80a297 | ||
|   | aa9ff01030 | ||
|   | 7f8ecf57d7 | ||
|   | 6be6755f6f | ||
|   | 64459a06c6 | ||
|   | df35496c6e | ||
|   | aa988c758d | ||
|   | 1dd1095d19 | ||
|   | 7e68393c84 | ||
|   | 540c06c9f7 | ||
|   | f633cc2b0d | ||
|   | 1baaf76471 | ||
|   | 8263e299a8 | ||
|   | ebd6a26554 | ||
|   | 5335772a7a | ||
|   | f5b5414461 | ||
|   | 1e6f402d0f | ||
|   | ed9d886009 | ||
|   | 940f5c0002 | ||
|   | 15d1b8b2ac | ||
|   | 73855e6f99 | ||
|   | d230541256 | ||
|   | b1f369a355 | ||
|   | e6d1e86c64 | ||
|   | eb1f94c370 | ||
|   | 27750b8b5d | ||
|   | 564a725284 | ||
|   | a5ee610af5 | ||
|   | eaf97ee7f5 | ||
|   | a14d75deec | ||
|   | 72b5721c88 | ||
|   | 94b4b818aa | ||
|   | 98699b640a | ||
|   | decc0d3e0d | ||
|   | 2281f5bafa | ||
|   | 6cac7eeff0 | ||
|   | 794bc161c8 | ||
|   | 28cd9b6408 | ||
|   | 9b4c6eea63 | ||
|   | afe044d152 | ||
|   | dc2038916b | ||
|   | cf8e2a6d02 | ||
|   | 3269b2878b | ||
|   | 29e1b7b452 | ||
|   | 3d6d07e5bd | ||
|   | 7bac41fe41 | ||
|   | 6e4b027575 | ||
|   | 728c391b5d | ||
|   | 8999ca2ea0 | ||
|   | 4fc0617289 | ||
|   | 494cc3a569 | ||
|   | cc177ef911 | ||
|   | 41ec65ef3d | ||
|   | 79e1e195a0 | ||
|   | dfbf7fb436 | ||
|   | f37a5fa021 | ||
|   | 5e2fcf928c | ||
|   | bc6ef7780c | ||
|   | b29563a254 | ||
|   | fe8a1152c4 | ||
|   | 26689a0a85 | ||
|   | 4f6a241817 | ||
|   | eae7e82127 | ||
|   | 9500ac498c | ||
|   | 5c5459bcaf | ||
|   | 246724c59e | ||
|   | 8f5c9295d3 | ||
|   | 0abafff4c9 | ||
|   | f88ce269a7 | ||
|   | 0dc56d7983 | ||
|   | cbd0ef6b65 | ||
|   | f923228078 | ||
|   | b55c7edd70 | ||
|   | bfb90632ac | ||
|   | 3a664d45a9 | ||
|   | 53607fe8c6 | ||
|   | 9dec0f8ccd | ||
|   | 89f4fe9d20 | ||
|   | f43655eea5 | ||
|   | 6563984fdd | ||
|   | 16d8eb0be3 | ||
|   | 965fc9bc4e | ||
|   | 56cb958a47 | ||
|   | f5feb1d8aa | ||
|   | e95065ed08 | ||
|   | 68a411838d | ||
|   | ba63ab8b7a | ||
|   | 26d4599ef4 | ||
|   | d049990f04 | ||
|   | 9c8d683a19 | ||
|   | 901677bbdf | ||
|   | 8bb2374b1b | ||
|   | 520896a3c2 | ||
|   | 92db272759 | ||
|   | fc654d86c6 | ||
|   | 523afe2f6f | ||
|   | 460b9003fc | ||
|   | 2ac0ad1d98 | ||
|   | a321432175 | ||
|   | 63c9b3f830 | ||
|   | 806b1296b0 | ||
|   | 7f90ffa82f | ||
|   | db33c38e21 | ||
|   | a8c1fdd21e | ||
|   | d86a18b80b | ||
|   | bef6591548 | ||
|   | e1c07f109c | ||
|   | fb66d224ae | ||
|   | ee1fd3e865 | ||
|   | a9bfea233c | ||
|   | 35cc291118 | ||
|   | 35a41b3490 | ||
|   | db7cac5782 | ||
|   | 099fa706a0 | ||
|   | f59cb661cd | ||
|   | ed84ce9692 | ||
|   | 9912d427f2 | ||
|   | 76f574f875 | ||
|   | ac90bb7088 | ||
|   | ce9f83e9a2 | ||
|   | 51938fb51f | ||
|   | c85236e251 | 
| @@ -16,6 +16,9 @@ | ||||
|     "runem.lit-plugin", | ||||
|     "ms-python.vscode-pylance" | ||||
|   ], | ||||
|   "containerEnv": { | ||||
|     "WORKSPACE_DIRECTORY": "${containerWorkspaceFolder}" | ||||
|   }, | ||||
|   "settings": { | ||||
|     "terminal.integrated.shell.linux": "/bin/bash", | ||||
|     "files.eol": "\n", | ||||
|   | ||||
							
								
								
									
										15
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										15
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -10,10 +10,18 @@ env: | ||||
|   NODE_VERSION: 14 | ||||
|   NODE_OPTIONS: --max_old_space_size=6144 | ||||
|  | ||||
| # Set default workflow permissions | ||||
| # All scopes not mentioned here are set to no access | ||||
| # https://docs.github.com/en/actions/security-guides/automatic-token-authentication#permissions-for-the-github_token | ||||
| permissions: | ||||
|   actions: none | ||||
|  | ||||
| jobs: | ||||
|   release: | ||||
|     name: Release | ||||
|     runs-on: ubuntu-latest | ||||
|     permissions: | ||||
|       contents: write  # Required to upload release assets | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@v2 | ||||
| @@ -47,6 +55,13 @@ jobs: | ||||
|  | ||||
|           script/release | ||||
|  | ||||
|       - name: Upload release assets | ||||
|         uses: softprops/action-gh-release@v0.1.14 | ||||
|         with: | ||||
|           files: | | ||||
|             dist/*.whl | ||||
|             dist/*.tar.gz | ||||
|  | ||||
|   wheels-init: | ||||
|     name: Init wheels build | ||||
|     needs: release | ||||
|   | ||||
							
								
								
									
										631
									
								
								.yarn/releases/yarn-3.0.2.cjs
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										631
									
								
								.yarn/releases/yarn-3.0.2.cjs
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										785
									
								
								.yarn/releases/yarn-3.2.0.cjs
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										785
									
								
								.yarn/releases/yarn-3.2.0.cjs
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -6,4 +6,4 @@ plugins: | ||||
|   - path: .yarn/plugins/@yarnpkg/plugin-interactive-tools.cjs | ||||
|     spec: "@yarnpkg/plugin-interactive-tools" | ||||
|  | ||||
| yarnPath: .yarn/releases/yarn-3.0.2.cjs | ||||
| yarnPath: .yarn/releases/yarn-3.2.0.cjs | ||||
|   | ||||
| @@ -2,7 +2,7 @@ | ||||
|  | ||||
| This is the repository for the official [Home Assistant](https://home-assistant.io) frontend. | ||||
|  | ||||
| [](https://demo.home-assistant.io/) | ||||
| [](https://demo.home-assistant.io/) | ||||
|  | ||||
| - [View demo of Home Assistant](https://demo.home-assistant.io/) | ||||
| - [More information about Home Assistant](https://home-assistant.io) | ||||
|   | ||||
| @@ -33,6 +33,10 @@ module.exports.emptyPackages = ({ latestBuild, isHassioBuild }) => | ||||
|       require.resolve( | ||||
|         path.resolve(paths.polymer_dir, "src/components/ha-icon.ts") | ||||
|       ), | ||||
|     isHassioBuild && | ||||
|       require.resolve( | ||||
|         path.resolve(paths.polymer_dir, "src/components/ha-icon-picker.ts") | ||||
|       ), | ||||
|   ].filter(Boolean); | ||||
|  | ||||
| module.exports.definedVars = ({ isProdBuild, latestBuild, defineOverlay }) => ({ | ||||
|   | ||||
| @@ -3,7 +3,7 @@ | ||||
| const gulp = require("gulp"); | ||||
| const fs = require("fs"); | ||||
| const path = require("path"); | ||||
| const marked = require("marked"); | ||||
| const { marked } = require("marked"); | ||||
| const glob = require("glob"); | ||||
| const yaml = require("js-yaml"); | ||||
|  | ||||
|   | ||||
| @@ -7,7 +7,7 @@ const source = require("vinyl-source-stream"); | ||||
| const vinylBuffer = require("vinyl-buffer"); | ||||
| const gulp = require("gulp"); | ||||
| const fs = require("fs"); | ||||
| const foreach = require("gulp-foreach"); | ||||
| const flatmap = require("gulp-flatmap"); | ||||
| const merge = require("gulp-merge-json"); | ||||
| const rename = require("gulp-rename"); | ||||
| const transform = require("gulp-json-transform"); | ||||
| @@ -183,7 +183,7 @@ gulp.task("build-merged-translations", () => | ||||
|     }) | ||||
|     .pipe(transform((data, file) => lokaliseTransform(data, data, file))) | ||||
|     .pipe( | ||||
|       foreach((stream, file) => { | ||||
|       flatmap((stream, file) => { | ||||
|         // For each language generate a merged json file. It begins with the master | ||||
|         // translation as a failsafe for untranslated strings, and merges all parent | ||||
|         // tags into one file for each specific subtag | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import { customElement, property, query } from "lit/decorators"; | ||||
| import { fireEvent } from "../../../../src/common/dom/fire_event"; | ||||
| import { LovelaceConfig } from "../../../../src/data/lovelace"; | ||||
| import { Lovelace } from "../../../../src/panels/lovelace/types"; | ||||
| @@ -20,6 +20,8 @@ class HcLovelace extends LitElement { | ||||
|  | ||||
|   @property() public urlPath: string | null = null; | ||||
|  | ||||
|   @query("hui-view") private _huiView?: HTMLElement; | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     const index = this._viewIndex; | ||||
|     if (index === undefined) { | ||||
| @@ -78,12 +80,12 @@ class HcLovelace extends LitElement { | ||||
|           this.lovelaceConfig.background; | ||||
|  | ||||
|         if (configBackground) { | ||||
|           (this.shadowRoot!.querySelector( | ||||
|             "hui-view" | ||||
|           ) as HTMLElement)!.style.setProperty( | ||||
|           this._huiView!.style.setProperty( | ||||
|             "--lovelace-background", | ||||
|             configBackground | ||||
|           ); | ||||
|         } else { | ||||
|           this._huiView!.style.removeProperty("--lovelace-background"); | ||||
|         } | ||||
|       } | ||||
|     } | ||||
| @@ -116,6 +118,9 @@ class HcLovelace extends LitElement { | ||||
|       :host > * { | ||||
|         flex: 1; | ||||
|       } | ||||
|       hui-view { | ||||
|         background: var(--lovelace-background, var(--primary-background-color)); | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1,4 +1,3 @@ | ||||
| import "web-animations-js/web-animations-next-lite.min"; | ||||
| import "../../../src/resources/ha-style"; | ||||
| import "../../../src/resources/roboto"; | ||||
| import "./layout/hc-lovelace"; | ||||
|   | ||||
| @@ -2,8 +2,3 @@ import "../../src/resources/ha-style"; | ||||
| import "../../src/resources/roboto"; | ||||
| import "../../src/resources/safari-14-attachshadow-patch"; | ||||
| import "./ha-demo"; | ||||
|  | ||||
| /* polyfill for paper-dropdown */ | ||||
| setTimeout(() => { | ||||
|   import("web-animations-js/web-animations-next-lite.min"); | ||||
| }, 1000); | ||||
|   | ||||
							
								
								
									
										
											BIN
										
									
								
								gallery/public/images/clearspace.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								gallery/public/images/clearspace.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 44 KiB | 
							
								
								
									
										
											BIN
										
									
								
								gallery/public/images/logo-variants.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								gallery/public/images/logo-variants.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 35 KiB | 
							
								
								
									
										
											BIN
										
									
								
								gallery/public/images/logo-with-text.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								gallery/public/images/logo-with-text.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 67 KiB | 
							
								
								
									
										
											BIN
										
									
								
								gallery/public/images/logo.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								gallery/public/images/logo.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 27 KiB | 
							
								
								
									
										
											BIN
										
									
								
								gallery/public/images/using-our-logo.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								gallery/public/images/using-our-logo.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 32 KiB | 
| @@ -23,7 +23,7 @@ if [[ "${PULL_REQUEST}" == "true" ]]; then | ||||
|     createStatus "pending" "Building design preview" "https://app.netlify.com/sites/home-assistant-gallery/deploys/$BUILD_ID" | ||||
|     gulp build-gallery | ||||
|     if [ $? -eq 0 ]; then | ||||
|       createStatus "success" "Build complete" "$DEPLOY_URL" | ||||
|       createStatus "success" "Build complete" "$DEPLOY_PRIME_URL" | ||||
|     else | ||||
|       createStatus "error" "Build failed" "https://app.netlify.com/sites/home-assistant-gallery/deploys/$BUILD_ID" | ||||
|     fi | ||||
|   | ||||
| @@ -20,7 +20,6 @@ module.exports = [ | ||||
|       "editor-trigger", | ||||
|       "editor-condition", | ||||
|       "editor-action", | ||||
|       "selectors", | ||||
|       "trace", | ||||
|       "trace-timeline", | ||||
|     ], | ||||
| @@ -37,12 +36,17 @@ module.exports = [ | ||||
|     category: "misc", | ||||
|     header: "Miscelaneous", | ||||
|   }, | ||||
|   { | ||||
|     category: "brand", | ||||
|     header: "Brand", | ||||
|   }, | ||||
|   { | ||||
|     category: "user-test", | ||||
|     header: "User Tests", | ||||
|     header: "Users", | ||||
|     pages: ["user-types", "configuration-menu"], | ||||
|   }, | ||||
|   { | ||||
|     category: "design.home-assistant.io", | ||||
|     header: "Design Documentation", | ||||
|     header: "About", | ||||
|   }, | ||||
| ]; | ||||
|   | ||||
| @@ -78,6 +78,9 @@ class DemoCards extends LitElement { | ||||
|     ha-formfield { | ||||
|       margin-right: 16px; | ||||
|     } | ||||
|     #container { | ||||
|       background-color: var(--primary-background-color); | ||||
|     } | ||||
|   `; | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -12,7 +12,14 @@ class PageDescription extends HaMarkdown { | ||||
|     if (!PAGES[this.page].description) { | ||||
|       return html``; | ||||
|     } | ||||
|  | ||||
|     return html` | ||||
|       <div class="heading"> | ||||
|         <div class="title"> | ||||
|           ${PAGES[this.page].metadata.title || this.page.split("/")[1]} | ||||
|         </div> | ||||
|         <div class="subtitle">${PAGES[this.page].metadata.subtitle}</div> | ||||
|       </div> | ||||
|       ${until( | ||||
|         PAGES[this.page] | ||||
|           .description() | ||||
| @@ -25,9 +32,22 @@ class PageDescription extends HaMarkdown { | ||||
|   static styles = [ | ||||
|     HaMarkdown.styles, | ||||
|     css` | ||||
|       .heading { | ||||
|         padding: 16px; | ||||
|         border-bottom: 1px solid var(--secondary-background-color); | ||||
|       } | ||||
|       .title { | ||||
|         font-size: 42px; | ||||
|         line-height: 56px; | ||||
|         padding-bottom: 8px; | ||||
|       } | ||||
|       .subtitle { | ||||
|         font-size: 18px; | ||||
|         line-height: 24px; | ||||
|       } | ||||
|       .root { | ||||
|         max-width: 800px; | ||||
|         margin: 0 auto; | ||||
|         margin: 16px auto; | ||||
|       } | ||||
|       .root > *:first-child { | ||||
|         margin-top: 0; | ||||
|   | ||||
| @@ -5,6 +5,7 @@ import { html, css, LitElement, PropertyValues } from "lit"; | ||||
| import { customElement, property, query } from "lit/decorators"; | ||||
| import "../../src/components/ha-icon-button"; | ||||
| import "../../src/managers/notification-manager"; | ||||
| import "../../src/components/ha-expansion-panel"; | ||||
| import { haStyle } from "../../src/resources/styles"; | ||||
| import { PAGES, SIDEBAR } from "../build/import-pages"; | ||||
| import { dynamicElement } from "../../src/common/dom/dynamic-element-directive"; | ||||
| @@ -44,6 +45,10 @@ class HaGallery extends LitElement { | ||||
|       for (const page of group.pages!) { | ||||
|         const key = `${group.category}/${page}`; | ||||
|         const active = this._page === key; | ||||
|         if (!(key in PAGES)) { | ||||
|           console.error("Undefined page referenced in sidebar.js:", key); | ||||
|           continue; | ||||
|         } | ||||
|         const title = PAGES[key].metadata.title || page; | ||||
|         links.push(html` | ||||
|           <a ?active=${active} href=${`#${group.category}/${page}`}>${title}</a> | ||||
| @@ -53,10 +58,9 @@ class HaGallery extends LitElement { | ||||
|       sidebar.push( | ||||
|         group.header | ||||
|           ? html` | ||||
|               <details> | ||||
|                 <summary class="section">${group.header}</summary> | ||||
|               <ha-expansion-panel .header=${group.header}> | ||||
|                 ${links} | ||||
|               </details> | ||||
|               </ha-expansion-panel> | ||||
|             ` | ||||
|           : links | ||||
|       ); | ||||
| @@ -92,27 +96,34 @@ class HaGallery extends LitElement { | ||||
|             ${dynamicElement(`demo-${this._page.replace("/", "-")}`)} | ||||
|           </div> | ||||
|           <div class="page-footer"> | ||||
|             ${PAGES[this._page].description || | ||||
|             Object.keys(PAGES[this._page].metadata).length > 0 | ||||
|               ? html` | ||||
|                   <a | ||||
|                     href=${`${GITHUB_DEMO_URL}${this._page}.markdown`} | ||||
|                     target="_blank" | ||||
|                   > | ||||
|                     Edit text | ||||
|                   </a> | ||||
|                 ` | ||||
|               : ""} | ||||
|             ${PAGES[this._page].demo | ||||
|               ? html` | ||||
|                   <a | ||||
|                     href=${`${GITHUB_DEMO_URL}${this._page}.ts`} | ||||
|                     target="_blank" | ||||
|                   > | ||||
|                     Edit demo | ||||
|                   </a> | ||||
|                 ` | ||||
|               : ""} | ||||
|             <div class="header">Help us to improve our documentation</div> | ||||
|             <div class="secondary"> | ||||
|               Suggest an edit to this page, or provide/view feedback for this | ||||
|               page. | ||||
|             </div> | ||||
|             <div> | ||||
|               ${PAGES[this._page].description || | ||||
|               Object.keys(PAGES[this._page].metadata).length > 0 | ||||
|                 ? html` | ||||
|                     <a | ||||
|                       href=${`${GITHUB_DEMO_URL}${this._page}.markdown`} | ||||
|                       target="_blank" | ||||
|                     > | ||||
|                       Edit text | ||||
|                     </a> | ||||
|                   ` | ||||
|                 : ""} | ||||
|               ${PAGES[this._page].demo | ||||
|                 ? html` | ||||
|                     <a | ||||
|                       href=${`${GITHUB_DEMO_URL}${this._page}.ts`} | ||||
|                       target="_blank" | ||||
|                     > | ||||
|                       Edit demo | ||||
|                     </a> | ||||
|                   ` | ||||
|                 : ""} | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|       </mwc-drawer> | ||||
| @@ -186,27 +197,16 @@ class HaGallery extends LitElement { | ||||
|         padding: 4px; | ||||
|       } | ||||
|  | ||||
|       .sidebar details { | ||||
|         margin-top: 1em; | ||||
|         margin-left: 1em; | ||||
|       } | ||||
|  | ||||
|       .sidebar summary { | ||||
|         cursor: pointer; | ||||
|         font-weight: bold; | ||||
|         margin-bottom: 8px; | ||||
|       } | ||||
|  | ||||
|       .sidebar a { | ||||
|         color: var(--primary-text-color); | ||||
|         display: block; | ||||
|         padding: 4px 12px; | ||||
|         padding: 12px; | ||||
|         text-decoration: none; | ||||
|         position: relative; | ||||
|       } | ||||
|  | ||||
|       .sidebar a[active]::before { | ||||
|         border-radius: 4px; | ||||
|         border-radius: 12px; | ||||
|         position: absolute; | ||||
|         top: 0; | ||||
|         right: 2px; | ||||
| @@ -237,14 +237,32 @@ class HaGallery extends LitElement { | ||||
|  | ||||
|       .page-footer { | ||||
|         text-align: center; | ||||
|         margin: 16px 0; | ||||
|         padding-top: 16px; | ||||
|         border-top: 1px solid rgba(0, 0, 0, 0.12); | ||||
|         margin: 16px; | ||||
|         padding: 16px; | ||||
|         border-radius: 12px; | ||||
|         background-color: var(--primary-background-color); | ||||
|       } | ||||
|  | ||||
|       .page-footer div { | ||||
|         margin-top: 4px; | ||||
|       } | ||||
|  | ||||
|       .page-footer .header { | ||||
|         font-size: 16px; | ||||
|         font-weight: 500; | ||||
|         line-height: 28px; | ||||
|         text-align: center; | ||||
|       } | ||||
|  | ||||
|       .page-footer .secondary { | ||||
|         line-height: 23px; | ||||
|         text-align: center; | ||||
|       } | ||||
|  | ||||
|       .page-footer a { | ||||
|         display: inline-block; | ||||
|         margin: 0 8px; | ||||
|         text-decoration: none; | ||||
|       } | ||||
|     `, | ||||
|   ]; | ||||
|   | ||||
| @@ -3,10 +3,20 @@ import { html, css, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import { describeAction } from "../../../../src/data/script_i18n"; | ||||
| import { getEntity } from "../../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import { HomeAssistant } from "../../../../src/types"; | ||||
|  | ||||
| const actions = [ | ||||
| const ENTITIES = [ | ||||
|   getEntity("scene", "kitchen_morning", "scening", { | ||||
|     friendly_name: "Kitchen Morning", | ||||
|   }), | ||||
|   getEntity("media_player", "kitchen", "playing", { | ||||
|     friendly_name: "Sonos Kitchen", | ||||
|   }), | ||||
| ]; | ||||
|  | ||||
| const ACTIONS = [ | ||||
|   { wait_template: "{{ true }}", alias: "Something with an alias" }, | ||||
|   { delay: "0:05" }, | ||||
|   { wait_template: "{{ true }}" }, | ||||
| @@ -19,8 +29,20 @@ const actions = [ | ||||
|     device_id: "abcdefgh", | ||||
|     domain: "plex", | ||||
|     entity_id: "media_player.kitchen", | ||||
|     type: "turn_on", | ||||
|   }, | ||||
|   { scene: "scene.kitchen_morning" }, | ||||
|   { | ||||
|     service: "scene.turn_on", | ||||
|     target: { entity_id: "scene.kitchen_morning" }, | ||||
|     metadata: {}, | ||||
|   }, | ||||
|   { | ||||
|     service: "media_player.play_media", | ||||
|     target: { entity_id: "media_player.kitchen" }, | ||||
|     data: { media_content_id: "", media_content_type: "" }, | ||||
|     metadata: { title: "Happy Song" }, | ||||
|   }, | ||||
|   { | ||||
|     wait_for_trigger: [ | ||||
|       { | ||||
| @@ -52,7 +74,7 @@ export class DemoAutomationDescribeAction extends LitElement { | ||||
|     } | ||||
|     return html` | ||||
|       <ha-card header="Actions"> | ||||
|         ${actions.map( | ||||
|         ${ACTIONS.map( | ||||
|           (conf) => html` | ||||
|             <div class="action"> | ||||
|               <span>${describeAction(this.hass, conf as any)}</span> | ||||
| @@ -68,6 +90,7 @@ export class DemoAutomationDescribeAction extends LitElement { | ||||
|     super.firstUpdated(changedProps); | ||||
|     const hass = provideHass(this); | ||||
|     hass.updateTranslations(null, "en"); | ||||
|     hass.addEntities(ENTITIES); | ||||
|   } | ||||
|  | ||||
|   static get styles() { | ||||
|   | ||||
| @@ -14,7 +14,7 @@ import { HaDelayAction } from "../../../../src/panels/config/automation/action/t | ||||
| import { HaDeviceAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-device_id"; | ||||
| import { HaEventAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-event"; | ||||
| import { HaRepeatAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-repeat"; | ||||
| import { HaSceneAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-scene"; | ||||
| import { HaSceneAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-activate_scene"; | ||||
| import { HaServiceAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-service"; | ||||
| import { HaWaitForTriggerAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-wait_for_trigger"; | ||||
| import { HaWaitAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-wait_template"; | ||||
|   | ||||
| @@ -1,3 +0,0 @@ | ||||
| --- | ||||
| title: Selectors | ||||
| --- | ||||
| @@ -1,102 +0,0 @@ | ||||
| /* eslint-disable lit/no-template-arrow */ | ||||
| import { LitElement, TemplateResult, html } from "lit"; | ||||
| import { customElement, state } from "lit/decorators"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import type { HomeAssistant } from "../../../../src/types"; | ||||
| import "../../components/demo-black-white-row"; | ||||
| import { mockEntityRegistry } from "../../../../demo/src/stubs/entity_registry"; | ||||
| import { mockDeviceRegistry } from "../../../../demo/src/stubs/device_registry"; | ||||
| import { mockAreaRegistry } from "../../../../demo/src/stubs/area_registry"; | ||||
| import { mockHassioSupervisor } from "../../../../demo/src/stubs/hassio_supervisor"; | ||||
| import "../../../../src/panels/config/automation/trigger/ha-automation-trigger"; | ||||
| import { Selector } from "../../../../src/data/selector"; | ||||
| import "../../../../src/components/ha-selector/ha-selector"; | ||||
|  | ||||
| const SCHEMAS: { name: string; selector: Selector }[] = [ | ||||
|   { name: "Addon", selector: { addon: {} } }, | ||||
|  | ||||
|   { name: "Entity", selector: { entity: {} } }, | ||||
|   { name: "Device", selector: { device: {} } }, | ||||
|   { name: "Area", selector: { area: {} } }, | ||||
|   { name: "Target", selector: { target: {} } }, | ||||
|   { | ||||
|     name: "Number", | ||||
|     selector: { | ||||
|       number: { | ||||
|         min: 0, | ||||
|         max: 10, | ||||
|       }, | ||||
|     }, | ||||
|   }, | ||||
|   { name: "Boolean", selector: { boolean: {} } }, | ||||
|   { name: "Time", selector: { time: {} } }, | ||||
|   { name: "Action", selector: { action: {} } }, | ||||
|   { name: "Text", selector: { text: { multiline: false } } }, | ||||
|   { name: "Text Multiline", selector: { text: { multiline: true } } }, | ||||
|   { name: "Object", selector: { object: {} } }, | ||||
|   { | ||||
|     name: "Select", | ||||
|     selector: { | ||||
|       select: { | ||||
|         options: ["Everyone Home", "Some Home", "All gone"], | ||||
|       }, | ||||
|     }, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| @customElement("demo-automation-selectors") | ||||
| class DemoHaSelector extends LitElement { | ||||
|   @state() private hass!: HomeAssistant; | ||||
|  | ||||
|   private data: any = SCHEMAS.map(() => undefined); | ||||
|  | ||||
|   constructor() { | ||||
|     super(); | ||||
|     const hass = provideHass(this); | ||||
|     hass.updateTranslations(null, "en"); | ||||
|     hass.updateTranslations("config", "en"); | ||||
|     mockEntityRegistry(hass); | ||||
|     mockDeviceRegistry(hass); | ||||
|     mockAreaRegistry(hass); | ||||
|     mockHassioSupervisor(hass); | ||||
|   } | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     const valueChanged = (ev) => { | ||||
|       const sampleIdx = ev.target.sampleIdx; | ||||
|       this.data[sampleIdx] = ev.detail.value; | ||||
|       this.requestUpdate(); | ||||
|     }; | ||||
|     return html` | ||||
|       ${SCHEMAS.map( | ||||
|         (info, sampleIdx) => html` | ||||
|           <demo-black-white-row | ||||
|             .title=${info.name} | ||||
|             .value=${{ selector: info.selector, data: this.data[sampleIdx] }} | ||||
|           > | ||||
|             ${["light", "dark"].map( | ||||
|               (slot) => | ||||
|                 html` | ||||
|                   <ha-selector | ||||
|                     slot=${slot} | ||||
|                     .hass=${this.hass} | ||||
|                     .selector=${info.selector} | ||||
|                     .label=${info.name} | ||||
|                     .value=${this.data[sampleIdx]} | ||||
|                     .sampleIdx=${sampleIdx} | ||||
|                     @value-changed=${valueChanged} | ||||
|                   ></ha-selector> | ||||
|                 ` | ||||
|             )} | ||||
|           </demo-black-white-row> | ||||
|         ` | ||||
|       )} | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-automation-selectors": DemoHaSelector; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										34
									
								
								gallery/src/pages/brand/logo.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								gallery/src/pages/brand/logo.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | ||||
| --- | ||||
| title: "Logo" | ||||
| --- | ||||
|  | ||||
|  | ||||
|  | ||||
| # Using our logo | ||||
|  | ||||
| As a community, we are proud of our logo. Follow these guidelines to ensure it always looks its best. Our logo follows Google's material design spec and uses the blue interface color. | ||||
|  | ||||
| [Download Logo](https://github.com/home-assistant/assets/tree/master/logo) | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| ## Using the icon | ||||
|  | ||||
| Our icon is a shorter and most used version of our logo. The icon can exist without the wordmark, the wordmark should never exist without the icon. | ||||
|  | ||||
|  | ||||
|  | ||||
| ## Using the right variant | ||||
|  | ||||
| The pretty blue logo with a background shadow, pictured top left, is our primary logo. It should only be used with black, white, and non-duotone photography. | ||||
|  | ||||
| When needed you can use our logo without a shadow, as seen as the second variant.  | ||||
|  | ||||
| The outlined logo should only be used on packaging. | ||||
|  | ||||
| ## Exclusion zone | ||||
|  | ||||
| The logo needs some personal space. It's exclusion zone is equal to a quarter the height of the icon. | ||||
|  | ||||
|  | ||||
							
								
								
									
										41
									
								
								gallery/src/pages/brand/our-story.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								gallery/src/pages/brand/our-story.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,41 @@ | ||||
| --- | ||||
| title: "Our story" | ||||
| --- | ||||
|  | ||||
| ## Open source home automation that puts local control and privacy first | ||||
|  | ||||
| Home Assistant is a free and open-source software for home automation that is designed to be the central control system for smart home devices with a focus on local control and privacy. It can be accessed via a web-based user interface, via apps for Android and iOS, or using voice commands via a supported virtual assistant like Google Assistant and Amazon Alexa. | ||||
|  | ||||
| IoT devices and services are supported by modular support for controlling proprietary ecosystems if they provide public access via an Open API for third-party integrations and protocols like Bluetooth, MQTT, Zigbee, and Z-Wave, After the Home Assistant software application is installed as a computer appliance it will act as a central control system for home automation. Information from all entities it sees can be used and controlled from within scripts trigger automations using scheduling and "blueprint" subroutines, e.g. for controlling lighting, climate, entertainment systems, and appliances. | ||||
|  | ||||
| # Open Home | ||||
|  | ||||
| The Open Home is our vision for the smart home. It defines the values that we put at the heart of every decision we make at Home Assistant. It’s woven into our architecture, licensing, community, and everything else. | ||||
|  | ||||
| The Open Home is about privacy, choice, and durability. | ||||
|  | ||||
| ## Privacy | ||||
|  | ||||
| Your home should be your safe space. A place where you can be your true self without having to bother about what the world thinks of you. A place where you don’t need to act differently to avoid an algorithm categorizing your behavior. Privacy for the Open Home means that devices need to work locally. No one else needs to know if you turn on a light bulb or change the thermostat. | ||||
|  | ||||
| It is okay for a product to offer a cloud connection, but it should be extra and opt-in. | ||||
|  | ||||
| ## Choice | ||||
|  | ||||
| Devices in your home gather data about themselves and their surroundings. Your data. Vendors shouldn’t be able to limit your access to your data or limit the interoperability of your devices with the rest of your smart home. | ||||
|  | ||||
| Choice for the Open Home means that devices need to make the gathered data available through local APIs. This avoids vendor lock-in and allows users to create their own smart home with devices from different manufacturers. | ||||
|  | ||||
| ## Durability | ||||
|  | ||||
| If there is one thing that technology firms are very good at, it is launching new products. However, maintaining the products and making sure they keep working is an afterthought for most. The result is that vendors can decide to no longer support your device, crippling its features or even preventing it from working at all. As we install more and more devices in our home, durability is becoming more and more important. We shouldn’t have to buy everything new every couple of years because the manufacturer decided to move on. | ||||
|  | ||||
| Durability for the Open Home means that devices are designed and built to keep working. Not just this year, but for the next decade. | ||||
|  | ||||
| # Our history | ||||
|  | ||||
| The project was started as a Python application by Paulus Schoutsen in September 2013 and first published publicly on GitHub in November 2013. In July 2017, a managed operating system called Hass.io was initially introduced to make it easier use to use Home Assistant on single-board computers like the Raspberry Pi series. Its bundled "supervisor" management system allowed users to manage, backup, and update the local installation and introduced the option to extend the functionality of the software with add-ons. | ||||
|  | ||||
| An optional subscription service was introduced in December 2017 for $5/month to solve the complexities associated with secured remote access, as well as linking to Amazon Alexa and Google Assistant. Nabu Casa, Inc. was formed in September 2018 to take over the subscription service. The company's funding is based solely on revenue from the subscription service. It is used to finance the project's infrastructure and to pay for full-time employees contributing to the project. | ||||
|  | ||||
| In January 2020, branding was adjusted to make it easier to refer to different parts of the project. The main piece of software was renamed to Home Assistant Core, while the full suite of software with the embedded operating system and bundled "supervisor" management system was renamed to Home Assistant. | ||||
| @@ -1,5 +1,6 @@ | ||||
| --- | ||||
| title: Alerts | ||||
| subtitle: An alert displays a short, important message in a way that attracts the user's attention without interrupting the user's task. | ||||
| --- | ||||
|  | ||||
| # Alert `<ha-alert>` | ||||
|   | ||||
| @@ -12,6 +12,98 @@ import { mockEntityRegistry } from "../../../../demo/src/stubs/entity_registry"; | ||||
| import { mockHassioSupervisor } from "../../../../demo/src/stubs/hassio_supervisor"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import { HomeAssistant } from "../../../../src/types"; | ||||
| import { getEntity } from "../../../../src/fake_data/entity"; | ||||
|  | ||||
| const ENTITIES = [ | ||||
|   getEntity("alarm_control_panel", "alarm", "disarmed", { | ||||
|     friendly_name: "Alarm", | ||||
|   }), | ||||
|   getEntity("media_player", "livingroom", "playing", { | ||||
|     friendly_name: "Livingroom", | ||||
|   }), | ||||
|   getEntity("media_player", "lounge", "idle", { | ||||
|     friendly_name: "Lounge", | ||||
|     supported_features: 444983, | ||||
|   }), | ||||
|   getEntity("light", "bedroom", "on", { | ||||
|     friendly_name: "Bedroom", | ||||
|   }), | ||||
|   getEntity("switch", "coffee", "off", { | ||||
|     friendly_name: "Coffee", | ||||
|   }), | ||||
| ]; | ||||
|  | ||||
| const DEVICES = [ | ||||
|   { | ||||
|     area_id: "bedroom", | ||||
|     configuration_url: null, | ||||
|     config_entries: ["config_entry_1"], | ||||
|     connections: [], | ||||
|     disabled_by: null, | ||||
|     entry_type: null, | ||||
|     id: "device_1", | ||||
|     identifiers: [["demo", "volume1"] as [string, string]], | ||||
|     manufacturer: null, | ||||
|     model: null, | ||||
|     name_by_user: null, | ||||
|     name: "Dishwasher", | ||||
|     sw_version: null, | ||||
|     hw_version: null, | ||||
|     via_device_id: null, | ||||
|   }, | ||||
|   { | ||||
|     area_id: "backyard", | ||||
|     configuration_url: null, | ||||
|     config_entries: ["config_entry_2"], | ||||
|     connections: [], | ||||
|     disabled_by: null, | ||||
|     entry_type: null, | ||||
|     id: "device_2", | ||||
|     identifiers: [["demo", "pwm1"] as [string, string]], | ||||
|     manufacturer: null, | ||||
|     model: null, | ||||
|     name_by_user: null, | ||||
|     name: "Lamp", | ||||
|     sw_version: null, | ||||
|     hw_version: null, | ||||
|     via_device_id: null, | ||||
|   }, | ||||
|   { | ||||
|     area_id: null, | ||||
|     configuration_url: null, | ||||
|     config_entries: ["config_entry_3"], | ||||
|     connections: [], | ||||
|     disabled_by: null, | ||||
|     entry_type: null, | ||||
|     id: "device_3", | ||||
|     identifiers: [["demo", "pwm1"] as [string, string]], | ||||
|     manufacturer: null, | ||||
|     model: null, | ||||
|     name_by_user: "User name", | ||||
|     name: "Technical name", | ||||
|     sw_version: null, | ||||
|     hw_version: null, | ||||
|     via_device_id: null, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| const AREAS = [ | ||||
|   { | ||||
|     area_id: "backyard", | ||||
|     name: "Backyard", | ||||
|     picture: null, | ||||
|   }, | ||||
|   { | ||||
|     area_id: "bedroom", | ||||
|     name: "Bedroom", | ||||
|     picture: null, | ||||
|   }, | ||||
|   { | ||||
|     area_id: "livingroom", | ||||
|     name: "Livingroom", | ||||
|     picture: null, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| const SCHEMAS: { | ||||
|   title: string; | ||||
| @@ -36,6 +128,10 @@ const SCHEMAS: { | ||||
|       text_multiline: "Text Multiline", | ||||
|       object: "Object", | ||||
|       select: "Select", | ||||
|       icon: "Icon", | ||||
|       media: "Media", | ||||
|       location: "Location", | ||||
|       entities: "Entities", | ||||
|     }, | ||||
|     schema: [ | ||||
|       { name: "addon", selector: { addon: {} } }, | ||||
| @@ -43,6 +139,7 @@ const SCHEMAS: { | ||||
|       { | ||||
|         name: "Attribute", | ||||
|         selector: { attribute: { entity_id: "" } }, | ||||
|         context: { filter_entity: "entity" }, | ||||
|       }, | ||||
|       { name: "Device", selector: { device: {} } }, | ||||
|       { name: "Duration", selector: { duration: {} } }, | ||||
| @@ -61,6 +158,26 @@ const SCHEMAS: { | ||||
|           select: { options: ["Everyone Home", "Some Home", "All gone"] }, | ||||
|         }, | ||||
|       }, | ||||
|       { | ||||
|         name: "icon", | ||||
|         selector: { | ||||
|           icon: {}, | ||||
|         }, | ||||
|       }, | ||||
|       { | ||||
|         name: "media", | ||||
|         selector: { | ||||
|           media: {}, | ||||
|         }, | ||||
|       }, | ||||
|       { | ||||
|         name: "location", | ||||
|         selector: { location: { radius: true, icon: "mdi:home" } }, | ||||
|       }, | ||||
|       { | ||||
|         name: "entities", | ||||
|         selector: { entity: { multiple: true } }, | ||||
|       }, | ||||
|     ], | ||||
|   }, | ||||
|   { | ||||
| @@ -301,9 +418,10 @@ class DemoHaForm extends LitElement { | ||||
|     const hass = provideHass(this); | ||||
|     hass.updateTranslations(null, "en"); | ||||
|     hass.updateTranslations("config", "en"); | ||||
|     hass.addEntities(ENTITIES); | ||||
|     mockEntityRegistry(hass); | ||||
|     mockDeviceRegistry(hass); | ||||
|     mockAreaRegistry(hass); | ||||
|     mockDeviceRegistry(hass, DEVICES); | ||||
|     mockAreaRegistry(hass, AREAS); | ||||
|     mockHassioSupervisor(hass); | ||||
|   } | ||||
|  | ||||
|   | ||||
| @@ -1,3 +1,5 @@ | ||||
| --- | ||||
| title: Target Selectors | ||||
| title: Selectors | ||||
| --- | ||||
|  | ||||
| See the website for [list of available selectors](https://www.home-assistant.io/docs/blueprint/selectors/). | ||||
|   | ||||
| @@ -12,6 +12,100 @@ import { mockEntityRegistry } from "../../../../demo/src/stubs/entity_registry"; | ||||
| import { mockDeviceRegistry } from "../../../../demo/src/stubs/device_registry"; | ||||
| import { mockAreaRegistry } from "../../../../demo/src/stubs/area_registry"; | ||||
| import { mockHassioSupervisor } from "../../../../demo/src/stubs/hassio_supervisor"; | ||||
| import { getEntity } from "../../../../src/fake_data/entity"; | ||||
| import { ProvideHassElement } from "../../../../src/mixins/provide-hass-lit-mixin"; | ||||
| import { showDialog } from "../../../../src/dialogs/make-dialog-manager"; | ||||
|  | ||||
| const ENTITIES = [ | ||||
|   getEntity("alarm_control_panel", "alarm", "disarmed", { | ||||
|     friendly_name: "Alarm", | ||||
|   }), | ||||
|   getEntity("media_player", "livingroom", "playing", { | ||||
|     friendly_name: "Livingroom", | ||||
|   }), | ||||
|   getEntity("media_player", "lounge", "idle", { | ||||
|     friendly_name: "Lounge", | ||||
|     supported_features: 444983, | ||||
|   }), | ||||
|   getEntity("light", "bedroom", "on", { | ||||
|     friendly_name: "Bedroom", | ||||
|   }), | ||||
|   getEntity("switch", "coffee", "off", { | ||||
|     friendly_name: "Coffee", | ||||
|   }), | ||||
| ]; | ||||
|  | ||||
| const DEVICES = [ | ||||
|   { | ||||
|     area_id: "bedroom", | ||||
|     configuration_url: null, | ||||
|     config_entries: ["config_entry_1"], | ||||
|     connections: [], | ||||
|     disabled_by: null, | ||||
|     entry_type: null, | ||||
|     id: "device_1", | ||||
|     identifiers: [["demo", "volume1"] as [string, string]], | ||||
|     manufacturer: null, | ||||
|     model: null, | ||||
|     name_by_user: null, | ||||
|     name: "Dishwasher", | ||||
|     sw_version: null, | ||||
|     hw_version: null, | ||||
|     via_device_id: null, | ||||
|   }, | ||||
|   { | ||||
|     area_id: "backyard", | ||||
|     configuration_url: null, | ||||
|     config_entries: ["config_entry_2"], | ||||
|     connections: [], | ||||
|     disabled_by: null, | ||||
|     entry_type: null, | ||||
|     id: "device_2", | ||||
|     identifiers: [["demo", "pwm1"] as [string, string]], | ||||
|     manufacturer: null, | ||||
|     model: null, | ||||
|     name_by_user: null, | ||||
|     name: "Lamp", | ||||
|     sw_version: null, | ||||
|     hw_version: null, | ||||
|     via_device_id: null, | ||||
|   }, | ||||
|   { | ||||
|     area_id: null, | ||||
|     configuration_url: null, | ||||
|     config_entries: ["config_entry_3"], | ||||
|     connections: [], | ||||
|     disabled_by: null, | ||||
|     entry_type: null, | ||||
|     id: "device_3", | ||||
|     identifiers: [["demo", "pwm1"] as [string, string]], | ||||
|     manufacturer: null, | ||||
|     model: null, | ||||
|     name_by_user: "User name", | ||||
|     name: "Technical name", | ||||
|     sw_version: null, | ||||
|     hw_version: null, | ||||
|     via_device_id: null, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| const AREAS = [ | ||||
|   { | ||||
|     area_id: "backyard", | ||||
|     name: "Backyard", | ||||
|     picture: null, | ||||
|   }, | ||||
|   { | ||||
|     area_id: "bedroom", | ||||
|     name: "Bedroom", | ||||
|     picture: null, | ||||
|   }, | ||||
|   { | ||||
|     area_id: "livingroom", | ||||
|     name: "Livingroom", | ||||
|     picture: null, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| const SCHEMAS: { | ||||
|   name: string; | ||||
| @@ -52,6 +146,8 @@ const SCHEMAS: { | ||||
|       }, | ||||
|       boolean: { name: "Boolean", selector: { boolean: {} } }, | ||||
|       time: { name: "Time", selector: { time: {} } }, | ||||
|       date: { name: "Date", selector: { date: {} } }, | ||||
|       datetime: { name: "Date Time", selector: { datetime: {} } }, | ||||
|       action: { name: "Action", selector: { action: {} } }, | ||||
|       text: { | ||||
|         name: "Text", | ||||
| @@ -68,17 +164,51 @@ const SCHEMAS: { | ||||
|         }, | ||||
|       }, | ||||
|       object: { name: "Object", selector: { object: {} } }, | ||||
|       select: { | ||||
|         name: "Select", | ||||
|       select_radio: { | ||||
|         name: "Select (Radio)", | ||||
|         selector: { select: { options: ["Option 1", "Option 2"] } }, | ||||
|       }, | ||||
|       select: { | ||||
|         name: "Select", | ||||
|         selector: { | ||||
|           select: { | ||||
|             options: [ | ||||
|               "Option 1", | ||||
|               "Option 2", | ||||
|               "Option 3", | ||||
|               "Option 4", | ||||
|               "Option 5", | ||||
|               "Option 6", | ||||
|             ], | ||||
|           }, | ||||
|         }, | ||||
|       }, | ||||
|       icon: { name: "Icon", selector: { icon: {} } }, | ||||
|       media: { name: "Media", selector: { media: {} } }, | ||||
|       location: { name: "Location", selector: { location: {} } }, | ||||
|       location_radius: { | ||||
|         name: "Location with radius", | ||||
|         selector: { location: { radius: true, icon: "mdi:home" } }, | ||||
|       }, | ||||
|       color_temp: { | ||||
|         name: "Color Temperature", | ||||
|         selector: { color_temp: {} }, | ||||
|       }, | ||||
|       color_rgb: { name: "Color", selector: { color_rgb: {} } }, | ||||
|     }, | ||||
|   }, | ||||
|   { | ||||
|     name: "Multiples", | ||||
|     input: { | ||||
|       entity: { name: "Entity", selector: { entity: { multiple: true } } }, | ||||
|       device: { name: "Device", selector: { device: { multiple: true } } }, | ||||
|     }, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| @customElement("demo-components-ha-selector") | ||||
| class DemoHaSelector extends LitElement { | ||||
|   @state() private hass!: HomeAssistant; | ||||
| class DemoHaSelector extends LitElement implements ProvideHassElement { | ||||
|   @state() public hass!: HomeAssistant; | ||||
|  | ||||
|   private data = SCHEMAS.map(() => ({})); | ||||
|  | ||||
| @@ -87,12 +217,130 @@ class DemoHaSelector extends LitElement { | ||||
|     const hass = provideHass(this); | ||||
|     hass.updateTranslations(null, "en"); | ||||
|     hass.updateTranslations("config", "en"); | ||||
|     hass.addEntities(ENTITIES); | ||||
|     mockEntityRegistry(hass); | ||||
|     mockDeviceRegistry(hass); | ||||
|     mockAreaRegistry(hass); | ||||
|     mockDeviceRegistry(hass, DEVICES); | ||||
|     mockAreaRegistry(hass, AREAS); | ||||
|     mockHassioSupervisor(hass); | ||||
|     hass.mockWS("auth/sign_path", (params) => params); | ||||
|     hass.mockWS("media_player/browse_media", this._browseMedia); | ||||
|   } | ||||
|  | ||||
|   public provideHass(el) { | ||||
|     el.hass = this.hass; | ||||
|   } | ||||
|  | ||||
|   public connectedCallback() { | ||||
|     super.connectedCallback(); | ||||
|     this.addEventListener("show-dialog", this._dialogManager); | ||||
|   } | ||||
|  | ||||
|   public disconnectedCallback() { | ||||
|     super.disconnectedCallback(); | ||||
|     this.removeEventListener("show-dialog", this._dialogManager); | ||||
|   } | ||||
|  | ||||
|   private _browseMedia = ({ media_content_id }) => { | ||||
|     if (media_content_id === undefined) { | ||||
|       return { | ||||
|         title: "Media", | ||||
|         media_class: "directory", | ||||
|         media_content_type: "", | ||||
|         media_content_id: "media-source://media_source/local/.", | ||||
|         can_play: false, | ||||
|         can_expand: true, | ||||
|         children_media_class: "directory", | ||||
|         thumbnail: null, | ||||
|         children: [ | ||||
|           { | ||||
|             title: "Misc", | ||||
|             media_class: "directory", | ||||
|             media_content_type: "", | ||||
|             media_content_id: "media-source://media_source/local/misc", | ||||
|             can_play: false, | ||||
|             can_expand: true, | ||||
|             children_media_class: null, | ||||
|             thumbnail: null, | ||||
|           }, | ||||
|           { | ||||
|             title: "Movies", | ||||
|             media_class: "directory", | ||||
|             media_content_type: "", | ||||
|             media_content_id: "media-source://media_source/local/movies", | ||||
|             can_play: true, | ||||
|             can_expand: true, | ||||
|             children_media_class: "movie", | ||||
|             thumbnail: null, | ||||
|           }, | ||||
|           { | ||||
|             title: "Music", | ||||
|             media_class: "album", | ||||
|             media_content_type: "", | ||||
|             media_content_id: "media-source://media_source/local/music", | ||||
|             can_play: false, | ||||
|             can_expand: true, | ||||
|             children_media_class: "music", | ||||
|             thumbnail: "/images/album_cover_2.jpg", | ||||
|           }, | ||||
|         ], | ||||
|       }; | ||||
|     } | ||||
|     return { | ||||
|       title: "Subfolder", | ||||
|       media_class: "directory", | ||||
|       media_content_type: "", | ||||
|       media_content_id: "media-source://media_source/local/sub", | ||||
|       can_play: false, | ||||
|       can_expand: true, | ||||
|       children_media_class: "directory", | ||||
|       thumbnail: null, | ||||
|       children: [ | ||||
|         { | ||||
|           title: "audio.mp3", | ||||
|           media_class: "music", | ||||
|           media_content_type: "audio/mpeg", | ||||
|           media_content_id: "media-source://media_source/local/audio.mp3", | ||||
|           can_play: true, | ||||
|           can_expand: false, | ||||
|           children_media_class: null, | ||||
|           thumbnail: "/images/album_cover.jpg", | ||||
|         }, | ||||
|         { | ||||
|           title: "image.jpg", | ||||
|           media_class: "image", | ||||
|           media_content_type: "image/jpeg", | ||||
|           media_content_id: "media-source://media_source/local/image.jpg", | ||||
|           can_play: true, | ||||
|           can_expand: false, | ||||
|           children_media_class: null, | ||||
|           thumbnail: "https://brands.home-assistant.io/_/image/logo.png", | ||||
|         }, | ||||
|         { | ||||
|           title: "movie.mp4", | ||||
|           media_class: "movie", | ||||
|           media_content_type: "image/jpeg", | ||||
|           media_content_id: "media-source://media_source/local/movie.mp4", | ||||
|           can_play: true, | ||||
|           can_expand: false, | ||||
|           children_media_class: null, | ||||
|           thumbnail: null, | ||||
|         }, | ||||
|       ], | ||||
|     }; | ||||
|   }; | ||||
|  | ||||
|   private _dialogManager = (e) => { | ||||
|     const { dialogTag, dialogImport, dialogParams, addHistory } = e.detail; | ||||
|     showDialog( | ||||
|       this, | ||||
|       this.shadowRoot!, | ||||
|       dialogTag, | ||||
|       dialogParams, | ||||
|       dialogImport, | ||||
|       addHistory | ||||
|     ); | ||||
|   }; | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
|       ${SCHEMAS.map((info, idx) => { | ||||
| @@ -131,7 +379,6 @@ class DemoHaSelector extends LitElement { | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|     paper-input, | ||||
|     ha-selector { | ||||
|       width: 60; | ||||
|     } | ||||
|   | ||||
| @@ -2,6 +2,8 @@ | ||||
| title: Editing design.home-assistant.io | ||||
| --- | ||||
|  | ||||
|  | ||||
|  | ||||
| # How to edit design.home-assistant.io | ||||
|  | ||||
| All pages are stored in [the pages folder][pages-folder] on GitHub. Pages are grouped in a folder per sidebar section. Each page can contain a `<page name>.markdown` description file, a `<page name>.ts` demo file or both. If both are defined the description is rendered first. The description can contain metadata to specify the title of the page. | ||||
| @@ -41,15 +43,12 @@ import { html, css, LitElement } from "lit"; | ||||
| import { customElement } from "lit/decorators"; | ||||
| import "../../../../src/components/ha-card"; | ||||
|  | ||||
|  | ||||
| @customElement("demo-user-experience-usability") | ||||
| export class DemoUserExperienceUsability extends LitElement { | ||||
|   protected render() { | ||||
|     return html` | ||||
|       <ha-card> | ||||
|         <div class="card-content"> | ||||
|           Hello world! | ||||
|         </div> | ||||
|         <div class="card-content">Hello world!</div> | ||||
|       </ha-card> | ||||
|     `; | ||||
|   } | ||||
|   | ||||
| @@ -29,6 +29,7 @@ const createConfigEntry = ( | ||||
|   source: "zeroconf", | ||||
|   state: "loaded", | ||||
|   supports_options: false, | ||||
|   supports_remove_device: false, | ||||
|   supports_unload: true, | ||||
|   disabled_by: null, | ||||
|   pref_disable_new_entities: false, | ||||
| @@ -187,6 +188,7 @@ const createEntityRegistryEntries = ( | ||||
|     device_id: "mock-device-id", | ||||
|     area_id: null, | ||||
|     disabled_by: null, | ||||
|     hidden_by: null, | ||||
|     entity_category: null, | ||||
|     entity_id: "binary_sensor.updater", | ||||
|     name: null, | ||||
|   | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/more-info/update.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/more-info/update.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Update | ||||
| --- | ||||
							
								
								
									
										140
									
								
								gallery/src/pages/more-info/update.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										140
									
								
								gallery/src/pages/more-info/update.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,140 @@ | ||||
| import { html, LitElement, PropertyValues, TemplateResult } from "lit"; | ||||
| import { customElement, property, query } from "lit/decorators"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import { | ||||
|   UPDATE_SUPPORT_BACKUP, | ||||
|   UPDATE_SUPPORT_PROGRESS, | ||||
|   UPDATE_SUPPORT_INSTALL, | ||||
| } from "../../../../src/data/update"; | ||||
| import "../../../../src/dialogs/more-info/more-info-content"; | ||||
| import { getEntity } from "../../../../src/fake_data/entity"; | ||||
| import { | ||||
|   MockHomeAssistant, | ||||
|   provideHass, | ||||
| } from "../../../../src/fake_data/provide_hass"; | ||||
| import "../../components/demo-more-infos"; | ||||
|  | ||||
| const base_attributes = { | ||||
|   title: "Awesome", | ||||
|   current_version: "1.2.2", | ||||
|   latest_version: "1.2.3", | ||||
|   release_url: "https://home-assistant.io", | ||||
|   supported_features: UPDATE_SUPPORT_INSTALL, | ||||
|   skipped_version: null, | ||||
|   in_progress: false, | ||||
|   release_summary: | ||||
|     "Lorem ipsum dolor sit amet, consectetur adipiscing elit. In nec metus aliquet, porta mi ut, ultrices odio. Etiam egestas orci tellus, non semper metus blandit tincidunt. Praesent elementum turpis vel tempor pharetra. Sed quis cursus diam. Proin sem justo.", | ||||
| }; | ||||
|  | ||||
| const ENTITIES = [ | ||||
|   getEntity("update", "update1", "on", { | ||||
|     ...base_attributes, | ||||
|     friendly_name: "Update", | ||||
|   }), | ||||
|   getEntity("update", "update2", "on", { | ||||
|     ...base_attributes, | ||||
|     title: null, | ||||
|     friendly_name: "Update without title", | ||||
|   }), | ||||
|   getEntity("update", "update3", "on", { | ||||
|     ...base_attributes, | ||||
|     release_url: null, | ||||
|     friendly_name: "Update without release_url", | ||||
|   }), | ||||
|   getEntity("update", "update4", "on", { | ||||
|     ...base_attributes, | ||||
|     release_summary: null, | ||||
|     friendly_name: "Update without release_summary", | ||||
|   }), | ||||
|   getEntity("update", "update5", "off", { | ||||
|     ...base_attributes, | ||||
|     current_version: "1.2.3", | ||||
|     friendly_name: "No update", | ||||
|   }), | ||||
|   getEntity("update", "update6", "off", { | ||||
|     ...base_attributes, | ||||
|     skipped_version: "1.2.3", | ||||
|     friendly_name: "Skipped version", | ||||
|   }), | ||||
|   getEntity("update", "update7", "on", { | ||||
|     ...base_attributes, | ||||
|     supported_features: | ||||
|       base_attributes.supported_features + UPDATE_SUPPORT_BACKUP, | ||||
|     friendly_name: "With backup support", | ||||
|   }), | ||||
|   getEntity("update", "update8", "on", { | ||||
|     ...base_attributes, | ||||
|     in_progress: true, | ||||
|     friendly_name: "With true in_progress", | ||||
|   }), | ||||
|   getEntity("update", "update9", "on", { | ||||
|     ...base_attributes, | ||||
|     in_progress: 25, | ||||
|     supported_features: | ||||
|       base_attributes.supported_features + UPDATE_SUPPORT_PROGRESS, | ||||
|     friendly_name: "With 25 in_progress", | ||||
|   }), | ||||
|   getEntity("update", "update10", "on", { | ||||
|     ...base_attributes, | ||||
|     in_progress: 50, | ||||
|     supported_features: | ||||
|       base_attributes.supported_features + UPDATE_SUPPORT_PROGRESS, | ||||
|     friendly_name: "With 50 in_progress", | ||||
|   }), | ||||
|   getEntity("update", "update11", "on", { | ||||
|     ...base_attributes, | ||||
|     in_progress: 75, | ||||
|     supported_features: | ||||
|       base_attributes.supported_features + UPDATE_SUPPORT_PROGRESS, | ||||
|     friendly_name: "With 75 in_progress", | ||||
|   }), | ||||
|   getEntity("update", "update12", "unavailable", { | ||||
|     ...base_attributes, | ||||
|     in_progress: 50, | ||||
|     friendly_name: "Unavailable", | ||||
|   }), | ||||
|   getEntity("update", "update13", "on", { | ||||
|     ...base_attributes, | ||||
|     supported_features: 0, | ||||
|     friendly_name: "No install support", | ||||
|   }), | ||||
|   getEntity("update", "update14", "off", { | ||||
|     ...base_attributes, | ||||
|     current_version: null, | ||||
|     friendly_name: "Update without current_version", | ||||
|   }), | ||||
|   getEntity("update", "update15", "off", { | ||||
|     ...base_attributes, | ||||
|     latest_version: null, | ||||
|     friendly_name: "Update without latest_version", | ||||
|   }), | ||||
| ]; | ||||
|  | ||||
| @customElement("demo-more-info-update") | ||||
| class DemoMoreInfoUpdate extends LitElement { | ||||
|   @property() public hass!: MockHomeAssistant; | ||||
|  | ||||
|   @query("demo-more-infos") private _demoRoot!: HTMLElement; | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
|       <demo-more-infos | ||||
|         .hass=${this.hass} | ||||
|         .entities=${ENTITIES.map((ent) => ent.entityId)} | ||||
|       ></demo-more-infos> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   protected firstUpdated(changedProperties: PropertyValues) { | ||||
|     super.firstUpdated(changedProperties); | ||||
|     const hass = provideHass(this._demoRoot); | ||||
|     hass.updateTranslations(null, "en"); | ||||
|     hass.addEntities(ENTITIES); | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-more-info-update": DemoMoreInfoUpdate; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										17
									
								
								gallery/src/pages/user-test/user-types.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								gallery/src/pages/user-test/user-types.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | ||||
| --- | ||||
| title: "User types" | ||||
| --- | ||||
|  | ||||
| We have defined three user types for Home Assistant. They are a lean segmentation of users that helps us make decisions throughout the product. User types differ from traditional personas in that the segmentation criteria aren’t demographic and don’t personify a group into a single character with a fictitious background story.  | ||||
|  | ||||
| # Outgrowers | ||||
|  | ||||
| Users that outgrow big tech smart home solutions. It just needs to work with easy setup via an app. | ||||
|  | ||||
| # Tinkerers | ||||
|  | ||||
| Technoid users in home networking and development that know how to code. | ||||
|  | ||||
| # Questioner | ||||
|  | ||||
| Users who want more advanced home automation, but need support to make it work. | ||||
| @@ -14,7 +14,7 @@ import memoizeOne from "memoize-one"; | ||||
| import { atLeastVersion } from "../../../src/common/config/version"; | ||||
| import { fireEvent } from "../../../src/common/dom/fire_event"; | ||||
| import { navigate } from "../../../src/common/navigate"; | ||||
| import "../../../src/common/search/search-input"; | ||||
| import "../../../src/components/search-input"; | ||||
| import { extractSearchParam } from "../../../src/common/url/search-params"; | ||||
| import "../../../src/components/ha-button-menu"; | ||||
| import "../../../src/components/ha-icon-button"; | ||||
| @@ -110,8 +110,6 @@ class HassioAddonStore extends LitElement { | ||||
|               <div class="search"> | ||||
|                 <search-input | ||||
|                   .hass=${this.hass} | ||||
|                   no-label-float | ||||
|                   no-underline | ||||
|                   .filter=${this._filter} | ||||
|                   @value-changed=${this._filterChanged} | ||||
|                 ></search-input> | ||||
| @@ -221,13 +219,14 @@ class HassioAddonStore extends LitElement { | ||||
|         margin-top: 24px; | ||||
|       } | ||||
|       .search { | ||||
|         padding: 0 16px; | ||||
|         background: var(--sidebar-background-color); | ||||
|         border-bottom: 1px solid var(--divider-color); | ||||
|         position: sticky; | ||||
|         top: 0; | ||||
|         z-index: 2; | ||||
|       } | ||||
|       .search search-input { | ||||
|         position: relative; | ||||
|         top: 2px; | ||||
|       search-input { | ||||
|         display: block; | ||||
|         --mdc-text-field-fill-color: var(--sidebar-background-color); | ||||
|         --mdc-text-field-idle-line-color: var(--divider-color); | ||||
|       } | ||||
|       .advanced { | ||||
|         padding: 12px; | ||||
|   | ||||
| @@ -1,7 +1,5 @@ | ||||
| import "@material/mwc-button"; | ||||
| import "@polymer/paper-dropdown-menu/paper-dropdown-menu"; | ||||
| import "@polymer/paper-item/paper-item"; | ||||
| import "@polymer/paper-listbox/paper-listbox"; | ||||
| import "@material/mwc-list/mwc-list-item"; | ||||
| import { | ||||
|   css, | ||||
|   CSSResultGroup, | ||||
| @@ -11,10 +9,11 @@ import { | ||||
|   TemplateResult, | ||||
| } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import "web-animations-js/web-animations-next-lite.min"; | ||||
| import { stopPropagation } from "../../../../src/common/dom/stop_propagation"; | ||||
| import "../../../../src/components/buttons/ha-progress-button"; | ||||
| import "../../../../src/components/ha-alert"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import "../../../../src/components/ha-select"; | ||||
| import { | ||||
|   HassioAddonDetails, | ||||
|   HassioAddonSetOptionParams, | ||||
| @@ -57,49 +56,44 @@ class HassioAddonAudio extends LitElement { | ||||
|           ${this._error | ||||
|             ? html`<ha-alert alert-type="error">${this._error}</ha-alert>` | ||||
|             : ""} | ||||
|  | ||||
|           <paper-dropdown-menu | ||||
|           ${this._inputDevices && | ||||
|           html`<ha-select | ||||
|             .label=${this.supervisor.localize( | ||||
|               "addon.configuration.audio.input" | ||||
|             )} | ||||
|             @iron-select=${this._setInputDevice} | ||||
|             @selected=${this._setInputDevice} | ||||
|             @closed=${stopPropagation} | ||||
|             fixedMenuPosition | ||||
|             naturalMenuWidth | ||||
|             .value=${this._selectedInput!} | ||||
|           > | ||||
|             <paper-listbox | ||||
|               slot="dropdown-content" | ||||
|               attr-for-selected="device" | ||||
|               .selected=${this._selectedInput!} | ||||
|             > | ||||
|               ${this._inputDevices && | ||||
|               this._inputDevices.map( | ||||
|                 (item) => html` | ||||
|                   <paper-item device=${item.device || ""}> | ||||
|                     ${item.name} | ||||
|                   </paper-item> | ||||
|                 ` | ||||
|               )} | ||||
|             </paper-listbox> | ||||
|           </paper-dropdown-menu> | ||||
|           <paper-dropdown-menu | ||||
|             ${this._inputDevices.map( | ||||
|               (item) => html` | ||||
|                 <mwc-list-item .value=${item.device || ""}> | ||||
|                   ${item.name} | ||||
|                 </mwc-list-item> | ||||
|               ` | ||||
|             )} | ||||
|           </ha-select>`} | ||||
|           ${this._outputDevices && | ||||
|           html`<ha-select | ||||
|             .label=${this.supervisor.localize( | ||||
|               "addon.configuration.audio.output" | ||||
|             )} | ||||
|             @iron-select=${this._setOutputDevice} | ||||
|             @selected=${this._setOutputDevice} | ||||
|             @closed=${stopPropagation} | ||||
|             fixedMenuPosition | ||||
|             naturalMenuWidth | ||||
|             .value=${this._selectedOutput!} | ||||
|           > | ||||
|             <paper-listbox | ||||
|               slot="dropdown-content" | ||||
|               attr-for-selected="device" | ||||
|               .selected=${this._selectedOutput!} | ||||
|             > | ||||
|               ${this._outputDevices && | ||||
|               this._outputDevices.map( | ||||
|                 (item) => html` | ||||
|                   <paper-item device=${item.device || ""} | ||||
|                     >${item.name}</paper-item | ||||
|                   > | ||||
|                 ` | ||||
|               )} | ||||
|             </paper-listbox> | ||||
|           </paper-dropdown-menu> | ||||
|             ${this._outputDevices.map( | ||||
|               (item) => html` | ||||
|                 <mwc-list-item .value=${item.device || ""} | ||||
|                   >${item.name}</mwc-list-item | ||||
|                 > | ||||
|               ` | ||||
|             )} | ||||
|           </ha-select>`} | ||||
|         </div> | ||||
|         <div class="card-actions"> | ||||
|           <ha-progress-button @click=${this._saveSettings}> | ||||
| @@ -116,8 +110,7 @@ class HassioAddonAudio extends LitElement { | ||||
|       hassioStyle, | ||||
|       css` | ||||
|         :host, | ||||
|         ha-card, | ||||
|         paper-dropdown-menu { | ||||
|         ha-card { | ||||
|           display: block; | ||||
|         } | ||||
|         paper-item { | ||||
| @@ -126,24 +119,30 @@ class HassioAddonAudio extends LitElement { | ||||
|         .card-actions { | ||||
|           text-align: right; | ||||
|         } | ||||
|         ha-select { | ||||
|           width: 100%; | ||||
|         } | ||||
|         ha-select:last-child { | ||||
|           margin-top: 8px; | ||||
|         } | ||||
|       `, | ||||
|     ]; | ||||
|   } | ||||
|  | ||||
|   protected update(changedProperties: PropertyValues): void { | ||||
|     super.update(changedProperties); | ||||
|   protected willUpdate(changedProperties: PropertyValues): void { | ||||
|     super.willUpdate(changedProperties); | ||||
|     if (changedProperties.has("addon")) { | ||||
|       this._addonChanged(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   private _setInputDevice(ev): void { | ||||
|     const device = ev.detail.item.getAttribute("device"); | ||||
|     const device = ev.target.value; | ||||
|     this._selectedInput = device; | ||||
|   } | ||||
|  | ||||
|   private _setOutputDevice(ev): void { | ||||
|     const device = ev.detail.item.getAttribute("device"); | ||||
|     const device = ev.target.value; | ||||
|     this._selectedOutput = device; | ||||
|   } | ||||
|  | ||||
|   | ||||
| @@ -9,6 +9,7 @@ import { | ||||
|   mdiFlask, | ||||
|   mdiHomeAssistant, | ||||
|   mdiKey, | ||||
|   mdiLinkLock, | ||||
|   mdiNetwork, | ||||
|   mdiNumeric1, | ||||
|   mdiNumeric2, | ||||
| @@ -16,6 +17,8 @@ import { | ||||
|   mdiNumeric4, | ||||
|   mdiNumeric5, | ||||
|   mdiNumeric6, | ||||
|   mdiNumeric7, | ||||
|   mdiNumeric8, | ||||
|   mdiPound, | ||||
|   mdiShield, | ||||
| } from "@mdi/js"; | ||||
| @@ -31,6 +34,7 @@ import "../../../../src/components/buttons/ha-progress-button"; | ||||
| import "../../../../src/components/ha-alert"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import "../../../../src/components/ha-chip"; | ||||
| import "../../../../src/components/ha-chip-set"; | ||||
| import "../../../../src/components/ha-markdown"; | ||||
| import "../../../../src/components/ha-settings-row"; | ||||
| import "../../../../src/components/ha-svg-icon"; | ||||
| @@ -84,6 +88,8 @@ const RATING_ICON = { | ||||
|   4: mdiNumeric4, | ||||
|   5: mdiNumeric5, | ||||
|   6: mdiNumeric6, | ||||
|   7: mdiNumeric7, | ||||
|   8: mdiNumeric8, | ||||
| }; | ||||
|  | ||||
| @customElement("hassio-addon-info") | ||||
| @@ -209,7 +215,7 @@ class HassioAddonInfo extends LitElement { | ||||
|                 >`} | ||||
|           </div> | ||||
|  | ||||
|           <div class="capabilities"> | ||||
|           <ha-chip-set class="capabilities"> | ||||
|             ${this.addon.stage !== "stable" | ||||
|               ? html` <ha-chip | ||||
|                   hasIcon | ||||
| @@ -234,9 +240,9 @@ class HassioAddonInfo extends LitElement { | ||||
|             <ha-chip | ||||
|               hasIcon | ||||
|               class=${classMap({ | ||||
|                 green: [5, 6].includes(Number(this.addon.rating)), | ||||
|                 yellow: [3, 4].includes(Number(this.addon.rating)), | ||||
|                 red: [1, 2].includes(Number(this.addon.rating)), | ||||
|                 green: Number(this.addon.rating) >= 6, | ||||
|                 yellow: [3, 4, 5].includes(Number(this.addon.rating)), | ||||
|                 red: Number(this.addon.rating) >= 2, | ||||
|               })} | ||||
|               @click=${this._showMoreInfo} | ||||
|               id="rating" | ||||
| @@ -364,7 +370,17 @@ class HassioAddonInfo extends LitElement { | ||||
|                   </ha-chip> | ||||
|                 ` | ||||
|               : ""} | ||||
|           </div> | ||||
|             ${this.addon.signed | ||||
|               ? html` | ||||
|                   <ha-chip hasIcon @click=${this._showMoreInfo} id="signed"> | ||||
|                     <ha-svg-icon slot="icon" .path=${mdiLinkLock}></ha-svg-icon> | ||||
|                     ${this.supervisor.localize( | ||||
|                       "addon.dashboard.capability.label.signed" | ||||
|                     )} | ||||
|                   </ha-chip> | ||||
|                 ` | ||||
|               : ""} | ||||
|           </ha-chip-set> | ||||
|  | ||||
|           <div class="description light-color"> | ||||
|             ${this.addon.description}.<br /> | ||||
|   | ||||
| @@ -1,5 +1,4 @@ | ||||
| import { mdiFolderUpload } from "@mdi/js"; | ||||
| import "@polymer/paper-input/paper-input-container"; | ||||
| import { html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, state } from "lit/decorators"; | ||||
| import { fireEvent } from "../../../src/common/dom/fire_event"; | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| import { mdiFolder, mdiHomeAssistant, mdiPuzzle } from "@mdi/js"; | ||||
| import { PaperInputElement } from "@polymer/paper-input/paper-input"; | ||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import { customElement, property, query } from "lit/decorators"; | ||||
| import { atLeastVersion } from "../../../src/common/config/version"; | ||||
| import { formatDate } from "../../../src/common/datetime/format_date"; | ||||
| import { formatDateTime } from "../../../src/common/datetime/format_date_time"; | ||||
| @@ -92,6 +92,8 @@ export class SupervisorBackupContent extends LitElement { | ||||
|  | ||||
|   @property() public confirmBackupPassword = ""; | ||||
|  | ||||
|   @query("paper-input, ha-radio, ha-checkbox", true) private _focusTarget; | ||||
|  | ||||
|   public willUpdate(changedProps) { | ||||
|     super.willUpdate(changedProps); | ||||
|     if (!this.hasUpdated) { | ||||
| @@ -109,6 +111,10 @@ export class SupervisorBackupContent extends LitElement { | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   public override focus() { | ||||
|     this._focusTarget?.focus(); | ||||
|   } | ||||
|  | ||||
|   private _localize = (string: string) => | ||||
|     this.supervisor?.localize(`backup.${string}`) || | ||||
|     this.localize!(`ui.panel.page-onboarding.restore.${string}`); | ||||
| @@ -169,24 +175,23 @@ export class SupervisorBackupContent extends LitElement { | ||||
|         : ""} | ||||
|       ${this.backupType === "partial" | ||||
|         ? html`<div class="partial-picker"> | ||||
|             ${this.backup && this.backup.homeassistant | ||||
|               ? html` | ||||
|                   <ha-formfield | ||||
|                     .label=${html`<supervisor-formfield-label | ||||
|                       label="Home Assistant" | ||||
|                       .iconPath=${mdiHomeAssistant} | ||||
|                       .version=${this.backup.homeassistant} | ||||
|                     > | ||||
|                     </supervisor-formfield-label>`} | ||||
|                   > | ||||
|                     <ha-checkbox | ||||
|                       .checked=${this.homeAssistant} | ||||
|                       @click=${this.toggleHomeAssistant} | ||||
|                     > | ||||
|                     </ha-checkbox> | ||||
|                   </ha-formfield> | ||||
|                 ` | ||||
|               : ""} | ||||
|             <ha-formfield | ||||
|               .label=${html`<supervisor-formfield-label | ||||
|                 label="Home Assistant" | ||||
|                 .iconPath=${mdiHomeAssistant} | ||||
|                 .version=${this.backup | ||||
|                   ? this.backup.homeassistant | ||||
|                   : this.hass.config.version} | ||||
|               > | ||||
|               </supervisor-formfield-label>`} | ||||
|             > | ||||
|               <ha-checkbox | ||||
|                 .checked=${this.homeAssistant} | ||||
|                 @click=${this.toggleHomeAssistant} | ||||
|               > | ||||
|               </ha-checkbox> | ||||
|             </ha-formfield> | ||||
|  | ||||
|             ${foldersSection?.templates.length | ||||
|               ? html` | ||||
|                   <ha-formfield | ||||
|   | ||||
| @@ -148,7 +148,6 @@ export class HassioUpdate extends LitElement { | ||||
|         } | ||||
|         ha-settings-row { | ||||
|           padding: 0; | ||||
|           --paper-item-body-two-line-min-height: 32px; | ||||
|         } | ||||
|       `, | ||||
|     ]; | ||||
|   | ||||
| @@ -64,6 +64,7 @@ export class DialogHassioBackupUpload | ||||
|               .path=${mdiClose} | ||||
|               slot="actionItems" | ||||
|               dialogAction="cancel" | ||||
|               dialogInitialFocus | ||||
|             ></ha-icon-button> | ||||
|           </ha-header-bar> | ||||
|         </div> | ||||
|   | ||||
| @@ -92,6 +92,7 @@ class HassioBackupDialog | ||||
|               .backup=${this._backup} | ||||
|               .onboarding=${this._dialogParams.onboarding || false} | ||||
|               .localize=${this._dialogParams.localize} | ||||
|               dialogInitialFocus | ||||
|             > | ||||
|             </supervisor-backup-content>`} | ||||
|         ${this._error | ||||
|   | ||||
| @@ -61,6 +61,7 @@ class HassioCreateBackupDialog extends LitElement { | ||||
|           : html`<supervisor-backup-content | ||||
|               .hass=${this.hass} | ||||
|               .supervisor=${this._dialogParams.supervisor} | ||||
|               dialogInitialFocus | ||||
|             > | ||||
|             </supervisor-backup-content>`} | ||||
|         ${this._error | ||||
|   | ||||
| @@ -1,12 +1,11 @@ | ||||
| import "@polymer/paper-dropdown-menu/paper-dropdown-menu"; | ||||
| import "@polymer/paper-item/paper-item"; | ||||
| import "@polymer/paper-listbox/paper-listbox"; | ||||
| import "@material/mwc-list/mwc-list-item"; | ||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import memoizeOne from "memoize-one"; | ||||
| import { fireEvent } from "../../../../src/common/dom/fire_event"; | ||||
| import "../../../../src/components/ha-circular-progress"; | ||||
| import "../../../../src/components/ha-markdown"; | ||||
| import "../../../../src/components/ha-select"; | ||||
| import { | ||||
|   extractApiErrorMessage, | ||||
|   ignoreSupervisorError, | ||||
| @@ -90,18 +89,20 @@ class HassioDatadiskDialog extends LitElement { | ||||
|                     )} | ||||
|                     <br /><br /> | ||||
|  | ||||
|                     <paper-dropdown-menu | ||||
|                     <ha-select | ||||
|                       .label=${this.dialogParams.supervisor.localize( | ||||
|                         "dialog.datadisk_move.select_device" | ||||
|                       )} | ||||
|                       @value-changed=${this._select_device} | ||||
|                       @selected=${this._select_device} | ||||
|                       dialogInitialFocus | ||||
|                     > | ||||
|                       <paper-listbox slot="dropdown-content"> | ||||
|                         ${this.devices.map( | ||||
|                           (device) => html`<paper-item>${device}</paper-item>` | ||||
|                         )} | ||||
|                       </paper-listbox> | ||||
|                     </paper-dropdown-menu> | ||||
|                       ${this.devices.map( | ||||
|                         (device) => | ||||
|                           html`<mwc-list-item .value=${device} | ||||
|                             >${device}</mwc-list-item | ||||
|                           >` | ||||
|                       )} | ||||
|                     </ha-select> | ||||
|                   ` | ||||
|                 : this.devices === undefined | ||||
|                 ? this.dialogParams.supervisor.localize( | ||||
| @@ -111,7 +112,11 @@ class HassioDatadiskDialog extends LitElement { | ||||
|                     "dialog.datadisk_move.no_devices" | ||||
|                   )} | ||||
|  | ||||
|               <mwc-button slot="secondaryAction" @click=${this.closeDialog}> | ||||
|               <mwc-button | ||||
|                 slot="secondaryAction" | ||||
|                 @click=${this.closeDialog} | ||||
|                 dialogInitialFocus | ||||
|               > | ||||
|                 ${this.dialogParams.supervisor.localize( | ||||
|                   "dialog.datadisk_move.cancel" | ||||
|                 )} | ||||
| @@ -130,8 +135,8 @@ class HassioDatadiskDialog extends LitElement { | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   private _select_device(event) { | ||||
|     this.selectedDevice = event.detail.value; | ||||
|   private _select_device(ev) { | ||||
|     this.selectedDevice = ev.target.value; | ||||
|   } | ||||
|  | ||||
|   private async _moveDatadisk() { | ||||
| @@ -156,7 +161,7 @@ class HassioDatadiskDialog extends LitElement { | ||||
|       haStyle, | ||||
|       haStyleDialog, | ||||
|       css` | ||||
|         paper-dropdown-menu { | ||||
|         ha-select { | ||||
|           width: 100%; | ||||
|         } | ||||
|         ha-circular-progress { | ||||
|   | ||||
| @@ -3,7 +3,7 @@ import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import memoizeOne from "memoize-one"; | ||||
| import { fireEvent } from "../../../../src/common/dom/fire_event"; | ||||
| import "../../../../src/common/search/search-input"; | ||||
| import "../../../../src/components/search-input"; | ||||
| import { stringCompare } from "../../../../src/common/string/compare"; | ||||
| import "../../../../src/components/ha-dialog"; | ||||
| import "../../../../src/components/ha-expansion-panel"; | ||||
| @@ -80,8 +80,6 @@ class HassioHardwareDialog extends LitElement { | ||||
|           ></ha-icon-button> | ||||
|           <search-input | ||||
|             .hass=${this.hass} | ||||
|             autofocus | ||||
|             no-label-float | ||||
|             .filter=${this._filter} | ||||
|             @value-changed=${this._handleSearchChange} | ||||
|             .label=${this._dialogParams.supervisor.localize( | ||||
| @@ -178,7 +176,7 @@ class HassioHardwareDialog extends LitElement { | ||||
|           padding: 0.2em 0.4em; | ||||
|         } | ||||
|         search-input { | ||||
|           margin: 0 16px; | ||||
|           margin: 8px 16px 0; | ||||
|           display: block; | ||||
|         } | ||||
|         .device-property { | ||||
|   | ||||
| @@ -37,7 +37,10 @@ class HassioMarkdownDialog extends LitElement { | ||||
|         @closed=${this.closeDialog} | ||||
|         .heading=${createCloseHeading(this.hass, this.title)} | ||||
|       > | ||||
|         <ha-markdown .content=${this.content || ""}></ha-markdown> | ||||
|         <ha-markdown | ||||
|           .content=${this.content || ""} | ||||
|           dialogInitialFocus | ||||
|         ></ha-markdown> | ||||
|       </ha-dialog> | ||||
|     `; | ||||
|   } | ||||
|   | ||||
| @@ -119,6 +119,7 @@ export class DialogHassioNetwork | ||||
|                     html`<mwc-tab | ||||
|                       .id=${device.interface} | ||||
|                       .label=${device.interface} | ||||
|                       dialogInitialFocus | ||||
|                     > | ||||
|                     </mwc-tab>` | ||||
|                 )} | ||||
| @@ -315,6 +316,7 @@ export class DialogHassioNetwork | ||||
|               value="auto" | ||||
|               name="${version}method" | ||||
|               .checked=${this._interface![version]?.method === "auto"} | ||||
|               dialogInitialFocus | ||||
|             > | ||||
|             </ha-radio> | ||||
|           </ha-formfield> | ||||
|   | ||||
| @@ -80,6 +80,7 @@ class HassioRegistriesDialog extends LitElement { | ||||
|                 .schema=${SCHEMA} | ||||
|                 @value-changed=${this._valueChanged} | ||||
|                 .computeLabel=${this._computeLabel} | ||||
|                 dialogInitialFocus | ||||
|               ></ha-form> | ||||
|               <div class="action"> | ||||
|                 <mwc-button | ||||
| @@ -124,7 +125,7 @@ class HassioRegistriesDialog extends LitElement { | ||||
|                     </ha-alert> | ||||
|                   `} | ||||
|               <div class="action"> | ||||
|                 <mwc-button @click=${this._addRegistry}> | ||||
|                 <mwc-button @click=${this._addRegistry} dialogInitialFocus> | ||||
|                   ${this.supervisor.localize( | ||||
|                     "dialog.registries.add_new_registry" | ||||
|                   )} | ||||
|   | ||||
| @@ -106,6 +106,9 @@ class HassioRepositoriesDialog extends LitElement { | ||||
|                     </paper-item-body> | ||||
|                     <div class="delete"> | ||||
|                       <ha-icon-button | ||||
|                         .label=${this._dialogParams!.supervisor.localize( | ||||
|                           "dialog.repositories.remove" | ||||
|                         )} | ||||
|                         .disabled=${usedRepositories.includes(repo.slug)} | ||||
|                         .slug=${repo.slug} | ||||
|                         .path=${usedRepositories.includes(repo.slug) | ||||
| @@ -139,6 +142,7 @@ class HassioRepositoriesDialog extends LitElement { | ||||
|                 "dialog.repositories.add" | ||||
|               )} | ||||
|               @keydown=${this._handleKeyAdd} | ||||
|               dialogInitialFocus | ||||
|             ></paper-input> | ||||
|             <mwc-button @click=${this._addRepository}> | ||||
|               ${this._processing | ||||
|   | ||||
| @@ -1,9 +1,12 @@ | ||||
| // Compat needs to be first import | ||||
| import "../../src/resources/compatibility"; | ||||
| import { setCancelSyntheticClickEvents } from "@polymer/polymer/lib/utils/settings"; | ||||
| import "../../src/resources/roboto"; | ||||
| import "../../src/resources/safari-14-attachshadow-patch"; | ||||
| import "./hassio-main"; | ||||
|  | ||||
| setCancelSyntheticClickEvents(false); | ||||
|  | ||||
| const styleEl = document.createElement("style"); | ||||
| styleEl.innerHTML = ` | ||||
| body { | ||||
|   | ||||
| @@ -121,7 +121,8 @@ export class HassioMain extends SupervisorBaseElement { | ||||
|       this.parentElement, | ||||
|       this.hass.themes, | ||||
|       themeName, | ||||
|       themeSettings | ||||
|       themeSettings, | ||||
|       true | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1,12 +1,10 @@ | ||||
| import "@material/mwc-button"; | ||||
| import "@polymer/paper-dropdown-menu/paper-dropdown-menu"; | ||||
| import "@polymer/paper-item/paper-item"; | ||||
| import "@polymer/paper-listbox/paper-listbox"; | ||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import "../../../src/components/buttons/ha-progress-button"; | ||||
| import "../../../src/components/ha-alert"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import "../../../src/components/ha-select"; | ||||
| import { extractApiErrorMessage } from "../../../src/data/hassio/common"; | ||||
| import { fetchHassioLogs } from "../../../src/data/hassio/supervisor"; | ||||
| import { Supervisor } from "../../../src/data/supervisor/supervisor"; | ||||
| @@ -73,24 +71,19 @@ class HassioSupervisorLog extends LitElement { | ||||
|           : ""} | ||||
|         ${this.hass.userData?.showAdvanced | ||||
|           ? html` | ||||
|               <paper-dropdown-menu | ||||
|               <ha-select | ||||
|                 .label=${this.supervisor.localize("system.log.log_provider")} | ||||
|                 @iron-select=${this._setLogProvider} | ||||
|                 @selected=${this._setLogProvider} | ||||
|                 .value=${this._selectedLogProvider} | ||||
|               > | ||||
|                 <paper-listbox | ||||
|                   slot="dropdown-content" | ||||
|                   attr-for-selected="provider" | ||||
|                   .selected=${this._selectedLogProvider} | ||||
|                 > | ||||
|                   ${logProviders.map( | ||||
|                     (provider) => html` | ||||
|                       <paper-item provider=${provider.key}> | ||||
|                         ${provider.name} | ||||
|                       </paper-item> | ||||
|                     ` | ||||
|                   )} | ||||
|                 </paper-listbox> | ||||
|               </paper-dropdown-menu> | ||||
|                 ${logProviders.map( | ||||
|                   (provider) => html` | ||||
|                     <mwc-list-item .value=${provider.key}> | ||||
|                       ${provider.name} | ||||
|                     </mwc-list-item> | ||||
|                   ` | ||||
|                 )} | ||||
|               </ha-select> | ||||
|             ` | ||||
|           : ""} | ||||
|  | ||||
| @@ -110,7 +103,7 @@ class HassioSupervisorLog extends LitElement { | ||||
|   } | ||||
|  | ||||
|   private async _setLogProvider(ev): Promise<void> { | ||||
|     const provider = ev.detail.item.getAttribute("provider"); | ||||
|     const provider = ev.target.value; | ||||
|     this._selectedLogProvider = provider; | ||||
|     this._loadData(); | ||||
|   } | ||||
| @@ -153,9 +146,9 @@ class HassioSupervisorLog extends LitElement { | ||||
|         pre { | ||||
|           white-space: pre-wrap; | ||||
|         } | ||||
|         paper-dropdown-menu { | ||||
|           padding: 0 2%; | ||||
|           width: 96%; | ||||
|         ha-select { | ||||
|           width: 100%; | ||||
|           margin-bottom: 4px; | ||||
|         } | ||||
|       `, | ||||
|     ]; | ||||
|   | ||||
| @@ -10,7 +10,6 @@ import { | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import memoizeOne from "memoize-one"; | ||||
| import { fireEvent } from "../../../src/common/dom/fire_event"; | ||||
| import "../../../src/common/search/search-input"; | ||||
| import "../../../src/components/buttons/ha-progress-button"; | ||||
| import "../../../src/components/ha-alert"; | ||||
| import "../../../src/components/ha-button-menu"; | ||||
|   | ||||
							
								
								
									
										35
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										35
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,8 +1,8 @@ | ||||
| { | ||||
|   "description": "A frontend for Home Assistant using the Polymer framework", | ||||
|   "description": "A frontend for Home Assistant", | ||||
|   "repository": { | ||||
|     "type": "git", | ||||
|     "url": "https://github.com/home-assistant/home-assistant-polymer" | ||||
|     "url": "https://github.com/home-assistant/frontend" | ||||
|   }, | ||||
|   "name": "home-assistant-frontend", | ||||
|   "version": "1.0.0", | ||||
| @@ -46,6 +46,7 @@ | ||||
|     "@fullcalendar/daygrid": "5.9.0", | ||||
|     "@fullcalendar/interaction": "5.9.0", | ||||
|     "@fullcalendar/list": "5.9.0", | ||||
|     "@lit-labs/motion": "^1.0.2", | ||||
|     "@lit-labs/virtualizer": "patch:@lit-labs/virtualizer@0.7.0-pre.2#./.yarn/patches/@lit-labs/virtualizer/event-target-shim.patch", | ||||
|     "@material/chips": "14.0.0-canary.261f2db59.0", | ||||
|     "@material/data-table": "14.0.0-canary.261f2db59.0", | ||||
| @@ -78,7 +79,6 @@ | ||||
|     "@polymer/iron-icon": "^3.0.1", | ||||
|     "@polymer/iron-input": "^3.0.1", | ||||
|     "@polymer/iron-resizable-behavior": "^3.0.1", | ||||
|     "@polymer/paper-dropdown-menu": "^3.2.0", | ||||
|     "@polymer/paper-input": "^3.2.1", | ||||
|     "@polymer/paper-item": "^3.0.1", | ||||
|     "@polymer/paper-listbox": "^3.0.1", | ||||
| @@ -95,6 +95,7 @@ | ||||
|     "@vibrant/core": "^3.2.1-alpha.1", | ||||
|     "@vibrant/quantizer-mmcq": "^3.2.1-alpha.1", | ||||
|     "@vue/web-component-wrapper": "^1.2.0", | ||||
|     "@webcomponents/scoped-custom-element-registry": "^0.0.5", | ||||
|     "@webcomponents/webcomponentsjs": "^2.2.10", | ||||
|     "app-datepicker": "^5.0.1", | ||||
|     "chart.js": "^3.3.2", | ||||
| @@ -106,8 +107,8 @@ | ||||
|     "deep-freeze": "^0.0.1", | ||||
|     "fuse.js": "^6.0.0", | ||||
|     "google-timezones-json": "^1.0.2", | ||||
|     "hls.js": "^1.0.11", | ||||
|     "home-assistant-js-websocket": "^6.0.1", | ||||
|     "hls.js": "^1.1.5", | ||||
|     "home-assistant-js-websocket": "^7.0.1", | ||||
|     "idb-keyval": "^5.1.3", | ||||
|     "intl-messageformat": "^9.9.1", | ||||
|     "js-yaml": "^4.1.0", | ||||
| @@ -115,7 +116,7 @@ | ||||
|     "leaflet-draw": "^1.0.4", | ||||
|     "lit": "^2.1.2", | ||||
|     "lit-vaadin-helpers": "^0.3.0", | ||||
|     "marked": "^3.0.2", | ||||
|     "marked": "^4.0.12", | ||||
|     "memoize-one": "^5.2.1", | ||||
|     "node-vibrant": "3.2.1-alpha.1", | ||||
|     "proxy-polyfill": "^0.3.2", | ||||
| @@ -134,13 +135,12 @@ | ||||
|     "vis-network": "^8.5.4", | ||||
|     "vue": "^2.6.12", | ||||
|     "vue2-daterange-picker": "^0.5.1", | ||||
|     "web-animations-js": "^2.3.2", | ||||
|     "workbox-cacheable-response": "^6.1.5", | ||||
|     "workbox-core": "^6.1.5", | ||||
|     "workbox-expiration": "^6.1.5", | ||||
|     "workbox-precaching": "^6.1.5", | ||||
|     "workbox-routing": "^6.1.5", | ||||
|     "workbox-strategies": "^6.1.5", | ||||
|     "workbox-cacheable-response": "^6.4.2", | ||||
|     "workbox-core": "^6.4.2", | ||||
|     "workbox-expiration": "^6.4.2", | ||||
|     "workbox-precaching": "^6.4.2", | ||||
|     "workbox-routing": "^6.4.2", | ||||
|     "workbox-strategies": "^6.4.2", | ||||
|     "xss": "^1.0.9" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
| @@ -169,7 +169,7 @@ | ||||
|     "@types/js-yaml": "^4", | ||||
|     "@types/leaflet": "^1", | ||||
|     "@types/leaflet-draw": "^1", | ||||
|     "@types/marked": "^2", | ||||
|     "@types/marked": "^4", | ||||
|     "@types/mocha": "^8", | ||||
|     "@types/qrcode": "^1.4.2", | ||||
|     "@types/sortablejs": "^1", | ||||
| @@ -196,7 +196,7 @@ | ||||
|     "fs-extra": "^7.0.1", | ||||
|     "glob": "^7.2.0", | ||||
|     "gulp": "^4.0.2", | ||||
|     "gulp-foreach": "^0.1.0", | ||||
|     "gulp-flatmap": "^1.0.2", | ||||
|     "gulp-json-transform": "^0.4.6", | ||||
|     "gulp-merge-json": "^1.3.1", | ||||
|     "gulp-rename": "^2.0.0", | ||||
| @@ -233,7 +233,7 @@ | ||||
|     "webpack-dev-server": "^4.3.0", | ||||
|     "webpack-manifest-plugin": "^4.0.2", | ||||
|     "webpackbar": "^5.0.0-3", | ||||
|     "workbox-build": "^6.1.5" | ||||
|     "workbox-build": "^6.4.2" | ||||
|   }, | ||||
|   "_comment": "Polymer 3.2 contained a bug, fixed in https://github.com/Polymer/polymer/pull/5569, add as patch", | ||||
|   "resolutions": { | ||||
| @@ -253,5 +253,6 @@ | ||||
|   "prettier": { | ||||
|     "trailingComma": "es5", | ||||
|     "arrowParens": "always" | ||||
|   } | ||||
|   }, | ||||
|   "packageManager": "yarn@3.2.0" | ||||
| } | ||||
|   | ||||
| @@ -2,6 +2,6 @@ | ||||
| from pathlib import Path | ||||
|  | ||||
|  | ||||
| def where(): | ||||
| def where() -> Path: | ||||
|     """Return path to the frontend.""" | ||||
|     return Path(__file__).parent | ||||
|   | ||||
							
								
								
									
										0
									
								
								public/py.typed
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								public/py.typed
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										18
									
								
								script/core
									
									
									
									
									
								
							
							
						
						
									
										18
									
								
								script/core
									
									
									
									
									
								
							| @@ -4,6 +4,8 @@ | ||||
| # Stop on errors | ||||
| set -e | ||||
|  | ||||
| WD="${WORKSPACE_DIRECTORY:=/workspaces/frontend}" | ||||
|  | ||||
| if [ -z "${DEVCONTAINER}" ]; then | ||||
|   echo "This task should only run inside a devcontainer, for local install HA Core in a venv." | ||||
|   exit 1 | ||||
| @@ -16,9 +18,9 @@ if [ -z $(which hass) ]; then | ||||
|     git+git://github.com/home-assistant/home-assistant.git@dev | ||||
| fi | ||||
|  | ||||
| if [ ! -d "/workspaces/frontend/config" ]; then | ||||
| if [ ! -d "${WD}/config" ]; then | ||||
|   echo "Creating default configuration." | ||||
|   mkdir -p "/workspaces/frontend/config"; | ||||
|   mkdir -p "${WD}/config"; | ||||
|   hass --script ensure_config -c config | ||||
|   echo "demo: | ||||
|  | ||||
| @@ -26,24 +28,24 @@ logger: | ||||
|   default: info | ||||
|   logs: | ||||
|     homeassistant.components.frontend: debug | ||||
| " >> /workspaces/frontend/config/configuration.yaml | ||||
| " >> "${WD}/config/configuration.yaml" | ||||
|  | ||||
|   if [ ! -z "${HASSIO}" ]; then | ||||
|   echo " | ||||
| # frontend: | ||||
| #   development_repo: /workspaces/frontend | ||||
| #   development_repo: ${WD} | ||||
|  | ||||
| hassio: | ||||
|   development_repo: /workspaces/frontend" >> /workspaces/frontend/config/configuration.yaml | ||||
|   development_repo: ${WD}" >> "${WD}/config/configuration.yaml" | ||||
|   else | ||||
|   echo " | ||||
| frontend: | ||||
|   development_repo: /workspaces/frontend | ||||
|   development_repo: ${WD} | ||||
|  | ||||
| # hassio: | ||||
| #   development_repo: /workspaces/frontend" >> /workspaces/frontend/config/configuration.yaml | ||||
| #   development_repo: ${WD}" >> "${WD}/config/configuration.yaml" | ||||
|   fi | ||||
|  | ||||
| fi | ||||
|  | ||||
| hass -c /workspaces/frontend/config | ||||
| hass -c "${WD}/config" | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| [metadata] | ||||
| name         = home-assistant-frontend | ||||
| version      = 20220203.0 | ||||
| version      = 20220322.0 | ||||
| author       = The Home Assistant Authors | ||||
| author_email = hello@home-assistant.io | ||||
| license      = Apache-2.0 | ||||
| @@ -19,3 +19,8 @@ python_requires = >= 3.4.0 | ||||
| [options.packages.find] | ||||
| include = | ||||
|     hass_frontend* | ||||
|  | ||||
| [mypy] | ||||
| python_version = 3.4 | ||||
| show_error_codes = True | ||||
| strict = True | ||||
|   | ||||
							
								
								
									
										7
									
								
								setup.py
									
									
									
									
									
								
							
							
						
						
									
										7
									
								
								setup.py
									
									
									
									
									
								
							| @@ -1,7 +0,0 @@ | ||||
| """ | ||||
| Entry point for setuptools. Required for editable installs. | ||||
| TODO: Remove file after updating to pip 21.3 | ||||
| """ | ||||
| from setuptools import setup | ||||
|  | ||||
| setup() | ||||
| @@ -101,13 +101,19 @@ class HaAuthorize extends litLocalizeLiteMixin(LitElement) { | ||||
|     this._fetchAuthProviders(); | ||||
|  | ||||
|     if (matchMedia("(prefers-color-scheme: dark)").matches) { | ||||
|       applyThemesOnElement(document.documentElement, { | ||||
|         default_theme: "default", | ||||
|         default_dark_theme: null, | ||||
|         themes: {}, | ||||
|         darkMode: true, | ||||
|         theme: "default", | ||||
|       }); | ||||
|       applyThemesOnElement( | ||||
|         document.documentElement, | ||||
|         { | ||||
|           default_theme: "default", | ||||
|           default_dark_theme: null, | ||||
|           themes: {}, | ||||
|           darkMode: true, | ||||
|           theme: "default", | ||||
|         }, | ||||
|         undefined, | ||||
|         undefined, | ||||
|         true | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     if (!this.redirectUri) { | ||||
|   | ||||
| @@ -187,6 +187,7 @@ export const DOMAINS_WITH_MORE_INFO = [ | ||||
|   "scene", | ||||
|   "sun", | ||||
|   "timer", | ||||
|   "update", | ||||
|   "vacuum", | ||||
|   "water_heater", | ||||
|   "weather", | ||||
| @@ -200,6 +201,7 @@ export const DOMAINS_HIDE_DEFAULT_MORE_INFO = [ | ||||
|   "input_text", | ||||
|   "number", | ||||
|   "scene", | ||||
|   "update", | ||||
|   "select", | ||||
| ]; | ||||
|  | ||||
|   | ||||
| @@ -3,9 +3,9 @@ import type { ForDict } from "../../data/automation"; | ||||
|  | ||||
| export const createDurationData = ( | ||||
|   duration: string | number | ForDict | undefined | ||||
| ): HaDurationData => { | ||||
| ): HaDurationData | undefined => { | ||||
|   if (duration === undefined) { | ||||
|     return {}; | ||||
|     return undefined; | ||||
|   } | ||||
|   if (typeof duration !== "object") { | ||||
|     if (typeof duration === "string" || isNaN(duration)) { | ||||
|   | ||||
| @@ -31,11 +31,12 @@ export const applyThemesOnElement = ( | ||||
|   element, | ||||
|   themes: HomeAssistant["themes"], | ||||
|   selectedTheme?: string, | ||||
|   themeSettings?: Partial<HomeAssistant["selectedTheme"]> | ||||
|   themeSettings?: Partial<HomeAssistant["selectedTheme"]>, | ||||
|   main?: boolean | ||||
| ) => { | ||||
|   // If there is no explicitly desired theme provided, we automatically | ||||
|   // If there is no explicitly desired theme provided, and the element is the main element we automatically | ||||
|   // use the active one from `themes`. | ||||
|   const themeToApply = selectedTheme || themes.theme; | ||||
|   const themeToApply = selectedTheme || (main ? themes.theme : undefined); | ||||
|  | ||||
|   // If there is no explicitly desired dark mode provided, we automatically | ||||
|   // use the active one from `themes`. | ||||
| @@ -47,7 +48,7 @@ export const applyThemesOnElement = ( | ||||
|   let cacheKey = themeToApply; | ||||
|   let themeRules: Partial<ThemeVars> = {}; | ||||
|  | ||||
|   if (darkMode) { | ||||
|   if (themeToApply && darkMode) { | ||||
|     cacheKey = `${cacheKey}__dark`; | ||||
|     themeRules = { ...darkStyles }; | ||||
|   } | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| import { HomeAssistant } from "../../types"; | ||||
| import type { HomeAssistant } from "../../types"; | ||||
|  | ||||
| export const canToggleDomain = (hass: HomeAssistant, domain: string) => { | ||||
|   const services = hass.services[domain]; | ||||
|   | ||||
| @@ -1,14 +1,30 @@ | ||||
| import { HassEntity } from "home-assistant-js-websocket"; | ||||
| import { HomeAssistant } from "../../types"; | ||||
| import type { HassEntity } from "home-assistant-js-websocket"; | ||||
| import type { HomeAssistant } from "../../types"; | ||||
| import { canToggleDomain } from "./can_toggle_domain"; | ||||
| import { computeStateDomain } from "./compute_state_domain"; | ||||
| import { supportsFeature } from "./supports-feature"; | ||||
|  | ||||
| export const canToggleState = (hass: HomeAssistant, stateObj: HassEntity) => { | ||||
|   const domain = computeStateDomain(stateObj); | ||||
|  | ||||
|   if (domain === "group") { | ||||
|     return stateObj.state === "on" || stateObj.state === "off"; | ||||
|     if ( | ||||
|       stateObj.attributes?.entity_id?.some((entity) => { | ||||
|         const entityStateObj = hass.states[entity]; | ||||
|         if (!entityStateObj) { | ||||
|           return false; | ||||
|         } | ||||
|  | ||||
|         const entityDomain = computeStateDomain(entityStateObj); | ||||
|         return canToggleDomain(hass, entityDomain); | ||||
|       }) | ||||
|     ) { | ||||
|       return stateObj.state === "on" || stateObj.state === "off"; | ||||
|     } | ||||
|  | ||||
|     return false; | ||||
|   } | ||||
|  | ||||
|   if (domain === "climate") { | ||||
|     return supportsFeature(stateObj, 4096); | ||||
|   } | ||||
|   | ||||
| @@ -1,12 +1,18 @@ | ||||
| import { HassEntity } from "home-assistant-js-websocket"; | ||||
| import { UNAVAILABLE, UNKNOWN } from "../../data/entity"; | ||||
| import { FrontendLocaleData } from "../../data/translation"; | ||||
| import { | ||||
|   updateIsInstalling, | ||||
|   UpdateEntity, | ||||
|   UPDATE_SUPPORT_PROGRESS, | ||||
| } from "../../data/update"; | ||||
| import { formatDate } from "../datetime/format_date"; | ||||
| import { formatDateTime } from "../datetime/format_date_time"; | ||||
| import { formatTime } from "../datetime/format_time"; | ||||
| import { formatNumber, isNumericState } from "../number/format_number"; | ||||
| import { LocalizeFunc } from "../translations/localize"; | ||||
| import { computeStateDomain } from "./compute_state_domain"; | ||||
| import { supportsFeature } from "./supports-feature"; | ||||
|  | ||||
| export const computeStateDisplay = ( | ||||
|   localize: LocalizeFunc, | ||||
| @@ -123,7 +129,33 @@ export const computeStateDisplay = ( | ||||
|     domain === "scene" || | ||||
|     (domain === "sensor" && stateObj.attributes.device_class === "timestamp") | ||||
|   ) { | ||||
|     return formatDateTime(new Date(compareState), locale); | ||||
|     try { | ||||
|       return formatDateTime(new Date(compareState), locale); | ||||
|     } catch (_err) { | ||||
|       return compareState; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   if (domain === "update") { | ||||
|     // When updating, and entity does not support % show "Installing" | ||||
|     // When updating, and entity does support % show "Installing (xx%)" | ||||
|     // When update available, show the version | ||||
|     // When the latest version is skipped, show the latest version | ||||
|     // When update is not available, show "Up-to-date" | ||||
|     // When update is not available and there is no latest_version show "Unavailable" | ||||
|     return compareState === "on" | ||||
|       ? updateIsInstalling(stateObj as UpdateEntity) | ||||
|         ? supportsFeature(stateObj, UPDATE_SUPPORT_PROGRESS) | ||||
|           ? localize("ui.card.update.installing_with_progress", { | ||||
|               progress: stateObj.attributes.in_progress, | ||||
|             }) | ||||
|           : localize("ui.card.update.installing") | ||||
|         : stateObj.attributes.latest_version | ||||
|       : stateObj.attributes.skipped_version === | ||||
|         stateObj.attributes.latest_version | ||||
|       ? stateObj.attributes.latest_version ?? | ||||
|         localize("state.default.unavailable") | ||||
|       : localize("ui.card.update.up_to_date"); | ||||
|   } | ||||
|  | ||||
|   return ( | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| import { HassEntity } from "home-assistant-js-websocket"; | ||||
| import type { HassEntity } from "home-assistant-js-websocket"; | ||||
| import { computeDomain } from "./compute_domain"; | ||||
|  | ||||
| export const computeStateDomain = (stateObj: HassEntity) => | ||||
|   | ||||
| @@ -120,6 +120,7 @@ export const computeOpenIcon = (stateObj: HassEntity): string => { | ||||
|     case "awning": | ||||
|     case "door": | ||||
|     case "gate": | ||||
|     case "curtain": | ||||
|       return mdiArrowExpandHorizontal; | ||||
|     default: | ||||
|       return mdiArrowUp; | ||||
| @@ -131,6 +132,7 @@ export const computeCloseIcon = (stateObj: HassEntity): string => { | ||||
|     case "awning": | ||||
|     case "door": | ||||
|     case "gate": | ||||
|     case "curtain": | ||||
|       return mdiArrowCollapseHorizontal; | ||||
|     default: | ||||
|       return mdiArrowDown; | ||||
|   | ||||
| @@ -9,11 +9,10 @@ import { | ||||
|   mdiCast, | ||||
|   mdiCastConnected, | ||||
|   mdiClock, | ||||
|   mdiEmoticonDead, | ||||
|   mdiFlash, | ||||
|   mdiGestureTapButton, | ||||
|   mdiLanConnect, | ||||
|   mdiLanDisconnect, | ||||
|   mdiLightSwitch, | ||||
|   mdiLock, | ||||
|   mdiLockAlert, | ||||
|   mdiLockClock, | ||||
| @@ -22,16 +21,16 @@ import { | ||||
|   mdiPowerPlug, | ||||
|   mdiPowerPlugOff, | ||||
|   mdiRestart, | ||||
|   mdiSleep, | ||||
|   mdiTimerSand, | ||||
|   mdiToggleSwitch, | ||||
|   mdiToggleSwitchOff, | ||||
|   mdiCheckCircleOutline, | ||||
|   mdiCloseCircleOutline, | ||||
|   mdiWeatherNight, | ||||
|   mdiZWave, | ||||
|   mdiPackage, | ||||
|   mdiPackageDown, | ||||
| } from "@mdi/js"; | ||||
| import { HassEntity } from "home-assistant-js-websocket"; | ||||
| import { updateIsInstalling, UpdateEntity } from "../../data/update"; | ||||
| /** | ||||
|  * Return the icon to be used for a domain. | ||||
|  * | ||||
| @@ -112,19 +111,7 @@ export const domainIcon = ( | ||||
|         case "switch": | ||||
|           return compareState === "on" ? mdiToggleSwitch : mdiToggleSwitchOff; | ||||
|         default: | ||||
|           return mdiFlash; | ||||
|       } | ||||
|  | ||||
|     case "zwave": | ||||
|       switch (compareState) { | ||||
|         case "dead": | ||||
|           return mdiEmoticonDead; | ||||
|         case "sleeping": | ||||
|           return mdiSleep; | ||||
|         case "initializing": | ||||
|           return mdiTimerSand; | ||||
|         default: | ||||
|           return mdiZWave; | ||||
|           return mdiLightSwitch; | ||||
|       } | ||||
|  | ||||
|     case "sensor": { | ||||
| @@ -149,6 +136,13 @@ export const domainIcon = ( | ||||
|       return stateObj?.state === "above_horizon" | ||||
|         ? FIXED_DOMAIN_ICONS[domain] | ||||
|         : mdiWeatherNight; | ||||
|  | ||||
|     case "update": | ||||
|       return compareState === "on" | ||||
|         ? updateIsInstalling(stateObj as UpdateEntity) | ||||
|           ? mdiPackageDown | ||||
|           : mdiPackageUp | ||||
|         : mdiPackage; | ||||
|   } | ||||
|  | ||||
|   if (domain in FIXED_DOMAIN_ICONS) { | ||||
|   | ||||
| @@ -1,24 +1,32 @@ | ||||
| const SUFFIXES = [" ", ": "]; | ||||
|  | ||||
| /** | ||||
|  * Strips a device name from an entity name. | ||||
|  * @param entityName the entity name | ||||
|  * @param lowerCasedPrefixWithSpaceSuffix the prefix to strip, lower cased with a space suffix | ||||
|  * @param lowerCasedPrefix the prefix to strip, lower cased | ||||
|  * @returns | ||||
|  */ | ||||
| export const stripPrefixFromEntityName = ( | ||||
|   entityName: string, | ||||
|   lowerCasedPrefixWithSpaceSuffix: string | ||||
|   lowerCasedPrefix: string | ||||
| ) => { | ||||
|   if (!entityName.toLowerCase().startsWith(lowerCasedPrefixWithSpaceSuffix)) { | ||||
|     return undefined; | ||||
|   const lowerCasedEntityName = entityName.toLowerCase(); | ||||
|  | ||||
|   for (const suffix of SUFFIXES) { | ||||
|     const lowerCasedPrefixWithSuffix = `${lowerCasedPrefix}${suffix}`; | ||||
|  | ||||
|     if (lowerCasedEntityName.startsWith(lowerCasedPrefixWithSuffix)) { | ||||
|       const newName = entityName.substring(lowerCasedPrefixWithSuffix.length); | ||||
|  | ||||
|       // If first word already has an upper case letter (e.g. from brand name) | ||||
|       // leave as-is, otherwise capitalize the first word. | ||||
|       return hasUpperCase(newName.substr(0, newName.indexOf(" "))) | ||||
|         ? newName | ||||
|         : newName[0].toUpperCase() + newName.slice(1); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   const newName = entityName.substring(lowerCasedPrefixWithSpaceSuffix.length); | ||||
|  | ||||
|   // If first word already has an upper case letter (e.g. from brand name) | ||||
|   // leave as-is, otherwise capitalize the first word. | ||||
|   return hasUpperCase(newName.substr(0, newName.indexOf(" "))) | ||||
|     ? newName | ||||
|     : newName[0].toUpperCase() + newName.slice(1); | ||||
|   return undefined; | ||||
| }; | ||||
|  | ||||
| const hasUpperCase = (str: string): boolean => str.toLowerCase() !== str; | ||||
|   | ||||
							
								
								
									
										4
									
								
								src/common/string/is_ip_address.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								src/common/string/is_ip_address.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| const regexp = | ||||
|   /^(?:(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.){3}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)$/; | ||||
|  | ||||
| export const isIPAddress = (input: string): boolean => regexp.test(input); | ||||
| @@ -15,6 +15,7 @@ export const iconColorCSS = css` | ||||
|   ha-state-icon[data-domain="media_player"][data-state="on"], | ||||
|   ha-state-icon[data-domain="media_player"][data-state="paused"], | ||||
|   ha-state-icon[data-domain="media_player"][data-state="playing"], | ||||
|   ha-state-icon[data-domain="remote"][data-state="on"], | ||||
|   ha-state-icon[data-domain="script"][data-state="on"], | ||||
|   ha-state-icon[data-domain="sun"][data-state="above_horizon"], | ||||
|   ha-state-icon[data-domain="switch"][data-state="on"], | ||||
| @@ -69,9 +70,6 @@ export const iconColorCSS = css` | ||||
|   } | ||||
|  | ||||
|   ha-state-icon[data-domain="plant"][data-state="problem"], | ||||
|   ha-state-icon[data-domain="zwave"][data-state="dead"] { | ||||
|     color: var(--state-icon-error-color); | ||||
|   } | ||||
|  | ||||
|   /* Color the icon if unavailable */ | ||||
|   ha-state-icon[data-state="unavailable"] { | ||||
|   | ||||
| @@ -11,7 +11,7 @@ export const debounce = <T extends any[]>( | ||||
|   immediate = false | ||||
| ) => { | ||||
|   let timeout: number | undefined; | ||||
|   return (...args: T): void => { | ||||
|   const debouncedFunc = (...args: T): void => { | ||||
|     const later = () => { | ||||
|       timeout = undefined; | ||||
|       if (!immediate) { | ||||
| @@ -25,4 +25,8 @@ export const debounce = <T extends any[]>( | ||||
|       func(...args); | ||||
|     } | ||||
|   }; | ||||
|   debouncedFunc.cancel = () => { | ||||
|     clearTimeout(timeout); | ||||
|   }; | ||||
|   return debouncedFunc; | ||||
| }; | ||||
|   | ||||
| @@ -1,8 +1,9 @@ | ||||
| import "@material/mwc-button"; | ||||
| import type { Button } from "@material/mwc-button"; | ||||
| import { mdiAlertOctagram, mdiCheckBold } from "@mdi/js"; | ||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property, query } from "lit/decorators"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import "../ha-circular-progress"; | ||||
| import "../ha-svg-icon"; | ||||
|  | ||||
| @customElement("ha-progress-button") | ||||
| export class HaProgressButton extends LitElement { | ||||
| @@ -12,38 +13,53 @@ export class HaProgressButton extends LitElement { | ||||
|  | ||||
|   @property({ type: Boolean }) public raised = false; | ||||
|  | ||||
|   @query("mwc-button", true) private _button?: Button; | ||||
|   @state() private _result?: "success" | "error"; | ||||
|  | ||||
|   public render(): TemplateResult { | ||||
|     const overlay = this._result || this.progress; | ||||
|     return html` | ||||
|       <mwc-button | ||||
|         ?raised=${this.raised} | ||||
|         .disabled=${this.disabled || this.progress} | ||||
|         @click=${this._buttonTapped} | ||||
|         class=${this._result || ""} | ||||
|       > | ||||
|         <slot></slot> | ||||
|       </mwc-button> | ||||
|       ${this.progress | ||||
|         ? html`<div class="progress"> | ||||
|             <ha-circular-progress size="small" active></ha-circular-progress> | ||||
|           </div>` | ||||
|         : ""} | ||||
|       ${!overlay | ||||
|         ? "" | ||||
|         : html` | ||||
|             <div class="progress"> | ||||
|               ${this._result === "success" | ||||
|                 ? html`<ha-svg-icon .path=${mdiCheckBold}></ha-svg-icon>` | ||||
|                 : this._result === "error" | ||||
|                 ? html`<ha-svg-icon .path=${mdiAlertOctagram}></ha-svg-icon>` | ||||
|                 : this.progress | ||||
|                 ? html` | ||||
|                     <ha-circular-progress | ||||
|                       size="small" | ||||
|                       active | ||||
|                     ></ha-circular-progress> | ||||
|                   ` | ||||
|                 : ""} | ||||
|             </div> | ||||
|           `} | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   public actionSuccess(): void { | ||||
|     this._tempClass("success"); | ||||
|     this._setResult("success"); | ||||
|   } | ||||
|  | ||||
|   public actionError(): void { | ||||
|     this._tempClass("error"); | ||||
|     this._setResult("error"); | ||||
|   } | ||||
|  | ||||
|   private _tempClass(className: string): void { | ||||
|     this._button!.classList.add(className); | ||||
|   private _setResult(result: "success" | "error"): void { | ||||
|     this._result = result; | ||||
|     setTimeout(() => { | ||||
|       this._button!.classList.remove(className); | ||||
|     }, 1000); | ||||
|       this._result = undefined; | ||||
|     }, 2000); | ||||
|   } | ||||
|  | ||||
|   private _buttonTapped(ev: Event): void { | ||||
| @@ -69,6 +85,7 @@ export class HaProgressButton extends LitElement { | ||||
|         background-color: var(--success-color); | ||||
|         transition: none; | ||||
|         border-radius: 4px; | ||||
|         pointer-events: none; | ||||
|       } | ||||
|  | ||||
|       mwc-button[raised].success { | ||||
| @@ -81,6 +98,7 @@ export class HaProgressButton extends LitElement { | ||||
|         background-color: var(--error-color); | ||||
|         transition: none; | ||||
|         border-radius: 4px; | ||||
|         pointer-events: none; | ||||
|       } | ||||
|  | ||||
|       mwc-button[raised].error { | ||||
| @@ -89,13 +107,21 @@ export class HaProgressButton extends LitElement { | ||||
|       } | ||||
|  | ||||
|       .progress { | ||||
|         bottom: 0; | ||||
|         margin-top: 4px; | ||||
|         bottom: 4px; | ||||
|         position: absolute; | ||||
|         text-align: center; | ||||
|         top: 0; | ||||
|         top: 4px; | ||||
|         width: 100%; | ||||
|       } | ||||
|  | ||||
|       ha-svg-icon { | ||||
|         color: white; | ||||
|       } | ||||
|  | ||||
|       mwc-button.success slot, | ||||
|       mwc-button.error slot { | ||||
|         visibility: hidden; | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -21,7 +21,7 @@ import { styleMap } from "lit/directives/style-map"; | ||||
| import memoizeOne from "memoize-one"; | ||||
| import { restoreScroll } from "../../common/decorators/restore-scroll"; | ||||
| import { fireEvent } from "../../common/dom/fire_event"; | ||||
| import "../../common/search/search-input"; | ||||
| import "../search-input"; | ||||
| import { debounce } from "../../common/util/debounce"; | ||||
| import { nextRender } from "../../common/util/render-status"; | ||||
| import { haStyleScrollbar } from "../../resources/styles"; | ||||
|   | ||||
| @@ -115,6 +115,9 @@ class DateRangePickerElement extends WrappedElement { | ||||
|             color: var(--primary-text-color); | ||||
|             min-width: initial !important; | ||||
|           } | ||||
|           .daterangepicker:before { | ||||
|             display: none; | ||||
|           } | ||||
|           .daterangepicker:after { | ||||
|             border-bottom: 6px solid var(--card-background-color); | ||||
|           } | ||||
|   | ||||
| @@ -1,5 +1,4 @@ | ||||
| import "@material/mwc-list/mwc-list-item"; | ||||
| import "@material/mwc-select/mwc-select"; | ||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { property, state } from "lit/decorators"; | ||||
| import { fireEvent } from "../../common/dom/fire_event"; | ||||
| @@ -8,6 +7,7 @@ import { | ||||
|   deviceAutomationsEqual, | ||||
| } from "../../data/device_automation"; | ||||
| import { HomeAssistant } from "../../types"; | ||||
| import "../ha-select"; | ||||
|  | ||||
| const NO_AUTOMATION_KEY = "NO_AUTOMATION"; | ||||
| const UNKNOWN_AUTOMATION_KEY = "UNKNOWN_AUTOMATION"; | ||||
| @@ -90,7 +90,7 @@ export abstract class HaDeviceAutomationPicker< | ||||
|     } | ||||
|     const value = this._value; | ||||
|     return html` | ||||
|       <mwc-select | ||||
|       <ha-select | ||||
|         .label=${this.label} | ||||
|         .value=${value} | ||||
|         @selected=${this._automationChanged} | ||||
| @@ -113,7 +113,7 @@ export abstract class HaDeviceAutomationPicker< | ||||
|             </mwc-list-item> | ||||
|           ` | ||||
|         )} | ||||
|       </mwc-select> | ||||
|       </ha-select> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
| @@ -167,7 +167,7 @@ export abstract class HaDeviceAutomationPicker< | ||||
|  | ||||
|   static get styles(): CSSResultGroup { | ||||
|     return css` | ||||
|       mwc-select { | ||||
|       ha-select { | ||||
|         width: 100%; | ||||
|         margin-top: 4px; | ||||
|       } | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| import { html, LitElement, TemplateResult } from "lit"; | ||||
| import { css, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import { fireEvent } from "../../common/dom/fire_event"; | ||||
| import { PolymerChangedEvent } from "../../polymer-types"; | ||||
| @@ -116,6 +116,12 @@ class HaDevicesPicker extends LitElement { | ||||
|  | ||||
|     this._updateDevices([...currentDevices, toAdd]); | ||||
|   } | ||||
|  | ||||
|   static override styles = css` | ||||
|     div { | ||||
|       margin-top: 8px; | ||||
|     } | ||||
|   `; | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   | ||||
| @@ -46,11 +46,29 @@ class HaEntitiesPickerLight extends LitElement { | ||||
|   @property({ type: Array, attribute: "include-unit-of-measurement" }) | ||||
|   public includeUnitOfMeasurement?: string[]; | ||||
|  | ||||
|   /** | ||||
|    * List of allowed entities to show. Will ignore all other filters. | ||||
|    * @type {Array} | ||||
|    * @attr include-entities | ||||
|    */ | ||||
|   @property({ type: Array, attribute: "include-entities" }) | ||||
|   public includeEntities?: string[]; | ||||
|  | ||||
|   /** | ||||
|    * List of entities to be excluded. | ||||
|    * @type {Array} | ||||
|    * @attr exclude-entities | ||||
|    */ | ||||
|   @property({ type: Array, attribute: "exclude-entities" }) | ||||
|   public excludeEntities?: string[]; | ||||
|  | ||||
|   @property({ attribute: "picked-entity-label" }) | ||||
|   public pickedEntityLabel?: string; | ||||
|  | ||||
|   @property({ attribute: "pick-entity-label" }) public pickEntityLabel?: string; | ||||
|  | ||||
|   @property() public entityFilter?: HaEntityPickerEntityFilterFunc; | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     if (!this.hass) { | ||||
|       return html``; | ||||
| @@ -67,6 +85,8 @@ class HaEntitiesPickerLight extends LitElement { | ||||
|               .hass=${this.hass} | ||||
|               .includeDomains=${this.includeDomains} | ||||
|               .excludeDomains=${this.excludeDomains} | ||||
|               .includeEntities=${this.includeEntities} | ||||
|               .excludeEntities=${this.excludeEntities} | ||||
|               .includeDeviceClasses=${this.includeDeviceClasses} | ||||
|               .includeUnitOfMeasurement=${this.includeUnitOfMeasurement} | ||||
|               .entityFilter=${this._entityFilter} | ||||
| @@ -82,6 +102,8 @@ class HaEntitiesPickerLight extends LitElement { | ||||
|           .hass=${this.hass} | ||||
|           .includeDomains=${this.includeDomains} | ||||
|           .excludeDomains=${this.excludeDomains} | ||||
|           .includeEntities=${this.includeEntities} | ||||
|           .excludeEntities=${this.excludeEntities} | ||||
|           .includeDeviceClasses=${this.includeDeviceClasses} | ||||
|           .includeUnitOfMeasurement=${this.includeUnitOfMeasurement} | ||||
|           .entityFilter=${this._entityFilter} | ||||
| @@ -94,7 +116,9 @@ class HaEntitiesPickerLight extends LitElement { | ||||
|  | ||||
|   private _entityFilter: HaEntityPickerEntityFilterFunc = ( | ||||
|     stateObj: HassEntity | ||||
|   ) => !this.value || !this.value.includes(stateObj.entity_id); | ||||
|   ) => | ||||
|     (!this.value || !this.value.includes(stateObj.entity_id)) && | ||||
|     (!this.entityFilter || this.entityFilter(stateObj)); | ||||
|  | ||||
|   private get _currentEntities() { | ||||
|     return this.value || []; | ||||
| @@ -114,7 +138,7 @@ class HaEntitiesPickerLight extends LitElement { | ||||
|     const newValue = event.detail.value; | ||||
|     if ( | ||||
|       newValue === curValue || | ||||
|       (newValue !== "" && !isValidEntityId(newValue)) | ||||
|       (newValue !== undefined && !isValidEntityId(newValue)) | ||||
|     ) { | ||||
|       return; | ||||
|     } | ||||
| @@ -147,7 +171,7 @@ class HaEntitiesPickerLight extends LitElement { | ||||
|   } | ||||
|  | ||||
|   static override styles = css` | ||||
|     ha-entity-picker { | ||||
|     div { | ||||
|       margin-top: 8px; | ||||
|     } | ||||
|   `; | ||||
|   | ||||
| @@ -7,6 +7,7 @@ import memoizeOne from "memoize-one"; | ||||
| import { fireEvent } from "../../common/dom/fire_event"; | ||||
| import { computeDomain } from "../../common/entity/compute_domain"; | ||||
| import { computeStateName } from "../../common/entity/compute_state_name"; | ||||
| import { caseInsensitiveStringCompare } from "../../common/string/compare"; | ||||
| import { PolymerChangedEvent } from "../../polymer-types"; | ||||
| import { HomeAssistant } from "../../types"; | ||||
| import "../ha-combo-box"; | ||||
| @@ -15,18 +16,21 @@ import "../ha-icon-button"; | ||||
| import "../ha-svg-icon"; | ||||
| import "./state-badge"; | ||||
|  | ||||
| interface HassEntityWithCachedName extends HassEntity { | ||||
|   friendly_name: string; | ||||
| } | ||||
|  | ||||
| export type HaEntityPickerEntityFilterFunc = (entityId: HassEntity) => boolean; | ||||
|  | ||||
| // eslint-disable-next-line lit/prefer-static-styles | ||||
| const rowRenderer: ComboBoxLitRenderer<HassEntity & { friendly_name: string }> = | ||||
|   (item) => | ||||
|     html`<mwc-list-item graphic="avatar" .twoline=${!!item.entity_id}> | ||||
|       ${item.state | ||||
|         ? html`<state-badge slot="graphic" .stateObj=${item}></state-badge>` | ||||
|         : ""} | ||||
|       <span>${item.friendly_name}</span> | ||||
|       <span slot="secondary">${item.entity_id}</span> | ||||
|     </mwc-list-item>`; | ||||
| const rowRenderer: ComboBoxLitRenderer<HassEntityWithCachedName> = (item) => | ||||
|   html`<mwc-list-item graphic="avatar" .twoline=${!!item.entity_id}> | ||||
|     ${item.state | ||||
|       ? html`<state-badge slot="graphic" .stateObj=${item}></state-badge>` | ||||
|       : ""} | ||||
|     <span>${item.friendly_name}</span> | ||||
|     <span slot="secondary">${item.entity_id}</span> | ||||
|   </mwc-list-item>`; | ||||
| @customElement("ha-entity-picker") | ||||
| export class HaEntityPicker extends LitElement { | ||||
|   @property({ attribute: false }) public hass!: HomeAssistant; | ||||
| @@ -74,6 +78,22 @@ export class HaEntityPicker extends LitElement { | ||||
|   @property({ type: Array, attribute: "include-unit-of-measurement" }) | ||||
|   public includeUnitOfMeasurement?: string[]; | ||||
|  | ||||
|   /** | ||||
|    * List of allowed entities to show. Will ignore all other filters. | ||||
|    * @type {Array} | ||||
|    * @attr include-entities | ||||
|    */ | ||||
|   @property({ type: Array, attribute: "include-entities" }) | ||||
|   public includeEntities?: string[]; | ||||
|  | ||||
|   /** | ||||
|    * List of entities to be excluded. | ||||
|    * @type {Array} | ||||
|    * @attr exclude-entities | ||||
|    */ | ||||
|   @property({ type: Array, attribute: "exclude-entities" }) | ||||
|   public excludeEntities?: string[]; | ||||
|  | ||||
|   @property() public entityFilter?: HaEntityPickerEntityFilterFunc; | ||||
|  | ||||
|   @property({ type: Boolean }) public hideClearIcon = false; | ||||
| @@ -96,7 +116,7 @@ export class HaEntityPicker extends LitElement { | ||||
|  | ||||
|   private _initedStates = false; | ||||
|  | ||||
|   private _states: HassEntity[] = []; | ||||
|   private _states: HassEntityWithCachedName[] = []; | ||||
|  | ||||
|   private _getStates = memoizeOne( | ||||
|     ( | ||||
| @@ -106,9 +126,11 @@ export class HaEntityPicker extends LitElement { | ||||
|       excludeDomains: this["excludeDomains"], | ||||
|       entityFilter: this["entityFilter"], | ||||
|       includeDeviceClasses: this["includeDeviceClasses"], | ||||
|       includeUnitOfMeasurement: this["includeUnitOfMeasurement"] | ||||
|     ) => { | ||||
|       let states: HassEntity[] = []; | ||||
|       includeUnitOfMeasurement: this["includeUnitOfMeasurement"], | ||||
|       includeEntities: this["includeEntities"], | ||||
|       excludeEntities: this["excludeEntities"] | ||||
|     ): HassEntityWithCachedName[] => { | ||||
|       let states: HassEntityWithCachedName[] = []; | ||||
|  | ||||
|       if (!hass) { | ||||
|         return []; | ||||
| @@ -122,7 +144,7 @@ export class HaEntityPicker extends LitElement { | ||||
|             state: "", | ||||
|             last_changed: "", | ||||
|             last_updated: "", | ||||
|             context: { id: "", user_id: null }, | ||||
|             context: { id: "", user_id: null, parent_id: null }, | ||||
|             friendly_name: this.hass!.localize( | ||||
|               "ui.components.entity.entity-picker.no_entities" | ||||
|             ), | ||||
| @@ -136,6 +158,30 @@ export class HaEntityPicker extends LitElement { | ||||
|         ]; | ||||
|       } | ||||
|  | ||||
|       if (includeEntities) { | ||||
|         entityIds = entityIds.filter((entityId) => | ||||
|           this.includeEntities!.includes(entityId) | ||||
|         ); | ||||
|  | ||||
|         return entityIds | ||||
|           .map((key) => ({ | ||||
|             ...hass!.states[key], | ||||
|             friendly_name: computeStateName(hass!.states[key]) || key, | ||||
|           })) | ||||
|           .sort((entityA, entityB) => | ||||
|             caseInsensitiveStringCompare( | ||||
|               entityA.friendly_name, | ||||
|               entityB.friendly_name | ||||
|             ) | ||||
|           ); | ||||
|       } | ||||
|  | ||||
|       if (excludeEntities) { | ||||
|         entityIds = entityIds.filter( | ||||
|           (entityId) => !excludeEntities!.includes(entityId) | ||||
|         ); | ||||
|       } | ||||
|  | ||||
|       if (includeDomains) { | ||||
|         entityIds = entityIds.filter((eid) => | ||||
|           includeDomains.includes(computeDomain(eid)) | ||||
| @@ -148,10 +194,17 @@ export class HaEntityPicker extends LitElement { | ||||
|         ); | ||||
|       } | ||||
|  | ||||
|       states = entityIds.sort().map((key) => ({ | ||||
|         ...hass!.states[key], | ||||
|         friendly_name: computeStateName(hass!.states[key]) || key, | ||||
|       })); | ||||
|       states = entityIds | ||||
|         .map((key) => ({ | ||||
|           ...hass!.states[key], | ||||
|           friendly_name: computeStateName(hass!.states[key]) || key, | ||||
|         })) | ||||
|         .sort((entityA, entityB) => | ||||
|           caseInsensitiveStringCompare( | ||||
|             entityA.friendly_name, | ||||
|             entityB.friendly_name | ||||
|           ) | ||||
|         ); | ||||
|  | ||||
|       if (includeDeviceClasses) { | ||||
|         states = states.filter( | ||||
| @@ -190,7 +243,7 @@ export class HaEntityPicker extends LitElement { | ||||
|             state: "", | ||||
|             last_changed: "", | ||||
|             last_updated: "", | ||||
|             context: { id: "", user_id: null }, | ||||
|             context: { id: "", user_id: null, parent_id: null }, | ||||
|             friendly_name: this.hass!.localize( | ||||
|               "ui.components.entity.entity-picker.no_match" | ||||
|             ), | ||||
| @@ -228,7 +281,9 @@ export class HaEntityPicker extends LitElement { | ||||
|         this.excludeDomains, | ||||
|         this.entityFilter, | ||||
|         this.includeDeviceClasses, | ||||
|         this.includeUnitOfMeasurement | ||||
|         this.includeUnitOfMeasurement, | ||||
|         this.includeEntities, | ||||
|         this.excludeEntities | ||||
|       ); | ||||
|       if (this._initedStates) { | ||||
|         (this.comboBox as any).filteredItems = this._states; | ||||
|   | ||||
| @@ -1,6 +1,3 @@ | ||||
| import "@polymer/paper-input/paper-input"; | ||||
| import "@polymer/paper-item/paper-icon-item"; | ||||
| import "@polymer/paper-item/paper-item-body"; | ||||
| import { HassEntity } from "home-assistant-js-websocket"; | ||||
| import { html, LitElement, PropertyValues, TemplateResult } from "lit"; | ||||
| import { ComboBoxLitRenderer } from "lit-vaadin-helpers"; | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| import { LitElement, html, TemplateResult, css } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import "@material/mwc-select/mwc-select"; | ||||
| import "./ha-select"; | ||||
| import "@material/mwc-list/mwc-list-item"; | ||||
| import "./ha-textfield"; | ||||
| import { fireEvent } from "../common/dom/fire_event"; | ||||
| @@ -193,7 +193,7 @@ export class HaBaseTimeInput extends LitElement { | ||||
|           : ""} | ||||
|         ${this.format === 24 | ||||
|           ? "" | ||||
|           : html`<mwc-select | ||||
|           : html`<ha-select | ||||
|               .required=${this.required} | ||||
|               .value=${this.amPm} | ||||
|               .disabled=${this.disabled} | ||||
| @@ -205,7 +205,7 @@ export class HaBaseTimeInput extends LitElement { | ||||
|             > | ||||
|               <mwc-list-item value="AM">AM</mwc-list-item> | ||||
|               <mwc-list-item value="PM">PM</mwc-list-item> | ||||
|             </mwc-select>`} | ||||
|             </ha-select>`} | ||||
|       </div> | ||||
|     `; | ||||
|   } | ||||
| @@ -280,7 +280,7 @@ export class HaBaseTimeInput extends LitElement { | ||||
|     ha-textfield:last-child { | ||||
|       --text-field-border-top-right-radius: var(--mdc-shape-medium); | ||||
|     } | ||||
|     mwc-select { | ||||
|     ha-select { | ||||
|       --mdc-shape-small: 0; | ||||
|       width: 85px; | ||||
|     } | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import "@material/mwc-list/mwc-list-item"; | ||||
| import "@material/mwc-select/mwc-select"; | ||||
| import "./ha-select"; | ||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import memoizeOne from "memoize-one"; | ||||
| @@ -24,7 +24,7 @@ class HaBluePrintPicker extends LitElement { | ||||
|   @property({ type: Boolean }) public disabled = false; | ||||
|  | ||||
|   public open() { | ||||
|     const select = this.shadowRoot?.querySelector("mwc-select"); | ||||
|     const select = this.shadowRoot?.querySelector("ha-select"); | ||||
|     if (select) { | ||||
|       // @ts-expect-error | ||||
|       select.menuOpen = true; | ||||
| @@ -49,7 +49,7 @@ class HaBluePrintPicker extends LitElement { | ||||
|       return html``; | ||||
|     } | ||||
|     return html` | ||||
|       <mwc-select | ||||
|       <ha-select | ||||
|         .label=${this.label || | ||||
|         this.hass.localize("ui.components.blueprint-picker.label")} | ||||
|         fixedMenuPosition | ||||
| @@ -71,7 +71,7 @@ class HaBluePrintPicker extends LitElement { | ||||
|             </mwc-list-item> | ||||
|           ` | ||||
|         )} | ||||
|       </mwc-select> | ||||
|       </ha-select> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
| @@ -101,7 +101,7 @@ class HaBluePrintPicker extends LitElement { | ||||
|       :host { | ||||
|         display: inline-block; | ||||
|       } | ||||
|       mwc-select { | ||||
|       ha-select { | ||||
|         width: 100%; | ||||
|         min-width: 200px; | ||||
|         display: block; | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import "@material/mwc-menu"; | ||||
| import type { Corner, Menu } from "@material/mwc-menu"; | ||||
| import type { Corner, Menu, MenuCorner } from "@material/mwc-menu"; | ||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property, query } from "lit/decorators"; | ||||
|  | ||||
| @@ -7,6 +7,12 @@ import { customElement, property, query } from "lit/decorators"; | ||||
| export class HaButtonMenu extends LitElement { | ||||
|   @property() public corner: Corner = "TOP_START"; | ||||
|  | ||||
|   @property() public menuCorner: MenuCorner = "START"; | ||||
|  | ||||
|   @property({ type: Number }) public x?: number; | ||||
|  | ||||
|   @property({ type: Number }) public y?: number; | ||||
|  | ||||
|   @property({ type: Boolean }) public multi = false; | ||||
|  | ||||
|   @property({ type: Boolean }) public activatable = false; | ||||
| @@ -32,9 +38,12 @@ export class HaButtonMenu extends LitElement { | ||||
|       </div> | ||||
|       <mwc-menu | ||||
|         .corner=${this.corner} | ||||
|         .menuCorner=${this.menuCorner} | ||||
|         .fixed=${this.fixed} | ||||
|         .multi=${this.multi} | ||||
|         .activatable=${this.activatable} | ||||
|         .y=${this.y} | ||||
|         .x=${this.x} | ||||
|       > | ||||
|         <slot></slot> | ||||
|       </mwc-menu> | ||||
|   | ||||
| @@ -4,6 +4,7 @@ import { mdiFilterVariant } from "@mdi/js"; | ||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import { fireEvent } from "../common/dom/fire_event"; | ||||
| import { stopPropagation } from "../common/dom/stop_propagation"; | ||||
| import { computeStateName } from "../common/entity/compute_state_name"; | ||||
| import { computeDeviceName } from "../data/device_registry"; | ||||
| import { findRelated, RelatedResult } from "../data/search"; | ||||
| @@ -65,6 +66,7 @@ export class HaRelatedFilterButtonMenu extends LitElement { | ||||
|         .fullwidth=${this.narrow} | ||||
|         .corner=${this.corner} | ||||
|         @closed=${this._onClosed} | ||||
|         @input=${stopPropagation} | ||||
|       > | ||||
|         <ha-area-picker | ||||
|           .label=${this.hass.localize( | ||||
| @@ -74,6 +76,7 @@ export class HaRelatedFilterButtonMenu extends LitElement { | ||||
|           .value=${this.value?.area} | ||||
|           no-add | ||||
|           @value-changed=${this._areaPicked} | ||||
|           @click=${this._preventDefault} | ||||
|         ></ha-area-picker> | ||||
|         <ha-device-picker | ||||
|           .label=${this.hass.localize( | ||||
| @@ -82,6 +85,7 @@ export class HaRelatedFilterButtonMenu extends LitElement { | ||||
|           .hass=${this.hass} | ||||
|           .value=${this.value?.device} | ||||
|           @value-changed=${this._devicePicked} | ||||
|           @click=${this._preventDefault} | ||||
|         ></ha-device-picker> | ||||
|         <ha-entity-picker | ||||
|           .label=${this.hass.localize( | ||||
| @@ -91,6 +95,7 @@ export class HaRelatedFilterButtonMenu extends LitElement { | ||||
|           .value=${this.value?.entity} | ||||
|           .excludeDomains=${this.excludeDomains} | ||||
|           @value-changed=${this._entityPicked} | ||||
|           @click=${this._preventDefault} | ||||
|         ></ha-entity-picker> | ||||
|       </mwc-menu-surface> | ||||
|     `; | ||||
| @@ -103,11 +108,17 @@ export class HaRelatedFilterButtonMenu extends LitElement { | ||||
|     this._open = true; | ||||
|   } | ||||
|  | ||||
|   private _onClosed(): void { | ||||
|   private _onClosed(ev): void { | ||||
|     ev.stopPropagation(); | ||||
|     this._open = false; | ||||
|   } | ||||
|  | ||||
|   private _preventDefault(ev) { | ||||
|     ev.preventDefault(); | ||||
|   } | ||||
|  | ||||
|   private async _entityPicked(ev: CustomEvent) { | ||||
|     ev.stopPropagation(); | ||||
|     const entityId = ev.detail.value; | ||||
|     if (!entityId) { | ||||
|       fireEvent(this, "related-changed", { value: undefined }); | ||||
| @@ -127,6 +138,7 @@ export class HaRelatedFilterButtonMenu extends LitElement { | ||||
|   } | ||||
|  | ||||
|   private async _devicePicked(ev: CustomEvent) { | ||||
|     ev.stopPropagation(); | ||||
|     const deviceId = ev.detail.value; | ||||
|     if (!deviceId) { | ||||
|       fireEvent(this, "related-changed", { value: undefined }); | ||||
| @@ -150,6 +162,7 @@ export class HaRelatedFilterButtonMenu extends LitElement { | ||||
|   } | ||||
|  | ||||
|   private async _areaPicked(ev: CustomEvent) { | ||||
|     ev.stopPropagation(); | ||||
|     const areaId = ev.detail.value; | ||||
|     if (!areaId) { | ||||
|       fireEvent(this, "related-changed", { value: undefined }); | ||||
| @@ -173,9 +186,7 @@ export class HaRelatedFilterButtonMenu extends LitElement { | ||||
|       :host { | ||||
|         display: inline-block; | ||||
|         position: relative; | ||||
|       } | ||||
|       :host([narrow]) { | ||||
|         position: static; | ||||
|         --mdc-menu-min-width: 250px; | ||||
|       } | ||||
|       ha-area-picker, | ||||
|       ha-device-picker, | ||||
| @@ -185,8 +196,15 @@ export class HaRelatedFilterButtonMenu extends LitElement { | ||||
|         padding: 4px 16px; | ||||
|         box-sizing: border-box; | ||||
|       } | ||||
|       ha-area-picker { | ||||
|         padding-top: 16px; | ||||
|       } | ||||
|       ha-entity-picker { | ||||
|         padding-bottom: 16px; | ||||
|       } | ||||
|       :host([narrow]) ha-area-picker, | ||||
|       :host([narrow]) ha-device-picker { | ||||
|       :host([narrow]) ha-device-picker, | ||||
|       :host([narrow]) ha-entity-picker { | ||||
|         width: 100%; | ||||
|       } | ||||
|     `; | ||||
|   | ||||
| @@ -1,11 +1,11 @@ | ||||
| import { mdiCalendar } from "@mdi/js"; | ||||
| import "@polymer/paper-input/paper-input"; | ||||
| import { css, CSSResultGroup, html, LitElement } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import { formatDateNumeric } from "../common/datetime/format_date"; | ||||
| import { fireEvent } from "../common/dom/fire_event"; | ||||
| import { HomeAssistant } from "../types"; | ||||
| import "./ha-svg-icon"; | ||||
| import "./ha-textfield"; | ||||
|  | ||||
| const loadDatePickerDialog = () => import("./ha-dialog-date-picker"); | ||||
|  | ||||
| @@ -38,17 +38,17 @@ export class HaDateInput extends LitElement { | ||||
|   @property() public label?: string; | ||||
|  | ||||
|   render() { | ||||
|     return html`<paper-input | ||||
|     return html`<ha-textfield | ||||
|       .label=${this.label} | ||||
|       .disabled=${this.disabled} | ||||
|       no-label-float | ||||
|       iconTrailing | ||||
|       @click=${this._openDialog} | ||||
|       .value=${this.value | ||||
|         ? formatDateNumeric(new Date(this.value), this.locale) | ||||
|         : ""} | ||||
|     > | ||||
|       <ha-svg-icon slot="suffix" .path=${mdiCalendar}></ha-svg-icon> | ||||
|     </paper-input>`; | ||||
|       <ha-svg-icon slot="trailingIcon" .path=${mdiCalendar}></ha-svg-icon> | ||||
|     </ha-textfield>`; | ||||
|   } | ||||
|  | ||||
|   private _openDialog() { | ||||
| @@ -73,9 +73,6 @@ export class HaDateInput extends LitElement { | ||||
|  | ||||
|   static get styles(): CSSResultGroup { | ||||
|     return css` | ||||
|       paper-input { | ||||
|         width: 110px; | ||||
|       } | ||||
|       ha-svg-icon { | ||||
|         color: var(--secondary-text-color); | ||||
|       } | ||||
|   | ||||
| @@ -3,7 +3,6 @@ import "@material/mwc-list/mwc-list"; | ||||
| import { ActionDetail } from "@material/mwc-list/mwc-list-foundation"; | ||||
| import "@material/mwc-list/mwc-list-item"; | ||||
| import { mdiCalendar } from "@mdi/js"; | ||||
| import "@polymer/paper-input/paper-input"; | ||||
| import { | ||||
|   css, | ||||
|   CSSResultGroup, | ||||
| @@ -19,6 +18,7 @@ import { computeRTLDirection } from "../common/util/compute_rtl"; | ||||
| import { HomeAssistant } from "../types"; | ||||
| import "./date-range-picker"; | ||||
| import "./ha-svg-icon"; | ||||
| import "./ha-textfield"; | ||||
|  | ||||
| export interface DateRangePickerRanges { | ||||
|   [key: string]: [Date, Date]; | ||||
| @@ -61,7 +61,7 @@ export class HaDateRangePicker extends LitElement { | ||||
|       > | ||||
|         <div slot="input" class="date-range-inputs"> | ||||
|           <ha-svg-icon .path=${mdiCalendar}></ha-svg-icon> | ||||
|           <paper-input | ||||
|           <ha-textfield | ||||
|             .value=${formatDateTime(this.startDate, this.hass.locale)} | ||||
|             .label=${this.hass.localize( | ||||
|               "ui.components.date-range-picker.start_date" | ||||
| @@ -69,16 +69,16 @@ export class HaDateRangePicker extends LitElement { | ||||
|             .disabled=${this.disabled} | ||||
|             @click=${this._handleInputClick} | ||||
|             readonly | ||||
|           ></paper-input> | ||||
|           <paper-input | ||||
|           ></ha-textfield> | ||||
|           <ha-textfield | ||||
|             .value=${formatDateTime(this.endDate, this.hass.locale)} | ||||
|             label=${this.hass.localize( | ||||
|             .label=${this.hass.localize( | ||||
|               "ui.components.date-range-picker.end_date" | ||||
|             )} | ||||
|             .disabled=${this.disabled} | ||||
|             @click=${this._handleInputClick} | ||||
|             readonly | ||||
|           ></paper-input> | ||||
|           ></ha-textfield> | ||||
|         </div> | ||||
|         ${this.ranges | ||||
|           ? html`<div | ||||
| @@ -158,13 +158,13 @@ export class HaDateRangePicker extends LitElement { | ||||
|         border-top: 1px solid var(--divider-color); | ||||
|       } | ||||
|  | ||||
|       paper-input { | ||||
|       ha-textfield { | ||||
|         display: inline-block; | ||||
|         max-width: 250px; | ||||
|         min-width: 200px; | ||||
|       } | ||||
|  | ||||
|       paper-input:last-child { | ||||
|       ha-textfield:last-child { | ||||
|         margin-left: 8px; | ||||
|       } | ||||
|  | ||||
| @@ -176,7 +176,7 @@ export class HaDateRangePicker extends LitElement { | ||||
|       } | ||||
|  | ||||
|       @media only screen and (max-width: 500px) { | ||||
|         paper-input { | ||||
|         ha-textfield { | ||||
|           min-width: inherit; | ||||
|         } | ||||
|  | ||||
|   | ||||
| @@ -1,6 +1,13 @@ | ||||
| import { mdiChevronDown } from "@mdi/js"; | ||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property, query } from "lit/decorators"; | ||||
| import { | ||||
|   css, | ||||
|   CSSResultGroup, | ||||
|   html, | ||||
|   LitElement, | ||||
|   PropertyValues, | ||||
|   TemplateResult, | ||||
| } from "lit"; | ||||
| import { customElement, property, query, state } from "lit/decorators"; | ||||
| import { classMap } from "lit/directives/class-map"; | ||||
| import { fireEvent } from "../common/dom/fire_event"; | ||||
| import { nextRender } from "../common/util/render-status"; | ||||
| @@ -16,11 +23,21 @@ class HaExpansionPanel extends LitElement { | ||||
|  | ||||
|   @property() secondary?: string; | ||||
|  | ||||
|   @state() _showContent = this.expanded; | ||||
|  | ||||
|   @query(".container") private _container!: HTMLDivElement; | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
|       <div class="summary" @click=${this._toggleContainer}> | ||||
|       <div | ||||
|         id="summary" | ||||
|         @click=${this._toggleContainer} | ||||
|         @keydown=${this._toggleContainer} | ||||
|         role="button" | ||||
|         tabindex="0" | ||||
|         aria-expanded=${this.expanded} | ||||
|         aria-controls="sect1" | ||||
|       > | ||||
|         <slot class="header" name="header"> | ||||
|           ${this.header} | ||||
|           <slot class="secondary" name="secondary">${this.secondary}</slot> | ||||
| @@ -33,21 +50,37 @@ class HaExpansionPanel extends LitElement { | ||||
|       <div | ||||
|         class="container ${classMap({ expanded: this.expanded })}" | ||||
|         @transitionend=${this._handleTransitionEnd} | ||||
|         role="region" | ||||
|         aria-labelledby="summary" | ||||
|         aria-hidden=${!this.expanded} | ||||
|         tabindex="-1" | ||||
|       > | ||||
|         <slot></slot> | ||||
|         ${this._showContent ? html`<slot></slot>` : ""} | ||||
|       </div> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   private _handleTransitionEnd() { | ||||
|     this._container.style.removeProperty("height"); | ||||
|   protected willUpdate(changedProps: PropertyValues) { | ||||
|     if (changedProps.has("expanded") && this.expanded) { | ||||
|       this._showContent = this.expanded; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   private async _toggleContainer(): Promise<void> { | ||||
|   private _handleTransitionEnd() { | ||||
|     this._container.style.removeProperty("height"); | ||||
|     this._showContent = this.expanded; | ||||
|   } | ||||
|  | ||||
|   private async _toggleContainer(ev): Promise<void> { | ||||
|     if (ev.type === "keydown" && ev.key !== "Enter" && ev.key !== " ") { | ||||
|       return; | ||||
|     } | ||||
|     ev.preventDefault(); | ||||
|     const newExpanded = !this.expanded; | ||||
|     fireEvent(this, "expanded-will-change", { expanded: newExpanded }); | ||||
|  | ||||
|     if (newExpanded) { | ||||
|       this._showContent = true; | ||||
|       // allow for dynamic content to be rendered | ||||
|       await nextRender(); | ||||
|     } | ||||
| @@ -80,17 +113,21 @@ class HaExpansionPanel extends LitElement { | ||||
|           var(--divider-color, #e0e0e0) | ||||
|         ); | ||||
|         border-radius: var(--ha-card-border-radius, 4px); | ||||
|         padding: 0 8px; | ||||
|       } | ||||
|  | ||||
|       .summary { | ||||
|       #summary { | ||||
|         display: flex; | ||||
|         padding: var(--expansion-panel-summary-padding, 0); | ||||
|         padding: var(--expansion-panel-summary-padding, 0 8px); | ||||
|         min-height: 48px; | ||||
|         align-items: center; | ||||
|         cursor: pointer; | ||||
|         overflow: hidden; | ||||
|         font-weight: 500; | ||||
|         outline: none; | ||||
|       } | ||||
|  | ||||
|       #summary:focus { | ||||
|         background: var(--input-fill-color); | ||||
|       } | ||||
|  | ||||
|       .summary-icon { | ||||
| @@ -103,6 +140,7 @@ class HaExpansionPanel extends LitElement { | ||||
|       } | ||||
|  | ||||
|       .container { | ||||
|         padding: var(--expansion-panel-content-padding, 0 8px); | ||||
|         overflow: hidden; | ||||
|         transition: height 300ms cubic-bezier(0.4, 0, 0.2, 1); | ||||
|         height: 0px; | ||||
|   | ||||
| @@ -1,6 +1,5 @@ | ||||
| import { styles } from "@material/mwc-textfield/mwc-textfield.css"; | ||||
| import { mdiClose } from "@mdi/js"; | ||||
| import "@polymer/iron-input/iron-input"; | ||||
| import "@polymer/paper-input/paper-input-container"; | ||||
| import { css, html, LitElement, PropertyValues, TemplateResult } from "lit"; | ||||
| import { customElement, property, query, state } from "lit/decorators"; | ||||
| import { classMap } from "lit/directives/class-map"; | ||||
| @@ -21,7 +20,7 @@ export class HaFileUpload extends LitElement { | ||||
|  | ||||
|   @property() public accept!: string; | ||||
|  | ||||
|   @property() public icon!: string; | ||||
|   @property() public icon?: string; | ||||
|  | ||||
|   @property() public label!: string; | ||||
|  | ||||
| @@ -39,15 +38,7 @@ export class HaFileUpload extends LitElement { | ||||
|   protected firstUpdated(changedProperties: PropertyValues) { | ||||
|     super.firstUpdated(changedProperties); | ||||
|     if (this.autoOpenFileDialog) { | ||||
|       this._input?.click(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   protected updated(changedProperties: PropertyValues) { | ||||
|     if (changedProperties.has("_drag") && !this.uploading) { | ||||
|       ( | ||||
|         this.shadowRoot!.querySelector("paper-input-container") as any | ||||
|       )._setFocused(this._drag); | ||||
|       this._openFilePicker(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
| @@ -60,51 +51,75 @@ export class HaFileUpload extends LitElement { | ||||
|             active | ||||
|           ></ha-circular-progress>` | ||||
|         : html` | ||||
|             <label for="input"> | ||||
|               <paper-input-container | ||||
|                 .alwaysFloatLabel=${Boolean(this.value)} | ||||
|                 @drop=${this._handleDrop} | ||||
|                 @dragenter=${this._handleDragStart} | ||||
|                 @dragover=${this._handleDragStart} | ||||
|                 @dragleave=${this._handleDragEnd} | ||||
|                 @dragend=${this._handleDragEnd} | ||||
|                 class=${classMap({ | ||||
|                   dragged: this._drag, | ||||
|                 })} | ||||
|             <label | ||||
|               for="input" | ||||
|               class="mdc-text-field mdc-text-field--filled ${classMap({ | ||||
|                 "mdc-text-field--focused": this._drag, | ||||
|                 "mdc-text-field--with-leading-icon": Boolean(this.icon), | ||||
|                 "mdc-text-field--with-trailing-icon": Boolean(this.value), | ||||
|               })}" | ||||
|               @drop=${this._handleDrop} | ||||
|               @dragenter=${this._handleDragStart} | ||||
|               @dragover=${this._handleDragStart} | ||||
|               @dragleave=${this._handleDragEnd} | ||||
|               @dragend=${this._handleDragEnd} | ||||
|             > | ||||
|               <span class="mdc-text-field__ripple"></span> | ||||
|               <span | ||||
|                 class="mdc-floating-label ${this.value || this._drag | ||||
|                   ? "mdc-floating-label--float-above" | ||||
|                   : ""}" | ||||
|                 id="label" | ||||
|                 >${this.label}</span | ||||
|               > | ||||
|                 <label for="input" slot="label"> ${this.label} </label> | ||||
|                 <iron-input slot="input"> | ||||
|                   <input | ||||
|                     id="input" | ||||
|                     type="file" | ||||
|                     class="file" | ||||
|                     accept=${this.accept} | ||||
|                     @change=${this._handleFilePicked} | ||||
|                   /> | ||||
|                   ${this.value} | ||||
|                 </iron-input> | ||||
|                 ${this.value | ||||
|                   ? html` | ||||
|                       <ha-icon-button | ||||
|                         slot="suffix" | ||||
|                         @click=${this._clearValue} | ||||
|                         .label=${this.hass?.localize("ui.common.close") || | ||||
|                         "close"} | ||||
|                         .path=${mdiClose} | ||||
|                       ></ha-icon-button> | ||||
|                     ` | ||||
|                   : html` | ||||
|                       <ha-icon-button | ||||
|                         slot="suffix" | ||||
|                         .path=${this.icon} | ||||
|                       ></ha-icon-button> | ||||
|                     `} | ||||
|               </paper-input-container> | ||||
|               ${this.icon | ||||
|                 ? html`<span | ||||
|                     class="mdc-text-field__icon mdc-text-field__icon--leading" | ||||
|                     tabindex="-1" | ||||
|                   > | ||||
|                     <ha-icon-button | ||||
|                       @click=${this._openFilePicker} | ||||
|                       .path=${this.icon} | ||||
|                     ></ha-icon-button> | ||||
|                   </span>` | ||||
|                 : ""} | ||||
|               <div class="value">${this.value}</div> | ||||
|               <input | ||||
|                 id="input" | ||||
|                 type="file" | ||||
|                 class="mdc-text-field__input file" | ||||
|                 accept=${this.accept} | ||||
|                 @change=${this._handleFilePicked} | ||||
|                 aria-labelledby="label" | ||||
|               /> | ||||
|               ${this.value | ||||
|                 ? html`<span | ||||
|                     class="mdc-text-field__icon mdc-text-field__icon--trailing" | ||||
|                     tabindex="1" | ||||
|                   > | ||||
|                     <ha-icon-button | ||||
|                       slot="suffix" | ||||
|                       @click=${this._clearValue} | ||||
|                       .label=${this.hass?.localize("ui.common.close") || | ||||
|                       "close"} | ||||
|                       .path=${mdiClose} | ||||
|                     ></ha-icon-button> | ||||
|                   </span>` | ||||
|                 : ""} | ||||
|               <span | ||||
|                 class="mdc-line-ripple ${this._drag | ||||
|                   ? "mdc-line-ripple--active" | ||||
|                   : ""}" | ||||
|               ></span> | ||||
|             </label> | ||||
|           `} | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   private _openFilePicker() { | ||||
|     this._input?.click(); | ||||
|   } | ||||
|  | ||||
|   private _handleDrop(ev: DragEvent) { | ||||
|     ev.preventDefault(); | ||||
|     ev.stopPropagation(); | ||||
| @@ -137,40 +152,66 @@ export class HaFileUpload extends LitElement { | ||||
|   } | ||||
|  | ||||
|   static get styles() { | ||||
|     return css` | ||||
|       paper-input-container { | ||||
|         position: relative; | ||||
|         padding: 8px; | ||||
|         margin: 0 -8px; | ||||
|       } | ||||
|       paper-input-container.dragged:before { | ||||
|         position: var(--layout-fit_-_position); | ||||
|         top: var(--layout-fit_-_top); | ||||
|         right: var(--layout-fit_-_right); | ||||
|         bottom: var(--layout-fit_-_bottom); | ||||
|         left: var(--layout-fit_-_left); | ||||
|         background: currentColor; | ||||
|         content: ""; | ||||
|         opacity: var(--dark-divider-opacity); | ||||
|         pointer-events: none; | ||||
|         border-radius: 4px; | ||||
|       } | ||||
|       input.file { | ||||
|         display: none; | ||||
|       } | ||||
|       img { | ||||
|         max-width: 125px; | ||||
|         max-height: 125px; | ||||
|       } | ||||
|       ha-icon-button { | ||||
|         --mdc-icon-button-size: 24px; | ||||
|         --mdc-icon-size: 20px; | ||||
|       } | ||||
|       ha-circular-progress { | ||||
|         display: block; | ||||
|         text-align-last: center; | ||||
|       } | ||||
|     `; | ||||
|     return [ | ||||
|       styles, | ||||
|       css` | ||||
|         :host { | ||||
|           display: block; | ||||
|         } | ||||
|         .mdc-text-field--filled { | ||||
|           height: auto; | ||||
|           padding-top: 16px; | ||||
|           cursor: pointer; | ||||
|         } | ||||
|         .mdc-text-field--filled.mdc-text-field--with-trailing-icon { | ||||
|           padding-top: 28px; | ||||
|         } | ||||
|         .mdc-text-field:not(.mdc-text-field--disabled) .mdc-text-field__icon { | ||||
|           color: var(--secondary-text-color); | ||||
|         } | ||||
|         .mdc-text-field--filled.mdc-text-field--with-trailing-icon | ||||
|           .mdc-text-field__icon { | ||||
|           align-self: flex-end; | ||||
|         } | ||||
|         .mdc-text-field__icon--leading { | ||||
|           margin-bottom: 12px; | ||||
|         } | ||||
|         .mdc-text-field--filled .mdc-floating-label--float-above { | ||||
|           transform: scale(0.75); | ||||
|           top: 8px; | ||||
|         } | ||||
|         .dragged:before { | ||||
|           position: var(--layout-fit_-_position); | ||||
|           top: var(--layout-fit_-_top); | ||||
|           right: var(--layout-fit_-_right); | ||||
|           bottom: var(--layout-fit_-_bottom); | ||||
|           left: var(--layout-fit_-_left); | ||||
|           background: currentColor; | ||||
|           content: ""; | ||||
|           opacity: var(--dark-divider-opacity); | ||||
|           pointer-events: none; | ||||
|           border-radius: 4px; | ||||
|         } | ||||
|         .value { | ||||
|           width: 100%; | ||||
|         } | ||||
|         input.file { | ||||
|           display: none; | ||||
|         } | ||||
|         img { | ||||
|           max-width: 100%; | ||||
|           max-height: 125px; | ||||
|         } | ||||
|         ha-icon-button { | ||||
|           --mdc-icon-button-size: 24px; | ||||
|           --mdc-icon-size: 20px; | ||||
|         } | ||||
|         ha-circular-progress { | ||||
|           display: block; | ||||
|           text-align-last: center; | ||||
|         } | ||||
|       `, | ||||
|     ]; | ||||
|   } | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| import { HaFormSchema } from "./types"; | ||||
| import type { Selector } from "../../data/selector"; | ||||
| import type { HaFormSchema } from "./types"; | ||||
|  | ||||
| export const computeInitialHaFormData = ( | ||||
|   schema: HaFormSchema[] | ||||
| @@ -31,6 +32,25 @@ export const computeInitialHaFormData = ( | ||||
|         minutes: 0, | ||||
|         seconds: 0, | ||||
|       }; | ||||
|     } else if ("selector" in field) { | ||||
|       const selector: Selector = field.selector; | ||||
|       if ("boolean" in selector) { | ||||
|         data[field.name] = false; | ||||
|       } else if ("text" in selector) { | ||||
|         data[field.name] = ""; | ||||
|       } else if ("number" in selector) { | ||||
|         data[field.name] = "min" in selector.number ? selector.number.min : 0; | ||||
|       } else if ("select" in selector) { | ||||
|         if (selector.select.options.length) { | ||||
|           data[field.name] = selector.select.options[0][0]; | ||||
|         } | ||||
|       } else if ("duration" in selector) { | ||||
|         data[field.name] = { | ||||
|           hours: 0, | ||||
|           minutes: 0, | ||||
|           seconds: 0, | ||||
|         }; | ||||
|       } | ||||
|     } | ||||
|   }); | ||||
|   return data; | ||||
|   | ||||
| @@ -9,7 +9,9 @@ export class HaFormConstant extends LitElement implements HaFormElement { | ||||
|   @property() public label!: string; | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     return html`<span class="label">${this.label}</span>: ${this.schema.value}`; | ||||
|     return html`<span class="label">${this.label}</span>${this.schema.value | ||||
|         ? `: ${this.schema.value}` | ||||
|         : ""}`; | ||||
|   } | ||||
|  | ||||
|   static get styles(): CSSResultGroup { | ||||
|   | ||||
							
								
								
									
										95
									
								
								src/components/ha-form/ha-form-grid.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										95
									
								
								src/components/ha-form/ha-form-grid.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,95 @@ | ||||
| import "./ha-form"; | ||||
| import { | ||||
|   css, | ||||
|   CSSResultGroup, | ||||
|   html, | ||||
|   LitElement, | ||||
|   PropertyValues, | ||||
|   TemplateResult, | ||||
| } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import type { | ||||
|   HaFormGridSchema, | ||||
|   HaFormDataContainer, | ||||
|   HaFormElement, | ||||
|   HaFormSchema, | ||||
| } from "./types"; | ||||
| import type { HomeAssistant } from "../../types"; | ||||
|  | ||||
| @customElement("ha-form-grid") | ||||
| export class HaFormGrid extends LitElement implements HaFormElement { | ||||
|   @property({ attribute: false }) public hass!: HomeAssistant; | ||||
|  | ||||
|   @property({ attribute: false }) public data!: HaFormDataContainer; | ||||
|  | ||||
|   @property({ attribute: false }) public schema!: HaFormGridSchema; | ||||
|  | ||||
|   @property({ type: Boolean }) public disabled = false; | ||||
|  | ||||
|   @property() public computeLabel?: ( | ||||
|     schema: HaFormSchema, | ||||
|     data?: HaFormDataContainer | ||||
|   ) => string; | ||||
|  | ||||
|   @property() public computeHelper?: (schema: HaFormSchema) => string; | ||||
|  | ||||
|   protected firstUpdated(changedProps: PropertyValues) { | ||||
|     super.firstUpdated(changedProps); | ||||
|     this.setAttribute("own-margin", ""); | ||||
|   } | ||||
|  | ||||
|   protected updated(changedProps: PropertyValues): void { | ||||
|     super.updated(changedProps); | ||||
|     if (changedProps.has("schema")) { | ||||
|       if (this.schema.column_min_width) { | ||||
|         this.style.setProperty( | ||||
|           "--form-grid-min-width", | ||||
|           this.schema.column_min_width | ||||
|         ); | ||||
|       } else { | ||||
|         this.style.setProperty("--form-grid-min-width", ""); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
|       ${this.schema.schema.map( | ||||
|         (item) => | ||||
|           html` | ||||
|             <ha-form | ||||
|               .hass=${this.hass} | ||||
|               .data=${this.data} | ||||
|               .schema=${[item]} | ||||
|               .disabled=${this.disabled} | ||||
|               .computeLabel=${this.computeLabel} | ||||
|               .computeHelper=${this.computeHelper} | ||||
|             ></ha-form> | ||||
|           ` | ||||
|       )} | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   static get styles(): CSSResultGroup { | ||||
|     return css` | ||||
|       :host { | ||||
|         display: grid !important; | ||||
|         grid-template-columns: repeat( | ||||
|           var(--form-grid-column-count, auto-fit), | ||||
|           minmax(var(--form-grid-min-width, 200px), 1fr) | ||||
|         ); | ||||
|         grid-gap: 8px; | ||||
|       } | ||||
|       :host > ha-form { | ||||
|         display: block; | ||||
|         margin-bottom: 24px; | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "ha-form-grid": HaFormGrid; | ||||
|   } | ||||
| } | ||||
| @@ -1,4 +1,3 @@ | ||||
| import "@material/mwc-select/mwc-select"; | ||||
| import { mdiMenuDown, mdiMenuUp } from "@mdi/js"; | ||||
| import { | ||||
|   css, | ||||
| @@ -11,7 +10,8 @@ import { | ||||
| import { customElement, property, query, state } from "lit/decorators"; | ||||
| import { fireEvent } from "../../common/dom/fire_event"; | ||||
| import "../ha-button-menu"; | ||||
| import { HaCheckListItem } from "../ha-check-list-item"; | ||||
| import "../ha-check-list-item"; | ||||
| import type { HaCheckListItem } from "../ha-check-list-item"; | ||||
| import "../ha-checkbox"; | ||||
| import type { HaCheckbox } from "../ha-checkbox"; | ||||
| import "../ha-formfield"; | ||||
|   | ||||
| @@ -1,17 +1,20 @@ | ||||
| import "@material/mwc-select"; | ||||
| import type { Select } from "@material/mwc-select"; | ||||
| import "@material/mwc-list/mwc-list-item"; | ||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property, query } from "lit/decorators"; | ||||
| import memoizeOne from "memoize-one"; | ||||
| import { html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import { fireEvent } from "../../common/dom/fire_event"; | ||||
| import "../ha-radio"; | ||||
| import { HaFormElement, HaFormSelectData, HaFormSelectSchema } from "./types"; | ||||
|  | ||||
| import { stopPropagation } from "../../common/dom/stop_propagation"; | ||||
| import type { HaRadio } from "../ha-radio"; | ||||
| import type { HomeAssistant } from "../../types"; | ||||
| import type { | ||||
|   HaFormElement, | ||||
|   HaFormSelectData, | ||||
|   HaFormSelectSchema, | ||||
| } from "./types"; | ||||
| import type { SelectSelector } from "../../data/selector"; | ||||
| import "../ha-selector/ha-selector-select"; | ||||
|  | ||||
| @customElement("ha-form-select") | ||||
| export class HaFormSelect extends LitElement implements HaFormElement { | ||||
|   @property({ attribute: false }) public hass!: HomeAssistant; | ||||
|  | ||||
|   @property({ attribute: false }) public schema!: HaFormSelectSchema; | ||||
|  | ||||
|   @property() public data!: HaFormSelectData; | ||||
| @@ -20,60 +23,35 @@ export class HaFormSelect extends LitElement implements HaFormElement { | ||||
|  | ||||
|   @property({ type: Boolean }) public disabled = false; | ||||
|  | ||||
|   @query("mwc-select", true) private _input?: HTMLElement; | ||||
|  | ||||
|   public focus() { | ||||
|     if (this._input) { | ||||
|       this._input.focus(); | ||||
|     } | ||||
|   } | ||||
|   private _selectSchema = memoizeOne( | ||||
|     (options): SelectSelector => ({ | ||||
|       select: { | ||||
|         options: options.map((option) => ({ | ||||
|           value: option[0], | ||||
|           label: option[1], | ||||
|         })), | ||||
|       }, | ||||
|     }) | ||||
|   ); | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     if (this.schema.required && this.schema.options!.length < 6) { | ||||
|       return html` | ||||
|         <div> | ||||
|           ${this.label} | ||||
|           ${this.schema.options.map( | ||||
|             ([value, label]) => html` | ||||
|               <mwc-formfield .label=${label}> | ||||
|                 <ha-radio | ||||
|                   .checked=${value === this.data} | ||||
|                   .value=${value} | ||||
|                   .disabled=${this.disabled} | ||||
|                   @change=${this._valueChanged} | ||||
|                 ></ha-radio> | ||||
|               </mwc-formfield> | ||||
|             ` | ||||
|           )} | ||||
|         </div> | ||||
|       `; | ||||
|     } | ||||
|  | ||||
|     return html` | ||||
|       <mwc-select | ||||
|         fixedMenuPosition | ||||
|         naturalMenuWidth | ||||
|         .label=${this.label} | ||||
|       <ha-selector-select | ||||
|         .hass=${this.hass} | ||||
|         .schema=${this.schema} | ||||
|         .value=${this.data} | ||||
|         .label=${this.label} | ||||
|         .disabled=${this.disabled} | ||||
|         @closed=${stopPropagation} | ||||
|         @selected=${this._valueChanged} | ||||
|       > | ||||
|         ${!this.schema.required | ||||
|           ? html`<mwc-list-item value=""></mwc-list-item>` | ||||
|           : ""} | ||||
|         ${this.schema.options!.map( | ||||
|           ([value, label]) => html` | ||||
|             <mwc-list-item .value=${value}>${label}</mwc-list-item> | ||||
|           ` | ||||
|         )} | ||||
|       </mwc-select> | ||||
|         .required=${this.schema.required} | ||||
|         .selector=${this._selectSchema(this.schema.options)} | ||||
|         @value-changed=${this._valueChanged} | ||||
|       ></ha-selector-select> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   private _valueChanged(ev: CustomEvent) { | ||||
|     ev.stopPropagation(); | ||||
|     let value: string | undefined = (ev.target as Select | HaRadio).value; | ||||
|     let value: string | undefined = ev.detail.value; | ||||
|  | ||||
|     if (value === this.data) { | ||||
|       return; | ||||
| @@ -87,15 +65,6 @@ export class HaFormSelect extends LitElement implements HaFormElement { | ||||
|       value, | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   static get styles(): CSSResultGroup { | ||||
|     return css` | ||||
|       mwc-select, | ||||
|       mwc-formfield { | ||||
|         display: block; | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   | ||||
| @@ -1,10 +1,18 @@ | ||||
| import { css, CSSResultGroup, html, LitElement, PropertyValues } from "lit"; | ||||
| import { | ||||
|   css, | ||||
|   CSSResultGroup, | ||||
|   html, | ||||
|   LitElement, | ||||
|   PropertyValues, | ||||
|   TemplateResult, | ||||
| } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import { dynamicElement } from "../../common/dom/dynamic-element-directive"; | ||||
| import { fireEvent } from "../../common/dom/fire_event"; | ||||
| import "../ha-alert"; | ||||
| import "./ha-form-boolean"; | ||||
| import "./ha-form-constant"; | ||||
| import "./ha-form-grid"; | ||||
| import "./ha-form-float"; | ||||
| import "./ha-form-integer"; | ||||
| import "./ha-form-multi_select"; | ||||
| @@ -14,17 +22,20 @@ import "./ha-form-string"; | ||||
| import { HaFormElement, HaFormDataContainer, HaFormSchema } from "./types"; | ||||
| import { HomeAssistant } from "../../types"; | ||||
|  | ||||
| const getValue = (obj, item) => (obj ? obj[item.name] : null); | ||||
| const getValue = (obj, item) => | ||||
|   obj ? (!item.name ? obj : obj[item.name]) : null; | ||||
|  | ||||
| const getError = (obj, item) => (obj && item.name ? obj[item.name] : null); | ||||
|  | ||||
| let selectorImported = false; | ||||
|  | ||||
| @customElement("ha-form") | ||||
| export class HaForm extends LitElement implements HaFormElement { | ||||
|   @property() public hass!: HomeAssistant; | ||||
|   @property({ attribute: false }) public hass!: HomeAssistant; | ||||
|  | ||||
|   @property() public data!: HaFormDataContainer; | ||||
|   @property({ attribute: false }) public data!: HaFormDataContainer; | ||||
|  | ||||
|   @property() public schema!: HaFormSchema[]; | ||||
|   @property({ attribute: false }) public schema!: HaFormSchema[]; | ||||
|  | ||||
|   @property() public error?: Record<string, string>; | ||||
|  | ||||
| @@ -32,7 +43,12 @@ export class HaForm extends LitElement implements HaFormElement { | ||||
|  | ||||
|   @property() public computeError?: (schema: HaFormSchema, error) => string; | ||||
|  | ||||
|   @property() public computeLabel?: (schema: HaFormSchema) => string; | ||||
|   @property() public computeLabel?: ( | ||||
|     schema: HaFormSchema, | ||||
|     data?: HaFormDataContainer | ||||
|   ) => string; | ||||
|  | ||||
|   @property() public computeHelper?: (schema: HaFormSchema) => string; | ||||
|  | ||||
|   public focus() { | ||||
|     const root = this.shadowRoot?.querySelector(".root"); | ||||
| @@ -59,7 +75,7 @@ export class HaForm extends LitElement implements HaFormElement { | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   protected render() { | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
|       <div class="root"> | ||||
|         ${this.error && this.error.base | ||||
| @@ -70,7 +86,8 @@ export class HaForm extends LitElement implements HaFormElement { | ||||
|             ` | ||||
|           : ""} | ||||
|         ${this.schema.map((item) => { | ||||
|           const error = getValue(this.error, item); | ||||
|           const error = getError(this.error, item); | ||||
|  | ||||
|           return html` | ||||
|             ${error | ||||
|               ? html` | ||||
| @@ -85,15 +102,21 @@ export class HaForm extends LitElement implements HaFormElement { | ||||
|                   .hass=${this.hass} | ||||
|                   .selector=${item.selector} | ||||
|                   .value=${getValue(this.data, item)} | ||||
|                   .label=${this._computeLabel(item)} | ||||
|                   .label=${this._computeLabel(item, this.data)} | ||||
|                   .disabled=${this.disabled} | ||||
|                   .required=${item.required} | ||||
|                   .helper=${this._computeHelper(item)} | ||||
|                   .required=${item.required || false} | ||||
|                   .context=${this._generateContext(item)} | ||||
|                 ></ha-selector>` | ||||
|               : dynamicElement(`ha-form-${item.type}`, { | ||||
|                   schema: item, | ||||
|                   data: getValue(this.data, item), | ||||
|                   label: this._computeLabel(item), | ||||
|                   label: this._computeLabel(item, this.data), | ||||
|                   disabled: this.disabled, | ||||
|                   hass: this.hass, | ||||
|                   computeLabel: this.computeLabel, | ||||
|                   computeHelper: this.computeHelper, | ||||
|                   context: this._generateContext(item), | ||||
|                 })} | ||||
|           `; | ||||
|         })} | ||||
| @@ -101,27 +124,50 @@ export class HaForm extends LitElement implements HaFormElement { | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   private _generateContext( | ||||
|     schema: HaFormSchema | ||||
|   ): Record<string, any> | undefined { | ||||
|     if (!schema.context) { | ||||
|       return undefined; | ||||
|     } | ||||
|  | ||||
|     const context = {}; | ||||
|     for (const [context_key, data_key] of Object.entries(schema.context)) { | ||||
|       context[context_key] = this.data[data_key]; | ||||
|     } | ||||
|     return context; | ||||
|   } | ||||
|  | ||||
|   protected createRenderRoot() { | ||||
|     const root = super.createRenderRoot(); | ||||
|     // attach it as soon as possible to make sure we fetch all events. | ||||
|     root.addEventListener("value-changed", (ev) => { | ||||
|       ev.stopPropagation(); | ||||
|       const schema = (ev.target as HaFormElement).schema as HaFormSchema; | ||||
|  | ||||
|       const newValue = !schema.name | ||||
|         ? ev.detail.value | ||||
|         : { [schema.name]: ev.detail.value }; | ||||
|  | ||||
|       fireEvent(this, "value-changed", { | ||||
|         value: { ...this.data, [schema.name]: ev.detail.value }, | ||||
|         value: { ...this.data, ...newValue }, | ||||
|       }); | ||||
|     }); | ||||
|     return root; | ||||
|   } | ||||
|  | ||||
|   private _computeLabel(schema: HaFormSchema) { | ||||
|   private _computeLabel(schema: HaFormSchema, data: HaFormDataContainer) { | ||||
|     return this.computeLabel | ||||
|       ? this.computeLabel(schema) | ||||
|       ? this.computeLabel(schema, data) | ||||
|       : schema | ||||
|       ? schema.name | ||||
|       : ""; | ||||
|   } | ||||
|  | ||||
|   private _computeHelper(schema: HaFormSchema) { | ||||
|     return this.computeHelper ? this.computeHelper(schema) : ""; | ||||
|   } | ||||
|  | ||||
|   private _computeError(error, schema: HaFormSchema | HaFormSchema[]) { | ||||
|     return this.computeError ? this.computeError(error, schema) : error; | ||||
|   } | ||||
|   | ||||
| @@ -11,7 +11,8 @@ export type HaFormSchema = | ||||
|   | HaFormSelectSchema | ||||
|   | HaFormMultiSelectSchema | ||||
|   | HaFormTimeSchema | ||||
|   | HaFormSelector; | ||||
|   | HaFormSelector | ||||
|   | HaFormGridSchema; | ||||
|  | ||||
| export interface HaFormBaseSchema { | ||||
|   name: string; | ||||
| @@ -23,6 +24,14 @@ export interface HaFormBaseSchema { | ||||
|     // This value will be set initially when form is loaded | ||||
|     suggested_value?: HaFormData; | ||||
|   }; | ||||
|   context?: Record<string, string>; | ||||
| } | ||||
|  | ||||
| export interface HaFormGridSchema extends HaFormBaseSchema { | ||||
|   type: "grid"; | ||||
|   name: ""; | ||||
|   column_min_width?: string; | ||||
|   schema: HaFormSchema[]; | ||||
| } | ||||
|  | ||||
| export interface HaFormSelector extends HaFormBaseSchema { | ||||
| @@ -32,7 +41,7 @@ export interface HaFormSelector extends HaFormBaseSchema { | ||||
|  | ||||
| export interface HaFormConstantSchema extends HaFormBaseSchema { | ||||
|   type: "constant"; | ||||
|   value: string; | ||||
|   value?: string; | ||||
| } | ||||
|  | ||||
| export interface HaFormIntegerSchema extends HaFormBaseSchema { | ||||
| @@ -49,7 +58,7 @@ export interface HaFormSelectSchema extends HaFormBaseSchema { | ||||
|  | ||||
| export interface HaFormMultiSelectSchema extends HaFormBaseSchema { | ||||
|   type: "multi_select"; | ||||
|   options: Record<string, string> | string[]; | ||||
|   options: Record<string, string> | string[] | Array<[string, string]>; | ||||
| } | ||||
|  | ||||
| export interface HaFormFloatSchema extends HaFormBaseSchema { | ||||
|   | ||||
| @@ -43,6 +43,8 @@ class HaHLSPlayer extends LitElement { | ||||
|  | ||||
|   @state() private _error?: string; | ||||
|  | ||||
|   @state() private _errorIsFatal = false; | ||||
|  | ||||
|   private _hlsPolyfillInstance?: HlsLite; | ||||
|  | ||||
|   private _exoPlayer = false; | ||||
| @@ -53,6 +55,7 @@ class HaHLSPlayer extends LitElement { | ||||
|     super.connectedCallback(); | ||||
|     HaHLSPlayer.streamCount += 1; | ||||
|     if (this.hasUpdated) { | ||||
|       this._resetError(); | ||||
|       this._startHls(); | ||||
|     } | ||||
|   } | ||||
| @@ -64,16 +67,23 @@ class HaHLSPlayer extends LitElement { | ||||
|   } | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     if (this._error) { | ||||
|       return html`<ha-alert alert-type="error">${this._error}</ha-alert>`; | ||||
|     } | ||||
|     return html` | ||||
|       <video | ||||
|         ?autoplay=${this.autoPlay} | ||||
|         .muted=${this.muted} | ||||
|         ?playsinline=${this.playsInline} | ||||
|         ?controls=${this.controls} | ||||
|       ></video> | ||||
|       ${this._error | ||||
|         ? html`<ha-alert | ||||
|             alert-type="error" | ||||
|             class=${this._errorIsFatal ? "fatal" : "retry"} | ||||
|           > | ||||
|             ${this._error} | ||||
|           </ha-alert>` | ||||
|         : ""} | ||||
|       ${!this._errorIsFatal | ||||
|         ? html`<video | ||||
|             ?autoplay=${this.autoPlay} | ||||
|             .muted=${this.muted} | ||||
|             ?playsinline=${this.playsInline} | ||||
|             ?controls=${this.controls} | ||||
|           ></video>` | ||||
|         : ""} | ||||
|     `; | ||||
|   } | ||||
|  | ||||
| @@ -87,12 +97,11 @@ class HaHLSPlayer extends LitElement { | ||||
|     } | ||||
|  | ||||
|     this._cleanUp(); | ||||
|     this._resetError(); | ||||
|     this._startHls(); | ||||
|   } | ||||
|  | ||||
|   private async _startHls(): Promise<void> { | ||||
|     this._error = undefined; | ||||
|  | ||||
|     const masterPlaylistPromise = fetch(this.url); | ||||
|  | ||||
|     const Hls: typeof HlsType = (await import("hls.js/dist/hls.light.min")) | ||||
| @@ -110,8 +119,8 @@ class HaHLSPlayer extends LitElement { | ||||
|     } | ||||
|  | ||||
|     if (!hlsSupported) { | ||||
|       this._error = this.hass.localize( | ||||
|         "ui.components.media-browser.video_not_supported" | ||||
|       this._setFatalError( | ||||
|         this.hass.localize("ui.components.media-browser.video_not_supported") | ||||
|       ); | ||||
|       return; | ||||
|     } | ||||
| @@ -219,9 +228,16 @@ class HaHLSPlayer extends LitElement { | ||||
|     this._hlsPolyfillInstance = hls; | ||||
|     hls.attachMedia(videoEl); | ||||
|     hls.on(Hls.Events.MEDIA_ATTACHED, () => { | ||||
|       this._resetError(); | ||||
|       hls.loadSource(url); | ||||
|     }); | ||||
|     hls.on(Hls.Events.ERROR, (_, data: any) => { | ||||
|     hls.on(Hls.Events.FRAG_LOADED, (_event, _data: any) => { | ||||
|       this._resetError(); | ||||
|     }); | ||||
|     hls.on(Hls.Events.ERROR, (_event, data: any) => { | ||||
|       // Some errors are recovered automatically by the hls player itself, and the others handled | ||||
|       // in this function require special actions to recover. Errors retried in this function | ||||
|       // are done with backoff to not cause unecessary failures. | ||||
|       if (!data.fatal) { | ||||
|         return; | ||||
|       } | ||||
| @@ -241,22 +257,22 @@ class HaHLSPlayer extends LitElement { | ||||
|                 error += " (" + data.response.code + ")"; | ||||
|               } | ||||
|             } | ||||
|             this._error = error; | ||||
|             return; | ||||
|             this._setRetryableError(error); | ||||
|             break; | ||||
|           } | ||||
|           case Hls.ErrorDetails.MANIFEST_LOAD_TIMEOUT: | ||||
|             this._error = "Timeout while starting stream"; | ||||
|             return; | ||||
|             this._setRetryableError("Timeout while starting stream"); | ||||
|             break; | ||||
|           default: | ||||
|             this._error = "Unknown stream network error (" + data.details + ")"; | ||||
|             return; | ||||
|             this._setRetryableError("Stream network error"); | ||||
|             break; | ||||
|         } | ||||
|         this._error = "Error with media stream contents (" + data.details + ")"; | ||||
|         hls.startLoad(); | ||||
|       } else if (data.type === Hls.ErrorTypes.MEDIA_ERROR) { | ||||
|         this._error = "Error with media stream contents (" + data.details + ")"; | ||||
|         this._setRetryableError("Error with media stream contents"); | ||||
|         hls.recoverMediaError(); | ||||
|       } else { | ||||
|         this._error = | ||||
|           "Unknown error with stream (" + data.type + ", " + data.details + ")"; | ||||
|         this._setFatalError("Error playing stream"); | ||||
|       } | ||||
|     }); | ||||
|   } | ||||
| @@ -284,6 +300,21 @@ class HaHLSPlayer extends LitElement { | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   private _resetError() { | ||||
|     this._error = undefined; | ||||
|     this._errorIsFatal = false; | ||||
|   } | ||||
|  | ||||
|   private _setFatalError(errorMessage: string) { | ||||
|     this._error = errorMessage; | ||||
|     this._errorIsFatal = true; | ||||
|   } | ||||
|  | ||||
|   private _setRetryableError(errorMessage: string) { | ||||
|     this._error = errorMessage; | ||||
|     this._errorIsFatal = false; | ||||
|   } | ||||
|  | ||||
|   static get styles(): CSSResultGroup { | ||||
|     return css` | ||||
|       :host, | ||||
| @@ -296,10 +327,14 @@ class HaHLSPlayer extends LitElement { | ||||
|         max-height: var(--video-max-height, calc(100vh - 97px)); | ||||
|       } | ||||
|  | ||||
|       ha-alert { | ||||
|       .fatal { | ||||
|         display: block; | ||||
|         padding: 100px 16px; | ||||
|       } | ||||
|  | ||||
|       .retry { | ||||
|         display: block; | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user